#[macro_use]
extern crate anyhow;
#[macro_use]
extern crate log;
use abstio::MapName;
use abstutil::{CmdArgs, Timer};
use geom::{Duration, LonLat, Pt2D};
use map_gui::options::Options;
use map_model::Map;
use sim::{Sim, SimFlags};
use widgetry::{EventCtx, State, Transition};
use crate::app::{App, Flags};
use crate::pregame::TitleScreen;
use crate::sandbox::{GameplayMode, SandboxMode};
mod app;
mod challenges;
mod common;
mod debug;
mod devtools;
mod edit;
mod info;
mod layer;
mod pregame;
mod sandbox;
pub fn main() {
let mut args = CmdArgs::new();
if args.enabled("--prebake") {
challenges::prebake::prebake_all();
return;
}
let mut flags = Flags {
sim_flags: SimFlags::from_args(&mut args),
live_map_edits: args.enabled("--live_map_edits"),
study_area: args.optional("--study_area"),
};
let mut opts = Options::default();
opts.toggle_day_night_colors = true;
opts.update_from_args(&mut args);
let mut settings = widgetry::Settings::new("A/B Street")
.read_svg(Box::new(abstio::slurp_bytes))
.window_icon(abstio::path("system/assets/pregame/icon.png"))
.loading_tips(map_gui::tools::loading_tips());
if args.enabled("--dump_raw_events") {
settings = settings.dump_raw_events();
}
if let Some(s) = args.optional_parse("--scale_factor", |s| s.parse::<f64>()) {
settings = settings.scale_factor(s);
}
let mut mode = None;
let mut initialize_tutorial = false;
if let Some(x) = args.optional("--challenge") {
let mut aliases = Vec::new();
'OUTER: for (_, stages) in challenges::Challenge::all() {
for challenge in stages {
if challenge.alias == x {
flags.sim_flags.load = challenge.gameplay.map_name().path();
mode = Some(challenge.gameplay);
break 'OUTER;
} else {
aliases.push(challenge.alias);
}
}
}
if mode.is_none() {
panic!("Invalid --challenge={}. Choices: {}", x, aliases.join(", "));
}
}
if let Some(n) = args.optional_parse("--tutorial", |s| s.parse::<usize>()) {
initialize_tutorial = true;
mode = Some(sandbox::GameplayMode::Tutorial(
sandbox::TutorialPointer::new(n - 1, 0),
));
}
let modifiers = flags.sim_flags.modifiers.drain(..).collect();
if mode.is_none() && flags.sim_flags.load.contains("scenarios/") {
let (map_name, scenario) = abstio::parse_scenario_path(&flags.sim_flags.load);
flags.sim_flags.load = map_name.path();
mode = Some(sandbox::GameplayMode::PlayScenario(
map_name, scenario, modifiers,
));
}
let start_with_edits = args.optional("--edits");
let center_camera = args.optional("--cam");
if let Some(site) = args.optional("--actdev") {
let city = site.replace("-", "_");
let name = MapName::new(&city, "center");
flags.sim_flags.load = name.path();
flags.study_area = Some(site);
mode = Some(sandbox::GameplayMode::Blog(name));
}
args.done();
widgetry::run(settings, |ctx| {
setup_app(
ctx,
flags,
opts,
start_with_edits,
mode,
initialize_tutorial,
center_camera,
)
});
}
fn setup_app(
ctx: &mut EventCtx,
flags: Flags,
mut opts: Options,
start_with_edits: Option<String>,
maybe_mode: Option<GameplayMode>,
initialize_tutorial: bool,
center_camera: Option<String>,
) -> (App, Vec<Box<dyn State<App>>>) {
let title = !opts.dev
&& !flags.sim_flags.load.contains("player/save")
&& !flags.sim_flags.load.contains("/scenarios/")
&& maybe_mode.is_none();
if let Some(GameplayMode::PlayScenario(_, _, _))
| Some(GameplayMode::FixTrafficSignals)
| Some(GameplayMode::OptimizeCommute(_, _))
| Some(GameplayMode::Tutorial(_)) = maybe_mode
{
opts.color_scheme = map_gui::colors::ColorSchemeChoice::NightMode;
}
let cs = map_gui::colors::ColorScheme::new(ctx, opts.color_scheme);
if flags.sim_flags.load.contains("/maps/") {
let map = Map::blank();
let sim = Sim::new(&map, flags.sim_flags.opts.clone());
let primary = crate::app::PerMap::map_loaded(
map,
sim,
flags,
&opts,
&cs,
ctx,
&mut Timer::throwaway(),
);
let app = App {
primary,
cs,
opts,
per_obj: crate::app::PerObjectActions::new(),
session: crate::app::SessionState::empty(),
};
let map_name = MapName::from_path(&app.primary.current_flags.sim_flags.load).unwrap();
let states = vec![map_gui::load::MapLoader::new(
ctx,
&app,
map_name,
Box::new(move |ctx, app| {
Transition::Clear(finish_app_setup(
ctx,
app,
title,
start_with_edits,
maybe_mode,
initialize_tutorial,
center_camera,
))
}),
)];
(app, states)
} else {
let primary = ctx.loading_screen("load map", |ctx, mut timer| {
assert!(flags.sim_flags.modifiers.is_empty());
let (map, sim, _) = flags.sim_flags.load(timer);
crate::app::PerMap::map_loaded(map, sim, flags, &opts, &cs, ctx, &mut timer)
});
let mut app = App {
primary,
cs,
opts,
per_obj: crate::app::PerObjectActions::new(),
session: crate::app::SessionState::empty(),
};
let states = finish_app_setup(
ctx,
&mut app,
title,
start_with_edits,
maybe_mode,
initialize_tutorial,
center_camera,
);
(app, states)
}
}
fn finish_app_setup(
ctx: &mut EventCtx,
app: &mut App,
title: bool,
start_with_edits: Option<String>,
maybe_mode: Option<GameplayMode>,
initialize_tutorial: bool,
center_camera: Option<String>,
) -> Vec<Box<dyn State<App>>> {
if let Some((pt, zoom)) =
center_camera.and_then(|cam| parse_center_camera(ctx, &app.primary.map, cam))
{
ctx.canvas.cam_zoom = zoom;
ctx.canvas.center_on_map_pt(pt);
} else {
app.primary.init_camera_for_loaded_map(ctx, title);
}
let savestate = if app
.primary
.current_flags
.sim_flags
.load
.contains("player/saves/")
{
assert!(maybe_mode.is_none());
Some(app.primary.clear_sim())
} else {
None
};
if let Some(edits_name) = start_with_edits {
let mut timer = Timer::new("apply initial edits");
let edits = map_model::MapEdits::load(
&app.primary.map,
abstio::path_edits(app.primary.map.get_name(), &edits_name),
&mut timer,
)
.unwrap();
crate::edit::apply_map_edits(ctx, app, edits);
app.primary
.map
.recalculate_pathfinding_after_edits(&mut timer);
app.primary.clear_sim();
}
if initialize_tutorial {
crate::sandbox::gameplay::Tutorial::initialize(ctx, app);
}
let start_daytime = Box::new(|ctx: &mut EventCtx, app: &mut App| {
ctx.loading_screen("start in the daytime", |_, mut timer| {
app.primary
.sim
.timed_step(&app.primary.map, Duration::hours(6), &mut None, &mut timer);
});
vec![Transition::Keep]
});
let states: Vec<Box<dyn State<App>>> = if title {
vec![Box::new(TitleScreen::new(ctx, app))]
} else if let Some(mode) = maybe_mode {
if let GameplayMode::Blog(_) = mode {
vec![SandboxMode::async_new(app, mode, start_daytime)]
} else {
vec![SandboxMode::simple_new(app, mode)]
}
} else {
vec![SandboxMode::async_new(
app,
GameplayMode::Freeform(app.primary.map.get_name().clone()),
start_daytime,
)]
};
if let Some(ss) = savestate {
app.primary.sim = ss;
}
states
}
fn parse_center_camera(ctx: &EventCtx, map: &Map, raw: String) -> Option<(Pt2D, f64)> {
let parts: Vec<&str> = raw.split("/").collect();
if parts.len() != 3 {
return None;
}
let zoom_lvl = parts[0].parse::<f64>().ok()?;
let lat = parts[1].parse::<f64>().ok()?;
let lon = parts[2].parse::<f64>().ok()?;
let gps = LonLat::new(lon, lat);
if !map.get_gps_bounds().contains(gps) {
return None;
}
let pt = gps.to_pt(map.get_gps_bounds());
let earth_circumference_equator = 40_075_016.686;
let horiz_meters_per_pixel =
earth_circumference_equator * gps.y().to_radians().cos() / 2.0_f64.powf(zoom_lvl + 8.0);
let horiz_meters_per_screen = ctx.canvas.window_width * horiz_meters_per_pixel;
let cam_zoom = ctx.canvas.window_width / horiz_meters_per_screen;
Some((pt, cam_zoom))
}
#[cfg(target_arch = "wasm32")]
use wasm_bindgen::prelude::*;
#[cfg(target_arch = "wasm32")]
#[wasm_bindgen(start)]
pub fn run() {
main();
}