From 32ec6c43cc75f0b7908e7c1c90fe968a9970f407 Mon Sep 17 00:00:00 2001 From: Evan Pratten Date: Tue, 29 Mar 2022 14:13:18 -0400 Subject: [PATCH] wip --- .gitignore | 2 + .vscode/tasks.json | 12 ++ .../test_debugTexture.anim_meta.json | 15 ++ .../test_debugTexture/test_debugTexture.png | Bin 0 -> 2599 bytes game/dist/known-sprite-types.json | 4 + game/game_logic/Cargo.toml | 7 +- game/game_logic/src/asset_manager/mod.rs | 6 +- .../src/asset_manager/sprite_types.rs | 24 +++ game/game_logic/src/asset_manager/texture.rs | 64 ++++++++ game/game_logic/src/lib.rs | 2 +- game/game_logic/src/rendering/event_loop.rs | 20 ++- .../src/rendering/utilities/anim_texture.rs | 148 +++++++++++++++++- game/game_logic/src/scenes/mod.rs | 4 +- game/game_logic/src/scenes/test_fox.rs | 23 ++- renderdoc_settings.cap | 27 ++++ third_party/raylib-rs | 2 +- 16 files changed, 342 insertions(+), 18 deletions(-) create mode 100644 game/dist/assets/anm/test/test_debugTexture/test_debugTexture.anim_meta.json create mode 100644 game/dist/assets/anm/test/test_debugTexture/test_debugTexture.png create mode 100644 game/game_logic/src/asset_manager/sprite_types.rs create mode 100644 game/game_logic/src/asset_manager/texture.rs create mode 100644 renderdoc_settings.cap diff --git a/.gitignore b/.gitignore index f4ade874..8c90d182 100644 --- a/.gitignore +++ b/.gitignore @@ -167,3 +167,5 @@ cython_debug/ # and can be added to the global gitignore or merged into this file. For a more nuclear # option (not recommended) you can uncomment the following to ignore the entire idea folder. #.idea/ + +/screenshot*.png \ No newline at end of file diff --git a/.vscode/tasks.json b/.vscode/tasks.json index 472de6b2..388ab5f5 100644 --- a/.vscode/tasks.json +++ b/.vscode/tasks.json @@ -18,6 +18,18 @@ ], "label": "Launch Game" }, + { + "type": "cargo", + "command": "run", + "args": [ + "--", + "--verbose" + ], + "problemMatcher": [ + "$rustc" + ], + "label": "Launch Game [DEBUG LOGS]" + }, { "type": "cargo", "command": "doc", diff --git a/game/dist/assets/anm/test/test_debugTexture/test_debugTexture.anim_meta.json b/game/dist/assets/anm/test/test_debugTexture/test_debugTexture.anim_meta.json new file mode 100644 index 00000000..7a85d82a --- /dev/null +++ b/game/dist/assets/anm/test/test_debugTexture/test_debugTexture.anim_meta.json @@ -0,0 +1,15 @@ +{ + "sheet_height": 64, + "sheet_width": 64, + "published_at": "2022-03-29 16:46:34", + "published_by": "ewpratten", + "fps": 24.0, + "frames": [ + { + "x": 0, + "y": 0, + "width": 64, + "height": 64 + } + ] +} \ No newline at end of file diff --git a/game/dist/assets/anm/test/test_debugTexture/test_debugTexture.png b/game/dist/assets/anm/test/test_debugTexture/test_debugTexture.png new file mode 100644 index 0000000000000000000000000000000000000000..63706c1ddbe6f384185daf742dafbef5f543bf12 GIT binary patch literal 2599 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 Result, serde_json::Error> { + // Load the json file from the embedded data as a string + let data = InternalData::get("known-sprite-types.json").unwrap().data; + + // Deserialize the json string into a rust structure + let json_structure: Vec = serde_json::from_slice(&data)?; + Ok(json_structure) +} diff --git a/game/game_logic/src/asset_manager/texture.rs b/game/game_logic/src/asset_manager/texture.rs new file mode 100644 index 00000000..36a819d8 --- /dev/null +++ b/game/game_logic/src/asset_manager/texture.rs @@ -0,0 +1,64 @@ +//! Code for loading textures from RAM to VRAM +//! +//! Largely coppied from last year: https://github.com/Ewpratten/ludum-dare-49/blob/master/game/src/utilities/datastore.rs + +use std::path::Path; + +use raylib::{texture::Texture2D, RaylibHandle, RaylibThread}; +use tempfile::tempdir; + +use crate::asset_manager::InternalData; + +#[derive(Debug, thiserror::Error)] +pub enum ResourceLoadError { + #[error(transparent)] + Io(#[from] std::io::Error), + #[error("Could not load embedded asset: {0}")] + AssetNotFound(String), + #[error("Generic error: {0}")] + Generic(String), +} + +/// Loads an embedded texture into VRAM. +/// +/// # Technical Info +/// In this application, we are using `rust_embed` to embed static assets directly inside the executable. +/// This has the limitation of none of the assets being "real files", which causes an issue with Raylib. +/// Raylib requires a "real file" in order to load data into VRAM (without digging into `unsafe` dark magic). +/// The solution is to temporarily write the assets to disk, and then load them from disk. +/// We must also preserve the file extension, so the Raylib file loader can parse them correctly. +pub fn load_texture_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 image data to a real file on the local filesystem so raylib will read it correctly + std::fs::write( + &tmp_path, + &InternalData::get(path) + .ok_or(ResourceLoadError::AssetNotFound(path.to_string()))? + .data, + )?; + + // Call through via FFI to re-load the file + let texture = raylib_handle + .load_texture(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) +} diff --git a/game/game_logic/src/lib.rs b/game/game_logic/src/lib.rs index a4685174..02137318 100644 --- a/game/game_logic/src/lib.rs +++ b/game/game_logic/src/lib.rs @@ -73,7 +73,7 @@ pub async fn entrypoint(force_recreate_savefiles: bool) { |builder| { builder .msaa_4x() - .vsync() + // .vsync() .title(project_constants.game_name.as_str()) .height(project_constants.base_window_size.1 as i32) .width(project_constants.base_window_size.0 as i32); diff --git a/game/game_logic/src/rendering/event_loop.rs b/game/game_logic/src/rendering/event_loop.rs index b067b6db..54aa6d00 100644 --- a/game/game_logic/src/rendering/event_loop.rs +++ b/game/game_logic/src/rendering/event_loop.rs @@ -15,6 +15,8 @@ use crate::rendering::core_renderer_sm::{PreloadState, RenderBackendStates}; use crate::rendering::screens::sm_failure_screen; use crate::scenes::SceneRenderDelegate; use raylib::RaylibBuilder; +use raylib::consts::KeyboardKey; +use raylib::prelude::RaylibDraw; /// Will begin rendering graphics. Returns when the window closes pub async fn handle_graphics_blocking( @@ -45,7 +47,8 @@ pub async fn handle_graphics_blocking( let mut sm_failure_screen = sm_failure_screen::SmFailureScreen::new(); // Set up the main render delegate - let mut render_delegate = SceneRenderDelegate::on_game_start(); + let mut render_delegate = + SceneRenderDelegate::on_game_start(&mut raylib_handle, &raylib_thread); // Handle loading the resources and rendering the loading screen log::trace!("Running event loop"); @@ -74,13 +77,16 @@ pub async fn handle_graphics_blocking( // Tell the profiler that we ended the frame profiling::finish_frame!(); } - log::trace!("Finished loading game"); + log::info!("Finished loading game"); // Get access to the global resources let global_resources = loading_screen .resources .expect("Failed to get global resources"); + // Tracker for if we are showing the FPS counter + let mut show_fps_counter = false; + // Run the event loop while !raylib_handle.window_should_close() { // Handle state machine updates @@ -106,6 +112,16 @@ pub async fn handle_graphics_blocking( _ => backend_sm = RenderBackendStates::sm_failed(), }; + // Check for F3 being pressed + if raylib_handle.is_key_pressed(KeyboardKey::KEY_F3) { + show_fps_counter = !show_fps_counter; + } + + // Show the FPS counter + if show_fps_counter { + raylib_handle.begin_drawing(&raylib_thread).draw_fps(10, 10); + } + // Tell the profiler that we ended the frame profiling::finish_frame!(); } diff --git a/game/game_logic/src/rendering/utilities/anim_texture.rs b/game/game_logic/src/rendering/utilities/anim_texture.rs index 42bb4e79..2d5e85da 100644 --- a/game/game_logic/src/rendering/utilities/anim_texture.rs +++ b/game/game_logic/src/rendering/utilities/anim_texture.rs @@ -1,23 +1,157 @@ //! This module handles the code for rendering framerate-locked animations from textures -use raylib::texture::Texture2D; +use nalgebra::Vector2; +use raylib::{ + color::Color, + math::Rectangle, + prelude::{RaylibDraw, RaylibDrawHandle}, + texture::Texture2D, + RaylibHandle, RaylibThread, +}; +use serde::Deserialize; + +use crate::asset_manager::{ + load_json_structure, load_known_sprite_types, load_texture_from_internal_data, + InternalJsonLoadError, +}; + +/// Possible errors to be thrown during the animation texture loading process +#[derive(Debug, thiserror::Error)] +pub enum AnimatedTextureLoadError { + #[error(transparent)] + MetadataLoadError(#[from] InternalJsonLoadError), + #[error(transparent)] + KnownSpriteTypesLoadError(#[from] serde_json::Error), + #[error("Invalid Sprite Type: {0}")] + InvalidSpriteType(String), + #[error(transparent)] + TextureLoadError(#[from] crate::asset_manager::ResourceLoadError), +} + +/// Definition for the structure describing a frame's size and position in a texture +#[derive(Debug, Clone, Deserialize)] +struct FrameTextureDescriptor { + pub x: f32, + pub y: f32, + pub width: f32, + pub height: f32, +} + +impl Into for FrameTextureDescriptor { + fn into(self) -> Rectangle { + Rectangle::new(self.x, self.y, self.width, self.height) + } +} + +/// Definition for the metadata structure attached to each spritesheet +#[derive(Debug, Clone, Deserialize)] +struct AnimatedTextureMetadata { + pub sheet_height: u64, + pub sheet_width: u64, + pub fps: f32, + pub frames: Vec, +} #[derive(Debug)] pub struct AnimatedTexture { texture: Texture2D, - target_fps: f32, + texture_metadata: AnimatedTextureMetadata, + // a list of source rects to reduce memory allocation needs during render time + texture_source_rects: Vec, } impl AnimatedTexture { /// Construct a new `AnimatedTexture` - pub fn new(texture: Texture2D, target_frames_per_second: f32) -> Self { - Self { - texture, - target_fps: target_frames_per_second, + #[profiling::function] + pub fn new( + raylib_handle: &mut RaylibHandle, + thread: &RaylibThread, + sprite_type: &str, + sprite_name: &str, + ) -> Result { + // Try to convert the sprite type string to a real type + let known_sprite_types = load_known_sprite_types()?; + let sprite_type_obj = known_sprite_types.iter().find(|known_sprite_type| { + known_sprite_type.short_name == sprite_type + || known_sprite_type.friendly_name == sprite_type + }); + if let None = sprite_type_obj { + error!("Invalid sprite type supplied: {}", sprite_type); + return Err(AnimatedTextureLoadError::InvalidSpriteType( + sprite_type.to_string(), + )); } + let sprite_type_obj = sprite_type_obj.unwrap(); + + // Now, we can construct the paths to the texture and metadata + let parent_dir_path = format!( + "assets/anm/{}/{}_{}", + sprite_type_obj.short_name, sprite_type_obj.short_name, sprite_name + ); + let metadata_file_path = format!( + "{}/{}_{}.anim_meta.json", + parent_dir_path, sprite_type_obj.short_name, sprite_name + ); + let texture_file_path = format!( + "{}/{}_{}.png", + parent_dir_path, sprite_type_obj.short_name, sprite_name + ); + + // Attempt to load the metadata + let texture_metadata: AnimatedTextureMetadata = load_json_structure(&metadata_file_path)?; + let source_rects = texture_metadata + .frames + .iter() + .map(|frame_descriptor| frame_descriptor.clone().into()) + .collect(); + + // Attempt to load the texture itself + let texture = load_texture_from_internal_data(raylib_handle, thread, &texture_file_path)?; + + Ok(Self { + texture, + texture_metadata, + texture_source_rects: source_rects, + }) } - pub fn render_frame_by_index(&self, index: usize) { + #[profiling::function] + pub fn render_frame_by_index( + &self, + draw_handle: &mut RaylibDrawHandle, + index: usize, + position: Vector2, + percent_scale: Option>, + origin: Option>, + rotation: Option, + tint: Option, + ) { + // Get the frame-specific metadata + let metadata = &self.texture_metadata.frames[index]; + // Build a source rectangle + let source = self.texture_source_rects[index]; + + // Build a destination rectangle + let scaler = percent_scale.unwrap_or(Vector2::new(1.0, 1.0)); + let destination = Rectangle::new( + position.x, + position.y, + metadata.width * scaler.x, + metadata.height * scaler.y, + ); + let origin: raylib::core::math::Vector2 = + origin.unwrap_or_else(|| Vector2::::zeros()).into(); + debug!("{:?} -> {:?}", source, destination); + + // Render the frame + draw_handle.draw_texture_pro( + &self.texture, + source, + destination, + origin, + rotation.unwrap_or(0.0), + tint.unwrap_or(Color::WHITE), + ); } } diff --git a/game/game_logic/src/scenes/mod.rs b/game/game_logic/src/scenes/mod.rs index f264f964..6fcbb5c7 100644 --- a/game/game_logic/src/scenes/mod.rs +++ b/game/game_logic/src/scenes/mod.rs @@ -19,11 +19,11 @@ pub struct SceneRenderDelegate { impl SceneRenderDelegate { /// This is called when the game first loads - pub fn on_game_start() -> Self { + pub fn on_game_start(raylib: &mut RaylibHandle, rl_thread: &RaylibThread) -> Self { // TODO: Stick any init code you want here. // Init some scenes - let scene_test_fox = TestFoxScene::new(); + let scene_test_fox = TestFoxScene::new(raylib, rl_thread); Self { scene_test_fox } } diff --git a/game/game_logic/src/scenes/test_fox.rs b/game/game_logic/src/scenes/test_fox.rs index e575a181..297c610a 100644 --- a/game/game_logic/src/scenes/test_fox.rs +++ b/game/game_logic/src/scenes/test_fox.rs @@ -1,7 +1,8 @@ //! This "scene" is used only for testing animation and resource loading //! It should be removed once the game is being worked on -use raylib::{RaylibHandle, RaylibThread}; +use raylib::prelude::*; +use nalgebra as na; use crate::{ discord::DiscordChannel, global_resource_package::GlobalResources, @@ -15,9 +16,9 @@ pub struct TestFoxScene { impl TestFoxScene { /// Construct a new `TestFoxScene` - pub fn new() -> Self { + pub fn new(raylib_handle: &mut RaylibHandle, thread: &RaylibThread) -> Self { // Load the fox texture - let fox = AnimatedTexture::new(); + let fox = AnimatedTexture::new(raylib_handle, thread, "test", "debugTexture").unwrap(); Self { fox_animation: fox } } @@ -30,5 +31,21 @@ impl TestFoxScene { discord: &DiscordChannel, global_resources: &GlobalResources, ) { + // Get a drawing handle + let mut draw = raylib.begin_drawing(rl_thread); + + // Clear the screen + draw.clear_background(Color::WHITE); + + // Render the fox + self.fox_animation.render_frame_by_index( + &mut draw, + 0, + na::Vector2::new(0.0, 0.0), + None, + None, + None, + None, + ); } } diff --git a/renderdoc_settings.cap b/renderdoc_settings.cap new file mode 100644 index 00000000..309662ba --- /dev/null +++ b/renderdoc_settings.cap @@ -0,0 +1,27 @@ +{ + "rdocCaptureSettings": 1, + "settings": { + "autoStart": false, + "commandLine": "--verbose", + "environment": [ + ], + "executable": "/home/ewpratten/projects/ludum-dare-50/target/debug/desktop_wrapper", + "inject": false, + "numQueuedFrames": 0, + "options": { + "allowFullscreen": true, + "allowVSync": true, + "apiValidation": false, + "captureAllCmdLists": false, + "captureCallstacks": false, + "captureCallstacksOnlyDraws": false, + "debugOutputMute": true, + "delayForDebugger": 0, + "hookIntoChildren": false, + "refAllResources": false, + "verifyBufferAccess": false + }, + "queuedFrameCap": 0, + "workingDir": "/home/ewpratten/projects/ludum-dare-50" + } +} diff --git a/third_party/raylib-rs b/third_party/raylib-rs index 3aff1382..abae275a 160000 --- a/third_party/raylib-rs +++ b/third_party/raylib-rs @@ -1 +1 @@ -Subproject commit 3aff138276b374f5e07187a652a71d9eb59e97d1 +Subproject commit abae275a63ee527cfe16d35a7d00d7532426d5a5