diff --git a/assets/ewpratten/chr_testFox/preview-17.jpg b/assets/ewpratten/chr_testFox/preview-17.jpg deleted file mode 100644 index 0063c74f..00000000 Binary files a/assets/ewpratten/chr_testFox/preview-17.jpg and /dev/null differ diff --git a/assets/ewpratten/chr_testFox/preview-18.jpg b/assets/ewpratten/chr_testFox/preview-18.jpg deleted file mode 100644 index ebde156c..00000000 Binary files a/assets/ewpratten/chr_testFox/preview-18.jpg and /dev/null differ diff --git a/automation/anim_stitcher/stitcher.py b/automation/anim_stitcher/stitcher.py index 055fa611..9f92d04f 100644 --- a/automation/anim_stitcher/stitcher.py +++ b/automation/anim_stitcher/stitcher.py @@ -35,6 +35,10 @@ def check_sprite_exists(sprite_type: str, sprite_name: str) -> bool: def stitch_images_and_write_to_disk(sprite_type: str, sprite_name: str, images: List[str], quantize: bool, fps: float, reverse_order: bool) -> None: + # Handle reverse order + if reverse_order: + images = images[::-1] + # Load all the images images_to_stitch = [] for image_path in images: @@ -74,13 +78,9 @@ def stitch_images_and_write_to_disk(sprite_type: str, sprite_name: str, images: "frames": [] } - # Handle reverse order - if reverse_order: - images_to_stitch.reverse() - # Add the metadata for each image x_offset = 0 - for image in images_to_stitch: + for i in range(len(images_to_stitch)): metadata["frames"].append({ "x": x_offset, "y": 0, diff --git a/game/dist/assets/anm/chr/chr_testFox/chr_testFox.anim_meta.json b/game/dist/assets/anm/chr/chr_testFox/chr_testFox.anim_meta.json index 7333301b..d0457605 100644 --- a/game/dist/assets/anm/chr/chr_testFox/chr_testFox.anim_meta.json +++ b/game/dist/assets/anm/chr/chr_testFox/chr_testFox.anim_meta.json @@ -1,7 +1,7 @@ { "sheet_height": 338, - "sheet_width": 10800, - "published_at": "2022-03-29 18:21:28", + "sheet_width": 9600, + "published_at": "2022-03-29 18:40:22", "published_by": "ewpratten", "fps": 24.0, "frames": [ @@ -100,18 +100,6 @@ "y": 0, "width": 600, "height": 338 - }, - { - "x": 9600, - "y": 0, - "width": 600, - "height": 338 - }, - { - "x": 10200, - "y": 0, - "width": 600, - "height": 338 } ] } \ No newline at end of file diff --git a/game/dist/assets/anm/chr/chr_testFox/chr_testFox.png b/game/dist/assets/anm/chr/chr_testFox/chr_testFox.png index ee8e60ca..5d82aca2 100644 Binary files a/game/dist/assets/anm/chr/chr_testFox/chr_testFox.png and b/game/dist/assets/anm/chr/chr_testFox/chr_testFox.png differ diff --git a/game/game_logic/src/rendering/utilities/anim_texture.rs b/game/game_logic/src/rendering/utilities/anim_texture.rs index 68c50611..bbe593de 100644 --- a/game/game_logic/src/rendering/utilities/anim_texture.rs +++ b/game/game_logic/src/rendering/utilities/anim_texture.rs @@ -1,5 +1,6 @@ //! This module handles the code for rendering framerate-locked animations from textures +use chrono::{DateTime, Utc}; use nalgebra::Vector2; use raylib::{ color::Color, @@ -54,14 +55,20 @@ struct AnimatedTextureMetadata { #[derive(Debug)] pub struct AnimatedTexture { + /// The whole internal spritesheet texture: Texture2D, + /// The metadata describing the spritesheet texture_metadata: AnimatedTextureMetadata, - // a list of source rects to reduce memory allocation needs during render time + /// a list of source rects to reduce memory allocation needs during render time texture_source_rects: Vec, + /// The animation start timestamp + start_time: Option>, } impl AnimatedTexture { /// Construct a new `AnimatedTexture` + /// + /// This will load all resources from RAM or disk. May take a while. #[profiling::function] pub fn new( raylib_handle: &mut RaylibHandle, @@ -112,9 +119,11 @@ impl AnimatedTexture { texture, texture_metadata, texture_source_rects: source_rects, + start_time: None, }) } + /// Render a single frame to the screen #[profiling::function] pub fn render_frame_by_index( &self, @@ -154,4 +163,57 @@ impl AnimatedTexture { tint.unwrap_or(Color::WHITE), ); } + + /// Clear the internal tracker for when the animation started + /// + /// This will bring the animation back to frame 1. Useful for non-looping rendering + pub fn reset_animation(&mut self) { + self.start_time = None; + } + + /// Get the current frame index + pub fn get_current_frame_index(&self) -> Option { + self.start_time.map(|start_time| { + let elapsed_time_ms = Utc::now() + .signed_duration_since(start_time) + .num_milliseconds() as f32; + let elapsed_time_s = elapsed_time_ms / 1000.0; + let frame_index = (elapsed_time_s * self.texture_metadata.fps) as usize; + frame_index % self.texture_metadata.frames.len() + }) + } + + /// Render the animation based on timestamp + pub fn render_automatic( + &mut self, + draw_handle: &mut RaylibDrawHandle, + position: Vector2, + percent_scale: Option>, + origin: Option>, + rotation: Option, + tint: Option, + ) { + // If this is the first time we're rendering, set the start time + if self.start_time.is_none() { + self.start_time = Some(Utc::now()); + } + + // Get the current frame index + let current_frame_index = self.get_current_frame_index(); + + // If we have a valid index, render it + if let Some(current_frame_index) = current_frame_index { + self.render_frame_by_index( + draw_handle, + current_frame_index, + position, + percent_scale, + origin, + rotation, + tint, + ); + } else { + warn!("We somehow got a frame index of None"); + } + } } diff --git a/game/game_logic/src/scenes/test_fox.rs b/game/game_logic/src/scenes/test_fox.rs index fb089b85..526dc57e 100644 --- a/game/game_logic/src/scenes/test_fox.rs +++ b/game/game_logic/src/scenes/test_fox.rs @@ -38,9 +38,8 @@ impl TestFoxScene { draw.clear_background(Color::WHITE); // Render the fox - self.fox_animation.render_frame_by_index( + self.fox_animation.render_automatic( &mut draw, - 0, na::Vector2::new(0.0, 0.0), None, None,