diff --git a/src/pswui/components/Button.tsx b/src/pswui/components/Button.tsx new file mode 100644 index 0000000..586027b --- /dev/null +++ b/src/pswui/components/Button.tsx @@ -0,0 +1,124 @@ +import React from "react"; +import { vcn, VariantProps, Slot, AsChild } from "@pswui-lib"; + +const colors = { + outline: { + focus: "dark:focus-visible:outline-white/20 focus-visible:outline-black/10", + }, + border: { + default: "border-neutral-300 dark:border-neutral-700", + success: "border-green-400 dark:border-green-600", + warning: "border-yellow-400 dark:border-yellow-600", + danger: "border-red-400 dark:border-red-600", + }, + background: { + default: + "bg-white dark:bg-black hover:bg-neutral-200 dark:hover:bg-neutral-800", + ghost: + "bg-black/0 dark:bg-white/0 hover:bg-black/20 dark:hover:bg-white/20", + success: + "bg-green-100 dark:bg-green-900 hover:bg-green-200 dark:hover:bg-green-800", + warning: + "bg-yellow-100 dark:bg-yellow-900 hover:bg-yellow-200 dark:hover:bg-yellow-800", + danger: "bg-red-100 dark:bg-red-900 hover:bg-red-200 dark:hover:bg-red-800", + }, + underline: "decoration-current", +}; + +const [buttonVariants, resolveVariants] = vcn({ + base: `w-fit flex flex-row items-center justify-between rounded-md outline outline-1 outline-transparent outline-offset-2 ${colors.outline.focus} transition-all`, + variants: { + size: { + link: "p-0 text-base", + sm: "px-2 py-1 text-sm", + md: "px-4 py-2 text-base", + lg: "px-5 py-3 text-lg", + icon: "p-2 text-base", + }, + border: { + none: "border-0", + solid: `border ${colors.border.default}`, + success: `border ${colors.border.success}`, + warning: `border ${colors.border.warning}`, + danger: `border ${colors.border.danger}`, + }, + background: { + default: colors.background.default, + ghost: colors.background.ghost, + success: colors.background.success, + warning: colors.background.warning, + danger: colors.background.danger, + transparent: "bg-transparent hover:bg-transparent", + }, + decoration: { + none: "no-underline", + link: `underline decoration-1 underline-offset-2 hover:underline-offset-4 ${colors.underline}`, + }, + }, + defaults: { + size: "md", + border: "solid", + background: "default", + decoration: "none", + }, + presets: { + default: { + border: "solid", + background: "default", + decoration: "none", + size: "md", + }, + ghost: { + border: "none", + background: "ghost", + decoration: "none", + size: "md", + }, + link: { + border: "none", + background: "transparent", + decoration: "link", + size: "link", + }, + success: { + border: "success", + background: "success", + decoration: "none", + size: "md", + }, + warning: { + border: "warning", + background: "warning", + decoration: "none", + size: "md", + }, + danger: { + border: "danger", + background: "danger", + decoration: "none", + size: "md", + }, + }, +}); + +export interface ButtonProps + extends Omit, "className">, + VariantProps, + AsChild {} + +const Button = React.forwardRef( + (props, ref) => { + const [variantProps, otherPropsCompressed] = resolveVariants(props); + const { asChild, ...otherPropsExtracted } = otherPropsCompressed; + + const Comp = asChild ? Slot : "button"; + const compProps = { + ...otherPropsExtracted, + className: buttonVariants(variantProps), + }; + + return ; + }, +); + +export { Button }; diff --git a/src/pswui/components/Checkbox.tsx b/src/pswui/components/Checkbox.tsx new file mode 100644 index 0000000..b37b5ff --- /dev/null +++ b/src/pswui/components/Checkbox.tsx @@ -0,0 +1,113 @@ +import React from "react"; +import { VariantProps, vcn } from "@pswui-lib"; + +const checkboxColors = { + background: { + default: "bg-neutral-200 dark:bg-neutral-700", + hover: "hover:bg-neutral-300 dark:hover:bg-neutral-600", + checked: + "has-[input[type=checkbox]:checked]:bg-neutral-300 dark:has-[input[type=checkbox]:checked]:bg-neutral-400", + checkedHover: + "has-[input[type=checkbox]:checked]:hover:bg-neutral-400 dark:has-[input[type=checkbox]:checked]:hover:bg-neutral-300", + disabled: + 'has-[input[type="checkbox"]:disabled]:bg-neutral-100 dark:has-[input[type="checkbox"]:disabled]:bg-neutral-800', + disabledHover: + "has-[input[type='checkbox']:disabled]:hover:bg-neutral-100 dark:has-[input[type='checkbox']:disabled]:hover:bg-neutral-800", + disabledChecked: + "has-[input[type='checkbox']:disabled:checked]:bg-neutral-300 dark:has-[input[type='checkbox']:disabled:checked]:bg-neutral-700", + disabledCheckedHover: + "has-[input[type='checkbox']:disabled:checked]:hover:bg-neutral-300 dark:has-[input[type='checkbox']:disabled:checked]:hover:bg-neutral-700", + }, + checkmark: + "text-black dark:text-white has-[input[type=checkbox]:disabled]:text-neutral-400 dark:has-[input[type=checkbox]:disabled]:text-neutral-500", +}; + +const [checkboxVariant, resolveCheckboxVariantProps] = vcn({ + base: `inline-block rounded-md ${checkboxColors.checkmark} ${checkboxColors.background.disabled} ${checkboxColors.background.default} ${checkboxColors.background.hover} ${checkboxColors.background.checked} ${checkboxColors.background.checkedHover} ${checkboxColors.background.disabledChecked} ${checkboxColors.background.disabledCheckedHover} has-[input[type="checkbox"]:disabled]:cursor-not-allowed transition-colors duration-75 ease-in-out`, + variants: { + size: { + base: "size-[1em] p-0 [&>svg]:size-[1em]", + md: "size-[1.5em] p-0.5 [&>svg]:size-[1.25em]", + lg: "size-[1.75em] p-1 [&>svg]:size-[1.25em]", + }, + }, + defaults: { + size: "md", + }, +}); + +interface CheckboxProps + extends VariantProps, + Omit< + React.ComponentPropsWithoutRef<"input">, + "type" | "className" | "size" + > {} + +const Checkbox = React.forwardRef( + (props, ref) => { + const [variantProps, otherPropsCompressed] = + resolveCheckboxVariantProps(props); + const { + defaultChecked, + checked: propChecked, + onChange, + ...otherPropsExtracted + } = otherPropsCompressed; + + // internally handles checked, so we can use checked value without prop + const [checked, setChecked] = React.useState(defaultChecked ?? false); + React.useEffect(() => { + if (typeof propChecked === "boolean") { + setChecked(propChecked); + } + }, [propChecked]); + + const internalRef = React.useRef(null); + + return ( + <> + + + ); + }, +); + +export { Checkbox }; diff --git a/src/pswui/components/Dialog.tsx b/src/pswui/components/Dialog.tsx new file mode 100644 index 0000000..704db6f --- /dev/null +++ b/src/pswui/components/Dialog.tsx @@ -0,0 +1,424 @@ +import React, { Dispatch, SetStateAction, useState } from "react"; +import { Slot, VariantProps, vcn } from "@pswui-lib"; +import ReactDOM from "react-dom"; + +/** + * ========================= + * DialogContext + * ========================= + */ + +interface DialogContext { + opened: boolean; +} + +const initialDialogContext: DialogContext = { opened: false }; +const DialogContext = React.createContext< + [DialogContext, Dispatch>] +>([ + initialDialogContext, + () => { + if (process.env.NODE_ENV && process.env.NODE_ENV === "development") { + console.warn( + "It seems like you're using DialogContext outside of a provider.", + ); + } + }, +]); + +const useDialogContext = () => React.useContext(DialogContext); + +/** + * ========================= + * DialogRoot + * ========================= + */ + +interface DialogRootProps { + children: React.ReactNode; +} + +const DialogRoot = ({ children }: DialogRootProps) => { + const state = useState(initialDialogContext); + return ( + {children} + ); +}; + +/** + * ========================= + * DialogTrigger + * ========================= + */ + +interface DialogTriggerProps { + children: React.ReactNode; +} + +const DialogTrigger = ({ children }: DialogTriggerProps) => { + const [_, setState] = useDialogContext(); + const onClick = () => setState((p) => ({ ...p, opened: true })); + + const slotProps = { + onClick, + children, + }; + + return ; +}; + +/** + * ========================= + * DialogOverlay + * ========================= + */ + +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", + variants: { + opened: { + true: "pointer-events-auto opacity-100", + false: "pointer-events-none opacity-0", + }, + blur: { + sm: "backdrop-blur-sm", + md: "backdrop-blur-md", + lg: "backdrop-blur-lg", + }, + darken: { + sm: "backdrop-brightness-90", + md: "backdrop-brightness-75", + lg: "backdrop-brightness-50", + }, + padding: { + sm: "p-4", + md: "p-6", + lg: "p-8", + }, + }, + defaults: { + opened: false, + blur: "md", + darken: "md", + padding: "md", + }, +}); + +interface DialogOverlay + extends React.ComponentPropsWithoutRef<"div">, + Omit, "opened"> { + closeOnClick?: boolean; +} + +const DialogOverlay = React.forwardRef( + (props, ref) => { + const [{ opened }, setContext] = useDialogContext(); + const [variantProps, otherPropsCompressed] = resolveDialogOverlayVariant({ + ...props, + opened, + }); + const { children, closeOnClick, onClick, ...otherPropsExtracted } = + otherPropsCompressed; + return ( + <> + {ReactDOM.createPortal( +
{ + if (closeOnClick) { + setContext((p) => ({ ...p, opened: false })); + } + onClick?.(e); + }} + > + {children} +
, + document.body, + )} + + ); + }, +); + +/** + * ========================= + * DialogContent + * ========================= + */ + +const [dialogContentVariant, resolveDialogContentVariant] = vcn({ + base: "transition-transform duration-300 bg-white dark:bg-black border border-neutral-200 dark:border-neutral-800", + variants: { + opened: { + true: "scale-100", + false: "scale-50", + }, + size: { + fit: "w-fit", + fullSm: "w-full max-w-sm", + fullMd: "w-full max-w-md", + fullLg: "w-full max-w-lg", + fullXl: "w-full max-w-xl", + full2xl: "w-full max-w-2xl", + }, + rounded: { + sm: "rounded-sm", + md: "rounded-md", + lg: "rounded-lg", + xl: "rounded-xl", + }, + padding: { + sm: "p-4", + md: "p-6", + lg: "p-8", + }, + gap: { + sm: "space-y-4", + md: "space-y-6", + lg: "space-y-8", + }, + }, + defaults: { + opened: false, + size: "fit", + rounded: "md", + padding: "md", + gap: "md", + }, +}); + +interface DialogContent + extends React.ComponentPropsWithoutRef<"div">, + Omit, "opened"> {} + +const DialogContent = React.forwardRef( + (props, ref) => { + const [{ opened }] = useDialogContext(); + const [variantProps, otherPropsCompressed] = resolveDialogContentVariant({ + ...props, + opened, + }); + const { children, ...otherPropsExtracted } = otherPropsCompressed; + return ( +
+ {children} +
+ ); + }, +); + +/** + * ========================= + * DialogClose + * ========================= + */ + +interface DialogCloseProps { + children: React.ReactNode; +} + +const DialogClose = ({ children }: DialogCloseProps) => { + const [_, setState] = useDialogContext(); + const onClick = () => setState((p) => ({ ...p, opened: false })); + + const slotProps = { + onClick, + children, + }; + + return ; +}; + +/** + * ========================= + * DialogHeader + * ========================= + */ + +const [dialogHeaderVariant, resolveDialogHeaderVariant] = vcn({ + base: "flex flex-col", + variants: { + gap: { + sm: "gap-2", + md: "gap-4", + lg: "gap-6", + }, + }, + defaults: { + gap: "sm", + }, +}); + +interface DialogHeaderProps + extends React.ComponentPropsWithoutRef<"header">, + VariantProps {} + +const DialogHeader = React.forwardRef( + (props, ref) => { + const [variantProps, otherPropsCompressed] = + resolveDialogHeaderVariant(props); + const { children, ...otherPropsExtracted } = otherPropsCompressed; + return ( +
+ {children} +
+ ); + }, +); + +/** + * ========================= + * DialogTitle / DialogSubtitle + * ========================= + */ + +const [dialogTitleVariant, resolveDialogTitleVariant] = vcn({ + variants: { + size: { + sm: "text-lg", + md: "text-xl", + lg: "text-2xl", + }, + weight: { + sm: "font-medium", + md: "font-semibold", + lg: "font-bold", + }, + }, + defaults: { + size: "md", + weight: "lg", + }, +}); + +interface DialogTitleProps + extends React.ComponentPropsWithoutRef<"h1">, + VariantProps {} + +const [dialogSubtitleVariant, resolveDialogSubtitleVariant] = vcn({ + variants: { + size: { + sm: "text-sm", + md: "text-base", + lg: "text-lg", + }, + opacity: { + sm: "opacity-60", + md: "opacity-70", + lg: "opacity-80", + }, + weight: { + sm: "font-light", + md: "font-normal", + lg: "font-medium", + }, + }, + defaults: { + size: "sm", + opacity: "sm", + weight: "md", + }, +}); + +interface DialogSubtitleProps + extends React.ComponentPropsWithoutRef<"h2">, + VariantProps {} + +const DialogTitle = React.forwardRef( + (props, ref) => { + const [variantProps, otherPropsCompressed] = + resolveDialogTitleVariant(props); + const { children, ...otherPropsExtracted } = otherPropsCompressed; + return ( +

+ {children} +

+ ); + }, +); + +const DialogSubtitle = React.forwardRef< + HTMLHeadingElement, + DialogSubtitleProps +>((props, ref) => { + const [variantProps, otherPropsCompressed] = + resolveDialogSubtitleVariant(props); + const { children, ...otherPropsExtracted } = otherPropsCompressed; + return ( +

+ {children} +

+ ); +}); + +/** + * ========================= + * DialogFooter + * ========================= + */ + +const [dialogFooterVariant, resolveDialogFooterVariant] = vcn({ + base: "flex flex-col items-end sm:flex-row sm:items-center sm:justify-end", + variants: { + gap: { + sm: "gap-2", + md: "gap-4", + lg: "gap-6", + }, + }, + defaults: { + gap: "md", + }, +}); + +interface DialogFooterProps + extends React.ComponentPropsWithoutRef<"footer">, + VariantProps {} + +const DialogFooter = React.forwardRef( + (props, ref) => { + const [variantProps, otherPropsCompressed] = + resolveDialogFooterVariant(props); + const { children, ...otherPropsExtracted } = otherPropsCompressed; + return ( +
+ {children} +
+ ); + }, +); + +export { + useDialogContext, + DialogRoot, + DialogTrigger, + DialogOverlay, + DialogContent, + DialogClose, + DialogHeader, + DialogTitle, + DialogSubtitle, + DialogFooter, +}; diff --git a/src/pswui/components/Drawer.tsx b/src/pswui/components/Drawer.tsx new file mode 100644 index 0000000..fbaa187 --- /dev/null +++ b/src/pswui/components/Drawer.tsx @@ -0,0 +1,476 @@ +import React, { + ComponentPropsWithoutRef, + TouchEvent as ReactTouchEvent, + forwardRef, + useContext, + useEffect, + useRef, + useState, +} from "react"; +import { AsChild, Slot, VariantProps, vcn } from "@pswui-lib"; +import { createPortal } from "react-dom"; + +interface IDrawerContext { + opened: boolean; + closeThreshold: number; + movePercentage: number; + isDragging: boolean; + leaveWhileDragging: boolean; +} +const DrawerContextInitial: IDrawerContext = { + opened: false, + closeThreshold: 0.3, + movePercentage: 0, + isDragging: false, + leaveWhileDragging: false, +}; +const DrawerContext = React.createContext< + [IDrawerContext, React.Dispatch>] +>([ + DrawerContextInitial, + () => { + if (process.env.NODE_ENV && process.env.NODE_ENV === "development") { + console.warn( + "It seems like you're using DrawerContext outside of a provider.", + ); + } + }, +]); + +interface DrawerRootProps { + children: React.ReactNode; + closeThreshold?: number; + opened?: boolean; +} + +const DrawerRoot = ({ children, closeThreshold, opened }: DrawerRootProps) => { + const state = useState({ + ...DrawerContextInitial, + opened: opened ?? DrawerContextInitial.opened, + closeThreshold: closeThreshold ?? DrawerContextInitial.closeThreshold, + }); + + useEffect(() => { + state[1]((prev) => ({ + ...prev, + opened: opened ?? prev.opened, + closeThreshold: closeThreshold ?? prev.closeThreshold, + })); + }, [closeThreshold, opened]); + + return ( + {children} + ); +}; + +const DrawerTrigger = ({ children }: { children: React.ReactNode }) => { + const [_, setState] = useContext(DrawerContext); + + function onClick() { + setState((prev) => ({ ...prev, opened: true })); + } + + return {children}; +}; + +const [drawerOverlayVariant, resolveDrawerOverlayVariantProps] = vcn({ + base: "fixed inset-0 transition-[backdrop-filter] duration-75", + variants: { + opened: { + true: "pointer-events-auto select-auto", + false: "pointer-events-none select-none", + }, + }, + defaults: { + opened: false, + }, +}); + +const DRAWER_OVERLAY_BACKDROP_FILTER_BRIGHTNESS = 0.3; + +interface DrawerOverlayProps + extends Omit, "opened">, + AsChild, + ComponentPropsWithoutRef<"div"> {} + +const DrawerOverlay = forwardRef( + (props, ref) => { + const [state, setState] = useContext(DrawerContext); + + const [variantProps, restPropsCompressed] = + resolveDrawerOverlayVariantProps(props); + const { asChild, ...restPropsExtracted } = restPropsCompressed; + + function onOutsideClick() { + if (state.leaveWhileDragging) { + setState((prev) => ({ ...prev, leaveWhileDragging: false })); + return; + } + setState((prev) => ({ ...prev, opened: false })); + } + + const Comp = asChild ? Slot : "div"; + const backdropFilter = `brightness(${ + state.isDragging + ? state.movePercentage + DRAWER_OVERLAY_BACKDROP_FILTER_BRIGHTNESS + : state.opened + ? DRAWER_OVERLAY_BACKDROP_FILTER_BRIGHTNESS + : 1 + })`; + + return createPortal( + , + document.body, + ); + }, +); + +const drawerContentColors = { + background: "bg-white dark:bg-black", + border: "border-neutral-200 dark:border-neutral-800", +}; + +const [drawerContentVariant, resolveDrawerContentVariantProps] = vcn({ + base: `fixed ${drawerContentColors.background} ${drawerContentColors.border} transition-all p-4 flex flex-col justify-between gap-8`, + variants: { + position: { + top: "top-0 inset-x-0 w-full max-w-screen rounded-t-lg border-b-2", + bottom: "bottom-0 inset-x-0 w-full max-w-screen rounded-b-lg border-t-2", + left: "left-0 inset-y-0 h-screen rounded-l-lg border-r-2", + right: "right-0 inset-y-0 h-screen rounded-r-lg border-l-2", + }, + opened: { + true: "touch-none", + false: + "[&.top-0]:-translate-y-full [&.bottom-0]:translate-y-full [&.left-0]:-translate-x-full [&.right-0]:translate-x-full", + }, + }, + defaults: { + position: "left", + opened: false, + }, +}); + +interface DrawerContentProps + extends Omit, "opened">, + AsChild, + ComponentPropsWithoutRef<"div"> {} + +const DrawerContent = forwardRef( + (props, ref) => { + const [state, setState] = useContext(DrawerContext); + const [dragState, setDragState] = useState({ + isDragging: false, + prevTouch: { x: 0, y: 0 }, + delta: 0, + }); + + const [variantProps, restPropsCompressed] = + resolveDrawerContentVariantProps(props); + const { position = "left" } = variantProps; + const { asChild, onClick, ...restPropsExtracted } = restPropsCompressed; + + const Comp = asChild ? Slot : "div"; + + const internalRef = useRef(null); + + function onMouseDown() { + setState((prev) => ({ ...prev, isDragging: true })); + setDragState({ + isDragging: true, + delta: 0, + prevTouch: { x: 0, y: 0 }, + }); + } + + function onTouchStart(e: ReactTouchEvent) { + setState((prev) => ({ ...prev, isDragging: true })); + setDragState({ + isDragging: true, + delta: 0, + prevTouch: { x: e.touches[0].pageX, y: e.touches[0].pageY }, + }); + } + + useEffect(() => { + function onMouseUp(e: TouchEvent): void; + function onMouseUp(e: MouseEvent): void; + function onMouseUp(e: TouchEvent | MouseEvent) { + if ( + e.target instanceof Element && + internalRef.current && + internalRef.current.contains(e.target) + ) { + const size = ["top", "bottom"].includes(position) + ? e.target.getBoundingClientRect().height + : e.target.getBoundingClientRect().width; + setState((prev) => ({ + ...prev, + isDragging: false, + opened: + Math.abs(dragState.delta) > state.closeThreshold * size + ? false + : prev.opened, + movePercentage: 0, + })); + } else { + setState((prev) => ({ + ...prev, + isDragging: false, + movePercentage: 0, + })); + } + setDragState({ + isDragging: false, + delta: 0, + prevTouch: { x: 0, y: 0 }, + }); + } + + function onMouseMove(e: TouchEvent): void; + function onMouseMove(e: MouseEvent): void; + function onMouseMove(e: MouseEvent | TouchEvent) { + if (dragState.isDragging) { + setDragState((prev) => { + let movement = ["top", "bottom"].includes(position) + ? "movementY" in e + ? e.movementY + : e.touches[0].pageY - prev.prevTouch.y + : "movementX" in e + ? e.movementX + : e.touches[0].pageX - prev.prevTouch.x; + if ( + (["top", "left"].includes(position) && + dragState.delta >= 0 && + movement > 0) || + (["bottom", "right"].includes(position) && + dragState.delta <= 0 && + movement < 0) + ) { + movement = + movement / + Math.abs(dragState.delta === 0 ? 1 : dragState.delta); + } + return { + ...prev, + delta: prev.delta + movement, + ...("touches" in e + ? { + prevTouch: { x: e.touches[0].pageX, y: e.touches[0].pageY }, + } + : {}), + }; + }); + + if (internalRef.current) { + const size = ["top", "bottom"].includes(position) + ? internalRef.current.getBoundingClientRect().height + : internalRef.current.getBoundingClientRect().width; + const movePercentage = dragState.delta / size; + setState((prev) => ({ + ...prev, + movePercentage: ["top", "left"].includes(position) + ? -movePercentage + : movePercentage, + })); + } + } + } + + window.addEventListener("mousemove", onMouseMove); + window.addEventListener("mouseup", onMouseUp); + window.addEventListener("touchmove", onMouseMove); + window.addEventListener("touchend", onMouseUp); + return () => { + window.removeEventListener("mousemove", onMouseMove); + window.removeEventListener("mouseup", onMouseUp); + window.removeEventListener("touchmove", onMouseMove); + window.removeEventListener("touchend", onMouseUp); + }; + }, [state, dragState]); + + return ( +
+ { + internalRef.current = el; + if (typeof ref === "function") { + ref(el); + } else if (ref) { + ref.current = el; + } + }} + onClick={(e) => { + e.stopPropagation(); + onClick?.(e); + }} + onMouseDown={onMouseDown} + onMouseLeave={() => + dragState.isDragging && + setState((prev) => ({ ...prev, leaveWhileDragging: true })) + } + onMouseEnter={() => + dragState.isDragging && + setState((prev) => ({ ...prev, leaveWhileDragging: false })) + } + onTouchStart={onTouchStart} + /> +
+ ); + }, +); + +const DrawerClose = forwardRef< + HTMLButtonElement, + ComponentPropsWithoutRef<"button"> +>((props, ref) => { + const [_, setState] = useContext(DrawerContext); + return ( + setState((prev) => ({ ...prev, opened: false }))} + /> + ); +}); + +const [drawerHeaderVariant, resolveDrawerHeaderVariantProps] = vcn({ + base: "flex flex-col gap-2", + variants: {}, + defaults: {}, +}); + +interface DrawerHeaderProps + extends ComponentPropsWithoutRef<"div">, + VariantProps, + AsChild {} + +const DrawerHeader = forwardRef( + (props, ref) => { + const [variantProps, restPropsCompressed] = + resolveDrawerHeaderVariantProps(props); + const { asChild, ...restPropsExtracted } = restPropsCompressed; + return ( +
+ ); + }, +); + +const [drawerBodyVariant, resolveDrawerBodyVariantProps] = vcn({ + base: "flex-grow", + variants: {}, + defaults: {}, +}); + +interface DrawerBodyProps + extends ComponentPropsWithoutRef<"div">, + VariantProps, + AsChild {} + +const DrawerBody = forwardRef((props, ref) => { + const [variantProps, restPropsCompressed] = + resolveDrawerBodyVariantProps(props); + const { asChild, ...restPropsExtracted } = restPropsCompressed; + return ( +
+ ); +}); + +const [drawerFooterVariant, resolveDrawerFooterVariantProps] = vcn({ + base: "flex flex-row justify-end gap-2", + variants: {}, + defaults: {}, +}); + +interface DrawerFooterProps + extends ComponentPropsWithoutRef<"div">, + VariantProps, + AsChild {} + +const DrawerFooter = forwardRef( + (props, ref) => { + const [variantProps, restPropsCompressed] = + resolveDrawerFooterVariantProps(props); + const { asChild, ...restPropsExtracted } = restPropsCompressed; + return ( +
+ ); + }, +); + +export { + DrawerRoot, + DrawerTrigger, + DrawerOverlay, + DrawerContent, + DrawerClose, + DrawerHeader, + DrawerBody, + DrawerFooter, +}; diff --git a/src/pswui/components/Input.tsx b/src/pswui/components/Input.tsx new file mode 100644 index 0000000..daf50e9 --- /dev/null +++ b/src/pswui/components/Input.tsx @@ -0,0 +1,117 @@ +import React from "react"; +import { VariantProps, vcn } from "@pswui-lib"; + +const inputColors = { + background: { + default: "bg-neutral-50 dark:bg-neutral-900", + hover: "hover:bg-neutral-100 dark:hover:bg-neutral-800", + invalid: + "invalid:bg-red-100 invalid:dark:bg-red-900 has-[input:invalid]:bg-red-100 dark:has-[input:invalid]:bg-red-900", + invalidHover: + "hover:invalid:bg-red-200 dark:hover:invalid:bg-red-800 has-[input:invalid:hover]:bg-red-200 dark:has-[input:invalid:hover]:bg-red-800", + }, + border: { + default: "border-neutral-400 dark:border-neutral-600", + invalid: + "invalid:border-red-400 invalid:dark:border-red-600 has-[input:invalid]:border-red-400 dark:has-[input:invalid]:border-red-600", + }, + ring: { + default: "ring-transparent focus-within:ring-current", + invalid: + "invalid:focus-within:ring-red-400 invalid:focus-within:dark:ring-red-600 has-[input:invalid]:focus-within:ring-red-400 dark:has-[input:invalid]:focus-within:ring-red-600", + }, +}; + +const [inputVariant, resolveInputVariantProps] = vcn({ + base: `rounded-md p-2 border ring-1 outline-none transition-all duration-200 [appearance:textfield] disabled:brightness-50 disabled:saturate-0 disabled:cursor-not-allowed ${inputColors.background.default} ${inputColors.background.hover} ${inputColors.border.default} ${inputColors.ring.default} ${inputColors.background.invalid} ${inputColors.background.invalidHover} ${inputColors.border.invalid} ${inputColors.ring.invalid} [&:has(input)]:flex [&:has(input)]:w-fit`, + variants: { + unstyled: { + true: "bg-transparent border-none p-0 ring-0 hover:bg-transparent invalid:hover:bg-transparent invalid:focus-within:bg-transparent invalid:focus-within:ring-0", + false: "", + }, + full: { + true: "w-full", + false: "w-fit", + }, + }, + defaults: { + unstyled: false, + full: false, + }, +}); + +interface InputFrameProps + extends VariantProps, + React.ComponentPropsWithoutRef<"label"> { + children?: React.ReactNode; +} + +const InputFrame = React.forwardRef( + (props, ref) => { + const [variantProps, otherPropsCompressed] = + resolveInputVariantProps(props); + const { children, ...otherPropsExtracted } = otherPropsCompressed; + + return ( + + ); + }, +); + +interface InputProps + extends VariantProps, + React.ComponentPropsWithoutRef<"input"> { + type: Exclude< + React.InputHTMLAttributes["type"], + | "button" + | "checkbox" + | "color" + | "date" + | "datetime-local" + | "file" + | "radio" + | "range" + | "reset" + | "image" + | "submit" + | "time" + >; + invalid?: string; +} + +const Input = React.forwardRef((props, ref) => { + const [variantProps, otherPropsCompressed] = resolveInputVariantProps(props); + const { type, invalid, ...otherPropsExtracted } = otherPropsCompressed; + + const innerRef = React.useRef(null); + + React.useEffect(() => { + if (innerRef && innerRef.current) { + innerRef.current.setCustomValidity(invalid ?? ""); + } + }, [invalid]); + + return ( + { + innerRef.current = el; + if (typeof ref === "function") { + ref(el); + } else if (ref) { + ref.current = el; + } + }} + className={inputVariant(variantProps)} + {...otherPropsExtracted} + /> + ); +}); + +export { InputFrame, Input }; diff --git a/src/pswui/components/Label.tsx b/src/pswui/components/Label.tsx new file mode 100644 index 0000000..d2ff753 --- /dev/null +++ b/src/pswui/components/Label.tsx @@ -0,0 +1,33 @@ +import React from "react"; +import { VariantProps, vcn } from "@pswui-lib"; + +const [labelVariant, resolveLabelVariantProps] = vcn({ + base: "has-[input[disabled]]:brightness-75 has-[input[disabled]]:cursor-not-allowed has-[input:invalid]:text-red-500", + variants: { + direction: { + vertical: "flex flex-col gap-2 justify-center items-start", + horizontal: "flex flex-row gap-2 justify-start items-center", + }, + }, + defaults: { + direction: "vertical", + }, +}); + +interface LabelProps + extends VariantProps, + React.ComponentPropsWithoutRef<"label"> {} + +const Label = React.forwardRef((props, ref) => { + const [variantProps, otherPropsCompressed] = resolveLabelVariantProps(props); + + return ( +