Jofthomas's picture
Upload 4781 files
5c2ed06 verified
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;
},
},
};