feat: add slot component in shared

This commit is contained in:
p-sw 2024-05-24 22:03:16 +09:00
parent 7d3413451f
commit 27c4ab0470

View File

@ -1,3 +1,4 @@
import React from "react";
import { twMerge } from "tailwind-merge"; import { twMerge } from "tailwind-merge";
/** /**
@ -88,7 +89,10 @@ type PresetType<V extends VariantType, N extends string> = {
* @returns function (variantProps) -> class name, * @returns function (variantProps) -> class name,
* @returns function (anyProps) -> [variantProps, otherProps] * @returns function (anyProps) -> [variantProps, otherProps]
*/ */
export function vcn<V extends VariantType, N extends string /* Preset names */>({ export function vcn<
V extends VariantType,
N extends string /* Preset names */,
>({
base, base,
variants, variants,
defaults, defaults,
@ -106,7 +110,7 @@ export function vcn<V extends VariantType, N extends string /* Preset names */>(
} }
) => string, ) => string,
<AnyPropBeforeResolve extends Record<string, any>>( <AnyPropBeforeResolve extends Record<string, any>>(
anyProps: AnyPropBeforeResolve, anyProps: AnyPropBeforeResolve
) => [ ) => [
Partial<VariantKV<V>>, Partial<VariantKV<V>>,
Omit< Omit<
@ -158,9 +162,7 @@ export function vcn<V extends VariantType, N extends string /* Preset names */>(
return Object.entries(anyProps).reduce( return Object.entries(anyProps).reduce(
([variantProps, otherProps], [key, value]) => { ([variantProps, otherProps], [key, value]) => {
if ( if (variantKeys.includes(key)) {
variantKeys.includes(key)
) {
return [{ ...variantProps, [key]: value }, otherProps]; return [{ ...variantProps, [key]: value }, otherProps];
} }
return [variantProps, { ...otherProps, [key]: value }]; return [variantProps, { ...otherProps, [key]: value }];
@ -168,7 +170,10 @@ export function vcn<V extends VariantType, N extends string /* Preset names */>(
[{}, {}] [{}, {}]
) as [ ) as [
Partial<VariantKV<V>>, Partial<VariantKV<V>>,
Omit<typeof anyProps, keyof Partial<VariantKV<V>> | "preset" | "className">, Omit<
typeof anyProps,
keyof Partial<VariantKV<V>> | "preset" | "className"
>,
]; ];
}, },
]; ];
@ -193,3 +198,72 @@ export type VariantProps<F extends (props: any) => string> = F extends (
) => string ) => string
? P ? P
: never; : never;
function mergeReactProps(
parentProps: Record<string, any>,
childProps: Record<string, any>
) {
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) {
overrideProps[propName] = (...args: unknown[]) => {
childPropValue?.(...args);
parentPropValue?.(...args);
};
} else if (parentPropValue) {
overrideProps[propName] = parentPropValue;
}
} else if (propName === "style") {
overrideProps[propName] = { ...parentPropValue, ...childPropValue };
} else if (propName === "className") {
overrideProps[propName] = twMerge(parentPropValue, childPropValue);
}
}
return { ...parentProps, ...overrideProps };
}
function combinedRef<I>(refs: React.Ref<I | null>[]) {
return (instance: I | null) =>
refs.forEach((ref) => {
if (ref instanceof Function) {
ref(instance);
} else if (!!ref) {
(ref as React.MutableRefObject<I | null>).current = instance;
}
});
}
interface SlotProps {
children?: Exclude<React.ReactNode, Iterable<React.ReactNode>> | string;
}
export const Slot = React.forwardRef<any, SlotProps>((props, ref) => {
const { children, ...slotProps } = props;
if (!React.isValidElement(children)) {
return null;
}
return React.cloneElement(children, {
...mergeReactProps(slotProps, children.props),
ref: combinedRef([ref, (children as any).ref]),
} as any);
});
export interface MustAsChild {
children: React.ReactElement<
unknown,
string | React.JSXElementConstructor<any>
>;
}
export interface OptionalAsChild<T extends boolean> {
children?: T extends true
? React.ReactElement<unknown, string | React.JSXElementConstructor<any>>
: React.ReactNode;
asChild?: T;
}