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 std::cell::RefCell;
use std::collections::HashMap;
use map_model::{LaneType, PathConstraints, Road};
use widgetry::{Color, Drawable, Fill, GeomBatch, GfxCtx, Texture};
use crate::app::App;
lazy_static::lazy_static! {
pub static ref DEDICATED_TRAIL: Color = Color::GREEN;
pub static ref PROTECTED_BIKE_LANE: Color = Color::hex("#A4DE02");
pub static ref PAINTED_BIKE_LANE: Color = Color::hex("#76BA1B");
pub static ref GREENWAY: Color = Color::hex("#4C9A2A");
}
pub struct DrawNetworkLayer {
per_zoom: RefCell<[Option<Drawable>; 11]>,
}
impl DrawNetworkLayer {
pub fn new() -> DrawNetworkLayer {
DrawNetworkLayer {
per_zoom: Default::default(),
}
}
pub fn draw(&self, g: &mut GfxCtx, app: &App) {
let (zoom, idx) = DrawNetworkLayer::discretize_zoom(g.canvas.cam_zoom);
let value = &mut self.per_zoom.borrow_mut()[idx];
if value.is_none() {
*value = Some(DrawNetworkLayer::render_network_layer(g, app, zoom));
}
g.redraw(value.as_ref().unwrap());
}
fn discretize_zoom(zoom: f64) -> (f64, usize) {
if zoom >= 1.0 {
return (1.0, 10);
}
let rounded = (zoom * 10.0).round();
let idx = rounded as usize;
(rounded / 10.0, idx)
}
fn render_network_layer(g: &mut GfxCtx, app: &App, zoom: f64) -> Drawable {
let mut batch = GeomBatch::new();
let map = &app.primary.map;
batch.push(Color::BLACK.alpha(0.4), map.get_boundary_polygon().clone());
let thickness = (0.5 / zoom).max(1.0);
let mut intersections = HashMap::new();
for r in map.all_roads() {
let mut bike_lane = false;
let mut buffer = false;
for (_, _, lt) in r.lanes_ltr() {
if lt == LaneType::Biking {
bike_lane = true;
} else if matches!(lt, LaneType::Buffer(_)) {
buffer = true;
}
}
let color = if r.is_cycleway() {
*DEDICATED_TRAIL
} else if bike_lane && buffer {
*PROTECTED_BIKE_LANE
} else if bike_lane {
*PAINTED_BIKE_LANE
} else if is_greenway(r) {
*GREENWAY
} else {
continue;
};
batch.push(
if map.get_edits().changed_roads.contains(&r.id) {
Fill::ColoredTexture(color, Texture::CROSS_HATCH)
} else {
Fill::Color(color)
},
r.center_pts.make_polygons(thickness * r.get_width(map)),
);
intersections.insert(r.src_i, color);
intersections.insert(r.dst_i, color);
}
for (i, color) in intersections {
batch.push(color, map.get_i(i).polygon.clone());
}
g.upload(batch)
}
}
pub fn is_greenway(road: &Road) -> bool {
!road
.access_restrictions
.allow_through_traffic
.contains(PathConstraints::Car)
&& road
.access_restrictions
.allow_through_traffic
.contains(PathConstraints::Bike)
}