mod client_init; #[cfg(feature = "singleplayer")] mod start_singleplayer; mod ui; use super::char_selection::CharSelectionState; use crate::{window::Event, Direction, GlobalState, PlayState, PlayStateResult}; use argon2::{self, Config}; use client_init::{ClientInit, Error as InitError}; use common::{clock::Clock, comp}; use log::warn; #[cfg(feature = "singleplayer")] use start_singleplayer::StartSingleplayerState; use std::time::Duration; use ui::{Event as MainMenuEvent, MainMenuUi}; pub struct MainMenuState { main_menu_ui: MainMenuUi, } impl MainMenuState { /// Create a new `MainMenuState`. pub fn new(global_state: &mut GlobalState) -> Self { Self { main_menu_ui: MainMenuUi::new(global_state), } } } const DEFAULT_PORT: u16 = 14004; impl PlayState for MainMenuState { fn play(&mut self, _: Direction, global_state: &mut GlobalState) -> PlayStateResult { // Set up an fps clock. let mut clock = Clock::start(); // Used for client creation. let mut client_init: Option = None; // Kick off title music global_state .audio .play_music("voxygen.audio.soundtrack.veloren_title_tune-3"); loop { // Handle window events. for event in global_state.window.fetch_events() { match event { Event::Close => return PlayStateResult::Shutdown, // Pass events to ui. Event::Ui(event) => { self.main_menu_ui.handle_event(event); } // Ignore all other events. _ => {} } } global_state.window.renderer_mut().clear(); // Poll client creation. match client_init.as_ref().and_then(|init| init.poll()) { Some(Ok(client)) => { self.main_menu_ui.connected(); return PlayStateResult::Push(Box::new(CharSelectionState::new( global_state, std::rc::Rc::new(std::cell::RefCell::new(client)), ))); } Some(Err(err)) => { client_init = None; self.main_menu_ui.login_error( match err { InitError::BadAddress(_) | InitError::NoAddress => "Server not found", InitError::InvalidAuth => "Invalid credentials", InitError::ServerIsFull => "Server is Full!", InitError::ConnectionFailed(_) => "Connection failed", InitError::ClientCrashed => "Client crashed", } .to_string(), ); } None => {} } // Maintain global_state global_state.maintain(clock.get_last_delta().as_secs_f32()); // Maintain the UI. for event in self.main_menu_ui.maintain(global_state) { match event { MainMenuEvent::LoginAttempt { username, password, server_address, } => { let mut net_settings = &mut global_state.settings.networking; net_settings.username = username.clone(); net_settings.password = password.clone(); if !net_settings.servers.contains(&server_address) { net_settings.servers.push(server_address.clone()); } if let Err(err) = global_state.settings.save_to_file() { warn!("Failed to save settings: {:?}", err); } let player = comp::Player::new( username.clone(), Some(global_state.settings.graphics.view_distance), ); if player.is_valid() { // Don't try to connect if there is already a connection in progress. client_init = client_init.or(Some(ClientInit::new( (server_address, DEFAULT_PORT, false), player, { let salt = b"staticsalt_zTuGkGvybZIjZbNUDtw15"; let config = Config::default(); argon2::hash_encoded(password.as_bytes(), salt, &config) .unwrap() }, false, ))); } else { self.main_menu_ui .login_error("Invalid username or password".to_string()); } } MainMenuEvent::CancelLoginAttempt => { // client_init contains Some(ClientInit), which spawns a thread which contains a TcpStream::connect() call // This call is blocking // TODO fix when the network rework happens client_init = None; self.main_menu_ui.cancel_connection(); } #[cfg(feature = "singleplayer")] MainMenuEvent::StartSingleplayer => { return PlayStateResult::Push(Box::new(StartSingleplayerState::new())); } MainMenuEvent::Settings => {} // TODO MainMenuEvent::Quit => return PlayStateResult::Shutdown, MainMenuEvent::DisclaimerClosed => { global_state.settings.show_disclaimer = false } } } // Draw the UI to the screen. self.main_menu_ui.render(global_state.window.renderer_mut()); // Finish the frame. global_state.window.renderer_mut().flush(); global_state .window .swap_buffers() .expect("Failed to swap window buffers!"); // Wait for the next tick clock.tick(Duration::from_millis( 1000 / (global_state.settings.graphics.max_fps as u64), )); } } fn name(&self) -> &'static str { "Title" } }