mirror of
https://github.com/nushell/nushell
synced 2025-01-08 03:09:00 +00:00
147 lines
4.5 KiB
Rust
147 lines
4.5 KiB
Rust
|
use nu_data::utils::Model;
|
||
|
use nu_errors::ShellError;
|
||
|
|
||
|
use tui::{
|
||
|
layout::{Constraint, Direction, Layout},
|
||
|
style::{Color, Modifier, Style},
|
||
|
symbols,
|
||
|
text::Span,
|
||
|
widgets::{Axis, Chart, Dataset, GraphType},
|
||
|
};
|
||
|
|
||
|
const DEFAULT_COLOR: Color = Color::Green;
|
||
|
|
||
|
const DEFAULT_LINE_COLORS: [Color; 5] = [
|
||
|
Color::Green,
|
||
|
Color::Cyan,
|
||
|
Color::Magenta,
|
||
|
Color::Yellow,
|
||
|
Color::Red,
|
||
|
];
|
||
|
|
||
|
#[derive(Debug)]
|
||
|
pub struct Line<'a> {
|
||
|
title: &'a str,
|
||
|
x_labels: Vec<String>,
|
||
|
x_range: [f64; 2],
|
||
|
y_range: [f64; 2],
|
||
|
datasets_names: Vec<String>,
|
||
|
data: Vec<Vec<(f64, f64)>>,
|
||
|
}
|
||
|
|
||
|
impl<'a> Line<'a> {
|
||
|
pub fn from_model(model: &'a Model) -> Result<Line<'a>, ShellError> {
|
||
|
Ok(Line {
|
||
|
title: "Line Chart",
|
||
|
x_labels: model.labels.x.to_vec(),
|
||
|
x_range: [
|
||
|
model.ranges.0.start.as_u64()? as f64,
|
||
|
model.labels.x.len() as f64,
|
||
|
],
|
||
|
y_range: [
|
||
|
model.ranges.1.start.as_u64()? as f64,
|
||
|
model.ranges.1.end.as_u64()? as f64,
|
||
|
],
|
||
|
datasets_names: if model.labels.y.len() == 1 {
|
||
|
vec!["".to_string()]
|
||
|
} else {
|
||
|
model.labels.y.to_vec()
|
||
|
},
|
||
|
data: model
|
||
|
.data
|
||
|
.table_entries()
|
||
|
.collect::<Vec<_>>()
|
||
|
.iter()
|
||
|
.map(|subset| {
|
||
|
subset
|
||
|
.table_entries()
|
||
|
.enumerate()
|
||
|
.map(|(idx, data_point)| {
|
||
|
(
|
||
|
idx as f64,
|
||
|
if let Ok(point) = data_point.as_u64() {
|
||
|
point as f64
|
||
|
} else {
|
||
|
0.0
|
||
|
},
|
||
|
)
|
||
|
})
|
||
|
.collect::<Vec<_>>()
|
||
|
})
|
||
|
.collect::<Vec<_>>(),
|
||
|
})
|
||
|
}
|
||
|
|
||
|
pub fn draw<T>(&mut self, ui: &mut tui::Terminal<T>) -> std::io::Result<()>
|
||
|
where
|
||
|
T: tui::backend::Backend,
|
||
|
{
|
||
|
ui.draw(|f| {
|
||
|
let chunks = Layout::default()
|
||
|
.direction(Direction::Vertical)
|
||
|
.margin(1)
|
||
|
.constraints([Constraint::Percentage(100)].as_ref())
|
||
|
.split(f.size());
|
||
|
|
||
|
let x_labels = self
|
||
|
.x_labels
|
||
|
.iter()
|
||
|
.map(move |label| {
|
||
|
Span::styled(label, Style::default().add_modifier(Modifier::BOLD))
|
||
|
})
|
||
|
.collect::<Vec<_>>();
|
||
|
|
||
|
let y_labels = vec![
|
||
|
Span::styled(
|
||
|
format!("{}", self.y_range[0]),
|
||
|
Style::default().add_modifier(Modifier::BOLD),
|
||
|
),
|
||
|
Span::raw(format!("{}", (self.y_range[0] + self.y_range[1]) / 2.0)),
|
||
|
Span::styled(
|
||
|
format!("{}", self.y_range[1]),
|
||
|
Style::default().add_modifier(Modifier::BOLD),
|
||
|
),
|
||
|
];
|
||
|
|
||
|
let marker = if x_labels.len() > 60 {
|
||
|
symbols::Marker::Braille
|
||
|
} else {
|
||
|
symbols::Marker::Dot
|
||
|
};
|
||
|
|
||
|
let datasets = self
|
||
|
.data
|
||
|
.iter()
|
||
|
.enumerate()
|
||
|
.map(|(idx, data_series)| {
|
||
|
Dataset::default()
|
||
|
.name(&self.datasets_names[idx])
|
||
|
.marker(marker)
|
||
|
.graph_type(GraphType::Line)
|
||
|
.style(
|
||
|
Style::default().fg(*DEFAULT_LINE_COLORS
|
||
|
.get(idx)
|
||
|
.unwrap_or_else(|| &DEFAULT_COLOR)),
|
||
|
)
|
||
|
.data(data_series)
|
||
|
})
|
||
|
.collect();
|
||
|
|
||
|
let chart = Chart::new(datasets)
|
||
|
.x_axis(
|
||
|
Axis::default()
|
||
|
.style(Style::default().fg(Color::Gray))
|
||
|
.labels(x_labels)
|
||
|
.bounds(self.x_range),
|
||
|
)
|
||
|
.y_axis(
|
||
|
Axis::default()
|
||
|
.style(Style::default().fg(Color::Gray))
|
||
|
.labels(y_labels)
|
||
|
.bounds(self.y_range),
|
||
|
);
|
||
|
f.render_widget(chart, chunks[0]);
|
||
|
})
|
||
|
}
|
||
|
}
|