use bevy::prelude::*;
use game_library::events::CastSpell;
use super::avatar::PlayerAvatar;
#[derive(Debug, Clone, Copy, Default, Eq, PartialEq, Hash, States)]
pub(super) enum PlayerAnimation {
#[default]
Idle,
Walking,
}
#[derive(Debug, Clone, Copy, Default, Eq, PartialEq, Hash, States)]
pub(super) enum PlayerAnimationSupplemental {
#[default]
None,
Casting,
}
#[derive(Debug, Clone, Copy, Default, Eq, PartialEq, Hash, States)]
pub(super) enum PlayerFacing {
#[default]
Left,
Right,
}
#[derive(Debug, Default, Resource)]
pub(super) struct PlayerAnimationTimer(pub Timer);
const WALKING_ANIMATION: [usize; 4] = [0, 1, 0, 2];
const CASTING_WALKING_ANIMATION: [usize; 4] = [5, 6, 7, 8];
const IDLE_ANIMATION: [usize; 1] = [3];
const CASTING_IDLE_ANIMATION: [usize; 1] = [4];
#[derive(Debug, Default, Resource, Clone, Copy, PartialEq, Eq, Hash)]
pub(super) struct AnimationFrame(pub usize);
impl AnimationFrame {
pub const MAX: usize = 3;
pub fn reset(&mut self) {
self.0 = 0;
}
pub fn next(&mut self) {
self.0 = (self.0 + 1) % Self::MAX;
}
#[must_use]
pub const fn get(self, length: usize) -> usize {
self.0 % length
}
#[must_use]
pub const fn is_fin(self) -> bool {
self.0 == Self::MAX - 1
}
}
pub(super) fn set_casting_animation(
mut supplemental_state_next: ResMut<NextState<PlayerAnimationSupplemental>>,
mut er_cast_spell: EventReader<CastSpell>,
mut frame: ResMut<AnimationFrame>,
) {
if er_cast_spell.read().next().is_some() {
supplemental_state_next.set(PlayerAnimationSupplemental::Casting);
frame.reset();
}
}
pub(super) fn advance_animation_timer(
time: Res<Time>,
mut timer: ResMut<PlayerAnimationTimer>,
mut frame: ResMut<AnimationFrame>,
) {
if !timer.0.tick(time.delta()).just_finished() {
return;
}
frame.next();
}
pub fn update_avatar_animation(
mut sprite_query: Query<&mut TextureAtlasSprite, With<PlayerAvatar>>,
state: Res<State<PlayerAnimation>>,
supplemental_state: Res<State<PlayerAnimationSupplemental>>,
facing: Res<State<PlayerFacing>>,
mut supplemental_state_next: ResMut<NextState<PlayerAnimationSupplemental>>,
frame: Res<AnimationFrame>,
) {
let Ok(mut sprite) = sprite_query.get_single_mut() else {
tracing::error!("update_avatar_animation: failed to get player sprite");
return;
};
sprite.flip_x = *facing.get() == PlayerFacing::Left;
match (state.get(), supplemental_state.get()) {
(PlayerAnimation::Walking, PlayerAnimationSupplemental::None) => {
sprite.index = WALKING_ANIMATION[frame.get(WALKING_ANIMATION.len())];
}
(PlayerAnimation::Walking, PlayerAnimationSupplemental::Casting) => {
sprite.index = CASTING_WALKING_ANIMATION[frame.get(CASTING_WALKING_ANIMATION.len())];
}
(PlayerAnimation::Idle, PlayerAnimationSupplemental::None) => {
sprite.index = IDLE_ANIMATION[frame.get(IDLE_ANIMATION.len())];
}
(PlayerAnimation::Idle, PlayerAnimationSupplemental::Casting) => {
sprite.index = CASTING_IDLE_ANIMATION[frame.get(CASTING_IDLE_ANIMATION.len())];
}
}
if frame.is_fin() && supplemental_state.get() == &PlayerAnimationSupplemental::Casting {
supplemental_state_next.set(PlayerAnimationSupplemental::None);
}
}