fix: upgrade all component to fix biomejs errors
This commit is contained in:
parent
417deb4aa6
commit
8c7f9bd257
@ -1,8 +1,9 @@
|
|||||||
|
import { type AsChild, Slot, type VariantProps, vcn } from "@pswui-lib";
|
||||||
import React from "react";
|
import React from "react";
|
||||||
import { vcn, VariantProps, Slot, AsChild } from "@pswui-lib";
|
|
||||||
|
|
||||||
const colors = {
|
const colors = {
|
||||||
disabled: "disabled:brightness-50 disabled:cursor-not-allowed disabled:opacity-50 disabled:saturate-50",
|
disabled:
|
||||||
|
"disabled:brightness-50 disabled:cursor-not-allowed disabled:opacity-50 disabled:saturate-50",
|
||||||
outline: {
|
outline: {
|
||||||
focus: "dark:focus-visible:outline-white/20 focus-visible:outline-black/10",
|
focus: "dark:focus-visible:outline-white/20 focus-visible:outline-black/10",
|
||||||
},
|
},
|
||||||
@ -118,7 +119,12 @@ const Button = React.forwardRef<HTMLButtonElement, ButtonProps>(
|
|||||||
className: buttonVariants(variantProps),
|
className: buttonVariants(variantProps),
|
||||||
};
|
};
|
||||||
|
|
||||||
return <Comp ref={ref} {...compProps} />;
|
return (
|
||||||
|
<Comp
|
||||||
|
ref={ref}
|
||||||
|
{...compProps}
|
||||||
|
/>
|
||||||
|
);
|
||||||
},
|
},
|
||||||
);
|
);
|
||||||
|
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
|
import { type VariantProps, vcn } from "@pswui-lib";
|
||||||
import React from "react";
|
import React from "react";
|
||||||
import { VariantProps, vcn } from "@pswui-lib";
|
|
||||||
|
|
||||||
const checkboxColors = {
|
const checkboxColors = {
|
||||||
background: {
|
background: {
|
||||||
@ -99,10 +99,11 @@ const Checkbox = React.forwardRef<HTMLInputElement, CheckboxProps>(
|
|||||||
viewBox="0 0 24 24"
|
viewBox="0 0 24 24"
|
||||||
className={`${checked ? "opacity-100" : "opacity-0"} transition-opacity duration-75 ease-in-out`}
|
className={`${checked ? "opacity-100" : "opacity-0"} transition-opacity duration-75 ease-in-out`}
|
||||||
>
|
>
|
||||||
|
<title>checked</title>
|
||||||
<path
|
<path
|
||||||
fill="currentColor"
|
fill="currentColor"
|
||||||
d="M21 7L9 19l-5.5-5.5l1.41-1.41L9 16.17L19.59 5.59z"
|
d="M21 7L9 19l-5.5-5.5l1.41-1.41L9 16.17L19.59 5.59z"
|
||||||
></path>
|
/>
|
||||||
</svg>
|
</svg>
|
||||||
</label>
|
</label>
|
||||||
</>
|
</>
|
||||||
|
@ -1,12 +1,12 @@
|
|||||||
|
import { Slot, type VariantProps, vcn } from "@pswui-lib";
|
||||||
import React, { useState } from "react";
|
import React, { useState } from "react";
|
||||||
import { Slot, VariantProps, vcn } from "@pswui-lib";
|
|
||||||
import ReactDOM from "react-dom";
|
import ReactDOM from "react-dom";
|
||||||
|
|
||||||
import {
|
import {
|
||||||
DialogContext,
|
DialogContext,
|
||||||
|
type IDialogContext,
|
||||||
initialDialogContext,
|
initialDialogContext,
|
||||||
useDialogContext,
|
useDialogContext,
|
||||||
IDialogContext,
|
|
||||||
} from "./Context";
|
} from "./Context";
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -55,7 +55,7 @@ const DialogTrigger = ({ children }: DialogTriggerProps) => {
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
const [dialogOverlayVariant, resolveDialogOverlayVariant] = vcn({
|
const [dialogOverlayVariant, resolveDialogOverlayVariant] = vcn({
|
||||||
base: "fixed inset-0 z-50 w-full h-full max-w-screen transition-all duration-300 flex flex-col justify-center items-center",
|
base: "fixed inset-0 z-50 w-full h-screen overflow-y-auto max-w-screen transition-all duration-300",
|
||||||
variants: {
|
variants: {
|
||||||
opened: {
|
opened: {
|
||||||
true: "pointer-events-auto opacity-100",
|
true: "pointer-events-auto opacity-100",
|
||||||
@ -72,9 +72,9 @@ const [dialogOverlayVariant, resolveDialogOverlayVariant] = vcn({
|
|||||||
lg: "backdrop-brightness-50",
|
lg: "backdrop-brightness-50",
|
||||||
},
|
},
|
||||||
padding: {
|
padding: {
|
||||||
sm: "p-4",
|
sm: "[&>div]:p-4",
|
||||||
md: "p-6",
|
md: "[&>div]:p-6",
|
||||||
lg: "p-8",
|
lg: "[&>div]:p-8",
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
defaults: {
|
defaults: {
|
||||||
@ -114,7 +114,14 @@ const DialogOverlay = React.forwardRef<HTMLDivElement, DialogOverlay>(
|
|||||||
onClick?.(e);
|
onClick?.(e);
|
||||||
}}
|
}}
|
||||||
>
|
>
|
||||||
|
<div
|
||||||
|
className={
|
||||||
|
"w-screen max-w-full min-h-screen flex flex-col justify-center items-center"
|
||||||
|
}
|
||||||
|
>
|
||||||
|
{/* Layer for overflow positioning */}
|
||||||
{children}
|
{children}
|
||||||
|
</div>
|
||||||
</div>,
|
</div>,
|
||||||
document.body,
|
document.body,
|
||||||
)}
|
)}
|
||||||
@ -170,23 +177,27 @@ const [dialogContentVariant, resolveDialogContentVariant] = vcn({
|
|||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
interface DialogContent
|
interface DialogContentProps
|
||||||
extends React.ComponentPropsWithoutRef<"div">,
|
extends React.ComponentPropsWithoutRef<"div">,
|
||||||
Omit<VariantProps<typeof dialogContentVariant>, "opened"> {}
|
Omit<VariantProps<typeof dialogContentVariant>, "opened"> {}
|
||||||
|
|
||||||
const DialogContent = React.forwardRef<HTMLDivElement, DialogContent>(
|
const DialogContent = React.forwardRef<HTMLDivElement, DialogContentProps>(
|
||||||
(props, ref) => {
|
(props, ref) => {
|
||||||
const [{ opened }] = useDialogContext();
|
const [{ opened }] = useDialogContext();
|
||||||
const [variantProps, otherPropsCompressed] = resolveDialogContentVariant({
|
const [variantProps, otherPropsCompressed] = resolveDialogContentVariant({
|
||||||
...props,
|
...props,
|
||||||
opened,
|
opened,
|
||||||
});
|
});
|
||||||
const { children, ...otherPropsExtracted } = otherPropsCompressed;
|
const { children, onClick, ...otherPropsExtracted } = otherPropsCompressed;
|
||||||
return (
|
return (
|
||||||
<div
|
<div
|
||||||
{...otherPropsExtracted}
|
{...otherPropsExtracted}
|
||||||
ref={ref}
|
ref={ref}
|
||||||
className={dialogContentVariant(variantProps)}
|
className={dialogContentVariant(variantProps)}
|
||||||
|
onClick={(e) => {
|
||||||
|
e.stopPropagation();
|
||||||
|
onClick?.(e);
|
||||||
|
}}
|
||||||
>
|
>
|
||||||
{children}
|
{children}
|
||||||
</div>
|
</div>
|
||||||
|
@ -1,4 +1,9 @@
|
|||||||
import { Dispatch, SetStateAction, useContext, createContext } from "react";
|
import {
|
||||||
|
type Dispatch,
|
||||||
|
type SetStateAction,
|
||||||
|
createContext,
|
||||||
|
useContext,
|
||||||
|
} from "react";
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* =========================
|
* =========================
|
||||||
|
@ -1,13 +1,13 @@
|
|||||||
|
import { type AsChild, Slot, type VariantProps, vcn } from "@pswui-lib";
|
||||||
import React, {
|
import React, {
|
||||||
ComponentPropsWithoutRef,
|
type ComponentPropsWithoutRef,
|
||||||
TouchEvent as ReactTouchEvent,
|
type TouchEvent as ReactTouchEvent,
|
||||||
forwardRef,
|
forwardRef,
|
||||||
useContext,
|
useContext,
|
||||||
useEffect,
|
useEffect,
|
||||||
useRef,
|
useRef,
|
||||||
useState,
|
useState,
|
||||||
} from "react";
|
} from "react";
|
||||||
import { AsChild, Slot, VariantProps, vcn } from "@pswui-lib";
|
|
||||||
import { createPortal } from "react-dom";
|
import { createPortal } from "react-dom";
|
||||||
|
|
||||||
interface IDrawerContext {
|
interface IDrawerContext {
|
||||||
@ -57,8 +57,7 @@ const DrawerRoot = ({ children, closeThreshold, opened }: DrawerRootProps) => {
|
|||||||
opened: opened ?? prev.opened,
|
opened: opened ?? prev.opened,
|
||||||
closeThreshold: closeThreshold ?? prev.closeThreshold,
|
closeThreshold: closeThreshold ?? prev.closeThreshold,
|
||||||
}));
|
}));
|
||||||
// eslint-disable-next-line react-hooks/exhaustive-deps
|
}, [closeThreshold, opened, setState]);
|
||||||
}, [closeThreshold, opened]);
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<DrawerContext.Provider value={state}>{children}</DrawerContext.Provider>
|
<DrawerContext.Provider value={state}>{children}</DrawerContext.Provider>
|
||||||
@ -302,8 +301,7 @@ const DrawerContent = forwardRef<HTMLDivElement, DrawerContentProps>(
|
|||||||
window.removeEventListener("touchmove", onMouseMove);
|
window.removeEventListener("touchmove", onMouseMove);
|
||||||
window.removeEventListener("touchend", onMouseUp);
|
window.removeEventListener("touchend", onMouseUp);
|
||||||
};
|
};
|
||||||
// eslint-disable-next-line react-hooks/exhaustive-deps
|
}, [state, setState, dragState, position]);
|
||||||
}, [state, dragState, position]);
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div
|
<div
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
|
import { type VariantProps, vcn } from "@pswui-lib";
|
||||||
import React from "react";
|
import React from "react";
|
||||||
import { VariantProps, vcn } from "@pswui-lib";
|
|
||||||
|
|
||||||
const inputColors = {
|
const inputColors = {
|
||||||
background: {
|
background: {
|
||||||
@ -92,7 +92,7 @@ const Input = React.forwardRef<HTMLInputElement, InputProps>((props, ref) => {
|
|||||||
const innerRef = React.useRef<HTMLInputElement | null>(null);
|
const innerRef = React.useRef<HTMLInputElement | null>(null);
|
||||||
|
|
||||||
React.useEffect(() => {
|
React.useEffect(() => {
|
||||||
if (innerRef && innerRef.current) {
|
if (innerRef?.current) {
|
||||||
innerRef.current.setCustomValidity(invalid ?? "");
|
innerRef.current.setCustomValidity(invalid ?? "");
|
||||||
}
|
}
|
||||||
}, [invalid]);
|
}, [invalid]);
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
|
import { type VariantProps, vcn } from "@pswui-lib";
|
||||||
import React from "react";
|
import React from "react";
|
||||||
import { VariantProps, vcn } from "@pswui-lib";
|
|
||||||
|
|
||||||
const [labelVariant, resolveLabelVariantProps] = vcn({
|
const [labelVariant, resolveLabelVariantProps] = vcn({
|
||||||
base: "has-[input[disabled]]:brightness-75 has-[input[disabled]]:cursor-not-allowed has-[input:invalid]:text-red-500",
|
base: "has-[input[disabled]]:brightness-75 has-[input[disabled]]:cursor-not-allowed has-[input:invalid]:text-red-500",
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
|
import { type VariantProps, vcn } from "@pswui-lib";
|
||||||
import React from "react";
|
import React from "react";
|
||||||
import { VariantProps, vcn } from "@pswui-lib";
|
|
||||||
|
|
||||||
const switchColors = {
|
const switchColors = {
|
||||||
background: {
|
background: {
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
import { AsChild, Slot, VariantProps, vcn } from "@pswui-lib";
|
import { type AsChild, Slot, type VariantProps, vcn } from "@pswui-lib";
|
||||||
import React from "react";
|
import React from "react";
|
||||||
|
|
||||||
import { TabContextBody, TabContext, Tab } from "./Context";
|
import { type Tab, TabContext, type TabContextBody } from "./Context";
|
||||||
|
|
||||||
interface TabProviderProps {
|
interface TabProviderProps {
|
||||||
defaultName: string;
|
defaultName: string;
|
||||||
@ -30,7 +30,12 @@ interface TabListProps
|
|||||||
const TabList = (props: TabListProps) => {
|
const TabList = (props: TabListProps) => {
|
||||||
const [variantProps, restProps] = resolveTabListVariantProps(props);
|
const [variantProps, restProps] = resolveTabListVariantProps(props);
|
||||||
|
|
||||||
return <div className={TabListVariant(variantProps)} {...restProps} />;
|
return (
|
||||||
|
<div
|
||||||
|
className={TabListVariant(variantProps)}
|
||||||
|
{...restProps}
|
||||||
|
/>
|
||||||
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
const [TabTriggerVariant, resolveTabTriggerVariantProps] = vcn({
|
const [TabTriggerVariant, resolveTabTriggerVariantProps] = vcn({
|
||||||
@ -76,7 +81,7 @@ const TabTrigger = (props: TabTriggerProps) => {
|
|||||||
});
|
});
|
||||||
};
|
};
|
||||||
// eslint-disable-next-line react-hooks/exhaustive-deps
|
// eslint-disable-next-line react-hooks/exhaustive-deps
|
||||||
}, [name]);
|
}, [name, setContext]);
|
||||||
|
|
||||||
const Comp = props.asChild ? Slot : "button";
|
const Comp = props.asChild ? Slot : "button";
|
||||||
|
|
||||||
@ -119,7 +124,10 @@ const TabContent = (props: TabContentProps) => {
|
|||||||
const { name, ...restProps } = restPropsBeforeParse;
|
const { name, ...restProps } = restPropsBeforeParse;
|
||||||
const [context] = React.useContext(TabContext);
|
const [context] = React.useContext(TabContext);
|
||||||
|
|
||||||
if (context.active[1] === name) {
|
if (context.active[1] !== name) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Slot
|
<Slot
|
||||||
className={tabContentVariant({
|
className={tabContentVariant({
|
||||||
@ -128,9 +136,6 @@ const TabContent = (props: TabContentProps) => {
|
|||||||
{...restProps}
|
{...restProps}
|
||||||
/>
|
/>
|
||||||
);
|
);
|
||||||
} else {
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
};
|
};
|
||||||
|
|
||||||
export { TabProvider, TabList, TabTrigger, TabContent };
|
export { TabProvider, TabList, TabTrigger, TabContent };
|
||||||
|
@ -1,20 +1,20 @@
|
|||||||
|
import { type VariantProps, vcn } from "@pswui-lib";
|
||||||
import React, { useEffect, useId, useRef } from "react";
|
import React, { useEffect, useId, useRef } from "react";
|
||||||
import ReactDOM from "react-dom";
|
import ReactDOM from "react-dom";
|
||||||
import { VariantProps, vcn } from "@pswui-lib";
|
|
||||||
|
|
||||||
import { toastVariant } from "./Variant";
|
|
||||||
import {
|
import {
|
||||||
ToastOption,
|
type ToastOption,
|
||||||
toasts,
|
|
||||||
subscribeSingle,
|
|
||||||
getSingleSnapshot,
|
|
||||||
notifySingle,
|
|
||||||
close,
|
close,
|
||||||
notify,
|
|
||||||
defaultToastOption,
|
defaultToastOption,
|
||||||
subscribe,
|
getSingleSnapshot,
|
||||||
getSnapshot,
|
getSnapshot,
|
||||||
|
notify,
|
||||||
|
notifySingle,
|
||||||
|
subscribe,
|
||||||
|
subscribeSingle,
|
||||||
|
toasts,
|
||||||
} from "./Store";
|
} from "./Store";
|
||||||
|
import { toastVariant } from "./Variant";
|
||||||
|
|
||||||
const ToastTemplate = ({
|
const ToastTemplate = ({
|
||||||
id,
|
id,
|
||||||
@ -74,7 +74,7 @@ const ToastTemplate = ({
|
|||||||
);
|
);
|
||||||
transitionDuration = style
|
transitionDuration = style
|
||||||
? {
|
? {
|
||||||
value: parseFloat(style[1] ?? "0"),
|
value: Number.parseFloat(style[1] ?? "0"),
|
||||||
unit: style[3] ?? style[2] ?? "s",
|
unit: style[3] ?? style[2] ?? "s",
|
||||||
}
|
}
|
||||||
: null;
|
: null;
|
||||||
@ -96,7 +96,7 @@ const ToastTemplate = ({
|
|||||||
}, calculatedTransitionDuration);
|
}, calculatedTransitionDuration);
|
||||||
return () => clearTimeout(timeout);
|
return () => clearTimeout(timeout);
|
||||||
}
|
}
|
||||||
}, [id, toastData.life, toastData.closeTimeout, toastData.closeButton]);
|
}, [id, toastData.life, toastData.closeTimeout]);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div
|
<div
|
||||||
@ -107,13 +107,18 @@ const ToastTemplate = ({
|
|||||||
ref={ref}
|
ref={ref}
|
||||||
>
|
>
|
||||||
{toastData.closeButton && (
|
{toastData.closeButton && (
|
||||||
<button className="absolute top-2 right-2" onClick={() => close(id)}>
|
<button
|
||||||
|
className="absolute top-2 right-2"
|
||||||
|
onClick={() => close(id)}
|
||||||
|
type={"button"}
|
||||||
|
>
|
||||||
<svg
|
<svg
|
||||||
xmlns="http://www.w3.org/2000/svg"
|
xmlns="http://www.w3.org/2000/svg"
|
||||||
width="1.2rem"
|
width="1.2rem"
|
||||||
height="1.2rem"
|
height="1.2rem"
|
||||||
viewBox="0 0 24 24"
|
viewBox="0 0 24 24"
|
||||||
>
|
>
|
||||||
|
<title>Close</title>
|
||||||
<path
|
<path
|
||||||
fill="currentColor"
|
fill="currentColor"
|
||||||
d="M19 6.41L17.59 5L12 10.59L6.41 5L5 6.41L10.59 12L5 17.59L6.41 19L12 13.41L17.59 19L19 17.59L13.41 12z"
|
d="M19 6.41L17.59 5L12 10.59L6.41 5L5 6.41L10.59 12L5 17.59L6.41 19L12 13.41L17.59 19L19 17.59L13.41 12z"
|
||||||
@ -167,7 +172,7 @@ const Toaster = React.forwardRef<HTMLDivElement, ToasterProps>((props, ref) => {
|
|||||||
if (toasterInstance && id !== toasterInstance.id) {
|
if (toasterInstance && id !== toasterInstance.id) {
|
||||||
if (process.env.NODE_ENV === "development" && !muteDuplicationWarning) {
|
if (process.env.NODE_ENV === "development" && !muteDuplicationWarning) {
|
||||||
console.warn(
|
console.warn(
|
||||||
`Multiple Toaster instances detected. Only one Toaster is allowed.`,
|
"Multiple Toaster instances detected. Only one Toaster is allowed.",
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
return null;
|
return null;
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
import { addToast, update, close } from "./Store";
|
import { addToast, close, update } from "./Store";
|
||||||
|
|
||||||
export function useToast() {
|
export function useToast() {
|
||||||
return {
|
return {
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
import { ToastBody } from "./Variant";
|
import type { ToastBody } from "./Variant";
|
||||||
|
|
||||||
export interface ToastOption {
|
export interface ToastOption {
|
||||||
closeButton: boolean;
|
closeButton: boolean;
|
||||||
@ -54,11 +54,11 @@ export function getSingleSnapshot(id: `${number}`) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
export function notify() {
|
export function notify() {
|
||||||
subscribers.forEach((subscriber) => subscriber());
|
for (const subscriber of subscribers) subscriber();
|
||||||
}
|
}
|
||||||
|
|
||||||
export function notifySingle(id: `${number}`) {
|
export function notifySingle(id: `${number}`) {
|
||||||
toasts[id].subscribers.forEach((subscriber) => subscriber());
|
for (const subscriber of toasts[id].subscribers) subscriber();
|
||||||
}
|
}
|
||||||
|
|
||||||
export function close(id: `${number}`) {
|
export function close(id: `${number}`) {
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
import { VariantProps, vcn } from "@pswui-lib";
|
import { type VariantProps, vcn } from "@pswui-lib";
|
||||||
|
|
||||||
const toastColors = {
|
const toastColors = {
|
||||||
background: "bg-white dark:bg-black",
|
background: "bg-white dark:bg-black",
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
|
import { type AsChild, Slot, type VariantProps, vcn } from "@pswui-lib";
|
||||||
import React, { useState } from "react";
|
import React, { useState } from "react";
|
||||||
import { AsChild, Slot, VariantProps, vcn } from "@pswui-lib";
|
|
||||||
|
|
||||||
interface TooltipContextBody {
|
interface TooltipContextBody {
|
||||||
position: "top" | "bottom" | "left" | "right";
|
position: "top" | "bottom" | "left" | "right";
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
import { twMerge } from "tailwind-merge";
|
|
||||||
import React from "react";
|
import React from "react";
|
||||||
|
import { twMerge } from "tailwind-merge";
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Merges the react props.
|
* Merges the react props.
|
||||||
@ -60,14 +60,15 @@ function mergeReactProps(
|
|||||||
* @returns The single ref.
|
* @returns The single ref.
|
||||||
*/
|
*/
|
||||||
function combinedRef<I>(refs: React.Ref<I | null>[]) {
|
function combinedRef<I>(refs: React.Ref<I | null>[]) {
|
||||||
return (instance: I | null) =>
|
return (instance: I | null) => {
|
||||||
refs.forEach((ref) => {
|
for (const ref of refs) {
|
||||||
if (ref instanceof Function) {
|
if (ref instanceof Function) {
|
||||||
ref(instance);
|
ref(instance);
|
||||||
} else if (ref) {
|
} else if (ref) {
|
||||||
(ref as React.MutableRefObject<I | null>).current = instance;
|
(ref as React.MutableRefObject<I | null>).current = instance;
|
||||||
}
|
}
|
||||||
});
|
}
|
||||||
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
interface SlotProps {
|
interface SlotProps {
|
||||||
|
@ -57,6 +57,31 @@ type VariantKV<V extends VariantType> = {
|
|||||||
[VariantKey in keyof V]: BooleanString<keyof V[VariantKey] & string>;
|
[VariantKey in keyof V]: BooleanString<keyof V[VariantKey] & string>;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Used for safely casting `Object.entries(<VariantKV>)`
|
||||||
|
*/
|
||||||
|
type VariantKVEntry<V extends VariantType> = [
|
||||||
|
keyof V,
|
||||||
|
BooleanString<keyof V[keyof V] & string>,
|
||||||
|
][];
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Takes VariantKV as parameter, return className string.
|
||||||
|
*
|
||||||
|
* @example
|
||||||
|
* vcn({
|
||||||
|
* /* ... *\/
|
||||||
|
* dynamics: [
|
||||||
|
* ({ a, b }) => {
|
||||||
|
* return a === "something" ? "asdf" : b
|
||||||
|
* },
|
||||||
|
* ]
|
||||||
|
* })
|
||||||
|
*/
|
||||||
|
type DynamicClassName<V extends VariantType> = (
|
||||||
|
variantProps: VariantKV<V>,
|
||||||
|
) => string;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Takes VariantType, and returns a type that represents the preset object.
|
* Takes VariantType, and returns a type that represents the preset object.
|
||||||
*
|
*
|
||||||
@ -94,6 +119,7 @@ export function vcn<V extends VariantType>(param: {
|
|||||||
*/
|
*/
|
||||||
base?: string | undefined;
|
base?: string | undefined;
|
||||||
variants: V;
|
variants: V;
|
||||||
|
dynamics?: DynamicClassName<V>[];
|
||||||
defaults: VariantKV<V>;
|
defaults: VariantKV<V>;
|
||||||
presets?: undefined;
|
presets?: undefined;
|
||||||
}): [
|
}): [
|
||||||
@ -108,7 +134,7 @@ export function vcn<V extends VariantType>(param: {
|
|||||||
/**
|
/**
|
||||||
* Any Props -> Variant Props, Other Props
|
* Any Props -> Variant Props, Other Props
|
||||||
*/
|
*/
|
||||||
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
// biome-ignore lint/suspicious/noExplicitAny: using unknown causes error `Index signature for type 'string' is missing in type --Props`.
|
||||||
<AnyPropBeforeResolve extends Record<string, any>>(
|
<AnyPropBeforeResolve extends Record<string, any>>(
|
||||||
anyProps: AnyPropBeforeResolve,
|
anyProps: AnyPropBeforeResolve,
|
||||||
) => [
|
) => [
|
||||||
@ -124,6 +150,7 @@ export function vcn<V extends VariantType, P extends PresetType<V>>(param: {
|
|||||||
*/
|
*/
|
||||||
base?: string | undefined;
|
base?: string | undefined;
|
||||||
variants: V /* VariantType */;
|
variants: V /* VariantType */;
|
||||||
|
dynamics?: DynamicClassName<V>[];
|
||||||
defaults: VariantKV<V>;
|
defaults: VariantKV<V>;
|
||||||
presets: P;
|
presets: P;
|
||||||
}): [
|
}): [
|
||||||
@ -139,7 +166,7 @@ export function vcn<V extends VariantType, P extends PresetType<V>>(param: {
|
|||||||
/**
|
/**
|
||||||
* Any Props -> Variant Props, Other Props
|
* Any Props -> Variant Props, Other Props
|
||||||
*/
|
*/
|
||||||
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
// biome-ignore lint/suspicious/noExplicitAny: using unknown causes error `Index signature for type 'string' is missing in type --Props`.
|
||||||
<AnyPropBeforeResolve extends Record<string, any>>(
|
<AnyPropBeforeResolve extends Record<string, any>>(
|
||||||
anyProps: AnyPropBeforeResolve,
|
anyProps: AnyPropBeforeResolve,
|
||||||
) => [
|
) => [
|
||||||
@ -159,19 +186,58 @@ export function vcn<
|
|||||||
>({
|
>({
|
||||||
base,
|
base,
|
||||||
variants,
|
variants,
|
||||||
|
dynamics = [],
|
||||||
defaults,
|
defaults,
|
||||||
presets,
|
presets,
|
||||||
}: {
|
}: {
|
||||||
base?: string | undefined;
|
base?: string | undefined;
|
||||||
variants: V;
|
variants: V;
|
||||||
|
dynamics?: DynamicClassName<V>[];
|
||||||
defaults: VariantKV<V>;
|
defaults: VariantKV<V>;
|
||||||
presets?: P;
|
presets?: P;
|
||||||
}) {
|
}) {
|
||||||
|
/**
|
||||||
|
* --Internal utility function--
|
||||||
|
* After transforming props to final version (which means "after overriding default, preset, and variant props sent via component props")
|
||||||
|
* It turns final version of variant props to className
|
||||||
|
*/
|
||||||
|
function __transformer__(
|
||||||
|
final: VariantKV<V>,
|
||||||
|
dynamics: string[],
|
||||||
|
propClassName?: string,
|
||||||
|
): string {
|
||||||
|
const classNames: string[] = [];
|
||||||
|
|
||||||
|
for (const [variantName, variantKey] of Object.entries(
|
||||||
|
final,
|
||||||
|
) as VariantKVEntry<V>) {
|
||||||
|
classNames.push(variants[variantName][variantKey.toString()]);
|
||||||
|
}
|
||||||
|
|
||||||
|
return twMerge(base, ...classNames, ...dynamics, propClassName);
|
||||||
|
}
|
||||||
|
|
||||||
return [
|
return [
|
||||||
/**
|
/**
|
||||||
* Takes any props (including className), and returns the class name.
|
* 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.
|
* If there is no variant specified in props, then it will fallback to preset, and then default.
|
||||||
*
|
*
|
||||||
|
*
|
||||||
|
* Process priority of variant will be:
|
||||||
|
*
|
||||||
|
* --- Processed as string
|
||||||
|
* 1. Base
|
||||||
|
*
|
||||||
|
* --- Processed as object (it will ignore rest of "not duplicated classname" in lower priority)
|
||||||
|
* 2. Default
|
||||||
|
* 3. Preset (overriding default)
|
||||||
|
* 4. Variant props via component (overriding preset)
|
||||||
|
*
|
||||||
|
* --- Processed as string
|
||||||
|
* 5. Dynamic classNames using variant props
|
||||||
|
* 6. User's className (overriding dynamic)
|
||||||
|
*
|
||||||
|
*
|
||||||
* @param variantProps - The variant props including className.
|
* @param variantProps - The variant props including className.
|
||||||
* @returns The class name.
|
* @returns The class name.
|
||||||
*/
|
*/
|
||||||
@ -180,42 +246,42 @@ export function vcn<
|
|||||||
VariantKV<V>
|
VariantKV<V>
|
||||||
>,
|
>,
|
||||||
) => {
|
) => {
|
||||||
const { className, preset, ...otherVariantProps } = variantProps;
|
const { className, preset, ..._otherVariantProps } = variantProps;
|
||||||
|
|
||||||
const currentPreset: P[keyof P] | null =
|
|
||||||
presets && preset ? (presets as NonNullable<P>)[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<string>(([variantKey, defaultValue]) => {
|
|
||||||
// Omit<Partial<VariantKV<V>> & { className; preset; }, className | preset> = Partial<VariantKV<V>> (safe to cast)
|
// Omit<Partial<VariantKV<V>> & { className; preset; }, className | preset> = Partial<VariantKV<V>> (safe to cast)
|
||||||
// Partial<VariantKV<V>>[keyof V] => { [k in keyof V]?: BooleanString<keyof V[keyof V] & string> } => BooleanString<keyof V[keyof V]>
|
// We all know `keyof V` = string, right? (but typescript says it's not, so.. attacking typescript with unknown lol)
|
||||||
|
const otherVariantProps = _otherVariantProps as unknown as Partial<
|
||||||
|
VariantKV<V>
|
||||||
|
>;
|
||||||
|
|
||||||
const directVariantValue: (keyof V[keyof V] & string) | undefined = (
|
const kv: VariantKV<V> = { ...defaults };
|
||||||
otherVariantProps as unknown as Partial<VariantKV<V>>
|
|
||||||
)[variantKey]?.toString?.(); // BooleanString<> -> string (safe to index V[keyof V])
|
|
||||||
|
|
||||||
const currentPresetVariantValue:
|
// Preset Processing
|
||||||
| (keyof V[keyof V] & string)
|
if (presets && preset && preset in presets) {
|
||||||
| undefined =
|
for (const [variantName, variantKey] of Object.entries(
|
||||||
!!currentPreset && presetVariantKeys.includes(variantKey)
|
// typescript bug (casting to NonNullable<P> required)
|
||||||
? (currentPreset as Partial<VariantKV<V>>)[
|
(presets as NonNullable<P>)[preset],
|
||||||
variantKey
|
) as VariantKVEntry<V>) {
|
||||||
]?.toString?.()
|
kv[variantName] = variantKey;
|
||||||
: undefined;
|
}
|
||||||
|
}
|
||||||
|
|
||||||
const variantValue: keyof V[keyof V] & string =
|
// VariantProps Processing
|
||||||
directVariantValue ?? currentPresetVariantValue ?? defaultValue;
|
for (const [variantName, variantKey] of Object.entries(
|
||||||
return variants[variantKey][variantValue];
|
otherVariantProps,
|
||||||
}),
|
) as VariantKVEntry<V>) {
|
||||||
(
|
kv[variantName] = variantKey;
|
||||||
currentPreset as Partial<VariantKV<V>> | null
|
}
|
||||||
)?.className?.toString?.(), // preset's classname comes after user's variant props? huh..
|
|
||||||
className,
|
// make dynamics result
|
||||||
);
|
const dynamicClasses: string[] = [];
|
||||||
|
for (const dynamicFunction of dynamics) {
|
||||||
|
dynamicClasses.push(dynamicFunction(kv));
|
||||||
|
}
|
||||||
|
|
||||||
|
return __transformer__(kv, dynamicClasses, className);
|
||||||
},
|
},
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Takes any props, parse variant props and other props.
|
* Takes any props, parse variant props and other props.
|
||||||
* If `options.excludeA` is true, then it will parse `A` as "other" props.
|
* If `options.excludeA` is true, then it will parse `A` as "other" props.
|
||||||
|
Loading…
x
Reference in New Issue
Block a user