2024-08-22 06:49:33 -07:00
|
|
|
import { ArenaTagSide } from "#app/data/arena-tag";
|
|
|
|
import Move, { allMoves } from "#app/data/move";
|
|
|
|
import { Abilities } from "#app/enums/abilities";
|
|
|
|
import { ArenaTagType } from "#app/enums/arena-tag-type";
|
|
|
|
import Pokemon from "#app/field/pokemon";
|
|
|
|
import { TurnEndPhase } from "#app/phases/turn-end-phase";
|
|
|
|
import { NumberHolder } from "#app/utils";
|
2024-06-22 22:09:23 +08:00
|
|
|
import { Moves } from "#enums/moves";
|
|
|
|
import { Species } from "#enums/species";
|
2024-08-22 06:49:33 -07:00
|
|
|
import GameManager from "#test/utils/gameManager";
|
2024-07-25 16:18:47 -07:00
|
|
|
import Phaser from "phaser";
|
|
|
|
import { afterEach, beforeAll, beforeEach, describe, expect, it } from "vitest";
|
2024-06-22 22:09:23 +08:00
|
|
|
|
|
|
|
|
|
|
|
describe("Moves - Light Screen", () => {
|
|
|
|
let phaserGame: Phaser.Game;
|
|
|
|
let game: GameManager;
|
|
|
|
const singleBattleMultiplier = 0.5;
|
2024-08-22 06:49:33 -07:00
|
|
|
const doubleBattleMultiplier = 2732 / 4096;
|
2024-06-22 22:09:23 +08: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-07-25 15:35:20 -07:00
|
|
|
game.override.ability(Abilities.NONE);
|
2024-07-25 15:51:05 -07:00
|
|
|
game.override.moveset([Moves.ABSORB, Moves.DAZZLING_GLEAM, Moves.TACKLE]);
|
2024-07-25 16:10:38 -07:00
|
|
|
game.override.enemyLevel(100);
|
2024-07-25 14:57:47 -07:00
|
|
|
game.override.enemySpecies(Species.MAGIKARP);
|
2024-07-25 15:18:14 -07:00
|
|
|
game.override.enemyMoveset([Moves.LIGHT_SCREEN, Moves.LIGHT_SCREEN, Moves.LIGHT_SCREEN, Moves.LIGHT_SCREEN]);
|
2024-07-25 16:18:47 -07:00
|
|
|
game.override.disableCrits();
|
2024-06-22 22:09:23 +08:00
|
|
|
});
|
|
|
|
|
2024-08-22 06:49:33 -07:00
|
|
|
it("reduces damage of special attacks by half in a single battle", async () => {
|
2024-06-22 22:09:23 +08:00
|
|
|
const moveToUse = Moves.ABSORB;
|
|
|
|
await game.startBattle([Species.SHUCKLE]);
|
|
|
|
|
2024-08-22 06:49:33 -07:00
|
|
|
game.move.select(moveToUse);
|
2024-06-22 22:09:23 +08:00
|
|
|
|
|
|
|
await game.phaseInterceptor.to(TurnEndPhase);
|
|
|
|
|
2024-08-07 09:23:12 -07:00
|
|
|
const mockedDmg = getMockedMoveDamage(game.scene.getEnemyPokemon()!, game.scene.getPlayerPokemon()!, allMoves[moveToUse]);
|
2024-06-22 22:09:23 +08:00
|
|
|
|
|
|
|
expect(mockedDmg).toBe(allMoves[moveToUse].power * singleBattleMultiplier);
|
|
|
|
});
|
|
|
|
|
2024-08-22 06:49:33 -07:00
|
|
|
it("reduces damage of special attacks by a third in a double battle", async () => {
|
2024-07-25 14:48:48 -07:00
|
|
|
game.override.battleType("double");
|
2024-06-22 22:09:23 +08:00
|
|
|
|
|
|
|
const moveToUse = Moves.DAZZLING_GLEAM;
|
|
|
|
await game.startBattle([Species.SHUCKLE, Species.SHUCKLE]);
|
|
|
|
|
2024-08-22 06:49:33 -07:00
|
|
|
game.move.select(moveToUse);
|
|
|
|
game.move.select(moveToUse, 1);
|
2024-06-22 22:09:23 +08:00
|
|
|
|
|
|
|
await game.phaseInterceptor.to(TurnEndPhase);
|
2024-08-07 09:23:12 -07:00
|
|
|
const mockedDmg = getMockedMoveDamage(game.scene.getEnemyPokemon()!, game.scene.getPlayerPokemon()!, allMoves[moveToUse]);
|
2024-06-22 22:09:23 +08:00
|
|
|
|
|
|
|
expect(mockedDmg).toBe(allMoves[moveToUse].power * doubleBattleMultiplier);
|
|
|
|
});
|
|
|
|
|
2024-08-22 06:49:33 -07:00
|
|
|
it("does not affect physical attacks", async () => {
|
2024-06-22 22:09:23 +08:00
|
|
|
const moveToUse = Moves.TACKLE;
|
|
|
|
await game.startBattle([Species.SHUCKLE]);
|
|
|
|
|
2024-08-22 06:49:33 -07:00
|
|
|
game.move.select(moveToUse);
|
2024-06-22 22:09:23 +08:00
|
|
|
|
|
|
|
await game.phaseInterceptor.to(TurnEndPhase);
|
2024-08-07 09:23:12 -07:00
|
|
|
const mockedDmg = getMockedMoveDamage(game.scene.getEnemyPokemon()!, game.scene.getPlayerPokemon()!, allMoves[moveToUse]);
|
2024-06-22 22:09:23 +08:00
|
|
|
|
|
|
|
expect(mockedDmg).toBe(allMoves[moveToUse].power);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Calculates the damage of a move multiplied by screen's multiplier, Light Screen in this case {@linkcode Moves.LIGHT_SCREEN}.
|
|
|
|
* 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 (defender.scene.arena.getTagOnSide(ArenaTagType.LIGHT_SCREEN, side)) {
|
|
|
|
defender.scene.arena.applyTagsForSide(ArenaTagType.LIGHT_SCREEN, side, move.category, defender.scene.currentBattle.double, multiplierHolder);
|
|
|
|
}
|
|
|
|
|
|
|
|
return move.power * multiplierHolder.value;
|
|
|
|
};
|