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
use crate::app::App;
use crate::game::{ChooseSomething, State, Transition};
use crate::sandbox::{spawn_agents_around, SpeedControls, TimePanel};
use abstutil::Timer;
use ezgui::{
hotkey, Btn, Choice, Composite, EventCtx, GfxCtx, HorizontalAlignment, Key, Outcome, TextExt,
UpdateType, VerticalAlignment, Widget,
};
use geom::Duration;
use map_model::IntersectionID;
use std::collections::BTreeSet;
struct PreviewTrafficSignal {
composite: Composite,
speed: SpeedControls,
time_panel: TimePanel,
}
impl PreviewTrafficSignal {
fn new(ctx: &mut EventCtx, app: &App) -> PreviewTrafficSignal {
PreviewTrafficSignal {
composite: Composite::new(Widget::col(vec![
"Previewing traffic signal".draw_text(ctx),
Btn::text_fg("back to editing").build_def(ctx, hotkey(Key::Escape)),
]))
.aligned(HorizontalAlignment::Center, VerticalAlignment::Top)
.build(ctx),
speed: SpeedControls::new(ctx, app),
time_panel: TimePanel::new(ctx, app),
}
}
}
impl State for PreviewTrafficSignal {
fn event(&mut self, ctx: &mut EventCtx, app: &mut App) -> Transition {
ctx.canvas_movement();
match self.composite.event(ctx) {
Outcome::Clicked(x) => match x.as_ref() {
"back to editing" => {
app.primary.clear_sim();
return Transition::Pop;
}
_ => unreachable!(),
},
_ => {}
}
self.time_panel.event(ctx, app);
if let Some(t) = self.speed.event(ctx, app, None) {
return t;
}
if self.speed.is_paused() {
Transition::Keep
} else {
ctx.request_update(UpdateType::Game);
Transition::Keep
}
}
fn draw(&self, g: &mut GfxCtx, _: &App) {
self.composite.draw(g);
self.speed.draw(g);
self.time_panel.draw(g);
}
}
pub fn make_previewer(
ctx: &mut EventCtx,
app: &App,
members: BTreeSet<IntersectionID>,
phase: usize,
) -> Box<dyn State> {
let random = "random agents around these intersections".to_string();
let right_now = format!(
"change the traffic signal live at {}",
app.suspended_sim.as_ref().unwrap().time()
);
ChooseSomething::new(
ctx,
"Preview the traffic signal with what kind of traffic?",
Choice::strings(vec![random, right_now]),
Box::new(move |x, ctx, app| {
if x == "random agents around these intersections" {
for (idx, i) in members.iter().enumerate() {
if idx == 0 {
let signal = app.primary.map.get_traffic_signal(*i);
let mut step = Duration::ZERO;
for idx in 0..phase {
step += signal.phases[idx].phase_type.simple_duration();
}
app.primary.sim.timed_step(
&app.primary.map,
step,
&mut app.primary.sim_cb,
&mut Timer::throwaway(),
);
}
spawn_agents_around(*i, app);
}
} else {
app.primary.sim = app.suspended_sim.as_ref().unwrap().clone();
app.primary
.sim
.handle_live_edited_traffic_signals(&app.primary.map);
}
Transition::Replace(Box::new(PreviewTrafficSignal::new(ctx, app)))
}),
)
}