fold trip explorer into info panel

This commit is contained in:
Dustin Carlino 2020-01-16 17:11:51 -08:00
parent eb0281a294
commit 3636f7afdc
4 changed files with 146 additions and 22 deletions

View File

@ -1,8 +1,7 @@
use crate::common::route_viewer::RouteViewer; use crate::common::route_viewer::RouteViewer;
use crate::common::TripExplorer;
use crate::game::Transition; use crate::game::Transition;
use crate::ui::UI; use crate::ui::UI;
use ezgui::{EventCtx, GfxCtx, Key, ModalMenu}; use ezgui::{EventCtx, GfxCtx, ModalMenu};
pub struct AgentTools { pub struct AgentTools {
route_viewer: RouteViewer, route_viewer: RouteViewer,
@ -22,19 +21,6 @@ impl AgentTools {
menu: &mut ModalMenu, menu: &mut ModalMenu,
) -> Option<Transition> { ) -> Option<Transition> {
self.route_viewer.event(ctx, ui, menu); self.route_viewer.event(ctx, ui, menu);
if let Some(trip) = ui
.primary
.current_selection
.as_ref()
.and_then(|id| id.agent_id())
.and_then(|agent| ui.primary.sim.agent_to_trip(agent))
{
if ui.per_obj.action(ctx, Key::T, format!("explore {}", trip)) {
return Some(Transition::Push(Box::new(TripExplorer::new(trip, ctx, ui))));
}
}
None None
} }

View File

@ -1,15 +1,17 @@
use crate::common::{CommonState, Warping}; use crate::common::{CommonState, Warping};
use crate::game::{msg, Transition}; use crate::game::{msg, Transition};
use crate::helpers::{rotating_color, ID}; use crate::helpers::{rotating_color, rotating_color_map, ID};
use crate::render::{dashed_lines, MIN_ZOOM_FOR_DETAIL};
use crate::ui::UI; use crate::ui::UI;
use abstutil::prettyprint_usize; use abstutil::prettyprint_usize;
use ezgui::{ use ezgui::{
hotkey, Button, Color, Composite, EventCtx, EventLoopMode, HorizontalAlignment, Key, Line, hotkey, Button, Color, Composite, Drawable, EventCtx, EventLoopMode, GeomBatch, GfxCtx,
ManagedWidget, Outcome, Plot, RewriteColor, Series, Text, VerticalAlignment, HorizontalAlignment, Key, Line, ManagedWidget, Outcome, Plot, RewriteColor, Series, Text,
VerticalAlignment,
}; };
use geom::{Duration, Statistic, Time}; use geom::{Circle, Distance, Duration, Pt2D, Statistic, Time};
use map_model::{IntersectionID, RoadID}; use map_model::{IntersectionID, RoadID};
use sim::{CarID, TripMode}; use sim::{CarID, TripEnd, TripID, TripMode, TripStart};
use std::collections::BTreeMap; use std::collections::BTreeMap;
pub struct InfoPanel { pub struct InfoPanel {
@ -17,6 +19,8 @@ pub struct InfoPanel {
pub time: Time, pub time: Time,
pub composite: Composite, pub composite: Composite,
trip_details: Option<(Drawable, Drawable)>,
actions: Vec<(Key, String)>, actions: Vec<(Key, String)>,
} }
@ -94,6 +98,15 @@ impl InfoPanel {
_ => {} _ => {}
} }
let trip_details =
if let Some(trip) = id.agent_id().and_then(|a| ui.primary.sim.agent_to_trip(a)) {
let (rows, unzoomed, zoomed) = trip_details(trip, ctx, ui);
col.push(rows);
Some((unzoomed, zoomed))
} else {
None
};
// Follow the agent. When the sim is paused, this lets the player naturally pan away, // Follow the agent. When the sim is paused, this lets the player naturally pan away,
// because the InfoPanel isn't being updated. // because the InfoPanel isn't being updated.
// TODO Should we pin to the trip, not the specific agent? // TODO Should we pin to the trip, not the specific agent?
@ -107,6 +120,7 @@ impl InfoPanel {
InfoPanel { InfoPanel {
id, id,
actions, actions,
trip_details,
time: ui.primary.sim.time(), time: ui.primary.sim.time(),
composite: Composite::new(ManagedWidget::col(col).bg(Color::grey(0.3))) composite: Composite::new(ManagedWidget::col(col).bg(Color::grey(0.3)))
.aligned( .aligned(
@ -175,6 +189,17 @@ impl InfoPanel {
None => (false, None), None => (false, None),
} }
} }
pub fn draw(&self, g: &mut GfxCtx) {
self.composite.draw(g);
if let Some((ref unzoomed, ref zoomed)) = self.trip_details {
if g.canvas.cam_zoom < MIN_ZOOM_FOR_DETAIL {
g.redraw(unzoomed);
} else {
g.redraw(zoomed);
}
}
}
} }
fn info_for(id: ID, ui: &UI) -> Text { fn info_for(id: ID, ui: &UI) -> Text {
@ -457,3 +482,116 @@ fn color_for_mode(m: TripMode, ui: &UI) -> Color {
TripMode::Drive => ui.cs.get("unzoomed car"), TripMode::Drive => ui.cs.get("unzoomed car"),
} }
} }
// (extra rows to display, unzoomed view, zoomed view)
fn trip_details(trip: TripID, ctx: &mut EventCtx, ui: &UI) -> (ManagedWidget, Drawable, Drawable) {
let map = &ui.primary.map;
let phases = ui.primary.sim.get_analytics().get_trip_phases(trip, map);
let mut col = vec![ManagedWidget::draw_text(
ctx,
Text::from(Line(trip.to_string())),
)];
let mut unzoomed = GeomBatch::new();
let mut zoomed = GeomBatch::new();
for (idx, p) in phases.into_iter().enumerate() {
let color = rotating_color_map(idx + 1);
col.push(trip_line(ctx, color, p.describe(ui.primary.sim.time())));
// TODO Could really cache this between live updates
if let Some((dist, ref path)) = p.path {
if let Some(trace) = path.trace(map, dist, None) {
unzoomed.push(color, trace.make_polygons(Distance::meters(10.0)));
zoomed.extend(
ui.cs.get_def("route", Color::ORANGE.alpha(0.5)),
dashed_lines(
&trace,
Distance::meters(0.75),
Distance::meters(1.0),
Distance::meters(0.4),
),
);
}
}
}
// Handle endpoints
let (trip_start, trip_end) = ui.primary.sim.trip_endpoints(trip);
let start_color = rotating_color_map(0);
match trip_start {
TripStart::Bldg(b) => {
let bldg = map.get_b(b);
col.insert(
0,
trip_line(ctx, start_color, format!("start at {}", bldg.get_name(map))),
);
unzoomed.push(start_color, bldg.polygon.clone());
zoomed.push(start_color, bldg.polygon.clone());
}
TripStart::Border(i) => {
let i = map.get_i(i);
col.insert(
0,
trip_line(ctx, start_color, format!("enter map via {}", i.id)),
);
unzoomed.push(start_color, i.polygon.clone());
zoomed.push(start_color, i.polygon.clone());
}
};
// Is the trip ongoing?
if let Some(pt) = ui.primary.sim.get_canonical_pt_per_trip(trip, map).ok() {
let color = rotating_color_map(col.len());
unzoomed.push(color, Circle::new(pt, Distance::meters(10.0)).to_polygon());
// Don't need anything when zoomed; the info panel already focuses on them.
col.push(trip_line(ctx, color, format!("currently here")));
}
let end_color = rotating_color_map(col.len());
match trip_end {
TripEnd::Bldg(b) => {
let bldg = map.get_b(b);
col.push(trip_line(
ctx,
end_color,
format!("end at {}", bldg.get_name(map)),
));
unzoomed.push(end_color, bldg.polygon.clone());
zoomed.push(end_color, bldg.polygon.clone());
}
TripEnd::Border(i) => {
let i = map.get_i(i);
col.push(trip_line(ctx, end_color, format!("leave map via {}", i.id)));
unzoomed.push(end_color, i.polygon.clone());
zoomed.push(end_color, i.polygon.clone());
}
TripEnd::ServeBusRoute(br) => {
col.push(trip_line(
ctx,
end_color,
format!("serve route {} forever", map.get_br(br).name),
));
}
};
(
ManagedWidget::col(col),
unzoomed.upload(ctx),
zoomed.upload(ctx),
)
}
fn trip_line(ctx: &EventCtx, color: Color, descr: String) -> ManagedWidget {
let radius = 15.0;
ManagedWidget::row(vec![
ManagedWidget::draw_batch(
ctx,
GeomBatch::from(vec![(
color,
Circle::new(Pt2D::new(radius, radius), Distance::meters(radius)).to_polygon(),
)]),
),
ManagedWidget::draw_text(ctx, Text::from(Line(descr))),
])
}

View File

@ -80,7 +80,7 @@ impl CommonState {
pub fn draw_no_osd(&self, g: &mut GfxCtx, ui: &UI) { pub fn draw_no_osd(&self, g: &mut GfxCtx, ui: &UI) {
self.turn_cycler.draw(g, ui); self.turn_cycler.draw(g, ui);
if let Some(ref info) = self.info_panel { if let Some(ref info) = self.info_panel {
info.composite.draw(g); info.draw(g);
} }
} }

View File

@ -23,7 +23,7 @@ impl RouteViewer {
if let Some(trace) = ui.primary.sim.trace_route(agent, &ui.primary.map, None) { if let Some(trace) = ui.primary.sim.trace_route(agent, &ui.primary.map, None) {
let mut batch = GeomBatch::new(); let mut batch = GeomBatch::new();
batch.extend( batch.extend(
ui.cs.get_def("route", Color::ORANGE.alpha(0.5)), ui.cs.get("route"),
dashed_lines( dashed_lines(
&trace, &trace,
Distance::meters(0.75), Distance::meters(0.75),