use std::cell::RefCell;
use geom::{Angle, Distance, Line, Polygon, Pt2D, Ring};
use map_model::{Building, BuildingID, LaneType, Map, OffstreetParking, NORMAL_LANE_THICKNESS};
use widgetry::{Color, Drawable, EventCtx, GeomBatch, GfxCtx, Line, Text};
use crate::colors::{ColorScheme, ColorSchemeChoice};
use crate::options::{CameraAngle, Options};
use crate::render::{DrawOptions, Renderable, OUTLINE_THICKNESS};
use crate::{AppLike, ID};
pub struct DrawBuilding {
pub id: BuildingID,
label: RefCell<Option<Drawable>>,
}
impl DrawBuilding {
pub fn new(
ctx: &EventCtx,
bldg: &Building,
map: &Map,
cs: &ColorScheme,
opts: &Options,
bldg_batch: &mut GeomBatch,
outlines_batch: &mut GeomBatch,
) -> DrawBuilding {
let bldg_color = if bldg.amenities.is_empty() {
cs.residential_building
} else {
cs.commercial_building
};
match &opts.camera_angle {
CameraAngle::TopDown => {
bldg_batch.push(bldg_color, bldg.polygon.clone());
if let Ok(p) = bldg.polygon.to_outline(Distance::meters(0.1)) {
outlines_batch.push(cs.building_outline, p);
}
let parking_icon = match bldg.parking {
OffstreetParking::PublicGarage(_, _) => true,
OffstreetParking::Private(_, garage) => garage,
};
if parking_icon {
bldg_batch.append(
GeomBatch::load_svg(ctx, "system/assets/map/parking.svg")
.scale(0.1)
.centered_on(bldg.label_center),
);
}
}
CameraAngle::Abstract => {
bldg_batch.push(
bldg_color,
Polygon::rectangle_centered(
bldg.polygon.center(),
Distance::meters(5.0),
Distance::meters(5.0),
),
);
}
x => {
let angle = match x {
CameraAngle::IsometricNE => Angle::degrees(-45.0),
CameraAngle::IsometricNW => Angle::degrees(-135.0),
CameraAngle::IsometricSE => Angle::degrees(45.0),
CameraAngle::IsometricSW => Angle::degrees(135.0),
CameraAngle::TopDown | CameraAngle::Abstract => unreachable!(),
};
let bldg_height_per_level = 3.5;
let bldg_rendered_meters = bldg_height_per_level * bldg.levels.powf(0.8);
let height = Distance::meters(bldg_rendered_meters);
let map_bounds = map.get_gps_bounds().to_bounds();
let (map_width, map_height) = (map_bounds.width(), map_bounds.height());
let map_length = map_width.hypot(map_height);
let distance = |pt: &Pt2D| {
let projection_origin = match x {
CameraAngle::IsometricNE => Pt2D::new(0.0, map_height),
CameraAngle::IsometricNW => Pt2D::new(map_width, map_height),
CameraAngle::IsometricSE => Pt2D::new(0.0, 0.0),
CameraAngle::IsometricSW => Pt2D::new(map_width, 0.0),
CameraAngle::TopDown | CameraAngle::Abstract => unreachable!(),
};
let abs_pt = Pt2D::new(
(pt.x() - projection_origin.x()).abs(),
(pt.y() - projection_origin.y()).abs(),
);
let a = f64::hypot(abs_pt.x(), abs_pt.y());
let theta = f64::atan(abs_pt.y() / abs_pt.x());
let distance = a * f64::sin(theta + std::f64::consts::PI / 4.0);
Distance::meters(distance)
};
let closest_pt = bldg
.polygon
.points()
.iter()
.min_by(|a, b| distance(a).cmp(&distance(b)));
let distance_from_projection_axis = closest_pt
.map(|pt| distance(pt).inner_meters())
.unwrap_or(0.0);
let scale_factor = map_length;
let groundfloor_z = distance_from_projection_axis / scale_factor - 1.0;
let roof_z = groundfloor_z - height.inner_meters() / scale_factor;
if let Ok(roof) = Ring::new(
bldg.polygon
.points()
.iter()
.map(|pt| pt.project_away(height, angle))
.collect(),
) {
if let Ok(p) = bldg.polygon.to_outline(Distance::meters(0.3)) {
bldg_batch.push(Color::BLACK, p);
}
let wall_z = (groundfloor_z + roof_z) / 2.0;
let mut wall_beams = Vec::new();
for (low, high) in bldg.polygon.points().iter().zip(roof.points().iter()) {
if let Some(l) = Line::new(*low, *high) {
wall_beams.push(l);
}
}
let wall_color = Color::hex("#BBBEC3");
for (wall1, wall2) in wall_beams.iter().zip(wall_beams.iter().skip(1)) {
bldg_batch.push_with_z(
wall_color,
Ring::must_new(vec![
wall1.pt1(),
wall1.pt2(),
wall2.pt2(),
wall2.pt1(),
wall1.pt1(),
])
.into_polygon(),
wall_z,
);
}
for wall in wall_beams {
bldg_batch.push_with_z(
Color::BLACK,
wall.make_polygons(Distance::meters(0.1)),
wall_z,
);
}
bldg_batch.push_with_z(bldg_color, roof.clone().into_polygon(), roof_z);
bldg_batch.push_with_z(
Color::BLACK,
roof.to_outline(Distance::meters(0.3)),
roof_z,
);
} else {
bldg_batch.push(bldg_color, bldg.polygon.clone());
if let Ok(p) = bldg.polygon.to_outline(Distance::meters(0.1)) {
outlines_batch.push(cs.building_outline, p);
}
}
}
}
DrawBuilding {
id: bldg.id,
label: RefCell::new(None),
}
}
pub fn draw_driveway(
bldg: &Building,
map: &Map,
cs: &ColorScheme,
opts: &Options,
batch: &mut GeomBatch,
) {
if opts.camera_angle != CameraAngle::Abstract {
let orig_pl = &bldg.driveway_geom;
let driveway = orig_pl
.slice(
Distance::ZERO,
orig_pl.length() - map.get_l(bldg.sidewalk()).width / 2.0,
)
.map(|(pl, _)| pl)
.unwrap_or_else(|_| orig_pl.clone());
batch.push(
if opts.color_scheme == ColorSchemeChoice::NightMode {
Color::hex("#4B4B4B")
} else {
cs.zoomed_road_surface(
LaneType::Sidewalk,
map.get_parent(bldg.sidewalk()).get_rank(),
)
},
driveway.make_polygons(NORMAL_LANE_THICKNESS),
);
}
}
}
impl Renderable for DrawBuilding {
fn get_id(&self) -> ID {
ID::Building(self.id)
}
fn draw(&self, g: &mut GfxCtx, app: &dyn AppLike, opts: &DrawOptions) {
if opts.label_buildings {
let mut label = self.label.borrow_mut();
if label.is_none() {
let mut batch = GeomBatch::new();
let b = app.map().get_b(self.id);
if let Some(a) = b.amenities.get(0) {
let mut txt = Text::from(
Line(a.names.get(app.opts().language.as_ref())).fg(Color::BLACK),
);
if b.amenities.len() > 1 {
txt.append(Line(format!(" (+{})", b.amenities.len() - 1)).fg(Color::BLACK));
}
batch.append(
txt.render_autocropped(g)
.scale(0.1)
.centered_on(b.label_center),
);
}
*label = Some(g.prerender.upload(batch));
}
g.redraw(label.as_ref().unwrap());
}
}
fn get_zorder(&self) -> isize {
0
}
fn get_outline(&self, map: &Map) -> Polygon {
let b = map.get_b(self.id);
if let Ok(p) = b.polygon.to_outline(OUTLINE_THICKNESS) {
p
} else {
b.polygon.clone()
}
}
fn contains_pt(&self, pt: Pt2D, map: &Map) -> bool {
map.get_b(self.id).polygon.contains_pt(pt)
}
}