2024-08-12 03:52:39 -07:00
|
|
|
import { afterEach, beforeAll, beforeEach, describe, expect, it } from "vitest";
|
2024-07-25 16:10:38 -07:00
|
|
|
import GameManager from "#test/utils/gameManager";
|
|
|
|
import { getMovePosition } from "#test/utils/gameManagerUtils";
|
2024-08-12 03:52:39 -07:00
|
|
|
import { Moves } from "#enums/moves";
|
|
|
|
import { Species } from "#enums/species";
|
|
|
|
import { StatusEffect } from "#app/data/status-effect.js";
|
2024-08-17 21:05:04 -07:00
|
|
|
import { CommandPhase, MoveEffectPhase, MoveEndPhase, TurnEndPhase, TurnInitPhase } from "#app/phases.js";
|
2024-08-12 03:52:39 -07:00
|
|
|
import { BattleStat } from "#app/data/battle-stat.js";
|
|
|
|
import { SPLASH_ONLY } from "../utils/testUtils";
|
2024-08-17 21:05:04 -07:00
|
|
|
import { Mode } from "#app/ui/ui.js";
|
2024-06-13 05:36:12 -04:00
|
|
|
|
|
|
|
const TIMEOUT = 20 * 1000;
|
|
|
|
|
2024-08-12 03:52:39 -07:00
|
|
|
describe("Abilities - Disguise", () => {
|
2024-06-13 05:36:12 -04:00
|
|
|
let phaserGame: Phaser.Game;
|
|
|
|
let game: GameManager;
|
2024-08-12 03:52:39 -07:00
|
|
|
const bustedForm = 1;
|
|
|
|
const disguisedForm = 0;
|
2024-06-13 05:36:12 -04:00
|
|
|
|
|
|
|
beforeAll(() => {
|
|
|
|
phaserGame = new Phaser.Game({
|
|
|
|
type: Phaser.HEADLESS,
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
afterEach(() => {
|
|
|
|
game.phaseInterceptor.restoreOg();
|
|
|
|
});
|
|
|
|
|
|
|
|
beforeEach(() => {
|
|
|
|
game = new GameManager(phaserGame);
|
2024-07-25 14:48:48 -07:00
|
|
|
game.override.battleType("single");
|
2024-06-13 05:36:12 -04:00
|
|
|
|
2024-08-12 03:52:39 -07:00
|
|
|
game.override.enemySpecies(Species.MIMIKYU);
|
|
|
|
game.override.enemyMoveset(SPLASH_ONLY);
|
|
|
|
|
|
|
|
game.override.starterSpecies(Species.REGIELEKI);
|
|
|
|
game.override.moveset([Moves.SHADOW_SNEAK, Moves.VACUUM_WAVE, Moves.TOXIC_THREAD, Moves.SPLASH]);
|
|
|
|
}, TIMEOUT);
|
|
|
|
|
2024-08-17 21:05:04 -07:00
|
|
|
it("takes no damage from attacking move and transforms to Busted form, takes 1/8 max HP damage from the disguise breaking", async () => {
|
2024-08-12 03:52:39 -07:00
|
|
|
await game.startBattle();
|
|
|
|
|
|
|
|
const mimikyu = game.scene.getEnemyPokemon()!;
|
|
|
|
const maxHp = mimikyu.getMaxHp();
|
|
|
|
const disguiseDamage = Math.floor(maxHp / 8);
|
|
|
|
|
|
|
|
expect(mimikyu.formIndex).toBe(disguisedForm);
|
|
|
|
|
|
|
|
game.doAttack(getMovePosition(game.scene, 0, Moves.SHADOW_SNEAK));
|
|
|
|
|
|
|
|
await game.phaseInterceptor.to(MoveEndPhase);
|
|
|
|
|
|
|
|
expect(mimikyu.hp).equals(maxHp - disguiseDamage);
|
|
|
|
expect(mimikyu.formIndex).toBe(bustedForm);
|
|
|
|
}, TIMEOUT);
|
|
|
|
|
|
|
|
it("doesn't break disguise when attacked with ineffective move", async () => {
|
|
|
|
await game.startBattle();
|
|
|
|
|
|
|
|
const mimikyu = game.scene.getEnemyPokemon()!;
|
|
|
|
|
|
|
|
expect(mimikyu.formIndex).toBe(disguisedForm);
|
|
|
|
|
|
|
|
game.doAttack(getMovePosition(game.scene, 0, Moves.VACUUM_WAVE));
|
|
|
|
|
|
|
|
await game.phaseInterceptor.to(MoveEndPhase);
|
|
|
|
|
|
|
|
expect(mimikyu.formIndex).toBe(disguisedForm);
|
|
|
|
}, TIMEOUT);
|
|
|
|
|
|
|
|
it("takes no damage from the first hit of a multihit move and transforms to Busted form, then takes damage from the second hit", async () => {
|
|
|
|
game.override.moveset([Moves.SURGING_STRIKES]);
|
|
|
|
game.override.enemyLevel(5);
|
|
|
|
await game.startBattle();
|
|
|
|
|
|
|
|
const mimikyu = game.scene.getEnemyPokemon()!;
|
|
|
|
const maxHp = mimikyu.getMaxHp();
|
|
|
|
const disguiseDamage = Math.floor(maxHp / 8);
|
|
|
|
|
|
|
|
expect(mimikyu.formIndex).toBe(disguisedForm);
|
|
|
|
|
|
|
|
game.doAttack(getMovePosition(game.scene, 0, Moves.SURGING_STRIKES));
|
|
|
|
|
|
|
|
// First hit
|
|
|
|
await game.phaseInterceptor.to(MoveEffectPhase);
|
|
|
|
expect(mimikyu.hp).equals(maxHp - disguiseDamage);
|
|
|
|
expect(mimikyu.formIndex).toBe(disguisedForm);
|
|
|
|
|
|
|
|
// Second hit
|
|
|
|
await game.phaseInterceptor.to(MoveEffectPhase);
|
|
|
|
expect(mimikyu.hp).lessThan(maxHp - disguiseDamage);
|
|
|
|
expect(mimikyu.formIndex).toBe(bustedForm);
|
|
|
|
}, TIMEOUT);
|
|
|
|
|
|
|
|
it("takes effects from status moves and damage from status effects", async () => {
|
|
|
|
await game.startBattle();
|
|
|
|
|
|
|
|
const mimikyu = game.scene.getEnemyPokemon()!;
|
|
|
|
expect(mimikyu.hp).toBe(mimikyu.getMaxHp());
|
|
|
|
|
|
|
|
game.doAttack(getMovePosition(game.scene, 0, Moves.TOXIC_THREAD));
|
|
|
|
|
|
|
|
await game.phaseInterceptor.to(TurnEndPhase);
|
|
|
|
|
|
|
|
expect(mimikyu.formIndex).toBe(disguisedForm);
|
|
|
|
expect(mimikyu.status?.effect).toBe(StatusEffect.POISON);
|
|
|
|
expect(mimikyu.summonData.battleStats[BattleStat.SPD]).toBe(-1);
|
|
|
|
expect(mimikyu.hp).toBeLessThan(mimikyu.getMaxHp());
|
|
|
|
}, TIMEOUT);
|
|
|
|
|
|
|
|
it("persists form change when switched out", async () => {
|
|
|
|
game.override.enemyMoveset(Array(4).fill(Moves.SHADOW_SNEAK));
|
|
|
|
game.override.starterSpecies(0);
|
|
|
|
|
|
|
|
await game.startBattle([Species.MIMIKYU, Species.FURRET]);
|
|
|
|
|
|
|
|
const mimikyu = game.scene.getPlayerPokemon()!;
|
|
|
|
const maxHp = mimikyu.getMaxHp();
|
|
|
|
const disguiseDamage = Math.floor(maxHp / 8);
|
|
|
|
|
|
|
|
game.doAttack(getMovePosition(game.scene, 0, Moves.SPLASH));
|
|
|
|
|
|
|
|
await game.phaseInterceptor.to(TurnEndPhase);
|
|
|
|
|
|
|
|
expect(mimikyu.formIndex).toBe(bustedForm);
|
|
|
|
expect(mimikyu.hp).equals(maxHp - disguiseDamage);
|
|
|
|
|
|
|
|
await game.toNextTurn();
|
|
|
|
game.doSwitchPokemon(1);
|
|
|
|
|
|
|
|
await game.phaseInterceptor.to(TurnEndPhase);
|
|
|
|
|
|
|
|
expect(mimikyu.formIndex).toBe(bustedForm);
|
|
|
|
}, TIMEOUT);
|
|
|
|
|
2024-08-17 21:05:04 -07:00
|
|
|
it("persists form change when wave changes with no arena reset", async () => {
|
|
|
|
game.override.starterSpecies(0);
|
|
|
|
game.override.starterForms({
|
|
|
|
[Species.MIMIKYU]: bustedForm
|
|
|
|
});
|
|
|
|
await game.startBattle([Species.FURRET, Species.MIMIKYU]);
|
|
|
|
|
|
|
|
const mimikyu = game.scene.getParty()[1]!;
|
|
|
|
expect(mimikyu.formIndex).toBe(bustedForm);
|
|
|
|
|
|
|
|
game.doAttack(getMovePosition(game.scene, 0, Moves.SPLASH));
|
|
|
|
await game.doKillOpponents();
|
|
|
|
await game.toNextWave();
|
|
|
|
|
|
|
|
expect(mimikyu.formIndex).toBe(bustedForm);
|
|
|
|
}, TIMEOUT);
|
2024-08-12 03:52:39 -07:00
|
|
|
|
2024-08-17 21:05:04 -07:00
|
|
|
it("reverts to Disguised form on arena reset", async () => {
|
|
|
|
game.override.startingWave(4);
|
2024-08-12 03:52:39 -07:00
|
|
|
game.override.starterSpecies(Species.MIMIKYU);
|
|
|
|
game.override.starterForms({
|
|
|
|
[Species.MIMIKYU]: bustedForm
|
|
|
|
});
|
|
|
|
|
|
|
|
await game.startBattle();
|
|
|
|
|
|
|
|
const mimikyu = game.scene.getPlayerPokemon()!;
|
|
|
|
|
|
|
|
expect(mimikyu.formIndex).toBe(bustedForm);
|
|
|
|
|
|
|
|
game.doAttack(getMovePosition(game.scene, 0, Moves.SPLASH));
|
|
|
|
await game.doKillOpponents();
|
2024-08-17 21:05:04 -07:00
|
|
|
await game.toNextWave();
|
2024-08-12 03:52:39 -07:00
|
|
|
|
|
|
|
expect(mimikyu.formIndex).toBe(disguisedForm);
|
|
|
|
}, TIMEOUT);
|
2024-08-17 21:05:04 -07:00
|
|
|
|
|
|
|
it("reverts to Disguised form on biome change when fainted", async () => {
|
|
|
|
game.override.startingWave(10);
|
|
|
|
game.override.starterSpecies(0);
|
|
|
|
game.override.starterForms({
|
|
|
|
[Species.MIMIKYU]: bustedForm
|
|
|
|
});
|
|
|
|
|
|
|
|
await game.startBattle([Species.MIMIKYU, Species.FURRET]);
|
|
|
|
|
|
|
|
const mimikyu1 = game.scene.getPlayerPokemon()!;
|
|
|
|
|
|
|
|
expect(mimikyu1.formIndex).toBe(bustedForm);
|
|
|
|
|
|
|
|
game.doAttack(getMovePosition(game.scene, 0, Moves.SPLASH));
|
|
|
|
await game.killPokemon(mimikyu1);
|
|
|
|
game.doSelectPartyPokemon(1);
|
|
|
|
await game.toNextTurn();
|
|
|
|
game.doAttack(getMovePosition(game.scene, 0, Moves.SPLASH));
|
|
|
|
await game.doKillOpponents();
|
|
|
|
game.onNextPrompt("CheckSwitchPhase", Mode.CONFIRM, () => { // TODO: Make tests run in set mode instead of switch mode
|
|
|
|
game.setMode(Mode.MESSAGE);
|
|
|
|
game.endPhase();
|
|
|
|
}, () => game.isCurrentPhase(CommandPhase) || game.isCurrentPhase(TurnInitPhase));
|
|
|
|
|
|
|
|
game.onNextPrompt("CheckSwitchPhase", Mode.CONFIRM, () => {
|
|
|
|
game.setMode(Mode.MESSAGE);
|
|
|
|
game.endPhase();
|
|
|
|
}, () => game.isCurrentPhase(CommandPhase) || game.isCurrentPhase(TurnInitPhase));
|
|
|
|
await game.phaseInterceptor.to("PartyHealPhase");
|
|
|
|
|
|
|
|
expect(mimikyu1.formIndex).toBe(disguisedForm);
|
|
|
|
}, TIMEOUT);
|
2024-06-13 05:36:12 -04:00
|
|
|
});
|