use std::collections::{BTreeSet, VecDeque};
use abstutil::Timer;
use geom::{Distance, Line, Polygon, Pt2D};
use map_gui::options::TrafficSignalStyle;
use map_gui::render::{traffic_signal, DrawMovement, DrawOptions};
use map_gui::tools::PopupMsg;
use map_model::{
ControlTrafficSignal, EditCmd, EditIntersection, IntersectionID, MovementID, Stage, StageType,
TurnPriority,
};
use widgetry::{
include_labeled_bytes, lctrl, Color, ControlState, DrawBaselayer, Drawable, EventCtx,
GeomBatch, GfxCtx, HorizontalAlignment, Key, Line, MultiButton, Outcome, Panel, RewriteColor,
State, Text, TextExt, VerticalAlignment, Widget,
};
use crate::app::{App, ShowEverything, Transition};
use crate::common::{CommonState, Warping};
use crate::edit::{apply_map_edits, ConfirmDiscard};
use crate::sandbox::GameplayMode;
mod edits;
mod offsets;
mod picker;
mod preview;
pub struct TrafficSignalEditor {
side_panel: Panel,
top_panel: Panel,
mode: GameplayMode,
members: BTreeSet<IntersectionID>,
current_stage: usize,
movements: Vec<DrawMovement>,
movement_selected: Option<(MovementID, Option<TurnPriority>)>,
draw_current: Drawable,
tooltip: Option<Text>,
command_stack: Vec<BundleEdits>,
redo_stack: Vec<BundleEdits>,
original: BundleEdits,
warn_changed: bool,
fade_irrelevant: Drawable,
}
#[derive(Clone, PartialEq)]
pub struct BundleEdits {
signals: Vec<ControlTrafficSignal>,
}
impl TrafficSignalEditor {
pub fn new(
ctx: &mut EventCtx,
app: &mut App,
members: BTreeSet<IntersectionID>,
mode: GameplayMode,
) -> Box<dyn State<App>> {
app.primary.current_selection = None;
let original = BundleEdits::get_current(app, &members);
let synced = BundleEdits::synchronize(app, &members);
let warn_changed = original != synced;
synced.apply(app);
let mut editor = TrafficSignalEditor {
side_panel: make_side_panel(ctx, app, &members, 0),
top_panel: make_top_panel(ctx, app, false, false),
mode,
current_stage: 0,
movements: Vec::new(),
movement_selected: None,
draw_current: Drawable::empty(ctx),
tooltip: None,
command_stack: Vec::new(),
redo_stack: Vec::new(),
warn_changed,
original,
fade_irrelevant: fade_irrelevant(app, &members).upload(ctx),
members,
};
editor.recalc_draw_current(ctx, app);
Box::new(editor)
}
fn change_stage(&mut self, ctx: &mut EventCtx, app: &App, idx: usize) {
if self.current_stage == idx {
let mut new = make_side_panel(ctx, app, &self.members, self.current_stage);
new.restore(ctx, &self.side_panel);
self.side_panel = new;
} else {
self.current_stage = idx;
self.side_panel = make_side_panel(ctx, app, &self.members, self.current_stage);
self.side_panel
.scroll_to_member(ctx, format!("change duration of stage {}", idx + 1));
}
self.recalc_draw_current(ctx, app);
}
fn add_new_edit<F: Fn(&mut ControlTrafficSignal)>(
&mut self,
ctx: &mut EventCtx,
app: &mut App,
idx: usize,
fxn: F,
) {
let mut bundle = BundleEdits::get_current(app, &self.members);
self.command_stack.push(bundle.clone());
self.redo_stack.clear();
for ts in &mut bundle.signals {
fxn(ts);
}
bundle.apply(app);
self.top_panel = make_top_panel(ctx, app, true, false);
self.change_stage(ctx, app, idx);
}
fn recalc_draw_current(&mut self, ctx: &mut EventCtx, app: &App) {
let mut batch = GeomBatch::new();
let mut movements = Vec::new();
for i in &self.members {
let stage = &app.primary.map.get_traffic_signal(*i).stages[self.current_stage];
for (m, draw) in DrawMovement::for_i(
ctx.prerender,
&app.primary.map,
&app.cs,
*i,
self.current_stage,
) {
if self
.movement_selected
.map(|(x, _)| x != m.id)
.unwrap_or(true)
|| m.id.crosswalk
{
batch.append(draw);
} else if !stage.protected_movements.contains(&m.id)
&& !stage.yield_movements.contains(&m.id)
{
batch.append(draw.color(RewriteColor::Change(
Color::hex("#7C7C7C"),
Color::hex("#72CE36"),
)));
}
movements.push(m);
}
traffic_signal::draw_stage_number(
app,
ctx.prerender,
*i,
self.current_stage,
&mut batch,
);
}
if let Some((selected, next_priority)) = self.movement_selected {
for m in &movements {
if m.id == selected {
m.draw_selected_movement(app, &mut batch, next_priority);
break;
}
}
}
self.draw_current = ctx.upload(batch);
self.movements = movements;
}
}
impl State<App> for TrafficSignalEditor {
fn event(&mut self, ctx: &mut EventCtx, app: &mut App) -> Transition {
self.tooltip = None;
if self.warn_changed {
self.warn_changed = false;
return Transition::Push(PopupMsg::new(
ctx,
"Note",
vec!["Some signals were modified to match the number and duration of stages"],
));
}
ctx.canvas_movement();
let canonical_signal = app
.primary
.map
.get_traffic_signal(*self.members.iter().next().unwrap());
let num_stages = canonical_signal.stages.len();
match self.side_panel.event(ctx) {
Outcome::Clicked(x) => {
if x == "Edit entire signal" {
return Transition::Push(edits::edit_entire_signal(
ctx,
app,
canonical_signal.id,
self.mode.clone(),
self.original.clone(),
));
}
if x == "Tune offsets between signals" {
return Transition::Push(offsets::ShowAbsolute::new(
ctx,
app,
self.members.clone(),
));
}
if x == "Add a new stage" {
self.add_new_edit(ctx, app, num_stages, |ts| {
ts.stages.push(Stage::new());
});
return Transition::Keep;
}
if let Some(x) = x.strip_prefix("change duration of stage ") {
let idx = x.parse::<usize>().unwrap() - 1;
return Transition::Push(edits::ChangeDuration::new(
ctx,
app,
&canonical_signal,
idx,
));
}
if let Some(x) = x.strip_prefix("delete stage ") {
let idx = x.parse::<usize>().unwrap() - 1;
self.add_new_edit(ctx, app, 0, |ts| {
ts.stages.remove(idx);
});
return Transition::Keep;
}
if let Some(x) = x.strip_prefix("move up stage ") {
let idx = x.parse::<usize>().unwrap() - 1;
self.add_new_edit(ctx, app, idx - 1, |ts| {
ts.stages.swap(idx, idx - 1);
});
return Transition::Keep;
}
if let Some(x) = x.strip_prefix("move down stage ") {
let idx = x.parse::<usize>().unwrap() - 1;
self.add_new_edit(ctx, app, idx + 1, |ts| {
ts.stages.swap(idx, idx + 1);
});
return Transition::Keep;
}
if let Some(x) = x.strip_prefix("stage ") {
let parts = x.split(", Intersection #").collect::<Vec<_>>();
let idx = parts[0].parse::<usize>().unwrap() - 1;
let i = IntersectionID(parts[1].parse::<usize>().unwrap());
self.change_stage(ctx, app, idx);
let center = app.primary.map.get_i(i).polygon.center();
if ctx.canvas.get_screen_bounds().contains(center) {
return Transition::Keep;
} else {
return Transition::Push(Warping::new(
ctx,
center,
Some(15.0),
None,
&mut app.primary,
));
}
}
unreachable!()
}
_ => {}
}
match self.top_panel.event(ctx) {
Outcome::Clicked(x) => match x.as_ref() {
"Finish" => {
if let Some(bundle) = check_for_missing_turns(app, &self.members) {
bundle.apply(app);
self.command_stack.push(bundle.clone());
self.redo_stack.clear();
self.top_panel = make_top_panel(ctx, app, true, false);
self.change_stage(ctx, app, 0);
return Transition::Push(PopupMsg::new(
ctx,
"Error: missing turns",
vec![
"Some turns are missing from this traffic signal",
"They've all been added as a new first stage. Please update your \
changes to include them.",
],
));
} else {
let changes = BundleEdits::get_current(app, &self.members);
self.original.apply(app);
changes.commit(ctx, app);
return Transition::Pop;
}
}
"Cancel" => {
if BundleEdits::get_current(app, &self.members) == self.original {
self.original.apply(app);
return Transition::Pop;
}
let original = self.original.clone();
return Transition::Push(ConfirmDiscard::new(
ctx,
Box::new(move |app| {
original.apply(app);
}),
));
}
"Edit multiple signals" => {
let changes = check_for_missing_turns(app, &self.members)
.unwrap_or_else(|| BundleEdits::get_current(app, &self.members));
self.original.apply(app);
changes.commit(ctx, app);
return Transition::Replace(picker::SignalPicker::new(
ctx,
self.members.clone(),
self.mode.clone(),
));
}
"Export" => {
for signal in BundleEdits::get_current(app, &self.members).signals {
let ts = signal.export(&app.primary.map);
abstio::write_json(
format!("traffic_signal_data/{}.json", ts.intersection_osm_node_id),
&ts,
);
}
}
"Preview" => {
app.primary
.map
.recalculate_pathfinding_after_edits(&mut Timer::throwaway());
return Transition::Push(preview::make_previewer(
ctx,
app,
self.members.clone(),
self.current_stage,
));
}
"undo" => {
self.redo_stack
.push(BundleEdits::get_current(app, &self.members));
self.command_stack.pop().unwrap().apply(app);
self.top_panel = make_top_panel(ctx, app, !self.command_stack.is_empty(), true);
self.change_stage(ctx, app, 0);
return Transition::Keep;
}
"redo" => {
self.command_stack
.push(BundleEdits::get_current(app, &self.members));
self.redo_stack.pop().unwrap().apply(app);
self.top_panel = make_top_panel(ctx, app, true, !self.redo_stack.is_empty());
self.change_stage(ctx, app, 0);
return Transition::Keep;
}
_ => unreachable!(),
},
_ => {}
}
{
if self.current_stage != 0 && ctx.input.pressed(Key::UpArrow) {
self.change_stage(ctx, app, self.current_stage - 1);
}
if self.current_stage != num_stages - 1 && ctx.input.pressed(Key::DownArrow) {
self.change_stage(ctx, app, self.current_stage + 1);
}
}
if ctx.redo_mouseover() {
let old = self.movement_selected.clone();
self.movement_selected = None;
if let Some(pt) = ctx.canvas.get_cursor_in_map_space() {
for m in &self.movements {
let signal = app.primary.map.get_traffic_signal(m.id.parent);
if m.hitbox.contains_pt(pt) {
let stage = &signal.stages[self.current_stage];
let next_priority = match stage.get_priority_of_movement(m.id) {
TurnPriority::Banned => {
if stage.could_be_protected(m.id, &signal.movements) {
Some(TurnPriority::Protected)
} else if m.id.crosswalk {
None
} else {
Some(TurnPriority::Yield)
}
}
TurnPriority::Yield => Some(TurnPriority::Banned),
TurnPriority::Protected => {
if m.id.crosswalk {
Some(TurnPriority::Banned)
} else {
Some(TurnPriority::Yield)
}
}
};
self.movement_selected = Some((m.id, next_priority));
break;
}
}
}
if self.movement_selected != old {
self.change_stage(ctx, app, self.current_stage);
}
}
if let Some((id, next_priority)) = self.movement_selected {
if let Some(pri) = next_priority {
let signal = app.primary.map.get_traffic_signal(id.parent);
let mut txt = Text::new();
txt.add_line(Line(format!(
"{} {}",
match signal.stages[self.current_stage].get_priority_of_movement(id) {
TurnPriority::Protected => "Protected",
TurnPriority::Yield => "Yielding",
TurnPriority::Banned => "Forbidden",
},
if id.crosswalk { "crosswalk" } else { "turn" },
)));
txt.add_appended(vec![
Line("Click").fg(ctx.style().text_hotkey_color),
Line(format!(
" to {}",
match pri {
TurnPriority::Protected => "add it as protected",
TurnPriority::Yield => "allow it after yielding",
TurnPriority::Banned => "forbid it",
}
)),
]);
self.tooltip = Some(txt);
if app.per_obj.left_click(
ctx,
format!(
"toggle from {:?} to {:?}",
signal.stages[self.current_stage].get_priority_of_movement(id),
pri
),
) {
let idx = self.current_stage;
let signal = signal.clone();
self.add_new_edit(ctx, app, idx, |ts| {
if ts.id == id.parent {
ts.stages[idx].edit_movement(&signal.movements[&id], pri);
}
});
return Transition::KeepWithMouseover;
}
}
}
Transition::Keep
}
fn draw_baselayer(&self) -> DrawBaselayer {
DrawBaselayer::Custom
}
fn draw(&self, g: &mut GfxCtx, app: &App) {
{
let mut opts = DrawOptions::new();
opts.suppress_traffic_signal_details
.extend(self.members.clone());
app.draw(g, opts, &ShowEverything::new());
}
g.redraw(&self.fade_irrelevant);
g.redraw(&self.draw_current);
self.top_panel.draw(g);
self.side_panel.draw(g);
if let Some((id, _)) = self.movement_selected {
let osd = if id.crosswalk {
Text::from(format!(
"Crosswalk across {}",
app.primary
.map
.get_r(id.from.id)
.get_name(app.opts.language.as_ref())
))
} else {
Text::from(format!(
"Turn from {} to {}",
app.primary
.map
.get_r(id.from.id)
.get_name(app.opts.language.as_ref()),
app.primary
.map
.get_r(id.to.id)
.get_name(app.opts.language.as_ref())
))
};
CommonState::draw_custom_osd(g, app, osd);
} else {
CommonState::draw_osd(g, app);
}
if let Some(txt) = self.tooltip.clone() {
g.draw_mouse_tooltip(txt);
}
}
}
fn make_top_panel(ctx: &mut EventCtx, app: &App, can_undo: bool, can_redo: bool) -> Panel {
let row = vec![
ctx.style()
.btn_solid_primary
.text("Finish")
.hotkey(Key::Enter)
.build_def(ctx),
ctx.style()
.btn_outline
.text("Preview")
.hotkey(lctrl(Key::P))
.build_def(ctx),
ctx.style()
.btn_plain
.icon("system/assets/tools/undo.svg")
.disabled(!can_undo)
.hotkey(lctrl(Key::Z))
.build_widget(ctx, "undo"),
ctx.style()
.btn_plain
.icon("system/assets/tools/redo.svg")
.disabled(!can_redo)
.hotkey(lctrl(Key::Y))
.build_widget(ctx, "redo"),
ctx.style()
.btn_plain_destructive
.text("Cancel")
.hotkey(Key::Escape)
.build_def(ctx)
.align_right(),
];
Panel::new(Widget::col(vec![
Widget::row(vec![
Line("Traffic signal editor")
.small_heading()
.into_widget(ctx),
ctx.style()
.btn_plain
.text("+ Edit multiple")
.label_color(Color::hex("#4CA7E9"), ControlState::Default)
.hotkey(Key::M)
.build_widget(ctx, "Edit multiple signals"),
]),
Widget::row(row),
if app.opts.dev {
ctx.style()
.btn_outline
.text("Export")
.tooltip(Text::from_multiline(vec![
Line("This will create a JSON file in traffic_signal_data/.").small(),
Line(
"Contribute this to map how this traffic signal is currently timed in \
real life.",
)
.small(),
]))
.build_def(ctx)
} else {
Widget::nothing()
},
]))
.aligned(HorizontalAlignment::Center, VerticalAlignment::Top)
.build(ctx)
}
fn make_side_panel(
ctx: &mut EventCtx,
app: &App,
members: &BTreeSet<IntersectionID>,
selected: usize,
) -> Panel {
let map = &app.primary.map;
let canonical_signal = map.get_traffic_signal(*members.iter().next().unwrap());
let mut txt = Text::new();
if members.len() == 1 {
let i = *members.iter().next().unwrap();
txt.add_line(Line(i.to_string()).big_heading_plain());
let mut road_names = BTreeSet::new();
for r in &app.primary.map.get_i(i).roads {
road_names.insert(
app.primary
.map
.get_r(*r)
.get_name(app.opts.language.as_ref()),
);
}
for r in road_names {
txt.add_line(Line(format!(" {}", r)).secondary());
}
} else {
txt.add_line(Line(format!("{} intersections", members.len())).big_heading_plain());
txt.add_line(
Line(
members
.iter()
.map(|i| format!("#{}", i.0))
.collect::<Vec<_>>()
.join(", "),
)
.secondary(),
);
}
let mut col = vec![txt.into_widget(ctx)];
col.push(Widget::horiz_separator(ctx, 0.2));
col.push(
format!(
"One full cycle lasts {}",
canonical_signal.simple_cycle_duration()
)
.text_widget(ctx),
);
if members.len() == 1 {
col.push(
ctx.style()
.btn_outline
.text("Edit entire signal")
.hotkey(Key::E)
.build_def(ctx),
);
} else {
col.push(
ctx.style()
.btn_outline
.text("Tune offsets between signals")
.hotkey(Key::O)
.build_def(ctx),
);
}
let translations = squish_polygons_together(
members
.iter()
.map(|i| app.primary.map.get_i(*i).polygon.clone())
.collect(),
);
for (idx, canonical_stage) in canonical_signal.stages.iter().enumerate() {
let stage_btn = draw_multiple_signals(ctx, app, members, idx, &translations);
let up_button = ctx
.style()
.btn_plain
.icon_bytes(include_labeled_bytes!(
"../../../../widgetry/icons/arrow_up.svg"
))
.disabled(idx == 0);
let down_button = ctx
.style()
.btn_plain
.icon_bytes(include_labeled_bytes!(
"../../../../widgetry/icons/arrow_down.svg"
))
.disabled(idx == canonical_signal.stages.len() - 1);
let stage_controls = Widget::row(vec![
Widget::col(vec![
up_button.build_widget(ctx, format!("move up stage {}", idx + 1)),
down_button.build_widget(ctx, format!("move down stage {}", idx + 1)),
])
.centered_vert(),
Widget::col(vec![
Widget::row(vec![
match canonical_stage.stage_type {
StageType::Fixed(d) => format!("Stage {}: {}", idx + 1, d),
StageType::Variable(min, delay, additional) => format!(
"Stage {}: {}, {}, {} (variable)",
idx + 1,
min,
delay,
additional
),
}
.text_widget(ctx)
.centered_vert(),
{
let mut button =
ctx.style().btn_plain.icon("system/assets/tools/pencil.svg");
if selected == idx {
button = button.hotkey(Key::X);
}
button
.build_widget(ctx, format!("change duration of stage {}", idx + 1))
.align_right()
},
if canonical_signal.stages.len() > 1 {
ctx.style()
.btn_solid_destructive
.icon("system/assets/tools/trash.svg")
.build_widget(ctx, format!("delete stage {}", idx + 1))
.align_right()
} else {
Widget::nothing()
},
]),
stage_btn,
]),
])
.padding(10)
.bg(app.cs.inner_panel_bg);
if idx == selected {
col.push(stage_controls.outline(ctx.style().btn_outline.outline));
} else {
col.push(stage_controls);
}
}
col.push(
ctx.style()
.btn_outline
.text("Add a new stage")
.build_def(ctx)
.centered_horiz(),
);
Panel::new(Widget::col(col))
.aligned(HorizontalAlignment::Left, VerticalAlignment::Top)
.exact_size_percent(30, 85)
.build(ctx)
}
impl BundleEdits {
fn apply(&self, app: &mut App) {
for s in &self.signals {
app.primary.map.incremental_edit_traffic_signal(s.clone());
}
}
fn commit(self, ctx: &mut EventCtx, app: &mut App) {
if self == BundleEdits::get_current(app, &self.signals.iter().map(|s| s.id).collect()) {
return;
}
let mut edits = app.primary.map.get_edits().clone();
for signal in self.signals {
edits.commands.push(EditCmd::ChangeIntersection {
i: signal.id,
old: app.primary.map.get_i_edit(signal.id),
new: EditIntersection::TrafficSignal(signal.export(&app.primary.map)),
});
}
apply_map_edits(ctx, app, edits);
}
fn get_current(app: &App, members: &BTreeSet<IntersectionID>) -> BundleEdits {
let signals = members
.iter()
.map(|i| app.primary.map.get_traffic_signal(*i).clone())
.collect();
BundleEdits { signals }
}
fn synchronize(app: &App, members: &BTreeSet<IntersectionID>) -> BundleEdits {
let map = &app.primary.map;
let canonical = map.get_traffic_signal(
*members
.iter()
.max_by_key(|i| map.get_traffic_signal(**i).stages.len())
.unwrap(),
);
let mut signals = Vec::new();
for i in members {
let mut signal = map.get_traffic_signal(*i).clone();
for (idx, canonical_stage) in canonical.stages.iter().enumerate() {
if signal.stages.len() == idx {
signal.stages.push(Stage::new());
}
signal.stages[idx].stage_type = canonical_stage.stage_type.clone();
}
signals.push(signal);
}
BundleEdits { signals }
}
}
fn check_for_missing_turns(app: &App, members: &BTreeSet<IntersectionID>) -> Option<BundleEdits> {
let mut all_missing = BTreeSet::new();
for i in members {
all_missing.extend(app.primary.map.get_traffic_signal(*i).missing_turns());
}
if all_missing.is_empty() {
return None;
}
let mut bundle = BundleEdits::get_current(app, members);
for signal in &mut bundle.signals {
let mut stage = Stage::new();
for m in &all_missing {
if m.parent != signal.id {
continue;
}
if m.crosswalk {
stage.protected_movements.insert(*m);
} else {
stage.yield_movements.insert(*m);
}
}
signal.stages.insert(0, stage);
}
Some(bundle)
}
fn draw_multiple_signals(
ctx: &mut EventCtx,
app: &App,
members: &BTreeSet<IntersectionID>,
idx: usize,
translations: &Vec<(f64, f64)>,
) -> Widget {
let mut batch = GeomBatch::new();
for (i, (dx, dy)) in members.iter().zip(translations) {
let mut piece = GeomBatch::new();
piece.push(
app.cs.normal_intersection,
app.primary.map.get_i(*i).polygon.clone(),
);
traffic_signal::draw_signal_stage(
ctx.prerender,
&app.primary.map.get_traffic_signal(*i).stages[idx],
idx,
*i,
None,
&mut piece,
app,
TrafficSignalStyle::Yuwen,
);
batch.append(piece.translate(*dx, *dy));
}
let bounds_before = batch.get_bounds();
batch = batch.autocrop();
let bounds = batch.get_bounds();
let zoom = (300.0 / bounds.width()).min(300.0 / bounds.height());
let batch = batch.scale(zoom);
let mut hitboxes = Vec::new();
for (i, (dx, dy)) in members.iter().zip(translations) {
hitboxes.push((
app.primary
.map
.get_i(*i)
.polygon
.clone()
.translate(*dx - bounds_before.min_x, *dy - bounds_before.min_y)
.scale(zoom),
format!("stage {}, {}", idx + 1, i),
));
}
MultiButton::new(ctx, batch, hitboxes).named(format!("stage {}", idx + 1))
}
fn squish_polygons_together(mut polygons: Vec<Polygon>) -> Vec<(f64, f64)> {
if polygons.len() == 1 {
return vec![(0.0, 0.0)];
}
let step_size = 0.8
* polygons.iter().fold(std::f64::MAX, |x, p| {
x.min(p.get_bounds().width()).min(p.get_bounds().height())
});
let mut translations: Vec<(f64, f64)> =
std::iter::repeat((0.0, 0.0)).take(polygons.len()).collect();
let mut indices: VecDeque<usize> = (0..polygons.len()).collect();
let mut attempts = 0;
while !indices.is_empty() {
let idx = indices.pop_front().unwrap();
let center = Pt2D::center(&polygons.iter().map(|p| p.center()).collect());
let angle = Line::must_new(polygons[idx].center(), center).angle();
let pt = Pt2D::new(0.0, 0.0).project_away(Distance::meters(step_size), angle);
let translated = polygons[idx].translate(pt.x(), pt.y());
if polygons
.iter()
.enumerate()
.any(|(i, p)| i != idx && !translated.intersection(p).is_empty())
{
} else {
translations[idx].0 += pt.x();
translations[idx].1 += pt.y();
polygons[idx] = translated;
indices.push_back(idx);
}
attempts += 1;
if attempts == 100 {
break;
}
}
translations
}
pub fn fade_irrelevant(app: &App, members: &BTreeSet<IntersectionID>) -> GeomBatch {
let mut holes = Vec::new();
for i in members {
let i = app.primary.map.get_i(*i);
holes.push(i.polygon.clone());
for r in &i.roads {
holes.push(
app.primary
.map
.get_r(*r)
.get_thick_polygon(&app.primary.map),
);
}
}
let fade_area = Polygon::with_holes(
app.primary.map.get_boundary_polygon().clone().into_ring(),
vec![Polygon::convex_hull(holes).into_ring()],
);
GeomBatch::from(vec![(app.cs.fade_map_dark, fade_area)])
}