feat: add presets in vcn

This commit is contained in:
p-sw 2024-05-19 15:23:33 +09:00
parent 6ef88893bf
commit 2caa5dbea8

View File

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