pokemon-showdown/data/mods/trademarked/scripts.ts
Guangcong Luo 78439b4a02
Update to ESLint 9 (#10926)
ESLint has a whole new config format, so I figure it's a good time to
make the config system saner.

- First, we no longer have separate eslint-no-types configs. Lint
  performance shouldn't be enough of a problem to justify the
  relevant maintenance complexity.

- Second, our base config should work out-of-the-box now. `npx eslint`
  will work as expected, without any CLI flags. You should still use
  `npm run lint` which adds the `--cached` flag for performance.

- Third, whatever updates I did fixed style linting, which apparently
  has been bugged for quite some time, considering all the obvious
  mixed-tabs-and-spaces issues I found in the upgrade.

Also here are some changes to our style rules. In particular:

- Curly brackets (for objects etc) now have spaces inside them. Sorry
  for the huge change. ESLint doesn't support our old style, and most
  projects use Prettier style, so we might as well match them in this way.
  See https://github.com/eslint-stylistic/eslint-stylistic/issues/415

- String + number concatenation is no longer allowed. We now
  consistently use template strings for this.
2025-02-25 20:03:46 -08:00

322 lines
12 KiB
TypeScript

export const Scripts: ModdedBattleScriptsData = {
gen: 9,
inherit: 'gen9',
endTurn() {
this.turn++;
this.lastSuccessfulMoveThisTurn = null;
for (const pokemon of this.getAllPokemon()) {
pokemon.m.trademarkUsedThisTurn = false;
// Somehow things went terribly wrong and I don't know what happened
pokemon.switchFlag = false;
pokemon.forceSwitchFlag = false;
}
const dynamaxEnding: Pokemon[] = [];
for (const pokemon of this.getAllActive()) {
if (pokemon.volatiles['dynamax']?.turns === 3) {
dynamaxEnding.push(pokemon);
}
}
if (dynamaxEnding.length > 1) {
this.updateSpeed();
this.speedSort(dynamaxEnding);
}
for (const pokemon of dynamaxEnding) {
pokemon.removeVolatile('dynamax');
}
// Gen 1 partial trapping ends when either Pokemon or a switch in faints to residual damage
if (this.gen === 1) {
for (const pokemon of this.getAllActive()) {
if (pokemon.volatiles['partialtrappinglock']) {
const target = pokemon.volatiles['partialtrappinglock'].locked;
if (target.hp <= 0 || !target.volatiles['partiallytrapped']) {
delete pokemon.volatiles['partialtrappinglock'];
}
}
if (pokemon.volatiles['partiallytrapped']) {
const source = pokemon.volatiles['partiallytrapped'].source;
if (source.hp <= 0 || !source.volatiles['partialtrappinglock']) {
delete pokemon.volatiles['partiallytrapped'];
}
}
}
}
const trappedBySide: boolean[] = [];
const stalenessBySide: ('internal' | 'external' | undefined)[] = [];
for (const side of this.sides) {
let sideTrapped = true;
let sideStaleness: 'internal' | 'external' | undefined;
for (const pokemon of side.active) {
if (!pokemon) continue;
pokemon.moveThisTurn = '';
pokemon.newlySwitched = false;
pokemon.moveLastTurnResult = pokemon.moveThisTurnResult;
pokemon.moveThisTurnResult = undefined;
if (this.turn !== 1) {
pokemon.usedItemThisTurn = false;
pokemon.statsRaisedThisTurn = false;
pokemon.statsLoweredThisTurn = false;
// It shouldn't be possible in a normal battle for a Pokemon to be damaged before turn 1's move selection
// However, this could be potentially relevant in certain OMs
pokemon.hurtThisTurn = null;
}
pokemon.maybeDisabled = false;
for (const moveSlot of pokemon.moveSlots) {
moveSlot.disabled = false;
moveSlot.disabledSource = '';
}
this.runEvent('DisableMove', pokemon);
for (const moveSlot of pokemon.moveSlots) {
const activeMove = this.dex.getActiveMove(moveSlot.id);
this.singleEvent('DisableMove', activeMove, null, pokemon);
if (activeMove.flags['cantusetwice'] && pokemon.lastMove?.id === moveSlot.id) {
pokemon.disableMove(pokemon.lastMove.id);
}
}
// If it was an illusion, it's not any more
if (pokemon.getLastAttackedBy() && this.gen >= 7) pokemon.knownType = true;
for (let i = pokemon.attackedBy.length - 1; i >= 0; i--) {
const attack = pokemon.attackedBy[i];
if (attack.source.isActive) {
attack.thisTurn = false;
} else {
pokemon.attackedBy.splice(pokemon.attackedBy.indexOf(attack), 1);
}
}
if (this.gen >= 7 && !pokemon.terastallized) {
// In Gen 7, the real type of every Pokemon is visible to all players via the bottom screen while making choices
const seenPokemon = pokemon.illusion || pokemon;
const realTypeString = seenPokemon.getTypes(true).join('/');
if (realTypeString !== seenPokemon.apparentType) {
this.add('-start', pokemon, 'typechange', realTypeString, '[silent]');
seenPokemon.apparentType = realTypeString;
if (pokemon.addedType) {
// The typechange message removes the added type, so put it back
this.add('-start', pokemon, 'typeadd', pokemon.addedType, '[silent]');
}
}
}
pokemon.trapped = pokemon.maybeTrapped = false;
this.runEvent('TrapPokemon', pokemon);
if (!pokemon.knownType || this.dex.getImmunity('trapped', pokemon)) {
this.runEvent('MaybeTrapPokemon', pokemon);
}
// canceling switches would leak information
// if a foe might have a trapping ability
if (this.gen > 2) {
for (const source of pokemon.foes()) {
const species = (source.illusion || source).species;
if (!species.abilities) continue;
for (const abilitySlot in species.abilities) {
const abilityName = species.abilities[abilitySlot as keyof Species['abilities']];
if (abilityName === source.ability) {
// pokemon event was already run above so we don't need
// to run it again.
continue;
}
const ruleTable = this.ruleTable;
if ((ruleTable.has('+hackmons') || !ruleTable.has('obtainableabilities')) && !this.format.team) {
// hackmons format
continue;
} else if (abilitySlot === 'H' && species.unreleasedHidden) {
// unreleased hidden ability
continue;
}
const ability = this.dex.abilities.get(abilityName);
if (ruleTable.has('-ability:' + ability.id)) continue;
if (pokemon.knownType && !this.dex.getImmunity('trapped', pokemon)) continue;
this.singleEvent('FoeMaybeTrapPokemon', ability, {}, pokemon, source);
}
}
}
if (pokemon.fainted) continue;
sideTrapped = sideTrapped && pokemon.trapped;
const staleness = pokemon.volatileStaleness || pokemon.staleness;
if (staleness) sideStaleness = sideStaleness === 'external' ? sideStaleness : staleness;
pokemon.activeTurns++;
}
trappedBySide.push(sideTrapped);
stalenessBySide.push(sideStaleness);
side.faintedLastTurn = side.faintedThisTurn;
side.faintedThisTurn = null;
}
if (this.maybeTriggerEndlessBattleClause(trappedBySide, stalenessBySide)) return;
if (this.gameType === 'triples' && this.sides.every(side => side.pokemonLeft === 1)) {
// If both sides have one Pokemon left in triples and they are not adjacent, they are both moved to the center.
const actives = this.getAllActive();
if (actives.length > 1 && !actives[0].isAdjacent(actives[1])) {
this.swapPosition(actives[0], 1, '[silent]');
this.swapPosition(actives[1], 1, '[silent]');
this.add('-center');
}
}
this.add('turn', this.turn);
if (this.gameType === 'multi') {
for (const side of this.sides) {
if (side.canDynamaxNow()) {
if (this.turn === 1) {
this.addSplit(side.id, ['-candynamax', side.id]);
} else {
this.add('-candynamax', side.id);
}
}
}
}
if (this.gen === 2) this.quickClawRoll = this.randomChance(60, 256);
if (this.gen === 3) this.quickClawRoll = this.randomChance(1, 5);
this.makeRequest('move');
},
pokemon: {
getAbility() {
const move = this.battle.dex.moves.get(this.battle.toID(this.ability));
if (!move.exists) return Object.getPrototypeOf(this).getAbility.call(this);
return {
id: move.id,
name: move.name,
flags: {},
// Does not need activation message with this
fullname: 'ability: ' + move.name,
onStart(this: Battle, pokemon: Pokemon) {
if (pokemon.m.trademarkUsedThisTurn) {
// no.
this.add('cant', pokemon, 'ability: ' + move.name);
this.add('-hint', "Each Pokemon can use a trademark only once per turn to prevent infinite loops.");
} else {
pokemon.m.trademarkUsedThisTurn = true;
const trademark = this.dex.getActiveMove(move.id);
trademark.accuracy = true;
this.actions.useMove(trademark, pokemon);
}
},
toString() {
return move.name;
},
};
},
transformInto(pokemon, effect) {
const species = pokemon.species;
if (
pokemon.fainted || this.illusion || pokemon.illusion || (pokemon.volatiles['substitute'] && this.battle.gen >= 5) ||
(pokemon.transformed && this.battle.gen >= 2) || (this.transformed && this.battle.gen >= 5) ||
species.name === 'Eternatus-Eternamax' || (['Ogerpon', 'Terapagos'].includes(species.baseSpecies) &&
(this.terastallized || pokemon.terastallized)) || this.terastallized === 'Stellar'
) {
return false;
}
if (this.battle.dex.currentMod === 'gen1stadium' && (
species.name === 'Ditto' ||
(this.species.name === 'Ditto' && pokemon.moves.includes('transform'))
)) {
return false;
}
if (!this.setSpecies(species, effect, true)) return false;
this.transformed = true;
this.weighthg = pokemon.weighthg;
const types = pokemon.getTypes(true, true);
this.setType(pokemon.volatiles['roost'] ? pokemon.volatiles['roost'].typeWas : types, true);
this.addedType = pokemon.addedType;
this.knownType = this.isAlly(pokemon) && pokemon.knownType;
this.apparentType = pokemon.apparentType;
let statName: StatIDExceptHP;
for (statName in this.storedStats) {
this.storedStats[statName] = pokemon.storedStats[statName];
if (this.modifiedStats) this.modifiedStats[statName] = pokemon.modifiedStats![statName]; // Gen 1: Copy modified stats.
}
this.moveSlots = [];
this.hpType = (this.battle.gen >= 5 ? this.hpType : pokemon.hpType);
this.hpPower = (this.battle.gen >= 5 ? this.hpPower : pokemon.hpPower);
this.timesAttacked = pokemon.timesAttacked;
for (const moveSlot of pokemon.moveSlots) {
let moveName = moveSlot.move;
if (moveSlot.id === 'hiddenpower') {
moveName = 'Hidden Power ' + this.hpType;
}
this.moveSlots.push({
move: moveName,
id: moveSlot.id,
pp: moveSlot.maxpp === 1 ? 1 : 5,
maxpp: this.battle.gen >= 5 ? (moveSlot.maxpp === 1 ? 1 : 5) : moveSlot.maxpp,
target: moveSlot.target,
disabled: false,
used: false,
virtual: true,
});
}
let boostName: BoostID;
for (boostName in pokemon.boosts) {
this.boosts[boostName] = pokemon.boosts[boostName];
}
if (this.battle.gen >= 6) {
const volatilesToCopy = ['dragoncheer', 'focusenergy', 'gmaxchistrike', 'laserfocus'];
// we need to remove all the crit volatiles before adding any crit volatile
for (const volatile of volatilesToCopy) this.removeVolatile(volatile);
for (const volatile of volatilesToCopy) {
if (pokemon.volatiles[volatile]) {
this.addVolatile(volatile);
if (volatile === 'gmaxchistrike') this.volatiles[volatile].layers = pokemon.volatiles[volatile].layers;
if (volatile === 'dragoncheer') this.volatiles[volatile].hasDragonType = pokemon.volatiles[volatile].hasDragonType;
}
}
}
if (effect) {
this.battle.add('-transform', this, pokemon, '[from] ' + effect.fullname);
} else {
this.battle.add('-transform', this, pokemon);
}
if (this.terastallized) {
this.knownType = true;
this.apparentType = this.terastallized;
}
// Changed to be compatible with trademarks
if (this.battle.gen > 2) this.setAbility(pokemon.getAbility(), this, true, true);
// Change formes based on held items (for Transform)
// Only ever relevant in Generation 4 since Generation 3 didn't have item-based forme changes
if (this.battle.gen === 4) {
if (this.species.num === 487) {
// Giratina formes
if (this.species.name === 'Giratina' && this.item === 'griseousorb') {
this.formeChange('Giratina-Origin');
} else if (this.species.name === 'Giratina-Origin' && this.item !== 'griseousorb') {
this.formeChange('Giratina');
}
}
if (this.species.num === 493) {
// Arceus formes
const item = this.getItem();
const targetForme = (item?.onPlate ? 'Arceus-' + item.onPlate : 'Arceus');
if (this.species.name !== targetForme) {
this.formeChange(targetForme);
}
}
}
// Pokemon transformed into Ogerpon cannot Terastallize
// restoring their ability to tera after they untransform is handled ELSEWHERE
if (this.species.baseSpecies === 'Ogerpon' && this.canTerastallize) this.canTerastallize = false;
if (this.species.baseSpecies === 'Terapagos' && this.canTerastallize) this.canTerastallize = false;
return true;
},
},
};