mirror of
https://github.com/a-b-street/abstreet.git
synced 2025-01-07 23:20:42 +03:00
prep for all plots to have toggleable series.
This commit is contained in:
parent
fadae3aa85
commit
694b1f6cc9
@ -1,5 +1,5 @@
|
|||||||
use crate::{
|
use crate::{
|
||||||
Color, Drawable, EventCtx, GeomBatch, GfxCtx, Line, Outcome, ScreenDims, ScreenPt,
|
Checkbox, Color, Drawable, EventCtx, GeomBatch, GfxCtx, Line, Outcome, ScreenDims, ScreenPt,
|
||||||
ScreenRectangle, Text, TextExt, Widget, WidgetImpl,
|
ScreenRectangle, Text, TextExt, Widget, WidgetImpl,
|
||||||
};
|
};
|
||||||
use abstutil::prettyprint_usize;
|
use abstutil::prettyprint_usize;
|
||||||
@ -7,7 +7,8 @@ use geom::{Angle, Bounds, Circle, Distance, Duration, FindClosest, PolyLine, Pt2
|
|||||||
|
|
||||||
// The X is always time
|
// The X is always time
|
||||||
pub struct Plot<T: Yvalue<T>> {
|
pub struct Plot<T: Yvalue<T>> {
|
||||||
draw: Drawable,
|
series: Vec<SeriesState>,
|
||||||
|
draw_grid: Drawable,
|
||||||
|
|
||||||
// The geometry here is in screen-space.
|
// The geometry here is in screen-space.
|
||||||
max_x: Time,
|
max_x: Time,
|
||||||
@ -18,6 +19,12 @@ pub struct Plot<T: Yvalue<T>> {
|
|||||||
dims: ScreenDims,
|
dims: ScreenDims,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
struct SeriesState {
|
||||||
|
label: String,
|
||||||
|
enabled: bool,
|
||||||
|
draw: Drawable,
|
||||||
|
}
|
||||||
|
|
||||||
pub struct PlotOptions {
|
pub struct PlotOptions {
|
||||||
pub max_x: Option<Time>,
|
pub max_x: Option<Time>,
|
||||||
}
|
}
|
||||||
@ -30,34 +37,33 @@ impl PlotOptions {
|
|||||||
|
|
||||||
impl<T: Yvalue<T>> Plot<T> {
|
impl<T: Yvalue<T>> Plot<T> {
|
||||||
pub fn new(ctx: &EventCtx, series: Vec<Series<T>>, opts: PlotOptions) -> Widget {
|
pub fn new(ctx: &EventCtx, series: Vec<Series<T>>, opts: PlotOptions) -> Widget {
|
||||||
let mut batch = GeomBatch::new();
|
let legend = if series.len() == 1 {
|
||||||
|
|
||||||
// TODO Tuned to fit the info panel. Instead these should somehow stretch to fill their
|
|
||||||
// container.
|
|
||||||
let width = 0.25 * ctx.canvas.window_width;
|
|
||||||
let height = 0.2 * ctx.canvas.window_height;
|
|
||||||
|
|
||||||
let radius = 15.0;
|
let radius = 15.0;
|
||||||
let legend = Widget::row(
|
// Can't hide if there's just one series
|
||||||
series
|
|
||||||
.iter()
|
|
||||||
.map(|s| {
|
|
||||||
Widget::row(vec![
|
Widget::row(vec![
|
||||||
Widget::draw_batch(
|
Widget::draw_batch(
|
||||||
ctx,
|
ctx,
|
||||||
GeomBatch::from(vec![(
|
GeomBatch::from(vec![(
|
||||||
s.color,
|
series[0].color,
|
||||||
Circle::new(Pt2D::new(radius, radius), Distance::meters(radius))
|
Circle::new(Pt2D::new(radius, radius), Distance::meters(radius))
|
||||||
.to_polygon(),
|
.to_polygon(),
|
||||||
)]),
|
)]),
|
||||||
)
|
)
|
||||||
.margin(5),
|
.margin(5),
|
||||||
s.label.clone().draw_text(ctx),
|
series[0].label.clone().draw_text(ctx),
|
||||||
])
|
])
|
||||||
|
} else {
|
||||||
|
Widget::row(
|
||||||
|
series
|
||||||
|
.iter()
|
||||||
|
.map(|s| {
|
||||||
|
// TODO Colored checkbox
|
||||||
|
Checkbox::text(ctx, &s.label, None, true)
|
||||||
})
|
})
|
||||||
.collect(),
|
.collect(),
|
||||||
)
|
)
|
||||||
.flex_wrap(ctx, 24);
|
.flex_wrap(ctx, 24)
|
||||||
|
};
|
||||||
|
|
||||||
// Assume min_x is Time::START_OF_DAY and min_y is T::zero()
|
// Assume min_x is Time::START_OF_DAY and min_y is T::zero()
|
||||||
let max_x = opts.max_x.unwrap_or_else(|| {
|
let max_x = opts.max_x.unwrap_or_else(|| {
|
||||||
@ -85,6 +91,12 @@ impl<T: Yvalue<T>> Plot<T> {
|
|||||||
.max()
|
.max()
|
||||||
.unwrap_or(T::zero());
|
.unwrap_or(T::zero());
|
||||||
|
|
||||||
|
// TODO Tuned to fit the info panel. Instead these should somehow stretch to fill their
|
||||||
|
// container.
|
||||||
|
let width = 0.25 * ctx.canvas.window_width;
|
||||||
|
let height = 0.2 * ctx.canvas.window_height;
|
||||||
|
|
||||||
|
let mut grid_batch = GeomBatch::new();
|
||||||
// Grid lines for the Y scale. Draw up to 10 lines max to cover the order of magnitude of
|
// Grid lines for the Y scale. Draw up to 10 lines max to cover the order of magnitude of
|
||||||
// the range.
|
// the range.
|
||||||
// TODO This caps correctly, but if the max is 105, then suddenly we just have 2 grid
|
// TODO This caps correctly, but if the max is 105, then suddenly we just have 2 grid
|
||||||
@ -97,7 +109,7 @@ impl<T: Yvalue<T>> Plot<T> {
|
|||||||
if pct > 1.0 {
|
if pct > 1.0 {
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
batch.push(
|
grid_batch.push(
|
||||||
Color::BLACK,
|
Color::BLACK,
|
||||||
PolyLine::new(vec![
|
PolyLine::new(vec![
|
||||||
Pt2D::new(0.0, (1.0 - pct) * height),
|
Pt2D::new(0.0, (1.0 - pct) * height),
|
||||||
@ -116,7 +128,7 @@ impl<T: Yvalue<T>> Plot<T> {
|
|||||||
if pct > 1.0 {
|
if pct > 1.0 {
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
batch.push(
|
grid_batch.push(
|
||||||
Color::BLACK,
|
Color::BLACK,
|
||||||
PolyLine::new(vec![
|
PolyLine::new(vec![
|
||||||
Pt2D::new(pct * width, 0.0),
|
Pt2D::new(pct * width, 0.0),
|
||||||
@ -131,10 +143,18 @@ impl<T: Yvalue<T>> Plot<T> {
|
|||||||
Pt2D::new(0.0, 0.0),
|
Pt2D::new(0.0, 0.0),
|
||||||
Pt2D::new(width, height),
|
Pt2D::new(width, height),
|
||||||
]));
|
]));
|
||||||
|
let mut series_state = Vec::new();
|
||||||
for s in series {
|
for s in series {
|
||||||
|
let mut batch = GeomBatch::new();
|
||||||
if max_x == Time::START_OF_DAY {
|
if max_x == Time::START_OF_DAY {
|
||||||
|
series_state.push(SeriesState {
|
||||||
|
label: s.label,
|
||||||
|
enabled: true,
|
||||||
|
draw: ctx.upload(batch),
|
||||||
|
});
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
let mut pts = Vec::new();
|
let mut pts = Vec::new();
|
||||||
for (t, y) in s.pts {
|
for (t, y) in s.pts {
|
||||||
let percent_x = t.to_percent(max_x);
|
let percent_x = t.to_percent(max_x);
|
||||||
@ -156,10 +176,16 @@ impl<T: Yvalue<T>> Plot<T> {
|
|||||||
.make_polygons_with_miter_threshold(Distance::meters(5.0), 10.0),
|
.make_polygons_with_miter_threshold(Distance::meters(5.0), 10.0),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
series_state.push(SeriesState {
|
||||||
|
label: s.label,
|
||||||
|
enabled: true,
|
||||||
|
draw: ctx.upload(batch),
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
let plot = Plot {
|
let plot = Plot {
|
||||||
draw: ctx.upload(batch),
|
series: series_state,
|
||||||
|
draw_grid: ctx.upload(grid_batch),
|
||||||
closest,
|
closest,
|
||||||
max_x,
|
max_x,
|
||||||
max_y: max_y,
|
max_y: max_y,
|
||||||
@ -214,7 +240,12 @@ impl<T: Yvalue<T>> WidgetImpl for Plot<T> {
|
|||||||
}
|
}
|
||||||
|
|
||||||
fn draw(&self, g: &mut GfxCtx) {
|
fn draw(&self, g: &mut GfxCtx) {
|
||||||
g.redraw_at(self.top_left, &self.draw);
|
g.redraw_at(self.top_left, &self.draw_grid);
|
||||||
|
for series in &self.series {
|
||||||
|
if series.enabled {
|
||||||
|
g.redraw_at(self.top_left, &series.draw);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if let Some(cursor) = g.canvas.get_cursor_in_screen_space() {
|
if let Some(cursor) = g.canvas.get_cursor_in_screen_space() {
|
||||||
if ScreenRectangle::top_left(self.top_left, self.dims).contains(cursor) {
|
if ScreenRectangle::top_left(self.top_left, self.dims).contains(cursor) {
|
||||||
@ -224,6 +255,7 @@ impl<T: Yvalue<T>> WidgetImpl for Plot<T> {
|
|||||||
Pt2D::new(cursor.x - self.top_left.x, cursor.y - self.top_left.y),
|
Pt2D::new(cursor.x - self.top_left.x, cursor.y - self.top_left.y),
|
||||||
radius,
|
radius,
|
||||||
) {
|
) {
|
||||||
|
if self.series.iter().any(|s| s.label == label && s.enabled) {
|
||||||
// TODO If some/all of the matches have the same t, write it once?
|
// TODO If some/all of the matches have the same t, write it once?
|
||||||
let t = self.max_x.percent_of(pt.x() / self.dims.width);
|
let t = self.max_x.percent_of(pt.x() / self.dims.width);
|
||||||
let y_percent = 1.0 - (pt.y() / self.dims.height);
|
let y_percent = 1.0 - (pt.y() / self.dims.height);
|
||||||
@ -236,6 +268,7 @@ impl<T: Yvalue<T>> WidgetImpl for Plot<T> {
|
|||||||
self.max_y.from_percent(y_percent).prettyprint()
|
self.max_y.from_percent(y_percent).prettyprint()
|
||||||
)));
|
)));
|
||||||
}
|
}
|
||||||
|
}
|
||||||
if !txt.is_empty() {
|
if !txt.is_empty() {
|
||||||
g.fork_screenspace();
|
g.fork_screenspace();
|
||||||
g.draw_circle(Color::RED, &Circle::new(cursor.to_pt(), radius));
|
g.draw_circle(Color::RED, &Circle::new(cursor.to_pt(), radius));
|
||||||
|
Loading…
Reference in New Issue
Block a user