ng-prime-tools 1.0.4 → 1.0.5

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/README.md CHANGED
@@ -12,6 +12,10 @@ An advanced PrimeNG tools library for Angular.
12
12
  - [Inputs](#inputs)
13
13
  - [Outputs](#outputs)
14
14
  - [Example](#example)
15
+ - [multi-search-criteria](#multi-search-criteria)
16
+ - [Inputs](#inputs-1)
17
+ - [Outputs](#outputs-1)
18
+ - [Example](#example-1)
15
19
  - [Changelog](#changelog)
16
20
  - [License](#license)
17
21
 
@@ -186,8 +190,216 @@ export class AppComponent implements OnInit {
186
190
  <ng-advanced-prime-table [columns]="columns" [data]="data" [hasSearchFilter]="true" [hasColumnFilter]="true" [isPaginated]="true" [totalRecords]="totalRecords" [isSortable]="true"></ng-advanced-prime-table>
187
191
  ```
188
192
 
193
+ ### multi-search-criteria
194
+
195
+ To use the `multi-search-criteria` component, follow these steps:
196
+
197
+ 1. **Import the Module:**
198
+
199
+ Import the `MultiSearchCriteriaModule` in your Angular module or standalone component.
200
+
201
+ ```typescript
202
+ import { MultiSearchCriteriaModule, SearchCriteria, SearchCriteriaTypeEnum } from "ng-prime-tools";
203
+ ```
204
+
205
+ 2. **Add the Component:**
206
+
207
+ Add the `multi-search-criteria` component to your template:
208
+
209
+ ```html
210
+ <multi-search-criteria [title]="'Multi-Criteria Search'" [criteria]="criteria" [data]="data" [mode]="'dynamic'" (filteredData)="onFilteredData($event)" (searchCriteria)="onSearchData($event)"></multi-search-criteria>
211
+ ```
212
+
213
+ 3. **Define the Inputs:**
214
+
215
+ In your component, define the necessary inputs for the criteria:
216
+
217
+ ```typescript
218
+ import { Component, OnInit } from "@angular/core";
219
+ import { CommonModule } from "@angular/common";
220
+ import { MultiSearchCriteriaModule, SearchCriteria, SearchCriteriaTypeEnum, FilterOption } from "ng-prime-tools";
221
+
222
+ @Component({
223
+ selector: "app-root",
224
+ standalone: true,
225
+ imports: [CommonModule, MultiSearchCriteriaModule],
226
+ templateUrl: "./app.component.html",
227
+ styleUrls: ["./app.component.css"],
228
+ })
229
+ export class AppComponent implements OnInit {
230
+ criteria: SearchCriteria[] = [];
231
+ data: any[] = [];
232
+ filteredData: any[] = [];
233
+ totalRecords: number = 0;
234
+
235
+ ngOnInit(): void {
236
+ this.criteria = [
237
+ {
238
+ title: "Name",
239
+ code: "name",
240
+ type: SearchCriteriaTypeEnum.STRING,
241
+ },
242
+ {
243
+ title: "Date Range",
244
+ code: "dateRange",
245
+ type: SearchCriteriaTypeEnum.DATERANGE,
246
+ },
247
+ {
248
+ title: "Amount",
249
+ code: "amount",
250
+ type: SearchCriteriaTypeEnum.AMOUNT,
251
+ currency: "USD",
252
+ },
253
+ {
254
+ title: "Type",
255
+ code: "type",
256
+ type: SearchCriteriaTypeEnum.MULTISELECT,
257
+ filterOptions: [
258
+ { label: "Invoice", value: "Invoice" },
259
+ { label: "Bill", value: "Bill" },
260
+ ],
261
+ },
262
+ ];
263
+
264
+ this.data = [
265
+ { name: "ZAK JAAI", date: "11/06/2024", amount: 100, type: "Invoice" },
266
+ { name: "ZAK2 JAAI", date: "20/06/2024", amount: 200, type: "Bill" },
267
+ ];
268
+
269
+ this.filteredData = [...this.data];
270
+ this.totalRecords = this.data.length;
271
+ }
272
+
273
+ onFilteredData(filteredData: any[]): void {
274
+ this.filteredData = filteredData;
275
+ this.totalRecords = filteredData.length;
276
+ }
277
+
278
+ onSearchData(searchCriteria: { [key: string]: any }): void {
279
+ console.log(searchCriteria);
280
+ }
281
+ }
282
+ ```
283
+
284
+ ### Inputs
285
+
286
+ - **title** (`
287
+
288
+ string`): The title of the multi-search criteria panel.
289
+
290
+ - **criteria** (`SearchCriteria[]`): The search criteria configuration.
291
+ - **data** (`any[]`): The data to be filtered (used in static mode).
292
+ - **mode** (`'static' | 'dynamic'`): The mode of the multi-search criteria component.
293
+ - `'static'`: Filters the data directly within the component.
294
+ - `'dynamic'`: Emits the search criteria to be handled externally (e.g., for server-side filtering).
295
+
296
+ ### Outputs
297
+
298
+ - **filteredData** (`EventEmitter<any[]>`): Emitted with the filtered data (only in static mode).
299
+ - **searchCriteria** (`EventEmitter<{ [key: string]: any }>`): Emitted with the search criteria (only in dynamic mode).
300
+
301
+ ### Example
302
+
303
+ Here's a complete example of how to use the `multi-search-criteria` component in dynamic mode:
304
+
305
+ ```typescript
306
+ import { Component, OnInit } from "@angular/core";
307
+ import { CommonModule } from "@angular/common";
308
+ import { MultiSearchCriteriaModule, SearchCriteria, SearchCriteriaTypeEnum, FilterOption } from "ng-prime-tools";
309
+
310
+ @Component({
311
+ selector: "app-root",
312
+ standalone: true,
313
+ imports: [CommonModule, MultiSearchCriteriaModule],
314
+ templateUrl: "./app.component.html",
315
+ styleUrls: ["./app.component.css"],
316
+ })
317
+ export class AppComponent implements OnInit {
318
+ criteria: SearchCriteria[] = [];
319
+ data: any[] = [];
320
+ filteredData: any[] = [];
321
+ totalRecords: number = 0;
322
+
323
+ ngOnInit(): void {
324
+ this.criteria = [
325
+ {
326
+ title: "Name",
327
+ code: "name",
328
+ type: SearchCriteriaTypeEnum.STRING,
329
+ },
330
+ {
331
+ title: "Date Range",
332
+ code: "dateRange",
333
+ type: SearchCriteriaTypeEnum.DATERANGE,
334
+ },
335
+ {
336
+ title: "Amount",
337
+ code: "amount",
338
+ type: SearchCriteriaTypeEnum.AMOUNT,
339
+ currency: "USD",
340
+ },
341
+ {
342
+ title: "Type",
343
+ code: "type",
344
+ type: SearchCriteriaTypeEnum.MULTISELECT,
345
+ filterOptions: [
346
+ { label: "Invoice", value: "Invoice" },
347
+ { label: "Bill", value: "Bill" },
348
+ ],
349
+ },
350
+ ];
351
+
352
+ this.data = [
353
+ { name: "ZAK JAAI", date: "11/06/2024", amount: 100, type: "Invoice" },
354
+ { name: "ZAK2 JAAI", date: "20/06/2024", amount: 200, type: "Bill" },
355
+ ];
356
+
357
+ this.filteredData = [...this.data];
358
+ this.totalRecords = this.data.length;
359
+ }
360
+
361
+ onFilteredData(filteredData: any[]): void {
362
+ this.filteredData = filteredData;
363
+ this.totalRecords = filteredData.length;
364
+ }
365
+
366
+ onSearchData(searchCriteria: { [key: string]: any }): void {
367
+ console.log(searchCriteria);
368
+ // Perform server-side search or any other external data fetching logic
369
+ }
370
+ }
371
+ ```
372
+
373
+ ```html
374
+ <multi-search-criteria [title]="'Multi-Criteria Search'" [criteria]="criteria" [data]="data" [mode]="'dynamic'" (filteredData)="onFilteredData($event)" (searchCriteria)="onSearchData($event)"></multi-search-criteria>
375
+
376
+ <ng-advanced-prime-table [columns]="columns" [data]="filteredData" [hasSearchFilter]="true" [hasColumnFilter]="true" [isPaginated]="true" [totalRecords]="totalRecords" [isSortable]="true"></ng-advanced-prime-table>
377
+ ```
378
+
189
379
  ## Changelog
190
380
 
381
+ ### Version 1.0.5
382
+
383
+ #### New Features
384
+
385
+ - **`multi-search-criteria` Component**
386
+
387
+ - Added support for `mode` input to handle dynamic and static filtering.
388
+ - `mode` input can be set to `'dynamic'` or `'static'`. When set to `'dynamic'`, the component emits search criteria for backend filtering. When set to `'static'`, the component filters data locally.
389
+ - Fixed the bug in the `selectAll` functionality for multi-select filters.
390
+ - Users can select or deselect all options in a multi-select filter.
391
+
392
+ - **`ng-advanced-prime-table` Component**
393
+ - Added support for customizable amount formatting.
394
+ - New inputs for columns of type `AMOUNT`:
395
+ - `thousandSeparator` (`comma` or `space`): Specifies the character used for thousands separator.
396
+ - `decimalSeparator` (`comma` or `dot`): Specifies the character used for decimal separator.
397
+ - Updated the `customCurrency` pipe to handle optional currency display and customizable separators.
398
+
399
+ ### Version 1.0.4
400
+
401
+ - Added `multi-search-criteria` component.
402
+
191
403
  ### Version 1.0.3
192
404
 
193
405
  - Initial release with `ng-advanced-prime-table` component.
@@ -1,2 +1,2 @@
1
1
  export {};
2
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidGFibGUtY29sdW1uLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvbmctcHJpbWUtdG9vbHMvc3JjL2xpYi9tb2RlbHMvdGFibGUtY29sdW1uLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiIiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBUYWJsZVR5cGVFbnVtIH0gZnJvbSAnLi4vZW51bXMvdGFibGUtdHlwZS1lbnVtJztcblxuZXhwb3J0IGludGVyZmFjZSBUYWJsZUNvbHVtbiB7XG4gIHRpdGxlOiBzdHJpbmc7XG4gIGNvZGU/OiBzdHJpbmc7XG4gIHR5cGU/OiBUYWJsZVR5cGVFbnVtO1xuICBvcHRpb25zPzogYW55W107XG4gIGlzRWRpdGFibGU/OiBib29sZWFuO1xuICBpc0ZpbHRlcj86IGJvb2xlYW47XG4gIGRhdGVGb3JtYXQ/OiBzdHJpbmc7XG4gIGN1cnJlbmN5Pzogc3RyaW5nO1xuICBmaWx0ZXJPcHRpb25zPzogYW55W107XG4gIGRlY2ltYWxQbGFjZXM/OiBudW1iZXI7XG4gIGlzU29ydGFibGU/OiBib29sZWFuO1xufVxuIl19
2
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidGFibGUtY29sdW1uLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvbmctcHJpbWUtdG9vbHMvc3JjL2xpYi9tb2RlbHMvdGFibGUtY29sdW1uLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiIiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBUYWJsZVR5cGVFbnVtIH0gZnJvbSAnLi4vZW51bXMvdGFibGUtdHlwZS1lbnVtJztcblxuZXhwb3J0IGludGVyZmFjZSBUYWJsZUNvbHVtbiB7XG4gIHRpdGxlOiBzdHJpbmc7XG4gIGNvZGU/OiBzdHJpbmc7XG4gIHR5cGU/OiBUYWJsZVR5cGVFbnVtO1xuICBvcHRpb25zPzogYW55W107XG4gIGlzRWRpdGFibGU/OiBib29sZWFuO1xuICBpc0ZpbHRlcj86IGJvb2xlYW47XG4gIGRhdGVGb3JtYXQ/OiBzdHJpbmc7XG4gIGN1cnJlbmN5Pzogc3RyaW5nO1xuICBmaWx0ZXJPcHRpb25zPzogYW55W107XG4gIGRlY2ltYWxQbGFjZXM/OiBudW1iZXI7XG4gIGlzU29ydGFibGU/OiBib29sZWFuO1xuICB0aG91c2FuZFNlcGFyYXRvcj86ICdjb21tYScgfCAnc3BhY2UnO1xuICBkZWNpbWFsU2VwYXJhdG9yPzogJ2NvbW1hJyB8ICdkb3QnO1xufVxuIl19
@@ -17,7 +17,9 @@ export class MultiSearchCriteriaComponent {
17
17
  this.criteria = [];
18
18
  this.inputsPerRow = 3;
19
19
  this.data = [];
20
+ this.mode = 'static'; // Add mode input
20
21
  this.filteredData = new EventEmitter();
22
+ this.searchCriteria = new EventEmitter();
21
23
  this.selectAll = false;
22
24
  this.selected = [];
23
25
  }
@@ -34,48 +36,56 @@ export class MultiSearchCriteriaComponent {
34
36
  return input.currency || 'EUR';
35
37
  }
36
38
  search() {
37
- console.log('Original Data:', this.data);
38
- const filtered = this.data.filter((item) => {
39
- return this.criteria.every((criterion) => {
40
- console.log('Criterion:', criterion);
41
- if (criterion.type === SearchCriteriaTypeEnum.DATERANGE) {
42
- const [startDate, endDate] = criterion.value || [
43
- undefined,
44
- undefined,
45
- ];
46
- const itemDate = this.parseDate(item[criterion.code] || item['date']);
47
- console.log('Item Date:', itemDate, 'Start Date:', startDate, 'End Date:', endDate);
48
- if (!startDate && !endDate) {
49
- return true;
39
+ if (this.mode === 'dynamic') {
40
+ const criteriaValues = {};
41
+ this.criteria.forEach((criterion) => {
42
+ if (criterion.value !== null && criterion.value !== undefined) {
43
+ criteriaValues[criterion.code] = criterion.value;
44
+ }
45
+ });
46
+ this.searchCriteria.emit(criteriaValues);
47
+ }
48
+ else {
49
+ const filtered = this.data.filter((item) => {
50
+ return this.criteria.every((criterion) => {
51
+ if (criterion.type === SearchCriteriaTypeEnum.DATERANGE) {
52
+ const [startDate, endDate] = criterion.value || [
53
+ undefined,
54
+ undefined,
55
+ ];
56
+ const itemDate = this.parseDate(item[criterion.code] || item['date']);
57
+ if (!startDate && !endDate) {
58
+ return true;
59
+ }
60
+ if (itemDate === null) {
61
+ return false;
62
+ }
63
+ const parsedStartDate = startDate
64
+ ? this.parseDate(startDate)
65
+ : null;
66
+ const parsedEndDate = endDate ? this.parseDate(endDate) : null;
67
+ return ((!parsedStartDate || itemDate >= parsedStartDate) &&
68
+ (!parsedEndDate || itemDate <= parsedEndDate));
50
69
  }
51
- if (itemDate === null) {
52
- return false;
70
+ else if (criterion.type === SearchCriteriaTypeEnum.AMOUNT) {
71
+ return !criterion.value || item[criterion.code] === criterion.value;
53
72
  }
54
- const parsedStartDate = startDate ? this.parseDate(startDate) : null;
55
- const parsedEndDate = endDate ? this.parseDate(endDate) : null;
56
- console.log('Parsed Start Date:', parsedStartDate, 'Parsed End Date:', parsedEndDate);
57
- return ((!parsedStartDate || itemDate >= parsedStartDate) &&
58
- (!parsedEndDate || itemDate <= parsedEndDate));
59
- }
60
- else if (criterion.type === SearchCriteriaTypeEnum.AMOUNT) {
61
- return !criterion.value || item[criterion.code] === criterion.value;
62
- }
63
- else if (criterion.type === SearchCriteriaTypeEnum.MULTISELECT) {
64
- return (!criterion.value ||
65
- criterion.value.some((option) => option.label === item[criterion.code]));
66
- }
67
- else if (criterion.type === SearchCriteriaTypeEnum.STRING) {
68
- return (!criterion.value ||
69
- item[criterion.code]
70
- .toString()
71
- .toLowerCase()
72
- .includes(criterion.value.toString().toLowerCase()));
73
- }
74
- return true;
73
+ else if (criterion.type === SearchCriteriaTypeEnum.MULTISELECT) {
74
+ return (!criterion.value ||
75
+ criterion.value.some((option) => option.label === item[criterion.code]));
76
+ }
77
+ else if (criterion.type === SearchCriteriaTypeEnum.STRING) {
78
+ return (!criterion.value ||
79
+ item[criterion.code]
80
+ .toString()
81
+ .toLowerCase()
82
+ .includes(criterion.value.toString().toLowerCase()));
83
+ }
84
+ return true;
85
+ });
75
86
  });
76
- });
77
- console.log('Filtered Data:', filtered);
78
- this.filteredData.emit(filtered);
87
+ this.filteredData.emit(filtered);
88
+ }
79
89
  }
80
90
  clear() {
81
91
  this.criteria.forEach((input) => {
@@ -83,13 +93,9 @@ export class MultiSearchCriteriaComponent {
83
93
  });
84
94
  this.filteredData.emit(this.data);
85
95
  }
86
- onSelectAllChange(event) {
96
+ onSelectAllChange(event, criterion) {
87
97
  this.selectAll = event.checked;
88
- this.selected = event.checked
89
- ? [
90
- ...(this.criteria.find((criterion) => criterion.type === SearchCriteriaTypeEnum.MULTISELECT)?.filterOptions || []),
91
- ]
92
- : [];
98
+ criterion.value = event.checked ? [...(criterion.filterOptions ?? [])] : [];
93
99
  }
94
100
  parseDate(dateString) {
95
101
  if (!dateString) {
@@ -109,11 +115,11 @@ export class MultiSearchCriteriaComponent {
109
115
  return null;
110
116
  }
111
117
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.11", ngImport: i0, type: MultiSearchCriteriaComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
112
- static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "17.3.11", type: MultiSearchCriteriaComponent, selector: "multi-search-criteria", inputs: { title: "title", criteria: "criteria", inputsPerRow: "inputsPerRow", data: "data" }, outputs: { filteredData: "filteredData" }, ngImport: i0, template: "<p-panel header=\"{{ title }}\" [toggleable]=\"true\" [collapsed]=\"true\">\n <div\n class=\"criteria-container\"\n [ngStyle]=\"{ 'grid-template-columns': 'repeat(' + inputsPerRow + ', 1fr)' }\"\n >\n <div *ngFor=\"let input of criteria\" class=\"criteria-item\">\n <label class=\"bsc-label\">{{ input.title }}</label>\n\n <ng-container [ngSwitch]=\"input.type\">\n <p-calendar\n *ngSwitchCase=\"SearchCriteriaTypeEnum.DATERANGE\"\n [(ngModel)]=\"input.value\"\n selectionMode=\"range\"\n [dateFormat]=\"'dd/mm/yy'\"\n class=\"full-width-input\"\n [showIcon]=\"true\"\n (ngModelChange)=\"input.value = $event\"\n ></p-calendar>\n <p-calendar\n *ngSwitchCase=\"SearchCriteriaTypeEnum.DATE\"\n [ngModel]=\"input.value\"\n (ngModelChange)=\"input.value = $event\"\n [dateFormat]=\"'dd/mm/yy'\"\n [showIcon]=\"true\"\n class=\"full-width-input\"\n ></p-calendar>\n <input\n *ngSwitchCase=\"SearchCriteriaTypeEnum.STRING\"\n type=\"text\"\n pInputText\n [(ngModel)]=\"input.value\"\n class=\"full-width-input\"\n />\n\n <p-inputNumber\n *ngSwitchCase=\"SearchCriteriaTypeEnum.AMOUNT\"\n type=\"number\"\n [(ngModel)]=\"input.value\"\n mode=\"decimal\"\n inputId=\"minmaxfraction\"\n [minFractionDigits]=\"2\"\n [maxFractionDigits]=\"5\"\n [placeholder]=\"getCurrencySymbol(input)\"\n class=\"full-width-input\"\n ></p-inputNumber>\n\n <p-inputNumber\n *ngSwitchCase=\"SearchCriteriaTypeEnum.NUMBER\"\n type=\"number\"\n [(ngModel)]=\"input.value\"\n class=\"full-width-input\"\n ></p-inputNumber>\n\n <p-multiSelect\n *ngSwitchCase=\"SearchCriteriaTypeEnum.MULTISELECT\"\n [options]=\"input.filterOptions\"\n [display]=\"'chip'\"\n placeholder=\"Select\"\n [selectAll]=\"selectAll\"\n [(ngModel)]=\"input.value\"\n optionLabel=\"label\"\n (onSelectAllChange)=\"onSelectAllChange($event)\"\n class=\"custom-multiselect full-width-input\"\n ></p-multiSelect>\n </ng-container>\n </div>\n </div>\n\n <ng-template pTemplate=\"footer\">\n <div class=\"footer-buttons\">\n <p-button\n label=\"Effacer\"\n icon=\"pi pi-times\"\n (click)=\"clear()\"\n class=\"footer-button\"\n ></p-button>\n <p-button\n label=\"Rechercher\"\n icon=\"pi pi-search\"\n (click)=\"search()\"\n class=\"footer-button\"\n ></p-button>\n </div>\n </ng-template>\n</p-panel>\n", styles: ["::ng-deep .p-element .p-hidden-accessible input{display:none}::ng-deep .bsc-label{display:block;margin-bottom:5px;font-weight:700}::ng-deep .custom-multiselect{width:100%}::ng-deep .p-multiselect .p-multiselect-label{font-size:14px;color:#333}::ng-deep .p-inputtext{width:100%}::ng-deep .p-calendar .p-inputtext{width:100%}::ng-deep .p-inputnumber{width:100%}.criteria-container{display:grid;gap:10px}.criteria-item{margin-bottom:10px;box-sizing:border-box}.full-width-input,::ng-deep .p-calendar,::ng-deep .p-multiselect{width:100%}::ng-deep .p-calendar .p-inputtext,::ng-deep .p-multiselect .p-multiselect-label{width:100%}.footer-buttons{display:flex;justify-content:flex-end;gap:10px}::ng-deep .footer-button button{width:140px;cursor:pointer}::ng-deep .p-panel .p-panel-footer{background:#f8f9fa}\n"], dependencies: [{ kind: "directive", type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i1.NgStyle, selector: "[ngStyle]", inputs: ["ngStyle"] }, { kind: "directive", type: i1.NgSwitch, selector: "[ngSwitch]", inputs: ["ngSwitch"] }, { kind: "directive", type: i1.NgSwitchCase, selector: "[ngSwitchCase]", inputs: ["ngSwitchCase"] }, { kind: "component", type: i2.Calendar, selector: "p-calendar", inputs: ["iconDisplay", "style", "styleClass", "inputStyle", "inputId", "name", "inputStyleClass", "placeholder", "ariaLabelledBy", "ariaLabel", "iconAriaLabel", "disabled", "dateFormat", "multipleSeparator", "rangeSeparator", "inline", "showOtherMonths", "selectOtherMonths", "showIcon", "icon", "appendTo", "readonlyInput", "shortYearCutoff", "monthNavigator", "yearNavigator", "hourFormat", "timeOnly", "stepYearPicker", "stepHour", "stepMinute", "stepSecond", "showSeconds", "required", "showOnFocus", "showWeek", "startWeekFromFirstDayOfYear", "showClear", "dataType", "selectionMode", "maxDateCount", "showButtonBar", "todayButtonStyleClass", "clearButtonStyleClass", "autofocus", "autoZIndex", "baseZIndex", "panelStyleClass", "panelStyle", "keepInvalid", "hideOnDateTimeSelect", "touchUI", "timeSeparator", "focusTrap", "showTransitionOptions", "hideTransitionOptions", "tabindex", "variant", "minDate", "maxDate", "disabledDates", "disabledDays", "yearRange", "showTime", "responsiveOptions", "numberOfMonths", "firstDayOfWeek", "locale", "view", "defaultDate"], outputs: ["onFocus", "onBlur", "onClose", "onSelect", "onClear", "onInput", "onTodayClick", "onClearClick", "onMonthChange", "onYearChange", "onClickOutside", "onShow"] }, { kind: "component", type: i3.Button, selector: "p-button", inputs: ["type", "iconPos", "icon", "badge", "label", "disabled", "loading", "loadingIcon", "raised", "rounded", "text", "plain", "severity", "outlined", "link", "tabindex", "size", "style", "styleClass", "badgeClass", "ariaLabel", "autofocus"], outputs: ["onClick", "onFocus", "onBlur"] }, { kind: "directive", type: i4.PrimeTemplate, selector: "[pTemplate]", inputs: ["type", "pTemplate"] }, { kind: "directive", type: i5.InputText, selector: "[pInputText]", inputs: ["variant"] }, { kind: "component", type: i6.MultiSelect, selector: "p-multiSelect", inputs: ["id", "ariaLabel", "style", "styleClass", "panelStyle", "panelStyleClass", "inputId", "disabled", "readonly", "group", "filter", "filterPlaceHolder", "filterLocale", "overlayVisible", "tabindex", "variant", "appendTo", "dataKey", "name", "ariaLabelledBy", "displaySelectedLabel", "maxSelectedLabels", "selectionLimit", "selectedItemsLabel", "showToggleAll", "emptyFilterMessage", "emptyMessage", "resetFilterOnHide", "dropdownIcon", "optionLabel", "optionValue", "optionDisabled", "optionGroupLabel", "optionGroupChildren", "showHeader", "filterBy", "scrollHeight", "lazy", "virtualScroll", "loading", "virtualScrollItemSize", "loadingIcon", "virtualScrollOptions", "overlayOptions", "ariaFilterLabel", "filterMatchMode", "tooltip", "tooltipPosition", "tooltipPositionStyle", "tooltipStyleClass", "autofocusFilter", "display", "autocomplete", "showClear", "autofocus", "autoZIndex", "baseZIndex", "showTransitionOptions", "hideTransitionOptions", "defaultLabel", "placeholder", "options", "filterValue", "itemSize", "selectAll", "focusOnHover", "filterFields", "selectOnFocus", "autoOptionFocus"], outputs: ["onChange", "onFilter", "onFocus", "onBlur", "onClick", "onClear", "onPanelShow", "onPanelHide", "onLazyLoad", "onRemove", "onSelectAllChange"] }, { kind: "directive", type: i7.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { kind: "directive", type: i7.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i7.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }, { kind: "component", type: i8.InputNumber, selector: "p-inputNumber", inputs: ["showButtons", "format", "buttonLayout", "inputId", "styleClass", "style", "placeholder", "size", "maxlength", "tabindex", "title", "ariaLabelledBy", "ariaLabel", "ariaRequired", "name", "required", "autocomplete", "min", "max", "incrementButtonClass", "decrementButtonClass", "incrementButtonIcon", "decrementButtonIcon", "readonly", "step", "allowEmpty", "locale", "localeMatcher", "mode", "currency", "currencyDisplay", "useGrouping", "variant", "minFractionDigits", "maxFractionDigits", "prefix", "suffix", "inputStyle", "inputStyleClass", "showClear", "autofocus", "disabled"], outputs: ["onInput", "onFocus", "onBlur", "onKeyDown", "onClear"] }, { kind: "component", type: i9.Panel, selector: "p-panel", inputs: ["toggleable", "header", "collapsed", "style", "styleClass", "iconPos", "expandIcon", "collapseIcon", "showHeader", "toggler", "transitionOptions"], outputs: ["collapsedChange", "onBeforeToggle", "onAfterToggle"] }] }); }
118
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "17.3.11", type: MultiSearchCriteriaComponent, selector: "multi-search-criteria", inputs: { title: "title", criteria: "criteria", inputsPerRow: "inputsPerRow", data: "data", mode: "mode" }, outputs: { filteredData: "filteredData", searchCriteria: "searchCriteria" }, ngImport: i0, template: "<p-panel header=\"{{ title }}\" [toggleable]=\"true\" [collapsed]=\"true\">\n <div\n class=\"criteria-container\"\n [ngStyle]=\"{ 'grid-template-columns': 'repeat(' + inputsPerRow + ', 1fr)' }\"\n >\n <div *ngFor=\"let input of criteria\" class=\"criteria-item\">\n <label class=\"bsc-label\">{{ input.title }}</label>\n\n <ng-container [ngSwitch]=\"input.type\">\n <p-calendar\n *ngSwitchCase=\"SearchCriteriaTypeEnum.DATERANGE\"\n [(ngModel)]=\"input.value\"\n selectionMode=\"range\"\n [dateFormat]=\"'dd/mm/yy'\"\n class=\"full-width-input\"\n [showIcon]=\"true\"\n (ngModelChange)=\"input.value = $event\"\n ></p-calendar>\n <p-calendar\n *ngSwitchCase=\"SearchCriteriaTypeEnum.DATE\"\n [ngModel]=\"input.value\"\n (ngModelChange)=\"input.value = $event\"\n [dateFormat]=\"'dd/mm/yy'\"\n [showIcon]=\"true\"\n class=\"full-width-input\"\n ></p-calendar>\n <input\n *ngSwitchCase=\"SearchCriteriaTypeEnum.STRING\"\n type=\"text\"\n pInputText\n [(ngModel)]=\"input.value\"\n class=\"full-width-input\"\n />\n\n <p-inputNumber\n *ngSwitchCase=\"SearchCriteriaTypeEnum.AMOUNT\"\n type=\"number\"\n [(ngModel)]=\"input.value\"\n mode=\"decimal\"\n inputId=\"minmaxfraction\"\n [minFractionDigits]=\"2\"\n [maxFractionDigits]=\"5\"\n [placeholder]=\"getCurrencySymbol(input)\"\n class=\"full-width-input\"\n ></p-inputNumber>\n\n <p-inputNumber\n *ngSwitchCase=\"SearchCriteriaTypeEnum.NUMBER\"\n type=\"number\"\n [(ngModel)]=\"input.value\"\n class=\"full-width-input\"\n ></p-inputNumber>\n\n <p-multiSelect\n *ngSwitchCase=\"SearchCriteriaTypeEnum.MULTISELECT\"\n [options]=\"input.filterOptions\"\n [display]=\"'chip'\"\n placeholder=\"Select\"\n [selectAll]=\"selectAll\"\n [(ngModel)]=\"input.value\"\n optionLabel=\"label\"\n (onSelectAllChange)=\"onSelectAllChange($event, input)\"\n class=\"custom-multiselect full-width-input\"\n ></p-multiSelect>\n </ng-container>\n </div>\n </div>\n\n <ng-template pTemplate=\"footer\">\n <div class=\"footer-buttons\">\n <p-button\n label=\"Effacer\"\n icon=\"pi pi-times\"\n (click)=\"clear()\"\n class=\"footer-button\"\n ></p-button>\n <p-button\n label=\"Rechercher\"\n icon=\"pi pi-search\"\n (click)=\"search()\"\n class=\"footer-button\"\n ></p-button>\n </div>\n </ng-template>\n</p-panel>\n", styles: ["::ng-deep .p-element .p-hidden-accessible input{display:none}::ng-deep .bsc-label{display:block;margin-bottom:5px;font-weight:700}::ng-deep .custom-multiselect{width:100%}::ng-deep .p-multiselect .p-multiselect-label{font-size:14px;color:#333}::ng-deep .p-inputtext{width:100%}::ng-deep .p-calendar .p-inputtext{width:100%}::ng-deep .p-inputnumber{width:100%}.criteria-container{display:grid;gap:10px}.criteria-item{margin-bottom:10px;box-sizing:border-box}.full-width-input,::ng-deep .p-calendar,::ng-deep .p-multiselect{width:100%}::ng-deep .p-calendar .p-inputtext,::ng-deep .p-multiselect .p-multiselect-label{width:100%}.footer-buttons{display:flex;justify-content:flex-end;gap:10px}::ng-deep .footer-button button{width:140px;cursor:pointer}::ng-deep .p-panel .p-panel-footer{background:#f8f9fa}\n"], dependencies: [{ kind: "directive", type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i1.NgStyle, selector: "[ngStyle]", inputs: ["ngStyle"] }, { kind: "directive", type: i1.NgSwitch, selector: "[ngSwitch]", inputs: ["ngSwitch"] }, { kind: "directive", type: i1.NgSwitchCase, selector: "[ngSwitchCase]", inputs: ["ngSwitchCase"] }, { kind: "component", type: i2.Calendar, selector: "p-calendar", inputs: ["iconDisplay", "style", "styleClass", "inputStyle", "inputId", "name", "inputStyleClass", "placeholder", "ariaLabelledBy", "ariaLabel", "iconAriaLabel", "disabled", "dateFormat", "multipleSeparator", "rangeSeparator", "inline", "showOtherMonths", "selectOtherMonths", "showIcon", "icon", "appendTo", "readonlyInput", "shortYearCutoff", "monthNavigator", "yearNavigator", "hourFormat", "timeOnly", "stepYearPicker", "stepHour", "stepMinute", "stepSecond", "showSeconds", "required", "showOnFocus", "showWeek", "startWeekFromFirstDayOfYear", "showClear", "dataType", "selectionMode", "maxDateCount", "showButtonBar", "todayButtonStyleClass", "clearButtonStyleClass", "autofocus", "autoZIndex", "baseZIndex", "panelStyleClass", "panelStyle", "keepInvalid", "hideOnDateTimeSelect", "touchUI", "timeSeparator", "focusTrap", "showTransitionOptions", "hideTransitionOptions", "tabindex", "variant", "minDate", "maxDate", "disabledDates", "disabledDays", "yearRange", "showTime", "responsiveOptions", "numberOfMonths", "firstDayOfWeek", "locale", "view", "defaultDate"], outputs: ["onFocus", "onBlur", "onClose", "onSelect", "onClear", "onInput", "onTodayClick", "onClearClick", "onMonthChange", "onYearChange", "onClickOutside", "onShow"] }, { kind: "component", type: i3.Button, selector: "p-button", inputs: ["type", "iconPos", "icon", "badge", "label", "disabled", "loading", "loadingIcon", "raised", "rounded", "text", "plain", "severity", "outlined", "link", "tabindex", "size", "style", "styleClass", "badgeClass", "ariaLabel", "autofocus"], outputs: ["onClick", "onFocus", "onBlur"] }, { kind: "directive", type: i4.PrimeTemplate, selector: "[pTemplate]", inputs: ["type", "pTemplate"] }, { kind: "directive", type: i5.InputText, selector: "[pInputText]", inputs: ["variant"] }, { kind: "component", type: i6.MultiSelect, selector: "p-multiSelect", inputs: ["id", "ariaLabel", "style", "styleClass", "panelStyle", "panelStyleClass", "inputId", "disabled", "readonly", "group", "filter", "filterPlaceHolder", "filterLocale", "overlayVisible", "tabindex", "variant", "appendTo", "dataKey", "name", "ariaLabelledBy", "displaySelectedLabel", "maxSelectedLabels", "selectionLimit", "selectedItemsLabel", "showToggleAll", "emptyFilterMessage", "emptyMessage", "resetFilterOnHide", "dropdownIcon", "optionLabel", "optionValue", "optionDisabled", "optionGroupLabel", "optionGroupChildren", "showHeader", "filterBy", "scrollHeight", "lazy", "virtualScroll", "loading", "virtualScrollItemSize", "loadingIcon", "virtualScrollOptions", "overlayOptions", "ariaFilterLabel", "filterMatchMode", "tooltip", "tooltipPosition", "tooltipPositionStyle", "tooltipStyleClass", "autofocusFilter", "display", "autocomplete", "showClear", "autofocus", "autoZIndex", "baseZIndex", "showTransitionOptions", "hideTransitionOptions", "defaultLabel", "placeholder", "options", "filterValue", "itemSize", "selectAll", "focusOnHover", "filterFields", "selectOnFocus", "autoOptionFocus"], outputs: ["onChange", "onFilter", "onFocus", "onBlur", "onClick", "onClear", "onPanelShow", "onPanelHide", "onLazyLoad", "onRemove", "onSelectAllChange"] }, { kind: "directive", type: i7.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { kind: "directive", type: i7.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i7.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }, { kind: "component", type: i8.InputNumber, selector: "p-inputNumber", inputs: ["showButtons", "format", "buttonLayout", "inputId", "styleClass", "style", "placeholder", "size", "maxlength", "tabindex", "title", "ariaLabelledBy", "ariaLabel", "ariaRequired", "name", "required", "autocomplete", "min", "max", "incrementButtonClass", "decrementButtonClass", "incrementButtonIcon", "decrementButtonIcon", "readonly", "step", "allowEmpty", "locale", "localeMatcher", "mode", "currency", "currencyDisplay", "useGrouping", "variant", "minFractionDigits", "maxFractionDigits", "prefix", "suffix", "inputStyle", "inputStyleClass", "showClear", "autofocus", "disabled"], outputs: ["onInput", "onFocus", "onBlur", "onKeyDown", "onClear"] }, { kind: "component", type: i9.Panel, selector: "p-panel", inputs: ["toggleable", "header", "collapsed", "style", "styleClass", "iconPos", "expandIcon", "collapseIcon", "showHeader", "toggler", "transitionOptions"], outputs: ["collapsedChange", "onBeforeToggle", "onAfterToggle"] }] }); }
113
119
  }
114
120
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.11", ngImport: i0, type: MultiSearchCriteriaComponent, decorators: [{
115
121
  type: Component,
116
- args: [{ selector: 'multi-search-criteria', template: "<p-panel header=\"{{ title }}\" [toggleable]=\"true\" [collapsed]=\"true\">\n <div\n class=\"criteria-container\"\n [ngStyle]=\"{ 'grid-template-columns': 'repeat(' + inputsPerRow + ', 1fr)' }\"\n >\n <div *ngFor=\"let input of criteria\" class=\"criteria-item\">\n <label class=\"bsc-label\">{{ input.title }}</label>\n\n <ng-container [ngSwitch]=\"input.type\">\n <p-calendar\n *ngSwitchCase=\"SearchCriteriaTypeEnum.DATERANGE\"\n [(ngModel)]=\"input.value\"\n selectionMode=\"range\"\n [dateFormat]=\"'dd/mm/yy'\"\n class=\"full-width-input\"\n [showIcon]=\"true\"\n (ngModelChange)=\"input.value = $event\"\n ></p-calendar>\n <p-calendar\n *ngSwitchCase=\"SearchCriteriaTypeEnum.DATE\"\n [ngModel]=\"input.value\"\n (ngModelChange)=\"input.value = $event\"\n [dateFormat]=\"'dd/mm/yy'\"\n [showIcon]=\"true\"\n class=\"full-width-input\"\n ></p-calendar>\n <input\n *ngSwitchCase=\"SearchCriteriaTypeEnum.STRING\"\n type=\"text\"\n pInputText\n [(ngModel)]=\"input.value\"\n class=\"full-width-input\"\n />\n\n <p-inputNumber\n *ngSwitchCase=\"SearchCriteriaTypeEnum.AMOUNT\"\n type=\"number\"\n [(ngModel)]=\"input.value\"\n mode=\"decimal\"\n inputId=\"minmaxfraction\"\n [minFractionDigits]=\"2\"\n [maxFractionDigits]=\"5\"\n [placeholder]=\"getCurrencySymbol(input)\"\n class=\"full-width-input\"\n ></p-inputNumber>\n\n <p-inputNumber\n *ngSwitchCase=\"SearchCriteriaTypeEnum.NUMBER\"\n type=\"number\"\n [(ngModel)]=\"input.value\"\n class=\"full-width-input\"\n ></p-inputNumber>\n\n <p-multiSelect\n *ngSwitchCase=\"SearchCriteriaTypeEnum.MULTISELECT\"\n [options]=\"input.filterOptions\"\n [display]=\"'chip'\"\n placeholder=\"Select\"\n [selectAll]=\"selectAll\"\n [(ngModel)]=\"input.value\"\n optionLabel=\"label\"\n (onSelectAllChange)=\"onSelectAllChange($event)\"\n class=\"custom-multiselect full-width-input\"\n ></p-multiSelect>\n </ng-container>\n </div>\n </div>\n\n <ng-template pTemplate=\"footer\">\n <div class=\"footer-buttons\">\n <p-button\n label=\"Effacer\"\n icon=\"pi pi-times\"\n (click)=\"clear()\"\n class=\"footer-button\"\n ></p-button>\n <p-button\n label=\"Rechercher\"\n icon=\"pi pi-search\"\n (click)=\"search()\"\n class=\"footer-button\"\n ></p-button>\n </div>\n </ng-template>\n</p-panel>\n", styles: ["::ng-deep .p-element .p-hidden-accessible input{display:none}::ng-deep .bsc-label{display:block;margin-bottom:5px;font-weight:700}::ng-deep .custom-multiselect{width:100%}::ng-deep .p-multiselect .p-multiselect-label{font-size:14px;color:#333}::ng-deep .p-inputtext{width:100%}::ng-deep .p-calendar .p-inputtext{width:100%}::ng-deep .p-inputnumber{width:100%}.criteria-container{display:grid;gap:10px}.criteria-item{margin-bottom:10px;box-sizing:border-box}.full-width-input,::ng-deep .p-calendar,::ng-deep .p-multiselect{width:100%}::ng-deep .p-calendar .p-inputtext,::ng-deep .p-multiselect .p-multiselect-label{width:100%}.footer-buttons{display:flex;justify-content:flex-end;gap:10px}::ng-deep .footer-button button{width:140px;cursor:pointer}::ng-deep .p-panel .p-panel-footer{background:#f8f9fa}\n"] }]
122
+ args: [{ selector: 'multi-search-criteria', template: "<p-panel header=\"{{ title }}\" [toggleable]=\"true\" [collapsed]=\"true\">\n <div\n class=\"criteria-container\"\n [ngStyle]=\"{ 'grid-template-columns': 'repeat(' + inputsPerRow + ', 1fr)' }\"\n >\n <div *ngFor=\"let input of criteria\" class=\"criteria-item\">\n <label class=\"bsc-label\">{{ input.title }}</label>\n\n <ng-container [ngSwitch]=\"input.type\">\n <p-calendar\n *ngSwitchCase=\"SearchCriteriaTypeEnum.DATERANGE\"\n [(ngModel)]=\"input.value\"\n selectionMode=\"range\"\n [dateFormat]=\"'dd/mm/yy'\"\n class=\"full-width-input\"\n [showIcon]=\"true\"\n (ngModelChange)=\"input.value = $event\"\n ></p-calendar>\n <p-calendar\n *ngSwitchCase=\"SearchCriteriaTypeEnum.DATE\"\n [ngModel]=\"input.value\"\n (ngModelChange)=\"input.value = $event\"\n [dateFormat]=\"'dd/mm/yy'\"\n [showIcon]=\"true\"\n class=\"full-width-input\"\n ></p-calendar>\n <input\n *ngSwitchCase=\"SearchCriteriaTypeEnum.STRING\"\n type=\"text\"\n pInputText\n [(ngModel)]=\"input.value\"\n class=\"full-width-input\"\n />\n\n <p-inputNumber\n *ngSwitchCase=\"SearchCriteriaTypeEnum.AMOUNT\"\n type=\"number\"\n [(ngModel)]=\"input.value\"\n mode=\"decimal\"\n inputId=\"minmaxfraction\"\n [minFractionDigits]=\"2\"\n [maxFractionDigits]=\"5\"\n [placeholder]=\"getCurrencySymbol(input)\"\n class=\"full-width-input\"\n ></p-inputNumber>\n\n <p-inputNumber\n *ngSwitchCase=\"SearchCriteriaTypeEnum.NUMBER\"\n type=\"number\"\n [(ngModel)]=\"input.value\"\n class=\"full-width-input\"\n ></p-inputNumber>\n\n <p-multiSelect\n *ngSwitchCase=\"SearchCriteriaTypeEnum.MULTISELECT\"\n [options]=\"input.filterOptions\"\n [display]=\"'chip'\"\n placeholder=\"Select\"\n [selectAll]=\"selectAll\"\n [(ngModel)]=\"input.value\"\n optionLabel=\"label\"\n (onSelectAllChange)=\"onSelectAllChange($event, input)\"\n class=\"custom-multiselect full-width-input\"\n ></p-multiSelect>\n </ng-container>\n </div>\n </div>\n\n <ng-template pTemplate=\"footer\">\n <div class=\"footer-buttons\">\n <p-button\n label=\"Effacer\"\n icon=\"pi pi-times\"\n (click)=\"clear()\"\n class=\"footer-button\"\n ></p-button>\n <p-button\n label=\"Rechercher\"\n icon=\"pi pi-search\"\n (click)=\"search()\"\n class=\"footer-button\"\n ></p-button>\n </div>\n </ng-template>\n</p-panel>\n", styles: ["::ng-deep .p-element .p-hidden-accessible input{display:none}::ng-deep .bsc-label{display:block;margin-bottom:5px;font-weight:700}::ng-deep .custom-multiselect{width:100%}::ng-deep .p-multiselect .p-multiselect-label{font-size:14px;color:#333}::ng-deep .p-inputtext{width:100%}::ng-deep .p-calendar .p-inputtext{width:100%}::ng-deep .p-inputnumber{width:100%}.criteria-container{display:grid;gap:10px}.criteria-item{margin-bottom:10px;box-sizing:border-box}.full-width-input,::ng-deep .p-calendar,::ng-deep .p-multiselect{width:100%}::ng-deep .p-calendar .p-inputtext,::ng-deep .p-multiselect .p-multiselect-label{width:100%}.footer-buttons{display:flex;justify-content:flex-end;gap:10px}::ng-deep .footer-button button{width:140px;cursor:pointer}::ng-deep .p-panel .p-panel-footer{background:#f8f9fa}\n"] }]
117
123
  }], propDecorators: { title: [{
118
124
  type: Input
119
125
  }], criteria: [{
@@ -122,7 +128,11 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.11", ngImpo
122
128
  type: Input
123
129
  }], data: [{
124
130
  type: Input
131
+ }], mode: [{
132
+ type: Input
125
133
  }], filteredData: [{
126
134
  type: Output
135
+ }], searchCriteria: [{
136
+ type: Output
127
137
  }] } });
128
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibXVsdGktc2VhcmNoLWNyaXRlcmlhLmNvbXBvbmVudC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL25nLXByaW1lLXRvb2xzL3NyYy9saWIvbXVsdGktc2VhcmNoLWNyaXRlcmlhL211bHRpLXNlYXJjaC1jcml0ZXJpYS5jb21wb25lbnQudHMiLCIuLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9uZy1wcmltZS10b29scy9zcmMvbGliL211bHRpLXNlYXJjaC1jcml0ZXJpYS9tdWx0aS1zZWFyY2gtY3JpdGVyaWEuY29tcG9uZW50Lmh0bWwiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLFNBQVMsRUFBRSxLQUFLLEVBQUUsTUFBTSxFQUFFLFlBQVksRUFBVSxNQUFNLGVBQWUsQ0FBQztBQUMvRSxPQUFPLEVBQUUsc0JBQXNCLEVBQUUsTUFBTSxVQUFVLENBQUM7Ozs7Ozs7Ozs7O0FBU2xELE1BQU0sT0FBTyw0QkFBNEI7SUFMekM7UUFNRSwyQkFBc0IsR0FBRyxzQkFBc0IsQ0FBQztRQUV2QyxVQUFLLEdBQVcsRUFBRSxDQUFDO1FBQ25CLGFBQVEsR0FBcUIsRUFBRSxDQUFDO1FBQ2hDLGlCQUFZLEdBQVcsQ0FBQyxDQUFDO1FBQ3pCLFNBQUksR0FBVSxFQUFFLENBQUM7UUFDaEIsaUJBQVksR0FBRyxJQUFJLFlBQVksRUFBUyxDQUFDO1FBRW5ELGNBQVMsR0FBWSxLQUFLLENBQUM7UUFDM0IsYUFBUSxHQUFtQixFQUFFLENBQUM7S0FxSC9CO0lBbkhDLFFBQVE7UUFDTixJQUFJLENBQUMsUUFBUSxDQUFDLE9BQU8sQ0FBQyxDQUFDLElBQUksRUFBRSxFQUFFO1lBQzdCLElBQ0UsSUFBSSxDQUFDLElBQUksS0FBSyxzQkFBc0IsQ0FBQyxXQUFXO2dCQUNoRCxJQUFJLENBQUMsYUFBYSxFQUNsQixDQUFDO2dCQUNELElBQUksQ0FBQyxRQUFRLEdBQUcsQ0FBQyxHQUFHLElBQUksQ0FBQyxh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cnQgeyBDb21wb25lbnQsIElucHV0LCBPdXRwdXQsIEV2ZW50RW1pdHRlciwgT25Jbml0IH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQgeyBTZWFyY2hDcml0ZXJpYVR5cGVFbnVtIH0gZnJvbSAnLi4vZW51bXMnO1xuaW1wb3J0IHsgU2VhcmNoQ3JpdGVyaWEgfSBmcm9tICcuLi9tb2RlbHMvc2VhcmNoLWNyaXRlcmlhJztcbmltcG9ydCB7IEZpbHRlck9wdGlvbiB9IGZyb20gJy4uL21vZGVscy9maWx0ZXItb3B0aW9uJztcblxuQENvbXBvbmVudCh7XG4gIHNlbGVjdG9yOiAnbXVsdGktc2VhcmNoLWNyaXRlcmlhJyxcbiAgdGVtcGxhdGVVcmw6ICcuL211bHRpLXNlYXJjaC1jcml0ZXJpYS5jb21wb25lbnQuaHRtbCcsXG4gIHN0eWxlVXJsczogWycuL211bHRpLXNlYXJjaC1jcml0ZXJpYS5jb21wb25lbnQuY3NzJ10sXG59KVxuZXhwb3J0IGNsYXNzIE11bHRpU2VhcmNoQ3JpdGVyaWFDb21wb25lbnQgaW1wbGVtZW50cyBPbkluaXQge1xuICBTZWFyY2hDcml0ZXJpYVR5cGVFbnVtID0gU2VhcmNoQ3JpdGVyaWFUeXBlRW51bTtcblxuICBASW5wdXQoKSB0aXRsZTogU3RyaW5nID0gJyc7XG4gIEBJbnB1dCgpIGNyaXRlcmlhOiBTZWFyY2hDcml0ZXJpYVtdID0gW107XG4gIEBJbnB1dCgpIGlucHV0c1BlclJvdzogbnVtYmVyID0gMztcbiAgQElucHV0KCkgZGF0YTogYW55W10gPSBbXTtcbiAgQE91dHB1dCgpIGZpbHRlcmVkRGF0YSA9IG5ldyBFdmVudEVtaXR0ZXI8YW55W10+KCk7XG5cbiAgc2VsZWN0QWxsOiBib29sZWFuID0gZmFsc2U7XG4gIHNlbGVjdGVkOiBGaWx0ZXJPcHRpb25bXSA9IFtdO1xuXG4gIG5nT25Jbml0KCkge1xuICAgIHRoaXMuY3JpdGVyaWEuZm9yRWFjaCgoaXRlbSkgPT4ge1xuICAgICAgaWYgKFxuICAgICAgICBpdGVtLnR5cGUgPT09IFNlYXJjaENyaXRlcmlhVHlwZUVudW0uTVVMVElTRUxFQ1QgJiZcbiAgICAgICAgaXRlbS5maWx0ZXJPcHRpb25zXG4gICAgICApIHtcbiAgICAgICAgdGhpcy5zZWxlY3RlZCA9IFsuLi5pdGVtLmZpbHRlck9wdGlvbnNdO1xuICAgICAgICBpdGVtLnZhbHVlID0gdGhpcy5zZWxlY3RlZDtcbiAgICAgIH1cbiAgICB9KTtcbiAgfVxuXG4gIGdldEN1cnJlbmN5U3ltYm9sKGlucHV0OiBTZWFyY2hDcml0ZXJpYSk6IHN0cmluZyB7XG4gICAgcmV0dXJuIGlucHV0LmN1cnJlbmN5IHx8ICdFVVInO1xuICB9XG5cbiAgc2VhcmNoKCk6IHZvaWQge1xuICAgIGNvbnNvbGUubG9nKCdPcmlnaW5hbCBEYXRhOicsIHRoaXMuZGF0YSk7XG4gICAgY29uc3QgZmlsdGVyZWQgPSB0aGlzLmRhdGEuZmlsdGVyKChpdGVtKSA9PiB7XG4gICAgICByZXR1cm4gdGhpcy5jcml0ZXJpYS5ldmVyeSgoY3JpdGVyaW9uKSA9PiB7XG4gICAgICAgIGNvbnNvbGUubG9nKCdDcml0ZXJpb246JywgY3JpdGVyaW9uKTtcbiAgICAgICAgaWYgKGNyaXRlcmlvbi50eXBlID09PSBTZWFyY2hDcml0ZXJpYVR5cGVFbnVtLkRBVEVSQU5HRSkge1xuICAgICAgICAgIGNvbnN0IFtzdGFydERhdGUsIGVuZERhdGVdID0gY3JpdGVyaW9uLnZhbHVlIHx8IFtcbiAgICAgICAgICAgIHVuZGVmaW5lZCxcbiAgICAgICAgICAgIHVuZGVmaW5lZCxcbiAgICAgICAgICBdO1xuICAgICAgICAgIGNvbnN0IGl0ZW1EYXRlID0gdGhpcy5wYXJzZURhdGUoaXRlbVtjcml0ZXJpb24uY29kZV0gfHwgaXRlbVsnZGF0ZSddKTtcbiAgICAgICAgICBjb25zb2xlLmxvZyhcbiAgICAgICAgICAgICdJdGVtIERhdGU6JyxcbiAgICAgICAgICAgIGl0ZW1EYXRlLFxuICAgICAgICAgICAgJ1N0YXJ0IERhdGU6JyxcbiAgICAgICAgICAgIHN0YXJ0RGF0ZSxcbiAgICAgICAgICAgICdFbmQgRGF0ZTonLFxuICAgICAgICAgICAgZW5kRGF0ZVxuICAgICAgICAgICk7XG4gICAgICAgICAgaWYgKCFzdGFydERhdGUgJiYgIWVuZERhdGUpIHtcbiAgICAgICAgICAgIHJldHVybiB0cnVlO1xuICAgICAgICAgIH1cbiAgICAgICAgICBpZiAoaXRlbURhdGUgPT09IG51bGwpIHtcbiAgICAgICAgICAgIHJldHVybiBmYWxzZTtcbiAgICAgICAgICB9XG4gICAgICAgICAgY29uc3QgcGFyc2VkU3RhcnREYXRlID0gc3RhcnREYXRlID8gdGhpcy5wYXJzZURhdGUoc3RhcnREYXRlKSA6IG51bGw7XG4gICAgICAgICAgY29uc3QgcGFyc2VkRW5kRGF0ZSA9IGVuZERhdGUgPyB0aGlzLnBhcnNlRGF0ZShlbmREYXRlKSA6IG51bGw7XG4gICAgICAgICAgY29uc29sZS5sb2coXG4gICAgICAgICAgICAnUGFyc2VkIFN0YXJ0IERhdGU6JyxcbiAgICAgICAgICAgIHBhcnNlZFN0YXJ0RGF0ZSxcbiAgICAgICAgICAgICdQYXJzZWQgRW5kIERhdGU6JyxcbiAgICAgICAgICAgIHBhcnNlZEVuZERhdGVcbiAgICAgICAgICApO1xuICAgICAgICAgIHJldHVybiAoXG4gICAgICAgICAgICAoIXBhcnNlZFN0YXJ0RGF0ZSB8fCBpdGVtRGF0ZSA+PSBwYXJzZWRTdGFydERhdGUpICYmXG4gICAgICAgICAgICAoIXBhcnNlZEVuZERhdGUgfHwgaXRlbURhdGUgPD0gcGFyc2VkRW5kRGF0ZSlcbiAgICAgICAgICApO1xuICAgICAgICB9IGVsc2UgaWYgKGNyaXRlcmlvbi50eXBlID09PSBTZWFyY2hDcml0ZXJpYVR5cGVFbnVtLkFNT1VOVCkge1xuICAgICAgICAgIHJldHVybiAhY3JpdGVyaW9uLnZhbHVlIHx8IGl0ZW1bY3JpdGVyaW9uLmNvZGVdID09PSBjcml0ZXJpb24udmFsdWU7XG4gICAgICAgIH0gZWxzZSBpZiAoY3JpdGVyaW9uLnR5cGUgPT09IFNlYXJjaENyaXRlcmlhVHlwZUVudW0uTVVMVElTRUxFQ1QpIHtcbiAgICAgICAgICByZXR1cm4gKFxuICAgICAgICAgICAgIWNyaXRlcmlvbi52YWx1ZSB8fFxuICAgICAgICAgICAgY3JpdGVyaW9uLnZhbHVlLnNvbWUoXG4gICAgICAgICAgICAgIChvcHRpb246IEZpbHRlck9wdGlvbikgPT4gb3B0aW9uLmxhYmVsID09PSBpdGVtW2NyaXRlcmlvbi5jb2RlXVxuICAgICAgICAgICAgKVxuICAgICAgICAgICk7XG4gICAgICAgIH0gZWxzZSBpZiAoY3JpdGVyaW9uLnR5cGUgPT09IFNlYXJjaENyaXRlcmlhVHlwZUVudW0uU1RSSU5HKSB7XG4gICAgICAgICAgcmV0dXJuIChcbiAgICAgICAgICAgICFjcml0ZXJpb24udmFsdWUgfHxcbiAgICAgICAgICAgIGl0ZW1bY3JpdGVyaW9uLmNvZGVdXG4gICAgICAgICAgICAgIC50b1N0cmluZygpXG4gICAgICAgICAgICAgIC50b0xvd2VyQ2FzZSgpXG4gICAgICAgICAgICAgIC5pbmNsdWRlcyhjcml0ZXJpb24udmFsdWUudG9TdHJpbmcoKS50b0xvd2VyQ2FzZSgpKVxuICAgICAgICAgICk7XG4gICAgICAgIH1cbiAgICAgICAgcmV0dXJuIHRydWU7XG4gICAgICB9KTtcbiAgICB9KTtcbiAgICBjb25zb2xlLmxvZygnRmlsdGVyZWQgRGF0YTonLCBmaWx0ZXJlZCk7XG4gICAgdGhpcy5maWx0ZXJlZERhdGEuZW1pdChmaWx0ZXJlZCk7XG4gIH1cblxuICBjbGVhcigpOiB2b2lkIHtcbiAgICB0aGlzLmNyaXRlcmlhLmZvckVhY2goKGlucHV0KSA9PiB7XG4gICAgICBpbnB1dC52YWx1ZSA9IG51bGw7XG4gICAgfSk7XG4gICAgdGhpcy5maWx0ZXJlZERhdGEuZW1pdCh0aGlzLmRhdGEpO1xuICB9XG5cbiAgb25TZWxlY3RBbGxDaGFuZ2UoZXZlbnQ6IGFueSkge1xuICAgIHRoaXMuc2VsZWN0QWxsID0gZXZlbnQuY2hlY2tlZDtcbiAgICB0aGlzLnNlbGVjdGVkID0gZXZlbnQuY2hlY2tlZFxuICAgICAgPyBbXG4gICAgICAgICAgLi4uKHRoaXMuY3JpdGVyaWEuZmluZChcbiAgICAgICAgICAgIChjcml0ZXJpb24pID0+IGNyaXRlcmlvbi50eXBlID09PSBTZWFyY2hDcml0ZXJpYVR5cGVFbnVtLk1VTFRJU0VMRUNUXG4gICAgICAgICAgKT8uZmlsdGVyT3B0aW9ucyB8fCBbXSksXG4gICAgICAgIF1cbiAgICAgIDogW107XG4gIH1cblxuICBwcml2YXRlIHBhcnNlRGF0ZShkYXRlU3RyaW5nOiBzdHJpbmcgfCBEYXRlIHwgbnVsbCB8IHVuZGVmaW5lZCk6IERhdGUgfCBudWxsIHtcbiAgICBpZiAoIWRhdGVTdHJpbmcpIHtcbiAgICAgIHJldHVybiBudWxsO1xuICAgIH1cblxuICAgIGlmIChPYmplY3QucHJvdG90eXBlLnRvU3RyaW5nLmNhbGwoZGF0ZVN0cmluZykgPT09ICdbb2JqZWN0IERhdGVdJykge1xuICAgICAgcmV0dXJuIGRhdGVTdHJpbmcgYXMgRGF0ZTtcbiAgICB9XG5cbiAgICBjb25zdCBwYXJ0cyA9IChkYXRlU3RyaW5nIGFzIHN0cmluZykuc3BsaXQoJy8nKTtcbiAgICBpZiAocGFydHMubGVuZ3RoID09PSAzKSB7XG4gICAgICBjb25zdCBkYXkgPSBwYXJzZUludChwYXJ0c1swXSwgMTApO1xuICAgICAgY29uc3QgbW9udGggPSBwYXJzZUludChwYXJ0c1sxXSwgMTApIC0gMTtcbiAgICAgIGNvbnN0IHllYXIgPSBwYXJzZUludChwYXJ0c1syXSwgMTApO1xuICAgICAgY29uc3QgZGF0ZSA9IG5ldyBEYXRlKHllYXIsIG1vbnRoLCBkYXkpO1xuICAgICAgcmV0dXJuIGlzTmFOKGRhdGUuZ2V0VGltZSgpKSA/IG51bGwgOiBkYXRlO1xuICAgIH1cbiAgICByZXR1cm4gbnVsbDtcbiAgfVxufVxuIiwiPHAtcGFuZWwgaGVhZGVyPVwie3sgdGl0bGUgfX1cIiBbdG9nZ2xlYWJsZV09XCJ0cnVlXCIgW2NvbGxhcHNlZF09XCJ0cnVlXCI+XG4gIDxkaXZcbiAgICBjbGFzcz1cImNyaXRlcmlhLWNvbnRhaW5lclwiXG4gICAgW25nU3R5bGVdPVwieyAnZ3JpZC10ZW1wbGF0ZS1jb2x1bW5zJzogJ3JlcGVhdCgnICsgaW5wdXRzUGVyUm93ICsgJywgMWZyKScgfVwiXG4gID5cbiAgICA8ZGl2ICpuZ0Zvcj1cImxldCBpbnB1dCBvZiBjcml0ZXJpYVwiIGNsYXNzPVwiY3JpdGVyaWEtaXRlbVwiPlxuICAgICAgPGxhYmVsIGNsYXNzPVwiYnNjLWxhYmVsXCI+e3sgaW5wdXQudGl0bGUgfX08L2xhYmVsPlxuXG4gICAgICA8bmctY29udGFpbmVyIFtuZ1N3aXRjaF09XCJpbnB1dC50eXBlXCI+XG4gICAgICAgIDxwLWNhbGVuZGFyXG4gICAgICAgICAgKm5nU3dpdGNoQ2FzZT1cIlNlYXJjaENyaXRlcmlhVHlwZUVudW0uREFURVJBTkdFXCJcbiAgICAgICAgICBbKG5nTW9kZWwpXT1cImlucHV0LnZhbHVlXCJcbiAgICAgICAgICBzZWxlY3Rpb25Nb2RlPVwicmFuZ2VcIlxuICAgICAgICAgIFtkYXRlRm9ybWF0XT1cIidkZC9tbS95eSdcIlxuICAgICAgICAgIGNsYXNzPVwiZnVsbC13aWR0aC1pbnB1dFwiXG4gICAgICAgICAgW3Nob3dJY29uXT1cInRydWVcIlxuICAgICAgICAgIChuZ01vZGVsQ2hhbmdlKT1cImlucHV0LnZhbHVlID0gJGV2ZW50XCJcbiAgICAgICAgPjwvcC1jYWxlbmRhcj5cbiAgICAgICAgPHAtY2FsZW5kYXJcbiAgICAgICAgICAqbmdTd2l0Y2hDYXNlPVwiU2VhcmNoQ3JpdGVyaWFUeXBlRW51bS5EQVRFXCJcbiAgICAgICAgICBbbmdNb2RlbF09XCJpbnB1dC52YWx1ZVwiXG4gICAgICAgICAgKG5nTW9kZWxDaGFuZ2UpPVwiaW5wdXQudmFsdWUgPSAkZXZlbnRcIlxuICAgICAgICAgIFtkYXRlRm9ybWF0XT1cIidkZC9tbS95eSdcIlxuICAgICAgICAgIFtzaG93SWNvbl09XCJ0cnVlXCJcbiAgICAgICAgICBjbGFzcz1cImZ1bGwtd2lkdGgtaW5wdXRcIlxuICAgICAgICA+PC9wLWNhbGVuZGFyPlxuICAgICAgICA8aW5wdXRcbiAgICAgICAgICAqbmdTd2l0Y2hDYXNlPVwiU2VhcmNoQ3JpdGVyaWFUeXBlRW51bS5TVFJJTkdcIlxuICAgICAgICAgIHR5cGU9XCJ0ZXh0XCJcbiAgICAgICAgICBwSW5wdXRUZXh0XG4gICAgICAgICAgWyhuZ01vZGVsKV09XCJpbnB1dC52YWx1ZVwiXG4gICAgICAgICAgY2xhc3M9XCJmdWxsLXdpZHRoLWlucHV0XCJcbiAgICAgICAgLz5cblxuICAgICAgICA8cC1pbnB1dE51bWJlclxuICAgICAgICAgICpuZ1N3aXRjaENhc2U9XCJTZWFyY2hDcml0ZXJpYVR5cGVFbnVtLkFNT1VOVFwiXG4gICAgICAgICAgdHlwZT1cIm51bWJlclwiXG4gICAgICAgICAgWyhuZ01vZGVsKV09XCJpbnB1dC52YWx1ZVwiXG4gICAgICAgICAgbW9kZT1cImRlY2ltYWxcIlxuICAgICAgICAgIGlucHV0SWQ9XCJtaW5tYXhmcmFjdGlvblwiXG4gICAgICAgICAgW21pbkZyYWN0aW9uRGlnaXRzXT1cIjJcIlxuICAgICAgICAgIFttYXhGcmFjdGlvbkRpZ2l0c109XCI1XCJcbiAgICAgICAgICBbcGxhY2Vob2xkZXJdPVwiZ2V0Q3VycmVuY3lTeW1ib2woaW5wdXQpXCJcbiAgICAgICAgICBjbGFzcz1cImZ1bGwtd2lkdGgtaW5wdXRcIlxuICAgICAgICA+PC9wLWlucHV0TnVtYmVyPlxuXG4gICAgICAgIDxwLWlucHV0TnVtYmVyXG4gICAgICAgICAgKm5nU3dpdGNoQ2FzZT1cIlNlYXJjaENyaXRlcmlhVHlwZUVudW0uTlVNQkVSXCJcbiAgICAgICAgICB0eXBlPVwibnVtYmVyXCJcbiAgICAgICAgICBbKG5nTW9kZWwpXT1cImlucHV0LnZhbHVlXCJcbiAgICAgICAgICBjbGFzcz1cImZ1bGwtd2lkdGgtaW5wdXRcIlxuICAgICAgICA+PC9wLWlucHV0TnVtYmVyPlxuXG4gICAgICAgIDxwLW11bHRpU2VsZWN0XG4gICAgICAgICAgKm5nU3dpdGNoQ2FzZT1cIlNlYXJjaENyaXRlcmlhVHlwZUVudW0uTVVMVElTRUxFQ1RcIlxuICAgICAgICAgIFtvcHRpb25zXT1cImlucHV0LmZpbHRlck9wdGlvbnNcIlxuICAgICAgICAgIFtkaXNwbGF5XT1cIidjaGlwJ1wiXG4gICAgICAgICAgcGxhY2Vob2xkZXI9XCJTZWxlY3RcIlxuICAgICAgICAgIFtzZWxlY3RBbGxdPVwic2VsZWN0QWxsXCJcbiAgICAgICAgICBbKG5nTW9kZWwpXT1cImlucHV0LnZhbHVlXCJcbiAgICAgICAgICBvcHRpb25MYWJlbD1cImxhYmVsXCJcbiAgICAgICAgICAob25TZWxlY3RBbGxDaGFuZ2UpPVwib25TZWxlY3RBbGxDaGFuZ2UoJGV2ZW50KVwiXG4gICAgICAgICAgY2xhc3M9XCJjdXN0b20tbXVsdGlzZWxlY3QgZnVsbC13aWR0aC1pbnB1dFwiXG4gICAgICAgID48L3AtbXVsdGlTZWxlY3Q+XG4gICAgICA8L25nLWNvbnRhaW5lcj5cbiAgICA8L2Rpdj5cbiAgPC9kaXY+XG5cbiAgPG5nLXRlbXBsYXRlIHBUZW1wbGF0ZT1cImZvb3RlclwiPlxuICAgIDxkaXYgY2xhc3M9XCJmb290ZXItYnV0dG9uc1wiPlxuICAgICAgPHAtYnV0dG9uXG4gICAgICAgIGxhYmVsPVwiRWZmYWNlclwiXG4gICAgICAgIGljb249XCJwaSBwaS10aW1lc1wiXG4gICAgICAgIChjbGljayk9XCJjbGVhcigpXCJcbiAgICAgICAgY2xhc3M9XCJmb290ZXItYnV0dG9uXCJcbiAgICAgID48L3AtYnV0dG9uPlxuICAgICAgPHAtYnV0dG9uXG4gICAgICAgIGxhYmVsPVwiUmVjaGVyY2hlclwiXG4gICAgICAgIGljb249XCJwaSBwaS1zZWFyY2hcIlxuICAgICAgICAoY2xpY2spPVwic2VhcmNoKClcIlxuICAgICAgICBjbGFzcz1cImZvb3Rlci1idXR0b25cIlxuICAgICAgPjwvcC1idXR0b24+XG4gICAgPC9kaXY+XG4gIDwvbmctdGVtcGxhdGU+XG48L3AtcGFuZWw+XG4iXX0=
138
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibXVsdGktc2VhcmNoLWNyaXRlcmlhLmNvbXBvbmVudC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL25nLXByaW1lLXRvb2xzL3NyYy9saWIvbXVsdGktc2VhcmNoLWNyaXRlcmlhL211bHRpLXNlYXJjaC1jcml0ZXJpYS5jb21wb25lbnQudHMiLCIuLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9uZy1wcmltZS10b29scy9zcmMvbGliL211bHRpLXNlYXJjaC1jcml0ZXJpYS9tdWx0aS1zZWFyY2gtY3JpdGVyaWEuY29tcG9uZW50Lmh0bWwiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLFNBQVMsRUFBRSxLQUFLLEVBQUUsTUFBTSxFQUFFLFlBQVksRUFBVSxNQUFNLGVBQWUsQ0FBQztBQUMvRSxPQUFPLEVBQUUsc0JBQXNCLEVBQUUsTUFBTSxVQUFVLENBQUM7Ozs7Ozs7Ozs7O0FBU2xELE1BQU0sT0FBTyw0QkFBNEI7SUFMekM7UUFNRSwyQkFBc0IsR0FBRyxzQkFBc0IsQ0FBQztRQUV2QyxVQUFLLEdBQVcsRUFBRSxDQUFDO1FBQ25CLGFBQVEsR0FBcUIsRUFBRSxDQUFDO1FBQ2hDLGlCQUFZLEdBQVcsQ0FBQyxDQUFDO1FBQ3pCLFNBQUksR0FBVSxFQUFFLENBQUM7UUFDakIsU0FBSSxHQUF5QixRQUFRLENBQUMsQ0FBQyxpQkFBaUI7UUFDdkQsaUJBQVksR0FBRyxJQUFJLFlBQVksRUFBUyxDQUFDO1FBQ3pDLG1CQUFjLEdBQUcsSUFBSSxZQUFZLEVBQTBCLENBQUM7UUFFdEUsY0FBUyxHQUFZLEtBQUssQ0FBQztRQUMzQixhQUFRLEdBQW1CLEVBQUUsQ0FBQztLQTRHL0I7SUExR0MsUUFBUTtRQUNOLElBQUksQ0FBQyxRQUFRLENBQUMsT0FBTyxDQUFDLENBQUMsSUFBSSxFQUFFLEVBQUU7WUFDN0IsSUFDRSxJQUFJLENBQUMsSUFBSSxLQUFLLHNCQUFzQixDQUFDLFdBQVc7Z0JB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Z0JBQ0csSUFBSTtzQkFBWixLQUFLO2dCQUNJLFlBQVk7c0JBQXJCLE1BQU07Z0JBQ0csY0FBYztzQkFBdkIsTUFBTSIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IENvbXBvbmVudCwgSW5wdXQsIE91dHB1dCwgRXZlbnRFbWl0dGVyLCBPbkluaXQgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcbmltcG9ydCB7IFNlYXJjaENyaXRlcmlhVHlwZUVudW0gfSBmcm9tICcuLi9lbnVtcyc7XG5pbXBvcnQgeyBTZWFyY2hDcml0ZXJpYSB9IGZyb20gJy4uL21vZGVscy9zZWFyY2gtY3JpdGVyaWEnO1xuaW1wb3J0IHsgRmlsdGVyT3B0aW9uIH0gZnJvbSAnLi4vbW9kZWxzL2ZpbHRlci1vcHRpb24nO1xuXG5AQ29tcG9uZW50KHtcbiAgc2VsZWN0b3I6ICdtdWx0aS1zZWFyY2gtY3JpdGVyaWEnLFxuICB0ZW1wbGF0ZVVybDogJy4vbXVsdGktc2VhcmNoLWNyaXRlcmlhLmNvbXBvbmVudC5odG1sJyxcbiAgc3R5bGVVcmxzOiBbJy4vbXVsdGktc2VhcmNoLWNyaXRlcmlhLmNvbXBvbmVudC5jc3MnXSxcbn0pXG5leHBvcnQgY2xhc3MgTXVsdGlTZWFyY2hDcml0ZXJpYUNvbXBvbmVudCBpbXBsZW1lbnRzIE9uSW5pdCB7XG4gIFNlYXJjaENyaXRlcmlhVHlwZUVudW0gPSBTZWFyY2hDcml0ZXJpYVR5cGVFbnVtO1xuXG4gIEBJbnB1dCgpIHRpdGxlOiBTdHJpbmcgPSAnJztcbiAgQElucHV0KCkgY3JpdGVyaWE6IFNlYXJjaENyaXRlcmlhW10gPSBbXTtcbiAgQElucHV0KCkgaW5wdXRzUGVyUm93OiBudW1iZXIgPSAzO1xuICBASW5wdXQoKSBkYXRhOiBhbnlbXSA9IFtdO1xuICBASW5wdXQoKSBtb2RlOiAnc3RhdGljJyB8ICdkeW5hbWljJyA9ICdzdGF0aWMnOyAvLyBBZGQgbW9kZSBpbnB1dFxuICBAT3V0cHV0KCkgZmlsdGVyZWREYXRhID0gbmV3IEV2ZW50RW1pdHRlcjxhbnlbXT4oKTtcbiAgQE91dHB1dCgpIHNlYXJjaENyaXRlcmlhID0gbmV3IEV2ZW50RW1pdHRlcjx7IFtrZXk6IHN0cmluZ106IGFueSB9PigpO1xuXG4gIHNlbGVjdEFsbDogYm9vbGVhbiA9IGZhbHNlO1xuICBzZWxlY3RlZDogRmlsdGVyT3B0aW9uW10gPSBbXTtcblxuICBuZ09uSW5pdCgpIHtcbiAgICB0aGlzLmNyaXRlcmlhLmZvckVhY2goKGl0ZW0pID0+IHtcbiAgICAgIGlmIChcbiAgICAgICAgaXRlbS50eXBlID09PSBTZWFyY2hDcml0ZXJpYVR5cGVFbnVtLk1VTFRJU0VMRUNUICYmXG4gICAgICAgIGl0ZW0uZmlsdGVyT3B0aW9uc1xuICAgICAgKSB7XG4gICAgICAgIHRoaXMuc2VsZWN0ZWQgPSBbLi4uaXRlbS5maWx0ZXJPcHRpb25zXTtcbiAgICAgICAgaXRlbS52YWx1ZSA9IHRoaXMuc2VsZWN0ZWQ7XG4gICAgICB9XG4gICAgfSk7XG4gIH1cblxuICBnZXRDdXJyZW5jeVN5bWJvbChpbnB1dDogU2VhcmNoQ3JpdGVyaWEpOiBzdHJpbmcge1xuICAgIHJldHVybiBpbnB1dC5jdXJyZW5jeSB8fCAnRVVSJztcbiAgfVxuXG4gIHNlYXJjaCgpOiB2b2lkIHtcbiAgICBpZiAodGhpcy5tb2RlID09PSAnZHluYW1pYycpIHtcbiAgICAgIGNvbnN0IGNyaXRlcmlhVmFsdWVzOiB7IFtrZXk6IHN0cmluZ106IGFueSB9ID0ge307XG4gICAgICB0aGlzLmNyaXRlcmlhLmZvckVhY2goKGNyaXRlcmlvbikgPT4ge1xuICAgICAgICBpZiAoY3JpdGVyaW9uLnZhbHVlICE9PSBudWxsICYmIGNyaXRlcmlvbi52YWx1ZSAhPT0gdW5kZWZpbmVkKSB7XG4gICAgICAgICAgY3JpdGVyaWFWYWx1ZXNbY3JpdGVyaW9uLmNvZGVdID0gY3JpdGVyaW9uLnZhbHVlO1xuICAgICAgICB9XG4gICAgICB9KTtcbiAgICAgIHRoaXMuc2VhcmNoQ3JpdGVyaWEuZW1pdChjcml0ZXJpYVZhbHVlcyk7XG4gICAgfSBlbHNlIHtcbiAgICAgIGNvbnN0IGZpbHRlcmVkID0gdGhpcy5kYXRhLmZpbHRlcigoaXRlbSkgPT4ge1xuICAgICAgICByZXR1cm4gdGhpcy5jcml0ZXJpYS5ldmVyeSgoY3JpdGVyaW9uKSA9PiB7XG4gICAgICAgICAgaWYgKGNyaXRlcmlvbi50eXBlID09PSBTZWFyY2hDcml0ZXJpYVR5cGVFbnVtLkRBVEVSQU5HRSkge1xuICAgICAgICAgICAgY29uc3QgW3N0YXJ0RGF0ZSwgZW5kRGF0ZV0gPSBjcml0ZXJpb24udmFsdWUgfHwgW1xuICAgICAgICAgICAgICB1bmRlZmluZWQsXG4gICAgICAgICAgICAgIHVuZGVmaW5lZCxcbiAgICAgICAgICAgIF07XG4gICAgICAgICAgICBjb25zdCBpdGVtRGF0ZSA9IHRoaXMucGFyc2VEYXRlKFxuICAgICAgICAgICAgICBpdGVtW2NyaXRlcmlvbi5jb2RlXSB8fCBpdGVtWydkYXRlJ11cbiAgICAgICAgICAgICk7XG4gICAgICAgICAgICBpZiAoIXN0YXJ0RGF0ZSAmJiAhZW5kRGF0ZSkge1xuICAgICAgICAgICAgICByZXR1cm4gdHJ1ZTtcbiAgICAgICAgICAgIH1cbiAgICAgICAgICAgIGlmIChpdGVtRGF0ZSA9PT0gbnVsbCkge1xuICAgICAgICAgICAgICByZXR1cm4gZmFsc2U7XG4gICAgICAgICAgICB9XG4gICAgICAgICAgICBjb25zdCBwYXJzZWRTdGFydERhdGUgPSBzdGFydERhdGVcbiAgICAgICAgICAgICAgPyB0aGlzLnBhcnNlRGF0ZShzdGFydERhdGUpXG4gICAgICAgICAgICAgIDogbnVsbDtcbiAgICAgICAgICAgIGNvbnN0IHBhcnNlZEVuZERhdGUgPSBlbmREYXRlID8gdGhpcy5wYXJzZURhdGUoZW5kRGF0ZSkgOiBudWxsO1xuICAgICAgICAgICAgcmV0dXJuIChcbiAgICAgICAgICAgICAgKCFwYXJzZWRTdGFydERhdGUgfHwgaXRlbURhdGUgPj0gcGFyc2VkU3RhcnREYXRlKSAmJlxuICAgICAgICAgICAgICAoIXBhcnNlZEVuZERhdGUgfHwgaXRlbURhdGUgPD0gcGFyc2VkRW5kRGF0ZSlcbiAgICAgICAgICAgICk7XG4gICAgICAgICAgfSBlbHNlIGlmIChjcml0ZXJpb24udHlwZSA9PT0gU2VhcmNoQ3JpdGVyaWFUeXBlRW51bS5BTU9VTlQpIHtcbiAgICAgICAgICAgIHJldHVybiAhY3JpdGVyaW9uLnZhbHVlIHx8IGl0ZW1bY3JpdGVyaW9uLmNvZGVdID09PSBjcml0ZXJpb24udmFsdWU7XG4gICAgICAgICAgfSBlbHNlIGlmIChjcml0ZXJpb24udHlwZSA9PT0gU2VhcmNoQ3JpdGVyaWFUeXBlRW51bS5NVUxUSVNFTEVDVCkge1xuICAgICAgICAgICAgcmV0dXJuIChcbiAgICAgICAgICAgICAgIWNyaXRlcmlvbi52YWx1ZSB8fFxuICAgICAgICAgICAgICBjcml0ZXJpb24udmFsdWUuc29tZShcbiAgICAgICAgICAgICAgICAob3B0aW9uOiBGaWx0ZXJPcHRpb24pID0+IG9wdGlvbi5sYWJlbCA9PT0gaXRlbVtjcml0ZXJpb24uY29kZV1cbiAgICAgICAgICAgICAgKVxuICAgICAgICAgICAgKTtcbiAgICAgICAgICB9IGVsc2UgaWYgKGNyaXRlcmlvbi50eXBlID09PSBTZWFyY2hDcml0ZXJpYVR5cGVFbnVtLlNUUklORykge1xuICAgICAgICAgICAgcmV0dXJuIChcbiAgICAgICAgICAgICAgIWNyaXRlcmlvbi52YWx1ZSB8fFxuICAgICAgICAgICAgICBpdGVtW2NyaXRlcmlvbi5jb2RlXVxuICAgICAgICAgICAgICAgIC50b1N0cmluZygpXG4gICAgICAgICAgICAgICAgLnRvTG93ZXJDYXNlKClcbiAgICAgICAgICAgICAgICAuaW5jbHVkZXMoY3JpdGVyaW9uLnZhbHVlLnRvU3RyaW5nKCkudG9Mb3dlckNhc2UoKSlcbiAgICAgICAgICAgICk7XG4gICAgICAgICAgfVxuICAgICAgICAgIHJldHVybiB0cnVlO1xuICAgICAgICB9KTtcbiAgICAgIH0pO1xuICAgICAgdGhpcy5maWx0ZXJlZERhdGEuZW1pdChmaWx0ZXJlZCk7XG4gICAgfVxuICB9XG5cbiAgY2xlYXIoKTogdm9pZCB7XG4gICAgdGhpcy5jcml0ZXJpYS5mb3JFYWNoKChpbnB1dCkgPT4ge1xuICAgICAgaW5wdXQudmFsdWUgPSBudWxsO1xuICAgIH0pO1xuICAgIHRoaXMuZmlsdGVyZWREYXRhLmVtaXQodGhpcy5kYXRhKTtcbiAgfVxuXG4gIG9uU2VsZWN0QWxsQ2hhbmdlKGV2ZW50OiBhbnksIGNyaXRlcmlvbjogU2VhcmNoQ3JpdGVyaWEpIHtcbiAgICB0aGlzLnNlbGVjdEFsbCA9IGV2ZW50LmNoZWNrZWQ7XG4gICAgY3JpdGVyaW9uLnZhbHVlID0gZXZlbnQuY2hlY2tlZCA/IFsuLi4oY3JpdGVyaW9uLmZpbHRlck9wdGlvbnMgPz8gW10pXSA6IFtdO1xuICB9XG5cbiAgcHJpdmF0ZSBwYXJzZURhdGUoZGF0ZVN0cmluZzogc3RyaW5nIHwgRGF0ZSB8IG51bGwgfCB1bmRlZmluZWQpOiBEYXRlIHwgbnVsbCB7XG4gICAgaWYgKCFkYXRlU3RyaW5nKSB7XG4gICAgICByZXR1cm4gbnVsbDtcbiAgICB9XG5cbiAgICBpZiAoT2JqZWN0LnByb3RvdHlwZS50b1N0cmluZy5jYWxsKGRhdGVTdHJpbmcpID09PSAnW29iamVjdCBEYXRlXScpIHtcbiAgICAgIHJldHVybiBkYXRlU3RyaW5nIGFzIERhdGU7XG4gICAgfVxuXG4gICAgY29uc3QgcGFydHMgPSAoZGF0ZVN0cmluZyBhcyBzdHJpbmcpLnNwbGl0KCcvJyk7XG4gICAgaWYgKHBhcnRzLmxlbmd0aCA9PT0gMykge1xuICAgICAgY29uc3QgZGF5ID0gcGFyc2VJbnQocGFydHNbMF0sIDEwKTtcbiAgICAgIGNvbnN0IG1vbnRoID0gcGFyc2VJbnQocGFydHNbMV0sIDEwKSAtIDE7XG4gICAgICBjb25zdCB5ZWFyID0gcGFyc2VJbnQocGFydHNbMl0sIDEwKTtcbiAgICAgIGNvbnN0IGRhdGUgPSBuZXcgRGF0ZSh5ZWFyLCBtb250aCwgZGF5KTtcbiAgICAgIHJldHVybiBpc05hTihkYXRlLmdldFRpbWUoKSkgPyBudWxsIDogZGF0ZTtcbiAgICB9XG4gICAgcmV0dXJuIG51bGw7XG4gIH1cbn1cbiIsIjxwLXBhbmVsIGhlYWRlcj1cInt7IHRpdGxlIH19XCIgW3RvZ2dsZWFibGVdPVwidHJ1ZVwiIFtjb2xsYXBzZWRdPVwidHJ1ZVwiPlxuICA8ZGl2XG4gICAgY2xhc3M9XCJjcml0ZXJpYS1jb250YWluZXJcIlxuICAgIFtuZ1N0eWxlXT1cInsgJ2dyaWQtdGVtcGxhdGUtY29sdW1ucyc6ICdyZXBlYXQoJyArIGlucHV0c1BlclJvdyArICcsIDFmciknIH1cIlxuICA+XG4gICAgPGRpdiAqbmdGb3I9XCJsZXQgaW5wdXQgb2YgY3JpdGVyaWFcIiBjbGFzcz1cImNyaXRlcmlhLWl0ZW1cIj5cbiAgICAgIDxsYWJlbCBjbGFzcz1cImJzYy1sYWJlbFwiPnt7IGlucHV0LnRpdGxlIH19PC9sYWJlbD5cblxuICAgICAgPG5nLWNvbnRhaW5lciBbbmdTd2l0Y2hdPVwiaW5wdXQudHlwZVwiPlxuICAgICAgICA8cC1jYWxlbmRhclxuICAgICAgICAgICpuZ1N3aXRjaENhc2U9XCJTZWFyY2hDcml0ZXJpYVR5cGVFbnVtLkRBVEVSQU5HRVwiXG4gICAgICAgICAgWyhuZ01vZGVsKV09XCJpbnB1dC52YWx1ZVwiXG4gICAgICAgICAgc2VsZWN0aW9uTW9kZT1cInJhbmdlXCJcbiAgICAgICAgICBbZGF0ZUZvcm1hdF09XCInZGQvbW0veXknXCJcbiAgICAgICAgICBjbGFzcz1cImZ1bGwtd2lkdGgtaW5wdXRcIlxuICAgICAgICAgIFtzaG93SWNvbl09XCJ0cnVlXCJcbiAgICAgICAgICAobmdNb2RlbENoYW5nZSk9XCJpbnB1dC52YWx1ZSA9ICRldmVudFwiXG4gICAgICAgID48L3AtY2FsZW5kYXI+XG4gICAgICAgIDxwLWNhbGVuZGFyXG4gICAgICAgICAgKm5nU3dpdGNoQ2FzZT1cIlNlYXJjaENyaXRlcmlhVHlwZUVudW0uREFURVwiXG4gICAgICAgICAgW25nTW9kZWxdPVwiaW5wdXQudmFsdWVcIlxuICAgICAgICAgIChuZ01vZGVsQ2hhbmdlKT1cImlucHV0LnZhbHVlID0gJGV2ZW50XCJcbiAgICAgICAgICBbZGF0ZUZvcm1hdF09XCInZGQvbW0veXknXCJcbiAgICAgICAgICBbc2hvd0ljb25dPVwidHJ1ZVwiXG4gICAgICAgICAgY2xhc3M9XCJmdWxsLXdpZHRoLWlucHV0XCJcbiAgICAgICAgPjwvcC1jYWxlbmRhcj5cbiAgICAgICAgPGlucHV0XG4gICAgICAgICAgKm5nU3dpdGNoQ2FzZT1cIlNlYXJjaENyaXRlcmlhVHlwZUVudW0uU1RSSU5HXCJcbiAgICAgICAgICB0eXBlPVwidGV4dFwiXG4gICAgICAgICAgcElucHV0VGV4dFxuICAgICAgICAgIFsobmdNb2RlbCldPVwiaW5wdXQudmFsdWVcIlxuICAgICAgICAgIGNsYXNzPVwiZnVsbC13aWR0aC1pbnB1dFwiXG4gICAgICAgIC8+XG5cbiAgICAgICAgPHAtaW5wdXROdW1iZXJcbiAgICAgICAgICAqbmdTd2l0Y2hDYXNlPVwiU2VhcmNoQ3JpdGVyaWFUeXBlRW51bS5BTU9VTlRcIlxuICAgICAgICAgIHR5cGU9XCJudW1iZXJcIlxuICAgICAgICAgIFsobmdNb2RlbCldPVwiaW5wdXQudmFsdWVcIlxuICAgICAgICAgIG1vZGU9XCJkZWNpbWFsXCJcbiAgICAgICAgICBpbnB1dElkPVwibWlubWF4ZnJhY3Rpb25cIlxuICAgICAgICAgIFttaW5GcmFjdGlvbkRpZ2l0c109XCIyXCJcbiAgICAgICAgICBbbWF4RnJhY3Rpb25EaWdpdHNdPVwiNVwiXG4gICAgICAgICAgW3BsYWNlaG9sZGVyXT1cImdldEN1cnJlbmN5U3ltYm9sKGlucHV0KVwiXG4gICAgICAgICAgY2xhc3M9XCJmdWxsLXdpZHRoLWlucHV0XCJcbiAgICAgICAgPjwvcC1pbnB1dE51bWJlcj5cblxuICAgICAgICA8cC1pbnB1dE51bWJlclxuICAgICAgICAgICpuZ1N3aXRjaENhc2U9XCJTZWFyY2hDcml0ZXJpYVR5cGVFbnVtLk5VTUJFUlwiXG4gICAgICAgICAgdHlwZT1cIm51bWJlclwiXG4gICAgICAgICAgWyhuZ01vZGVsKV09XCJpbnB1dC52YWx1ZVwiXG4gICAgICAgICAgY2xhc3M9XCJmdWxsLXdpZHRoLWlucHV0XCJcbiAgICAgICAgPjwvcC1pbnB1dE51bWJlcj5cblxuICAgICAgICA8cC1tdWx0aVNlbGVjdFxuICAgICAgICAgICpuZ1N3aXRjaENhc2U9XCJTZWFyY2hDcml0ZXJpYVR5cGVFbnVtLk1VTFRJU0VMRUNUXCJcbiAgICAgICAgICBbb3B0aW9uc109XCJpbnB1dC5maWx0ZXJPcHRpb25zXCJcbiAgICAgICAgICBbZGlzcGxheV09XCInY2hpcCdcIlxuICAgICAgICAgIHBsYWNlaG9sZGVyPVwiU2VsZWN0XCJcbiAgICAgICAgICBbc2VsZWN0QWxsXT1cInNlbGVjdEFsbFwiXG4gICAgICAgICAgWyhuZ01vZGVsKV09XCJpbnB1dC52YWx1ZVwiXG4gICAgICAgICAgb3B0aW9uTGFiZWw9XCJsYWJlbFwiXG4gICAgICAgICAgKG9uU2VsZWN0QWxsQ2hhbmdlKT1cIm9uU2VsZWN0QWxsQ2hhbmdlKCRldmVudCwgaW5wdXQpXCJcbiAgICAgICAgICBjbGFzcz1cImN1c3RvbS1tdWx0aXNlbGVjdCBmdWxsLXdpZHRoLWlucHV0XCJcbiAgICAgICAgPjwvcC1tdWx0aVNlbGVjdD5cbiAgICAgIDwvbmctY29udGFpbmVyPlxuICAgIDwvZGl2PlxuICA8L2Rpdj5cblxuICA8bmctdGVtcGxhdGUgcFRlbXBsYXRlPVwiZm9vdGVyXCI+XG4gICAgPGRpdiBjbGFzcz1cImZvb3Rlci1idXR0b25zXCI+XG4gICAgICA8cC1idXR0b25cbiAgICAgICAgbGFiZWw9XCJFZmZhY2VyXCJcbiAgICAgICAgaWNvbj1cInBpIHBpLXRpbWVzXCJcbiAgICAgICAgKGNsaWNrKT1cImNsZWFyKClcIlxuICAgICAgICBjbGFzcz1cImZvb3Rlci1idXR0b25cIlxuICAgICAgPjwvcC1idXR0b24+XG4gICAgICA8cC1idXR0b25cbiAgICAgICAgbGFiZWw9XCJSZWNoZXJjaGVyXCJcbiAgICAgICAgaWNvbj1cInBpIHBpLXNlYXJjaFwiXG4gICAgICAgIChjbGljayk9XCJzZWFyY2goKVwiXG4gICAgICAgIGNsYXNzPVwiZm9vdGVyLWJ1dHRvblwiXG4gICAgICA+PC9wLWJ1dHRvbj5cbiAgICA8L2Rpdj5cbiAgPC9uZy10ZW1wbGF0ZT5cbjwvcC1wYW5lbD5cbiJdfQ==