@decaf-ts/for-angular 0.0.5 → 0.0.7
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.
- package/dist/lib/components/component-renderer/component-renderer.component.d.ts +16 -7
- package/dist/lib/components/crud-field/crud-field.component.d.ts +1 -3
- package/dist/lib/components/crud-form/crud-form.component.d.ts +1 -1
- package/dist/lib/components/model-renderer/model-renderer.component.d.ts +5 -6
- package/dist/lib/engine/NgxCrudFormField.d.ts +3 -7
- package/dist/lib/engine/NgxRenderingEngine.d.ts +2 -3
- package/dist/lib/engine/NgxRenderingEngine2.d.ts +5 -6
- package/dist/lib/engine/ValidatorFactory.d.ts +2 -2
- package/dist/lib/engine/types.d.ts +69 -25
- package/dist/lib/esm2022/components/component-renderer/component-renderer.component.mjs +14 -10
- package/dist/lib/esm2022/components/crud-field/crud-field.component.mjs +20 -11
- package/dist/lib/esm2022/components/crud-form/crud-form.component.mjs +3 -4
- package/dist/lib/esm2022/components/model-renderer/model-renderer.component.mjs +10 -9
- package/dist/lib/esm2022/engine/NgxCrudFormField.mjs +21 -15
- package/dist/lib/esm2022/engine/NgxFormService.mjs +1 -1
- package/dist/lib/esm2022/engine/NgxRenderingEngine.mjs +2 -4
- package/dist/lib/esm2022/engine/NgxRenderingEngine2.mjs +5 -6
- package/dist/lib/esm2022/engine/ValidatorFactory.mjs +4 -4
- package/dist/lib/esm2022/engine/types.mjs +1 -1
- package/dist/lib/esm2022/for-angular.module.mjs +1 -1
- package/dist/lib/fesm2022/decaf-ts-for-angular.mjs +71 -56
- package/dist/lib/fesm2022/decaf-ts-for-angular.mjs.map +1 -1
- package/dist/lib/for-angular.module.d.ts +2 -1
- package/package.json +4 -6
|
@@ -1,6 +1,7 @@
|
|
|
1
1
|
import { ComponentRef, EnvironmentInjector, EventEmitter, OnDestroy, OnInit, TemplateRef, ViewContainerRef } from '@angular/core';
|
|
2
|
-
import {
|
|
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~
|
|
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<
|
|
108
|
+
* @type {EventEmitter<RendererCustomEvent>}
|
|
108
109
|
* @memberOf ComponentRendererComponent
|
|
109
110
|
*/
|
|
110
|
-
listenEvent: EventEmitter<
|
|
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,8 +120,16 @@ export declare class ComponentRendererComponent implements OnInit, OnDestroy {
|
|
|
119
120
|
* @memberOf ComponentRendererComponent
|
|
120
121
|
*/
|
|
121
122
|
logger: Logger;
|
|
122
|
-
|
|
123
|
-
|
|
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;
|
|
131
|
+
parent: undefined | KeyValue;
|
|
132
|
+
inner?: TemplateRef<unknown>;
|
|
124
133
|
/**
|
|
125
134
|
* @description Creates an instance of ComponentRendererComponent.
|
|
126
135
|
* @summary Initializes a new ComponentRendererComponent. This component doesn't require
|
|
@@ -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
|
}
|
|
@@ -49,8 +49,6 @@ import * as i0 from "@angular/core";
|
|
|
49
49
|
* @memberOf module:for-angular
|
|
50
50
|
*/
|
|
51
51
|
export declare class CrudFieldComponent extends NgxCrudFormField implements OnInit, OnDestroy, AfterViewInit {
|
|
52
|
-
protected elementRef: ElementRef;
|
|
53
|
-
constructor(elementRef: ElementRef);
|
|
54
52
|
/**
|
|
55
53
|
* @description The CRUD operation being performed.
|
|
56
54
|
* @summary Specifies which CRUD operation (Create, Read, Update, Delete) the field is being used for.
|
|
@@ -324,7 +322,7 @@ export declare class CrudFieldComponent extends NgxCrudFormField implements OnIn
|
|
|
324
322
|
* @type {SelectInterface}
|
|
325
323
|
* @memberOf CrudFieldComponent
|
|
326
324
|
*/
|
|
327
|
-
interface
|
|
325
|
+
interface: SelectInterface;
|
|
328
326
|
/**
|
|
329
327
|
* @description Options for select or radio inputs.
|
|
330
328
|
* @summary Provides the list of options for select or radio inputs. Each option can have a value and a label.
|
|
@@ -42,7 +42,7 @@ export declare class CrudFormComponent implements OnInit, FormElement, OnDestroy
|
|
|
42
42
|
options: CrudFormOptions;
|
|
43
43
|
action?: string;
|
|
44
44
|
operation: CrudOperations;
|
|
45
|
-
handlers: Record<string, (...args:
|
|
45
|
+
handlers: Record<string, (...args: unknown[]) => unknown | Promise<unknown>>;
|
|
46
46
|
formGroup: FormGroup | undefined;
|
|
47
47
|
/**
|
|
48
48
|
* @description Path to the parent FormGroup, if nested.
|
|
@@ -1,7 +1,7 @@
|
|
|
1
|
-
import { EventEmitter,
|
|
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 {
|
|
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
|
|
@@ -32,7 +32,6 @@ import * as i0 from "@angular/core";
|
|
|
32
32
|
* ModelRenderer-->>App: Emit events
|
|
33
33
|
*/
|
|
34
34
|
export declare class ModelRendererComponent<M extends Model> implements OnChanges, OnDestroy, RenderedModel {
|
|
35
|
-
private injector;
|
|
36
35
|
/**
|
|
37
36
|
* @description Input model to be rendered
|
|
38
37
|
* @summary Can be a Model instance or a JSON string representation of a model
|
|
@@ -45,7 +44,7 @@ export declare class ModelRendererComponent<M extends Model> implements OnChange
|
|
|
45
44
|
/**
|
|
46
45
|
* @description Template reference for inner content
|
|
47
46
|
*/
|
|
48
|
-
inner?: TemplateRef<
|
|
47
|
+
inner?: TemplateRef<unknown>;
|
|
49
48
|
/**
|
|
50
49
|
* @description Output of the rendered model
|
|
51
50
|
*/
|
|
@@ -61,7 +60,7 @@ export declare class ModelRendererComponent<M extends Model> implements OnChange
|
|
|
61
60
|
/**
|
|
62
61
|
* @description Event emitter for custom events from the rendered component
|
|
63
62
|
*/
|
|
64
|
-
listenEvent: EventEmitter<
|
|
63
|
+
listenEvent: EventEmitter<RendererCustomEvent>;
|
|
65
64
|
/**
|
|
66
65
|
* @description Instance of the NgxRenderingEngine2
|
|
67
66
|
*/
|
|
@@ -70,7 +69,7 @@ export declare class ModelRendererComponent<M extends Model> implements OnChange
|
|
|
70
69
|
* @description Instance of the rendered component
|
|
71
70
|
*/
|
|
72
71
|
private instance;
|
|
73
|
-
|
|
72
|
+
private injector;
|
|
74
73
|
/**
|
|
75
74
|
* @description Refreshes the rendered model
|
|
76
75
|
* @param {string | M} model - The model to be rendered
|
|
@@ -12,7 +12,6 @@ import { ElementRef } from '@angular/core';
|
|
|
12
12
|
* implementing both CrudFormField and ControlValueAccessor interfaces.
|
|
13
13
|
*/
|
|
14
14
|
export declare abstract class NgxCrudFormField implements ControlValueAccessor, FieldProperties {
|
|
15
|
-
protected elementRef: ElementRef;
|
|
16
15
|
/**
|
|
17
16
|
* @summary Reference to the component's element
|
|
18
17
|
* @description ElementRef representing the component's native element
|
|
@@ -51,7 +50,7 @@ export declare abstract class NgxCrudFormField implements ControlValueAccessor,
|
|
|
51
50
|
greaterThan?: string;
|
|
52
51
|
greaterThanOrEqual?: string;
|
|
53
52
|
value: string | number | Date;
|
|
54
|
-
|
|
53
|
+
private translateService;
|
|
55
54
|
/**
|
|
56
55
|
* @summary Parent HTML element
|
|
57
56
|
* @description Reference to the parent HTML element of the field
|
|
@@ -113,10 +112,7 @@ export declare abstract class NgxCrudFormField implements ControlValueAccessor,
|
|
|
113
112
|
/**
|
|
114
113
|
* @summary Get field errors
|
|
115
114
|
* @description Retrieves all errors associated with the field
|
|
116
|
-
* @returns {
|
|
115
|
+
* @returns {string|void} An array of error objects
|
|
117
116
|
*/
|
|
118
|
-
getErrors(parent: HTMLElement):
|
|
119
|
-
key: string;
|
|
120
|
-
message: string;
|
|
121
|
-
}[];
|
|
117
|
+
getErrors(parent: HTMLElement): string | void;
|
|
122
118
|
}
|
|
@@ -87,15 +87,14 @@ export declare class NgxRenderingEngine extends RenderingEngine<AngularFieldDefi
|
|
|
87
87
|
* FromField-->>Render: AngularDynamicOutput
|
|
88
88
|
* Render-->>Client: return AngularDynamicOutput
|
|
89
89
|
*/
|
|
90
|
-
render<M extends Model>(model: M, globalProps: Record<string, unknown>, vcr: ViewContainerRef, injector: Injector, tpl: TemplateRef<
|
|
90
|
+
render<M extends Model>(model: M, globalProps: Record<string, unknown>, vcr: ViewContainerRef, injector: Injector, tpl: TemplateRef<unknown>): AngularDynamicOutput;
|
|
91
91
|
/**
|
|
92
92
|
* @description Initializes the rendering engine
|
|
93
93
|
* @summary This method initializes the rendering engine. It checks if the engine is already initialized
|
|
94
94
|
* and sets the initialized flag to true. This method is called before the engine is used.
|
|
95
|
-
* @param {...any[]} args - Initialization arguments
|
|
96
95
|
* @return {Promise<void>} A promise that resolves when initialization is complete
|
|
97
96
|
*/
|
|
98
|
-
initialize(
|
|
97
|
+
initialize(): Promise<void>;
|
|
99
98
|
/**
|
|
100
99
|
* @description Registers a component with the rendering engine
|
|
101
100
|
* @summary This static method registers a component constructor with the rendering engine
|
|
@@ -115,10 +115,10 @@ export declare class NgxRenderingEngine2 extends RenderingEngine<AngularFieldDef
|
|
|
115
115
|
* @param {ComponentMirror<unknown>} metadata - The component metadata for input validation
|
|
116
116
|
* @param {ViewContainerRef} vcr - The view container reference for component creation
|
|
117
117
|
* @param {Injector} injector - The Angular injector for dependency injection
|
|
118
|
-
* @param {
|
|
118
|
+
* @param {Node[]} [template=[]] - The template nodes to project into the component
|
|
119
119
|
* @return {ComponentRef<unknown>} The created component reference
|
|
120
120
|
*/
|
|
121
|
-
static createComponent(component: Type<unknown>, inputs: KeyValue | undefined, metadata: ComponentMirror<unknown>, vcr: ViewContainerRef, injector: Injector, template?:
|
|
121
|
+
static createComponent(component: Type<unknown>, inputs: KeyValue | undefined, metadata: ComponentMirror<unknown>, vcr: ViewContainerRef, injector: Injector, template?: Node[]): ComponentRef<unknown>;
|
|
122
122
|
/**
|
|
123
123
|
* @description Extracts decorator metadata from a model
|
|
124
124
|
* @summary This method provides access to the field definition generated from a model's
|
|
@@ -170,17 +170,16 @@ export declare class NgxRenderingEngine2 extends RenderingEngine<AngularFieldDef
|
|
|
170
170
|
* FromField-->>Render: AngularDynamicOutput
|
|
171
171
|
* Render-->>Client: return AngularDynamicOutput
|
|
172
172
|
*/
|
|
173
|
-
render<M extends Model>(model: M, globalProps: Record<string, unknown>, vcr: ViewContainerRef, injector: Injector, tpl: TemplateRef<
|
|
173
|
+
render<M extends Model>(model: M, globalProps: Record<string, unknown>, vcr: ViewContainerRef, injector: Injector, tpl: TemplateRef<unknown>): AngularDynamicOutput;
|
|
174
174
|
/**
|
|
175
175
|
* @description Initializes the rendering engine
|
|
176
176
|
* @summary This method initializes the rendering engine. It checks if the engine is already initialized
|
|
177
177
|
* and sets the initialized flag to true. This method is called before the engine is used
|
|
178
178
|
* to ensure it's properly set up for rendering operations.
|
|
179
179
|
*
|
|
180
|
-
* @param {...any[]} args - Initialization arguments
|
|
181
180
|
* @return {Promise<void>} A promise that resolves when initialization is complete
|
|
182
181
|
*/
|
|
183
|
-
initialize(
|
|
182
|
+
initialize(): Promise<void>;
|
|
184
183
|
/**
|
|
185
184
|
* @description Registers a component with the rendering engine
|
|
186
185
|
* @summary This static method registers a component constructor with the rendering engine
|
|
@@ -203,7 +202,7 @@ export declare class NgxRenderingEngine2 extends RenderingEngine<AngularFieldDef
|
|
|
203
202
|
* @param {string} [selector] - Optional selector to retrieve a specific component
|
|
204
203
|
* @return {Object|Array} Either a specific component or an array of all components
|
|
205
204
|
*/
|
|
206
|
-
static components(selector?: string): object |
|
|
205
|
+
static components(selector?: string): object | string[];
|
|
207
206
|
/**
|
|
208
207
|
* @description Generates a key for reflection metadata
|
|
209
208
|
* @summary This static method generates a key for reflection metadata by prefixing the input key
|
|
@@ -9,7 +9,7 @@ export declare class ValidatorFactory {
|
|
|
9
9
|
* Enables Validators handling method to access parent and child properties using consistent dot-notation in Angular forms.
|
|
10
10
|
*
|
|
11
11
|
* @param {AbstractControl} control - The control to wrap in a proxy.
|
|
12
|
-
* @returns {PathProxy<
|
|
12
|
+
* @returns {PathProxy<unknown>} A proxy object exposing form values and enabling recursive parent access.
|
|
13
13
|
*/
|
|
14
|
-
static createProxy(control: AbstractControl): PathProxy<
|
|
14
|
+
static createProxy(control: AbstractControl): PathProxy<unknown>;
|
|
15
15
|
}
|
|
@@ -1,9 +1,25 @@
|
|
|
1
1
|
import { IonCheckbox, IonInput, IonSelect, IonTextarea } from '@ionic/angular';
|
|
2
2
|
import { TextFieldTypes } from '@ionic/core';
|
|
3
|
-
import { Injector, Type } from '@angular/core';
|
|
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
|
+
*/
|
|
22
|
+
export type FunctionLike = (...args: any[]) => any;
|
|
7
23
|
/**
|
|
8
24
|
* @description Element size options for UI components
|
|
9
25
|
* @summary Defines the possible size values that can be applied to UI elements.
|
|
@@ -203,6 +219,15 @@ export type HTMLFormTarget = '_blank' | '_self' | '_parent' | '_top' | string;
|
|
|
203
219
|
export interface IListComponentRefreshEvent {
|
|
204
220
|
data: KeyValue[];
|
|
205
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
|
+
*/
|
|
206
231
|
export interface FormServiceControl {
|
|
207
232
|
control: FormGroup;
|
|
208
233
|
props: AngularFieldDefinition;
|
|
@@ -217,20 +242,14 @@ export interface FormServiceControl {
|
|
|
217
242
|
*/
|
|
218
243
|
export type FormServiceControls = Record<string, Record<string, FormServiceControl>>;
|
|
219
244
|
/**
|
|
220
|
-
* @description
|
|
221
|
-
* @summary
|
|
222
|
-
*
|
|
223
|
-
*
|
|
224
|
-
* @
|
|
225
|
-
* @property {string} component - The component that triggered the event
|
|
226
|
-
* @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
|
|
227
250
|
* @memberOf module:engine
|
|
228
251
|
*/
|
|
229
|
-
export
|
|
230
|
-
detail: BaseCustomEvent;
|
|
231
|
-
component: string;
|
|
232
|
-
name: string;
|
|
233
|
-
}
|
|
252
|
+
export type RendererCustomEvent = BaseCustomEvent & KeyValue;
|
|
234
253
|
/**
|
|
235
254
|
* @description Interface for list item custom events
|
|
236
255
|
* @summary Defines the structure of custom events triggered by list items.
|
|
@@ -260,34 +279,59 @@ export interface ListItemCustomEvent extends BaseCustomEvent {
|
|
|
260
279
|
* @memberOf module:engine
|
|
261
280
|
*/
|
|
262
281
|
export interface BaseCustomEvent {
|
|
263
|
-
|
|
264
|
-
target?: HTMLElement;
|
|
265
|
-
name?: string;
|
|
282
|
+
name: string;
|
|
266
283
|
component: string;
|
|
284
|
+
data?: unknown;
|
|
285
|
+
target?: HTMLElement;
|
|
267
286
|
}
|
|
268
287
|
/**
|
|
269
|
-
* @description
|
|
270
|
-
* @summary
|
|
271
|
-
*
|
|
272
|
-
* @
|
|
273
|
-
* @property {any}
|
|
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
|
|
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
|
|
277
293
|
* @memberOf module:engine
|
|
278
294
|
*/
|
|
279
295
|
export type CrudFormEvent = BaseCustomEvent & {
|
|
280
296
|
handlers?: Record<string, any>;
|
|
281
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
|
+
*/
|
|
282
308
|
export interface ComponentInput extends FieldProperties {
|
|
283
309
|
updateMode?: FieldUpdateMode;
|
|
284
310
|
formGroup?: FormGroup;
|
|
285
311
|
formControl?: FormControl;
|
|
286
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
|
+
*/
|
|
287
324
|
export interface ComponentConfig {
|
|
288
325
|
component: string;
|
|
289
326
|
inputs: ComponentInput;
|
|
290
|
-
injector:
|
|
327
|
+
injector: EnvironmentInjector | Injector;
|
|
291
328
|
children?: ComponentConfig[];
|
|
292
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
|
+
*/
|
|
293
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~
|
|
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<
|
|
85
|
+
* @type {EventEmitter<RendererCustomEvent>}
|
|
86
86
|
* @memberOf ComponentRendererComponent
|
|
87
87
|
*/
|
|
88
88
|
this.listenEvent = new EventEmitter();
|
|
@@ -113,9 +113,12 @@ export class ComponentRendererComponent {
|
|
|
113
113
|
* @memberOf ComponentRendererComponent
|
|
114
114
|
*/
|
|
115
115
|
ngOnInit() {
|
|
116
|
-
if (!this.parent)
|
|
116
|
+
if (!this.parent) {
|
|
117
117
|
this.createComponent(this.tag, this.globals);
|
|
118
|
-
|
|
118
|
+
}
|
|
119
|
+
else {
|
|
120
|
+
this.createParentComponent();
|
|
121
|
+
}
|
|
119
122
|
}
|
|
120
123
|
/**
|
|
121
124
|
* @description Cleans up resources when the component is destroyed.
|
|
@@ -178,8 +181,9 @@ export class ComponentRendererComponent {
|
|
|
178
181
|
?.constructor;
|
|
179
182
|
const metadata = reflectComponentType(component);
|
|
180
183
|
const componentInputs = metadata.inputs;
|
|
181
|
-
const props = globals?.['item'];
|
|
182
|
-
|
|
184
|
+
const props = globals?.['item'] || globals?.['props'] || {};
|
|
185
|
+
if (props?.['tag'])
|
|
186
|
+
delete props['tag'];
|
|
183
187
|
const inputKeys = Object.keys(props);
|
|
184
188
|
const unmappedKeys = [];
|
|
185
189
|
for (const input of inputKeys) {
|
|
@@ -191,8 +195,6 @@ export class ComponentRendererComponent {
|
|
|
191
195
|
unmappedKeys.push(input);
|
|
192
196
|
}
|
|
193
197
|
}
|
|
194
|
-
if (unmappedKeys.length)
|
|
195
|
-
this.logger.info(`Unmapped input properties for component ${tag}: ${unmappedKeys.join(', ')}`);
|
|
196
198
|
this.vcr.clear();
|
|
197
199
|
this.component = NgxRenderingEngine2.createComponent(component, props, metadata, this.vcr, this.injector, []);
|
|
198
200
|
this.subscribeEvents();
|
|
@@ -285,7 +287,7 @@ export class ComponentRendererComponent {
|
|
|
285
287
|
}
|
|
286
288
|
}
|
|
287
289
|
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"] }] }); }
|
|
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"] }] }); }
|
|
289
291
|
}
|
|
290
292
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: ComponentRendererComponent, decorators: [{
|
|
291
293
|
type: Component,
|
|
@@ -300,10 +302,12 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
|
|
|
300
302
|
type: Input
|
|
301
303
|
}], listenEvent: [{
|
|
302
304
|
type: Output
|
|
305
|
+
}], model: [{
|
|
306
|
+
type: Input
|
|
303
307
|
}], parent: [{
|
|
304
308
|
type: Input
|
|
305
309
|
}], inner: [{
|
|
306
310
|
type: ViewChild,
|
|
307
311
|
args: ['inner', { read: TemplateRef, static: true }]
|
|
308
312
|
}] } });
|
|
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":"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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: 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"]}
|
|
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"]}
|