/* * SPDX-FileCopyrightText: syuilo and misskey-project * SPDX-License-Identifier: AGPL-3.0-only */ import { EventEmitter } from 'eventemitter3'; import { IRouter, Resolved, RouteDef, RouterEvent } from '@/nirax.js'; import type { App, ShallowRef } from 'vue'; /** * {@link Router}による画面遷移を可能とするために{@link mainRouter}をセットアップする。 * また、{@link Router}のインスタンスを作成するためのファクトリも{@link provide}経由で公開する(`routerFactory`というキーで取得可能) */ export function setupRouter(app: App, routerFactory: ((path: string) => IRouter)): void { app.provide('routerFactory', routerFactory); const mainRouter = routerFactory(location.pathname + location.search + location.hash); window.addEventListener('popstate', (event) => { mainRouter.replace(location.pathname + location.search + location.hash, event.state?.key); }); mainRouter.addListener('push', ctx => { window.history.pushState({ key: ctx.key }, '', ctx.path); }); mainRouter.addListener('replace', ctx => { window.history.replaceState({ key: ctx.key }, '', ctx.path); }); mainRouter.init(); setMainRouter(mainRouter); } function getMainRouter(): IRouter { const router = mainRouterHolder; if (!router) { throw new Error('mainRouter is not found.'); } return router; } /** * メインルータを設定する。一度設定すると、それ以降は変更できない。 * {@link setupRouter}から呼び出されることのみを想定している。 */ export function setMainRouter(router: IRouter) { if (mainRouterHolder) { throw new Error('mainRouter is already exists.'); } mainRouterHolder = router; } /** * {@link mainRouter}用のプロキシ実装。 * {@link mainRouter}は起動シーケンスの一部にて初期化されるため、僅かにundefinedになる期間がある。 * その僅かな期間のためだけに型をundefined込みにしたくないのでこのクラスを緩衝材として使用する。 */ class MainRouterProxy implements IRouter { private supplier: () => IRouter; constructor(supplier: () => IRouter) { this.supplier = supplier; } get current(): Resolved { return this.supplier().current; } get currentRef(): ShallowRef { return this.supplier().currentRef; } get currentRoute(): ShallowRef { return this.supplier().currentRoute; } get navHook(): ((path: string, flag?: any) => boolean) | null { return this.supplier().navHook; } set navHook(value) { this.supplier().navHook = value; } getCurrentKey(): string { return this.supplier().getCurrentKey(); } getCurrentPath(): any { return this.supplier().getCurrentPath(); } push(path: string, flag?: any): void { this.supplier().push(path, flag); } replace(path: string, key?: string | null): void { this.supplier().replace(path, key); } resolve(path: string): Resolved | null { return this.supplier().resolve(path); } init(): void { this.supplier().init(); } eventNames(): Array> { return this.supplier().eventNames(); } listeners>( event: T, ): Array> { return this.supplier().listeners(event); } listenerCount( event: EventEmitter.EventNames, ): number { return this.supplier().listenerCount(event); } emit>( event: T, ...args: EventEmitter.EventArgs ): boolean { return this.supplier().emit(event, ...args); } on>( event: T, fn: EventEmitter.EventListener, context?: any, ): this { this.supplier().on(event, fn, context); return this; } addListener>( event: T, fn: EventEmitter.EventListener, context?: any, ): this { this.supplier().addListener(event, fn, context); return this; } once>( event: T, fn: EventEmitter.EventListener, context?: any, ): this { this.supplier().once(event, fn, context); return this; } removeListener>( event: T, fn?: EventEmitter.EventListener, context?: any, once?: boolean, ): this { this.supplier().removeListener(event, fn, context, once); return this; } off>( event: T, fn?: EventEmitter.EventListener, context?: any, once?: boolean, ): this { this.supplier().off(event, fn, context, once); return this; } removeAllListeners( event?: EventEmitter.EventNames, ): this { this.supplier().removeAllListeners(event); return this; } } let mainRouterHolder: IRouter | null = null; export const mainRouter: IRouter = new MainRouterProxy(getMainRouter);