Adds attract

This commit is contained in:
Deukhoofd 2022-09-10 10:29:00 +02:00
parent 4275816fd9
commit 19d2705221
Signed by: Deukhoofd
GPG Key ID: F63E044490819F6F
8 changed files with 131 additions and 3 deletions

View File

@ -13,6 +13,7 @@ use pkmn_lib_interface::set_load_script_fn;
pub mod registered_scripts;
pub mod moves;
pub mod pokemon;
pub mod util_scripts;
pub(crate) mod utils;

View File

@ -0,0 +1,40 @@
use crate::pokemon::infatuated::Infatuated;
use crate::script;
use core::any::Any;
use pkmn_lib_interface::app_interface::{ExecutingMove, Gender, Pokemon};
use pkmn_lib_interface::handling::{Script, ScriptCapabilities};
script!(Attract, "attract");
impl Script for Attract {
fn new() -> Self {
Self {}
}
fn get_name(&self) -> &'static str {
Self::get_const_name()
}
fn get_capabilities(&self) -> &[ScriptCapabilities] {
&[ScriptCapabilities::OnSecondaryEffect]
}
fn on_secondary_effect(&self, mv: ExecutingMove, target: Pokemon, hit: u8) {
let user_gender = mv.user().gender();
let target_gender = target.gender();
// If the move is used on a Pokémon that is the same gender as the user, it will fail
if target_gender == user_gender {
return mv.get_hit_data(&target, hit).fail();
}
// It will also fail if used by or on a gender-unknown Pokémon
if user_gender == Gender::Genderless || target_gender == Gender::Genderless {
return mv.get_hit_data(&target, hit).fail();
}
// If the target is the opposite gender of the Pokémon who launched the move, the target becomes infatuated
target.add_volatile_by_name(Infatuated::get_const_name());
}
fn as_any(&self) -> &dyn Any {
self
}
}

View File

@ -1,6 +1,7 @@
pub mod acrobatics;
pub mod acupressure;
pub mod after_you;
pub mod multi_hit_move;
pub mod assist;
pub mod assurance;
pub mod attract;
pub mod multi_hit_move;

View File

@ -0,0 +1,30 @@
use crate::script;
use core::any::Any;
use pkmn_lib_interface::app_interface::ExecutingMove;
use pkmn_lib_interface::handling::{Script, ScriptCapabilities};
script!(Infatuated, "infatuated");
impl Script for Infatuated {
fn new() -> Self {
Self {}
}
fn get_name(&self) -> &'static str {
Self::get_const_name()
}
fn get_capabilities(&self) -> &[ScriptCapabilities] {
&[ScriptCapabilities::PreventMove]
}
fn prevent_move(&self, mv: ExecutingMove, prevent: &mut bool) {
if mv.user().battle().unwrap().random().get_max(2) == 0 {
*prevent = true
}
}
fn as_any(&self) -> &dyn Any {
self
}
}

View File

@ -0,0 +1 @@
pub mod infatuated;

View File

@ -1,4 +1,5 @@
use crate::moves::*;
use crate::pokemon::*;
use alloc::boxed::Box;
use pkmn_lib_interface::app_interface::{get_hash, StringKey};
use pkmn_lib_interface::handling::{Script, ScriptCategory};
@ -32,11 +33,14 @@ pub fn get_script(category: ScriptCategory, name: &StringKey) -> Option<Box<dyn
assist::Assist,
assurance::Assurance,
multi_hit_move::MultiHitMove,
attract::Attract,
);
}
ScriptCategory::Ability => {}
ScriptCategory::Status => {}
ScriptCategory::Pokemon => {}
ScriptCategory::Pokemon => {
resolve_match!(name.hash(), infatuated::Infatuated,)
}
ScriptCategory::Battle => {
resolve_match!(name.hash(), crate::util_scripts::ForceEffectTriggerScript,)
}

View File

@ -7,7 +7,8 @@ use crate::handling::cached_value::CachedValue;
use crate::handling::Cacheable;
use crate::{
cached_value, cached_value_getters, wasm_optional_reference_getters, wasm_reference_getters,
wasm_value_getters, DynamicLibrary, ExternRef, ExternalReferenceType, Script, TypeIdentifier,
wasm_value_getters, DynamicLibrary, ExternRef, ExternalReferenceType, Script, ScriptPtr,
TypeIdentifier,
};
use alloc::boxed::Box;
use alloc::rc::Rc;
@ -178,6 +179,49 @@ impl Pokemon {
.get(self.battle_side_index() as u32)
.unwrap()
}
#[cfg(not(feature = "mock_data"))]
pub fn add_volatile(&self, script: Box<dyn Script>) -> &dyn Script {
unsafe {
pokemon_add_volatile(self.inner.reference, ScriptPtr::new(script))
.val()
.unwrap()
}
}
#[cfg(not(feature = "mock_data"))]
pub fn add_volatile_by_name(&self, script_name: &str) -> &dyn Script {
unsafe {
let ptr = CString::new(script_name).unwrap();
pokemon_add_volatile_by_name(self.inner.reference, ptr.into_raw())
.val()
.unwrap()
}
}
#[cfg(not(feature = "mock_data"))]
pub fn remove_volatile(&self, script: &dyn Script) {
unsafe {
let name = CString::new(script.get_name()).unwrap();
pokemon_remove_volatile(self.inner.reference, name.into_raw());
}
}
#[cfg(not(feature = "mock_data"))]
pub fn get_volatile<T>(&self, script_name: &str) -> Option<&T>
where
T: Script + 'static,
{
unsafe {
let script_name = CString::new(script_name).unwrap();
let s = pokemon_get_volatile(self.inner.reference, script_name.into_raw()).val();
if let Some(s) = s {
Some(s.as_any().downcast_ref().unwrap())
} else {
None
}
}
}
}
#[cfg(not(feature = "mock_data"))]
@ -275,6 +319,12 @@ extern "wasm" {
fn pokemon_change_form(r: ExternRef<Pokemon>, form: ExternRef<Form>);
fn pokemon_damage(r: ExternRef<Pokemon>, damage: u32, source: DamageSource);
fn pokemon_heal(r: ExternRef<Pokemon>, amount: u32, allow_revive: bool) -> bool;
fn pokemon_add_volatile_by_name(r: ExternRef<Pokemon>, name: *const c_char) -> ScriptPtr;
fn pokemon_add_volatile(r: ExternRef<Pokemon>, script: ScriptPtr) -> ScriptPtr;
fn pokemon_has_volatile(r: ExternRef<Pokemon>, name: *const c_char) -> bool;
fn pokemon_remove_volatile(r: ExternRef<Pokemon>, name: *const c_char);
fn pokemon_get_volatile(r: ExternRef<Pokemon>, name: *const c_char) -> ScriptPtr;
}
#[cfg(feature = "mock_data")]

View File

@ -10,6 +10,7 @@ use alloc::vec::Vec;
use spin::RwLock;
#[repr(u8)]
#[derive(Eq, PartialEq)]
pub enum Gender {
Male = 0,
Female = 1,