@jdlinker/ui
Version:
jdLinker UI库,基于ant-design-vue封装
245 lines (225 loc) • 8.42 kB
text/typescript
import { warn } from 'vue';
import { isObject } from '@vue/shared';
import { fromPairs } from 'lodash-es';
import type { ExtractPropTypes, PropType } from 'vue';
import { ContextMenuOptions, TreeActionItem, TreeItem } from '../Tree';
const wrapperKey = Symbol();
export type PropWrapper<T> = { [wrapperKey]: T };
export const propKey = Symbol();
type ResolveProp<T> = ExtractPropTypes<{
key: { type: T; required: true };
}>['key'];
type ResolvePropType<T> = ResolveProp<T> extends { type: infer V } ? V : ResolveProp<T>;
type ResolvePropTypeWithReadonly<T> = Readonly<T> extends Readonly<Array<infer A>>
? ResolvePropType<A[]>
: ResolvePropType<T>;
type IfUnknown<T, V> = [unknown] extends [T] ? V : T;
export type BuildPropOption<T, D extends BuildPropType<T, V, C>, R, V, C> = {
type?: T;
values?: readonly V[];
required?: R;
default?: R extends true ? never : D extends Record<string, unknown> | Array<any> ? () => D : (() => D) | D;
validator?: ((val: any) => val is C) | ((val: any) => boolean);
};
type _BuildPropType<T, V, C> =
| (T extends PropWrapper<unknown>
? T[typeof wrapperKey]
: [V] extends [never]
? ResolvePropTypeWithReadonly<T>
: never)
| V
| C;
export type BuildPropType<T, V, C> = _BuildPropType<IfUnknown<T, never>, IfUnknown<V, never>, IfUnknown<C, never>>;
type _BuildPropDefault<T, D> = [T] extends [
// eslint-disable-next-line @typescript-eslint/ban-types
Record<string, unknown> | Array<any> | Function
]
? D
: D extends () => T
? ReturnType<D>
: D;
export type BuildPropDefault<T, D, R> = R extends true
? { readonly default?: undefined }
: {
readonly default: Exclude<D, undefined> extends never ? undefined : Exclude<_BuildPropDefault<T, D>, undefined>;
};
export type BuildPropReturn<T, D, R, V, C> = {
readonly type: PropType<BuildPropType<T, V, C>>;
readonly required: IfUnknown<R, false>;
readonly validator: ((val: unknown) => boolean) | undefined;
[propKey]: true;
} & BuildPropDefault<BuildPropType<T, V, C>, IfUnknown<D, never>, IfUnknown<R, false>>;
/**
* @description Build prop. It can better optimize prop types
* @description 生成 prop,能更好地优化类型
* @example
// limited options
// the type will be PropType<'light' | 'dark'>
buildProp({
type: String,
values: ['light', 'dark'],
} as const)
* @example
// limited options and other types
// the type will be PropType<'small' | 'medium' | number>
buildProp({
type: [String, Number],
values: ['small', 'medium'],
validator: (val: unknown): val is number => typeof val === 'number',
} as const)
@link see more: https://github.com/element-plus/element-plus/pull/3341
*/
export function buildProp<
T = never,
D extends BuildPropType<T, V, C> = never,
R extends boolean = false,
V = never,
C = never
>(option: BuildPropOption<T, D, R, V, C>, key?: string): BuildPropReturn<T, D, R, V, C> {
// filter native prop type and nested prop, e.g `null`, `undefined` (from `buildProps`)
if (!isObject(option) || !!option[propKey]) return option as any;
const { values, required, default: defaultValue, type, validator } = option;
const _validator =
values || validator
? (val: unknown) => {
let valid = false;
let allowedValues: unknown[] = [];
if (values) {
allowedValues = [...values, defaultValue];
valid ||= allowedValues.includes(val);
}
if (validator) valid ||= validator(val);
if (!valid && allowedValues.length > 0) {
// @ts-ignore
const allowValuesText = [...new Set(allowedValues)].map((value) => JSON.stringify(value)).join(', ');
warn(
`Invalid prop: validation failed${
key ? ` for prop "${key}"` : ''
}. Expected one of [${allowValuesText}], got value ${JSON.stringify(val)}.`
);
}
return valid;
}
: undefined;
return {
type: typeof type === 'object' && Object.getOwnPropertySymbols(type).includes(wrapperKey) ? type[wrapperKey] : type,
required: !!required,
default: defaultValue,
validator: _validator,
[propKey]: true
} as unknown as BuildPropReturn<T, D, R, V, C>;
}
type NativePropType = [((...args: any) => any) | { new (...args: any): any } | undefined | null];
export const buildProps = <
O extends {
[K in keyof O]: O[K] extends BuildPropReturn<any, any, any, any, any>
? O[K]
: [O[K]] extends NativePropType
? O[K]
: O[K] extends BuildPropOption<infer T, infer D, infer R, infer V, infer C>
? D extends BuildPropType<T, V, C>
? BuildPropOption<T, D, R, V, C>
: never
: never;
}
>(props: {
expandedKeys: {
default: () => any[];
type: { new (...args: any[]): KeyType[] & {} } | { (): KeyType[] } | any[];
};
fieldNames: { type: { new (...args: any[]): any } | { (): any } | any[] };
selectedKeys: {
default: () => any[];
type: { new (...args: any[]): KeyType[] & {} } | { (): KeyType[] } | any[];
};
renderIcon: {
type:
| { new (...args: any[]): (params: any) => string }
| { (): (params: any) => string }
| any
| any[];
};
title: { default: string; type: StringConstructor };
clickRowToExpand: { default: boolean; type: BooleanConstructor };
selectedOnSearch: BooleanConstructor;
beforeRightClick: {
default: undefined;
type:
| { new (...args: any[]): (...arg: any) => any[] | ContextMenuOptions }
| { (): (...arg: any) => any[] | ContextMenuOptions }
| any
| any[];
};
highlight: {
default: boolean;
type: { new (...args: any[]): Boolean | String } | { (): Boolean | String } | any[];
};
search: BooleanConstructor;
checkOnSearch: BooleanConstructor;
defaultExpandAll: BooleanConstructor;
checkStrictly: BooleanConstructor;
value: {
type:
| { new (...args: any[]): (KeyType[] & {}) | { checked: string[] | number[]; halfChecked: string[] | number[] } }
| { (): KeyType[] | { checked: string[] | number[]; halfChecked: string[] | number[] } }
| any[];
};
defaultExpandLevel: {
default: string;
type: { new (...args: any[]): string | number } | { (): string | number } | any[];
};
filterFn: {
default: undefined;
type:
| { new (...args: any[]): (searchValue: any, node: TreeItem, fieldNames: any) => boolean }
| { (): (searchValue: any, node: TreeItem, fieldNames: any) => boolean }
| any
| any[];
};
checkable: BooleanConstructor;
actionList: {
default: () => any[];
type: { new (...args: any[]): TreeActionItem[] & {} } | { (): TreeActionItem[] } | any[];
};
loading: { default: boolean; type: BooleanConstructor };
helpMessage: {
default: string;
type: { new (...args: any[]): string | (string[] & {}) } | { (): string | string[] } | any[];
};
checkedKeys: {
default: () => any[];
type:
| { new (...args: any[]): (KeyType[] & {}) | { checked: string[] | number[]; halfChecked: string[] | number[] } }
| { (): KeyType[] | { checked: string[] | number[]; halfChecked: string[] | number[] } }
| any[];
};
toolbar: BooleanConstructor;
rightMenuList: {
type: { new (...args: any[]): any[] & {} } | { (): any[] } | any[];
};
searchValue: { default: string; type: StringConstructor };
expandOnSearch: BooleanConstructor;
treeData: {
type: { new (...args: any[]): any[] & {} } | { (): any[] } | any[];
};
}) =>
fromPairs(Object.entries(props).map(([key, option]) => [key, buildProp(option as any, key)])) as unknown as {
[K in keyof O]: O[K] extends { [propKey]: boolean }
? O[K]
: [O[K]] extends NativePropType
? O[K]
: O[K] extends BuildPropOption<
infer T,
// eslint-disable-next-line @typescript-eslint/no-unused-vars
infer _D,
infer R,
infer V,
infer C
>
? BuildPropReturn<T, O[K]['default'], R, V, C>
: never;
};
export const definePropType = <T>(val: any) => ({ [wrapperKey]: val } as PropWrapper<T>);
export const keyOf = <T>(arr: T) => Object.keys(arr) as Array<keyof T>;
export const mutable = <T extends readonly any[] | Record<string, unknown>>(val: T) => val as any;
export const componentSize = ['large', 'medium', 'small', 'mini'] as const;