import React from "react"; 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 * ``` * const kvs: VariantKV = { * opened: true // BooleanString<"true" | "false"> = boolean; * size: "sm" // BooleanString<"sm" | "md" | "lg"> = "sm" | "md" | "lg"; * color: "red" // BooleanString<"red" | "green" | "blue"> = "red" | "green" | "blue"; * } * ``` */ type VariantKV = { [VariantKey in keyof V]: BooleanString; }; /** * Takes VariantType, and returns a type that represents the preset object. * * @example * ``` * const presets: PresetType = { * preset1: { * opened: true, * size: "sm", * color: "red", * }, * preset2: { * opened: false, * size: "md", * color: "green", * className: "transition-opacity", * }, * } * ``` */ type PresetType = { [PresetName: string]: Partial> & { className?: string }; }; /** * A utility function to provide variants and presets to the component * * @param param - Variant Configuration * @returns function (variantProps) -> class name, * @returns function (anyProps) -> [variantProps, otherProps] */ export function vcn(param: { /** * First definition: without presets */ base?: string | undefined; variants: V; defaults: VariantKV; presets?: undefined; }): [ /** * Variant Props -> Class Name */ ( variantProps: Partial> & { className?: string; }, ) => string, /** * Any Props -> Variant Props, Other Props */ >( anyProps: AnyPropBeforeResolve, ) => [ Partial> & { className?: string; }, Omit> | "className">, ], ]; export function vcn>(param: { /** * Second definition: with presets */ base?: string | undefined; variants: V /* VariantType */; defaults: VariantKV; presets: P; }): [ /** * Variant Props -> Class Name */ ( variantProps: Partial> & { className?: string; preset?: keyof P; }, ) => string, /** * Any Props -> Variant Props, Other Props */ >( anyProps: AnyPropBeforeResolve, ) => [ Partial> & { className?: string; preset?: keyof P; }, Omit< AnyPropBeforeResolve, keyof Partial> | "preset" | "className" >, ], ]; export function vcn< V extends VariantType, P extends PresetType | undefined, >({ base, variants, defaults, presets, }: { base?: string | undefined; variants: V; defaults: VariantKV; presets?: P; }) { return [ /** * Takes any props (including className), and returns the class name. * If there is no variant specified in props, then it will fallback to preset, and then default. * * @param variantProps - The variant props including className. * @returns The class name. */ ( variantProps: { className?: string; preset?: keyof P } & Partial< VariantKV >, ) => { 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 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, ); }, /** * Takes any props, parse variant props and other props. * If `options.excludeA` is true, then it will parse `A` as "other" props. * Otherwise, it will parse A as variant props. * * @param anyProps - Any props that have passed to the component. * @returns [variantProps, otherProps] */ >( anyProps: AnyPropBeforeResolve, ) => { const variantKeys = Object.keys(variants) as (keyof V)[]; return Object.entries(anyProps).reduce( ([variantProps, otherProps], [key, value]) => { if ( variantKeys.includes(key) || key === "className" || key === "preset" ) { return [{ ...variantProps, [key]: value }, otherProps]; } return [variantProps, { ...otherProps, [key]: value }]; }, [{}, {}], ) as [ Partial> & { className?: string; preset?: keyof P; }, Omit< typeof anyProps, keyof Partial> | "preset" | "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 string> = F extends ( props: infer P, ) => string ? P : never; /** * Merges the react props. * Basically childProps will override parentProps. * But if it is a event handler, style, or className, it will be merged. * * @param parentProps - The parent props. * @param childProps - The child props. * @returns The merged props. */ function mergeReactProps( parentProps: Record, childProps: Record, ) { const overrideProps = { ...childProps }; for (const propName in childProps) { const parentPropValue = parentProps[propName]; const childPropValue = childProps[propName]; const isHandler = /^on[A-Z]/.test(propName); if (isHandler) { if ( childPropValue && parentPropValue && typeof childPropValue === "function" && typeof parentPropValue === "function" ) { overrideProps[propName] = (...args: unknown[]) => { childPropValue?.(...args); parentPropValue?.(...args); }; } else if (parentPropValue) { overrideProps[propName] = parentPropValue; } } else if ( propName === "style" && typeof parentPropValue === "object" && typeof childPropValue === "object" ) { overrideProps[propName] = { ...parentPropValue, ...childPropValue }; } else if ( propName === "className" && typeof parentPropValue === "string" && typeof childPropValue === "string" ) { overrideProps[propName] = twMerge(parentPropValue, childPropValue); } } return { ...parentProps, ...overrideProps }; } /** * Takes an array of refs, and returns a single ref. * * @param refs - The array of refs. * @returns The single ref. */ function combinedRef(refs: React.Ref[]) { return (instance: I | null) => refs.forEach((ref) => { if (ref instanceof Function) { ref(instance); } else if (ref) { (ref as React.MutableRefObject).current = instance; } }); } interface SlotProps { children?: React.ReactNode; } export const Slot = React.forwardRef< HTMLElement, SlotProps & Record >((props, ref) => { const { children, ...slotProps } = props; const { asChild: _1, ...safeSlotProps } = slotProps; if (!React.isValidElement(children)) { console.warn(`given children "${children}" is not valid for asChild`); return null; } return React.cloneElement(children, { ...mergeReactProps(safeSlotProps, children.props), ref: combinedRef([ ref, (children as unknown as { ref: React.Ref }).ref, ]), } as never); }); export interface AsChild { asChild?: boolean; }