diff --git a/data/abilities.ts b/data/abilities.ts index 8048f5204e6c..4a2d0449a685 100644 --- a/data/abilities.ts +++ b/data/abilities.ts @@ -2030,8 +2030,7 @@ export const Abilities: import('../sim/dex-abilities').AbilityDataTable = { if (pokemon.illusion) { this.debug('illusion cleared'); pokemon.illusion = null; - const details = pokemon.species.name + (pokemon.level === 100 ? '' : ', L' + pokemon.level) + - (pokemon.gender === '' ? '' : ', ' + pokemon.gender) + (pokemon.set.shiny ? ', shiny' : ''); + const details = pokemon.getSimpleDetails(); this.add('replace', pokemon, details); this.add('-end', pokemon, 'Illusion'); if (this.ruleTable.has('illusionlevelmod')) { diff --git a/data/mods/gen7pokebilities/scripts.ts b/data/mods/gen7pokebilities/scripts.ts index b0199e4bad6b..b74ad6949f05 100644 --- a/data/mods/gen7pokebilities/scripts.ts +++ b/data/mods/gen7pokebilities/scripts.ts @@ -169,8 +169,7 @@ export const Scripts: ModdedBattleScriptsData = { this.illusion ? this.illusion.species.name : species.baseSpecies; if (isPermanent) { this.baseSpecies = rawSpecies; - this.details = species.name + (this.level === 100 ? '' : ', L' + this.level) + - (this.gender === '' ? '' : ', ' + this.gender) + (this.set.shiny ? ', shiny' : ''); + this.details = this.getSimpleDetails(); this.battle.add('detailschange', this, (this.illusion || this).details); if (source.effectType === 'Item') { this.canTerastallize = null; // National Dex behavior diff --git a/data/mods/gen9ssb/abilities.ts b/data/mods/gen9ssb/abilities.ts index badd191ed69f..11e74ec300a7 100644 --- a/data/mods/gen9ssb/abilities.ts +++ b/data/mods/gen9ssb/abilities.ts @@ -1755,8 +1755,7 @@ export const Abilities: import('../../../sim/dex-abilities').ModdedAbilityDataTa target.clearBoosts(); this.add('-clearboost', target); this.boost({atk: 1, def: -1, spa: 1, spd: -1, spe: 1}); - const details = target.species.name + (target.level === 100 ? '' : ', L' + target.level) + - (target.gender === '' ? '' : ', ' + target.gender) + (target.set.shiny ? ', shiny' : ''); + const details = target.getSimpleDetails(); target.details = details; this.add('replace', target, details); } diff --git a/data/mods/gen9ssb/moves.ts b/data/mods/gen9ssb/moves.ts index 35c1284202f1..fd2d9e278187 100644 --- a/data/mods/gen9ssb/moves.ts +++ b/data/mods/gen9ssb/moves.ts @@ -4236,8 +4236,7 @@ export const Moves: import('../../../sim/dex-moves').ModdedMoveDataTable = { pokemon.cureStatus(); this.boost({def: 1, spd: 1}); (pokemon as any).level += 5; - pokemon.details = pokemon.species.name + (pokemon.level === 100 ? '' : ', L' + pokemon.level) + - (pokemon.gender === '' ? '' : ', ' + pokemon.gender) + (pokemon.set.shiny ? ', shiny' : ''); + pokemon.details = pokemon.getSimpleDetails(); this.add('-anim', pokemon, 'Geomancy', pokemon); this.add('replace', pokemon, pokemon.details); this.add('-message', `${pokemon.name} gained 5 levels!`); @@ -5905,8 +5904,7 @@ export const Moves: import('../../../sim/dex-moves').ModdedMoveDataTable = { this.add(`c:|${getName((source.illusion || source).name)}|lol never do that ever again thanks`); this.add('custom', '-endterastallize', pokemon); delete pokemon.terastallized; - const details = pokemon.species.name + (pokemon.level === 100 ? '' : ', L' + pokemon.level) + - (pokemon.gender === '' ? '' : ', ' + pokemon.gender) + (pokemon.set.shiny ? ', shiny' : ''); + const details = pokemon.getSimpleDetails(); this.add('detailschange', pokemon, details); } }, diff --git a/data/mods/gen9ssb/scripts.ts b/data/mods/gen9ssb/scripts.ts index 3fe0e55a73b8..98f8877ce552 100644 --- a/data/mods/gen9ssb/scripts.ts +++ b/data/mods/gen9ssb/scripts.ts @@ -89,8 +89,7 @@ export function changeSet(context: Battle, pokemon: Pokemon, newSet: SSBSet, cha pokemon.teraType = newSet.teraType === 'Any' ? context.sample(allTypes) : Array.isArray(newSet.teraType) ? context.sample(newSet.teraType) : newSet.teraType; } - const details = pokemon.species.name + (pokemon.level === 100 ? '' : ', L' + pokemon.level) + - (pokemon.gender === '' ? '' : ', ' + pokemon.gender) + (pokemon.set.shiny ? ', shiny' : ''); + const details = pokemon.getSimpleDetails(); if (oldShiny !== pokemon.set.shiny || oldGender !== pokemon.gender) context.add('replace', pokemon, details); if (changeAbility) pokemon.setAbility(newSet.ability as string, undefined, true); @@ -386,8 +385,7 @@ export const Scripts: ModdedBattleScriptsData = { const species = pokemon.setSpecies(rawSpecies); if (!species) continue; pokemon.baseSpecies = rawSpecies; - pokemon.details = species.name + (pokemon.level === 100 ? '' : ', L' + pokemon.level) + - (pokemon.gender === '' ? '' : ', ' + pokemon.gender) + (pokemon.set.shiny ? ', shiny' : ''); + pokemon.details = pokemon.getSimpleDetails(); // pokemon.setAbility(species.abilities['0'], null, true); // pokemon.baseAbility = pokemon.ability; diff --git a/data/mods/mixandmega/scripts.ts b/data/mods/mixandmega/scripts.ts index a687f445ba51..2bed1943aac2 100644 --- a/data/mods/mixandmega/scripts.ts +++ b/data/mods/mixandmega/scripts.ts @@ -67,8 +67,7 @@ export const Scripts: ModdedBattleScriptsData = { const species = pokemon.setSpecies(rawSpecies); if (!species) continue; pokemon.baseSpecies = rawSpecies; - pokemon.details = species.name + (pokemon.level === 100 ? '' : ', L' + pokemon.level) + - (pokemon.gender === '' ? '' : ', ' + pokemon.gender) + (pokemon.set.shiny ? ', shiny' : ''); + pokemon.details = pokemon.getSimpleDetails(); pokemon.ability = this.toID(species.abilities['0']); pokemon.baseAbility = pokemon.ability; } @@ -118,8 +117,7 @@ export const Scripts: ModdedBattleScriptsData = { const species = pokemon.setSpecies(rawSpecies); if (!species) continue; pokemon.baseSpecies = rawSpecies; - pokemon.details = species.name + (pokemon.level === 100 ? '' : ', L' + pokemon.level) + - (pokemon.gender === '' ? '' : ', ' + pokemon.gender) + (pokemon.set.shiny ? ', shiny' : ''); + pokemon.details = pokemon.getSimpleDetails(); pokemon.ability = this.toID(species.abilities['0']); pokemon.baseAbility = pokemon.ability; diff --git a/data/mods/pokebilities/scripts.ts b/data/mods/pokebilities/scripts.ts index e9564991a2ac..ecc2cf98647d 100644 --- a/data/mods/pokebilities/scripts.ts +++ b/data/mods/pokebilities/scripts.ts @@ -169,8 +169,7 @@ export const Scripts: ModdedBattleScriptsData = { this.illusion ? this.illusion.species.name : species.baseSpecies; if (isPermanent) { this.baseSpecies = rawSpecies; - this.details = species.name + (this.level === 100 ? '' : ', L' + this.level) + - (this.gender === '' ? '' : ', ' + this.gender) + (this.set.shiny ? ', shiny' : ''); + this.details = this.getSimpleDetails(); this.battle.add('detailschange', this, (this.illusion || this).details); if (source.effectType === 'Item') { this.canTerastallize = null; // National Dex behavior diff --git a/sim/battle-actions.ts b/sim/battle-actions.ts index 4128c3f85d54..ad8e9eafd707 100644 --- a/sim/battle-actions.ts +++ b/sim/battle-actions.ts @@ -1959,7 +1959,7 @@ export class BattleActions { } if (pokemon.species.baseSpecies === 'Morpeko') { pokemon.baseSpecies = pokemon.species; - pokemon.details = pokemon.details.replace('Morpeko', pokemon.species.name); + pokemon.details = pokemon.getSimpleDetails(); } this.battle.runEvent('AfterTerastallization', pokemon); } diff --git a/sim/battle.ts b/sim/battle.ts index dba8f006b52c..28b426ac0dd8 100644 --- a/sim/battle.ts +++ b/sim/battle.ts @@ -2375,6 +2375,18 @@ export class Battle { pokemon.illusion = null; pokemon.isActive = false; pokemon.isStarted = false; + if ( + (pokemon.terastallized && ['Morpeko', 'Ogerpon', 'Terapagos'].includes(pokemon.baseSpecies.baseSpecies)) || + ['Greninja-Bond', 'Necrozma-Ultra', 'Zygarde-Complete'].includes(pokemon.baseSpecies.baseSpecies) || + pokemon.baseSpecies.isMega || pokemon.baseSpecies.isPrimal + ) { + const baseSpecies = this.dex.species.get(pokemon.set.species || pokemon.set.name); + pokemon.setSpecies(baseSpecies); + pokemon.baseSpecies = baseSpecies; + pokemon.baseAbility = pokemon.ability = toID(pokemon.set.ability); + pokemon.details = pokemon.getSimpleDetails(); + this.add('detailschange', pokemon, pokemon.details, '[silent]'); + } delete pokemon.terastallized; pokemon.side.faintedThisTurn = pokemon; if (this.faintQueue.length >= faintQueueLeft) checkWin = true; @@ -2494,8 +2506,7 @@ export class Battle { const species = pokemon.setSpecies(rawSpecies); if (!species) continue; pokemon.baseSpecies = rawSpecies; - pokemon.details = species.name + (pokemon.level === 100 ? '' : ', L' + pokemon.level) + - (pokemon.gender === '' ? '' : ', ' + pokemon.gender) + (pokemon.set.shiny ? ', shiny' : ''); + pokemon.details = pokemon.getSimpleDetails(); pokemon.setAbility(species.abilities['0'], null, true); pokemon.baseAbility = pokemon.ability; diff --git a/sim/pokemon.ts b/sim/pokemon.ts index bb954c86b7d5..2d58191e2c00 100644 --- a/sim/pokemon.ts +++ b/sim/pokemon.ts @@ -353,8 +353,7 @@ export class Pokemon { this.position = 0; let displayedSpeciesName = this.species.name; if (displayedSpeciesName === 'Greninja-Bond') displayedSpeciesName = 'Greninja'; - this.details = displayedSpeciesName + (this.level === 100 ? '' : ', L' + this.level) + - (this.gender === '' ? '' : ', ' + this.gender) + (this.set.shiny ? ', shiny' : ''); + this.details = this.getSimpleDetails(displayedSpeciesName); this.status = ''; this.statusState = {}; @@ -509,17 +508,21 @@ export class Pokemon { return this.isActive ? this.getSlot() + fullname.slice(2) : fullname; } + getSimpleDetails(displayedSpeciesName: string | null = null, illusionDetails = false) { + let name = displayedSpeciesName || this.species.name; + let level = this.level; + if (illusionDetails && this.illusion) { + if (this.battle.ruleTable.has('illusionlevelmod')) level = this.illusion.level; + name = this.illusion.species.name; + if (name === 'Greninja-Bond') name = 'Greninja'; + } + return name + (level === 100 ? '' : ', L' + level) + + (this.gender === '' ? '' : ', ' + this.gender) + (this.set.shiny ? ', shiny' : ''); + } + getDetails = () => { const health = this.getHealth(); - let details = this.details; - if (this.illusion) { - const level = this.battle.ruleTable.has('illusionlevelmod') ? this.illusion.level : this.level; - let displayedSpeciesName = this.illusion.species.name; - if (displayedSpeciesName === 'Greninja-Bond') displayedSpeciesName = 'Greninja'; - const illusionDetails = displayedSpeciesName + (level === 100 ? '' : ', L' + level) + - (this.illusion.gender === '' ? '' : ', ' + this.illusion.gender) + (this.illusion.set.shiny ? ', shiny' : ''); - details = illusionDetails; - } + let details = this.getSimpleDetails(null, true); if (this.terastallized) details += `, tera:${this.terastallized}`; return {side: health.side, secret: `${details}|${health.secret}`, shared: `${details}|${health.shared}`}; }; @@ -1320,8 +1323,7 @@ export class Pokemon { // 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; + if (['Ogerpon', 'Terapagos'].includes(this.species.baseSpecies) && this.canTerastallize) this.canTerastallize = false; return true; } @@ -1387,8 +1389,7 @@ export class Pokemon { this.illusion ? this.illusion.species.name : species.baseSpecies; if (isPermanent) { this.baseSpecies = rawSpecies; - this.details = species.name + (this.level === 100 ? '' : ', L' + this.level) + - (this.gender === '' ? '' : ', ' + this.gender) + (this.set.shiny ? ', shiny' : ''); + this.details = this.getSimpleDetails(); let details = (this.illusion || this).details; if (this.terastallized) details += `, tera:${this.terastallized}`; this.battle.add('detailschange', this, details); diff --git a/test/sim/misc/terapagos.js b/test/sim/misc/terapagos.js index a1276f21a290..64815320878f 100644 --- a/test/sim/misc/terapagos.js +++ b/test/sim/misc/terapagos.js @@ -24,6 +24,32 @@ describe(`Terapagos`, function () { assert.false(!!ditto.terastallized); }); + it(`shouldn't revert to Terapagos-Normal if it faints while not Terastallized`, function () { + battle = common.createBattle([[ + {species: 'terapagos', ability: 'terashift', moves: ['memento']}, + {species: 'darkrai', moves: ['darkpulse']}, + ], [ + {species: 'mareep', ability: 'static', moves: ['sleeptalk']}, + ]]); + const terapagos = battle.p1.active[0]; + battle.makeChoices(); + assert.species(terapagos, 'Terapagos-Terastal'); + assert.equal(terapagos.ability, 'terashell', `Expected ${terapagos}'ability to be Tera Shell, not ${terapagos.ability}`); + }); + + it(`should revert to Terapagos-Normal if it faints while Terastallized`, function () { + battle = common.createBattle([[ + {species: 'terapagos', ability: 'terashift', moves: ['memento']}, + {species: 'darkrai', moves: ['darkpulse']}, + ], [ + {species: 'mareep', ability: 'static', moves: ['sleeptalk']}, + ]]); + const terapagos = battle.p1.active[0]; + battle.makeChoices('move memento terastallize', 'auto'); + assert.species(terapagos, 'Terapagos'); + assert.equal(terapagos.ability, 'terashift', `Expected ${terapagos}'ability to be Tera Shift, not ${terapagos.ability}`); + }); + it(`[Hackmons] should not cause Terapagos-Terastal to become Terapagos-Stellar if the user is Transformed`, function () { battle = common.createBattle([[ {species: 'terapagos', ability: 'terashift', moves: ['transform'], teraType: 'Stellar'},