2024-02-13 18:42:11 -05:00
|
|
|
import BattleScene from "./battle-scene";
|
2024-02-14 10:44:55 -05:00
|
|
|
import AwaitableUiHandler from "./ui/awaitable-ui-handler";
|
2024-09-16 21:19:34 +02:00
|
|
|
import UiHandler from "./ui/ui-handler";
|
2024-02-14 10:44:55 -05:00
|
|
|
import { Mode } from "./ui/ui";
|
2024-06-17 17:05:33 -04:00
|
|
|
import i18next from "i18next";
|
2024-09-16 21:19:34 +02:00
|
|
|
import Overrides from "#app/overrides";
|
2024-02-13 18:42:11 -05:00
|
|
|
|
|
|
|
export enum Tutorial {
|
|
|
|
Intro = "INTRO",
|
2024-02-14 10:44:55 -05:00
|
|
|
Access_Menu = "ACCESS_MENU",
|
2024-02-13 18:42:11 -05:00
|
|
|
Menu = "MENU",
|
|
|
|
Starter_Select = "STARTER_SELECT",
|
2024-04-01 22:50:00 -04:00
|
|
|
Pokerus = "POKERUS",
|
2024-04-30 23:02:16 -04:00
|
|
|
Stat_Change = "STAT_CHANGE",
|
2024-02-13 18:42:11 -05:00
|
|
|
Select_Item = "SELECT_ITEM",
|
2024-02-14 10:44:55 -05:00
|
|
|
Egg_Gacha = "EGG_GACHA"
|
2024-02-13 18:42:11 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
const tutorialHandlers = {
|
|
|
|
[Tutorial.Intro]: (scene: BattleScene) => {
|
|
|
|
return new Promise<void>(resolve => {
|
2024-04-29 18:33:18 +02:00
|
|
|
scene.ui.showText(i18next.t("tutorial:intro"), null, () => resolve(), null, true);
|
2024-02-13 18:42:11 -05:00
|
|
|
});
|
|
|
|
},
|
2024-02-14 10:44:55 -05:00
|
|
|
[Tutorial.Access_Menu]: (scene: BattleScene) => {
|
2024-02-13 18:42:11 -05:00
|
|
|
return new Promise<void>(resolve => {
|
2024-05-23 17:03:10 +02:00
|
|
|
if (scene.enableTouchControls) {
|
2024-02-13 18:42:11 -05:00
|
|
|
return resolve();
|
2024-05-23 17:03:10 +02:00
|
|
|
}
|
2024-04-29 18:33:18 +02:00
|
|
|
scene.showFieldOverlay(1000).then(() => scene.ui.showText(i18next.t("tutorial:accessMenu"), null, () => scene.hideFieldOverlay(1000).then(() => resolve()), null, true));
|
2024-02-14 10:44:55 -05:00
|
|
|
});
|
|
|
|
},
|
|
|
|
[Tutorial.Menu]: (scene: BattleScene) => {
|
|
|
|
return new Promise<void>(resolve => {
|
|
|
|
scene.gameData.saveTutorialFlag(Tutorial.Access_Menu, true);
|
2024-05-23 17:03:10 +02:00
|
|
|
scene.ui.showText(i18next.t("tutorial:menu"), null, () => scene.ui.showText("", null, () => resolve()), null, true);
|
2024-02-13 18:42:11 -05:00
|
|
|
});
|
|
|
|
},
|
|
|
|
[Tutorial.Starter_Select]: (scene: BattleScene) => {
|
|
|
|
return new Promise<void>(resolve => {
|
2024-05-23 17:03:10 +02:00
|
|
|
scene.ui.showText(i18next.t("tutorial:starterSelect"), null, () => scene.ui.showText("", null, () => resolve()), null, true);
|
2024-02-14 10:44:55 -05:00
|
|
|
});
|
|
|
|
},
|
2024-09-16 21:19:34 +02:00
|
|
|
[Tutorial.Pokerus]: (scene: BattleScene) => {
|
2024-04-01 22:50:00 -04:00
|
|
|
return new Promise<void>(resolve => {
|
2024-05-23 17:03:10 +02:00
|
|
|
scene.ui.showText(i18next.t("tutorial:pokerus"), null, () => scene.ui.showText("", null, () => resolve()), null, true);
|
2024-04-01 22:50:00 -04:00
|
|
|
});
|
|
|
|
},
|
2024-04-30 23:02:16 -04:00
|
|
|
[Tutorial.Stat_Change]: (scene: BattleScene) => {
|
|
|
|
return new Promise<void>(resolve => {
|
2024-05-23 17:03:10 +02:00
|
|
|
scene.showFieldOverlay(1000).then(() => scene.ui.showText(i18next.t("tutorial:statChange"), null, () => scene.ui.showText("", null, () => scene.hideFieldOverlay(1000).then(() => resolve())), null, true));
|
2024-04-30 23:02:16 -04:00
|
|
|
});
|
|
|
|
},
|
2024-02-14 10:44:55 -05:00
|
|
|
[Tutorial.Select_Item]: (scene: BattleScene) => {
|
|
|
|
return new Promise<void>(resolve => {
|
|
|
|
scene.ui.setModeWithoutClear(Mode.MESSAGE).then(() => {
|
2024-05-23 17:03:10 +02:00
|
|
|
scene.ui.showText(i18next.t("tutorial:selectItem"), null, () => scene.ui.showText("", null, () => scene.ui.setModeWithoutClear(Mode.MODIFIER_SELECT).then(() => resolve())), null, true);
|
2024-02-14 10:44:55 -05:00
|
|
|
});
|
|
|
|
});
|
|
|
|
},
|
|
|
|
[Tutorial.Egg_Gacha]: (scene: BattleScene) => {
|
|
|
|
return new Promise<void>(resolve => {
|
2024-05-23 17:03:10 +02:00
|
|
|
scene.ui.showText(i18next.t("tutorial:eggGacha"), null, () => scene.ui.showText("", null, () => resolve()), null, true);
|
2024-02-13 18:42:11 -05:00
|
|
|
});
|
|
|
|
},
|
|
|
|
};
|
|
|
|
|
2024-09-16 21:19:34 +02:00
|
|
|
/**
|
|
|
|
* Run through the specified tutorial if it hasn't been seen before and mark it as seen once done
|
|
|
|
* This will show a tutorial overlay if defined in the current {@linkcode AwaitableUiHandler}
|
|
|
|
* The main menu will also get disabled while the tutorial is running
|
|
|
|
* @param scene the current {@linkcode BattleScene}
|
|
|
|
* @param tutorial the {@linkcode Tutorial} to play
|
|
|
|
* @returns a promise with result `true` if the tutorial was run and finished, `false` otherwise
|
|
|
|
*/
|
|
|
|
export async function handleTutorial(scene: BattleScene, tutorial: Tutorial): Promise<boolean> {
|
2024-09-26 04:39:59 -04:00
|
|
|
if (!scene.enableTutorials && !Overrides.BYPASS_TUTORIAL_SKIP_OVERRIDE) {
|
2024-09-16 21:19:34 +02:00
|
|
|
return false;
|
|
|
|
}
|
2024-02-13 18:42:11 -05:00
|
|
|
|
2024-09-26 04:39:59 -04:00
|
|
|
if (scene.gameData.getTutorialFlags()[tutorial] && !Overrides.BYPASS_TUTORIAL_SKIP_OVERRIDE) {
|
2024-09-16 21:19:34 +02:00
|
|
|
return false;
|
|
|
|
}
|
2024-02-13 18:42:11 -05:00
|
|
|
|
2024-09-16 21:19:34 +02:00
|
|
|
const handler = scene.ui.getHandler();
|
|
|
|
const isMenuDisabled = scene.disableMenu;
|
|
|
|
|
|
|
|
// starting tutorial, disable menu
|
|
|
|
scene.disableMenu = true;
|
|
|
|
if (handler instanceof AwaitableUiHandler) {
|
|
|
|
handler.tutorialActive = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
await showTutorialOverlay(scene, handler);
|
|
|
|
await tutorialHandlers[tutorial](scene);
|
|
|
|
await hideTutorialOverlay(scene, handler);
|
|
|
|
|
|
|
|
// tutorial finished and overlay gone, re-enable menu, save tutorial as seen
|
|
|
|
scene.disableMenu = isMenuDisabled;
|
|
|
|
scene.gameData.saveTutorialFlag(tutorial, true);
|
|
|
|
if (handler instanceof AwaitableUiHandler) {
|
|
|
|
handler.tutorialActive = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Show the tutorial overlay if there is one
|
|
|
|
* @param scene the current BattleScene
|
|
|
|
* @param handler the current UiHandler
|
|
|
|
* @returns `true` once the overlay has finished appearing, or if there is no overlay
|
|
|
|
*/
|
|
|
|
async function showTutorialOverlay(scene: BattleScene, handler: UiHandler) {
|
|
|
|
if (handler instanceof AwaitableUiHandler && handler.tutorialOverlay) {
|
|
|
|
scene.tweens.add({
|
|
|
|
targets: handler.tutorialOverlay,
|
|
|
|
alpha: 0.5,
|
|
|
|
duration: 750,
|
|
|
|
ease: "Sine.easeOut",
|
|
|
|
onComplete: () => {
|
|
|
|
return true;
|
2024-05-23 17:03:10 +02:00
|
|
|
}
|
2024-02-13 18:42:11 -05:00
|
|
|
});
|
2024-09-16 21:19:34 +02:00
|
|
|
} else {
|
|
|
|
return true;
|
|
|
|
}
|
2024-04-29 18:33:18 +02:00
|
|
|
}
|
2024-09-16 21:19:34 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Hide the tutorial overlay if there is one
|
|
|
|
* @param scene the current BattleScene
|
|
|
|
* @param handler the current UiHandler
|
|
|
|
* @returns `true` once the overlay has finished disappearing, or if there is no overlay
|
|
|
|
*/
|
|
|
|
async function hideTutorialOverlay(scene: BattleScene, handler: UiHandler) {
|
|
|
|
if (handler instanceof AwaitableUiHandler && handler.tutorialOverlay) {
|
|
|
|
scene.tweens.add({
|
|
|
|
targets: handler.tutorialOverlay,
|
|
|
|
alpha: 0,
|
|
|
|
duration: 500,
|
|
|
|
ease: "Sine.easeOut",
|
|
|
|
onComplete: () => {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
} else {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|