From 617b7be249a304e7145f29de7a28220bbe628dd7 Mon Sep 17 00:00:00 2001 From: p-sw Date: Sun, 4 Aug 2024 16:02:27 +0900 Subject: [PATCH] feat: add Form --- packages/react/components/Form.tsx | 183 +++++++++++++++++++++++++++++ 1 file changed, 183 insertions(+) diff --git a/packages/react/components/Form.tsx b/packages/react/components/Form.tsx index e69de29..b03ef3f 100644 --- a/packages/react/components/Form.tsx +++ b/packages/react/components/Form.tsx @@ -0,0 +1,183 @@ +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, + ComponentPropsWithoutRef<"span"> {} + +const FormError = forwardRef((props, ref) => { + const [variantProps, otherPropsCompressed] = + resolveFormErrorVariantProps(props); + const { asChild, children, ...otherPropsExtracted } = otherPropsCompressed; + + const item = useContext(FormItemContext); + + const Comp = asChild ? Slot : "span"; + + return item.invalid ? ( + + {children} + + ) : null; +}); +FormError.displayName = "FormError"; + +export { FormItem, FormLabel, FormHelper, FormError };