@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.
- package/LICENSE +21 -0
- package/README.md +1 -0
- package/dist/cjs/components/Button/Button.cjs +37 -0
- package/dist/cjs/components/Button/Button.cjs.map +1 -0
- package/dist/cjs/components/Form/Checkbox.cjs +145 -0
- package/dist/cjs/components/Form/Checkbox.cjs.map +1 -0
- package/dist/cjs/components/Form/Form.cjs +93 -0
- package/dist/cjs/components/Form/Form.cjs.map +1 -0
- package/dist/cjs/components/Form/Radio.cjs +178 -0
- package/dist/cjs/components/Form/Radio.cjs.map +1 -0
- package/dist/cjs/components/Form/Select.cjs +168 -0
- package/dist/cjs/components/Form/Select.cjs.map +1 -0
- package/dist/cjs/components/Form/Switch.cjs +125 -0
- package/dist/cjs/components/Form/Switch.cjs.map +1 -0
- package/dist/cjs/components/Form/TextArea.cjs +168 -0
- package/dist/cjs/components/Form/TextArea.cjs.map +1 -0
- package/dist/cjs/components/Form/TextBox.cjs +211 -0
- package/dist/cjs/components/Form/TextBox.cjs.map +1 -0
- package/dist/cjs/components/Form/context.cjs +131 -0
- package/dist/cjs/components/Form/context.cjs.map +1 -0
- package/dist/cjs/components/Form/hooks.cjs +36 -0
- package/dist/cjs/components/Form/hooks.cjs.map +1 -0
- package/dist/cjs/components/Input/Input.cjs +44 -0
- package/dist/cjs/components/Input/Input.cjs.map +1 -0
- package/dist/cjs/components/ui/checkbox.cjs +45 -0
- package/dist/cjs/components/ui/checkbox.cjs.map +1 -0
- package/dist/cjs/components/ui/input.cjs +41 -0
- package/dist/cjs/components/ui/input.cjs.map +1 -0
- package/dist/cjs/components/ui/label.cjs +38 -0
- package/dist/cjs/components/ui/label.cjs.map +1 -0
- package/dist/cjs/components/ui/radio-group.cjs +53 -0
- package/dist/cjs/components/ui/radio-group.cjs.map +1 -0
- package/dist/cjs/components/ui/select.cjs +144 -0
- package/dist/cjs/components/ui/select.cjs.map +1 -0
- package/dist/cjs/components/ui/switch.cjs +45 -0
- package/dist/cjs/components/ui/switch.cjs.map +1 -0
- package/dist/cjs/components/ui/textarea.cjs +38 -0
- package/dist/cjs/components/ui/textarea.cjs.map +1 -0
- package/dist/cjs/hooks/useDebounce.cjs +17 -0
- package/dist/cjs/hooks/useDebounce.cjs.map +1 -0
- package/dist/cjs/hooks/useToggle.cjs +12 -0
- package/dist/cjs/hooks/useToggle.cjs.map +1 -0
- package/dist/cjs/index.cjs +65 -0
- package/dist/cjs/index.cjs.map +1 -0
- package/dist/cjs/nebula-ui.css +212 -0
- package/dist/cjs/utils/cn.cjs +9 -0
- package/dist/cjs/utils/cn.cjs.map +1 -0
- package/dist/components/Button/Button.d.ts +18 -0
- package/dist/components/Button/Button.d.ts.map +1 -0
- package/dist/components/Button/Button.js +37 -0
- package/dist/components/Button/Button.js.map +1 -0
- package/dist/components/Button/index.d.ts +3 -0
- package/dist/components/Button/index.d.ts.map +1 -0
- package/dist/components/Form/Checkbox.d.ts +32 -0
- package/dist/components/Form/Checkbox.d.ts.map +1 -0
- package/dist/components/Form/Checkbox.js +145 -0
- package/dist/components/Form/Checkbox.js.map +1 -0
- package/dist/components/Form/Form.d.ts +62 -0
- package/dist/components/Form/Form.d.ts.map +1 -0
- package/dist/components/Form/Form.js +93 -0
- package/dist/components/Form/Form.js.map +1 -0
- package/dist/components/Form/Radio.d.ts +80 -0
- package/dist/components/Form/Radio.d.ts.map +1 -0
- package/dist/components/Form/Radio.js +178 -0
- package/dist/components/Form/Radio.js.map +1 -0
- package/dist/components/Form/Select.d.ts +75 -0
- package/dist/components/Form/Select.d.ts.map +1 -0
- package/dist/components/Form/Select.js +168 -0
- package/dist/components/Form/Select.js.map +1 -0
- package/dist/components/Form/Switch.d.ts +34 -0
- package/dist/components/Form/Switch.d.ts.map +1 -0
- package/dist/components/Form/Switch.js +125 -0
- package/dist/components/Form/Switch.js.map +1 -0
- package/dist/components/Form/TextArea.d.ts +47 -0
- package/dist/components/Form/TextArea.d.ts.map +1 -0
- package/dist/components/Form/TextArea.js +168 -0
- package/dist/components/Form/TextArea.js.map +1 -0
- package/dist/components/Form/TextBox.d.ts +61 -0
- package/dist/components/Form/TextBox.d.ts.map +1 -0
- package/dist/components/Form/TextBox.js +211 -0
- package/dist/components/Form/TextBox.js.map +1 -0
- package/dist/components/Form/context.d.ts +107 -0
- package/dist/components/Form/context.d.ts.map +1 -0
- package/dist/components/Form/context.js +131 -0
- package/dist/components/Form/context.js.map +1 -0
- package/dist/components/Form/hooks.d.ts +21 -0
- package/dist/components/Form/hooks.d.ts.map +1 -0
- package/dist/components/Form/hooks.js +36 -0
- package/dist/components/Form/hooks.js.map +1 -0
- package/dist/components/Form/index.d.ts +18 -0
- package/dist/components/Form/index.d.ts.map +1 -0
- package/dist/components/Input/Input.d.ts +18 -0
- package/dist/components/Input/Input.d.ts.map +1 -0
- package/dist/components/Input/Input.js +44 -0
- package/dist/components/Input/Input.js.map +1 -0
- package/dist/components/Input/index.d.ts +3 -0
- package/dist/components/Input/index.d.ts.map +1 -0
- package/dist/components/ui/checkbox.d.ts +5 -0
- package/dist/components/ui/checkbox.d.ts.map +1 -0
- package/dist/components/ui/checkbox.js +28 -0
- package/dist/components/ui/checkbox.js.map +1 -0
- package/dist/components/ui/index.d.ts +16 -0
- package/dist/components/ui/index.d.ts.map +1 -0
- package/dist/components/ui/input.d.ts +11 -0
- package/dist/components/ui/input.d.ts.map +1 -0
- package/dist/components/ui/input.js +24 -0
- package/dist/components/ui/input.js.map +1 -0
- package/dist/components/ui/label.d.ts +6 -0
- package/dist/components/ui/label.d.ts.map +1 -0
- package/dist/components/ui/label.js +21 -0
- package/dist/components/ui/label.js.map +1 -0
- package/dist/components/ui/radio-group.d.ts +6 -0
- package/dist/components/ui/radio-group.d.ts.map +1 -0
- package/dist/components/ui/radio-group.js +36 -0
- package/dist/components/ui/radio-group.js.map +1 -0
- package/dist/components/ui/select.d.ts +14 -0
- package/dist/components/ui/select.d.ts.map +1 -0
- package/dist/components/ui/select.js +127 -0
- package/dist/components/ui/select.js.map +1 -0
- package/dist/components/ui/switch.d.ts +5 -0
- package/dist/components/ui/switch.d.ts.map +1 -0
- package/dist/components/ui/switch.js +28 -0
- package/dist/components/ui/switch.js.map +1 -0
- package/dist/components/ui/textarea.d.ts +11 -0
- package/dist/components/ui/textarea.d.ts.map +1 -0
- package/dist/components/ui/textarea.js +21 -0
- package/dist/components/ui/textarea.js.map +1 -0
- package/dist/hooks/useDebounce.d.ts +8 -0
- package/dist/hooks/useDebounce.d.ts.map +1 -0
- package/dist/hooks/useDebounce.js +17 -0
- package/dist/hooks/useDebounce.js.map +1 -0
- package/dist/hooks/useToggle.d.ts +7 -0
- package/dist/hooks/useToggle.d.ts.map +1 -0
- package/dist/hooks/useToggle.js +12 -0
- package/dist/hooks/useToggle.js.map +1 -0
- package/dist/index.d.ts +13 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +62 -0
- package/dist/index.js.map +1 -0
- package/dist/nebula-ui.css +212 -0
- package/dist/utils/cn.d.ts +7 -0
- package/dist/utils/cn.d.ts.map +1 -0
- package/dist/utils/cn.js +9 -0
- package/dist/utils/cn.js.map +1 -0
- package/package.json +201 -0
|
@@ -0,0 +1,168 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, Symbol.toStringTag, { value: "Module" });
|
|
3
|
+
const jsxRuntime = require("react/jsx-runtime");
|
|
4
|
+
const React = require("react");
|
|
5
|
+
const reactHookForm = require("react-hook-form");
|
|
6
|
+
const index = require("../../node_modules/class-variance-authority/dist/index.cjs");
|
|
7
|
+
const cn = require("../../utils/cn.cjs");
|
|
8
|
+
const context = require("./context.cjs");
|
|
9
|
+
const select = require("../ui/select.cjs");
|
|
10
|
+
const label = require("../ui/label.cjs");
|
|
11
|
+
const selectSizeVariants = index.cva(
|
|
12
|
+
"",
|
|
13
|
+
{
|
|
14
|
+
variants: {
|
|
15
|
+
size: {
|
|
16
|
+
sm: "h-8 text-xs",
|
|
17
|
+
md: "h-10 text-sm",
|
|
18
|
+
lg: "h-12 text-base"
|
|
19
|
+
}
|
|
20
|
+
},
|
|
21
|
+
defaultVariants: {
|
|
22
|
+
size: "md"
|
|
23
|
+
}
|
|
24
|
+
}
|
|
25
|
+
);
|
|
26
|
+
const labelVariants = index.cva(
|
|
27
|
+
"block text-sm font-medium mb-1.5",
|
|
28
|
+
{
|
|
29
|
+
variants: {
|
|
30
|
+
required: {
|
|
31
|
+
true: "after:content-['*'] after:ml-0.5 after:text-destructive",
|
|
32
|
+
false: ""
|
|
33
|
+
}
|
|
34
|
+
},
|
|
35
|
+
defaultVariants: {
|
|
36
|
+
required: false
|
|
37
|
+
}
|
|
38
|
+
}
|
|
39
|
+
);
|
|
40
|
+
function Select({
|
|
41
|
+
name,
|
|
42
|
+
label: label$1,
|
|
43
|
+
helperText,
|
|
44
|
+
showError = true,
|
|
45
|
+
error: customError,
|
|
46
|
+
size,
|
|
47
|
+
variant,
|
|
48
|
+
fullWidth = true,
|
|
49
|
+
className,
|
|
50
|
+
disabled,
|
|
51
|
+
options = [],
|
|
52
|
+
placeholder,
|
|
53
|
+
allowClear,
|
|
54
|
+
id: providedId,
|
|
55
|
+
control: externalControl,
|
|
56
|
+
children,
|
|
57
|
+
// Validation props
|
|
58
|
+
required,
|
|
59
|
+
validate
|
|
60
|
+
}) {
|
|
61
|
+
const generatedId = React.useId();
|
|
62
|
+
const inputId = providedId ?? generatedId;
|
|
63
|
+
const formConfigContext = React.useContext(context.FormConfigContext);
|
|
64
|
+
const formConfig = formConfigContext ?? {};
|
|
65
|
+
const rhfContext = reactHookForm.useFormContext();
|
|
66
|
+
const control = externalControl ?? rhfContext?.control;
|
|
67
|
+
React.useEffect(() => {
|
|
68
|
+
if (formConfigContext?.registerFieldValidation) {
|
|
69
|
+
const rules = {};
|
|
70
|
+
if (required !== void 0) rules.required = required;
|
|
71
|
+
if (validate !== void 0) rules.validate = validate;
|
|
72
|
+
formConfigContext.registerFieldValidation({
|
|
73
|
+
name,
|
|
74
|
+
type: "string",
|
|
75
|
+
rules
|
|
76
|
+
});
|
|
77
|
+
return () => {
|
|
78
|
+
formConfigContext.unregisterFieldValidation(name);
|
|
79
|
+
};
|
|
80
|
+
}
|
|
81
|
+
}, [formConfigContext, name, required, validate]);
|
|
82
|
+
const { field, fieldState } = reactHookForm.useController({
|
|
83
|
+
name,
|
|
84
|
+
control
|
|
85
|
+
});
|
|
86
|
+
const fieldError = fieldState.error?.message;
|
|
87
|
+
const errorMessage = customError ?? fieldError;
|
|
88
|
+
const hasError = !!errorMessage;
|
|
89
|
+
const effectiveSize = size ?? formConfig.size ?? "md";
|
|
90
|
+
const effectiveDisabled = disabled ?? formConfig.disabled;
|
|
91
|
+
const effectiveVariant = hasError ? "error" : variant;
|
|
92
|
+
return /* @__PURE__ */ jsxRuntime.jsxs("div", { className: cn.cn("space-y-1.5", fullWidth && "w-full"), children: [
|
|
93
|
+
label$1 && /* @__PURE__ */ jsxRuntime.jsxs(
|
|
94
|
+
label.Label,
|
|
95
|
+
{
|
|
96
|
+
htmlFor: inputId,
|
|
97
|
+
className: labelVariants({ required: !!required }),
|
|
98
|
+
children: [
|
|
99
|
+
label$1,
|
|
100
|
+
formConfig.colon && ":"
|
|
101
|
+
]
|
|
102
|
+
}
|
|
103
|
+
),
|
|
104
|
+
/* @__PURE__ */ jsxRuntime.jsxs(
|
|
105
|
+
select.Select,
|
|
106
|
+
{
|
|
107
|
+
value: field.value?.toString() ?? "",
|
|
108
|
+
onValueChange: (value) => {
|
|
109
|
+
if (value === "" && allowClear) {
|
|
110
|
+
field.onChange("");
|
|
111
|
+
} else {
|
|
112
|
+
field.onChange(value);
|
|
113
|
+
}
|
|
114
|
+
},
|
|
115
|
+
disabled: effectiveDisabled,
|
|
116
|
+
children: [
|
|
117
|
+
/* @__PURE__ */ jsxRuntime.jsx(
|
|
118
|
+
select.SelectTrigger,
|
|
119
|
+
{
|
|
120
|
+
id: inputId,
|
|
121
|
+
"aria-invalid": hasError,
|
|
122
|
+
"aria-describedby": hasError ? `${inputId}-error` : helperText ? `${inputId}-helper` : void 0,
|
|
123
|
+
className: cn.cn(
|
|
124
|
+
selectSizeVariants({ size: effectiveSize }),
|
|
125
|
+
effectiveVariant === "error" && "border-destructive focus:ring-destructive",
|
|
126
|
+
effectiveVariant === "success" && "border-green-500 focus:ring-green-500",
|
|
127
|
+
className
|
|
128
|
+
),
|
|
129
|
+
children: /* @__PURE__ */ jsxRuntime.jsx(select.SelectValue, { placeholder })
|
|
130
|
+
}
|
|
131
|
+
),
|
|
132
|
+
/* @__PURE__ */ jsxRuntime.jsxs(select.SelectContent, { children: [
|
|
133
|
+
allowClear && field.value && /* @__PURE__ */ jsxRuntime.jsx(select.SelectItem, { value: "", children: /* @__PURE__ */ jsxRuntime.jsx("span", { className: "text-muted-foreground", children: "Clear selection" }) }),
|
|
134
|
+
children ?? options.map((option) => /* @__PURE__ */ jsxRuntime.jsx(
|
|
135
|
+
select.SelectItem,
|
|
136
|
+
{
|
|
137
|
+
value: option.value.toString(),
|
|
138
|
+
disabled: option.disabled,
|
|
139
|
+
children: option.label
|
|
140
|
+
},
|
|
141
|
+
option.value
|
|
142
|
+
))
|
|
143
|
+
] })
|
|
144
|
+
]
|
|
145
|
+
}
|
|
146
|
+
),
|
|
147
|
+
showError && hasError && /* @__PURE__ */ jsxRuntime.jsx(
|
|
148
|
+
"p",
|
|
149
|
+
{
|
|
150
|
+
id: `${inputId}-error`,
|
|
151
|
+
className: "text-sm text-destructive",
|
|
152
|
+
role: "alert",
|
|
153
|
+
children: errorMessage
|
|
154
|
+
}
|
|
155
|
+
),
|
|
156
|
+
helperText && !hasError && /* @__PURE__ */ jsxRuntime.jsx(
|
|
157
|
+
"p",
|
|
158
|
+
{
|
|
159
|
+
id: `${inputId}-helper`,
|
|
160
|
+
className: "text-sm text-muted-foreground",
|
|
161
|
+
children: helperText
|
|
162
|
+
}
|
|
163
|
+
)
|
|
164
|
+
] });
|
|
165
|
+
}
|
|
166
|
+
Select.displayName = "Select";
|
|
167
|
+
exports.Select = Select;
|
|
168
|
+
//# sourceMappingURL=Select.cjs.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"Select.cjs","sources":["../../../../src/components/Form/Select.tsx"],"sourcesContent":["import { useId, useContext, useEffect } from \"react\";\r\nimport type { 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 {\r\n Select as ShadcnSelect,\r\n SelectContent,\r\n SelectItem,\r\n SelectTrigger,\r\n SelectValue,\r\n} from \"../ui/select\";\r\nimport { Label } from \"../ui/label\";\r\n\r\nconst selectSizeVariants = cva(\r\n \"\",\r\n {\r\n variants: {\r\n size: {\r\n sm: \"h-8 text-xs\",\r\n md: \"h-10 text-sm\",\r\n lg: \"h-12 text-base\",\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 \"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\nexport interface SelectOption {\r\n label: string;\r\n value: string | number;\r\n disabled?: boolean;\r\n}\r\n\r\nexport interface SelectProps<\r\n TFieldValues extends FieldValues = FieldValues,\r\n TName extends FieldPath<TFieldValues> = FieldPath<TFieldValues>\r\n> extends VariantProps<typeof selectSizeVariants> {\r\n /** Field name - required for form integration */\r\n name: TName;\r\n /** Label text for the select */\r\n label?: string;\r\n /** Helper text displayed below the select */\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 select should take full width */\r\n fullWidth?: boolean;\r\n /** Options for the select */\r\n options?: SelectOption[];\r\n /** Placeholder text */\r\n placeholder?: string;\r\n /** Allow clear selection */\r\n allowClear?: boolean;\r\n /** External control (for use outside Form) */\r\n control?: Control<TFieldValues>;\r\n /** Children (alternative to options prop) */\r\n children?: ReactNode;\r\n /** Disabled state */\r\n disabled?: boolean;\r\n /** Additional class name */\r\n className?: string;\r\n /** ID for the select */\r\n id?: string;\r\n /** Variant style */\r\n variant?: \"default\" | \"error\" | \"success\";\r\n \r\n // Validation props\r\n /** Field is required */\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 * Select component with form integration\r\n * \r\n * This is a wrapper around the shadcn/ui Select 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 * \r\n * @example\r\n * ```tsx\r\n * // Inside a Form component\r\n * <Select \r\n * name=\"country\" \r\n * label=\"Country\" \r\n * required\r\n * options={[\r\n * { label: \"USA\", value: \"us\" },\r\n * { label: \"Canada\", value: \"ca\" },\r\n * ]} \r\n * />\r\n * ```\r\n */\r\nexport function Select<\r\n TFieldValues extends FieldValues = FieldValues,\r\n TName extends FieldPath<TFieldValues> = FieldPath<TFieldValues>\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 options = [],\r\n placeholder,\r\n allowClear,\r\n id: providedId,\r\n control: externalControl,\r\n children,\r\n // Validation props\r\n required,\r\n validate,\r\n}: SelectProps<TFieldValues, TName>) {\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: \"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, 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 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 <ShadcnSelect\r\n value={field.value?.toString() ?? \"\"}\r\n onValueChange={(value) => {\r\n // Handle clear selection\r\n if (value === \"\" && allowClear) {\r\n field.onChange(\"\");\r\n } else {\r\n field.onChange(value);\r\n }\r\n }}\r\n disabled={effectiveDisabled}\r\n >\r\n <SelectTrigger\r\n id={inputId}\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 selectSizeVariants({ size: effectiveSize }),\r\n effectiveVariant === \"error\" && \"border-destructive focus:ring-destructive\",\r\n effectiveVariant === \"success\" && \"border-green-500 focus:ring-green-500\",\r\n className\r\n )}\r\n >\r\n <SelectValue placeholder={placeholder} />\r\n </SelectTrigger>\r\n <SelectContent>\r\n {allowClear && field.value && (\r\n <SelectItem value=\"\">\r\n <span className=\"text-muted-foreground\">Clear selection</span>\r\n </SelectItem>\r\n )}\r\n {children ?? options.map((option) => (\r\n <SelectItem\r\n key={option.value}\r\n value={option.value.toString()}\r\n disabled={option.disabled}\r\n >\r\n {option.label}\r\n </SelectItem>\r\n ))}\r\n </SelectContent>\r\n </ShadcnSelect>\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\nSelect.displayName = \"Select\";\r\n"],"names":["cva","label","useId","useContext","FormConfigContext","useRHFFormContext","useEffect","useController","cn","jsxs","Label","ShadcnSelect","jsx","SelectTrigger","SelectValue","SelectContent","SelectItem"],"mappings":"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"}
|
|
@@ -0,0 +1,125 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, Symbol.toStringTag, { value: "Module" });
|
|
3
|
+
const jsxRuntime = require("react/jsx-runtime");
|
|
4
|
+
const React = require("react");
|
|
5
|
+
const reactHookForm = require("react-hook-form");
|
|
6
|
+
const index = require("../../node_modules/class-variance-authority/dist/index.cjs");
|
|
7
|
+
const cn = require("../../utils/cn.cjs");
|
|
8
|
+
const context = require("./context.cjs");
|
|
9
|
+
const _switch = require("../ui/switch.cjs");
|
|
10
|
+
const label = require("../ui/label.cjs");
|
|
11
|
+
const switchSizeVariants = index.cva(
|
|
12
|
+
"",
|
|
13
|
+
{
|
|
14
|
+
variants: {
|
|
15
|
+
size: {
|
|
16
|
+
sm: "h-5 w-9",
|
|
17
|
+
md: "h-6 w-11",
|
|
18
|
+
lg: "h-7 w-14"
|
|
19
|
+
}
|
|
20
|
+
},
|
|
21
|
+
defaultVariants: {
|
|
22
|
+
size: "md"
|
|
23
|
+
}
|
|
24
|
+
}
|
|
25
|
+
);
|
|
26
|
+
const labelVariants = index.cva(
|
|
27
|
+
"text-sm font-medium cursor-pointer select-none",
|
|
28
|
+
{
|
|
29
|
+
variants: {
|
|
30
|
+
disabled: {
|
|
31
|
+
true: "cursor-not-allowed opacity-50",
|
|
32
|
+
false: ""
|
|
33
|
+
}
|
|
34
|
+
},
|
|
35
|
+
defaultVariants: {
|
|
36
|
+
disabled: false
|
|
37
|
+
}
|
|
38
|
+
}
|
|
39
|
+
);
|
|
40
|
+
function SwitchInner({
|
|
41
|
+
name,
|
|
42
|
+
label: label$1,
|
|
43
|
+
helperText,
|
|
44
|
+
showError = true,
|
|
45
|
+
error: customError,
|
|
46
|
+
size,
|
|
47
|
+
className,
|
|
48
|
+
disabled,
|
|
49
|
+
id: providedId,
|
|
50
|
+
control: externalControl,
|
|
51
|
+
checkedText,
|
|
52
|
+
uncheckedText,
|
|
53
|
+
// Validation props
|
|
54
|
+
required,
|
|
55
|
+
validate,
|
|
56
|
+
...props
|
|
57
|
+
}, ref) {
|
|
58
|
+
const generatedId = React.useId();
|
|
59
|
+
const inputId = providedId ?? generatedId;
|
|
60
|
+
const formConfigContext = React.useContext(context.FormConfigContext);
|
|
61
|
+
const formConfig = formConfigContext ?? {};
|
|
62
|
+
const rhfContext = reactHookForm.useFormContext();
|
|
63
|
+
const control = externalControl ?? rhfContext?.control;
|
|
64
|
+
React.useEffect(() => {
|
|
65
|
+
if (formConfigContext?.registerFieldValidation) {
|
|
66
|
+
const rules = {};
|
|
67
|
+
if (required !== void 0) rules.required = required;
|
|
68
|
+
if (validate !== void 0) rules.validate = validate;
|
|
69
|
+
formConfigContext.registerFieldValidation({
|
|
70
|
+
name,
|
|
71
|
+
type: "boolean",
|
|
72
|
+
rules
|
|
73
|
+
});
|
|
74
|
+
return () => {
|
|
75
|
+
formConfigContext.unregisterFieldValidation(name);
|
|
76
|
+
};
|
|
77
|
+
}
|
|
78
|
+
}, [formConfigContext, name, required, validate]);
|
|
79
|
+
const { field, fieldState } = reactHookForm.useController({
|
|
80
|
+
name,
|
|
81
|
+
control
|
|
82
|
+
});
|
|
83
|
+
const fieldError = fieldState.error?.message;
|
|
84
|
+
const errorMessage = customError ?? fieldError;
|
|
85
|
+
const hasError = !!errorMessage;
|
|
86
|
+
const effectiveSize = size ?? formConfig.size ?? "md";
|
|
87
|
+
const effectiveDisabled = disabled ?? formConfig.disabled;
|
|
88
|
+
const isChecked = field.value ?? false;
|
|
89
|
+
return /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "space-y-1.5", children: [
|
|
90
|
+
/* @__PURE__ */ jsxRuntime.jsxs("div", { className: "flex items-center gap-3", children: [
|
|
91
|
+
/* @__PURE__ */ jsxRuntime.jsx(
|
|
92
|
+
_switch.Switch,
|
|
93
|
+
{
|
|
94
|
+
...props,
|
|
95
|
+
ref,
|
|
96
|
+
id: inputId,
|
|
97
|
+
checked: isChecked,
|
|
98
|
+
onCheckedChange: (checked) => field.onChange(checked),
|
|
99
|
+
disabled: effectiveDisabled,
|
|
100
|
+
"aria-labelledby": label$1 ? `${inputId}-label` : void 0,
|
|
101
|
+
className: cn.cn(
|
|
102
|
+
switchSizeVariants({ size: effectiveSize }),
|
|
103
|
+
className
|
|
104
|
+
)
|
|
105
|
+
}
|
|
106
|
+
),
|
|
107
|
+
(checkedText || uncheckedText) && /* @__PURE__ */ jsxRuntime.jsx("span", { className: "text-sm text-muted-foreground", children: isChecked ? checkedText : uncheckedText }),
|
|
108
|
+
label$1 && /* @__PURE__ */ jsxRuntime.jsx(
|
|
109
|
+
label.Label,
|
|
110
|
+
{
|
|
111
|
+
id: `${inputId}-label`,
|
|
112
|
+
htmlFor: inputId,
|
|
113
|
+
className: labelVariants({ disabled: effectiveDisabled }),
|
|
114
|
+
children: label$1
|
|
115
|
+
}
|
|
116
|
+
)
|
|
117
|
+
] }),
|
|
118
|
+
showError && hasError && /* @__PURE__ */ jsxRuntime.jsx("p", { className: "text-sm text-destructive", role: "alert", children: errorMessage }),
|
|
119
|
+
helperText && !hasError && /* @__PURE__ */ jsxRuntime.jsx("p", { className: "text-sm text-muted-foreground", children: helperText })
|
|
120
|
+
] });
|
|
121
|
+
}
|
|
122
|
+
const Switch = React.forwardRef(SwitchInner);
|
|
123
|
+
Switch.displayName = "Switch";
|
|
124
|
+
exports.Switch = Switch;
|
|
125
|
+
//# sourceMappingURL=Switch.cjs.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"Switch.cjs","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":["cva","label","useId","useContext","FormConfigContext","useRHFFormContext","useEffect","useController","jsxs","jsx","ShadcnSwitch","cn","Label","forwardRef"],"mappings":";;;;;;;;;;AAUA,MAAM,qBAAqBA,MAAAA;AAAAA,EACzB;AAAA,EACA;AAAA,IACE,UAAU;AAAA,MACR,MAAM;AAAA,QACJ,IAAI;AAAA,QACJ,IAAI;AAAA,QACJ,IAAI;AAAA,MAAA;AAAA,IACN;AAAA,IAEF,iBAAiB;AAAA,MACf,MAAM;AAAA,IAAA;AAAA,EACR;AAEJ;AAEA,MAAM,gBAAgBA,MAAAA;AAAAA,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;AAgDA,SAAS,YAIP;AAAA,EACE;AAAA,EAAA,OACAC;AAAAA,EACA;AAAA,EACA,YAAY;AAAA,EACZ,OAAO;AAAA,EACP;AAAA,EACA;AAAA,EACA;AAAA,EACA,IAAI;AAAA,EACJ,SAAS;AAAA,EACT;AAAA,EACA;AAAA;AAAA,EAEA;AAAA,EACA;AAAA,EACA,GAAG;AACL,GACA,KACA;AACA,QAAM,cAAcC,MAAAA,MAAA;AACpB,QAAM,UAAU,cAAc;AAG9B,QAAM,oBAAoBC,MAAAA,WAAWC,yBAAiB;AACtD,QAAM,aAAyB,qBAAqB,CAAA;AAGpD,QAAM,aAAaC,cAAAA,eAAA;AACnB,QAAM,UAAU,mBAAmB,YAAY;AAG/CC,QAAAA,UAAU,MAAM;AACd,QAAI,mBAAmB,yBAAyB;AAC9C,YAAM,QAA8B,CAAA;AAEpC,UAAI,aAAa,OAAW,OAAM,WAAW;AAC7C,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,QAAQ,CAAC;AAGhD,QAAM,EAAE,OAAO,WAAA,IAAeC,4BAAmC;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;AACjD,QAAM,YAAY,MAAM,SAAS;AAEjC,SACEC,2BAAAA,KAAC,OAAA,EAAI,WAAU,eACb,UAAA;AAAA,IAAAA,2BAAAA,KAAC,OAAA,EAAI,WAAU,2BACb,UAAA;AAAA,MAAAC,2BAAAA;AAAAA,QAACC,QAAAA;AAAAA,QAAA;AAAA,UACE,GAAG;AAAA,UACJ;AAAA,UACA,IAAI;AAAA,UACJ,SAAS;AAAA,UACT,iBAAiB,CAAC,YAAY,MAAM,SAAS,OAAO;AAAA,UACpD,UAAU;AAAA,UACV,mBAAiBT,UAAQ,GAAG,OAAO,WAAW;AAAA,UAC9C,WAAWU,GAAAA;AAAAA,YACT,mBAAmB,EAAE,MAAM,eAAe;AAAA,YAC1C;AAAA,UAAA;AAAA,QACF;AAAA,MAAA;AAAA,OAGA,eAAe,kBACfF,2BAAAA,IAAC,QAAA,EAAK,WAAU,iCACb,UAAA,YAAY,cAAc,cAAA,CAC7B;AAAA,MAGDR,WACCQ,2BAAAA;AAAAA,QAACG,MAAAA;AAAAA,QAAA;AAAA,UACC,IAAI,GAAG,OAAO;AAAA,UACd,SAAS;AAAA,UACT,WAAW,cAAc,EAAE,UAAU,mBAAmB;AAAA,UAEvD,UAAAX;AAAAA,QAAA;AAAA,MAAA;AAAA,IACH,GAEJ;AAAA,IAEC,aAAa,YACZQ,+BAAC,KAAA,EAAE,WAAU,4BAA2B,MAAK,SAC1C,UAAA,aAAA,CACH;AAAA,IAGD,cAAc,CAAC,2CACb,KAAA,EAAE,WAAU,iCACV,UAAA,WAAA,CACH;AAAA,EAAA,GAEJ;AAEJ;AAGO,MAAM,SAASI,MAAAA,WAAW,WAAW;AAO3C,OAAoB,cAAc;;"}
|
|
@@ -0,0 +1,168 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, Symbol.toStringTag, { value: "Module" });
|
|
3
|
+
const jsxRuntime = require("react/jsx-runtime");
|
|
4
|
+
const React = require("react");
|
|
5
|
+
const reactHookForm = require("react-hook-form");
|
|
6
|
+
const index = require("../../node_modules/class-variance-authority/dist/index.cjs");
|
|
7
|
+
const cn = require("../../utils/cn.cjs");
|
|
8
|
+
const context = require("./context.cjs");
|
|
9
|
+
const textarea = require("../ui/textarea.cjs");
|
|
10
|
+
const label = require("../ui/label.cjs");
|
|
11
|
+
const textAreaSizeVariants = index.cva(
|
|
12
|
+
"",
|
|
13
|
+
{
|
|
14
|
+
variants: {
|
|
15
|
+
size: {
|
|
16
|
+
sm: "text-xs px-2 py-1.5 min-h-[60px]",
|
|
17
|
+
md: "text-sm px-3 py-2 min-h-[80px]",
|
|
18
|
+
lg: "text-base px-4 py-3 min-h-[100px]"
|
|
19
|
+
},
|
|
20
|
+
variant: {
|
|
21
|
+
default: "border-input focus-visible:ring-ring",
|
|
22
|
+
error: "border-destructive focus-visible:ring-destructive",
|
|
23
|
+
success: "border-green-500 focus-visible:ring-green-500"
|
|
24
|
+
}
|
|
25
|
+
},
|
|
26
|
+
defaultVariants: {
|
|
27
|
+
size: "md",
|
|
28
|
+
variant: "default"
|
|
29
|
+
}
|
|
30
|
+
}
|
|
31
|
+
);
|
|
32
|
+
const labelVariants = index.cva(
|
|
33
|
+
"block text-sm font-medium mb-1.5",
|
|
34
|
+
{
|
|
35
|
+
variants: {
|
|
36
|
+
required: {
|
|
37
|
+
true: "after:content-['*'] after:ml-0.5 after:text-destructive",
|
|
38
|
+
false: ""
|
|
39
|
+
}
|
|
40
|
+
},
|
|
41
|
+
defaultVariants: {
|
|
42
|
+
required: false
|
|
43
|
+
}
|
|
44
|
+
}
|
|
45
|
+
);
|
|
46
|
+
function TextAreaInner({
|
|
47
|
+
name,
|
|
48
|
+
label: label$1,
|
|
49
|
+
helperText,
|
|
50
|
+
showError = true,
|
|
51
|
+
error: customError,
|
|
52
|
+
size,
|
|
53
|
+
variant,
|
|
54
|
+
fullWidth = true,
|
|
55
|
+
className,
|
|
56
|
+
disabled,
|
|
57
|
+
showCount,
|
|
58
|
+
maxCount,
|
|
59
|
+
id: providedId,
|
|
60
|
+
control: externalControl,
|
|
61
|
+
// Validation props
|
|
62
|
+
required,
|
|
63
|
+
minLength,
|
|
64
|
+
maxLength,
|
|
65
|
+
validate,
|
|
66
|
+
...props
|
|
67
|
+
}, ref) {
|
|
68
|
+
const generatedId = React.useId();
|
|
69
|
+
const inputId = providedId ?? generatedId;
|
|
70
|
+
const formConfigContext = React.useContext(context.FormConfigContext);
|
|
71
|
+
const formConfig = formConfigContext ?? {};
|
|
72
|
+
const rhfContext = reactHookForm.useFormContext();
|
|
73
|
+
const control = externalControl ?? rhfContext?.control;
|
|
74
|
+
React.useEffect(() => {
|
|
75
|
+
if (formConfigContext?.registerFieldValidation) {
|
|
76
|
+
const rules = {};
|
|
77
|
+
if (required !== void 0) rules.required = required;
|
|
78
|
+
if (minLength !== void 0) rules.minLength = minLength;
|
|
79
|
+
if (maxLength !== void 0) rules.maxLength = maxLength;
|
|
80
|
+
if (validate !== void 0) rules.validate = validate;
|
|
81
|
+
formConfigContext.registerFieldValidation({
|
|
82
|
+
name,
|
|
83
|
+
type: "string",
|
|
84
|
+
rules
|
|
85
|
+
});
|
|
86
|
+
return () => {
|
|
87
|
+
formConfigContext.unregisterFieldValidation(name);
|
|
88
|
+
};
|
|
89
|
+
}
|
|
90
|
+
}, [formConfigContext, name, required, minLength, maxLength, validate]);
|
|
91
|
+
const { field, fieldState } = reactHookForm.useController({
|
|
92
|
+
name,
|
|
93
|
+
control
|
|
94
|
+
});
|
|
95
|
+
const fieldError = fieldState.error?.message;
|
|
96
|
+
const errorMessage = customError ?? fieldError;
|
|
97
|
+
const hasError = !!errorMessage;
|
|
98
|
+
const effectiveSize = size ?? formConfig.size ?? "md";
|
|
99
|
+
const effectiveDisabled = disabled ?? formConfig.disabled;
|
|
100
|
+
const effectiveVariant = hasError ? "error" : variant;
|
|
101
|
+
const currentLength = typeof field.value === "string" ? field.value.length : 0;
|
|
102
|
+
const maxLengthValue = maxCount ?? (typeof maxLength === "number" ? maxLength : maxLength?.value);
|
|
103
|
+
return /* @__PURE__ */ jsxRuntime.jsxs("div", { className: cn.cn("space-y-1.5", fullWidth && "w-full"), children: [
|
|
104
|
+
label$1 && /* @__PURE__ */ jsxRuntime.jsxs(
|
|
105
|
+
label.Label,
|
|
106
|
+
{
|
|
107
|
+
htmlFor: inputId,
|
|
108
|
+
className: labelVariants({ required: !!required }),
|
|
109
|
+
children: [
|
|
110
|
+
label$1,
|
|
111
|
+
formConfig.colon && ":"
|
|
112
|
+
]
|
|
113
|
+
}
|
|
114
|
+
),
|
|
115
|
+
/* @__PURE__ */ jsxRuntime.jsxs("div", { className: "relative", children: [
|
|
116
|
+
/* @__PURE__ */ jsxRuntime.jsx(
|
|
117
|
+
textarea.Textarea,
|
|
118
|
+
{
|
|
119
|
+
...props,
|
|
120
|
+
...field,
|
|
121
|
+
ref: (node) => {
|
|
122
|
+
if (typeof ref === "function") {
|
|
123
|
+
ref(node);
|
|
124
|
+
} else if (ref) {
|
|
125
|
+
ref.current = node;
|
|
126
|
+
}
|
|
127
|
+
field.ref(node);
|
|
128
|
+
},
|
|
129
|
+
id: inputId,
|
|
130
|
+
disabled: effectiveDisabled,
|
|
131
|
+
maxLength: maxLengthValue,
|
|
132
|
+
"aria-invalid": hasError,
|
|
133
|
+
"aria-describedby": hasError ? `${inputId}-error` : helperText ? `${inputId}-helper` : void 0,
|
|
134
|
+
className: cn.cn(
|
|
135
|
+
textAreaSizeVariants({ size: effectiveSize, variant: effectiveVariant }),
|
|
136
|
+
showCount && "pb-6",
|
|
137
|
+
className
|
|
138
|
+
)
|
|
139
|
+
}
|
|
140
|
+
),
|
|
141
|
+
showCount && /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "absolute bottom-2 right-2 text-xs text-muted-foreground", children: [
|
|
142
|
+
currentLength,
|
|
143
|
+
maxCount ? `/${maxCount}` : ""
|
|
144
|
+
] })
|
|
145
|
+
] }),
|
|
146
|
+
showError && hasError && /* @__PURE__ */ jsxRuntime.jsx(
|
|
147
|
+
"p",
|
|
148
|
+
{
|
|
149
|
+
id: `${inputId}-error`,
|
|
150
|
+
className: "text-sm text-destructive",
|
|
151
|
+
role: "alert",
|
|
152
|
+
children: errorMessage
|
|
153
|
+
}
|
|
154
|
+
),
|
|
155
|
+
helperText && !hasError && /* @__PURE__ */ jsxRuntime.jsx(
|
|
156
|
+
"p",
|
|
157
|
+
{
|
|
158
|
+
id: `${inputId}-helper`,
|
|
159
|
+
className: "text-sm text-muted-foreground",
|
|
160
|
+
children: helperText
|
|
161
|
+
}
|
|
162
|
+
)
|
|
163
|
+
] });
|
|
164
|
+
}
|
|
165
|
+
const TextArea = React.forwardRef(TextAreaInner);
|
|
166
|
+
TextArea.displayName = "TextArea";
|
|
167
|
+
exports.TextArea = TextArea;
|
|
168
|
+
//# sourceMappingURL=TextArea.cjs.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"TextArea.cjs","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":["cva","label","useId","useContext","FormConfigContext","useRHFFormContext","useEffect","useController","cn","jsxs","Label","jsx","Textarea","forwardRef"],"mappings":"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"}
|