@rlucan/ui 17.1.4 → 18.2.1

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 (43) hide show
  1. package/esm2022/lib/action-button/action-button.component.mjs +3 -3
  2. package/esm2022/lib/action-icon/action-icon.component.mjs +3 -3
  3. package/esm2022/lib/auto-complete/auto-complete.component.mjs +403 -0
  4. package/esm2022/lib/autocomplete/autocomplete.component.mjs +3 -3
  5. package/esm2022/lib/avatar/avatar.component.mjs +3 -3
  6. package/esm2022/lib/button/button.component.mjs +3 -3
  7. package/esm2022/lib/checkbox/checkbox.component.mjs +3 -3
  8. package/esm2022/lib/checkbox-group/checkbox-group.component.mjs +3 -3
  9. package/esm2022/lib/currency/currency.component.mjs +3 -3
  10. package/esm2022/lib/date/date.component.mjs +3 -3
  11. package/esm2022/lib/dialog/dialog.component.mjs +3 -3
  12. package/esm2022/lib/directives/force-visibility/force-visibility.directive.mjs +3 -3
  13. package/esm2022/lib/editor/editor.component.mjs +4 -4
  14. package/esm2022/lib/elements/burger/burger.component.mjs +3 -3
  15. package/esm2022/lib/elements/expander/expander.component.mjs +3 -3
  16. package/esm2022/lib/elements/validation-message/validation-message.component.mjs +3 -3
  17. package/esm2022/lib/file/file.component.mjs +3 -3
  18. package/esm2022/lib/file-uploader/ui-file-uploader.component.mjs +3 -3
  19. package/esm2022/lib/input/input.component.mjs +3 -3
  20. package/esm2022/lib/input-autocomplete/input-autocomplete.component.mjs +3 -3
  21. package/esm2022/lib/layouts/base/ui-base-layout.component.mjs +3 -3
  22. package/esm2022/lib/layouts/base/ui-base.component.mjs +3 -3
  23. package/esm2022/lib/layouts/simple/ui-simple-layout.component.mjs +3 -3
  24. package/esm2022/lib/layouts/simple/ui-simple.component.mjs +3 -3
  25. package/esm2022/lib/radio/radio.component.mjs +3 -3
  26. package/esm2022/lib/radio-group/radio-group.component.mjs +3 -3
  27. package/esm2022/lib/select/select.component.mjs +3 -3
  28. package/esm2022/lib/services/message-box.service.mjs +7 -7
  29. package/esm2022/lib/services/toast.service.mjs +3 -3
  30. package/esm2022/lib/services/ui-file.service.mjs +3 -3
  31. package/esm2022/lib/services/ui-translate.service.mjs +5 -5
  32. package/esm2022/lib/submit-button/submit-button.component.mjs +3 -3
  33. package/esm2022/lib/table/table.component.mjs +3 -3
  34. package/esm2022/lib/text-area/text-area.component.mjs +3 -3
  35. package/esm2022/lib/ui.module.mjs +10 -5
  36. package/esm2022/public-api.mjs +2 -1
  37. package/fesm2022/rlucan-ui.mjs +511 -117
  38. package/fesm2022/rlucan-ui.mjs.map +1 -1
  39. package/lib/auto-complete/auto-complete.component.d.ts +65 -0
  40. package/lib/ui.module.d.ts +36 -35
  41. package/package.json +9 -9
  42. package/public-api.d.ts +1 -0
  43. package/scss/ui-defaults.scss +46 -46
@@ -66,10 +66,10 @@ export class ActionButtonComponent extends ButtonComponent {
66
66
  $event.stopPropagation();
67
67
  $event.preventDefault();
68
68
  }
69
- static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.1.2", ngImport: i0, type: ActionButtonComponent, deps: null, target: i0.ɵɵFactoryTarget.Component }); }
70
- static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "17.1.2", type: ActionButtonComponent, selector: "ui-action-button", inputs: { withPrimary: "withPrimary", overlayAlignment: "overlayAlignment", hasBackdrop: "hasBackdrop", triggerTpl: "triggerTpl", autoClose: "autoClose" }, outputs: { primaryClick: "primaryClick", overlayToggled: "overlayToggled" }, host: { listeners: { "document:keydown.escape": "onKeydownHandler($event)" }, properties: { "class": "this.class" } }, viewQueries: [{ propertyName: "trigger1", first: true, predicate: ["trigger1"], descendants: true }], usesInheritance: true, ngImport: i0, template: "<ng-container *ngIf=\"triggerTpl\">\r\n <div style=\"display: flex;\" (click)=\"toggleOverlay(true)\" cdkOverlayOrigin #trigger1=\"cdkOverlayOrigin\">\r\n <ng-template [ngTemplateOutlet]=\"triggerTpl\"></ng-template>\r\n </div>\r\n</ng-container>\r\n\r\n<ng-container *ngIf=\"!triggerTpl && withPrimary\">\r\n <button matRipple [type]=\"type\" [disabled]=\"isDisabled || busy\" [ngClass]=\"{busy: busy, 'icon-hover': iconHover}\" [class]=\"class\" (click)=\"primaryClick.emit()\">\r\n <div class=\"hover-container\"></div>\r\n {{label}}\r\n <div class=\"mat-icon-container\" (mouseenter)=\"iconHover = true\" (mouseleave)=\"iconHover = false\" cdkOverlayOrigin #trigger1=\"cdkOverlayOrigin\" (click)=\"togglePrimaryOverlay(true, $event)\">\r\n \u25BC\r\n </div>\r\n <mat-progress-bar *ngIf=\"busy\" mode=\"indeterminate\" [color]=\"color\"></mat-progress-bar>\r\n </button>\r\n</ng-container>\r\n\r\n<ng-container *ngIf=\"!triggerTpl && !withPrimary\">\r\n <button matRipple [type]=\"type\" [disabled]=\"isDisabled || busy\" [ngClass]=\"{busy: busy}\" [class]=\"class\" cdkOverlayOrigin #trigger1=\"cdkOverlayOrigin\" (click)=\"toggleOverlay(true)\">\r\n {{label}}\r\n <div class=\"mat-icon-container\">\r\n \u25BC\r\n </div>\r\n <mat-progress-bar *ngIf=\"busy\" mode=\"indeterminate\" [color]=\"color\"></mat-progress-bar>\r\n </button>\r\n</ng-container>\r\n\r\n<ng-template *ngIf=\"deferredRender\" cdkConnectedOverlay [cdkConnectedOverlayPositions]=\"positions\" [cdkConnectedOverlayOrigin]=\"trigger1\" [cdkConnectedOverlayOpen]=\"openOverlay\" [cdkConnectedOverlayHasBackdrop]=\"hasBackdrop\" [cdkConnectedOverlayDisableClose]=\"!autoClose\" (overlayOutsideClick)=\"toggleOverlay(false)\">\r\n <ng-content></ng-content>\r\n</ng-template>\r\n", styles: [":host{display:flex;align-items:center}:host button{font-size:1em;width:100%;cursor:pointer;outline:none;position:relative;display:flex;align-items:center;justify-content:center;border:1px solid transparent;transition:all .15s ease-in-out}:host button mat-progress-bar{position:absolute;bottom:1px;height:2px;border-bottom-left-radius:10px;border-bottom-right-radius:10px}:host button mat-icon{margin-right:4px;height:16px;width:16px;font-size:16px}:host.small button mat-icon{width:14px;height:14px;font-size:14px;margin-right:2px}\n", ":host .mat-icon-container{margin-left:.25em}:host button{font-size:1em}:host.small.with-primary button{padding-right:2.9em!important}:host.small.with-primary button .mat-icon-container{position:absolute;top:0;right:0;bottom:0;display:flex;align-items:center;justify-content:center;border-left:1px solid transparent;padding-right:.6em;padding-left:.6em}:host.small.with-primary button .hover-container{position:absolute;inset:0 2.2em 0 0}\n"], dependencies: [{ kind: "directive", type: i1.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i1.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "component", type: i2.MatProgressBar, selector: "mat-progress-bar", inputs: ["color", "value", "bufferValue", "mode"], outputs: ["animationEnd"], exportAs: ["matProgressBar"] }, { kind: "directive", type: i3.CdkConnectedOverlay, selector: "[cdk-connected-overlay], [connected-overlay], [cdkConnectedOverlay]", inputs: ["cdkConnectedOverlayOrigin", "cdkConnectedOverlayPositions", "cdkConnectedOverlayPositionStrategy", "cdkConnectedOverlayOffsetX", "cdkConnectedOverlayOffsetY", "cdkConnectedOverlayWidth", "cdkConnectedOverlayHeight", "cdkConnectedOverlayMinWidth", "cdkConnectedOverlayMinHeight", "cdkConnectedOverlayBackdropClass", "cdkConnectedOverlayPanelClass", "cdkConnectedOverlayViewportMargin", "cdkConnectedOverlayScrollStrategy", "cdkConnectedOverlayOpen", "cdkConnectedOverlayDisableClose", "cdkConnectedOverlayTransformOriginOn", "cdkConnectedOverlayHasBackdrop", "cdkConnectedOverlayLockPosition", "cdkConnectedOverlayFlexibleDimensions", "cdkConnectedOverlayGrowAfterOpen", "cdkConnectedOverlayPush", "cdkConnectedOverlayDisposeOnNavigation"], outputs: ["backdropClick", "positionChange", "attach", "detach", "overlayKeydown", "overlayOutsideClick"], exportAs: ["cdkConnectedOverlay"] }, { kind: "directive", type: i3.CdkOverlayOrigin, selector: "[cdk-overlay-origin], [overlay-origin], [cdkOverlayOrigin]", exportAs: ["cdkOverlayOrigin"] }, { kind: "directive", type: i4.MatRipple, selector: "[mat-ripple], [matRipple]", inputs: ["matRippleColor", "matRippleUnbounded", "matRippleCentered", "matRippleRadius", "matRippleAnimation", "matRippleDisabled", "matRippleTrigger"], exportAs: ["matRipple"] }] }); }
69
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.1", ngImport: i0, type: ActionButtonComponent, deps: null, target: i0.ɵɵFactoryTarget.Component }); }
70
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.1", type: ActionButtonComponent, selector: "ui-action-button", inputs: { withPrimary: "withPrimary", overlayAlignment: "overlayAlignment", hasBackdrop: "hasBackdrop", triggerTpl: "triggerTpl", autoClose: "autoClose" }, outputs: { primaryClick: "primaryClick", overlayToggled: "overlayToggled" }, host: { listeners: { "document:keydown.escape": "onKeydownHandler($event)" }, properties: { "class": "this.class" } }, viewQueries: [{ propertyName: "trigger1", first: true, predicate: ["trigger1"], descendants: true }], usesInheritance: true, ngImport: i0, template: "<ng-container *ngIf=\"triggerTpl\">\r\n <div style=\"display: flex;\" (click)=\"toggleOverlay(true)\" cdkOverlayOrigin #trigger1=\"cdkOverlayOrigin\">\r\n <ng-template [ngTemplateOutlet]=\"triggerTpl\"></ng-template>\r\n </div>\r\n</ng-container>\r\n\r\n<ng-container *ngIf=\"!triggerTpl && withPrimary\">\r\n <button matRipple [type]=\"type\" [disabled]=\"isDisabled || busy\" [ngClass]=\"{busy: busy, 'icon-hover': iconHover}\" [class]=\"class\" (click)=\"primaryClick.emit()\">\r\n <div class=\"hover-container\"></div>\r\n {{label}}\r\n <div class=\"mat-icon-container\" (mouseenter)=\"iconHover = true\" (mouseleave)=\"iconHover = false\" cdkOverlayOrigin #trigger1=\"cdkOverlayOrigin\" (click)=\"togglePrimaryOverlay(true, $event)\">\r\n \u25BC\r\n </div>\r\n <mat-progress-bar *ngIf=\"busy\" mode=\"indeterminate\" [color]=\"color\"></mat-progress-bar>\r\n </button>\r\n</ng-container>\r\n\r\n<ng-container *ngIf=\"!triggerTpl && !withPrimary\">\r\n <button matRipple [type]=\"type\" [disabled]=\"isDisabled || busy\" [ngClass]=\"{busy: busy}\" [class]=\"class\" cdkOverlayOrigin #trigger1=\"cdkOverlayOrigin\" (click)=\"toggleOverlay(true)\">\r\n {{label}}\r\n <div class=\"mat-icon-container\">\r\n \u25BC\r\n </div>\r\n <mat-progress-bar *ngIf=\"busy\" mode=\"indeterminate\" [color]=\"color\"></mat-progress-bar>\r\n </button>\r\n</ng-container>\r\n\r\n<ng-template *ngIf=\"deferredRender\" cdkConnectedOverlay [cdkConnectedOverlayPositions]=\"positions\" [cdkConnectedOverlayOrigin]=\"trigger1\" [cdkConnectedOverlayOpen]=\"openOverlay\" [cdkConnectedOverlayHasBackdrop]=\"hasBackdrop\" [cdkConnectedOverlayDisableClose]=\"!autoClose\" (overlayOutsideClick)=\"toggleOverlay(false)\">\r\n <ng-content></ng-content>\r\n</ng-template>\r\n", styles: [":host{display:flex;align-items:center}:host button{font-size:1em;width:100%;cursor:pointer;outline:none;position:relative;display:flex;align-items:center;justify-content:center;border:1px solid transparent;transition:all .15s ease-in-out}:host button mat-progress-bar{position:absolute;bottom:1px;height:2px;border-bottom-left-radius:10px;border-bottom-right-radius:10px}:host button mat-icon{margin-right:4px;height:16px;width:16px;font-size:16px}:host.small button mat-icon{width:14px;height:14px;font-size:14px;margin-right:2px}\n", ":host .mat-icon-container{margin-left:.25em}:host button{font-size:1em}:host.small.with-primary button{padding-right:2.9em!important}:host.small.with-primary button .mat-icon-container{position:absolute;top:0;right:0;bottom:0;display:flex;align-items:center;justify-content:center;border-left:1px solid transparent;padding-right:.6em;padding-left:.6em}:host.small.with-primary button .hover-container{position:absolute;inset:0 2.2em 0 0}\n"], dependencies: [{ kind: "directive", type: i1.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i1.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "component", type: i2.MatProgressBar, selector: "mat-progress-bar", inputs: ["color", "value", "bufferValue", "mode"], outputs: ["animationEnd"], exportAs: ["matProgressBar"] }, { kind: "directive", type: i3.CdkConnectedOverlay, selector: "[cdk-connected-overlay], [connected-overlay], [cdkConnectedOverlay]", inputs: ["cdkConnectedOverlayOrigin", "cdkConnectedOverlayPositions", "cdkConnectedOverlayPositionStrategy", "cdkConnectedOverlayOffsetX", "cdkConnectedOverlayOffsetY", "cdkConnectedOverlayWidth", "cdkConnectedOverlayHeight", "cdkConnectedOverlayMinWidth", "cdkConnectedOverlayMinHeight", "cdkConnectedOverlayBackdropClass", "cdkConnectedOverlayPanelClass", "cdkConnectedOverlayViewportMargin", "cdkConnectedOverlayScrollStrategy", "cdkConnectedOverlayOpen", "cdkConnectedOverlayDisableClose", "cdkConnectedOverlayTransformOriginOn", "cdkConnectedOverlayHasBackdrop", "cdkConnectedOverlayLockPosition", "cdkConnectedOverlayFlexibleDimensions", "cdkConnectedOverlayGrowAfterOpen", "cdkConnectedOverlayPush", "cdkConnectedOverlayDisposeOnNavigation"], outputs: ["backdropClick", "positionChange", "attach", "detach", "overlayKeydown", "overlayOutsideClick"], exportAs: ["cdkConnectedOverlay"] }, { kind: "directive", type: i3.CdkOverlayOrigin, selector: "[cdk-overlay-origin], [overlay-origin], [cdkOverlayOrigin]", exportAs: ["cdkOverlayOrigin"] }, { kind: "directive", type: i4.MatRipple, selector: "[mat-ripple], [matRipple]", inputs: ["matRippleColor", "matRippleUnbounded", "matRippleCentered", "matRippleRadius", "matRippleAnimation", "matRippleDisabled", "matRippleTrigger"], exportAs: ["matRipple"] }] }); }
71
71
  }
72
- i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.1.2", ngImport: i0, type: ActionButtonComponent, decorators: [{
72
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.1", ngImport: i0, type: ActionButtonComponent, decorators: [{
73
73
  type: Component,
74
74
  args: [{ selector: 'ui-action-button', template: "<ng-container *ngIf=\"triggerTpl\">\r\n <div style=\"display: flex;\" (click)=\"toggleOverlay(true)\" cdkOverlayOrigin #trigger1=\"cdkOverlayOrigin\">\r\n <ng-template [ngTemplateOutlet]=\"triggerTpl\"></ng-template>\r\n </div>\r\n</ng-container>\r\n\r\n<ng-container *ngIf=\"!triggerTpl && withPrimary\">\r\n <button matRipple [type]=\"type\" [disabled]=\"isDisabled || busy\" [ngClass]=\"{busy: busy, 'icon-hover': iconHover}\" [class]=\"class\" (click)=\"primaryClick.emit()\">\r\n <div class=\"hover-container\"></div>\r\n {{label}}\r\n <div class=\"mat-icon-container\" (mouseenter)=\"iconHover = true\" (mouseleave)=\"iconHover = false\" cdkOverlayOrigin #trigger1=\"cdkOverlayOrigin\" (click)=\"togglePrimaryOverlay(true, $event)\">\r\n \u25BC\r\n </div>\r\n <mat-progress-bar *ngIf=\"busy\" mode=\"indeterminate\" [color]=\"color\"></mat-progress-bar>\r\n </button>\r\n</ng-container>\r\n\r\n<ng-container *ngIf=\"!triggerTpl && !withPrimary\">\r\n <button matRipple [type]=\"type\" [disabled]=\"isDisabled || busy\" [ngClass]=\"{busy: busy}\" [class]=\"class\" cdkOverlayOrigin #trigger1=\"cdkOverlayOrigin\" (click)=\"toggleOverlay(true)\">\r\n {{label}}\r\n <div class=\"mat-icon-container\">\r\n \u25BC\r\n </div>\r\n <mat-progress-bar *ngIf=\"busy\" mode=\"indeterminate\" [color]=\"color\"></mat-progress-bar>\r\n </button>\r\n</ng-container>\r\n\r\n<ng-template *ngIf=\"deferredRender\" cdkConnectedOverlay [cdkConnectedOverlayPositions]=\"positions\" [cdkConnectedOverlayOrigin]=\"trigger1\" [cdkConnectedOverlayOpen]=\"openOverlay\" [cdkConnectedOverlayHasBackdrop]=\"hasBackdrop\" [cdkConnectedOverlayDisableClose]=\"!autoClose\" (overlayOutsideClick)=\"toggleOverlay(false)\">\r\n <ng-content></ng-content>\r\n</ng-template>\r\n", styles: [":host{display:flex;align-items:center}:host button{font-size:1em;width:100%;cursor:pointer;outline:none;position:relative;display:flex;align-items:center;justify-content:center;border:1px solid transparent;transition:all .15s ease-in-out}:host button mat-progress-bar{position:absolute;bottom:1px;height:2px;border-bottom-left-radius:10px;border-bottom-right-radius:10px}:host button mat-icon{margin-right:4px;height:16px;width:16px;font-size:16px}:host.small button mat-icon{width:14px;height:14px;font-size:14px;margin-right:2px}\n", ":host .mat-icon-container{margin-left:.25em}:host button{font-size:1em}:host.small.with-primary button{padding-right:2.9em!important}:host.small.with-primary button .mat-icon-container{position:absolute;top:0;right:0;bottom:0;display:flex;align-items:center;justify-content:center;border-left:1px solid transparent;padding-right:.6em;padding-left:.6em}:host.small.with-primary button .hover-container{position:absolute;inset:0 2.2em 0 0}\n"] }]
75
75
  }], propDecorators: { withPrimary: [{
@@ -14,10 +14,10 @@ export class ActionIconComponent {
14
14
  }
15
15
  ngOnInit() {
16
16
  }
17
- static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.1.2", ngImport: i0, type: ActionIconComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
18
- static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "17.1.2", type: ActionIconComponent, selector: "ui-action-icon", inputs: { size: "size", color: "color", disabled: "disabled", busy: "busy", matIcon: "matIcon" }, host: { properties: { "class": "this.class" } }, ngImport: i0, template: "<mat-icon>{{matIcon}}</mat-icon>\r\n", styles: [":host{display:flex}:host:not(.disabled){cursor:pointer}:host mat-icon{width:1em;height:1em;transition:color .25s}:host.small mat-icon{font-size:1em}:host.smaller mat-icon{font-size:1.25em}:host.normal mat-icon{font-size:1.5em}:host.larger mat-icon{font-size:1.75em}:host.large mat-icon{font-size:2em}\n"], dependencies: [{ kind: "component", type: i1.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }] }); }
17
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.1", ngImport: i0, type: ActionIconComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
18
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.1", type: ActionIconComponent, selector: "ui-action-icon", inputs: { size: "size", color: "color", disabled: "disabled", busy: "busy", matIcon: "matIcon" }, host: { properties: { "class": "this.class" } }, ngImport: i0, template: "<mat-icon>{{matIcon}}</mat-icon>\r\n", styles: [":host{display:flex}:host:not(.disabled){cursor:pointer}:host mat-icon{width:1em;height:1em;transition:color .25s}:host.small mat-icon{font-size:1em}:host.smaller mat-icon{font-size:1.25em}:host.normal mat-icon{font-size:1.5em}:host.larger mat-icon{font-size:1.75em}:host.large mat-icon{font-size:2em}\n"], dependencies: [{ kind: "component", type: i1.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }] }); }
19
19
  }
20
- i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.1.2", ngImport: i0, type: ActionIconComponent, decorators: [{
20
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.1", ngImport: i0, type: ActionIconComponent, decorators: [{
21
21
  type: Component,
22
22
  args: [{ selector: 'ui-action-icon', template: "<mat-icon>{{matIcon}}</mat-icon>\r\n", styles: [":host{display:flex}:host:not(.disabled){cursor:pointer}:host mat-icon{width:1em;height:1em;transition:color .25s}:host.small mat-icon{font-size:1em}:host.smaller mat-icon{font-size:1.25em}:host.normal mat-icon{font-size:1.5em}:host.larger mat-icon{font-size:1.75em}:host.large mat-icon{font-size:2em}\n"] }]
23
23
  }], ctorParameters: () => [], propDecorators: { size: [{
@@ -0,0 +1,403 @@
1
+ import { Component, Host, Input, Optional, SkipSelf, ViewChild } from '@angular/core';
2
+ import { NG_VALUE_ACCESSOR, UntypedFormControl } from '@angular/forms';
3
+ import { debounceTime } from 'rxjs/operators';
4
+ import { UiBaseComponent } from '../layouts/base/ui-base.component';
5
+ import * as i0 from "@angular/core";
6
+ import * as i1 from "@angular/forms";
7
+ import * as i2 from "@angular/common";
8
+ import * as i3 from "@angular/material/icon";
9
+ import * as i4 from "@angular/material/checkbox";
10
+ import * as i5 from "@angular/cdk/overlay";
11
+ import * as i6 from "../button/button.component";
12
+ import * as i7 from "../input/input.component";
13
+ import * as i8 from "../directives/force-visibility/force-visibility.directive";
14
+ export class AutoCompleteComponent extends UiBaseComponent {
15
+ constructor(controlContainer,
16
+ // @Optional() @Self() public ngControl: NgControl,
17
+ cdr) {
18
+ super();
19
+ this.controlContainer = controlContainer;
20
+ this.cdr = cdr;
21
+ this.SHOWN_USERS_DELTA = 10;
22
+ this.templates = {};
23
+ this.multiple = true;
24
+ this.allowNew = false;
25
+ this.displayAttribute = 'name';
26
+ this.idAttribute = 'id';
27
+ this.panelClass = '';
28
+ this.filteredOptions = [];
29
+ this.filteredMaxItemsShown = this.SHOWN_USERS_DELTA;
30
+ // skipBlur = false;
31
+ // typingStarted = false;
32
+ this.messageShown = false;
33
+ this.alreadyAdded = false;
34
+ this.optionsVisible = false;
35
+ this.inputFocused = false;
36
+ this.inputControl = new UntypedFormControl('');
37
+ this.checkboxSelection = new Set();
38
+ this.optionDisabled = () => false;
39
+ this.optionFormatter = (o) => o[this.displayAttribute];
40
+ }
41
+ buildSearchableOptions() {
42
+ return this.options.map(o => ({
43
+ ...o,
44
+ unselectable: this.optionDisabled(o),
45
+ searchString: `${this.optionFormatter(o)}`.toLowerCase()
46
+ }));
47
+ }
48
+ filterOptions(text) {
49
+ this.filteredMaxItemsShown = this.SHOWN_USERS_DELTA;
50
+ return this.searchableOptions.filter(u => u.searchString.includes(text));
51
+ }
52
+ ngOnChanges(changes) {
53
+ if (changes['options']) {
54
+ this.searchableOptions = this.buildSearchableOptions();
55
+ this.inputControl.setValue('');
56
+ }
57
+ }
58
+ ngOnInit() {
59
+ console.log('ng on init');
60
+ super.ngOnInit();
61
+ // if (this.formControlName && this.controlContainer.control) {
62
+ // this.control = this.controlContainer.control.get(this.formControlName) as FormControl;
63
+ // }
64
+ //
65
+ // this.user = this.control.value;
66
+ // this.value = this.control.value;
67
+ // this.searchableOptions = this.buildSearchableOptions();
68
+ // if (this.control.disabled) {
69
+ // this.inputControl.disable();
70
+ // }
71
+ // this.control.statusChanges.subscribe(v => {
72
+ // if (v === 'DISABLED') {
73
+ // this.inputControl.disable();
74
+ // }
75
+ // if (v === 'VALID') {
76
+ // this.inputControl.enable();
77
+ // }
78
+ // });
79
+ this.inputControl.valueChanges.pipe(debounceTime(200)).subscribe(v => {
80
+ const lcv = v.toLowerCase();
81
+ this.filteredOptions = this.filterOptions(lcv);
82
+ if (this.filteredOptions.length > 0) {
83
+ this.focusedOption = this.filteredOptions[0];
84
+ }
85
+ else {
86
+ this.focusedOption = undefined;
87
+ }
88
+ if (v === '') {
89
+ this.focusedOption = undefined;
90
+ this.selectedOption = undefined;
91
+ }
92
+ // if (v !== '') {
93
+ // if (this.newUsername === v) {
94
+ // this.selectedUser = {id: null, fullName: v};
95
+ // this.initialUser = this.selectedUser;
96
+ // } else {
97
+ // if (this.filteredGroupsAndUsers.length > 0) {
98
+ // this.focusedGroup = this.filteredGroupsAndUsers[0];
99
+ // this.focusedUser = this.focusedGroup.filteredUsers[0];
100
+ // } else {
101
+ // this.focusedGroup = null;
102
+ // this.focusedUser = null;
103
+ // }
104
+ // if ((this.focusedUser && !this.focusedUser.unselectable) || this.focusedUser === this.initialUser) {
105
+ // this.selectedUser = this.focusedUser;
106
+ // } else {
107
+ // this.selectedUser = null;
108
+ // }
109
+ // }
110
+ // } else {
111
+ // this.selectedUser = null;
112
+ // this.focusedUser = null;
113
+ // this.focusedGroup = null;
114
+ // }
115
+ // this.onChange(v);
116
+ });
117
+ // if (this.user !== null) {
118
+ // let selectedUser = null;
119
+ // this.groupsAndUsers.find(g => g.users.find(u => {
120
+ // if (u.id === this.user.id) {
121
+ // selectedUser = u;
122
+ // return true;
123
+ // } else {
124
+ // return false;
125
+ // }
126
+ // }));
127
+ //
128
+ // if (selectedUser) {
129
+ // this.inputControl.setValue(selectedUser.fullName);
130
+ // this.selectedUser = selectedUser;
131
+ // } else {
132
+ // this.newUsername = this.user.fullName;
133
+ // this.inputControl.setValue(this.user.fullName);
134
+ // this.selectedUser = { id: null, fullName: this.newUsername };
135
+ // }
136
+ // } else {
137
+ // this.selectedUser = null;
138
+ // this.inputControl.setValue('');
139
+ // }
140
+ // this.initialUser = this.selectedUser;
141
+ // }
142
+ }
143
+ inputFocusChanged(focused) {
144
+ if (focused) {
145
+ this.optionsVisible = true;
146
+ }
147
+ }
148
+ // setTimeout(() => {
149
+ // if (!focused && this.skipBlur) {
150
+ // this.skipBlur = false;
151
+ // return;
152
+ // }
153
+ // if (!this.typingStarted && focused) {
154
+ // this.inputControl.setValue(this.inputControl.value);
155
+ // this.typingStarted = true;
156
+ // }
157
+ // if (!this.messageShown) {
158
+ // this.inputFocused = focused;
159
+ // if (!focused) {
160
+ // this.emitSelection('focus lost');
161
+ // }
162
+ // }
163
+ // }, 250);
164
+ // }
165
+ get hasCheckboxSelected() {
166
+ return this.checkboxSelection.size > 0;
167
+ }
168
+ optionChecked(u) {
169
+ return this.checkboxSelection.has(u[this.idAttribute]);
170
+ }
171
+ select(u) {
172
+ if (this.hasCheckboxSelected || u.unselectable) {
173
+ return;
174
+ }
175
+ if (!u.unselectable) {
176
+ this.selectedOption = u;
177
+ this.initialOption = u;
178
+ this.toggleDropdown();
179
+ }
180
+ }
181
+ toggleOptionCheckboxed(u) {
182
+ if (!u.unselectable) {
183
+ this.checkboxSelection.has(u[this.idAttribute]) ?
184
+ this.checkboxSelection.delete(u[this.idAttribute]) : this.checkboxSelection.add(u[this.idAttribute]);
185
+ }
186
+ this.input?.focus();
187
+ }
188
+ // get optionsVisible(): boolean {
189
+ // return this.inputFocused && this.typingStarted;
190
+ // }
191
+ keyPressed(keyEvent) {
192
+ this.lastEmittedId = -1;
193
+ this.optionsVisible = true;
194
+ // this.typingStarted = keyEvent.key !== 'Tab' && keyEvent.key !== 'Escape';
195
+ if (keyEvent.key === 'Escape') {
196
+ keyEvent.stopPropagation();
197
+ keyEvent.preventDefault();
198
+ this.optionsVisible = false;
199
+ return;
200
+ }
201
+ if (keyEvent.key === 'Enter' && this.hasCheckboxSelected) {
202
+ if (this.focusedOption && !this.focusedOption.unselectable) {
203
+ this.toggleOptionCheckboxed(this.focusedOption);
204
+ }
205
+ keyEvent.stopPropagation();
206
+ keyEvent.preventDefault();
207
+ return;
208
+ }
209
+ // if (keyEvent.key === 'Enter' && this.filteredGroupsAndUsers.length === 0 && this.focusedUser === null && !this.alreadyAdded) {
210
+ // this.newUser();
211
+ // keyEvent.stopPropagation();
212
+ // keyEvent.preventDefault();
213
+ // return;
214
+ // }
215
+ if (keyEvent.key === 'Enter') {
216
+ this.selectedOption = this.focusedOption;
217
+ keyEvent.stopPropagation();
218
+ keyEvent.preventDefault();
219
+ this.toggleDropdown();
220
+ return;
221
+ }
222
+ if ((keyEvent.key === 'ArrowDown' || keyEvent.key === 'ArrowUp') && !this.focusedOption) {
223
+ if (this.filteredOptions.length > 0) {
224
+ this.focusedOption = this.filteredOptions[0];
225
+ // if (!this.focusedUser.unselectable || this.focusedUser === this.initialUser) {
226
+ // this.selectedUser = this.focusedUser;
227
+ // }
228
+ this.selectedOption = this.focusedOption;
229
+ return;
230
+ }
231
+ }
232
+ if (keyEvent.key === 'ArrowDown' && this.focusedOption) {
233
+ const idx = this.filteredOptions.indexOf(this.focusedOption);
234
+ if (idx === this.filteredMaxItemsShown - 1) {
235
+ this.filteredMaxItemsShown += this.SHOWN_USERS_DELTA;
236
+ }
237
+ if (idx < this.filteredOptions.length - 1) {
238
+ this.focusedOption = this.filteredOptions[idx + 1];
239
+ this.selectedOption = this.focusedOption;
240
+ }
241
+ }
242
+ if (keyEvent.key === 'ArrowUp' && this.focusedOption) {
243
+ const idx = this.filteredOptions.indexOf(this.focusedOption);
244
+ if (idx > 0) {
245
+ this.focusedOption = this.filteredOptions[idx - 1];
246
+ this.selectedOption = this.focusedOption;
247
+ }
248
+ }
249
+ this.cdr.detectChanges();
250
+ }
251
+ toggleDropdown() {
252
+ this.optionsVisible = false;
253
+ this.emitSelection('toggle');
254
+ }
255
+ emitSelection(src) {
256
+ this.initialOption = this.selectedOption;
257
+ if (this.selectedOption) {
258
+ const checkId = this.selectedOption.id === null ? 0 : this.selectedOption.id;
259
+ if (this.lastEmittedId !== checkId) {
260
+ this.lastEmittedId = checkId;
261
+ this.onChange(this.options.find(o => o[this.idAttribute] === this.selectedOption[this.idAttribute]));
262
+ this.inputControl.setValue(this.selectedOption[this.displayAttribute]);
263
+ }
264
+ }
265
+ else {
266
+ // this.control.setValue(null);
267
+ this.onChange(null);
268
+ this.inputControl.setValue('', { emitEvent: false });
269
+ }
270
+ // if (this.selectedOption) {
271
+ // const checkId = this.selectedUser.id === null ? 0 : this.selectedUser.id;
272
+ // if (this.lastEmittedId !== checkId) {
273
+ // this.inputControl.setValue(this.selectedUser.fullName, {emitEvent: false});
274
+ // if (this.selectedUser.id) {
275
+ // this.groupsAndUsers.find(g => g.users.find(u => {
276
+ // if (u.id === this.selectedUser.id) {
277
+ // this.control.setValue(u);
278
+ // return true;
279
+ // } else {
280
+ // return false;
281
+ // }
282
+ // }));
283
+ // } else {
284
+ // this.control.setValue(this.selectedUser);
285
+ // }
286
+ // this.lastEmittedId = checkId;
287
+ // }
288
+ // } else {
289
+ // if (this.lastEmittedId !== null || this.lastEmittedId === -1/* || this.selectedUser?.unselectable*/) {
290
+ // this.control?.setValue(null);
291
+ // this.inputControl.setValue('', {emitEvent: false});
292
+ // this.lastEmittedId = null;
293
+ // }
294
+ // }
295
+ // this.typingStarted = false;
296
+ }
297
+ cancelCheckboxSelection() {
298
+ this.checkboxSelection.clear();
299
+ // this.skipBlur = true;
300
+ // this.input?.focus();
301
+ }
302
+ useCheckboxSelection() {
303
+ // this.newUsername = '';
304
+ // const users = [];
305
+ // this.groupsAndUsers.forEach(g => g.users.forEach(u => {
306
+ // if (!users.includes(u) && this.checkboxSelection.has(u.id)) {
307
+ // users.push(u);
308
+ // }
309
+ // }));
310
+ const outputOptions = this.options.filter(o => this.checkboxSelection.has(o[this.idAttribute]) && !this.optionDisabled(o));
311
+ if (outputOptions.length > 0) {
312
+ this.selectedOption = outputOptions[0];
313
+ this.inputControl.setValue(this.selectedOption[this.displayAttribute], { emitEvent: false });
314
+ this.onChange(outputOptions);
315
+ this.toggleDropdown();
316
+ // this.control?.setValue(outputOptions);
317
+ // this.typingStarted = false;
318
+ // this.skipBlur = true;
319
+ }
320
+ this.checkboxSelection.clear();
321
+ }
322
+ newOption() {
323
+ }
324
+ loadMore(visible) {
325
+ if (visible) {
326
+ this.filteredMaxItemsShown += this.SHOWN_USERS_DELTA;
327
+ }
328
+ }
329
+ focus() {
330
+ this.input?.focus();
331
+ }
332
+ registerOnChange(fn) {
333
+ this.onChange = fn;
334
+ }
335
+ registerOnTouched(fn) {
336
+ }
337
+ setDisabledState(isDisabled) {
338
+ isDisabled ? this.inputControl.disable() : this.inputControl.enable();
339
+ }
340
+ writeValue(obj) {
341
+ if (obj) {
342
+ if (obj.id) {
343
+ this.filteredOptions = this.searchableOptions.filter(so => so.id === obj.id);
344
+ }
345
+ else {
346
+ this.filteredOptions = this.filterOptions(obj[this.displayAttribute].toLowerCase());
347
+ }
348
+ if (this.filteredOptions.length === 1) {
349
+ this.selectedOption = this.filteredOptions[0];
350
+ this.focusedOption = this.filteredOptions[0];
351
+ }
352
+ this.inputControl.setValue(obj[this.displayAttribute]);
353
+ }
354
+ else {
355
+ this.filteredOptions = this.filterOptions('');
356
+ }
357
+ }
358
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.1", ngImport: i0, type: AutoCompleteComponent, deps: [{ token: i1.ControlContainer, host: true, optional: true, skipSelf: true }, { token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Component }); }
359
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.1", type: AutoCompleteComponent, selector: "ui-auto-complete", inputs: { templates: "templates", options: "options", multiple: "multiple", allowNew: "allowNew", displayAttribute: "displayAttribute", idAttribute: "idAttribute", panelClass: "panelClass", optionDisabled: "optionDisabled", optionFormatter: "optionFormatter" }, providers: [{
360
+ provide: NG_VALUE_ACCESSOR,
361
+ multi: true,
362
+ useExisting: AutoCompleteComponent
363
+ }], viewQueries: [{ propertyName: "input", first: true, predicate: ["input"], descendants: true }, { propertyName: "optionsTrigger", first: true, predicate: ["trigger"], descendants: true }], usesInheritance: true, usesOnChanges: true, ngImport: i0, template: "<div class=\"wrapper\" #wrapper>\r\n <div *ngIf=\"optionsVisible\" (click)=\"toggleDropdown()\" class=\"overlay\"></div>\r\n <ui-input [formControl]=\"inputControl\" #input #trigger=\"cdkOverlayOrigin\" cdkOverlayOrigin\r\n (focusChanged)=\"inputFocusChanged($event)\" (keyPressed)=\"keyPressed($event)\" [placeholder]=\"placeholder\"\r\n [useInputMessages]=\"'never'\" [label]=\"label\">\r\n <div slot=\"prefix\">\r\n <ng-content select=\"[slot=acprefix]\"></ng-content>\r\n </div>\r\n </ui-input>\r\n\r\n <ng-template cdkConnectedOverlay [cdkConnectedOverlayOrigin]=\"trigger\" [cdkConnectedOverlayOpen]=\"optionsVisible\">\r\n <div class=\"options-box\" [class]=\"panelClass\" style=\"display: flex; flex-direction: column\"\r\n [ngStyle]=\"{width: wrapper.offsetWidth + 'px'}\">\r\n <div class=\"options\" #options>\r\n\r\n <ng-container *ngFor=\"let option of filteredOptions; index as i\">\r\n <div *ngIf=\"i < filteredMaxItemsShown\" class=\"option-container\"\r\n [ngClass]=\"{selected: focusedOption === option}\">\r\n @if (multiple) {\r\n <mat-checkbox [checked]=\"optionChecked(option) || option.unselectable\"\r\n [disabled]=\"option.unselectable\" (change)=\"toggleOptionCheckboxed(option)\">\r\n <div class=\"option-container-inner\" (click)=\"select(option)\">\r\n <div class=\"option-option\" uiForceVisibility [visibilityWithin]=\"options\"\r\n [visibilityPadding]=\"4\" *ngIf=\"focusedOption === option\"\r\n [ngClass]=\"{unselectable: option.unselectable}\">\r\n {{ optionFormatter(option) }}\r\n </div>\r\n <div class=\"option-option\" *ngIf=\"focusedOption !== option\"\r\n [ngClass]=\"{unselectable: option.unselectable}\">\r\n {{ optionFormatter(option) }}\r\n </div>\r\n <mat-icon *ngIf=\"!hasCheckboxSelected && !option.unselectable\">north_west</mat-icon>\r\n </div>\r\n </mat-checkbox>\r\n } @else {\r\n <div class=\"option-container-inner\" (click)=\"select(option)\">\r\n <div class=\"option-option\" uiForceVisibility [visibilityWithin]=\"options\"\r\n [visibilityPadding]=\"4\" *ngIf=\"focusedOption === option\"\r\n [ngClass]=\"{unselectable: option.unselectable}\">\r\n @if (templates?.optionTemplate) {\r\n <ng-template [ngTemplateOutlet]=\"templates?.optionTemplate\"\r\n [ngTemplateOutletContext]=\"{option: option}\"></ng-template>\r\n } @else {\r\n {{ optionFormatter(option) }}\r\n }\r\n </div>\r\n <div class=\"option-option\" *ngIf=\"focusedOption !== option\"\r\n [ngClass]=\"{unselectable: option.unselectable}\">\r\n @if (templates?.optionTemplate) {\r\n <ng-template [ngTemplateOutlet]=\"templates?.optionTemplate\"\r\n [ngTemplateOutletContext]=\"{option: option}\"></ng-template>\r\n } @else {\r\n {{ optionFormatter(option) }}\r\n }\r\n </div>\r\n <mat-icon *ngIf=\"!hasCheckboxSelected && !option.unselectable\">north_west</mat-icon>\r\n </div>\r\n }\r\n </div>\r\n </ng-container>\r\n\r\n <div *ngIf=\"filteredOptions.length >= filteredMaxItemsShown\" class=\"option-container\" uiForceVisibility\r\n [visibilityWithin]=\"options\" [visibilityEmitChange]=\"true\" (visibilityChanged)=\"loadMore($event)\">\r\n &nbsp;\r\n </div>\r\n\r\n\r\n <div *ngIf=\"filteredOptions.length === 0 && !focusedOption\" class=\"new-user\">\r\n @if (templates?.notFound) {\r\n <ng-template [ngTemplateOutlet]=\"templates?.notFound\"\r\n [ngTemplateOutletContext]=\"{searchText: inputControl.value}\"></ng-template>\r\n } @else {\r\n <ng-container *ngIf=\"alreadyAdded\">\r\n <div style=\"text-align: center\">U\u017Eivatel {{ inputControl.value }} u\u017E je mezi dlu\u017En\u00EDky</div>\r\n </ng-container>\r\n <ng-container *ngIf=\"!alreadyAdded\">\r\n <div style=\"text-align: center; padding: 1em 0\">U\u017Eivatele\r\n <strong>{{ inputControl.value }}</strong> nezn\u00E1me\r\n </div>\r\n <ui-button *ngIf=\"allowNew\" style=\"margin-top: 6px;\" [size]=\"'small'\" [label]=\"'P\u0159idat jej?'\"\r\n (click)=\"newOption()\"></ui-button>\r\n </ng-container>\r\n }\r\n </div>\r\n <div *ngIf=\"filteredOptions.length === 0 && focusedOption\" class=\"new-user\">\r\n <div style=\"text-align: center\">Nezn\u00E1m\u00FD u\u017Eivatel {{ inputControl.value }}</div>\r\n </div>\r\n </div>\r\n\r\n <div *ngIf=\"hasCheckboxSelected && filteredOptions.length > 0\" class=\"checkbox-buttons\">\r\n <ui-button [size]=\"'small'\" [kind]=\"'basic'\" [label]=\"'Zru\u0161it v\u00FDb\u011Br'\"\r\n (click)=\"cancelCheckboxSelection()\"></ui-button>\r\n <ui-button [size]=\"'small'\" [label]=\"'Pou\u017E\u00EDt v\u00FDb\u011Br (' + checkboxSelection.size + ')'\"\r\n (click)=\"useCheckboxSelection()\"></ui-button>\r\n </div>\r\n </div>\r\n </ng-template>\r\n</div>\r\n\r\n", styles: [":host .wrapper{position:relative;outline:none;display:flex}:host .overlay{position:fixed;inset:0}.options-box{max-height:248px;min-width:310px;box-shadow:0 2px 4px -1px #07254833,0 4px 5px #07254824,0 1px 10px #0725481f}.options-box .checkbox-buttons{display:flex;padding:8px 4px 4px;justify-content:space-between}.options-box .options{padding:4px;overflow:auto;scroll-behavior:smooth;display:flex;flex-direction:column}.options-box .options .option-container{padding:0 0 0 12px;display:flex;align-items:center}.options-box .options .option-container:hover{cursor:pointer}.options-box .options .option-container mat-checkbox{width:100%}.options-box .options .option-container mat-checkbox ::ng-deep .mdc-form-field{width:100%}.options-box .options .option-container mat-checkbox ::ng-deep .mdc-form-field label{display:flex;width:100%}.options-box .options .option-container-inner{flex:1 1 100%;display:flex;align-items:center}.options-box .options .option-container-inner .option-option{flex:1 1 100%;padding:10px 0 10px 8px}.options-box .options .option-container-inner mat-icon{display:none;font-size:20px;height:20px;margin-right:8px}.options-box .options .option-container-inner:hover mat-icon{display:block}\n"], dependencies: [{ kind: "directive", type: i2.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i2.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i2.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "directive", type: i2.NgStyle, selector: "[ngStyle]", inputs: ["ngStyle"] }, { kind: "component", type: i3.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { kind: "component", type: i4.MatCheckbox, selector: "mat-checkbox", inputs: ["aria-label", "aria-labelledby", "aria-describedby", "id", "required", "labelPosition", "name", "value", "disableRipple", "tabIndex", "color", "checked", "disabled", "indeterminate"], outputs: ["change", "indeterminateChange"], exportAs: ["matCheckbox"] }, { kind: "directive", type: i1.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i1.FormControlDirective, selector: "[formControl]", inputs: ["formControl", "disabled", "ngModel"], outputs: ["ngModelChange"], exportAs: ["ngForm"] }, { kind: "directive", type: i5.CdkConnectedOverlay, selector: "[cdk-connected-overlay], [connected-overlay], [cdkConnectedOverlay]", inputs: ["cdkConnectedOverlayOrigin", "cdkConnectedOverlayPositions", "cdkConnectedOverlayPositionStrategy", "cdkConnectedOverlayOffsetX", "cdkConnectedOverlayOffsetY", "cdkConnectedOverlayWidth", "cdkConnectedOverlayHeight", "cdkConnectedOverlayMinWidth", "cdkConnectedOverlayMinHeight", "cdkConnectedOverlayBackdropClass", "cdkConnectedOverlayPanelClass", "cdkConnectedOverlayViewportMargin", "cdkConnectedOverlayScrollStrategy", "cdkConnectedOverlayOpen", "cdkConnectedOverlayDisableClose", "cdkConnectedOverlayTransformOriginOn", "cdkConnectedOverlayHasBackdrop", "cdkConnectedOverlayLockPosition", "cdkConnectedOverlayFlexibleDimensions", "cdkConnectedOverlayGrowAfterOpen", "cdkConnectedOverlayPush", "cdkConnectedOverlayDisposeOnNavigation"], outputs: ["backdropClick", "positionChange", "attach", "detach", "overlayKeydown", "overlayOutsideClick"], exportAs: ["cdkConnectedOverlay"] }, { kind: "directive", type: i5.CdkOverlayOrigin, selector: "[cdk-overlay-origin], [overlay-origin], [cdkOverlayOrigin]", exportAs: ["cdkOverlayOrigin"] }, { kind: "component", type: i6.ButtonComponent, selector: "ui-button", inputs: ["type", "matIconPrefix", "label", "disabled", "busy", "kind", "color", "size", "formInvalid"] }, { kind: "component", type: i7.InputComponent, selector: "ui-input", inputs: ["prefixIcon", "suffixIcon", "type", "clearButton", "activeIcons", "forceHasPrefix", "forceHasSuffix", "textAlign"], outputs: ["focusChanged", "keyPressed"] }, { kind: "directive", type: i8.ForceVisibilityDirective, selector: "[uiForceVisibility]", inputs: ["visibilityWithin", "visibilityPadding", "visibilityOnRequestOnly", "visibilityEmitChange", "visibilityFromTop", "visibilityCheckEnabled"], outputs: ["visibilityChanged"], exportAs: ["ForceVisibilityDirective"] }] }); }
364
+ }
365
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.1", ngImport: i0, type: AutoCompleteComponent, decorators: [{
366
+ type: Component,
367
+ args: [{ selector: 'ui-auto-complete', providers: [{
368
+ provide: NG_VALUE_ACCESSOR,
369
+ multi: true,
370
+ useExisting: AutoCompleteComponent
371
+ }], template: "<div class=\"wrapper\" #wrapper>\r\n <div *ngIf=\"optionsVisible\" (click)=\"toggleDropdown()\" class=\"overlay\"></div>\r\n <ui-input [formControl]=\"inputControl\" #input #trigger=\"cdkOverlayOrigin\" cdkOverlayOrigin\r\n (focusChanged)=\"inputFocusChanged($event)\" (keyPressed)=\"keyPressed($event)\" [placeholder]=\"placeholder\"\r\n [useInputMessages]=\"'never'\" [label]=\"label\">\r\n <div slot=\"prefix\">\r\n <ng-content select=\"[slot=acprefix]\"></ng-content>\r\n </div>\r\n </ui-input>\r\n\r\n <ng-template cdkConnectedOverlay [cdkConnectedOverlayOrigin]=\"trigger\" [cdkConnectedOverlayOpen]=\"optionsVisible\">\r\n <div class=\"options-box\" [class]=\"panelClass\" style=\"display: flex; flex-direction: column\"\r\n [ngStyle]=\"{width: wrapper.offsetWidth + 'px'}\">\r\n <div class=\"options\" #options>\r\n\r\n <ng-container *ngFor=\"let option of filteredOptions; index as i\">\r\n <div *ngIf=\"i < filteredMaxItemsShown\" class=\"option-container\"\r\n [ngClass]=\"{selected: focusedOption === option}\">\r\n @if (multiple) {\r\n <mat-checkbox [checked]=\"optionChecked(option) || option.unselectable\"\r\n [disabled]=\"option.unselectable\" (change)=\"toggleOptionCheckboxed(option)\">\r\n <div class=\"option-container-inner\" (click)=\"select(option)\">\r\n <div class=\"option-option\" uiForceVisibility [visibilityWithin]=\"options\"\r\n [visibilityPadding]=\"4\" *ngIf=\"focusedOption === option\"\r\n [ngClass]=\"{unselectable: option.unselectable}\">\r\n {{ optionFormatter(option) }}\r\n </div>\r\n <div class=\"option-option\" *ngIf=\"focusedOption !== option\"\r\n [ngClass]=\"{unselectable: option.unselectable}\">\r\n {{ optionFormatter(option) }}\r\n </div>\r\n <mat-icon *ngIf=\"!hasCheckboxSelected && !option.unselectable\">north_west</mat-icon>\r\n </div>\r\n </mat-checkbox>\r\n } @else {\r\n <div class=\"option-container-inner\" (click)=\"select(option)\">\r\n <div class=\"option-option\" uiForceVisibility [visibilityWithin]=\"options\"\r\n [visibilityPadding]=\"4\" *ngIf=\"focusedOption === option\"\r\n [ngClass]=\"{unselectable: option.unselectable}\">\r\n @if (templates?.optionTemplate) {\r\n <ng-template [ngTemplateOutlet]=\"templates?.optionTemplate\"\r\n [ngTemplateOutletContext]=\"{option: option}\"></ng-template>\r\n } @else {\r\n {{ optionFormatter(option) }}\r\n }\r\n </div>\r\n <div class=\"option-option\" *ngIf=\"focusedOption !== option\"\r\n [ngClass]=\"{unselectable: option.unselectable}\">\r\n @if (templates?.optionTemplate) {\r\n <ng-template [ngTemplateOutlet]=\"templates?.optionTemplate\"\r\n [ngTemplateOutletContext]=\"{option: option}\"></ng-template>\r\n } @else {\r\n {{ optionFormatter(option) }}\r\n }\r\n </div>\r\n <mat-icon *ngIf=\"!hasCheckboxSelected && !option.unselectable\">north_west</mat-icon>\r\n </div>\r\n }\r\n </div>\r\n </ng-container>\r\n\r\n <div *ngIf=\"filteredOptions.length >= filteredMaxItemsShown\" class=\"option-container\" uiForceVisibility\r\n [visibilityWithin]=\"options\" [visibilityEmitChange]=\"true\" (visibilityChanged)=\"loadMore($event)\">\r\n &nbsp;\r\n </div>\r\n\r\n\r\n <div *ngIf=\"filteredOptions.length === 0 && !focusedOption\" class=\"new-user\">\r\n @if (templates?.notFound) {\r\n <ng-template [ngTemplateOutlet]=\"templates?.notFound\"\r\n [ngTemplateOutletContext]=\"{searchText: inputControl.value}\"></ng-template>\r\n } @else {\r\n <ng-container *ngIf=\"alreadyAdded\">\r\n <div style=\"text-align: center\">U\u017Eivatel {{ inputControl.value }} u\u017E je mezi dlu\u017En\u00EDky</div>\r\n </ng-container>\r\n <ng-container *ngIf=\"!alreadyAdded\">\r\n <div style=\"text-align: center; padding: 1em 0\">U\u017Eivatele\r\n <strong>{{ inputControl.value }}</strong> nezn\u00E1me\r\n </div>\r\n <ui-button *ngIf=\"allowNew\" style=\"margin-top: 6px;\" [size]=\"'small'\" [label]=\"'P\u0159idat jej?'\"\r\n (click)=\"newOption()\"></ui-button>\r\n </ng-container>\r\n }\r\n </div>\r\n <div *ngIf=\"filteredOptions.length === 0 && focusedOption\" class=\"new-user\">\r\n <div style=\"text-align: center\">Nezn\u00E1m\u00FD u\u017Eivatel {{ inputControl.value }}</div>\r\n </div>\r\n </div>\r\n\r\n <div *ngIf=\"hasCheckboxSelected && filteredOptions.length > 0\" class=\"checkbox-buttons\">\r\n <ui-button [size]=\"'small'\" [kind]=\"'basic'\" [label]=\"'Zru\u0161it v\u00FDb\u011Br'\"\r\n (click)=\"cancelCheckboxSelection()\"></ui-button>\r\n <ui-button [size]=\"'small'\" [label]=\"'Pou\u017E\u00EDt v\u00FDb\u011Br (' + checkboxSelection.size + ')'\"\r\n (click)=\"useCheckboxSelection()\"></ui-button>\r\n </div>\r\n </div>\r\n </ng-template>\r\n</div>\r\n\r\n", styles: [":host .wrapper{position:relative;outline:none;display:flex}:host .overlay{position:fixed;inset:0}.options-box{max-height:248px;min-width:310px;box-shadow:0 2px 4px -1px #07254833,0 4px 5px #07254824,0 1px 10px #0725481f}.options-box .checkbox-buttons{display:flex;padding:8px 4px 4px;justify-content:space-between}.options-box .options{padding:4px;overflow:auto;scroll-behavior:smooth;display:flex;flex-direction:column}.options-box .options .option-container{padding:0 0 0 12px;display:flex;align-items:center}.options-box .options .option-container:hover{cursor:pointer}.options-box .options .option-container mat-checkbox{width:100%}.options-box .options .option-container mat-checkbox ::ng-deep .mdc-form-field{width:100%}.options-box .options .option-container mat-checkbox ::ng-deep .mdc-form-field label{display:flex;width:100%}.options-box .options .option-container-inner{flex:1 1 100%;display:flex;align-items:center}.options-box .options .option-container-inner .option-option{flex:1 1 100%;padding:10px 0 10px 8px}.options-box .options .option-container-inner mat-icon{display:none;font-size:20px;height:20px;margin-right:8px}.options-box .options .option-container-inner:hover mat-icon{display:block}\n"] }]
372
+ }], ctorParameters: () => [{ type: i1.ControlContainer, decorators: [{
373
+ type: Optional
374
+ }, {
375
+ type: Host
376
+ }, {
377
+ type: SkipSelf
378
+ }] }, { type: i0.ChangeDetectorRef }], propDecorators: { templates: [{
379
+ type: Input
380
+ }], options: [{
381
+ type: Input
382
+ }], multiple: [{
383
+ type: Input
384
+ }], allowNew: [{
385
+ type: Input
386
+ }], displayAttribute: [{
387
+ type: Input
388
+ }], idAttribute: [{
389
+ type: Input
390
+ }], panelClass: [{
391
+ type: Input
392
+ }], input: [{
393
+ type: ViewChild,
394
+ args: ['input']
395
+ }], optionsTrigger: [{
396
+ type: ViewChild,
397
+ args: ['trigger']
398
+ }], optionDisabled: [{
399
+ type: Input
400
+ }], optionFormatter: [{
401
+ type: Input
402
+ }] } });
403
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYXV0by1jb21wbGV0ZS5jb21wb25lbnQuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy91aS9zcmMvbGliL2F1dG8tY29tcGxldGUvYXV0by1jb21wbGV0ZS5jb21wb25lbnQudHMiLCIuLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy91aS9zcmMvbGliL2F1dG8tY29tcGxldGUvYXV0by1jb21wbGV0ZS5jb21wb25lbnQuaHRtbCJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBRUwsU0FBUyxFQUNULElBQUksRUFDSixLQUFLLEVBR0wsUUFBUSxFQUNSLFFBQVEsRUFDUixTQUFTLEVBQ1YsTUFBTSxlQUFlLENBQUM7QUFDdkIsT0FBTyxFQUEwQyxpQkFBaUIsRUFBRSxrQkFBa0IsRUFBRSxNQUFNLGdCQUFnQixDQUFDO0FBQy9HLE9BQU8sRUFBRSxZQUFZLEVBQUUsTUFBTSxnQkFBZ0IsQ0FBQztBQUM5QyxPQUFPLEVBQUUsZUFBZSxFQUFFLE1BQU0sbUNBQW1DLENBQUM7Ozs7Ozs7Ozs7QUFjcEUsTUFBTSxPQUFPLHFCQUFzQixTQUFRLGVBQWU7SUFJeEQsWUFBc0QsZ0JBQWtDO0lBQzVFLG1EQUFtRDtJQUMzQyxHQUFzQjtRQUN4QyxLQUFLLEVBQUUsQ0FBQztRQUg0QyxxQkFBZ0IsR0FBaEIsZ0JBQWdCLENBQWtCO1FBRXBFLFFBQUcsR0FBSCxHQUFHLENBQW1CO1FBSmxDLHNCQUFpQixHQUFHLEVBQUUsQ0FBQztRQVF0QixjQUFTLEdBQThGLEVBQUUsQ0FBQTtRQUd6RyxhQUFRLEdBQUcsSUFBSSxDQUFDO1FBQ2hCLGFBQVEsR0FBRyxLQUFLLENBQUM7UUFDakIscUJBQWdCLEdBQUcsTUFBTSxDQUFDO1FBQzFCLGdCQUFXLEdBQUcsSUFBSSxDQUFDO1FBQ25CLGVBQVUsR0FBRyxFQUFFLENBQUM7UUFJekIsb0JBQWUsR0FBVSxFQUFFLENBQUM7UUFDNUIsMEJBQXFCLEdBQUcsSUFBSSxDQUFD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LEtBQUs7Z0JBRUcsT0FBTztzQkFBZixLQUFLO2dCQUNHLFFBQVE7c0JBQWhCLEtBQUs7Z0JBQ0csUUFBUTtzQkFBaEIsS0FBSztnQkFDRyxnQkFBZ0I7c0JBQXhCLEtBQUs7Z0JBQ0csV0FBVztzQkFBbkIsS0FBSztnQkFDRyxVQUFVO3NCQUFsQixLQUFLO2dCQWlCYyxLQUFLO3NCQUF4QixTQUFTO3VCQUFDLE9BQU87Z0JBQ0ksY0FBYztzQkFBbkMsU0FBUzt1QkFBQyxTQUFTO2dCQVFYLGNBQWM7c0JBQXRCLEtBQUs7Z0JBQ0csZUFBZTtzQkFBdkIsS0FBSyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7XHJcbiAgQ2hhbmdlRGV0ZWN0b3JSZWYsXHJcbiAgQ29tcG9uZW50LFxyXG4gIEhvc3QsXHJcbiAgSW5wdXQsXHJcbiAgT25DaGFuZ2VzLFxyXG4gIE9uSW5pdCxcclxuICBPcHRpb25hbCwgU2ltcGxlQ2hhbmdlcyxcclxuICBTa2lwU2VsZixcclxuICBWaWV3Q2hpbGRcclxufSBmcm9tICdAYW5ndWxhci9jb3JlJztcclxuaW1wb3J0IHsgQ29udHJvbENvbnRhaW5lciwgQ29udHJvbFZhbHVlQWNjZXNzb3IsIE5HX1ZBTFVFX0FDQ0VTU09SLCBVbnR5cGVkRm9ybUNvbnRyb2wgfSBmcm9tICdAYW5ndWxhci9mb3Jtcyc7XHJcbmltcG9ydCB7IGRlYm91bmNlVGltZSB9IGZyb20gJ3J4anMvb3BlcmF0b3JzJztcclxuaW1wb3J0IHsgVWlCYXNlQ29tcG9uZW50IH0gZnJvbSAnLi4vbGF5b3V0cy9iYXNlL3VpLWJhc2UuY29tcG9uZW50JztcclxuaW1wb3J0IHsgSW5wdXRDb21wb25lbnQgfSBmcm9tICcuLi9pbnB1dC9pbnB1dC5jb21wb25lbnQnO1xyXG5pbXBvcnQgeyBDZGtPdmVybGF5T3JpZ2luIH0gZnJvbSAnQGFuZ3VsYXIvY2RrL292ZXJsYXknO1xyXG5cclxuQENvbXBvbmVudCh7XHJcbiAgc2VsZWN0b3I6ICd1aS1hdXRvLWNvbXBsZXRlJyxcclxuICB0ZW1wbGF0ZVVybDogJy4vYXV0by1jb21wbGV0ZS5jb21wb25lbnQuaHRtbCcsXHJcbiAgc3R5bGVVcmxzOiBbJy4vYXV0by1jb21wbGV0ZS5jb21wb25lbnQuc2NzcyddLFxyXG4gIHByb3ZpZGVyczogW3tcclxuICAgIHByb3ZpZGU6IE5HX1ZBTFVFX0FDQ0VTU09SLFxyXG4gICAgbXVsdGk6IHRydWUsXHJcbiAgICB1c2VFeGlzdGluZzogQXV0b0NvbXBsZXRlQ29tcG9uZW50XHJcbiAgfV1cclxufSlcclxuZXhwb3J0IGNsYXNzIEF1dG9Db21wbGV0ZUNvbXBvbmVudCBleHRlbmRzIFVpQmFzZUNvbXBvbmVudCBpbXBsZW1lbnRzIE9uSW5pdCwgQ29udHJvbFZhbHVlQWNjZXNzb3IsIE9uQ2hhbmdlcyB7XHJcblxyXG4gIHByaXZhdGUgU0hPV05fVVNFUlNfREVMVEEgPSAxMDtcclxuXHJcbiAgY29uc3RydWN0b3IoQE9wdGlvbmFsKCkgQEhvc3QoKSBAU2tpcFNlbGYoKSBwcm90ZWN0ZWQgY29udHJvbENvbnRhaW5lcjogQ29udHJvbENvbnRhaW5lcixcclxuICAgICAgICAgICAgICAvLyBAT3B0aW9uYWwoKSBAU2VsZigpIHB1YmxpYyBuZ0NvbnRyb2w6IE5nQ29udHJvbCxcclxuICAgICAgICAgICAgICBwcml2YXRlIGNkcjogQ2hhbmdlRGV0ZWN0b3JSZWYpIHtcclxuICAgIHN1cGVyKCk7XHJcbiAgfVxyXG5cclxuICBASW5wdXQoKSB0ZW1wbGF0ZXM6IHsgb3B0aW9uVGVtcGxhdGU/OiBhbnksIG5vdEZvdW5kPzogYW55LCBmaWxlVGVtcGxhdGU/OiBhbnksIGN1c3RvbUNvbnRlbnRUZW1wbGF0ZT86IGFueSB9ID0ge31cclxuXHJcbiAgQElucHV0KCkgb3B0aW9ucz86IGFueVtdO1xyXG4gIEBJbnB1dCgpIG11bHRpcGxlID0gdHJ1ZTtcclxuICBASW5wdXQoKSBhbGxvd05ldyA9IGZhbHNlO1xyXG4gIEBJbnB1dCgpIGRpc3BsYXlBdHRyaWJ1dGUgPSAnbmFtZSc7XHJcbiAgQElucHV0KCkgaWRBdHRyaWJ1dGUgPSAnaWQnO1xyXG4gIEBJbnB1dCgpIHBhbmVsQ2xhc3MgPSAnJztcclxuXHJcbiAgLy8gdmFsdWU/OiBhbnk7XHJcbiAgc2VhcmNoYWJsZU9wdGlvbnM/OiBhbnlbXTtcclxuICBmaWx0ZXJlZE9wdGlvbnM6IGFueVtdID0gW107XHJcbiAgZmlsdGVyZWRNYXhJdGVtc1Nob3duID0gdGhpcy5TSE9XTl9VU0VSU19ERUxUQTtcclxuXHJcbiAgLy8gc2tpcEJsdXIgPSBmYWxzZTtcclxuICAvLyB0eXBpbmdTdGFydGVkID0gZmFsc2U7XHJcbiAgbWVzc2FnZVNob3duID0gZmFsc2U7XHJcbiAgaW5pdGlhbE9wdGlvbj86IGFueTtcclxuICBzZWxlY3RlZE9wdGlvbj86IGFueTtcclxuICBsYXN0RW1pdHRlZElkPzogYW55O1xyXG4gIGZvY3VzZWRPcHRpb24/OiBhbnk7XHJcbiAgYWxyZWFkeUFkZGVkID0gZmFsc2U7XHJcbiAgb3B0aW9uc1Zpc2libGUgPSBmYWxzZTtcclxuXHJcbiAgQFZpZXdDaGlsZCgnaW5wdXQnKSBpbnB1dD86IElucHV0Q29tcG9uZW50O1xyXG4gIEBWaWV3Q2hpbGQoJ3RyaWdnZXInKSBvcHRpb25zVHJpZ2dlcjogQ2RrT3ZlcmxheU9yaWdpbjtcclxuXHJcbiAgaW5wdXRGb2N1c2VkID0gZmFsc2U7XHJcbiAgaW5wdXRDb250cm9sID0gbmV3IFVudHlwZWRGb3JtQ29udHJvbCgnJyk7XHJcbiAgY2hlY2tib3hTZWxlY3Rpb246IFNldDxhbnk+ID0gbmV3IFNldDxhbnk+KCk7XHJcblxyXG4gIG9uQ2hhbmdlOiAodmFsdWUpID0+IHZvaWQ7XHJcblxyXG4gIEBJbnB1dCgpIG9wdGlvbkRpc2FibGVkOiAoYW55KSA9PiBib29sZWFuID0gKCkgPT4gZmFsc2U7XHJcbiAgQElucHV0KCkgb3B0aW9uRm9ybWF0dGVyOiAoYW55KSA9PiBzdHJpbmcgPSAobykgPT4gb1t0aGlzLmRpc3BsYXlBdHRyaWJ1dGVdO1xyXG5cclxuICBwcm90ZWN0ZWQgYnVpbGRTZWFyY2hhYmxlT3B0aW9ucygpIHtcclxuICAgIHJldHVybiB0aGlzLm9wdGlvbnMubWFwKG8gPT4gKHtcclxuICAgICAgLi4ubyxcclxuICAgICAgdW5zZWxlY3RhYmxlOiB0aGlzLm9wdGlvbkRpc2FibGVkKG8pLFxyXG4gICAgICBzZWFyY2hTdHJpbmc6IGAke3RoaXMub3B0aW9uRm9ybWF0dGVyKG8pfWAudG9Mb3dlckNhc2UoKVxyXG4gICAgfSkpO1xyXG4gIH1cclxuXHJcbiAgcHJvdGVjdGVkIGZpbHRlck9wdGlvbnModGV4dDogc3RyaW5nKSB7XHJcbiAgICB0aGlzLmZpbHRlcmVkTWF4SXRlbXNTaG93biA9IHRoaXMuU0hPV05fVVNFUlNfREVMVEE7XHJcbiAgICByZXR1cm4gdGhpcy5zZWFyY2hhYmxlT3B0aW9ucy5maWx0ZXIodSA9PiB1LnNlYXJjaFN0cmluZy5pbmNsdWRlcyh0ZXh0KSk7XHJcbiAgfVxyXG5cclxuICBuZ09uQ2hhbmdlcyhjaGFuZ2VzOiBTaW1wbGVDaGFuZ2VzKTogdm9pZCB7XHJcbiAgICBpZiAoY2hhbmdlc1snb3B0aW9ucyddKSB7XHJcbiAgICAgIHRoaXMuc2VhcmNoYWJsZU9wdGlvbnMgPSB0aGlzLmJ1aWxkU2VhcmNoYWJsZU9wdGlvbnMoKTtcclxuICAgICAgdGhpcy5pbnB1dENvbnRyb2wuc2V0VmFsdWUoJycpO1xyXG4gICAgfVxyXG4gIH1cclxuXHJcblxyXG4gIG5nT25Jbml0KCk6IHZvaWQge1xyXG5cclxuICAgIGNvbnNvbGUubG9nKCduZyBvbiBpbml0Jyk7XHJcblxyXG4gICAgc3VwZXIubmdPbkluaXQoKTtcclxuXHJcbiAgICAvLyBpZiAodGhpcy5mb3JtQ29udHJvbE5hbWUgJiYgdGhpcy5jb250cm9sQ29udGFpbmVyLmNvbnRyb2wpIHtcclxuICAgIC8vICAgdGhpcy5jb250cm9sID0gdGhpcy5jb250cm9sQ29udGFpbmVyLmNvbnRyb2wuZ2V0KHRoaXMuZm9ybUNvbnRyb2xOYW1lKSBhcyBGb3JtQ29udHJvbDtcclxuICAgIC8vIH1cclxuICAgIC8vXHJcbiAgICAvLyB0aGlzLnVzZXIgPSB0aGlzLmNvbnRyb2wudmFsdWU7XHJcblxyXG4gICAgLy8gdGhpcy52YWx1ZSA9IHRoaXMuY29udHJvbC52YWx1ZTtcclxuXHJcbiAgICAvLyB0aGlzLnNlYXJjaGFibGVPcHRpb25zID0gdGhpcy5idWlsZFNlYXJjaGFibGVPcHRpb25zKCk7XHJcblxyXG4gICAgLy8gaWYgKHRoaXMuY29udHJvbC5kaXNhYmxlZCkge1xyXG4gICAgLy8gICB0aGlzLmlucHV0Q29udHJvbC5kaXNhYmxlKCk7XHJcbiAgICAvLyB9XHJcblxyXG4gICAgLy8gdGhpcy5jb250cm9sLnN0YXR1c0NoYW5nZXMuc3Vic2NyaWJlKHYgPT4ge1xyXG4gICAgLy8gICBpZiAodiA9PT0gJ0RJU0FCTEVEJykge1xyXG4gICAgLy8gICAgIHRoaXMuaW5wdXRDb250cm9sLmRpc2FibGUoKTtcclxuICAgIC8vICAgfVxyXG4gICAgLy8gICBpZiAodiA9PT0gJ1ZBTElEJykge1xyXG4gICAgLy8gICAgIHRoaXMuaW5wdXRDb250cm9sLmVuYWJsZSgpO1xyXG4gICAgLy8gICB9XHJcbiAgICAvLyB9KTtcclxuXHJcblxyXG5cclxuICAgIHRoaXMuaW5wdXRDb250cm9sLnZhbHVlQ2hhbmdlcy5waXBlKGRlYm91bmNlVGltZSgyMDApKS5zdWJzY3JpYmUodiA9PiB7XHJcbiAgICAgIGNvbnN0IGxjdiA9IHYudG9Mb3dlckNhc2UoKTtcclxuXHJcbiAgICAgIHRoaXMuZmlsdGVyZWRPcHRpb25zID0gdGhpcy5maWx0ZXJPcHRpb25zKGxjdik7XHJcblxyXG4gICAgICBpZiAodGhpcy5maWx0ZXJlZE9wdGlvbnMubGVuZ3RoID4gMCkge1xyXG4gICAgICAgIHRoaXMuZm9jdXNlZE9wdGlvbiA9IHRoaXMuZmlsdGVyZWRPcHRpb25zWzBdO1xyXG4gICAgICB9IGVsc2Uge1xyXG4gICAgICAgIHRoaXMuZm9jdXNlZE9wdGlvbiA9IHVuZGVmaW5lZDtcclxuICAgICAgfVxyXG5cclxuICAgICAgaWYgKHYgPT09ICcnKSB7XHJcbiAgICAgICAgdGhpcy5mb2N1c2VkT3B0aW9uID0gdW5kZWZpbmVkO1xyXG4gICAgICAgIHRoaXMuc2VsZWN0ZWRPcHRpb24gPSB1bmRlZmluZWQ7XHJcbiAgICAgIH1cclxuXHJcbiAgICAgIC8vIGlmICh2ICE9PSAnJykge1xyXG4gICAgICAvLyAgIGlmICh0aGlzLm5ld1VzZXJuYW1lID09PSB2KSB7XHJcbiAgICAgIC8vICAgICB0aGlzLnNlbGVjdGVkVXNlciA9IHtpZDogbnVsbCwgZnVsbE5hbWU6IHZ9O1xyXG4gICAgICAvLyAgICAgdGhpcy5pbml0aWFsVXNlciA9IHRoaXMuc2VsZWN0ZWRVc2VyO1xyXG4gICAgICAvLyAgIH0gZWxzZSB7XHJcbiAgICAgIC8vICAgICBpZiAodGhpcy5maWx0ZXJlZEdyb3Vwc0FuZFVzZXJzLmxlbmd0aCA+IDApIHtcclxuICAgICAgLy8gICAgICAgdGhpcy5mb2N1c2VkR3JvdXAgPSB0aGlzLmZpbHRlcmVkR3JvdXBzQW5kVXNlcnNbMF07XHJcbiAgICAgIC8vICAgICAgIHRoaXMuZm9jdXNlZFVzZXIgPSB0aGlzLmZvY3VzZWRHcm91cC5maWx0ZXJlZFVzZXJzWzBdO1xyXG4gICAgICAvLyAgICAgfSBlbHNlIHtcclxuICAgICAgLy8gICAgICAgdGhpcy5mb2N1c2VkR3JvdXAgPSBudWxsO1xyXG4gICAgICAvLyAgICAgICB0aGlzLmZvY3VzZWRVc2VyID0gbnVsbDtcclxuICAgICAgLy8gICAgIH1cclxuICAgICAgLy8gICAgIGlmICgodGhpcy5mb2N1c2VkVXNlciAmJiAhdGhpcy5mb2N1c2VkVXNlci51bnNlbGVjdGFibGUpIHx8IHRoaXMuZm9jdXNlZFVzZXIgPT09IHRoaXMuaW5pdGlhbFVzZXIpIHtcclxuICAgICAgLy8gICAgICAgdGhpcy5zZWxlY3RlZFVzZXIgPSB0aGlzLmZvY3VzZWRVc2VyO1xyXG4gICAgICAvLyAgICAgfSBlbHNlIHtcclxuICAgICAgLy8gICAgICAgdGhpcy5zZWxlY3RlZFVzZXIgPSBudWxsO1xyXG4gICAgICAvLyAgICAgfVxyXG4gICAgICAvLyAgIH1cclxuICAgICAgLy8gfSBlbHNlIHtcclxuICAgICAgLy8gICB0aGlzLnNlbGVjdGVkVXNlciA9IG51bGw7XHJcbiAgICAgIC8vICAgdGhpcy5mb2N1c2VkVXNlciA9IG51bGw7XHJcbiAgICAgIC8vICAgdGhpcy5mb2N1c2VkR3JvdXAgPSBudWxsO1xyXG4gICAgICAvLyB9XHJcblxyXG4gICAgICAvLyB0aGlzLm9uQ2hhbmdlKHYpO1xyXG4gICAgfSk7XHJcblxyXG4gICAgLy8gICBpZiAodGhpcy51c2VyICE9PSBudWxsKSB7XHJcbiAgICAvLyAgICAgbGV0IHNlbGVjdGVkVXNlciA9IG51bGw7XHJcbiAgICAvLyAgICAgdGhpcy5ncm91cHNBbmRVc2Vycy5maW5kKGcgPT4gZy51c2Vycy5maW5kKHUgPT4ge1xyXG4gICAgLy8gICAgICAgaWYgKHUuaWQgPT09IHRoaXMudXNlci5pZCkge1xyXG4gICAgLy8gICAgICAgICBzZWxlY3RlZFVzZXIgPSB1O1xyXG4gICAgLy8gICAgICAgICByZXR1cm4gdHJ1ZTtcclxuICAgIC8vICAgICAgIH0gZWxzZSB7XHJcbiAgICAvLyAgICAgICAgIHJldHVybiBmYWxzZTtcclxuICAgIC8vICAgICAgIH1cclxuICAgIC8vICAgICB9KSk7XHJcbiAgICAvL1xyXG4gICAgLy8gICAgIGlmIChzZWxlY3RlZFVzZXIpIHtcclxuICAgIC8vICAgICAgIHRoaXMuaW5wdXRDb250cm9sLnNldFZhbHVlKHNlbGVjdGVkVXNlci5mdWxsTmFtZSk7XHJcbiAgICAvLyAgICAgICB0aGlzLnNlbGVjdGVkVXNlciA9IHNlbGVjdGVkVXNlcjtcclxuICAgIC8vICAgICB9IGVsc2Uge1xyXG4gICAgLy8gICAgICAgdGhpcy5uZXdVc2VybmFtZSA9IHRoaXMudXNlci5mdWxsTmFtZTtcclxuICAgIC8vICAgICAgIHRoaXMuaW5wdXRDb250cm9sLnNldFZhbHVlKHRoaXMudXNlci5mdWxsTmFtZSk7XHJcbiAgICAvLyAgICAgICB0aGlzLnNlbGVjdGVkVXNlciA9IHsgaWQ6IG51bGwsIGZ1bGxOYW1lOiB0aGlzLm5ld1VzZXJuYW1lIH07XHJcbiAgICAvLyAgICAgfVxyXG4gICAgLy8gICB9IGVsc2Uge1xyXG4gICAgLy8gICAgIHRoaXMuc2VsZWN0ZWRVc2VyID0gbnVsbDtcclxuICAgIC8vICAgICB0aGlzLmlucHV0Q29udHJvbC5zZXRWYWx1ZSgnJyk7XHJcbiAgICAvLyAgIH1cclxuICAgIC8vICAgdGhpcy5pbml0aWFsVXNlciA9IHRoaXMuc2VsZWN0ZWRVc2VyO1xyXG4gICAgLy8gfVxyXG4gIH1cclxuXHJcbiAgaW5wdXRGb2N1c0NoYW5nZWQoZm9jdXNlZDogYm9vbGVhbik6IHZvaWQge1xyXG4gICAgaWYgKGZvY3VzZWQpIHtcclxuICAgICAgdGhpcy5vcHRpb25zVmlzaWJsZSA9IHRydWU7XHJcbiAgICB9XHJcbiAgfVxyXG4gIC8vICAgc2V0VGltZW91dCgoKSA9PiB7XHJcbiAgLy8gICAgIGlmICghZm9jdXNlZCAmJiB0aGlzLnNraXBCbHVyKSB7XHJcbiAgLy8gICAgICAgdGhpcy5za2lwQmx1ciA9IGZhbHNlO1xyXG4gIC8vICAgICAgIHJldHVybjtcclxuICAvLyAgICAgfVxyXG4gIC8vICAgICBpZiAoIXRoaXMudHlwaW5nU3RhcnRlZCAmJiBmb2N1c2VkKSB7XHJcbiAgLy8gICAgICAgdGhpcy5pbnB1dENvbnRyb2wuc2V0VmFsdWUodGhpcy5pbnB1dENvbnRyb2wudmFsdWUpO1xyXG4gIC8vICAgICAgIHRoaXMudHlwaW5nU3RhcnRlZCA9IHRydWU7XHJcbiAgLy8gICAgIH1cclxuICAvLyAgICAgaWYgKCF0aGlzLm1lc3NhZ2VTaG93bikge1xyXG4gIC8vICAgICAgIHRoaXMuaW5wdXRGb2N1c2VkID0gZm9jdXNlZDtcclxuICAvLyAgICAgICBpZiAoIWZvY3VzZWQpIHtcclxuICAvLyAgICAgICAgIHRoaXMuZW1pdFNlbGVjdGlvbignZm9jdXMgbG9zdCcpO1xyXG4gIC8vICAgICAgIH1cclxuICAvLyAgICAgfVxyXG4gIC8vICAgfSwgMjUwKTtcclxuICAvLyB9XHJcblxyXG4gIGdldCBoYXNDaGVja2JveFNlbGVjdGVkKCk6IGJvb2xlYW4ge1xyXG4gICAgcmV0dXJuIHRoaXMuY2hlY2tib3hTZWxlY3Rpb24uc2l6ZSA+IDA7XHJcbiAgfVxyXG5cclxuICBvcHRpb25DaGVja2VkKHU6IGFueSk6IGJvb2xlYW4ge1xyXG4gICAgcmV0dXJuIHRoaXMuY2hlY2tib3hTZWxlY3Rpb24uaGFzKHVbdGhpcy5pZEF0dHJpYnV0ZV0pO1xyXG4gIH1cclxuXHJcbiAgc2VsZWN0KHU6IGFueSk6IHZvaWQge1xyXG4gICAgaWYgKHRoaXMuaGFzQ2hlY2tib3hTZWxlY3RlZCB8fCB1LnVuc2VsZWN0YWJsZSkge1xyXG4gICAgICByZXR1cm47XHJcbiAgICB9XHJcbiAgICBpZiAoIXUudW5zZWxlY3RhYmxlKSB7XHJcbiAgICAgIHRoaXMuc2VsZWN0ZWRPcHRpb24gPSB1O1xyXG4gICAgICB0aGlzLmluaXRpYWxPcHRpb24gPSB1O1xyXG4gICAgICB0aGlzLnRvZ2dsZURyb3Bkb3duKCk7XHJcbiAgICB9XHJcbiAgfVxyXG5cclxuICB0b2dnbGVPcHRpb25DaGVja2JveGVkKHU6IGFueSk6IHZvaWQge1xyXG4gICAgaWYgKCF1LnVuc2VsZWN0YWJsZSkge1xyXG4gICAgICB0aGlzLmNoZWNrYm94U2VsZWN0aW9uLmhhcyh1W3RoaXMuaWRBdHRyaWJ1dGVdKSA/XHJcbiAgICAgICAgdGhpcy5jaGVja2JveFNlbGVjdGlvbi5kZWxldGUodVt0aGlzLmlkQXR0cmlidXRlXSkgOiB0aGlzLmNoZWNrYm94U2VsZWN0aW9uLmFkZCh1W3RoaXMuaWRBdHRyaWJ1dGVdKTtcclxuICAgIH1cclxuICAgIHRoaXMuaW5wdXQ/LmZvY3VzKCk7XHJcbiAgfVxyXG5cclxuXHJcbiAgLy8gZ2V0IG9wdGlvbnNWaXNpYmxlKCk6IGJvb2xlYW4ge1xyXG4gIC8vICAgcmV0dXJuIHRoaXMuaW5wdXRGb2N1c2VkICYmIHRoaXMudHlwaW5nU3RhcnRlZDtcclxuICAvLyB9XHJcblxyXG4gIGtleVByZXNzZWQoa2V5RXZlbnQ6IEtleWJvYXJkRXZlbnQpOiB2b2lkIHtcclxuICAgIHRoaXMubGFzdEVtaXR0ZWRJZCA9IC0xO1xyXG4gICAgdGhpcy5vcHRpb25zVmlzaWJsZSA9IHRydWU7XHJcbiAgICAvLyB0aGlzLnR5cGluZ1N0YXJ0ZWQgPSBrZXlFdmVudC5rZXkgIT09ICdUYWInICYmIGtleUV2ZW50LmtleSAhPT0gJ0VzY2FwZSc7XHJcbiAgICBpZiAoa2V5RXZlbnQua2V5ID09PSAnRXNjYXBlJykge1xyXG4gICAgICBrZXlFdmVudC5zdG9wUHJvcGFnYXRpb24oKTtcclxuICAgICAga2V5RXZlbnQucHJldmVudERlZmF1bHQoKTtcclxuICAgICAgdGhpcy5vcHRpb25zVmlzaWJsZSA9IGZhbHNlO1xyXG4gICAgICByZXR1cm47XHJcbiAgICB9XHJcbiAgICBpZiAoa2V5RXZlbnQua2V5ID09PSAnRW50ZXInICYmIHRoaXMuaGFzQ2hlY2tib3hTZWxlY3RlZCkge1xyXG4gICAgICBpZiAodGhpcy5mb2N1c2VkT3B0aW9uICYmICF0aGlzLmZvY3VzZWRPcHRpb24udW5zZWxlY3RhYmxlKSB7XHJcbiAgICAgICAgdGhpcy50b2dnbGVPcHRpb25DaGVja2JveGVkKHRoaXMuZm9jdXNlZE9wdGlvbik7XHJcbiAgICAgIH1cclxuICAgICAga2V5RXZlbnQuc3RvcFByb3BhZ2F0aW9uKCk7XHJcbiAgICAgIGtleUV2ZW50LnByZXZlbnREZWZhdWx0KCk7XHJcbiAgICAgIHJldHVybjtcclxuICAgIH1cclxuICAgIC8vIGlmIChrZXlFdmVudC5rZXkgPT09ICdFbnRlcicgJiYgdGhpcy5maWx0ZXJlZEdyb3Vwc0FuZFVzZXJzLmxlbmd0aCA9PT0gMCAmJiB0aGlzLmZvY3VzZWRVc2VyID09PSBudWxsICYmICF0aGlzLmFscmVhZHlBZGRlZCkge1xyXG4gICAgLy8gICB0aGlzLm5ld1VzZXIoKTtcclxuICAgIC8vICAga2V5RXZlbnQuc3RvcFByb3BhZ2F0aW9uKCk7XHJcbiAgICAvLyAgIGtleUV2ZW50LnByZXZlbnREZWZhdWx0KCk7XHJcbiAgICAvLyAgIHJldHVybjtcclxuICAgIC8vIH1cclxuICAgIGlmIChrZXlFdmVudC5rZXkgPT09ICdFbnRlcicpIHtcclxuICAgICAgdGhpcy5zZWxlY3RlZE9wdGlvbiA9IHRoaXMuZm9jdXNlZE9wdGlvbjtcclxuICAgICAga2V5RXZlbnQuc3RvcFByb3BhZ2F0aW9uKCk7XHJcbiAgICAgIGtleUV2ZW50LnByZXZlbnREZWZhdWx0KCk7XHJcbiAgICAgIHRoaXMudG9nZ2xlRHJvcGRvd24oKTtcclxuICAgICAgcmV0dXJuO1xyXG4gICAgfVxyXG4gICAgaWYgKChrZXlFdmVudC5rZXkgPT09ICdBcnJvd0Rvd24nIHx8IGtleUV2ZW50LmtleSA9PT0gJ0Fycm93VXAnKSAmJiAhdGhpcy5mb2N1c2VkT3B0aW9uKSB7XHJcbiAgICAgIGlmICh0aGlzLmZpbHRlcmVkT3B0aW9ucy5sZW5ndGggPiAwKSB7XHJcbiAgICAgICAgdGhpcy5mb2N1c2VkT3B0aW9uID0gdGhpcy5maWx0ZXJlZE9wdGlvbnNbMF07XHJcbiAgICAgICAgLy8gaWYgKCF0aGlzLmZvY3VzZWRVc2VyLnVuc2VsZWN0YWJsZSB8fCB0aGlzLmZvY3VzZWRVc2VyID09PSB0aGlzLmluaXRpYWxVc2VyKSB7XHJcbiAgICAgICAgLy8gICB0aGlzLnNlbGVjdGVkVXNlciA9IHRoaXMuZm9jdXNlZFVzZXI7XHJcbiAgICAgICAgLy8gfVxyXG4gICAgICAgIHRoaXMuc2VsZWN0ZWRPcHRpb24gPSB0aGlzLmZvY3VzZWRPcHRpb247XHJcbiAgICAgICAgcmV0dXJuO1xyXG4gICAgICB9XHJcbiAgICB9XHJcbiAgICBpZiAoa2V5RXZlbnQua2V5ID09PSAnQXJyb3dEb3duJyAmJiB0aGlzLmZvY3VzZWRPcHRpb24pIHtcclxuICAgICAgY29uc3QgaWR4ID0gdGhpcy5maWx0ZXJlZE9wdGlvbnMuaW5kZXhPZih0aGlzLmZvY3VzZWRPcHRpb24pO1xyXG4gICAgICBpZiAoaWR4ID09PSB0aGlzLmZpbHRlcmVkTWF4SXRlbXNTaG93biAtIDEpIHtcclxuICAgICAgICB0aGlzLmZpbHRlcmVkTWF4SXRlbXNTaG93biArPSB0aGlzLlNIT1dOX1VTRVJTX0RFTFRBO1xyXG4gICAgICB9XHJcbiAgICAgIGlmIChpZHggPCB0aGlzLmZpbHRlcmVkT3B0aW9ucy5sZW5ndGggLSAxKSB7XHJcbiAgICAgICAgdGhpcy5mb2N1c2VkT3B0aW9uID0gdGhpcy5maWx0ZXJlZE9wdGlvbnNbaWR4ICsgMV07XHJcbiAgICAgICAgdGhpcy5zZWxlY3RlZE9wdGlvbiA9IHRoaXMuZm9jdXNlZE9wdGlvbjtcclxuICAgICAgfVxyXG4gICAgfVxyXG4gICAgaWYgKGtleUV2ZW50LmtleSA9PT0gJ0Fycm93VXAnICYmIHRoaXMuZm9jdXNlZE9wdGlvbikge1xyXG4gICAgICBjb25zdCBpZHggPSB0aGlzLmZpbHRlcmVkT3B0aW9ucy5pbmRleE9mKHRoaXMuZm9jdXNlZE9wdGlvbik7XHJcbiAgICAgIGlmIChpZHggPiAwKSB7XHJcbiAgICAgICAgdGhpcy5mb2N1c2VkT3B0aW9uID0gdGhpcy5maWx0ZXJlZE9wdGlvbnNbaWR4IC0gMV07XHJcbiAgICAgICAgdGhpcy5zZWxlY3RlZE9wdGlvbiA9IHRoaXMuZm9jdXNlZE9wdGlvbjtcclxuICAgICAgfVxyXG4gICAgfVxyXG5cclxuICAgIHRoaXMuY2RyLmRldGVjdENoYW5nZXMoKTtcclxuICB9XHJcblxyXG4gIHRvZ2dsZURyb3Bkb3duKCkge1xyXG4gICAgdGhpcy5vcHRpb25zVmlzaWJsZSA9IGZhbHNlO1xyXG4gICAgdGhpcy5lbWl0U2VsZWN0aW9uKCd0b2dnbGUnKTtcclxuICB9XHJcblxyXG4gIGVtaXRTZWxlY3Rpb24oc3JjPzogc3RyaW5nKTogdm9pZCB7XHJcbiAgICB0aGlzLmluaXRpYWxPcHRpb24gPSB0aGlzLnNlbGVjdGVkT3B0aW9uO1xyXG5cclxuICAgIGlmICh0aGlzLnNlbGVjdGVkT3B0aW9uKSB7XHJcbiAgICAgIGNvbnN0IGNoZWNrSWQgPSB0aGlzLnNlbGVjdGVkT3B0aW9uLmlkID09PSBudWxsID8gMCA6IHRoaXMuc2VsZWN0ZWRPcHRpb24uaWQ7XHJcbiAgICAgIGlmICh0aGlzLmxhc3RFbWl0dGVkSWQgIT09IGNoZWNrSWQpIHtcclxuICAgICAgICB0aGlzLmxhc3RFbWl0dGVkSWQgPSBjaGVja0lkO1xyXG4gICAgICAgIHRoaXMub25DaGFuZ2UodGhpcy5vcHRpb25zLmZpbmQobyA9PiBvW3RoaXMuaWRBdHRyaWJ1dGVdID09PSB0aGlzLnNlbGVjdGVkT3B0aW9uW3RoaXMuaWRBdHRyaWJ1dGVdKSk7XHJcbiAgICAgICAgdGhpcy5pbnB1dENvbnRyb2wuc2V0VmFsdWUodGhpcy5zZWxlY3RlZE9wdGlvblt0aGlzLmRpc3BsYXlBdHRyaWJ1dGVdKTtcclxuICAgICAgfVxyXG4gICAgfSBlbHNlIHtcclxuICAgICAgLy8gdGhpcy5jb250cm9sLnNldFZhbHVlKG51bGwpO1xyXG4gICAgICB0aGlzLm9uQ2hhbmdlKG51bGwpO1xyXG4gICAgICB0aGlzLmlucHV0Q29udHJvbC5zZXRWYWx1ZSgnJywge2VtaXRFdmVudDogZmFsc2V9KTtcclxuICAgIH1cclxuXHJcblxyXG5cclxuICAgIC8vIGlmICh0aGlzLnNlbGVjdGVkT3B0aW9uKSB7XHJcbiAgICAvLyAgIGNvbnN0IGNoZWNrSWQgPSB0aGlzLnNlbGVjdGVkVXNlci5pZCA9PT0gbnVsbCA/IDAgOiB0aGlzLnNlbGVjdGVkVXNlci5pZDtcclxuICAgIC8vICAgaWYgKHRoaXMubGFzdEVtaXR0ZWRJZCAhPT0gY2hlY2tJZCkge1xyXG4gICAgLy8gICAgIHRoaXMuaW5wdXRDb250cm9sLnNldFZhbHVlKHRoaXMuc2VsZWN0ZWRVc2VyLmZ1bGxOYW1lLCB7ZW1pdEV2ZW50OiBmYWxzZX0pO1xyXG4gICAgLy8gICAgIGlmICh0aGlzLnNlbGVjdGVkVXNlci5pZCkge1xyXG4gICAgLy8gICAgICAgdGhpcy5ncm91cHNBbmRVc2Vycy5maW5kKGcgPT4gZy51c2Vycy5maW5kKHUgPT4ge1xyXG4gICAgLy8gICAgICAgICBpZiAodS5pZCA9PT0gdGhpcy5zZWxlY3RlZFVzZXIuaWQpIHtcclxuICAgIC8vICAgICAgICAgICB0aGlzLmNvbnRyb2wuc2V0VmFsdWUodSk7XHJcbiAgICAvLyAgICAgICAgICAgcmV0dXJuIHRydWU7XHJcbiAgICAvLyAgICAgICAgIH0gZWxzZSB7XHJcbiAgICAvLyAgICAgICAgICAgcmV0dXJuIGZhbHNlO1xyXG4gICAgLy8gICAgICAgICB9XHJcbiAgICAvLyAgICAgICB9KSk7XHJcbiAgICAvLyAgICAgfSBlbHNlIHtcclxuICAgIC8vICAgICAgIHRoaXMuY29udHJvbC5zZXRWYWx1ZSh0aGlzLnNlbGVjdGVkVXNlcik7XHJcbiAgICAvLyAgICAgfVxyXG4gICAgLy8gICAgIHRoaXMubGFzdEVtaXR0ZWRJZCA9IGNoZWNrSWQ7XHJcbiAgICAvLyAgIH1cclxuICAgIC8vIH0gZWxzZSB7XHJcbiAgICAvLyAgIGlmICh0aGlzLmxhc3RFbWl0dGVkSWQgIT09IG51bGwgfHwgdGhpcy5sYXN0RW1pdHRlZElkID09PSAtMS8qIHx8IHRoaXMuc2VsZWN0ZWRVc2VyPy51bnNlbGVjdGFibGUqLykge1xyXG4gICAgLy8gICAgIHRoaXMuY29udHJvbD8uc2V0VmFsdWUobnVsbCk7XHJcbiAgICAvLyAgICAgdGhpcy5pbnB1dENvbnRyb2wuc2V0VmFsdWUoJycsIHtlbWl0RXZlbnQ6IGZhbHNlfSk7XHJcbiAgICAvLyAgICAgdGhpcy5sYXN0RW1pdHRlZElkID0gbnVsbDtcclxuICAgIC8vICAgfVxyXG4gICAgLy8gfVxyXG4gICAgLy8gdGhpcy50eXBpbmdTdGFydGVkID0gZmFsc2U7XHJcbiAgfVxyXG5cclxuICBjYW5jZWxDaGVja2JveFNlbGVjdGlvbigpOiB2b2lkIHtcclxuICAgIHRoaXMuY2hlY2tib3hTZWxlY3Rpb24uY2xlYXIoKTtcclxuICAgIC8vIHRoaXMuc2tpcEJsdXIgPSB0cnVlO1xyXG4gICAgLy8gdGhpcy5pbnB1dD8uZm9jdXMoKTtcclxuICB9XHJcblxyXG4gIHVzZUNoZWNrYm94U2VsZWN0aW9uKCk6IHZvaWQge1xyXG4gICAgLy8gdGhpcy5uZXdVc2VybmFtZSA9ICcnO1xyXG4gICAgLy8gY29uc3QgdXNlcnMgPSBbXTtcclxuICAgIC8vIHRoaXMuZ3JvdXBzQW5kVXNlcnMuZm9yRWFjaChnID0+IGcudXNlcnMuZm9yRWFjaCh1ID0+IHtcclxuICAgIC8vICAgaWYgKCF1c2Vycy5pbmNsdWRlcyh1KSAmJiB0aGlzLmNoZWNrYm94U2VsZWN0aW9uLmhhcyh1LmlkKSkge1xyXG4gICAgLy8gICAgIHVzZXJzLnB1c2godSk7XHJcbiAgICAvLyAgIH1cclxuICAgIC8vIH0pKTtcclxuICAgIGNvbnN0IG91dHB1dE9wdGlvbnMgPSB0aGlzLm9wdGlvbnMuZmlsdGVyKG8gPT4gdGhpcy5jaGVja2JveFNlbGVjdGlvbi5oYXMob1t0aGlzLmlkQXR0cmlidXRlXSkgJiYgIXRoaXMub3B0aW9uRGlzYWJsZWQobykpO1xyXG4gICAgaWYgKG91dHB1dE9wdGlvbnMubGVuZ3RoID4gMCkge1xyXG4gICAgICB0aGlzLnNlbGVjdGVkT3B0aW9uID0gb3V0cHV0T3B0aW9uc1swXTtcclxuICAgICAgdGhpcy5pbnB1dENvbnRyb2wuc2V0VmFsdWUodGhpcy5zZWxlY3RlZE9wdGlvblt0aGlzLmRpc3BsYXlBdHRyaWJ1dGVdLCB7ZW1pdEV2ZW50OiBmYWxzZX0pO1xyXG4gICAgICB0aGlzLm9uQ2hhbmdlKG91dHB1dE9wdGlvbnMpO1xyXG4gICAgICB0aGlzLnRvZ2dsZURyb3Bkb3duKCk7XHJcbiAgICAgIC8vIHRoaXMuY29udHJvbD8uc2V0VmFsdWUob3V0cHV0T3B0aW9ucyk7XHJcbiAgICAgIC8vIHRoaXMudHlwaW5nU3RhcnRlZCA9IGZhbHNlO1xyXG4gICAgICAvLyB0aGlzLnNraXBCbHVyID0gdHJ1ZTtcclxuICAgIH1cclxuICAgIHRoaXMuY2hlY2tib3hTZWxlY3Rpb24uY2xlYXIoKTtcclxuICB9XHJcblxyXG4gIG5ld09wdGlvbigpIHtcclxuXHJcbiAgfVxyXG5cclxuICBsb2FkTW9yZSh2aXNpYmxlOiBib29sZWFuKSB7XHJcbiAgICBpZiAodmlzaWJsZSkge1xyXG4gICAgICB0aGlzLmZpbHRlcmVkTWF4SXRlbXNTaG93biArPSB0aGlzLlNIT1dOX1VTRVJTX0RFTFRBO1xyXG4gICAgfVxyXG4gIH1cclxuXHJcbiAgcHVibGljIGZvY3VzKCkge1xyXG4gICAgdGhpcy5pbnB1dD8uZm9jdXMoKTtcclxuICB9XHJcblxyXG4gIHJlZ2lzdGVyT25DaGFuZ2UoZm46IGFueSk6IHZvaWQge1xyXG4gICAgdGhpcy5vbkNoYW5nZSA9IGZuO1xyXG4gIH1cclxuXHJcbiAgcmVnaXN0ZXJPblRvdWNoZWQoZm46IGFueSk6IHZvaWQge1xyXG4gIH1cclxuXHJcbiAgc2V0RGlzYWJsZWRTdGF0ZShpc0Rpc2FibGVkOiBib29sZWFuKTogdm9pZCB7XHJcbiAgICBpc0Rpc2FibGVkID8gdGhpcy5pbnB1dENvbnRyb2wuZGlzYWJsZSgpIDogdGhpcy5pbnB1dENvbnRyb2wuZW5hYmxlKCk7XHJcbiAgfVxyXG5cclxuICB3cml0ZVZhbHVlKG9iajogYW55KTogdm9pZCB7XHJcbiAgICBpZiAob2JqKSB7XHJcbiAgICAgIGlmIChvYmouaWQpIHtcclxuICAgICAgICB0aGlzLmZpbHRlcmVkT3B0aW9ucyA9IHRoaXMuc2VhcmNoYWJsZU9wdGlvbnMuZmlsdGVyKHNvID0+IHNvLmlkID09PSBvYmouaWQpO1xyXG4gICAgICB9IGVsc2Uge1xyXG4gICAgICAgIHRoaXMuZmlsdGVyZWRPcHRpb25zID0gdGhpcy5maWx0ZXJPcHRpb25zKG9ialt0aGlzLmRpc3BsYXlBdHRyaWJ1dGVdLnRvTG93ZXJDYXNlKCkpO1xyXG4gICAgICB9XHJcbiAgICAgIGlmICh0aGlzLmZpbHRlcmVkT3B0aW9ucy5sZW5ndGggPT09IDEpIHtcclxuICAgICAgICB0aGlzLnNlbGVjdGVkT3B0aW9uID0gdGhpcy5maWx0ZXJlZE9wdGlvbnNbMF07XHJcbiAgICAgICAgdGhpcy5mb2N1c2VkT3B0aW9uID0gdGhpcy5maWx0ZXJlZE9wdGlvbnNbMF07XHJcbiAgICAgIH1cclxuICAgICAgdGhpcy5pbnB1dENvbnRyb2wuc2V0VmFsdWUob2JqW3RoaXMuZGlzcGxheUF0dHJpYnV0ZV0pO1xyXG4gICAgfSBlbHNlIHtcclxuICAgICAgdGhpcy5maWx0ZXJlZE9wdGlvbnMgPSB0aGlzLmZpbHRlck9wdGlvbnMoJycpO1xyXG4gICAgfVxyXG4gIH1cclxufVxyXG4iLCI8ZGl2IGNsYXNzPVwid3JhcHBlclwiICN3cmFwcGVyPlxyXG4gICAgPGRpdiAqbmdJZj1cIm9wdGlvbnNWaXNpYmxlXCIgKGNsaWNrKT1cInRvZ2dsZURyb3Bkb3duKClcIiBjbGFzcz1cIm92ZXJsYXlcIj48L2Rpdj5cclxuICAgIDx1aS1pbnB1dCBbZm9ybUNvbnRyb2xdPVwiaW5wdXRDb250cm9sXCIgI2lucHV0ICN0cmlnZ2VyPVwiY2RrT3ZlcmxheU9yaWdpblwiIGNka092ZXJsYXlPcmlnaW5cclxuICAgICAgICAgICAgICAoZm9jdXNDaGFuZ2VkKT1cImlucHV0Rm9jdXNDaGFuZ2VkKCRldmVudClcIiAoa2V5UHJlc3NlZCk9XCJrZXlQcmVzc2VkKCRldmVudClcIiBbcGxhY2Vob2xkZXJdPVwicGxhY2Vob2xkZXJcIlxyXG4gICAgICAgICAgICAgIFt1c2VJbnB1dE1lc3NhZ2VzXT1cIiduZXZlcidcIiBbbGFiZWxdPVwibGFiZWxcIj5cclxuICAgICAgICA8ZGl2IHNsb3Q9XCJwcmVmaXhcIj5cclxuICAgICAgICAgICAgPG5nLWNvbnRlbnQgc2VsZWN0PVwiW3Nsb3Q9YWNwcmVmaXhdXCI+PC9uZy1jb250ZW50PlxyXG4gICAgICAgIDwvZGl2PlxyXG4gICAgPC91aS1pbnB1dD5cclxuXHJcbiAgICA8bmctdGVtcGxhdGUgY2RrQ29ubmVjdGVkT3ZlcmxheSBbY2RrQ29ubmVjdGVkT3ZlcmxheU9yaWdpbl09XCJ0cmlnZ2VyXCIgW2Nka0Nvbm5lY3RlZE92ZXJsYXlPcGVuXT1cIm9wdGlvbnNWaXNpYmxlXCI+XHJcbiAgICAgICAgPGRpdiBjbGFzcz1cIm9wdGlvbnMtYm94XCIgW2NsYXNzXT1cInBhbmVsQ2xhc3NcIiBzdHlsZT1cImRpc3BsYXk6IGZsZXg7IGZsZXgtZGlyZWN0aW9uOiBjb2x1bW5cIlxyXG4gICAgICAgICAgICAgW25nU3R5bGVdPVwie3dpZHRoOiB3cmFwcGVyLm9mZnNldFdpZHRoICsgJ3B4J31cIj5cclxuICAgICAgICAgICAgPGRpdiBjbGFzcz1cIm9wdGlvbnNcIiAjb3B0aW9ucz5cclxuXHJcbiAgICAgICAgICAgICAgICA8bmctY29udGFpbmVyICpuZ0Zvcj1cImxldCBvcHRpb24gb2YgZmlsdGVyZWRPcHRpb25zOyBpbmRleCBhcyBpXCI+XHJcbiAgICAgICAgICAgICAgICAgICAgPGRpdiAqbmdJZj1cImkgPCBmaWx0ZXJlZE1heEl0ZW1zU2hvd25cIiBjbGFzcz1cIm9wdGlvbi1jb250YWluZXJcIlxyXG4gICAgICAgICAgICAgICAgICAgICAgICAgW25nQ2xhc3NdPVwie3NlbGVjdGVkOiBmb2N1c2VkT3B0aW9uID09PSBvcHRpb259XCI+XHJcbiAgICAgICAgICAgICAgICAgICAgICAgIEBpZiAobXVsdGlwbGUpIHtcclxuICAgICAgICAgICAgICAgICAgICAgICAgICAgIDxtYXQtY2hlY2tib3ggW2NoZWNrZWRdPVwib3B0aW9uQ2hlY2tlZChvcHRpb24pIHx8IG9wdGlvbi51bnNlbGVjdGFibGVcIlxyXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBbZGlzYWJsZWRdPVwib3B0aW9uLnVuc2VsZWN0YWJsZVwiIChjaGFuZ2UpPVwidG9nZ2xlT3B0aW9uQ2hlY2tib3hlZChvcHRpb24pXCI+XHJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPGRpdiBjbGFzcz1cIm9wdGlvbi1jb250YWluZXItaW5uZXJcIiAoY2xpY2spPVwic2VsZWN0KG9wdGlvbilcIj5cclxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPGRpdiBjbGFzcz1cIm9wdGlvbi1vcHRpb25cIiB1aUZvcmNlVmlzaWJpbGl0eSBbdmlzaWJpbGl0eVdpdGhpbl09XCJvcHRpb25zXCJcclxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBbdmlzaWJpbGl0eVBhZGRpbmddPVwiNFwiICpuZ0lmPVwiZm9jdXNlZE9wdGlvbiA9PT0gb3B0aW9uXCJcclxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBbbmdDbGFzc109XCJ7dW5zZWxlY3RhYmxlOiBvcHRpb24udW5zZWxlY3RhYmxlfVwiPlxyXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAge3sgb3B0aW9uRm9ybWF0dGVyKG9wdGlvbikgfX1cclxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPC9kaXY+XHJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIDxkaXYgY2xhc3M9XCJvcHRpb24tb3B0aW9uXCIgKm5nSWY9XCJmb2N1c2VkT3B0aW9uICE9PSBvcHRpb25cIlxyXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIFtuZ0NsYXNzXT1cInt1bnNlbGVjdGFibGU6IG9wdGlvbi51bnNlbGVjdGFibGV9XCI+XHJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB7eyBvcHRpb25Gb3JtYXR0ZXIob3B0aW9uKSB9fVxyXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICA8L2Rpdj5cclxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPG1hdC1pY29uICpuZ0lmPVwiIWhhc0NoZWNrYm94U2VsZWN0ZWQgJiYgIW9wdGlvbi51bnNlbGVjdGFibGVcIj5ub3J0aF93ZXN0PC9tYXQtaWNvbj5cclxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICA8L2Rpdj5cclxuICAgICAgICAgICAgICAgICAgICAgICAgICAgIDwvbWF0LWNoZWNrYm94PlxyXG4gICAgICAgICAgICAgICAgICAgICAgICB9IEBlbHNlIHtcclxuICAgICAgICAgICAgICAgICAgICAgICAgICAgIDxkaXYgY2xhc3M9XCJvcHRpb24tY29udGFpbmVyLWlubmVyXCIgKGNsaWNrKT1cInNlbGVjdChvcHRpb24pXCI+XHJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPGRpdiBjbGFzcz1cIm9wdGlvbi1vcHRpb25cIiB1aUZvcmNlVmlzaWJpbGl0eSBbdmlzaWJpbGl0eVdpdGhpbl09XCJvcHRpb25zXCJcclxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIFt2aXNpYmlsaXR5UGFkZGluZ109XCI0XCIgKm5nSWY9XCJmb2N1c2VkT3B0aW9uID09PSBvcHRpb25cIlxyXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgW25nQ2xhc3NdPVwie3Vuc2VsZWN0YWJsZTogb3B0aW9uLnVuc2VsZWN0YWJsZX1cIj5cclxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgQGlmICh0ZW1wbGF0ZXM/Lm9wdGlvblRlbXBsYXRlKSB7XHJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICA8bmctdGVtcGxhdGUgW25nVGVtcGxhdGVPdXRsZXRdPVwidGVtcGxhdGVzPy5vcHRpb25UZW1wbGF0ZVwiXHJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgW25nVGVtcGxhdGVPdXRsZXRDb250ZXh0XT1cIntvcHRpb246IG9wdGlvbn1cIj48L25nLXRlbXBsYXRlPlxyXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB9IEBlbHNlIHtcclxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHt7IG9wdGlvbkZvcm1hdHRlcihvcHRpb24pIH19XHJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIH1cclxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICA8L2Rpdj5cclxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICA8ZGl2IGNsYXNzPVwib3B0aW9uLW9wdGlvblwiICpuZ0lmPVwiZm9jdXNlZE9wdGlvbiAhPT0gb3B0aW9uXCJcclxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIFtuZ0NsYXNzXT1cInt1bnNlbGVjdGFibGU6IG9wdGlvbi51bnNlbGVjdGFibGV9XCI+XHJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIEBpZiAodGVtcGxhdGVzPy5vcHRpb25UZW1wbGF0ZSkge1xyXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPG5nLXRlbXBsYXRlIFtuZ1RlbXBsYXRlT3V0bGV0XT1cInRlbXBsYXRlcz8ub3B0aW9uVGVtcGxhdGVcIlxyXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIFtuZ1RlbXBsYXRlT3V0bGV0Q29udGV4dF09XCJ7b3B0aW9uOiBvcHRpb259XCI+PC9uZy10ZW1wbGF0ZT5cclxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgfSBAZWxzZSB7XHJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB7eyBvcHRpb25Gb3JtYXR0ZXIob3B0aW9uKSB9fVxyXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB9XHJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPC9kaXY+XHJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPG1hdC1pY29uICpuZ0lmPVwiIWhhc0NoZWNrYm94U2VsZWN0ZWQgJiYgIW9wdGlvbi51bnNlbGVjdGFibGVcIj5ub3J0aF93ZXN0PC9tYXQtaWNvbj5cclxuICAgICAgICAgICAgICAgICAgICAgICAgICAgIDwvZGl2PlxyXG4gICAgICAgICAgICAgICAgICAgICAgICB9XHJcbiAgICAgICAgICAgICAgICAgICAgPC9kaXY+XHJcbiAgICAgICAgICAgICAgICA8L25nLWNvbnRhaW5lcj5cclxuXHJcbiAgICAgICAgICAgICAgICA8ZGl2ICpuZ0lmPVwiZmlsdGVyZWRPcHRpb25zLmxlbmd0aCA+PSBmaWx0ZXJlZE1heEl0ZW1zU2hvd25cIiBjbGFzcz1cIm9wdGlvbi1jb250YWluZXJcIiB1aUZvcmNlVmlzaWJpbGl0eVxyXG4gICAgICAgICAgICAgICAgICAgICBbdmlzaWJpbGl0eVdpdGhpbl09XCJvcHRpb25zXCIgW3Zpc2liaWxpdHlFbWl0Q2hhbmdlXT1cInRydWVcIiAodmlzaWJpbGl0eUNoYW5nZWQpPVwibG9hZE1vcmUoJGV2ZW50KVwiPlxyXG4gICAgICAgICAgICAgICAgICAgICZuYnNwO1xyXG4gICAgICAgICAgICAgICAgPC9kaXY+XHJcblxyXG5cclxuICAgICAgICAgICAgICAgIDxkaXYgKm5nSWY9XCJmaWx0ZXJlZE9wdGlvbnMubGVuZ3RoID09PSAwICYmICFmb2N1c2VkT3B0aW9uXCIgY2xhc3M9XCJuZXctdXNlclwiPlxyXG4gICAgICAgICAgICAgICAgICAgIEBpZiAodGVtcGxhdGVzPy5ub3RGb3VuZCkge1xyXG4gICAgICAgICAgICAgICAgICAgICAgICA8bmctdGVtcGxhdGUgW25nVGVtcGxhdGVPdXRsZXRdPVwidGVtcGxhdGVzPy5ub3RGb3VuZFwiXHJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBbbmdUZW1wbGF0ZU91dGxldENvbnRleHRdPVwie3NlYXJjaFRleHQ6IGlucHV0Q29udHJvbC52YWx1ZX1cIj48L25nLXRlbXBsYXRlPlxyXG4gICAgICAgICAgICAgICAgICAgIH0gQGVsc2Uge1xyXG4gICAgICAgICAgICAgICAgICAgICAgICA8bmctY29udGFpbmVyICpuZ0lmPVwiYWxyZWFkeUFkZGVkXCI+XHJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICA8ZGl2IHN0eWxlPVwidGV4dC1hbGlnbjogY2VudGVyXCI+VcW+aXZhdGVsIHt7IGlucHV0Q29udHJvbC52YWx1ZSB9fSB1xb4gamUgbWV6aSBkbHXFvm7DrWt5PC9kaXY+XHJcbiAgICAgICAgICAgICAgICAgICAgICAgIDwvbmctY29udGFpbmVyPlxyXG4gICAgICAgICAgICAgICAgICAgICAgICA8bmctY29udGFpbmVyICpuZ0lmPVwiIWFscmVhZHlBZGRlZFwiPlxyXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgPGRpdiBzdHlsZT1cInRleHQtYWxpZ246IGNlbnRlcjsgcGFkZGluZzogMWVtIDBcIj5Vxb5pdmF0ZWxlXHJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPHN0cm9uZz57eyBpbnB1dENvbnRyb2wudmFsdWUgfX08L3N0cm9uZz4gbmV6bsOhbWVcclxuICAgICAgICAgICAgICAgICAgICAgICAgICAgIDwvZGl2PlxyXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgPHVpLWJ1dHRvbiAqbmdJZj1cImFsbG93TmV3XCIgc3R5bGU9XCJtYXJnaW4tdG9wOiA2cHg7XCIgW3NpemVdPVwiJ3NtYWxsJ1wiIFtsYWJlbF09XCInUMWZaWRhdCBqZWo/J1wiXHJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIChjbGljayk9XCJuZXdPcHRpb24oKVwiPjwvdWktYnV0dG9uPlxyXG4gICAgICAgICAgICAgICAgICAgICAgICA8L25nLWNvbnRhaW5lcj5cclxuICAgICAgICAgICAgICAgICAgICB9XHJcbiAgICAgICAgICAgICAgICA8L2Rpdj5cclxuICAgICAgICAgICAgICAgIDxkaXYgKm5nSWY9XCJmaWx0ZXJlZE9wdGlvbnMubGVuZ3RoID09PSAwICYmIGZvY3VzZWRPcHRpb25cIiBjbGFzcz1cIm5ldy11c2VyXCI+XHJcbiAgICAgICAgICAgICAgICAgICAgPGRpdiBzdHlsZT1cInRleHQtYWxpZ246IGNlbnRlclwiPk5lem7DoW3DvSB1xb5pdmF0ZWwge3sgaW5wdXRDb250cm9sLnZhbHVlIH19PC9kaXY+XHJcbiAgICAgICAgICAgICAgICA8L2Rpdj5cclxuICAgICAgICAgICAgPC9kaXY+XHJcblxyXG4gICAgICAgICAgICA8ZGl2ICpuZ0lmPVwiaGFzQ2hlY2tib3hTZWxlY3RlZCAmJiBmaWx0ZXJlZE9wdGlvbnMubGVuZ3RoID4gMFwiIGNsYXNzPVwiY2hlY2tib3gtYnV0dG9uc1wiPlxyXG4gICAgICAgICAgICAgICAgPHVpLWJ1dHRvbiBbc2l6ZV09XCInc21hbGwnXCIgW2tpbmRdPVwiJ2Jhc2ljJ1wiIFtsYWJlbF09XCInWnJ1xaFpdCB2w71ixJtyJ1wiXHJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgIChjbGljayk9XCJjYW5jZWxDaGVja2JveFNlbGVjdGlvbigpXCI+PC91aS1idXR0b24+XHJcbiAgICAgICAgICAgICAgICA8dWktYnV0dG9uIFtzaXplXT1cIidzbWFsbCdcIiBbbGFiZWxdPVwiJ1BvdcW+w610IHbDvWLEm3IgKCcgKyBjaGVja2JveFNlbGVjdGlvbi5zaXplICsgJyknXCJcclxuICAgICAgICAgICAgICAgICAgICAgICAgICAgKGNsaWNrKT1cInVzZUNoZWNrYm94U2VsZWN0aW9uKClcIj48L3VpLWJ1dHRvbj5cclxuICAgICAgICAgICAgPC9kaXY+XHJcbiAgICAgICAgPC9kaXY+XHJcbiAgICA8L25nLXRlbXBsYXRlPlxyXG48L2Rpdj5cclxuXHJcbiJdfQ==