@gnggln/ng-ui-system 1.0.0-alpha.0

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 (141) hide show
  1. package/esm2022/gnggln-ng-ui-system.mjs +5 -0
  2. package/esm2022/lib/components/accordion/accordion.component.mjs +353 -0
  3. package/esm2022/lib/components/accordion/accordion.types.mjs +6 -0
  4. package/esm2022/lib/components/accordion/index.mjs +2 -0
  5. package/esm2022/lib/components/base-layout/base-layout.component.mjs +218 -0
  6. package/esm2022/lib/components/base-layout/base-layout.types.mjs +6 -0
  7. package/esm2022/lib/components/base-layout/index.mjs +14 -0
  8. package/esm2022/lib/components/button/button-area.component.mjs +196 -0
  9. package/esm2022/lib/components/button/button.component.mjs +164 -0
  10. package/esm2022/lib/components/button/button.types.mjs +6 -0
  11. package/esm2022/lib/components/button/index.mjs +16 -0
  12. package/esm2022/lib/components/crud-table/crud-table.component.mjs +789 -0
  13. package/esm2022/lib/components/crud-table/crud-table.types.mjs +6 -0
  14. package/esm2022/lib/components/crud-table/index.mjs +16 -0
  15. package/esm2022/lib/components/form-builder/adapters/it-date-adapter.mjs +82 -0
  16. package/esm2022/lib/components/form-builder/directives/currency-input.directive.mjs +184 -0
  17. package/esm2022/lib/components/form-builder/form-builder.component.mjs +824 -0
  18. package/esm2022/lib/components/form-builder/form-wizard.component.mjs +510 -0
  19. package/esm2022/lib/components/form-builder/index.mjs +19 -0
  20. package/esm2022/lib/components/form-builder/services/form-condition.service.mjs +132 -0
  21. package/esm2022/lib/components/form-builder/services/form-validation.service.mjs +381 -0
  22. package/esm2022/lib/components/form-builder/services/location.service.mjs +140 -0
  23. package/esm2022/lib/components/form-builder/services/wizard-sync.service.mjs +84 -0
  24. package/esm2022/lib/components/form-builder/sub-components/error-summary/form-error-summary.component.mjs +161 -0
  25. package/esm2022/lib/components/form-builder/sub-components/file-input/file-input.component.mjs +310 -0
  26. package/esm2022/lib/components/form-builder/sub-components/specifica-territoriale/specifica-territoriale.component.mjs +648 -0
  27. package/esm2022/lib/components/form-builder/sub-components/table-territoriale/table-territoriale.component.mjs +432 -0
  28. package/esm2022/lib/components/form-builder/types/condition.types.mjs +6 -0
  29. package/esm2022/lib/components/form-builder/types/field.types.mjs +6 -0
  30. package/esm2022/lib/components/form-builder/types/index.mjs +2 -0
  31. package/esm2022/lib/components/form-builder/types/schema.types.mjs +6 -0
  32. package/esm2022/lib/components/form-builder/types/territoriale.types.mjs +6 -0
  33. package/esm2022/lib/components/form-builder/types/validation.types.mjs +6 -0
  34. package/esm2022/lib/components/form-builder-editor/form-builder-editor.component.mjs +730 -0
  35. package/esm2022/lib/components/form-builder-editor/form-builder-editor.service.mjs +56 -0
  36. package/esm2022/lib/components/form-builder-editor/index.mjs +21 -0
  37. package/esm2022/lib/components/form-builder-editor/services/editor-persistence.service.mjs +190 -0
  38. package/esm2022/lib/components/form-builder-editor/services/editor-state.service.mjs +324 -0
  39. package/esm2022/lib/components/form-builder-editor/services/field-factory.service.mjs +188 -0
  40. package/esm2022/lib/components/form-builder-editor/sub-components/condition-editor/condition-editor.component.mjs +667 -0
  41. package/esm2022/lib/components/form-builder-editor/sub-components/editor-toolbar/editor-toolbar.component.mjs +317 -0
  42. package/esm2022/lib/components/form-builder-editor/sub-components/field-config-panel/field-config-panel.component.mjs +611 -0
  43. package/esm2022/lib/components/form-builder-editor/sub-components/field-palette/field-palette.component.mjs +267 -0
  44. package/esm2022/lib/components/form-builder-editor/sub-components/form-values-panel/form-values-panel.component.mjs +276 -0
  45. package/esm2022/lib/components/form-builder-editor/sub-components/options-editor/options-editor.component.mjs +323 -0
  46. package/esm2022/lib/components/form-builder-editor/sub-components/preview-container/preview-container.component.mjs +238 -0
  47. package/esm2022/lib/components/form-builder-editor/sub-components/section-editor/section-editor.component.mjs +472 -0
  48. package/esm2022/lib/components/form-builder-editor/sub-components/validation-editor/validation-editor.component.mjs +473 -0
  49. package/esm2022/lib/components/form-builder-editor/types/editor.types.mjs +6 -0
  50. package/esm2022/lib/components/layout-builder/index.mjs +18 -0
  51. package/esm2022/lib/components/layout-builder/layout-builder.component.mjs +1730 -0
  52. package/esm2022/lib/components/layout-builder/layout-builder.types.mjs +9 -0
  53. package/esm2022/lib/components/layout-builder/layout.service.mjs +239 -0
  54. package/esm2022/lib/components/modal/confirm-dialog.component.mjs +151 -0
  55. package/esm2022/lib/components/modal/index.mjs +4 -0
  56. package/esm2022/lib/components/modal/modal.component.mjs +139 -0
  57. package/esm2022/lib/components/modal/modal.service.mjs +194 -0
  58. package/esm2022/lib/components/modal/modal.types.mjs +6 -0
  59. package/esm2022/lib/components/page-header/breadcrumb.service.mjs +242 -0
  60. package/esm2022/lib/components/page-header/index.mjs +20 -0
  61. package/esm2022/lib/components/page-header/page-header.component.mjs +243 -0
  62. package/esm2022/lib/components/page-header/page-header.types.mjs +21 -0
  63. package/esm2022/lib/components/table/index.mjs +2 -0
  64. package/esm2022/lib/components/table/paginated-table.component.mjs +407 -0
  65. package/esm2022/lib/components/table/table.types.mjs +6 -0
  66. package/esm2022/lib/core/types/index.mjs +6 -0
  67. package/esm2022/lib/core/utils/index.mjs +53 -0
  68. package/esm2022/lib/sources/location-data.opt.json +8942 -0
  69. package/esm2022/lib/sources/nazioni.opt.json +215 -0
  70. package/esm2022/public-api.mjs +34 -0
  71. package/fesm2022/gnggln-ng-ui-system.mjs +55752 -0
  72. package/fesm2022/gnggln-ng-ui-system.mjs.map +1 -0
  73. package/index.d.ts +5 -0
  74. package/lib/components/accordion/accordion.component.d.ts +118 -0
  75. package/lib/components/accordion/accordion.types.d.ts +62 -0
  76. package/lib/components/accordion/index.d.ts +2 -0
  77. package/lib/components/base-layout/base-layout.component.d.ts +83 -0
  78. package/lib/components/base-layout/base-layout.types.d.ts +26 -0
  79. package/lib/components/base-layout/index.d.ts +13 -0
  80. package/lib/components/button/button-area.component.d.ts +88 -0
  81. package/lib/components/button/button.component.d.ts +55 -0
  82. package/lib/components/button/button.types.d.ts +70 -0
  83. package/lib/components/button/index.d.ts +15 -0
  84. package/lib/components/crud-table/crud-table.component.d.ts +143 -0
  85. package/lib/components/crud-table/crud-table.types.d.ts +207 -0
  86. package/lib/components/crud-table/index.d.ts +15 -0
  87. package/lib/components/form-builder/adapters/it-date-adapter.d.ts +32 -0
  88. package/lib/components/form-builder/directives/currency-input.directive.d.ts +48 -0
  89. package/lib/components/form-builder/form-builder.component.d.ts +183 -0
  90. package/lib/components/form-builder/form-wizard.component.d.ts +87 -0
  91. package/lib/components/form-builder/index.d.ts +13 -0
  92. package/lib/components/form-builder/services/form-condition.service.d.ts +46 -0
  93. package/lib/components/form-builder/services/form-validation.service.d.ts +63 -0
  94. package/lib/components/form-builder/services/location.service.d.ts +83 -0
  95. package/lib/components/form-builder/services/wizard-sync.service.d.ts +63 -0
  96. package/lib/components/form-builder/sub-components/error-summary/form-error-summary.component.d.ts +28 -0
  97. package/lib/components/form-builder/sub-components/file-input/file-input.component.d.ts +41 -0
  98. package/lib/components/form-builder/sub-components/specifica-territoriale/specifica-territoriale.component.d.ts +145 -0
  99. package/lib/components/form-builder/sub-components/table-territoriale/table-territoriale.component.d.ts +108 -0
  100. package/lib/components/form-builder/types/condition.types.d.ts +51 -0
  101. package/lib/components/form-builder/types/field.types.d.ts +288 -0
  102. package/lib/components/form-builder/types/index.d.ts +5 -0
  103. package/lib/components/form-builder/types/schema.types.d.ts +227 -0
  104. package/lib/components/form-builder/types/territoriale.types.d.ts +170 -0
  105. package/lib/components/form-builder/types/validation.types.d.ts +174 -0
  106. package/lib/components/form-builder-editor/form-builder-editor.component.d.ts +117 -0
  107. package/lib/components/form-builder-editor/form-builder-editor.service.d.ts +38 -0
  108. package/lib/components/form-builder-editor/index.d.ts +15 -0
  109. package/lib/components/form-builder-editor/services/editor-persistence.service.d.ts +42 -0
  110. package/lib/components/form-builder-editor/services/editor-state.service.d.ts +66 -0
  111. package/lib/components/form-builder-editor/services/field-factory.service.d.ts +28 -0
  112. package/lib/components/form-builder-editor/sub-components/condition-editor/condition-editor.component.d.ts +139 -0
  113. package/lib/components/form-builder-editor/sub-components/editor-toolbar/editor-toolbar.component.d.ts +43 -0
  114. package/lib/components/form-builder-editor/sub-components/field-config-panel/field-config-panel.component.d.ts +83 -0
  115. package/lib/components/form-builder-editor/sub-components/field-palette/field-palette.component.d.ts +40 -0
  116. package/lib/components/form-builder-editor/sub-components/form-values-panel/form-values-panel.component.d.ts +51 -0
  117. package/lib/components/form-builder-editor/sub-components/options-editor/options-editor.component.d.ts +63 -0
  118. package/lib/components/form-builder-editor/sub-components/preview-container/preview-container.component.d.ts +68 -0
  119. package/lib/components/form-builder-editor/sub-components/section-editor/section-editor.component.d.ts +82 -0
  120. package/lib/components/form-builder-editor/sub-components/validation-editor/validation-editor.component.d.ts +112 -0
  121. package/lib/components/form-builder-editor/types/editor.types.d.ts +124 -0
  122. package/lib/components/layout-builder/index.d.ts +16 -0
  123. package/lib/components/layout-builder/layout-builder.component.d.ts +85 -0
  124. package/lib/components/layout-builder/layout-builder.types.d.ts +436 -0
  125. package/lib/components/layout-builder/layout.service.d.ts +100 -0
  126. package/lib/components/modal/confirm-dialog.component.d.ts +46 -0
  127. package/lib/components/modal/index.d.ts +4 -0
  128. package/lib/components/modal/modal.component.d.ts +44 -0
  129. package/lib/components/modal/modal.service.d.ts +93 -0
  130. package/lib/components/modal/modal.types.d.ts +110 -0
  131. package/lib/components/page-header/breadcrumb.service.d.ts +96 -0
  132. package/lib/components/page-header/index.d.ts +16 -0
  133. package/lib/components/page-header/page-header.component.d.ts +59 -0
  134. package/lib/components/page-header/page-header.types.d.ts +96 -0
  135. package/lib/components/table/index.d.ts +2 -0
  136. package/lib/components/table/paginated-table.component.d.ts +85 -0
  137. package/lib/components/table/table.types.d.ts +81 -0
  138. package/lib/core/types/index.d.ts +57 -0
  139. package/lib/core/utils/index.d.ts +29 -0
  140. package/package.json +44 -0
  141. package/public-api.d.ts +22 -0
@@ -0,0 +1,789 @@
1
+ import { Component, Input, Output, EventEmitter, ViewChild, ChangeDetectionStrategy, ChangeDetectorRef, ViewEncapsulation, inject, } from '@angular/core';
2
+ import { NgTemplateOutlet } from '@angular/common';
3
+ import { MatTooltipModule } from '@angular/material/tooltip';
4
+ import { LucideAngularModule } from 'lucide-angular';
5
+ import { Subject } from 'rxjs';
6
+ import { UiButtonAreaComponent } from '../button/button-area.component';
7
+ import { UiFormBuilderComponent } from '../form-builder/form-builder.component';
8
+ import { UiModalService } from '../modal/modal.service';
9
+ import * as i0 from "@angular/core";
10
+ import * as i1 from "@angular/material/tooltip";
11
+ import * as i2 from "lucide-angular";
12
+ /**
13
+ * Tabella CRUD standalone con form builder integrato.
14
+ *
15
+ * Gestisce il ciclo completo delle operazioni CRUD:
16
+ * inserimento, modifica, duplicazione, eliminazione e
17
+ * visualizzazione in sola lettura. Il form e generato
18
+ * automaticamente dallo schema `UiFormSchema`.
19
+ *
20
+ * Utilizza `UiModalService` per la conferma di eliminazione
21
+ * e `UiFormBuilderComponent` per la gestione del form.
22
+ *
23
+ * @selector ui-crud-table
24
+ *
25
+ * @example
26
+ * ```html
27
+ * <ui-crud-table
28
+ * [data]="items"
29
+ * [columns]="columns"
30
+ * [config]="crudConfig"
31
+ * (dataChange)="onDataChange($event)"
32
+ * (itemAdd)="onAdd($event)"
33
+ * (itemEdit)="onEdit($event)"
34
+ * (itemDelete)="onDelete($event)"
35
+ * />
36
+ * ```
37
+ */
38
+ export class UiCrudTableComponent {
39
+ constructor() {
40
+ this.cdr = inject(ChangeDetectorRef);
41
+ this.modalService = inject(UiModalService);
42
+ this.destroy$ = new Subject();
43
+ // ── Inputs ──────────────────────────────────────────────────────
44
+ /** Array di dati da visualizzare nella tabella. */
45
+ this.data = [];
46
+ /** Definizione delle colonne. */
47
+ this.columns = [];
48
+ /** Mostra l'overlay di caricamento. */
49
+ this.loading = false;
50
+ /** Disabilita tutte le azioni CRUD. */
51
+ this.disabled = false;
52
+ // ── Outputs ─────────────────────────────────────────────────────
53
+ /** Emesso ogni volta che la lista dati cambia (insert/edit/duplicate). */
54
+ this.dataChange = new EventEmitter();
55
+ /** Emesso quando viene inserito un nuovo elemento. */
56
+ this.itemAdd = new EventEmitter();
57
+ /** Emesso quando viene modificato un elemento. */
58
+ this.itemEdit = new EventEmitter();
59
+ /** Emesso quando viene eliminato un elemento (dopo conferma). */
60
+ this.itemDelete = new EventEmitter();
61
+ /** Emesso quando viene duplicato un elemento. */
62
+ this.itemDuplicate = new EventEmitter();
63
+ /** Emesso quando l'utente clicca su una riga in modalita view. */
64
+ this.rowClick = new EventEmitter();
65
+ // ── Stato interno ───────────────────────────────────────────────
66
+ /** @internal Modalita corrente. */
67
+ this.currentMode = 'view';
68
+ /** @internal Elemento in fase di modifica/visualizzazione. */
69
+ this.editingItem = null;
70
+ /** @internal Indice dell'elemento in modifica. */
71
+ this.editingIndex = null;
72
+ /** @internal Dati originali dell'elemento prima della modifica. */
73
+ this.lastOriginalData = null;
74
+ /** @internal Cache per i dati iniziali del form. */
75
+ this._formInitialData = {};
76
+ this._lastCacheKey = null;
77
+ /** @internal Pulsanti per il form. */
78
+ this.formButtons = [];
79
+ /** @internal Pulsante aggiungi. */
80
+ this.addButton = [];
81
+ /** @internal Track function per il loop @for. */
82
+ this.trackByFn = (index, item) => {
83
+ const trackBy = this.config?.trackByProperty || 'id';
84
+ return item[trackBy] ?? index;
85
+ };
86
+ }
87
+ // ── Lifecycle ───────────────────────────────────────────────────
88
+ ngOnInit() {
89
+ this.setupButtons();
90
+ }
91
+ ngOnDestroy() {
92
+ this.destroy$.next();
93
+ this.destroy$.complete();
94
+ }
95
+ // ── Getter calcolati ────────────────────────────────────────────
96
+ /** La tabella e in modalita form (insert/edit/duplicate/readonly). */
97
+ get isFormMode() {
98
+ return this.currentMode !== 'view';
99
+ }
100
+ /** La tabella e in modalita sola lettura. */
101
+ get isReadonlyMode() {
102
+ return this.currentMode === 'readonly';
103
+ }
104
+ /** Dati iniziali calcolati per il form builder. */
105
+ get formInitialData() {
106
+ const cacheKey = this.generateCacheKey();
107
+ if (this._lastCacheKey === cacheKey && this._formInitialData) {
108
+ return this._formInitialData;
109
+ }
110
+ let data = {};
111
+ if (this.currentMode === 'edit' || this.currentMode === 'duplicate' || this.currentMode === 'readonly') {
112
+ data = this.initMapper ? this.initMapper(this.editingItem) : this.editingItem || {};
113
+ }
114
+ this._formInitialData = data;
115
+ this._lastCacheKey = cacheKey;
116
+ return this._formInitialData;
117
+ }
118
+ // ── Permessi ────────────────────────────────────────────────────
119
+ /** Verifica se ci sono azioni disponibili (per mostrare la colonna azioni). */
120
+ hasAnyActions() {
121
+ return (this.config?.allowEdit !== false ||
122
+ this.config?.allowDuplicate === true ||
123
+ this.config?.allowDelete !== false ||
124
+ this.config?.allowView === true);
125
+ }
126
+ /** Verifica se l'elemento puo essere modificato. */
127
+ canEditItem(item) {
128
+ if (this.config?.canEdit) {
129
+ return this.config.canEdit(item, this.currentMode === 'readonly');
130
+ }
131
+ return this.config?.allowEdit !== false;
132
+ }
133
+ /** Verifica se l'elemento puo essere eliminato. */
134
+ canDeleteItem(item) {
135
+ if (this.config?.canDelete) {
136
+ return this.config.canDelete(item, this.currentMode === 'readonly');
137
+ }
138
+ return this.config?.allowDelete !== false;
139
+ }
140
+ /** Verifica se l'elemento puo essere duplicato. */
141
+ canDuplicateItem(item) {
142
+ if (this.config?.canDuplicate) {
143
+ return this.config.canDuplicate(item, this.currentMode === 'readonly');
144
+ }
145
+ return this.config?.allowDuplicate === true;
146
+ }
147
+ /** Verifica se l'elemento puo essere visualizzato in readonly. */
148
+ canViewItem(item) {
149
+ if (this.config?.canView) {
150
+ return this.config.canView(item);
151
+ }
152
+ return this.config?.allowView === true;
153
+ }
154
+ // ── Tooltip ─────────────────────────────────────────────────────
155
+ getEditTooltip(item) {
156
+ if (this.config?.getEditTooltip)
157
+ return this.config.getEditTooltip(item);
158
+ return this.canEditItem(item) ? 'Modifica' : 'Modifica non disponibile';
159
+ }
160
+ getDeleteTooltip(item) {
161
+ if (this.config?.getDeleteTooltip)
162
+ return this.config.getDeleteTooltip(item);
163
+ return this.canDeleteItem(item) ? 'Elimina' : 'Eliminazione non disponibile';
164
+ }
165
+ getDuplicateTooltip(item) {
166
+ if (this.config?.getDuplicateTooltip)
167
+ return this.config.getDuplicateTooltip(item);
168
+ return this.canDuplicateItem(item) ? 'Duplica' : 'Duplicazione non disponibile';
169
+ }
170
+ getViewTooltip(item) {
171
+ if (this.config?.getViewTooltip)
172
+ return this.config.getViewTooltip(item);
173
+ return this.canViewItem(item) ? 'Visualizza' : 'Visualizzazione non disponibile';
174
+ }
175
+ // ── Classi CSS per riga ─────────────────────────────────────────
176
+ getRowClasses(item, index) {
177
+ let classes = 'ui-crud-table__row';
178
+ if (this.config?.getRowClass) {
179
+ const extra = this.config.getRowClass(item, index);
180
+ if (Array.isArray(extra)) {
181
+ classes += ' ' + extra.join(' ');
182
+ }
183
+ else if (extra) {
184
+ classes += ' ' + extra;
185
+ }
186
+ }
187
+ return classes;
188
+ }
189
+ // ── Ellipsis ────────────────────────────────────────────────────
190
+ shouldShowEllipsis(col, value) {
191
+ if (!col.ellipsis || value == null)
192
+ return false;
193
+ return String(value).length > (col.maxLength || 50);
194
+ }
195
+ truncateText(col, value) {
196
+ if (value == null)
197
+ return '\u2014';
198
+ const str = String(value);
199
+ const max = col.maxLength || 50;
200
+ return str.length > max ? str.substring(0, max) + '\u2026' : str;
201
+ }
202
+ getFullText(value) {
203
+ return String(value ?? '\u2014');
204
+ }
205
+ // ── Operazioni CRUD ─────────────────────────────────────────────
206
+ /** Attiva la modalita inserimento. */
207
+ activateInsertMode() {
208
+ this.config?.onBeforeFormOpen?.('insert', null, this.data);
209
+ this.currentMode = 'insert';
210
+ this.editingItem = null;
211
+ this.editingIndex = null;
212
+ this.invalidateFormCache();
213
+ this.setupButtons();
214
+ this.cdr.detectChanges();
215
+ }
216
+ /** Attiva la modalita modifica per un elemento. */
217
+ activateEditMode(item, index) {
218
+ if (!this.canEditItem(item))
219
+ return;
220
+ this.config?.onBeforeFormOpen?.('edit', item, this.data);
221
+ this.currentMode = 'edit';
222
+ this.lastOriginalData = item;
223
+ this.editingItem = { ...item };
224
+ this.editingIndex = index;
225
+ this.invalidateFormCache();
226
+ this.setupButtons();
227
+ this.cdr.detectChanges();
228
+ }
229
+ /** Attiva la modalita visualizzazione readonly per un elemento. */
230
+ activateViewMode(item, index) {
231
+ if (!this.canViewItem(item))
232
+ return;
233
+ this.currentMode = 'readonly';
234
+ this.lastOriginalData = item;
235
+ this.editingItem = { ...item };
236
+ this.editingIndex = index;
237
+ this.invalidateFormCache();
238
+ this.setupButtons();
239
+ this.cdr.detectChanges();
240
+ }
241
+ /** Attiva la modalita duplicazione per un elemento. */
242
+ activateDuplicateMode(item) {
243
+ if (!this.canDuplicateItem(item))
244
+ return;
245
+ this.config?.onBeforeFormOpen?.('duplicate', item, this.data);
246
+ this.currentMode = 'duplicate';
247
+ this.editingItem = { ...item };
248
+ this.editingIndex = null;
249
+ this.invalidateFormCache();
250
+ this.setupButtons();
251
+ this.cdr.detectChanges();
252
+ }
253
+ /** Annulla il form e torna alla vista tabella. */
254
+ cancelForm() {
255
+ this.currentMode = 'view';
256
+ this.editingItem = null;
257
+ this.editingIndex = null;
258
+ this.lastOriginalData = null;
259
+ this.invalidateFormCache();
260
+ this.setupButtons();
261
+ this.cdr.detectChanges();
262
+ }
263
+ /** Salva i dati del form (insert/edit/duplicate). */
264
+ saveForm() {
265
+ if (!this.formBuilder)
266
+ return;
267
+ if (!this.formBuilder.isFormValid()) {
268
+ this.formBuilder.validateAllFields();
269
+ return;
270
+ }
271
+ const formData = this.formBuilder.getFormValue();
272
+ const newData = [...this.data];
273
+ switch (this.currentMode) {
274
+ case 'insert':
275
+ newData.push(formData);
276
+ this.itemAdd.emit(formData);
277
+ break;
278
+ case 'duplicate':
279
+ newData.push(formData);
280
+ this.itemDuplicate.emit(formData);
281
+ break;
282
+ case 'edit':
283
+ if (this.editingIndex !== null) {
284
+ newData[this.editingIndex] = formData;
285
+ this.itemEdit.emit({
286
+ item: formData,
287
+ index: this.editingIndex,
288
+ originalData: this.lastOriginalData,
289
+ });
290
+ }
291
+ break;
292
+ }
293
+ this.data = newData;
294
+ this.dataChange.emit(this.data);
295
+ this.cancelForm();
296
+ }
297
+ /** Elimina un elemento con conferma tramite UiModalService. */
298
+ deleteItem(item, index) {
299
+ if (!this.canDeleteItem(item))
300
+ return;
301
+ const message = this.config?.deleteConfirmMessage || 'Sei sicuro di voler eliminare questo elemento?';
302
+ this.modalService
303
+ .confirm({
304
+ title: 'Conferma eliminazione',
305
+ message,
306
+ confirmText: 'Elimina',
307
+ cancelText: 'Annulla',
308
+ variant: 'delete',
309
+ })
310
+ .subscribe((confirmed) => {
311
+ if (confirmed) {
312
+ this.itemDelete.emit({ item, index });
313
+ }
314
+ });
315
+ }
316
+ // ── Event handlers template ─────────────────────────────────────
317
+ onRowClick(item) {
318
+ if (this.currentMode === 'view') {
319
+ this.rowClick.emit(item);
320
+ }
321
+ }
322
+ /** @internal */
323
+ onViewBtnClick(event, item, index) {
324
+ event.stopPropagation();
325
+ if (this.disabled || !this.canViewItem(item))
326
+ return;
327
+ this.activateViewMode(item, index);
328
+ }
329
+ /** @internal */
330
+ onEditBtnClick(event, item, index) {
331
+ event.stopPropagation();
332
+ if (this.disabled || !this.canEditItem(item))
333
+ return;
334
+ this.activateEditMode(item, index);
335
+ }
336
+ /** @internal */
337
+ onDuplicateBtnClick(event, item) {
338
+ event.stopPropagation();
339
+ if (this.disabled || !this.canDuplicateItem(item))
340
+ return;
341
+ this.activateDuplicateMode(item);
342
+ }
343
+ /** @internal */
344
+ onDeleteBtnClick(event, item, index) {
345
+ event.stopPropagation();
346
+ if (this.disabled || !this.canDeleteItem(item))
347
+ return;
348
+ this.deleteItem(item, index);
349
+ }
350
+ // ── Utilita private ─────────────────────────────────────────────
351
+ /** @internal Genera una chiave per la cache dei dati del form. */
352
+ generateCacheKey() {
353
+ if (this.currentMode === 'view' || this.currentMode === 'insert') {
354
+ return this.currentMode;
355
+ }
356
+ const itemId = this.editingItem ? JSON.stringify(this.editingItem) : 'null';
357
+ return `${this.currentMode}-${itemId}`;
358
+ }
359
+ /** @internal Invalida la cache dei dati del form. */
360
+ invalidateFormCache() {
361
+ this._formInitialData = {};
362
+ this._lastCacheKey = null;
363
+ }
364
+ /** @internal Configura i pulsanti in base alla modalita corrente. */
365
+ setupButtons() {
366
+ // Pulsante "Aggiungi" (visibile solo in modalita view)
367
+ this.addButton =
368
+ this.currentMode === 'view' && this.config?.allowAdd !== false && !this.disabled
369
+ ? [
370
+ {
371
+ id: 'crud-add',
372
+ label: this.config?.addButtonLabel || 'Aggiungi',
373
+ variant: 'outline',
374
+ icon: 'plus',
375
+ action: () => this.activateInsertMode(),
376
+ },
377
+ ]
378
+ : [];
379
+ // Pulsanti form (visibili in modalita form)
380
+ this.formButtons = this.isFormMode
381
+ ? [
382
+ {
383
+ id: 'crud-cancel',
384
+ label: this.isReadonlyMode ? 'Chiudi' : this.config?.cancelButtonLabel || 'Annulla',
385
+ variant: 'outline',
386
+ action: () => this.cancelForm(),
387
+ },
388
+ ...(this.isReadonlyMode
389
+ ? []
390
+ : [
391
+ {
392
+ id: 'crud-save',
393
+ label: this.getSaveButtonLabel(),
394
+ variant: 'primary',
395
+ icon: 'check',
396
+ action: () => this.saveForm(),
397
+ },
398
+ ]),
399
+ ]
400
+ : [];
401
+ }
402
+ /** @internal Restituisce la label del pulsante salva in base alla modalita. */
403
+ getSaveButtonLabel() {
404
+ switch (this.currentMode) {
405
+ case 'edit':
406
+ return this.config?.updateButtonLabel || 'Aggiorna';
407
+ case 'duplicate':
408
+ return this.config?.duplicateButtonLabel || 'Duplica';
409
+ default:
410
+ return this.config?.saveButtonLabel || 'Salva';
411
+ }
412
+ }
413
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: UiCrudTableComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
414
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.14", type: UiCrudTableComponent, isStandalone: true, selector: "ui-crud-table", inputs: { data: "data", columns: "columns", config: "config", loading: "loading", disabled: "disabled", initMapper: "initMapper" }, outputs: { dataChange: "dataChange", itemAdd: "itemAdd", itemEdit: "itemEdit", itemDelete: "itemDelete", itemDuplicate: "itemDuplicate", rowClick: "rowClick" }, host: { classAttribute: "ui-crud-table-host" }, viewQueries: [{ propertyName: "formBuilder", first: true, predicate: UiFormBuilderComponent, descendants: true }], ngImport: i0, template: `
415
+ <div class="ui-crud-table">
416
+
417
+ <!-- ═══ Stato vuoto ═══════════════════════════════════════ -->
418
+ @if (currentMode === 'view' && (!data || data.length === 0) && !loading) {
419
+ <div class="ui-crud-table__empty">
420
+ <div class="ui-crud-table__empty-message">
421
+ <lucide-icon name="info" [size]="20" class="ui-crud-table__empty-icon" aria-hidden="true" />
422
+ <span>{{ config?.emptyMessage || 'Nessun elemento presente' }}</span>
423
+ </div>
424
+ <ui-button-area [buttons]="addButton" align="end" gap="xs" />
425
+ </div>
426
+ }
427
+
428
+ <!-- ═══ Stato di caricamento ══════════════════════════════ -->
429
+ @if (loading && currentMode === 'view') {
430
+ <div class="ui-crud-table__loading" role="status" aria-live="polite">
431
+ <span class="ui-crud-table__spinner" aria-hidden="true"></span>
432
+ <span class="ui-crud-table__sr-only">Caricamento in corso</span>
433
+ </div>
434
+ }
435
+
436
+ <!-- ═══ Vista tabella ═════════════════════════════════════ -->
437
+ @if (currentMode === 'view' && data && data.length > 0 && !loading) {
438
+ <div class="ui-crud-table__container">
439
+ <table class="ui-crud-table__table" role="grid" [attr.aria-label]="config?.tableLabel || 'Tabella CRUD'">
440
+ <thead>
441
+ <tr>
442
+ @for (col of columns; track col.key) {
443
+ <th
444
+ class="ui-crud-table__th"
445
+ [style.width]="col.width || null"
446
+ [style.text-align]="col.align || 'left'"
447
+ >
448
+ @if (col.headerTemplate) {
449
+ <ng-container
450
+ [ngTemplateOutlet]="col.headerTemplate"
451
+ [ngTemplateOutletContext]="{ $implicit: col, column: col }"
452
+ />
453
+ } @else {
454
+ {{ col.header }}
455
+ }
456
+ </th>
457
+ }
458
+ @if (!disabled && hasAnyActions()) {
459
+ <th class="ui-crud-table__th ui-crud-table__th--actions">
460
+ Azioni
461
+ </th>
462
+ }
463
+ </tr>
464
+ </thead>
465
+ <tbody>
466
+ @for (item of data; track trackByFn($index, item); let i = $index) {
467
+ <tr
468
+ [class]="getRowClasses(item, i)"
469
+ (click)="onRowClick(item)"
470
+ >
471
+ @for (col of columns; track col.key) {
472
+ <td
473
+ class="ui-crud-table__td"
474
+ [style.text-align]="col.align || 'left'"
475
+ >
476
+ @if (col.template) {
477
+ <ng-container
478
+ [ngTemplateOutlet]="col.template"
479
+ [ngTemplateOutletContext]="{
480
+ $implicit: item,
481
+ row: item,
482
+ column: col,
483
+ index: i,
484
+ value: item[col.key]
485
+ }"
486
+ />
487
+ } @else if (col.ellipsis && shouldShowEllipsis(col, item[col.key])) {
488
+ <span
489
+ class="ui-crud-table__ellipsis"
490
+ [matTooltip]="getFullText(item[col.key])"
491
+ >{{ truncateText(col, item[col.key]) }}</span>
492
+ } @else {
493
+ <span class="ui-crud-table__cell-text">{{ item[col.key] ?? '\u2014' }}</span>
494
+ }
495
+ </td>
496
+ }
497
+
498
+ <!-- Colonna azioni -->
499
+ @if (!disabled && hasAnyActions()) {
500
+ <td class="ui-crud-table__td ui-crud-table__td--actions">
501
+ <div class="ui-crud-table__actions">
502
+ @if (config?.allowView === true) {
503
+ <button
504
+ class="ui-crud-table__action-btn"
505
+ [class.ui-crud-table__action-btn--disabled]="!canViewItem(item)"
506
+ [matTooltip]="getViewTooltip(item)"
507
+ (click)="onViewBtnClick($event, item, i)"
508
+ type="button"
509
+ >
510
+ <lucide-icon name="eye" [size]="16" aria-hidden="true" />
511
+ </button>
512
+ }
513
+ @if (config?.allowEdit !== false) {
514
+ <button
515
+ class="ui-crud-table__action-btn"
516
+ [class.ui-crud-table__action-btn--disabled]="!canEditItem(item)"
517
+ [matTooltip]="getEditTooltip(item)"
518
+ (click)="onEditBtnClick($event, item, i)"
519
+ type="button"
520
+ >
521
+ <lucide-icon name="pencil" [size]="16" aria-hidden="true" />
522
+ </button>
523
+ }
524
+ @if (config?.allowDuplicate === true) {
525
+ <button
526
+ class="ui-crud-table__action-btn"
527
+ [class.ui-crud-table__action-btn--disabled]="!canDuplicateItem(item)"
528
+ [matTooltip]="getDuplicateTooltip(item)"
529
+ (click)="onDuplicateBtnClick($event, item)"
530
+ type="button"
531
+ >
532
+ <lucide-icon name="copy" [size]="16" aria-hidden="true" />
533
+ </button>
534
+ }
535
+ @if (config?.allowDelete !== false) {
536
+ <button
537
+ class="ui-crud-table__action-btn ui-crud-table__action-btn--warn"
538
+ [class.ui-crud-table__action-btn--disabled]="!canDeleteItem(item)"
539
+ [matTooltip]="getDeleteTooltip(item)"
540
+ (click)="onDeleteBtnClick($event, item, i)"
541
+ type="button"
542
+ >
543
+ <lucide-icon name="trash-2" [size]="16" aria-hidden="true" />
544
+ </button>
545
+ }
546
+ </div>
547
+ </td>
548
+ }
549
+ </tr>
550
+ }
551
+ </tbody>
552
+ </table>
553
+ </div>
554
+
555
+ <!-- Footer con pulsante aggiungi -->
556
+ <div class="ui-crud-table__footer">
557
+ <ui-button-area [buttons]="addButton" align="end" gap="xs" />
558
+ </div>
559
+ }
560
+
561
+ <!-- ═══ Vista form (insert/edit/duplicate/readonly) ══════ -->
562
+ @if (isFormMode) {
563
+ <div class="ui-crud-table__form">
564
+ <div class="ui-crud-table__form-header">
565
+ <h4 class="ui-crud-table__form-title">
566
+ @switch (currentMode) {
567
+ @case ('insert') { Nuovo elemento }
568
+ @case ('edit') { Modifica elemento }
569
+ @case ('duplicate') { Duplica elemento }
570
+ @case ('readonly') { Visualizza elemento }
571
+ }
572
+ </h4>
573
+ </div>
574
+ <div class="ui-crud-table__form-content">
575
+ <ui-form-builder
576
+ [schema]="config.formSchema"
577
+ [initialData]="formInitialData"
578
+ [readonly]="isReadonlyMode"
579
+ [buttonsOverride]="formButtons"
580
+ />
581
+ </div>
582
+ </div>
583
+ }
584
+ </div>
585
+ `, isInline: true, styles: [".ui-crud-table-host{display:block;width:100%}.ui-crud-table{display:flex;flex-direction:column;width:100%;font-family:var(--ui-font-family)}.ui-crud-table__empty{display:flex;justify-content:space-between;align-items:center;padding:var(--ui-spacing-5);border:1px dashed var(--ui-color-border);border-radius:var(--ui-radius-lg);background:var(--ui-color-bg-subtle);min-height:80px}.ui-crud-table__empty-message{display:flex;align-items:center;gap:var(--ui-spacing-2);color:var(--ui-color-text-muted);font-size:var(--ui-font-size-sm);font-style:italic}.ui-crud-table__empty-icon{color:var(--ui-color-text-muted);flex-shrink:0}.ui-crud-table__loading{display:flex;align-items:center;justify-content:center;min-height:120px;padding:var(--ui-spacing-6);border:1px dashed var(--ui-color-border);border-radius:var(--ui-radius-lg);background:var(--ui-color-bg-subtle)}.ui-crud-table__spinner{width:24px;height:24px;border:3px solid var(--ui-color-primary-light);border-top-color:var(--ui-color-primary);border-radius:50%;animation:ui-crud-table-spin .7s linear infinite}@keyframes ui-crud-table-spin{to{transform:rotate(360deg)}}.ui-crud-table__sr-only{position:absolute;width:1px;height:1px;padding:0;margin:-1px;overflow:hidden;clip:rect(0,0,0,0);white-space:nowrap;border:0}.ui-crud-table__container{border:1px solid var(--ui-color-border);border-radius:var(--ui-radius-lg);overflow:hidden;overflow-x:auto;box-shadow:var(--ui-shadow-sm)}.ui-crud-table__table{width:100%;border-collapse:collapse;font-size:var(--ui-font-size-sm)}.ui-crud-table__th{padding:var(--ui-spacing-3) var(--ui-spacing-4);border-bottom:2px solid var(--ui-color-border);background:var(--ui-color-bg-subtle);font-weight:var(--ui-font-weight-semibold);color:var(--ui-color-text-secondary);text-align:left;white-space:nowrap;-webkit-user-select:none;user-select:none;vertical-align:middle}.ui-crud-table__th--actions{text-align:right;width:140px;min-width:140px}.ui-crud-table__row{transition-property:background-color;transition-duration:var(--ui-transition-fast);transition-timing-function:ease}.ui-crud-table__row:hover{background:var(--ui-color-surface-hover)}.ui-crud-table__row:last-child .ui-crud-table__td{border-bottom:none}.ui-crud-table__td{padding:var(--ui-spacing-3) var(--ui-spacing-4);border-bottom:1px solid var(--ui-color-border);color:var(--ui-color-text);vertical-align:middle;word-break:break-word}.ui-crud-table__td--actions{text-align:right;white-space:nowrap}.ui-crud-table__cell-text{color:var(--ui-color-text)}.ui-crud-table__ellipsis{cursor:help}.ui-crud-table__ellipsis:hover{text-decoration:underline;text-decoration-style:dotted;text-underline-offset:2px}.ui-crud-table__actions{display:inline-flex;gap:var(--ui-spacing-1);align-items:center;justify-content:flex-end}.ui-crud-table__action-btn{appearance:none;border:none;background:transparent;cursor:pointer;display:inline-flex;align-items:center;justify-content:center;width:32px;height:32px;border-radius:var(--ui-radius-md);color:var(--ui-color-primary);transition-property:\"background-color, color, transform, opacity\";transition-duration:var(--ui-transition-fast);transition-timing-function:ease}.ui-crud-table__action-btn:focus{outline:none}.ui-crud-table__action-btn:focus-visible{outline:var(--ui-focus-ring-width) solid var(--ui-focus-ring-color);outline-offset:var(--ui-focus-ring-offset)}.ui-crud-table__action-btn:hover:not(.ui-crud-table__action-btn--disabled){background:var(--ui-color-primary-light);transform:scale(1.08)}.ui-crud-table__action-btn--warn{color:var(--ui-color-warn)}.ui-crud-table__action-btn--warn:hover:not(.ui-crud-table__action-btn--disabled){background:var(--ui-color-warn-light)}.ui-crud-table__action-btn--disabled{opacity:.35;cursor:not-allowed;color:var(--ui-color-text-muted)!important}.ui-crud-table__action-btn--disabled:hover{transform:none;background:transparent}.ui-crud-table__footer{margin-top:var(--ui-spacing-4)}.ui-crud-table__form{border:1px solid var(--ui-color-border);border-radius:var(--ui-radius-lg);background:var(--ui-color-bg-subtle);box-shadow:var(--ui-shadow-sm);overflow:hidden}.ui-crud-table__form-header{padding:var(--ui-spacing-4) var(--ui-spacing-5);border-bottom:1px solid var(--ui-color-border);background:var(--ui-color-surface)}.ui-crud-table__form-title{margin:0;font-size:var(--ui-font-size-lg);font-weight:var(--ui-font-weight-semibold);color:var(--ui-color-text)}.ui-crud-table__form-content{padding:var(--ui-spacing-5)}@media (max-width: 768px){.ui-crud-table__th,.ui-crud-table__td{padding:var(--ui-spacing-2) var(--ui-spacing-3);font-size:var(--ui-font-size-xs)}.ui-crud-table__th--actions{width:auto;min-width:auto}.ui-crud-table__empty{flex-direction:column;gap:var(--ui-spacing-3);text-align:center}.ui-crud-table__form-content{padding:var(--ui-spacing-3)}}\n"], dependencies: [{ kind: "directive", type: NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "ngmodule", type: MatTooltipModule }, { kind: "directive", type: i1.MatTooltip, selector: "[matTooltip]", inputs: ["matTooltipPosition", "matTooltipPositionAtOrigin", "matTooltipDisabled", "matTooltipShowDelay", "matTooltipHideDelay", "matTooltipTouchGestures", "matTooltip", "matTooltipClass"], exportAs: ["matTooltip"] }, { kind: "ngmodule", type: LucideAngularModule }, { kind: "component", type: i2.LucideAngularComponent, selector: "lucide-angular, lucide-icon, i-lucide, span-lucide", inputs: ["class", "name", "img", "color", "absoluteStrokeWidth", "size", "strokeWidth"] }, { kind: "component", type: UiButtonAreaComponent, selector: "ui-button-area", inputs: ["buttons", "align", "ariaLabel", "gap", "stackOnMobile", "disableWhileLoading", "loadingIds"] }, { kind: "component", type: UiFormBuilderComponent, selector: "ui-form-builder", inputs: ["schema", "initialData", "readonly", "disabled", "buttonsOverride", "loadingFor"], outputs: ["valueChange", "validationChange", "formSubmit", "formReset", "customEvent"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush, encapsulation: i0.ViewEncapsulation.None }); }
586
+ }
587
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: UiCrudTableComponent, decorators: [{
588
+ type: Component,
589
+ args: [{ selector: 'ui-crud-table', standalone: true, imports: [NgTemplateOutlet, MatTooltipModule, LucideAngularModule, UiButtonAreaComponent, UiFormBuilderComponent], changeDetection: ChangeDetectionStrategy.OnPush, encapsulation: ViewEncapsulation.None, host: { class: 'ui-crud-table-host' }, template: `
590
+ <div class="ui-crud-table">
591
+
592
+ <!-- ═══ Stato vuoto ═══════════════════════════════════════ -->
593
+ @if (currentMode === 'view' && (!data || data.length === 0) && !loading) {
594
+ <div class="ui-crud-table__empty">
595
+ <div class="ui-crud-table__empty-message">
596
+ <lucide-icon name="info" [size]="20" class="ui-crud-table__empty-icon" aria-hidden="true" />
597
+ <span>{{ config?.emptyMessage || 'Nessun elemento presente' }}</span>
598
+ </div>
599
+ <ui-button-area [buttons]="addButton" align="end" gap="xs" />
600
+ </div>
601
+ }
602
+
603
+ <!-- ═══ Stato di caricamento ══════════════════════════════ -->
604
+ @if (loading && currentMode === 'view') {
605
+ <div class="ui-crud-table__loading" role="status" aria-live="polite">
606
+ <span class="ui-crud-table__spinner" aria-hidden="true"></span>
607
+ <span class="ui-crud-table__sr-only">Caricamento in corso</span>
608
+ </div>
609
+ }
610
+
611
+ <!-- ═══ Vista tabella ═════════════════════════════════════ -->
612
+ @if (currentMode === 'view' && data && data.length > 0 && !loading) {
613
+ <div class="ui-crud-table__container">
614
+ <table class="ui-crud-table__table" role="grid" [attr.aria-label]="config?.tableLabel || 'Tabella CRUD'">
615
+ <thead>
616
+ <tr>
617
+ @for (col of columns; track col.key) {
618
+ <th
619
+ class="ui-crud-table__th"
620
+ [style.width]="col.width || null"
621
+ [style.text-align]="col.align || 'left'"
622
+ >
623
+ @if (col.headerTemplate) {
624
+ <ng-container
625
+ [ngTemplateOutlet]="col.headerTemplate"
626
+ [ngTemplateOutletContext]="{ $implicit: col, column: col }"
627
+ />
628
+ } @else {
629
+ {{ col.header }}
630
+ }
631
+ </th>
632
+ }
633
+ @if (!disabled && hasAnyActions()) {
634
+ <th class="ui-crud-table__th ui-crud-table__th--actions">
635
+ Azioni
636
+ </th>
637
+ }
638
+ </tr>
639
+ </thead>
640
+ <tbody>
641
+ @for (item of data; track trackByFn($index, item); let i = $index) {
642
+ <tr
643
+ [class]="getRowClasses(item, i)"
644
+ (click)="onRowClick(item)"
645
+ >
646
+ @for (col of columns; track col.key) {
647
+ <td
648
+ class="ui-crud-table__td"
649
+ [style.text-align]="col.align || 'left'"
650
+ >
651
+ @if (col.template) {
652
+ <ng-container
653
+ [ngTemplateOutlet]="col.template"
654
+ [ngTemplateOutletContext]="{
655
+ $implicit: item,
656
+ row: item,
657
+ column: col,
658
+ index: i,
659
+ value: item[col.key]
660
+ }"
661
+ />
662
+ } @else if (col.ellipsis && shouldShowEllipsis(col, item[col.key])) {
663
+ <span
664
+ class="ui-crud-table__ellipsis"
665
+ [matTooltip]="getFullText(item[col.key])"
666
+ >{{ truncateText(col, item[col.key]) }}</span>
667
+ } @else {
668
+ <span class="ui-crud-table__cell-text">{{ item[col.key] ?? '\u2014' }}</span>
669
+ }
670
+ </td>
671
+ }
672
+
673
+ <!-- Colonna azioni -->
674
+ @if (!disabled && hasAnyActions()) {
675
+ <td class="ui-crud-table__td ui-crud-table__td--actions">
676
+ <div class="ui-crud-table__actions">
677
+ @if (config?.allowView === true) {
678
+ <button
679
+ class="ui-crud-table__action-btn"
680
+ [class.ui-crud-table__action-btn--disabled]="!canViewItem(item)"
681
+ [matTooltip]="getViewTooltip(item)"
682
+ (click)="onViewBtnClick($event, item, i)"
683
+ type="button"
684
+ >
685
+ <lucide-icon name="eye" [size]="16" aria-hidden="true" />
686
+ </button>
687
+ }
688
+ @if (config?.allowEdit !== false) {
689
+ <button
690
+ class="ui-crud-table__action-btn"
691
+ [class.ui-crud-table__action-btn--disabled]="!canEditItem(item)"
692
+ [matTooltip]="getEditTooltip(item)"
693
+ (click)="onEditBtnClick($event, item, i)"
694
+ type="button"
695
+ >
696
+ <lucide-icon name="pencil" [size]="16" aria-hidden="true" />
697
+ </button>
698
+ }
699
+ @if (config?.allowDuplicate === true) {
700
+ <button
701
+ class="ui-crud-table__action-btn"
702
+ [class.ui-crud-table__action-btn--disabled]="!canDuplicateItem(item)"
703
+ [matTooltip]="getDuplicateTooltip(item)"
704
+ (click)="onDuplicateBtnClick($event, item)"
705
+ type="button"
706
+ >
707
+ <lucide-icon name="copy" [size]="16" aria-hidden="true" />
708
+ </button>
709
+ }
710
+ @if (config?.allowDelete !== false) {
711
+ <button
712
+ class="ui-crud-table__action-btn ui-crud-table__action-btn--warn"
713
+ [class.ui-crud-table__action-btn--disabled]="!canDeleteItem(item)"
714
+ [matTooltip]="getDeleteTooltip(item)"
715
+ (click)="onDeleteBtnClick($event, item, i)"
716
+ type="button"
717
+ >
718
+ <lucide-icon name="trash-2" [size]="16" aria-hidden="true" />
719
+ </button>
720
+ }
721
+ </div>
722
+ </td>
723
+ }
724
+ </tr>
725
+ }
726
+ </tbody>
727
+ </table>
728
+ </div>
729
+
730
+ <!-- Footer con pulsante aggiungi -->
731
+ <div class="ui-crud-table__footer">
732
+ <ui-button-area [buttons]="addButton" align="end" gap="xs" />
733
+ </div>
734
+ }
735
+
736
+ <!-- ═══ Vista form (insert/edit/duplicate/readonly) ══════ -->
737
+ @if (isFormMode) {
738
+ <div class="ui-crud-table__form">
739
+ <div class="ui-crud-table__form-header">
740
+ <h4 class="ui-crud-table__form-title">
741
+ @switch (currentMode) {
742
+ @case ('insert') { Nuovo elemento }
743
+ @case ('edit') { Modifica elemento }
744
+ @case ('duplicate') { Duplica elemento }
745
+ @case ('readonly') { Visualizza elemento }
746
+ }
747
+ </h4>
748
+ </div>
749
+ <div class="ui-crud-table__form-content">
750
+ <ui-form-builder
751
+ [schema]="config.formSchema"
752
+ [initialData]="formInitialData"
753
+ [readonly]="isReadonlyMode"
754
+ [buttonsOverride]="formButtons"
755
+ />
756
+ </div>
757
+ </div>
758
+ }
759
+ </div>
760
+ `, styles: [".ui-crud-table-host{display:block;width:100%}.ui-crud-table{display:flex;flex-direction:column;width:100%;font-family:var(--ui-font-family)}.ui-crud-table__empty{display:flex;justify-content:space-between;align-items:center;padding:var(--ui-spacing-5);border:1px dashed var(--ui-color-border);border-radius:var(--ui-radius-lg);background:var(--ui-color-bg-subtle);min-height:80px}.ui-crud-table__empty-message{display:flex;align-items:center;gap:var(--ui-spacing-2);color:var(--ui-color-text-muted);font-size:var(--ui-font-size-sm);font-style:italic}.ui-crud-table__empty-icon{color:var(--ui-color-text-muted);flex-shrink:0}.ui-crud-table__loading{display:flex;align-items:center;justify-content:center;min-height:120px;padding:var(--ui-spacing-6);border:1px dashed var(--ui-color-border);border-radius:var(--ui-radius-lg);background:var(--ui-color-bg-subtle)}.ui-crud-table__spinner{width:24px;height:24px;border:3px solid var(--ui-color-primary-light);border-top-color:var(--ui-color-primary);border-radius:50%;animation:ui-crud-table-spin .7s linear infinite}@keyframes ui-crud-table-spin{to{transform:rotate(360deg)}}.ui-crud-table__sr-only{position:absolute;width:1px;height:1px;padding:0;margin:-1px;overflow:hidden;clip:rect(0,0,0,0);white-space:nowrap;border:0}.ui-crud-table__container{border:1px solid var(--ui-color-border);border-radius:var(--ui-radius-lg);overflow:hidden;overflow-x:auto;box-shadow:var(--ui-shadow-sm)}.ui-crud-table__table{width:100%;border-collapse:collapse;font-size:var(--ui-font-size-sm)}.ui-crud-table__th{padding:var(--ui-spacing-3) var(--ui-spacing-4);border-bottom:2px solid var(--ui-color-border);background:var(--ui-color-bg-subtle);font-weight:var(--ui-font-weight-semibold);color:var(--ui-color-text-secondary);text-align:left;white-space:nowrap;-webkit-user-select:none;user-select:none;vertical-align:middle}.ui-crud-table__th--actions{text-align:right;width:140px;min-width:140px}.ui-crud-table__row{transition-property:background-color;transition-duration:var(--ui-transition-fast);transition-timing-function:ease}.ui-crud-table__row:hover{background:var(--ui-color-surface-hover)}.ui-crud-table__row:last-child .ui-crud-table__td{border-bottom:none}.ui-crud-table__td{padding:var(--ui-spacing-3) var(--ui-spacing-4);border-bottom:1px solid var(--ui-color-border);color:var(--ui-color-text);vertical-align:middle;word-break:break-word}.ui-crud-table__td--actions{text-align:right;white-space:nowrap}.ui-crud-table__cell-text{color:var(--ui-color-text)}.ui-crud-table__ellipsis{cursor:help}.ui-crud-table__ellipsis:hover{text-decoration:underline;text-decoration-style:dotted;text-underline-offset:2px}.ui-crud-table__actions{display:inline-flex;gap:var(--ui-spacing-1);align-items:center;justify-content:flex-end}.ui-crud-table__action-btn{appearance:none;border:none;background:transparent;cursor:pointer;display:inline-flex;align-items:center;justify-content:center;width:32px;height:32px;border-radius:var(--ui-radius-md);color:var(--ui-color-primary);transition-property:\"background-color, color, transform, opacity\";transition-duration:var(--ui-transition-fast);transition-timing-function:ease}.ui-crud-table__action-btn:focus{outline:none}.ui-crud-table__action-btn:focus-visible{outline:var(--ui-focus-ring-width) solid var(--ui-focus-ring-color);outline-offset:var(--ui-focus-ring-offset)}.ui-crud-table__action-btn:hover:not(.ui-crud-table__action-btn--disabled){background:var(--ui-color-primary-light);transform:scale(1.08)}.ui-crud-table__action-btn--warn{color:var(--ui-color-warn)}.ui-crud-table__action-btn--warn:hover:not(.ui-crud-table__action-btn--disabled){background:var(--ui-color-warn-light)}.ui-crud-table__action-btn--disabled{opacity:.35;cursor:not-allowed;color:var(--ui-color-text-muted)!important}.ui-crud-table__action-btn--disabled:hover{transform:none;background:transparent}.ui-crud-table__footer{margin-top:var(--ui-spacing-4)}.ui-crud-table__form{border:1px solid var(--ui-color-border);border-radius:var(--ui-radius-lg);background:var(--ui-color-bg-subtle);box-shadow:var(--ui-shadow-sm);overflow:hidden}.ui-crud-table__form-header{padding:var(--ui-spacing-4) var(--ui-spacing-5);border-bottom:1px solid var(--ui-color-border);background:var(--ui-color-surface)}.ui-crud-table__form-title{margin:0;font-size:var(--ui-font-size-lg);font-weight:var(--ui-font-weight-semibold);color:var(--ui-color-text)}.ui-crud-table__form-content{padding:var(--ui-spacing-5)}@media (max-width: 768px){.ui-crud-table__th,.ui-crud-table__td{padding:var(--ui-spacing-2) var(--ui-spacing-3);font-size:var(--ui-font-size-xs)}.ui-crud-table__th--actions{width:auto;min-width:auto}.ui-crud-table__empty{flex-direction:column;gap:var(--ui-spacing-3);text-align:center}.ui-crud-table__form-content{padding:var(--ui-spacing-3)}}\n"] }]
761
+ }], propDecorators: { data: [{
762
+ type: Input
763
+ }], columns: [{
764
+ type: Input
765
+ }], config: [{
766
+ type: Input
767
+ }], loading: [{
768
+ type: Input
769
+ }], disabled: [{
770
+ type: Input
771
+ }], initMapper: [{
772
+ type: Input
773
+ }], dataChange: [{
774
+ type: Output
775
+ }], itemAdd: [{
776
+ type: Output
777
+ }], itemEdit: [{
778
+ type: Output
779
+ }], itemDelete: [{
780
+ type: Output
781
+ }], itemDuplicate: [{
782
+ type: Output
783
+ }], rowClick: [{
784
+ type: Output
785
+ }], formBuilder: [{
786
+ type: ViewChild,
787
+ args: [UiFormBuilderComponent]
788
+ }] } });
789
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY3J1ZC10YWJsZS5jb21wb25lbnQuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9wYWNrYWdlcy9uZy11aS1zeXN0ZW0vc3JjL2xpYi9jb21wb25lbnRzL2NydWQtdGFibGUvY3J1ZC10YWJsZS5jb21wb25lbnQudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUNMLFNBQVMsRUFDVCxLQUFLLEVBQ0wsTUFBTSxFQUNOLFlBQVksRUFDWixTQUFTLEVBQ1QsdUJBQXVCLEVBQ3ZCLGlCQUFpQixFQUNqQixpQkFBaUIsRUFHakIsTUFBTSxHQUNQLE1BQU0sZUFBZSxDQUFDO0FBQ3ZCLE9BQU8sRUFBRSxnQkFBZ0IsRUFBRSxNQUFNLGlCQUFpQixDQUFDO0FBQ25ELE9BQU8sRUFBRSxnQkFBZ0IsRUFBRSxNQUFNLDJCQUEyQixDQUFDO0FBQzdELE9BQU8sRUFBRSxtQkFBbUIsRUFBRSxNQUFNLGdCQUFnQixDQUFDO0FBQ3JELE9BQU8sRUFBRSxPQUFPLEVBQUUsTUFBTSxNQUFNLENBQUM7QUFFL0IsT0FBTyxFQUFFLHFCQUFxQixFQUFFLE1BQU0saUNBQWlDLENBQUM7QUFFeEUsT0FBTyxFQUFFLHNCQUFzQixFQUFFLE1BQU0sd0NBQXdDLENBQUM7QUFDaEYsT0FBTyxFQUFFLGNBQWMsRUFBRSxNQUFNLHdCQUF3QixDQUFDOzs7O0FBV3hEOzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7O0dBeUJHO0FBc0xILE1BQU0sT0FBTyxvQkFBb0I7SUFyTGpDO1FBc0xtQixRQUFHLEdBQUcsTUFBTSxDQUFDLGlCQUFpQixDQUFDLENBQUM7UUFDaEMsaUJBQVksR0FBRyxNQUFNLENBQUMsY0FBYyxDQUFDLENBQUM7UUFDdEMsYUFBUSxHQUFHLElBQUksT0FBTyxFQUFRLENBQUM7UUFFaEQsbUVBQW1FO1FBRW5FLG1EQUFtRDtRQUMxQyxTQUFJLEdBQVEsRUFBRSxDQUFDO1FBRXhCLGlDQUFpQztRQUN4QixZQUFPLEdB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TUFBTTtzQkFBZCxLQUFLO2dCQUdHLE9BQU87c0JBQWYsS0FBSztnQkFHRyxRQUFRO3NCQUFoQixLQUFLO2dCQU9HLFVBQVU7c0JBQWxCLEtBQUs7Z0JBS0ksVUFBVTtzQkFBbkIsTUFBTTtnQkFHRyxPQUFPO3NCQUFoQixNQUFNO2dCQUdHLFFBQVE7c0JBQWpCLE1BQU07Z0JBR0csVUFBVTtzQkFBbkIsTUFBTTtnQkFHRyxhQUFhO3NCQUF0QixNQUFNO2dCQUdHLFFBQVE7c0JBQWpCLE1BQU07Z0JBcUI0QixXQUFXO3NCQUE3QyxTQUFTO3VCQUFDLHNCQUFzQiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7XHJcbiAgQ29tcG9uZW50LFxyXG4gIElucHV0LFxyXG4gIE91dHB1dCxcclxuICBFdmVudEVtaXR0ZXIsXHJcbiAgVmlld0NoaWxkLFxyXG4gIENoYW5nZURldGVjdGlvblN0cmF0ZWd5LFxyXG4gIENoYW5nZURldGVjdG9yUmVmLFxyXG4gIFZpZXdFbmNhcHN1bGF0aW9uLFxyXG4gIE9uSW5pdCxcclxuICBPbkRlc3Ryb3ksXHJcbiAgaW5qZWN0LFxyXG59IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xyXG5pbXBvcnQgeyBOZ1RlbXBsYXRlT3V0bGV0IH0gZnJvbSAnQGFuZ3VsYXIvY29tbW9uJztcclxuaW1wb3J0IHsgTWF0VG9vbHRpcE1vZHVsZSB9IGZyb20gJ0Bhbmd1bGFyL21hdGVyaWFsL3Rvb2x0aXAnO1xyXG5pbXBvcnQgeyBMdWNpZGVBbmd1bGFyTW9kdWxlIH0gZnJvbSAnbHVjaWRlLWFuZ3VsYXInO1xyXG5pbXBvcnQgeyBTdWJqZWN0IH0gZnJvbSAncnhqcyc7XHJcblxyXG5pbXBvcnQgeyBVaUJ1dHRvbkFyZWFDb21wb25lbnQgfSBmcm9tICcuLi9idXR0b24vYnV0dG9uLWFyZWEuY29tcG9uZW50JztcclxuaW1wb3J0IHsgVWlCdXR0b25EZXNjcmlwdG9yIH0gZnJvbSAnLi4vYnV0dG9uL2J1dHRvbi50eXBlcyc7XHJcbmltcG9ydCB7IFVpRm9ybUJ1aWxkZXJDb21wb25lbnQgfSBmcm9tICcuLi9mb3JtLWJ1aWxkZXIvZm9ybS1idWlsZGVyLmNvbXBvbmVudCc7XHJcbmltcG9ydCB7IFVpTW9kYWxTZXJ2aWNlIH0gZnJvbSAnLi4vbW9kYWwvbW9kYWwuc2VydmljZSc7XHJcbmltcG9ydCB7IFVpRm9ybURhdGEgfSBmcm9tICcuLi9mb3JtLWJ1aWxkZXIvdHlwZXMvc2NoZW1hLnR5cGVzJztcclxuXHJcbmltcG9ydCB7XHJcbiAgVWlDcnVkVGFibGVDb2x1bW4sXHJcbiAgVWlDcnVkVGFibGVDb25maWcsXHJcbiAgVWlDcnVkTW9kZSxcclxuICBVaUNydWRFZGl0RXZlbnQsXHJcbiAgVWlDcnVkRGVsZXRlRXZlbnQsXHJcbn0gZnJvbSAnLi9jcnVkLXRhYmxlLnR5cGVzJztcclxuXHJcbi8qKlxyXG4gKiBUYWJlbGxhIENSVUQgc3RhbmRhbG9uZSBjb24gZm9ybSBidWlsZGVyIGludGVncmF0by5cclxuICpcclxuICogR2VzdGlzY2UgaWwgY2ljbG8gY29tcGxldG8gZGVsbGUgb3BlcmF6aW9uaSBDUlVEOlxyXG4gKiBpbnNlcmltZW50bywgbW9kaWZpY2EsIGR1cGxpY2F6aW9uZSwgZWxpbWluYXppb25lIGVcclxuICogdmlzdWFsaXp6YXppb25lIGluIHNvbGEgbGV0dHVyYS4gSWwgZm9ybSBlIGdlbmVyYXRvXHJcbiAqIGF1dG9tYXRpY2FtZW50ZSBkYWxsbyBzY2hlbWEgYFVpRm9ybVNjaGVtYWAuXHJcbiAqXHJcbiAqIFV0aWxpenphIGBVaU1vZGFsU2VydmljZWAgcGVyIGxhIGNvbmZlcm1hIGRpIGVsaW1pbmF6aW9uZVxyXG4gKiBlIGBVaUZvcm1CdWlsZGVyQ29tcG9uZW50YCBwZXIgbGEgZ2VzdGlvbmUgZGVsIGZvcm0uXHJcbiAqXHJcbiAqIEBzZWxlY3RvciB1aS1jcnVkLXRhYmxlXHJcbiAqXHJcbiAqIEBleGFtcGxlXHJcbiAqIGBgYGh0bWxcclxuICogPHVpLWNydWQtdGFibGVcclxuICogICBbZGF0YV09XCJpdGVtc1wiXHJcbiAqICAgW2NvbHVtbnNdPVwiY29sdW1uc1wiXHJcbiAqICAgW2NvbmZpZ109XCJjcnVkQ29uZmlnXCJcclxuICogICAoZGF0YUNoYW5nZSk9XCJvbkRhdGFDaGFuZ2UoJGV2ZW50KVwiXHJcbiAqICAgKGl0ZW1BZGQpPVwib25BZGQoJGV2ZW50KVwiXHJcbiAqICAgKGl0ZW1FZGl0KT1cIm9uRWRpdCgkZXZlbnQpXCJcclxuICogICAoaXRlbURlbGV0ZSk9XCJvbkRlbGV0ZSgkZXZlbnQpXCJcclxuICogLz5cclxuICogYGBgXHJcbiAqL1xyXG5AQ29tcG9uZW50KHtcclxuICBzZWxlY3RvcjogJ3VpLWNydWQtdGFibGUnLFxyXG4gIHN0YW5kYWxvbmU6IHRydWUsXHJcbiAgaW1wb3J0czogW05nVGVtcGxhdGVPdXRsZXQsIE1hdFRvb2x0aXBNb2R1bGUsIEx1Y2lkZUFuZ3VsYXJNb2R1bGUsIFVpQnV0dG9uQXJlYUNvbXBvbmVudCwgVWlGb3JtQnVpbGRlckNvbXBvbmVudF0sXHJcbiAgY2hhbmdlRGV0ZWN0aW9uOiBDaGFuZ2VEZXRlY3Rpb25TdHJhdGVneS5PblB1c2gsXHJcbiAgZW5jYXBzdWxhdGlvbjogVmlld0VuY2Fwc3VsYXRpb24uTm9uZSxcclxuICBob3N0OiB7IGNsYXNzOiAndWktY3J1ZC10YWJsZS1ob3N0JyB9LFxyXG4gIHRlbXBsYXRlOiBgXHJcbiAgICA8ZGl2IGNsYXNzPVwidWktY3J1ZC10YWJsZVwiPlxyXG5cclxuICAgICAgPCEtLSDilZDilZDilZAgU3RhdG8gdnVvdG8g4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQ4pWQIC0tPlxyXG4gICAgICBAaWYgKGN1cnJlbnRNb2RlID09PSAndmlldycgJiYgKCFkYXRhIHx8IGRhdGEubGVuZ3RoID09PSAwKSAmJiAhbG9hZGluZykge1xyXG4gICAgICAgIDxkaXYgY2xhc3M9XCJ1aS1jcnVkLXRhYmxlX19lbXB0eVwiPlxyXG4gICAgICAgICAgPGRpdiBjbGFzcz1cInVpLWNydWQtdGFibGVfX2VtcHR5LW1lc3NhZ2VcIj5cclxuICAgICAgICAgICAgPGx1Y2lkZS1pY29uIG5hbWU9XCJpbmZvXCIgW3NpemVdPVwiMjBcIiBjbGFzcz1cInVpLWNydWQtdGFibGVfX2VtcHR5LWljb25cIiBhcmlhLWhpZGRlbj1cInRydWVcIiAvPlxyXG4gICAgICAgICAgICA8c3Bhbj57eyBjb25maWc/LmVtcHR5TWVzc2FnZSB8fCAnTmVzc3VuIGVsZW1lbnRvIHByZXNlbnRlJyB9fTwvc3Bhbj5cclxuICAgICAgICAgIDwvZGl2PlxyXG4gICAgICAgICAgPHVpLWJ1dHRvbi1hcmVhIFtidXR0b25zXT1cImFkZEJ1dHRvblwiIGFsaWduPVwiZW5kXCIgZ2FwPVwieHNcIiAvPlxyXG4gICAgICAgIDwvZGl2PlxyXG4gICAgICB9XHJcblxyXG4gICAgICA8IS0tIOKVkOKVkOKVkCBTdGF0byBkaSBjYXJpY2FtZW50byDilZDilZDilZDilZDilZDilZDilZDilZDilZDilZDilZDilZDilZDilZDilZDilZDilZDilZDilZDilZDilZDilZDilZDilZDilZDilZDilZDilZDilZDilZAgLS0+XHJcbiAgICAgIEBpZiAobG9hZGluZyAmJiBjdXJyZW50TW9kZSA9PT0gJ3ZpZXcnKSB7XHJcbiAgICAgICAgPGRpdiBjbGFzcz1cInVpLWNydWQtdGFibGVfX2xvYWRpbmdcIiByb2xlPVwic3RhdHVzXCIgYXJpYS1saXZlPVwicG9saXRlXCI+XHJcbiAgICAgICAgICA8c3BhbiBjbGFzcz1cInVpLWNydWQtdGFibGVfX3NwaW5uZXJcIiBhcmlhLWhpZGRlbj1cInRydWVcIj48L3NwYW4+XHJcbiAgICAgICAgICA8c3BhbiBjbGFzcz1cInVpLWNydWQtdGFibGVfX3NyLW9ubHlcIj5DYXJpY2FtZW50byBpbiBjb3Jzbzwvc3Bhbj5cclxuICAgICAgICA8L2Rpdj5cclxuICAgICAgfVxyXG5cclxuICAgICAgPCEtLSDilZDilZDilZAgVmlzdGEgdGFiZWxsYSDilZDilZDilZDilZDilZDilZDilZDilZDilZDilZDilZDilZDilZDilZDilZDilZDilZDilZDilZDilZDilZDilZDilZDilZDilZDilZDilZDilZDilZDilZDilZDilZDilZDilZDilZDilZDilZAgLS0+XHJcbiAgICAgIEBpZiAoY3VycmVudE1vZGUgPT09ICd2aWV3JyAmJiBkYXRhICYmIGRhdGEubGVuZ3RoID4gMCAmJiAhbG9hZGluZykge1xyXG4gICAgICAgIDxkaXYgY2xhc3M9XCJ1aS1jcnVkLXRhYmxlX19jb250YWluZXJcIj5cclxuICAgICAgICAgIDx0YWJsZSBjbGFzcz1cInVpLWNydWQtdGFibGVfX3RhYmxlXCIgcm9sZT1cImdyaWRcIiBbYXR0ci5hcmlhLWxhYmVsXT1cImNvbmZpZz8udGFibGVMYWJlbCB8fCAnVGFiZWxsYSBDUlVEJ1wiPlxyXG4gICAgICAgICAgICA8dGhlYWQ+XHJcbiAgICAgICAgICAgICAgPHRyPlxyXG4gICAgICAgICAgICAgICAgQGZvciAoY29sIG9mIGNvbHVtbnM7IHRyYWNrIGNvbC5rZXkpIHtcclxuICAgICAgICAgICAgICAgICAgPHRoXHJcbiAgICAgICAgICAgICAgICAgICAgY2xhc3M9XCJ1aS1jcnVkLXRhYmxlX190aFwiXHJcbiAgICAgICAgICAgICAgICAgICAgW3N0eWxlLndpZHRoXT1cImNvbC53aWR0aCB8fCBudWxsXCJcclxuICAgICAgICAgICAgICAgICAgICBbc3R5bGUudGV4dC1hbGlnbl09XCJjb2wuYWxpZ24gfHwgJ2xlZnQnXCJcclxuICAgICAgICAgICAgICAgICAgPlxyXG4gICAgICAgICAgICAgICAgICAgIEBpZiAoY29sLmhlYWRlclRlbXBsYXRlKSB7XHJcbiAgICAgICAgICAgICAgICAgICAgICA8bmctY29udGFpbmVyXHJcbiAgICAgICAgICAgICAgICAgICAgICAgIFtuZ1RlbXBsYXRlT3V0bGV0XT1cImNvbC5oZWFkZXJUZW1wbGF0ZVwiXHJcbiAgICAgICAgICAgICAgICAgICAgICAgIFtuZ1RlbXBsYXRlT3V0bGV0Q29udGV4dF09XCJ7ICRpbXBsaWNpdDogY29sLCBjb2x1bW46IGNvbCB9XCJcclxuICAgICAgICAgICAgICAgICAgICAgIC8+XHJcbiAgICAgICAgICAgICAgICAgICAgfSBAZWxzZSB7XHJcbiAgICAgICAgICAgICAgICAgICAgICB7eyBjb2wuaGVhZGVyIH19XHJcbiAgICAgICAgICAgICAgICAgICAgfVxyXG4gICAgICAgICAgICAgICAgICA8L3RoPlxyXG4gICAgICAgICAgICAgICAgfVxyXG4gICAgICAgICAgICAgICAgQGlmICghZGlzYWJsZWQgJiYgaGFzQW55QWN0aW9ucygpKSB7XHJcbiAgICAgICAgICAgICAgICAgIDx0aCBjbGFzcz1cInVpLWNydWQtdGFibGVfX3RoIHVpLWNydWQtdGFibGVfX3RoLS1hY3Rpb25zXCI+XHJcbiAgICAgICAgICAgICAgICAgICAgQXppb25pXHJcbiAgICAgICAgICAgICAgICAgIDwvdGg+XHJcbiAgICAgICAgICAgICAgICB9XHJcbiAgICAgICAgICAgICAgPC90cj5cclxuICAgICAgICAgICAgPC90aGVhZD5cclxuICAgICAgICAgICAgPHRib2R5PlxyXG4gICAgICAgICAgICAgIEBmb3IgKGl0ZW0gb2YgZGF0YTsgdHJhY2sgdHJhY2tCeUZuKCRpbmRleCwgaXRlbSk7IGxldCBpID0gJGluZGV4KSB7XHJcbiAgICAgICAgICAgICAgICA8dHJcclxuICAgICAgICAgICAgICAgICAgW2NsYXNzXT1cImdldFJvd0NsYXNzZXMoaXRlbSwgaSlcIlxyXG4gICAgICAgICAgICAgICAgICAoY2xpY2spPVwib25Sb3dDbGljayhpdGVtKVwiXHJcbiAgICAgICAgICAgICAgICA+XHJcbiAgICAgICAgICAgICAgICAgIEBmb3IgKGNvbCBvZiBjb2x1bW5zOyB0cmFjayBjb2wua2V5KSB7XHJcbiAgICAgICAgICAgICAgICAgICAgPHRkXHJcbiAgICAgICAgICAgICAgICAgICAgICBjbGFzcz1cInVpLWNydWQtdGFibGVfX3RkXCJcclxuICAgICAgICAgICAgICAgICAgICAgIFtzdHlsZS50ZXh0LWFsaWduXT1cImNvbC5hbGlnbiB8fCAnbGVmdCdcIlxyXG4gICAgICAgICAgICAgICAgICAgID5cclxuICAgICAgICAgICAgICAgICAgICAgIEBpZiAoY29sLnRlbXBsYXRlKSB7XHJcbiAgICAgICAgICAgICAgICAgICAgICAgIDxuZy1jb250YWluZXJcclxuICAgICAgICAgICAgICAgICAgICAgICAgICBbbmdUZW1wbGF0ZU91dGxldF09XCJjb2wudGVtcGxhdGVcIlxyXG4gICAgICAgICAgICAgICAgICAgICAgICAgIFtuZ1RlbXBsYXRlT3V0bGV0Q29udGV4dF09XCJ7XHJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAkaW1wbGljaXQ6IGl0ZW0sXHJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICByb3c6IGl0ZW0sXHJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICBjb2x1bW46IGNvbCxcclxuICAgICAgICAgICAgICAgICAgICAgICAgICAgIGluZGV4OiBpLFxyXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgdmFsdWU6IGl0ZW1bY29sLmtleV1cclxuICAgICAgICAgICAgICAgICAgICAgICAgICB9XCJcclxuICAgICAgICAgICAgICAgICAgICAgICAgLz5cclxuICAgICAgICAgICAgICAgICAgICAgIH0gQGVsc2UgaWYgKGNvbC5lbGxpcHNpcyAmJiBzaG91bGRTaG93RWxsaXBzaXMoY29sLCBpdGVtW2NvbC5rZXldKSkge1xyXG4gICAgICAgICAgICAgICAgICAgICAgICA8c3BhblxyXG4gICAgICAgICAgICAgICAgICAgICAgICAgIGNsYXNzPVwidWktY3J1ZC10YWJsZV9fZWxsaXBzaXNcIlxyXG4gICAgICAgICAgICAgICAgICAgICAgICAgIFttYXRUb29sdGlwXT1cImdldEZ1bGxUZXh0KGl0ZW1bY29sLmtleV0pXCJcclxuICAgICAgICAgICAgICAgICAgICAgICAgPnt7IHRydW5jYXRlVGV4dChjb2wsIGl0ZW1bY29sLmtleV0pIH19PC9zcGFuPlxyXG4gICAgICAgICAgICAgICAgICAgICAgfSBAZWxzZSB7XHJcbiAgICAgICAgICAgICAgICAgICAgICAgIDxzcGFuIGNsYXNzPVwidWktY3J1ZC10YWJsZV9fY2VsbC10ZXh0XCI+e3sgaXRlbVtjb2wua2V5XSA/PyAnXFx1MjAxNCcgfX08L3NwYW4+XHJcbiAgICAgICAgICAgICAgICAgICAgICB9XHJcbiAgICAgICAgICAgICAgICAgICAgPC90ZD5cclxuICAgICAgICAgICAgICAgICAgfVxyXG5cclxuICAgICAgICAgICAgICAgICAgPCEtLSBDb2xvbm5hIGF6aW9uaSAtLT5cclxuICAgICAgICAgICAgICAgICAgQGlmICghZGlzYWJsZWQgJiYgaGFzQW55QWN0aW9ucygpKSB7XHJcbiAgICAgICAgICAgICAgICAgICAgPHRkIGNsYXNzPVwidWktY3J1ZC10YWJsZV9fdGQgdWktY3J1ZC10YWJsZV9fdGQtLWFjdGlvbnNcIj5cclxuICAgICAgICAgICAgICAgICAgICAgIDxkaXYgY2xhc3M9XCJ1aS1jcnVkLXRhYmxlX19hY3Rpb25zXCI+XHJcbiAgICAgICAgICAgICAgICAgICAgICAgIEBpZiAoY29uZmlnPy5hbGxvd1ZpZXcgPT09IHRydWUpIHtcclxuICAgICAgICAgICAgICAgICAgICAgICAgICA8YnV0dG9uXHJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICBjbGFzcz1cInVpLWNydWQtdGFibGVfX2FjdGlvbi1idG5cIlxyXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgW2NsYXNzLnVpLWNydWQtdGFibGVfX2FjdGlvbi1idG4tLWRpc2FibGVkXT1cIiFjYW5WaWV3SXRlbShpdGVtKVwiXHJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICBbbWF0VG9vbHRpcF09XCJnZXRWaWV3VG9vbHRpcChpdGVtKVwiXHJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAoY2xpY2spPVwib25WaWV3QnRuQ2xpY2soJGV2ZW50LCBpdGVtLCBpKVwiXHJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICB0eXBlPVwiYnV0dG9uXCJcclxuICAgICAgICAgICAgICAgICAgICAgICAgICA+XHJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICA8bHVjaWRlLWljb24gbmFtZT1cImV5ZVwiIFtzaXplXT1cIjE2XCIgYXJpYS1oaWRkZW49XCJ0cnVlXCIgLz5cclxuICAgICAgICAgICAgICAgICAgICAgICAgICA8L2J1dHRvbj5cclxuICAgICAgICAgICAgICAgICAgICAgICAgfVxyXG4gICAgICAgICAgICAgICAgICAgICAgICBAaWYgKGNvbmZpZz8uYWxsb3dFZGl0ICE9PSBmYWxzZSkge1xyXG4gICAgICAgICAgICAgICAgICAgICAgICAgIDxidXR0b25cclxuICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNsYXNzPVwidWktY3J1ZC10YWJsZV9fYWN0aW9uLWJ0blwiXHJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICBbY2xhc3MudWktY3J1ZC10YWJsZV9fYWN0aW9uLWJ0bi0tZGlzYWJsZWRdPVwiIWNhbkVkaXRJdGVtKGl0ZW0pXCJcclxuICAgICAgICAgICAgICAgICAgICAgICAgICAgIFttYXRUb29sdGlwXT1cImdldEVkaXRUb29sdGlwKGl0ZW0pXCJcclxuICAgICAgICAgICAgICAgICAgICAgICAgICAgIChjbGljayk9XCJvbkVkaXRCdG5DbGljaygkZXZlbnQsIGl0ZW0sIGkpXCJcclxuICAgICAgICAgICAgICAgICAgICAgICAgICAgIHR5cGU9XCJidXR0b25cIlxyXG4gICAgICAgICAgICAgICAgICAgICAgICAgID5cclxuICAgICAgICAgICAgICAgICAgICAgICAgICAgIDxsdWNpZGUtaWNvbiBuYW1lPVwicGVuY2lsXCIgW3NpemVdPVwiMTZcIiBhcmlhLWhpZGRlbj1cInRydWVcIiAvPlxyXG4gICAgICAgICAgICAgICAgICAgICAgICAgIDwvYnV0dG9uPlxyXG4gICAgICAgICAgICAgICAgICAgICAgICB9XHJcbiAgICAgICAgICAgICAgICAgICAgICAgIEBpZiAoY29uZmlnPy5hbGxvd0R1cGxpY2F0ZSA9PT0gdHJ1ZSkge1xyXG4gICAgICAgICAgICAgICAgICAgICAgICAgIDxidXR0b25cclxuICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNsYXNzPVwidWktY3J1ZC10YWJsZV9fYWN0aW9uLWJ0blwiXHJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICBbY2xhc3MudWktY3J1ZC10YWJsZV9fYWN0aW9uLWJ0bi0tZGlzYWJsZWRdPVwiIWNhbkR1cGxpY2F0ZUl0ZW0oaXRlbSlcIlxyXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgW21hdFRvb2x0aXBdPVwiZ2V0RHVwbGljYXRlVG9vbHRpcChpdGVtKVwiXHJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAoY2xpY2spPVwib25EdXBsaWNhdGVCdG5DbGljaygkZXZlbnQsIGl0ZW0pXCJcclxuICAgICAgICAgICAgICAgICAgICAgICAgICAgIHR5cGU9XCJidXR0b25cIlxyXG4gICAgICAgICAgICAgICAgICAgICAgICAgID5cclxuICAgICAgICAgICAgICAgICAgICAgICAgICAgIDxsdWNpZGUtaWNvbiBuYW1lPVwiY29weVwiIFtzaXplXT1cIjE2XCIgYXJpYS1oaWRkZW49XCJ0cnVlXCIgLz5cclxuICAgICAgICAgICAgICAgICAgICAgICAgICA8L2J1dHRvbj5cclxuICAgICAgICAgICAgICAgICAgICAgICAgfVxyXG4gICAgICAgICAgICAgICAgICAgICAgICBAaWYgKGNvbmZpZz8uYWxsb3dEZWxldGUgIT09IGZhbHNlKSB7XHJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgPGJ1dHRvblxyXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgY2xhc3M9XCJ1aS1jcnVkLXRhYmxlX19hY3Rpb24tYnRuIHVpLWNydWQtdGFibGVfX2FjdGlvbi1idG4tLXdhcm5cIlxyXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgW2NsYXNzLnVpLWNydWQtdGFibGVfX2FjdGlvbi1idG4tLWRpc2FibGVkXT1cIiFjYW5EZWxldGVJdGVtKGl0ZW0pXCJcclxuICAgICAgICAgICAgICAgICAgICAgICAgICAgIFttYXRUb29sdGlwXT1cImdldERlbGV0ZVRvb2x0aXAoaXRlbSlcIlxyXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgKGNsaWNrKT1cIm9uRGVsZXRlQnRuQ2xpY2soJGV2ZW50LCBpdGVtLCBpKVwiXHJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICB0eXBlPVwiYnV0dG9uXCJcclxuICAgICAgICAgICAgICAgICAgICAgICAgICA+XHJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICA8bHVjaWRlLWljb24gbmFtZT1cInRyYXNoLTJcIiBbc2l6ZV09XCIxNlwiIGFyaWEtaGlkZGVuPVwidHJ1ZVwiIC8+XHJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgPC9idXR0b24+XHJcbiAgICAgICAgICAgICAgICAgICAgICAgIH1cclxuICAgICAgICAgICAgICAgICAgICAgIDwvZGl2PlxyXG4gICAgICAgICAgICAgICAgICAgIDwvdGQ+XHJcbiAgICAgICAgICAgICAgICAgIH1cclxuICAgICAgICAgICAgICAgIDwvdHI+XHJcbiAgICAgICAgICAgICAgfVxyXG4gICAgICAgICAgICA8L3Rib2R5PlxyXG4gICAgICAgICAgPC90YWJsZT5cclxuICAgICAgICA8L2Rpdj5cclxuXHJcbiAgICAgICAgPCEtLSBGb290ZXIgY29uIHB1bHNhbnRlIGFnZ2l1bmdpIC0tPlxyXG4gICAgICAgIDxkaXYgY2xhc3M9XCJ1aS1jcnVkLXRhYmxlX19mb290ZXJcIj5cclxuICAgICAgICAgIDx1aS1idXR0b24tYXJlYSBbYnV0dG9uc109XCJhZGRCdXR0b25cIiBhbGlnbj1cImVuZFwiIGdhcD1cInhzXCIgLz5cclxuICAgICAgICA8L2Rpdj5cclxuICAgICAgfVxyXG5cclxuICAgICAgPCEtLSDilZDilZDilZAgVmlzdGEgZm9ybSAoaW5zZXJ0L2VkaXQvZHVwbGljYXRlL3JlYWRvbmx5KSDilZDilZDilZDilZDilZDilZAgLS0+XHJcbiAgICAgIEBpZiAoaXNGb3JtTW9kZSkge1xyXG4gICAgICAgIDxkaXYgY2xhc3M9XCJ1aS1jcnVkLXRhYmxlX19mb3JtXCI+XHJcbiAgICAgICAgICA8ZGl2IGNsYXNzPVwidWktY3J1ZC10YWJsZV9fZm9ybS1oZWFkZXJcIj5cclxuICAgICAgICAgICAgPGg0IGNsYXNzPVwidWktY3J1ZC10YWJsZV9fZm9ybS10aXRsZVwiPlxyXG4gICAgICAgICAgICAgIEBzd2l0Y2ggKGN1cnJlbnRNb2RlKSB7XHJcbiAgICAgICAgICAgICAgICBAY2FzZSAoJ2luc2VydCcpICAgIHsgTnVvdm8gZWxlbWVudG8gfVxyXG4gICAgICAgICAgICAgICAgQGNhc2UgKCdlZGl0JykgICAgICB7IE1vZGlmaWNhIGVsZW1lbnRvIH1cclxuICAgICAgICAgICAgICAgIEBjYXNlICgnZHVwbGljYXRlJykgeyBEdXBsaWNhIGVsZW1lbnRvIH1cclxuICAgICAgICAgICAgICAgIEBjYXNlICgncmVhZG9ubHknKSAgeyBWaXN1YWxpenphIGVsZW1lbnRvIH1cclxuICAgICAgICAgICAgICB9XHJcbiAgICAgICAgICAgIDwvaDQ+XHJcbiAgICAgICAgICA8L2Rpdj5cclxuICAgICAgICAgIDxkaXYgY2xhc3M9XCJ1aS1jcnVkLXRhYmxlX19mb3JtLWNvbnRlbnRcIj5cclxuICAgICAgICAgICAgPHVpLWZvcm0tYnVpbGRlclxyXG4gICAgICAgICAgICAgIFtzY2hlbWFdPVwiY29uZmlnLmZvcm1TY2hlbWFcIlxyXG4gICAgICAgICAgICAgIFtpbml0aWFsRGF0YV09XCJmb3JtSW5pdGlhbERhdGFcIlxyXG4gICAgICAgICAgICAgIFtyZWFkb25seV09XCJpc1JlYWRvbmx5TW9kZVwiXHJcbiAgICAgICAgICAgICAgW2J1dHRvbnNPdmVycmlkZV09XCJmb3JtQnV0dG9uc1wiXHJcbiAgICAgICAgICAgIC8+XHJcbiAgICAgICAgICA8L2Rpdj5cclxuICAgICAgICA8L2Rpdj5cclxuICAgICAgfVxyXG4gICAgPC9kaXY+XHJcbiAgYCxcclxuICBzdHlsZVVybDogJy4vY3J1ZC10YWJsZS5jb21wb25lbnQuc2NzcycsXHJcbn0pXHJcbmV4cG9ydCBjbGFzcyBVaUNydWRUYWJsZUNvbXBvbmVudDxUIGV4dGVuZHMgUmVjb3JkPHN0cmluZywgYW55PiA9IFJlY29yZDxzdHJpbmcsIGFueT4+IGltcGxlbWVudHMgT25Jbml0LCBPbkRlc3Ryb3kge1xyXG4gIHByaXZhdGUgcmVhZG9ubHkgY2RyID0gaW5qZWN0KENoYW5nZURldGVjdG9yUmVmKTtcclxuICBwcml2YXRlIHJlYWRvbmx5IG1vZGFsU2VydmljZSA9IGluamVjdChVaU1vZGFsU2VydmljZSk7XHJcbiAgcHJpdmF0ZSByZWFkb25seSBkZXN0cm95JCA9IG5ldyBTdWJqZWN0PHZvaWQ+KCk7XHJcblxyXG4gIC8vIOKUgOKUgCBJbnB1dHMg4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSAXHJcblxyXG4gIC8qKiBBcnJheSBkaSBkYXRpIGRhIHZpc3VhbGl6emFyZSBuZWxsYSB0YWJlbGxhLiAqL1xyXG4gIEBJbnB1dCgpIGRhdGE6IFRbXSA9IFtdO1xyXG5cclxuICAvKiogRGVmaW5pemlvbmUgZGVsbGUgY29sb25uZS4gKi9cclxuICBASW5wdXQoKSBjb2x1bW5zOiBVaUNydWRUYWJsZUNvbHVtbltdID0gW107XHJcblxyXG4gIC8qKiBDb25maWd1cmF6aW9uZSBDUlVEIChzY2hlbWEgZm9ybSwgcGVybWVzc2ksIGxhYmVsKS4gKi9cclxuICBASW5wdXQoKSBjb25maWchOiBVaUNydWRUYWJsZUNvbmZpZztcclxuXHJcbiAgLyoqIE1vc3RyYSBsJ292ZXJsYXkgZGkgY2FyaWNhbWVudG8uICovXHJcbiAgQElucHV0KCkgbG9hZGluZyA9IGZhbHNlO1xyXG5cclxuICAvKiogRGlzYWJpbGl0YSB0dXR0ZSBsZSBhemlvbmkgQ1JVRC4gKi9cclxuICBASW5wdXQoKSBkaXNhYmxlZCA9IGZhbHNlO1xyXG5cclxuICAvKipcclxuICAgKiBGdW56aW9uZSBkaSBtYXBwYXR1cmEgYXBwbGljYXRhIGFpIGRhdGkgZGkgdW4gZWxlbWVudG9cclxuICAgKiBwcmltYSBkaSBwYXNzYXJsaSBjb21lIGBpbml0aWFsRGF0YWAgYWwgZm9ybSBidWlsZGVyLlxyXG4gICAqIFV0aWxlIHBlciB0cmFzZm9ybWFyZSBsYSBzdHJ1dHR1cmEgZGF0aSBkZWxsJ29nZ2V0dG8uXHJcbiAgICovXHJcbiAgQElucHV0KCkgaW5pdE1hcHBlcj86IChvcmlnaW5hbERhdGE6IFQpID0+IFJlY29yZDxzdHJpbmcsIGFueT47XHJcblxyXG4gIC8vIOKUgOKUgCBPdXRwdXRzIOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgFxyXG5cclxuICAvKiogRW1lc3NvIG9nbmkgdm9sdGEgY2hlIGxhIGxpc3RhIGRhdGkgY2FtYmlhIChpbnNlcnQvZWRpdC9kdXBsaWNhdGUpLiAqL1xyXG4gIEBPdXRwdXQoKSBkYXRhQ2hhbmdlID0gbmV3IEV2ZW50RW1pdHRlcjxUW10+KCk7XHJcblxyXG4gIC8qKiBFbWVzc28gcXVhbmRvIHZpZW5lIGluc2VyaXRvIHVuIG51b3ZvIGVsZW1lbnRvLiAqL1xyXG4gIEBPdXRwdXQoKSBpdGVtQWRkID0gbmV3IEV2ZW50RW1pdHRlcjxUPigpO1xyXG5cclxuICAvKiogRW1lc3NvIHF1YW5kbyB2aWVuZSBtb2RpZmljYXRvIHVuIGVsZW1lbnRvLiAqL1xyXG4gIEBPdXRwdXQoKSBpdGVtRWRpdCA9IG5ldyBFdmVudEVtaXR0ZXI8VWlDcnVkRWRpdEV2ZW50PFQ+PigpO1xyXG5cclxuICAvKiogRW1lc3NvIHF1YW5kbyB2aWVuZSBlbGltaW5hdG8gdW4gZWxlbWVudG8gKGRvcG8gY29uZmVybWEpLiAqL1xyXG4gIEBPdXRwdXQoKSBpdGVtRGVsZXRlID0gbmV3IEV2ZW50RW1pdHRlcjxVaUNydWREZWxldGVFdmVudDxUPj4oKTtcclxuXHJcbiAgLyoqIEVtZXNzbyBxdWFuZG8gdmllbmUgZHVwbGljYXRvIHVuIGVsZW1lbnRvLiAqL1xyXG4gIEBPdXRwdXQoKSBpdGVtRHVwbGljYXRlID0gbmV3IEV2ZW50RW1pdHRlcjxUPigpO1xyXG5cclxuICAvKiogRW1lc3NvIHF1YW5kbyBsJ3V0ZW50ZSBjbGljY2Egc3UgdW5hIHJpZ2EgaW4gbW9kYWxpdGEgdmlldy4gKi9cclxuICBAT3V0cHV0KCkgcm93Q2xpY2sgPSBuZXcgRXZlbnRFbWl0dGVyPFQ+KCk7XHJcblxyXG4gIC8vIOKUgOKUgCBTdGF0byBpbnRlcm5vIOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgFxyXG5cclxuICAvKiogQGludGVybmFsIE1vZGFsaXRhIGNvcnJlbnRlLiAqL1xyXG4gIGN1cnJlbnRNb2RlOiBVaUNydWRNb2RlID0gJ3ZpZXcnO1xyXG5cclxuICAvKiogQGludGVybmFsIEVsZW1lbnRvIGluIGZhc2UgZGkgbW9kaWZpY2EvdmlzdWFsaXp6YXppb25lLiAqL1xyXG4gIGVkaXRpbmdJdGVtOiBUIHwgbnVsbCA9IG51bGw7XHJcblxyXG4gIC8qKiBAaW50ZXJuYWwgSW5kaWNlIGRlbGwnZWxlbWVudG8gaW4gbW9kaWZpY2EuICovXHJcbiAgZWRpdGluZ0luZGV4OiBudW1iZXIgfCBudWxsID0gbnVsbDtcclxuXHJcbiAgLyoqIEBpbnRlcm5hbCBEYXRpIG9yaWdpbmFsaSBkZWxsJ2VsZW1lbnRvIHByaW1hIGRlbGxhIG1vZGlmaWNhLiAqL1xyXG4gIHByaXZhdGUgbGFzdE9yaWdpbmFsRGF0YTogYW55ID0gbnVsbDtcclxuXHJcbiAgLyoqIEBpbnRlcm5hbCBDYWNoZSBwZXIgaSBkYXRpIGluaXppYWxpIGRlbCBmb3JtLiAqL1xyXG4gIHByaXZhdGUgX2Zvcm1Jbml0aWFsRGF0YTogVWlGb3JtRGF0YSA9IHt9O1xyXG4gIHByaXZhdGUgX2xhc3RDYWNoZUtleTogc3RyaW5nIHwgbnVsbCA9IG51bGw7XHJcblxyXG4gIC8qKiBAaW50ZXJuYWwgUmlmZXJpbWVudG8gYWwgZm9ybSBidWlsZGVyLiAqL1xyXG4gIEBWaWV3Q2hpbGQoVWlGb3JtQnVpbGRlckNvbXBvbmVudCkgZm9ybUJ1aWxkZXIhOiBVaUZvcm1CdWlsZGVyQ29tcG9uZW50O1xyXG5cclxuICAvKiogQGludGVybmFsIFB1bHNhbnRpIHBlciBpbCBmb3JtLiAqL1xyXG4gIGZvcm1CdXR0b25zOiBVaUJ1dHRvbkRlc2NyaXB0b3JbXSA9IFtdO1xyXG5cclxuICAvKiogQGludGVybmFsIFB1bHNhbnRlIGFnZ2l1bmdpLiAqL1xyXG4gIGFkZEJ1dHRvbjogVWlCdXR0b25EZXNjcmlwdG9yW10gPSBbXTtcclxuXHJcbiAgLy8g4pSA4pSAIExpZmVjeWNsZSDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIBcclxuXHJcbiAgbmdPbkluaXQoKTogdm9pZCB7XHJcbiAgICB0aGlzLnNldHVwQnV0dG9ucygpO1xyXG4gIH1cclxuXHJcbiAgbmdPbkRlc3Ryb3koKTogdm9pZCB7XHJcbiAgICB0aGlzLmRlc3Ryb3kkLm5leHQoKTtcclxuICAgIHRoaXMuZGVzdHJveSQuY29tcGxldGUoKTtcclxuICB9XHJcblxyXG4gIC8vIOKUgOKUgCBHZXR0ZXIgY2FsY29sYXRpIOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgFxyXG5cclxuICAvKiogTGEgdGFiZWxsYSBlIGluIG1vZGFsaXRhIGZvcm0gKGluc2VydC9lZGl0L2R1cGxpY2F0ZS9yZWFkb25seSkuICovXHJcbiAgZ2V0IGlzRm9ybU1vZGUoKTogYm9vbGVhbiB7XHJcbiAgICByZXR1cm4gdGhpcy5jdXJyZW50TW9kZSAhPT0gJ3ZpZXcnO1xyXG4gIH1cclxuXHJcbiAgLyoqIExhIHRhYmVsbGEgZSBpbiBtb2RhbGl0YSBzb2xhIGxldHR1cmEuICovXHJcbiAgZ2V0IGlzUmVhZG9ubHlNb2RlKCk6IGJvb2xlYW4ge1xyXG4gICAgcmV0dXJuIHRoaXMuY3VycmVudE1vZGUgPT09ICdyZWFkb25seSc7XHJcbiAgfVxyXG5cclxuICAvKiogRGF0aSBpbml6aWFsaSBjYWxjb2xhdGkgcGVyIGlsIGZvcm0gYnVpbGRlci4gKi9cclxuICBnZXQgZm9ybUluaXRpYWxEYXRhKCk6IFVpRm9ybURhdGEge1xyXG4gICAgY29uc3QgY2FjaGVLZXkgPSB0aGlzLmdlbmVyYXRlQ2FjaGVLZXkoKTtcclxuICAgIGlmICh0aGlzLl9sYXN0Q2FjaGVLZXkgPT09IGNhY2hlS2V5ICYmIHRoaXMuX2Zvcm1Jbml0aWFsRGF0YSkge1xyXG4gICAgICByZXR1cm4gdGhpcy5fZm9ybUluaXRpYWxEYXRhO1xyXG4gICAgfVxyXG5cclxuICAgIGxldCBkYXRhOiBVaUZvcm1EYXRhID0ge307XHJcbiAgICBpZiAodGhpcy5jdXJyZW50TW9kZSA9PT0gJ2VkaXQnIHx8IHRoaXMuY3VycmVudE1vZGUgPT09ICdkdXBsaWNhdGUnIHx8IHRoaXMuY3VycmVudE1vZGUgPT09ICdyZWFkb25seScpIHtcclxuICAgICAgZGF0YSA9IHRoaXMuaW5pdE1hcHBlciA/IHRoaXMuaW5pdE1hcHBlcih0aGlzLmVkaXRpbmdJdGVtIGFzIFQpIDogKHRoaXMuZWRpdGluZ0l0ZW0gYXMgVWlGb3JtRGF0YSkgfHwge307XHJcbiAgICB9XHJcblxyXG4gICAgdGhpcy5fZm9ybUluaXRpYWxEYXRhID0gZGF0YTtcclxuICAgIHRoaXMuX2xhc3RDYWNoZUtleSA9IGNhY2hlS2V5O1xyXG4gICAgcmV0dXJuIHRoaXMuX2Zvcm1Jbml0aWFsRGF0YTtcclxuICB9XHJcblxyXG4gIC8vIOKUgOKUgCBQZXJtZXNzaSDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIBcclxuXHJcbiAgLyoqIFZlcmlmaWNhIHNlIGNpIHNvbm8gYXppb25pIGRpc3BvbmliaWxpIChwZXIgbW9zdHJhcmUgbGEgY29sb25uYSBhemlvbmkpLiAqL1xyXG4gIGhhc0FueUFjdGlvbnMoKTogYm9vbGVhbiB7XHJcbiAgICByZXR1cm4gKFxyXG4gICAgICB0aGlzLmNvbmZpZz8uYWxsb3dFZGl0ICE9PSBmYWxzZSB8fFxyXG4gICAgICB0aGlzLmNvbmZpZz8uYWxsb3dEdXBsaWNhdGUgPT09IHRydWUgfHxcclxuICAgICAgdGhpcy5jb25maWc/LmFsbG93RGVsZXRlICE9PSBmYWxzZSB8fFxyXG4gICAgICB0aGlzLmNvbmZpZz8uYWxsb3dWaWV3ID09PSB0cnVlXHJcbiAgICApO1xyXG4gIH1cclxuXHJcbiAgLyoqIFZlcmlmaWNhIHNlIGwnZWxlbWVudG8gcHVvIGVzc2VyZSBtb2RpZmljYXRvLiAqL1xyXG4gIGNhbkVkaXRJdGVtKGl0ZW06IGFueSk6IGJvb2xlYW4ge1xyXG4gICAgaWYgKHRoaXMuY29uZmlnPy5jYW5FZGl0KSB7XHJcbiAgICAgIHJldHVybiB0aGlzLmNvbmZpZy5jYW5FZGl0KGl0ZW0sIHRoaXMuY3VycmVudE1vZGUgPT09ICdyZWFkb25seScpO1xyXG4gICAgfVxyXG4gICAgcmV0dXJuIHRoaXMuY29uZmlnPy5hbGxvd0VkaXQgIT09IGZhbHNlO1xyXG4gIH1cclxuXHJcbiAgLyoqIFZlcmlmaWNhIHNlIGwnZWxlbWVudG8gcHVvIGVzc2VyZSBlbGltaW5hdG8uICovXHJcbiAgY2FuRGVsZXRlSXRlbShpdGVtOiBhbnkpOiBib29sZWFuIHtcclxuICAgIGlmICh0aGlzLmNvbmZpZz8uY2FuRGVsZXRlKSB7XHJcbiAgICAgIHJldHVybiB0aGlzLmNvbmZpZy5jYW5EZWxldGUoaXRlbSwgdGhpcy5jdXJyZW50TW9kZSA9PT0gJ3JlYWRvbmx5Jyk7XHJcbiAgICB9XHJcbiAgICByZXR1cm4gdGhpcy5jb25maWc/LmFsbG93RGVsZXRlICE9PSBmYWxzZTtcclxuICB9XHJcblxyXG4gIC8qKiBWZXJpZmljYSBzZSBsJ2VsZW1lbnRvIHB1byBlc3NlcmUgZHVwbGljYXRvLiAqL1xyXG4gIGNhbkR1cGxpY2F0ZUl0ZW0oaXRlbTogYW55KTogYm9vbGVhbiB7XHJcbiAgICBpZiAodGhpcy5jb25maWc/LmNhbkR1cGxpY2F0ZSkge1xyXG4gICAgICByZXR1cm4gdGhpcy5jb25maWcuY2FuRHVwbGljYXRlKGl0ZW0sIHRoaXMuY3VycmVudE1vZGUgPT09ICdyZWFkb25seScpO1xyXG4gICAgfVxyXG4gICAgcmV0dXJuIHRoaXMuY29uZmlnPy5hbGxvd0R1cGxpY2F0ZSA9PT0gdHJ1ZTtcclxuICB9XHJcblxyXG4gIC8qKiBWZXJpZmljYSBzZSBsJ2VsZW1lbnRvIHB1byBlc3NlcmUgdmlzdWFsaXp6YXRvIGluIHJlYWRvbmx5LiAqL1xyXG4gIGNhblZpZXdJdGVtKGl0ZW06IGFueSk6IGJvb2xlYW4ge1xyXG4gICAgaWYgKHRoaXMuY29uZmlnPy5jYW5WaWV3KSB7XHJcbiAgICAgIHJldHVybiB0aGlzLmNvbmZpZy5jYW5WaWV3KGl0ZW0pO1xyXG4gICAgfVxyXG4gICAgcmV0dXJuIHRoaXMuY29uZmlnPy5hbGxvd1ZpZXcgPT09IHRydWU7XHJcbiAgfVxyXG5cclxuICAvLyDilIDilIAgVG9vbHRpcCDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIBcclxuXHJcbiAgZ2V0RWRpdFRvb2x0aXAoaXRlbTogYW55KTogc3RyaW5nIHtcclxuICAgIGlmICh0aGlzLmNvbmZpZz8uZ2V0RWRpdFRvb2x0aXApIHJldHVybiB0aGlzLmNvbmZpZy5nZXRFZGl0VG9vbHRpcChpdGVtKTtcclxuICAgIHJldHVybiB0aGlzLmNhbkVkaXRJdGVtKGl0ZW0pID8gJ01vZGlmaWNhJyA6ICdNb2RpZmljYSBub24gZGlzcG9uaWJpbGUnO1xyXG4gIH1cclxuXHJcbiAgZ2V0RGVsZXRlVG9vbHRpcChpdGVtOiBhbnkpOiBzdHJpbmcge1xyXG4gICAgaWYgKHRoaXMuY29uZmlnPy5nZXREZWxldGVUb29sdGlwKSByZXR1cm4gdGhpcy5jb25maWcuZ2V0RGVsZXRlVG9vbHRpcChpdGVtKTtcclxuICAgIHJldHVybiB0aGlzLmNhbkRlbGV0ZUl0ZW0oaXRlbSkgPyAnRWxpbWluYScgOiAnRWxpbWluYXppb25lIG5vbiBkaXNwb25pYmlsZSc7XHJcbiAgfVxyXG5cclxuICBnZXREdXBsaWNhdGVUb29sdGlwKGl0ZW06IGFueSk6IHN0cmluZyB7XHJcbiAgICBpZiAodGhpcy5jb25maWc/LmdldER1cGxpY2F0ZVRvb2x0aXApIHJldHVybiB0aGlzLmNvbmZpZy5nZXREdXBsaWNhdGVUb29sdGlwKGl0ZW0pO1xyXG4gICAgcmV0dXJuIHRoaXMuY2FuRHVwbGljYXRlSXRlbShpdGVtKSA/ICdEdXBsaWNhJyA6ICdEdXBsaWNhemlvbmUgbm9uIGRpc3BvbmliaWxlJztcclxuICB9XHJcblxyXG4gIGdldFZpZXdUb29sdGlwKGl0ZW06IGFueSk6IHN0cmluZyB7XHJcbiAgICBpZiAodGhpcy5jb25maWc/LmdldFZpZXdUb29sdGlwKSByZXR1cm4gdGhpcy5jb25maWcuZ2V0Vmlld1Rvb2x0aXAoaXRlbSk7XHJcbiAgICByZXR1cm4gdGhpcy5jYW5WaWV3SXRlbShpdGVtKSA/ICdWaXN1YWxpenphJyA6ICdWaXN1YWxpenphemlvbmUgbm9uIGRpc3BvbmliaWxlJztcclxuICB9XHJcblxyXG4gIC8vIOKUgOKUgCBDbGFzc2kgQ1NTIHBlciByaWdhIOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgFxyXG5cclxuICBnZXRSb3dDbGFzc2VzKGl0ZW06IGFueSwgaW5kZXg6IG51bWJlcik6IHN0cmluZyB7XHJcbiAgICBsZXQgY2xhc3NlcyA9ICd1aS1jcnVkLXRhYmxlX19yb3cnO1xyXG4gICAgaWYgKHRoaXMuY29uZmlnPy5nZXRSb3dDbGFzcykge1xyXG4gICAgICBjb25zdCBleHRyYSA9IHRoaXMuY29uZmlnLmdldFJvd0NsYXNzKGl0ZW0sIGluZGV4KTtcclxuICAgICAgaWYgKEFycmF5LmlzQXJyYXkoZXh0cmEpKSB7XHJcbiAgICAgICAgY2xhc3NlcyArPSAnICcgKyBleHRyYS5qb2luKCcgJyk7XHJcbiAgICAgIH0gZWxzZSBpZiAoZXh0cmEpIHtcclxuICAgICAgICBjbGFzc2VzICs9ICcgJyArIGV4dHJhO1xyXG4gICAgICB9XHJcbiAgICB9XHJcbiAgICByZXR1cm4gY2xhc3NlcztcclxuICB9XHJcblxyXG4gIC8vIOKUgOKUgCBFbGxpcHNpcyDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIBcclxuXHJcbiAgc2hvdWxkU2hvd0VsbGlwc2lzKGNvbDogVWlDcnVkVGFibGVDb2x1bW4sIHZhbHVlOiBhbnkpOiBib29sZWFuIHtcclxuICAgIGlmICghY29sLmVsbGlwc2lzIHx8IHZhbHVlID09IG51bGwpIHJldHVybiBmYWxzZTtcclxuICAgIHJldHVybiBTdHJpbmcodmFsdWUpLmxlbmd0aCA+IChjb2wubWF4TGVuZ3RoIHx8IDUwKTtcclxuICB9XHJcblxyXG4gIHRydW5jYXRlVGV4dChjb2w6IFVpQ3J1ZFRhYmxlQ29sdW1uLCB2YWx1ZTogYW55KTogc3RyaW5nIHtcclxuICAgIGlmICh2YWx1ZSA9PSBudWxsKSByZXR1cm4gJ1xcdTIwMTQnO1xyXG4gICAgY29uc3Qgc3RyID0gU3RyaW5nKHZhbHVlKTtcclxuICAgIGNvbnN0IG1heCA9IGNvbC5tYXhMZW5ndGggfHwgNTA7XHJcbiAgICByZXR1cm4gc3RyLmxlbmd0aCA+IG1heCA/IHN0ci5zdWJzdHJpbmcoMCwgbWF4KSArICdcXHUyMDI2JyA6IHN0cjtcclxuICB9XHJcblxyXG4gIGdldEZ1bGxUZXh0KHZhbHVlOiBhbnkpOiBzdHJpbmcge1xyXG4gICAgcmV0dXJuIFN0cmluZyh2YWx1ZSA/PyAnXFx1MjAxNCcpO1xyXG4gIH1cclxuXHJcbiAgLy8g4pSA4pSAIE9wZXJhemlvbmkgQ1JVRCDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIBcclxuXHJcbiAgLyoqIEF0dGl2YSBsYSBtb2RhbGl0YSBpbnNlcmltZW50by4gKi9cclxuICBhY3RpdmF0ZUluc2VydE1vZGUoKTogdm9pZCB7XHJcbiAgICB0aGlzLmNvbmZpZz8ub25CZWZvcmVGb3JtT3Blbj8uKCdpbnNlcnQnLCBudWxsLCB0aGlzLmRhdGEpO1xyXG4gICAgdGhpcy5jdXJyZW50TW9kZSA9ICdpbnNlcnQnO1xyXG4gICAgdGhpcy5lZGl0aW5nSXRlbSA9IG51bGw7XHJcbiAgICB0aGlzLmVkaXRpbmdJbmRleCA9IG51bGw7XHJcbiAgICB0aGlzLmludmFsaWRhdGVGb3JtQ2FjaGUoKTtcclxuICAgIHRoaXMuc2V0dXBCdXR0b25zKCk7XHJcbiAgICB0aGlzLmNkci5kZXRlY3RDaGFuZ2VzKCk7XHJcbiAgfVxyXG5cclxuICAvKiogQXR0aXZhIGxhIG1vZGFsaXRhIG1vZGlmaWNhIHBlciB1biBlbGVtZW50by4gKi9cclxuICBhY3RpdmF0ZUVkaXRNb2RlKGl0ZW06IFQsIGluZGV4OiBudW1iZXIpOiB2b2lkIHtcclxuICAgIGlmICghdGhpcy5jYW5FZGl0SXRlbShpdGVtKSkgcmV0dXJuO1xyXG4gICAgdGhpcy5jb25maWc/Lm9uQmVmb3JlRm9ybU9wZW4/LignZWRpdCcsIGl0ZW0sIHRoaXMuZGF0YSk7XHJcbiAgICB0aGlzLmN1cnJlbnRNb2RlID0gJ2VkaXQnO1xyXG4gICAgdGhpcy5sYXN0T3JpZ2luYWxEYXRhID0gaXRlbTtcclxuICAgIHRoaXMuZWRpdGluZ0l0ZW0gPSB7IC4uLml0ZW0gfTtcclxuICAgIHRoaXMuZWRpdGluZ0luZGV4ID0gaW5kZXg7XHJcbiAgICB0aGlzLmludmFsaWRhdGVGb3JtQ2FjaGUoKTtcclxuICAgIHRoaXMuc2V0dXBCdXR0b25zKCk7XHJcbiAgICB0aGlzLmNkci5kZXRlY3RDaGFuZ2VzKCk7XHJcbiAgfVxyXG5cclxuICAvKiogQXR0aXZhIGxhIG1vZGFsaXRhIHZpc3VhbGl6emF6aW9uZSByZWFkb25seSBwZXIgdW4gZWxlbWVudG8uICovXHJcbiAgYWN0aXZhdGVWaWV3TW9kZShpdGVtOiBULCBpbmRleDogbnVtYmVyKTogdm9pZCB7XHJcbiAgICBpZiAoIXRoaXMuY2FuVmlld0l0ZW0oaXRlbSkpIHJldHVybjtcclxuICAgIHRoaXMuY3VycmVudE1vZGUgPSAncmVhZG9ubHknO1xyXG4gICAgdGhpcy5sYXN0T3JpZ2luYWxEYXRhID0gaXRlbTtcclxuICAgIHRoaXMuZWRpdGluZ0l0ZW0gPSB7IC4uLml0ZW0gfTtcclxuICAgIHRoaXMuZWRpdGluZ0luZGV4ID0gaW5kZXg7XHJcbiAgICB0aGlzLmludmFsaWRhdGVGb3JtQ2FjaGUoKTtcclxuICAgIHRoaXMuc2V0dXBCdXR0b25zKCk7XHJcbiAgICB0aGlzLmNkci5kZXRlY3RDaGFuZ2VzKCk7XHJcbiAgfVxyXG5cclxuICAvKiogQXR0aXZhIGxhIG1vZGFsaXRhIGR1cGxpY2F6aW9uZSBwZXIgdW4gZWxlbWVudG8uICovXHJcbiAgYWN0aXZhdGVEdXBsaWNhdGVNb2RlKGl0ZW06IFQpOiB2b2lkIHtcclxuICAgIGlmICghdGhpcy5jYW5EdXBsaWNhdGVJdGVtKGl0ZW0pKSByZXR1cm47XHJcbiAgICB0aGlzLmNvbmZpZz8ub25CZWZvcmVGb3JtT3Blbj8uKCdkdXBsaWNhdGUnLCBpdGVtLCB0aGlzLmRhdGEpO1xyXG4gICAgdGhpcy5jdXJyZW50TW9kZSA9ICdkdXBsaWNhdGUnO1xyXG4gICAgdGhpcy5lZGl0aW5nSXRlbSA9IHsgLi4uaXRlbSB9O1xyXG4gICAgdGhpcy5lZGl0aW5nSW5kZXggPSBudWxsO1xyXG4gICAgdGhpcy5pbnZhbGlkYXRlRm9ybUNhY2hlKCk7XHJcbiAgICB0aGlzLnNldHVwQnV0dG9ucygpO1xyXG4gICAgdGhpcy5jZHIuZGV0ZWN0Q2hhbmdlcygpO1xyXG4gIH1cclxuXHJcbiAgLyoqIEFubnVsbGEgaWwgZm9ybSBlIHRvcm5hIGFsbGEgdmlzdGEgdGFiZWxsYS4gKi9cclxuICBjYW5jZWxGb3JtKCk6IHZvaWQge1xyXG4gICAgdGhpcy5jdXJyZW50TW9kZSA9ICd2aWV3JztcclxuICAgIHRoaXMuZWRpdGluZ0l0ZW0gPSBudWxsO1xyXG4gICAgdGhpcy5lZGl0aW5nSW5kZXggPSBudWxsO1xyXG4gICAgdGhpcy5sYXN0T3JpZ2luYWxEYXRhID0gbnVsbDtcclxuICAgIHRoaXMuaW52YWxpZGF0ZUZvcm1DYWNoZSgpO1xyXG4gICAgdGhpcy5zZXR1cEJ1dHRvbnMoKTtcclxuICAgIHRoaXMuY2RyLmRldGVjdENoYW5nZXMoKTtcclxuICB9XHJcblxyXG4gIC8qKiBTYWx2YSBpIGRhdGkgZGVsIGZvcm0gKGluc2VydC9lZGl0L2R1cGxpY2F0ZSkuICovXHJcbiAgc2F2ZUZvcm0oKTogdm9pZCB7XHJcbiAgICBpZiAoIXRoaXMuZm9ybUJ1aWxkZXIpIHJldHVybjtcclxuICAgIGlmICghdGhpcy5mb3JtQnVpbGRlci5pc0Zvcm1WYWxpZCgpKSB7XHJcbiAgICAgIHRoaXMuZm9ybUJ1aWxkZXIudmFsaWRhdGVBbGxGaWVsZHMoKTtcclxuICAgICAgcmV0dXJuO1xyXG4gICAgfVxyXG5cclxuICAgIGNvbnN0IGZvcm1EYXRhID0gdGhpcy5mb3JtQnVpbGRlci5nZXRGb3JtVmFsdWUoKSBhcyBUO1xyXG4gICAgY29uc3QgbmV3RGF0YSA9IFsuLi50aGlzLmRhdGFdO1xyXG5cclxuICAgIHN3aXRjaCAodGhpcy5jdXJyZW50TW9kZSkge1xyXG4gICAgICBjYXNlICdpbnNlcnQnOlxyXG4gICAgICAgIG5ld0RhdGEucHVzaChmb3JtRGF0YSk7XHJcbiAgICAgICAgdGhpcy5pdGVtQWRkLmVtaXQoZm9ybURhdGEpO1xyXG4gICAgICAgIGJyZWFrO1xyXG4gICAgICBjYXNlICdkdXBsaWNhdGUnOlxyXG4gICAgICAgIG5ld0RhdGEucHVzaChmb3JtRGF0YSk7XHJcbiAgICAgICAgdGhpcy5pdGVtRHVwbGljYXRlLmVtaXQoZm9ybURhdGEpO1xyXG4gICAgICAgIGJyZWFrO1xyXG4gICAgICBjYXNlICdlZGl0JzpcclxuICAgICAgICBpZiAodGhpcy5lZGl0aW5nSW5kZXggIT09IG51bGwpIHtcclxuICAgICAgICAgIG5ld0RhdGFbdGhpcy5lZGl0aW5nSW5kZXhdID0gZm9ybURhdGE7XHJcbiAgICAgICAgICB0aGlzLml0ZW1FZGl0LmVtaXQoe1xyXG4gICAgICAgICAgICBpdGVtOiBmb3JtRGF0YSxcclxuICAgICAgICAgICAgaW5kZXg6IHRoaXMuZWRpdGluZ0luZGV4LFxyXG4gICAgICAgICAgICBvcmlnaW5hbERhdGE6IHRoaXMubGFzdE9yaWdpbmFsRGF0YSxcclxuICAgICAgICAgIH0pO1xyXG4gICAgICAgIH1cclxuICAgICAgICBicmVhaztcclxuICAgIH1cclxuXHJcbiAgICB0aGlzLmRhdGEgPSBuZXdEYXRhO1xyXG4gICAgdGhpcy5kYXRhQ2hhbmdlLmVtaXQodGhpcy5kYXRhKTtcclxuICAgIHRoaXMuY2FuY2VsRm9ybSgpO1xyXG4gIH1cclxuXHJcbiAgLyoqIEVsaW1pbmEgdW4gZWxlbWVudG8gY29uIGNvbmZlcm1hIHRyYW1pdGUgVWlNb2RhbFNlcnZpY2UuICovXHJcbiAgZGVsZXRlSXRlbShpdGVtOiBULCBpbmRleDogbnVtYmVyKTogdm9pZCB7XHJcbiAgICBpZiAoIXRoaXMuY2FuRGVsZXRlSXRlbShpdGVtKSkgcmV0dXJuO1xyXG5cclxuICAgIGNvbnN0IG1lc3NhZ2UgPSB0aGlzLmNvbmZpZz8uZGVsZXRlQ29uZmlybU1lc3NhZ2UgfHwgJ1NlaSBzaWN1cm8gZGkgdm9sZXIgZWxpbWluYXJlIHF1ZXN0byBlbGVtZW50bz8nO1xyXG5cclxuICAgIHRoaXMubW9kYWxTZXJ2aWNlXHJcbiAgICAgIC5jb25maXJtKHtcclxuICAgICAgICB0aXRsZTogJ0NvbmZlcm1hIGVsaW1pbmF6aW9uZScsXHJcbiAgICAgICAgbWVzc2FnZSxcclxuICAgICAgICBjb25maXJtVGV4dDogJ0VsaW1pbmEnLFxyXG4gICAgICAgIGNhbmNlbFRleHQ6ICdBbm51bGxhJyxcclxuICAgICAgICB2YXJpYW50OiAnZGVsZXRlJyxcclxuICAgICAgfSlcclxuICAgICAgLnN1YnNjcmliZSgoY29uZmlybWVkKSA9PiB7XHJcbiAgICAgICAgaWYgKGNvbmZpcm1lZCkge1xyXG4gICAgICAgICAgdGhpcy5pdGVtRGVsZXRlLmVtaXQoeyBpdGVtLCBpbmRleCB9KTtcclxuICAgICAgICB9XHJcbiAgICAgIH0pO1xyXG4gIH1cclxuXHJcbiAgLy8g4pSA4pSAIEV2ZW50IGhhbmRsZXJzIHRlbXBsYXRlIOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgFxyXG5cclxuICBvblJvd0NsaWNrKGl0ZW06IFQpOiB2b2lkIHtcclxuICAgIGlmICh0aGlzLmN1cnJlbnRNb2RlID09PSAndmlldycpIHtcclxuICAgICAgdGhpcy5yb3dDbGljay5lbWl0KGl0ZW0pO1xyXG4gICAgfVxyXG4gIH1cclxuXHJcbiAgLyoqIEBpbnRlcm5hbCAqL1xyXG4gIG9uVmlld0J0bkNsaWNrKGV2ZW50OiBFdmVudCwgaXRlbTogVCwgaW5kZXg6IG51bWJlcik6IHZvaWQge1xyXG4gICAgZXZlbnQuc3RvcFByb3BhZ2F0aW9uKCk7XHJcbiAgICBpZiAodGhpcy5kaXNhYmxlZCB8fCAhdGhpcy5jYW5WaWV3SXRlbShpdGVtKSkgcmV0dXJuO1xyXG4gICAgdGhpcy5hY3RpdmF0ZVZpZXdNb2RlKGl0ZW0sIGluZGV4KTtcclxuICB9XHJcblxyXG4gIC8qKiBAaW50ZXJuYWwgKi9cclxuICBvbkVkaXRCdG5DbGljayhldmVudDogRXZlbnQsIGl0ZW06IFQsIGluZGV4OiBudW1iZXIpOiB2b2lkIHtcclxuICAgIGV2ZW50LnN0b3BQcm9wYWdhdGlvbigpO1xyXG4gICAgaWYgKHRoaXMuZGlzYWJsZWQgfHwgIXRoaXMuY2FuRWRpdEl0ZW0oaXRlbSkpIHJldHVybjtcclxuICAgIHRoaXMuYWN0aXZhdGVFZGl0TW9kZShpdGVtLCBpbmRleCk7XHJcbiAgfVxyXG5cclxuICAvKiogQGludGVybmFsICovXHJcbiAgb25EdXBsaWNhdGVCdG5DbGljayhldmVudDogRXZlbnQsIGl0ZW06IFQpOiB2b2lkIHtcclxuICAgIGV2ZW50LnN0b3BQcm9wYWdhdGlvbigpO1xyXG4gICAgaWYgKHRoaXMuZGlzYWJsZWQgfHwgIXRoaXMuY2FuRHVwbGljYXRlSXRlbShpdGVtKSkgcmV0dXJuO1xyXG4gICAgdGhpcy5hY3RpdmF0ZUR1cGxpY2F0ZU1vZGUoaXRlbSk7XHJcbiAgfVxyXG5cclxuICAvKiogQGludGVybmFsICovXHJcbiAgb25EZWxldGVCdG5DbGljayhldmVudDogRXZlbnQsIGl0ZW06IFQsIGluZGV4OiBudW1iZXIpOiB2b2lkIHtcclxuICAgIGV2ZW50LnN0b3BQcm9wYWdhdGlvbigpO1xyXG4gICAgaWYgKHRoaXMuZGlzYWJsZWQgfHwgIXRoaXMuY2FuRGVsZXRlSXRlbShpdGVtKSkgcmV0dXJuO1xyXG4gICAgdGhpcy5kZWxldGVJdGVtKGl0ZW0sIGluZGV4KTtcclxuICB9XHJcblxyXG4gIC8qKiBAaW50ZXJuYWwgVHJhY2sgZnVuY3Rpb24gcGVyIGlsIGxvb3AgQGZvci4gKi9cclxuICB0cmFja0J5Rm4gPSAoaW5kZXg6IG51bWJlciwgaXRlbTogYW55KTogYW55ID0+IHtcclxuICAgIGNvbnN0IHRyYWNrQnkgPSB0aGlzLmNvbmZpZz8udHJhY2tCeVByb3BlcnR5IHx8ICdpZCc7XHJcbiAgICByZXR1cm4gaXRlbVt0cmFja0J5XSA/PyBpbmRleDtcclxuICB9O1xyXG5cclxuICAvLyDilIDilIAgVXRpbGl0YSBwcml2YXRlIOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgFxyXG5cclxuICAvKiogQGludGVybmFsIEdlbmVyYSB1bmEgY2hpYXZlIHBlciBsYSBjYWNoZSBkZWkgZGF0aSBkZWwgZm9ybS4gKi9cclxuICBwcml2YXRlIGdlbmVyYXRlQ2FjaGVLZXkoKTogc3RyaW5nIHtcclxuICAgIGlmICh0aGlzLmN1cnJlbnRNb2RlID09PSAndmlldycgfHwgdGhpcy5jdXJyZW50TW9kZSA9PT0gJ2luc2VydCcpIHtcclxuICAgICAgcmV0dXJuIHRoaXMuY3VycmVudE1vZGU7XHJcbiAgICB9XHJcbiAgICBjb25zdCBpdGVtSWQgPSB0aGlzLmVkaXRpbmdJdGVtID8gSlNPTi5zdHJpbmdpZnkodGhpcy5lZGl0aW5nSXRlbSkgOiAnbnVsbCc7XHJcbiAgICByZXR1cm4gYCR7dGhpcy5jdXJyZW50TW9kZX0tJHtpdGVtSWR9YDtcclxuICB9XHJcblxyXG4gIC8qKiBAaW50ZXJuYWwgSW52YWxpZGEgbGEgY2FjaGUgZGVpIGRhdGkgZGVsIGZvcm0uICovXHJcbiAgcHJpdmF0ZSBpbnZhbGlkYXRlRm9ybUNhY2hlKCk6IHZvaWQge1xyXG4gICAgdGhpcy5fZm9ybUluaXRpYWxEYXRhID0ge307XHJcbiAgICB0aGlzLl9sYXN0Q2FjaGVLZXkgPSBudWxsO1xyXG4gIH1cclxuXHJcbiAgLyoqIEBpbnRlcm5hbCBDb25maWd1cmEgaSBwdWxzYW50aSBpbiBiYXNlIGFsbGEgbW9kYWxpdGEgY29ycmVudGUuICovXHJcbiAgcHJpdmF0ZSBzZXR1cEJ1dHRvbnMoKTogdm9pZCB7XHJcbiAgICAvLyBQdWxzYW50ZSBcIkFnZ2l1bmdpXCIgKHZpc2liaWxlIHNvbG8gaW4gbW9kYWxpdGEgdmlldylcclxuICAgIHRoaXMuYWRkQnV0dG9uID1cclxuICAgICAgdGhpcy5jdXJyZW50TW9kZSA9PT0gJ3ZpZXcnICYmIHRoaXMuY29uZmlnPy5hbGxvd0FkZCAhPT0gZmFsc2UgJiYgIXRoaXMuZGlzYWJsZWRcclxuICAgICAgICA/IFtcclxuICAgICAgICAgICAge1xyXG4gICAgICAgICAgICAgIGlkOiAnY3J1ZC1hZGQnLFxyXG4gICAgICAgICAgICAgIGxhYmVsOiB0aGlzLmNvbmZpZz8uYWRkQnV0dG9uTGFiZWwgfHwgJ0FnZ2l1bmdpJyxcclxuICAgICAgICAgICAgICB2YXJpYW50OiAnb3V0bGluZScsXHJcbiAgICAgICAgICAgICAgaWNvbjogJ3BsdXMnLFxyXG4gICAgICAgICAgICAgIGFjdGlvbjogKCkgPT4gdGhpcy5hY3RpdmF0ZUluc2VydE1vZGUoKSxcclxuICAgICAgICAgICAgfSxcclxuICAgICAgICAgIF1cclxuICAgICAgICA6IFtdO1xyXG5cclxuICAgIC8vIFB1bHNhbnRpIGZvcm0gKHZpc2liaWxpIGluIG1vZGFsaXRhIGZvcm0pXHJcbiAgICB0aGlzLmZvcm1CdXR0b25zID0gdGhpcy5pc0Zvcm1Nb2RlXHJcbiAgICAgID8gW1xyXG4gICAgICAgICAge1xyXG4gICAgICAgICAgICBpZDogJ2NydWQtY2FuY2VsJyxcclxuICAgICAgICAgICAgbGFiZWw6IHRoaXMuaXNSZWFkb25seU1vZGUgPyAnQ2hpdWRpJyA6IHRoaXMuY29uZmlnPy5jYW5jZWxCdXR0b25MYWJlbCB8fCAnQW5udWxsYScsXHJcbiAgICAgICAgICAgIHZhcmlhbnQ6ICdvdXRsaW5lJyxcclxuICAgICAgICAgICAgYWN0aW9uOiAoKSA9PiB0aGlzLmNhbmNlbEZvcm0oKSxcclxuICAgICAgICAgIH0sXHJcbiAgICAgICAgICAuLi4odGhpcy5pc1JlYWRvbmx5TW9kZVxyXG4gICAgICAgICAgICA/IFtdXHJcbiAgICAgICAgICAgIDogW1xyXG4gICAgICAgICAgICAgICAge1xyXG4gICAgICAgICAgICAgICAgICBpZDogJ2NydWQtc2F2ZScsXHJcbiAgICAgICAgICAgICAgICAgIGxhYmVsOiB0aGlzLmdldFNhdmVCdXR0b25MYWJlbCgpLFxyXG4gICAgICAgICAgICAgICAgICB2YXJpYW50OiAncHJpbWFyeScgYXMgY29uc3QsXHJcbiAgICAgICAgICAgICAgICAgIGljb246ICdjaGVjaycgYXMgY29uc3QsXHJcbiAgICAgICAgICAgICAgICAgIGFjdGlvbjogKCkgPT4gdGhpcy5zYXZlRm9ybSgpLFxyXG4gICAgICAgICAgICAgICAgfSxcclxuICAgICAgICAgICAgICBdKSxcclxuICAgICAgICBdXHJcbiAgICAgIDogW107XHJcbiAgfVxyXG5cclxuICAvKiogQGludGVybmFsIFJlc3RpdHVpc2NlIGxhIGxhYmVsIGRlbCBwdWxzYW50ZSBzYWx2YSBpbiBiYXNlIGFsbGEgbW9kYWxpdGEuICovXHJcbiAgcHJpdmF0ZSBnZXRTYXZlQnV0dG9uTGFiZWwoKTogc3RyaW5nIHtcclxuICAgIHN3aXRjaCAodGhpcy5jdXJyZW50TW9kZSkge1xyXG4gICAgICBjYXNlICdlZGl0JzpcclxuICAgICAgICByZXR1cm4gdGhpcy5jb25maWc/LnVwZGF0ZUJ1dHRvbkxhYmVsIHx8ICdBZ2dpb3JuYSc7XHJcbiAgICAgIGNhc2UgJ2R1cGxpY2F0ZSc6XHJcbiAgICAgICAgcmV0dXJuIHRoaXMuY29uZmlnPy5kdXBsaWNhdGVCdXR0b25MYWJlbCB8fCAnRHVwbGljYSc7XHJcbiAgICAgIGRlZmF1bHQ6XHJcbiAgICAgICAgcmV0dXJuIHRoaXMuY29uZmlnPy5zYXZlQnV0dG9uTGFiZWwgfHwgJ1NhbHZhJztcclxuICAgIH1cclxuICB9XHJcbn1cclxuIl19