abstreet/editor/src/ui.rs

680 lines
25 KiB
Rust
Raw Normal View History

// Copyright 2018 Google LLC, licensed under http://www.apache.org/licenses/LICENSE-2.0
// TODO this should just be a way to handle interactions between plugins
2018-06-22 21:01:44 +03:00
use abstutil;
2018-06-21 20:03:51 +03:00
use colors::{ColorScheme, Colors};
use control::ControlMap;
//use cpuprofiler;
use ezgui::{Canvas, Color, EventLoopMode, GfxCtx, Text, UserInput, BOTTOM_LEFT, GUI};
use flame;
use kml;
use map_model::{IntersectionID, Map};
use objects::{Ctx, ID, ROOT_MENU};
2018-10-06 01:44:12 +03:00
use piston::input::Key;
use plugins::a_b_tests::ABTestManager;
2018-10-13 21:34:00 +03:00
use plugins::chokepoints::ChokepointsFinder;
use plugins::classification::OsmClassifier;
2018-06-21 04:30:38 +03:00
use plugins::color_picker::ColorPicker;
use plugins::debug_objects::DebugObjectsState;
use plugins::diff_worlds::DiffWorldsState;
use plugins::draw_neighborhoods::DrawNeighborhoodState;
use plugins::floodfill::Floodfiller;
2018-08-29 21:11:51 +03:00
use plugins::follow::FollowState;
2018-07-06 01:37:35 +03:00
use plugins::geom_validation::Validator;
2018-09-13 19:43:04 +03:00
use plugins::hider::Hider;
use plugins::layers::ToggleableLayers;
2018-09-22 00:16:38 +03:00
use plugins::logs::DisplayLogs;
use plugins::map_edits::EditsManager;
use plugins::road_editor::RoadEditor;
use plugins::scenarios::ScenarioManager;
use plugins::search::SearchState;
use plugins::show_owner::ShowOwnerState;
2018-08-31 01:15:37 +03:00
use plugins::show_route::ShowRouteState;
use plugins::sim_controls::SimController;
use plugins::steep::SteepnessVisualizer;
use plugins::stop_sign_editor::StopSignEditor;
use plugins::traffic_signal_editor::TrafficSignalEditor;
use plugins::turn_cycler::TurnCyclerState;
use plugins::warp::WarpState;
use plugins::Colorizer;
use render::{DrawMap, RenderOptions};
2018-08-27 23:03:35 +03:00
use sim;
2018-10-03 17:47:59 +03:00
use sim::{Sim, SimFlags};
use std::process;
2018-09-14 23:23:39 +03:00
const MIN_ZOOM_FOR_MOUSEOVER: f64 = 4.0;
2018-10-22 05:23:47 +03:00
pub struct UI {
primary: PerMapUI,
2018-10-22 05:59:37 +03:00
primary_plugins: PluginsPerMap,
2018-10-22 05:23:47 +03:00
// When running an A/B test, this is populated too.
2018-10-22 05:59:37 +03:00
secondary: Option<(PerMapUI, PluginsPerMap)>,
2018-10-22 05:23:47 +03:00
plugins: PluginsPerUI,
// TODO describe An index into plugin_handlers.
2018-10-22 05:23:47 +03:00
active_plugin: Option<usize>,
canvas: Canvas,
cs: ColorScheme,
// Remember this to support loading a new PerMapUI
kml: Option<String>,
}
2018-10-22 05:23:47 +03:00
impl GUI for UI {
fn event(&mut self, mut input: UserInput, osd: &mut Text) -> EventLoopMode {
// First update the camera and handle zoom
let old_zoom = self.canvas.cam_zoom;
self.canvas.handle_event(&mut input);
let new_zoom = self.canvas.cam_zoom;
self.plugins.layers.handle_zoom(old_zoom, new_zoom);
// Always handle mouseover
if old_zoom >= MIN_ZOOM_FOR_MOUSEOVER && new_zoom < MIN_ZOOM_FOR_MOUSEOVER {
self.primary.current_selection = None;
}
if !self.canvas.is_dragging()
&& input.get_moved_mouse().is_some()
&& new_zoom >= MIN_ZOOM_FOR_MOUSEOVER
{
self.primary.current_selection = self.mouseover_something();
}
// TODO Normally we'd return InputOnly here if there was an active plugin, but actually, we
// want some keys to always be pressable (sim controller stuff, quitting the game?)
// If there's an active plugin, just run it.
if let Some(idx) = self.active_plugin {
if !self.run_plugin(idx, &mut input, osd) {
2018-10-22 05:23:47 +03:00
self.active_plugin = None;
}
} else {
// Run each plugin, short-circuiting if the plugin claimed it was active.
for idx in 0..=1 {
// TODO 23, and dont hardcode it
if self.run_plugin(idx, &mut input, osd) {
2018-10-22 05:23:47 +03:00
self.active_plugin = Some(idx);
break;
}
}
}
if input.unimportant_key_pressed(Key::Escape, ROOT_MENU, "quit") {
let state = EditorState {
map_name: self.primary.map.get_name().clone(),
cam_x: self.canvas.cam_x,
cam_y: self.canvas.cam_y,
cam_zoom: self.canvas.cam_zoom,
};
// TODO maybe make state line up with the map, so loading from a new map doesn't break
abstutil::write_json("editor_state", &state).expect("Saving editor_state failed");
abstutil::write_json("color_scheme", &self.cs).expect("Saving color_scheme failed");
info!("Saved editor_state and color_scheme");
//cpuprofiler::PROFILER.lock().unwrap().stop().unwrap();
process::exit(0);
}
// Sim controller plugin is kind of always active? If nothing else ran, let it use keys.
2018-10-22 05:59:37 +03:00
let result = self.plugins.sim_ctrl.event(
&mut input,
&mut self.primary,
&mut self.primary_plugins,
&mut self.secondary,
osd,
);
2018-10-22 05:23:47 +03:00
if self.primary.recalculate_current_selection {
self.primary.recalculate_current_selection = false;
self.primary.current_selection = self.mouseover_something();
}
input.populate_osd(osd);
result
}
2018-09-13 03:39:53 +03:00
fn get_mut_canvas(&mut self) -> &mut Canvas {
2018-10-22 05:23:47 +03:00
&mut self.canvas
}
fn draw(&self, g: &mut GfxCtx, osd: Text) {
2018-10-22 05:23:47 +03:00
g.clear(self.cs.get(Colors::Background));
let (statics, dynamics) = self.primary.draw_map.get_objects_onscreen(
self.canvas.get_screen_bbox(),
2018-10-22 05:59:37 +03:00
&self.primary_plugins.hider,
2018-10-22 05:23:47 +03:00
&self.primary.map,
&self.primary.sim,
&self.plugins.layers,
self,
);
for obj in statics.into_iter() {
let opts = RenderOptions {
color: self.color_obj(obj.get_id()),
cam_zoom: self.canvas.cam_zoom,
debug_mode: self.plugins.layers.debug_mode.is_enabled(),
};
obj.draw(
g,
opts,
Ctx {
cs: &self.cs,
map: &self.primary.map,
control_map: &self.primary.control_map,
canvas: &self.canvas,
sim: &self.primary.sim,
},
);
}
for obj in dynamics.into_iter() {
let opts = RenderOptions {
color: self.color_obj(obj.get_id()),
cam_zoom: self.canvas.cam_zoom,
debug_mode: self.plugins.layers.debug_mode.is_enabled(),
};
obj.draw(
g,
opts,
Ctx {
cs: &self.cs,
map: &self.primary.map,
control_map: &self.primary.control_map,
canvas: &self.canvas,
sim: &self.primary.sim,
},
);
}
// TODO Only if active?
2018-10-22 05:59:37 +03:00
self.primary_plugins.turn_cycler.draw(
2018-10-22 05:23:47 +03:00
&self.primary.map,
&self.primary.draw_map,
&self.primary.control_map,
self.primary.sim.time,
&self.cs,
g,
);
2018-10-22 05:59:37 +03:00
self.primary_plugins.debug_objects.draw(
2018-10-22 05:23:47 +03:00
&self.primary.map,
&self.canvas,
&self.primary.draw_map,
&self.primary.sim,
g,
);
self.plugins.color_picker.draw(&self.canvas, g);
2018-10-22 05:59:37 +03:00
self.primary_plugins
.draw_neighborhoods
.draw(g, &self.canvas);
self.primary_plugins.scenarios.draw(g, &self.canvas);
self.primary_plugins.edits_manager.draw(g, &self.canvas);
2018-10-22 05:23:47 +03:00
self.plugins.ab_test_manager.draw(g, &self.canvas);
self.plugins.logs.draw(g, &self.canvas);
self.plugins.search_state.draw(g, &self.canvas);
self.plugins.warp.draw(g, &self.canvas);
self.plugins.sim_ctrl.draw(g, &self.canvas);
2018-10-22 05:59:37 +03:00
self.primary_plugins.show_route.draw(g, &self.cs);
2018-10-22 05:23:47 +03:00
self.plugins.diff_worlds.draw(g, &self.cs);
self.canvas.draw_text(g, osd, BOTTOM_LEFT);
}
}
// All of the state that's bound to a specific map+edit has to live here.
// TODO How can we arrange the code so that we statically know that we don't pass anything from UI
// to something in PerMapUI?
pub struct PerMapUI {
pub map: Map,
draw_map: DrawMap,
pub control_map: ControlMap,
pub sim: Sim,
pub current_selection: Option<ID>,
pub recalculate_current_selection: bool,
current_flags: SimFlags,
2018-10-22 05:59:37 +03:00
}
2018-10-22 05:23:47 +03:00
2018-10-22 05:59:37 +03:00
pub struct PluginsPerMap {
2018-10-22 05:23:47 +03:00
// Anything that holds onto any kind of ID has to live here!
hider: Hider,
debug_objects: DebugObjectsState,
follow: FollowState,
show_route: ShowRouteState,
floodfiller: Floodfiller,
steepness_viz: SteepnessVisualizer,
traffic_signal_editor: TrafficSignalEditor,
stop_sign_editor: StopSignEditor,
road_editor: RoadEditor,
geom_validator: Validator,
turn_cycler: TurnCyclerState,
show_owner: ShowOwnerState,
draw_neighborhoods: DrawNeighborhoodState,
scenarios: ScenarioManager,
edits_manager: EditsManager,
chokepoints: ChokepointsFinder,
}
impl PerMapUI {
2018-10-22 05:59:37 +03:00
pub fn new(flags: SimFlags, kml: &Option<String>) -> (PerMapUI, PluginsPerMap) {
2018-10-22 05:23:47 +03:00
flame::start("setup");
let (map, control_map, sim) = sim::load(flags.clone(), Some(sim::Tick::from_seconds(30)));
let extra_shapes = if let Some(path) = kml {
kml::load(&path, &map.get_gps_bounds()).expect("Couldn't load extra KML shapes")
} else {
Vec::new()
};
flame::start("draw_map");
let draw_map = DrawMap::new(&map, &control_map, extra_shapes);
flame::end("draw_map");
flame::end("setup");
flame::dump_stdout();
let steepness_viz = SteepnessVisualizer::new(&map);
let road_editor = RoadEditor::new(map.get_road_edits().clone());
2018-10-22 05:59:37 +03:00
let state = PerMapUI {
2018-10-22 05:23:47 +03:00
map,
draw_map,
control_map,
sim,
current_selection: None,
recalculate_current_selection: false,
current_flags: flags,
2018-10-22 05:59:37 +03:00
};
let plugins = PluginsPerMap {
2018-10-22 05:23:47 +03:00
hider: Hider::new(),
debug_objects: DebugObjectsState::new(),
follow: FollowState::Empty,
show_route: ShowRouteState::Empty,
floodfiller: Floodfiller::new(),
steepness_viz,
traffic_signal_editor: TrafficSignalEditor::new(),
stop_sign_editor: StopSignEditor::new(),
road_editor,
geom_validator: Validator::new(),
turn_cycler: TurnCyclerState::new(),
show_owner: ShowOwnerState::new(),
draw_neighborhoods: DrawNeighborhoodState::new(),
scenarios: ScenarioManager::new(),
edits_manager: EditsManager::new(),
chokepoints: ChokepointsFinder::new(),
2018-10-22 05:59:37 +03:00
};
(state, plugins)
}
}
2018-10-22 05:23:47 +03:00
// aka plugins that don't depend on map
struct PluginsPerUI {
layers: ToggleableLayers,
search_state: SearchState,
warp: WarpState,
osm_classifier: OsmClassifier,
sim_ctrl: SimController,
color_picker: ColorPicker,
ab_test_manager: ABTestManager,
logs: DisplayLogs,
diff_worlds: DiffWorldsState,
}
impl UI {
pub fn new(flags: SimFlags, kml: Option<String>) -> UI {
2018-09-28 00:55:59 +03:00
// Do this first, so anything logged by sim::load isn't lost.
let logs = DisplayLogs::new();
2018-10-22 05:59:37 +03:00
let (primary, primary_plugins) = PerMapUI::new(flags, &kml);
let mut ui = UI {
2018-10-22 05:59:37 +03:00
primary,
primary_plugins,
secondary: None,
plugins: PluginsPerUI {
layers: ToggleableLayers::new(),
search_state: SearchState::Empty,
warp: WarpState::Empty,
osm_classifier: OsmClassifier::new(),
sim_ctrl: SimController::new(),
color_picker: ColorPicker::new(),
ab_test_manager: ABTestManager::new(),
logs,
diff_worlds: DiffWorldsState::new(),
},
2018-09-13 19:21:41 +03:00
active_plugin: None,
2018-09-10 03:10:34 +03:00
canvas: Canvas::new(),
cs: ColorScheme::load("color_scheme").unwrap(),
kml,
/*plugin_handlers: vec![
2018-10-03 17:47:59 +03:00
Box::new(|ctx| {
2018-10-22 05:59:37 +03:00
ctx.primary_plugins.stop_sign_editor.event(
2018-10-03 17:47:59 +03:00
ctx.input,
2018-10-22 05:11:20 +03:00
&ctx.primary.map,
&mut ctx.primary.control_map,
ctx.primary.current_selection,
)
}),
2018-10-03 17:47:59 +03:00
Box::new(|ctx| {
2018-10-22 05:59:37 +03:00
ctx.primary_plugins.road_editor.event(
2018-10-03 17:47:59 +03:00
ctx.input,
2018-10-22 05:11:20 +03:00
ctx.primary.current_selection,
&mut ctx.primary.map,
&mut ctx.primary.draw_map,
&ctx.primary.control_map,
&mut ctx.primary.sim,
)
}),
2018-10-22 05:11:20 +03:00
Box::new(|ctx| ctx.plugins.search_state.event(ctx.input)),
2018-10-03 17:47:59 +03:00
Box::new(|ctx| {
2018-10-22 05:11:20 +03:00
ctx.plugins.warp.event(
2018-10-03 17:47:59 +03:00
ctx.input,
2018-10-22 05:11:20 +03:00
&ctx.primary.map,
&ctx.primary.sim,
&ctx.primary.draw_map,
ctx.canvas,
&mut ctx.primary.current_selection,
)
}),
2018-10-03 17:47:59 +03:00
Box::new(|ctx| {
2018-10-22 05:59:37 +03:00
ctx.primary_plugins.follow.event(
2018-10-03 17:47:59 +03:00
ctx.input,
2018-10-22 05:11:20 +03:00
&ctx.primary.map,
&ctx.primary.sim,
ctx.canvas,
ctx.primary.current_selection,
)
}),
2018-10-03 17:47:59 +03:00
Box::new(|ctx| {
2018-10-22 05:59:37 +03:00
ctx.primary_plugins.show_route.event(
ctx.input,
2018-10-22 05:11:20 +03:00
&ctx.primary.sim,
&ctx.primary.map,
ctx.primary.current_selection,
)
2018-10-03 17:47:59 +03:00
}),
Box::new(|ctx| {
2018-10-22 05:11:20 +03:00
ctx.plugins
2018-10-03 17:47:59 +03:00
.color_picker
2018-10-22 05:11:20 +03:00
.event(ctx.input, ctx.canvas, ctx.cs)
2018-10-03 17:47:59 +03:00
}),
2018-10-22 05:59:37 +03:00
Box::new(|ctx| ctx.primary_plugins.steepness_viz.event(ctx.input)),
2018-10-22 05:11:20 +03:00
Box::new(|ctx| ctx.plugins.osm_classifier.event(ctx.input)),
2018-10-03 17:47:59 +03:00
Box::new(|ctx| {
2018-10-22 05:59:37 +03:00
ctx.primary_plugins
.hider
2018-10-22 05:11:20 +03:00
.event(ctx.input, &mut ctx.primary.current_selection)
}),
Box::new(|ctx| {
2018-10-22 05:59:37 +03:00
ctx.primary_plugins.debug_objects.event(
2018-10-22 05:11:20 +03:00
ctx.primary.current_selection,
2018-10-03 17:47:59 +03:00
ctx.input,
2018-10-22 05:11:20 +03:00
&ctx.primary.map,
&mut ctx.primary.sim,
&ctx.primary.control_map,
2018-10-03 17:47:59 +03:00
)
}),
2018-10-03 17:47:59 +03:00
Box::new(|ctx| {
2018-10-22 05:59:37 +03:00
ctx.primary_plugins.floodfiller.event(
2018-10-22 05:11:20 +03:00
&ctx.primary.map,
ctx.input,
2018-10-22 05:11:20 +03:00
ctx.primary.current_selection,
)
}),
2018-10-03 17:47:59 +03:00
Box::new(|ctx| {
2018-10-22 05:59:37 +03:00
ctx.primary_plugins.geom_validator.event(
2018-10-03 17:47:59 +03:00
ctx.input,
2018-10-22 05:11:20 +03:00
ctx.canvas,
&ctx.primary.map,
&ctx.primary.sim,
&ctx.primary.draw_map,
2018-09-13 20:50:59 +03:00
)
}),
2018-10-03 17:47:59 +03:00
Box::new(|ctx| {
2018-10-22 05:59:37 +03:00
ctx.primary_plugins
2018-10-03 17:47:59 +03:00
.turn_cycler
2018-10-22 05:11:20 +03:00
.event(ctx.input, ctx.primary.current_selection)
}),
2018-10-03 17:47:59 +03:00
Box::new(|ctx| {
2018-10-22 05:59:37 +03:00
ctx.primary_plugins.draw_neighborhoods.event(
ctx.input,
2018-10-22 05:11:20 +03:00
ctx.canvas,
&ctx.primary.map,
ctx.osd,
)
}),
2018-10-03 17:47:59 +03:00
Box::new(|ctx| {
2018-10-22 05:59:37 +03:00
ctx.primary_plugins.scenarios.event(
ctx.input,
&ctx.primary.map,
&mut ctx.primary.sim,
)
}),
2018-10-03 17:47:59 +03:00
Box::new(|ctx| {
2018-10-22 05:59:37 +03:00
let (active, new_primary) = ctx.primary_plugins.edits_manager.event(
2018-10-03 17:47:59 +03:00
ctx.input,
2018-10-22 05:11:20 +03:00
&ctx.primary.map,
&ctx.primary.control_map,
2018-10-22 05:59:37 +03:00
&ctx.primary_plugins.road_editor,
2018-10-22 05:11:20 +03:00
&mut ctx.primary.current_flags,
ctx.kml,
);
2018-10-22 05:59:37 +03:00
if let Some((state, plugins)) = new_primary {
*ctx.primary = state;
*ctx.primary_plugins = plugins;
}
active
2018-10-03 17:47:59 +03:00
}),
2018-10-13 21:34:00 +03:00
Box::new(|ctx| {
2018-10-22 05:59:37 +03:00
ctx.primary_plugins.chokepoints.event(
ctx.input,
&ctx.primary.sim,
&ctx.primary.map,
)
2018-10-13 21:34:00 +03:00
}),
Box::new(|ctx| {
2018-10-22 05:11:20 +03:00
let (active, new_ui) = ctx.plugins.ab_test_manager.event(
2018-10-14 03:31:30 +03:00
ctx.input,
2018-10-22 05:11:20 +03:00
ctx.primary.current_selection,
&ctx.primary.map,
ctx.kml,
&ctx.primary.current_flags,
2018-10-14 03:31:30 +03:00
);
2018-10-22 05:59:37 +03:00
if let Some(((new_primary, new_primary_plugins), new_secondary)) = new_ui {
2018-10-22 05:11:20 +03:00
*ctx.primary = new_primary;
2018-10-22 05:59:37 +03:00
*ctx.primary_plugins = new_primary_plugins;
2018-10-22 05:11:20 +03:00
*ctx.secondary = Some(new_secondary);
}
active
}),
2018-10-22 05:11:20 +03:00
Box::new(|ctx| ctx.plugins.logs.event(ctx.input)),
Box::new(|ctx| {
2018-10-22 05:23:47 +03:00
ctx.plugins
.diff_worlds
.event(ctx.input, &ctx.primary, ctx.secondary)
}),
Box::new(|ctx| {
2018-10-22 05:59:37 +03:00
ctx.primary_plugins
2018-10-22 05:23:47 +03:00
.show_owner
.event(ctx.primary.current_selection, &ctx.primary.sim);
// TODO This is a weird exception -- this plugin doesn't consume input, so
// never treat it as active for blocking input
false
}),
],*/
};
2018-10-22 05:23:47 +03:00
match abstutil::read_json::<EditorState>("editor_state") {
Ok(ref state) if ui.primary.map.get_name().to_string() == state.map_name => {
info!("Loaded previous editor_state");
ui.canvas.cam_x = state.cam_x;
ui.canvas.cam_y = state.cam_y;
ui.canvas.cam_zoom = state.cam_zoom;
}
_ => {
warn!("Couldn't load editor_state or it's for a different map, so just centering initial view");
ui.canvas.center_on_map_pt(ui.primary.draw_map.center_pt);
}
}
2018-10-22 05:23:47 +03:00
ui.plugins.layers.handle_zoom(-1.0, ui.canvas.cam_zoom);
2018-10-22 05:23:47 +03:00
ui
}
2018-07-06 20:02:01 +03:00
fn mouseover_something(&self) -> Option<ID> {
let pt = self.canvas.get_cursor_in_map_space();
2018-06-23 19:01:53 +03:00
let (statics, dynamics) = self.primary.draw_map.get_objects_onscreen(
self.canvas.get_screen_bbox(),
2018-10-22 05:59:37 +03:00
&self.primary_plugins.hider,
&self.primary.map,
&self.primary.sim,
&self.plugins.layers,
self,
);
// Check front-to-back
for obj in dynamics.into_iter() {
if obj.contains_pt(pt) {
return Some(obj.get_id());
2018-06-23 19:01:53 +03:00
}
}
for obj in statics.into_iter().rev() {
if obj.contains_pt(pt) {
return Some(obj.get_id());
2018-09-14 00:09:55 +03:00
}
}
2018-06-23 19:01:53 +03:00
None
}
fn color_obj(&self, id: ID) -> Option<Color> {
if Some(id) == self.primary.current_selection {
return Some(self.cs.get(Colors::Selected));
}
let ctx = Ctx {
cs: &self.cs,
map: &self.primary.map,
control_map: &self.primary.control_map,
canvas: &self.canvas,
sim: &self.primary.sim,
};
if let Some(p) = self.get_active_plugin() {
return p.color_for(id, ctx);
}
// TODO Ew, this is a weird ambient plugin that doesn't consume input but has an opinion on
// color.
2018-10-22 05:59:37 +03:00
self.primary_plugins.show_owner.color_for(id, ctx)
}
fn get_active_plugin(&self) -> Option<Box<&Colorizer>> {
let idx = self.active_plugin?;
// Match instead of array, because can't move the Box out of the temporary vec. :\
2018-10-22 05:23:47 +03:00
// This must line up with the list of plugins in UI::new.
match idx {
0 => Some(Box::new(&self.plugins.layers)),
2018-10-22 05:59:37 +03:00
1 => Some(Box::new(&self.primary_plugins.traffic_signal_editor)),
2 => Some(Box::new(&self.primary_plugins.stop_sign_editor)),
3 => Some(Box::new(&self.primary_plugins.road_editor)),
4 => Some(Box::new(&self.plugins.search_state)),
5 => Some(Box::new(&self.plugins.warp)),
2018-10-22 05:59:37 +03:00
6 => Some(Box::new(&self.primary_plugins.follow)),
7 => Some(Box::new(&self.primary_plugins.show_route)),
8 => Some(Box::new(&self.plugins.color_picker)),
2018-10-22 05:59:37 +03:00
9 => Some(Box::new(&self.primary_plugins.steepness_viz)),
10 => Some(Box::new(&self.plugins.osm_classifier)),
2018-10-22 05:59:37 +03:00
11 => Some(Box::new(&self.primary_plugins.hider)),
12 => Some(Box::new(&self.primary_plugins.debug_objects)),
13 => Some(Box::new(&self.primary_plugins.floodfiller)),
14 => Some(Box::new(&self.primary_plugins.geom_validator)),
15 => Some(Box::new(&self.primary_plugins.turn_cycler)),
16 => Some(Box::new(&self.primary_plugins.draw_neighborhoods)),
17 => Some(Box::new(&self.primary_plugins.scenarios)),
18 => Some(Box::new(&self.primary_plugins.edits_manager)),
19 => Some(Box::new(&self.primary_plugins.chokepoints)),
20 => Some(Box::new(&self.plugins.ab_test_manager)),
21 => Some(Box::new(&self.plugins.logs)),
22 => Some(Box::new(&self.plugins.diff_worlds)),
2018-10-22 05:59:37 +03:00
23 => Some(Box::new(&self.primary_plugins.show_owner)),
_ => panic!("Active plugin {} is too high", idx),
}
}
fn run_plugin(&mut self, idx: usize, input: &mut UserInput, osd: &mut Text) -> bool {
let ctx = PluginCtx {
primary: &mut self.primary,
secondary: &mut self.secondary,
canvas: &mut self.canvas,
cs: &mut self.cs,
input,
osd,
kml: &self.kml,
};
match idx {
0 => self.plugins.layers.event(ctx),
1 => self.primary_plugins.traffic_signal_editor.event(ctx),
/*2 => self.primary_plugins.stop_sign_editor.event(ctx),
3 => self.primary_plugins.road_editor.event(ctx),
4 => self.plugins.search_state.event(ctx),
5 => self.plugins.warp.event(ctx),
6 => self.primary_plugins.follow.event(ctx),
7 => self.primary_plugins.show_route.event(ctx),
8 => self.plugins.color_picker.event(ctx),
9 => self.primary_plugins.steepness_viz.event(ctx),
10 => self.plugins.osm_classifier.event(ctx),
11 => self.primary_plugins.hider.event(ctx),
12 => self.primary_plugins.debug_objects.event(ctx),
13 => self.primary_plugins.floodfiller.event(ctx),
14 => self.primary_plugins.geom_validator.event(ctx),
15 => self.primary_plugins.turn_cycler.event(ctx),
16 => self.primary_plugins.draw_neighborhoods.event(ctx),
17 => self.primary_plugins.scenarios.event(ctx),
18 => self.primary_plugins.edits_manager.event(ctx),
19 => self.primary_plugins.chokepoints.event(ctx),
20 => self.plugins.ab_test_manager.event(ctx),
21 => self.plugins.logs.event(ctx),
22 => self.plugins.diff_worlds.event(ctx),
23 => self.primary_plugins.show_owner.event(ctx),*/
_ => panic!("Plugin {} is too high", idx),
}
}
}
2018-06-22 21:01:44 +03:00
#[derive(Serialize, Deserialize, Debug)]
pub struct EditorState {
pub map_name: String,
2018-06-22 21:01:44 +03:00
pub cam_x: f64,
pub cam_y: f64,
pub cam_zoom: f64,
}
pub trait ShowTurnIcons {
fn show_icons_for(&self, id: IntersectionID) -> bool;
}
impl ShowTurnIcons for UI {
fn show_icons_for(&self, id: IntersectionID) -> bool {
self.plugins.layers.show_all_turn_icons.is_enabled()
2018-10-22 05:59:37 +03:00
|| self.primary_plugins.stop_sign_editor.show_turn_icons(id)
|| self
.primary_plugins
.traffic_signal_editor
.show_turn_icons(id)
}
}
2018-10-03 17:47:59 +03:00
2018-10-22 05:11:20 +03:00
// TODO I can't help but noticing this is just UI but with references. Can we be more direct?
pub struct PluginCtx<'a> {
pub primary: &'a mut PerMapUI,
pub secondary: &'a mut Option<(PerMapUI, PluginsPerMap)>,
pub canvas: &'a mut Canvas,
pub cs: &'a mut ColorScheme,
pub input: &'a mut UserInput,
pub osd: &'a mut Text,
pub kml: &'a Option<String>,
2018-10-03 17:47:59 +03:00
}