pokerogue/src/field/pokemon.ts

2703 lines
99 KiB
TypeScript
Raw Normal View History

2023-03-28 14:54:52 -04:00
import Phaser from 'phaser';
2024-02-29 20:08:50 -05:00
import BattleScene, { AnySound } from '../battle-scene';
import BattleInfo, { PlayerBattleInfo, EnemyBattleInfo } from '../ui/battle-info';
import { Moves } from "../data/enums/moves";
2024-03-11 18:18:55 -04:00
import Move, { HighCritAttr, HitsTagAttr, applyMoveAttrs, FixedDamageAttr, VariableAtkAttr, VariablePowerAttr, allMoves, MoveCategory, TypelessAttr, CritOnlyAttr, getMoveTargets, OneHitKOAttr, MultiHitAttr, StatusMoveTypeImmunityAttr, MoveTarget, VariableDefAttr } from "../data/move";
2024-02-29 20:08:50 -05:00
import { default as PokemonSpecies, PokemonSpeciesForm, SpeciesFormKey, getFusedSpeciesName, getPokemonSpecies } from '../data/pokemon-species';
import * as Utils from '../utils';
import { Type, TypeDamageMultiplier, getTypeDamageMultiplier, getTypeRgb } from '../data/type';
import { getLevelTotalExp } from '../data/exp';
import { Stat } from '../data/pokemon-stat';
import { AttackTypeBoosterModifier, DamageMoneyRewardModifier, EnemyDamageBoosterModifier, EnemyDamageReducerModifier, EnemyFusionChanceModifier, HiddenAbilityRateBoosterModifier, PokemonBaseStatModifier, PokemonHeldItemModifier, PokemonNatureWeightModifier, ShinyRateBoosterModifier, SurviveDamageModifier, TempBattleStatBoosterModifier, TerastallizeModifier } from '../modifier/modifier';
2024-02-29 20:08:50 -05:00
import { PokeballType } from '../data/pokeball';
import { Gender } from '../data/gender';
import { initMoveAnim, loadMoveAnimAssets } from '../data/battle-anims';
import { Status, StatusEffect } from '../data/status-effect';
import { pokemonEvolutions, pokemonPrevolutions, SpeciesFormEvolution, SpeciesEvolutionCondition } from '../data/pokemon-evolutions';
import { reverseCompatibleTms, tmSpecies } from '../data/tms';
import { DamagePhase, FaintPhase, LearnMovePhase, ObtainStatusEffectPhase, StatChangePhase, SwitchSummonPhase } from '../phases';
2024-02-29 20:08:50 -05:00
import { BattleStat } from '../data/battle-stat';
import { BattlerTag, BattlerTagLapseType, EncoreTag, TypeBoostTag, getBattlerTag } from '../data/battler-tags';
import { BattlerTagType } from "../data/enums/battler-tag-type";
import { Species } from '../data/enums/species';
import { WeatherType } from '../data/weather';
import { TempBattleStat } from '../data/temp-battle-stat';
import { WeakenMoveTypeTag } from '../data/arena-tag';
import { ArenaTagType } from "../data/enums/arena-tag-type";
import { Biome } from "../data/enums/biome";
import { Abilities, Ability, BattleStatMultiplierAbAttr, BlockCritAbAttr, BypassBurnDamageReductionAbAttr, FieldVariableMovePowerAbAttr, IgnoreOpponentStatChangesAbAttr, MoveAbilityBypassAbAttr, MoveImmunityAbAttr, NonSuperEffectiveImmunityAbAttr, PreApplyBattlerTagAbAttr, PreDefendEndureAbAttr, ReceivedMoveDamageMultiplierAbAttr, ReduceStatusEffectDurationAbAttr, StabBoostAbAttr, StatusEffectImmunityAbAttr, TypeImmunityAbAttr, VariableMovePowerAbAttr, WeightMultiplierAbAttr, allAbilities, applyAbAttrs, applyBattleStatMultiplierAbAttrs, applyPostDefendAbAttrs, applyPreApplyBattlerTagAbAttrs, applyPreAttackAbAttrs, applyPreDefendAbAttrs, applyPreSetStatusAbAttrs } from '../data/ability';
2024-02-29 20:08:50 -05:00
import PokemonData from '../system/pokemon-data';
import { BattlerIndex } from '../battle';
import { BattleSpec } from "../enums/battle-spec";
import { Mode } from '../ui/ui';
import PartyUiHandler, { PartyOption, PartyUiMode } from '../ui/party-ui-handler';
2023-11-04 00:32:12 -04:00
import SoundFade from 'phaser3-rex-plugins/plugins/soundfade';
2024-02-29 20:08:50 -05:00
import { LevelMoves } from '../data/pokemon-level-moves';
import { DamageAchv, achvs } from '../system/achv';
import { DexAttr, StarterMoveset } from '../system/game-data';
2023-11-23 23:52:13 -05:00
import { QuantizerCelebi, argbFromRgba, rgbaFromArgb } from '@material/material-color-utilities';
2024-02-29 20:08:50 -05:00
import { Nature, getNatureStatMultiplier } from '../data/nature';
import { SpeciesFormChange, SpeciesFormChangeActiveTrigger, SpeciesFormChangeMoveLearnedTrigger, SpeciesFormChangeMoveUsedTrigger, SpeciesFormChangeStatusEffectTrigger } from '../data/pokemon-forms';
import { TerrainType } from '../data/terrain';
import { TrainerSlot } from '../data/trainer-config';
export enum FieldPosition {
CENTER,
LEFT,
RIGHT
}
2023-03-28 14:54:52 -04:00
2023-07-11 11:32:56 -04:00
const ABILITY_OVERRIDE = Abilities.NONE;
const MOVE_OVERRIDE = Moves.NONE;
2023-07-11 11:32:56 -04:00
const OPP_ABILITY_OVERRIDE = Abilities.NONE;
const OPP_MOVE_OVERRIDE = Moves.NONE;
2023-03-28 14:54:52 -04:00
export default abstract class Pokemon extends Phaser.GameObjects.Container {
public id: integer;
public name: string;
public species: PokemonSpecies;
public formIndex: integer;
public abilityIndex: integer;
2023-03-28 14:54:52 -04:00
public shiny: boolean;
public pokeball: PokeballType;
protected battleInfo: BattleInfo;
public level: integer;
public exp: integer;
public levelExp: integer;
2023-04-01 20:06:44 -04:00
public gender: Gender;
2023-03-28 14:54:52 -04:00
public hp: integer;
public stats: integer[];
public ivs: integer[];
2024-01-05 22:24:05 -05:00
public nature: Nature;
2023-03-28 14:54:52 -04:00
public moveset: PokemonMove[];
2023-04-11 19:08:03 -04:00
public status: Status;
public friendship: integer;
2024-01-05 11:29:34 -05:00
public metLevel: integer;
public metBiome: Biome | -1;
public pauseEvolutions: boolean;
public pokerus: boolean;
2023-03-28 14:54:52 -04:00
2023-11-04 00:32:12 -04:00
public fusionSpecies: PokemonSpecies;
public fusionFormIndex: integer;
public fusionAbilityIndex: integer;
public fusionShiny: boolean;
public fusionGender: Gender;
private summonDataPrimer: PokemonSummonData;
2023-04-03 23:38:31 -04:00
public summonData: PokemonSummonData;
2023-11-28 21:35:52 -05:00
public battleData: PokemonBattleData;
2023-04-03 23:38:31 -04:00
public battleSummonData: PokemonBattleSummonData;
public turnData: PokemonTurnData;
public fieldPosition: FieldPosition;
2023-04-10 23:15:06 -04:00
public maskEnabled: boolean;
public maskSprite: Phaser.GameObjects.Sprite;
2023-03-28 14:54:52 -04:00
private shinySparkle: Phaser.GameObjects.Sprite;
2024-01-05 22:24:05 -05:00
constructor(scene: BattleScene, x: number, y: number, species: PokemonSpecies, level: integer, abilityIndex?: integer, formIndex?: integer, gender?: Gender, shiny?: boolean, ivs?: integer[], nature?: Nature, dataSource?: Pokemon | PokemonData) {
2023-03-28 14:54:52 -04:00
super(scene, x, y);
2023-04-26 16:07:29 -04:00
if (!species.isObtainable() && this.isPlayer())
2023-12-07 17:43:56 -05:00
throw `Cannot create a player Pokemon for species '${species.getName(formIndex)}'`;
2023-04-26 16:07:29 -04:00
const hiddenAbilityChance = new Utils.IntegerHolder(256);
if (!this.hasTrainer())
this.scene.applyModifiers(HiddenAbilityRateBoosterModifier, true, hiddenAbilityChance);
const hasHiddenAbility = !Utils.randSeedInt(hiddenAbilityChance.value);
const randAbilityIndex = Utils.randSeedInt(2);
2023-03-28 14:54:52 -04:00
this.species = species;
this.pokeball = dataSource?.pokeball || PokeballType.POKEBALL;
this.level = level;
this.abilityIndex = abilityIndex !== undefined
? abilityIndex
: (species.abilityHidden && hasHiddenAbility ? species.ability2 ? 2 : 1 : species.ability2 ? randAbilityIndex : 0);
if (formIndex !== undefined)
this.formIndex = formIndex;
if (gender !== undefined)
this.gender = gender;
if (shiny !== undefined)
this.shiny = shiny;
2023-03-28 14:54:52 -04:00
this.exp = dataSource?.exp || getLevelTotalExp(this.level, species.growthRate);
this.levelExp = dataSource?.levelExp || 0;
if (dataSource) {
this.id = dataSource.id;
this.hp = dataSource.hp;
this.stats = dataSource.stats;
this.ivs = dataSource.ivs;
2024-01-05 22:24:05 -05:00
this.nature = dataSource.nature || 0 as Nature;
2023-03-28 14:54:52 -04:00
this.moveset = dataSource.moveset;
2023-04-03 23:38:31 -04:00
this.status = dataSource.status;
this.friendship = dataSource.friendship !== undefined ? dataSource.friendship : this.species.baseFriendship;
2024-01-05 11:29:34 -05:00
this.metLevel = dataSource.metLevel || 5;
this.metBiome = dataSource.metBiome;
this.pauseEvolutions = dataSource.pauseEvolutions;
this.pokerus = !!dataSource.pokerus;
2023-11-04 00:32:12 -04:00
this.fusionSpecies = dataSource.fusionSpecies instanceof PokemonSpecies ? dataSource.fusionSpecies : getPokemonSpecies(dataSource.fusionSpecies);
this.fusionFormIndex = dataSource.fusionFormIndex;
this.fusionAbilityIndex = dataSource.fusionAbilityIndex;
this.fusionShiny = dataSource.fusionShiny;
this.fusionGender = dataSource.fusionGender;
2023-03-28 14:54:52 -04:00
} else {
this.id = Utils.randSeedInt(4294967296);
2023-11-12 23:47:04 -05:00
this.ivs = ivs || [
2023-03-28 14:54:52 -04:00
Utils.binToDec(Utils.decToBin(this.id).substring(0, 5)),
Utils.binToDec(Utils.decToBin(this.id).substring(5, 10)),
Utils.binToDec(Utils.decToBin(this.id).substring(10, 15)),
Utils.binToDec(Utils.decToBin(this.id).substring(15, 20)),
Utils.binToDec(Utils.decToBin(this.id).substring(20, 25)),
Utils.binToDec(Utils.decToBin(this.id).substring(25, 30))
];
if (this.gender === undefined) {
if (this.species.malePercent === null)
this.gender = Gender.GENDERLESS;
else {
const genderChance = (this.id % 256) * 0.390625;
if (genderChance < this.species.malePercent)
this.gender = Gender.MALE;
else
this.gender = Gender.FEMALE;
}
2023-03-28 14:54:52 -04:00
}
if (this.formIndex === undefined)
2024-01-09 23:34:43 -05:00
this.formIndex = this.scene.getSpeciesFormIndex(species, this.gender, this.nature, this.isPlayer());
if (this.shiny === undefined)
this.trySetShiny();
2023-03-28 14:54:52 -04:00
2024-01-09 23:34:43 -05:00
if (nature !== undefined)
this.setNature(nature);
else
this.generateNature();
this.friendship = species.baseFriendship;
2024-01-05 11:29:34 -05:00
this.metLevel = level;
this.metBiome = scene.currentBattle ? scene.arena.biomeType : -1;
this.pokerus = false;
2024-03-14 16:26:57 -04:00
const fused = new Utils.BooleanHolder(scene.gameMode.isSplicedOnly);
if (!fused.value && !this.isPlayer() && !this.hasTrainer())
this.scene.applyModifier(EnemyFusionChanceModifier, false, fused);
if (fused.value) {
2023-12-31 11:15:57 -05:00
this.calculateStats();
2023-11-08 18:36:30 -05:00
this.generateFusionSpecies();
2023-12-31 11:15:57 -05:00
}
this.generateAndPopulateMoveset();
2023-03-28 14:54:52 -04:00
}
this.generateName();
2023-04-26 16:07:29 -04:00
if (!species.isObtainable())
this.shiny = false;
2023-03-28 14:54:52 -04:00
this.calculateStats();
2024-01-07 23:17:24 -05:00
}
2023-03-28 14:54:52 -04:00
2024-01-07 23:17:24 -05:00
init(): void {
this.fieldPosition = FieldPosition.CENTER;
2024-01-07 23:17:24 -05:00
this.initBattleInfo();
this.scene.fieldUI.addAt(this.battleInfo, 0);
2023-03-28 14:54:52 -04:00
2023-06-04 21:47:43 -04:00
const getSprite = (hasShadow?: boolean) => {
const ret = this.scene.addPokemonSprite(this, 0, 0, `pkmn__${this.isPlayer() ? 'back__' : ''}sub`, undefined, true);
2023-03-28 14:54:52 -04:00
ret.setOrigin(0.5, 1);
ret.setPipeline(this.scene.spritePipeline, { tone: [ 0.0, 0.0, 0.0, 0.0 ], hasShadow: !!hasShadow, teraColor: getTypeRgb(this.getTeraType()) });
2023-03-28 14:54:52 -04:00
return ret;
};
this.setScale(this.getSpriteScale());
2023-03-28 14:54:52 -04:00
2023-06-04 21:47:43 -04:00
const sprite = getSprite(true);
2023-03-28 14:54:52 -04:00
const tintSprite = getSprite();
tintSprite.setVisible(false);
this.addAt(sprite, 0);
this.addAt(tintSprite, 1);
2023-03-28 14:54:52 -04:00
if (this.isShiny() && !this.shinySparkle)
this.initShinySparkle();
2023-03-28 14:54:52 -04:00
}
2024-01-07 23:17:24 -05:00
abstract initBattleInfo(): void;
2023-06-06 10:14:53 -04:00
isOnField(): boolean {
2023-10-31 14:09:33 -04:00
if (!this.scene)
return false;
2023-06-06 10:14:53 -04:00
return this.scene.field.getIndex(this) > -1;
}
isFainted(checkStatus?: boolean): boolean {
return !this.hp && (!checkStatus || this.status?.effect === StatusEffect.FAINT);
}
isActive(onField?: boolean): boolean {
2023-10-31 14:09:33 -04:00
if (!this.scene)
return false;
2023-06-06 10:14:53 -04:00
return !this.isFainted() && !!this.scene && (!onField || this.isOnField());
}
getDexAttr(): bigint {
2023-11-12 23:47:04 -05:00
let ret = 0n;
ret |= this.gender !== Gender.FEMALE ? DexAttr.MALE : DexAttr.FEMALE;
ret |= !this.shiny ? DexAttr.NON_SHINY : DexAttr.SHINY;
ret |= !this.abilityIndex ? DexAttr.ABILITY_1 : this.species.ability2 && this.abilityIndex === 1 ? DexAttr.ABILITY_2 : DexAttr.ABILITY_HIDDEN;
ret |= this.scene.gameData.getFormAttr(this.formIndex);
return ret;
}
generateName(): void {
if (!this.fusionSpecies) {
2023-12-07 17:43:56 -05:00
this.name = this.species.getName(this.formIndex);
return;
}
2023-12-07 17:43:56 -05:00
this.name = getFusedSpeciesName(this.species.getName(this.formIndex), this.fusionSpecies.getName(this.fusionFormIndex));
if (this.battleInfo)
this.updateInfo(true);
}
2023-03-28 14:54:52 -04:00
abstract isPlayer(): boolean;
abstract hasTrainer(): boolean;
abstract getFieldIndex(): integer;
abstract getBattlerIndex(): BattlerIndex;
2023-03-29 00:31:25 -04:00
loadAssets(): Promise<void> {
return new Promise(resolve => {
const moveIds = this.getMoveset().map(m => m.getMove().id);
2023-04-11 19:08:03 -04:00
Promise.allSettled(moveIds.map(m => initMoveAnim(m)))
2023-04-03 20:47:41 -04:00
.then(() => {
2023-04-14 18:21:33 -04:00
loadMoveAnimAssets(this.scene, moveIds);
this.getSpeciesForm().loadAssets(this.scene, this.getGender() === Gender.FEMALE, this.formIndex, this.shiny);
if (this.isPlayer() || this.getFusionSpeciesForm())
2024-02-24 21:16:19 -05:00
this.scene.loadPokemonAtlas(this.getBattleSpriteKey(true, true), this.getBattleSpriteAtlasPath(true, true));
if (this.getFusionSpeciesForm()) {
this.getFusionSpeciesForm().loadAssets(this.scene, this.getFusionGender() === Gender.FEMALE, this.fusionFormIndex, this.fusionShiny);
2024-02-24 21:16:19 -05:00
this.scene.loadPokemonAtlas(this.getFusionBattleSpriteKey(true, true), this.getFusionBattleSpriteAtlasPath(true, true));
}
2023-04-03 20:47:41 -04:00
this.scene.load.once(Phaser.Loader.Events.COMPLETE, () => {
if (this.isPlayer()) {
2023-04-12 19:09:15 -04:00
const originalWarn = console.warn;
// Ignore warnings for missing frames, because there will be a lot
console.warn = () => {};
const battleFrameNames = this.scene.anims.generateFrameNames(this.getBattleSpriteKey(), { zeroPad: 4, suffix: ".png", start: 1, end: 400 });
2023-04-12 19:09:15 -04:00
console.warn = originalWarn;
this.scene.anims.create({
key: this.getBattleSpriteKey(),
frames: battleFrameNames,
frameRate: 12,
repeat: -1
});
}
2023-04-04 21:10:11 -04:00
this.playAnim();
this.updateFusionPalette();
if (this.summonData?.speciesForm)
this.updateFusionPalette(true);
2023-04-03 20:47:41 -04:00
resolve();
});
if (!this.scene.load.isLoading())
this.scene.load.start();
2023-03-29 00:31:25 -04:00
});
});
}
2023-12-07 17:43:56 -05:00
getFormKey(): string {
if (!this.species.forms.length || this.species.forms.length <= this.formIndex)
return '';
return this.species.forms[this.formIndex].formKey;
}
getFusionFormKey(): string {
if (!this.fusionSpecies)
return null;
if (!this.fusionSpecies.forms.length || this.fusionSpecies.forms.length <= this.fusionFormIndex)
return '';
return this.fusionSpecies.forms[this.fusionFormIndex].formKey;
}
getSpriteAtlasPath(ignoreOverride?: boolean): string {
return this.getSpriteId(ignoreOverride).replace(/\_{2}/g, '/');
2023-03-28 14:54:52 -04:00
}
getBattleSpriteAtlasPath(back?: boolean, ignoreOverride?: boolean): string {
return this.getBattleSpriteId(back, ignoreOverride).replace(/\_{2}/g, '/');
}
getSpriteId(ignoreOverride?: boolean): string {
return this.getSpeciesForm(ignoreOverride).getSpriteId(this.getGender(ignoreOverride) === Gender.FEMALE, this.formIndex, this.shiny);
}
getBattleSpriteId(back?: boolean, ignoreOverride?: boolean): string {
if (back === undefined)
back = this.isPlayer();
return `${back ? 'back__' : ''}${this.getSpriteId(ignoreOverride)}`;
2023-03-28 14:54:52 -04:00
}
getSpriteKey(ignoreOverride?: boolean): string {
return this.getSpeciesForm(ignoreOverride).getSpriteKey(this.getGender(ignoreOverride) === Gender.FEMALE, this.formIndex, this.shiny);
2023-03-28 14:54:52 -04:00
}
getBattleSpriteKey(back?: boolean, ignoreOverride?: boolean): string {
return `pkmn__${this.getBattleSpriteId(back, ignoreOverride)}`;
}
getFusionSpriteId(ignoreOverride?: boolean): string {
return this.getFusionSpeciesForm(ignoreOverride).getSpriteId(this.getFusionGender(ignoreOverride) === Gender.FEMALE, this.fusionFormIndex, this.fusionShiny);
}
getFusionBattleSpriteId(back?: boolean, ignoreOverride?: boolean): string {
if (back === undefined)
back = this.isPlayer();
return `${back ? 'back__' : ''}${this.getFusionSpriteId(ignoreOverride)}`;
}
getFusionBattleSpriteKey(back?: boolean, ignoreOverride?: boolean): string {
return `pkmn__${this.getFusionBattleSpriteId(back, ignoreOverride)}`;
}
getFusionBattleSpriteAtlasPath(back?: boolean, ignoreOverride?: boolean): string {
return this.getFusionBattleSpriteId(back, ignoreOverride).replace(/\_{2}/g, '/');
}
getIconAtlasKey(ignoreOverride?: boolean): string {
return this.getSpeciesForm(ignoreOverride).getIconAtlasKey(this.formIndex);
}
getIconId(ignoreOverride?: boolean): string {
return this.getSpeciesForm(ignoreOverride).getIconId(this.getGender(ignoreOverride) === Gender.FEMALE, this.formIndex, this.isShiny());
2023-03-28 14:54:52 -04:00
}
getSpeciesForm(ignoreOverride?: boolean): PokemonSpeciesForm {
if (!ignoreOverride && this.summonData?.speciesForm)
return this.summonData.speciesForm;
if (!this.species.forms?.length)
return this.species;
return this.species.forms[this.formIndex];
}
getFusionSpeciesForm(ignoreOverride?: boolean): PokemonSpeciesForm {
if (!ignoreOverride && this.summonData?.speciesForm)
return this.summonData.fusionSpeciesForm;
if (!this.fusionSpecies?.forms?.length || this.fusionFormIndex >= this.fusionSpecies?.forms.length)
2023-11-04 00:32:12 -04:00
return this.fusionSpecies;
return this.fusionSpecies?.forms[this.fusionFormIndex];
2023-11-04 00:32:12 -04:00
}
2023-03-29 00:31:25 -04:00
getSprite(): Phaser.GameObjects.Sprite {
2023-03-28 14:54:52 -04:00
return this.getAt(0) as Phaser.GameObjects.Sprite;
}
2023-03-29 00:31:25 -04:00
getTintSprite(): Phaser.GameObjects.Sprite {
2023-04-10 23:15:06 -04:00
return !this.maskEnabled
? this.getAt(1) as Phaser.GameObjects.Sprite
: this.maskSprite;
2023-03-28 14:54:52 -04:00
}
getSpriteScale(): number {
2024-02-14 23:25:12 -05:00
const formKey = this.getFormKey();
if (formKey.indexOf(SpeciesFormKey.GIGANTAMAX) > -1 || formKey.indexOf(SpeciesFormKey.ETERNAMAX) > -1)
return 1.5;
return 1;
}
2024-02-27 22:50:27 -05:00
updateScale(): void {
this.setScale(this.getSpriteScale());
}
updateSpritePipelineData(): void {
[ this.getSprite(), this.getTintSprite() ].map(s => s.pipelineData['teraColor'] = getTypeRgb(this.getTeraType()));
this.updateInfo(true);
}
initShinySparkle(): void {
2023-12-29 21:04:40 -05:00
const shinySparkle = this.scene.addFieldSprite(0, 0, 'shiny');
shinySparkle.setVisible(false);
shinySparkle.setOrigin(0.5, 1);
const frameNames = this.scene.anims.generateFrameNames('shiny', { suffix: '.png', end: 34 });
this.scene.anims.create({
key: 'sparkle',
frames: frameNames,
frameRate: 32,
showOnStart: true,
hideOnComplete: true,
});
this.add(shinySparkle);
this.shinySparkle = shinySparkle;
}
playAnim(): void {
this.getSprite().play(this.getBattleSpriteKey());
this.getTintSprite().play(this.getBattleSpriteKey());
2023-04-04 21:10:11 -04:00
}
getFieldPositionOffset(): [ number, number ] {
switch (this.fieldPosition) {
case FieldPosition.CENTER:
return [ 0, 0 ];
case FieldPosition.LEFT:
return [ -32, -8 ];
case FieldPosition.RIGHT:
return [ 32, 0 ];
}
}
setFieldPosition(fieldPosition: FieldPosition, duration?: integer): Promise<void> {
return new Promise(resolve => {
if (fieldPosition === this.fieldPosition) {
resolve();
return;
}
const initialOffset = this.getFieldPositionOffset();
this.fieldPosition = fieldPosition;
this.battleInfo.setMini(fieldPosition !== FieldPosition.CENTER);
this.battleInfo.setOffset(fieldPosition === FieldPosition.RIGHT);
const newOffset = this.getFieldPositionOffset();
let relX = newOffset[0] - initialOffset[0];
let relY = newOffset[1] - initialOffset[1];
if (duration) {
this.scene.tweens.add({
targets: this,
x: (_target, _key, value: number) => value + relX,
y: (_target, _key, value: number) => value + relY,
duration: duration,
ease: 'Sine.easeOut',
onComplete: () => resolve()
});
} else {
this.x += relX;
this.y += relY;
}
});
}
getStat(stat: Stat): integer {
return this.stats[stat];
}
2023-11-05 23:27:40 -05:00
getBattleStat(stat: Stat, opponent?: Pokemon): integer {
2023-04-10 23:15:06 -04:00
if (stat === Stat.HP)
return this.getStat(Stat.HP);
const battleStat = (stat - 1) as BattleStat;
const statLevel = new Utils.IntegerHolder(this.summonData.battleStats[battleStat]);
2023-11-05 23:27:40 -05:00
if (opponent)
applyAbAttrs(IgnoreOpponentStatChangesAbAttr, opponent, null, statLevel);
if (this.isPlayer())
2023-04-20 19:44:56 -04:00
this.scene.applyModifiers(TempBattleStatBoosterModifier, this.isPlayer(), battleStat as integer as TempBattleStat, statLevel);
const statValue = new Utils.NumberHolder(this.getStat(stat));
2023-05-02 15:56:41 -04:00
applyBattleStatMultiplierAbAttrs(BattleStatMultiplierAbAttr, this, battleStat, statValue);
let ret = statValue.value * (Math.max(2, 2 + statLevel.value) / Math.max(2, 2 - statLevel.value));
2023-12-22 22:46:05 -05:00
switch (stat) {
case Stat.ATK:
if (this.getTag(BattlerTagType.SLOW_START))
ret >>= 1;
break;
case Stat.DEF:
break;
case Stat.SPATK:
break;
case Stat.SPDEF:
if (this.scene.arena.weather?.weatherType === WeatherType.SANDSTORM)
ret *= 1.5;
break;
case Stat.SPD:
if (this.getTag(BattlerTagType.SLOW_START))
ret >>= 1;
if (this.status && this.status.effect === StatusEffect.PARALYSIS)
ret >>= 2;
break;
}
return Math.floor(ret);
2023-04-10 23:15:06 -04:00
}
2023-04-10 07:59:00 -04:00
calculateStats(): void {
2023-03-28 14:54:52 -04:00
if (!this.stats)
this.stats = [ 0, 0, 0, 0, 0, 0 ];
const baseStats = this.getSpeciesForm().baseStats.slice(0);
2023-11-04 00:32:12 -04:00
if (this.fusionSpecies) {
const fusionBaseStats = this.getFusionSpeciesForm().baseStats;
for (let s = 0; s < this.stats.length; s++)
baseStats[s] = Math.ceil((baseStats[s] + fusionBaseStats[s]) / 2);
2024-03-14 16:26:57 -04:00
} else if (this.scene.gameMode.isSplicedOnly) {
for (let s = 0; s < this.stats.length; s++)
baseStats[s] = Math.ceil(baseStats[s] / 2);
2023-11-04 00:32:12 -04:00
}
2023-04-20 19:44:56 -04:00
this.scene.applyModifiers(PokemonBaseStatModifier, this.isPlayer(), this, baseStats);
2023-03-28 14:54:52 -04:00
const stats = Utils.getEnumValues(Stat);
for (let s of stats) {
const isHp = s === Stat.HP;
let baseStat = baseStats[s];
2023-11-12 23:47:04 -05:00
let value = Math.floor(((2 * baseStat + this.ivs[s]) * this.level) * 0.01);
2023-03-28 14:54:52 -04:00
if (isHp) {
2024-02-29 15:25:15 -05:00
value = value + this.level + 10;
2023-05-29 13:25:36 -04:00
if (this.getAbility().hasAttr(NonSuperEffectiveImmunityAbAttr))
value = 1;
2023-03-28 14:54:52 -04:00
if (this.hp > value || this.hp === undefined)
this.hp = value;
2023-04-24 22:32:12 -04:00
else if (this.hp) {
2023-03-28 14:54:52 -04:00
const lastMaxHp = this.getMaxHp();
if (lastMaxHp && value > lastMaxHp)
this.hp += value - lastMaxHp;
}
2024-01-05 22:24:05 -05:00
} else {
value += 5;
const natureStatMultiplier = new Utils.NumberHolder(getNatureStatMultiplier(this.nature, s));
this.scene.applyModifier(PokemonNatureWeightModifier, this.isPlayer(), this, natureStatMultiplier);
if (natureStatMultiplier.value !== 1)
value = Math.max(Math[natureStatMultiplier.value > 1 ? 'ceil' : 'floor'](value * natureStatMultiplier.value), 1);
2024-01-05 22:24:05 -05:00
}
2023-03-28 14:54:52 -04:00
this.stats[s] = value;
}
}
2024-01-05 22:24:05 -05:00
setNature(nature: Nature): void {
this.nature = nature;
this.calculateStats();
}
2024-01-09 23:34:43 -05:00
generateNature(naturePool?: Nature[]): void {
if (naturePool === undefined)
naturePool = Utils.getEnumValues(Nature);
const nature = naturePool[Utils.randSeedInt(naturePool.length)];
this.setNature(nature);
}
2023-04-10 07:59:00 -04:00
getMaxHp(): integer {
return this.getStat(Stat.HP);
2023-03-28 14:54:52 -04:00
}
2023-04-11 11:04:39 -04:00
getInverseHp(): integer {
return this.getMaxHp() - this.hp;
}
2023-04-10 07:59:00 -04:00
getHpRatio(): number {
2023-03-29 12:23:52 -04:00
return Math.floor((this.hp / this.getMaxHp()) * 100) / 100;
}
getGender(ignoreOverride?: boolean): Gender {
if (!ignoreOverride && this.summonData?.gender !== undefined)
return this.summonData.gender;
return this.gender;
}
getFusionGender(ignoreOverride?: boolean): Gender {
if (!ignoreOverride && this.summonData?.fusionGender !== undefined)
return this.summonData.fusionGender;
return this.fusionGender;
}
2023-11-05 23:48:04 -05:00
isShiny(): boolean {
return this.shiny || (this.fusionSpecies && this.fusionShiny);
}
2024-01-07 23:17:24 -05:00
abstract isBoss(): boolean;
getMoveset(ignoreOverride?: boolean): PokemonMove[] {
2023-12-11 21:46:49 -05:00
const ret = !ignoreOverride && this.summonData?.moveset
? this.summonData.moveset
: this.moveset;
if (MOVE_OVERRIDE && this.isPlayer())
this.moveset[0] = new PokemonMove(MOVE_OVERRIDE, Math.min(this.moveset[0].ppUsed, allMoves[MOVE_OVERRIDE].pp));
2023-12-11 21:46:49 -05:00
else if (OPP_MOVE_OVERRIDE && !this.isPlayer())
this.moveset[0] = new PokemonMove(OPP_MOVE_OVERRIDE, Math.min(this.moveset[0].ppUsed, allMoves[OPP_MOVE_OVERRIDE].pp));
2023-12-11 21:46:49 -05:00
return ret;
}
getLearnableLevelMoves(): Moves[] {
return this.getLevelMoves(1, true).map(lm => lm[1]).filter(lm => !this.moveset.filter(m => m.moveId === lm).length).filter((move: Moves, i: integer, array: Moves[]) => array.indexOf(move) === i);
}
getTypes(includeTeraType = false, ignoreOverride?: boolean): Type[] {
2023-04-21 19:30:04 -04:00
const types = [];
if (includeTeraType) {
const teraType = this.getTeraType();
if (teraType != Type.UNKNOWN)
types.push(teraType);
}
if (!types.length || !includeTeraType) {
if (!ignoreOverride && this.summonData?.types)
this.summonData.types.forEach(t => types.push(t));
else {
const speciesForm = this.getSpeciesForm();
types.push(speciesForm.type1);
const fusionSpeciesForm = this.getFusionSpeciesForm();
if (fusionSpeciesForm) {
if (fusionSpeciesForm.type2 !== null && fusionSpeciesForm.type2 !== speciesForm.type1)
types.push(fusionSpeciesForm.type2);
else if (fusionSpeciesForm.type1 !== speciesForm.type1)
types.push(fusionSpeciesForm.type1);
}
if (types.length === 1 && speciesForm.type2 !== null)
types.push(speciesForm.type2);
}
2023-04-21 19:30:04 -04:00
}
2023-04-16 00:29:55 -04:00
2023-05-08 18:48:35 -04:00
if (this.getTag(BattlerTagType.IGNORE_FLYING) || this.scene.arena.getTag(ArenaTagType.GRAVITY)) {
2023-04-21 19:30:04 -04:00
const flyingIndex = types.indexOf(Type.FLYING);
2023-04-16 00:29:55 -04:00
if (flyingIndex > -1)
2023-04-21 19:30:04 -04:00
types.splice(flyingIndex, 1);
2023-04-16 00:29:55 -04:00
}
2023-04-21 19:30:04 -04:00
if (!types.length)
types.push(Type.NORMAL);
2023-04-16 00:29:55 -04:00
2023-04-21 19:30:04 -04:00
return types;
2023-04-16 00:29:55 -04:00
}
isOfType(type: Type): boolean {
return !!this.getTypes(true).find(t => t === type);
2023-04-22 22:14:53 -04:00
}
getAbility(ignoreOverride?: boolean): Ability {
if (!ignoreOverride && this.summonData?.ability)
return allAbilities[this.summonData.ability];
2023-07-11 11:32:56 -04:00
if (ABILITY_OVERRIDE && this.isPlayer())
return allAbilities[ABILITY_OVERRIDE];
if (OPP_ABILITY_OVERRIDE && !this.isPlayer())
return allAbilities[OPP_ABILITY_OVERRIDE];
2023-11-04 00:32:12 -04:00
if (this.fusionSpecies)
return allAbilities[this.getFusionSpeciesForm().getAbility(this.fusionAbilityIndex)];
let abilityId = this.getSpeciesForm().getAbility(this.abilityIndex);
if (abilityId === Abilities.NONE)
abilityId = this.species.ability1;
return allAbilities[abilityId];
2023-04-26 23:33:13 -04:00
}
canApplyAbility(): boolean {
2023-12-22 23:57:05 -05:00
const ability = this.getAbility();
if (ability.isIgnorable && this.scene.arena.ignoreAbilities)
return false;
2023-12-22 23:57:05 -05:00
return (this.hp || ability.isPassive) && !this.getAbility().conditions.find(condition => !condition(this));
}
2023-12-10 22:29:13 -05:00
getWeight(): number {
const weight = new Utils.NumberHolder(this.species.weight);
// This will trigger the ability overlay so only call this function when necessary
applyAbAttrs(WeightMultiplierAbAttr, this, null, weight);
return weight.value;
}
getTeraType(): Type {
const teraModifier = this.scene.findModifier(m => m instanceof TerastallizeModifier
&& m.pokemonId === this.id && !!m.getBattlesLeft(), this.isPlayer()) as TerastallizeModifier;
if (teraModifier)
return teraModifier.teraType;
return Type.UNKNOWN;
}
isTerastallized(): boolean {
return this.getTeraType() !== Type.UNKNOWN;
}
isGrounded(): boolean {
return !this.isOfType(Type.FLYING);
}
getAttackMoveEffectiveness(source: Pokemon, move: PokemonMove): TypeDamageMultiplier {
const typeless = !!move.getMove().getAttrs(TypelessAttr).length;
const typeMultiplier = new Utils.NumberHolder(this.getAttackTypeEffectiveness(move.getMove().type));
const cancelled = new Utils.BooleanHolder(false);
if (!typeless)
applyPreDefendAbAttrs(TypeImmunityAbAttr, this, source, move, cancelled, typeMultiplier, true);
if (!cancelled.value)
applyPreDefendAbAttrs(MoveImmunityAbAttr, this, source, move, cancelled, typeMultiplier, true);
return (!cancelled.value ? typeMultiplier.value : 0) as TypeDamageMultiplier;
}
getAttackTypeEffectiveness(moveType: Type): TypeDamageMultiplier {
if (moveType === Type.STELLAR)
return this.isTerastallized() ? 2 : 1;
const types = this.getTypes(true);
return getTypeDamageMultiplier(moveType, types[0]) * (types.length > 1 ? getTypeDamageMultiplier(moveType, types[1]) : 1) as TypeDamageMultiplier;
}
2023-10-07 16:08:33 -04:00
getMatchupScore(pokemon: Pokemon): number {
const types = this.getTypes(true);
const enemyTypes = pokemon.getTypes(true);
2024-02-17 10:51:11 -05:00
const outspeed = (this.isActive(true) ? this.getBattleStat(Stat.SPD, pokemon) : this.getStat(Stat.SPD)) <= pokemon.getBattleStat(Stat.SPD, this);
let atkScore = pokemon.getAttackTypeEffectiveness(types[0]) * (outspeed ? 1.25 : 1);
let defScore = 1 / Math.max(this.getAttackTypeEffectiveness(enemyTypes[0]), 0.25);
2023-10-07 16:08:33 -04:00
if (types.length > 1)
atkScore *= pokemon.getAttackTypeEffectiveness(types[1]);
2023-10-07 16:08:33 -04:00
if (enemyTypes.length > 1)
defScore *= (1 / this.getAttackTypeEffectiveness(enemyTypes[1]));
2024-02-17 10:51:11 -05:00
let hpDiffRatio = this.getHpRatio() + (1 - pokemon.getHpRatio());
if (outspeed)
hpDiffRatio = Math.min(hpDiffRatio * 1.5, 1);
return (atkScore + defScore) * hpDiffRatio;
2023-10-07 16:08:33 -04:00
}
getEvolution(): SpeciesFormEvolution {
2023-04-10 07:59:00 -04:00
if (!pokemonEvolutions.hasOwnProperty(this.species.speciesId))
return null;
const evolutions = pokemonEvolutions[this.species.speciesId];
for (let e of evolutions) {
if (!e.item && this.level >= e.level && (!e.preFormKey || this.getFormKey() === e.preFormKey)) {
2023-04-15 22:51:33 -04:00
if (e.condition === null || (e.condition as SpeciesEvolutionCondition).predicate(this))
2023-04-10 07:59:00 -04:00
return e;
}
}
return null;
}
getLevelMoves(startingLevel?: integer, includeEvolutionMoves: boolean = false, simulateEvolutionChain: boolean = false): LevelMoves {
const ret: LevelMoves = [];
let levelMoves: LevelMoves = [];
if (!startingLevel)
startingLevel = this.level;
if (simulateEvolutionChain) {
const evolutionChain = this.species.getSimulatedEvolutionChain(this.level, this.hasTrainer(), this.isBoss(), this.isPlayer());
for (let e = 0; e < evolutionChain.length; e++) {
const speciesLevelMoves = getPokemonSpecies(evolutionChain[e][0] as Species).getLevelMoves();
levelMoves.push(...speciesLevelMoves.filter(lm => (includeEvolutionMoves && !lm[0]) || (!e && (e === evolutionChain.length - 1 || lm[0] < evolutionChain[e + 1][0]))));
}
const uniqueMoves: Moves[] = [];
levelMoves = levelMoves.filter(lm => {
if (uniqueMoves.find(m => m === lm[1]))
return false;
uniqueMoves.push(lm[1]);
return true;
});
} else
levelMoves = this.getSpeciesForm().getLevelMoves();
if (this.fusionSpecies) {
2023-12-14 08:55:08 -05:00
const evolutionLevelMoves = levelMoves.slice(0, Math.max(levelMoves.findIndex(lm => !!lm[0]), 0));
const fusionLevelMoves = this.getFusionSpeciesForm().getLevelMoves();
const newLevelMoves: LevelMoves = [];
while (levelMoves.length && levelMoves[0][0] < startingLevel)
levelMoves.shift();
while (fusionLevelMoves.length && fusionLevelMoves[0][0] < startingLevel)
fusionLevelMoves.shift();
2023-12-14 08:55:08 -05:00
if (includeEvolutionMoves) {
for (let elm of evolutionLevelMoves.reverse())
levelMoves.unshift(elm);
}
2023-12-09 10:03:36 -05:00
for (let l = includeEvolutionMoves ? 0 : startingLevel; l <= this.level; l++) {
if (l === 1 && startingLevel > 1)
l = startingLevel;
while (levelMoves.length && levelMoves[0][0] === l) {
const levelMove = levelMoves.shift();
if (!newLevelMoves.find(lm => lm[1] === levelMove[1]))
newLevelMoves.push(levelMove);
}
while (fusionLevelMoves.length && fusionLevelMoves[0][0] === l) {
const fusionLevelMove = fusionLevelMoves.shift();
if (!newLevelMoves.find(lm => lm[1] === fusionLevelMove[1]))
newLevelMoves.push(fusionLevelMove);
}
}
levelMoves = newLevelMoves;
}
if (levelMoves) {
2023-04-10 13:54:06 -04:00
for (let lm of levelMoves) {
const level = lm[0];
2023-12-14 08:55:08 -05:00
if ((!includeEvolutionMoves || level) && level < startingLevel)
2023-04-10 13:54:06 -04:00
continue;
else if (level > this.level)
break;
ret.push(lm);
2023-04-10 13:54:06 -04:00
}
}
return ret;
2023-04-10 07:59:00 -04:00
}
setMove(moveIndex: integer, moveId: Moves): void {
const move = moveId ? new PokemonMove(moveId) : null;
2023-05-06 17:31:45 -04:00
this.moveset[moveIndex] = move;
if (this.summonData?.moveset)
this.summonData.moveset[moveIndex] = move;
}
2023-04-10 07:59:00 -04:00
trySetShiny(thresholdOverride?: integer): boolean {
const rand1 = Utils.binToDec(Utils.decToBin(this.id).substring(0, 16));
const rand2 = Utils.binToDec(Utils.decToBin(this.id).substring(16, 32));
const E = this.scene.gameData.trainerId ^ this.scene.gameData.secretId;
const F = rand1 ^ rand2;
let shinyThreshold = new Utils.IntegerHolder(32);
if (thresholdOverride === undefined) {
2023-12-29 21:04:40 -05:00
if (!this.hasTrainer())
this.scene.applyModifiers(ShinyRateBoosterModifier, true, shinyThreshold);
} else
shinyThreshold.value = thresholdOverride;
this.shiny = (E ^ F) < shinyThreshold.value;
if ((E ^ F) < 32)
console.log('REAL SHINY!!');
if (this.shiny)
this.initShinySparkle();
return this.shiny;
}
2023-11-08 18:36:30 -05:00
generateFusionSpecies(forStarter?: boolean): void {
const hiddenAbilityChance = new Utils.IntegerHolder(256);
if (!this.hasTrainer())
this.scene.applyModifiers(HiddenAbilityRateBoosterModifier, true, hiddenAbilityChance);
const hasHiddenAbility = !Utils.randSeedInt(hiddenAbilityChance.value);
const randAbilityIndex = Utils.randSeedInt(2);
const filter = !forStarter ? this.species.getCompatibleFusionSpeciesFilter()
: species => {
2024-01-12 20:16:29 -05:00
return pokemonEvolutions.hasOwnProperty(species.speciesId)
2023-11-08 18:36:30 -05:00
&& !pokemonPrevolutions.hasOwnProperty(species.speciesId)
&& !species.pseudoLegendary
&& !species.legendary
&& !species.mythical
&& species.speciesId !== this.species.speciesId
2023-11-08 18:36:30 -05:00
};
this.fusionSpecies = this.scene.randomSpecies(this.scene.currentBattle?.waveIndex || 0, this.level, false, filter, true);
this.fusionAbilityIndex = (this.fusionSpecies.abilityHidden && hasHiddenAbility ? this.fusionSpecies.ability2 ? 2 : 1 : this.fusionSpecies.ability2 ? randAbilityIndex : 0);
this.fusionShiny = this.shiny;
if (this.fusionSpecies.malePercent === null)
2023-11-08 18:36:30 -05:00
this.fusionGender = Gender.GENDERLESS;
else {
const genderChance = (this.id % 256) * 0.390625;
if (genderChance < this.fusionSpecies.malePercent)
2023-11-08 18:36:30 -05:00
this.fusionGender = Gender.MALE;
else
this.fusionGender = Gender.FEMALE;
}
2024-01-09 23:34:43 -05:00
this.fusionFormIndex = this.scene.getSpeciesFormIndex(this.fusionSpecies, this.fusionGender, this.nature, true);
this.generateName();
2023-11-08 18:36:30 -05:00
}
clearFusionSpecies(): void {
this.fusionSpecies = undefined;
this.fusionFormIndex = 0;
this.fusionAbilityIndex = 0;
this.fusionShiny = false;
this.fusionGender = 0;
this.generateName();
this.calculateStats();
}
2023-04-10 07:59:00 -04:00
generateAndPopulateMoveset(): void {
2023-03-28 14:54:52 -04:00
this.moveset = [];
const movePool = [];
const allLevelMoves = this.getLevelMoves(1, true, true);
2023-03-28 14:54:52 -04:00
if (!allLevelMoves) {
console.log(this.species.speciesId, 'ERROR')
return;
}
2023-04-29 01:40:24 -04:00
2023-03-28 14:54:52 -04:00
for (let m = 0; m < allLevelMoves.length; m++) {
const levelMove = allLevelMoves[m];
if (this.level < levelMove[0])
break;
if (movePool.indexOf(levelMove[1]) === -1)
movePool.push(levelMove[1]);
}
const attackMovePool = movePool.filter(m => {
2023-04-20 21:32:48 -04:00
const move = allMoves[m];
2023-03-28 14:54:52 -04:00
return move.category !== MoveCategory.STATUS;
});
if (attackMovePool.length) {
2024-03-24 18:57:24 -04:00
const randomAttackMove = Utils.randSeedWeightedItem(attackMovePool.reverse());
this.moveset.push(new PokemonMove(randomAttackMove, 0, 0));
movePool.splice(movePool.findIndex(m => m === randomAttackMove), 1);
2023-03-28 14:54:52 -04:00
}
while (movePool.length && this.moveset.length < 4) {
2024-03-24 18:57:24 -04:00
const randomMove = Utils.randSeedWeightedItem(movePool.reverse());
this.moveset.push(new PokemonMove(randomMove, 0, 0));
console.log(allMoves[randomMove]);
movePool.splice(movePool.indexOf(randomMove), 1);
2023-03-28 14:54:52 -04:00
}
this.scene.triggerPokemonFormChange(this, SpeciesFormChangeMoveLearnedTrigger);
2023-03-28 14:54:52 -04:00
}
trySelectMove(moveIndex: integer, ignorePp?: boolean): boolean {
const move = this.getMoveset().length > moveIndex
? this.getMoveset()[moveIndex]
2023-03-28 14:54:52 -04:00
: null;
2023-10-25 09:41:37 -04:00
return move?.isUsable(this, ignorePp);
2023-03-28 14:54:52 -04:00
}
2024-01-07 23:17:24 -05:00
showInfo(): void {
2023-03-28 14:54:52 -04:00
if (!this.battleInfo.visible) {
const otherBattleInfo = this.scene.fieldUI.getAll().slice(0, 4).filter(ui => ui instanceof BattleInfo && ((ui as BattleInfo) instanceof PlayerBattleInfo) === this.isPlayer()).find(() => true);
if (!otherBattleInfo || !this.getFieldIndex())
this.scene.fieldUI.sendToBack(this.battleInfo);
else
this.scene.fieldUI.moveAbove(this.battleInfo, otherBattleInfo);
2024-01-07 23:17:24 -05:00
this.battleInfo.setX(this.battleInfo.x + (this.isPlayer() ? 150 : !this.isBoss() ? -150 : -198));
2023-03-28 14:54:52 -04:00
this.battleInfo.setVisible(true);
this.scene.tweens.add({
targets: this.battleInfo,
2024-01-07 23:17:24 -05:00
x: this.isPlayer() ? '-=150' : `+=${!this.isBoss() ? 150 : 246}`,
2023-03-28 14:54:52 -04:00
duration: 1000,
ease: 'Sine.easeOut'
});
}
}
hideInfo(): Promise<void> {
return new Promise(resolve => {
if (this.battleInfo.visible) {
this.scene.tweens.add({
targets: this.battleInfo,
x: this.isPlayer() ? '+=150' : `-=${!this.isBoss() ? 150 : 246}`,
duration: 500,
ease: 'Sine.easeIn',
onComplete: () => {
this.battleInfo.setVisible(false);
2024-01-07 23:17:24 -05:00
this.battleInfo.setX(this.battleInfo.x - (this.isPlayer() ? 150 : !this.isBoss() ? -150 : -198));
resolve();
}
});
} else
resolve();
});
2023-03-28 14:54:52 -04:00
}
2023-04-10 07:59:00 -04:00
updateInfo(instant?: boolean): Promise<void> {
return this.battleInfo.updateInfo(this, instant);
2023-03-28 14:54:52 -04:00
}
addExp(exp: integer) {
const maxExpLevel = this.scene.getMaxExpLevel();
const initialExp = this.exp;
2023-03-28 14:54:52 -04:00
this.exp += exp;
while (this.level < maxExpLevel && this.exp >= getLevelTotalExp(this.level + 1, this.species.growthRate))
2023-03-28 14:54:52 -04:00
this.level++;
if (this.level >= maxExpLevel) {
console.log(initialExp, this.exp, getLevelTotalExp(this.level, this.species.growthRate));
this.exp = Math.max(getLevelTotalExp(this.level, this.species.growthRate), initialExp);
}
this.levelExp = this.exp - getLevelTotalExp(this.level, this.species.growthRate);
2023-03-28 14:54:52 -04:00
}
getOpponent(targetIndex: integer): Pokemon {
const ret = this.getOpponents()[targetIndex];
if (ret.summonData)
return ret;
return null;
}
getOpponents(): Pokemon[] {
return ((this.isPlayer() ? this.scene.getEnemyField() : this.scene.getPlayerField()) as Pokemon[]).filter(p => p.isActive());
}
getOpponentDescriptor(): string {
const opponents = this.getOpponents();
if (opponents.length === 1)
return opponents[0].name;
return this.isPlayer() ? 'the opposing team' : 'your team';
}
getAlly(): Pokemon {
return (this.isPlayer() ? this.scene.getPlayerField() : this.scene.getEnemyField())[this.getFieldIndex() ? 0 : 1];
}
apply(source: Pokemon, battlerMove: PokemonMove): HitResult {
let result: HitResult;
2023-04-15 01:32:16 -04:00
const move = battlerMove.getMove();
const moveCategory = move.category;
let damage = new Utils.NumberHolder(0);
const cancelled = new Utils.BooleanHolder(false);
const typeless = !!move.getAttrs(TypelessAttr).length;
const types = this.getTypes(true);
const typeMultiplier = new Utils.NumberHolder(!typeless && (moveCategory !== MoveCategory.STATUS || move.getAttrs(StatusMoveTypeImmunityAttr).find(attr => (attr as StatusMoveTypeImmunityAttr).immuneType === move.type))
? getTypeDamageMultiplier(move.type, types[0]) * (types.length > 1 ? getTypeDamageMultiplier(move.type, types[1]) : 1)
: 1);
if (typeless)
typeMultiplier.value = 1;
2023-04-15 01:32:16 -04:00
switch (moveCategory) {
case MoveCategory.PHYSICAL:
case MoveCategory.SPECIAL:
const isPhysical = moveCategory === MoveCategory.PHYSICAL;
const power = new Utils.NumberHolder(move.power);
const sourceTeraType = source.getTeraType();
if (sourceTeraType !== Type.UNKNOWN && sourceTeraType === move.type && power.value < 60 && move.priority <= 0 && !move.getAttrs(MultiHitAttr).length)
power.value = 60;
applyPreAttackAbAttrs(VariableMovePowerAbAttr, source, this, battlerMove, power);
this.scene.getField(true).map(p => applyPreAttackAbAttrs(FieldVariableMovePowerAbAttr, this, source, battlerMove, power));
applyPreDefendAbAttrs(ReceivedMoveDamageMultiplierAbAttr, this, source, battlerMove, cancelled, power);
if (!typeless)
applyPreDefendAbAttrs(TypeImmunityAbAttr, this, source, battlerMove, cancelled, typeMultiplier);
2023-12-22 01:16:56 -05:00
if (!cancelled.value)
applyPreDefendAbAttrs(MoveImmunityAbAttr, this, source, battlerMove, cancelled, typeMultiplier);
2023-04-26 23:33:13 -04:00
if (cancelled.value)
result = HitResult.NO_EFFECT;
2023-04-26 23:33:13 -04:00
else {
if (source.findTag(t => t instanceof TypeBoostTag && (t as TypeBoostTag).boostedType === move.type))
power.value *= 1.5;
const arenaAttackTypeMultiplier = this.scene.arena.getAttackTypeMultiplier(move.type, this.isGrounded());
if (this.scene.arena.terrain?.terrainType === TerrainType.GRASSY && this.isGrounded() && move.type === Type.GROUND && move.moveTarget === MoveTarget.ALL_NEAR_OTHERS)
power.value /= 2;
2023-04-26 23:33:13 -04:00
applyMoveAttrs(VariablePowerAttr, source, this, move, power);
if (!typeless) {
this.scene.arena.applyTags(WeakenMoveTypeTag, move.type, power);
this.scene.applyModifiers(AttackTypeBoosterModifier, source.isPlayer(), source, power);
}
2023-05-05 18:20:55 -04:00
let isCritical: boolean;
const critOnly = new Utils.BooleanHolder(false);
applyMoveAttrs(CritOnlyAttr, source, this, move, critOnly);
if (critOnly.value)
isCritical = true;
else {
const critLevel = new Utils.IntegerHolder(0);
applyMoveAttrs(HighCritAttr, source, this, move, critLevel);
this.scene.applyModifiers(TempBattleStatBoosterModifier, source.isPlayer(), TempBattleStat.CRIT, critLevel);
if (source.getTag(BattlerTagType.CRIT_BOOST))
critLevel.value += 2;
const critChance = Math.ceil(16 / Math.pow(2, critLevel.value));
2024-03-16 22:06:56 -04:00
isCritical = !source.getTag(BattlerTagType.NO_CRIT) && (critChance === 1 || !this.scene.currentBattle.randSeedInt(critChance));
if (isCritical) {
const blockCrit = new Utils.BooleanHolder(false);
applyAbAttrs(BlockCritAbAttr, this, null, blockCrit);
if (blockCrit.value)
isCritical = false;
}
2023-05-05 18:20:55 -04:00
}
2024-03-11 21:45:32 +01:00
const sourceAtk = new Utils.IntegerHolder(source.getBattleStat(isPhysical ? Stat.ATK : Stat.SPATK, this));
const targetDef = new Utils.IntegerHolder(this.getBattleStat(isPhysical ? Stat.DEF : Stat.SPDEF, source));
2023-04-26 23:33:13 -04:00
const criticalMultiplier = isCritical ? 2 : 1;
const isTypeImmune = (typeMultiplier.value * arenaAttackTypeMultiplier) === 0;
const sourceTypes = source.getTypes();
const matchesSourceType = sourceTypes[0] === move.type || (sourceTypes.length > 1 && sourceTypes[1] === move.type);
let stabMultiplier = new Utils.NumberHolder(1);
if (sourceTeraType === Type.UNKNOWN && matchesSourceType)
stabMultiplier.value += 0.5;
else if (sourceTeraType !== Type.UNKNOWN && sourceTeraType === move.type)
stabMultiplier.value += 0.5;
2023-10-26 20:02:30 -04:00
applyAbAttrs(StabBoostAbAttr, source, null, stabMultiplier);
if (sourceTeraType !== Type.UNKNOWN && matchesSourceType)
stabMultiplier.value = Math.min(stabMultiplier.value + 0.5, 2.25);
2024-03-11 21:45:32 +01:00
applyMoveAttrs(VariableAtkAttr, source, this, move, sourceAtk);
2024-03-11 18:18:55 -04:00
applyMoveAttrs(VariableDefAttr, source, this, move, targetDef);
2024-03-11 21:45:32 +01:00
if (!isTypeImmune) {
2024-03-11 21:45:32 +01:00
damage.value = Math.ceil(((((2 * source.level / 5 + 2) * power.value * sourceAtk.value / targetDef.value) / 50) + 2) * stabMultiplier.value * typeMultiplier.value * arenaAttackTypeMultiplier * ((this.scene.currentBattle.randSeedInt(15) + 85) / 100)) * criticalMultiplier;
if (isPhysical && source.status && source.status.effect === StatusEffect.BURN) {
const burnDamageReductionCancelled = new Utils.BooleanHolder(false);
applyAbAttrs(BypassBurnDamageReductionAbAttr, this, burnDamageReductionCancelled);
if (!burnDamageReductionCancelled.value)
damage.value = Math.floor(damage.value / 2);
}
move.getAttrs(HitsTagAttr).map(hta => hta as HitsTagAttr).filter(hta => hta.doubleDamage).forEach(hta => {
if (this.getTag(hta.tagType))
damage.value *= 2;
});
}
2023-04-13 12:16:36 -04:00
if (this.scene.arena.terrain?.terrainType === TerrainType.MISTY && this.isGrounded() && move.type === Type.DRAGON)
damage.value = Math.floor(damage.value / 2);
2023-04-26 23:33:13 -04:00
const fixedDamage = new Utils.IntegerHolder(0);
applyMoveAttrs(FixedDamageAttr, source, this, move, fixedDamage);
if (!isTypeImmune && fixedDamage.value) {
damage.value = fixedDamage.value;
2023-04-26 23:33:13 -04:00
isCritical = false;
result = HitResult.EFFECTIVE;
2023-04-26 23:33:13 -04:00
}
2023-04-15 01:32:16 -04:00
console.log('damage', damage, move.name, power.value, sourceAtk, targetDef);
2023-04-26 23:33:13 -04:00
if (!result) {
2023-11-07 22:23:42 -05:00
if (!typeMultiplier.value)
result = HitResult.NO_EFFECT;
2023-11-07 22:23:42 -05:00
else {
const oneHitKo = new Utils.BooleanHolder(false);
applyMoveAttrs(OneHitKOAttr, source, this, move, oneHitKo);
if (oneHitKo.value) {
result = HitResult.ONE_HIT_KO;
isCritical = false;
damage.value = this.hp;
} else if (typeMultiplier.value >= 2)
result = HitResult.SUPER_EFFECTIVE;
else if (typeMultiplier.value >= 1)
result = HitResult.EFFECTIVE;
else
result = HitResult.NOT_VERY_EFFECTIVE;
}
2023-04-26 23:33:13 -04:00
}
2023-04-15 01:32:16 -04:00
if (!fixedDamage.value) {
if (!source.isPlayer())
this.scene.applyModifiers(EnemyDamageBoosterModifier, false, damage);
if (!this.isPlayer())
this.scene.applyModifiers(EnemyDamageReducerModifier, false, damage);
}
2023-04-26 23:33:13 -04:00
if (damage) {
2024-03-09 23:49:00 +01:00
applyPreDefendAbAttrs(PreDefendEndureAbAttr, this, source, battlerMove, cancelled, damage);
2024-03-09 10:40:51 -05:00
const oneHitKo = result === HitResult.ONE_HIT_KO;
damage.value = this.damageAndUpdate(damage.value, result as DamageResult, isCritical, oneHitKo, oneHitKo);
2023-04-26 23:33:13 -04:00
if (isCritical)
this.scene.queueMessage('A critical hit!');
this.scene.setPhaseQueueSplice();
if (source.isPlayer()) {
this.scene.validateAchvs(DamageAchv, damage);
if (damage.value > this.scene.gameData.gameStats.highestDamage)
this.scene.gameData.gameStats.highestDamage = damage.value;
}
source.turnData.damageDealt += damage.value;
2023-11-28 21:35:52 -05:00
this.battleData.hitCount++;
2024-03-01 00:27:46 -05:00
const attackResult = { move: move.id, result: result as DamageResult, damage: damage.value, critical: isCritical, sourceId: source.id };
this.turnData.attacksReceived.unshift(attackResult);
if (source.isPlayer() && !this.isPlayer())
this.scene.applyModifiers(DamageMoneyRewardModifier, true, source, damage)
2023-04-26 23:33:13 -04:00
}
2023-10-30 12:33:20 -04:00
if (source.turnData.hitsLeft === 1) {
switch (result) {
case HitResult.SUPER_EFFECTIVE:
this.scene.queueMessage('It\'s super effective!');
break;
case HitResult.NOT_VERY_EFFECTIVE:
this.scene.queueMessage('It\'s not very effective!');
break;
case HitResult.NO_EFFECT:
this.scene.queueMessage(`It doesn\'t affect ${this.name}!`);
break;
2023-11-07 22:23:42 -05:00
case HitResult.ONE_HIT_KO:
this.scene.queueMessage('It\'s a one-hit KO!');
break;
2023-10-30 12:33:20 -04:00
}
2023-04-26 23:33:13 -04:00
}
if (damage)
this.scene.clearPhaseQueueSplice();
2023-04-15 01:32:16 -04:00
}
break;
case MoveCategory.STATUS:
if (!typeless)
applyPreDefendAbAttrs(TypeImmunityAbAttr, this, source, battlerMove, cancelled, typeMultiplier);
2023-12-22 01:16:56 -05:00
if (!cancelled.value)
applyPreDefendAbAttrs(MoveImmunityAbAttr, this, source, battlerMove, cancelled, typeMultiplier);
if (!typeMultiplier.value)
this.scene.queueMessage(`It doesn\'t affect ${this.name}!`);
result = cancelled.value || !typeMultiplier.value ? HitResult.NO_EFFECT : HitResult.STATUS;
2023-04-15 01:32:16 -04:00
break;
}
return result;
2023-04-13 12:16:36 -04:00
}
2024-01-07 23:17:24 -05:00
damage(damage: integer, ignoreSegments: boolean = false, preventEndure: boolean = false): integer {
if (this.isFainted())
2024-01-07 23:17:24 -05:00
return 0;
2023-04-16 18:40:32 -04:00
2023-04-23 21:31:06 -04:00
if (this.hp > 1 && this.hp - damage <= 0 && !preventEndure) {
const surviveDamage = new Utils.BooleanHolder(false);
if (this.lapseTag(BattlerTagType.ENDURING))
surviveDamage.value = true;
if (!surviveDamage.value)
this.scene.applyModifiers(SurviveDamageModifier, this.isPlayer(), this, surviveDamage);
2023-04-23 21:31:06 -04:00
if (surviveDamage.value)
damage = this.hp - 1;
}
2024-01-07 23:17:24 -05:00
damage = Math.min(damage, this.hp);
this.hp = this.hp - damage;
if (this.isFainted()) {
this.scene.unshiftPhase(new FaintPhase(this.scene, this.getBattlerIndex(), preventEndure));
2023-04-20 21:32:48 -04:00
this.resetSummonData();
2023-04-16 18:40:32 -04:00
}
2024-01-07 23:17:24 -05:00
return damage;
2023-04-16 18:40:32 -04:00
}
damageAndUpdate(damage: integer, result?: DamageResult, critical: boolean = false, ignoreSegments: boolean = false, preventEndure: boolean = false): integer {
const damagePhase = new DamagePhase(this.scene, this.getBattlerIndex(), damage, result as DamageResult, critical);
this.scene.unshiftPhase(damagePhase);
damage = this.damage(damage, ignoreSegments, preventEndure);
// Damage amount may have changed, but needed to be queued before calling damage function
damagePhase.updateAmount(damage);
return damage;
}
2024-01-07 23:17:24 -05:00
heal(amount: integer): integer {
const healAmount = Math.min(amount, this.getMaxHp() - this.hp);
this.hp += healAmount;
return healAmount;
2023-10-21 20:52:19 -04:00
}
2023-12-18 21:53:28 -04:00
isBossImmune(): boolean {
2024-01-07 23:17:24 -05:00
return this.isBoss();
2023-12-18 21:53:28 -04:00
}
addTag(tagType: BattlerTagType, turnCount: integer = 0, sourceMove?: Moves, sourceId?: integer): boolean {
2023-04-15 01:32:16 -04:00
const existingTag = this.getTag(tagType);
if (existingTag) {
existingTag.onOverlap(this);
2023-04-13 12:16:36 -04:00
return false;
2023-04-15 01:32:16 -04:00
}
2023-04-13 12:16:36 -04:00
const newTag = getBattlerTag(tagType, turnCount, sourceMove, sourceId);
2023-04-21 19:30:04 -04:00
const cancelled = new Utils.BooleanHolder(false);
applyPreApplyBattlerTagAbAttrs(PreApplyBattlerTagAbAttr, this, newTag, cancelled);
if (!cancelled.value && newTag.canAdd(this)) {
2023-04-22 22:14:53 -04:00
this.summonData.tags.push(newTag);
newTag.onAdd(this);
return true;
}
return false;
2023-04-13 12:16:36 -04:00
}
2023-04-21 19:30:04 -04:00
getTag(tagType: BattlerTagType | { new(...args: any[]): BattlerTag }): BattlerTag {
if (!this.summonData)
return null;
2023-12-22 22:46:05 -05:00
return typeof(tagType) === 'string'
2023-04-15 01:32:16 -04:00
? this.summonData.tags.find(t => t.tagType === tagType)
: this.summonData.tags.find(t => t instanceof tagType);
2023-04-13 12:16:36 -04:00
}
2023-04-21 19:30:04 -04:00
findTag(tagFilter: ((tag: BattlerTag) => boolean)) {
if (!this.summonData)
return null;
2023-04-15 01:32:16 -04:00
return this.summonData.tags.find(t => tagFilter(t));
2023-04-13 12:16:36 -04:00
}
2023-04-21 19:30:04 -04:00
getTags(tagType: BattlerTagType | { new(...args: any[]): BattlerTag }): BattlerTag[] {
if (!this.summonData)
return [];
2023-12-22 22:46:05 -05:00
return typeof(tagType) === 'string'
2023-04-15 01:32:16 -04:00
? this.summonData.tags.filter(t => t.tagType === tagType)
: this.summonData.tags.filter(t => t instanceof tagType);
}
findTags(tagFilter: ((tag: BattlerTag) => boolean)): BattlerTag[] {
if (!this.summonData)
return [];
2023-04-15 01:32:16 -04:00
return this.summonData.tags.filter(t => tagFilter(t));
}
2023-04-21 19:30:04 -04:00
lapseTag(tagType: BattlerTagType): boolean {
2023-04-13 12:16:36 -04:00
const tags = this.summonData.tags;
2023-04-15 01:32:16 -04:00
const tag = tags.find(t => t.tagType === tagType);
2023-04-21 19:30:04 -04:00
if (tag && !(tag.lapse(this, BattlerTagLapseType.CUSTOM))) {
2023-04-15 01:32:16 -04:00
tag.onRemove(this);
tags.splice(tags.indexOf(tag), 1);
}
2023-04-18 12:30:47 -04:00
return !!tag;
2023-04-15 01:32:16 -04:00
}
2023-04-21 19:30:04 -04:00
lapseTags(lapseType: BattlerTagLapseType): void {
2023-04-15 01:32:16 -04:00
const tags = this.summonData.tags;
2023-04-21 19:30:04 -04:00
tags.filter(t => lapseType === BattlerTagLapseType.FAINT || ((t.lapseType === lapseType) && !(t.lapse(this, lapseType))) || (lapseType === BattlerTagLapseType.TURN_END && t.turnCount < 1)).forEach(t => {
2023-04-15 01:32:16 -04:00
t.onRemove(this);
tags.splice(tags.indexOf(t), 1);
});
2023-04-13 12:16:36 -04:00
}
2024-02-20 12:27:38 -05:00
removeTag(tagType: BattlerTagType): boolean {
const tags = this.summonData.tags;
const tag = tags.find(t => t.tagType === tagType);
if (tag) {
tag.turnCount = 0;
tag.onRemove(this);
tags.splice(tags.indexOf(tag), 1);
}
return !!tag;
}
findAndRemoveTags(tagFilter: ((tag: BattlerTag) => boolean)): boolean {
if (!this.summonData)
return false;
2023-04-22 22:14:53 -04:00
const tags = this.summonData.tags;
const tagsToRemove = tags.filter(t => tagFilter(t));
for (let tag of tagsToRemove) {
tag.turnCount = 0;
tag.onRemove(this);
tags.splice(tags.indexOf(tag), 1);
}
return true;
}
removeTagsBySourceId(sourceId: integer): void {
this.findAndRemoveTags(t => t.sourceId === sourceId);
2023-04-22 22:14:53 -04:00
}
2023-04-28 19:26:41 -04:00
transferTagsBySourceId(sourceId: integer, newSourceId: integer): void {
if (!this.summonData)
return;
2023-04-28 19:26:41 -04:00
const tags = this.summonData.tags;
tags.filter(t => t.sourceId === sourceId).forEach(t => t.sourceId = newSourceId);
}
transferSummon(source: Pokemon): void {
const battleStats = Utils.getEnumValues(BattleStat);
for (let stat of battleStats)
this.summonData.battleStats[stat] = source.summonData.battleStats[stat];
for (let tag of source.summonData.tags)
this.summonData.tags.push(tag);
2024-02-21 18:25:47 -05:00
if (this instanceof PlayerPokemon && source.summonData.battleStats.find(bs => bs === 6))
this.scene.validateAchv(achvs.TRANSFER_MAX_BATTLE_STAT);
2023-04-28 19:26:41 -04:00
}
2023-04-20 21:32:48 -04:00
getMoveHistory(): TurnMove[] {
2023-04-28 15:03:42 -04:00
return this.battleSummonData.moveHistory;
2023-04-20 21:32:48 -04:00
}
2023-04-25 01:32:48 -04:00
pushMoveHistory(turnMove: TurnMove) {
turnMove.turn = this.scene.currentBattle?.turn;
this.getMoveHistory().push(turnMove);
}
2023-04-13 12:16:36 -04:00
getLastXMoves(turnCount?: integer): TurnMove[] {
2023-04-20 21:32:48 -04:00
const moveHistory = this.getMoveHistory();
2023-04-18 12:30:47 -04:00
return moveHistory.slice(turnCount >= 0 ? Math.max(moveHistory.length - (turnCount || 1), 0) : 0, moveHistory.length).reverse();
2023-03-28 14:54:52 -04:00
}
2023-04-20 21:32:48 -04:00
getMoveQueue(): QueuedMove[] {
return this.summonData.moveQueue;
}
2024-01-09 23:34:43 -05:00
changeForm(formChange: SpeciesFormChange): Promise<void> {
return new Promise(resolve => {
this.formIndex = Math.max(this.species.forms.findIndex(f => f.formKey === formChange.formKey), 0);
this.generateName();
const abilityCount = this.getSpeciesForm().getAbilityCount();
if (this.abilityIndex >= abilityCount) // Shouldn't happen
this.abilityIndex = abilityCount - 1;
this.scene.gameData.setPokemonSeen(this, true);
this.setScale(this.getSpriteScale());
2024-01-09 23:34:43 -05:00
this.loadAssets().then(() => {
this.calculateStats();
this.scene.updateModifiers(this.isPlayer(), true);
2024-02-14 23:25:12 -05:00
Promise.all([ this.updateInfo(), this.scene.updateFieldScale() ]).then(() => resolve());
2024-01-09 23:34:43 -05:00
});
});
}
2023-11-04 00:32:12 -04:00
cry(soundConfig?: Phaser.Types.Sound.SoundConfig): AnySound {
const cry = this.getSpeciesForm().cry(this.scene, soundConfig);
let duration = cry.totalDuration * 1000;
if (this.fusionSpecies) {
let fusionCry = this.getFusionSpeciesForm().cry(this.scene, soundConfig, true);
duration = Math.min(duration, fusionCry.totalDuration * 1000);
fusionCry.destroy();
this.scene.time.delayedCall(Utils.fixedInt(Math.ceil(duration * 0.4)), () => {
try {
SoundFade.fadeOut(this.scene, cry, Utils.fixedInt(Math.ceil(duration * 0.2)));
fusionCry = this.getFusionSpeciesForm().cry(this.scene, Object.assign({ seek: Math.max(fusionCry.totalDuration * 0.4, 0) }, soundConfig));
SoundFade.fadeIn(this.scene, fusionCry, Utils.fixedInt(Math.ceil(duration * 0.2)), this.scene.masterVolume * this.scene.seVolume, 0);
} catch (err) {
console.error(err);
}
});
}
return cry;
2023-03-28 14:54:52 -04:00
}
2023-11-04 00:32:12 -04:00
faintCry(callback: Function): void {
if (this.fusionSpecies)
return this.fusionFaintCry(callback);
2023-11-05 22:32:08 -05:00
const key = this.getSpeciesForm().getCryKey(this.formIndex);
2023-03-28 14:54:52 -04:00
let i = 0;
let rate = 0.85;
2023-11-04 00:32:12 -04:00
const cry = this.scene.playSound(key, { rate: rate }) as AnySound;
2023-03-28 14:54:52 -04:00
const sprite = this.getSprite();
2023-04-03 20:47:41 -04:00
const tintSprite = this.getTintSprite();
2023-11-04 00:32:12 -04:00
2023-03-28 14:54:52 -04:00
const delay = Math.max(this.scene.sound.get(key).totalDuration * 50, 25);
2023-11-04 00:32:12 -04:00
2023-03-28 14:54:52 -04:00
let frameProgress = 0;
let frameThreshold: number;
2023-11-04 00:32:12 -04:00
2023-03-28 14:54:52 -04:00
sprite.anims.pause();
2023-04-03 20:47:41 -04:00
tintSprite.anims.pause();
2023-11-04 00:32:12 -04:00
2023-03-28 14:54:52 -04:00
let faintCryTimer = this.scene.time.addEvent({
2023-10-26 16:33:59 -04:00
delay: Utils.fixedInt(delay),
2023-03-28 14:54:52 -04:00
repeat: -1,
callback: () => {
++i;
frameThreshold = sprite.anims.msPerFrame / rate;
frameProgress += delay;
while (frameProgress > frameThreshold) {
2023-04-03 20:47:41 -04:00
if (sprite.anims.duration) {
2023-03-28 14:54:52 -04:00
sprite.anims.nextFrame();
2023-04-03 20:47:41 -04:00
tintSprite.anims.nextFrame();
}
2023-03-28 14:54:52 -04:00
frameProgress -= frameThreshold;
}
2023-11-04 00:32:12 -04:00
if (cry && !cry.pendingRemove) {
2023-03-28 14:54:52 -04:00
rate *= 0.99;
2023-11-04 00:32:12 -04:00
cry.setRate(rate);
2023-11-10 16:41:02 -05:00
} else {
2023-03-28 14:54:52 -04:00
faintCryTimer.destroy();
faintCryTimer = null;
if (callback)
callback();
}
}
});
2023-11-04 00:32:12 -04:00
2023-03-28 14:54:52 -04:00
// Failsafe
2023-10-26 16:33:59 -04:00
this.scene.time.delayedCall(Utils.fixedInt(3000), () => {
2023-03-28 14:54:52 -04:00
if (!faintCryTimer || !this.scene)
return;
2023-11-04 00:32:12 -04:00
if (cry?.isPlaying)
cry.stop();
faintCryTimer.destroy();
if (callback)
callback();
});
}
private fusionFaintCry(callback: Function): void {
const key = this.getSpeciesForm().getCryKey(this.formIndex);
2023-11-04 00:32:12 -04:00
let i = 0;
let rate = 0.85;
let cry = this.scene.playSound(key, { rate: rate }) as AnySound;
const sprite = this.getSprite();
const tintSprite = this.getTintSprite();
let duration = cry.totalDuration * 1000;
let fusionCry = this.scene.playSound(this.getFusionSpeciesForm().getCryKey(this.fusionFormIndex), { rate: rate }) as AnySound;
fusionCry.stop();
duration = Math.min(duration, fusionCry.totalDuration * 1000);
fusionCry.destroy();
const delay = Math.max(duration * 0.05, 25);
let transitionIndex = 0;
let durationProgress = 0;
const transitionThreshold = Math.ceil(duration * 0.4);
while (durationProgress < transitionThreshold) {
++i;
durationProgress += delay * rate;
rate *= 0.99;
}
transitionIndex = i;
i = 0;
rate = 0.85;
let frameProgress = 0;
let frameThreshold: number;
sprite.anims.pause();
tintSprite.anims.pause();
let faintCryTimer = this.scene.time.addEvent({
delay: Utils.fixedInt(delay),
repeat: -1,
callback: () => {
++i;
frameThreshold = sprite.anims.msPerFrame / rate;
frameProgress += delay;
while (frameProgress > frameThreshold) {
if (sprite.anims.duration) {
sprite.anims.nextFrame();
tintSprite.anims.nextFrame();
}
frameProgress -= frameThreshold;
}
if (i === transitionIndex) {
SoundFade.fadeOut(this.scene, cry, Utils.fixedInt(Math.ceil((duration / rate) * 0.2)));
fusionCry = this.scene.playSound(this.getFusionSpeciesForm().getCryKey(this.fusionFormIndex), Object.assign({ seek: Math.max(fusionCry.totalDuration * 0.4, 0), rate: rate }));
SoundFade.fadeIn(this.scene, fusionCry, Utils.fixedInt(Math.ceil((duration / rate) * 0.2)), this.scene.masterVolume * this.scene.seVolume, 0);
}
rate *= 0.99;
if (cry && !cry.pendingRemove)
cry.setRate(rate);
if (fusionCry && !fusionCry.pendingRemove)
fusionCry.setRate(rate);
if ((!cry || cry.pendingRemove) && (!fusionCry || fusionCry.pendingRemove)) {
faintCryTimer.destroy();
faintCryTimer = null;
if (callback)
callback();
}
}
});
// Failsafe
this.scene.time.delayedCall(Utils.fixedInt(3000), () => {
if (!faintCryTimer || !this.scene)
return;
if (cry?.isPlaying)
cry.stop();
if (fusionCry?.isPlaying)
fusionCry.stop();
2023-03-28 14:54:52 -04:00
faintCryTimer.destroy();
if (callback)
callback();
});
}
isOppositeGender(pokemon: Pokemon): boolean {
return this.gender !== Gender.GENDERLESS && pokemon.gender === (this.gender === Gender.MALE ? Gender.FEMALE : Gender.MALE);
}
canSetStatus(effect: StatusEffect, quiet: boolean = false): boolean {
2024-03-13 20:36:25 -04:00
if (effect !== StatusEffect.FAINT) {
if (this.status)
return false;
if (this.isGrounded() && this.scene.arena.terrain?.terrainType === TerrainType.MISTY)
return false;
}
2023-04-11 19:08:03 -04:00
switch (effect) {
case StatusEffect.POISON:
case StatusEffect.TOXIC:
2023-04-22 22:14:53 -04:00
if (this.isOfType(Type.POISON) || this.isOfType(Type.STEEL))
2023-04-11 19:08:03 -04:00
return false;
break;
case StatusEffect.PARALYSIS:
if (this.isOfType(Type.ELECTRIC))
return false;
break;
case StatusEffect.SLEEP:
if (this.isGrounded() && this.scene.arena.terrain?.terrainType === TerrainType.ELECTRIC)
return false;
2023-04-11 19:08:03 -04:00
case StatusEffect.FREEZE:
2023-04-22 22:14:53 -04:00
if (this.isOfType(Type.ICE))
2023-04-11 19:08:03 -04:00
return false;
break;
case StatusEffect.BURN:
2023-04-22 22:14:53 -04:00
if (this.isOfType(Type.FIRE))
2023-04-11 19:08:03 -04:00
return false;
break;
}
const cancelled = new Utils.BooleanHolder(false);
applyPreSetStatusAbAttrs(StatusEffectImmunityAbAttr, this, effect, cancelled, quiet);
if (cancelled.value)
return false;
return true;
}
trySetStatus(effect: StatusEffect, asPhase: boolean = false, cureTurn: integer = 0, sourceText: string = null): boolean {
if (!this.canSetStatus(effect, asPhase))
return false;
if (asPhase) {
this.scene.unshiftPhase(new ObtainStatusEffectPhase(this.scene, this.getBattlerIndex(), effect, cureTurn, sourceText));
return true;
}
let statusCureTurn: Utils.IntegerHolder;
if (effect === StatusEffect.SLEEP) {
statusCureTurn = new Utils.IntegerHolder(this.randSeedIntRange(2, 4));
applyAbAttrs(ReduceStatusEffectDurationAbAttr, this, null, effect, statusCureTurn);
2024-01-16 00:28:03 -05:00
this.setFrameRate(4);
}
this.status = new Status(effect, 0, statusCureTurn?.value);
2024-01-09 23:34:43 -05:00
if (effect !== StatusEffect.FAINT)
this.scene.triggerPokemonFormChange(this, SpeciesFormChangeStatusEffectTrigger, true);
2023-04-11 19:08:03 -04:00
return true;
}
resetStatus(): void {
const lastStatus = this.status?.effect;
2023-04-11 19:08:03 -04:00
this.status = undefined;
2023-04-16 18:40:32 -04:00
if (lastStatus === StatusEffect.SLEEP) {
this.setFrameRate(12);
2023-04-21 19:30:04 -04:00
if (this.getTag(BattlerTagType.NIGHTMARE))
this.lapseTag(BattlerTagType.NIGHTMARE);
2023-04-16 18:40:32 -04:00
}
2023-04-11 19:08:03 -04:00
}
primeSummonData(summonDataPrimer: PokemonSummonData): void {
this.summonDataPrimer = summonDataPrimer;
}
2023-04-11 19:08:03 -04:00
resetSummonData(): void {
if (this.summonData?.speciesForm) {
this.summonData.speciesForm = null;
this.updateFusionPalette();
}
2023-04-03 23:38:31 -04:00
this.summonData = new PokemonSummonData();
2023-11-28 21:35:52 -05:00
if (!this.battleData)
this.resetBattleData();
2023-04-03 23:38:31 -04:00
this.resetBattleSummonData();
if (this.summonDataPrimer) {
for (let k of Object.keys(this.summonData)) {
if (this.summonDataPrimer[k])
this.summonData[k] = this.summonDataPrimer[k];
}
this.summonDataPrimer = null;
}
2023-04-03 23:38:31 -04:00
}
2023-11-28 21:35:52 -05:00
resetBattleData(): void {
this.battleData = new PokemonBattleData();
}
2023-04-11 19:08:03 -04:00
resetBattleSummonData(): void {
2023-04-03 23:38:31 -04:00
this.battleSummonData = new PokemonBattleSummonData();
2023-04-21 19:30:04 -04:00
if (this.getTag(BattlerTagType.SEEDED))
this.lapseTag(BattlerTagType.SEEDED);
2024-01-09 23:53:31 -05:00
if (this.scene)
this.scene.triggerPokemonFormChange(this, SpeciesFormChangeMoveUsedTrigger, true);
2023-04-03 23:38:31 -04:00
}
2023-04-11 19:08:03 -04:00
resetTurnData(): void {
2023-04-03 23:38:31 -04:00
this.turnData = new PokemonTurnData();
}
getExpValue(): integer {
// Logic to factor in victor level has been removed for balancing purposes, so the player doesn't have to focus on EXP maxxing
2024-03-17 11:36:19 -04:00
return ((this.getSpeciesForm().getBaseExp() * this.level) / 5 + 1);
2023-03-28 14:54:52 -04:00
}
setFrameRate(frameRate: integer) {
this.scene.anims.get(this.getBattleSpriteKey()).frameRate = frameRate;
this.getSprite().play(this.getBattleSpriteKey());
this.getTintSprite().play(this.getBattleSpriteKey());
}
2023-03-28 14:54:52 -04:00
tint(color: number, alpha?: number, duration?: integer, ease?: string) {
const tintSprite = this.getTintSprite();
tintSprite.setTintFill(color);
tintSprite.setVisible(true);
if (duration) {
tintSprite.setAlpha(0);
this.scene.tweens.add({
targets: tintSprite,
alpha: alpha || 1,
duration: duration,
ease: ease || 'Linear'
});
} else
tintSprite.setAlpha(alpha);
}
untint(duration: integer, ease?: string) {
const tintSprite = this.getTintSprite();
if (duration) {
this.scene.tweens.add({
targets: tintSprite,
alpha: 0,
duration: duration,
ease: ease || 'Linear',
2023-04-10 23:15:06 -04:00
onComplete: () => {
tintSprite.setVisible(false);
tintSprite.setAlpha(1);
}
2023-03-28 14:54:52 -04:00
});
2023-04-10 23:15:06 -04:00
} else {
2023-03-28 14:54:52 -04:00
tintSprite.setVisible(false);
2023-04-10 23:15:06 -04:00
tintSprite.setAlpha(1);
}
}
enableMask() {
if (!this.maskEnabled) {
this.maskSprite = this.getTintSprite();
this.maskSprite.setVisible(true);
this.maskSprite.setPosition(this.x * this.parentContainer.scale + this.parentContainer.x,
this.y * this.parentContainer.scale + this.parentContainer.y);
this.maskSprite.setScale(this.getSpriteScale() * this.parentContainer.scale);
2023-04-10 23:15:06 -04:00
this.maskEnabled = true;
}
}
disableMask() {
if (this.maskEnabled) {
this.maskSprite.setVisible(false);
this.maskSprite.setPosition(0, 0);
this.maskSprite.setScale(this.getSpriteScale());
2023-04-10 23:15:06 -04:00
this.maskSprite = null;
this.maskEnabled = false;
}
2023-03-28 14:54:52 -04:00
}
sparkle(): void {
if (this.shinySparkle) {
this.shinySparkle.play('sparkle');
2023-10-21 08:58:39 -04:00
this.scene.playSound('sparkle');
2023-03-28 14:54:52 -04:00
}
}
updateFusionPalette(ignoreOveride?: boolean): void {
if (!this.getFusionSpeciesForm(ignoreOveride)) {
2023-11-23 23:52:13 -05:00
[ this.getSprite(), this.getTintSprite() ].map(s => {
s.pipelineData[`spriteColors${ignoreOveride && this.summonData?.speciesForm ? 'Base' : ''}`] = [];
s.pipelineData[`fusionSpriteColors${ignoreOveride && this.summonData?.speciesForm ? 'Base' : ''}`] = [];
2023-11-23 23:52:13 -05:00
});
return;
}
const speciesForm = this.getSpeciesForm(ignoreOveride);
const fusionSpeciesForm = this.getFusionSpeciesForm(ignoreOveride);
const sourceTexture = this.scene.textures.get(speciesForm.getSpriteKey(this.getGender(ignoreOveride) === Gender.FEMALE, speciesForm.formIndex, this.shiny));
const sourceBackTexture = this.scene.textures.get(speciesForm.getSpriteKey(this.getGender(ignoreOveride) === Gender.FEMALE, speciesForm.formIndex, this.shiny).replace('pkmn__', 'pkmn__back__'));
const fusionTexture = this.scene.textures.get(fusionSpeciesForm.getSpriteKey(this.getFusionGender(ignoreOveride) === Gender.FEMALE, fusionSpeciesForm.formIndex, this.fusionShiny));
const fusionBackTexture = this.scene.textures.get(fusionSpeciesForm.getSpriteKey(this.getFusionGender(ignoreOveride) === Gender.FEMALE, fusionSpeciesForm.formIndex, this.fusionShiny).replace('pkmn__', 'pkmn__back__'));
2023-11-23 23:52:13 -05:00
const [ sourceFrame, sourceBackFrame, fusionFrame, fusionBackFrame ] = [ sourceTexture, sourceBackTexture, fusionTexture, fusionBackTexture ].map(texture => texture.frames[texture.firstFrame]);
const [ sourceImage, sourceBackImage, fusionImage, fusionBackImage ] = [ sourceTexture, sourceBackTexture, fusionTexture, fusionBackTexture ].map(i => i.getSourceImage() as HTMLImageElement);
2023-11-23 23:52:13 -05:00
const canvas = document.createElement('canvas');
const backCanvas = document.createElement('canvas');
2023-11-23 23:52:13 -05:00
const fusionCanvas = document.createElement('canvas');
const fusionBackCanvas = document.createElement('canvas');
2023-11-23 23:52:13 -05:00
const spriteColors: integer[][] = [];
const pixelData: Uint8ClampedArray[] = [];
[ canvas, backCanvas, fusionCanvas, fusionBackCanvas ].forEach((canv: HTMLCanvasElement, c: integer) => {
2023-11-23 23:52:13 -05:00
const context = canv.getContext('2d');
const frame = [ sourceFrame, sourceBackFrame, fusionFrame, fusionBackFrame ][c];
2023-11-23 23:52:13 -05:00
canv.width = frame.width;
canv.height = frame.height;
context.drawImage([ sourceImage, sourceBackImage, fusionImage, fusionBackImage ][c], frame.cutX, frame.cutY, frame.width, frame.height, 0, 0, frame.width, frame.height);
2023-11-23 23:52:13 -05:00
const imageData = context.getImageData(frame.cutX, frame.cutY, frame.width, frame.height);
pixelData.push(imageData.data);
});
for (let f = 0; f < 2; f++) {
for (let i = 0; i < pixelData[f].length; i += 4) {
if (pixelData[f][i + 3]) {
const pixel = pixelData[f].slice(i, i + 4);
const [ r, g, b, a ] = pixel;
if (!spriteColors.find(c => c[0] === r && c[1] === g && c[2] === b))
spriteColors.push([ r, g, b, a ]);
}
2023-11-23 23:52:13 -05:00
}
}
const fusionSpriteColors = JSON.parse(JSON.stringify(spriteColors));
const pixelColors = [];
for (let f = 0; f < 2; f++) {
for (let i = 0; i < pixelData[f].length; i += 4) {
const total = pixelData[f].slice(i, i + 3).reduce((total: integer, value: integer) => total + value, 0);
if (!total)
continue;
pixelColors.push(argbFromRgba({ r: pixelData[f][i], g: pixelData[f][i + 1], b: pixelData[f][i + 2], a: pixelData[f][i + 3] }));
}
2023-11-23 23:52:13 -05:00
}
const fusionPixelColors = [];
for (let f = 0; f < 2; f++) {
for (let i = 0; i < pixelData[2 + f].length; i += 4) {
const total = pixelData[2 + f].slice(i, i + 3).reduce((total: integer, value: integer) => total + value, 0);
if (!total)
continue;
fusionPixelColors.push(argbFromRgba({ r: pixelData[2 + f][i], g: pixelData[2 + f][i + 1], b: pixelData[2 + f][i + 2], a: pixelData[2 + f][i + 3] }));
}
2023-11-23 23:52:13 -05:00
}
let paletteColors: Map<number, number>;
let fusionPaletteColors: Map<number, number>;
const originalRandom = Math.random;
Math.random = () => Phaser.Math.RND.realInRange(0, 1);
this.scene.executeWithSeedOffset(() => {
paletteColors = QuantizerCelebi.quantize(pixelColors, 4);
fusionPaletteColors = QuantizerCelebi.quantize(fusionPixelColors, 4);
}, 0, 'This result should not vary');
Math.random = originalRandom;
const [ palette, fusionPalette ] = [ paletteColors, fusionPaletteColors ]
.map(paletteColors => {
let keys = Array.from(paletteColors.keys()).sort((a: integer, b: integer) => paletteColors.get(a) < paletteColors.get(b) ? 1 : -1);
let rgbaColors: Map<number, integer[]>;
let hsvColors: Map<number, number[]>;
const mappedColors = new Map<integer, integer[]>();
do {
mappedColors.clear();
rgbaColors = keys.reduce((map: Map<number, integer[]>, k: number) => { map.set(k, Object.values(rgbaFromArgb(k))); return map; }, new Map<number, integer[]>());
hsvColors = Array.from(rgbaColors.keys()).reduce((map: Map<number, number[]>, k: number) => {
const rgb = rgbaColors.get(k).slice(0, 3);
map.set(k, Utils.rgbToHsv(rgb[0], rgb[1], rgb[2]));
return map;
}, new Map<number, number[]>());
for (let c = keys.length - 1; c >= 0; c--) {
const hsv = hsvColors.get(keys[c]);
for (let c2 = 0; c2 < c; c2++) {
const hsv2 = hsvColors.get(keys[c2]);
const diff = Math.abs(hsv[0] - hsv2[0]);
if (diff < 30 || diff >= 330) {
2023-11-23 23:52:13 -05:00
if (mappedColors.has(keys[c]))
mappedColors.get(keys[c]).push(keys[c2]);
else
mappedColors.set(keys[c], [ keys[c2] ]);
break;
}
}
}
mappedColors.forEach((values: integer[], key: integer) => {
const keyColor = rgbaColors.get(key);
const valueColors = values.map(v => rgbaColors.get(v));
let color = keyColor.slice(0);
let count = paletteColors.get(key);
for (let value of values) {
const valueCount = paletteColors.get(value);
if (!valueCount)
continue;
count += valueCount;
}
for (let c = 0; c < 3; c++) {
color[c] *= (paletteColors.get(key) / count);
values.forEach((value: integer, i: integer) => {
if (paletteColors.has(value)) {
const valueCount = paletteColors.get(value);
color[c] += valueColors[i][c] * (valueCount / count);
}
});
color[c] = Math.round(color[c]);
}
paletteColors.delete(key);
for (let value of values) {
paletteColors.delete(value);
if (mappedColors.has(value))
mappedColors.delete(value);
}
paletteColors.set(argbFromRgba({ r: color[0], g: color[1], b: color[2], a: color[3] }), count);
});
keys = Array.from(paletteColors.keys()).sort((a: integer, b: integer) => paletteColors.get(a) < paletteColors.get(b) ? 1 : -1);
} while (mappedColors.size);
return keys.map(c => Object.values(rgbaFromArgb(c)))
}
);
const paletteDeltas: number[][] = [];
spriteColors.forEach((sc: integer[], i: integer) => {
paletteDeltas.push([]);
for (let p = 0; p < palette.length; p++)
paletteDeltas[i].push(Utils.deltaRgb(sc, palette[p]));
});
const easeFunc = Phaser.Tweens.Builders.GetEaseFunction('Cubic.easeIn');
for (let sc = 0; sc < spriteColors.length; sc++) {
const delta = Math.min(...paletteDeltas[sc]);
const paletteIndex = Math.min(paletteDeltas[sc].findIndex(pd => pd === delta), fusionPalette.length - 1);
if (delta < 255) {
const ratio = easeFunc(delta / 255);
let color = [ 0, 0, 0, fusionSpriteColors[sc][3] ];
for (let c = 0; c < 3; c++)
color[c] = Math.round((fusionSpriteColors[sc][c] * ratio) + (fusionPalette[paletteIndex][c] * (1 - ratio)));
fusionSpriteColors[sc] = color;
}
}
[ this.getSprite(), this.getTintSprite() ].map(s => {
s.pipelineData[`spriteColors${ignoreOveride && this.summonData?.speciesForm ? 'Base' : ''}`] = spriteColors;
s.pipelineData[`fusionSpriteColors${ignoreOveride && this.summonData?.speciesForm ? 'Base' : ''}`] = fusionSpriteColors;
2023-11-23 23:52:13 -05:00
});
canvas.remove();
fusionCanvas.remove();
}
randSeedInt(range: integer, min: integer = 0): integer {
return this.scene.currentBattle
? this.scene.currentBattle.randSeedInt(range, min)
: Utils.randSeedInt(range, min);
}
randSeedIntRange(min: integer, max: integer): integer {
return this.randSeedInt((max - min) + 1, min);
}
destroy(): void {
this.battleInfo?.destroy();
super.destroy();
}
2023-03-28 14:54:52 -04:00
}
2023-04-14 18:21:33 -04:00
export default interface Pokemon {
scene: BattleScene
}
2023-03-28 14:54:52 -04:00
export class PlayerPokemon extends Pokemon {
2023-04-08 00:21:44 -04:00
public compatibleTms: Moves[];
2024-01-07 23:17:24 -05:00
constructor(scene: BattleScene, species: PokemonSpecies, level: integer, abilityIndex: integer, formIndex: integer, gender: Gender, shiny: boolean, ivs: integer[], nature: Nature, dataSource: Pokemon | PokemonData) {
2024-01-05 22:24:05 -05:00
super(scene, 106, 148, species, level, abilityIndex, formIndex, gender, shiny, ivs, nature, dataSource);
2023-04-08 00:21:44 -04:00
this.generateCompatibleTms();
2023-03-28 14:54:52 -04:00
}
2024-01-07 23:17:24 -05:00
initBattleInfo(): void {
this.battleInfo = new PlayerBattleInfo(this.scene);
this.battleInfo.initInfo(this);
}
2023-04-08 00:21:44 -04:00
isPlayer(): boolean {
2023-03-28 14:54:52 -04:00
return true;
}
hasTrainer(): boolean {
return true;
}
2024-01-07 23:17:24 -05:00
isBoss(): boolean {
return false;
}
getFieldIndex(): integer {
return this.scene.getPlayerField().indexOf(this);
}
getBattlerIndex(): BattlerIndex {
return this.getFieldIndex();
}
2023-04-08 00:21:44 -04:00
generateCompatibleTms(): void {
this.compatibleTms = [];
const tms = Object.keys(tmSpecies);
for (let tm of tms) {
const moveId = parseInt(tm) as Moves;
let compatible = false;
2023-04-08 00:21:44 -04:00
for (let p of tmSpecies[tm]) {
if (Array.isArray(p)) {
2023-12-09 20:09:09 -05:00
if (p[0] === this.species.speciesId || (this.fusionSpecies && p[0] === this.fusionSpecies.speciesId) && p.slice(1).indexOf(this.species.forms[this.formIndex]) > -1) {
compatible = true;
2023-04-08 00:21:44 -04:00
break;
}
} else if (p === this.species.speciesId || (this.fusionSpecies && p === this.fusionSpecies.speciesId)) {
compatible = true;
2023-04-08 00:21:44 -04:00
break;
}
}
if (reverseCompatibleTms.indexOf(moveId) > -1)
compatible = !compatible;
if (compatible)
this.compatibleTms.push(moveId);
2023-04-08 00:21:44 -04:00
}
}
2023-04-10 13:54:06 -04:00
tryPopulateMoveset(moveset: StarterMoveset): boolean {
2024-02-25 12:45:41 -05:00
if (!this.getSpeciesForm().validateStarterMoveset(moveset, this.scene.gameData.starterEggMoveData[this.species.getRootSpeciesId()]))
return false;
this.moveset = moveset.map(m => new PokemonMove(m));
return true;
}
2023-10-31 14:09:33 -04:00
switchOut(batonPass: boolean): Promise<void> {
return new Promise(resolve => {
this.resetTurnData();
this.resetSummonData();
this.hideInfo();
this.setVisible(false);
this.scene.ui.setMode(Mode.PARTY, PartyUiMode.FAINT_SWITCH, this.getFieldIndex(), (slotIndex: integer, option: PartyOption) => {
if (slotIndex >= this.scene.currentBattle.getBattlerCount() && slotIndex < 6)
this.scene.unshiftPhase(new SwitchSummonPhase(this.scene, this.getFieldIndex(), slotIndex, false, batonPass));
this.scene.ui.setMode(Mode.MESSAGE).then(() => resolve());
}, PartyUiHandler.FilterNonFainted);
});
}
2023-12-14 11:54:56 -05:00
getPossibleEvolution(evolution: SpeciesFormEvolution): Promise<Pokemon> {
return new Promise(resolve => {
const species = getPokemonSpecies(evolution.speciesId);
const formIndex = evolution.evoFormKey !== null ? Math.max(this.species.forms.findIndex(f => f.formKey === evolution.evoFormKey), 0) : this.formIndex;
2024-01-07 23:17:24 -05:00
const ret = this.scene.addPlayerPokemon(species, this.level, this.abilityIndex, formIndex, this.gender, this.shiny, this.ivs, this.nature, this);
ret.loadAssets().then(() => resolve(ret));
});
2023-12-14 11:54:56 -05:00
}
2023-10-31 14:09:33 -04:00
evolve(evolution: SpeciesFormEvolution): Promise<void> {
2023-04-10 13:54:06 -04:00
return new Promise(resolve => {
this.pauseEvolutions = false;
2023-04-14 18:21:33 -04:00
this.handleSpecialEvolutions(evolution);
2023-04-10 13:54:06 -04:00
this.species = getPokemonSpecies(evolution.speciesId);
if (evolution.preFormKey !== null)
this.formIndex = Math.max(this.species.forms.findIndex(f => f.formKey === evolution.evoFormKey), 0);
this.generateName();
2023-10-20 14:19:22 -04:00
const abilityCount = this.getSpeciesForm().getAbilityCount();
2023-04-26 12:50:21 -04:00
if (this.abilityIndex >= abilityCount) // Shouldn't happen
this.abilityIndex = abilityCount - 1;
2023-04-10 13:54:06 -04:00
this.compatibleTms.splice(0, this.compatibleTms.length);
this.generateCompatibleTms();
if (!this.scene.gameMode.isDaily || this.metBiome > -1) {
this.scene.gameData.setPokemonSeen(this, false);
this.scene.gameData.setPokemonCaught(this, false);
}
2023-04-10 13:54:06 -04:00
this.loadAssets().then(() => {
this.calculateStats();
this.updateInfo(true).then(() => resolve());
2023-04-10 13:54:06 -04:00
});
});
}
2023-04-14 18:21:33 -04:00
private handleSpecialEvolutions(evolution: SpeciesFormEvolution) {
2023-04-14 18:21:33 -04:00
if (this.species.speciesId === Species.NINCADA && evolution.speciesId === Species.NINJASK) {
const newEvolution = pokemonEvolutions[this.species.speciesId][1];
if (newEvolution.condition.predicate(this)) {
2024-01-07 23:17:24 -05:00
const newPokemon = this.scene.addPlayerPokemon(this.species, this.level, this.abilityIndex, this.formIndex, this.gender, this.shiny, this.ivs, this.nature);
2023-04-14 18:21:33 -04:00
this.scene.getParty().push(newPokemon);
newPokemon.evolve(newEvolution);
2023-10-24 18:44:38 -04:00
const modifiers = this.scene.findModifiers(m => m instanceof PokemonHeldItemModifier
&& (m as PokemonHeldItemModifier).pokemonId === this.id, true) as PokemonHeldItemModifier[];
modifiers.forEach(m => {
const clonedModifier = m.clone() as PokemonHeldItemModifier;
clonedModifier.pokemonId = newPokemon.id;
this.scene.addModifier(clonedModifier, true);
2023-10-24 18:44:38 -04:00
});
this.scene.updateModifiers(true);
2023-04-14 18:21:33 -04:00
}
}
}
2023-11-04 00:32:12 -04:00
2024-01-09 23:34:43 -05:00
getPossibleForm(formChange: SpeciesFormChange): Promise<Pokemon> {
return new Promise(resolve => {
const formIndex = Math.max(this.species.forms.findIndex(f => f.formKey === formChange.formKey), 0);
const ret = this.scene.addPlayerPokemon(this.species, this.level, this.abilityIndex, formIndex, this.gender, this.shiny, this.ivs, this.nature, this);
ret.loadAssets().then(() => resolve(ret));
});
}
changeForm(formChange: SpeciesFormChange): Promise<void> {
return new Promise(resolve => {
this.formIndex = Math.max(this.species.forms.findIndex(f => f.formKey === formChange.formKey), 0);
this.generateName();
const abilityCount = this.getSpeciesForm().getAbilityCount();
if (this.abilityIndex >= abilityCount) // Shouldn't happen
this.abilityIndex = abilityCount - 1;
this.compatibleTms.splice(0, this.compatibleTms.length);
this.generateCompatibleTms();
if (!this.scene.gameMode.isDaily || this.metBiome > -1) {
this.scene.gameData.setPokemonSeen(this, false);
this.scene.gameData.setPokemonCaught(this, false);
}
2024-01-09 23:34:43 -05:00
this.loadAssets().then(() => {
this.calculateStats();
this.scene.updateModifiers(true, true);
this.updateInfo(true).then(() => resolve());
2024-01-09 23:34:43 -05:00
});
});
}
2023-11-04 00:32:12 -04:00
isFusion(): boolean {
return !!(this.fusionSpecies || (this.species.speciesId === Species.KYUREM && this.formIndex));
}
clearFusionSpecies(): void {
super.clearFusionSpecies();
this.generateCompatibleTms();
}
2023-11-04 00:32:12 -04:00
fuse(pokemon: PlayerPokemon): Promise<void> {
return new Promise(resolve => {
2024-02-27 21:34:21 -05:00
this.fusionSpecies = pokemon.species;
this.fusionFormIndex = pokemon.formIndex;
this.fusionAbilityIndex = pokemon.abilityIndex;
this.fusionShiny = pokemon.shiny;
this.fusionGender = pokemon.gender;
2023-11-04 00:32:12 -04:00
this.scene.validateAchv(achvs.SPLICE);
this.scene.gameData.gameStats.pokemonFused++;
this.generateName();
2023-11-04 00:32:12 -04:00
this.calculateStats();
this.generateCompatibleTms();
2023-11-04 00:32:12 -04:00
this.updateInfo(true).then(() => {
const fusedPartyMemberIndex = this.scene.getParty().indexOf(pokemon);
let partyMemberIndex = this.scene.getParty().indexOf(this);
if (partyMemberIndex > fusedPartyMemberIndex)
partyMemberIndex--;
pokemon.getMoveset(true).map(m => this.scene.unshiftPhase(new LearnMovePhase(this.scene, partyMemberIndex, m.getMove().id)));
2023-11-04 00:32:12 -04:00
const fusedPartyMemberHeldModifiers = this.scene.findModifiers(m => m instanceof PokemonHeldItemModifier
&& (m as PokemonHeldItemModifier).pokemonId === pokemon.id, true) as PokemonHeldItemModifier[];
const transferModifiers: Promise<boolean>[] = [];
for (let modifier of fusedPartyMemberHeldModifiers)
transferModifiers.push(this.scene.tryTransferHeldItemModifier(modifier, this, true, false, true, true));
2023-11-04 00:32:12 -04:00
Promise.allSettled(transferModifiers).then(() => {
this.scene.updateModifiers(true, true).then(() => {
this.scene.removePartyMemberModifiers(fusedPartyMemberIndex);
this.scene.getParty().splice(fusedPartyMemberIndex, 1)[0];
pokemon.destroy();
this.updateFusionPalette();
resolve();
});
2023-11-04 00:32:12 -04:00
});
});
});
}
unfuse(): Promise<void> {
return new Promise(resolve => {
this.clearFusionSpecies();
2023-11-04 00:32:12 -04:00
this.updateInfo(true).then(() => resolve());
2023-11-23 23:52:13 -05:00
this.updateFusionPalette();
2023-11-04 00:32:12 -04:00
});
}
2023-03-28 14:54:52 -04:00
}
export class EnemyPokemon extends Pokemon {
public trainerSlot: TrainerSlot;
2023-03-28 14:54:52 -04:00
public aiType: AiType;
2024-01-07 23:17:24 -05:00
public bossSegments: integer;
public bossSegmentIndex: integer;
2023-03-28 14:54:52 -04:00
constructor(scene: BattleScene, species: PokemonSpecies, level: integer, trainerSlot: TrainerSlot, boss: boolean, dataSource: PokemonData) {
super(scene, 236, 84, species, level, dataSource?.abilityIndex, dataSource?.formIndex,
2024-01-05 22:24:05 -05:00
dataSource?.gender, dataSource ? dataSource.shiny : false, null, dataSource ? dataSource.nature : undefined, dataSource);
2023-04-28 15:03:42 -04:00
this.trainerSlot = trainerSlot;
2024-01-07 23:17:24 -05:00
if (boss)
this.setBoss();
2023-04-28 15:03:42 -04:00
if (!dataSource) {
this.trySetShiny();
2023-04-28 15:03:42 -04:00
let prevolution: Species;
let speciesId = species.speciesId;
while ((prevolution = pokemonPrevolutions[speciesId])) {
2023-12-07 17:43:56 -05:00
const evolution = pokemonEvolutions[prevolution].find(pe => pe.speciesId === speciesId && (!pe.evoFormKey || pe.evoFormKey === this.getFormKey()));
2023-04-28 15:03:42 -04:00
if (evolution.condition?.enforceFunc)
evolution.condition.enforceFunc(this);
speciesId = prevolution;
}
2023-04-18 15:07:10 -04:00
}
2023-03-29 00:31:25 -04:00
this.aiType = AiType.SMART_RANDOM;
2023-03-28 14:54:52 -04:00
}
2024-01-07 23:17:24 -05:00
initBattleInfo(): void {
if (!this.battleInfo) {
this.battleInfo = new EnemyBattleInfo(this.scene);
this.battleInfo.updateBossSegments(this);
this.battleInfo.initInfo(this);
} else
this.battleInfo.updateBossSegments(this);
2024-01-07 23:17:24 -05:00
}
2024-01-07 23:17:24 -05:00
setBoss(boss: boolean = true): void {
if (boss) {
this.bossSegments = this.scene.getEncounterBossSegments(this.scene.currentBattle.waveIndex, this.level, this.species, true);
this.bossSegmentIndex = this.bossSegments - 1;
} else {
this.bossSegments = 0;
this.bossSegmentIndex = 0;
}
}
generateAndPopulateMoveset(formIndex?: integer): void {
2023-04-29 01:40:24 -04:00
switch (true) {
2023-12-25 20:43:56 -05:00
case (this.species.speciesId === Species.SMEARGLE):
this.moveset = [
new PokemonMove(Moves.SKETCH),
new PokemonMove(Moves.SKETCH),
new PokemonMove(Moves.SKETCH),
new PokemonMove(Moves.SKETCH)
];
break;
2023-04-29 01:40:24 -04:00
case (this.species.speciesId === Species.ETERNATUS):
this.moveset = (formIndex !== undefined ? formIndex : this.formIndex)
2023-12-16 00:46:01 -05:00
? [
new PokemonMove(Moves.DYNAMAX_CANNON),
new PokemonMove(Moves.CROSS_POISON),
2023-12-16 00:46:01 -05:00
new PokemonMove(Moves.FLAMETHROWER),
2024-03-24 19:52:30 -04:00
new PokemonMove(Moves.RECOVER, 0, -4)
2023-12-16 00:46:01 -05:00
]
: [
new PokemonMove(Moves.ETERNABEAM),
new PokemonMove(Moves.SLUDGE_BOMB),
2023-12-16 00:46:01 -05:00
new PokemonMove(Moves.DRAGON_DANCE),
new PokemonMove(Moves.COSMIC_POWER)
2023-12-16 00:46:01 -05:00
];
2023-04-29 01:40:24 -04:00
break;
default:
super.generateAndPopulateMoveset();
break;
}
}
getNextMove(): QueuedMove {
2023-04-20 21:32:48 -04:00
const queuedMove = this.getMoveQueue().length
? this.getMoveset().find(m => m.moveId === this.getMoveQueue()[0].move)
2023-04-13 12:16:36 -04:00
: null;
2023-04-19 18:19:55 -04:00
if (queuedMove) {
2023-10-25 09:41:37 -04:00
if (queuedMove.isUsable(this, this.getMoveQueue()[0].ignorePP))
return { move: queuedMove.moveId, targets: this.getMoveQueue()[0].targets, ignorePP: this.getMoveQueue()[0].ignorePP };
2023-04-19 18:19:55 -04:00
else {
2023-04-20 21:32:48 -04:00
this.getMoveQueue().shift();
2023-04-19 18:19:55 -04:00
return this.getNextMove();
}
}
2023-04-13 12:16:36 -04:00
2023-10-25 09:41:37 -04:00
const movePool = this.getMoveset().filter(m => m.isUsable(this));
2023-03-28 14:54:52 -04:00
if (movePool.length) {
if (movePool.length === 1)
return { move: movePool[0].moveId, targets: this.getNextTargets(movePool[0].moveId) };
2023-11-16 00:58:57 -05:00
const encoreTag = this.getTag(EncoreTag) as EncoreTag;
if (encoreTag) {
const encoreMove = movePool.find(m => m.moveId === encoreTag.moveId);
if (encoreMove)
return { move: encoreMove.moveId, targets: this.getNextTargets(encoreMove.moveId) };
}
2023-03-28 14:54:52 -04:00
switch (this.aiType) {
case AiType.RANDOM:
const moveId = movePool[this.scene.currentBattle.randSeedInt(movePool.length)].moveId;
return { move: moveId, targets: this.getNextTargets(moveId) };
2023-03-29 00:31:25 -04:00
case AiType.SMART_RANDOM:
2023-03-28 14:54:52 -04:00
case AiType.SMART:
const moveScores = movePool.map(() => 0);
const moveTargets = Object.fromEntries(movePool.map(m => [ m.moveId, this.getNextTargets(m.moveId) ]));
2023-03-28 14:54:52 -04:00
for (let m in movePool) {
const pokemonMove = movePool[m];
const move = pokemonMove.getMove();
2023-04-10 23:15:06 -04:00
let moveScore = moveScores[m];
let targetScores: integer[] = [];
for (let mt of moveTargets[move.id]) {
const target = this.scene.getField()[mt];
let targetScore = move.getUserBenefitScore(this, target, move) + move.getTargetBenefitScore(this, target, move) * (mt < BattlerIndex.ENEMY === this.isPlayer() ? 1 : -1);
if (mt !== this.getBattlerIndex())
targetScore *= target.getAttackMoveEffectiveness(this, pokemonMove);
targetScores.push(targetScore);
2023-04-10 23:15:06 -04:00
}
moveScore += Math.max(...targetScores);
2023-04-10 23:15:06 -04:00
// could make smarter by checking opponent def/spdef
moveScores[m] = moveScore;
2023-03-28 14:54:52 -04:00
}
2023-03-29 00:31:25 -04:00
console.log(moveScores);
2023-03-28 14:54:52 -04:00
const sortedMovePool = movePool.slice(0);
sortedMovePool.sort((a, b) => {
const scoreA = moveScores[movePool.indexOf(a)];
const scoreB = moveScores[movePool.indexOf(b)];
return scoreA < scoreB ? 1 : scoreA > scoreB ? -1 : 0;
});
let r = 0;
2023-03-29 00:31:25 -04:00
if (this.aiType === AiType.SMART_RANDOM) {
while (r < sortedMovePool.length - 1 && this.scene.currentBattle.randSeedInt(8) >= 5)
2023-03-29 00:31:25 -04:00
r++;
}
2023-03-28 14:54:52 -04:00
console.log(movePool.map(m => m.getName()), moveScores, r, sortedMovePool.map(m => m.getName()));
return { move: sortedMovePool[r].moveId, targets: moveTargets[sortedMovePool[r].moveId] };
2023-03-28 14:54:52 -04:00
}
}
2023-04-19 18:19:55 -04:00
return { move: Moves.STRUGGLE, targets: this.getNextTargets(Moves.STRUGGLE) };
}
getNextTargets(moveId: Moves): BattlerIndex[] {
const moveTargets = getMoveTargets(this, moveId);
const targets = this.scene.getField(true).filter(p => moveTargets.targets.indexOf(p.getBattlerIndex()) > -1);
if (moveTargets.multiple)
return targets.map(p => p.getBattlerIndex());
const move = allMoves[moveId];
const benefitScores = targets
.map(p => [ p.getBattlerIndex(), move.getTargetBenefitScore(this, p, move) * (p.isPlayer() === this.isPlayer() ? 1 : -1) ]);
const sortedBenefitScores = benefitScores.slice(0);
sortedBenefitScores.sort((a, b) => {
const scoreA = a[1];
const scoreB = b[1];
return scoreA < scoreB ? 1 : scoreA > scoreB ? -1 : 0;
});
2023-05-23 10:44:33 -04:00
if (!sortedBenefitScores.length)
return [];
let targetWeights = sortedBenefitScores.map(s => s[1]);
const lowestWeight = targetWeights[targetWeights.length - 1];
if (lowestWeight < 1) {
for (let w = 0; w < targetWeights.length; w++)
targetWeights[w] += Math.abs(lowestWeight - 1);
}
const benefitCutoffIndex = targetWeights.findIndex(s => s < targetWeights[0] / 2);
if (benefitCutoffIndex > -1)
targetWeights = targetWeights.slice(0, benefitCutoffIndex);
const thresholds: integer[] = [];
let totalWeight: integer;
targetWeights.reduce((total: integer, w: integer) => {
total += w;
thresholds.push(total);
totalWeight = total;
return total;
}, 0);
const randValue = this.scene.currentBattle.randSeedInt(totalWeight);
let targetIndex: integer;
thresholds.every((t, i) => {
if (randValue >= t)
return true;
targetIndex = i;
return false;
});
return [ sortedBenefitScores[targetIndex][0] ];
2023-03-28 14:54:52 -04:00
}
isPlayer() {
return false;
}
hasTrainer(): boolean {
return !!this.trainerSlot;
}
2024-01-07 23:17:24 -05:00
isBoss(): boolean {
return !!this.bossSegments;
}
getBossSegmentIndex(): integer {
const segments = (this as EnemyPokemon).bossSegments;
const segmentSize = this.getMaxHp() / segments;
for (let s = segments - 1; s > 0; s--) {
const hpThreshold = Math.round(segmentSize * s);
if (this.hp > hpThreshold) {
return s;
}
}
return 0;
}
damage(damage: integer, ignoreSegments: boolean = false, preventEndure: boolean = false): integer {
if (this.isFainted())
return 0;
let clearedBossSegmentIndex = this.isBoss()
? this.bossSegmentIndex + 1
: 0;
2024-03-01 11:35:13 -05:00
if (this.isBoss() && !ignoreSegments) {
const segmentSize = this.getMaxHp() / this.bossSegments;
for (let s = this.bossSegmentIndex; s > 0; s--) {
const hpThreshold = segmentSize * s;
const roundedHpThreshold = Math.round(hpThreshold);
if (this.hp >= roundedHpThreshold) {
if (this.hp - damage <= roundedHpThreshold) {
const hpRemainder = this.hp - roundedHpThreshold;
let segmentsBypassed = 0;
while (segmentsBypassed < this.bossSegmentIndex && this.canBypassBossSegments(segmentsBypassed + 1) && (damage - hpRemainder) >= Math.round(segmentSize * Math.pow(2, segmentsBypassed + 1))) {
segmentsBypassed++;
//console.log('damage', damage, 'segment', segmentsBypassed + 1, 'segment size', segmentSize, 'damage needed', Math.round(segmentSize * Math.pow(2, segmentsBypassed + 1)));
2024-02-20 00:02:44 -05:00
}
2024-03-01 11:35:13 -05:00
damage = hpRemainder + Math.round(segmentSize * segmentsBypassed);
clearedBossSegmentIndex = s - segmentsBypassed;
2024-01-07 23:17:24 -05:00
}
2024-03-01 11:35:13 -05:00
break;
2024-01-07 23:17:24 -05:00
}
}
}
const ret = super.damage(damage, ignoreSegments, preventEndure);
2024-03-01 11:35:13 -05:00
if (this.isBoss()) {
if (ignoreSegments) {
const segmentSize = this.getMaxHp() / this.bossSegments;
clearedBossSegmentIndex = Math.ceil(this.hp / segmentSize);
}
if (clearedBossSegmentIndex <= this.bossSegmentIndex)
this.handleBossSegmentCleared(clearedBossSegmentIndex);
2024-03-01 11:35:13 -05:00
this.battleInfo.updateBossSegments(this);
}
return ret;
2024-01-07 23:17:24 -05:00
}
canBypassBossSegments(segmentCount: integer = 1): boolean {
if (this.scene.currentBattle.battleSpec === BattleSpec.FINAL_BOSS) {
if (!this.formIndex && (this.bossSegmentIndex - segmentCount) < 1)
return false;
}
return true;
}
2024-01-07 23:17:24 -05:00
handleBossSegmentCleared(segmentIndex: integer): void {
while (segmentIndex - 1 < this.bossSegmentIndex) {
let boostedStat = BattleStat.RAND;
const battleStats = Utils.getEnumValues(BattleStat).slice(0, -2);
const statWeights = new Array().fill(battleStats.length).filter((bs: BattleStat) => this.summonData.battleStats[bs] < 6).map((bs: BattleStat) => this.getStat(bs + 1));
const statThresholds: integer[] = [];
let totalWeight = 0;
for (let bs of battleStats) {
totalWeight += statWeights[bs];
statThresholds.push(totalWeight);
}
const randInt = Utils.randSeedInt(totalWeight);
for (let bs of battleStats) {
if (randInt < statThresholds[bs]) {
boostedStat = bs;
break;
}
}
let statLevels = 1;
switch (segmentIndex) {
case 1:
if (this.bossSegments >= 3)
statLevels++;
break;
case 2:
if (this.bossSegments >= 5)
statLevels++;
break;
}
this.scene.unshiftPhase(new StatChangePhase(this.scene, this.getBattlerIndex(), true, [ boostedStat ], statLevels));
this.bossSegmentIndex--;
}
}
heal(amount: integer): integer {
if (this.isBoss()) {
let amountRatio = amount / this.getMaxHp();
let segmentBypassCount = Math.floor(amountRatio / (1 / this.bossSegments));
const segmentSize = this.getMaxHp() / this.bossSegments;
for (let s = 1; s < this.bossSegments; s++) {
const hpThreshold = segmentSize * s;
if (this.hp <= Math.round(hpThreshold)) {
2024-01-07 23:17:24 -05:00
const healAmount = Math.min(amount, this.getMaxHp() - this.hp, Math.round(hpThreshold + (segmentSize * segmentBypassCount) - this.hp));
this.hp += healAmount;
return healAmount;
} else if (s >= this.bossSegmentIndex)
return super.heal(amount);
2024-01-07 23:17:24 -05:00
}
}
return super.heal(amount);
}
getFieldIndex(): integer {
return this.scene.getEnemyField().indexOf(this);
}
getBattlerIndex(): BattlerIndex {
return BattlerIndex.ENEMY + this.getFieldIndex();
}
addToParty(pokeballType: PokeballType) {
2023-04-14 18:21:33 -04:00
const party = this.scene.getParty();
2023-03-31 20:19:57 -04:00
let ret: PlayerPokemon = null;
2023-03-28 14:54:52 -04:00
2023-03-31 20:19:57 -04:00
if (party.length < 6) {
this.pokeball = pokeballType;
2024-01-05 11:29:34 -05:00
this.metLevel = this.level;
this.metBiome = this.scene.arena.biomeType;
2024-01-07 23:17:24 -05:00
const newPokemon = this.scene.addPlayerPokemon(this.species, this.level, this.abilityIndex, this.formIndex, this.gender, this.shiny, this.ivs, this.nature, this);
2023-03-31 20:19:57 -04:00
party.push(newPokemon);
ret = newPokemon;
2024-01-09 23:53:31 -05:00
this.scene.triggerPokemonFormChange(newPokemon, SpeciesFormChangeActiveTrigger, true);
2023-03-31 20:19:57 -04:00
}
2023-03-31 22:31:20 -04:00
2023-03-31 20:19:57 -04:00
return ret;
2023-03-28 14:54:52 -04:00
}
}
2023-04-13 12:16:36 -04:00
export interface TurnMove {
move: Moves;
targets?: BattlerIndex[];
2023-04-13 12:16:36 -04:00
result: MoveResult;
2023-04-19 18:19:55 -04:00
virtual?: boolean;
2023-04-25 01:32:48 -04:00
turn?: integer;
2023-04-13 12:16:36 -04:00
}
export interface QueuedMove {
move: Moves;
targets: BattlerIndex[];
2023-04-13 12:16:36 -04:00
ignorePP?: boolean;
}
export interface AttackMoveResult {
move: Moves;
result: DamageResult;
damage: integer;
2023-10-26 21:12:53 -07:00
critical: boolean;
sourceId: integer;
}
2023-04-03 23:38:31 -04:00
export class PokemonSummonData {
2023-04-10 23:15:06 -04:00
public battleStats: integer[] = [ 0, 0, 0, 0, 0, 0, 0 ];
2023-04-13 12:16:36 -04:00
public moveQueue: QueuedMove[] = [];
2023-10-25 09:41:37 -04:00
public disabledMove: Moves = Moves.NONE;
public disabledTurns: integer = 0;
2023-04-21 19:30:04 -04:00
public tags: BattlerTag[] = [];
public speciesForm: PokemonSpeciesForm;
public fusionSpeciesForm: PokemonSpeciesForm;
public ability: Abilities = Abilities.NONE;
public gender: Gender;
public fusionGender: Gender;
public stats: integer[];
public moveset: PokemonMove[];
2023-04-21 19:30:04 -04:00
public types: Type[];
2023-04-03 23:38:31 -04:00
}
2023-11-28 21:35:52 -05:00
export class PokemonBattleData {
public hitCount: integer = 0;
public revived: boolean = false;
public maxRevived: boolean = false;
2023-11-28 21:35:52 -05:00
}
2023-04-03 23:38:31 -04:00
export class PokemonBattleSummonData {
public turnCount: integer = 1;
2023-04-28 15:03:42 -04:00
public moveHistory: TurnMove[] = [];
2023-04-03 23:38:31 -04:00
}
export class PokemonTurnData {
public flinched: boolean;
2023-04-10 16:17:25 -04:00
public hitCount: integer;
2023-04-03 23:38:31 -04:00
public hitsLeft: integer;
2023-04-14 01:08:44 -04:00
public damageDealt: integer = 0;
public attacksReceived: AttackMoveResult[] = [];
2023-04-03 23:38:31 -04:00
}
2023-03-29 00:31:25 -04:00
export enum AiType {
2023-03-28 14:54:52 -04:00
RANDOM,
2023-03-29 00:31:25 -04:00
SMART_RANDOM,
2023-03-28 14:54:52 -04:00
SMART
2023-04-29 01:40:24 -04:00
}
2023-03-28 14:54:52 -04:00
export enum MoveResult {
PENDING,
SUCCESS,
FAIL,
MISS,
OTHER
}
export enum HitResult {
EFFECTIVE = 1,
2023-03-28 14:54:52 -04:00
SUPER_EFFECTIVE,
NOT_VERY_EFFECTIVE,
2023-11-07 22:23:42 -05:00
ONE_HIT_KO,
2023-12-22 01:16:56 -05:00
NO_EFFECT,
2023-04-13 12:16:36 -04:00
STATUS,
HEAL,
FAIL,
MISS,
2023-03-28 14:54:52 -04:00
OTHER
2023-04-29 01:40:24 -04:00
}
2023-03-28 14:54:52 -04:00
2023-11-07 22:23:42 -05:00
export type DamageResult = HitResult.EFFECTIVE | HitResult.SUPER_EFFECTIVE | HitResult.NOT_VERY_EFFECTIVE | HitResult.ONE_HIT_KO | HitResult.OTHER;
2023-04-15 01:32:16 -04:00
2023-03-28 14:54:52 -04:00
export class PokemonMove {
2023-04-13 12:16:36 -04:00
public moveId: Moves;
2023-03-28 14:54:52 -04:00
public ppUsed: integer;
public ppUp: integer;
2023-04-19 18:19:55 -04:00
public virtual: boolean;
2023-03-28 14:54:52 -04:00
2023-04-19 18:19:55 -04:00
constructor(moveId: Moves, ppUsed?: integer, ppUp?: integer, virtual?: boolean) {
2023-03-28 14:54:52 -04:00
this.moveId = moveId;
2023-04-13 23:04:51 -04:00
this.ppUsed = ppUsed || 0;
this.ppUp = ppUp || 0;
2023-04-19 18:19:55 -04:00
this.virtual = !!virtual;
2023-03-28 14:54:52 -04:00
}
2023-10-25 09:41:37 -04:00
isUsable(pokemon: Pokemon, ignorePp?: boolean): boolean {
2023-11-27 13:47:32 -05:00
if (this.moveId && pokemon.summonData?.disabledMove === this.moveId)
2023-03-28 14:54:52 -04:00
return false;
2024-01-12 19:05:00 -05:00
return ignorePp || this.ppUsed < this.getMovePp() || this.getMove().pp === -1;
2023-04-19 18:19:55 -04:00
}
2023-03-28 14:54:52 -04:00
getMove(): Move {
2023-04-20 21:32:48 -04:00
return allMoves[this.moveId];
2023-03-28 14:54:52 -04:00
}
2024-01-12 19:05:00 -05:00
getMovePp(): integer {
return this.getMove().pp + this.ppUp * Math.max(Math.floor(this.getMove().pp / 5), 1);
}
2023-04-28 00:25:33 -04:00
getPpRatio(): number {
2024-01-12 19:05:00 -05:00
return 1 - (this.ppUsed / this.getMovePp());
2023-04-28 00:25:33 -04:00
}
2023-03-28 14:54:52 -04:00
getName(): string {
return this.getMove().name;
2023-03-28 14:54:52 -04:00
}
}