diff --git a/packages/react/lib/vcn.ts b/packages/react/lib/vcn.ts index d232f2c..d5e558b 100644 --- a/packages/react/lib/vcn.ts +++ b/packages/react/lib/vcn.ts @@ -57,6 +57,11 @@ type VariantKV = { [VariantKey in keyof V]: BooleanString; }; +/** + * Used for safely casting `Object.entries()` + */ +type VariantKVEntry = [keyof V, BooleanString][] + /** * Takes VariantType, and returns a type that represents the preset object. * @@ -167,6 +172,26 @@ export function vcn< defaults: VariantKV; presets?: P; }) { + /** + * --Internal utility function-- + * After transforming props to final version (which means "after overriding default, preset, and variant props sent via component props") + * It turns final version of variant props to className + */ + function __transformer__(final: VariantKV, dynamics: string[], propClassName?: string): string { + const classNames: string[] = []; + + for (const [variantName, variantKey] of (Object.entries(final) as VariantKVEntry)) { + classNames.push(variants[variantName][variantKey.toString()]) + } + + return twMerge( + base, + ...classNames, + ...dynamics, + propClassName, + ) + } + return [ /** * Takes any props (including className), and returns the class name. @@ -180,42 +205,31 @@ export function vcn< VariantKV >, ) => { - const { className, preset, ...otherVariantProps } = variantProps; + const { className, preset, ..._otherVariantProps } = variantProps; - const currentPreset: P[keyof P] | null = - presets && preset ? (presets as NonNullable

)[preset] ?? null : null; - const presetVariantKeys: (keyof V)[] = Object.keys(currentPreset ?? {}); - return twMerge( - base, - ...( - 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 + // Omit> & { className; preset; }, className | preset> = Partial> (safe to cast) + // We all know `keyof V` = string, right? (but typescript says it's not, so.. attacking typescript with unknown lol) + const otherVariantProps = _otherVariantProps as unknown as Partial> - const directVariantValue: (keyof V[keyof V] & string) | undefined = ( - otherVariantProps as unknown as Partial> - )[variantKey]?.toString?.(); // BooleanString<> -> string (safe to index V[keyof V]) + const kv: VariantKV = { ...defaults }; - const currentPresetVariantValue: - | (keyof V[keyof V] & string) - | undefined = - !!currentPreset && presetVariantKeys.includes(variantKey) - ? (currentPreset as Partial>)[ - variantKey - ]?.toString?.() - : undefined; + // Preset Processing + if (presets && preset && preset in presets) { + for (const [variantName, variantKey] of (Object.entries((presets)[preset]) as VariantKVEntry)) { + kv[variantName] = variantKey + } + } - 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, - ); + // VariantProps Processing + for (const [variantName, variantKey] of (Object.entries(otherVariantProps) as VariantKVEntry)) { + kv[variantName] = variantKey + } + + // make dynamics result + const dynamicClasses: string[] = [] + return __transformer__(kv, dynamicClasses, className); }, + /** * Takes any props, parse variant props and other props. * If `options.excludeA` is true, then it will parse `A` as "other" props.