import { action, computed, makeObservable, observable, runInAction, } from "mobx"; import type { Client, API } from "revolt.js"; import { ulid } from "ulid"; import MFAFlow from "./components/MFAFlow"; import MFARecovery from "./components/MFARecovery"; import Test from "./components/Test"; import { Modal } from "./types"; type Components = Record>; /** * Handles layering and displaying modals to the user. */ class ModalController { stack: T[] = []; components: Components; constructor(components: Components) { this.components = components; makeObservable(this, { stack: observable, push: action, remove: action, rendered: computed, }); } /** * Display a new modal on the stack * @param modal Modal data */ push(modal: T) { this.stack = [ ...this.stack, { ...modal, key: ulid(), }, ]; } /** * Remove the keyed modal from the stack */ remove(key: string) { this.stack = this.stack.filter((x) => x.key !== key); } /** * Render modals */ get rendered() { return ( <> {this.stack.map((modal) => { const Component = this.components[modal.type]; return ( this.remove(modal.key!)} /> ); })} ); } } /** * Modal controller with additional helpers. */ class ModalControllerExtended extends ModalController { /** * Perform MFA flow * @param client Client */ mfaFlow(client: Client) { return runInAction( () => new Promise((callback: (ticket: API.MFATicket) => void) => this.push({ type: "mfa_flow", state: "known", client, callback, }), ), ); } } export const modalController = new ModalControllerExtended({ mfa_flow: MFAFlow, mfa_recovery: MFARecovery, test: Test, });