Profectus-Demo/src/game/layers.tsx

215 lines
6.4 KiB
TypeScript
Raw Normal View History

2022-03-04 03:39:48 +00:00
import Modal from "components/Modal.vue";
import {
CoercableComponent,
jsx,
JSXFunction,
Replace,
setDefault,
StyleValue
2022-03-04 03:39:48 +00:00
} from "features/feature";
import { createNanoEvents, Emitter } from "nanoevents";
import {
2022-01-14 04:25:47 +00:00
Computable,
GetComputableType,
GetComputableTypeWithDefault,
processComputable,
ProcessedComputable
2022-03-04 03:39:48 +00:00
} from "util/computed";
import { createLazyProxy } from "util/proxies";
import { InjectionKey, Ref, ref, shallowReactive, unref } from "vue";
2022-01-25 04:23:30 +00:00
import { globalBus } from "./events";
import { Persistent, persistent } from "./persistence";
2022-01-14 04:25:47 +00:00
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<Ref<Record<string, FeatureNode | undefined>>> =
Symbol("Nodes");
export const BoundsInjectionKey: InjectionKey<Ref<DOMRect | undefined>> = Symbol("Bounds");
2022-01-25 04:23:30 +00:00
export interface LayerEvents {
// Generation
preUpdate: (diff: number) => void;
2022-01-25 04:23:30 +00:00
// Actions (e.g. automation)
update: (diff: number) => void;
2022-01-25 04:23:30 +00:00
// Effects (e.g. milestones)
postUpdate: (diff: number) => void;
2022-01-25 04:23:30 +00:00
}
export const layers: Record<string, Readonly<GenericLayer> | undefined> = shallowReactive({});
window.layers = layers;
declare module "@vue/runtime-dom" {
interface CSSProperties {
"--layer-color"?: string;
}
}
2022-01-14 04:25:47 +00:00
export interface Position {
x: number;
y: number;
}
2022-01-14 04:25:47 +00:00
export interface LayerOptions {
color?: Computable<string>;
display: Computable<CoercableComponent>;
classes?: Computable<Record<string, boolean>>;
style?: Computable<StyleValue>;
name?: Computable<string>;
minimizable?: Computable<boolean>;
forceHideGoBack?: Computable<boolean>;
minWidth?: Computable<number | string>;
2022-01-14 04:25:47 +00:00
}
2022-01-14 04:25:47 +00:00
export interface BaseLayer {
id: string;
minimized: Persistent<boolean>;
2022-01-14 04:25:47 +00:00
emitter: Emitter<LayerEvents>;
on: OmitThisParameter<Emitter<LayerEvents>["on"]>;
emit: <K extends keyof LayerEvents>(event: K, ...args: Parameters<LayerEvents[K]>) => void;
nodes: Ref<Record<string, FeatureNode | undefined>>;
2022-01-14 04:25:47 +00:00
}
2022-01-14 04:25:47 +00:00
export type Layer<T extends LayerOptions> = Replace<
T & BaseLayer,
{
color: GetComputableType<T["color"]>;
display: GetComputableType<T["display"]>;
classes: GetComputableType<T["classes"]>;
style: GetComputableType<T["style"]>;
name: GetComputableTypeWithDefault<T["name"], string>;
2022-01-14 04:25:47 +00:00
minWidth: GetComputableTypeWithDefault<T["minWidth"], 600>;
minimizable: GetComputableTypeWithDefault<T["minimizable"], true>;
forceHideGoBack: GetComputableType<T["forceHideGoBack"]>;
}
>;
export type GenericLayer = Replace<
Layer<LayerOptions>,
{
name: ProcessedComputable<string>;
minWidth: ProcessedComputable<number>;
minimizable: ProcessedComputable<boolean>;
}
>;
export const persistentRefs: Record<string, Set<Persistent>> = {};
export const addingLayers: string[] = [];
export function createLayer<T extends LayerOptions>(
id: string,
optionsFunc: (this: BaseLayer) => T & Partial<BaseLayer>
): Layer<T> {
return createLazyProxy(() => {
const layer = {} as T & Partial<BaseLayer>;
const emitter = (layer.emitter = createNanoEvents<LayerEvents>());
layer.on = emitter.on.bind(emitter);
layer.emit = emitter.emit.bind(emitter);
layer.nodes = ref({});
layer.id = id;
2022-01-28 04:47:26 +00:00
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);
2022-01-28 04:47:26 +00:00
return layer as unknown as Layer<T>;
2022-01-28 04:47:26 +00:00
});
2022-01-14 04:25:47 +00:00
}
2022-01-14 04:25:47 +00:00
export function addLayer(
layer: GenericLayer,
player: { layers?: Record<string, Record<string, unknown>> }
): void {
2022-01-25 04:23:30 +00:00
console.info("Adding layer", layer.id);
if (layers[layer.id]) {
2022-01-14 04:25:47 +00:00
console.error(
"Attempted to add layer with same ID as existing layer",
2021-08-18 05:18:23 +00:00
layer.id,
2022-01-14 04:25:47 +00:00
layers[layer.id]
);
2022-01-14 04:25:47 +00:00
return;
}
2022-01-14 04:25:47 +00:00
setDefault(player, "layers", {});
if (player.layers[layer.id] == null) {
player.layers[layer.id] = {};
}
2022-01-14 04:25:47 +00:00
layers[layer.id] = layer;
2022-01-14 04:25:47 +00:00
globalBus.emit("addLayer", layer, player.layers[layer.id]);
}
2022-01-28 04:47:26 +00:00
export function getLayer<T extends GenericLayer>(layerID: string): T {
return layers[layerID] as T;
}
2022-01-14 04:25:47 +00:00
export function removeLayer(layer: GenericLayer): void {
2022-01-25 04:23:30 +00:00
console.info("Removing layer", layer.id);
2022-01-14 04:25:47 +00:00
globalBus.emit("removeLayer", layer);
2022-01-25 04:23:30 +00:00
layers[layer.id] = undefined;
}
2022-01-14 04:25:47 +00:00
export function reloadLayer(layer: GenericLayer): void {
removeLayer(layer);
// Re-create layer
addLayer(layer, player);
}
2022-01-25 04:23:30 +00:00
export function setupLayerModal(layer: GenericLayer): {
openModal: VoidFunction;
modal: JSXFunction;
} {
const showModal = ref(false);
return {
openModal: () => (showModal.value = true),
modal: jsx(() => (
<Modal
modelValue={showModal.value}
onUpdate:modelValue={value => (showModal.value = value)}
v-slots={{
header: () => <h2>{unref(layer.name)}</h2>,
body: unref(layer.display)
}}
/>
))
};
}
2022-01-25 04:23:30 +00:00
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);
});
});