@eqproject/eqp-dynamic-module 2.4.5 → 2.4.6

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (36) hide show
  1. package/esm2020/lib/components/private/add-form-field/add-form-field.component.mjs +37 -77
  2. package/esm2020/lib/components/private/dynamic-module-field-fix/dynamic-module-field.component.mjs +9 -3
  3. package/esm2020/lib/components/private/field-templates/image-field-selector-template/image-field-selector-template.component.mjs +2 -2
  4. package/esm2020/lib/components/private/field-templates/image-field-template/image-field-template.component.mjs +7 -2
  5. package/esm2020/lib/components/private/field-templates/image-with-markers-field-template/image-with-markers-field-template.component.mjs +248 -0
  6. package/esm2020/lib/components/private/form-records/add-form-record/add-form-record.component.mjs +40 -5
  7. package/esm2020/lib/components/private/form-records/list-view-form-record/list-view-form-record.component.mjs +2 -1
  8. package/esm2020/lib/components/private/tmw-image-marker/tmw-image-marker.component.mjs +301 -0
  9. package/esm2020/lib/eqp-dynamic-module.module.mjs +9 -3
  10. package/esm2020/lib/models/baseField.model.mjs +2 -1
  11. package/esm2020/lib/models/dynAttachment.mjs +21 -0
  12. package/esm2020/lib/models/endPointConfiguration.model.mjs +13 -1
  13. package/esm2020/lib/models/fields/imageField.model.mjs +1 -1
  14. package/esm2020/lib/models/fields/imageSelectorField.model.mjs +1 -15
  15. package/esm2020/lib/models/fields/imageWithMarkersField.model.mjs +8 -0
  16. package/esm2020/lib/services/utilityHelper.services.mjs +1 -1
  17. package/fesm2015/eqproject-eqp-dynamic-module.mjs +661 -96
  18. package/fesm2015/eqproject-eqp-dynamic-module.mjs.map +1 -1
  19. package/fesm2020/eqproject-eqp-dynamic-module.mjs +659 -96
  20. package/fesm2020/eqproject-eqp-dynamic-module.mjs.map +1 -1
  21. package/lib/components/private/add-form-field/add-form-field.component.d.ts +4 -12
  22. package/lib/components/private/dynamic-module-field-fix/dynamic-module-field.component.d.ts +2 -1
  23. package/lib/components/private/field-templates/image-field-selector-template/image-field-selector-template.component.d.ts +2 -1
  24. package/lib/components/private/field-templates/image-field-template/image-field-template.component.d.ts +1 -1
  25. package/lib/components/private/field-templates/image-with-markers-field-template/image-with-markers-field-template.component.d.ts +77 -0
  26. package/lib/components/private/form-records/add-form-record/add-form-record.component.d.ts +2 -1
  27. package/lib/components/private/tmw-image-marker/tmw-image-marker.component.d.ts +57 -0
  28. package/lib/eqp-dynamic-module.module.d.ts +14 -12
  29. package/lib/models/baseField.model.d.ts +2 -1
  30. package/lib/models/dynAttachment.d.ts +48 -0
  31. package/lib/models/endPointConfiguration.model.d.ts +1 -0
  32. package/lib/models/fields/imageField.model.d.ts +1 -1
  33. package/lib/models/fields/imageSelectorField.model.d.ts +1 -41
  34. package/lib/models/fields/imageWithMarkersField.model.d.ts +7 -0
  35. package/lib/services/utilityHelper.services.d.ts +2 -2
  36. package/package.json +1 -1
@@ -0,0 +1,248 @@
1
+ import { Component, EventEmitter, Input, Output, ViewChild } from '@angular/core';
2
+ import { ImageFolderEnum } from '../../../../models/fields/imageField.model';
3
+ import { UtilityHelperService } from '../../../../services/utilityHelper.services';
4
+ import { ParamTypeEnum } from '../../../../models/endPointConfiguration.model';
5
+ import * as i0 from "@angular/core";
6
+ import * as i1 from "@angular/material/dialog";
7
+ import * as i2 from "../../../../services/utilityHelper.services";
8
+ import * as i3 from "@angular/common";
9
+ import * as i4 from "../../tmw-image-marker/tmw-image-marker.component";
10
+ /**
11
+ * Metodo per recuperare, a partire da un Blob, la stringa contenente il base64 del file.
12
+ * Usato per prendere il base64 dell'immagine dopo che l'utente ci ha disegnato sopra.
13
+ * @param file Blob che rappresenta l'immagine.
14
+ */
15
+ const toBase64 = file => new Promise((resolve, reject) => {
16
+ const reader = new FileReader();
17
+ reader.readAsDataURL(file);
18
+ reader.onload = () => resolve(reader.result.toString());
19
+ reader.onerror = error => reject(error);
20
+ });
21
+ export class ImageWithMarkersFieldTemplateComponent {
22
+ //#endregion
23
+ constructor(dialog, utilityHelperService) {
24
+ this.dialog = dialog;
25
+ this.utilityHelperService = utilityHelperService;
26
+ this.inConfig = false;
27
+ this.recordChange = new EventEmitter();
28
+ this.imageMark = new EventEmitter();
29
+ this.imageWidth = 0;
30
+ this.MarkingLoaded = false;
31
+ this.FileBase64 = "";
32
+ this.MarkFileBase64 = "";
33
+ this.FileDataBase64 = "";
34
+ //preLoadedImageMetadataFormGroup: UntypedFormGroup;
35
+ //#region Proprietà per configurare il componente ngx-image-Marking e aprire il relativo dialog.
36
+ this.width = null;
37
+ this.height = null;
38
+ }
39
+ ngOnInit() {
40
+ if (this.record[this.field.Name] == null || this.record[this.field.Name] == 0) {
41
+ this.FileDataBase64 = this.field.DynAttachment.FileDataBase64;
42
+ this.imageWidth = (this.field.DynAttachment.ResizedImageHeightPx / this.field.DynAttachment.ImageHeightPx * this.field.DynAttachment.ImageWidthPx) + 50;
43
+ this.record[this.field.Name] = this.field.DynAttachmentID;
44
+ this.imageMarkAttachment = JSON.parse(JSON.stringify(this.field.DynAttachment));
45
+ this.imageMarkAttachment.ID = 0;
46
+ this.AttachmentID = this.field.DynAttachment.ID;
47
+ }
48
+ else {
49
+ if (typeof this.record[this.field.Name] === 'string') {
50
+ this.record[this.field.Name] = JSON.parse(this.record[this.field.Name]);
51
+ }
52
+ this.imageWidth = (this.field.DynAttachment.ResizedImageHeightPx / this.field.DynAttachment.ImageHeightPx * this.field.DynAttachment.ImageWidthPx) + 50;
53
+ if (this.record[this.field.Name].FileBases64 != null) {
54
+ this.FileDataBase64 = this.record[this.field.Name].FileBases64[ImageFolderEnum.Regular];
55
+ }
56
+ else {
57
+ this.FileDataBase64 = this.record[this.field.Name].FileDataBase64;
58
+ }
59
+ this.record[this.field.Name].FileDataBase64 = this.FileDataBase64;
60
+ this.imageMarkAttachment = this.record[this.field.Name];
61
+ this.imageMarkAttachment.FileDataBase64 = this.FileDataBase64;
62
+ this.AttachmentID = this.imageMarkAttachment.ID;
63
+ }
64
+ console.log("image-with-markers-field-template", this.imageWidth);
65
+ this.recordChange.emit(this.record);
66
+ // Clono l'attachment originale, gli cambio il FileBase e gli azzero l'ID
67
+ // this.imageMarkAttachment.FileName = "";
68
+ // this.imageMarkAttachment.FilePath = "";
69
+ this.imageMark.emit(this.imageMarkAttachment);
70
+ this.updateField();
71
+ }
72
+ ngOnChanges(changes) {
73
+ // Se viene modificato il valore di "ngModelInput" allora aggiorna l'input
74
+ if (changes['record'] != undefined && changes['record'].firstChange == false && JSON.stringify(changes['record'].currentValue) != JSON.stringify(changes['record'].previousValue)) {
75
+ this.record = changes['record'].currentValue;
76
+ this.updateField();
77
+ }
78
+ }
79
+ // updateMetadataValidity() {
80
+ // if (this.preLoadedImageMetadataFormGroup.invalid) {
81
+ // this.field.FormFormGroup.controls[this.field.Name].setErrors({ 'incorrect': true });
82
+ // } else {
83
+ // this.field.FormFormGroup.controls[this.field.Name].setErrors(null);
84
+ // }
85
+ // this.onRecordValueChange();
86
+ // }
87
+ /**
88
+ * Metodo per aggiornare il valore del campo quando questo è rappresentato da una formula.
89
+ */
90
+ updateField() {
91
+ if (this.field.Formula) {
92
+ this.record[this.field.Name] = UtilityHelperService.EvaluateFieldFormula(this.field.Formula, this.record, null);
93
+ }
94
+ if (this.field.VisibleIf) {
95
+ this.field.InListView = UtilityHelperService.EvaluateFieldFormula(this.field.VisibleIf, this.record, null);
96
+ }
97
+ }
98
+ resetPreLoadedImage() {
99
+ this.width = null;
100
+ this.height = null;
101
+ /*this.record[this.field.Name] = [JSON.parse(JSON.stringify(this.field.ContextualAttachment))];*/
102
+ this.getImageDimensions();
103
+ }
104
+ /**
105
+ * Metodo invocato quando vengono aggiunti o eliminati gli allegati nel componente AttachmentFieldTemplate.
106
+ * Se l'ImageField è abilitato al disegno (EnableMarking == true), non permette allegati multipli (IsMulitAttach == false)
107
+ * e nel Record corrente è presente l'immagine caricata dall'utente allora recupera le dimensioni dell'immagine e
108
+ * apre il dialog con il componente image-Marking per permettere il disegno sull'immagine appena caricata.
109
+ */
110
+ onAttachmentsChange() {
111
+ /*
112
+ if (!this.field.AttachDefinition.IsMultiAttach && this.field.EnableMarking && this.record[this.field.Name] && this.record[this.field.Name][0]) {
113
+ this.getImageDimensions();
114
+ this.dialogImageDrowingRef = this.dialog.open(this.dialogImageDrowing, {
115
+ disableClose: true,
116
+ hasBackdrop: true,
117
+ width: '75%'
118
+ });
119
+ }
120
+ */
121
+ }
122
+ /**
123
+ * Metodo invocato al salvataggio dell'immagine dal componente image-Marking.
124
+ * Trasforma il Blob ricevuto in un base64, aggiorna il FileContetType e il FileDataBase64
125
+ * dell'allegato caricato e chiude il dialog in cui disegnare.
126
+ * @param MarknPicture Blob dell'immagine disegnata.
127
+ */
128
+ async saveMark(markedImage) {
129
+ let base64File = await toBase64(markedImage.image);
130
+ if (base64File) {
131
+ this.MarkFileBase64 = base64File.split(",")[1];
132
+ this.FileDataBase64 = this.MarkFileBase64;
133
+ // Clono l'attachment originale, gli cambio il FileBase e gli azzero l'ID
134
+ this.imageMarkAttachment.FileDataBase64 = this.MarkFileBase64;
135
+ this.imageMarkAttachment.ExtraJSON = JSON.stringify(markedImage.outmarkers);
136
+ this.imageMark.emit(this.imageMarkAttachment);
137
+ }
138
+ if (this.dialogImageMarkingRef) {
139
+ this.closeDialog();
140
+ }
141
+ }
142
+ /**
143
+ * Metodo per chiudere il dialog in cui disegnare sull'immagine cariata e svuotare le dimensioni.
144
+ */
145
+ closeDialog() {
146
+ this.dialogImageMarkingRef.close();
147
+ this.width = null;
148
+ this.height = null;
149
+ }
150
+ /**
151
+ * Metodo per recuperare le dimensioni dell'immagine caricata per mostrare correttamente
152
+ * il componente image-Marking. Se non viene eseguito questo passaggio l'area in cui disegnare
153
+ * è più grande dell'immagine stessa e al salvataggio i tratti disegnati vengono spostati.
154
+ */
155
+ getImageDimensions() {
156
+ var blob = this.getBlobFromBase64();
157
+ const reader = new FileReader();
158
+ reader.onload = (e) => {
159
+ const image = new Image();
160
+ image.src = e.target.result;
161
+ image.onload = rs => {
162
+ this.height = rs.currentTarget['height'];
163
+ this.width = rs.currentTarget['width'];
164
+ };
165
+ };
166
+ reader.readAsDataURL(blob);
167
+ }
168
+ /**
169
+ * Metodo per convertire il base64 di un'immagine in un Blob.
170
+ * Necessario per il recupero delle dimensioni dell'immagine caricata a partire dal suo base64.
171
+ * Viene eseguito solo nel caso in cui l'ImageField non ammette allegati multipli.
172
+ * @returns Restituisce il Blob dell'immagine caricata.
173
+ */
174
+ getBlobFromBase64() {
175
+ const byteCharacters = atob(this.record[this.field.Name][0].FileDataBase64);
176
+ const byteNumbers = new Array(byteCharacters.length);
177
+ for (let i = 0; i < byteCharacters.length; i++) {
178
+ byteNumbers[i] = byteCharacters.charCodeAt(i);
179
+ }
180
+ const byteArray = new Uint8Array(byteNumbers);
181
+ return new Blob([byteArray], { type: this.record[this.field.Name][0].FileContentType });
182
+ }
183
+ /**
184
+ * Metodo per emettere l'evento che il valore del record è cambiato.
185
+ */
186
+ onRecordValueChange() {
187
+ if (!this.field.Formula) {
188
+ this.recordChange.emit(this.record);
189
+ }
190
+ }
191
+ openMark() {
192
+ this.dialogImageMarkingRef = this.dialog.open(this.dialogImageMarking, {
193
+ disableClose: true,
194
+ hasBackdrop: true,
195
+ width: '75%'
196
+ });
197
+ if (this.AttachmentID != null && Number(this.AttachmentID) > 0) {
198
+ this.load(this.AttachmentID, [ImageFolderEnum.Regular]);
199
+ }
200
+ else {
201
+ this.FileBase64 = this.FileDataBase64;
202
+ this.MarkingLoaded = true;
203
+ }
204
+ }
205
+ load(ID, Types) {
206
+ this.MarkingLoaded = false;
207
+ // preparo la versione corretta del form
208
+ var dynamicModuleParams = [
209
+ {
210
+ ParamName: "AttachmentID",
211
+ ParamValue: ID,
212
+ ParamType: ParamTypeEnum["Query param"],
213
+ },
214
+ {
215
+ ParamName: "Types",
216
+ ParamValue: Types,
217
+ ParamType: ParamTypeEnum["Query param"],
218
+ },
219
+ ];
220
+ this.utilityHelperService.RunEndPointCall(this.endPointConfiguration.Forms.GetAttachmentWithBases64, dynamicModuleParams, (res) => {
221
+ this.FileBase64 = res.FileBases64[ImageFolderEnum.Regular];
222
+ this.MarkingLoaded = true;
223
+ }, (err) => {
224
+ });
225
+ }
226
+ }
227
+ ImageWithMarkersFieldTemplateComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: ImageWithMarkersFieldTemplateComponent, deps: [{ token: i1.MatDialog }, { token: i2.UtilityHelperService }], target: i0.ɵɵFactoryTarget.Component });
228
+ ImageWithMarkersFieldTemplateComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.2.10", type: ImageWithMarkersFieldTemplateComponent, selector: "image-with-markers-field-template", inputs: { endPointConfiguration: "endPointConfiguration", record: "record", field: "field", inConfig: "inConfig" }, outputs: { recordChange: "recordChange", imageMark: "imageMark" }, viewQueries: [{ propertyName: "dialogImageMarking", first: true, predicate: ["dialogImageMarking"], descendants: true, static: true }], usesOnChanges: true, ngImport: i0, template: "<div class=\"row imageContainer\" style=\"width: {{imageWidth}}px;\">\r\n <div class=\"row\">\r\n <div class=\"col-12\">{{field.Description}}</div>\r\n <div class=\"col-12\"><img *ngIf=\"field.DynAttachment\"\r\n src=\"data:{{field.DynAttachment.FileContentType}};base64,{{FileDataBase64}}\"\r\n [height]=\"field.DynAttachment.ResizedImageHeightPx != null ? field.DynAttachment.ResizedImageHeightPx : 128\"\r\n class=\"singleImage\"></div>\r\n <div class=\"col-12 drawButton\">\r\n <button class=\"btn btn-primary btn-w100\" (click)=\"openMark()\">Marca</button>\r\n </div>\r\n <div class=\"col-12 flex justify-content-around buttonkey\" *ngIf=\"field.DynAttachment.ButtonKey != null && field.DynAttachment.ButtonKey != ''\">\r\n <div>Etichetta</div><div>{{field.DynAttachment.ButtonKey}}</div>\r\n </div>\r\n <div class=\"col-12 flex justify-content-around buttonvalue\" *ngIf=\"field.DynAttachment.ButtonValue != null && field.DynAttachment.ButtonValue != ''\"> \r\n <div>Valore</div><div>{{field.DynAttachment.ButtonValue}}</div>\r\n </div>\r\n </div> \r\n</div>\r\n\r\n<!-- DIALOG PER DISEGNARE SUL NUOVO ALLEGATO CARICATO -->\r\n<ng-template #dialogImageMarking>\r\n <div class=\"padder\">\r\n <div\r\n class=\"row\">\r\n <div class=\"col-sm-12 col-md-12\">\r\n <tmw-image-marker\r\n *ngIf=\"MarkingLoaded\"\r\n [src]=\"'data:'+ field.DynAttachment.FileContentType + ';base64,'+ FileBase64\" [inConfig]=\"inConfig\"\r\n [width]=\"field.DynAttachment.ImageWidthPx\" [height]=\"field.DynAttachment.ImageHeightPx\" \r\n (save)=\"saveMark($event)\" (closeDialog)=\"closeDialog()\">\r\n </tmw-image-marker>\r\n </div>\r\n </div>\r\n </div>\r\n</ng-template>", styles: ["::ng-deep image-drawing>button{background-color:var(--primary)!important;color:#fff!important}.imageContainer .singleImage{margin-top:5px;display:block}.drawButton{margin:5px 0}.buttonkey{border-top:1px solid black;padding-top:10px}\n"], dependencies: [{ kind: "directive", type: i3.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: i4.TmwImageMarkerComponent, selector: "tmw-image-marker", inputs: ["src", "width", "height", "inConfig", "outputMimeType", "outputQuality", "forceSizeCanvas"], outputs: ["closeDialog", "save"] }] });
229
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: ImageWithMarkersFieldTemplateComponent, decorators: [{
230
+ type: Component,
231
+ args: [{ selector: 'image-with-markers-field-template', template: "<div class=\"row imageContainer\" style=\"width: {{imageWidth}}px;\">\r\n <div class=\"row\">\r\n <div class=\"col-12\">{{field.Description}}</div>\r\n <div class=\"col-12\"><img *ngIf=\"field.DynAttachment\"\r\n src=\"data:{{field.DynAttachment.FileContentType}};base64,{{FileDataBase64}}\"\r\n [height]=\"field.DynAttachment.ResizedImageHeightPx != null ? field.DynAttachment.ResizedImageHeightPx : 128\"\r\n class=\"singleImage\"></div>\r\n <div class=\"col-12 drawButton\">\r\n <button class=\"btn btn-primary btn-w100\" (click)=\"openMark()\">Marca</button>\r\n </div>\r\n <div class=\"col-12 flex justify-content-around buttonkey\" *ngIf=\"field.DynAttachment.ButtonKey != null && field.DynAttachment.ButtonKey != ''\">\r\n <div>Etichetta</div><div>{{field.DynAttachment.ButtonKey}}</div>\r\n </div>\r\n <div class=\"col-12 flex justify-content-around buttonvalue\" *ngIf=\"field.DynAttachment.ButtonValue != null && field.DynAttachment.ButtonValue != ''\"> \r\n <div>Valore</div><div>{{field.DynAttachment.ButtonValue}}</div>\r\n </div>\r\n </div> \r\n</div>\r\n\r\n<!-- DIALOG PER DISEGNARE SUL NUOVO ALLEGATO CARICATO -->\r\n<ng-template #dialogImageMarking>\r\n <div class=\"padder\">\r\n <div\r\n class=\"row\">\r\n <div class=\"col-sm-12 col-md-12\">\r\n <tmw-image-marker\r\n *ngIf=\"MarkingLoaded\"\r\n [src]=\"'data:'+ field.DynAttachment.FileContentType + ';base64,'+ FileBase64\" [inConfig]=\"inConfig\"\r\n [width]=\"field.DynAttachment.ImageWidthPx\" [height]=\"field.DynAttachment.ImageHeightPx\" \r\n (save)=\"saveMark($event)\" (closeDialog)=\"closeDialog()\">\r\n </tmw-image-marker>\r\n </div>\r\n </div>\r\n </div>\r\n</ng-template>", styles: ["::ng-deep image-drawing>button{background-color:var(--primary)!important;color:#fff!important}.imageContainer .singleImage{margin-top:5px;display:block}.drawButton{margin:5px 0}.buttonkey{border-top:1px solid black;padding-top:10px}\n"] }]
232
+ }], ctorParameters: function () { return [{ type: i1.MatDialog }, { type: i2.UtilityHelperService }]; }, propDecorators: { endPointConfiguration: [{
233
+ type: Input
234
+ }], record: [{
235
+ type: Input
236
+ }], field: [{
237
+ type: Input
238
+ }], inConfig: [{
239
+ type: Input
240
+ }], recordChange: [{
241
+ type: Output
242
+ }], imageMark: [{
243
+ type: Output
244
+ }], dialogImageMarking: [{
245
+ type: ViewChild,
246
+ args: ["dialogImageMarking", { static: true }]
247
+ }] } });
248
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW1hZ2Utd2l0aC1tYXJrZXJzLWZpZWxkLXRlbXBsYXRlLmNvbXBvbmVudC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL2VxcC1keW5hbWljLW1vZHVsZS9zcmMvbGliL2NvbXBvbmVudHMvcHJpdmF0ZS9maWVsZC10ZW1wbGF0ZXMvaW1hZ2Utd2l0aC1tYXJrZXJzLWZpZWxkLXRlbXBsYXRlL2ltYWdlLXdpdGgtbWFya2Vycy1maWVsZC10ZW1wbGF0ZS5jb21wb25lbnQudHMiLCIuLi8uLi8uLi8uLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9lcXAtZHluYW1pYy1tb2R1bGUvc3JjL2xpYi9jb21wb25lbnRzL3ByaXZhdGUvZmllbGQtdGVtcGxhdGVzL2ltYWdlLXdpdGgtbWFya2Vycy1maWVsZC10ZW1wbGF0ZS9pbWFnZS13aXRoLW1hcmtlcnMtZmllbGQtdGVtcGxhdGUuY29tcG9uZW50Lmh0bWwiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLFNBQVMsRUFBRSxZQUFZLEVBQUUsS0FBSyxFQUFxQixNQUFNLEVBQThCLFNBQVMsRUFBRSxNQUFNLGVBQWUsQ0FBQztBQUNqSSxPQUFPLEVBQWMsZUFBZSxFQUFFLE1BQU0sNENBQTRDLENBQUM7QUFFekYsT0FBTyxFQUFFLG9CQUFvQixFQUFFLE1BQU0sNkNBQTZDLENBQUM7QUFFbkYsT0FBTyxFQUF5QixhQUFhLEVBQUUsTUFBTSxnREFBZ0QsQ0FBQzs7Ozs7O0FBS3RHOzs7O0dBSUc7QUFDSCxNQUFNLFFBQVEsR0FBRyxJQUFJLENBQUMsRUFBRSxDQUFDLElBQUksT0FBTyxDQUFTLENBQUMsT0FBTyxFQUFFLE1BQU0sRUFBRSxFQUFFO0lBQy9ELE1BQU0sTUFBTSxHQUFHLElBQUksVUFBVSxFQUFFLENBQUM7SUFDaEMsTUFBTSxDQUFDLGFBQWEsQ0FBQyxJQUFJLENBQUMsQ0FBQztJQUMzQixNQUFNLENBQUMsTUFBTSxH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QUFDLGVBQWUsQ0FDdkMsSUFBSSxDQUFDLHFCQUFxQixDQUFDLEtBQUssQ0FBQyx3QkFBd0IsRUFDekQsbUJBQW1CLEVBQ25CLENBQUMsR0FBa0IsRUFBRSxFQUFFO1lBRXJCLElBQUksQ0FBQyxVQUFVLEdBQUcsR0FBRyxDQUFDLFdBQVcsQ0FBQyxlQUFlLENBQUMsT0FBTyxDQUFDLENBQUM7WUFDM0QsSUFBSSxDQUFDLGFBQWEsR0FBRyxJQUFJLENBQUM7UUFFNUIsQ0FBQyxFQUNELENBQUMsR0FBRyxFQUFFLEVBQUU7UUFFUixDQUFDLENBQ0YsQ0FBQztJQUNKLENBQUM7O29JQTdRVSxzQ0FBc0M7d0hBQXRDLHNDQUFzQyw2WkM1Qm5ELHUrREFrQ2M7NEZETkQsc0NBQXNDO2tCQU5sRCxTQUFTOytCQUNFLG1DQUFtQzttSUFPcEMscUJBQXFCO3NCQUE3QixLQUFLO2dCQUNHLE1BQU07c0JBQWQsS0FBSztnQkFDRyxLQUFLO3NCQUFiLEtBQUs7Z0JBQ0csUUFBUTtzQkFBaEIsS0FBSztnQkFDSSxZQUFZO3NCQUFyQixNQUFNO2dCQUNHLFNBQVM7c0JBQWxCLE1BQU07Z0JBb0I0QyxrQkFBa0I7c0JBQXBFLFNBQVM7dUJBQUMsb0JBQW9CLEVBQUUsRUFBRSxNQUFNLEVBQUUsSUFBSSxFQUFFIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgQ29tcG9uZW50LCBFdmVudEVtaXR0ZXIsIElucHV0LCBPbkNoYW5nZXMsIE9uSW5pdCwgT3V0cHV0LCBTaW1wbGVDaGFuZ2VzLCBUZW1wbGF0ZVJlZiwgVmlld0NoaWxkIH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XHJcbmltcG9ydCB7IEltYWdlRmllbGQsIEltYWdlRm9sZGVyRW51bSB9IGZyb20gJy4uLy4uLy4uLy4uL21vZGVscy9maWVsZHMvaW1hZ2VGaWVsZC5tb2RlbCc7XHJcbmltcG9ydCB7IFJlY29yZCB9IGZyb20gJy4uLy4uLy4uLy4uL21vZGVscy9yZWNvcmQubW9kZWwnO1xyXG5pbXBvcnQgeyBVdGlsaXR5SGVscGVyU2VydmljZSB9IGZyb20gJy4uLy4uLy4uLy4uL3NlcnZpY2VzL3V0aWxpdHlIZWxwZXIuc2VydmljZXMnO1xyXG5pbXBvcnQgeyBNYXREaWFsb2csIE1hdERpYWxvZ1JlZiB9IGZyb20gJ0Bhbmd1bGFyL21hdGVyaWFsL2RpYWxvZyc7XHJcbmltcG9ydCB7IEVuZFBvaW50Q29uZmlndXJhdGlvbiwgUGFyYW1UeXBlRW51bSB9IGZyb20gJy4uLy4uLy4uLy4uL21vZGVscy9lbmRQb2ludENvbmZpZ3VyYXRpb24ubW9kZWwnO1xyXG5pbXBvcnQgeyBEeW5BdHRhY2htZW50IH0gZnJvbSAnLi4vLi4vLi4vLi4vbW9kZWxzL2R5bkF0dGFjaG1lbnQnO1xyXG5pbXBvcnQgeyBNYXJrZWRJbWFnZSB9IGZyb20gJy4uLy4uL3Rtdy1pbWFnZS1tYXJrZXIvdG13LWltYWdlLW1hcmtlci5jb21wb25lbnQnO1xyXG5pbXBvcnQgeyBJbWFnZVdpdGhNYXJrZXJzRmllbGQgfSBmcm9tICcuLi8uLi8uLi8uLi9tb2RlbHMvZmllbGRzL2ltYWdlV2l0aE1hcmtlcnNGaWVsZC5tb2RlbCc7XHJcblxyXG4vKipcclxuICogTWV0b2RvIHBlciByZWN1cGVyYXJlLCBhIHBhcnRpcmUgZGEgdW4gQmxvYiwgbGEgc3RyaW5nYSBjb250ZW5lbnRlIGlsIGJhc2U2NCBkZWwgZmlsZS5cclxuICogVXNhdG8gcGVyIHByZW5kZXJlIGlsIGJhc2U2NCBkZWxsJ2ltbWFnaW5lIGRvcG8gY2hlIGwndXRlbnRlIGNpIGhhIGRpc2VnbmF0byBzb3ByYS5cclxuICogQHBhcmFtIGZpbGUgQmxvYiBjaGUgcmFwcHJlc2VudGEgbCdpbW1hZ2luZS5cclxuICovXHJcbmNvbnN0IHRvQmFzZTY0ID0gZmlsZSA9PiBuZXcgUHJvbWlzZTxzdHJpbmc+KChyZXNvbHZlLCByZWplY3QpID0+IHtcclxuICBjb25zdCByZWFkZXIgPSBuZXcgRmlsZVJlYWRlcigpO1xyXG4gIHJlYWRlci5yZWFkQXNEYXRhVVJMKGZpbGUpO1xyXG4gIHJlYWRlci5vbmxvYWQgPSAoKSA9PiByZXNvbHZlKHJlYWRlci5yZXN1bHQudG9TdHJpbmcoKSk7XHJcbiAgcmVhZGVyLm9uZXJyb3IgPSBlcnJvciA9PiByZWplY3QoZXJyb3IpO1xyXG59KTtcclxuXHJcbkBDb21wb25lbnQoe1xyXG4gIHNlbGVjdG9yOiAnaW1hZ2Utd2l0aC1tYXJrZXJzLWZpZWxkLXRlbXBsYXRlJyxcclxuICB0ZW1wbGF0ZVVybDogJy4vaW1hZ2Utd2l0aC1tYXJrZXJzLWZpZWxkLXRlbXBsYXRlLmNvbXBvbmVudC5odG1sJyxcclxuICBzdHlsZVVybHM6IFsnLi9pbWFnZS13aXRoLW1hcmtlcnMtZmllbGQtdGVtcGxhdGUuY29tcG9uZW50LnNjc3MnXVxyXG59KVxyXG5cclxuZXhwb3J0IGNsYXNzIEltYWdlV2l0aE1hcmtlcnNGaWVsZFRlbXBsYXRlQ29tcG9uZW50IHtcclxuXHJcbiAgQElucHV0KCkgZW5kUG9pbnRDb25maWd1cmF0aW9uOiBFbmRQb2ludENvbmZpZ3VyYXRpb247XHJcbiAgQElucHV0KCkgcmVjb3JkOiBSZWNvcmQ7XHJcbiAgQElucHV0KCkgZmllbGQ6IEltYWdlV2l0aE1hcmtlcnNGaWVsZDtcclxuICBASW5wdXQoKSBpbkNvbmZpZzogYm9vbGVhbiA9IGZhbHNlO1xyXG4gIEBPdXRwdXQoKSByZWNvcmRDaGFuZ2U6IEV2ZW50RW1pdHRlcjxSZWNvcmQ+ID0gbmV3IEV2ZW50RW1pdHRlcjxSZWNvcmQ+KCk7XHJcbiAgQE91dHB1dCgpIGltYWdlTWFyazogRXZlbnRFbWl0dGVyPER5bkF0dGFjaG1lbnQ+ID0gbmV3IEV2ZW50RW1pdHRlcjxEeW5BdHRhY2htZW50PigpO1xyXG5cclxuICBpbWFnZVdpZHRoOiBudW1iZXIgPSAwO1xyXG4gIE1hcmtpbmdMb2FkZWQ6IGJvb2xlYW4gPSBmYWxzZTtcclxuICBGaWxlQmFzZTY0OiBzdHJpbmcgPSBcIlwiO1xyXG4gIE1hcmtGaWxlQmFzZTY0OiBzdHJpbmcgPSBcIlwiO1xyXG4gIEZpbGVEYXRhQmFzZTY0OiBzdHJpbmcgPSBcIlwiO1xyXG5cclxuICBBdHRhY2htZW50SUQ6IHN0cmluZyB8IG51bWJlcjtcclxuXHJcbiAgaW1hZ2VNYXJrQXR0YWNobWVudDogRHluQXR0YWNobWVudDtcclxuICBcclxuXHJcbiAgLy9wcmVMb2FkZWRJbWFnZU1ldGFkYXRhRm9ybUdyb3VwOiBVbnR5cGVkRm9ybUdyb3VwO1xyXG5cclxuICAvLyNyZWdpb24gUHJvcHJpZXTDoCBwZXIgY29uZmlndXJhcmUgaWwgY29tcG9uZW50ZSBuZ3gtaW1hZ2UtTWFya2luZyBlIGFwcmlyZSBpbCByZWxhdGl2byBkaWFsb2cuXHJcbiAgcHVibGljIHdpZHRoID0gbnVsbDtcclxuICBwdWJsaWMgaGVpZ2h0ID0gbnVsbDtcclxuXHJcbiAgZGlhbG9nSW1hZ2VNYXJraW5nUmVmOiBNYXREaWFsb2dSZWY8VGVtcGxhdGVSZWY8YW55Pj47XHJcbiAgQFZpZXdDaGlsZChcImRpYWxvZ0ltYWdlTWFya2luZ1wiLCB7IHN0YXRpYzogdHJ1ZSB9KSBkaWFsb2dJbWFnZU1hcmtpbmc6IFRlbXBsYXRlUmVmPGFueT47XHJcbiAgLy8jZW5kcmVnaW9uXHJcblxyXG4gIGNvbnN0cnVjdG9yKFxyXG4gICAgcHJpdmF0ZSBkaWFsb2c6IE1hdERpYWxvZyxcclxuICAgIHByaXZhdGUgdXRpbGl0eUhlbHBlclNlcnZpY2U6IFV0aWxpdHlIZWxwZXJTZXJ2aWNlXHJcbiAgKSB7IH1cclxuXHJcbiAgbmdPbkluaXQoKTogdm9pZCB7XHJcblxyXG4gICAgaWYgKHRoaXMucmVjb3JkW3RoaXMuZmllbGQuTmFtZV0gPT0gbnVsbCB8fCB0aGlzLnJlY29yZFt0aGlzLmZpZWxkLk5hbWVdID09IDApe1xyXG4gICAgICB0aGlzLkZpbGVEYXRhQmFzZTY0ID0gdGhpcy5maWVsZC5EeW5BdHRhY2htZW50LkZpbGVEYXRhQmFzZTY0O1xyXG4gICAgICB0aGlzLmltYWdlV2lkdGggPSAgKHRoaXMuZmllbGQuRHluQXR0YWNobWVudC5SZXNpemVkSW1hZ2VIZWlnaHRQeCAvIHRoaXMuZmllbGQuRHluQXR0YWNobWVudC5JbWFnZUhlaWdodFB4ICogdGhpcy5maWVsZC5EeW5BdHRhY2htZW50LkltYWdlV2lkdGhQeCkgKyA1MDtcclxuICAgICAgdGhpcy5yZWNvcmRbdGhpcy5maWVsZC5OYW1lXSA9IHRoaXMuZmllbGQuRHluQXR0YWNobWVudElEO1xyXG5cclxuICAgICAgdGhpcy5pbWFnZU1hcmtBdHRhY2htZW50ID0gSlNPTi5wYXJzZShKU09OLnN0cmluZ2lmeSh0aGlzLmZpZWxkLkR5bkF0dGFjaG1lbnQpKTtcclxuICAgICAgdGhpcy5pbWFnZU1hcmtBdHRhY2htZW50LklEID0gMDtcclxuXHJcbiAgICAgIHRoaXMuQXR0YWNobWVudElEID0gdGhpcy5maWVsZC5EeW5BdHRhY2htZW50LklEO1xyXG5cclxuICAgIH0gZWxzZSB7XHJcbiAgICAgIGlmICh0eXBlb2YgdGhpcy5yZWNvcmRbdGhpcy5maWVsZC5OYW1lXSA9PT0gJ3N0cmluZycpe1xyXG4gICAgICAgIHRoaXMucmVjb3JkW3RoaXMuZmllbGQuTmFtZV0gPSBKU09OLnBhcnNlKHRoaXMucmVjb3JkW3RoaXMuZmllbGQuTmFtZV0pO1xyXG4gICAgICB9XHJcbiAgICAgIHRoaXMuaW1hZ2VXaWR0aCA9ICAodGhpcy5maWVsZC5EeW5BdHRhY2htZW50LlJlc2l6ZWRJbWFnZUhlaWdodFB4IC8gdGhpcy5maWVsZC5EeW5BdHRhY2htZW50LkltYWdlSGVpZ2h0UHggKiB0aGlzLmZpZWxkLkR5bkF0dGFjaG1lbnQuSW1hZ2VXaWR0aFB4KSArIDUwO1xyXG5cclxuICAgICAgaWYgKCh0aGlzLnJlY29yZFt0aGlzLmZpZWxkLk5hbWVdIGFzIER5bkF0dGFjaG1lbnQpLkZpbGVCYXNlczY0ICE9IG51bGwpe1xyXG4gICAgICAgIHRoaXMuRmlsZURhdGFCYXNlNjQgPSAodGhpcy5yZWNvcmRbdGhpcy5maWVsZC5OYW1lXSBhcyBEeW5BdHRhY2htZW50KS5GaWxlQmFzZXM2NFtJbWFnZUZvbGRlckVudW0uUmVndWxhcl07XHJcbiAgICAgIH0gZWxzZSB7XHJcbiAgICAgICAgdGhpcy5GaWxlRGF0YUJhc2U2NCA9ICh0aGlzLnJlY29yZFt0aGlzLmZpZWxkLk5hbWVdIGFzIER5bkF0dGFjaG1lbnQpLkZpbGVEYXRhQmFzZTY0O1xyXG4gICAgICB9ICAgICAgXHJcblxyXG4gICAgICAodGhpcy5yZWNvcmRbdGhpcy5maWVsZC5OYW1lXSBhcyBEeW5BdHRhY2htZW50KS5GaWxlRGF0YUJhc2U2NCA9IHRoaXMuRmlsZURhdGFCYXNlNjQ7XHJcblxyXG4gICAgICB0aGlzLmltYWdlTWFya0F0dGFjaG1lbnQgPSB0aGlzLnJlY29yZFt0aGlzLmZpZWxkLk5hbWVdO1xyXG4gICAgICB0aGlzLmltYWdlTWFya0F0dGFjaG1lbnQuRmlsZURhdGFCYXNlNjQgPSB0aGlzLkZpbGVEYXRhQmFzZTY0O1xyXG5cclxuICAgICAgdGhpcy5BdHRhY2htZW50SUQgPSB0aGlzLmltYWdlTWFya0F0dGFjaG1lbnQuSUQ7XHJcbiAgICB9XHJcblxyXG4gICAgY29uc29sZS5sb2coXCJpbWFnZS13aXRoLW1hcmtlcnMtZmllbGQtdGVtcGxhdGVcIix0aGlzLmltYWdlV2lkdGgpO1xyXG5cclxuICAgIHRoaXMucmVjb3JkQ2hhbmdlLmVtaXQodGhpcy5yZWNvcmQpO1xyXG5cclxuXHJcbiAgICAvLyBDbG9ubyBsJ2F0dGFjaG1lbnQgb3JpZ2luYWxlLCBnbGkgY2FtYmlvIGlsIEZpbGVCYXNlIGUgZ2xpIGF6emVybyBsJ0lEXHJcblxyXG4gICAgLy8gdGhpcy5pbWFnZU1hcmtBdHRhY2htZW50LkZpbGVOYW1lID0gXCJcIjtcclxuICAgIC8vIHRoaXMuaW1hZ2VNYXJrQXR0YWNobWVudC5GaWxlUGF0aCA9IFwiXCI7XHJcblxyXG4gICAgdGhpcy5pbWFnZU1hcmsuZW1pdCh0aGlzLmltYWdlTWFya0F0dGFjaG1lbnQpO1xyXG5cclxuXHJcblxyXG5cclxuXHJcbiAgICB0aGlzLnVwZGF0ZUZpZWxkKCk7XHJcbiAgfVxyXG5cclxuICBuZ09uQ2hhbmdlcyhjaGFuZ2VzOiBTaW1wbGVDaGFuZ2VzKSB7XHJcbiAgICAvLyBTZSB2aWVuZSBtb2RpZmljYXRvIGlsIHZhbG9yZSBkaSBcIm5nTW9kZWxJbnB1dFwiIGFsbG9yYSBhZ2dpb3JuYSBsJ2lucHV0XHJcbiAgICBpZiAoY2hhbmdlc1sncmVjb3JkJ10gIT0gdW5kZWZpbmVkICYmIGNoYW5nZXNbJ3JlY29yZCddLmZpcnN0Q2hhbmdlID09IGZhbHNlICYmIEpTT04uc3RyaW5naWZ5KGNoYW5nZXNbJ3JlY29yZCddLmN1cnJlbnRWYWx1ZSkgIT0gSlNPTi5zdHJpbmdpZnkoY2hhbmdlc1sncmVjb3JkJ10ucHJldmlvdXNWYWx1ZSkpIHtcclxuICAgICAgdGhpcy5yZWNvcmQgPSBjaGFuZ2VzWydyZWNvcmQnXS5jdXJyZW50VmFsdWU7XHJcbiAgICAgIHRoaXMudXBkYXRlRmllbGQoKTtcclxuICAgIH1cclxuICB9XHJcblxyXG5cclxuICAvLyB1cGRhdGVNZXRhZGF0YVZhbGlkaXR5KCkge1xyXG4gIC8vICAgaWYgKHRoaXMucHJlTG9hZGVkSW1hZ2VNZXRhZGF0YUZvcm1Hcm91cC5pbnZhbGlkKSB7XHJcbiAgLy8gICAgIHRoaXMuZmllbGQuRm9ybUZvcm1Hcm91cC5jb250cm9sc1t0aGlzLmZpZWxkLk5hbWVdLnNldEVycm9ycyh7ICdpbmNvcnJlY3QnOiB0cnVlIH0pO1xyXG4gIC8vICAgfSBlbHNlIHtcclxuICAvLyAgICAgdGhpcy5maWVsZC5Gb3JtRm9ybUdyb3VwLmNvbnRyb2xzW3RoaXMuZmllbGQuTmFtZV0uc2V0RXJyb3JzKG51bGwpO1xyXG4gIC8vICAgfVxyXG4gIC8vICAgdGhpcy5vblJlY29yZFZhbHVlQ2hhbmdlKCk7XHJcbiAgLy8gfVxyXG5cclxuXHJcbiAgLyoqXHJcbiAgICogTWV0b2RvIHBlciBhZ2dpb3JuYXJlIGlsIHZhbG9yZSBkZWwgY2FtcG8gcXVhbmRvIHF1ZXN0byDDqCByYXBwcmVzZW50YXRvIGRhIHVuYSBmb3JtdWxhLlxyXG4gICAqL1xyXG4gIHVwZGF0ZUZpZWxkKCkge1xyXG4gICAgaWYgKHRoaXMuZmllbGQuRm9ybXVsYSkge1xyXG4gICAgICB0aGlzLnJlY29yZFt0aGlzLmZpZWxkLk5hbWVdID0gVXRpbGl0eUhlbHBlclNlcnZpY2UuRXZhbHVhdGVGaWVsZEZvcm11bGEodGhpcy5maWVsZC5Gb3JtdWxhLCB0aGlzLnJlY29yZCwgbnVsbCk7XHJcbiAgICB9XHJcbiAgICBpZiAodGhpcy5maWVsZC5WaXNpYmxlSWYpIHtcclxuICAgICAgdGhpcy5maWVsZC5Jbkxpc3RWaWV3ID0gVXRpbGl0eUhlbHBlclNlcnZpY2UuRXZhbHVhdGVGaWVsZEZvcm11bGEodGhpcy5maWVsZC5WaXNpYmxlSWYsIHRoaXMucmVjb3JkLCBudWxsKTtcclxuICAgIH1cclxuICB9XHJcblxyXG5cclxuICByZXNldFByZUxvYWRlZEltYWdlKCkge1xyXG4gICAgdGhpcy53aWR0aCA9IG51bGw7XHJcbiAgICB0aGlzLmhlaWdodCA9IG51bGw7XHJcbiAgICAvKnRoaXMucmVjb3JkW3RoaXMuZmllbGQuTmFtZV0gPSBbSlNPTi5wYXJzZShKU09OLnN0cmluZ2lmeSh0aGlzLmZpZWxkLkNvbnRleHR1YWxBdHRhY2htZW50KSldOyovXHJcbiAgICB0aGlzLmdldEltYWdlRGltZW5zaW9ucygpO1xyXG4gIH1cclxuXHJcbiAgLyoqXHJcbiAgICogTWV0b2RvIGludm9jYXRvIHF1YW5kbyB2ZW5nb25vIGFnZ2l1bnRpIG8gZWxpbWluYXRpIGdsaSBhbGxlZ2F0aSBuZWwgY29tcG9uZW50ZSBBdHRhY2htZW50RmllbGRUZW1wbGF0ZS5cclxuICAgKiBTZSBsJ0ltYWdlRmllbGQgw6ggYWJpbGl0YXRvIGFsIGRpc2Vnbm8gKEVuYWJsZU1hcmtpbmcgPT0gdHJ1ZSksIG5vbiBwZXJtZXR0ZSBhbGxlZ2F0aSBtdWx0aXBsaSAoSXNNdWxpdEF0dGFjaCA9PSBmYWxzZSlcclxuICAgKiBlIG5lbCBSZWNvcmQgY29ycmVudGUgw6ggcHJlc2VudGUgbCdpbW1hZ2luZSBjYXJpY2F0YSBkYWxsJ3V0ZW50ZSBhbGxvcmEgcmVjdXBlcmEgbGUgZGltZW5zaW9uaSBkZWxsJ2ltbWFnaW5lIGVcclxuICAgKiBhcHJlIGlsIGRpYWxvZyBjb24gaWwgY29tcG9uZW50ZSBpbWFnZS1NYXJraW5nIHBlciBwZXJtZXR0ZXJlIGlsIGRpc2Vnbm8gc3VsbCdpbW1hZ2luZSBhcHBlbmEgY2FyaWNhdGEuXHJcbiAgICovXHJcbiAgb25BdHRhY2htZW50c0NoYW5nZSgpIHtcclxuICAgIC8qXHJcbiAgICBpZiAoIXRoaXMuZmllbGQuQXR0YWNoRGVmaW5pdGlvbi5Jc011bHRpQXR0YWNoICYmIHRoaXMuZmllbGQuRW5hYmxlTWFya2luZyAmJiB0aGlzLnJlY29yZFt0aGlzLmZpZWxkLk5hbWVdICYmIHRoaXMucmVjb3JkW3RoaXMuZmllbGQuTmFtZV1bMF0pIHtcclxuICAgICAgdGhpcy5nZXRJbWFnZURpbWVuc2lvbnMoKTtcclxuICAgICAgdGhpcy5kaWFsb2dJbWFnZURyb3dpbmdSZWYgPSB0aGlzLmRpYWxvZy5vcGVuKHRoaXMuZGlhbG9nSW1hZ2VEcm93aW5nLCB7XHJcbiAgICAgICAgZGlzYWJsZUNsb3NlOiB0cnVlLFxyXG4gICAgICAgIGhhc0JhY2tkcm9wOiB0cnVlLFxyXG4gICAgICAgIHdpZHRoOiAnNzUlJ1xyXG4gICAgICB9KTtcclxuICAgIH1cclxuICAgICovXHJcbiAgfVxyXG5cclxuICAvKipcclxuICAgKiBNZXRvZG8gaW52b2NhdG8gYWwgc2FsdmF0YWdnaW8gZGVsbCdpbW1hZ2luZSBkYWwgY29tcG9uZW50ZSBpbWFnZS1NYXJraW5nLlxyXG4gICAqIFRyYXNmb3JtYSBpbCBCbG9iIHJpY2V2dXRvIGluIHVuIGJhc2U2NCwgYWdnaW9ybmEgaWwgRmlsZUNvbnRldFR5cGUgZSBpbCBGaWxlRGF0YUJhc2U2NFxyXG4gICAqIGRlbGwnYWxsZWdhdG8gY2FyaWNhdG8gZSBjaGl1ZGUgaWwgZGlhbG9nIGluIGN1aSBkaXNlZ25hcmUuXHJcbiAgICogQHBhcmFtIE1hcmtuUGljdHVyZSBCbG9iIGRlbGwnaW1tYWdpbmUgZGlzZWduYXRhLlxyXG4gICAqL1xyXG4gIGFzeW5jIHNhdmVNYXJrKG1hcmtlZEltYWdlOiBNYXJrZWRJbWFnZSkge1xyXG4gICAgbGV0IGJhc2U2NEZpbGUgPSBhd2FpdCB0b0Jhc2U2NChtYXJrZWRJbWFnZS5pbWFnZSk7XHJcbiAgICBpZiAoYmFzZTY0RmlsZSkge1xyXG4gICAgICB0aGlzLk1hcmtGaWxlQmFzZTY0ID0gYmFzZTY0RmlsZS5zcGxpdChcIixcIilbMV07XHJcbiAgICAgIHRoaXMuRmlsZURhdGFCYXNlNjQgPSB0aGlzLk1hcmtGaWxlQmFzZTY0O1xyXG5cclxuICAgICAgLy8gQ2xvbm8gbCdhdHRhY2htZW50IG9yaWdpbmFsZSwgZ2xpIGNhbWJpbyBpbCBGaWxlQmFzZSBlIGdsaSBhenplcm8gbCdJRFxyXG4gICAgICB0aGlzLmltYWdlTWFya0F0dGFjaG1lbnQuRmlsZURhdGFCYXNlNjQgPSB0aGlzLk1hcmtGaWxlQmFzZTY0O1xyXG4gICAgICB0aGlzLmltYWdlTWFya0F0dGFjaG1lbnQuRXh0cmFKU09OID0gSlNPTi5zdHJpbmdpZnkobWFya2VkSW1hZ2Uub3V0bWFya2Vycyk7XHJcblxyXG4gICAgICB0aGlzLmltYWdlTWFyay5lbWl0KHRoaXMuaW1hZ2VNYXJrQXR0YWNobWVudCk7XHJcblxyXG4gICAgfVxyXG5cclxuICAgIGlmICh0aGlzLmRpYWxvZ0ltYWdlTWFya2luZ1JlZikge1xyXG4gICAgICB0aGlzLmNsb3NlRGlhbG9nKCk7XHJcbiAgICB9XHJcbiAgfVxyXG5cclxuICAvKipcclxuICAgKiBNZXRvZG8gcGVyIGNoaXVkZXJlIGlsIGRpYWxvZyBpbiBjdWkgZGlzZWduYXJlIHN1bGwnaW1tYWdpbmUgY2FyaWF0YSBlIHN2dW90YXJlIGxlIGRpbWVuc2lvbmkuXHJcbiAgICovXHJcbiAgY2xvc2VEaWFsb2coKSB7XHJcbiAgICB0aGlzLmRpYWxvZ0ltYWdlTWFya2luZ1JlZi5jbG9zZSgpO1xyXG4gICAgdGhpcy53aWR0aCA9IG51bGw7XHJcbiAgICB0aGlzLmhlaWdodCA9IG51bGw7XHJcbiAgfVxyXG5cclxuICAvKipcclxuICAgKiBNZXRvZG8gcGVyIHJlY3VwZXJhcmUgbGUgZGltZW5zaW9uaSBkZWxsJ2ltbWFnaW5lIGNhcmljYXRhIHBlciBtb3N0cmFyZSBjb3JyZXR0YW1lbnRlXHJcbiAgICogaWwgY29tcG9uZW50ZSBpbWFnZS1NYXJraW5nLiBTZSBub24gdmllbmUgZXNlZ3VpdG8gcXVlc3RvIHBhc3NhZ2dpbyBsJ2FyZWEgaW4gY3VpIGRpc2VnbmFyZSBcclxuICAgKiDDqCBwacO5IGdyYW5kZSBkZWxsJ2ltbWFnaW5lIHN0ZXNzYSBlIGFsIHNhbHZhdGFnZ2lvIGkgdHJhdHRpIGRpc2VnbmF0aSB2ZW5nb25vIHNwb3N0YXRpLiBcclxuICAgKi9cclxuICBwcml2YXRlIGdldEltYWdlRGltZW5zaW9ucygpIHtcclxuICAgIHZhciBibG9iOiBCbG9iID0gdGhpcy5nZXRCbG9iRnJvbUJhc2U2NCgpO1xyXG4gICAgY29uc3QgcmVhZGVyID0gbmV3IEZpbGVSZWFkZXIoKTtcclxuICAgIHJlYWRlci5vbmxvYWQgPSAoZTogYW55KSA9PiB7XHJcbiAgICAgIGNvbnN0IGltYWdlID0gbmV3IEltYWdlKCk7XHJcbiAgICAgIGltYWdlLnNyYyA9IGUudGFyZ2V0LnJlc3VsdDtcclxuICAgICAgaW1hZ2Uub25sb2FkID0gcnMgPT4ge1xyXG4gICAgICAgIHRoaXMuaGVpZ2h0ID0gcnMuY3VycmVudFRhcmdldFsnaGVpZ2h0J107XHJcbiAgICAgICAgdGhpcy53aWR0aCA9IHJzLmN1cnJlbnRUYXJnZXRbJ3dpZHRoJ107XHJcbiAgICAgIH07XHJcbiAgICB9O1xyXG4gICAgcmVhZGVyLnJlYWRBc0RhdGFVUkwoYmxvYik7XHJcbiAgfVxyXG5cclxuICAvKipcclxuICAgKiBNZXRvZG8gcGVyIGNvbnZlcnRpcmUgaWwgYmFzZTY0IGRpIHVuJ2ltbWFnaW5lIGluIHVuIEJsb2IuXHJcbiAgICogTmVjZXNzYXJpbyBwZXIgaWwgcmVjdXBlcm8gZGVsbGUgZGltZW5zaW9uaSBkZWxsJ2ltbWFnaW5lIGNhcmljYXRhIGEgcGFydGlyZSBkYWwgc3VvIGJhc2U2NC5cclxuICAgKiBWaWVuZSBlc2VndWl0byBzb2xvIG5lbCBjYXNvIGluIGN1aSBsJ0ltYWdlRmllbGQgbm9uIGFtbWV0dGUgYWxsZWdhdGkgbXVsdGlwbGkuXHJcbiAgICogQHJldHVybnMgUmVzdGl0dWlzY2UgaWwgQmxvYiBkZWxsJ2ltbWFnaW5lIGNhcmljYXRhLlxyXG4gICAqL1xyXG4gIHByaXZhdGUgZ2V0QmxvYkZyb21CYXNlNjQoKSB7XHJcbiAgICBjb25zdCBieXRlQ2hhcmFjdGVycyA9IGF0b2IodGhpcy5yZWNvcmRbdGhpcy5maWVsZC5OYW1lXVswXS5GaWxlRGF0YUJhc2U2NCk7XHJcbiAgICBjb25zdCBieXRlTnVtYmVycyA9IG5ldyBBcnJheShieXRlQ2hhcmFjdGVycy5sZW5ndGgpO1xyXG4gICAgZm9yIChsZXQgaSA9IDA7IGkgPCBieXRlQ2hhcmFjdGVycy5sZW5ndGg7IGkrKykge1xyXG4gICAgICBieXRlTnVtYmVyc1tpXSA9IGJ5dGVDaGFyYWN0ZXJzLmNoYXJDb2RlQXQoaSk7XHJcbiAgICB9XHJcbiAgICBjb25zdCBieXRlQXJyYXkgPSBuZXcgVWludDhBcnJheShieXRlTnVtYmVycyk7XHJcbiAgICByZXR1cm4gbmV3IEJsb2IoW2J5dGVBcnJheV0sIHsgdHlwZTogdGhpcy5yZWNvcmRbdGhpcy5maWVsZC5OYW1lXVswXS5GaWxlQ29udGVudFR5cGUgfSk7XHJcbiAgfVxyXG5cclxuICAvKipcclxuICAgKiBNZXRvZG8gcGVyIGVtZXR0ZXJlIGwnZXZlbnRvIGNoZSBpbCB2YWxvcmUgZGVsIHJlY29yZCDDqCBjYW1iaWF0by5cclxuICAgKi9cclxuICBwcml2YXRlIG9uUmVjb3JkVmFsdWVDaGFuZ2UoKSB7XHJcbiAgICBpZiAoIXRoaXMuZmllbGQuRm9ybXVsYSkge1xyXG4gICAgICB0aGlzLnJlY29yZENoYW5nZS5lbWl0KHRoaXMucmVjb3JkKTtcclxuICAgIH1cclxuICB9XHJcblxyXG4gIHB1YmxpYyBvcGVuTWFyaygpe1xyXG4gICAgdGhpcy5kaWFsb2dJbWFnZU1hcmtpbmdSZWYgPSB0aGlzLmRpYWxvZy5vcGVuKHRoaXMuZGlhbG9nSW1hZ2VNYXJraW5nLCB7XHJcbiAgICAgIGRpc2FibGVDbG9zZTogdHJ1ZSxcclxuICAgICAgaGFzQmFja2Ryb3A6IHRydWUsXHJcbiAgICAgIHdpZHRoOiAnNzUlJ1xyXG4gICAgfSk7XHJcblxyXG4gICAgaWYgKHRoaXMuQXR0YWNobWVudElEICE9IG51bGwgJiYgTnVtYmVyKHRoaXMuQXR0YWNobWVudElEKSA+IDApe1xyXG4gICAgICB0aGlzLmxvYWQodGhpcy5BdHRhY2htZW50SUQsW0ltYWdlRm9sZGVyRW51bS5SZWd1bGFyXSk7XHJcbiAgICB9IGVsc2Uge1xyXG4gICAgICB0aGlzLkZpbGVCYXNlNjQgPSB0aGlzLkZpbGVEYXRhQmFzZTY0O1xyXG4gICAgICB0aGlzLk1hcmtpbmdMb2FkZWQgPSB0cnVlO1xyXG4gICAgfVxyXG5cclxuICB9XHJcblxyXG4gIGxvYWQoSUQsVHlwZXMpe1xyXG5cclxuICAgIHRoaXMuTWFya2luZ0xvYWRlZCA9IGZhbHNlO1xyXG5cclxuICAgICAgLy8gcHJlcGFybyBsYSB2ZXJzaW9uZSBjb3JyZXR0YSBkZWwgZm9ybVxyXG4gICAgICB2YXIgZHluYW1pY01vZHVsZVBhcmFtcyA9IFtcclxuICAgICAgICB7XHJcbiAgICAgICAgICBQYXJhbU5hbWU6IFwiQXR0YWNobWVudElEXCIsXHJcbiAgICAgICAgICBQYXJhbVZhbHVlOiBJRCxcclxuICAgICAgICAgIFBhcmFtVHlwZTogUGFyYW1UeXBlRW51bVtcIlF1ZXJ5IHBhcmFtXCJdLFxyXG4gICAgICAgIH0sXHJcbiAgICAgICAge1xyXG4gICAgICAgICAgUGFyYW1OYW1lOiBcIlR5cGVzXCIsXHJcbiAgICAgICAgICBQYXJhbVZhbHVlOiBUeXBlcyxcclxuICAgICAgICAgIFBhcmFtVHlwZTogUGFyYW1UeXBlRW51bVtcIlF1ZXJ5IHBhcmFtXCJdLFxyXG4gICAgICAgIH0sXHJcbiAgICAgIF07ICAgIFxyXG4gICAgICBcclxuICAgIHRoaXMudXRpbGl0eUhlbHBlclNlcnZpY2UuUnVuRW5kUG9pbnRDYWxsKFxyXG4gICAgICB0aGlzLmVuZFBvaW50Q29uZmlndXJhdGlvbi5Gb3Jtcy5HZXRBdHRhY2htZW50V2l0aEJhc2VzNjQsXHJcbiAgICAgIGR5bmFtaWNNb2R1bGVQYXJhbXMsXHJcbiAgICAgIChyZXM6IER5bkF0dGFjaG1lbnQpID0+IHtcclxuXHJcbiAgICAgICAgdGhpcy5GaWxlQmFzZTY0ID0gcmVzLkZpbGVCYXNlczY0W0ltYWdlRm9sZGVyRW51bS5SZWd1bGFyXTtcclxuICAgICAgICB0aGlzLk1hcmtpbmdMb2FkZWQgPSB0cnVlO1xyXG5cclxuICAgICAgfSxcclxuICAgICAgKGVycikgPT4ge1xyXG5cclxuICAgICAgfVxyXG4gICAgKTsgICAgXHJcbiAgfVxyXG5cclxufVxyXG4iLCI8ZGl2IGNsYXNzPVwicm93IGltYWdlQ29udGFpbmVyXCIgc3R5bGU9XCJ3aWR0aDoge3tpbWFnZVdpZHRofX1weDtcIj5cclxuICAgIDxkaXYgY2xhc3M9XCJyb3dcIj5cclxuICAgICAgICAgPGRpdiBjbGFzcz1cImNvbC0xMlwiPnt7ZmllbGQuRGVzY3JpcHRpb259fTwvZGl2PlxyXG4gICAgICAgICA8ZGl2IGNsYXNzPVwiY29sLTEyXCI+PGltZyAqbmdJZj1cImZpZWxkLkR5bkF0dGFjaG1lbnRcIlxyXG4gICAgICAgICAgICAgIHNyYz1cImRhdGE6e3tmaWVsZC5EeW5BdHRhY2htZW50LkZpbGVDb250ZW50VHlwZX19O2Jhc2U2NCx7e0ZpbGVEYXRhQmFzZTY0fX1cIlxyXG4gICAgICAgICAgICAgIFtoZWlnaHRdPVwiZmllbGQuRHluQXR0YWNobWVudC5SZXNpemVkSW1hZ2VIZWlnaHRQeCAhPSBudWxsID8gZmllbGQuRHluQXR0YWNobWVudC5SZXNpemVkSW1hZ2VIZWlnaHRQeCA6IDEyOFwiXHJcbiAgICAgICAgICAgICAgY2xhc3M9XCJzaW5nbGVJbWFnZVwiPjwvZGl2PlxyXG4gICAgICAgICAgICAgIDxkaXYgY2xhc3M9XCJjb2wtMTIgZHJhd0J1dHRvblwiPlxyXG4gICAgICAgICAgICAgICAgICAgPGJ1dHRvbiBjbGFzcz1cImJ0biBidG4tcHJpbWFyeSBidG4tdzEwMFwiIChjbGljayk9XCJvcGVuTWFyaygpXCI+TWFyY2E8L2J1dHRvbj5cclxuICAgICAgICAgICAgICA8L2Rpdj5cclxuICAgICAgICAgICAgICA8ZGl2IGNsYXNzPVwiY29sLTEyIGZsZXgganVzdGlmeS1jb250ZW50LWFyb3VuZCBidXR0b25rZXlcIiAqbmdJZj1cImZpZWxkLkR5bkF0dGFjaG1lbnQuQnV0dG9uS2V5ICE9IG51bGwgJiYgZmllbGQuRHluQXR0YWNobWVudC5CdXR0b25LZXkgIT0gJydcIj5cclxuICAgICAgICAgICAgICAgICAgIDxkaXY+RXRpY2hldHRhPC9kaXY+PGRpdj57e2ZpZWxkLkR5bkF0dGFjaG1lbnQuQnV0dG9uS2V5fX08L2Rpdj5cclxuICAgICAgICAgICAgICA8L2Rpdj5cclxuICAgICAgICAgICAgICA8ZGl2IGNsYXNzPVwiY29sLTEyIGZsZXgganVzdGlmeS1jb250ZW50LWFyb3VuZCBidXR0b252YWx1ZVwiICpuZ0lmPVwiZmllbGQuRHluQXR0YWNobWVudC5CdXR0b25WYWx1ZSAhPSBudWxsICYmIGZpZWxkLkR5bkF0dGFjaG1lbnQuQnV0dG9uVmFsdWUgIT0gJydcIj4gXHJcbiAgICAgICAgICAgICAgICAgICA8ZGl2PlZhbG9yZTwvZGl2PjxkaXY+e3tmaWVsZC5EeW5BdHRhY2htZW50LkJ1dHRvblZhbHVlfX08L2Rpdj5cclxuICAgICAgICAgICAgICA8L2Rpdj5cclxuICAgIDwvZGl2PiAgIFxyXG48L2Rpdj5cclxuXHJcbjwhLS0gRElBTE9HIFBFUiBESVNFR05BUkUgU1VMIE5VT1ZPIEFMTEVHQVRPIENBUklDQVRPIC0tPlxyXG48bmctdGVtcGxhdGUgI2RpYWxvZ0ltYWdlTWFya2luZz5cclxuICAgIDxkaXYgY2xhc3M9XCJwYWRkZXJcIj5cclxuICAgICAgICAgPGRpdlxyXG4gICAgICAgICAgICAgIGNsYXNzPVwicm93XCI+XHJcbiAgICAgICAgICAgICAgPGRpdiBjbGFzcz1cImNvbC1zbS0xMiBjb2wtbWQtMTJcIj5cclxuICAgICAgICAgICAgICAgICAgIDx0bXctaW1hZ2UtbWFya2VyXHJcbiAgICAgICAgICAgICAgICAgICAqbmdJZj1cIk1hcmtpbmdMb2FkZWRcIlxyXG4gICAgICAgICAgICAgICAgICAgW3NyY109XCInZGF0YTonKyBmaWVsZC5EeW5BdHRhY2htZW50LkZpbGVDb250ZW50VHlwZSArICc7YmFzZTY0LCcrIEZpbGVCYXNlNjRcIiBbaW5Db25maWddPVwiaW5Db25maWdcIlxyXG4gICAgICAgICAgICAgICAgICAgW3dpZHRoXT1cImZpZWxkLkR5bkF0dGFjaG1lbnQuSW1hZ2VXaWR0aFB4XCIgW2hlaWdodF09XCJmaWVsZC5EeW5BdHRhY2htZW50LkltYWdlSGVpZ2h0UHhcIiBcclxuICAgICAgICAgICAgICAgICAgIChzYXZlKT1cInNhdmVNYXJrKCRldmVudClcIiAoY2xvc2VEaWFsb2cpPVwiY2xvc2VEaWFsb2coKVwiPlxyXG4gICAgICAgICAgICAgICAgICAgPC90bXctaW1hZ2UtbWFya2VyPlxyXG4gICAgICAgICAgICAgIDwvZGl2PlxyXG4gICAgICAgICA8L2Rpdj5cclxuICAgIDwvZGl2PlxyXG48L25nLXRlbXBsYXRlPiJdfQ==