@algodomain/smart-forms 0.1.10 → 0.1.11
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/dist/{SmartFormProvider-BdyRQakk.d.cts → SmartFormProvider-DtK3qRsV.d.cts} +8 -2
- package/dist/{SmartFormProvider-BdyRQakk.d.ts → SmartFormProvider-DtK3qRsV.d.ts} +8 -2
- package/dist/{chunk-WUYS7DMR.cjs → chunk-AEN4A4ST.cjs} +49 -12
- package/dist/chunk-AEN4A4ST.cjs.map +1 -0
- package/dist/{chunk-6VKQ7EMR.js → chunk-DYTQTHGE.js} +50 -13
- package/dist/chunk-DYTQTHGE.js.map +1 -0
- package/dist/{chunk-4XK6HAJ2.js → chunk-N3SIQIJR.js} +48 -18
- package/dist/chunk-N3SIQIJR.js.map +1 -0
- package/dist/{chunk-DRMVY7TX.cjs → chunk-TX7JD2XS.cjs} +135 -105
- package/dist/chunk-TX7JD2XS.cjs.map +1 -0
- package/dist/fields.cjs +124 -114
- package/dist/fields.cjs.map +1 -1
- package/dist/fields.d.cts +1 -1
- package/dist/fields.d.ts +1 -1
- package/dist/fields.js +27 -17
- package/dist/fields.js.map +1 -1
- package/dist/index.cjs +76 -68
- package/dist/index.cjs.map +1 -1
- package/dist/index.d.cts +2 -2
- package/dist/index.d.ts +2 -2
- package/dist/index.js +28 -20
- package/dist/index.js.map +1 -1
- package/dist/opinionated.cjs +18 -18
- package/dist/opinionated.d.cts +1 -1
- package/dist/opinionated.d.ts +1 -1
- package/dist/opinionated.js +2 -2
- package/package.json +1 -1
- package/dist/chunk-4XK6HAJ2.js.map +0 -1
- package/dist/chunk-6VKQ7EMR.js.map +0 -1
- package/dist/chunk-DRMVY7TX.cjs.map +0 -1
- package/dist/chunk-WUYS7DMR.cjs.map +0 -1
|
@@ -72,8 +72,12 @@ interface FormContextType {
|
|
|
72
72
|
saveDraft: () => Promise<void>;
|
|
73
73
|
resetForm: () => void;
|
|
74
74
|
fieldRefs: React__default.MutableRefObject<any>;
|
|
75
|
-
registerValidation: (field: string, validation: any
|
|
75
|
+
registerValidation: (field: string, validation: any, options?: {
|
|
76
|
+
label?: string;
|
|
77
|
+
}) => void;
|
|
76
78
|
validationRegistry: any;
|
|
79
|
+
validationLabels: Record<string, string>;
|
|
80
|
+
getValidationErrorMessage: (field: string, zodMessage: string) => string;
|
|
77
81
|
config: FormConfig;
|
|
78
82
|
setErrors: (errors: any) => void;
|
|
79
83
|
registerSubmitHook: (key: string, hook: () => Promise<void>) => void;
|
|
@@ -91,7 +95,9 @@ declare const useFormField: (field: string) => {
|
|
|
91
95
|
error: any;
|
|
92
96
|
onChange: (value: any) => void;
|
|
93
97
|
fieldRef: (el: any) => any;
|
|
94
|
-
registerValidation: (field: string, validation: any
|
|
98
|
+
registerValidation: (field: string, validation: any, options?: {
|
|
99
|
+
label?: string;
|
|
100
|
+
}) => void;
|
|
95
101
|
};
|
|
96
102
|
|
|
97
103
|
export { type AuthenticationConfig as A, type FormContextType as F, SmartFormProvider as S, useFormField as a, type FormConfig as b, useSmartForm as u };
|
|
@@ -72,8 +72,12 @@ interface FormContextType {
|
|
|
72
72
|
saveDraft: () => Promise<void>;
|
|
73
73
|
resetForm: () => void;
|
|
74
74
|
fieldRefs: React__default.MutableRefObject<any>;
|
|
75
|
-
registerValidation: (field: string, validation: any
|
|
75
|
+
registerValidation: (field: string, validation: any, options?: {
|
|
76
|
+
label?: string;
|
|
77
|
+
}) => void;
|
|
76
78
|
validationRegistry: any;
|
|
79
|
+
validationLabels: Record<string, string>;
|
|
80
|
+
getValidationErrorMessage: (field: string, zodMessage: string) => string;
|
|
77
81
|
config: FormConfig;
|
|
78
82
|
setErrors: (errors: any) => void;
|
|
79
83
|
registerSubmitHook: (key: string, hook: () => Promise<void>) => void;
|
|
@@ -91,7 +95,9 @@ declare const useFormField: (field: string) => {
|
|
|
91
95
|
error: any;
|
|
92
96
|
onChange: (value: any) => void;
|
|
93
97
|
fieldRef: (el: any) => any;
|
|
94
|
-
registerValidation: (field: string, validation: any
|
|
98
|
+
registerValidation: (field: string, validation: any, options?: {
|
|
99
|
+
label?: string;
|
|
100
|
+
}) => void;
|
|
95
101
|
};
|
|
96
102
|
|
|
97
103
|
export { type AuthenticationConfig as A, type FormContextType as F, SmartFormProvider as S, useFormField as a, type FormConfig as b, useSmartForm as u };
|
|
@@ -31,6 +31,15 @@ var LabelPrimitive__namespace = /*#__PURE__*/_interopNamespace(LabelPrimitive);
|
|
|
31
31
|
var TooltipPrimitive__namespace = /*#__PURE__*/_interopNamespace(TooltipPrimitive);
|
|
32
32
|
|
|
33
33
|
var FormContext = react.createContext(null);
|
|
34
|
+
function isUglyValidationMessage(message) {
|
|
35
|
+
if (!message || typeof message !== "string") return false;
|
|
36
|
+
const m = message.trim().toLowerCase();
|
|
37
|
+
if (m === "required") return true;
|
|
38
|
+
if (/expected (string|number|boolean|array|object), received (undefined|null)/.test(m)) return true;
|
|
39
|
+
if (/invalid (type|input)[\s:]*.*expected \w+, received (undefined|null)/.test(m)) return true;
|
|
40
|
+
if (/invalid (type|input)/.test(m) && /expected \w+, received (undefined|null)/.test(m)) return true;
|
|
41
|
+
return false;
|
|
42
|
+
}
|
|
34
43
|
var SmartFormProvider = ({
|
|
35
44
|
children,
|
|
36
45
|
config,
|
|
@@ -98,6 +107,7 @@ var SmartFormProvider = ({
|
|
|
98
107
|
const [isLoading, setIsLoading] = react.useState(false);
|
|
99
108
|
const [isDraftSaving, setIsDraftSaving] = react.useState(false);
|
|
100
109
|
const [validationRegistry, setValidationRegistry] = react.useState({});
|
|
110
|
+
const [validationLabels, setValidationLabels] = react.useState({});
|
|
101
111
|
const fieldRefs = react.useRef({});
|
|
102
112
|
const submitHooksRef = react.useRef(/* @__PURE__ */ new Map());
|
|
103
113
|
const updateField = react.useCallback((field, value) => {
|
|
@@ -117,6 +127,14 @@ var SmartFormProvider = ({
|
|
|
117
127
|
});
|
|
118
128
|
}
|
|
119
129
|
}, [errors, config.enableLocalStorage, saveToStorage]);
|
|
130
|
+
const getValidationErrorMessage = react.useCallback((field, zodMessage) => {
|
|
131
|
+
const label = validationLabels[field];
|
|
132
|
+
const raw = zodMessage?.trim() || `Invalid ${field}`;
|
|
133
|
+
if (isUglyValidationMessage(raw) && label) {
|
|
134
|
+
return `${label} is required`;
|
|
135
|
+
}
|
|
136
|
+
return raw || `Invalid ${field}`;
|
|
137
|
+
}, [validationLabels]);
|
|
120
138
|
const validateField = react.useCallback((field, value) => {
|
|
121
139
|
const validation = validationRegistry[field];
|
|
122
140
|
if (!validation) return true;
|
|
@@ -125,20 +143,28 @@ var SmartFormProvider = ({
|
|
|
125
143
|
return true;
|
|
126
144
|
} catch (error) {
|
|
127
145
|
if (error instanceof zod.z.ZodError) {
|
|
146
|
+
const msg = error.issues[0]?.message;
|
|
128
147
|
setErrors((prev) => ({
|
|
129
148
|
...prev,
|
|
130
|
-
[field]:
|
|
149
|
+
[field]: getValidationErrorMessage(field, msg ?? "")
|
|
131
150
|
}));
|
|
132
151
|
return false;
|
|
133
152
|
}
|
|
134
153
|
return false;
|
|
135
154
|
}
|
|
136
|
-
}, [validationRegistry]);
|
|
137
|
-
const registerValidation = react.useCallback((field, validation) => {
|
|
155
|
+
}, [validationRegistry, getValidationErrorMessage]);
|
|
156
|
+
const registerValidation = react.useCallback((field, validation, options) => {
|
|
138
157
|
setValidationRegistry((prev) => ({
|
|
139
158
|
...prev,
|
|
140
159
|
[field]: validation
|
|
141
160
|
}));
|
|
161
|
+
if (options?.label != null) {
|
|
162
|
+
const label = options.label;
|
|
163
|
+
setValidationLabels((prev) => ({
|
|
164
|
+
...prev,
|
|
165
|
+
[field]: label
|
|
166
|
+
}));
|
|
167
|
+
}
|
|
142
168
|
}, []);
|
|
143
169
|
const validateAllFields = react.useCallback(() => {
|
|
144
170
|
const allErrors = {};
|
|
@@ -149,7 +175,8 @@ var SmartFormProvider = ({
|
|
|
149
175
|
validation.parse(formData[field]);
|
|
150
176
|
} catch (error) {
|
|
151
177
|
if (error instanceof zod.z.ZodError) {
|
|
152
|
-
|
|
178
|
+
const msg = error.issues[0]?.message;
|
|
179
|
+
allErrors[field] = getValidationErrorMessage(field, msg ?? "");
|
|
153
180
|
isValid = false;
|
|
154
181
|
}
|
|
155
182
|
}
|
|
@@ -157,7 +184,7 @@ var SmartFormProvider = ({
|
|
|
157
184
|
}
|
|
158
185
|
setErrors(allErrors);
|
|
159
186
|
return isValid;
|
|
160
|
-
}, [formData, validationRegistry]);
|
|
187
|
+
}, [formData, validationRegistry, getValidationErrorMessage]);
|
|
161
188
|
const validateFields = react.useCallback((fields) => {
|
|
162
189
|
const allErrors = {};
|
|
163
190
|
let isValid = true;
|
|
@@ -168,7 +195,8 @@ var SmartFormProvider = ({
|
|
|
168
195
|
validation.parse(formData[field]);
|
|
169
196
|
} catch (error) {
|
|
170
197
|
if (error instanceof zod.z.ZodError) {
|
|
171
|
-
|
|
198
|
+
const msg = error.issues[0]?.message;
|
|
199
|
+
allErrors[field] = getValidationErrorMessage(field, msg ?? "");
|
|
172
200
|
isValid = false;
|
|
173
201
|
}
|
|
174
202
|
}
|
|
@@ -179,7 +207,7 @@ var SmartFormProvider = ({
|
|
|
179
207
|
...allErrors
|
|
180
208
|
}));
|
|
181
209
|
return isValid;
|
|
182
|
-
}, [formData, validationRegistry]);
|
|
210
|
+
}, [formData, validationRegistry, getValidationErrorMessage]);
|
|
183
211
|
const submitForm = react.useCallback(async () => {
|
|
184
212
|
if (!validateAllFields()) {
|
|
185
213
|
return;
|
|
@@ -408,6 +436,8 @@ var SmartFormProvider = ({
|
|
|
408
436
|
fieldRefs,
|
|
409
437
|
registerValidation,
|
|
410
438
|
validationRegistry,
|
|
439
|
+
validationLabels,
|
|
440
|
+
getValidationErrorMessage,
|
|
411
441
|
config,
|
|
412
442
|
setErrors,
|
|
413
443
|
registerSubmitHook: (key, hook) => {
|
|
@@ -564,12 +594,19 @@ var SmartInput = ({
|
|
|
564
594
|
const togglePasswordVisibility = () => {
|
|
565
595
|
setShowPassword(!showPassword);
|
|
566
596
|
};
|
|
597
|
+
const displayName = label || field;
|
|
598
|
+
const builtinValidation = react.useMemo(() => {
|
|
599
|
+
if (validation || !required) return void 0;
|
|
600
|
+
const preprocess = (v) => v === void 0 || v === null ? "" : v;
|
|
601
|
+
return zod.z.preprocess(preprocess, zod.z.string().min(1, `${displayName} is required`));
|
|
602
|
+
}, [validation, required, displayName]);
|
|
603
|
+
const schemaToRegister = validation ?? builtinValidation;
|
|
567
604
|
react.useEffect(() => {
|
|
568
|
-
if (
|
|
605
|
+
if (schemaToRegister && !hasRegistered.current) {
|
|
569
606
|
hasRegistered.current = true;
|
|
570
|
-
registerValidation(field,
|
|
607
|
+
registerValidation(field, schemaToRegister, { label: displayName });
|
|
571
608
|
}
|
|
572
|
-
}, [
|
|
609
|
+
}, [schemaToRegister, field, registerValidation, displayName]);
|
|
573
610
|
react.useEffect(() => {
|
|
574
611
|
if (fieldDetection?.registerField) {
|
|
575
612
|
fieldDetection.registerField(field);
|
|
@@ -740,5 +777,5 @@ exports.cn = cn;
|
|
|
740
777
|
exports.useFieldDetection = useFieldDetection;
|
|
741
778
|
exports.useFormField = useFormField;
|
|
742
779
|
exports.useSmartForm = useSmartForm;
|
|
743
|
-
//# sourceMappingURL=chunk-
|
|
744
|
-
//# sourceMappingURL=chunk-
|
|
780
|
+
//# sourceMappingURL=chunk-AEN4A4ST.cjs.map
|
|
781
|
+
//# sourceMappingURL=chunk-AEN4A4ST.cjs.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../src/SmartFormProvider.tsx","../src/FieldDetectionHook.tsx","../src/lib/utils.ts","../src/components/ui/input.tsx","../src/components/ui/textarea.tsx","../src/components/ui/label.tsx","../src/components/ui/tooltip.tsx","../src/smart-fields/SmartInput.tsx"],"names":["createContext","useCallback","useState","useRef","z","useContext","twMerge","clsx","jsx","LabelPrimitive","TooltipPrimitive","jsxs","useMemo","useEffect","EyeOffIcon","EyeIcon","InfoIcon"],"mappings":"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","file":"chunk-AEN4A4ST.cjs","sourcesContent":["import React, { createContext, useContext, useState, useRef, useCallback, type ReactNode } from 'react';\nimport { z } from 'zod';\n\n// =============================================================================\n// FORM PROVIDER TYPES\n// =============================================================================\n\n/**\n * Authentication configuration for SmartForm\n * \n * @interface AuthenticationConfig\n * @property {boolean} enable - Enable authentication (sends Bearer token in Authorization header)\n * @property {string} [refreshTokenEndpoint] - Optional endpoint for token refresh when accessToken expires\n * @property {string} [accessTokenKey] - localStorage key for access token (default: 'accessToken')\n * @property {string} [refreshTokenKey] - localStorage key for refresh token (default: 'refreshToken')\n * \n * @example\n * ```typescript\n * // Basic authentication without token refresh\n * authentication={{ enable: true }}\n * \n * // Authentication with token refresh\n * authentication={{\n * enable: true,\n * refreshTokenEndpoint: \"/api/auth/refresh\"\n * }}\n * \n * // Custom token keys\n * authentication={{\n * enable: true,\n * refreshTokenEndpoint: \"/api/auth/refresh\",\n * accessTokenKey: \"myAccessToken\",\n * refreshTokenKey: \"myRefreshToken\"\n * }}\n * ```\n */\nexport interface AuthenticationConfig {\n enable: boolean;\n refreshTokenEndpoint?: string; // Optional endpoint for token refresh when accessToken expires\n accessTokenKey?: string; // localStorage key for access token (default: 'accessToken')\n refreshTokenKey?: string; // localStorage key for refresh token (default: 'refreshToken')\n}\n\nexport interface FormConfig {\n showProgressBar?: any;\n api?: string;\n method?: 'POST' | 'PUT' | 'PATCH'; // HTTP method for form submission\n submitButtonText?: string;\n submitButtonIcon?: ReactNode;\n allowSaveDraft?: boolean;\n saveDraftApi?: string;\n onSuccess?: (data: any) => void;\n onError?: (error: any) => void;\n transformData?: (data: any) => any; // Transform data before submission\n className?: string;\n title?: string;\n subTitle?: string; // Optional subtitle displayed under the title\n storageKey?: string; // Key for localStorage persistence\n enableLocalStorage?: boolean; // Enable/disable localStorage\n logFormData?: boolean; // Enable/disable console logging of form data\n showReset?: boolean; // Show reset button regardless of localStorage setting\n showTabNumbers?: boolean; // Show tab numbers in tab headers\n authentication?: AuthenticationConfig; // Authentication configuration\n includeQueryParams?: boolean; // Include URL query parameters in form submission (default: false)\n queryParamsToInclude?: string[]; // Optional array of specific query param names to include\n submitDisabled?: boolean | ((formData: any) => boolean); // Disable submit button conditionally\n}\n\nexport interface FormContextType {\n formData: any;\n errors: any;\n isLoading: boolean;\n isDraftSaving: boolean;\n updateField: (field: string, value: any) => void;\n validateField: (field: string, value: any) => boolean;\n validateFields: (fields: string[]) => boolean;\n submitForm: () => Promise<void>;\n validateFormAndGetResult: () => boolean;\n saveDraft: () => Promise<void>;\n resetForm: () => void;\n fieldRefs: React.MutableRefObject<any>;\n registerValidation: (field: string, validation: any, options?: { label?: string }) => void;\n validationRegistry: any;\n validationLabels: Record<string, string>;\n getValidationErrorMessage: (field: string, zodMessage: string) => string;\n config: FormConfig;\n setErrors: (errors: any) => void;\n registerSubmitHook: (key: string, hook: () => Promise<void>) => void;\n unregisterSubmitHook: (key: string) => void;\n}\n\n// =============================================================================\n// FORM CONTEXT\n// =============================================================================\n\nconst FormContext = createContext<FormContextType | null>(null);\n\n/** Matches generic Zod messages we replace with \"{label} is required\". */\nfunction isUglyValidationMessage(message: string): boolean {\n if (!message || typeof message !== 'string') return false;\n const m = message.trim().toLowerCase();\n if (m === 'required') return true;\n if (/expected (string|number|boolean|array|object), received (undefined|null)/.test(m)) return true;\n if (/invalid (type|input)[\\s:]*.*expected \\w+, received (undefined|null)/.test(m)) return true;\n if (/invalid (type|input)/.test(m) && /expected \\w+, received (undefined|null)/.test(m)) return true;\n return false;\n}\n\n// =============================================================================\n// FORM PROVIDER COMPONENT\n// =============================================================================\n\nexport interface SmartFormProviderProps {\n children: ReactNode;\n config: FormConfig;\n initialData?: any;\n}\n\nexport const SmartFormProvider: React.FC<SmartFormProviderProps> = ({\n children,\n config,\n initialData = {}\n}) => {\n // Helper function to extract query parameters from URL\n const getQueryParams = useCallback((allowedParams?: string[]): Record<string, any> => {\n const params = new URLSearchParams(window.location.search);\n const result: Record<string, any> = {};\n \n if (allowedParams && allowedParams.length > 0) {\n // Only include specified params\n allowedParams.forEach(key => {\n const value = params.get(key);\n if (value !== null) {\n result[key] = value;\n }\n });\n } else {\n // Include all params\n params.forEach((value, key) => {\n result[key] = value;\n });\n }\n \n return result;\n }, []);\n\n // Helper functions for localStorage\n const getStorageKey = () => {\n return config.storageKey || 'smart-form-data';\n };\n\n const loadFromStorage = () => {\n if (!config.enableLocalStorage) return initialData;\n\n try {\n const stored = localStorage.getItem(getStorageKey());\n if (stored) {\n const parsedStored = JSON.parse(stored);\n // Merge stored data with initialData\n // For each field: use stored value if it's not empty/null, otherwise use initialData value\n const merged = { ...initialData };\n Object.keys(parsedStored).forEach((key) => {\n const storedValue = parsedStored[key];\n // Only use stored value if it's not null, undefined, or empty string\n if (storedValue !== null && storedValue !== undefined && storedValue !== '') {\n merged[key] = storedValue;\n }\n });\n return merged;\n }\n return initialData;\n } catch (error) {\n console.warn('Failed to load form data from localStorage:', error);\n return initialData;\n }\n };\n\n const saveToStorage = (data: any) => {\n if (!config.enableLocalStorage) return;\n\n try {\n localStorage.setItem(getStorageKey(), JSON.stringify(data));\n } catch (error) {\n console.warn('Failed to save form data to localStorage:', error);\n }\n };\n\n const clearStorage = () => {\n if (!config.enableLocalStorage) return;\n\n try {\n localStorage.removeItem(getStorageKey());\n } catch (error) {\n console.warn('Failed to clear form data from localStorage:', error);\n }\n };\n\n const [formData, setFormData] = useState(loadFromStorage);\n const [errors, setErrors] = useState<any>({});\n const [isLoading, setIsLoading] = useState(false);\n const [isDraftSaving, setIsDraftSaving] = useState(false);\n const [validationRegistry, setValidationRegistry] = useState<any>({});\n const [validationLabels, setValidationLabels] = useState<Record<string, string>>({});\n const fieldRefs = useRef<any>({});\n const submitHooksRef = useRef<Map<string, () => Promise<void>>>(new Map());\n\n // Update form field\n const updateField = useCallback((field: string, value: any) => {\n setFormData((prev: any) => {\n const newData = {\n ...prev,\n [field]: value\n };\n // Save to localStorage after state update\n setTimeout(() => saveToStorage(newData), 0);\n return newData;\n });\n\n // Clear error when user starts typing\n if (errors[field]) {\n setErrors((prev: any) => {\n const newErrors = { ...prev };\n delete newErrors[field];\n return newErrors;\n });\n }\n }, [errors, config.enableLocalStorage, saveToStorage]);\n\n const getValidationErrorMessage = useCallback((field: string, zodMessage: string) => {\n const label = validationLabels[field];\n const raw = zodMessage?.trim() || `Invalid ${field}`;\n if (isUglyValidationMessage(raw) && label) {\n return `${label} is required`;\n }\n return raw || `Invalid ${field}`;\n }, [validationLabels]);\n\n // Validate single field\n const validateField = useCallback((field: string, value: any) => {\n const validation = validationRegistry[field];\n if (!validation) return true;\n\n try {\n validation.parse(value);\n return true;\n } catch (error) {\n if (error instanceof z.ZodError) {\n const msg = error.issues[0]?.message;\n setErrors((prev: any) => ({\n ...prev,\n [field]: getValidationErrorMessage(field, msg ?? '')\n }));\n return false;\n }\n return false;\n }\n }, [validationRegistry, getValidationErrorMessage]);\n\n // Register validation for a field\n const registerValidation = useCallback((field: string, validation: any, options?: { label?: string }) => {\n setValidationRegistry((prev: any) => ({\n ...prev,\n [field]: validation\n }));\n if (options?.label != null) {\n const label = options.label;\n setValidationLabels((prev) => ({\n ...prev,\n [field]: label\n }));\n }\n }, []);\n\n // Validate all fields\n const validateAllFields = useCallback(() => {\n const allErrors: any = {};\n let isValid = true;\n\n for (const [field, validation] of Object.entries(validationRegistry)) {\n if (validation && typeof (validation as any).parse === 'function') {\n try {\n (validation as any).parse(formData[field]);\n } catch (error) {\n if (error instanceof z.ZodError) {\n const msg = error.issues[0]?.message;\n allErrors[field] = getValidationErrorMessage(field, msg ?? '');\n isValid = false;\n }\n }\n }\n }\n\n setErrors(allErrors);\n return isValid;\n }, [formData, validationRegistry, getValidationErrorMessage]);\n\n // Validate specific fields (for tab validation)\n const validateFields = useCallback((fields: string[]) => {\n const allErrors: any = {};\n let isValid = true;\n\n for (const field of fields) {\n const validation = validationRegistry[field];\n if (validation) {\n try {\n validation.parse(formData[field]);\n } catch (error) {\n if (error instanceof z.ZodError) {\n const msg = error.issues[0]?.message;\n allErrors[field] = getValidationErrorMessage(field, msg ?? '');\n isValid = false;\n }\n }\n }\n }\n\n // Only update errors for the fields being validated\n setErrors((prev: any) => ({\n ...prev,\n ...allErrors\n }));\n\n return isValid;\n }, [formData, validationRegistry, getValidationErrorMessage]);\n\n // Submit form\n const submitForm = useCallback(async () => {\n if (!validateAllFields()) {\n return;\n }\n\n if (!config.api) {\n console.warn('No API endpoint provided');\n return;\n }\n\n // Merge query params with form data if enabled\n let dataToSubmit = formData;\n if (config.includeQueryParams) {\n const queryParams = getQueryParams(config.queryParamsToInclude);\n // Form data overrides query params in case of conflicts\n dataToSubmit = { ...queryParams, ...formData };\n }\n\n // Log form data if enabled\n if (config.logFormData) {\n console.log('📤 Submitting form data:', dataToSubmit);\n console.log('🎯 API endpoint:', config.api);\n }\n\n // Build a JSON-safe payload: replace File(s) with filename(s), Dates with ISO\n const toJsonSafe = (val: any): any => {\n if (val instanceof File) return val.name;\n if (val instanceof Date) return val.toISOString();\n if (Array.isArray(val)) {\n return val.map((item) => (item instanceof File ? item.name : toJsonSafe(item)));\n }\n if (val && typeof val === 'object') {\n // shallow map object values\n const out: any = {};\n for (const [k, v] of Object.entries(val)) out[k] = toJsonSafe(v);\n return out;\n }\n return val;\n };\n\n let jsonPayload = toJsonSafe(dataToSubmit);\n \n // Apply transformation if provided\n if (config.transformData) {\n jsonPayload = config.transformData(jsonPayload);\n }\n\n setIsLoading(true);\n try {\n // Prepare headers\n const headers: Record<string, string> = {\n 'Content-Type': 'application/json',\n };\n\n // Add Authorization header if authentication is enabled and token exists\n if (config.authentication?.enable) {\n const { accessTokenKey = 'accessToken' } = config.authentication;\n const token = localStorage.getItem(accessTokenKey);\n if (token) {\n headers.Authorization = `Bearer ${token}`;\n }\n }\n\n // Prepare fetch options\n const fetchOptions: RequestInit = {\n method: config.method || 'POST',\n headers,\n body: JSON.stringify(jsonPayload),\n };\n\n // Add credentials if authentication is enabled\n if (config.authentication?.enable) {\n fetchOptions.credentials = 'include';\n }\n\n let response = await fetch(config.api, fetchOptions);\n\n // Handle token expiration with automatic refresh\n if (response.status === 403 && config.authentication?.enable && config.authentication.refreshTokenEndpoint) {\n const newAccessToken = await refreshAccessToken(config.authentication);\n \n if (newAccessToken) {\n // Retry the request with the new token\n headers.Authorization = `Bearer ${newAccessToken}`;\n response = await fetch(config.api, {\n ...fetchOptions,\n headers,\n });\n }\n }\n\n // Parse response body (JSON or text)\n let responseBody;\n try {\n responseBody = await response.json();\n } catch {\n responseBody = await response.text();\n }\n\n const responseData = {\n status: response.status,\n statusText: response.statusText,\n body: responseBody\n };\n\n // Check if response is successful (2xx status codes)\n if (response.status >= 200 && response.status < 300) {\n if (config.onSuccess) {\n config.onSuccess(responseData);\n }\n\n // After successful submit, run any registered submit hooks (e.g., file uploads)\n for (const hook of submitHooksRef.current.values()) {\n try {\n // eslint-disable-next-line no-await-in-loop\n await hook();\n } catch (hookError) {\n console.error('Submit hook error:', hookError);\n }\n }\n } else {\n // Handle error response\n if (config.onError) {\n config.onError(responseData);\n }\n }\n } catch (error: any) {\n // Handle network errors or other fetch failures\n if (config.onError) {\n config.onError({\n status: 0,\n statusText: 'Network Error',\n body: error.message || 'An error occurred'\n });\n }\n console.error('Form submission error:', error);\n } finally {\n setIsLoading(false);\n }\n }, [formData, config, validateAllFields, getQueryParams]);\n\n // Validate form and return validation result\n const validateFormAndGetResult = useCallback(() => {\n return validateAllFields();\n }, [validateAllFields]);\n\n /**\n * Helper function to refresh access token\n * This function handles token refresh internally and is reusable across projects\n * \n * @param authConfig - The authentication configuration object\n * @returns Promise<string | null> - New access token or null if refresh failed\n */\n const refreshAccessToken = useCallback(async (authConfig: AuthenticationConfig): Promise<string | null> => {\n const { refreshTokenEndpoint, accessTokenKey = 'accessToken', refreshTokenKey = 'refreshToken' } = authConfig;\n \n if (!refreshTokenEndpoint) {\n throw new Error('Refresh token endpoint not provided');\n }\n \n try {\n const refreshToken = localStorage.getItem(refreshTokenKey);\n if (!refreshToken) {\n throw new Error('No refresh token available');\n }\n\n const response = await fetch(refreshTokenEndpoint, {\n method: 'POST',\n headers: {\n 'Content-Type': 'application/json',\n },\n body: JSON.stringify({ refreshToken }),\n });\n\n if (!response.ok) {\n throw new Error(`Token refresh failed: ${response.status}`);\n }\n\n const result = await response.json();\n // Support different response formats: { data: { accessToken } } or { accessToken }\n const newAccessToken = result.data?.accessToken || result.accessToken;\n \n if (newAccessToken) {\n localStorage.setItem(accessTokenKey, newAccessToken);\n return newAccessToken;\n }\n \n throw new Error('No access token in refresh response');\n } catch (error) {\n console.error('Token refresh failed:', error);\n // Clear tokens on refresh failure to prevent infinite retry loops\n localStorage.removeItem(accessTokenKey);\n localStorage.removeItem(refreshTokenKey);\n return null;\n }\n }, []);\n\n // Save draft\n const saveDraft = useCallback(async () => {\n if (!config.saveDraftApi) {\n console.warn('No save draft API endpoint provided');\n return;\n }\n\n // Merge query params with form data if enabled\n let dataToSave = formData;\n if (config.includeQueryParams) {\n const queryParams = getQueryParams(config.queryParamsToInclude);\n // Form data overrides query params in case of conflicts\n dataToSave = { ...queryParams, ...formData };\n }\n\n // Log form data if enabled\n if (config.logFormData) {\n console.log('💾 Saving draft data:', dataToSave);\n console.log('🎯 Draft API endpoint:', config.saveDraftApi);\n }\n\n setIsDraftSaving(true);\n try {\n // Prepare headers\n const headers: Record<string, string> = {\n 'Content-Type': 'application/json',\n };\n\n // Add Authorization header if authentication is enabled and token exists\n if (config.authentication?.enable) {\n const { accessTokenKey = 'accessToken' } = config.authentication;\n const token = localStorage.getItem(accessTokenKey);\n if (token) {\n headers.Authorization = `Bearer ${token}`;\n }\n }\n\n // Prepare fetch options\n const fetchOptions: RequestInit = {\n method: 'POST',\n headers,\n body: JSON.stringify(dataToSave),\n };\n\n // Add credentials if authentication is enabled\n if (config.authentication?.enable) {\n fetchOptions.credentials = 'include';\n }\n\n let response = await fetch(config.saveDraftApi, fetchOptions);\n\n // Handle token expiration with automatic refresh\n if (response.status === 403 && config.authentication?.enable && config.authentication.refreshTokenEndpoint) {\n const newAccessToken = await refreshAccessToken(config.authentication);\n \n if (newAccessToken) {\n // Retry the request with the new token\n headers.Authorization = `Bearer ${newAccessToken}`;\n response = await fetch(config.saveDraftApi, {\n ...fetchOptions,\n headers,\n });\n }\n }\n\n // Parse response body (JSON or text)\n let responseBody;\n try {\n responseBody = await response.json();\n } catch {\n responseBody = await response.text();\n }\n\n // Check if response is successful (2xx status codes)\n if (response.status >= 200 && response.status < 300) {\n console.log('Draft saved successfully:', responseBody);\n } else {\n console.error('Draft save failed:', {\n status: response.status,\n statusText: response.statusText,\n body: responseBody\n });\n }\n } catch (error) {\n console.error('Draft save error:', error);\n } finally {\n setIsDraftSaving(false);\n }\n }, [formData, config, getQueryParams]);\n\n // Reset form and clear localStorage\n const resetForm = useCallback(() => {\n setFormData(initialData);\n setErrors({});\n clearStorage();\n }, [initialData, config.enableLocalStorage, clearStorage]);\n\n const contextValue: FormContextType = {\n formData,\n errors,\n isLoading,\n isDraftSaving,\n updateField,\n validateField,\n validateFields,\n submitForm,\n validateFormAndGetResult,\n saveDraft,\n resetForm,\n fieldRefs,\n registerValidation,\n validationRegistry,\n validationLabels,\n getValidationErrorMessage,\n config,\n setErrors,\n registerSubmitHook: (key: string, hook: () => Promise<void>) => {\n submitHooksRef.current.set(key, hook);\n },\n unregisterSubmitHook: (key: string) => {\n submitHooksRef.current.delete(key);\n }\n };\n\n return (\n <FormContext.Provider value={contextValue}>\n {children}\n </FormContext.Provider>\n );\n};\n\n// =============================================================================\n// FORM HOOKS\n// =============================================================================\n\nexport const useSmartForm = () => {\n const context = useContext(FormContext);\n if (!context) {\n throw new Error('useSmartForm must be used within a SmartFormProvider');\n }\n return context;\n};\n\nexport const useFormField = (field: string) => {\n const { formData, errors, updateField, fieldRefs, registerValidation } = useSmartForm();\n\n return {\n value: formData[field],\n error: errors[field],\n onChange: (value: any) => updateField(field, value),\n fieldRef: (el: any) => fieldRefs.current[field] = el,\n registerValidation\n };\n};\n","import { createContext, useContext } from 'react';\n\nexport interface FieldDetectionContextType {\n registerField: (fieldName: string) => void;\n}\n\nexport const FieldDetectionContext = createContext<FieldDetectionContextType | null>(null);\n\nexport const useFieldDetection = () => {\n const context = useContext(FieldDetectionContext);\n return context; // Return null if not in context (for backward compatibility)\n};\n","import { clsx, type ClassValue } from \"clsx\"\nimport { twMerge } from \"tailwind-merge\"\n\nexport function cn(...inputs: ClassValue[]) {\n return twMerge(clsx(inputs))\n}\n","import * as React from \"react\"\n\nimport { cn } from \"../../lib/utils\"\n\nfunction Input({ className, type, ...props }: React.ComponentProps<\"input\">) {\n return (\n <input\n type={type}\n data-slot=\"input\"\n className={cn(\n \"file:text-foreground placeholder:text-muted-foreground selection:bg-primary selection:text-primary-foreground dark:bg-input/30 border-input h-9 w-full min-w-0 rounded-md border bg-transparent px-3 py-1 text-base shadow-xs transition-[color,box-shadow] outline-none file:inline-flex file:h-7 file:border-0 file:bg-transparent file:text-sm file:font-medium disabled:pointer-events-none disabled:cursor-not-allowed disabled:opacity-50 md:text-sm\",\n \"focus-visible:border-ring focus-visible:ring-ring/50 focus-visible:ring-[3px]\",\n \"aria-invalid:ring-destructive/20 dark:aria-invalid:ring-destructive/40 aria-invalid:border-destructive\",\n className\n )}\n {...props}\n />\n )\n}\n\nexport { Input }\n","import * as React from \"react\"\n\nimport { cn } from \"../../lib/utils\"\n\nfunction Textarea({ className, ...props }: React.ComponentProps<\"textarea\">) {\n return (\n <textarea\n data-slot=\"textarea\"\n className={cn(\n \"border-input placeholder:text-muted-foreground focus-visible:border-ring focus-visible:ring-ring/50 aria-invalid:ring-destructive/20 dark:aria-invalid:ring-destructive/40 aria-invalid:border-destructive dark:bg-input/30 flex field-sizing-content min-h-16 w-full rounded-md border bg-transparent px-3 py-2 text-base shadow-xs transition-[color,box-shadow] outline-none focus-visible:ring-[3px] disabled:cursor-not-allowed disabled:opacity-50 md:text-sm\",\n className\n )}\n {...props}\n />\n )\n}\n\nexport { Textarea }\n","import * as React from \"react\"\nimport * as LabelPrimitive from \"@radix-ui/react-label\"\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 * as TooltipPrimitive from \"@radix-ui/react-tooltip\"\n\nimport { cn } from \"../../lib/utils\"\n\nfunction TooltipProvider({\n delayDuration = 0,\n ...props\n}: React.ComponentProps<typeof TooltipPrimitive.Provider>) {\n return (\n <TooltipPrimitive.Provider\n data-slot=\"tooltip-provider\"\n delayDuration={delayDuration}\n {...props}\n />\n )\n}\n\nfunction Tooltip({\n ...props\n}: React.ComponentProps<typeof TooltipPrimitive.Root>) {\n return (\n <TooltipProvider>\n <TooltipPrimitive.Root data-slot=\"tooltip\" {...props} />\n </TooltipProvider>\n )\n}\n\nfunction TooltipTrigger({\n ...props\n}: React.ComponentProps<typeof TooltipPrimitive.Trigger>) {\n return <TooltipPrimitive.Trigger data-slot=\"tooltip-trigger\" {...props} />\n}\n\nfunction TooltipContent({\n className,\n sideOffset = 0,\n children,\n ...props\n}: React.ComponentProps<typeof TooltipPrimitive.Content>) {\n return (\n <TooltipPrimitive.Portal>\n <TooltipPrimitive.Content\n data-slot=\"tooltip-content\"\n sideOffset={sideOffset}\n className={cn(\n \"bg-foreground text-background animate-in fade-in-0 zoom-in-95 data-[state=closed]:animate-out data-[state=closed]:fade-out-0 data-[state=closed]:zoom-out-95 data-[side=bottom]:slide-in-from-top-2 data-[side=left]:slide-in-from-right-2 data-[side=right]:slide-in-from-left-2 data-[side=top]:slide-in-from-bottom-2 z-50 w-fit origin-(--radix-tooltip-content-transform-origin) rounded-md px-3 py-1.5 text-xs text-balance\",\n className\n )}\n {...props}\n >\n {children}\n <TooltipPrimitive.Arrow className=\"bg-foreground fill-foreground z-50 size-2.5 translate-y-[calc(-50%_-_2px)] rotate-45 rounded-[2px]\" />\n </TooltipPrimitive.Content>\n </TooltipPrimitive.Portal>\n )\n}\n\nexport { Tooltip, TooltipTrigger, TooltipContent, TooltipProvider }\n","import React, { useEffect, useMemo, useRef, useState } from 'react';\nimport { z } from 'zod';\nimport { useFormField, useSmartForm } from '../SmartFormProvider';\nimport { useFieldDetection } from '../FieldDetectionHook';\nimport { Input } from '../components/ui/input';\nimport { Textarea } from '../components/ui/textarea';\nimport { Label } from '../components/ui/label';\nimport { Tooltip, TooltipContent, TooltipProvider, TooltipTrigger } from '../components/ui/tooltip';\nimport { EyeIcon, EyeOffIcon, InfoIcon } from 'lucide-react';\n\n// =============================================================================\n// SMART FORM INPUT TYPES\n// =============================================================================\n\nexport interface SmartInputProps {\n field: string;\n label?: string;\n type?: 'text' | 'email' | 'password' | 'tel' | 'number' | 'textarea' | 'date';\n placeholder?: string;\n validation?: any; // Zod schema\n className?: string;\n required?: boolean;\n defaultValue?: any; // Default value for the field (only applied once on mount)\n syncValue?: any; // Value that syncs to form whenever it changes (for external control)\n info?: string; // Info text to show on hover of info icon\n subLabel?: string; // Additional label text shown below the main label\n disabled?: boolean | ((formData: any) => boolean);\n hidden?: boolean | ((formData: any) => boolean); // Hide field conditionally\n icon?: React.ReactNode; // Icon to display inside the input\n iconPosition?: 'left' | 'right'; // Position of the icon (default: 'right')\n}\n\n// =============================================================================\n// SMART FORM INPUT COMPONENT\n// =============================================================================\n\nexport const SmartInput: React.FC<SmartInputProps> = ({\n field,\n label,\n type = 'text',\n placeholder,\n validation,\n className = '',\n required = false,\n defaultValue,\n syncValue,\n info,\n subLabel,\n disabled,\n hidden,\n icon,\n iconPosition = 'right'\n}) => {\n const { formData } = useSmartForm();\n const { value, error, onChange, fieldRef, registerValidation } = useFormField(field);\n const fieldDetection = useFieldDetection();\n const hasRegistered = useRef(false);\n const hasSetDefault = useRef(false);\n const [showPassword, setShowPassword] = useState(false);\n\n // Evaluate disabled state\n const isDisabled = typeof disabled === 'function' ? disabled(formData) : disabled || false;\n \n // Evaluate hidden state\n const isHidden = typeof hidden === 'function' ? hidden(formData) : hidden || false;\n \n // Return null if field is hidden\n if (isHidden) return null;\n\n const togglePasswordVisibility = () => {\n setShowPassword(!showPassword);\n };\n\n const displayName = label || field;\n\n const builtinValidation = useMemo(() => {\n if (validation || !required) return undefined;\n const preprocess = (v: unknown) => (v === undefined || v === null ? '' : v);\n return z.preprocess(preprocess, z.string().min(1, `${displayName} is required`));\n }, [validation, required, displayName]);\n\n const schemaToRegister = validation ?? builtinValidation;\n\n // Register validation for this field (pass label for friendly default error messages)\n useEffect(() => {\n if (schemaToRegister && !hasRegistered.current) {\n hasRegistered.current = true;\n registerValidation(field, schemaToRegister, { label: displayName });\n }\n }, [schemaToRegister, field, registerValidation, displayName]);\n\n // Auto-register field if in field detection context\n useEffect(() => {\n if (fieldDetection?.registerField) {\n fieldDetection.registerField(field);\n }\n }, [field, fieldDetection]);\n\n // Set default value if provided and field is empty\n useEffect(() => {\n if (defaultValue !== undefined && !hasSetDefault.current && (value === undefined || value === null || value === '')) {\n onChange(defaultValue);\n hasSetDefault.current = true;\n }\n }, [defaultValue, value, onChange]);\n\n // Sync value from external source - updates form whenever syncValue changes\n const prevSyncValue = useRef(syncValue);\n useEffect(() => {\n if (syncValue !== undefined && syncValue !== prevSyncValue.current) {\n onChange(syncValue);\n prevSyncValue.current = syncValue;\n }\n }, [syncValue, onChange]);\n\n // Generate placeholder text\n const getPlaceholder = () => {\n if (placeholder) return placeholder;\n if (!label) return `Enter ${field}`;\n if (type === 'email') return `Enter ${label.toLowerCase()}`;\n if (type === 'tel') return `Enter ${label.toLowerCase()}`;\n return `Enter ${label.toLowerCase()}`;\n };\n\n const renderInput = () => {\n // Icon positioning classes\n const iconPaddingClass = icon ? (iconPosition === 'left' ? 'pl-10' : 'pr-10') : '';\n const iconPositionClass = iconPosition === 'left' ? 'left-3' : 'right-3';\n\n switch (type) {\n case 'textarea':\n return (\n <Textarea\n ref={fieldRef}\n value={value || ''}\n onChange={(e) => onChange(e.target.value)}\n className={`w-full ${error ? 'border-destructive' : ''} ${className}`}\n placeholder={getPlaceholder()}\n rows={4}\n data-field={field}\n disabled={isDisabled}\n />\n );\n\n case 'password':\n return (\n <div className=\"relative\">\n <Input\n ref={fieldRef}\n type={showPassword ? \"text\" : \"password\"}\n value={value || ''}\n onChange={(e) => onChange(e.target.value)}\n className={`w-full pr-10 ${icon && iconPosition === 'left' ? 'pl-10' : ''} ${error ? 'border-destructive' : ''} ${className}`}\n placeholder={getPlaceholder()}\n data-field={field}\n disabled={isDisabled}\n />\n {icon && iconPosition === 'left' && (\n <span className=\"absolute left-3 top-1/2 -translate-y-1/2 text-muted-foreground pointer-events-none\">\n {icon}\n </span>\n )}\n <button \n type=\"button\"\n onClick={togglePasswordVisibility}\n className=\"absolute right-3 top-1/2 -translate-y-1/2 p-1 hover:bg-gray-100 rounded transition-colors\"\n >\n {showPassword ? (\n <EyeOffIcon className=\"h-4 w-4 text-muted-foreground\" />\n ) : (\n <EyeIcon className=\"h-4 w-4 text-muted-foreground\" />\n )}\n </button>\n </div>\n );\n\n case 'date':\n // Display formatted date string, but form stores the actual Date object\n const displayDate = value instanceof Date \n ? value.toLocaleDateString('en-US', { year: 'numeric', month: 'long', day: 'numeric' })\n : '';\n if (icon) {\n return (\n <div className=\"relative\">\n <Input\n ref={fieldRef}\n type=\"text\"\n value={displayDate}\n readOnly\n className={`w-full ${iconPaddingClass} ${error ? 'border-destructive' : ''} ${className}`}\n placeholder={getPlaceholder()}\n data-field={field}\n disabled={isDisabled}\n />\n <span className={`absolute ${iconPositionClass} top-1/2 -translate-y-1/2 text-muted-foreground pointer-events-none`}>\n {icon}\n </span>\n </div>\n );\n }\n return (\n <Input\n ref={fieldRef}\n type=\"text\"\n value={displayDate}\n readOnly\n className={`w-full ${error ? 'border-destructive' : ''} ${className}`}\n placeholder={getPlaceholder()}\n data-field={field}\n disabled={isDisabled}\n />\n );\n\n default:\n if (icon) {\n return (\n <div className=\"relative\">\n <Input\n ref={fieldRef}\n type={type}\n value={value || ''}\n onChange={(e) => onChange(e.target.value)}\n className={`w-full ${iconPaddingClass} ${error ? 'border-destructive' : ''} ${className}`}\n placeholder={getPlaceholder()}\n data-field={field}\n disabled={isDisabled}\n />\n <span className={`absolute ${iconPositionClass} top-1/2 -translate-y-1/2 text-muted-foreground pointer-events-none`}>\n {icon}\n </span>\n </div>\n );\n }\n return (\n <Input\n ref={fieldRef}\n type={type}\n value={value || ''}\n onChange={(e) => onChange(e.target.value)}\n className={`w-full ${error ? 'border-destructive' : ''} ${className}`}\n placeholder={getPlaceholder()}\n data-field={field}\n disabled={isDisabled}\n />\n );\n }\n };\n\n return (\n <div className={`flex-1 min-w-0 ${className}`}>\n {label && (\n <div className=\"mb-1\">\n <div className=\"flex items-center justify-between gap-2\">\n <Label className=\"text-sm font-medium text-foreground\">\n {label} {required && <span className=\"text-destructive\">*</span>}\n </Label>\n {info && (\n <TooltipProvider>\n <Tooltip>\n <TooltipTrigger asChild>\n <InfoIcon className=\"h-4 w-4 text-muted-foreground cursor-pointer mr-2\" />\n </TooltipTrigger>\n <TooltipContent>\n <p className=\"max-w-xs\">{info}</p>\n </TooltipContent>\n </Tooltip>\n </TooltipProvider>\n )}\n </div>\n {subLabel && (\n <p className=\"text-xs text-muted-foreground mt-1\">{subLabel}</p>\n )}\n </div>\n )}\n {renderInput()}\n {error && (\n <p className=\"text-destructive text-sm mt-1\">{error}</p>\n )}\n </div>\n );\n};"]}
|
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
import { createContext, useCallback, useState, useRef, useContext, useEffect } from 'react';
|
|
1
|
+
import { createContext, useCallback, useState, useRef, useContext, useMemo, useEffect } from 'react';
|
|
2
2
|
import { z } from 'zod';
|
|
3
3
|
import { jsx, jsxs } from 'react/jsx-runtime';
|
|
4
4
|
import { clsx } from 'clsx';
|
|
@@ -8,6 +8,15 @@ import * as TooltipPrimitive from '@radix-ui/react-tooltip';
|
|
|
8
8
|
import { InfoIcon, EyeOffIcon, EyeIcon } from 'lucide-react';
|
|
9
9
|
|
|
10
10
|
var FormContext = createContext(null);
|
|
11
|
+
function isUglyValidationMessage(message) {
|
|
12
|
+
if (!message || typeof message !== "string") return false;
|
|
13
|
+
const m = message.trim().toLowerCase();
|
|
14
|
+
if (m === "required") return true;
|
|
15
|
+
if (/expected (string|number|boolean|array|object), received (undefined|null)/.test(m)) return true;
|
|
16
|
+
if (/invalid (type|input)[\s:]*.*expected \w+, received (undefined|null)/.test(m)) return true;
|
|
17
|
+
if (/invalid (type|input)/.test(m) && /expected \w+, received (undefined|null)/.test(m)) return true;
|
|
18
|
+
return false;
|
|
19
|
+
}
|
|
11
20
|
var SmartFormProvider = ({
|
|
12
21
|
children,
|
|
13
22
|
config,
|
|
@@ -75,6 +84,7 @@ var SmartFormProvider = ({
|
|
|
75
84
|
const [isLoading, setIsLoading] = useState(false);
|
|
76
85
|
const [isDraftSaving, setIsDraftSaving] = useState(false);
|
|
77
86
|
const [validationRegistry, setValidationRegistry] = useState({});
|
|
87
|
+
const [validationLabels, setValidationLabels] = useState({});
|
|
78
88
|
const fieldRefs = useRef({});
|
|
79
89
|
const submitHooksRef = useRef(/* @__PURE__ */ new Map());
|
|
80
90
|
const updateField = useCallback((field, value) => {
|
|
@@ -94,6 +104,14 @@ var SmartFormProvider = ({
|
|
|
94
104
|
});
|
|
95
105
|
}
|
|
96
106
|
}, [errors, config.enableLocalStorage, saveToStorage]);
|
|
107
|
+
const getValidationErrorMessage = useCallback((field, zodMessage) => {
|
|
108
|
+
const label = validationLabels[field];
|
|
109
|
+
const raw = zodMessage?.trim() || `Invalid ${field}`;
|
|
110
|
+
if (isUglyValidationMessage(raw) && label) {
|
|
111
|
+
return `${label} is required`;
|
|
112
|
+
}
|
|
113
|
+
return raw || `Invalid ${field}`;
|
|
114
|
+
}, [validationLabels]);
|
|
97
115
|
const validateField = useCallback((field, value) => {
|
|
98
116
|
const validation = validationRegistry[field];
|
|
99
117
|
if (!validation) return true;
|
|
@@ -102,20 +120,28 @@ var SmartFormProvider = ({
|
|
|
102
120
|
return true;
|
|
103
121
|
} catch (error) {
|
|
104
122
|
if (error instanceof z.ZodError) {
|
|
123
|
+
const msg = error.issues[0]?.message;
|
|
105
124
|
setErrors((prev) => ({
|
|
106
125
|
...prev,
|
|
107
|
-
[field]:
|
|
126
|
+
[field]: getValidationErrorMessage(field, msg ?? "")
|
|
108
127
|
}));
|
|
109
128
|
return false;
|
|
110
129
|
}
|
|
111
130
|
return false;
|
|
112
131
|
}
|
|
113
|
-
}, [validationRegistry]);
|
|
114
|
-
const registerValidation = useCallback((field, validation) => {
|
|
132
|
+
}, [validationRegistry, getValidationErrorMessage]);
|
|
133
|
+
const registerValidation = useCallback((field, validation, options) => {
|
|
115
134
|
setValidationRegistry((prev) => ({
|
|
116
135
|
...prev,
|
|
117
136
|
[field]: validation
|
|
118
137
|
}));
|
|
138
|
+
if (options?.label != null) {
|
|
139
|
+
const label = options.label;
|
|
140
|
+
setValidationLabels((prev) => ({
|
|
141
|
+
...prev,
|
|
142
|
+
[field]: label
|
|
143
|
+
}));
|
|
144
|
+
}
|
|
119
145
|
}, []);
|
|
120
146
|
const validateAllFields = useCallback(() => {
|
|
121
147
|
const allErrors = {};
|
|
@@ -126,7 +152,8 @@ var SmartFormProvider = ({
|
|
|
126
152
|
validation.parse(formData[field]);
|
|
127
153
|
} catch (error) {
|
|
128
154
|
if (error instanceof z.ZodError) {
|
|
129
|
-
|
|
155
|
+
const msg = error.issues[0]?.message;
|
|
156
|
+
allErrors[field] = getValidationErrorMessage(field, msg ?? "");
|
|
130
157
|
isValid = false;
|
|
131
158
|
}
|
|
132
159
|
}
|
|
@@ -134,7 +161,7 @@ var SmartFormProvider = ({
|
|
|
134
161
|
}
|
|
135
162
|
setErrors(allErrors);
|
|
136
163
|
return isValid;
|
|
137
|
-
}, [formData, validationRegistry]);
|
|
164
|
+
}, [formData, validationRegistry, getValidationErrorMessage]);
|
|
138
165
|
const validateFields = useCallback((fields) => {
|
|
139
166
|
const allErrors = {};
|
|
140
167
|
let isValid = true;
|
|
@@ -145,7 +172,8 @@ var SmartFormProvider = ({
|
|
|
145
172
|
validation.parse(formData[field]);
|
|
146
173
|
} catch (error) {
|
|
147
174
|
if (error instanceof z.ZodError) {
|
|
148
|
-
|
|
175
|
+
const msg = error.issues[0]?.message;
|
|
176
|
+
allErrors[field] = getValidationErrorMessage(field, msg ?? "");
|
|
149
177
|
isValid = false;
|
|
150
178
|
}
|
|
151
179
|
}
|
|
@@ -156,7 +184,7 @@ var SmartFormProvider = ({
|
|
|
156
184
|
...allErrors
|
|
157
185
|
}));
|
|
158
186
|
return isValid;
|
|
159
|
-
}, [formData, validationRegistry]);
|
|
187
|
+
}, [formData, validationRegistry, getValidationErrorMessage]);
|
|
160
188
|
const submitForm = useCallback(async () => {
|
|
161
189
|
if (!validateAllFields()) {
|
|
162
190
|
return;
|
|
@@ -385,6 +413,8 @@ var SmartFormProvider = ({
|
|
|
385
413
|
fieldRefs,
|
|
386
414
|
registerValidation,
|
|
387
415
|
validationRegistry,
|
|
416
|
+
validationLabels,
|
|
417
|
+
getValidationErrorMessage,
|
|
388
418
|
config,
|
|
389
419
|
setErrors,
|
|
390
420
|
registerSubmitHook: (key, hook) => {
|
|
@@ -541,12 +571,19 @@ var SmartInput = ({
|
|
|
541
571
|
const togglePasswordVisibility = () => {
|
|
542
572
|
setShowPassword(!showPassword);
|
|
543
573
|
};
|
|
574
|
+
const displayName = label || field;
|
|
575
|
+
const builtinValidation = useMemo(() => {
|
|
576
|
+
if (validation || !required) return void 0;
|
|
577
|
+
const preprocess = (v) => v === void 0 || v === null ? "" : v;
|
|
578
|
+
return z.preprocess(preprocess, z.string().min(1, `${displayName} is required`));
|
|
579
|
+
}, [validation, required, displayName]);
|
|
580
|
+
const schemaToRegister = validation ?? builtinValidation;
|
|
544
581
|
useEffect(() => {
|
|
545
|
-
if (
|
|
582
|
+
if (schemaToRegister && !hasRegistered.current) {
|
|
546
583
|
hasRegistered.current = true;
|
|
547
|
-
registerValidation(field,
|
|
584
|
+
registerValidation(field, schemaToRegister, { label: displayName });
|
|
548
585
|
}
|
|
549
|
-
}, [
|
|
586
|
+
}, [schemaToRegister, field, registerValidation, displayName]);
|
|
550
587
|
useEffect(() => {
|
|
551
588
|
if (fieldDetection?.registerField) {
|
|
552
589
|
fieldDetection.registerField(field);
|
|
@@ -704,5 +741,5 @@ var SmartInput = ({
|
|
|
704
741
|
};
|
|
705
742
|
|
|
706
743
|
export { FieldDetectionContext, Input, Label, SmartFormProvider, SmartInput, Textarea, Tooltip, TooltipContent, TooltipProvider, TooltipTrigger, cn, useFieldDetection, useFormField, useSmartForm };
|
|
707
|
-
//# sourceMappingURL=chunk-
|
|
708
|
-
//# sourceMappingURL=chunk-
|
|
744
|
+
//# sourceMappingURL=chunk-DYTQTHGE.js.map
|
|
745
|
+
//# sourceMappingURL=chunk-DYTQTHGE.js.map
|