pokerogue/src/ui/mystery-encounter-ui-handler.ts
Sirz Benjie 93745f14b7
[Refactor] Decouple phase system from battle-scene (#5953)
* Move phase logic into its own class

* Move ts ignore comment
2025-06-07 17:59:30 -07:00

717 lines
25 KiB
TypeScript

import { addBBCodeTextObject, getBBCodeFrag, TextStyle } from "./text";
import { UiMode } from "#enums/ui-mode";
import UiHandler from "./ui-handler";
import { Button } from "#enums/buttons";
import { addWindow, WindowVariant } from "./ui-theme";
import type { MysteryEncounterPhase } from "../phases/mystery-encounter-phases";
import { PartyUiMode } from "./party-ui-handler";
import type MysteryEncounterOption from "#app/data/mystery-encounters/mystery-encounter-option";
import { fixedInt, isNullOrUndefined } from "#app/utils/common";
import { getPokeballAtlasKey } from "../data/pokeball";
import type { OptionSelectSettings } from "#app/data/mystery-encounters/utils/encounter-phase-utils";
import { getEncounterText } from "#app/data/mystery-encounters/utils/encounter-dialogue-utils";
import i18next from "i18next";
import { MysteryEncounterOptionMode } from "#enums/mystery-encounter-option-mode";
import { MysteryEncounterTier } from "#enums/mystery-encounter-tier";
import type BBCodeText from "phaser3-rex-plugins/plugins/bbcodetext";
import { globalScene } from "#app/global-scene";
export default class MysteryEncounterUiHandler extends UiHandler {
private cursorContainer: Phaser.GameObjects.Container;
private cursorObj?: Phaser.GameObjects.Image;
private optionsContainer: Phaser.GameObjects.Container;
// Length = max number of allowable options (4)
private optionScrollTweens: (Phaser.Tweens.Tween | null)[] = new Array(4).fill(null);
private tooltipWindow: Phaser.GameObjects.NineSlice;
private tooltipContainer: Phaser.GameObjects.Container;
private tooltipScrollTween?: Phaser.Tweens.Tween;
private descriptionWindow: Phaser.GameObjects.NineSlice;
private descriptionContainer: Phaser.GameObjects.Container;
private descriptionScrollTween?: Phaser.Tweens.Tween;
private rarityBall: Phaser.GameObjects.Sprite;
private dexProgressWindow: Phaser.GameObjects.NineSlice;
private dexProgressContainer: Phaser.GameObjects.Container;
private showDexProgress = false;
private overrideSettings?: OptionSelectSettings;
private encounterOptions: MysteryEncounterOption[] = [];
private optionsMeetsReqs: boolean[];
protected viewPartyIndex = 0;
protected viewPartyXPosition = 0;
protected blockInput = true;
constructor() {
super(UiMode.MYSTERY_ENCOUNTER);
}
override setup() {
const ui = this.getUi();
this.cursorContainer = globalScene.add.container(18, -38.7);
this.cursorContainer.setVisible(false);
ui.add(this.cursorContainer);
this.optionsContainer = globalScene.add.container(12, -38.7);
this.optionsContainer.setVisible(false);
ui.add(this.optionsContainer);
this.dexProgressContainer = globalScene.add.container(214, -43);
this.dexProgressContainer.setVisible(false);
ui.add(this.dexProgressContainer);
this.descriptionContainer = globalScene.add.container(0, -152);
this.descriptionContainer.setVisible(false);
ui.add(this.descriptionContainer);
this.tooltipContainer = globalScene.add.container(210, -48);
this.tooltipContainer.setVisible(false);
ui.add(this.tooltipContainer);
this.setCursor(this.getCursor());
this.descriptionWindow = addWindow(0, 0, 150, 105, false, false, 0, 0, WindowVariant.THIN);
this.descriptionContainer.add(this.descriptionWindow);
this.tooltipWindow = addWindow(0, 0, 110, 48, false, false, 0, 0, WindowVariant.THIN);
this.tooltipContainer.add(this.tooltipWindow);
this.dexProgressWindow = addWindow(0, 0, 24, 28, false, false, 0, 0, WindowVariant.THIN);
this.dexProgressContainer.add(this.dexProgressWindow);
this.rarityBall = globalScene.add.sprite(141, 9, "pb");
this.rarityBall.setScale(0.75);
this.descriptionContainer.add(this.rarityBall);
const dexProgressIndicator = globalScene.add.sprite(12, 10, "encounter_radar");
dexProgressIndicator.setScale(0.8);
this.dexProgressContainer.add(dexProgressIndicator);
this.dexProgressContainer.setInteractive(new Phaser.Geom.Rectangle(0, 0, 24, 28), Phaser.Geom.Rectangle.Contains);
}
override show(args: any[]): boolean {
super.show(args);
this.overrideSettings = (args[0] as OptionSelectSettings) ?? {};
const showDescriptionContainer = isNullOrUndefined(this.overrideSettings?.hideDescription)
? true
: !this.overrideSettings.hideDescription;
const slideInDescription = isNullOrUndefined(this.overrideSettings?.slideInDescription)
? true
: this.overrideSettings.slideInDescription;
const startingCursorIndex = this.overrideSettings?.startingCursorIndex ?? 0;
this.cursorContainer.setVisible(true);
this.descriptionContainer.setVisible(showDescriptionContainer);
this.optionsContainer.setVisible(true);
this.dexProgressContainer.setVisible(true);
this.displayEncounterOptions(slideInDescription);
const cursor = this.getCursor();
if (cursor === (this.optionsContainer?.length || 0) - 1) {
// Always resets cursor on view party button if it was last there
this.setCursor(cursor);
} else {
this.setCursor(startingCursorIndex);
}
if (this.blockInput) {
setTimeout(() => {
this.unblockInput();
}, 1000);
}
this.displayOptionTooltip();
return true;
}
override processInput(button: Button): boolean {
const ui = this.getUi();
let success = false;
const cursor = this.getCursor();
if (button === Button.CANCEL || button === Button.ACTION) {
if (button === Button.ACTION) {
const selected = this.encounterOptions[cursor];
if (cursor === this.viewPartyIndex) {
// Handle view party
success = true;
const overrideSettings: OptionSelectSettings = {
...this.overrideSettings,
slideInDescription: false,
};
globalScene.ui.setMode(UiMode.PARTY, PartyUiMode.CHECK, -1, () => {
globalScene.ui.setMode(UiMode.MYSTERY_ENCOUNTER, overrideSettings);
setTimeout(() => {
this.setCursor(this.viewPartyIndex);
this.unblockInput();
}, 300);
});
} else if (
this.blockInput ||
(!this.optionsMeetsReqs[cursor] &&
(selected.optionMode === MysteryEncounterOptionMode.DISABLED_OR_DEFAULT ||
selected.optionMode === MysteryEncounterOptionMode.DISABLED_OR_SPECIAL))
) {
success = false;
} else {
if (
(globalScene.phaseManager.getCurrentPhase() as MysteryEncounterPhase).handleOptionSelect(selected, cursor)
) {
success = true;
} else {
ui.playError();
}
}
} else {
// TODO: If we need to handle cancel option? Maybe default logic to leave/run from encounter idk
}
} else {
switch (this.optionsContainer.getAll()?.length) {
// biome-ignore lint/suspicious/useDefaultSwitchClauseLast: Default shares logic with case 3 and it makes more sense for the statements to be ordered by the case value
default:
case 3:
success = this.handleTwoOptionMoveInput(button);
break;
case 4:
success = this.handleThreeOptionMoveInput(button);
break;
case 5:
success = this.handleFourOptionMoveInput(button);
break;
}
this.displayOptionTooltip();
}
if (success) {
ui.playSelect();
}
return success;
}
private handleTwoOptionMoveInput(button: Button): boolean {
let success = false;
const cursor = this.getCursor();
switch (button) {
case Button.UP:
if (cursor < this.viewPartyIndex) {
success = this.setCursor(this.viewPartyIndex);
}
break;
case Button.DOWN:
if (cursor === this.viewPartyIndex) {
success = this.setCursor(1);
}
break;
case Button.LEFT:
if (cursor > 0) {
success = this.setCursor(cursor - 1);
}
break;
case Button.RIGHT:
if (cursor < this.viewPartyIndex) {
success = this.setCursor(cursor + 1);
}
break;
}
return success;
}
private handleThreeOptionMoveInput(button: Button): boolean {
let success = false;
const cursor = this.getCursor();
switch (button) {
case Button.UP:
if (cursor === 2) {
success = this.setCursor(cursor - 2);
} else {
success = this.setCursor(this.viewPartyIndex);
}
break;
case Button.DOWN:
if (cursor === this.viewPartyIndex) {
success = this.setCursor(1);
} else {
success = this.setCursor(2);
}
break;
case Button.LEFT:
if (cursor === this.viewPartyIndex) {
success = this.setCursor(1);
} else if (cursor === 1) {
success = this.setCursor(cursor - 1);
}
break;
case Button.RIGHT:
if (cursor === 1) {
success = this.setCursor(this.viewPartyIndex);
} else if (cursor < 1) {
success = this.setCursor(cursor + 1);
}
break;
}
return success;
}
private handleFourOptionMoveInput(button: Button): boolean {
let success = false;
const cursor = this.getCursor();
switch (button) {
case Button.UP:
if (cursor >= 2 && cursor !== this.viewPartyIndex) {
success = this.setCursor(cursor - 2);
} else {
success = this.setCursor(this.viewPartyIndex);
}
break;
case Button.DOWN:
if (cursor <= 1) {
success = this.setCursor(cursor + 2);
} else if (cursor === this.viewPartyIndex) {
success = this.setCursor(1);
}
break;
case Button.LEFT:
if (cursor === this.viewPartyIndex) {
success = this.setCursor(1);
} else if (cursor % 2 === 1) {
success = this.setCursor(cursor - 1);
}
break;
case Button.RIGHT:
if (cursor === 1) {
success = this.setCursor(this.viewPartyIndex);
} else if (cursor % 2 === 0 && cursor !== this.viewPartyIndex) {
success = this.setCursor(cursor + 1);
}
break;
}
return success;
}
/**
* When ME UI first displays, the option buttons will be disabled temporarily to prevent player accidentally clicking through hastily
* This method is automatically called after a short delay but can also be called manually
*/
unblockInput() {
if (this.blockInput) {
this.blockInput = false;
for (let i = 0; i < this.optionsContainer.length - 1; i++) {
const optionMode = this.encounterOptions[i].optionMode;
if (
!this.optionsMeetsReqs[i] &&
(optionMode === MysteryEncounterOptionMode.DISABLED_OR_DEFAULT ||
optionMode === MysteryEncounterOptionMode.DISABLED_OR_SPECIAL)
) {
continue;
}
(this.optionsContainer.getAt(i) as Phaser.GameObjects.Text).setAlpha(1);
}
}
}
override getCursor(): number {
return this.cursor ? this.cursor : 0;
}
override setCursor(cursor: number): boolean {
const prevCursor = this.getCursor();
const changed = prevCursor !== cursor;
if (changed) {
this.cursor = cursor;
}
this.viewPartyIndex = this.optionsContainer.getAll()?.length - 1;
if (!this.cursorObj) {
this.cursorObj = globalScene.add.image(0, 0, "cursor");
this.cursorContainer.add(this.cursorObj);
}
if (cursor === this.viewPartyIndex) {
this.cursorObj.setPosition(this.viewPartyXPosition, -17);
} else if (this.optionsContainer.getAll()?.length === 3) {
// 2 Options
this.cursorObj.setPosition(-10.5 + (cursor % 2 === 1 ? 100 : 0), 15);
} else if (this.optionsContainer.getAll()?.length === 4) {
// 3 Options
this.cursorObj.setPosition(-10.5 + (cursor % 2 === 1 ? 100 : 0), 7 + (cursor > 1 ? 16 : 0));
} else if (this.optionsContainer.getAll()?.length === 5) {
// 4 Options
this.cursorObj.setPosition(-10.5 + (cursor % 2 === 1 ? 100 : 0), 7 + (cursor > 1 ? 16 : 0));
}
return changed;
}
displayEncounterOptions(slideInDescription = true): void {
this.getUi().clearText();
const mysteryEncounter = globalScene.currentBattle.mysteryEncounter!;
this.encounterOptions = this.overrideSettings?.overrideOptions ?? mysteryEncounter.options;
this.optionsMeetsReqs = [];
const titleText: string | null = getEncounterText(
mysteryEncounter.dialogue.encounterOptionsDialogue?.title,
TextStyle.TOOLTIP_TITLE,
);
const descriptionText: string | null = getEncounterText(
mysteryEncounter.dialogue.encounterOptionsDialogue?.description,
TextStyle.TOOLTIP_CONTENT,
);
const queryText: string | null = getEncounterText(
mysteryEncounter.dialogue.encounterOptionsDialogue?.query,
TextStyle.TOOLTIP_CONTENT,
);
// Clear options container (except cursor)
this.optionsContainer.removeAll(true);
// Options Window
for (let i = 0; i < this.encounterOptions.length; i++) {
const option = this.encounterOptions[i];
let optionText: BBCodeText;
switch (this.encounterOptions.length) {
// biome-ignore lint/suspicious/useDefaultSwitchClauseLast: default shares logic with case 2 and it makes more sense for the statements to be ordered by the case number
default:
case 2:
optionText = addBBCodeTextObject(i % 2 === 0 ? 0 : 100, 8, "-", TextStyle.WINDOW, {
fontSize: "80px",
lineSpacing: -8,
});
break;
case 3:
optionText = addBBCodeTextObject(i % 2 === 0 ? 0 : 100, i < 2 ? 0 : 16, "-", TextStyle.WINDOW, {
fontSize: "80px",
lineSpacing: -8,
});
break;
case 4:
optionText = addBBCodeTextObject(i % 2 === 0 ? 0 : 100, i < 2 ? 0 : 16, "-", TextStyle.WINDOW, {
fontSize: "80px",
lineSpacing: -8,
});
break;
}
this.optionsMeetsReqs.push(option.meetsRequirements());
const optionDialogue = option.dialogue!;
const label =
!this.optionsMeetsReqs[i] && optionDialogue.disabledButtonLabel
? optionDialogue.disabledButtonLabel
: optionDialogue.buttonLabel;
let text: string | null;
if (
option.hasRequirements() &&
this.optionsMeetsReqs[i] &&
(option.optionMode === MysteryEncounterOptionMode.DEFAULT_OR_SPECIAL ||
option.optionMode === MysteryEncounterOptionMode.DISABLED_OR_SPECIAL)
) {
// Options with special requirements that are met are automatically colored green
text = getEncounterText(label, TextStyle.ME_OPTION_SPECIAL);
} else {
text = getEncounterText(label, optionDialogue.style ? optionDialogue.style : TextStyle.ME_OPTION_DEFAULT);
}
if (text) {
optionText.setText(text);
}
if (
!this.optionsMeetsReqs[i] &&
(option.optionMode === MysteryEncounterOptionMode.DISABLED_OR_DEFAULT ||
option.optionMode === MysteryEncounterOptionMode.DISABLED_OR_SPECIAL)
) {
optionText.setAlpha(0.5);
}
if (this.blockInput) {
optionText.setAlpha(0.5);
}
// Sets up the mask that hides the option text to give an illusion of scrolling
const nonScrollWidth = 90;
const optionTextMaskRect = globalScene.make.graphics({});
optionTextMaskRect.setScale(6);
optionTextMaskRect.fillStyle(0xffffff);
optionTextMaskRect.beginPath();
optionTextMaskRect.fillRect(optionText.x + 11, optionText.y + 140, nonScrollWidth, 18);
const optionTextMask = optionTextMaskRect.createGeometryMask();
optionText.setMask(optionTextMask);
const optionTextWidth = optionText.displayWidth;
const tween = this.optionScrollTweens[i];
if (tween) {
tween.remove();
this.optionScrollTweens[i] = null;
}
// Animates the option text scrolling sideways
if (optionTextWidth > nonScrollWidth) {
this.optionScrollTweens[i] = globalScene.tweens.add({
targets: optionText,
delay: fixedInt(2000),
loop: -1,
hold: fixedInt(2000),
duration: fixedInt(((optionTextWidth - nonScrollWidth) / 15) * 2000),
x: `-=${optionTextWidth - nonScrollWidth}`,
});
}
this.optionsContainer.add(optionText);
}
// View Party Button
const viewPartyText = addBBCodeTextObject(
globalScene.game.canvas.width / 6,
-24,
getBBCodeFrag(i18next.t("mysteryEncounterMessages:view_party_button"), TextStyle.PARTY),
TextStyle.PARTY,
);
this.optionsContainer.add(viewPartyText);
viewPartyText.x -= viewPartyText.displayWidth + 16;
this.viewPartyXPosition = viewPartyText.x - 10;
// Description Window
const titleTextObject = addBBCodeTextObject(0, 0, titleText ?? "", TextStyle.TOOLTIP_TITLE, {
wordWrap: { width: 750 },
align: "center",
lineSpacing: -8,
});
this.descriptionContainer.add(titleTextObject);
titleTextObject.setPosition(72 - titleTextObject.displayWidth / 2, 5.5);
// Rarity of encounter
const index =
mysteryEncounter.encounterTier === MysteryEncounterTier.COMMON
? 0
: mysteryEncounter.encounterTier === MysteryEncounterTier.GREAT
? 1
: mysteryEncounter.encounterTier === MysteryEncounterTier.ULTRA
? 2
: mysteryEncounter.encounterTier === MysteryEncounterTier.ROGUE
? 3
: 4;
const ballType = getPokeballAtlasKey(index);
this.rarityBall.setTexture("pb", ballType);
const descriptionTextObject = addBBCodeTextObject(6, 25, descriptionText ?? "", TextStyle.TOOLTIP_CONTENT, {
wordWrap: { width: 830 },
});
// Sets up the mask that hides the description text to give an illusion of scrolling
const descriptionTextMaskRect = globalScene.make.graphics({});
descriptionTextMaskRect.setScale(6);
descriptionTextMaskRect.fillStyle(0xffffff);
descriptionTextMaskRect.beginPath();
descriptionTextMaskRect.fillRect(6, 53, 206, 57);
const abilityDescriptionTextMask = descriptionTextMaskRect.createGeometryMask();
descriptionTextObject.setMask(abilityDescriptionTextMask);
const descriptionLineCount = Math.floor(descriptionTextObject.displayHeight / 10);
if (this.descriptionScrollTween) {
this.descriptionScrollTween.remove();
this.descriptionScrollTween = undefined;
}
// Animates the description text moving upwards
if (descriptionLineCount > 6) {
this.descriptionScrollTween = globalScene.tweens.add({
targets: descriptionTextObject,
delay: fixedInt(2000),
loop: -1,
hold: fixedInt(2000),
duration: fixedInt((descriptionLineCount - 6) * 2000),
y: `-=${10 * (descriptionLineCount - 6)}`,
});
}
this.descriptionContainer.add(descriptionTextObject);
const queryTextObject = addBBCodeTextObject(0, 0, queryText ?? "", TextStyle.TOOLTIP_CONTENT, {
wordWrap: { width: 830 },
});
this.descriptionContainer.add(queryTextObject);
queryTextObject.setPosition(75 - queryTextObject.displayWidth / 2, 90);
// Slide in description container
if (slideInDescription) {
this.descriptionContainer.x -= 150;
globalScene.tweens.add({
targets: this.descriptionContainer,
x: "+=150",
ease: "Sine.easeInOut",
duration: 1000,
});
}
}
/**
* Updates and displays the tooltip for a given option
* The tooltip will auto wrap and scroll if it is too long
*/
private displayOptionTooltip() {
const cursor = this.getCursor();
// Clear tooltip box
if (this.tooltipContainer.length > 1) {
this.tooltipContainer.removeBetween(1, this.tooltipContainer.length, true);
}
this.tooltipContainer.setVisible(true);
if (isNullOrUndefined(cursor) || cursor > this.optionsContainer.length - 2) {
// Ignore hovers on view party button
// Hide dex progress if visible
this.showHideDexProgress(false);
return;
}
let text: string | null;
const cursorOption = this.encounterOptions[cursor];
const optionDialogue = cursorOption.dialogue!;
if (
!this.optionsMeetsReqs[cursor] &&
(cursorOption.optionMode === MysteryEncounterOptionMode.DISABLED_OR_DEFAULT ||
cursorOption.optionMode === MysteryEncounterOptionMode.DISABLED_OR_SPECIAL) &&
optionDialogue.disabledButtonTooltip
) {
text = getEncounterText(optionDialogue.disabledButtonTooltip, TextStyle.TOOLTIP_CONTENT);
} else {
text = getEncounterText(optionDialogue.buttonTooltip, TextStyle.TOOLTIP_CONTENT);
}
// Auto-color options green/blue for good/bad by looking for (+)/(-)
if (text) {
const primaryStyleString = [...text.match(new RegExp(/\[color=[^\[]*\]\[shadow=[^\[]*\]/i))!][0];
text = text.replace(
/(\(\+\)[^\(\[]*)/gi,
substring =>
"[/color][/shadow]" +
getBBCodeFrag(substring, TextStyle.SUMMARY_GREEN) +
"[/color][/shadow]" +
primaryStyleString,
);
text = text.replace(
/(\(\-\)[^\(\[]*)/gi,
substring =>
"[/color][/shadow]" +
getBBCodeFrag(substring, TextStyle.SUMMARY_BLUE) +
"[/color][/shadow]" +
primaryStyleString,
);
}
if (text) {
const tooltipTextObject = addBBCodeTextObject(6, 7, text, TextStyle.TOOLTIP_CONTENT, {
wordWrap: { width: 600 },
fontSize: "72px",
});
this.tooltipContainer.add(tooltipTextObject);
// Sets up the mask that hides the description text to give an illusion of scrolling
const tooltipTextMaskRect = globalScene.make.graphics({});
tooltipTextMaskRect.setScale(6);
tooltipTextMaskRect.fillStyle(0xffffff);
tooltipTextMaskRect.beginPath();
tooltipTextMaskRect.fillRect(this.tooltipContainer.x, this.tooltipContainer.y + 188.5, 150, 32);
const textMask = tooltipTextMaskRect.createGeometryMask();
tooltipTextObject.setMask(textMask);
const tooltipLineCount = Math.floor(tooltipTextObject.displayHeight / 11.2);
if (this.tooltipScrollTween) {
this.tooltipScrollTween.remove();
this.tooltipScrollTween = undefined;
}
// Animates the tooltip text moving upwards
if (tooltipLineCount > 3) {
this.tooltipScrollTween = globalScene.tweens.add({
targets: tooltipTextObject,
delay: fixedInt(1200),
loop: -1,
hold: fixedInt(1200),
duration: fixedInt((tooltipLineCount - 3) * 1200),
y: `-=${11.2 * (tooltipLineCount - 3)}`,
});
}
}
// Dex progress indicator
if (cursorOption.hasDexProgress && !this.showDexProgress) {
this.showHideDexProgress(true);
} else if (!cursorOption.hasDexProgress) {
this.showHideDexProgress(false);
}
}
override clear(): void {
super.clear();
this.overrideSettings = undefined;
this.optionsContainer.setVisible(false);
this.optionsContainer.removeAll(true);
this.dexProgressContainer.setVisible(false);
this.descriptionContainer.setVisible(false);
this.tooltipContainer.setVisible(false);
// Keeps container background and pokeball
this.descriptionContainer.removeBetween(2, this.descriptionContainer.length, true);
this.getUi().getMessageHandler().clearText();
this.eraseCursor();
}
private eraseCursor(): void {
if (this.cursorObj) {
this.cursorObj.destroy();
}
this.cursorObj = undefined;
}
/**
* Will show or hide the Dex progress icon for an option that has dex progress
* @param show - if true does show, if false does hide
*/
private showHideDexProgress(show: boolean) {
if (show && !this.showDexProgress) {
this.showDexProgress = true;
globalScene.tweens.killTweensOf(this.dexProgressContainer);
globalScene.tweens.add({
targets: this.dexProgressContainer,
y: -63,
ease: "Sine.easeInOut",
duration: 750,
onComplete: () => {
this.dexProgressContainer.on("pointerover", () => {
globalScene.ui.showTooltip("", i18next.t("mysteryEncounterMessages:affects_pokedex"), true);
});
this.dexProgressContainer.on("pointerout", () => {
globalScene.ui.hideTooltip();
});
},
});
} else if (!show && this.showDexProgress) {
this.showDexProgress = false;
globalScene.tweens.killTweensOf(this.dexProgressContainer);
globalScene.tweens.add({
targets: this.dexProgressContainer,
y: -43,
ease: "Sine.easeInOut",
duration: 750,
onComplete: () => {
this.dexProgressContainer.off("pointerover");
this.dexProgressContainer.off("pointerout");
},
});
}
}
}