@decaf-ts/for-angular 0.0.19 → 0.0.20

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 (38) hide show
  1. package/README.md +2 -2
  2. package/components/crud-form/crud-form.component.d.ts +1 -1
  3. package/components/fieldset/fieldset.component.d.ts +1 -1
  4. package/components/filter/filter.component.d.ts +1 -1
  5. package/components/for-angular-components.module.d.ts +1 -2
  6. package/components/list/constants.d.ts +5 -9
  7. package/components/list-item/list-item.component.d.ts +1 -1
  8. package/engine/NgxBaseComponent.d.ts +1 -1
  9. package/esm2022/components/component-renderer/component-renderer.component.mjs +7 -7
  10. package/esm2022/components/crud-field/crud-field.component.mjs +13 -12
  11. package/esm2022/components/crud-form/crud-form.component.mjs +9 -9
  12. package/esm2022/components/empty-state/empty-state.component.mjs +8 -12
  13. package/esm2022/components/fieldset/fieldset.component.mjs +27 -19
  14. package/esm2022/components/filter/filter.component.mjs +14 -13
  15. package/esm2022/components/for-angular-components.module.mjs +17 -9
  16. package/esm2022/components/layout/layout.component.mjs +6 -7
  17. package/esm2022/components/list/constants.mjs +1 -1
  18. package/esm2022/components/list/list.component.mjs +6 -11
  19. package/esm2022/components/list-item/list-item.component.mjs +7 -12
  20. package/esm2022/components/model-renderer/model-renderer.component.mjs +5 -7
  21. package/esm2022/components/pagination/pagination.component.mjs +8 -9
  22. package/esm2022/components/searchbar/searchbar.component.mjs +5 -7
  23. package/esm2022/directives/collapsable.directive.mjs +3 -3
  24. package/esm2022/engine/NgxBaseComponent.mjs +11 -10
  25. package/esm2022/engine/NgxFormService.mjs +12 -3
  26. package/esm2022/engine/NgxRenderingEngine.mjs +3 -2
  27. package/esm2022/for-angular-common.module.mjs +82 -0
  28. package/esm2022/helpers/utils.mjs +2 -2
  29. package/esm2022/i18n/Loader.mjs +10 -6
  30. package/esm2022/i18n/data/en.json +80 -0
  31. package/esm2022/public-apis.mjs +3 -2
  32. package/fesm2022/decaf-ts-for-angular.mjs +254 -170
  33. package/fesm2022/decaf-ts-for-angular.mjs.map +1 -1
  34. package/for-angular-common.module.d.ts +44 -0
  35. package/package.json +6 -3
  36. package/public-apis.d.ts +2 -1
  37. package/esm2022/for-angular.module.mjs +0 -118
  38. package/for-angular.module.d.ts +0 -45
@@ -1,11 +1,11 @@
1
1
  import { Input, Component, Inject, ViewChild, ElementRef, Output, EventEmitter, } from '@angular/core';
2
- import { generateRandomValue, getInjectablesRegistry, getOnWindow, isDevelopmentMode, stringToBoolean } from '../helpers/utils';
2
+ import { generateRandomValue, getInjectablesRegistry, getOnWindow, isDevelopmentMode, stringToBoolean, } from '../helpers/utils';
3
3
  import { getLocaleContext } from '../i18n/Loader';
4
4
  import { Model } from '@decaf-ts/decorator-validation';
5
- import { InternalError, OperationKeys } from '@decaf-ts/db-decorators';
5
+ import { InternalError, OperationKeys, } from '@decaf-ts/db-decorators';
6
6
  import { BaseComponentProps } from './constants';
7
7
  import { NgxRenderingEngine } from './NgxRenderingEngine';
8
- import { getLogger } from '../for-angular.module';
8
+ import { getLogger } from '../for-angular-common.module';
9
9
  import { Repository } from '@decaf-ts/core';
10
10
  import { RamAdapter } from '@decaf-ts/core/ram';
11
11
  import * as i0 from "@angular/core";
@@ -261,13 +261,14 @@ export class NgxBaseComponent {
261
261
  throw new InternalError('Cannot find model. was it registered with @model?');
262
262
  let dbAdapterFlavour = getOnWindow('dbAdapterFlavour');
263
263
  if (!dbAdapterFlavour && isDevelopmentMode()) {
264
- const adapter = new RamAdapter();
264
+ const adapter = new RamAdapter({ user: 'user' });
265
265
  dbAdapterFlavour = adapter.flavour;
266
266
  }
267
267
  this._repository = Repository.forModel(constructor, dbAdapterFlavour);
268
268
  this.model = new constructor();
269
269
  if (this.model && !this.pk)
270
- this.pk = this._repository.pk || 'id';
270
+ this.pk =
271
+ this._repository.pk || 'id';
271
272
  }
272
273
  }
273
274
  catch (error) {
@@ -490,14 +491,14 @@ export class NgxBaseComponent {
490
491
  this[key] = this.props[key];
491
492
  });
492
493
  }
493
- static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: NgxBaseComponent, deps: [{ token: 'instanceToken' }], target: i0.ɵɵFactoryTarget.Component }); }
494
- static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: NgxBaseComponent, isStandalone: true, selector: "ng-component", inputs: { rendererId: "rendererId", model: "model", props: "props", item: "item", pk: "pk", route: "route", operations: "operations", uid: "uid", mapper: "mapper", locale: "locale", translatable: "translatable", className: "className", mode: "mode", renderChild: "renderChild" }, outputs: { listenEvent: "listenEvent" }, host: { properties: { "attr.id": "uid" } }, viewQueries: [{ propertyName: "component", first: true, predicate: ["component"], descendants: true, read: ElementRef, static: true }], usesOnChanges: true, ngImport: i0, template: '<div><div>', isInline: true }); }
494
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: NgxBaseComponent, deps: [{ token: 'instanceToken' }], target: i0.ɵɵFactoryTarget.Component }); }
495
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.14", type: NgxBaseComponent, isStandalone: true, selector: "ng-component", inputs: { rendererId: "rendererId", model: "model", props: "props", item: "item", pk: "pk", route: "route", operations: "operations", uid: "uid", mapper: "mapper", locale: "locale", translatable: "translatable", className: "className", mode: "mode", renderChild: "renderChild" }, outputs: { listenEvent: "listenEvent" }, host: { properties: { "attr.id": "uid" } }, viewQueries: [{ propertyName: "component", first: true, predicate: ["component"], descendants: true, read: ElementRef, static: true }], usesOnChanges: true, ngImport: i0, template: '<div></div>', isInline: true }); }
495
496
  }
496
- i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: NgxBaseComponent, decorators: [{
497
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: NgxBaseComponent, decorators: [{
497
498
  type: Component,
498
499
  args: [{
499
500
  standalone: true,
500
- template: '<div><div>',
501
+ template: '<div></div>',
501
502
  host: { '[attr.id]': 'uid' },
502
503
  }]
503
504
  }], ctorParameters: () => [{ type: undefined, decorators: [{
@@ -537,4 +538,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
537
538
  }], listenEvent: [{
538
539
  type: Output
539
540
  }] } });
540
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"NgxBaseComponent.js","sourceRoot":"","sources":["../../../../src/lib/engine/NgxBaseComponent.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  Input,\n  Component,\n  Inject,\n  ViewChild,\n  ElementRef,\n  OnChanges,\n  SimpleChanges,\n  Output,\n  EventEmitter,\n} from '@angular/core';\nimport { KeyValue, RendererCustomEvent, StringOrBoolean } from './types';\nimport {\n  generateRandomValue,\n  getInjectablesRegistry,\n  getOnWindow,\n  isDevelopmentMode,\n  stringToBoolean\n} from '../helpers/utils';\nimport { getLocaleContext } from '../i18n/Loader';\nimport { Model } from '@decaf-ts/decorator-validation';\nimport {\n  CrudOperations,\n  InternalError,\n  OperationKeys\n} from '@decaf-ts/db-decorators';\nimport { BaseComponentProps } from './constants';\nimport { NgxRenderingEngine } from './NgxRenderingEngine';\nimport { Logger } from '@decaf-ts/logging';\nimport { getLogger } from '../for-angular.module';\nimport { DecafRepository } from '../components/list/constants';\nimport { Repository } from '@decaf-ts/core';\nimport { RamAdapter } from '@decaf-ts/core/ram';\n\n/**\n * @description Base component class that provides common functionality for all Decaf components.\n * @summary The NgxBaseComponent serves as the foundation for all Decaf UI components, providing\n * shared functionality for localization, element references, and styling. This abstract class\n * implements common properties and methods that are used across the component library, ensuring\n * consistent behavior and reducing code duplication. Components that extend this class inherit\n * its capabilities for handling translations, accessing DOM elements, and applying custom styling.\n *\n * @template M - The model type that this component works with\n * @param {string} instance - The component instance token used for identification\n * @param {string} locale - The locale to be used for translations\n * @param {StringOrBoolean} translatable - Whether the component should be translated\n * @param {string} className - Additional CSS classes to apply to the component\n * @param {\"ios\" | \"md\" | undefined} mode - Component platform style\n *\n * @component NgxBaseComponent\n * @example\n * ```typescript\n * @Component({\n *   selector: 'app-my-component',\n *   templateUrl: './my-component.component.html',\n *   styleUrls: ['./my-component.component.scss']\n * })\n * export class MyComponent extends NgxBaseComponent {\n *   constructor(@Inject('instanceToken') instance: string) {\n *     super(instance);\n *   }\n *\n *   ngOnInit() {\n *     this.initialize();\n *     // Component-specific initialization\n *   }\n * }\n * ```\n * @mermaid\n * sequenceDiagram\n *   participant App as Application\n *   participant Comp as Component\n *   participant Base as NgxBaseComponent\n *   participant Engine as NgxRenderingEngine\n *\n *   App->>Comp: Create component\n *   Comp->>Base: super(instance)\n *   Base->>Base: Set componentName & componentLocale\n *\n *   App->>Comp: Set @Input properties\n *   Comp->>Base: ngOnChanges(changes)\n *\n *   alt model changed\n *     Base->>Base: getModel(model)\n *     Base->>Engine: getDecorators(model, {})\n *     Engine-->>Base: Return decorator metadata\n *     Base->>Base: Configure mapper and item\n *     Base->>Base: getLocale(translatable)\n *   else locale/translatable changed\n *     Base->>Base: getLocale(translatable)\n *   end\n *\n *   App->>Comp: ngOnInit()\n *   Comp->>Base: initialize()\n *   Base->>Base: Set initialized flag\n */\n@Component({\n  standalone: true,\n  template: '<div><div>',\n  host: {'[attr.id]': 'uid'},\n})\nexport abstract class NgxBaseComponent implements OnChanges {\n  /**\n   * @description Reference to the component's element.\n   * @summary Provides direct access to the native DOM element of the component through Angular's\n   * ViewChild decorator. This reference can be used to manipulate the DOM element directly,\n   * apply custom styles, or access native element properties and methods. The element is\n   * identified by the 'component' template reference variable.\n   *\n   * @type {ElementRef}\n   * @memberOf NgxBaseComponent\n   */\n  @ViewChild('component', { read: ElementRef, static: true })\n  component!: ElementRef;\n\n  /**\n   * @description The name of the component.\n   * @summary Stores the name of the component, which is typically derived from the class name.\n   * This property is used internally for various purposes such as logging, deriving the default\n   * locale, and potentially for component identification in debugging or error reporting.\n   *\n   * The `componentName` is set during the component's initialization process and should not\n   * be modified externally. It's marked as protected to allow access in derived classes while\n   * preventing direct access from outside the component hierarchy.\n   *\n   * @type {string}\n   * @protected\n   * @memberOf NgxBaseComponent\n   *\n   * @example\n   * // Inside a derived component class\n   * console.log(this.componentName); // Outputs: \"MyCustomComponent\"\n   */\n  componentName!: string;\n\n  /**\n   * @description Unique identifier for the renderer.\n   * @summary A unique identifier used to reference the component's renderer instance.\n   * This can be used for targeting specific renderer instances when multiple components\n   * are present on the same page.\n   *\n   * @type {string}\n   * @memberOf NgxBaseComponent\n   */\n  @Input()\n  rendererId!: string;\n\n  /**\n   * @description Repository model for data operations.\n   * @summary The data model repository that this component will use for CRUD operations.\n   * This provides a connection to the data layer for retrieving and manipulating data.\n   *\n   * @type {Model| undefined}\n   * @memberOf NgxBaseComponent\n   */\n  @Input()\n  model!:  Model | undefined;\n\n  /**\n   * @description The repository for interacting with the data model.\n   * @summary Provides a connection to the data layer for retrieving and manipulating data.\n   * This is an instance of the `DecafRepository` class from the `@decaf-ts/core` package,\n   * which is initialized in the `repository` getter method.\n   *\n   * The repository is used to perform CRUD (Create, Read, Update, Delete) operations on the\n   * data model, such as fetching data, creating new items, updating existing items, and deleting\n   * items. It also provides methods for querying and filtering data based on specific criteria.\n   *\n   * @type {DecafRepository<Model>}\n   * @private\n   * @memberOf NgxBaseComponent\n   */\n  protected _repository?: DecafRepository<Model>;\n\n  /**\n   * @description Dynamic properties configuration object.\n   * @summary Contains key-value pairs of dynamic properties that can be applied to the component\n   * at runtime. This flexible configuration object allows for dynamic property assignment without\n   * requiring explicit input bindings for every possible configuration option. Properties from\n   * this object are parsed and applied to the component instance through the parseProps method,\n   * enabling customizable component behavior based on external configuration.\n   *\n   * @type {Record<string, unknown>}\n   * @default {}\n   * @memberOf NgxBaseComponent\n   */\n  @Input()\n  props: Record<string, unknown> = {};\n\n  /**\n   * @description Configuration for list item rendering\n   * @summary Defines how list items should be rendered in the component.\n   * This property holds a configuration object that specifies the tag name\n   * and other properties needed to render list items correctly. The tag property\n   * identifies which component should be used to render each item in a list.\n   * Additional properties can be included to customize the rendering behavior.\n   *\n   * @type {Record<string, unknown>}\n   * @default {tag: \"\"}\n   * @memberOf NgxBaseComponent\n   */\n  @Input()\n  item: Record<string, unknown> = { tag: '' };\n\n  /**\n   * @description Primary key field name for the model.\n   * @summary Specifies which field in the model should be used as the primary key.\n   * This is typically used for identifying unique records in operations like update and delete.\n   *\n   * @type {string}\n   * @default 'id'\n   * @memberOf NgxBaseComponent\n   */\n  @Input()\n  pk!: string;\n\n  /**\n   * @description Base route for navigation related to this component.\n   * @summary Defines the base route path used for navigation actions related to this component.\n   * This is often used as a prefix for constructing navigation URLs.\n   *\n   * @type {string}\n   * @memberOf NgxBaseComponent\n   */\n  @Input()\n  route!: string;\n\n  /**\n   * @description Available CRUD operations for this component.\n   * @summary Defines which CRUD operations (Create, Read, Update, Delete) are available\n   * for this component. This affects which operations can be performed on the data.\n   *\n   * @default [OperationKeys.READ]\n   * @memberOf NgxBaseComponent\n   */\n  @Input()\n  operations: CrudOperations[] = [OperationKeys.READ];\n\n  /**\n   * @description Unique identifier for the current record.\n   * @summary A unique identifier for the current record being displayed or manipulated.\n   * This is typically used in conjunction with the primary key for operations on specific records.\n   *\n   * @type {string | number}\n   * @memberOf NgxBaseComponent\n   */\n  @Input()\n  uid!: string | number;\n\n  /**\n   * @description Field mapping configuration.\n   * @summary Defines how fields from the data model should be mapped to properties used by the component.\n   * This allows for flexible data binding between the model and the component's display logic.\n   *\n   * @type {Record<string, string>}\n   * @memberOf NgxBaseComponent\n   */\n  @Input()\n  mapper: Record<string, string> = {};\n\n  /**\n   * @description The locale to be used for translations.\n   * @summary Specifies the locale identifier to use when translating component text.\n   * This can be set explicitly via input property to override the automatically derived\n   * locale from the component name. The locale is typically a language code (e.g., 'en', 'fr')\n   * or a language-region code (e.g., 'en-US', 'fr-CA') that determines which translation\n   * set to use for the component's text content.\n   *\n   * @type {string}\n   * @memberOf NgxBaseComponent\n   */\n  @Input()\n  locale!: string;\n\n  /**\n   * @description Determines if the component should be translated.\n   * @summary Controls whether the component's text content should be processed for translation.\n   * When true, the component will attempt to translate text using the specified locale.\n   * When false, text is displayed as-is without translation. This property accepts either\n   * a boolean value or a string that can be converted to a boolean (e.g., 'true', 'false', '1', '0').\n   *\n   * @type {StringOrBoolean}\n   * @default false\n   * @memberOf NgxBaseComponent\n   */\n  @Input()\n  translatable: StringOrBoolean = true;\n\n  /**\n   * @description Additional CSS class names to apply to the component.\n   * @summary Allows custom CSS classes to be added to the component's root element.\n   * These classes are appended to any automatically generated classes based on other\n   * component properties. Multiple classes can be provided as a space-separated string.\n   * This provides a way to customize the component's appearance beyond the built-in styling options.\n   *\n   * @type {string}\n   * @default \"\"\n   * @memberOf NgxBaseComponent\n   */\n  @Input()\n  className: string = '';\n\n  /**\n   * @description Component platform style.\n   * @summary Controls the visual appearance of the component based on platform design guidelines.\n   * The 'ios' mode follows iOS design patterns, while 'md' (Material Design) follows Android/Google\n   * design patterns. This property affects various visual aspects such as animations, form elements,\n   * and icons. Setting this property allows components to maintain platform-specific styling\n   * for a more native look and feel.\n   *\n   * @type {(\"ios\" | \"md\" | undefined)}\n   * @default \"md\"\n   * @memberOf NgxBaseComponent\n   */\n  @Input()\n  mode: 'ios' | 'md' | undefined = 'md';\n\n  /**\n   * @description The locale derived from the component's class name.\n   * @summary Stores the automatically derived locale based on the component's class name.\n   * This is determined during component initialization and serves as a fallback when no\n   * explicit locale is provided via the locale input property. The derivation is handled\n   * by the getLocaleContext utility function, which extracts a locale identifier\n   * from the component's class name.\n   *\n   * @type {string}\n   * @memberOf NgxBaseComponent\n   */\n  componentLocale!: string;\n\n  /**\n   * @description Controls whether child components should be rendered\n   * @summary Determines if child components should be rendered by the component.\n   * This can be set to a boolean value or a string that can be converted to a boolean.\n   * When true, child components defined in the model will be rendered. When false,\n   * child components will be skipped. This provides control over the rendering depth.\n   *\n   * @type {string | StringOrBoolean}\n   * @default true\n   * @memberOf NgxBaseComponent\n   */\n  @Input()\n  renderChild: string | StringOrBoolean = true;\n\n  /**\n   * @description Flag indicating if the component has been initialized\n   * @summary Tracks whether the component has completed its initialization process.\n   * This flag is used to prevent duplicate initialization and to determine if\n   * certain operations that require initialization can be performed.\n   *\n   * @type {boolean}\n   * @default false\n   */\n  initialized: boolean = false;\n\n  /**\n   * @description Event emitter for custom renderer events.\n   * @summary Emits custom events that occur within child components or the layout itself.\n   * This allows parent components to listen for and respond to user interactions or\n   * state changes within the grid layout. Events are passed up the component hierarchy\n   * to enable coordinated behavior across the application.\n   *\n   * @type {EventEmitter<RendererCustomEvent>}\n   * @memberOf NgxBaseComponent\n   */\n  @Output()\n  listenEvent: EventEmitter<RendererCustomEvent> = new EventEmitter<RendererCustomEvent>();\n\n  /**\n   * @description Reference to the rendering engine instance\n   * @summary Provides access to the NgxRenderingEngine singleton instance,\n   * which handles the rendering of components based on model definitions.\n   * This engine is used to extract decorator metadata and render child components.\n   *\n   * @type {NgxRenderingEngine}\n   */\n  renderingEngine: NgxRenderingEngine =\n    NgxRenderingEngine.get() as unknown as NgxRenderingEngine;\n\n  /**\n   * @description Logger instance for the component.\n   * @summary Provides logging capabilities for the component, allowing for consistent\n   * and structured logging of information, warnings, and errors. This logger is initialized\n   * in the ngOnInit method using the getLogger function from the ForAngularModule.\n   *\n   * The logger is used throughout the component to record important events, debug information,\n   * and potential issues. It helps in monitoring the component's behavior, tracking the flow\n   * of operations, and facilitating easier debugging and maintenance.\n   *\n   * @type {Logger}\n   * @private\n   * @memberOf NgxBaseComponent\n   */\n  logger!: Logger;\n\n  /**\n   * @description Creates an instance of NgxBaseComponent.\n   * @summary Initializes a new instance of the base component with the provided instance token.\n   * This constructor sets up the fundamental properties required by all Decaf components,\n   * including the component name, locale settings, and logging capabilities. The instance\n   * token is used for component identification and locale derivation.\n   *\n   * The constructor performs the following initialization steps:\n   * 1. Sets the componentName from the provided instance token\n   * 2. Derives the componentLocale from the class name using utility functions\n   * 3. Initializes the logger instance for the component\n   *\n   * @param {string} instance - The component instance token used for identification\n   *\n   * @mermaid\n   * sequenceDiagram\n   *   participant A as Angular\n   *   participant C as Component\n   *   participant B as NgxBaseComponent\n   *   participant U as Utils\n   *   participant L as Logger\n   *\n   *   A->>C: new Component(instance)\n   *   C->>B: super(instance)\n   *   B->>B: Set componentName = instance\n   *   B->>U: getLocaleContext(instance)\n   *   U-->>B: Return derived locale\n   *   B->>B: Set componentLocale\n   *   B->>L: getLogger(this)\n   *   L-->>B: Return logger instance\n   *   B->>B: Set logger\n   *\n   * @memberOf NgxBaseComponent\n   */\n  // eslint-disable-next-line @angular-eslint/prefer-inject\n  protected constructor(@Inject('instanceToken') protected instance: string) {\n    this.componentName = instance;\n    this.componentLocale = getLocaleContext(instance);\n    this.logger = getLogger(this);\n    this.getLocale(this.translatable);\n    this.uid = generateRandomValue(12);\n  }\n\n  /**\n   * @description Getter for the repository instance.\n   * @summary Provides a connection to the data layer for retrieving and manipulating data.\n   * This method initializes the `_repository` property if it is not already set, ensuring\n   * that a single instance of the repository is used throughout the component.\n   *\n   * The repository is used to perform CRUD operations on the data model, such as fetching data,\n   * creating new items, updating existing items, and deleting items. It also provides methods\n   * for querying and filtering data based on specific criteria.\n   *\n   * @returns {DecafRepository<Model>} The initialized repository instance.\n   * @private\n   * @memberOf NgxBaseComponent\n   */\n  protected get repository(): DecafRepository<Model> {\n    try {\n      if (!this._repository) {\n        const modelName  = (this.model as Model).constructor.name\n        const constructor = Model.get(modelName);\n        if (!constructor)\n          throw new InternalError(\n            'Cannot find model. was it registered with @model?',\n          );\n        let dbAdapterFlavour = getOnWindow('dbAdapterFlavour');\n        if(!dbAdapterFlavour && isDevelopmentMode()) {\n          const adapter = new RamAdapter();\n          dbAdapterFlavour = adapter.flavour;\n        }\n        this._repository = Repository.forModel(constructor, dbAdapterFlavour as string);\n        this.model = new constructor() as Model;\n        if(this.model && !this.pk)\n          this.pk = (this._repository as unknown as DecafRepository<Model>).pk || 'id';\n      }\n    } catch (error: unknown) {\n      throw new InternalError(\n        (error as Error)?.message || error as string\n      );\n    }\n    return this._repository;\n  }\n\n  /**\n   * @description Handles changes to component inputs\n   * @summary This Angular lifecycle hook is called when input properties change.\n   * It responds to changes in the model, locale, or translatable properties by\n   * updating the component's internal state accordingly. When the model changes,\n   * it calls getModel to process the new model and getLocale to update the locale.\n   * When locale or translatable properties change, it calls getLocale to update\n   * the translation settings.\n   *\n   * @param {SimpleChanges} changes - Object containing changed properties\n   * @return {void}\n   */\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes[BaseComponentProps.MODEL]) {\n      const { currentValue } = changes[BaseComponentProps.MODEL];\n      if (currentValue) this.getModel(currentValue);\n      this.getLocale(this.translatable);\n    }\n    if (\n      changes[BaseComponentProps.INITIALIZED] ||\n      changes[BaseComponentProps.LOCALE] ||\n      changes[BaseComponentProps.TRANSLATABLE]\n    )\n      this.getLocale(this.translatable);\n  }\n\n  /**\n   * @description Gets the appropriate locale string based on the translatable flag and available locales.\n   * @summary Determines which locale string to use for translation based on the translatable flag\n   * and available locale settings. This method first converts the translatable parameter to a boolean\n   * using the stringToBoolean utility function. If translatable is false, it returns an empty string,\n   * indicating no translation should be performed. If translatable is true, it checks for an explicitly\n   * provided locale via the locale property. If no explicit locale is available, it falls back to the\n   * componentLocale derived from the component's class name.\n   *\n   * @param {StringOrBoolean} translatable - Whether the component should be translated\n   * @return {string} The locale string to use for translation, or empty string if not translatable\n   *\n   * @mermaid\n   * sequenceDiagram\n   *   participant C as Component\n   *   participant N as NgxBaseComponent\n   *   participant S as StringUtils\n   *\n   *   C->>N: getLocale(translatable)\n   *   N->>S: stringToBoolean(translatable)\n   *   S-->>N: Return boolean value\n   *   N->>N: Store in this.translatable\n   *   alt translatable is false\n   *     N-->>C: Return empty string\n   *   else translatable is true\n   *     alt this.locale is defined\n   *       N-->>C: Return this.locale\n   *     else this.locale is not defined\n   *       N-->>C: Return this.componentLocale\n   *     end\n   *   end\n   */\n  getLocale(translatable: StringOrBoolean): string {\n    this.translatable = stringToBoolean(translatable);\n    if (!this.translatable)\n      return '';\n    if (!this.locale)\n      this.locale = this.componentLocale;\n    return this.locale;\n  }\n\n  /**\n   * @description Gets the route for the component\n   * @summary Retrieves the route path for the component, generating one based on the model\n   * if no route is explicitly set. This method checks if a route is already defined, and if not,\n   * it creates a default route based on the model's constructor name. The generated route follows\n   * the pattern '/model/{ModelName}'. This is useful for automatic routing in CRUD operations.\n   *\n   * @return {string} The route path for the component, or empty string if no route is available\n   */\n  getRoute(): string {\n    if (!this.route && this.model instanceof Model)\n      this.route = `/model/${this.model?.constructor.name}`;\n    return this.route || '';\n  }\n\n  /**\n   * @description Resolves and sets the component's model\n   * @summary Processes the provided model parameter, which can be either a Model instance\n   * or a string identifier. If a string is provided, it attempts to resolve the actual model\n   * from the injectables registry. After resolving the model, it calls setModelDefinitions\n   * to configure the component based on the model's metadata.\n   *\n   * @param {string | Model} model - The model instance or identifier string\n   * @return {void}\n   */\n  getModel(model: string | Model): void {\n    if (!(model instanceof Model))\n      this.model = getInjectablesRegistry().get(model) as Model;\n    this.setModelDefinitions(this.model as Model);\n  }\n\n  /**\n   * @description Configures component properties based on model metadata\n   * @summary Extracts and applies configuration from the model's decorators to set up\n   * the component. This method uses the rendering engine to retrieve decorator metadata\n   * from the model, then configures the component's mapper and item properties accordingly.\n   * It ensures the route is properly set and merges various properties from the model's\n   * metadata into the component's configuration.\n   *\n   * @param {Model} model - The model to extract configuration from\n   * @return {void}\n   */\n  setModelDefinitions(model: Model): void {\n    if (model instanceof Model) {\n      this.getRoute();\n      this.model = model;\n      const field = this.renderingEngine.getDecorators(this.model as Model, {});\n      const{ props, item, children } = field;\n      this.props = Object.assign(props || {}, {children: children || []});\n      if(item?.props?.['mapper'])\n        this.mapper = item?.props!['mapper'] || {};\n      this.item = {\n        tag: item?.tag || '',\n        ...item?.props,\n        ...(this.mapper ? {mapper: this.mapper} : {}),\n        ...{ route: item?.props?.['route'] || this.route },\n      };\n    }\n  }\n\n  /**\n   * @description Initializes the component\n   * @summary Performs one-time initialization of the component. This method checks if\n   * the component has already been initialized to prevent duplicate initialization.\n   * When called for the first time, it sets the initialized flag to true and logs\n   * an initialization message with the component name. This method is typically called\n   * during the component's lifecycle setup.\n   */\n  initialize(): void {\n    this.initialized = true;\n  }\n\n  /**\n   * @description Handles custom events from child components.\n   * @summary Receives events from child renderer components and forwards them to parent\n   * components through the listenEvent output. This creates an event propagation chain\n   * that allows events to bubble up through the component hierarchy, enabling coordinated\n   * responses to user interactions across the layout structure.\n   *\n   * @param {RendererCustomEvent} event - The custom event from a child component\n   * @return {void}\n   *\n   * @mermaid\n   * sequenceDiagram\n   *   participant C as Child Component\n   *   participant L as NgxBaseComponent\n   *   participant P as Parent Component\n   *\n   *   C->>L: Emit RendererCustomEvent\n   *   L->>L: handleEvent(event)\n   *   L->>P: listenEvent.emit(event)\n   *   Note over P: Handle event in parent\n   *\n   * @memberOf NgxBaseComponent\n   */\n  handleEvent(event: RendererCustomEvent): void {\n    this.listenEvent.emit(event);\n  }\n\n  /**\n   * @description Tracks items in ngFor loops for optimal change detection.\n   * @summary Provides a tracking function for Angular's *ngFor directive to optimize rendering\n   * performance. This method generates unique identifiers for list items based on their index\n   * and content, allowing Angular to efficiently track changes and minimize DOM manipulations\n   * during list updates. The tracking function is essential for maintaining component state\n   * and preventing unnecessary re-rendering of unchanged items.\n   *\n   * @param {number} index - The index of the item in the list\n   * @param {KeyValue | string | number} item - The item data to track\n   * @returns {string | number} A unique identifier for the item\n   * @memberOf NgxBaseComponent\n   */\n  trackItemFn(index: number, item: KeyValue | string | number): string | number {\n    return `${index}-${item}`;\n  }\n\n  /**\n   * @description Parses and applies properties from the props object to the component instance.\n   * @summary This method iterates through the properties of the provided instance object\n   * and applies any matching properties from the component's props configuration to the\n   * component instance. This allows for dynamic property assignment based on configuration\n   * stored in the props object, enabling flexible component customization without requiring\n   * explicit property binding for every possible configuration option.\n   *\n   * The method performs a safe property assignment by checking if each key from the instance\n   * exists in the props object before applying it. This prevents accidental property\n   * overwriting and ensures only intended properties are modified.\n   *\n   * @param {KeyValue} instance - The component instance object to process\n   * @return {void}\n   *\n   * @mermaid\n   * sequenceDiagram\n   *   participant C as Component\n   *   participant B as NgxBaseComponent\n   *   participant P as Props Object\n   *\n   *   C->>B: parseProps(instance)\n   *   B->>B: Get Object.keys(instance)\n   *   loop For each key in instance\n   *     B->>P: Check if key exists in this.props\n   *     alt Key exists in props\n   *       B->>B: Set this[key] = this.props[key]\n   *     else Key not in props\n   *       Note over B: Skip this key\n   *     end\n   *   end\n   *\n   * @protected\n   * @memberOf NgxBaseComponent\n   */\n  protected parseProps(instance: KeyValue): void {\n    Object.keys(instance).forEach((key) => {\n      if(Object.keys(this.props).includes(key))\n        (this as KeyValue)[key] = this.props[key];\n    })\n  }\n}\n"]}
541
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"NgxBaseComponent.js","sourceRoot":"","sources":["../../../../src/lib/engine/NgxBaseComponent.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  Input,\n  Component,\n  Inject,\n  ViewChild,\n  ElementRef,\n  OnChanges,\n  SimpleChanges,\n  Output,\n  EventEmitter,\n} from '@angular/core';\nimport { KeyValue, RendererCustomEvent, StringOrBoolean } from './types';\nimport {\n  generateRandomValue,\n  getInjectablesRegistry,\n  getOnWindow,\n  isDevelopmentMode,\n  stringToBoolean,\n} from '../helpers/utils';\nimport { getLocaleContext } from '../i18n/Loader';\nimport { Model } from '@decaf-ts/decorator-validation';\nimport {\n  CrudOperations,\n  InternalError,\n  OperationKeys,\n} from '@decaf-ts/db-decorators';\nimport { BaseComponentProps } from './constants';\nimport { NgxRenderingEngine } from './NgxRenderingEngine';\nimport { Logger } from '@decaf-ts/logging';\nimport { getLogger } from '../for-angular-common.module';\nimport { DecafRepository } from '../components/list/constants';\nimport { Repository } from '@decaf-ts/core';\nimport { RamAdapter } from '@decaf-ts/core/ram';\n\n/**\n * @description Base component class that provides common functionality for all Decaf components.\n * @summary The NgxBaseComponent serves as the foundation for all Decaf UI components, providing\n * shared functionality for localization, element references, and styling. This abstract class\n * implements common properties and methods that are used across the component library, ensuring\n * consistent behavior and reducing code duplication. Components that extend this class inherit\n * its capabilities for handling translations, accessing DOM elements, and applying custom styling.\n *\n * @template M - The model type that this component works with\n * @param {string} instance - The component instance token used for identification\n * @param {string} locale - The locale to be used for translations\n * @param {StringOrBoolean} translatable - Whether the component should be translated\n * @param {string} className - Additional CSS classes to apply to the component\n * @param {\"ios\" | \"md\" | undefined} mode - Component platform style\n *\n * @component NgxBaseComponent\n * @example\n * ```typescript\n * @Component({\n *   selector: 'app-my-component',\n *   templateUrl: './my-component.component.html',\n *   styleUrls: ['./my-component.component.scss']\n * })\n * export class MyComponent extends NgxBaseComponent {\n *   constructor(@Inject('instanceToken') instance: string) {\n *     super(instance);\n *   }\n *\n *   ngOnInit() {\n *     this.initialize();\n *     // Component-specific initialization\n *   }\n * }\n * ```\n * @mermaid\n * sequenceDiagram\n *   participant App as Application\n *   participant Comp as Component\n *   participant Base as NgxBaseComponent\n *   participant Engine as NgxRenderingEngine\n *\n *   App->>Comp: Create component\n *   Comp->>Base: super(instance)\n *   Base->>Base: Set componentName & componentLocale\n *\n *   App->>Comp: Set @Input properties\n *   Comp->>Base: ngOnChanges(changes)\n *\n *   alt model changed\n *     Base->>Base: getModel(model)\n *     Base->>Engine: getDecorators(model, {})\n *     Engine-->>Base: Return decorator metadata\n *     Base->>Base: Configure mapper and item\n *     Base->>Base: getLocale(translatable)\n *   else locale/translatable changed\n *     Base->>Base: getLocale(translatable)\n *   end\n *\n *   App->>Comp: ngOnInit()\n *   Comp->>Base: initialize()\n *   Base->>Base: Set initialized flag\n */\n@Component({\n  standalone: true,\n  template: '<div></div>',\n  host: { '[attr.id]': 'uid' },\n})\nexport abstract class NgxBaseComponent implements OnChanges {\n  /**\n   * @description Reference to the component's element.\n   * @summary Provides direct access to the native DOM element of the component through Angular's\n   * ViewChild decorator. This reference can be used to manipulate the DOM element directly,\n   * apply custom styles, or access native element properties and methods. The element is\n   * identified by the 'component' template reference variable.\n   *\n   * @type {ElementRef}\n   * @memberOf NgxBaseComponent\n   */\n  @ViewChild('component', { read: ElementRef, static: true })\n  component!: ElementRef;\n\n  /**\n   * @description The name of the component.\n   * @summary Stores the name of the component, which is typically derived from the class name.\n   * This property is used internally for various purposes such as logging, deriving the default\n   * locale, and potentially for component identification in debugging or error reporting.\n   *\n   * The `componentName` is set during the component's initialization process and should not\n   * be modified externally. It's marked as protected to allow access in derived classes while\n   * preventing direct access from outside the component hierarchy.\n   *\n   * @type {string}\n   * @protected\n   * @memberOf NgxBaseComponent\n   *\n   * @example\n   * // Inside a derived component class\n   * console.log(this.componentName); // Outputs: \"MyCustomComponent\"\n   */\n  componentName!: string;\n\n  /**\n   * @description Unique identifier for the renderer.\n   * @summary A unique identifier used to reference the component's renderer instance.\n   * This can be used for targeting specific renderer instances when multiple components\n   * are present on the same page.\n   *\n   * @type {string}\n   * @memberOf NgxBaseComponent\n   */\n  @Input()\n  rendererId!: string;\n\n  /**\n   * @description Repository model for data operations.\n   * @summary The data model repository that this component will use for CRUD operations.\n   * This provides a connection to the data layer for retrieving and manipulating data.\n   *\n   * @type {Model| undefined}\n   * @memberOf NgxBaseComponent\n   */\n  @Input()\n  model!: Model | undefined;\n\n  /**\n   * @description The repository for interacting with the data model.\n   * @summary Provides a connection to the data layer for retrieving and manipulating data.\n   * This is an instance of the `DecafRepository` class from the `@decaf-ts/core` package,\n   * which is initialized in the `repository` getter method.\n   *\n   * The repository is used to perform CRUD (Create, Read, Update, Delete) operations on the\n   * data model, such as fetching data, creating new items, updating existing items, and deleting\n   * items. It also provides methods for querying and filtering data based on specific criteria.\n   *\n   * @type {DecafRepository<Model>}\n   * @private\n   * @memberOf NgxBaseComponent\n   */\n  protected _repository?: DecafRepository<Model>;\n\n  /**\n   * @description Dynamic properties configuration object.\n   * @summary Contains key-value pairs of dynamic properties that can be applied to the component\n   * at runtime. This flexible configuration object allows for dynamic property assignment without\n   * requiring explicit input bindings for every possible configuration option. Properties from\n   * this object are parsed and applied to the component instance through the parseProps method,\n   * enabling customizable component behavior based on external configuration.\n   *\n   * @type {Record<string, unknown>}\n   * @default {}\n   * @memberOf NgxBaseComponent\n   */\n  @Input()\n  props: Record<string, unknown> = {};\n\n  /**\n   * @description Configuration for list item rendering\n   * @summary Defines how list items should be rendered in the component.\n   * This property holds a configuration object that specifies the tag name\n   * and other properties needed to render list items correctly. The tag property\n   * identifies which component should be used to render each item in a list.\n   * Additional properties can be included to customize the rendering behavior.\n   *\n   * @type {Record<string, unknown>}\n   * @default {tag: \"\"}\n   * @memberOf NgxBaseComponent\n   */\n  @Input()\n  item: Record<string, unknown> = { tag: '' };\n\n  /**\n   * @description Primary key field name for the model.\n   * @summary Specifies which field in the model should be used as the primary key.\n   * This is typically used for identifying unique records in operations like update and delete.\n   *\n   * @type {string}\n   * @default 'id'\n   * @memberOf NgxBaseComponent\n   */\n  @Input()\n  pk!: string;\n\n  /**\n   * @description Base route for navigation related to this component.\n   * @summary Defines the base route path used for navigation actions related to this component.\n   * This is often used as a prefix for constructing navigation URLs.\n   *\n   * @type {string}\n   * @memberOf NgxBaseComponent\n   */\n  @Input()\n  route!: string;\n\n  /**\n   * @description Available CRUD operations for this component.\n   * @summary Defines which CRUD operations (Create, Read, Update, Delete) are available\n   * for this component. This affects which operations can be performed on the data.\n   *\n   * @default [OperationKeys.READ]\n   * @memberOf NgxBaseComponent\n   */\n  @Input()\n  operations: CrudOperations[] = [OperationKeys.READ];\n\n  /**\n   * @description Unique identifier for the current record.\n   * @summary A unique identifier for the current record being displayed or manipulated.\n   * This is typically used in conjunction with the primary key for operations on specific records.\n   *\n   * @type {string | number}\n   * @memberOf NgxBaseComponent\n   */\n  @Input()\n  uid!: string | number;\n\n  /**\n   * @description Field mapping configuration.\n   * @summary Defines how fields from the data model should be mapped to properties used by the component.\n   * This allows for flexible data binding between the model and the component's display logic.\n   *\n   * @type {Record<string, string>}\n   * @memberOf NgxBaseComponent\n   */\n  @Input()\n  mapper: Record<string, string> = {};\n\n  /**\n   * @description The locale to be used for translations.\n   * @summary Specifies the locale identifier to use when translating component text.\n   * This can be set explicitly via input property to override the automatically derived\n   * locale from the component name. The locale is typically a language code (e.g., 'en', 'fr')\n   * or a language-region code (e.g., 'en-US', 'fr-CA') that determines which translation\n   * set to use for the component's text content.\n   *\n   * @type {string}\n   * @memberOf NgxBaseComponent\n   */\n  @Input()\n  locale!: string;\n\n  /**\n   * @description Determines if the component should be translated.\n   * @summary Controls whether the component's text content should be processed for translation.\n   * When true, the component will attempt to translate text using the specified locale.\n   * When false, text is displayed as-is without translation. This property accepts either\n   * a boolean value or a string that can be converted to a boolean (e.g., 'true', 'false', '1', '0').\n   *\n   * @type {StringOrBoolean}\n   * @default false\n   * @memberOf NgxBaseComponent\n   */\n  @Input()\n  translatable: StringOrBoolean = true;\n\n  /**\n   * @description Additional CSS class names to apply to the component.\n   * @summary Allows custom CSS classes to be added to the component's root element.\n   * These classes are appended to any automatically generated classes based on other\n   * component properties. Multiple classes can be provided as a space-separated string.\n   * This provides a way to customize the component's appearance beyond the built-in styling options.\n   *\n   * @type {string}\n   * @default \"\"\n   * @memberOf NgxBaseComponent\n   */\n  @Input()\n  className: string = '';\n\n  /**\n   * @description Component platform style.\n   * @summary Controls the visual appearance of the component based on platform design guidelines.\n   * The 'ios' mode follows iOS design patterns, while 'md' (Material Design) follows Android/Google\n   * design patterns. This property affects various visual aspects such as animations, form elements,\n   * and icons. Setting this property allows components to maintain platform-specific styling\n   * for a more native look and feel.\n   *\n   * @type {(\"ios\" | \"md\" | undefined)}\n   * @default \"md\"\n   * @memberOf NgxBaseComponent\n   */\n  @Input()\n  mode: 'ios' | 'md' | undefined = 'md';\n\n  /**\n   * @description The locale derived from the component's class name.\n   * @summary Stores the automatically derived locale based on the component's class name.\n   * This is determined during component initialization and serves as a fallback when no\n   * explicit locale is provided via the locale input property. The derivation is handled\n   * by the getLocaleContext utility function, which extracts a locale identifier\n   * from the component's class name.\n   *\n   * @type {string}\n   * @memberOf NgxBaseComponent\n   */\n  componentLocale!: string;\n\n  /**\n   * @description Controls whether child components should be rendered\n   * @summary Determines if child components should be rendered by the component.\n   * This can be set to a boolean value or a string that can be converted to a boolean.\n   * When true, child components defined in the model will be rendered. When false,\n   * child components will be skipped. This provides control over the rendering depth.\n   *\n   * @type {string | StringOrBoolean}\n   * @default true\n   * @memberOf NgxBaseComponent\n   */\n  @Input()\n  renderChild: string | StringOrBoolean = true;\n\n  /**\n   * @description Flag indicating if the component has been initialized\n   * @summary Tracks whether the component has completed its initialization process.\n   * This flag is used to prevent duplicate initialization and to determine if\n   * certain operations that require initialization can be performed.\n   *\n   * @type {boolean}\n   * @default false\n   */\n  initialized: boolean = false;\n\n  /**\n   * @description Event emitter for custom renderer events.\n   * @summary Emits custom events that occur within child components or the layout itself.\n   * This allows parent components to listen for and respond to user interactions or\n   * state changes within the grid layout. Events are passed up the component hierarchy\n   * to enable coordinated behavior across the application.\n   *\n   * @type {EventEmitter<RendererCustomEvent>}\n   * @memberOf NgxBaseComponent\n   */\n  @Output()\n  listenEvent: EventEmitter<RendererCustomEvent> =\n    new EventEmitter<RendererCustomEvent>();\n\n  /**\n   * @description Reference to the rendering engine instance\n   * @summary Provides access to the NgxRenderingEngine singleton instance,\n   * which handles the rendering of components based on model definitions.\n   * This engine is used to extract decorator metadata and render child components.\n   *\n   * @type {NgxRenderingEngine}\n   */\n  renderingEngine: NgxRenderingEngine =\n    NgxRenderingEngine.get() as unknown as NgxRenderingEngine;\n\n  /**\n   * @description Logger instance for the component.\n   * @summary Provides logging capabilities for the component, allowing for consistent\n   * and structured logging of information, warnings, and errors. This logger is initialized\n   * in the ngOnInit method using the getLogger function from the ForAngularCommonModule.\n   *\n   * The logger is used throughout the component to record important events, debug information,\n   * and potential issues. It helps in monitoring the component's behavior, tracking the flow\n   * of operations, and facilitating easier debugging and maintenance.\n   *\n   * @type {Logger}\n   * @private\n   * @memberOf NgxBaseComponent\n   */\n  logger!: Logger;\n\n  /**\n   * @description Creates an instance of NgxBaseComponent.\n   * @summary Initializes a new instance of the base component with the provided instance token.\n   * This constructor sets up the fundamental properties required by all Decaf components,\n   * including the component name, locale settings, and logging capabilities. The instance\n   * token is used for component identification and locale derivation.\n   *\n   * The constructor performs the following initialization steps:\n   * 1. Sets the componentName from the provided instance token\n   * 2. Derives the componentLocale from the class name using utility functions\n   * 3. Initializes the logger instance for the component\n   *\n   * @param {string} instance - The component instance token used for identification\n   *\n   * @mermaid\n   * sequenceDiagram\n   *   participant A as Angular\n   *   participant C as Component\n   *   participant B as NgxBaseComponent\n   *   participant U as Utils\n   *   participant L as Logger\n   *\n   *   A->>C: new Component(instance)\n   *   C->>B: super(instance)\n   *   B->>B: Set componentName = instance\n   *   B->>U: getLocaleContext(instance)\n   *   U-->>B: Return derived locale\n   *   B->>B: Set componentLocale\n   *   B->>L: getLogger(this)\n   *   L-->>B: Return logger instance\n   *   B->>B: Set logger\n   *\n   * @memberOf NgxBaseComponent\n   */\n  // eslint-disable-next-line @angular-eslint/prefer-inject\n  protected constructor(@Inject('instanceToken') protected instance: string) {\n    this.componentName = instance;\n    this.componentLocale = getLocaleContext(instance);\n    this.logger = getLogger(this);\n    this.getLocale(this.translatable);\n    this.uid = generateRandomValue(12);\n  }\n\n  /**\n   * @description Getter for the repository instance.\n   * @summary Provides a connection to the data layer for retrieving and manipulating data.\n   * This method initializes the `_repository` property if it is not already set, ensuring\n   * that a single instance of the repository is used throughout the component.\n   *\n   * The repository is used to perform CRUD operations on the data model, such as fetching data,\n   * creating new items, updating existing items, and deleting items. It also provides methods\n   * for querying and filtering data based on specific criteria.\n   *\n   * @returns {DecafRepository<Model>} The initialized repository instance.\n   * @private\n   * @memberOf NgxBaseComponent\n   */\n  protected get repository(): DecafRepository<Model> {\n    try {\n      if (!this._repository) {\n        const modelName = (this.model as Model).constructor.name;\n        const constructor = Model.get(modelName);\n        if (!constructor)\n          throw new InternalError(\n            'Cannot find model. was it registered with @model?'\n          );\n        let dbAdapterFlavour = getOnWindow('dbAdapterFlavour');\n        if (!dbAdapterFlavour && isDevelopmentMode()) {\n          const adapter = new RamAdapter({ user: 'user' });\n          dbAdapterFlavour = adapter.flavour;\n        }\n        this._repository = Repository.forModel(\n          constructor,\n          dbAdapterFlavour as string\n        );\n        this.model = new constructor() as Model;\n        if (this.model && !this.pk)\n          this.pk =\n            (this._repository as unknown as DecafRepository<Model>).pk || 'id';\n      }\n    } catch (error: unknown) {\n      throw new InternalError((error as Error)?.message || (error as string));\n    }\n    return this._repository;\n  }\n\n  /**\n   * @description Handles changes to component inputs\n   * @summary This Angular lifecycle hook is called when input properties change.\n   * It responds to changes in the model, locale, or translatable properties by\n   * updating the component's internal state accordingly. When the model changes,\n   * it calls getModel to process the new model and getLocale to update the locale.\n   * When locale or translatable properties change, it calls getLocale to update\n   * the translation settings.\n   *\n   * @param {SimpleChanges} changes - Object containing changed properties\n   * @return {void}\n   */\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes[BaseComponentProps.MODEL]) {\n      const { currentValue } = changes[BaseComponentProps.MODEL];\n      if (currentValue) this.getModel(currentValue);\n      this.getLocale(this.translatable);\n    }\n    if (\n      changes[BaseComponentProps.INITIALIZED] ||\n      changes[BaseComponentProps.LOCALE] ||\n      changes[BaseComponentProps.TRANSLATABLE]\n    )\n      this.getLocale(this.translatable);\n  }\n\n  /**\n   * @description Gets the appropriate locale string based on the translatable flag and available locales.\n   * @summary Determines which locale string to use for translation based on the translatable flag\n   * and available locale settings. This method first converts the translatable parameter to a boolean\n   * using the stringToBoolean utility function. If translatable is false, it returns an empty string,\n   * indicating no translation should be performed. If translatable is true, it checks for an explicitly\n   * provided locale via the locale property. If no explicit locale is available, it falls back to the\n   * componentLocale derived from the component's class name.\n   *\n   * @param {StringOrBoolean} translatable - Whether the component should be translated\n   * @return {string} The locale string to use for translation, or empty string if not translatable\n   *\n   * @mermaid\n   * sequenceDiagram\n   *   participant C as Component\n   *   participant N as NgxBaseComponent\n   *   participant S as StringUtils\n   *\n   *   C->>N: getLocale(translatable)\n   *   N->>S: stringToBoolean(translatable)\n   *   S-->>N: Return boolean value\n   *   N->>N: Store in this.translatable\n   *   alt translatable is false\n   *     N-->>C: Return empty string\n   *   else translatable is true\n   *     alt this.locale is defined\n   *       N-->>C: Return this.locale\n   *     else this.locale is not defined\n   *       N-->>C: Return this.componentLocale\n   *     end\n   *   end\n   */\n  getLocale(translatable: StringOrBoolean): string {\n    this.translatable = stringToBoolean(translatable);\n    if (!this.translatable) return '';\n    if (!this.locale) this.locale = this.componentLocale;\n    return this.locale;\n  }\n\n  /**\n   * @description Gets the route for the component\n   * @summary Retrieves the route path for the component, generating one based on the model\n   * if no route is explicitly set. This method checks if a route is already defined, and if not,\n   * it creates a default route based on the model's constructor name. The generated route follows\n   * the pattern '/model/{ModelName}'. This is useful for automatic routing in CRUD operations.\n   *\n   * @return {string} The route path for the component, or empty string if no route is available\n   */\n  getRoute(): string {\n    if (!this.route && this.model instanceof Model)\n      this.route = `/model/${this.model?.constructor.name}`;\n    return this.route || '';\n  }\n\n  /**\n   * @description Resolves and sets the component's model\n   * @summary Processes the provided model parameter, which can be either a Model instance\n   * or a string identifier. If a string is provided, it attempts to resolve the actual model\n   * from the injectables registry. After resolving the model, it calls setModelDefinitions\n   * to configure the component based on the model's metadata.\n   *\n   * @param {string | Model} model - The model instance or identifier string\n   * @return {void}\n   */\n  getModel(model: string | Model): void {\n    if (!(model instanceof Model))\n      this.model = getInjectablesRegistry().get(model) as Model;\n    this.setModelDefinitions(this.model as Model);\n  }\n\n  /**\n   * @description Configures component properties based on model metadata\n   * @summary Extracts and applies configuration from the model's decorators to set up\n   * the component. This method uses the rendering engine to retrieve decorator metadata\n   * from the model, then configures the component's mapper and item properties accordingly.\n   * It ensures the route is properly set and merges various properties from the model's\n   * metadata into the component's configuration.\n   *\n   * @param {Model} model - The model to extract configuration from\n   * @return {void}\n   */\n  setModelDefinitions(model: Model): void {\n    if (model instanceof Model) {\n      this.getRoute();\n      this.model = model;\n      const field = this.renderingEngine.getDecorators(this.model as Model, {});\n      const { props, item, children } = field;\n      this.props = Object.assign(props || {}, { children: children || [] });\n      if (item?.props?.['mapper']) this.mapper = item?.props!['mapper'] || {};\n      this.item = {\n        tag: item?.tag || '',\n        ...item?.props,\n        ...(this.mapper ? { mapper: this.mapper } : {}),\n        ...{ route: item?.props?.['route'] || this.route },\n      };\n    }\n  }\n\n  /**\n   * @description Initializes the component\n   * @summary Performs one-time initialization of the component. This method checks if\n   * the component has already been initialized to prevent duplicate initialization.\n   * When called for the first time, it sets the initialized flag to true and logs\n   * an initialization message with the component name. This method is typically called\n   * during the component's lifecycle setup.\n   */\n  initialize(): void {\n    this.initialized = true;\n  }\n\n  /**\n   * @description Handles custom events from child components.\n   * @summary Receives events from child renderer components and forwards them to parent\n   * components through the listenEvent output. This creates an event propagation chain\n   * that allows events to bubble up through the component hierarchy, enabling coordinated\n   * responses to user interactions across the layout structure.\n   *\n   * @param {RendererCustomEvent} event - The custom event from a child component\n   * @return {void}\n   *\n   * @mermaid\n   * sequenceDiagram\n   *   participant C as Child Component\n   *   participant L as NgxBaseComponent\n   *   participant P as Parent Component\n   *\n   *   C->>L: Emit RendererCustomEvent\n   *   L->>L: handleEvent(event)\n   *   L->>P: listenEvent.emit(event)\n   *   Note over P: Handle event in parent\n   *\n   * @memberOf NgxBaseComponent\n   */\n  handleEvent(event: RendererCustomEvent): void {\n    this.listenEvent.emit(event);\n  }\n\n  /**\n   * @description Tracks items in ngFor loops for optimal change detection.\n   * @summary Provides a tracking function for Angular's *ngFor directive to optimize rendering\n   * performance. This method generates unique identifiers for list items based on their index\n   * and content, allowing Angular to efficiently track changes and minimize DOM manipulations\n   * during list updates. The tracking function is essential for maintaining component state\n   * and preventing unnecessary re-rendering of unchanged items.\n   *\n   * @param {number} index - The index of the item in the list\n   * @param {KeyValue | string | number} item - The item data to track\n   * @returns {string | number} A unique identifier for the item\n   * @memberOf NgxBaseComponent\n   */\n  trackItemFn(\n    index: number,\n    item: KeyValue | string | number\n  ): string | number {\n    return `${index}-${item}`;\n  }\n\n  /**\n   * @description Parses and applies properties from the props object to the component instance.\n   * @summary This method iterates through the properties of the provided instance object\n   * and applies any matching properties from the component's props configuration to the\n   * component instance. This allows for dynamic property assignment based on configuration\n   * stored in the props object, enabling flexible component customization without requiring\n   * explicit property binding for every possible configuration option.\n   *\n   * The method performs a safe property assignment by checking if each key from the instance\n   * exists in the props object before applying it. This prevents accidental property\n   * overwriting and ensures only intended properties are modified.\n   *\n   * @param {KeyValue} instance - The component instance object to process\n   * @return {void}\n   *\n   * @mermaid\n   * sequenceDiagram\n   *   participant C as Component\n   *   participant B as NgxBaseComponent\n   *   participant P as Props Object\n   *\n   *   C->>B: parseProps(instance)\n   *   B->>B: Get Object.keys(instance)\n   *   loop For each key in instance\n   *     B->>P: Check if key exists in this.props\n   *     alt Key exists in props\n   *       B->>B: Set this[key] = this.props[key]\n   *     else Key not in props\n   *       Note over B: Skip this key\n   *     end\n   *   end\n   *\n   * @protected\n   * @memberOf NgxBaseComponent\n   */\n  protected parseProps(instance: KeyValue): void {\n    Object.keys(instance).forEach((key) => {\n      if (Object.keys(this.props).includes(key))\n        (this as KeyValue)[key] = this.props[key];\n    });\n  }\n}\n"]}
@@ -215,7 +215,10 @@ export class NgxFormService {
215
215
  const controlValue = cleanSpaces(`${formGroup.get(pk)?.value}`, true);
216
216
  if (operation === OperationKeys.CREATE)
217
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));
218
+ return !formGroupArray.controls.some((group, i) => {
219
+ const value = cleanSpaces(`${group.get(pk)?.value}`, true);
220
+ return i !== index && controlValue === value;
221
+ });
219
222
  }
220
223
  /**
221
224
  * @description Enables all controls within a FormGroup or FormArray.
@@ -416,7 +419,13 @@ export class NgxFormService {
416
419
  });
417
420
  }
418
421
  }
419
- return control.valid;
422
+ function getControlName(control) {
423
+ const group = control.parent;
424
+ if (!group)
425
+ return null;
426
+ return Object.keys(group.controls).find(name => control === group.get(name)) || null;
427
+ }
428
+ return !getControlName(control) ? true : control.valid;
420
429
  }
421
430
  /**
422
431
  * @description Generates validators from component properties.
@@ -524,4 +533,4 @@ export class NgxFormService {
524
533
  }
525
534
  }
526
535
  }
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"]}
536
+ //# 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      const value = cleanSpaces(`${group.get(pk)?.value}`, true);\n      return i !== index && controlValue === value;\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    function getControlName(control: AbstractControl): string | null {\n      const group = control.parent as FormGroup;\n      if (!group)\n          return null;\n      return Object.keys(group.controls).find(name => control === group.get(name)) || null;\n    }\n\n    return !getControlName(control) ? true : 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"]}