close intersections from signal and stop sign editor

This commit is contained in:
Dustin Carlino 2020-02-05 16:34:42 -08:00
parent 2f2f0506fa
commit 950238af81
4 changed files with 57 additions and 52 deletions

View File

@ -1,17 +1,14 @@
use crate::colors; use crate::colors;
use crate::common::{Colorer, CommonState}; use crate::common::CommonState;
use crate::edit::apply_map_edits; use crate::edit::apply_map_edits;
use crate::game::{msg, State, Transition, WizardState}; use crate::game::{msg, State, Transition, WizardState};
use crate::helpers::ID;
use crate::managed::WrappedComposite; use crate::managed::WrappedComposite;
use crate::ui::UI; use crate::ui::UI;
use ezgui::{ use ezgui::{
hotkey, Button, Choice, Color, Composite, EventCtx, GfxCtx, HorizontalAlignment, Key, Line, hotkey, Button, Choice, Color, Composite, EventCtx, GfxCtx, HorizontalAlignment, Key, Line,
ManagedWidget, Outcome, RewriteColor, Text, VerticalAlignment, ManagedWidget, Outcome, RewriteColor, Text, VerticalAlignment,
}; };
use map_model::{ use map_model::{EditCmd, LaneID, LaneType, Map, RoadID};
connectivity, EditCmd, IntersectionType, LaneID, LaneType, Map, PathConstraints, RoadID,
};
use std::collections::BTreeSet; use std::collections::BTreeSet;
pub struct LaneEditor { pub struct LaneEditor {
@ -157,47 +154,6 @@ impl State for LaneEditor {
None => {} None => {}
} }
/*// Woo, a special case! The construction tool also applies to intersections.
if let Some(ID::Intersection(i)) = ui.primary.current_selection {
if self.brush == Brush::Construction
&& ui
.per_obj
.action(ctx, Key::Space, "closed for construction")
{
let it = ui.primary.map.get_i(i).intersection_type;
if it != IntersectionType::Construction && it != IntersectionType::Border {
let mut edits = ui.primary.map.get_edits().clone();
edits
.commands
.push(EditCmd::CloseIntersection { id: i, orig_it: it });
apply_map_edits(ctx, ui, edits);
let (_, disconnected) =
connectivity::find_scc(&ui.primary.map, PathConstraints::Pedestrian);
if !disconnected.is_empty() {
let mut edits = ui.primary.map.get_edits().clone();
edits.commands.pop();
apply_map_edits(ctx, ui, edits);
let mut err_state = msg(
"Error",
vec![format!("{} sidewalks disconnected", disconnected.len())],
);
let color = ui.cs.get("unreachable lane");
let mut c = Colorer::new(Text::new(), vec![("", color)]);
for l in disconnected {
c.add_l(l, color, &ui.primary.map);
}
err_state.downcast_mut::<WizardState>().unwrap().also_draw =
Some(c.build_zoomed(ctx, ui));
return Some(Transition::Push(err_state));
}
}
}
}*/
Transition::Keep Transition::Keep
} }

View File

@ -5,9 +5,9 @@ mod traffic_signals;
pub use self::lanes::LaneEditor; pub use self::lanes::LaneEditor;
pub use self::stop_signs::StopSignEditor; pub use self::stop_signs::StopSignEditor;
pub use self::traffic_signals::TrafficSignalEditor; pub use self::traffic_signals::TrafficSignalEditor;
use crate::common::{tool_panel, CommonState, Overlays, Warping}; use crate::common::{tool_panel, Colorer, CommonState, Overlays, Warping};
use crate::debug::DebugMode; use crate::debug::DebugMode;
use crate::game::{State, Transition, WizardState}; use crate::game::{msg, State, Transition, WizardState};
use crate::helpers::ID; use crate::helpers::ID;
use crate::managed::{WrappedComposite, WrappedOutcome}; use crate::managed::{WrappedComposite, WrappedOutcome};
use crate::render::{DrawIntersection, DrawLane, DrawRoad}; use crate::render::{DrawIntersection, DrawLane, DrawRoad};
@ -15,7 +15,9 @@ use crate::sandbox::{GameplayMode, SandboxMode};
use crate::ui::UI; use crate::ui::UI;
use abstutil::Timer; use abstutil::Timer;
use ezgui::{hotkey, lctrl, Choice, EventCtx, GfxCtx, Key, Line, ModalMenu, Text, WrappedWizard}; use ezgui::{hotkey, lctrl, Choice, EventCtx, GfxCtx, Key, Line, ModalMenu, Text, WrappedWizard};
use map_model::{EditCmd, LaneID, LaneType, MapEdits}; use map_model::{
connectivity, EditCmd, IntersectionID, LaneID, LaneType, MapEdits, PathConstraints,
};
use sim::Sim; use sim::Sim;
use std::collections::BTreeSet; use std::collections::BTreeSet;
@ -131,7 +133,9 @@ impl State for EditMode {
self.suspended_sim.clone(), self.suspended_sim.clone(),
))); )));
} }
if ui.primary.map.get_i(id).is_closed() && ui.per_obj.action(ctx, Key::R, "revert") { if ui.primary.map.get_i(id).is_closed()
&& ui.per_obj.left_click(ctx, "re-open closed intersection")
{
let mut edits = ui.primary.map.get_edits().clone(); let mut edits = ui.primary.map.get_edits().clone();
edits edits
.commands .commands
@ -316,3 +320,37 @@ pub fn can_edit_lane(mode: &GameplayMode, l: LaneID, ui: &UI) -> bool {
&& !ui.primary.map.get_l(l).is_sidewalk() && !ui.primary.map.get_l(l).is_sidewalk()
&& ui.primary.map.get_l(l).lane_type != LaneType::SharedLeftTurn && ui.primary.map.get_l(l).lane_type != LaneType::SharedLeftTurn
} }
pub fn close_intersection(ctx: &mut EventCtx, ui: &mut UI, i: IntersectionID) -> Transition {
let it = ui.primary.map.get_i(i).intersection_type;
let mut edits = ui.primary.map.get_edits().clone();
edits
.commands
.push(EditCmd::CloseIntersection { id: i, orig_it: it });
apply_map_edits(ctx, ui, edits);
let (_, disconnected) = connectivity::find_scc(&ui.primary.map, PathConstraints::Pedestrian);
if disconnected.is_empty() {
// Success! Quit the stop sign / signal editor.
return Transition::Pop;
}
let mut edits = ui.primary.map.get_edits().clone();
edits.commands.pop();
apply_map_edits(ctx, ui, edits);
let mut err_state = msg(
"Error",
vec![format!("{} sidewalks disconnected", disconnected.len())],
);
let color = ui.cs.get("unreachable lane");
let mut c = Colorer::new(Text::new(), vec![("", color)]);
for l in disconnected {
c.add_l(l, color, &ui.primary.map);
}
err_state.downcast_mut::<WizardState>().unwrap().also_draw = Some(c.build_zoomed(ctx, ui));
Transition::Push(err_state)
}

View File

@ -1,5 +1,5 @@
use crate::common::CommonState; use crate::common::CommonState;
use crate::edit::apply_map_edits; use crate::edit::{apply_map_edits, close_intersection};
use crate::game::{State, Transition}; use crate::game::{State, Transition};
use crate::render::{DrawIntersection, DrawOptions}; use crate::render::{DrawIntersection, DrawOptions};
use crate::ui::{ShowEverything, UI}; use crate::ui::{ShowEverything, UI};
@ -39,6 +39,7 @@ impl StopSignEditor {
vec![ vec![
(hotkey(Key::Escape), "quit"), (hotkey(Key::Escape), "quit"),
(hotkey(Key::R), "reset to default"), (hotkey(Key::R), "reset to default"),
(None, "close intersection for construction"),
], ],
ctx, ctx,
), ),
@ -91,6 +92,8 @@ impl State for StopSignEditor {
self.id, self.id,
))); )));
apply_map_edits(ctx, ui, edits); apply_map_edits(ctx, ui, edits);
} else if self.menu.action("close intersection for construction") {
return close_intersection(ctx, ui, self.id);
} }
Transition::Keep Transition::Keep
} }

View File

@ -1,6 +1,6 @@
use crate::colors; use crate::colors;
use crate::common::CommonState; use crate::common::CommonState;
use crate::edit::apply_map_edits; use crate::edit::{apply_map_edits, close_intersection};
use crate::game::{msg, DrawBaselayer, State, Transition, WizardState}; use crate::game::{msg, DrawBaselayer, State, Transition, WizardState};
use crate::helpers::plain_list_names; use crate::helpers::plain_list_names;
use crate::managed::{WrappedComposite, WrappedOutcome}; use crate::managed::{WrappedComposite, WrappedOutcome};
@ -110,6 +110,9 @@ impl State for TrafficSignalEditor {
x if x == "Use preset" => { x if x == "Use preset" => {
return Transition::Push(change_preset(self.i)); return Transition::Push(change_preset(self.i));
} }
x if x == "Close intersection for construction" => {
return close_intersection(ctx, ui, self.i);
}
x if x == "Make all-walk" => { x if x == "Make all-walk" => {
let mut new_signal = orig_signal.clone(); let mut new_signal = orig_signal.clone();
if new_signal.convert_to_ped_scramble(&ui.primary.map) { if new_signal.convert_to_ped_scramble(&ui.primary.map) {
@ -443,6 +446,11 @@ fn make_diagram(i: IntersectionID, selected: usize, ui: &UI, ctx: &mut EventCtx)
hotkey(Key::B), hotkey(Key::B),
)); ));
} }
col.push(WrappedComposite::text_button(
ctx,
"Close intersection for construction",
None,
));
for (idx, phase) in signal.phases.iter().enumerate() { for (idx, phase) in signal.phases.iter().enumerate() {
col.push( col.push(