mirror of
https://github.com/pagefaultgames/pokerogue.git
synced 2025-10-23 21:45:50 +02:00
* Add isPartner method to trainer class * Ensure force switches cannot pull pokemon from the wrong trainer * Add override for battle type * Fixup tests and broken assumptions * Make move fail override semi-invuln check Bandaid fix because move effect phase does not allow for the move to fail if all of its conditions fail * Restore overrides * Apply kev's suggestions from code review Co-authored-by: NightKev <34855794+DayKev@users.noreply.github.com> * Fix illusion test battle type invocation * Update struggle and healer tests to use battleStyle --------- Co-authored-by: NightKev <34855794+DayKev@users.noreply.github.com>
307 lines
11 KiB
TypeScript
307 lines
11 KiB
TypeScript
import { BattlerIndex } from "#app/battle";
|
|
import type Pokemon from "#app/field/pokemon";
|
|
import { Abilities } from "#enums/abilities";
|
|
import { BattlerTagType } from "#enums/battler-tag-type";
|
|
import { Moves } from "#enums/moves";
|
|
import { Species } from "#enums/species";
|
|
import { Stat } from "#enums/stat";
|
|
import { StatusEffect } from "#enums/status-effect";
|
|
import GameManager from "#test/testUtils/gameManager";
|
|
import Phaser from "phaser";
|
|
import { afterEach, beforeAll, beforeEach, describe, expect, it, vi } from "vitest";
|
|
|
|
describe("Abilities - Gulp Missile", () => {
|
|
let phaserGame: Phaser.Game;
|
|
let game: GameManager;
|
|
|
|
const NORMAL_FORM = 0;
|
|
const GULPING_FORM = 1;
|
|
const GORGING_FORM = 2;
|
|
|
|
/**
|
|
* Gets the effect damage of Gulp Missile
|
|
* See Gulp Missile {@link https://bulbapedia.bulbagarden.net/wiki/Gulp_Missile_(Ability)}
|
|
* @param {Pokemon} pokemon The pokemon taking the effect damage.
|
|
* @returns The effect damage of Gulp Missile
|
|
*/
|
|
const getEffectDamage = (pokemon: Pokemon): number => {
|
|
return Math.max(1, Math.floor((pokemon.getMaxHp() * 1) / 4));
|
|
};
|
|
|
|
beforeAll(() => {
|
|
phaserGame = new Phaser.Game({
|
|
type: Phaser.HEADLESS,
|
|
});
|
|
});
|
|
|
|
afterEach(() => {
|
|
game.phaseInterceptor.restoreOg();
|
|
});
|
|
|
|
beforeEach(() => {
|
|
game = new GameManager(phaserGame);
|
|
game.override
|
|
.disableCrits()
|
|
.battleStyle("single")
|
|
.moveset([Moves.SURF, Moves.DIVE, Moves.SPLASH, Moves.SUBSTITUTE])
|
|
.enemySpecies(Species.SNORLAX)
|
|
.enemyAbility(Abilities.BALL_FETCH)
|
|
.enemyMoveset(Moves.SPLASH)
|
|
.enemyLevel(5);
|
|
});
|
|
|
|
it("changes to Gulping Form if HP is over half when Surf or Dive is used", async () => {
|
|
await game.classicMode.startBattle([Species.CRAMORANT]);
|
|
const cramorant = game.scene.getPlayerPokemon()!;
|
|
|
|
game.move.select(Moves.DIVE);
|
|
await game.toNextTurn();
|
|
game.move.select(Moves.DIVE);
|
|
await game.phaseInterceptor.to("MoveEndPhase");
|
|
|
|
expect(cramorant.getHpRatio()).toBeGreaterThanOrEqual(0.5);
|
|
expect(cramorant.getTag(BattlerTagType.GULP_MISSILE_ARROKUDA)).toBeDefined();
|
|
expect(cramorant.formIndex).toBe(GULPING_FORM);
|
|
});
|
|
|
|
it("changes to Gorging Form if HP is under half when Surf or Dive is used", async () => {
|
|
await game.classicMode.startBattle([Species.CRAMORANT]);
|
|
const cramorant = game.scene.getPlayerPokemon()!;
|
|
|
|
vi.spyOn(cramorant, "getHpRatio").mockReturnValue(0.49);
|
|
expect(cramorant.getHpRatio()).toBe(0.49);
|
|
|
|
game.move.select(Moves.SURF);
|
|
await game.phaseInterceptor.to("MoveEndPhase");
|
|
|
|
expect(cramorant.getTag(BattlerTagType.GULP_MISSILE_PIKACHU)).toBeDefined();
|
|
expect(cramorant.formIndex).toBe(GORGING_FORM);
|
|
});
|
|
|
|
it("changes to base form when switched out after Surf or Dive is used", async () => {
|
|
await game.classicMode.startBattle([Species.CRAMORANT, Species.MAGIKARP]);
|
|
const cramorant = game.scene.getPlayerPokemon()!;
|
|
|
|
game.move.select(Moves.SURF);
|
|
await game.toNextTurn();
|
|
|
|
game.doSwitchPokemon(1);
|
|
await game.toNextTurn(); // form change is delayed until after end of turn
|
|
|
|
expect(cramorant.formIndex).toBe(NORMAL_FORM);
|
|
expect(cramorant.getTag(BattlerTagType.GULP_MISSILE_ARROKUDA)).toBeUndefined();
|
|
expect(cramorant.getTag(BattlerTagType.GULP_MISSILE_PIKACHU)).toBeUndefined();
|
|
});
|
|
|
|
it("changes form during Dive's charge turn", async () => {
|
|
await game.classicMode.startBattle([Species.CRAMORANT]);
|
|
const cramorant = game.scene.getPlayerPokemon()!;
|
|
|
|
game.move.select(Moves.DIVE);
|
|
await game.phaseInterceptor.to("MoveEndPhase");
|
|
|
|
expect(cramorant.getTag(BattlerTagType.GULP_MISSILE_ARROKUDA)).toBeDefined();
|
|
expect(cramorant.formIndex).toBe(GULPING_FORM);
|
|
});
|
|
|
|
it("deals 1/4 of the attacker's maximum HP when hit by a damaging attack", async () => {
|
|
game.override.enemyMoveset(Moves.TACKLE);
|
|
await game.classicMode.startBattle([Species.CRAMORANT]);
|
|
|
|
const enemy = game.scene.getEnemyPokemon()!;
|
|
vi.spyOn(enemy, "damageAndUpdate");
|
|
|
|
game.move.select(Moves.SURF);
|
|
await game.phaseInterceptor.to("TurnEndPhase");
|
|
|
|
expect(enemy.damageAndUpdate).toHaveReturnedWith(getEffectDamage(enemy));
|
|
});
|
|
|
|
it("does not have any effect when hit by non-damaging attack", async () => {
|
|
game.override.enemyMoveset(Moves.TAIL_WHIP);
|
|
await game.classicMode.startBattle([Species.CRAMORANT]);
|
|
|
|
const cramorant = game.scene.getPlayerPokemon()!;
|
|
vi.spyOn(cramorant, "getHpRatio").mockReturnValue(0.55);
|
|
|
|
game.move.select(Moves.SURF);
|
|
await game.phaseInterceptor.to("MoveEndPhase");
|
|
|
|
expect(cramorant.getTag(BattlerTagType.GULP_MISSILE_ARROKUDA)).toBeDefined();
|
|
expect(cramorant.formIndex).toBe(GULPING_FORM);
|
|
|
|
await game.phaseInterceptor.to("TurnEndPhase");
|
|
|
|
expect(cramorant.getTag(BattlerTagType.GULP_MISSILE_ARROKUDA)).toBeDefined();
|
|
expect(cramorant.formIndex).toBe(GULPING_FORM);
|
|
});
|
|
|
|
it("lowers attacker's DEF stat stage by 1 when hit in Gulping form", async () => {
|
|
game.override.enemyMoveset(Moves.TACKLE);
|
|
await game.classicMode.startBattle([Species.CRAMORANT]);
|
|
|
|
const cramorant = game.scene.getPlayerPokemon()!;
|
|
const enemy = game.scene.getEnemyPokemon()!;
|
|
|
|
vi.spyOn(enemy, "damageAndUpdate");
|
|
vi.spyOn(cramorant, "getHpRatio").mockReturnValue(0.55);
|
|
|
|
game.move.select(Moves.SURF);
|
|
await game.phaseInterceptor.to("MoveEndPhase");
|
|
|
|
expect(cramorant.getTag(BattlerTagType.GULP_MISSILE_ARROKUDA)).toBeDefined();
|
|
expect(cramorant.formIndex).toBe(GULPING_FORM);
|
|
|
|
await game.phaseInterceptor.to("TurnEndPhase");
|
|
|
|
expect(enemy.damageAndUpdate).toHaveReturnedWith(getEffectDamage(enemy));
|
|
expect(enemy.getStatStage(Stat.DEF)).toBe(-1);
|
|
expect(cramorant.getTag(BattlerTagType.GULP_MISSILE_ARROKUDA)).toBeUndefined();
|
|
expect(cramorant.formIndex).toBe(NORMAL_FORM);
|
|
});
|
|
|
|
it("paralyzes the enemy when hit in Gorging form", async () => {
|
|
game.override.enemyMoveset(Moves.TACKLE);
|
|
await game.classicMode.startBattle([Species.CRAMORANT]);
|
|
|
|
const cramorant = game.scene.getPlayerPokemon()!;
|
|
const enemy = game.scene.getEnemyPokemon()!;
|
|
|
|
vi.spyOn(enemy, "damageAndUpdate");
|
|
vi.spyOn(cramorant, "getHpRatio").mockReturnValue(0.45);
|
|
|
|
game.move.select(Moves.SURF);
|
|
await game.phaseInterceptor.to("MoveEndPhase");
|
|
|
|
expect(cramorant.getTag(BattlerTagType.GULP_MISSILE_PIKACHU)).toBeDefined();
|
|
expect(cramorant.formIndex).toBe(GORGING_FORM);
|
|
|
|
await game.phaseInterceptor.to("TurnEndPhase");
|
|
|
|
expect(enemy.damageAndUpdate).toHaveReturnedWith(getEffectDamage(enemy));
|
|
expect(enemy.status?.effect).toBe(StatusEffect.PARALYSIS);
|
|
expect(cramorant.getTag(BattlerTagType.GULP_MISSILE_PIKACHU)).toBeUndefined();
|
|
expect(cramorant.formIndex).toBe(NORMAL_FORM);
|
|
});
|
|
|
|
it("does not activate the ability when underwater", async () => {
|
|
game.override.enemyMoveset(Moves.SURF);
|
|
await game.classicMode.startBattle([Species.CRAMORANT]);
|
|
|
|
const cramorant = game.scene.getPlayerPokemon()!;
|
|
|
|
game.move.select(Moves.DIVE);
|
|
await game.phaseInterceptor.to("BerryPhase", false);
|
|
|
|
expect(cramorant.getTag(BattlerTagType.GULP_MISSILE_ARROKUDA)).toBeDefined();
|
|
expect(cramorant.formIndex).toBe(GULPING_FORM);
|
|
});
|
|
|
|
it("prevents effect damage but inflicts secondary effect on attacker with Magic Guard", async () => {
|
|
game.override.enemyMoveset(Moves.TACKLE).enemyAbility(Abilities.MAGIC_GUARD);
|
|
await game.classicMode.startBattle([Species.CRAMORANT]);
|
|
|
|
const cramorant = game.scene.getPlayerPokemon()!;
|
|
const enemy = game.scene.getEnemyPokemon()!;
|
|
|
|
vi.spyOn(cramorant, "getHpRatio").mockReturnValue(0.55);
|
|
|
|
game.move.select(Moves.SURF);
|
|
await game.phaseInterceptor.to("MoveEndPhase");
|
|
const enemyHpPreEffect = enemy.hp;
|
|
|
|
expect(cramorant.getTag(BattlerTagType.GULP_MISSILE_ARROKUDA)).toBeDefined();
|
|
expect(cramorant.formIndex).toBe(GULPING_FORM);
|
|
|
|
await game.phaseInterceptor.to("TurnEndPhase");
|
|
|
|
expect(enemy.hp).toBe(enemyHpPreEffect);
|
|
expect(enemy.getStatStage(Stat.DEF)).toBe(-1);
|
|
expect(cramorant.getTag(BattlerTagType.GULP_MISSILE_ARROKUDA)).toBeUndefined();
|
|
expect(cramorant.formIndex).toBe(NORMAL_FORM);
|
|
});
|
|
|
|
it("activates on faint", async () => {
|
|
game.override.enemyMoveset(Moves.THUNDERBOLT);
|
|
await game.classicMode.startBattle([Species.CRAMORANT]);
|
|
|
|
const cramorant = game.scene.getPlayerPokemon()!;
|
|
|
|
game.move.select(Moves.SURF);
|
|
await game.phaseInterceptor.to("FaintPhase");
|
|
|
|
expect(cramorant.hp).toBe(0);
|
|
expect(cramorant.getTag(BattlerTagType.GULP_MISSILE_ARROKUDA)).toBeUndefined();
|
|
expect(cramorant.formIndex).toBe(NORMAL_FORM);
|
|
expect(game.scene.getEnemyPokemon()!.getStatStage(Stat.DEF)).toBe(-1);
|
|
});
|
|
|
|
it("doesn't trigger if user is behind a substitute", async () => {
|
|
game.override.enemyAbility(Abilities.STURDY).enemyMoveset([Moves.SPLASH, Moves.POWER_TRIP]);
|
|
await game.classicMode.startBattle([Species.CRAMORANT]);
|
|
|
|
game.move.select(Moves.SURF);
|
|
await game.forceEnemyMove(Moves.SPLASH);
|
|
await game.toNextTurn();
|
|
|
|
expect(game.scene.getPlayerPokemon()!.formIndex).toBe(GULPING_FORM);
|
|
|
|
game.move.select(Moves.SUBSTITUTE);
|
|
await game.forceEnemyMove(Moves.POWER_TRIP);
|
|
await game.setTurnOrder([BattlerIndex.PLAYER, BattlerIndex.ENEMY]);
|
|
await game.toNextTurn();
|
|
|
|
expect(game.scene.getPlayerPokemon()!.formIndex).toBe(GULPING_FORM);
|
|
});
|
|
|
|
it("cannot be suppressed", async () => {
|
|
game.override.enemyMoveset(Moves.GASTRO_ACID);
|
|
await game.classicMode.startBattle([Species.CRAMORANT]);
|
|
|
|
const cramorant = game.scene.getPlayerPokemon()!;
|
|
vi.spyOn(cramorant, "getHpRatio").mockReturnValue(0.55);
|
|
|
|
game.move.select(Moves.SURF);
|
|
await game.phaseInterceptor.to("MoveEndPhase");
|
|
|
|
expect(cramorant.getTag(BattlerTagType.GULP_MISSILE_ARROKUDA)).toBeDefined();
|
|
expect(cramorant.formIndex).toBe(GULPING_FORM);
|
|
|
|
await game.phaseInterceptor.to("TurnEndPhase");
|
|
|
|
expect(cramorant.hasAbility(Abilities.GULP_MISSILE)).toBe(true);
|
|
expect(cramorant.getTag(BattlerTagType.GULP_MISSILE_ARROKUDA)).toBeDefined();
|
|
expect(cramorant.formIndex).toBe(GULPING_FORM);
|
|
});
|
|
|
|
it("cannot be swapped with another ability", async () => {
|
|
game.override.enemyMoveset(Moves.SKILL_SWAP);
|
|
await game.classicMode.startBattle([Species.CRAMORANT]);
|
|
|
|
const cramorant = game.scene.getPlayerPokemon()!;
|
|
vi.spyOn(cramorant, "getHpRatio").mockReturnValue(0.55);
|
|
|
|
game.move.select(Moves.SURF);
|
|
await game.phaseInterceptor.to("MoveEndPhase");
|
|
|
|
expect(cramorant.getTag(BattlerTagType.GULP_MISSILE_ARROKUDA)).toBeDefined();
|
|
expect(cramorant.formIndex).toBe(GULPING_FORM);
|
|
|
|
await game.phaseInterceptor.to("TurnEndPhase");
|
|
|
|
expect(cramorant.hasAbility(Abilities.GULP_MISSILE)).toBe(true);
|
|
expect(cramorant.getTag(BattlerTagType.GULP_MISSILE_ARROKUDA)).toBeDefined();
|
|
expect(cramorant.formIndex).toBe(GULPING_FORM);
|
|
});
|
|
|
|
it("cannot be copied", async () => {
|
|
game.override.enemyAbility(Abilities.TRACE);
|
|
|
|
await game.classicMode.startBattle([Species.CRAMORANT]);
|
|
game.move.select(Moves.SPLASH);
|
|
await game.phaseInterceptor.to("TurnStartPhase");
|
|
|
|
expect(game.scene.getEnemyPokemon()?.hasAbility(Abilities.GULP_MISSILE)).toBe(false);
|
|
});
|
|
});
|