70 lines
2.3 KiB
TypeScript
70 lines
2.3 KiB
TypeScript
import { NonPersistent } from "game/persistence";
|
|
import Decimal from "util/bignum";
|
|
|
|
export const ProxyState = Symbol("ProxyState");
|
|
export const ProxyPath = Symbol("ProxyPath");
|
|
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
export type ProxiedWithState<T> = NonNullable<T> extends Record<PropertyKey, any>
|
|
? NonNullable<T> extends Decimal
|
|
? T
|
|
: {
|
|
[K in keyof T]: ProxiedWithState<T[K]>;
|
|
} & {
|
|
[ProxyState]: T;
|
|
[ProxyPath]: string[];
|
|
}
|
|
: T;
|
|
|
|
// Takes a function that returns an object and pretends to be that object
|
|
// Note that the object is lazily calculated
|
|
export function createLazyProxy<T extends object, S extends T>(
|
|
objectFunc: (baseObject: S) => T & S,
|
|
baseObject: S = {} as S
|
|
): T {
|
|
const obj: S & Partial<T> = baseObject;
|
|
let calculated = false;
|
|
function calculateObj(): T {
|
|
if (!calculated) {
|
|
Object.assign(obj, objectFunc(obj));
|
|
calculated = true;
|
|
}
|
|
return obj as S & T;
|
|
}
|
|
|
|
return new Proxy(obj, {
|
|
get(target, key) {
|
|
if (key === ProxyState) {
|
|
return calculateObj();
|
|
}
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
const val = (calculateObj() as any)[key];
|
|
if (val != null && typeof val === "object" && NonPersistent in val) {
|
|
return val[NonPersistent];
|
|
}
|
|
return val;
|
|
},
|
|
set(target, key, value) {
|
|
// TODO give warning about this? It should only be done with caution
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
(calculateObj() as any)[key] = value;
|
|
return true;
|
|
},
|
|
has(target, key) {
|
|
if (key === ProxyState) {
|
|
return true;
|
|
}
|
|
return Reflect.has(calculateObj(), key);
|
|
},
|
|
ownKeys() {
|
|
return Reflect.ownKeys(calculateObj());
|
|
},
|
|
getOwnPropertyDescriptor(target, key) {
|
|
if (!calculated) {
|
|
Object.assign(obj, objectFunc(obj));
|
|
calculated = true;
|
|
}
|
|
return Object.getOwnPropertyDescriptor(target, key);
|
|
}
|
|
}) as S & T;
|
|
}
|