diff --git a/Cargo.toml b/Cargo.toml index d17519a..8d72bb7 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -9,4 +9,5 @@ description = "" raylib = { version = "3.5", git = "https://github.com/ewpratten/raylib-rs", branch = "master" } serialstudio = "0.1.0" serde = "1.0.125" -serde_json = "1.0.64" \ No newline at end of file +serde_json = "1.0.64" +failure = "0.1.8" \ No newline at end of file diff --git a/src/gamecore.rs b/src/gamecore.rs index 37daab7..25526e3 100644 --- a/src/gamecore.rs +++ b/src/gamecore.rs @@ -1,13 +1,38 @@ +//! This file contains the global state of the game. Data here is passed around to all handler functions. + +use std::fmt; + +use crate::resources::GlobalResources; + /// Overall states for the game +#[derive(Debug)] pub enum GameState { Loading, MainMenu, } +impl fmt::Display for GameState { + fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result { + write!(f, "{:?}", self) + } +} + /// This structure contains the entire game state, and should be passed around to various logic functions. pub struct GameCore { - /// The game's overall state - pub state: GameState + pub state: GameState, + pub last_state_change_time: f64, + /// Resources + pub resources: Option, +} + +impl GameCore { + pub fn new() -> Self { + Self { + state: GameState::Loading, + last_state_change_time: 0.0, + resources: None, + } + } } diff --git a/src/lib/mod.rs b/src/lib/mod.rs index 2610b8a..7fcdc86 100644 --- a/src/lib/mod.rs +++ b/src/lib/mod.rs @@ -1,2 +1,2 @@ pub mod wrappers; -pub mod profiler; \ No newline at end of file +pub mod utils; \ No newline at end of file diff --git a/src/lib/profiler/mod.rs b/src/lib/profiler/mod.rs deleted file mode 100644 index b054cef..0000000 --- a/src/lib/profiler/mod.rs +++ /dev/null @@ -1,83 +0,0 @@ -use serde_json::json; -use serialstudio::{ - data::{DataGroup, DataSet, TelemetryFrame}, - SerialStudioSource, -}; - -/// The development profiler -#[derive(Default)] -pub struct GameProfiler { - /// The SerialStudio server - server: Option, - - /// The data - pub frames_per_second: u32, - pub seconds_per_frame: f32, -} - -/// Dev mode -#[cfg(debug_assertions)] -impl GameProfiler { - pub fn new() -> Self { - Self { - server: Some(SerialStudioSource::new()), - ..Default::default() - } - } - - pub fn start(&mut self) { - println!("Starting debug server on: tcp://localhost:8019"); - self.server - .as_mut() - .unwrap() - .start("localhost:8019".to_string()); - } - - pub fn stop(&mut self) { - println!("Stopping debug server"); - self.server.as_mut().unwrap().stop(); - } - - pub fn update(&mut self) { - // Build telemetry frame - let frame = TelemetryFrame { - title: "Game status".to_string(), - groups: vec![DataGroup { - title: "Rendering engine".to_string(), - widget_type: None, - datasets: vec![ - DataSet { - title: Some("Frames per Second".to_string()), - value: json!(self.frames_per_second), - graph: Some(true), - unit: Some("fps".to_string()), - w_type: None, - }, - DataSet { - title: Some("Seconds per Frame".to_string()), - value: json!(self.seconds_per_frame), - graph: Some(true), - unit: Some("seconds".to_string()), - w_type: None, - }, - ], - }], - }; - - // Send the frame - self.server.as_mut().unwrap().publish(frame); - } -} - -/// Release mode: We do nothing here -#[cfg(not(debug_assertions))] -impl GameProfiler { - pub fn new() -> Self { - Self { - ..Default::default() - } - } - pub fn start(&mut self) {} - pub fn stop(&mut self) {} - pub fn update(&mut self) {} -} diff --git a/src/lib/utils/mod.rs b/src/lib/utils/mod.rs new file mode 100644 index 0000000..c570215 --- /dev/null +++ b/src/lib/utils/mod.rs @@ -0,0 +1 @@ +pub mod profiler; \ No newline at end of file diff --git a/src/lib/utils/profiler.rs b/src/lib/utils/profiler.rs new file mode 100644 index 0000000..37293c2 --- /dev/null +++ b/src/lib/utils/profiler.rs @@ -0,0 +1,139 @@ +use serde_json::json; +use serialstudio::{ + data::{DataGroup, DataSet, TelemetryFrame}, + SerialStudioSource, +}; + +#[derive(Default)] +pub struct ProfilerData { + // Rendering + pub frames_per_second: u32, + pub seconds_per_frame: f32, + pub monitor_count: i32, + + // Audio + pub audio_volume: f32, + pub active_sounds: i32, + + // Game core + pub game_state: String +} + +/// The development profiler +#[derive(Default)] +pub struct GameProfiler { + /// The SerialStudio server + server: Option, + + /// The data + pub data: ProfilerData, +} + +/// Dev mode +#[cfg(debug_assertions)] +impl GameProfiler { + pub fn new() -> Self { + Self { + server: Some(SerialStudioSource::new()), + ..Default::default() + } + } + + pub fn start(&mut self) { + println!("Starting debug server on: tcp://localhost:8019"); + self.server + .as_mut() + .unwrap() + .start("localhost:8019".to_string()); + } + + pub fn stop(&mut self) { + println!("Stopping debug server"); + self.server.as_mut().unwrap().stop(); + } + + pub fn update(&mut self) { + // Build telemetry frame + let frame = TelemetryFrame { + title: "Game status".to_string(), + groups: vec![ + DataGroup { + title: "Rendering engine".to_string(), + widget_type: None, + datasets: vec![ + DataSet { + title: Some("Frames per Second".to_string()), + value: json!(self.data.frames_per_second), + graph: Some(true), + unit: Some("fps".to_string()), + w_type: None, + }, + DataSet { + title: Some("Seconds per Frame".to_string()), + value: json!(self.data.seconds_per_frame), + graph: Some(true), + unit: Some("seconds".to_string()), + w_type: None, + }, + DataSet { + title: Some("Monitor Count".to_string()), + value: json!(self.data.monitor_count), + graph: Some(false), + unit: None, + w_type: None, + }, + ], + }, + DataGroup { + title: "Audio engine".to_string(), + widget_type: None, + datasets: vec![ + DataSet { + title: Some("Master Volume".to_string()), + value: json!(self.data.audio_volume), + graph: Some(false), + unit: None, + w_type: None, + }, + DataSet { + title: Some("Active Sounds".to_string()), + value: json!(self.data.active_sounds), + graph: Some(true), + unit: None, + w_type: None, + }, + ], + }, + DataGroup { + title: "Game".to_string(), + widget_type: None, + datasets: vec![ + DataSet { + title: Some("Global State".to_string()), + value: json!(self.data.game_state), + graph: Some(false), + unit: None, + w_type: None, + }, + ], + }, + ], + }; + + // Send the frame + self.server.as_mut().unwrap().publish(frame); + } +} + +/// Release mode: We do nothing here +#[cfg(not(debug_assertions))] +impl GameProfiler { + pub fn new() -> Self { + Self { + ..Default::default() + } + } + pub fn start(&mut self) {} + pub fn stop(&mut self) {} + pub fn update(&mut self) {} +} diff --git a/src/lib/wrappers/audio.rs b/src/lib/wrappers/audio.rs deleted file mode 100644 index 67a76f1..0000000 --- a/src/lib/wrappers/audio.rs +++ /dev/null @@ -1,41 +0,0 @@ -use raylib::{ - audio::{Music, RaylibAudio}, - prelude::RaylibDrawHandle, -}; - -/// A simple wrapper around a single audio clip. -pub struct AudioWrapper { - music: Music, -} - -impl AudioWrapper { - /// Create a new AudioWrapper from a `Music` struct - pub fn new(music: Music) -> Self { - Self { music } - } - - /// Begin playing the audio - pub fn play(&mut self, audio_handle: &mut RaylibAudio) { - audio_handle.play_music_stream(&mut self.music); - } - - /// Stop playing the audio - pub fn stop(&mut self, audio_handle: &mut RaylibAudio) { - audio_handle.stop_music_stream(&mut self.music); - } - - /// Pause the audio - pub fn pause(&mut self, audio_handle: &mut RaylibAudio) { - audio_handle.pause_music_stream(&mut self.music); - } - - /// Call this every frame - pub fn update(&mut self, audio_handle: &mut RaylibAudio) { - audio_handle.update_music_stream(&mut self.music); - } - - /// Check if this audio clip is playing - pub fn is_playing(&mut self, audio_handle: &mut RaylibAudio) -> bool { - return audio_handle.is_music_playing(&mut self.music); - } -} diff --git a/src/lib/wrappers/audio.rs.old b/src/lib/wrappers/audio.rs.old new file mode 100644 index 0000000..d16baab --- /dev/null +++ b/src/lib/wrappers/audio.rs.old @@ -0,0 +1,41 @@ +// use raylib::{ +// audio::{Music, RaylibAudio}, +// prelude::RaylibDrawHandle, +// }; + +// /// A simple wrapper around a single audio clip. +// pub struct AudioWrapper { +// music: Music, +// } + +// impl AudioWrapper { +// /// Create a new AudioWrapper from a `Music` struct +// pub fn new(music: Music) -> Self { +// Self { music } +// } + +// /// Begin playing the audio +// pub fn play(&mut self, audio_handle: &mut RaylibAudio) { +// audio_handle.play_music_stream(&mut self.music); +// } + +// /// Stop playing the audio +// pub fn stop(&mut self, audio_handle: &mut RaylibAudio) { +// audio_handle.stop_music_stream(&mut self.music); +// } + +// /// Pause the audio +// pub fn pause(&mut self, audio_handle: &mut RaylibAudio) { +// audio_handle.pause_music_stream(&mut self.music); +// } + +// /// Call this every frame +// pub fn update(&mut self, audio_handle: &mut RaylibAudio) { +// audio_handle.update_music_stream(&mut self.music); +// } + +// /// Check if this audio clip is playing +// pub fn is_playing(&mut self, audio_handle: &mut RaylibAudio) -> bool { +// return audio_handle.is_music_playing(&mut self.music); +// } +// } diff --git a/src/lib/wrappers/audio/mod.rs b/src/lib/wrappers/audio/mod.rs new file mode 100644 index 0000000..d44230b --- /dev/null +++ b/src/lib/wrappers/audio/mod.rs @@ -0,0 +1 @@ +pub mod player; \ No newline at end of file diff --git a/src/lib/wrappers/audio/player.rs b/src/lib/wrappers/audio/player.rs new file mode 100644 index 0000000..4ca9a25 --- /dev/null +++ b/src/lib/wrappers/audio/player.rs @@ -0,0 +1,41 @@ +use raylib::audio::RaylibAudio; + +/// A thin wrapper around `raylib::core::audio::RaylibAudio` that keeps track of the volume of its audio channels. +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 + } +} diff --git a/src/logic/loadingscreen.rs b/src/logic/loadingscreen.rs new file mode 100644 index 0000000..05520c4 --- /dev/null +++ b/src/logic/loadingscreen.rs @@ -0,0 +1,13 @@ +use raylib::prelude::*; + +use crate::gamecore::{GameCore, GameState}; + + +pub fn handle_loading_screen(draw_handle: &mut RaylibDrawHandle, game_core: &mut GameCore) -> Option { + + // Clear frame + draw_handle.clear_background(Color::WHITE); + + + return None; +} \ No newline at end of file diff --git a/src/logic/mainmenu.rs b/src/logic/mainmenu.rs new file mode 100644 index 0000000..c393a13 --- /dev/null +++ b/src/logic/mainmenu.rs @@ -0,0 +1,11 @@ +use raylib::prelude::*; + +use crate::gamecore::{GameCore, GameState}; + + +pub fn handle_main_menu(draw_handle: &mut RaylibDrawHandle, game_core: &mut GameCore) -> Option{ + + + return None; + +} \ No newline at end of file diff --git a/src/logic/mod.rs b/src/logic/mod.rs new file mode 100644 index 0000000..e4105ac --- /dev/null +++ b/src/logic/mod.rs @@ -0,0 +1,2 @@ +pub mod loadingscreen; +pub mod mainmenu; \ No newline at end of file diff --git a/src/main.rs b/src/main.rs index cb827f0..984e328 100644 --- a/src/main.rs +++ b/src/main.rs @@ -1,8 +1,11 @@ -mod lib; mod gamecore; +mod lib; +mod logic; +mod resources; use gamecore::{GameCore, GameState}; -use lib::profiler::GameProfiler; +use lib::{utils::profiler::GameProfiler, wrappers::audio::player::AudioPlayer}; +use logic::{loadingscreen::handle_loading_screen, mainmenu::handle_main_menu}; use raylib::prelude::*; // Game Launch Configuration @@ -21,31 +24,49 @@ fn main() { .build(); raylib.set_target_fps(MAX_FPS); + // Override the default exit key + raylib.set_exit_key(None); + // Set up the game's core state - let mut game_core = GameCore{ - state: GameState::Loading - }; + let mut game_core = GameCore::new(); // Set up the game's profiler let mut profiler = GameProfiler::new(); profiler.start(); + // Init the audio subsystem + let mut audio_system = AudioPlayer::new(RaylibAudio::init_audio_device()); + // Main rendering loop while !raylib.window_should_close() { let mut draw_handle = raylib.begin_drawing(&raylib_thread); - // Clear frame - draw_handle.clear_background(Color::WHITE); - // Call appropriate render function - // TODO: the usual match statement on `game_core.state` + let new_state: Option = match game_core.state { + GameState::Loading => handle_loading_screen(&mut draw_handle, &mut game_core), + GameState::MainMenu => handle_main_menu(&mut draw_handle, &mut game_core), + }; + + if new_state.is_some() { + game_core.state = new_state.unwrap(); + game_core.last_state_change_time = draw_handle.get_time(); + } // Feed the profiler - profiler.seconds_per_frame = draw_handle.get_frame_time(); - profiler.frames_per_second = draw_handle.get_fps(); + // This only runs in the dev profile. + #[cfg(debug_assertions)] + { + // Update all data + profiler.data.seconds_per_frame = draw_handle.get_frame_time(); + profiler.data.frames_per_second = draw_handle.get_fps(); + profiler.data.monitor_count = raylib::core::window::get_monitor_count(); + profiler.data.audio_volume = audio_system.get_master_volume(); + profiler.data.active_sounds = audio_system.get_sounds_playing(); + profiler.data.game_state = game_core.state.to_string(); - // Send telemetry data - profiler.update(); + // Send telemetry data + profiler.update(); + } } // Cleanup diff --git a/src/resources.rs b/src/resources.rs new file mode 100644 index 0000000..9133592 --- /dev/null +++ b/src/resources.rs @@ -0,0 +1,11 @@ +use failure::Error; + +/// This struct contains all textures and sounds that must be loaded into (V)RAM at the start of the game +pub struct GlobalResources {} + +impl GlobalResources { + /// Load all resources. **THIS WILL HANG!** + pub fn load_all(&mut self) -> Result { + Ok(GlobalResources {}) + } +}