diff --git a/packages/react/shared.ts b/packages/react/shared.ts index 473b9fe..957fdd6 100644 --- a/packages/react/shared.ts +++ b/packages/react/shared.ts @@ -46,47 +46,87 @@ type VariantType = Record>; * * @example * ``` - * type VariantKV = VariantKV - * // VariantKV = { - * // opened: BooleanString<"true" | "false"> = boolean; - * // size: BooleanString<"sm" | "md" | "lg">; - * // color: BooleanString<"red" | "green" | "blue">; - * // } + * const kvs: VariantKV = { + * opened: true // BooleanString<"true" | "false"> = boolean; + * size: "sm" // BooleanString<"sm" | "md" | "lg"> = "sm" | "md" | "lg"; + * color: "red" // BooleanString<"red" | "green" | "blue"> = "red" | "green" | "blue"; + * } * ``` */ type VariantKV = { [VariantKey in keyof V]: BooleanString; }; +/** + * Takes VariantType, and returns a type that represents the preset object. + * + * @example + * ``` + * const presets: PresetType = { + * preset1: { + * opened: true, + * size: "sm", + * color: "red", + * }, + * preset2: { + * opened: false, + * size: "md", + * color: "green", + * }, + * } + * ``` + */ +type PresetType = Record< + string, + Partial> & { className?: string } +>; + export function vcn({ base, variants, defaults, + presets, }: { base?: string | undefined; variants: V /* VariantType */; defaults: VariantKV; + presets?: PresetType; }): [ - (variantProps: Partial> & { className?: string }) => string, + ( + variantProps: Partial> & { + className?: string; + preset?: keyof PresetType; + } + ) => string, >( anyProps: AnyPropBeforeResolve, options?: { + excludePreset?: boolean; excludeClassName?: boolean; } ) => [ - Partial> & { className?: string }, - Omit> | "className">, + Partial> & { + className?: string; + preset?: keyof PresetType; + }, + Omit< + AnyPropBeforeResolve, + keyof Partial> | "className" | "preset" + >, ], ] { return [ /** * Takes any props (including className), and returns the class name. - * If there is no variant specified in props, then it will fallback to default. + * If there is no variant specified in props, then it will fallback to preset, and then default. * * @param variantProps - The variant props including className. * @returns The class name. */ - ({ className, ...variantProps }) => { + ({ className, preset, ...variantProps }) => { + const currentPreset: PresetType[string] | null = + presets && preset ? presets[preset] ?? null : null; + const presetVariantKeys: (keyof V)[] = Object.keys(currentPreset ?? {}); return twMerge( base, ...( @@ -95,16 +135,20 @@ export function vcn({ ([variantKey, defaultValue]) => variants[variantKey][ (variantProps as unknown as Partial>)[variantKey] ?? - defaultValue + (currentPreset !== null && + presetVariantKeys.includes(variantKey) + ? currentPreset[variantKey] ?? defaultValue + : defaultValue) ] ), + currentPreset?.className, className ); }, /** * Takes any props, parse variant props and other props. - * If `options.excludeClassName` is true, then it will parse className as "other" props. - * Otherwise, it will parse className as variant props. + * If `options.excludeA` is true, then it will parse `A` as "other" props. + * Otherwise, it will parse A as variant props. * * @param anyProps - Any props that have passed to the component. * @param options - Options. @@ -117,7 +161,8 @@ export function vcn({ ([variantProps, otherProps], [key, value]) => { if ( variantKeys.includes(key) || - (!options.excludeClassName && key === "className") + (!options.excludeClassName && key === "className") || + (!options.excludePreset && key === "preset") ) { return [{ ...variantProps, [key]: value }, otherProps]; }