From 6ef88893bff0efa15146ae7dff7f2ac9ac5fbf44 Mon Sep 17 00:00:00 2001 From: p-sw Date: Sun, 19 May 2024 14:30:11 +0900 Subject: [PATCH] docs: add comments to shared.ts --- packages/react/shared.ts | 102 ++++++++++++++++++++++++++++++++++----- 1 file changed, 89 insertions(+), 13 deletions(-) diff --git a/packages/react/shared.ts b/packages/react/shared.ts index 8652051..473b9fe 100644 --- a/packages/react/shared.ts +++ b/packages/react/shared.ts @@ -1,38 +1,91 @@ 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; -type RawVariantProps>> = { - [VariantKey in keyof V]?: BooleanString; +/** + * 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 + * ``` + * type VariantKV = VariantKV + * // VariantKV = { + * // opened: BooleanString<"true" | "false"> = boolean; + * // size: BooleanString<"sm" | "md" | "lg">; + * // color: BooleanString<"red" | "green" | "blue">; + * // } + * ``` + */ +type VariantKV = { + [VariantKey in keyof V]: BooleanString; }; -export function vcn>>({ +export function vcn({ base, variants, defaults, }: { base?: string | undefined; - variants: V; - defaults: { - [VariantKey in keyof V]: BooleanString; - }; + variants: V /* VariantType */; + defaults: VariantKV; }): [ - (variantProps: RawVariantProps & { className?: string }) => string, + (variantProps: Partial> & { className?: string }) => string, >( anyProps: AnyPropBeforeResolve, options?: { excludeClassName?: boolean; } ) => [ - RawVariantProps & { className?: string }, - Omit | "className">, + Partial> & { className?: string }, + Omit> | "className">, ], ] { return [ + /** + * Takes any props (including className), and returns the class name. + * If there is no variant specified in props, then it will fallback to default. + * + * @param variantProps - The variant props including className. + * @returns The class name. + */ ({ className, ...variantProps }) => { return twMerge( base, @@ -41,13 +94,22 @@ export function vcn>>({ ).map( ([variantKey, defaultValue]) => variants[variantKey][ - (variantProps as unknown as RawVariantProps)[variantKey] ?? + (variantProps as unknown as Partial>)[variantKey] ?? defaultValue ] ), className ); }, + /** + * Takes any props, parse variant props and other props. + * If `options.excludeClassName` is true, then it will parse className as "other" props. + * Otherwise, it will parse className as variant props. + * + * @param anyProps - Any props that have passed to the component. + * @param options - Options. + * @returns [variantProps, otherProps] + */ (anyProps, options = {}) => { const variantKeys = Object.keys(variants) as (keyof V)[]; @@ -63,13 +125,27 @@ export function vcn>>({ }, [{}, {}] ) as [ - RawVariantProps & { className?: string }, - Omit | "className">, + Partial> & { className?: string }, + Omit> | "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[0]> = F extends ( props: infer P ) => string