feat: add slot component in shared
This commit is contained in:
parent
7d3413451f
commit
27c4ab0470
@ -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;
|
||||||
|
}
|
Loading…
x
Reference in New Issue
Block a user