2019-01-12 15:57:19 +00:00
|
|
|
// Standard
|
|
|
|
use std::time::Duration;
|
|
|
|
|
2019-01-11 23:18:34 +00:00
|
|
|
// Library
|
|
|
|
use vek::*;
|
|
|
|
|
2019-01-12 15:57:19 +00:00
|
|
|
// Project
|
|
|
|
use common::clock::Clock;
|
2019-01-14 15:47:57 +00:00
|
|
|
use client::{
|
|
|
|
self,
|
|
|
|
Client,
|
|
|
|
};
|
2019-01-12 15:57:19 +00:00
|
|
|
|
2019-01-11 23:18:34 +00:00
|
|
|
// Crate
|
|
|
|
use crate::{
|
2019-01-14 15:47:57 +00:00
|
|
|
Error,
|
2019-01-11 23:18:34 +00:00
|
|
|
PlayState,
|
|
|
|
PlayStateResult,
|
|
|
|
GlobalState,
|
2019-03-02 03:48:30 +00:00
|
|
|
key_state::KeyState,
|
2019-02-16 03:01:42 +00:00
|
|
|
window::{Event, Key, Window},
|
2019-01-11 23:18:34 +00:00
|
|
|
render::Renderer,
|
|
|
|
scene::Scene,
|
2019-03-16 02:03:21 +00:00
|
|
|
hud::{Hud, Event as HudEvent},
|
2019-01-11 23:18:34 +00:00
|
|
|
};
|
|
|
|
|
2019-01-12 15:57:19 +00:00
|
|
|
const FPS: u64 = 60;
|
|
|
|
|
2019-01-11 23:18:34 +00:00
|
|
|
pub struct SessionState {
|
|
|
|
scene: Scene,
|
2019-01-14 15:47:57 +00:00
|
|
|
client: Client,
|
2019-03-02 03:48:30 +00:00
|
|
|
key_state: KeyState,
|
2019-03-15 04:55:52 +00:00
|
|
|
hud: Hud,
|
2019-01-11 23:18:34 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Represents an active game session (i.e: one that is being played)
|
|
|
|
impl SessionState {
|
|
|
|
/// Create a new `SessionState`
|
2019-02-16 03:01:42 +00:00
|
|
|
pub fn new(window: &mut Window) -> Result<Self, Error> {
|
2019-03-03 22:11:38 +00:00
|
|
|
let client = Client::new(([127, 0, 0, 1], 59003))?.with_test_state(); // <--- TODO: Remove this
|
|
|
|
Ok(Self {
|
2019-01-11 23:18:34 +00:00
|
|
|
// Create a scene for this session. The scene handles visible elements of the game world
|
2019-02-16 03:01:42 +00:00
|
|
|
scene: Scene::new(window.renderer_mut(), &client),
|
2019-01-15 15:13:11 +00:00
|
|
|
client,
|
2019-03-02 03:48:30 +00:00
|
|
|
key_state: KeyState::new(),
|
2019-03-15 04:55:52 +00:00
|
|
|
hud: Hud::new(window),
|
2019-03-03 22:11:38 +00:00
|
|
|
})
|
2019-01-11 23:18:34 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// The background colour
|
|
|
|
const BG_COLOR: Rgba<f32> = Rgba { r: 0.0, g: 0.3, b: 1.0, a: 1.0 };
|
|
|
|
|
2019-01-12 15:57:19 +00:00
|
|
|
impl SessionState {
|
2019-01-14 15:47:57 +00:00
|
|
|
/// Tick the session (and the client attached to it)
|
|
|
|
pub fn tick(&mut self, dt: Duration) -> Result<(), Error> {
|
2019-03-02 03:48:30 +00:00
|
|
|
// Calculate the movement input vector of the player from the current key presses and the camera direction
|
|
|
|
let ori = self.scene.camera().get_orientation();
|
|
|
|
let unit_vecs = (
|
|
|
|
Vec2::new(ori[0].cos(), -ori[0].sin()),
|
|
|
|
Vec2::new(ori[0].sin(), ori[0].cos()),
|
|
|
|
);
|
|
|
|
let dir_vec = self.key_state.dir_vec();
|
2019-03-02 19:43:51 +00:00
|
|
|
let move_dir = unit_vecs.0 * dir_vec[0] + unit_vecs.1 * dir_vec[1];
|
2019-03-02 03:48:30 +00:00
|
|
|
|
2019-03-17 05:26:51 +00:00
|
|
|
for event in self.client.tick(client::Input { move_dir }, dt)? {
|
|
|
|
match event {
|
|
|
|
client::Event::Chat(msg) => {
|
|
|
|
self.hud.new_message(msg);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-01-14 15:47:57 +00:00
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
2019-01-23 20:01:58 +00:00
|
|
|
/// Clean up the session (and the client attached to it) after a tick
|
|
|
|
pub fn cleanup(&mut self) {
|
|
|
|
self.client.cleanup();
|
|
|
|
}
|
|
|
|
|
2019-01-12 15:57:19 +00:00
|
|
|
/// Render the session to the screen.
|
|
|
|
///
|
|
|
|
/// This method should be called once per frame.
|
|
|
|
pub fn render(&mut self, renderer: &mut Renderer) {
|
|
|
|
// Clear the screen
|
|
|
|
renderer.clear(BG_COLOR);
|
|
|
|
|
|
|
|
// Render the screen using the global renderer
|
|
|
|
self.scene.render_to(renderer);
|
2019-02-16 03:01:42 +00:00
|
|
|
// Draw the UI to the screen
|
2019-03-15 04:55:52 +00:00
|
|
|
self.hud.render(renderer);
|
2019-01-12 15:57:19 +00:00
|
|
|
|
|
|
|
// Finish the frame
|
|
|
|
renderer.flush();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-01-11 23:18:34 +00:00
|
|
|
impl PlayState for SessionState {
|
|
|
|
fn play(&mut self, global_state: &mut GlobalState) -> PlayStateResult {
|
2019-01-12 01:14:58 +00:00
|
|
|
// Trap the cursor
|
2019-01-23 22:21:47 +00:00
|
|
|
global_state.window.grab_cursor(true);
|
2019-01-12 01:14:58 +00:00
|
|
|
|
2019-01-12 15:57:19 +00:00
|
|
|
// Set up an fps clock
|
|
|
|
let mut clock = Clock::new();
|
|
|
|
|
2019-01-23 20:01:58 +00:00
|
|
|
// Load a few chunks TODO: Remove this
|
2019-01-23 22:39:31 +00:00
|
|
|
for x in -6..7 {
|
|
|
|
for y in -6..7 {
|
2019-01-23 20:01:58 +00:00
|
|
|
for z in -1..2 {
|
|
|
|
self.client.load_chunk(Vec3::new(x, y, z));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-01-11 23:18:34 +00:00
|
|
|
// Game loop
|
|
|
|
loop {
|
|
|
|
// Handle window events
|
|
|
|
for event in global_state.window.fetch_events() {
|
2019-03-15 04:55:52 +00:00
|
|
|
|
2019-03-22 03:55:42 +00:00
|
|
|
// Pass all events to the ui first
|
|
|
|
if self.hud.handle_event(event.clone()) {
|
|
|
|
continue;
|
|
|
|
}
|
2019-01-12 13:56:34 +00:00
|
|
|
let _handled = match event {
|
2019-01-11 23:18:34 +00:00
|
|
|
Event::Close => return PlayStateResult::Shutdown,
|
|
|
|
// When 'q' is pressed, exit the session
|
|
|
|
Event::Char('q') => return PlayStateResult::Pop,
|
2019-02-23 02:41:52 +00:00
|
|
|
// When 'm' is pressed, open/close the in-game test menu
|
2019-03-15 04:55:52 +00:00
|
|
|
Event::Char('m') => self.hud.toggle_menu(),
|
|
|
|
// Close windows on esc
|
|
|
|
Event::KeyDown(Key::Escape) => self.hud.toggle_windows(),
|
2019-01-23 22:21:47 +00:00
|
|
|
// Toggle cursor grabbing
|
|
|
|
Event::KeyDown(Key::ToggleCursor) => {
|
|
|
|
global_state.window.grab_cursor(!global_state.window.is_cursor_grabbed());
|
2019-03-22 03:55:42 +00:00
|
|
|
self.hud.update_grab(global_state.window.is_cursor_grabbed());
|
2019-01-23 22:21:47 +00:00
|
|
|
},
|
2019-03-02 03:48:30 +00:00
|
|
|
// Movement Key Pressed
|
|
|
|
Event::KeyDown(Key::MoveForward) => self.key_state.up = true,
|
|
|
|
Event::KeyDown(Key::MoveBack) => self.key_state.down = true,
|
|
|
|
Event::KeyDown(Key::MoveLeft) => self.key_state.left = true,
|
|
|
|
Event::KeyDown(Key::MoveRight) => self.key_state.right = true,
|
|
|
|
// Movement Key Released
|
|
|
|
Event::KeyUp(Key::MoveForward) => self.key_state.up = false,
|
|
|
|
Event::KeyUp(Key::MoveBack) => self.key_state.down = false,
|
|
|
|
Event::KeyUp(Key::MoveLeft) => self.key_state.left = false,
|
|
|
|
Event::KeyUp(Key::MoveRight) => self.key_state.right = false,
|
2019-01-12 13:56:34 +00:00
|
|
|
// Pass all other events to the scene
|
2019-01-23 22:21:47 +00:00
|
|
|
event => { self.scene.handle_input_event(event); },
|
2019-01-12 13:56:34 +00:00
|
|
|
};
|
|
|
|
// TODO: Do something if the event wasn't handled?
|
2019-01-11 23:18:34 +00:00
|
|
|
}
|
|
|
|
|
2019-01-12 15:57:19 +00:00
|
|
|
// Perform an in-game tick
|
2019-01-14 15:47:57 +00:00
|
|
|
self.tick(clock.get_last_delta())
|
2019-01-12 15:57:19 +00:00
|
|
|
.expect("Failed to tick the scene");
|
2019-01-11 23:18:34 +00:00
|
|
|
|
2019-01-23 20:01:58 +00:00
|
|
|
// Maintain the scene
|
|
|
|
self.scene.maintain(global_state.window.renderer_mut(), &self.client);
|
2019-02-16 03:01:42 +00:00
|
|
|
// Maintain the UI
|
2019-03-16 02:03:21 +00:00
|
|
|
for event in self.hud.maintain(global_state.window.renderer_mut()) {
|
|
|
|
match event {
|
2019-03-17 05:26:51 +00:00
|
|
|
HudEvent::SendMessage(msg) => {
|
|
|
|
// TODO: Handle result
|
|
|
|
self.client.send_chat(msg);
|
|
|
|
},
|
2019-03-16 02:03:21 +00:00
|
|
|
HudEvent::Logout => return PlayStateResult::Pop,
|
|
|
|
HudEvent::Quit => return PlayStateResult::Shutdown,
|
|
|
|
}
|
|
|
|
}
|
2019-01-23 20:01:58 +00:00
|
|
|
|
2019-01-12 15:57:19 +00:00
|
|
|
// Render the session
|
|
|
|
self.render(global_state.window.renderer_mut());
|
2019-01-11 23:18:34 +00:00
|
|
|
|
2019-01-12 15:57:19 +00:00
|
|
|
// Display the frame on the window
|
2019-01-12 01:14:58 +00:00
|
|
|
global_state.window
|
|
|
|
.swap_buffers()
|
|
|
|
.expect("Failed to swap window buffers");
|
2019-01-12 15:57:19 +00:00
|
|
|
|
|
|
|
// Wait for the next tick
|
|
|
|
clock.tick(Duration::from_millis(1000 / FPS));
|
2019-01-23 20:01:58 +00:00
|
|
|
|
|
|
|
// Clean things up after the tick
|
|
|
|
self.cleanup();
|
2019-01-11 23:18:34 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn name(&self) -> &'static str { "Session" }
|
|
|
|
}
|