import { twMerge } from "tailwind-merge"; /** * Takes a string, and returns boolean if it is "true" or "false". * Otherwise, returns the string. * * @example * ``` * type BooleanString = BooleanString<"true" | "false" | "other"> * // BooleanString = true | false | "other" = boolean | "other" * ``` */ type BooleanString = T extends "true" ? true : T extends "false" ? false : T; /** * A type that represents a variant object. * * @example * ``` * const variant: VariantType = { * opened: { * true: "opacity-100", * false: "opacity-0", * } * size: { * sm: "small", * md: "medium", * lg: "large", * }, * color: { * red: "#ff0000", * green: "#00ff00", * blue: "#0000ff", * }, * } * ``` */ type VariantType = Record>; /** * Takes VariantType, and returns a type that represents the variant object. * * @example * ``` * type VariantKV = VariantKV * // VariantKV = { * // opened: BooleanString<"true" | "false"> = boolean; * // size: BooleanString<"sm" | "md" | "lg">; * // color: BooleanString<"red" | "green" | "blue">; * // } * ``` */ type VariantKV = { [VariantKey in keyof V]: BooleanString; }; export function vcn({ base, variants, defaults, }: { base?: string | undefined; variants: V /* VariantType */; defaults: VariantKV; }): [ (variantProps: Partial> & { className?: string }) => string, >( anyProps: AnyPropBeforeResolve, options?: { excludeClassName?: boolean; } ) => [ Partial> & { className?: string }, Omit> | "className">, ], ] { 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. * * @param variantProps - The variant props including className. * @returns The class name. */ ({ className, ...variantProps }) => { return twMerge( base, ...( Object.entries(defaults) as [keyof V, keyof V[keyof V]][] ).map( ([variantKey, defaultValue]) => variants[variantKey][ (variantProps as unknown as Partial>)[variantKey] ?? defaultValue ] ), 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. * * @param anyProps - Any props that have passed to the component. * @param options - Options. * @returns [variantProps, otherProps] */ (anyProps, options = {}) => { const variantKeys = Object.keys(variants) as (keyof V)[]; return Object.entries(anyProps).reduce( ([variantProps, otherProps], [key, value]) => { if ( variantKeys.includes(key) || (!options.excludeClassName && key === "className") ) { return [{ ...variantProps, [key]: value }, otherProps]; } return [variantProps, { ...otherProps, [key]: value }]; }, [{}, {}] ) as [ Partial> & { className?: string }, Omit> | "className">, ]; }, ]; } /** * Extract the props type from return value of `vcn` function. * * @example * ``` * const [variantProps, otherProps] = vcn({ ... }) * interface Props * extends VariantProps, OtherProps { ... } * * function Component(props: Props) { * ... * } * ``` */ export type VariantProps[0]> = F extends ( props: infer P ) => string ? P : never;