@reformer/core 1.1.0-beta.7 → 1.1.0-beta.8

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (80) hide show
  1. package/dist/behaviors-BRaiR-UY.js +528 -0
  2. package/dist/behaviors.d.ts +6 -2
  3. package/dist/behaviors.js +18 -227
  4. package/dist/index.d.ts +2 -0
  5. package/dist/index.js +3380 -10
  6. package/dist/validators-DjXtDVoE.js +455 -0
  7. package/dist/validators.d.ts +6 -2
  8. package/dist/validators.js +29 -281
  9. package/package.json +1 -1
  10. package/dist/core/behavior/behavior-applicator.js +0 -92
  11. package/dist/core/behavior/behavior-context.js +0 -43
  12. package/dist/core/behavior/behavior-registry.js +0 -198
  13. package/dist/core/behavior/behaviors/compute-from.js +0 -84
  14. package/dist/core/behavior/behaviors/copy-from.js +0 -64
  15. package/dist/core/behavior/behaviors/enable-when.js +0 -81
  16. package/dist/core/behavior/behaviors/index.js +0 -11
  17. package/dist/core/behavior/behaviors/reset-when.js +0 -63
  18. package/dist/core/behavior/behaviors/revalidate-when.js +0 -51
  19. package/dist/core/behavior/behaviors/sync-fields.js +0 -66
  20. package/dist/core/behavior/behaviors/transform-value.js +0 -110
  21. package/dist/core/behavior/behaviors/watch-field.js +0 -56
  22. package/dist/core/behavior/compose-behavior.js +0 -166
  23. package/dist/core/behavior/create-field-path.js +0 -69
  24. package/dist/core/behavior/index.js +0 -17
  25. package/dist/core/behavior/types.js +0 -7
  26. package/dist/core/context/form-context-impl.js +0 -37
  27. package/dist/core/factories/index.js +0 -6
  28. package/dist/core/factories/node-factory.js +0 -281
  29. package/dist/core/nodes/array-node.js +0 -534
  30. package/dist/core/nodes/field-node.js +0 -510
  31. package/dist/core/nodes/form-node.js +0 -343
  32. package/dist/core/nodes/group-node/field-registry.js +0 -215
  33. package/dist/core/nodes/group-node/index.js +0 -11
  34. package/dist/core/nodes/group-node/proxy-builder.js +0 -161
  35. package/dist/core/nodes/group-node/state-manager.js +0 -265
  36. package/dist/core/nodes/group-node.js +0 -770
  37. package/dist/core/types/deep-schema.js +0 -11
  38. package/dist/core/types/field-path.js +0 -4
  39. package/dist/core/types/form-context.js +0 -25
  40. package/dist/core/types/group-node-proxy.js +0 -31
  41. package/dist/core/types/index.js +0 -4
  42. package/dist/core/types/validation-schema.js +0 -10
  43. package/dist/core/utils/create-form.js +0 -24
  44. package/dist/core/utils/debounce.js +0 -197
  45. package/dist/core/utils/error-handler.js +0 -226
  46. package/dist/core/utils/field-path-navigator.js +0 -374
  47. package/dist/core/utils/index.js +0 -14
  48. package/dist/core/utils/registry-helpers.js +0 -79
  49. package/dist/core/utils/registry-stack.js +0 -86
  50. package/dist/core/utils/resources.js +0 -69
  51. package/dist/core/utils/subscription-manager.js +0 -214
  52. package/dist/core/utils/type-guards.js +0 -169
  53. package/dist/core/validation/core/apply-when.js +0 -41
  54. package/dist/core/validation/core/apply.js +0 -38
  55. package/dist/core/validation/core/index.js +0 -8
  56. package/dist/core/validation/core/validate-async.js +0 -45
  57. package/dist/core/validation/core/validate-tree.js +0 -43
  58. package/dist/core/validation/core/validate.js +0 -38
  59. package/dist/core/validation/field-path.js +0 -147
  60. package/dist/core/validation/index.js +0 -33
  61. package/dist/core/validation/validate-form.js +0 -152
  62. package/dist/core/validation/validation-applicator.js +0 -217
  63. package/dist/core/validation/validation-context.js +0 -75
  64. package/dist/core/validation/validation-registry.js +0 -298
  65. package/dist/core/validation/validators/array-validators.js +0 -86
  66. package/dist/core/validation/validators/date.js +0 -117
  67. package/dist/core/validation/validators/email.js +0 -60
  68. package/dist/core/validation/validators/index.js +0 -14
  69. package/dist/core/validation/validators/max-length.js +0 -60
  70. package/dist/core/validation/validators/max.js +0 -60
  71. package/dist/core/validation/validators/min-length.js +0 -60
  72. package/dist/core/validation/validators/min.js +0 -60
  73. package/dist/core/validation/validators/number.js +0 -90
  74. package/dist/core/validation/validators/pattern.js +0 -62
  75. package/dist/core/validation/validators/phone.js +0 -58
  76. package/dist/core/validation/validators/required.js +0 -69
  77. package/dist/core/validation/validators/url.js +0 -55
  78. package/dist/create-field-path-nXfTtl55.js +0 -283
  79. package/dist/hooks/useFormControl.js +0 -298
  80. package/dist/validation-context-cWXmh_Ho.js +0 -156
@@ -1,343 +0,0 @@
1
- /**
2
- * FormNode - абстрактный базовый класс для всех узлов формы
3
- *
4
- * Аналог AbstractControl из Angular Forms
5
- * Унифицирует работу с полями (FieldNode), группами (GroupNode) и массивами (ArrayNode)
6
- *
7
- * Использует Template Method паттерн для управления состоянием:
8
- * - Публичные методы (markAsTouched, disable и т.д.) реализованы в базовом классе
9
- * - Protected hooks (onMarkAsTouched, onDisable и т.д.) переопределяются в наследниках
10
- *
11
- * @group Nodes
12
- */
13
- import { signal, computed } from '@preact/signals-core';
14
- /**
15
- * Абстрактный базовый класс для всех узлов формы
16
- *
17
- * Все узлы (поля, группы, массивы) наследуют от этого класса
18
- * и реализуют единый интерфейс для работы с состоянием и валидацией
19
- *
20
- * Template Method паттерн используется для управления состоянием:
21
- * - Общие signals (_touched, _dirty, _status) определены в базовом классе
22
- * - Публичные методы (markAsTouched, disable и т.д.) реализованы здесь
23
- * - Protected hooks (onMarkAsTouched, onDisable и т.д.) переопределяются в наследниках
24
- *
25
- * @group Nodes
26
- */
27
- export class FormNode {
28
- // ============================================================================
29
- // Protected состояние (для Template Method паттерна)
30
- // ============================================================================
31
- /**
32
- * Пользователь взаимодействовал с узлом (touched)
33
- * Protected: наследники могут читать/изменять через методы
34
- */
35
- _touched = signal(false);
36
- /**
37
- * Значение узла было изменено (dirty)
38
- * Protected: наследники могут читать/изменять через методы
39
- */
40
- _dirty = signal(false);
41
- /**
42
- * Текущий статус узла
43
- * Protected: наследники могут читать/изменять через методы
44
- */
45
- _status = signal('valid');
46
- // ============================================================================
47
- // Публичные computed signals (readonly для внешнего мира)
48
- // ============================================================================
49
- /**
50
- * Пользователь взаимодействовал с узлом (touched)
51
- * Computed из _touched для предоставления readonly интерфейса
52
- */
53
- touched = computed(() => this._touched.value);
54
- /**
55
- * Пользователь не взаимодействовал с узлом (untouched)
56
- */
57
- untouched = computed(() => !this._touched.value);
58
- /**
59
- * Значение узла было изменено (dirty)
60
- * Computed из _dirty для предоставления readonly интерфейса
61
- */
62
- dirty = computed(() => this._dirty.value);
63
- /**
64
- * Значение узла не было изменено (pristine)
65
- */
66
- pristine = computed(() => !this._dirty.value);
67
- /**
68
- * Текущий статус узла
69
- * Computed из _status для предоставления readonly интерфейса
70
- */
71
- status = computed(() => this._status.value);
72
- /**
73
- * Узел отключен (disabled)
74
- */
75
- disabled = computed(() => this._status.value === 'disabled');
76
- /**
77
- * Узел включен (enabled)
78
- */
79
- enabled = computed(() => this._status.value !== 'disabled');
80
- /**
81
- * Получить ошибки валидации с фильтрацией
82
- *
83
- * Позволяет фильтровать ошибки по различным критериям:
84
- * - По коду ошибки
85
- * - По сообщению (частичное совпадение)
86
- * - По параметрам
87
- * - Через кастомный предикат
88
- *
89
- * Без параметров возвращает все ошибки (эквивалент errors.value)
90
- *
91
- * @param options - Опции фильтрации ошибок
92
- * @returns Отфильтрованный массив ошибок валидации
93
- *
94
- * @example
95
- * ```typescript
96
- * // Все ошибки
97
- * const allErrors = form.getErrors();
98
- *
99
- * // Ошибки с конкретным кодом
100
- * const requiredErrors = form.getErrors({ code: 'required' });
101
- *
102
- * // Ошибки с несколькими кодами
103
- * const errors = form.getErrors({ code: ['required', 'email'] });
104
- *
105
- * // Ошибки по сообщению
106
- * const passwordErrors = form.getErrors({ message: 'Password' });
107
- *
108
- * // Ошибки по параметрам
109
- * const minLengthErrors = form.getErrors({
110
- * params: { minLength: 8 }
111
- * });
112
- *
113
- * // Кастомная фильтрация
114
- * const customErrors = form.getErrors({
115
- * predicate: (err) => err.code.startsWith('custom_')
116
- * });
117
- * ```
118
- */
119
- getErrors(options) {
120
- const allErrors = this.errors.value;
121
- // Без фильтрации - вернуть все ошибки
122
- if (!options) {
123
- return allErrors;
124
- }
125
- return allErrors.filter((error) => {
126
- // Фильтр по коду
127
- if (options.code !== undefined) {
128
- const codes = Array.isArray(options.code) ? options.code : [options.code];
129
- if (!codes.includes(error.code)) {
130
- return false;
131
- }
132
- }
133
- // Фильтр по сообщению (частичное совпадение, регистронезависимый)
134
- if (options.message !== undefined) {
135
- if (!error.message.toLowerCase().includes(options.message.toLowerCase())) {
136
- return false;
137
- }
138
- }
139
- // Фильтр по параметрам
140
- if (options.params !== undefined) {
141
- if (!error.params) {
142
- return false;
143
- }
144
- // Проверяем, что все ключи из options.params присутствуют в error.params
145
- // и имеют те же значения
146
- for (const [key, value] of Object.entries(options.params)) {
147
- if (error.params[key] !== value) {
148
- return false;
149
- }
150
- }
151
- }
152
- // Кастомный предикат
153
- if (options.predicate !== undefined) {
154
- if (!options.predicate(error)) {
155
- return false;
156
- }
157
- }
158
- return true;
159
- });
160
- }
161
- // ============================================================================
162
- // Методы управления состоянием (Template Method)
163
- // ============================================================================
164
- /**
165
- * Отметить узел как touched (пользователь взаимодействовал)
166
- *
167
- * Template Method: обновляет signal в базовом классе,
168
- * вызывает hook для кастомной логики в наследниках
169
- */
170
- markAsTouched() {
171
- this._touched.value = true;
172
- this.onMarkAsTouched();
173
- }
174
- /**
175
- * Отметить узел как untouched
176
- *
177
- * Template Method: обновляет signal в базовом классе,
178
- * вызывает hook для кастомной логики в наследниках
179
- */
180
- markAsUntouched() {
181
- this._touched.value = false;
182
- this.onMarkAsUntouched();
183
- }
184
- /**
185
- * Отметить узел как dirty (значение изменено)
186
- *
187
- * Template Method: обновляет signal в базовом классе,
188
- * вызывает hook для кастомной логики в наследниках
189
- */
190
- markAsDirty() {
191
- this._dirty.value = true;
192
- this.onMarkAsDirty();
193
- }
194
- /**
195
- * Отметить узел как pristine (значение не изменено)
196
- *
197
- * Template Method: обновляет signal в базовом классе,
198
- * вызывает hook для кастомной логики в наследниках
199
- */
200
- markAsPristine() {
201
- this._dirty.value = false;
202
- this.onMarkAsPristine();
203
- }
204
- /**
205
- * Пометить все поля (включая вложенные) как touched
206
- * Алиас для markAsTouched(), но более явно показывает намерение
207
- * пометить ВСЕ поля рекурсивно
208
- *
209
- * Полезно для:
210
- * - Показа всех ошибок валидации перед submit
211
- * - Принудительного отображения ошибок при нажатии "Validate All"
212
- * - Отображения невалидных полей в wizard/step form
213
- *
214
- * @example
215
- * ```typescript
216
- * // Показать все ошибки перед submit
217
- * form.touchAll();
218
- * const isValid = await form.validate();
219
- * if (!isValid) {
220
- * // Все ошибки теперь видны пользователю
221
- * }
222
- *
223
- * // Или использовать submit() который уже вызывает touchAll
224
- * await form.submit(async (values) => {
225
- * await api.save(values);
226
- * });
227
- * ```
228
- */
229
- touchAll() {
230
- this.markAsTouched();
231
- }
232
- // ============================================================================
233
- // Методы управления доступностью (Template Method)
234
- // ============================================================================
235
- /**
236
- * Отключить узел
237
- *
238
- * Template Method: обновляет статус в базовом классе,
239
- * вызывает hook для кастомной логики в наследниках
240
- *
241
- * Отключенные узлы не проходят валидацию и не включаются в getValue()
242
- */
243
- disable() {
244
- this._status.value = 'disabled';
245
- this.onDisable();
246
- }
247
- /**
248
- * Включить узел
249
- *
250
- * Template Method: обновляет статус в базовом классе,
251
- * вызывает hook для кастомной логики в наследниках
252
- */
253
- enable() {
254
- this._status.value = 'valid';
255
- this.onEnable();
256
- }
257
- // ============================================================================
258
- // Protected hooks (для переопределения в наследниках)
259
- // ============================================================================
260
- /**
261
- * Hook: вызывается после markAsTouched()
262
- *
263
- * Переопределите в наследниках для дополнительной логики:
264
- * - GroupNode: пометить все дочерние узлы как touched
265
- * - ArrayNode: пометить все элементы массива как touched
266
- * - FieldNode: пустая реализация (нет дочерних узлов)
267
- *
268
- * @example
269
- * ```typescript
270
- * // GroupNode
271
- * protected onMarkAsTouched(): void {
272
- * this.fields.forEach(field => field.markAsTouched());
273
- * }
274
- * ```
275
- */
276
- onMarkAsTouched() {
277
- // Пустая реализация по умолчанию
278
- // Наследники переопределяют при необходимости
279
- }
280
- /**
281
- * Hook: вызывается после markAsUntouched()
282
- *
283
- * Переопределите в наследниках для дополнительной логики:
284
- * - GroupNode: пометить все дочерние узлы как untouched
285
- * - ArrayNode: пометить все элементы массива как untouched
286
- * - FieldNode: пустая реализация (нет дочерних узлов)
287
- */
288
- onMarkAsUntouched() {
289
- // Пустая реализация по умолчанию
290
- }
291
- /**
292
- * Hook: вызывается после markAsDirty()
293
- *
294
- * Переопределите в наследниках для дополнительной логики:
295
- * - GroupNode: может обновить родительскую форму
296
- * - ArrayNode: может обновить родительскую форму
297
- * - FieldNode: пустая реализация
298
- */
299
- onMarkAsDirty() {
300
- // Пустая реализация по умолчанию
301
- }
302
- /**
303
- * Hook: вызывается после markAsPristine()
304
- *
305
- * Переопределите в наследниках для дополнительной логики:
306
- * - GroupNode: пометить все дочерние узлы как pristine
307
- * - ArrayNode: пометить все элементы массива как pristine
308
- * - FieldNode: пустая реализация
309
- */
310
- onMarkAsPristine() {
311
- // Пустая реализация по умолчанию
312
- }
313
- /**
314
- * Hook: вызывается после disable()
315
- *
316
- * Переопределите в наследниках для дополнительной логики:
317
- * - GroupNode: отключить все дочерние узлы
318
- * - ArrayNode: отключить все элементы массива
319
- * - FieldNode: очистить ошибки валидации
320
- *
321
- * @example
322
- * ```typescript
323
- * // GroupNode
324
- * protected onDisable(): void {
325
- * this.fields.forEach(field => field.disable());
326
- * }
327
- * ```
328
- */
329
- onDisable() {
330
- // Пустая реализация по умолчанию
331
- }
332
- /**
333
- * Hook: вызывается после enable()
334
- *
335
- * Переопределите в наследниках для дополнительной логики:
336
- * - GroupNode: включить все дочерние узлы
337
- * - ArrayNode: включить все элементы массива
338
- * - FieldNode: пустая реализация
339
- */
340
- onEnable() {
341
- // Пустая реализация по умолчанию
342
- }
343
- }
@@ -1,215 +0,0 @@
1
- /**
2
- * FieldRegistry - управление полями в GroupNode
3
- *
4
- * Извлечено из GroupNode для соблюдения SRP (Single Responsibility Principle).
5
- * Отвечает только за хранение и управление коллекцией полей формы.
6
- *
7
- * @template T Тип формы (объект)
8
- *
9
- * @example
10
- * ```typescript
11
- * const registry = new FieldRegistry<{ email: string; name: string }>();
12
- * registry.set('email', emailField);
13
- * registry.set('name', nameField);
14
- *
15
- * const emailField = registry.get('email');
16
- * registry.forEach((field, key) => {
17
- * console.log(key, field.value.value);
18
- * });
19
- * ```
20
- */
21
- /**
22
- * Реестр полей формы
23
- *
24
- * Предоставляет типобезопасный доступ к полям формы
25
- * через Map-подобный интерфейс
26
- *
27
- * @template T Тип формы (объект)
28
- */
29
- export class FieldRegistry {
30
- /**
31
- * Внутреннее хранилище полей
32
- * Map обеспечивает быструю lookup производительность O(1)
33
- */
34
- // eslint-disable-next-line @typescript-eslint/no-explicit-any
35
- fields = new Map();
36
- /**
37
- * Установить поле в реестр
38
- *
39
- * @param key - Ключ поля (имя свойства в типе T)
40
- * @param node - FormNode для этого поля
41
- *
42
- * @example
43
- * ```typescript
44
- * registry.set('email', new FieldNode({ value: '' }));
45
- * ```
46
- */
47
- set(key, node) {
48
- this.fields.set(key, node);
49
- }
50
- /**
51
- * Получить поле из реестра
52
- *
53
- * @param key - Ключ поля
54
- * @returns FormNode или undefined, если поле не найдено
55
- *
56
- * @example
57
- * ```typescript
58
- * const emailField = registry.get('email');
59
- * if (emailField) {
60
- * console.log(emailField.value.value);
61
- * }
62
- * ```
63
- */
64
- get(key) {
65
- // eslint-disable-next-line @typescript-eslint/no-explicit-any
66
- return this.fields.get(key);
67
- }
68
- /**
69
- * Проверить наличие поля в реестре
70
- *
71
- * @param key - Ключ поля
72
- * @returns true если поле существует
73
- *
74
- * @example
75
- * ```typescript
76
- * if (registry.has('email')) {
77
- * console.log('Email field exists');
78
- * }
79
- * ```
80
- */
81
- has(key) {
82
- return this.fields.has(key);
83
- }
84
- /**
85
- * Удалить поле из реестра
86
- *
87
- * @param key - Ключ поля
88
- * @returns true если поле было удалено, false если поля не было
89
- *
90
- * @example
91
- * ```typescript
92
- * registry.delete('email');
93
- * ```
94
- */
95
- delete(key) {
96
- return this.fields.delete(key);
97
- }
98
- /**
99
- * Перебрать все поля
100
- *
101
- * @param callback - Функция обратного вызова для каждого поля
102
- *
103
- * @example
104
- * ```typescript
105
- * registry.forEach((field, key) => {
106
- * console.log(`${key}: ${field.value.value}`);
107
- * });
108
- * ```
109
- */
110
- forEach(callback) {
111
- this.fields.forEach(callback);
112
- }
113
- /**
114
- * Получить итератор значений (полей)
115
- *
116
- * @returns Итератор по всем полям
117
- *
118
- * @example
119
- * ```typescript
120
- * for (const field of registry.values()) {
121
- * await field.validate();
122
- * }
123
- * ```
124
- */
125
- values() {
126
- return this.fields.values();
127
- }
128
- /**
129
- * Получить итератор пар [ключ, значение]
130
- *
131
- * @returns Итератор по всем записям
132
- *
133
- * @example
134
- * ```typescript
135
- * for (const [key, field] of registry.entries()) {
136
- * console.log(key, field.value.value);
137
- * }
138
- * ```
139
- */
140
- entries() {
141
- return this.fields.entries();
142
- }
143
- /**
144
- * Получить итератор ключей полей
145
- *
146
- * @returns Итератор по всем ключам
147
- *
148
- * @example
149
- * ```typescript
150
- * const fieldNames = Array.from(registry.keys());
151
- * // ['email', 'name', 'age']
152
- * ```
153
- */
154
- keys() {
155
- return this.fields.keys();
156
- }
157
- /**
158
- * Получить количество полей
159
- *
160
- * @returns Количество зарегистрированных полей
161
- *
162
- * @example
163
- * ```typescript
164
- * console.log(`Form has ${registry.size()} fields`);
165
- * ```
166
- */
167
- size() {
168
- return this.fields.size;
169
- }
170
- /**
171
- * Очистить все поля
172
- *
173
- * Удаляет все поля из реестра
174
- *
175
- * @example
176
- * ```typescript
177
- * registry.clear();
178
- * console.log(registry.size()); // 0
179
- * ```
180
- */
181
- clear() {
182
- this.fields.clear();
183
- }
184
- /**
185
- * Получить все поля как массив
186
- *
187
- * Полезно для операций, требующих работу с массивом
188
- *
189
- * @returns Массив всех полей
190
- *
191
- * @example
192
- * ```typescript
193
- * const allValid = registry.toArray().every(field => field.valid.value);
194
- * ```
195
- */
196
- toArray() {
197
- return Array.from(this.fields.values());
198
- }
199
- /**
200
- * Получить Map-представление реестра (readonly)
201
- *
202
- * Используйте для совместимости с существующим кодом
203
- *
204
- * @returns ReadonlyMap с полями
205
- * @internal
206
- *
207
- * @example
208
- * ```typescript
209
- * const mapView = registry.asMap();
210
- * ```
211
- */
212
- asMap() {
213
- return this.fields;
214
- }
215
- }
@@ -1,11 +0,0 @@
1
- /**
2
- * GroupNode modules
3
- *
4
- * Модульная структура для GroupNode:
5
- * - FieldRegistry: Управление коллекцией полей
6
- * - ProxyBuilder: Создание Proxy для типобезопасного доступа к полям
7
- * - StateManager: Управление состоянием формы (signals)
8
- */
9
- export { FieldRegistry } from './field-registry';
10
- export { ProxyBuilder } from './proxy-builder';
11
- export { StateManager } from './state-manager';