import { SelectModifierPhase } from "../battle-phases"; import BattleScene, { Button } from "../battle-scene"; import { ModifierType, ModifierTypeOption, PokemonBaseStatBoosterModifierType, PokemonHpRestoreModifierType, PokemonReviveModifierType } from "../modifier/modifier-type"; import Pokemon, { AiType, EnemyPokemon, PlayerPokemon, PokemonMove } from "../pokemon"; import { Species } from "../data/enums/species"; import BattleMessageUiHandler from "../ui/battle-message-ui-handler"; import CommandUiHandler from "../ui/command-ui-handler"; import FightUiHandler from "../ui/fight-ui-handler"; import MessageUiHandler from "../ui/message-ui-handler"; import ModifierSelectUiHandler from "../ui/modifier-select-ui-handler"; import PartyUiHandler, { PartyUiMode } from "../ui/party-ui-handler"; import ConfirmUiHandler from "../ui/confirm-ui-handler"; import { Mode } from "../ui/ui"; import { ModifierTier } from "../modifier/modifier-tier"; export function initAutoPlay() { const thisArg = this as BattleScene; PlayerPokemon.prototype.getNextMove = EnemyPokemon.prototype.getNextMove; const playerPokemon = this.getParty()[0] as PlayerPokemon; const messageUiHandler = this.ui.handlers[Mode.MESSAGE] as BattleMessageUiHandler; const commandUiHandler = this.ui.handlers[Mode.COMMAND] as CommandUiHandler; const fightUiHandler = this.ui.handlers[Mode.FIGHT] as FightUiHandler; const partyUiHandler = this.ui.handlers[Mode.PARTY] as PartyUiHandler; const confirmUiHandler = this.ui.handlers[Mode.CONFIRM] as ConfirmUiHandler; const modifierSelectUiHandler = this.ui.handlers[Mode.MODIFIER_SELECT] as ModifierSelectUiHandler; const getBestPartyMemberIndex = () => { const enemyPokemon = thisArg.getEnemyPokemon(); const party = thisArg.getParty(); let bestPartyMemberIndex = -1; let bestPartyMemberEffectiveness = 0.5; for (let p = 0; p < party.length; p++) { const pokemon = party[p]; if (pokemon.getHpRatio() <= 0.2) continue; const effectiveness = enemyPokemon ? getMaxMoveEffectiveness(pokemon, enemyPokemon) / getMaxMoveEffectiveness(enemyPokemon, pokemon) : 1; if (effectiveness > bestPartyMemberEffectiveness) { bestPartyMemberIndex = p; bestPartyMemberEffectiveness = effectiveness; } if (enemyPokemon) console.log(p, Species[pokemon.species.speciesId], '->', Species[enemyPokemon.species.speciesId], effectiveness); } if (bestPartyMemberIndex === -1) { let highestHpValue = -1; for (let p = 0; p < party.length; p++) { const pokemon = party[p]; if (pokemon.hp > highestHpValue) { highestHpValue = pokemon.hp; bestPartyMemberIndex = p; } } } return bestPartyMemberIndex; }; const getMaxMoveEffectiveness = (attacker: Pokemon, defender: Pokemon) => { let maxEffectiveness = 0.5; for (let m of attacker.getMoveset()) { const moveType = m.getMove().type; const effectiveness = defender.getAttackMoveEffectiveness(moveType); if (effectiveness > maxEffectiveness) maxEffectiveness = effectiveness; } return maxEffectiveness; }; let nextPartyMemberIndex = -1; const originalMessageUiHandlerShowText = MessageUiHandler.prototype.showText; MessageUiHandler.prototype.showText = function (text: string, delay?: integer, callback?: Function, callbackDelay?: integer, prompt?: boolean, promptDelay?: integer) { if (thisArg.auto) { delay = 1; callbackDelay = 0; promptDelay = 0; } originalMessageUiHandlerShowText.apply(this, [ text, delay, callback, callbackDelay, prompt, promptDelay ]); }; const originalMessageUiHandlerShowPrompt = MessageUiHandler.prototype.showPrompt; MessageUiHandler.prototype.showPrompt = function (callback: Function, callbackDelay: integer) { if (thisArg.auto) callbackDelay = 0; originalMessageUiHandlerShowPrompt.apply(this, [ callback, callbackDelay ]); if (thisArg.auto) thisArg.time.delayedCall(20, () => this.processInput(Button.ACTION)); }; const originalMessageUiHandlerPromptLevelUpStats = messageUiHandler.promptLevelUpStats; messageUiHandler.promptLevelUpStats = function (partyMemberIndex: integer, prevStats: integer[], showTotals: boolean): Promise { return new Promise(resolve => { originalMessageUiHandlerPromptLevelUpStats.apply(this, [ partyMemberIndex, prevStats, showTotals ]).then(() => resolve()); if (thisArg.auto) thisArg.time.delayedCall(20, () => this.processInput(Button.ACTION)); }); }; const originalCommandUiHandlerShow = commandUiHandler.show; commandUiHandler.show = function (args: any[]) { originalCommandUiHandlerShow.apply(this, [ args ]); if (thisArg.auto) { thisArg.time.delayedCall(20, () => { const bestPartyMemberIndex = getBestPartyMemberIndex(); if (bestPartyMemberIndex) { console.log(bestPartyMemberIndex, thisArg.getParty()) console.log('Switching to ', Species[thisArg.getParty()[bestPartyMemberIndex].species.speciesId]); nextPartyMemberIndex = bestPartyMemberIndex; commandUiHandler.setCursor(2); thisArg.time.delayedCall(20, () => this.processInput(Button.ACTION)); } else { commandUiHandler.setCursor(0); thisArg.time.delayedCall(20, () => this.processInput(Button.ACTION)); } }); } }; const originalFightUiHandlerShow = fightUiHandler.show; fightUiHandler.show = function (args: any[]) { originalFightUiHandlerShow.apply(this, [ args ]); if (thisArg.auto) { if (!playerPokemon.aiType) playerPokemon.aiType = AiType.SMART; thisArg.time.delayedCall(20, () => { const nextMove = playerPokemon.getNextMove() as PokemonMove; fightUiHandler.setCursor(playerPokemon.getMoveset().indexOf(nextMove)); thisArg.time.delayedCall(20, () => this.processInput(Button.ACTION)); }); } }; const originalPartyUiHandlerShow = partyUiHandler.show; partyUiHandler.show = function (args: any[]) { originalPartyUiHandlerShow.apply(this, [ args ]); if (thisArg.auto) { thisArg.time.delayedCall(20, () => { if (nextPartyMemberIndex === -1) nextPartyMemberIndex = getBestPartyMemberIndex(); partyUiHandler.setCursor(nextPartyMemberIndex); nextPartyMemberIndex = -1; if ((partyUiHandler.partyUiMode === PartyUiMode.MODIFIER || partyUiHandler.partyUiMode === PartyUiMode.TM_MODIFIER) || partyUiHandler.getCursor()) { this.processInput(Button.ACTION); thisArg.time.delayedCall(250, () => this.processInput(Button.ACTION)); } else this.processInput(Button.CANCEL); }); } }; const originalSwitchCheckUiHandlerShow = confirmUiHandler.show; confirmUiHandler.show = function (args: any[]) { originalSwitchCheckUiHandlerShow.apply(this, [ args ]); if (thisArg.auto) { const bestPartyMemberIndex = getBestPartyMemberIndex(); thisArg.time.delayedCall(20, () => { if (bestPartyMemberIndex) nextPartyMemberIndex = bestPartyMemberIndex; confirmUiHandler.setCursor(bestPartyMemberIndex ? 1 : 0); thisArg.time.delayedCall(20, () => this.processInput(Button.ACTION)); }); } } const tryGetBestModifier = (modifierTypeOptions: Array, predicate: Function) => { for (let mt = 0; mt < modifierTypeOptions.length; mt++) { const modifierTypeOption = modifierTypeOptions[mt]; if (predicate(modifierTypeOption.type)) { return mt; } } return -1; }; const originalModifierSelectUiHandlerShow = modifierSelectUiHandler.show; modifierSelectUiHandler.show = function (args: any[]) { if (!thisArg.auto) { originalModifierSelectUiHandlerShow.apply(this, [ args ]); return; } if (modifierSelectUiHandler.active) return; thisArg.time.delayedCall(20, () => { originalModifierSelectUiHandlerShow.apply(this, [ args ]); const party = thisArg.getParty(); const modifierTypeOptions = modifierSelectUiHandler.options.map(o => o.modifierTypeOption); const faintedPartyMemberIndex = party.findIndex(p => p.isFainted()); const lowHpPartyMemberIndex = party.findIndex(p => p.getHpRatio() <= 0.5); const criticalHpPartyMemberIndex = party.findIndex(p => p.getHpRatio() <= 0.25); let optionIndex = tryGetBestModifier(modifierTypeOptions, (modifierType: ModifierType) => { if (modifierType instanceof PokemonHpRestoreModifierType) { if (modifierType instanceof PokemonReviveModifierType) { if (faintedPartyMemberIndex > -1) { nextPartyMemberIndex = faintedPartyMemberIndex; return true; } } else if (criticalHpPartyMemberIndex > -1){ nextPartyMemberIndex = criticalHpPartyMemberIndex; return true; } } }); if (optionIndex === -1) { optionIndex = tryGetBestModifier(modifierTypeOptions, (modifierType: ModifierType) => { if (modifierType.tier >= ModifierTier.ULTRA) { nextPartyMemberIndex = 0; return true; } }); } if (optionIndex === -1) { optionIndex = tryGetBestModifier(modifierTypeOptions, (modifierType: ModifierType) => { if (modifierType instanceof PokemonBaseStatBoosterModifierType) { nextPartyMemberIndex = 0; return true; } }); } if (optionIndex === -1) { optionIndex = tryGetBestModifier(modifierTypeOptions, (modifierType: ModifierType) => { if (lowHpPartyMemberIndex && modifierType instanceof PokemonHpRestoreModifierType && !(ModifierType instanceof PokemonReviveModifierType)) { nextPartyMemberIndex = lowHpPartyMemberIndex; return true; } }); } if (optionIndex === -1) optionIndex = 0; const trySelectModifier = () => { modifierSelectUiHandler.setCursor(optionIndex); thisArg.time.delayedCall(20, () => { modifierSelectUiHandler.processInput(Button.ACTION); thisArg.time.delayedCall(250, () => { console.log(modifierTypeOptions[optionIndex]?.type.name); if (thisArg.getCurrentPhase() instanceof SelectModifierPhase) { if (optionIndex < modifierSelectUiHandler.options.length - 1) { optionIndex++; thisArg.time.delayedCall(250, () => trySelectModifier()); } else modifierSelectUiHandler.processInput(Button.CANCEL); } }); }); }; thisArg.time.delayedCall(4000, () => trySelectModifier()); }); } }