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
use geom::{Angle, Circle, Distance, Line, Polygon, Pt2D};
use map_model::{Map, TransitStop, TransitStopID};
use widgetry::{Drawable, EventCtx, GeomBatch, GfxCtx};
use crate::colors::ColorScheme;
use crate::render::{DrawOptions, Renderable, OUTLINE_THICKNESS};
use crate::{AppLike, ID};
const RADIUS: Distance = Distance::const_meters(1.0);
pub struct DrawTransitStop {
pub id: TransitStopID,
center: Pt2D,
zorder: isize,
draw_default: Drawable,
}
impl DrawTransitStop {
pub fn new(ctx: &EventCtx, stop: &TransitStop, map: &Map, cs: &ColorScheme) -> DrawTransitStop {
let (pt, angle) = stop.sidewalk_pos.pt_and_angle(map);
let center = pt.project_away(
map.get_l(stop.sidewalk_pos.lane()).width / 2.0,
angle.rotate_degs(90.0),
);
let mut icon = GeomBatch::new();
icon.append(
GeomBatch::load_svg(
ctx.prerender,
if stop.is_train_stop {
"system/assets/map/light_rail.svg"
} else {
"system/assets/meters/bus.svg"
},
)
.scale(0.05)
.centered_on(center),
);
let mut batch = GeomBatch::new();
batch.push(
cs.bus_layer.alpha(0.8),
Circle::new(center, RADIUS).to_polygon(),
);
batch.append(icon.autocrop().centered_on(center));
batch.push(
cs.stop_sign_pole,
Line::must_new(
center.project_away(RADIUS, Angle::degrees(90.0)),
center.project_away(1.5 * RADIUS, Angle::degrees(90.0)),
)
.make_polygons(Distance::meters(0.3)),
);
DrawTransitStop {
id: stop.id,
center,
zorder: map.get_parent(stop.sidewalk_pos.lane()).zorder,
draw_default: ctx.upload(batch),
}
}
}
impl Renderable for DrawTransitStop {
fn get_id(&self) -> ID {
ID::TransitStop(self.id)
}
fn draw(&self, g: &mut GfxCtx, _: &dyn AppLike, _: &DrawOptions) {
g.redraw(&self.draw_default);
}
fn get_outline(&self, _: &Map) -> Polygon {
Circle::new(self.center, RADIUS)
.to_outline(OUTLINE_THICKNESS)
.expect("constants defined wrong")
}
fn contains_pt(&self, pt: Pt2D, _: &Map) -> bool {
Circle::new(self.center, RADIUS).contains_pt(pt)
}
fn get_zorder(&self) -> isize {
self.zorder
}
}