From e1e174903430d0cea06cd97f56e9ff49d9bcdb43 Mon Sep 17 00:00:00 2001 From: Sean <34076225+Rezzo64@users.noreply.github.com> Date: Sat, 28 Oct 2023 13:00:45 +0100 Subject: [PATCH] October 28, 2023: Merge with PS (#12) * Populate Tooltips with information from OTS (#2161) * Fix `showteam` in replays (#2164) * Fix speed buttons on downloaded replays (#2162) * fix package.json * remove DH2 sub --------- Co-authored-by: Karthik <32044378+Karthik99999@users.noreply.github.com> --- js/replay-embed.template.js | 24 +++- replays/battle.php | 2 +- src/battle-log.ts | 29 +++- src/battle-text-parser.ts | 2 +- src/battle.ts | 278 +++++++++++++++++++++++++++++++++++- 5 files changed, 318 insertions(+), 17 deletions(-) diff --git a/js/replay-embed.template.js b/js/replay-embed.template.js index 426bafa2d..b037fec05 100644 --- a/js/replay-embed.template.js +++ b/js/replay-embed.template.js @@ -84,7 +84,7 @@ var Replays = { paused: true, }); - this.$('.replay-controls-2').html('
Speed:
Color scheme:
'); + this.$('.replay-controls-2').html('
Speed:
Color scheme:
'); // this works around a WebKit/Blink bug relating to float layout var rc2 = this.$('.replay-controls-2')[0]; @@ -141,13 +141,23 @@ var Replays = { break; case 'speed': - var speedTable = { - fast: 8, - normal: 800, - slow: 2500, - reallyslow: 5000 + var fadeTable = { + hyperfast: 40, + fast: 50, + normal: 300, + slow: 500, + reallyslow: 1000 }; - this.battle.messageDelay = speedTable[value]; + var delayTable = { + hyperfast: 1, + fast: 1, + normal: 1, + slow: 1000, + reallyslow: 3000 + }; + this.battle.messageShownTime = delayTable[value]; + this.battle.messageFadeTime = fadeTable[value]; + this.battle.scene.updateAcceleration(); break; } }, diff --git a/replays/battle.php b/replays/battle.php index d369f72d7..6e7e0b0c1 100644 --- a/replays/battle.php +++ b/replays/battle.php @@ -113,7 +113,7 @@ function userid($username) {
Speed: -
+
Color scheme: diff --git a/src/battle-log.ts b/src/battle-log.ts index df9439ada..71cfd52cf 100644 --- a/src/battle-log.ts +++ b/src/battle-log.ts @@ -99,8 +99,8 @@ export class BattleLog { } add(args: Args, kwArgs?: KWArgs, preempt?: boolean) { if (kwArgs?.silent) return; - if (this.scene?.battle.seeking) { - const battle = this.scene.battle; + const battle = this.scene?.battle; + if (battle?.seeking) { if (battle.stepQueue.length > 2000) { // adding elements gets slower and slower the more there are // (so showing 100 turns takes around 2 seconds, and 1000 turns takes around a minute) @@ -122,7 +122,6 @@ export class BattleLog { if (!['name', 'n'].includes(args[0])) this.lastRename = null; switch (args[0]) { case 'chat': case 'c': case 'c:': - let battle = this.scene?.battle; let name; let message; if (args[0] === 'c:') { @@ -259,6 +258,27 @@ export class BattleLog { app.rooms[roomid].notifyOnce(title, body, 'highlight'); break; + case 'showteam': { + if (!battle) return; + const team = battle.unpackTeam(args[2]); + if (!team.length) return; + const side = battle.getSide(args[1]); + const exportedTeam = team.map(set => { + let buf = battle.exportTeam([set]).replace(/\n/g, '
'); + if (set.name && set.name !== set.species) { + buf = buf.replace(set.name, BattleLog.sanitizeHTML(`
${set.name}`)); + } else { + buf = buf.replace(set.species, `
${set.species}`); + } + if (set.item) { + buf = buf.replace(set.item, `${set.item} `); + } + return buf; + }).join(''); + divHTML = `
Open Team Sheet for ${side.name}${exportedTeam}
`; + break; + } + case 'seed': case 'choice': case ':': case 'timer': case 't:': case 'J': case 'L': case 'N': case 'spectator': case 'spectatorleave': case 'initdone': @@ -1140,9 +1160,6 @@ export class BattleLog { buf += '\n'; buf += '\n'; buf += `${BattleLog.escapeHTML(battle.tier)} replay: ${BattleLog.escapeHTML(battle.p1.name)} vs. ${BattleLog.escapeHTML(battle.p2.name)}\n`; - buf += '\n'; buf += '
\n'; buf += '\n'; buf += '
\n'; diff --git a/src/battle-text-parser.ts b/src/battle-text-parser.ts index b64e399e0..11da55e34 100644 --- a/src/battle-text-parser.ts +++ b/src/battle-text-parser.ts @@ -46,7 +46,7 @@ class BattleTextParser { case 'fieldhtml': case 'controlshtml': case 'bigerror': case 'debug': case 'tier': case 'challstr': case 'popup': case '': return [cmd, line.slice(index + 1)]; - case 'c': case 'chat': case 'uhtml': case 'uhtmlchange': case 'queryresponse': + case 'c': case 'chat': case 'uhtml': case 'uhtmlchange': case 'queryresponse': case 'showteam': // three parts const index2a = line.indexOf('|', index + 1); return [cmd, line.slice(index + 1, index2a), line.slice(index2a + 1)]; diff --git a/src/battle.ts b/src/battle.ts index 8127a3b85..c8669997b 100644 --- a/src/battle.ts +++ b/src/battle.ts @@ -741,14 +741,19 @@ export class Side { this.battle.scene.removeSideCondition(this.n, id); } addPokemon(name: string, ident: string, details: string, replaceSlot = -1) { - const oldItem = replaceSlot >= 0 ? this.pokemon[replaceSlot].item : undefined; + const oldPokemon = replaceSlot >= 0 ? this.pokemon[replaceSlot] : undefined; const data = this.battle.parseDetails(name, ident, details); const poke = new Pokemon(data, this); - if (oldItem) poke.item = oldItem; + if (oldPokemon) { + poke.item = oldPokemon.item; + poke.baseAbility = oldPokemon.baseAbility; + poke.teraType = oldPokemon.teraType; + } if (!poke.ability && poke.baseAbility) poke.ability = poke.baseAbility; poke.reset(); + if (oldPokemon?.moveTrack.length) poke.moveTrack = oldPokemon.moveTrack; if (replaceSlot >= 0) { this.pokemon[replaceSlot] = poke; @@ -3339,6 +3344,256 @@ export class Battle { } as any; } + // Taken from Storage, they need to be implemented here so `showteam` can work + unpackTeam(buf: string) { + if (!buf) return []; + + const team = []; + let i = 0; + let j = 0; + + while (true) { + const set: PokemonSet = {} as any; + team.push(set); + + // name + j = buf.indexOf('|', i); + set.name = buf.substring(i, j); + i = j + 1; + + // species + j = buf.indexOf('|', i); + set.species = Dex.species.get(buf.substring(i, j)).name || set.name; + i = j + 1; + + // item + j = buf.indexOf('|', i); + set.item = Dex.items.get(buf.substring(i, j)).name; + i = j + 1; + + // ability + j = buf.indexOf('|', i); + const ability = Dex.abilities.get(buf.substring(i, j)).name; + const species = Dex.species.get(set.species); + set.ability = (species.abilities && + ['', '0', '1', 'H', 'S'].includes(ability) ? species.abilities[ability as '0' || '0'] : ability); + i = j + 1; + + // moves + j = buf.indexOf('|', i); + set.moves = buf.substring(i, j).split(',').map(function (moveid) { + return Dex.moves.get(moveid).name; + }); + i = j + 1; + + // nature + j = buf.indexOf('|', i); + set.nature = buf.substring(i, j) as NatureName; + if (set.nature as any === 'undefined') delete set.nature; + i = j + 1; + + // evs + j = buf.indexOf('|', i); + if (j !== i) { + const evstring = buf.substring(i, j); + if (evstring.length > 5) { + const evs = evstring.split(','); + set.evs = { + hp: Number(evs[0]) || 0, + atk: Number(evs[1]) || 0, + def: Number(evs[2]) || 0, + spa: Number(evs[3]) || 0, + spd: Number(evs[4]) || 0, + spe: Number(evs[5]) || 0, + }; + } else if (evstring === '0') { + set.evs = {hp: 0, atk: 0, def: 0, spa: 0, spd: 0, spe: 0}; + } + } + i = j + 1; + + // gender + j = buf.indexOf('|', i); + if (i !== j) set.gender = buf.substring(i, j); + i = j + 1; + + // ivs + j = buf.indexOf('|', i); + if (j !== i) { + const ivs = buf.substring(i, j).split(','); + set.ivs = { + hp: ivs[0] === '' ? 31 : Number(ivs[0]), + atk: ivs[1] === '' ? 31 : Number(ivs[1]), + def: ivs[2] === '' ? 31 : Number(ivs[2]), + spa: ivs[3] === '' ? 31 : Number(ivs[3]), + spd: ivs[4] === '' ? 31 : Number(ivs[4]), + spe: ivs[5] === '' ? 31 : Number(ivs[5]), + }; + } + i = j + 1; + + // shiny + j = buf.indexOf('|', i); + if (i !== j) set.shiny = true; + i = j + 1; + + // level + j = buf.indexOf('|', i); + if (i !== j) set.level = parseInt(buf.substring(i, j), 10); + i = j + 1; + + // happiness + j = buf.indexOf(']', i); + let misc; + if (j < 0) { + if (i < buf.length) misc = buf.substring(i).split(',', 6); + } else { + if (i !== j) misc = buf.substring(i, j).split(',', 6); + } + if (misc) { + set.happiness = (misc[0] ? Number(misc[0]) : 255); + set.hpType = misc[1]; + set.pokeball = misc[2]; + set.gigantamax = !!misc[3]; + set.dynamaxLevel = (misc[4] ? Number(misc[4]) : 10); + set.teraType = misc[5]; + } + if (j < 0) break; + i = j + 1; + } + + return team; + } + exportTeam(team: PokemonSet[] | string, gen = this.gen, hidestats = false) { + if (!team) return ''; + if (typeof team === 'string') { + if (team.indexOf('\n') >= 0) return team; + team = this.unpackTeam(team); + } + let text = ''; + for (const curSet of team) { + if (curSet.name && curSet.name !== curSet.species) { + text += '' + curSet.name + ' (' + curSet.species + ')'; + } else { + text += '' + curSet.species; + } + if (curSet.gender === 'M') text += ' (M)'; + if (curSet.gender === 'F') text += ' (F)'; + if (curSet.item) { + text += ' @ ' + curSet.item; + } + text += " \n"; + if (curSet.ability) { + text += 'Ability: ' + curSet.ability + " \n"; + } + if (curSet.level && curSet.level !== 100) { + text += 'Level: ' + curSet.level + " \n"; + } + if (curSet.shiny) { + text += 'Shiny: Yes \n'; + } + if (typeof curSet.happiness === 'number' && curSet.happiness !== 255 && !isNaN(curSet.happiness)) { + text += 'Happiness: ' + curSet.happiness + " \n"; + } + if (curSet.pokeball) { + text += 'Pokeball: ' + curSet.pokeball + " \n"; + } + if (curSet.hpType) { + text += 'Hidden Power: ' + curSet.hpType + " \n"; + } + if (typeof curSet.dynamaxLevel === 'number' && curSet.dynamaxLevel !== 10 && !isNaN(curSet.dynamaxLevel)) { + text += 'Dynamax Level: ' + curSet.dynamaxLevel + " \n"; + } + if (curSet.gigantamax) { + text += 'Gigantamax: Yes \n'; + } + if (gen === 9) { + const species = Dex.species.get(curSet.species); + text += 'Tera Type: ' + (species.forceTeraType || curSet.teraType || species.types[0]) + " \n"; + } + if (!hidestats) { + let first = true; + if (curSet.evs) { + let j: StatName; + for (j in BattleStatNames) { + if (!curSet.evs[j]) continue; + if (first) { + text += 'EVs: '; + first = false; + } else { + text += ' / '; + } + text += '' + curSet.evs[j] + ' ' + BattleStatNames[j]; + } + } + if (!first) { + text += " \n"; + } + if (curSet.nature) { + text += '' + curSet.nature + ' Nature' + " \n"; + } + first = true; + if (curSet.ivs) { + let defaultIvs = true; + let hpType = ''; + for (const move of curSet.moves) { + if (move.substr(0, 13) === 'Hidden Power ' && move.substr(0, 14) !== 'Hidden Power [') { + hpType = move.substr(13); + if (!Dex.types.isName(hpType)) { + alert(move + " is not a valid Hidden Power type."); + continue; + } + let stat: StatName; + for (stat in BattleStatNames) { + if ((curSet.ivs[stat] === undefined ? 31 : curSet.ivs[stat]) !== (Dex.types.get(hpType).HPivs?.[stat] || 31)) { + defaultIvs = false; + break; + } + } + } + } + if (defaultIvs && !hpType) { + let stat: StatName; + for (stat in BattleStatNames) { + if (curSet.ivs[stat] !== 31 && curSet.ivs[stat] !== undefined) { + defaultIvs = false; + break; + } + } + } + if (!defaultIvs) { + let stat: StatName; + for (stat in BattleStatNames) { + if (typeof curSet.ivs[stat] === 'undefined' || isNaN(curSet.ivs[stat]) || curSet.ivs[stat] === 31) continue; + if (first) { + text += 'IVs: '; + first = false; + } else { + text += ' / '; + } + text += '' + curSet.ivs[stat] + ' ' + BattleStatNames[stat]; + } + } + } + if (!first) { + text += " \n"; + } + } + if (curSet.moves) { + for (let move of curSet.moves) { + if (move.substr(0, 13) === 'Hidden Power ') { + move = move.substr(0, 13) + '[' + move.substr(13) + ']'; + } + if (move) { + text += '- ' + move + " \n"; + } + } + } + text += "\n"; + } + return text; + } + add(command?: string) { if (command) this.stepQueue.push(command); @@ -3590,6 +3845,25 @@ export class Battle { this.scene.teamPreview(); break; } + case 'showteam': { + const team = this.unpackTeam(args[2]); + if (!team.length) return; + const side = this.getSide(args[1]); + side.clearPokemon(); + for (const set of team) { + const details = set.species + (!set.level || set.level === 100 ? '' : ', L' + set.level) + + (!set.gender || set.gender === 'N' ? '' : ', ' + set.gender) + (set.shiny ? ', shiny' : ''); + const pokemon = side.addPokemon('', '', details); + if (set.item) pokemon.item = set.item; + if (set.ability) pokemon.rememberAbility(set.ability); + for (const move of set.moves) { + pokemon.rememberMove(move, 0); + } + if (set.teraType) pokemon.teraType = set.teraType; + } + this.log(args, kwArgs); + break; + } case 'switch': case 'drag': case 'replace': { this.endLastTurn(); let poke = this.getSwitchedPokemon(args[1], args[2])!;