diff --git a/README.md b/README.md index d3ba8d5..6f531d6 100644 --- a/README.md +++ b/README.md @@ -31,8 +31,9 @@ This game is developed by a team of 6 students from *Sheridan College* and *Tren - [**Emilia Frias**](https://www.instagram.com/demilurii/) - Character art - Animations - - Map assets + - Tilesets - [**Kori**](https://www.instagram.com/korigama/) - Concept art + - Tilesets A special thanks goes out to: [James Nickoli](https://github.com/rsninja722/) for insight on 2D collision detection, as well as [Ray](https://github.com/raysan5) and the members of the [raylib community](https://discord.gg/raylib) on discord for their support with the past two game jam projects. diff --git a/game/assets/audio/button-press.mp3 b/game/assets/audio/button-press.mp3 new file mode 100644 index 0000000..82cbcf3 Binary files /dev/null and b/game/assets/audio/button-press.mp3 differ diff --git a/game/assets/audio/soundtrack.mp3 b/game/assets/audio/soundtrack.mp3 new file mode 100644 index 0000000..3f8edab Binary files /dev/null and b/game/assets/audio/soundtrack.mp3 differ diff --git a/game/src/context.rs b/game/src/context.rs index b08f8e8..6612f52 100644 --- a/game/src/context.rs +++ b/game/src/context.rs @@ -1,9 +1,10 @@ -use std::{cell::RefCell, sync::mpsc::Sender}; +use std::{cell::RefCell, collections::HashMap, sync::mpsc::Sender}; use chrono::{DateTime, Duration, Utc}; use discord_sdk::activity::ActivityBuilder; +use raylib::audio::Sound; -use crate::{progress::ProgressData, utilities::non_ref_raylib::HackedRaylibHandle, GameConfig}; +use crate::{GameConfig, progress::ProgressData, utilities::{audio_player::AudioPlayer, non_ref_raylib::HackedRaylibHandle}}; #[derive(Debug)] pub enum ControlFlag { @@ -11,12 +12,15 @@ pub enum ControlFlag { SwitchLevel(usize), UpdateLevelStart(DateTime), SaveProgress, - MaybeUpdateHighScore(usize, Duration) + MaybeUpdateHighScore(usize, Duration), + SoundTrigger(String) } #[derive(Debug)] pub struct GameContext { pub renderer: RefCell, + pub audio: AudioPlayer, + pub sounds: HashMap, pub config: GameConfig, pub player_progress: ProgressData, pub current_level: usize, diff --git a/game/src/lib.rs b/game/src/lib.rs index 31aee85..b768fee 100644 --- a/game/src/lib.rs +++ b/game/src/lib.rs @@ -70,7 +70,7 @@ )] #![clippy::msrv = "1.57.0"] -use std::{borrow::BorrowMut, cell::RefCell, sync::mpsc::TryRecvError}; +use std::{borrow::BorrowMut, cell::RefCell, collections::HashMap, sync::mpsc::TryRecvError}; use chrono::Utc; use discord_sdk::activity::ActivityBuilder; @@ -84,6 +84,8 @@ use crate::{ progress::ProgressData, scenes::{build_screen_state_machine, Scenes}, utilities::{ + audio_player::AudioPlayer, + datastore::{load_music_from_internal_data, load_sound_from_internal_data}, game_config::FinalShaderConfig, shaders::{ shader::ShaderWrapper, @@ -172,10 +174,23 @@ pub async fn game_begin(game_config: &mut GameConfig) -> Result<(), Box Result<(), Box Result<(), Box Result<(), Box { + context.audio.play_sound( + context.sounds.get(&name).unwrap(), + ); + } } } } diff --git a/game/src/scenes/death_screen.rs b/game/src/scenes/death_screen.rs index d3ae105..ee2d4d3 100644 --- a/game/src/scenes/death_screen.rs +++ b/game/src/scenes/death_screen.rs @@ -6,17 +6,13 @@ use discord_sdk::activity::{ActivityBuilder, Assets}; use pkg_version::pkg_version_major; use raylib::prelude::*; -use crate::{ - context::GameContext, - utilities::{ +use crate::{GameConfig, context::{ControlFlag, GameContext}, utilities::{ datastore::{load_texture_from_internal_data, ResourceLoadError}, game_version::get_version_string, math::interpolate_exp, non_ref_raylib::HackedRaylibHandle, render_layer::ScreenSpaceRender, - }, - GameConfig, -}; + }}; use super::{Scenes, ScreenError}; use tracing::{debug, error, info, trace}; @@ -69,6 +65,10 @@ impl Action for DeathScreen { self.timer_value = format!("{:02}:{:02}", elapsed.num_minutes(), elapsed.num_seconds() % 60); if self.is_retry_pressed { + context + .flag_send + .send(Some(ControlFlag::SoundTrigger("button-press".to_string()))) + .unwrap(); Ok(ActionFlag::SwitchState(Scenes::InGameScene)) } else { Ok(ActionFlag::Continue) diff --git a/game/src/scenes/how_to_play_screen.rs b/game/src/scenes/how_to_play_screen.rs index bcfb308..1592b81 100644 --- a/game/src/scenes/how_to_play_screen.rs +++ b/game/src/scenes/how_to_play_screen.rs @@ -6,17 +6,13 @@ use discord_sdk::activity::{ActivityBuilder, Assets}; use pkg_version::pkg_version_major; use raylib::prelude::*; -use crate::{ - context::GameContext, - utilities::{ +use crate::{GameConfig, context::{ControlFlag, GameContext}, utilities::{ datastore::{load_texture_from_internal_data, ResourceLoadError}, game_version::get_version_string, math::interpolate_exp, non_ref_raylib::HackedRaylibHandle, render_layer::ScreenSpaceRender, - }, - GameConfig, -}; + }}; use super::{Scenes, ScreenError}; use tracing::{debug, error, info, trace}; @@ -66,6 +62,10 @@ impl Action for HowToPlayScreen { self.render_screen_space(&mut context.renderer.borrow_mut(), &context.config); if self.is_btm_pressed { + context + .flag_send + .send(Some(ControlFlag::SoundTrigger("button-press".to_string()))) + .unwrap(); Ok(ActionFlag::SwitchState(Scenes::MainMenuScreen)) } else { Ok(ActionFlag::Continue) diff --git a/game/src/scenes/main_menu_screen.rs b/game/src/scenes/main_menu_screen.rs index d32187e..d8fa2c4 100644 --- a/game/src/scenes/main_menu_screen.rs +++ b/game/src/scenes/main_menu_screen.rs @@ -71,12 +71,28 @@ impl Action for MainMenuScreen { self.render_screen_space(&mut context.renderer.borrow_mut(), &context.config); if self.is_start_pressed { + context + .flag_send + .send(Some(ControlFlag::SoundTrigger("button-press".to_string()))) + .unwrap(); Ok(ActionFlag::SwitchState(Scenes::InGameScene)) } else if self.is_htp_pressed { + context + .flag_send + .send(Some(ControlFlag::SoundTrigger("button-press".to_string()))) + .unwrap(); Ok(ActionFlag::SwitchState(Scenes::HowToPlayScreen)) } else if self.is_options_pressed { + context + .flag_send + .send(Some(ControlFlag::SoundTrigger("button-press".to_string()))) + .unwrap(); Ok(ActionFlag::SwitchState(Scenes::OptionsScreen)) } else if self.is_quit_pressed { + context + .flag_send + .send(Some(ControlFlag::SoundTrigger("button-press".to_string()))) + .unwrap(); context.flag_send.send(Some(ControlFlag::Quit)).unwrap(); Ok(ActionFlag::Continue) } else { @@ -265,5 +281,7 @@ impl ScreenSpaceRender for MainMenuScreen { ); }; self.is_quit_pressed = mouse_pressed && hovering_quit; + + // for } } diff --git a/game/src/scenes/mod.rs b/game/src/scenes/mod.rs index 1a1c055..1143503 100644 --- a/game/src/scenes/mod.rs +++ b/game/src/scenes/mod.rs @@ -6,13 +6,7 @@ use self::{ death_screen::DeathScreen, win_screen::WinScreen, next_level_screen::NextLevelScreen }; -use crate::{ - context::GameContext, - utilities::{ - datastore::{load_texture_from_internal_data, ResourceLoadError}, - non_ref_raylib::HackedRaylibHandle, - }, -}; +use crate::{context::GameContext, utilities::{datastore::{ResourceLoadError, load_music_from_internal_data, load_sound_from_internal_data, load_texture_from_internal_data}, non_ref_raylib::HackedRaylibHandle}}; use dirty_fsm::StateMachine; use raylib::{texture::Texture2D, RaylibThread}; diff --git a/game/src/scenes/next_level_screen.rs b/game/src/scenes/next_level_screen.rs index b677855..464ff8c 100644 --- a/game/src/scenes/next_level_screen.rs +++ b/game/src/scenes/next_level_screen.rs @@ -6,17 +6,13 @@ use discord_sdk::activity::{ActivityBuilder, Assets}; use pkg_version::pkg_version_major; use raylib::prelude::*; -use crate::{ - context::GameContext, - utilities::{ +use crate::{GameConfig, context::{ControlFlag, GameContext}, utilities::{ datastore::{load_texture_from_internal_data, ResourceLoadError}, game_version::get_version_string, math::interpolate_exp, non_ref_raylib::HackedRaylibHandle, render_layer::ScreenSpaceRender, - }, - GameConfig, -}; + }}; use super::{Scenes, ScreenError}; use tracing::{debug, error, info, trace}; @@ -87,6 +83,10 @@ impl Action for NextLevelScreen { ); if self.is_next_pressed { + context + .flag_send + .send(Some(ControlFlag::SoundTrigger("button-press".to_string()))) + .unwrap(); Ok(ActionFlag::SwitchState(Scenes::InGameScene)) } else { Ok(ActionFlag::Continue) diff --git a/game/src/scenes/options_screen.rs b/game/src/scenes/options_screen.rs index b0b3832..cff3716 100644 --- a/game/src/scenes/options_screen.rs +++ b/game/src/scenes/options_screen.rs @@ -6,17 +6,13 @@ use discord_sdk::activity::{ActivityBuilder, Assets}; use pkg_version::pkg_version_major; use raylib::prelude::*; -use crate::{ - context::GameContext, - utilities::{ +use crate::{GameConfig, context::{ControlFlag, GameContext}, utilities::{ datastore::{load_texture_from_internal_data, ResourceLoadError}, game_version::get_version_string, math::interpolate_exp, non_ref_raylib::HackedRaylibHandle, render_layer::ScreenSpaceRender, - }, - GameConfig, -}; + }}; use super::{Scenes, ScreenError}; use tracing::{debug, error, info, trace}; @@ -69,6 +65,10 @@ impl Action for OptionsScreen { self.render_screen_space(&mut context.renderer.borrow_mut(), &context.config); if self.is_btm_pressed { + context + .flag_send + .send(Some(ControlFlag::SoundTrigger("button-press".to_string()))) + .unwrap(); Ok(ActionFlag::SwitchState(Scenes::MainMenuScreen)) } else { Ok(ActionFlag::Continue) diff --git a/game/src/scenes/pause_screen.rs b/game/src/scenes/pause_screen.rs index 1dde2e6..c2e59ae 100644 --- a/game/src/scenes/pause_screen.rs +++ b/game/src/scenes/pause_screen.rs @@ -6,17 +6,13 @@ use discord_sdk::activity::{ActivityBuilder, Assets}; use pkg_version::pkg_version_major; use raylib::prelude::*; -use crate::{ - context::GameContext, - utilities::{ +use crate::{GameConfig, context::{ControlFlag, GameContext}, utilities::{ datastore::{load_texture_from_internal_data, ResourceLoadError}, game_version::get_version_string, math::interpolate_exp, non_ref_raylib::HackedRaylibHandle, render_layer::ScreenSpaceRender, - }, - GameConfig, -}; + }}; use super::{Scenes, ScreenError}; use tracing::{debug, error, info, trace}; @@ -82,6 +78,10 @@ impl Action for PauseScreen { && Rectangle::new(centered_x_paused, centered_y_paused, 435.0, 80.0) .check_collision_point_rec(mouse_position) { + context + .flag_send + .send(Some(ControlFlag::SoundTrigger("button-press".to_string()))) + .unwrap(); return Ok(ActionFlag::SwitchState(Scenes::InGameScene)); } //For Menu @@ -89,6 +89,10 @@ impl Action for PauseScreen { && Rectangle::new(centered_x_menu, centered_y_menu, 200.0, 50.0) .check_collision_point_rec(mouse_position) { + context + .flag_send + .send(Some(ControlFlag::SoundTrigger("button-press".to_string()))) + .unwrap(); return Ok(ActionFlag::SwitchState(Scenes::MainMenuScreen)); } diff --git a/game/src/scenes/win_screen.rs b/game/src/scenes/win_screen.rs index dc2851f..2f2dca2 100644 --- a/game/src/scenes/win_screen.rs +++ b/game/src/scenes/win_screen.rs @@ -6,17 +6,13 @@ use discord_sdk::activity::{ActivityBuilder, Assets}; use pkg_version::pkg_version_major; use raylib::prelude::*; -use crate::{ - context::GameContext, - utilities::{ +use crate::{GameConfig, context::{ControlFlag, GameContext}, utilities::{ datastore::{load_texture_from_internal_data, ResourceLoadError}, game_version::get_version_string, math::interpolate_exp, non_ref_raylib::HackedRaylibHandle, render_layer::ScreenSpaceRender, - }, - GameConfig, -}; + }}; use super::{Scenes, ScreenError}; use tracing::{debug, error, info, trace}; @@ -69,6 +65,10 @@ impl Action for WinScreen { self.counter += 1; if self.is_menu_pressed { + context + .flag_send + .send(Some(ControlFlag::SoundTrigger("button-press".to_string()))) + .unwrap(); Ok(ActionFlag::SwitchState(Scenes::MainMenuScreen)) } else { Ok(ActionFlag::Continue) diff --git a/game/src/utilities/audio_player.rs b/game/src/utilities/audio_player.rs new file mode 100644 index 0000000..e33c61b --- /dev/null +++ b/game/src/utilities/audio_player.rs @@ -0,0 +1,49 @@ +use raylib::audio::RaylibAudio; + +/// A thin wrapper around `raylib::core::audio::RaylibAudio` that keeps track of the volume of its audio channels. +#[derive(Debug)] +pub struct AudioPlayer { + backend: RaylibAudio, + + // Volume + pub master_volume: f32, +} + +impl AudioPlayer { + /// Construct an AudioPlayer around a RaylibAudio + pub fn new(backend: RaylibAudio) -> Self { + Self { + backend, + master_volume: 1.0, + } + } + + /// Set the master volume for all tracks. `0.0` to `1.0` + pub fn set_master_volume(&mut self, volume: f32) { + // The volume must be 0-1 + let volume = volume.clamp(0.0, 1.0); + + // Set the volume + self.master_volume = volume; + self.backend.set_master_volume(volume); + } + + /// Get the master volume + pub fn get_master_volume(&self) -> f32 { + self.master_volume + } +} + +impl std::ops::Deref for AudioPlayer { + type Target = RaylibAudio; + fn deref(&self) -> &Self::Target { + &self.backend + } +} + + +impl std::ops::DerefMut for AudioPlayer { + fn deref_mut(&mut self) -> &mut Self::Target { + &mut self.backend + } +} diff --git a/game/src/utilities/datastore.rs b/game/src/utilities/datastore.rs index d181204..a8dfd0e 100644 --- a/game/src/utilities/datastore.rs +++ b/game/src/utilities/datastore.rs @@ -1,6 +1,10 @@ use std::{io::Write, path::Path}; -use raylib::{texture::Texture2D, RaylibHandle, RaylibThread}; +use raylib::{ + audio::{Music, Sound}, + texture::Texture2D, + RaylibHandle, RaylibThread, +}; use tempfile::{tempdir, NamedTempFile}; use tracing::debug; @@ -67,3 +71,71 @@ pub fn load_texture_from_internal_data( Ok(texture) } + +pub fn load_music_from_internal_data( + raylib_handle: &mut RaylibHandle, + thread: &RaylibThread, + path: &str, +) -> Result { + // Create a temp file path to work with + let temp_dir = tempdir()?; + debug!( + "Created temporary directory for passing embedded data to Raylib: {}", + temp_dir.path().display() + ); + let tmp_path = temp_dir.path().join(Path::new(path).file_name().unwrap()); + + // Unpack the raw sound data to a real file on the local filesystem so raylib will read it correctly + std::fs::write( + &tmp_path, + &StaticGameData::get(path) + .ok_or(ResourceLoadError::AssetNotFound(path.to_string()))? + .data, + )?; + + // Call through via FFI to re-load the file + let texture = Music::load_music_stream(thread, tmp_path.to_str().unwrap()) + .map_err(ResourceLoadError::Generic)?; + + // Close the file + debug!( + "Dropping temporary directory: {}", + temp_dir.path().display() + ); + temp_dir.close()?; + + Ok(texture) +} + +pub fn load_sound_from_internal_data( + path: &str, +) -> Result { + // Create a temp file path to work with + let temp_dir = tempdir()?; + debug!( + "Created temporary directory for passing embedded data to Raylib: {}", + temp_dir.path().display() + ); + let tmp_path = temp_dir.path().join(Path::new(path).file_name().unwrap()); + + // Unpack the raw sound data to a real file on the local filesystem so raylib will read it correctly + std::fs::write( + &tmp_path, + &StaticGameData::get(path) + .ok_or(ResourceLoadError::AssetNotFound(path.to_string()))? + .data, + )?; + + // Call through via FFI to re-load the file + let texture = + Sound::load_sound(tmp_path.to_str().unwrap()).map_err(ResourceLoadError::Generic)?; + + // Close the file + debug!( + "Dropping temporary directory: {}", + temp_dir.path().display() + ); + temp_dir.close()?; + + Ok(texture) +} diff --git a/game/src/utilities/mod.rs b/game/src/utilities/mod.rs index 1f05091..666e389 100644 --- a/game/src/utilities/mod.rs +++ b/game/src/utilities/mod.rs @@ -8,3 +8,4 @@ pub mod non_ref_raylib; pub mod render_layer; pub mod shaders; pub mod world_paint_texture; +pub mod audio_player;