From 7452803ba9ccf7521a7895c3586e694202083137 Mon Sep 17 00:00:00 2001 From: Evan Pratten Date: Sun, 25 Apr 2021 09:49:16 -0400 Subject: [PATCH 01/27] Create a base screen for the shop --- src/gamecore.rs | 1 + src/logic/mod.rs | 3 +- src/logic/shopscreen.rs | 130 ++++++++++++++++++++++++++++++++++++++++ src/main.rs | 9 ++- 4 files changed, 141 insertions(+), 2 deletions(-) create mode 100644 src/logic/shopscreen.rs diff --git a/src/gamecore.rs b/src/gamecore.rs index 19b9592..fcbe880 100644 --- a/src/gamecore.rs +++ b/src/gamecore.rs @@ -25,6 +25,7 @@ pub enum GameState { GameQuit, InGame, GameEnd, + InShop } impl fmt::Display for GameState { diff --git a/src/logic/mod.rs b/src/logic/mod.rs index 362eee2..19c7044 100644 --- a/src/logic/mod.rs +++ b/src/logic/mod.rs @@ -3,4 +3,5 @@ pub mod loadingscreen; pub mod mainmenu; pub mod pausemenu; pub mod ingame; -pub mod gameend; \ No newline at end of file +pub mod gameend; +pub mod shopscreen; \ No newline at end of file diff --git a/src/logic/shopscreen.rs b/src/logic/shopscreen.rs new file mode 100644 index 0000000..3c1db0a --- /dev/null +++ b/src/logic/shopscreen.rs @@ -0,0 +1,130 @@ +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: 380.0 }; + +pub struct ShopScreen {} + +impl ShopScreen { + pub fn new() -> Self { + Self {} + } +} + +impl Screen for ShopScreen { + 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? + + // If escape is pressed again, return to the previous render state + if draw_handle.is_key_pressed(KeyboardKey::KEY_ESCAPE) { + return Some(game_core.last_state); + } + + // 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( + "SHOP", + (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/main.rs b/src/main.rs index aa48d4a..a98eed6 100644 --- a/src/main.rs +++ b/src/main.rs @@ -11,7 +11,7 @@ mod items; 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}; +use logic::{gameend::GameEndScreen, ingame::InGameScreen, loadingscreen::LoadingScreen, mainmenu::MainMenuScreen, pausemenu::PauseMenuScreen, screen::Screen, shopscreen::ShopScreen}; use raylib::prelude::*; use world::{World, load_world_colliders}; @@ -63,6 +63,7 @@ fn main() { let mut pause_menu_screen = PauseMenuScreen::new(); let mut ingame_screen = InGameScreen::new(); let mut game_end_screen = GameEndScreen::new(); + let mut shop_screen = ShopScreen::new(); // Main rendering loop while !raylib.window_should_close() { @@ -101,6 +102,12 @@ fn main() { &mut audio_system, &mut game_core, ), + GameState::InShop => shop_screen.render( + &mut draw_handle, + &raylib_thread, + &mut audio_system, + &mut game_core, + ), }; // If needed, update the global state From c4dd17f25da39ab56341ab21f4965e4532c843eb Mon Sep 17 00:00:00 2001 From: Evan Pratten Date: Sun, 25 Apr 2021 09:52:15 -0400 Subject: [PATCH 02/27] Add airbag item data Co-authored-by: wm-c --- src/items.rs | 22 +++++++++++++++++++++- 1 file changed, 21 insertions(+), 1 deletion(-) diff --git a/src/items.rs b/src/items.rs index 547477e..8d46661 100644 --- a/src/items.rs +++ b/src/items.rs @@ -22,7 +22,27 @@ impl StunGun { } #[derive(Debug, PartialEq, Clone, Serialize, Deserialize)] -pub struct AirBag; +pub struct AirBag{ + extra_oxygen: u32, +} + +impl AirBag { + pub fn lvl1() -> Self { + Self { + extra_oxygen: 15, + } + } + pub fn lvl2() -> Self { + Self { + extra_oxygen: 30, + } + } + pub fn lvl3() -> Self { + Self { + extra_oxygen: 45, + } + } +} #[derive(Debug, PartialEq, Clone, Serialize, Deserialize)] pub struct Flashlight { From 73ac4d25a9944211b5afcd926d798e0e0dfda2f6 Mon Sep 17 00:00:00 2001 From: Evan Pratten Date: Sun, 25 Apr 2021 09:53:24 -0400 Subject: [PATCH 03/27] lvl3 stun gun Co-authored-by: wm-c --- src/items.rs | 6 ++++++ 1 file changed, 6 insertions(+) diff --git a/src/items.rs b/src/items.rs index 8d46661..fe5f47e 100644 --- a/src/items.rs +++ b/src/items.rs @@ -19,6 +19,12 @@ impl StunGun { duration: 1.25, } } + pub fn lvl3() -> Self { + Self { + range: 80.0, + duration: 1.0, + } + } } #[derive(Debug, PartialEq, Clone, Serialize, Deserialize)] From 6c288e91787412918a1fffbdd8ce07730f49eca4 Mon Sep 17 00:00:00 2001 From: Evan Pratten Date: Sun, 25 Apr 2021 09:53:56 -0400 Subject: [PATCH 04/27] Add flashlight Co-authored-by: wm-c --- src/items.rs | 14 ++++++++++++-- 1 file changed, 12 insertions(+), 2 deletions(-) diff --git a/src/items.rs b/src/items.rs index fe5f47e..1b5e155 100644 --- a/src/items.rs +++ b/src/items.rs @@ -55,10 +55,20 @@ pub struct Flashlight { pub radius: f32 } -impl Flashlight { +impl Flashlight{ pub fn lvl1() -> Self { Self { - radius: 0.25 + radius: 0.25, + } + } + pub fn lvl2() -> Self { + Self { + radius: 0.5, + } + } + pub fn lvl3() -> Self { + Self { + radius: 1.0, } } } From a68d3e8e709b63bf339f568ea45c135f4019b67a Mon Sep 17 00:00:00 2001 From: Evan Pratten Date: Sun, 25 Apr 2021 09:54:23 -0400 Subject: [PATCH 05/27] lvl3 flippers Co-authored-by: wm-c --- src/items.rs | 5 +++++ 1 file changed, 5 insertions(+) diff --git a/src/items.rs b/src/items.rs index 1b5e155..e52b92c 100644 --- a/src/items.rs +++ b/src/items.rs @@ -89,4 +89,9 @@ impl Flippers { speed_increase: 1.5 } } + pub fn lvl3() -> Self { + Self { + speed_increase: 1.8 + } + } } From 2a24d45ccef5ee97ab57cdf76538a98f403d5354 Mon Sep 17 00:00:00 2001 From: Evan Pratten Date: Sun, 25 Apr 2021 09:58:43 -0400 Subject: [PATCH 06/27] Pull in shop logic Co-authored-by: wm-c --- src/logic/shopscreen.rs | 187 +++++++++++++++++++++++++++++++++++++++- 1 file changed, 184 insertions(+), 3 deletions(-) diff --git a/src/logic/shopscreen.rs b/src/logic/shopscreen.rs index 3c1db0a..b57c9bf 100644 --- a/src/logic/shopscreen.rs +++ b/src/logic/shopscreen.rs @@ -9,11 +9,62 @@ use super::screen::Screen; const SCREEN_PANEL_SIZE: Vector2 = Vector2 { x: 300.0, y: 380.0 }; -pub struct ShopScreen {} +pub struct Item { + x_pose: i32, + y_pose: i32, + width: i32, + height: i32, + cost: u8, + level: u8, + name: String, +} + +#[derive(Debug, Default)] +pub struct ShopScreen { + shop_items: Vec, +} impl ShopScreen { pub fn new() -> Self { - Self {} + Self { + ..Default::default() + } + } + + // Creates all the items + pub fn create_items(&mut self, screen_dimension: Vector2) { + // gets every item.. hacky + let items = ShopItems::get_inital_items(); + + // sets sizes any random number is just a number I think looks good + let screen_width = screen_dimension.x as f32; + let screen_height = screen_dimension.y as f32; + + let box_height = screen_height * 0.15; + let box_width = screen_width * 0.1; + + let start_width = screen_width - (box_width * 4.0) - 40.0; + let draw_height = screen_height - 20.0 - box_height; + + let mut item_vec = Vec::new(); + + for box_num in 0..4 { + let x_pose = start_width + box_width * box_num as f32; + + // adds an item struct to the item list + item_vec.push(Item { + x_pose: ((x_pose + (5 * box_num) as f32) as i32), + y_pose: (draw_height as i32), + width: (box_width as i32), + height: (box_height as i32), + // Crazy hacky but this gets the data from the enum + cost: (ShopItems::get_cost(&items.get(box_num).unwrap())), + level: (ShopItems::get_level(&items.get(box_num).unwrap())), + name: (ShopItems::get_name(&items.get(box_num).unwrap())), + }); + } + + self.shop_items = item_vec; } } @@ -63,7 +114,6 @@ impl Screen for ShopScreen { 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, @@ -128,3 +178,134 @@ impl Screen for ShopScreen { return None; } } +pub fn render_shop( + draw_handle: &mut RaylibDrawHandle, + game_core: &mut GameCore, + inGameScreen: &mut InGameScreen, +) { + // Pressing F exits from buying + if draw_handle.is_key_pressed(KeyboardKey::KEY_F) { + inGameScreen.current_state = InGameState::SWIMMING; + } + + let mouse_position = draw_handle.get_mouse_position(); + + draw_handle.draw_text( + &format!("Coins: {}", game_core.player.coins), + 15, + 15, + 30, + Color::WHITE, + ); + + // Draws shop boxes + for mut item in inGameScreen.shop.shop_items.iter_mut() { + // If hovering on square draw full + if mouse_position.x >= item.x_pose as f32 + && mouse_position.x <= item.x_pose as f32 + item.width as f32 + && mouse_position.y >= item.y_pose as f32 + && mouse_position.y <= item.y_pose as f32 + item.width as f32 + { + // Draw rect + draw_handle.draw_rectangle( + item.x_pose, + item.y_pose, + item.width, + item.height, + Color::BLACK, + ); + + // Preform purchasing functions + if draw_handle.is_mouse_button_pressed(MouseButton::MOUSE_LEFT_BUTTON) + && game_core.player.coins >= item.cost as u32 + { + // Remove currency + game_core.world.spend_coins(item.cost.into()); + game_core.player.coins -= item.cost as u32; + + // Upgrade item in inventory + match &(item.name)[..] { + "Stun Gun" => { + match item.level { + 0 => game_core.player.inventory.stun_gun = Some(items::StunGun::lvl1()), + 1 => game_core.player.inventory.stun_gun = Some(items::StunGun::lvl2()), + 2 => game_core.player.inventory.stun_gun = Some(items::StunGun::lvl3()), + _ => (return), + }; + item.cost += 5; + item.level += 1; + } + "Air Bag" => { + match item.level { + 0 => { + game_core.player.inventory.air_bag = Some(items::AirBag::lvl1()); + } + 1 => { + game_core.player.inventory.air_bag = Some(items::AirBag::lvl2()); + } + 2 => { + game_core.player.inventory.air_bag = Some(items::AirBag::lvl3()); + } + _ => (return), + }; + item.cost += 5; + item.level += 1; + } + "Flash Light" => { + match item.level { + 0 => { + game_core.player.inventory.flashlight = + Some(items::Flashlight::lvl1()); + } + 1 => { + game_core.player.inventory.flashlight = + Some(items::Flashlight::lvl2()); + } + 2 => { + game_core.player.inventory.flashlight = + Some(items::Flashlight::lvl3()); + } + _ => (return), + }; + item.cost += 5; + item.level += 1; + } + "Flippers" => { + match item.level { + 0 => { + game_core.player.inventory.flippers = Some(items::Flippers::lvl1()); + } + 1 => { + game_core.player.inventory.flippers = Some(items::Flippers::lvl2()); + } + 2 => { + game_core.player.inventory.flippers = Some(items::Flippers::lvl3()); + } + _ => (return), + }; + item.cost += 5; + item.level += 1; + } + _ => (return), + }; + } + } else { + // outlines if not hovered + draw_handle.draw_rectangle_lines( + item.x_pose, + item.y_pose, + item.width, + item.height, + Color::BLACK, + ); + } + // Draw text about object + draw_handle.draw_text( + &format!("{}: ${}", item.name, item.cost), + item.x_pose + 5, + item.y_pose + 5, + 12, + Color::BLACK, + ); + } +} From bed3f407d6a6602bfe465df0734b602f083933e2 Mon Sep 17 00:00:00 2001 From: Evan Pratten Date: Sun, 25 Apr 2021 10:01:02 -0400 Subject: [PATCH 07/27] tmp disable shop code --- src/logic/shopscreen.rs | 310 ++++++++++++++++++++-------------------- 1 file changed, 155 insertions(+), 155 deletions(-) diff --git a/src/logic/shopscreen.rs b/src/logic/shopscreen.rs index b57c9bf..8ca2c6e 100644 --- a/src/logic/shopscreen.rs +++ b/src/logic/shopscreen.rs @@ -21,7 +21,7 @@ pub struct Item { #[derive(Debug, Default)] pub struct ShopScreen { - shop_items: Vec, + // shop_items: Vec, } impl ShopScreen { @@ -31,41 +31,41 @@ impl ShopScreen { } } - // Creates all the items - pub fn create_items(&mut self, screen_dimension: Vector2) { - // gets every item.. hacky - let items = ShopItems::get_inital_items(); + // // Creates all the items + // pub fn create_items(&mut self, screen_dimension: Vector2) { + // // gets every item.. hacky + // let items = ShopItems::get_inital_items(); - // sets sizes any random number is just a number I think looks good - let screen_width = screen_dimension.x as f32; - let screen_height = screen_dimension.y as f32; + // // sets sizes any random number is just a number I think looks good + // let screen_width = screen_dimension.x as f32; + // let screen_height = screen_dimension.y as f32; - let box_height = screen_height * 0.15; - let box_width = screen_width * 0.1; + // let box_height = screen_height * 0.15; + // let box_width = screen_width * 0.1; - let start_width = screen_width - (box_width * 4.0) - 40.0; - let draw_height = screen_height - 20.0 - box_height; + // let start_width = screen_width - (box_width * 4.0) - 40.0; + // let draw_height = screen_height - 20.0 - box_height; - let mut item_vec = Vec::new(); + // let mut item_vec = Vec::new(); - for box_num in 0..4 { - let x_pose = start_width + box_width * box_num as f32; + // for box_num in 0..4 { + // let x_pose = start_width + box_width * box_num as f32; - // adds an item struct to the item list - item_vec.push(Item { - x_pose: ((x_pose + (5 * box_num) as f32) as i32), - y_pose: (draw_height as i32), - width: (box_width as i32), - height: (box_height as i32), - // Crazy hacky but this gets the data from the enum - cost: (ShopItems::get_cost(&items.get(box_num).unwrap())), - level: (ShopItems::get_level(&items.get(box_num).unwrap())), - name: (ShopItems::get_name(&items.get(box_num).unwrap())), - }); - } + // // adds an item struct to the item list + // item_vec.push(Item { + // x_pose: ((x_pose + (5 * box_num) as f32) as i32), + // y_pose: (draw_height as i32), + // width: (box_width as i32), + // height: (box_height as i32), + // // Crazy hacky but this gets the data from the enum + // cost: (ShopItems::get_cost(&items.get(box_num).unwrap())), + // level: (ShopItems::get_level(&items.get(box_num).unwrap())), + // name: (ShopItems::get_name(&items.get(box_num).unwrap())), + // }); + // } - self.shop_items = item_vec; - } + // self.shop_items = item_vec; + // } } impl Screen for ShopScreen { @@ -178,134 +178,134 @@ impl Screen for ShopScreen { return None; } } -pub fn render_shop( - draw_handle: &mut RaylibDrawHandle, - game_core: &mut GameCore, - inGameScreen: &mut InGameScreen, -) { - // Pressing F exits from buying - if draw_handle.is_key_pressed(KeyboardKey::KEY_F) { - inGameScreen.current_state = InGameState::SWIMMING; - } +// pub fn render_shop( +// draw_handle: &mut RaylibDrawHandle, +// game_core: &mut GameCore, +// inGameScreen: &mut InGameScreen, +// ) { +// // Pressing F exits from buying +// if draw_handle.is_key_pressed(KeyboardKey::KEY_F) { +// inGameScreen.current_state = InGameState::SWIMMING; +// } - let mouse_position = draw_handle.get_mouse_position(); +// let mouse_position = draw_handle.get_mouse_position(); - draw_handle.draw_text( - &format!("Coins: {}", game_core.player.coins), - 15, - 15, - 30, - Color::WHITE, - ); +// draw_handle.draw_text( +// &format!("Coins: {}", game_core.player.coins), +// 15, +// 15, +// 30, +// Color::WHITE, +// ); - // Draws shop boxes - for mut item in inGameScreen.shop.shop_items.iter_mut() { - // If hovering on square draw full - if mouse_position.x >= item.x_pose as f32 - && mouse_position.x <= item.x_pose as f32 + item.width as f32 - && mouse_position.y >= item.y_pose as f32 - && mouse_position.y <= item.y_pose as f32 + item.width as f32 - { - // Draw rect - draw_handle.draw_rectangle( - item.x_pose, - item.y_pose, - item.width, - item.height, - Color::BLACK, - ); +// // Draws shop boxes +// for mut item in inGameScreen.shop.shop_items.iter_mut() { +// // If hovering on square draw full +// if mouse_position.x >= item.x_pose as f32 +// && mouse_position.x <= item.x_pose as f32 + item.width as f32 +// && mouse_position.y >= item.y_pose as f32 +// && mouse_position.y <= item.y_pose as f32 + item.width as f32 +// { +// // Draw rect +// draw_handle.draw_rectangle( +// item.x_pose, +// item.y_pose, +// item.width, +// item.height, +// Color::BLACK, +// ); - // Preform purchasing functions - if draw_handle.is_mouse_button_pressed(MouseButton::MOUSE_LEFT_BUTTON) - && game_core.player.coins >= item.cost as u32 - { - // Remove currency - game_core.world.spend_coins(item.cost.into()); - game_core.player.coins -= item.cost as u32; +// // Preform purchasing functions +// if draw_handle.is_mouse_button_pressed(MouseButton::MOUSE_LEFT_BUTTON) +// && game_core.player.coins >= item.cost as u32 +// { +// // Remove currency +// game_core.world.spend_coins(item.cost.into()); +// game_core.player.coins -= item.cost as u32; - // Upgrade item in inventory - match &(item.name)[..] { - "Stun Gun" => { - match item.level { - 0 => game_core.player.inventory.stun_gun = Some(items::StunGun::lvl1()), - 1 => game_core.player.inventory.stun_gun = Some(items::StunGun::lvl2()), - 2 => game_core.player.inventory.stun_gun = Some(items::StunGun::lvl3()), - _ => (return), - }; - item.cost += 5; - item.level += 1; - } - "Air Bag" => { - match item.level { - 0 => { - game_core.player.inventory.air_bag = Some(items::AirBag::lvl1()); - } - 1 => { - game_core.player.inventory.air_bag = Some(items::AirBag::lvl2()); - } - 2 => { - game_core.player.inventory.air_bag = Some(items::AirBag::lvl3()); - } - _ => (return), - }; - item.cost += 5; - item.level += 1; - } - "Flash Light" => { - match item.level { - 0 => { - game_core.player.inventory.flashlight = - Some(items::Flashlight::lvl1()); - } - 1 => { - game_core.player.inventory.flashlight = - Some(items::Flashlight::lvl2()); - } - 2 => { - game_core.player.inventory.flashlight = - Some(items::Flashlight::lvl3()); - } - _ => (return), - }; - item.cost += 5; - item.level += 1; - } - "Flippers" => { - match item.level { - 0 => { - game_core.player.inventory.flippers = Some(items::Flippers::lvl1()); - } - 1 => { - game_core.player.inventory.flippers = Some(items::Flippers::lvl2()); - } - 2 => { - game_core.player.inventory.flippers = Some(items::Flippers::lvl3()); - } - _ => (return), - }; - item.cost += 5; - item.level += 1; - } - _ => (return), - }; - } - } else { - // outlines if not hovered - draw_handle.draw_rectangle_lines( - item.x_pose, - item.y_pose, - item.width, - item.height, - Color::BLACK, - ); - } - // Draw text about object - draw_handle.draw_text( - &format!("{}: ${}", item.name, item.cost), - item.x_pose + 5, - item.y_pose + 5, - 12, - Color::BLACK, - ); - } -} +// // Upgrade item in inventory +// match &(item.name)[..] { +// "Stun Gun" => { +// match item.level { +// 0 => game_core.player.inventory.stun_gun = Some(items::StunGun::lvl1()), +// 1 => game_core.player.inventory.stun_gun = Some(items::StunGun::lvl2()), +// 2 => game_core.player.inventory.stun_gun = Some(items::StunGun::lvl3()), +// _ => (return), +// }; +// item.cost += 5; +// item.level += 1; +// } +// "Air Bag" => { +// match item.level { +// 0 => { +// game_core.player.inventory.air_bag = Some(items::AirBag::lvl1()); +// } +// 1 => { +// game_core.player.inventory.air_bag = Some(items::AirBag::lvl2()); +// } +// 2 => { +// game_core.player.inventory.air_bag = Some(items::AirBag::lvl3()); +// } +// _ => (return), +// }; +// item.cost += 5; +// item.level += 1; +// } +// "Flash Light" => { +// match item.level { +// 0 => { +// game_core.player.inventory.flashlight = +// Some(items::Flashlight::lvl1()); +// } +// 1 => { +// game_core.player.inventory.flashlight = +// Some(items::Flashlight::lvl2()); +// } +// 2 => { +// game_core.player.inventory.flashlight = +// Some(items::Flashlight::lvl3()); +// } +// _ => (return), +// }; +// item.cost += 5; +// item.level += 1; +// } +// "Flippers" => { +// match item.level { +// 0 => { +// game_core.player.inventory.flippers = Some(items::Flippers::lvl1()); +// } +// 1 => { +// game_core.player.inventory.flippers = Some(items::Flippers::lvl2()); +// } +// 2 => { +// game_core.player.inventory.flippers = Some(items::Flippers::lvl3()); +// } +// _ => (return), +// }; +// item.cost += 5; +// item.level += 1; +// } +// _ => (return), +// }; +// } +// } else { +// // outlines if not hovered +// draw_handle.draw_rectangle_lines( +// item.x_pose, +// item.y_pose, +// item.width, +// item.height, +// Color::BLACK, +// ); +// } +// // Draw text about object +// draw_handle.draw_text( +// &format!("{}: ${}", item.name, item.cost), +// item.x_pose + 5, +// item.y_pose + 5, +// 12, +// Color::BLACK, +// ); +// } +// } From 1bc5aa966d233f3eac3491a31c57925dbeea87f2 Mon Sep 17 00:00:00 2001 From: Evan Pratten Date: Sun, 25 Apr 2021 10:05:35 -0400 Subject: [PATCH 08/27] improved menu navigation --- src/logic/mainmenu.rs | 28 +++++++++++++++++++++++----- src/logic/pausemenu.rs | 4 ++-- src/logic/shopscreen.rs | 8 ++++---- 3 files changed, 29 insertions(+), 11 deletions(-) diff --git a/src/logic/mainmenu.rs b/src/logic/mainmenu.rs index e03d098..9f64a60 100644 --- a/src/logic/mainmenu.rs +++ b/src/logic/mainmenu.rs @@ -1,6 +1,10 @@ use raylib::prelude::*; -use crate::{gamecore::{GameCore, GameState}, lib::wrappers::audio::player::AudioPlayer, pallette::WATER_DARK}; +use crate::{ + gamecore::{GameCore, GameState}, + lib::wrappers::audio::player::AudioPlayer, + pallette::WATER_DARK, +}; use super::screen::Screen; @@ -38,10 +42,12 @@ impl Screen for MainMenuScreen { // Get mouse position data let mouse_position = draw_handle.get_mouse_position(); - let hovering_play_button = mouse_position.y > (win_width as f32 / 4.0) + let hovering_play_button = mouse_position.y > (win_width as f32 / 4.0) && mouse_position.y < (win_width as f32 / 4.0) + 60.0; - let hovering_quit_button = mouse_position.y > (win_width as f32 / 4.0) + 100.0 + let hovering_shop_button = mouse_position.y > (win_width as f32 / 4.0) + 100.0 && mouse_position.y < (win_width as f32 / 4.0) + 160.0; + let hovering_quit_button = mouse_position.y > (win_width as f32 / 4.0) + 200.0 + && mouse_position.y < (win_width as f32 / 4.0) + 260.0; // Play and quit draw_handle.draw_text( @@ -54,10 +60,20 @@ impl Screen for MainMenuScreen { false => Color::BLACK, }, ); + draw_handle.draw_text( + "Shop", + (win_height / 2) + 120, + (win_width / 4) + 100, + 60, + match hovering_shop_button { + true => Color::GREEN, + false => Color::BLACK, + }, + ); draw_handle.draw_text( "Quit", (win_height / 2) + 130, - (win_width / 4) + 100, + (win_width / 4) + 200, 60, match hovering_quit_button { true => Color::GREEN, @@ -72,7 +88,9 @@ impl Screen for MainMenuScreen { if mouse_clicked { if hovering_play_button { return Some(GameState::InGame); - } else if hovering_quit_button { + } else if hovering_shop_button { + return Some(GameState::InShop); + }else if hovering_quit_button { return Some(GameState::GameQuit); } } diff --git a/src/logic/pausemenu.rs b/src/logic/pausemenu.rs index 30c91f4..e89b0f1 100644 --- a/src/logic/pausemenu.rs +++ b/src/logic/pausemenu.rs @@ -157,7 +157,7 @@ impl Screen for PauseMenuScreen { }, ); draw_handle.draw_text( - "Quit", + "Menu", bottom_left_button_dimensions.x as i32 + 15, bottom_left_button_dimensions.y as i32 + 5, 30, @@ -182,7 +182,7 @@ impl Screen for PauseMenuScreen { // 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); + return Some(GameState::MainMenu); } else if mouse_over_bottom_right_button { return Some(game_core.last_state); } diff --git a/src/logic/shopscreen.rs b/src/logic/shopscreen.rs index 8ca2c6e..43e7ddb 100644 --- a/src/logic/shopscreen.rs +++ b/src/logic/shopscreen.rs @@ -144,7 +144,7 @@ impl Screen for ShopScreen { }, ); draw_handle.draw_text( - "Quit", + "Menu", bottom_left_button_dimensions.x as i32 + 15, bottom_left_button_dimensions.y as i32 + 5, 30, @@ -159,7 +159,7 @@ impl Screen for ShopScreen { }, ); draw_handle.draw_text( - "Close", + "Play", bottom_right_button_dimensions.x as i32 + 15, bottom_right_button_dimensions.y as i32 + 5, 30, @@ -169,9 +169,9 @@ impl Screen for ShopScreen { // 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); + return Some(GameState::MainMenu); } else if mouse_over_bottom_right_button { - return Some(game_core.last_state); + return Some(GameState::InGame); } } From 83aaa01133f88cf08190f74b72cb8b4f8c3c6eb4 Mon Sep 17 00:00:00 2001 From: Evan Pratten Date: Sun, 25 Apr 2021 10:18:23 -0400 Subject: [PATCH 09/27] button --- src/lib/utils/button.rs | 71 +++++++++++++++++++++++++++ src/lib/utils/mod.rs | 1 + src/logic/shopscreen.rs | 105 +++++----------------------------------- 3 files changed, 83 insertions(+), 94 deletions(-) create mode 100644 src/lib/utils/button.rs diff --git a/src/lib/utils/button.rs b/src/lib/utils/button.rs new file mode 100644 index 0000000..7a17239 --- /dev/null +++ b/src/lib/utils/button.rs @@ -0,0 +1,71 @@ +use raylib::prelude::*; + +pub struct OnScreenButton { + bounds: Rectangle, + text: String, + background: Color, + border: Color, + border_hover: Color, + font_px: i32, + draw_border: bool, +} + +impl OnScreenButton { + pub fn new( + text: String, + bounds: Rectangle, + background: Color, + border: Color, + border_hover: Color, + font_px: i32, + draw_border: bool, + ) -> Self { + Self { + bounds, + text, + background, + border, + border_hover, + font_px, + draw_border, + } + } + + pub fn is_hovered(&self, draw_handle: &RaylibDrawHandle) -> bool { + return self + .bounds + .check_collision_point_rec(draw_handle.get_mouse_position()); + } + + pub fn render(&self, draw_handle: &mut RaylibDrawHandle) { + // Draw the button background + draw_handle.draw_rectangle_rec(self.bounds, self.background); + + // Check mouse info + let is_being_hovered = self.is_hovered(draw_handle); + + // Render the border + if self.draw_border { + draw_handle.draw_rectangle_lines_ex( + self.bounds, + 3, + match is_being_hovered { + true => self.border_hover, + false => self.border, + }, + ); + } + + // Render the text + draw_handle.draw_text( + &self.text, + self.bounds.x as i32 + 10, + self.bounds.y as i32 + ((self.bounds.height as i32 - self.font_px) / 2), + self.font_px, + match is_being_hovered { + true => self.border_hover, + false => self.border, + }, + ) + } +} diff --git a/src/lib/utils/mod.rs b/src/lib/utils/mod.rs index baba8af..2f1353d 100644 --- a/src/lib/utils/mod.rs +++ b/src/lib/utils/mod.rs @@ -1,5 +1,6 @@ pub mod profiler; pub mod triangles; +pub mod button; pub fn calculate_linear_slide(playthrough_percent: f64) -> f64 { if playthrough_percent < 0.25 { diff --git a/src/logic/shopscreen.rs b/src/logic/shopscreen.rs index 43e7ddb..e1f37db 100644 --- a/src/logic/shopscreen.rs +++ b/src/logic/shopscreen.rs @@ -31,6 +31,16 @@ impl ShopScreen { } } + fn render_buy_section( + &mut self, + draw_handle: &mut RaylibDrawHandle, + _thread: &RaylibThread, + audio_system: &mut AudioPlayer, + game_core: &mut GameCore, + draw_bounds: Rectangle + ) { + } + // // Creates all the items // pub fn create_items(&mut self, screen_dimension: Vector2) { // // gets every item.. hacky @@ -80,100 +90,7 @@ impl Screen for ShopScreen { draw_handle.clear_background(Color::GRAY); // TODO: Maybe we can stick some art here? - // If escape is pressed again, return to the previous render state - if draw_handle.is_key_pressed(KeyboardKey::KEY_ESCAPE) { - return Some(game_core.last_state); - } - - // 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( - "SHOP", - (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( - "Menu", - 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( - "Play", - 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::MainMenu); - } else if mouse_over_bottom_right_button { - return Some(GameState::InGame); - } - } + return None; } From eff489d9ffa7ee9bd594e7fa0854dd68b129b293 Mon Sep 17 00:00:00 2001 From: Evan Pratten Date: Sun, 25 Apr 2021 10:33:11 -0400 Subject: [PATCH 10/27] Adding UI boxes --- src/logic/pausemenu.rs | 56 +++++++++++++-------------------- src/logic/shopscreen.rs | 68 ++++++++++++++++++++++++++++++++--------- 2 files changed, 74 insertions(+), 50 deletions(-) diff --git a/src/logic/pausemenu.rs b/src/logic/pausemenu.rs index e89b0f1..109d3f0 100644 --- a/src/logic/pausemenu.rs +++ b/src/logic/pausemenu.rs @@ -2,7 +2,7 @@ use raylib::prelude::*; use crate::{ gamecore::{GameCore, GameState}, - lib::wrappers::audio::player::AudioPlayer, + lib::{utils::button::OnScreenButton, wrappers::audio::player::AudioPlayer}, }; use super::screen::Screen; @@ -127,7 +127,8 @@ impl Screen for PauseMenuScreen { Color::BLACK, ); - // Close and quit buttons + // Bottom 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, @@ -141,49 +142,34 @@ impl Screen for PauseMenuScreen { 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( + let menu_button = OnScreenButton::new( + "Menu".to_string(), bottom_left_button_dimensions, - 3, - match mouse_over_bottom_left_button { - true => Color::GRAY, - false => Color::BLACK, - }, - ); - draw_handle.draw_text( - "Menu", - bottom_left_button_dimensions.x as i32 + 15, - bottom_left_button_dimensions.y as i32 + 5, - 30, + Color::WHITE, Color::BLACK, + Color::GRAY, + 30, + true, ); - draw_handle.draw_rectangle_lines_ex( + let close_button = OnScreenButton::new( + "Close".to_string(), 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::WHITE, Color::BLACK, + Color::GRAY, + 30, + true, ); + // Render both + menu_button.render(draw_handle); + close_button.render(draw_handle); + // Handle click actions on the buttons if draw_handle.is_mouse_button_pressed(MouseButton::MOUSE_LEFT_BUTTON) { - if mouse_over_bottom_left_button { + if menu_button.is_hovered(draw_handle) { return Some(GameState::MainMenu); - } else if mouse_over_bottom_right_button { + } else if menu_button.is_hovered(draw_handle) { return Some(game_core.last_state); } } diff --git a/src/logic/shopscreen.rs b/src/logic/shopscreen.rs index e1f37db..0b64f6a 100644 --- a/src/logic/shopscreen.rs +++ b/src/logic/shopscreen.rs @@ -9,16 +9,6 @@ use super::screen::Screen; const SCREEN_PANEL_SIZE: Vector2 = Vector2 { x: 300.0, y: 380.0 }; -pub struct Item { - x_pose: i32, - y_pose: i32, - width: i32, - height: i32, - cost: u8, - level: u8, - name: String, -} - #[derive(Debug, Default)] pub struct ShopScreen { // shop_items: Vec, @@ -31,14 +21,39 @@ impl ShopScreen { } } - fn render_buy_section( + fn render_shop( &mut self, draw_handle: &mut RaylibDrawHandle, _thread: &RaylibThread, audio_system: &mut AudioPlayer, game_core: &mut GameCore, - draw_bounds: Rectangle + bounds: Rectangle, + ) -> Option { + // Render background + draw_handle.draw_rectangle_rec(bounds, Color::WHITE); + draw_handle.draw_rectangle_lines_ex(bounds, 3, Color::BLACK); + + return None; + } + + fn render_stats( + &mut self, + draw_handle: &mut RaylibDrawHandle, + game_core: &mut GameCore, + bounds: Rectangle, ) { + // Render background + draw_handle.draw_rectangle_rec(bounds, Color::WHITE); + draw_handle.draw_rectangle_lines_ex(bounds, 3, Color::BLACK); + + // Coins + draw_handle.draw_text( + &format!("Fish: {}", game_core.player.coins.min(99)), + bounds.x as i32 + 5, + bounds.y as i32 + 5, + 20, + Color::BLACK, + ); } // // Creates all the items @@ -82,7 +97,7 @@ impl Screen for ShopScreen { fn render( &mut self, draw_handle: &mut RaylibDrawHandle, - _thread: &RaylibThread, + thread: &RaylibThread, audio_system: &mut AudioPlayer, game_core: &mut GameCore, ) -> Option { @@ -90,9 +105,32 @@ impl Screen for ShopScreen { 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(); - return None; + // Build a rect for the shop UI to sit inside + let shop_ui_bounds = Rectangle { + x: win_width as f32 - (win_width as f32 / 5.0), + y: 0.0, + width: win_width as f32 / 5.0, + height: win_height as f32, + }; + let stats_ui_bounds = Rectangle { + x: win_width as f32 - (win_width as f32 / 5.0) - 100.0, + y: 10.0, + width: 90.0, + height: 120.0, + }; + + // Render the shop UI + let next_state = + self.render_shop(draw_handle, thread, audio_system, game_core, shop_ui_bounds); + + // Render the stats UI + self.render_stats(draw_handle, game_core, stats_ui_bounds); + + return next_state; } } // pub fn render_shop( From fa7040f6267e0c1cdd43e0eb7a765971446e0019 Mon Sep 17 00:00:00 2001 From: Evan Pratten Date: Sun, 25 Apr 2021 10:38:41 -0400 Subject: [PATCH 11/27] Shop screen layout --- src/logic/shopscreen.rs | 21 +++++++++++++++------ 1 file changed, 15 insertions(+), 6 deletions(-) diff --git a/src/logic/shopscreen.rs b/src/logic/shopscreen.rs index 0b64f6a..7bae2d5 100644 --- a/src/logic/shopscreen.rs +++ b/src/logic/shopscreen.rs @@ -33,6 +33,15 @@ impl ShopScreen { draw_handle.draw_rectangle_rec(bounds, Color::WHITE); draw_handle.draw_rectangle_lines_ex(bounds, 3, Color::BLACK); + // Title + draw_handle.draw_text( + "SHOP", + bounds.x as i32 + 30, + bounds.y as i32 + 20, + 40, + Color::BLACK, + ); + return None; } @@ -111,15 +120,15 @@ impl Screen for ShopScreen { // Build a rect for the shop UI to sit inside let shop_ui_bounds = Rectangle { - x: win_width as f32 - (win_width as f32 / 5.0), - y: 0.0, - width: win_width as f32 / 5.0, - height: win_height as f32, + x: win_width as f32 - (win_width as f32 / 2.0), + y: 10.0, + width: (win_width as f32 / 2.0) - 10.0, + height: win_height as f32 - 20.0, }; let stats_ui_bounds = Rectangle { - x: win_width as f32 - (win_width as f32 / 5.0) - 100.0, + x: win_width as f32 - (win_width as f32 / 2.0) - 130.0, y: 10.0, - width: 90.0, + width: 120.0, height: 120.0, }; From 010ad7ce264866dd0f7b8825001008ae1b78a0b0 Mon Sep 17 00:00:00 2001 From: Evan Pratten Date: Sun, 25 Apr 2021 10:42:03 -0400 Subject: [PATCH 12/27] move shop file --- src/logic/mod.rs | 2 +- src/logic/{shopscreen.rs => shop/mod.rs} | 2 +- src/main.rs | 29 ++++++++++++++++-------- 3 files changed, 21 insertions(+), 12 deletions(-) rename src/logic/{shopscreen.rs => shop/mod.rs} (99%) diff --git a/src/logic/mod.rs b/src/logic/mod.rs index 19c7044..7b1e5eb 100644 --- a/src/logic/mod.rs +++ b/src/logic/mod.rs @@ -4,4 +4,4 @@ pub mod mainmenu; pub mod pausemenu; pub mod ingame; pub mod gameend; -pub mod shopscreen; \ No newline at end of file +pub mod shop; \ No newline at end of file diff --git a/src/logic/shopscreen.rs b/src/logic/shop/mod.rs similarity index 99% rename from src/logic/shopscreen.rs rename to src/logic/shop/mod.rs index 7bae2d5..5cf2458 100644 --- a/src/logic/shopscreen.rs +++ b/src/logic/shop/mod.rs @@ -36,7 +36,7 @@ impl ShopScreen { // Title draw_handle.draw_text( "SHOP", - bounds.x as i32 + 30, + bounds.x as i32 + (bounds.width / 2.0) as i32 - 50, bounds.y as i32 + 20, 40, Color::BLACK, diff --git a/src/main.rs b/src/main.rs index a98eed6..a8a260f 100644 --- a/src/main.rs +++ b/src/main.rs @@ -1,19 +1,22 @@ +mod entities; mod gamecore; +mod items; mod lib; mod logic; -mod resources; -mod player; -mod world; mod pallette; -mod entities; -mod items; +mod player; +mod resources; +mod world; 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, shopscreen::ShopScreen}; +use logic::{ + gameend::GameEndScreen, ingame::InGameScreen, loadingscreen::LoadingScreen, + mainmenu::MainMenuScreen, pausemenu::PauseMenuScreen, screen::Screen, shop::ShopScreen, +}; use raylib::prelude::*; -use world::{World, load_world_colliders}; +use world::{load_world_colliders, World}; // Game Launch Configuration const DEFAULT_WINDOW_DIMENSIONS: Vector2 = Vector2 { @@ -32,7 +35,8 @@ fn main() { .size( DEFAULT_WINDOW_DIMENSIONS.x as i32, DEFAULT_WINDOW_DIMENSIONS.y as i32, - ).msaa_4x() + ) + .msaa_4x() .title(WINDOW_TITLE) .build(); raylib.set_target_fps(MAX_FPS); @@ -41,8 +45,13 @@ fn main() { raylib.set_exit_key(None); // Load the world - let world_colliders = load_world_colliders("./assets/img/map/cave.json".to_string()).expect("Failed to load world colliders"); - let world = World::load_from_json("./assets/worlds/mainworld.json".to_string(), world_colliders).expect("Failed to load main world JSON"); + let world_colliders = load_world_colliders("./assets/img/map/cave.json".to_string()) + .expect("Failed to load world colliders"); + let world = World::load_from_json( + "./assets/worlds/mainworld.json".to_string(), + world_colliders, + ) + .expect("Failed to load main world JSON"); // Load the game progress let game_progress = GameProgress::try_from_file("./assets/savestate.json".to_string()); From 3a2caecfff1785c92dd6a48b8c903987def4e029 Mon Sep 17 00:00:00 2001 From: Evan Pratten Date: Sun, 25 Apr 2021 11:09:38 -0400 Subject: [PATCH 13/27] working on buy buttons --- src/items.rs | 34 ++++++++++++++----- src/lib/utils/button.rs | 2 +- src/logic/shop/itemui.rs | 62 +++++++++++++++++++++++++++++++++++ src/logic/shop/mainui.rs | 71 ++++++++++++++++++++++++++++++++++++++++ src/logic/shop/mod.rs | 53 ++++-------------------------- 5 files changed, 166 insertions(+), 56 deletions(-) create mode 100644 src/logic/shop/itemui.rs create mode 100644 src/logic/shop/mainui.rs diff --git a/src/items.rs b/src/items.rs index e52b92c..d157dbc 100644 --- a/src/items.rs +++ b/src/items.rs @@ -4,6 +4,7 @@ use serde::{Deserialize, Serialize}; pub struct StunGun { pub range: f32, pub duration: f64, + pub level: u8, } impl StunGun { @@ -11,64 +12,75 @@ impl StunGun { Self { range: 30.0, duration: 0.75, + level: 1, } } pub fn lvl2() -> Self { Self { range: 60.0, duration: 1.25, + level: 2, } } pub fn lvl3() -> Self { Self { range: 80.0, duration: 1.0, + level: 3, } } } #[derive(Debug, PartialEq, Clone, Serialize, Deserialize)] -pub struct AirBag{ - extra_oxygen: u32, +pub struct AirBag { + extra_oxygen: u32, + pub level: u8, } impl AirBag { pub fn lvl1() -> Self { Self { extra_oxygen: 15, + level: 1, } } pub fn lvl2() -> Self { Self { extra_oxygen: 30, + level: 2, } } - pub fn lvl3() -> Self { + pub fn lvl3() -> Self { Self { extra_oxygen: 45, + level: 3, } } } #[derive(Debug, PartialEq, Clone, Serialize, Deserialize)] pub struct Flashlight { - pub radius: f32 + pub radius: f32, + pub level: u8, } -impl Flashlight{ +impl Flashlight { pub fn lvl1() -> Self { Self { radius: 0.25, + level: 1, } } pub fn lvl2() -> Self { Self { radius: 0.5, + level: 2, } } - pub fn lvl3() -> Self { + pub fn lvl3() -> Self { Self { radius: 1.0, + level: 3, } } } @@ -76,22 +88,26 @@ impl Flashlight{ #[derive(Debug, PartialEq, Clone, Serialize, Deserialize)] pub struct Flippers { pub speed_increase: f32, + pub level: u8, } impl Flippers { pub fn lvl1() -> Self { Self { - speed_increase: 1.2 + speed_increase: 1.2, + level: 1, } } pub fn lvl2() -> Self { Self { - speed_increase: 1.5 + speed_increase: 1.5, + level: 2, } } pub fn lvl3() -> Self { Self { - speed_increase: 1.8 + speed_increase: 1.8, + level: 3, } } } diff --git a/src/lib/utils/button.rs b/src/lib/utils/button.rs index 7a17239..8960a30 100644 --- a/src/lib/utils/button.rs +++ b/src/lib/utils/button.rs @@ -62,7 +62,7 @@ impl OnScreenButton { self.bounds.x as i32 + 10, self.bounds.y as i32 + ((self.bounds.height as i32 - self.font_px) / 2), self.font_px, - match is_being_hovered { + match is_being_hovered && !self.draw_border { true => self.border_hover, false => self.border, }, diff --git a/src/logic/shop/itemui.rs b/src/logic/shop/itemui.rs new file mode 100644 index 0000000..bf251e8 --- /dev/null +++ b/src/logic/shop/itemui.rs @@ -0,0 +1,62 @@ +use crate::lib::utils::button::OnScreenButton; +use raylib::prelude::*; + +pub struct ShopItemUi { + name: String, + current_level: u8, + max_level: u8, + pub cost: u32, + buy_button_hovered: bool, +} + +impl ShopItemUi { + pub fn new(name: String, current_level: u8, max_level: u8, cost: u32) -> Self { + Self { + name, + current_level, + max_level, + cost, + buy_button_hovered: false, + } + } + + pub fn render( + &self, + draw_handle: &mut RaylibDrawHandle, + bounds: Rectangle, + can_be_bought: bool, + ) { + // Render the background box + draw_handle.draw_rectangle_rec(bounds, Color::WHITE); + draw_handle.draw_rectangle_lines_ex(bounds, 3, Color::BLACK); + + // Render the name + draw_handle.draw_text( + &format!("{}: {}/{}", self.name, self.current_level, self.max_level), + bounds.x as i32 + 10, + bounds.y as i32 + ((bounds.height as i32 - 20) / 2), + 20, + Color::BLACK, + ); + + // Render the buy button + let buy_button = OnScreenButton::new( + format!("Buy - {}f", self.cost), + Rectangle { + x: bounds.x + bounds.width - 150.0, + y: bounds.y + 5.0, + width: 145.0, + height: bounds.height - 10.0, + }, + match can_be_bought { + true => Color::WHITE, + false => Color::GRAY, + }, + Color::BLACK, + Color::GRAY, + 20, + true, + ); + buy_button.render(draw_handle); + } +} diff --git a/src/logic/shop/mainui.rs b/src/logic/shop/mainui.rs new file mode 100644 index 0000000..192f516 --- /dev/null +++ b/src/logic/shop/mainui.rs @@ -0,0 +1,71 @@ +use raylib::prelude::*; + +use crate::{ + gamecore::{GameCore, GameState}, + lib::wrappers::audio::player::AudioPlayer, +}; + +use super::itemui::ShopItemUi; + +pub fn render_shop( + draw_handle: &mut RaylibDrawHandle, + _thread: &RaylibThread, + audio_system: &mut AudioPlayer, + game_core: &mut GameCore, + bounds: Rectangle, +) -> Option { + // Render background + draw_handle.draw_rectangle_rec(bounds, Color::WHITE); + draw_handle.draw_rectangle_lines_ex(bounds, 3, Color::BLACK); + + // Title + draw_handle.draw_text( + "SHOP", + bounds.x as i32 + (bounds.width / 2.0) as i32 - 50, + bounds.y as i32 + 20, + 40, + Color::BLACK, + ); + + // Items + let stun_gun_buy_ui = ShopItemUi::new( + "Stun Gun".to_string(), + match &game_core.player.inventory.stun_gun { + Some(x) => x.level, + None => 0, + }, + 3, + 10, + ); + stun_gun_buy_ui.render( + draw_handle, + Rectangle { + x: bounds.x + 5.0, + y: bounds.y + 100.0, + width: bounds.width - 10.0, + height: 50.0, + }, + game_core.player.coins >= stun_gun_buy_ui.cost, + ); + + return None; +} + +pub fn render_stats( + draw_handle: &mut RaylibDrawHandle, + game_core: &mut GameCore, + bounds: Rectangle, +) { + // Render background + draw_handle.draw_rectangle_rec(bounds, Color::WHITE); + draw_handle.draw_rectangle_lines_ex(bounds, 3, Color::BLACK); + + // Coins + draw_handle.draw_text( + &format!("Fish: {}", game_core.player.coins.min(99)), + bounds.x as i32 + 5, + bounds.y as i32 + 5, + 20, + Color::BLACK, + ); +} diff --git a/src/logic/shop/mod.rs b/src/logic/shop/mod.rs index 5cf2458..4a07865 100644 --- a/src/logic/shop/mod.rs +++ b/src/logic/shop/mod.rs @@ -1,3 +1,6 @@ +mod mainui; +mod itemui; + use raylib::prelude::*; use crate::{ @@ -5,6 +8,8 @@ use crate::{ lib::wrappers::audio::player::AudioPlayer, }; +use self::mainui::{render_shop, render_stats}; + use super::screen::Screen; const SCREEN_PANEL_SIZE: Vector2 = Vector2 { x: 300.0, y: 380.0 }; @@ -21,50 +26,6 @@ impl ShopScreen { } } - fn render_shop( - &mut self, - draw_handle: &mut RaylibDrawHandle, - _thread: &RaylibThread, - audio_system: &mut AudioPlayer, - game_core: &mut GameCore, - bounds: Rectangle, - ) -> Option { - // Render background - draw_handle.draw_rectangle_rec(bounds, Color::WHITE); - draw_handle.draw_rectangle_lines_ex(bounds, 3, Color::BLACK); - - // Title - draw_handle.draw_text( - "SHOP", - bounds.x as i32 + (bounds.width / 2.0) as i32 - 50, - bounds.y as i32 + 20, - 40, - Color::BLACK, - ); - - return None; - } - - fn render_stats( - &mut self, - draw_handle: &mut RaylibDrawHandle, - game_core: &mut GameCore, - bounds: Rectangle, - ) { - // Render background - draw_handle.draw_rectangle_rec(bounds, Color::WHITE); - draw_handle.draw_rectangle_lines_ex(bounds, 3, Color::BLACK); - - // Coins - draw_handle.draw_text( - &format!("Fish: {}", game_core.player.coins.min(99)), - bounds.x as i32 + 5, - bounds.y as i32 + 5, - 20, - Color::BLACK, - ); - } - // // Creates all the items // pub fn create_items(&mut self, screen_dimension: Vector2) { // // gets every item.. hacky @@ -134,10 +95,10 @@ impl Screen for ShopScreen { // Render the shop UI let next_state = - self.render_shop(draw_handle, thread, audio_system, game_core, shop_ui_bounds); + render_shop(draw_handle, thread, audio_system, game_core, shop_ui_bounds); // Render the stats UI - self.render_stats(draw_handle, game_core, stats_ui_bounds); + render_stats(draw_handle, game_core, stats_ui_bounds); return next_state; } From ad5d6bc1abf9b46c0165623b0b305d01368e53c1 Mon Sep 17 00:00:00 2001 From: Evan Pratten Date: Sun, 25 Apr 2021 11:12:37 -0400 Subject: [PATCH 14/27] background image --- src/logic/shop/mod.rs | 6 ++++-- src/resources.rs | 10 +++++++++- 2 files changed, 13 insertions(+), 3 deletions(-) diff --git a/src/logic/shop/mod.rs b/src/logic/shop/mod.rs index 4a07865..96f44f7 100644 --- a/src/logic/shop/mod.rs +++ b/src/logic/shop/mod.rs @@ -72,8 +72,10 @@ impl Screen for ShopScreen { 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? + + + // Render the background + draw_handle.draw_texture(&game_core.resources.shop_background, 0, 0, Color::WHITE); // Window dimensions let win_height = draw_handle.get_screen_height(); diff --git a/src/resources.rs b/src/resources.rs index 03cd78c..4dfe7c7 100644 --- a/src/resources.rs +++ b/src/resources.rs @@ -26,7 +26,11 @@ pub struct GlobalResources { pub jellyfish_animation_attack: FrameAnimationWrapper, // Darkness layer - pub darkness_overlay: Texture2D + pub darkness_overlay: Texture2D, + + // Shop & items + pub shop_background: Texture2D, + } impl GlobalResources { @@ -102,6 +106,10 @@ impl GlobalResources { &thread, &Image::load_image("./assets/img/map/darkness.png")?, )?, + shop_background: raylib.load_texture_from_image( + &thread, + &Image::load_image("./assets/img/map/shopHighRes.png")?, + )?, }) } } From 198edc6556ce9cf4ff1674c59c56b6c398354e80 Mon Sep 17 00:00:00 2001 From: Evan Pratten Date: Sun, 25 Apr 2021 11:17:48 -0400 Subject: [PATCH 15/27] more items --- src/logic/shop/mainui.rs | 65 +++++++++++++++++++++++++++++++++++++++- src/logic/shop/mod.rs | 2 +- 2 files changed, 65 insertions(+), 2 deletions(-) diff --git a/src/logic/shop/mainui.rs b/src/logic/shop/mainui.rs index 192f516..8de60aa 100644 --- a/src/logic/shop/mainui.rs +++ b/src/logic/shop/mainui.rs @@ -27,7 +27,7 @@ pub fn render_shop( Color::BLACK, ); - // Items + // Stun Gun let stun_gun_buy_ui = ShopItemUi::new( "Stun Gun".to_string(), match &game_core.player.inventory.stun_gun { @@ -48,6 +48,69 @@ pub fn render_shop( game_core.player.coins >= stun_gun_buy_ui.cost, ); + // Flippers + let flippers_buy_ui = ShopItemUi::new( + "Flippers".to_string(), + match &game_core.player.inventory.flippers { + Some(x) => x.level, + None => 0, + }, + 3, + 10, + ); + flippers_buy_ui.render( + draw_handle, + Rectangle { + x: bounds.x + 5.0, + y: bounds.y + 160.0, + width: bounds.width - 10.0, + height: 50.0, + }, + game_core.player.coins >= flippers_buy_ui.cost, + ); + + // Flashlight + let flashlight_buy_ui = ShopItemUi::new( + "Flashlight".to_string(), + match &game_core.player.inventory.flashlight { + Some(x) => x.level, + None => 0, + }, + 3, + 10, + ); + flashlight_buy_ui.render( + draw_handle, + Rectangle { + x: bounds.x + 5.0, + y: bounds.y + 220.0, + width: bounds.width - 10.0, + height: 50.0, + }, + game_core.player.coins >= flashlight_buy_ui.cost, + ); + + // Air Bag + let air_bag_buy_ui = ShopItemUi::new( + "Bag of Air".to_string(), + match &game_core.player.inventory.air_bag { + Some(x) => x.level, + None => 0, + }, + 3, + 10, + ); + air_bag_buy_ui.render( + draw_handle, + Rectangle { + x: bounds.x + 5.0, + y: bounds.y + 280.0, + width: bounds.width - 10.0, + height: 50.0, + }, + game_core.player.coins >= air_bag_buy_ui.cost, + ); + return None; } diff --git a/src/logic/shop/mod.rs b/src/logic/shop/mod.rs index 96f44f7..d025468 100644 --- a/src/logic/shop/mod.rs +++ b/src/logic/shop/mod.rs @@ -92,7 +92,7 @@ impl Screen for ShopScreen { x: win_width as f32 - (win_width as f32 / 2.0) - 130.0, y: 10.0, width: 120.0, - height: 120.0, + height: 30.0, }; // Render the shop UI From 0d862760bdcd4156d98a5efbc2b226f9a2bb88fe Mon Sep 17 00:00:00 2001 From: Evan Pratten Date: Sun, 25 Apr 2021 11:26:25 -0400 Subject: [PATCH 16/27] item trait --- src/items.rs | 96 ++++++++++++++++++++++++++++++++++++++++++ src/logic/shop/item.rs | 16 +++++++ src/logic/shop/mod.rs | 1 + 3 files changed, 113 insertions(+) create mode 100644 src/logic/shop/item.rs diff --git a/src/items.rs b/src/items.rs index d157dbc..ee04336 100644 --- a/src/items.rs +++ b/src/items.rs @@ -1,10 +1,19 @@ +use raylib::texture::Texture2D; use serde::{Deserialize, Serialize}; +pub trait ItemBase { + fn get_cost(&self) -> u32; + fn get_name(&self) -> String; + fn get_description(&self) -> String; + fn get_texture(&self) -> &Texture2D; +} + #[derive(Debug, PartialEq, Clone, Serialize, Deserialize)] pub struct StunGun { pub range: f32, pub duration: f64, pub level: u8, + cost: u32, } impl StunGun { @@ -13,6 +22,7 @@ impl StunGun { range: 30.0, duration: 0.75, level: 1, + cost: 30, } } pub fn lvl2() -> Self { @@ -20,6 +30,7 @@ impl StunGun { range: 60.0, duration: 1.25, level: 2, + cost: 40, } } pub fn lvl3() -> Self { @@ -27,14 +38,34 @@ impl StunGun { range: 80.0, duration: 1.0, level: 3, + cost: 50, } } } +impl ItemBase for StunGun { + fn get_cost(&self) -> u32 { + self.cost + } + + fn get_name(&self) -> String { + return "Stun Gun".to_string(); + } + + fn get_description(&self) -> String { + return "Stun your enemies! Just don't point it at yourself.".to_string(); + } + + fn get_texture(&self) -> &Texture2D { + todo!() + } +} + #[derive(Debug, PartialEq, Clone, Serialize, Deserialize)] pub struct AirBag { extra_oxygen: u32, pub level: u8, + cost: u32, } impl AirBag { @@ -42,26 +73,48 @@ impl AirBag { Self { extra_oxygen: 15, level: 1, + cost: 30, } } pub fn lvl2() -> Self { Self { extra_oxygen: 30, level: 2, + cost: 40, } } pub fn lvl3() -> Self { Self { extra_oxygen: 45, level: 3, + cost: 50, } } } +impl ItemBase for AirBag { + fn get_cost(&self) -> u32 { + self.cost + } + + fn get_name(&self) -> String { + return "Bag of Air".to_string(); + } + + fn get_description(&self) -> String { + return "Its.. a bag. Filled with air. Duh".to_string(); + } + + fn get_texture(&self) -> &Texture2D { + todo!() + } +} + #[derive(Debug, PartialEq, Clone, Serialize, Deserialize)] pub struct Flashlight { pub radius: f32, pub level: u8, + cost: u32, } impl Flashlight { @@ -69,26 +122,48 @@ impl Flashlight { Self { radius: 0.25, level: 1, + cost: 40, } } pub fn lvl2() -> Self { Self { radius: 0.5, level: 2, + cost: 50, } } pub fn lvl3() -> Self { Self { radius: 1.0, level: 3, + cost: 60, } } } +impl ItemBase for Flashlight { + fn get_cost(&self) -> u32 { + self.cost + } + + fn get_name(&self) -> String { + return "Flashlight".to_string(); + } + + fn get_description(&self) -> String { + return "See better for longer".to_string(); + } + + fn get_texture(&self) -> &Texture2D { + todo!() + } +} + #[derive(Debug, PartialEq, Clone, Serialize, Deserialize)] pub struct Flippers { pub speed_increase: f32, pub level: u8, + cost: u32, } impl Flippers { @@ -96,18 +171,39 @@ impl Flippers { Self { speed_increase: 1.2, level: 1, + cost: 30, } } pub fn lvl2() -> Self { Self { speed_increase: 1.5, level: 2, + cost: 40, } } pub fn lvl3() -> Self { Self { speed_increase: 1.8, level: 3, + cost: 50, } } } + +impl ItemBase for Flippers { + fn get_cost(&self) -> u32 { + self.cost + } + + fn get_name(&self) -> String { + return "Flippers".to_string(); + } + + fn get_description(&self) -> String { + return "Swim faster, and look stupid at the same time!".to_string(); + } + + fn get_texture(&self) -> &Texture2D { + todo!() + } +} diff --git a/src/logic/shop/item.rs b/src/logic/shop/item.rs new file mode 100644 index 0000000..6af28cb --- /dev/null +++ b/src/logic/shop/item.rs @@ -0,0 +1,16 @@ +use raylib::prelude::*; + +use super::itemui::ShopItemUi; + +// pub struct ShopItemWrapper { +// bounds: Rectangle, +// ui: ShopItemUi +// } + +// impl ShopItemWrapper { + +// pub fn new(name: String, from_inventory: Option, first_item_bounds: Rectangle, index: u8){ + +// } + +// } \ No newline at end of file diff --git a/src/logic/shop/mod.rs b/src/logic/shop/mod.rs index d025468..0013b5a 100644 --- a/src/logic/shop/mod.rs +++ b/src/logic/shop/mod.rs @@ -1,5 +1,6 @@ mod mainui; mod itemui; +mod item; use raylib::prelude::*; From 5bb8261119a729ca8e79b83dce58b68ac379b7f0 Mon Sep 17 00:00:00 2001 From: Evan Pratten Date: Sun, 25 Apr 2021 11:37:53 -0400 Subject: [PATCH 17/27] Item wrapper --- src/items.rs | 13 +++++++++ src/logic/shop/item.rs | 66 +++++++++++++++++++++++++++++++++++++----- src/world.rs | 10 +++---- 3 files changed, 76 insertions(+), 13 deletions(-) diff --git a/src/items.rs b/src/items.rs index ee04336..66638b3 100644 --- a/src/items.rs +++ b/src/items.rs @@ -3,6 +3,7 @@ use serde::{Deserialize, Serialize}; pub trait ItemBase { fn get_cost(&self) -> u32; + fn get_level(&self) -> u8; fn get_name(&self) -> String; fn get_description(&self) -> String; fn get_texture(&self) -> &Texture2D; @@ -59,6 +60,9 @@ impl ItemBase for StunGun { fn get_texture(&self) -> &Texture2D { todo!() } + fn get_level(&self) -> u8 { + self.level + } } #[derive(Debug, PartialEq, Clone, Serialize, Deserialize)] @@ -108,6 +112,9 @@ impl ItemBase for AirBag { fn get_texture(&self) -> &Texture2D { todo!() } + fn get_level(&self) -> u8 { + self.level + } } #[derive(Debug, PartialEq, Clone, Serialize, Deserialize)] @@ -157,6 +164,9 @@ impl ItemBase for Flashlight { fn get_texture(&self) -> &Texture2D { todo!() } + fn get_level(&self) -> u8 { + self.level + } } #[derive(Debug, PartialEq, Clone, Serialize, Deserialize)] @@ -206,4 +216,7 @@ impl ItemBase for Flippers { fn get_texture(&self) -> &Texture2D { todo!() } + fn get_level(&self) -> u8 { + self.level + } } diff --git a/src/logic/shop/item.rs b/src/logic/shop/item.rs index 6af28cb..1f15335 100644 --- a/src/logic/shop/item.rs +++ b/src/logic/shop/item.rs @@ -1,16 +1,66 @@ +use std::marker::PhantomData; + use raylib::prelude::*; +use crate::{items::ItemBase, player::Player, world::World}; + use super::itemui::ShopItemUi; -// pub struct ShopItemWrapper { -// bounds: Rectangle, -// ui: ShopItemUi -// } +pub struct ShopItemWrapper { + bounds: Rectangle, + ui: ShopItemUi, + item: T, +} -// impl ShopItemWrapper { +impl ShopItemWrapper { + pub fn new( + name: String, + item: T, + from_inventory: &Option, + first_item_bounds: Rectangle, + index: u8, + player: &Player, + ) -> Self { + // Build new bounds for the UI row + let new_bounds = Rectangle { + x: first_item_bounds.x, + y: first_item_bounds.y + (first_item_bounds.height + 5.0 * index as f32), + width: first_item_bounds.width, + height: first_item_bounds.height, + }; -// pub fn new(name: String, from_inventory: Option, first_item_bounds: Rectangle, index: u8){ + Self { + bounds: new_bounds, + ui: ShopItemUi::new( + name, + match from_inventory { + Some(x) => x.get_level(), + None => 0, + }, + 3, + item.get_cost(), + ), + item, + } + } -// } + pub fn get_item(&self) -> &T { + &self.item + } -// } \ No newline at end of file + pub fn can_player_afford(&self, player: &Player) -> bool { + return player.coins >= self.item.get_cost(); + } + + pub fn purchase(&self, player: &mut Player) -> T { + // Take the currency from the player + player.coins -= self.item.get_cost(); + + // Return a clone of the item + return self.item.clone(); + } + + pub fn render(&self, draw_handle: &mut RaylibDrawHandle, player: &Player) { + self.ui.render(draw_handle, self.bounds, self.can_player_afford(player)); + } +} diff --git a/src/world.rs b/src/world.rs index b4237c4..db722b3 100644 --- a/src/world.rs +++ b/src/world.rs @@ -52,11 +52,11 @@ impl World { Ok(result) } - pub fn spend_coins(&mut self, count: usize) { - for _ in 0..count { - self.fish.pop(); - } - } + // 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() { From 289befc8024de3c231c7ec295494230f4b787360 Mon Sep 17 00:00:00 2001 From: Evan Pratten Date: Sun, 25 Apr 2021 11:45:54 -0400 Subject: [PATCH 18/27] much improved item buy button rendering --- src/logic/shop/item.rs | 8 +-- src/logic/shop/mainui.rs | 140 +++++++++++++++++---------------------- 2 files changed, 62 insertions(+), 86 deletions(-) diff --git a/src/logic/shop/item.rs b/src/logic/shop/item.rs index 1f15335..6d5eaaf 100644 --- a/src/logic/shop/item.rs +++ b/src/logic/shop/item.rs @@ -14,17 +14,15 @@ pub struct ShopItemWrapper { impl ShopItemWrapper { pub fn new( - name: String, item: T, from_inventory: &Option, first_item_bounds: Rectangle, - index: u8, - player: &Player, + index: u8 ) -> Self { // Build new bounds for the UI row let new_bounds = Rectangle { x: first_item_bounds.x, - y: first_item_bounds.y + (first_item_bounds.height + 5.0 * index as f32), + y: first_item_bounds.y + ((first_item_bounds.height + 5.0) * index as f32), width: first_item_bounds.width, height: first_item_bounds.height, }; @@ -32,7 +30,7 @@ impl ShopItemWrapper { Self { bounds: new_bounds, ui: ShopItemUi::new( - name, + item.get_name(), match from_inventory { Some(x) => x.get_level(), None => 0, diff --git a/src/logic/shop/mainui.rs b/src/logic/shop/mainui.rs index 8de60aa..e8ba083 100644 --- a/src/logic/shop/mainui.rs +++ b/src/logic/shop/mainui.rs @@ -1,11 +1,8 @@ use raylib::prelude::*; -use crate::{ - gamecore::{GameCore, GameState}, - lib::wrappers::audio::player::AudioPlayer, -}; +use crate::{gamecore::{GameCore, GameState}, items::{AirBag, Flashlight, Flippers, ItemBase, StunGun}, lib::wrappers::audio::player::AudioPlayer}; -use super::itemui::ShopItemUi; +use super::{item::ShopItemWrapper, itemui::ShopItemUi}; pub fn render_shop( draw_handle: &mut RaylibDrawHandle, @@ -27,90 +24,71 @@ pub fn render_shop( Color::BLACK, ); - // Stun Gun - let stun_gun_buy_ui = ShopItemUi::new( - "Stun Gun".to_string(), + // Bounds for use in item row sizing + let first_bounds = Rectangle { + x: bounds.x + 5.0, + y: bounds.y + 100.0, + width: bounds.width - 10.0, + height: 50.0, + }; + + // Create items + let stun_gun_buy_ui = ShopItemWrapper::new( match &game_core.player.inventory.stun_gun { - Some(x) => x.level, - None => 0, + Some(x) => match x.get_level() { + 1 => StunGun::lvl2(), + _ => StunGun::lvl3(), + }, + None => StunGun::lvl1(), }, - 3, - 10, + &game_core.player.inventory.stun_gun, + first_bounds, + 0, ); - stun_gun_buy_ui.render( - draw_handle, - Rectangle { - x: bounds.x + 5.0, - y: bounds.y + 100.0, - width: bounds.width - 10.0, - height: 50.0, - }, - game_core.player.coins >= stun_gun_buy_ui.cost, - ); - - // Flippers - let flippers_buy_ui = ShopItemUi::new( - "Flippers".to_string(), - match &game_core.player.inventory.flippers { - Some(x) => x.level, - None => 0, - }, - 3, - 10, - ); - flippers_buy_ui.render( - draw_handle, - Rectangle { - x: bounds.x + 5.0, - y: bounds.y + 160.0, - width: bounds.width - 10.0, - height: 50.0, - }, - game_core.player.coins >= flippers_buy_ui.cost, - ); - - // Flashlight - let flashlight_buy_ui = ShopItemUi::new( - "Flashlight".to_string(), - match &game_core.player.inventory.flashlight { - Some(x) => x.level, - None => 0, - }, - 3, - 10, - ); - flashlight_buy_ui.render( - draw_handle, - Rectangle { - x: bounds.x + 5.0, - y: bounds.y + 220.0, - width: bounds.width - 10.0, - height: 50.0, - }, - game_core.player.coins >= flashlight_buy_ui.cost, - ); - - // Air Bag - let air_bag_buy_ui = ShopItemUi::new( - "Bag of Air".to_string(), + let air_bag_buy_ui = ShopItemWrapper::new( match &game_core.player.inventory.air_bag { - Some(x) => x.level, - None => 0, + Some(x) => match x.get_level() { + 1 => AirBag::lvl2(), + _ => AirBag::lvl3(), + }, + None => AirBag::lvl1(), }, + &game_core.player.inventory.air_bag, + first_bounds, + 1, + ); + let flashlight_buy_ui = ShopItemWrapper::new( + match &game_core.player.inventory.flashlight { + Some(x) => match x.get_level() { + 1 => Flashlight::lvl2(), + _ => Flashlight::lvl3(), + }, + None => Flashlight::lvl1(), + }, + &game_core.player.inventory.flashlight, + first_bounds, + 2, + ); + let flippers_buy_ui = ShopItemWrapper::new( + match &game_core.player.inventory.flippers { + Some(x) => match x.get_level() { + 1 => Flippers::lvl2(), + _ => Flippers::lvl3(), + }, + None => Flippers::lvl1(), + }, + &game_core.player.inventory.flippers, + first_bounds, 3, - 10, - ); - air_bag_buy_ui.render( - draw_handle, - Rectangle { - x: bounds.x + 5.0, - y: bounds.y + 280.0, - width: bounds.width - 10.0, - height: 50.0, - }, - game_core.player.coins >= air_bag_buy_ui.cost, ); + // Render items + stun_gun_buy_ui.render(draw_handle, &game_core.player); + air_bag_buy_ui.render(draw_handle, &game_core.player); + flashlight_buy_ui.render(draw_handle, &game_core.player); + flippers_buy_ui.render(draw_handle, &game_core.player); + + return None; } From 82d5b336c6af7894847e2803d94c489fa58ccb16 Mon Sep 17 00:00:00 2001 From: Evan Pratten Date: Sun, 25 Apr 2021 11:51:38 -0400 Subject: [PATCH 19/27] Full game navigation --- src/logic/pausemenu.rs | 2 +- src/logic/shop/mainui.rs | 48 +++++++++++++++++++++++++++++++++++++++- 2 files changed, 48 insertions(+), 2 deletions(-) diff --git a/src/logic/pausemenu.rs b/src/logic/pausemenu.rs index 109d3f0..48a3d3d 100644 --- a/src/logic/pausemenu.rs +++ b/src/logic/pausemenu.rs @@ -169,7 +169,7 @@ impl Screen for PauseMenuScreen { if draw_handle.is_mouse_button_pressed(MouseButton::MOUSE_LEFT_BUTTON) { if menu_button.is_hovered(draw_handle) { return Some(GameState::MainMenu); - } else if menu_button.is_hovered(draw_handle) { + } else if close_button.is_hovered(draw_handle) { return Some(game_core.last_state); } } diff --git a/src/logic/shop/mainui.rs b/src/logic/shop/mainui.rs index e8ba083..680d139 100644 --- a/src/logic/shop/mainui.rs +++ b/src/logic/shop/mainui.rs @@ -1,6 +1,6 @@ use raylib::prelude::*; -use crate::{gamecore::{GameCore, GameState}, items::{AirBag, Flashlight, Flippers, ItemBase, StunGun}, lib::wrappers::audio::player::AudioPlayer}; +use crate::{gamecore::{GameCore, GameState}, items::{AirBag, Flashlight, Flippers, ItemBase, StunGun}, lib::{utils::button::OnScreenButton, wrappers::audio::player::AudioPlayer}}; use super::{item::ShopItemWrapper, itemui::ShopItemUi}; @@ -89,6 +89,52 @@ pub fn render_shop( flippers_buy_ui.render(draw_handle, &game_core.player); + // Handle exit buttons + let bottom_left_button_dimensions = Rectangle { + x: bounds.x + 5.0, + y: bounds.y + bounds.height - 50.0, + width: (bounds.width / 2.0) - 15.0, + height: 40.0, + }; + let bottom_right_button_dimensions = Rectangle { + x: (bounds.x + bottom_left_button_dimensions.width ) + 15.0, + y: bottom_left_button_dimensions.y, + width: bottom_left_button_dimensions.width, + height: bottom_left_button_dimensions.height, + }; + + let menu_button = OnScreenButton::new( + "Menu".to_string(), + bottom_left_button_dimensions, + Color::WHITE, + Color::BLACK, + Color::GRAY, + 30, + true, + ); + let play_button = OnScreenButton::new( + "Play".to_string(), + bottom_right_button_dimensions, + Color::WHITE, + Color::BLACK, + Color::GRAY, + 30, + true, + ); + + // Render both + menu_button.render(draw_handle); + play_button.render(draw_handle); + + // Handle click actions on the buttons + if draw_handle.is_mouse_button_pressed(MouseButton::MOUSE_LEFT_BUTTON) { + if menu_button.is_hovered(draw_handle) { + return Some(GameState::MainMenu); + } else if play_button.is_hovered(draw_handle) { + return Some(GameState::InGame); + } + } + return None; } From a88cae2875afcec9ba8d1074c8a40d30bbca0284 Mon Sep 17 00:00:00 2001 From: Evan Pratten Date: Sun, 25 Apr 2021 11:55:35 -0400 Subject: [PATCH 20/27] buy button --- src/logic/shop/item.rs | 4 ++++ src/logic/shop/itemui.rs | 2 +- src/logic/shop/mainui.rs | 5 +++++ 3 files changed, 10 insertions(+), 1 deletion(-) diff --git a/src/logic/shop/item.rs b/src/logic/shop/item.rs index 6d5eaaf..6afdb8b 100644 --- a/src/logic/shop/item.rs +++ b/src/logic/shop/item.rs @@ -58,6 +58,10 @@ impl ShopItemWrapper { return self.item.clone(); } + pub fn user_clicked_buy(&self, draw_handle: &mut RaylibDrawHandle) -> bool { + return self.ui.buy_button_hovered && draw_handle.is_mouse_button_pressed(MouseButton::MOUSE_LEFT_BUTTON); + } + pub fn render(&self, draw_handle: &mut RaylibDrawHandle, player: &Player) { self.ui.render(draw_handle, self.bounds, self.can_player_afford(player)); } diff --git a/src/logic/shop/itemui.rs b/src/logic/shop/itemui.rs index bf251e8..c6b98f2 100644 --- a/src/logic/shop/itemui.rs +++ b/src/logic/shop/itemui.rs @@ -6,7 +6,7 @@ pub struct ShopItemUi { current_level: u8, max_level: u8, pub cost: u32, - buy_button_hovered: bool, + pub buy_button_hovered: bool, } impl ShopItemUi { diff --git a/src/logic/shop/mainui.rs b/src/logic/shop/mainui.rs index 680d139..39cdf76 100644 --- a/src/logic/shop/mainui.rs +++ b/src/logic/shop/mainui.rs @@ -88,6 +88,11 @@ pub fn render_shop( flashlight_buy_ui.render(draw_handle, &game_core.player); flippers_buy_ui.render(draw_handle, &game_core.player); + // Handle buying items + if stun_gun_buy_ui.can_player_afford(&game_core.player) && stun_gun_buy_ui.user_clicked_buy(draw_handle) { + stun_gun_buy_ui.purchase(&mut game_core.player); + } + // Handle exit buttons let bottom_left_button_dimensions = Rectangle { From 7dac0bbfd1d64afb596c7fe24098a15bb1592958 Mon Sep 17 00:00:00 2001 From: Evan Pratten Date: Sun, 25 Apr 2021 12:23:46 -0400 Subject: [PATCH 21/27] user can now buy items! --- src/logic/shop/item.rs | 2 +- src/logic/shop/itemui.rs | 3 ++- src/logic/shop/mainui.rs | 23 ++++++++++++++++++----- src/player.rs | 5 +++-- 4 files changed, 24 insertions(+), 9 deletions(-) diff --git a/src/logic/shop/item.rs b/src/logic/shop/item.rs index 6afdb8b..1bddfd2 100644 --- a/src/logic/shop/item.rs +++ b/src/logic/shop/item.rs @@ -62,7 +62,7 @@ impl ShopItemWrapper { return self.ui.buy_button_hovered && draw_handle.is_mouse_button_pressed(MouseButton::MOUSE_LEFT_BUTTON); } - pub fn render(&self, draw_handle: &mut RaylibDrawHandle, player: &Player) { + pub fn render(&mut self, draw_handle: &mut RaylibDrawHandle, player: &Player) { self.ui.render(draw_handle, self.bounds, self.can_player_afford(player)); } } diff --git a/src/logic/shop/itemui.rs b/src/logic/shop/itemui.rs index c6b98f2..0084e84 100644 --- a/src/logic/shop/itemui.rs +++ b/src/logic/shop/itemui.rs @@ -21,7 +21,7 @@ impl ShopItemUi { } pub fn render( - &self, + &mut self, draw_handle: &mut RaylibDrawHandle, bounds: Rectangle, can_be_bought: bool, @@ -58,5 +58,6 @@ impl ShopItemUi { true, ); buy_button.render(draw_handle); + self.buy_button_hovered = buy_button.is_hovered(draw_handle); } } diff --git a/src/logic/shop/mainui.rs b/src/logic/shop/mainui.rs index 39cdf76..d6d02a7 100644 --- a/src/logic/shop/mainui.rs +++ b/src/logic/shop/mainui.rs @@ -33,7 +33,7 @@ pub fn render_shop( }; // Create items - let stun_gun_buy_ui = ShopItemWrapper::new( + let mut stun_gun_buy_ui = ShopItemWrapper::new( match &game_core.player.inventory.stun_gun { Some(x) => match x.get_level() { 1 => StunGun::lvl2(), @@ -45,7 +45,7 @@ pub fn render_shop( first_bounds, 0, ); - let air_bag_buy_ui = ShopItemWrapper::new( + let mut air_bag_buy_ui = ShopItemWrapper::new( match &game_core.player.inventory.air_bag { Some(x) => match x.get_level() { 1 => AirBag::lvl2(), @@ -57,7 +57,7 @@ pub fn render_shop( first_bounds, 1, ); - let flashlight_buy_ui = ShopItemWrapper::new( + let mut flashlight_buy_ui = ShopItemWrapper::new( match &game_core.player.inventory.flashlight { Some(x) => match x.get_level() { 1 => Flashlight::lvl2(), @@ -69,7 +69,7 @@ pub fn render_shop( first_bounds, 2, ); - let flippers_buy_ui = ShopItemWrapper::new( + let mut flippers_buy_ui = ShopItemWrapper::new( match &game_core.player.inventory.flippers { Some(x) => match x.get_level() { 1 => Flippers::lvl2(), @@ -90,7 +90,20 @@ pub fn render_shop( // Handle buying items if stun_gun_buy_ui.can_player_afford(&game_core.player) && stun_gun_buy_ui.user_clicked_buy(draw_handle) { - stun_gun_buy_ui.purchase(&mut game_core.player); + let item = stun_gun_buy_ui.purchase(&mut game_core.player); + game_core.player.inventory.stun_gun = Some(item); + } + if air_bag_buy_ui.can_player_afford(&game_core.player) && air_bag_buy_ui.user_clicked_buy(draw_handle) { + let item = air_bag_buy_ui.purchase(&mut game_core.player); + game_core.player.inventory.air_bag = Some(item); + } + if flashlight_buy_ui.can_player_afford(&game_core.player) && flashlight_buy_ui.user_clicked_buy(draw_handle) { + let item = flashlight_buy_ui.purchase(&mut game_core.player); + game_core.player.inventory.flashlight = Some(item); + } + if flippers_buy_ui.can_player_afford(&game_core.player) && flippers_buy_ui.user_clicked_buy(draw_handle) { + let item = flippers_buy_ui.purchase(&mut game_core.player); + game_core.player.inventory.flippers = Some(item); } diff --git a/src/player.rs b/src/player.rs index e7f367e..ea81463 100644 --- a/src/player.rs +++ b/src/player.rs @@ -24,8 +24,8 @@ pub struct PlayerInventory { impl PlayerInventory { pub fn new() -> Self { Self { - stun_gun: Some(StunGun::lvl1()), //TMP - flashlight: Some(Flashlight::lvl1()), //TMP + // stun_gun: Some(StunGun::lvl1()), //TMP + // flashlight: Some(Flashlight::lvl1()), //TMP ..Default::default() } } @@ -58,6 +58,7 @@ impl Player { radius: 4.5, position: spawn.clone(), inventory: PlayerInventory::new(), + coins: 50, //TMP ..Default::default() } } From 7979295f7aed963c8f64fb2d8f34672b6b0ed896 Mon Sep 17 00:00:00 2001 From: Evan Pratten Date: Sun, 25 Apr 2021 12:39:10 -0400 Subject: [PATCH 22/27] Add world resetting and state saving --- src/gamecore.rs | 16 ++++++++++++++++ src/logic/mainmenu.rs | 4 ++++ src/logic/shop/mainui.rs | 11 +++++++++++ src/main.rs | 14 ++++++++++++++ src/player.rs | 6 ++++++ src/world.rs | 12 +++++++----- 6 files changed, 58 insertions(+), 5 deletions(-) diff --git a/src/gamecore.rs b/src/gamecore.rs index fcbe880..0954c0f 100644 --- a/src/gamecore.rs +++ b/src/gamecore.rs @@ -77,6 +77,22 @@ impl GameProgress { Ok(()) } + + pub fn update(&mut self, new_progress: &GameProgress) { + + // Bring in new data + self.coins = new_progress.coins; + self.inventory = new_progress.inventory.clone(); + self.max_depth = self.max_depth.max(new_progress.max_depth); + // self.fastest_time = self.fastest_time.min(new_progress.fastest_time); + + // Write to file + let result = self.to_file("./assets/savestate.json".to_string()); + if result.is_err() { + println!("Could not save game state. Holding in RAM"); + } + + } } /// This structure contains the entire game state, and should be passed around to various logic functions. diff --git a/src/logic/mainmenu.rs b/src/logic/mainmenu.rs index 9f64a60..3dc3f90 100644 --- a/src/logic/mainmenu.rs +++ b/src/logic/mainmenu.rs @@ -87,6 +87,10 @@ impl Screen for MainMenuScreen { // Check clicks if mouse_clicked { if hovering_play_button { + // Reset the world + game_core.world.reset(&mut game_core.player); + + // Start playing return Some(GameState::InGame); } else if hovering_shop_button { return Some(GameState::InShop); diff --git a/src/logic/shop/mainui.rs b/src/logic/shop/mainui.rs index d6d02a7..03d0b46 100644 --- a/src/logic/shop/mainui.rs +++ b/src/logic/shop/mainui.rs @@ -146,9 +146,20 @@ pub fn render_shop( // Handle click actions on the buttons if draw_handle.is_mouse_button_pressed(MouseButton::MOUSE_LEFT_BUTTON) { + + // Handle saving core state + if menu_button.is_hovered(draw_handle) || play_button.is_hovered(draw_handle) { + let new_progress = game_core.player.create_statistics(game_core, draw_handle.get_time()); + game_core.progress.update(&new_progress); + } + if menu_button.is_hovered(draw_handle) { return Some(GameState::MainMenu); } else if play_button.is_hovered(draw_handle) { + // Reset the world + game_core.world.reset(&mut game_core.player); + + // Start playing return Some(GameState::InGame); } } diff --git a/src/main.rs b/src/main.rs index a8a260f..64c8e2a 100644 --- a/src/main.rs +++ b/src/main.rs @@ -58,6 +58,8 @@ fn main() { // Set up the game's core state let mut game_core = GameCore::new(&mut raylib, &raylib_thread, world, game_progress); + game_core.player.inventory = game_core.progress.inventory.clone(); + game_core.player.coins = game_core.progress.coins; // Set up the game's profiler let mut profiler = GameProfiler::new(); @@ -125,6 +127,12 @@ fn main() { // Handle game quit if new_state == GameState::GameQuit { + // Save the game state + let new_progress = game_core + .player + .create_statistics(&game_core, draw_handle.get_time()); + game_core.progress.update(&new_progress); + // For now, just quit // This also throws a SEGFAULT.. yay for unsafe code.. info!("User quit game"); @@ -181,6 +189,12 @@ fn main() { game_core.last_frame_time = draw_handle.get_time(); } + // Save the game state + let new_progress = game_core + .player + .create_statistics(&game_core, raylib.get_time()); + game_core.progress.update(&new_progress); + // Cleanup profiler.stop(); } diff --git a/src/player.rs b/src/player.rs index ea81463..574172f 100644 --- a/src/player.rs +++ b/src/player.rs @@ -63,6 +63,12 @@ impl Player { } } + pub fn reset(&mut self, position: Vector2) { + self.position = position; + self.breath_percent = 1.0; + self.boost_percent = 1.0; + } + pub fn collides_with_rec(&self, rectangle: &Rectangle) -> bool { // // Build a bounding box of the player by their corners // let top_left_corner = self.position - (self.size / 2.0); diff --git a/src/world.rs b/src/world.rs index db722b3..ed58b00 100644 --- a/src/world.rs +++ b/src/world.rs @@ -5,7 +5,7 @@ use serde::{Deserialize, Serialize}; use std::io::Read; use failure::Error; -use crate::entities::{enemy::{jellyfish::JellyFish, octopus::Octopus}, fish::FishEntity}; +use crate::{entities::{enemy::{jellyfish::JellyFish, octopus::Octopus}, fish::FishEntity}, player::Player}; #[derive(Debug, Serialize, Deserialize, Clone)] pub struct World { @@ -58,10 +58,12 @@ impl World { // } // } - pub fn reset(&mut self) { - for fish in self.fish.iter_mut() { - fish.following_player = false; - } + pub fn reset(&mut self, player: &mut Player) { + // Init all fish + self.fish = FishEntity::new_from_positions(&self.fish_positions); + + // Reset the player + player.reset(self.player_spawn); } } From cc4affdbd16ad7cef90c947a4574df35feef09ed Mon Sep 17 00:00:00 2001 From: Evan Pratten Date: Sun, 25 Apr 2021 12:39:53 -0400 Subject: [PATCH 23/27] remove player extra coins --- src/player.rs | 1 - 1 file changed, 1 deletion(-) diff --git a/src/player.rs b/src/player.rs index 574172f..48fcdbe 100644 --- a/src/player.rs +++ b/src/player.rs @@ -58,7 +58,6 @@ impl Player { radius: 4.5, position: spawn.clone(), inventory: PlayerInventory::new(), - coins: 50, //TMP ..Default::default() } } From 1abc80be6416c19e41f628965cb8bfbfd9ef5911 Mon Sep 17 00:00:00 2001 From: Evan Pratten Date: Sun, 25 Apr 2021 13:06:48 -0400 Subject: [PATCH 24/27] placeholder item rendering --- src/logic/shop/item.rs | 4 + src/logic/shop/mainui.rs | 65 ++++++++++++-- src/logic/shop/mod.rs | 177 +-------------------------------------- 3 files changed, 64 insertions(+), 182 deletions(-) diff --git a/src/logic/shop/item.rs b/src/logic/shop/item.rs index 1bddfd2..2c94f3b 100644 --- a/src/logic/shop/item.rs +++ b/src/logic/shop/item.rs @@ -62,6 +62,10 @@ impl ShopItemWrapper { return self.ui.buy_button_hovered && draw_handle.is_mouse_button_pressed(MouseButton::MOUSE_LEFT_BUTTON); } + pub fn user_hovering_row(&self, draw_handle: &mut RaylibDrawHandle) -> bool { + return self.bounds.check_collision_point_rec(draw_handle.get_mouse_position()); + } + pub fn render(&mut self, draw_handle: &mut RaylibDrawHandle, player: &Player) { self.ui.render(draw_handle, self.bounds, self.can_player_afford(player)); } diff --git a/src/logic/shop/mainui.rs b/src/logic/shop/mainui.rs index 03d0b46..1b25b42 100644 --- a/src/logic/shop/mainui.rs +++ b/src/logic/shop/mainui.rs @@ -1,6 +1,10 @@ use raylib::prelude::*; -use crate::{gamecore::{GameCore, GameState}, items::{AirBag, Flashlight, Flippers, ItemBase, StunGun}, lib::{utils::button::OnScreenButton, wrappers::audio::player::AudioPlayer}}; +use crate::{ + gamecore::{GameCore, GameState}, + items::{AirBag, Flashlight, Flippers, ItemBase, StunGun}, + lib::{utils::button::OnScreenButton, wrappers::audio::player::AudioPlayer}, +}; use super::{item::ShopItemWrapper, itemui::ShopItemUi}; @@ -89,23 +93,65 @@ pub fn render_shop( flippers_buy_ui.render(draw_handle, &game_core.player); // Handle buying items - if stun_gun_buy_ui.can_player_afford(&game_core.player) && stun_gun_buy_ui.user_clicked_buy(draw_handle) { + if stun_gun_buy_ui.can_player_afford(&game_core.player) + && stun_gun_buy_ui.user_clicked_buy(draw_handle) + { let item = stun_gun_buy_ui.purchase(&mut game_core.player); game_core.player.inventory.stun_gun = Some(item); } - if air_bag_buy_ui.can_player_afford(&game_core.player) && air_bag_buy_ui.user_clicked_buy(draw_handle) { + if air_bag_buy_ui.can_player_afford(&game_core.player) + && air_bag_buy_ui.user_clicked_buy(draw_handle) + { let item = air_bag_buy_ui.purchase(&mut game_core.player); game_core.player.inventory.air_bag = Some(item); } - if flashlight_buy_ui.can_player_afford(&game_core.player) && flashlight_buy_ui.user_clicked_buy(draw_handle) { + if flashlight_buy_ui.can_player_afford(&game_core.player) + && flashlight_buy_ui.user_clicked_buy(draw_handle) + { let item = flashlight_buy_ui.purchase(&mut game_core.player); game_core.player.inventory.flashlight = Some(item); } - if flippers_buy_ui.can_player_afford(&game_core.player) && flippers_buy_ui.user_clicked_buy(draw_handle) { + if flippers_buy_ui.can_player_afford(&game_core.player) + && flippers_buy_ui.user_clicked_buy(draw_handle) + { let item = flippers_buy_ui.purchase(&mut game_core.player); game_core.player.inventory.flippers = Some(item); } + // Render the tooltip box + let hovering_stun_gun = stun_gun_buy_ui.user_hovering_row(draw_handle); + let hovering_air_bag = air_bag_buy_ui.user_hovering_row(draw_handle); + let hovering_flashlight = flashlight_buy_ui.user_hovering_row(draw_handle); + let hovering_flippers = flippers_buy_ui.user_hovering_row(draw_handle); + let should_show_tooltip = + hovering_stun_gun || hovering_air_bag || hovering_flashlight || hovering_flippers; + + if should_show_tooltip { + // Create bounds + let box_bounds = Rectangle { + x: bounds.x + 5.0, + y: bounds.y + 350.0, + width: bounds.width - 10.0, + height: 250.0, + }; + + // Draw background box + draw_handle.draw_rectangle_rec(box_bounds, Color::WHITE); + draw_handle.draw_rectangle_lines_ex(box_bounds, 3, Color::BLACK); + + // TODO: draw item sprite + draw_handle.draw_rectangle_v( + Vector2 { + x: box_bounds.x + (box_bounds.width / 2.0) - 40.0, + y: box_bounds.y + 10.0, + }, + Vector2 { + x: 80.0, + y: 80.0 + }, + Color::BLACK, + ); + } // Handle exit buttons let bottom_left_button_dimensions = Rectangle { @@ -115,7 +161,7 @@ pub fn render_shop( height: 40.0, }; let bottom_right_button_dimensions = Rectangle { - x: (bounds.x + bottom_left_button_dimensions.width ) + 15.0, + x: (bounds.x + bottom_left_button_dimensions.width) + 15.0, y: bottom_left_button_dimensions.y, width: bottom_left_button_dimensions.width, height: bottom_left_button_dimensions.height, @@ -146,10 +192,11 @@ pub fn render_shop( // Handle click actions on the buttons if draw_handle.is_mouse_button_pressed(MouseButton::MOUSE_LEFT_BUTTON) { - // Handle saving core state - if menu_button.is_hovered(draw_handle) || play_button.is_hovered(draw_handle) { - let new_progress = game_core.player.create_statistics(game_core, draw_handle.get_time()); + if menu_button.is_hovered(draw_handle) || play_button.is_hovered(draw_handle) { + let new_progress = game_core + .player + .create_statistics(game_core, draw_handle.get_time()); game_core.progress.update(&new_progress); } diff --git a/src/logic/shop/mod.rs b/src/logic/shop/mod.rs index 0013b5a..9885e04 100644 --- a/src/logic/shop/mod.rs +++ b/src/logic/shop/mod.rs @@ -1,6 +1,6 @@ -mod mainui; -mod itemui; mod item; +mod itemui; +mod mainui; use raylib::prelude::*; @@ -26,42 +26,6 @@ impl ShopScreen { ..Default::default() } } - - // // Creates all the items - // pub fn create_items(&mut self, screen_dimension: Vector2) { - // // gets every item.. hacky - // let items = ShopItems::get_inital_items(); - - // // sets sizes any random number is just a number I think looks good - // let screen_width = screen_dimension.x as f32; - // let screen_height = screen_dimension.y as f32; - - // let box_height = screen_height * 0.15; - // let box_width = screen_width * 0.1; - - // let start_width = screen_width - (box_width * 4.0) - 40.0; - // let draw_height = screen_height - 20.0 - box_height; - - // let mut item_vec = Vec::new(); - - // for box_num in 0..4 { - // let x_pose = start_width + box_width * box_num as f32; - - // // adds an item struct to the item list - // item_vec.push(Item { - // x_pose: ((x_pose + (5 * box_num) as f32) as i32), - // y_pose: (draw_height as i32), - // width: (box_width as i32), - // height: (box_height as i32), - // // Crazy hacky but this gets the data from the enum - // cost: (ShopItems::get_cost(&items.get(box_num).unwrap())), - // level: (ShopItems::get_level(&items.get(box_num).unwrap())), - // name: (ShopItems::get_name(&items.get(box_num).unwrap())), - // }); - // } - - // self.shop_items = item_vec; - // } } impl Screen for ShopScreen { @@ -73,8 +37,7 @@ impl Screen for ShopScreen { game_core: &mut GameCore, ) -> Option { let mouse_position = draw_handle.get_mouse_position(); - - + // Render the background draw_handle.draw_texture(&game_core.resources.shop_background, 0, 0, Color::WHITE); @@ -97,8 +60,7 @@ impl Screen for ShopScreen { }; // Render the shop UI - let next_state = - render_shop(draw_handle, thread, audio_system, game_core, shop_ui_bounds); + let next_state = render_shop(draw_handle, thread, audio_system, game_core, shop_ui_bounds); // Render the stats UI render_stats(draw_handle, game_core, stats_ui_bounds); @@ -106,134 +68,3 @@ impl Screen for ShopScreen { return next_state; } } -// pub fn render_shop( -// draw_handle: &mut RaylibDrawHandle, -// game_core: &mut GameCore, -// inGameScreen: &mut InGameScreen, -// ) { -// // Pressing F exits from buying -// if draw_handle.is_key_pressed(KeyboardKey::KEY_F) { -// inGameScreen.current_state = InGameState::SWIMMING; -// } - -// let mouse_position = draw_handle.get_mouse_position(); - -// draw_handle.draw_text( -// &format!("Coins: {}", game_core.player.coins), -// 15, -// 15, -// 30, -// Color::WHITE, -// ); - -// // Draws shop boxes -// for mut item in inGameScreen.shop.shop_items.iter_mut() { -// // If hovering on square draw full -// if mouse_position.x >= item.x_pose as f32 -// && mouse_position.x <= item.x_pose as f32 + item.width as f32 -// && mouse_position.y >= item.y_pose as f32 -// && mouse_position.y <= item.y_pose as f32 + item.width as f32 -// { -// // Draw rect -// draw_handle.draw_rectangle( -// item.x_pose, -// item.y_pose, -// item.width, -// item.height, -// Color::BLACK, -// ); - -// // Preform purchasing functions -// if draw_handle.is_mouse_button_pressed(MouseButton::MOUSE_LEFT_BUTTON) -// && game_core.player.coins >= item.cost as u32 -// { -// // Remove currency -// game_core.world.spend_coins(item.cost.into()); -// game_core.player.coins -= item.cost as u32; - -// // Upgrade item in inventory -// match &(item.name)[..] { -// "Stun Gun" => { -// match item.level { -// 0 => game_core.player.inventory.stun_gun = Some(items::StunGun::lvl1()), -// 1 => game_core.player.inventory.stun_gun = Some(items::StunGun::lvl2()), -// 2 => game_core.player.inventory.stun_gun = Some(items::StunGun::lvl3()), -// _ => (return), -// }; -// item.cost += 5; -// item.level += 1; -// } -// "Air Bag" => { -// match item.level { -// 0 => { -// game_core.player.inventory.air_bag = Some(items::AirBag::lvl1()); -// } -// 1 => { -// game_core.player.inventory.air_bag = Some(items::AirBag::lvl2()); -// } -// 2 => { -// game_core.player.inventory.air_bag = Some(items::AirBag::lvl3()); -// } -// _ => (return), -// }; -// item.cost += 5; -// item.level += 1; -// } -// "Flash Light" => { -// match item.level { -// 0 => { -// game_core.player.inventory.flashlight = -// Some(items::Flashlight::lvl1()); -// } -// 1 => { -// game_core.player.inventory.flashlight = -// Some(items::Flashlight::lvl2()); -// } -// 2 => { -// game_core.player.inventory.flashlight = -// Some(items::Flashlight::lvl3()); -// } -// _ => (return), -// }; -// item.cost += 5; -// item.level += 1; -// } -// "Flippers" => { -// match item.level { -// 0 => { -// game_core.player.inventory.flippers = Some(items::Flippers::lvl1()); -// } -// 1 => { -// game_core.player.inventory.flippers = Some(items::Flippers::lvl2()); -// } -// 2 => { -// game_core.player.inventory.flippers = Some(items::Flippers::lvl3()); -// } -// _ => (return), -// }; -// item.cost += 5; -// item.level += 1; -// } -// _ => (return), -// }; -// } -// } else { -// // outlines if not hovered -// draw_handle.draw_rectangle_lines( -// item.x_pose, -// item.y_pose, -// item.width, -// item.height, -// Color::BLACK, -// ); -// } -// // Draw text about object -// draw_handle.draw_text( -// &format!("{}: ${}", item.name, item.cost), -// item.x_pose + 5, -// item.y_pose + 5, -// 12, -// Color::BLACK, -// ); -// } -// } From 53b3a33b4e43a6516da61bbda6dce3a73de3a007 Mon Sep 17 00:00:00 2001 From: Evan Pratten Date: Sun, 25 Apr 2021 13:15:05 -0400 Subject: [PATCH 25/27] Store tooltips --- src/items.rs | 6 +++--- src/logic/shop/mainui.rs | 26 ++++++++++++++++++++++---- 2 files changed, 25 insertions(+), 7 deletions(-) diff --git a/src/items.rs b/src/items.rs index 66638b3..d30dabc 100644 --- a/src/items.rs +++ b/src/items.rs @@ -54,7 +54,7 @@ impl ItemBase for StunGun { } fn get_description(&self) -> String { - return "Stun your enemies! Just don't point it at yourself.".to_string(); + return "Stun your enemies!\nJust don't point it at yourself.".to_string(); } fn get_texture(&self) -> &Texture2D { @@ -106,7 +106,7 @@ impl ItemBase for AirBag { } fn get_description(&self) -> String { - return "Its.. a bag. Filled with air. Duh".to_string(); + return "Its.. a bag.\nFilled with air. Duh".to_string(); } fn get_texture(&self) -> &Texture2D { @@ -210,7 +210,7 @@ impl ItemBase for Flippers { } fn get_description(&self) -> String { - return "Swim faster, and look stupid at the same time!".to_string(); + return "Swim faster, and look stupid\nat the same time!".to_string(); } fn get_texture(&self) -> &Texture2D { diff --git a/src/logic/shop/mainui.rs b/src/logic/shop/mainui.rs index 1b25b42..759da23 100644 --- a/src/logic/shop/mainui.rs +++ b/src/logic/shop/mainui.rs @@ -135,6 +135,18 @@ pub fn render_shop( height: 250.0, }; + // Get the hovered item + let hovered_item: &dyn ItemBase; + if hovering_stun_gun { + hovered_item = stun_gun_buy_ui.get_item(); + } else if hovering_air_bag { + hovered_item = air_bag_buy_ui.get_item(); + } else if hovering_flashlight { + hovered_item = flashlight_buy_ui.get_item(); + } else { + hovered_item = flippers_buy_ui.get_item(); + } + // Draw background box draw_handle.draw_rectangle_rec(box_bounds, Color::WHITE); draw_handle.draw_rectangle_lines_ex(box_bounds, 3, Color::BLACK); @@ -145,10 +157,16 @@ pub fn render_shop( x: box_bounds.x + (box_bounds.width / 2.0) - 40.0, y: box_bounds.y + 10.0, }, - Vector2 { - x: 80.0, - y: 80.0 - }, + Vector2 { x: 80.0, y: 80.0 }, + Color::BLACK, + ); + + // Render item description + draw_handle.draw_text( + &hovered_item.get_description(), + box_bounds.x as i32 + 10, + box_bounds.y as i32 + 100, + 30, Color::BLACK, ); } From da3b52e39516060b6be576d9f4f1574c5eaa012f Mon Sep 17 00:00:00 2001 From: Evan Pratten Date: Sun, 25 Apr 2021 13:25:05 -0400 Subject: [PATCH 26/27] Add flipper logic --- src/logic/ingame/playerlogic.rs | 7 +++++++ 1 file changed, 7 insertions(+) diff --git a/src/logic/ingame/playerlogic.rs b/src/logic/ingame/playerlogic.rs index a506daa..9ff92ed 100644 --- a/src/logic/ingame/playerlogic.rs +++ b/src/logic/ingame/playerlogic.rs @@ -147,6 +147,13 @@ pub fn update_player_movement( // Only do this if the mouse is far enough away let player_stunned = game_core.player.stun_timer > 0.0; let mut player_real_movement = game_core.player.direction * speed_multiplier; + + // Handle the player wearing flippers + if game_core.player.inventory.flippers.is_some() { + player_real_movement *= game_core.player.inventory.flippers.as_ref().unwrap().speed_increase; + } + + // Handle movement and collisions if raw_movement_direction.distance_to(Vector2::zero()) > game_core.player.size.y / 2.0 && !game_core.player.is_stunned() { From 2d6d1f3629fcf498d1d147fb001ce6f01b528029 Mon Sep 17 00:00:00 2001 From: Evan Pratten Date: Sun, 25 Apr 2021 13:28:34 -0400 Subject: [PATCH 27/27] air bag impl --- src/items.rs | 8 ++++---- src/logic/ingame/playerlogic.rs | 2 +- src/player.rs | 5 +++++ 3 files changed, 10 insertions(+), 5 deletions(-) diff --git a/src/items.rs b/src/items.rs index d30dabc..f83afd1 100644 --- a/src/items.rs +++ b/src/items.rs @@ -67,7 +67,7 @@ impl ItemBase for StunGun { #[derive(Debug, PartialEq, Clone, Serialize, Deserialize)] pub struct AirBag { - extra_oxygen: u32, + pub extra_oxygen: f32, pub level: u8, cost: u32, } @@ -75,21 +75,21 @@ pub struct AirBag { impl AirBag { pub fn lvl1() -> Self { Self { - extra_oxygen: 15, + extra_oxygen: 0.15, level: 1, cost: 30, } } pub fn lvl2() -> Self { Self { - extra_oxygen: 30, + extra_oxygen: 0.30, level: 2, cost: 40, } } pub fn lvl3() -> Self { Self { - extra_oxygen: 45, + extra_oxygen: 0.45, level: 3, cost: 50, } diff --git a/src/logic/ingame/playerlogic.rs b/src/logic/ingame/playerlogic.rs index 9ff92ed..91e2cef 100644 --- a/src/logic/ingame/playerlogic.rs +++ b/src/logic/ingame/playerlogic.rs @@ -142,7 +142,7 @@ pub fn update_player_movement( // 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 - BREATH_DECREASE_PER_SECOND * dt as f32).max(0.0); // Only do this if the mouse is far enough away let player_stunned = game_core.player.stun_timer > 0.0; diff --git a/src/player.rs b/src/player.rs index 48fcdbe..0a78a20 100644 --- a/src/player.rs +++ b/src/player.rs @@ -66,6 +66,11 @@ impl Player { self.position = position; self.breath_percent = 1.0; self.boost_percent = 1.0; + + // Handle an air bag being used + if self.inventory.air_bag.is_some() { + self.breath_percent += self.inventory.air_bag.as_ref().unwrap().extra_oxygen; + } } pub fn collides_with_rec(&self, rectangle: &Rectangle) -> bool {