mirror of
https://github.com/a-b-street/abstreet.git
synced 2025-01-08 07:41:57 +03:00
275 lines
10 KiB
Rust
275 lines
10 KiB
Rust
// You have to run from the ezgui crate (abstreet/ezgui), due to relative paths to fonts and
|
|
// images.
|
|
//
|
|
// To run:
|
|
// > cargo run --example demo
|
|
//
|
|
// Try the web version, but there's no text rendering yet:
|
|
// > cargo web start --target wasm32-unknown-unknown --no-default-features \
|
|
// --features wasm-backend --example demo
|
|
|
|
use ezgui::{
|
|
hotkey, lctrl, Btn, Checkbox, Color, Composite, Drawable, EventCtx, EventLoopMode, GeomBatch,
|
|
GfxCtx, HorizontalAlignment, Key, Line, LinePlot, Outcome, PlotOptions, Series, Text, TextExt,
|
|
VerticalAlignment, Widget, GUI,
|
|
};
|
|
use geom::{Angle, Duration, Polygon, Pt2D, Time};
|
|
|
|
fn main() {
|
|
// Control flow surrendered here. App implements State, which has an event handler and a draw
|
|
// callback.
|
|
ezgui::run(
|
|
ezgui::Settings::new("ezgui demo", "../data/system/fonts"),
|
|
|ctx| App::new(ctx),
|
|
);
|
|
}
|
|
|
|
struct App {
|
|
controls: Composite,
|
|
timeseries_panel: Option<(Duration, Composite)>,
|
|
scrollable_canvas: Drawable,
|
|
elapsed: Duration,
|
|
}
|
|
|
|
impl App {
|
|
fn new(ctx: &mut EventCtx) -> App {
|
|
App {
|
|
controls: make_controls(ctx),
|
|
timeseries_panel: None,
|
|
scrollable_canvas: setup_scrollable_canvas(ctx),
|
|
elapsed: Duration::ZERO,
|
|
}
|
|
}
|
|
|
|
fn make_timeseries_panel(&self, ctx: &mut EventCtx) -> Composite {
|
|
// Make a table with 3 columns.
|
|
let mut col1 = vec![Line("Time").draw(ctx)];
|
|
let mut col2 = vec![Line("Linear").draw(ctx)];
|
|
let mut col3 = vec![Line("Quadratic").draw(ctx)];
|
|
for s in 0..(self.elapsed.inner_seconds() as usize) {
|
|
col1.push(
|
|
Line(Duration::seconds(s as f64).to_string())
|
|
.secondary()
|
|
.draw(ctx),
|
|
);
|
|
col2.push(Line(s.to_string()).secondary().draw(ctx));
|
|
col3.push(Line(s.pow(2).to_string()).secondary().draw(ctx));
|
|
}
|
|
|
|
let mut c = Composite::new(
|
|
Widget::col(vec![
|
|
Widget::row(vec![{
|
|
let mut txt = Text::from(
|
|
Line("Here's a bunch of text to force some scrolling.").small_heading(),
|
|
);
|
|
txt.add(
|
|
Line(
|
|
"Bug: scrolling by clicking and dragging doesn't work while the \
|
|
stopwatch is running.",
|
|
)
|
|
.fg(Color::RED),
|
|
);
|
|
txt.draw(ctx)
|
|
}]),
|
|
Widget::row(vec![
|
|
// Examples of styling widgets
|
|
Widget::col(col1).outline(3.0, Color::BLACK).margin(5),
|
|
Widget::col(col2).outline(3.0, Color::BLACK).margin(5),
|
|
Widget::col(col3).outline(3.0, Color::BLACK).margin(5),
|
|
]),
|
|
LinePlot::new(
|
|
ctx,
|
|
"timeseries",
|
|
vec![
|
|
Series {
|
|
label: "Linear".to_string(),
|
|
color: Color::GREEN,
|
|
// These points are (x axis = Time, y axis = usize)
|
|
pts: (0..(self.elapsed.inner_seconds() as usize))
|
|
.map(|s| (Time::START_OF_DAY + Duration::seconds(s as f64), s))
|
|
.collect(),
|
|
},
|
|
Series {
|
|
label: "Quadratic".to_string(),
|
|
color: Color::BLUE,
|
|
pts: (0..(self.elapsed.inner_seconds() as usize))
|
|
.map(|s| {
|
|
(Time::START_OF_DAY + Duration::seconds(s as f64), s.pow(2))
|
|
})
|
|
.collect(),
|
|
},
|
|
],
|
|
PlotOptions {
|
|
// Without this, the plot doesn't stretch to cover times in between whole
|
|
// seconds.
|
|
max_x: Some(Time::START_OF_DAY + self.elapsed),
|
|
max_y: None,
|
|
},
|
|
),
|
|
])
|
|
.bg(Color::grey(0.4)),
|
|
)
|
|
// Don't let the panel exceed this percentage of the window. Scrollbars appear
|
|
// automatically if needed.
|
|
.max_size_percent(30, 40)
|
|
// We take up 30% width, and we want to leave 10% window width as buffer.
|
|
.aligned(HorizontalAlignment::Percent(0.6), VerticalAlignment::Center)
|
|
.build(ctx);
|
|
|
|
// Since we're creating an entirely new panel when the time changes, we need to preserve
|
|
// some internal state, like scroll and whether plot checkboxes were enabled.
|
|
if let Some((_, ref old)) = self.timeseries_panel {
|
|
c.restore(ctx, old);
|
|
}
|
|
c
|
|
}
|
|
}
|
|
|
|
impl GUI for App {
|
|
fn event(&mut self, ctx: &mut EventCtx) -> EventLoopMode {
|
|
// Allow panning and zooming to work.
|
|
ctx.canvas_movement();
|
|
|
|
// This dispatches event handling to all of the widgets inside.
|
|
match self.controls.event(ctx) {
|
|
Some(Outcome::Clicked(x)) => match x.as_ref() {
|
|
// These outcomes should probably be a custom enum per Composite, to be more
|
|
// typesafe.
|
|
"reset the stopwatch" => {
|
|
self.elapsed = Duration::ZERO;
|
|
// We can replace any named widget with another one. Layout gets recalculated.
|
|
self.controls.replace(
|
|
ctx,
|
|
"stopwatch",
|
|
format!("Stopwatch: {}", self.elapsed)
|
|
.draw_text(ctx)
|
|
.named("stopwatch"),
|
|
);
|
|
}
|
|
_ => unreachable!(),
|
|
},
|
|
None => {}
|
|
}
|
|
|
|
// An update event means that no keyboard/mouse input happened, but time has passed.
|
|
// (Ignore the "nonblocking"; this API is funky right now. Only one caller "consumes" an
|
|
// event, so that multiple things don't all respond to one keypress, but that's set up
|
|
// oddly for update events.)
|
|
if let Some(dt) = ctx.input.nonblocking_is_update_event() {
|
|
ctx.input.use_update_event();
|
|
self.elapsed += dt;
|
|
self.controls.replace(
|
|
ctx,
|
|
"stopwatch",
|
|
format!("Stopwatch: {}", self.elapsed)
|
|
.draw_text(ctx)
|
|
.named("stopwatch"),
|
|
);
|
|
}
|
|
|
|
if self.controls.is_checked("Show timeseries") {
|
|
// Update the panel when time changes.
|
|
if self
|
|
.timeseries_panel
|
|
.as_ref()
|
|
.map(|(dt, _)| *dt != self.elapsed)
|
|
.unwrap_or(true)
|
|
{
|
|
self.timeseries_panel = Some((self.elapsed, self.make_timeseries_panel(ctx)));
|
|
}
|
|
} else {
|
|
self.timeseries_panel = None;
|
|
}
|
|
|
|
if let Some((_, ref mut p)) = self.timeseries_panel {
|
|
match p.event(ctx) {
|
|
// No buttons in there
|
|
Some(Outcome::Clicked(_)) => unreachable!(),
|
|
None => {}
|
|
}
|
|
}
|
|
|
|
// If we're paused, only call event() again when there's some kind of input. If not, also
|
|
// sprinkle in periodic update events as time passes.
|
|
if self.controls.is_checked("paused") {
|
|
EventLoopMode::InputOnly
|
|
} else {
|
|
EventLoopMode::Animation
|
|
}
|
|
}
|
|
|
|
fn draw(&self, g: &mut GfxCtx) {
|
|
g.clear(Color::BLACK);
|
|
|
|
if self.controls.is_checked("Draw scrollable canvas") {
|
|
g.redraw(&self.scrollable_canvas);
|
|
}
|
|
|
|
self.controls.draw(g);
|
|
|
|
if let Some((_, ref p)) = self.timeseries_panel {
|
|
p.draw(g);
|
|
}
|
|
}
|
|
}
|
|
|
|
// This prepares a bunch of geometry (colored polygons) and uploads it to the GPU once. Then it can
|
|
// be redrawn cheaply later.
|
|
fn setup_scrollable_canvas(ctx: &mut EventCtx) -> Drawable {
|
|
let mut batch = GeomBatch::new();
|
|
batch.push(
|
|
Color::hex("#4E30A6"),
|
|
Polygon::rounded_rectangle(5000.0, 5000.0, Some(25.0)),
|
|
);
|
|
// SVG support using lyon and usvg. Map-space means don't scale for high DPI monitors.
|
|
batch.append(
|
|
GeomBatch::mapspace_svg(&ctx.prerender, "../data/system/assets/pregame/logo.svg")
|
|
.translate(300.0, 300.0),
|
|
);
|
|
// Text rendering also goes through lyon and usvg.
|
|
batch.append(
|
|
Text::from(Line("Awesome vector text thanks to usvg and lyon").fg(Color::hex("#DF8C3D")))
|
|
.render_to_batch(&ctx.prerender)
|
|
.scale(2.0)
|
|
.centered_on(Pt2D::new(600.0, 500.0))
|
|
.rotate(Angle::new_degs(30.0)),
|
|
);
|
|
// This is a bit of a hack; it's needed so that zooming in/out has reasonable limits.
|
|
ctx.canvas.map_dims = (5000.0, 5000.0);
|
|
batch.upload(ctx)
|
|
}
|
|
|
|
fn make_controls(ctx: &mut EventCtx) -> Composite {
|
|
Composite::new(
|
|
Widget::col(vec![
|
|
{
|
|
let mut txt = Text::from(Line("ezgui demo").small_heading());
|
|
txt.add(Line(
|
|
"Click and drag to pan, use touchpad or scroll wheel to zoom",
|
|
));
|
|
txt.draw(ctx)
|
|
},
|
|
Widget::row(vec![
|
|
// This just cycles between two arbitrary buttons
|
|
Checkbox::new(
|
|
false,
|
|
Btn::text_bg1("Pause").build(ctx, "pause the stopwatch", hotkey(Key::Space)),
|
|
Btn::text_bg1("Resume").build(ctx, "resume the stopwatch", hotkey(Key::Space)),
|
|
)
|
|
.named("paused")
|
|
.margin(5),
|
|
Btn::text_fg("Reset")
|
|
.build(ctx, "reset the stopwatch", None)
|
|
.margin(5),
|
|
Checkbox::text(ctx, "Draw scrollable canvas", None, true).margin(5),
|
|
Checkbox::text(ctx, "Show timeseries", lctrl(Key::T), false).margin(5),
|
|
])
|
|
.evenly_spaced(),
|
|
"Stopwatch: ...".draw_text(ctx).named("stopwatch"),
|
|
])
|
|
.bg(Color::grey(0.4)),
|
|
)
|
|
.aligned(HorizontalAlignment::Center, VerticalAlignment::Top)
|
|
.build(ctx)
|
|
}
|