@tanstack/form-core 0.3.1 → 0.3.3
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/build/legacy/FieldApi.cjs +285 -0
- package/build/legacy/FieldApi.cjs.map +1 -0
- package/build/legacy/FieldApi.d.cts +3 -0
- package/build/legacy/FieldApi.d.ts +3 -0
- package/build/legacy/FieldApi.js +252 -0
- package/build/legacy/FieldApi.js.map +1 -0
- package/build/legacy/FormApi.cjs +257 -0
- package/build/legacy/FormApi.cjs.map +1 -0
- package/build/legacy/FormApi.d.cts +3 -0
- package/build/legacy/FormApi.d.ts +3 -0
- package/build/legacy/FormApi.js +234 -0
- package/build/legacy/FormApi.js.map +1 -0
- package/build/legacy/chunk-4QZDOMDG.js +19 -0
- package/build/legacy/chunk-4QZDOMDG.js.map +1 -0
- package/build/legacy/index.cjs +29 -0
- package/build/legacy/index.cjs.map +1 -0
- package/build/legacy/index.d.cts +172 -0
- package/build/legacy/index.d.ts +172 -0
- package/build/legacy/index.js +5 -0
- package/build/legacy/index.js.map +1 -0
- package/build/legacy/utils.cjs +103 -0
- package/build/legacy/utils.cjs.map +1 -0
- package/build/legacy/utils.d.cts +33 -0
- package/build/legacy/utils.d.ts +33 -0
- package/build/legacy/utils.js +77 -0
- package/build/legacy/utils.js.map +1 -0
- package/build/modern/FieldApi.cjs +267 -0
- package/build/modern/FieldApi.cjs.map +1 -0
- package/build/modern/FieldApi.d.cts +3 -0
- package/build/modern/FieldApi.d.ts +3 -0
- package/build/modern/FieldApi.js +242 -0
- package/build/modern/FieldApi.js.map +1 -0
- package/build/modern/FormApi.cjs +251 -0
- package/build/modern/FormApi.cjs.map +1 -0
- package/build/modern/FormApi.d.cts +3 -0
- package/build/modern/FormApi.d.ts +3 -0
- package/build/modern/FormApi.js +226 -0
- package/build/modern/FormApi.js.map +1 -0
- package/build/modern/index.cjs +29 -0
- package/build/modern/index.cjs.map +1 -0
- package/build/modern/index.d.cts +172 -0
- package/build/modern/index.d.ts +172 -0
- package/build/modern/index.js +5 -0
- package/build/modern/index.js.map +1 -0
- package/build/modern/utils.cjs +103 -0
- package/build/modern/utils.cjs.map +1 -0
- package/build/modern/utils.d.cts +33 -0
- package/build/modern/utils.d.ts +33 -0
- package/build/modern/utils.js +75 -0
- package/build/modern/utils.js.map +1 -0
- package/package.json +1 -1
- package/src/FieldApi.ts +62 -48
|
@@ -0,0 +1,267 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var __defProp = Object.defineProperty;
|
|
3
|
+
var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
|
|
4
|
+
var __getOwnPropNames = Object.getOwnPropertyNames;
|
|
5
|
+
var __hasOwnProp = Object.prototype.hasOwnProperty;
|
|
6
|
+
var __export = (target, all) => {
|
|
7
|
+
for (var name in all)
|
|
8
|
+
__defProp(target, name, { get: all[name], enumerable: true });
|
|
9
|
+
};
|
|
10
|
+
var __copyProps = (to, from, except, desc) => {
|
|
11
|
+
if (from && typeof from === "object" || typeof from === "function") {
|
|
12
|
+
for (let key of __getOwnPropNames(from))
|
|
13
|
+
if (!__hasOwnProp.call(to, key) && key !== except)
|
|
14
|
+
__defProp(to, key, { get: () => from[key], enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable });
|
|
15
|
+
}
|
|
16
|
+
return to;
|
|
17
|
+
};
|
|
18
|
+
var __toCommonJS = (mod) => __copyProps(__defProp({}, "__esModule", { value: true }), mod);
|
|
19
|
+
|
|
20
|
+
// src/FieldApi.ts
|
|
21
|
+
var FieldApi_exports = {};
|
|
22
|
+
__export(FieldApi_exports, {
|
|
23
|
+
FieldApi: () => FieldApi
|
|
24
|
+
});
|
|
25
|
+
module.exports = __toCommonJS(FieldApi_exports);
|
|
26
|
+
var import_store = require("@tanstack/store");
|
|
27
|
+
var uid = 0;
|
|
28
|
+
var FieldApi = class _FieldApi {
|
|
29
|
+
constructor(opts) {
|
|
30
|
+
this.options = {};
|
|
31
|
+
this.mount = () => {
|
|
32
|
+
const info = this.getInfo();
|
|
33
|
+
info.instances[this.uid] = this;
|
|
34
|
+
const unsubscribe = this.form.store.subscribe(() => {
|
|
35
|
+
this.store.batch(() => {
|
|
36
|
+
const nextValue = this.getValue();
|
|
37
|
+
const nextMeta = this.getMeta();
|
|
38
|
+
if (nextValue !== this.state.value) {
|
|
39
|
+
this.store.setState((prev) => ({ ...prev, value: nextValue }));
|
|
40
|
+
}
|
|
41
|
+
if (nextMeta !== this.state.meta) {
|
|
42
|
+
this.store.setState((prev) => ({ ...prev, meta: nextMeta }));
|
|
43
|
+
}
|
|
44
|
+
});
|
|
45
|
+
});
|
|
46
|
+
this.update(this.options);
|
|
47
|
+
this.options.onMount?.(this);
|
|
48
|
+
return () => {
|
|
49
|
+
unsubscribe();
|
|
50
|
+
delete info.instances[this.uid];
|
|
51
|
+
if (!Object.keys(info.instances).length) {
|
|
52
|
+
delete this.form.fieldInfo[this.name];
|
|
53
|
+
}
|
|
54
|
+
};
|
|
55
|
+
};
|
|
56
|
+
this.update = (opts) => {
|
|
57
|
+
if (this.state.value === void 0) {
|
|
58
|
+
const formDefault = opts.form.options.defaultValues?.[opts.name];
|
|
59
|
+
if (opts.defaultValue !== void 0) {
|
|
60
|
+
this.setValue(opts.defaultValue);
|
|
61
|
+
} else if (formDefault !== void 0) {
|
|
62
|
+
this.setValue(formDefault);
|
|
63
|
+
}
|
|
64
|
+
}
|
|
65
|
+
if (this._getMeta() === void 0) {
|
|
66
|
+
this.setMeta(this.state.meta);
|
|
67
|
+
}
|
|
68
|
+
this.options = opts;
|
|
69
|
+
};
|
|
70
|
+
this.getValue = () => {
|
|
71
|
+
return this.form.getFieldValue(this.name);
|
|
72
|
+
};
|
|
73
|
+
this.setValue = (updater, options) => {
|
|
74
|
+
this.form.setFieldValue(this.name, updater, options);
|
|
75
|
+
this.validate("change", this.state.value);
|
|
76
|
+
};
|
|
77
|
+
this._getMeta = () => this.form.getFieldMeta(this.name);
|
|
78
|
+
this.getMeta = () => this._getMeta() ?? {
|
|
79
|
+
isValidating: false,
|
|
80
|
+
isTouched: false,
|
|
81
|
+
touchedErrors: [],
|
|
82
|
+
errors: [],
|
|
83
|
+
errorMap: {},
|
|
84
|
+
...this.options.defaultMeta
|
|
85
|
+
};
|
|
86
|
+
this.setMeta = (updater) => this.form.setFieldMeta(this.name, updater);
|
|
87
|
+
this.getInfo = () => this.form.getFieldInfo(this.name);
|
|
88
|
+
this.pushValue = (value) => this.form.pushFieldValue(this.name, value);
|
|
89
|
+
this.insertValue = (index, value) => this.form.insertFieldValue(this.name, index, value);
|
|
90
|
+
this.removeValue = (index) => this.form.removeFieldValue(this.name, index);
|
|
91
|
+
this.swapValues = (aIndex, bIndex) => this.form.swapFieldValues(this.name, aIndex, bIndex);
|
|
92
|
+
this.getSubField = (name) => new _FieldApi({
|
|
93
|
+
name: `${this.name}.${name}`,
|
|
94
|
+
form: this.form
|
|
95
|
+
});
|
|
96
|
+
this.validateSync = (value = this.state.value, cause) => {
|
|
97
|
+
const { onChange, onBlur } = this.options;
|
|
98
|
+
const validate = cause === "submit" ? void 0 : cause === "change" ? onChange : onBlur;
|
|
99
|
+
if (!validate)
|
|
100
|
+
return;
|
|
101
|
+
const validationCount = (this.getInfo().validationCount || 0) + 1;
|
|
102
|
+
this.getInfo().validationCount = validationCount;
|
|
103
|
+
const error = normalizeError(validate(value, this));
|
|
104
|
+
const errorMapKey = getErrorMapKey(cause);
|
|
105
|
+
if (error && this.state.meta.errorMap[errorMapKey] !== error) {
|
|
106
|
+
this.setMeta((prev) => ({
|
|
107
|
+
...prev,
|
|
108
|
+
errors: [...prev.errors, error],
|
|
109
|
+
errorMap: {
|
|
110
|
+
...prev.errorMap,
|
|
111
|
+
[getErrorMapKey(cause)]: error
|
|
112
|
+
}
|
|
113
|
+
}));
|
|
114
|
+
}
|
|
115
|
+
if (this.state.meta.errorMap[errorMapKey]) {
|
|
116
|
+
this.cancelValidateAsync();
|
|
117
|
+
}
|
|
118
|
+
};
|
|
119
|
+
this.#leaseValidateAsync = () => {
|
|
120
|
+
const count = (this.getInfo().validationAsyncCount || 0) + 1;
|
|
121
|
+
this.getInfo().validationAsyncCount = count;
|
|
122
|
+
return count;
|
|
123
|
+
};
|
|
124
|
+
this.cancelValidateAsync = () => {
|
|
125
|
+
this.#leaseValidateAsync();
|
|
126
|
+
this.setMeta((prev) => ({
|
|
127
|
+
...prev,
|
|
128
|
+
isValidating: false
|
|
129
|
+
}));
|
|
130
|
+
};
|
|
131
|
+
this.validateAsync = async (value = this.state.value, cause) => {
|
|
132
|
+
const {
|
|
133
|
+
onChangeAsync,
|
|
134
|
+
onBlurAsync,
|
|
135
|
+
onSubmitAsync,
|
|
136
|
+
asyncDebounceMs,
|
|
137
|
+
onBlurAsyncDebounceMs,
|
|
138
|
+
onChangeAsyncDebounceMs
|
|
139
|
+
} = this.options;
|
|
140
|
+
const validate = cause === "change" ? onChangeAsync : cause === "submit" ? onSubmitAsync : onBlurAsync;
|
|
141
|
+
if (!validate)
|
|
142
|
+
return [];
|
|
143
|
+
const debounceMs = cause === "submit" ? 0 : (cause === "change" ? onChangeAsyncDebounceMs : onBlurAsyncDebounceMs) ?? asyncDebounceMs ?? 0;
|
|
144
|
+
if (this.state.meta.isValidating !== true)
|
|
145
|
+
this.setMeta((prev) => ({ ...prev, isValidating: true }));
|
|
146
|
+
const validationAsyncCount = this.#leaseValidateAsync();
|
|
147
|
+
const checkLatest = () => validationAsyncCount === this.getInfo().validationAsyncCount;
|
|
148
|
+
if (!this.getInfo().validationPromise) {
|
|
149
|
+
this.getInfo().validationPromise = new Promise((resolve, reject) => {
|
|
150
|
+
this.getInfo().validationResolve = resolve;
|
|
151
|
+
this.getInfo().validationReject = reject;
|
|
152
|
+
});
|
|
153
|
+
}
|
|
154
|
+
if (debounceMs > 0) {
|
|
155
|
+
await new Promise((r) => setTimeout(r, debounceMs));
|
|
156
|
+
}
|
|
157
|
+
if (checkLatest()) {
|
|
158
|
+
const prevErrors = this.getMeta().errors;
|
|
159
|
+
try {
|
|
160
|
+
const rawError = await validate(value, this);
|
|
161
|
+
if (checkLatest()) {
|
|
162
|
+
const error = normalizeError(rawError);
|
|
163
|
+
this.setMeta((prev) => ({
|
|
164
|
+
...prev,
|
|
165
|
+
isValidating: false,
|
|
166
|
+
errors: [...prev.errors, error],
|
|
167
|
+
errorMap: {
|
|
168
|
+
...prev.errorMap,
|
|
169
|
+
[getErrorMapKey(cause)]: error
|
|
170
|
+
}
|
|
171
|
+
}));
|
|
172
|
+
this.getInfo().validationResolve?.([...prevErrors, error]);
|
|
173
|
+
}
|
|
174
|
+
} catch (error) {
|
|
175
|
+
if (checkLatest()) {
|
|
176
|
+
this.getInfo().validationReject?.([...prevErrors, error]);
|
|
177
|
+
throw error;
|
|
178
|
+
}
|
|
179
|
+
} finally {
|
|
180
|
+
if (checkLatest()) {
|
|
181
|
+
this.setMeta((prev) => ({ ...prev, isValidating: false }));
|
|
182
|
+
delete this.getInfo().validationPromise;
|
|
183
|
+
}
|
|
184
|
+
}
|
|
185
|
+
}
|
|
186
|
+
return this.getInfo().validationPromise ?? [];
|
|
187
|
+
};
|
|
188
|
+
this.validate = (cause, value) => {
|
|
189
|
+
if (!this.state.meta.isTouched)
|
|
190
|
+
return [];
|
|
191
|
+
this.validateSync(value, cause);
|
|
192
|
+
const errorMapKey = getErrorMapKey(cause);
|
|
193
|
+
if (this.getMeta().errorMap[errorMapKey]) {
|
|
194
|
+
if (!this.options.asyncAlways) {
|
|
195
|
+
return this.state.meta.errors;
|
|
196
|
+
}
|
|
197
|
+
}
|
|
198
|
+
return this.validateAsync(value, cause);
|
|
199
|
+
};
|
|
200
|
+
this.handleChange = (updater) => {
|
|
201
|
+
this.setValue(updater, { touch: true });
|
|
202
|
+
};
|
|
203
|
+
this.handleBlur = () => {
|
|
204
|
+
const prevTouched = this.state.meta.isTouched;
|
|
205
|
+
if (!prevTouched) {
|
|
206
|
+
this.setMeta((prev) => ({ ...prev, isTouched: true }));
|
|
207
|
+
this.validate("change");
|
|
208
|
+
}
|
|
209
|
+
this.validate("blur");
|
|
210
|
+
};
|
|
211
|
+
this.form = opts.form;
|
|
212
|
+
this.uid = uid++;
|
|
213
|
+
this.name = opts.name;
|
|
214
|
+
this.store = new import_store.Store(
|
|
215
|
+
{
|
|
216
|
+
value: this.getValue(),
|
|
217
|
+
// eslint-disable-next-line @typescript-eslint/no-unnecessary-condition
|
|
218
|
+
meta: this._getMeta() ?? {
|
|
219
|
+
isValidating: false,
|
|
220
|
+
isTouched: false,
|
|
221
|
+
touchedErrors: [],
|
|
222
|
+
errors: [],
|
|
223
|
+
errorMap: {},
|
|
224
|
+
...opts.defaultMeta
|
|
225
|
+
}
|
|
226
|
+
},
|
|
227
|
+
{
|
|
228
|
+
onUpdate: () => {
|
|
229
|
+
const state = this.store.state;
|
|
230
|
+
state.meta.touchedErrors = state.meta.isTouched ? state.meta.errors : [];
|
|
231
|
+
this.prevState = state;
|
|
232
|
+
this.state = state;
|
|
233
|
+
}
|
|
234
|
+
}
|
|
235
|
+
);
|
|
236
|
+
this.state = this.store.state;
|
|
237
|
+
this.prevState = this.state;
|
|
238
|
+
this.options = opts;
|
|
239
|
+
}
|
|
240
|
+
#leaseValidateAsync;
|
|
241
|
+
};
|
|
242
|
+
function normalizeError(rawError) {
|
|
243
|
+
if (rawError) {
|
|
244
|
+
if (typeof rawError !== "string") {
|
|
245
|
+
return "Invalid Form Values";
|
|
246
|
+
}
|
|
247
|
+
return rawError;
|
|
248
|
+
}
|
|
249
|
+
return void 0;
|
|
250
|
+
}
|
|
251
|
+
function getErrorMapKey(cause) {
|
|
252
|
+
switch (cause) {
|
|
253
|
+
case "submit":
|
|
254
|
+
return "onSubmit";
|
|
255
|
+
case "change":
|
|
256
|
+
return "onChange";
|
|
257
|
+
case "blur":
|
|
258
|
+
return "onBlur";
|
|
259
|
+
case "mount":
|
|
260
|
+
return "onMount";
|
|
261
|
+
}
|
|
262
|
+
}
|
|
263
|
+
// Annotate the CommonJS export names for ESM import in node:
|
|
264
|
+
0 && (module.exports = {
|
|
265
|
+
FieldApi
|
|
266
|
+
});
|
|
267
|
+
//# sourceMappingURL=FieldApi.cjs.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../../src/FieldApi.ts"],"sourcesContent":["import { type DeepKeys, type DeepValue, type Updater } from './utils'\nimport type { FormApi, ValidationError, ValidationErrorMap } from './FormApi'\nimport { Store } from '@tanstack/store'\n\nexport type ValidationCause = 'change' | 'blur' | 'submit' | 'mount'\n\ntype ValidateFn<TData, TFormData> = (\n value: TData,\n fieldApi: FieldApi<TData, TFormData>,\n) => ValidationError\n\ntype ValidateAsyncFn<TData, TFormData> = (\n value: TData,\n fieldApi: FieldApi<TData, TFormData>,\n) => ValidationError | Promise<ValidationError>\n\nexport interface FieldOptions<\n _TData,\n TFormData,\n /**\n * This allows us to restrict the name to only be a valid field name while\n * also assigning it to a generic\n */\n TName = unknown extends TFormData ? string : DeepKeys<TFormData>,\n /**\n * If TData is unknown, we can use the TName generic to determine the type\n */\n TData = unknown extends _TData ? DeepValue<TFormData, TName> : _TData,\n> {\n name: TName\n index?: TData extends any[] ? number : never\n defaultValue?: TData\n asyncDebounceMs?: number\n asyncAlways?: boolean\n onMount?: (formApi: FieldApi<TData, TFormData>) => void\n onChange?: ValidateFn<TData, TFormData>\n onChangeAsync?: ValidateAsyncFn<TData, TFormData>\n onChangeAsyncDebounceMs?: number\n onBlur?: ValidateFn<TData, TFormData>\n onBlurAsync?: ValidateAsyncFn<TData, TFormData>\n onBlurAsyncDebounceMs?: number\n onSubmitAsync?: ValidateAsyncFn<TData, TFormData>\n defaultMeta?: Partial<FieldMeta>\n}\n\nexport interface FieldApiOptions<\n _TData,\n TFormData,\n /**\n * This allows us to restrict the name to only be a valid field name while\n * also assigning it to a generic\n */\n TName = unknown extends TFormData ? string : DeepKeys<TFormData>,\n /**\n * If TData is unknown, we can use the TName generic to determine the type\n */\n TData = unknown extends _TData ? DeepValue<TFormData, TName> : _TData,\n> extends FieldOptions<_TData, TFormData, TName, TData> {\n form: FormApi<TFormData>\n}\n\nexport type FieldMeta = {\n isTouched: boolean\n touchedErrors: ValidationError[]\n errors: ValidationError[]\n errorMap: ValidationErrorMap\n isValidating: boolean\n}\n\nlet uid = 0\n\nexport type FieldState<TData> = {\n value: TData\n meta: FieldMeta\n}\n\ntype GetTData<\n TData,\n TFormData,\n Opts extends FieldApiOptions<TData, TFormData>,\n> = Opts extends FieldApiOptions<\n infer _TData,\n infer _TFormData,\n infer _TName,\n infer RealTData\n>\n ? RealTData\n : never\n\nexport class FieldApi<\n _TData,\n TFormData,\n Opts extends FieldApiOptions<_TData, TFormData> = FieldApiOptions<\n _TData,\n TFormData\n >,\n TData extends GetTData<_TData, TFormData, Opts> = GetTData<\n _TData,\n TFormData,\n Opts\n >,\n> {\n uid: number\n form: Opts['form']\n name!: DeepKeys<TFormData>\n options: Opts = {} as any\n store!: Store<FieldState<TData>>\n state!: FieldState<TData>\n prevState!: FieldState<TData>\n\n constructor(\n opts: Opts & {\n form: FormApi<TFormData>\n },\n ) {\n this.form = opts.form\n this.uid = uid++\n // Support field prefixing from FieldScope\n // let fieldPrefix = ''\n // if (this.form.fieldName) {\n // fieldPrefix = `${this.form.fieldName}.`\n // }\n\n this.name = opts.name as any\n\n this.store = new Store<FieldState<TData>>(\n {\n value: this.getValue(),\n // eslint-disable-next-line @typescript-eslint/no-unnecessary-condition\n meta: this._getMeta() ?? {\n isValidating: false,\n isTouched: false,\n touchedErrors: [],\n errors: [],\n errorMap: {},\n ...opts.defaultMeta,\n },\n },\n {\n onUpdate: () => {\n const state = this.store.state\n\n state.meta.touchedErrors = state.meta.isTouched\n ? state.meta.errors\n : []\n\n this.prevState = state\n this.state = state\n },\n },\n )\n\n this.state = this.store.state\n this.prevState = this.state\n this.options = opts as never\n }\n\n mount = () => {\n const info = this.getInfo()\n info.instances[this.uid] = this as never\n\n const unsubscribe = this.form.store.subscribe(() => {\n this.store.batch(() => {\n const nextValue = this.getValue()\n const nextMeta = this.getMeta()\n\n if (nextValue !== this.state.value) {\n this.store.setState((prev) => ({ ...prev, value: nextValue }))\n }\n\n if (nextMeta !== this.state.meta) {\n this.store.setState((prev) => ({ ...prev, meta: nextMeta }))\n }\n })\n })\n\n this.update(this.options as never)\n this.options.onMount?.(this as never)\n\n return () => {\n unsubscribe()\n delete info.instances[this.uid]\n if (!Object.keys(info.instances).length) {\n delete this.form.fieldInfo[this.name]\n }\n }\n }\n\n update = (opts: FieldApiOptions<TData, TFormData>) => {\n // Default Value\n // eslint-disable-next-line @typescript-eslint/no-unnecessary-condition\n if (this.state.value === undefined) {\n const formDefault =\n opts.form.options.defaultValues?.[opts.name as keyof TFormData]\n\n if (opts.defaultValue !== undefined) {\n this.setValue(opts.defaultValue as never)\n } else if (formDefault !== undefined) {\n this.setValue(formDefault as never)\n }\n }\n\n // Default Meta\n if (this._getMeta() === undefined) {\n this.setMeta(this.state.meta)\n }\n\n this.options = opts as never\n }\n\n getValue = (): TData => {\n return this.form.getFieldValue(this.name)\n }\n\n setValue = (\n updater: Updater<TData>,\n options?: { touch?: boolean; notify?: boolean },\n ) => {\n this.form.setFieldValue(this.name, updater as never, options)\n this.validate('change', this.state.value)\n }\n\n _getMeta = () => this.form.getFieldMeta(this.name)\n getMeta = () =>\n this._getMeta() ??\n ({\n isValidating: false,\n isTouched: false,\n touchedErrors: [],\n errors: [],\n errorMap: {},\n ...this.options.defaultMeta,\n } as FieldMeta)\n\n setMeta = (updater: Updater<FieldMeta>) =>\n this.form.setFieldMeta(this.name, updater)\n\n getInfo = () => this.form.getFieldInfo(this.name)\n\n pushValue = (value: TData extends any[] ? TData[number] : never) =>\n this.form.pushFieldValue(this.name, value as any)\n\n insertValue = (\n index: number,\n value: TData extends any[] ? TData[number] : never,\n ) => this.form.insertFieldValue(this.name, index, value as any)\n\n removeValue = (index: number) => this.form.removeFieldValue(this.name, index)\n\n swapValues = (aIndex: number, bIndex: number) =>\n this.form.swapFieldValues(this.name, aIndex, bIndex)\n\n getSubField = <TName extends DeepKeys<TData>>(name: TName) =>\n new FieldApi<DeepValue<TData, TName>, TFormData>({\n name: `${this.name}.${name}` as never,\n form: this.form,\n })\n\n validateSync = (value = this.state.value, cause: ValidationCause) => {\n const { onChange, onBlur } = this.options\n const validate =\n cause === 'submit' ? undefined : cause === 'change' ? onChange : onBlur\n if (!validate) return\n\n // Use the validationCount for all field instances to\n // track freshness of the validation\n const validationCount = (this.getInfo().validationCount || 0) + 1\n this.getInfo().validationCount = validationCount\n const error = normalizeError(validate(value as never, this as never))\n const errorMapKey = getErrorMapKey(cause)\n if (error && this.state.meta.errorMap[errorMapKey] !== error) {\n this.setMeta((prev) => ({\n ...prev,\n errors: [...prev.errors, error],\n errorMap: {\n ...prev.errorMap,\n [getErrorMapKey(cause)]: error,\n },\n }))\n }\n\n // If a sync error is encountered for the errorMapKey (eg. onChange), cancel any async validation\n if (this.state.meta.errorMap[errorMapKey]) {\n this.cancelValidateAsync()\n }\n }\n\n #leaseValidateAsync = () => {\n const count = (this.getInfo().validationAsyncCount || 0) + 1\n this.getInfo().validationAsyncCount = count\n return count\n }\n\n cancelValidateAsync = () => {\n // Lease a new validation count to ignore any pending validations\n this.#leaseValidateAsync()\n // Cancel any pending validation state\n this.setMeta((prev) => ({\n ...prev,\n isValidating: false,\n }))\n }\n\n validateAsync = async (value = this.state.value, cause: ValidationCause) => {\n const {\n onChangeAsync,\n onBlurAsync,\n onSubmitAsync,\n asyncDebounceMs,\n onBlurAsyncDebounceMs,\n onChangeAsyncDebounceMs,\n } = this.options\n\n const validate =\n cause === 'change'\n ? onChangeAsync\n : cause === 'submit'\n ? onSubmitAsync\n : onBlurAsync\n if (!validate) return []\n const debounceMs =\n cause === 'submit'\n ? 0\n : (cause === 'change'\n ? onChangeAsyncDebounceMs\n : onBlurAsyncDebounceMs) ??\n asyncDebounceMs ??\n 0\n\n if (this.state.meta.isValidating !== true)\n this.setMeta((prev) => ({ ...prev, isValidating: true }))\n\n // Use the validationCount for all field instances to\n // track freshness of the validation\n const validationAsyncCount = this.#leaseValidateAsync()\n\n const checkLatest = () =>\n validationAsyncCount === this.getInfo().validationAsyncCount\n\n if (!this.getInfo().validationPromise) {\n this.getInfo().validationPromise = new Promise((resolve, reject) => {\n this.getInfo().validationResolve = resolve\n this.getInfo().validationReject = reject\n })\n }\n\n if (debounceMs > 0) {\n await new Promise((r) => setTimeout(r, debounceMs))\n }\n\n // Only kick off validation if this validation is the latest attempt\n if (checkLatest()) {\n const prevErrors = this.getMeta().errors\n try {\n const rawError = await validate(value as never, this as never)\n if (checkLatest()) {\n const error = normalizeError(rawError)\n this.setMeta((prev) => ({\n ...prev,\n isValidating: false,\n errors: [...prev.errors, error],\n errorMap: {\n ...prev.errorMap,\n [getErrorMapKey(cause)]: error,\n },\n }))\n this.getInfo().validationResolve?.([...prevErrors, error])\n }\n } catch (error) {\n if (checkLatest()) {\n this.getInfo().validationReject?.([...prevErrors, error])\n throw error\n }\n } finally {\n if (checkLatest()) {\n this.setMeta((prev) => ({ ...prev, isValidating: false }))\n delete this.getInfo().validationPromise\n }\n }\n }\n\n // Always return the latest validation promise to the caller\n return this.getInfo().validationPromise ?? []\n }\n\n validate = (\n cause: ValidationCause,\n value?: TData,\n ): ValidationError[] | Promise<ValidationError[]> => {\n // If the field is pristine and validatePristine is false, do not validate\n if (!this.state.meta.isTouched) return []\n // Attempt to sync validate first\n this.validateSync(value, cause)\n\n const errorMapKey = getErrorMapKey(cause)\n // If there is an error mapped to the errorMapKey (eg. onChange, onBlur, onSubmit), return the errors array, do not attempt async validation\n if (this.getMeta().errorMap[errorMapKey]) {\n if (!this.options.asyncAlways) {\n return this.state.meta.errors\n }\n }\n // No error? Attempt async validation\n return this.validateAsync(value, cause)\n }\n\n handleChange = (updater: Updater<TData>) => {\n this.setValue(updater, { touch: true })\n }\n\n handleBlur = () => {\n const prevTouched = this.state.meta.isTouched\n if (!prevTouched) {\n this.setMeta((prev) => ({ ...prev, isTouched: true }))\n this.validate('change')\n }\n this.validate('blur')\n }\n}\n\nfunction normalizeError(rawError?: ValidationError) {\n if (rawError) {\n if (typeof rawError !== 'string') {\n return 'Invalid Form Values'\n }\n\n return rawError\n }\n\n return undefined\n}\n\nfunction getErrorMapKey(cause: ValidationCause) {\n switch (cause) {\n case 'submit':\n return 'onSubmit'\n case 'change':\n return 'onChange'\n case 'blur':\n return 'onBlur'\n case 'mount':\n return 'onMount'\n }\n}\n"],"mappings":"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","names":[]}
|
|
@@ -0,0 +1,242 @@
|
|
|
1
|
+
// src/FieldApi.ts
|
|
2
|
+
import { Store } from "@tanstack/store";
|
|
3
|
+
var uid = 0;
|
|
4
|
+
var FieldApi = class _FieldApi {
|
|
5
|
+
constructor(opts) {
|
|
6
|
+
this.options = {};
|
|
7
|
+
this.mount = () => {
|
|
8
|
+
const info = this.getInfo();
|
|
9
|
+
info.instances[this.uid] = this;
|
|
10
|
+
const unsubscribe = this.form.store.subscribe(() => {
|
|
11
|
+
this.store.batch(() => {
|
|
12
|
+
const nextValue = this.getValue();
|
|
13
|
+
const nextMeta = this.getMeta();
|
|
14
|
+
if (nextValue !== this.state.value) {
|
|
15
|
+
this.store.setState((prev) => ({ ...prev, value: nextValue }));
|
|
16
|
+
}
|
|
17
|
+
if (nextMeta !== this.state.meta) {
|
|
18
|
+
this.store.setState((prev) => ({ ...prev, meta: nextMeta }));
|
|
19
|
+
}
|
|
20
|
+
});
|
|
21
|
+
});
|
|
22
|
+
this.update(this.options);
|
|
23
|
+
this.options.onMount?.(this);
|
|
24
|
+
return () => {
|
|
25
|
+
unsubscribe();
|
|
26
|
+
delete info.instances[this.uid];
|
|
27
|
+
if (!Object.keys(info.instances).length) {
|
|
28
|
+
delete this.form.fieldInfo[this.name];
|
|
29
|
+
}
|
|
30
|
+
};
|
|
31
|
+
};
|
|
32
|
+
this.update = (opts) => {
|
|
33
|
+
if (this.state.value === void 0) {
|
|
34
|
+
const formDefault = opts.form.options.defaultValues?.[opts.name];
|
|
35
|
+
if (opts.defaultValue !== void 0) {
|
|
36
|
+
this.setValue(opts.defaultValue);
|
|
37
|
+
} else if (formDefault !== void 0) {
|
|
38
|
+
this.setValue(formDefault);
|
|
39
|
+
}
|
|
40
|
+
}
|
|
41
|
+
if (this._getMeta() === void 0) {
|
|
42
|
+
this.setMeta(this.state.meta);
|
|
43
|
+
}
|
|
44
|
+
this.options = opts;
|
|
45
|
+
};
|
|
46
|
+
this.getValue = () => {
|
|
47
|
+
return this.form.getFieldValue(this.name);
|
|
48
|
+
};
|
|
49
|
+
this.setValue = (updater, options) => {
|
|
50
|
+
this.form.setFieldValue(this.name, updater, options);
|
|
51
|
+
this.validate("change", this.state.value);
|
|
52
|
+
};
|
|
53
|
+
this._getMeta = () => this.form.getFieldMeta(this.name);
|
|
54
|
+
this.getMeta = () => this._getMeta() ?? {
|
|
55
|
+
isValidating: false,
|
|
56
|
+
isTouched: false,
|
|
57
|
+
touchedErrors: [],
|
|
58
|
+
errors: [],
|
|
59
|
+
errorMap: {},
|
|
60
|
+
...this.options.defaultMeta
|
|
61
|
+
};
|
|
62
|
+
this.setMeta = (updater) => this.form.setFieldMeta(this.name, updater);
|
|
63
|
+
this.getInfo = () => this.form.getFieldInfo(this.name);
|
|
64
|
+
this.pushValue = (value) => this.form.pushFieldValue(this.name, value);
|
|
65
|
+
this.insertValue = (index, value) => this.form.insertFieldValue(this.name, index, value);
|
|
66
|
+
this.removeValue = (index) => this.form.removeFieldValue(this.name, index);
|
|
67
|
+
this.swapValues = (aIndex, bIndex) => this.form.swapFieldValues(this.name, aIndex, bIndex);
|
|
68
|
+
this.getSubField = (name) => new _FieldApi({
|
|
69
|
+
name: `${this.name}.${name}`,
|
|
70
|
+
form: this.form
|
|
71
|
+
});
|
|
72
|
+
this.validateSync = (value = this.state.value, cause) => {
|
|
73
|
+
const { onChange, onBlur } = this.options;
|
|
74
|
+
const validate = cause === "submit" ? void 0 : cause === "change" ? onChange : onBlur;
|
|
75
|
+
if (!validate)
|
|
76
|
+
return;
|
|
77
|
+
const validationCount = (this.getInfo().validationCount || 0) + 1;
|
|
78
|
+
this.getInfo().validationCount = validationCount;
|
|
79
|
+
const error = normalizeError(validate(value, this));
|
|
80
|
+
const errorMapKey = getErrorMapKey(cause);
|
|
81
|
+
if (error && this.state.meta.errorMap[errorMapKey] !== error) {
|
|
82
|
+
this.setMeta((prev) => ({
|
|
83
|
+
...prev,
|
|
84
|
+
errors: [...prev.errors, error],
|
|
85
|
+
errorMap: {
|
|
86
|
+
...prev.errorMap,
|
|
87
|
+
[getErrorMapKey(cause)]: error
|
|
88
|
+
}
|
|
89
|
+
}));
|
|
90
|
+
}
|
|
91
|
+
if (this.state.meta.errorMap[errorMapKey]) {
|
|
92
|
+
this.cancelValidateAsync();
|
|
93
|
+
}
|
|
94
|
+
};
|
|
95
|
+
this.#leaseValidateAsync = () => {
|
|
96
|
+
const count = (this.getInfo().validationAsyncCount || 0) + 1;
|
|
97
|
+
this.getInfo().validationAsyncCount = count;
|
|
98
|
+
return count;
|
|
99
|
+
};
|
|
100
|
+
this.cancelValidateAsync = () => {
|
|
101
|
+
this.#leaseValidateAsync();
|
|
102
|
+
this.setMeta((prev) => ({
|
|
103
|
+
...prev,
|
|
104
|
+
isValidating: false
|
|
105
|
+
}));
|
|
106
|
+
};
|
|
107
|
+
this.validateAsync = async (value = this.state.value, cause) => {
|
|
108
|
+
const {
|
|
109
|
+
onChangeAsync,
|
|
110
|
+
onBlurAsync,
|
|
111
|
+
onSubmitAsync,
|
|
112
|
+
asyncDebounceMs,
|
|
113
|
+
onBlurAsyncDebounceMs,
|
|
114
|
+
onChangeAsyncDebounceMs
|
|
115
|
+
} = this.options;
|
|
116
|
+
const validate = cause === "change" ? onChangeAsync : cause === "submit" ? onSubmitAsync : onBlurAsync;
|
|
117
|
+
if (!validate)
|
|
118
|
+
return [];
|
|
119
|
+
const debounceMs = cause === "submit" ? 0 : (cause === "change" ? onChangeAsyncDebounceMs : onBlurAsyncDebounceMs) ?? asyncDebounceMs ?? 0;
|
|
120
|
+
if (this.state.meta.isValidating !== true)
|
|
121
|
+
this.setMeta((prev) => ({ ...prev, isValidating: true }));
|
|
122
|
+
const validationAsyncCount = this.#leaseValidateAsync();
|
|
123
|
+
const checkLatest = () => validationAsyncCount === this.getInfo().validationAsyncCount;
|
|
124
|
+
if (!this.getInfo().validationPromise) {
|
|
125
|
+
this.getInfo().validationPromise = new Promise((resolve, reject) => {
|
|
126
|
+
this.getInfo().validationResolve = resolve;
|
|
127
|
+
this.getInfo().validationReject = reject;
|
|
128
|
+
});
|
|
129
|
+
}
|
|
130
|
+
if (debounceMs > 0) {
|
|
131
|
+
await new Promise((r) => setTimeout(r, debounceMs));
|
|
132
|
+
}
|
|
133
|
+
if (checkLatest()) {
|
|
134
|
+
const prevErrors = this.getMeta().errors;
|
|
135
|
+
try {
|
|
136
|
+
const rawError = await validate(value, this);
|
|
137
|
+
if (checkLatest()) {
|
|
138
|
+
const error = normalizeError(rawError);
|
|
139
|
+
this.setMeta((prev) => ({
|
|
140
|
+
...prev,
|
|
141
|
+
isValidating: false,
|
|
142
|
+
errors: [...prev.errors, error],
|
|
143
|
+
errorMap: {
|
|
144
|
+
...prev.errorMap,
|
|
145
|
+
[getErrorMapKey(cause)]: error
|
|
146
|
+
}
|
|
147
|
+
}));
|
|
148
|
+
this.getInfo().validationResolve?.([...prevErrors, error]);
|
|
149
|
+
}
|
|
150
|
+
} catch (error) {
|
|
151
|
+
if (checkLatest()) {
|
|
152
|
+
this.getInfo().validationReject?.([...prevErrors, error]);
|
|
153
|
+
throw error;
|
|
154
|
+
}
|
|
155
|
+
} finally {
|
|
156
|
+
if (checkLatest()) {
|
|
157
|
+
this.setMeta((prev) => ({ ...prev, isValidating: false }));
|
|
158
|
+
delete this.getInfo().validationPromise;
|
|
159
|
+
}
|
|
160
|
+
}
|
|
161
|
+
}
|
|
162
|
+
return this.getInfo().validationPromise ?? [];
|
|
163
|
+
};
|
|
164
|
+
this.validate = (cause, value) => {
|
|
165
|
+
if (!this.state.meta.isTouched)
|
|
166
|
+
return [];
|
|
167
|
+
this.validateSync(value, cause);
|
|
168
|
+
const errorMapKey = getErrorMapKey(cause);
|
|
169
|
+
if (this.getMeta().errorMap[errorMapKey]) {
|
|
170
|
+
if (!this.options.asyncAlways) {
|
|
171
|
+
return this.state.meta.errors;
|
|
172
|
+
}
|
|
173
|
+
}
|
|
174
|
+
return this.validateAsync(value, cause);
|
|
175
|
+
};
|
|
176
|
+
this.handleChange = (updater) => {
|
|
177
|
+
this.setValue(updater, { touch: true });
|
|
178
|
+
};
|
|
179
|
+
this.handleBlur = () => {
|
|
180
|
+
const prevTouched = this.state.meta.isTouched;
|
|
181
|
+
if (!prevTouched) {
|
|
182
|
+
this.setMeta((prev) => ({ ...prev, isTouched: true }));
|
|
183
|
+
this.validate("change");
|
|
184
|
+
}
|
|
185
|
+
this.validate("blur");
|
|
186
|
+
};
|
|
187
|
+
this.form = opts.form;
|
|
188
|
+
this.uid = uid++;
|
|
189
|
+
this.name = opts.name;
|
|
190
|
+
this.store = new Store(
|
|
191
|
+
{
|
|
192
|
+
value: this.getValue(),
|
|
193
|
+
// eslint-disable-next-line @typescript-eslint/no-unnecessary-condition
|
|
194
|
+
meta: this._getMeta() ?? {
|
|
195
|
+
isValidating: false,
|
|
196
|
+
isTouched: false,
|
|
197
|
+
touchedErrors: [],
|
|
198
|
+
errors: [],
|
|
199
|
+
errorMap: {},
|
|
200
|
+
...opts.defaultMeta
|
|
201
|
+
}
|
|
202
|
+
},
|
|
203
|
+
{
|
|
204
|
+
onUpdate: () => {
|
|
205
|
+
const state = this.store.state;
|
|
206
|
+
state.meta.touchedErrors = state.meta.isTouched ? state.meta.errors : [];
|
|
207
|
+
this.prevState = state;
|
|
208
|
+
this.state = state;
|
|
209
|
+
}
|
|
210
|
+
}
|
|
211
|
+
);
|
|
212
|
+
this.state = this.store.state;
|
|
213
|
+
this.prevState = this.state;
|
|
214
|
+
this.options = opts;
|
|
215
|
+
}
|
|
216
|
+
#leaseValidateAsync;
|
|
217
|
+
};
|
|
218
|
+
function normalizeError(rawError) {
|
|
219
|
+
if (rawError) {
|
|
220
|
+
if (typeof rawError !== "string") {
|
|
221
|
+
return "Invalid Form Values";
|
|
222
|
+
}
|
|
223
|
+
return rawError;
|
|
224
|
+
}
|
|
225
|
+
return void 0;
|
|
226
|
+
}
|
|
227
|
+
function getErrorMapKey(cause) {
|
|
228
|
+
switch (cause) {
|
|
229
|
+
case "submit":
|
|
230
|
+
return "onSubmit";
|
|
231
|
+
case "change":
|
|
232
|
+
return "onChange";
|
|
233
|
+
case "blur":
|
|
234
|
+
return "onBlur";
|
|
235
|
+
case "mount":
|
|
236
|
+
return "onMount";
|
|
237
|
+
}
|
|
238
|
+
}
|
|
239
|
+
export {
|
|
240
|
+
FieldApi
|
|
241
|
+
};
|
|
242
|
+
//# sourceMappingURL=FieldApi.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../../src/FieldApi.ts"],"sourcesContent":["import { type DeepKeys, type DeepValue, type Updater } from './utils'\nimport type { FormApi, ValidationError, ValidationErrorMap } from './FormApi'\nimport { Store } from '@tanstack/store'\n\nexport type ValidationCause = 'change' | 'blur' | 'submit' | 'mount'\n\ntype ValidateFn<TData, TFormData> = (\n value: TData,\n fieldApi: FieldApi<TData, TFormData>,\n) => ValidationError\n\ntype ValidateAsyncFn<TData, TFormData> = (\n value: TData,\n fieldApi: FieldApi<TData, TFormData>,\n) => ValidationError | Promise<ValidationError>\n\nexport interface FieldOptions<\n _TData,\n TFormData,\n /**\n * This allows us to restrict the name to only be a valid field name while\n * also assigning it to a generic\n */\n TName = unknown extends TFormData ? string : DeepKeys<TFormData>,\n /**\n * If TData is unknown, we can use the TName generic to determine the type\n */\n TData = unknown extends _TData ? DeepValue<TFormData, TName> : _TData,\n> {\n name: TName\n index?: TData extends any[] ? number : never\n defaultValue?: TData\n asyncDebounceMs?: number\n asyncAlways?: boolean\n onMount?: (formApi: FieldApi<TData, TFormData>) => void\n onChange?: ValidateFn<TData, TFormData>\n onChangeAsync?: ValidateAsyncFn<TData, TFormData>\n onChangeAsyncDebounceMs?: number\n onBlur?: ValidateFn<TData, TFormData>\n onBlurAsync?: ValidateAsyncFn<TData, TFormData>\n onBlurAsyncDebounceMs?: number\n onSubmitAsync?: ValidateAsyncFn<TData, TFormData>\n defaultMeta?: Partial<FieldMeta>\n}\n\nexport interface FieldApiOptions<\n _TData,\n TFormData,\n /**\n * This allows us to restrict the name to only be a valid field name while\n * also assigning it to a generic\n */\n TName = unknown extends TFormData ? string : DeepKeys<TFormData>,\n /**\n * If TData is unknown, we can use the TName generic to determine the type\n */\n TData = unknown extends _TData ? DeepValue<TFormData, TName> : _TData,\n> extends FieldOptions<_TData, TFormData, TName, TData> {\n form: FormApi<TFormData>\n}\n\nexport type FieldMeta = {\n isTouched: boolean\n touchedErrors: ValidationError[]\n errors: ValidationError[]\n errorMap: ValidationErrorMap\n isValidating: boolean\n}\n\nlet uid = 0\n\nexport type FieldState<TData> = {\n value: TData\n meta: FieldMeta\n}\n\ntype GetTData<\n TData,\n TFormData,\n Opts extends FieldApiOptions<TData, TFormData>,\n> = Opts extends FieldApiOptions<\n infer _TData,\n infer _TFormData,\n infer _TName,\n infer RealTData\n>\n ? RealTData\n : never\n\nexport class FieldApi<\n _TData,\n TFormData,\n Opts extends FieldApiOptions<_TData, TFormData> = FieldApiOptions<\n _TData,\n TFormData\n >,\n TData extends GetTData<_TData, TFormData, Opts> = GetTData<\n _TData,\n TFormData,\n Opts\n >,\n> {\n uid: number\n form: Opts['form']\n name!: DeepKeys<TFormData>\n options: Opts = {} as any\n store!: Store<FieldState<TData>>\n state!: FieldState<TData>\n prevState!: FieldState<TData>\n\n constructor(\n opts: Opts & {\n form: FormApi<TFormData>\n },\n ) {\n this.form = opts.form\n this.uid = uid++\n // Support field prefixing from FieldScope\n // let fieldPrefix = ''\n // if (this.form.fieldName) {\n // fieldPrefix = `${this.form.fieldName}.`\n // }\n\n this.name = opts.name as any\n\n this.store = new Store<FieldState<TData>>(\n {\n value: this.getValue(),\n // eslint-disable-next-line @typescript-eslint/no-unnecessary-condition\n meta: this._getMeta() ?? {\n isValidating: false,\n isTouched: false,\n touchedErrors: [],\n errors: [],\n errorMap: {},\n ...opts.defaultMeta,\n },\n },\n {\n onUpdate: () => {\n const state = this.store.state\n\n state.meta.touchedErrors = state.meta.isTouched\n ? state.meta.errors\n : []\n\n this.prevState = state\n this.state = state\n },\n },\n )\n\n this.state = this.store.state\n this.prevState = this.state\n this.options = opts as never\n }\n\n mount = () => {\n const info = this.getInfo()\n info.instances[this.uid] = this as never\n\n const unsubscribe = this.form.store.subscribe(() => {\n this.store.batch(() => {\n const nextValue = this.getValue()\n const nextMeta = this.getMeta()\n\n if (nextValue !== this.state.value) {\n this.store.setState((prev) => ({ ...prev, value: nextValue }))\n }\n\n if (nextMeta !== this.state.meta) {\n this.store.setState((prev) => ({ ...prev, meta: nextMeta }))\n }\n })\n })\n\n this.update(this.options as never)\n this.options.onMount?.(this as never)\n\n return () => {\n unsubscribe()\n delete info.instances[this.uid]\n if (!Object.keys(info.instances).length) {\n delete this.form.fieldInfo[this.name]\n }\n }\n }\n\n update = (opts: FieldApiOptions<TData, TFormData>) => {\n // Default Value\n // eslint-disable-next-line @typescript-eslint/no-unnecessary-condition\n if (this.state.value === undefined) {\n const formDefault =\n opts.form.options.defaultValues?.[opts.name as keyof TFormData]\n\n if (opts.defaultValue !== undefined) {\n this.setValue(opts.defaultValue as never)\n } else if (formDefault !== undefined) {\n this.setValue(formDefault as never)\n }\n }\n\n // Default Meta\n if (this._getMeta() === undefined) {\n this.setMeta(this.state.meta)\n }\n\n this.options = opts as never\n }\n\n getValue = (): TData => {\n return this.form.getFieldValue(this.name)\n }\n\n setValue = (\n updater: Updater<TData>,\n options?: { touch?: boolean; notify?: boolean },\n ) => {\n this.form.setFieldValue(this.name, updater as never, options)\n this.validate('change', this.state.value)\n }\n\n _getMeta = () => this.form.getFieldMeta(this.name)\n getMeta = () =>\n this._getMeta() ??\n ({\n isValidating: false,\n isTouched: false,\n touchedErrors: [],\n errors: [],\n errorMap: {},\n ...this.options.defaultMeta,\n } as FieldMeta)\n\n setMeta = (updater: Updater<FieldMeta>) =>\n this.form.setFieldMeta(this.name, updater)\n\n getInfo = () => this.form.getFieldInfo(this.name)\n\n pushValue = (value: TData extends any[] ? TData[number] : never) =>\n this.form.pushFieldValue(this.name, value as any)\n\n insertValue = (\n index: number,\n value: TData extends any[] ? TData[number] : never,\n ) => this.form.insertFieldValue(this.name, index, value as any)\n\n removeValue = (index: number) => this.form.removeFieldValue(this.name, index)\n\n swapValues = (aIndex: number, bIndex: number) =>\n this.form.swapFieldValues(this.name, aIndex, bIndex)\n\n getSubField = <TName extends DeepKeys<TData>>(name: TName) =>\n new FieldApi<DeepValue<TData, TName>, TFormData>({\n name: `${this.name}.${name}` as never,\n form: this.form,\n })\n\n validateSync = (value = this.state.value, cause: ValidationCause) => {\n const { onChange, onBlur } = this.options\n const validate =\n cause === 'submit' ? undefined : cause === 'change' ? onChange : onBlur\n if (!validate) return\n\n // Use the validationCount for all field instances to\n // track freshness of the validation\n const validationCount = (this.getInfo().validationCount || 0) + 1\n this.getInfo().validationCount = validationCount\n const error = normalizeError(validate(value as never, this as never))\n const errorMapKey = getErrorMapKey(cause)\n if (error && this.state.meta.errorMap[errorMapKey] !== error) {\n this.setMeta((prev) => ({\n ...prev,\n errors: [...prev.errors, error],\n errorMap: {\n ...prev.errorMap,\n [getErrorMapKey(cause)]: error,\n },\n }))\n }\n\n // If a sync error is encountered for the errorMapKey (eg. onChange), cancel any async validation\n if (this.state.meta.errorMap[errorMapKey]) {\n this.cancelValidateAsync()\n }\n }\n\n #leaseValidateAsync = () => {\n const count = (this.getInfo().validationAsyncCount || 0) + 1\n this.getInfo().validationAsyncCount = count\n return count\n }\n\n cancelValidateAsync = () => {\n // Lease a new validation count to ignore any pending validations\n this.#leaseValidateAsync()\n // Cancel any pending validation state\n this.setMeta((prev) => ({\n ...prev,\n isValidating: false,\n }))\n }\n\n validateAsync = async (value = this.state.value, cause: ValidationCause) => {\n const {\n onChangeAsync,\n onBlurAsync,\n onSubmitAsync,\n asyncDebounceMs,\n onBlurAsyncDebounceMs,\n onChangeAsyncDebounceMs,\n } = this.options\n\n const validate =\n cause === 'change'\n ? onChangeAsync\n : cause === 'submit'\n ? onSubmitAsync\n : onBlurAsync\n if (!validate) return []\n const debounceMs =\n cause === 'submit'\n ? 0\n : (cause === 'change'\n ? onChangeAsyncDebounceMs\n : onBlurAsyncDebounceMs) ??\n asyncDebounceMs ??\n 0\n\n if (this.state.meta.isValidating !== true)\n this.setMeta((prev) => ({ ...prev, isValidating: true }))\n\n // Use the validationCount for all field instances to\n // track freshness of the validation\n const validationAsyncCount = this.#leaseValidateAsync()\n\n const checkLatest = () =>\n validationAsyncCount === this.getInfo().validationAsyncCount\n\n if (!this.getInfo().validationPromise) {\n this.getInfo().validationPromise = new Promise((resolve, reject) => {\n this.getInfo().validationResolve = resolve\n this.getInfo().validationReject = reject\n })\n }\n\n if (debounceMs > 0) {\n await new Promise((r) => setTimeout(r, debounceMs))\n }\n\n // Only kick off validation if this validation is the latest attempt\n if (checkLatest()) {\n const prevErrors = this.getMeta().errors\n try {\n const rawError = await validate(value as never, this as never)\n if (checkLatest()) {\n const error = normalizeError(rawError)\n this.setMeta((prev) => ({\n ...prev,\n isValidating: false,\n errors: [...prev.errors, error],\n errorMap: {\n ...prev.errorMap,\n [getErrorMapKey(cause)]: error,\n },\n }))\n this.getInfo().validationResolve?.([...prevErrors, error])\n }\n } catch (error) {\n if (checkLatest()) {\n this.getInfo().validationReject?.([...prevErrors, error])\n throw error\n }\n } finally {\n if (checkLatest()) {\n this.setMeta((prev) => ({ ...prev, isValidating: false }))\n delete this.getInfo().validationPromise\n }\n }\n }\n\n // Always return the latest validation promise to the caller\n return this.getInfo().validationPromise ?? []\n }\n\n validate = (\n cause: ValidationCause,\n value?: TData,\n ): ValidationError[] | Promise<ValidationError[]> => {\n // If the field is pristine and validatePristine is false, do not validate\n if (!this.state.meta.isTouched) return []\n // Attempt to sync validate first\n this.validateSync(value, cause)\n\n const errorMapKey = getErrorMapKey(cause)\n // If there is an error mapped to the errorMapKey (eg. onChange, onBlur, onSubmit), return the errors array, do not attempt async validation\n if (this.getMeta().errorMap[errorMapKey]) {\n if (!this.options.asyncAlways) {\n return this.state.meta.errors\n }\n }\n // No error? Attempt async validation\n return this.validateAsync(value, cause)\n }\n\n handleChange = (updater: Updater<TData>) => {\n this.setValue(updater, { touch: true })\n }\n\n handleBlur = () => {\n const prevTouched = this.state.meta.isTouched\n if (!prevTouched) {\n this.setMeta((prev) => ({ ...prev, isTouched: true }))\n this.validate('change')\n }\n this.validate('blur')\n }\n}\n\nfunction normalizeError(rawError?: ValidationError) {\n if (rawError) {\n if (typeof rawError !== 'string') {\n return 'Invalid Form Values'\n }\n\n return rawError\n }\n\n return undefined\n}\n\nfunction getErrorMapKey(cause: ValidationCause) {\n switch (cause) {\n case 'submit':\n return 'onSubmit'\n case 'change':\n return 'onChange'\n case 'blur':\n return 'onBlur'\n case 'mount':\n return 'onMount'\n }\n}\n"],"mappings":"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","names":[]}
|