Add outage handling
parent
5f0815bc3c
commit
eb5bdb07a8
|
@ -8,8 +8,8 @@ export interface UserInfo {
|
||||||
|
|
||||||
export let loggedInUser: UserInfo = null;
|
export let loggedInUser: UserInfo = null;
|
||||||
|
|
||||||
export function updateUserInfo(): Promise<boolean> {
|
export function updateUserInfo(): Promise<[boolean, integer]> {
|
||||||
return new Promise<boolean>(resolve => {
|
return new Promise<[boolean, integer]>(resolve => {
|
||||||
if (bypassLogin) {
|
if (bypassLogin) {
|
||||||
let lastSessionSlot = -1;
|
let lastSessionSlot = -1;
|
||||||
for (let s = 0; s < 2; s++) {
|
for (let s = 0; s < 2; s++) {
|
||||||
|
@ -19,18 +19,21 @@ export function updateUserInfo(): Promise<boolean> {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
loggedInUser = { username: 'Guest', lastSessionSlot: lastSessionSlot };
|
loggedInUser = { username: 'Guest', lastSessionSlot: lastSessionSlot };
|
||||||
return resolve(true);
|
return resolve([ true, 200 ]);
|
||||||
}
|
}
|
||||||
Utils.apiFetch('account/info').then(response => {
|
Utils.apiFetch('account/info').then(response => {
|
||||||
|
console.log(response.status);
|
||||||
if (!response.ok) {
|
if (!response.ok) {
|
||||||
loggedInUser = null;
|
resolve([ false, response.status ]);
|
||||||
resolve(false);
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
return response.json();
|
return response.json();
|
||||||
}).then(jsonResponse => {
|
}).then(jsonResponse => {
|
||||||
loggedInUser = jsonResponse;
|
loggedInUser = jsonResponse;
|
||||||
resolve(true);
|
resolve([ true, 200 ]);
|
||||||
|
}).catch(err => {
|
||||||
|
console.error(err);
|
||||||
|
resolve([ false, 500 ]);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
}
|
}
|
|
@ -69,39 +69,48 @@ export class LoginPhase extends Phase {
|
||||||
start(): void {
|
start(): void {
|
||||||
super.start();
|
super.start();
|
||||||
|
|
||||||
this.scene.ui.setMode(Mode.LOADING, { buttonActions: [] });
|
const hasSession = !!Utils.getCookie(Utils.sessionIdKey);
|
||||||
Utils.executeIf(bypassLogin || !!Utils.getCookie(Utils.sessionIdKey), updateUserInfo).then(success => {
|
|
||||||
if (!success) {
|
|
||||||
if (this.showText)
|
|
||||||
this.scene.ui.showText('Log in or create an account to start. No email required!');
|
|
||||||
|
|
||||||
this.scene.playSound('menu_open');
|
|
||||||
|
|
||||||
const loadData = () => {
|
this.scene.ui.setMode(Mode.LOADING, { buttonActions: [] });
|
||||||
updateUserInfo().then(() => this.scene.gameData.loadSystem().then(() => this.end()));
|
Utils.executeIf(bypassLogin || hasSession, updateUserInfo).then(response => {
|
||||||
};
|
const success = response ? response[0] : false;
|
||||||
|
const statusCode = response ? response[1] : null;
|
||||||
|
if (!success) {
|
||||||
|
if (!statusCode || statusCode === 400) {
|
||||||
|
if (this.showText)
|
||||||
|
this.scene.ui.showText('Log in or create an account to start. No email required!');
|
||||||
|
|
||||||
this.scene.ui.setMode(Mode.LOGIN_FORM, {
|
this.scene.playSound('menu_open');
|
||||||
buttonActions: [
|
|
||||||
() => {
|
const loadData = () => {
|
||||||
this.scene.ui.playSelect();
|
updateUserInfo().then(() => this.scene.gameData.loadSystem().then(() => this.end()));
|
||||||
loadData();
|
};
|
||||||
}, () => {
|
|
||||||
this.scene.playSound('menu_open');
|
this.scene.ui.setMode(Mode.LOGIN_FORM, {
|
||||||
this.scene.ui.setMode(Mode.REGISTRATION_FORM, {
|
buttonActions: [
|
||||||
buttonActions: [
|
() => {
|
||||||
() => {
|
this.scene.ui.playSelect();
|
||||||
this.scene.ui.playSelect();
|
loadData();
|
||||||
updateUserInfo().then(() => this.end());
|
}, () => {
|
||||||
}, () => {
|
this.scene.playSound('menu_open');
|
||||||
this.scene.unshiftPhase(new LoginPhase(this.scene, false));
|
this.scene.ui.setMode(Mode.REGISTRATION_FORM, {
|
||||||
this.end();
|
buttonActions: [
|
||||||
}
|
() => {
|
||||||
]
|
this.scene.ui.playSelect();
|
||||||
});
|
updateUserInfo().then(() => this.end());
|
||||||
}
|
}, () => {
|
||||||
]
|
this.scene.unshiftPhase(new LoginPhase(this.scene, false));
|
||||||
});
|
this.end();
|
||||||
|
}
|
||||||
|
]
|
||||||
|
});
|
||||||
|
}
|
||||||
|
]
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
this.scene.unshiftPhase(new UnavailablePhase(this.scene));
|
||||||
|
super.end();
|
||||||
|
}
|
||||||
return null;
|
return null;
|
||||||
} else {
|
} else {
|
||||||
this.scene.gameData.loadSystem().then(success => {
|
this.scene.gameData.loadSystem().then(success => {
|
||||||
|
@ -336,6 +345,19 @@ export class TitlePhase extends Phase {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export class UnavailablePhase extends Phase {
|
||||||
|
constructor(scene: BattleScene) {
|
||||||
|
super(scene);
|
||||||
|
}
|
||||||
|
|
||||||
|
start(): void {
|
||||||
|
this.scene.ui.setMode(Mode.UNAVAILABLE, () => {
|
||||||
|
this.scene.unshiftPhase(new LoginPhase(this.scene, true));
|
||||||
|
this.end();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
export class SelectGenderPhase extends Phase {
|
export class SelectGenderPhase extends Phase {
|
||||||
constructor(scene: BattleScene) {
|
constructor(scene: BattleScene) {
|
||||||
super(scene);
|
super(scene);
|
||||||
|
|
|
@ -224,8 +224,8 @@ export class GameData {
|
||||||
public saveSystem(): Promise<boolean> {
|
public saveSystem(): Promise<boolean> {
|
||||||
return new Promise<boolean>(resolve => {
|
return new Promise<boolean>(resolve => {
|
||||||
this.scene.ui.savingIcon.show();
|
this.scene.ui.savingIcon.show();
|
||||||
updateUserInfo().then((success: boolean) => {
|
updateUserInfo().then(response => {
|
||||||
if (!success) {
|
if (!response[0]) {
|
||||||
this.scene.ui.savingIcon.hide();
|
this.scene.ui.savingIcon.hide();
|
||||||
return resolve(false);
|
return resolve(false);
|
||||||
}
|
}
|
||||||
|
|
10
src/ui/ui.ts
10
src/ui/ui.ts
|
@ -33,6 +33,7 @@ import AwaitableUiHandler from './awaitable-ui-handler';
|
||||||
import SaveSlotSelectUiHandler from './save-slot-select-ui-handler';
|
import SaveSlotSelectUiHandler from './save-slot-select-ui-handler';
|
||||||
import TitleUiHandler from './title-ui-handler';
|
import TitleUiHandler from './title-ui-handler';
|
||||||
import SavingIconHandler from './saving-icon-handler';
|
import SavingIconHandler from './saving-icon-handler';
|
||||||
|
import UnavailableModalUiHandler from './unavailable-modal-ui-handler';
|
||||||
|
|
||||||
export enum Mode {
|
export enum Mode {
|
||||||
MESSAGE,
|
MESSAGE,
|
||||||
|
@ -61,7 +62,8 @@ export enum Mode {
|
||||||
EGG_GACHA,
|
EGG_GACHA,
|
||||||
LOGIN_FORM,
|
LOGIN_FORM,
|
||||||
REGISTRATION_FORM,
|
REGISTRATION_FORM,
|
||||||
LOADING
|
LOADING,
|
||||||
|
UNAVAILABLE
|
||||||
};
|
};
|
||||||
|
|
||||||
const transitionModes = [
|
const transitionModes = [
|
||||||
|
@ -87,7 +89,8 @@ const noTransitionModes = [
|
||||||
Mode.VOUCHERS,
|
Mode.VOUCHERS,
|
||||||
Mode.LOGIN_FORM,
|
Mode.LOGIN_FORM,
|
||||||
Mode.REGISTRATION_FORM,
|
Mode.REGISTRATION_FORM,
|
||||||
Mode.LOADING
|
Mode.LOADING,
|
||||||
|
Mode.UNAVAILABLE
|
||||||
];
|
];
|
||||||
|
|
||||||
export default class UI extends Phaser.GameObjects.Container {
|
export default class UI extends Phaser.GameObjects.Container {
|
||||||
|
@ -137,7 +140,8 @@ export default class UI extends Phaser.GameObjects.Container {
|
||||||
new EggGachaUiHandler(scene),
|
new EggGachaUiHandler(scene),
|
||||||
new LoginFormUiHandler(scene),
|
new LoginFormUiHandler(scene),
|
||||||
new RegistrationFormUiHandler(scene),
|
new RegistrationFormUiHandler(scene),
|
||||||
new LoadingModalUiHandler(scene)
|
new LoadingModalUiHandler(scene),
|
||||||
|
new UnavailableModalUiHandler(scene)
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -0,0 +1,68 @@
|
||||||
|
import BattleScene from "../battle-scene";
|
||||||
|
import { ModalConfig, ModalUiHandler } from "./modal-ui-handler";
|
||||||
|
import { addTextObject, TextStyle } from "./text";
|
||||||
|
import { Mode } from "./ui";
|
||||||
|
import { updateUserInfo } from "#app/account";
|
||||||
|
|
||||||
|
export default class UnavailableModalUiHandler extends ModalUiHandler {
|
||||||
|
private reconnectTimer: number;
|
||||||
|
private reconnectCallback: () => void;
|
||||||
|
|
||||||
|
constructor(scene: BattleScene, mode?: Mode) {
|
||||||
|
super(scene, mode);
|
||||||
|
}
|
||||||
|
|
||||||
|
getModalTitle(): string {
|
||||||
|
return '';
|
||||||
|
}
|
||||||
|
|
||||||
|
getWidth(): number {
|
||||||
|
return 160;
|
||||||
|
}
|
||||||
|
|
||||||
|
getHeight(): number {
|
||||||
|
return 64;
|
||||||
|
}
|
||||||
|
|
||||||
|
getMargin(): [number, number, number, number] {
|
||||||
|
return [ 0, 0, 48, 0 ];
|
||||||
|
}
|
||||||
|
|
||||||
|
getButtonLabels(): string[] {
|
||||||
|
return [ ];
|
||||||
|
}
|
||||||
|
|
||||||
|
setup(): void {
|
||||||
|
super.setup();
|
||||||
|
|
||||||
|
const label = addTextObject(this.scene, this.getWidth() / 2, this.getHeight() / 2, 'Oops! There was an issue contacting the server.\n\nYou may leave this window open,\nthe game will automatically reconnect.', TextStyle.WINDOW, { fontSize: '48px', align: 'center' });
|
||||||
|
label.setOrigin(0.5, 0.5);
|
||||||
|
|
||||||
|
this.modalContainer.add(label);
|
||||||
|
}
|
||||||
|
|
||||||
|
show(args: any[]): boolean {
|
||||||
|
if (args.length >= 1 && args[0] instanceof Function) {
|
||||||
|
const config: ModalConfig = {
|
||||||
|
buttonActions: []
|
||||||
|
};
|
||||||
|
|
||||||
|
this.reconnectCallback = args[0];
|
||||||
|
|
||||||
|
this.reconnectTimer = setInterval(() => {
|
||||||
|
updateUserInfo().then(response => {
|
||||||
|
if (response[0] || [200, 400].includes(response[1])) {
|
||||||
|
clearInterval(this.reconnectTimer);
|
||||||
|
this.reconnectTimer = null;
|
||||||
|
this.scene.playSound('pb_bounce_1');
|
||||||
|
this.reconnectCallback();
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}, 5000);
|
||||||
|
|
||||||
|
return super.show([ config ]);
|
||||||
|
}
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in New Issue