pokerogue/src/battle-scene.ts

673 lines
20 KiB
TypeScript
Raw Normal View History

2023-03-28 19:54:52 +01:00
import Phaser from 'phaser';
import { Biome } from './biome';
2023-03-28 19:54:52 +01:00
import UI from './ui/ui';
import { EncounterPhase, SummonPhase, CommandPhase, NextEncounterPhase, NewBiomeEncounterPhase, SelectBiomePhase, SelectStarterPhase } from './battle-phases';
2023-03-28 19:54:52 +01:00
import { PlayerPokemon, EnemyPokemon } from './pokemon';
2023-03-29 05:31:25 +01:00
import PokemonSpecies, { allSpecies, getPokemonSpecies } from './pokemon-species';
2023-03-28 19:54:52 +01:00
import * as Utils from './utils';
2023-04-14 23:21:33 +01:00
import { Modifier, ModifierBar, ConsumablePokemonModifier, ConsumableModifier, PartyShareModifier, PokemonHpRestoreModifier, HealingBoosterModifier, PersistentModifier, PokemonHeldItemModifier, ConsumablePokemonMoveModifier, ModifierPredicate } from './modifier';
2023-03-28 19:54:52 +01:00
import { PokeballType } from './pokeball';
2023-03-29 05:31:25 +01:00
import { Species } from './species';
import { initAutoPlay } from './auto-play';
2023-03-31 04:02:35 +01:00
import { Battle } from './battle';
2023-04-12 00:08:03 +01:00
import { initCommonAnims, loadCommonAnimAssets, populateAnims } from './battle-anims';
2023-04-10 19:12:01 +01:00
import { BattlePhase } from './battle-phase';
2023-04-12 16:30:47 +01:00
import { initGameSpeed } from './game-speed';
import { Arena } from './arena';
2023-04-18 06:32:26 +01:00
import { GameData } from './game-data';
import StarterSelectUiHandler from './ui/starter-select-ui-handler';
2023-03-28 19:54:52 +01:00
2023-04-09 05:22:14 +01:00
const enableAuto = true;
export enum Button {
UP,
DOWN,
LEFT,
RIGHT,
ACTION,
CANCEL,
CYCLE_SHINY,
CYCLE_FORM,
CYCLE_GENDER,
2023-04-13 00:09:15 +01:00
QUICK_START,
RANDOM,
AUTO,
SPEED_UP,
SLOW_DOWN
}
2023-03-28 19:54:52 +01:00
export default class BattleScene extends Phaser.Scene {
2023-04-09 05:22:14 +01:00
public auto: boolean;
2023-04-12 16:30:47 +01:00
public gameSpeed: integer = 1;
2023-04-13 00:09:15 +01:00
public quickStart: boolean;
2023-03-29 05:31:25 +01:00
2023-04-18 06:32:26 +01:00
public gameData: GameData;
2023-04-10 19:12:01 +01:00
private phaseQueue: BattlePhase[];
private phaseQueuePrepend: BattlePhase[];
2023-03-28 19:54:52 +01:00
private currentPhase: BattlePhase;
public field: Phaser.GameObjects.Container;
public fieldUI: Phaser.GameObjects.Container;
public arenaBg: Phaser.GameObjects.Image;
2023-03-31 21:04:39 +01:00
public arenaBgTransition: Phaser.GameObjects.Image;
2023-03-28 19:54:52 +01:00
public arenaPlayer: Phaser.GameObjects.Image;
2023-03-31 21:04:39 +01:00
public arenaPlayerTransition: Phaser.GameObjects.Image;
2023-03-28 19:54:52 +01:00
public arenaEnemy: Phaser.GameObjects.Image;
2023-03-31 21:04:39 +01:00
public arenaEnemyTransition: Phaser.GameObjects.Image;
public arenaNextEnemy: Phaser.GameObjects.Image;
public arena: Arena;
2023-03-28 19:54:52 +01:00
public trainer: Phaser.GameObjects.Sprite;
2023-03-31 04:02:35 +01:00
public currentBattle: Battle;
2023-04-02 03:59:07 +01:00
public pokeballCounts = Object.fromEntries(Utils.getEnumValues(PokeballType).filter(p => p <= PokeballType.MASTER_BALL).map(t => [ t, 0 ]));
2023-03-28 19:54:52 +01:00
private party: PlayerPokemon[];
private modifierBar: ModifierBar;
2023-04-10 00:15:21 +01:00
private modifiers: PersistentModifier[];
2023-03-28 19:54:52 +01:00
public uiContainer: Phaser.GameObjects.Container;
public ui: UI;
2023-04-04 01:47:41 +01:00
//public spritePipeline: SpritePipeline;
2023-03-28 19:54:52 +01:00
private bgm: Phaser.Sound.BaseSound;
2023-04-18 06:32:26 +01:00
private bgmResumeTimer: Phaser.Time.TimerEvent;
2023-03-28 19:54:52 +01:00
private buttonKeys: Phaser.Input.Keyboard.Key[][];
2023-03-28 19:54:52 +01:00
private blockInput: boolean;
constructor() {
super('battle');
2023-04-18 06:32:26 +01:00
this.gameData = new GameData(this);
2023-03-28 19:54:52 +01:00
this.phaseQueue = [];
this.phaseQueuePrepend = [];
}
loadImage(key: string, folder: string, filename?: string) {
if (!filename)
filename = `${key}.png`;
this.load.image(key, `images/${folder}/${filename}`);
}
loadAtlas(key: string, folder: string, filenameRoot?: string) {
if (!filenameRoot)
filenameRoot = key;
if (folder)
folder += '/';
2023-04-10 21:17:25 +01:00
this.load.atlas(key, `images/${folder}${filenameRoot}.png`, `images/${folder}/${filenameRoot}.json`);
2023-03-28 19:54:52 +01:00
}
2023-04-04 01:47:41 +01:00
loadSpritesheet(key: string, folder: string, size: integer, filename?: string) {
if (!filename)
filename = `${key}.png`;
this.load.spritesheet(key, `images/${folder}/${filename}`, { frameWidth: size, frameHeight: size });
}
2023-03-28 19:54:52 +01:00
loadSe(key: string, folder?: string, filenames?: string | string[]) {
if (!filenames)
filenames = `${key}.wav`;
if (!folder)
folder = '';
else
folder += '/';
if (!Array.isArray(filenames))
filenames = [ filenames ];
for (let f of filenames as string[]) {
this.load.audio(key, `audio/se/${folder}${f}`);
}
}
loadBgm(key: string, filename?: string) {
if (!filename)
filename = `${key}.mp3`;
this.load.audio(key, `audio/bgm/${filename}`);
}
preload() {
// Load menu images
this.loadImage('bg', 'ui');
this.loadImage('bg_command', 'ui');
this.loadImage('bg_fight', 'ui');
this.loadAtlas('prompt', 'ui');
this.loadImage('cursor', 'ui');
this.loadImage('pbinfo_player', 'ui');
this.loadImage('pbinfo_enemy', 'ui');
this.loadImage('overlay_lv', 'ui');
this.loadAtlas('numbers', 'ui');
this.loadAtlas('overlay_hp', 'ui');
this.loadImage('overlay_exp', 'ui');
2023-04-18 06:32:26 +01:00
this.loadImage('icon_owned', 'ui');
2023-03-28 19:54:52 +01:00
this.loadImage('level_up_stats', 'ui');
2023-04-02 03:59:07 +01:00
this.loadImage('ball_window', 'ui');
2023-03-28 19:54:52 +01:00
this.loadImage('boolean_window', 'ui');
this.loadImage('party_bg', 'ui');
this.loadAtlas('party_slot_main', 'ui');
this.loadAtlas('party_slot', 'ui');
this.loadImage('party_slot_overlay_lv', 'ui');
this.loadImage('party_slot_hp_bar', 'ui');
this.loadAtlas('party_slot_hp_overlay', 'ui');
this.loadAtlas('party_pb', 'ui');
this.loadImage('party_message', 'ui');
this.loadImage('party_message_large', 'ui');
this.loadImage('party_message_options', 'ui');
this.loadImage('party_options_top', 'ui');
this.loadImage('party_options_center', 'ui');
this.loadImage('party_options_bottom', 'ui');
2023-03-28 19:54:52 +01:00
this.loadAtlas('party_cancel', 'ui');
this.loadImage('summary_bg', 'ui');
2023-04-06 15:05:12 +01:00
this.loadImage('summary_overlay_shiny', 'ui');
this.loadImage('summary_profile', 'ui');
this.loadImage('summary_moves', 'ui');
this.loadImage('summary_moves_effect', 'ui');
this.loadImage('summary_moves_overlay_row', 'ui');
2023-04-09 01:35:45 +01:00
this.loadImage('summary_moves_overlay_pp', 'ui');
this.loadAtlas('summary_moves_cursor', 'ui');
2023-04-12 05:37:56 +01:00
this.loadImage('biome_select_window_2', 'ui');
this.loadImage('biome_select_window_3', 'ui');
2023-04-13 00:09:15 +01:00
this.loadImage('starter_select_bg', 'ui');
this.loadImage('starter_select_message', 'ui');
this.loadImage('starter_select_cursor', 'ui');
this.loadImage('starter_select_cursor_highlight', 'ui');
this.loadImage('starter_select_gen_cursor', 'ui');
this.loadImage('starter_select_gen_cursor_highlight', 'ui');
2023-03-28 19:54:52 +01:00
// Load arena images
2023-03-31 04:02:35 +01:00
Utils.getEnumValues(Biome).map(at => {
const atKey = Biome[at].toLowerCase();
2023-03-29 23:55:41 +01:00
this.loadImage(`${atKey}_bg`, 'arenas', `${atKey}_bg.png`);
this.loadImage(`${atKey}_a`, 'arenas', `${atKey}_a.png`);
this.loadImage(`${atKey}_b`, 'arenas', `${atKey}_b.png`);
});
2023-03-28 19:54:52 +01:00
// Load trainer images
this.loadImage('trainer_m', 'trainer');
this.loadAtlas('trainer_m_pb', 'trainer');
// Load pokemon-related images
this.loadImage(`pkmn__back__sub`, 'pokemon/back', 'sub.png');
this.loadImage(`pkmn__sub`, 'pokemon', 'sub.png');
2023-04-11 04:15:06 +01:00
this.loadAtlas('battle_stats', 'effects');
2023-04-10 21:52:27 +01:00
this.loadAtlas('shiny', 'effects');
2023-04-10 12:59:00 +01:00
this.loadImage('evo_sparkle', 'effects');
this.load.video('evo_bg', 'images/effects/evo_bg.mp4', null, false, true);
2023-03-28 19:54:52 +01:00
this.loadAtlas('pb', '');
this.loadAtlas('items', '');
2023-04-01 01:19:57 +01:00
this.loadAtlas('types', '');
this.loadAtlas('statuses', '');
2023-04-09 01:35:45 +01:00
this.loadAtlas('categories', '');
2023-03-28 19:54:52 +01:00
for (let i = 0; i < 6; i++)
this.loadAtlas(`pokemon_icons_${i}`, 'ui');
this.loadSe('select');
this.loadSe('menu_open');
this.loadSe('hit');
this.loadSe('hit_strong');
this.loadSe('hit_weak');
2023-04-11 04:15:06 +01:00
this.loadSe('stat_up');
this.loadSe('stat_down');
2023-03-28 19:54:52 +01:00
this.loadSe('faint');
this.loadSe('flee');
2023-04-12 17:48:02 +01:00
this.loadSe('low_hp');
2023-03-28 19:54:52 +01:00
this.loadSe('exp');
this.loadSe('level_up');
2023-04-10 18:54:06 +01:00
this.loadSe('sparkle');
2023-03-28 19:54:52 +01:00
this.loadSe('restore');
2023-04-10 18:54:06 +01:00
this.loadSe('shine');
this.loadSe('charge');
this.loadSe('beam');
2023-04-12 17:48:02 +01:00
this.loadSe('upgrade');
2023-03-28 19:54:52 +01:00
this.loadSe('error');
this.loadSe('pb');
this.loadSe('pb_rel');
this.loadSe('pb_throw');
this.loadSe('pb_bounce_1');
this.loadSe('pb_bounce_2');
this.loadSe('pb_move');
this.loadSe('pb_catch');
this.loadSe('pb_lock');
2023-04-13 00:09:15 +01:00
this.loadBgm('menu');
2023-03-28 19:54:52 +01:00
this.loadBgm('level_up_fanfare');
2023-04-10 12:59:00 +01:00
this.loadBgm('evolution');
this.loadBgm('evolution_fanfare');
2023-04-04 01:47:41 +01:00
2023-04-06 16:30:22 +01:00
//this.load.glsl('sprite', 'shaders/sprite.frag');
2023-04-04 01:47:41 +01:00
populateAnims();
2023-03-28 19:54:52 +01:00
}
create() {
2023-04-12 16:30:47 +01:00
initGameSpeed.apply(this);
this.setupControls();
2023-03-28 19:54:52 +01:00
this.load.setBaseURL();
2023-04-04 01:47:41 +01:00
//this.spritePipeline = (this.renderer as Phaser.Renderer.WebGL.WebGLRenderer).pipelines.get('Sprite') as SpritePipeline;
this.time.delayedCall(20, () => this.launchBattle());
}
update() {
this.checkInput();
}
launchBattle() {
2023-03-28 19:54:52 +01:00
const field = this.add.container(0, 0);
field.setScale(6);
this.field = field;
const fieldUI = this.add.container(0, this.game.canvas.height);
2023-04-11 04:15:06 +01:00
fieldUI.setDepth(1);
2023-03-28 19:54:52 +01:00
fieldUI.setScale(6);
this.fieldUI = fieldUI;
const uiContainer = this.add.container(0, 0);
2023-04-11 04:15:06 +01:00
uiContainer.setDepth(2);
2023-03-28 19:54:52 +01:00
uiContainer.setScale(6);
this.uiContainer = uiContainer;
this.modifiers = [];
this.modifierBar = new ModifierBar(this);
2023-03-29 05:31:25 +01:00
this.add.existing(this.modifierBar);
uiContainer.add(this.modifierBar);
2023-03-28 19:54:52 +01:00
this.party = [];
2023-03-29 05:31:25 +01:00
let loadPokemonAssets = [];
const isRandom = this.isButtonPressed(Button.RANDOM); // For testing purposes
2023-04-14 23:21:33 +01:00
this.quickStart = this.quickStart || isRandom || this.isButtonPressed(Button.QUICK_START);
2023-04-12 05:37:56 +01:00
if (isRandom) {
const biomes = Utils.getEnumValues(Biome);
this.newArena(biomes[Utils.randInt(biomes.length)]);
2023-04-12 05:37:56 +01:00
} else
this.newArena(Biome.PLAINS);
2023-04-12 05:37:56 +01:00
const biomeKey = this.arena.getBiomeKey();
this.arenaBg = this.add.sprite(0, 0, `${biomeKey}_bg`);
this.arenaBgTransition = this.add.sprite(0, 0, `${biomeKey}_bg`);
this.arenaPlayer = this.add.sprite(340, 20, `${biomeKey}_a`);
this.arenaPlayerTransition = this.add.sprite(40, 20, `${biomeKey}_a`);
this.arenaEnemy = this.add.sprite(-240, 13, `${biomeKey}_b`);
this.arenaNextEnemy = this.add.sprite(-240, 13, `${biomeKey}_b`);
this.arenaBgTransition.setVisible(false);
this.arenaPlayerTransition.setVisible(false);
[this.arenaBg, this.arenaBgTransition, this.arenaPlayer, this.arenaPlayerTransition, this.arenaEnemy, this.arenaNextEnemy].forEach(a => {
a.setOrigin(0, 0);
field.add(a);
});
2023-04-13 00:09:15 +01:00
if (this.quickStart) {
for (let s = 0; s < 3; s++) {
const playerSpecies = !isRandom ? getPokemonSpecies(s === 0 ? Species.TORCHIC : s === 1 ? Species.TREECKO : Species.MUDKIP) : this.randomSpecies(5);
const playerPokemon = new PlayerPokemon(this, playerSpecies, 5, 0);
2023-04-13 00:09:15 +01:00
playerPokemon.setVisible(false);
loadPokemonAssets.push(playerPokemon.loadAssets());
2023-03-28 19:54:52 +01:00
2023-04-13 00:09:15 +01:00
this.party.push(playerPokemon);
}
2023-03-28 19:54:52 +01:00
}
const trainerPbFrameNames = this.anims.generateFrameNames('trainer_m_pb', { zeroPad: 2, start: 1, end: 12 });
this.anims.create({
key: 'trainer_m_pb',
frames: trainerPbFrameNames,
frameRate: 16
});
const trainer = this.add.sprite(406, 132, 'trainer_m');
trainer.setOrigin(0.5, 1);
field.add(trainer);
this.trainer = trainer;
this.anims.create({
key: 'prompt',
frames: this.anims.generateFrameNumbers('prompt', { start: 1, end: 4 }),
frameRate: 6,
repeat: -1,
showOnStart: true
});
const ui = new UI(this);
this.uiContainer.add(ui);
this.ui = ui;
ui.setup();
2023-04-12 00:08:03 +01:00
Promise.all([ Promise.all(loadPokemonAssets), initCommonAnims().then(() => loadCommonAnimAssets(this, true)) ]).then(() => {
2023-04-09 05:22:14 +01:00
if (enableAuto)
initAutoPlay.apply(this);
2023-04-02 03:59:07 +01:00
this.pokeballCounts[PokeballType.POKEBALL] += 5;
2023-03-29 05:31:25 +01:00
2023-03-31 21:04:39 +01:00
this.newBattle();
2023-03-29 05:31:25 +01:00
this.shiftPhase();
});
2023-03-28 19:54:52 +01:00
}
setupControls() {
const keyCodes = Phaser.Input.Keyboard.KeyCodes;
const keyConfig = {
[Button.UP]: [keyCodes.UP, keyCodes.W],
[Button.DOWN]: [keyCodes.DOWN, keyCodes.S],
[Button.LEFT]: [keyCodes.LEFT, keyCodes.A],
[Button.RIGHT]: [keyCodes.RIGHT, keyCodes.D],
[Button.ACTION]: [keyCodes.ENTER, keyCodes.SPACE, keyCodes.Z],
[Button.CANCEL]: [keyCodes.BACKSPACE, keyCodes.ESC, keyCodes.X],
[Button.CYCLE_SHINY]: [keyCodes.R],
[Button.CYCLE_FORM]: [keyCodes.F],
[Button.CYCLE_GENDER]: [keyCodes.G],
2023-04-13 00:09:15 +01:00
[Button.QUICK_START]: [keyCodes.Q],
[Button.RANDOM]: [keyCodes.R],
[Button.AUTO]: [keyCodes.F2],
[Button.SPEED_UP]: [keyCodes.PLUS],
[Button.SLOW_DOWN]: [keyCodes.MINUS]
};
this.buttonKeys = [];
for (let b of Utils.getEnumValues(Button)) {
const keys: Phaser.Input.Keyboard.Key[] = [];
if (keyConfig.hasOwnProperty(b)) {
for (let k of keyConfig[b])
keys.push(this.input.keyboard.addKey(k));
}
this.buttonKeys[b] = keys;
}
2023-03-28 19:54:52 +01:00
}
getParty(): PlayerPokemon[] {
return this.party;
}
getPlayerPokemon(): PlayerPokemon {
return this.getParty()[0];
}
getEnemyPokemon(): EnemyPokemon {
2023-03-31 04:02:35 +01:00
return this.currentBattle.enemyPokemon;
2023-03-28 19:54:52 +01:00
}
2023-03-31 21:04:39 +01:00
newBattle(): Battle {
if (this.currentBattle) {
this.getEnemyPokemon().destroy();
if (this.currentBattle.waveIndex % 10)
this.pushPhase(new NextEncounterPhase(this));
2023-03-31 21:04:39 +01:00
else {
this.pushPhase(new SelectBiomePhase(this));
this.pushPhase(new NewBiomeEncounterPhase(this));
2023-03-31 21:04:39 +01:00
}
} else {
2023-04-13 00:09:15 +01:00
if (!this.quickStart) {
this.arena.preloadBgm();
this.pushPhase(new SelectStarterPhase(this));
} else
this.arena.playBgm();
2023-03-31 21:04:39 +01:00
this.pushPhase(new EncounterPhase(this));
this.pushPhase(new SummonPhase(this));
}
this.currentBattle = new Battle((this.currentBattle?.waveIndex || 0) + 1);
2023-03-31 04:02:35 +01:00
return this.currentBattle;
2023-03-28 19:54:52 +01:00
}
newArena(biome: Biome): Arena {
this.arena = new Arena(this, biome, Biome[biome].toLowerCase());
2023-03-31 21:04:39 +01:00
return this.arena;
2023-03-28 19:54:52 +01:00
}
2023-03-31 21:04:39 +01:00
randomSpecies(level: integer, fromArenaPool?: boolean): PokemonSpecies {
return fromArenaPool
? this.arena.randomSpecies(1, level)
: getPokemonSpecies(allSpecies[(Utils.randInt(allSpecies.length)) - 1].getSpeciesForLevel(level));
2023-03-28 19:54:52 +01:00
}
checkInput(): boolean {
if (this.blockInput)
return;
if (this.isButtonPressed(Button.UP))
this.ui.processInput(Button.UP);
else if (this.isButtonPressed(Button.DOWN))
this.ui.processInput(Button.DOWN);
else if (this.isButtonPressed(Button.LEFT))
this.ui.processInput(Button.LEFT);
else if (this.isButtonPressed(Button.RIGHT))
this.ui.processInput(Button.RIGHT);
else if (this.isButtonPressed(Button.ACTION))
this.ui.processInput(Button.ACTION);
else if (this.isButtonPressed(Button.CANCEL))
this.ui.processInput(Button.CANCEL);
else if (this.ui?.getHandler() instanceof StarterSelectUiHandler) {
if (this.isButtonPressed(Button.CYCLE_SHINY))
this.ui.processInput(Button.CYCLE_SHINY);
else if (this.isButtonPressed(Button.CYCLE_FORM))
this.ui.processInput(Button.CYCLE_FORM);
else if (this.isButtonPressed(Button.CYCLE_GENDER))
this.ui.processInput(Button.CYCLE_GENDER);
else
return;
}
2023-04-12 16:30:47 +01:00
else if (this.isButtonPressed(Button.SPEED_UP)) {
if (!this.auto) {
if (this.gameSpeed < 2)
this.gameSpeed += 0.25;
} else if (this.gameSpeed < 20)
this.gameSpeed++;
} else if (this.isButtonPressed(Button.SLOW_DOWN)) {
if (this.gameSpeed > 1) {
if (!this.auto)
this.gameSpeed -= 0.25;
else
this.gameSpeed--;
2023-04-09 05:22:14 +01:00
}
2023-04-12 16:30:47 +01:00
} else if (enableAuto) {
if (this.isButtonPressed(Button.AUTO)) {
this.auto = !this.auto;
if (this.auto)
this.gameSpeed = Math.floor(this.gameSpeed);
else if (this.gameSpeed > 2)
this.gameSpeed = 2;
} else
return;
2023-04-09 05:22:14 +01:00
} else
2023-03-28 19:54:52 +01:00
return;
this.blockInput = true;
2023-04-12 16:30:47 +01:00
this.time.delayedCall(new Utils.FixedInt(250) as unknown as integer, () => this.blockInput = false);
2023-03-28 19:54:52 +01:00
}
isButtonPressed(button: Button): boolean {
return this.buttonKeys[button].filter(k => k.isDown).length >= 1;
}
2023-04-10 18:54:06 +01:00
playBgm(bgmName?: string): void {
if (!bgmName && this.bgm) {
this.bgm.play({
volume: 1
});
return;
}
2023-03-31 21:04:39 +01:00
if (this.bgm && this.bgm.isPlaying)
2023-03-28 19:54:52 +01:00
this.bgm.stop();
this.bgm = this.sound.add(bgmName, { loop: true });
this.bgm.play();
2023-03-28 19:54:52 +01:00
}
pauseBgm(): void {
if (this.bgm)
this.bgm.pause();
}
resumeBgm(): void {
if (this.bgm && this.bgm.isPaused)
this.bgm.resume();
}
2023-04-10 18:54:06 +01:00
fadeOutBgm(destroy?: boolean): void {
this.arena.fadeOutBgm(500, destroy);
}
2023-04-18 06:32:26 +01:00
playSoundWithoutBgm(soundName: string, pauseDuration?: integer): void {
this.pauseBgm();
this.sound.play(soundName);
const sound = this.sound.get(soundName);
if (this.bgmResumeTimer)
this.bgmResumeTimer.destroy();
this.bgmResumeTimer = this.time.delayedCall((pauseDuration || (sound.totalDuration * 1000)), () => {
this.resumeBgm();
this.bgmResumeTimer = null;
});
}
2023-03-28 19:54:52 +01:00
getCurrentPhase(): BattlePhase {
return this.currentPhase;
}
pushPhase(phase: BattlePhase): void {
this.phaseQueue.push(phase);
}
unshiftPhase(phase: BattlePhase): void {
this.phaseQueuePrepend.push(phase);
}
clearPhaseQueue(): void {
this.phaseQueue.splice(0, this.phaseQueue.length);
}
shiftPhase(): void {
if (this.phaseQueuePrepend.length) {
while (this.phaseQueuePrepend.length)
this.phaseQueue.unshift(this.phaseQueuePrepend.pop());
}
if (!this.phaseQueue.length)
this.populatePhaseQueue();
this.currentPhase = this.phaseQueue.shift();
this.currentPhase.start();
}
populatePhaseQueue(): void {
this.phaseQueue.push(new CommandPhase(this));
}
2023-04-10 00:15:21 +01:00
addModifier(modifier: Modifier, virtual?: boolean): Promise<void> {
2023-04-04 23:28:21 +01:00
return new Promise(resolve => {
2023-04-10 00:15:21 +01:00
if (modifier instanceof PersistentModifier) {
if ((modifier as PersistentModifier).add(this.modifiers, !!virtual) && !virtual)
this.sound.play('restore');
2023-03-28 19:54:52 +01:00
2023-04-10 00:15:21 +01:00
if (!virtual)
this.updateModifiers().then(() => resolve());
} else if (modifier instanceof ConsumableModifier) {
this.sound.play('restore');
2023-03-28 19:54:52 +01:00
2023-04-10 00:15:21 +01:00
if (modifier instanceof ConsumablePokemonModifier) {
for (let p in this.party) {
const pokemon = this.party[p];
2023-04-04 23:28:21 +01:00
2023-04-10 00:15:21 +01:00
const args: any[] = [ pokemon ];
if (modifier instanceof PokemonHpRestoreModifier) {
const hpRestoreMultiplier = new Utils.IntegerHolder(1);
this.applyModifiers(HealingBoosterModifier, hpRestoreMultiplier);
args.push(hpRestoreMultiplier.value);
}
2023-04-11 16:04:39 +01:00
2023-04-04 23:28:21 +01:00
if (modifier.shouldApply(args))
modifier.apply(args);
}
2023-04-10 00:15:21 +01:00
Promise.allSettled(this.party.map(p => p.updateInfo())).then(() => resolve());
} else {
const args = [ this ];
if (modifier.shouldApply(args))
modifier.apply(args);
resolve();
}
}
});
}
2023-03-28 19:54:52 +01:00
2023-04-11 14:41:11 +01:00
removePartyMemberModifiers(partyMemberIndex: integer): Promise<void> {
2023-04-10 00:15:21 +01:00
return new Promise(resolve => {
2023-04-11 14:41:11 +01:00
const pokemonId = this.getParty()[partyMemberIndex].id;
const modifiersToRemove = this.modifiers.filter(m => (m instanceof PokemonHeldItemModifier) && (m as PokemonHeldItemModifier).pokemonId === pokemonId);
for (let m of modifiersToRemove)
this.modifiers.splice(this.modifiers.indexOf(m), 1);
this.updateModifiers().then(() => resolve());
2023-04-10 00:15:21 +01:00
});
}
2023-04-04 23:28:21 +01:00
2023-04-10 00:15:21 +01:00
updateModifiers(): Promise<void> {
return new Promise(resolve => {
for (let modifier of this.modifiers) {
if (modifier instanceof PersistentModifier)
(modifier as PersistentModifier).virtualStackCount = 0;
}
this.applyModifiers(PartyShareModifier, this, this.modifiers);
const modifiers = this.modifiers.slice(0);
for (let modifier of modifiers) {
if (!modifier.getStackCount())
this.modifiers.splice(this.modifiers.indexOf(modifier), 1);
2023-03-28 19:54:52 +01:00
}
2023-04-04 23:28:21 +01:00
2023-04-10 00:15:21 +01:00
this.updatePartyForModifiers().then(() => {
this.modifierBar.updateModifiers(this.modifiers);
2023-04-04 23:28:21 +01:00
resolve();
2023-04-10 00:15:21 +01:00
});
2023-04-04 23:28:21 +01:00
});
2023-03-28 19:54:52 +01:00
}
2023-04-11 14:41:11 +01:00
updatePartyForModifiers(): Promise<void> {
return new Promise(resolve => {
Promise.allSettled(this.party.map(p => {
p.calculateStats();
return p.updateInfo();
})).then(() => resolve());
});
}
removeModifier(modifier: PersistentModifier): boolean {
const modifierIndex = this.modifiers.indexOf(modifier);
if (modifierIndex > -1) {
this.modifiers.splice(modifierIndex, 1);
return true;
}
return false;
}
getModifiers(modifierType: { new(...args: any[]): Modifier }): Modifier[] {
return this.modifiers.filter(m => m instanceof modifierType);
2023-03-31 04:02:35 +01:00
}
2023-04-14 23:21:33 +01:00
findModifier(modifierFilter: ModifierPredicate): Modifier {
return this.modifiers.find(m => (modifierFilter as ModifierPredicate)(m));
}
2023-03-28 19:54:52 +01:00
applyModifiers(modifierType: { new(...args: any[]): Modifier }, ...args: any[]): void {
const modifiers = this.modifiers.filter(m => m instanceof modifierType && m.shouldApply(args));
for (let modifier of modifiers) {
if (modifier.apply(args))
console.log('Applied', modifier.type.name);
}
}
}