mirror of
https://github.com/pagefaultgames/pokerogue.git
synced 2024-11-30 02:36:11 +00:00
[Move] Implement Flame Burst (after beta fix) (#3239)
* add integration tests * account for magic guard * update test name * remove test code * fix magic guard interaction * set 1 as min damage * fix tests * only apply magic guard attr if ally is active * nit: remove new line * update docs * add the move attr
This commit is contained in:
parent
cb7dbe601a
commit
208f5af62a
@ -1435,6 +1435,39 @@ export class PartyStatusCureAttr extends MoveEffectAttr {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Applies damage to the target's ally equal to 1/16 of that ally's max HP.
|
||||||
|
* @extends MoveEffectAttr
|
||||||
|
*/
|
||||||
|
export class FlameBurstAttr extends MoveEffectAttr {
|
||||||
|
/**
|
||||||
|
* @param user - n/a
|
||||||
|
* @param target - The target Pokémon.
|
||||||
|
* @param move - n/a
|
||||||
|
* @param args - n/a
|
||||||
|
* @returns A boolean indicating whether the effect was successfully applied.
|
||||||
|
*/
|
||||||
|
apply(user: Pokemon, target: Pokemon, move: Move, args: any[]): boolean | Promise<boolean> {
|
||||||
|
const targetAlly = target.getAlly();
|
||||||
|
const cancelled = new Utils.BooleanHolder(false);
|
||||||
|
|
||||||
|
if (targetAlly) {
|
||||||
|
applyAbAttrs(BlockNonDirectDamageAbAttr, targetAlly, cancelled);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (cancelled.value || !targetAlly) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
targetAlly.damageAndUpdate(Math.max(1, Math.floor(1/16 * targetAlly.getMaxHp())), HitResult.OTHER);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
getTargetBenefitScore(user: Pokemon, target: Pokemon, move: Move): integer {
|
||||||
|
return target.getAlly() ? -5 : 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
export class SacrificialFullRestoreAttr extends SacrificialAttr {
|
export class SacrificialFullRestoreAttr extends SacrificialAttr {
|
||||||
constructor() {
|
constructor() {
|
||||||
super();
|
super();
|
||||||
@ -7292,7 +7325,7 @@ export function initMoves() {
|
|||||||
new AttackMove(Moves.STORM_THROW, Type.FIGHTING, MoveCategory.PHYSICAL, 60, 100, 10, -1, 0, 5)
|
new AttackMove(Moves.STORM_THROW, Type.FIGHTING, MoveCategory.PHYSICAL, 60, 100, 10, -1, 0, 5)
|
||||||
.attr(CritOnlyAttr),
|
.attr(CritOnlyAttr),
|
||||||
new AttackMove(Moves.FLAME_BURST, Type.FIRE, MoveCategory.SPECIAL, 70, 100, 15, -1, 0, 5)
|
new AttackMove(Moves.FLAME_BURST, Type.FIRE, MoveCategory.SPECIAL, 70, 100, 15, -1, 0, 5)
|
||||||
.partial(),
|
.attr(FlameBurstAttr),
|
||||||
new AttackMove(Moves.SLUDGE_WAVE, Type.POISON, MoveCategory.SPECIAL, 95, 100, 10, 10, 0, 5)
|
new AttackMove(Moves.SLUDGE_WAVE, Type.POISON, MoveCategory.SPECIAL, 95, 100, 10, 10, 0, 5)
|
||||||
.attr(StatusEffectAttr, StatusEffect.POISON)
|
.attr(StatusEffectAttr, StatusEffect.POISON)
|
||||||
.target(MoveTarget.ALL_NEAR_OTHERS),
|
.target(MoveTarget.ALL_NEAR_OTHERS),
|
||||||
|
118
src/test/moves/flame_burst.test.ts
Normal file
118
src/test/moves/flame_burst.test.ts
Normal file
@ -0,0 +1,118 @@
|
|||||||
|
import { afterEach, beforeAll, beforeEach, describe, expect, it, vi } from "vitest";
|
||||||
|
import Phaser from "phaser";
|
||||||
|
import GameManager from "#app/test/utils/gameManager";
|
||||||
|
import Overrides from "#app/overrides";
|
||||||
|
import { Species } from "#enums/species";
|
||||||
|
import {
|
||||||
|
SelectTargetPhase,
|
||||||
|
TurnEndPhase,
|
||||||
|
} from "#app/phases";
|
||||||
|
import { Moves } from "#enums/moves";
|
||||||
|
import { getMovePosition } from "#app/test/utils/gameManagerUtils";
|
||||||
|
import { Abilities } from "#app/enums/abilities.js";
|
||||||
|
import { allAbilities } from "#app/data/ability.js";
|
||||||
|
import Pokemon from "#app/field/pokemon.js";
|
||||||
|
|
||||||
|
describe("Moves - Flame Burst", () => {
|
||||||
|
let phaserGame: Phaser.Game;
|
||||||
|
let game: GameManager;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Calculates the effect damage of Flame Burst which is 1/16 of the target ally's max HP
|
||||||
|
* See Flame Burst {@link https://bulbapedia.bulbagarden.net/wiki/Flame_Burst_(move)}
|
||||||
|
* See Flame Burst's move attribute {@linkcode FlameBurstAttr}
|
||||||
|
* @param pokemon {@linkcode Pokemon} - The ally of the move's target
|
||||||
|
* @returns Effect damage of Flame Burst
|
||||||
|
*/
|
||||||
|
const getEffectDamage = (pokemon: Pokemon): number => {
|
||||||
|
return Math.max(1, Math.floor(pokemon.getMaxHp() * 1/16));
|
||||||
|
};
|
||||||
|
|
||||||
|
beforeAll(() => {
|
||||||
|
phaserGame = new Phaser.Game({
|
||||||
|
type: Phaser.HEADLESS,
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
afterEach(() => {
|
||||||
|
game.phaseInterceptor.restoreOg();
|
||||||
|
});
|
||||||
|
|
||||||
|
beforeEach(() => {
|
||||||
|
game = new GameManager(phaserGame);
|
||||||
|
vi.spyOn(Overrides, "BATTLE_TYPE_OVERRIDE", "get").mockReturnValue("double");
|
||||||
|
vi.spyOn(Overrides, "MOVESET_OVERRIDE", "get").mockReturnValue([Moves.FLAME_BURST, Moves.SPLASH]);
|
||||||
|
vi.spyOn(Overrides, "NEVER_CRIT_OVERRIDE", "get").mockReturnValue(true);
|
||||||
|
vi.spyOn(Overrides, "ABILITY_OVERRIDE", "get").mockReturnValue(Abilities.UNNERVE);
|
||||||
|
vi.spyOn(Overrides, "STARTING_WAVE_OVERRIDE", "get").mockReturnValue(4);
|
||||||
|
vi.spyOn(Overrides, "OPP_SPECIES_OVERRIDE", "get").mockReturnValue(Species.SHUCKLE);
|
||||||
|
vi.spyOn(Overrides, "OPP_ABILITY_OVERRIDE", "get").mockReturnValue(Abilities.BALL_FETCH);
|
||||||
|
vi.spyOn(Overrides, "OPP_MOVESET_OVERRIDE", "get").mockReturnValue(new Array(4).fill(Moves.SPLASH));
|
||||||
|
});
|
||||||
|
|
||||||
|
it("inflicts damage to the target's ally equal to 1/16 of its max HP", async () => {
|
||||||
|
await game.startBattle([Species.PIKACHU, Species.PIKACHU]);
|
||||||
|
const [ leftEnemy, rightEnemy ] = game.scene.getEnemyField();
|
||||||
|
|
||||||
|
game.doAttack(getMovePosition(game.scene, 0, Moves.FLAME_BURST));
|
||||||
|
await game.phaseInterceptor.to(SelectTargetPhase, false);
|
||||||
|
game.doSelectTarget(leftEnemy.getBattlerIndex());
|
||||||
|
game.doAttack(getMovePosition(game.scene, 1, Moves.SPLASH));
|
||||||
|
await game.phaseInterceptor.to(TurnEndPhase);
|
||||||
|
|
||||||
|
expect(leftEnemy.hp).toBeLessThan(leftEnemy.getMaxHp());
|
||||||
|
expect(rightEnemy.hp).toBe(rightEnemy.getMaxHp() - getEffectDamage(rightEnemy));
|
||||||
|
});
|
||||||
|
|
||||||
|
it("does not inflict damage to the target's ally if the target was not affected by Flame Burst", async () => {
|
||||||
|
vi.spyOn(Overrides, "OPP_ABILITY_OVERRIDE", "get").mockReturnValue(Abilities.FLASH_FIRE);
|
||||||
|
|
||||||
|
await game.startBattle([Species.PIKACHU, Species.PIKACHU]);
|
||||||
|
const [ leftEnemy, rightEnemy ] = game.scene.getEnemyField();
|
||||||
|
|
||||||
|
game.doAttack(getMovePosition(game.scene, 0, Moves.FLAME_BURST));
|
||||||
|
await game.phaseInterceptor.to(SelectTargetPhase, false);
|
||||||
|
game.doSelectTarget(leftEnemy.getBattlerIndex());
|
||||||
|
game.doAttack(getMovePosition(game.scene, 1, Moves.SPLASH));
|
||||||
|
await game.phaseInterceptor.to(TurnEndPhase);
|
||||||
|
|
||||||
|
expect(leftEnemy.hp).toBe(leftEnemy.getMaxHp());
|
||||||
|
expect(rightEnemy.hp).toBe(rightEnemy.getMaxHp());
|
||||||
|
});
|
||||||
|
|
||||||
|
it("does not interact with the target ally's abilities", async () => {
|
||||||
|
await game.startBattle([Species.PIKACHU, Species.PIKACHU]);
|
||||||
|
const [ leftEnemy, rightEnemy ] = game.scene.getEnemyField();
|
||||||
|
|
||||||
|
vi.spyOn(rightEnemy, "getAbility").mockReturnValue(allAbilities[Abilities.FLASH_FIRE]);
|
||||||
|
|
||||||
|
game.doAttack(getMovePosition(game.scene, 0, Moves.FLAME_BURST));
|
||||||
|
await game.phaseInterceptor.to(SelectTargetPhase, false);
|
||||||
|
game.doSelectTarget(leftEnemy.getBattlerIndex());
|
||||||
|
game.doAttack(getMovePosition(game.scene, 1, Moves.SPLASH));
|
||||||
|
await game.phaseInterceptor.to(TurnEndPhase);
|
||||||
|
|
||||||
|
expect(leftEnemy.hp).toBeLessThan(leftEnemy.getMaxHp());
|
||||||
|
expect(rightEnemy.hp).toBe(rightEnemy.getMaxHp() - getEffectDamage(rightEnemy));
|
||||||
|
});
|
||||||
|
|
||||||
|
it("effect damage is prevented by Magic Guard", async () => {
|
||||||
|
await game.startBattle([Species.PIKACHU, Species.PIKACHU]);
|
||||||
|
const [ leftEnemy, rightEnemy ] = game.scene.getEnemyField();
|
||||||
|
|
||||||
|
vi.spyOn(rightEnemy, "getAbility").mockReturnValue(allAbilities[Abilities.MAGIC_GUARD]);
|
||||||
|
|
||||||
|
game.doAttack(getMovePosition(game.scene, 0, Moves.FLAME_BURST));
|
||||||
|
await game.phaseInterceptor.to(SelectTargetPhase, false);
|
||||||
|
game.doSelectTarget(leftEnemy.getBattlerIndex());
|
||||||
|
game.doAttack(getMovePosition(game.scene, 1, Moves.SPLASH));
|
||||||
|
await game.phaseInterceptor.to(TurnEndPhase);
|
||||||
|
|
||||||
|
expect(leftEnemy.hp).toBeLessThan(leftEnemy.getMaxHp());
|
||||||
|
expect(rightEnemy.hp).toBe(rightEnemy.getMaxHp());
|
||||||
|
});
|
||||||
|
|
||||||
|
it("is not affected by protection moves and Endure", async () => {
|
||||||
|
// TODO: update this test when it's possible to select move for each enemy
|
||||||
|
}, { skip: true });
|
||||||
|
});
|
Loading…
Reference in New Issue
Block a user