import { action, computed, makeObservable, observable, runInAction, } from "mobx"; import type { Client, API } from "revolt.js"; import { ulid } from "ulid"; import { determineLink } from "../../lib/links"; import { getApplicationState } from "../../mobx/State"; import { history } from "../../context/history"; import { __thisIsAHack } from "../../context/intermediate/Intermediate"; // import { determineLink } from "../../lib/links"; import Changelog from "./components/Changelog"; import Clipboard from "./components/Clipboard"; import Error from "./components/Error"; import LinkWarning from "./components/LinkWarning"; import MFAEnableTOTP from "./components/MFAEnableTOTP"; import MFAFlow from "./components/MFAFlow"; import MFARecovery from "./components/MFARecovery"; import ModifyAccount from "./components/ModifyAccount"; import OutOfDate from "./components/OutOfDate"; import PendingFriendRequests from "./components/PendingFriendRequests"; import ServerIdentity from "./components/ServerIdentity"; import ShowToken from "./components/ShowToken"; import SignOutSessions from "./components/SignOutSessions"; import SignedOut from "./components/SignedOut"; 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, pop: action, remove: action, rendered: computed, isVisible: computed, }); } /** * Display a new modal on the stack * @param modal Modal data */ push(modal: T) { this.stack = [ ...this.stack, { ...modal, key: ulid(), }, ]; } /** * Remove the top modal from the screen * @param signal What action to trigger */ pop(signal: "close" | "confirm" | "force") { this.stack = this.stack.map((entry, index) => index === this.stack.length - 1 ? { ...entry, signal } : entry, ); } /** * 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 ( // ESLint does not understand spread operator // eslint-disable-next-line this.remove(modal.key!)} /> ); })} ); } /** * Whether a modal is currently visible */ get isVisible() { return this.stack.length > 0; } } /** * 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, }), ), ); } /** * Open TOTP secret modal * @param client Client */ mfaEnableTOTP(secret: string, identifier: string) { return runInAction( () => new Promise((callback: (value?: string) => void) => this.push({ type: "mfa_enable_totp", identifier, secret, callback, }), ), ); } /** * Write text to the clipboard * @param text Text to write */ writeText(text: string) { if (navigator.clipboard) { navigator.clipboard.writeText(text); } else { this.push({ type: "clipboard", text, }); } } openLink(href?: string, trusted?: boolean) { const link = determineLink(href); const settings = getApplicationState().settings; switch (link.type) { case "profile": { __thisIsAHack({ id: "profile", user_id: link.id }); break; } case "navigate": { history.push(link.path); break; } case "external": { if ( !trusted && !settings.security.isTrustedOrigin(link.url.hostname) ) { modalController.push({ type: "link_warning", link: link.href, callback: () => this.openLink(href, true) as true, }); } else { window.open(link.href, "_blank", "noreferrer"); } } } return true; } } export const modalController = new ModalControllerExtended({ changelog: Changelog, clipboard: Clipboard, error: Error, link_warning: LinkWarning, mfa_flow: MFAFlow, mfa_recovery: MFARecovery, mfa_enable_totp: MFAEnableTOTP, modify_account: ModifyAccount, out_of_date: OutOfDate, pending_friend_requests: PendingFriendRequests, server_identity: ServerIdentity, show_token: ShowToken, signed_out: SignedOut, sign_out_sessions: SignOutSessions, });