mirror of
https://github.com/pagefaultgames/pokerogue.git
synced 2025-10-23 21:45:50 +02:00
* Rename `Abilities` to `AbilityId` * Rename `abilities.ts` to `ability-id.ts` * Rename `Moves` to `MoveId` * Rename `moves.ts` to `move-id.ts` * Rename `Species` to `SpeciesId` * Rename `species.ts` to `species-id.ts` * Rename `Biome` to `BiomeId` * Rename `biome.ts` to `biome-id.ts` * Replace `Abilities` with `AbilityId` in comments * Replace `Biome` with `BiomeId` in comments * Replace `Moves` with `MoveId` in comments * Replace `Species` with `SpeciesId` in comments
155 lines
4.9 KiB
TypeScript
155 lines
4.9 KiB
TypeScript
import type BattleScene from "#app/battle-scene";
|
|
import { ArenaTagSide } from "#app/data/arena-tag";
|
|
import type Move from "#app/data/moves/move";
|
|
import { CritOnlyAttr } from "#app/data/moves/move";
|
|
import { allMoves } from "#app/data/data-lists";
|
|
import { AbilityId } from "#enums/ability-id";
|
|
import { ArenaTagType } from "#app/enums/arena-tag-type";
|
|
import type Pokemon from "#app/field/pokemon";
|
|
import { TurnEndPhase } from "#app/phases/turn-end-phase";
|
|
import { NumberHolder } from "#app/utils/common";
|
|
import { MoveId } from "#enums/move-id";
|
|
import { SpeciesId } from "#enums/species-id";
|
|
import GameManager from "#test/testUtils/gameManager";
|
|
import Phaser from "phaser";
|
|
import { afterEach, beforeAll, beforeEach, describe, expect, it } from "vitest";
|
|
|
|
let globalScene: BattleScene;
|
|
|
|
describe("Moves - Reflect", () => {
|
|
let phaserGame: Phaser.Game;
|
|
let game: GameManager;
|
|
const singleBattleMultiplier = 0.5;
|
|
const doubleBattleMultiplier = 2732 / 4096;
|
|
|
|
beforeAll(() => {
|
|
phaserGame = new Phaser.Game({
|
|
type: Phaser.HEADLESS,
|
|
});
|
|
});
|
|
|
|
afterEach(() => {
|
|
game.phaseInterceptor.restoreOg();
|
|
});
|
|
|
|
beforeEach(() => {
|
|
game = new GameManager(phaserGame);
|
|
globalScene = game.scene;
|
|
game.override
|
|
.battleStyle("single")
|
|
.ability(AbilityId.NONE)
|
|
.moveset([MoveId.ABSORB, MoveId.ROCK_SLIDE, MoveId.TACKLE])
|
|
.enemyLevel(100)
|
|
.enemySpecies(SpeciesId.MAGIKARP)
|
|
.enemyMoveset(MoveId.REFLECT)
|
|
.disableCrits();
|
|
});
|
|
|
|
it("reduces damage of physical attacks by half in a single battle", async () => {
|
|
const moveToUse = MoveId.TACKLE;
|
|
await game.classicMode.startBattle([SpeciesId.SHUCKLE]);
|
|
|
|
game.move.select(moveToUse);
|
|
|
|
await game.phaseInterceptor.to(TurnEndPhase);
|
|
const mockedDmg = getMockedMoveDamage(
|
|
game.scene.getEnemyPokemon()!,
|
|
game.scene.getPlayerPokemon()!,
|
|
allMoves[moveToUse],
|
|
);
|
|
|
|
expect(mockedDmg).toBe(allMoves[moveToUse].power * singleBattleMultiplier);
|
|
});
|
|
|
|
it("reduces damage of physical attacks by a third in a double battle", async () => {
|
|
game.override.battleStyle("double");
|
|
|
|
const moveToUse = MoveId.ROCK_SLIDE;
|
|
await game.classicMode.startBattle([SpeciesId.SHUCKLE, SpeciesId.SHUCKLE]);
|
|
|
|
game.move.select(moveToUse);
|
|
game.move.select(moveToUse, 1);
|
|
|
|
await game.phaseInterceptor.to(TurnEndPhase);
|
|
const mockedDmg = getMockedMoveDamage(
|
|
game.scene.getEnemyPokemon()!,
|
|
game.scene.getPlayerPokemon()!,
|
|
allMoves[moveToUse],
|
|
);
|
|
|
|
expect(mockedDmg).toBe(allMoves[moveToUse].power * doubleBattleMultiplier);
|
|
});
|
|
|
|
it("does not affect special attacks", async () => {
|
|
const moveToUse = MoveId.ABSORB;
|
|
await game.classicMode.startBattle([SpeciesId.SHUCKLE]);
|
|
|
|
game.move.select(moveToUse);
|
|
|
|
await game.phaseInterceptor.to(TurnEndPhase);
|
|
|
|
const mockedDmg = getMockedMoveDamage(
|
|
game.scene.getEnemyPokemon()!,
|
|
game.scene.getPlayerPokemon()!,
|
|
allMoves[moveToUse],
|
|
);
|
|
|
|
expect(mockedDmg).toBe(allMoves[moveToUse].power);
|
|
});
|
|
|
|
it("does not affect critical hits", async () => {
|
|
game.override.moveset([MoveId.WICKED_BLOW]);
|
|
const moveToUse = MoveId.WICKED_BLOW;
|
|
await game.classicMode.startBattle([SpeciesId.SHUCKLE]);
|
|
|
|
game.move.select(moveToUse);
|
|
await game.phaseInterceptor.to(TurnEndPhase);
|
|
|
|
const mockedDmg = getMockedMoveDamage(
|
|
game.scene.getEnemyPokemon()!,
|
|
game.scene.getPlayerPokemon()!,
|
|
allMoves[moveToUse],
|
|
);
|
|
|
|
expect(mockedDmg).toBe(allMoves[moveToUse].power);
|
|
});
|
|
|
|
it("does not affect critical hits", async () => {
|
|
game.override.moveset([MoveId.WICKED_BLOW]);
|
|
const moveToUse = MoveId.WICKED_BLOW;
|
|
await game.classicMode.startBattle([SpeciesId.SHUCKLE]);
|
|
|
|
game.move.select(moveToUse);
|
|
await game.phaseInterceptor.to(TurnEndPhase);
|
|
|
|
const mockedDmg = getMockedMoveDamage(
|
|
game.scene.getEnemyPokemon()!,
|
|
game.scene.getPlayerPokemon()!,
|
|
allMoves[moveToUse],
|
|
);
|
|
expect(mockedDmg).toBe(allMoves[moveToUse].power);
|
|
});
|
|
});
|
|
|
|
/**
|
|
* Calculates the damage of a move multiplied by screen's multiplier, Reflect in this case {@linkcode MoveId.REFLECT}.
|
|
* Please note this does not consider other damage calculations except the screen multiplier.
|
|
*
|
|
* @param defender - The defending Pokémon.
|
|
* @param attacker - The attacking Pokémon.
|
|
* @param move - The move being used.
|
|
* @returns The calculated move damage considering any weakening effects.
|
|
*/
|
|
const getMockedMoveDamage = (defender: Pokemon, attacker: Pokemon, move: Move) => {
|
|
const multiplierHolder = new NumberHolder(1);
|
|
const side = defender.isPlayer() ? ArenaTagSide.PLAYER : ArenaTagSide.ENEMY;
|
|
|
|
if (globalScene.arena.getTagOnSide(ArenaTagType.REFLECT, side)) {
|
|
if (move.getAttrs(CritOnlyAttr).length === 0) {
|
|
globalScene.arena.applyTagsForSide(ArenaTagType.REFLECT, side, false, attacker, move.category, multiplierHolder);
|
|
}
|
|
}
|
|
|
|
return move.power * multiplierHolder.value;
|
|
};
|