mirror of
https://github.com/a-b-street/abstreet.git
synced 2025-01-07 06:57:25 +03:00
Come take a ride on my skele-skeletonvator
(I am so sorry)
This commit is contained in:
parent
82865b4e08
commit
4935ea4fd6
@ -198,7 +198,7 @@ impl ElevationContours {
|
|||||||
high = high.max(i.elevation);
|
high = high.max(i.elevation);
|
||||||
}
|
}
|
||||||
|
|
||||||
let (closest_elevation, unzoomed) = make_contours(ctx, app, low, high);
|
let (closest_elevation, unzoomed) = make_elevation_contours(ctx, app, low, high);
|
||||||
|
|
||||||
let panel = Panel::new_builder(Widget::col(vec![
|
let panel = Panel::new_builder(Widget::col(vec![
|
||||||
header(ctx, "Elevation"),
|
header(ctx, "Elevation"),
|
||||||
@ -221,7 +221,7 @@ impl ElevationContours {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn make_contours(
|
pub fn make_elevation_contours(
|
||||||
ctx: &mut EventCtx,
|
ctx: &mut EventCtx,
|
||||||
app: &App,
|
app: &App,
|
||||||
low: Distance,
|
low: Distance,
|
||||||
|
@ -8,7 +8,7 @@ use widgetry::{
|
|||||||
use crate::app::{App, Transition};
|
use crate::app::{App, Transition};
|
||||||
use crate::sandbox::dashboards;
|
use crate::sandbox::dashboards;
|
||||||
|
|
||||||
mod elevation;
|
pub mod elevation;
|
||||||
pub mod favorites;
|
pub mod favorites;
|
||||||
pub mod map;
|
pub mod map;
|
||||||
mod pandemic;
|
mod pandemic;
|
||||||
|
@ -4,7 +4,7 @@ use map_gui::ID;
|
|||||||
use map_model::{LaneType, PathConstraints, Road, RoadID};
|
use map_model::{LaneType, PathConstraints, Road, RoadID};
|
||||||
use widgetry::{
|
use widgetry::{
|
||||||
lctrl, Cached, Color, Drawable, EventCtx, GeomBatch, GeomBatchStack, GfxCtx,
|
lctrl, Cached, Color, Drawable, EventCtx, GeomBatch, GeomBatchStack, GfxCtx,
|
||||||
HorizontalAlignment, Key, Line, Outcome, Panel, State, Text, VerticalAlignment, Widget,
|
HorizontalAlignment, Key, Line, Outcome, Panel, State, Text, Toggle, VerticalAlignment, Widget,
|
||||||
};
|
};
|
||||||
|
|
||||||
use crate::app::{App, Transition};
|
use crate::app::{App, Transition};
|
||||||
@ -20,6 +20,7 @@ pub struct ExploreMap {
|
|||||||
legend: Panel,
|
legend: Panel,
|
||||||
tooltip: Cached<RoadID, (RoadID, Drawable, Drawable)>,
|
tooltip: Cached<RoadID, (RoadID, Drawable, Drawable)>,
|
||||||
unzoomed_layer: Drawable,
|
unzoomed_layer: Drawable,
|
||||||
|
elevation: Option<Drawable>,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl ExploreMap {
|
impl ExploreMap {
|
||||||
@ -28,9 +29,10 @@ impl ExploreMap {
|
|||||||
|
|
||||||
Box::new(ExploreMap {
|
Box::new(ExploreMap {
|
||||||
top_panel: make_top_panel(ctx),
|
top_panel: make_top_panel(ctx),
|
||||||
legend: make_legend(ctx, app),
|
legend: make_legend(ctx, app, false),
|
||||||
tooltip: Cached::new(),
|
tooltip: Cached::new(),
|
||||||
unzoomed_layer: make_unzoomed_layer(ctx, app),
|
unzoomed_layer: make_unzoomed_layer(ctx, app),
|
||||||
|
elevation: None,
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -76,7 +78,8 @@ impl State<App> for ExploreMap {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if let Outcome::Clicked(x) = self.legend.event(ctx) {
|
match self.legend.event(ctx) {
|
||||||
|
Outcome::Clicked(x) => {
|
||||||
return Transition::Push(match x.as_ref() {
|
return Transition::Push(match x.as_ref() {
|
||||||
"change map" => CityPicker::new_state(
|
"change map" => CityPicker::new_state(
|
||||||
ctx,
|
ctx,
|
||||||
@ -93,9 +96,31 @@ impl State<App> for ExploreMap {
|
|||||||
"protected bike lane" => PopupMsg::new_state(ctx, "Protected bike lanes", vec!["Bike lanes separated from vehicle traffic by physical barriers or a few feet of striping"]),
|
"protected bike lane" => PopupMsg::new_state(ctx, "Protected bike lanes", vec!["Bike lanes separated from vehicle traffic by physical barriers or a few feet of striping"]),
|
||||||
"painted bike lane" => PopupMsg::new_state(ctx, "Painted bike lanes", vec!["Bike lanes without any separation from vehicle traffic. Often uncomfortably close to the \"door zone\" of parked cars."]),
|
"painted bike lane" => PopupMsg::new_state(ctx, "Painted bike lanes", vec!["Bike lanes without any separation from vehicle traffic. Often uncomfortably close to the \"door zone\" of parked cars."]),
|
||||||
"Stay Healthy Street / greenway" => PopupMsg::new_state(ctx, "Stay Healthy Streets and neighborhood greenways", vec!["Residential streets with additional signage and light barriers. These are intended to be low traffic, dedicated for people walking and biking."]),
|
"Stay Healthy Street / greenway" => PopupMsg::new_state(ctx, "Stay Healthy Streets and neighborhood greenways", vec!["Residential streets with additional signage and light barriers. These are intended to be low traffic, dedicated for people walking and biking."]),
|
||||||
|
// TODO Add URLs
|
||||||
|
"about the elevation data" => PopupMsg::new_state(ctx, "About the elevation data", vec!["Biking uphill next to traffic without any dedicated space isn't fun.", "Biking downhill next to traffic, especially in the door-zone of parked cars, and especially on Seattle's bumpy roads... is downright terrifying.", "", "Note the elevation data is incorrect near bridges.", "Thanks to King County LIDAR for the data, and Eldan Goldenberg for processing it."]),
|
||||||
_ => unreachable!(),
|
_ => unreachable!(),
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
Outcome::Changed(_) => {
|
||||||
|
if self.legend.is_checked("elevation") {
|
||||||
|
let mut low = Distance::ZERO;
|
||||||
|
let mut high = Distance::ZERO;
|
||||||
|
for i in app.primary.map.all_intersections() {
|
||||||
|
low = low.min(i.elevation);
|
||||||
|
high = high.max(i.elevation);
|
||||||
|
}
|
||||||
|
// TODO Cache the elevation layer per map, maybe in the session state
|
||||||
|
// TODO Maybe also draw the uphill arrows on the steepest streets?
|
||||||
|
self.elevation = Some(
|
||||||
|
crate::layer::elevation::make_elevation_contours(ctx, app, low, high).1,
|
||||||
|
);
|
||||||
|
} else {
|
||||||
|
self.elevation = None;
|
||||||
|
}
|
||||||
|
self.legend = make_legend(ctx, app, self.elevation.is_some());
|
||||||
|
}
|
||||||
|
_ => {}
|
||||||
|
}
|
||||||
|
|
||||||
Transition::Keep
|
Transition::Keep
|
||||||
}
|
}
|
||||||
@ -113,6 +138,9 @@ impl State<App> for ExploreMap {
|
|||||||
}
|
}
|
||||||
if g.canvas.cam_zoom < app.opts.min_zoom_for_detail {
|
if g.canvas.cam_zoom < app.opts.min_zoom_for_detail {
|
||||||
g.redraw(&self.unzoomed_layer);
|
g.redraw(&self.unzoomed_layer);
|
||||||
|
if let Some(ref draw) = self.elevation {
|
||||||
|
g.redraw(draw);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -147,7 +175,7 @@ fn make_top_panel(ctx: &mut EventCtx) -> Panel {
|
|||||||
.build(ctx)
|
.build(ctx)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn make_legend(ctx: &mut EventCtx, app: &App) -> Panel {
|
fn make_legend(ctx: &mut EventCtx, app: &App, elevation: bool) -> Panel {
|
||||||
Panel::new_builder(Widget::col(vec![
|
Panel::new_builder(Widget::col(vec![
|
||||||
Widget::custom_row(vec![
|
Widget::custom_row(vec![
|
||||||
Line("Bike Network")
|
Line("Bike Network")
|
||||||
@ -174,6 +202,14 @@ fn make_legend(ctx: &mut EventCtx, app: &App) -> Panel {
|
|||||||
// TODO Distinguish door-zone bike lanes?
|
// TODO Distinguish door-zone bike lanes?
|
||||||
// TODO Call out bike turning boxes?
|
// TODO Call out bike turning boxes?
|
||||||
// TODO Call out bike signals?
|
// TODO Call out bike signals?
|
||||||
|
Widget::row(vec![
|
||||||
|
Toggle::checkbox(ctx, "elevation", Key::E, elevation),
|
||||||
|
ctx.style()
|
||||||
|
.btn_plain
|
||||||
|
.icon("system/assets/tools/info.svg")
|
||||||
|
.build_widget(ctx, "about the elevation data")
|
||||||
|
.centered_vert(),
|
||||||
|
]),
|
||||||
]))
|
]))
|
||||||
.aligned(HorizontalAlignment::Right, VerticalAlignment::Bottom)
|
.aligned(HorizontalAlignment::Right, VerticalAlignment::Bottom)
|
||||||
.build(ctx)
|
.build(ctx)
|
||||||
@ -217,7 +253,14 @@ fn make_tooltip(ctx: &mut EventCtx, app: &App, r: RoadID) -> (RoadID, Drawable,
|
|||||||
let zoom = (fit_dims / bounds.width()).min(fit_dims / bounds.height());
|
let zoom = (fit_dims / bounds.width()).min(fit_dims / bounds.height());
|
||||||
screen_batch = screen_batch.scale(zoom);
|
screen_batch = screen_batch.scale(zoom);
|
||||||
|
|
||||||
let label = Text::from(Line(road.get_name(app.opts.language.as_ref())).small_heading())
|
let label = Text::from_multiline(vec![
|
||||||
|
Line(road.get_name(app.opts.language.as_ref())).small_heading(),
|
||||||
|
// TODO Indicate which direction is uphill
|
||||||
|
Line(format!(
|
||||||
|
"{}% incline",
|
||||||
|
(road.percent_incline.abs() * 100.0).round()
|
||||||
|
)),
|
||||||
|
])
|
||||||
.render_autocropped(ctx);
|
.render_autocropped(ctx);
|
||||||
screen_batch = GeomBatchStack::vertical(vec![label, screen_batch]).batch();
|
screen_batch = GeomBatchStack::vertical(vec![label, screen_batch]).batch();
|
||||||
screen_batch = magnifying_glass(screen_batch);
|
screen_batch = magnifying_glass(screen_batch);
|
||||||
|
Loading…
Reference in New Issue
Block a user