@skybin-tech/nebula-ui 0.0.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (145) hide show
  1. package/LICENSE +21 -0
  2. package/README.md +1 -0
  3. package/dist/cjs/components/Button/Button.cjs +37 -0
  4. package/dist/cjs/components/Button/Button.cjs.map +1 -0
  5. package/dist/cjs/components/Form/Checkbox.cjs +145 -0
  6. package/dist/cjs/components/Form/Checkbox.cjs.map +1 -0
  7. package/dist/cjs/components/Form/Form.cjs +93 -0
  8. package/dist/cjs/components/Form/Form.cjs.map +1 -0
  9. package/dist/cjs/components/Form/Radio.cjs +178 -0
  10. package/dist/cjs/components/Form/Radio.cjs.map +1 -0
  11. package/dist/cjs/components/Form/Select.cjs +168 -0
  12. package/dist/cjs/components/Form/Select.cjs.map +1 -0
  13. package/dist/cjs/components/Form/Switch.cjs +125 -0
  14. package/dist/cjs/components/Form/Switch.cjs.map +1 -0
  15. package/dist/cjs/components/Form/TextArea.cjs +168 -0
  16. package/dist/cjs/components/Form/TextArea.cjs.map +1 -0
  17. package/dist/cjs/components/Form/TextBox.cjs +211 -0
  18. package/dist/cjs/components/Form/TextBox.cjs.map +1 -0
  19. package/dist/cjs/components/Form/context.cjs +131 -0
  20. package/dist/cjs/components/Form/context.cjs.map +1 -0
  21. package/dist/cjs/components/Form/hooks.cjs +36 -0
  22. package/dist/cjs/components/Form/hooks.cjs.map +1 -0
  23. package/dist/cjs/components/Input/Input.cjs +44 -0
  24. package/dist/cjs/components/Input/Input.cjs.map +1 -0
  25. package/dist/cjs/components/ui/checkbox.cjs +45 -0
  26. package/dist/cjs/components/ui/checkbox.cjs.map +1 -0
  27. package/dist/cjs/components/ui/input.cjs +41 -0
  28. package/dist/cjs/components/ui/input.cjs.map +1 -0
  29. package/dist/cjs/components/ui/label.cjs +38 -0
  30. package/dist/cjs/components/ui/label.cjs.map +1 -0
  31. package/dist/cjs/components/ui/radio-group.cjs +53 -0
  32. package/dist/cjs/components/ui/radio-group.cjs.map +1 -0
  33. package/dist/cjs/components/ui/select.cjs +144 -0
  34. package/dist/cjs/components/ui/select.cjs.map +1 -0
  35. package/dist/cjs/components/ui/switch.cjs +45 -0
  36. package/dist/cjs/components/ui/switch.cjs.map +1 -0
  37. package/dist/cjs/components/ui/textarea.cjs +38 -0
  38. package/dist/cjs/components/ui/textarea.cjs.map +1 -0
  39. package/dist/cjs/hooks/useDebounce.cjs +17 -0
  40. package/dist/cjs/hooks/useDebounce.cjs.map +1 -0
  41. package/dist/cjs/hooks/useToggle.cjs +12 -0
  42. package/dist/cjs/hooks/useToggle.cjs.map +1 -0
  43. package/dist/cjs/index.cjs +65 -0
  44. package/dist/cjs/index.cjs.map +1 -0
  45. package/dist/cjs/nebula-ui.css +212 -0
  46. package/dist/cjs/utils/cn.cjs +9 -0
  47. package/dist/cjs/utils/cn.cjs.map +1 -0
  48. package/dist/components/Button/Button.d.ts +18 -0
  49. package/dist/components/Button/Button.d.ts.map +1 -0
  50. package/dist/components/Button/Button.js +37 -0
  51. package/dist/components/Button/Button.js.map +1 -0
  52. package/dist/components/Button/index.d.ts +3 -0
  53. package/dist/components/Button/index.d.ts.map +1 -0
  54. package/dist/components/Form/Checkbox.d.ts +32 -0
  55. package/dist/components/Form/Checkbox.d.ts.map +1 -0
  56. package/dist/components/Form/Checkbox.js +145 -0
  57. package/dist/components/Form/Checkbox.js.map +1 -0
  58. package/dist/components/Form/Form.d.ts +62 -0
  59. package/dist/components/Form/Form.d.ts.map +1 -0
  60. package/dist/components/Form/Form.js +93 -0
  61. package/dist/components/Form/Form.js.map +1 -0
  62. package/dist/components/Form/Radio.d.ts +80 -0
  63. package/dist/components/Form/Radio.d.ts.map +1 -0
  64. package/dist/components/Form/Radio.js +178 -0
  65. package/dist/components/Form/Radio.js.map +1 -0
  66. package/dist/components/Form/Select.d.ts +75 -0
  67. package/dist/components/Form/Select.d.ts.map +1 -0
  68. package/dist/components/Form/Select.js +168 -0
  69. package/dist/components/Form/Select.js.map +1 -0
  70. package/dist/components/Form/Switch.d.ts +34 -0
  71. package/dist/components/Form/Switch.d.ts.map +1 -0
  72. package/dist/components/Form/Switch.js +125 -0
  73. package/dist/components/Form/Switch.js.map +1 -0
  74. package/dist/components/Form/TextArea.d.ts +47 -0
  75. package/dist/components/Form/TextArea.d.ts.map +1 -0
  76. package/dist/components/Form/TextArea.js +168 -0
  77. package/dist/components/Form/TextArea.js.map +1 -0
  78. package/dist/components/Form/TextBox.d.ts +61 -0
  79. package/dist/components/Form/TextBox.d.ts.map +1 -0
  80. package/dist/components/Form/TextBox.js +211 -0
  81. package/dist/components/Form/TextBox.js.map +1 -0
  82. package/dist/components/Form/context.d.ts +107 -0
  83. package/dist/components/Form/context.d.ts.map +1 -0
  84. package/dist/components/Form/context.js +131 -0
  85. package/dist/components/Form/context.js.map +1 -0
  86. package/dist/components/Form/hooks.d.ts +21 -0
  87. package/dist/components/Form/hooks.d.ts.map +1 -0
  88. package/dist/components/Form/hooks.js +36 -0
  89. package/dist/components/Form/hooks.js.map +1 -0
  90. package/dist/components/Form/index.d.ts +18 -0
  91. package/dist/components/Form/index.d.ts.map +1 -0
  92. package/dist/components/Input/Input.d.ts +18 -0
  93. package/dist/components/Input/Input.d.ts.map +1 -0
  94. package/dist/components/Input/Input.js +44 -0
  95. package/dist/components/Input/Input.js.map +1 -0
  96. package/dist/components/Input/index.d.ts +3 -0
  97. package/dist/components/Input/index.d.ts.map +1 -0
  98. package/dist/components/ui/checkbox.d.ts +5 -0
  99. package/dist/components/ui/checkbox.d.ts.map +1 -0
  100. package/dist/components/ui/checkbox.js +28 -0
  101. package/dist/components/ui/checkbox.js.map +1 -0
  102. package/dist/components/ui/index.d.ts +16 -0
  103. package/dist/components/ui/index.d.ts.map +1 -0
  104. package/dist/components/ui/input.d.ts +11 -0
  105. package/dist/components/ui/input.d.ts.map +1 -0
  106. package/dist/components/ui/input.js +24 -0
  107. package/dist/components/ui/input.js.map +1 -0
  108. package/dist/components/ui/label.d.ts +6 -0
  109. package/dist/components/ui/label.d.ts.map +1 -0
  110. package/dist/components/ui/label.js +21 -0
  111. package/dist/components/ui/label.js.map +1 -0
  112. package/dist/components/ui/radio-group.d.ts +6 -0
  113. package/dist/components/ui/radio-group.d.ts.map +1 -0
  114. package/dist/components/ui/radio-group.js +36 -0
  115. package/dist/components/ui/radio-group.js.map +1 -0
  116. package/dist/components/ui/select.d.ts +14 -0
  117. package/dist/components/ui/select.d.ts.map +1 -0
  118. package/dist/components/ui/select.js +127 -0
  119. package/dist/components/ui/select.js.map +1 -0
  120. package/dist/components/ui/switch.d.ts +5 -0
  121. package/dist/components/ui/switch.d.ts.map +1 -0
  122. package/dist/components/ui/switch.js +28 -0
  123. package/dist/components/ui/switch.js.map +1 -0
  124. package/dist/components/ui/textarea.d.ts +11 -0
  125. package/dist/components/ui/textarea.d.ts.map +1 -0
  126. package/dist/components/ui/textarea.js +21 -0
  127. package/dist/components/ui/textarea.js.map +1 -0
  128. package/dist/hooks/useDebounce.d.ts +8 -0
  129. package/dist/hooks/useDebounce.d.ts.map +1 -0
  130. package/dist/hooks/useDebounce.js +17 -0
  131. package/dist/hooks/useDebounce.js.map +1 -0
  132. package/dist/hooks/useToggle.d.ts +7 -0
  133. package/dist/hooks/useToggle.d.ts.map +1 -0
  134. package/dist/hooks/useToggle.js +12 -0
  135. package/dist/hooks/useToggle.js.map +1 -0
  136. package/dist/index.d.ts +13 -0
  137. package/dist/index.d.ts.map +1 -0
  138. package/dist/index.js +62 -0
  139. package/dist/index.js.map +1 -0
  140. package/dist/nebula-ui.css +212 -0
  141. package/dist/utils/cn.d.ts +7 -0
  142. package/dist/utils/cn.d.ts.map +1 -0
  143. package/dist/utils/cn.js +9 -0
  144. package/dist/utils/cn.js.map +1 -0
  145. package/package.json +201 -0
@@ -0,0 +1 @@
1
+ {"version":3,"file":"Switch.js","sources":["../../../src/components/Form/Switch.tsx"],"sourcesContent":["import { forwardRef, useId, useContext, useEffect } from \"react\";\r\nimport type { ReactNode, ComponentPropsWithoutRef } from \"react\";\r\nimport { useController, useFormContext as useRHFFormContext, type FieldValues, type FieldPath, type Control } from \"react-hook-form\";\r\nimport { cva, type VariantProps } from \"class-variance-authority\";\r\nimport { cn } from \"../../utils/cn\";\r\nimport { FormConfigContext, type FormConfig, type FieldValidationRules } from \"../Form/context\";\r\nimport { Switch as ShadcnSwitch } from \"../ui/switch\";\r\nimport { Label } from \"../ui/label\";\r\nimport type * as SwitchPrimitives from \"@radix-ui/react-switch\";\r\n\r\nconst switchSizeVariants = cva(\r\n \"\",\r\n {\r\n variants: {\r\n size: {\r\n sm: \"h-5 w-9\",\r\n md: \"h-6 w-11\",\r\n lg: \"h-7 w-14\",\r\n },\r\n },\r\n defaultVariants: {\r\n size: \"md\",\r\n },\r\n }\r\n);\r\n\r\nconst labelVariants = cva(\r\n \"text-sm font-medium cursor-pointer select-none\",\r\n {\r\n variants: {\r\n disabled: {\r\n true: \"cursor-not-allowed opacity-50\",\r\n false: \"\",\r\n },\r\n },\r\n defaultVariants: {\r\n disabled: false,\r\n },\r\n }\r\n);\r\n\r\nexport interface SwitchProps<\r\n TFieldValues extends FieldValues = FieldValues,\r\n TName extends FieldPath<TFieldValues> = FieldPath<TFieldValues>\r\n> extends Omit<ComponentPropsWithoutRef<typeof SwitchPrimitives.Root>, \"name\" | \"checked\" | \"onCheckedChange\" | \"required\">,\r\n VariantProps<typeof switchSizeVariants> {\r\n /** Field name - required for form integration */\r\n name: TName;\r\n /** Label text for the switch */\r\n label?: ReactNode;\r\n /** Helper text displayed below the switch */\r\n helperText?: string;\r\n /** Whether to show the error message */\r\n showError?: boolean;\r\n /** Custom error message (overrides form error) */\r\n error?: string;\r\n /** External control (for use outside Form) */\r\n control?: Control<TFieldValues>;\r\n /** Text to show when checked */\r\n checkedText?: ReactNode;\r\n /** Text to show when unchecked */\r\n uncheckedText?: ReactNode;\r\n \r\n // Validation props\r\n /** Field is required (must be on) */\r\n required?: boolean | string;\r\n /** Custom validation function */\r\n validate?: (value: unknown) => boolean | string | Promise<boolean | string>;\r\n}\r\n\r\n/**\r\n * Switch component with form integration\r\n * \r\n * This is a wrapper around the shadcn/ui Switch primitive that adds:\r\n * - Form integration with react-hook-form\r\n * - Automatic validation registration\r\n * - Label, helper text, and error message support\r\n * - Checked/unchecked text display\r\n * \r\n * @example\r\n * ```tsx\r\n * // Inside a Form component\r\n * <Switch name=\"notifications\" label=\"Enable notifications\" />\r\n * <Switch name=\"terms\" label=\"Accept terms\" required=\"You must accept the terms\" />\r\n * <Switch name=\"darkMode\" label=\"Dark mode\" checkedText=\"On\" uncheckedText=\"Off\" />\r\n * ```\r\n */\r\nfunction SwitchInner<\r\n TFieldValues extends FieldValues = FieldValues,\r\n TName extends FieldPath<TFieldValues> = FieldPath<TFieldValues>\r\n>(\r\n {\r\n name,\r\n label,\r\n helperText,\r\n showError = true,\r\n error: customError,\r\n size,\r\n className,\r\n disabled,\r\n id: providedId,\r\n control: externalControl,\r\n checkedText,\r\n uncheckedText,\r\n // Validation props\r\n required,\r\n validate,\r\n ...props\r\n }: SwitchProps<TFieldValues, TName>,\r\n ref: React.ForwardedRef<React.ElementRef<typeof SwitchPrimitives.Root>>\r\n) {\r\n const generatedId = useId();\r\n const inputId = providedId ?? generatedId;\r\n \r\n // Try to get form context\r\n const formConfigContext = useContext(FormConfigContext);\r\n const formConfig: FormConfig = formConfigContext ?? {};\r\n \r\n // Get form context from react-hook-form\r\n const rhfContext = useRHFFormContext<TFieldValues>();\r\n const control = externalControl ?? rhfContext?.control;\r\n\r\n // Register validation rules with the form\r\n useEffect(() => {\r\n if (formConfigContext?.registerFieldValidation) {\r\n const rules: FieldValidationRules = {};\r\n \r\n if (required !== undefined) rules.required = required;\r\n if (validate !== undefined) rules.validate = validate;\r\n\r\n formConfigContext.registerFieldValidation({\r\n name: name as string,\r\n type: \"boolean\",\r\n rules,\r\n });\r\n\r\n return () => {\r\n formConfigContext.unregisterFieldValidation(name as string);\r\n };\r\n }\r\n }, [formConfigContext, name, required, validate]);\r\n\r\n // Use controller for form integration\r\n const { field, fieldState } = useController<TFieldValues, TName>({\r\n name,\r\n control,\r\n });\r\n\r\n const fieldError = fieldState.error?.message;\r\n const errorMessage = customError ?? fieldError;\r\n const hasError = !!errorMessage;\r\n \r\n // Merge sizes - prop takes precedence over form config\r\n const effectiveSize = size ?? formConfig.size ?? \"md\";\r\n const effectiveDisabled = disabled ?? formConfig.disabled;\r\n const isChecked = field.value ?? false;\r\n\r\n return (\r\n <div className=\"space-y-1.5\">\r\n <div className=\"flex items-center gap-3\">\r\n <ShadcnSwitch\r\n {...props}\r\n ref={ref}\r\n id={inputId}\r\n checked={isChecked}\r\n onCheckedChange={(checked) => field.onChange(checked)}\r\n disabled={effectiveDisabled}\r\n aria-labelledby={label ? `${inputId}-label` : undefined}\r\n className={cn(\r\n switchSizeVariants({ size: effectiveSize }),\r\n className\r\n )}\r\n />\r\n \r\n {(checkedText || uncheckedText) && (\r\n <span className=\"text-sm text-muted-foreground\">\r\n {isChecked ? checkedText : uncheckedText}\r\n </span>\r\n )}\r\n \r\n {label && (\r\n <Label\r\n id={`${inputId}-label`}\r\n htmlFor={inputId}\r\n className={labelVariants({ disabled: effectiveDisabled })}\r\n >\r\n {label}\r\n </Label>\r\n )}\r\n </div>\r\n\r\n {showError && hasError && (\r\n <p className=\"text-sm text-destructive\" role=\"alert\">\r\n {errorMessage}\r\n </p>\r\n )}\r\n \r\n {helperText && !hasError && (\r\n <p className=\"text-sm text-muted-foreground\">\r\n {helperText}\r\n </p>\r\n )}\r\n </div>\r\n );\r\n}\r\n\r\n// Use forwardRef with generic support\r\nexport const Switch = forwardRef(SwitchInner) as <\r\n TFieldValues extends FieldValues = FieldValues,\r\n TName extends FieldPath<TFieldValues> = FieldPath<TFieldValues>\r\n>(\r\n props: SwitchProps<TFieldValues, TName> & { ref?: React.ForwardedRef<React.ElementRef<typeof SwitchPrimitives.Root>> }\r\n) => React.ReactElement;\r\n\r\n(Switch as React.FC).displayName = \"Switch\";\r\n"],"names":["useRHFFormContext","ShadcnSwitch"],"mappings":"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"}
@@ -0,0 +1,47 @@
1
+ import { TextareaHTMLAttributes } from 'react';
2
+ import { FieldValues, FieldPath, Control } from 'react-hook-form';
3
+ import { VariantProps } from 'class-variance-authority';
4
+ declare const textAreaSizeVariants: (props?: ({
5
+ size?: "sm" | "md" | "lg" | null | undefined;
6
+ variant?: "error" | "success" | "default" | null | undefined;
7
+ } & import('class-variance-authority/types').ClassProp) | undefined) => string;
8
+ /**
9
+ * Validation rule with optional custom message
10
+ */
11
+ type ValidationRule<T> = T | {
12
+ value: T;
13
+ message: string;
14
+ };
15
+ export interface TextAreaProps<TFieldValues extends FieldValues = FieldValues, TName extends FieldPath<TFieldValues> = FieldPath<TFieldValues>> extends Omit<TextareaHTMLAttributes<HTMLTextAreaElement>, "name" | "required" | "minLength" | "maxLength">, VariantProps<typeof textAreaSizeVariants> {
16
+ /** Field name - required for form integration */
17
+ name: TName;
18
+ /** Label text for the textarea */
19
+ label?: string;
20
+ /** Helper text displayed below the textarea */
21
+ helperText?: string;
22
+ /** Whether to show the error message */
23
+ showError?: boolean;
24
+ /** Custom error message (overrides form error) */
25
+ error?: string;
26
+ /** Whether the textarea should take full width */
27
+ fullWidth?: boolean;
28
+ /** Show character count */
29
+ showCount?: boolean;
30
+ /** Maximum character count (for display) */
31
+ maxCount?: number;
32
+ /** External control (for use outside Form) */
33
+ control?: Control<TFieldValues>;
34
+ /** Field is required */
35
+ required?: boolean | string;
36
+ /** Minimum length */
37
+ minLength?: ValidationRule<number>;
38
+ /** Maximum length */
39
+ maxLength?: ValidationRule<number>;
40
+ /** Custom validation function */
41
+ validate?: (value: unknown) => boolean | string | Promise<boolean | string>;
42
+ }
43
+ export declare const TextArea: <TFieldValues extends FieldValues = FieldValues, TName extends FieldPath<TFieldValues> = FieldPath<TFieldValues>>(props: TextAreaProps<TFieldValues, TName> & {
44
+ ref?: React.ForwardedRef<HTMLTextAreaElement>;
45
+ }) => React.ReactElement;
46
+ export {};
47
+ //# sourceMappingURL=TextArea.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"TextArea.d.ts","sourceRoot":"","sources":["../../../src/components/Form/TextArea.tsx"],"names":[],"mappings":"AACA,OAAO,KAAK,EAAE,sBAAsB,EAAE,MAAM,OAAO,CAAC;AACpD,OAAO,EAAsD,KAAK,WAAW,EAAE,KAAK,SAAS,EAAE,KAAK,OAAO,EAAE,MAAM,iBAAiB,CAAC;AACrI,OAAO,EAAO,KAAK,YAAY,EAAE,MAAM,0BAA0B,CAAC;AAMlE,QAAA,MAAM,oBAAoB;;;8EAoBzB,CAAC;AAiBF;;GAEG;AACH,KAAK,cAAc,CAAC,CAAC,IAAI,CAAC,GAAG;IAAE,KAAK,EAAE,CAAC,CAAC;IAAC,OAAO,EAAE,MAAM,CAAA;CAAE,CAAC;AAE3D,MAAM,WAAW,aAAa,CAC5B,YAAY,SAAS,WAAW,GAAG,WAAW,EAC9C,KAAK,SAAS,SAAS,CAAC,YAAY,CAAC,GAAG,SAAS,CAAC,YAAY,CAAC,CAC/D,SAAQ,IAAI,CAAC,sBAAsB,CAAC,mBAAmB,CAAC,EAAE,MAAM,GAAG,UAAU,GAAG,WAAW,GAAG,WAAW,CAAC,EACxG,YAAY,CAAC,OAAO,oBAAoB,CAAC;IAC3C,iDAAiD;IACjD,IAAI,EAAE,KAAK,CAAC;IACZ,kCAAkC;IAClC,KAAK,CAAC,EAAE,MAAM,CAAC;IACf,+CAA+C;IAC/C,UAAU,CAAC,EAAE,MAAM,CAAC;IACpB,wCAAwC;IACxC,SAAS,CAAC,EAAE,OAAO,CAAC;IACpB,kDAAkD;IAClD,KAAK,CAAC,EAAE,MAAM,CAAC;IACf,kDAAkD;IAClD,SAAS,CAAC,EAAE,OAAO,CAAC;IACpB,2BAA2B;IAC3B,SAAS,CAAC,EAAE,OAAO,CAAC;IACpB,4CAA4C;IAC5C,QAAQ,CAAC,EAAE,MAAM,CAAC;IAClB,8CAA8C;IAC9C,OAAO,CAAC,EAAE,OAAO,CAAC,YAAY,CAAC,CAAC;IAGhC,wBAAwB;IACxB,QAAQ,CAAC,EAAE,OAAO,GAAG,MAAM,CAAC;IAC5B,qBAAqB;IACrB,SAAS,CAAC,EAAE,cAAc,CAAC,MAAM,CAAC,CAAC;IACnC,qBAAqB;IACrB,SAAS,CAAC,EAAE,cAAc,CAAC,MAAM,CAAC,CAAC;IACnC,iCAAiC;IACjC,QAAQ,CAAC,EAAE,CAAC,KAAK,EAAE,OAAO,KAAK,OAAO,GAAG,MAAM,GAAG,OAAO,CAAC,OAAO,GAAG,MAAM,CAAC,CAAC;CAC7E;AA6KD,eAAO,MAAM,QAAQ,EAAgC,CACnD,YAAY,SAAS,WAAW,GAAG,WAAW,EAC9C,KAAK,SAAS,SAAS,CAAC,YAAY,CAAC,GAAG,SAAS,CAAC,YAAY,CAAC,EAE/D,KAAK,EAAE,aAAa,CAAC,YAAY,EAAE,KAAK,CAAC,GAAG;IAAE,GAAG,CAAC,EAAE,KAAK,CAAC,YAAY,CAAC,mBAAmB,CAAC,CAAA;CAAE,KAC1F,KAAK,CAAC,YAAY,CAAC"}
@@ -0,0 +1,168 @@
1
+ import { jsxs, jsx } from "react/jsx-runtime";
2
+ import { forwardRef, useId, useContext, useEffect } from "react";
3
+ import { useFormContext, useController } from "react-hook-form";
4
+ import { cva } from "../../node_modules/class-variance-authority/dist/index.js";
5
+ import { cn } from "../../utils/cn.js";
6
+ import { FormConfigContext } from "./context.js";
7
+ import { Textarea } from "../ui/textarea.js";
8
+ import { Label } from "../ui/label.js";
9
+ const textAreaSizeVariants = cva(
10
+ "",
11
+ {
12
+ variants: {
13
+ size: {
14
+ sm: "text-xs px-2 py-1.5 min-h-[60px]",
15
+ md: "text-sm px-3 py-2 min-h-[80px]",
16
+ lg: "text-base px-4 py-3 min-h-[100px]"
17
+ },
18
+ variant: {
19
+ default: "border-input focus-visible:ring-ring",
20
+ error: "border-destructive focus-visible:ring-destructive",
21
+ success: "border-green-500 focus-visible:ring-green-500"
22
+ }
23
+ },
24
+ defaultVariants: {
25
+ size: "md",
26
+ variant: "default"
27
+ }
28
+ }
29
+ );
30
+ const labelVariants = cva(
31
+ "block text-sm font-medium mb-1.5",
32
+ {
33
+ variants: {
34
+ required: {
35
+ true: "after:content-['*'] after:ml-0.5 after:text-destructive",
36
+ false: ""
37
+ }
38
+ },
39
+ defaultVariants: {
40
+ required: false
41
+ }
42
+ }
43
+ );
44
+ function TextAreaInner({
45
+ name,
46
+ label,
47
+ helperText,
48
+ showError = true,
49
+ error: customError,
50
+ size,
51
+ variant,
52
+ fullWidth = true,
53
+ className,
54
+ disabled,
55
+ showCount,
56
+ maxCount,
57
+ id: providedId,
58
+ control: externalControl,
59
+ // Validation props
60
+ required,
61
+ minLength,
62
+ maxLength,
63
+ validate,
64
+ ...props
65
+ }, ref) {
66
+ const generatedId = useId();
67
+ const inputId = providedId ?? generatedId;
68
+ const formConfigContext = useContext(FormConfigContext);
69
+ const formConfig = formConfigContext ?? {};
70
+ const rhfContext = useFormContext();
71
+ const control = externalControl ?? rhfContext?.control;
72
+ useEffect(() => {
73
+ if (formConfigContext?.registerFieldValidation) {
74
+ const rules = {};
75
+ if (required !== void 0) rules.required = required;
76
+ if (minLength !== void 0) rules.minLength = minLength;
77
+ if (maxLength !== void 0) rules.maxLength = maxLength;
78
+ if (validate !== void 0) rules.validate = validate;
79
+ formConfigContext.registerFieldValidation({
80
+ name,
81
+ type: "string",
82
+ rules
83
+ });
84
+ return () => {
85
+ formConfigContext.unregisterFieldValidation(name);
86
+ };
87
+ }
88
+ }, [formConfigContext, name, required, minLength, maxLength, validate]);
89
+ const { field, fieldState } = useController({
90
+ name,
91
+ control
92
+ });
93
+ const fieldError = fieldState.error?.message;
94
+ const errorMessage = customError ?? fieldError;
95
+ const hasError = !!errorMessage;
96
+ const effectiveSize = size ?? formConfig.size ?? "md";
97
+ const effectiveDisabled = disabled ?? formConfig.disabled;
98
+ const effectiveVariant = hasError ? "error" : variant;
99
+ const currentLength = typeof field.value === "string" ? field.value.length : 0;
100
+ const maxLengthValue = maxCount ?? (typeof maxLength === "number" ? maxLength : maxLength?.value);
101
+ return /* @__PURE__ */ jsxs("div", { className: cn("space-y-1.5", fullWidth && "w-full"), children: [
102
+ label && /* @__PURE__ */ jsxs(
103
+ Label,
104
+ {
105
+ htmlFor: inputId,
106
+ className: labelVariants({ required: !!required }),
107
+ children: [
108
+ label,
109
+ formConfig.colon && ":"
110
+ ]
111
+ }
112
+ ),
113
+ /* @__PURE__ */ jsxs("div", { className: "relative", children: [
114
+ /* @__PURE__ */ jsx(
115
+ Textarea,
116
+ {
117
+ ...props,
118
+ ...field,
119
+ ref: (node) => {
120
+ if (typeof ref === "function") {
121
+ ref(node);
122
+ } else if (ref) {
123
+ ref.current = node;
124
+ }
125
+ field.ref(node);
126
+ },
127
+ id: inputId,
128
+ disabled: effectiveDisabled,
129
+ maxLength: maxLengthValue,
130
+ "aria-invalid": hasError,
131
+ "aria-describedby": hasError ? `${inputId}-error` : helperText ? `${inputId}-helper` : void 0,
132
+ className: cn(
133
+ textAreaSizeVariants({ size: effectiveSize, variant: effectiveVariant }),
134
+ showCount && "pb-6",
135
+ className
136
+ )
137
+ }
138
+ ),
139
+ showCount && /* @__PURE__ */ jsxs("div", { className: "absolute bottom-2 right-2 text-xs text-muted-foreground", children: [
140
+ currentLength,
141
+ maxCount ? `/${maxCount}` : ""
142
+ ] })
143
+ ] }),
144
+ showError && hasError && /* @__PURE__ */ jsx(
145
+ "p",
146
+ {
147
+ id: `${inputId}-error`,
148
+ className: "text-sm text-destructive",
149
+ role: "alert",
150
+ children: errorMessage
151
+ }
152
+ ),
153
+ helperText && !hasError && /* @__PURE__ */ jsx(
154
+ "p",
155
+ {
156
+ id: `${inputId}-helper`,
157
+ className: "text-sm text-muted-foreground",
158
+ children: helperText
159
+ }
160
+ )
161
+ ] });
162
+ }
163
+ const TextArea = forwardRef(TextAreaInner);
164
+ TextArea.displayName = "TextArea";
165
+ export {
166
+ TextArea
167
+ };
168
+ //# sourceMappingURL=TextArea.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"TextArea.js","sources":["../../../src/components/Form/TextArea.tsx"],"sourcesContent":["import { forwardRef, useId, useContext, useEffect } from \"react\";\r\nimport type { TextareaHTMLAttributes } from \"react\";\r\nimport { useController, useFormContext as useRHFFormContext, type FieldValues, type FieldPath, type Control } from \"react-hook-form\";\r\nimport { cva, type VariantProps } from \"class-variance-authority\";\r\nimport { cn } from \"../../utils/cn\";\r\nimport { FormConfigContext, type FormConfig, type FieldValidationRules } from \"../Form/context\";\r\nimport { Textarea } from \"../ui/textarea\";\r\nimport { Label } from \"../ui/label\";\r\n\r\nconst textAreaSizeVariants = cva(\r\n \"\",\r\n {\r\n variants: {\r\n size: {\r\n sm: \"text-xs px-2 py-1.5 min-h-[60px]\",\r\n md: \"text-sm px-3 py-2 min-h-[80px]\",\r\n lg: \"text-base px-4 py-3 min-h-[100px]\",\r\n },\r\n variant: {\r\n default: \"border-input focus-visible:ring-ring\",\r\n error: \"border-destructive focus-visible:ring-destructive\",\r\n success: \"border-green-500 focus-visible:ring-green-500\",\r\n },\r\n },\r\n defaultVariants: {\r\n size: \"md\",\r\n variant: \"default\",\r\n },\r\n }\r\n);\r\n\r\nconst labelVariants = cva(\r\n \"block text-sm font-medium mb-1.5\",\r\n {\r\n variants: {\r\n required: {\r\n true: \"after:content-['*'] after:ml-0.5 after:text-destructive\",\r\n false: \"\",\r\n },\r\n },\r\n defaultVariants: {\r\n required: false,\r\n },\r\n }\r\n);\r\n\r\n/**\r\n * Validation rule with optional custom message\r\n */\r\ntype ValidationRule<T> = T | { value: T; message: string };\r\n\r\nexport interface TextAreaProps<\r\n TFieldValues extends FieldValues = FieldValues,\r\n TName extends FieldPath<TFieldValues> = FieldPath<TFieldValues>\r\n> extends Omit<TextareaHTMLAttributes<HTMLTextAreaElement>, \"name\" | \"required\" | \"minLength\" | \"maxLength\">,\r\n VariantProps<typeof textAreaSizeVariants> {\r\n /** Field name - required for form integration */\r\n name: TName;\r\n /** Label text for the textarea */\r\n label?: string;\r\n /** Helper text displayed below the textarea */\r\n helperText?: string;\r\n /** Whether to show the error message */\r\n showError?: boolean;\r\n /** Custom error message (overrides form error) */\r\n error?: string;\r\n /** Whether the textarea should take full width */\r\n fullWidth?: boolean;\r\n /** Show character count */\r\n showCount?: boolean;\r\n /** Maximum character count (for display) */\r\n maxCount?: number;\r\n /** External control (for use outside Form) */\r\n control?: Control<TFieldValues>;\r\n \r\n // Validation props\r\n /** Field is required */\r\n required?: boolean | string;\r\n /** Minimum length */\r\n minLength?: ValidationRule<number>;\r\n /** Maximum length */\r\n maxLength?: ValidationRule<number>;\r\n /** Custom validation function */\r\n validate?: (value: unknown) => boolean | string | Promise<boolean | string>;\r\n}\r\n\r\n/**\r\n * TextArea component with form integration\r\n * \r\n * This is a wrapper around the shadcn/ui Textarea primitive that adds:\r\n * - Form integration with react-hook-form\r\n * - Automatic validation registration\r\n * - Label, helper text, and error message support\r\n * - Character count display\r\n * \r\n * @example\r\n * ```tsx\r\n * // Inside a Form component\r\n * <TextArea name=\"description\" label=\"Description\" required minLength={10} />\r\n * <TextArea name=\"bio\" label=\"Bio\" showCount maxCount={500} maxLength={500} />\r\n * ```\r\n */\r\nfunction TextAreaInner<\r\n TFieldValues extends FieldValues = FieldValues,\r\n TName extends FieldPath<TFieldValues> = FieldPath<TFieldValues>\r\n>(\r\n {\r\n name,\r\n label,\r\n helperText,\r\n showError = true,\r\n error: customError,\r\n size,\r\n variant,\r\n fullWidth = true,\r\n className,\r\n disabled,\r\n showCount,\r\n maxCount,\r\n id: providedId,\r\n control: externalControl,\r\n // Validation props\r\n required,\r\n minLength,\r\n maxLength,\r\n validate,\r\n ...props\r\n }: TextAreaProps<TFieldValues, TName>,\r\n ref: React.ForwardedRef<HTMLTextAreaElement>\r\n) {\r\n const generatedId = useId();\r\n const inputId = providedId ?? generatedId;\r\n \r\n // Try to get form context\r\n const formConfigContext = useContext(FormConfigContext);\r\n const formConfig: FormConfig = formConfigContext ?? {};\r\n \r\n // Get form context from react-hook-form\r\n const rhfContext = useRHFFormContext<TFieldValues>();\r\n const control = externalControl ?? rhfContext?.control;\r\n\r\n // Register validation rules with the form\r\n useEffect(() => {\r\n if (formConfigContext?.registerFieldValidation) {\r\n const rules: FieldValidationRules = {};\r\n \r\n if (required !== undefined) rules.required = required;\r\n if (minLength !== undefined) rules.minLength = minLength;\r\n if (maxLength !== undefined) rules.maxLength = maxLength;\r\n if (validate !== undefined) rules.validate = validate;\r\n\r\n formConfigContext.registerFieldValidation({\r\n name: name as string,\r\n type: \"string\",\r\n rules,\r\n });\r\n\r\n return () => {\r\n formConfigContext.unregisterFieldValidation(name as string);\r\n };\r\n }\r\n }, [formConfigContext, name, required, minLength, maxLength, validate]);\r\n\r\n // Use controller for form integration\r\n const { field, fieldState } = useController<TFieldValues, TName>({\r\n name,\r\n control,\r\n });\r\n\r\n const fieldError = fieldState.error?.message;\r\n const errorMessage = customError ?? fieldError;\r\n const hasError = !!errorMessage;\r\n \r\n // Merge sizes - prop takes precedence over form config\r\n const effectiveSize = size ?? formConfig.size ?? \"md\";\r\n const effectiveDisabled = disabled ?? formConfig.disabled;\r\n \r\n // Determine variant based on error state\r\n const effectiveVariant = hasError ? \"error\" : variant;\r\n\r\n const currentLength = typeof field.value === \"string\" ? field.value.length : 0;\r\n \r\n // Get maxLength value for HTML attribute\r\n const maxLengthValue = maxCount ?? (typeof maxLength === \"number\" ? maxLength : (maxLength as { value: number })?.value);\r\n\r\n return (\r\n <div className={cn(\"space-y-1.5\", fullWidth && \"w-full\")}>\r\n {label && (\r\n <Label\r\n htmlFor={inputId}\r\n className={labelVariants({ required: !!required })}\r\n >\r\n {label}\r\n {formConfig.colon && \":\"}\r\n </Label>\r\n )}\r\n \r\n <div className=\"relative\">\r\n <Textarea\r\n {...props}\r\n {...field}\r\n ref={(node) => {\r\n if (typeof ref === \"function\") {\r\n ref(node);\r\n } else if (ref) {\r\n ref.current = node;\r\n }\r\n field.ref(node);\r\n }}\r\n id={inputId}\r\n disabled={effectiveDisabled}\r\n maxLength={maxLengthValue}\r\n aria-invalid={hasError}\r\n aria-describedby={\r\n hasError\r\n ? `${inputId}-error`\r\n : helperText\r\n ? `${inputId}-helper`\r\n : undefined\r\n }\r\n className={cn(\r\n textAreaSizeVariants({ size: effectiveSize, variant: effectiveVariant }),\r\n showCount && \"pb-6\",\r\n className\r\n )}\r\n />\r\n \r\n {showCount && (\r\n <div className=\"absolute bottom-2 right-2 text-xs text-muted-foreground\">\r\n {currentLength}{maxCount ? `/${maxCount}` : \"\"}\r\n </div>\r\n )}\r\n </div>\r\n\r\n {showError && hasError && (\r\n <p\r\n id={`${inputId}-error`}\r\n className=\"text-sm text-destructive\"\r\n role=\"alert\"\r\n >\r\n {errorMessage}\r\n </p>\r\n )}\r\n \r\n {helperText && !hasError && (\r\n <p\r\n id={`${inputId}-helper`}\r\n className=\"text-sm text-muted-foreground\"\r\n >\r\n {helperText}\r\n </p>\r\n )}\r\n </div>\r\n );\r\n}\r\n\r\n// Use forwardRef with generic support\r\nexport const TextArea = forwardRef(TextAreaInner) as <\r\n TFieldValues extends FieldValues = FieldValues,\r\n TName extends FieldPath<TFieldValues> = FieldPath<TFieldValues>\r\n>(\r\n props: TextAreaProps<TFieldValues, TName> & { ref?: React.ForwardedRef<HTMLTextAreaElement> }\r\n) => React.ReactElement;\r\n\r\n(TextArea as React.FC).displayName = \"TextArea\";\r\n"],"names":["useRHFFormContext"],"mappings":";;;;;;;;AASA,MAAM,uBAAuB;AAAA,EAC3B;AAAA,EACA;AAAA,IACE,UAAU;AAAA,MACR,MAAM;AAAA,QACJ,IAAI;AAAA,QACJ,IAAI;AAAA,QACJ,IAAI;AAAA,MAAA;AAAA,MAEN,SAAS;AAAA,QACP,SAAS;AAAA,QACT,OAAO;AAAA,QACP,SAAS;AAAA,MAAA;AAAA,IACX;AAAA,IAEF,iBAAiB;AAAA,MACf,MAAM;AAAA,MACN,SAAS;AAAA,IAAA;AAAA,EACX;AAEJ;AAEA,MAAM,gBAAgB;AAAA,EACpB;AAAA,EACA;AAAA,IACE,UAAU;AAAA,MACR,UAAU;AAAA,QACR,MAAM;AAAA,QACN,OAAO;AAAA,MAAA;AAAA,IACT;AAAA,IAEF,iBAAiB;AAAA,MACf,UAAU;AAAA,IAAA;AAAA,EACZ;AAEJ;AA0DA,SAAS,cAIP;AAAA,EACE;AAAA,EACA;AAAA,EACA;AAAA,EACA,YAAY;AAAA,EACZ,OAAO;AAAA,EACP;AAAA,EACA;AAAA,EACA,YAAY;AAAA,EACZ;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA,IAAI;AAAA,EACJ,SAAS;AAAA;AAAA,EAET;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA,GAAG;AACL,GACA,KACA;AACA,QAAM,cAAc,MAAA;AACpB,QAAM,UAAU,cAAc;AAG9B,QAAM,oBAAoB,WAAW,iBAAiB;AACtD,QAAM,aAAyB,qBAAqB,CAAA;AAGpD,QAAM,aAAaA,eAAA;AACnB,QAAM,UAAU,mBAAmB,YAAY;AAG/C,YAAU,MAAM;AACd,QAAI,mBAAmB,yBAAyB;AAC9C,YAAM,QAA8B,CAAA;AAEpC,UAAI,aAAa,OAAW,OAAM,WAAW;AAC7C,UAAI,cAAc,OAAW,OAAM,YAAY;AAC/C,UAAI,cAAc,OAAW,OAAM,YAAY;AAC/C,UAAI,aAAa,OAAW,OAAM,WAAW;AAE7C,wBAAkB,wBAAwB;AAAA,QACxC;AAAA,QACA,MAAM;AAAA,QACN;AAAA,MAAA,CACD;AAED,aAAO,MAAM;AACX,0BAAkB,0BAA0B,IAAc;AAAA,MAC5D;AAAA,IACF;AAAA,EACF,GAAG,CAAC,mBAAmB,MAAM,UAAU,WAAW,WAAW,QAAQ,CAAC;AAGtE,QAAM,EAAE,OAAO,WAAA,IAAe,cAAmC;AAAA,IAC/D;AAAA,IACA;AAAA,EAAA,CACD;AAED,QAAM,aAAa,WAAW,OAAO;AACrC,QAAM,eAAe,eAAe;AACpC,QAAM,WAAW,CAAC,CAAC;AAGnB,QAAM,gBAAgB,QAAQ,WAAW,QAAQ;AACjD,QAAM,oBAAoB,YAAY,WAAW;AAGjD,QAAM,mBAAmB,WAAW,UAAU;AAE9C,QAAM,gBAAgB,OAAO,MAAM,UAAU,WAAW,MAAM,MAAM,SAAS;AAG7E,QAAM,iBAAiB,aAAa,OAAO,cAAc,WAAW,YAAa,WAAiC;AAElH,8BACG,OAAA,EAAI,WAAW,GAAG,eAAe,aAAa,QAAQ,GACpD,UAAA;AAAA,IAAA,SACC;AAAA,MAAC;AAAA,MAAA;AAAA,QACC,SAAS;AAAA,QACT,WAAW,cAAc,EAAE,UAAU,CAAC,CAAC,UAAU;AAAA,QAEhD,UAAA;AAAA,UAAA;AAAA,UACA,WAAW,SAAS;AAAA,QAAA;AAAA,MAAA;AAAA,IAAA;AAAA,IAIzB,qBAAC,OAAA,EAAI,WAAU,YACb,UAAA;AAAA,MAAA;AAAA,QAAC;AAAA,QAAA;AAAA,UACE,GAAG;AAAA,UACH,GAAG;AAAA,UACJ,KAAK,CAAC,SAAS;AACb,gBAAI,OAAO,QAAQ,YAAY;AAC7B,kBAAI,IAAI;AAAA,YACV,WAAW,KAAK;AACd,kBAAI,UAAU;AAAA,YAChB;AACA,kBAAM,IAAI,IAAI;AAAA,UAChB;AAAA,UACA,IAAI;AAAA,UACJ,UAAU;AAAA,UACV,WAAW;AAAA,UACX,gBAAc;AAAA,UACd,oBACE,WACI,GAAG,OAAO,WACV,aACA,GAAG,OAAO,YACV;AAAA,UAEN,WAAW;AAAA,YACT,qBAAqB,EAAE,MAAM,eAAe,SAAS,kBAAkB;AAAA,YACvE,aAAa;AAAA,YACb;AAAA,UAAA;AAAA,QACF;AAAA,MAAA;AAAA,MAGD,aACC,qBAAC,OAAA,EAAI,WAAU,2DACZ,UAAA;AAAA,QAAA;AAAA,QAAe,WAAW,IAAI,QAAQ,KAAK;AAAA,MAAA,EAAA,CAC9C;AAAA,IAAA,GAEJ;AAAA,IAEC,aAAa,YACZ;AAAA,MAAC;AAAA,MAAA;AAAA,QACC,IAAI,GAAG,OAAO;AAAA,QACd,WAAU;AAAA,QACV,MAAK;AAAA,QAEJ,UAAA;AAAA,MAAA;AAAA,IAAA;AAAA,IAIJ,cAAc,CAAC,YACd;AAAA,MAAC;AAAA,MAAA;AAAA,QACC,IAAI,GAAG,OAAO;AAAA,QACd,WAAU;AAAA,QAET,UAAA;AAAA,MAAA;AAAA,IAAA;AAAA,EACH,GAEJ;AAEJ;AAGO,MAAM,WAAW,WAAW,aAAa;AAO/C,SAAsB,cAAc;"}
@@ -0,0 +1,61 @@
1
+ import { InputHTMLAttributes, ReactNode } from 'react';
2
+ import { FieldValues, FieldPath, Control } from 'react-hook-form';
3
+ import { VariantProps } from 'class-variance-authority';
4
+ declare const textBoxVariants: (props?: ({
5
+ size?: "sm" | "md" | "lg" | null | undefined;
6
+ variant?: "error" | "success" | "default" | null | undefined;
7
+ } & import('class-variance-authority/types').ClassProp) | undefined) => string;
8
+ /**
9
+ * Validation rule with optional custom message
10
+ */
11
+ type ValidationRule<T> = T | {
12
+ value: T;
13
+ message: string;
14
+ };
15
+ export interface TextBoxProps<TFieldValues extends FieldValues = FieldValues, TName extends FieldPath<TFieldValues> = FieldPath<TFieldValues>> extends Omit<InputHTMLAttributes<HTMLInputElement>, "size" | "name" | "prefix" | "pattern" | "required" | "minLength" | "maxLength">, VariantProps<typeof textBoxVariants> {
16
+ /** Field name - required for form integration */
17
+ name: TName;
18
+ /** Label text for the input */
19
+ label?: string;
20
+ /** Helper text displayed below the input */
21
+ helperText?: string;
22
+ /** Whether to show the error message */
23
+ showError?: boolean;
24
+ /** Custom error message (overrides form error) */
25
+ error?: string;
26
+ /** Whether the input should take full width */
27
+ fullWidth?: boolean;
28
+ /** Prefix element */
29
+ prefix?: ReactNode;
30
+ /** Suffix element */
31
+ suffix?: ReactNode;
32
+ /** Allow clear button */
33
+ allowClear?: boolean;
34
+ /** Callback when clear is clicked */
35
+ onClear?: () => void;
36
+ /** External control (for use outside Form) */
37
+ control?: Control<TFieldValues>;
38
+ /** Field is required */
39
+ required?: boolean | string;
40
+ /** Minimum length for strings */
41
+ minLength?: ValidationRule<number>;
42
+ /** Maximum length for strings */
43
+ maxLength?: ValidationRule<number>;
44
+ /** Minimum value for numbers */
45
+ minValue?: ValidationRule<number>;
46
+ /** Maximum value for numbers */
47
+ maxValue?: ValidationRule<number>;
48
+ /** Regex pattern for validation */
49
+ pattern?: ValidationRule<RegExp>;
50
+ /** Email validation */
51
+ email?: boolean | string;
52
+ /** URL validation */
53
+ url?: boolean | string;
54
+ /** Custom validation function */
55
+ validate?: (value: unknown) => boolean | string | Promise<boolean | string>;
56
+ }
57
+ export declare const TextBox: <TFieldValues extends FieldValues = FieldValues, TName extends FieldPath<TFieldValues> = FieldPath<TFieldValues>>(props: TextBoxProps<TFieldValues, TName> & {
58
+ ref?: React.ForwardedRef<HTMLInputElement>;
59
+ }) => React.ReactElement;
60
+ export {};
61
+ //# sourceMappingURL=TextBox.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"TextBox.d.ts","sourceRoot":"","sources":["../../../src/components/Form/TextBox.tsx"],"names":[],"mappings":"AACA,OAAO,KAAK,EAAE,mBAAmB,EAAE,SAAS,EAAE,MAAM,OAAO,CAAC;AAC5D,OAAO,EAAsD,KAAK,WAAW,EAAE,KAAK,SAAS,EAAE,KAAK,OAAO,EAAE,MAAM,iBAAiB,CAAC;AACrI,OAAO,EAAO,KAAK,YAAY,EAAE,MAAM,0BAA0B,CAAC;AAOlE,QAAA,MAAM,eAAe;;;8EAoBpB,CAAC;AAiBF;;GAEG;AACH,KAAK,cAAc,CAAC,CAAC,IAAI,CAAC,GAAG;IAAE,KAAK,EAAE,CAAC,CAAC;IAAC,OAAO,EAAE,MAAM,CAAA;CAAE,CAAC;AAE3D,MAAM,WAAW,YAAY,CAC3B,YAAY,SAAS,WAAW,GAAG,WAAW,EAC9C,KAAK,SAAS,SAAS,CAAC,YAAY,CAAC,GAAG,SAAS,CAAC,YAAY,CAAC,CAC/D,SAAQ,IAAI,CAAC,mBAAmB,CAAC,gBAAgB,CAAC,EAAE,MAAM,GAAG,MAAM,GAAG,QAAQ,GAAG,SAAS,GAAG,UAAU,GAAG,WAAW,GAAG,WAAW,CAAC,EAClI,YAAY,CAAC,OAAO,eAAe,CAAC;IACtC,iDAAiD;IACjD,IAAI,EAAE,KAAK,CAAC;IACZ,+BAA+B;IAC/B,KAAK,CAAC,EAAE,MAAM,CAAC;IACf,4CAA4C;IAC5C,UAAU,CAAC,EAAE,MAAM,CAAC;IACpB,wCAAwC;IACxC,SAAS,CAAC,EAAE,OAAO,CAAC;IACpB,kDAAkD;IAClD,KAAK,CAAC,EAAE,MAAM,CAAC;IACf,+CAA+C;IAC/C,SAAS,CAAC,EAAE,OAAO,CAAC;IACpB,qBAAqB;IACrB,MAAM,CAAC,EAAE,SAAS,CAAC;IACnB,qBAAqB;IACrB,MAAM,CAAC,EAAE,SAAS,CAAC;IACnB,yBAAyB;IACzB,UAAU,CAAC,EAAE,OAAO,CAAC;IACrB,qCAAqC;IACrC,OAAO,CAAC,EAAE,MAAM,IAAI,CAAC;IACrB,8CAA8C;IAC9C,OAAO,CAAC,EAAE,OAAO,CAAC,YAAY,CAAC,CAAC;IAGhC,wBAAwB;IACxB,QAAQ,CAAC,EAAE,OAAO,GAAG,MAAM,CAAC;IAC5B,iCAAiC;IACjC,SAAS,CAAC,EAAE,cAAc,CAAC,MAAM,CAAC,CAAC;IACnC,iCAAiC;IACjC,SAAS,CAAC,EAAE,cAAc,CAAC,MAAM,CAAC,CAAC;IACnC,gCAAgC;IAChC,QAAQ,CAAC,EAAE,cAAc,CAAC,MAAM,CAAC,CAAC;IAClC,gCAAgC;IAChC,QAAQ,CAAC,EAAE,cAAc,CAAC,MAAM,CAAC,CAAC;IAClC,mCAAmC;IACnC,OAAO,CAAC,EAAE,cAAc,CAAC,MAAM,CAAC,CAAC;IACjC,uBAAuB;IACvB,KAAK,CAAC,EAAE,OAAO,GAAG,MAAM,CAAC;IACzB,qBAAqB;IACrB,GAAG,CAAC,EAAE,OAAO,GAAG,MAAM,CAAC;IACvB,iCAAiC;IACjC,QAAQ,CAAC,EAAE,CAAC,KAAK,EAAE,OAAO,KAAK,OAAO,GAAG,MAAM,GAAG,OAAO,CAAC,OAAO,GAAG,MAAM,CAAC,CAAC;CAC7E;AAqOD,eAAO,MAAM,OAAO,EAA+B,CACjD,YAAY,SAAS,WAAW,GAAG,WAAW,EAC9C,KAAK,SAAS,SAAS,CAAC,YAAY,CAAC,GAAG,SAAS,CAAC,YAAY,CAAC,EAE/D,KAAK,EAAE,YAAY,CAAC,YAAY,EAAE,KAAK,CAAC,GAAG;IAAE,GAAG,CAAC,EAAE,KAAK,CAAC,YAAY,CAAC,gBAAgB,CAAC,CAAA;CAAE,KACtF,KAAK,CAAC,YAAY,CAAC"}
@@ -0,0 +1,211 @@
1
+ import { jsxs, jsx } from "react/jsx-runtime";
2
+ import { forwardRef, useId, useContext, useEffect } from "react";
3
+ import { useFormContext, useController } from "react-hook-form";
4
+ import { cva } from "../../node_modules/class-variance-authority/dist/index.js";
5
+ import { cn } from "../../utils/cn.js";
6
+ import { FormConfigContext } from "./context.js";
7
+ import { Input } from "../ui/input.js";
8
+ import { Label } from "../ui/label.js";
9
+ import X from "../../node_modules/lucide-react/dist/esm/icons/x.js";
10
+ const textBoxVariants = cva(
11
+ "",
12
+ {
13
+ variants: {
14
+ size: {
15
+ sm: "h-8 text-xs px-2",
16
+ md: "h-10 text-sm px-3",
17
+ lg: "h-12 text-base px-4"
18
+ },
19
+ variant: {
20
+ default: "border-input focus-visible:ring-ring",
21
+ error: "border-destructive focus-visible:ring-destructive",
22
+ success: "border-green-500 focus-visible:ring-green-500"
23
+ }
24
+ },
25
+ defaultVariants: {
26
+ size: "md",
27
+ variant: "default"
28
+ }
29
+ }
30
+ );
31
+ const labelVariants = cva(
32
+ "block text-sm font-medium mb-1.5",
33
+ {
34
+ variants: {
35
+ required: {
36
+ true: "after:content-['*'] after:ml-0.5 after:text-destructive",
37
+ false: ""
38
+ }
39
+ },
40
+ defaultVariants: {
41
+ required: false
42
+ }
43
+ }
44
+ );
45
+ function TextBoxInner({
46
+ name,
47
+ label,
48
+ helperText,
49
+ showError = true,
50
+ error: customError,
51
+ size,
52
+ variant,
53
+ fullWidth = true,
54
+ className,
55
+ disabled,
56
+ prefix,
57
+ suffix,
58
+ allowClear,
59
+ onClear,
60
+ id: providedId,
61
+ control: externalControl,
62
+ // Validation props
63
+ required,
64
+ minLength,
65
+ maxLength,
66
+ minValue,
67
+ maxValue,
68
+ pattern,
69
+ email,
70
+ url,
71
+ validate,
72
+ ...props
73
+ }, ref) {
74
+ const generatedId = useId();
75
+ const inputId = providedId ?? generatedId;
76
+ const formConfigContext = useContext(FormConfigContext);
77
+ const formConfig = formConfigContext ?? {};
78
+ const rhfContext = useFormContext();
79
+ const control = externalControl ?? rhfContext?.control;
80
+ useEffect(() => {
81
+ if (formConfigContext?.registerFieldValidation) {
82
+ const rules = {};
83
+ if (required !== void 0) rules.required = required;
84
+ if (minLength !== void 0) rules.minLength = minLength;
85
+ if (maxLength !== void 0) rules.maxLength = maxLength;
86
+ if (minValue !== void 0) rules.min = minValue;
87
+ if (maxValue !== void 0) rules.max = maxValue;
88
+ if (pattern !== void 0) rules.pattern = pattern;
89
+ if (email !== void 0) rules.email = email;
90
+ if (url !== void 0) rules.url = url;
91
+ if (validate !== void 0) rules.validate = validate;
92
+ const inputType = props.type ?? "text";
93
+ let fieldType = "string";
94
+ if (inputType === "number") {
95
+ fieldType = "number";
96
+ }
97
+ formConfigContext.registerFieldValidation({
98
+ name,
99
+ type: fieldType,
100
+ rules
101
+ });
102
+ return () => {
103
+ formConfigContext.unregisterFieldValidation(name);
104
+ };
105
+ }
106
+ }, [
107
+ formConfigContext,
108
+ name,
109
+ required,
110
+ minLength,
111
+ maxLength,
112
+ minValue,
113
+ maxValue,
114
+ pattern,
115
+ email,
116
+ url,
117
+ validate,
118
+ props.type
119
+ ]);
120
+ const { field, fieldState } = useController({
121
+ name,
122
+ control
123
+ });
124
+ const fieldError = fieldState.error?.message;
125
+ const errorMessage = customError ?? fieldError;
126
+ const hasError = !!errorMessage;
127
+ const effectiveSize = size ?? formConfig.size ?? "md";
128
+ const effectiveDisabled = disabled ?? formConfig.disabled;
129
+ const effectiveVariant = hasError ? "error" : variant;
130
+ const handleClear = () => {
131
+ field.onChange("");
132
+ onClear?.();
133
+ };
134
+ return /* @__PURE__ */ jsxs("div", { className: cn("space-y-1.5", fullWidth && "w-full"), children: [
135
+ label && /* @__PURE__ */ jsxs(
136
+ Label,
137
+ {
138
+ htmlFor: inputId,
139
+ className: labelVariants({ required: !!required }),
140
+ children: [
141
+ label,
142
+ formConfig.colon && ":"
143
+ ]
144
+ }
145
+ ),
146
+ /* @__PURE__ */ jsxs("div", { className: "relative", children: [
147
+ prefix && /* @__PURE__ */ jsx("div", { className: "absolute left-3 top-1/2 -translate-y-1/2 text-muted-foreground", children: prefix }),
148
+ /* @__PURE__ */ jsx(
149
+ Input,
150
+ {
151
+ ...props,
152
+ ...field,
153
+ ref: (node) => {
154
+ if (typeof ref === "function") {
155
+ ref(node);
156
+ } else if (ref) {
157
+ ref.current = node;
158
+ }
159
+ field.ref(node);
160
+ },
161
+ id: inputId,
162
+ disabled: effectiveDisabled,
163
+ "aria-invalid": hasError,
164
+ "aria-describedby": hasError ? `${inputId}-error` : helperText ? `${inputId}-helper` : void 0,
165
+ className: cn(
166
+ textBoxVariants({ size: effectiveSize, variant: effectiveVariant }),
167
+ prefix && "pl-10",
168
+ (suffix || allowClear) && "pr-10",
169
+ className
170
+ )
171
+ }
172
+ ),
173
+ (suffix || allowClear && field.value) && /* @__PURE__ */ jsxs("div", { className: "absolute right-3 top-1/2 -translate-y-1/2 flex items-center gap-1", children: [
174
+ allowClear && field.value && /* @__PURE__ */ jsx(
175
+ "button",
176
+ {
177
+ type: "button",
178
+ onClick: handleClear,
179
+ className: "text-muted-foreground hover:text-foreground transition-colors",
180
+ tabIndex: -1,
181
+ children: /* @__PURE__ */ jsx(X, { className: "h-4 w-4" })
182
+ }
183
+ ),
184
+ suffix && /* @__PURE__ */ jsx("span", { className: "text-muted-foreground", children: suffix })
185
+ ] })
186
+ ] }),
187
+ showError && hasError && /* @__PURE__ */ jsx(
188
+ "p",
189
+ {
190
+ id: `${inputId}-error`,
191
+ className: "text-sm text-destructive",
192
+ role: "alert",
193
+ children: errorMessage
194
+ }
195
+ ),
196
+ helperText && !hasError && /* @__PURE__ */ jsx(
197
+ "p",
198
+ {
199
+ id: `${inputId}-helper`,
200
+ className: "text-sm text-muted-foreground",
201
+ children: helperText
202
+ }
203
+ )
204
+ ] });
205
+ }
206
+ const TextBox = forwardRef(TextBoxInner);
207
+ TextBox.displayName = "TextBox";
208
+ export {
209
+ TextBox
210
+ };
211
+ //# sourceMappingURL=TextBox.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"TextBox.js","sources":["../../../src/components/Form/TextBox.tsx"],"sourcesContent":["import { forwardRef, useId, useContext, useEffect } from \"react\";\r\nimport type { InputHTMLAttributes, ReactNode } from \"react\";\r\nimport { useController, useFormContext as useRHFFormContext, type FieldValues, type FieldPath, type Control } from \"react-hook-form\";\r\nimport { cva, type VariantProps } from \"class-variance-authority\";\r\nimport { cn } from \"../../utils/cn\";\r\nimport { FormConfigContext, type FormConfig, type FieldValidationRules } from \"../Form/context\";\r\nimport { Input } from \"../ui/input\";\r\nimport { Label } from \"../ui/label\";\r\nimport { X } from \"lucide-react\";\r\n\r\nconst textBoxVariants = cva(\r\n \"\",\r\n {\r\n variants: {\r\n size: {\r\n sm: \"h-8 text-xs px-2\",\r\n md: \"h-10 text-sm px-3\",\r\n lg: \"h-12 text-base px-4\",\r\n },\r\n variant: {\r\n default: \"border-input focus-visible:ring-ring\",\r\n error: \"border-destructive focus-visible:ring-destructive\",\r\n success: \"border-green-500 focus-visible:ring-green-500\",\r\n },\r\n },\r\n defaultVariants: {\r\n size: \"md\",\r\n variant: \"default\",\r\n },\r\n }\r\n);\r\n\r\nconst labelVariants = cva(\r\n \"block text-sm font-medium mb-1.5\",\r\n {\r\n variants: {\r\n required: {\r\n true: \"after:content-['*'] after:ml-0.5 after:text-destructive\",\r\n false: \"\",\r\n },\r\n },\r\n defaultVariants: {\r\n required: false,\r\n },\r\n }\r\n);\r\n\r\n/**\r\n * Validation rule with optional custom message\r\n */\r\ntype ValidationRule<T> = T | { value: T; message: string };\r\n\r\nexport interface TextBoxProps<\r\n TFieldValues extends FieldValues = FieldValues,\r\n TName extends FieldPath<TFieldValues> = FieldPath<TFieldValues>\r\n> extends Omit<InputHTMLAttributes<HTMLInputElement>, \"size\" | \"name\" | \"prefix\" | \"pattern\" | \"required\" | \"minLength\" | \"maxLength\">,\r\n VariantProps<typeof textBoxVariants> {\r\n /** Field name - required for form integration */\r\n name: TName;\r\n /** Label text for the input */\r\n label?: string;\r\n /** Helper text displayed below the input */\r\n helperText?: string;\r\n /** Whether to show the error message */\r\n showError?: boolean;\r\n /** Custom error message (overrides form error) */\r\n error?: string;\r\n /** Whether the input should take full width */\r\n fullWidth?: boolean;\r\n /** Prefix element */\r\n prefix?: ReactNode;\r\n /** Suffix element */\r\n suffix?: ReactNode;\r\n /** Allow clear button */\r\n allowClear?: boolean;\r\n /** Callback when clear is clicked */\r\n onClear?: () => void;\r\n /** External control (for use outside Form) */\r\n control?: Control<TFieldValues>;\r\n \r\n // Validation props\r\n /** Field is required */\r\n required?: boolean | string;\r\n /** Minimum length for strings */\r\n minLength?: ValidationRule<number>;\r\n /** Maximum length for strings */\r\n maxLength?: ValidationRule<number>;\r\n /** Minimum value for numbers */\r\n minValue?: ValidationRule<number>;\r\n /** Maximum value for numbers */\r\n maxValue?: ValidationRule<number>;\r\n /** Regex pattern for validation */\r\n pattern?: ValidationRule<RegExp>;\r\n /** Email validation */\r\n email?: boolean | string;\r\n /** URL validation */\r\n url?: boolean | string;\r\n /** Custom validation function */\r\n validate?: (value: unknown) => boolean | string | Promise<boolean | string>;\r\n}\r\n\r\n/**\r\n * TextBox component with form integration and automatic validation registration\r\n * \r\n * This is a wrapper around the shadcn/ui Input primitive that adds:\r\n * - Form integration with react-hook-form\r\n * - Automatic validation registration\r\n * - Label, helper text, and error message support\r\n * - Prefix/suffix elements\r\n * - Clear button functionality\r\n * \r\n * @example\r\n * ```tsx\r\n * // Inside a Form component - validation is automatically registered\r\n * <Form onSubmit={handleSubmit} defaultValues={{ username: \"\", email: \"\" }}>\r\n * <TextBox name=\"username\" label=\"Username\" required minLength={3} maxLength={50} />\r\n * <TextBox name=\"email\" label=\"Email\" type=\"email\" required email />\r\n * <TextBox name=\"website\" label=\"Website\" url />\r\n * <Button type=\"submit\">Submit</Button>\r\n * </Form>\r\n * ```\r\n */\r\nfunction TextBoxInner<\r\n TFieldValues extends FieldValues = FieldValues,\r\n TName extends FieldPath<TFieldValues> = FieldPath<TFieldValues>\r\n>(\r\n {\r\n name,\r\n label,\r\n helperText,\r\n showError = true,\r\n error: customError,\r\n size,\r\n variant,\r\n fullWidth = true,\r\n className,\r\n disabled,\r\n prefix,\r\n suffix,\r\n allowClear,\r\n onClear,\r\n id: providedId,\r\n control: externalControl,\r\n // Validation props\r\n required,\r\n minLength,\r\n maxLength,\r\n minValue,\r\n maxValue,\r\n pattern,\r\n email,\r\n url,\r\n validate,\r\n ...props\r\n }: TextBoxProps<TFieldValues, TName>,\r\n ref: React.ForwardedRef<HTMLInputElement>\r\n) {\r\n const generatedId = useId();\r\n const inputId = providedId ?? generatedId;\r\n \r\n // Try to get form context\r\n const formConfigContext = useContext(FormConfigContext);\r\n const formConfig: FormConfig = formConfigContext ?? {};\r\n \r\n // Get form context from react-hook-form\r\n const rhfContext = useRHFFormContext<TFieldValues>();\r\n const control = externalControl ?? rhfContext?.control;\r\n\r\n // Register validation rules with the form\r\n useEffect(() => {\r\n if (formConfigContext?.registerFieldValidation) {\r\n const rules: FieldValidationRules = {};\r\n \r\n if (required !== undefined) rules.required = required;\r\n if (minLength !== undefined) rules.minLength = minLength;\r\n if (maxLength !== undefined) rules.maxLength = maxLength;\r\n if (minValue !== undefined) rules.min = minValue;\r\n if (maxValue !== undefined) rules.max = maxValue;\r\n if (pattern !== undefined) rules.pattern = pattern;\r\n if (email !== undefined) rules.email = email;\r\n if (url !== undefined) rules.url = url;\r\n if (validate !== undefined) rules.validate = validate;\r\n\r\n // Determine field type based on input type\r\n const inputType = props.type ?? \"text\";\r\n let fieldType: \"string\" | \"number\" = \"string\";\r\n if (inputType === \"number\") {\r\n fieldType = \"number\";\r\n }\r\n\r\n formConfigContext.registerFieldValidation({\r\n name: name as string,\r\n type: fieldType,\r\n rules,\r\n });\r\n\r\n return () => {\r\n formConfigContext.unregisterFieldValidation(name as string);\r\n };\r\n }\r\n }, [\r\n formConfigContext,\r\n name,\r\n required,\r\n minLength,\r\n maxLength,\r\n minValue,\r\n maxValue,\r\n pattern,\r\n email,\r\n url,\r\n validate,\r\n props.type,\r\n ]);\r\n\r\n // Use controller for form integration\r\n const { field, fieldState } = useController<TFieldValues, TName>({\r\n name,\r\n control,\r\n });\r\n\r\n const fieldError = fieldState.error?.message;\r\n const errorMessage = customError ?? fieldError;\r\n const hasError = !!errorMessage;\r\n \r\n // Merge sizes - prop takes precedence over form config\r\n const effectiveSize = size ?? formConfig.size ?? \"md\";\r\n const effectiveDisabled = disabled ?? formConfig.disabled;\r\n \r\n // Determine variant based on error state\r\n const effectiveVariant = hasError ? \"error\" : variant;\r\n\r\n const handleClear = () => {\r\n field.onChange(\"\");\r\n onClear?.();\r\n };\r\n\r\n return (\r\n <div className={cn(\"space-y-1.5\", fullWidth && \"w-full\")}>\r\n {label && (\r\n <Label\r\n htmlFor={inputId}\r\n className={labelVariants({ required: !!required })}\r\n >\r\n {label}\r\n {formConfig.colon && \":\"}\r\n </Label>\r\n )}\r\n \r\n <div className=\"relative\">\r\n {prefix && (\r\n <div className=\"absolute left-3 top-1/2 -translate-y-1/2 text-muted-foreground\">\r\n {prefix}\r\n </div>\r\n )}\r\n \r\n <Input\r\n {...props}\r\n {...field}\r\n ref={(node) => {\r\n // Handle both refs\r\n if (typeof ref === \"function\") {\r\n ref(node);\r\n } else if (ref) {\r\n ref.current = node;\r\n }\r\n field.ref(node);\r\n }}\r\n id={inputId}\r\n disabled={effectiveDisabled}\r\n aria-invalid={hasError}\r\n aria-describedby={\r\n hasError\r\n ? `${inputId}-error`\r\n : helperText\r\n ? `${inputId}-helper`\r\n : undefined\r\n }\r\n className={cn(\r\n textBoxVariants({ size: effectiveSize, variant: effectiveVariant }),\r\n prefix && \"pl-10\",\r\n (suffix || allowClear) && \"pr-10\",\r\n className\r\n )}\r\n />\r\n \r\n {(suffix || (allowClear && field.value)) && (\r\n <div className=\"absolute right-3 top-1/2 -translate-y-1/2 flex items-center gap-1\">\r\n {allowClear && field.value && (\r\n <button\r\n type=\"button\"\r\n onClick={handleClear}\r\n className=\"text-muted-foreground hover:text-foreground transition-colors\"\r\n tabIndex={-1}\r\n >\r\n <X className=\"h-4 w-4\" />\r\n </button>\r\n )}\r\n {suffix && (\r\n <span className=\"text-muted-foreground\">{suffix}</span>\r\n )}\r\n </div>\r\n )}\r\n </div>\r\n\r\n {showError && hasError && (\r\n <p\r\n id={`${inputId}-error`}\r\n className=\"text-sm text-destructive\"\r\n role=\"alert\"\r\n >\r\n {errorMessage}\r\n </p>\r\n )}\r\n \r\n {helperText && !hasError && (\r\n <p\r\n id={`${inputId}-helper`}\r\n className=\"text-sm text-muted-foreground\"\r\n >\r\n {helperText}\r\n </p>\r\n )}\r\n </div>\r\n );\r\n}\r\n\r\n// Use forwardRef with generic support\r\nexport const TextBox = forwardRef(TextBoxInner) as <\r\n TFieldValues extends FieldValues = FieldValues,\r\n TName extends FieldPath<TFieldValues> = FieldPath<TFieldValues>\r\n>(\r\n props: TextBoxProps<TFieldValues, TName> & { ref?: React.ForwardedRef<HTMLInputElement> }\r\n) => React.ReactElement;\r\n\r\n(TextBox as React.FC).displayName = \"TextBox\";\r\n"],"names":["useRHFFormContext"],"mappings":"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"}