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;
|
|
|
|
protected pendingPrompt: boolean;
|
|
|
|
|
|
|
|
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) {
|
2023-10-18 23:01:15 +01:00
|
|
|
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) {
|
2023-03-28 19:54:52 +01:00
|
|
|
if (delay === null || delay === undefined)
|
|
|
|
delay = 20;
|
2023-04-10 18:54:06 +01:00
|
|
|
let delayMap = new Map<integer, integer>();
|
|
|
|
let soundMap = new Map<integer, string>();
|
|
|
|
const actionPattern = /@(d|s)\{(.*?)\}/;
|
|
|
|
let actionMatch: RegExpExecArray;
|
|
|
|
while ((actionMatch = actionPattern.exec(text))) {
|
|
|
|
switch (actionMatch[1]) {
|
|
|
|
case 'd':
|
|
|
|
delayMap.set(actionMatch.index, parseInt(actionMatch[2]));
|
|
|
|
break;
|
|
|
|
case 's':
|
|
|
|
soundMap.set(actionMatch.index, actionMatch[2]);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
text = text.slice(0, actionMatch.index) + text.slice(actionMatch.index + actionMatch[2].length + 4);
|
|
|
|
}
|
2023-03-28 19:54:52 +01:00
|
|
|
if (this.textTimer) {
|
|
|
|
this.textTimer.remove();
|
|
|
|
if (this.textCallbackTimer)
|
|
|
|
this.textCallbackTimer.callback();
|
|
|
|
};
|
|
|
|
if (prompt) {
|
|
|
|
const originalCallback = callback;
|
2023-04-10 18:54:06 +01:00
|
|
|
callback = () => {
|
|
|
|
const showPrompt = () => this.showPrompt(originalCallback, callbackDelay);
|
|
|
|
if (promptDelay)
|
|
|
|
this.scene.time.delayedCall(promptDelay, showPrompt);
|
|
|
|
else
|
|
|
|
showPrompt();
|
|
|
|
};
|
2023-03-28 19:54:52 +01:00
|
|
|
}
|
|
|
|
if (delay) {
|
|
|
|
this.clearText();
|
|
|
|
if (prompt)
|
|
|
|
this.pendingPrompt = true;
|
|
|
|
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 charSound = soundMap.get(charIndex);
|
|
|
|
const charDelay = delayMap.get(charIndex);
|
|
|
|
this.message.setText(text.slice(0, charIndex));
|
|
|
|
const advance = () => {
|
|
|
|
if (charSound)
|
|
|
|
this.scene.sound.play(charSound);
|
|
|
|
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
|
|
|
|
advance();
|
2023-03-28 19:54:52 +01:00
|
|
|
},
|
|
|
|
repeat: text.length
|
|
|
|
});
|
|
|
|
} else {
|
|
|
|
this.message.setText(text);
|
|
|
|
if (prompt)
|
|
|
|
this.pendingPrompt = true;
|
|
|
|
if (callback)
|
|
|
|
callback();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
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' });
|
|
|
|
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');
|
|
|
|
}
|
|
|
|
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
|
|
|
|
callback();
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2023-03-28 19:54:52 +01:00
|
|
|
clearText() {
|
|
|
|
this.message.setText('');
|
|
|
|
this.pendingPrompt = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
clear() {
|
|
|
|
super.clear();
|
|
|
|
}
|
|
|
|
}
|