@decaf-ts/for-angular 0.0.3 → 0.0.5

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 (87) hide show
  1. package/LICENSE.md +659 -21
  2. package/README.md +37 -242
  3. package/dist/lib/README.md +92 -0
  4. package/dist/lib/assets/i18n/en.json +131 -0
  5. package/dist/lib/assets/images/angular-logo.svg +45 -0
  6. package/dist/lib/assets/images/decaf-logo-black.svg +22 -0
  7. package/dist/lib/assets/images/decaf-logo-lw.svg +50 -0
  8. package/dist/lib/assets/images/decaf-logo-white.svg +22 -0
  9. package/dist/lib/assets/images/decaf-logo.svg +54 -0
  10. package/dist/lib/components/component-renderer/component-renderer.component.d.ts +267 -0
  11. package/dist/lib/components/crud-field/crud-field.component.d.ts +447 -0
  12. package/dist/lib/components/crud-form/crud-form.component.d.ts +102 -0
  13. package/dist/lib/components/model-renderer/model-renderer.component.d.ts +97 -0
  14. package/dist/lib/engine/DynamicModule.d.ts +17 -0
  15. package/dist/{for-angular → lib}/engine/NgxCrudFormField.d.ts +37 -30
  16. package/dist/lib/engine/NgxFormService.d.ts +167 -0
  17. package/dist/lib/engine/NgxRenderingEngine.d.ts +128 -0
  18. package/dist/lib/engine/NgxRenderingEngine2.d.ts +251 -0
  19. package/dist/lib/engine/ValidatorFactory.d.ts +15 -0
  20. package/dist/lib/engine/constants.d.ts +151 -0
  21. package/dist/lib/engine/decorators.d.ts +25 -0
  22. package/dist/lib/engine/index.d.ts +15 -0
  23. package/dist/lib/engine/types.d.ts +293 -0
  24. package/dist/lib/esm2022/components/component-renderer/component-renderer.component.mjs +309 -0
  25. package/dist/lib/esm2022/components/crud-field/crud-field.component.mjs +288 -0
  26. package/dist/lib/esm2022/components/crud-form/constants.mjs +14 -0
  27. package/dist/lib/esm2022/components/crud-form/crud-form.component.mjs +140 -0
  28. package/dist/lib/esm2022/components/crud-form/types.mjs +2 -0
  29. package/dist/lib/esm2022/components/model-renderer/model-renderer.component.mjs +137 -0
  30. package/dist/lib/esm2022/engine/DynamicModule.mjs +18 -0
  31. package/dist/lib/esm2022/engine/NgxCrudFormField.mjs +117 -0
  32. package/dist/lib/esm2022/engine/NgxFormService.mjs +315 -0
  33. package/dist/lib/esm2022/engine/NgxRenderingEngine.mjs +194 -0
  34. package/dist/lib/esm2022/engine/NgxRenderingEngine2.mjs +333 -0
  35. package/dist/lib/esm2022/engine/ValidatorFactory.mjs +102 -0
  36. package/dist/lib/esm2022/engine/constants.mjs +160 -0
  37. package/dist/lib/esm2022/engine/decorators.mjs +38 -0
  38. package/dist/lib/esm2022/engine/index.mjs +16 -0
  39. package/dist/lib/esm2022/engine/types.mjs +2 -0
  40. package/dist/lib/esm2022/for-angular.module.mjs +118 -0
  41. package/dist/lib/esm2022/interfaces.mjs +2 -0
  42. package/dist/lib/esm2022/public-apis.mjs +13 -0
  43. package/dist/lib/fesm2022/decaf-ts-for-angular.mjs +2138 -0
  44. package/dist/lib/fesm2022/decaf-ts-for-angular.mjs.map +1 -0
  45. package/dist/lib/for-angular.module.d.ts +44 -0
  46. package/dist/lib/interfaces.d.ts +28 -0
  47. package/dist/lib/public-apis.d.ts +12 -0
  48. package/package.json +74 -25
  49. package/dist/for-angular/README.md +0 -297
  50. package/dist/for-angular/assets/i18n/en.json +0 -21
  51. package/dist/for-angular/components/decaf-crud-field/decaf-crud-field.component.d.ts +0 -22
  52. package/dist/for-angular/components/decaf-crud-form/decaf-crud-form.component.d.ts +0 -28
  53. package/dist/for-angular/components/decaf-model-renderer/decaf-model-renderer.component.d.ts +0 -20
  54. package/dist/for-angular/directives/decaf-field.directive.d.ts +0 -8
  55. package/dist/for-angular/engine/DynamicModule.d.ts +0 -2
  56. package/dist/for-angular/engine/NgxFormService.d.ts +0 -119
  57. package/dist/for-angular/engine/NgxRenderingEngine.d.ts +0 -17
  58. package/dist/for-angular/engine/ValidatorFactory.d.ts +0 -4
  59. package/dist/for-angular/engine/constants.d.ts +0 -10
  60. package/dist/for-angular/engine/decorators.d.ts +0 -1
  61. package/dist/for-angular/engine/index.d.ts +0 -5
  62. package/dist/for-angular/engine/types.d.ts +0 -32
  63. package/dist/for-angular/esm2022/components/decaf-crud-field/decaf-crud-field.component.mjs +0 -66
  64. package/dist/for-angular/esm2022/components/decaf-crud-form/constants.mjs +0 -14
  65. package/dist/for-angular/esm2022/components/decaf-crud-form/decaf-crud-form.component.mjs +0 -84
  66. package/dist/for-angular/esm2022/components/decaf-crud-form/types.mjs +0 -2
  67. package/dist/for-angular/esm2022/components/decaf-model-renderer/decaf-model-renderer.component.mjs +0 -46
  68. package/dist/for-angular/esm2022/directives/decaf-field.directive.mjs +0 -23
  69. package/dist/for-angular/esm2022/engine/DynamicModule.mjs +0 -3
  70. package/dist/for-angular/esm2022/engine/NgxCrudFormField.mjs +0 -118
  71. package/dist/for-angular/esm2022/engine/NgxFormService.mjs +0 -232
  72. package/dist/for-angular/esm2022/engine/NgxRenderingEngine.mjs +0 -35
  73. package/dist/for-angular/esm2022/engine/ValidatorFactory.mjs +0 -48
  74. package/dist/for-angular/esm2022/engine/constants.mjs +0 -12
  75. package/dist/for-angular/esm2022/engine/decorators.mjs +0 -17
  76. package/dist/for-angular/esm2022/engine/index.mjs +0 -6
  77. package/dist/for-angular/esm2022/engine/types.mjs +0 -2
  78. package/dist/for-angular/esm2022/interfaces.mjs +0 -2
  79. package/dist/for-angular/esm2022/public-apis.mjs +0 -5
  80. package/dist/for-angular/fesm2022/decaf-ts-for-angular.mjs +0 -675
  81. package/dist/for-angular/fesm2022/decaf-ts-for-angular.mjs.map +0 -1
  82. package/dist/for-angular/interfaces.d.ts +0 -8
  83. package/dist/for-angular/public-apis.d.ts +0 -4
  84. /package/dist/{for-angular/components/decaf-crud-form → lib/components/crud-form}/constants.d.ts +0 -0
  85. /package/dist/{for-angular/components/decaf-crud-form → lib/components/crud-form}/types.d.ts +0 -0
  86. /package/dist/{for-angular → lib}/esm2022/decaf-ts-for-angular.mjs +0 -0
  87. /package/dist/{for-angular → lib}/index.d.ts +0 -0
@@ -0,0 +1,293 @@
1
+ import { IonCheckbox, IonInput, IonSelect, IonTextarea } from '@ionic/angular';
2
+ import { TextFieldTypes } from '@ionic/core';
3
+ import { Injector, Type } from '@angular/core';
4
+ import { FormControl, FormGroup } from '@angular/forms';
5
+ import { FieldProperties } from '@decaf-ts/ui-decorators';
6
+ export type KeyValue = Record<string, any>;
7
+ /**
8
+ * @description Element size options for UI components
9
+ * @summary Defines the possible size values that can be applied to UI elements.
10
+ * These sizes control the dimensions and layout behavior of components.
11
+ * @typedef {('small'|'medium'|'large'|'xlarge'|'2xlarge'|'auto'|'expand'|'block')} ElementSizes
12
+ * @memberOf module:engine
13
+ */
14
+ export type ElementSizes = 'small' | 'medium' | 'large' | 'xlarge' | '2xlarge' | 'auto' | 'expand' | 'block';
15
+ /**
16
+ * @description Basic position options for UI elements
17
+ * @summary Defines the possible position values that can be applied to UI elements.
18
+ * These positions control the alignment and placement of components.
19
+ * @typedef {('left'|'center'|'right'|'top'|'bottom')} ElementPositions
20
+ * @memberOf module:engine
21
+ */
22
+ export type ElementPositions = 'left' | 'center' | 'right' | 'top' | 'bottom';
23
+ /**
24
+ * @description Extended position options for flex layouts
25
+ * @summary Extends the basic ElementPositions with additional flex-specific position values.
26
+ * These positions are used for controlling alignment and distribution in flex containers.
27
+ * @typedef {(ElementPositions|'stretch'|'middle'|'around'|'between')} FlexPositions
28
+ * @memberOf module:engine
29
+ */
30
+ export type FlexPositions = ElementPositions | 'stretch' | 'middle' | 'around' | 'between';
31
+ /**
32
+ * @description Update mode options for form fields
33
+ * @summary Defines when form field values should be updated in the model.
34
+ * - 'change': Update on every change event
35
+ * - 'blur': Update when the field loses focus
36
+ * - 'submit': Update only when the form is submitted
37
+ * @typedef {('change'|'blur'|'submit')} FieldUpdateMode
38
+ * @memberOf module:engine
39
+ */
40
+ export type FieldUpdateMode = 'change' | 'blur' | 'submit';
41
+ /**
42
+ * @description Metadata structure for Angular components
43
+ * @summary Defines the structure of metadata for Angular components, including
44
+ * change detection strategy, selector, standalone status, imports, template, and styles.
45
+ * This is used for reflection and dynamic component creation.
46
+ * @interface ComponentMetadata
47
+ * @property {number} changeDetection - The change detection strategy number
48
+ * @property {string} selector - The CSS selector for the component
49
+ * @property {boolean} standalone - Whether the component is standalone
50
+ * @property imports - Array of imported modules/components
51
+ * @property {string} template - The HTML template for the component
52
+ * @property {string[]} styles - Array of CSS styles for the component
53
+ * @memberOf module:engine
54
+ */
55
+ export interface ComponentMetadata {
56
+ changeDetection: number;
57
+ selector: string;
58
+ standalone: boolean;
59
+ imports: (new (...args: unknown[]) => unknown)[];
60
+ template: string;
61
+ styles: string[];
62
+ }
63
+ /**
64
+ * @description Output structure from the Angular rendering engine
65
+ * @summary Defines the structure of the output produced by the NgxRenderingEngine
66
+ * when rendering a component. Contains the component type, inputs, injector,
67
+ * content nodes, and child components.
68
+ * @typedef {Object} AngularDynamicOutput
69
+ * @property {Type<unknown>} component - The Angular component type
70
+ * @property {string} [rendererId] - Optional unique ID for the rendered component
71
+ * @property {Record<string, unknown>} [inputs] - Optional input properties for the component
72
+ * @property {Injector} [injector] - Optional Angular injector for dependency injection
73
+ * @property {Node[][]} [content] - Optional content nodes for projection
74
+ * @property {AngularDynamicOutput[]} [children] - Optional child components
75
+ * @property {Type<unknown>} [instance] - Optional component instance
76
+ * @property {FormGroup} [formGroup] - Optional component FormGroup
77
+ * @property {FormControl} [formControl] - Optional component FormControl
78
+ * @memberOf module:engine
79
+ */
80
+ export interface AngularDynamicOutput {
81
+ component: Type<unknown>;
82
+ rendererId?: string;
83
+ inputs?: Record<string, unknown>;
84
+ injector?: Injector;
85
+ content?: Node[][];
86
+ children?: AngularDynamicOutput[];
87
+ instance?: Type<unknown>;
88
+ formGroup?: FormGroup;
89
+ formControl?: FormControl;
90
+ }
91
+ /**
92
+ * @description Interface for models that can be rendered
93
+ * @summary Defines the basic structure for models that can be rendered by the engine.
94
+ * Contains an optional rendererId that uniquely identifies the rendered instance.
95
+ * @interface RenderedModel
96
+ * @property {string} [rendererId] - Optional unique ID for the rendered model instance
97
+ * @memberOf module:engine
98
+ */
99
+ export interface RenderedModel {
100
+ rendererId?: string;
101
+ }
102
+ /**
103
+ * @description Possible input types for form fields
104
+ * @summary Defines the possible input types that can be used in form fields.
105
+ * Includes standard HTML input types like checkbox, radio, and select,
106
+ * as well as Ionic's TextFieldTypes and textarea.
107
+ * @typedef {('checkbox'|'radio'|'select'|TextFieldTypes|'textarea')} PossibleInputTypes
108
+ * @memberOf module:engine
109
+ */
110
+ export type PossibleInputTypes = 'checkbox' | 'radio' | 'select' | TextFieldTypes | 'textarea';
111
+ /**
112
+ * @description Field definition for Angular components
113
+ * @summary A comprehensive type that combines properties from various Ionic components
114
+ * to define the structure of a field in an Angular form. It omits certain properties
115
+ * from IonInput, picks specific properties from IonSelect, IonTextarea, and IonCheckbox,
116
+ * and adds custom properties like type and className.
117
+ * @typedef {Object} AngularFieldDefinition
118
+ * @property {PossibleInputTypes} type - The type of input field
119
+ * @property {string|string[]} className - CSS class name(s) for the field
120
+ * @property {string} [cancelText] - Text for the cancel button (from IonSelect)
121
+ * @property {string} [interface] - Interface style for select (from IonSelect)
122
+ * @property {string} [selectedText] - Text for selected option (from IonSelect)
123
+ * @property {Object} [interfaceOptions] - Options for the interface (from IonSelect)
124
+ * @property {number} [rows] - Number of rows for textarea (from IonTextarea)
125
+ * @property {number} [cols] - Number of columns for textarea (from IonTextarea)
126
+ * @property {string} [alignment] - Alignment of checkbox (from IonCheckbox)
127
+ * @property {string} [justify] - Justification of checkbox (from IonCheckbox)
128
+ * @property {boolean} [checked] - Whether checkbox is checked (from IonCheckbox)
129
+ * @memberOf module:engine
130
+ */
131
+ export type AngularFieldDefinition = Omit<IonInput, 'ionInput' | 'ionFocus' | 'ionChange' | 'ionBlur' | 'getInputElement' | 'setFocus' | 'label' | 'el' | 'z' | 'type'> & Pick<IonSelect, 'cancelText' | 'interface' | 'selectedText' | 'interfaceOptions'> & Pick<IonTextarea, 'rows' | 'cols'> & Pick<IonCheckbox, 'alignment' | 'justify' | 'checked'> & {
132
+ type: PossibleInputTypes;
133
+ className: string | string[];
134
+ } & Record<string, unknown>;
135
+ /**
136
+ * @description String or boolean representation of a boolean value
137
+ * @summary Represents a value that can be either a boolean or a string representation of a boolean.
138
+ * This is useful for handling attribute values that can be specified as either strings or booleans.
139
+ * @typedef {('true'|'false'|boolean)} StringOrBoolean
140
+ * @memberOf module:engine
141
+ */
142
+ export type StringOrBoolean = 'true' | 'false' | boolean;
143
+ /**
144
+ * @description Option type for select inputs
145
+ * @summary Extends the InputOption interface with a selected property to indicate
146
+ * whether the option is selected by default.
147
+ * @memberOf module:engine
148
+ */
149
+ export type SelectOption = InputOption & {
150
+ selected?: boolean;
151
+ };
152
+ /**
153
+ * @description Option type for radio inputs
154
+ * @summary Extends the InputOption interface with a checked property to indicate
155
+ * whether the option is checked by default.
156
+ * @memberOf module:engine
157
+ */
158
+ export type RadioOption = InputOption & {
159
+ checked?: boolean;
160
+ };
161
+ /**
162
+ * @description Option type for checkbox inputs
163
+ * @summary Alias for RadioOption, as checkbox options have the same structure as radio options.
164
+ * @typedef {RadioOption} CheckboxOption
165
+ * @memberOf module:engine
166
+ */
167
+ export type CheckboxOption = RadioOption;
168
+ /**
169
+ * @description Base option type for input components
170
+ * @summary Defines the common structure for options used in select, radio, and checkbox inputs.
171
+ * Contains properties for the display text, value, disabled state, CSS class, and icon.
172
+ * @interface InputOption
173
+ * @property {string} text - The display text for the option
174
+ * @property {string|number} value - The value associated with the option
175
+ * @property {StringOrBoolean} [disabled] - Whether the option is disabled
176
+ * @property {string} [className] - CSS class name for styling the option
177
+ * @property {string} [icon] - Icon to display with the option
178
+ * @memberOf module:engine
179
+ */
180
+ export interface InputOption {
181
+ text: string;
182
+ value: string | number;
183
+ disabled?: StringOrBoolean;
184
+ className?: string;
185
+ icon?: string;
186
+ }
187
+ /**
188
+ * @description Target options for HTML forms
189
+ * @summary Defines the possible target values for HTML forms, including standard targets
190
+ * like '_blank', '_self', '_parent', and '_top', as well as custom string values.
191
+ * @typedef {('_blank'|'_self'|'_parent'|'_top'|string)} HTMLFormTarget
192
+ * @memberOf module:engine
193
+ */
194
+ export type HTMLFormTarget = '_blank' | '_self' | '_parent' | '_top' | string;
195
+ /**
196
+ * @description Interface for list component refresh events
197
+ * @summary Defines the structure of a refresh event for list components.
198
+ * Contains an array of key-value pairs representing the new data for the list.
199
+ * @interface IListComponentRefreshEvent
200
+ * @property {KeyValue[]} data - Array of key-value pairs representing the new data
201
+ * @memberOf module:engine
202
+ */
203
+ export interface IListComponentRefreshEvent {
204
+ data: KeyValue[];
205
+ }
206
+ export interface FormServiceControl {
207
+ control: FormGroup;
208
+ props: AngularFieldDefinition;
209
+ }
210
+ /**
211
+ * @description Type for form service controls
212
+ * @summary Defines the structure of form controls managed by the form service.
213
+ * It's a nested record where the outer key is the form group name, the inner key
214
+ * is the control name, and the value contains the form group and field properties.
215
+ * @typedef {Record<string, Record<string, { control: FormGroup; props: AngularFieldDefinition }>>} FormServiceControls
216
+ * @memberOf module:engine
217
+ */
218
+ export type FormServiceControls = Record<string, Record<string, FormServiceControl>>;
219
+ /**
220
+ * @description Interface for model render custom events
221
+ * @summary Defines the structure of custom events triggered during model rendering.
222
+ * Contains the event detail, component name, and event name.
223
+ * @interface ModelRenderCustomEvent
224
+ * @property {BaseCustomEvent} detail - The detailed event information
225
+ * @property {string} component - The component that triggered the event
226
+ * @property {string} name - The name of the event
227
+ * @memberOf module:engine
228
+ */
229
+ export interface ModelRenderCustomEvent {
230
+ detail: BaseCustomEvent;
231
+ component: string;
232
+ name: string;
233
+ }
234
+ /**
235
+ * @description Interface for list item custom events
236
+ * @summary Defines the structure of custom events triggered by list items.
237
+ * Extends BaseCustomEvent with additional properties for the action and primary key.
238
+ * @interface ListItemCustomEvent
239
+ * @property {string} action - The action performed on the list item
240
+ * @property {string} [pk] - Optional primary key of the affected item
241
+ * @property {any} data - The data associated with the event (inherited from BaseCustomEvent)
242
+ * @property {HTMLElement} [target] - The target element (inherited from BaseCustomEvent)
243
+ * @property {string} [name] - The name of the event (inherited from BaseCustomEvent)
244
+ * @property {string} component - The component that triggered the event (inherited from BaseCustomEvent)
245
+ * @memberOf module:engine
246
+ */
247
+ export interface ListItemCustomEvent extends BaseCustomEvent {
248
+ action: string;
249
+ pk?: string;
250
+ }
251
+ /**
252
+ * @description Base interface for custom events
253
+ * @summary Defines the base structure for custom events in the application.
254
+ * Contains properties for the event data, target element, name, and component.
255
+ * @interface BaseCustomEvent
256
+ * @property {any} data - The data associated with the event
257
+ * @property {HTMLElement} [target] - The target element that triggered the event
258
+ * @property {string} [name] - The name of the event
259
+ * @property {string} component - The component that triggered the event
260
+ * @memberOf module:engine
261
+ */
262
+ export interface BaseCustomEvent {
263
+ data: any;
264
+ target?: HTMLElement;
265
+ name?: string;
266
+ component: string;
267
+ }
268
+ /**
269
+ * @description Base interface for custom events
270
+ * @summary Defines the base structure for custom events in the application.
271
+ * Contains properties for the event data, target element, name, and component.
272
+ * @interface BaseCustomEvent
273
+ * @property {any} data - The data associated with the event
274
+ * @property {HTMLElement} [target] - The target element that triggered the event
275
+ * @property {string} [name] - The name of the event
276
+ * @property {string} component - The component that triggered the event
277
+ * @memberOf module:engine
278
+ */
279
+ export type CrudFormEvent = BaseCustomEvent & {
280
+ handlers?: Record<string, any>;
281
+ };
282
+ export interface ComponentInput extends FieldProperties {
283
+ updateMode?: FieldUpdateMode;
284
+ formGroup?: FormGroup;
285
+ formControl?: FormControl;
286
+ }
287
+ export interface ComponentConfig {
288
+ component: string;
289
+ inputs: ComponentInput;
290
+ injector: any;
291
+ children?: ComponentConfig[];
292
+ }
293
+ export type FormParentGroup = [FormGroup, string];
@@ -0,0 +1,309 @@
1
+ import { Component, EnvironmentInjector, EventEmitter, inject, Input, Output, reflectComponentType, TemplateRef, ViewChild, ViewContainerRef, } from '@angular/core';
2
+ import { NgxRenderingEngine2 } from 'src/lib/engine/NgxRenderingEngine2';
3
+ import { ForAngularModule, getLogger } from 'src/lib/for-angular.module';
4
+ import * as i0 from "@angular/core";
5
+ import * as i1 from "@angular/common";
6
+ /**
7
+ * @description Dynamic component renderer for Decaf Angular applications.
8
+ * @summary This component provides a flexible way to dynamically render Angular components
9
+ * at runtime based on a tag name. It handles the creation, property binding, and event
10
+ * subscription for dynamically loaded components. This is particularly useful for
11
+ * building configurable UIs where components need to be determined at runtime.
12
+ *
13
+ * @component {ComponentRendererComponent}
14
+ * @example
15
+ * <ngx-decaf-component-renderer
16
+ * [tag]="tag"
17
+ * [globals]="globals"
18
+ * (listenEvent)="listenEvent($event)">
19
+ * </ngx-decaf-component-renderer>
20
+ *
21
+ * @mermaid
22
+ * classDiagram
23
+ * class ComponentRendererComponent {
24
+ * +ViewContainerRef vcr
25
+ * +string tag
26
+ * +Record~string, unknown~ globals
27
+ * +EnvironmentInjector injector
28
+ * +ComponentRef~unknown~ component
29
+ * +EventEmitter~ModelRenderCustomEvent~ listenEvent
30
+ * +ngOnInit()
31
+ * +ngOnDestroy()
32
+ * +ngOnChanges(changes)
33
+ * -createComponent(tag, globals)
34
+ * -subscribeEvents()
35
+ * -unsubscribeEvents()
36
+ * }
37
+ * ComponentRendererComponent --|> OnInit
38
+ * ComponentRendererComponent --|> OnChanges
39
+ * ComponentRendererComponent --|> OnDestroy
40
+ *
41
+ * @implements {OnInit}
42
+ * @implements {OnChanges}
43
+ * @implements {OnDestroy}
44
+ */
45
+ export class ComponentRendererComponent {
46
+ /**
47
+ * @description Creates an instance of ComponentRendererComponent.
48
+ * @summary Initializes a new ComponentRendererComponent. This component doesn't require
49
+ * any dependencies to be injected in its constructor as it uses the inject function to
50
+ * obtain the EnvironmentInjector.
51
+ *
52
+ * @memberOf ComponentRendererComponent
53
+ */
54
+ constructor() {
55
+ /**
56
+ * @description Global properties to pass to the rendered component.
57
+ * @summary This input property allows passing a set of properties to the dynamically
58
+ * rendered component. These properties will be mapped to the component's inputs if they
59
+ * match. Properties that don't match any input on the target component will be filtered out
60
+ * with a warning.
61
+ *
62
+ * @type {Record<string, unknown>}
63
+ * @default {}
64
+ * @memberOf ComponentRendererComponent
65
+ */
66
+ this.globals = {};
67
+ /**
68
+ * @description Injector used for dependency injection in the dynamic component.
69
+ * @summary This injector is used when creating the dynamic component to provide it with
70
+ * access to the application's dependency injection system. It ensures that the dynamically
71
+ * created component can access the same services and dependencies as statically created
72
+ * components.
73
+ *
74
+ * @type {EnvironmentInjector}
75
+ * @memberOf ComponentRendererComponent
76
+ */
77
+ this.injector = inject(EnvironmentInjector);
78
+ /**
79
+ * @description Event emitter for events from the rendered component.
80
+ * @summary This output property emits events that originate from the dynamically rendered
81
+ * component. It allows the parent component to listen for and respond to events from the
82
+ * dynamic component, creating a communication channel between the parent and the dynamically
83
+ * rendered child.
84
+ *
85
+ * @type {EventEmitter<ModelRenderCustomEvent>}
86
+ * @memberOf ComponentRendererComponent
87
+ */
88
+ this.listenEvent = new EventEmitter();
89
+ this.parent = undefined;
90
+ this.logger = getLogger(this);
91
+ }
92
+ /**
93
+ * @description Initializes the component after Angular first displays the data-bound properties.
94
+ * @summary Sets up the component by creating the dynamic component specified by the tag input.
95
+ * This method is called once when the component is initialized and triggers the dynamic
96
+ * component creation process with the provided tag name and global properties.
97
+ *
98
+ * @mermaid
99
+ * sequenceDiagram
100
+ * participant A as Angular Lifecycle
101
+ * participant C as ComponentRendererComponent
102
+ * participant R as NgxRenderingEngine2
103
+ *
104
+ * A->>C: ngOnInit()
105
+ * C->>C: createComponent(tag, globals)
106
+ * C->>R: components(tag)
107
+ * R-->>C: Return component constructor
108
+ * C->>C: Process component inputs
109
+ * C->>C: Create component instance
110
+ * C->>C: subscribeEvents()
111
+ *
112
+ * @return {void}
113
+ * @memberOf ComponentRendererComponent
114
+ */
115
+ ngOnInit() {
116
+ if (!this.parent)
117
+ this.createComponent(this.tag, this.globals);
118
+ this.createParentComponent();
119
+ }
120
+ /**
121
+ * @description Cleans up resources when the component is destroyed.
122
+ * @summary Performs cleanup operations when the component is being destroyed by Angular.
123
+ * This includes unsubscribing from all event emitters of the dynamic component and
124
+ * destroying the rendering engine instance to prevent memory leaks.
125
+ *
126
+ * @mermaid
127
+ * sequenceDiagram
128
+ * participant A as Angular Lifecycle
129
+ * participant C as ComponentRendererComponent
130
+ * participant R as NgxRenderingEngine2
131
+ *
132
+ * A->>C: ngOnDestroy()
133
+ * alt component exists
134
+ * C->>C: unsubscribeEvents()
135
+ * C->>R: destroy()
136
+ * end
137
+ *
138
+ * @return {Promise<void>} A promise that resolves when cleanup is complete
139
+ * @memberOf ComponentRendererComponent
140
+ */
141
+ async ngOnDestroy() {
142
+ if (this.component) {
143
+ this.unsubscribeEvents();
144
+ NgxRenderingEngine2.destroy();
145
+ }
146
+ }
147
+ /**
148
+ * @description Creates and renders a dynamic component.
149
+ * @summary This method handles the creation of a dynamic component based on the provided tag.
150
+ * It retrieves the component constructor from the rendering engine, processes its inputs,
151
+ * filters out unmapped properties, creates the component instance, and sets up event subscriptions.
152
+ *
153
+ * @param {string} tag - The tag name of the component to create
154
+ * @param {KeyValue} globals - Global properties to pass to the component
155
+ * @return {void}
156
+ *
157
+ * @mermaid
158
+ * sequenceDiagram
159
+ * participant C as ComponentRendererComponent
160
+ * participant R as NgxRenderingEngine2
161
+ * participant V as ViewContainerRef
162
+ *
163
+ * C->>R: components(tag)
164
+ * R-->>C: Return component constructor
165
+ * C->>C: reflectComponentType(component)
166
+ * C->>C: Process input properties
167
+ * C->>C: Filter unmapped properties
168
+ * C->>V: clear()
169
+ * C->>R: createComponent(component, props, metadata, vcr, injector, [])
170
+ * R-->>C: Return component reference
171
+ * C->>C: subscribeEvents()
172
+ *
173
+ * @private
174
+ * @memberOf ComponentRendererComponent
175
+ */
176
+ createComponent(tag, globals = {}) {
177
+ const component = NgxRenderingEngine2.components(tag)
178
+ ?.constructor;
179
+ const metadata = reflectComponentType(component);
180
+ const componentInputs = metadata.inputs;
181
+ const props = globals?.['item'];
182
+ delete props['tag'];
183
+ const inputKeys = Object.keys(props);
184
+ const unmappedKeys = [];
185
+ for (const input of inputKeys) {
186
+ if (!inputKeys.length)
187
+ break;
188
+ const prop = componentInputs.find((item) => item.propName === input);
189
+ if (!prop) {
190
+ delete props[input];
191
+ unmappedKeys.push(input);
192
+ }
193
+ }
194
+ if (unmappedKeys.length)
195
+ this.logger.info(`Unmapped input properties for component ${tag}: ${unmappedKeys.join(', ')}`);
196
+ this.vcr.clear();
197
+ this.component = NgxRenderingEngine2.createComponent(component, props, metadata, this.vcr, this.injector, []);
198
+ this.subscribeEvents();
199
+ }
200
+ createParentComponent() {
201
+ const { component, inputs } = this.parent;
202
+ const metadata = reflectComponentType(component);
203
+ const template = this.vcr.createEmbeddedView(this.inner, this.injector).rootNodes;
204
+ this.component = NgxRenderingEngine2.createComponent(component, inputs, metadata, this.vcr, this.injector, template);
205
+ this.subscribeEvents();
206
+ }
207
+ /**
208
+ * @description Subscribes to events emitted by the dynamic component.
209
+ * @summary This method sets up subscriptions to all EventEmitter properties of the
210
+ * dynamically created component. When an event is emitted by the dynamic component,
211
+ * it is captured and re-emitted through the listenEvent output property with additional
212
+ * metadata about the event source.
213
+ *
214
+ * @mermaid
215
+ * sequenceDiagram
216
+ * participant C as ComponentRendererComponent
217
+ * participant D as Dynamic Component
218
+ * participant P as Parent Component
219
+ *
220
+ * C->>C: subscribeEvents()
221
+ * C->>D: Get instance properties
222
+ * loop For each property
223
+ * C->>C: Check if property is EventEmitter
224
+ * alt is EventEmitter
225
+ * C->>D: Subscribe to event
226
+ * D-->>C: Event emitted
227
+ * C->>P: Re-emit event with metadata
228
+ * end
229
+ * end
230
+ *
231
+ * @private
232
+ * @return {void}
233
+ * @memberOf ComponentRendererComponent
234
+ */
235
+ subscribeEvents() {
236
+ if (this.component) {
237
+ const instance = this.component?.instance;
238
+ const componentKeys = Object.keys(instance);
239
+ for (const key of componentKeys) {
240
+ const value = instance[key];
241
+ if (value instanceof EventEmitter)
242
+ instance[key].subscribe((event) => {
243
+ this.listenEvent.emit({
244
+ name: key,
245
+ ...event,
246
+ });
247
+ });
248
+ }
249
+ }
250
+ }
251
+ /**
252
+ * @description Unsubscribes from all events of the dynamic component.
253
+ * @summary This method cleans up event subscriptions when the component is being destroyed.
254
+ * It iterates through all properties of the dynamic component instance and unsubscribes
255
+ * from any EventEmitter properties to prevent memory leaks and unexpected behavior after
256
+ * the component is destroyed.
257
+ *
258
+ * @mermaid
259
+ * sequenceDiagram
260
+ * participant C as ComponentRendererComponent
261
+ * participant D as Dynamic Component
262
+ *
263
+ * C->>C: unsubscribeEvents()
264
+ * C->>D: Get instance properties
265
+ * loop For each property
266
+ * C->>C: Check if property is EventEmitter
267
+ * alt is EventEmitter
268
+ * C->>D: Unsubscribe from event
269
+ * end
270
+ * end
271
+ *
272
+ * @private
273
+ * @return {void}
274
+ * @memberOf ComponentRendererComponent
275
+ */
276
+ unsubscribeEvents() {
277
+ if (this.component) {
278
+ const instance = this.component?.instance;
279
+ const componentKeys = Object.keys(instance);
280
+ for (const key of componentKeys) {
281
+ const value = instance[key];
282
+ if (value instanceof EventEmitter)
283
+ instance[key].unsubscribe();
284
+ }
285
+ }
286
+ }
287
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: ComponentRendererComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
288
+ 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"] }] }); }
289
+ }
290
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: ComponentRendererComponent, decorators: [{
291
+ type: Component,
292
+ args: [{ selector: 'ngx-decaf-component-renderer', imports: [ForAngularModule], standalone: true, 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" }]
293
+ }], ctorParameters: () => [], propDecorators: { vcr: [{
294
+ type: ViewChild,
295
+ args: ['componentViewContainer', { static: true, read: ViewContainerRef }]
296
+ }], tag: [{
297
+ type: Input,
298
+ args: [{ required: true }]
299
+ }], globals: [{
300
+ type: Input
301
+ }], listenEvent: [{
302
+ type: Output
303
+ }], parent: [{
304
+ type: Input
305
+ }], inner: [{
306
+ type: ViewChild,
307
+ args: ['inner', { read: TemplateRef, static: true }]
308
+ }] } });
309
+ //# 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":"AAAA,OAAO,EACL,SAAS,EAGT,mBAAmB,EACnB,YAAY,EACZ,MAAM,EAEN,KAAK,EAGL,MAAM,EACN,oBAAoB,EACpB,WAAW,EAEX,SAAS,EACT,gBAAgB,GACjB,MAAM,eAAe,CAAC;AACvB,OAAO,EAAE,mBAAmB,EAAE,MAAM,oCAAoC,CAAC;AAEzE,OAAO,EAAE,gBAAgB,EAAE,SAAS,EAAE,MAAM,4BAA4B,CAAC;;;AAGzE;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAsCG;AAQH,MAAM,OAAO,0BAA0B;IAkGrC;;;;;;;OAOG;IACH;QA9EA;;;;;;;;;;WAUG;QAEH,YAAO,GAA4B,EAAE,CAAC;QAEtC;;;;;;;;;WASG;QACH,aAAQ,GAAwB,MAAM,CAAC,mBAAmB,CAAC,CAAC;QAa5D;;;;;;;;;WASG;QAEH,gBAAW,GACT,IAAI,YAAY,EAA0B,CAAC;QAe7C,WAAM,GAAQ,SAAS,CAAC;QAetB,IAAI,CAAC,MAAM,GAAG,SAAS,CAAC,IAAI,CAAC,CAAC;IAChC,CAAC;IAED;;;;;;;;;;;;;;;;;;;;;;OAsBG;IACH,QAAQ;QACN,IAAI,CAAC,IAAI,CAAC,MAAM;YACd,IAAI,CAAC,eAAe,CAAC,IAAI,CAAC,GAAG,EAAE,IAAI,CAAC,OAAO,CAAC,CAAC;QAC/C,IAAI,CAAC,qBAAqB,EAAE,CAAC;IAC/B,CAAC;IAED;;;;;;;;;;;;;;;;;;;;OAoBG;IACH,KAAK,CAAC,WAAW;QACf,IAAI,IAAI,CAAC,SAAS,EAAE,CAAC;YACnB,IAAI,CAAC,iBAAiB,EAAE,CAAC;YACzB,mBAAmB,CAAC,OAAO,EAAE,CAAC;QAChC,CAAC;IACH,CAAC;IAED;;;;;;;;;;;;;;;;;;;;;;;;;;;;OA4BG;IACK,eAAe,CAAC,GAAW,EAAE,UAAoB,EAAE;QACzD,MAAM,SAAS,GAAG,mBAAmB,CAAC,UAAU,CAAC,GAAG,CAAC;YACnD,EAAE,WAA4B,CAAC;QACjC,MAAM,QAAQ,GAAG,oBAAoB,CAAC,SAAS,CAAC,CAAC;QACjD,MAAM,eAAe,GAAI,QAAqC,CAAC,MAAM,CAAC;QACtE,MAAM,KAAK,GAAG,OAAO,EAAE,CAAC,MAAM,CAAC,CAAC;QAChC,OAAO,KAAK,CAAC,KAAK,CAAC,CAAC;QACpB,MAAM,SAAS,GAAG,MAAM,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;QACrC,MAAM,YAAY,GAAG,EAAE,CAAC;QAExB,KAAK,MAAM,KAAK,IAAI,SAAS,EAAE,CAAC;YAC9B,IAAI,CAAC,SAAS,CAAC,MAAM;gBAAE,MAAM;YAC7B,MAAM,IAAI,GAAG,eAAe,CAAC,IAAI,CAC/B,CAAC,IAA0B,EAAE,EAAE,CAAC,IAAI,CAAC,QAAQ,KAAK,KAAK,CACxD,CAAC;YACF,IAAI,CAAC,IAAI,EAAE,CAAC;gBACV,OAAO,KAAK,CAAC,KAAK,CAAC,CAAC;gBACpB,YAAY,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;YAC3B,CAAC;QACH,CAAC;QACD,IAAI,YAAY,CAAC,MAAM;YACrB,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,2CAA2C,GAAG,KAAK,YAAY,CAAC,IAAI,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;QAEjG,IAAI,CAAC,GAAG,CAAC,KAAK,EAAE,CAAC;QACjB,IAAI,CAAC,SAAS,GAAG,mBAAmB,CAAC,eAAe,CAClD,SAAS,EACT,KAAK,EACL,QAAoC,EACpC,IAAI,CAAC,GAAG,EACR,IAAI,CAAC,QAAoB,EACzB,EAAE,CACH,CAAC;QACF,IAAI,CAAC,eAAe,EAAE,CAAC;IACzB,CAAC;IAED,qBAAqB;QACnB,MAAM,EAAE,SAAS,EAAE,MAAM,EAAE,GAAG,IAAI,CAAC,MAAM,CAAC;QAC1C,MAAM,QAAQ,GAAG,oBAAoB,CAAC,SAAS,CAA6B,CAAC;QAC7E,MAAM,QAAQ,GAAG,IAAI,CAAC,GAAG,CAAC,kBAAkB,CAAC,IAAI,CAAC,KAAyB,EAAE,IAAI,CAAC,QAAQ,CAAC,CAAC,SAAS,CAAC;QACtG,IAAI,CAAC,SAAS,GAAG,mBAAmB,CAAC,eAAe,CAClD,SAAS,EACT,MAAM,EACN,QAAQ,EACR,IAAI,CAAC,GAAG,EACR,IAAI,CAAC,QAAQ,EACb,QAAQ,CACT,CAAC;QACF,IAAI,CAAC,eAAe,EAAE,CAAC;IACzB,CAAC;IAED;;;;;;;;;;;;;;;;;;;;;;;;;;;OA2BG;IACK,eAAe;QACrB,IAAI,IAAI,CAAC,SAAS,EAAE,CAAC;YACnB,MAAM,QAAQ,GAAG,IAAI,CAAC,SAAS,EAAE,QAAe,CAAC;YACjD,MAAM,aAAa,GAAG,MAAM,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;YAC5C,KAAK,MAAM,GAAG,IAAI,aAAa,EAAE,CAAC;gBAChC,MAAM,KAAK,GAAG,QAAQ,CAAC,GAAG,CAAC,CAAC;gBAC5B,IAAI,KAAK,YAAY,YAAY;oBAC9B,QAAqB,CAAC,GAAG,CAAC,CAAC,SAAS,CACnC,CAAC,KAA+B,EAAE,EAAE;wBAClC,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC;4BACpB,IAAI,EAAE,GAAG;4BACT,GAAG,KAAK;yBACiB,CAAC,CAAC;oBAC/B,CAAC,CACF,CAAC;YACN,CAAC;QACH,CAAC;IACH,CAAC;IAED;;;;;;;;;;;;;;;;;;;;;;;;OAwBG;IACK,iBAAiB;QACvB,IAAI,IAAI,CAAC,SAAS,EAAE,CAAC;YACnB,MAAM,QAAQ,GAAG,IAAI,CAAC,SAAS,EAAE,QAAe,CAAC;YACjD,MAAM,aAAa,GAAG,MAAM,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;YAC5C,KAAK,MAAM,GAAG,IAAI,aAAa,EAAE,CAAC;gBAChC,MAAM,KAAK,GAAG,QAAQ,CAAC,GAAG,CAAC,CAAC;gBAC5B,IAAI,KAAK,YAAY,YAAY;oBAAE,QAAQ,CAAC,GAAG,CAAC,CAAC,WAAW,EAAE,CAAC;YACjE,CAAC;QACH,CAAC;IACH,CAAC;+GAvUU,0BAA0B;mGAA1B,0BAA0B,yRAWsB,gBAAgB,uGAoF/C,WAAW,2CCpKzC,+mBAsBA,0DD+Ca,0BAA0B,wIAH3B,gBAAgB;;4FAGf,0BAA0B;kBAPtC,SAAS;+BACE,8BAA8B,WAG/B,CAAC,gBAAgB,CAAC,cACf,IAAI;wDAchB,GAAG;sBADF,SAAS;uBAAC,wBAAwB,EAAE,EAAE,MAAM,EAAE,IAAI,EAAE,IAAI,EAAE,gBAAgB,EAAE;gBAe7E,GAAG;sBADF,KAAK;uBAAC,EAAE,QAAQ,EAAE,IAAI,EAAE;gBAezB,OAAO;sBADN,KAAK;gBAqCN,WAAW;sBADV,MAAM;gBAiBP,MAAM;sBADL,KAAK;gBAKN,KAAK;sBADJ,SAAS;uBAAC,OAAO,EAAE,EAAE,IAAI,EAAE,WAAW,EAAE,MAAM,EAAE,IAAI,EAAE","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: any = undefined;\n\n\n  @ViewChild('inner', { read: TemplateRef, static: true })\n  inner?: TemplateRef<any>;\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    this.createParentComponent();\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;\n    const metadata = reflectComponentType(component) as ComponentMirror<unknown>;\n    const template = this.vcr.createEmbeddedView(this.inner as TemplateRef<any>, 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 any;\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 any;\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"]}