use std::collections::HashSet;
use abstutil::prettyprint_usize;
use geom::{ArrowCap, Circle, Distance, Duration, PolyLine, Pt2D, Time};
use map_gui::tools::{ChooseSomething, ColorLegend, SimpleMinimap};
use map_gui::SimpleApp;
use map_model::BuildingID;
use widgetry::{
Btn, Choice, Color, Drawable, EventCtx, GeomBatch, GfxCtx, HorizontalAlignment, Key, Line,
Outcome, Panel, State, Text, TextExt, Transition, UpdateType, VerticalAlignment, Widget,
};
use crate::animation::{Animator, SnowEffect};
use crate::buildings::{BldgState, Buildings};
use crate::levels::Level;
use crate::meters::{custom_bar, make_bar};
use crate::movement::Player;
use crate::vehicles::Vehicle;
const ACQUIRE_BOOST_RATE: f64 = 0.5;
const BOOST_SPEED_MULTIPLIER: f64 = 1.5;
pub struct Game {
title_panel: Panel,
status_panel: Panel,
time_panel: Panel,
boost_panel: Panel,
minimap: SimpleMinimap,
animator: Animator,
snow: SnowEffect,
time: Time,
state: GameState,
player: Player,
}
impl Game {
pub fn new(
ctx: &mut EventCtx,
app: &SimpleApp,
level: Level,
vehicle: Vehicle,
upzones: HashSet<BuildingID>,
) -> Box<dyn State<SimpleApp>> {
let title_panel = Panel::new(Widget::row(vec![
Btn::svg_def("system/assets/tools/home.svg").build(ctx, "back", Key::Escape),
"15 min Santa".draw_text(ctx),
Widget::draw_svg(ctx, "system/assets/tools/map.svg"),
level.title.draw_text(ctx),
]))
.aligned(HorizontalAlignment::Center, VerticalAlignment::TopInset)
.build(ctx);
let status_panel = Panel::new(Widget::col(vec![
"Complete Deliveries".draw_text(ctx),
Widget::draw_batch(ctx, GeomBatch::new()).named("score"),
"Remaining Gifts:".draw_text(ctx),
Widget::draw_batch(ctx, GeomBatch::new()).named("energy"),
Widget::horiz_separator(ctx, 0.2),
ColorLegend::row(ctx, app.cs.residential_building, "single-family house"),
ColorLegend::row(ctx, Color::CYAN, "apartment building"),
ColorLegend::row(ctx, Color::YELLOW, "store"),
]))
.aligned(HorizontalAlignment::RightInset, VerticalAlignment::TopInset)
.build(ctx);
let time_panel = Panel::new(Widget::row(vec![
"Time spent:".draw_text(ctx),
Widget::draw_batch(ctx, GeomBatch::new())
.named("time")
.align_right(),
]))
.aligned(HorizontalAlignment::LeftInset, VerticalAlignment::TopInset)
.build(ctx);
let boost_panel = Panel::new(Widget::row(vec![
"Boost".draw_text(ctx),
Widget::draw_batch(ctx, GeomBatch::new())
.named("boost")
.align_right(),
]))
.aligned(HorizontalAlignment::Center, VerticalAlignment::BottomInset)
.build(ctx);
let start = app
.map
.find_i_by_osm_id(level.start)
.expect(&format!("can't find {}", level.start));
let player = Player::new(ctx, app, start);
let bldgs = Buildings::new(ctx, app, upzones);
let state = GameState::new(ctx, app, level, vehicle, bldgs);
let with_zorder = false;
let mut game = Game {
title_panel,
status_panel,
time_panel,
boost_panel,
minimap: SimpleMinimap::new(ctx, app, with_zorder),
animator: Animator::new(ctx),
snow: SnowEffect::new(ctx),
time: Time::START_OF_DAY,
state,
player,
};
game.update_panels(ctx);
game.minimap
.set_zoom(ctx, app, game.state.level.minimap_zoom);
Box::new(game)
}
fn update_panels(&mut self, ctx: &mut EventCtx) {
let time = format!("{}", self.time - Time::START_OF_DAY).draw_text(ctx);
self.time_panel.replace(ctx, "time", time);
let score_bar = make_bar(
ctx,
Color::hex("#83AA51"),
self.state.score,
self.state.bldgs.total_housing_units,
);
self.status_panel.replace(ctx, "score", score_bar);
let energy_bar = make_bar(
ctx,
Color::hex("#D8B830"),
self.state.energy,
self.state.vehicle.max_energy,
);
self.status_panel.replace(ctx, "energy", energy_bar);
let boost_bar = custom_bar(
ctx,
Color::hex("#A32015"),
self.state.boost / self.state.vehicle.max_boost,
if self.state.boost == Duration::ZERO {
Text::from(Line("Find a bike or bus lane to get a boost"))
} else {
Text::from(Line("Press space to boost"))
},
);
self.boost_panel.replace(ctx, "boost", boost_bar);
}
}
impl State<SimpleApp> for Game {
fn event(&mut self, ctx: &mut EventCtx, app: &mut SimpleApp) -> Transition<SimpleApp> {
if let Some(dt) = ctx.input.nonblocking_is_update_event() {
self.time += dt;
}
let base_speed = if self.state.has_energy() {
self.state.vehicle.normal_speed
} else {
self.state.vehicle.tired_speed
};
let speed = if ctx.is_key_down(Key::Space) && self.state.boost > Duration::ZERO {
if let Some(dt) = ctx.input.nonblocking_is_update_event() {
self.state.boost -= dt;
self.state.boost = self.state.boost.max(Duration::ZERO);
}
base_speed * BOOST_SPEED_MULTIPLIER
} else {
base_speed
};
for b in self.player.update_with_speed(ctx, app, speed) {
match self.state.bldgs.buildings[&b] {
BldgState::Undelivered(_) => {
if let Some(increase) = self.state.present_dropped(ctx, app, b) {
self.animator.add(
self.time,
Duration::seconds(0.5),
(1.0, 4.0),
app.map.get_b(b).label_center,
Text::from(Line(format!("+{}", prettyprint_usize(increase))))
.bg(Color::hex("#83AA51"))
.render_to_batch(ctx.prerender)
.scale(0.1),
);
}
}
BldgState::Store => {
let refill = self.state.vehicle.max_energy - self.state.energy;
if refill > 0 {
self.state.energy += refill;
self.animator.add(
self.time,
Duration::seconds(0.5),
(1.0, 4.0),
app.map.get_b(b).label_center,
Text::from(Line(format!("Refilled {}", prettyprint_usize(refill))))
.bg(Color::BLUE)
.render_to_batch(ctx.prerender)
.scale(0.1),
);
}
}
BldgState::Done => {}
}
}
if let Some(dt) = ctx.input.nonblocking_is_update_event() {
if self.player.on_good_road(app) {
self.state.boost += dt * ACQUIRE_BOOST_RATE;
self.state.boost = self.state.boost.min(self.state.vehicle.max_boost);
}
}
if let Some(t) = self.minimap.event(ctx, app) {
return t;
}
self.animator.event(ctx, self.time);
self.snow.event(ctx, self.time);
if self.state.has_energy() {
self.state.energyless_arrow = None;
} else {
if self.state.energyless_arrow.is_none() {
self.state.energyless_arrow = Some(EnergylessArrow::new(ctx, self.time));
}
let stores = self.state.bldgs.all_stores();
self.state.energyless_arrow.as_mut().unwrap().update(
ctx,
app,
self.time,
self.player.get_pos(),
stores,
);
}
match self.title_panel.event(ctx) {
Outcome::Clicked(x) => match x.as_ref() {
"back" => {
return Transition::Push(ChooseSomething::new(
ctx,
"Game Paused",
vec![
Choice::string("Resume").key(Key::Escape),
Choice::string("Quit"),
],
Box::new(|resp, _, _| match resp.as_ref() {
"Resume" => Transition::Pop,
"Quit" => Transition::Multi(vec![Transition::Pop, Transition::Pop]),
_ => unreachable!(),
}),
));
}
_ => unreachable!(),
},
_ => {}
}
self.update_panels(ctx);
ctx.request_update(UpdateType::Game);
Transition::Keep
}
fn draw(&self, g: &mut GfxCtx, app: &SimpleApp) {
self.title_panel.draw(g);
self.status_panel.draw(g);
self.time_panel.draw(g);
self.boost_panel.draw(g);
let santa_tracker = g.upload(GeomBatch::from(vec![(
Color::RED,
Circle::new(self.player.get_pos(), Distance::meters(20.0)).to_polygon(),
)]));
self.minimap.draw_with_extra_layers(
g,
app,
vec![
&self.state.bldgs.draw_all,
&self.state.draw_done_houses,
&santa_tracker,
],
);
g.redraw(&self.state.bldgs.draw_all);
g.redraw(&self.state.draw_done_houses);
if true {
GeomBatch::load_svg(g.prerender, &self.state.vehicle.animate(self.time))
.scale(0.1)
.centered_on(self.player.get_pos())
.rotate_around_batch_center(self.player.get_angle())
.draw(g);
} else {
g.draw_polygon(
Color::RED,
Circle::new(self.player.get_pos(), Distance::meters(2.0)).to_polygon(),
);
}
self.snow.draw(g);
self.animator.draw(g);
if let Some(ref arrow) = self.state.energyless_arrow {
g.redraw(&arrow.draw);
}
}
}
struct GameState {
level: Level,
vehicle: Vehicle,
bldgs: Buildings,
score: usize,
energy: usize,
boost: Duration,
draw_done_houses: Drawable,
energyless_arrow: Option<EnergylessArrow>,
}
impl GameState {
fn new(
ctx: &mut EventCtx,
app: &SimpleApp,
level: Level,
vehicle: Vehicle,
bldgs: Buildings,
) -> GameState {
let energy = vehicle.max_energy;
let mut s = GameState {
level,
vehicle,
bldgs,
score: 0,
energy,
boost: Duration::ZERO,
draw_done_houses: Drawable::empty(ctx),
energyless_arrow: None,
};
s.recalc_deliveries(ctx, app);
s
}
fn recalc_deliveries(&mut self, ctx: &mut EventCtx, app: &SimpleApp) {
let mut batch = GeomBatch::new();
for (b, state) in &self.bldgs.buildings {
if let BldgState::Done = state {
batch.push(Color::BLACK, app.map.get_b(*b).polygon.clone());
}
}
self.draw_done_houses = ctx.upload(batch);
}
fn present_dropped(
&mut self,
ctx: &mut EventCtx,
app: &SimpleApp,
id: BuildingID,
) -> Option<usize> {
if !self.has_energy() {
return None;
}
if let BldgState::Undelivered(num_housing_units) = self.bldgs.buildings[&id] {
let deliveries = num_housing_units.min(self.energy);
self.score += deliveries;
self.bldgs.buildings.insert(id, BldgState::Done);
self.energy -= deliveries;
self.recalc_deliveries(ctx, app);
return Some(deliveries);
}
None
}
fn has_energy(&self) -> bool {
self.energy > 0
}
}
struct EnergylessArrow {
draw: Drawable,
started: Time,
last_update: Time,
}
impl EnergylessArrow {
fn new(ctx: &EventCtx, started: Time) -> EnergylessArrow {
EnergylessArrow {
draw: Drawable::empty(ctx),
started,
last_update: Time::START_OF_DAY,
}
}
fn update(
&mut self,
ctx: &mut EventCtx,
app: &SimpleApp,
time: Time,
sleigh: Pt2D,
all_stores: Vec<BuildingID>,
) {
if self.last_update == time {
return;
}
self.last_update = time;
let store = app.map.get_b(
all_stores
.into_iter()
.min_by_key(|b| app.map.get_b(*b).label_center.fast_dist(sleigh))
.unwrap(),
);
let period = Duration::seconds(0.5);
let pct = ((time - self.started) % period) / period;
let shift = (pct * std::f64::consts::PI).sin();
let thickness = Distance::meters(5.0 + shift);
let angle = sleigh.angle_to(store.label_center);
let arrow = PolyLine::must_new(vec![
sleigh.project_away(Distance::meters(20.0), angle),
sleigh.project_away(Distance::meters(40.0), angle),
])
.make_arrow(thickness, ArrowCap::Triangle);
self.draw = ctx.upload(GeomBatch::from(vec![(Color::RED.alpha(0.8), arrow)]));
}
}