[Ability] Implemented Aura Break (#2252)
* re-adjust according to new changes * added unit tests * add test fixes * add documentation * more fixes
This commit is contained in:
parent
3bbe01b288
commit
77d25d18f9
|
@ -4592,7 +4592,8 @@ export function initAbilities() {
|
||||||
.attr(FieldMoveTypePowerBoostAbAttr, Type.FAIRY, 4 / 3),
|
.attr(FieldMoveTypePowerBoostAbAttr, Type.FAIRY, 4 / 3),
|
||||||
new Ability(Abilities.AURA_BREAK, 6)
|
new Ability(Abilities.AURA_BREAK, 6)
|
||||||
.ignorable()
|
.ignorable()
|
||||||
.unimplemented(),
|
.conditionalAttr(target => target.hasAbility(Abilities.DARK_AURA), FieldMoveTypePowerBoostAbAttr, Type.DARK, 9 / 16)
|
||||||
|
.conditionalAttr(target => target.hasAbility(Abilities.FAIRY_AURA), FieldMoveTypePowerBoostAbAttr, Type.FAIRY, 9 / 16),
|
||||||
new Ability(Abilities.PRIMORDIAL_SEA, 6)
|
new Ability(Abilities.PRIMORDIAL_SEA, 6)
|
||||||
.attr(PostSummonWeatherChangeAbAttr, WeatherType.HEAVY_RAIN)
|
.attr(PostSummonWeatherChangeAbAttr, WeatherType.HEAVY_RAIN)
|
||||||
.attr(PostBiomeChangeWeatherChangeAbAttr, WeatherType.HEAVY_RAIN)
|
.attr(PostBiomeChangeWeatherChangeAbAttr, WeatherType.HEAVY_RAIN)
|
||||||
|
|
|
@ -0,0 +1,95 @@
|
||||||
|
import { afterEach, beforeAll, beforeEach, describe, expect, it, vi } from "vitest";
|
||||||
|
import Phaser from "phaser";
|
||||||
|
import GameManager from "#app/test/utils/gameManager";
|
||||||
|
import * as overrides from "#app/overrides";
|
||||||
|
import { Species } from "#enums/species";
|
||||||
|
import { MoveEffectPhase } from "#app/phases";
|
||||||
|
import { Moves } from "#enums/moves";
|
||||||
|
import { getMovePosition } from "#app/test/utils/gameManagerUtils";
|
||||||
|
import { Abilities } from "#enums/abilities";
|
||||||
|
import Move, { allMoves } from "#app/data/move.js";
|
||||||
|
import Pokemon from "#app/field/pokemon.js";
|
||||||
|
import { FieldMoveTypePowerBoostAbAttr } from "#app/data/ability.js";
|
||||||
|
import { NumberHolder } from "#app/utils.js";
|
||||||
|
|
||||||
|
describe("Abilities - Aura Break", () => {
|
||||||
|
let phaserGame: Phaser.Game;
|
||||||
|
let game: GameManager;
|
||||||
|
|
||||||
|
beforeAll(() => {
|
||||||
|
phaserGame = new Phaser.Game({
|
||||||
|
type: Phaser.HEADLESS,
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
afterEach(() => {
|
||||||
|
game.phaseInterceptor.restoreOg();
|
||||||
|
});
|
||||||
|
|
||||||
|
beforeEach(() => {
|
||||||
|
game = new GameManager(phaserGame);
|
||||||
|
vi.spyOn(overrides, "SINGLE_BATTLE_OVERRIDE", "get").mockReturnValue(true);
|
||||||
|
vi.spyOn(overrides, "MOVESET_OVERRIDE", "get").mockReturnValue([Moves.MOONBLAST, Moves.DARK_PULSE, Moves.MOONBLAST, Moves.DARK_PULSE]);
|
||||||
|
vi.spyOn(overrides, "OPP_MOVESET_OVERRIDE", "get").mockReturnValue([Moves.SPLASH, Moves.SPLASH, Moves.SPLASH, Moves.SPLASH]);
|
||||||
|
vi.spyOn(overrides, "OPP_ABILITY_OVERRIDE", "get").mockReturnValue(Abilities.AURA_BREAK);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("reverses the effect of fairy aura", async () => {
|
||||||
|
vi.spyOn(overrides, "ABILITY_OVERRIDE", "get").mockReturnValue(Abilities.FAIRY_AURA);
|
||||||
|
const basePower = allMoves[Moves.MOONBLAST].power;
|
||||||
|
const multiplier = 9 / 16;
|
||||||
|
await game.startBattle([Species.MAGIKARP]);
|
||||||
|
|
||||||
|
game.doAttack(getMovePosition(game.scene, 0, Moves.MOONBLAST));
|
||||||
|
|
||||||
|
const appliedPower = getMockedMovePower(game.scene.getEnemyField()[0], game.scene.getPlayerField()[0], allMoves[Moves.MOONBLAST]);
|
||||||
|
|
||||||
|
await game.phaseInterceptor.to(MoveEffectPhase);
|
||||||
|
|
||||||
|
expect(appliedPower).not.toBe(undefined);
|
||||||
|
expect(appliedPower).not.toBe(basePower);
|
||||||
|
expect(appliedPower).toBe(basePower * multiplier);
|
||||||
|
|
||||||
|
});
|
||||||
|
it("reverses the effect of dark aura", async () => {
|
||||||
|
vi.spyOn(overrides, "ABILITY_OVERRIDE", "get").mockReturnValue(Abilities.DARK_AURA);
|
||||||
|
const basePower = allMoves[Moves.DARK_PULSE].power;
|
||||||
|
const multiplier = 9 / 16;
|
||||||
|
await game.startBattle([Species.MAGIKARP]);
|
||||||
|
|
||||||
|
game.doAttack(getMovePosition(game.scene, 0, Moves.DARK_PULSE));
|
||||||
|
|
||||||
|
const appliedPower = getMockedMovePower(game.scene.getEnemyField()[0], game.scene.getPlayerField()[0], allMoves[Moves.DARK_PULSE]);
|
||||||
|
|
||||||
|
await game.phaseInterceptor.to(MoveEffectPhase);
|
||||||
|
|
||||||
|
expect(appliedPower).not.toBe(undefined);
|
||||||
|
expect(appliedPower).not.toBe(basePower);
|
||||||
|
expect(appliedPower).toBe(basePower * multiplier);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Calculates the mocked power of a move in a Pokémon battle, taking into account certain abilities.
|
||||||
|
*
|
||||||
|
* @param defender - The defending Pokémon.
|
||||||
|
* @param attacker - The attacking Pokémon.
|
||||||
|
* @param move - The move being used in the attack.
|
||||||
|
* @returns The calculated power of the move after applying any relevant ability effects.
|
||||||
|
*
|
||||||
|
* @remarks
|
||||||
|
* This function creates a NumberHolder with the initial power of the move.
|
||||||
|
* It then checks if the defender has an ability with the FieldMoveTypePowerBoostAbAttr.
|
||||||
|
* If so, it applies a power modification of 9/16 using an instance of FieldMoveTypePowerBoostAbAttr.
|
||||||
|
* The final calculated power is then returned.
|
||||||
|
*/
|
||||||
|
const getMockedMovePower = (defender: Pokemon, attacker: Pokemon, move: Move): number => {
|
||||||
|
const powerHolder = new NumberHolder(move.power);
|
||||||
|
|
||||||
|
if (defender.hasAbilityWithAttr(FieldMoveTypePowerBoostAbAttr)) {
|
||||||
|
const auraBreakInstance = new FieldMoveTypePowerBoostAbAttr(move.type, 9 / 16);
|
||||||
|
auraBreakInstance.applyPreAttack(attacker, false, defender, move, [powerHolder]);
|
||||||
|
}
|
||||||
|
|
||||||
|
return powerHolder.value;
|
||||||
|
};
|
Loading…
Reference in New Issue