@decaf-ts/for-angular 0.0.16 → 0.0.18

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (65) hide show
  1. package/assets/i18n/en.json +9 -69
  2. package/assets/i18n/pt.json +80 -0
  3. package/assets/icons/icon-128.webp +0 -0
  4. package/assets/icons/icon-192.webp +0 -0
  5. package/assets/icons/icon-256.webp +0 -0
  6. package/assets/icons/icon-48.webp +0 -0
  7. package/assets/icons/icon-512.webp +0 -0
  8. package/assets/icons/icon-72.webp +0 -0
  9. package/assets/icons/icon-96.webp +0 -0
  10. package/assets/images/apple-touch-icon.png +0 -0
  11. package/assets/images/favicon.png +0 -0
  12. package/assets/images/favicon.svg +29 -0
  13. package/components/component-renderer/component-renderer.component.d.ts +5 -4
  14. package/components/crud-field/crud-field.component.d.ts +186 -22
  15. package/components/crud-form/crud-form.component.d.ts +194 -8
  16. package/components/empty-state/empty-state.component.d.ts +9 -10
  17. package/components/fieldset/fieldset.component.d.ts +383 -36
  18. package/components/filter/filter.component.d.ts +11 -2
  19. package/components/list/list.component.d.ts +1 -1
  20. package/components/list-item/list-item.component.d.ts +2 -2
  21. package/components/model-renderer/model-renderer.component.d.ts +1 -5
  22. package/directives/collapsable.directive.d.ts +1 -0
  23. package/engine/NgxBaseComponent.d.ts +43 -43
  24. package/engine/NgxCrudFormField.d.ts +7 -3
  25. package/engine/NgxFormService.d.ts +113 -12
  26. package/engine/NgxRenderingEngine.d.ts +178 -25
  27. package/engine/constants.d.ts +11 -6
  28. package/engine/decorators.d.ts +2 -2
  29. package/engine/index.d.ts +4 -2
  30. package/engine/interfaces.d.ts +271 -0
  31. package/engine/types.d.ts +11 -206
  32. package/esm2022/components/component-renderer/component-renderer.component.mjs +13 -11
  33. package/esm2022/components/crud-field/crud-field.component.mjs +213 -8
  34. package/esm2022/components/crud-form/crud-form.component.mjs +133 -13
  35. package/esm2022/components/empty-state/empty-state.component.mjs +13 -12
  36. package/esm2022/components/fieldset/fieldset.component.mjs +485 -43
  37. package/esm2022/components/filter/filter.component.mjs +16 -6
  38. package/esm2022/components/layout/layout.component.mjs +3 -3
  39. package/esm2022/components/list/list.component.mjs +4 -5
  40. package/esm2022/components/list-item/list-item.component.mjs +10 -10
  41. package/esm2022/components/model-renderer/model-renderer.component.mjs +9 -8
  42. package/esm2022/components/pagination/pagination.component.mjs +7 -7
  43. package/esm2022/components/searchbar/searchbar.component.mjs +3 -3
  44. package/esm2022/directives/collapsable.directive.mjs +3 -2
  45. package/esm2022/engine/NgxBaseComponent.mjs +64 -63
  46. package/esm2022/engine/NgxCrudFormField.mjs +14 -4
  47. package/esm2022/engine/NgxFormService.mjs +239 -27
  48. package/esm2022/engine/NgxRenderingEngine.mjs +218 -46
  49. package/esm2022/engine/ValidatorFactory.mjs +6 -4
  50. package/esm2022/engine/constants.mjs +14 -9
  51. package/esm2022/engine/decorators.mjs +6 -6
  52. package/esm2022/engine/index.mjs +5 -3
  53. package/esm2022/engine/interfaces.mjs +4 -0
  54. package/esm2022/engine/types.mjs +1 -3
  55. package/esm2022/helpers/utils.mjs +53 -32
  56. package/esm2022/i18n/Loader.mjs +82 -0
  57. package/fesm2022/decaf-ts-for-angular.mjs +3030 -2097
  58. package/fesm2022/decaf-ts-for-angular.mjs.map +1 -1
  59. package/helpers/utils.d.ts +42 -16
  60. package/i18n/Loader.d.ts +48 -0
  61. package/package.json +11 -1
  62. package/engine/NgxRenderingEngine2.d.ts +0 -250
  63. package/esm2022/engine/NgxRenderingEngine2.mjs +0 -332
  64. package/esm2022/interfaces.mjs +0 -2
  65. package/interfaces.d.ts +0 -28
@@ -2,38 +2,64 @@ import { RenderingEngine } from '@decaf-ts/ui-decorators';
2
2
  import { AngularEngineKeys } from './constants';
3
3
  import { InternalError } from '@decaf-ts/db-decorators';
4
4
  import { reflectComponentType, } from '@angular/core';
5
+ import { NgxFormService } from './NgxFormService';
5
6
  /**
6
- * @description Angular implementation of the RenderingEngine
7
- * @summary This class extends the base RenderingEngine to provide Angular-specific rendering capabilities.
8
- * It handles the conversion of field definitions to Angular components and manages component registration.
7
+ * @description Angular implementation of the RenderingEngine with enhanced features
8
+ * @summary This class extends the base RenderingEngine to provide Angular-specific rendering capabilities
9
+ * with additional features compared to NgxRenderingEngine. It handles the conversion of field definitions
10
+ * to Angular components, manages component registration, and provides utilities for component creation
11
+ * and input handling. This implementation uses Angular's newer component APIs.
12
+ *
9
13
  * @template AngularFieldDefinition - Type for Angular-specific field definitions
10
14
  * @template AngularDynamicOutput - Type for Angular-specific component output
11
- * @param {Injector} injector - Angular injector for dependency injection
12
- * @param {ViewContainerRef} vcr - View container reference for component creation
13
- * @param {TemplateRef<any>} tpl - Template reference for content projection
15
+ *
14
16
  * @class NgxRenderingEngine
15
17
  * @example
16
18
  * ```typescript
17
- * const engine = new NgxRenderingEngine();
19
+ * const engine = NgxRenderingEngine.get();
18
20
  * engine.initialize();
19
21
  * const output = engine.render(myModel, {}, viewContainerRef, injector, templateRef);
20
22
  * ```
23
+ *
21
24
  * @mermaid
22
25
  * sequenceDiagram
23
26
  * participant Client
24
27
  * participant Engine as NgxRenderingEngine
25
28
  * participant Components as RegisteredComponents
26
29
  *
27
- * Client->>Engine: new NgxRenderingEngine()
30
+ * Client->>Engine: get()
28
31
  * Client->>Engine: initialize()
29
32
  * Client->>Engine: render(model, props, vcr, injector, tpl)
30
33
  * Engine->>Engine: toFieldDefinition(model, props)
31
34
  * Engine->>Engine: fromFieldDefinition(fieldDef, vcr, injector, tpl)
32
35
  * Engine->>Components: components(fieldDef.tag)
33
36
  * Components-->>Engine: component constructor
34
- * Engine->>Client: return AngularDynamicOutput
37
+ * Engine->>Engine: createComponent(component, inputs, metadata, vcr, injector, template)
38
+ * Engine-->>Client: return AngularDynamicOutput
35
39
  */
36
40
  export class NgxRenderingEngine extends RenderingEngine {
41
+ /**
42
+ * @description Current operation context for component visibility control
43
+ * @summary Static property that stores the current operation being performed,
44
+ * which is used to determine component visibility through the 'hidden' property.
45
+ * Components can specify operations where they should be hidden, and this property
46
+ * provides the context for those visibility checks. The value is typically extracted
47
+ * from the global properties during the rendering process.
48
+ *
49
+ * @private
50
+ * @static
51
+ * @type {string | undefined}
52
+ */
53
+ static { this._operation = undefined; }
54
+ /**
55
+ * @description Constructs a new NgxRenderingEngine instance
56
+ * @summary Initializes a new instance of the Angular rendering engine by calling the parent
57
+ * constructor with the 'angular' engine type identifier. This constructor sets up the base
58
+ * rendering engine functionality with Angular-specific configurations and prepares the
59
+ * instance for component registration and rendering operations.
60
+ *
61
+ * @constructor
62
+ */
37
63
  constructor() {
38
64
  super('angular');
39
65
  }
@@ -41,69 +67,142 @@ export class NgxRenderingEngine extends RenderingEngine {
41
67
  * @description Converts a field definition to an Angular component output
42
68
  * @summary This private method takes a field definition and creates the corresponding Angular component.
43
69
  * It handles component instantiation, input property mapping, and child component rendering.
70
+ * The method validates input properties against the component's metadata and processes
71
+ * child components recursively.
72
+ *
44
73
  * @param {FieldDefinition<AngularFieldDefinition>} fieldDef - The field definition to convert
45
74
  * @param {ViewContainerRef} vcr - The view container reference for component creation
46
75
  * @param {Injector} injector - The Angular injector for dependency injection
47
76
  * @param {TemplateRef<any>} tpl - The template reference for content projection
77
+ * @param {string} registryFormId - Form identifier for the component renderer
48
78
  * @return {AngularDynamicOutput} The Angular component output with component reference and inputs
79
+ *
49
80
  * @mermaid
50
81
  * sequenceDiagram
51
82
  * participant Method as fromFieldDefinition
52
83
  * participant Components as NgxRenderingEngine.components
53
84
  * participant Angular as Angular Core
85
+ * participant Process as processChild
54
86
  *
55
87
  * Method->>Components: components(fieldDef.tag)
56
88
  * Components-->>Method: component constructor
57
89
  * Method->>Angular: reflectComponentType(component)
58
90
  * Angular-->>Method: componentMetadata
59
- * Method->>Method: Check input properties
91
+ * Method->>Method: Validate input properties
60
92
  * Method->>Method: Create result object
61
- * Method->>Method: Process children if any
93
+ * alt Has children
94
+ * Method->>Process: Process children recursively
95
+ * Process->>Method: Return processed children
96
+ * Method->>Angular: Create embedded view
97
+ * Method->>Method: Create component instance
98
+ * end
62
99
  * Method-->>Caller: return AngularDynamicOutput
63
100
  */
64
- fromFieldDefinition(fieldDef, vcr, injector, tpl) {
65
- const component = NgxRenderingEngine.components(fieldDef.tag)
66
- .constructor;
101
+ fromFieldDefinition(fieldDef, vcr, injector, tpl, registryFormId = Date.now().toString(36).toUpperCase()) {
102
+ const cmp = fieldDef?.['component'] || NgxRenderingEngine.components(fieldDef.tag);
103
+ const component = cmp.constructor;
67
104
  const componentMetadata = reflectComponentType(component);
68
105
  if (!componentMetadata) {
69
106
  throw new InternalError(`Metadata for component ${fieldDef.tag} not found.`);
70
107
  }
71
- const inputs = fieldDef.props;
72
- const possibleInputs = componentMetadata.inputs;
73
- const inputKeys = Object.keys(inputs);
74
- for (const input of possibleInputs) {
75
- const index = inputKeys.indexOf(input.propName);
76
- if (index !== -1) {
77
- inputKeys.splice(index, 1);
78
- }
79
- if (!inputKeys.length)
80
- break;
81
- }
82
- if (inputKeys.length)
83
- console.warn(`Unmapped input properties for component ${fieldDef.tag}: ${inputKeys.join(', ')}`);
108
+ const { inputs: possibleInputs } = componentMetadata;
109
+ const inputs = { ...fieldDef.props };
110
+ const unmappedKeys = Object.keys(inputs).filter(input => {
111
+ const isMapped = possibleInputs.find(({ propName }) => propName === input);
112
+ if (!isMapped)
113
+ delete inputs[input];
114
+ return !isMapped;
115
+ });
116
+ if (unmappedKeys.length > 0)
117
+ console.warn(`Unmapped input properties for component ${fieldDef.tag}: ${unmappedKeys.join(', ')}`);
118
+ const operation = NgxRenderingEngine._operation;
119
+ const hiddenOn = inputs?.hidden || [];
120
+ if (hiddenOn.includes(operation))
121
+ return { inputs, injector };
122
+ // const hiddenOn = inputs?.hidden || [];
84
123
  const result = {
85
- component: component,
86
- inputs: inputs || {},
87
- injector: injector,
124
+ component,
125
+ inputs,
126
+ injector,
88
127
  };
89
- if (fieldDef.rendererId) {
90
- result.inputs['rendererId'] =
91
- fieldDef.rendererId;
92
- }
93
- if (fieldDef.children && fieldDef.children.length) {
128
+ if (fieldDef.rendererId)
129
+ result.inputs['rendererId'] = fieldDef.rendererId;
130
+ // process children
131
+ if (fieldDef.children?.length) {
94
132
  result.children = fieldDef.children.map((child) => {
95
- return this.fromFieldDefinition(child, vcr, injector, tpl);
133
+ if (child?.children?.length) {
134
+ child.children = child.children.filter(c => {
135
+ const hiddenOn = c?.props?.hidden || [];
136
+ if (!hiddenOn.includes(operation))
137
+ return c;
138
+ });
139
+ }
140
+ // create a child form and add its controls as properties of child.props
141
+ NgxFormService.addControlFromProps(registryFormId, child.props, inputs);
142
+ return this.fromFieldDefinition(child, vcr, injector, tpl, registryFormId);
96
143
  });
97
- const template = vcr.createEmbeddedView(tpl, injector).rootNodes;
98
- result.content = [template];
99
144
  }
145
+ // generating DOM
146
+ vcr.clear();
147
+ const template = vcr.createEmbeddedView(tpl, injector).rootNodes;
148
+ const componentInstance = NgxRenderingEngine.createComponent(component, { ...inputs, model: this._model }, componentMetadata, vcr, injector, template);
149
+ result.instance = NgxRenderingEngine._instance = componentInstance.instance;
100
150
  return result;
101
151
  }
152
+ /**
153
+ * @description Creates an Angular component instance
154
+ * @summary This static utility method creates an Angular component instance with the specified
155
+ * inputs and template. It uses Angular's component creation API to instantiate the component
156
+ * and then sets the input properties using the provided metadata.
157
+ *
158
+ * @param {Type<unknown>} component - The component type to create
159
+ * @param {KeyValue} [inputs={}] - The input properties to set on the component
160
+ * @param {ComponentMirror<unknown>} metadata - The component metadata for input validation
161
+ * @param {ViewContainerRef} vcr - The view container reference for component creation
162
+ * @param {Injector} injector - The Angular injector for dependency injection
163
+ * @param {Node[]} [template=[]] - The template nodes to project into the component
164
+ * @return {ComponentRef<unknown>} The created component reference
165
+ */
166
+ static createComponent(component, inputs = {}, metadata, vcr, injector, template = []) {
167
+ const componentInstance = vcr.createComponent(component, {
168
+ environmentInjector: injector,
169
+ projectableNodes: [template],
170
+ });
171
+ this.setInputs(componentInstance, inputs, metadata);
172
+ return componentInstance;
173
+ }
174
+ /**
175
+ * @description Extracts decorator metadata from a model
176
+ * @summary This method provides access to the field definition generated from a model's
177
+ * decorators. It's a convenience wrapper around the toFieldDefinition method that
178
+ * converts a model to a field definition based on its decorators and the provided
179
+ * global properties.
180
+ *
181
+ * @param {Model} model - The model to extract decorators from
182
+ * @param {Record<string, unknown>} globalProps - Global properties to include in the field definition
183
+ * @return {FieldDefinition<AngularFieldDefinition>} The field definition generated from the model
184
+ */
185
+ getDecorators(model, globalProps) {
186
+ return this.toFieldDefinition(model, globalProps);
187
+ }
188
+ /**
189
+ * @description Destroys the current engine instance
190
+ * @summary This static method clears the current instance reference, effectively
191
+ * destroying the singleton instance of the rendering engine. This can be used
192
+ * to reset the engine state or to prepare for a new instance creation.
193
+ *
194
+ * @return {Promise<void>} A promise that resolves when the instance is destroyed
195
+ */
196
+ static async destroy() {
197
+ NgxRenderingEngine._instance = undefined;
198
+ }
102
199
  /**
103
200
  * @description Renders a model into an Angular component output
104
201
  * @summary This method takes a model and converts it to an Angular component output.
105
- * It first converts the model to a field definition using the base RenderingEngine's
106
- * toFieldDefinition method, then converts that field definition to an Angular component output.
202
+ * It first stores a reference to the model, then converts it to a field definition
203
+ * using the base RenderingEngine's toFieldDefinition method, and finally converts
204
+ * that field definition to an Angular component output using fromFieldDefinition.
205
+ *
107
206
  * @template M - Type extending Model
108
207
  * @param {M} model - The model to render
109
208
  * @param {Record<string, unknown>} globalProps - Global properties to pass to the component
@@ -111,6 +210,7 @@ export class NgxRenderingEngine extends RenderingEngine {
111
210
  * @param {Injector} injector - The Angular injector for dependency injection
112
211
  * @param {TemplateRef<any>} tpl - The template reference for content projection
113
212
  * @return {AngularDynamicOutput} The Angular component output with component reference and inputs
213
+ *
114
214
  * @mermaid
115
215
  * sequenceDiagram
116
216
  * participant Client as Client Code
@@ -119,6 +219,7 @@ export class NgxRenderingEngine extends RenderingEngine {
119
219
  * participant FromField as fromFieldDefinition
120
220
  *
121
221
  * Client->>Render: render(model, globalProps, vcr, injector, tpl)
222
+ * Render->>Render: Store model reference
122
223
  * Render->>ToField: toFieldDefinition(model, globalProps)
123
224
  * ToField-->>Render: fieldDef
124
225
  * Render->>FromField: fromFieldDefinition(fieldDef, vcr, injector, tpl)
@@ -128,8 +229,15 @@ export class NgxRenderingEngine extends RenderingEngine {
128
229
  render(model, globalProps, vcr, injector, tpl) {
129
230
  let result;
130
231
  try {
232
+ this._model = model;
233
+ const formId = Date.now().toString(36).toUpperCase();
131
234
  const fieldDef = this.toFieldDefinition(model, globalProps);
132
- result = this.fromFieldDefinition(fieldDef, vcr, injector, tpl);
235
+ const props = fieldDef.props;
236
+ if (!NgxRenderingEngine._operation)
237
+ NgxRenderingEngine._operation = props?.['operation'] || undefined;
238
+ result = this.fromFieldDefinition(fieldDef, vcr, injector, tpl, formId);
239
+ result.instance['formGroup'] = NgxFormService.getControlFromForm(formId);
240
+ NgxFormService.removeRegistry(formId);
133
241
  }
134
242
  catch (e) {
135
243
  throw new InternalError(`Failed to render Model ${model.constructor.name}: ${e}`);
@@ -139,18 +247,24 @@ export class NgxRenderingEngine extends RenderingEngine {
139
247
  /**
140
248
  * @description Initializes the rendering engine
141
249
  * @summary This method initializes the rendering engine. It checks if the engine is already initialized
142
- * and sets the initialized flag to true. This method is called before the engine is used.
250
+ * and sets the initialized flag to true. This method is called before the engine is used
251
+ * to ensure it's properly set up for rendering operations.
252
+ *
143
253
  * @return {Promise<void>} A promise that resolves when initialization is complete
144
254
  */
145
255
  async initialize() {
146
256
  if (this.initialized)
147
257
  return;
258
+ // ValidatableByType[]
148
259
  this.initialized = true;
149
260
  }
150
261
  /**
151
262
  * @description Registers a component with the rendering engine
152
263
  * @summary This static method registers a component constructor with the rendering engine
153
- * under a specific name. It throws an error if a component is already registered under the same name.
264
+ * under a specific name. It initializes the components registry if needed and throws
265
+ * an error if a component is already registered under the same name to prevent
266
+ * accidental overrides.
267
+ *
154
268
  * @param {string} name - The name to register the component under
155
269
  * @param {Constructor<unknown>} constructor - The component constructor
156
270
  * @return {void}
@@ -167,7 +281,10 @@ export class NgxRenderingEngine extends RenderingEngine {
167
281
  /**
168
282
  * @description Retrieves registered components from the rendering engine
169
283
  * @summary This static method retrieves either all registered components or a specific component
170
- * by its selector. It throws an error if the requested component is not registered.
284
+ * by its selector. When called without a selector, it returns an array of all registered
285
+ * components. When called with a selector, it returns the specific component if found,
286
+ * or throws an error if the component is not registered.
287
+ *
171
288
  * @param {string} [selector] - Optional selector to retrieve a specific component
172
289
  * @return {Object|Array} Either a specific component or an array of all components
173
290
  */
@@ -181,12 +298,67 @@ export class NgxRenderingEngine extends RenderingEngine {
181
298
  /**
182
299
  * @description Generates a key for reflection metadata
183
300
  * @summary This static method generates a key for reflection metadata by prefixing the input key
184
- * with the Angular engine's reflection prefix. This is used for storing and retrieving metadata.
301
+ * with the Angular engine's reflection prefix. This is used for storing and retrieving
302
+ * metadata in a namespaced way to avoid conflicts with other metadata.
303
+ *
185
304
  * @param {string} key - The base key to prefix
186
305
  * @return {string} The prefixed key for reflection metadata
187
306
  */
188
307
  static key(key) {
189
308
  return `${AngularEngineKeys.REFLECT}${key}`;
190
309
  }
310
+ /**
311
+ * @description Sets input properties on a component instance
312
+ * @summary This static utility method sets input properties on a component instance
313
+ * based on the provided inputs object and component metadata. It handles both simple
314
+ * values and nested objects, recursively processing object properties. The method
315
+ * validates each input against the component's metadata to ensure only valid inputs
316
+ * are set.
317
+ *
318
+ * @param {ComponentRef<unknown>} component - The component reference to set inputs on
319
+ * @param {KeyValue} inputs - The input properties to set
320
+ * @param {ComponentMirror<unknown>} metadata - The component metadata for input validation
321
+ * @return {void}
322
+ *
323
+ * @mermaid
324
+ * sequenceDiagram
325
+ * participant Caller
326
+ * participant SetInputs as setInputs
327
+ * participant Parse as parseInputValue
328
+ * participant Component as ComponentRef
329
+ *
330
+ * Caller->>SetInputs: setInputs(component, inputs, metadata)
331
+ * SetInputs->>SetInputs: Iterate through inputs
332
+ * loop For each input
333
+ * SetInputs->>SetInputs: Check if input exists in metadata
334
+ * alt Input is 'props'
335
+ * SetInputs->>Parse: parseInputValue(component, value)
336
+ * Parse->>Parse: Recursively process nested objects
337
+ * Parse->>Component: setInput(key, value)
338
+ * else Input is valid
339
+ * SetInputs->>Component: setInput(key, value)
340
+ * end
341
+ * end
342
+ */
343
+ static setInputs(component, inputs, metadata) {
344
+ function parseInputValue(component, input) {
345
+ Object.keys(input).forEach(key => {
346
+ const value = input[key];
347
+ if (typeof value === 'object' && !!value)
348
+ return parseInputValue(component, value);
349
+ component.setInput(key, value);
350
+ });
351
+ }
352
+ Object.entries(inputs).forEach(([key, value]) => {
353
+ const prop = metadata.inputs.find((item) => item.propName === key);
354
+ if (prop) {
355
+ if (key === 'props')
356
+ parseInputValue(component, value);
357
+ // if(key === 'locale' && !value)
358
+ // value = getLocaleFromClassName(this._componentName);
359
+ component.setInput(key, value);
360
+ }
361
+ });
362
+ }
191
363
  }
192
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"NgxRenderingEngine.js","sourceRoot":"","sources":["../../../../src/lib/engine/NgxRenderingEngine.ts"],"names":[],"mappings":"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","sourcesContent":["import { FieldDefinition, RenderingEngine } from '@decaf-ts/ui-decorators';\nimport { AngularDynamicOutput, AngularFieldDefinition } from './types';\nimport { AngularEngineKeys } from './constants';\nimport { Constructor, Model } from '@decaf-ts/decorator-validation';\nimport { InternalError } from '@decaf-ts/db-decorators';\nimport {\n  Injector,\n  reflectComponentType,\n  TemplateRef,\n  Type,\n  ViewContainerRef,\n} from '@angular/core';\n\n/**\n * @description Angular implementation of the RenderingEngine\n * @summary This class extends the base RenderingEngine to provide Angular-specific rendering capabilities.\n * It handles the conversion of field definitions to Angular components and manages component registration.\n * @template AngularFieldDefinition - Type for Angular-specific field definitions\n * @template AngularDynamicOutput - Type for Angular-specific component output\n * @param {Injector} injector - Angular injector for dependency injection\n * @param {ViewContainerRef} vcr - View container reference for component creation\n * @param {TemplateRef<any>} tpl - Template reference for content projection\n * @class NgxRenderingEngine\n * @example\n * ```typescript\n * const engine = new NgxRenderingEngine();\n * engine.initialize();\n * const output = engine.render(myModel, {}, viewContainerRef, injector, templateRef);\n * ```\n * @mermaid\n * sequenceDiagram\n *   participant Client\n *   participant Engine as NgxRenderingEngine\n *   participant Components as RegisteredComponents\n *\n *   Client->>Engine: new NgxRenderingEngine()\n *   Client->>Engine: initialize()\n *   Client->>Engine: render(model, props, vcr, injector, tpl)\n *   Engine->>Engine: toFieldDefinition(model, props)\n *   Engine->>Engine: fromFieldDefinition(fieldDef, vcr, injector, tpl)\n *   Engine->>Components: components(fieldDef.tag)\n *   Components-->>Engine: component constructor\n *   Engine->>Client: return AngularDynamicOutput\n */\nexport class NgxRenderingEngine extends RenderingEngine<\n  AngularFieldDefinition,\n  AngularDynamicOutput\n> {\n  private static _components: Record<\n    string,\n    { constructor: Constructor<unknown> }\n  >;\n\n  constructor() {\n    super('angular');\n  }\n\n  /**\n   * @description Converts a field definition to an Angular component output\n   * @summary This private method takes a field definition and creates the corresponding Angular component.\n   * It handles component instantiation, input property mapping, and child component rendering.\n   * @param {FieldDefinition<AngularFieldDefinition>} fieldDef - The field definition to convert\n   * @param {ViewContainerRef} vcr - The view container reference for component creation\n   * @param {Injector} injector - The Angular injector for dependency injection\n   * @param {TemplateRef<any>} tpl - The template reference for content projection\n   * @return {AngularDynamicOutput} The Angular component output with component reference and inputs\n   * @mermaid\n   * sequenceDiagram\n   *   participant Method as fromFieldDefinition\n   *   participant Components as NgxRenderingEngine.components\n   *   participant Angular as Angular Core\n   *\n   *   Method->>Components: components(fieldDef.tag)\n   *   Components-->>Method: component constructor\n   *   Method->>Angular: reflectComponentType(component)\n   *   Angular-->>Method: componentMetadata\n   *   Method->>Method: Check input properties\n   *   Method->>Method: Create result object\n   *   Method->>Method: Process children if any\n   *   Method-->>Caller: return AngularDynamicOutput\n   */\n  private fromFieldDefinition(\n    fieldDef: FieldDefinition<AngularFieldDefinition>,\n    vcr: ViewContainerRef,\n    injector: Injector,\n    tpl: TemplateRef<unknown>,\n  ): AngularDynamicOutput {\n    const component = NgxRenderingEngine.components(fieldDef.tag)\n      .constructor as unknown as Type<unknown>;\n\n    const componentMetadata = reflectComponentType(component);\n    if (!componentMetadata) {\n      throw new InternalError(\n        `Metadata for component ${fieldDef.tag} not found.`,\n      );\n    }\n    const inputs = fieldDef.props;\n\n    const possibleInputs = componentMetadata.inputs;\n    const inputKeys = Object.keys(inputs);\n    for (const input of possibleInputs) {\n      const index = inputKeys.indexOf(input.propName);\n      if (index !== -1) {\n        inputKeys.splice(index, 1);\n      }\n      if (!inputKeys.length) break;\n    }\n\n    if (inputKeys.length)\n      console.warn(\n        `Unmapped input properties for component ${fieldDef.tag}: ${inputKeys.join(', ')}`,\n      );\n\n    const result: AngularDynamicOutput = {\n      component: component,\n      inputs: inputs || {},\n      injector: injector,\n    };\n\n    if (fieldDef.rendererId) {\n      (result.inputs as Record<string, unknown>)['rendererId'] =\n        fieldDef.rendererId;\n    }\n\n    if (fieldDef.children && fieldDef.children.length) {\n      result.children = fieldDef.children.map((child) => {\n        return this.fromFieldDefinition(child, vcr, injector, tpl);\n      });\n\n      const template = vcr.createEmbeddedView(tpl, injector).rootNodes;\n      result.content = [template];\n    }\n\n    return result;\n  }\n\n  /**\n   * @description Renders a model into an Angular component output\n   * @summary This method takes a model and converts it to an Angular component output.\n   * It first converts the model to a field definition using the base RenderingEngine's\n   * toFieldDefinition method, then converts that field definition to an Angular component output.\n   * @template M - Type extending Model\n   * @param {M} model - The model to render\n   * @param {Record<string, unknown>} globalProps - Global properties to pass to the component\n   * @param {ViewContainerRef} vcr - The view container reference for component creation\n   * @param {Injector} injector - The Angular injector for dependency injection\n   * @param {TemplateRef<any>} tpl - The template reference for content projection\n   * @return {AngularDynamicOutput} The Angular component output with component reference and inputs\n   * @mermaid\n   * sequenceDiagram\n   *   participant Client as Client Code\n   *   participant Render as render method\n   *   participant ToField as toFieldDefinition\n   *   participant FromField as fromFieldDefinition\n   *\n   *   Client->>Render: render(model, globalProps, vcr, injector, tpl)\n   *   Render->>ToField: toFieldDefinition(model, globalProps)\n   *   ToField-->>Render: fieldDef\n   *   Render->>FromField: fromFieldDefinition(fieldDef, vcr, injector, tpl)\n   *   FromField-->>Render: AngularDynamicOutput\n   *   Render-->>Client: return AngularDynamicOutput\n   */\n  override render<M extends Model>(\n    model: M,\n    globalProps: Record<string, unknown>,\n    vcr: ViewContainerRef,\n    injector: Injector,\n    tpl: TemplateRef<unknown>,\n  ): AngularDynamicOutput {\n    let result: AngularDynamicOutput;\n    try {\n      const fieldDef = this.toFieldDefinition(model, globalProps);\n      result = this.fromFieldDefinition(fieldDef, vcr, injector, tpl);\n    } catch (e: unknown) {\n      throw new InternalError(\n        `Failed to render Model ${model.constructor.name}: ${e}`,\n      );\n    }\n\n    return result;\n  }\n\n  /**\n   * @description Initializes the rendering engine\n   * @summary This method initializes the rendering engine. It checks if the engine is already initialized\n   * and sets the initialized flag to true. This method is called before the engine is used.\n   * @return {Promise<void>} A promise that resolves when initialization is complete\n   */\n  override async initialize(): Promise<void> {\n    if (this.initialized) return;\n    this.initialized = true;\n  }\n\n  /**\n   * @description Registers a component with the rendering engine\n   * @summary This static method registers a component constructor with the rendering engine\n   * under a specific name. It throws an error if a component is already registered under the same name.\n   * @param {string} name - The name to register the component under\n   * @param {Constructor<unknown>} constructor - The component constructor\n   * @return {void}\n   */\n  static registerComponent(name: string, constructor: Constructor<unknown>) {\n    if (!this._components) this._components = {};\n    if (name in this._components)\n      throw new InternalError(`Component already registered under ${name}`);\n    this._components[name] = {\n      constructor: constructor,\n    };\n  }\n\n  /**\n   * @description Retrieves registered components from the rendering engine\n   * @summary This static method retrieves either all registered components or a specific component\n   * by its selector. It throws an error if the requested component is not registered.\n   * @param {string} [selector] - Optional selector to retrieve a specific component\n   * @return {Object|Array} Either a specific component or an array of all components\n   */\n  static components(selector?: string) {\n    if (!selector) return Object.values(this._components);\n    if (!(selector in this._components))\n      throw new InternalError(`No Component registered under ${selector}`);\n    return this._components[selector];\n  }\n\n  /**\n   * @description Generates a key for reflection metadata\n   * @summary This static method generates a key for reflection metadata by prefixing the input key\n   * with the Angular engine's reflection prefix. This is used for storing and retrieving metadata.\n   * @param {string} key - The base key to prefix\n   * @return {string} The prefixed key for reflection metadata\n   */\n  static override key(key: string) {\n    return `${AngularEngineKeys.REFLECT}${key}`;\n  }\n}\n"]}
364
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"NgxRenderingEngine.js","sourceRoot":"","sources":["../../../../src/lib/engine/NgxRenderingEngine.ts"],"names":[],"mappings":"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","sourcesContent":["import { FieldDefinition, RenderingEngine } from '@decaf-ts/ui-decorators';\nimport { AngularFieldDefinition, KeyValue } from './types';\nimport { AngularDynamicOutput } from './interfaces';\nimport { AngularEngineKeys } from './constants';\nimport { Constructor, Model} from '@decaf-ts/decorator-validation';\nimport { InternalError } from '@decaf-ts/db-decorators';\nimport {\n  ComponentMirror,\n  ComponentRef,\n  EnvironmentInjector,\n  Injector,\n  reflectComponentType,\n  TemplateRef,\n  Type,\n  ViewContainerRef,\n} from '@angular/core';\nimport { NgxFormService } from './NgxFormService';\n\n/**\n * @description Angular implementation of the RenderingEngine with enhanced features\n * @summary This class extends the base RenderingEngine to provide Angular-specific rendering capabilities\n * with additional features compared to NgxRenderingEngine. It handles the conversion of field definitions\n * to Angular components, manages component registration, and provides utilities for component creation\n * and input handling. This implementation uses Angular's newer component APIs.\n *\n * @template AngularFieldDefinition - Type for Angular-specific field definitions\n * @template AngularDynamicOutput - Type for Angular-specific component output\n *\n * @class NgxRenderingEngine\n * @example\n * ```typescript\n * const engine = NgxRenderingEngine.get();\n * engine.initialize();\n * const output = engine.render(myModel, {}, viewContainerRef, injector, templateRef);\n * ```\n *\n * @mermaid\n * sequenceDiagram\n *   participant Client\n *   participant Engine as NgxRenderingEngine\n *   participant Components as RegisteredComponents\n *\n *   Client->>Engine: get()\n *   Client->>Engine: initialize()\n *   Client->>Engine: render(model, props, vcr, injector, tpl)\n *   Engine->>Engine: toFieldDefinition(model, props)\n *   Engine->>Engine: fromFieldDefinition(fieldDef, vcr, injector, tpl)\n *   Engine->>Components: components(fieldDef.tag)\n *   Components-->>Engine: component constructor\n *   Engine->>Engine: createComponent(component, inputs, metadata, vcr, injector, template)\n *   Engine-->>Client: return AngularDynamicOutput\n */\nexport class NgxRenderingEngine extends RenderingEngine<AngularFieldDefinition, AngularDynamicOutput> {\n\n  /**\n   * @description Registry of components available for rendering\n   * @summary Static registry that stores all registered components indexed by their selector name.\n   * Each component entry contains a constructor reference that can be used to instantiate\n   * the component during the rendering process. This registry is shared across all instances\n   * of the rendering engine and is populated through the registerComponent method.\n   *\n   * @private\n   * @static\n   * @type {Record<string, { constructor: Constructor<unknown> }>}\n   */\n  private static _components: Record<string, { constructor: Constructor<unknown> }>;\n\n  /**\n   * @description Currently active model being rendered\n   * @summary Stores a reference to the model instance that is currently being processed\n   * by the rendering engine. This property is set during the render method execution\n   * and is used throughout the rendering lifecycle to access model data and metadata.\n   * The definite assignment assertion (!) is used because this property is always\n   * initialized before use within the render method.\n   *\n   * @private\n   * @type {Model}\n   */\n  private _model!: Model;\n\n  /**\n   * @description Current operation context for component visibility control\n   * @summary Static property that stores the current operation being performed,\n   * which is used to determine component visibility through the 'hidden' property.\n   * Components can specify operations where they should be hidden, and this property\n   * provides the context for those visibility checks. The value is typically extracted\n   * from the global properties during the rendering process.\n   *\n   * @private\n   * @static\n   * @type {string | undefined}\n   */\n  private static _operation: string | undefined = undefined;\n\n  /**\n   * @description Reference to the currently active component instance\n   * @summary Static property that maintains a reference to the most recently created\n   * component instance. This is used internally for component lifecycle management\n   * and can be cleared through the destroy method. The reference allows access to\n   * the active component instance for operations that need to interact with the\n   * currently rendered component.\n   *\n   * @private\n   * @static\n   * @type {Type<unknown> | undefined}\n   */\n  private static _instance: Type<unknown> | undefined;\n\n  /**\n   * @description Constructs a new NgxRenderingEngine instance\n   * @summary Initializes a new instance of the Angular rendering engine by calling the parent\n   * constructor with the 'angular' engine type identifier. This constructor sets up the base\n   * rendering engine functionality with Angular-specific configurations and prepares the\n   * instance for component registration and rendering operations.\n   *\n   * @constructor\n   */\n  constructor() {\n    super('angular');\n  }\n\n  /**\n   * @description Converts a field definition to an Angular component output\n   * @summary This private method takes a field definition and creates the corresponding Angular component.\n   * It handles component instantiation, input property mapping, and child component rendering.\n   * The method validates input properties against the component's metadata and processes\n   * child components recursively.\n   *\n   * @param {FieldDefinition<AngularFieldDefinition>} fieldDef - The field definition to convert\n   * @param {ViewContainerRef} vcr - The view container reference for component creation\n   * @param {Injector} injector - The Angular injector for dependency injection\n   * @param {TemplateRef<any>} tpl - The template reference for content projection\n   * @param {string} registryFormId - Form identifier for the component renderer\n   * @return {AngularDynamicOutput} The Angular component output with component reference and inputs\n   *\n   * @mermaid\n   * sequenceDiagram\n   *   participant Method as fromFieldDefinition\n   *   participant Components as NgxRenderingEngine.components\n   *   participant Angular as Angular Core\n   *   participant Process as processChild\n   *\n   *   Method->>Components: components(fieldDef.tag)\n   *   Components-->>Method: component constructor\n   *   Method->>Angular: reflectComponentType(component)\n   *   Angular-->>Method: componentMetadata\n   *   Method->>Method: Validate input properties\n   *   Method->>Method: Create result object\n   *   alt Has children\n   *     Method->>Process: Process children recursively\n   *     Process->>Method: Return processed children\n   *     Method->>Angular: Create embedded view\n   *     Method->>Method: Create component instance\n   *   end\n   *   Method-->>Caller: return AngularDynamicOutput\n   */\n  private fromFieldDefinition(\n    fieldDef: FieldDefinition<AngularFieldDefinition>,\n    vcr: ViewContainerRef,\n    injector: Injector,\n    tpl: TemplateRef<unknown>,\n    registryFormId: string = Date.now().toString(36).toUpperCase(),\n  ): AngularDynamicOutput {\n    const cmp = (fieldDef as KeyValue)?.['component'] || NgxRenderingEngine.components(fieldDef.tag);\n    const component = cmp.constructor as unknown as Type<unknown>;\n\n    const componentMetadata = reflectComponentType(component);\n    if (!componentMetadata) {\n      throw new InternalError(`Metadata for component ${fieldDef.tag} not found.`);\n    }\n\n    const { inputs: possibleInputs } = componentMetadata;\n    const inputs = { ...fieldDef.props };\n\n    const unmappedKeys = Object.keys(inputs).filter(input => {\n      const isMapped = possibleInputs.find(({ propName }) => propName === input);\n      if (!isMapped) delete inputs[input];\n      return !isMapped;\n    });\n\n    if (unmappedKeys.length > 0)\n      console.warn(`Unmapped input properties for component ${fieldDef.tag}: ${unmappedKeys.join(', ')}`);\n\n    const operation = NgxRenderingEngine._operation;\n\n    const hiddenOn = inputs?.hidden || [];\n    if((hiddenOn as string[]).includes(operation as string))\n      return {inputs, injector};\n    // const hiddenOn = inputs?.hidden || [];\n    const result: AngularDynamicOutput = {\n      component,\n      inputs,\n      injector,\n    };\n\n    if (fieldDef.rendererId)\n      (result.inputs as Record<string, unknown>)['rendererId'] = fieldDef.rendererId;\n\n    // process children\n    if (fieldDef.children?.length) {\n      result.children = fieldDef.children.map((child) => {\n        if(child?.children?.length) {\n          child.children = child.children.filter(c => {\n            const hiddenOn = c?.props?.hidden || [];\n            if(!(hiddenOn as string[]).includes(operation as string))\n              return c\n          })\n        }\n        // create a child form and add its controls as properties of child.props\n        NgxFormService.addControlFromProps(registryFormId, child.props, inputs);\n        return this.fromFieldDefinition(child, vcr, injector, tpl, registryFormId);\n      });\n    }\n\n    // generating DOM\n    vcr.clear();\n    const template = vcr.createEmbeddedView(tpl, injector).rootNodes;\n    const componentInstance = NgxRenderingEngine.createComponent(\n      component,\n      { ...inputs, model: this._model },\n      componentMetadata,\n      vcr,\n      injector,\n      template,\n    );\n\n    result.instance = NgxRenderingEngine._instance = componentInstance.instance as Type<unknown>;\n\n    return result;\n  }\n\n\n  /**\n   * @description Creates an Angular component instance\n   * @summary This static utility method creates an Angular component instance with the specified\n   * inputs and template. It uses Angular's component creation API to instantiate the component\n   * and then sets the input properties using the provided metadata.\n   *\n   * @param {Type<unknown>} component - The component type to create\n   * @param {KeyValue} [inputs={}] - The input properties to set on the component\n   * @param {ComponentMirror<unknown>} metadata - The component metadata for input validation\n   * @param {ViewContainerRef} vcr - The view container reference for component creation\n   * @param {Injector} injector - The Angular injector for dependency injection\n   * @param {Node[]} [template=[]] - The template nodes to project into the component\n   * @return {ComponentRef<unknown>} The created component reference\n   */\n  static createComponent(component: Type<unknown>, inputs: KeyValue = {}, metadata: ComponentMirror<unknown>, vcr: ViewContainerRef, injector: Injector, template: Node[] = []): ComponentRef<unknown> {\n    const componentInstance = vcr.createComponent(component as Type<unknown>, {\n      environmentInjector: injector as EnvironmentInjector,\n      projectableNodes: [template],\n    });\n    this.setInputs(componentInstance, inputs, metadata);\n    return componentInstance;\n  }\n\n  /**\n   * @description Extracts decorator metadata from a model\n   * @summary This method provides access to the field definition generated from a model's\n   * decorators. It's a convenience wrapper around the toFieldDefinition method that\n   * converts a model to a field definition based on its decorators and the provided\n   * global properties.\n   *\n   * @param {Model} model - The model to extract decorators from\n   * @param {Record<string, unknown>} globalProps - Global properties to include in the field definition\n   * @return {FieldDefinition<AngularFieldDefinition>} The field definition generated from the model\n   */\n  getDecorators(model: Model, globalProps: Record<string, unknown>): FieldDefinition<AngularFieldDefinition> {\n    return this.toFieldDefinition(model, globalProps);\n  }\n\n  /**\n   * @description Destroys the current engine instance\n   * @summary This static method clears the current instance reference, effectively\n   * destroying the singleton instance of the rendering engine. This can be used\n   * to reset the engine state or to prepare for a new instance creation.\n   *\n   * @return {Promise<void>} A promise that resolves when the instance is destroyed\n   */\n  static async destroy(): Promise<void> {\n    NgxRenderingEngine._instance = undefined;\n  }\n\n\n  /**\n   * @description Renders a model into an Angular component output\n   * @summary This method takes a model and converts it to an Angular component output.\n   * It first stores a reference to the model, then converts it to a field definition\n   * using the base RenderingEngine's toFieldDefinition method, and finally converts\n   * that field definition to an Angular component output using fromFieldDefinition.\n   *\n   * @template M - Type extending Model\n   * @param {M} model - The model to render\n   * @param {Record<string, unknown>} globalProps - Global properties to pass to the component\n   * @param {ViewContainerRef} vcr - The view container reference for component creation\n   * @param {Injector} injector - The Angular injector for dependency injection\n   * @param {TemplateRef<any>} tpl - The template reference for content projection\n   * @return {AngularDynamicOutput} The Angular component output with component reference and inputs\n   *\n   * @mermaid\n   * sequenceDiagram\n   *   participant Client as Client Code\n   *   participant Render as render method\n   *   participant ToField as toFieldDefinition\n   *   participant FromField as fromFieldDefinition\n   *\n   *   Client->>Render: render(model, globalProps, vcr, injector, tpl)\n   *   Render->>Render: Store model reference\n   *   Render->>ToField: toFieldDefinition(model, globalProps)\n   *   ToField-->>Render: fieldDef\n   *   Render->>FromField: fromFieldDefinition(fieldDef, vcr, injector, tpl)\n   *   FromField-->>Render: AngularDynamicOutput\n   *   Render-->>Client: return AngularDynamicOutput\n   */\n  override render<M extends Model>(\n    model: M,\n    globalProps: Record<string, unknown>,\n    vcr: ViewContainerRef,\n    injector: Injector,\n    tpl: TemplateRef<unknown>,\n  ): AngularDynamicOutput {\n    let result: AngularDynamicOutput;\n    try {\n      this._model = model;\n      const formId = Date.now().toString(36).toUpperCase();\n      const fieldDef = this.toFieldDefinition(model, globalProps);\n      const props = fieldDef.props as KeyValue;\n      if(!NgxRenderingEngine._operation)\n        NgxRenderingEngine._operation = props?.['operation'] || undefined;\n      result = this.fromFieldDefinition(fieldDef, vcr, injector, tpl, formId);\n\n      (result!.instance! as KeyValue)['formGroup'] = NgxFormService.getControlFromForm(formId);\n      NgxFormService.removeRegistry(formId);\n    } catch (e: unknown) {\n      throw new InternalError(\n        `Failed to render Model ${model.constructor.name}: ${e}`,\n      );\n    }\n\n    return result;\n  }\n\n  /**\n   * @description Initializes the rendering engine\n   * @summary This method initializes the rendering engine. It checks if the engine is already initialized\n   * and sets the initialized flag to true. This method is called before the engine is used\n   * to ensure it's properly set up for rendering operations.\n   *\n   * @return {Promise<void>} A promise that resolves when initialization is complete\n   */\n  override async initialize(): Promise<void> {\n    if (this.initialized)\n      return;\n    // ValidatableByType[]\n    this.initialized = true;\n  }\n\n  /**\n   * @description Registers a component with the rendering engine\n   * @summary This static method registers a component constructor with the rendering engine\n   * under a specific name. It initializes the components registry if needed and throws\n   * an error if a component is already registered under the same name to prevent\n   * accidental overrides.\n   *\n   * @param {string} name - The name to register the component under\n   * @param {Constructor<unknown>} constructor - The component constructor\n   * @return {void}\n   */\n  static registerComponent(name: string, constructor: Constructor<unknown>): void {\n    if (!this._components) this._components = {};\n    if (name in this._components)\n      throw new InternalError(`Component already registered under ${name}`);\n    this._components[name] = {\n      constructor: constructor,\n    };\n  }\n\n  /**\n   * @description Retrieves registered components from the rendering engine\n   * @summary This static method retrieves either all registered components or a specific component\n   * by its selector. When called without a selector, it returns an array of all registered\n   * components. When called with a selector, it returns the specific component if found,\n   * or throws an error if the component is not registered.\n   *\n   * @param {string} [selector] - Optional selector to retrieve a specific component\n   * @return {Object|Array} Either a specific component or an array of all components\n   */\n  static components(selector?: string): object | string[] {\n    if (!selector) return Object.values(this._components);\n    if (!(selector in this._components))\n      throw new InternalError(`No Component registered under ${selector}`);\n    return this._components[selector];\n  }\n\n  /**\n   * @description Generates a key for reflection metadata\n   * @summary This static method generates a key for reflection metadata by prefixing the input key\n   * with the Angular engine's reflection prefix. This is used for storing and retrieving\n   * metadata in a namespaced way to avoid conflicts with other metadata.\n   *\n   * @param {string} key - The base key to prefix\n   * @return {string} The prefixed key for reflection metadata\n   */\n  static override key(key: string): string {\n    return `${AngularEngineKeys.REFLECT}${key}`;\n  }\n\n  /**\n   * @description Sets input properties on a component instance\n   * @summary This static utility method sets input properties on a component instance\n   * based on the provided inputs object and component metadata. It handles both simple\n   * values and nested objects, recursively processing object properties. The method\n   * validates each input against the component's metadata to ensure only valid inputs\n   * are set.\n   *\n   * @param {ComponentRef<unknown>} component - The component reference to set inputs on\n   * @param {KeyValue} inputs - The input properties to set\n   * @param {ComponentMirror<unknown>} metadata - The component metadata for input validation\n   * @return {void}\n   *\n   * @mermaid\n   * sequenceDiagram\n   *   participant Caller\n   *   participant SetInputs as setInputs\n   *   participant Parse as parseInputValue\n   *   participant Component as ComponentRef\n   *\n   *   Caller->>SetInputs: setInputs(component, inputs, metadata)\n   *   SetInputs->>SetInputs: Iterate through inputs\n   *   loop For each input\n   *     SetInputs->>SetInputs: Check if input exists in metadata\n   *     alt Input is 'props'\n   *       SetInputs->>Parse: parseInputValue(component, value)\n   *       Parse->>Parse: Recursively process nested objects\n   *       Parse->>Component: setInput(key, value)\n   *     else Input is valid\n   *       SetInputs->>Component: setInput(key, value)\n   *     end\n   *   end\n   */\n  static setInputs(component: ComponentRef<unknown>, inputs: KeyValue, metadata: ComponentMirror<unknown>): void {\n    function parseInputValue(component: ComponentRef<unknown>, input: KeyValue) {\n      Object.keys(input).forEach(key => {\n        const value = input[key];\n        if (typeof value === 'object' && !!value)\n          return parseInputValue(component, value);\n        component.setInput(key, value);\n      });\n    }\n\n    Object.entries(inputs).forEach(([key, value]) => {\n      const prop = metadata.inputs.find((item: { propName: string }) => item.propName === key);\n      if (prop) {\n        if (key === 'props')\n          parseInputValue(component, value);\n        // if(key === 'locale' && !value)\n        //   value = getLocaleFromClassName(this._componentName);\n        component.setInput(key, value);\n      }\n    });\n  }\n}\n"]}
@@ -1,7 +1,8 @@
1
1
  import { FormControl, FormGroup } from '@angular/forms';
2
- import { ComparisonValidationKeys, DEFAULT_PATTERNS, PathProxyEngine, Primitives, Validation, ValidationKeys, } from '@decaf-ts/decorator-validation';
2
+ import { ComparisonValidationKeys, DEFAULT_PATTERNS, PathProxyEngine, Validation, ValidationKeys, } from '@decaf-ts/decorator-validation';
3
3
  import { HTML5InputTypes, parseValueByType } from '@decaf-ts/ui-decorators';
4
4
  import { AngularEngineKeys } from './constants';
5
+ import { NgxRenderingEngine } from './NgxRenderingEngine';
5
6
  /**
6
7
  *
7
8
  * Resolves the correct validator key and its associated properties based on the input key and type.
@@ -25,7 +26,8 @@ const resolveValidatorKeyProps = (key, value, type) => {
25
26
  const isTypeBased = key === ValidationKeys.TYPE && Object.keys(patternValidators).includes(type);
26
27
  const validatorKey = isTypeBased ? type : key;
27
28
  const props = {
28
- [validatorKey]: value,
29
+ // [validatorKey]: (!isTypeBased && key === 'type') ? parseType(type) : value,
30
+ [validatorKey]: (!isTypeBased && validatorKey === ValidationKeys.TYPE) ? NgxRenderingEngine.get().translate(value, false) : value,
29
31
  // Email, Password, and URL are validated using the "pattern" key
30
32
  ...(isTypeBased && { [ValidationKeys.PATTERN]: patternValidators[type] }),
31
33
  };
@@ -52,7 +54,7 @@ export class ValidatorFactory {
52
54
  let errs;
53
55
  try {
54
56
  if (!props['types'] && !props['customTypes'])
55
- props['types'] = props['type'] === 'text' ? Primitives.STRING : type;
57
+ props['types'] = props['type'];
56
58
  errs = validator.hasErrors(value, props, proxy);
57
59
  }
58
60
  catch (e) {
@@ -101,4 +103,4 @@ export class ValidatorFactory {
101
103
  });
102
104
  }
103
105
  }
104
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ValidatorFactory.js","sourceRoot":"","sources":["../../../../src/lib/engine/ValidatorFactory.ts"],"names":[],"mappings":"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","sourcesContent":["import { AbstractControl, FormControl, FormGroup, ValidationErrors, ValidatorFn } from '@angular/forms';\nimport {\n  ComparisonValidationKeys,\n  DEFAULT_PATTERNS,\n  PathProxy,\n  PathProxyEngine,\n  Primitives,\n  Validation,\n  ValidationKeys,\n  Validator,\n} from '@decaf-ts/decorator-validation';\nimport { FieldProperties, HTML5InputTypes, parseValueByType } from '@decaf-ts/ui-decorators';\nimport { AngularEngineKeys } from './constants';\nimport { KeyValue } from './types';\n\ntype ComparisonValidationKey = typeof ComparisonValidationKeys[keyof typeof ComparisonValidationKeys];\n\n/**\n *\n * Resolves the correct validator key and its associated properties based on the input key and type.\n *\n * When the validation key is TYPE, it's necessary to resolve the actual validator based on the\n * field's type (e.g., 'password', 'email', 'url') instead of using the generic getValidator(\"type\") logic.\n * This allows directly invoking specific validators like getValidator('password'), ensuring the correct\n * behavior for type-based validation.\n *\n * @param key - The validation key (e.g., 'type', 'required', etc.).\n * @param value - The value that needs be provided to the validator.\n * @param type - The field's declared type.\n * @returns An object containing the resolved validator key and its corresponding props.\n */\nconst resolveValidatorKeyProps = (key: string, value: unknown, type: string): {\n  validatorKey: string;\n  props: Record<string, unknown>;\n} => {\n  const patternValidators: Record<string, unknown> = {\n    [ValidationKeys.PASSWORD]: DEFAULT_PATTERNS.PASSWORD.CHAR8_ONE_OF_EACH,\n    [ValidationKeys.EMAIL]: DEFAULT_PATTERNS.EMAIL,\n    [ValidationKeys.URL]: DEFAULT_PATTERNS.URL,\n  };\n\n  const isTypeBased = key === ValidationKeys.TYPE && Object.keys(patternValidators).includes(type);\n  const validatorKey = isTypeBased ? type : key;\n  const props: Record<string, unknown> = {\n    [validatorKey]: value,\n    // Email, Password, and URL are validated using the \"pattern\" key\n    ...(isTypeBased && { [ValidationKeys.PATTERN]: patternValidators[type] }),\n  };\n\n  return { validatorKey, props };\n};\n\n\nexport class ValidatorFactory {\n  static spawn(fieldProps: FieldProperties, key: string): ValidatorFn {\n    if (!Validation.keys().includes(key))\n      throw new Error('Unsupported custom validation');\n\n    const validatorFn: ValidatorFn = (control: AbstractControl): ValidationErrors | null => {\n      const { name, type } = fieldProps;\n      const { validatorKey, props } = resolveValidatorKeyProps(key, fieldProps[key as keyof FieldProperties], type);\n      const validator = Validation.get(validatorKey) as Validator;\n\n      // parseValueByType does not support undefined values\n      const value = typeof control.value !== 'undefined'\n        ? parseValueByType(type, type === HTML5InputTypes.CHECKBOX ? name : control.value, fieldProps)\n        : undefined;\n\n      // Create a proxy to enable access to parent and child values\n      let proxy: PathProxy<unknown> = ValidatorFactory.createProxy({} as AbstractControl);\n      if (Object.values(ComparisonValidationKeys).includes(key as ComparisonValidationKey)) {\n        const parent: FormGroup = control instanceof FormGroup ? control : (control as KeyValue)[AngularEngineKeys.PARENT];\n        proxy = ValidatorFactory.createProxy(parent) as PathProxy<unknown>;\n      }\n\n      let errs: string | undefined;\n      try {\n        if(!props['types'] && !props['customTypes'])\n          props['types'] = props['type'] === 'text' ? Primitives.STRING : type;\n        errs = validator.hasErrors(value, props, proxy);\n      } catch (e: unknown) {\n        errs = `${key} validator failed to validate: ${e}`;\n        console.warn(errs);\n      }\n\n      return errs ? { [validatorKey]: true } : null;\n    };\n\n    Object.defineProperty(validatorFn, 'name', {\n      value: `${key}Validator`,\n    });\n\n    return validatorFn;\n  }\n\n  /**\n   * @summary Creates a proxy wrapper for an Angular AbstractControl to assist with custom validation logic.\n   * @description Returns a structured proxy object that simulates a hierarchical tree of form values.\n   * Enables Validators handling method to access parent and child properties using consistent dot-notation in Angular forms.\n   *\n   * @param {AbstractControl} control - The control to wrap in a proxy.\n   * @returns {PathProxy<unknown>} A proxy object exposing form values and enabling recursive parent access.\n   */\n  static createProxy(control: AbstractControl): PathProxy<unknown> {\n    return PathProxyEngine.create(control, {\n      getValue(target: AbstractControl, prop: string): unknown {\n        if (target instanceof FormControl)\n          return target.value;\n\n        if (target instanceof FormGroup) {\n          const control = target.controls[prop];\n          return control instanceof FormControl ? control.value : control;\n        }\n\n        // const value = target[prop];\n        // if (value instanceof FormControl)\n        //   return value.value;\n        //\n        // if (value instanceof FormGroup) {\n        //   const control = value.controls[prop];\n        //   return control instanceof FormControl ? control.value : control;\n        // }\n\n        return (target as KeyValue)?.[prop];\n      },\n      getParent: function(target: AbstractControl)  {\n        return target?.['_parent'];\n      },\n      ignoreUndefined: true,\n      ignoreNull: true,\n    });\n  }\n}\n"]}
106
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ValidatorFactory.js","sourceRoot":"","sources":["../../../../src/lib/engine/ValidatorFactory.ts"],"names":[],"mappings":"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","sourcesContent":["import { AbstractControl, FormControl, FormGroup, ValidationErrors, ValidatorFn } from '@angular/forms';\nimport {\n  ComparisonValidationKeys,\n  DEFAULT_PATTERNS,\n  PathProxy,\n  PathProxyEngine,\n  Validation,\n  ValidationKeys,\n  Validator,\n} from '@decaf-ts/decorator-validation';\nimport { FieldProperties, HTML5InputTypes, parseValueByType } from '@decaf-ts/ui-decorators';\nimport { AngularEngineKeys } from './constants';\nimport { KeyValue } from './types';\nimport { NgxRenderingEngine } from './NgxRenderingEngine';\n\n\ntype ComparisonValidationKey = typeof ComparisonValidationKeys[keyof typeof ComparisonValidationKeys];\n\n/**\n *\n * Resolves the correct validator key and its associated properties based on the input key and type.\n *\n * When the validation key is TYPE, it's necessary to resolve the actual validator based on the\n * field's type (e.g., 'password', 'email', 'url') instead of using the generic getValidator(\"type\") logic.\n * This allows directly invoking specific validators like getValidator('password'), ensuring the correct\n * behavior for type-based validation.\n *\n * @param key - The validation key (e.g., 'type', 'required', etc.).\n * @param value - The value that needs be provided to the validator.\n * @param type - The field's declared type.\n * @returns An object containing the resolved validator key and its corresponding props.\n */\nconst resolveValidatorKeyProps = (key: string, value: unknown, type: string): {\n  validatorKey: string;\n  props: Record<string, unknown>;\n} => {\n  const patternValidators: Record<string, unknown> = {\n    [ValidationKeys.PASSWORD]: DEFAULT_PATTERNS.PASSWORD.CHAR8_ONE_OF_EACH,\n    [ValidationKeys.EMAIL]: DEFAULT_PATTERNS.EMAIL,\n    [ValidationKeys.URL]: DEFAULT_PATTERNS.URL,\n  };\n  const isTypeBased = key === ValidationKeys.TYPE && Object.keys(patternValidators).includes(type);\n  const validatorKey = isTypeBased ? type : key;\n  const props: Record<string, unknown> = {\n    // [validatorKey]: (!isTypeBased && key === 'type') ? parseType(type) : value,\n    [validatorKey]: (!isTypeBased && validatorKey === ValidationKeys.TYPE) ? NgxRenderingEngine.get().translate(value as string, false) : value,\n    // Email, Password, and URL are validated using the \"pattern\" key\n    ...(isTypeBased && { [ValidationKeys.PATTERN] : patternValidators[type] }),\n  };\n\n  return { validatorKey, props };\n};\n\n\nexport class ValidatorFactory {\n  static spawn(fieldProps: FieldProperties, key: string): ValidatorFn {\n    if (!Validation.keys().includes(key))\n      throw new Error('Unsupported custom validation');\n\n    const validatorFn: ValidatorFn = (control: AbstractControl): ValidationErrors | null => {\n      const { name, type } = fieldProps;\n      const { validatorKey, props } = resolveValidatorKeyProps(key, fieldProps[key as keyof FieldProperties], type);\n      const validator = Validation.get(validatorKey) as Validator;\n\n      // parseValueByType does not support undefined values\n      const value = typeof control.value !== 'undefined'\n        ? parseValueByType(type, type === HTML5InputTypes.CHECKBOX ? name : control.value, fieldProps)\n        : undefined;\n\n      // Create a proxy to enable access to parent and child values\n      let proxy: PathProxy<unknown> = ValidatorFactory.createProxy({} as AbstractControl);\n      if (Object.values(ComparisonValidationKeys).includes(key as ComparisonValidationKey)) {\n        const parent: FormGroup = control instanceof FormGroup ? control : (control as KeyValue)[AngularEngineKeys.PARENT];\n        proxy = ValidatorFactory.createProxy(parent) as PathProxy<unknown>;\n      }\n\n      let errs: string | undefined;\n      try {\n        if(!props['types'] && !props['customTypes'])\n          props['types'] = props['type'];\n        errs = validator.hasErrors(value, props, proxy);\n      } catch (e: unknown) {\n        errs = `${key} validator failed to validate: ${e}`;\n        console.warn(errs);\n      }\n\n      return errs ? { [validatorKey]: true } : null;\n    };\n\n    Object.defineProperty(validatorFn, 'name', {\n      value: `${key}Validator`,\n    });\n\n    return validatorFn;\n  }\n\n  /**\n   * @summary Creates a proxy wrapper for an Angular AbstractControl to assist with custom validation logic.\n   * @description Returns a structured proxy object that simulates a hierarchical tree of form values.\n   * Enables Validators handling method to access parent and child properties using consistent dot-notation in Angular forms.\n   *\n   * @param {AbstractControl} control - The control to wrap in a proxy.\n   * @returns {PathProxy<unknown>} A proxy object exposing form values and enabling recursive parent access.\n   */\n  static createProxy(control: AbstractControl): PathProxy<unknown> {\n    return PathProxyEngine.create(control, {\n      getValue(target: AbstractControl, prop: string): unknown {\n        if (target instanceof FormControl)\n          return target.value;\n\n        if (target instanceof FormGroup) {\n          const control = target.controls[prop];\n          return control instanceof FormControl ? control.value : control;\n        }\n\n        // const value = target[prop];\n        // if (value instanceof FormControl)\n        //   return value.value;\n        //\n        // if (value instanceof FormGroup) {\n        //   const control = value.controls[prop];\n        //   return control instanceof FormControl ? control.value : control;\n        // }\n\n        return (target as KeyValue)?.[prop];\n      },\n      getParent: function(target: AbstractControl)  {\n        return target?.['_parent'];\n      },\n      ignoreUndefined: true,\n      ignoreNull: true,\n    });\n  }\n}\n"]}