From af342d8dc61ce57ee32df3df18241b981fb48976 Mon Sep 17 00:00:00 2001 From: Evan Pratten Date: Fri, 23 Apr 2021 20:41:04 -0400 Subject: [PATCH 01/27] nalgebra --- Cargo.toml | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/Cargo.toml b/Cargo.toml index 5231d76..f53aea5 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -13,4 +13,5 @@ serde_json = "1.0.64" failure = "0.1.8" parry2d = "0.4.0" log = "0.4.14" -env_logger = "0.8.3" \ No newline at end of file +env_logger = "0.8.3" +nalgebra = "0.26.1" \ No newline at end of file From 51fc8adac190f8ff2343b307145a74c2f277806f Mon Sep 17 00:00:00 2001 From: Evan Pratten Date: Fri, 23 Apr 2021 22:39:28 -0400 Subject: [PATCH 02/27] Begining player rendering Co-authored-by: William Meathrel Co-authored-by: Catarina Burghi --- src/logic/ingame/ingame.rs | 12 ++++++++++++ 1 file changed, 12 insertions(+) create mode 100644 src/logic/ingame/ingame.rs diff --git a/src/logic/ingame/ingame.rs b/src/logic/ingame/ingame.rs new file mode 100644 index 0000000..40e99da --- /dev/null +++ b/src/logic/ingame/ingame.rs @@ -0,0 +1,12 @@ + + + +pub enum ingame_states{ + BUYING, + SWIMMING, + DEAD, +} + + + + From 534c9bd800715823e951e96fbc70771927ab7c9c Mon Sep 17 00:00:00 2001 From: Evan Pratten Date: Fri, 23 Apr 2021 22:39:43 -0400 Subject: [PATCH 03/27] player rendering --- assets/worlds/mainworld.json | 3 ++ src/gamecore.rs | 19 ++++++--- src/logic/ingame/mod.rs | 77 +++++++++++++++++++++++++++++++++++- src/logic/mainmenu.rs | 46 +++++++++++++++++++-- src/main.rs | 19 +++++++-- src/player.rs | 19 +++++++++ src/world.rs | 19 +++++++++ 7 files changed, 189 insertions(+), 13 deletions(-) create mode 100644 assets/worlds/mainworld.json create mode 100644 src/player.rs create mode 100644 src/world.rs diff --git a/assets/worlds/mainworld.json b/assets/worlds/mainworld.json new file mode 100644 index 0000000..544b7b4 --- /dev/null +++ b/assets/worlds/mainworld.json @@ -0,0 +1,3 @@ +{ + +} \ No newline at end of file diff --git a/src/gamecore.rs b/src/gamecore.rs index edc827d..ef7f71a 100644 --- a/src/gamecore.rs +++ b/src/gamecore.rs @@ -6,7 +6,7 @@ use raylib::{ camera::Camera2D, math::Vector2, prelude::RaylibDrawHandle, RaylibHandle, RaylibThread, }; -use crate::resources::GlobalResources; +use crate::{player::Player, resources::GlobalResources, world::World}; use log::debug; @@ -16,7 +16,8 @@ pub enum GameState { Loading, MainMenu, PauseMenu, - GameQuit + GameQuit, + InGame } impl fmt::Display for GameState { @@ -40,11 +41,17 @@ pub struct GameCore { pub master_camera: Camera2D, /// Debug features - pub show_simple_debug_info: bool + pub show_simple_debug_info: bool, + + /// The world + pub world: World, + + /// The player + pub player: Player, } impl GameCore { - pub fn new(raylib: &mut RaylibHandle, thread: &RaylibThread) -> Self { + pub fn new(raylib: &mut RaylibHandle, thread: &RaylibThread, world: World) -> Self { Self { state: GameState::Loading, last_state: GameState::Loading, @@ -58,7 +65,9 @@ impl GameCore { rotation: 0.0, zoom: 1.0, }, - show_simple_debug_info: false + show_simple_debug_info: false, + world: world, + player: Player::new() } } diff --git a/src/logic/ingame/mod.rs b/src/logic/ingame/mod.rs index 81e542c..8f58e8d 100644 --- a/src/logic/ingame/mod.rs +++ b/src/logic/ingame/mod.rs @@ -7,11 +7,60 @@ use crate::{ use super::screen::Screen; -pub struct InGameScreen {} +pub enum InGameState { + BUYING, + SWIMMING, + DEAD, +} + +pub struct InGameScreen { + current_state: InGameState, +} impl InGameScreen { pub fn new() -> Self { - Self {} + Self { + current_state: InGameState::SWIMMING, + } + } + + fn update_player_movement( + &mut self, + draw_handle: &mut RaylibDrawHandle, + game_core: &mut GameCore, + ) { + let player_screen_position = + draw_handle.get_screen_to_world2D(game_core.player.position, game_core.master_camera); + + // Handle player movement + let mouse_pose = draw_handle.get_mouse_position(); + let mut raw_movement_direction = mouse_pose - player_screen_position; + raw_movement_direction.normalize(); + game_core.player.direction = raw_movement_direction; + + // Handle action buttons + let user_request_boost = + draw_handle.is_mouse_button_pressed(MouseButton::MOUSE_LEFT_BUTTON); + let user_request_action = + draw_handle.is_mouse_button_pressed(MouseButton::MOUSE_RIGHT_BUTTON); + } + + fn render_player( + &mut self, + context_2d: &mut RaylibMode2D, + game_core: &mut GameCore, + ) { + // Get the player + let player = &game_core.player; + + // TODO: tmp rect + context_2d.draw_rectangle( + player.position.x as i32 - 10, + player.position.y as i32 - 10, + 20, + 30, + Color::BLACK, + ); } } @@ -26,6 +75,30 @@ impl Screen for InGameScreen { // Clear frame draw_handle.clear_background(Color::WHITE); + // Handle the pause menu being opened + if draw_handle.is_key_pressed(KeyboardKey::KEY_ESCAPE) { + return Some(GameState::PauseMenu); + } + + // Window dimensions + let win_height = draw_handle.get_screen_height(); + let win_width = draw_handle.get_screen_width(); + let window_center = Vector2 { + x: (win_width as f32 / 2.0), + y: (win_height as f32 / 2.0), + }; + + // Update player movement + self.update_player_movement(draw_handle, game_core); + + // Open a 2D context + { + let mut context_2d = draw_handle.begin_mode2D(game_core.master_camera); + + // Render Player + self.render_player(&mut context_2d, game_core); + } + return None; } } diff --git a/src/logic/mainmenu.rs b/src/logic/mainmenu.rs index 662c9ad..893494e 100644 --- a/src/logic/mainmenu.rs +++ b/src/logic/mainmenu.rs @@ -23,13 +23,53 @@ impl Screen for MainMenuScreen { audio_system: &mut AudioPlayer, game_core: &mut GameCore, ) -> Option { + // Window dimensions + let win_height = draw_handle.get_screen_height(); + let win_width = draw_handle.get_screen_width(); // Clear frame draw_handle.clear_background(Color::WHITE); - // TODO: This is only for testing - if draw_handle.is_key_pressed(KeyboardKey::KEY_ESCAPE) { - return Some(GameState::PauseMenu); + // Render title + draw_handle.draw_text( + "TMP TITLE", + (win_height / 2) - 80, + win_width / 4, + 40, + Color::BLACK, + ); + + // Play and quit + draw_handle.draw_text( + "Play", + (win_height / 2) - 80, + (win_width / 4) + 100, + 20, + Color::BLACK, + ); + draw_handle.draw_text( + "Quit", + (win_height / 2) - 80, + (win_width / 4) + 140, + 20, + Color::BLACK, + ); + + // Handle button presses + let mouse_position = draw_handle.get_mouse_position(); + let mouse_clicked = draw_handle.is_mouse_button_pressed(MouseButton::MOUSE_LEFT_BUTTON); + + // Check clicks + if mouse_clicked { + if mouse_position.y > (win_width as f32 / 4.0) + 100.0 + && mouse_position.y < (win_width as f32 / 4.0) + 120.0 + { + return Some(GameState::InGame); + } else if mouse_position.y > (win_width as f32 / 4.0) + 140.0 + && mouse_position.y < (win_width as f32 / 4.0) + 180.0 + { + return Some(GameState::GameQuit); + } } return None; diff --git a/src/main.rs b/src/main.rs index 186107e..241418d 100644 --- a/src/main.rs +++ b/src/main.rs @@ -2,15 +2,18 @@ mod gamecore; mod lib; mod logic; mod resources; +mod player; +mod world; use gamecore::{GameCore, GameState}; use lib::{utils::profiler::GameProfiler, wrappers::audio::player::AudioPlayer}; use log::info; use logic::{ - loadingscreen::LoadingScreen, mainmenu::MainMenuScreen, pausemenu::PauseMenuScreen, - screen::Screen, + ingame::InGameScreen, loadingscreen::LoadingScreen, mainmenu::MainMenuScreen, + pausemenu::PauseMenuScreen, screen::Screen, }; use raylib::prelude::*; +use world::World; // Game Launch Configuration const DEFAULT_WINDOW_DIMENSIONS: Vector2 = Vector2 { @@ -37,8 +40,11 @@ fn main() { // Override the default exit key raylib.set_exit_key(None); + // Load the world + let world = World::load_from_json("./assets/worlds/mainworld.json".to_string()).expect("Failed to load main world JSON"); + // Set up the game's core state - let mut game_core = GameCore::new(&mut raylib, &raylib_thread); + let mut game_core = GameCore::new(&mut raylib, &raylib_thread, world); // Set up the game's profiler let mut profiler = GameProfiler::new(); @@ -51,6 +57,7 @@ fn main() { let mut loading_screen = LoadingScreen::new(); let mut main_menu_screen = MainMenuScreen::new(); let mut pause_menu_screen = PauseMenuScreen::new(); + let mut ingame_screen = InGameScreen::new(); // Main rendering loop while !raylib.window_should_close() { @@ -77,6 +84,12 @@ fn main() { &mut game_core, ), GameState::GameQuit => None, + GameState::InGame => ingame_screen.render( + &mut draw_handle, + &raylib_thread, + &mut audio_system, + &mut game_core, + ), }; // If needed, update the global state diff --git a/src/player.rs b/src/player.rs new file mode 100644 index 0000000..b5a79aa --- /dev/null +++ b/src/player.rs @@ -0,0 +1,19 @@ +use raylib::math::Vector2; + + + +#[derive(Debug, Default)] +pub struct Player { + pub position: Vector2, + pub direction: Vector2, + pub coins: u32 +} + +impl Player { + pub fn new() -> Self { + Self { + ..Default::default() + + } + } +} \ No newline at end of file diff --git a/src/world.rs b/src/world.rs new file mode 100644 index 0000000..4600c51 --- /dev/null +++ b/src/world.rs @@ -0,0 +1,19 @@ +use std::{fs::File, io::BufReader}; + +use serde::{Deserialize, Serialize}; +use std::io::Read; +use failure::Error; + +#[derive(Debug, Serialize, Deserialize, Clone)] +pub struct World {} + +impl World { + pub fn load_from_json(file: String) -> Result { + // Load the file + let file = File::open(file)?; + let reader = BufReader::new(file); + + // Deserialize + Ok(serde_json::from_reader(reader)?) + } +} From 50634c7516243e7314720afabac6bffff8c68bf7 Mon Sep 17 00:00:00 2001 From: Evan Pratten Date: Fri, 23 Apr 2021 22:54:04 -0400 Subject: [PATCH 04/27] update for will --- src/logic/ingame/mod.rs | 19 ++++++++++++++++++- src/player.rs | 4 +++- 2 files changed, 21 insertions(+), 2 deletions(-) diff --git a/src/logic/ingame/mod.rs b/src/logic/ingame/mod.rs index 8f58e8d..fd48cce 100644 --- a/src/logic/ingame/mod.rs +++ b/src/logic/ingame/mod.rs @@ -7,6 +7,10 @@ use crate::{ use super::screen::Screen; +const NORMAL_PLAYER_SPEED: i32 = 4; +const BOOST_PLAYER_SPEED: i32 = NORMAL_PLAYER_SPEED * 2; +const CAMERA_FOLLOW_SPEED: f32 = 3.0; + pub enum InGameState { BUYING, SWIMMING, @@ -40,9 +44,22 @@ impl InGameScreen { // Handle action buttons let user_request_boost = - draw_handle.is_mouse_button_pressed(MouseButton::MOUSE_LEFT_BUTTON); + draw_handle.is_mouse_button_down(MouseButton::MOUSE_LEFT_BUTTON); let user_request_action = draw_handle.is_mouse_button_pressed(MouseButton::MOUSE_RIGHT_BUTTON); + + // Move the player in their direction + let speed_multiplier = match user_request_boost && game_core.player.boost_percent >= 0.0 { + true => BOOST_PLAYER_SPEED as f32, + false => NORMAL_PLAYER_SPEED as f32 + }; + game_core.player.position += game_core.player.direction * speed_multiplier; + + // Move the camera to follow the player + let direction_from_cam_to_player = player_screen_position; + // game_core.master_camera.offset -= direction_from_cam_to_player * CAMERA_FOLLOW_SPEED; + game_core.master_camera.target = game_core.player.position + 0; + } fn render_player( diff --git a/src/player.rs b/src/player.rs index b5a79aa..3fc7c14 100644 --- a/src/player.rs +++ b/src/player.rs @@ -6,12 +6,14 @@ use raylib::math::Vector2; pub struct Player { pub position: Vector2, pub direction: Vector2, - pub coins: u32 + pub coins: u32, + pub boost_percent: f32 } impl Player { pub fn new() -> Self { Self { + boost_percent: 1.0, ..Default::default() } From 40b7421f24cbdf68d3675afb4f0487f589df022e Mon Sep 17 00:00:00 2001 From: Evan Pratten Date: Fri, 23 Apr 2021 22:59:37 -0400 Subject: [PATCH 05/27] camera lock --- src/logic/ingame/mod.rs | 9 +++++---- 1 file changed, 5 insertions(+), 4 deletions(-) diff --git a/src/logic/ingame/mod.rs b/src/logic/ingame/mod.rs index fd48cce..ce2e475 100644 --- a/src/logic/ingame/mod.rs +++ b/src/logic/ingame/mod.rs @@ -9,7 +9,7 @@ use super::screen::Screen; const NORMAL_PLAYER_SPEED: i32 = 4; const BOOST_PLAYER_SPEED: i32 = NORMAL_PLAYER_SPEED * 2; -const CAMERA_FOLLOW_SPEED: f32 = 3.0; +const CAMERA_FOLLOW_SPEED: f32 = 1.0; pub enum InGameState { BUYING, @@ -32,6 +32,7 @@ impl InGameScreen { &mut self, draw_handle: &mut RaylibDrawHandle, game_core: &mut GameCore, + window_center: Vector2 ) { let player_screen_position = draw_handle.get_screen_to_world2D(game_core.player.position, game_core.master_camera); @@ -56,9 +57,9 @@ impl InGameScreen { game_core.player.position += game_core.player.direction * speed_multiplier; // Move the camera to follow the player - let direction_from_cam_to_player = player_screen_position; + let direction_from_cam_to_player = (game_core.player.position - window_center) - game_core.master_camera.target; // game_core.master_camera.offset -= direction_from_cam_to_player * CAMERA_FOLLOW_SPEED; - game_core.master_camera.target = game_core.player.position + 0; + game_core.master_camera.target += direction_from_cam_to_player * CAMERA_FOLLOW_SPEED; } @@ -106,7 +107,7 @@ impl Screen for InGameScreen { }; // Update player movement - self.update_player_movement(draw_handle, game_core); + self.update_player_movement(draw_handle, game_core, window_center); // Open a 2D context { From 79cf3e727415287a7c7e887e4953f9471f8e3aa1 Mon Sep 17 00:00:00 2001 From: Evan Pratten Date: Fri, 23 Apr 2021 23:34:07 -0400 Subject: [PATCH 06/27] Basic movement --- src/logic/ingame/mod.rs | 74 ++++++++++++++++++++++++++++++----------- src/player.rs | 5 +++ 2 files changed, 59 insertions(+), 20 deletions(-) diff --git a/src/logic/ingame/mod.rs b/src/logic/ingame/mod.rs index ce2e475..b876de3 100644 --- a/src/logic/ingame/mod.rs +++ b/src/logic/ingame/mod.rs @@ -9,7 +9,7 @@ use super::screen::Screen; const NORMAL_PLAYER_SPEED: i32 = 4; const BOOST_PLAYER_SPEED: i32 = NORMAL_PLAYER_SPEED * 2; -const CAMERA_FOLLOW_SPEED: f32 = 1.0; +const CAMERA_FOLLOW_SPEED: f32 = 0.7; pub enum InGameState { BUYING, @@ -32,35 +32,49 @@ impl InGameScreen { &mut self, draw_handle: &mut RaylibDrawHandle, game_core: &mut GameCore, - window_center: Vector2 + window_center: Vector2, ) { - let player_screen_position = - draw_handle.get_screen_to_world2D(game_core.player.position, game_core.master_camera); // Handle player movement let mouse_pose = draw_handle.get_mouse_position(); - let mut raw_movement_direction = mouse_pose - player_screen_position; - raw_movement_direction.normalize(); - game_core.player.direction = raw_movement_direction; + let mouse_world_pose = draw_handle.get_screen_to_world2D(mouse_pose, game_core.master_camera); + let raw_movement_direction = mouse_world_pose - game_core.player.position; + let mut normalized_movement_direction = raw_movement_direction; + normalized_movement_direction.normalize(); + game_core.player.direction = normalized_movement_direction; + + // In the case the player is in "null", just jump the camera to them + if game_core.player.position == Vector2::zero() { + game_core.master_camera.target = game_core.player.position - (window_center / 2.0); + } // Handle action buttons - let user_request_boost = - draw_handle.is_mouse_button_down(MouseButton::MOUSE_LEFT_BUTTON); + let user_request_boost = draw_handle.is_mouse_button_down(MouseButton::MOUSE_LEFT_BUTTON); let user_request_action = draw_handle.is_mouse_button_pressed(MouseButton::MOUSE_RIGHT_BUTTON); // Move the player in their direction let speed_multiplier = match user_request_boost && game_core.player.boost_percent >= 0.0 { true => BOOST_PLAYER_SPEED as f32, - false => NORMAL_PLAYER_SPEED as f32 + false => NORMAL_PLAYER_SPEED as f32, }; - game_core.player.position += game_core.player.direction * speed_multiplier; + + // Only do this if the mouse is far enough away + let player_real_movement = game_core.player.direction * speed_multiplier; + if raw_movement_direction.distance_to(Vector2::zero()) > game_core.player.size.y / 2.0 { + game_core.player.position += player_real_movement; + } // Move the camera to follow the player - let direction_from_cam_to_player = (game_core.player.position - window_center) - game_core.master_camera.target; - // game_core.master_camera.offset -= direction_from_cam_to_player * CAMERA_FOLLOW_SPEED; - game_core.master_camera.target += direction_from_cam_to_player * CAMERA_FOLLOW_SPEED; - + let direction_from_cam_to_player = + (game_core.player.position - window_center) - game_core.master_camera.target; + let player_screen_position = draw_handle.get_world_to_screen2D(game_core.player.position, game_core.master_camera); + + // Camera only moves if you get close to the edge of the screen + if player_screen_position.distance_to(window_center).abs() > (window_center.y - 40.0) { + game_core.master_camera.target += player_real_movement; + } + } fn render_player( @@ -71,15 +85,32 @@ impl InGameScreen { // Get the player let player = &game_core.player; + // Convert the player direction to a rotation + let player_rotation = Vector2::zero().angle_to(player.direction); + // TODO: tmp rect - context_2d.draw_rectangle( - player.position.x as i32 - 10, - player.position.y as i32 - 10, - 20, - 30, + context_2d.draw_rectangle_pro( + Rectangle { + x: player.position.x, + y: player.position.y, + width: player.size.x, + height: player.size.y, + }, + player.size / 2.0, + player_rotation.to_degrees() + 90.0, Color::BLACK, ); } + + fn render_world( + &mut self, + context_2d: &mut RaylibMode2D, + game_core: &mut GameCore, + ) { + + context_2d.draw_circle(0, 0, 10.0, Color::BLACK); + + } } impl Screen for InGameScreen { @@ -113,6 +144,9 @@ impl Screen for InGameScreen { { let mut context_2d = draw_handle.begin_mode2D(game_core.master_camera); + // Render the world + self.render_world(&mut context_2d, game_core); + // Render Player self.render_player(&mut context_2d, game_core); } diff --git a/src/player.rs b/src/player.rs index 3fc7c14..3fa3a58 100644 --- a/src/player.rs +++ b/src/player.rs @@ -6,6 +6,7 @@ use raylib::math::Vector2; pub struct Player { pub position: Vector2, pub direction: Vector2, + pub size: Vector2, pub coins: u32, pub boost_percent: f32 } @@ -14,6 +15,10 @@ impl Player { pub fn new() -> Self { Self { boost_percent: 1.0, + size: Vector2 { + x: 11.0 * 4.0, + y: 21.0 * 4.0 + }, ..Default::default() } From bd24162fbd3d29a27775f22312f023f20dd1a539 Mon Sep 17 00:00:00 2001 From: Evan Pratten Date: Fri, 23 Apr 2021 23:56:11 -0400 Subject: [PATCH 07/27] Adding hud data --- Cargo.toml | 6 ++++-- assets/worlds/mainworld.json | 5 ++++- src/lib/utils/profiler.rs | 34 +++++++++++++++++++++++++++++++++- src/logic/ingame/mod.rs | 26 +++++++++++++++++++------- src/main.rs | 3 +++ src/player.rs | 4 +++- src/world.rs | 5 ++++- 7 files changed, 70 insertions(+), 13 deletions(-) diff --git a/Cargo.toml b/Cargo.toml index f53aea5..36bf495 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -6,7 +6,9 @@ edition = "2018" description = "" [dependencies] -raylib = { version = "3.5", git = "https://github.com/ewpratten/raylib-rs", branch = "master" } +raylib = { version = "3.5", git = "https://github.com/ewpratten/raylib-rs", branch = "master", features = [ + "with_serde" +] } serialstudio = "0.1.0" serde = "1.0.125" serde_json = "1.0.64" @@ -14,4 +16,4 @@ failure = "0.1.8" parry2d = "0.4.0" log = "0.4.14" env_logger = "0.8.3" -nalgebra = "0.26.1" \ No newline at end of file +nalgebra = "0.26.1" diff --git a/assets/worlds/mainworld.json b/assets/worlds/mainworld.json index 544b7b4..1b50365 100644 --- a/assets/worlds/mainworld.json +++ b/assets/worlds/mainworld.json @@ -1,3 +1,6 @@ { - + "end_position": { + "x": 100.0, + "y": 100.0 + } } \ No newline at end of file diff --git a/src/lib/utils/profiler.rs b/src/lib/utils/profiler.rs index 37293c2..6634e5b 100644 --- a/src/lib/utils/profiler.rs +++ b/src/lib/utils/profiler.rs @@ -16,7 +16,12 @@ pub struct ProfilerData { pub active_sounds: i32, // Game core - pub game_state: String + pub game_state: String, + + // Player + pub player_coins: u32, + pub player_boost_percent: f32, + pub player_breath_percent: f32 } /// The development profiler @@ -117,6 +122,33 @@ impl GameProfiler { }, ], }, + DataGroup { + title: "Player".to_string(), + widget_type: None, + datasets: vec![ + DataSet { + title: Some("Coins".to_string()), + value: json!(self.data.player_coins), + graph: Some(false), + unit: Some("coins".to_string()), + w_type: None, + }, + DataSet { + title: Some("Breath".to_string()), + value: json!(self.data.player_boost_percent), + graph: Some(false), + unit: Some("%".to_string()), + w_type: None, + }, + DataSet { + title: Some("Breath".to_string()), + value: json!(self.data.player_breath_percent), + graph: Some(false), + unit: Some("%".to_string()), + w_type: None, + }, + ], + }, ], }; diff --git a/src/logic/ingame/mod.rs b/src/logic/ingame/mod.rs index b876de3..62f69e0 100644 --- a/src/logic/ingame/mod.rs +++ b/src/logic/ingame/mod.rs @@ -34,15 +34,15 @@ impl InGameScreen { game_core: &mut GameCore, window_center: Vector2, ) { - // Handle player movement let mouse_pose = draw_handle.get_mouse_position(); - let mouse_world_pose = draw_handle.get_screen_to_world2D(mouse_pose, game_core.master_camera); + let mouse_world_pose = + draw_handle.get_screen_to_world2D(mouse_pose, game_core.master_camera); let raw_movement_direction = mouse_world_pose - game_core.player.position; let mut normalized_movement_direction = raw_movement_direction; normalized_movement_direction.normalize(); game_core.player.direction = normalized_movement_direction; - + // In the case the player is in "null", just jump the camera to them if game_core.player.position == Vector2::zero() { game_core.master_camera.target = game_core.player.position - (window_center / 2.0); @@ -68,13 +68,13 @@ impl InGameScreen { // Move the camera to follow the player let direction_from_cam_to_player = (game_core.player.position - window_center) - game_core.master_camera.target; - let player_screen_position = draw_handle.get_world_to_screen2D(game_core.player.position, game_core.master_camera); + let player_screen_position = + draw_handle.get_world_to_screen2D(game_core.player.position, game_core.master_camera); // Camera only moves if you get close to the edge of the screen if player_screen_position.distance_to(window_center).abs() > (window_center.y - 40.0) { game_core.master_camera.target += player_real_movement; } - } fn render_player( @@ -107,9 +107,21 @@ impl InGameScreen { context_2d: &mut RaylibMode2D, game_core: &mut GameCore, ) { - context_2d.draw_circle(0, 0, 10.0, Color::BLACK); - + } + + fn render_hud(&mut self, draw_handle: &mut RaylibDrawHandle, game_core: &mut GameCore) { + // Get the relevant data + let breath = game_core.player.breath_percent; + let dist_from_player_to_end = game_core + .player + .position + .distance_to(game_core.world.end_position); + let dist_from_start_to_end = Vector2::zero().distance_to(game_core.world.end_position); + let progress = (dist_from_start_to_end - dist_from_player_to_end) / dist_from_start_to_end; + + + } } diff --git a/src/main.rs b/src/main.rs index 241418d..65d79ad 100644 --- a/src/main.rs +++ b/src/main.rs @@ -120,6 +120,9 @@ fn main() { profiler.data.audio_volume = audio_system.get_master_volume(); profiler.data.active_sounds = audio_system.get_sounds_playing(); profiler.data.game_state = game_core.state.to_string(); + profiler.data.player_coins = game_core.player.coins; + profiler.data.player_boost_percent = game_core.player.boost_percent; + profiler.data.player_breath_percent = game_core.player.breath_percent; // Send telemetry data profiler.update(); diff --git a/src/player.rs b/src/player.rs index 3fa3a58..a455e9b 100644 --- a/src/player.rs +++ b/src/player.rs @@ -8,7 +8,8 @@ pub struct Player { pub direction: Vector2, pub size: Vector2, pub coins: u32, - pub boost_percent: f32 + pub boost_percent: f32, + pub breath_percent: f32 } impl Player { @@ -19,6 +20,7 @@ impl Player { x: 11.0 * 4.0, y: 21.0 * 4.0 }, + breath_percent: 1.0, ..Default::default() } diff --git a/src/world.rs b/src/world.rs index 4600c51..309fd41 100644 --- a/src/world.rs +++ b/src/world.rs @@ -1,11 +1,14 @@ use std::{fs::File, io::BufReader}; +use raylib::math::Vector2; use serde::{Deserialize, Serialize}; use std::io::Read; use failure::Error; #[derive(Debug, Serialize, Deserialize, Clone)] -pub struct World {} +pub struct World { + pub end_position: Vector2 +} impl World { pub fn load_from_json(file: String) -> Result { From 1c34ba905863b7c6aa39e106fc937f10498f27d2 Mon Sep 17 00:00:00 2001 From: rsninja722 Date: Sat, 24 Apr 2021 00:06:37 -0400 Subject: [PATCH 08/27] turns smoother --- src/logic/ingame/mod.rs | 37 ++++++++++++++++++++++++++++++++++++- 1 file changed, 36 insertions(+), 1 deletion(-) diff --git a/src/logic/ingame/mod.rs b/src/logic/ingame/mod.rs index b876de3..3cc48d6 100644 --- a/src/logic/ingame/mod.rs +++ b/src/logic/ingame/mod.rs @@ -10,6 +10,7 @@ use super::screen::Screen; const NORMAL_PLAYER_SPEED: i32 = 4; const BOOST_PLAYER_SPEED: i32 = NORMAL_PLAYER_SPEED * 2; const CAMERA_FOLLOW_SPEED: f32 = 0.7; +const TURN_SPEED: f32 = 0.05; pub enum InGameState { BUYING, @@ -41,7 +42,41 @@ impl InGameScreen { let raw_movement_direction = mouse_world_pose - game_core.player.position; let mut normalized_movement_direction = raw_movement_direction; normalized_movement_direction.normalize(); - game_core.player.direction = normalized_movement_direction; + + let tau: f32 = PI as f32 * 2.0; + let mut player_angle: f32 = Vector2::zero().angle_to(game_core.player.direction); + let mut desired_angle: f32 = Vector2::zero().angle_to(normalized_movement_direction); + + if desired_angle < 0.0 { + desired_angle += tau; + } + + if player_angle % tau > desired_angle { + if (player_angle % tau) - desired_angle > PI as f32 { + player_angle += TURN_SPEED; + } else { + player_angle -= TURN_SPEED; + } + } else { + if desired_angle - (player_angle % tau) > PI as f32 { + player_angle -= TURN_SPEED; + } else { + player_angle += TURN_SPEED; + } + } + + if f32::abs(player_angle - desired_angle) < (TURN_SPEED * 1.1) { + player_angle = desired_angle; + } + if player_angle > tau { + player_angle -= tau; + } + if player_angle < 0.0 { + player_angle += tau; + } + + + game_core.player.direction = Vector2::new(f32::cos(player_angle), f32::sin(player_angle)); // In the case the player is in "null", just jump the camera to them if game_core.player.position == Vector2::zero() { From e5a5fb86dce546b4d4abfb04baf79548e1f2bbc1 Mon Sep 17 00:00:00 2001 From: rsninja722 Date: Sat, 24 Apr 2021 00:09:36 -0400 Subject: [PATCH 09/27] forgot comments --- src/logic/ingame/mod.rs | 6 +++++- 1 file changed, 5 insertions(+), 1 deletion(-) diff --git a/src/logic/ingame/mod.rs b/src/logic/ingame/mod.rs index 3cc48d6..df5ade2 100644 --- a/src/logic/ingame/mod.rs +++ b/src/logic/ingame/mod.rs @@ -44,13 +44,16 @@ impl InGameScreen { normalized_movement_direction.normalize(); let tau: f32 = PI as f32 * 2.0; + // get angles as floats let mut player_angle: f32 = Vector2::zero().angle_to(game_core.player.direction); let mut desired_angle: f32 = Vector2::zero().angle_to(normalized_movement_direction); + // make angle positive if desired_angle < 0.0 { desired_angle += tau; } + // turn towards mouse at turn speed if player_angle % tau > desired_angle { if (player_angle % tau) - desired_angle > PI as f32 { player_angle += TURN_SPEED; @@ -65,6 +68,7 @@ impl InGameScreen { } } + // snap to mouse if close enough if f32::abs(player_angle - desired_angle) < (TURN_SPEED * 1.1) { player_angle = desired_angle; } @@ -75,7 +79,7 @@ impl InGameScreen { player_angle += tau; } - + // set angle game_core.player.direction = Vector2::new(f32::cos(player_angle), f32::sin(player_angle)); // In the case the player is in "null", just jump the camera to them From 8d6b08823276b727febcf6e7a8d012d0a9da00ef Mon Sep 17 00:00:00 2001 From: Evan Pratten Date: Sat, 24 Apr 2021 00:15:21 -0400 Subject: [PATCH 10/27] Add progress slider --- assets/worlds/mainworld.json | 4 ++-- src/logic/ingame/mod.rs | 26 ++++++++++++++++++++++++-- 2 files changed, 26 insertions(+), 4 deletions(-) diff --git a/assets/worlds/mainworld.json b/assets/worlds/mainworld.json index 1b50365..f34b38e 100644 --- a/assets/worlds/mainworld.json +++ b/assets/worlds/mainworld.json @@ -1,6 +1,6 @@ { "end_position": { - "x": 100.0, - "y": 100.0 + "x": 10000.0, + "y": 10000.0 } } \ No newline at end of file diff --git a/src/logic/ingame/mod.rs b/src/logic/ingame/mod.rs index 62f69e0..218264a 100644 --- a/src/logic/ingame/mod.rs +++ b/src/logic/ingame/mod.rs @@ -110,7 +110,7 @@ impl InGameScreen { context_2d.draw_circle(0, 0, 10.0, Color::BLACK); } - fn render_hud(&mut self, draw_handle: &mut RaylibDrawHandle, game_core: &mut GameCore) { + fn render_hud(&mut self, draw_handle: &mut RaylibDrawHandle, game_core: &mut GameCore, window_center: Vector2) { // Get the relevant data let breath = game_core.player.breath_percent; let dist_from_player_to_end = game_core @@ -120,7 +120,26 @@ impl InGameScreen { let dist_from_start_to_end = Vector2::zero().distance_to(game_core.world.end_position); let progress = (dist_from_start_to_end - dist_from_player_to_end) / dist_from_start_to_end; - + // Render the base of the progress bar + let progress_bar_rect = Rectangle { + x: 20.0, + y: (window_center.y * 2.0) - 20.0 - 40.0, + width: (window_center.x * 2.0) - 40.0, + height: 40.0 + }; + draw_handle.draw_rectangle_rec(progress_bar_rect, Color::BLUE); + draw_handle.draw_rectangle_lines_ex(progress_bar_rect, 6, Color::WHITE); + + // Render the slider of the progress bar + let progress_bar_slider = Rectangle { + x: (((window_center.x * 2.0) - 40.0) * progress.abs().clamp(0.0, 1.0)) + 10.0, + y: (window_center.y * 2.0) - 20.0 - 50.0, + width: 40.0, + height: 60.0 + }; + draw_handle.draw_rectangle_rec(progress_bar_slider, Color::BLUE); + //TODO: This causes a render bug + draw_handle.draw_rectangle_lines_ex(progress_bar_slider, 6, Color::WHITE); } } @@ -163,6 +182,9 @@ impl Screen for InGameScreen { self.render_player(&mut context_2d, game_core); } + // Render the hud + self.render_hud(draw_handle, game_core, window_center); + return None; } } From fe1df618c0ae725689e4c47d0d6f450a51b5f6ab Mon Sep 17 00:00:00 2001 From: Evan Pratten Date: Sat, 24 Apr 2021 08:40:36 -0400 Subject: [PATCH 11/27] coloriing --- src/logic/ingame/mod.rs | 5 ++++- src/pallette.rs | 0 2 files changed, 4 insertions(+), 1 deletion(-) create mode 100644 src/pallette.rs diff --git a/src/logic/ingame/mod.rs b/src/logic/ingame/mod.rs index 218264a..4d1961e 100644 --- a/src/logic/ingame/mod.rs +++ b/src/logic/ingame/mod.rs @@ -141,6 +141,9 @@ impl InGameScreen { //TODO: This causes a render bug draw_handle.draw_rectangle_lines_ex(progress_bar_slider, 6, Color::WHITE); + // TODO: Breath bar + // TODO: Boost bar + } } @@ -153,7 +156,7 @@ impl Screen for InGameScreen { game_core: &mut GameCore, ) -> Option { // Clear frame - draw_handle.clear_background(Color::WHITE); + draw_handle.clear_background(Color::RAYWHITE); // Handle the pause menu being opened if draw_handle.is_key_pressed(KeyboardKey::KEY_ESCAPE) { diff --git a/src/pallette.rs b/src/pallette.rs new file mode 100644 index 0000000..e69de29 From bfef2269e83b8761ff8408a04d82e2c9919acd81 Mon Sep 17 00:00:00 2001 From: Evan Pratten Date: Sat, 24 Apr 2021 08:46:10 -0400 Subject: [PATCH 12/27] Some quick reorg of ingame logic --- src/logic/ingame/hud.rs | 42 +++++++++++ src/logic/ingame/ingame.rs | 12 ---- src/logic/ingame/mod.rs | 123 ++------------------------------ src/logic/ingame/playerlogic.rs | 74 +++++++++++++++++++ 4 files changed, 123 insertions(+), 128 deletions(-) create mode 100644 src/logic/ingame/hud.rs delete mode 100644 src/logic/ingame/ingame.rs create mode 100644 src/logic/ingame/playerlogic.rs diff --git a/src/logic/ingame/hud.rs b/src/logic/ingame/hud.rs new file mode 100644 index 0000000..e067cba --- /dev/null +++ b/src/logic/ingame/hud.rs @@ -0,0 +1,42 @@ +use raylib::prelude::*; + +use crate::gamecore::GameCore; + +pub fn render_hud( + draw_handle: &mut RaylibDrawHandle, + game_core: &mut GameCore, + window_center: Vector2, +) { + // Get the relevant data + let breath = game_core.player.breath_percent; + let dist_from_player_to_end = game_core + .player + .position + .distance_to(game_core.world.end_position); + let dist_from_start_to_end = Vector2::zero().distance_to(game_core.world.end_position); + let progress = (dist_from_start_to_end - dist_from_player_to_end) / dist_from_start_to_end; + + // Render the base of the progress bar + let progress_bar_rect = Rectangle { + x: 20.0, + y: (window_center.y * 2.0) - 20.0 - 40.0, + width: (window_center.x * 2.0) - 40.0, + height: 40.0, + }; + draw_handle.draw_rectangle_rec(progress_bar_rect, Color::BLUE); + draw_handle.draw_rectangle_lines_ex(progress_bar_rect, 6, Color::WHITE); + + // Render the slider of the progress bar + let progress_bar_slider = Rectangle { + x: (((window_center.x * 2.0) - 40.0) * progress.abs().clamp(0.0, 1.0)) + 10.0, + y: (window_center.y * 2.0) - 20.0 - 50.0, + width: 40.0, + height: 60.0, + }; + draw_handle.draw_rectangle_rec(progress_bar_slider, Color::BLUE); + //TODO: This causes a render bug + draw_handle.draw_rectangle_lines_ex(progress_bar_slider, 6, Color::WHITE); + + // TODO: Breath bar + // TODO: Boost bar +} diff --git a/src/logic/ingame/ingame.rs b/src/logic/ingame/ingame.rs deleted file mode 100644 index 40e99da..0000000 --- a/src/logic/ingame/ingame.rs +++ /dev/null @@ -1,12 +0,0 @@ - - - -pub enum ingame_states{ - BUYING, - SWIMMING, - DEAD, -} - - - - diff --git a/src/logic/ingame/mod.rs b/src/logic/ingame/mod.rs index 4d1961e..3934e09 100644 --- a/src/logic/ingame/mod.rs +++ b/src/logic/ingame/mod.rs @@ -1,3 +1,6 @@ +mod playerlogic; +mod hud; + use raylib::prelude::*; use crate::{ @@ -7,10 +10,6 @@ use crate::{ use super::screen::Screen; -const NORMAL_PLAYER_SPEED: i32 = 4; -const BOOST_PLAYER_SPEED: i32 = NORMAL_PLAYER_SPEED * 2; -const CAMERA_FOLLOW_SPEED: f32 = 0.7; - pub enum InGameState { BUYING, SWIMMING, @@ -28,80 +27,6 @@ impl InGameScreen { } } - fn update_player_movement( - &mut self, - draw_handle: &mut RaylibDrawHandle, - game_core: &mut GameCore, - window_center: Vector2, - ) { - // Handle player movement - let mouse_pose = draw_handle.get_mouse_position(); - let mouse_world_pose = - draw_handle.get_screen_to_world2D(mouse_pose, game_core.master_camera); - let raw_movement_direction = mouse_world_pose - game_core.player.position; - let mut normalized_movement_direction = raw_movement_direction; - normalized_movement_direction.normalize(); - game_core.player.direction = normalized_movement_direction; - - // In the case the player is in "null", just jump the camera to them - if game_core.player.position == Vector2::zero() { - game_core.master_camera.target = game_core.player.position - (window_center / 2.0); - } - - // Handle action buttons - let user_request_boost = draw_handle.is_mouse_button_down(MouseButton::MOUSE_LEFT_BUTTON); - let user_request_action = - draw_handle.is_mouse_button_pressed(MouseButton::MOUSE_RIGHT_BUTTON); - - // Move the player in their direction - let speed_multiplier = match user_request_boost && game_core.player.boost_percent >= 0.0 { - true => BOOST_PLAYER_SPEED as f32, - false => NORMAL_PLAYER_SPEED as f32, - }; - - // Only do this if the mouse is far enough away - let player_real_movement = game_core.player.direction * speed_multiplier; - if raw_movement_direction.distance_to(Vector2::zero()) > game_core.player.size.y / 2.0 { - game_core.player.position += player_real_movement; - } - - // Move the camera to follow the player - let direction_from_cam_to_player = - (game_core.player.position - window_center) - game_core.master_camera.target; - let player_screen_position = - draw_handle.get_world_to_screen2D(game_core.player.position, game_core.master_camera); - - // Camera only moves if you get close to the edge of the screen - if player_screen_position.distance_to(window_center).abs() > (window_center.y - 40.0) { - game_core.master_camera.target += player_real_movement; - } - } - - fn render_player( - &mut self, - context_2d: &mut RaylibMode2D, - game_core: &mut GameCore, - ) { - // Get the player - let player = &game_core.player; - - // Convert the player direction to a rotation - let player_rotation = Vector2::zero().angle_to(player.direction); - - // TODO: tmp rect - context_2d.draw_rectangle_pro( - Rectangle { - x: player.position.x, - y: player.position.y, - width: player.size.x, - height: player.size.y, - }, - player.size / 2.0, - player_rotation.to_degrees() + 90.0, - Color::BLACK, - ); - } - fn render_world( &mut self, context_2d: &mut RaylibMode2D, @@ -110,41 +35,7 @@ impl InGameScreen { context_2d.draw_circle(0, 0, 10.0, Color::BLACK); } - fn render_hud(&mut self, draw_handle: &mut RaylibDrawHandle, game_core: &mut GameCore, window_center: Vector2) { - // Get the relevant data - let breath = game_core.player.breath_percent; - let dist_from_player_to_end = game_core - .player - .position - .distance_to(game_core.world.end_position); - let dist_from_start_to_end = Vector2::zero().distance_to(game_core.world.end_position); - let progress = (dist_from_start_to_end - dist_from_player_to_end) / dist_from_start_to_end; - - // Render the base of the progress bar - let progress_bar_rect = Rectangle { - x: 20.0, - y: (window_center.y * 2.0) - 20.0 - 40.0, - width: (window_center.x * 2.0) - 40.0, - height: 40.0 - }; - draw_handle.draw_rectangle_rec(progress_bar_rect, Color::BLUE); - draw_handle.draw_rectangle_lines_ex(progress_bar_rect, 6, Color::WHITE); - - // Render the slider of the progress bar - let progress_bar_slider = Rectangle { - x: (((window_center.x * 2.0) - 40.0) * progress.abs().clamp(0.0, 1.0)) + 10.0, - y: (window_center.y * 2.0) - 20.0 - 50.0, - width: 40.0, - height: 60.0 - }; - draw_handle.draw_rectangle_rec(progress_bar_slider, Color::BLUE); - //TODO: This causes a render bug - draw_handle.draw_rectangle_lines_ex(progress_bar_slider, 6, Color::WHITE); - - // TODO: Breath bar - // TODO: Boost bar - - } + } impl Screen for InGameScreen { @@ -172,7 +63,7 @@ impl Screen for InGameScreen { }; // Update player movement - self.update_player_movement(draw_handle, game_core, window_center); + playerlogic::update_player_movement(draw_handle, game_core, window_center); // Open a 2D context { @@ -182,11 +73,11 @@ impl Screen for InGameScreen { self.render_world(&mut context_2d, game_core); // Render Player - self.render_player(&mut context_2d, game_core); + playerlogic::render_player(&mut context_2d, game_core); } // Render the hud - self.render_hud(draw_handle, game_core, window_center); + hud::render_hud(draw_handle, game_core, window_center); return None; } diff --git a/src/logic/ingame/playerlogic.rs b/src/logic/ingame/playerlogic.rs new file mode 100644 index 0000000..76958e4 --- /dev/null +++ b/src/logic/ingame/playerlogic.rs @@ -0,0 +1,74 @@ +use raylib::prelude::*; + +use crate::gamecore::GameCore; + +const NORMAL_PLAYER_SPEED: i32 = 4; +const BOOST_PLAYER_SPEED: i32 = NORMAL_PLAYER_SPEED * 2; +const CAMERA_FOLLOW_SPEED: f32 = 0.7; + +pub fn update_player_movement( + draw_handle: &mut RaylibDrawHandle, + game_core: &mut GameCore, + window_center: Vector2, +) { + // Handle player movement + let mouse_pose = draw_handle.get_mouse_position(); + let mouse_world_pose = draw_handle.get_screen_to_world2D(mouse_pose, game_core.master_camera); + let raw_movement_direction = mouse_world_pose - game_core.player.position; + let mut normalized_movement_direction = raw_movement_direction; + normalized_movement_direction.normalize(); + game_core.player.direction = normalized_movement_direction; + + // In the case the player is in "null", just jump the camera to them + if game_core.player.position == Vector2::zero() { + game_core.master_camera.target = game_core.player.position - (window_center / 2.0); + } + + // Handle action buttons + let user_request_boost = draw_handle.is_mouse_button_down(MouseButton::MOUSE_LEFT_BUTTON); + let user_request_action = draw_handle.is_mouse_button_pressed(MouseButton::MOUSE_RIGHT_BUTTON); + + // Move the player in their direction + let speed_multiplier = match user_request_boost && game_core.player.boost_percent >= 0.0 { + true => BOOST_PLAYER_SPEED as f32, + false => NORMAL_PLAYER_SPEED as f32, + }; + + // Only do this if the mouse is far enough away + let player_real_movement = game_core.player.direction * speed_multiplier; + if raw_movement_direction.distance_to(Vector2::zero()) > game_core.player.size.y / 2.0 { + game_core.player.position += player_real_movement; + } + + // Move the camera to follow the player + let direction_from_cam_to_player = + (game_core.player.position - window_center) - game_core.master_camera.target; + let player_screen_position = + draw_handle.get_world_to_screen2D(game_core.player.position, game_core.master_camera); + + // Camera only moves if you get close to the edge of the screen + if player_screen_position.distance_to(window_center).abs() > (window_center.y - 40.0) { + game_core.master_camera.target += player_real_movement; + } +} + +pub fn render_player(context_2d: &mut RaylibMode2D, game_core: &mut GameCore) { + // Get the player + let player = &game_core.player; + + // Convert the player direction to a rotation + let player_rotation = Vector2::zero().angle_to(player.direction); + + // TODO: tmp rect + context_2d.draw_rectangle_pro( + Rectangle { + x: player.position.x, + y: player.position.y, + width: player.size.x, + height: player.size.y, + }, + player.size / 2.0, + player_rotation.to_degrees() + 90.0, + Color::BLACK, + ); +} From fa9bc120e355095c47f1a14b93fd215a4fca9b41 Mon Sep 17 00:00:00 2001 From: Evan Pratten Date: Sat, 24 Apr 2021 09:10:14 -0400 Subject: [PATCH 13/27] Add breath and boost hud --- src/logic/ingame/hud.rs | 36 ++++++++++++++++----------------- src/logic/ingame/mod.rs | 2 +- src/logic/ingame/playerlogic.rs | 24 +++++++++++++++++++++- src/main.rs | 1 + src/pallette.rs | 15 ++++++++++++++ src/player.rs | 2 +- 6 files changed, 59 insertions(+), 21 deletions(-) diff --git a/src/logic/ingame/hud.rs b/src/logic/ingame/hud.rs index e067cba..3f576d0 100644 --- a/src/logic/ingame/hud.rs +++ b/src/logic/ingame/hud.rs @@ -17,25 +17,25 @@ pub fn render_hud( let progress = (dist_from_start_to_end - dist_from_player_to_end) / dist_from_start_to_end; // Render the base of the progress bar - let progress_bar_rect = Rectangle { - x: 20.0, - y: (window_center.y * 2.0) - 20.0 - 40.0, - width: (window_center.x * 2.0) - 40.0, - height: 40.0, - }; - draw_handle.draw_rectangle_rec(progress_bar_rect, Color::BLUE); - draw_handle.draw_rectangle_lines_ex(progress_bar_rect, 6, Color::WHITE); + // let progress_bar_rect = Rectangle { + // x: 20.0, + // y: (window_center.y * 2.0) - 20.0 - 40.0, + // width: (window_center.x * 2.0) - 40.0, + // height: 40.0, + // }; + // draw_handle.draw_rectangle_rec(progress_bar_rect, Color::BLUE); + // draw_handle.draw_rectangle_lines_ex(progress_bar_rect, 6, Color::WHITE); - // Render the slider of the progress bar - let progress_bar_slider = Rectangle { - x: (((window_center.x * 2.0) - 40.0) * progress.abs().clamp(0.0, 1.0)) + 10.0, - y: (window_center.y * 2.0) - 20.0 - 50.0, - width: 40.0, - height: 60.0, - }; - draw_handle.draw_rectangle_rec(progress_bar_slider, Color::BLUE); - //TODO: This causes a render bug - draw_handle.draw_rectangle_lines_ex(progress_bar_slider, 6, Color::WHITE); + // // Render the slider of the progress bar + // let progress_bar_slider = Rectangle { + // x: (((window_center.x * 2.0) - 40.0) * progress.abs().clamp(0.0, 1.0)) + 10.0, + // y: (window_center.y * 2.0) - 20.0 - 50.0, + // width: 40.0, + // height: 60.0, + // }; + // draw_handle.draw_rectangle_rec(progress_bar_slider, Color::BLUE); + // //TODO: This causes a render bug + // draw_handle.draw_rectangle_lines_ex(progress_bar_slider, 6, Color::WHITE); // TODO: Breath bar // TODO: Boost bar diff --git a/src/logic/ingame/mod.rs b/src/logic/ingame/mod.rs index 3934e09..7ef15cc 100644 --- a/src/logic/ingame/mod.rs +++ b/src/logic/ingame/mod.rs @@ -47,7 +47,7 @@ impl Screen for InGameScreen { game_core: &mut GameCore, ) -> Option { // Clear frame - draw_handle.clear_background(Color::RAYWHITE); + draw_handle.clear_background(Color::BLUE); // Handle the pause menu being opened if draw_handle.is_key_pressed(KeyboardKey::KEY_ESCAPE) { diff --git a/src/logic/ingame/playerlogic.rs b/src/logic/ingame/playerlogic.rs index 76958e4..6dc036e 100644 --- a/src/logic/ingame/playerlogic.rs +++ b/src/logic/ingame/playerlogic.rs @@ -1,6 +1,9 @@ use raylib::prelude::*; -use crate::gamecore::GameCore; +use crate::{ + gamecore::GameCore, + pallette::{TRANSLUCENT_WHITE_128, TRANSLUCENT_WHITE_64}, +}; const NORMAL_PLAYER_SPEED: i32 = 4; const BOOST_PLAYER_SPEED: i32 = NORMAL_PLAYER_SPEED * 2; @@ -59,6 +62,25 @@ pub fn render_player(context_2d: &mut RaylibMode2D, game_core: // Convert the player direction to a rotation let player_rotation = Vector2::zero().angle_to(player.direction); + // Render the player's boost ring + // This functions both as a breath meter, and as a boost meter + let boost_ring_max_radius = player.size.x + 5.0; + context_2d.draw_circle( + player.position.x as i32, + player.position.y as i32, + boost_ring_max_radius * player.boost_percent, + TRANSLUCENT_WHITE_64, + ); + context_2d.draw_ring( + player.position, + boost_ring_max_radius - 2.0, + boost_ring_max_radius + 2.0, + 0, + (360.0 * player.breath_percent) as i32, + 0, + TRANSLUCENT_WHITE_128, + ); + // TODO: tmp rect context_2d.draw_rectangle_pro( Rectangle { diff --git a/src/main.rs b/src/main.rs index 65d79ad..50cc74d 100644 --- a/src/main.rs +++ b/src/main.rs @@ -4,6 +4,7 @@ mod logic; mod resources; mod player; mod world; +mod pallette; use gamecore::{GameCore, GameState}; use lib::{utils::profiler::GameProfiler, wrappers::audio::player::AudioPlayer}; diff --git a/src/pallette.rs b/src/pallette.rs index e69de29..2517a57 100644 --- a/src/pallette.rs +++ b/src/pallette.rs @@ -0,0 +1,15 @@ +use raylib::color::Color; + +pub const TRANSLUCENT_WHITE_128: Color = Color { + r: 255, + g: 255, + b: 255, + a: 128, +}; + +pub const TRANSLUCENT_WHITE_64: Color = Color { + r: 255, + g: 255, + b: 255, + a: 64, +}; \ No newline at end of file diff --git a/src/player.rs b/src/player.rs index a455e9b..4ce8bfe 100644 --- a/src/player.rs +++ b/src/player.rs @@ -20,7 +20,7 @@ impl Player { x: 11.0 * 4.0, y: 21.0 * 4.0 }, - breath_percent: 1.0, + breath_percent: 0.5, ..Default::default() } From 6469a36c5651ebfab60e4d602528817eed0ce10f Mon Sep 17 00:00:00 2001 From: Evan Pratten Date: Sat, 24 Apr 2021 09:18:09 -0400 Subject: [PATCH 14/27] Real boost logic --- src/gamecore.rs | 6 ++++-- src/logic/ingame/playerlogic.rs | 32 +++++++++++++++++++++++--------- src/main.rs | 3 +++ src/pallette.rs | 7 +++++++ 4 files changed, 37 insertions(+), 11 deletions(-) diff --git a/src/gamecore.rs b/src/gamecore.rs index ef7f71a..3eb8f20 100644 --- a/src/gamecore.rs +++ b/src/gamecore.rs @@ -17,7 +17,7 @@ pub enum GameState { MainMenu, PauseMenu, GameQuit, - InGame + InGame, } impl fmt::Display for GameState { @@ -32,6 +32,7 @@ pub struct GameCore { pub state: GameState, pub last_state: GameState, pub last_state_change_time: f64, + pub last_frame_time: f64, pub has_rendered_first_frame: bool, /// Resources @@ -56,6 +57,7 @@ impl GameCore { state: GameState::Loading, last_state: GameState::Loading, last_state_change_time: 0.0, + last_frame_time: 0.0, has_rendered_first_frame: false, resources: GlobalResources::load_all(raylib, thread) .expect("Failed to load game assets. Can not launch!"), @@ -67,7 +69,7 @@ impl GameCore { }, show_simple_debug_info: false, world: world, - player: Player::new() + player: Player::new(), } } diff --git a/src/logic/ingame/playerlogic.rs b/src/logic/ingame/playerlogic.rs index 6dc036e..db01d5c 100644 --- a/src/logic/ingame/playerlogic.rs +++ b/src/logic/ingame/playerlogic.rs @@ -1,19 +1,22 @@ use raylib::prelude::*; -use crate::{ - gamecore::GameCore, - pallette::{TRANSLUCENT_WHITE_128, TRANSLUCENT_WHITE_64}, -}; +use crate::{gamecore::GameCore, pallette::{TRANSLUCENT_WHITE_128, TRANSLUCENT_WHITE_64, TRANSLUCENT_WHITE_96}}; const NORMAL_PLAYER_SPEED: i32 = 4; const BOOST_PLAYER_SPEED: i32 = NORMAL_PLAYER_SPEED * 2; const CAMERA_FOLLOW_SPEED: f32 = 0.7; +const BOOST_DECREASE_PER_SECOND: f32 = 0.75; +const BOOST_REGEN_PER_SECOND: f32 = 0.25; +const BREATH_DECREASE_PER_SECOND: f32 = 0.01; pub fn update_player_movement( draw_handle: &mut RaylibDrawHandle, game_core: &mut GameCore, window_center: Vector2, ) { + // Calculate DT + let dt = draw_handle.get_time() - game_core.last_frame_time; + // Handle player movement let mouse_pose = draw_handle.get_mouse_position(); let mouse_world_pose = draw_handle.get_screen_to_world2D(mouse_pose, game_core.master_camera); @@ -32,10 +35,21 @@ pub fn update_player_movement( let user_request_action = draw_handle.is_mouse_button_pressed(MouseButton::MOUSE_RIGHT_BUTTON); // Move the player in their direction - let speed_multiplier = match user_request_boost && game_core.player.boost_percent >= 0.0 { - true => BOOST_PLAYER_SPEED as f32, - false => NORMAL_PLAYER_SPEED as f32, - }; + let speed_multiplier; + if user_request_boost && game_core.player.boost_percent >= 0.0 { + // Set the speed multiplier + speed_multiplier = BOOST_PLAYER_SPEED as f32; + + // Decrease the boost + game_core.player.boost_percent -= BOOST_DECREASE_PER_SECOND * dt as f32; + } else { + // Set the speed multiplier + speed_multiplier = NORMAL_PLAYER_SPEED as f32; + + // Handle boost regen + game_core.player.boost_percent = + (game_core.player.boost_percent + BOOST_REGEN_PER_SECOND * dt as f32).clamp(0.0, 1.0); + } // Only do this if the mouse is far enough away let player_real_movement = game_core.player.direction * speed_multiplier; @@ -78,7 +92,7 @@ pub fn render_player(context_2d: &mut RaylibMode2D, game_core: 0, (360.0 * player.breath_percent) as i32, 0, - TRANSLUCENT_WHITE_128, + TRANSLUCENT_WHITE_96, ); // TODO: tmp rect diff --git a/src/main.rs b/src/main.rs index 50cc74d..c0ede7f 100644 --- a/src/main.rs +++ b/src/main.rs @@ -149,6 +149,9 @@ fn main() { // Set the first frame flag game_core.has_rendered_first_frame = true; + + // Update the frame time + game_core.last_frame_time = draw_handle.get_time(); } // Cleanup diff --git a/src/pallette.rs b/src/pallette.rs index 2517a57..9d70e98 100644 --- a/src/pallette.rs +++ b/src/pallette.rs @@ -7,6 +7,13 @@ pub const TRANSLUCENT_WHITE_128: Color = Color { a: 128, }; +pub const TRANSLUCENT_WHITE_96: Color = Color { + r: 255, + g: 255, + b: 255, + a: 96, +}; + pub const TRANSLUCENT_WHITE_64: Color = Color { r: 255, g: 255, From 68f90199a7ba35369207437a77efc86d1bb07b90 Mon Sep 17 00:00:00 2001 From: Evan Pratten Date: Sat, 24 Apr 2021 09:23:57 -0400 Subject: [PATCH 15/27] finish player logic --- src/gamecore.rs | 1 + src/logic/gameend.rs | 126 ++++++++++++++++++++++++++++++++ src/logic/ingame/mod.rs | 6 +- src/logic/ingame/playerlogic.rs | 3 + src/logic/mod.rs | 3 +- src/main.rs | 12 ++- src/player.rs | 2 +- 7 files changed, 146 insertions(+), 7 deletions(-) create mode 100644 src/logic/gameend.rs diff --git a/src/gamecore.rs b/src/gamecore.rs index 3eb8f20..fc6b26f 100644 --- a/src/gamecore.rs +++ b/src/gamecore.rs @@ -18,6 +18,7 @@ pub enum GameState { PauseMenu, GameQuit, InGame, + GameEnd } impl fmt::Display for GameState { diff --git a/src/logic/gameend.rs b/src/logic/gameend.rs new file mode 100644 index 0000000..8cee91f --- /dev/null +++ b/src/logic/gameend.rs @@ -0,0 +1,126 @@ +use raylib::prelude::*; + +use crate::{ + gamecore::{GameCore, GameState}, + lib::wrappers::audio::player::AudioPlayer, +}; + +use super::screen::Screen; + +const SCREEN_PANEL_SIZE: Vector2 = Vector2 { x: 300.0, y: 300.0 }; + +pub struct GameEndScreen {} + +impl GameEndScreen { + pub fn new() -> Self { + Self {} + } +} + +impl Screen for GameEndScreen { + fn render( + &mut self, + draw_handle: &mut RaylibDrawHandle, + _thread: &RaylibThread, + audio_system: &mut AudioPlayer, + game_core: &mut GameCore, + ) -> Option { + let mouse_position = draw_handle.get_mouse_position(); + draw_handle.clear_background(Color::GRAY); + // TODO: Maybe we can stick some art here? + + + // Window dimensions + let win_height = draw_handle.get_screen_height(); + let win_width = draw_handle.get_screen_width(); + + // Render the backing to the menu itself + draw_handle.draw_rectangle( + (win_width / 2) - ((SCREEN_PANEL_SIZE.x as i32 + 6) / 2), + (win_height / 2) - ((SCREEN_PANEL_SIZE.y as i32 + 6) / 2), + SCREEN_PANEL_SIZE.x as i32 + 6, + SCREEN_PANEL_SIZE.y as i32 + 6, + Color::BLACK, + ); + draw_handle.draw_rectangle( + (win_width / 2) - (SCREEN_PANEL_SIZE.x as i32 / 2), + (win_height / 2) - (SCREEN_PANEL_SIZE.y as i32 / 2), + SCREEN_PANEL_SIZE.x as i32, + SCREEN_PANEL_SIZE.y as i32, + Color::WHITE, + ); + + // Render heading text + draw_handle.draw_text( + "OUT OF BREATH", + (win_width / 2) - 80, + (win_height / 2) - (SCREEN_PANEL_SIZE.y as i32 / 2) + 10, + 40, + Color::BLACK, + ); + + + // // Close and quit buttons + // let bottom_left_button_dimensions = Rectangle { + // x: (win_width as f32 / 2.0) - (SCREEN_PANEL_SIZE.x / 2.0) + 5.0, + // y: (win_height as f32 / 2.0) + (SCREEN_PANEL_SIZE.y / 2.0) - 50.0, + // width: (SCREEN_PANEL_SIZE.x / 2.0) - 15.0, + // height: 40.0, + // }; + // let bottom_right_button_dimensions = Rectangle { + // x: (win_width as f32 / 2.0) + 5.0, + // y: bottom_left_button_dimensions.y, + // width: bottom_left_button_dimensions.width, + // height: bottom_left_button_dimensions.height, + // }; + + // // Check if the mouse is over either button + // let mouse_over_bottom_left_button = + // bottom_left_button_dimensions.check_collision_point_rec(mouse_position); + // let mouse_over_bottom_right_button = + // bottom_right_button_dimensions.check_collision_point_rec(mouse_position); + + // // Render buttons + // draw_handle.draw_rectangle_lines_ex( + // bottom_left_button_dimensions, + // 3, + // match mouse_over_bottom_left_button { + // true => Color::GRAY, + // false => Color::BLACK, + // }, + // ); + // draw_handle.draw_text( + // "Quit", + // bottom_left_button_dimensions.x as i32 + 15, + // bottom_left_button_dimensions.y as i32 + 5, + // 30, + // Color::BLACK, + // ); + // draw_handle.draw_rectangle_lines_ex( + // bottom_right_button_dimensions, + // 3, + // match mouse_over_bottom_right_button { + // true => Color::GRAY, + // false => Color::BLACK, + // }, + // ); + // draw_handle.draw_text( + // "Close", + // bottom_right_button_dimensions.x as i32 + 15, + // bottom_right_button_dimensions.y as i32 + 5, + // 30, + // Color::BLACK, + // ); + + // // Handle click actions on the buttons + // if draw_handle.is_mouse_button_pressed(MouseButton::MOUSE_LEFT_BUTTON) { + // if mouse_over_bottom_left_button { + // return Some(GameState::GameQuit); + // } else if mouse_over_bottom_right_button { + // return Some(game_core.last_state); + // } + // } + + return None; + } +} diff --git a/src/logic/ingame/mod.rs b/src/logic/ingame/mod.rs index 7ef15cc..7297289 100644 --- a/src/logic/ingame/mod.rs +++ b/src/logic/ingame/mod.rs @@ -13,7 +13,6 @@ use super::screen::Screen; pub enum InGameState { BUYING, SWIMMING, - DEAD, } pub struct InGameScreen { @@ -79,6 +78,11 @@ impl Screen for InGameScreen { // Render the hud hud::render_hud(draw_handle, game_core, window_center); + // Handle player out of breath + if game_core.player.breath_percent == 0.0 { + return Some(GameState::GameEnd); + } + return None; } } diff --git a/src/logic/ingame/playerlogic.rs b/src/logic/ingame/playerlogic.rs index db01d5c..72a694c 100644 --- a/src/logic/ingame/playerlogic.rs +++ b/src/logic/ingame/playerlogic.rs @@ -51,6 +51,9 @@ pub fn update_player_movement( (game_core.player.boost_percent + BOOST_REGEN_PER_SECOND * dt as f32).clamp(0.0, 1.0); } + // Update the player's breath + game_core.player.breath_percent = (game_core.player.breath_percent - BREATH_DECREASE_PER_SECOND * dt as f32).clamp(0.0, 1.0); + // Only do this if the mouse is far enough away let player_real_movement = game_core.player.direction * speed_multiplier; if raw_movement_direction.distance_to(Vector2::zero()) > game_core.player.size.y / 2.0 { diff --git a/src/logic/mod.rs b/src/logic/mod.rs index 6d476b4..362eee2 100644 --- a/src/logic/mod.rs +++ b/src/logic/mod.rs @@ -2,4 +2,5 @@ pub mod screen; pub mod loadingscreen; pub mod mainmenu; pub mod pausemenu; -pub mod ingame; \ No newline at end of file +pub mod ingame; +pub mod gameend; \ No newline at end of file diff --git a/src/main.rs b/src/main.rs index c0ede7f..85ff39b 100644 --- a/src/main.rs +++ b/src/main.rs @@ -9,10 +9,7 @@ mod pallette; use gamecore::{GameCore, GameState}; use lib::{utils::profiler::GameProfiler, wrappers::audio::player::AudioPlayer}; use log::info; -use logic::{ - ingame::InGameScreen, loadingscreen::LoadingScreen, mainmenu::MainMenuScreen, - pausemenu::PauseMenuScreen, screen::Screen, -}; +use logic::{gameend::GameEndScreen, ingame::InGameScreen, loadingscreen::LoadingScreen, mainmenu::MainMenuScreen, pausemenu::PauseMenuScreen, screen::Screen}; use raylib::prelude::*; use world::World; @@ -59,6 +56,7 @@ fn main() { let mut main_menu_screen = MainMenuScreen::new(); let mut pause_menu_screen = PauseMenuScreen::new(); let mut ingame_screen = InGameScreen::new(); + let mut game_end_screen = GameEndScreen::new(); // Main rendering loop while !raylib.window_should_close() { @@ -91,6 +89,12 @@ fn main() { &mut audio_system, &mut game_core, ), + GameState::GameEnd => game_end_screen.render( + &mut draw_handle, + &raylib_thread, + &mut audio_system, + &mut game_core, + ), }; // If needed, update the global state diff --git a/src/player.rs b/src/player.rs index 4ce8bfe..a455e9b 100644 --- a/src/player.rs +++ b/src/player.rs @@ -20,7 +20,7 @@ impl Player { x: 11.0 * 4.0, y: 21.0 * 4.0 }, - breath_percent: 0.5, + breath_percent: 1.0, ..Default::default() } From 7836d5e8892f00b0fd4669b50ccb3d6bab3ed76d Mon Sep 17 00:00:00 2001 From: Evan Pratten Date: Sat, 24 Apr 2021 09:43:38 -0400 Subject: [PATCH 16/27] improved boost logic and hud --- src/logic/ingame/hud.rs | 107 +++++++++++++++++++++++++------- src/logic/ingame/playerlogic.rs | 15 +++-- 2 files changed, 94 insertions(+), 28 deletions(-) diff --git a/src/logic/ingame/hud.rs b/src/logic/ingame/hud.rs index 3f576d0..6a70ff9 100644 --- a/src/logic/ingame/hud.rs +++ b/src/logic/ingame/hud.rs @@ -1,6 +1,6 @@ use raylib::prelude::*; -use crate::gamecore::GameCore; +use crate::{gamecore::GameCore, pallette::TRANSLUCENT_WHITE_96}; pub fn render_hud( draw_handle: &mut RaylibDrawHandle, @@ -8,35 +8,94 @@ pub fn render_hud( window_center: Vector2, ) { // Get the relevant data - let breath = game_core.player.breath_percent; let dist_from_player_to_end = game_core .player .position .distance_to(game_core.world.end_position); let dist_from_start_to_end = Vector2::zero().distance_to(game_core.world.end_position); - let progress = (dist_from_start_to_end - dist_from_player_to_end) / dist_from_start_to_end; + let progress = ((dist_from_start_to_end - dist_from_player_to_end) / dist_from_start_to_end) + .clamp(0.0, 1.0); - // Render the base of the progress bar - // let progress_bar_rect = Rectangle { - // x: 20.0, - // y: (window_center.y * 2.0) - 20.0 - 40.0, - // width: (window_center.x * 2.0) - 40.0, - // height: 40.0, - // }; - // draw_handle.draw_rectangle_rec(progress_bar_rect, Color::BLUE); - // draw_handle.draw_rectangle_lines_ex(progress_bar_rect, 6, Color::WHITE); + // Determine the progress slider position + let slider_bound_height = 20.0; + let progress_slider_position = Vector2 { + x: window_center.x * 2.0, + y: (((window_center.y * 2.0) - (slider_bound_height * 2.0)) * progress) + + slider_bound_height, + }; - // // Render the slider of the progress bar - // let progress_bar_slider = Rectangle { - // x: (((window_center.x * 2.0) - 40.0) * progress.abs().clamp(0.0, 1.0)) + 10.0, - // y: (window_center.y * 2.0) - 20.0 - 50.0, - // width: 40.0, - // height: 60.0, - // }; - // draw_handle.draw_rectangle_rec(progress_bar_slider, Color::BLUE); - // //TODO: This causes a render bug - // draw_handle.draw_rectangle_lines_ex(progress_bar_slider, 6, Color::WHITE); + // Render the base of the slider + draw_handle.draw_rectangle( + (progress_slider_position.x - slider_bound_height) as i32, + (progress_slider_position.y - slider_bound_height / 2.0) as i32, + slider_bound_height as i32, + slider_bound_height as i32, + TRANSLUCENT_WHITE_96, + ); + draw_handle.draw_triangle( + Vector2 { + x: (progress_slider_position.x - slider_bound_height), + y: (progress_slider_position.y - slider_bound_height / 2.0), + }, + Vector2 { + x: (progress_slider_position.x - slider_bound_height - (slider_bound_height / 2.0)), + y: progress_slider_position.y, + }, + Vector2 { + x: (progress_slider_position.x - slider_bound_height), + y: (progress_slider_position.y + slider_bound_height / 2.0), + }, + TRANSLUCENT_WHITE_96, + ); + + // Render the outline of the slider + draw_handle.draw_line_ex( + Vector2 { + x: (progress_slider_position.x - slider_bound_height), + y: (progress_slider_position.y - slider_bound_height / 2.0), + }, + Vector2 { + x: progress_slider_position.x, + y: (progress_slider_position.y - slider_bound_height / 2.0), + }, + 3.0, + Color::BLACK, + ); + draw_handle.draw_line_ex( + Vector2 { + x: (progress_slider_position.x - slider_bound_height), + y: (progress_slider_position.y + slider_bound_height / 2.0), + }, + Vector2 { + x: progress_slider_position.x, + y: (progress_slider_position.y + slider_bound_height / 2.0), + }, + 3.0, + Color::BLACK, + ); + draw_handle.draw_line_ex( + Vector2 { + x: (progress_slider_position.x - slider_bound_height), + y: (progress_slider_position.y - slider_bound_height / 2.0), + }, + Vector2 { + x: (progress_slider_position.x - slider_bound_height - (slider_bound_height / 2.0)), + y: progress_slider_position.y, + }, + 3.0, + Color::BLACK, + ); + draw_handle.draw_line_ex( + Vector2 { + x: (progress_slider_position.x - slider_bound_height), + y: (progress_slider_position.y + slider_bound_height / 2.0), + }, + Vector2 { + x: (progress_slider_position.x - slider_bound_height - (slider_bound_height / 2.0)), + y: progress_slider_position.y, + }, + 3.0, + Color::BLACK, + ); - // TODO: Breath bar - // TODO: Boost bar } diff --git a/src/logic/ingame/playerlogic.rs b/src/logic/ingame/playerlogic.rs index 72a694c..01d5f0d 100644 --- a/src/logic/ingame/playerlogic.rs +++ b/src/logic/ingame/playerlogic.rs @@ -1,6 +1,9 @@ use raylib::prelude::*; -use crate::{gamecore::GameCore, pallette::{TRANSLUCENT_WHITE_128, TRANSLUCENT_WHITE_64, TRANSLUCENT_WHITE_96}}; +use crate::{ + gamecore::GameCore, + pallette::{TRANSLUCENT_WHITE_128, TRANSLUCENT_WHITE_64, TRANSLUCENT_WHITE_96}, +}; const NORMAL_PLAYER_SPEED: i32 = 4; const BOOST_PLAYER_SPEED: i32 = NORMAL_PLAYER_SPEED * 2; @@ -47,12 +50,16 @@ pub fn update_player_movement( speed_multiplier = NORMAL_PLAYER_SPEED as f32; // Handle boost regen - game_core.player.boost_percent = - (game_core.player.boost_percent + BOOST_REGEN_PER_SECOND * dt as f32).clamp(0.0, 1.0); + if !user_request_boost { + game_core.player.boost_percent = (game_core.player.boost_percent + + BOOST_REGEN_PER_SECOND * dt as f32) + .clamp(0.0, 1.0); + } } // Update the player's breath - game_core.player.breath_percent = (game_core.player.breath_percent - BREATH_DECREASE_PER_SECOND * dt as f32).clamp(0.0, 1.0); + game_core.player.breath_percent = + (game_core.player.breath_percent - BREATH_DECREASE_PER_SECOND * dt as f32).clamp(0.0, 1.0); // Only do this if the mouse is far enough away let player_real_movement = game_core.player.direction * speed_multiplier; From b4e74a5694f2031a13cd5e1401c9af37acc6a102 Mon Sep 17 00:00:00 2001 From: Evan Pratten Date: Sat, 24 Apr 2021 09:49:39 -0400 Subject: [PATCH 17/27] Move James's changes to new file Co-authored-by: rsninja722 --- src/logic/ingame/mod.rs | 117 -------------------------------- src/logic/ingame/playerlogic.rs | 41 ++++++++++- 2 files changed, 40 insertions(+), 118 deletions(-) diff --git a/src/logic/ingame/mod.rs b/src/logic/ingame/mod.rs index e014d08..7297289 100644 --- a/src/logic/ingame/mod.rs +++ b/src/logic/ingame/mod.rs @@ -10,11 +10,6 @@ use crate::{ use super::screen::Screen; -const NORMAL_PLAYER_SPEED: i32 = 4; -const BOOST_PLAYER_SPEED: i32 = NORMAL_PLAYER_SPEED * 2; -const CAMERA_FOLLOW_SPEED: f32 = 0.7; -const TURN_SPEED: f32 = 0.05; - pub enum InGameState { BUYING, SWIMMING, @@ -31,118 +26,6 @@ impl InGameScreen { } } - fn update_player_movement( - &mut self, - draw_handle: &mut RaylibDrawHandle, - game_core: &mut GameCore, - window_center: Vector2, - ) { - - // Handle player movement - let mouse_pose = draw_handle.get_mouse_position(); - let mouse_world_pose = draw_handle.get_screen_to_world2D(mouse_pose, game_core.master_camera); - let raw_movement_direction = mouse_world_pose - game_core.player.position; - let mut normalized_movement_direction = raw_movement_direction; - normalized_movement_direction.normalize(); - - let tau: f32 = PI as f32 * 2.0; - // get angles as floats - let mut player_angle: f32 = Vector2::zero().angle_to(game_core.player.direction); - let mut desired_angle: f32 = Vector2::zero().angle_to(normalized_movement_direction); - - // make angle positive - if desired_angle < 0.0 { - desired_angle += tau; - } - - // turn towards mouse at turn speed - if player_angle % tau > desired_angle { - if (player_angle % tau) - desired_angle > PI as f32 { - player_angle += TURN_SPEED; - } else { - player_angle -= TURN_SPEED; - } - } else { - if desired_angle - (player_angle % tau) > PI as f32 { - player_angle -= TURN_SPEED; - } else { - player_angle += TURN_SPEED; - } - } - - // snap to mouse if close enough - if f32::abs(player_angle - desired_angle) < (TURN_SPEED * 1.1) { - player_angle = desired_angle; - } - if player_angle > tau { - player_angle -= tau; - } - if player_angle < 0.0 { - player_angle += tau; - } - - // set angle - game_core.player.direction = Vector2::new(f32::cos(player_angle), f32::sin(player_angle)); - - // In the case the player is in "null", just jump the camera to them - if game_core.player.position == Vector2::zero() { - game_core.master_camera.target = game_core.player.position - (window_center / 2.0); - } - - // Handle action buttons - let user_request_boost = draw_handle.is_mouse_button_down(MouseButton::MOUSE_LEFT_BUTTON); - let user_request_action = - draw_handle.is_mouse_button_pressed(MouseButton::MOUSE_RIGHT_BUTTON); - - // Move the player in their direction - let speed_multiplier = match user_request_boost && game_core.player.boost_percent >= 0.0 { - true => BOOST_PLAYER_SPEED as f32, - false => NORMAL_PLAYER_SPEED as f32, - }; - - // Only do this if the mouse is far enough away - let player_real_movement = game_core.player.direction * speed_multiplier; - if raw_movement_direction.distance_to(Vector2::zero()) > game_core.player.size.y / 2.0 { - game_core.player.position += player_real_movement; - } - - // Move the camera to follow the player - let direction_from_cam_to_player = - (game_core.player.position - window_center) - game_core.master_camera.target; - let player_screen_position = draw_handle.get_world_to_screen2D(game_core.player.position, game_core.master_camera); - - // Camera only moves if you get close to the edge of the screen - if player_screen_position.distance_to(window_center).abs() > (window_center.y - 40.0) { - game_core.master_camera.target += player_real_movement; - } - - } - - fn render_player( - &mut self, - context_2d: &mut RaylibMode2D, - game_core: &mut GameCore, - ) { - // Get the player - let player = &game_core.player; - - // Convert the player direction to a rotation - let player_rotation = Vector2::zero().angle_to(player.direction); - - // TODO: tmp rect - context_2d.draw_rectangle_pro( - Rectangle { - x: player.position.x, - y: player.position.y, - width: player.size.x, - height: player.size.y, - }, - player.size / 2.0, - player_rotation.to_degrees() + 90.0, - Color::BLACK, - ); - } - fn render_world( &mut self, context_2d: &mut RaylibMode2D, diff --git a/src/logic/ingame/playerlogic.rs b/src/logic/ingame/playerlogic.rs index 01d5f0d..37c0a16 100644 --- a/src/logic/ingame/playerlogic.rs +++ b/src/logic/ingame/playerlogic.rs @@ -8,6 +8,7 @@ use crate::{ const NORMAL_PLAYER_SPEED: i32 = 4; const BOOST_PLAYER_SPEED: i32 = NORMAL_PLAYER_SPEED * 2; const CAMERA_FOLLOW_SPEED: f32 = 0.7; +const TURN_SPEED: f32 = 0.05; const BOOST_DECREASE_PER_SECOND: f32 = 0.75; const BOOST_REGEN_PER_SECOND: f32 = 0.25; const BREATH_DECREASE_PER_SECOND: f32 = 0.01; @@ -26,7 +27,45 @@ pub fn update_player_movement( let raw_movement_direction = mouse_world_pose - game_core.player.position; let mut normalized_movement_direction = raw_movement_direction; normalized_movement_direction.normalize(); - game_core.player.direction = normalized_movement_direction; + + let tau: f32 = PI as f32 * 2.0; + // get angles as floats + let mut player_angle: f32 = Vector2::zero().angle_to(game_core.player.direction); + let mut desired_angle: f32 = Vector2::zero().angle_to(normalized_movement_direction); + + // make angle positive + if desired_angle < 0.0 { + desired_angle += tau; + } + + // turn towards mouse at turn speed + if player_angle % tau > desired_angle { + if (player_angle % tau) - desired_angle > PI as f32 { + player_angle += TURN_SPEED; + } else { + player_angle -= TURN_SPEED; + } + } else { + if desired_angle - (player_angle % tau) > PI as f32 { + player_angle -= TURN_SPEED; + } else { + player_angle += TURN_SPEED; + } + } + + // snap to mouse if close enough + if f32::abs(player_angle - desired_angle) < (TURN_SPEED * 1.1) { + player_angle = desired_angle; + } + if player_angle > tau { + player_angle -= tau; + } + if player_angle < 0.0 { + player_angle += tau; + } + + // set angle + game_core.player.direction = Vector2::new(f32::cos(player_angle), f32::sin(player_angle)); // In the case the player is in "null", just jump the camera to them if game_core.player.position == Vector2::zero() { From dd843e7a3f38689a9a1acbd2b33bcbfb0aa9690f Mon Sep 17 00:00:00 2001 From: Evan Pratten Date: Sat, 24 Apr 2021 10:14:45 -0400 Subject: [PATCH 18/27] Game progress --- Cargo.toml | 1 + assets/.gitignore | 1 + src/gamecore.rs | 62 +++++++++++++++++++++++++++++++++++++++++--- src/items.rs | 10 +++++++ src/logic/gameend.rs | 2 ++ src/main.rs | 8 ++++-- 6 files changed, 78 insertions(+), 6 deletions(-) create mode 100644 assets/.gitignore create mode 100644 src/items.rs diff --git a/Cargo.toml b/Cargo.toml index 36bf495..ed60464 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -17,3 +17,4 @@ parry2d = "0.4.0" log = "0.4.14" env_logger = "0.8.3" nalgebra = "0.26.1" +tiled = "0.9.4" \ No newline at end of file diff --git a/assets/.gitignore b/assets/.gitignore new file mode 100644 index 0000000..97ec57d --- /dev/null +++ b/assets/.gitignore @@ -0,0 +1 @@ +savestate.json \ No newline at end of file diff --git a/src/gamecore.rs b/src/gamecore.rs index fc6b26f..2cd14c7 100644 --- a/src/gamecore.rs +++ b/src/gamecore.rs @@ -1,14 +1,16 @@ //! This file contains the global state of the game. Data here is passed around to all handler functions. -use std::fmt; +use std::{fmt, fs::File, io::BufReader}; use raylib::{ camera::Camera2D, math::Vector2, prelude::RaylibDrawHandle, RaylibHandle, RaylibThread, }; -use crate::{player::Player, resources::GlobalResources, world::World}; +use crate::{items::ShopItems, player::Player, resources::GlobalResources, world::World}; +use failure::Error; use log::debug; +use serde::{Deserialize, Serialize}; /// Overall states for the game #[derive(Debug, PartialEq, Copy, Clone)] @@ -18,7 +20,7 @@ pub enum GameState { PauseMenu, GameQuit, InGame, - GameEnd + GameEnd, } impl fmt::Display for GameState { @@ -27,6 +29,51 @@ impl fmt::Display for GameState { } } +#[derive(Debug, Serialize, Deserialize, Default)] +pub struct GameProgress { + coins: u32, + max_depth: f32, + fastest_time: Option, + inventory: Vec, +} + +impl GameProgress { + pub fn new() -> Self { + Self { + ..Default::default() + } + } + + pub fn from_file(file: String) -> Result { + // Load the file + let file = File::open(file)?; + let reader = BufReader::new(file); + + // Deserialize + Ok(serde_json::from_reader(reader)?) + } + + pub fn try_from_file(file: String) -> Self { + // Load from file + let loaded = GameProgress::from_file(file); + if loaded.is_ok() { + return loaded.unwrap(); + } else { + return GameProgress::new(); + } + } + + pub fn to_file(&self, file: String) -> Result<(), Error> { + // Serialize + let json = serde_json::to_string(self)?; + + // Write to file + std::fs::write(file, json)?; + + Ok(()) + } +} + /// This structure contains the entire game state, and should be passed around to various logic functions. pub struct GameCore { /// The game's overall state @@ -50,10 +97,16 @@ pub struct GameCore { /// The player pub player: Player, + pub progress: GameProgress, } impl GameCore { - pub fn new(raylib: &mut RaylibHandle, thread: &RaylibThread, world: World) -> Self { + pub fn new( + raylib: &mut RaylibHandle, + thread: &RaylibThread, + world: World, + progress: GameProgress, + ) -> Self { Self { state: GameState::Loading, last_state: GameState::Loading, @@ -71,6 +124,7 @@ impl GameCore { show_simple_debug_info: false, world: world, player: Player::new(), + progress: progress, } } diff --git a/src/items.rs b/src/items.rs new file mode 100644 index 0000000..b49d49b --- /dev/null +++ b/src/items.rs @@ -0,0 +1,10 @@ +use serde::{Serialize, Deserialize}; + +#[derive(Debug, PartialEq, Clone, Serialize, Deserialize)] +#[serde(tag = "t", content = "c")] +pub enum ShopItems { + StunGun(u8), + AirBag, + Flashlight(u8), + Flippers(u8) +} \ No newline at end of file diff --git a/src/logic/gameend.rs b/src/logic/gameend.rs index 8cee91f..94690dd 100644 --- a/src/logic/gameend.rs +++ b/src/logic/gameend.rs @@ -59,6 +59,8 @@ impl Screen for GameEndScreen { Color::BLACK, ); + // TODO: Save game progress + // // Close and quit buttons // let bottom_left_button_dimensions = Rectangle { diff --git a/src/main.rs b/src/main.rs index 85ff39b..67f4c0b 100644 --- a/src/main.rs +++ b/src/main.rs @@ -5,8 +5,9 @@ mod resources; mod player; mod world; mod pallette; +mod items; -use gamecore::{GameCore, GameState}; +use gamecore::{GameCore, GameProgress, GameState}; use lib::{utils::profiler::GameProfiler, wrappers::audio::player::AudioPlayer}; use log::info; use logic::{gameend::GameEndScreen, ingame::InGameScreen, loadingscreen::LoadingScreen, mainmenu::MainMenuScreen, pausemenu::PauseMenuScreen, screen::Screen}; @@ -41,8 +42,11 @@ fn main() { // Load the world let world = World::load_from_json("./assets/worlds/mainworld.json".to_string()).expect("Failed to load main world JSON"); + // Load the game progress + let game_progress = GameProgress::try_from_file("./assets/savestate.json".to_string()); + // Set up the game's core state - let mut game_core = GameCore::new(&mut raylib, &raylib_thread, world); + let mut game_core = GameCore::new(&mut raylib, &raylib_thread, world, game_progress); // Set up the game's profiler let mut profiler = GameProfiler::new(); From a0735b774baeb38bec679a58b88ed82e375d0900 Mon Sep 17 00:00:00 2001 From: Evan Pratten Date: Sat, 24 Apr 2021 11:17:34 -0400 Subject: [PATCH 19/27] triangle fish --- assets/worlds/mainworld.json | 8 ++- src/entities/fish.rs | 98 ++++++++++++++++++++++++++++++++++++ src/entities/mod.rs | 1 + src/lib/utils/mod.rs | 3 +- src/lib/utils/triangles.rs | 14 ++++++ src/logic/ingame/mod.rs | 14 ++++-- src/main.rs | 1 + src/world.rs | 17 ++++++- 8 files changed, 149 insertions(+), 7 deletions(-) create mode 100644 src/entities/fish.rs create mode 100644 src/entities/mod.rs create mode 100644 src/lib/utils/triangles.rs diff --git a/assets/worlds/mainworld.json b/assets/worlds/mainworld.json index f34b38e..70fb440 100644 --- a/assets/worlds/mainworld.json +++ b/assets/worlds/mainworld.json @@ -2,5 +2,11 @@ "end_position": { "x": 10000.0, "y": 10000.0 - } + }, + "fish": [ + { + "x": 100.0, + "y": 100.0 + } + ] } \ No newline at end of file diff --git a/src/entities/fish.rs b/src/entities/fish.rs new file mode 100644 index 0000000..7b01a6b --- /dev/null +++ b/src/entities/fish.rs @@ -0,0 +1,98 @@ +use raylib::prelude::*; + +use crate::{gamecore::GameCore, lib::utils::triangles::rotate_vector, player::Player}; + +const FISH_FOLLOW_PLAYER_DISTANCE: f32 = 80.0; +const FISH_FOLLOW_PLAYER_SPEED: f32 = 2.0; +const FISH_FOLLOW_PLAYER_SPEED_FAST: f32 = 6.0; + +#[derive(Debug, Clone)] +pub struct FishEntity { + position: Vector2, + direction: Vector2, + following_player: bool, + size: Vector2 +} + +impl FishEntity { + pub fn new(position: Vector2) -> Self { + Self { + position: position, + direction: Vector2::zero(), + following_player: true, + size: Vector2 { + x: 5.0, + y: 8.0 + } + } + } + + pub fn new_from_positions(positions: &Vec) -> Vec { + let mut output = Vec::new(); + for position in positions { + output.push(FishEntity::new(*position)); + } + return output; + } + + pub fn handle_follow_player(&mut self, player: &Player) { + // Distance and direction to player + let dist_to_player = player.position - self.position; + let dist_to_player_lin = self.position.distance_to(player.position); + let mut direction_to_player = dist_to_player; + direction_to_player.normalize(); + + // Fish movement + let movement; + + // If the fish is double its follow distance from the player + if dist_to_player_lin.abs() > (FISH_FOLLOW_PLAYER_DISTANCE * 2.0) { + movement = direction_to_player * FISH_FOLLOW_PLAYER_SPEED_FAST; + } else { + // Move slowly in the direction of the player unless too close + if dist_to_player_lin.abs() > FISH_FOLLOW_PLAYER_DISTANCE { + movement = direction_to_player * FISH_FOLLOW_PLAYER_SPEED; + } else { + movement = Vector2::zero(); + } + } + + // Move the fish + self.direction = direction_to_player; + self.position += movement; + } + + pub fn handle_free_movement(&mut self) {} + + pub fn update_position(&mut self, player: &Player, dt: f64) { + if self.following_player { + self.handle_follow_player(player); + } else { + self.handle_free_movement(); + } + } + + pub fn render(&self, context_2d: &mut RaylibMode2D) { + + // Direction + let direction = Vector2::zero().angle_to(self.direction.normalized()) + (90.0 as f32).to_radians(); + + // Get the corners of the fish + let fish_front = rotate_vector(Vector2 { + x: 0.0, + y: (self.size.y / 2.0) * -1.0, + }, direction); + let fish_bl = rotate_vector(Vector2 { + x: (self.size.x / 2.0) * -1.0, + y: (self.size.y / 2.0), + }, direction); + let fish_br = rotate_vector(Vector2 { + x: (self.size.x / 2.0), + y: (self.size.y / 2.0), + }, direction); + + // Draw the fish as a triangle with rotation + context_2d.draw_triangle(self.position + fish_front, self.position + fish_bl, self.position + fish_br, Color::BLACK); + + } +} diff --git a/src/entities/mod.rs b/src/entities/mod.rs new file mode 100644 index 0000000..e948307 --- /dev/null +++ b/src/entities/mod.rs @@ -0,0 +1 @@ +pub mod fish; \ No newline at end of file diff --git a/src/lib/utils/mod.rs b/src/lib/utils/mod.rs index c570215..9d6dd04 100644 --- a/src/lib/utils/mod.rs +++ b/src/lib/utils/mod.rs @@ -1 +1,2 @@ -pub mod profiler; \ No newline at end of file +pub mod profiler; +pub mod triangles; \ No newline at end of file diff --git a/src/lib/utils/triangles.rs b/src/lib/utils/triangles.rs new file mode 100644 index 0000000..a0f1c10 --- /dev/null +++ b/src/lib/utils/triangles.rs @@ -0,0 +1,14 @@ +use raylib::math::Vector2; + + +pub fn rotate_vector(vector: Vector2, angle_rad: f32) -> Vector2{ + + // let dist = (vector.x * vector.x) + (vector.y * vector.y); + // let angle = (vector.x.abs() / vector.y.abs()).atan(); + // let angle = angle + angle_rad; + return Vector2 { + x: (vector.x * angle_rad.cos()) - (vector.y * angle_rad.sin()), + y: (vector.y * angle_rad.cos()) + (vector.x * angle_rad.sin()), + }; + +} \ No newline at end of file diff --git a/src/logic/ingame/mod.rs b/src/logic/ingame/mod.rs index 7297289..fb93e50 100644 --- a/src/logic/ingame/mod.rs +++ b/src/logic/ingame/mod.rs @@ -1,5 +1,5 @@ -mod playerlogic; mod hud; +mod playerlogic; use raylib::prelude::*; @@ -33,8 +33,6 @@ impl InGameScreen { ) { context_2d.draw_circle(0, 0, 10.0, Color::BLACK); } - - } impl Screen for InGameScreen { @@ -45,6 +43,9 @@ impl Screen for InGameScreen { audio_system: &mut AudioPlayer, game_core: &mut GameCore, ) -> Option { + // Calculate DT + let dt = draw_handle.get_time() - game_core.last_frame_time; + // Clear frame draw_handle.clear_background(Color::BLUE); @@ -71,6 +72,13 @@ impl Screen for InGameScreen { // Render the world self.render_world(&mut context_2d, game_core); + // Render entities + let mut fish = &mut game_core.world.fish; + for fish in fish.iter_mut() { + fish.update_position(&game_core.player, dt); + fish.render(&mut context_2d); + } + // Render Player playerlogic::render_player(&mut context_2d, game_core); } diff --git a/src/main.rs b/src/main.rs index 85ff39b..8fd784b 100644 --- a/src/main.rs +++ b/src/main.rs @@ -5,6 +5,7 @@ mod resources; mod player; mod world; mod pallette; +mod entities; use gamecore::{GameCore, GameState}; use lib::{utils::profiler::GameProfiler, wrappers::audio::player::AudioPlayer}; diff --git a/src/world.rs b/src/world.rs index 309fd41..0212480 100644 --- a/src/world.rs +++ b/src/world.rs @@ -5,9 +5,17 @@ use serde::{Deserialize, Serialize}; use std::io::Read; use failure::Error; +use crate::entities::fish::FishEntity; + #[derive(Debug, Serialize, Deserialize, Clone)] pub struct World { - pub end_position: Vector2 + pub end_position: Vector2, + + #[serde(rename = "fish")] + pub fish_positions: Vec, + + #[serde(skip)] + pub fish: Vec } impl World { @@ -17,6 +25,11 @@ impl World { let reader = BufReader::new(file); // Deserialize - Ok(serde_json::from_reader(reader)?) + let mut result: World = serde_json::from_reader(reader)?; + + // Init all fish + result.fish = FishEntity::new_from_positions(&result.fish_positions); + + Ok(result) } } From e3b523ecc152eeb058c79454d0eebab22ac15bb5 Mon Sep 17 00:00:00 2001 From: Evan Pratten Date: Sat, 24 Apr 2021 11:34:19 -0400 Subject: [PATCH 20/27] fish movement --- Cargo.toml | 1 + assets/worlds/mainworld.json | 8 +++- src/entities/fish.rs | 77 +++++++++++++++++++++++------------- 3 files changed, 56 insertions(+), 30 deletions(-) diff --git a/Cargo.toml b/Cargo.toml index 36bf495..6c805d4 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -17,3 +17,4 @@ parry2d = "0.4.0" log = "0.4.14" env_logger = "0.8.3" nalgebra = "0.26.1" +rand = "0.8.3" \ No newline at end of file diff --git a/assets/worlds/mainworld.json b/assets/worlds/mainworld.json index 70fb440..9d52a17 100644 --- a/assets/worlds/mainworld.json +++ b/assets/worlds/mainworld.json @@ -5,8 +5,12 @@ }, "fish": [ { - "x": 100.0, - "y": 100.0 + "x": 500.0, + "y": 500.0 + }, + { + "x": 800.0, + "y": 500.0 } ] } \ No newline at end of file diff --git a/src/entities/fish.rs b/src/entities/fish.rs index 7b01a6b..f2a784c 100644 --- a/src/entities/fish.rs +++ b/src/entities/fish.rs @@ -1,17 +1,20 @@ +use rand::{Rng, prelude::ThreadRng}; use raylib::prelude::*; use crate::{gamecore::GameCore, lib::utils::triangles::rotate_vector, player::Player}; const FISH_FOLLOW_PLAYER_DISTANCE: f32 = 80.0; const FISH_FOLLOW_PLAYER_SPEED: f32 = 2.0; -const FISH_FOLLOW_PLAYER_SPEED_FAST: f32 = 6.0; +const FISH_FOLLOW_PLAYER_SPEED_FAST: f32 = FISH_FOLLOW_PLAYER_SPEED * 3.0; +const FISH_ATTACH_RADIUS: f32 = 20.0; #[derive(Debug, Clone)] pub struct FishEntity { position: Vector2, direction: Vector2, following_player: bool, - size: Vector2 + size: Vector2, + rng: ThreadRng } impl FishEntity { @@ -20,10 +23,8 @@ impl FishEntity { position: position, direction: Vector2::zero(), following_player: true, - size: Vector2 { - x: 5.0, - y: 8.0 - } + size: Vector2 { x: 5.0, y: 8.0 }, + rng: rand::thread_rng() } } @@ -35,7 +36,7 @@ impl FishEntity { return output; } - pub fn handle_follow_player(&mut self, player: &Player) { + pub fn handle_follow_player(&mut self, player: &Player, dt: f64) { // Distance and direction to player let dist_to_player = player.position - self.position; let dist_to_player_lin = self.position.distance_to(player.position); @@ -45,13 +46,16 @@ impl FishEntity { // Fish movement let movement; + // Random variance + let variance = self.rng.gen_range(500.0..1000.0) / 1000.0; + // If the fish is double its follow distance from the player if dist_to_player_lin.abs() > (FISH_FOLLOW_PLAYER_DISTANCE * 2.0) { - movement = direction_to_player * FISH_FOLLOW_PLAYER_SPEED_FAST; + movement = direction_to_player * FISH_FOLLOW_PLAYER_SPEED_FAST * variance; } else { // Move slowly in the direction of the player unless too close if dist_to_player_lin.abs() > FISH_FOLLOW_PLAYER_DISTANCE { - movement = direction_to_player * FISH_FOLLOW_PLAYER_SPEED; + movement = direction_to_player * FISH_FOLLOW_PLAYER_SPEED * variance; } else { movement = Vector2::zero(); } @@ -62,37 +66,54 @@ impl FishEntity { self.position += movement; } - pub fn handle_free_movement(&mut self) {} + pub fn handle_free_movement(&mut self, player: &Player, dt: f64) { + if player.position.distance_to(self.position).abs() <= player.size.y * 2.0 { + self.following_player = true; + } + } pub fn update_position(&mut self, player: &Player, dt: f64) { if self.following_player { - self.handle_follow_player(player); + self.handle_follow_player(player, dt); } else { - self.handle_free_movement(); + self.handle_free_movement(player, dt); } } pub fn render(&self, context_2d: &mut RaylibMode2D) { - // Direction - let direction = Vector2::zero().angle_to(self.direction.normalized()) + (90.0 as f32).to_radians(); + let direction = + Vector2::zero().angle_to(self.direction.normalized()) + (90.0 as f32).to_radians(); // Get the corners of the fish - let fish_front = rotate_vector(Vector2 { - x: 0.0, - y: (self.size.y / 2.0) * -1.0, - }, direction); - let fish_bl = rotate_vector(Vector2 { - x: (self.size.x / 2.0) * -1.0, - y: (self.size.y / 2.0), - }, direction); - let fish_br = rotate_vector(Vector2 { - x: (self.size.x / 2.0), - y: (self.size.y / 2.0), - }, direction); + let fish_front = rotate_vector( + Vector2 { + x: 0.0, + y: (self.size.y / 2.0) * -1.0, + }, + direction, + ); + let fish_bl = rotate_vector( + Vector2 { + x: (self.size.x / 2.0) * -1.0, + y: (self.size.y / 2.0), + }, + direction, + ); + let fish_br = rotate_vector( + Vector2 { + x: (self.size.x / 2.0), + y: (self.size.y / 2.0), + }, + direction, + ); // Draw the fish as a triangle with rotation - context_2d.draw_triangle(self.position + fish_front, self.position + fish_bl, self.position + fish_br, Color::BLACK); - + context_2d.draw_triangle( + self.position + fish_front, + self.position + fish_bl, + self.position + fish_br, + Color::BLACK, + ); } } From 877a5ba4484f70faabe5468a4ab52bec52d99ccf Mon Sep 17 00:00:00 2001 From: Evan Pratten Date: Sat, 24 Apr 2021 11:37:17 -0400 Subject: [PATCH 21/27] fish pickup --- assets/worlds/mainworld.json | 4 ++-- src/entities/fish.rs | 15 +++++++++++++-- 2 files changed, 15 insertions(+), 4 deletions(-) diff --git a/assets/worlds/mainworld.json b/assets/worlds/mainworld.json index 9d52a17..45a78a5 100644 --- a/assets/worlds/mainworld.json +++ b/assets/worlds/mainworld.json @@ -6,11 +6,11 @@ "fish": [ { "x": 500.0, - "y": 500.0 + "y": 300.0 }, { "x": 800.0, - "y": 500.0 + "y": 200.0 } ] } \ No newline at end of file diff --git a/src/entities/fish.rs b/src/entities/fish.rs index f2a784c..869fb61 100644 --- a/src/entities/fish.rs +++ b/src/entities/fish.rs @@ -22,7 +22,7 @@ impl FishEntity { Self { position: position, direction: Vector2::zero(), - following_player: true, + following_player: false, size: Vector2 { x: 5.0, y: 8.0 }, rng: rand::thread_rng() } @@ -67,9 +67,20 @@ impl FishEntity { } pub fn handle_free_movement(&mut self, player: &Player, dt: f64) { - if player.position.distance_to(self.position).abs() <= player.size.y * 2.0 { + // Distance and direction to player + let dist_to_player = player.position - self.position; + let dist_to_player_lin = self.position.distance_to(player.position); + let mut direction_to_player = dist_to_player; + direction_to_player.normalize(); + + // Handle player picking up fish + if player.position.distance_to(self.position).abs() <= player.size.y * 1.2 { self.following_player = true; } + + // Look at the player; + self.position = self.position; + self.direction = direction_to_player; } pub fn update_position(&mut self, player: &Player, dt: f64) { From 09c776ba3f1c6165927acf3b2da2b4e9b8fd0d6c Mon Sep 17 00:00:00 2001 From: Evan Pratten Date: Sat, 24 Apr 2021 12:16:38 -0400 Subject: [PATCH 22/27] resizing everything --- src/entities/fish.rs | 6 +-- src/gamecore.rs | 2 +- src/logic/ingame/mod.rs | 1 + src/logic/ingame/playerlogic.rs | 79 +++++++++++++++++---------------- src/player.rs | 4 +- 5 files changed, 48 insertions(+), 44 deletions(-) diff --git a/src/entities/fish.rs b/src/entities/fish.rs index 869fb61..f78fbb3 100644 --- a/src/entities/fish.rs +++ b/src/entities/fish.rs @@ -3,8 +3,8 @@ use raylib::prelude::*; use crate::{gamecore::GameCore, lib::utils::triangles::rotate_vector, player::Player}; -const FISH_FOLLOW_PLAYER_DISTANCE: f32 = 80.0; -const FISH_FOLLOW_PLAYER_SPEED: f32 = 2.0; +const FISH_FOLLOW_PLAYER_DISTANCE: f32 = 30.0; +const FISH_FOLLOW_PLAYER_SPEED: f32 = 1.8; const FISH_FOLLOW_PLAYER_SPEED_FAST: f32 = FISH_FOLLOW_PLAYER_SPEED * 3.0; const FISH_ATTACH_RADIUS: f32 = 20.0; @@ -74,7 +74,7 @@ impl FishEntity { direction_to_player.normalize(); // Handle player picking up fish - if player.position.distance_to(self.position).abs() <= player.size.y * 1.2 { + if player.position.distance_to(self.position).abs() <= player.size.y * 2.2 { self.following_player = true; } diff --git a/src/gamecore.rs b/src/gamecore.rs index 2cd14c7..a29e2fa 100644 --- a/src/gamecore.rs +++ b/src/gamecore.rs @@ -119,7 +119,7 @@ impl GameCore { offset: Vector2::zero(), target: Vector2::zero(), rotation: 0.0, - zoom: 1.0, + zoom: 2.0, }, show_simple_debug_info: false, world: world, diff --git a/src/logic/ingame/mod.rs b/src/logic/ingame/mod.rs index fb93e50..843134e 100644 --- a/src/logic/ingame/mod.rs +++ b/src/logic/ingame/mod.rs @@ -61,6 +61,7 @@ impl Screen for InGameScreen { x: (win_width as f32 / 2.0), y: (win_height as f32 / 2.0), }; + let camera_window_center = window_center * (1.0 / game_core.master_camera.zoom); // Update player movement playerlogic::update_player_movement(draw_handle, game_core, window_center); diff --git a/src/logic/ingame/playerlogic.rs b/src/logic/ingame/playerlogic.rs index 37c0a16..0234ec0 100644 --- a/src/logic/ingame/playerlogic.rs +++ b/src/logic/ingame/playerlogic.rs @@ -5,10 +5,10 @@ use crate::{ pallette::{TRANSLUCENT_WHITE_128, TRANSLUCENT_WHITE_64, TRANSLUCENT_WHITE_96}, }; -const NORMAL_PLAYER_SPEED: i32 = 4; +const NORMAL_PLAYER_SPEED: i32 = 3; const BOOST_PLAYER_SPEED: i32 = NORMAL_PLAYER_SPEED * 2; const CAMERA_FOLLOW_SPEED: f32 = 0.7; -const TURN_SPEED: f32 = 0.05; +const TURN_SPEED: f32 = 0.15; const BOOST_DECREASE_PER_SECOND: f32 = 0.75; const BOOST_REGEN_PER_SECOND: f32 = 0.25; const BREATH_DECREASE_PER_SECOND: f32 = 0.01; @@ -27,45 +27,45 @@ pub fn update_player_movement( let raw_movement_direction = mouse_world_pose - game_core.player.position; let mut normalized_movement_direction = raw_movement_direction; normalized_movement_direction.normalize(); - - let tau: f32 = PI as f32 * 2.0; - // get angles as floats - let mut player_angle: f32 = Vector2::zero().angle_to(game_core.player.direction); - let mut desired_angle: f32 = Vector2::zero().angle_to(normalized_movement_direction); - // make angle positive - if desired_angle < 0.0 { - desired_angle += tau; - } + let tau: f32 = PI as f32 * 2.0; + // get angles as floats + let mut player_angle: f32 = Vector2::zero().angle_to(game_core.player.direction); + let mut desired_angle: f32 = Vector2::zero().angle_to(normalized_movement_direction); - // turn towards mouse at turn speed - if player_angle % tau > desired_angle { - if (player_angle % tau) - desired_angle > PI as f32 { - player_angle += TURN_SPEED; - } else { - player_angle -= TURN_SPEED; - } + // make angle positive + if desired_angle < 0.0 { + desired_angle += tau; + } + + // turn towards mouse at turn speed + if player_angle % tau > desired_angle { + if (player_angle % tau) - desired_angle > PI as f32 { + player_angle += TURN_SPEED; } else { - if desired_angle - (player_angle % tau) > PI as f32 { - player_angle -= TURN_SPEED; - } else { - player_angle += TURN_SPEED; - } + player_angle -= TURN_SPEED; } + } else { + if desired_angle - (player_angle % tau) > PI as f32 { + player_angle -= TURN_SPEED; + } else { + player_angle += TURN_SPEED; + } + } - // snap to mouse if close enough - if f32::abs(player_angle - desired_angle) < (TURN_SPEED * 1.1) { - player_angle = desired_angle; - } - if player_angle > tau { - player_angle -= tau; - } - if player_angle < 0.0 { - player_angle += tau; - } + // snap to mouse if close enough + if f32::abs(player_angle - desired_angle) < (TURN_SPEED * 1.1) { + player_angle = desired_angle; + } + if player_angle > tau { + player_angle -= tau; + } + if player_angle < 0.0 { + player_angle += tau; + } - // set angle - game_core.player.direction = Vector2::new(f32::cos(player_angle), f32::sin(player_angle)); + // set angle + game_core.player.direction = Vector2::new(f32::cos(player_angle), f32::sin(player_angle)); // In the case the player is in "null", just jump the camera to them if game_core.player.position == Vector2::zero() { @@ -135,9 +135,12 @@ pub fn render_player(context_2d: &mut RaylibMode2D, game_core: TRANSLUCENT_WHITE_64, ); context_2d.draw_ring( - player.position, - boost_ring_max_radius - 2.0, - boost_ring_max_radius + 2.0, + Vector2 { + x: player.position.x as i32 as f32, + y: player.position.y as i32 as f32, + }, + boost_ring_max_radius , + boost_ring_max_radius + 1.0, 0, (360.0 * player.breath_percent) as i32, 0, diff --git a/src/player.rs b/src/player.rs index a455e9b..d37f539 100644 --- a/src/player.rs +++ b/src/player.rs @@ -17,8 +17,8 @@ impl Player { Self { boost_percent: 1.0, size: Vector2 { - x: 11.0 * 4.0, - y: 21.0 * 4.0 + x: 11.0, + y: 21.0 }, breath_percent: 1.0, ..Default::default() From c0ad24029ab158bece20db77f67724540c3a01b1 Mon Sep 17 00:00:00 2001 From: Evan Pratten Date: Sat, 24 Apr 2021 12:39:53 -0400 Subject: [PATCH 23/27] working on animaiton --- src/lib/wrappers/animation.rs | 13 ++++++++--- src/lib/wrappers/complexanimation.rs | 32 ++++++++++++++++++++++++++++ src/lib/wrappers/mod.rs | 3 ++- 3 files changed, 44 insertions(+), 4 deletions(-) create mode 100644 src/lib/wrappers/complexanimation.rs diff --git a/src/lib/wrappers/animation.rs b/src/lib/wrappers/animation.rs index 3632dee..b6672df 100644 --- a/src/lib/wrappers/animation.rs +++ b/src/lib/wrappers/animation.rs @@ -48,9 +48,9 @@ impl FrameAnimationWrapper { } /// Draw the next frame to the screen at `position` - pub fn draw(&mut self, handle: &mut RaylibDrawHandle, position: Vector2) { + pub fn draw(&mut self, handle: &mut RaylibDrawHandle, position: Vector2, rotation: f32) { let frame_id = self.get_current_frame_id(handle); - self.draw_frame(handle, position, frame_id); + self.draw_frame(handle, position, rotation, frame_id); } /// Draw a specified frame to the screen at `position` @@ -58,6 +58,7 @@ impl FrameAnimationWrapper { &mut self, handle: &mut RaylibDrawHandle, position: Vector2, + rotation: f32, frame_number: u32, ) { // Determine the col number @@ -76,7 +77,13 @@ impl FrameAnimationWrapper { height: self.size.y, }; + // Rotation origin + let origin = Vector2 { + x: self.size.x, + y: self.size.y + }; + // Render - handle.draw_texture_rec(&mut self.sprite_sheet, frame_box, position, Color::WHITE); + handle.draw_texture_pro(&mut self.sprite_sheet, frame_box, position, origin, rotation, Color::WHITE); } } diff --git a/src/lib/wrappers/complexanimation.rs b/src/lib/wrappers/complexanimation.rs new file mode 100644 index 0000000..ad87572 --- /dev/null +++ b/src/lib/wrappers/complexanimation.rs @@ -0,0 +1,32 @@ +use std::usize; + +use raylib::prelude::*; + +pub struct FrameRange { + pub min: usize, + pub max: usize, +} + +pub struct ComplexAnimationTool { + sprite_sheet: Texture2D, + frames_per_second: f32, + frame_size: Vector2, + sprite_sheet_size_frames: Vector2 +} + +impl ComplexAnimationTool { + pub fn render_loop(&self, context_2d: &mut RaylibMode2D, bounds: Rectangle, rotation: f32, range: &FrameRange) { + + } + + pub fn render_frame(&self, context_2d: &mut RaylibMode2D, bounds: Rectangle, rotation: f32, id: usize) { + + // Convert the ID to an xy + let col_id = id % self.sprite_sheet_size_frames.x as usize; + let row_id = id / self.sprite_sheet_size_frames.y as usize; + + + + + } +} diff --git a/src/lib/wrappers/mod.rs b/src/lib/wrappers/mod.rs index 9af9ea2..1a8f43a 100644 --- a/src/lib/wrappers/mod.rs +++ b/src/lib/wrappers/mod.rs @@ -1,2 +1,3 @@ pub mod audio; -pub mod animation; \ No newline at end of file +pub mod animation; +pub mod complexanimation; \ No newline at end of file From f97ff9dec4710ea0af0cb86049df27b90f45ff12 Mon Sep 17 00:00:00 2001 From: Evan Pratten Date: Sat, 24 Apr 2021 12:48:22 -0400 Subject: [PATCH 24/27] animation --- src/lib/wrappers/animation.rs | 19 ++++++++++--------- src/logic/ingame/playerlogic.rs | 25 ++++++++++++++----------- src/resources.rs | 32 +++++++++++++++++++++++++++----- 3 files changed, 51 insertions(+), 25 deletions(-) diff --git a/src/lib/wrappers/animation.rs b/src/lib/wrappers/animation.rs index b6672df..3949089 100644 --- a/src/lib/wrappers/animation.rs +++ b/src/lib/wrappers/animation.rs @@ -1,9 +1,4 @@ -use raylib::{ - core::color::Color, - math::{Rectangle, Vector2}, - prelude::{RaylibDraw, RaylibDrawHandle}, - texture::Texture2D, -}; +use raylib::{core::color::Color, math::{Rectangle, Vector2}, prelude::{RaylibDraw, RaylibDrawHandle, RaylibMode2D}, texture::Texture2D}; /// A wrapper around an animation spritesheet pub struct FrameAnimationWrapper { @@ -48,7 +43,7 @@ impl FrameAnimationWrapper { } /// Draw the next frame to the screen at `position` - pub fn draw(&mut self, handle: &mut RaylibDrawHandle, position: Vector2, rotation: f32) { + pub fn draw(&mut self, handle: &mut RaylibMode2D, position: Vector2, rotation: f32) { let frame_id = self.get_current_frame_id(handle); self.draw_frame(handle, position, rotation, frame_id); } @@ -56,7 +51,7 @@ impl FrameAnimationWrapper { /// Draw a specified frame to the screen at `position` pub fn draw_frame( &mut self, - handle: &mut RaylibDrawHandle, + handle: &mut RaylibMode2D, position: Vector2, rotation: f32, frame_number: u32, @@ -76,6 +71,12 @@ impl FrameAnimationWrapper { width: self.size.x, height: self.size.y, }; + let frame_dest = Rectangle { + x: position.x, + y: position.y, + width: self.size.x, + height: self.size.y, + }; // Rotation origin let origin = Vector2 { @@ -84,6 +85,6 @@ impl FrameAnimationWrapper { }; // Render - handle.draw_texture_pro(&mut self.sprite_sheet, frame_box, position, origin, rotation, Color::WHITE); + handle.draw_texture_pro(&mut self.sprite_sheet, frame_box, frame_dest, origin, rotation, Color::WHITE); } } diff --git a/src/logic/ingame/playerlogic.rs b/src/logic/ingame/playerlogic.rs index 37c0a16..52d6262 100644 --- a/src/logic/ingame/playerlogic.rs +++ b/src/logic/ingame/playerlogic.rs @@ -144,16 +144,19 @@ pub fn render_player(context_2d: &mut RaylibMode2D, game_core: TRANSLUCENT_WHITE_96, ); + // Render the player + game_core.resources.player_animation_regular.draw(context_2d, player.position, player_rotation.to_degrees() + 90.0); + // TODO: tmp rect - context_2d.draw_rectangle_pro( - Rectangle { - x: player.position.x, - y: player.position.y, - width: player.size.x, - height: player.size.y, - }, - player.size / 2.0, - player_rotation.to_degrees() + 90.0, - Color::BLACK, - ); + // context_2d.draw_rectangle_pro( + // Rectangle { + // x: player.position.x, + // y: player.position.y, + // width: player.size.x, + // height: player.size.y, + // }, + // player.size / 2.0, + // player_rotation.to_degrees() + 90.0, + // Color::BLACK, + // ); } diff --git a/src/resources.rs b/src/resources.rs index 9ca9a7d..0857689 100644 --- a/src/resources.rs +++ b/src/resources.rs @@ -1,19 +1,41 @@ use failure::Error; -use raylib::{RaylibHandle, RaylibThread, texture::{Image, Texture2D}}; +use raylib::{ + math::Vector2, + texture::{Image, Texture2D}, + RaylibHandle, RaylibThread, +}; + +use crate::lib::wrappers::animation::FrameAnimationWrapper; /// This struct contains all textures and sounds that must be loaded into (V)RAM at the start of the game pub struct GlobalResources { - // Branding - pub game_logo: Texture2D + pub game_logo: Texture2D, + // Player + pub player_animation_regular: FrameAnimationWrapper, } impl GlobalResources { /// Load all resources. **THIS WILL HANG!** - pub fn load_all(raylib: &mut RaylibHandle, thread: &RaylibThread) -> Result { + pub fn load_all( + raylib: &mut RaylibHandle, + thread: &RaylibThread, + ) -> Result { Ok(GlobalResources { - game_logo: raylib.load_texture_from_image(&thread, &Image::load_image("./assets/img/logos/game-logo.png")?)? + game_logo: raylib.load_texture_from_image( + &thread, + &Image::load_image("./assets/img/logos/game-logo.png")?, + )?, + player_animation_regular: FrameAnimationWrapper::new( + raylib.load_texture_from_image( + &thread, + &Image::load_image("./assets/img/character/diveNormal.png")?, + )?, + Vector2 { x: 11.0, y: 21.0 }, + 8, + 100 / 8, + ), }) } } From 9cac7f83da7bb3fc6062a7c9ad76b218a6ccdb2f Mon Sep 17 00:00:00 2001 From: Evan Pratten Date: Sat, 24 Apr 2021 13:11:02 -0400 Subject: [PATCH 25/27] animated chr --- src/lib/wrappers/animation.rs | 8 ++-- src/logic/ingame/playerlogic.rs | 75 +++++++++++++++++++++++++-------- src/player.rs | 5 ++- src/resources.rs | 20 +++++++++ 4 files changed, 87 insertions(+), 21 deletions(-) diff --git a/src/lib/wrappers/animation.rs b/src/lib/wrappers/animation.rs index 3949089..053cf2b 100644 --- a/src/lib/wrappers/animation.rs +++ b/src/lib/wrappers/animation.rs @@ -24,7 +24,9 @@ impl FrameAnimationWrapper { /// Start the animation pub fn start(&mut self, handle: &RaylibDrawHandle) { - self.start_time_seconds = handle.get_time(); + if self.start_time_seconds == 0.0 { + self.start_time_seconds = handle.get_time(); + } } /// Stop (and reset) the animation @@ -80,8 +82,8 @@ impl FrameAnimationWrapper { // Rotation origin let origin = Vector2 { - x: self.size.x, - y: self.size.y + x: self.size.x / 2.0, + y: self.size.y / 2.0 }; // Render diff --git a/src/logic/ingame/playerlogic.rs b/src/logic/ingame/playerlogic.rs index e71554b..1460984 100644 --- a/src/logic/ingame/playerlogic.rs +++ b/src/logic/ingame/playerlogic.rs @@ -9,7 +9,7 @@ const NORMAL_PLAYER_SPEED: i32 = 3; const BOOST_PLAYER_SPEED: i32 = NORMAL_PLAYER_SPEED * 2; const CAMERA_FOLLOW_SPEED: f32 = 0.7; const TURN_SPEED: f32 = 0.15; -const BOOST_DECREASE_PER_SECOND: f32 = 0.75; +const BOOST_DECREASE_PER_SECOND: f32 = 0.65; const BOOST_REGEN_PER_SECOND: f32 = 0.25; const BREATH_DECREASE_PER_SECOND: f32 = 0.01; @@ -84,10 +84,36 @@ pub fn update_player_movement( // Decrease the boost game_core.player.boost_percent -= BOOST_DECREASE_PER_SECOND * dt as f32; + game_core.player.is_boosting = true; + if game_core.player.boost_percent >= 0.9 { + game_core + .resources + .player_animation_boost_charge + .start(draw_handle); + game_core.resources.player_animation_regular.stop(); + game_core.player.is_boost_charging = true; + } else { + game_core.resources.player_animation_boost_charge.stop(); + game_core + .resources + .player_animation_boost + .start(draw_handle); + game_core.player.is_boost_charging = false; + } } else { // Set the speed multiplier speed_multiplier = NORMAL_PLAYER_SPEED as f32; + // Reset boost animation + game_core.player.is_boosting = false; + game_core.player.is_boost_charging = false; + game_core.resources.player_animation_boost_charge.stop(); + game_core.resources.player_animation_boost.stop(); + game_core + .resources + .player_animation_regular + .start(draw_handle); + // Handle boost regen if !user_request_boost { game_core.player.boost_percent = (game_core.player.boost_percent @@ -104,6 +130,9 @@ pub fn update_player_movement( let player_real_movement = game_core.player.direction * speed_multiplier; if raw_movement_direction.distance_to(Vector2::zero()) > game_core.player.size.y / 2.0 { game_core.player.position += player_real_movement; + game_core.player.is_moving = true; + } else { + game_core.player.is_moving = false; } // Move the camera to follow the player @@ -139,7 +168,7 @@ pub fn render_player(context_2d: &mut RaylibMode2D, game_core: x: player.position.x as i32 as f32, y: player.position.y as i32 as f32, }, - boost_ring_max_radius , + boost_ring_max_radius, boost_ring_max_radius + 1.0, 0, (360.0 * player.breath_percent) as i32, @@ -147,19 +176,31 @@ pub fn render_player(context_2d: &mut RaylibMode2D, game_core: TRANSLUCENT_WHITE_96, ); - // Render the player - game_core.resources.player_animation_regular.draw(context_2d, player.position, player_rotation.to_degrees() + 90.0); - - // TODO: tmp rect - // context_2d.draw_rectangle_pro( - // Rectangle { - // x: player.position.x, - // y: player.position.y, - // width: player.size.x, - // height: player.size.y, - // }, - // player.size / 2.0, - // player_rotation.to_degrees() + 90.0, - // Color::BLACK, - // ); + // Render the player based on what is happening + if player.is_boost_charging { + game_core.resources.player_animation_boost_charge.draw( + context_2d, + player.position, + player_rotation.to_degrees() - 90.0, + ); + } else if player.is_boosting { + game_core.resources.player_animation_boost.draw( + context_2d, + player.position, + player_rotation.to_degrees() - 90.0, + ); + } else if player.is_moving { + game_core.resources.player_animation_regular.draw( + context_2d, + player.position, + player_rotation.to_degrees() - 90.0, + ); + } else { + game_core.resources.player_animation_regular.draw_frame( + context_2d, + player.position, + player_rotation.to_degrees() - 90.0, + 0, + ); + } } diff --git a/src/player.rs b/src/player.rs index d37f539..95c34a2 100644 --- a/src/player.rs +++ b/src/player.rs @@ -9,7 +9,10 @@ pub struct Player { pub size: Vector2, pub coins: u32, pub boost_percent: f32, - pub breath_percent: f32 + pub breath_percent: f32, + pub is_moving: bool, + pub is_boosting: bool, + pub is_boost_charging: bool } impl Player { diff --git a/src/resources.rs b/src/resources.rs index 0857689..7397358 100644 --- a/src/resources.rs +++ b/src/resources.rs @@ -14,6 +14,8 @@ pub struct GlobalResources { // Player pub player_animation_regular: FrameAnimationWrapper, + pub player_animation_boost_charge: FrameAnimationWrapper, + pub player_animation_boost: FrameAnimationWrapper, } impl GlobalResources { @@ -36,6 +38,24 @@ impl GlobalResources { 8, 100 / 8, ), + player_animation_boost_charge: FrameAnimationWrapper::new( + raylib.load_texture_from_image( + &thread, + &Image::load_image("./assets/img/character/diveStrokeCharge.png")?, + )?, + Vector2 { x: 11.0, y: 21.0 }, + 21, + 100 / 4, + ), + player_animation_boost: FrameAnimationWrapper::new( + raylib.load_texture_from_image( + &thread, + &Image::load_image("./assets/img/character/diveStroke.png")?, + )?, + Vector2 { x: 17.0, y: 21.0 }, + 21, + 30, + ), }) } } From 0f243e69175ae1a9c64ff85de51bf45f4a037a6c Mon Sep 17 00:00:00 2001 From: Evan Pratten Date: Sat, 24 Apr 2021 13:19:34 -0400 Subject: [PATCH 26/27] properly handle currency --- src/entities/fish.rs | 9 ++++++--- src/logic/ingame/mod.rs | 2 +- src/world.rs | 12 ++++++++++++ 3 files changed, 19 insertions(+), 4 deletions(-) diff --git a/src/entities/fish.rs b/src/entities/fish.rs index f78fbb3..7d994bc 100644 --- a/src/entities/fish.rs +++ b/src/entities/fish.rs @@ -12,7 +12,7 @@ const FISH_ATTACH_RADIUS: f32 = 20.0; pub struct FishEntity { position: Vector2, direction: Vector2, - following_player: bool, + pub following_player: bool, size: Vector2, rng: ThreadRng } @@ -66,7 +66,7 @@ impl FishEntity { self.position += movement; } - pub fn handle_free_movement(&mut self, player: &Player, dt: f64) { + pub fn handle_free_movement(&mut self, player: &mut Player, dt: f64) { // Distance and direction to player let dist_to_player = player.position - self.position; let dist_to_player_lin = self.position.distance_to(player.position); @@ -76,6 +76,9 @@ impl FishEntity { // Handle player picking up fish if player.position.distance_to(self.position).abs() <= player.size.y * 2.2 { self.following_player = true; + + // Add currency to the player + player.coins += 1; } // Look at the player; @@ -83,7 +86,7 @@ impl FishEntity { self.direction = direction_to_player; } - pub fn update_position(&mut self, player: &Player, dt: f64) { + pub fn update_position(&mut self, player: &mut Player, dt: f64) { if self.following_player { self.handle_follow_player(player, dt); } else { diff --git a/src/logic/ingame/mod.rs b/src/logic/ingame/mod.rs index 843134e..4a4e0a8 100644 --- a/src/logic/ingame/mod.rs +++ b/src/logic/ingame/mod.rs @@ -76,7 +76,7 @@ impl Screen for InGameScreen { // Render entities let mut fish = &mut game_core.world.fish; for fish in fish.iter_mut() { - fish.update_position(&game_core.player, dt); + fish.update_position(&mut game_core.player, dt); fish.render(&mut context_2d); } diff --git a/src/world.rs b/src/world.rs index 0212480..050d284 100644 --- a/src/world.rs +++ b/src/world.rs @@ -32,4 +32,16 @@ impl World { Ok(result) } + + pub fn spend_coins(&mut self, count: usize) { + for _ in 0..count { + self.fish.pop(); + } + } + + pub fn reset(&mut self) { + for fish in self.fish.iter_mut() { + fish.following_player = false; + } + } } From dc110c9bfe83ff6407ecb576f161db68cd99a9a1 Mon Sep 17 00:00:00 2001 From: Evan Pratten Date: Sat, 24 Apr 2021 13:20:47 -0400 Subject: [PATCH 27/27] fix profiler name --- src/lib/utils/profiler.rs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/lib/utils/profiler.rs b/src/lib/utils/profiler.rs index 6634e5b..c813885 100644 --- a/src/lib/utils/profiler.rs +++ b/src/lib/utils/profiler.rs @@ -134,7 +134,7 @@ impl GameProfiler { w_type: None, }, DataSet { - title: Some("Breath".to_string()), + title: Some("Boost".to_string()), value: json!(self.data.player_boost_percent), graph: Some(false), unit: Some("%".to_string()),