diff --git a/packages/react/lib/Slot.tsx b/packages/react/lib/Slot.tsx new file mode 100644 index 0000000..51ee483 --- /dev/null +++ b/packages/react/lib/Slot.tsx @@ -0,0 +1,97 @@ +import { twMerge } from "tailwind-merge"; +import React from "react"; + +/** + * 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; +} diff --git a/packages/react/lib.tsx b/packages/react/lib/vcn.ts similarity index 73% rename from packages/react/lib.tsx rename to packages/react/lib/vcn.ts index a3fad95..41f7669 100644 --- a/packages/react/lib.tsx +++ b/packages/react/lib/vcn.ts @@ -1,4 +1,3 @@ -import React from "react"; import { twMerge } from "tailwind-merge"; /** @@ -273,98 +272,3 @@ export type VariantProps string> = F extends ( ) => 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; -}