@decaf-ts/for-angular 0.0.16 → 0.0.18

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 (65) hide show
  1. package/assets/i18n/en.json +9 -69
  2. package/assets/i18n/pt.json +80 -0
  3. package/assets/icons/icon-128.webp +0 -0
  4. package/assets/icons/icon-192.webp +0 -0
  5. package/assets/icons/icon-256.webp +0 -0
  6. package/assets/icons/icon-48.webp +0 -0
  7. package/assets/icons/icon-512.webp +0 -0
  8. package/assets/icons/icon-72.webp +0 -0
  9. package/assets/icons/icon-96.webp +0 -0
  10. package/assets/images/apple-touch-icon.png +0 -0
  11. package/assets/images/favicon.png +0 -0
  12. package/assets/images/favicon.svg +29 -0
  13. package/components/component-renderer/component-renderer.component.d.ts +5 -4
  14. package/components/crud-field/crud-field.component.d.ts +186 -22
  15. package/components/crud-form/crud-form.component.d.ts +194 -8
  16. package/components/empty-state/empty-state.component.d.ts +9 -10
  17. package/components/fieldset/fieldset.component.d.ts +383 -36
  18. package/components/filter/filter.component.d.ts +11 -2
  19. package/components/list/list.component.d.ts +1 -1
  20. package/components/list-item/list-item.component.d.ts +2 -2
  21. package/components/model-renderer/model-renderer.component.d.ts +1 -5
  22. package/directives/collapsable.directive.d.ts +1 -0
  23. package/engine/NgxBaseComponent.d.ts +43 -43
  24. package/engine/NgxCrudFormField.d.ts +7 -3
  25. package/engine/NgxFormService.d.ts +113 -12
  26. package/engine/NgxRenderingEngine.d.ts +178 -25
  27. package/engine/constants.d.ts +11 -6
  28. package/engine/decorators.d.ts +2 -2
  29. package/engine/index.d.ts +4 -2
  30. package/engine/interfaces.d.ts +271 -0
  31. package/engine/types.d.ts +11 -206
  32. package/esm2022/components/component-renderer/component-renderer.component.mjs +13 -11
  33. package/esm2022/components/crud-field/crud-field.component.mjs +213 -8
  34. package/esm2022/components/crud-form/crud-form.component.mjs +133 -13
  35. package/esm2022/components/empty-state/empty-state.component.mjs +13 -12
  36. package/esm2022/components/fieldset/fieldset.component.mjs +485 -43
  37. package/esm2022/components/filter/filter.component.mjs +16 -6
  38. package/esm2022/components/layout/layout.component.mjs +3 -3
  39. package/esm2022/components/list/list.component.mjs +4 -5
  40. package/esm2022/components/list-item/list-item.component.mjs +10 -10
  41. package/esm2022/components/model-renderer/model-renderer.component.mjs +9 -8
  42. package/esm2022/components/pagination/pagination.component.mjs +7 -7
  43. package/esm2022/components/searchbar/searchbar.component.mjs +3 -3
  44. package/esm2022/directives/collapsable.directive.mjs +3 -2
  45. package/esm2022/engine/NgxBaseComponent.mjs +64 -63
  46. package/esm2022/engine/NgxCrudFormField.mjs +14 -4
  47. package/esm2022/engine/NgxFormService.mjs +239 -27
  48. package/esm2022/engine/NgxRenderingEngine.mjs +218 -46
  49. package/esm2022/engine/ValidatorFactory.mjs +6 -4
  50. package/esm2022/engine/constants.mjs +14 -9
  51. package/esm2022/engine/decorators.mjs +6 -6
  52. package/esm2022/engine/index.mjs +5 -3
  53. package/esm2022/engine/interfaces.mjs +4 -0
  54. package/esm2022/engine/types.mjs +1 -3
  55. package/esm2022/helpers/utils.mjs +53 -32
  56. package/esm2022/i18n/Loader.mjs +82 -0
  57. package/fesm2022/decaf-ts-for-angular.mjs +3030 -2097
  58. package/fesm2022/decaf-ts-for-angular.mjs.map +1 -1
  59. package/helpers/utils.d.ts +42 -16
  60. package/i18n/Loader.d.ts +48 -0
  61. package/package.json +11 -1
  62. package/engine/NgxRenderingEngine2.d.ts +0 -250
  63. package/esm2022/engine/NgxRenderingEngine2.mjs +0 -332
  64. package/esm2022/interfaces.mjs +0 -2
  65. package/interfaces.d.ts +0 -28
@@ -1,7 +1,10 @@
1
1
  import { escapeHtml, HTML5CheckTypes, HTML5InputTypes, parseToNumber } from '@decaf-ts/ui-decorators';
2
- import { FormControl, FormGroup, Validators } from '@angular/forms';
3
- import { isValidDate, parseDate, Validation } from '@decaf-ts/decorator-validation';
2
+ import { AbstractControl, FormArray, FormControl, FormGroup, Validators } from '@angular/forms';
3
+ import { isValidDate, ModelKeys, parseDate, Primitives, Validation } from '@decaf-ts/decorator-validation';
4
4
  import { ValidatorFactory } from './ValidatorFactory';
5
+ import { cleanSpaces } from '../helpers';
6
+ import { OperationKeys } from '@decaf-ts/db-decorators';
7
+ import { AngularEngineKeys, BaseComponentProps } from '../engine/constants';
5
8
  /**
6
9
  * @description Service for managing Angular forms and form controls.
7
10
  * @summary The NgxFormService provides utility methods for creating, managing, and validating Angular forms and form controls. It includes functionality for registering forms, adding controls, validating fields, and handling form data.
@@ -41,7 +44,27 @@ import { ValidatorFactory } from './ValidatorFactory';
41
44
  * NFS-->>C: Return form data
42
45
  */
43
46
  export class NgxFormService {
47
+ /**
48
+ * @description WeakMap that stores control properties for form controls.
49
+ * @summary A WeakMap that associates AbstractControl instances with their corresponding FieldProperties.
50
+ * This allows the service to track metadata for form controls without creating memory leaks.
51
+ *
52
+ * @type {WeakMap<AbstractControl, FieldProperties>}
53
+ * @private
54
+ * @static
55
+ * @memberOf NgxFormService
56
+ */
44
57
  static { this.controls = new WeakMap(); }
58
+ /**
59
+ * @description Registry of form groups indexed by their unique identifiers.
60
+ * @summary A Map that stores FormGroup instances with their unique string identifiers.
61
+ * This allows global access to registered forms throughout the application.
62
+ *
63
+ * @type {Map<string, FormGroup>}
64
+ * @private
65
+ * @static
66
+ * @memberOf NgxFormService
67
+ */
45
68
  static { this.formRegistry = new Map(); }
46
69
  /**
47
70
  * @description Adds a form to the registry.
@@ -70,28 +93,177 @@ export class NgxFormService {
70
93
  * @param {string} path - The path to the control.
71
94
  * @return {FormParentGroup} A tuple containing the parent FormGroup and the control name.
72
95
  */
73
- static resolveParentGroup(formGroup, path) {
96
+ static resolveParentGroup(formGroup, path, componentProps, parentProps) {
97
+ const isMultiple = parentProps?.['multiple'] || parentProps?.['type'] === 'Array' || false;
74
98
  const parts = path.split('.');
75
99
  const controlName = parts.pop();
100
+ const { childOf } = componentProps;
76
101
  let currentGroup = formGroup;
102
+ function setArrayComponentProps(formGroupArray) {
103
+ const props = formGroupArray[AngularEngineKeys.FORM_GROUP_COMPONENT_PROPS] || {};
104
+ if (!props[ModelKeys.MODEL][controlName])
105
+ props[ModelKeys.MODEL] = Object.assign({}, props[ModelKeys.MODEL], { [controlName]: { ...componentProps } });
106
+ }
77
107
  for (const part of parts) {
78
108
  if (!currentGroup.get(part)) {
79
- currentGroup.addControl(part, new FormGroup({}));
109
+ const partFormGroup = (isMultiple && part === childOf) ? new FormArray([new FormGroup({})]) : new FormGroup({});
110
+ partFormGroup[AngularEngineKeys.FORM_GROUP_COMPONENT_PROPS] = {
111
+ childOf: childOf || '',
112
+ isMultiple: isMultiple,
113
+ name: part,
114
+ pk: componentProps?.['pk'] || parentProps?.['pk'] || '',
115
+ [ModelKeys.MODEL]: {},
116
+ };
117
+ if (currentGroup instanceof FormArray) {
118
+ currentGroup.push(partFormGroup);
119
+ }
120
+ else {
121
+ for (const control of Object.values(partFormGroup.controls)) {
122
+ if (control instanceof FormControl)
123
+ this.register(control, componentProps);
124
+ }
125
+ if (partFormGroup instanceof AbstractControl)
126
+ this.register(partFormGroup, componentProps);
127
+ currentGroup.addControl(part, partFormGroup);
128
+ }
80
129
  }
130
+ if (childOf && currentGroup instanceof FormArray)
131
+ setArrayComponentProps(currentGroup);
81
132
  currentGroup = currentGroup.get(part);
82
133
  }
83
134
  return [currentGroup, controlName];
84
135
  }
85
136
  /**
86
- * @description Adds a form control to a form group.
87
- * @summary Creates and adds a form control to the specified form group based on the provided component properties.
88
- * @param {FormGroup} formGroup - The form group to add the control to.
89
- * @param {ComponentInput} componentProps - The properties of the component to create the control from.
137
+ * @description Retrieves component properties from a FormGroup or FormArray.
138
+ * @summary Extracts component properties stored in the form group metadata. If a FormGroup is provided
139
+ * and groupArrayName is specified, it will look for the FormArray within the form structure.
140
+ *
141
+ * @param {FormGroup | FormArray} formGroup - The form group or form array to extract properties from
142
+ * @param {string} [key] - Optional key to retrieve a specific property
143
+ * @param {string} [groupArrayName] - Optional name of the group array if formGroup is not a FormArray
144
+ * @return {Partial<FieldProperties>} The component properties or a specific property if key is provided
145
+ *
146
+ * @static
147
+ * @memberOf NgxFormService
148
+ */
149
+ static getComponentPropsFromGroupArray(formGroup, key, groupArrayName) {
150
+ if (!(formGroup instanceof FormArray) && typeof groupArrayName === Primitives.STRING)
151
+ formGroup = formGroup.root.get(groupArrayName) || {};
152
+ const props = formGroup?.[AngularEngineKeys.FORM_GROUP_COMPONENT_PROPS] || {};
153
+ return (!key ? props : props?.[key]) || {};
154
+ }
155
+ /**
156
+ * @description Adds a new group to a parent FormArray.
157
+ * @summary Creates and adds a new FormGroup to the specified parent FormArray based on the
158
+ * component properties stored in the parent's metadata. This is used for dynamic form arrays
159
+ * where new groups need to be added at runtime.
160
+ *
161
+ * @param {FormGroup} formGroup - The root form group containing the parent FormArray
162
+ * @param {string} parentName - The name of the parent FormArray to add the group to
163
+ * @param {number} [index=1] - The index position where the new group should be added
164
+ * @return {FormGroup} The newly created and added FormGroup
165
+ *
166
+ * @static
167
+ * @memberOf NgxFormService
168
+ */
169
+ static addGroupToParent(formGroup, parentName, index = 1) {
170
+ const componentProps = this.getComponentPropsFromGroupArray(formGroup, ModelKeys.MODEL, parentName);
171
+ Object.entries(componentProps).forEach(([, value]) => {
172
+ return this.addFormControl(formGroup, value, { multiple: true }, index);
173
+ });
174
+ return this.getGroupFromParent(formGroup, parentName, index);
175
+ }
176
+ /**
177
+ * @description Retrieves a FormGroup from a parent FormArray at the specified index.
178
+ * @summary Gets a FormGroup from the specified parent FormArray. If the group doesn't exist
179
+ * at the given index, it will create a new one using addGroupToParent.
180
+ *
181
+ * @param {FormGroup} formGroup - The root form group containing the parent FormArray
182
+ * @param {string} parentName - The name of the parent FormArray to retrieve the group from
183
+ * @param {number} [index=1] - The index of the group to retrieve
184
+ * @return {FormGroup} The FormGroup at the specified index
185
+ *
186
+ * @static
187
+ * @memberOf NgxFormService
188
+ */
189
+ static getGroupFromParent(formGroup, parentName, index = 1) {
190
+ const childGroup = (formGroup.get(parentName) || formGroup).at(index);
191
+ if (childGroup instanceof FormGroup)
192
+ return childGroup;
193
+ return this.addGroupToParent(formGroup, parentName, index);
194
+ }
195
+ /**
196
+ * @description Checks if a value is unique within a FormArray group.
197
+ * @summary Validates that the primary key value in a FormGroup is unique among all groups
198
+ * in the parent FormArray. The uniqueness check behavior differs based on the operation type.
199
+ *
200
+ * @param {FormGroup} formGroup - The FormGroup to check for uniqueness
201
+ * @param {number} index - The index of the current group within the FormArray
202
+ * @param {OperationKeys} [operation=OperationKeys.CREATE] - The type of operation being performed
203
+ * @return {boolean} True if the value is unique, false otherwise
204
+ *
205
+ * @static
206
+ * @memberOf NgxFormService
207
+ */
208
+ static isUniqueOnGroup(formGroup, index, operation = OperationKeys.CREATE) {
209
+ const formGroupArray = formGroup.parent;
210
+ const pk = this.getComponentPropsFromGroupArray(formGroupArray, BaseComponentProps.PK);
211
+ const controlName = Object.keys(formGroup.controls)[0];
212
+ // only check for unique if is the pk control
213
+ if (controlName !== pk)
214
+ return true;
215
+ const controlValue = cleanSpaces(`${formGroup.get(pk)?.value}`, true);
216
+ if (operation === OperationKeys.CREATE)
217
+ return !formGroupArray.controls.some((group, i) => i !== index && cleanSpaces(`${group.get(pk)?.value}`, true) === controlValue);
218
+ return !formGroupArray.controls.some((group, i) => i !== index && controlValue === cleanSpaces(`${group.get(pk)?.value}`, true));
219
+ }
220
+ /**
221
+ * @description Enables all controls within a FormGroup or FormArray.
222
+ * @summary Recursively enables all form controls within the provided FormGroup or FormArray.
223
+ * This is useful for making all controls interactive after they have been disabled.
224
+ *
225
+ * @param {FormArray | FormGroup} formGroup - The FormGroup or FormArray to enable all controls for
226
+ * @return {void}
227
+ *
228
+ * @static
229
+ * @memberOf NgxFormService
90
230
  */
91
- static addFormControl(formGroup, componentProps) {
92
- const { name, childOf } = componentProps;
93
- const fullPath = childOf ? `${childOf}.${name}` : name;
94
- const [parentGroup, controlName] = this.resolveParentGroup(formGroup, fullPath);
231
+ static enableAllGroupControls(formGroup) {
232
+ Object.keys(formGroup.controls).forEach(key => {
233
+ const control = formGroup.get(key);
234
+ if (control instanceof FormArray) {
235
+ control.controls.forEach(child => {
236
+ if (child instanceof FormGroup) {
237
+ child.enable({ emitEvent: false });
238
+ child.updateValueAndValidity({ emitEvent: true });
239
+ }
240
+ });
241
+ }
242
+ });
243
+ }
244
+ /**
245
+ * @description Adds a form control to a form group based on component properties.
246
+ * @summary Creates and configures a FormControl within the specified FormGroup using the provided
247
+ * component properties. Handles nested paths, multiple controls (FormArrays), and control registration.
248
+ * This method supports complex form structures with nested groups and arrays.
249
+ *
250
+ * @param {FormGroup} formGroup - The form group to add the control to
251
+ * @param {IComponentInput} componentProps - The component properties defining the control configuration
252
+ * @param {KeyValue} [parentProps={}] - Properties from the parent component for context
253
+ * @param {number} [index=0] - The index for multiple controls in FormArrays
254
+ * @return {void}
255
+ *
256
+ * @private
257
+ * @static
258
+ * @memberOf NgxFormService
259
+ */
260
+ static addFormControl(formGroup, componentProps, parentProps = {}, index = 0) {
261
+ const isMultiple = parentProps?.['multiple'] || parentProps?.['type'] === 'Array' || false;
262
+ const { name, childOf, } = componentProps;
263
+ if (isMultiple)
264
+ componentProps['pk'] = componentProps['pk'] || parentProps?.['pk'] || '';
265
+ const fullPath = childOf ? isMultiple ? `${childOf}.${index}.${name}` : `${childOf}.${name}` : name;
266
+ const [parentGroup, controlName] = this.resolveParentGroup(formGroup, fullPath, componentProps, parentProps);
95
267
  if (!parentGroup.get(controlName)) {
96
268
  const control = NgxFormService.fromProps(componentProps, componentProps.updateMode || 'change');
97
269
  NgxFormService.register(control, componentProps);
@@ -99,6 +271,7 @@ export class NgxFormService {
99
271
  }
100
272
  componentProps['formGroup'] = parentGroup;
101
273
  componentProps['formControl'] = parentGroup.get(controlName);
274
+ componentProps['multiple'] = isMultiple;
102
275
  }
103
276
  /**
104
277
  * @description Retrieves a control from a registered form.
@@ -123,7 +296,7 @@ export class NgxFormService {
123
296
  * @description Creates a form from component configurations.
124
297
  * @summary Generates a FormGroup based on an array of component configurations and optionally registers it.
125
298
  * @param {string} id - The unique identifier for the form.
126
- * @param {ComponentConfig[]} components - An array of component configurations.
299
+ * @param {IComponentConfig[]} components - An array of component configurations.
127
300
  * @param {boolean} [registry=false] - Whether to register the created form.
128
301
  * @return {FormGroup} The created FormGroup.
129
302
  */
@@ -143,12 +316,12 @@ export class NgxFormService {
143
316
  * @param {FieldProperties} componentProperties - The properties of the component to create the control from.
144
317
  * @return {AbstractControl} The form or created control.
145
318
  */
146
- static addControlFromProps(id, componentProperties) {
319
+ static addControlFromProps(id, componentProperties, parentProps) {
147
320
  const form = this.formRegistry.get(id) ?? new FormGroup({});
148
321
  if (!this.formRegistry.has(id))
149
322
  this.addRegistry(id, form);
150
323
  if (componentProperties.path)
151
- this.addFormControl(form, componentProperties);
324
+ this.addFormControl(form, componentProperties, parentProps);
152
325
  return form;
153
326
  }
154
327
  /**
@@ -161,8 +334,20 @@ export class NgxFormService {
161
334
  const data = {};
162
335
  for (const key in formGroup.controls) {
163
336
  const control = formGroup.controls[key];
337
+ const parentProps = NgxFormService.getPropsFromControl(formGroup);
164
338
  if (!(control instanceof FormControl)) {
165
- data[key] = NgxFormService.getFormData(control);
339
+ const value = NgxFormService.getFormData(control);
340
+ const isValid = control.valid;
341
+ if (parentProps.multiple) {
342
+ if (isValid) {
343
+ data[key] = value;
344
+ }
345
+ else {
346
+ this.reset(control);
347
+ }
348
+ continue;
349
+ }
350
+ data[key] = value;
166
351
  continue;
167
352
  }
168
353
  const props = NgxFormService.getPropsFromControl(control);
@@ -182,6 +367,7 @@ export class NgxFormService {
182
367
  }
183
368
  data[key] = value;
184
369
  }
370
+ NgxFormService.enableAllGroupControls(formGroup);
185
371
  return data;
186
372
  }
187
373
  /**
@@ -192,21 +378,44 @@ export class NgxFormService {
192
378
  * @return {boolean} True if all fields are valid, false otherwise.
193
379
  * @throws {Error} If no control is found at the specified path or if the control type is unknown.
194
380
  */
195
- static validateFields(control, path) {
381
+ static validateFields(control, pk, path) {
196
382
  control = path ? control.get(path) : control;
197
383
  if (!control)
198
384
  throw new Error(`No control found at path: ${path || 'root'}.`);
199
- const isAllowed = [FormGroup, FormControl].some(type => control instanceof type);
385
+ const isAllowed = [FormArray, FormGroup, FormControl].some(type => control instanceof type);
200
386
  if (!isAllowed)
201
387
  throw new Error(`Unknown control type at: ${path || 'root'}`);
202
388
  control.markAsTouched();
203
389
  control.markAsDirty();
204
390
  control.updateValueAndValidity({ emitEvent: true });
205
391
  if (control instanceof FormGroup) {
206
- Object.values(control.controls).forEach((childControl) => {
392
+ Object.values(control.controls).forEach(childControl => {
207
393
  this.validateFields(childControl);
208
394
  });
209
395
  }
396
+ if (control instanceof FormArray) {
397
+ const totalGroups = control.length;
398
+ const hasValid = control.controls.some(control => control.valid);
399
+ if (totalGroups > 1 && hasValid) {
400
+ for (let i = control.length - 1; i >= 0; i--) {
401
+ const childControl = control.at(i);
402
+ // disable no valid groups on array
403
+ if (!childControl.valid) {
404
+ childControl.parent.setErrors(null);
405
+ childControl.parent.updateValueAndValidity({ emitEvent: true });
406
+ childControl.disable();
407
+ }
408
+ else {
409
+ this.validateFields(childControl);
410
+ }
411
+ }
412
+ }
413
+ else {
414
+ Object.values(control.controls).forEach(childControl => {
415
+ this.validateFields(childControl);
416
+ });
417
+ }
418
+ }
210
419
  return control.valid;
211
420
  }
212
421
  /**
@@ -296,20 +505,23 @@ export class NgxFormService {
296
505
  * @param {FormGroup} formGroup - The form group to reset.
297
506
  */
298
507
  static reset(formGroup) {
299
- for (const key in formGroup.controls) {
300
- const control = formGroup.controls[key];
301
- if (!(control instanceof FormControl)) {
302
- NgxFormService.reset(control);
303
- continue;
304
- }
508
+ if (formGroup instanceof FormControl) {
509
+ const control = formGroup;
305
510
  const { type } = NgxFormService.getPropsFromControl(control);
306
511
  if (!HTML5CheckTypes.includes(type))
307
- control.setValue(undefined);
512
+ control.setValue("");
308
513
  control.markAsPristine();
309
514
  control.markAsUntouched();
310
515
  control.setErrors(null);
311
516
  control.updateValueAndValidity();
312
517
  }
518
+ else {
519
+ for (const key in formGroup.controls) {
520
+ const control = formGroup.controls[key];
521
+ NgxFormService.reset(control);
522
+ continue;
523
+ }
524
+ }
313
525
  }
314
526
  }
315
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"NgxFormService.js","sourceRoot":"","sources":["../../../../src/lib/engine/NgxFormService.ts"],"names":[],"mappings":"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","sourcesContent":["import { escapeHtml, FieldProperties, HTML5CheckTypes, HTML5InputTypes, parseToNumber } from '@decaf-ts/ui-decorators';\nimport {  ComponentConfig, ComponentInput, FieldUpdateMode, FormParentGroup } from './types';\nimport { AbstractControl, FormControl, FormGroup, ValidatorFn, Validators } from '@angular/forms';\nimport { isValidDate, parseDate, Validation } from '@decaf-ts/decorator-validation';\nimport { ValidatorFactory } from './ValidatorFactory';\n\n\n/**\n * @description Service for managing Angular forms and form controls.\n * @summary The NgxFormService provides utility methods for creating, managing, and validating Angular forms and form controls. It includes functionality for registering forms, adding controls, validating fields, and handling form data.\n *\n * @class\n * @param {WeakMap<AbstractControl, FieldProperties>} controls - A WeakMap to store control properties.\n * @param {Map<string, FormGroup>} formRegistry - A Map to store registered forms.\n *\n * @example\n * // Creating a form from components\n * const components = [\n *   { inputs: { name: 'username', type: 'text', required: true } },\n *   { inputs: { name: 'password', type: 'password', minLength: 8 } }\n * ];\n * const form = NgxFormService.createFormFromComponents('loginForm', components, true);\n *\n * // Validating fields\n * NgxFormService.validateFields(form);\n *\n * // Getting form data\n * const formData = NgxFormService.getFormData(form);\n *\n * @mermaid\n * sequenceDiagram\n *   participant C as Component\n *   participant NFS as NgxFormService\n *   participant AF as Angular Forms\n *   C->>NFS: createFormFromComponents()\n *   NFS->>AF: new FormGroup()\n *   NFS->>NFS: addFormControl()\n *   NFS->>AF: addControl()\n *   NFS-->>C: Return FormGroup\n *   C->>NFS: validateFields()\n *   NFS->>AF: markAsTouched(), markAsDirty(), updateValueAndValidity()\n *   C->>NFS: getFormData()\n *   NFS->>AF: Get control values\n *   NFS-->>C: Return form data\n */\nexport class NgxFormService {\n  private static controls = new WeakMap<AbstractControl, FieldProperties>();\n  private static formRegistry = new Map<string, FormGroup>();\n\n  /**\n   * @description Adds a form to the registry.\n   * @summary Registers a FormGroup with a unique identifier. Throws an error if the identifier is already in use.\n   * @param {string} formId - The unique identifier for the form.\n   * @param {FormGroup} formGroup - The FormGroup to be registered.\n   * @throws {Error} If a FormGroup with the given id is already registered.\n   */\n  static addRegistry(formId: string, formGroup: FormGroup): void {\n    if (this.formRegistry.has(formId))\n      throw new Error(`A FormGroup with id '${formId}' is already registered.`);\n    this.formRegistry.set(formId, formGroup);\n  }\n\n  /**\n   * @description Removes a form from the registry.\n   * @summary Deletes a FormGroup from the registry using its unique identifier.\n   * @param {string} formId - The unique identifier of the form to be removed.\n   */\n  static removeRegistry(formId: string): void {\n    this.formRegistry.delete(formId);\n  }\n\n  /**\n   * @description Resolves the parent group and control name from a path.\n   * @summary Traverses the form group structure to find the parent group and control name for a given path.\n   * @param {FormGroup} formGroup - The root FormGroup.\n   * @param {string} path - The path to the control.\n   * @return {FormParentGroup} A tuple containing the parent FormGroup and the control name.\n   */\n  private static resolveParentGroup(formGroup: FormGroup, path: string): FormParentGroup {\n    const parts = path.split('.');\n    const controlName = parts.pop() as string;\n    let currentGroup = formGroup;\n    for (const part of parts) {\n      if (!currentGroup.get(part)) {\n        currentGroup.addControl(part, new FormGroup({}));\n      }\n      currentGroup = currentGroup.get(part) as FormGroup;\n    }\n    return [currentGroup, controlName];\n  }\n\n  /**\n   * @description Adds a form control to a form group.\n   * @summary Creates and adds a form control to the specified form group based on the provided component properties.\n   * @param {FormGroup} formGroup - The form group to add the control to.\n   * @param {ComponentInput} componentProps - The properties of the component to create the control from.\n   */\n  private static addFormControl(formGroup: FormGroup, componentProps: ComponentInput): void {\n\n    const { name, childOf } = componentProps;\n    const fullPath = childOf ? `${childOf}.${name}` : name;\n    const [parentGroup, controlName] = this.resolveParentGroup(formGroup, fullPath);\n\n    if (!parentGroup.get(controlName)) {\n      const control = NgxFormService.fromProps(\n        componentProps,\n        componentProps.updateMode || 'change',\n      );\n      NgxFormService.register(control, componentProps);\n      parentGroup.addControl(controlName, control);\n    }\n\n    componentProps['formGroup'] = parentGroup;\n    componentProps['formControl'] = parentGroup.get(controlName) as FormControl;\n  }\n\n  /**\n   * @description Retrieves a control from a registered form.\n   * @summary Finds and returns an AbstractControl from a registered form using the form id and optional path.\n   * @param {string} formId - The unique identifier of the form.\n   * @param {string} [path] - The path to the control within the form.\n   * @return {AbstractControl} The requested AbstractControl.\n   * @throws {Error} If the form is not found in the registry or the control is not found in the form.\n   */\n  static getControlFromForm(formId: string, path?: string): AbstractControl {\n    const form = this.formRegistry.get(formId);\n    if (!form)\n      throw new Error(`Form with id '${formId}' not found in the registry.`);\n\n    if (!path)\n      return form;\n\n    const control = form.get(path);\n    if (!control)\n      throw new Error(`Control with path '${path}' not found in form '${formId}'.`);\n    return control;\n  }\n\n  /**\n   * @description Creates a form from component configurations.\n   * @summary Generates a FormGroup based on an array of component configurations and optionally registers it.\n   * @param {string} id - The unique identifier for the form.\n   * @param {ComponentConfig[]} components - An array of component configurations.\n   * @param {boolean} [registry=false] - Whether to register the created form.\n   * @return {FormGroup} The created FormGroup.\n   */\n  static createFormFromComponents(id: string, components: ComponentConfig[], registry: boolean = false): FormGroup {\n    const form = new FormGroup({});\n    components.forEach(component => {\n      this.addFormControl(form, component.inputs);\n    });\n\n    if (registry)\n      this.addRegistry(id, form);\n\n    return form;\n  }\n\n  /**\n   * @description Adds a control to a form based on component properties.\n   * @summary Creates and adds a form control to a form (existing or new) based on the provided component properties.\n   * @param {string} id - The unique identifier of the form.\n   * @param {FieldProperties} componentProperties - The properties of the component to create the control from.\n   * @return {AbstractControl} The form or created control.\n   */\n  static addControlFromProps(id: string, componentProperties: FieldProperties): AbstractControl {\n    const form = this.formRegistry.get(id) ?? new FormGroup({});\n    if (!this.formRegistry.has(id))\n      this.addRegistry(id, form);\n\n    if (componentProperties.path)\n      this.addFormControl(form, componentProperties);\n\n    return form;\n  }\n\n  /**\n   * @description Retrieves form data from a FormGroup.\n   * @summary Extracts and processes the data from a FormGroup, handling different input types and nested form groups.\n   * @param {FormGroup} formGroup - The FormGroup to extract data from.\n   * @return {Record<string, unknown>} An object containing the form data.\n   */\n  static getFormData(formGroup: FormGroup): Record<string, unknown> {\n    const data: Record<string, unknown> = {};\n    for (const key in formGroup.controls) {\n      const control = formGroup.controls[key];\n      if (!(control instanceof FormControl)) {\n        data[key] = NgxFormService.getFormData(control as FormGroup);\n        continue;\n      }\n\n      const props = NgxFormService.getPropsFromControl(control);\n      let value = control.value;\n      if (!HTML5CheckTypes.includes(props['type'])) {\n        switch (props['type']) {\n          case HTML5InputTypes.NUMBER:\n            value = parseToNumber(value);\n            break;\n          case HTML5InputTypes.DATE:\n          case HTML5InputTypes.DATETIME_LOCAL:\n            value = new Date(value);\n            break;\n          default:\n            value = escapeHtml(value);\n        }\n      }\n      data[key] = value;\n    }\n\n    return data;\n  }\n\n  /**\n   * @description Validates fields in a form control or form group.\n   * @summary Recursively validates all fields in a form control or form group, marking them as touched and dirty.\n   * @param {AbstractControl} control - The control or form group to validate.\n   * @param {string} [path] - The path to the control within the form.\n   * @return {boolean} True if all fields are valid, false otherwise.\n   * @throws {Error} If no control is found at the specified path or if the control type is unknown.\n   */\n  static validateFields(control: AbstractControl, path?: string): boolean {\n    control = path ? control.get(path) as AbstractControl : control;\n    if (!control)\n      throw new Error(`No control found at path: ${path || 'root'}.`);\n\n    const isAllowed = [FormGroup, FormControl].some(type => control instanceof type);\n    if (!isAllowed)\n      throw new Error(`Unknown control type at: ${path || 'root'}`);\n\n    control.markAsTouched();\n    control.markAsDirty();\n    control.updateValueAndValidity({ emitEvent: true });\n\n    if (control instanceof FormGroup) {\n      Object.values(control.controls).forEach((childControl) => {\n        this.validateFields(childControl);\n      });\n    }\n\n    return control.valid;\n  }\n\n  /**\n   * @description Generates validators from component properties.\n   * @summary Creates an array of ValidatorFn based on the supported validation keys in the component properties.\n   * @param {FieldProperties} props - The component properties.\n   * @return {ValidatorFn[]} An array of validator functions.\n   */\n  private static validatorsFromProps(props: FieldProperties): ValidatorFn[] {\n    const supportedValidationKeys = Validation.keys();\n    return Object.keys(props)\n      .filter((k: string) => supportedValidationKeys.includes(k))\n      .map((k: string) => {\n        return ValidatorFactory.spawn(props, k);\n      });\n  }\n\n  /**\n   * @description Creates a FormControl from component properties.\n   * @summary Generates a FormControl with validators based on the provided component properties.\n   * @param {FieldProperties} props - The component properties.\n   * @param {FieldUpdateMode} [updateMode='change'] - The update mode for the control.\n   * @return {FormControl} The created FormControl.\n   */\n  static fromProps(props: FieldProperties, updateMode: FieldUpdateMode = 'change'): FormControl {\n    const validators = this.validatorsFromProps(props);\n    const composed = validators.length ? Validators.compose(validators) : null;\n    return new FormControl(\n      {\n        value:\n          props.value && props.type !== HTML5InputTypes.CHECKBOX\n            ? props.type === HTML5InputTypes.DATE\n              ? !isValidDate(parseDate(props.format as string, props.value as string))\n                ? undefined : props.value :\n              (props.value as unknown) : undefined,\n        disabled: props.disabled,\n      },\n      {\n        validators: composed,\n        updateOn: updateMode,\n      },\n    );\n  }\n\n  /**\n   * @description Retrieves properties from a FormControl.\n   * @summary Gets the FieldProperties associated with a FormControl from the internal WeakMap.\n   * @param {FormControl} control - The FormControl to get properties for.\n   * @return {FieldProperties} The properties associated with the control.\n   */\n  static getPropsFromControl(control: FormControl): FieldProperties {\n    return this.controls.get(control) || {} as FieldProperties;\n  }\n\n  /**\n   * @description Finds a parent element with a specific tag.\n   * @summary Traverses up the DOM tree to find the nearest parent element with the specified tag.\n   * @param {HTMLElement} el - The starting element.\n   * @param {string} tag - The tag name to search for.\n   * @return {HTMLElement} The found parent element.\n   * @throws {Error} If no parent with the specified tag is found.\n   */\n  static getParentEl(el: HTMLElement, tag: string) {\n    let parent: HTMLElement | null;\n    while ((parent = el.parentElement) !== null) {\n      if (parent.tagName.toLowerCase() === tag.toLowerCase()) {\n        return parent;\n      }\n      el = parent;\n    }\n    throw new Error(\n      `No parent with the tag ${tag} was found for provided element`,\n    );\n  }\n\n  /**\n   * @description Registers a control with its properties.\n   * @summary Associates a control with its properties in the internal WeakMap.\n   * @param {AbstractControl} control - The control to register.\n   * @param {FieldProperties} props - The properties to associate with the control.\n   */\n  static register(control: AbstractControl, props: FieldProperties) {\n    this.controls.set(control, props);\n  }\n\n  /**\n   * @description Unregisters a control.\n   * @summary Removes a control and its associated properties from the internal WeakMap.\n   * @param {AbstractControl} control - The control to unregister.\n   * @return {boolean} True if the control was successfully unregistered, false otherwise.\n   */\n  static unregister(control: AbstractControl): boolean {\n    return this.controls.delete(control);\n  }\n\n  /**\n   * @description Resets a form group.\n   * @summary Recursively resets all controls in a form group, clearing values, errors, and marking them as pristine and untouched.\n   * @param {FormGroup} formGroup - The form group to reset.\n   */\n  static reset(formGroup: FormGroup) {\n    for (const key in formGroup.controls) {\n      const control = formGroup.controls[key];\n      if (!(control instanceof FormControl)) {\n        NgxFormService.reset(control as FormGroup);\n        continue;\n      }\n\n      const { type } = NgxFormService.getPropsFromControl(control);\n      if (!HTML5CheckTypes.includes(type))\n        control.setValue(undefined);\n      control.markAsPristine();\n      control.markAsUntouched();\n      control.setErrors(null);\n      control.updateValueAndValidity();\n    }\n  }\n}\n"]}
527
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"NgxFormService.js","sourceRoot":"","sources":["../../../../src/lib/engine/NgxFormService.ts"],"names":[],"mappings":"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","sourcesContent":["import { escapeHtml, FieldProperties, HTML5CheckTypes, HTML5InputTypes, parseToNumber } from '@decaf-ts/ui-decorators';\nimport { FieldUpdateMode, FormParentGroup, KeyValue } from './types';\nimport { IComponentConfig, IComponentInput } from './interfaces';\nimport { AbstractControl, FormArray, FormControl, FormGroup, ValidatorFn, Validators } from '@angular/forms';\nimport { isValidDate, ModelKeys, parseDate, Primitives, Validation } from '@decaf-ts/decorator-validation';\nimport { ValidatorFactory } from './ValidatorFactory';\nimport { cleanSpaces } from '../helpers';\nimport { OperationKeys } from '@decaf-ts/db-decorators';\nimport { AngularEngineKeys, BaseComponentProps } from '../engine/constants';\n\n\n/**\n * @description Service for managing Angular forms and form controls.\n * @summary The NgxFormService provides utility methods for creating, managing, and validating Angular forms and form controls. It includes functionality for registering forms, adding controls, validating fields, and handling form data.\n *\n * @class\n * @param {WeakMap<AbstractControl, FieldProperties>} controls - A WeakMap to store control properties.\n * @param {Map<string, FormGroup>} formRegistry - A Map to store registered forms.\n *\n * @example\n * // Creating a form from components\n * const components = [\n *   { inputs: { name: 'username', type: 'text', required: true } },\n *   { inputs: { name: 'password', type: 'password', minLength: 8 } }\n * ];\n * const form = NgxFormService.createFormFromComponents('loginForm', components, true);\n *\n * // Validating fields\n * NgxFormService.validateFields(form);\n *\n * // Getting form data\n * const formData = NgxFormService.getFormData(form);\n *\n * @mermaid\n * sequenceDiagram\n *   participant C as Component\n *   participant NFS as NgxFormService\n *   participant AF as Angular Forms\n *   C->>NFS: createFormFromComponents()\n *   NFS->>AF: new FormGroup()\n *   NFS->>NFS: addFormControl()\n *   NFS->>AF: addControl()\n *   NFS-->>C: Return FormGroup\n *   C->>NFS: validateFields()\n *   NFS->>AF: markAsTouched(), markAsDirty(), updateValueAndValidity()\n *   C->>NFS: getFormData()\n *   NFS->>AF: Get control values\n *   NFS-->>C: Return form data\n */\nexport class NgxFormService {\n  /**\n   * @description WeakMap that stores control properties for form controls.\n   * @summary A WeakMap that associates AbstractControl instances with their corresponding FieldProperties.\n   * This allows the service to track metadata for form controls without creating memory leaks.\n   *\n   * @type {WeakMap<AbstractControl, FieldProperties>}\n   * @private\n   * @static\n   * @memberOf NgxFormService\n   */\n  private static controls = new WeakMap<AbstractControl, FieldProperties>();\n\n  /**\n   * @description Registry of form groups indexed by their unique identifiers.\n   * @summary A Map that stores FormGroup instances with their unique string identifiers.\n   * This allows global access to registered forms throughout the application.\n   *\n   * @type {Map<string, FormGroup>}\n   * @private\n   * @static\n   * @memberOf NgxFormService\n   */\n  private static formRegistry = new Map<string, FormGroup>();\n  /**\n   * @description Adds a form to the registry.\n   * @summary Registers a FormGroup with a unique identifier. Throws an error if the identifier is already in use.\n   * @param {string} formId - The unique identifier for the form.\n   * @param {FormGroup} formGroup - The FormGroup to be registered.\n   * @throws {Error} If a FormGroup with the given id is already registered.\n   */\n  static addRegistry(formId: string, formGroup: FormGroup): void {\n    if (this.formRegistry.has(formId))\n      throw new Error(`A FormGroup with id '${formId}' is already registered.`);\n    this.formRegistry.set(formId, formGroup);\n  }\n\n  /**\n   * @description Removes a form from the registry.\n   * @summary Deletes a FormGroup from the registry using its unique identifier.\n   * @param {string} formId - The unique identifier of the form to be removed.\n   */\n  static removeRegistry(formId: string): void {\n    this.formRegistry.delete(formId);\n  }\n\n  /**\n   * @description Resolves the parent group and control name from a path.\n   * @summary Traverses the form group structure to find the parent group and control name for a given path.\n   * @param {FormGroup} formGroup - The root FormGroup.\n   * @param {string} path - The path to the control.\n   * @return {FormParentGroup} A tuple containing the parent FormGroup and the control name.\n   */\n  private static resolveParentGroup(formGroup: FormGroup, path: string, componentProps: IComponentInput, parentProps: KeyValue): FormParentGroup {\n    const isMultiple = parentProps?.['multiple'] || parentProps?.['type'] === 'Array' || false;\n    const parts = path.split('.');\n    const controlName = parts.pop() as string;\n    const {childOf} = componentProps\n    let currentGroup = formGroup;\n\n    function setArrayComponentProps(formGroupArray: FormArray) {\n      const props = (formGroupArray as KeyValue)[AngularEngineKeys.FORM_GROUP_COMPONENT_PROPS] || {};\n        if(!props[ModelKeys.MODEL][controlName])\n          props[ModelKeys.MODEL] = Object.assign({}, props[ModelKeys.MODEL], {[controlName]: {...componentProps}});\n    }\n\n    for (const part of parts) {\n      if (!currentGroup.get(part)) {\n        const partFormGroup = (isMultiple && part === childOf) ? new FormArray([new FormGroup({})]) : new FormGroup({});\n        (partFormGroup as KeyValue)[AngularEngineKeys.FORM_GROUP_COMPONENT_PROPS] = {\n          childOf: childOf || '',\n          isMultiple: isMultiple,\n          name: part,\n          pk: componentProps?.['pk'] || parentProps?.['pk'] || '',\n          [ModelKeys.MODEL]: {},\n        } as Partial<FieldProperties> & {model: KeyValue};\n\n        if(currentGroup instanceof FormArray) {\n          (currentGroup as FormArray).push(partFormGroup);\n        } else {\n\n          for(const control of Object.values(partFormGroup.controls)) {\n            if(control instanceof FormControl)\n              this.register(control as AbstractControl, componentProps);\n          }\n\n          if(partFormGroup instanceof AbstractControl)\n            this.register(partFormGroup as AbstractControl, componentProps);\n\n          currentGroup.addControl(part, partFormGroup);\n        }\n      }\n      if(childOf && currentGroup instanceof FormArray)\n        setArrayComponentProps(currentGroup);\n\n      currentGroup = currentGroup.get(part) as FormGroup;\n    }\n    return [currentGroup, controlName];\n  }\n\n  /**\n   * @description Retrieves component properties from a FormGroup or FormArray.\n   * @summary Extracts component properties stored in the form group metadata. If a FormGroup is provided\n   * and groupArrayName is specified, it will look for the FormArray within the form structure.\n   *\n   * @param {FormGroup | FormArray} formGroup - The form group or form array to extract properties from\n   * @param {string} [key] - Optional key to retrieve a specific property\n   * @param {string} [groupArrayName] - Optional name of the group array if formGroup is not a FormArray\n   * @return {Partial<FieldProperties>} The component properties or a specific property if key is provided\n   *\n   * @static\n   * @memberOf NgxFormService\n   */\n  static getComponentPropsFromGroupArray(formGroup: FormGroup | FormArray, key?: string, groupArrayName?: string | undefined): Partial<FieldProperties> {\n    if(!(formGroup instanceof FormArray) && typeof groupArrayName === Primitives.STRING)\n      formGroup = formGroup.root.get(groupArrayName as string) as FormArray || {};\n    const props = (formGroup as KeyValue)?.[AngularEngineKeys.FORM_GROUP_COMPONENT_PROPS] || {};\n    return (!key ? props : props?.[key]) || {};\n  }\n\n  /**\n   * @description Adds a new group to a parent FormArray.\n   * @summary Creates and adds a new FormGroup to the specified parent FormArray based on the\n   * component properties stored in the parent's metadata. This is used for dynamic form arrays\n   * where new groups need to be added at runtime.\n   *\n   * @param {FormGroup} formGroup - The root form group containing the parent FormArray\n   * @param {string} parentName - The name of the parent FormArray to add the group to\n   * @param {number} [index=1] - The index position where the new group should be added\n   * @return {FormGroup} The newly created and added FormGroup\n   *\n   * @static\n   * @memberOf NgxFormService\n   */\n  static addGroupToParent(formGroup: FormGroup, parentName: string, index: number = 1): FormGroup {\n    const componentProps = this.getComponentPropsFromGroupArray(formGroup, ModelKeys.MODEL, parentName);\n    Object.entries(componentProps as KeyValue).forEach(([, value]) => {\n      return this.addFormControl(formGroup, value, {multiple: true}, index);\n    });\n\n    return this.getGroupFromParent(formGroup, parentName, index);\n  }\n\n  /**\n   * @description Retrieves a FormGroup from a parent FormArray at the specified index.\n   * @summary Gets a FormGroup from the specified parent FormArray. If the group doesn't exist\n   * at the given index, it will create a new one using addGroupToParent.\n   *\n   * @param {FormGroup} formGroup - The root form group containing the parent FormArray\n   * @param {string} parentName - The name of the parent FormArray to retrieve the group from\n   * @param {number} [index=1] - The index of the group to retrieve\n   * @return {FormGroup} The FormGroup at the specified index\n   *\n   * @static\n   * @memberOf NgxFormService\n   */\n  static getGroupFromParent(formGroup: FormGroup, parentName: string, index: number = 1): FormGroup {\n    const childGroup = ((formGroup.get(parentName) || formGroup) as FormArray).at(index);\n    if(childGroup instanceof FormGroup)\n      return childGroup;\n    return this.addGroupToParent(formGroup, parentName, index);\n  }\n\n  /**\n   * @description Checks if a value is unique within a FormArray group.\n   * @summary Validates that the primary key value in a FormGroup is unique among all groups\n   * in the parent FormArray. The uniqueness check behavior differs based on the operation type.\n   *\n   * @param {FormGroup} formGroup - The FormGroup to check for uniqueness\n   * @param {number} index - The index of the current group within the FormArray\n   * @param {OperationKeys} [operation=OperationKeys.CREATE] - The type of operation being performed\n   * @return {boolean} True if the value is unique, false otherwise\n   *\n   * @static\n   * @memberOf NgxFormService\n   */\n  static isUniqueOnGroup(formGroup: FormGroup, index: number, operation: OperationKeys = OperationKeys.CREATE): boolean {\n    const formGroupArray = formGroup.parent as FormArray;\n    const pk = this.getComponentPropsFromGroupArray(formGroupArray, BaseComponentProps.PK as string) as string;\n    const controlName = Object.keys(formGroup.controls)[0];\n\n    // only check for unique if is the pk control\n    if(controlName !== pk)\n      return true;\n    const controlValue = cleanSpaces(`${formGroup.get(pk)?.value}`, true);\n    if(operation === OperationKeys.CREATE)\n      return !formGroupArray.controls.some((group, i) => i !== index && cleanSpaces(`${group.get(pk)?.value}`, true) === controlValue);\n\n    return !formGroupArray.controls.some((group, i) =>\n      i !== index && controlValue === cleanSpaces(`${group.get(pk)?.value}`, true)\n    );\n  }\n\n  /**\n   * @description Enables all controls within a FormGroup or FormArray.\n   * @summary Recursively enables all form controls within the provided FormGroup or FormArray.\n   * This is useful for making all controls interactive after they have been disabled.\n   *\n   * @param {FormArray | FormGroup} formGroup - The FormGroup or FormArray to enable all controls for\n   * @return {void}\n   *\n   * @static\n   * @memberOf NgxFormService\n   */\n  static enableAllGroupControls(formGroup: FormArray | FormGroup): void {\n    Object.keys(formGroup.controls).forEach(key => {\n      const control = formGroup.get(key);\n      if (control instanceof FormArray) {\n        control.controls.forEach(child => {\n          if (child instanceof FormGroup) {\n            child.enable({ emitEvent: false });\n            child.updateValueAndValidity({ emitEvent: true });\n          }\n        });\n      }\n    });\n  }\n\n  /**\n   * @description Adds a form control to a form group based on component properties.\n   * @summary Creates and configures a FormControl within the specified FormGroup using the provided\n   * component properties. Handles nested paths, multiple controls (FormArrays), and control registration.\n   * This method supports complex form structures with nested groups and arrays.\n   *\n   * @param {FormGroup} formGroup - The form group to add the control to\n   * @param {IComponentInput} componentProps - The component properties defining the control configuration\n   * @param {KeyValue} [parentProps={}] - Properties from the parent component for context\n   * @param {number} [index=0] - The index for multiple controls in FormArrays\n   * @return {void}\n   *\n   * @private\n   * @static\n   * @memberOf NgxFormService\n   */\n  private static addFormControl(formGroup: FormGroup, componentProps: IComponentInput, parentProps: KeyValue = {}, index: number = 0): void {\n\n    const isMultiple = parentProps?.['multiple'] || parentProps?.['type'] === 'Array' || false;\n    const { name, childOf, } = componentProps;\n    if(isMultiple)\n      componentProps['pk'] = componentProps['pk'] || parentProps?.['pk'] || '';\n    const fullPath = childOf ? isMultiple ? `${childOf}.${index}.${name}` : `${childOf}.${name}` : name;\n    const [parentGroup, controlName] = this.resolveParentGroup(formGroup, fullPath, componentProps, parentProps);\n\n    if (!parentGroup.get(controlName)) {\n      const control = NgxFormService.fromProps(\n        componentProps,\n        componentProps.updateMode || 'change',\n      );\n      NgxFormService.register(control, componentProps);\n      parentGroup.addControl(controlName, control);\n    }\n\n    componentProps['formGroup'] = parentGroup;\n    componentProps['formControl'] = parentGroup.get(controlName) as FormControl;\n    componentProps['multiple'] = isMultiple\n\n  }\n\n  /**\n   * @description Retrieves a control from a registered form.\n   * @summary Finds and returns an AbstractControl from a registered form using the form id and optional path.\n   * @param {string} formId - The unique identifier of the form.\n   * @param {string} [path] - The path to the control within the form.\n   * @return {AbstractControl} The requested AbstractControl.\n   * @throws {Error} If the form is not found in the registry or the control is not found in the form.\n   */\n  static getControlFromForm(formId: string, path?: string): AbstractControl {\n    const form = this.formRegistry.get(formId);\n    if (!form)\n      throw new Error(`Form with id '${formId}' not found in the registry.`);\n\n    if (!path)\n      return form;\n\n    const control = form.get(path);\n    if (!control)\n      throw new Error(`Control with path '${path}' not found in form '${formId}'.`);\n    return control;\n  }\n\n  /**\n   * @description Creates a form from component configurations.\n   * @summary Generates a FormGroup based on an array of component configurations and optionally registers it.\n   * @param {string} id - The unique identifier for the form.\n   * @param {IComponentConfig[]} components - An array of component configurations.\n   * @param {boolean} [registry=false] - Whether to register the created form.\n   * @return {FormGroup} The created FormGroup.\n   */\n  static createFormFromComponents(id: string, components: IComponentConfig[], registry: boolean = false): FormGroup {\n    const form = new FormGroup({});\n    components.forEach(component => {\n      this.addFormControl(form, component.inputs);\n    });\n\n    if (registry)\n      this.addRegistry(id, form);\n\n    return form;\n  }\n\n  /**\n   * @description Adds a control to a form based on component properties.\n   * @summary Creates and adds a form control to a form (existing or new) based on the provided component properties.\n   * @param {string} id - The unique identifier of the form.\n   * @param {FieldProperties} componentProperties - The properties of the component to create the control from.\n   * @return {AbstractControl} The form or created control.\n   */\n  static addControlFromProps(id: string, componentProperties: FieldProperties, parentProps?: FieldProperties): AbstractControl {\n    const form = this.formRegistry.get(id) ?? new FormGroup({});\n    if (!this.formRegistry.has(id))\n      this.addRegistry(id, form);\n\n    if (componentProperties.path)\n      this.addFormControl(form, componentProperties, parentProps);\n\n    return form;\n  }\n\n  /**\n   * @description Retrieves form data from a FormGroup.\n   * @summary Extracts and processes the data from a FormGroup, handling different input types and nested form groups.\n   * @param {FormGroup} formGroup - The FormGroup to extract data from.\n   * @return {Record<string, unknown>} An object containing the form data.\n   */\n  static getFormData(formGroup: FormGroup): Record<string, unknown> {\n    const data: Record<string, unknown> = {};\n    for (const key in formGroup.controls) {\n      const control = formGroup.controls[key];\n      const parentProps = NgxFormService.getPropsFromControl(formGroup as FormGroup | FormArray);\n      if (!(control instanceof FormControl)) {\n        const value = NgxFormService.getFormData(control as FormGroup);\n        const isValid = control.valid;\n        if(parentProps.multiple) {\n            if(isValid) {\n               data[key] = value;\n            } else {\n              this.reset(control as FormControl);\n            }\n\n            continue;\n        }\n        data[key] = value;\n        continue;\n      }\n\n      const props = NgxFormService.getPropsFromControl(control as FormControl | FormArray);\n      let value = control.value;\n      if (!HTML5CheckTypes.includes(props['type'])) {\n        switch (props['type']) {\n          case HTML5InputTypes.NUMBER:\n            value = parseToNumber(value);\n            break;\n          case HTML5InputTypes.DATE:\n          case HTML5InputTypes.DATETIME_LOCAL:\n            value = new Date(value);\n            break;\n          default:\n            value = escapeHtml(value);\n        }\n      }\n      data[key] = value;\n    }\n    NgxFormService.enableAllGroupControls(formGroup as FormGroup);\n    return data;\n  }\n\n  /**\n   * @description Validates fields in a form control or form group.\n   * @summary Recursively validates all fields in a form control or form group, marking them as touched and dirty.\n   * @param {AbstractControl} control - The control or form group to validate.\n   * @param {string} [path] - The path to the control within the form.\n   * @return {boolean} True if all fields are valid, false otherwise.\n   * @throws {Error} If no control is found at the specified path or if the control type is unknown.\n   */\n  static validateFields(control: AbstractControl, pk?: string,  path?: string): boolean {\n    control = path ? control.get(path) as AbstractControl : control;\n    if (!control)\n      throw new Error(`No control found at path: ${path || 'root'}.`);\n\n    const isAllowed = [FormArray, FormGroup, FormControl].some(type => control instanceof type);\n    if (!isAllowed)\n      throw new Error(`Unknown control type at: ${path || 'root'}`);\n\n    control.markAsTouched();\n    control.markAsDirty();\n    control.updateValueAndValidity({ emitEvent: true });\n\n    if (control instanceof FormGroup) {\n      Object.values(control.controls).forEach(childControl => {\n        this.validateFields(childControl);\n      });\n    }\n\n    if (control instanceof FormArray) {\n      const totalGroups = control.length;\n      const hasValid = control.controls.some(control => control.valid);\n      if(totalGroups > 1 && hasValid) {\n         for (let i = control.length - 1; i >= 0; i--) {\n          const childControl = control.at(i);\n          // disable no valid groups on array\n          if (!childControl.valid) {\n            (childControl.parent as FormGroup).setErrors(null);\n             (childControl.parent as FormGroup).updateValueAndValidity({ emitEvent: true });\n            childControl.disable();\n          } else {\n            this.validateFields(childControl);\n          }\n        }\n      } else {\n        Object.values(control.controls).forEach(childControl => {\n          this.validateFields(childControl);\n        });\n      }\n    }\n\n    return control.valid;\n  }\n\n  /**\n   * @description Generates validators from component properties.\n   * @summary Creates an array of ValidatorFn based on the supported validation keys in the component properties.\n   * @param {FieldProperties} props - The component properties.\n   * @return {ValidatorFn[]} An array of validator functions.\n   */\n  private static validatorsFromProps(props: FieldProperties): ValidatorFn[] {\n    const supportedValidationKeys = Validation.keys();\n    return Object.keys(props)\n      .filter((k: string) => supportedValidationKeys.includes(k))\n      .map((k: string) => {\n        return ValidatorFactory.spawn(props, k);\n      });\n  }\n\n  /**\n   * @description Creates a FormControl from component properties.\n   * @summary Generates a FormControl with validators based on the provided component properties.\n   * @param {FieldProperties} props - The component properties.\n   * @param {FieldUpdateMode} [updateMode='change'] - The update mode for the control.\n   * @return {FormControl} The created FormControl.\n   */\n  static fromProps(props: FieldProperties, updateMode: FieldUpdateMode = 'change'): FormControl {\n    const validators = this.validatorsFromProps(props);\n    const composed = validators.length ? Validators.compose(validators) : null;\n    return new FormControl(\n      {\n        value:\n          props.value && props.type !== HTML5InputTypes.CHECKBOX\n            ? props.type === HTML5InputTypes.DATE\n              ? !isValidDate(parseDate(props.format as string, props.value as string))\n                ? undefined : props.value :\n              (props.value as unknown) : undefined,\n        disabled: props.disabled,\n      },\n      {\n        validators: composed,\n        updateOn: updateMode,\n      },\n    );\n  }\n\n  /**\n   * @description Retrieves properties from a FormControl.\n   * @summary Gets the FieldProperties associated with a FormControl from the internal WeakMap.\n   * @param {FormControl} control - The FormControl to get properties for.\n   * @return {FieldProperties} The properties associated with the control.\n   */\n  static getPropsFromControl(control: FormControl | FormArray | FormGroup): FieldProperties {\n    return this.controls.get(control) || {} as FieldProperties;\n  }\n\n  /**\n   * @description Finds a parent element with a specific tag.\n   * @summary Traverses up the DOM tree to find the nearest parent element with the specified tag.\n   * @param {HTMLElement} el - The starting element.\n   * @param {string} tag - The tag name to search for.\n   * @return {HTMLElement} The found parent element.\n   * @throws {Error} If no parent with the specified tag is found.\n   */\n  static getParentEl(el: HTMLElement, tag: string): HTMLElement {\n    let parent: HTMLElement | null;\n    while ((parent = el.parentElement) !== null) {\n      if (parent.tagName.toLowerCase() === tag.toLowerCase()) {\n        return parent;\n      }\n      el = parent;\n    }\n    throw new Error(\n      `No parent with the tag ${tag} was found for provided element`,\n    );\n  }\n\n  /**\n   * @description Registers a control with its properties.\n   * @summary Associates a control with its properties in the internal WeakMap.\n   * @param {AbstractControl} control - The control to register.\n   * @param {FieldProperties} props - The properties to associate with the control.\n   */\n  static register(control: AbstractControl, props: FieldProperties) {\n    this.controls.set(control, props);\n  }\n\n  /**\n   * @description Unregisters a control.\n   * @summary Removes a control and its associated properties from the internal WeakMap.\n   * @param {AbstractControl} control - The control to unregister.\n   * @return {boolean} True if the control was successfully unregistered, false otherwise.\n   */\n  static unregister(control: AbstractControl): boolean {\n    return this.controls.delete(control);\n  }\n\n  /**\n   * @description Resets a form group.\n   * @summary Recursively resets all controls in a form group, clearing values, errors, and marking them as pristine and untouched.\n   * @param {FormGroup} formGroup - The form group to reset.\n   */\n  static reset(formGroup: FormGroup | FormControl): void {\n    if(formGroup instanceof FormControl) {\n      const control = formGroup as FormControl;\n      const { type } = NgxFormService.getPropsFromControl(control);\n      if (!HTML5CheckTypes.includes(type))\n        control.setValue(\"\");\n      control.markAsPristine();\n      control.markAsUntouched();\n      control.setErrors(null);\n      control.updateValueAndValidity();\n    } else {\n      for (const key in formGroup.controls) {\n        const control = formGroup.controls[key];\n        NgxFormService.reset(control as FormControl);\n        continue;\n      }\n    }\n  }\n}\n"]}