pokerogue/src/ui/message-ui-handler.ts

198 lines
6.2 KiB
TypeScript
Raw Normal View History

2023-03-28 19:54:52 +01:00
import BattleScene from "../battle-scene";
import AwaitableUiHandler from "./awaitable-ui-handler";
import { Mode } from "./ui";
2023-06-05 02:47:43 +01:00
import * as Utils from "../utils";
2023-03-28 19:54:52 +01:00
export default abstract class MessageUiHandler extends AwaitableUiHandler {
protected textTimer: Phaser.Time.TimerEvent;
protected textCallbackTimer: Phaser.Time.TimerEvent;
2023-10-26 21:33:59 +01:00
public pendingPrompt: boolean;
2023-03-28 19:54:52 +01:00
public message: Phaser.GameObjects.Text;
public prompt: Phaser.GameObjects.Sprite;
constructor(scene: BattleScene, mode: Mode) {
super(scene, mode);
this.pendingPrompt = false;
}
2023-04-10 18:54:06 +01:00
showText(text: string, delay?: integer, callback?: Function, callbackDelay?: integer, prompt?: boolean, promptDelay?: integer) {
this.showTextInternal(text, delay, callback, callbackDelay, prompt, promptDelay);
}
showDialogue(text: string, name: string, delay?: integer, callback?: Function, callbackDelay?: integer, prompt?: boolean, promptDelay?: integer) {
this.showTextInternal(text, delay, callback, callbackDelay, prompt, promptDelay);
}
private showTextInternal(text: string, delay: integer, callback: Function, callbackDelay: integer, prompt: boolean, promptDelay: integer) {
if (delay === null || delay === undefined) {
2023-03-28 19:54:52 +01:00
delay = 20;
}
const charVarMap = new Map<integer, string>();
const delayMap = new Map<integer, integer>();
const soundMap = new Map<integer, string>();
const actionPattern = /@(c|d|s)\{(.*?)\}/;
2023-04-10 18:54:06 +01:00
let actionMatch: RegExpExecArray;
while ((actionMatch = actionPattern.exec(text))) {
switch (actionMatch[1]) {
case "c":
charVarMap.set(actionMatch.index, actionMatch[2]);
break;
case "d":
delayMap.set(actionMatch.index, parseInt(actionMatch[2]));
break;
case "s":
soundMap.set(actionMatch.index, actionMatch[2]);
break;
2023-04-10 18:54:06 +01:00
}
text = text.slice(0, actionMatch.index) + text.slice(actionMatch.index + actionMatch[2].length + 4);
}
2024-04-07 04:03:20 +01:00
2024-04-08 00:11:34 +01:00
if (text) {
// Predetermine overflow line breaks to avoid words breaking while displaying
const textWords = text.split(" ");
2024-04-08 00:11:34 +01:00
let lastLineCount = 1;
let newText = "";
for (let w = 0; w < textWords.length; w++) {
2024-04-09 02:10:53 +01:00
const nextWordText = newText ? `${newText} ${textWords[w]}` : textWords[w];
2024-05-24 00:45:04 +01:00
if (textWords[w].includes("\n")) {
2024-04-07 04:03:20 +01:00
newText = nextWordText;
2024-04-08 00:11:34 +01:00
lastLineCount++;
} else {
const lineCount = this.message.runWordWrap(nextWordText).split(/\n/g).length;
if (lineCount > lastLineCount) {
lastLineCount = lineCount;
newText = `${newText}\n${textWords[w]}`;
} else {
2024-04-08 00:11:34 +01:00
newText = nextWordText;
}
2024-04-08 00:11:34 +01:00
}
2024-04-07 04:03:20 +01:00
}
2024-04-08 00:11:34 +01:00
2024-04-09 02:10:53 +01:00
text = newText;
2023-04-10 18:54:06 +01:00
}
2023-03-28 19:54:52 +01:00
if (this.textTimer) {
this.textTimer.remove();
if (this.textCallbackTimer) {
2023-03-28 19:54:52 +01:00
this.textCallbackTimer.callback();
}
}
2023-03-28 19:54:52 +01:00
if (prompt) {
const originalCallback = callback;
2023-04-10 18:54:06 +01:00
callback = () => {
const showPrompt = () => this.showPrompt(originalCallback, callbackDelay);
if (promptDelay) {
2023-04-10 18:54:06 +01:00
this.scene.time.delayedCall(promptDelay, showPrompt);
} else {
2023-04-10 18:54:06 +01:00
showPrompt();
}
2023-04-10 18:54:06 +01:00
};
2023-03-28 19:54:52 +01:00
}
if (delay) {
this.clearText();
if (prompt) {
2023-03-28 19:54:52 +01:00
this.pendingPrompt = true;
}
2023-03-28 19:54:52 +01:00
this.textTimer = this.scene.time.addEvent({
delay: delay,
callback: () => {
2023-04-10 18:54:06 +01:00
const charIndex = text.length - this.textTimer.repeatCount;
const charVar = charVarMap.get(charIndex);
2023-04-10 18:54:06 +01:00
const charSound = soundMap.get(charIndex);
const charDelay = delayMap.get(charIndex);
this.message.setText(text.slice(0, charIndex));
const advance = () => {
if (charVar) {
this.scene.charSprite.setVariant(charVar);
}
if (charSound) {
2023-10-21 13:58:39 +01:00
this.scene.playSound(charSound);
}
2023-04-10 18:54:06 +01:00
if (callback && !this.textTimer.repeatCount) {
if (callbackDelay && !prompt) {
this.textCallbackTimer = this.scene.time.delayedCall(callbackDelay, () => {
if (this.textCallbackTimer) {
this.textCallbackTimer.destroy();
this.textCallbackTimer = null;
}
callback();
});
} else {
2023-03-28 19:54:52 +01:00
callback();
}
2023-04-10 18:54:06 +01:00
}
};
if (charDelay) {
this.textTimer.paused = true;
this.scene.tweens.addCounter({
2023-06-05 02:47:43 +01:00
duration: Utils.getFrameMs(charDelay),
2023-04-10 18:54:06 +01:00
onComplete: () => {
this.textTimer.paused = false;
advance();
}
});
} else {
2023-04-10 18:54:06 +01:00
advance();
}
2023-03-28 19:54:52 +01:00
},
repeat: text.length
});
} else {
this.message.setText(text);
if (prompt) {
2023-03-28 19:54:52 +01:00
this.pendingPrompt = true;
}
if (callback) {
2023-03-28 19:54:52 +01:00
callback();
}
2023-03-28 19:54:52 +01:00
}
}
2023-03-29 05:31:25 +01:00
showPrompt(callback: Function, callbackDelay: integer) {
const wrappedTextLines = this.message.runWordWrap(this.message.text).split(/\n/g);
const textLinesCount = wrappedTextLines.length;
const lastTextLine = wrappedTextLines[wrappedTextLines.length - 1];
const lastLineTest = this.scene.add.text(0, 0, lastTextLine, { font: "96px emerald" });
2023-03-29 05:31:25 +01:00
lastLineTest.setScale(this.message.scale);
const lastLineWidth = lastLineTest.displayWidth;
lastLineTest.destroy();
if (this.prompt) {
this.prompt.setPosition(lastLineWidth + 2, (textLinesCount - 1) * 18 + 2);
this.prompt.play("prompt");
2023-03-29 05:31:25 +01:00
}
this.pendingPrompt = false;
this.awaitingActionInput = true;
this.onActionInput = () => {
if (this.prompt) {
this.prompt.anims.stop();
this.prompt.setVisible(false);
}
if (callback) {
if (callbackDelay) {
this.textCallbackTimer = this.scene.time.delayedCall(callbackDelay, () => {
if (this.textCallbackTimer) {
this.textCallbackTimer.destroy();
this.textCallbackTimer = null;
}
callback();
});
} else {
2023-03-29 05:31:25 +01:00
callback();
}
2023-03-29 05:31:25 +01:00
}
};
}
2023-03-28 19:54:52 +01:00
clearText() {
this.message.setText("");
2023-03-28 19:54:52 +01:00
this.pendingPrompt = false;
}
clear() {
super.clear();
}
}