import { type AsChild, Slot, type VariantProps, vcn } from "@pswui-lib"; import { type ComponentPropsWithoutRef, createContext, forwardRef, useContext, useEffect, useRef, } from "react"; /** Form Item Context **/ interface IFormItemContext { invalid?: string | null | undefined; } const FormItemContext = createContext({}); /** FormItem **/ const [formItemVariant, resolveFormItemVariantProps] = vcn({ base: "flex flex-col gap-2 items-start w-full", variants: {}, defaults: {}, }); interface FormItemProps extends VariantProps, AsChild, ComponentPropsWithoutRef<"label"> { invalid?: string | null | undefined; } const FormItem = forwardRef((props, ref) => { const [variantProps, restPropsCompressed] = resolveFormItemVariantProps(props); const { asChild, children, invalid, ...restPropsExtracted } = restPropsCompressed; const innerRef = useRef(null); useEffect(() => { const invalidAsString = invalid ? invalid : ""; const input = innerRef.current?.querySelector?.("input"); if (!input) return; input.setCustomValidity(invalidAsString); }, [invalid]); const Comp = asChild ? Slot : "label"; return ( { innerRef.current = el; if (typeof ref === "function") { ref(el); } else if (ref) { ref.current = el; } }} className={formItemVariant(variantProps)} {...restPropsExtracted} > {children} ); }); FormItem.displayName = "FormItem"; /** FormLabel **/ const [formLabelVariant, resolveFormLabelVariantProps] = vcn({ base: "text-sm font-bold", variants: {}, defaults: {}, }); interface FormLabelProps extends VariantProps, AsChild, ComponentPropsWithoutRef<"span"> {} const FormLabel = forwardRef((props, ref) => { const [variantProps, otherPropsCompressed] = resolveFormLabelVariantProps(props); const { children, asChild, ...otherPropsExtracted } = otherPropsCompressed; const Comp = asChild ? Slot : "span"; return ( {children} ); }); FormLabel.displayName = "FormLabel"; /** FormHelper **/ const [formHelperVariant, resolveFormHelperVariantProps] = vcn({ base: "opacity-75 text-sm font-light", variants: {}, defaults: {}, }); interface FormHelperProps extends VariantProps, AsChild, ComponentPropsWithoutRef<"span"> { hiddenOnInvalid?: boolean; } const FormHelper = forwardRef( (props, ref) => { const [variantProps, otherPropsCompressed] = resolveFormHelperVariantProps(props); const { asChild, children, hiddenOnInvalid, ...otherPropsExtracted } = otherPropsCompressed; const item = useContext(FormItemContext); if (item.invalid && hiddenOnInvalid) return null; const Comp = asChild ? Slot : "span"; return ( {children} ); }, ); FormHelper.displayName = "FormHelper"; /** FormError **/ const [formErrorVariant, resolveFormErrorVariantProps] = vcn({ base: "text-sm text-red-500", variants: {}, defaults: {}, }); interface FormErrorProps extends VariantProps, AsChild, Omit, "children"> {} const FormError = forwardRef((props, ref) => { const [variantProps, otherPropsCompressed] = resolveFormErrorVariantProps(props); const { asChild, ...otherPropsExtracted } = otherPropsCompressed; const item = useContext(FormItemContext); const Comp = asChild ? Slot : "span"; return item.invalid ? ( {item.invalid} ) : null; }); FormError.displayName = "FormError"; export { FormItem, FormLabel, FormHelper, FormError };