diff --git a/packages/react/shared.tsx b/packages/react/shared.tsx index d07392e..2f4d675 100644 --- a/packages/react/shared.tsx +++ b/packages/react/shared.tsx @@ -104,13 +104,13 @@ export function vcn(param: { ( variantProps: Partial> & { className?: string; - } + }, ) => string, /** * Any Props -> Variant Props, Other Props */ >( - anyProps: AnyPropBeforeResolve + anyProps: AnyPropBeforeResolve, ) => [ Partial> & { className?: string; @@ -134,13 +134,13 @@ export function vcn>(param: { variantProps: Partial> & { className?: string; preset?: keyof P; - } + }, ) => string, /** * Any Props -> Variant Props, Other Props */ >( - anyProps: AnyPropBeforeResolve + anyProps: AnyPropBeforeResolve, ) => [ Partial> & { className?: string; @@ -174,7 +174,11 @@ export function vcn< * @param variantProps - The variant props including className. * @returns The class name. */ - (variantProps: { className?: string; preset?: keyof P } & Partial>) => { + ( + variantProps: { className?: string; preset?: keyof P } & Partial< + VariantKV + >, + ) => { const { className, preset, ...otherVariantProps } = variantProps; const currentPreset: P[keyof P] | null = @@ -183,19 +187,32 @@ export function vcn< return twMerge( base, ...( - Object.entries(defaults) as [keyof V, keyof V[keyof V]][] - ).map( - ([variantKey, defaultValue]) => - variants[variantKey][ - (otherVariantProps as unknown as Partial>)[variantKey] ?? - (!!currentPreset && presetVariantKeys.includes(variantKey) - ? (currentPreset as Partial>)[variantKey] ?? - defaultValue - : defaultValue) - ] - ), - (currentPreset as Partial> | null)?.className, // preset's classname comes after user's variant props? huh.. - className + Object.entries(defaults) as [keyof V, keyof V[keyof V] & string][] + ).map(([variantKey, defaultValue]) => { + // Omit> & { className; preset; }, className | preset> = Partial> (safe to cast) + // Partial>[keyof V] => { [k in keyof V]?: BooleanString } => BooleanString + + const directVariantValue: (keyof V[keyof V] & string) | undefined = ( + otherVariantProps as unknown as Partial> + )[variantKey]?.toString?.(); // BooleanString<> -> string (safe to index V[keyof V]) + + const currentPresetVariantValue: + | (keyof V[keyof V] & string) + | undefined = + !!currentPreset && presetVariantKeys.includes(variantKey) + ? (currentPreset as Partial>)[ + variantKey + ]?.toString?.() + : undefined; + + const variantValue: keyof V[keyof V] & string = + directVariantValue ?? currentPresetVariantValue ?? defaultValue; + return variants[variantKey][variantValue]; + }), + ( + currentPreset as Partial> | null + )?.className?.toString?.(), // preset's classname comes after user's variant props? huh.. + className, ); }, /** @@ -207,7 +224,7 @@ export function vcn< * @returns [variantProps, otherProps] */ >( - anyProps: AnyPropBeforeResolve + anyProps: AnyPropBeforeResolve, ) => { const variantKeys = Object.keys(variants) as (keyof V)[]; @@ -222,7 +239,7 @@ export function vcn< } return [variantProps, { ...otherProps, [key]: value }]; }, - [{}, {}] + [{}, {}], ) as [ Partial> & { className?: string; @@ -252,7 +269,7 @@ export function vcn< * ``` */ export type VariantProps string> = F extends ( - props: infer P + props: infer P, ) => string ? P : never; @@ -268,7 +285,7 @@ export type VariantProps string> = F extends ( */ function mergeReactProps( parentProps: Record, - childProps: Record + childProps: Record, ) { const overrideProps = { ...childProps }; @@ -328,7 +345,7 @@ export const Slot = React.forwardRef>( ...mergeReactProps(safeSlotProps, children.props), ref: combinedRef([ref, (children as any).ref]), } as any); - } + }, ); export interface AsChild {