@decaf-ts/for-angular 0.0.23 → 0.0.25

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 (36) hide show
  1. package/components/component-renderer/component-renderer.component.d.ts +3 -2
  2. package/components/crud-field/crud-field.component.d.ts +4 -2
  3. package/components/fieldset/fieldset.component.d.ts +10 -1
  4. package/components/for-angular-components.module.d.ts +3 -2
  5. package/components/index.d.ts +1 -0
  6. package/components/layout/layout.component.d.ts +1 -24
  7. package/components/model-renderer/model-renderer.component.d.ts +6 -1
  8. package/components/stepped-form/stepped-form.component.d.ts +255 -0
  9. package/engine/NgxBaseComponent.d.ts +2 -2
  10. package/engine/NgxCrudFormField.d.ts +1 -0
  11. package/engine/NgxFormService.d.ts +381 -48
  12. package/engine/NgxRenderingEngine.d.ts +4 -2
  13. package/engine/interfaces.d.ts +1 -1
  14. package/engine/types.d.ts +4 -3
  15. package/esm2022/components/component-renderer/component-renderer.component.mjs +10 -4
  16. package/esm2022/components/crud-field/crud-field.component.mjs +14 -3
  17. package/esm2022/components/crud-form/crud-form.component.mjs +3 -3
  18. package/esm2022/components/empty-state/empty-state.component.mjs +2 -2
  19. package/esm2022/components/fieldset/fieldset.component.mjs +5 -3
  20. package/esm2022/components/for-angular-components.module.mjs +12 -7
  21. package/esm2022/components/index.mjs +2 -1
  22. package/esm2022/components/layout/layout.component.mjs +4 -29
  23. package/esm2022/components/list/list.component.mjs +3 -3
  24. package/esm2022/components/model-renderer/model-renderer.component.mjs +10 -3
  25. package/esm2022/components/stepped-form/stepped-form.component.mjs +306 -0
  26. package/esm2022/engine/NgxBaseComponent.mjs +10 -4
  27. package/esm2022/engine/NgxCrudFormField.mjs +19 -17
  28. package/esm2022/engine/NgxFormService.mjs +438 -57
  29. package/esm2022/engine/NgxRenderingEngine.mjs +21 -10
  30. package/esm2022/engine/ValidatorFactory.mjs +4 -4
  31. package/esm2022/engine/interfaces.mjs +1 -1
  32. package/esm2022/engine/types.mjs +1 -1
  33. package/esm2022/i18n/data/en.json +5 -0
  34. package/fesm2022/decaf-ts-for-angular.mjs +837 -137
  35. package/fesm2022/decaf-ts-for-angular.mjs.map +1 -1
  36. package/package.json +1 -1
@@ -38,6 +38,11 @@ export class ModelRendererComponent {
38
38
  * @description Global variables to be passed to the rendered component
39
39
  */
40
40
  this.globals = {};
41
+ /**
42
+ * @description Set if render content projection is allowed
43
+ * @default true
44
+ */
45
+ this.projectable = true;
41
46
  /**
42
47
  * @description Event emitter for custom events from the rendered component
43
48
  */
@@ -55,7 +60,7 @@ export class ModelRendererComponent {
55
60
  typeof model === 'string'
56
61
  ? Model.build({}, model)
57
62
  : model;
58
- this.output = model.render(this.globals || {}, this.vcr, this.injector, this.inner);
63
+ this.output = model.render(this.globals || {}, this.vcr, this.injector, this.inner, this.projectable);
59
64
  if (this.output?.inputs)
60
65
  this.rendererId = sf(AngularEngineKeys.RENDERED_ID, this.output.inputs['rendererId']);
61
66
  this.instance = this.output?.instance;
@@ -113,7 +118,7 @@ export class ModelRendererComponent {
113
118
  }
114
119
  }
115
120
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: ModelRendererComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
116
- static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.14", type: ModelRendererComponent, isStandalone: true, selector: "ngx-decaf-model-renderer", inputs: { model: "model", globals: "globals", rendererId: "rendererId" }, outputs: { listenEvent: "listenEvent" }, host: { properties: { "attr.id": "rendererId" } }, viewQueries: [{ propertyName: "inner", first: true, predicate: ["inner"], descendants: true, read: TemplateRef, static: true }, { propertyName: "vcr", first: true, predicate: ["componentOuter"], descendants: true, read: ViewContainerRef, static: true }], usesOnChanges: true, ngImport: i0, template: " <!-- Keep to avoid id conflicts -->\n <div [id]=\"rendererId\"></div>\n\n <ng-template #componentOuter></ng-template>\n <ng-template #inner>\n <div [id]=\"rendererId || null\">\n @for (child of output?.children; track child) {\n @if(child?.children?.length) {\n <ngx-decaf-component-renderer [parent]=\"child\" />\n } @else {\n <ng-container\n #childComponents\n *ngComponentOutlet=\"\n child.component;\n injector: child.injector;\n inputs: child.inputs;\n content:child.content;\n \"\n />\n }\n }\n </div>\n </ng-template>\n", styles: [""], dependencies: [{ kind: "directive", type: NgComponentOutlet, selector: "[ngComponentOutlet]", inputs: ["ngComponentOutlet", "ngComponentOutletInputs", "ngComponentOutletInjector", "ngComponentOutletContent", "ngComponentOutletNgModule", "ngComponentOutletNgModuleFactory"] }, { kind: "component", type: ComponentRendererComponent, selector: "ngx-decaf-component-renderer", inputs: ["tag", "globals", "model", "parent"], outputs: ["listenEvent"] }] }); }
121
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.14", type: ModelRendererComponent, isStandalone: true, selector: "ngx-decaf-model-renderer", inputs: { model: "model", globals: "globals", projectable: "projectable", rendererId: "rendererId" }, outputs: { listenEvent: "listenEvent" }, host: { properties: { "attr.id": "rendererId" } }, viewQueries: [{ propertyName: "inner", first: true, predicate: ["inner"], descendants: true, read: TemplateRef, static: true }, { propertyName: "vcr", first: true, predicate: ["componentOuter"], descendants: true, read: ViewContainerRef, static: true }], usesOnChanges: true, ngImport: i0, template: " <!-- Keep to avoid id conflicts -->\n <div [id]=\"rendererId\"></div>\n\n <ng-template #componentOuter></ng-template>\n <ng-template #inner>\n <div [id]=\"rendererId || null\">\n @for (child of output?.children; track child) {\n @if(child?.children?.length) {\n <ngx-decaf-component-renderer [parent]=\"child\" />\n } @else {\n <ng-container\n #childComponents\n *ngComponentOutlet=\"\n child.component;\n injector: child.injector;\n inputs: child.inputs;\n content:child.content;\n \"\n />\n }\n }\n </div>\n </ng-template>\n", styles: [""], dependencies: [{ kind: "directive", type: NgComponentOutlet, selector: "[ngComponentOutlet]", inputs: ["ngComponentOutlet", "ngComponentOutletInputs", "ngComponentOutletInjector", "ngComponentOutletContent", "ngComponentOutletNgModule", "ngComponentOutletNgModuleFactory"] }, { kind: "component", type: ComponentRendererComponent, selector: "ngx-decaf-component-renderer", inputs: ["tag", "globals", "children", "model", "parent"], outputs: ["listenEvent"] }] }); }
117
122
  }
118
123
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: ModelRendererComponent, decorators: [{
119
124
  type: Component,
@@ -123,6 +128,8 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
123
128
  args: [{ required: true }]
124
129
  }], globals: [{
125
130
  type: Input
131
+ }], projectable: [{
132
+ type: Input
126
133
  }], inner: [{
127
134
  type: ViewChild,
128
135
  args: ['inner', { read: TemplateRef, static: true }]
@@ -134,4 +141,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
134
141
  }], listenEvent: [{
135
142
  type: Output
136
143
  }] } });
137
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"model-renderer.component.js","sourceRoot":"","sources":["../../../../../src/lib/components/model-renderer/model-renderer.component.ts","../../../../../src/lib/components/model-renderer/model-renderer.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component,\n  EventEmitter,\n  inject,\n  Injector,\n  Input,\n  OnChanges,\n  OnDestroy,\n  Output,\n  SimpleChanges,\n  TemplateRef,\n  ViewChild,\n  ViewContainerRef,\n} from '@angular/core';\nimport { Model, sf } from '@decaf-ts/decorator-validation';\nimport { NgComponentOutlet } from '@angular/common';\nimport {\n  AngularDynamicOutput,\n  AngularEngineKeys,\n  BaseComponentProps,\n  BaseCustomEvent,\n  NgxRenderingEngine,\n  RenderedModel,\n} from '../../engine';\nimport { KeyValue, RendererCustomEvent } from '../../engine/types';\nimport { Renderable } from '@decaf-ts/ui-decorators';\nimport { ComponentRendererComponent } from '../component-renderer/component-renderer.component';\n\n/**\n * @description Component for rendering dynamic models\n * @summary This component is responsible for dynamically rendering models,\n * handling model changes, and managing event subscriptions for the rendered components.\n * It uses the NgxRenderingEngine to render the models and supports both string and Model inputs.\n * @class\n * @template M - Type extending Model\n * @param {Injector} injector - Angular Injector for dependency injection\n * @example\n * <ngx-decaf-model-renderer\n *   [model]=\"myModel\"\n *   [globals]=\"globalVariables\"\n *   (listenEvent)=\"handleEvent($event)\">\n * </ngx-decaf-model-renderer>\n * @mermaid\n * sequenceDiagram\n *   participant App\n *   participant ModelRenderer\n *   participant RenderingEngine\n *   participant Model\n *   App->>ModelRenderer: Input model\n *   ModelRenderer->>Model: Parse if string\n *   Model-->>ModelRenderer: Parsed model\n *   ModelRenderer->>RenderingEngine: Render model\n *   RenderingEngine-->>ModelRenderer: Rendered output\n *   ModelRenderer->>ModelRenderer: Subscribe to events\n *   ModelRenderer-->>App: Emit events\n */\n@Component({\n  standalone: true,\n  imports: [ NgComponentOutlet, ComponentRendererComponent],\n  selector: 'ngx-decaf-model-renderer',\n  templateUrl: './model-renderer.component.html',\n  styleUrl: './model-renderer.component.scss',\n  host: {'[attr.id]': 'rendererId'},\n})\nexport class ModelRendererComponent<M extends Model>\n  implements OnChanges, OnDestroy, RenderedModel {\n\n  /**\n   * @description Input model to be rendered\n   * @summary Can be a Model instance or a JSON string representation of a model\n   */\n  @Input({ required: true })\n  model!: M | string | undefined;\n\n  /**\n   * @description Global variables to be passed to the rendered component\n   */\n  @Input()\n  globals: Record<string, unknown> = {};\n\n  /**\n   * @description Template reference for inner content\n   */\n  @ViewChild('inner', { read: TemplateRef, static: true })\n  inner?: TemplateRef<unknown>;\n\n  /**\n   * @description Output of the rendered model\n   */\n  output?: AngularDynamicOutput;\n\n  /**\n   * @description Unique identifier for the renderer\n   */\n  @Input()\n  rendererId?: string;\n\n  /**\n   * @description View container reference for dynamic component rendering\n   */\n  @ViewChild('componentOuter', { static: true, read: ViewContainerRef })\n  vcr!: ViewContainerRef;\n\n  /**\n   * @description Event emitter for custom events from the rendered component\n   */\n  @Output()\n  listenEvent = new EventEmitter<RendererCustomEvent>();\n\n  /**\n   * @description Instance of the rendered component\n   */\n  private instance!: KeyValue | undefined;\n\n  private injector: Injector = inject(Injector);\n\n  // constructor() {}\n\n  /**\n   * @description Refreshes the rendered model\n   * @param {string | M} model - The model to be rendered\n   */\n  private refresh(model: string | M) {\n    model =\n      typeof model === 'string'\n        ? (Model.build({}, model) as M)\n        : model;\n    this.output = (model as unknown as Renderable).render<AngularDynamicOutput>(\n      this.globals || {},\n      this.vcr,\n      this.injector,\n      this.inner,\n    );\n    if (this.output?.inputs)\n      this.rendererId = sf(\n        AngularEngineKeys.RENDERED_ID,\n        (this.output.inputs as Record<string, unknown>)['rendererId'] as string,\n      );\n    this.instance = this.output?.instance;\n    this.subscribeEvents();\n  }\n\n  /**\n   * @description Lifecycle hook that is called when data-bound properties of a directive change\n   * @param {SimpleChanges} changes - Object containing changes\n   */\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes[BaseComponentProps.MODEL]) {\n      const { currentValue } = changes[BaseComponentProps.MODEL];\n      this.refresh(currentValue);\n    }\n  }\n\n  /**\n   * @description Lifecycle hook that is called when a directive, pipe, or service is destroyed\n   * @return {Promise<void>}\n   */\n  async ngOnDestroy(): Promise<void> {\n    if (this.instance) {\n      this.unsubscribeEvents();\n      await NgxRenderingEngine.destroy();\n    }\n    this.output = undefined;\n  }\n\n  private subscribeEvents(): void {\n    const component = this?.output?.component;\n    if (this.instance && component) {\n      const componentKeys = Object.keys(this.instance);\n      for (const key of componentKeys) {\n        const value = this.instance[key];\n        if (value instanceof EventEmitter)\n          (this.instance as KeyValue)[key].subscribe((event: Partial<BaseCustomEvent>) => {\n            this.listenEvent.emit({\n              component: component.name || '',\n              name: key,\n              ...event,\n            } as RendererCustomEvent);\n          });\n      }\n    }\n  }\n\n  /**\n   * @description Unsubscribes from events emitted by the rendered component\n   */\n  private unsubscribeEvents(): void {\n    if (this.instance) {\n      const componentKeys = Object.keys(this.instance);\n      for (const key of componentKeys) {\n        const value = this.instance[key];\n        if (value instanceof EventEmitter)\n          this.instance[key].unsubscribe();\n      }\n    }\n  }\n\n  protected readonly JSON = JSON;\n}\n","  <!-- Keep to avoid id conflicts -->\n  <div [id]=\"rendererId\"></div>\n\n  <ng-template #componentOuter></ng-template>\n  <ng-template #inner>\n    <div  [id]=\"rendererId || null\">\n      @for (child of output?.children; track child) {\n        @if(child?.children?.length) {\n          <ngx-decaf-component-renderer [parent]=\"child\" />\n        } @else {\n          <ng-container\n            #childComponents\n            *ngComponentOutlet=\"\n              child.component;\n              injector: child.injector;\n              inputs: child.inputs;\n              content:child.content;\n            \"\n          />\n        }\n      }\n    </div>\n  </ng-template>\n"]}
144
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"model-renderer.component.js","sourceRoot":"","sources":["../../../../../src/lib/components/model-renderer/model-renderer.component.ts","../../../../../src/lib/components/model-renderer/model-renderer.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component,\n  EventEmitter,\n  inject,\n  Injector,\n  Input,\n  OnChanges,\n  OnDestroy,\n  Output,\n  SimpleChanges,\n  TemplateRef,\n  ViewChild,\n  ViewContainerRef,\n} from '@angular/core';\nimport { Model, sf } from '@decaf-ts/decorator-validation';\nimport { NgComponentOutlet } from '@angular/common';\nimport {\n  AngularDynamicOutput,\n  AngularEngineKeys,\n  BaseComponentProps,\n  BaseCustomEvent,\n  NgxRenderingEngine,\n  RenderedModel,\n} from '../../engine';\nimport { KeyValue, RendererCustomEvent } from '../../engine/types';\nimport { Renderable } from '@decaf-ts/ui-decorators';\nimport { ComponentRendererComponent } from '../component-renderer/component-renderer.component';\n\n/**\n * @description Component for rendering dynamic models\n * @summary This component is responsible for dynamically rendering models,\n * handling model changes, and managing event subscriptions for the rendered components.\n * It uses the NgxRenderingEngine to render the models and supports both string and Model inputs.\n * @class\n * @template M - Type extending Model\n * @param {Injector} injector - Angular Injector for dependency injection\n * @example\n * <ngx-decaf-model-renderer\n *   [model]=\"myModel\"\n *   [globals]=\"globalVariables\"\n *   (listenEvent)=\"handleEvent($event)\">\n * </ngx-decaf-model-renderer>\n * @mermaid\n * sequenceDiagram\n *   participant App\n *   participant ModelRenderer\n *   participant RenderingEngine\n *   participant Model\n *   App->>ModelRenderer: Input model\n *   ModelRenderer->>Model: Parse if string\n *   Model-->>ModelRenderer: Parsed model\n *   ModelRenderer->>RenderingEngine: Render model\n *   RenderingEngine-->>ModelRenderer: Rendered output\n *   ModelRenderer->>ModelRenderer: Subscribe to events\n *   ModelRenderer-->>App: Emit events\n */\n@Component({\n  standalone: true,\n  imports: [ NgComponentOutlet, ComponentRendererComponent],\n  selector: 'ngx-decaf-model-renderer',\n  templateUrl: './model-renderer.component.html',\n  styleUrl: './model-renderer.component.scss',\n  host: {'[attr.id]': 'rendererId'},\n})\nexport class ModelRendererComponent<M extends Model>\n  implements OnChanges, OnDestroy, RenderedModel {\n\n  /**\n   * @description Input model to be rendered\n   * @summary Can be a Model instance or a JSON string representation of a model\n   */\n  @Input({ required: true })\n  model!: M | string | undefined;\n\n  /**\n   * @description Global variables to be passed to the rendered component\n   */\n  @Input()\n  globals: Record<string, unknown> = {};\n\n  /**\n   * @description Set if render content projection is allowed\n   * @default true\n   */\n  @Input()\n  projectable: boolean = true;\n\n  /**\n   * @description Template reference for inner content\n   */\n  @ViewChild('inner', { read: TemplateRef, static: true })\n  inner?: TemplateRef<unknown>;\n\n  /**\n   * @description Output of the rendered model\n   */\n  output?: AngularDynamicOutput;\n\n  /**\n   * @description Unique identifier for the renderer\n   */\n  @Input()\n  rendererId?: string;\n\n  /**\n   * @description View container reference for dynamic component rendering\n   */\n  @ViewChild('componentOuter', { static: true, read: ViewContainerRef })\n  vcr!: ViewContainerRef;\n\n  /**\n   * @description Event emitter for custom events from the rendered component\n   */\n  @Output()\n  listenEvent = new EventEmitter<RendererCustomEvent>();\n\n  /**\n   * @description Instance of the rendered component\n   */\n  private instance!: KeyValue | undefined;\n\n  private injector: Injector = inject(Injector);\n\n  // constructor() {}\n\n  /**\n   * @description Refreshes the rendered model\n   * @param {string | M} model - The model to be rendered\n   */\n  private refresh(model: string | M) {\n    model =\n      typeof model === 'string'\n        ? (Model.build({}, model) as M)\n        : model;\n    this.output = (model as unknown as Renderable).render<AngularDynamicOutput>(\n      this.globals || {},\n      this.vcr,\n      this.injector,\n      this.inner,\n      this.projectable\n    );\n    if (this.output?.inputs)\n      this.rendererId = sf(\n        AngularEngineKeys.RENDERED_ID,\n        (this.output.inputs as Record<string, unknown>)['rendererId'] as string,\n      );\n    this.instance = this.output?.instance;\n    this.subscribeEvents();\n  }\n\n  /**\n   * @description Lifecycle hook that is called when data-bound properties of a directive change\n   * @param {SimpleChanges} changes - Object containing changes\n   */\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes[BaseComponentProps.MODEL]) {\n      const { currentValue } = changes[BaseComponentProps.MODEL];\n      this.refresh(currentValue);\n    }\n  }\n\n  /**\n   * @description Lifecycle hook that is called when a directive, pipe, or service is destroyed\n   * @return {Promise<void>}\n   */\n  async ngOnDestroy(): Promise<void> {\n    if (this.instance) {\n      this.unsubscribeEvents();\n      await NgxRenderingEngine.destroy();\n    }\n    this.output = undefined;\n  }\n\n  private subscribeEvents(): void {\n    const component = this?.output?.component;\n    if (this.instance && component) {\n      const componentKeys = Object.keys(this.instance);\n      for (const key of componentKeys) {\n        const value = this.instance[key];\n        if (value instanceof EventEmitter)\n          (this.instance as KeyValue)[key].subscribe((event: Partial<BaseCustomEvent>) => {\n            this.listenEvent.emit({\n              component: component.name || '',\n              name: key,\n              ...event,\n            } as RendererCustomEvent);\n          });\n      }\n    }\n  }\n\n  /**\n   * @description Unsubscribes from events emitted by the rendered component\n   */\n  private unsubscribeEvents(): void {\n    if (this.instance) {\n      const componentKeys = Object.keys(this.instance);\n      for (const key of componentKeys) {\n        const value = this.instance[key];\n        if (value instanceof EventEmitter)\n          this.instance[key].unsubscribe();\n      }\n    }\n  }\n\n  protected readonly JSON = JSON;\n}\n","  <!-- Keep to avoid id conflicts -->\n  <div [id]=\"rendererId\"></div>\n\n  <ng-template #componentOuter></ng-template>\n  <ng-template #inner>\n    <div  [id]=\"rendererId || null\">\n      @for (child of output?.children; track child) {\n        @if(child?.children?.length) {\n          <ngx-decaf-component-renderer [parent]=\"child\" />\n        } @else {\n          <ng-container\n            #childComponents\n            *ngComponentOutlet=\"\n              child.component;\n              injector: child.injector;\n              inputs: child.inputs;\n              content:child.content;\n            \"\n          />\n        }\n      }\n    </div>\n  </ng-template>\n"]}
@@ -0,0 +1,306 @@
1
+ import { __decorate, __metadata } from "tslib";
2
+ import { Component, Input, Output, EventEmitter } from '@angular/core';
3
+ import { ReactiveFormsModule } from '@angular/forms';
4
+ import { IonButton, IonIcon, IonSkeletonText, IonText } from '@ionic/angular/standalone';
5
+ import { arrowForwardOutline, arrowBackOutline } from 'ionicons/icons';
6
+ import { addIcons } from 'ionicons';
7
+ import { OperationKeys } from '@decaf-ts/db-decorators';
8
+ import { Dynamic, EventConstants, NgxFormService } from '../../engine';
9
+ import { ComponentRendererComponent } from '../component-renderer/component-renderer.component';
10
+ import { ModelRendererComponent } from '../model-renderer/model-renderer.component';
11
+ import { timer } from 'rxjs';
12
+ import { getLocaleContext } from '../../i18n/Loader';
13
+ import { TranslatePipe } from '@ngx-translate/core';
14
+ import * as i0 from "@angular/core";
15
+ import * as i1 from "@angular/forms";
16
+ let SteppedFormComponent = class SteppedFormComponent {
17
+ /**
18
+ * @description Creates an instance of SteppedFormComponent.
19
+ * @summary Initializes a new SteppedFormComponent instance and registers the required
20
+ * Ionic icons for navigation buttons (forward and back arrows).
21
+ *
22
+ * @memberOf SteppedFormComponent
23
+ */
24
+ constructor() {
25
+ /**
26
+ * @description Number of pages in the stepped form.
27
+ * @summary Represents the total number of steps/pages in the multi-step form.
28
+ * This value is automatically calculated based on the page properties of the children
29
+ * or can be explicitly set. Each page represents a logical group of form fields.
30
+ *
31
+ * @type {number}
32
+ * @default 1
33
+ * @memberOf SteppedFormComponent
34
+ */
35
+ this.pages = 1;
36
+ /**
37
+ * @description The CRUD operation type for this form.
38
+ * @summary Defines the type of operation being performed (CREATE, READ, UPDATE, DELETE).
39
+ * This affects form behavior, validation rules, and field accessibility. For example,
40
+ * READ operations might disable form fields, while CREATE operations enable all fields.
41
+ *
42
+ * @type {CrudOperations}
43
+ * @default OperationKeys.CREATE
44
+ * @memberOf SteppedFormComponent
45
+ */
46
+ this.operation = OperationKeys.CREATE;
47
+ /**
48
+ * @description The initial page to display when the form loads.
49
+ * @summary Specifies which page of the multi-step form should be shown first.
50
+ * This allows starting the form at any step, useful for scenarios like editing
51
+ * existing data where you might want to jump to a specific section.
52
+ *
53
+ * @type {number}
54
+ * @default 1
55
+ * @memberOf SteppedFormComponent
56
+ */
57
+ this.startPage = 1;
58
+ /**
59
+ * @description Array of UI model metadata for the currently active page.
60
+ * @summary Contains only the UI model metadata for fields that should be displayed
61
+ * on the currently active page. This is a filtered subset of the children array,
62
+ * updated whenever the user navigates between pages.
63
+ *
64
+ * @type {UIModelMetadata[] | undefined}
65
+ * @memberOf SteppedFormComponent
66
+ */
67
+ this.activeChildren = undefined;
68
+ /**
69
+ * @description FormGroup for the currently active page.
70
+ * @summary The FormGroup instance that manages form controls and validation
71
+ * for the current page only. This is extracted from the main formGroup
72
+ * when using FormArray structure.
73
+ *
74
+ * @type {FormGroup | undefined}
75
+ * @memberOf SteppedFormComponent
76
+ */
77
+ this.activeFormGroup = undefined;
78
+ /**
79
+ * @description The currently active page number.
80
+ * @summary Tracks which page of the multi-step form is currently being displayed.
81
+ * This property is updated as users navigate through the form steps using
82
+ * the next/back buttons or programmatic navigation.
83
+ *
84
+ * @type {number}
85
+ * @memberOf SteppedFormComponent
86
+ */
87
+ this.activePage = 1;
88
+ /**
89
+ * @description Array representing the structure of pages.
90
+ * @summary Contains metadata about each page, including page numbers and indices.
91
+ * This array is built during initialization to organize the form fields into
92
+ * logical pages and provide navigation structure.
93
+ *
94
+ * @type {UIModelMetadata[]}
95
+ * @memberOf SteppedFormComponent
96
+ */
97
+ this.pagesArray = [];
98
+ /**
99
+ * @description Event emitter for form submission.
100
+ * @summary Emits events when the form is submitted, typically on the last page
101
+ * when all validation passes. The emitted event contains the form data and
102
+ * event type information for parent components to handle.
103
+ *
104
+ * @type {EventEmitter<BaseCustomEvent>}
105
+ * @memberOf SteppedFormComponent
106
+ */
107
+ this.submitEvent = new EventEmitter();
108
+ addIcons({ arrowForwardOutline, arrowBackOutline });
109
+ }
110
+ /**
111
+ * @description Initializes the component after Angular first displays the data-bound properties.
112
+ * @summary Sets up the stepped form by organizing children into pages, calculating the total
113
+ * number of pages, and initializing the active page. This method processes the UI model metadata
114
+ * to create a logical page structure and ensures proper page assignments for all form fields.
115
+ *
116
+ * @mermaid
117
+ * sequenceDiagram
118
+ * participant A as Angular Lifecycle
119
+ * participant S as SteppedFormComponent
120
+ * participant F as Form Service
121
+ *
122
+ * A->>S: ngOnInit()
123
+ * S->>S: Set activePage = startPage
124
+ * S->>S: Process children into pagesArray
125
+ * S->>S: Calculate total pages
126
+ * S->>S: Assign page props to children
127
+ * S->>S: getCurrentFormGroup(activePage)
128
+ * S->>F: Extract FormGroup for active page
129
+ * F-->>S: Return activeFormGroup
130
+ *
131
+ * @memberOf SteppedFormComponent
132
+ */
133
+ ngOnInit() {
134
+ if (!this.locale)
135
+ this.locale = getLocaleContext("SteppedFormComponent");
136
+ this.activePage = this.startPage;
137
+ this.pagesArray = this.children.reduce((acc, curr, index) => {
138
+ const page = curr.props?.['page'] || index + 1;
139
+ if (!acc[page])
140
+ acc[page] = [];
141
+ acc[page].push({ index: page });
142
+ return acc;
143
+ }, []).filter(Boolean);
144
+ this.pages = this.pagesArray.length;
145
+ this.children = [...this.children.map((c) => {
146
+ if (!c.props)
147
+ c.props = {};
148
+ const page = c.props['page'] || 1;
149
+ // prevent page overflow
150
+ c.props['page'] = page > this.pages ? this.pages : page;
151
+ return c;
152
+ })];
153
+ this.getCurrentFormGroup(this.activePage);
154
+ }
155
+ /**
156
+ * @description Cleanup method called when the component is destroyed.
157
+ * @summary Unsubscribes from any active timer subscriptions to prevent memory leaks.
158
+ * This is part of Angular's component lifecycle and ensures proper resource cleanup.
159
+ *
160
+ * @memberOf SteppedFormComponent
161
+ */
162
+ ngOnDestroy() {
163
+ if (this.timerSubscription)
164
+ this.timerSubscription.unsubscribe();
165
+ }
166
+ /**
167
+ * @description Handles navigation to the next page or form submission.
168
+ * @summary Validates the current page's form fields and either navigates to the next page
169
+ * or submits the entire form if on the last page. Form validation must pass before
170
+ * proceeding. On successful submission, emits a submit event with form data.
171
+ *
172
+ * @param {boolean} lastPage - Whether this is the last page of the form
173
+ * @return {void}
174
+ *
175
+ * @mermaid
176
+ * sequenceDiagram
177
+ * participant U as User
178
+ * participant S as SteppedFormComponent
179
+ * participant F as Form Service
180
+ * participant P as Parent Component
181
+ *
182
+ * U->>S: Click Next/Submit
183
+ * S->>F: validateFields(activeFormGroup)
184
+ * F-->>S: Return validation result
185
+ * alt Not last page and valid
186
+ * S->>S: activePage++
187
+ * S->>S: getCurrentFormGroup(activePage)
188
+ * else Last page and valid
189
+ * S->>F: getFormData(formGroup)
190
+ * F-->>S: Return form data
191
+ * S->>P: submitEvent.emit({data, name: SUBMIT})
192
+ * end
193
+ *
194
+ * @memberOf SteppedFormComponent
195
+ */
196
+ handleNext(lastPage = false) {
197
+ const isValid = NgxFormService.validateFields(this.activeFormGroup);
198
+ if (!lastPage) {
199
+ if (isValid) {
200
+ this.activePage = this.activePage + 1;
201
+ this.getCurrentFormGroup(this.activePage);
202
+ }
203
+ }
204
+ else {
205
+ if (isValid) {
206
+ const data = Object.assign({}, ...Object.values(NgxFormService.getFormData(this.formGroup)));
207
+ this.submitEvent.emit({
208
+ data,
209
+ name: EventConstants.SUBMIT,
210
+ });
211
+ }
212
+ }
213
+ }
214
+ /**
215
+ * @description Handles navigation to the previous page.
216
+ * @summary Moves the user back to the previous page in the stepped form.
217
+ * This method decrements the active page number and updates the form
218
+ * group and children to display the previous page's content.
219
+ *
220
+ * @return {void}
221
+ *
222
+ * @mermaid
223
+ * sequenceDiagram
224
+ * participant U as User
225
+ * participant S as SteppedFormComponent
226
+ *
227
+ * U->>S: Click Back
228
+ * S->>S: activePage--
229
+ * S->>S: getCurrentFormGroup(activePage)
230
+ * Note over S: Update active form and children
231
+ *
232
+ * @memberOf SteppedFormComponent
233
+ */
234
+ handleBack() {
235
+ this.activePage = this.activePage - 1;
236
+ this.getCurrentFormGroup(this.activePage);
237
+ }
238
+ /**
239
+ * @description Updates the active form group and children for the specified page.
240
+ * @summary Extracts the FormGroup for the given page from the FormArray and filters
241
+ * the children to show only fields belonging to that page. Uses a timer to ensure
242
+ * proper Angular change detection when updating the activeChildren.
243
+ *
244
+ * @param {number} page - The page number to activate
245
+ * @return {void}
246
+ *
247
+ * @private
248
+ * @mermaid
249
+ * sequenceDiagram
250
+ * participant S as SteppedFormComponent
251
+ * participant F as FormArray
252
+ * participant T as Timer
253
+ *
254
+ * S->>F: Extract FormGroup at index (page - 1)
255
+ * F-->>S: Return page FormGroup
256
+ * S->>S: Set activeChildren = undefined
257
+ * S->>T: timer(10).subscribe()
258
+ * T-->>S: Filter children for active page
259
+ * S->>S: Set activeChildren
260
+ *
261
+ * @memberOf SteppedFormComponent
262
+ */
263
+ getCurrentFormGroup(page) {
264
+ this.activeFormGroup = this.formGroup.at(page - 1);
265
+ this.activeChildren = undefined;
266
+ this.timerSubscription = timer(10).subscribe(() => {
267
+ this.activeChildren = this.children.filter(c => c.props?.['page'] === page);
268
+ console.log(this.activeChildren);
269
+ });
270
+ }
271
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: SteppedFormComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
272
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.14", type: SteppedFormComponent, isStandalone: true, selector: "ngx-decaf-stepped-form", inputs: { locale: "locale", pages: "pages", operation: "operation", startPage: "startPage", children: "children", formGroup: "formGroup" }, outputs: { submitEvent: "submitEvent" }, ngImport: i0, template: "<form class=\"dcf-steped-form\" novalidate>\n <div class=\"dcf-page-steps\">\n <div>\n @for(page of pagesArray; track $index;) {\n <div [class.dcf-active]=\"activePage === $index + 1\" [class.dcf-passed]=\"($index + 1) < activePage\">{{ $index + 1 }}</div>\n }\n </div>\n </div>\n @if(formGroup) {\n @for(child of activeChildren; track $index) {\n <ngx-decaf-component-renderer\n [tag]=\"child?.tag\"\n (listenEvent)=\"handleEvent($event)\"\n [children]=\"child?.children || []\"\n [globals]=\"{props: child.props}\"\n />\n }\n } @else {\n <ion-skeleton-text [animated]=\"true\"></ion-skeleton-text>\n <ion-text class=\"date\" style=\"width: 20%;\"><ion-skeleton-text [animated]=\"true\"></ion-skeleton-text></ion-text>\n <br />\n <ion-skeleton-text [animated]=\"true\"></ion-skeleton-text>\n <ion-text class=\"date\" style=\"width: 20%;\"><ion-skeleton-text [animated]=\"true\"></ion-skeleton-text></ion-text>\n }\n\n <div class=\"dcf-buttons-container dcf-grid dcf-grid-collapse dcf-flex dcf-flex-left\">\n <div class=\"dcf-width-1-2@s\">\n <ion-button fill=\"clear\" (click)=\"handleBack()\" [disabled]=\"activePage <= 1\">\n <ion-icon aria-hidden=\"true\" name=\"arrow-back-outline\"></ion-icon>\n {{locale + '.previous' | translate}}\n </ion-button>\n </div>\n\n <div class=\"dcf-width-1-2@s\">\n <ion-button [fill]=\"activePage === pages ? 'solid' : 'outline'\" (click)=\"handleNext(activePage === pages ? true : false)\">\n @if(activePage === pages) {\n {{locale + '.submit' | translate}}\n } @else {\n {{locale + '.next' | translate}}\n <ion-icon aria-hidden=\"true\" name=\"arrow-forward-outline\"></ion-icon>\n }\n </ion-button>\n </div>\n </div>\n</form>\n", styles: [".dcf-buttons-container{margin-top:1.8rem;margin-bottom:0}@media (min-width: 639px){.dcf-buttons-container.dcf-flex div:nth-child(2){display:flex;justify-content:flex-end}}@media (max-width: 638px){.dcf-buttons-container.dcf-flex div{width:100%}.dcf-buttons-container.dcf-flex ion-button{width:100%;margin-bottom:1rem}}.dcf-steped-form{padding:2rem 1rem}.dcf-page-steps{display:flex;justify-content:center;margin-bottom:2rem}.dcf-page-steps>div{justify-content:center;min-width:200px;max-width:200px;column-gap:.25em;display:flex}.dcf-page-steps>div>div{width:30px;text-align:center;border-bottom:solid var(--dcf-color-gray-3);box-sizing:border-box;border-width:3px;font-size:0px;font-weight:600}.dcf-page-steps>div>div.dcf-active{border-width:5px;font-size:1rem;color:var(--ion-color-gray-7);border-color:var(--ion-color-primary);line-height:1rem}.dcf-page-steps>div>div.dcf-passed{border-width:4px;font-size:.5rem;line-height:1.2rem;border-color:var(--dcf-color-gray-5);color:var(--ion-color-primary)}\n"], dependencies: [{ kind: "pipe", type: TranslatePipe, name: "translate" }, { kind: "ngmodule", type: ReactiveFormsModule }, { kind: "directive", type: i1.ɵNgNoValidate, selector: "form:not([ngNoForm]):not([ngNativeValidate])" }, { kind: "directive", type: i1.NgControlStatusGroup, selector: "[formGroupName],[formArrayName],[ngModelGroup],[formGroup],form:not([ngNoForm]),[ngForm]" }, { kind: "component", type: IonSkeletonText, selector: "ion-skeleton-text", inputs: ["animated"] }, { kind: "component", type: IonText, selector: "ion-text", inputs: ["color", "mode"] }, { kind: "component", type: IonButton, selector: "ion-button", inputs: ["buttonType", "color", "disabled", "download", "expand", "fill", "form", "href", "mode", "rel", "routerAnimation", "routerDirection", "shape", "size", "strong", "target", "type"] }, { kind: "component", type: IonIcon, selector: "ion-icon", inputs: ["color", "flipRtl", "icon", "ios", "lazy", "md", "mode", "name", "sanitize", "size", "src"] }, { kind: "component", type: ComponentRendererComponent, selector: "ngx-decaf-component-renderer", inputs: ["tag", "globals", "children", "model", "parent"], outputs: ["listenEvent"] }] }); }
273
+ };
274
+ SteppedFormComponent = __decorate([
275
+ Dynamic(),
276
+ __metadata("design:paramtypes", [])
277
+ ], SteppedFormComponent);
278
+ export { SteppedFormComponent };
279
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: SteppedFormComponent, decorators: [{
280
+ type: Component,
281
+ args: [{ selector: 'ngx-decaf-stepped-form', imports: [
282
+ TranslatePipe,
283
+ ReactiveFormsModule,
284
+ IonSkeletonText,
285
+ IonText,
286
+ IonButton,
287
+ IonIcon,
288
+ ModelRendererComponent,
289
+ ComponentRendererComponent
290
+ ], standalone: true, template: "<form class=\"dcf-steped-form\" novalidate>\n <div class=\"dcf-page-steps\">\n <div>\n @for(page of pagesArray; track $index;) {\n <div [class.dcf-active]=\"activePage === $index + 1\" [class.dcf-passed]=\"($index + 1) < activePage\">{{ $index + 1 }}</div>\n }\n </div>\n </div>\n @if(formGroup) {\n @for(child of activeChildren; track $index) {\n <ngx-decaf-component-renderer\n [tag]=\"child?.tag\"\n (listenEvent)=\"handleEvent($event)\"\n [children]=\"child?.children || []\"\n [globals]=\"{props: child.props}\"\n />\n }\n } @else {\n <ion-skeleton-text [animated]=\"true\"></ion-skeleton-text>\n <ion-text class=\"date\" style=\"width: 20%;\"><ion-skeleton-text [animated]=\"true\"></ion-skeleton-text></ion-text>\n <br />\n <ion-skeleton-text [animated]=\"true\"></ion-skeleton-text>\n <ion-text class=\"date\" style=\"width: 20%;\"><ion-skeleton-text [animated]=\"true\"></ion-skeleton-text></ion-text>\n }\n\n <div class=\"dcf-buttons-container dcf-grid dcf-grid-collapse dcf-flex dcf-flex-left\">\n <div class=\"dcf-width-1-2@s\">\n <ion-button fill=\"clear\" (click)=\"handleBack()\" [disabled]=\"activePage <= 1\">\n <ion-icon aria-hidden=\"true\" name=\"arrow-back-outline\"></ion-icon>\n {{locale + '.previous' | translate}}\n </ion-button>\n </div>\n\n <div class=\"dcf-width-1-2@s\">\n <ion-button [fill]=\"activePage === pages ? 'solid' : 'outline'\" (click)=\"handleNext(activePage === pages ? true : false)\">\n @if(activePage === pages) {\n {{locale + '.submit' | translate}}\n } @else {\n {{locale + '.next' | translate}}\n <ion-icon aria-hidden=\"true\" name=\"arrow-forward-outline\"></ion-icon>\n }\n </ion-button>\n </div>\n </div>\n</form>\n", styles: [".dcf-buttons-container{margin-top:1.8rem;margin-bottom:0}@media (min-width: 639px){.dcf-buttons-container.dcf-flex div:nth-child(2){display:flex;justify-content:flex-end}}@media (max-width: 638px){.dcf-buttons-container.dcf-flex div{width:100%}.dcf-buttons-container.dcf-flex ion-button{width:100%;margin-bottom:1rem}}.dcf-steped-form{padding:2rem 1rem}.dcf-page-steps{display:flex;justify-content:center;margin-bottom:2rem}.dcf-page-steps>div{justify-content:center;min-width:200px;max-width:200px;column-gap:.25em;display:flex}.dcf-page-steps>div>div{width:30px;text-align:center;border-bottom:solid var(--dcf-color-gray-3);box-sizing:border-box;border-width:3px;font-size:0px;font-weight:600}.dcf-page-steps>div>div.dcf-active{border-width:5px;font-size:1rem;color:var(--ion-color-gray-7);border-color:var(--ion-color-primary);line-height:1rem}.dcf-page-steps>div>div.dcf-passed{border-width:4px;font-size:.5rem;line-height:1.2rem;border-color:var(--dcf-color-gray-5);color:var(--ion-color-primary)}\n"] }]
291
+ }], ctorParameters: () => [], propDecorators: { locale: [{
292
+ type: Input
293
+ }], pages: [{
294
+ type: Input
295
+ }], operation: [{
296
+ type: Input
297
+ }], startPage: [{
298
+ type: Input
299
+ }], children: [{
300
+ type: Input
301
+ }], formGroup: [{
302
+ type: Input
303
+ }], submitEvent: [{
304
+ type: Output
305
+ }] } });
306
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"stepped-form.component.js","sourceRoot":"","sources":["../../../../../src/lib/components/stepped-form/stepped-form.component.ts","../../../../../src/lib/components/stepped-form/stepped-form.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, Input, OnInit, OnDestroy, Output, EventEmitter  } from '@angular/core';\nimport { FormArray, FormGroup, ReactiveFormsModule } from '@angular/forms';\nimport { IonButton, IonIcon, IonSkeletonText, IonText } from '@ionic/angular/standalone';\nimport { arrowForwardOutline, arrowBackOutline } from 'ionicons/icons';\nimport { addIcons } from 'ionicons';\nimport { UIModelMetadata} from '@decaf-ts/ui-decorators';\nimport { CrudOperations, OperationKeys } from '@decaf-ts/db-decorators';\nimport { BaseCustomEvent, Dynamic, EventConstants, KeyValue, NgxFormService } from '../../engine';\nimport { ComponentRendererComponent } from '../component-renderer/component-renderer.component';\nimport { ModelRendererComponent } from '../model-renderer/model-renderer.component';\nimport { Subscription, timer } from 'rxjs';\nimport { getLocaleContext } from '../../i18n/Loader';\nimport { TranslatePipe } from '@ngx-translate/core';\n\n\n@Dynamic()\n@Component({\n  selector: 'ngx-decaf-stepped-form',\n  templateUrl: './stepped-form.component.html',\n  styleUrls: ['./stepped-form.component.scss'],\n  imports: [\n    TranslatePipe,\n    ReactiveFormsModule,\n    IonSkeletonText,\n    IonText,\n    IonButton,\n    IonIcon,\n    ModelRendererComponent,\n    ComponentRendererComponent\n  ],\n  standalone: true,\n})\nexport class SteppedFormComponent implements OnInit, OnDestroy {\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 SteppedFormComponent\n   */\n  @Input()\n  locale!: string;\n\n\n  /**\n   * @description Number of pages in the stepped form.\n   * @summary Represents the total number of steps/pages in the multi-step form.\n   * This value is automatically calculated based on the page properties of the children\n   * or can be explicitly set. Each page represents a logical group of form fields.\n   *\n   * @type {number}\n   * @default 1\n   * @memberOf SteppedFormComponent\n   */\n  @Input()\n  pages: number = 1;\n\n  /**\n   * @description The CRUD operation type for this form.\n   * @summary Defines the type of operation being performed (CREATE, READ, UPDATE, DELETE).\n   * This affects form behavior, validation rules, and field accessibility. For example,\n   * READ operations might disable form fields, while CREATE operations enable all fields.\n   *\n   * @type {CrudOperations}\n   * @default OperationKeys.CREATE\n   * @memberOf SteppedFormComponent\n   */\n  @Input()\n  operation: CrudOperations = OperationKeys.CREATE;\n\n  /**\n   * @description The initial page to display when the form loads.\n   * @summary Specifies which page of the multi-step form should be shown first.\n   * This allows starting the form at any step, useful for scenarios like editing\n   * existing data where you might want to jump to a specific section.\n   *\n   * @type {number}\n   * @default 1\n   * @memberOf SteppedFormComponent\n   */\n  @Input()\n  startPage: number = 1;\n\n  /**\n   * @description Array of UI model metadata for all form fields.\n   * @summary Contains the complete collection of UI model metadata that defines\n   * the structure, validation, and presentation of form fields across all pages.\n   * Each metadata object contains information about field type, validation rules,\n   * page assignment, and display properties.\n   *\n   * @type {UIModelMetadata[]}\n   * @memberOf SteppedFormComponent\n   */\n  @Input()\n  children!: UIModelMetadata[];\n\n  /**\n   * @description Angular reactive FormGroup or FormArray for form state management.\n   * @summary The form instance that manages all form controls, validation, and form state.\n   * When using FormArray, each array element represents a page's FormGroup. When using\n   * FormGroup, it contains all form controls for the entire stepped form.\n   *\n   * @type {FormGroup | FormArray | undefined}\n   * @memberOf SteppedFormComponent\n   */\n  @Input()\n  formGroup!: FormGroup | FormArray | undefined;\n\n  /**\n   * @description Array of UI model metadata for the currently active page.\n   * @summary Contains only the UI model metadata for fields that should be displayed\n   * on the currently active page. This is a filtered subset of the children array,\n   * updated whenever the user navigates between pages.\n   *\n   * @type {UIModelMetadata[] | undefined}\n   * @memberOf SteppedFormComponent\n   */\n  activeChildren: UIModelMetadata[] | undefined = undefined;\n\n  /**\n   * @description FormGroup for the currently active page.\n   * @summary The FormGroup instance that manages form controls and validation\n   * for the current page only. This is extracted from the main formGroup\n   * when using FormArray structure.\n   *\n   * @type {FormGroup | undefined}\n   * @memberOf SteppedFormComponent\n   */\n  activeFormGroup: FormGroup | undefined = undefined;\n\n  /**\n   * @description The currently active page number.\n   * @summary Tracks which page of the multi-step form is currently being displayed.\n   * This property is updated as users navigate through the form steps using\n   * the next/back buttons or programmatic navigation.\n   *\n   * @type {number}\n   * @memberOf SteppedFormComponent\n   */\n  activePage: number = 1;\n\n  /**\n   * @description Array representing the structure of pages.\n   * @summary Contains metadata about each page, including page numbers and indices.\n   * This array is built during initialization to organize the form fields into\n   * logical pages and provide navigation structure.\n   *\n   * @type {UIModelMetadata[]}\n   * @memberOf SteppedFormComponent\n   */\n  pagesArray: UIModelMetadata[] = [];\n\n  /**\n   * @description Subscription for timer-based operations.\n   * @summary Manages the timer subscription used for asynchronous operations\n   * like updating active children after page transitions. This subscription\n   * is cleaned up in ngOnDestroy to prevent memory leaks.\n   *\n   * @private\n   * @type {Subscription}\n   * @memberOf SteppedFormComponent\n   */\n  private timerSubscription!: Subscription;\n\n  /**\n   * @description Event emitter for form submission.\n   * @summary Emits events when the form is submitted, typically on the last page\n   * when all validation passes. The emitted event contains the form data and\n   * event type information for parent components to handle.\n   *\n   * @type {EventEmitter<BaseCustomEvent>}\n   * @memberOf SteppedFormComponent\n   */\n  @Output()\n  submitEvent: EventEmitter<BaseCustomEvent> = new EventEmitter<BaseCustomEvent>();\n\n  /**\n   * @description Creates an instance of SteppedFormComponent.\n   * @summary Initializes a new SteppedFormComponent instance and registers the required\n   * Ionic icons for navigation buttons (forward and back arrows).\n   *\n   * @memberOf SteppedFormComponent\n   */\n  constructor() {\n    addIcons({arrowForwardOutline, arrowBackOutline});\n  }\n\n  /**\n   * @description Initializes the component after Angular first displays the data-bound properties.\n   * @summary Sets up the stepped form by organizing children into pages, calculating the total\n   * number of pages, and initializing the active page. This method processes the UI model metadata\n   * to create a logical page structure and ensures proper page assignments for all form fields.\n   *\n   * @mermaid\n   * sequenceDiagram\n   *   participant A as Angular Lifecycle\n   *   participant S as SteppedFormComponent\n   *   participant F as Form Service\n   *\n   *   A->>S: ngOnInit()\n   *   S->>S: Set activePage = startPage\n   *   S->>S: Process children into pagesArray\n   *   S->>S: Calculate total pages\n   *   S->>S: Assign page props to children\n   *   S->>S: getCurrentFormGroup(activePage)\n   *   S->>F: Extract FormGroup for active page\n   *   F-->>S: Return activeFormGroup\n   *\n   * @memberOf SteppedFormComponent\n   */\n  ngOnInit(): void  {\n    if(!this.locale)\n      this.locale = getLocaleContext(\"SteppedFormComponent\")\n    this.activePage = this.startPage;\n\n    this.pagesArray = (this.children.reduce((acc, curr, index) => {\n      const page = curr.props?.['page'] || index + 1;\n      if(!acc[page])\n        acc[page] = [];\n      acc[page].push({index: page});\n      return acc;\n    }, [] as KeyValue) as []).filter(Boolean);\n\n    this.pages = this.pagesArray.length;\n    this.children = [... this.children.map((c) => {\n      if(!c.props)\n        c.props = {};\n      const page = c.props['page'] || 1;\n      // prevent page overflow\n      c.props['page'] = page > this.pages ? this.pages : page;\n      return c;\n    })];\n    this.getCurrentFormGroup(this.activePage);\n  }\n\n  /**\n   * @description Cleanup method called when the component is destroyed.\n   * @summary Unsubscribes from any active timer subscriptions to prevent memory leaks.\n   * This is part of Angular's component lifecycle and ensures proper resource cleanup.\n   *\n   * @memberOf SteppedFormComponent\n   */\n  ngOnDestroy(): void {\n    if(this.timerSubscription)\n      this.timerSubscription.unsubscribe();\n  }\n\n  /**\n   * @description Handles navigation to the next page or form submission.\n   * @summary Validates the current page's form fields and either navigates to the next page\n   * or submits the entire form if on the last page. Form validation must pass before\n   * proceeding. On successful submission, emits a submit event with form data.\n   *\n   * @param {boolean} lastPage - Whether this is the last page of the form\n   * @return {void}\n   *\n   * @mermaid\n   * sequenceDiagram\n   *   participant U as User\n   *   participant S as SteppedFormComponent\n   *   participant F as Form Service\n   *   participant P as Parent Component\n   *\n   *   U->>S: Click Next/Submit\n   *   S->>F: validateFields(activeFormGroup)\n   *   F-->>S: Return validation result\n   *   alt Not last page and valid\n   *     S->>S: activePage++\n   *     S->>S: getCurrentFormGroup(activePage)\n   *   else Last page and valid\n   *     S->>F: getFormData(formGroup)\n   *     F-->>S: Return form data\n   *     S->>P: submitEvent.emit({data, name: SUBMIT})\n   *   end\n   *\n   * @memberOf SteppedFormComponent\n   */\n  handleNext(lastPage: boolean = false): void {\n    const isValid = NgxFormService.validateFields(this.activeFormGroup as FormGroup);\n    if (!lastPage) {\n      if (isValid) {\n        this.activePage = this.activePage + 1;\n        this.getCurrentFormGroup(this.activePage);\n      }\n    } else {\n     if (isValid) {\n      const data = Object.assign({}, ...Object.values(NgxFormService.getFormData(this.formGroup as FormGroup)));\n      this.submitEvent.emit({\n        data,\n        name: EventConstants.SUBMIT,\n      });\n     }\n    }\n  }\n\n  /**\n   * @description Handles navigation to the previous page.\n   * @summary Moves the user back to the previous page in the stepped form.\n   * This method decrements the active page number and updates the form\n   * group and children to display the previous page's content.\n   *\n   * @return {void}\n   *\n   * @mermaid\n   * sequenceDiagram\n   *   participant U as User\n   *   participant S as SteppedFormComponent\n   *\n   *   U->>S: Click Back\n   *   S->>S: activePage--\n   *   S->>S: getCurrentFormGroup(activePage)\n   *   Note over S: Update active form and children\n   *\n   * @memberOf SteppedFormComponent\n   */\n  handleBack(): void {\n    this.activePage = this.activePage - 1;\n    this.getCurrentFormGroup(this.activePage);\n  }\n\n  /**\n   * @description Updates the active form group and children for the specified page.\n   * @summary Extracts the FormGroup for the given page from the FormArray and filters\n   * the children to show only fields belonging to that page. Uses a timer to ensure\n   * proper Angular change detection when updating the activeChildren.\n   *\n   * @param {number} page - The page number to activate\n   * @return {void}\n   *\n   * @private\n   * @mermaid\n   * sequenceDiagram\n   *   participant S as SteppedFormComponent\n   *   participant F as FormArray\n   *   participant T as Timer\n   *\n   *   S->>F: Extract FormGroup at index (page - 1)\n   *   F-->>S: Return page FormGroup\n   *   S->>S: Set activeChildren = undefined\n   *   S->>T: timer(10).subscribe()\n   *   T-->>S: Filter children for active page\n   *   S->>S: Set activeChildren\n   *\n   * @memberOf SteppedFormComponent\n   */\n  private getCurrentFormGroup(page: number): void {\n    this.activeFormGroup = (this.formGroup as FormArray).at(page - 1) as FormGroup;\n    this.activeChildren = undefined;\n    this.timerSubscription = timer(10).subscribe(() => {\n      this.activeChildren = this.children.filter(c => c.props?.['page'] === page);\n      console.log(this.activeChildren);\n    });\n  }\n\n}\n","<form class=\"dcf-steped-form\" novalidate>\n  <div class=\"dcf-page-steps\">\n    <div>\n        @for(page of pagesArray; track $index;) {\n        <div [class.dcf-active]=\"activePage === $index + 1\" [class.dcf-passed]=\"($index + 1) < activePage\">{{ $index + 1 }}</div>\n      }\n    </div>\n  </div>\n  @if(formGroup) {\n    @for(child of activeChildren;  track $index) {\n      <ngx-decaf-component-renderer\n        [tag]=\"child?.tag\"\n        (listenEvent)=\"handleEvent($event)\"\n        [children]=\"child?.children || []\"\n        [globals]=\"{props: child.props}\"\n      />\n    }\n  } @else {\n    <ion-skeleton-text [animated]=\"true\"></ion-skeleton-text>\n    <ion-text class=\"date\" style=\"width: 20%;\"><ion-skeleton-text [animated]=\"true\"></ion-skeleton-text></ion-text>\n    <br />\n    <ion-skeleton-text [animated]=\"true\"></ion-skeleton-text>\n    <ion-text class=\"date\" style=\"width: 20%;\"><ion-skeleton-text [animated]=\"true\"></ion-skeleton-text></ion-text>\n  }\n\n  <div class=\"dcf-buttons-container dcf-grid dcf-grid-collapse dcf-flex dcf-flex-left\">\n    <div class=\"dcf-width-1-2@s\">\n      <ion-button fill=\"clear\" (click)=\"handleBack()\" [disabled]=\"activePage <= 1\">\n        <ion-icon aria-hidden=\"true\" name=\"arrow-back-outline\"></ion-icon>\n          {{locale + '.previous' | translate}}\n      </ion-button>\n    </div>\n\n    <div class=\"dcf-width-1-2@s\">\n      <ion-button [fill]=\"activePage === pages ? 'solid' : 'outline'\"  (click)=\"handleNext(activePage === pages ? true : false)\">\n        @if(activePage === pages) {\n          {{locale + '.submit' | translate}}\n        } @else {\n          {{locale + '.next' | translate}}\n          <ion-icon aria-hidden=\"true\" name=\"arrow-forward-outline\"></ion-icon>\n        }\n      </ion-button>\n    </div>\n  </div>\n</form>\n"]}