import { twMerge } from "tailwind-merge"; type BooleanString = T extends "true" ? true : T extends "false" ? false : T; type RawVariantProps>> = { [VariantKey in keyof V]?: BooleanString; }; export function vcn>>({ base, variants, defaults, }: { base?: string | undefined; variants: V; defaults: { [VariantKey in keyof V]: BooleanString; }; }): [ (variantProps: RawVariantProps & { className?: string }) => string, ( anyProps: Record, options?: { excludeClassName?: boolean; } ) => [RawVariantProps & { className?: string }, Record], ] { return [ ({ className, ...variantProps }) => { return twMerge( base, ...( Object.entries(defaults) as [keyof V, keyof V[keyof V]][] ).map( ([variantKey, defaultValue]) => variants[variantKey][ (variantProps as unknown as RawVariantProps)[variantKey] ?? defaultValue ] ), className ); }, (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 }]; }, [{}, {}] ); }, ]; } export type VariantProps[0]> = F extends ( props: infer P ) => string ? P : never;