1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
use abstutil::Counter;
use map_gui::tools::{ColorLegend, ColorNetwork};
use map_gui::ID;
use map_model::{IntersectionID, PathStep, RoadID, Traversable};
use widgetry::mapspace::ToggleZoomed;
use widgetry::{
Color, EventCtx, GfxCtx, HorizontalAlignment, Line, Outcome, Panel, State, Text,
VerticalAlignment, Widget,
};
use crate::app::App;
use crate::app::Transition;
use crate::common::CommonState;
pub struct PathCounter {
panel: Panel,
draw: ToggleZoomed,
cnt: Counter<RoadID>,
tooltip: Option<Text>,
}
impl PathCounter {
pub fn demand_across_intersection(
ctx: &mut EventCtx,
app: &App,
i: IntersectionID,
) -> Box<dyn State<App>> {
let map = &app.primary.map;
let sim = &app.primary.sim;
let mut cnt = Counter::new();
for agent in sim.active_agents() {
if let Some(path) = sim.get_path(agent) {
if path.get_steps().iter().any(|step| match step {
PathStep::Turn(t) | PathStep::ContraflowTurn(t) => t.parent == i,
_ => false,
}) {
for step in path.get_steps() {
if let Traversable::Lane(l) = step.as_traversable() {
cnt.inc(l.road);
}
}
}
}
}
let mut colorer = ColorNetwork::new(app);
colorer
.draw
.unzoomed
.push(Color::CYAN, map.get_i(i).polygon.clone());
colorer
.draw
.zoomed
.push(Color::CYAN.alpha(0.5), map.get_i(i).polygon.clone());
colorer.pct_roads(cnt.clone(), &app.cs.good_to_bad_red);
Box::new(PathCounter {
draw: colorer.build(ctx),
tooltip: None,
cnt,
panel: Panel::new_builder(Widget::col(vec![
Widget::row(vec![
Line(format!("Paths across {}", i))
.small_heading()
.into_widget(ctx),
ctx.style().btn_close_widget(ctx),
]),
ColorLegend::gradient(
ctx,
&app.cs.good_to_bad_red,
vec!["lowest count", "highest"],
),
]))
.aligned(HorizontalAlignment::Center, VerticalAlignment::Top)
.build(ctx),
})
}
}
impl State<App> for PathCounter {
fn event(&mut self, ctx: &mut EventCtx, app: &mut App) -> Transition {
ctx.canvas_movement();
if ctx.redo_mouseover() {
app.primary.current_selection = app.mouseover_unzoomed_roads_and_intersections(ctx);
self.tooltip = None;
if let Some(r) = match app.primary.current_selection {
Some(ID::Lane(l)) => Some(l.road),
Some(ID::Road(r)) => Some(r),
_ => None,
} {
let n = self.cnt.get(r);
if n > 0 {
self.tooltip = Some(Text::from(abstutil::prettyprint_usize(n)));
}
} else {
app.primary.current_selection = None;
}
}
if let Outcome::Clicked(x) = self.panel.event(ctx) {
match x.as_ref() {
"close" => {
return Transition::Pop;
}
_ => unreachable!(),
}
}
Transition::Keep
}
fn draw(&self, g: &mut GfxCtx, app: &App) {
self.panel.draw(g);
CommonState::draw_osd(g, app);
self.draw.draw(g);
if let Some(ref txt) = self.tooltip {
g.draw_mouse_tooltip(txt.clone());
}
}
}