UNPKG

mobx-keystone

Version:

A MobX powered state management solution based on data trees with first class support for TypeScript, snapshots, patches and much more

125 lines (102 loc) 3.35 kB
import { modelTypeKey } from "../../model/metadata" import { modelInfoByClass } from "../../modelShared/modelInfo" import { isObject } from "../../utils" import { ObjectMap } from "../../wrappers/ObjectMap" import { getTypeInfo } from "../getTypeInfo" import { resolveStandardType, resolveTypeChecker } from "../resolveTypeChecker" import type { AnyStandardType, AnyType, ModelType, TypeToData } from "../schemas" import { lateTypeChecker, TypeChecker, TypeCheckerBaseType, TypeInfo, TypeInfoGen, } from "../TypeChecker" import { TypeCheckError } from "../TypeCheckError" import { typesObject } from "./typesObject" import { typesRecord } from "./typesRecord" /** * A type that represents an object-like map ObjectMap. * * Example: * ```ts * const numberMapType = types.objectMap(types.number) * ``` * * @template T Value type. * @param valueType Value type. * @returns */ export function typesObjectMap<T extends AnyType>( valueType: T ): ModelType<ObjectMap<TypeToData<T>>> { const typeInfoGen: TypeInfoGen = (t) => new ObjectMapTypeInfo(t, resolveStandardType(valueType)) return lateTypeChecker(() => { const modelInfo = modelInfoByClass.get(ObjectMap)! const valueChecker = resolveTypeChecker(valueType) const getTypeName = (...recursiveTypeCheckers: TypeChecker[]) => `ObjectMap<${valueChecker.getTypeName(...recursiveTypeCheckers, valueChecker)}>` const dataTypeChecker = typesObject(() => ({ items: typesRecord(valueChecker as any), })) const resolvedDataTypeChecker = resolveTypeChecker(dataTypeChecker) const thisTc: TypeChecker = new TypeChecker( TypeCheckerBaseType.Object, (obj, path, typeCheckedValue) => { if (!(obj instanceof ObjectMap)) { return new TypeCheckError(path, getTypeName(thisTc), obj, typeCheckedValue) } return resolvedDataTypeChecker.check(obj.$, path, typeCheckedValue) }, getTypeName, typeInfoGen, (obj) => { if (!isObject(obj)) { return null } if (obj[modelTypeKey] !== undefined) { // fast check return obj[modelTypeKey] === modelInfo.name ? thisTc : null } return resolvedDataTypeChecker.snapshotType(obj) ? thisTc : null }, (sn: { items: Record<string, unknown> }) => { const newItems: (typeof sn)["items"] = {} for (const k of Object.keys(sn.items)) { newItems[k] = valueChecker.fromSnapshotProcessor(sn.items[k]) } return { ...sn, [modelTypeKey]: modelInfo.name, items: newItems, } }, (sn: { items: Record<string, unknown>; [modelTypeKey]?: string }) => { const newItems: (typeof sn)["items"] = {} for (const k of Object.keys(sn.items)) { newItems[k] = valueChecker.toSnapshotProcessor(sn.items[k]) } const snCopy = { ...sn, items: newItems, } return snCopy } ) return thisTc }, typeInfoGen) as any } /** * `types.objectMap` type info. */ export class ObjectMapTypeInfo extends TypeInfo { get valueTypeInfo(): TypeInfo { return getTypeInfo(this.valueType) } constructor( thisType: AnyStandardType, readonly valueType: AnyStandardType ) { super(thisType) } }