@tanstack/form-core 0.1.3 → 0.2.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (107) hide show
  1. package/build/legacy/FieldApi.cjs +256 -0
  2. package/build/legacy/FieldApi.cjs.map +1 -0
  3. package/build/legacy/FieldApi.d.cts +3 -0
  4. package/build/legacy/FieldApi.d.ts +3 -0
  5. package/build/legacy/FieldApi.js +223 -0
  6. package/build/legacy/FieldApi.js.map +1 -0
  7. package/build/legacy/FormApi.cjs +255 -0
  8. package/build/legacy/FormApi.cjs.map +1 -0
  9. package/build/legacy/FormApi.d.cts +3 -0
  10. package/build/legacy/FormApi.d.ts +3 -0
  11. package/build/legacy/FormApi.js +232 -0
  12. package/build/legacy/FormApi.js.map +1 -0
  13. package/build/legacy/chunk-4QZDOMDG.js +19 -0
  14. package/build/legacy/chunk-4QZDOMDG.js.map +1 -0
  15. package/build/legacy/index.cjs +29 -0
  16. package/build/legacy/index.cjs.map +1 -0
  17. package/build/legacy/index.d.cts +171 -0
  18. package/build/legacy/index.d.ts +171 -0
  19. package/build/legacy/index.js +5 -0
  20. package/build/legacy/index.js.map +1 -0
  21. package/build/legacy/utils.cjs +98 -0
  22. package/build/legacy/utils.cjs.map +1 -0
  23. package/build/legacy/utils.d.cts +32 -0
  24. package/build/legacy/utils.d.ts +32 -0
  25. package/build/legacy/utils.js +73 -0
  26. package/build/legacy/utils.js.map +1 -0
  27. package/build/modern/FieldApi.cjs +238 -0
  28. package/build/modern/FieldApi.cjs.map +1 -0
  29. package/build/modern/FieldApi.d.cts +3 -0
  30. package/build/modern/FieldApi.d.ts +3 -0
  31. package/build/modern/FieldApi.js +213 -0
  32. package/build/modern/FieldApi.js.map +1 -0
  33. package/build/modern/FormApi.cjs +249 -0
  34. package/build/modern/FormApi.cjs.map +1 -0
  35. package/build/modern/FormApi.d.cts +3 -0
  36. package/build/modern/FormApi.d.ts +3 -0
  37. package/build/modern/FormApi.js +224 -0
  38. package/build/modern/FormApi.js.map +1 -0
  39. package/build/modern/index.cjs +29 -0
  40. package/build/modern/index.cjs.map +1 -0
  41. package/build/modern/index.d.cts +171 -0
  42. package/build/modern/index.d.ts +171 -0
  43. package/build/modern/index.js +5 -0
  44. package/build/modern/index.js.map +1 -0
  45. package/build/modern/utils.cjs +98 -0
  46. package/build/modern/utils.cjs.map +1 -0
  47. package/build/modern/utils.d.cts +32 -0
  48. package/build/modern/utils.d.ts +32 -0
  49. package/build/modern/utils.js +71 -0
  50. package/build/modern/utils.js.map +1 -0
  51. package/package.json +23 -12
  52. package/build/lib/FieldApi.cjs +0 -293
  53. package/build/lib/FieldApi.cjs.map +0 -1
  54. package/build/lib/FieldApi.d.ts +0 -95
  55. package/build/lib/FieldApi.d.ts.map +0 -1
  56. package/build/lib/FieldApi.js +0 -291
  57. package/build/lib/FieldApi.js.map +0 -1
  58. package/build/lib/FieldApi.legacy.cjs +0 -293
  59. package/build/lib/FieldApi.legacy.cjs.map +0 -1
  60. package/build/lib/FieldApi.legacy.js +0 -291
  61. package/build/lib/FieldApi.legacy.js.map +0 -1
  62. package/build/lib/FormApi.cjs +0 -239
  63. package/build/lib/FormApi.cjs.map +0 -1
  64. package/build/lib/FormApi.d.ts +0 -78
  65. package/build/lib/FormApi.d.ts.map +0 -1
  66. package/build/lib/FormApi.js +0 -237
  67. package/build/lib/FormApi.js.map +0 -1
  68. package/build/lib/FormApi.legacy.cjs +0 -239
  69. package/build/lib/FormApi.legacy.cjs.map +0 -1
  70. package/build/lib/FormApi.legacy.js +0 -237
  71. package/build/lib/FormApi.legacy.js.map +0 -1
  72. package/build/lib/_virtual/_rollupPluginBabelHelpers.cjs +0 -65
  73. package/build/lib/_virtual/_rollupPluginBabelHelpers.cjs.map +0 -1
  74. package/build/lib/_virtual/_rollupPluginBabelHelpers.js +0 -56
  75. package/build/lib/_virtual/_rollupPluginBabelHelpers.js.map +0 -1
  76. package/build/lib/_virtual/_rollupPluginBabelHelpers.legacy.cjs +0 -65
  77. package/build/lib/_virtual/_rollupPluginBabelHelpers.legacy.cjs.map +0 -1
  78. package/build/lib/_virtual/_rollupPluginBabelHelpers.legacy.js +0 -56
  79. package/build/lib/_virtual/_rollupPluginBabelHelpers.legacy.js.map +0 -1
  80. package/build/lib/index.cjs +0 -14
  81. package/build/lib/index.cjs.map +0 -1
  82. package/build/lib/index.d.ts +0 -4
  83. package/build/lib/index.d.ts.map +0 -1
  84. package/build/lib/index.js +0 -4
  85. package/build/lib/index.js.map +0 -1
  86. package/build/lib/index.legacy.cjs +0 -14
  87. package/build/lib/index.legacy.cjs.map +0 -1
  88. package/build/lib/index.legacy.js +0 -4
  89. package/build/lib/index.legacy.js.map +0 -1
  90. package/build/lib/tests/FieldApi.spec.d.ts +0 -2
  91. package/build/lib/tests/FieldApi.spec.d.ts.map +0 -1
  92. package/build/lib/tests/FieldApi.test-d.d.ts +0 -2
  93. package/build/lib/tests/FieldApi.test-d.d.ts.map +0 -1
  94. package/build/lib/tests/FormApi.spec.d.ts +0 -2
  95. package/build/lib/tests/FormApi.spec.d.ts.map +0 -1
  96. package/build/lib/tests/utils.d.ts +0 -2
  97. package/build/lib/tests/utils.d.ts.map +0 -1
  98. package/build/lib/utils.cjs +0 -81
  99. package/build/lib/utils.cjs.map +0 -1
  100. package/build/lib/utils.d.ts +0 -32
  101. package/build/lib/utils.d.ts.map +0 -1
  102. package/build/lib/utils.js +0 -77
  103. package/build/lib/utils.js.map +0 -1
  104. package/build/lib/utils.legacy.cjs +0 -81
  105. package/build/lib/utils.legacy.cjs.map +0 -1
  106. package/build/lib/utils.legacy.js +0 -77
  107. package/build/lib/utils.legacy.js.map +0 -1
@@ -0,0 +1,238 @@
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
+ ...this.options.defaultMeta
82
+ };
83
+ this.setMeta = (updater) => this.form.setFieldMeta(this.name, updater);
84
+ this.getInfo = () => this.form.getFieldInfo(this.name);
85
+ this.pushValue = (value) => this.form.pushFieldValue(this.name, value);
86
+ this.insertValue = (index, value) => this.form.insertFieldValue(this.name, index, value);
87
+ this.removeValue = (index) => this.form.removeFieldValue(this.name, index);
88
+ this.swapValues = (aIndex, bIndex) => this.form.swapFieldValues(this.name, aIndex, bIndex);
89
+ this.getSubField = (name) => new _FieldApi({
90
+ name: `${this.name}.${name}`,
91
+ form: this.form
92
+ });
93
+ this.validateSync = (value = this.state.value, cause) => {
94
+ const { onChange, onBlur } = this.options;
95
+ const validate = cause === "submit" ? void 0 : cause === "change" ? onChange : onBlur;
96
+ if (!validate)
97
+ return;
98
+ const validationCount = (this.getInfo().validationCount || 0) + 1;
99
+ this.getInfo().validationCount = validationCount;
100
+ const error = normalizeError(validate(value, this));
101
+ if (this.state.meta.error !== error) {
102
+ this.setMeta((prev) => ({
103
+ ...prev,
104
+ error
105
+ }));
106
+ }
107
+ if (this.state.meta.error) {
108
+ this.cancelValidateAsync();
109
+ }
110
+ };
111
+ this.#leaseValidateAsync = () => {
112
+ const count = (this.getInfo().validationAsyncCount || 0) + 1;
113
+ this.getInfo().validationAsyncCount = count;
114
+ return count;
115
+ };
116
+ this.cancelValidateAsync = () => {
117
+ this.#leaseValidateAsync();
118
+ this.setMeta((prev) => ({
119
+ ...prev,
120
+ isValidating: false
121
+ }));
122
+ };
123
+ this.validateAsync = async (value = this.state.value, cause) => {
124
+ const {
125
+ onChangeAsync,
126
+ onBlurAsync,
127
+ onSubmitAsync,
128
+ asyncDebounceMs,
129
+ onBlurAsyncDebounceMs,
130
+ onChangeAsyncDebounceMs
131
+ } = this.options;
132
+ const validate = cause === "change" ? onChangeAsync : cause === "submit" ? onSubmitAsync : onBlurAsync;
133
+ if (!validate)
134
+ return;
135
+ const debounceMs = cause === "submit" ? 0 : (cause === "change" ? onChangeAsyncDebounceMs : onBlurAsyncDebounceMs) ?? asyncDebounceMs ?? 0;
136
+ if (this.state.meta.isValidating !== true)
137
+ this.setMeta((prev) => ({ ...prev, isValidating: true }));
138
+ const validationAsyncCount = this.#leaseValidateAsync();
139
+ const checkLatest = () => validationAsyncCount === this.getInfo().validationAsyncCount;
140
+ if (!this.getInfo().validationPromise) {
141
+ this.getInfo().validationPromise = new Promise((resolve, reject) => {
142
+ this.getInfo().validationResolve = resolve;
143
+ this.getInfo().validationReject = reject;
144
+ });
145
+ }
146
+ if (debounceMs > 0) {
147
+ await new Promise((r) => setTimeout(r, debounceMs));
148
+ }
149
+ if (checkLatest()) {
150
+ try {
151
+ const rawError = await validate(value, this);
152
+ if (checkLatest()) {
153
+ const error = normalizeError(rawError);
154
+ this.setMeta((prev) => ({
155
+ ...prev,
156
+ isValidating: false,
157
+ error
158
+ }));
159
+ this.getInfo().validationResolve?.(error);
160
+ }
161
+ } catch (error) {
162
+ if (checkLatest()) {
163
+ this.getInfo().validationReject?.(error);
164
+ throw error;
165
+ }
166
+ } finally {
167
+ if (checkLatest()) {
168
+ this.setMeta((prev) => ({ ...prev, isValidating: false }));
169
+ delete this.getInfo().validationPromise;
170
+ }
171
+ }
172
+ }
173
+ return this.getInfo().validationPromise;
174
+ };
175
+ this.validate = (cause, value) => {
176
+ if (!this.state.meta.isTouched)
177
+ return;
178
+ this.validateSync(value, cause);
179
+ if (this.state.meta.error) {
180
+ if (!this.options.asyncAlways) {
181
+ return this.state.meta.error;
182
+ }
183
+ }
184
+ return this.validateAsync(value, cause);
185
+ };
186
+ this.handleChange = (updater) => {
187
+ this.setValue(updater, { touch: true });
188
+ };
189
+ this.handleBlur = () => {
190
+ const prevTouched = this.state.meta.isTouched;
191
+ if (!prevTouched) {
192
+ this.setMeta((prev) => ({ ...prev, isTouched: true }));
193
+ this.validate("change");
194
+ }
195
+ this.validate("blur");
196
+ };
197
+ this.form = opts.form;
198
+ this.uid = uid++;
199
+ this.name = opts.name;
200
+ this.store = new import_store.Store(
201
+ {
202
+ value: this.getValue(),
203
+ // eslint-disable-next-line @typescript-eslint/no-unnecessary-condition
204
+ meta: this._getMeta() ?? {
205
+ isValidating: false,
206
+ isTouched: false,
207
+ ...opts.defaultMeta
208
+ }
209
+ },
210
+ {
211
+ onUpdate: () => {
212
+ const state = this.store.state;
213
+ state.meta.touchedError = state.meta.isTouched ? state.meta.error : void 0;
214
+ this.prevState = state;
215
+ this.state = state;
216
+ }
217
+ }
218
+ );
219
+ this.state = this.store.state;
220
+ this.prevState = this.state;
221
+ this.options = opts;
222
+ }
223
+ #leaseValidateAsync;
224
+ };
225
+ function normalizeError(rawError) {
226
+ if (rawError) {
227
+ if (typeof rawError !== "string") {
228
+ return "Invalid Form Values";
229
+ }
230
+ return rawError;
231
+ }
232
+ return void 0;
233
+ }
234
+ // Annotate the CommonJS export names for ESM import in node:
235
+ 0 && (module.exports = {
236
+ FieldApi
237
+ });
238
+ //# sourceMappingURL=FieldApi.cjs.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../../src/FieldApi.ts"],"sourcesContent":["import type { DeepKeys, DeepValue, Updater } from './utils'\nimport type { FormApi, ValidationError } from './FormApi'\nimport { Store } from '@tanstack/store'\n\nexport type ValidationCause = 'change' | 'blur' | 'submit'\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 type FieldApiOptions<TData, TFormData> = FieldOptions<\n TData,\n TFormData\n> & {\n form: FormApi<TFormData>\n}\n\nexport type FieldMeta = {\n isTouched: boolean\n touchedError?: ValidationError\n error?: ValidationError\n isValidating: boolean\n}\n\nlet uid = 0\n\nexport type FieldState<TData> = {\n value: TData\n meta: FieldMeta\n}\n\n/**\n * TData may not be known at the time of FieldApi construction, so we need to\n * use a conditional type to determine if TData is known or not.\n *\n * If TData is not known, we use the TFormData type to determine the type of\n * the field value based on the field name.\n */\ntype GetTData<Name, TData, TFormData> = unknown extends TData\n ? DeepValue<TFormData, Name>\n : TData\n\nexport class FieldApi<TData, TFormData> {\n uid: number\n form: FormApi<TFormData>\n name!: DeepKeys<TFormData>\n /**\n * This is a hack that allows us to use `GetTData` without calling it everywhere\n *\n * Unfortunately this hack appears to be needed alongside the `TName` hack\n * further up in this file. This properly types all of the internal methods,\n * while the `TName` hack types the options properly\n */\n _tdata!: GetTData<typeof this.name, TData, TFormData>\n store!: Store<FieldState<typeof this._tdata>>\n state!: FieldState<typeof this._tdata>\n prevState!: FieldState<typeof this._tdata>\n options: FieldOptions<typeof this._tdata, TFormData> = {} as any\n\n constructor(opts: FieldApiOptions<TData, TFormData>) {\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<typeof this._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 ...opts.defaultMeta,\n },\n },\n {\n onUpdate: () => {\n const state = this.store.state\n\n state.meta.touchedError = state.meta.isTouched\n ? state.meta.error\n : undefined\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\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<typeof this._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 = (): typeof this._tdata => {\n return this.form.getFieldValue(this.name)\n }\n\n setValue = (\n updater: Updater<typeof this._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 ...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 = (\n value: typeof this._tdata extends any[]\n ? (typeof this._tdata)[number]\n : never,\n ) => this.form.pushFieldValue(this.name, value as any)\n\n insertValue = (\n index: number,\n value: typeof this._tdata extends any[]\n ? (typeof this._tdata)[number]\n : 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<typeof this._tdata>>(name: TName) =>\n new FieldApi<DeepValue<typeof this._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\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\n if (this.state.meta.error !== error) {\n this.setMeta((prev) => ({\n ...prev,\n error,\n }))\n }\n\n // If a sync error is encountered, cancel any async validation\n if (this.state.meta.error) {\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\n if (!validate) return\n\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 try {\n const rawError = await validate(value as never, this as never)\n\n if (checkLatest()) {\n const error = normalizeError(rawError)\n this.setMeta((prev) => ({\n ...prev,\n isValidating: false,\n error,\n }))\n this.getInfo().validationResolve?.(error)\n }\n } catch (error) {\n if (checkLatest()) {\n this.getInfo().validationReject?.(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?: typeof this._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\n // Attempt to sync validate first\n this.validateSync(value, cause)\n\n // If there is an error, return it, do not attempt async validation\n if (this.state.meta.error) {\n if (!this.options.asyncAlways) {\n return this.state.meta.error\n }\n }\n\n // No error? Attempt async validation\n return this.validateAsync(value, cause)\n }\n\n handleChange = (updater: Updater<typeof this._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"],"mappings":";;;;;;;;;;;;;;;;;;;;AAAA;AAAA;AAAA;AAAA;AAAA;AAEA,mBAAsB;AAyDtB,IAAI,MAAM;AAkBH,IAAM,WAAN,MAAM,UAA2B;AAAA,EAiBtC,YAAY,MAAyC;AAFrD,mBAAuD,CAAC;AA0CxD,iBAAQ,MAAM;AACZ,YAAM,OAAO,KAAK,QAAQ;AAC1B,WAAK,UAAU,KAAK,GAAG,IAAI;AAE3B,YAAM,cAAc,KAAK,KAAK,MAAM,UAAU,MAAM;AAClD,aAAK,MAAM,MAAM,MAAM;AACrB,gBAAM,YAAY,KAAK,SAAS;AAChC,gBAAM,WAAW,KAAK,QAAQ;AAE9B,cAAI,cAAc,KAAK,MAAM,OAAO;AAClC,iBAAK,MAAM,SAAS,CAAC,UAAU,EAAE,GAAG,MAAM,OAAO,UAAU,EAAE;AAAA,UAC/D;AAEA,cAAI,aAAa,KAAK,MAAM,MAAM;AAChC,iBAAK,MAAM,SAAS,CAAC,UAAU,EAAE,GAAG,MAAM,MAAM,SAAS,EAAE;AAAA,UAC7D;AAAA,QACF,CAAC;AAAA,MACH,CAAC;AAED,WAAK,OAAO,KAAK,OAAgB;AACjC,WAAK,QAAQ,UAAU,IAAa;AAEpC,aAAO,MAAM;AACX,oBAAY;AACZ,eAAO,KAAK,UAAU,KAAK,GAAG;AAC9B,YAAI,CAAC,OAAO,KAAK,KAAK,SAAS,EAAE,QAAQ;AACvC,iBAAO,KAAK,KAAK,UAAU,KAAK,IAAI;AAAA,QACtC;AAAA,MACF;AAAA,IACF;AAEA,kBAAS,CAAC,SAAyD;AAGjE,UAAI,KAAK,MAAM,UAAU,QAAW;AAClC,cAAM,cACJ,KAAK,KAAK,QAAQ,gBAAgB,KAAK,IAAuB;AAEhE,YAAI,KAAK,iBAAiB,QAAW;AACnC,eAAK,SAAS,KAAK,YAAqB;AAAA,QAC1C,WAAW,gBAAgB,QAAW;AACpC,eAAK,SAAS,WAAoB;AAAA,QACpC;AAAA,MACF;AAGA,UAAI,KAAK,SAAS,MAAM,QAAW;AACjC,aAAK,QAAQ,KAAK,MAAM,IAAI;AAAA,MAC9B;AAEA,WAAK,UAAU;AAAA,IACjB;AAEA,oBAAW,MAA0B;AACnC,aAAO,KAAK,KAAK,cAAc,KAAK,IAAI;AAAA,IAC1C;AAEA,oBAAW,CACT,SACA,YACG;AACH,WAAK,KAAK,cAAc,KAAK,MAAM,SAAkB,OAAO;AAC5D,WAAK,SAAS,UAAU,KAAK,MAAM,KAAK;AAAA,IAC1C;AAEA,oBAAW,MAAM,KAAK,KAAK,aAAa,KAAK,IAAI;AACjD,mBAAU,MACR,KAAK,SAAS,KACb;AAAA,MACC,cAAc;AAAA,MACd,WAAW;AAAA,MACX,GAAG,KAAK,QAAQ;AAAA,IAClB;AAEF,mBAAU,CAAC,YACT,KAAK,KAAK,aAAa,KAAK,MAAM,OAAO;AAE3C,mBAAU,MAAM,KAAK,KAAK,aAAa,KAAK,IAAI;AAEhD,qBAAY,CACV,UAGG,KAAK,KAAK,eAAe,KAAK,MAAM,KAAY;AAErD,uBAAc,CACZ,OACA,UAGG,KAAK,KAAK,iBAAiB,KAAK,MAAM,OAAO,KAAY;AAE9D,uBAAc,CAAC,UAAkB,KAAK,KAAK,iBAAiB,KAAK,MAAM,KAAK;AAE5E,sBAAa,CAAC,QAAgB,WAC5B,KAAK,KAAK,gBAAgB,KAAK,MAAM,QAAQ,MAAM;AAErD,uBAAc,CAA6C,SACzD,IAAI,UAA0D;AAAA,MAC5D,MAAM,GAAG,KAAK,IAAI,IAAI,IAAI;AAAA,MAC1B,MAAM,KAAK;AAAA,IACb,CAAC;AAEH,wBAAe,CAAC,QAAQ,KAAK,MAAM,OAAO,UAA2B;AACnE,YAAM,EAAE,UAAU,OAAO,IAAI,KAAK;AAClC,YAAM,WACJ,UAAU,WAAW,SAAY,UAAU,WAAW,WAAW;AAEnE,UAAI,CAAC;AAAU;AAIf,YAAM,mBAAmB,KAAK,QAAQ,EAAE,mBAAmB,KAAK;AAChE,WAAK,QAAQ,EAAE,kBAAkB;AACjC,YAAM,QAAQ,eAAe,SAAS,OAAgB,IAAa,CAAC;AAEpE,UAAI,KAAK,MAAM,KAAK,UAAU,OAAO;AACnC,aAAK,QAAQ,CAAC,UAAU;AAAA,UACtB,GAAG;AAAA,UACH;AAAA,QACF,EAAE;AAAA,MACJ;AAGA,UAAI,KAAK,MAAM,KAAK,OAAO;AACzB,aAAK,oBAAoB;AAAA,MAC3B;AAAA,IACF;AAEA,+BAAsB,MAAM;AAC1B,YAAM,SAAS,KAAK,QAAQ,EAAE,wBAAwB,KAAK;AAC3D,WAAK,QAAQ,EAAE,uBAAuB;AACtC,aAAO;AAAA,IACT;AAEA,+BAAsB,MAAM;AAE1B,WAAK,oBAAoB;AAEzB,WAAK,QAAQ,CAAC,UAAU;AAAA,QACtB,GAAG;AAAA,QACH,cAAc;AAAA,MAChB,EAAE;AAAA,IACJ;AAEA,yBAAgB,OAAO,QAAQ,KAAK,MAAM,OAAO,UAA2B;AAC1E,YAAM;AAAA,QACJ;AAAA,QACA;AAAA,QACA;AAAA,QACA;AAAA,QACA;AAAA,QACA;AAAA,MACF,IAAI,KAAK;AAET,YAAM,WACJ,UAAU,WACN,gBACA,UAAU,WACV,gBACA;AAEN,UAAI,CAAC;AAAU;AAEf,YAAM,aACJ,UAAU,WACN,KACC,UAAU,WACP,0BACA,0BACJ,mBACA;AAEN,UAAI,KAAK,MAAM,KAAK,iBAAiB;AACnC,aAAK,QAAQ,CAAC,UAAU,EAAE,GAAG,MAAM,cAAc,KAAK,EAAE;AAI1D,YAAM,uBAAuB,KAAK,oBAAoB;AAEtD,YAAM,cAAc,MAClB,yBAAyB,KAAK,QAAQ,EAAE;AAE1C,UAAI,CAAC,KAAK,QAAQ,EAAE,mBAAmB;AACrC,aAAK,QAAQ,EAAE,oBAAoB,IAAI,QAAQ,CAAC,SAAS,WAAW;AAClE,eAAK,QAAQ,EAAE,oBAAoB;AACnC,eAAK,QAAQ,EAAE,mBAAmB;AAAA,QACpC,CAAC;AAAA,MACH;AAEA,UAAI,aAAa,GAAG;AAClB,cAAM,IAAI,QAAQ,CAAC,MAAM,WAAW,GAAG,UAAU,CAAC;AAAA,MACpD;AAGA,UAAI,YAAY,GAAG;AACjB,YAAI;AACF,gBAAM,WAAW,MAAM,SAAS,OAAgB,IAAa;AAE7D,cAAI,YAAY,GAAG;AACjB,kBAAM,QAAQ,eAAe,QAAQ;AACrC,iBAAK,QAAQ,CAAC,UAAU;AAAA,cACtB,GAAG;AAAA,cACH,cAAc;AAAA,cACd;AAAA,YACF,EAAE;AACF,iBAAK,QAAQ,EAAE,oBAAoB,KAAK;AAAA,UAC1C;AAAA,QACF,SAAS,OAAO;AACd,cAAI,YAAY,GAAG;AACjB,iBAAK,QAAQ,EAAE,mBAAmB,KAAK;AACvC,kBAAM;AAAA,UACR;AAAA,QACF,UAAE;AACA,cAAI,YAAY,GAAG;AACjB,iBAAK,QAAQ,CAAC,UAAU,EAAE,GAAG,MAAM,cAAc,MAAM,EAAE;AACzD,mBAAO,KAAK,QAAQ,EAAE;AAAA,UACxB;AAAA,QACF;AAAA,MACF;AAGA,aAAO,KAAK,QAAQ,EAAE;AAAA,IACxB;AAEA,oBAAW,CACT,OACA,UAC+C;AAE/C,UAAI,CAAC,KAAK,MAAM,KAAK;AAAW;AAGhC,WAAK,aAAa,OAAO,KAAK;AAG9B,UAAI,KAAK,MAAM,KAAK,OAAO;AACzB,YAAI,CAAC,KAAK,QAAQ,aAAa;AAC7B,iBAAO,KAAK,MAAM,KAAK;AAAA,QACzB;AAAA,MACF;AAGA,aAAO,KAAK,cAAc,OAAO,KAAK;AAAA,IACxC;AAEA,wBAAe,CAAC,YAAyC;AACvD,WAAK,SAAS,SAAS,EAAE,OAAO,KAAK,CAAC;AAAA,IACxC;AAEA,sBAAa,MAAM;AACjB,YAAM,cAAc,KAAK,MAAM,KAAK;AACpC,UAAI,CAAC,aAAa;AAChB,aAAK,QAAQ,CAAC,UAAU,EAAE,GAAG,MAAM,WAAW,KAAK,EAAE;AACrD,aAAK,SAAS,QAAQ;AAAA,MACxB;AACA,WAAK,SAAS,MAAM;AAAA,IACtB;AAxSE,SAAK,OAAO,KAAK;AACjB,SAAK,MAAM;AAOX,SAAK,OAAO,KAAK;AAEjB,SAAK,QAAQ,IAAI;AAAA,MACf;AAAA,QACE,OAAO,KAAK,SAAS;AAAA;AAAA,QAErB,MAAM,KAAK,SAAS,KAAK;AAAA,UACvB,cAAc;AAAA,UACd,WAAW;AAAA,UACX,GAAG,KAAK;AAAA,QACV;AAAA,MACF;AAAA,MACA;AAAA,QACE,UAAU,MAAM;AACd,gBAAM,QAAQ,KAAK,MAAM;AAEzB,gBAAM,KAAK,eAAe,MAAM,KAAK,YACjC,MAAM,KAAK,QACX;AAEJ,eAAK,YAAY;AACjB,eAAK,QAAQ;AAAA,QACf;AAAA,MACF;AAAA,IACF;AAEA,SAAK,QAAQ,KAAK,MAAM;AACxB,SAAK,YAAY,KAAK;AACtB,SAAK,UAAU;AAAA,EACjB;AAAA,EAmIA;AAiIF;AAEA,SAAS,eAAe,UAA4B;AAClD,MAAI,UAAU;AACZ,QAAI,OAAO,aAAa,UAAU;AAChC,aAAO;AAAA,IACT;AAEA,WAAO;AAAA,EACT;AAEA,SAAO;AACT;","names":[]}
@@ -0,0 +1,3 @@
1
+ export { FieldApi, FieldApiOptions, FieldMeta, FieldOptions, FieldState, ValidationCause } from './index.cjs';
2
+ import './utils.cjs';
3
+ import '@tanstack/store';
@@ -0,0 +1,3 @@
1
+ export { FieldApi, FieldApiOptions, FieldMeta, FieldOptions, FieldState, ValidationCause } from './index.js';
2
+ import './utils.js';
3
+ import '@tanstack/store';
@@ -0,0 +1,213 @@
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
+ ...this.options.defaultMeta
58
+ };
59
+ this.setMeta = (updater) => this.form.setFieldMeta(this.name, updater);
60
+ this.getInfo = () => this.form.getFieldInfo(this.name);
61
+ this.pushValue = (value) => this.form.pushFieldValue(this.name, value);
62
+ this.insertValue = (index, value) => this.form.insertFieldValue(this.name, index, value);
63
+ this.removeValue = (index) => this.form.removeFieldValue(this.name, index);
64
+ this.swapValues = (aIndex, bIndex) => this.form.swapFieldValues(this.name, aIndex, bIndex);
65
+ this.getSubField = (name) => new _FieldApi({
66
+ name: `${this.name}.${name}`,
67
+ form: this.form
68
+ });
69
+ this.validateSync = (value = this.state.value, cause) => {
70
+ const { onChange, onBlur } = this.options;
71
+ const validate = cause === "submit" ? void 0 : cause === "change" ? onChange : onBlur;
72
+ if (!validate)
73
+ return;
74
+ const validationCount = (this.getInfo().validationCount || 0) + 1;
75
+ this.getInfo().validationCount = validationCount;
76
+ const error = normalizeError(validate(value, this));
77
+ if (this.state.meta.error !== error) {
78
+ this.setMeta((prev) => ({
79
+ ...prev,
80
+ error
81
+ }));
82
+ }
83
+ if (this.state.meta.error) {
84
+ this.cancelValidateAsync();
85
+ }
86
+ };
87
+ this.#leaseValidateAsync = () => {
88
+ const count = (this.getInfo().validationAsyncCount || 0) + 1;
89
+ this.getInfo().validationAsyncCount = count;
90
+ return count;
91
+ };
92
+ this.cancelValidateAsync = () => {
93
+ this.#leaseValidateAsync();
94
+ this.setMeta((prev) => ({
95
+ ...prev,
96
+ isValidating: false
97
+ }));
98
+ };
99
+ this.validateAsync = async (value = this.state.value, cause) => {
100
+ const {
101
+ onChangeAsync,
102
+ onBlurAsync,
103
+ onSubmitAsync,
104
+ asyncDebounceMs,
105
+ onBlurAsyncDebounceMs,
106
+ onChangeAsyncDebounceMs
107
+ } = this.options;
108
+ const validate = cause === "change" ? onChangeAsync : cause === "submit" ? onSubmitAsync : onBlurAsync;
109
+ if (!validate)
110
+ return;
111
+ const debounceMs = cause === "submit" ? 0 : (cause === "change" ? onChangeAsyncDebounceMs : onBlurAsyncDebounceMs) ?? asyncDebounceMs ?? 0;
112
+ if (this.state.meta.isValidating !== true)
113
+ this.setMeta((prev) => ({ ...prev, isValidating: true }));
114
+ const validationAsyncCount = this.#leaseValidateAsync();
115
+ const checkLatest = () => validationAsyncCount === this.getInfo().validationAsyncCount;
116
+ if (!this.getInfo().validationPromise) {
117
+ this.getInfo().validationPromise = new Promise((resolve, reject) => {
118
+ this.getInfo().validationResolve = resolve;
119
+ this.getInfo().validationReject = reject;
120
+ });
121
+ }
122
+ if (debounceMs > 0) {
123
+ await new Promise((r) => setTimeout(r, debounceMs));
124
+ }
125
+ if (checkLatest()) {
126
+ try {
127
+ const rawError = await validate(value, this);
128
+ if (checkLatest()) {
129
+ const error = normalizeError(rawError);
130
+ this.setMeta((prev) => ({
131
+ ...prev,
132
+ isValidating: false,
133
+ error
134
+ }));
135
+ this.getInfo().validationResolve?.(error);
136
+ }
137
+ } catch (error) {
138
+ if (checkLatest()) {
139
+ this.getInfo().validationReject?.(error);
140
+ throw error;
141
+ }
142
+ } finally {
143
+ if (checkLatest()) {
144
+ this.setMeta((prev) => ({ ...prev, isValidating: false }));
145
+ delete this.getInfo().validationPromise;
146
+ }
147
+ }
148
+ }
149
+ return this.getInfo().validationPromise;
150
+ };
151
+ this.validate = (cause, value) => {
152
+ if (!this.state.meta.isTouched)
153
+ return;
154
+ this.validateSync(value, cause);
155
+ if (this.state.meta.error) {
156
+ if (!this.options.asyncAlways) {
157
+ return this.state.meta.error;
158
+ }
159
+ }
160
+ return this.validateAsync(value, cause);
161
+ };
162
+ this.handleChange = (updater) => {
163
+ this.setValue(updater, { touch: true });
164
+ };
165
+ this.handleBlur = () => {
166
+ const prevTouched = this.state.meta.isTouched;
167
+ if (!prevTouched) {
168
+ this.setMeta((prev) => ({ ...prev, isTouched: true }));
169
+ this.validate("change");
170
+ }
171
+ this.validate("blur");
172
+ };
173
+ this.form = opts.form;
174
+ this.uid = uid++;
175
+ this.name = opts.name;
176
+ this.store = new Store(
177
+ {
178
+ value: this.getValue(),
179
+ // eslint-disable-next-line @typescript-eslint/no-unnecessary-condition
180
+ meta: this._getMeta() ?? {
181
+ isValidating: false,
182
+ isTouched: false,
183
+ ...opts.defaultMeta
184
+ }
185
+ },
186
+ {
187
+ onUpdate: () => {
188
+ const state = this.store.state;
189
+ state.meta.touchedError = state.meta.isTouched ? state.meta.error : void 0;
190
+ this.prevState = state;
191
+ this.state = state;
192
+ }
193
+ }
194
+ );
195
+ this.state = this.store.state;
196
+ this.prevState = this.state;
197
+ this.options = opts;
198
+ }
199
+ #leaseValidateAsync;
200
+ };
201
+ function normalizeError(rawError) {
202
+ if (rawError) {
203
+ if (typeof rawError !== "string") {
204
+ return "Invalid Form Values";
205
+ }
206
+ return rawError;
207
+ }
208
+ return void 0;
209
+ }
210
+ export {
211
+ FieldApi
212
+ };
213
+ //# sourceMappingURL=FieldApi.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../../src/FieldApi.ts"],"sourcesContent":["import type { DeepKeys, DeepValue, Updater } from './utils'\nimport type { FormApi, ValidationError } from './FormApi'\nimport { Store } from '@tanstack/store'\n\nexport type ValidationCause = 'change' | 'blur' | 'submit'\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 type FieldApiOptions<TData, TFormData> = FieldOptions<\n TData,\n TFormData\n> & {\n form: FormApi<TFormData>\n}\n\nexport type FieldMeta = {\n isTouched: boolean\n touchedError?: ValidationError\n error?: ValidationError\n isValidating: boolean\n}\n\nlet uid = 0\n\nexport type FieldState<TData> = {\n value: TData\n meta: FieldMeta\n}\n\n/**\n * TData may not be known at the time of FieldApi construction, so we need to\n * use a conditional type to determine if TData is known or not.\n *\n * If TData is not known, we use the TFormData type to determine the type of\n * the field value based on the field name.\n */\ntype GetTData<Name, TData, TFormData> = unknown extends TData\n ? DeepValue<TFormData, Name>\n : TData\n\nexport class FieldApi<TData, TFormData> {\n uid: number\n form: FormApi<TFormData>\n name!: DeepKeys<TFormData>\n /**\n * This is a hack that allows us to use `GetTData` without calling it everywhere\n *\n * Unfortunately this hack appears to be needed alongside the `TName` hack\n * further up in this file. This properly types all of the internal methods,\n * while the `TName` hack types the options properly\n */\n _tdata!: GetTData<typeof this.name, TData, TFormData>\n store!: Store<FieldState<typeof this._tdata>>\n state!: FieldState<typeof this._tdata>\n prevState!: FieldState<typeof this._tdata>\n options: FieldOptions<typeof this._tdata, TFormData> = {} as any\n\n constructor(opts: FieldApiOptions<TData, TFormData>) {\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<typeof this._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 ...opts.defaultMeta,\n },\n },\n {\n onUpdate: () => {\n const state = this.store.state\n\n state.meta.touchedError = state.meta.isTouched\n ? state.meta.error\n : undefined\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\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<typeof this._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 = (): typeof this._tdata => {\n return this.form.getFieldValue(this.name)\n }\n\n setValue = (\n updater: Updater<typeof this._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 ...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 = (\n value: typeof this._tdata extends any[]\n ? (typeof this._tdata)[number]\n : never,\n ) => this.form.pushFieldValue(this.name, value as any)\n\n insertValue = (\n index: number,\n value: typeof this._tdata extends any[]\n ? (typeof this._tdata)[number]\n : 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<typeof this._tdata>>(name: TName) =>\n new FieldApi<DeepValue<typeof this._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\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\n if (this.state.meta.error !== error) {\n this.setMeta((prev) => ({\n ...prev,\n error,\n }))\n }\n\n // If a sync error is encountered, cancel any async validation\n if (this.state.meta.error) {\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\n if (!validate) return\n\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 try {\n const rawError = await validate(value as never, this as never)\n\n if (checkLatest()) {\n const error = normalizeError(rawError)\n this.setMeta((prev) => ({\n ...prev,\n isValidating: false,\n error,\n }))\n this.getInfo().validationResolve?.(error)\n }\n } catch (error) {\n if (checkLatest()) {\n this.getInfo().validationReject?.(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?: typeof this._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\n // Attempt to sync validate first\n this.validateSync(value, cause)\n\n // If there is an error, return it, do not attempt async validation\n if (this.state.meta.error) {\n if (!this.options.asyncAlways) {\n return this.state.meta.error\n }\n }\n\n // No error? Attempt async validation\n return this.validateAsync(value, cause)\n }\n\n handleChange = (updater: Updater<typeof this._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"],"mappings":"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","names":[]}