@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.
- package/LICENSE.md +659 -21
- package/README.md +37 -242
- package/dist/lib/README.md +92 -0
- package/dist/lib/assets/i18n/en.json +131 -0
- package/dist/lib/assets/images/angular-logo.svg +45 -0
- package/dist/lib/assets/images/decaf-logo-black.svg +22 -0
- package/dist/lib/assets/images/decaf-logo-lw.svg +50 -0
- package/dist/lib/assets/images/decaf-logo-white.svg +22 -0
- package/dist/lib/assets/images/decaf-logo.svg +54 -0
- package/dist/lib/components/component-renderer/component-renderer.component.d.ts +267 -0
- package/dist/lib/components/crud-field/crud-field.component.d.ts +447 -0
- package/dist/lib/components/crud-form/crud-form.component.d.ts +102 -0
- package/dist/lib/components/model-renderer/model-renderer.component.d.ts +97 -0
- package/dist/lib/engine/DynamicModule.d.ts +17 -0
- package/dist/{for-angular → lib}/engine/NgxCrudFormField.d.ts +37 -30
- package/dist/lib/engine/NgxFormService.d.ts +167 -0
- package/dist/lib/engine/NgxRenderingEngine.d.ts +128 -0
- package/dist/lib/engine/NgxRenderingEngine2.d.ts +251 -0
- package/dist/lib/engine/ValidatorFactory.d.ts +15 -0
- package/dist/lib/engine/constants.d.ts +151 -0
- package/dist/lib/engine/decorators.d.ts +25 -0
- package/dist/lib/engine/index.d.ts +15 -0
- package/dist/lib/engine/types.d.ts +293 -0
- package/dist/lib/esm2022/components/component-renderer/component-renderer.component.mjs +309 -0
- package/dist/lib/esm2022/components/crud-field/crud-field.component.mjs +288 -0
- package/dist/lib/esm2022/components/crud-form/constants.mjs +14 -0
- package/dist/lib/esm2022/components/crud-form/crud-form.component.mjs +140 -0
- package/dist/lib/esm2022/components/crud-form/types.mjs +2 -0
- package/dist/lib/esm2022/components/model-renderer/model-renderer.component.mjs +137 -0
- package/dist/lib/esm2022/engine/DynamicModule.mjs +18 -0
- package/dist/lib/esm2022/engine/NgxCrudFormField.mjs +117 -0
- package/dist/lib/esm2022/engine/NgxFormService.mjs +315 -0
- package/dist/lib/esm2022/engine/NgxRenderingEngine.mjs +194 -0
- package/dist/lib/esm2022/engine/NgxRenderingEngine2.mjs +333 -0
- package/dist/lib/esm2022/engine/ValidatorFactory.mjs +102 -0
- package/dist/lib/esm2022/engine/constants.mjs +160 -0
- package/dist/lib/esm2022/engine/decorators.mjs +38 -0
- package/dist/lib/esm2022/engine/index.mjs +16 -0
- package/dist/lib/esm2022/engine/types.mjs +2 -0
- package/dist/lib/esm2022/for-angular.module.mjs +118 -0
- package/dist/lib/esm2022/interfaces.mjs +2 -0
- package/dist/lib/esm2022/public-apis.mjs +13 -0
- package/dist/lib/fesm2022/decaf-ts-for-angular.mjs +2138 -0
- package/dist/lib/fesm2022/decaf-ts-for-angular.mjs.map +1 -0
- package/dist/lib/for-angular.module.d.ts +44 -0
- package/dist/lib/interfaces.d.ts +28 -0
- package/dist/lib/public-apis.d.ts +12 -0
- package/package.json +74 -25
- package/dist/for-angular/README.md +0 -297
- package/dist/for-angular/assets/i18n/en.json +0 -21
- package/dist/for-angular/components/decaf-crud-field/decaf-crud-field.component.d.ts +0 -22
- package/dist/for-angular/components/decaf-crud-form/decaf-crud-form.component.d.ts +0 -28
- package/dist/for-angular/components/decaf-model-renderer/decaf-model-renderer.component.d.ts +0 -20
- package/dist/for-angular/directives/decaf-field.directive.d.ts +0 -8
- package/dist/for-angular/engine/DynamicModule.d.ts +0 -2
- package/dist/for-angular/engine/NgxFormService.d.ts +0 -119
- package/dist/for-angular/engine/NgxRenderingEngine.d.ts +0 -17
- package/dist/for-angular/engine/ValidatorFactory.d.ts +0 -4
- package/dist/for-angular/engine/constants.d.ts +0 -10
- package/dist/for-angular/engine/decorators.d.ts +0 -1
- package/dist/for-angular/engine/index.d.ts +0 -5
- package/dist/for-angular/engine/types.d.ts +0 -32
- package/dist/for-angular/esm2022/components/decaf-crud-field/decaf-crud-field.component.mjs +0 -66
- package/dist/for-angular/esm2022/components/decaf-crud-form/constants.mjs +0 -14
- package/dist/for-angular/esm2022/components/decaf-crud-form/decaf-crud-form.component.mjs +0 -84
- package/dist/for-angular/esm2022/components/decaf-crud-form/types.mjs +0 -2
- package/dist/for-angular/esm2022/components/decaf-model-renderer/decaf-model-renderer.component.mjs +0 -46
- package/dist/for-angular/esm2022/directives/decaf-field.directive.mjs +0 -23
- package/dist/for-angular/esm2022/engine/DynamicModule.mjs +0 -3
- package/dist/for-angular/esm2022/engine/NgxCrudFormField.mjs +0 -118
- package/dist/for-angular/esm2022/engine/NgxFormService.mjs +0 -232
- package/dist/for-angular/esm2022/engine/NgxRenderingEngine.mjs +0 -35
- package/dist/for-angular/esm2022/engine/ValidatorFactory.mjs +0 -48
- package/dist/for-angular/esm2022/engine/constants.mjs +0 -12
- package/dist/for-angular/esm2022/engine/decorators.mjs +0 -17
- package/dist/for-angular/esm2022/engine/index.mjs +0 -6
- package/dist/for-angular/esm2022/engine/types.mjs +0 -2
- package/dist/for-angular/esm2022/interfaces.mjs +0 -2
- package/dist/for-angular/esm2022/public-apis.mjs +0 -5
- package/dist/for-angular/fesm2022/decaf-ts-for-angular.mjs +0 -675
- package/dist/for-angular/fesm2022/decaf-ts-for-angular.mjs.map +0 -1
- package/dist/for-angular/interfaces.d.ts +0 -8
- package/dist/for-angular/public-apis.d.ts +0 -4
- /package/dist/{for-angular/components/decaf-crud-form → lib/components/crud-form}/constants.d.ts +0 -0
- /package/dist/{for-angular/components/decaf-crud-form → lib/components/crud-form}/types.d.ts +0 -0
- /package/dist/{for-angular → lib}/esm2022/decaf-ts-for-angular.mjs +0 -0
- /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":"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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"]}
|