@decaf-ts/for-angular 0.0.8 → 0.0.10

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 (105) hide show
  1. package/dist/{lib/assets → assets}/i18n/en.json +10 -1
  2. package/dist/components/empty-state/empty-state.component.d.ts +301 -0
  3. package/dist/components/fieldset/fieldset.component.d.ts +199 -0
  4. package/dist/components/filter/filter.component.d.ts +505 -0
  5. package/dist/components/for-angular-components.module.d.ts +20 -0
  6. package/dist/components/index.d.ts +16 -0
  7. package/dist/components/layout/layout.component.d.ts +133 -0
  8. package/dist/components/list/constants.d.ts +29 -0
  9. package/dist/components/list/list.component.d.ts +849 -0
  10. package/dist/components/list-item/list-item.component.d.ts +390 -0
  11. package/dist/{lib/components → components}/model-renderer/model-renderer.component.d.ts +1 -1
  12. package/dist/components/pagination/constants.d.ts +7 -0
  13. package/dist/components/pagination/pagination.component.d.ts +264 -0
  14. package/dist/components/searchbar/searchbar.component.d.ts +407 -0
  15. package/dist/directives/collapsable.directive.d.ts +8 -0
  16. package/dist/directives/index.d.ts +1 -0
  17. package/dist/engine/NgxBaseComponent.d.ts +541 -0
  18. package/dist/{lib/engine → engine}/index.d.ts +1 -0
  19. package/dist/{lib/engine → engine}/types.d.ts +44 -0
  20. package/dist/{lib/esm2022 → esm2022}/components/component-renderer/component-renderer.component.mjs +3 -3
  21. package/dist/esm2022/components/crud-field/crud-field.component.mjs +301 -0
  22. package/dist/esm2022/components/crud-form/constants.mjs +14 -0
  23. package/dist/esm2022/components/crud-form/crud-form.component.mjs +139 -0
  24. package/dist/esm2022/components/crud-form/types.mjs +2 -0
  25. package/dist/esm2022/components/empty-state/empty-state.component.mjs +348 -0
  26. package/dist/esm2022/components/fieldset/fieldset.component.mjs +225 -0
  27. package/dist/esm2022/components/filter/filter.component.mjs +689 -0
  28. package/dist/esm2022/components/for-angular-components.module.mjs +71 -0
  29. package/dist/esm2022/components/index.mjs +20 -0
  30. package/dist/esm2022/components/layout/layout.component.mjs +176 -0
  31. package/dist/esm2022/components/list/constants.mjs +6 -0
  32. package/dist/esm2022/components/list/list.component.mjs +1236 -0
  33. package/dist/esm2022/components/list-item/list-item.component.mjs +408 -0
  34. package/dist/esm2022/components/model-renderer/model-renderer.component.mjs +138 -0
  35. package/dist/esm2022/components/pagination/constants.mjs +2 -0
  36. package/dist/esm2022/components/pagination/pagination.component.mjs +323 -0
  37. package/dist/esm2022/components/searchbar/searchbar.component.mjs +493 -0
  38. package/dist/esm2022/decaf-ts-for-angular.mjs +5 -0
  39. package/dist/esm2022/directives/collapsable.directive.mjs +28 -0
  40. package/dist/esm2022/directives/index.mjs +2 -0
  41. package/dist/esm2022/engine/DynamicModule.mjs +18 -0
  42. package/dist/esm2022/engine/NgxBaseComponent.mjs +539 -0
  43. package/dist/esm2022/engine/NgxCrudFormField.mjs +125 -0
  44. package/dist/esm2022/engine/NgxFormService.mjs +315 -0
  45. package/dist/esm2022/engine/NgxRenderingEngine.mjs +192 -0
  46. package/dist/esm2022/engine/NgxRenderingEngine2.mjs +332 -0
  47. package/dist/esm2022/engine/ValidatorFactory.mjs +102 -0
  48. package/dist/esm2022/engine/constants.mjs +160 -0
  49. package/dist/esm2022/engine/decorators.mjs +38 -0
  50. package/dist/esm2022/engine/index.mjs +17 -0
  51. package/dist/esm2022/engine/types.mjs +4 -0
  52. package/dist/esm2022/for-angular.module.mjs +118 -0
  53. package/dist/esm2022/helpers/index.mjs +13 -0
  54. package/dist/esm2022/helpers/utils.mjs +415 -0
  55. package/dist/esm2022/interfaces.mjs +2 -0
  56. package/dist/esm2022/public-apis.mjs +14 -0
  57. package/dist/fesm2022/decaf-ts-for-angular.mjs +7109 -0
  58. package/dist/fesm2022/decaf-ts-for-angular.mjs.map +1 -0
  59. package/dist/helpers/index.d.ts +12 -0
  60. package/dist/helpers/utils.d.ts +253 -0
  61. package/dist/{lib/public-apis.d.ts → public-apis.d.ts} +4 -3
  62. package/package.json +5 -3
  63. package/dist/lib/esm2022/components/crud-field/crud-field.component.mjs +0 -297
  64. package/dist/lib/esm2022/components/crud-form/constants.mjs +0 -14
  65. package/dist/lib/esm2022/components/crud-form/crud-form.component.mjs +0 -139
  66. package/dist/lib/esm2022/components/crud-form/types.mjs +0 -2
  67. package/dist/lib/esm2022/components/model-renderer/model-renderer.component.mjs +0 -138
  68. package/dist/lib/esm2022/decaf-ts-for-angular.mjs +0 -5
  69. package/dist/lib/esm2022/engine/DynamicModule.mjs +0 -18
  70. package/dist/lib/esm2022/engine/NgxCrudFormField.mjs +0 -123
  71. package/dist/lib/esm2022/engine/NgxFormService.mjs +0 -315
  72. package/dist/lib/esm2022/engine/NgxRenderingEngine.mjs +0 -192
  73. package/dist/lib/esm2022/engine/NgxRenderingEngine2.mjs +0 -332
  74. package/dist/lib/esm2022/engine/ValidatorFactory.mjs +0 -102
  75. package/dist/lib/esm2022/engine/constants.mjs +0 -160
  76. package/dist/lib/esm2022/engine/decorators.mjs +0 -38
  77. package/dist/lib/esm2022/engine/index.mjs +0 -16
  78. package/dist/lib/esm2022/engine/types.mjs +0 -2
  79. package/dist/lib/esm2022/for-angular.module.mjs +0 -118
  80. package/dist/lib/esm2022/interfaces.mjs +0 -2
  81. package/dist/lib/esm2022/public-apis.mjs +0 -13
  82. package/dist/lib/fesm2022/decaf-ts-for-angular.mjs +0 -2153
  83. package/dist/lib/fesm2022/decaf-ts-for-angular.mjs.map +0 -1
  84. /package/dist/{lib/README.md → README.md} +0 -0
  85. /package/dist/{lib/assets → assets}/images/angular-logo.svg +0 -0
  86. /package/dist/{lib/assets → assets}/images/decaf-logo-black.svg +0 -0
  87. /package/dist/{lib/assets → assets}/images/decaf-logo-lw.svg +0 -0
  88. /package/dist/{lib/assets → assets}/images/decaf-logo-white.svg +0 -0
  89. /package/dist/{lib/assets → assets}/images/decaf-logo.svg +0 -0
  90. /package/dist/{lib/components → components}/component-renderer/component-renderer.component.d.ts +0 -0
  91. /package/dist/{lib/components → components}/crud-field/crud-field.component.d.ts +0 -0
  92. /package/dist/{lib/components → components}/crud-form/constants.d.ts +0 -0
  93. /package/dist/{lib/components → components}/crud-form/crud-form.component.d.ts +0 -0
  94. /package/dist/{lib/components → components}/crud-form/types.d.ts +0 -0
  95. /package/dist/{lib/engine → engine}/DynamicModule.d.ts +0 -0
  96. /package/dist/{lib/engine → engine}/NgxCrudFormField.d.ts +0 -0
  97. /package/dist/{lib/engine → engine}/NgxFormService.d.ts +0 -0
  98. /package/dist/{lib/engine → engine}/NgxRenderingEngine.d.ts +0 -0
  99. /package/dist/{lib/engine → engine}/NgxRenderingEngine2.d.ts +0 -0
  100. /package/dist/{lib/engine → engine}/ValidatorFactory.d.ts +0 -0
  101. /package/dist/{lib/engine → engine}/constants.d.ts +0 -0
  102. /package/dist/{lib/engine → engine}/decorators.d.ts +0 -0
  103. /package/dist/{lib/for-angular.module.d.ts → for-angular.module.d.ts} +0 -0
  104. /package/dist/{lib/index.d.ts → index.d.ts} +0 -0
  105. /package/dist/{lib/interfaces.d.ts → interfaces.d.ts} +0 -0
@@ -1,315 +0,0 @@
1
- import { escapeHtml, HTML5CheckTypes, HTML5InputTypes, parseToNumber } from '@decaf-ts/ui-decorators';
2
- import { FormControl, FormGroup, Validators } from '@angular/forms';
3
- import { isValidDate, parseDate, Validation } from '@decaf-ts/decorator-validation';
4
- import { ValidatorFactory } from './ValidatorFactory';
5
- /**
6
- * @description Service for managing Angular forms and form controls.
7
- * @summary The NgxFormService provides utility methods for creating, managing, and validating Angular forms and form controls. It includes functionality for registering forms, adding controls, validating fields, and handling form data.
8
- *
9
- * @class
10
- * @param {WeakMap<AbstractControl, FieldProperties>} controls - A WeakMap to store control properties.
11
- * @param {Map<string, FormGroup>} formRegistry - A Map to store registered forms.
12
- *
13
- * @example
14
- * // Creating a form from components
15
- * const components = [
16
- * { inputs: { name: 'username', type: 'text', required: true } },
17
- * { inputs: { name: 'password', type: 'password', minLength: 8 } }
18
- * ];
19
- * const form = NgxFormService.createFormFromComponents('loginForm', components, true);
20
- *
21
- * // Validating fields
22
- * NgxFormService.validateFields(form);
23
- *
24
- * // Getting form data
25
- * const formData = NgxFormService.getFormData(form);
26
- *
27
- * @mermaid
28
- * sequenceDiagram
29
- * participant C as Component
30
- * participant NFS as NgxFormService
31
- * participant AF as Angular Forms
32
- * C->>NFS: createFormFromComponents()
33
- * NFS->>AF: new FormGroup()
34
- * NFS->>NFS: addFormControl()
35
- * NFS->>AF: addControl()
36
- * NFS-->>C: Return FormGroup
37
- * C->>NFS: validateFields()
38
- * NFS->>AF: markAsTouched(), markAsDirty(), updateValueAndValidity()
39
- * C->>NFS: getFormData()
40
- * NFS->>AF: Get control values
41
- * NFS-->>C: Return form data
42
- */
43
- export class NgxFormService {
44
- static { this.controls = new WeakMap(); }
45
- static { this.formRegistry = new Map(); }
46
- /**
47
- * @description Adds a form to the registry.
48
- * @summary Registers a FormGroup with a unique identifier. Throws an error if the identifier is already in use.
49
- * @param {string} formId - The unique identifier for the form.
50
- * @param {FormGroup} formGroup - The FormGroup to be registered.
51
- * @throws {Error} If a FormGroup with the given id is already registered.
52
- */
53
- static addRegistry(formId, formGroup) {
54
- if (this.formRegistry.has(formId))
55
- throw new Error(`A FormGroup with id '${formId}' is already registered.`);
56
- this.formRegistry.set(formId, formGroup);
57
- }
58
- /**
59
- * @description Removes a form from the registry.
60
- * @summary Deletes a FormGroup from the registry using its unique identifier.
61
- * @param {string} formId - The unique identifier of the form to be removed.
62
- */
63
- static removeRegistry(formId) {
64
- this.formRegistry.delete(formId);
65
- }
66
- /**
67
- * @description Resolves the parent group and control name from a path.
68
- * @summary Traverses the form group structure to find the parent group and control name for a given path.
69
- * @param {FormGroup} formGroup - The root FormGroup.
70
- * @param {string} path - The path to the control.
71
- * @return {FormParentGroup} A tuple containing the parent FormGroup and the control name.
72
- */
73
- static resolveParentGroup(formGroup, path) {
74
- const parts = path.split('.');
75
- const controlName = parts.pop();
76
- let currentGroup = formGroup;
77
- for (const part of parts) {
78
- if (!currentGroup.get(part)) {
79
- currentGroup.addControl(part, new FormGroup({}));
80
- }
81
- currentGroup = currentGroup.get(part);
82
- }
83
- return [currentGroup, controlName];
84
- }
85
- /**
86
- * @description Adds a form control to a form group.
87
- * @summary Creates and adds a form control to the specified form group based on the provided component properties.
88
- * @param {FormGroup} formGroup - The form group to add the control to.
89
- * @param {ComponentInput} componentProps - The properties of the component to create the control from.
90
- */
91
- static addFormControl(formGroup, componentProps) {
92
- const { name, childOf } = componentProps;
93
- const fullPath = childOf ? `${childOf}.${name}` : name;
94
- const [parentGroup, controlName] = this.resolveParentGroup(formGroup, fullPath);
95
- if (!parentGroup.get(controlName)) {
96
- const control = NgxFormService.fromProps(componentProps, componentProps.updateMode || 'change');
97
- NgxFormService.register(control, componentProps);
98
- parentGroup.addControl(controlName, control);
99
- }
100
- componentProps['formGroup'] = parentGroup;
101
- componentProps['formControl'] = parentGroup.get(controlName);
102
- }
103
- /**
104
- * @description Retrieves a control from a registered form.
105
- * @summary Finds and returns an AbstractControl from a registered form using the form id and optional path.
106
- * @param {string} formId - The unique identifier of the form.
107
- * @param {string} [path] - The path to the control within the form.
108
- * @return {AbstractControl} The requested AbstractControl.
109
- * @throws {Error} If the form is not found in the registry or the control is not found in the form.
110
- */
111
- static getControlFromForm(formId, path) {
112
- const form = this.formRegistry.get(formId);
113
- if (!form)
114
- throw new Error(`Form with id '${formId}' not found in the registry.`);
115
- if (!path)
116
- return form;
117
- const control = form.get(path);
118
- if (!control)
119
- throw new Error(`Control with path '${path}' not found in form '${formId}'.`);
120
- return control;
121
- }
122
- /**
123
- * @description Creates a form from component configurations.
124
- * @summary Generates a FormGroup based on an array of component configurations and optionally registers it.
125
- * @param {string} id - The unique identifier for the form.
126
- * @param {ComponentConfig[]} components - An array of component configurations.
127
- * @param {boolean} [registry=false] - Whether to register the created form.
128
- * @return {FormGroup} The created FormGroup.
129
- */
130
- static createFormFromComponents(id, components, registry = false) {
131
- const form = new FormGroup({});
132
- components.forEach(component => {
133
- this.addFormControl(form, component.inputs);
134
- });
135
- if (registry)
136
- this.addRegistry(id, form);
137
- return form;
138
- }
139
- /**
140
- * @description Adds a control to a form based on component properties.
141
- * @summary Creates and adds a form control to a form (existing or new) based on the provided component properties.
142
- * @param {string} id - The unique identifier of the form.
143
- * @param {FieldProperties} componentProperties - The properties of the component to create the control from.
144
- * @return {AbstractControl} The form or created control.
145
- */
146
- static addControlFromProps(id, componentProperties) {
147
- const form = this.formRegistry.get(id) ?? new FormGroup({});
148
- if (!this.formRegistry.has(id))
149
- this.addRegistry(id, form);
150
- if (componentProperties.path)
151
- this.addFormControl(form, componentProperties);
152
- return form;
153
- }
154
- /**
155
- * @description Retrieves form data from a FormGroup.
156
- * @summary Extracts and processes the data from a FormGroup, handling different input types and nested form groups.
157
- * @param {FormGroup} formGroup - The FormGroup to extract data from.
158
- * @return {Record<string, unknown>} An object containing the form data.
159
- */
160
- static getFormData(formGroup) {
161
- const data = {};
162
- for (const key in formGroup.controls) {
163
- const control = formGroup.controls[key];
164
- if (!(control instanceof FormControl)) {
165
- data[key] = NgxFormService.getFormData(control);
166
- continue;
167
- }
168
- const props = NgxFormService.getPropsFromControl(control);
169
- let value = control.value;
170
- if (!HTML5CheckTypes.includes(props['type'])) {
171
- switch (props['type']) {
172
- case HTML5InputTypes.NUMBER:
173
- value = parseToNumber(value);
174
- break;
175
- case HTML5InputTypes.DATE:
176
- case HTML5InputTypes.DATETIME_LOCAL:
177
- value = new Date(value);
178
- break;
179
- default:
180
- value = escapeHtml(value);
181
- }
182
- }
183
- data[key] = value;
184
- }
185
- return data;
186
- }
187
- /**
188
- * @description Validates fields in a form control or form group.
189
- * @summary Recursively validates all fields in a form control or form group, marking them as touched and dirty.
190
- * @param {AbstractControl} control - The control or form group to validate.
191
- * @param {string} [path] - The path to the control within the form.
192
- * @return {boolean} True if all fields are valid, false otherwise.
193
- * @throws {Error} If no control is found at the specified path or if the control type is unknown.
194
- */
195
- static validateFields(control, path) {
196
- control = path ? control.get(path) : control;
197
- if (!control)
198
- throw new Error(`No control found at path: ${path || 'root'}.`);
199
- const isAllowed = [FormGroup, FormControl].some(type => control instanceof type);
200
- if (!isAllowed)
201
- throw new Error(`Unknown control type at: ${path || 'root'}`);
202
- control.markAsTouched();
203
- control.markAsDirty();
204
- control.updateValueAndValidity({ emitEvent: true });
205
- if (control instanceof FormGroup) {
206
- Object.values(control.controls).forEach((childControl) => {
207
- this.validateFields(childControl);
208
- });
209
- }
210
- return control.valid;
211
- }
212
- /**
213
- * @description Generates validators from component properties.
214
- * @summary Creates an array of ValidatorFn based on the supported validation keys in the component properties.
215
- * @param {FieldProperties} props - The component properties.
216
- * @return {ValidatorFn[]} An array of validator functions.
217
- */
218
- static validatorsFromProps(props) {
219
- const supportedValidationKeys = Validation.keys();
220
- return Object.keys(props)
221
- .filter((k) => supportedValidationKeys.includes(k))
222
- .map((k) => {
223
- return ValidatorFactory.spawn(props, k);
224
- });
225
- }
226
- /**
227
- * @description Creates a FormControl from component properties.
228
- * @summary Generates a FormControl with validators based on the provided component properties.
229
- * @param {FieldProperties} props - The component properties.
230
- * @param {FieldUpdateMode} [updateMode='change'] - The update mode for the control.
231
- * @return {FormControl} The created FormControl.
232
- */
233
- static fromProps(props, updateMode = 'change') {
234
- const validators = this.validatorsFromProps(props);
235
- const composed = validators.length ? Validators.compose(validators) : null;
236
- return new FormControl({
237
- value: props.value && props.type !== HTML5InputTypes.CHECKBOX
238
- ? props.type === HTML5InputTypes.DATE
239
- ? !isValidDate(parseDate(props.format, props.value))
240
- ? undefined : props.value :
241
- props.value : undefined,
242
- disabled: props.disabled,
243
- }, {
244
- validators: composed,
245
- updateOn: updateMode,
246
- });
247
- }
248
- /**
249
- * @description Retrieves properties from a FormControl.
250
- * @summary Gets the FieldProperties associated with a FormControl from the internal WeakMap.
251
- * @param {FormControl} control - The FormControl to get properties for.
252
- * @return {FieldProperties} The properties associated with the control.
253
- */
254
- static getPropsFromControl(control) {
255
- return this.controls.get(control) || {};
256
- }
257
- /**
258
- * @description Finds a parent element with a specific tag.
259
- * @summary Traverses up the DOM tree to find the nearest parent element with the specified tag.
260
- * @param {HTMLElement} el - The starting element.
261
- * @param {string} tag - The tag name to search for.
262
- * @return {HTMLElement} The found parent element.
263
- * @throws {Error} If no parent with the specified tag is found.
264
- */
265
- static getParentEl(el, tag) {
266
- let parent;
267
- while ((parent = el.parentElement) !== null) {
268
- if (parent.tagName.toLowerCase() === tag.toLowerCase()) {
269
- return parent;
270
- }
271
- el = parent;
272
- }
273
- throw new Error(`No parent with the tag ${tag} was found for provided element`);
274
- }
275
- /**
276
- * @description Registers a control with its properties.
277
- * @summary Associates a control with its properties in the internal WeakMap.
278
- * @param {AbstractControl} control - The control to register.
279
- * @param {FieldProperties} props - The properties to associate with the control.
280
- */
281
- static register(control, props) {
282
- this.controls.set(control, props);
283
- }
284
- /**
285
- * @description Unregisters a control.
286
- * @summary Removes a control and its associated properties from the internal WeakMap.
287
- * @param {AbstractControl} control - The control to unregister.
288
- * @return {boolean} True if the control was successfully unregistered, false otherwise.
289
- */
290
- static unregister(control) {
291
- return this.controls.delete(control);
292
- }
293
- /**
294
- * @description Resets a form group.
295
- * @summary Recursively resets all controls in a form group, clearing values, errors, and marking them as pristine and untouched.
296
- * @param {FormGroup} formGroup - The form group to reset.
297
- */
298
- static reset(formGroup) {
299
- for (const key in formGroup.controls) {
300
- const control = formGroup.controls[key];
301
- if (!(control instanceof FormControl)) {
302
- NgxFormService.reset(control);
303
- continue;
304
- }
305
- const { type } = NgxFormService.getPropsFromControl(control);
306
- if (!HTML5CheckTypes.includes(type))
307
- control.setValue(undefined);
308
- control.markAsPristine();
309
- control.markAsUntouched();
310
- control.setErrors(null);
311
- control.updateValueAndValidity();
312
- }
313
- }
314
- }
315
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiTmd4Rm9ybVNlcnZpY2UuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9zcmMvbGliL2VuZ2luZS9OZ3hGb3JtU2VydmljZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsVUFBVSxFQUFtQixlQUFlLEVBQUUsZUFBZSxFQUFFLGFBQWEsRUFBRSxNQUFNLHlCQUF5QixDQUFDO0FBRXZILE9BQU8sRUFBbUIsV0FBVyxFQUFFLFNBQVMsRUFBZSxVQUFVLEVBQUUsTUFBTSxnQkFBZ0IsQ0FBQztBQUNsRyxPQUFPLEVBQUUsV0FBVyxFQUFFLFNBQVMsRUFBRSxVQUFVLEVBQUUsTUFBTSxnQ0FBZ0MsQ0FBQztBQUNwRixPQUFPLEVBQUUsZ0JBQWdCLEVBQUUsTUFBTSxvQkFBb0IsQ0FBQztBQUd0RDs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7OztHQXFDRztBQUNILE1BQU0sT0FBTyxjQUFjO2FBQ1YsYUFBUSxHQUFHLElBQUksT0FBTyxFQUFvQyxDQUFDO2FBQzNELGlCQUFZLEdBQUcsSUFBSSxHQUFHLEVBQXFCLENBQUM7SUFFM0Q7Ozs7OztPQU1HO0lBQ0gsTUFBTSxDQUFDLFdBQVcsQ0FBQyxNQUFjLEVBQUUsU0FBb0I7UUFDckQsSUFBSSxJQUFJLENBQUMsWUFBWSxDQUFDLEdBQUcsQ0FBQyxNQUFNLENBQUM7WUFDL0IsTUFBTSxJQUFJLEtBQUssQ0FBQyx3QkFBd0IsTUFBTSwwQkFBMEIsQ0FBQyxDQUFDO1FBQzVFLElBQUksQ0FBQyxZQUFZLENBQUMsR0FBRyxDQUFDLE1BQU0sRUFBRSxTQUFTLENBQUMsQ0FBQztJQUMzQyxDQUFDO0lBRUQ7Ozs7T0FJRztJQUNILE1BQU0sQ0FBQyxjQUFjLENBQUMsTUFBYztRQUNsQyxJQUFJLENBQUMsWUFBWSxDQUFDLE1BQU0sQ0FBQyxNQUFNLENBQUMsQ0FBQztJQUNuQyxDQUFDO0lBRUQ7Ozs7OztPQU1HO0lBQ0ssTUFBTSxDQUFDLGtCQUFr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Ozs7T0FLRztJQUNILE1BQU0sQ0FBQyxRQUFRLENBQUMsT0FBd0IsRUFBRSxLQUFzQjtRQUM5RCxJQUFJLENBQUMsUUFBUSxDQUFDLEdBQUcsQ0FBQyxPQUFPLEVBQUUsS0FBSyxDQUFDLENBQUM7SUFDcEMsQ0FBQztJQUVEOzs7OztPQUtHO0lBQ0gsTUFBTSxDQUFDLFVBQVUsQ0FBQyxPQUF3QjtRQUN4QyxPQUFPLElBQUksQ0FBQyxRQUFRLENBQUMsTUFBTSxDQUFDLE9BQU8sQ0FBQyxDQUFDO0lBQ3ZDLENBQUM7SUFFRDs7OztPQUlHO0lBQ0gsTUFBTSxDQUFDLEtBQUssQ0FBQyxTQUFvQjtRQUMvQixLQUFLLE1BQU0sR0FBRyxJQUFJLFNBQVMsQ0FBQyxRQUFRLEVBQUUsQ0FBQztZQUNyQyxNQUFNLE9BQU8sR0FBRyxTQUFTLENBQUMsUUFBUSxDQUFDLEdBQUcsQ0FBQyxDQUFDO1lBQ3hDLElBQUksQ0FBQyxDQUFDLE9BQU8sWUFBWSxXQUFXLENBQUMsRUFBRSxDQUFDO2dCQUN0QyxjQUFjLENBQUMsS0FBSyxDQUFDLE9BQW9CLENBQUMsQ0FBQztnQkFDM0MsU0FBUztZQUNYLENBQUM7WUFFRCxNQUFNLEVBQUUsSUFBSSxFQUFFLEdBQUcsY0FBYyxDQUFDLG1CQUFtQixDQUFDLE9BQU8sQ0FBQyxDQUFDO1lBQzdELElBQUksQ0FBQyxlQUFlLENBQUMsUUFBUSxDQUFDLElBQUksQ0FBQztnQkFDakMsT0FBTyxDQUFDLFFBQVEsQ0FBQyxTQUFTLENBQUMsQ0FBQztZQUM5QixPQUFPLENBQUMsY0FBYyxFQUFFLENBQUM7WUFDekIsT0FBTyxDQUFDLGVBQWUsRUFBRSxDQUFDO1lBQzFCLE9BQU8sQ0FBQyxTQUFTLENBQUMsSUFBSSxDQUFDLENBQUM7WUFDeEIsT0FBTyxDQUFDLHNCQUFzQixFQUFFLENBQUM7UUFDbkMsQ0FBQztJQUNILENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBlc2NhcGVIdG1sLCBGaWVsZFByb3BlcnRpZXMsIEhUTUw1Q2hlY2tUeXBlcywgSFRNTDVJbnB1dFR5cGVzLCBwYXJzZVRvTnVtYmVyIH0gZnJvbSAnQGRlY2FmLXRzL3VpLWRlY29yYXRvcnMnO1xuaW1wb3J0IHsgIENvbXBvbmVudENvbmZpZywgQ29tcG9uZW50SW5wdXQsIEZpZWxkVXBkYXRlTW9kZSwgRm9ybVBhcmVudEdyb3VwIH0gZnJvbSAnLi90eXBlcyc7XG5pbXBvcnQgeyBBYnN0cmFjdENvbnRyb2wsIEZvcm1Db250cm9sLCBGb3JtR3JvdXAsIFZhbGlkYXRvckZuLCBWYWxpZGF0b3JzIH0gZnJvbSAnQGFuZ3VsYXIvZm9ybXMnO1xuaW1wb3J0IHsgaXNWYWxpZERhdGUsIHBhcnNlRGF0ZSwgVmFsaWRhdGlvbiB9IGZyb20gJ0BkZWNhZi10cy9kZWNvcmF0b3ItdmFsaWRhdGlvbic7XG5pbXBvcnQgeyBWYWxpZGF0b3JGYWN0b3J5IH0gZnJvbSAnLi9WYWxpZGF0b3JGYWN0b3J5JztcblxuXG4vKipcbiAqIEBkZXNjcmlwdGlvbiBTZXJ2aWNlIGZvciBtYW5hZ2luZyBBbmd1bGFyIGZvcm1zIGFuZCBmb3JtIGNvbnRyb2xzLlxuICogQHN1bW1hcnkgVGhlIE5neEZvcm1TZXJ2aWNlIHByb3ZpZGVzIHV0aWxpdHkgbWV0aG9kcyBmb3IgY3JlYXRpbmcsIG1hbmFnaW5nLCBhbmQgdmFsaWRhdGluZyBBbmd1bGFyIGZvcm1zIGFuZCBmb3JtIGNvbnRyb2xzLiBJdCBpbmNsdWRlcyBmdW5jdGlvbmFsaXR5IGZvciByZWdpc3RlcmluZyBmb3JtcywgYWRkaW5nIGNvbnRyb2xzLCB2YWxpZGF0aW5nIGZpZWxkcywgYW5kIGhhbmRsaW5nIGZvcm0gZGF0YS5cbiAqXG4gKiBAY2xhc3NcbiAqIEBwYXJhbSB7V2Vha01hcDxBYnN0cmFjdENvbnRyb2wsIEZpZWxkUHJvcGVydGllcz59IGNvbnRyb2xzIC0gQSBXZWFrTWFwIHRvIHN0b3JlIGNvbnRyb2wgcHJvcGVydGllcy5cbiAqIEBwYXJhbSB7TWFwPHN0cmluZywgRm9ybUdyb3VwPn0gZm9ybVJlZ2lzdHJ5IC0gQSBNYXAgdG8gc3RvcmUgcmVnaXN0ZXJlZCBmb3Jtcy5cbiAqXG4gKiBAZXhhbXBsZVxuICogLy8gQ3JlYXRpbmcgYSBmb3JtIGZyb20gY29tcG9uZW50c1xuICogY29uc3QgY29tcG9uZW50cyA9IFtcbiAqICAgeyBpbnB1dHM6IHsgbmFtZTogJ3VzZXJuYW1lJywgdHlwZTogJ3RleHQnLCByZXF1aXJlZDogdHJ1ZSB9IH0sXG4gKiAgIHsgaW5wdXRzOiB7IG5hbWU6ICdwYXNzd29yZCcsIHR5cGU6ICdwYXNzd29yZCcsIG1pbkxlbmd0aDogOCB9IH1cbiAqIF07XG4gKiBjb25zdCBmb3JtID0gTmd4Rm9ybVNlcnZpY2UuY3JlYXRlRm9ybUZyb21Db21wb25lbnRzKCdsb2dpbkZvcm0nLCBjb21wb25lbnRzLCB0cnVlKTtcbiAqXG4gKiAvLyBWYWxpZGF0aW5nIGZpZWxkc1xuICogTmd4Rm9ybVNlcnZpY2UudmFsaWRhdGVGaWVsZHMoZm9ybSk7XG4gKlxuICogLy8gR2V0dGluZyBmb3JtIGRhdGFcbiAqIGNvbnN0IGZvcm1EYXRhID0gTmd4Rm9ybVNlcnZpY2UuZ2V0Rm9ybURhdGEoZm9ybSk7XG4gKlxuICogQG1lcm1haWRcbiAqIHNlcXVlbmNlRGlhZ3JhbVxuICogICBwYXJ0aWNpcGFudCBDIGFzIENvbXBvbmVudFxuICogICBwYXJ0aWNpcGFudCBORlMgYXMgTmd4Rm9ybVNlcnZpY2VcbiAqICAgcGFydGljaXBhbnQgQUYgYXMgQW5ndWxhciBGb3Jtc1xuICogICBDLT4+TkZTOiBjcmVhdGVGb3JtRnJvbUNvbXBvbmVudHMoKVxuICogICBORlMtPj5BRjogbmV3IEZvcm1Hcm91cCgpXG4gKiAgIE5GUy0+Pk5GUzogYWRkRm9ybUNvbnRyb2woKVxuICogICBORlMtPj5BRjogYWRkQ29udHJvbCgpXG4gKiAgIE5GUy0tPj5DOiBSZXR1cm4gRm9ybUdyb3VwXG4gKiAgIEMtPj5ORlM6IHZhbGlkYXRlRmllbGRzKClcbiAqICAgTkZTLT4+QUY6IG1hcmtBc1RvdWNoZWQoKSwgbWFya0FzRGlydHkoKSwgdXBkYXRlVmFsdWVBbmRWYWxpZGl0eSgpXG4gKiAgIEMtPj5ORlM6IGdldEZvcm1EYXRhKClcbiAqICAgTkZTLT4+QUY6IEdldCBjb250cm9sIHZhbHVlc1xuICogICBORlMtLT4+QzogUmV0dXJuIGZvcm0gZGF0YVxuICovXG5leHBvcnQgY2xhc3MgTmd4Rm9ybVNlcnZpY2Uge1xuICBwcml2YXRlIHN0YXRpYyBjb250cm9scyA9IG5ldyBXZWFrTWFwPEFic3RyYWN0Q29udHJvbCwgRmllbGRQcm9wZXJ0aWVzPigpO1xuICBwcml2YXRlIHN0YXRpYyBmb3JtUmVnaXN0cnkgPSBuZXcgTWFwPHN0cmluZywgRm9ybUdyb3VwPigpO1xuXG4gIC8qKlxuICAgKiBAZGVzY3JpcHRpb24gQWRkcyBhIGZvcm0gdG8gdGhlIHJlZ2lzdHJ5LlxuICAgKiBAc3VtbWFyeSBSZWdpc3RlcnMgYSBGb3JtR3JvdXAgd2l0aCBhIHVuaXF1ZSBpZGVudGlmaWVyLiBUaHJvd3MgYW4gZXJyb3IgaWYgdGhlIGlkZW50aWZpZXIgaXMgYWxyZWFkeSBpbiB1c2UuXG4gICAqIEBwYXJhbSB7c3RyaW5nfSBmb3JtSWQgLSBUaGUgdW5pcXVlIGlkZW50aWZpZXIgZm9yIHRoZSBmb3JtLlxuICAgKiBAcGFyYW0ge0Zvcm1Hcm91cH0gZm9ybUdyb3VwIC0gVGhlIEZvcm1Hcm91cCB0byBiZSByZWdpc3RlcmVkLlxuICAgKiBAdGhyb3dzIHtFcnJvcn0gSWYgYSBGb3JtR3JvdXAgd2l0aCB0aGUgZ2l2ZW4gaWQgaXMgYWxyZWFkeSByZWdpc3RlcmVkLlxuICAgKi9cbiAgc3RhdGljIGFkZFJlZ2lzdHJ5KGZvcm1JZDogc3RyaW5nLCBmb3JtR3JvdXA6IEZvcm1Hcm91cCk6IHZvaWQge1xuICAgIGlmICh0aGlzLmZvcm1SZWdpc3RyeS5oYXMoZm9ybUlkKSlcbiAgICAgIHRocm93IG5ldyBFcnJvcihgQSBGb3JtR3JvdXAgd2l0aCBpZCAnJHtmb3JtSWR9JyBpcyBhbHJlYWR5IHJlZ2lzdGVyZWQuYCk7XG4gICAgdGhpcy5mb3JtUmVnaXN0cnkuc2V0KGZvcm1JZCwgZm9ybUdyb3VwKTtcbiAgfVxuXG4gIC8qKlxuICAgKiBAZGVzY3JpcHRpb24gUmVtb3ZlcyBhIGZvcm0gZnJvbSB0aGUgcmVnaXN0cnkuXG4gICAqIEBzdW1tYXJ5IERlbGV0ZXMgYSBGb3JtR3JvdXAgZnJvbSB0aGUgcmVnaXN0cnkgdXNpbmcgaXRzIHVuaXF1ZSBpZGVudGlmaWVyLlxuICAgKiBAcGFyYW0ge3N0cmluZ30gZm9ybUlkIC0gVGhlIHVuaXF1ZSBpZGVudGlmaWVyIG9mIHRoZSBmb3JtIHRvIGJlIHJlbW92ZWQuXG4gICAqL1xuICBzdGF0aWMgcmVtb3ZlUmVnaXN0cnkoZm9ybUlkOiBzdHJpbmcpOiB2b2lkIHtcbiAgICB0aGlzLmZvcm1SZWdpc3RyeS5kZWxldGUoZm9ybUlkKTtcbiAgfVxuXG4gIC8qKlxuICAgKiBAZGVzY3JpcHRpb24gUmVzb2x2ZXMgdGhlIHBhcmVudCBncm91cCBhbmQgY29udHJvbCBuYW1lIGZyb20gYSBwYXRoLlxuICAgKiBAc3VtbWFyeSBUcmF2ZXJzZXMgdGhlIGZvcm0gZ3JvdXAgc3RydWN0dXJlIHRvIGZpbmQgdGhlIHBhcmVudCBncm91cCBhbmQgY29udHJvbCBuYW1lIGZvciBhIGdpdmVuIHBhdGguXG4gICAqIEBwYXJhbSB7Rm9ybUdyb3VwfSBmb3JtR3JvdXAgLSBUaGUgcm9vdCBGb3JtR3JvdXAuXG4gICAqIEBwYXJhbSB7c3RyaW5nfSBwYXRoIC0gVGhlIHBhdGggdG8gdGhlIGNvbnRyb2wuXG4gICAqIEByZXR1cm4ge0Zvcm1QYXJlbnRHcm91cH0gQSB0dXBsZSBjb250YWluaW5nIHRoZSBwYXJlbnQgRm9ybUdyb3VwIGFuZCB0aGUgY29udHJvbCBuYW1lLlxuICAgKi9cbiAgcHJpdmF0ZSBzdGF0aWMgcmVzb2x2ZVBhcmVudEdyb3VwKGZvcm1Hcm91cDogRm9ybUdyb3VwLCBwYXRoOiBzdHJpbmcpOiBGb3JtUGFyZW50R3JvdXAge1xuICAgIGNvbnN0IHBhcnRzID0gcGF0aC5zcGxpdCgnLicpO1xuICAgIGNvbnN0IGNvbnRyb2xOYW1lID0gcGFydHMucG9wKCkgYXMgc3RyaW5nO1xuICAgIGxldCBjdXJyZW50R3JvdXAgPSBmb3JtR3JvdXA7XG4gICAgZm9yIChjb25zdCBwYXJ0IG9mIHBhcnRzKSB7XG4gICAgICBpZiAoIWN1cnJlbnRHcm91cC5nZXQocGFydCkpIHtcbiAgICAgICAgY3VycmVudEdyb3VwLmFkZENvbnRyb2wocGFydCwgbmV3IEZvcm1Hcm91cCh7fSkpO1xuICAgICAgfVxuICAgICAgY3VycmVudEdyb3VwID0gY3VycmVudEdyb3VwLmdldChwYXJ0KSBhcyBGb3JtR3JvdXA7XG4gICAgfVxuICAgIHJldHVybiBbY3VycmVudEdyb3VwLCBjb250cm9sTmFtZV07XG4gIH1cblxuICAvKipcbiAgICogQGRlc2NyaXB0aW9uIEFkZHMgYSBmb3JtIGNvbnRyb2wgdG8gYSBmb3JtIGdyb3VwLlxuICAgKiBAc3VtbWFyeSBDcmVhdGVzIGFuZCBhZGRzIGEgZm9ybSBjb250cm9sIHRvIHRoZSBzcGVjaWZpZWQgZm9ybSBncm91cCBiYXNlZCBvbiB0aGUgcHJvdmlkZWQgY29tcG9uZW50IHByb3BlcnRpZXMuXG4gICAqIEBwYXJhbSB7Rm9ybUdyb3VwfSBmb3JtR3JvdXAgLSBUaGUgZm9ybSBncm91cCB0byBhZGQgdGhlIGNvbnRyb2wgdG8uXG4gICAqIEBwYXJhbSB7Q29tcG9uZW50SW5wdXR9IGNvbXBvbmVudFByb3BzIC0gVGhlIHByb3BlcnRpZXMgb2YgdGhlIGNvbXBvbmVudCB0byBjcmVhdGUgdGhlIGNvbnRyb2wgZnJvbS5cbiAgICovXG4gIHByaXZhdGUgc3RhdGljIGFkZEZvcm1Db250cm9sKGZvcm1Hcm91cDogRm9ybUdyb3VwLCBjb21wb25lbnRQcm9wczogQ29tcG9uZW50SW5wdXQpOiB2b2lkIHtcblxuICAgIGNvbnN0IHsgbmFtZSwgY2hpbGRPZiB9ID0gY29tcG9uZW50UHJvcHM7XG4gICAgY29uc3QgZnVsbFBhdGggPSBjaGlsZE9mID8gYCR7Y2hpbGRPZn0uJHtuYW1lfWAgOiBuYW1lO1xuICAgIGNvbnN0IFtwYXJlbnRHcm91cCwgY29udHJvbE5hbWVdID0gdGhpcy5yZXNvbHZlUGFyZW50R3JvdXAoZm9ybUdyb3VwLCBmdWxsUGF0aCk7XG5cbiAgICBpZiAoIXBhcmVudEdyb3VwLmdldChjb250cm9sTmFtZSkpIHtcbiAgICAgIGNvbnN0IGNvbnRyb2wgPSBOZ3hGb3JtU2VydmljZS5mcm9tUHJvcHMoXG4gICAgICAgIGNvbXBvbmVudFByb3BzLFxuICAgICAgICBjb21wb25lbnRQcm9wcy51cGRhdGVNb2RlIHx8ICdjaGFuZ2UnLFxuICAgICAgKTtcbiAgICAgIE5neEZvcm1TZXJ2aWNlLnJlZ2lzdGVyKGNvbnRyb2wsIGNvbXBvbmVudFByb3BzKTtcbiAgICAgIHBhcmVudEdyb3VwLmFkZENvbnRyb2woY29udHJvbE5hbWUsIGNvbnRyb2wpO1xuICAgIH1cblxuICAgIGNvbXBvbmVudFByb3BzWydmb3JtR3JvdXAnXSA9IHBhcmVudEdyb3VwO1xuICAgIGNvbXBvbmVudFByb3BzWydmb3JtQ29udHJvbCddID0gcGFyZW50R3JvdXAuZ2V0KGNvbnRyb2xOYW1lKSBhcyBGb3JtQ29udHJvbDtcbiAgfVxuXG4gIC8qKlxuICAgKiBAZGVzY3JpcHRpb24gUmV0cmlldmVzIGEgY29udHJvbCBmcm9tIGEgcmVnaXN0ZXJlZCBmb3JtLlxuICAgKiBAc3VtbWFyeSBGaW5kcyBhbmQgcmV0dXJucyBhbiBBYnN0cmFjdENvbnRyb2wgZnJvbSBhIHJlZ2lzdGVyZWQgZm9ybSB1c2luZyB0aGUgZm9ybSBpZCBhbmQgb3B0aW9uYWwgcGF0aC5cbiAgICogQHBhcmFtIHtzdHJpbmd9IGZvcm1JZCAtIFRoZSB1bmlxdWUgaWRlbnRpZmllciBvZiB0aGUgZm9ybS5cbiAgICogQHBhcmFtIHtzdHJpbmd9IFtwYXRoXSAtIFRoZSBwYXRoIHRvIHRoZSBjb250cm9sIHdpdGhpbiB0aGUgZm9ybS5cbiAgICogQHJldHVybiB7QWJzdHJhY3RDb250cm9sfSBUaGUgcmVxdWVzdGVkIEFic3RyYWN0Q29udHJvbC5cbiAgICogQHRocm93cyB7RXJyb3J9IElmIHRoZSBmb3JtIGlzIG5vdCBmb3VuZCBpbiB0aGUgcmVnaXN0cnkgb3IgdGhlIGNvbnRyb2wgaXMgbm90IGZvdW5kIGluIHRoZSBmb3JtLlxuICAgKi9cbiAgc3RhdGljIGdldENvbnRyb2xGcm9tRm9ybShmb3JtSWQ6IHN0cmluZywgcGF0aD86IHN0cmluZyk6IEFic3RyYWN0Q29udHJvbCB7XG4gICAgY29uc3QgZm9ybSA9IHRoaXMuZm9ybVJlZ2lzdHJ5LmdldChmb3JtSWQpO1xuICAgIGlmICghZm9ybSlcbiAgICAgIHRocm93IG5ldyBFcnJvcihgRm9ybSB3aXRoIGlkICcke2Zvcm1JZH0nIG5vdCBmb3VuZCBpbiB0aGUgcmVnaXN0cnkuYCk7XG5cbiAgICBpZiAoIXBhdGgpXG4gICAgICByZXR1cm4gZm9ybTtcblxuICAgIGNvbnN0IGNvbnRyb2wgPSBmb3JtLmdldChwYXRoKTtcbiAgICBpZiAoIWNvbnRyb2wpXG4gICAgICB0aHJvdyBuZXcgRXJyb3IoYENvbnRyb2wgd2l0aCBwYXRoICcke3BhdGh9JyBub3QgZm91bmQgaW4gZm9ybSAnJHtmb3JtSWR9Jy5gKTtcbiAgICByZXR1cm4gY29udHJvbDtcbiAgfVxuXG4gIC8qKlxuICAgKiBAZGVzY3JpcHRpb24gQ3JlYXRlcyBhIGZvcm0gZnJvbSBjb21wb25lbnQgY29uZmlndXJhdGlvbnMuXG4gICAqIEBzdW1tYXJ5IEdlbmVyYXRlcyBhIEZvcm1Hcm91cCBiYXNlZCBvbiBhbiBhcnJheSBvZiBjb21wb25lbnQgY29uZmlndXJhdGlvbnMgYW5kIG9wdGlvbmFsbHkgcmVnaXN0ZXJzIGl0LlxuICAgKiBAcGFyYW0ge3N0cmluZ30gaWQgLSBUaGUgdW5pcXVlIGlkZW50aWZpZXIgZm9yIHRoZSBmb3JtLlxuICAgKiBAcGFyYW0ge0NvbXBvbmVudENvbmZpZ1tdfSBjb21wb25lbnRzIC0gQW4gYXJyYXkgb2YgY29tcG9uZW50IGNvbmZpZ3VyYXRpb25zLlxuICAgKiBAcGFyYW0ge2Jvb2xlYW59IFtyZWdpc3RyeT1mYWxzZV0gLSBXaGV0aGVyIHRvIHJlZ2lzdGVyIHRoZSBjcmVhdGVkIGZvcm0uXG4gICAqIEByZXR1cm4ge0Zvcm1Hcm91cH0gVGhlIGNyZWF0ZWQgRm9ybUdyb3VwLlxuICAgKi9cbiAgc3RhdGljIGNyZWF0ZUZvcm1Gcm9tQ29tcG9uZW50cyhpZDogc3RyaW5nLCBjb21wb25lbnRzOiBDb21wb25lbnRDb25maWdbXSwgcmVnaXN0cnk6IGJvb2xlYW4gPSBmYWxzZSk6IEZvcm1Hcm91cCB7XG4gICAgY29uc3QgZm9ybSA9IG5ldyBGb3JtR3JvdXAoe30pO1xuICAgIGNvbXBvbmVudHMuZm9yRWFjaChjb21wb25lbnQgPT4ge1xuICAgICAgdGhpcy5hZGRGb3JtQ29udHJvbChmb3JtLCBjb21wb25lbnQuaW5wdXRzKTtcbiAgICB9KTtcblxuICAgIGlmIChyZWdpc3RyeSlcbiAgICAgIHRoaXMuYWRkUmVnaXN0cnkoaWQsIGZvcm0pO1xuXG4gICAgcmV0dXJuIGZvcm07XG4gIH1cblxuICAvKipcbiAgICogQGRlc2NyaXB0aW9uIEFkZHMgYSBjb250cm9sIHRvIGEgZm9ybSBiYXNlZCBvbiBjb21wb25lbnQgcHJvcGVydGllcy5cbiAgICogQHN1bW1hcnkgQ3JlYXRlcyBhbmQgYWRkcyBhIGZvcm0gY29udHJvbCB0byBhIGZvcm0gKGV4aXN0aW5nIG9yIG5ldykgYmFzZWQgb24gdGhlIHByb3ZpZGVkIGNvbXBvbmVudCBwcm9wZXJ0aWVzLlxuICAgKiBAcGFyYW0ge3N0cmluZ30gaWQgLSBUaGUgdW5pcXVlIGlkZW50aWZpZXIgb2YgdGhlIGZvcm0uXG4gICAqIEBwYXJhbSB7RmllbGRQcm9wZXJ0aWVzfSBjb21wb25lbnRQcm9wZXJ0aWVzIC0gVGhlIHByb3BlcnRpZXMgb2YgdGhlIGNvbXBvbmVudCB0byBjcmVhdGUgdGhlIGNvbnRyb2wgZnJvbS5cbiAgICogQHJldHVybiB7QWJzdHJhY3RDb250cm9sfSBUaGUgZm9ybSBvciBjcmVhdGVkIGNvbnRyb2wuXG4gICAqL1xuICBzdGF0aWMgYWRkQ29udHJvbEZyb21Qcm9wcyhpZDogc3RyaW5nLCBjb21wb25lbnRQcm9wZXJ0aWVzOiBGaWVsZFByb3BlcnRpZXMpOiBBYnN0cmFjdENvbnRyb2wge1xuICAgIGNvbnN0IGZvcm0gPSB0aGlzLmZvcm1SZWdpc3RyeS5nZXQoaWQpID8/IG5ldyBGb3JtR3JvdXAoe30pO1xuICAgIGlmICghdGhpcy5mb3JtUmVnaXN0cnkuaGFzKGlkKSlcbiAgICAgIHRoaXMuYWRkUmVnaXN0cnkoaWQsIGZvcm0pO1xuXG4gICAgaWYgKGNvbXBvbmVudFByb3BlcnRpZXMucGF0aClcbiAgICAgIHRoaXMuYWRkRm9ybUNvbnRyb2woZm9ybSwgY29tcG9uZW50UHJvcGVydGllcyk7XG5cbiAgICByZXR1cm4gZm9ybTtcbiAgfVxuXG4gIC8qKlxuICAgKiBAZGVzY3JpcHRpb24gUmV0cmlldmVzIGZvcm0gZGF0YSBmcm9tIGEgRm9ybUdyb3VwLlxuICAgKiBAc3VtbWFyeSBFeHRyYWN0cyBhbmQgcHJvY2Vzc2VzIHRoZSBkYXRhIGZyb20gYSBGb3JtR3JvdXAsIGhhbmRsaW5nIGRpZmZlcmVudCBpbnB1dCB0eXBlcyBhbmQgbmVzdGVkIGZvcm0gZ3JvdXBzLlxuICAgKiBAcGFyYW0ge0Zvcm1Hcm91cH0gZm9ybUdyb3VwIC0gVGhlIEZvcm1Hcm91cCB0byBleHRyYWN0IGRhdGEgZnJvbS5cbiAgICogQHJldHVybiB7UmVjb3JkPHN0cmluZywgdW5rbm93bj59IEFuIG9iamVjdCBjb250YWluaW5nIHRoZSBmb3JtIGRhdGEuXG4gICAqL1xuICBzdGF0aWMgZ2V0Rm9ybURhdGEoZm9ybUdyb3VwOiBGb3JtR3JvdXApOiBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPiB7XG4gICAgY29uc3QgZGF0YTogUmVjb3JkPHN0cmluZywgdW5rbm93bj4gPSB7fTtcbiAgICBmb3IgKGNvbnN0IGtleSBpbiBmb3JtR3JvdXAuY29udHJvbHMpIHtcbiAgICAgIGNvbnN0IGNvbnRyb2wgPSBmb3JtR3JvdXAuY29udHJvbHNba2V5XTtcbiAgICAgIGlmICghKGNvbnRyb2wgaW5zdGFuY2VvZiBGb3JtQ29udHJvbCkpIHtcbiAgICAgICAgZGF0YVtrZXldID0gTmd4Rm9ybVNlcnZpY2UuZ2V0Rm9ybURhdGEoY29udHJvbCBhcyBGb3JtR3JvdXApO1xuICAgICAgICBjb250aW51ZTtcbiAgICAgIH1cblxuICAgICAgY29uc3QgcHJvcHMgPSBOZ3hGb3JtU2VydmljZS5nZXRQcm9wc0Zyb21Db250cm9sKGNvbnRyb2wpO1xuICAgICAgbGV0IHZhbHVlID0gY29udHJvbC52YWx1ZTtcbiAgICAgIGlmICghSFRNTDVDaGVja1R5cGVzLmluY2x1ZGVzKHByb3BzWyd0eXBlJ10pKSB7XG4gICAgICAgIHN3aXRjaCAocHJvcHNbJ3R5cGUnXSkge1xuICAgICAgICAgIGNhc2UgSFRNTDVJbnB1dFR5cGVzLk5VTUJFUjpcbiAgICAgICAgICAgIHZhbHVlID0gcGFyc2VUb051bWJlcih2YWx1ZSk7XG4gICAgICAgICAgICBicmVhaztcbiAgICAgICAgICBjYXNlIEhUTUw1SW5wdXRUeXBlcy5EQVRFOlxuICAgICAgICAgIGNhc2UgSFRNTDVJbnB1dFR5cGVzLkRBVEVUSU1FX0xPQ0FMOlxuICAgICAgICAgICAgdmFsdWUgPSBuZXcgRGF0ZSh2YWx1ZSk7XG4gICAgICAgICAgICBicmVhaztcbiAgICAgICAgICBkZWZhdWx0OlxuICAgICAgICAgICAgdmFsdWUgPSBlc2NhcGVIdG1sKHZhbHVlKTtcbiAgICAgICAgfVxuICAgICAgfVxuICAgICAgZGF0YVtrZXldID0gdmFsdWU7XG4gICAgfVxuXG4gICAgcmV0dXJuIGRhdGE7XG4gIH1cblxuICAvKipcbiAgICogQGRlc2NyaXB0aW9uIFZhbGlkYXRlcyBmaWVsZHMgaW4gYSBmb3JtIGNvbnRyb2wgb3IgZm9ybSBncm91cC5cbiAgICogQHN1bW1hcnkgUmVjdXJzaXZlbHkgdmFsaWRhdGVzIGFsbCBmaWVsZHMgaW4gYSBmb3JtIGNvbnRyb2wgb3IgZm9ybSBncm91cCwgbWFya2luZyB0aGVtIGFzIHRvdWNoZWQgYW5kIGRpcnR5LlxuICAgKiBAcGFyYW0ge0Fic3RyYWN0Q29udHJvbH0gY29udHJvbCAtIFRoZSBjb250cm9sIG9yIGZvcm0gZ3JvdXAgdG8gdmFsaWRhdGUuXG4gICAqIEBwYXJhbSB7c3RyaW5nfSBbcGF0aF0gLSBUaGUgcGF0aCB0byB0aGUgY29udHJvbCB3aXRoaW4gdGhlIGZvcm0uXG4gICAqIEByZXR1cm4ge2Jvb2xlYW59IFRydWUgaWYgYWxsIGZpZWxkcyBhcmUgdmFsaWQsIGZhbHNlIG90aGVyd2lzZS5cbiAgICogQHRocm93cyB7RXJyb3J9IElmIG5vIGNvbnRyb2wgaXMgZm91bmQgYXQgdGhlIHNwZWNpZmllZCBwYXRoIG9yIGlmIHRoZSBjb250cm9sIHR5cGUgaXMgdW5rbm93bi5cbiAgICovXG4gIHN0YXRpYyB2YWxpZGF0ZUZpZWxkcyhjb250cm9sOiBBYnN0cmFjdENvbnRyb2wsIHBhdGg/OiBzdHJpbmcpOiBib29sZWFuIHtcbiAgICBjb250cm9sID0gcGF0aCA/IGNvbnRyb2wuZ2V0KHBhdGgpIGFzIEFic3RyYWN0Q29udHJvbCA6IGNvbnRyb2w7XG4gICAgaWYgKCFjb250cm9sKVxuICAgICAgdGhyb3cgbmV3IEVycm9yKGBObyBjb250cm9sIGZvdW5kIGF0IHBhdGg6ICR7cGF0aCB8fCAncm9vdCd9LmApO1xuXG4gICAgY29uc3QgaXNBbGxvd2VkID0gW0Zvcm1Hcm91cCwgRm9ybUNvbnRyb2xdLnNvbWUodHlwZSA9PiBjb250cm9sIGluc3RhbmNlb2YgdHlwZSk7XG4gICAgaWYgKCFpc0FsbG93ZWQpXG4gICAgICB0aHJvdyBuZXcgRXJyb3IoYFVua25vd24gY29udHJvbCB0eXBlIGF0OiAke3BhdGggfHwgJ3Jvb3QnfWApO1xuXG4gICAgY29udHJvbC5tYXJrQXNUb3VjaGVkKCk7XG4gICAgY29udHJvbC5tYXJrQXNEaXJ0eSgpO1xuICAgIGNvbnRyb2wudXBkYXRlVmFsdWVBbmRWYWxpZGl0eSh7IGVtaXRFdmVudDogdHJ1ZSB9KTtcblxuICAgIGlmIChjb250cm9sIGluc3RhbmNlb2YgRm9ybUdyb3VwKSB7XG4gICAgICBPYmplY3QudmFsdWVzKGNvbnRyb2wuY29udHJvbHMpLmZvckVhY2goKGNoaWxkQ29udHJvbCkgPT4ge1xuICAgICAgICB0aGlzLnZhbGlkYXRlRmllbGRzKGNoaWxkQ29udHJvbCk7XG4gICAgICB9KTtcbiAgICB9XG5cbiAgICByZXR1cm4gY29udHJvbC52YWxpZDtcbiAgfVxuXG4gIC8qKlxuICAgKiBAZGVzY3JpcHRpb24gR2VuZXJhdGVzIHZhbGlkYXRvcnMgZnJvbSBjb21wb25lbnQgcHJvcGVydGllcy5cbiAgICogQHN1bW1hcnkgQ3JlYXRlcyBhbiBhcnJheSBvZiBWYWxpZGF0b3JGbiBiYXNlZCBvbiB0aGUgc3VwcG9ydGVkIHZhbGlkYXRpb24ga2V5cyBpbiB0aGUgY29tcG9uZW50IHByb3BlcnRpZXMuXG4gICAqIEBwYXJhbSB7RmllbGRQcm9wZXJ0aWVzfSBwcm9wcyAtIFRoZSBjb21wb25lbnQgcHJvcGVydGllcy5cbiAgICogQHJldHVybiB7VmFsaWRhdG9yRm5bXX0gQW4gYXJyYXkgb2YgdmFsaWRhdG9yIGZ1bmN0aW9ucy5cbiAgICovXG4gIHByaXZhdGUgc3RhdGljIHZhbGlkYXRvcnNGcm9tUHJvcHMocHJvcHM6IEZpZWxkUHJvcGVydGllcyk6IFZhbGlkYXRvckZuW10ge1xuICAgIGNvbnN0IHN1cHBvcnRlZFZhbGlkYXRpb25LZXlzID0gVmFsaWRhdGlvbi5rZXlzKCk7XG4gICAgcmV0dXJuIE9iamVjdC5rZXlzKHByb3BzKVxuICAgICAgLmZpbHRlcigoazogc3RyaW5nKSA9PiBzdXBwb3J0ZWRWYWxpZGF0aW9uS2V5cy5pbmNsdWRlcyhrKSlcbiAgICAgIC5tYXAoKGs6IHN0cmluZykgPT4ge1xuICAgICAgICByZXR1cm4gVmFsaWRhdG9yRmFjdG9yeS5zcGF3bihwcm9wcywgayk7XG4gICAgICB9KTtcbiAgfVxuXG4gIC8qKlxuICAgKiBAZGVzY3JpcHRpb24gQ3JlYXRlcyBhIEZvcm1Db250cm9sIGZyb20gY29tcG9uZW50IHByb3BlcnRpZXMuXG4gICAqIEBzdW1tYXJ5IEdlbmVyYXRlcyBhIEZvcm1Db250cm9sIHdpdGggdmFsaWRhdG9ycyBiYXNlZCBvbiB0aGUgcHJvdmlkZWQgY29tcG9uZW50IHByb3BlcnRpZXMuXG4gICAqIEBwYXJhbSB7RmllbGRQcm9wZXJ0aWVzfSBwcm9wcyAtIFRoZSBjb21wb25lbnQgcHJvcGVydGllcy5cbiAgICogQHBhcmFtIHtGaWVsZFVwZGF0ZU1vZGV9IFt1cGRhdGVNb2RlPSdjaGFuZ2UnXSAtIFRoZSB1cGRhdGUgbW9kZSBmb3IgdGhlIGNvbnRyb2wuXG4gICAqIEByZXR1cm4ge0Zvcm1Db250cm9sfSBUaGUgY3JlYXRlZCBGb3JtQ29udHJvbC5cbiAgICovXG4gIHN0YXRpYyBmcm9tUHJvcHMocHJvcHM6IEZpZWxkUHJvcGVydGllcywgdXBkYXRlTW9kZTogRmllbGRVcGRhdGVNb2RlID0gJ2NoYW5nZScpOiBGb3JtQ29udHJvbCB7XG4gICAgY29uc3QgdmFsaWRhdG9ycyA9IHRoaXMudmFsaWRhdG9yc0Zyb21Qcm9wcyhwcm9wcyk7XG4gICAgY29uc3QgY29tcG9zZWQgPSB2YWxpZGF0b3JzLmxlbmd0aCA/IFZhbGlkYXRvcnMuY29tcG9zZSh2YWxpZGF0b3JzKSA6IG51bGw7XG4gICAgcmV0dXJuIG5ldyBGb3JtQ29udHJvbChcbiAgICAgIHtcbiAgICAgICAgdmFsdWU6XG4gICAgICAgICAgcHJvcHMudmFsdWUgJiYgcHJvcHMudHlwZSAhPT0gSFRNTDVJbnB1dFR5cGVzLkNIRUNLQk9YXG4gICAgICAgICAgICA/IHByb3BzLnR5cGUgPT09IEhUTUw1SW5wdXRUeXBlcy5EQVRFXG4gICAgICAgICAgICAgID8gIWlzVmFsaWREYXRlKHBhcnNlRGF0ZShwcm9wcy5mb3JtYXQgYXMgc3RyaW5nLCBwcm9wcy52YWx1ZSBhcyBzdHJpbmcpKVxuICAgICAgICAgICAgICAgID8gdW5kZWZpbmVkIDogcHJvcHMudmFsdWUgOlxuICAgICAgICAgICAgICAocHJvcHMudmFsdWUgYXMgdW5rbm93bikgOiB1bmRlZmluZWQsXG4gICAgICAgIGRpc2FibGVkOiBwcm9wcy5kaXNhYmxlZCxcbiAgICAgIH0sXG4gICAgICB7XG4gICAgICAgIHZhbGlkYXRvcnM6IGNvbXBvc2VkLFxuICAgICAgICB1cGRhdGVPbjogdXBkYXRlTW9kZSxcbiAgICAgIH0sXG4gICAgKTtcbiAgfVxuXG4gIC8qKlxuICAgKiBAZGVzY3JpcHRpb24gUmV0cmlldmVzIHByb3BlcnRpZXMgZnJvbSBhIEZvcm1Db250cm9sLlxuICAgKiBAc3VtbWFyeSBHZXRzIHRoZSBGaWVsZFByb3BlcnRpZXMgYXNzb2NpYXRlZCB3aXRoIGEgRm9ybUNvbnRyb2wgZnJvbSB0aGUgaW50ZXJuYWwgV2Vha01hcC5cbiAgICogQHBhcmFtIHtGb3JtQ29udHJvbH0gY29udHJvbCAtIFRoZSBGb3JtQ29udHJvbCB0byBnZXQgcHJvcGVydGllcyBmb3IuXG4gICAqIEByZXR1cm4ge0ZpZWxkUHJvcGVydGllc30gVGhlIHByb3BlcnRpZXMgYXNzb2NpYXRlZCB3aXRoIHRoZSBjb250cm9sLlxuICAgKi9cbiAgc3RhdGljIGdldFByb3BzRnJvbUNvbnRyb2woY29udHJvbDogRm9ybUNvbnRyb2wpOiBGaWVsZFByb3BlcnRpZXMge1xuICAgIHJldHVybiB0aGlzLmNvbnRyb2xzLmdldChjb250cm9sKSB8fCB7fSBhcyBGaWVsZFByb3BlcnRpZXM7XG4gIH1cblxuICAvKipcbiAgICogQGRlc2NyaXB0aW9uIEZpbmRzIGEgcGFyZW50IGVsZW1lbnQgd2l0aCBhIHNwZWNpZmljIHRhZy5cbiAgICogQHN1bW1hcnkgVHJhdmVyc2VzIHVwIHRoZSBET00gdHJlZSB0byBmaW5kIHRoZSBuZWFyZXN0IHBhcmVudCBlbGVtZW50IHdpdGggdGhlIHNwZWNpZmllZCB0YWcuXG4gICAqIEBwYXJhbSB7SFRNTEVsZW1lbnR9IGVsIC0gVGhlIHN0YXJ0aW5nIGVsZW1lbnQuXG4gICAqIEBwYXJhbSB7c3RyaW5nfSB0YWcgLSBUaGUgdGFnIG5hbWUgdG8gc2VhcmNoIGZvci5cbiAgICogQHJldHVybiB7SFRNTEVsZW1lbnR9IFRoZSBmb3VuZCBwYXJlbnQgZWxlbWVudC5cbiAgICogQHRocm93cyB7RXJyb3J9IElmIG5vIHBhcmVudCB3aXRoIHRoZSBzcGVjaWZpZWQgdGFnIGlzIGZvdW5kLlxuICAgKi9cbiAgc3RhdGljIGdldFBhcmVudEVsKGVsOiBIVE1MRWxlbWVudCwgdGFnOiBzdHJpbmcpIHtcbiAgICBsZXQgcGFyZW50OiBIVE1MRWxlbWVudCB8IG51bGw7XG4gICAgd2hpbGUgKChwYXJlbnQgPSBlbC5wYXJlbnRFbGVtZW50KSAhPT0gbnVsbCkge1xuICAgICAgaWYgKHBhcmVudC50YWdOYW1lLnRvTG93ZXJDYXNlKCkgPT09IHRhZy50b0xvd2VyQ2FzZSgpKSB7XG4gICAgICAgIHJldHVybiBwYXJlbnQ7XG4gICAgICB9XG4gICAgICBlbCA9IHBhcmVudDtcbiAgICB9XG4gICAgdGhyb3cgbmV3IEVycm9yKFxuICAgICAgYE5vIHBhcmVudCB3aXRoIHRoZSB0YWcgJHt0YWd9IHdhcyBmb3VuZCBmb3IgcHJvdmlkZWQgZWxlbWVudGAsXG4gICAgKTtcbiAgfVxuXG4gIC8qKlxuICAgKiBAZGVzY3JpcHRpb24gUmVnaXN0ZXJzIGEgY29udHJvbCB3aXRoIGl0cyBwcm9wZXJ0aWVzLlxuICAgKiBAc3VtbWFyeSBBc3NvY2lhdGVzIGEgY29udHJvbCB3aXRoIGl0cyBwcm9wZXJ0aWVzIGluIHRoZSBpbnRlcm5hbCBXZWFrTWFwLlxuICAgKiBAcGFyYW0ge0Fic3RyYWN0Q29udHJvbH0gY29udHJvbCAtIFRoZSBjb250cm9sIHRvIHJlZ2lzdGVyLlxuICAgKiBAcGFyYW0ge0ZpZWxkUHJvcGVydGllc30gcHJvcHMgLSBUaGUgcHJvcGVydGllcyB0byBhc3NvY2lhdGUgd2l0aCB0aGUgY29udHJvbC5cbiAgICovXG4gIHN0YXRpYyByZWdpc3Rlcihjb250cm9sOiBBYnN0cmFjdENvbnRyb2wsIHByb3BzOiBGaWVsZFByb3BlcnRpZXMpIHtcbiAgICB0aGlzLmNvbnRyb2xzLnNldChjb250cm9sLCBwcm9wcyk7XG4gIH1cblxuICAvKipcbiAgICogQGRlc2NyaXB0aW9uIFVucmVnaXN0ZXJzIGEgY29udHJvbC5cbiAgICogQHN1bW1hcnkgUmVtb3ZlcyBhIGNvbnRyb2wgYW5kIGl0cyBhc3NvY2lhdGVkIHByb3BlcnRpZXMgZnJvbSB0aGUgaW50ZXJuYWwgV2Vha01hcC5cbiAgICogQHBhcmFtIHtBYnN0cmFjdENvbnRyb2x9IGNvbnRyb2wgLSBUaGUgY29udHJvbCB0byB1bnJlZ2lzdGVyLlxuICAgKiBAcmV0dXJuIHtib29sZWFufSBUcnVlIGlmIHRoZSBjb250cm9sIHdhcyBzdWNjZXNzZnVsbHkgdW5yZWdpc3RlcmVkLCBmYWxzZSBvdGhlcndpc2UuXG4gICAqL1xuICBzdGF0aWMgdW5yZWdpc3Rlcihjb250cm9sOiBBYnN0cmFjdENvbnRyb2wpOiBib29sZWFuIHtcbiAgICByZXR1cm4gdGhpcy5jb250cm9scy5kZWxldGUoY29udHJvbCk7XG4gIH1cblxuICAvKipcbiAgICogQGRlc2NyaXB0aW9uIFJlc2V0cyBhIGZvcm0gZ3JvdXAuXG4gICAqIEBzdW1tYXJ5IFJlY3Vyc2l2ZWx5IHJlc2V0cyBhbGwgY29udHJvbHMgaW4gYSBmb3JtIGdyb3VwLCBjbGVhcmluZyB2YWx1ZXMsIGVycm9ycywgYW5kIG1hcmtpbmcgdGhlbSBhcyBwcmlzdGluZSBhbmQgdW50b3VjaGVkLlxuICAgKiBAcGFyYW0ge0Zvcm1Hcm91cH0gZm9ybUdyb3VwIC0gVGhlIGZvcm0gZ3JvdXAgdG8gcmVzZXQuXG4gICAqL1xuICBzdGF0aWMgcmVzZXQoZm9ybUdyb3VwOiBGb3JtR3JvdXApIHtcbiAgICBmb3IgKGNvbnN0IGtleSBpbiBmb3JtR3JvdXAuY29udHJvbHMpIHtcbiAgICAgIGNvbnN0IGNvbnRyb2wgPSBmb3JtR3JvdXAuY29udHJvbHNba2V5XTtcbiAgICAgIGlmICghKGNvbnRyb2wgaW5zdGFuY2VvZiBGb3JtQ29udHJvbCkpIHtcbiAgICAgICAgTmd4Rm9ybVNlcnZpY2UucmVzZXQoY29udHJvbCBhcyBGb3JtR3JvdXApO1xuICAgICAgICBjb250aW51ZTtcbiAgICAgIH1cblxuICAgICAgY29uc3QgeyB0eXBlIH0gPSBOZ3hGb3JtU2VydmljZS5nZXRQcm9wc0Zyb21Db250cm9sKGNvbnRyb2wpO1xuICAgICAgaWYgKCFIVE1MNUNoZWNrVHlwZXMuaW5jbHVkZXModHlwZSkpXG4gICAgICAgIGNvbnRyb2wuc2V0VmFsdWUodW5kZWZpbmVkKTtcbiAgICAgIGNvbnRyb2wubWFya0FzUHJpc3RpbmUoKTtcbiAgICAgIGNvbnRyb2wubWFya0FzVW50b3VjaGVkKCk7XG4gICAgICBjb250cm9sLnNldEVycm9ycyhudWxsKTtcbiAgICAgIGNvbnRyb2wudXBkYXRlVmFsdWVBbmRWYWxpZGl0eSgpO1xuICAgIH1cbiAgfVxufVxuIl19
@@ -1,192 +0,0 @@
1
- import { RenderingEngine } from '@decaf-ts/ui-decorators';
2
- import { AngularEngineKeys } from './constants';
3
- import { InternalError } from '@decaf-ts/db-decorators';
4
- import { reflectComponentType, } from '@angular/core';
5
- /**
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.
9
- * @template AngularFieldDefinition - Type for Angular-specific field definitions
10
- * @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
14
- * @class NgxRenderingEngine
15
- * @example
16
- * ```typescript
17
- * const engine = new NgxRenderingEngine();
18
- * engine.initialize();
19
- * const output = engine.render(myModel, {}, viewContainerRef, injector, templateRef);
20
- * ```
21
- * @mermaid
22
- * sequenceDiagram
23
- * participant Client
24
- * participant Engine as NgxRenderingEngine
25
- * participant Components as RegisteredComponents
26
- *
27
- * Client->>Engine: new NgxRenderingEngine()
28
- * Client->>Engine: initialize()
29
- * Client->>Engine: render(model, props, vcr, injector, tpl)
30
- * Engine->>Engine: toFieldDefinition(model, props)
31
- * Engine->>Engine: fromFieldDefinition(fieldDef, vcr, injector, tpl)
32
- * Engine->>Components: components(fieldDef.tag)
33
- * Components-->>Engine: component constructor
34
- * Engine->>Client: return AngularDynamicOutput
35
- */
36
- export class NgxRenderingEngine extends RenderingEngine {
37
- constructor() {
38
- super('angular');
39
- }
40
- /**
41
- * @description Converts a field definition to an Angular component output
42
- * @summary This private method takes a field definition and creates the corresponding Angular component.
43
- * It handles component instantiation, input property mapping, and child component rendering.
44
- * @param {FieldDefinition<AngularFieldDefinition>} fieldDef - The field definition to convert
45
- * @param {ViewContainerRef} vcr - The view container reference for component creation
46
- * @param {Injector} injector - The Angular injector for dependency injection
47
- * @param {TemplateRef<any>} tpl - The template reference for content projection
48
- * @return {AngularDynamicOutput} The Angular component output with component reference and inputs
49
- * @mermaid
50
- * sequenceDiagram
51
- * participant Method as fromFieldDefinition
52
- * participant Components as NgxRenderingEngine.components
53
- * participant Angular as Angular Core
54
- *
55
- * Method->>Components: components(fieldDef.tag)
56
- * Components-->>Method: component constructor
57
- * Method->>Angular: reflectComponentType(component)
58
- * Angular-->>Method: componentMetadata
59
- * Method->>Method: Check input properties
60
- * Method->>Method: Create result object
61
- * Method->>Method: Process children if any
62
- * Method-->>Caller: return AngularDynamicOutput
63
- */
64
- fromFieldDefinition(fieldDef, vcr, injector, tpl) {
65
- const component = NgxRenderingEngine.components(fieldDef.tag)
66
- .constructor;
67
- const componentMetadata = reflectComponentType(component);
68
- if (!componentMetadata) {
69
- throw new InternalError(`Metadata for component ${fieldDef.tag} not found.`);
70
- }
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(', ')}`);
84
- const result = {
85
- component: component,
86
- inputs: inputs || {},
87
- injector: injector,
88
- };
89
- if (fieldDef.rendererId) {
90
- result.inputs['rendererId'] =
91
- fieldDef.rendererId;
92
- }
93
- if (fieldDef.children && fieldDef.children.length) {
94
- result.children = fieldDef.children.map((child) => {
95
- return this.fromFieldDefinition(child, vcr, injector, tpl);
96
- });
97
- const template = vcr.createEmbeddedView(tpl, injector).rootNodes;
98
- result.content = [template];
99
- }
100
- return result;
101
- }
102
- /**
103
- * @description Renders a model into an Angular component output
104
- * @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.
107
- * @template M - Type extending Model
108
- * @param {M} model - The model to render
109
- * @param {Record<string, unknown>} globalProps - Global properties to pass to the component
110
- * @param {ViewContainerRef} vcr - The view container reference for component creation
111
- * @param {Injector} injector - The Angular injector for dependency injection
112
- * @param {TemplateRef<any>} tpl - The template reference for content projection
113
- * @return {AngularDynamicOutput} The Angular component output with component reference and inputs
114
- * @mermaid
115
- * sequenceDiagram
116
- * participant Client as Client Code
117
- * participant Render as render method
118
- * participant ToField as toFieldDefinition
119
- * participant FromField as fromFieldDefinition
120
- *
121
- * Client->>Render: render(model, globalProps, vcr, injector, tpl)
122
- * Render->>ToField: toFieldDefinition(model, globalProps)
123
- * ToField-->>Render: fieldDef
124
- * Render->>FromField: fromFieldDefinition(fieldDef, vcr, injector, tpl)
125
- * FromField-->>Render: AngularDynamicOutput
126
- * Render-->>Client: return AngularDynamicOutput
127
- */
128
- render(model, globalProps, vcr, injector, tpl) {
129
- let result;
130
- try {
131
- const fieldDef = this.toFieldDefinition(model, globalProps);
132
- result = this.fromFieldDefinition(fieldDef, vcr, injector, tpl);
133
- }
134
- catch (e) {
135
- throw new InternalError(`Failed to render Model ${model.constructor.name}: ${e}`);
136
- }
137
- return result;
138
- }
139
- /**
140
- * @description Initializes the rendering engine
141
- * @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.
143
- * @return {Promise<void>} A promise that resolves when initialization is complete
144
- */
145
- async initialize() {
146
- if (this.initialized)
147
- return;
148
- this.initialized = true;
149
- }
150
- /**
151
- * @description Registers a component with the rendering engine
152
- * @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.
154
- * @param {string} name - The name to register the component under
155
- * @param {Constructor<unknown>} constructor - The component constructor
156
- * @return {void}
157
- */
158
- static registerComponent(name, constructor) {
159
- if (!this._components)
160
- this._components = {};
161
- if (name in this._components)
162
- throw new InternalError(`Component already registered under ${name}`);
163
- this._components[name] = {
164
- constructor: constructor,
165
- };
166
- }
167
- /**
168
- * @description Retrieves registered components from the rendering engine
169
- * @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.
171
- * @param {string} [selector] - Optional selector to retrieve a specific component
172
- * @return {Object|Array} Either a specific component or an array of all components
173
- */
174
- static components(selector) {
175
- if (!selector)
176
- return Object.values(this._components);
177
- if (!(selector in this._components))
178
- throw new InternalError(`No Component registered under ${selector}`);
179
- return this._components[selector];
180
- }
181
- /**
182
- * @description Generates a key for reflection metadata
183
- * @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.
185
- * @param {string} key - The base key to prefix
186
- * @return {string} The prefixed key for reflection metadata
187
- */
188
- static key(key) {
189
- return `${AngularEngineKeys.REFLECT}${key}`;
190
- }
191
- }
192
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiTmd4UmVuZGVyaW5nRW5naW5lLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vc3JjL2xpYi9lbmdpbmUvTmd4UmVuZGVyaW5nRW5naW5lLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBbUIsZUFBZSxFQUFFLE1BQU0seUJBQXlCLENBQUM7QUFFM0UsT0FBTyxFQUFFLGlCQUFpQixFQUFFLE1BQU0sYUFBYSxDQUFDO0FBRWhELE9BQU8sRUFBRSxhQUFhLEVBQUUsTUFBTSx5QkFBeUIsQ0FBQztBQUN4RCxPQUFPLEVBRUwsb0JBQW9CLEdBSXJCLE1BQU0sZUFBZSxDQUFDO0FBRXZCOzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7R0E4Qkc7QUFDSCxNQUFNLE9BQU8sa0JBQW1CLFNBQVEsZUFHdkM7SUFNQztRQUNFLEtBQUssQ0FBQyxTQUFTLENBQUMsQ0FBQztJQUNuQixDQUFDO0lBRUQ7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7O09BdUJHO0lBQ0ssbUJBQW1CLENBQ3pCLFFBQWlELEVBQ2pELEdBQXFCLEVBQ3JCLFFBQWtCLEVBQ2xCLEdBQXlCO1FBRXpCLE1BQU0sU0FBUyxHQUFHLGtCQUFrQixDQUFDLFVBQVUsQ0FBQyxRQUFRLENBQUMsR0FBRyxDQUFDO2FBQzFELFdBQXVDLENBQUM7UUFFM0MsTUFBTSxpQkFBaUIsR0FBRyxvQkFBb0IsQ0FBQyxTQUFTLENBQUMsQ0FBQztRQUMxRCxJQUFJLENBQUMsaUJBQWlCLEVBQUUsQ0FBQztZQUN2QixNQUFNLElBQUksYUFBYSxDQUNyQiwwQkFBMEIsUUFBUSxDQUFDLEdBQUcsYUFBYSxDQUNwRCxDQUFDO1FBQ0osQ0FBQztRQUNELE1BQU0sTUFBTSxHQUFHLFFBQVEsQ0FBQyxLQUFLLENBQUM7UUFFOUIsTUFBTSxjQUFjLEdBQUcsaUJBQWlCLENBQUMsTUFBTSxDQUFDO1FBQ2hELE1BQU0sU0FBUyxHQUFHLE1BQU0sQ0FBQyxJQUFJLENBQUMsTUFBTSxD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UUFDeEUsSUFBSSxDQUFDLFdBQVcsQ0FBQyxJQUFJLENBQUMsR0FBRztZQUN2QixXQUFXLEVBQUUsV0FBVztTQUN6QixDQUFDO0lBQ0osQ0FBQztJQUVEOzs7Ozs7T0FNRztJQUNILE1BQU0sQ0FBQyxVQUFVLENBQUMsUUFBaUI7UUFDakMsSUFBSSxDQUFDLFFBQVE7WUFBRSxPQUFPLE1BQU0sQ0FBQyxNQUFNLENBQUMsSUFBSSxDQUFDLFdBQVcsQ0FBQyxDQUFDO1FBQ3RELElBQUksQ0FBQyxDQUFDLFFBQVEsSUFBSSxJQUFJLENBQUMsV0FBVyxDQUFDO1lBQ2pDLE1BQU0sSUFBSSxhQUFhLENBQUMsaUNBQWlDLFFBQVEsRUFBRSxDQUFDLENBQUM7UUFDdkUsT0FBTyxJQUFJLENBQUMsV0FBVyxDQUFDLFFBQVEsQ0FBQyxDQUFDO0lBQ3BDLENBQUM7SUFFRDs7Ozs7O09BTUc7SUFDSCxNQUFNLENBQVUsR0FBRyxDQUFDLEdBQVc7UUFDN0IsT0FBTyxHQUFHLGlCQUFpQixDQUFDLE9BQU8sR0FBRyxHQUFHLEVBQUUsQ0FBQztJQUM5QyxDQUFDO0NBQ0YiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBGaWVsZERlZmluaXRpb24sIFJlbmRlcmluZ0VuZ2luZSB9IGZyb20gJ0BkZWNhZi10cy91aS1kZWNvcmF0b3JzJztcbmltcG9ydCB7IEFuZ3VsYXJEeW5hbWljT3V0cHV0LCBBbmd1bGFyRmllbGREZWZpbml0aW9uIH0gZnJvbSAnLi90eXBlcyc7XG5pbXBvcnQgeyBBbmd1bGFyRW5naW5lS2V5cyB9IGZyb20gJy4vY29uc3RhbnRzJztcbmltcG9ydCB7IENvbnN0cnVjdG9yLCBNb2RlbCB9IGZyb20gJ0BkZWNhZi10cy9kZWNvcmF0b3ItdmFsaWRhdGlvbic7XG5pbXBvcnQgeyBJbnRlcm5hbEVycm9yIH0gZnJvbSAnQGRlY2FmLXRzL2RiLWRlY29yYXRvcnMnO1xuaW1wb3J0IHtcbiAgSW5qZWN0b3IsXG4gIHJlZmxlY3RDb21wb25lbnRUeXBlLFxuICBUZW1wbGF0ZVJlZixcbiAgVHlwZSxcbiAgVmlld0NvbnRhaW5lclJlZixcbn0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5cbi8qKlxuICogQGRlc2NyaXB0aW9uIEFuZ3VsYXIgaW1wbGVtZW50YXRpb24gb2YgdGhlIFJlbmRlcmluZ0VuZ2luZVxuICogQHN1bW1hcnkgVGhpcyBjbGFzcyBleHRlbmRzIHRoZSBiYXNlIFJlbmRlcmluZ0VuZ2luZSB0byBwcm92aWRlIEFuZ3VsYXItc3BlY2lmaWMgcmVuZGVyaW5nIGNhcGFiaWxpdGllcy5cbiAqIEl0IGhhbmRsZXMgdGhlIGNvbnZlcnNpb24gb2YgZmllbGQgZGVmaW5pdGlvbnMgdG8gQW5ndWxhciBjb21wb25lbnRzIGFuZCBtYW5hZ2VzIGNvbXBvbmVudCByZWdpc3RyYXRpb24uXG4gKiBAdGVtcGxhdGUgQW5ndWxhckZpZWxkRGVmaW5pdGlvbiAtIFR5cGUgZm9yIEFuZ3VsYXItc3BlY2lmaWMgZmllbGQgZGVmaW5pdGlvbnNcbiAqIEB0ZW1wbGF0ZSBBbmd1bGFyRHluYW1pY091dHB1dCAtIFR5cGUgZm9yIEFuZ3VsYXItc3BlY2lmaWMgY29tcG9uZW50IG91dHB1dFxuICogQHBhcmFtIHtJbmplY3Rvcn0gaW5qZWN0b3IgLSBBbmd1bGFyIGluamVjdG9yIGZvciBkZXBlbmRlbmN5IGluamVjdGlvblxuICogQHBhcmFtIHtWaWV3Q29udGFpbmVyUmVmfSB2Y3IgLSBWaWV3IGNvbnRhaW5lciByZWZlcmVuY2UgZm9yIGNvbXBvbmVudCBjcmVhdGlvblxuICogQHBhcmFtIHtUZW1wbGF0ZVJlZjxhbnk+fSB0cGwgLSBUZW1wbGF0ZSByZWZlcmVuY2UgZm9yIGNvbnRlbnQgcHJvamVjdGlvblxuICogQGNsYXNzIE5neFJlbmRlcmluZ0VuZ2luZVxuICogQGV4YW1wbGVcbiAqIGBgYHR5cGVzY3JpcHRcbiAqIGNvbnN0IGVuZ2luZSA9IG5ldyBOZ3hSZW5kZXJpbmdFbmdpbmUoKTtcbiAqIGVuZ2luZS5pbml0aWFsaXplKCk7XG4gKiBjb25zdCBvdXRwdXQgPSBlbmdpbmUucmVuZGVyKG15TW9kZWwsIHt9LCB2aWV3Q29udGFpbmVyUmVmLCBpbmplY3RvciwgdGVtcGxhdGVSZWYpO1xuICogYGBgXG4gKiBAbWVybWFpZFxuICogc2VxdWVuY2VEaWFncmFtXG4gKiAgIHBhcnRpY2lwYW50IENsaWVudFxuICogICBwYXJ0aWNpcGFudCBFbmdpbmUgYXMgTmd4UmVuZGVyaW5nRW5naW5lXG4gKiAgIHBhcnRpY2lwYW50IENvbXBvbmVudHMgYXMgUmVnaXN0ZXJlZENvbXBvbmVudHNcbiAqXG4gKiAgIENsaWVudC0+PkVuZ2luZTogbmV3IE5neFJlbmRlcmluZ0VuZ2luZSgpXG4gKiAgIENsaWVudC0+PkVuZ2luZTogaW5pdGlhbGl6ZSgpXG4gKiAgIENsaWVudC0+PkVuZ2luZTogcmVuZGVyKG1vZGVsLCBwcm9wcywgdmNyLCBpbmplY3RvciwgdHBsKVxuICogICBFbmdpbmUtPj5FbmdpbmU6IHRvRmllbGREZWZpbml0aW9uKG1vZGVsLCBwcm9wcylcbiAqICAgRW5naW5lLT4+RW5naW5lOiBmcm9tRmllbGREZWZpbml0aW9uKGZpZWxkRGVmLCB2Y3IsIGluamVjdG9yLCB0cGwpXG4gKiAgIEVuZ2luZS0+PkNvbXBvbmVudHM6IGNvbXBvbmVudHMoZmllbGREZWYudGFnKVxuICogICBDb21wb25lbnRzLS0+PkVuZ2luZTogY29tcG9uZW50IGNvbnN0cnVjdG9yXG4gKiAgIEVuZ2luZS0+PkNsaWVudDogcmV0dXJuIEFuZ3VsYXJEeW5hbWljT3V0cHV0XG4gKi9cbmV4cG9ydCBjbGFzcyBOZ3hSZW5kZXJpbmdFbmdpbmUgZXh0ZW5kcyBSZW5kZXJpbmdFbmdpbmU8XG4gIEFuZ3VsYXJGaWVsZERlZmluaXRpb24sXG4gIEFuZ3VsYXJEeW5hbWljT3V0cHV0XG4+IHtcbiAgcHJpdmF0ZSBzdGF0aWMgX2NvbXBvbmVudHM6IFJlY29yZDxcbiAgICBzdHJpbmcsXG4gICAgeyBjb25zdHJ1Y3RvcjogQ29uc3RydWN0b3I8dW5rbm93bj4gfVxuICA+O1xuXG4gIGNvbnN0cnVjdG9yKCkge1xuICAgIHN1cGVyKCdhbmd1bGFyJyk7XG4gIH1cblxuICAvKipcbiAgICogQGRlc2NyaXB0aW9uIENvbnZlcnRzIGEgZmllbGQgZGVmaW5pdGlvbiB0byBhbiBBbmd1bGFyIGNvbXBvbmVudCBvdXRwdXRcbiAgICogQHN1bW1hcnkgVGhpcyBwcml2YXRlIG1ldGhvZCB0YWtlcyBhIGZpZWxkIGRlZmluaXRpb24gYW5kIGNyZWF0ZXMgdGhlIGNvcnJlc3BvbmRpbmcgQW5ndWxhciBjb21wb25lbnQuXG4gICAqIEl0IGhhbmRsZXMgY29tcG9uZW50IGluc3RhbnRpYXRpb24sIGlucHV0IHByb3BlcnR5IG1hcHBpbmcsIGFuZCBjaGlsZCBjb21wb25lbnQgcmVuZGVyaW5nLlxuICAgKiBAcGFyYW0ge0ZpZWxkRGVmaW5pdGlvbjxBbmd1bGFyRmllbGREZWZpbml0aW9uPn0gZmllbGREZWYgLSBUaGUgZmllbGQgZGVmaW5pdGlvbiB0byBjb252ZXJ0XG4gICAqIEBwYXJhbSB7Vmlld0NvbnRhaW5lclJlZn0gdmNyIC0gVGhlIHZpZXcgY29udGFpbmVyIHJlZmVyZW5jZSBmb3IgY29tcG9uZW50IGNyZWF0aW9uXG4gICAqIEBwYXJhbSB7SW5qZWN0b3J9IGluamVjdG9yIC0gVGhlIEFuZ3VsYXIgaW5qZWN0b3IgZm9yIGRlcGVuZGVuY3kgaW5qZWN0aW9uXG4gICAqIEBwYXJhbSB7VGVtcGxhdGVSZWY8YW55Pn0gdHBsIC0gVGhlIHRlbXBsYXRlIHJlZmVyZW5jZSBmb3IgY29udGVudCBwcm9qZWN0aW9uXG4gICAqIEByZXR1cm4ge0FuZ3VsYXJEeW5hbWljT3V0cHV0fSBUaGUgQW5ndWxhciBjb21wb25lbnQgb3V0cHV0IHdpdGggY29tcG9uZW50IHJlZmVyZW5jZSBhbmQgaW5wdXRzXG4gICAqIEBtZXJtYWlkXG4gICAqIHNlcXVlbmNlRGlhZ3JhbVxuICAgKiAgIHBhcnRpY2lwYW50IE1ldGhvZCBhcyBmcm9tRmllbGREZWZpbml0aW9uXG4gICAqICAgcGFydGljaXBhbnQgQ29tcG9uZW50cyBhcyBOZ3hSZW5kZXJpbmdFbmdpbmUuY29tcG9uZW50c1xuICAgKiAgIHBhcnRpY2lwYW50IEFuZ3VsYXIgYXMgQW5ndWxhciBDb3JlXG4gICAqXG4gICAqICAgTWV0aG9kLT4+Q29tcG9uZW50czogY29tcG9uZW50cyhmaWVsZERlZi50YWcpXG4gICAqICAgQ29tcG9uZW50cy0tPj5NZXRob2Q6IGNvbXBvbmVudCBjb25zdHJ1Y3RvclxuICAgKiAgIE1ldGhvZC0+PkFuZ3VsYXI6IHJlZmxlY3RDb21wb25lbnRUeXBlKGNvbXBvbmVudClcbiAgICogICBBbmd1bGFyLS0+Pk1ldGhvZDogY29tcG9uZW50TWV0YWRhdGFcbiAgICogICBNZXRob2QtPj5NZXRob2Q6IENoZWNrIGlucHV0IHByb3BlcnRpZXNcbiAgICogICBNZXRob2QtPj5NZXRob2Q6IENyZWF0ZSByZXN1bHQgb2JqZWN0XG4gICAqICAgTWV0aG9kLT4+TWV0aG9kOiBQcm9jZXNzIGNoaWxkcmVuIGlmIGFueVxuICAgKiAgIE1ldGhvZC0tPj5DYWxsZXI6IHJldHVybiBBbmd1bGFyRHluYW1pY091dHB1dFxuICAgKi9cbiAgcHJpdmF0ZSBmcm9tRmllbGREZWZpbml0aW9uKFxuICAgIGZpZWxkRGVmOiBGaWVsZERlZmluaXRpb248QW5ndWxhckZpZWxkRGVmaW5pdGlvbj4sXG4gICAgdmNyOiBWaWV3Q29udGFpbmVyUmVmLFxuICAgIGluamVjdG9yOiBJbmplY3RvcixcbiAgICB0cGw6IFRlbXBsYXRlUmVmPHVua25vd24+LFxuICApOiBBbmd1bGFyRHluYW1pY091dHB1dCB7XG4gICAgY29uc3QgY29tcG9uZW50ID0gTmd4UmVuZGVyaW5nRW5naW5lLmNvbXBvbmVudHMoZmllbGREZWYudGFnKVxuICAgICAgLmNvbnN0cnVjdG9yIGFzIHVua25vd24gYXMgVHlwZTx1bmtub3duPjtcblxuICAgIGNvbnN0IGNvbXBvbmVudE1ldGFkYXRhID0gcmVmbGVjdENvbXBvbmVudFR5cGUoY29tcG9uZW50KTtcbiAgICBpZiAoIWNvbXBvbmVudE1ldGFkYXRhKSB7XG4gICAgICB0aHJvdyBuZXcgSW50ZXJuYWxFcnJvcihcbiAgICAgICAgYE1ldGFkYXRhIGZvciBjb21wb25lbnQgJHtmaWVsZERlZi50YWd9IG5vdCBmb3VuZC5gLFxuICAgICAgKTtcbiAgICB9XG4gICAgY29uc3QgaW5wdXRzID0gZmllbGREZWYucHJvcHM7XG5cbiAgICBjb25zdCBwb3NzaWJsZUlucHV0cyA9IGNvbXBvbmVudE1ldGFkYXRhLmlucHV0cztcbiAgICBjb25zdCBpbnB1dEtleXMgPSBPYmplY3Qua2V5cyhpbnB1dHMpO1xuICAgIGZvciAoY29uc3QgaW5wdXQgb2YgcG9zc2libGVJbnB1dHMpIHtcbiAgICAgIGNvbnN0IGluZGV4ID0gaW5wdXRLZXlzLmluZGV4T2YoaW5wdXQucHJvcE5hbWUpO1xuICAgICAgaWYgKGluZGV4ICE9PSAtMSkge1xuICAgICAgICBpbnB1dEtleXMuc3BsaWNlKGluZGV4LCAxKTtcbiAgICAgIH1cbiAgICAgIGlmICghaW5wdXRLZXlzLmxlbmd0aCkgYnJlYWs7XG4gICAgfVxuXG4gICAgaWYgKGlucHV0S2V5cy5sZW5ndGgpXG4gICAgICBjb25zb2xlLndhcm4oXG4gICAgICAgIGBVbm1hcHBlZCBpbnB1dCBwcm9wZXJ0aWVzIGZvciBjb21wb25lbnQgJHtmaWVsZERlZi50YWd9OiAke2lucHV0S2V5cy5qb2luKCcsICcpfWAsXG4gICAgICApO1xuXG4gICAgY29uc3QgcmVzdWx0OiBBbmd1bGFyRHluYW1pY091dHB1dCA9IHtcbiAgICAgIGNvbXBvbmVudDogY29tcG9uZW50LFxuICAgICAgaW5wdXRzOiBpbnB1dHMgfHwge30sXG4gICAgICBpbmplY3RvcjogaW5qZWN0b3IsXG4gICAgfTtcblxuICAgIGlmIChmaWVsZERlZi5yZW5kZXJlcklkKSB7XG4gICAgICAocmVzdWx0LmlucHV0cyBhcyBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPilbJ3JlbmRlcmVySWQnXSA9XG4gICAgICAgIGZpZWxkRGVmLnJlbmRlcmVySWQ7XG4gICAgfVxuXG4gICAgaWYgKGZpZWxkRGVmLmNoaWxkcmVuICYmIGZpZWxkRGVmLmNoaWxkcmVuLmxlbmd0aCkge1xuICAgICAgcmVzdWx0LmNoaWxkcmVuID0gZmllbGREZWYuY2hpbGRyZW4ubWFwKChjaGlsZCkgPT4ge1xuICAgICAgICByZXR1cm4gdGhpcy5mcm9tRmllbGREZWZpbml0aW9uKGNoaWxkLCB2Y3IsIGluamVjdG9yLCB0cGwpO1xuICAgICAgfSk7XG5cbiAgICAgIGNvbnN0IHRlbXBsYXRlID0gdmNyLmNyZWF0ZUVtYmVkZGVkVmlldyh0cGwsIGluamVjdG9yKS5yb290Tm9kZXM7XG4gICAgICByZXN1bHQuY29udGVudCA9IFt0ZW1wbGF0ZV07XG4gICAgfVxuXG4gICAgcmV0dXJuIHJlc3VsdDtcbiAgfVxuXG4gIC8qKlxuICAgKiBAZGVzY3JpcHRpb24gUmVuZGVycyBhIG1vZGVsIGludG8gYW4gQW5ndWxhciBjb21wb25lbnQgb3V0cHV0XG4gICAqIEBzdW1tYXJ5IFRoaXMgbWV0aG9kIHRha2VzIGEgbW9kZWwgYW5kIGNvbnZlcnRzIGl0IHRvIGFuIEFuZ3VsYXIgY29tcG9uZW50IG91dHB1dC5cbiAgICogSXQgZmlyc3QgY29udmVydHMgdGhlIG1vZGVsIHRvIGEgZmllbGQgZGVmaW5pdGlvbiB1c2luZyB0aGUgYmFzZSBSZW5kZXJpbmdFbmdpbmUnc1xuICAgKiB0b0ZpZWxkRGVmaW5pdGlvbiBtZXRob2QsIHRoZW4gY29udmVydHMgdGhhdCBmaWVsZCBkZWZpbml0aW9uIHRvIGFuIEFuZ3VsYXIgY29tcG9uZW50IG91dHB1dC5cbiAgICogQHRlbXBsYXRlIE0gLSBUeXBlIGV4dGVuZGluZyBNb2RlbFxuICAgKiBAcGFyYW0ge019IG1vZGVsIC0gVGhlIG1vZGVsIHRvIHJlbmRlclxuICAgKiBAcGFyYW0ge1JlY29yZDxzdHJpbmcsIHVua25vd24+fSBnbG9iYWxQcm9wcyAtIEdsb2JhbCBwcm9wZXJ0aWVzIHRvIHBhc3MgdG8gdGhlIGNvbXBvbmVudFxuICAgKiBAcGFyYW0ge1ZpZXdDb250YWluZXJSZWZ9IHZjciAtIFRoZSB2aWV3IGNvbnRhaW5lciByZWZlcmVuY2UgZm9yIGNvbXBvbmVudCBjcmVhdGlvblxuICAgKiBAcGFyYW0ge0luamVjdG9yfSBpbmplY3RvciAtIFRoZSBBbmd1bGFyIGluamVjdG9yIGZvciBkZXBlbmRlbmN5IGluamVjdGlvblxuICAgKiBAcGFyYW0ge1RlbXBsYXRlUmVmPGFueT59IHRwbCAtIFRoZSB0ZW1wbGF0ZSByZWZlcmVuY2UgZm9yIGNvbnRlbnQgcHJvamVjdGlvblxuICAgKiBAcmV0dXJuIHtBbmd1bGFyRHluYW1pY091dHB1dH0gVGhlIEFuZ3VsYXIgY29tcG9uZW50IG91dHB1dCB3aXRoIGNvbXBvbmVudCByZWZlcmVuY2UgYW5kIGlucHV0c1xuICAgKiBAbWVybWFpZFxuICAgKiBzZXF1ZW5jZURpYWdyYW1cbiAgICogICBwYXJ0aWNpcGFudCBDbGllbnQgYXMgQ2xpZW50IENvZGVcbiAgICogICBwYXJ0aWNpcGFudCBSZW5kZXIgYXMgcmVuZGVyIG1ldGhvZFxuICAgKiAgIHBhcnRpY2lwYW50IFRvRmllbGQgYXMgdG9GaWVsZERlZmluaXRpb25cbiAgICogICBwYXJ0aWNpcGFudCBGcm9tRmllbGQgYXMgZnJvbUZpZWxkRGVmaW5pdGlvblxuICAgKlxuICAgKiAgIENsaWVudC0+PlJlbmRlcjogcmVuZGVyKG1vZGVsLCBnbG9iYWxQcm9wcywgdmNyLCBpbmplY3RvciwgdHBsKVxuICAgKiAgIFJlbmRlci0+PlRvRmllbGQ6IHRvRmllbGREZWZpbml0aW9uKG1vZGVsLCBnbG9iYWxQcm9wcylcbiAgICogICBUb0ZpZWxkLS0+PlJlbmRlcjogZmllbGREZWZcbiAgICogICBSZW5kZXItPj5Gcm9tRmllbGQ6IGZyb21GaWVsZERlZmluaXRpb24oZmllbGREZWYsIHZjciwgaW5qZWN0b3IsIHRwbClcbiAgICogICBGcm9tRmllbGQtLT4+UmVuZGVyOiBBbmd1bGFyRHluYW1pY091dHB1dFxuICAgKiAgIFJlbmRlci0tPj5DbGllbnQ6IHJldHVybiBBbmd1bGFyRHluYW1pY091dHB1dFxuICAgKi9cbiAgb3ZlcnJpZGUgcmVuZGVyPE0gZXh0ZW5kcyBNb2RlbD4oXG4gICAgbW9kZWw6IE0sXG4gICAgZ2xvYmFsUHJvcHM6IFJlY29yZDxzdHJpbmcsIHVua25vd24+LFxuICAgIHZjcjogVmlld0NvbnRhaW5lclJlZixcbiAgICBpbmplY3RvcjogSW5qZWN0b3IsXG4gICAgdHBsOiBUZW1wbGF0ZVJlZjx1bmtub3duPixcbiAgKTogQW5ndWxhckR5bmFtaWNPdXRwdXQge1xuICAgIGxldCByZXN1bHQ6IEFuZ3VsYXJEeW5hbWljT3V0cHV0O1xuICAgIHRyeSB7XG4gICAgICBjb25zdCBmaWVsZERlZiA9IHRoaXMudG9GaWVsZERlZmluaXRpb24obW9kZWwsIGdsb2JhbFByb3BzKTtcbiAgICAgIHJlc3VsdCA9IHRoaXMuZnJvbUZpZWxkRGVmaW5pdGlvbihmaWVsZERlZiwgdmNyLCBpbmplY3RvciwgdHBsKTtcbiAgICB9IGNhdGNoIChlOiB1bmtub3duKSB7XG4gICAgICB0aHJvdyBuZXcgSW50ZXJuYWxFcnJvcihcbiAgICAgICAgYEZhaWxlZCB0byByZW5kZXIgTW9kZWwgJHttb2RlbC5jb25zdHJ1Y3Rvci5uYW1lfTogJHtlfWAsXG4gICAgICApO1xuICAgIH1cblxuICAgIHJldHVybiByZXN1bHQ7XG4gIH1cblxuICAvKipcbiAgICogQGRlc2NyaXB0aW9uIEluaXRpYWxpemVzIHRoZSByZW5kZXJpbmcgZW5naW5lXG4gICAqIEBzdW1tYXJ5IFRoaXMgbWV0aG9kIGluaXRpYWxpemVzIHRoZSByZW5kZXJpbmcgZW5naW5lLiBJdCBjaGVja3MgaWYgdGhlIGVuZ2luZSBpcyBhbHJlYWR5IGluaXRpYWxpemVkXG4gICAqIGFuZCBzZXRzIHRoZSBpbml0aWFsaXplZCBmbGFnIHRvIHRydWUuIFRoaXMgbWV0aG9kIGlzIGNhbGxlZCBiZWZvcmUgdGhlIGVuZ2luZSBpcyB1c2VkLlxuICAgKiBAcmV0dXJuIHtQcm9taXNlPHZvaWQ+fSBBIHByb21pc2UgdGhhdCByZXNvbHZlcyB3aGVuIGluaXRpYWxpemF0aW9uIGlzIGNvbXBsZXRlXG4gICAqL1xuICBvdmVycmlkZSBhc3luYyBpbml0aWFsaXplKCk6IFByb21pc2U8dm9pZD4ge1xuICAgIGlmICh0aGlzLmluaXRpYWxpemVkKSByZXR1cm47XG4gICAgdGhpcy5pbml0aWFsaXplZCA9IHRydWU7XG4gIH1cblxuICAvKipcbiAgICogQGRlc2NyaXB0aW9uIFJlZ2lzdGVycyBhIGNvbXBvbmVudCB3aXRoIHRoZSByZW5kZXJpbmcgZW5naW5lXG4gICAqIEBzdW1tYXJ5IFRoaXMgc3RhdGljIG1ldGhvZCByZWdpc3RlcnMgYSBjb21wb25lbnQgY29uc3RydWN0b3Igd2l0aCB0aGUgcmVuZGVyaW5nIGVuZ2luZVxuICAgKiB1bmRlciBhIHNwZWNpZmljIG5hbWUuIEl0IHRocm93cyBhbiBlcnJvciBpZiBhIGNvbXBvbmVudCBpcyBhbHJlYWR5IHJlZ2lzdGVyZWQgdW5kZXIgdGhlIHNhbWUgbmFtZS5cbiAgICogQHBhcmFtIHtzdHJpbmd9IG5hbWUgLSBUaGUgbmFtZSB0byByZWdpc3RlciB0aGUgY29tcG9uZW50IHVuZGVyXG4gICAqIEBwYXJhbSB7Q29uc3RydWN0b3I8dW5rbm93bj59IGNvbnN0cnVjdG9yIC0gVGhlIGNvbXBvbmVudCBjb25zdHJ1Y3RvclxuICAgKiBAcmV0dXJuIHt2b2lkfVxuICAgKi9cbiAgc3RhdGljIHJlZ2lzdGVyQ29tcG9uZW50KG5hbWU6IHN0cmluZywgY29uc3RydWN0b3I6IENvbnN0cnVjdG9yPHVua25vd24+KSB7XG4gICAgaWYgKCF0aGlzLl9jb21wb25lbnRzKSB0aGlzLl9jb21wb25lbnRzID0ge307XG4gICAgaWYgKG5hbWUgaW4gdGhpcy5fY29tcG9uZW50cylcbiAgICAgIHRocm93IG5ldyBJbnRlcm5hbEVycm9yKGBDb21wb25lbnQgYWxyZWFkeSByZWdpc3RlcmVkIHVuZGVyICR7bmFtZX1gKTtcbiAgICB0aGlzLl9jb21wb25lbnRzW25hbWVdID0ge1xuICAgICAgY29uc3RydWN0b3I6IGNvbnN0cnVjdG9yLFxuICAgIH07XG4gIH1cblxuICAvKipcbiAgICogQGRlc2NyaXB0aW9uIFJldHJpZXZlcyByZWdpc3RlcmVkIGNvbXBvbmVudHMgZnJvbSB0aGUgcmVuZGVyaW5nIGVuZ2luZVxuICAgKiBAc3VtbWFyeSBUaGlzIHN0YXRpYyBtZXRob2QgcmV0cmlldmVzIGVpdGhlciBhbGwgcmVnaXN0ZXJlZCBjb21wb25lbnRzIG9yIGEgc3BlY2lmaWMgY29tcG9uZW50XG4gICAqIGJ5IGl0cyBzZWxlY3Rvci4gSXQgdGhyb3dzIGFuIGVycm9yIGlmIHRoZSByZXF1ZXN0ZWQgY29tcG9uZW50IGlzIG5vdCByZWdpc3RlcmVkLlxuICAgKiBAcGFyYW0ge3N0cmluZ30gW3NlbGVjdG9yXSAtIE9wdGlvbmFsIHNlbGVjdG9yIHRvIHJldHJpZXZlIGEgc3BlY2lmaWMgY29tcG9uZW50XG4gICAqIEByZXR1cm4ge09iamVjdHxBcnJheX0gRWl0aGVyIGEgc3BlY2lmaWMgY29tcG9uZW50IG9yIGFuIGFycmF5IG9mIGFsbCBjb21wb25lbnRzXG4gICAqL1xuICBzdGF0aWMgY29tcG9uZW50cyhzZWxlY3Rvcj86IHN0cmluZykge1xuICAgIGlmICghc2VsZWN0b3IpIHJldHVybiBPYmplY3QudmFsdWVzKHRoaXMuX2NvbXBvbmVudHMpO1xuICAgIGlmICghKHNlbGVjdG9yIGluIHRoaXMuX2NvbXBvbmVudHMpKVxuICAgICAgdGhyb3cgbmV3IEludGVybmFsRXJyb3IoYE5vIENvbXBvbmVudCByZWdpc3RlcmVkIHVuZGVyICR7c2VsZWN0b3J9YCk7XG4gICAgcmV0dXJuIHRoaXMuX2NvbXBvbmVudHNbc2VsZWN0b3JdO1xuICB9XG5cbiAgLyoqXG4gICAqIEBkZXNjcmlwdGlvbiBHZW5lcmF0ZXMgYSBrZXkgZm9yIHJlZmxlY3Rpb24gbWV0YWRhdGFcbiAgICogQHN1bW1hcnkgVGhpcyBzdGF0aWMgbWV0aG9kIGdlbmVyYXRlcyBhIGtleSBmb3IgcmVmbGVjdGlvbiBtZXRhZGF0YSBieSBwcmVmaXhpbmcgdGhlIGlucHV0IGtleVxuICAgKiB3aXRoIHRoZSBBbmd1bGFyIGVuZ2luZSdzIHJlZmxlY3Rpb24gcHJlZml4LiBUaGlzIGlzIHVzZWQgZm9yIHN0b3JpbmcgYW5kIHJldHJpZXZpbmcgbWV0YWRhdGEuXG4gICAqIEBwYXJhbSB7c3RyaW5nfSBrZXkgLSBUaGUgYmFzZSBrZXkgdG8gcHJlZml4XG4gICAqIEByZXR1cm4ge3N0cmluZ30gVGhlIHByZWZpeGVkIGtleSBmb3IgcmVmbGVjdGlvbiBtZXRhZGF0YVxuICAgKi9cbiAgc3RhdGljIG92ZXJyaWRlIGtleShrZXk6IHN0cmluZykge1xuICAgIHJldHVybiBgJHtBbmd1bGFyRW5naW5lS2V5cy5SRUZMRUNUfSR7a2V5fWA7XG4gIH1cbn1cbiJdfQ==