pokerogue/src/pokemon.ts

1070 lines
35 KiB
TypeScript
Raw Normal View History

2023-03-28 11:54:52 -07:00
import Phaser from 'phaser';
import BattleScene from './battle-scene';
2023-04-20 12:46:05 -07:00
import BattleInfo, { PlayerBattleInfo, EnemyBattleInfo } from './ui/battle-info';
import Move, { StatChangeAttr, HighCritAttr, HitsTagAttr, applyMoveAttrs, FixedDamageAttr, VariablePowerAttr, Moves, allMoves, MoveCategory } from "./data/move";
import { pokemonLevelMoves } from './data/pokemon-level-moves';
import { default as PokemonSpecies, PokemonSpeciesForm, getPokemonSpecies } from './data/pokemon-species';
2023-03-28 11:54:52 -07:00
import * as Utils from './utils';
2023-04-20 12:46:05 -07:00
import { Type, getTypeDamageMultiplier } from './data/type';
import { getLevelTotalExp } from './data/exp';
import { Stat } from './data/pokemon-stat';
import { AttackTypeBoosterModifier, PokemonBaseStatModifier, ShinyRateBoosterModifier, TempBattleStatBoosterModifier } from './modifier/modifier';
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 { tmSpecies } from './data/tms';
import { pokemonEvolutions, pokemonPrevolutions, SpeciesEvolution, SpeciesEvolutionCondition } from './data/pokemon-evolutions';
2023-04-16 15:40:32 -07:00
import { DamagePhase, FaintPhase, MessagePhase } from './battle-phases';
2023-04-20 12:46:05 -07:00
import { BattleStat } from './data/battle-stat';
import { BattleTag, BattleTagLapseType, BattleTagType, getBattleTag } from './data/battle-tag';
import { Species } from './data/species';
import { WeatherType } from './data/weather';
import { TempBattleStat } from './data/temp-battle-stat';
2023-03-28 11:54:52 -07:00
export default abstract class Pokemon extends Phaser.GameObjects.Container {
public id: integer;
public name: string;
public species: PokemonSpecies;
public formIndex: integer;
2023-03-28 11:54:52 -07:00
public shiny: boolean;
public pokeball: PokeballType;
protected battleInfo: BattleInfo;
public level: integer;
public exp: integer;
public levelExp: integer;
2023-04-01 17:06:44 -07:00
public gender: Gender;
2023-03-28 11:54:52 -07:00
public hp: integer;
public stats: integer[];
public ivs: integer[];
public moveset: PokemonMove[];
2023-04-11 16:08:03 -07:00
public status: Status;
2023-03-28 11:54:52 -07:00
public winCount: integer;
2023-04-03 20:38:31 -07:00
public summonData: PokemonSummonData;
public battleSummonData: PokemonBattleSummonData;
public turnData: PokemonTurnData;
2023-04-10 20:15:06 -07:00
public maskEnabled: boolean;
public maskSprite: Phaser.GameObjects.Sprite;
2023-03-28 11:54:52 -07:00
private shinySparkle: Phaser.GameObjects.Sprite;
constructor(scene: BattleScene, x: number, y: number, species: PokemonSpecies, level: integer, formIndex?: integer, gender?: Gender, shiny?: boolean, dataSource?: Pokemon) {
2023-03-28 11:54:52 -07:00
super(scene, x, y);
this.name = Utils.toPokemonUpperCase(species.name);
this.species = species;
this.battleInfo = this.isPlayer()
? new PlayerBattleInfo(scene)
: new EnemyBattleInfo(scene);
this.pokeball = dataSource?.pokeball || PokeballType.POKEBALL;
this.level = level;
this.formIndex = formIndex || 0;
if (gender !== undefined)
this.gender = gender;
if (shiny !== undefined)
this.shiny = shiny;
2023-03-28 11:54:52 -07: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;
this.moveset = dataSource.moveset;
2023-04-03 20:38:31 -07:00
this.status = dataSource.status;
2023-03-28 11:54:52 -07:00
this.winCount = dataSource.winCount;
} else {
this.generateAndPopulateMoveset();
this.id = Utils.randInt(4294967295);
this.ivs = [
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))
];
//} else
//this.id = parseInt(Utils.decToBin(this.ivs[Stat.HP]) + Utils.decToBin(this.ivs[Stat.ATK]) + Utils.decToBin(this.ivs[Stat.DEF]) + Utils.decToBin(this.ivs[Stat.SPATK]) + Utils.decToBin(this.ivs[Stat.SPDEF]) + Utils.decToBin(this.ivs[Stat.SPD]) + this.id.toString(2).slice(30));
2023-03-28 11:54:52 -07:00
if (this.gender === undefined) {
if (this.getSpeciesForm().malePercent === null)
this.gender = Gender.GENDERLESS;
else {
const genderChance = (this.id % 256) * 0.390625;
if (genderChance < this.getSpeciesForm().malePercent)
this.gender = Gender.MALE;
else
this.gender = Gender.FEMALE;
}
2023-03-28 11:54:52 -07:00
}
const rand1 = Utils.binToDec(Utils.decToBin(this.id).substring(0, 16));
const rand2 = Utils.binToDec(Utils.decToBin(this.id).substring(16, 32));
2023-04-17 22:32:26 -07:00
const E = this.scene.gameData.trainerId ^ this.scene.gameData.secretId;
2023-03-28 11:54:52 -07:00
const F = rand1 ^ rand2;
if (this.shiny === undefined) {
let shinyThreshold = new Utils.IntegerHolder(32);
2023-04-20 16:44:56 -07:00
this.scene.applyModifiers(ShinyRateBoosterModifier, this.isPlayer(), shinyThreshold);
console.log(shinyThreshold.value);
this.shiny = (E ^ F) < shinyThreshold.value;
if ((E ^ F) < 32)
console.log('REAL SHINY!!');
if (this.shiny)
console.log((E ^ F), shinyThreshold.value);
/*else
this.shiny = Utils.randInt(16) === 0;*/
}
2023-03-28 11:54:52 -07:00
this.winCount = 0;
}
2023-04-14 15:21:33 -07:00
//this.setPipeline(this.scene).spritePipeline);
2023-04-03 17:47:41 -07:00
2023-03-28 11:54:52 -07:00
this.calculateStats();
2023-04-14 15:21:33 -07:00
scene.fieldUI.addAt(this.battleInfo, 0);
2023-03-28 11:54:52 -07:00
this.battleInfo.initInfo(this);
const getSprite = () => {
const ret = this.scene.add.sprite(0, 0, `pkmn__${this.isPlayer() ? 'back__' : ''}sub`);
ret.setOrigin(0.5, 1);
return ret;
};
const sprite = getSprite();
const tintSprite = getSprite();
2023-04-03 17:47:41 -07:00
const zoomSprite = getSprite();
2023-03-28 11:54:52 -07:00
tintSprite.setVisible(false);
2023-04-03 17:47:41 -07:00
zoomSprite.setAlpha(0.5);
2023-03-28 11:54:52 -07:00
this.add(sprite);
this.add(tintSprite);
2023-04-03 17:47:41 -07:00
this.add(zoomSprite);
2023-03-28 11:54:52 -07:00
this.getSpeciesForm().generateIconAnim(scene, this.gender === Gender.FEMALE, formIndex);
2023-03-28 11:54:52 -07:00
if (this.shiny) {
const shinySparkle = this.scene.add.sprite(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;
}
}
abstract isPlayer(): boolean;
2023-03-28 21:31:25 -07:00
loadAssets(): Promise<void> {
return new Promise(resolve => {
2023-04-03 17:47:41 -07:00
const moveIds = this.moveset.map(m => m.getMove().id);
2023-04-11 16:08:03 -07:00
Promise.allSettled(moveIds.map(m => initMoveAnim(m)))
2023-04-03 17:47:41 -07:00
.then(() => {
2023-04-14 15:21:33 -07:00
loadMoveAnimAssets(this.scene, moveIds);
this.getSpeciesForm().loadAssets(this.scene, this.gender === Gender.FEMALE, this.formIndex, this.shiny);
if (this.isPlayer())
2023-04-14 15:21:33 -07:00
this.scene.loadAtlas(this.getBattleSpriteKey(), 'pokemon', this.getBattleSpriteAtlasPath());
2023-04-03 17:47:41 -07:00
this.scene.load.once(Phaser.Loader.Events.COMPLETE, () => {
if (this.isPlayer()) {
2023-04-12 16:09:15 -07: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: 256 });
console.warn = originalWarn;
if (this.isPlayer()) {
this.scene.anims.create({
key: this.getBattleSpriteKey(),
frames: battleFrameNames,
frameRate: 12,
repeat: -1
});
}
}
2023-04-04 18:10:11 -07:00
this.playAnim();
2023-04-03 17:47:41 -07:00
resolve();
});
if (!this.scene.load.isLoading())
this.scene.load.start();
2023-03-28 21:31:25 -07:00
});
});
}
getSpriteAtlasPath(): string {
2023-03-28 11:54:52 -07:00
return this.getSpriteId().replace(/\_{2}/g, '/');
}
getBattleSpriteAtlasPath(): string {
return this.getBattleSpriteId().replace(/\_{2}/g, '/');
}
2023-03-28 21:31:25 -07:00
getSpriteId(): string {
return this.getSpeciesForm().getSpriteId(this.gender === Gender.FEMALE, this.formIndex, this.shiny);
}
getBattleSpriteId(): string {
return `${this.isPlayer() ? 'back__' : ''}${this.getSpriteId()}`;
2023-03-28 11:54:52 -07:00
}
2023-03-28 21:31:25 -07:00
getSpriteKey(): string {
return this.getSpeciesForm().getSpriteKey(this.gender === Gender.FEMALE, this.formIndex, this.shiny);
2023-03-28 11:54:52 -07:00
}
getBattleSpriteKey(): string {
return `pkmn__${this.getBattleSpriteId()}`;
}
2023-03-28 21:31:25 -07:00
getIconId(): string {
return this.getSpeciesForm().getIconId(this.gender === Gender.FEMALE, this.formIndex);
2023-03-28 11:54:52 -07:00
}
2023-03-28 21:31:25 -07:00
getIconKey(): string {
2023-03-28 11:54:52 -07:00
return `pkmn_icon__${this.getIconId()}`;
}
getSpeciesForm(): PokemonSpeciesForm {
if (!this.species.forms?.length)
return this.species;
return this.species.forms[this.formIndex];
}
2023-03-28 21:31:25 -07:00
getSprite(): Phaser.GameObjects.Sprite {
2023-03-28 11:54:52 -07:00
return this.getAt(0) as Phaser.GameObjects.Sprite;
}
2023-03-28 21:31:25 -07:00
getTintSprite(): Phaser.GameObjects.Sprite {
2023-04-10 20:15:06 -07:00
return !this.maskEnabled
? this.getAt(1) as Phaser.GameObjects.Sprite
: this.maskSprite;
2023-03-28 11:54:52 -07:00
}
2023-04-03 17:47:41 -07:00
getZoomSprite(): Phaser.GameObjects.Sprite {
2023-04-10 20:15:06 -07:00
return this.getAt(!this.maskEnabled ? 2 : 1) as Phaser.GameObjects.Sprite;
2023-04-03 17:47:41 -07:00
}
2023-04-10 04:59:00 -07:00
playAnim(): void{
this.getSprite().play(this.getBattleSpriteKey());
this.getTintSprite().play(this.getBattleSpriteKey());
this.getZoomSprite().play(this.getBattleSpriteKey());
2023-04-04 18:10:11 -07:00
}
getBattleStat(stat: Stat): integer {
2023-04-10 20:15:06 -07:00
if (stat === Stat.HP)
return this.stats[Stat.HP];
const battleStat = (stat - 1) as BattleStat;
const statLevel = new Utils.IntegerHolder(this.summonData.battleStats[battleStat]);
if (this.isPlayer())
2023-04-20 16:44:56 -07:00
this.scene.applyModifiers(TempBattleStatBoosterModifier, this.isPlayer(), battleStat as integer as TempBattleStat, statLevel);
let ret = this.stats[stat] * (Math.max(2, 2 + statLevel.value) / Math.max(2, 2 - statLevel.value));
if (stat === Stat.SPDEF && this.scene.arena.weather?.weatherType === WeatherType.SANDSTORM)
ret *= 1.5;
2023-04-20 21:10:45 -07:00
if (stat === Stat.SPD && this.status && this.status.effect === StatusEffect.PARALYSIS)
2023-04-11 16:08:03 -07:00
ret >>= 2;
return ret;
2023-04-10 20:15:06 -07:00
}
2023-04-10 04:59:00 -07:00
calculateStats(): void {
2023-03-28 11:54:52 -07:00
if (!this.stats)
this.stats = [ 0, 0, 0, 0, 0, 0 ];
const baseStats = this.getSpeciesForm().baseStats.slice(0);
2023-04-20 16:44:56 -07:00
this.scene.applyModifiers(PokemonBaseStatModifier, this.isPlayer(), this, baseStats);
2023-03-28 11:54:52 -07:00
const stats = Utils.getEnumValues(Stat);
for (let s of stats) {
const isHp = s === Stat.HP;
let baseStat = baseStats[s];
let value = Math.floor(((2 * baseStat + this.ivs[s] + (0 / 4)) * this.level) * 0.01);
if (isHp) {
value = Math.min(value + this.level + 10, 99999);
if (this.hp > value || this.hp === undefined)
this.hp = value;
else {
const lastMaxHp = this.getMaxHp();
if (lastMaxHp && value > lastMaxHp)
this.hp += value - lastMaxHp;
}
} else
value = Math.min(value + 5, 99999);
this.stats[s] = value;
}
}
2023-04-10 04:59:00 -07:00
getMaxHp(): integer {
2023-03-28 11:54:52 -07:00
return this.stats[Stat.HP];
}
2023-04-11 08:04:39 -07:00
getInverseHp(): integer {
return this.getMaxHp() - this.hp;
}
2023-04-10 04:59:00 -07:00
getHpRatio(): number {
2023-03-29 09:23:52 -07:00
return Math.floor((this.hp / this.getMaxHp()) * 100) / 100;
}
2023-04-15 21:29:55 -07:00
getTypes(): Type[] {
const speciesForm = this.getSpeciesForm();
const speciesTypes = [ speciesForm.type1 ];
if (speciesForm.type2 !== null)
speciesTypes.push(speciesForm.type1);
2023-04-15 21:29:55 -07:00
if (this.getTag(BattleTagType.IGNORE_FLYING)) {
const flyingIndex = speciesTypes.indexOf(Type.FLYING);
if (flyingIndex > -1)
speciesTypes.splice(flyingIndex, 1);
}
if (!speciesTypes.length)
speciesTypes.push(Type.NORMAL);
return speciesTypes;
}
2023-04-10 04:59:00 -07:00
getEvolution(): SpeciesEvolution {
if (!pokemonEvolutions.hasOwnProperty(this.species.speciesId))
return null;
const evolutions = pokemonEvolutions[this.species.speciesId];
for (let e of evolutions) {
2023-04-15 19:51:33 -07:00
if (!e.item && this.level >= e.level) {
if (e.condition === null || (e.condition as SpeciesEvolutionCondition).predicate(this))
2023-04-10 04:59:00 -07:00
return e;
}
}
return null;
}
2023-04-10 10:54:06 -07:00
getLevelMoves(startingLevel?: integer): Moves[] {
const ret: Moves[] = [];
const levelMoves = pokemonLevelMoves[this.species.speciesId];
if (levelMoves) {
if (!startingLevel)
startingLevel = this.level;
for (let lm of levelMoves) {
const level = lm[0];
if (level < startingLevel)
continue;
else if (level > this.level)
break;
ret.push(lm[1]);
}
}
return ret;
2023-04-10 04:59:00 -07:00
}
generateAndPopulateMoveset(): void {
2023-03-28 11:54:52 -07:00
this.moveset = [];
const movePool = [];
const allLevelMoves = pokemonLevelMoves[this.species.speciesId];
if (!allLevelMoves) {
console.log(this.species.speciesId, 'ERROR')
return;
}
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 18:32:48 -07:00
const move = allMoves[m];
2023-03-28 11:54:52 -07:00
return move.category !== MoveCategory.STATUS;
});
if (attackMovePool.length) {
const moveIndex = Utils.randInt(attackMovePool.length);
this.moveset.push(new PokemonMove(attackMovePool[moveIndex], 0, 0));
2023-04-20 18:32:48 -07:00
console.log(allMoves[attackMovePool[moveIndex]]);
2023-03-28 11:54:52 -07:00
movePool.splice(movePool.findIndex(m => m === attackMovePool[moveIndex]), 1);
}
while (movePool.length && this.moveset.length < 4) {
const moveIndex = Utils.randInt(movePool.length);
this.moveset.push(new PokemonMove(movePool[moveIndex], 0, 0));
2023-04-20 18:32:48 -07:00
console.log(allMoves[movePool[moveIndex]]);
2023-03-28 11:54:52 -07:00
movePool.splice(moveIndex, 1);
}
}
trySelectMove(moveIndex: integer): boolean {
const move = this.moveset.length > moveIndex
? this.moveset[moveIndex]
: null;
return move?.isUsable();
}
showInfo() {
if (!this.battleInfo.visible) {
this.battleInfo.setX(this.battleInfo.x + (this.isPlayer() ? 150 : -150));
this.battleInfo.setVisible(true);
this.scene.tweens.add({
targets: this.battleInfo,
x: this.isPlayer() ? '-=150' : '+=150',
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' : '-=150',
duration: 500,
ease: 'Sine.easeIn',
onComplete: () => {
this.battleInfo.setVisible(false);
this.battleInfo.setX(this.battleInfo.x - (this.isPlayer() ? 150 : -150));
resolve();
}
});
} else
resolve();
});
2023-03-28 11:54:52 -07:00
}
2023-04-10 04:59:00 -07:00
updateInfo(instant?: boolean): Promise<void> {
return this.battleInfo.updateInfo(this, instant);
2023-03-28 11:54:52 -07:00
}
addExp(exp: integer) {
this.exp += exp;
while (this.exp >= getLevelTotalExp(this.level + 1, this.getSpeciesForm().growthRate))
2023-03-28 11:54:52 -07:00
this.level++;
this.levelExp = this.exp - getLevelTotalExp(this.level, this.getSpeciesForm().growthRate);
2023-03-28 11:54:52 -07:00
}
2023-04-14 22:32:16 -07:00
apply(source: Pokemon, battlerMove: PokemonMove): MoveResult {
let result: MoveResult;
2023-04-14 22:32:16 -07:00
const move = battlerMove.getMove();
const moveCategory = move.category;
let damage = 0;
switch (moveCategory) {
case MoveCategory.PHYSICAL:
case MoveCategory.SPECIAL:
const isPhysical = moveCategory === MoveCategory.PHYSICAL;
const power = new Utils.NumberHolder(move.power);
const typeMultiplier = getTypeDamageMultiplier(move.type, this.getSpeciesForm().type1) * (this.getSpeciesForm().type2 !== null ? getTypeDamageMultiplier(move.type, this.getSpeciesForm().type2) : 1);
const weatherTypeMultiplier = this.scene.arena.getAttackTypeMultiplier(move.type);
applyMoveAttrs(VariablePowerAttr, source, this, move, power);
2023-04-20 16:44:56 -07:00
this.scene.applyModifiers(AttackTypeBoosterModifier, source.isPlayer(), source, power);
const critLevel = new Utils.IntegerHolder(0);
applyMoveAttrs(HighCritAttr, source, this, move, critLevel);
2023-04-20 16:44:56 -07:00
this.scene.applyModifiers(TempBattleStatBoosterModifier, source.isPlayer(), TempBattleStat.CRIT, critLevel);
const critChance = Math.ceil(16 / Math.pow(2, critLevel.value));
2023-04-19 13:52:14 -07:00
let isCritical = !source.getTag(BattleTagType.NO_CRIT) && (critChance === 1 || !Utils.randInt(critChance));
2023-04-14 22:32:16 -07:00
const sourceAtk = source.getBattleStat(isPhysical ? Stat.ATK : Stat.SPATK);
const targetDef = this.getBattleStat(isPhysical ? Stat.DEF : Stat.SPDEF);
const stabMultiplier = source.species.type1 === move.type || (source.species.type2 !== null && source.species.type2 === move.type) ? 1.5 : 1;
2023-04-14 22:32:16 -07:00
const criticalMultiplier = isCritical ? 2 : 1;
damage = Math.ceil(((((2 * source.level / 5 + 2) * power.value * sourceAtk / targetDef) / 50) + 2) * stabMultiplier * typeMultiplier * weatherTypeMultiplier * ((Utils.randInt(15) + 85) / 100)) * criticalMultiplier;
2023-04-14 22:32:16 -07:00
if (isPhysical && source.status && source.status.effect === StatusEffect.BURN)
damage = Math.floor(damage / 2);
move.getAttrs(HitsTagAttr).map(hta => hta as HitsTagAttr).filter(hta => hta.doubleDamage).forEach(hta => {
if (this.getTag(hta.tagType))
damage *= 2;
});
const fixedDamage = new Utils.IntegerHolder(0);
2023-04-15 21:29:55 -07:00
applyMoveAttrs(FixedDamageAttr, source, this, move, fixedDamage);
if (damage && fixedDamage.value) {
damage = fixedDamage.value;
isCritical = false;
2023-04-14 22:32:16 -07:00
result = MoveResult.EFFECTIVE;
}
2023-04-13 09:16:36 -07:00
console.log('damage', damage, move.name, move.power, sourceAtk, targetDef);
2023-04-14 22:32:16 -07:00
if (!result) {
if (typeMultiplier >= 2)
result = MoveResult.SUPER_EFFECTIVE;
else if (typeMultiplier >= 1)
result = MoveResult.EFFECTIVE;
else if (typeMultiplier > 0)
result = MoveResult.NOT_VERY_EFFECTIVE;
else
result = MoveResult.NO_EFFECT;
}
2023-04-14 22:32:16 -07:00
if (damage) {
2023-04-16 15:40:32 -07:00
this.damage(damage);
2023-04-14 22:32:16 -07:00
source.turnData.damageDealt += damage;
this.scene.unshiftPhase(new DamagePhase(this.scene, this.isPlayer(), result as DamageResult))
if (isCritical)
this.scene.unshiftPhase(new MessagePhase(this.scene, 'A critical hit!'));
2023-04-13 09:16:36 -07:00
}
2023-04-14 22:32:16 -07:00
switch (result) {
case MoveResult.SUPER_EFFECTIVE:
this.scene.unshiftPhase(new MessagePhase(this.scene, 'It\'s super effective!'));
break;
case MoveResult.NOT_VERY_EFFECTIVE:
this.scene.unshiftPhase(new MessagePhase(this.scene, 'It\'s not very effective!'));
2023-04-14 22:32:16 -07:00
break;
case MoveResult.NO_EFFECT:
this.scene.unshiftPhase(new MessagePhase(this.scene, `It doesn\'t affect ${this.name}!`));
2023-04-14 22:32:16 -07:00
break;
}
break;
case MoveCategory.STATUS:
result = MoveResult.STATUS;
break;
}
return result;
2023-04-13 09:16:36 -07:00
}
2023-04-16 15:40:32 -07:00
damage(damage: integer): void {
if (!this.hp)
return;
this.hp = Math.max(this.hp - damage, 0);
if (!this.hp) {
this.scene.pushPhase(new FaintPhase(this.scene, this.isPlayer()));
2023-04-20 18:32:48 -07:00
this.resetSummonData();
2023-04-16 15:40:32 -07:00
(this.isPlayer() ? this.scene.getEnemyPokemon() : this.scene.getPlayerPokemon()).resetBattleSummonData();
}
}
2023-04-14 22:32:16 -07:00
addTag(tagType: BattleTagType, turnCount?: integer): boolean {
const existingTag = this.getTag(tagType);
if (existingTag) {
existingTag.onOverlap(this);
2023-04-13 09:16:36 -07:00
return false;
2023-04-14 22:32:16 -07:00
}
2023-04-13 09:16:36 -07:00
const newTag = getBattleTag(tagType, turnCount || 0);
2023-04-13 09:16:36 -07:00
this.summonData.tags.push(newTag);
2023-04-14 22:32:16 -07:00
newTag.onAdd(this);
2023-04-13 09:16:36 -07:00
}
2023-04-14 22:32:16 -07:00
getTag(tagType: BattleTagType | { new(...args: any[]): BattleTag }): BattleTag {
return typeof(tagType) === 'number'
? this.summonData.tags.find(t => t.tagType === tagType)
: this.summonData.tags.find(t => t instanceof tagType);
2023-04-13 09:16:36 -07:00
}
2023-04-14 22:32:16 -07:00
findTag(tagFilter: ((tag: BattleTag) => boolean)) {
return this.summonData.tags.find(t => tagFilter(t));
2023-04-13 09:16:36 -07:00
}
2023-04-14 22:32:16 -07:00
getTags(tagType: BattleTagType | { new(...args: any[]): BattleTag }): BattleTag[] {
return typeof(tagType) === 'number'
? this.summonData.tags.filter(t => t.tagType === tagType)
: this.summonData.tags.filter(t => t instanceof tagType);
}
findTags(tagFilter: ((tag: BattleTag) => boolean)) {
return this.summonData.tags.filter(t => tagFilter(t));
}
2023-04-18 09:30:47 -07:00
lapseTag(tagType: BattleTagType): boolean {
2023-04-13 09:16:36 -07:00
const tags = this.summonData.tags;
2023-04-14 22:32:16 -07:00
const tag = tags.find(t => t.tagType === tagType);
2023-04-16 15:40:32 -07:00
if (tag && !(tag.lapse(this, BattleTagLapseType.CUSTOM))) {
2023-04-14 22:32:16 -07:00
tag.onRemove(this);
tags.splice(tags.indexOf(tag), 1);
}
2023-04-18 09:30:47 -07:00
return !!tag;
2023-04-14 22:32:16 -07:00
}
lapseTags(lapseType: BattleTagLapseType): void {
const tags = this.summonData.tags;
2023-04-16 15:40:32 -07:00
tags.filter(t => lapseType === BattleTagLapseType.FAINT || ((t.lapseType === lapseType) && !(t.lapse(this, lapseType))) || (lapseType === BattleTagLapseType.TURN_END && t.turnCount < 1)).forEach(t => {
2023-04-14 22:32:16 -07:00
t.onRemove(this);
tags.splice(tags.indexOf(t), 1);
});
2023-04-13 09:16:36 -07:00
}
2023-04-20 18:32:48 -07:00
getMoveHistory(): TurnMove[] {
return this.summonData.moveHistory;
}
2023-04-13 09:16:36 -07:00
getLastXMoves(turnCount?: integer): TurnMove[] {
2023-04-20 18:32:48 -07:00
const moveHistory = this.getMoveHistory();
2023-04-18 09:30:47 -07:00
return moveHistory.slice(turnCount >= 0 ? Math.max(moveHistory.length - (turnCount || 1), 0) : 0, moveHistory.length).reverse();
2023-03-28 11:54:52 -07:00
}
2023-04-20 18:32:48 -07:00
getMoveQueue(): QueuedMove[] {
return this.summonData.moveQueue;
}
2023-04-11 16:08:03 -07:00
cry(soundConfig?: Phaser.Types.Sound.SoundConfig): integer {
return this.getSpeciesForm().cry(this.scene, soundConfig);
2023-03-28 11:54:52 -07:00
}
2023-03-28 21:31:25 -07:00
faintCry(callback: Function) {
2023-03-28 11:54:52 -07:00
const key = this.species.speciesId.toString();
let i = 0;
let rate = 0.85;
this.scene.sound.play(key, {
rate: rate
});
const sprite = this.getSprite();
2023-04-03 17:47:41 -07:00
const tintSprite = this.getTintSprite();
const zoomSprite = this.getZoomSprite();
2023-03-28 11:54:52 -07:00
const delay = Math.max(this.scene.sound.get(key).totalDuration * 50, 25);
let frameProgress = 0;
let frameThreshold: number;
sprite.anims.pause();
2023-04-03 17:47:41 -07:00
tintSprite.anims.pause();
zoomSprite.anims.pause();
2023-03-28 11:54:52 -07:00
let faintCryTimer = this.scene.time.addEvent({
delay: delay,
repeat: -1,
callback: () => {
++i;
frameThreshold = sprite.anims.msPerFrame / rate;
frameProgress += delay;
while (frameProgress > frameThreshold) {
2023-04-03 17:47:41 -07:00
if (sprite.anims.duration) {
2023-03-28 11:54:52 -07:00
sprite.anims.nextFrame();
2023-04-03 17:47:41 -07:00
tintSprite.anims.nextFrame();
zoomSprite.anims.nextFrame();
}
2023-03-28 11:54:52 -07:00
frameProgress -= frameThreshold;
}
const crySound = this.scene.sound.get(key);
if (crySound) {
rate *= 0.99;
crySound.play({
rate: rate,
seek: (i * delay * 0.001) * rate
});
}
else {
faintCryTimer.destroy();
faintCryTimer = null;
if (callback)
callback();
}
}
});
// Failsafe
2023-04-11 16:08:03 -07:00
this.scene.time.delayedCall(3000, () => {
2023-03-28 11:54:52 -07:00
if (!faintCryTimer || !this.scene)
return;
const crySound = this.scene.sound.get(key);
if (crySound?.isPlaying)
crySound.stop();
faintCryTimer.destroy();
if (callback)
callback();
});
}
2023-04-11 16:08:03 -07:00
trySetStatus(effect: StatusEffect): boolean {
if (this.status)
return false;
const speciesForm = this.getSpeciesForm();
2023-04-11 16:08:03 -07:00
switch (effect) {
case StatusEffect.POISON:
case StatusEffect.TOXIC:
if (speciesForm.isOfType(Type.POISON) || speciesForm.isOfType(Type.STEEL))
2023-04-11 16:08:03 -07:00
return false;
break;
case StatusEffect.FREEZE:
if (speciesForm.isOfType(Type.ICE))
2023-04-11 16:08:03 -07:00
return false;
break;
case StatusEffect.BURN:
if (speciesForm.isOfType(Type.FIRE))
2023-04-11 16:08:03 -07:00
return false;
break;
}
this.status = new Status(effect);
return true;
}
resetStatus(): void {
2023-04-16 15:40:32 -07:00
const lastStatus = this.status.effect;
2023-04-11 16:08:03 -07:00
this.status = undefined;
2023-04-16 15:40:32 -07:00
if (lastStatus === StatusEffect.SLEEP) {
if (this.getTag(BattleTagType.NIGHTMARE))
this.lapseTag(BattleTagType.NIGHTMARE);
}
2023-04-11 16:08:03 -07:00
}
resetSummonData(): void {
2023-04-03 20:38:31 -07:00
this.summonData = new PokemonSummonData();
this.resetBattleSummonData();
}
2023-04-11 16:08:03 -07:00
resetBattleSummonData(): void {
2023-04-03 20:38:31 -07:00
this.battleSummonData = new PokemonBattleSummonData();
2023-04-18 09:30:47 -07:00
if (this.getTag(BattleTagType.SEEDED))
this.lapseTag(BattleTagType.SEEDED);
2023-04-03 20:38:31 -07:00
}
2023-04-11 16:08:03 -07:00
resetTurnData(): void {
2023-04-03 20:38:31 -07: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
return (this.getSpeciesForm().baseExp * this.level) / 5 + 1;
2023-03-28 11:54:52 -07: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 20:15:06 -07:00
onComplete: () => {
tintSprite.setVisible(false);
tintSprite.setAlpha(1);
}
2023-03-28 11:54:52 -07:00
});
2023-04-10 20:15:06 -07:00
} else {
2023-03-28 11:54:52 -07:00
tintSprite.setVisible(false);
2023-04-10 20:15:06 -07:00
tintSprite.setAlpha(1);
}
}
enableMask() {
if (!this.maskEnabled) {
this.maskSprite = this.getTintSprite();
this.maskSprite.setVisible(true);
this.maskSprite.setPosition(this.x * 6, this.y * 6);
this.maskSprite.setScale(6);
this.maskEnabled = true;
}
}
disableMask() {
if (this.maskEnabled) {
this.maskSprite.setVisible(false);
this.maskSprite.setPosition(0, 0);
this.maskSprite.setScale(1);
this.maskSprite = null;
this.maskEnabled = false;
}
2023-03-28 11:54:52 -07:00
}
sparkle(): void {
if (this.shinySparkle) {
this.shinySparkle.play('sparkle');
2023-04-10 10:54:06 -07:00
this.scene.sound.play('sparkle');
2023-03-28 11:54:52 -07:00
}
}
destroy(): void {
this.battleInfo.destroy();
super.destroy();
}
2023-03-28 11:54:52 -07:00
}
2023-04-14 15:21:33 -07:00
export default interface Pokemon {
scene: BattleScene
}
2023-03-28 11:54:52 -07:00
export class PlayerPokemon extends Pokemon {
2023-04-07 21:21:44 -07:00
public compatibleTms: Moves[];
2023-04-17 22:32:26 -07:00
constructor(scene: BattleScene, species: PokemonSpecies, level: integer, formIndex: integer, gender?: Gender, shiny?: boolean, dataSource?: Pokemon) {
super(scene, 106, 148, species, level, formIndex, gender, shiny, dataSource);
2023-04-07 21:21:44 -07:00
this.generateCompatibleTms();
2023-03-28 11:54:52 -07:00
}
2023-04-07 21:21:44 -07:00
isPlayer(): boolean {
2023-03-28 11:54:52 -07:00
return true;
}
2023-04-07 21:21:44 -07:00
generateCompatibleTms(): void {
this.compatibleTms = [];
const tms = Object.keys(tmSpecies);
for (let tm of tms) {
const moveId = parseInt(tm) as Moves;
for (let p of tmSpecies[tm]) {
if (Array.isArray(p)) {
if (p[0] === this.species.speciesId) {
this.compatibleTms.push(moveId);
break;
}
} else if (p === this.species.speciesId) {
this.compatibleTms.push(moveId);
break;
}
}
}
}
2023-04-10 10:54:06 -07:00
evolve(evolution: SpeciesEvolution): Promise<void> {
return new Promise(resolve => {
2023-04-14 15:21:33 -07:00
this.handleSpecialEvolutions(evolution);
2023-04-10 10:54:06 -07:00
this.species = getPokemonSpecies(evolution.speciesId);
this.name = this.species.name.toUpperCase();
this.getSpeciesForm().generateIconAnim(this.scene, this.gender === Gender.FEMALE, this.formIndex);
2023-04-10 10:54:06 -07:00
this.compatibleTms.splice(0, this.compatibleTms.length);
this.generateCompatibleTms();
2023-04-17 22:32:26 -07:00
this.scene.gameData.setPokemonSeen(this);
this.scene.gameData.setPokemonCaught(this);
2023-04-10 10:54:06 -07:00
this.loadAssets().then(() => {
this.calculateStats();
this.updateInfo().then(() => resolve());
});
});
}
2023-04-14 15:21:33 -07:00
private handleSpecialEvolutions(evolution: SpeciesEvolution) {
if (this.species.speciesId === Species.NINCADA && evolution.speciesId === Species.NINJASK) {
const newEvolution = pokemonEvolutions[this.species.speciesId][1];
if (newEvolution.condition.predicate(this)) {
const newPokemon = new PlayerPokemon(this.scene, this.species, this.level, this.formIndex, this.gender, this.shiny);
2023-04-14 15:21:33 -07:00
this.scene.getParty().push(newPokemon);
newPokemon.evolve(newEvolution);
}
}
}
2023-03-28 11:54:52 -07:00
}
export class EnemyPokemon extends Pokemon {
public aiType: AiType;
constructor(scene: BattleScene, species: PokemonSpecies, level: integer) {
super(scene, -66, 84, species, level, scene.arena.getFormIndex(species));
2023-03-28 11:54:52 -07:00
2023-04-18 12:07:10 -07:00
let prevolution: Species;
let speciesId = species.speciesId;
while ((prevolution = pokemonPrevolutions[speciesId])) {
const evolution = pokemonEvolutions[prevolution].find(pe => pe.speciesId === speciesId);
if (evolution.condition?.enforceFunc)
evolution.condition.enforceFunc(this);
speciesId = prevolution;
}
2023-03-28 21:31:25 -07:00
this.aiType = AiType.SMART_RANDOM;
2023-03-28 11:54:52 -07:00
}
getNextMove(): PokemonMove {
2023-04-20 18:32:48 -07:00
const queuedMove = this.getMoveQueue().length
? this.moveset.find(m => m.moveId === this.getMoveQueue()[0].move)
2023-04-13 09:16:36 -07:00
: null;
2023-04-19 15:19:55 -07:00
if (queuedMove) {
2023-04-20 18:32:48 -07:00
if (queuedMove.isUsable(this.getMoveQueue()[0].ignorePP))
2023-04-19 15:19:55 -07:00
return queuedMove;
else {
2023-04-20 18:32:48 -07:00
this.getMoveQueue().shift();
2023-04-19 15:19:55 -07:00
return this.getNextMove();
}
}
2023-04-13 09:16:36 -07:00
2023-03-28 11:54:52 -07:00
const movePool = this.moveset.filter(m => m.isUsable());
if (movePool.length) {
if (movePool.length === 1)
return movePool[0];
switch (this.aiType) {
case AiType.RANDOM:
return movePool[Utils.randInt(movePool.length)];
2023-03-28 21:31:25 -07:00
case AiType.SMART_RANDOM:
2023-03-28 11:54:52 -07:00
case AiType.SMART:
2023-04-14 15:21:33 -07:00
const target = this.scene.getPlayerPokemon();
2023-03-28 11:54:52 -07:00
const moveScores = movePool.map(() => 0);
for (let m in movePool) {
const pokemonMove = movePool[m];
const move = pokemonMove.getMove();
2023-04-10 20:15:06 -07:00
let moveScore = moveScores[m];
if (move.category === MoveCategory.STATUS)
moveScore++;
else {
2023-04-15 21:29:55 -07:00
const targetTypes = target.getTypes();
const effectiveness = getTypeDamageMultiplier(move.type, targetTypes[0]) * (targetTypes.length > 1 ? getTypeDamageMultiplier(move.type, targetTypes[1]) : 1);
2023-04-10 20:15:06 -07:00
moveScore = Math.pow(effectiveness - 1, 2) * effectiveness < 1 ? -2 : 2;
if (moveScore) {
2023-03-28 11:54:52 -07:00
if (move.category === MoveCategory.PHYSICAL) {
2023-04-10 20:15:06 -07:00
if (this.getBattleStat(Stat.ATK) > this.getBattleStat(Stat.SPATK)) {
const statRatio = this.getBattleStat(Stat.SPATK) / this.getBattleStat(Stat.ATK);
2023-03-28 11:54:52 -07:00
if (statRatio <= 0.75)
2023-04-10 20:15:06 -07:00
moveScore *= 2;
2023-03-28 11:54:52 -07:00
else if (statRatio <= 0.875)
2023-04-10 20:15:06 -07:00
moveScore *= 1.5;
2023-03-28 11:54:52 -07:00
}
} else {
2023-04-10 20:15:06 -07:00
if (this.getBattleStat(Stat.SPATK) > this.getBattleStat(Stat.ATK)) {
const statRatio = this.getBattleStat(Stat.ATK) / this.getBattleStat(Stat.SPATK);
2023-03-28 11:54:52 -07:00
if (statRatio <= 0.75)
2023-04-10 20:15:06 -07:00
moveScore *= 2;
2023-03-28 11:54:52 -07:00
else if (statRatio <= 0.875)
2023-04-10 20:15:06 -07:00
moveScore *= 1.5;
2023-03-28 11:54:52 -07:00
}
}
2023-04-10 20:15:06 -07:00
moveScore += Math.floor(move.power / 5);
2023-03-28 11:54:52 -07:00
}
}
2023-04-10 20:15:06 -07:00
const statChangeAttrs = move.getAttrs(StatChangeAttr) as StatChangeAttr[];
for (let sc of statChangeAttrs) {
moveScore += ((sc.levels >= 1) === sc.selfTarget ? -2 : 2) + sc.levels * (sc.selfTarget ? 4 : -4);
// TODO: Add awareness of current levels
}
// could make smarter by checking opponent def/spdef
moveScores[m] = moveScore;
2023-03-28 11:54:52 -07:00
}
2023-03-28 21:31:25 -07:00
console.log(moveScores);
2023-03-28 11:54:52 -07: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-28 21:31:25 -07:00
if (this.aiType === AiType.SMART_RANDOM) {
2023-04-10 04:59:00 -07:00
while (r < sortedMovePool.length - 1 && Utils.randInt(8) >= 5)
2023-03-28 21:31:25 -07:00
r++;
}
2023-03-28 11:54:52 -07:00
console.log(movePool.map(m => m.getName()), moveScores, r, sortedMovePool.map(m => m.getName()));
2023-03-28 21:31:25 -07:00
return sortedMovePool[r];
2023-03-28 11:54:52 -07:00
}
}
2023-04-19 15:19:55 -07:00
2023-03-28 11:54:52 -07:00
return new PokemonMove(Moves.STRUGGLE, 0, 0);
}
isPlayer() {
return false;
}
addToParty() {
2023-04-14 15:21:33 -07:00
const party = this.scene.getParty();
2023-03-31 17:19:57 -07:00
let ret: PlayerPokemon = null;
2023-03-28 11:54:52 -07:00
2023-03-31 17:19:57 -07:00
if (party.length < 6) {
2023-04-20 16:44:56 -07:00
const newPokemon = new PlayerPokemon(this.scene, this.species, this.level, this.formIndex, this.gender, this.shiny, this);
2023-03-31 17:19:57 -07:00
party.push(newPokemon);
ret = newPokemon;
}
2023-03-31 19:31:20 -07:00
2023-03-31 17:19:57 -07:00
return ret;
2023-03-28 11:54:52 -07:00
}
}
2023-04-13 09:16:36 -07:00
export interface TurnMove {
move: Moves;
result: MoveResult;
2023-04-19 15:19:55 -07:00
virtual?: boolean;
2023-04-13 09:16:36 -07:00
}
export interface QueuedMove {
move: Moves;
ignorePP?: boolean;
}
2023-04-03 20:38:31 -07:00
export class PokemonSummonData {
2023-04-10 20:15:06 -07:00
public battleStats: integer[] = [ 0, 0, 0, 0, 0, 0, 0 ];
2023-04-13 09:16:36 -07:00
public moveHistory: TurnMove[] = [];
public moveQueue: QueuedMove[] = [];
public tags: BattleTag[] = [];
2023-04-03 20:38:31 -07:00
}
export class PokemonBattleSummonData {
public turnCount: integer = 1;
2023-04-03 20:38:31 -07:00
}
export class PokemonTurnData {
public flinched: boolean;
2023-04-10 13:17:25 -07:00
public hitCount: integer;
2023-04-03 20:38:31 -07:00
public hitsLeft: integer;
2023-04-11 16:08:03 -07:00
public hitsTotal: integer;
2023-04-13 22:08:44 -07:00
public damageDealt: integer = 0;
2023-04-03 20:38:31 -07:00
}
2023-03-28 21:31:25 -07:00
export enum AiType {
2023-03-28 11:54:52 -07:00
RANDOM,
2023-03-28 21:31:25 -07:00
SMART_RANDOM,
2023-03-28 11:54:52 -07:00
SMART
};
export enum MoveResult {
EFFECTIVE = 1,
2023-03-28 11:54:52 -07:00
SUPER_EFFECTIVE,
NOT_VERY_EFFECTIVE,
NO_EFFECT,
2023-04-13 09:16:36 -07:00
STATUS,
FAILED,
MISSED,
2023-03-28 11:54:52 -07:00
OTHER
};
2023-04-18 09:30:47 -07:00
export type DamageResult = MoveResult.EFFECTIVE | MoveResult.SUPER_EFFECTIVE | MoveResult.NOT_VERY_EFFECTIVE | MoveResult.OTHER;
2023-04-14 22:32:16 -07:00
2023-03-28 11:54:52 -07:00
export class PokemonMove {
2023-04-13 09:16:36 -07:00
public moveId: Moves;
2023-03-28 11:54:52 -07:00
public ppUsed: integer;
public ppUp: integer;
2023-04-19 15:19:55 -07:00
public virtual: boolean;
2023-03-28 11:54:52 -07:00
public disableTurns: integer;
2023-04-19 15:19:55 -07:00
constructor(moveId: Moves, ppUsed?: integer, ppUp?: integer, virtual?: boolean) {
2023-03-28 11:54:52 -07:00
this.moveId = moveId;
2023-04-13 20:04:51 -07:00
this.ppUsed = ppUsed || 0;
this.ppUp = ppUp || 0;
2023-04-19 15:19:55 -07:00
this.virtual = !!virtual;
2023-03-28 11:54:52 -07:00
this.disableTurns = 0;
}
2023-04-19 15:19:55 -07:00
isUsable(ignorePp?: boolean): boolean {
if (this.isDisabled())
2023-03-28 11:54:52 -07:00
return false;
2023-04-19 15:19:55 -07:00
return ignorePp || this.ppUsed < this.getMove().pp + this.ppUp || this.getMove().pp === -1;
}
isDisabled(): boolean {
return !!this.disableTurns;
2023-03-28 11:54:52 -07:00
}
getMove(): Move {
2023-04-20 18:32:48 -07:00
return allMoves[this.moveId];
2023-03-28 11:54:52 -07:00
}
getName(): string {
return this.getMove().name;
2023-03-28 11:54:52 -07:00
}
}