@sbb-esta/lyne-elements-dev 5.0.0-next.2-dev.1778120187 → 5.0.0-next.2-dev.1778141158

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 (55) hide show
  1. package/autocomplete/autocomplete-base-element.js +1 -1
  2. package/autocomplete/autocomplete.component.js +1 -1
  3. package/{autocomplete-base-element-BuCAxBy3.js → autocomplete-base-element-CAWFmfLW.js} +20 -23
  4. package/autocomplete.js +1 -1
  5. package/autocomplete.pure.js +1 -1
  6. package/core/overlay/overlay-option-panel.js +6 -5
  7. package/core/styles/_index.scss +0 -1
  8. package/core/styles/core.scss +4 -2
  9. package/core/styles/option-panel-common.global.scss +11 -0
  10. package/core/styles/option-panel-common.scss +157 -0
  11. package/core/styles/styles.js +2 -2
  12. package/core.css +7 -5
  13. package/core.js +6 -7
  14. package/custom-elements.json +985 -1035
  15. package/development/autocomplete/autocomplete-base-element.d.ts +0 -1
  16. package/development/autocomplete/autocomplete-base-element.d.ts.map +1 -1
  17. package/development/autocomplete/autocomplete-base-element.js +1 -1
  18. package/development/autocomplete/autocomplete.component.js +1 -1
  19. package/development/autocomplete-base-element-DZx53PO1.js +683 -0
  20. package/development/autocomplete.js +1 -1
  21. package/development/autocomplete.pure.js +1 -1
  22. package/development/core/overlay/overlay-option-panel.d.ts +1 -1
  23. package/development/core/overlay/overlay-option-panel.d.ts.map +1 -1
  24. package/development/core/overlay/overlay-option-panel.js +5 -2
  25. package/development/core/styles/styles.d.ts +1 -0
  26. package/development/core/styles/styles.d.ts.map +1 -1
  27. package/development/core/styles/styles.js +2 -2
  28. package/development/core.d.ts +0 -1
  29. package/development/core.d.ts.map +1 -1
  30. package/development/core.js +2 -3
  31. package/development/select/select.component.d.ts +1 -2
  32. package/development/select/select.component.d.ts.map +1 -1
  33. package/development/select/select.component.js +1 -1
  34. package/development/select.component-VODtrUee.js +807 -0
  35. package/development/select.js +1 -1
  36. package/development/select.pure.js +1 -1
  37. package/development/styles-CmwkvLqU.js +22 -0
  38. package/off-brand-theme.css +7 -5
  39. package/package.json +2 -2
  40. package/safety-theme.css +7 -5
  41. package/select/select.component.js +1 -1
  42. package/{select.component-PLjAiAA3.js → select.component-CO-42jqH.js} +61 -61
  43. package/select.js +1 -1
  44. package/select.pure.js +1 -1
  45. package/standard-theme.css +7 -5
  46. package/styles-CgXa3TG9.js +5 -0
  47. package/core/overlay/overlay.js +0 -14
  48. package/core/styles/mixins/overlay.scss +0 -175
  49. package/development/autocomplete-base-element-CG5m0PpY.js +0 -686
  50. package/development/core/overlay/overlay.d.ts +0 -7
  51. package/development/core/overlay/overlay.d.ts.map +0 -1
  52. package/development/core/overlay/overlay.js +0 -20
  53. package/development/select.component-B8_tnz6D.js +0 -806
  54. package/development/styles-BzEbxmCd.js +0 -18
  55. package/styles-OHFAsnJq.js +0 -5
@@ -0,0 +1,683 @@
1
+ import { __esDecorate, __runInitializers } from "tslib";
2
+ import { html, isServer, nothing, unsafeCSS } from "lit";
3
+ import { property } from "lit/decorators.js";
4
+ import { SbbEscapableOverlayController, SbbNegativeMixin, SbbOpenCloseBaseElement, SbbPropertyWatcherController, forceType, idReference, isEventOnElement, isLean, isSafari, isZeroAnimationDuration, optionPanelStyles, removeAriaComboBoxAttributes, scrollbarStyles, setOverlayPosition } from "./core.js";
5
+ import { ResizeController } from "@lit-labs/observers/resize-controller.js";
6
+ import { ref } from "lit/directives/ref.js";
7
+ //#region src/elements/autocomplete/autocomplete-base-element.scss?inline
8
+ var autocomplete_base_element_default = ":host {\n display: none;\n}\n\n:host(:is(:is(:state(state-opening),[state--state-opening]), :is(:state(state-opened),[state--state-opened]), :is(:state(state-closing),[state--state-closing]))) {\n display: block;\n}";
9
+ //#endregion
10
+ //#region src/elements/autocomplete/autocomplete-base-element.ts
11
+ /**
12
+ * On Safari, the aria role 'listbox' must be on the host element, or else VoiceOver won't work at all.
13
+ * On the other hand, JAWS and NVDA need the role to be "closer" to the options, or else optgroups won't work.
14
+ */
15
+ var ariaRoleOnHost = isSafari;
16
+ /**
17
+ * Base class for autocomplete components.
18
+ *
19
+ * @event {Event} change - The change event is fired on the autocomplete's trigger when the user modifies the element's value. Unlike the input event, the change event is not necessarily fired for each alteration to an element's value.
20
+ * @event {InputEvent} input - The input event fires on the autocomplete's trigger when the value has been changed as a direct result of a user action.
21
+ */
22
+ var SbbAutocompleteBaseElement = (() => {
23
+ let _classSuper = SbbNegativeMixin(SbbOpenCloseBaseElement);
24
+ let _origin_decorators;
25
+ let _origin_initializers = [];
26
+ let _origin_extraInitializers = [];
27
+ let _trigger_decorators;
28
+ let _trigger_initializers = [];
29
+ let _trigger_extraInitializers = [];
30
+ let _preserveIconSpace_decorators;
31
+ let _preserveIconSpace_initializers = [];
32
+ let _preserveIconSpace_extraInitializers = [];
33
+ let _autoActiveFirstOption_decorators;
34
+ let _autoActiveFirstOption_initializers = [];
35
+ let _autoActiveFirstOption_extraInitializers = [];
36
+ let _displayWith_decorators;
37
+ let _displayWith_initializers = [];
38
+ let _displayWith_extraInitializers = [];
39
+ let _size_decorators;
40
+ let _size_initializers = [];
41
+ let _size_extraInitializers = [];
42
+ let _autoSelectActiveOption_decorators;
43
+ let _autoSelectActiveOption_initializers = [];
44
+ let _autoSelectActiveOption_extraInitializers = [];
45
+ let _autoSelectActiveOptionOnBlur_decorators;
46
+ let _autoSelectActiveOptionOnBlur_initializers = [];
47
+ let _autoSelectActiveOptionOnBlur_extraInitializers = [];
48
+ let _requireSelection_decorators;
49
+ let _requireSelection_initializers = [];
50
+ let _requireSelection_extraInitializers = [];
51
+ let _position_decorators;
52
+ let _position_initializers = [];
53
+ let _position_extraInitializers = [];
54
+ return class SbbAutocompleteBaseElement extends _classSuper {
55
+ static {
56
+ const _metadata = typeof Symbol === "function" && Symbol.metadata ? Object.create(_classSuper[Symbol.metadata] ?? null) : void 0;
57
+ _origin_decorators = [idReference(), property()];
58
+ _trigger_decorators = [idReference(), property()];
59
+ _preserveIconSpace_decorators = [forceType(), property({
60
+ attribute: "preserve-icon-space",
61
+ reflect: true,
62
+ type: Boolean
63
+ })];
64
+ _autoActiveFirstOption_decorators = [forceType(), property({
65
+ attribute: "auto-active-first-option",
66
+ type: Boolean
67
+ })];
68
+ _displayWith_decorators = [property({ attribute: false })];
69
+ _size_decorators = [property({ reflect: true })];
70
+ _autoSelectActiveOption_decorators = [forceType(), property({
71
+ attribute: "auto-select-active-option",
72
+ type: Boolean
73
+ })];
74
+ _autoSelectActiveOptionOnBlur_decorators = [forceType(), property({
75
+ attribute: "auto-select-active-option-on-blur",
76
+ type: Boolean
77
+ })];
78
+ _requireSelection_decorators = [forceType(), property({
79
+ attribute: "require-selection",
80
+ type: Boolean
81
+ })];
82
+ _position_decorators = [property()];
83
+ __esDecorate(this, null, _origin_decorators, {
84
+ kind: "accessor",
85
+ name: "origin",
86
+ static: false,
87
+ private: false,
88
+ access: {
89
+ has: (obj) => "origin" in obj,
90
+ get: (obj) => obj.origin,
91
+ set: (obj, value) => {
92
+ obj.origin = value;
93
+ }
94
+ },
95
+ metadata: _metadata
96
+ }, _origin_initializers, _origin_extraInitializers);
97
+ __esDecorate(this, null, _trigger_decorators, {
98
+ kind: "accessor",
99
+ name: "trigger",
100
+ static: false,
101
+ private: false,
102
+ access: {
103
+ has: (obj) => "trigger" in obj,
104
+ get: (obj) => obj.trigger,
105
+ set: (obj, value) => {
106
+ obj.trigger = value;
107
+ }
108
+ },
109
+ metadata: _metadata
110
+ }, _trigger_initializers, _trigger_extraInitializers);
111
+ __esDecorate(this, null, _preserveIconSpace_decorators, {
112
+ kind: "accessor",
113
+ name: "preserveIconSpace",
114
+ static: false,
115
+ private: false,
116
+ access: {
117
+ has: (obj) => "preserveIconSpace" in obj,
118
+ get: (obj) => obj.preserveIconSpace,
119
+ set: (obj, value) => {
120
+ obj.preserveIconSpace = value;
121
+ }
122
+ },
123
+ metadata: _metadata
124
+ }, _preserveIconSpace_initializers, _preserveIconSpace_extraInitializers);
125
+ __esDecorate(this, null, _autoActiveFirstOption_decorators, {
126
+ kind: "accessor",
127
+ name: "autoActiveFirstOption",
128
+ static: false,
129
+ private: false,
130
+ access: {
131
+ has: (obj) => "autoActiveFirstOption" in obj,
132
+ get: (obj) => obj.autoActiveFirstOption,
133
+ set: (obj, value) => {
134
+ obj.autoActiveFirstOption = value;
135
+ }
136
+ },
137
+ metadata: _metadata
138
+ }, _autoActiveFirstOption_initializers, _autoActiveFirstOption_extraInitializers);
139
+ __esDecorate(this, null, _displayWith_decorators, {
140
+ kind: "accessor",
141
+ name: "displayWith",
142
+ static: false,
143
+ private: false,
144
+ access: {
145
+ has: (obj) => "displayWith" in obj,
146
+ get: (obj) => obj.displayWith,
147
+ set: (obj, value) => {
148
+ obj.displayWith = value;
149
+ }
150
+ },
151
+ metadata: _metadata
152
+ }, _displayWith_initializers, _displayWith_extraInitializers);
153
+ __esDecorate(this, null, _size_decorators, {
154
+ kind: "accessor",
155
+ name: "size",
156
+ static: false,
157
+ private: false,
158
+ access: {
159
+ has: (obj) => "size" in obj,
160
+ get: (obj) => obj.size,
161
+ set: (obj, value) => {
162
+ obj.size = value;
163
+ }
164
+ },
165
+ metadata: _metadata
166
+ }, _size_initializers, _size_extraInitializers);
167
+ __esDecorate(this, null, _autoSelectActiveOption_decorators, {
168
+ kind: "accessor",
169
+ name: "autoSelectActiveOption",
170
+ static: false,
171
+ private: false,
172
+ access: {
173
+ has: (obj) => "autoSelectActiveOption" in obj,
174
+ get: (obj) => obj.autoSelectActiveOption,
175
+ set: (obj, value) => {
176
+ obj.autoSelectActiveOption = value;
177
+ }
178
+ },
179
+ metadata: _metadata
180
+ }, _autoSelectActiveOption_initializers, _autoSelectActiveOption_extraInitializers);
181
+ __esDecorate(this, null, _autoSelectActiveOptionOnBlur_decorators, {
182
+ kind: "accessor",
183
+ name: "autoSelectActiveOptionOnBlur",
184
+ static: false,
185
+ private: false,
186
+ access: {
187
+ has: (obj) => "autoSelectActiveOptionOnBlur" in obj,
188
+ get: (obj) => obj.autoSelectActiveOptionOnBlur,
189
+ set: (obj, value) => {
190
+ obj.autoSelectActiveOptionOnBlur = value;
191
+ }
192
+ },
193
+ metadata: _metadata
194
+ }, _autoSelectActiveOptionOnBlur_initializers, _autoSelectActiveOptionOnBlur_extraInitializers);
195
+ __esDecorate(this, null, _requireSelection_decorators, {
196
+ kind: "accessor",
197
+ name: "requireSelection",
198
+ static: false,
199
+ private: false,
200
+ access: {
201
+ has: (obj) => "requireSelection" in obj,
202
+ get: (obj) => obj.requireSelection,
203
+ set: (obj, value) => {
204
+ obj.requireSelection = value;
205
+ }
206
+ },
207
+ metadata: _metadata
208
+ }, _requireSelection_initializers, _requireSelection_extraInitializers);
209
+ __esDecorate(this, null, _position_decorators, {
210
+ kind: "accessor",
211
+ name: "position",
212
+ static: false,
213
+ private: false,
214
+ access: {
215
+ has: (obj) => "position" in obj,
216
+ get: (obj) => obj.position,
217
+ set: (obj, value) => {
218
+ obj.position = value;
219
+ }
220
+ },
221
+ metadata: _metadata
222
+ }, _position_initializers, _position_extraInitializers);
223
+ if (_metadata) Object.defineProperty(this, Symbol.metadata, {
224
+ enumerable: true,
225
+ configurable: true,
226
+ writable: true,
227
+ value: _metadata
228
+ });
229
+ }
230
+ static {
231
+ this.styles = [
232
+ scrollbarStyles,
233
+ optionPanelStyles,
234
+ unsafeCSS(autocomplete_base_element_default)
235
+ ];
236
+ }
237
+ #origin_accessor_storage;
238
+ /**
239
+ * The element where the autocomplete will attach.
240
+ * If not set, as fallback there are two elements which can act as origin with following priority order:
241
+ * 1. `sbb-form-field` if it is an ancestor.
242
+ * 2. trigger element if set.
243
+ *
244
+ * For attribute usage, provide an id reference.
245
+ */
246
+ get origin() {
247
+ return this.#origin_accessor_storage;
248
+ }
249
+ set origin(value) {
250
+ this.#origin_accessor_storage = value;
251
+ }
252
+ #trigger_accessor_storage;
253
+ /**
254
+ * The input element that will trigger the autocomplete opening.
255
+ * By default, the autocomplete will open on focus, click, input or `ArrowDown` keypress of the 'trigger' element.
256
+ * If not set, will search for the first 'input' child of a 'sbb-form-field' ancestor.
257
+ *
258
+ * For attribute usage, provide an id reference.
259
+ */
260
+ get trigger() {
261
+ return this.#trigger_accessor_storage;
262
+ }
263
+ set trigger(value) {
264
+ this.#trigger_accessor_storage = value;
265
+ }
266
+ #preserveIconSpace_accessor_storage;
267
+ /** Whether the icon space is preserved when no icon is set. */
268
+ get preserveIconSpace() {
269
+ return this.#preserveIconSpace_accessor_storage;
270
+ }
271
+ set preserveIconSpace(value) {
272
+ this.#preserveIconSpace_accessor_storage = value;
273
+ }
274
+ #autoActiveFirstOption_accessor_storage;
275
+ /** Whether the first option is automatically activated when the autocomplete is opened. */
276
+ get autoActiveFirstOption() {
277
+ return this.#autoActiveFirstOption_accessor_storage;
278
+ }
279
+ set autoActiveFirstOption(value) {
280
+ this.#autoActiveFirstOption_accessor_storage = value;
281
+ }
282
+ #displayWith_accessor_storage;
283
+ /** Function that maps an option's control value to its display value in the trigger. */
284
+ get displayWith() {
285
+ return this.#displayWith_accessor_storage;
286
+ }
287
+ set displayWith(value) {
288
+ this.#displayWith_accessor_storage = value;
289
+ }
290
+ #size_accessor_storage;
291
+ /**
292
+ * Size variant, either m or s.
293
+ * @default 'm' / 's' (lean)
294
+ */
295
+ get size() {
296
+ return this.#size_accessor_storage;
297
+ }
298
+ set size(value) {
299
+ this.#size_accessor_storage = value;
300
+ }
301
+ #autoSelectActiveOption_accessor_storage;
302
+ /** Whether the active option should be selected as the user is navigating. */
303
+ get autoSelectActiveOption() {
304
+ return this.#autoSelectActiveOption_accessor_storage;
305
+ }
306
+ set autoSelectActiveOption(value) {
307
+ this.#autoSelectActiveOption_accessor_storage = value;
308
+ }
309
+ #autoSelectActiveOptionOnBlur_accessor_storage;
310
+ /**
311
+ * When enabled, the active option is automatically selected on blur.
312
+ * This is an experimental feature. It might be subject to changes.
313
+ */
314
+ get autoSelectActiveOptionOnBlur() {
315
+ return this.#autoSelectActiveOptionOnBlur_accessor_storage;
316
+ }
317
+ set autoSelectActiveOptionOnBlur(value) {
318
+ this.#autoSelectActiveOptionOnBlur_accessor_storage = value;
319
+ }
320
+ #requireSelection_accessor_storage;
321
+ /**
322
+ * Whether the user is required to make a selection when they're interacting with the
323
+ * autocomplete. If the user moves away from the autocomplete without selecting an option from
324
+ * the list, the value will be reset. If the user opens the panel and closes it without
325
+ * interacting or selecting a value, the initial value will be kept.
326
+ */
327
+ get requireSelection() {
328
+ return this.#requireSelection_accessor_storage;
329
+ }
330
+ set requireSelection(value) {
331
+ this.#requireSelection_accessor_storage = value;
332
+ }
333
+ #position_accessor_storage;
334
+ /**
335
+ * The position of the autocomplete panel relative to the trigger.
336
+ * @default 'auto'
337
+ */
338
+ get position() {
339
+ return this.#position_accessor_storage;
340
+ }
341
+ set position(value) {
342
+ this.#position_accessor_storage = value;
343
+ }
344
+ /** Returns the element where the autocomplete overlay is attached to. */
345
+ get originElement() {
346
+ return this.origin ?? this.closest?.("sbb-form-field")?.shadowRoot?.querySelector?.("#overlay-anchor") ?? this.trigger ?? null;
347
+ }
348
+ /** Returns the trigger element. */
349
+ get triggerElement() {
350
+ return this._triggerElement ?? null;
351
+ }
352
+ constructor() {
353
+ super();
354
+ this.#origin_accessor_storage = __runInitializers(this, _origin_initializers, null);
355
+ this.#trigger_accessor_storage = (__runInitializers(this, _origin_extraInitializers), __runInitializers(this, _trigger_initializers, null));
356
+ this.#preserveIconSpace_accessor_storage = (__runInitializers(this, _trigger_extraInitializers), __runInitializers(this, _preserveIconSpace_initializers, false));
357
+ this.#autoActiveFirstOption_accessor_storage = (__runInitializers(this, _preserveIconSpace_extraInitializers), __runInitializers(this, _autoActiveFirstOption_initializers, false));
358
+ this.#displayWith_accessor_storage = (__runInitializers(this, _autoActiveFirstOption_extraInitializers), __runInitializers(this, _displayWith_initializers, null));
359
+ this.#size_accessor_storage = (__runInitializers(this, _displayWith_extraInitializers), __runInitializers(this, _size_initializers, isLean() ? "s" : "m"));
360
+ this.#autoSelectActiveOption_accessor_storage = (__runInitializers(this, _size_extraInitializers), __runInitializers(this, _autoSelectActiveOption_initializers, false));
361
+ this.#autoSelectActiveOptionOnBlur_accessor_storage = (__runInitializers(this, _autoSelectActiveOption_extraInitializers), __runInitializers(this, _autoSelectActiveOptionOnBlur_initializers, false));
362
+ this.#requireSelection_accessor_storage = (__runInitializers(this, _autoSelectActiveOptionOnBlur_extraInitializers), __runInitializers(this, _requireSelection_initializers, false));
363
+ this.#position_accessor_storage = (__runInitializers(this, _requireSelection_extraInitializers), __runInitializers(this, _position_initializers, "auto"));
364
+ this._triggerElement = __runInitializers(this, _position_extraInitializers);
365
+ this.activeOption = null;
366
+ this.pendingAutoSelectedOption = null;
367
+ this._resizeObserver = new ResizeController(this, {
368
+ target: null,
369
+ skipInitial: true,
370
+ callback: (() => {
371
+ let timeoutId;
372
+ return () => {
373
+ clearTimeout(timeoutId);
374
+ timeoutId = setTimeout(() => {
375
+ if (this.state !== "closed") this._setOverlayPosition();
376
+ }, 10);
377
+ };
378
+ })()
379
+ });
380
+ /** Listens to the changes on the `disabled` or `readonly` attribute of the trigger. */
381
+ this._triggerAttributeObserver = !isServer ? new MutationObserver((mutations) => {
382
+ const input = mutations[0].target;
383
+ if (input.hasAttribute("disabled") || input.hasAttribute("readonly")) this.close();
384
+ }) : null;
385
+ this._isPointerDownEventOnMenu = false;
386
+ this._escapableOverlayController = new SbbEscapableOverlayController(this);
387
+ this._optionsCount = 0;
388
+ /** Tracks input from keyboard. */
389
+ this._lastUserInput = null;
390
+ /** If true, the 'change' event source is an option selection or a 'requireSelection' field cleanup */
391
+ this._isCustomChangeEvent = false;
392
+ this._pointerDownListener = (event) => {
393
+ this._isPointerDownEventOnMenu = isEventOnElement(this._overlay, event);
394
+ };
395
+ this._closeOnBackdropClick = (event) => {
396
+ if (!this._isPointerDownEventOnMenu && !isEventOnElement(this._overlay, event) && !isEventOnElement(this.originElement, event)) this.close();
397
+ };
398
+ this.addEventListener?.("optionselected", (e) => this.onOptionSelected(e));
399
+ this.addEventListener?.("ɵoptgroupslotchange", () => this._handleSlotchange(), { capture: true });
400
+ this.addController(new SbbPropertyWatcherController(this, () => this.closest("sbb-form-field"), {
401
+ negative: (e) => {
402
+ this.negative = e.negative;
403
+ this.syncNegative();
404
+ },
405
+ borderless: (e) => this.toggleState("option-panel-origin-borderless", e.borderless)
406
+ }));
407
+ }
408
+ /** Opens the autocomplete. */
409
+ open() {
410
+ if (this.state === "opening" || this.state === "opened" || !this._overlay || this.options.length === 0 || this._readonly() || !this.dispatchBeforeOpenEvent()) return;
411
+ this.showPopover?.();
412
+ this.state = "opening";
413
+ this.triggerElement?.toggleAttribute("data-expanded", true);
414
+ const originElement = this.originElement;
415
+ if (!originElement) throw new Error("Cannot find the origin element. Please specify a valid element or check the usage of the \"origin\" property from the documentation");
416
+ this._setOverlayPosition(originElement);
417
+ this._setNextActiveOptionIfAutoActiveFirstOption();
418
+ this._attachOpenPanelEvents();
419
+ this._escapableOverlayController.connect();
420
+ if (this._isZeroAnimationDuration()) this._handleOpening();
421
+ }
422
+ /** Closes the autocomplete. */
423
+ close() {
424
+ if (this.state === "closing" || this.state === "closed" || !this.dispatchBeforeCloseEvent()) return;
425
+ if (this.pendingAutoSelectedOption) {
426
+ this.pendingAutoSelectedOption.selected = true;
427
+ this._setValueAndDispatchEvents(this.pendingAutoSelectedOption, true);
428
+ }
429
+ this.state = "closing";
430
+ this.triggerElement?.removeAttribute("data-expanded");
431
+ this._openPanelEventsController.abort();
432
+ if (this.originElement) this._resizeObserver.unobserve(this.originElement);
433
+ if (this._isZeroAnimationDuration()) this._handleClosing();
434
+ }
435
+ _isZeroAnimationDuration() {
436
+ return isZeroAnimationDuration(this, "--sbb-options-panel-animation-duration");
437
+ }
438
+ connectedCallback() {
439
+ this.popover = "manual";
440
+ super.connectedCallback();
441
+ if (ariaRoleOnHost) this.id ||= this.overlayId;
442
+ if (this.hasUpdated) this._componentSetup();
443
+ this.syncNegative();
444
+ }
445
+ willUpdate(changedProperties) {
446
+ super.willUpdate(changedProperties);
447
+ if (changedProperties.has("negative")) this.syncNegative();
448
+ if (changedProperties.has("autoActiveFirstOption") && this.isOpen) this._setNextActiveOptionIfAutoActiveFirstOption();
449
+ }
450
+ firstUpdated(changedProperties) {
451
+ super.firstUpdated(changedProperties);
452
+ this._componentSetup();
453
+ }
454
+ disconnectedCallback() {
455
+ super.disconnectedCallback();
456
+ this._triggerElement = null;
457
+ this._triggerAbortController?.abort();
458
+ this._openPanelEventsController?.abort();
459
+ }
460
+ requestUpdate(name, oldValue, options) {
461
+ super.requestUpdate(name, oldValue, options);
462
+ if (isServer || !this.hasUpdated) return;
463
+ if (!name || name === "trigger") this._configureTrigger();
464
+ else if ((!name || name === "origin") && this.isOpen) this._setOverlayPosition();
465
+ }
466
+ /** When an option is selected, update the input value and close the autocomplete. */
467
+ onOptionSelected(event) {
468
+ const target = event.target;
469
+ this._setValueAndDispatchEvents(target);
470
+ this.close();
471
+ }
472
+ /** @deprecated */
473
+ onOptionArrowsSelected(_activeOption) {}
474
+ /**
475
+ * A 'pending selection' sets the option value in the input element without emitting events.
476
+ * A 'pending selection' is confirmed when the panel closes. Any other user interaction
477
+ * will reset the pending value.
478
+ */
479
+ setPendingSelection(activeOption) {
480
+ this.pendingAutoSelectedOption = activeOption;
481
+ this._setInputValue(activeOption);
482
+ }
483
+ _setValueAndDispatchEvents(selectedOption, preventFocus = false) {
484
+ this.options.filter((option) => option.id !== selectedOption.id && option.selected).forEach((option) => option.selected = false);
485
+ this.pendingAutoSelectedOption = null;
486
+ if (this.triggerElement) {
487
+ this._setInputValue(selectedOption);
488
+ this._isCustomChangeEvent = true;
489
+ this.triggerElement.dispatchEvent(new Event("change", { bubbles: true }));
490
+ this.triggerElement.dispatchEvent(new InputEvent("input", {
491
+ bubbles: true,
492
+ composed: true
493
+ }));
494
+ this._lastUserInput = null;
495
+ this.triggerElement.dispatchEvent(new CustomEvent("inputAutocomplete", { detail: { option: selectedOption } }));
496
+ if (!preventFocus) this.triggerElement.focus();
497
+ }
498
+ }
499
+ /** Set the option value within the input element */
500
+ _setInputValue(option) {
501
+ if (!this.triggerElement) return;
502
+ const toDisplay = this.displayWith?.(option.value) ?? option.value;
503
+ Object.getOwnPropertyDescriptor(HTMLInputElement.prototype, "value").set.call(this.triggerElement, toDisplay);
504
+ }
505
+ _handleSlotchange() {
506
+ this._highlightOptions(this.triggerElement?.value);
507
+ const currentElements = Array.from(this.querySelectorAll("*"));
508
+ this._previousElements?.forEach((e) => this._resizeObserver.unobserve(e));
509
+ this._previousElements = currentElements;
510
+ this._previousElements.forEach((e) => this._resizeObserver.observe(e));
511
+ /**
512
+ * It's possible to filter out options with an opened panel on input change.
513
+ * In this case, the panel's position must be recalculated considering the new option's list.
514
+ */
515
+ if (this.isOpen) {
516
+ this._setOverlayPosition();
517
+ this._setNextActiveOptionIfAutoActiveFirstOption();
518
+ if (this._optionsCount > 0 && this.options.length === 0) this.close();
519
+ } else if (document?.activeElement === this.triggerElement && this._optionsCount === 0 && this.options.length > 0) this.open();
520
+ this._optionsCount = this.options.length;
521
+ }
522
+ _setNextActiveOptionIfAutoActiveFirstOption() {
523
+ if (this.autoActiveFirstOption) {
524
+ this.resetActiveElement();
525
+ this.setNextActiveOption();
526
+ }
527
+ }
528
+ /** The autocomplete should inherit 'readonly' state from the trigger. */
529
+ _readonly() {
530
+ return this.triggerElement?.hasAttribute("readonly") ?? false;
531
+ }
532
+ _componentSetup() {
533
+ if (isServer) return;
534
+ this._configureTrigger();
535
+ }
536
+ _configureTrigger() {
537
+ const triggerElement = this.trigger ?? this.closest?.("sbb-form-field")?.querySelector("input");
538
+ if (triggerElement === this.triggerElement) return;
539
+ this._triggerAbortController?.abort();
540
+ removeAriaComboBoxAttributes(this.triggerElement);
541
+ this.triggerElement?.removeAttribute("data-expanded");
542
+ this._triggerElement = triggerElement;
543
+ if (!this.triggerElement) return;
544
+ const originElement = this.originElement;
545
+ if (this.triggerElement === originElement && this.isOpen) this._setOverlayPosition(originElement);
546
+ this._triggerAttributeObserver?.observe(this.triggerElement, {
547
+ attributes: true,
548
+ attributeFilter: ["disabled", "readonly"]
549
+ });
550
+ this.setTriggerAttributes(this.triggerElement);
551
+ this._triggerAbortController = new AbortController();
552
+ this.triggerElement.addEventListener("focus", () => this.open(), { signal: this._triggerAbortController.signal });
553
+ this.triggerElement.addEventListener("click", () => this.open(), { signal: this._triggerAbortController.signal });
554
+ this.triggerElement.addEventListener("input", (event) => {
555
+ const value = event.target.value;
556
+ if (value && event.isTrusted) this.open();
557
+ this._highlightOptions(value);
558
+ this._lastUserInput = value;
559
+ this.pendingAutoSelectedOption = null;
560
+ }, { signal: this._triggerAbortController.signal });
561
+ this.triggerElement.addEventListener("change", (event) => {
562
+ /**
563
+ * In 'requireSelection' mode, we block the native change events and
564
+ * let only pass the ones that come with a valid value (when an option is selected)
565
+ */
566
+ if (this.requireSelection && !this._isCustomChangeEvent) event.stopImmediatePropagation();
567
+ this._isCustomChangeEvent = false;
568
+ }, {
569
+ signal: this._triggerAbortController.signal,
570
+ capture: true
571
+ });
572
+ this.triggerElement.addEventListener("keydown", (event) => this._closedPanelKeyboardInteraction(event), {
573
+ signal: this._triggerAbortController.signal,
574
+ capture: true
575
+ });
576
+ this.triggerElement.addEventListener("blur", (e) => {
577
+ if (this.contains(e.relatedTarget)) return;
578
+ if (this.autoSelectActiveOptionOnBlur && this.activeOption && this._lastUserInput && this.triggerElement?.value) {
579
+ this.activeOption.selected = true;
580
+ this._setValueAndDispatchEvents(this.activeOption, true);
581
+ }
582
+ if (this.requireSelection && this.triggerElement && this._lastUserInput != null) {
583
+ Object.getOwnPropertyDescriptor(HTMLInputElement.prototype, "value").set.call(this.triggerElement, "");
584
+ this._highlightOptions("");
585
+ this._isCustomChangeEvent = true;
586
+ this.triggerElement.dispatchEvent(new Event("change", { bubbles: true }));
587
+ this.triggerElement.dispatchEvent(new InputEvent("input", {
588
+ bubbles: true,
589
+ composed: true
590
+ }));
591
+ }
592
+ this.close();
593
+ }, {
594
+ signal: this._triggerAbortController.signal,
595
+ capture: true
596
+ });
597
+ }
598
+ _setOverlayPosition(originElement = this.originElement) {
599
+ if (!originElement) return;
600
+ setOverlayPosition(this._overlay, originElement, this._overlay, this.shadowRoot.querySelector(".sbb-option-panel__overlay-container"), this, this.position);
601
+ }
602
+ /**
603
+ * On open/close animation end.
604
+ * In rare cases it can be that the animationEnd event is triggered twice.
605
+ * To avoid entering a corrupt state, exit when state is not expected.
606
+ */
607
+ _onAnimationEnd(event) {
608
+ if (event.animationName === "open" && this.state === "opening") this._handleOpening();
609
+ else if (event.animationName === "close" && this.state === "closing") this._handleClosing();
610
+ }
611
+ _handleOpening() {
612
+ this.state = "opened";
613
+ if (this.originElement) this._resizeObserver.observe(this.originElement);
614
+ this.triggerElement?.setAttribute("aria-expanded", "true");
615
+ this.dispatchOpenEvent();
616
+ }
617
+ _handleClosing() {
618
+ this.state = "closed";
619
+ this.hidePopover?.();
620
+ this.triggerElement?.setAttribute("aria-expanded", "false");
621
+ this.resetActiveElement();
622
+ this._overlay.scrollTop = 0;
623
+ this._escapableOverlayController.disconnect();
624
+ this.dispatchCloseEvent();
625
+ }
626
+ _attachOpenPanelEvents() {
627
+ this._openPanelEventsController = new AbortController();
628
+ document.addEventListener("scroll", () => this._setOverlayPosition(), {
629
+ passive: true,
630
+ signal: this._openPanelEventsController.signal,
631
+ capture: true
632
+ });
633
+ window.addEventListener("resize", () => this._setOverlayPosition(), {
634
+ passive: true,
635
+ signal: this._openPanelEventsController.signal
636
+ });
637
+ window.addEventListener("pointerdown", (ev) => this._pointerDownListener(ev), { signal: this._openPanelEventsController.signal });
638
+ window.addEventListener("pointerup", (ev) => this._closeOnBackdropClick(ev), { signal: this._openPanelEventsController.signal });
639
+ this.addEventListener("ɵdisabledchange", () => {
640
+ if (this.activeOption?.disabled) this.resetActiveElement();
641
+ this._setNextActiveOptionIfAutoActiveFirstOption();
642
+ }, { signal: this._openPanelEventsController.signal });
643
+ this.triggerElement?.addEventListener("keydown", (event) => this.openedPanelKeyboardInteraction(event), {
644
+ signal: this._openPanelEventsController.signal,
645
+ capture: true
646
+ });
647
+ }
648
+ _closedPanelKeyboardInteraction(event) {
649
+ if (this.state === "opening" || this.state === "opened") return;
650
+ switch (event.key) {
651
+ case "ArrowDown":
652
+ case "ArrowUp":
653
+ this.open();
654
+ break;
655
+ }
656
+ }
657
+ /** Highlight the searched text on the options. */
658
+ _highlightOptions(searchTerm) {
659
+ if (searchTerm === null || searchTerm === void 0) return;
660
+ this.options.forEach((option) => option.highlight(searchTerm));
661
+ }
662
+ render() {
663
+ return html`
664
+ <div class="sbb-option-panel__overlay-container">
665
+ <div
666
+ class="sbb-option-panel__overlay ${this.negative ? "sbb-scrollbar-negative" : "sbb-scrollbar"}"
667
+ role=${!ariaRoleOnHost ? this.panelRole : nothing}
668
+ id=${!ariaRoleOnHost ? this.overlayId : nothing}
669
+ tabindex="-1"
670
+ @animationend=${this._onAnimationEnd}
671
+ ${ref((overlayRef) => this._overlay = overlayRef)}
672
+ >
673
+ <slot @slotchange=${this._handleSlotchange}></slot>
674
+ </div>
675
+ </div>
676
+ `;
677
+ }
678
+ };
679
+ })();
680
+ //#endregion
681
+ export { SbbAutocompleteBaseElement as t };
682
+
683
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYXV0b2NvbXBsZXRlLWJhc2UtZWxlbWVudC1EWng1M1BPMS5qcyIsIm5hbWVzIjpbXSwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvZWxlbWVudHMvYXV0b2NvbXBsZXRlL2F1dG9jb21wbGV0ZS1iYXNlLWVsZW1lbnQuc2Nzcz9pbmxpbmUiLCIuLi8uLi8uLi9zcmMvZWxlbWVudHMvYXV0b2NvbXBsZXRlL2F1dG9jb21wbGV0ZS1iYXNlLWVsZW1lbnQudHMiXSwic291cmNlc0NvbnRlbnQiOlsiQHVzZSAnLi4vY29yZS9zdHlsZXMnIGFzIHNiYjtcblxuOmhvc3Qge1xuICBkaXNwbGF5OiBub25lO1xufVxuXG46aG9zdCg6aXMoOnN0YXRlKHN0YXRlLW9wZW5pbmcpLCA6c3RhdGUoc3RhdGUtb3BlbmVkKSwgOnN0YXRlKHN0YXRlLWNsb3NpbmcpKSkge1xuICBkaXNwbGF5OiBibG9jaztcbn1cbiIsImltcG9ydCB7IFJlc2l6ZUNvbnRyb2xsZXIgfSBmcm9tICdAbGl0LWxhYnMvb2JzZXJ2ZXJzL3Jlc2l6ZS1jb250cm9sbGVyLmpzJztcbmltcG9ydCB7XG4gIHR5cGUgQ1NTUmVzdWx0R3JvdXAsXG4gIGh0bWwsXG4gIGlzU2VydmVyLFxuICBub3RoaW5nLFxuICB0eXBlIFByb3BlcnR5RGVjbGFyYXRpb24sXG4gIHR5cGUgUHJvcGVydHlWYWx1ZXMsXG4gIHR5cGUgVGVtcGxhdGVSZXN1bHQsXG4gIHVuc2FmZUNTUyxcbn0gZnJvbSAnbGl0JztcbmltcG9ydCB7IHByb3BlcnR5IH0gZnJvbSAnbGl0L2RlY29yYXRvcnMuanMnO1xuaW1wb3J0IHsgcmVmIH0gZnJvbSAnbGl0L2RpcmVjdGl2ZXMvcmVmLmpzJztcblxuaW1wb3J0IHtcbiAgZm9yY2VUeXBlLFxuICBpZFJlZmVyZW5jZSxcbiAgaXNFdmVudE9uRWxlbWVudCxcbiAgaXNMZWFuLFxuICBpc1NhZmFyaSxcbiAgaXNaZXJvQW5pbWF0aW9uRHVyYXRpb24sXG4gIG9wdGlvblBhbmVsU3R5bGVzLFxuICByZW1vdmVBcmlhQ29tYm9Cb3hBdHRyaWJ1dGVzLFxuICBTYmJFc2NhcGFibGVPdmVybGF5Q29udHJvbGxlcixcbiAgU2JiTmVnYXRpdmVNaXhpbixcbiAgU2JiT3BlbkNsb3NlQmFzZUVsZW1lbnQsXG4gIFNiYlByb3BlcnR5V2F0Y2hlckNvbnRyb2xsZXIsXG4gIHNjcm9sbGJhclN0eWxlcyxcbiAgc2V0T3ZlcmxheVBvc2l0aW9uLFxufSBmcm9tICcuLi9jb3JlLnRzJztcbmltcG9ydCB0eXBlIHsgU2JiRm9ybUZpZWxkRWxlbWVudCB9IGZyb20gJy4uL2Zvcm0tZmllbGQvZm9ybS1maWVsZC9mb3JtLWZpZWxkLmNvbXBvbmVudC50cyc7XG5pbXBvcnQgdHlwZSB7IFNiYk9wdGlvbkJhc2VFbGVtZW50IH0gZnJvbSAnLi4vb3B0aW9uLnB1cmUudHMnO1xuXG5pbXBvcnQgc3R5bGUgZnJvbSAnLi9hdXRvY29tcGxldGUtYmFzZS1lbGVtZW50LnNjc3M/aW5saW5lJztcblxuLyoqXG4gKiBPbiBTYWZhcmksIHRoZSBhcmlhIHJvbGUgJ2xpc3Rib3gnIG11c3QgYmUgb24gdGhlIGhvc3QgZWxlbWVudCwgb3IgZWxzZSBWb2ljZU92ZXIgd29uJ3Qgd29yayBhdCBhbGwuXG4gKiBPbiB0aGUgb3RoZXIgaGFuZCwgSkFXUyBhbmQgTlZEQSBuZWVkIHRoZSByb2xlIHRvIGJlIFwiY2xvc2VyXCIgdG8gdGhlIG9wdGlvbnMsIG9yIGVsc2Ugb3B0Z3JvdXBzIHdvbid0IHdvcmsuXG4gKi9cbmNvbnN0IGFyaWFSb2xlT25Ib3N0ID0gaXNTYWZhcmk7XG5cbi8qKlxuICogQmFzZSBjbGFzcyBmb3IgYXV0b2NvbXBsZXRlIGNvbXBvbmVudHMuXG4gKlxuICogQGV2ZW50IHtFdmVudH0gY2hhbmdlIC0gVGhlIGNoYW5nZSBldmVudCBpcyBmaXJlZCBvbiB0aGUgYXV0b2NvbXBsZXRlJ3MgdHJpZ2dlciB3aGVuIHRoZSB1c2VyIG1vZGlmaWVzIHRoZSBlbGVtZW50J3MgdmFsdWUuIFVubGlrZSB0aGUgaW5wdXQgZXZlbnQsIHRoZSBjaGFuZ2UgZXZlbnQgaXMgbm90IG5lY2Vzc2FyaWx5IGZpcmVkIGZvciBlYWNoIGFsdGVyYXRpb24gdG8gYW4gZWxlbWVudCdzIHZhbHVlLlxuICogQGV2ZW50IHtJbnB1dEV2ZW50fSBpbnB1dCAtIFRoZSBpbnB1dCBldmVudCBmaXJlcyAgb24gdGhlIGF1dG9jb21wbGV0ZSdzIHRyaWdnZXIgd2hlbiB0aGUgdmFsdWUgaGFzIGJlZW4gY2hhbmdlZCBhcyBhIGRpcmVjdCByZXN1bHQgb2YgYSB1c2VyIGFjdGlvbi5cbiAqL1xuZXhwb3J0IGFic3RyYWN0IGNsYXNzIFNiYkF1dG9jb21wbGV0ZUJhc2VFbGVtZW50PFQgPSBzdHJpbmc+IGV4dGVuZHMgU2JiTmVnYXRpdmVNaXhpbihcbiAgU2JiT3BlbkNsb3NlQmFzZUVsZW1lbnQsXG4pIHtcbiAgcHVibGljIHN0YXRpYyBvdmVycmlkZSBzdHlsZXM6IENTU1Jlc3VsdEdyb3VwID0gW1xuICAgIHNjcm9sbGJhclN0eWxlcyxcbiAgICBvcHRpb25QYW5lbFN0eWxlcyxcbiAgICB1bnNhZmVDU1Moc3R5bGUpLFxuICBdO1xuXG4gIC8qKlxuICAgKiBUaGUgZWxlbWVudCB3aGVyZSB0aGUgYXV0b2NvbXBsZXRlIHdpbGwgYXR0YWNoLlxuICAgKiBJZiBub3Qgc2V0LCBhcyBmYWxsYmFjayB0aGVyZSBhcmUgdHdvIGVsZW1lbnRzIHdoaWNoIGNhbiBhY3QgYXMgb3JpZ2luIHdpdGggZm9sbG93aW5nIHByaW9yaXR5IG9yZGVyOlxuICAgKiAxLiBgc2JiLWZvcm0tZmllbGRgIGlmIGl0IGlzIGFuIGFuY2VzdG9yLlxuICAgKiAyLiB0cmlnZ2VyIGVsZW1lbnQgaWYgc2V0LlxuICAgKlxuICAgKiBGb3IgYXR0cmlidXRlIHVzYWdlLCBwcm92aWRlIGFuIGlkIHJlZmVyZW5jZS5cbiAgICovXG4gIEBpZFJlZmVyZW5jZSgpXG4gIEBwcm9wZXJ0eSgpXG4gIHB1YmxpYyBhY2Nlc3NvciBvcmlnaW46IEhUTUxFbGVtZW50IHwgbnVsbCA9IG51bGw7XG5cbiAgLyoqXG4gICAqIFRoZSBpbnB1dCBlbGVtZW50IHRoYXQgd2lsbCB0cmlnZ2VyIHRoZSBhdXRvY29tcGxldGUgb3BlbmluZy5cbiAgICogQnkgZGVmYXVsdCwgdGhlIGF1dG9jb21wbGV0ZSB3aWxsIG9wZW4gb24gZm9jdXMsIGNsaWNrLCBpbnB1dCBvciBgQXJyb3dEb3duYCBrZXlwcmVzcyBvZiB0aGUgJ3RyaWdnZXInIGVsZW1lbnQuXG4gICAqIElmIG5vdCBzZXQsIHdpbGwgc2VhcmNoIGZvciB0aGUgZmlyc3QgJ2lucHV0JyBjaGlsZCBvZiBhICdzYmItZm9ybS1maWVsZCcgYW5jZXN0b3IuXG4gICAqXG4gICAqIEZvciBhdHRyaWJ1dGUgdXNhZ2UsIHByb3ZpZGUgYW4gaWQgcmVmZXJlbmNlLlxuICAgKi9cbiAgQGlkUmVmZXJlbmNlKClcbiAgQHByb3BlcnR5KClcbiAgcHVibGljIGFjY2Vzc29yIHRyaWdnZXI6IEhUTUxJbnB1dEVsZW1lbnQgfCBudWxsID0gbnVsbDtcblxuICAvKiogV2hldGhlciB0aGUgaWNvbiBzcGFjZSBpcyBwcmVzZXJ2ZWQgd2hlbiBubyBpY29uIGlzIHNldC4gKi9cbiAgQGZvcmNlVHlwZSgpXG4gIEBwcm9wZXJ0eSh7IGF0dHJpYnV0ZTogJ3ByZXNlcnZlLWljb24tc3BhY2UnLCByZWZsZWN0OiB0cnVlLCB0eXBlOiBCb29sZWFuIH0pXG4gIHB1YmxpYyBhY2Nlc3NvciBwcmVzZXJ2ZUljb25TcGFjZTogYm9vbGVhbiA9IGZhbHNlO1xuXG4gIC8qKiBXaGV0aGVyIHRoZSBmaXJzdCBvcHRpb24gaXMgYXV0b21hdGljYWxseSBhY3RpdmF0ZWQgd2hlbiB0aGUgYXV0b2NvbXBsZXRlIGlzIG9wZW5lZC4gKi9cbiAgQGZvcmNlVHlwZSgpXG4gIEBwcm9wZXJ0eSh7IGF0dHJpYnV0ZTogJ2F1dG8tYWN0aXZlLWZpcnN0LW9wdGlvbicsIHR5cGU6IEJvb2xlYW4gfSlcbiAgcHVibGljIGFjY2Vzc29yIGF1dG9BY3RpdmVGaXJzdE9wdGlvbjogYm9vbGVhbiA9IGZhbHNlO1xuXG4gIC8qKiBGdW5jdGlvbiB0aGF0IG1hcHMgYW4gb3B0aW9uJ3MgY29udHJvbCB2YWx1ZSB0byBpdHMgZGlzcGxheSB2YWx1ZSBpbiB0aGUgdHJpZ2dlci4gKi9cbiAgQHByb3BlcnR5KHsgYXR0cmlidXRlOiBmYWxzZSB9KVxuICBwdWJsaWMgYWNjZXNzb3IgZGlzcGxheVdpdGg6ICgodmFsdWU6IFQpID0+IHN0cmluZykgfCBudWxsID0gbnVsbDtcblxuICAvKipcbiAgICogU2l6ZSB2YXJpYW50LCBlaXRoZXIgbSBvciBzLlxuICAgKiBAZGVmYXVsdCAnbScgLyAncycgKGxlYW4pXG4gICAqL1xuICBAcHJvcGVydHkoeyByZWZsZWN0OiB0cnVlIH0pIHB1YmxpYyBhY2Nlc3NvciBzaXplOiAnbScgfCAncycgPSBpc0xlYW4oKSA/ICdzJyA6ICdtJztcblxuICAvKiogV2hldGhlciB0aGUgYWN0aXZlIG9wdGlvbiBzaG91bGQgYmUgc2VsZWN0ZWQgYXMgdGhlIHVzZXIgaXMgbmF2aWdhdGluZy4gKi9cbiAgQGZvcmNlVHlwZSgpXG4gIEBwcm9wZXJ0eSh7IGF0dHJpYnV0ZTogJ2F1dG8tc2VsZWN0LWFjdGl2ZS1vcHRpb24nLCB0eXBlOiBCb29sZWFuIH0pXG4gIHB1YmxpYyBhY2Nlc3NvciBhdXRvU2VsZWN0QWN0aXZlT3B0aW9uOiBib29sZWFuID0gZmFsc2U7XG5cbiAgLyoqXG4gICAqIFdoZW4gZW5hYmxlZCwgdGhlIGFjdGl2ZSBvcHRpb24gaXMgYXV0b21hdGljYWxseSBzZWxlY3RlZCBvbiBibHVyLlxuICAgKiBUaGlzIGlzIGFuIGV4cGVyaW1lbnRhbCBmZWF0dXJlLiBJdCBtaWdodCBiZSBzdWJqZWN0IHRvIGNoYW5nZXMuXG4gICAqL1xuICBAZm9yY2VUeXBlKClcbiAgQHByb3BlcnR5KHsgYXR0cmlidXRlOiAnYXV0by1zZWxlY3QtYWN0aXZlLW9wdGlvbi1vbi1ibHVyJywgdHlwZTogQm9vbGVhbiB9KVxuICBwdWJsaWMgYWNjZXNzb3IgYXV0b1NlbGVjdEFjdGl2ZU9wdGlvbk9uQmx1cjogYm9vbGVhbiA9IGZhbHNlO1xuXG4gIC8qKlxuICAgKiBXaGV0aGVyIHRoZSB1c2VyIGlzIHJlcXVpcmVkIHRvIG1ha2UgYSBzZWxlY3Rpb24gd2hlbiB0aGV5J3JlIGludGVyYWN0aW5nIHdpdGggdGhlXG4gICAqIGF1dG9jb21wbGV0ZS4gSWYgdGhlIHVzZXIgbW92ZXMgYXdheSBmcm9tIHRoZSBhdXRvY29tcGxldGUgd2l0aG91dCBzZWxlY3RpbmcgYW4gb3B0aW9uIGZyb21cbiAgICogdGhlIGxpc3QsIHRoZSB2YWx1ZSB3aWxsIGJlIHJlc2V0LiBJZiB0aGUgdXNlciBvcGVucyB0aGUgcGFuZWwgYW5kIGNsb3NlcyBpdCB3aXRob3V0XG4gICAqIGludGVyYWN0aW5nIG9yIHNlbGVjdGluZyBhIHZhbHVlLCB0aGUgaW5pdGlhbCB2YWx1ZSB3aWxsIGJlIGtlcHQuXG4gICAqL1xuICBAZm9yY2VUeXBlKClcbiAgQHByb3BlcnR5KHsgYXR0cmlidXRlOiAncmVxdWlyZS1zZWxlY3Rpb24nLCB0eXBlOiBCb29sZWFuIH0pXG4gIHB1YmxpYyBhY2Nlc3NvciByZXF1aXJlU2VsZWN0aW9uOiBib29sZWFuID0gZmFsc2U7XG5cbiAgLyoqXG4gICAqIFRoZSBwb3NpdGlvbiBvZiB0aGUgYXV0b2NvbXBsZXRlIHBhbmVsIHJlbGF0aXZlIHRvIHRoZSB0cmlnZ2VyLlxuICAgKiBAZGVmYXVsdCAnYXV0bydcbiAgICovXG4gIEBwcm9wZXJ0eSgpXG4gIHB1YmxpYyBhY2Nlc3NvciBwb3NpdGlvbjogJ2F1dG8nIHwgJ2Fib3ZlJyB8ICdiZWxvdycgPSAnYXV0byc7XG5cbiAgLyoqIFJldHVybnMgdGhlIGVsZW1lbnQgd2hlcmUgdGhlIGF1dG9jb21wbGV0ZSBvdmVybGF5IGlzIGF0dGFjaGVkIHRvLiAqL1xuICBwdWJsaWMgZ2V0IG9yaWdpbkVsZW1lbnQoKTogSFRNTEVsZW1lbnQgfCBudWxsIHtcbiAgICByZXR1cm4gKFxuICAgICAgdGhpcy5vcmlnaW4gPz9cbiAgICAgIHRoaXMuY2xvc2VzdD8uKCdzYmItZm9ybS1maWVsZCcpPy5zaGFkb3dSb290Py5xdWVyeVNlbGVjdG9yPy4oJyNvdmVybGF5LWFuY2hvcicpID8/XG4gICAgICB0aGlzLnRyaWdnZXIgPz9cbiAgICAgIG51bGxcbiAgICApO1xuICB9XG5cbiAgLyoqIFJldHVybnMgdGhlIHRyaWdnZXIgZWxlbWVudC4gKi9cbiAgcHVibGljIGdldCB0cmlnZ2VyRWxlbWVudCgpOiBIVE1MSW5wdXRFbGVtZW50IHwgbnVsbCB7XG4gICAgcmV0dXJuIHRoaXMuX3RyaWdnZXJFbGVtZW50ID8/IG51bGw7XG4gIH1cbiAgcHJpdmF0ZSBfdHJpZ2dlckVsZW1lbnQ/OiBIVE1MSW5wdXRFbGVtZW50IHwgbnVsbDtcblxuICBwcm90ZWN0ZWQgYWJzdHJhY3Qgb3ZlcmxheUlkOiBzdHJpbmc7XG4gIHByb3RlY3RlZCBhYnN0cmFjdCBwYW5lbFJvbGU6IHN0cmluZztcbiAgcHJvdGVjdGVkIGFjdGl2ZU9wdGlvbjogU2JiT3B0aW9uQmFzZUVsZW1lbnQ8VD4gfCBudWxsID0gbnVsbDtcbiAgcHJvdGVjdGVkIHBlbmRpbmdBdXRvU2VsZWN0ZWRPcHRpb246IFNiYk9wdGlvbkJhc2VFbGVtZW50PFQ+IHwgbnVsbCA9IG51bGw7XG4gIHByaXZhdGUgX3Jlc2l6ZU9ic2VydmVyID0gbmV3IFJlc2l6ZUNvbnRyb2xsZXIodGhpcywge1xuICAgIHRhcmdldDogbnVsbCxcbiAgICBza2lwSW5pdGlhbDogdHJ1ZSxcbiAgICAvLyBUaGlzIGlzIGFuIElJRkUsIGJlY2F1c2Ugd2UgbmVlZCB0byBrZWVwIHRyYWNrIG9mIHRoZSB0aW1lb3V0IHN0YXRlXG4gICAgLy8gZm9yIGRlYm91bmNpbmcgdGhlIHJlc2l6ZSBjYWxsYmFja3MuXG4gICAgY2FsbGJhY2s6ICgoKSA9PiB7XG4gICAgICBsZXQgdGltZW91dElkOiBSZXR1cm5UeXBlPHR5cGVvZiBzZXRUaW1lb3V0PjtcbiAgICAgIHJldHVybiAoKSA9PiB7XG4gICAgICAgIGNsZWFyVGltZW91dCh0aW1lb3V0SWQpO1xuICAgICAgICB0aW1lb3V0SWQgPSBzZXRUaW1lb3V0KCgpID0+IHtcbiAgICAgICAgICBpZiAodGhpcy5zdGF0ZSAhPT0gJ2Nsb3NlZCcpIHtcbiAgICAgICAgICAgIHRoaXMuX3NldE92ZXJsYXlQb3NpdGlvbigpO1xuICAgICAgICAgIH1cbiAgICAgICAgfSwgMTApO1xuICAgICAgfTtcbiAgICB9KSgpLFxuICB9KTtcbiAgLyoqIExpc3RlbnMgdG8gdGhlIGNoYW5nZXMgb24gdGhlIGBkaXNhYmxlZGAgb3IgYHJlYWRvbmx5YCBhdHRyaWJ1dGUgb2YgdGhlIHRyaWdnZXIuICovXG4gIHByaXZhdGUgX3RyaWdnZXJBdHRyaWJ1dGVPYnNlcnZlciA9ICFpc1NlcnZlclxuICAgID8gbmV3IE11dGF0aW9uT2JzZXJ2ZXIoKG11dGF0aW9uczogTXV0YXRpb25SZWNvcmRbXSk6IHZvaWQgPT4ge1xuICAgICAgICBjb25zdCBpbnB1dCA9IG11dGF0aW9uc1swXS50YXJnZXQgYXMgSFRNTElucHV0RWxlbWVudDtcbiAgICAgICAgaWYgKGlucHV0Lmhhc0F0dHJpYnV0ZSgnZGlzYWJsZWQnKSB8fCBpbnB1dC5oYXNBdHRyaWJ1dGUoJ3JlYWRvbmx5JykpIHtcbiAgICAgICAgICB0aGlzLmNsb3NlKCk7XG4gICAgICAgIH1cbiAgICAgIH0pXG4gICAgOiBudWxsO1xuICBwcml2YXRlIF9vdmVybGF5ITogSFRNTEVsZW1lbnQ7XG4gIHByaXZhdGUgX3RyaWdnZXJBYm9ydENvbnRyb2xsZXIhOiBBYm9ydENvbnRyb2xsZXI7XG4gIHByaXZhdGUgX29wZW5QYW5lbEV2ZW50c0NvbnRyb2xsZXIhOiBBYm9ydENvbnRyb2xsZXI7XG4gIHByaXZhdGUgX2lzUG9pbnRlckRvd25FdmVudE9uTWVudTogYm9vbGVhbiA9IGZhbHNlO1xuICBwcml2YXRlIF9lc2NhcGFibGVPdmVybGF5Q29udHJvbGxlciA9IG5ldyBTYmJFc2NhcGFibGVPdmVybGF5Q29udHJvbGxlcih0aGlzKTtcbiAgcHJpdmF0ZSBfb3B0aW9uc0NvdW50ID0gMDtcbiAgcHJpdmF0ZSBfcHJldmlvdXNFbGVtZW50cz86IEVsZW1lbnRbXTtcblxuICAvKiogVHJhY2tzIGlucHV0IGZyb20ga2V5Ym9hcmQuICovXG4gIHByaXZhdGUgX2xhc3RVc2VySW5wdXQ6IHN0cmluZyB8IG51bGwgPSBudWxsO1xuXG4gIC8qKiBJZiB0cnVlLCB0aGUgJ2NoYW5nZScgZXZlbnQgc291cmNlIGlzIGFuIG9wdGlvbiBzZWxlY3Rpb24gb3IgYSAncmVxdWlyZVNlbGVjdGlvbicgZmllbGQgY2xlYW51cCAqL1xuICBwcml2YXRlIF9pc0N1c3RvbUNoYW5nZUV2ZW50ID0gZmFsc2U7XG5cbiAgcHJvdGVjdGVkIGFic3RyYWN0IGdldCBvcHRpb25zKCk6IFNiYk9wdGlvbkJhc2VFbGVtZW50PFQ+W107XG5cbiAgcHVibGljIGNvbnN0cnVjdG9yKCkge1xuICAgIHN1cGVyKCk7XG4gICAgdGhpcy5hZGRFdmVudExpc3RlbmVyPy4oJ29wdGlvbnNlbGVjdGVkJywgKGU6IEV2ZW50KSA9PiB0aGlzLm9uT3B0aW9uU2VsZWN0ZWQoZSkpO1xuICAgIHRoaXMuYWRkRXZlbnRMaXN0ZW5lcj8uKCfJtW9wdGdyb3Vwc2xvdGNoYW5nZScsICgpID0+IHRoaXMuX2hhbmRsZVNsb3RjaGFuZ2UoKSwge1xuICAgICAgY2FwdHVyZTogdHJ1ZSxcbiAgICB9KTtcbiAgICB0aGlzLmFkZENvbnRyb2xsZXIoXG4gICAgICBuZXcgU2JiUHJvcGVydHlXYXRjaGVyQ29udHJvbGxlcihcbiAgICAgICAgdGhpcyxcbiAgICAgICAgKCkgPT4gdGhpcy5jbG9zZXN0PFNiYkZvcm1GaWVsZEVsZW1lbnQ+KCdzYmItZm9ybS1maWVsZCcpLFxuICAgICAgICB7XG4gICAgICAgICAgbmVnYXRpdmU6IChlKSA9PiB7XG4gICAgICAgICAgICB0aGlzLm5lZ2F0aXZlID0gZS5uZWdhdGl2ZTtcbiAgICAgICAgICAgIHRoaXMuc3luY05lZ2F0aXZlKCk7XG4gICAgICAgICAgfSxcbiAgICAgICAgICBib3JkZXJsZXNzOiAoZSkgPT4gdGhpcy50b2dnbGVTdGF0ZSgnb3B0aW9uLXBhbmVsLW9yaWdpbi1ib3JkZXJsZXNzJywgZS5ib3JkZXJsZXNzKSxcbiAgICAgICAgfSxcbiAgICAgICksXG4gICAgKTtcbiAgfVxuXG4gIHByb3RlY3RlZCBhYnN0cmFjdCBzeW5jTmVnYXRpdmUoKTogdm9pZDtcbiAgcHJvdGVjdGVkIGFic3RyYWN0IHNldFRyaWdnZXJBdHRyaWJ1dGVzKGVsZW1lbnQ6IEhUTUxJbnB1dEVsZW1lbnQpOiB2b2lkO1xuICBwcm90ZWN0ZWQgYWJzdHJhY3Qgb3BlbmVkUGFuZWxLZXlib2FyZEludGVyYWN0aW9uKGV2ZW50OiBLZXlib2FyZEV2ZW50KTogdm9pZDtcbiAgcHJvdGVjdGVkIGFic3RyYWN0IHNlbGVjdEJ5S2V5Ym9hcmQoZXZlbnQ6IEtleWJvYXJkRXZlbnQpOiB2b2lkO1xuICBwcm90ZWN0ZWQgYWJzdHJhY3Qgc2V0TmV4dEFjdGl2ZU9wdGlvbihldmVudD86IEtleWJvYXJkRXZlbnQpOiB2b2lkO1xuICBwcm90ZWN0ZWQgYWJzdHJhY3QgcmVzZXRBY3RpdmVFbGVtZW50KCk6IHZvaWQ7XG5cbiAgLyoqIE9wZW5zIHRoZSBhdXRvY29tcGxldGUuICovXG4gIHB1YmxpYyBvcGVuKCk6IHZvaWQge1xuICAgIGlmIChcbiAgICAgIHRoaXMuc3RhdGUgPT09ICdvcGVuaW5nJyB8fFxuICAgICAgdGhpcy5zdGF0ZSA9PT0gJ29wZW5lZCcgfHxcbiAgICAgICF0aGlzLl9vdmVybGF5IHx8XG4gICAgICB0aGlzLm9wdGlvbnMubGVuZ3RoID09PSAwIHx8XG4gICAgICB0aGlzLl9yZWFkb25seSgpIHx8XG4gICAgICAhdGhpcy5kaXNwYXRjaEJlZm9yZU9wZW5FdmVudCgpXG4gICAgKSB7XG4gICAgICByZXR1cm47XG4gICAgfVxuXG4gICAgdGhpcy5zaG93UG9wb3Zlcj8uKCk7XG4gICAgdGhpcy5zdGF0ZSA9ICdvcGVuaW5nJztcbiAgICB0aGlzLnRyaWdnZXJFbGVtZW50Py50b2dnbGVBdHRyaWJ1dGUoJ2RhdGEtZXhwYW5kZWQnLCB0cnVlKTtcbiAgICBjb25zdCBvcmlnaW5FbGVtZW50ID0gdGhpcy5vcmlnaW5FbGVtZW50O1xuICAgIGlmICghb3JpZ2luRWxlbWVudCkge1xuICAgICAgdGhyb3cgbmV3IEVycm9yKFxuICAgICAgICAnQ2Fubm90IGZpbmQgdGhlIG9yaWdpbiBlbGVtZW50LiBQbGVhc2Ugc3BlY2lmeSBhIHZhbGlkIGVsZW1lbnQgb3IgY2hlY2sgdGhlIHVzYWdlIG9mIHRoZSBcIm9yaWdpblwiIHByb3BlcnR5IGZyb20gdGhlIGRvY3VtZW50YXRpb24nLFxuICAgICAgKTtcbiAgICB9XG4gICAgdGhpcy5fc2V0T3ZlcmxheVBvc2l0aW9uKG9yaWdpbkVsZW1lbnQpO1xuICAgIHRoaXMuX3NldE5leHRBY3RpdmVPcHRpb25JZkF1dG9BY3RpdmVGaXJzdE9wdGlvbigpO1xuICAgIHRoaXMuX2F0dGFjaE9wZW5QYW5lbEV2ZW50cygpO1xuICAgIHRoaXMuX2VzY2FwYWJsZU92ZXJsYXlDb250cm9sbGVyLmNvbm5lY3QoKTtcblxuICAgIC8vIElmIHRoZSBhbmltYXRpb24gZHVyYXRpb24gaXMgemVybywgdGhlIGFuaW1hdGlvbmVuZCBldmVudCBpcyBub3QgYWx3YXlzIGZpcmVkIHJlbGlhYmx5LlxuICAgIC8vIEluIHRoaXMgY2FzZSB3ZSBkaXJlY3RseSBzZXQgdGhlIGBvcGVuZWRgIHN0YXRlLlxuICAgIGlmICh0aGlzLl9pc1plcm9BbmltYXRpb25EdXJhdGlvbigpKSB7XG4gICAgICB0aGlzLl9oYW5kbGVPcGVuaW5nKCk7XG4gICAgfVxuICB9XG5cbiAgLyoqIENsb3NlcyB0aGUgYXV0b2NvbXBsZXRlLiAqL1xuICBwdWJsaWMgY2xvc2UoKTogdm9pZCB7XG4gICAgaWYgKHRoaXMuc3RhdGUgPT09ICdjbG9zaW5nJyB8fCB0aGlzLnN0YXRlID09PSAnY2xvc2VkJyB8fCAhdGhpcy5kaXNwYXRjaEJlZm9yZUNsb3NlRXZlbnQoKSkge1xuICAgICAgcmV0dXJuO1xuICAgIH1cblxuICAgIC8vIEEgJ3BlbmRpbmcgc2VsZWN0aW9uJyBpcyBjb25maXJtZWQgb24gcGFuZWwgY2xvc2VcbiAgICBpZiAodGhpcy5wZW5kaW5nQXV0b1NlbGVjdGVkT3B0aW9uKSB7XG4gICAgICB0aGlzLnBlbmRpbmdBdXRvU2VsZWN0ZWRPcHRpb24uc2VsZWN0ZWQgPSB0cnVlO1xuICAgICAgdGhpcy5fc2V0VmFsdWVBbmREaXNwYXRjaEV2ZW50cyh0aGlzLnBlbmRpbmdBdXRvU2VsZWN0ZWRPcHRpb24sIHRydWUpO1xuICAgIH1cblxuICAgIHRoaXMuc3RhdGUgPSAnY2xvc2luZyc7XG4gICAgdGhpcy50cmlnZ2VyRWxlbWVudD8ucmVtb3ZlQXR0cmlidXRlKCdkYXRhLWV4cGFuZGVkJyk7XG4gICAgdGhpcy5fb3BlblBhbmVsRXZlbnRzQ29udHJvbGxlci5hYm9ydCgpO1xuICAgIGlmICh0aGlzLm9yaWdpbkVsZW1lbnQpIHtcbiAgICAgIHRoaXMuX3Jlc2l6ZU9ic2VydmVyLnVub2JzZXJ2ZSh0aGlzLm9yaWdpbkVsZW1lbnQpO1xuICAgIH1cblxuICAgIC8vIElmIHRoZSBhbmltYXRpb24gZHVyYXRpb24gaXMgemVybywgdGhlIGFuaW1hdGlvbmVuZCBldmVudCBpcyBub3QgYWx3YXlzIGZpcmVkIHJlbGlhYmx5LlxuICAgIC8vIEluIHRoaXMgY2FzZSB3ZSBkaXJlY3RseSBzZXQgdGhlIGBjbG9zZWRgIHN0YXRlLlxuICAgIGlmICh0aGlzLl9pc1plcm9BbmltYXRpb25EdXJhdGlvbigpKSB7XG4gICAgICB0aGlzLl9oYW5kbGVDbG9zaW5nKCk7XG4gICAgfVxuICB9XG5cbiAgcHJpdmF0ZSBfaXNaZXJvQW5pbWF0aW9uRHVyYXRpb24oKTogYm9vbGVhbiB7XG4gICAgcmV0dXJuIGlzWmVyb0FuaW1hdGlvbkR1cmF0aW9uKHRoaXMsICctLXNiYi1vcHRpb25zLXBhbmVsLWFuaW1hdGlvbi1kdXJhdGlvbicpO1xuICB9XG5cbiAgcHVibGljIG92ZXJyaWRlIGNvbm5lY3RlZENhbGxiYWNrKCk6IHZvaWQge1xuICAgIHRoaXMucG9wb3ZlciA9ICdtYW51YWwnO1xuICAgIHN1cGVyLmNvbm5lY3RlZENhbGxiYWNrKCk7XG4gICAgaWYgKGFyaWFSb2xlT25Ib3N0KSB7XG4gICAgICB0aGlzLmlkIHx8PSB0aGlzLm92ZXJsYXlJZDtcbiAgICB9XG5cbiAgICBpZiAodGhpcy5oYXNVcGRhdGVkKSB7XG4gICAgICB0aGlzLl9jb21wb25lbnRTZXR1cCgpO1xuICAgIH1cbiAgICB0aGlzLnN5bmNOZWdhdGl2ZSgpO1xuICB9XG5cbiAgcHJvdGVjdGVkIG92ZXJyaWRlIHdpbGxVcGRhdGUoY2hhbmdlZFByb3BlcnRpZXM6IFByb3BlcnR5VmFsdWVzPHRoaXM+KTogdm9pZCB7XG4gICAgc3VwZXIud2lsbFVwZGF0ZShjaGFuZ2VkUHJvcGVydGllcyk7XG5cbiAgICBpZiAoY2hhbmdlZFByb3BlcnRpZXMuaGFzKCduZWdhdGl2ZScpKSB7XG4gICAgICB0aGlzLnN5bmNOZWdhdGl2ZSgpO1xuICAgIH1cblxuICAgIGlmIChjaGFuZ2VkUHJvcGVydGllcy5oYXMoJ2F1dG9BY3RpdmVGaXJzdE9wdGlvbicpICYmIHRoaXMuaXNPcGVuKSB7XG4gICAgICB0aGlzLl9zZXROZXh0QWN0aXZlT3B0aW9uSWZBdXRvQWN0aXZlRmlyc3RPcHRpb24oKTtcbiAgICB9XG4gIH1cblxuICBwcm90ZWN0ZWQgb3ZlcnJpZGUgZmlyc3RVcGRhdGVkKGNoYW5nZWRQcm9wZXJ0aWVzOiBQcm9wZXJ0eVZhbHVlczx0aGlzPik6IHZvaWQge1xuICAgIHN1cGVyLmZpcnN0VXBkYXRlZChjaGFuZ2VkUHJvcGVydGllcyk7XG4gICAgdGhpcy5fY29tcG9uZW50U2V0dXAoKTtcbiAgfVxuXG4gIHB1YmxpYyBvdmVycmlkZSBkaXNjb25uZWN0ZWRDYWxsYmFjaygpOiB2b2lkIHtcbiAgICBzdXBlci5kaXNjb25uZWN0ZWRDYWxsYmFjaygpO1xuICAgIHRoaXMuX3RyaWdnZXJFbGVtZW50ID0gbnVsbDtcbiAgICB0aGlzLl90cmlnZ2VyQWJvcnRDb250cm9sbGVyPy5hYm9ydCgpO1xuICAgIHRoaXMuX29wZW5QYW5lbEV2ZW50c0NvbnRyb2xsZXI/LmFib3J0KCk7XG4gIH1cblxuICBwdWJsaWMgb3ZlcnJpZGUgcmVxdWVzdFVwZGF0ZShcbiAgICBuYW1lPzogUHJvcGVydHlLZXksXG4gICAgb2xkVmFsdWU/OiB1bmtub3duLFxuICAgIG9wdGlvbnM/OiBQcm9wZXJ0eURlY2xhcmF0aW9uLFxuICApOiB2b2lkIHtcbiAgICBzdXBlci5yZXF1ZXN0VXBkYXRlKG5hbWUsIG9sZFZhbHVlLCBvcHRpb25zKTtcblxuICAgIGlmIChpc1NlcnZlciB8fCAhdGhpcy5oYXNVcGRhdGVkKSB7XG4gICAgICByZXR1cm47XG4gICAgfVxuICAgIGlmICghbmFtZSB8fCBuYW1lID09PSAndHJpZ2dlcicpIHtcbiAgICAgIHRoaXMuX2NvbmZpZ3VyZVRyaWdnZXIoKTtcbiAgICB9IGVsc2UgaWYgKCghbmFtZSB8fCBuYW1lID09PSAnb3JpZ2luJykgJiYgdGhpcy5pc09wZW4pIHtcbiAgICAgIHRoaXMuX3NldE92ZXJsYXlQb3NpdGlvbigpO1xuICAgIH1cbiAgfVxuXG4gIC8qKiBXaGVuIGFuIG9wdGlvbiBpcyBzZWxlY3RlZCwgdXBkYXRlIHRoZSBpbnB1dCB2YWx1ZSBhbmQgY2xvc2UgdGhlIGF1dG9jb21wbGV0ZS4gKi9cbiAgcHJvdGVjdGVkIG9uT3B0aW9uU2VsZWN0ZWQoZXZlbnQ6IEV2ZW50KTogdm9pZCB7XG4gICAgY29uc3QgdGFyZ2V0ID0gZXZlbnQudGFyZ2V0IGFzIFNiYk9wdGlvbkJhc2VFbGVtZW50PFQ+O1xuICAgIHRoaXMuX3NldFZhbHVlQW5kRGlzcGF0Y2hFdmVudHModGFyZ2V0KTtcbiAgICB0aGlzLmNsb3NlKCk7XG4gIH1cblxuICAvKiogQGRlcHJlY2F0ZWQgKi9cbiAgcHJvdGVjdGVkIG9uT3B0aW9uQXJyb3dzU2VsZWN0ZWQoX2FjdGl2ZU9wdGlvbjogU2JiT3B0aW9uQmFzZUVsZW1lbnQ8VD4pOiB2b2lkIHt9XG5cbiAgLyoqXG4gICAqIEEgJ3BlbmRpbmcgc2VsZWN0aW9uJyBzZXRzIHRoZSBvcHRpb24gdmFsdWUgaW4gdGhlIGlucHV0IGVsZW1lbnQgd2l0aG91dCBlbWl0dGluZyBldmVudHMuXG4gICAqIEEgJ3BlbmRpbmcgc2VsZWN0aW9uJyBpcyBjb25maXJtZWQgd2hlbiB0aGUgcGFuZWwgY2xvc2VzLiBBbnkgb3RoZXIgdXNlciBpbnRlcmFjdGlvblxuICAgKiB3aWxsIHJlc2V0IHRoZSBwZW5kaW5nIHZhbHVlLlxuICAgKi9cbiAgcHJvdGVjdGVkIHNldFBlbmRpbmdTZWxlY3Rpb24oYWN0aXZlT3B0aW9uOiBTYmJPcHRpb25CYXNlRWxlbWVudDxUPik6IHZvaWQge1xuICAgIHRoaXMucGVuZGluZ0F1dG9TZWxlY3RlZE9wdGlvbiA9IGFjdGl2ZU9wdGlvbjtcbiAgICB0aGlzLl9zZXRJbnB1dFZhbHVlKGFjdGl2ZU9wdGlvbik7XG4gIH1cblxuICBwcml2YXRlIF9zZXRWYWx1ZUFuZERpc3BhdGNoRXZlbnRzKFxuICAgIHNlbGVjdGVkT3B0aW9uOiBTYmJPcHRpb25CYXNlRWxlbWVudDxUPixcbiAgICBwcmV2ZW50Rm9jdXMgPSBmYWxzZSxcbiAgKTogdm9pZCB7XG4gICAgLy8gRGVzZWxlY3QgdGhlIHByZXZpb3VzIG9wdGlvbnNcbiAgICB0aGlzLm9wdGlvbnNcbiAgICAgIC5maWx0ZXIoKG9wdGlvbikgPT4gb3B0aW9uLmlkICE9PSBzZWxlY3RlZE9wdGlvbi5pZCAmJiBvcHRpb24uc2VsZWN0ZWQpXG4gICAgICAuZm9yRWFjaCgob3B0aW9uKSA9PiAob3B0aW9uLnNlbGVjdGVkID0gZmFsc2UpKTtcbiAgICB0aGlzLnBlbmRpbmdBdXRvU2VsZWN0ZWRPcHRpb24gPSBudWxsO1xuXG4gICAgaWYgKHRoaXMudHJpZ2dlckVsZW1lbnQpIHtcbiAgICAgIHRoaXMuX3NldElucHV0VmFsdWUoc2VsZWN0ZWRPcHRpb24pO1xuXG4gICAgICAvLyBNYW51YWxseSB0cmlnZ2VyIHRoZSBjaGFuZ2UgZXZlbnRzXG4gICAgICB0aGlzLl9pc0N1c3RvbUNoYW5nZUV2ZW50ID0gdHJ1ZTtcbiAgICAgIHRoaXMudHJpZ2dlckVsZW1lbnQuZGlzcGF0Y2hFdmVudChuZXcgRXZlbnQoJ2NoYW5nZScsIHsgYnViYmxlczogdHJ1ZSB9KSk7XG4gICAgICB0aGlzLnRyaWdnZXJFbGVtZW50LmRpc3BhdGNoRXZlbnQobmV3IElucHV0RXZlbnQoJ2lucHV0JywgeyBidWJibGVzOiB0cnVlLCBjb21wb3NlZDogdHJ1ZSB9KSk7XG5cbiAgICAgIC8vIERpc3BhdGNoaW5nIHRoZSBpbnB1dCBldmVudCBjaGFuZ2VzIF9sYXN0SW5wdXQsIHdoaWNoIHNob3VsZCBzdGF5IGVtcHR5IHNpbmNlIHRoZXJlJ3Mgbm8ga2V5Ym9hcmQgaW50ZXJhY3Rpb24uXG4gICAgICB0aGlzLl9sYXN0VXNlcklucHV0ID0gbnVsbDtcblxuICAgICAgLy8gQ3VzdG9tIGlucHV0IGV2ZW50IGVtaXR0ZWQgd2hlbiBpbnB1dCB2YWx1ZSBjaGFuZ2VzIGFmdGVyIGFuIG9wdGlvbiBpcyBzZWxlY3RlZFxuICAgICAgdGhpcy50cmlnZ2VyRWxlbWVudC5kaXNwYXRjaEV2ZW50KFxuICAgICAgICBuZXcgQ3VzdG9tRXZlbnQ8eyBvcHRpb246IFNiYk9wdGlvbkJhc2VFbGVtZW50PFQ+IH0+KCdpbnB1dEF1dG9jb21wbGV0ZScsIHtcbiAgICAgICAgICBkZXRhaWw6IHsgb3B0aW9uOiBzZWxlY3RlZE9wdGlvbiB9LFxuICAgICAgICB9KSxcbiAgICAgICk7XG4gICAgICBpZiAoIXByZXZlbnRGb2N1cykge1xuICAgICAgICB0aGlzLnRyaWdnZXJFbGVtZW50LmZvY3VzKCk7XG4gICAgICB9XG4gICAgfVxuICB9XG5cbiAgLyoqIFNldCB0aGUgb3B0aW9uIHZhbHVlIHdpdGhpbiB0aGUgaW5wdXQgZWxlbWVudCAqL1xuICBwcml2YXRlIF9zZXRJbnB1dFZhbHVlKG9wdGlvbjogU2JiT3B0aW9uQmFzZUVsZW1lbnQ8VD4pOiB2b2lkIHtcbiAgICBpZiAoIXRoaXMudHJpZ2dlckVsZW1lbnQpIHtcbiAgICAgIHJldHVybjtcbiAgICB9XG5cbiAgICAvLyBHaXZlbiBhIHZhbHVlLCByZXR1cm5zIHRoZSBzdHJpbmcgdGhhdCBzaG91bGQgYmUgc2hvd24gd2l0aGluIHRoZSBpbnB1dC5cbiAgICBjb25zdCB0b0Rpc3BsYXkgPSB0aGlzLmRpc3BsYXlXaXRoPy4ob3B0aW9uLnZhbHVlIGFzIFQpID8/IG9wdGlvbi52YWx1ZTtcblxuICAgIC8vIFNldCB0aGUgb3B0aW9uIHZhbHVlXG4gICAgLy8gSW4gb3JkZXIgdG8gc3VwcG9ydCBSZWFjdCBvbkNoYW5nZSBldmVudCwgd2UgaGF2ZSB0byBnZXQgdGhlIHNldHRlciBhbmQgY2FsbCBpdC5cbiAgICAvLyBodHRwczovL2dpdGh1Yi5jb20vZmFjZWJvb2svcmVhY3QvaXNzdWVzLzExNjAwI2lzc3VlY29tbWVudC0zNDU4MTMxMzBcbiAgICBjb25zdCBzZXRWYWx1ZSA9IE9iamVjdC5nZXRPd25Qcm9wZXJ0eURlc2NyaXB0b3IoSFRNTElucHV0RWxlbWVudC5wcm90b3R5cGUsICd2YWx1ZScpIS5zZXQhO1xuICAgIHNldFZhbHVlLmNhbGwodGhpcy50cmlnZ2VyRWxlbWVudCwgdG9EaXNwbGF5KTtcbiAgfVxuXG4gIHByaXZhdGUgX2hhbmRsZVNsb3RjaGFuZ2UoKTogdm9pZCB7XG4gICAgdGhpcy5faGlnaGxpZ2h0T3B0aW9ucyh0aGlzLnRyaWdnZXJFbGVtZW50Py52YWx1ZSk7XG5cbiAgICAvLyBJdCBpcyBwb3NzaWJsZSB0aGF0IGFuIGVsZW1lbnQgaXMgYWRkZWQgdGhhdCBoYXMgbm90IGJlZW4gcmVuZGVyZWRcbiAgICAvLyB5ZXQgYW5kIHRoZXJlZm9yZSBoYXMgaGVpZ2h0IDAuIFRoZXJlZm9yZSwgd2UgYWxzbyBvYnNlcnZlIHRoZSBzaXplXG4gICAgLy8gb2YgYWxsIGNoaWxkIGVsZW1lbnRzLlxuICAgIGNvbnN0IGN1cnJlbnRFbGVtZW50cyA9IEFycmF5LmZyb20odGhpcy5xdWVyeVNlbGVjdG9yQWxsKCcqJykpO1xuICAgIHRoaXMuX3ByZXZpb3VzRWxlbWVudHM/LmZvckVhY2goKGUpID0+IHRoaXMuX3Jlc2l6ZU9ic2VydmVyLnVub2JzZXJ2ZShlKSk7XG4gICAgdGhpcy5fcHJldmlvdXNFbGVtZW50cyA9IGN1cnJlbnRFbGVtZW50cztcbiAgICB0aGlzLl9wcmV2aW91c0VsZW1lbnRzLmZvckVhY2goKGUpID0+IHRoaXMuX3Jlc2l6ZU9ic2VydmVyLm9ic2VydmUoZSkpO1xuXG4gICAgLyoqXG4gICAgICogSXQncyBwb3NzaWJsZSB0byBmaWx0ZXIgb3V0IG9wdGlvbnMgd2l0aCBhbiBvcGVuZWQgcGFuZWwgb24gaW5wdXQgY2hhbmdlLlxuICAgICAqIEluIHRoaXMgY2FzZSwgdGhlIHBhbmVsJ3MgcG9zaXRpb24gbXVzdCBiZSByZWNhbGN1bGF0ZWQgY29uc2lkZXJpbmcgdGhlIG5ldyBvcHRpb24ncyBsaXN0LlxuICAgICAqL1xuICAgIGlmICh0aGlzLmlzT3Blbikge1xuICAgICAgdGhpcy5fc2V0T3ZlcmxheVBvc2l0aW9uKCk7XG4gICAgICB0aGlzLl9zZXROZXh0QWN0aXZlT3B0aW9uSWZBdXRvQWN0aXZlRmlyc3RPcHRpb24oKTtcblxuICAgICAgLy8gSWYgdGhlIGF1dG9jb21wbGV0ZSBpcyBvcGVuIGFuZCB0aGUgb3B0aW9uIGNvdW50IGdldHMgdG8gemVybywgd2UgY2xvc2UgdGhlIGF1dG9jb21wbGV0ZS5cbiAgICAgIGlmICh0aGlzLl9vcHRpb25zQ291bnQgPiAwICYmIHRoaXMub3B0aW9ucy5sZW5ndGggPT09IDApIHtcbiAgICAgICAgdGhpcy5jbG9zZSgpO1xuICAgICAgfVxuICAgIH0gZWxzZSBpZiAoXG4gICAgICAvLyBJZiB0aGUgJ2lucHV0JyBpcyBmb2N1c2VkIGFuZCB0aGUgY291bnQgb2Ygb3B0aW9ucyBjaGFuZ2VzIGZyb20gMCB0byA+IDAsXG4gICAgICAvLyB0aGUgYXV0b2NvbXBsZXRlIHNob3VsZCBvcGVuIGF1dG9tYXRpY2FsbHkuXG4gICAgICBkb2N1bWVudD8uYWN0aXZlRWxlbWVudCA9PT0gdGhpcy50cmlnZ2VyRWxlbWVudCAmJlxuICAgICAgdGhpcy5fb3B0aW9uc0NvdW50ID09PSAwICYmXG4gICAgICB0aGlzLm9wdGlvbnMubGVuZ3RoID4gMFxuICAgICkge1xuICAgICAgdGhpcy5vcGVuKCk7XG4gICAgfVxuXG4gICAgdGhpcy5fb3B0aW9uc0NvdW50ID0gdGhpcy5vcHRpb25zLmxlbmd0aDtcbiAgfVxuXG4gIHByaXZhdGUgX3NldE5leHRBY3RpdmVPcHRpb25JZkF1dG9BY3RpdmVGaXJzdE9wdGlvbigpOiB2b2lkIHtcbiAgICBpZiAodGhpcy5hdXRvQWN0aXZlRmlyc3RPcHRpb24pIHtcbiAgICAgIHRoaXMucmVzZXRBY3RpdmVFbGVtZW50KCk7XG4gICAgICB0aGlzLnNldE5leHRBY3RpdmVPcHRpb24oKTtcbiAgICB9XG4gIH1cblxuICAvKiogVGhlIGF1dG9jb21wbGV0ZSBzaG91bGQgaW5oZXJpdCAncmVhZG9ubHknIHN0YXRlIGZyb20gdGhlIHRyaWdnZXIuICovXG4gIHByaXZhdGUgX3JlYWRvbmx5KCk6IGJvb2xlYW4ge1xuICAgIHJldHVybiB0aGlzLnRyaWdnZXJFbGVtZW50Py5oYXNBdHRyaWJ1dGUoJ3JlYWRvbmx5JykgPz8gZmFsc2U7XG4gIH1cblxuICBwcml2YXRlIF9jb21wb25lbnRTZXR1cCgpOiB2b2lkIHtcbiAgICBpZiAoaXNTZXJ2ZXIpIHtcbiAgICAgIHJldHVybjtcbiAgICB9XG5cbiAgICB0aGlzLl9jb25maWd1cmVUcmlnZ2VyKCk7XG4gIH1cblxuICBwcml2YXRlIF9jb25maWd1cmVUcmlnZ2VyKCk6IHZvaWQge1xuICAgIGNvbnN0IHRyaWdnZXJFbGVtZW50ID0gKHRoaXMudHJpZ2dlciA/P1xuICAgICAgdGhpcy5jbG9zZXN0Py4oJ3NiYi1mb3JtLWZpZWxkJyk/LnF1ZXJ5U2VsZWN0b3IoJ2lucHV0JykpIGFzIEhUTUxJbnB1dEVsZW1lbnQgfCBudWxsO1xuICAgIGlmICh0cmlnZ2VyRWxlbWVudCA9PT0gdGhpcy50cmlnZ2VyRWxlbWVudCkge1xuICAgICAgcmV0dXJuO1xuICAgIH1cblxuICAgIHRoaXMuX3RyaWdnZXJBYm9ydENvbnRyb2xsZXI/LmFib3J0KCk7XG4gICAgcmVtb3ZlQXJpYUNvbWJvQm94QXR0cmlidXRlcyh0aGlzLnRyaWdnZXJFbGVtZW50KTtcbiAgICB0aGlzLnRyaWdnZXJFbGVtZW50Py5yZW1vdmVBdHRyaWJ1dGUoJ2RhdGEtZXhwYW5kZWQnKTtcbiAgICB0aGlzLl90cmlnZ2VyRWxlbWVudCA9IHRyaWdnZXJFbGVtZW50O1xuXG4gICAgaWYgKCF0aGlzLnRyaWdnZXJFbGVtZW50KSB7XG4gICAgICByZXR1cm47XG4gICAgfVxuXG4gICAgLy8gQXMgdGhlIHRyaWdnZXIgY2FuIGJlIHRoZSBmYWxsYmFjayBvZiB0aGUgb3JpZ2luLCB3ZSBldmVudHVhbGx5IGhhdmUgdG8gdXBkYXRlIHRoZSBwb3NpdGlvbi5cbiAgICBjb25zdCBvcmlnaW5FbGVtZW50ID0gdGhpcy5vcmlnaW5FbGVtZW50O1xuICAgIGlmICh0aGlzLnRyaWdnZXJFbGVtZW50ID09PSBvcmlnaW5FbGVtZW50ICYmIHRoaXMuaXNPcGVuKSB7XG4gICAgICB0aGlzLl9zZXRPdmVybGF5UG9zaXRpb24ob3JpZ2luRWxlbWVudCk7XG4gICAgfVxuXG4gICAgdGhpcy5fdHJpZ2dlckF0dHJpYnV0ZU9ic2VydmVyPy5vYnNlcnZlKHRoaXMudHJpZ2dlckVsZW1lbnQhLCB7XG4gICAgICBhdHRyaWJ1dGVzOiB0cnVlLFxuICAgICAgYXR0cmlidXRlRmlsdGVyOiBbJ2Rpc2FibGVkJywgJ3JlYWRvbmx5J10sXG4gICAgfSk7XG5cbiAgICB0aGlzLnNldFRyaWdnZXJBdHRyaWJ1dGVzKHRoaXMudHJpZ2dlckVsZW1lbnQpO1xuICAgIHRoaXMuX3RyaWdnZXJBYm9ydENvbnRyb2xsZXIgPSBuZXcgQWJvcnRDb250cm9sbGVyKCk7XG5cbiAgICAvLyBPcGVuIHRoZSBvdmVybGF5IG9uIGZvY3VzLCBjbGljaywgaW5wdXQgYW5kIGBBcnJvd0Rvd25gIGV2ZW50XG4gICAgdGhpcy50cmlnZ2VyRWxlbWVudC5hZGRFdmVudExpc3RlbmVyKCdmb2N1cycsICgpID0+IHRoaXMub3BlbigpLCB7XG4gICAgICBzaWduYWw6IHRoaXMuX3RyaWdnZXJBYm9ydENvbnRyb2xsZXIuc2lnbmFsLFxuICAgIH0pO1xuICAgIHRoaXMudHJpZ2dlckVsZW1lbnQuYWRkRXZlbnRMaXN0ZW5lcignY2xpY2snLCAoKSA9PiB0aGlzLm9wZW4oKSwge1xuICAgICAgc2lnbmFsOiB0aGlzLl90cmlnZ2VyQWJvcnRDb250cm9sbGVyLnNpZ25hbCxcbiAgICB9KTtcbiAgICB0aGlzLnRyaWdnZXJFbGVtZW50LmFkZEV2ZW50TGlzdGVuZXIoXG4gICAgICAnaW5wdXQnLFxuICAgICAgKGV2ZW50KSA9PiB7XG4gICAgICAgIGNvbnN0IHZhbHVlOiBzdHJpbmcgPSAoZXZlbnQudGFyZ2V0IGFzIEhUTUxJbnB1dEVsZW1lbnQpLnZhbHVlO1xuXG4gICAgICAgIC8vIERvIG5vdCBvcGVuIGlmIHRoZSBldmVudCBpcyB0cmlnZ2VyZWQgdmlhIGRpc3BhdGNoRXZlbnQgKGUuZy4gY2xpY2sgb24gdGltZXRhYmxlLXN3YXAtYnV0dG9uKVxuICAgICAgICBpZiAodmFsdWUgJiYgZXZlbnQuaXNUcnVzdGVkKSB7XG4gICAgICAgICAgdGhpcy5vcGVuKCk7XG4gICAgICAgIH1cbiAgICAgICAgdGhpcy5faGlnaGxpZ2h0T3B0aW9ucyh2YWx1ZSk7XG4gICAgICAgIHRoaXMuX2xhc3RVc2VySW5wdXQgPSB2YWx1ZTtcbiAgICAgICAgdGhpcy5wZW5kaW5nQXV0b1NlbGVjdGVkT3B0aW9uID0gbnVsbDtcbiAgICAgIH0sXG4gICAgICB7IHNpZ25hbDogdGhpcy5fdHJpZ2dlckFib3J0Q29udHJvbGxlci5zaWduYWwgfSxcbiAgICApO1xuICAgIHRoaXMudHJpZ2dlckVsZW1lbnQuYWRkRXZlbnRMaXN0ZW5lcihcbiAgICAgICdjaGFuZ2UnLFxuICAgICAgKGV2ZW50KSA9PiB7XG4gICAgICAgIC8qKlxuICAgICAgICAgKiBJbiAncmVxdWlyZVNlbGVjdGlvbicgbW9kZSwgd2UgYmxvY2sgdGhlIG5hdGl2ZSBjaGFuZ2UgZXZlbnRzIGFuZFxuICAgICAgICAgKiBsZXQgb25seSBwYXNzIHRoZSBvbmVzIHRoYXQgY29tZSB3aXRoIGEgdmFsaWQgdmFsdWUgKHdoZW4gYW4gb3B0aW9uIGlzIHNlbGVjdGVkKVxuICAgICAgICAgKi9cbiAgICAgICAgaWYgKHRoaXMucmVxdWlyZVNlbGVjdGlvbiAmJiAhdGhpcy5faXNDdXN0b21DaGFuZ2VFdmVudCkge1xuICAgICAgICAgIGV2ZW50LnN0b3BJbW1lZGlhdGVQcm9wYWdhdGlvbigpO1xuICAgICAgICB9XG4gICAgICAgIHRoaXMuX2lzQ3VzdG9tQ2hhbmdlRXZlbnQgPSBmYWxzZTtcbiAgICAgIH0sXG4gICAgICB7IHNpZ25hbDogdGhpcy5fdHJpZ2dlckFib3J0Q29udHJvbGxlci5zaWduYWwsIGNhcHR1cmU6IHRydWUgfSxcbiAgICApO1xuICAgIHRoaXMudHJpZ2dlckVsZW1lbnQuYWRkRXZlbnRMaXN0ZW5lcihcbiAgICAgICdrZXlkb3duJyxcbiAgICAgIChldmVudDogS2V5Ym9hcmRFdmVudCkgPT4gdGhpcy5fY2xvc2VkUGFuZWxLZXlib2FyZEludGVyYWN0aW9uKGV2ZW50KSxcbiAgICAgIHtcbiAgICAgICAgc2lnbmFsOiB0aGlzLl90cmlnZ2VyQWJvcnRDb250cm9sbGVyLnNpZ25hbCxcbiAgICAgICAgLy8gV2UgbmVlZCBrZXkgZXZlbnQgdG8gcnVuIGJlZm9yZSBhbnkgb3RoZXIgc3Vic2NyaXB0aW9uIHRvIGd1YXJhbnRlZSBhIGNvcnJlY3RcbiAgICAgICAgLy8gaW50ZXJhY3Rpb24gd2l0aCBvdGhlciBjb21wb25lbnRzIChuZWNlc3NhcnkgZm9yIHRoZSAnc2JiLWNoaXAtZ3JvdXAnIHVzZSBjYXNlKS5cbiAgICAgICAgY2FwdHVyZTogdHJ1ZSxcbiAgICAgIH0sXG4gICAgKTtcblxuICAgIHRoaXMudHJpZ2dlckVsZW1lbnQuYWRkRXZlbnRMaXN0ZW5lcihcbiAgICAgICdibHVyJyxcbiAgICAgIChlKSA9PiB7XG4gICAgICAgIC8vIElmIHRoZSBuZXcgZm9jdXMgaXMgdGhlIGF1dG9jb21wbGV0ZSBvciBpbnNpZGUgb2YgaXQgdGhlbiBhbiBvcHRpb25cbiAgICAgICAgLy8gd2FzIHNlbGVjdGVkLiBUaGVyZWZvcmUsIHRoZSBmb2N1cyBpcyBzdGlsbCBvbiB0aGUgY29tcG9uZW50LlxuICAgICAgICBpZiAodGhpcy5jb250YWlucyhlLnJlbGF0ZWRUYXJnZXQgYXMgTm9kZSkpIHtcbiAgICAgICAgICByZXR1cm47XG4gICAgICAgIH1cblxuICAgICAgICAvLyBJZiAnYXV0b1NlbGVjdEFjdGl2ZU9wdGlvbk9uQmx1cicgaXMgZW5hYmxlZCwgc2VsZWN0IHRoZSBhY3RpdmUgb3B0aW9uIG9uIGJsdXJcbiAgICAgICAgaWYgKFxuICAgICAgICAgIHRoaXMuYXV0b1NlbGVjdEFjdGl2ZU9wdGlvbk9uQmx1ciAmJlxuICAgICAgICAgIHRoaXMuYWN0aXZlT3B0aW9uICYmXG4gICAgICAgICAgdGhpcy5fbGFzdFVzZXJJbnB1dCAmJlxuICAgICAgICAgIHRoaXMudHJpZ2dlckVsZW1lbnQ/LnZhbHVlXG4gICAgICAgICkge1xuICAgICAgICAgIHRoaXMuYWN0aXZlT3B0aW9uLnNlbGVjdGVkID0gdHJ1ZTtcbiAgICAgICAgICB0aGlzLl9zZXRWYWx1ZUFuZERpc3BhdGNoRXZlbnRzKHRoaXMuYWN0aXZlT3B0aW9uLCB0cnVlKTtcbiAgICAgICAgfVxuXG4gICAgICAgIC8vIENsZWFycyB0aGUgaW5wdXQgaWYgdGhlcmUncyB1c2VyIGludGVyYWN0aW9uIHdpdGhvdXQgc2VsZWN0aW9uIChzZWxlY3Rpb24gY2xlYXJzIGBfbGFzdFVzZXJJbnB1dGApLlxuICAgICAgICBpZiAodGhpcy5yZXF1aXJlU2VsZWN0aW9uICYmIHRoaXMudHJpZ2dlckVsZW1lbnQgJiYgdGhpcy5fbGFzdFVzZXJJbnB1dCAhPSBudWxsKSB7XG4gICAgICAgICAgY29uc3Qgc2V0VmFsdWUgPSBPYmplY3QuZ2V0T3duUHJvcGVydHlEZXNjcmlwdG9yKFxuICAgICAgICAgICAgSFRNTElucHV0RWxlbWVudC5wcm90b3R5cGUsXG4gICAgICAgICAgICAndmFsdWUnLFxuICAgICAgICAgICkhLnNldCE7XG4gICAgICAgICAgc2V0VmFsdWUuY2FsbCh0aGlzLnRyaWdnZXJFbGVtZW50LCAnJyk7XG4gICAgICAgICAgdGhpcy5faGlnaGxpZ2h0T3B0aW9ucygnJyk7XG4gICAgICAgICAgdGhpcy5faXNDdXN0b21DaGFuZ2VFdmVudCA9IHRydWU7XG4gICAgICAgICAgdGhpcy50cmlnZ2VyRWxlbWVudC5kaXNwYXRjaEV2ZW50KG5ldyBFdmVudCgnY2hhbmdlJywgeyBidWJibGVzOiB0cnVlIH0pKTtcbiAgICAgICAgICB0aGlzLnRyaWdnZXJFbGVtZW50LmRpc3BhdGNoRXZlbnQoXG4gICAgICAgICAgICBuZXcgSW5wdXRFdmVudCgnaW5wdXQnLCB7IGJ1YmJsZXM6IHRydWUsIGNvbXBvc2VkOiB0cnVlIH0pLFxuICAgICAgICAgICk7XG4gICAgICAgIH1cblxuICAgICAgICB0aGlzLmNsb3NlKCk7XG4gICAgICB9LFxuICAgICAgeyBzaWduYWw6IHRoaXMuX3RyaWdnZXJBYm9ydENvbnRyb2xsZXIuc2lnbmFsLCBjYXB0dXJlOiB0cnVlIH0sXG4gICAgKTtcbiAgfVxuXG4gIC8vIFNldCBvdmVybGF5IHBvc2l0aW9uLCB3aWR0aCBhbmQgbWF4IGhlaWdodFxuICBwcml2YXRlIF9zZXRPdmVybGF5UG9zaXRpb24ob3JpZ2luRWxlbWVudCA9IHRoaXMub3JpZ2luRWxlbWVudCk6IHZvaWQge1xuICAgIC8vIEFuIHVuZGVmaW5lZCBvcmlnaW5FbGVtZW50IHNob3VsZCBvbmx5IG9jY3VyIGluIHRoZSB1bmxpa2VseSBldmVudFxuICAgIC8vIHRoYXQgdGhlIGF1dG9jb21wbGV0ZSBsb3NlcyBpdHMgb3JpZ2luRWxlbWVudCBhbmQgdHJpZ2dlckVsZW1lbnQgZHVyaW5nIGFuIG9wZW4gc3RhdGUuXG4gICAgaWYgKCFvcmlnaW5FbGVtZW50KSB7XG4gICAgICByZXR1cm47XG4gICAgfVxuICAgIHNldE92ZXJsYXlQb3NpdGlvbihcbiAgICAgIHRoaXMuX292ZXJsYXksXG4gICAgICBvcmlnaW5FbGVtZW50LFxuICAgICAgdGhpcy5fb3ZlcmxheSxcbiAgICAgIHRoaXMuc2hhZG93Um9vdCEucXVlcnlTZWxlY3RvcignLnNiYi1vcHRpb24tcGFuZWxfX292ZXJsYXktY29udGFpbmVyJykhLFxuICAgICAgdGhpcyxcbiAgICAgIHRoaXMucG9zaXRpb24sXG4gICAgKTtcbiAgfVxuXG4gIC8qKlxuICAgKiBPbiBvcGVuL2Nsb3NlIGFuaW1hdGlvbiBlbmQuXG4gICAqIEluIHJhcmUgY2FzZXMgaXQgY2FuIGJlIHRoYXQgdGhlIGFuaW1hdGlvbkVuZCBldmVudCBpcyB0cmlnZ2VyZWQgdHdpY2UuXG4gICAqIFRvIGF2b2lkIGVudGVyaW5nIGEgY29ycnVwdCBzdGF0ZSwgZXhpdCB3aGVuIHN0YXRlIGlzIG5vdCBleHBlY3RlZC5cbiAgICovXG4gIHByaXZhdGUgX29uQW5pbWF0aW9uRW5kKGV2ZW50OiBBbmltYXRpb25FdmVudCk6IHZvaWQge1xuICAgIGlmIChldmVudC5hbmltYXRpb25OYW1lID09PSAnb3BlbicgJiYgdGhpcy5zdGF0ZSA9PT0gJ29wZW5pbmcnKSB7XG4gICAgICB0aGlzLl9oYW5kbGVPcGVuaW5nKCk7XG4gICAgfSBlbHNlIGlmIChldmVudC5hbmltYXRpb25OYW1lID09PSAnY2xvc2UnICYmIHRoaXMuc3RhdGUgPT09ICdjbG9zaW5nJykge1xuICAgICAgdGhpcy5faGFuZGxlQ2xvc2luZygpO1xuICAgIH1cbiAgfVxuXG4gIHByaXZhdGUgX2hhbmRsZU9wZW5pbmcoKTogdm9pZCB7XG4gICAgdGhpcy5zdGF0ZSA9ICdvcGVuZWQnO1xuICAgIGlmICh0aGlzLm9yaWdpbkVsZW1lbnQpIHtcbiAgICAgIHRoaXMuX3Jlc2l6ZU9ic2VydmVyLm9ic2VydmUodGhpcy5vcmlnaW5FbGVtZW50KTtcbiAgICB9XG4gICAgdGhpcy50cmlnZ2VyRWxlbWVudD8uc2V0QXR0cmlidXRlKCdhcmlhLWV4cGFuZGVkJywgJ3RydWUnKTtcbiAgICB0aGlzLmRpc3BhdGNoT3BlbkV2ZW50KCk7XG4gIH1cblxuICBwcml2YXRlIF9oYW5kbGVDbG9zaW5nKCk6IHZvaWQge1xuICAgIHRoaXMuc3RhdGUgPSAnY2xvc2VkJztcbiAgICB0aGlzLmhpZGVQb3BvdmVyPy4oKTtcbiAgICB0aGlzLnRyaWdnZXJFbGVtZW50Py5zZXRBdHRyaWJ1dGUoJ2FyaWEtZXhwYW5kZWQnLCAnZmFsc2UnKTtcblxuICAgIHRoaXMucmVzZXRBY3RpdmVFbGVtZW50KCk7XG4gICAgdGhpcy5fb3ZlcmxheS5zY3JvbGxUb3AgPSAwO1xuICAgIHRoaXMuX2VzY2FwYWJsZU92ZXJsYXlDb250cm9sbGVyLmRpc2Nvbm5lY3QoKTtcbiAgICB0aGlzLmRpc3BhdGNoQ2xvc2VFdmVudCgpO1xuICB9XG5cbiAgcHJpdmF0ZSBfYXR0YWNoT3BlblBhbmVsRXZlbnRzKCk6IHZvaWQge1xuICAgIHRoaXMuX29wZW5QYW5lbEV2ZW50c0NvbnRyb2xsZXIgPSBuZXcgQWJvcnRDb250cm9sbGVyKCk7XG5cbiAgICAvLyBSZWNhbGN1bGF0ZSB0aGUgb3ZlcmxheSBwb3NpdGlvbiBvbiBzY3JvbGwgYW5kIHdpbmRvdyByZXNpemVcbiAgICBkb2N1bWVudC5hZGRFdmVudExpc3RlbmVyKCdzY3JvbGwnLCAoKSA9PiB0aGlzLl9zZXRPdmVybGF5UG9zaXRpb24oKSwge1xuICAgICAgcGFzc2l2ZTogdHJ1ZSxcbiAgICAgIHNpZ25hbDogdGhpcy5fb3BlblBhbmVsRXZlbnRzQ29udHJvbGxlci5zaWduYWwsXG4gICAgICAvLyBXaXRob3V0IGNhcHR1cmUsIG90aGVyIHNjcm9sbCBjb250ZXh0cyB3b3VsZCBub3QgYnViYmxlIHRvIHRoaXMgZXZlbnQgbGlzdGVuZXIuXG4gICAgICAvLyBDYXB0dXJlIGFsbG93cyB1cyB0byByZWFjdCB0byBhbGwgc2Nyb2xsIGNvbnRleHRzIGluIHRoaXMgRE9NLlxuICAgICAgY2FwdHVyZTogdHJ1ZSxcbiAgICB9KTtcbiAgICB3aW5kb3cuYWRkRXZlbnRMaXN0ZW5lcigncmVzaXplJywgKCkgPT4gdGhpcy5fc2V0T3ZlcmxheVBvc2l0aW9uKCksIHtcbiAgICAgIHBhc3NpdmU6IHRydWUsXG4gICAgICBzaWduYWw6IHRoaXMuX29wZW5QYW5lbEV2ZW50c0NvbnRyb2xsZXIuc2lnbmFsLFxuICAgIH0pO1xuXG4gICAgLy8gQ2xvc2UgYXV0b2NvbXBsZXRlIG9uIGJhY2tkcm9wIGNsaWNrXG4gICAgd2luZG93LmFkZEV2ZW50TGlzdGVuZXIoJ3BvaW50ZXJkb3duJywgKGV2KSA9PiB0aGlzLl9wb2ludGVyRG93bkxpc3RlbmVyKGV2KSwge1xuICAgICAgc2lnbmFsOiB0aGlzLl9vcGVuUGFuZWxFdmVudHNDb250cm9sbGVyLnNpZ25hbCxcbiAgICB9KTtcbiAgICB3aW5kb3cuYWRkRXZlbnRMaXN0ZW5lcigncG9pbnRlcnVwJywgKGV2KSA9PiB0aGlzLl9jbG9zZU9uQmFja2Ryb3BDbGljayhldiksIHtcbiAgICAgIHNpZ25hbDogdGhpcy5fb3BlblBhbmVsRXZlbnRzQ29udHJvbGxlci5zaWduYWwsXG4gICAgfSk7XG5cbiAgICB0aGlzLmFkZEV2ZW50TGlzdGVuZXIoXG4gICAgICAnybVkaXNhYmxlZGNoYW5nZScsXG4gICAgICAoKSA9PiB7XG4gICAgICAgIGlmICh0aGlzLmFjdGl2ZU9wdGlvbj8uZGlzYWJsZWQpIHtcbiAgICAgICAgICB0aGlzLnJlc2V0QWN0aXZlRWxlbWVudCgpO1xuICAgICAgICB9XG4gICAgICAgIHRoaXMuX3NldE5leHRBY3RpdmVPcHRpb25JZkF1dG9BY3RpdmVGaXJzdE9wdGlvbigpO1xuICAgICAgfSxcbiAgICAgIHtcbiAgICAgICAgc2lnbmFsOiB0aGlzLl9vcGVuUGFuZWxFdmVudHNDb250cm9sbGVyLnNpZ25hbCxcbiAgICAgIH0sXG4gICAgKTtcblxuICAgIC8vIEtleWJvYXJkIGludGVyYWN0aW9uc1xuICAgIHRoaXMudHJpZ2dlckVsZW1lbnQ/LmFkZEV2ZW50TGlzdGVuZXIoXG4gICAgICAna2V5ZG93bicsXG4gICAgICAoZXZlbnQ6IEtleWJvYXJkRXZlbnQpID0+IHRoaXMub3BlbmVkUGFuZWxLZXlib2FyZEludGVyYWN0aW9uKGV2ZW50KSxcbiAgICAgIHtcbiAgICAgICAgc2lnbmFsOiB0aGlzLl9vcGVuUGFuZWxFdmVudHNDb250cm9sbGVyLnNpZ25hbCxcbiAgICAgICAgLy8gV2UgbmVlZCBrZXkgZXZlbnQgdG8gcnVuIGJlZm9yZSBhbnkgb3RoZXIgc3Vic2NyaXB0aW9uIHRvIGd1YXJhbnRlZSBhIGNvcnJlY3RcbiAgICAgICAgLy8gaW50ZXJhY3Rpb24gd2l0aCBvdGhlciBjb21wb25lbnRzIChuZWNlc3NhcnkgZm9yIHRoZSAnc2JiLWNoaXAtZ3JvdXAnIHVzZSBjYXNlKS5cbiAgICAgICAgY2FwdHVyZTogdHJ1ZSxcbiAgICAgIH0sXG4gICAgKTtcbiAgfVxuXG4gIC8vIENoZWNrIGlmIHRoZSBwb2ludGVyZG93biBldmVudCB0YXJnZXQgaXMgdHJpZ2dlcmVkIG9uIHRoZSBtZW51LlxuICBwcml2YXRlIF9wb2ludGVyRG93bkxpc3RlbmVyID0gKGV2ZW50OiBQb2ludGVyRXZlbnQpOiB2b2lkID0+IHtcbiAgICB0aGlzLl9pc1BvaW50ZXJEb3duRXZlbnRPbk1lbnUgPSBpc0V2ZW50T25FbGVtZW50KHRoaXMuX292ZXJsYXksIGV2ZW50KTtcbiAgfTtcblxuICAvLyBJZiB0aGUgY2xpY2sgaXMgb3V0c2lkZSB0aGUgYXV0b2NvbXBsZXRlLCBjbG9zZXMgdGhlIHBhbmVsLlxuICBwcml2YXRlIF9jbG9zZU9uQmFja2Ryb3BDbGljayA9IChldmVudDogUG9pbnRlckV2ZW50KTogdm9pZCA9PiB7XG4gICAgaWYgKFxuICAgICAgIXRoaXMuX2lzUG9pbnRlckRvd25FdmVudE9uTWVudSAmJlxuICAgICAgIWlzRXZlbnRPbkVsZW1lbnQodGhpcy5fb3ZlcmxheSwgZXZlbnQpICYmXG4gICAgICAhaXNFdmVudE9uRWxlbWVudCh0aGlzLm9yaWdpbkVsZW1lbnQsIGV2ZW50KVxuICAgICkge1xuICAgICAgdGhpcy5jbG9zZSgpO1xuICAgIH1cbiAgfTtcblxuICBwcml2YXRlIF9jbG9zZWRQYW5lbEtleWJvYXJkSW50ZXJhY3Rpb24oZXZlbnQ6IEtleWJvYXJkRXZlbnQpOiB2b2lkIHtcbiAgICBpZiAodGhpcy5zdGF0ZSA9PT0gJ29wZW5pbmcnIHx8IHRoaXMuc3RhdGUgPT09ICdvcGVuZWQnKSB7XG4gICAgICByZXR1cm47XG4gICAgfVxuXG4gICAgc3dpdGNoIChldmVudC5rZXkpIHtcbiAgICAgIGNhc2UgJ0Fycm93RG93bic6XG4gICAgICBjYXNlICdBcnJvd1VwJzpcbiAgICAgICAgdGhpcy5vcGVuKCk7XG4gICAgICAgIGJyZWFrO1xuICAgIH1cbiAgfVxuXG4gIC8qKiBIaWdobGlnaHQgdGhlIHNlYXJjaGVkIHRleHQgb24gdGhlIG9wdGlvbnMuICovXG4gIHByaXZhdGUgX2hpZ2hsaWdodE9wdGlvbnMoc2VhcmNoVGVybT86IHN0cmluZyk6IHZvaWQge1xuICAgIGlmIChzZWFyY2hUZXJtID09PSBudWxsIHx8IHNlYXJjaFRlcm0gPT09IHVuZGVmaW5lZCkge1xuICAgICAgcmV0dXJuO1xuICAgIH1cbiAgICB0aGlzLm9wdGlvbnMuZm9yRWFjaCgob3B0aW9uKSA9PiBvcHRpb24uaGlnaGxpZ2h0KHNlYXJjaFRlcm0pKTtcbiAgfVxuXG4gIHByb3RlY3RlZCBvdmVycmlkZSByZW5kZXIoKTogVGVtcGxhdGVSZXN1bHQge1xuICAgIC8vIFNjcm9sbCBhcmVhcyB3aXRob3V0IGNvbnRhaW5pbmcgYW4gaW50ZXJhY3RpdmUgZWxlbWVudCB3aWxsIHJlY2VpdmUgZm9jdXMgd2hlbiB0YWJiaW5nIHRocm91Z2ggdGhlIGRvY3VtZW50LlxuICAgIC8vIElmIHRoZXJlIGFyZSBhIGxvdCBvZiBvcHRpb25zIGFuZCB3aGVuIHByZXNzaW5nIHRhYiBrZXksIHRoZSBzY3JvbGwgYXJlYSBvbiBzYmItYXV0b2NvbXBsZXRlX19vcHRpb25zIGdldHMgZm9jdXMuXG4gICAgLy8gQXMgZWxlbWVudHMgaW5zaWRlIHRoZSBwYW5lbCBzaG91bGQgbmV2ZXIgZ2V0IGZvY3VzLCB3ZSBoYXZlIHRvIGF2b2lkIHRoYXQgYnkgc2V0dGluZyB0YWJpbmRleD0tMS5cbiAgICByZXR1cm4gaHRtbGBcbiAgICAgIDxkaXYgY2xhc3M9XCJzYmItb3B0aW9uLXBhbmVsX19vdmVybGF5LWNvbnRhaW5lclwiPlxuICAgICAgICA8ZGl2XG4gICAgICAgICAgY2xhc3M9XCJzYmItb3B0aW9uLXBhbmVsX19vdmVybGF5ICR7dGhpcy5uZWdhdGl2ZVxuICAgICAgICAgICAgPyAnc2JiLXNjcm9sbGJhci1uZWdhdGl2ZSdcbiAgICAgICAgICAgIDogJ3NiYi1zY3JvbGxiYXInfVwiXG4gICAgICAgICAgcm9sZT0keyFhcmlhUm9sZU9uSG9zdCA/IHRoaXMucGFuZWxSb2xlIDogbm90aGluZ31cbiAgICAgICAgICBpZD0keyFhcmlhUm9sZU9uSG9zdCA/IHRoaXMub3ZlcmxheUlkIDogbm90aGluZ31cbiAgICAgICAgICB0YWJpbmRleD1cIi0xXCJcbiAgICAgICAgICBAYW5pbWF0aW9uZW5kPSR7dGhpcy5fb25BbmltYXRpb25FbmR9XG4gICAgICAgICAgJHtyZWYoKG92ZXJsYXlSZWY/OiBFbGVtZW50KSA9PiAodGhpcy5fb3ZlcmxheSA9IG92ZXJsYXlSZWYgYXMgSFRNTEVsZW1lbnQpKX1cbiAgICAgICAgPlxuICAgICAgICAgIDxzbG90IEBzbG90Y2hhbmdlPSR7dGhpcy5faGFuZGxlU2xvdGNoYW5nZX0+PC9zbG90PlxuICAgICAgICA8L2Rpdj5cbiAgICAgIDwvZGl2PlxuICAgIGA7XG4gIH1cbn1cblxuZGVjbGFyZSBnbG9iYWwge1xuICBpbnRlcmZhY2UgSFRNTEVsZW1lbnRFdmVudE1hcCB7XG4gICAgaW5wdXRBdXRvY29tcGxldGU6IEN1c3RvbUV2ZW50PHsgb3B0aW9uOiBTYmJPcHRpb25CYXNlRWxlbWVudDxhbnk+IH0+O1xuICB9XG59XG4iXSwibWFwcGluZ3MiOiI7Ozs7Ozs7Ozs7Ozs7O0FDdUNBLElBQU0saUJBQWlCOzs7Ozs7O0lBUUQsb0NBQTBCO21CQUFxQixpQkFDbkUsd0JBQ0Q7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Y0FGcUIsbUNBQStDLFlBRXBFOzs7eUJBZUUsYUFBYSxFQUNiLFVBQVUsQ0FBQTswQkFVVixhQUFhLEVBQ2IsVUFBVSxDQUFBO29DQUlWLFdBQVcsRUFDWCxTQUFTO0lBQUUsV0FBVztJQUF1QixTQUFTO0lBQU0sTUFBTTtJQUFTLENBQUMsQ0FBQTt3Q0FJNUUsV0FBVyxFQUNYLFNBQVM7SUFBRSxXQUFXO0lBQTRCLE1BQU07SUFBUyxDQUFDLENBQUE7OEJBSWxFLFNBQVMsRUFBRSxXQUFXLE9BQU8sQ0FBQyxDQUFBO3VCQU85QixTQUFTLEVBQUUsU0FBUyxNQUFNLENBQUMsQ0FBQTt5Q0FHM0IsV0FBVyxFQUNYLFNBQVM7SUFBRSxXQUFXO0lBQTZCLE1BQU07SUFBUyxDQUFDLENBQUE7K0NBT25FLFdBQVcsRUFDWCxTQUFTO0lBQUUsV0FBVztJQUFxQyxNQUFNO0lBQVMsQ0FBQyxDQUFBO21DQVMzRSxXQUFXLEVBQ1gsU0FBUztJQUFFLFdBQVc7SUFBcUIsTUFBTTtJQUFTLENBQUMsQ0FBQTsyQkFPM0QsVUFBVSxDQUFBO0FBNURYLGdCQUFBLE1BQUEsTUFBQSxvQkFBQTtJQUFBLE1BQUE7SUFBQSxNQUFBO0lBQUEsUUFBQTtJQUFBLFNBQUE7SUFBQSxRQUFBO0tBQUEsTUFBQSxRQUFBLFlBQUE7S0FBQSxNQUFBLFFBQUEsSUFBZ0I7S0FBTSxNQUFBLEtBQUEsVUFBQTtBQUFBLFVBQU4sU0FBTTs7S0FBQTtJQUFBLFVBQUE7SUFBQSxFQUFBLHNCQUFBLDBCQUFBO0FBV3RCLGdCQUFBLE1BQUEsTUFBQSxxQkFBQTtJQUFBLE1BQUE7SUFBQSxNQUFBO0lBQUEsUUFBQTtJQUFBLFNBQUE7SUFB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