@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.
- package/esm2022/gnggln-ng-ui-system.mjs +5 -0
- package/esm2022/lib/components/accordion/accordion.component.mjs +353 -0
- package/esm2022/lib/components/accordion/accordion.types.mjs +6 -0
- package/esm2022/lib/components/accordion/index.mjs +2 -0
- package/esm2022/lib/components/base-layout/base-layout.component.mjs +218 -0
- package/esm2022/lib/components/base-layout/base-layout.types.mjs +6 -0
- package/esm2022/lib/components/base-layout/index.mjs +14 -0
- package/esm2022/lib/components/button/button-area.component.mjs +196 -0
- package/esm2022/lib/components/button/button.component.mjs +164 -0
- package/esm2022/lib/components/button/button.types.mjs +6 -0
- package/esm2022/lib/components/button/index.mjs +16 -0
- package/esm2022/lib/components/crud-table/crud-table.component.mjs +789 -0
- package/esm2022/lib/components/crud-table/crud-table.types.mjs +6 -0
- package/esm2022/lib/components/crud-table/index.mjs +16 -0
- package/esm2022/lib/components/form-builder/adapters/it-date-adapter.mjs +82 -0
- package/esm2022/lib/components/form-builder/directives/currency-input.directive.mjs +184 -0
- package/esm2022/lib/components/form-builder/form-builder.component.mjs +824 -0
- package/esm2022/lib/components/form-builder/form-wizard.component.mjs +510 -0
- package/esm2022/lib/components/form-builder/index.mjs +19 -0
- package/esm2022/lib/components/form-builder/services/form-condition.service.mjs +132 -0
- package/esm2022/lib/components/form-builder/services/form-validation.service.mjs +381 -0
- package/esm2022/lib/components/form-builder/services/location.service.mjs +140 -0
- package/esm2022/lib/components/form-builder/services/wizard-sync.service.mjs +84 -0
- package/esm2022/lib/components/form-builder/sub-components/error-summary/form-error-summary.component.mjs +161 -0
- package/esm2022/lib/components/form-builder/sub-components/file-input/file-input.component.mjs +310 -0
- package/esm2022/lib/components/form-builder/sub-components/specifica-territoriale/specifica-territoriale.component.mjs +648 -0
- package/esm2022/lib/components/form-builder/sub-components/table-territoriale/table-territoriale.component.mjs +432 -0
- package/esm2022/lib/components/form-builder/types/condition.types.mjs +6 -0
- package/esm2022/lib/components/form-builder/types/field.types.mjs +6 -0
- package/esm2022/lib/components/form-builder/types/index.mjs +2 -0
- package/esm2022/lib/components/form-builder/types/schema.types.mjs +6 -0
- package/esm2022/lib/components/form-builder/types/territoriale.types.mjs +6 -0
- package/esm2022/lib/components/form-builder/types/validation.types.mjs +6 -0
- package/esm2022/lib/components/form-builder-editor/form-builder-editor.component.mjs +730 -0
- package/esm2022/lib/components/form-builder-editor/form-builder-editor.service.mjs +56 -0
- package/esm2022/lib/components/form-builder-editor/index.mjs +21 -0
- package/esm2022/lib/components/form-builder-editor/services/editor-persistence.service.mjs +190 -0
- package/esm2022/lib/components/form-builder-editor/services/editor-state.service.mjs +324 -0
- package/esm2022/lib/components/form-builder-editor/services/field-factory.service.mjs +188 -0
- package/esm2022/lib/components/form-builder-editor/sub-components/condition-editor/condition-editor.component.mjs +667 -0
- package/esm2022/lib/components/form-builder-editor/sub-components/editor-toolbar/editor-toolbar.component.mjs +317 -0
- package/esm2022/lib/components/form-builder-editor/sub-components/field-config-panel/field-config-panel.component.mjs +611 -0
- package/esm2022/lib/components/form-builder-editor/sub-components/field-palette/field-palette.component.mjs +267 -0
- package/esm2022/lib/components/form-builder-editor/sub-components/form-values-panel/form-values-panel.component.mjs +276 -0
- package/esm2022/lib/components/form-builder-editor/sub-components/options-editor/options-editor.component.mjs +323 -0
- package/esm2022/lib/components/form-builder-editor/sub-components/preview-container/preview-container.component.mjs +238 -0
- package/esm2022/lib/components/form-builder-editor/sub-components/section-editor/section-editor.component.mjs +472 -0
- package/esm2022/lib/components/form-builder-editor/sub-components/validation-editor/validation-editor.component.mjs +473 -0
- package/esm2022/lib/components/form-builder-editor/types/editor.types.mjs +6 -0
- package/esm2022/lib/components/layout-builder/index.mjs +18 -0
- package/esm2022/lib/components/layout-builder/layout-builder.component.mjs +1730 -0
- package/esm2022/lib/components/layout-builder/layout-builder.types.mjs +9 -0
- package/esm2022/lib/components/layout-builder/layout.service.mjs +239 -0
- package/esm2022/lib/components/modal/confirm-dialog.component.mjs +151 -0
- package/esm2022/lib/components/modal/index.mjs +4 -0
- package/esm2022/lib/components/modal/modal.component.mjs +139 -0
- package/esm2022/lib/components/modal/modal.service.mjs +194 -0
- package/esm2022/lib/components/modal/modal.types.mjs +6 -0
- package/esm2022/lib/components/page-header/breadcrumb.service.mjs +242 -0
- package/esm2022/lib/components/page-header/index.mjs +20 -0
- package/esm2022/lib/components/page-header/page-header.component.mjs +243 -0
- package/esm2022/lib/components/page-header/page-header.types.mjs +21 -0
- package/esm2022/lib/components/table/index.mjs +2 -0
- package/esm2022/lib/components/table/paginated-table.component.mjs +407 -0
- package/esm2022/lib/components/table/table.types.mjs +6 -0
- package/esm2022/lib/core/types/index.mjs +6 -0
- package/esm2022/lib/core/utils/index.mjs +53 -0
- package/esm2022/lib/sources/location-data.opt.json +8942 -0
- package/esm2022/lib/sources/nazioni.opt.json +215 -0
- package/esm2022/public-api.mjs +34 -0
- package/fesm2022/gnggln-ng-ui-system.mjs +55752 -0
- package/fesm2022/gnggln-ng-ui-system.mjs.map +1 -0
- package/index.d.ts +5 -0
- package/lib/components/accordion/accordion.component.d.ts +118 -0
- package/lib/components/accordion/accordion.types.d.ts +62 -0
- package/lib/components/accordion/index.d.ts +2 -0
- package/lib/components/base-layout/base-layout.component.d.ts +83 -0
- package/lib/components/base-layout/base-layout.types.d.ts +26 -0
- package/lib/components/base-layout/index.d.ts +13 -0
- package/lib/components/button/button-area.component.d.ts +88 -0
- package/lib/components/button/button.component.d.ts +55 -0
- package/lib/components/button/button.types.d.ts +70 -0
- package/lib/components/button/index.d.ts +15 -0
- package/lib/components/crud-table/crud-table.component.d.ts +143 -0
- package/lib/components/crud-table/crud-table.types.d.ts +207 -0
- package/lib/components/crud-table/index.d.ts +15 -0
- package/lib/components/form-builder/adapters/it-date-adapter.d.ts +32 -0
- package/lib/components/form-builder/directives/currency-input.directive.d.ts +48 -0
- package/lib/components/form-builder/form-builder.component.d.ts +183 -0
- package/lib/components/form-builder/form-wizard.component.d.ts +87 -0
- package/lib/components/form-builder/index.d.ts +13 -0
- package/lib/components/form-builder/services/form-condition.service.d.ts +46 -0
- package/lib/components/form-builder/services/form-validation.service.d.ts +63 -0
- package/lib/components/form-builder/services/location.service.d.ts +83 -0
- package/lib/components/form-builder/services/wizard-sync.service.d.ts +63 -0
- package/lib/components/form-builder/sub-components/error-summary/form-error-summary.component.d.ts +28 -0
- package/lib/components/form-builder/sub-components/file-input/file-input.component.d.ts +41 -0
- package/lib/components/form-builder/sub-components/specifica-territoriale/specifica-territoriale.component.d.ts +145 -0
- package/lib/components/form-builder/sub-components/table-territoriale/table-territoriale.component.d.ts +108 -0
- package/lib/components/form-builder/types/condition.types.d.ts +51 -0
- package/lib/components/form-builder/types/field.types.d.ts +288 -0
- package/lib/components/form-builder/types/index.d.ts +5 -0
- package/lib/components/form-builder/types/schema.types.d.ts +227 -0
- package/lib/components/form-builder/types/territoriale.types.d.ts +170 -0
- package/lib/components/form-builder/types/validation.types.d.ts +174 -0
- package/lib/components/form-builder-editor/form-builder-editor.component.d.ts +117 -0
- package/lib/components/form-builder-editor/form-builder-editor.service.d.ts +38 -0
- package/lib/components/form-builder-editor/index.d.ts +15 -0
- package/lib/components/form-builder-editor/services/editor-persistence.service.d.ts +42 -0
- package/lib/components/form-builder-editor/services/editor-state.service.d.ts +66 -0
- package/lib/components/form-builder-editor/services/field-factory.service.d.ts +28 -0
- package/lib/components/form-builder-editor/sub-components/condition-editor/condition-editor.component.d.ts +139 -0
- package/lib/components/form-builder-editor/sub-components/editor-toolbar/editor-toolbar.component.d.ts +43 -0
- package/lib/components/form-builder-editor/sub-components/field-config-panel/field-config-panel.component.d.ts +83 -0
- package/lib/components/form-builder-editor/sub-components/field-palette/field-palette.component.d.ts +40 -0
- package/lib/components/form-builder-editor/sub-components/form-values-panel/form-values-panel.component.d.ts +51 -0
- package/lib/components/form-builder-editor/sub-components/options-editor/options-editor.component.d.ts +63 -0
- package/lib/components/form-builder-editor/sub-components/preview-container/preview-container.component.d.ts +68 -0
- package/lib/components/form-builder-editor/sub-components/section-editor/section-editor.component.d.ts +82 -0
- package/lib/components/form-builder-editor/sub-components/validation-editor/validation-editor.component.d.ts +112 -0
- package/lib/components/form-builder-editor/types/editor.types.d.ts +124 -0
- package/lib/components/layout-builder/index.d.ts +16 -0
- package/lib/components/layout-builder/layout-builder.component.d.ts +85 -0
- package/lib/components/layout-builder/layout-builder.types.d.ts +436 -0
- package/lib/components/layout-builder/layout.service.d.ts +100 -0
- package/lib/components/modal/confirm-dialog.component.d.ts +46 -0
- package/lib/components/modal/index.d.ts +4 -0
- package/lib/components/modal/modal.component.d.ts +44 -0
- package/lib/components/modal/modal.service.d.ts +93 -0
- package/lib/components/modal/modal.types.d.ts +110 -0
- package/lib/components/page-header/breadcrumb.service.d.ts +96 -0
- package/lib/components/page-header/index.d.ts +16 -0
- package/lib/components/page-header/page-header.component.d.ts +59 -0
- package/lib/components/page-header/page-header.types.d.ts +96 -0
- package/lib/components/table/index.d.ts +2 -0
- package/lib/components/table/paginated-table.component.d.ts +85 -0
- package/lib/components/table/table.types.d.ts +81 -0
- package/lib/core/types/index.d.ts +57 -0
- package/lib/core/utils/index.d.ts +29 -0
- package/package.json +44 -0
- 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
|