@page-speed/forms 0.5.6 → 0.5.8

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 (53) hide show
  1. package/dist/FormContext-089h0j0Q.d.ts +153 -0
  2. package/dist/FormContext-iHzBALZa.d.cts +153 -0
  3. package/dist/{chunk-A7R6GEMA.js → chunk-455PI4LV.js} +84 -3
  4. package/dist/chunk-455PI4LV.js.map +1 -0
  5. package/dist/{chunk-QZ4PK62A.js → chunk-4ROWNTY6.js} +335 -417
  6. package/dist/chunk-4ROWNTY6.js.map +1 -0
  7. package/dist/{chunk-WVSLOSNU.cjs → chunk-ED4UK63G.cjs} +440 -523
  8. package/dist/chunk-ED4UK63G.cjs.map +1 -0
  9. package/dist/chunk-IXHDJ7E5.js +263 -0
  10. package/dist/chunk-IXHDJ7E5.js.map +1 -0
  11. package/dist/{chunk-Q7VR374Y.js → chunk-MJYEXJ3U.js} +3 -3
  12. package/dist/{chunk-Q7VR374Y.js.map → chunk-MJYEXJ3U.js.map} +1 -1
  13. package/dist/{chunk-KPHMVGTU.cjs → chunk-MUBEMXI7.cjs} +6 -6
  14. package/dist/{chunk-KPHMVGTU.cjs.map → chunk-MUBEMXI7.cjs.map} +1 -1
  15. package/dist/chunk-Q7WXLZN5.cjs +287 -0
  16. package/dist/chunk-Q7WXLZN5.cjs.map +1 -0
  17. package/dist/{chunk-QQSBW6N3.cjs → chunk-QRI5TMES.cjs} +85 -2
  18. package/dist/chunk-QRI5TMES.cjs.map +1 -0
  19. package/dist/core.cjs +13 -9
  20. package/dist/core.d.cts +66 -141
  21. package/dist/core.d.ts +66 -141
  22. package/dist/core.js +3 -3
  23. package/dist/index.cjs +14 -14
  24. package/dist/index.d.cts +2 -2
  25. package/dist/index.d.ts +2 -2
  26. package/dist/index.js +3 -3
  27. package/dist/inputs.cjs +16 -16
  28. package/dist/inputs.d.cts +1 -1
  29. package/dist/inputs.d.ts +1 -1
  30. package/dist/inputs.js +2 -2
  31. package/dist/integration.cjs +17 -17
  32. package/dist/integration.d.cts +1 -1
  33. package/dist/integration.d.ts +1 -1
  34. package/dist/integration.js +3 -3
  35. package/dist/{types-CnOCn7b3.d.ts → types-BemGfSPA.d.cts} +27 -1
  36. package/dist/{types-CnOCn7b3.d.cts → types-BemGfSPA.d.ts} +27 -1
  37. package/dist/validation-rules.d.cts +1 -1
  38. package/dist/validation-rules.d.ts +1 -1
  39. package/dist/validation-utils.d.cts +1 -1
  40. package/dist/validation-utils.d.ts +1 -1
  41. package/dist/validation-valibot.d.cts +1 -1
  42. package/dist/validation-valibot.d.ts +1 -1
  43. package/dist/validation.d.cts +1 -1
  44. package/dist/validation.d.ts +1 -1
  45. package/package.json +1 -1
  46. package/dist/chunk-A7R6GEMA.js.map +0 -1
  47. package/dist/chunk-N2HOVRRN.js +0 -150
  48. package/dist/chunk-N2HOVRRN.js.map +0 -1
  49. package/dist/chunk-O4ZLR7AN.cjs +0 -173
  50. package/dist/chunk-O4ZLR7AN.cjs.map +0 -1
  51. package/dist/chunk-QQSBW6N3.cjs.map +0 -1
  52. package/dist/chunk-QZ4PK62A.js.map +0 -1
  53. package/dist/chunk-WVSLOSNU.cjs.map +0 -1
@@ -0,0 +1,153 @@
1
+ import { b as FormValues, U as UseFormOptions, m as UseFormReturn, p as UseFieldOptions, q as UseFieldReturn, r as FormProps, s as FieldProps } from './types-BemGfSPA.js';
2
+ import * as React from 'react';
3
+
4
+ /**
5
+ * useForm - High-performance form state management with field-level reactivity
6
+ *
7
+ * Built on @legendapp/state for optimal performance:
8
+ * - Field-level reactivity: Only re-render the specific field that changed
9
+ * - Observable-based state: ~1 re-render per change vs ~10 for traditional hooks
10
+ * - Tree-shakable: Only bundle what you use
11
+ *
12
+ * @example
13
+ * ```tsx
14
+ * const form = useForm({
15
+ * initialValues: { email: '', password: '' },
16
+ * onSubmit: async (values) => {
17
+ * await login(values);
18
+ * },
19
+ * validationSchema: {
20
+ * email: (value) => !value ? 'Required' : undefined,
21
+ * password: (value) => value.length < 8 ? 'Too short' : undefined,
22
+ * },
23
+ * });
24
+ *
25
+ * return (
26
+ * <form onSubmit={form.handleSubmit}>
27
+ * <input {...form.getFieldProps('email')} />
28
+ * {form.errors.email && <span>{form.errors.email}</span>}
29
+ * </form>
30
+ * );
31
+ * ```
32
+ *
33
+ * @see https://opensite.ai/developers/page-speed/forms/use-form
34
+ */
35
+ declare function useForm<T extends FormValues = FormValues>(options: UseFormOptions<T>): UseFormReturn<T>;
36
+
37
+ /**
38
+ * useField - Field-level reactive hook for form inputs
39
+ *
40
+ * Provides isolated reactivity for individual form fields.
41
+ * Only re-renders when the specific field changes, not when other fields update.
42
+ *
43
+ * Must be used within a FormContext (inside <Form> component).
44
+ *
45
+ * @example
46
+ * ```tsx
47
+ * function EmailInput() {
48
+ * const { field, meta, helpers } = useField({ name: 'email' });
49
+ *
50
+ * return (
51
+ * <div>
52
+ * <input {...field} type="email" />
53
+ * {meta.touched && meta.error && <span>{meta.error}</span>}
54
+ * </div>
55
+ * );
56
+ * }
57
+ * ```
58
+ *
59
+ * @see https://opensite.ai/developers/page-speed/forms/use-field
60
+ */
61
+ declare function useField<T = any>(options: UseFieldOptions<T>): UseFieldReturn<T>;
62
+
63
+ /**
64
+ * Form - Progressive enhancement form component
65
+ *
66
+ * Provides form context to child components and handles form submission.
67
+ * Supports progressive enhancement with server-side fallback.
68
+ *
69
+ * Features:
70
+ * - Provides FormContext for useField hook
71
+ * - Handles form submission with validation
72
+ * - Progressive enhancement support (works without JavaScript)
73
+ * - Accessible form semantics
74
+ *
75
+ * @example
76
+ * ```tsx
77
+ * const form = useForm({
78
+ * initialValues: { email: '' },
79
+ * onSubmit: async (values) => {
80
+ * await submitForm(values);
81
+ * },
82
+ * });
83
+ *
84
+ * return (
85
+ * <Form form={form} action="/api/submit" method="post">
86
+ * <input {...form.getFieldProps('email')} />
87
+ * <button type="submit">Submit</button>
88
+ * </Form>
89
+ * );
90
+ * ```
91
+ *
92
+ * @see https://opensite.ai/developers/page-speed/forms/form
93
+ */
94
+ declare function Form<T extends FormValues = FormValues>({ form, children, fields, className, action, method, noValidate, submissionConfig, successMessage, submissionError, successMessageClassName, errorMessageClassName, onNewSubmission, notificationConfig, styleConfig, formConfig, ...props }: FormProps<T> & React.FormHTMLAttributes<HTMLFormElement>): React.JSX.Element;
95
+ declare namespace Form {
96
+ var displayName: string;
97
+ }
98
+
99
+ /**
100
+ * Field - Field wrapper component with label, description, and error display
101
+ *
102
+ * Provides a complete field UI with automatic error handling and accessibility.
103
+ * Uses useField hook internally for field-level reactivity.
104
+ *
105
+ * Features:
106
+ * - Automatic label association
107
+ * - Error display with accessibility
108
+ * - Optional description text
109
+ * - Render prop pattern for flexibility
110
+ * - Full accessibility support
111
+ *
112
+ * @example
113
+ * ```tsx
114
+ * <Field name="email" label="Email Address" description="We'll never share your email">
115
+ * {({ field, meta }) => (
116
+ * <input
117
+ * {...field}
118
+ * type="email"
119
+ * className={meta.error && meta.touched ? 'error' : ''}
120
+ * />
121
+ * )}
122
+ * </Field>
123
+ * ```
124
+ *
125
+ * @see https://opensite.ai/developers/page-speed/forms/field
126
+ */
127
+ declare function Field({ name, label, description, children, showError, className, errorClassName, required, validate, }: FieldProps): React.JSX.Element;
128
+ declare namespace Field {
129
+ var displayName: string;
130
+ }
131
+
132
+ interface FormFeedbackProps {
133
+ successMessage?: React.ReactNode;
134
+ submissionError?: React.ReactNode;
135
+ successMessageClassName?: string;
136
+ errorMessageClassName?: string;
137
+ }
138
+ declare function FormFeedback({ successMessage, submissionError, successMessageClassName, errorMessageClassName, }: FormFeedbackProps): React.JSX.Element | null;
139
+ declare namespace FormFeedback {
140
+ var displayName: string;
141
+ }
142
+
143
+ /**
144
+ * FormContext - React context for providing form state to child components
145
+ *
146
+ * Allows useField hook to access form state without prop drilling.
147
+ * Automatically provided by the <Form> component.
148
+ *
149
+ * @internal
150
+ */
151
+ declare const FormContext: React.Context<UseFormReturn<any> | null>;
152
+
153
+ export { Form as F, useField as a, Field as b, FormFeedback as c, FormContext as d, type FormFeedbackProps as e, useForm as u };
@@ -0,0 +1,153 @@
1
+ import { b as FormValues, U as UseFormOptions, m as UseFormReturn, p as UseFieldOptions, q as UseFieldReturn, r as FormProps, s as FieldProps } from './types-BemGfSPA.cjs';
2
+ import * as React from 'react';
3
+
4
+ /**
5
+ * useForm - High-performance form state management with field-level reactivity
6
+ *
7
+ * Built on @legendapp/state for optimal performance:
8
+ * - Field-level reactivity: Only re-render the specific field that changed
9
+ * - Observable-based state: ~1 re-render per change vs ~10 for traditional hooks
10
+ * - Tree-shakable: Only bundle what you use
11
+ *
12
+ * @example
13
+ * ```tsx
14
+ * const form = useForm({
15
+ * initialValues: { email: '', password: '' },
16
+ * onSubmit: async (values) => {
17
+ * await login(values);
18
+ * },
19
+ * validationSchema: {
20
+ * email: (value) => !value ? 'Required' : undefined,
21
+ * password: (value) => value.length < 8 ? 'Too short' : undefined,
22
+ * },
23
+ * });
24
+ *
25
+ * return (
26
+ * <form onSubmit={form.handleSubmit}>
27
+ * <input {...form.getFieldProps('email')} />
28
+ * {form.errors.email && <span>{form.errors.email}</span>}
29
+ * </form>
30
+ * );
31
+ * ```
32
+ *
33
+ * @see https://opensite.ai/developers/page-speed/forms/use-form
34
+ */
35
+ declare function useForm<T extends FormValues = FormValues>(options: UseFormOptions<T>): UseFormReturn<T>;
36
+
37
+ /**
38
+ * useField - Field-level reactive hook for form inputs
39
+ *
40
+ * Provides isolated reactivity for individual form fields.
41
+ * Only re-renders when the specific field changes, not when other fields update.
42
+ *
43
+ * Must be used within a FormContext (inside <Form> component).
44
+ *
45
+ * @example
46
+ * ```tsx
47
+ * function EmailInput() {
48
+ * const { field, meta, helpers } = useField({ name: 'email' });
49
+ *
50
+ * return (
51
+ * <div>
52
+ * <input {...field} type="email" />
53
+ * {meta.touched && meta.error && <span>{meta.error}</span>}
54
+ * </div>
55
+ * );
56
+ * }
57
+ * ```
58
+ *
59
+ * @see https://opensite.ai/developers/page-speed/forms/use-field
60
+ */
61
+ declare function useField<T = any>(options: UseFieldOptions<T>): UseFieldReturn<T>;
62
+
63
+ /**
64
+ * Form - Progressive enhancement form component
65
+ *
66
+ * Provides form context to child components and handles form submission.
67
+ * Supports progressive enhancement with server-side fallback.
68
+ *
69
+ * Features:
70
+ * - Provides FormContext for useField hook
71
+ * - Handles form submission with validation
72
+ * - Progressive enhancement support (works without JavaScript)
73
+ * - Accessible form semantics
74
+ *
75
+ * @example
76
+ * ```tsx
77
+ * const form = useForm({
78
+ * initialValues: { email: '' },
79
+ * onSubmit: async (values) => {
80
+ * await submitForm(values);
81
+ * },
82
+ * });
83
+ *
84
+ * return (
85
+ * <Form form={form} action="/api/submit" method="post">
86
+ * <input {...form.getFieldProps('email')} />
87
+ * <button type="submit">Submit</button>
88
+ * </Form>
89
+ * );
90
+ * ```
91
+ *
92
+ * @see https://opensite.ai/developers/page-speed/forms/form
93
+ */
94
+ declare function Form<T extends FormValues = FormValues>({ form, children, fields, className, action, method, noValidate, submissionConfig, successMessage, submissionError, successMessageClassName, errorMessageClassName, onNewSubmission, notificationConfig, styleConfig, formConfig, ...props }: FormProps<T> & React.FormHTMLAttributes<HTMLFormElement>): React.JSX.Element;
95
+ declare namespace Form {
96
+ var displayName: string;
97
+ }
98
+
99
+ /**
100
+ * Field - Field wrapper component with label, description, and error display
101
+ *
102
+ * Provides a complete field UI with automatic error handling and accessibility.
103
+ * Uses useField hook internally for field-level reactivity.
104
+ *
105
+ * Features:
106
+ * - Automatic label association
107
+ * - Error display with accessibility
108
+ * - Optional description text
109
+ * - Render prop pattern for flexibility
110
+ * - Full accessibility support
111
+ *
112
+ * @example
113
+ * ```tsx
114
+ * <Field name="email" label="Email Address" description="We'll never share your email">
115
+ * {({ field, meta }) => (
116
+ * <input
117
+ * {...field}
118
+ * type="email"
119
+ * className={meta.error && meta.touched ? 'error' : ''}
120
+ * />
121
+ * )}
122
+ * </Field>
123
+ * ```
124
+ *
125
+ * @see https://opensite.ai/developers/page-speed/forms/field
126
+ */
127
+ declare function Field({ name, label, description, children, showError, className, errorClassName, required, validate, }: FieldProps): React.JSX.Element;
128
+ declare namespace Field {
129
+ var displayName: string;
130
+ }
131
+
132
+ interface FormFeedbackProps {
133
+ successMessage?: React.ReactNode;
134
+ submissionError?: React.ReactNode;
135
+ successMessageClassName?: string;
136
+ errorMessageClassName?: string;
137
+ }
138
+ declare function FormFeedback({ successMessage, submissionError, successMessageClassName, errorMessageClassName, }: FormFeedbackProps): React.JSX.Element | null;
139
+ declare namespace FormFeedback {
140
+ var displayName: string;
141
+ }
142
+
143
+ /**
144
+ * FormContext - React context for providing form state to child components
145
+ *
146
+ * Allows useField hook to access form state without prop drilling.
147
+ * Automatically provided by the <Form> component.
148
+ *
149
+ * @internal
150
+ */
151
+ declare const FormContext: React.Context<UseFormReturn<any> | null>;
152
+
153
+ export { Form as F, useField as a, Field as b, FormFeedback as c, FormContext as d, type FormFeedbackProps as e, useForm as u };
@@ -105,6 +105,87 @@ var FieldError = React2.forwardRef(({ className, ...props }, ref) => {
105
105
  );
106
106
  });
107
107
  FieldError.displayName = "FieldError";
108
+ var Input = React2.forwardRef(
109
+ ({ className, type, ...props }, ref) => {
110
+ return /* @__PURE__ */ React2.createElement(
111
+ "input",
112
+ {
113
+ ref,
114
+ type,
115
+ "data-slot": "input",
116
+ className: cn(
117
+ // Core structure - no hardcoded colors, uses CSS variables
118
+ "flex h-9 w-full min-w-0 rounded-md border border-input",
119
+ "bg-transparent px-3 py-1 text-base shadow-sm",
120
+ "transition-colors outline-none md:text-sm",
121
+ // Focus state - uses ring-ring CSS variable (adapts to theme)
122
+ "focus-visible:ring-1 focus-visible:ring-ring",
123
+ // Error state - uses destructive CSS variables (adapts to theme)
124
+ "aria-invalid:border-destructive aria-invalid:ring-1 aria-invalid:ring-destructive",
125
+ // Disabled state - no color hardcoding
126
+ "disabled:pointer-events-none disabled:cursor-not-allowed disabled:opacity-50",
127
+ // File input specific - inherits text color from parent
128
+ "file:inline-flex file:h-7 file:border-0 file:bg-transparent",
129
+ "file:text-sm file:font-medium",
130
+ // Autofill reset - prevents browser from overriding our dynamic colors
131
+ INPUT_AUTOFILL_RESET_CLASSES,
132
+ className
133
+ ),
134
+ ...props
135
+ }
136
+ );
137
+ }
138
+ );
139
+ Input.displayName = "Input";
140
+
141
+ // src/inputs/TextInput.tsx
142
+ function TextInput({
143
+ name,
144
+ value,
145
+ onChange,
146
+ onBlur,
147
+ placeholder,
148
+ disabled = false,
149
+ required = false,
150
+ error = false,
151
+ className = "",
152
+ type = "text",
153
+ id = "text",
154
+ ...props
155
+ }) {
156
+ const handleChange = (e) => {
157
+ onChange(e.target.value);
158
+ };
159
+ const handleBlur = () => {
160
+ onBlur?.();
161
+ };
162
+ const hasValue = String(value ?? "").trim().length > 0;
163
+ return /* @__PURE__ */ React2.createElement(
164
+ Input,
165
+ {
166
+ type,
167
+ id,
168
+ name,
169
+ value: value ?? "",
170
+ onChange: handleChange,
171
+ onBlur: handleBlur,
172
+ placeholder,
173
+ disabled,
174
+ required,
175
+ className: cn(
176
+ // Valid value indicator - ring-2 when has value and no error
177
+ !error && hasValue && "ring-2 ring-ring",
178
+ // Error state - handled by Input component via aria-invalid
179
+ className
180
+ ),
181
+ "aria-invalid": error || props["aria-invalid"],
182
+ "aria-describedby": props["aria-describedby"],
183
+ "aria-required": required || props["aria-required"],
184
+ ...props
185
+ }
186
+ );
187
+ }
188
+ TextInput.displayName = "TextInput";
108
189
  var buttonVariants = cva(
109
190
  "inline-flex items-center justify-center gap-2 whitespace-nowrap rounded-md text-sm font-medium transition-all disabled:pointer-events-none disabled:opacity-50 [&_svg]:pointer-events-none [&_svg:not([class*='size-'])]:size-4 shrink-0 [&_svg]:shrink-0 outline-none focus-visible:border-ring focus-visible:ring-ring/50 focus-visible:ring-[3px] aria-invalid:ring-destructive/20 aria-invalid:border-destructive",
110
191
  {
@@ -202,6 +283,6 @@ var LabelGroup = ({
202
283
  return /* @__PURE__ */ React2.createElement("div", { className: "flex flex-1 flex-col gap-0.5" }, primaryElement, secondaryElement);
203
284
  };
204
285
 
205
- export { Button, Field, FieldDescription, FieldError, FieldGroup, FieldLabel, INPUT_AUTOFILL_RESET_CLASSES, LabelGroup, buttonVariants, cn };
206
- //# sourceMappingURL=chunk-A7R6GEMA.js.map
207
- //# sourceMappingURL=chunk-A7R6GEMA.js.map
286
+ export { Button, Field, FieldDescription, FieldError, FieldGroup, FieldLabel, INPUT_AUTOFILL_RESET_CLASSES, Input, LabelGroup, TextInput, buttonVariants, cn };
287
+ //# sourceMappingURL=chunk-455PI4LV.js.map
288
+ //# sourceMappingURL=chunk-455PI4LV.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/lib/utils.ts","../src/components/ui/label.tsx","../src/components/ui/field.tsx","../src/components/ui/input.tsx","../src/inputs/TextInput.tsx","../src/components/ui/button.tsx","../src/core/label-group.tsx"],"names":["React","LabelPrimitive","React3","React4","React5","React6"],"mappings":"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","file":"chunk-455PI4LV.js","sourcesContent":["import { clsx, type ClassValue } from \"clsx\";\nimport { twMerge } from \"tailwind-merge\";\n\nexport function cn(...inputs: ClassValue[]) {\n return twMerge(clsx(inputs));\n}\n\n/**\n * Normalizes browser autofill colors so inputs keep theme colors.\n */\nexport const INPUT_AUTOFILL_RESET_CLASSES =\n \"autofill:bg-transparent autofill:text-foreground \" +\n \"[&:-webkit-autofill]:[-webkit-text-fill-color:hsl(var(--foreground))] \" +\n \"[&:-webkit-autofill]:[caret-color:hsl(var(--foreground))] \" +\n \"[&:-webkit-autofill]:[box-shadow:0_0_0px_1000px_hsl(var(--background))_inset] \" +\n \"[&:-webkit-autofill:hover]:[box-shadow:0_0_0px_1000px_hsl(var(--background))_inset] \" +\n \"[&:-webkit-autofill:focus]:[box-shadow:0_0_0px_1000px_hsl(var(--background))_inset] \" +\n \"[&:-webkit-autofill]:[transition:background-color_9999s_ease-out,color_9999s_ease-out]\";\n","import * as React from \"react\"\nimport { Label as LabelPrimitive } from \"radix-ui\"\n\nimport { cn } from \"../../lib/utils\"\n\nfunction Label({\n className,\n ...props\n}: React.ComponentProps<typeof LabelPrimitive.Root>) {\n return (\n <LabelPrimitive.Root\n data-slot=\"label\"\n className={cn(\n \"flex items-center gap-2 text-sm leading-none font-medium select-none group-data-[disabled=true]:pointer-events-none group-data-[disabled=true]:opacity-50 peer-disabled:cursor-not-allowed peer-disabled:opacity-50\",\n className\n )}\n {...props}\n />\n )\n}\n\nexport { Label }\n","import * as React from \"react\";\nimport { Label } from \"./label\";\nimport { cn } from \"../../lib/utils\";\n\ntype FieldOrientation = \"vertical\" | \"horizontal\";\n\ninterface FieldProps extends React.HTMLAttributes<HTMLDivElement> {\n orientation?: FieldOrientation;\n invalid?: boolean;\n}\n\n/**\n * Field - Container component for form inputs with validation display\n *\n * Provides consistent layout and spacing for form fields with labels,\n * inputs, descriptions, and error messages.\n */\nconst Field = React.forwardRef<HTMLDivElement, FieldProps>(\n({ className, orientation = \"vertical\", invalid = false, ...props }, ref) => {\n return (\n <div\n ref={ref}\n data-slot=\"field\"\n data-orientation={orientation}\n data-invalid={invalid || undefined}\n className={cn(\n // Use space-y instead of flex to avoid interfering with parent grid layouts\n orientation === \"horizontal\"\n ? \"flex items-center gap-2\"\n : \"space-y-1.5\",\n className,\n )}\n {...props}\n />\n );\n});\nField.displayName = \"Field\";\n\n/**\n * FieldGroup - Container for multiple related fields\n *\n * Used to group fields together (e.g., first name + last name in a row)\n */\nconst FieldGroup = React.forwardRef<\n HTMLDivElement,\n React.HTMLAttributes<HTMLDivElement>\n>(({ className, ...props }, ref) => {\n return (\n <div\n ref={ref}\n data-slot=\"field-group\"\n className={cn(\"flex flex-col gap-4\", className)}\n {...props}\n />\n );\n});\nFieldGroup.displayName = \"FieldGroup\";\n\n/**\n * FieldLabel - Label component for form fields\n *\n * Wrapper around ShadCN Label with consistent styling\n */\nconst FieldLabel = React.forwardRef<\n HTMLLabelElement,\n React.LabelHTMLAttributes<HTMLLabelElement> & {\n required?: boolean;\n }\n>(({ className, required, children, ...props }, ref) => {\n return (\n <Label\n ref={ref}\n data-slot=\"field-label\"\n className={cn(\n \"text-sm font-medium leading-none select-none\",\n \"peer-disabled:cursor-not-allowed peer-disabled:opacity-50\",\n className,\n )}\n {...props}\n >\n {children}\n {required && <span className=\"text-destructive ml-1\">*</span>}\n </Label>\n );\n});\nFieldLabel.displayName = \"FieldLabel\";\n\n/**\n * FieldDescription - Helper text for form fields\n *\n * Displays additional information or instructions for the field\n */\nconst FieldDescription = React.forwardRef<\n HTMLParagraphElement,\n React.HTMLAttributes<HTMLParagraphElement>\n>(({ className, ...props }, ref) => {\n return (\n <p\n ref={ref}\n data-slot=\"field-description\"\n className={cn(\"text-sm opacity-70\", className)}\n {...props}\n />\n );\n});\nFieldDescription.displayName = \"FieldDescription\";\n\n/**\n * FieldError - Error message display for form fields\n *\n * Shows validation errors with proper styling and accessibility\n */\nconst FieldError = React.forwardRef<\n HTMLParagraphElement,\n React.HTMLAttributes<HTMLParagraphElement>\n>(({ className, ...props }, ref) => {\n return (\n <p\n ref={ref}\n data-slot=\"field-error\"\n role=\"alert\"\n aria-live=\"polite\"\n className={cn(\"text-sm text-destructive\", className)}\n {...props}\n />\n );\n});\nFieldError.displayName = \"FieldError\";\n\nexport { Field, FieldGroup, FieldLabel, FieldDescription, FieldError };\n","import * as React from \"react\";\nimport { cn } from \"../../lib/utils\";\nimport { INPUT_AUTOFILL_RESET_CLASSES } from \"../../lib/utils\";\n\n/**\n * Input component - Optimized for dynamic theming across thousands of client brands\n *\n * CRITICAL: This component must work with dynamic Section backgrounds (light/dark/primary/etc)\n * Only uses CSS variables that adapt automatically - NO hardcoded semantic colors\n *\n * See: SHADCN_INTEGRATION_GUIDE.md for full documentation\n */\nconst Input = React.forwardRef<HTMLInputElement, React.ComponentProps<\"input\">>(\n ({ className, type, ...props }, ref) => {\n return (\n <input\n ref={ref}\n type={type}\n data-slot=\"input\"\n className={cn(\n // Core structure - no hardcoded colors, uses CSS variables\n \"flex h-9 w-full min-w-0 rounded-md border border-input\",\n \"bg-transparent px-3 py-1 text-base shadow-sm\",\n \"transition-colors outline-none md:text-sm\",\n\n // Focus state - uses ring-ring CSS variable (adapts to theme)\n \"focus-visible:ring-1 focus-visible:ring-ring\",\n\n // Error state - uses destructive CSS variables (adapts to theme)\n \"aria-invalid:border-destructive aria-invalid:ring-1 aria-invalid:ring-destructive\",\n\n // Disabled state - no color hardcoding\n \"disabled:pointer-events-none disabled:cursor-not-allowed disabled:opacity-50\",\n\n // File input specific - inherits text color from parent\n \"file:inline-flex file:h-7 file:border-0 file:bg-transparent\",\n \"file:text-sm file:font-medium\",\n\n // Autofill reset - prevents browser from overriding our dynamic colors\n INPUT_AUTOFILL_RESET_CLASSES,\n\n className,\n )}\n {...props}\n />\n );\n },\n);\nInput.displayName = \"Input\";\n\nexport { Input };\n","\"use client\";\n\nimport * as React from \"react\";\nimport type { InputProps } from \"../core/types\";\nimport { Input } from \"../components/ui/input\";\nimport { cn } from \"../lib/utils\";\n\n/**\n * TextInput - High-performance text input component (ShadCN-based)\n *\n * Built on ShadCN Input component with form-specific behavior:\n * - Error state handling\n * - Valid value indicator (ring-2)\n * - Form integration (onChange, onBlur)\n * - Full accessibility support\n *\n * @example\n * ```tsx\n * const form = useForm({ initialValues: { email: '' } });\n *\n * <TextInput\n * {...form.getFieldProps('email')}\n * type=\"email\"\n * placeholder=\"Enter your email\"\n * error={!!form.errors.email}\n * aria-invalid={!!form.errors.email}\n * aria-describedby={form.errors.email ? 'email-error' : undefined}\n * />\n * ```\n *\n * @see https://opensite.ai/developers/page-speed/forms/text-input\n */\nexport function TextInput({\n name,\n value,\n onChange,\n onBlur,\n placeholder,\n disabled = false,\n required = false,\n error = false,\n className = \"\",\n type = \"text\",\n id = \"text\",\n ...props\n}: InputProps<string> & {\n type?: \"text\" | \"email\" | \"password\" | \"url\" | \"tel\" | \"search\";\n}) {\n const handleChange = (e: React.ChangeEvent<HTMLInputElement>) => {\n onChange(e.target.value);\n };\n\n const handleBlur = () => {\n onBlur?.();\n };\n\n const hasValue = String(value ?? \"\").trim().length > 0;\n\n return (\n <Input\n type={type}\n id={id}\n name={name}\n value={value ?? \"\"}\n onChange={handleChange}\n onBlur={handleBlur}\n placeholder={placeholder}\n disabled={disabled}\n required={required}\n className={cn(\n // Valid value indicator - ring-2 when has value and no error\n !error && hasValue && \"ring-2 ring-ring\",\n // Error state - handled by Input component via aria-invalid\n className,\n )}\n aria-invalid={error || props[\"aria-invalid\"]}\n aria-describedby={props[\"aria-describedby\"]}\n aria-required={required || props[\"aria-required\"]}\n {...props}\n />\n );\n}\n\nTextInput.displayName = \"TextInput\";\n","import * as React from \"react\"\nimport { cva, type VariantProps } from \"class-variance-authority\"\nimport { Slot } from \"radix-ui\"\n\nimport { cn } from \"../../lib/utils\"\n\nconst buttonVariants = cva(\n \"inline-flex items-center justify-center gap-2 whitespace-nowrap rounded-md text-sm font-medium transition-all disabled:pointer-events-none disabled:opacity-50 [&_svg]:pointer-events-none [&_svg:not([class*='size-'])]:size-4 shrink-0 [&_svg]:shrink-0 outline-none focus-visible:border-ring focus-visible:ring-ring/50 focus-visible:ring-[3px] aria-invalid:ring-destructive/20 aria-invalid:border-destructive\",\n {\n variants: {\n variant: {\n default: \"bg-primary text-primary-foreground hover:bg-primary/90\",\n destructive:\n \"bg-destructive text-white hover:bg-destructive/90 focus-visible:ring-destructive/20\",\n outline:\n \"border border-input bg-transparent shadow-xs hover:bg-accent hover:text-accent-foreground\",\n secondary:\n \"bg-secondary text-secondary-foreground hover:bg-secondary/80\",\n ghost:\n \"hover:bg-accent hover:text-accent-foreground\",\n link: \"text-primary underline-offset-4 hover:underline\",\n },\n size: {\n default: \"h-9 px-4 py-2 has-[>svg]:px-3\",\n xs: \"h-6 gap-1 rounded-md px-2 text-xs has-[>svg]:px-1.5 [&_svg:not([class*='size-'])]:size-3\",\n sm: \"h-8 rounded-md gap-1.5 px-3 has-[>svg]:px-2.5\",\n lg: \"h-10 rounded-md px-6 has-[>svg]:px-4\",\n icon: \"size-9\",\n \"icon-xs\": \"size-6 rounded-md [&_svg:not([class*='size-'])]:size-3\",\n \"icon-sm\": \"size-8\",\n \"icon-lg\": \"size-10\",\n },\n },\n defaultVariants: {\n variant: \"default\",\n size: \"default\",\n },\n }\n)\n\nfunction Button({\n className,\n variant = \"default\",\n size = \"default\",\n asChild = false,\n ...props\n}: React.ComponentProps<\"button\"> &\n VariantProps<typeof buttonVariants> & {\n asChild?: boolean\n }) {\n const Comp = asChild ? Slot.Root : \"button\"\n\n return (\n <Comp\n data-slot=\"button\"\n data-variant={variant}\n data-size={size}\n className={cn(buttonVariants({ variant, size, className }))}\n {...props}\n />\n )\n}\n\nexport { Button, buttonVariants }\n","\"use client\";\n\nimport * as React from \"react\";\nimport type { ReactNode } from \"react\";\nimport { FieldDescription, FieldLabel } from \"../components/ui/field\";\nimport { cn } from \"../lib/utils\";\n\nexport type LabelGroupProps = {\n variant?: \"legend\" | \"label\" | \"text\";\n secondary?: ReactNode;\n secondaryId?: string;\n primary?: ReactNode;\n labelHtmlFor?: string;\n required?: boolean;\n primaryClassName?: string;\n secondaryClassName?: string;\n};\n\nconst LabelGroup = ({\n labelHtmlFor,\n required = false,\n variant = \"label\",\n secondaryId,\n secondary,\n primary,\n primaryClassName,\n secondaryClassName,\n}: LabelGroupProps) => {\n const primaryClasses = cn(\n \"text-sm font-medium leading-snug\",\n variant === \"legend\" ? \"mb-1.5\" : \"mb-1 block\",\n primaryClassName,\n );\n\n const requiredIndicator =\n required && variant !== \"label\" ? (\n <span className=\"text-destructive pl-0.5\" aria-label=\"required\">\n *\n </span>\n ) : null;\n\n let primaryElement: ReactNode = null;\n if (primary) {\n if (variant === \"label\") {\n primaryElement = (\n <FieldLabel\n htmlFor={labelHtmlFor}\n required={required}\n className={primaryClasses}\n >\n {primary}\n </FieldLabel>\n );\n } else if (variant === \"legend\") {\n primaryElement = (\n <legend data-slot=\"field-legend\" className={primaryClasses}>\n {primary}\n {requiredIndicator}\n </legend>\n );\n } else {\n primaryElement = (\n <div data-slot=\"field-label\" className={primaryClasses}>\n {primary}\n {requiredIndicator}\n </div>\n );\n }\n }\n\n const secondaryElement = secondary ? (\n <FieldDescription\n id={secondaryId}\n className={cn(\"leading-normal font-normal\", secondaryClassName)}\n >\n {secondary}\n </FieldDescription>\n ) : null;\n\n if (!primaryElement && !secondaryElement) return null;\n\n // Legend should remain a direct child of fieldset for proper semantics.\n if (variant === \"legend\") {\n return (\n <>\n {primaryElement}\n {secondaryElement}\n </>\n );\n }\n\n return (\n <div className=\"flex flex-1 flex-col gap-0.5\">\n {primaryElement}\n {secondaryElement}\n </div>\n );\n};\n\nexport { LabelGroup };\n"]}