use bevy::reflect::Reflect;
use serde_default_utils::{default_i32, default_usize};
use std::{any::Any, hash::Hash};
use crate::{
data_loader::DataFile,
enums::{
CastCategory, CastSlot, CastType, GameSystem, MagicType, ParticleAttachment, Skill,
SpellCollision,
},
shared_traits::KnownCastSlot,
InternalId, StatEffect,
};
#[derive(Debug, Clone, serde::Serialize, serde::Deserialize, Reflect)]
#[serde(rename_all = "camelCase")]
pub struct SpellData {
pub internal_id: Option<String>,
pub name: String,
pub description: String,
#[serde(default = "String::new")]
pub long_description: String,
pub spell_tier: usize,
pub magic: MagicType,
pub cast_slot: CastSlot,
#[serde(default = "spell_defaults::collision")]
pub collision: SpellCollision,
#[serde(default = "spell_defaults::cast_type")]
pub cast_type: CastType,
#[serde(default = "spell_defaults::cast_category")]
pub cast_category: CastCategory,
#[serde(default = "spell_defaults::placeholder_png_path")]
pub icon_tileset: String,
#[serde(default = "default_usize::<0>")]
pub icon_index: usize,
#[serde(default = "spell_defaults::placeholder_png_path")]
pub sprite_tileset: String,
#[serde(default = "default_usize::<0>")]
pub sprite_index: usize,
#[serde(default = "spell_defaults::spell_cooldown")]
pub cooldown: f32,
#[serde(default = "spell_defaults::spell_cast_time")]
pub cast_time: f32,
#[serde(default = "default_usize::<0>")]
pub mana_cost: usize,
#[serde(default = "spell_defaults::spell_range")]
pub range: f32,
#[serde(default = "spell_defaults::spell_speed")]
pub speed: f32,
#[serde(default = "spell_defaults::spell_duration")]
pub duration: f32,
#[serde(default = "default_i32::<0>")]
pub damage: i32,
#[serde(default = "default_i32::<0>")]
pub healing: i32,
#[serde(default = "default_i32::<0>")]
pub radius: i32,
#[serde(default = "default_i32::<0>")]
pub angle: i32,
#[serde(default = "Vec::new")]
pub buffs: Vec<StatEffect>,
#[serde(default = "Vec::new")]
pub debuffs: Vec<StatEffect>,
#[serde(default = "Vec::new")]
pub particles: Vec<SpellParticles>,
}
impl Hash for SpellData {
fn hash<H: std::hash::Hasher>(&self, state: &mut H) {
self.name.hash(state);
self.spell_tier.hash(state);
self.magic.hash(state);
self.cast_slot.hash(state);
}
}
impl InternalId for SpellData {
fn update_internal_id(&mut self) {
self.internal_id = Some(self.get_internal_id());
}
#[must_use]
fn get_internal_id(&self) -> String {
if self.internal_id.is_some() {
let id = self.internal_id.clone().unwrap_or_default();
if !id.is_empty() {
return id;
}
}
format!(
"{}{}{}{}",
self.name.replace(' ', ""),
self.spell_tier,
self.magic,
self.cast_slot
)
}
}
#[derive(Debug, Clone, serde::Serialize, serde::Deserialize, Reflect)]
#[serde(rename_all = "camelCase")]
pub struct SpellParticles {
pub particle_id: String,
pub attachment: ParticleAttachment,
}
impl KnownCastSlot for SpellData {
fn cast_slot(&self) -> CastSlot {
self.cast_slot
}
}
impl SpellData {
#[must_use]
pub fn skill(&self) -> Skill {
self.magic.into()
}
#[must_use]
pub fn texture_atlas_index(&self) -> bevy::sprite::TextureAtlasSprite {
bevy::sprite::TextureAtlasSprite::new(self.sprite_index)
}
}
mod spell_defaults {
use crate::enums::{CastCategory, CastType, SpellCollision};
pub(super) const fn collision() -> SpellCollision {
SpellCollision::Point
}
pub(super) const fn cast_type() -> CastType {
CastType::Instant
}
pub(super) const fn cast_category() -> CastCategory {
CastCategory::Projectile
}
pub(super) fn placeholder_png_path() -> String {
"placeholder.png".to_string()
}
pub(super) const fn spell_cooldown() -> f32 {
1.0
}
pub(super) const fn spell_cast_time() -> f32 {
0.0
}
pub(super) const fn spell_range() -> f32 {
5.0
}
pub(super) const fn spell_speed() -> f32 {
1.0
}
pub(super) const fn spell_duration() -> f32 {
5.0
}
}
impl<D: Hash + InternalId + 'static> TryInto<SpellData> for DataFile<D> {
type Error = ();
fn try_into(self) -> Result<SpellData, Self::Error> {
if self.header.system != GameSystem::Spell {
return Err(());
}
(&self.data as &dyn Any)
.downcast_ref::<SpellData>()
.cloned()
.ok_or(())
}
}
impl<D: Hash + InternalId + 'static> TryFrom<&DataFile<D>> for SpellData {
type Error = ();
fn try_from(data_file: &DataFile<D>) -> Result<Self, Self::Error> {
if data_file.header.system != GameSystem::Tileset {
return Err(());
}
(&data_file.data as &dyn Any)
.downcast_ref::<Self>()
.cloned()
.ok_or(())
}
}
impl Default for SpellData {
fn default() -> Self {
Self {
internal_id: None,
name: "Unnamed Spell".to_string(),
description: "No description provided.".to_string(),
long_description: String::new(),
spell_tier: 0,
magic: MagicType::Arcane,
cast_slot: CastSlot::Primary,
collision: SpellCollision::Point,
cast_type: CastType::Instant,
cast_category: CastCategory::Projectile,
icon_tileset: spell_defaults::placeholder_png_path(),
icon_index: 0,
sprite_tileset: spell_defaults::placeholder_png_path(),
sprite_index: 0,
cooldown: spell_defaults::spell_cooldown(),
cast_time: spell_defaults::spell_cast_time(),
mana_cost: 0,
range: spell_defaults::spell_range(),
speed: spell_defaults::spell_speed(),
duration: spell_defaults::spell_duration(),
damage: 0,
healing: 0,
radius: 0,
angle: 0,
buffs: Vec::new(),
debuffs: Vec::new(),
particles: Vec::new(),
}
}
}