@decaf-ts/for-angular 0.0.6 → 0.0.8

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -1,6 +1,7 @@
1
1
  import { ComponentRef, EnvironmentInjector, EventEmitter, OnDestroy, OnInit, TemplateRef, ViewContainerRef } from '@angular/core';
2
- import { KeyValue, ModelRenderCustomEvent } from '../../engine';
2
+ import { KeyValue, RendererCustomEvent } from '../../engine';
3
3
  import { Logger } from '@decaf-ts/logging';
4
+ import { Model } from '@decaf-ts/decorator-validation';
4
5
  import * as i0 from "@angular/core";
5
6
  /**
6
7
  * @description Dynamic component renderer for Decaf Angular applications.
@@ -25,7 +26,7 @@ import * as i0 from "@angular/core";
25
26
  * +Record~string, unknown~ globals
26
27
  * +EnvironmentInjector injector
27
28
  * +ComponentRef~unknown~ component
28
- * +EventEmitter~ModelRenderCustomEvent~ listenEvent
29
+ * +EventEmitter~RendererCustomEvent~ listenEvent
29
30
  * +ngOnInit()
30
31
  * +ngOnDestroy()
31
32
  * +ngOnChanges(changes)
@@ -104,10 +105,10 @@ export declare class ComponentRendererComponent implements OnInit, OnDestroy {
104
105
  * dynamic component, creating a communication channel between the parent and the dynamically
105
106
  * rendered child.
106
107
  *
107
- * @type {EventEmitter<ModelRenderCustomEvent>}
108
+ * @type {EventEmitter<RendererCustomEvent>}
108
109
  * @memberOf ComponentRendererComponent
109
110
  */
110
- listenEvent: EventEmitter<ModelRenderCustomEvent>;
111
+ listenEvent: EventEmitter<RendererCustomEvent>;
111
112
  /**
112
113
  * @description Logger instance for the component.
113
114
  * @summary This property holds a Logger instance specific to this component.
@@ -119,6 +120,14 @@ export declare class ComponentRendererComponent implements OnInit, OnDestroy {
119
120
  * @memberOf ComponentRendererComponent
120
121
  */
121
122
  logger: Logger;
123
+ /**
124
+ * @description Repository model for data operations.
125
+ * @summary The data model repository that this component will use for CRUD operations.
126
+ * This provides a connection to the data layer for retrieving and manipulating data.
127
+ *
128
+ * @type {Model| undefined}
129
+ */
130
+ model: Model | undefined;
122
131
  parent: undefined | KeyValue;
123
132
  inner?: TemplateRef<unknown>;
124
133
  /**
@@ -263,5 +272,5 @@ export declare class ComponentRendererComponent implements OnInit, OnDestroy {
263
272
  */
264
273
  private unsubscribeEvents;
265
274
  static ɵfac: i0.ɵɵFactoryDeclaration<ComponentRendererComponent, never>;
266
- static ɵcmp: i0.ɵɵComponentDeclaration<ComponentRendererComponent, "ngx-decaf-component-renderer", never, { "tag": { "alias": "tag"; "required": true; }; "globals": { "alias": "globals"; "required": false; }; "parent": { "alias": "parent"; "required": false; }; }, { "listenEvent": "listenEvent"; }, never, never, true, never>;
275
+ static ɵcmp: i0.ɵɵComponentDeclaration<ComponentRendererComponent, "ngx-decaf-component-renderer", never, { "tag": { "alias": "tag"; "required": true; }; "globals": { "alias": "globals"; "required": false; }; "model": { "alias": "model"; "required": false; }; "parent": { "alias": "parent"; "required": false; }; }, { "listenEvent": "listenEvent"; }, never, never, true, never>;
267
276
  }
@@ -1,7 +1,7 @@
1
1
  import { EventEmitter, OnChanges, OnDestroy, SimpleChanges, TemplateRef, ViewContainerRef } from '@angular/core';
2
2
  import { Model } from '@decaf-ts/decorator-validation';
3
3
  import { AngularDynamicOutput, RenderedModel } from '../../engine';
4
- import { ModelRenderCustomEvent } from 'src/lib/engine/types';
4
+ import { RendererCustomEvent } from 'src/lib/engine/types';
5
5
  import * as i0 from "@angular/core";
6
6
  /**
7
7
  * @description Component for rendering dynamic models
@@ -60,7 +60,7 @@ export declare class ModelRendererComponent<M extends Model> implements OnChange
60
60
  /**
61
61
  * @description Event emitter for custom events from the rendered component
62
62
  */
63
- listenEvent: EventEmitter<ModelRenderCustomEvent>;
63
+ listenEvent: EventEmitter<RendererCustomEvent>;
64
64
  /**
65
65
  * @description Instance of the NgxRenderingEngine2
66
66
  */
@@ -3,7 +3,22 @@ import { TextFieldTypes } from '@ionic/core';
3
3
  import { EnvironmentInjector, Injector, Type } from '@angular/core';
4
4
  import { FormControl, FormGroup } from '@angular/forms';
5
5
  import { FieldProperties } from '@decaf-ts/ui-decorators';
6
+ /**
7
+ * @description Generic key-value pair type
8
+ * @summary Represents a generic object with string keys and any type of values.
9
+ * This is commonly used for dynamic data structures where the properties are not known at compile time.
10
+ * @typedef {Record<string, any>} KeyValue
11
+ * @memberOf module:engine
12
+ */
6
13
  export type KeyValue = Record<string, any>;
14
+ /**
15
+ * @description Generic function type
16
+ * @summary Represents a function that accepts any number of arguments of any type
17
+ * and returns any type. This is useful for defining function parameters or variables
18
+ * where the exact function signature is not known at compile time.
19
+ * @typedef FunctionLike
20
+ * @memberOf module:engine
21
+ */
7
22
  export type FunctionLike = (...args: any[]) => any;
8
23
  /**
9
24
  * @description Element size options for UI components
@@ -204,6 +219,15 @@ export type HTMLFormTarget = '_blank' | '_self' | '_parent' | '_top' | string;
204
219
  export interface IListComponentRefreshEvent {
205
220
  data: KeyValue[];
206
221
  }
222
+ /**
223
+ * @description Form service control structure
224
+ * @summary Defines the structure for a form control managed by the form service.
225
+ * Contains the FormGroup control and the associated field properties for rendering.
226
+ * @interface FormServiceControl
227
+ * @property {FormGroup} control - The Angular FormGroup for the control
228
+ * @property {AngularFieldDefinition} props - The field properties for rendering the control
229
+ * @memberOf module:engine
230
+ */
207
231
  export interface FormServiceControl {
208
232
  control: FormGroup;
209
233
  props: AngularFieldDefinition;
@@ -218,20 +242,14 @@ export interface FormServiceControl {
218
242
  */
219
243
  export type FormServiceControls = Record<string, Record<string, FormServiceControl>>;
220
244
  /**
221
- * @description Interface for model render custom events
222
- * @summary Defines the structure of custom events triggered during model rendering.
223
- * Contains the event detail, component name, and event name.
224
- * @interface ModelRenderCustomEvent
225
- * @property {BaseCustomEvent} detail - The detailed event information
226
- * @property {string} component - The component that triggered the event
227
- * @property {string} name - The name of the event
245
+ * @description Renderer custom event type
246
+ * @summary Combines BaseCustomEvent with KeyValue properties to create a flexible
247
+ * custom event type for renderer components. This allows events to carry both
248
+ * standard event properties and additional custom data.
249
+ * @typedef RendererCustomEvent
228
250
  * @memberOf module:engine
229
251
  */
230
- export interface ModelRenderCustomEvent {
231
- detail: BaseCustomEvent;
232
- component: string;
233
- name: string;
234
- }
252
+ export type RendererCustomEvent = BaseCustomEvent & KeyValue;
235
253
  /**
236
254
  * @description Interface for list item custom events
237
255
  * @summary Defines the structure of custom events triggered by list items.
@@ -261,34 +279,59 @@ export interface ListItemCustomEvent extends BaseCustomEvent {
261
279
  * @memberOf module:engine
262
280
  */
263
281
  export interface BaseCustomEvent {
264
- data: unknown;
265
- target?: HTMLElement;
266
- name?: string;
282
+ name: string;
267
283
  component: string;
284
+ data?: unknown;
285
+ target?: HTMLElement;
268
286
  }
269
287
  /**
270
- * @description Base interface for custom events
271
- * @summary Defines the base structure for custom events in the application.
272
- * Contains properties for the event data, target element, name, and component.
273
- * @interface BaseCustomEvent
274
- * @property {any} data - The data associated with the event
275
- * @property {HTMLElement} [target] - The target element that triggered the event
276
- * @property {string} [name] - The name of the event
277
- * @property {string} component - The component that triggered the event
288
+ * @description CRUD form event type
289
+ * @summary Extends BaseCustomEvent to include optional handlers for CRUD form operations.
290
+ * This event type is used for form-related actions like create, read, update, and delete operations.
291
+ * @typedef CrudFormEvent
292
+ * @property {Record<string, any>} [handlers] - Optional handlers for form operations
278
293
  * @memberOf module:engine
279
294
  */
280
295
  export type CrudFormEvent = BaseCustomEvent & {
281
296
  handlers?: Record<string, any>;
282
297
  };
298
+ /**
299
+ * @description Component input properties
300
+ * @summary Extends FieldProperties with additional properties specific to Angular components.
301
+ * Includes update mode for form controls and optional FormGroup and FormControl references.
302
+ * @interface ComponentInput
303
+ * @property {FieldUpdateMode} [updateMode] - When the field value should be updated
304
+ * @property {FormGroup} [formGroup] - Optional FormGroup reference
305
+ * @property {FormControl} [formControl] - Optional FormControl reference
306
+ * @memberOf module:engine
307
+ */
283
308
  export interface ComponentInput extends FieldProperties {
284
309
  updateMode?: FieldUpdateMode;
285
310
  formGroup?: FormGroup;
286
311
  formControl?: FormControl;
287
312
  }
313
+ /**
314
+ * @description Component configuration structure
315
+ * @summary Defines the configuration for dynamically creating Angular components.
316
+ * Contains the component name, input properties, injector, and optional child components.
317
+ * @interface ComponentConfig
318
+ * @property {string} component - The name of the component to render
319
+ * @property {ComponentInput} inputs - The input properties for the component
320
+ * @property {EnvironmentInjector | Injector} injector - The Angular injector for dependency injection
321
+ * @property {ComponentConfig[]} [children] - Optional child component configurations
322
+ * @memberOf module:engine
323
+ */
288
324
  export interface ComponentConfig {
289
325
  component: string;
290
326
  inputs: ComponentInput;
291
327
  injector: EnvironmentInjector | Injector;
292
328
  children?: ComponentConfig[];
293
329
  }
330
+ /**
331
+ * @description Form parent group tuple
332
+ * @summary Represents a tuple containing a FormGroup and its associated string identifier.
333
+ * This is used for managing hierarchical form structures and parent-child relationships.
334
+ * @typedef {[FormGroup, string]} FormParentGroup
335
+ * @memberOf module:engine
336
+ */
294
337
  export type FormParentGroup = [FormGroup, string];
@@ -26,7 +26,7 @@ import * as i1 from "@angular/common";
26
26
  * +Record~string, unknown~ globals
27
27
  * +EnvironmentInjector injector
28
28
  * +ComponentRef~unknown~ component
29
- * +EventEmitter~ModelRenderCustomEvent~ listenEvent
29
+ * +EventEmitter~RendererCustomEvent~ listenEvent
30
30
  * +ngOnInit()
31
31
  * +ngOnDestroy()
32
32
  * +ngOnChanges(changes)
@@ -82,7 +82,7 @@ export class ComponentRendererComponent {
82
82
  * dynamic component, creating a communication channel between the parent and the dynamically
83
83
  * rendered child.
84
84
  *
85
- * @type {EventEmitter<ModelRenderCustomEvent>}
85
+ * @type {EventEmitter<RendererCustomEvent>}
86
86
  * @memberOf ComponentRendererComponent
87
87
  */
88
88
  this.listenEvent = new EventEmitter();
@@ -181,8 +181,9 @@ export class ComponentRendererComponent {
181
181
  ?.constructor;
182
182
  const metadata = reflectComponentType(component);
183
183
  const componentInputs = metadata.inputs;
184
- const props = globals?.['item'];
185
- delete props['tag'];
184
+ const props = globals?.['item'] || globals?.['props'] || {};
185
+ if (props?.['tag'])
186
+ delete props['tag'];
186
187
  const inputKeys = Object.keys(props);
187
188
  const unmappedKeys = [];
188
189
  for (const input of inputKeys) {
@@ -194,8 +195,6 @@ export class ComponentRendererComponent {
194
195
  unmappedKeys.push(input);
195
196
  }
196
197
  }
197
- if (unmappedKeys.length)
198
- this.logger.info(`Unmapped input properties for component ${tag}: ${unmappedKeys.join(', ')}`);
199
198
  this.vcr.clear();
200
199
  this.component = NgxRenderingEngine2.createComponent(component, props, metadata, this.vcr, this.injector, []);
201
200
  this.subscribeEvents();
@@ -288,7 +287,7 @@ export class ComponentRendererComponent {
288
287
  }
289
288
  }
290
289
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: ComponentRendererComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
291
- static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.13", type: ComponentRendererComponent, isStandalone: true, selector: "ngx-decaf-component-renderer", inputs: { tag: "tag", globals: "globals", parent: "parent" }, outputs: { listenEvent: "listenEvent" }, viewQueries: [{ propertyName: "vcr", first: true, predicate: ["componentViewContainer"], descendants: true, read: ViewContainerRef, static: true }, { propertyName: "inner", first: true, predicate: ["inner"], descendants: true, read: TemplateRef, static: true }], ngImport: i0, template: "<ng-template #componentViewContainer></ng-template>\n\n<ng-template #inner>\n @if(parent?.children?.length) {\n @for(child of parent.children; track child) {\n @if(!child.children?.length) {\n <ng-container\n *ngComponentOutlet=\"\n child.component;\n injector: child.injector;\n inputs: child.inputs;\n content:child.content;\n \"\n />\n } @else {\n <ngx-decaf-component-renderer [parent]=\"child\"> </ngx-decaf-component-renderer>\n }\n }\n }\n</ng-template>\n\n\n", styles: [""], dependencies: [{ kind: "component", type: ComponentRendererComponent, selector: "ngx-decaf-component-renderer", inputs: ["tag", "globals", "parent"], outputs: ["listenEvent"] }, { kind: "ngmodule", type: ForAngularModule }, { kind: "directive", type: i1.NgComponentOutlet, selector: "[ngComponentOutlet]", inputs: ["ngComponentOutlet", "ngComponentOutletInputs", "ngComponentOutletInjector", "ngComponentOutletContent", "ngComponentOutletNgModule", "ngComponentOutletNgModuleFactory"] }] }); }
290
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.13", type: ComponentRendererComponent, isStandalone: true, selector: "ngx-decaf-component-renderer", inputs: { tag: "tag", globals: "globals", model: "model", parent: "parent" }, outputs: { listenEvent: "listenEvent" }, viewQueries: [{ propertyName: "vcr", first: true, predicate: ["componentViewContainer"], descendants: true, read: ViewContainerRef, static: true }, { propertyName: "inner", first: true, predicate: ["inner"], descendants: true, read: TemplateRef, static: true }], ngImport: i0, template: "<ng-template #componentViewContainer></ng-template>\n\n<ng-template #inner>\n @if(parent?.children?.length) {\n @for(child of parent.children; track child) {\n @if(!child.children?.length) {\n <ng-container\n *ngComponentOutlet=\"\n child.component;\n injector: child.injector;\n inputs: child.inputs;\n content:child.content;\n \"\n />\n } @else {\n <ngx-decaf-component-renderer [parent]=\"child\"> </ngx-decaf-component-renderer>\n }\n }\n }\n</ng-template>\n\n\n", styles: [""], dependencies: [{ kind: "component", type: ComponentRendererComponent, selector: "ngx-decaf-component-renderer", inputs: ["tag", "globals", "model", "parent"], outputs: ["listenEvent"] }, { kind: "ngmodule", type: ForAngularModule }, { kind: "directive", type: i1.NgComponentOutlet, selector: "[ngComponentOutlet]", inputs: ["ngComponentOutlet", "ngComponentOutletInputs", "ngComponentOutletInjector", "ngComponentOutletContent", "ngComponentOutletNgModule", "ngComponentOutletNgModuleFactory"] }] }); }
292
291
  }
293
292
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: ComponentRendererComponent, decorators: [{
294
293
  type: Component,
@@ -303,10 +302,12 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
303
302
  type: Input
304
303
  }], listenEvent: [{
305
304
  type: Output
305
+ }], model: [{
306
+ type: Input
306
307
  }], parent: [{
307
308
  type: Input
308
309
  }], inner: [{
309
310
  type: ViewChild,
310
311
  args: ['inner', { read: TemplateRef, static: true }]
311
312
  }] } });
312
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"component-renderer.component.js","sourceRoot":"","sources":["../../../../../src/lib/components/component-renderer/component-renderer.component.ts","../../../../../src/lib/components/component-renderer/component-renderer.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component,\n  ComponentMirror,\n  ComponentRef,\n  EnvironmentInjector,\n  EventEmitter,\n  inject,\n  Injector,\n  Input,\n  OnDestroy,\n  OnInit,\n  Output,\n  reflectComponentType,\n  TemplateRef,\n  Type,\n  ViewChild,\n  ViewContainerRef,\n} from '@angular/core';\nimport { NgxRenderingEngine2 } from 'src/lib/engine/NgxRenderingEngine2';\nimport { BaseCustomEvent, KeyValue, ModelRenderCustomEvent } from '../../engine';\nimport { ForAngularModule, getLogger } from 'src/lib/for-angular.module';\nimport { Logger } from '@decaf-ts/logging';\n\n/**\n * @description Dynamic component renderer for Decaf Angular applications.\n * @summary This component provides a flexible way to dynamically render Angular components\n * at runtime based on a tag name. It handles the creation, property binding, and event\n * subscription for dynamically loaded components. This is particularly useful for\n * building configurable UIs where components need to be determined at runtime.\n *\n * @component {ComponentRendererComponent}\n * @example\n * <ngx-decaf-component-renderer\n *   [tag]=\"tag\"\n *   [globals]=\"globals\"\n *   (listenEvent)=\"listenEvent($event)\">\n * </ngx-decaf-component-renderer>\n *\n * @mermaid\n * classDiagram\n *   class ComponentRendererComponent {\n *     +ViewContainerRef vcr\n *     +string tag\n *     +Record~string, unknown~ globals\n *     +EnvironmentInjector injector\n *     +ComponentRef~unknown~ component\n *     +EventEmitter~ModelRenderCustomEvent~ listenEvent\n *     +ngOnInit()\n *     +ngOnDestroy()\n *     +ngOnChanges(changes)\n *     -createComponent(tag, globals)\n *     -subscribeEvents()\n *     -unsubscribeEvents()\n *   }\n *   ComponentRendererComponent --|> OnInit\n *   ComponentRendererComponent --|> OnChanges\n *   ComponentRendererComponent --|> OnDestroy\n *\n * @implements {OnInit}\n * @implements {OnChanges}\n * @implements {OnDestroy}\n */\n@Component({\n  selector: 'ngx-decaf-component-renderer',\n  templateUrl: './component-renderer.component.html',\n  styleUrls: ['./component-renderer.component.scss'],\n  imports: [ForAngularModule],\n  standalone: true,\n})\nexport class ComponentRendererComponent\n  implements OnInit, OnDestroy {\n  /**\n   * @description Reference to the container where the dynamic component will be rendered.\n   * @summary This ViewContainerRef provides the container where the dynamically created\n   * component will be inserted into the DOM. It's marked as static to ensure it's available\n   * during the ngOnInit lifecycle hook when the component is created.\n   *\n   * @type {ViewContainerRef}\n   * @memberOf ComponentRendererComponent\n   */\n  @ViewChild('componentViewContainer', { static: true, read: ViewContainerRef })\n  vcr!: ViewContainerRef;\n\n  /**\n   * @description The tag name of the component to be dynamically rendered.\n   * @summary This input property specifies which component should be rendered by providing\n   * its registered tag name. The tag must correspond to a component that has been registered\n   * with the NgxRenderingEngine2. This is a required input as it determines which component\n   * to create.\n   *\n   * @type {string}\n   * @required\n   * @memberOf ComponentRendererComponent\n   */\n  @Input({ required: true })\n  tag!: string;\n\n  /**\n   * @description Global properties to pass to the rendered component.\n   * @summary This input property allows passing a set of properties to the dynamically\n   * rendered component. These properties will be mapped to the component's inputs if they\n   * match. Properties that don't match any input on the target component will be filtered out\n   * with a warning.\n   *\n   * @type {Record<string, unknown>}\n   * @default {}\n   * @memberOf ComponentRendererComponent\n   */\n  @Input()\n  globals: Record<string, unknown> = {};\n\n  /**\n   * @description Injector used for dependency injection in the dynamic component.\n   * @summary This injector is used when creating the dynamic component to provide it with\n   * access to the application's dependency injection system. It ensures that the dynamically\n   * created component can access the same services and dependencies as statically created\n   * components.\n   *\n   * @type {EnvironmentInjector}\n   * @memberOf ComponentRendererComponent\n   */\n  injector: EnvironmentInjector = inject(EnvironmentInjector);\n\n  /**\n   * @description Reference to the dynamically created component.\n   * @summary This property holds a reference to the ComponentRef of the dynamically created\n   * component. It's used to interact with the component instance, subscribe to its events,\n   * and properly destroy it when the renderer is destroyed.\n   *\n   * @type {ComponentRef<unknown>}\n   * @memberOf ComponentRendererComponent\n   */\n  component!: ComponentRef<unknown>;\n\n  /**\n   * @description Event emitter for events from the rendered component.\n   * @summary This output property emits events that originate from the dynamically rendered\n   * component. It allows the parent component to listen for and respond to events from the\n   * dynamic component, creating a communication channel between the parent and the dynamically\n   * rendered child.\n   *\n   * @type {EventEmitter<ModelRenderCustomEvent>}\n   * @memberOf ComponentRendererComponent\n   */\n  @Output()\n  listenEvent: EventEmitter<ModelRenderCustomEvent> =\n    new EventEmitter<ModelRenderCustomEvent>();\n\n  /**\n   * @description Logger instance for the component.\n   * @summary This property holds a Logger instance specific to this component.\n   * It's used to log information, warnings, and errors related to the component's\n   * operations, particularly useful for debugging and monitoring the dynamic\n   * component rendering process.\n   *\n   * @type {Logger}\n   * @memberOf ComponentRendererComponent\n   */\n  logger!: Logger;\n\n  @Input()\n  parent: undefined | KeyValue = undefined;\n\n\n  @ViewChild('inner', { read: TemplateRef, static: true })\n  inner?: TemplateRef<unknown>;\n\n  /**\n   * @description Creates an instance of ComponentRendererComponent.\n   * @summary Initializes a new ComponentRendererComponent. This component doesn't require\n   * any dependencies to be injected in its constructor as it uses the inject function to\n   * obtain the EnvironmentInjector.\n   *\n   * @memberOf ComponentRendererComponent\n   */\n  constructor() {\n    this.logger = getLogger(this);\n  }\n\n  /**\n   * @description Initializes the component after Angular first displays the data-bound properties.\n   * @summary Sets up the component by creating the dynamic component specified by the tag input.\n   * This method is called once when the component is initialized and triggers the dynamic\n   * component creation process with the provided tag name and global properties.\n   *\n   * @mermaid\n   * sequenceDiagram\n   *   participant A as Angular Lifecycle\n   *   participant C as ComponentRendererComponent\n   *   participant R as NgxRenderingEngine2\n   *\n   *   A->>C: ngOnInit()\n   *   C->>C: createComponent(tag, globals)\n   *   C->>R: components(tag)\n   *   R-->>C: Return component constructor\n   *   C->>C: Process component inputs\n   *   C->>C: Create component instance\n   *   C->>C: subscribeEvents()\n   *\n   * @return {void}\n   * @memberOf ComponentRendererComponent\n   */\n  ngOnInit(): void {\n    if (!this.parent) {\n      this.createComponent(this.tag, this.globals);\n    } else {\n      this.createParentComponent();\n    }\n  }\n\n  /**\n   * @description Cleans up resources when the component is destroyed.\n   * @summary Performs cleanup operations when the component is being destroyed by Angular.\n   * This includes unsubscribing from all event emitters of the dynamic component and\n   * destroying the rendering engine instance to prevent memory leaks.\n   *\n   * @mermaid\n   * sequenceDiagram\n   *   participant A as Angular Lifecycle\n   *   participant C as ComponentRendererComponent\n   *   participant R as NgxRenderingEngine2\n   *\n   *   A->>C: ngOnDestroy()\n   *   alt component exists\n   *     C->>C: unsubscribeEvents()\n   *     C->>R: destroy()\n   *   end\n   *\n   * @return {Promise<void>} A promise that resolves when cleanup is complete\n   * @memberOf ComponentRendererComponent\n   */\n  async ngOnDestroy(): Promise<void> {\n    if (this.component) {\n      this.unsubscribeEvents();\n      NgxRenderingEngine2.destroy();\n    }\n  }\n\n  /**\n   * @description Creates and renders a dynamic component.\n   * @summary This method handles the creation of a dynamic component based on the provided tag.\n   * It retrieves the component constructor from the rendering engine, processes its inputs,\n   * filters out unmapped properties, creates the component instance, and sets up event subscriptions.\n   *\n   * @param {string} tag - The tag name of the component to create\n   * @param {KeyValue} globals - Global properties to pass to the component\n   * @return {void}\n   *\n   * @mermaid\n   * sequenceDiagram\n   *   participant C as ComponentRendererComponent\n   *   participant R as NgxRenderingEngine2\n   *   participant V as ViewContainerRef\n   *\n   *   C->>R: components(tag)\n   *   R-->>C: Return component constructor\n   *   C->>C: reflectComponentType(component)\n   *   C->>C: Process input properties\n   *   C->>C: Filter unmapped properties\n   *   C->>V: clear()\n   *   C->>R: createComponent(component, props, metadata, vcr, injector, [])\n   *   R-->>C: Return component reference\n   *   C->>C: subscribeEvents()\n   *\n   * @private\n   * @memberOf ComponentRendererComponent\n   */\n  private createComponent(tag: string, globals: KeyValue = {}): void {\n    const component = NgxRenderingEngine2.components(tag)\n      ?.constructor as Type<unknown>;\n    const metadata = reflectComponentType(component);\n    const componentInputs = (metadata as ComponentMirror<unknown>).inputs;\n    const props = globals?.['item'];\n    delete props['tag'];\n    const inputKeys = Object.keys(props);\n    const unmappedKeys = [];\n\n    for (const input of inputKeys) {\n      if (!inputKeys.length) break;\n      const prop = componentInputs.find(\n        (item: { propName: string }) => item.propName === input,\n      );\n      if (!prop) {\n        delete props[input];\n        unmappedKeys.push(input);\n      }\n    }\n    if (unmappedKeys.length)\n      this.logger.info(`Unmapped input properties for component ${tag}: ${unmappedKeys.join(', ')}`);\n\n    this.vcr.clear();\n    this.component = NgxRenderingEngine2.createComponent(\n      component,\n      props,\n      metadata as ComponentMirror<unknown>,\n      this.vcr,\n      this.injector as Injector,\n      [],\n    );\n    this.subscribeEvents();\n  }\n\n  createParentComponent() {\n    const { component, inputs } = this.parent as KeyValue;\n    const metadata = reflectComponentType(component) as ComponentMirror<unknown>;\n    const template = this.vcr.createEmbeddedView(this.inner as TemplateRef<unknown>, this.injector).rootNodes;\n    this.component = NgxRenderingEngine2.createComponent(\n      component,\n      inputs,\n      metadata,\n      this.vcr,\n      this.injector,\n      template,\n    );\n    this.subscribeEvents();\n  }\n\n  /**\n   * @description Subscribes to events emitted by the dynamic component.\n   * @summary This method sets up subscriptions to all EventEmitter properties of the\n   * dynamically created component. When an event is emitted by the dynamic component,\n   * it is captured and re-emitted through the listenEvent output property with additional\n   * metadata about the event source.\n   *\n   * @mermaid\n   * sequenceDiagram\n   *   participant C as ComponentRendererComponent\n   *   participant D as Dynamic Component\n   *   participant P as Parent Component\n   *\n   *   C->>C: subscribeEvents()\n   *   C->>D: Get instance properties\n   *   loop For each property\n   *     C->>C: Check if property is EventEmitter\n   *     alt is EventEmitter\n   *       C->>D: Subscribe to event\n   *       D-->>C: Event emitted\n   *       C->>P: Re-emit event with metadata\n   *     end\n   *   end\n   *\n   * @private\n   * @return {void}\n   * @memberOf ComponentRendererComponent\n   */\n  private subscribeEvents(): void {\n    if (this.component) {\n      const instance = this.component?.instance as KeyValue;\n      const componentKeys = Object.keys(instance);\n      for (const key of componentKeys) {\n        const value = instance[key];\n        if (value instanceof EventEmitter)\n          (instance as KeyValue)[key].subscribe(\n            (event: Partial<BaseCustomEvent>) => {\n              this.listenEvent.emit({\n                name: key,\n                ...event,\n              } as ModelRenderCustomEvent);\n            },\n          );\n      }\n    }\n  }\n\n  /**\n   * @description Unsubscribes from all events of the dynamic component.\n   * @summary This method cleans up event subscriptions when the component is being destroyed.\n   * It iterates through all properties of the dynamic component instance and unsubscribes\n   * from any EventEmitter properties to prevent memory leaks and unexpected behavior after\n   * the component is destroyed.\n   *\n   * @mermaid\n   * sequenceDiagram\n   *   participant C as ComponentRendererComponent\n   *   participant D as Dynamic Component\n   *\n   *   C->>C: unsubscribeEvents()\n   *   C->>D: Get instance properties\n   *   loop For each property\n   *     C->>C: Check if property is EventEmitter\n   *     alt is EventEmitter\n   *       C->>D: Unsubscribe from event\n   *     end\n   *   end\n   *\n   * @private\n   * @return {void}\n   * @memberOf ComponentRendererComponent\n   */\n  private unsubscribeEvents(): void {\n    if (this.component) {\n      const instance = this.component?.instance as KeyValue;\n      const componentKeys = Object.keys(instance);\n      for (const key of componentKeys) {\n        const value = instance[key];\n        if (value instanceof EventEmitter) instance[key].unsubscribe();\n      }\n    }\n  }\n}\n","<ng-template #componentViewContainer></ng-template>\n\n<ng-template #inner>\n  @if(parent?.children?.length) {\n    @for(child of parent.children; track child) {\n        @if(!child.children?.length) {\n          <ng-container\n                *ngComponentOutlet=\"\n                  child.component;\n                  injector: child.injector;\n                  inputs: child.inputs;\n                  content:child.content;\n                \"\n          />\n        } @else {\n          <ngx-decaf-component-renderer [parent]=\"child\"> </ngx-decaf-component-renderer>\n        }\n     }\n  }\n</ng-template>\n\n\n"]}
313
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"component-renderer.component.js","sourceRoot":"","sources":["../../../../../src/lib/components/component-renderer/component-renderer.component.ts","../../../../../src/lib/components/component-renderer/component-renderer.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component,\n  ComponentMirror,\n  ComponentRef,\n  EnvironmentInjector,\n  EventEmitter,\n  inject,\n  Injector,\n  Input,\n  OnDestroy,\n  OnInit,\n  Output,\n  reflectComponentType,\n  TemplateRef,\n  Type,\n  ViewChild,\n  ViewContainerRef,\n} from '@angular/core';\nimport { NgxRenderingEngine2 } from 'src/lib/engine/NgxRenderingEngine2';\nimport { BaseCustomEvent, KeyValue, RendererCustomEvent } from '../../engine';\nimport { ForAngularModule, getLogger } from 'src/lib/for-angular.module';\nimport { Logger } from '@decaf-ts/logging';\nimport { Model } from '@decaf-ts/decorator-validation';\n\n/**\n * @description Dynamic component renderer for Decaf Angular applications.\n * @summary This component provides a flexible way to dynamically render Angular components\n * at runtime based on a tag name. It handles the creation, property binding, and event\n * subscription for dynamically loaded components. This is particularly useful for\n * building configurable UIs where components need to be determined at runtime.\n *\n * @component {ComponentRendererComponent}\n * @example\n * <ngx-decaf-component-renderer\n *   [tag]=\"tag\"\n *   [globals]=\"globals\"\n *   (listenEvent)=\"listenEvent($event)\">\n * </ngx-decaf-component-renderer>\n *\n * @mermaid\n * classDiagram\n *   class ComponentRendererComponent {\n *     +ViewContainerRef vcr\n *     +string tag\n *     +Record~string, unknown~ globals\n *     +EnvironmentInjector injector\n *     +ComponentRef~unknown~ component\n *     +EventEmitter~RendererCustomEvent~ listenEvent\n *     +ngOnInit()\n *     +ngOnDestroy()\n *     +ngOnChanges(changes)\n *     -createComponent(tag, globals)\n *     -subscribeEvents()\n *     -unsubscribeEvents()\n *   }\n *   ComponentRendererComponent --|> OnInit\n *   ComponentRendererComponent --|> OnChanges\n *   ComponentRendererComponent --|> OnDestroy\n *\n * @implements {OnInit}\n * @implements {OnChanges}\n * @implements {OnDestroy}\n */\n@Component({\n  selector: 'ngx-decaf-component-renderer',\n  templateUrl: './component-renderer.component.html',\n  styleUrls: ['./component-renderer.component.scss'],\n  imports: [ForAngularModule],\n  standalone: true,\n})\nexport class ComponentRendererComponent\n  implements OnInit, OnDestroy {\n  /**\n   * @description Reference to the container where the dynamic component will be rendered.\n   * @summary This ViewContainerRef provides the container where the dynamically created\n   * component will be inserted into the DOM. It's marked as static to ensure it's available\n   * during the ngOnInit lifecycle hook when the component is created.\n   *\n   * @type {ViewContainerRef}\n   * @memberOf ComponentRendererComponent\n   */\n  @ViewChild('componentViewContainer', { static: true, read: ViewContainerRef })\n  vcr!: ViewContainerRef;\n\n  /**\n   * @description The tag name of the component to be dynamically rendered.\n   * @summary This input property specifies which component should be rendered by providing\n   * its registered tag name. The tag must correspond to a component that has been registered\n   * with the NgxRenderingEngine2. This is a required input as it determines which component\n   * to create.\n   *\n   * @type {string}\n   * @required\n   * @memberOf ComponentRendererComponent\n   */\n  @Input({ required: true })\n  tag!: string;\n\n  /**\n   * @description Global properties to pass to the rendered component.\n   * @summary This input property allows passing a set of properties to the dynamically\n   * rendered component. These properties will be mapped to the component's inputs if they\n   * match. Properties that don't match any input on the target component will be filtered out\n   * with a warning.\n   *\n   * @type {Record<string, unknown>}\n   * @default {}\n   * @memberOf ComponentRendererComponent\n   */\n  @Input()\n  globals: Record<string, unknown> = {};\n\n  /**\n   * @description Injector used for dependency injection in the dynamic component.\n   * @summary This injector is used when creating the dynamic component to provide it with\n   * access to the application's dependency injection system. It ensures that the dynamically\n   * created component can access the same services and dependencies as statically created\n   * components.\n   *\n   * @type {EnvironmentInjector}\n   * @memberOf ComponentRendererComponent\n   */\n  injector: EnvironmentInjector = inject(EnvironmentInjector);\n\n  /**\n   * @description Reference to the dynamically created component.\n   * @summary This property holds a reference to the ComponentRef of the dynamically created\n   * component. It's used to interact with the component instance, subscribe to its events,\n   * and properly destroy it when the renderer is destroyed.\n   *\n   * @type {ComponentRef<unknown>}\n   * @memberOf ComponentRendererComponent\n   */\n  component!: ComponentRef<unknown>;\n\n  /**\n   * @description Event emitter for events from the rendered component.\n   * @summary This output property emits events that originate from the dynamically rendered\n   * component. It allows the parent component to listen for and respond to events from the\n   * dynamic component, creating a communication channel between the parent and the dynamically\n   * rendered child.\n   *\n   * @type {EventEmitter<RendererCustomEvent>}\n   * @memberOf ComponentRendererComponent\n   */\n  @Output()\n  listenEvent: EventEmitter<RendererCustomEvent> =\n    new EventEmitter<RendererCustomEvent>();\n\n  /**\n   * @description Logger instance for the component.\n   * @summary This property holds a Logger instance specific to this component.\n   * It's used to log information, warnings, and errors related to the component's\n   * operations, particularly useful for debugging and monitoring the dynamic\n   * component rendering process.\n   *\n   * @type {Logger}\n   * @memberOf ComponentRendererComponent\n   */\n  logger!: Logger;\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   */\n  @Input()\n  model!:  Model | undefined;\n\n  @Input()\n  parent: undefined | KeyValue = undefined;\n\n\n  @ViewChild('inner', { read: TemplateRef, static: true })\n  inner?: TemplateRef<unknown>;\n\n  /**\n   * @description Creates an instance of ComponentRendererComponent.\n   * @summary Initializes a new ComponentRendererComponent. This component doesn't require\n   * any dependencies to be injected in its constructor as it uses the inject function to\n   * obtain the EnvironmentInjector.\n   *\n   * @memberOf ComponentRendererComponent\n   */\n  constructor() {\n    this.logger = getLogger(this);\n  }\n\n  /**\n   * @description Initializes the component after Angular first displays the data-bound properties.\n   * @summary Sets up the component by creating the dynamic component specified by the tag input.\n   * This method is called once when the component is initialized and triggers the dynamic\n   * component creation process with the provided tag name and global properties.\n   *\n   * @mermaid\n   * sequenceDiagram\n   *   participant A as Angular Lifecycle\n   *   participant C as ComponentRendererComponent\n   *   participant R as NgxRenderingEngine2\n   *\n   *   A->>C: ngOnInit()\n   *   C->>C: createComponent(tag, globals)\n   *   C->>R: components(tag)\n   *   R-->>C: Return component constructor\n   *   C->>C: Process component inputs\n   *   C->>C: Create component instance\n   *   C->>C: subscribeEvents()\n   *\n   * @return {void}\n   * @memberOf ComponentRendererComponent\n   */\n  ngOnInit(): void {\n    if (!this.parent) {\n      this.createComponent(this.tag, this.globals);\n    } else {\n      this.createParentComponent();\n    }\n  }\n\n  /**\n   * @description Cleans up resources when the component is destroyed.\n   * @summary Performs cleanup operations when the component is being destroyed by Angular.\n   * This includes unsubscribing from all event emitters of the dynamic component and\n   * destroying the rendering engine instance to prevent memory leaks.\n   *\n   * @mermaid\n   * sequenceDiagram\n   *   participant A as Angular Lifecycle\n   *   participant C as ComponentRendererComponent\n   *   participant R as NgxRenderingEngine2\n   *\n   *   A->>C: ngOnDestroy()\n   *   alt component exists\n   *     C->>C: unsubscribeEvents()\n   *     C->>R: destroy()\n   *   end\n   *\n   * @return {Promise<void>} A promise that resolves when cleanup is complete\n   * @memberOf ComponentRendererComponent\n   */\n  async ngOnDestroy(): Promise<void> {\n    if (this.component) {\n      this.unsubscribeEvents();\n      NgxRenderingEngine2.destroy();\n    }\n  }\n\n  /**\n   * @description Creates and renders a dynamic component.\n   * @summary This method handles the creation of a dynamic component based on the provided tag.\n   * It retrieves the component constructor from the rendering engine, processes its inputs,\n   * filters out unmapped properties, creates the component instance, and sets up event subscriptions.\n   *\n   * @param {string} tag - The tag name of the component to create\n   * @param {KeyValue} globals - Global properties to pass to the component\n   * @return {void}\n   *\n   * @mermaid\n   * sequenceDiagram\n   *   participant C as ComponentRendererComponent\n   *   participant R as NgxRenderingEngine2\n   *   participant V as ViewContainerRef\n   *\n   *   C->>R: components(tag)\n   *   R-->>C: Return component constructor\n   *   C->>C: reflectComponentType(component)\n   *   C->>C: Process input properties\n   *   C->>C: Filter unmapped properties\n   *   C->>V: clear()\n   *   C->>R: createComponent(component, props, metadata, vcr, injector, [])\n   *   R-->>C: Return component reference\n   *   C->>C: subscribeEvents()\n   *\n   * @private\n   * @memberOf ComponentRendererComponent\n   */\n  private createComponent(tag: string, globals: KeyValue = {}): void {\n    const component = NgxRenderingEngine2.components(tag)\n      ?.constructor as Type<unknown>;\n    const metadata = reflectComponentType(component);\n    const componentInputs = (metadata as ComponentMirror<unknown>).inputs;\n    const props = globals?.['item'] || globals?.['props'] || {};\n    if(props?.['tag'])\n      delete props['tag'];\n    const inputKeys = Object.keys(props);\n    const unmappedKeys = [];\n\n    for (const input of inputKeys) {\n      if (!inputKeys.length) break;\n      const prop = componentInputs.find(\n        (item: { propName: string }) => item.propName === input,\n      );\n      if (!prop) {\n        delete props[input];\n        unmappedKeys.push(input);\n      }\n    }\n    this.vcr.clear();\n    this.component = NgxRenderingEngine2.createComponent(\n      component,\n      props,\n      metadata as ComponentMirror<unknown>,\n      this.vcr,\n      this.injector as Injector,\n      [],\n    );\n    this.subscribeEvents();\n  }\n\n  createParentComponent() {\n    const { component, inputs } = this.parent as KeyValue;\n    const metadata = reflectComponentType(component) as ComponentMirror<unknown>;\n    const template = this.vcr.createEmbeddedView(this.inner as TemplateRef<unknown>, this.injector).rootNodes;\n    this.component = NgxRenderingEngine2.createComponent(\n      component,\n      inputs,\n      metadata,\n      this.vcr,\n      this.injector,\n      template,\n    );\n    this.subscribeEvents();\n  }\n\n  /**\n   * @description Subscribes to events emitted by the dynamic component.\n   * @summary This method sets up subscriptions to all EventEmitter properties of the\n   * dynamically created component. When an event is emitted by the dynamic component,\n   * it is captured and re-emitted through the listenEvent output property with additional\n   * metadata about the event source.\n   *\n   * @mermaid\n   * sequenceDiagram\n   *   participant C as ComponentRendererComponent\n   *   participant D as Dynamic Component\n   *   participant P as Parent Component\n   *\n   *   C->>C: subscribeEvents()\n   *   C->>D: Get instance properties\n   *   loop For each property\n   *     C->>C: Check if property is EventEmitter\n   *     alt is EventEmitter\n   *       C->>D: Subscribe to event\n   *       D-->>C: Event emitted\n   *       C->>P: Re-emit event with metadata\n   *     end\n   *   end\n   *\n   * @private\n   * @return {void}\n   * @memberOf ComponentRendererComponent\n   */\n  private subscribeEvents(): void {\n    if (this.component) {\n      const instance = this.component?.instance as KeyValue;\n      const componentKeys = Object.keys(instance);\n      for (const key of componentKeys) {\n        const value = instance[key];\n        if (value instanceof EventEmitter)\n          (instance as KeyValue)[key].subscribe(\n            (event: Partial<BaseCustomEvent>) => {\n              this.listenEvent.emit({\n                name: key,\n                ...event,\n              } as RendererCustomEvent);\n            },\n          );\n      }\n    }\n  }\n\n  /**\n   * @description Unsubscribes from all events of the dynamic component.\n   * @summary This method cleans up event subscriptions when the component is being destroyed.\n   * It iterates through all properties of the dynamic component instance and unsubscribes\n   * from any EventEmitter properties to prevent memory leaks and unexpected behavior after\n   * the component is destroyed.\n   *\n   * @mermaid\n   * sequenceDiagram\n   *   participant C as ComponentRendererComponent\n   *   participant D as Dynamic Component\n   *\n   *   C->>C: unsubscribeEvents()\n   *   C->>D: Get instance properties\n   *   loop For each property\n   *     C->>C: Check if property is EventEmitter\n   *     alt is EventEmitter\n   *       C->>D: Unsubscribe from event\n   *     end\n   *   end\n   *\n   * @private\n   * @return {void}\n   * @memberOf ComponentRendererComponent\n   */\n  private unsubscribeEvents(): void {\n    if (this.component) {\n      const instance = this.component?.instance as KeyValue;\n      const componentKeys = Object.keys(instance);\n      for (const key of componentKeys) {\n        const value = instance[key];\n        if (value instanceof EventEmitter) instance[key].unsubscribe();\n      }\n    }\n  }\n}\n","<ng-template #componentViewContainer></ng-template>\n\n<ng-template #inner>\n  @if(parent?.children?.length) {\n    @for(child of parent.children; track child) {\n        @if(!child.children?.length) {\n          <ng-container\n                *ngComponentOutlet=\"\n                  child.component;\n                  injector: child.injector;\n                  inputs: child.inputs;\n                  content:child.content;\n                \"\n          />\n        } @else {\n          <ngx-decaf-component-renderer [parent]=\"child\"> </ngx-decaf-component-renderer>\n        }\n     }\n  }\n</ng-template>\n\n\n"]}
@@ -76,7 +76,7 @@ let CrudFormComponent = class CrudFormComponent {
76
76
  const data = NgxFormService.getFormData(this.formGroup);
77
77
  this.submitEvent.emit({
78
78
  data,
79
- component: 'FormReactiveComponent',
79
+ component: 'CrudFormComponent',
80
80
  name: this.action || EventConstants.SUBMIT_EVENT,
81
81
  handlers: this.handlers,
82
82
  });
@@ -91,7 +91,7 @@ let CrudFormComponent = class CrudFormComponent {
91
91
  handleDelete() {
92
92
  this.submitEvent.emit({
93
93
  data: this.uid,
94
- component: 'FormReactiveComponent',
94
+ component: 'CrudFormComponent',
95
95
  name: EventConstants.SUBMIT_EVENT,
96
96
  });
97
97
  }
@@ -136,4 +136,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
136
136
  }], submitEvent: [{
137
137
  type: Output
138
138
  }] } });
139
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"crud-form.component.js","sourceRoot":"","sources":["../../../../../src/lib/components/crud-form/crud-form.component.ts","../../../../../src/lib/components/crud-form/crud-form.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component,\n  ElementRef,\n  EventEmitter,\n  inject,\n  Input,\n  OnDestroy,\n  OnInit,\n  Output,\n  ViewChild,\n} from '@angular/core';\nimport { Location } from '@angular/common';\nimport { FormGroup } from '@angular/forms';\nimport { FormElement } from '../../interfaces';\nimport { NgxFormService } from '../../engine/NgxFormService';\nimport { CrudFormEvent, Dynamic, EventConstants, FieldUpdateMode, HTMLFormTarget, RenderedModel } from '../../engine';\nimport { CrudFormOptions } from './types';\nimport { CrudOperations, OperationKeys } from '@decaf-ts/db-decorators';\nimport { DefaultFormReactiveOptions } from './constants';\nimport { ForAngularModule, getLogger } from 'src/lib/for-angular.module';\nimport { IonIcon } from '@ionic/angular/standalone';\nimport { Model } from '@decaf-ts/decorator-validation';\nimport { Logger } from '@decaf-ts/logging';\n\n\n/**\n * @component CrudFormComponent\n * @example <ngx-decaf-crud-form\n *   action=\"create\"\n *   operation=\"create\"\n *   formGroup=\"formGroup\"\n *   rendererId=\"rendererId\"\n *   submitEvent=\"submitEvent\"\n *   target=\"_self\"\n *   method=\"event\">\n * </ngx-decaf-crud-form>\n *\n * @param {string} action - The action to be performed (create, read, update, delete)\n * @param {CrudOperations} operation - The CRUD operation being performed (create, read, update, delete)\n * @param {FormGroup} formGroup - The form group\n * @param {string} rendererId - The renderer id\n * @param {SubmitEvent} submitEvent - The submit event\n * @param {string} target - The target\n * @param {string} method - The method\n */\n@Dynamic()\n@Component({\n  standalone: true,\n  selector: 'ngx-decaf-crud-form',\n  templateUrl: './crud-form.component.html',\n  styleUrls: ['./crud-form.component.scss'],\n  imports: [ForAngularModule, IonIcon],\n})\nexport class CrudFormComponent implements OnInit, FormElement, OnDestroy, RenderedModel {\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   */\n  @Input()\n  model!: Model | undefined;\n\n  @Input()\n  updateOn: FieldUpdateMode = 'change';\n\n  @ViewChild('reactiveForm', { static: false, read: ElementRef })\n  component!: ElementRef;\n\n  @Input()\n  target: HTMLFormTarget = '_self';\n\n  @Input()\n  method: 'get' | 'post' | 'event' = 'event';\n\n  @Input()\n  options!: CrudFormOptions;\n\n  @Input()\n  action?: string;\n\n  @Input({ required: true })\n  operation!: CrudOperations;\n\n  @Input()\n  handlers!: Record<string, (...args: unknown[]) => unknown | Promise<unknown>>;\n\n  @Input()\n  formGroup!: FormGroup | undefined;\n\n  /**\n   * @description Path to the parent FormGroup, if nested.\n   * @summary Full dot-delimited path of the parent FormGroup. Set only when is part of a nested structure.\n   *\n   * @type {string}\n   * @memberOf CrudFormComponent\n   */\n  @Input()\n  childOf?: string;\n\n  @Input()\n  rendererId!: string;\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   */\n  @Input()\n  uid!: string | number | undefined;\n\n\n  @Output()\n  submitEvent = new EventEmitter<CrudFormEvent>();\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 CrudFormComponent\n   */\n  private logger!: Logger;\n\n  /**\n   * @description Angular Location service.\n   * @summary Injected service that provides access to the browser's URL and history.\n   * This service is used for interacting with the browser's history API, allowing\n   * for back navigation and URL manipulation outside of Angular's router.\n   *\n   * @private\n   * @type {Location}\n   * @memberOf CrudFormComponent\n   */\n  private location: Location = inject(Location);\n\n  // ngAfterViewInit() {\n    // if (![OperationKeys.READ, OperationKeys.DELETE].includes(this.operation))\n    //   NgxFormService.formAfterViewInit(this, this.rendererId);\n  // }\n\n  async ngOnInit() {\n    if (!this.logger)\n      this.logger = getLogger(this);\n    if (this.operation === OperationKeys.READ || this.operation === OperationKeys.DELETE)\n      this.formGroup = undefined;\n    this.options = Object.assign(\n      {},\n      DefaultFormReactiveOptions,\n      this.options || {},\n    );\n\n  }\n\n  ngOnDestroy() {\n    if (this.formGroup)\n      NgxFormService.unregister(this.formGroup);\n  }\n\n  /**\n   * @param  {SubmitEvent} event\n   */\n  async submit(event: SubmitEvent): Promise<boolean | void> {\n    event.preventDefault();\n    event.stopImmediatePropagation();\n    event.stopPropagation();\n\n    if (!NgxFormService.validateFields(this.formGroup as FormGroup))\n      return false;\n\n    const data = NgxFormService.getFormData(this.formGroup as FormGroup);\n    this.submitEvent.emit({\n      data,\n      component: 'FormReactiveComponent',\n      name: this.action || EventConstants.SUBMIT_EVENT,\n      handlers: this.handlers,\n    });\n  }\n\n  handleReset() {\n    this.location.back();\n    // if(OperationKeys.DELETE !== this.operation)\n    //   NgxFormService.reset(this.formGroup);\n    // else\n    //   this.location.back();\n  }\n\n  handleDelete() {\n    this.submitEvent.emit({\n      data: this.uid,\n      component: 'FormReactiveComponent',\n      name: EventConstants.SUBMIT_EVENT,\n    });\n  }\n\n  protected readonly OperationKeys = OperationKeys;\n}\n","@if(operation !== 'read' && operation !== 'delete') {\n  <form #reactiveForm [id]=\"rendererId\" [formGroup]=\"formGroup\" (submit)=\"submit($event)\" novalidate [target]=\"target\">\n    <ng-content #formContent></ng-content>\n    <div class=\"dcf-flex dcf-flex-right\">\n      <div>\n        @if(options.buttons.clear) {\n          <ion-button fill=\"clear\" (click)=\"handleReset()\">\n            @if(options.buttons.clear?.icon) {\n              <ion-icon  aria-hidden=\"true\" [slot]=\"options.buttons.clear?.iconSlot\" [name]=\"options.buttons.clear?.icon\"></ion-icon>\n            }\n            {{operation === 'update' ? 'Back' : options.buttons.clear?.text}}\n          </ion-button>\n        }\n\n        <ion-button\n          type=\"submit\">\n          @if(options.buttons.submit.icon) {\n            <ion-icon aria-hidden=\"true\" [slot]=\"options.buttons.submit.iconSlot\" [name]=\"options.buttons.submit.icon\"></ion-icon>\n          }\n          {{ action ? action : options.buttons.submit.text}}\n        </ion-button>\n      </div>\n    </div>\n  </form>\n} @else {\n  <div [class]=\"'dcf-flex dcf-flex-right ' + operation\" id=\"dcf-buttons-container\">\n    <div>\n      @if(options.buttons.clear) {\n        <ion-button fill=\"clear\" (click)=\"handleReset()\">\n          @if(options.buttons.clear?.icon) {\n            <ion-icon  aria-hidden=\"true\" [slot]=\"options.buttons.clear?.iconSlot\" [name]=\"options.buttons.clear?.icon\"></ion-icon>\n          }\n          {{ ['delete', 'read', 'update'].includes(operation) ? 'Back' : options.buttons.clear?.text}}\n        </ion-button>\n      }\n      @if(operation === 'delete' && uid) {\n        <ion-button\n          (click)=\"handleDelete()\"\n          color=\"danger\"\n          type=\"button\">\n          @if(options.buttons.submit.icon) {\n            <ion-icon aria-hidden=\"true\" [slot]=\"options.buttons.submit.iconSlot\" [name]=\"options.buttons.submit.icon\"></ion-icon>\n          }\n          Delete\n       </ion-button>\n      }\n      @if(operation === OperationKeys.CREATE || operation === OperationKeys.UPDATE) {\n        <ion-button\n          type=\"submit\">\n          @if(options.buttons.submit.icon) {\n            <ion-icon aria-hidden=\"true\" [slot]=\"options.buttons.submit.iconSlot\" [name]=\"options.buttons.submit.icon\"></ion-icon>\n          }\n          {{options.buttons.submit.text}}\n        </ion-button>\n      }\n\n    </div>\n  </div>\n}\n\n"]}
139
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"crud-form.component.js","sourceRoot":"","sources":["../../../../../src/lib/components/crud-form/crud-form.component.ts","../../../../../src/lib/components/crud-form/crud-form.component.html"],"names":[],"mappings":";AAAA,OAAO,EACL,SAAS,EACT,UAAU,EACV,YAAY,EACZ,MAAM,EACN,KAAK,EAGL,MAAM,EACN,SAAS,GACV,MAAM,eAAe,CAAC;AACvB,OAAO,EAAE,QAAQ,EAAE,MAAM,iBAAiB,CAAC;AAG3C,OAAO,EAAE,cAAc,EAAE,MAAM,6BAA6B,CAAC;AAC7D,OAAO,EAAiB,OAAO,EAAE,cAAc,EAAkD,MAAM,cAAc,CAAC;AAEtH,OAAO,EAAkB,aAAa,EAAE,MAAM,yBAAyB,CAAC;AACxE,OAAO,EAAE,0BAA0B,EAAE,MAAM,aAAa,CAAC;AACzD,OAAO,EAAE,gBAAgB,EAAE,SAAS,EAAE,MAAM,4BAA4B,CAAC;AACzE,OAAO,EAAE,OAAO,EAAE,MAAM,2BAA2B,CAAC;;;;AAKpD;;;;;;;;;;;;;;;;;;;GAmBG;AASI,IAAM,iBAAiB,GAAvB,MAAM,iBAAiB;IAAvB;QAaL,aAAQ,GAAoB,QAAQ,CAAC;QAMrC,WAAM,GAAmB,OAAO,CAAC;QAGjC,WAAM,GAA6B,OAAO,CAAC;QA0C3C,gBAAW,GAAG,IAAI,YAAY,EAAiB,CAAC;QAkBhD;;;;;;;;;WASG;QACK,aAAQ,GAAa,MAAM,CAAC,QAAQ,CAAC,CAAC;QA6D3B,kBAAa,GAAG,aAAa,CAAC;KAClD;IA5DC,sBAAsB;IACpB,4EAA4E;IAC5E,6DAA6D;IAC/D,IAAI;IAEJ,KAAK,CAAC,QAAQ;QACZ,IAAI,CAAC,IAAI,CAAC,MAAM;YACd,IAAI,CAAC,MAAM,GAAG,SAAS,CAAC,IAAI,CAAC,CAAC;QAChC,IAAI,IAAI,CAAC,SAAS,KAAK,aAAa,CAAC,IAAI,IAAI,IAAI,CAAC,SAAS,KAAK,aAAa,CAAC,MAAM;YAClF,IAAI,CAAC,SAAS,GAAG,SAAS,CAAC;QAC7B,IAAI,CAAC,OAAO,GAAG,MAAM,CAAC,MAAM,CAC1B,EAAE,EACF,0BAA0B,EAC1B,IAAI,CAAC,OAAO,IAAI,EAAE,CACnB,CAAC;IAEJ,CAAC;IAED,WAAW;QACT,IAAI,IAAI,CAAC,SAAS;YAChB,cAAc,CAAC,UAAU,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;IAC9C,CAAC;IAED;;OAEG;IACH,KAAK,CAAC,MAAM,CAAC,KAAkB;QAC7B,KAAK,CAAC,cAAc,EAAE,CAAC;QACvB,KAAK,CAAC,wBAAwB,EAAE,CAAC;QACjC,KAAK,CAAC,eAAe,EAAE,CAAC;QAExB,IAAI,CAAC,cAAc,CAAC,cAAc,CAAC,IAAI,CAAC,SAAsB,CAAC;YAC7D,OAAO,KAAK,CAAC;QAEf,MAAM,IAAI,GAAG,cAAc,CAAC,WAAW,CAAC,IAAI,CAAC,SAAsB,CAAC,CAAC;QACrE,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC;YACpB,IAAI;YACJ,SAAS,EAAE,mBAAmB;YAC9B,IAAI,EAAE,IAAI,CAAC,MAAM,IAAI,cAAc,CAAC,YAAY;YAChD,QAAQ,EAAE,IAAI,CAAC,QAAQ;SACxB,CAAC,CAAC;IACL,CAAC;IAED,WAAW;QACT,IAAI,CAAC,QAAQ,CAAC,IAAI,EAAE,CAAC;QACrB,8CAA8C;QAC9C,0CAA0C;QAC1C,OAAO;QACP,0BAA0B;IAC5B,CAAC;IAED,YAAY;QACV,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC;YACpB,IAAI,EAAE,IAAI,CAAC,GAAG;YACd,SAAS,EAAE,mBAAmB;YAC9B,IAAI,EAAE,cAAc,CAAC,YAAY;SAClC,CAAC,CAAC;IACL,CAAC;+GAvJU,iBAAiB;mGAAjB,iBAAiB,0cAesB,UAAU,6BCpE9D,k6EA4DA,oHDTY,gBAAgB,qrBAAE,OAAO;;AAExB,iBAAiB;IAR7B,OAAO,EAAE;GAQG,iBAAiB,CA0J7B;;4FA1JY,iBAAiB;kBAP7B,SAAS;iCACI,IAAI,YACN,qBAAqB,WAGtB,CAAC,gBAAgB,EAAE,OAAO,CAAC;8BAYpC,KAAK;sBADJ,KAAK;gBAIN,QAAQ;sBADP,KAAK;gBAIN,SAAS;sBADR,SAAS;uBAAC,cAAc,EAAE,EAAE,MAAM,EAAE,KAAK,EAAE,IAAI,EAAE,UAAU,EAAE;gBAI9D,MAAM;sBADL,KAAK;gBAIN,MAAM;sBADL,KAAK;gBAIN,OAAO;sBADN,KAAK;gBAIN,MAAM;sBADL,KAAK;gBAIN,SAAS;sBADR,KAAK;uBAAC,EAAE,QAAQ,EAAE,IAAI,EAAE;gBAIzB,QAAQ;sBADP,KAAK;gBAIN,SAAS;sBADR,KAAK;gBAWN,OAAO;sBADN,KAAK;gBAIN,UAAU;sBADT,KAAK;gBAWN,GAAG;sBADF,KAAK;gBAKN,WAAW;sBADV,MAAM","sourcesContent":["import {\n  Component,\n  ElementRef,\n  EventEmitter,\n  inject,\n  Input,\n  OnDestroy,\n  OnInit,\n  Output,\n  ViewChild,\n} from '@angular/core';\nimport { Location } from '@angular/common';\nimport { FormGroup } from '@angular/forms';\nimport { FormElement } from '../../interfaces';\nimport { NgxFormService } from '../../engine/NgxFormService';\nimport { CrudFormEvent, Dynamic, EventConstants, FieldUpdateMode, HTMLFormTarget, RenderedModel } from '../../engine';\nimport { CrudFormOptions } from './types';\nimport { CrudOperations, OperationKeys } from '@decaf-ts/db-decorators';\nimport { DefaultFormReactiveOptions } from './constants';\nimport { ForAngularModule, getLogger } from 'src/lib/for-angular.module';\nimport { IonIcon } from '@ionic/angular/standalone';\nimport { Model } from '@decaf-ts/decorator-validation';\nimport { Logger } from '@decaf-ts/logging';\n\n\n/**\n * @component CrudFormComponent\n * @example <ngx-decaf-crud-form\n *   action=\"create\"\n *   operation=\"create\"\n *   formGroup=\"formGroup\"\n *   rendererId=\"rendererId\"\n *   submitEvent=\"submitEvent\"\n *   target=\"_self\"\n *   method=\"event\">\n * </ngx-decaf-crud-form>\n *\n * @param {string} action - The action to be performed (create, read, update, delete)\n * @param {CrudOperations} operation - The CRUD operation being performed (create, read, update, delete)\n * @param {FormGroup} formGroup - The form group\n * @param {string} rendererId - The renderer id\n * @param {SubmitEvent} submitEvent - The submit event\n * @param {string} target - The target\n * @param {string} method - The method\n */\n@Dynamic()\n@Component({\n  standalone: true,\n  selector: 'ngx-decaf-crud-form',\n  templateUrl: './crud-form.component.html',\n  styleUrls: ['./crud-form.component.scss'],\n  imports: [ForAngularModule, IonIcon],\n})\nexport class CrudFormComponent implements OnInit, FormElement, OnDestroy, RenderedModel {\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   */\n  @Input()\n  model!: Model | undefined;\n\n  @Input()\n  updateOn: FieldUpdateMode = 'change';\n\n  @ViewChild('reactiveForm', { static: false, read: ElementRef })\n  component!: ElementRef;\n\n  @Input()\n  target: HTMLFormTarget = '_self';\n\n  @Input()\n  method: 'get' | 'post' | 'event' = 'event';\n\n  @Input()\n  options!: CrudFormOptions;\n\n  @Input()\n  action?: string;\n\n  @Input({ required: true })\n  operation!: CrudOperations;\n\n  @Input()\n  handlers!: Record<string, (...args: unknown[]) => unknown | Promise<unknown>>;\n\n  @Input()\n  formGroup!: FormGroup | undefined;\n\n  /**\n   * @description Path to the parent FormGroup, if nested.\n   * @summary Full dot-delimited path of the parent FormGroup. Set only when is part of a nested structure.\n   *\n   * @type {string}\n   * @memberOf CrudFormComponent\n   */\n  @Input()\n  childOf?: string;\n\n  @Input()\n  rendererId!: string;\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   */\n  @Input()\n  uid!: string | number | undefined;\n\n\n  @Output()\n  submitEvent = new EventEmitter<CrudFormEvent>();\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 CrudFormComponent\n   */\n  private logger!: Logger;\n\n  /**\n   * @description Angular Location service.\n   * @summary Injected service that provides access to the browser's URL and history.\n   * This service is used for interacting with the browser's history API, allowing\n   * for back navigation and URL manipulation outside of Angular's router.\n   *\n   * @private\n   * @type {Location}\n   * @memberOf CrudFormComponent\n   */\n  private location: Location = inject(Location);\n\n  // ngAfterViewInit() {\n    // if (![OperationKeys.READ, OperationKeys.DELETE].includes(this.operation))\n    //   NgxFormService.formAfterViewInit(this, this.rendererId);\n  // }\n\n  async ngOnInit() {\n    if (!this.logger)\n      this.logger = getLogger(this);\n    if (this.operation === OperationKeys.READ || this.operation === OperationKeys.DELETE)\n      this.formGroup = undefined;\n    this.options = Object.assign(\n      {},\n      DefaultFormReactiveOptions,\n      this.options || {},\n    );\n\n  }\n\n  ngOnDestroy() {\n    if (this.formGroup)\n      NgxFormService.unregister(this.formGroup);\n  }\n\n  /**\n   * @param  {SubmitEvent} event\n   */\n  async submit(event: SubmitEvent): Promise<boolean | void> {\n    event.preventDefault();\n    event.stopImmediatePropagation();\n    event.stopPropagation();\n\n    if (!NgxFormService.validateFields(this.formGroup as FormGroup))\n      return false;\n\n    const data = NgxFormService.getFormData(this.formGroup as FormGroup);\n    this.submitEvent.emit({\n      data,\n      component: 'CrudFormComponent',\n      name: this.action || EventConstants.SUBMIT_EVENT,\n      handlers: this.handlers,\n    });\n  }\n\n  handleReset() {\n    this.location.back();\n    // if(OperationKeys.DELETE !== this.operation)\n    //   NgxFormService.reset(this.formGroup);\n    // else\n    //   this.location.back();\n  }\n\n  handleDelete() {\n    this.submitEvent.emit({\n      data: this.uid,\n      component: 'CrudFormComponent',\n      name: EventConstants.SUBMIT_EVENT,\n    });\n  }\n\n  protected readonly OperationKeys = OperationKeys;\n}\n","@if(operation !== 'read' && operation !== 'delete') {\n  <form #reactiveForm [id]=\"rendererId\" [formGroup]=\"formGroup\" (submit)=\"submit($event)\" novalidate [target]=\"target\">\n    <ng-content #formContent></ng-content>\n    <div class=\"dcf-flex dcf-flex-right\">\n      <div>\n        @if(options.buttons.clear) {\n          <ion-button fill=\"clear\" (click)=\"handleReset()\">\n            @if(options.buttons.clear?.icon) {\n              <ion-icon  aria-hidden=\"true\" [slot]=\"options.buttons.clear?.iconSlot\" [name]=\"options.buttons.clear?.icon\"></ion-icon>\n            }\n            {{operation === 'update' ? 'Back' : options.buttons.clear?.text}}\n          </ion-button>\n        }\n\n        <ion-button\n          type=\"submit\">\n          @if(options.buttons.submit.icon) {\n            <ion-icon aria-hidden=\"true\" [slot]=\"options.buttons.submit.iconSlot\" [name]=\"options.buttons.submit.icon\"></ion-icon>\n          }\n          {{ action ? action : options.buttons.submit.text}}\n        </ion-button>\n      </div>\n    </div>\n  </form>\n} @else {\n  <div [class]=\"'dcf-flex dcf-flex-right ' + operation\" id=\"dcf-buttons-container\">\n    <div>\n      @if(options.buttons.clear) {\n        <ion-button fill=\"clear\" (click)=\"handleReset()\">\n          @if(options.buttons.clear?.icon) {\n            <ion-icon  aria-hidden=\"true\" [slot]=\"options.buttons.clear?.iconSlot\" [name]=\"options.buttons.clear?.icon\"></ion-icon>\n          }\n          {{ ['delete', 'read', 'update'].includes(operation) ? 'Back' : options.buttons.clear?.text}}\n        </ion-button>\n      }\n      @if(operation === 'delete' && uid) {\n        <ion-button\n          (click)=\"handleDelete()\"\n          color=\"danger\"\n          type=\"button\">\n          @if(options.buttons.submit.icon) {\n            <ion-icon aria-hidden=\"true\" [slot]=\"options.buttons.submit.iconSlot\" [name]=\"options.buttons.submit.icon\"></ion-icon>\n          }\n          Delete\n       </ion-button>\n      }\n      @if(operation === OperationKeys.CREATE || operation === OperationKeys.UPDATE) {\n        <ion-button\n          type=\"submit\">\n          @if(options.buttons.submit.icon) {\n            <ion-icon aria-hidden=\"true\" [slot]=\"options.buttons.submit.iconSlot\" [name]=\"options.buttons.submit.icon\"></ion-icon>\n          }\n          {{options.buttons.submit.text}}\n        </ion-button>\n      }\n\n    </div>\n  </div>\n}\n\n"]}
@@ -55,7 +55,7 @@ export class ModelRendererComponent {
55
55
  refresh(model) {
56
56
  model =
57
57
  typeof model === 'string'
58
- ? Model.build({}, JSON.parse(model))
58
+ ? Model.build({}, model)
59
59
  : model;
60
60
  this.output = model.render(this.globals || {}, this.vcr, this.injector, this.inner);
61
61
  if (this.output?.inputs)
@@ -114,7 +114,7 @@ export class ModelRendererComponent {
114
114
  }
115
115
  }
116
116
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: ModelRendererComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
117
- static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.13", type: ModelRendererComponent, isStandalone: true, selector: "ngx-decaf-model-renderer", inputs: { model: "model", globals: "globals", rendererId: "rendererId" }, outputs: { listenEvent: "listenEvent" }, 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: " <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: "ngmodule", type: ForAngularModule }, { kind: "directive", type: i1.NgComponentOutlet, selector: "[ngComponentOutlet]", inputs: ["ngComponentOutlet", "ngComponentOutletInputs", "ngComponentOutletInjector", "ngComponentOutletContent", "ngComponentOutletNgModule", "ngComponentOutletNgModuleFactory"] }, { kind: "component", type: ComponentRendererComponent, selector: "ngx-decaf-component-renderer", inputs: ["tag", "globals", "parent"], outputs: ["listenEvent"] }] }); }
117
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.13", type: ModelRendererComponent, isStandalone: true, selector: "ngx-decaf-model-renderer", inputs: { model: "model", globals: "globals", rendererId: "rendererId" }, outputs: { listenEvent: "listenEvent" }, 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: " <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: "ngmodule", type: ForAngularModule }, { kind: "directive", type: i1.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"] }] }); }
118
118
  }
119
119
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: ModelRendererComponent, decorators: [{
120
120
  type: Component,
@@ -135,4 +135,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
135
135
  }], listenEvent: [{
136
136
  type: Output
137
137
  }] } });
138
- //# 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  NgxRenderingEngine2,\n  RenderedModel,\n} from '../../engine';\nimport { KeyValue, ModelRenderCustomEvent } from 'src/lib/engine/types';\nimport { ForAngularModule } from 'src/lib/for-angular.module';\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 NgxRenderingEngine2 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: [ForAngularModule, NgComponentOutlet, ComponentRendererComponent],\n  selector: 'ngx-decaf-model-renderer',\n  templateUrl: './model-renderer.component.html',\n  styleUrl: './model-renderer.component.scss',\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<ModelRenderCustomEvent>();\n\n  /**\n   * @description Instance of the NgxRenderingEngine2\n   */\n  private render!: NgxRenderingEngine2;\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({}, JSON.parse(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 NgxRenderingEngine2.destroy();\n    }\n    this.output = undefined;\n  }\n\n  private subscribeEvents(): 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 as KeyValue)[key].subscribe((event: Partial<BaseCustomEvent>) => {\n            this.listenEvent.emit({\n              component: this.output?.component.name || '',\n              name: key,\n              ...event,\n            } as ModelRenderCustomEvent);\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","  <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"]}
138
+ //# 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  NgxRenderingEngine2,\n  RenderedModel,\n} from '../../engine';\nimport { KeyValue, RendererCustomEvent } from 'src/lib/engine/types';\nimport { ForAngularModule } from 'src/lib/for-angular.module';\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 NgxRenderingEngine2 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: [ForAngularModule, NgComponentOutlet, ComponentRendererComponent],\n  selector: 'ngx-decaf-model-renderer',\n  templateUrl: './model-renderer.component.html',\n  styleUrl: './model-renderer.component.scss',\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 NgxRenderingEngine2\n   */\n  private render!: NgxRenderingEngine2;\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 NgxRenderingEngine2.destroy();\n    }\n    this.output = undefined;\n  }\n\n  private subscribeEvents(): 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 as KeyValue)[key].subscribe((event: Partial<BaseCustomEvent>) => {\n            this.listenEvent.emit({\n              component: this.output?.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","  <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"]}