import Modal from "components/Modal.vue"; import { CoercableComponent, jsx, JSXFunction, OptionsFunc, Replace, setDefault, StyleValue } from "features/feature"; import { createNanoEvents, Emitter } from "nanoevents"; import { Computable, GetComputableType, GetComputableTypeWithDefault, processComputable, ProcessedComputable } from "util/computed"; import { createLazyProxy } from "util/proxies"; import { InjectionKey, Ref, ref, shallowReactive, unref } from "vue"; import { globalBus } from "./events"; import { Persistent, persistent } from "./persistence"; import player from "./player"; export interface FeatureNode { rect: DOMRect; observer: MutationObserver; element: HTMLElement; } export const RegisterNodeInjectionKey: InjectionKey<(id: string, element: HTMLElement) => void> = Symbol("RegisterNode"); export const UnregisterNodeInjectionKey: InjectionKey<(id: string) => void> = Symbol("UnregisterNode"); export const NodesInjectionKey: InjectionKey>> = Symbol("Nodes"); export const BoundsInjectionKey: InjectionKey> = Symbol("Bounds"); export interface LayerEvents { // Generation preUpdate: (diff: number) => void; // Actions (e.g. automation) update: (diff: number) => void; // Effects (e.g. milestones) postUpdate: (diff: number) => void; } export const layers: Record | undefined> = shallowReactive({}); window.layers = layers; declare module "@vue/runtime-dom" { interface CSSProperties { "--layer-color"?: string; } } export interface Position { x: number; y: number; } export interface LayerOptions { color?: Computable; display: Computable; classes?: Computable>; style?: Computable; name?: Computable; minimizable?: Computable; forceHideGoBack?: Computable; minWidth?: Computable; } export interface BaseLayer { id: string; minimized: Persistent; emitter: Emitter; on: OmitThisParameter["on"]>; emit: (event: K, ...args: Parameters) => void; nodes: Ref>; } export type Layer = Replace< T & BaseLayer, { color: GetComputableType; display: GetComputableType; classes: GetComputableType; style: GetComputableType; name: GetComputableTypeWithDefault; minWidth: GetComputableTypeWithDefault; minimizable: GetComputableTypeWithDefault; forceHideGoBack: GetComputableType; } >; export type GenericLayer = Replace< Layer, { name: ProcessedComputable; minWidth: ProcessedComputable; minimizable: ProcessedComputable; } >; export const persistentRefs: Record> = {}; export const addingLayers: string[] = []; export function createLayer( id: string, optionsFunc: OptionsFunc ): Layer { return createLazyProxy(() => { const layer = {} as T & Partial; const emitter = (layer.emitter = createNanoEvents()); layer.on = emitter.on.bind(emitter); layer.emit = emitter.emit.bind(emitter); layer.nodes = ref({}); layer.id = id; addingLayers.push(id); persistentRefs[id] = new Set(); layer.minimized = persistent(false); Object.assign(layer, optionsFunc.call(layer as BaseLayer)); if ( addingLayers[addingLayers.length - 1] == null || addingLayers[addingLayers.length - 1] !== id ) { throw `Adding layers stack in invalid state. This should not happen\nStack: ${addingLayers}\nTrying to pop ${layer.id}`; } addingLayers.pop(); processComputable(layer as T, "color"); processComputable(layer as T, "display"); processComputable(layer as T, "name"); setDefault(layer, "name", layer.id); processComputable(layer as T, "minWidth"); setDefault(layer, "minWidth", 600); processComputable(layer as T, "minimizable"); setDefault(layer, "minimizable", true); return layer as unknown as Layer; }); } export function addLayer( layer: GenericLayer, player: { layers?: Record> } ): void { console.info("Adding layer", layer.id); if (layers[layer.id]) { console.error( "Attempted to add layer with same ID as existing layer", layer.id, layers[layer.id] ); return; } setDefault(player, "layers", {}); if (player.layers[layer.id] == null) { player.layers[layer.id] = {}; } layers[layer.id] = layer; globalBus.emit("addLayer", layer, player.layers[layer.id]); } export function getLayer(layerID: string): T { return layers[layerID] as T; } export function removeLayer(layer: GenericLayer): void { console.info("Removing layer", layer.id); globalBus.emit("removeLayer", layer); layers[layer.id] = undefined; } export function reloadLayer(layer: GenericLayer): void { removeLayer(layer); // Re-create layer addLayer(layer, player); } export function setupLayerModal(layer: GenericLayer): { openModal: VoidFunction; modal: JSXFunction; } { const showModal = ref(false); return { openModal: () => (showModal.value = true), modal: jsx(() => ( (showModal.value = value)} v-slots={{ header: () =>

{unref(layer.name)}

, body: unref(layer.display) }} /> )) }; } globalBus.on("update", function updateLayers(diff) { Object.values(layers).forEach(layer => { layer?.emit("preUpdate", diff); }); Object.values(layers).forEach(layer => { layer?.emit("update", diff); }); Object.values(layers).forEach(layer => { layer?.emit("postUpdate", diff); }); });