This commit is contained in:
		| @@ -3,10 +3,10 @@ use std::sync::Arc; | |||||||
|  |  | ||||||
| use parking_lot::RwLock; | use parking_lot::RwLock; | ||||||
|  |  | ||||||
| use crate::dynamic_data::LearnedMove; |  | ||||||
| use crate::dynamic_data::Pokemon; | use crate::dynamic_data::Pokemon; | ||||||
| use crate::dynamic_data::ScriptContainer; | use crate::dynamic_data::ScriptContainer; | ||||||
| use crate::dynamic_data::{ScriptSource, ScriptSourceData, ScriptWrapper}; | use crate::dynamic_data::{LearnedMove, ScriptWrapper}; | ||||||
|  | use crate::dynamic_data::{ScriptSource, ScriptSourceData}; | ||||||
|  |  | ||||||
| /// The data on a turn choice that should be contained in every turn choice, regardless of type. | /// The data on a turn choice that should be contained in every turn choice, regardless of type. | ||||||
| #[derive(Debug)] | #[derive(Debug)] | ||||||
|   | |||||||
| @@ -50,13 +50,22 @@ fn get_all_adjacent_opponent<'b, 'library>( | |||||||
|                 battle.get_pokemon(side, index).as_ref().cloned(), |                 battle.get_pokemon(side, index).as_ref().cloned(), | ||||||
|                 battle.get_pokemon(side, left as u8).as_ref().cloned(), |                 battle.get_pokemon(side, left as u8).as_ref().cloned(), | ||||||
|                 battle.get_pokemon(side, right).as_ref().cloned(), |                 battle.get_pokemon(side, right).as_ref().cloned(), | ||||||
|  |                 battle | ||||||
|  |                     .get_pokemon(get_opposite_side(side), left as u8) | ||||||
|  |                     .as_ref() | ||||||
|  |                     .cloned(), | ||||||
|                 battle.get_pokemon(get_opposite_side(side), index).as_ref().cloned(), |                 battle.get_pokemon(get_opposite_side(side), index).as_ref().cloned(), | ||||||
|  |                 battle.get_pokemon(get_opposite_side(side), right).as_ref().cloned(), | ||||||
|             ] |             ] | ||||||
|         } else { |         } else { | ||||||
|             vec![ |             vec![ | ||||||
|                 battle.get_pokemon(side, index).as_ref().cloned(), |                 battle.get_pokemon(side, index).as_ref().cloned(), | ||||||
|                 battle.get_pokemon(side, left as u8).as_ref().cloned(), |                 battle.get_pokemon(side, left as u8).as_ref().cloned(), | ||||||
|                 battle.get_pokemon(get_opposite_side(side), index).as_ref().cloned(), |                 battle.get_pokemon(get_opposite_side(side), index).as_ref().cloned(), | ||||||
|  |                 battle | ||||||
|  |                     .get_pokemon(get_opposite_side(side), left as u8) | ||||||
|  |                     .as_ref() | ||||||
|  |                     .cloned(), | ||||||
|             ] |             ] | ||||||
|         }; |         }; | ||||||
|     } |     } | ||||||
| @@ -65,6 +74,7 @@ fn get_all_adjacent_opponent<'b, 'library>( | |||||||
|         battle.get_pokemon(side, index).as_ref().cloned(), |         battle.get_pokemon(side, index).as_ref().cloned(), | ||||||
|         battle.get_pokemon(side, right).as_ref().cloned(), |         battle.get_pokemon(side, right).as_ref().cloned(), | ||||||
|         battle.get_pokemon(get_opposite_side(side), index).as_ref().cloned(), |         battle.get_pokemon(get_opposite_side(side), index).as_ref().cloned(), | ||||||
|  |         battle.get_pokemon(get_opposite_side(side), right).as_ref().cloned(), | ||||||
|     ] |     ] | ||||||
| } | } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -246,6 +246,9 @@ impl<'own, 'library> Battle<'own, 'library> { | |||||||
|             hit_data.set_damage(damage); |             hit_data.set_damage(damage); | ||||||
|  |  | ||||||
|             let mut accuracy = executing_move.use_move().accuracy(); |             let mut accuracy = executing_move.use_move().accuracy(); | ||||||
|  |             // If the accuracy is 255, the move should always hit, and as such we should not allow | ||||||
|  |             // modifying it. | ||||||
|  |             if accuracy != 255 { | ||||||
|                 script_hook!( |                 script_hook!( | ||||||
|                     change_accuracy, |                     change_accuracy, | ||||||
|                     executing_move, |                     executing_move, | ||||||
| @@ -254,6 +257,7 @@ impl<'own, 'library> Battle<'own, 'library> { | |||||||
|                     hit_index, |                     hit_index, | ||||||
|                     &mut accuracy |                     &mut accuracy | ||||||
|                 ); |                 ); | ||||||
|  |             } | ||||||
|             if accuracy < 100 && self.random().get_max(100) as u8 >= accuracy { |             if accuracy < 100 && self.random().get_max(100) as u8 >= accuracy { | ||||||
|                 script_hook!(on_move_miss, target, executing_move, target); |                 script_hook!(on_move_miss, target, executing_move, target); | ||||||
|                 self.event_hook().trigger(Event::Miss { |                 self.event_hook().trigger(Event::Miss { | ||||||
|   | |||||||
| @@ -40,7 +40,7 @@ impl<'library> Gen7MiscLibrary<'library> { | |||||||
|     pub fn new() -> Self { |     pub fn new() -> Self { | ||||||
|         let struggle_data = Box::new(MoveData::new( |         let struggle_data = Box::new(MoveData::new( | ||||||
|             &StringKey::new("struggle"), |             &StringKey::new("struggle"), | ||||||
|             0, |             0.into(), | ||||||
|             MoveCategory::Physical, |             MoveCategory::Physical, | ||||||
|             50, |             50, | ||||||
|             255, |             255, | ||||||
|   | |||||||
| @@ -7,7 +7,6 @@ use parking_lot::RwLock; | |||||||
|  |  | ||||||
| use crate::dynamic_data::choices::TurnChoice; | use crate::dynamic_data::choices::TurnChoice; | ||||||
| use crate::dynamic_data::event_hooks::{Event, EventHook}; | use crate::dynamic_data::event_hooks::{Event, EventHook}; | ||||||
| use crate::dynamic_data::is_valid_target; |  | ||||||
| use crate::dynamic_data::models::battle_party::BattleParty; | use crate::dynamic_data::models::battle_party::BattleParty; | ||||||
| use crate::dynamic_data::models::battle_random::BattleRandom; | use crate::dynamic_data::models::battle_random::BattleRandom; | ||||||
| use crate::dynamic_data::models::battle_result::BattleResult; | use crate::dynamic_data::models::battle_result::BattleResult; | ||||||
| @@ -17,8 +16,9 @@ use crate::dynamic_data::ChoiceQueue; | |||||||
| use crate::dynamic_data::DynamicLibrary; | use crate::dynamic_data::DynamicLibrary; | ||||||
| use crate::dynamic_data::Script; | use crate::dynamic_data::Script; | ||||||
| use crate::dynamic_data::ScriptSet; | use crate::dynamic_data::ScriptSet; | ||||||
| use crate::dynamic_data::VolatileScripts; | use crate::dynamic_data::VolatileScriptsOwner; | ||||||
| use crate::dynamic_data::{ScriptCategory, ScriptSource, ScriptSourceData, ScriptWrapper}; | use crate::dynamic_data::{is_valid_target, ScriptWrapper}; | ||||||
|  | use crate::dynamic_data::{ScriptCategory, ScriptSource, ScriptSourceData}; | ||||||
| use crate::{script_hook, PkmnResult, StringKey}; | use crate::{script_hook, PkmnResult, StringKey}; | ||||||
|  |  | ||||||
| /// A pokemon battle, with any amount of sides and pokemon per side. | /// A pokemon battle, with any amount of sides and pokemon per side. | ||||||
| @@ -322,7 +322,7 @@ impl<'own, 'library> Battle<'own, 'library> { | |||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
| impl<'own, 'library> VolatileScripts<'own> for Battle<'own, 'library> { | impl<'own, 'library> VolatileScriptsOwner<'own> for Battle<'own, 'library> { | ||||||
|     fn volatile_scripts(&self) -> &Arc<ScriptSet> { |     fn volatile_scripts(&self) -> &Arc<ScriptSet> { | ||||||
|         &self.volatile_scripts |         &self.volatile_scripts | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -13,7 +13,7 @@ use crate::dynamic_data::models::pokemon::Pokemon; | |||||||
| use crate::dynamic_data::script_handling::{ScriptSource, ScriptSourceData, ScriptWrapper}; | use crate::dynamic_data::script_handling::{ScriptSource, ScriptSourceData, ScriptWrapper}; | ||||||
| use crate::dynamic_data::Script; | use crate::dynamic_data::Script; | ||||||
| use crate::dynamic_data::ScriptSet; | use crate::dynamic_data::ScriptSet; | ||||||
| use crate::dynamic_data::VolatileScripts; | use crate::dynamic_data::VolatileScriptsOwner; | ||||||
| use crate::{script_hook, PkmnResult, StringKey}; | use crate::{script_hook, PkmnResult, StringKey}; | ||||||
|  |  | ||||||
| /// A side on a battle. | /// A side on a battle. | ||||||
| @@ -296,7 +296,7 @@ impl<'own, 'library> BattleSide<'own, 'library> { | |||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
| impl<'own, 'library> VolatileScripts<'own> for BattleSide<'own, 'library> { | impl<'own, 'library> VolatileScriptsOwner<'own> for BattleSide<'own, 'library> { | ||||||
|     fn volatile_scripts(&self) -> &Arc<ScriptSet> { |     fn volatile_scripts(&self) -> &Arc<ScriptSet> { | ||||||
|         &self.volatile_scripts |         &self.volatile_scripts | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -10,7 +10,7 @@ use crate::dynamic_data::models::pokemon::Pokemon; | |||||||
| use crate::dynamic_data::script_handling::{ScriptSource, ScriptSourceData, ScriptWrapper}; | use crate::dynamic_data::script_handling::{ScriptSource, ScriptSourceData, ScriptWrapper}; | ||||||
| use crate::dynamic_data::ScriptContainer; | use crate::dynamic_data::ScriptContainer; | ||||||
| use crate::dynamic_data::TargetList; | use crate::dynamic_data::TargetList; | ||||||
| use crate::static_data::MoveData; | use crate::static_data::{MoveData, TypeIdentifier}; | ||||||
| use crate::{PkmnResult, PokemonError}; | use crate::{PkmnResult, PokemonError}; | ||||||
|  |  | ||||||
| /// A hit data is the data for a single hit, on a single target. | /// A hit data is the data for a single hit, on a single target. | ||||||
| @@ -25,7 +25,7 @@ pub struct HitData { | |||||||
|     /// The actual damage of the hit. |     /// The actual damage of the hit. | ||||||
|     damage: AtomicU32, |     damage: AtomicU32, | ||||||
|     /// The type id of the type used for the hit. |     /// The type id of the type used for the hit. | ||||||
|     move_type: AtomicU8, |     move_type: Atomic<TypeIdentifier>, | ||||||
|     /// Whether or not the hit has failed. |     /// Whether or not the hit has failed. | ||||||
|     has_failed: AtomicBool, |     has_failed: AtomicBool, | ||||||
| } | } | ||||||
| @@ -48,7 +48,7 @@ impl HitData { | |||||||
|         self.damage.load(Ordering::Relaxed) |         self.damage.load(Ordering::Relaxed) | ||||||
|     } |     } | ||||||
|     /// The type id of the type used for the hit. |     /// The type id of the type used for the hit. | ||||||
|     pub fn move_type(&self) -> u8 { |     pub fn move_type(&self) -> TypeIdentifier { | ||||||
|         self.move_type.load(Ordering::Relaxed) |         self.move_type.load(Ordering::Relaxed) | ||||||
|     } |     } | ||||||
|     /// Whether or not the hit has failed. |     /// Whether or not the hit has failed. | ||||||
| @@ -73,7 +73,7 @@ impl HitData { | |||||||
|         self.damage.store(value, Ordering::SeqCst); |         self.damage.store(value, Ordering::SeqCst); | ||||||
|     } |     } | ||||||
|     /// Sets the move type id of the hit. |     /// Sets the move type id of the hit. | ||||||
|     pub fn set_move_type(&self, value: u8) { |     pub fn set_move_type(&self, value: TypeIdentifier) { | ||||||
|         self.move_type.store(value, Ordering::SeqCst); |         self.move_type.store(value, Ordering::SeqCst); | ||||||
|     } |     } | ||||||
|     /// Marks the hit as failed. |     /// Marks the hit as failed. | ||||||
|   | |||||||
| @@ -2,21 +2,32 @@ use std::sync::atomic::{AtomicU8, Ordering}; | |||||||
|  |  | ||||||
| use crate::static_data::MoveData; | use crate::static_data::MoveData; | ||||||
|  |  | ||||||
|  | /// A learned move is the data attached to a Pokemon for a move it has learned. It has information | ||||||
|  | /// such as the remaining amount of users, how it has been learned, etc. | ||||||
| #[derive(Debug)] | #[derive(Debug)] | ||||||
| pub struct LearnedMove<'library> { | pub struct LearnedMove<'library> { | ||||||
|  |     /// The immutable move information of the move. | ||||||
|     move_data: &'library MoveData, |     move_data: &'library MoveData, | ||||||
|  |     /// The maximal power points for this move. | ||||||
|     max_pp: u8, |     max_pp: u8, | ||||||
|  |     /// The amount of remaining power points. If this is 0, we can not use the move anymore. | ||||||
|     remaining_pp: AtomicU8, |     remaining_pp: AtomicU8, | ||||||
|  |     /// The way the move was learned. | ||||||
|     learn_method: MoveLearnMethod, |     learn_method: MoveLearnMethod, | ||||||
| } | } | ||||||
|  |  | ||||||
| #[derive(Copy, Clone, Debug)] | /// The different ways a move can be learned. | ||||||
|  | #[derive(Copy, Clone, Debug, Default)] | ||||||
| pub enum MoveLearnMethod { | pub enum MoveLearnMethod { | ||||||
|  |     /// We do not know the learn method. | ||||||
|  |     #[default] | ||||||
|     Unknown = 0, |     Unknown = 0, | ||||||
|  |     /// The move was learned through level up. | ||||||
|     Level = 1, |     Level = 1, | ||||||
| } | } | ||||||
|  |  | ||||||
| impl<'a> LearnedMove<'a> { | impl<'a> LearnedMove<'a> { | ||||||
|  |     /// Instantiate a new learned move. | ||||||
|     pub fn new(move_data: &'a MoveData, learn_method: MoveLearnMethod) -> Self { |     pub fn new(move_data: &'a MoveData, learn_method: MoveLearnMethod) -> Self { | ||||||
|         Self { |         Self { | ||||||
|             move_data, |             move_data, | ||||||
| @@ -26,36 +37,50 @@ impl<'a> LearnedMove<'a> { | |||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /// The immutable move information of the move. | ||||||
|     pub fn move_data(&self) -> &MoveData { |     pub fn move_data(&self) -> &MoveData { | ||||||
|         self.move_data |         self.move_data | ||||||
|     } |     } | ||||||
|  |     /// The maximal power points for this move. | ||||||
|     pub fn max_pp(&self) -> u8 { |     pub fn max_pp(&self) -> u8 { | ||||||
|         self.max_pp |         self.max_pp | ||||||
|     } |     } | ||||||
|  |     /// The amount of remaining power points. If this is 0, we can not use the move anymore. | ||||||
|     pub fn remaining_pp(&self) -> u8 { |     pub fn remaining_pp(&self) -> u8 { | ||||||
|         self.remaining_pp.load(Ordering::Relaxed) |         self.remaining_pp.load(Ordering::Relaxed) | ||||||
|     } |     } | ||||||
|  |     /// The way the move was learned. | ||||||
|     pub fn learn_method(&self) -> MoveLearnMethod { |     pub fn learn_method(&self) -> MoveLearnMethod { | ||||||
|         self.learn_method |         self.learn_method | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /// Try and reduce the PP by a certain amount. If the amount is higher than the current uses, | ||||||
|  |     /// return false. Otherwise, reduce the PP, and return true. | ||||||
|     pub fn try_use(&self, amount: u8) -> bool { |     pub fn try_use(&self, amount: u8) -> bool { | ||||||
|         if amount > self.remaining_pp() { |         let res = self.remaining_pp.fetch_update(Ordering::SeqCst, Ordering::SeqCst, |x| { | ||||||
|             return false; |             if amount > x { | ||||||
|  |                 None | ||||||
|  |             } else { | ||||||
|  |                 Some(x - amount) | ||||||
|             } |             } | ||||||
|         self.remaining_pp.fetch_sub(amount, Ordering::SeqCst); |         }); | ||||||
|         true |         res.is_ok() | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /// Set the remaining PP to the max amount of PP. | ||||||
|     pub fn restore_all_uses(&self) { |     pub fn restore_all_uses(&self) { | ||||||
|         self.remaining_pp.store(self.max_pp, Ordering::SeqCst); |         self.remaining_pp.store(self.max_pp, Ordering::SeqCst); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /// Restore the remaining PP by a certain amount. Will prevent it from going above max PP. | ||||||
|     pub fn restore_uses(&self, mut uses: u8) { |     pub fn restore_uses(&self, mut uses: u8) { | ||||||
|         if self.remaining_pp() + uses > self.max_pp { |         self.remaining_pp | ||||||
|             uses = self.remaining_pp() - uses; |             .fetch_update(Ordering::SeqCst, Ordering::SeqCst, |x| { | ||||||
|  |                 if x + uses > self.max_pp { | ||||||
|  |                     uses = self.max_pp - x; | ||||||
|                 } |                 } | ||||||
|         self.remaining_pp.fetch_add(uses, Ordering::SeqCst); |                 Some(x) | ||||||
|  |             }) | ||||||
|  |             .unwrap(); | ||||||
|     } |     } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -11,8 +11,7 @@ use crate::dynamic_data::models::battle::Battle; | |||||||
| use crate::dynamic_data::models::damage_source::DamageSource; | use crate::dynamic_data::models::damage_source::DamageSource; | ||||||
| use crate::dynamic_data::models::learned_move::{LearnedMove, MoveLearnMethod}; | use crate::dynamic_data::models::learned_move::{LearnedMove, MoveLearnMethod}; | ||||||
| use crate::dynamic_data::script_handling::{ScriptSource, ScriptSourceData, ScriptWrapper}; | use crate::dynamic_data::script_handling::{ScriptSource, ScriptSourceData, ScriptWrapper}; | ||||||
| use crate::dynamic_data::{DynamicLibrary, Script, ScriptCategory, ScriptContainer, ScriptSet, VolatileScripts}; | use crate::dynamic_data::{DynamicLibrary, Script, ScriptCategory, ScriptContainer, ScriptSet, VolatileScriptsOwner}; | ||||||
| use crate::static_data::AbilityIndex; |  | ||||||
| use crate::static_data::DataLibrary; | use crate::static_data::DataLibrary; | ||||||
| use crate::static_data::Form; | use crate::static_data::Form; | ||||||
| use crate::static_data::Gender; | use crate::static_data::Gender; | ||||||
| @@ -20,95 +19,107 @@ use crate::static_data::Item; | |||||||
| use crate::static_data::Nature; | use crate::static_data::Nature; | ||||||
| use crate::static_data::Species; | use crate::static_data::Species; | ||||||
| use crate::static_data::{Ability, Statistic}; | use crate::static_data::{Ability, Statistic}; | ||||||
|  | use crate::static_data::{AbilityIndex, TypeIdentifier}; | ||||||
| use crate::static_data::{ClampedStatisticSet, StatisticSet}; | use crate::static_data::{ClampedStatisticSet, StatisticSet}; | ||||||
| use crate::utils::Random; | use crate::utils::Random; | ||||||
| use crate::{script_hook, PkmnResult, StringKey}; | use crate::{script_hook, PkmnResult, StringKey}; | ||||||
|  |  | ||||||
| #[derive(Debug)] | /// An individual Pokemon as we know and love them. | ||||||
| pub struct PokemonBattleData<'pokemon, 'library> { |  | ||||||
|     battle: *mut Battle<'pokemon, 'library>, |  | ||||||
|     battle_side_index: AtomicU8, |  | ||||||
|     index: AtomicU8, |  | ||||||
|     on_battle_field: AtomicBool, |  | ||||||
|     seen_opponents: RwLock<Vec<Weak<Pokemon<'pokemon, 'library>>>>, |  | ||||||
| } |  | ||||||
|  |  | ||||||
| impl<'pokemon, 'library> PokemonBattleData<'pokemon, 'library> { |  | ||||||
|     pub fn battle_mut(&mut self) -> Option<&mut Battle<'pokemon, 'library>> { |  | ||||||
|         unsafe { self.battle.as_mut() } |  | ||||||
|     } |  | ||||||
|     pub fn battle(&self) -> Option<&Battle<'pokemon, 'library>> { |  | ||||||
|         unsafe { self.battle.as_ref() } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     pub fn battle_side_index(&self) -> u8 { |  | ||||||
|         self.battle_side_index.load(Ordering::Relaxed) |  | ||||||
|     } |  | ||||||
|     pub fn index(&self) -> u8 { |  | ||||||
|         self.index.load(Ordering::Relaxed) |  | ||||||
|     } |  | ||||||
|     pub fn on_battle_field(&self) -> bool { |  | ||||||
|         self.on_battle_field.load(Ordering::Relaxed) |  | ||||||
|     } |  | ||||||
|     pub fn seen_opponents(&self) -> &RwLock<Vec<Weak<Pokemon<'pokemon, 'library>>>> { |  | ||||||
|         &self.seen_opponents |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| #[derive(Debug)] | #[derive(Debug)] | ||||||
| pub struct Pokemon<'own, 'library> | pub struct Pokemon<'own, 'library> | ||||||
| where | where | ||||||
|     'own: 'library, |     'own: 'library, | ||||||
| { | { | ||||||
|  |     /// The library data of the Pokemon. | ||||||
|     library: &'own DynamicLibrary, |     library: &'own DynamicLibrary, | ||||||
|  |     /// The species of the Pokemon. | ||||||
|     species: &'own Species, |     species: &'own Species, | ||||||
|  |     /// The form of the Pokemon. | ||||||
|     form: &'own Form, |     form: &'own Form, | ||||||
|  |  | ||||||
|  |     /// An optional display species of the Pokemon. If this is set, the client should display this | ||||||
|  |     /// species. An example of usage for this is the Illusion ability. | ||||||
|     display_species: Option<&'own Species>, |     display_species: Option<&'own Species>, | ||||||
|  |     /// An optional display form of the Pokemon. If this is set, the client should display this | ||||||
|  |     //  species. An example of usage for this is the Illusion ability. | ||||||
|     display_form: Option<&'own Form>, |     display_form: Option<&'own Form>, | ||||||
|  |  | ||||||
|  |     /// The current level of the Pokemon. | ||||||
|     level: LevelInt, |     level: LevelInt, | ||||||
|  |     /// The amount of experience of the Pokemon. | ||||||
|     experience: AtomicU32, |     experience: AtomicU32, | ||||||
|  |     /// A unique random number for this Pokemon. | ||||||
|     unique_identifier: u32, |     unique_identifier: u32, | ||||||
|  |  | ||||||
|  |     /// The gender of the Pokemon. | ||||||
|     gender: Gender, |     gender: Gender, | ||||||
|  |     /// The coloring of the Pokemon. Value 0 is the default, value 1 means shiny. Other values are | ||||||
|  |     /// currently not used, and can be used for other implementations. | ||||||
|     coloring: u8, |     coloring: u8, | ||||||
|  |     /// The held item of the Pokemon. | ||||||
|     held_item: RwLock<Option<&'own Item>>, |     held_item: RwLock<Option<&'own Item>>, | ||||||
|  |     /// The remaining health points of the Pokemon. | ||||||
|     current_health: AtomicU32, |     current_health: AtomicU32, | ||||||
|  |  | ||||||
|  |     /// The weight of the Pokemon in kilograms. | ||||||
|     weight: Atomic<f32>, |     weight: Atomic<f32>, | ||||||
|  |     /// The height of the Pokemon in meters. | ||||||
|     height: Atomic<f32>, |     height: Atomic<f32>, | ||||||
|  |  | ||||||
|     stat_boost: ClampedStatisticSet<AtomicI8, -6, 6>, |     /// The stats of the Pokemon when disregarding any stat boosts. | ||||||
|     flat_stats: StatisticSet<AtomicU32>, |     flat_stats: StatisticSet<AtomicU32>, | ||||||
|  |     /// The statistics boosts of the Pokemon. Will prevent the value from going above 6, and below | ||||||
|  |     /// -6. | ||||||
|  |     stat_boost: ClampedStatisticSet<AtomicI8, -6, 6>, | ||||||
|  |     /// The stats of the Pokemon including the stat boosts | ||||||
|     boosted_stats: StatisticSet<AtomicU32>, |     boosted_stats: StatisticSet<AtomicU32>, | ||||||
|  |     /// The [individual values](https://bulbapedia.bulbagarden.net/wiki/Individual_values) of the Pokemon. | ||||||
|     individual_values: ClampedStatisticSet<AtomicU8, 0, 31>, |     individual_values: ClampedStatisticSet<AtomicU8, 0, 31>, | ||||||
|  |     /// The [effort values](https://bulbapedia.bulbagarden.net/wiki/Effort_values) of the Pokemon. | ||||||
|     effort_values: ClampedStatisticSet<AtomicU8, 0, 252>, |     effort_values: ClampedStatisticSet<AtomicU8, 0, 252>, | ||||||
|  |     /// The [nature](https://bulbapedia.bulbagarden.net/wiki/Nature) of the Pokemon. | ||||||
|     nature: &'own Nature, |     nature: &'own Nature, | ||||||
|  |  | ||||||
|  |     /// An optional nickname of the Pokemon. | ||||||
|     nickname: Option<String>, |     nickname: Option<String>, | ||||||
|  |  | ||||||
|  |     /// An index of the ability to find the actual ability on the form. | ||||||
|     ability_index: AbilityIndex, |     ability_index: AbilityIndex, | ||||||
|     is_ability_overridden: bool, |     /// An ability can be overriden to an arbitrary ability. This is for example used for the Mummy | ||||||
|  |     /// ability. | ||||||
|     override_ability: Option<Ability>, |     override_ability: Option<Ability>, | ||||||
|  |  | ||||||
|  |     /// If in battle, we have additional data. | ||||||
|     battle_data: RwLock<Option<PokemonBattleData<'own, 'library>>>, |     battle_data: RwLock<Option<PokemonBattleData<'own, 'library>>>, | ||||||
|  |  | ||||||
|  |     /// The moves the Pokemon has learned. This is of a set length of [`MAX_MOVES`]. Empty move slots | ||||||
|  |     /// are defined by None. | ||||||
|     moves: RwLock<[Option<Arc<LearnedMove<'library>>>; MAX_MOVES]>, |     moves: RwLock<[Option<Arc<LearnedMove<'library>>>; MAX_MOVES]>, | ||||||
|  |     /// Whether or not the Pokemon is allowed to gain experience. | ||||||
|     allowed_experience: bool, |     allowed_experience: bool, | ||||||
|  |  | ||||||
|     types: Vec<u8>, |     /// The current types of the Pokemon. | ||||||
|  |     types: Vec<TypeIdentifier>, | ||||||
|  |     /// Whether or not this Pokemon is an egg. | ||||||
|     is_egg: bool, |     is_egg: bool, | ||||||
|  |     /// Whether or not this Pokemon was caught this battle. | ||||||
|     is_caught: bool, |     is_caught: bool, | ||||||
|  |  | ||||||
|  |     /// The script for the held item. | ||||||
|     held_item_trigger_script: ScriptContainer, |     held_item_trigger_script: ScriptContainer, | ||||||
|  |     /// The script for the ability. | ||||||
|     ability_script: ScriptContainer, |     ability_script: ScriptContainer, | ||||||
|  |     /// The script for the status. | ||||||
|     status_script: ScriptContainer, |     status_script: ScriptContainer, | ||||||
|  |     /// The volatile status scripts of the Pokemon. | ||||||
|     volatile: Arc<ScriptSet>, |     volatile: Arc<ScriptSet>, | ||||||
|  |  | ||||||
|  |     /// Data required for the Pokemon to be a script source. | ||||||
|     script_source_data: RwLock<ScriptSourceData>, |     script_source_data: RwLock<ScriptSourceData>, | ||||||
| } | } | ||||||
|  |  | ||||||
| impl<'own, 'library> Pokemon<'own, 'library> { | impl<'own, 'library> Pokemon<'own, 'library> { | ||||||
|  |     /// Instantiates a new Pokemon. | ||||||
|     pub fn new( |     pub fn new( | ||||||
|         library: &'own DynamicLibrary, |         library: &'own DynamicLibrary, | ||||||
|         species: &'own Species, |         species: &'own Species, | ||||||
| @@ -155,7 +166,6 @@ impl<'own, 'library> Pokemon<'own, 'library> { | |||||||
|             nature, |             nature, | ||||||
|             nickname: None, |             nickname: None, | ||||||
|             ability_index: ability, |             ability_index: ability, | ||||||
|             is_ability_overridden: false, |  | ||||||
|             override_ability: None, |             override_ability: None, | ||||||
|             battle_data: RwLock::new(None), |             battle_data: RwLock::new(None), | ||||||
|             moves: RwLock::new([None, None, None, None]), |             moves: RwLock::new([None, None, None, None]), | ||||||
| @@ -176,15 +186,19 @@ impl<'own, 'library> Pokemon<'own, 'library> { | |||||||
|         pokemon |         pokemon | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /// The library data of the Pokemon. | ||||||
|     pub fn library(&self) -> &'own DynamicLibrary { |     pub fn library(&self) -> &'own DynamicLibrary { | ||||||
|         self.library |         self.library | ||||||
|     } |     } | ||||||
|  |     /// The species of the Pokemon. | ||||||
|     pub fn species(&self) -> &'own Species { |     pub fn species(&self) -> &'own Species { | ||||||
|         self.species |         self.species | ||||||
|     } |     } | ||||||
|  |     /// The form of the Pokemon. | ||||||
|     pub fn form(&self) -> &'own Form { |     pub fn form(&self) -> &'own Form { | ||||||
|         self.form |         self.form | ||||||
|     } |     } | ||||||
|  |     /// The species that should be displayed to the user. This handles stuff like the Illusion ability. | ||||||
|     pub fn display_species(&self) -> &'own Species { |     pub fn display_species(&self) -> &'own Species { | ||||||
|         if let Some(v) = self.display_species { |         if let Some(v) = self.display_species { | ||||||
|             v |             v | ||||||
| @@ -192,6 +206,7 @@ impl<'own, 'library> Pokemon<'own, 'library> { | |||||||
|             self.species |             self.species | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |     /// The form that should be displayed to the user. This handles stuff like the Illusion ability. | ||||||
|     pub fn display_form(&self) -> &'own Form { |     pub fn display_form(&self) -> &'own Form { | ||||||
|         if let Some(v) = self.display_form { |         if let Some(v) = self.display_form { | ||||||
|             v |             v | ||||||
| @@ -199,25 +214,32 @@ impl<'own, 'library> Pokemon<'own, 'library> { | |||||||
|             self.form |             self.form | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |     /// The current level of the Pokemon. | ||||||
|     pub fn level(&self) -> LevelInt { |     pub fn level(&self) -> LevelInt { | ||||||
|         self.level |         self.level | ||||||
|     } |     } | ||||||
|  |     /// The amount of experience of the Pokemon. | ||||||
|     pub fn experience(&self) -> u32 { |     pub fn experience(&self) -> u32 { | ||||||
|         self.experience.load(Ordering::Relaxed) |         self.experience.load(Ordering::Relaxed) | ||||||
|     } |     } | ||||||
|  |     /// A unique random number for this Pokemon. | ||||||
|     pub fn unique_identifier(&self) -> u32 { |     pub fn unique_identifier(&self) -> u32 { | ||||||
|         self.unique_identifier |         self.unique_identifier | ||||||
|     } |     } | ||||||
|  |     /// The gender of the Pokemon. | ||||||
|     pub fn gender(&self) -> Gender { |     pub fn gender(&self) -> Gender { | ||||||
|         self.gender |         self.gender | ||||||
|     } |     } | ||||||
|  |     /// The coloring of the Pokemon. Value 0 is the default, value 1 means shiny. Other values are | ||||||
|  |     /// currently not used, and can be used for other implementations. | ||||||
|     pub fn coloring(&self) -> u8 { |     pub fn coloring(&self) -> u8 { | ||||||
|         self.coloring |         self.coloring | ||||||
|     } |     } | ||||||
|  |     /// Checks whether the Pokemon is holding an item, | ||||||
|     pub fn held_item(&self) -> &RwLock<Option<&'own Item>> { |     pub fn held_item(&self) -> &RwLock<Option<&'own Item>> { | ||||||
|         &self.held_item |         &self.held_item | ||||||
|     } |     } | ||||||
|  |     /// Checks whether the Pokemon is holding a specific item. | ||||||
|     pub fn has_held_item(&self, name: &StringKey) -> bool { |     pub fn has_held_item(&self, name: &StringKey) -> bool { | ||||||
|         // Only true if we have an item, and the item name is the same as the requested item. |         // Only true if we have an item, and the item name is the same as the requested item. | ||||||
|         if let Some(v) = self.held_item.read().deref() { |         if let Some(v) = self.held_item.read().deref() { | ||||||
| @@ -225,12 +247,15 @@ impl<'own, 'library> Pokemon<'own, 'library> { | |||||||
|         } |         } | ||||||
|         false |         false | ||||||
|     } |     } | ||||||
|  |     /// Changes the held item of the Pokemon/ | ||||||
|     pub fn set_held_item(&self, item: &'own Item) -> Option<&'own Item> { |     pub fn set_held_item(&self, item: &'own Item) -> Option<&'own Item> { | ||||||
|         self.held_item.write().replace(item) |         self.held_item.write().replace(item) | ||||||
|     } |     } | ||||||
|  |     /// Removes the held item from the Pokemon. | ||||||
|     pub fn remove_held_item(&self) -> Option<&'own Item> { |     pub fn remove_held_item(&self) -> Option<&'own Item> { | ||||||
|         self.held_item.write().take() |         self.held_item.write().take() | ||||||
|     } |     } | ||||||
|  |     /// Makes the Pokemon uses its held item. | ||||||
|     pub fn consume_held_item(&self) -> bool { |     pub fn consume_held_item(&self) -> bool { | ||||||
|         if self.held_item.read().is_none() { |         if self.held_item.read().is_none() { | ||||||
|             return false; |             return false; | ||||||
| @@ -244,42 +269,53 @@ impl<'own, 'library> Pokemon<'own, 'library> { | |||||||
|         todo!(); |         todo!(); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /// The remaining health points of the Pokemon. | ||||||
|     pub fn current_health(&self) -> u32 { |     pub fn current_health(&self) -> u32 { | ||||||
|         self.current_health.load(Ordering::Relaxed) |         self.current_health.load(Ordering::Relaxed) | ||||||
|     } |     } | ||||||
|  |     /// The max health points of the Pokemon. | ||||||
|     pub fn max_health(&self) -> u32 { |     pub fn max_health(&self) -> u32 { | ||||||
|         self.boosted_stats.hp() |         self.boosted_stats.hp() | ||||||
|     } |     } | ||||||
|  |     /// The weight of the Pokemon in kilograms. | ||||||
|     pub fn weight(&self) -> f32 { |     pub fn weight(&self) -> f32 { | ||||||
|         self.weight.load(Ordering::Relaxed) |         self.weight.load(Ordering::Relaxed) | ||||||
|     } |     } | ||||||
|  |     /// The height of the Pokemon in meters. | ||||||
|     pub fn height(&self) -> f32 { |     pub fn height(&self) -> f32 { | ||||||
|         self.height.load(Ordering::Relaxed) |         self.height.load(Ordering::Relaxed) | ||||||
|     } |     } | ||||||
|  |     /// An optional nickname of the Pokemon. | ||||||
|     pub fn nickname(&self) -> &Option<String> { |     pub fn nickname(&self) -> &Option<String> { | ||||||
|         &self.nickname |         &self.nickname | ||||||
|     } |     } | ||||||
|  |     /// An index of the ability to find the actual ability on the form. | ||||||
|     pub fn real_ability(&self) -> &AbilityIndex { |     pub fn real_ability(&self) -> &AbilityIndex { | ||||||
|         &self.ability_index |         &self.ability_index | ||||||
|     } |     } | ||||||
|     pub fn types(&self) -> &Vec<u8> { |     /// The current types of the Pokemon. | ||||||
|  |     pub fn types(&self) -> &Vec<TypeIdentifier> { | ||||||
|         &self.types |         &self.types | ||||||
|     } |     } | ||||||
|  |     /// The moves the Pokemon has learned. This is of a set length of [`MAX_MOVES`]. Empty move slots | ||||||
|  |     /// are defined by None. | ||||||
|     pub fn learned_moves(&self) -> &RwLock<[Option<Arc<LearnedMove<'library>>>; MAX_MOVES]> { |     pub fn learned_moves(&self) -> &RwLock<[Option<Arc<LearnedMove<'library>>>; MAX_MOVES]> { | ||||||
|         &self.moves |         &self.moves | ||||||
|     } |     } | ||||||
|     pub fn status(&self) -> &ScriptContainer { |  | ||||||
|         &self.status_script |     /// The stats of the Pokemon when disregarding any stat boosts. | ||||||
|     } |  | ||||||
|     pub fn flat_stats(&self) -> &StatisticSet<AtomicU32> { |     pub fn flat_stats(&self) -> &StatisticSet<AtomicU32> { | ||||||
|         &self.flat_stats |         &self.flat_stats | ||||||
|     } |     } | ||||||
|  |     /// The stats of the Pokemon including the stat boosts | ||||||
|     pub fn boosted_stats(&self) -> &StatisticSet<AtomicU32> { |     pub fn boosted_stats(&self) -> &StatisticSet<AtomicU32> { | ||||||
|         &self.boosted_stats |         &self.boosted_stats | ||||||
|     } |     } | ||||||
|  |     /// Get the stat boosts for a specific stat. | ||||||
|     pub fn stat_boost(&self, stat: Statistic) -> i8 { |     pub fn stat_boost(&self, stat: Statistic) -> i8 { | ||||||
|         self.stat_boost.get_stat(stat) |         self.stat_boost.get_stat(stat) | ||||||
|     } |     } | ||||||
|  |     /// Change a boosted stat by a certain amount. | ||||||
|     pub fn change_stat_boost(&self, stat: Statistic, mut diff_amount: i8, self_inflicted: bool) -> bool { |     pub fn change_stat_boost(&self, stat: Statistic, mut diff_amount: i8, self_inflicted: bool) -> bool { | ||||||
|         let mut prevent = false; |         let mut prevent = false; | ||||||
|         script_hook!( |         script_hook!( | ||||||
| @@ -328,13 +364,16 @@ impl<'own, 'library> Pokemon<'own, 'library> { | |||||||
|         return changed; |         return changed; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /// The [individual values](https://bulbapedia.bulbagarden.net/wiki/Individual_values) of the Pokemon. | ||||||
|     pub fn individual_values(&self) -> &ClampedStatisticSet<AtomicU8, 0, 31> { |     pub fn individual_values(&self) -> &ClampedStatisticSet<AtomicU8, 0, 31> { | ||||||
|         &self.individual_values |         &self.individual_values | ||||||
|     } |     } | ||||||
|  |     /// The [effort values](https://bulbapedia.bulbagarden.net/wiki/Effort_values) of the Pokemon. | ||||||
|     pub fn effort_values(&self) -> &ClampedStatisticSet<AtomicU8, 0, 252> { |     pub fn effort_values(&self) -> &ClampedStatisticSet<AtomicU8, 0, 252> { | ||||||
|         &self.effort_values |         &self.effort_values | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /// Gets the battle the battle is currently in. | ||||||
|     pub fn get_battle(&self) -> Option<&Battle<'own, 'library>> { |     pub fn get_battle(&self) -> Option<&Battle<'own, 'library>> { | ||||||
|         let r = self.battle_data.read(); |         let r = self.battle_data.read(); | ||||||
|         if let Some(data) = &r.deref() { |         if let Some(data) = &r.deref() { | ||||||
| @@ -343,21 +382,25 @@ impl<'own, 'library> Pokemon<'own, 'library> { | |||||||
|             None |             None | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |     /// Get the index of the side of the battle the Pokemon is in. Only returns a value if the Pokemon | ||||||
|  |     /// is on the battlefield. | ||||||
|     pub fn get_battle_side_index(&self) -> Option<u8> { |     pub fn get_battle_side_index(&self) -> Option<u8> { | ||||||
|         self.battle_data.read().as_ref().map(|data| data.battle_side_index()) |         self.battle_data.read().as_ref().map(|data| data.battle_side_index()) | ||||||
|     } |     } | ||||||
|  |     /// Get the index of the slot on the side of the battle the Pokemon is in. Only returns a value | ||||||
|  |     /// if the Pokemon is on the battlefield. | ||||||
|     pub fn get_battle_index(&self) -> Option<u8> { |     pub fn get_battle_index(&self) -> Option<u8> { | ||||||
|         self.battle_data.read().as_ref().map(|data| data.index()) |         self.battle_data.read().as_ref().map(|data| data.index()) | ||||||
|     } |     } | ||||||
|  |     /// Returns whether something overrides the ability. | ||||||
|     pub fn is_ability_overriden(&self) -> bool { |     pub fn is_ability_overriden(&self) -> bool { | ||||||
|         self.is_ability_overridden |         self.override_ability.is_some() | ||||||
|     } |     } | ||||||
|  |     /// Returns the currently active ability. | ||||||
|     pub fn active_ability(&self) -> &Ability { |     pub fn active_ability(&self) -> &Ability { | ||||||
|         if self.is_ability_overridden { |  | ||||||
|         if let Some(v) = &self.override_ability { |         if let Some(v) = &self.override_ability { | ||||||
|             return v; |             return v; | ||||||
|         } |         } | ||||||
|         } |  | ||||||
|         self.library |         self.library | ||||||
|             .static_data() |             .static_data() | ||||||
|             .abilities() |             .abilities() | ||||||
| @@ -365,33 +408,41 @@ impl<'own, 'library> Pokemon<'own, 'library> { | |||||||
|             .unwrap() |             .unwrap() | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /// The script for the status. | ||||||
|  |     pub fn status(&self) -> &ScriptContainer { | ||||||
|  |         &self.status_script | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /// Returns the script for the currently active ability. | ||||||
|     pub fn ability_script(&self) -> &ScriptContainer { |     pub fn ability_script(&self) -> &ScriptContainer { | ||||||
|         &self.ability_script |         &self.ability_script | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     // pub fn seen_opponents(&self) -> &RwLock<Option<PokemonBattleData<'own, 'library>>> { |     /// Whether or not the Pokemon is allowed to gain experience. | ||||||
|     //     &self.battle_data.read |  | ||||||
|     // } |  | ||||||
|     pub fn allowed_experience_gain(&self) -> bool { |     pub fn allowed_experience_gain(&self) -> bool { | ||||||
|         self.allowed_experience |         self.allowed_experience | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /// The [nature](https://bulbapedia.bulbagarden.net/wiki/Nature) of the Pokemon. | ||||||
|     pub fn nature(&self) -> &'own Nature { |     pub fn nature(&self) -> &'own Nature { | ||||||
|         self.nature |         self.nature | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /// Calculates the flat stats on the Pokemon. | ||||||
|     pub fn recalculate_flat_stats(&self) { |     pub fn recalculate_flat_stats(&self) { | ||||||
|         self.library |         self.library | ||||||
|             .stat_calculator() |             .stat_calculator() | ||||||
|             .calculate_flat_stats(self, &self.flat_stats); |             .calculate_flat_stats(self, &self.flat_stats); | ||||||
|         self.recalculate_boosted_stats(); |         self.recalculate_boosted_stats(); | ||||||
|     } |     } | ||||||
|  |     /// Calculates the boosted stats on the Pokemon. | ||||||
|     pub fn recalculate_boosted_stats(&self) { |     pub fn recalculate_boosted_stats(&self) { | ||||||
|         self.library |         self.library | ||||||
|             .stat_calculator() |             .stat_calculator() | ||||||
|             .calculate_boosted_stats(self, &self.boosted_stats); |             .calculate_boosted_stats(self, &self.boosted_stats); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /// Change the species of the Pokemon. | ||||||
|     pub fn change_species(&mut self, species: &'own Species, form: &'own Form) { |     pub fn change_species(&mut self, species: &'own Species, form: &'own Form) { | ||||||
|         self.species = species; |         self.species = species; | ||||||
|         self.form = form; |         self.form = form; | ||||||
| @@ -424,6 +475,7 @@ impl<'own, 'library> Pokemon<'own, 'library> { | |||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /// Change the form of the Pokemon. | ||||||
|     pub fn change_form(&mut self, form: &'own Form) { |     pub fn change_form(&mut self, form: &'own Form) { | ||||||
|         if std::ptr::eq(self.form, form) { |         if std::ptr::eq(self.form, form) { | ||||||
|             return; |             return; | ||||||
| @@ -474,14 +526,17 @@ impl<'own, 'library> Pokemon<'own, 'library> { | |||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /// Whether or not the Pokemon is useable in a battle. | ||||||
|     pub fn is_usable(&self) -> bool { |     pub fn is_usable(&self) -> bool { | ||||||
|         !self.is_caught && !self.is_egg && !self.is_fainted() |         !self.is_caught && !self.is_egg && !self.is_fainted() | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /// Returns whether the Pokemon is fainted. | ||||||
|     pub fn is_fainted(&self) -> bool { |     pub fn is_fainted(&self) -> bool { | ||||||
|         self.current_health() == 0 |         self.current_health() == 0 | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /// Sets the current battle the Pokemon is in. | ||||||
|     pub fn set_battle_data(&self, battle: *mut Battle<'own, 'library>, battle_side_index: u8) { |     pub fn set_battle_data(&self, battle: *mut Battle<'own, 'library>, battle_side_index: u8) { | ||||||
|         let mut w = self.battle_data.write(); |         let mut w = self.battle_data.write(); | ||||||
|         if let Some(battle_data) = w.deref_mut() { |         if let Some(battle_data) = w.deref_mut() { | ||||||
| @@ -498,6 +553,7 @@ impl<'own, 'library> Pokemon<'own, 'library> { | |||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /// Sets whether or not the Pokemon is on the battlefield. | ||||||
|     pub fn set_on_battlefield(&self, value: bool) { |     pub fn set_on_battlefield(&self, value: bool) { | ||||||
|         let r = self.battle_data.read(); |         let r = self.battle_data.read(); | ||||||
|         if let Some(data) = &mut r.deref() { |         if let Some(data) = &mut r.deref() { | ||||||
| @@ -510,6 +566,7 @@ impl<'own, 'library> Pokemon<'own, 'library> { | |||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /// Sets the index of the slot of the side the Pokemon is on. | ||||||
|     pub fn set_battle_index(&self, index: u8) { |     pub fn set_battle_index(&self, index: u8) { | ||||||
|         let r = self.battle_data.read(); |         let r = self.battle_data.read(); | ||||||
|         if let Some(data) = r.deref() { |         if let Some(data) = r.deref() { | ||||||
| @@ -517,10 +574,12 @@ impl<'own, 'library> Pokemon<'own, 'library> { | |||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /// Whether or not the Pokemon is on the battlefield. | ||||||
|     pub fn is_on_battlefield(&self) -> bool { |     pub fn is_on_battlefield(&self) -> bool { | ||||||
|         self.battle_data.read().is_some_and(|a| a.on_battle_field()) |         self.battle_data.read().is_some_and(|a| a.on_battle_field()) | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /// Marks an opponent as seen, for use in experience gain. | ||||||
|     pub fn mark_opponent_as_seen(&self, pokemon: Weak<Pokemon<'own, 'library>>) { |     pub fn mark_opponent_as_seen(&self, pokemon: Weak<Pokemon<'own, 'library>>) { | ||||||
|         let r = self.battle_data.read(); |         let r = self.battle_data.read(); | ||||||
|         if let Some(battle_data) = &r.deref() { |         if let Some(battle_data) = &r.deref() { | ||||||
| @@ -534,6 +593,7 @@ impl<'own, 'library> Pokemon<'own, 'library> { | |||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /// Damages the Pokemon by a certain amount of damage, from a specific damage source. | ||||||
|     pub fn damage(&self, mut damage: u32, source: DamageSource) { |     pub fn damage(&self, mut damage: u32, source: DamageSource) { | ||||||
|         if damage > self.current_health() { |         if damage > self.current_health() { | ||||||
|             damage = self.current_health(); |             damage = self.current_health(); | ||||||
| @@ -563,7 +623,8 @@ impl<'own, 'library> Pokemon<'own, 'library> { | |||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     pub fn on_faint(&self, source: DamageSource) { |     /// Triggers when the Pokemon faints. | ||||||
|  |     fn on_faint(&self, source: DamageSource) { | ||||||
|         let r = self.battle_data.read(); |         let r = self.battle_data.read(); | ||||||
|         if let Some(battle_data) = r.deref() { |         if let Some(battle_data) = r.deref() { | ||||||
|             if let Some(battle) = battle_data.battle() { |             if let Some(battle) = battle_data.battle() { | ||||||
| @@ -581,6 +642,7 @@ impl<'own, 'library> Pokemon<'own, 'library> { | |||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /// Learn a move. | ||||||
|     pub fn learn_move(&self, move_name: &StringKey, learn_method: MoveLearnMethod) { |     pub fn learn_move(&self, move_name: &StringKey, learn_method: MoveLearnMethod) { | ||||||
|         let mut learned_moves = self.learned_moves().write(); |         let mut learned_moves = self.learned_moves().write(); | ||||||
|         let move_pos = learned_moves.iter().position(|a| a.is_none()); |         let move_pos = learned_moves.iter().position(|a| a.is_none()); | ||||||
| @@ -592,6 +654,49 @@ impl<'own, 'library> Pokemon<'own, 'library> { | |||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
|  | /// The data of the Pokemon related to being in a battle. | ||||||
|  | #[derive(Debug)] | ||||||
|  | pub struct PokemonBattleData<'pokemon, 'library> { | ||||||
|  |     /// The battle data of the Pokemon | ||||||
|  |     battle: *mut Battle<'pokemon, 'library>, | ||||||
|  |     /// The index of the side of the Pokemon | ||||||
|  |     battle_side_index: AtomicU8, | ||||||
|  |     /// The index of the slot on the side of the Pokemon. | ||||||
|  |     index: AtomicU8, | ||||||
|  |     /// Whether or not the Pokemon is on the battlefield. | ||||||
|  |     on_battle_field: AtomicBool, | ||||||
|  |     /// A list of opponents the Pokemon has seen this battle. | ||||||
|  |     seen_opponents: RwLock<Vec<Weak<Pokemon<'pokemon, 'library>>>>, | ||||||
|  | } | ||||||
|  |  | ||||||
|  | impl<'pokemon, 'library> PokemonBattleData<'pokemon, 'library> { | ||||||
|  |     /// The battle data of the Pokemon | ||||||
|  |     pub fn battle_mut(&mut self) -> Option<&mut Battle<'pokemon, 'library>> { | ||||||
|  |         unsafe { self.battle.as_mut() } | ||||||
|  |     } | ||||||
|  |     /// The battle data of the Pokemon | ||||||
|  |     pub fn battle(&self) -> Option<&Battle<'pokemon, 'library>> { | ||||||
|  |         unsafe { self.battle.as_ref() } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /// The index of the side of the Pokemon | ||||||
|  |     pub fn battle_side_index(&self) -> u8 { | ||||||
|  |         self.battle_side_index.load(Ordering::Relaxed) | ||||||
|  |     } | ||||||
|  |     /// The index of the slot on the side of the Pokemon. | ||||||
|  |     pub fn index(&self) -> u8 { | ||||||
|  |         self.index.load(Ordering::Relaxed) | ||||||
|  |     } | ||||||
|  |     /// Whether or not the Pokemon is on the battlefield. | ||||||
|  |     pub fn on_battle_field(&self) -> bool { | ||||||
|  |         self.on_battle_field.load(Ordering::Relaxed) | ||||||
|  |     } | ||||||
|  |     /// A list of opponents the Pokemon has seen this battle. | ||||||
|  |     pub fn seen_opponents(&self) -> &RwLock<Vec<Weak<Pokemon<'pokemon, 'library>>>> { | ||||||
|  |         &self.seen_opponents | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
| impl<'own, 'library> ScriptSource<'own> for Pokemon<'own, 'library> { | impl<'own, 'library> ScriptSource<'own> for Pokemon<'own, 'library> { | ||||||
|     fn get_script_count(&self) -> usize { |     fn get_script_count(&self) -> usize { | ||||||
|         let mut c = 3; |         let mut c = 3; | ||||||
| @@ -624,7 +729,7 @@ impl<'own, 'library> ScriptSource<'own> for Pokemon<'own, 'library> { | |||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
| impl<'own, 'library> VolatileScripts<'own> for Pokemon<'own, 'library> { | impl<'own, 'library> VolatileScriptsOwner<'own> for Pokemon<'own, 'library> { | ||||||
|     fn volatile_scripts(&self) -> &Arc<ScriptSet> { |     fn volatile_scripts(&self) -> &Arc<ScriptSet> { | ||||||
|         &self.volatile |         &self.volatile | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -3,30 +3,47 @@ use crate::dynamic_data::models::learned_move::MoveLearnMethod; | |||||||
| use crate::dynamic_data::models::pokemon::Pokemon; | use crate::dynamic_data::models::pokemon::Pokemon; | ||||||
| use crate::dynamic_data::DynamicLibrary; | use crate::dynamic_data::DynamicLibrary; | ||||||
| use crate::static_data::{AbilityIndex, DataLibrary, Gender}; | use crate::static_data::{AbilityIndex, DataLibrary, Gender}; | ||||||
| use crate::StringKey; | use crate::{Random, StringKey}; | ||||||
|  |  | ||||||
|  | /// This allows for the easy chain building of a Pokemon. | ||||||
| pub struct PokemonBuilder<'own> { | pub struct PokemonBuilder<'own> { | ||||||
|  |     /// The library of the Pokemon. | ||||||
|     library: &'own DynamicLibrary, |     library: &'own DynamicLibrary, | ||||||
|  |     /// The name of the species of the Pokemon. | ||||||
|     species: StringKey, |     species: StringKey, | ||||||
|  |     /// The level of the Pokemon. | ||||||
|     level: LevelInt, |     level: LevelInt, | ||||||
|  |     /// The moves the Pokemon will know. | ||||||
|     learned_moves: Vec<StringKey>, |     learned_moves: Vec<StringKey>, | ||||||
|  |     /// A random seed used for any randomization done. | ||||||
|  |     random_seed: Option<u128>, | ||||||
| } | } | ||||||
|  |  | ||||||
| impl<'own> PokemonBuilder<'own> { | impl<'own> PokemonBuilder<'own> { | ||||||
|  |     /// Creates a new PokemonBuilder with a library, species, and level. | ||||||
|     pub fn new(library: &'own DynamicLibrary, species: StringKey, level: LevelInt) -> Self { |     pub fn new(library: &'own DynamicLibrary, species: StringKey, level: LevelInt) -> Self { | ||||||
|         Self { |         Self { | ||||||
|             library, |             library, | ||||||
|             species, |             species, | ||||||
|             level, |             level, | ||||||
|             learned_moves: vec![], |             learned_moves: vec![], | ||||||
|  |             random_seed: None, | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |     /// Makes the Pokemon learn a move. | ||||||
|     pub fn learn_move(mut self, learned_move: StringKey) -> Self { |     pub fn learn_move(mut self, learned_move: StringKey) -> Self { | ||||||
|         self.learned_moves.push(learned_move); |         self.learned_moves.push(learned_move); | ||||||
|         self |         self | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /// Finally turn the builder into an actual Pokemon. | ||||||
|     pub fn build(self) -> Pokemon<'own, 'own> { |     pub fn build(self) -> Pokemon<'own, 'own> { | ||||||
|  |         let mut random = if let Some(seed) = self.random_seed { | ||||||
|  |             Random::new(seed) | ||||||
|  |         } else { | ||||||
|  |             Random::default() | ||||||
|  |         }; | ||||||
|  |  | ||||||
|         let species = self.library.static_data().species().get(&self.species).unwrap(); |         let species = self.library.static_data().species().get(&self.species).unwrap(); | ||||||
|         let form = species.get_default_form(); |         let form = species.get_default_form(); | ||||||
|         let p = Pokemon::new( |         let p = Pokemon::new( | ||||||
| @@ -38,7 +55,7 @@ impl<'own> PokemonBuilder<'own> { | |||||||
|                 index: 0, |                 index: 0, | ||||||
|             }, |             }, | ||||||
|             self.level, |             self.level, | ||||||
|             0, |             random.get_unsigned(), | ||||||
|             Gender::Male, |             Gender::Male, | ||||||
|             0, |             0, | ||||||
|             &"hardy".into(), |             &"hardy".into(), | ||||||
|   | |||||||
| @@ -1,12 +1,16 @@ | |||||||
| use crate::dynamic_data::models::pokemon::Pokemon; |  | ||||||
| use std::sync::Arc; | use std::sync::Arc; | ||||||
|  |  | ||||||
|  | use crate::dynamic_data::models::pokemon::Pokemon; | ||||||
|  |  | ||||||
|  | /// A list of Pokemon belonging to a trainer. | ||||||
| #[derive(Debug)] | #[derive(Debug)] | ||||||
| pub struct PokemonParty<'pokemon, 'library> { | pub struct PokemonParty<'pokemon, 'library> { | ||||||
|  |     /// The underlying list of Pokemon. | ||||||
|     pokemon: Vec<Option<Arc<Pokemon<'pokemon, 'library>>>>, |     pokemon: Vec<Option<Arc<Pokemon<'pokemon, 'library>>>>, | ||||||
| } | } | ||||||
|  |  | ||||||
| impl<'own, 'library> PokemonParty<'own, 'library> { | impl<'own, 'library> PokemonParty<'own, 'library> { | ||||||
|  |     /// Instantiates a party with a set size. | ||||||
|     pub fn new(size: usize) -> Self { |     pub fn new(size: usize) -> Self { | ||||||
|         let mut pokemon = Vec::with_capacity(size); |         let mut pokemon = Vec::with_capacity(size); | ||||||
|         for _i in 0..size { |         for _i in 0..size { | ||||||
| @@ -15,10 +19,12 @@ impl<'own, 'library> PokemonParty<'own, 'library> { | |||||||
|         Self { pokemon } |         Self { pokemon } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /// Instantiates a party with a list. | ||||||
|     pub fn new_from_vec(pokemon: Vec<Option<Arc<Pokemon<'own, 'library>>>>) -> Self { |     pub fn new_from_vec(pokemon: Vec<Option<Arc<Pokemon<'own, 'library>>>>) -> Self { | ||||||
|         Self { pokemon } |         Self { pokemon } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /// Gets a Pokemon at an index in the party. | ||||||
|     pub fn at(&self, index: usize) -> &Option<Arc<Pokemon<'own, 'library>>> { |     pub fn at(&self, index: usize) -> &Option<Arc<Pokemon<'own, 'library>>> { | ||||||
|         let opt = self.pokemon.get(index); |         let opt = self.pokemon.get(index); | ||||||
|         if let Some(v) = opt { |         if let Some(v) = opt { | ||||||
| @@ -28,10 +34,12 @@ impl<'own, 'library> PokemonParty<'own, 'library> { | |||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /// Swaps two Pokemon in the party around. | ||||||
|     pub fn switch(&mut self, a: usize, b: usize) { |     pub fn switch(&mut self, a: usize, b: usize) { | ||||||
|         self.pokemon.swap(a, b); |         self.pokemon.swap(a, b); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /// Sets the Pokemon at an index to a Pokemon, returning the old Pokemon. | ||||||
|     pub fn swap_into( |     pub fn swap_into( | ||||||
|         &mut self, |         &mut self, | ||||||
|         index: usize, |         index: usize, | ||||||
| @@ -45,6 +53,7 @@ impl<'own, 'library> PokemonParty<'own, 'library> { | |||||||
|         old |         old | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /// Whether or not the party still has Pokemon that can be used in battle. | ||||||
|     pub fn has_usable_pokemon(&self) -> bool { |     pub fn has_usable_pokemon(&self) -> bool { | ||||||
|         for pokemon in self.pokemon.iter().flatten() { |         for pokemon in self.pokemon.iter().flatten() { | ||||||
|             if pokemon.is_usable() { |             if pokemon.is_usable() { | ||||||
| @@ -54,14 +63,17 @@ impl<'own, 'library> PokemonParty<'own, 'library> { | |||||||
|         false |         false | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /// Get the length of the underlying list of Pokemon. | ||||||
|     pub fn length(&self) -> usize { |     pub fn length(&self) -> usize { | ||||||
|         self.pokemon.len() |         self.pokemon.len() | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /// Gets the underlying list of Pokemon. | ||||||
|     pub fn pokemon(&self) -> &Vec<Option<Arc<Pokemon<'own, 'library>>>> { |     pub fn pokemon(&self) -> &Vec<Option<Arc<Pokemon<'own, 'library>>>> { | ||||||
|         &self.pokemon |         &self.pokemon | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /// Makes sure there are no empty spots in the party anymore, leaving the length the same. | ||||||
|     pub fn pack_party(&mut self) { |     pub fn pack_party(&mut self) { | ||||||
|         let mut first_empty = None; |         let mut first_empty = None; | ||||||
|         let mut i = 0; |         let mut i = 0; | ||||||
|   | |||||||
| @@ -1 +1,2 @@ | |||||||
|  | /// The script functions that are relevant to item use. | ||||||
| pub trait ItemScript {} | pub trait ItemScript {} | ||||||
|   | |||||||
| @@ -9,13 +9,15 @@ pub use script::*; | |||||||
| #[doc(inline)] | #[doc(inline)] | ||||||
| pub use script_set::*; | pub use script_set::*; | ||||||
| #[doc(inline)] | #[doc(inline)] | ||||||
| pub use volatile_scripts::*; | pub use volatile_scripts_owner::*; | ||||||
|  |  | ||||||
| mod item_script; | mod item_script; | ||||||
| mod script; | mod script; | ||||||
| mod script_set; | mod script_set; | ||||||
| mod volatile_scripts; | mod volatile_scripts_owner; | ||||||
|  |  | ||||||
|  | /// This macro runs a script function on a given ScriptSource, and all its parents. It will ensure | ||||||
|  | /// to only run the script function if it is not suppressed, and can take any amount of parameters. | ||||||
| #[macro_export] | #[macro_export] | ||||||
| macro_rules! script_hook { | macro_rules! script_hook { | ||||||
|     ($hook_name: ident, $source: ident, $($parameters: expr),*) => { |     ($hook_name: ident, $source: ident, $($parameters: expr),*) => { | ||||||
| @@ -33,6 +35,8 @@ macro_rules! script_hook { | |||||||
|     }; |     }; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | /// This macro runs a script function on all scripts in a Vec of scripts. It will ensure it only | ||||||
|  | /// runs the script function if it is not suppressed, and can take any amount of parameters. | ||||||
| #[macro_export] | #[macro_export] | ||||||
| macro_rules! run_scripts { | macro_rules! run_scripts { | ||||||
|     ($hook_name: ident, $source: ident, $($parameters: expr),*) => { |     ($hook_name: ident, $source: ident, $($parameters: expr),*) => { | ||||||
| @@ -68,14 +72,20 @@ macro_rules! run_scripts { | |||||||
|     }; |     }; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | /// The script source data is the basic data required for any script source. | ||||||
| #[derive(Default, Debug)] | #[derive(Default, Debug)] | ||||||
| pub struct ScriptSourceData { | pub struct ScriptSourceData { | ||||||
|  |     /// Whether or not the data has been initialized yet. | ||||||
|     is_initialized: bool, |     is_initialized: bool, | ||||||
|  |     /// A list that references all possible scripts on this source, and it's parents. | ||||||
|     scripts: Vec<ScriptWrapper>, |     scripts: Vec<ScriptWrapper>, | ||||||
| } | } | ||||||
|  |  | ||||||
|  | /// A script source is a struct on which we can trigger scripts to be run from. | ||||||
| pub trait ScriptSource<'a> { | pub trait ScriptSource<'a> { | ||||||
|     fn get_script_iterator(&self) -> ScriptAggregator { |     /// Gets an iterator over all the scripts that are relevant to this script source. If the data | ||||||
|  |     /// has not been initialised, it will do so here. | ||||||
|  |     fn get_script_iterator(&self) -> ScriptIterator { | ||||||
|         let lock = self.get_script_source_data(); |         let lock = self.get_script_source_data(); | ||||||
|         if !lock.read().is_initialized { |         if !lock.read().is_initialized { | ||||||
|             let mut data = lock.write(); |             let mut data = lock.write(); | ||||||
| @@ -83,17 +93,28 @@ pub trait ScriptSource<'a> { | |||||||
|             self.collect_scripts(&mut data.scripts); |             self.collect_scripts(&mut data.scripts); | ||||||
|             data.is_initialized = true; |             data.is_initialized = true; | ||||||
|         } |         } | ||||||
|         ScriptAggregator::new(&lock.read().scripts as *const Vec<ScriptWrapper>) |         ScriptIterator::new(&lock.read().scripts as *const Vec<ScriptWrapper>) | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /// The number of scripts that are expected to be relevant for this source. This generally is | ||||||
|  |     /// the number of its own scripts + the number of scripts for any parents. | ||||||
|     fn get_script_count(&self) -> usize; |     fn get_script_count(&self) -> usize; | ||||||
|  |     /// Returns the underlying data required for us to be a script source. | ||||||
|     fn get_script_source_data(&self) -> &RwLock<ScriptSourceData>; |     fn get_script_source_data(&self) -> &RwLock<ScriptSourceData>; | ||||||
|  |     /// This should add all scripts belonging to this source to the scripts Vec, disregarding its | ||||||
|  |     /// potential parents. | ||||||
|     fn get_own_scripts(&self, scripts: &mut Vec<ScriptWrapper>); |     fn get_own_scripts(&self, scripts: &mut Vec<ScriptWrapper>); | ||||||
|  |     /// This should add all scripts that are relevant to the source the the scripts Vec, including | ||||||
|  |     /// everything that belongs to its parents. | ||||||
|     fn collect_scripts(&self, scripts: &mut Vec<ScriptWrapper>); |     fn collect_scripts(&self, scripts: &mut Vec<ScriptWrapper>); | ||||||
| } | } | ||||||
|  |  | ||||||
|  | /// Enum to store both ScriptSets and sets in a single value. | ||||||
| #[derive(Debug)] | #[derive(Debug)] | ||||||
| pub enum ScriptWrapper { | pub enum ScriptWrapper { | ||||||
|  |     /// A reference to a single script. | ||||||
|     Script(Weak<RwLock<Option<Arc<dyn Script>>>>), |     Script(Weak<RwLock<Option<Arc<dyn Script>>>>), | ||||||
|  |     /// A reference to a ScriptSet. | ||||||
|     Set(Weak<ScriptSet>), |     Set(Weak<ScriptSet>), | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -109,29 +130,31 @@ impl From<&Arc<ScriptSet>> for ScriptWrapper { | |||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
| pub struct ScriptAggregator { | /// This struct allows for the iteration over scripts. | ||||||
|  | pub struct ScriptIterator { | ||||||
|  |     /// A pointer to the vector of ScriptWrappers. This can be a pointer, as we know it remains valid | ||||||
|  |     /// while we're using it. | ||||||
|     scripts: *const Vec<ScriptWrapper>, |     scripts: *const Vec<ScriptWrapper>, | ||||||
|     size: i32, |     /// The current index in the scripts. | ||||||
|     index: i32, |     index: i32, | ||||||
|  |     /// If we're currently inside a set, the current index inside the set. | ||||||
|     set_index: i32, |     set_index: i32, | ||||||
| } | } | ||||||
|  |  | ||||||
| impl ScriptAggregator { | impl ScriptIterator { | ||||||
|  |     /// Instantiates an iterator. | ||||||
|     pub fn new(scripts: *const Vec<ScriptWrapper>) -> Self { |     pub fn new(scripts: *const Vec<ScriptWrapper>) -> Self { | ||||||
|         unsafe { |  | ||||||
|             let len = scripts.as_ref().unwrap().len(); |  | ||||||
|         Self { |         Self { | ||||||
|             scripts, |             scripts, | ||||||
|                 size: len as i32, |  | ||||||
|             index: -1, |             index: -1, | ||||||
|             set_index: -1, |             set_index: -1, | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|     } |  | ||||||
|  |  | ||||||
|  |     /// Move to the next valid value in the scripts. | ||||||
|     fn increment_to_next_value(&mut self) -> bool { |     fn increment_to_next_value(&mut self) -> bool { | ||||||
|         if self.index != -1 { |         if self.index != -1 { | ||||||
|             let wrapper = unsafe { &self.scripts.as_ref().unwrap()[self.index as usize] }; |             let wrapper = unsafe { &(*self.scripts)[self.index as usize] }; | ||||||
|             if let ScriptWrapper::Set(set) = wrapper { |             if let ScriptWrapper::Set(set) = wrapper { | ||||||
|                 if let Some(set) = set.upgrade() { |                 if let Some(set) = set.upgrade() { | ||||||
|                     self.set_index += 1; |                     self.set_index += 1; | ||||||
| @@ -144,7 +167,8 @@ impl ScriptAggregator { | |||||||
|             } |             } | ||||||
|         } |         } | ||||||
|         self.index += 1; |         self.index += 1; | ||||||
|         for index in self.index..self.size { |         let len = (unsafe { &*self.scripts }).len() as i32; | ||||||
|  |         for index in self.index..len { | ||||||
|             self.index = index; |             self.index = index; | ||||||
|             let wrapper = unsafe { &self.scripts.as_ref().unwrap()[self.index as usize] }; |             let wrapper = unsafe { &self.scripts.as_ref().unwrap()[self.index as usize] }; | ||||||
|             if let ScriptWrapper::Set(s) = wrapper { |             if let ScriptWrapper::Set(s) = wrapper { | ||||||
| @@ -166,6 +190,7 @@ impl ScriptAggregator { | |||||||
|         false |         false | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /// Gets the next valid script. If none is found, returns None. | ||||||
|     pub fn get_next(&mut self) -> Option<ScriptContainer> { |     pub fn get_next(&mut self) -> Option<ScriptContainer> { | ||||||
|         if !self.increment_to_next_value() { |         if !self.increment_to_next_value() { | ||||||
|             return None; |             return None; | ||||||
| @@ -183,6 +208,7 @@ impl ScriptAggregator { | |||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /// Resets the iterator to the start. | ||||||
|     pub fn reset(&mut self) { |     pub fn reset(&mut self) { | ||||||
|         self.index = -1; |         self.index = -1; | ||||||
|         self.set_index = -1; |         self.set_index = -1; | ||||||
| @@ -260,7 +286,7 @@ mod tests { | |||||||
|     fn script_aggregator_property_iterates_single_script() { |     fn script_aggregator_property_iterates_single_script() { | ||||||
|         let script = ScriptContainer::new(Arc::new(TestScript::new())); |         let script = ScriptContainer::new(Arc::new(TestScript::new())); | ||||||
|         let scripts = vec![ScriptWrapper::from(&script)]; |         let scripts = vec![ScriptWrapper::from(&script)]; | ||||||
|         let mut aggregator = ScriptAggregator::new(&scripts as *const Vec<ScriptWrapper>); |         let mut aggregator = ScriptIterator::new(&scripts as *const Vec<ScriptWrapper>); | ||||||
|         while let Some(v) = aggregator.get_next() { |         while let Some(v) = aggregator.get_next() { | ||||||
|             v.get().unwrap().read().as_ref().unwrap().on_initialize(&[]); |             v.get().unwrap().read().as_ref().unwrap().on_initialize(&[]); | ||||||
|         } |         } | ||||||
| @@ -272,7 +298,7 @@ mod tests { | |||||||
|     fn script_aggregator_property_iterates_single_script_with_resets() { |     fn script_aggregator_property_iterates_single_script_with_resets() { | ||||||
|         let script = ScriptContainer::new(Arc::new(TestScript::new())); |         let script = ScriptContainer::new(Arc::new(TestScript::new())); | ||||||
|         let scripts = vec![ScriptWrapper::from(&script)]; |         let scripts = vec![ScriptWrapper::from(&script)]; | ||||||
|         let mut aggregator = ScriptAggregator::new(&scripts as *const Vec<ScriptWrapper>); |         let mut aggregator = ScriptIterator::new(&scripts as *const Vec<ScriptWrapper>); | ||||||
|         for i in 1..11 { |         for i in 1..11 { | ||||||
|             aggregator.reset(); |             aggregator.reset(); | ||||||
|             while let Some(v) = aggregator.get_next() { |             while let Some(v) = aggregator.get_next() { | ||||||
| @@ -293,7 +319,7 @@ mod tests { | |||||||
|             ScriptWrapper::from(&script2), |             ScriptWrapper::from(&script2), | ||||||
|             ScriptWrapper::from(&script3), |             ScriptWrapper::from(&script3), | ||||||
|         ]; |         ]; | ||||||
|         let mut aggregator = ScriptAggregator::new(&scripts as *const Vec<ScriptWrapper>); |         let mut aggregator = ScriptIterator::new(&scripts as *const Vec<ScriptWrapper>); | ||||||
|         while let Some(v) = aggregator.get_next() { |         while let Some(v) = aggregator.get_next() { | ||||||
|             v.get().unwrap().read().as_ref().unwrap().on_initialize(&[]); |             v.get().unwrap().read().as_ref().unwrap().on_initialize(&[]); | ||||||
|         } |         } | ||||||
| @@ -315,7 +341,7 @@ mod tests { | |||||||
|             ScriptWrapper::from(&script2), |             ScriptWrapper::from(&script2), | ||||||
|             ScriptWrapper::from(&script3), |             ScriptWrapper::from(&script3), | ||||||
|         ]; |         ]; | ||||||
|         let mut aggregator = ScriptAggregator::new(&scripts as *const Vec<ScriptWrapper>); |         let mut aggregator = ScriptIterator::new(&scripts as *const Vec<ScriptWrapper>); | ||||||
|         for i in 1..11 { |         for i in 1..11 { | ||||||
|             aggregator.reset(); |             aggregator.reset(); | ||||||
|             while let Some(v) = aggregator.get_next() { |             while let Some(v) = aggregator.get_next() { | ||||||
| @@ -338,7 +364,7 @@ mod tests { | |||||||
|         set.add(Arc::new(TestScript::new_with_name("test_c"))); |         set.add(Arc::new(TestScript::new_with_name("test_c"))); | ||||||
|  |  | ||||||
|         let scripts = vec![ScriptWrapper::from(&set)]; |         let scripts = vec![ScriptWrapper::from(&set)]; | ||||||
|         let mut aggregator = ScriptAggregator::new(&scripts as *const Vec<ScriptWrapper>); |         let mut aggregator = ScriptIterator::new(&scripts as *const Vec<ScriptWrapper>); | ||||||
|         for i in 1..11 { |         for i in 1..11 { | ||||||
|             aggregator.reset(); |             aggregator.reset(); | ||||||
|             while let Some(v) = aggregator.get_next() { |             while let Some(v) = aggregator.get_next() { | ||||||
| @@ -367,7 +393,7 @@ mod tests { | |||||||
|         set.add(Arc::new(TestScript::new_with_name("test_c"))); |         set.add(Arc::new(TestScript::new_with_name("test_c"))); | ||||||
|  |  | ||||||
|         let scripts = vec![ScriptWrapper::from(&set)]; |         let scripts = vec![ScriptWrapper::from(&set)]; | ||||||
|         let mut aggregator = ScriptAggregator::new(&scripts as *const Vec<ScriptWrapper>); |         let mut aggregator = ScriptIterator::new(&scripts as *const Vec<ScriptWrapper>); | ||||||
|         assert_eq!( |         assert_eq!( | ||||||
|             aggregator |             aggregator | ||||||
|                 .get_next() |                 .get_next() | ||||||
| @@ -407,7 +433,7 @@ mod tests { | |||||||
|         set.add(Arc::new(TestScript::new_with_name("test_c"))); |         set.add(Arc::new(TestScript::new_with_name("test_c"))); | ||||||
|  |  | ||||||
|         let scripts = vec![ScriptWrapper::from(&set)]; |         let scripts = vec![ScriptWrapper::from(&set)]; | ||||||
|         let mut aggregator = ScriptAggregator::new(&scripts as *const Vec<ScriptWrapper>); |         let mut aggregator = ScriptIterator::new(&scripts as *const Vec<ScriptWrapper>); | ||||||
|         assert_eq!( |         assert_eq!( | ||||||
|             aggregator |             aggregator | ||||||
|                 .get_next() |                 .get_next() | ||||||
|   | |||||||
| @@ -12,7 +12,7 @@ use crate::dynamic_data::Battle; | |||||||
| use crate::dynamic_data::DamageSource; | use crate::dynamic_data::DamageSource; | ||||||
| use crate::dynamic_data::ExecutingMove; | use crate::dynamic_data::ExecutingMove; | ||||||
| use crate::dynamic_data::Pokemon; | use crate::dynamic_data::Pokemon; | ||||||
| use crate::static_data::EffectParameter; | use crate::static_data::{EffectParameter, TypeIdentifier}; | ||||||
| use crate::static_data::{Item, Statistic}; | use crate::static_data::{Item, Statistic}; | ||||||
| use crate::StringKey; | use crate::StringKey; | ||||||
|  |  | ||||||
| @@ -105,7 +105,14 @@ pub trait Script: Send + Sync { | |||||||
|     /// move, which include the scripts that are attached to the owner of the script. |     /// move, which include the scripts that are attached to the owner of the script. | ||||||
|     fn on_move_miss(&self, _move: &ExecutingMove, _target: &Arc<Pokemon>) {} |     fn on_move_miss(&self, _move: &ExecutingMove, _target: &Arc<Pokemon>) {} | ||||||
|     /// This function allows the script to change the actual type that is used for the move on a target. |     /// This function allows the script to change the actual type that is used for the move on a target. | ||||||
|     fn change_move_type(&self, _move: &ExecutingMove, _target: &Arc<Pokemon>, _hit: u8, _move_type: &mut u8) {} |     fn change_move_type( | ||||||
|  |         &self, | ||||||
|  |         _move: &ExecutingMove, | ||||||
|  |         _target: &Arc<Pokemon>, | ||||||
|  |         _hit: u8, | ||||||
|  |         _move_type: &mut TypeIdentifier, | ||||||
|  |     ) { | ||||||
|  |     } | ||||||
|     /// This function allows the script to change how effective a move is on a target. |     /// This function allows the script to change how effective a move is on a target. | ||||||
|     fn change_effectiveness(&self, _move: &ExecutingMove, _target: &Arc<Pokemon>, _hit: u8, _effectiveness: &mut f32) {} |     fn change_effectiveness(&self, _move: &ExecutingMove, _target: &Arc<Pokemon>, _hit: u8, _effectiveness: &mut f32) {} | ||||||
|     /// This function allows a script to block an outgoing move from being critical. |     /// This function allows a script to block an outgoing move from being critical. | ||||||
|   | |||||||
| @@ -1,16 +1,23 @@ | |||||||
| use crate::dynamic_data::script_handling::script::{Script, ScriptContainer}; |  | ||||||
| use crate::{PkmnResult, StringKey}; |  | ||||||
| use indexmap::IndexMap; |  | ||||||
| use parking_lot::RwLock; |  | ||||||
| use std::ops::Deref; | use std::ops::Deref; | ||||||
| use std::sync::Arc; | use std::sync::Arc; | ||||||
|  |  | ||||||
|  | use indexmap::IndexMap; | ||||||
|  | use parking_lot::RwLock; | ||||||
|  |  | ||||||
|  | use crate::dynamic_data::script_handling::script::{Script, ScriptContainer}; | ||||||
|  | use crate::{PkmnResult, StringKey}; | ||||||
|  |  | ||||||
|  | /// A collection of unique scripts. | ||||||
| #[derive(Debug, Default)] | #[derive(Debug, Default)] | ||||||
| pub struct ScriptSet { | pub struct ScriptSet { | ||||||
|  |     /// The scripts collection. This is an indexmap so we can iterate over them and always get the | ||||||
|  |     /// scripts in the same order., while still allowing fast lookup. | ||||||
|     scripts: RwLock<IndexMap<StringKey, ScriptContainer>>, |     scripts: RwLock<IndexMap<StringKey, ScriptContainer>>, | ||||||
| } | } | ||||||
|  |  | ||||||
| impl ScriptSet { | impl ScriptSet { | ||||||
|  |     /// Adds a script to the set. If the script with that name already exists in this set, this | ||||||
|  |     /// makes that script stack instead. The return value here is that script. | ||||||
|     pub fn add(&self, script: Arc<dyn Script>) -> ScriptContainer { |     pub fn add(&self, script: Arc<dyn Script>) -> ScriptContainer { | ||||||
|         if let Some(lock) = self.scripts.read().get(script.name()) { |         if let Some(lock) = self.scripts.read().get(script.name()) { | ||||||
|             if let Some(existing) = lock.get() { |             if let Some(existing) = lock.get() { | ||||||
| @@ -27,6 +34,8 @@ impl ScriptSet { | |||||||
|         self.scripts.read().last().unwrap().1.clone() |         self.scripts.read().last().unwrap().1.clone() | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /// Adds a script with a name to the set. If the script with that name already exists in this | ||||||
|  |     /// set, this makes that script stack instead. The return value here is that script. | ||||||
|     pub fn stack_or_add<'b, F>(&self, key: &StringKey, instantiation: &'b F) -> PkmnResult<Option<ScriptContainer>> |     pub fn stack_or_add<'b, F>(&self, key: &StringKey, instantiation: &'b F) -> PkmnResult<Option<ScriptContainer>> | ||||||
|     where |     where | ||||||
|         F: Fn() -> PkmnResult<Option<Arc<dyn Script>>>, |         F: Fn() -> PkmnResult<Option<Arc<dyn Script>>>, | ||||||
| @@ -51,10 +60,12 @@ impl ScriptSet { | |||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /// Gets a script from the set using its unique name. | ||||||
|     pub fn get(&self, key: &StringKey) -> Option<ScriptContainer> { |     pub fn get(&self, key: &StringKey) -> Option<ScriptContainer> { | ||||||
|         self.scripts.read().get(key).cloned() |         self.scripts.read().get(key).cloned() | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /// Removes a script from the set using its unique name. | ||||||
|     pub fn remove(&self, key: &StringKey) { |     pub fn remove(&self, key: &StringKey) { | ||||||
|         let value = self.scripts.write().shift_remove(key); |         let value = self.scripts.write().shift_remove(key); | ||||||
|         if let Some(script) = value { |         if let Some(script) = value { | ||||||
| @@ -66,6 +77,7 @@ impl ScriptSet { | |||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /// Clears all scripts from the set. | ||||||
|     pub fn clear(&self) { |     pub fn clear(&self) { | ||||||
|         for script in self.scripts.read().deref() { |         for script in self.scripts.read().deref() { | ||||||
|             if let Some(script) = script.1.get() { |             if let Some(script) = script.1.get() { | ||||||
| @@ -77,18 +89,24 @@ impl ScriptSet { | |||||||
|         self.scripts.write().clear(); |         self.scripts.write().clear(); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /// Checks if the set has a script with the given name. | ||||||
|     pub fn has(&self, key: &StringKey) -> bool { |     pub fn has(&self, key: &StringKey) -> bool { | ||||||
|         self.scripts.read().contains_key(key) |         self.scripts.read().contains_key(key) | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /// Gets a script from the set at a specific index. | ||||||
|     pub fn at(&self, index: usize) -> ScriptContainer { |     pub fn at(&self, index: usize) -> ScriptContainer { | ||||||
|         self.scripts.read()[index].clone() |         self.scripts.read()[index].clone() | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /// Gets the number of scripts in the set. | ||||||
|     pub fn count(&self) -> usize { |     pub fn count(&self) -> usize { | ||||||
|         self.scripts.read().len() |         self.scripts.read().len() | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /// Get a vector of the scripts in this set. This copies the current scripts into a Vec, and | ||||||
|  |     /// returns that. This allows modifying the scripts in the set, while still being able to iterate | ||||||
|  |     /// over them. | ||||||
|     pub(crate) fn get_owning_iterator(&self) -> Vec<ScriptContainer> { |     pub(crate) fn get_owning_iterator(&self) -> Vec<ScriptContainer> { | ||||||
|         let s = self.scripts.read(); |         let s = self.scripts.read(); | ||||||
|         let mut v = Vec::with_capacity(s.deref().len()); |         let mut v = Vec::with_capacity(s.deref().len()); | ||||||
|   | |||||||
| @@ -1,25 +1,33 @@ | |||||||
|  | use std::sync::Arc; | ||||||
|  | 
 | ||||||
| use crate::dynamic_data::script_handling::script::{Script, ScriptContainer}; | use crate::dynamic_data::script_handling::script::{Script, ScriptContainer}; | ||||||
| use crate::dynamic_data::script_handling::script_set::ScriptSet; | use crate::dynamic_data::script_handling::script_set::ScriptSet; | ||||||
| use crate::{PkmnResult, StringKey}; | use crate::{PkmnResult, StringKey}; | ||||||
| use std::sync::Arc; |  | ||||||
| 
 | 
 | ||||||
| pub trait VolatileScripts<'a> { | /// This trait adds a bunch of helper functions to deal with volatile scripts on a struct.
 | ||||||
|  | pub trait VolatileScriptsOwner<'a> { | ||||||
|  |     /// Return the [`ScriptSet`] that are our volatile scripts.
 | ||||||
|     fn volatile_scripts(&self) -> &Arc<ScriptSet>; |     fn volatile_scripts(&self) -> &Arc<ScriptSet>; | ||||||
|  |     /// Loads a volatile script by name.
 | ||||||
|     fn load_volatile_script(&self, key: &StringKey) -> PkmnResult<Option<Arc<dyn Script>>>; |     fn load_volatile_script(&self, key: &StringKey) -> PkmnResult<Option<Arc<dyn Script>>>; | ||||||
| 
 | 
 | ||||||
|  |     /// Check if a volatile script with given name exists.
 | ||||||
|     fn has_volatile_script(&self, key: &StringKey) -> bool { |     fn has_volatile_script(&self, key: &StringKey) -> bool { | ||||||
|         self.volatile_scripts().has(key) |         self.volatile_scripts().has(key) | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |     /// Gets a volatile script by name.
 | ||||||
|     fn get_volatile_script(&self, key: &StringKey) -> Option<ScriptContainer> { |     fn get_volatile_script(&self, key: &StringKey) -> Option<ScriptContainer> { | ||||||
|         self.volatile_scripts().get(key) |         self.volatile_scripts().get(key) | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |     /// Adds a volatile script by name.
 | ||||||
|     fn add_volatile_script(&mut self, key: &StringKey) -> PkmnResult<Option<ScriptContainer>> { |     fn add_volatile_script(&mut self, key: &StringKey) -> PkmnResult<Option<ScriptContainer>> { | ||||||
|         self.volatile_scripts() |         self.volatile_scripts() | ||||||
|             .stack_or_add(key, &|| self.load_volatile_script(key)) |             .stack_or_add(key, &|| self.load_volatile_script(key)) | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |     /// Removes a volatile script by name.
 | ||||||
|     fn remove_volatile_script(&mut self, key: &StringKey) { |     fn remove_volatile_script(&mut self, key: &StringKey) { | ||||||
|         self.volatile_scripts().remove(key) |         self.volatile_scripts().remove(key) | ||||||
|     } |     } | ||||||
| @@ -1,15 +1,21 @@ | |||||||
| use crate::defines::LevelInt; | use crate::defines::LevelInt; | ||||||
|  |  | ||||||
|  | /// A growth rate defines how much experience is required per level. | ||||||
| pub trait GrowthRate { | pub trait GrowthRate { | ||||||
|  |     /// Calculate the level something with this growth rate would have at a certain experience. | ||||||
|     fn calculate_level(&self, experience: u32) -> LevelInt; |     fn calculate_level(&self, experience: u32) -> LevelInt; | ||||||
|  |     /// Calculate the experience something with this growth rate would have at a certain level. | ||||||
|     fn calculate_experience(&self, level: LevelInt) -> u32; |     fn calculate_experience(&self, level: LevelInt) -> u32; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | /// An implementation of the growth rate that uses a lookup table for experience. | ||||||
| pub struct LookupGrowthRate { | pub struct LookupGrowthRate { | ||||||
|  |     /// The lookup Vec. | ||||||
|     experience: Vec<u32>, |     experience: Vec<u32>, | ||||||
| } | } | ||||||
|  |  | ||||||
| impl LookupGrowthRate { | impl LookupGrowthRate { | ||||||
|  |     /// Instantiates a new lookup growth rate. | ||||||
|     pub fn new(experience: Vec<u32>) -> LookupGrowthRate { |     pub fn new(experience: Vec<u32>) -> LookupGrowthRate { | ||||||
|         LookupGrowthRate { experience } |         LookupGrowthRate { experience } | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -4,41 +4,63 @@ use serde::{Deserialize, Serialize}; | |||||||
|  |  | ||||||
| use crate::StringKey; | use crate::StringKey; | ||||||
|  |  | ||||||
|  | /// An item category defines which bag slot items are stored in. | ||||||
| #[derive(Debug, Copy, Clone)] | #[derive(Debug, Copy, Clone)] | ||||||
| #[cfg_attr(feature = "serde", derive(Serialize, Deserialize))] | #[cfg_attr(feature = "serde", derive(Serialize, Deserialize))] | ||||||
| #[repr(u8)] | #[repr(u8)] | ||||||
| pub enum ItemCategory { | pub enum ItemCategory { | ||||||
|  |     /// This is where most items should go. | ||||||
|     MiscItem, |     MiscItem, | ||||||
|  |     /// Pokeballs are used for capturing Pokemons. | ||||||
|     Pokeball, |     Pokeball, | ||||||
|  |     /// Medicine is used for healing HP, PP, and status effects | ||||||
|     Medicine, |     Medicine, | ||||||
|  |     /// Berry is used for all berries. | ||||||
|     Berry, |     Berry, | ||||||
|  |     /// TMHM is used for Technical and Hidden Machines. | ||||||
|     TMHM, |     TMHM, | ||||||
|  |     /// Form Changer is used for items that change forms, such as mega stones. | ||||||
|     FormChanger, |     FormChanger, | ||||||
|  |     /// Key Items are single stored items, generally used for story progression. | ||||||
|     KeyItem, |     KeyItem, | ||||||
|  |     /// Mail is used for mail items. | ||||||
|     Mail, |     Mail, | ||||||
| } | } | ||||||
|  |  | ||||||
|  | /// A battle item category defines how the item is categorized when in battle. | ||||||
| #[derive(Debug, Copy, Clone)] | #[derive(Debug, Copy, Clone)] | ||||||
| #[cfg_attr(feature = "serde", derive(Serialize, Deserialize))] | #[cfg_attr(feature = "serde", derive(Serialize, Deserialize))] | ||||||
| #[repr(u8)] | #[repr(u8)] | ||||||
| pub enum BattleItemCategory { | pub enum BattleItemCategory { | ||||||
|  |     /// This item can't be used in battle. | ||||||
|     None, |     None, | ||||||
|  |     /// This item is used for healing Pokemon. | ||||||
|     Healing, |     Healing, | ||||||
|  |     /// This item is used for healing Pokemon from a status. | ||||||
|     StatusHealing, |     StatusHealing, | ||||||
|  |     /// This item is used for capturing Pokemon. | ||||||
|     Pokeball, |     Pokeball, | ||||||
|  |     /// This item does not belong in above categories, but is still a battle item. | ||||||
|     MiscBattleItem, |     MiscBattleItem, | ||||||
| } | } | ||||||
|  |  | ||||||
|  | /// An item is an object which the player can pick up, keep in their Bag, and use in some manner | ||||||
| #[derive(Debug)] | #[derive(Debug)] | ||||||
| pub struct Item { | pub struct Item { | ||||||
|  |     /// The name of the item. | ||||||
|     name: StringKey, |     name: StringKey, | ||||||
|  |     /// Which bag slot items are stored in. | ||||||
|     category: ItemCategory, |     category: ItemCategory, | ||||||
|  |     /// How the item is categorized when in battle. | ||||||
|     battle_category: BattleItemCategory, |     battle_category: BattleItemCategory, | ||||||
|  |     /// The buying value of the item. | ||||||
|     price: i32, |     price: i32, | ||||||
|  |     /// A set of arbitrary flags that can be set on the item. | ||||||
|     flags: HashSet<StringKey>, |     flags: HashSet<StringKey>, | ||||||
| } | } | ||||||
|  |  | ||||||
| impl Item { | impl Item { | ||||||
|  |     /// Instantiates an item. | ||||||
|     pub fn new( |     pub fn new( | ||||||
|         name: &StringKey, |         name: &StringKey, | ||||||
|         category: ItemCategory, |         category: ItemCategory, | ||||||
| @@ -55,22 +77,28 @@ impl Item { | |||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /// The name of the item. | ||||||
|     pub fn name(&self) -> &StringKey { |     pub fn name(&self) -> &StringKey { | ||||||
|         &self.name |         &self.name | ||||||
|     } |     } | ||||||
|  |     /// Which bag slot items are stored in. | ||||||
|     pub fn category(&self) -> ItemCategory { |     pub fn category(&self) -> ItemCategory { | ||||||
|         self.category |         self.category | ||||||
|     } |     } | ||||||
|  |     /// How the item is categorized when in battle. | ||||||
|     pub fn battle_category(&self) -> BattleItemCategory { |     pub fn battle_category(&self) -> BattleItemCategory { | ||||||
|         self.battle_category |         self.battle_category | ||||||
|     } |     } | ||||||
|  |     /// The buying value of the item. | ||||||
|     pub fn price(&self) -> i32 { |     pub fn price(&self) -> i32 { | ||||||
|         self.price |         self.price | ||||||
|     } |     } | ||||||
|  |     /// A set of arbitrary flags that can be set on the item. | ||||||
|     pub fn flags(&self) -> &HashSet<StringKey> { |     pub fn flags(&self) -> &HashSet<StringKey> { | ||||||
|         &self.flags |         &self.flags | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /// Checks whether the item has a specific flag. | ||||||
|     pub fn has_flag(&self, key: &StringKey) -> bool { |     pub fn has_flag(&self, key: &StringKey) -> bool { | ||||||
|         self.flags.contains(key) |         self.flags.contains(key) | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -1,35 +1,31 @@ | |||||||
| use hashbrown::HashMap; | use indexmap::IndexMap; | ||||||
|  |  | ||||||
| use crate::static_data::Ability; | use crate::static_data::Ability; | ||||||
| use crate::static_data::DataLibrary; | use crate::static_data::DataLibrary; | ||||||
| use crate::StringKey; | use crate::StringKey; | ||||||
|  |  | ||||||
|  | /// A storage for all abilities that can be used in this data library. | ||||||
| #[derive(Debug)] | #[derive(Debug)] | ||||||
| pub struct AbilityLibrary { | pub struct AbilityLibrary { | ||||||
|     map: HashMap<StringKey, Box<Ability>>, |     /// The underlying map for the library. | ||||||
|     list: Vec<StringKey>, |     map: IndexMap<StringKey, Box<Ability>>, | ||||||
| } | } | ||||||
|  |  | ||||||
| impl AbilityLibrary { | impl AbilityLibrary { | ||||||
|  |     /// Instantiates a new ability library. | ||||||
|     pub fn new(capacity: usize) -> AbilityLibrary { |     pub fn new(capacity: usize) -> AbilityLibrary { | ||||||
|         AbilityLibrary { |         AbilityLibrary { | ||||||
|             map: HashMap::with_capacity(capacity), |             map: IndexMap::with_capacity(capacity), | ||||||
|             list: Vec::with_capacity(capacity), |  | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
| impl DataLibrary<'_, Box<Ability>> for AbilityLibrary { | impl DataLibrary<'_, Box<Ability>> for AbilityLibrary { | ||||||
|     fn map(&self) -> &HashMap<StringKey, Box<Ability>> { |     fn map(&self) -> &IndexMap<StringKey, Box<Ability>> { | ||||||
|         &self.map |         &self.map | ||||||
|     } |     } | ||||||
|  |     fn get_modify(&mut self) -> &mut IndexMap<StringKey, Box<Ability>> { | ||||||
|     fn list_values(&self) -> &Vec<StringKey> { |         &mut self.map | ||||||
|         &self.list |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     fn get_modify(&mut self) -> (&mut HashMap<StringKey, Box<Ability>>, &mut Vec<StringKey>) { |  | ||||||
|         (&mut self.map, &mut self.list) |  | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -62,17 +58,8 @@ pub mod tests { | |||||||
|     #[test] |     #[test] | ||||||
|     fn get_ability_library_direct_map_access() { |     fn get_ability_library_direct_map_access() { | ||||||
|         let lib = build(); |         let lib = build(); | ||||||
|         let map = lib.map(); |         let ability = lib.get(&"test_ability".into()); | ||||||
|         let ability = map.get(&"test_ability".into()); |  | ||||||
|         assert!(ability.is_some()); |         assert!(ability.is_some()); | ||||||
|         assert_eq!(ability.unwrap().name(), &"test_ability".into()); |         assert_eq!(ability.unwrap().name(), &"test_ability".into()); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     #[test] |  | ||||||
|     fn get_ability_library_direct_list_access() { |  | ||||||
|         let lib = build(); |  | ||||||
|         let list = lib.list_values(); |  | ||||||
|         assert_eq!(list.len(), 1); |  | ||||||
|         assert!(list.contains(&StringKey::new("test_ability"))); |  | ||||||
|     } |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -1,43 +1,48 @@ | |||||||
|  | use indexmap::IndexMap; | ||||||
|  |  | ||||||
| use crate::Random; | use crate::Random; | ||||||
| use crate::StringKey; | use crate::StringKey; | ||||||
| use hashbrown::HashMap; |  | ||||||
|  |  | ||||||
|  | /// A data library is a collection of methods to set up a default library, where values are stored | ||||||
|  | /// by both key, while keeping their insertion order. | ||||||
| pub trait DataLibrary<'a, T: 'a> { | pub trait DataLibrary<'a, T: 'a> { | ||||||
|     fn map(&self) -> &HashMap<StringKey, T>; |     /// Returns the underlying map. | ||||||
|     fn list_values(&self) -> &Vec<StringKey>; |     fn map(&self) -> &IndexMap<StringKey, T>; | ||||||
|     fn get_modify(&mut self) -> (&mut HashMap<StringKey, T>, &mut Vec<StringKey>); |     /// Returns the underlying map in mutable manner. | ||||||
|  |     fn get_modify(&mut self) -> &mut IndexMap<StringKey, T>; | ||||||
|  |  | ||||||
|  |     /// Adds a new value to the library. | ||||||
|     fn add(&mut self, key: &StringKey, value: T) { |     fn add(&mut self, key: &StringKey, value: T) { | ||||||
|         let modifies = self.get_modify(); |         self.get_modify().insert(key.clone(), value); | ||||||
|         modifies.0.insert(key.clone(), value); |  | ||||||
|         modifies.1.push(key.clone()); |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /// Removes a value from the library. | ||||||
|     fn remove(&mut self, key: &StringKey) { |     fn remove(&mut self, key: &StringKey) { | ||||||
|         let modifies = self.get_modify(); |         self.get_modify().remove(key); | ||||||
|         let index = modifies.1.iter().position(|r| r == key).unwrap(); |  | ||||||
|         modifies.0.remove(key); |  | ||||||
|         modifies.1.remove(index); |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /// Gets a value from the library. | ||||||
|     fn get(&'a self, key: &StringKey) -> Option<&'a T> { |     fn get(&'a self, key: &StringKey) -> Option<&'a T> { | ||||||
|         self.map().get(key) |         self.map().get::<StringKey>(key) | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /// Gets a mutable value from the library. | ||||||
|     fn get_mut(&mut self, key: &StringKey) -> Option<&mut T> { |     fn get_mut(&mut self, key: &StringKey) -> Option<&mut T> { | ||||||
|         self.get_modify().0.get_mut(key) |         self.get_modify().get_mut(key) | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /// Gets the amount of values in the library. | ||||||
|     fn len(&self) -> usize { |     fn len(&self) -> usize { | ||||||
|         self.map().len() |         self.map().len() | ||||||
|     } |     } | ||||||
|  |     /// Returns whether the library has no values. | ||||||
|     fn is_empty(&self) -> bool { |     fn is_empty(&self) -> bool { | ||||||
|         self.map().is_empty() |         self.map().is_empty() | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /// Gets a random value from the library. | ||||||
|     fn random_value(&self, rand: &mut Random) -> &T { |     fn random_value(&self, rand: &mut Random) -> &T { | ||||||
|         let i = rand.get_between(0, self.list_values().len() as i32); |         let i = rand.get_between(0, self.len() as i32); | ||||||
|         let key = &self.list_values()[i as usize]; |         return &self.map().get_index(i as usize).unwrap().1; | ||||||
|         return &self.map()[key]; |  | ||||||
|     } |     } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -7,23 +7,30 @@ use crate::defines::LevelInt; | |||||||
| use crate::static_data::GrowthRate; | use crate::static_data::GrowthRate; | ||||||
| use crate::StringKey; | use crate::StringKey; | ||||||
|  |  | ||||||
|  | /// A library to store all growth rates. | ||||||
| pub struct GrowthRateLibrary { | pub struct GrowthRateLibrary { | ||||||
|  |     /// The underlying data structure. | ||||||
|     growth_rates: HashMap<StringKey, Box<dyn GrowthRate>>, |     growth_rates: HashMap<StringKey, Box<dyn GrowthRate>>, | ||||||
| } | } | ||||||
|  |  | ||||||
| impl GrowthRateLibrary { | impl GrowthRateLibrary { | ||||||
|  |     /// Instantiates a new growth rate library with a capacity. | ||||||
|     pub fn new(capacity: usize) -> GrowthRateLibrary { |     pub fn new(capacity: usize) -> GrowthRateLibrary { | ||||||
|         GrowthRateLibrary { |         GrowthRateLibrary { | ||||||
|             growth_rates: HashMap::with_capacity(capacity), |             growth_rates: HashMap::with_capacity(capacity), | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /// Calculates the level for a given growth key name and a certain experience. | ||||||
|     pub fn calculate_level(&self, growth_rate: &StringKey, experience: u32) -> LevelInt { |     pub fn calculate_level(&self, growth_rate: &StringKey, experience: u32) -> LevelInt { | ||||||
|         self.growth_rates[growth_rate].calculate_level(experience) |         self.growth_rates[growth_rate].calculate_level(experience) | ||||||
|     } |     } | ||||||
|  |     /// Calculates the experience for a given growth key name and a certain level. | ||||||
|     pub fn calculate_experience(&self, growth_rate: &StringKey, level: LevelInt) -> u32 { |     pub fn calculate_experience(&self, growth_rate: &StringKey, level: LevelInt) -> u32 { | ||||||
|         self.growth_rates[growth_rate].calculate_experience(level) |         self.growth_rates[growth_rate].calculate_experience(level) | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /// Adds a new growth rate with a name and value. | ||||||
|     pub fn add_growth_rate(&mut self, key: &StringKey, value: Box<dyn GrowthRate>) { |     pub fn add_growth_rate(&mut self, key: &StringKey, value: Box<dyn GrowthRate>) { | ||||||
|         self.growth_rates.insert(key.clone(), value); |         self.growth_rates.insert(key.clone(), value); | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -1,35 +1,32 @@ | |||||||
| use hashbrown::HashMap; | use indexmap::IndexMap; | ||||||
|  |  | ||||||
| use crate::static_data::DataLibrary; | use crate::static_data::DataLibrary; | ||||||
| use crate::static_data::Item; | use crate::static_data::Item; | ||||||
| use crate::StringKey; | use crate::StringKey; | ||||||
|  |  | ||||||
|  | /// A library to store all items. | ||||||
| #[derive(Debug)] | #[derive(Debug)] | ||||||
| pub struct ItemLibrary { | pub struct ItemLibrary { | ||||||
|     map: HashMap<StringKey, Box<Item>>, |     /// The underlying data structure. | ||||||
|     list: Vec<StringKey>, |     map: IndexMap<StringKey, Box<Item>>, | ||||||
| } | } | ||||||
|  |  | ||||||
| impl ItemLibrary { | impl ItemLibrary { | ||||||
|  |     /// Instantiates a new Item Library. | ||||||
|     pub fn new(capacity: usize) -> ItemLibrary { |     pub fn new(capacity: usize) -> ItemLibrary { | ||||||
|         ItemLibrary { |         ItemLibrary { | ||||||
|             map: HashMap::with_capacity(capacity), |             map: IndexMap::with_capacity(capacity), | ||||||
|             list: Vec::with_capacity(capacity), |  | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
| impl DataLibrary<'_, Box<Item>> for ItemLibrary { | impl DataLibrary<'_, Box<Item>> for ItemLibrary { | ||||||
|     fn map(&self) -> &HashMap<StringKey, Box<Item>> { |     fn map(&self) -> &IndexMap<StringKey, Box<Item>> { | ||||||
|         &self.map |         &self.map | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     fn list_values(&self) -> &Vec<StringKey> { |     fn get_modify(&mut self) -> &mut IndexMap<StringKey, Box<Item>> { | ||||||
|         &self.list |         &mut self.map | ||||||
|     } |  | ||||||
|  |  | ||||||
|     fn get_modify(&mut self) -> (&mut HashMap<StringKey, Box<Item>>, &mut Vec<StringKey>) { |  | ||||||
|         (&mut self.map, &mut self.list) |  | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1,15 +1,19 @@ | |||||||
| use crate::defines::LevelInt; | use crate::defines::LevelInt; | ||||||
|  |  | ||||||
|  | /// This library holds several misc settings for the library. | ||||||
| #[derive(Debug)] | #[derive(Debug)] | ||||||
| pub struct LibrarySettings { | pub struct LibrarySettings { | ||||||
|  |     /// The highest level a Pokemon can be. | ||||||
|     maximum_level: LevelInt, |     maximum_level: LevelInt, | ||||||
| } | } | ||||||
|  |  | ||||||
| impl LibrarySettings { | impl LibrarySettings { | ||||||
|  |     /// Creates a new settings library. | ||||||
|     pub fn new(maximum_level: LevelInt) -> Self { |     pub fn new(maximum_level: LevelInt) -> Self { | ||||||
|         Self { maximum_level } |         Self { maximum_level } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /// The highest level a Pokemon can be. | ||||||
|     pub fn maximum_level(&self) -> LevelInt { |     pub fn maximum_level(&self) -> LevelInt { | ||||||
|         self.maximum_level |         self.maximum_level | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -15,7 +15,7 @@ pub use species_library::SpeciesLibrary; | |||||||
| #[doc(inline)] | #[doc(inline)] | ||||||
| pub use static_data::StaticData; | pub use static_data::StaticData; | ||||||
| #[doc(inline)] | #[doc(inline)] | ||||||
| pub use type_library::TypeLibrary; | pub use type_library::*; | ||||||
|  |  | ||||||
| mod ability_library; | mod ability_library; | ||||||
| mod data_library; | mod data_library; | ||||||
|   | |||||||
| @@ -1,35 +1,31 @@ | |||||||
| use hashbrown::HashMap; | use indexmap::IndexMap; | ||||||
|  |  | ||||||
| use crate::static_data::DataLibrary; | use crate::static_data::DataLibrary; | ||||||
| use crate::static_data::MoveData; | use crate::static_data::MoveData; | ||||||
| use crate::StringKey; | use crate::StringKey; | ||||||
|  |  | ||||||
|  | /// A library to store all data for moves. | ||||||
| #[derive(Debug)] | #[derive(Debug)] | ||||||
| pub struct MoveLibrary { | pub struct MoveLibrary { | ||||||
|     map: HashMap<StringKey, MoveData>, |     /// The underlying map. | ||||||
|     list: Vec<StringKey>, |     map: IndexMap<StringKey, MoveData>, | ||||||
| } | } | ||||||
|  |  | ||||||
| impl MoveLibrary { | impl MoveLibrary { | ||||||
|  |     /// Instantiates a new Move Library. | ||||||
|     pub fn new(capacity: usize) -> MoveLibrary { |     pub fn new(capacity: usize) -> MoveLibrary { | ||||||
|         MoveLibrary { |         MoveLibrary { | ||||||
|             map: HashMap::with_capacity(capacity), |             map: IndexMap::with_capacity(capacity), | ||||||
|             list: Vec::with_capacity(capacity), |  | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
| impl DataLibrary<'_, MoveData> for MoveLibrary { | impl DataLibrary<'_, MoveData> for MoveLibrary { | ||||||
|     fn map(&self) -> &HashMap<StringKey, MoveData> { |     fn map(&self) -> &IndexMap<StringKey, MoveData> { | ||||||
|         &self.map |         &self.map | ||||||
|     } |     } | ||||||
|  |     fn get_modify(&mut self) -> &mut IndexMap<StringKey, MoveData> { | ||||||
|     fn list_values(&self) -> &Vec<StringKey> { |         &mut self.map | ||||||
|         &self.list |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     fn get_modify(&mut self) -> (&mut HashMap<StringKey, MoveData>, &mut Vec<StringKey>) { |  | ||||||
|         (&mut self.map, &mut self.list) |  | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -45,7 +41,7 @@ pub mod tests { | |||||||
|     fn build_move() -> MoveData { |     fn build_move() -> MoveData { | ||||||
|         MoveData::new( |         MoveData::new( | ||||||
|             &"foo".into(), |             &"foo".into(), | ||||||
|             0, |             0.into(), | ||||||
|             MoveCategory::Physical, |             MoveCategory::Physical, | ||||||
|             100, |             100, | ||||||
|             100, |             100, | ||||||
|   | |||||||
| @@ -1,35 +1,31 @@ | |||||||
| use hashbrown::HashMap; | use indexmap::IndexMap; | ||||||
|  |  | ||||||
| use crate::static_data::DataLibrary; | use crate::static_data::DataLibrary; | ||||||
| use crate::static_data::Species; | use crate::static_data::Species; | ||||||
| use crate::StringKey; | use crate::StringKey; | ||||||
|  |  | ||||||
|  | /// A library to store all data for Pokemon species. | ||||||
| #[derive(Debug)] | #[derive(Debug)] | ||||||
| pub struct SpeciesLibrary { | pub struct SpeciesLibrary { | ||||||
|     map: HashMap<StringKey, Box<Species>>, |     /// The underlying map. | ||||||
|     list: Vec<StringKey>, |     map: IndexMap<StringKey, Box<Species>>, | ||||||
| } | } | ||||||
|  |  | ||||||
| impl SpeciesLibrary { | impl SpeciesLibrary { | ||||||
|  |     /// Instantiates a new Species Library. | ||||||
|     pub fn new(capacity: usize) -> SpeciesLibrary { |     pub fn new(capacity: usize) -> SpeciesLibrary { | ||||||
|         SpeciesLibrary { |         SpeciesLibrary { | ||||||
|             map: HashMap::with_capacity(capacity), |             map: IndexMap::with_capacity(capacity), | ||||||
|             list: Vec::with_capacity(capacity), |  | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
| impl<'a> DataLibrary<'a, Box<Species>> for SpeciesLibrary { | impl<'a> DataLibrary<'a, Box<Species>> for SpeciesLibrary { | ||||||
|     fn map(&self) -> &HashMap<StringKey, Box<Species>> { |     fn map(&self) -> &IndexMap<StringKey, Box<Species>> { | ||||||
|         &self.map |         &self.map | ||||||
|     } |     } | ||||||
|  |     fn get_modify(&mut self) -> &mut IndexMap<StringKey, Box<Species>> { | ||||||
|     fn list_values(&self) -> &Vec<StringKey> { |         &mut self.map | ||||||
|         &self.list |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     fn get_modify(&mut self) -> (&mut HashMap<StringKey, Box<Species>>, &mut Vec<StringKey>) { |  | ||||||
|         (&mut self.map, &mut self.list) |  | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -7,19 +7,29 @@ use crate::static_data::NatureLibrary; | |||||||
| use crate::static_data::SpeciesLibrary; | use crate::static_data::SpeciesLibrary; | ||||||
| use crate::static_data::TypeLibrary; | use crate::static_data::TypeLibrary; | ||||||
|  |  | ||||||
|  | /// The storage for all different libraries. | ||||||
| #[derive(Debug)] | #[derive(Debug)] | ||||||
| pub struct StaticData { | pub struct StaticData { | ||||||
|  |     /// Several misc settings for the library. | ||||||
|     settings: LibrarySettings, |     settings: LibrarySettings, | ||||||
|  |     /// All data for Pokemon species. | ||||||
|     species: SpeciesLibrary, |     species: SpeciesLibrary, | ||||||
|  |     /// All data for the moves. | ||||||
|     moves: MoveLibrary, |     moves: MoveLibrary, | ||||||
|  |     /// All data for the items. | ||||||
|     items: ItemLibrary, |     items: ItemLibrary, | ||||||
|  |     /// All data for growth rates. | ||||||
|     growth_rates: GrowthRateLibrary, |     growth_rates: GrowthRateLibrary, | ||||||
|  |     /// All data related to types and type effectiveness. | ||||||
|     types: TypeLibrary, |     types: TypeLibrary, | ||||||
|  |     /// All data related to natures. | ||||||
|     natures: NatureLibrary, |     natures: NatureLibrary, | ||||||
|  |     /// All data related to abilities. | ||||||
|     abilities: AbilityLibrary, |     abilities: AbilityLibrary, | ||||||
| } | } | ||||||
|  |  | ||||||
| impl StaticData { | impl StaticData { | ||||||
|  |     /// Instantiates a new data collection. | ||||||
|     pub fn new(settings: LibrarySettings) -> Self { |     pub fn new(settings: LibrarySettings) -> Self { | ||||||
|         Self { |         Self { | ||||||
|             settings, |             settings, | ||||||
| @@ -32,50 +42,64 @@ impl StaticData { | |||||||
|             abilities: AbilityLibrary::new(0), |             abilities: AbilityLibrary::new(0), | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |     /// Several misc settings for the library. | ||||||
|     pub fn settings(&self) -> &LibrarySettings { |     pub fn settings(&self) -> &LibrarySettings { | ||||||
|         &self.settings |         &self.settings | ||||||
|     } |     } | ||||||
|  |     /// All data for Pokemon species. | ||||||
|     pub fn species(&self) -> &SpeciesLibrary { |     pub fn species(&self) -> &SpeciesLibrary { | ||||||
|         &self.species |         &self.species | ||||||
|     } |     } | ||||||
|  |     /// All data for Pokemon species. | ||||||
|     pub fn species_mut(&mut self) -> &mut SpeciesLibrary { |     pub fn species_mut(&mut self) -> &mut SpeciesLibrary { | ||||||
|         &mut self.species |         &mut self.species | ||||||
|     } |     } | ||||||
|  |     /// All data for the moves. | ||||||
|     pub fn moves(&self) -> &MoveLibrary { |     pub fn moves(&self) -> &MoveLibrary { | ||||||
|         &self.moves |         &self.moves | ||||||
|     } |     } | ||||||
|  |     /// All data for the moves. | ||||||
|     pub fn moves_mut(&mut self) -> &mut MoveLibrary { |     pub fn moves_mut(&mut self) -> &mut MoveLibrary { | ||||||
|         &mut self.moves |         &mut self.moves | ||||||
|     } |     } | ||||||
|  |     /// All data for the items. | ||||||
|     pub fn items(&self) -> &ItemLibrary { |     pub fn items(&self) -> &ItemLibrary { | ||||||
|         &self.items |         &self.items | ||||||
|     } |     } | ||||||
|  |     /// All data for the items. | ||||||
|     pub fn items_mut(&mut self) -> &mut ItemLibrary { |     pub fn items_mut(&mut self) -> &mut ItemLibrary { | ||||||
|         &mut self.items |         &mut self.items | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /// All data for growth rates. | ||||||
|     pub fn growth_rates(&self) -> &GrowthRateLibrary { |     pub fn growth_rates(&self) -> &GrowthRateLibrary { | ||||||
|         &self.growth_rates |         &self.growth_rates | ||||||
|     } |     } | ||||||
|  |     /// All data for growth rates. | ||||||
|     pub fn growth_rates_mut(&mut self) -> &mut GrowthRateLibrary { |     pub fn growth_rates_mut(&mut self) -> &mut GrowthRateLibrary { | ||||||
|         &mut self.growth_rates |         &mut self.growth_rates | ||||||
|     } |     } | ||||||
|  |     /// All data related to types and type effectiveness. | ||||||
|     pub fn types(&self) -> &TypeLibrary { |     pub fn types(&self) -> &TypeLibrary { | ||||||
|         &self.types |         &self.types | ||||||
|     } |     } | ||||||
|  |     /// All data related to types and type effectiveness. | ||||||
|     pub fn types_mut(&mut self) -> &mut TypeLibrary { |     pub fn types_mut(&mut self) -> &mut TypeLibrary { | ||||||
|         &mut self.types |         &mut self.types | ||||||
|     } |     } | ||||||
|  |     /// All data related to natures. | ||||||
|     pub fn natures(&self) -> &NatureLibrary { |     pub fn natures(&self) -> &NatureLibrary { | ||||||
|         &self.natures |         &self.natures | ||||||
|     } |     } | ||||||
|  |     /// All data related to natures. | ||||||
|     pub fn natures_mut(&mut self) -> &mut NatureLibrary { |     pub fn natures_mut(&mut self) -> &mut NatureLibrary { | ||||||
|         &mut self.natures |         &mut self.natures | ||||||
|     } |     } | ||||||
|  |     /// All data related to abilities. | ||||||
|     pub fn abilities(&self) -> &AbilityLibrary { |     pub fn abilities(&self) -> &AbilityLibrary { | ||||||
|         &self.abilities |         &self.abilities | ||||||
|     } |     } | ||||||
|  |     /// All data related to abilities. | ||||||
|     pub fn abilities_mut(&mut self) -> &mut AbilityLibrary { |     pub fn abilities_mut(&mut self) -> &mut AbilityLibrary { | ||||||
|         &mut self.abilities |         &mut self.abilities | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -1,13 +1,32 @@ | |||||||
| use crate::StringKey; |  | ||||||
| use hashbrown::HashMap; | use hashbrown::HashMap; | ||||||
|  |  | ||||||
|  | use crate::StringKey; | ||||||
|  |  | ||||||
|  | /// A unique key that can be used to store a reference to a type. Opaque reference to a byte | ||||||
|  | /// internally. | ||||||
|  | #[derive(Debug, Copy, Clone, Eq, PartialEq, Default, Hash)] | ||||||
|  | pub struct TypeIdentifier { | ||||||
|  |     /// The unique internal value. | ||||||
|  |     val: u8, | ||||||
|  | } | ||||||
|  |  | ||||||
|  | impl From<u8> for TypeIdentifier { | ||||||
|  |     fn from(val: u8) -> Self { | ||||||
|  |         Self { val } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /// All data related to types and effectiveness. | ||||||
| #[derive(Debug)] | #[derive(Debug)] | ||||||
| pub struct TypeLibrary { | pub struct TypeLibrary { | ||||||
|     types: HashMap<StringKey, u8>, |     /// A list of types | ||||||
|  |     types: HashMap<StringKey, TypeIdentifier>, | ||||||
|  |     /// The effectiveness of the different types against each other. | ||||||
|     effectiveness: Vec<Vec<f32>>, |     effectiveness: Vec<Vec<f32>>, | ||||||
| } | } | ||||||
|  |  | ||||||
| impl TypeLibrary { | impl TypeLibrary { | ||||||
|  |     /// Instantiates a new type library with a specific capacity. | ||||||
|     pub fn new(capacity: usize) -> TypeLibrary { |     pub fn new(capacity: usize) -> TypeLibrary { | ||||||
|         TypeLibrary { |         TypeLibrary { | ||||||
|             types: HashMap::with_capacity(capacity), |             types: HashMap::with_capacity(capacity), | ||||||
| @@ -15,15 +34,20 @@ impl TypeLibrary { | |||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     pub fn get_type_id(&self, key: &StringKey) -> u8 { |     /// Gets the type identifier for a type with a name. | ||||||
|  |     pub fn get_type_id(&self, key: &StringKey) -> TypeIdentifier { | ||||||
|         self.types[key] |         self.types[key] | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     pub fn get_single_effectiveness(&self, attacking: u8, defending: u8) -> f32 { |     /// Gets the effectiveness for a single attacking type against a single defending type. | ||||||
|         self.effectiveness[attacking as usize][defending as usize] |     pub fn get_single_effectiveness(&self, attacking: TypeIdentifier, defending: TypeIdentifier) -> f32 { | ||||||
|  |         self.effectiveness[attacking.val as usize][defending.val as usize] | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     pub fn get_effectiveness(&self, attacking: u8, defending: &[u8]) -> f32 { |     /// Gets the effectiveness for a single attacking type against an amount of defending types. | ||||||
|  |     /// This is equivalent to running [`get_single_effectiveness`] on each defending type, and | ||||||
|  |     /// multiplying the results with each other. | ||||||
|  |     pub fn get_effectiveness(&self, attacking: TypeIdentifier, defending: &[TypeIdentifier]) -> f32 { | ||||||
|         let mut e = 1.0; |         let mut e = 1.0; | ||||||
|         for def in defending { |         for def in defending { | ||||||
|             e *= self.get_single_effectiveness(attacking, *def); |             e *= self.get_single_effectiveness(attacking, *def); | ||||||
| @@ -31,37 +55,42 @@ impl TypeLibrary { | |||||||
|         e |         e | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     pub fn register_type(&mut self, name: &StringKey) -> u8 { |     /// Registers a new type in the library. | ||||||
|         let id = self.types.len() as u8; |     pub fn register_type(&mut self, name: &StringKey) -> TypeIdentifier { | ||||||
|  |         let id = TypeIdentifier { | ||||||
|  |             val: self.types.len() as u8, | ||||||
|  |         }; | ||||||
|         self.types.insert(name.clone(), id); |         self.types.insert(name.clone(), id); | ||||||
|         self.effectiveness.resize((id + 1) as usize, vec![]); |         self.effectiveness.resize((id.val + 1) as usize, vec![]); | ||||||
|         for effectiveness in &mut self.effectiveness { |         for effectiveness in &mut self.effectiveness { | ||||||
|             effectiveness.resize((id + 1) as usize, 1.0) |             effectiveness.resize((id.val + 1) as usize, 1.0) | ||||||
|         } |         } | ||||||
|         id |         id | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     pub fn set_effectiveness(&mut self, attacking: u8, defending: u8, effectiveness: f32) { |     /// Sets the effectiveness for an attacking type against a defending type. | ||||||
|         self.effectiveness[attacking as usize][defending as usize] = effectiveness; |     pub fn set_effectiveness(&mut self, attacking: TypeIdentifier, defending: TypeIdentifier, effectiveness: f32) { | ||||||
|  |         self.effectiveness[attacking.val as usize][defending.val as usize] = effectiveness; | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
| #[cfg(test)] | #[cfg(test)] | ||||||
| pub mod tests { | pub mod tests { | ||||||
|     use crate::static_data::libraries::type_library::TypeLibrary; |  | ||||||
|     use assert_approx_eq::assert_approx_eq; |     use assert_approx_eq::assert_approx_eq; | ||||||
|  |  | ||||||
|  |     use crate::static_data::libraries::type_library::TypeLibrary; | ||||||
|  |  | ||||||
|     pub fn build() -> TypeLibrary { |     pub fn build() -> TypeLibrary { | ||||||
|         let mut lib = TypeLibrary::new(2); |         let mut lib = TypeLibrary::new(2); | ||||||
|  |  | ||||||
|         // Borrow as mut so we can insert |         // Borrow as mut so we can insert | ||||||
|         let w = &mut lib; |         let w = &mut lib; | ||||||
|         w.register_type(&"foo".into()); |         let t0 = w.register_type(&"foo".into()); | ||||||
|         w.register_type(&"bar".into()); |         let t1 = w.register_type(&"bar".into()); | ||||||
|         // Drops borrow as mut |         // Drops borrow as mut | ||||||
|  |  | ||||||
|         w.set_effectiveness(0, 1, 0.5); |         w.set_effectiveness(t0, t1, 0.5); | ||||||
|         w.set_effectiveness(1, 0, 2.0); |         w.set_effectiveness(t1, t0, 2.0); | ||||||
|  |  | ||||||
|         lib |         lib | ||||||
|     } |     } | ||||||
| @@ -72,14 +101,14 @@ pub mod tests { | |||||||
|  |  | ||||||
|         // Borrow as mut so we can insert |         // Borrow as mut so we can insert | ||||||
|         let w = &mut lib; |         let w = &mut lib; | ||||||
|         w.register_type(&"foo".into()); |         let t0 = w.register_type(&"foo".into()); | ||||||
|         w.register_type(&"bar".into()); |         let t1 = w.register_type(&"bar".into()); | ||||||
|         // Drops borrow as mut |         // Drops borrow as mut | ||||||
|  |  | ||||||
|         // Borrow as read so we can read |         // Borrow as read so we can read | ||||||
|         let r = &lib; |         let r = &lib; | ||||||
|         assert_eq!(r.get_type_id(&"foo".into()), 0); |         assert_eq!(r.get_type_id(&"foo".into()), t0); | ||||||
|         assert_eq!(r.get_type_id(&"bar".into()), 1); |         assert_eq!(r.get_type_id(&"bar".into()), t1); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     #[test] |     #[test] | ||||||
| @@ -88,16 +117,16 @@ pub mod tests { | |||||||
|  |  | ||||||
|         // Borrow as mut so we can insert |         // Borrow as mut so we can insert | ||||||
|         let w = &mut lib; |         let w = &mut lib; | ||||||
|         w.register_type(&"foo".into()); |         let t0 = w.register_type(&"foo".into()); | ||||||
|         w.register_type(&"bar".into()); |         let t1 = w.register_type(&"bar".into()); | ||||||
|         w.set_effectiveness(0, 1, 0.5); |         w.set_effectiveness(t0, t1, 0.5); | ||||||
|         w.set_effectiveness(1, 0, 2.0); |         w.set_effectiveness(t1, t0, 2.0); | ||||||
|         // Drops borrow as mut |         // Drops borrow as mut | ||||||
|  |  | ||||||
|         // Borrow as read so we can read |         // Borrow as read so we can read | ||||||
|         let r = &lib; |         let r = &lib; | ||||||
|         assert_approx_eq!(r.get_single_effectiveness(0, 1), 0.5); |         assert_approx_eq!(r.get_single_effectiveness(t0, t1), 0.5); | ||||||
|         assert_approx_eq!(r.get_single_effectiveness(1, 0), 2.0); |         assert_approx_eq!(r.get_single_effectiveness(t1, t0), 2.0); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     #[test] |     #[test] | ||||||
| @@ -106,15 +135,15 @@ pub mod tests { | |||||||
|  |  | ||||||
|         // Borrow as mut so we can insert |         // Borrow as mut so we can insert | ||||||
|         let w = &mut lib; |         let w = &mut lib; | ||||||
|         w.register_type(&"foo".into()); |         let t0 = w.register_type(&"foo".into()); | ||||||
|         w.register_type(&"bar".into()); |         let t1 = w.register_type(&"bar".into()); | ||||||
|         w.set_effectiveness(0, 1, 0.5); |         w.set_effectiveness(t0, t1, 0.5); | ||||||
|         w.set_effectiveness(1, 0, 2.0); |         w.set_effectiveness(t1, t0, 2.0); | ||||||
|         // Drops borrow as mut |         // Drops borrow as mut | ||||||
|  |  | ||||||
|         // Borrow as read so we can read |         // Borrow as read so we can read | ||||||
|         let r = &lib; |         let r = &lib; | ||||||
|         assert_approx_eq!(r.get_effectiveness(0, &[1_u8, 1_u8]), 0.25); |         assert_approx_eq!(r.get_effectiveness(t0, &[t1, t1]), 0.25); | ||||||
|         assert_approx_eq!(r.get_effectiveness(1, &[0_u8, 0_u8]), 4.0); |         assert_approx_eq!(r.get_effectiveness(t1, &[t0, t0]), 4.0); | ||||||
|     } |     } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -1,58 +1,93 @@ | |||||||
| use crate::static_data::SecondaryEffect; |  | ||||||
| use crate::StringKey; |  | ||||||
| use hashbrown::HashSet; | use hashbrown::HashSet; | ||||||
|  |  | ||||||
| #[cfg(feature = "serde")] | #[cfg(feature = "serde")] | ||||||
| use serde::{Deserialize, Serialize}; | use serde::{Deserialize, Serialize}; | ||||||
|  |  | ||||||
|  | use crate::static_data::{SecondaryEffect, TypeIdentifier}; | ||||||
|  | use crate::StringKey; | ||||||
|  |  | ||||||
|  | /// The move category defines what global kind of move this move is. | ||||||
| #[derive(Copy, Clone, PartialEq, Eq, Debug)] | #[derive(Copy, Clone, PartialEq, Eq, Debug)] | ||||||
| #[cfg_attr(feature = "serde", derive(Serialize, Deserialize))] | #[cfg_attr(feature = "serde", derive(Serialize, Deserialize))] | ||||||
| #[cfg_attr(feature = "serde", serde(rename_all = "snake_case"))] | #[cfg_attr(feature = "serde", serde(rename_all = "snake_case"))] | ||||||
| pub enum MoveCategory { | pub enum MoveCategory { | ||||||
|  |     /// A physical move uses the physical attack stats and physical defense stats to calculate damage. | ||||||
|     Physical = 0, |     Physical = 0, | ||||||
|  |     /// A special move uses the special attack stats and special defense stats to calculate damage. | ||||||
|     Special = 1, |     Special = 1, | ||||||
|  |     /// A status move does not do damage, and only runs a secondary effect. | ||||||
|     Status = 2, |     Status = 2, | ||||||
| } | } | ||||||
|  |  | ||||||
| #[derive(Copy, Clone, PartialEq, Eq, Debug)] | /// The move target defines what kind of targets the move can touch. | ||||||
|  | #[derive(Copy, Clone, PartialEq, Eq, Debug, Default)] | ||||||
| #[cfg_attr(feature = "serde", derive(Serialize, Deserialize))] | #[cfg_attr(feature = "serde", derive(Serialize, Deserialize))] | ||||||
| pub enum MoveTarget { | pub enum MoveTarget { | ||||||
|  |     /// Adjacent allows a move to target any Pokemon that is either directly to the left or right of | ||||||
|  |     /// the user, opposed to the user, or left or right of the slot that is opposing the user. | ||||||
|  |     #[default] | ||||||
|     Adjacent = 0, |     Adjacent = 0, | ||||||
|  |     /// AdjacentAlly allows a move to target any Pokemon that is directly to the left or right of | ||||||
|  |     /// the user. | ||||||
|     AdjacentAlly, |     AdjacentAlly, | ||||||
|  |     /// AdjacentAllySelf allows a move to target any Pokemon that is either directly to the left or | ||||||
|  |     /// right of the user, or the user itself. | ||||||
|     AdjacentAllySelf, |     AdjacentAllySelf, | ||||||
|  |     /// AdjacentOpponent allows a move to target any Pokemon that is either the opponent, or directly | ||||||
|  |     /// to the left or right of it. | ||||||
|     AdjacentOpponent, |     AdjacentOpponent, | ||||||
|  |  | ||||||
|  |     /// All makes the move target everything on the field. | ||||||
|     All, |     All, | ||||||
|  |     /// AllAdjacent makes the move target everything adjacent on the field. | ||||||
|     AllAdjacent, |     AllAdjacent, | ||||||
|  |     /// AllAdjacentOpponent makes the move target everything adjacent to the opponent, and the opponent. | ||||||
|     AllAdjacentOpponent, |     AllAdjacentOpponent, | ||||||
|  |     /// AllAlly targets all Pokemon on the same side as the user. | ||||||
|     AllAlly, |     AllAlly, | ||||||
|  |     /// AllOpponent targets all Pokemon on an opposing side from the user. | ||||||
|     AllOpponent, |     AllOpponent, | ||||||
|  |  | ||||||
|  |     /// Any allows a move to target a single Pokemon, in any position. | ||||||
|     Any, |     Any, | ||||||
|  |  | ||||||
|  |     /// RandomOpponent allows a move to target a single Pokemon, in a random position. | ||||||
|     RandomOpponent, |     RandomOpponent, | ||||||
|  |     /// SelfUse makes the move target the user itself. | ||||||
|     #[cfg_attr(feature = "serde", serde(rename = "Self"))] |     #[cfg_attr(feature = "serde", serde(rename = "Self"))] | ||||||
|     SelfUse, |     SelfUse, | ||||||
| } | } | ||||||
|  |  | ||||||
|  | /// A move is the skill Pokémon primarily use in battle. This is the data related to that. | ||||||
| #[derive(PartialEq, Debug)] | #[derive(PartialEq, Debug)] | ||||||
| pub struct MoveData { | pub struct MoveData { | ||||||
|  |     /// The name of the move. | ||||||
|     name: StringKey, |     name: StringKey, | ||||||
|     move_type: u8, |     /// The attacking type of the move. | ||||||
|  |     move_type: TypeIdentifier, | ||||||
|  |     /// The category of the move. | ||||||
|     category: MoveCategory, |     category: MoveCategory, | ||||||
|  |     /// The base power, not considering any modifiers, the move has. | ||||||
|     base_power: u8, |     base_power: u8, | ||||||
|  |     /// The accuracy of the move in percentage. Should be 255 for moves that always hit. | ||||||
|     accuracy: u8, |     accuracy: u8, | ||||||
|  |     /// The number of times the move can be used. This can be modified on actually learned moves using | ||||||
|  |     /// PP-Ups | ||||||
|     base_usages: u8, |     base_usages: u8, | ||||||
|  |     /// How the move handles targets. | ||||||
|     target: MoveTarget, |     target: MoveTarget, | ||||||
|  |     /// The priority of the move. A higher priority means the move should go before other moves. | ||||||
|     priority: i8, |     priority: i8, | ||||||
|  |     /// The optional secondary effect the move has. | ||||||
|     secondary_effect: Option<SecondaryEffect>, |     secondary_effect: Option<SecondaryEffect>, | ||||||
|  |     /// Arbitrary flags that can be applied to the move. | ||||||
|     flags: HashSet<StringKey>, |     flags: HashSet<StringKey>, | ||||||
| } | } | ||||||
|  |  | ||||||
| impl MoveData { | impl MoveData { | ||||||
|  |     /// Instantiates a new move. | ||||||
|     pub fn new( |     pub fn new( | ||||||
|         name: &StringKey, |         name: &StringKey, | ||||||
|         move_type: u8, |         move_type: TypeIdentifier, | ||||||
|         category: MoveCategory, |         category: MoveCategory, | ||||||
|         base_power: u8, |         base_power: u8, | ||||||
|         accuracy: u8, |         accuracy: u8, | ||||||
| @@ -75,37 +110,47 @@ impl MoveData { | |||||||
|             flags, |             flags, | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |     /// The name of the move. | ||||||
|     pub fn name(&self) -> &StringKey { |     pub fn name(&self) -> &StringKey { | ||||||
|         &self.name |         &self.name | ||||||
|     } |     } | ||||||
|     pub fn move_type(&self) -> u8 { |     /// The attacking type of the move. | ||||||
|  |     pub fn move_type(&self) -> TypeIdentifier { | ||||||
|         self.move_type |         self.move_type | ||||||
|     } |     } | ||||||
|  |     /// The category of the move. | ||||||
|     pub fn category(&self) -> MoveCategory { |     pub fn category(&self) -> MoveCategory { | ||||||
|         self.category |         self.category | ||||||
|     } |     } | ||||||
|  |     /// The base power, not considering any modifiers, the move has. | ||||||
|     pub fn base_power(&self) -> u8 { |     pub fn base_power(&self) -> u8 { | ||||||
|         self.base_power |         self.base_power | ||||||
|     } |     } | ||||||
|  |     /// The accuracy of the move in percentage. Should be 255 for moves that always hit. | ||||||
|     pub fn accuracy(&self) -> u8 { |     pub fn accuracy(&self) -> u8 { | ||||||
|         self.accuracy |         self.accuracy | ||||||
|     } |     } | ||||||
|  |     /// The number of times the move can be used. This can be modified on actually learned moves using | ||||||
|  |     /// PP-Ups | ||||||
|     pub fn base_usages(&self) -> u8 { |     pub fn base_usages(&self) -> u8 { | ||||||
|         self.base_usages |         self.base_usages | ||||||
|     } |     } | ||||||
|  |     /// How the move handles targets. | ||||||
|     pub fn target(&self) -> MoveTarget { |     pub fn target(&self) -> MoveTarget { | ||||||
|         self.target |         self.target | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /// The priority of the move. A higher priority means the move should go before other moves. | ||||||
|     pub fn priority(&self) -> i8 { |     pub fn priority(&self) -> i8 { | ||||||
|         self.priority |         self.priority | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /// The optional secondary effect the move has. | ||||||
|     pub fn secondary_effect(&self) -> &Option<SecondaryEffect> { |     pub fn secondary_effect(&self) -> &Option<SecondaryEffect> { | ||||||
|         &self.secondary_effect |         &self.secondary_effect | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /// Arbitrary flags that can be applied to the move. | ||||||
|     pub fn has_flag(&self, key: &StringKey) -> bool { |     pub fn has_flag(&self, key: &StringKey) -> bool { | ||||||
|         self.flags.contains(key) |         self.flags.contains(key) | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -1,21 +1,32 @@ | |||||||
| use crate::StringKey; | use crate::StringKey; | ||||||
|  |  | ||||||
|  | /// A parameter for an effect. This is basically a simple way to dynamically store multiple different | ||||||
|  | /// primitives on data. | ||||||
| #[derive(PartialEq, Debug)] | #[derive(PartialEq, Debug)] | ||||||
| pub enum EffectParameter { | pub enum EffectParameter { | ||||||
|  |     /// A boolean value. | ||||||
|     Bool(bool), |     Bool(bool), | ||||||
|  |     /// An integer value. Stored as a 64 bit int to deal with potentially large numbers. | ||||||
|     Int(i64), |     Int(i64), | ||||||
|  |     /// A float value. Stored as a 32 bit float. | ||||||
|     Float(f32), |     Float(f32), | ||||||
|  |     /// A string value. | ||||||
|     String(String), |     String(String), | ||||||
| } | } | ||||||
|  |  | ||||||
|  | /// A secondary effect is an effect on a move that happens after it hits. | ||||||
| #[derive(PartialEq, Debug)] | #[derive(PartialEq, Debug)] | ||||||
| pub struct SecondaryEffect { | pub struct SecondaryEffect { | ||||||
|  |     /// The chance in percentages that the effect triggers. -1 to make it always trigger. | ||||||
|     chance: f32, |     chance: f32, | ||||||
|  |     /// The name of the effect. | ||||||
|     effect_name: StringKey, |     effect_name: StringKey, | ||||||
|  |     /// A list of parameters for the effect. | ||||||
|     parameters: Vec<EffectParameter>, |     parameters: Vec<EffectParameter>, | ||||||
| } | } | ||||||
|  |  | ||||||
| impl SecondaryEffect { | impl SecondaryEffect { | ||||||
|  |     /// Instantiates a new Secondary Effect. | ||||||
|     pub fn new(chance: f32, effect_name: StringKey, parameters: Vec<EffectParameter>) -> SecondaryEffect { |     pub fn new(chance: f32, effect_name: StringKey, parameters: Vec<EffectParameter>) -> SecondaryEffect { | ||||||
|         SecondaryEffect { |         SecondaryEffect { | ||||||
|             chance, |             chance, | ||||||
| @@ -24,12 +35,15 @@ impl SecondaryEffect { | |||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /// The chance in percentages that the effect triggers. -1 to make it always trigger. | ||||||
|     pub fn chance(&self) -> f32 { |     pub fn chance(&self) -> f32 { | ||||||
|         self.chance |         self.chance | ||||||
|     } |     } | ||||||
|  |     /// The name of the effect. | ||||||
|     pub fn effect_name(&self) -> &StringKey { |     pub fn effect_name(&self) -> &StringKey { | ||||||
|         &self.effect_name |         &self.effect_name | ||||||
|     } |     } | ||||||
|  |     /// A list of parameters for the effect. | ||||||
|     pub fn parameters(&self) -> &Vec<EffectParameter> { |     pub fn parameters(&self) -> &Vec<EffectParameter> { | ||||||
|         &self.parameters |         &self.parameters | ||||||
|     } |     } | ||||||
| @@ -37,9 +51,10 @@ impl SecondaryEffect { | |||||||
|  |  | ||||||
| #[cfg(test)] | #[cfg(test)] | ||||||
| mod tests { | mod tests { | ||||||
|     use crate::static_data::moves::secondary_effect::SecondaryEffect; |  | ||||||
|     use assert_approx_eq::assert_approx_eq; |     use assert_approx_eq::assert_approx_eq; | ||||||
|  |  | ||||||
|  |     use crate::static_data::moves::secondary_effect::SecondaryEffect; | ||||||
|  |  | ||||||
|     #[test] |     #[test] | ||||||
|     fn create_secondary_effect() { |     fn create_secondary_effect() { | ||||||
|         let empty = SecondaryEffect::new(0.0, "".into(), vec![]); |         let empty = SecondaryEffect::new(0.0, "".into(), vec![]); | ||||||
|   | |||||||
| @@ -1,16 +1,24 @@ | |||||||
| use crate::static_data::Statistic; |  | ||||||
| use crate::StringKey; |  | ||||||
| use hashbrown::HashMap; | use hashbrown::HashMap; | ||||||
|  |  | ||||||
|  | use crate::static_data::Statistic; | ||||||
|  | use crate::StringKey; | ||||||
|  |  | ||||||
|  | /// A nature is an attribute on a Pokemon that modifies the effective base stats on a Pokemon. They | ||||||
|  | /// can have an increased statistic and a decreased statistic, or be neutral. | ||||||
| #[derive(Debug)] | #[derive(Debug)] | ||||||
| pub struct Nature { | pub struct Nature { | ||||||
|  |     /// The stat that should receive the increased modifier. | ||||||
|     increase_stat: Statistic, |     increase_stat: Statistic, | ||||||
|  |     /// The stat that should receive the decreased modifier. | ||||||
|     decrease_stat: Statistic, |     decrease_stat: Statistic, | ||||||
|  |     /// The amount by which the increased stat is multiplied. | ||||||
|     increase_modifier: f32, |     increase_modifier: f32, | ||||||
|  |     /// The amount by which the decreased stat is multiplied. | ||||||
|     decrease_modifier: f32, |     decrease_modifier: f32, | ||||||
| } | } | ||||||
|  |  | ||||||
| impl Nature { | impl Nature { | ||||||
|  |     /// Instantiates a new statistic. | ||||||
|     pub fn new( |     pub fn new( | ||||||
|         increase_stat: Statistic, |         increase_stat: Statistic, | ||||||
|         decrease_stat: Statistic, |         decrease_stat: Statistic, | ||||||
| @@ -25,14 +33,18 @@ impl Nature { | |||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /// The stat that should receive the increased modifier. | ||||||
|     pub fn increased_stat(&self) -> Statistic { |     pub fn increased_stat(&self) -> Statistic { | ||||||
|         self.increase_stat |         self.increase_stat | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /// The stat that should receive the decreased modifier. | ||||||
|     pub fn decreased_stat(&self) -> Statistic { |     pub fn decreased_stat(&self) -> Statistic { | ||||||
|         self.decrease_stat |         self.decrease_stat | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /// Calculates the modifier for a given stat. If it's the increased stat, returns the increased | ||||||
|  |     /// modifier, if it's the decreased stat, returns the decreased modifier. Otherwise returns 1.0 | ||||||
|     pub fn get_stat_modifier(&self, stat: Statistic) -> f32 { |     pub fn get_stat_modifier(&self, stat: Statistic) -> f32 { | ||||||
|         if stat == self.increase_stat { |         if stat == self.increase_stat { | ||||||
|             self.increase_modifier |             self.increase_modifier | ||||||
| @@ -44,26 +56,32 @@ impl Nature { | |||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
|  | /// A library of all natures that can be used, stored by their names. | ||||||
| #[derive(Debug)] | #[derive(Debug)] | ||||||
| pub struct NatureLibrary { | pub struct NatureLibrary { | ||||||
|  |     /// The underlying data structure. | ||||||
|     map: HashMap<StringKey, Nature>, |     map: HashMap<StringKey, Nature>, | ||||||
| } | } | ||||||
|  |  | ||||||
| impl NatureLibrary { | impl NatureLibrary { | ||||||
|  |     /// Creates a new nature library with a given capacity. | ||||||
|     pub fn new(capacity: usize) -> Self { |     pub fn new(capacity: usize) -> Self { | ||||||
|         NatureLibrary { |         NatureLibrary { | ||||||
|             map: HashMap::with_capacity(capacity), |             map: HashMap::with_capacity(capacity), | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /// Adds a new nature with name to the library. | ||||||
|     pub fn load_nature(&mut self, name: StringKey, nature: Nature) { |     pub fn load_nature(&mut self, name: StringKey, nature: Nature) { | ||||||
|         self.map.insert(name, nature); |         self.map.insert(name, nature); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /// Gets a nature by name. | ||||||
|     pub fn get_nature(&self, key: &StringKey) -> Option<&Nature> { |     pub fn get_nature(&self, key: &StringKey) -> Option<&Nature> { | ||||||
|         self.map.get(key) |         self.map.get(key) | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /// Finds a nature name by nature. | ||||||
|     pub fn get_nature_name(&self, nature: &Nature) -> StringKey { |     pub fn get_nature_name(&self, nature: &Nature) -> StringKey { | ||||||
|         for kv in &self.map { |         for kv in &self.map { | ||||||
|             // As natures can't be copied, and should always be the same reference as the value |             // As natures can't be copied, and should always be the same reference as the value | ||||||
|   | |||||||
| @@ -1,32 +1,46 @@ | |||||||
| use crate::static_data::AbilityIndex; | use hashbrown::HashSet; | ||||||
|  |  | ||||||
| use crate::static_data::LearnableMoves; | use crate::static_data::LearnableMoves; | ||||||
| use crate::static_data::Statistic; | use crate::static_data::Statistic; | ||||||
| use crate::static_data::{Ability, StaticStatisticSet}; | use crate::static_data::{Ability, StaticStatisticSet}; | ||||||
|  | use crate::static_data::{AbilityIndex, TypeIdentifier}; | ||||||
| use crate::Random; | use crate::Random; | ||||||
| use crate::StringKey; | use crate::StringKey; | ||||||
| use hashbrown::HashSet; |  | ||||||
|  |  | ||||||
|  | /// A form is a variant of a specific species. A species always has at least one form, but can have | ||||||
|  | /// many more. | ||||||
| #[derive(Debug)] | #[derive(Debug)] | ||||||
| pub struct Form { | pub struct Form { | ||||||
|  |     /// The name of the form. | ||||||
|     name: StringKey, |     name: StringKey, | ||||||
|  |     /// The height of the form in meters. | ||||||
|     height: f32, |     height: f32, | ||||||
|  |     /// The weight of the form in kilograms. | ||||||
|     weight: f32, |     weight: f32, | ||||||
|  |     /// The base amount of experience that is gained when beating a Pokemon with this form. | ||||||
|     base_experience: u32, |     base_experience: u32, | ||||||
|     types: Vec<u8>, |     /// The normal types a Pokemon with this form has. | ||||||
|  |     types: Vec<TypeIdentifier>, | ||||||
|  |     /// The inherent values of a form of species that are used for the stats of a Pokemon. | ||||||
|     base_stats: StaticStatisticSet<u16>, |     base_stats: StaticStatisticSet<u16>, | ||||||
|  |     /// The possible abilities a Pokemon with this form can have. | ||||||
|     abilities: Vec<StringKey>, |     abilities: Vec<StringKey>, | ||||||
|  |     /// The possible hidden abilities a Pokemon with this form can have. | ||||||
|     hidden_abilities: Vec<StringKey>, |     hidden_abilities: Vec<StringKey>, | ||||||
|  |     /// The moves a Pokemon with this form can learn. | ||||||
|     moves: LearnableMoves, |     moves: LearnableMoves, | ||||||
|  |     /// Arbitrary flags can be set on a form for scripting use. | ||||||
|     flags: HashSet<StringKey>, |     flags: HashSet<StringKey>, | ||||||
| } | } | ||||||
|  |  | ||||||
| impl Form { | impl Form { | ||||||
|  |     /// Instantiates a new form. | ||||||
|     pub fn new( |     pub fn new( | ||||||
|         name: &StringKey, |         name: &StringKey, | ||||||
|         height: f32, |         height: f32, | ||||||
|         weight: f32, |         weight: f32, | ||||||
|         base_experience: u32, |         base_experience: u32, | ||||||
|         types: Vec<u8>, |         types: Vec<TypeIdentifier>, | ||||||
|         base_stats: StaticStatisticSet<u16>, |         base_stats: StaticStatisticSet<u16>, | ||||||
|         abilities: Vec<StringKey>, |         abilities: Vec<StringKey>, | ||||||
|         hidden_abilities: Vec<StringKey>, |         hidden_abilities: Vec<StringKey>, | ||||||
| @@ -47,45 +61,58 @@ impl Form { | |||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /// The name of the form. | ||||||
|     pub fn name(&self) -> &StringKey { |     pub fn name(&self) -> &StringKey { | ||||||
|         &self.name |         &self.name | ||||||
|     } |     } | ||||||
|  |     /// The height of the form in meters. | ||||||
|     pub fn height(&self) -> f32 { |     pub fn height(&self) -> f32 { | ||||||
|         self.height |         self.height | ||||||
|     } |     } | ||||||
|  |     /// The weight of the form in kilograms. | ||||||
|     pub fn weight(&self) -> f32 { |     pub fn weight(&self) -> f32 { | ||||||
|         self.weight |         self.weight | ||||||
|     } |     } | ||||||
|  |     /// The base amount of experience that is gained when beating a Pokemon with this form. | ||||||
|     pub fn base_experience(&self) -> u32 { |     pub fn base_experience(&self) -> u32 { | ||||||
|         self.base_experience |         self.base_experience | ||||||
|     } |     } | ||||||
|     pub fn types(&self) -> &Vec<u8> { |     /// The normal types a Pokemon with this form has. | ||||||
|  |     pub fn types(&self) -> &Vec<TypeIdentifier> { | ||||||
|         &self.types |         &self.types | ||||||
|     } |     } | ||||||
|  |     /// The inherent values of a form of species that are used for the stats of a Pokemon. | ||||||
|     pub fn base_stats(&self) -> &StaticStatisticSet<u16> { |     pub fn base_stats(&self) -> &StaticStatisticSet<u16> { | ||||||
|         &self.base_stats |         &self.base_stats | ||||||
|     } |     } | ||||||
|  |     /// The possible abilities a Pokemon with this form can have. | ||||||
|     pub fn abilities(&self) -> &Vec<StringKey> { |     pub fn abilities(&self) -> &Vec<StringKey> { | ||||||
|         &self.abilities |         &self.abilities | ||||||
|     } |     } | ||||||
|  |     /// The possible hidden abilities a Pokemon with this form can have. | ||||||
|     pub fn hidden_abilities(&self) -> &Vec<StringKey> { |     pub fn hidden_abilities(&self) -> &Vec<StringKey> { | ||||||
|         &self.hidden_abilities |         &self.hidden_abilities | ||||||
|     } |     } | ||||||
|  |     /// The moves a Pokemon with this form can learn. | ||||||
|     pub fn moves(&self) -> &LearnableMoves { |     pub fn moves(&self) -> &LearnableMoves { | ||||||
|         &self.moves |         &self.moves | ||||||
|     } |     } | ||||||
|  |     /// Arbitrary flags can be set on a form for scripting use. | ||||||
|     pub fn flags(&self) -> &HashSet<StringKey> { |     pub fn flags(&self) -> &HashSet<StringKey> { | ||||||
|         &self.flags |         &self.flags | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     pub fn get_type(&self, index: usize) -> u8 { |     /// Get a type of the move at a certain index. | ||||||
|  |     pub fn get_type(&self, index: usize) -> TypeIdentifier { | ||||||
|         self.types[index] |         self.types[index] | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /// Gets a single base stat value. | ||||||
|     pub fn get_base_stat(&self, stat: Statistic) -> u16 { |     pub fn get_base_stat(&self, stat: Statistic) -> u16 { | ||||||
|         self.base_stats.get_stat(stat) |         self.base_stats.get_stat(stat) | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /// Find the index of an ability that can be on this form. | ||||||
|     pub fn find_ability_index(&self, ability: &Ability) -> Option<AbilityIndex> { |     pub fn find_ability_index(&self, ability: &Ability) -> Option<AbilityIndex> { | ||||||
|         for (index, a) in self.abilities.iter().enumerate() { |         for (index, a) in self.abilities.iter().enumerate() { | ||||||
|             if a == ability.name() { |             if a == ability.name() { | ||||||
| @@ -106,6 +133,7 @@ impl Form { | |||||||
|         None |         None | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /// Gets an ability from the form. | ||||||
|     pub fn get_ability(&self, index: AbilityIndex) -> &StringKey { |     pub fn get_ability(&self, index: AbilityIndex) -> &StringKey { | ||||||
|         if index.hidden { |         if index.hidden { | ||||||
|             &self.hidden_abilities[index.index as usize] |             &self.hidden_abilities[index.index as usize] | ||||||
| @@ -114,13 +142,16 @@ impl Form { | |||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /// Gets a random ability from the form. | ||||||
|     pub fn get_random_ability(&self, rand: &mut Random) -> &StringKey { |     pub fn get_random_ability(&self, rand: &mut Random) -> &StringKey { | ||||||
|         &self.abilities[rand.get_between_unsigned(0, self.abilities.len() as u32) as usize] |         &self.abilities[rand.get_between_unsigned(0, self.abilities.len() as u32) as usize] | ||||||
|     } |     } | ||||||
|  |     /// Gets a random hidden ability from the form. | ||||||
|     pub fn get_random_hidden_ability(&self, rand: &mut Random) -> &StringKey { |     pub fn get_random_hidden_ability(&self, rand: &mut Random) -> &StringKey { | ||||||
|         &self.hidden_abilities[rand.get_between_unsigned(0, self.hidden_abilities.len() as u32) as usize] |         &self.hidden_abilities[rand.get_between_unsigned(0, self.hidden_abilities.len() as u32) as usize] | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /// Check if the form has a specific flag set. | ||||||
|     pub fn has_flag(&self, key: &StringKey) -> bool { |     pub fn has_flag(&self, key: &StringKey) -> bool { | ||||||
|         self.flags.contains(key) |         self.flags.contains(key) | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -2,25 +2,27 @@ use rand::distributions::{Distribution, Uniform}; | |||||||
| use rand::{Rng, SeedableRng}; | use rand::{Rng, SeedableRng}; | ||||||
| use rand_pcg::Pcg32; | use rand_pcg::Pcg32; | ||||||
|  |  | ||||||
|  | /// A random number generator. | ||||||
| #[derive(Clone)] | #[derive(Clone)] | ||||||
| pub struct Random { | pub struct Random { | ||||||
|  |     /// The seed of the random number generator. | ||||||
|     seed: u128, |     seed: u128, | ||||||
|  |     /// A float distribution. | ||||||
|     distribution: Uniform<f64>, |     distribution: Uniform<f64>, | ||||||
|  |     /// The underlying RNG. PCG for fast, hard to predict random number generation. | ||||||
|     random_gen: Pcg32, |     random_gen: Pcg32, | ||||||
| } | } | ||||||
|  |  | ||||||
| impl Default for Random { | impl Default for Random { | ||||||
|  |     /// The default for the RNG uses the nanoseconds since epoch as seed. | ||||||
|     fn default() -> Self { |     fn default() -> Self { | ||||||
|         let seed = chrono::Utc::now().timestamp_nanos() as u128; |         let seed = chrono::Utc::now().timestamp_nanos() as u128; | ||||||
|         Random { |         Random::new(seed) | ||||||
|             seed, |  | ||||||
|             distribution: Uniform::from(0.0..1.0), |  | ||||||
|             random_gen: Pcg32::from_seed(seed.to_be_bytes()), |  | ||||||
|         } |  | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
| impl Random { | impl Random { | ||||||
|  |     /// Creates a new RNG with a specific seed. | ||||||
|     pub fn new(seed: u128) -> Self { |     pub fn new(seed: u128) -> Self { | ||||||
|         Random { |         Random { | ||||||
|             seed, |             seed, | ||||||
| @@ -29,41 +31,60 @@ impl Random { | |||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /// The seed used for the RNG. | ||||||
|     pub fn get_seed(&self) -> u128 { |     pub fn get_seed(&self) -> u128 { | ||||||
|         self.seed |         self.seed | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /// Get a random 32 bit integer between minimal and maximal 32 bit integer | ||||||
|     pub fn get(&mut self) -> i32 { |     pub fn get(&mut self) -> i32 { | ||||||
|         self.random_gen.gen() |         self.random_gen.gen() | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /// Get a random 32 bit signed integer between 0 and max. If max equals 0, always returns 0. | ||||||
|     pub fn get_max(&mut self, max: i32) -> i32 { |     pub fn get_max(&mut self, max: i32) -> i32 { | ||||||
|         assert!(max > 0); |         if max <= 0 { | ||||||
|  |             return 0; | ||||||
|  |         } | ||||||
|         Uniform::from(0..max).sample(&mut self.random_gen) |         Uniform::from(0..max).sample(&mut self.random_gen) | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /// Get a random 32 bit signed integer between min and max. If max is equal or less than min, | ||||||
|  |     /// always returns min. | ||||||
|     pub fn get_between(&mut self, min: i32, max: i32) -> i32 { |     pub fn get_between(&mut self, min: i32, max: i32) -> i32 { | ||||||
|         assert!(max > min); |         if max <= min { | ||||||
|  |             return min; | ||||||
|  |         } | ||||||
|         Uniform::from(min..max).sample(&mut self.random_gen) |         Uniform::from(min..max).sample(&mut self.random_gen) | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /// Get a random 32 bit integer unsigned between 0 and maximal 32 bit unsigned int. | ||||||
|     pub fn get_unsigned(&mut self) -> u32 { |     pub fn get_unsigned(&mut self) -> u32 { | ||||||
|         self.random_gen.gen() |         self.random_gen.gen() | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /// Get a random 32 bit signed integer between 0 and max. If max equals 0, always returns 0. | ||||||
|     pub fn get_max_unsigned(&mut self, max: u32) -> u32 { |     pub fn get_max_unsigned(&mut self, max: u32) -> u32 { | ||||||
|         assert!(max > 0); |         if max == 0 { | ||||||
|  |             return 0; | ||||||
|  |         } | ||||||
|         Uniform::from(0..max).sample(&mut self.random_gen) |         Uniform::from(0..max).sample(&mut self.random_gen) | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /// Get a random 32 bit unsigned integer between min and max. If max is equal or less than min, | ||||||
|  |     /// always returns min. | ||||||
|     pub fn get_between_unsigned(&mut self, min: u32, max: u32) -> u32 { |     pub fn get_between_unsigned(&mut self, min: u32, max: u32) -> u32 { | ||||||
|         assert!(max > min); |         if max <= min { | ||||||
|  |             return min; | ||||||
|  |         } | ||||||
|         Uniform::from(min..max).sample(&mut self.random_gen) |         Uniform::from(min..max).sample(&mut self.random_gen) | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /// Gets a random 32 bit float between 0.0 and 1.0 | ||||||
|     pub fn get_float(&mut self) -> f32 { |     pub fn get_float(&mut self) -> f32 { | ||||||
|         self.get_double() as f32 |         self.get_double() as f32 | ||||||
|     } |     } | ||||||
|  |     /// Gets a random 64 bit float between 0.0 and 1.0 | ||||||
|     pub fn get_double(&mut self) -> f64 { |     pub fn get_double(&mut self) -> f64 { | ||||||
|         self.distribution.sample(&mut self.random_gen) |         self.distribution.sample(&mut self.random_gen) | ||||||
|     } |     } | ||||||
| @@ -71,11 +92,13 @@ impl Random { | |||||||
|  |  | ||||||
| #[cfg(test)] | #[cfg(test)] | ||||||
| mod tests { | mod tests { | ||||||
|     use crate::utils::random::Random; |  | ||||||
|     extern crate test; |  | ||||||
|     use std::hint::black_box; |     use std::hint::black_box; | ||||||
|     use test::Bencher; |     use test::Bencher; | ||||||
|  |  | ||||||
|  |     use crate::utils::random::Random; | ||||||
|  |  | ||||||
|  |     extern crate test; | ||||||
|  |  | ||||||
|     #[test] |     #[test] | ||||||
|     #[cfg_attr(miri, ignore)] |     #[cfg_attr(miri, ignore)] | ||||||
|     fn create_random() { |     fn create_random() { | ||||||
|   | |||||||
| @@ -1,23 +1,30 @@ | |||||||
| use hashbrown::HashMap; |  | ||||||
| use std::fmt::{Display, Formatter}; | use std::fmt::{Display, Formatter}; | ||||||
| use std::hash::{Hash, Hasher}; | use std::hash::{Hash, Hasher}; | ||||||
| use std::lazy::SyncLazy; | use std::lazy::SyncLazy; | ||||||
| use std::sync::{Arc, Mutex, Weak}; | use std::sync::{Arc, Mutex, Weak}; | ||||||
|  |  | ||||||
|  | use hashbrown::HashMap; | ||||||
|  |  | ||||||
| /// StringKey is an immutable string that is used for indexing of hashmaps or equality a lot. | /// StringKey is an immutable string that is used for indexing of hashmaps or equality a lot. | ||||||
| /// By reference counting the string instead of copying, and caching the hash, we can get some | /// By reference counting the string instead of copying, and caching the hash, we can get some | ||||||
| /// free speed out of it. Note that StringKeys also compare case insensitive, so that for example | /// free speed out of it. Note that StringKeys also compare case insensitive, so that for example | ||||||
| /// `charmander` == `Charmander`. | /// `charmander` == `Charmander`. | ||||||
| #[derive(Clone, Debug)] | #[derive(Clone, Debug)] | ||||||
| pub struct StringKey { | pub struct StringKey { | ||||||
|  |     /// The underlying reference counted string. | ||||||
|     str: Arc<str>, |     str: Arc<str>, | ||||||
|  |     /// The unique hash of the string. | ||||||
|     hash: u32, |     hash: u32, | ||||||
| } | } | ||||||
|  |  | ||||||
|  | /// A cache of all allocated strings. This allows us to re-use strings that are often used without | ||||||
|  | /// allocation. | ||||||
| static STRING_CACHE: SyncLazy<Mutex<HashMap<u32, Weak<str>>>> = SyncLazy::new(|| Mutex::new(HashMap::new())); | static STRING_CACHE: SyncLazy<Mutex<HashMap<u32, Weak<str>>>> = SyncLazy::new(|| Mutex::new(HashMap::new())); | ||||||
|  | /// An empty StringKey | ||||||
| static EMPTY: SyncLazy<StringKey> = SyncLazy::new(|| StringKey::new("")); | static EMPTY: SyncLazy<StringKey> = SyncLazy::new(|| StringKey::new("")); | ||||||
|  |  | ||||||
| impl StringKey { | impl StringKey { | ||||||
|  |     /// Calculates the hash of a string key in a const manner. | ||||||
|     pub const fn get_hash_const<const N: usize>(s: &[u8; N]) -> u32 { |     pub const fn get_hash_const<const N: usize>(s: &[u8; N]) -> u32 { | ||||||
|         let mut crc: u32 = 0xffffffff; |         let mut crc: u32 = 0xffffffff; | ||||||
|  |  | ||||||
| @@ -29,6 +36,7 @@ impl StringKey { | |||||||
|         crc ^ 0xffffffff |         crc ^ 0xffffffff | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /// Gets the hash of a string. | ||||||
|     pub fn get_hash(s: &str) -> u32 { |     pub fn get_hash(s: &str) -> u32 { | ||||||
|         let mut crc: u32 = 0xffffffff; |         let mut crc: u32 = 0xffffffff; | ||||||
|         for byte in s.bytes() { |         for byte in s.bytes() { | ||||||
| @@ -37,6 +45,8 @@ impl StringKey { | |||||||
|         crc ^ 0xffffffff |         crc ^ 0xffffffff | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /// Creates a new StringKey. If we can find a value for this StringKey in the cache, we re-use | ||||||
|  |     /// that value. | ||||||
|     pub fn new(s: &str) -> Self { |     pub fn new(s: &str) -> Self { | ||||||
|         let hash = StringKey::get_hash(s); |         let hash = StringKey::get_hash(s); | ||||||
|         let mut cache = STRING_CACHE.lock().unwrap(); |         let mut cache = STRING_CACHE.lock().unwrap(); | ||||||
| @@ -54,14 +64,17 @@ impl StringKey { | |||||||
|         v |         v | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /// Gets the empty StringKey. | ||||||
|     pub fn empty() -> Self { |     pub fn empty() -> Self { | ||||||
|         EMPTY.clone() |         EMPTY.clone() | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /// Gets the underlying string for the StringKey. | ||||||
|     pub fn str(&self) -> &str { |     pub fn str(&self) -> &str { | ||||||
|         &self.str |         &self.str | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /// Gets the hash of the string value. | ||||||
|     pub fn hash(&self) -> u32 { |     pub fn hash(&self) -> u32 { | ||||||
|         self.hash |         self.hash | ||||||
|     } |     } | ||||||
| @@ -93,6 +106,7 @@ impl Display for StringKey { | |||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
|  | /// Converts a character to lowercased in a const safe way. | ||||||
| const fn to_lower(c: u8) -> u8 { | const fn to_lower(c: u8) -> u8 { | ||||||
|     if c >= b'A' && c <= b'Z' { |     if c >= b'A' && c <= b'Z' { | ||||||
|         return c + (b'a' - b'A'); |         return c + (b'a' - b'A'); | ||||||
| @@ -100,6 +114,7 @@ const fn to_lower(c: u8) -> u8 { | |||||||
|     c |     c | ||||||
| } | } | ||||||
|  |  | ||||||
|  | /// A lookup table for use in CRC32 hash. | ||||||
| const CRC_TABLE: &[u32] = &[ | const CRC_TABLE: &[u32] = &[ | ||||||
|     0, 0x77073096, 0xEE0E612C, 0x990951BA, 0x076DC419, 0x706AF48F, 0xE963A535, 0x9E6495A3, 0x0EDB8832, 0x79DCB8A4, |     0, 0x77073096, 0xEE0E612C, 0x990951BA, 0x076DC419, 0x706AF48F, 0xE963A535, 0x9E6495A3, 0x0EDB8832, 0x79DCB8A4, | ||||||
|     0xE0D5E91E, 0x97D2D988, 0x09B64C2B, 0x7EB17CBD, 0xE7B82D07, 0x90BF1D91, 0x1DB71064, 0x6AB020F2, 0xF3B97148, |     0xE0D5E91E, 0x97D2D988, 0x09B64C2B, 0x7EB17CBD, 0xE7B82D07, 0x90BF1D91, 0x1DB71064, 0x6AB020F2, 0xF3B97148, | ||||||
|   | |||||||
| @@ -59,7 +59,7 @@ pub fn load_types(path: &String, type_library: &mut TypeLibrary) { | |||||||
|  |  | ||||||
|         for (i, v) in record.iter().skip(1).enumerate() { |         for (i, v) in record.iter().skip(1).enumerate() { | ||||||
|             let effectiveness = v.parse::<f32>().unwrap(); |             let effectiveness = v.parse::<f32>().unwrap(); | ||||||
|             type_library.set_effectiveness(offensive_type_id, i as u8, effectiveness); |             type_library.set_effectiveness(offensive_type_id, (i as u8).into(), effectiveness); | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| } | } | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user