import { Field, Opt } from "./Doc"; import { List } from "./List"; export type ToType = T extends "string" ? string : T extends "number" ? number : T extends "boolean" ? boolean : T extends ListSpec ? List : T extends { new(...args: any[]): infer R } ? R : never; export type ToConstructor = T extends string ? "string" : T extends number ? "number" : T extends boolean ? "boolean" : { new(...args: any[]): T }; export type ToInterface = { [P in keyof T]: ToType; }; // type ListSpec = { List: FieldCtor> | ListSpec> }; export type ListSpec = { List: FieldCtor }; // type ListType = { 0: List>>, 1: ToType> }[HasTail extends true ? 0 : 1]; export type Head = T extends [any, ...any[]] ? T[0] : never; export type Tail = ((...t: T) => any) extends ((_: any, ...tail: infer TT) => any) ? TT : []; export type HasTail = T extends ([] | [any]) ? false : true; //TODO Allow you to optionally specify default values for schemas, which should then make that field not be partial export interface Interface { [key: string]: ToConstructor | ListSpec; // [key: string]: ToConstructor | ListSpec; } export type FieldCtor = T extends List ? ListSpec : ToConstructor; export function Cast>(field: Field | null | undefined, ctor: T): ToType | null | undefined; export function Cast>(field: Field | null | undefined, ctor: T, defaultVal: ToType): ToType; export function Cast>(field: Field | null | undefined, ctor: T, defaultVal?: ToType): ToType | null | undefined { if (field !== undefined && field !== null) { if (typeof ctor === "string") { if (typeof field === ctor) { return field as ToType; } } else if (typeof ctor === "object") { if (field instanceof List) { return field as any; } } else if (field instanceof (ctor as any)) { return field as ToType; } } return defaultVal; } export function FieldValue(field: Opt | Promise>, defaultValue: U): T; export function FieldValue(field: Opt | Promise>): Opt; export function FieldValue(field: Opt | Promise>, defaultValue?: T): Opt { return field instanceof Promise ? defaultValue : field; }