@sbb-esta/lyne-elements-dev 4.9.0-dev.1774875578 → 4.9.0-dev.1774887348

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.
@@ -0,0 +1,1678 @@
1
+ import { __esDecorate, __runInitializers } from "tslib";
2
+ import { css, html, isServer, nothing } from "lit";
3
+ import { property, state } from "lit/decorators.js";
4
+ import { SbbElement } from "./core/base-elements.js";
5
+ import { forceType, handleDistinctChange, plainDate } from "./core/decorators.js";
6
+ import { boxSizingStyles } from "./core/styles.js";
7
+ import { SbbSecondaryButtonElement } from "./button.pure.js";
8
+ import { SbbLanguageController, SbbMediaMatcherController, SbbMediaQueryBreakpointLargeAndAbove } from "./core/controllers.js";
9
+ import { i18nCalendarDateSelection, i18nCalendarWeekNumber, i18nNextMonth, i18nNextYear, i18nNextYearRange, i18nPreviousMonth, i18nPreviousYear, i18nPreviousYearRange, i18nYearMonthSelection } from "./core/i18n.js";
10
+ import "./icon.js";
11
+ import { isArrowKeyOrPageKeysPressed } from "./core/a11y.js";
12
+ import { readConfig } from "./core/config.js";
13
+ import { DAYS_PER_ROW, MONDAY, MONTHS_PER_PAGE, MONTHS_PER_ROW, THURSDAY, TUESDAY, WEDNESDAY, YEARS_PER_PAGE, YEARS_PER_ROW, defaultDateAdapter } from "./core/datetime.js";
14
+ import { SbbCalendarDayElement } from "./calendar/calendar-day/calendar-day.component.js";
15
+ import { SbbCalendarMonthElement } from "./calendar/calendar-month/calendar-month.component.js";
16
+ import { SbbCalendarWeekdayElement } from "./calendar/calendar-weekday/calendar-weekday.component.js";
17
+ import { SbbCalendarWeeknumberElement } from "./calendar/calendar-weeknumber/calendar-weeknumber.component.js";
18
+ import { SbbCalendarYearElement } from "./calendar/calendar-year/calendar-year.component.js";
19
+ import "./screen-reader-only.js";
20
+ //#region src/elements/calendar/calendar/calendar.scss?lit&inline
21
+ var calendar_default = css`:host {
22
+ display: block;
23
+ width: max-content;
24
+ --sbb-calendar-cell-size: 2.75rem;
25
+ --sbb-calendar-cell-border-radius: calc(
26
+ var(--sbb-border-radius-4x) + var(--sbb-calendar-cell-border-width)
27
+ );
28
+ --sbb-calendar-cell-transition-duration: var(
29
+ --sbb-disable-animation-duration,
30
+ var(--sbb-animation-duration-2x)
31
+ );
32
+ }
33
+ @media (min-width: calc(37.5rem)) {
34
+ :host {
35
+ --sbb-calendar-control-view-change-height: 3rem;
36
+ }
37
+ }
38
+
39
+ :host(:is(:state(enhanced),[state--enhanced])) {
40
+ --sbb-calendar-cell-padding-block-end: var(--sbb-spacing-fixed-3x);
41
+ }
42
+
43
+ .sbb-calendar__wrapper {
44
+ width: 100%;
45
+ display: block;
46
+ transition-duration: var(--sbb-calendar-cell-transition-duration);
47
+ }
48
+
49
+ .sbb-calendar__controls {
50
+ width: 100%;
51
+ display: inline-flex;
52
+ align-items: center;
53
+ gap: var(--sbb-calendar-control-gap);
54
+ margin-block-end: var(--sbb-calendar-control-margin-block-end);
55
+ }
56
+
57
+ .sbb-calendar__controls-month {
58
+ width: 100%;
59
+ display: flex;
60
+ gap: var(--sbb-calendar-tables-gap);
61
+ }
62
+
63
+ #sbb-calendar__controls-previous,
64
+ #sbb-calendar__controls-next {
65
+ -webkit-tap-highlight-color: transparent;
66
+ }
67
+
68
+ .sbb-calendar__controls-change-date {
69
+ appearance: none;
70
+ box-sizing: border-box;
71
+ margin: 0;
72
+ outline: none;
73
+ border: none;
74
+ border-radius: 0;
75
+ padding: 0;
76
+ background: none;
77
+ -webkit-tap-highlight-color: transparent;
78
+ color: inherit;
79
+ display: flex;
80
+ align-items: center;
81
+ margin: auto;
82
+ height: var(--sbb-calendar-control-view-change-height);
83
+ font-size: var(--sbb-text-font-size-s);
84
+ letter-spacing: var(--sbb-typo-letter-spacing-text);
85
+ text-transform: capitalize;
86
+ cursor: var(--sbb-cursor-pointer);
87
+ padding-inline: var(--sbb-calendar-control-view-change-padding-inline);
88
+ border-radius: var(--sbb-border-radius-infinity);
89
+ background-color: var(--sbb-calendar-control-view-change-background);
90
+ color: var(--sbb-calendar-control-view-change-color);
91
+ transition-duration: var(--sbb-calendar-cell-transition-duration);
92
+ transition-timing-function: var(--sbb-calendar-cell-transition-easing-function);
93
+ transition-property: background-color, padding-block-end;
94
+ }
95
+ .sbb-calendar__controls-change-date:disabled {
96
+ --sbb-calendar-control-view-change-background: var(--sbb-color-milk);
97
+ --sbb-calendar-control-view-change-background: light-dark(
98
+ var(--sbb-color-milk),
99
+ var(--sbb-color-anthracite)
100
+ );
101
+ --sbb-calendar-control-view-change-color: var(--sbb-color-granite);
102
+ --sbb-calendar-control-view-change-color: light-dark(
103
+ var(--sbb-color-granite),
104
+ var(--sbb-color-aluminium)
105
+ );
106
+ cursor: unset;
107
+ }
108
+ .sbb-calendar__controls-change-date:focus-visible {
109
+ outline-offset: var(--sbb-focus-outline-offset);
110
+ outline: var(--sbb-focus-outline-color) var(--sbb-focus-outline-style, solid) var(--sbb-focus-outline-width);
111
+ outline-offset: var(--sbb-spacing-fixed-1x);
112
+ }
113
+ @media (any-hover: hover) {
114
+ .sbb-calendar__controls-change-date:not(:active, :disabled):hover {
115
+ padding-block-end: var(--sbb-calendar-cell-hover-shift);
116
+ }
117
+ }
118
+ .sbb-calendar__controls-change-date:not(:disabled):active {
119
+ --sbb-calendar-control-view-change-background: var(--sbb-background-color-3);
120
+ }
121
+
122
+ .sbb-calendar__table-month-view,
123
+ .sbb-calendar__table-year-view {
124
+ --sbb-calendar-table-column-spaces: 6;
125
+ }
126
+
127
+ .sbb-calendar__table-overflow-break {
128
+ contain: layout;
129
+ }
130
+
131
+ .sbb-calendar__table-container {
132
+ display: flex;
133
+ gap: var(--sbb-calendar-tables-gap);
134
+ margin-inline: var(--sbb-calendar-margin);
135
+ --sbb-calendar-min-width: calc(7 * var(--sbb-calendar-cell-size));
136
+ --sbb-calendar-overflow: calc(100% - var(--sbb-calendar-min-width));
137
+ --sbb-calendar-start-offset: min(
138
+ 0px,
139
+ -1 * (var(--sbb-calendar-overflow) / var(--sbb-calendar-table-column-spaces))
140
+ );
141
+ --sbb-calendar-margin: var(--sbb-calendar-start-offset);
142
+ }
143
+ :host(:is(:state(wide),[state--wide])) .sbb-calendar__table-container {
144
+ --sbb-calendar-min-width: calc(
145
+ 2 * 7 * var(--sbb-calendar-cell-size) + var(--sbb-calendar-tables-gap)
146
+ );
147
+ --sbb-calendar-margin: calc(0.5 * var(--sbb-calendar-start-offset));
148
+ }
149
+ :host([orientation=horizontal][week-numbers]) .sbb-calendar__table-container {
150
+ --sbb-calendar-min-width: calc(8 * var(--sbb-calendar-cell-size));
151
+ }
152
+ :host([orientation=horizontal][week-numbers]:is(:state(wide),[state--wide])) .sbb-calendar__table-container {
153
+ --sbb-calendar-min-width: calc(
154
+ 2 * 8 * var(--sbb-calendar-cell-size) + var(--sbb-calendar-tables-gap)
155
+ );
156
+ }
157
+ :host([orientation=vertical]) .sbb-calendar__table-container {
158
+ min-width: var(--sbb-calendar-min-width);
159
+ --sbb-calendar-start-offset: 0;
160
+ }
161
+ :host([orientation=vertical]:is(:state(wide),[state--wide])) .sbb-calendar__table-container {
162
+ --sbb-calendar-min-width: calc(
163
+ 13 * var(--sbb-calendar-cell-size) + var(--sbb-calendar-tables-gap)
164
+ );
165
+ }
166
+
167
+ .sbb-calendar__table {
168
+ width: 100%;
169
+ border-collapse: collapse;
170
+ height: max-content;
171
+ animation-name: show;
172
+ animation-duration: var(--sbb-calendar-table-animation-duration);
173
+ }
174
+ .sbb-calendar__table.sbb-calendar__table-hide {
175
+ --sbb-calendar-cell-transition-duration: 0ms;
176
+ animation-name: hide;
177
+ animation-duration: var(--sbb-calendar-table-animation-duration);
178
+ }
179
+ :host(:not(:is(:state(wide),[state--wide]))) .sbb-calendar__table {
180
+ min-width: 100%;
181
+ }
182
+
183
+ .sbb-calendar__table-header {
184
+ text-align: center;
185
+ }
186
+
187
+ .sbb-calendar__table-body {
188
+ text-align: center;
189
+ }
190
+
191
+ .sbb-calendar__table-header-cell,
192
+ .sbb-calendar__table-header-cell-vertical {
193
+ width: var(--sbb-calendar-cell-size);
194
+ color: var(--sbb-calendar-header-color);
195
+ padding: 0;
196
+ font-size: var(--sbb-text-font-size-xs);
197
+ font-weight: normal;
198
+ letter-spacing: var(--sbb-typo-letter-spacing-text);
199
+ }
200
+ :host(:not([multiple])) .sbb-calendar__table-header-cell,
201
+ :host(:not([multiple])) .sbb-calendar__table-header-cell-vertical {
202
+ height: var(--sbb-calendar-cell-size);
203
+ }
204
+
205
+ .sbb-calendar__table-header-cell-vertical {
206
+ padding-block-end: var(--sbb-calendar-cell-padding-block-end);
207
+ }
208
+
209
+ .sbb-calendar__table-data {
210
+ position: relative;
211
+ padding: 0;
212
+ text-align: center;
213
+ }
214
+
215
+ .sbb-calendar__day-cell {
216
+ padding-block-end: var(--sbb-calendar-cell-padding-block-end);
217
+ }
218
+
219
+ :host(:not(:is(:state(enhanced),[state--enhanced]))) sbb-calendar-day {
220
+ --sbb-calendar-cell-justify-content: center;
221
+ --sbb-calendar-day-height: 2.75rem;
222
+ --sbb-calendar-day-extra-display: none;
223
+ --sbb-calendar-day-value-height: unset;
224
+ --sbb-calendar-day-crossed-out-top: 50%;
225
+ --sbb-calendar-day-crossed-out-translate: translate(-50%, -50%) rotate(-45deg);
226
+ }
227
+
228
+ @keyframes show {
229
+ from {
230
+ opacity: 0;
231
+ transform: translateY(var(--sbb-calendar-table-animation-shift));
232
+ }
233
+ to {
234
+ opacity: 1;
235
+ transform: translateY(0%);
236
+ }
237
+ }
238
+ @keyframes hide {
239
+ from {
240
+ opacity: 1;
241
+ transform: translateY(0%);
242
+ }
243
+ to {
244
+ opacity: 0;
245
+ transform: translateY(var(--sbb-calendar-table-animation-shift));
246
+ }
247
+ }`;
248
+ //#endregion
249
+ //#region src/elements/calendar/calendar/calendar.component.ts
250
+ SbbSecondaryButtonElement.define();
251
+ SbbCalendarDayElement.define();
252
+ SbbCalendarMonthElement.define();
253
+ SbbCalendarYearElement.define();
254
+ SbbCalendarWeekdayElement.define();
255
+ SbbCalendarWeeknumberElement.define();
256
+ var SbbMonthChangeEvent = class extends Event {
257
+ get range() {
258
+ return this._range;
259
+ }
260
+ constructor(range) {
261
+ super("monthchange", {
262
+ bubbles: true,
263
+ composed: true
264
+ });
265
+ this._range = Object.freeze(range || []);
266
+ }
267
+ };
268
+ /**
269
+ * It displays a calendar which allows choosing a date.
270
+ *
271
+ * @slot - Use the unnamed slot to add customized `sbb-calendar-day` elements.
272
+ */
273
+ var SbbCalendarElement = (() => {
274
+ let _classSuper = SbbElement;
275
+ let _instanceExtraInitializers = [];
276
+ let _wide_decorators;
277
+ let _wide_initializers = [];
278
+ let _wide_extraInitializers = [];
279
+ let _view_decorators;
280
+ let _view_initializers = [];
281
+ let _view_extraInitializers = [];
282
+ let _min_decorators;
283
+ let _min_initializers = [];
284
+ let _min_extraInitializers = [];
285
+ let _max_decorators;
286
+ let _max_initializers = [];
287
+ let _max_extraInitializers = [];
288
+ let _multiple_decorators;
289
+ let _multiple_initializers = [];
290
+ let _multiple_extraInitializers = [];
291
+ let _set_selected_decorators;
292
+ let __selected_decorators;
293
+ let __selected_initializers = [];
294
+ let __selected_extraInitializers = [];
295
+ let _dateFilter_decorators;
296
+ let _dateFilter_initializers = [];
297
+ let _dateFilter_extraInitializers = [];
298
+ let _orientation_decorators;
299
+ let _orientation_initializers = [];
300
+ let _orientation_extraInitializers = [];
301
+ let _weekNumbers_decorators;
302
+ let _weekNumbers_initializers = [];
303
+ let _weekNumbers_extraInitializers = [];
304
+ let __activeDate_decorators;
305
+ let __activeDate_initializers = [];
306
+ let __activeDate_extraInitializers = [];
307
+ let _set__wide_decorators;
308
+ let __calendarView_decorators;
309
+ let __calendarView_initializers = [];
310
+ let __calendarView_extraInitializers = [];
311
+ let __initialized_decorators;
312
+ let __initialized_initializers = [];
313
+ let __initialized_extraInitializers = [];
314
+ return class SbbCalendarElement extends _classSuper {
315
+ static {
316
+ const _metadata = typeof Symbol === "function" && Symbol.metadata ? Object.create(_classSuper[Symbol.metadata] ?? null) : void 0;
317
+ _wide_decorators = [forceType(), property({ type: Boolean })];
318
+ _view_decorators = [property()];
319
+ _min_decorators = [plainDate(), property()];
320
+ _max_decorators = [plainDate(), property()];
321
+ _multiple_decorators = [
322
+ forceType(),
323
+ handleDistinctChange((e, newValue) => e._onMultipleChanged(newValue)),
324
+ property({ type: Boolean })
325
+ ];
326
+ _set_selected_decorators = [property()];
327
+ __selected_decorators = [state()];
328
+ _dateFilter_decorators = [property({ attribute: "date-filter" })];
329
+ _orientation_decorators = [property({ reflect: true })];
330
+ _weekNumbers_decorators = [forceType(), property({
331
+ attribute: "week-numbers",
332
+ type: Boolean
333
+ })];
334
+ __activeDate_decorators = [state()];
335
+ _set__wide_decorators = [state()];
336
+ __calendarView_decorators = [state()];
337
+ __initialized_decorators = [state()];
338
+ __esDecorate(this, null, _wide_decorators, {
339
+ kind: "accessor",
340
+ name: "wide",
341
+ static: false,
342
+ private: false,
343
+ access: {
344
+ has: (obj) => "wide" in obj,
345
+ get: (obj) => obj.wide,
346
+ set: (obj, value) => {
347
+ obj.wide = value;
348
+ }
349
+ },
350
+ metadata: _metadata
351
+ }, _wide_initializers, _wide_extraInitializers);
352
+ __esDecorate(this, null, _view_decorators, {
353
+ kind: "accessor",
354
+ name: "view",
355
+ static: false,
356
+ private: false,
357
+ access: {
358
+ has: (obj) => "view" in obj,
359
+ get: (obj) => obj.view,
360
+ set: (obj, value) => {
361
+ obj.view = value;
362
+ }
363
+ },
364
+ metadata: _metadata
365
+ }, _view_initializers, _view_extraInitializers);
366
+ __esDecorate(this, null, _min_decorators, {
367
+ kind: "accessor",
368
+ name: "min",
369
+ static: false,
370
+ private: false,
371
+ access: {
372
+ has: (obj) => "min" in obj,
373
+ get: (obj) => obj.min,
374
+ set: (obj, value) => {
375
+ obj.min = value;
376
+ }
377
+ },
378
+ metadata: _metadata
379
+ }, _min_initializers, _min_extraInitializers);
380
+ __esDecorate(this, null, _max_decorators, {
381
+ kind: "accessor",
382
+ name: "max",
383
+ static: false,
384
+ private: false,
385
+ access: {
386
+ has: (obj) => "max" in obj,
387
+ get: (obj) => obj.max,
388
+ set: (obj, value) => {
389
+ obj.max = value;
390
+ }
391
+ },
392
+ metadata: _metadata
393
+ }, _max_initializers, _max_extraInitializers);
394
+ __esDecorate(this, null, _multiple_decorators, {
395
+ kind: "accessor",
396
+ name: "multiple",
397
+ static: false,
398
+ private: false,
399
+ access: {
400
+ has: (obj) => "multiple" in obj,
401
+ get: (obj) => obj.multiple,
402
+ set: (obj, value) => {
403
+ obj.multiple = value;
404
+ }
405
+ },
406
+ metadata: _metadata
407
+ }, _multiple_initializers, _multiple_extraInitializers);
408
+ __esDecorate(this, null, _set_selected_decorators, {
409
+ kind: "setter",
410
+ name: "selected",
411
+ static: false,
412
+ private: false,
413
+ access: {
414
+ has: (obj) => "selected" in obj,
415
+ set: (obj, value) => {
416
+ obj.selected = value;
417
+ }
418
+ },
419
+ metadata: _metadata
420
+ }, null, _instanceExtraInitializers);
421
+ __esDecorate(this, null, __selected_decorators, {
422
+ kind: "accessor",
423
+ name: "_selected",
424
+ static: false,
425
+ private: false,
426
+ access: {
427
+ has: (obj) => "_selected" in obj,
428
+ get: (obj) => obj._selected,
429
+ set: (obj, value) => {
430
+ obj._selected = value;
431
+ }
432
+ },
433
+ metadata: _metadata
434
+ }, __selected_initializers, __selected_extraInitializers);
435
+ __esDecorate(this, null, _dateFilter_decorators, {
436
+ kind: "accessor",
437
+ name: "dateFilter",
438
+ static: false,
439
+ private: false,
440
+ access: {
441
+ has: (obj) => "dateFilter" in obj,
442
+ get: (obj) => obj.dateFilter,
443
+ set: (obj, value) => {
444
+ obj.dateFilter = value;
445
+ }
446
+ },
447
+ metadata: _metadata
448
+ }, _dateFilter_initializers, _dateFilter_extraInitializers);
449
+ __esDecorate(this, null, _orientation_decorators, {
450
+ kind: "accessor",
451
+ name: "orientation",
452
+ static: false,
453
+ private: false,
454
+ access: {
455
+ has: (obj) => "orientation" in obj,
456
+ get: (obj) => obj.orientation,
457
+ set: (obj, value) => {
458
+ obj.orientation = value;
459
+ }
460
+ },
461
+ metadata: _metadata
462
+ }, _orientation_initializers, _orientation_extraInitializers);
463
+ __esDecorate(this, null, _weekNumbers_decorators, {
464
+ kind: "accessor",
465
+ name: "weekNumbers",
466
+ static: false,
467
+ private: false,
468
+ access: {
469
+ has: (obj) => "weekNumbers" in obj,
470
+ get: (obj) => obj.weekNumbers,
471
+ set: (obj, value) => {
472
+ obj.weekNumbers = value;
473
+ }
474
+ },
475
+ metadata: _metadata
476
+ }, _weekNumbers_initializers, _weekNumbers_extraInitializers);
477
+ __esDecorate(this, null, __activeDate_decorators, {
478
+ kind: "accessor",
479
+ name: "_activeDate",
480
+ static: false,
481
+ private: false,
482
+ access: {
483
+ has: (obj) => "_activeDate" in obj,
484
+ get: (obj) => obj._activeDate,
485
+ set: (obj, value) => {
486
+ obj._activeDate = value;
487
+ }
488
+ },
489
+ metadata: _metadata
490
+ }, __activeDate_initializers, __activeDate_extraInitializers);
491
+ __esDecorate(this, null, _set__wide_decorators, {
492
+ kind: "setter",
493
+ name: "_wide",
494
+ static: false,
495
+ private: false,
496
+ access: {
497
+ has: (obj) => "_wide" in obj,
498
+ set: (obj, value) => {
499
+ obj._wide = value;
500
+ }
501
+ },
502
+ metadata: _metadata
503
+ }, null, _instanceExtraInitializers);
504
+ __esDecorate(this, null, __calendarView_decorators, {
505
+ kind: "accessor",
506
+ name: "_calendarView",
507
+ static: false,
508
+ private: false,
509
+ access: {
510
+ has: (obj) => "_calendarView" in obj,
511
+ get: (obj) => obj._calendarView,
512
+ set: (obj, value) => {
513
+ obj._calendarView = value;
514
+ }
515
+ },
516
+ metadata: _metadata
517
+ }, __calendarView_initializers, __calendarView_extraInitializers);
518
+ __esDecorate(this, null, __initialized_decorators, {
519
+ kind: "accessor",
520
+ name: "_initialized",
521
+ static: false,
522
+ private: false,
523
+ access: {
524
+ has: (obj) => "_initialized" in obj,
525
+ get: (obj) => obj._initialized,
526
+ set: (obj, value) => {
527
+ obj._initialized = value;
528
+ }
529
+ },
530
+ metadata: _metadata
531
+ }, __initialized_initializers, __initialized_extraInitializers);
532
+ if (_metadata) Object.defineProperty(this, Symbol.metadata, {
533
+ enumerable: true,
534
+ configurable: true,
535
+ writable: true,
536
+ value: _metadata
537
+ });
538
+ }
539
+ static {
540
+ this.elementName = "sbb-calendar";
541
+ }
542
+ static {
543
+ this.styles = [boxSizingStyles, calendar_default];
544
+ }
545
+ static {
546
+ this.events = {
547
+ dateselected: "dateselected",
548
+ monthchange: "monthchange"
549
+ };
550
+ }
551
+ #wide_accessor_storage;
552
+ /** If set to true, two months are displayed */
553
+ get wide() {
554
+ return this.#wide_accessor_storage;
555
+ }
556
+ set wide(value) {
557
+ this.#wide_accessor_storage = value;
558
+ }
559
+ #view_accessor_storage;
560
+ /** The initial view of the calendar which should be displayed on opening. */
561
+ get view() {
562
+ return this.#view_accessor_storage;
563
+ }
564
+ set view(value) {
565
+ this.#view_accessor_storage = value;
566
+ }
567
+ #min_accessor_storage;
568
+ /**
569
+ * The minimum valid date. Accepts a date object or null.
570
+ * Accepts an ISO8601 formatted string (e.g. 2024-12-24) as attribute.
571
+ */
572
+ get min() {
573
+ return this.#min_accessor_storage;
574
+ }
575
+ set min(value) {
576
+ this.#min_accessor_storage = value;
577
+ }
578
+ #max_accessor_storage;
579
+ /**
580
+ * The maximum valid date. Accepts a date object or null.
581
+ * Accepts an ISO8601 formatted string (e.g. 2024-12-24) as attribute.
582
+ */
583
+ get max() {
584
+ return this.#max_accessor_storage;
585
+ }
586
+ set max(value) {
587
+ this.#max_accessor_storage = value;
588
+ }
589
+ #multiple_accessor_storage;
590
+ /** Whether the calendar allows for multiple date selection. */
591
+ get multiple() {
592
+ return this.#multiple_accessor_storage;
593
+ }
594
+ set multiple(value) {
595
+ this.#multiple_accessor_storage = value;
596
+ }
597
+ /**
598
+ * The selected date: accepts a date object, or, if `multiple`, an array of dates.
599
+ */
600
+ set selected(value) {
601
+ if (Array.isArray(value)) this._selected = value.map((dateLike) => this._dateAdapter.getValidDateOrNull(this._dateAdapter.deserialize(dateLike))).filter((date) => date !== null).filter((date) => !this._isDayInRange(this._dateAdapter.toIso8601(date)) || this._dateFilter(date));
602
+ else {
603
+ const selectedDate = this._dateAdapter.getValidDateOrNull(this._dateAdapter.deserialize(value));
604
+ if (!!selectedDate && (!this._isDayInRange(this._dateAdapter.toIso8601(selectedDate)) || this._dateFilter(selectedDate))) this._selected = selectedDate;
605
+ else this._selected = null;
606
+ }
607
+ }
608
+ get selected() {
609
+ return this._selected;
610
+ }
611
+ #_selected_accessor_storage;
612
+ get _selected() {
613
+ return this.#_selected_accessor_storage;
614
+ }
615
+ set _selected(value) {
616
+ this.#_selected_accessor_storage = value;
617
+ }
618
+ #dateFilter_accessor_storage;
619
+ /** A function used to filter out dates. */
620
+ get dateFilter() {
621
+ return this.#dateFilter_accessor_storage;
622
+ }
623
+ set dateFilter(value) {
624
+ this.#dateFilter_accessor_storage = value;
625
+ }
626
+ #orientation_accessor_storage;
627
+ /** The orientation of days in the calendar. */
628
+ get orientation() {
629
+ return this.#orientation_accessor_storage;
630
+ }
631
+ set orientation(value) {
632
+ this.#orientation_accessor_storage = value;
633
+ }
634
+ #weekNumbers_accessor_storage;
635
+ /** Whether it has to display the week numbers in addition to week days. */
636
+ get weekNumbers() {
637
+ return this.#weekNumbers_accessor_storage;
638
+ }
639
+ set weekNumbers(value) {
640
+ this.#weekNumbers_accessor_storage = value;
641
+ }
642
+ #_activeDate_accessor_storage;
643
+ /** The currently active date. */
644
+ get _activeDate() {
645
+ return this.#_activeDate_accessor_storage;
646
+ }
647
+ set _activeDate(value) {
648
+ this.#_activeDate_accessor_storage = value;
649
+ }
650
+ /** The current wide property considering property value and breakpoints. From zero to small `wide` has always to be false. */
651
+ set _wide(wide) {
652
+ this.toggleState("wide", wide);
653
+ this._wideInternal = wide;
654
+ }
655
+ get _wide() {
656
+ return this._wideInternal;
657
+ }
658
+ #_calendarView_accessor_storage;
659
+ get _calendarView() {
660
+ return this.#_calendarView_accessor_storage;
661
+ }
662
+ set _calendarView(value) {
663
+ this.#_calendarView_accessor_storage = value;
664
+ }
665
+ /** A list of calendar's cells corresponding to days, months or years depending on the view. */
666
+ get _cells() {
667
+ return Array.from((this._calendarView === "day" ? Array.from(this.shadowRoot.querySelectorAll("slot")).flatMap((e) => e.assignedElements({ flatten: true })) : this.shadowRoot?.querySelectorAll(`sbb-calendar-${this._calendarView}`)) ?? []);
668
+ }
669
+ #_initialized_accessor_storage;
670
+ get _initialized() {
671
+ return this.#_initialized_accessor_storage;
672
+ }
673
+ set _initialized(value) {
674
+ this.#_initialized_accessor_storage = value;
675
+ }
676
+ constructor() {
677
+ super();
678
+ this.#wide_accessor_storage = (__runInitializers(this, _instanceExtraInitializers), __runInitializers(this, _wide_initializers, false));
679
+ this.#view_accessor_storage = (__runInitializers(this, _wide_extraInitializers), __runInitializers(this, _view_initializers, "day"));
680
+ this.#min_accessor_storage = (__runInitializers(this, _view_extraInitializers), __runInitializers(this, _min_initializers, null));
681
+ this.#max_accessor_storage = (__runInitializers(this, _min_extraInitializers), __runInitializers(this, _max_initializers, null));
682
+ this.#multiple_accessor_storage = (__runInitializers(this, _max_extraInitializers), __runInitializers(this, _multiple_initializers, false));
683
+ this.#_selected_accessor_storage = (__runInitializers(this, _multiple_extraInitializers), __runInitializers(this, __selected_initializers, null));
684
+ this.#dateFilter_accessor_storage = (__runInitializers(this, __selected_extraInitializers), __runInitializers(this, _dateFilter_initializers, null));
685
+ this.#orientation_accessor_storage = (__runInitializers(this, _dateFilter_extraInitializers), __runInitializers(this, _orientation_initializers, "horizontal"));
686
+ this.#weekNumbers_accessor_storage = (__runInitializers(this, _orientation_extraInitializers), __runInitializers(this, _weekNumbers_initializers, false));
687
+ this._dateAdapter = (__runInitializers(this, _weekNumbers_extraInitializers), readConfig().datetime?.dateAdapter ?? defaultDateAdapter);
688
+ this.#_activeDate_accessor_storage = __runInitializers(this, __activeDate_initializers, this._dateAdapter.today());
689
+ this._wideInternal = (__runInitializers(this, __activeDate_extraInitializers), false);
690
+ this.#_calendarView_accessor_storage = __runInitializers(this, __calendarView_initializers, "day");
691
+ this._nextCalendarView = (__runInitializers(this, __calendarView_extraInitializers), "day");
692
+ /** Information about the rendered day view; used in keyboard navigation. */
693
+ this._keyboardNavigationDayViewParameters = {
694
+ firstDayInView: null,
695
+ lastDayInView: null,
696
+ firstMonthOffset: 0,
697
+ firstMonthLength: 0,
698
+ secondMonthOffset: 0
699
+ };
700
+ /** Grid of calendar cells representing the dates of the month. */
701
+ this._weeks = [];
702
+ /** An array containing all the month names in the current language. */
703
+ this._monthNames = this._dateAdapter.getMonthNames("long");
704
+ this._enhancedVariant = false;
705
+ /** Whether the focus should be reset on focusCell. */
706
+ this._resetFocus = false;
707
+ /** Whether an element inside the calendar is currently focused. */
708
+ this._containingFocus = false;
709
+ this.#_initialized_accessor_storage = __runInitializers(this, __initialized_initializers, false);
710
+ this._language = (__runInitializers(this, __initialized_extraInitializers), new SbbLanguageController(this).withHandler(() => {
711
+ this._monthNames = this._dateAdapter.getMonthNames("long");
712
+ this._createMonthRows();
713
+ }));
714
+ this._mediaMatcher = new SbbMediaMatcherController(this, { [SbbMediaQueryBreakpointLargeAndAbove]: () => this._init() });
715
+ this._onSlotChange = () => {
716
+ this._enhancedVariant = Array.from(this.children).some((c) => c.localName === "sbb-calendar-day");
717
+ this.toggleState("enhanced", this._enhancedVariant);
718
+ this._setTabIndex();
719
+ };
720
+ this._createMonthRows();
721
+ this._setWeekdays();
722
+ this.addEventListener("focusin", () => this._containingFocus = true);
723
+ this.addEventListener("focusout", () => this._containingFocus = false);
724
+ this.addEventListener("click", (e) => {
725
+ const day = e.target.closest("sbb-calendar-day");
726
+ if (day) this._selectDate(day.value);
727
+ });
728
+ this.addEventListener("keydown", (e) => {
729
+ if (e.target.localName === "sbb-calendar-day") this._handleKeyboardEvent(e, this._mapDateToDay(e.target.value));
730
+ });
731
+ }
732
+ _dateFilter(date) {
733
+ return this.dateFilter?.(date) ?? true;
734
+ }
735
+ /** Resets the active month according to the new state of the calendar. */
736
+ resetPosition() {
737
+ this._resetCalendarView();
738
+ this._init();
739
+ }
740
+ connectedCallback() {
741
+ super.connectedCallback();
742
+ this.resetPosition();
743
+ this.shadowRoot?.addEventListener("slotchange", this._onSlotChange, { capture: true });
744
+ }
745
+ disconnectedCallback() {
746
+ super.disconnectedCallback();
747
+ this.shadowRoot?.removeEventListener("slotchange", this._onSlotChange, { capture: true });
748
+ }
749
+ /** @internal */
750
+ focus() {
751
+ this._resetFocus = true;
752
+ this._focusCell();
753
+ }
754
+ willUpdate(changedProperties) {
755
+ super.willUpdate(changedProperties);
756
+ if (!this._initialized) return;
757
+ if (changedProperties.has("wide") || changedProperties.has("orientation")) this.resetPosition();
758
+ if (changedProperties.has("view")) {
759
+ this._setChosenYear();
760
+ this._chosenMonth = void 0;
761
+ this._nextCalendarView = this._calendarView = this.view;
762
+ }
763
+ }
764
+ updated(changedProperties) {
765
+ super.updated(changedProperties);
766
+ this._setTabIndex();
767
+ this._focusCell();
768
+ }
769
+ /**
770
+ * The `_selected` state should be adapted when the `multiple` property changes:
771
+ * - if it changes to true, the '_selected' is set to an array;
772
+ * - if it changes to false, the first available option is set as 'value' otherwise it's set to null.
773
+ */
774
+ _onMultipleChanged(isMultiple) {
775
+ if (isMultiple && !Array.isArray(this._selected)) this._selected = this._selected ? [this._selected] : [];
776
+ if (!isMultiple && Array.isArray(this._selected)) this._selected = this._selected.length ? this._selected[0] : null;
777
+ }
778
+ /** Initializes the component. */
779
+ _init(activeDate) {
780
+ if (isServer) return;
781
+ else if (this.hydrationRequired) {
782
+ this.hydrationComplete.then(() => this._init());
783
+ return;
784
+ }
785
+ if (activeDate) this._assignActiveDate(activeDate);
786
+ this._wide = (this._mediaMatcher.matches(SbbMediaQueryBreakpointLargeAndAbove) ?? false) && this.wide;
787
+ this._weeks = this._createWeekRows(this._activeDate);
788
+ this._years = this._createYearRows();
789
+ this._weekNumbers = this._weeks.flat().sort((a, b) => a.value.localeCompare(b.value)).map((day) => day.weekValue).filter((v, i, a) => a.indexOf(v) === i);
790
+ this._nextMonthWeeks = [[]];
791
+ this._nextMonthYears = [[]];
792
+ if (this._wide) {
793
+ const nextMonthDate = this._dateAdapter.addCalendarMonths(this._activeDate, 1);
794
+ this._nextMonthWeeks = this._createWeekRows(nextMonthDate, true);
795
+ this._nextMonthYears = this._createYearRows(YEARS_PER_PAGE);
796
+ this._nextMonthWeekNumbers = this._nextMonthWeeks.flat().sort((a, b) => a.value.localeCompare(b.value)).map((day) => day.weekValue).filter((v, i, a) => a.indexOf(v) === i);
797
+ }
798
+ this._initialized = true;
799
+ }
800
+ /** Focuses on a day cell prioritizing the selected day, the current day, and lastly, the first selectable day. */
801
+ _focusCell() {
802
+ if (this._resetFocus) {
803
+ this._getFirstFocusable()?.focus();
804
+ this._resetFocus = false;
805
+ }
806
+ }
807
+ /** Creates the array of weekdays. */
808
+ _setWeekdays() {
809
+ const narrowWeekdays = this._dateAdapter.getDayOfWeekNames("narrow");
810
+ const weekdays = this._dateAdapter.getDayOfWeekNames("long").map((long, i) => ({
811
+ long,
812
+ narrow: narrowWeekdays[i]
813
+ }));
814
+ const firstDayOfWeek = this._dateAdapter.getFirstDayOfWeek();
815
+ this._weekdays = weekdays.slice(firstDayOfWeek).concat(weekdays.slice(0, firstDayOfWeek));
816
+ }
817
+ /** Creates the rows along the horizontal direction and sets the parameters used in keyboard navigation. */
818
+ _createWeekRows(value, isSecondMonthInView = false) {
819
+ const daysInMonth = this._dateAdapter.getNumDaysInMonth(value);
820
+ const weekOffset = this._dateAdapter.getFirstWeekOffset(value);
821
+ if (!isSecondMonthInView) {
822
+ this._keyboardNavigationDayViewParameters.firstMonthLength = daysInMonth;
823
+ this._keyboardNavigationDayViewParameters.firstMonthOffset = weekOffset;
824
+ this._keyboardNavigationDayViewParameters.firstDayInView = this._dateAdapter.toIso8601(this._dateAdapter.createDate(this._dateAdapter.getYear(value), this._dateAdapter.getMonth(value), 1));
825
+ this._keyboardNavigationDayViewParameters.lastDayInView = this._dateAdapter.toIso8601(this._dateAdapter.createDate(this._dateAdapter.getYear(value), this._dateAdapter.getMonth(value), daysInMonth));
826
+ } else {
827
+ this._keyboardNavigationDayViewParameters.secondMonthOffset = weekOffset;
828
+ this._keyboardNavigationDayViewParameters.lastDayInView = this._dateAdapter.toIso8601(this._dateAdapter.createDate(this._dateAdapter.getYear(value), this._dateAdapter.getMonth(value), daysInMonth));
829
+ }
830
+ return this.orientation === "horizontal" ? this._createWeekRowsHorizontal(value, daysInMonth, weekOffset) : this._createWeekRowsVertical(value, daysInMonth, weekOffset);
831
+ }
832
+ /**
833
+ * Creates the rows for each week in orientation='horizontal'.
834
+ *
835
+ * Iterates through the days of the months, creates a Day object for each and pushes it into and array.
836
+ * Each seven days (considering the offset at the beginning of the month) restarts from an empty array.
837
+ *
838
+ * The result is a matrix in which every row is a week (or part of it, considering offset).
839
+ */
840
+ _createWeekRowsHorizontal(value, daysInMonth, weekOffset) {
841
+ const weeks = [[]];
842
+ for (let i = 0, cell = weekOffset; i < daysInMonth; i++, cell++) {
843
+ if (cell === DAYS_PER_ROW) {
844
+ weeks.push([]);
845
+ cell = 0;
846
+ }
847
+ const date = this._dateAdapter.createDate(this._dateAdapter.getYear(value), this._dateAdapter.getMonth(value), i + 1);
848
+ weeks[weeks.length - 1].push(this._mapDateToDay(date));
849
+ }
850
+ return weeks;
851
+ }
852
+ /**
853
+ * Creates the rows for each week in orientation='vertical'.
854
+ *
855
+ * Creates a matrix with seven empty rows.
856
+ * Iterates through the days of the months, creates a Day object for each
857
+ * and pushes it into the correct array considering the offset at the beginning of the month.
858
+ * Each seven days (including offset) restarts from the first.
859
+ *
860
+ * The result is a matrix in which every row is a set of weekdays, so:
861
+ * - row 0: all the Mondays;
862
+ * - row 1: all the Tuesdays;
863
+ * - ...
864
+ * - row 7: all the Sundays.
865
+ */
866
+ _createWeekRowsVertical(value, daysInMonth, weekOffset) {
867
+ const weeks = Array.from({ length: DAYS_PER_ROW }, () => []);
868
+ for (let i = 0, cell = weekOffset; i < daysInMonth; i++, cell++) {
869
+ if (cell === DAYS_PER_ROW) cell = 0;
870
+ const date = this._dateAdapter.createDate(this._dateAdapter.getYear(value), this._dateAdapter.getMonth(value), i + 1);
871
+ weeks[cell].push(this._mapDateToDay(date));
872
+ }
873
+ return weeks;
874
+ }
875
+ _mapDateToDay(date) {
876
+ return {
877
+ value: this._dateAdapter.toIso8601(date),
878
+ dateValue: date,
879
+ dayValue: String(this._dateAdapter.getDate(date)),
880
+ monthValue: String(this._dateAdapter.getMonth(date)),
881
+ yearValue: String(this._dateAdapter.getYear(date)),
882
+ weekValue: this._getWeek(date),
883
+ weekDayValue: this._dateAdapter.getDayOfWeek(date)
884
+ };
885
+ }
886
+ _getWeek(date) {
887
+ const firstDayOfYear = this._dateAdapter.createDate(this._dateAdapter.getYear(date), 1, 1);
888
+ const weekday = this._dateAdapter.getDayOfWeek(firstDayOfYear);
889
+ let weekIndex = [
890
+ MONDAY,
891
+ TUESDAY,
892
+ WEDNESDAY,
893
+ THURSDAY
894
+ ].includes(weekday) ? 1 : 0;
895
+ let weekStart = this._dateAdapter.addCalendarDays(firstDayOfYear, this._dateAdapter.getFirstWeekOffset(firstDayOfYear) * -1);
896
+ while (this._dateAdapter.compareDate(weekStart, date) <= 0) {
897
+ const weekEnd = this._dateAdapter.addCalendarDays(weekStart, 6);
898
+ if (this._dateAdapter.compareDate(date, weekEnd) <= 0) return weekIndex > 0 ? weekIndex : this._getWeek(this._dateAdapter.addCalendarDays(firstDayOfYear, -1));
899
+ weekStart = this._dateAdapter.addCalendarDays(weekStart, DAYS_PER_ROW);
900
+ weekIndex++;
901
+ }
902
+ throw new Error("The provided date is invalid");
903
+ }
904
+ /** Force the conversion to ISO8601 formatted string. */
905
+ _mapValueToISODate(value) {
906
+ return typeof value === "string" ? value : this._dateAdapter.toIso8601(value);
907
+ }
908
+ /** Creates the rows for the month selection view. */
909
+ _createMonthRows() {
910
+ const months = new Array(12).fill(null).map((_, i) => ({
911
+ value: String(i + 1).padStart(2, "0"),
912
+ monthValue: i + 1
913
+ }));
914
+ const rows = 12 / MONTHS_PER_ROW;
915
+ const monthArray = [];
916
+ for (let i = 0; i < rows; i++) monthArray.push(months.slice(MONTHS_PER_ROW * i, MONTHS_PER_ROW * (i + 1)));
917
+ this._months = monthArray;
918
+ }
919
+ /** Creates the rows for the year selection view. */
920
+ _createYearRows(offset = 0) {
921
+ const startValueYearView = this._getStartValueYearView();
922
+ const allYears = new Array(YEARS_PER_PAGE).fill(0).map((_, i) => startValueYearView + offset + i);
923
+ const rows = YEARS_PER_PAGE / YEARS_PER_ROW;
924
+ const yearArray = [];
925
+ for (let i = 0; i < rows; i++) yearArray.push(allYears.slice(YEARS_PER_ROW * i, YEARS_PER_ROW * (i + 1)));
926
+ return yearArray;
927
+ }
928
+ /**
929
+ * Calculates the first year that will be shown in the year selection panel.
930
+ * If `minDate` and `maxDate` are both null, the starting year is calculated as
931
+ * the multiple of YEARS_PER_PAGE closest to and less than activeDate,
932
+ * e.g., with `YEARS_PER_PAGE` = 24 and `activeDate` = 2020, the function will return 2016 (24 * 83),
933
+ * while with `activeDate` = 2000, the function will return 1992 (24 * 82).
934
+ * If `minDate` is not null, it returns the corresponding year; if `maxDate` is not null,
935
+ * it returns the corresponding year minus `YEARS_PER_PAGE`, so that the `maxDate` is the last rendered year.
936
+ * If both are not null, `maxDate` has priority over `minDate`.
937
+ */
938
+ _getStartValueYearView() {
939
+ let startingYear = 0;
940
+ if (this.max) startingYear = this._dateAdapter.getYear(this.max) - YEARS_PER_PAGE + 1;
941
+ else if (this.min) startingYear = this._dateAdapter.getYear(this.min);
942
+ const activeYear = this._dateAdapter.getYear(this._activeDate);
943
+ return activeYear - ((activeYear - startingYear) % YEARS_PER_PAGE + YEARS_PER_PAGE) % YEARS_PER_PAGE;
944
+ }
945
+ /** Checks if date is within the min-max range. */
946
+ _isDayInRange(dateString) {
947
+ if (!this.min && !this.max) return true;
948
+ const date = this._dateAdapter.deserialize(dateString);
949
+ return this._dateAdapter.sameDate(date, this._dateAdapter.clampDate(date, this.min, this.max));
950
+ }
951
+ /** Emits the selected date and sets it internally. */
952
+ _selectDate(day) {
953
+ this._chosenMonth = void 0;
954
+ this._setChosenYear();
955
+ if (this.multiple) {
956
+ if (this._selected && this._selected.length > 0) {
957
+ const indexOfSelectedDay = this._selected.findIndex((sel) => this._dateAdapter.compareDate(sel, day) === 0);
958
+ if (indexOfSelectedDay !== -1) this._selected = this._selected.filter((_, i) => i !== indexOfSelectedDay);
959
+ else this._selected = [...this._selected, day];
960
+ } else this._selected = [day];
961
+ this._emitDateSelectedEvent(this._selected.map((e) => this._dateAdapter.deserialize(e)));
962
+ } else if (!this._selected || this._dateAdapter.compareDate(this._selected, day) !== 0) {
963
+ this._selected = day;
964
+ this._emitDateSelectedEvent(this._dateAdapter.deserialize(day));
965
+ }
966
+ }
967
+ /**
968
+ * Handle multiple dates selection via weekNumber / weekDay buttons:
969
+ * - if Cmd or Ctrl are pressed, add the new date to the current ones;
970
+ * - if not,
971
+ * - if the new dates are the same of the current ones, it means that the same button has been clicked twice, so do nothing;
972
+ * - if not, the selected dates are the new ones.
973
+ */
974
+ _selectMultipleDates(days) {
975
+ const enabledDays = this._cells.filter((e) => !e.disabled).map((e) => this._mapValueToISODate(e.value));
976
+ const daysToAdd = days.map((e) => e.value).filter((isoDate) => enabledDays.includes(isoDate));
977
+ const daysToAddSet = new Set(daysToAdd);
978
+ const selectedSet = new Set(this._selected.map((s) => this._dateAdapter.toIso8601(s)));
979
+ this._selected = this._updateSelectedWithMultipleDates(daysToAdd, daysToAddSet, selectedSet).map((s) => this._dateAdapter.deserialize(s));
980
+ this._emitDateSelectedEvent(this._selected.map((e) => this._dateAdapter.deserialize(e)));
981
+ }
982
+ /**
983
+ * Emits the dateselected event given the detail (as T or T[] based on the value of the multiple flag).
984
+ */
985
+ _emitDateSelectedEvent(detail) {
986
+ /** @type {CustomEvent<T | T[]>} Event emitted on date selection. */
987
+ this.dispatchEvent(new CustomEvent("dateselected", {
988
+ detail,
989
+ composed: true,
990
+ bubbles: true
991
+ }));
992
+ }
993
+ _emitMonthChange() {
994
+ const monthchange = (this.wide ? [...this._weeks, ...this._nextMonthWeeks] : this._weeks).flat().sort((a, b) => a.value.localeCompare(b.value));
995
+ /**
996
+ * @type {SbbMonthChangeEvent}
997
+ * Emits when the month changes.
998
+ * The `range` property contains the days array of the chosen month.
999
+ */
1000
+ this.dispatchEvent(new SbbMonthChangeEvent(monthchange));
1001
+ }
1002
+ /**
1003
+ * In case of multiple selection, newly added days must be added to the existing ones, without duplication.
1004
+ * If the days to add are exactly the same as the selected ones, the set must be emptied.
1005
+ */
1006
+ _updateSelectedWithMultipleDates(daysToAdd, daysToAddSet, selectedSet) {
1007
+ if (daysToAdd.every((day) => selectedSet.has(day))) daysToAddSet.forEach((day) => selectedSet.delete(day));
1008
+ else daysToAddSet.forEach((day) => selectedSet.add(day));
1009
+ return Array.from(selectedSet);
1010
+ }
1011
+ _setChosenYear() {
1012
+ if (this.view === "month") {
1013
+ let selectedDate;
1014
+ if (this.multiple) selectedDate = this.selected.at(-1);
1015
+ else selectedDate = this.selected;
1016
+ this._chosenYear = this._dateAdapter.getYear(selectedDate ?? this._dateAdapter.today());
1017
+ } else this._chosenYear = void 0;
1018
+ }
1019
+ _assignActiveDate(date) {
1020
+ if (this.min && this._dateAdapter.compareDate(this.min, date) > 0) {
1021
+ this._activeDate = this.min;
1022
+ return;
1023
+ }
1024
+ if (this.max && this._dateAdapter.compareDate(this.max, date) < 0) {
1025
+ this._activeDate = this.max;
1026
+ return;
1027
+ }
1028
+ this._activeDate = date;
1029
+ }
1030
+ /** Goes to the month identified by the shift. */
1031
+ _goToDifferentMonth(months) {
1032
+ this._init(this._dateAdapter.addCalendarMonths(this._activeDate, months));
1033
+ this._emitMonthChange();
1034
+ }
1035
+ _goToDifferentYear(years) {
1036
+ this._chosenYear += years;
1037
+ this._activeDate = this._dateAdapter.createDate(this._chosenYear, this._dateAdapter.getMonth(this._activeDate), this._dateAdapter.getDate(this._activeDate));
1038
+ this._init();
1039
+ }
1040
+ _goToDifferentYearRange(years) {
1041
+ this._init(this._dateAdapter.addCalendarYears(this._activeDate, years));
1042
+ }
1043
+ _prevDisabled(prevDate) {
1044
+ if (!this.min) return false;
1045
+ return this._dateAdapter.compareDate(prevDate, this.min) < 0;
1046
+ }
1047
+ _nextDisabled(nextDate) {
1048
+ if (!this.max) return false;
1049
+ return this._dateAdapter.compareDate(nextDate, this.max) > 0;
1050
+ }
1051
+ /** Checks if the "previous month" button should be disabled. */
1052
+ _previousMonthDisabled() {
1053
+ const prevMonth = this._dateAdapter.addCalendarDays(this._activeDate, this._dateAdapter.getDate(this._activeDate) * -1);
1054
+ return this._prevDisabled(prevMonth);
1055
+ }
1056
+ /** Checks if the "next month" button should be disabled. */
1057
+ _nextMonthDisabled() {
1058
+ let nextMonth = this._dateAdapter.addCalendarMonths(this._activeDate, this._wide ? 2 : 1);
1059
+ nextMonth = this._dateAdapter.createDate(this._dateAdapter.getYear(nextMonth), this._dateAdapter.getMonth(nextMonth), 1);
1060
+ return this._nextDisabled(nextMonth);
1061
+ }
1062
+ /** Checks if the "previous year" button should be disabled. */
1063
+ _previousYearDisabled() {
1064
+ const prevYear = this._dateAdapter.createDate(this._dateAdapter.getYear(this._activeDate) - 1, 12, 31);
1065
+ return this._prevDisabled(prevYear);
1066
+ }
1067
+ /** Checks if the "next year" button should be disabled. */
1068
+ _nextYearDisabled() {
1069
+ const nextYear = this._dateAdapter.createDate(this._dateAdapter.getYear(this._activeDate) + (this._wide ? 2 : 1), 1, 1);
1070
+ return this._nextDisabled(nextYear);
1071
+ }
1072
+ /** Checks if the "previous year" button should be disabled in year view. */
1073
+ _previousYearRangeDisabled() {
1074
+ const prevYear = this._dateAdapter.createDate(this._years[0][0] - 1, 12, 31);
1075
+ return this._prevDisabled(prevYear);
1076
+ }
1077
+ /** Checks if the "next year" button should be disabled in year view. */
1078
+ _nextYearRangeDisabled() {
1079
+ const years = this._wide ? this._nextMonthYears : this._years;
1080
+ const lastYearRange = years[years.length - 1];
1081
+ const lastYear = lastYearRange[lastYearRange.length - 1];
1082
+ const nextYear = this._dateAdapter.createDate(lastYear + 1, 1, 1);
1083
+ return this._nextDisabled(nextYear);
1084
+ }
1085
+ _handleTableBlur(eventTarget) {
1086
+ if (eventTarget?.localName !== "sbb-calendar-day") this._setTabIndex();
1087
+ }
1088
+ _setTabIndex() {
1089
+ Array.from(this._cells.filter((e) => e.tabIndex === 0) ?? []).forEach((day) => day.tabIndex = -1);
1090
+ const firstFocusable = this._getFirstFocusable();
1091
+ if (firstFocusable) firstFocusable.tabIndex = 0;
1092
+ }
1093
+ /** Get the element in the calendar to assign focus. */
1094
+ _getFirstFocusable() {
1095
+ if (this._calendarView === "day") {
1096
+ const selectedOrCurrent = this._cells.find((e) => e.matches(":is(:state(selected),[state--selected])")) ?? this._cells.find((e) => e.matches(":is(:state(current),[state--current])"));
1097
+ return selectedOrCurrent && !selectedOrCurrent.disabled ? selectedOrCurrent : this._getFirstFocusableDay();
1098
+ } else {
1099
+ const selectedOrCurrent = this.shadowRoot?.querySelector(":is(:state(selected),[state--selected])") ?? this.shadowRoot?.querySelector(":is(:state(current),[state--current])");
1100
+ return selectedOrCurrent && !selectedOrCurrent.disabled ? selectedOrCurrent : this.shadowRoot.querySelector(`sbb-calendar-${this._calendarView}:not([disabled])`);
1101
+ }
1102
+ }
1103
+ /**
1104
+ * In `day` view in `vertical` orientation,
1105
+ * if the first of the month is not a Monday, it is not the first rendered element in the table,
1106
+ * so `this.shadowRoot!.querySelector('sbb-calendar-day:not([disabled])')` will return a wrong value.
1107
+ *
1108
+ * To solve this, the element with the lowest `value` is taken (ISO String are ordered).
1109
+ */
1110
+ _getFirstFocusableDay() {
1111
+ const cells = this._cells;
1112
+ const daysInView = cells.filter((e) => !e.disabled);
1113
+ if (!daysInView || daysInView.length === 0) return null;
1114
+ else {
1115
+ const firstElement = daysInView.map((e) => this._dateAdapter.toIso8601(e.value)).sort()[0];
1116
+ return cells.find((e) => e.matches(`[slot="${firstElement}"]`)) ?? null;
1117
+ }
1118
+ }
1119
+ _handleKeyboardEvent(event, day) {
1120
+ if (isArrowKeyOrPageKeysPressed(event)) event.preventDefault();
1121
+ const cells = this._cells;
1122
+ const index = cells.findIndex((e) => e === event.target);
1123
+ let nextEl;
1124
+ if (day) nextEl = this._navigateByKeyboardDayView(event, index, cells, day);
1125
+ else nextEl = this._navigateByKeyboard(event, index, cells);
1126
+ const activeEl = (this._enhancedVariant ? document : this.shadowRoot).activeElement;
1127
+ if (nextEl !== activeEl) {
1128
+ nextEl.tabIndex = 0;
1129
+ nextEl?.focus();
1130
+ activeEl.tabIndex = -1;
1131
+ }
1132
+ }
1133
+ _navigateByKeyboardDayView(evt, index, cells, day) {
1134
+ const arrowsOffset = this.orientation === "horizontal" ? {
1135
+ leftRight: 1,
1136
+ upDown: DAYS_PER_ROW
1137
+ } : {
1138
+ leftRight: DAYS_PER_ROW,
1139
+ upDown: 1
1140
+ };
1141
+ const offsetForVertical = index < this._keyboardNavigationDayViewParameters.firstMonthLength ? this._keyboardNavigationDayViewParameters.firstMonthOffset : this._keyboardNavigationDayViewParameters.secondMonthOffset;
1142
+ switch (evt.key) {
1143
+ case "ArrowUp": return this._findDayArrows(cells, index, day.dateValue, -arrowsOffset.upDown);
1144
+ case "ArrowDown": return this._findDayArrows(cells, index, day.dateValue, arrowsOffset.upDown);
1145
+ case "ArrowLeft": return this._findDayArrows(cells, index, day.dateValue, -arrowsOffset.leftRight);
1146
+ case "ArrowRight": return this._findDayArrows(cells, index, day.dateValue, arrowsOffset.leftRight);
1147
+ case "PageUp": if (this.orientation === "horizontal") {
1148
+ const delta = (+day.dayValue % DAYS_PER_ROW || DAYS_PER_ROW) - +day.dayValue;
1149
+ return this._findDayPageUpDown(cells, index, day, delta, arrowsOffset.upDown);
1150
+ } else {
1151
+ const delta = (Math.ceil((+day.dayValue + offsetForVertical) / DAYS_PER_ROW) - 1) * DAYS_PER_ROW - offsetForVertical + 1 - +day.dayValue;
1152
+ return this._findDayPageUpDown(cells, index, day, delta, arrowsOffset.upDown);
1153
+ }
1154
+ case "PageDown": if (this.orientation === "horizontal") {
1155
+ const monthInBounds = +day.monthValue + 1 > 12 ? 1 : +day.monthValue + 1;
1156
+ const yearInBounds = +day.monthValue + 1 > 12 ? +day.yearValue + 1 : +day.yearValue;
1157
+ const firstNextMonth = this._dateAdapter.createDate(yearInBounds, monthInBounds, 1);
1158
+ const lastOfMonth = this._dateAdapter.getDate(this._dateAdapter.addCalendarDays(firstNextMonth, -1));
1159
+ const delta = Math.trunc((lastOfMonth - +day.dayValue) / DAYS_PER_ROW) * DAYS_PER_ROW;
1160
+ return this._findDayPageUpDown(cells, index, day, delta, -arrowsOffset.upDown);
1161
+ } else {
1162
+ const delta = Math.ceil((+day.dayValue + offsetForVertical) / DAYS_PER_ROW) * DAYS_PER_ROW - offsetForVertical - +day.dayValue;
1163
+ return this._findDayPageUpDown(cells, index, day, delta, -arrowsOffset.upDown);
1164
+ }
1165
+ case "Home": return this._findDayFirst(cells, index, day, 1);
1166
+ case "End": {
1167
+ const monthInBounds = +day.monthValue + 1 > 12 ? 1 : +day.monthValue + 1;
1168
+ const yearInBounds = +day.monthValue + 1 > 12 ? +day.yearValue + 1 : +day.yearValue;
1169
+ const firstNextMonth = this._dateAdapter.createDate(yearInBounds, monthInBounds, 1);
1170
+ return this._findDayLast(cells, index, firstNextMonth);
1171
+ }
1172
+ default: return cells[index];
1173
+ }
1174
+ }
1175
+ _isDayOutOfView(date) {
1176
+ return date < this._keyboardNavigationDayViewParameters.firstDayInView || date > this._keyboardNavigationDayViewParameters.lastDayInView;
1177
+ }
1178
+ _findDayArrows(cells, index, date, delta) {
1179
+ const newDateValue = this._dateAdapter.toIso8601(this._dateAdapter.addCalendarDays(date, delta));
1180
+ if (this._isDayOutOfView(newDateValue)) return cells[index];
1181
+ const nextCell = cells.find((e) => this._mapValueToISODate(e.value) === newDateValue);
1182
+ if (!nextCell || nextCell.disabled) return this._findDayArrows(cells, index, this._dateAdapter.deserialize(newDateValue), delta);
1183
+ return nextCell;
1184
+ }
1185
+ _findDayPageUpDown(cells, index, day, delta, deltaIfDisabled) {
1186
+ const newDateValue = this._dateAdapter.toIso8601(this._dateAdapter.addCalendarDays(day.dateValue, delta));
1187
+ if (this._isDayOutOfView(newDateValue)) return cells[index];
1188
+ const nextCell = cells.find((e) => this._mapValueToISODate(e.value) === newDateValue);
1189
+ if (!nextCell || nextCell.disabled) return this._findDayPageUpDown(cells, index, day, delta + deltaIfDisabled, deltaIfDisabled);
1190
+ return nextCell;
1191
+ }
1192
+ _findDayFirst(cells, index, day, date) {
1193
+ const newDateValue = this._dateAdapter.toIso8601(this._dateAdapter.createDate(+day.yearValue, +day.monthValue, date));
1194
+ if (this._isDayOutOfView(newDateValue)) return cells[index];
1195
+ const nextCell = cells.find((e) => this._mapValueToISODate(e.value) === newDateValue);
1196
+ if (!nextCell || nextCell.disabled) return this._findDayFirst(cells, index, day, date + 1);
1197
+ return nextCell;
1198
+ }
1199
+ _findDayLast(cells, index, firstNextMonth) {
1200
+ const newDateValue = this._dateAdapter.toIso8601(this._dateAdapter.addCalendarDays(firstNextMonth, -1));
1201
+ if (this._isDayOutOfView(newDateValue)) return cells[index];
1202
+ const nextCell = cells.find((e) => this._mapValueToISODate(e.value) === newDateValue);
1203
+ if (!nextCell || nextCell.disabled) return this._findDayLast(cells, index, this._dateAdapter.deserialize(newDateValue));
1204
+ return nextCell;
1205
+ }
1206
+ /**
1207
+ * Gets the index of the element to move to, based on a list of elements (which can be potentially disabled),
1208
+ * the keyboard input and the position of the current element in the list.
1209
+ * In the day view, the `day?: Day` parameter is mandatory for calculation,
1210
+ * while in month and year view it's not due to the fixed amount of rendered cells.
1211
+ */
1212
+ _navigateByKeyboard(evt, index, cells) {
1213
+ const { elementIndexForWideMode, offsetForWideMode, lastElementIndexForWideMode, verticalOffset } = this._calculateParametersForKeyboardNavigation(index, this._calendarView === "year");
1214
+ switch (evt.key) {
1215
+ case "ArrowUp": return this._findNext(cells, index, -verticalOffset);
1216
+ case "ArrowDown": return this._findNext(cells, index, verticalOffset);
1217
+ case "ArrowLeft": return this._findNext(cells, index, -1);
1218
+ case "ArrowRight": return this._findNext(cells, index, 1);
1219
+ case "Home": return this._findFirst(cells, offsetForWideMode);
1220
+ case "PageUp": return this._findFirstOnColumn(cells, elementIndexForWideMode, offsetForWideMode, verticalOffset);
1221
+ case "PageDown": return this._findLastOnColumn(cells, index, lastElementIndexForWideMode, verticalOffset);
1222
+ case "End": return this._findLast(cells, lastElementIndexForWideMode - 1);
1223
+ default: return cells[index];
1224
+ }
1225
+ }
1226
+ /**
1227
+ * Calculates the parameters needed in keyboard navigation in year and month view.
1228
+ * @param index The starting element's index in the cell array.
1229
+ * @param isYearView Whether the displayed `view` is the year one.
1230
+ */
1231
+ _calculateParametersForKeyboardNavigation(index, isYearView) {
1232
+ const elementsPerPage = isYearView ? YEARS_PER_PAGE : MONTHS_PER_PAGE;
1233
+ const offset = Math.trunc(index / elementsPerPage) * elementsPerPage;
1234
+ const indexInView = offset === 0 ? index : index - elementsPerPage;
1235
+ return {
1236
+ verticalOffset: isYearView ? YEARS_PER_ROW : MONTHS_PER_ROW,
1237
+ elementIndexForWideMode: indexInView,
1238
+ offsetForWideMode: index - indexInView,
1239
+ lastElementIndexForWideMode: offset === 0 ? elementsPerPage : elementsPerPage * 2
1240
+ };
1241
+ }
1242
+ /**
1243
+ * Gets the next element of the provided array starting from `index` by adding `delta`.
1244
+ * If the found element is disabled, it continues adding `delta` until it finds an enabled one in the array bounds.
1245
+ */
1246
+ _findNext(days, index, delta) {
1247
+ let nextIndex = index + delta;
1248
+ while (nextIndex < days.length && days[nextIndex]?.disabled) nextIndex += delta;
1249
+ return days[nextIndex] ?? days[index];
1250
+ }
1251
+ /** Find the first enabled element in the provided array. */
1252
+ _findFirst(days, firstOfCurrentMonth) {
1253
+ return !days[firstOfCurrentMonth].disabled ? days[firstOfCurrentMonth] : this._findNext(days, firstOfCurrentMonth, 1);
1254
+ }
1255
+ /** Find the last enabled element in the provided array. */
1256
+ _findLast(days, lastOfCurrentMonth) {
1257
+ return !days[lastOfCurrentMonth].disabled ? days[lastOfCurrentMonth] : this._findNext(days, lastOfCurrentMonth, -1);
1258
+ }
1259
+ /** Find the first enabled element in the same column of the provided array. */
1260
+ _findFirstOnColumn(days, index, offset, verticalOffset) {
1261
+ const nextIndex = index % verticalOffset + offset;
1262
+ return !days[nextIndex].disabled ? days[nextIndex] : this._findNext(days, nextIndex, verticalOffset);
1263
+ }
1264
+ /** Find the last enabled element in the same column of the provided array. */
1265
+ _findLastOnColumn(days, index, offset, verticalOffset) {
1266
+ const nextIndex = index + Math.trunc((offset - index - 1) / verticalOffset) * verticalOffset;
1267
+ return !days[nextIndex].disabled ? days[nextIndex] : this._findNext(days, nextIndex, -verticalOffset);
1268
+ }
1269
+ _resetCalendarViewAndEmitMonthChange(initTransition = false) {
1270
+ this._resetCalendarView(initTransition);
1271
+ this._emitMonthChange();
1272
+ }
1273
+ _resetCalendarView(initTransition = false) {
1274
+ if (this._containingFocus) this._resetFocus = true;
1275
+ this._activeDate = (this.multiple ? this._selected.at(-1) : this._selected) ?? this._dateAdapter.today();
1276
+ this._setChosenYear();
1277
+ this._chosenMonth = void 0;
1278
+ this._init();
1279
+ this._nextCalendarView = this._calendarView = this.view;
1280
+ if (initTransition) this._startTableTransition();
1281
+ }
1282
+ /** Render the view for the day selection. */
1283
+ _renderDayView() {
1284
+ const nextMonthActiveDate = this._wide ? this._dateAdapter.addCalendarMonths(this._activeDate, 1) : void 0;
1285
+ return html`
1286
+ <div class="sbb-calendar__controls">
1287
+ ${this._getArrow("left", () => this._goToDifferentMonth(-1), i18nPreviousMonth[this._language.current], this._previousMonthDisabled())}
1288
+ <div class="sbb-calendar__controls-month">
1289
+ ${this._createLabelForDayView(this._activeDate)}
1290
+ ${this._wide ? this._createLabelForDayView(nextMonthActiveDate) : nothing}
1291
+ <sbb-screen-reader-only role="status">
1292
+ ${this._createAriaLabelForDayView(this._activeDate, nextMonthActiveDate)}
1293
+ </sbb-screen-reader-only>
1294
+ </div>
1295
+ ${this._getArrow("right", () => this._goToDifferentMonth(1), i18nNextMonth[this._language.current], this._nextMonthDisabled())}
1296
+ </div>
1297
+ <div class="sbb-calendar__table-overflow-break">
1298
+ <div class="sbb-calendar__table-container sbb-calendar__table-day-view">
1299
+ ${this.orientation === "horizontal" ? html`
1300
+ ${this._createDayTable(this._weeks, this._weekNumbers)}
1301
+ ${this._wide ? this._createDayTable(this._nextMonthWeeks, this._nextMonthWeekNumbers, true) : nothing}
1302
+ ` : html`
1303
+ ${this._createDayTableVertical(this._weeks, this._weekNumbers)}
1304
+ ${this._wide ? this._createDayTableVertical(this._nextMonthWeeks, this._nextMonthWeekNumbers, nextMonthActiveDate) : nothing}
1305
+ `}
1306
+ </div>
1307
+ </div>
1308
+ `;
1309
+ }
1310
+ /** Creates the label with the month for the daily view. */
1311
+ _createLabelForDayView(d) {
1312
+ const monthLabel = `${this._monthNames[this._dateAdapter.getMonth(d) - 1]} ${this._dateAdapter.getYear(d)}`;
1313
+ return html`
1314
+ <button
1315
+ type="button"
1316
+ class="sbb-calendar__date-selection sbb-calendar__controls-change-date"
1317
+ aria-label="${i18nYearMonthSelection[this._language.current]} ${monthLabel}"
1318
+ @click=${() => {
1319
+ this._resetFocus = true;
1320
+ this._nextCalendarView = "year";
1321
+ this._startTableTransition();
1322
+ }}
1323
+ >
1324
+ ${monthLabel}
1325
+ <sbb-icon name="chevron-small-down-small"></sbb-icon>
1326
+ </button>
1327
+ `;
1328
+ }
1329
+ /** Creates the aria-label for the daily view. */
1330
+ _createAriaLabelForDayView(...dates) {
1331
+ let monthLabel = "";
1332
+ for (const d of dates) if (d) monthLabel += `${this._monthNames[this._dateAdapter.getMonth(d) - 1]} ${this._dateAdapter.getYear(d)} `;
1333
+ return monthLabel;
1334
+ }
1335
+ /** Creates the calendar table for the daily view. */
1336
+ _createDayTable(weeks, weekNumbers, isWideNextMonth = false) {
1337
+ const weeksForSelectMultipleWeekNumbers = (this._wide ? [...this._weeks, ...this._nextMonthWeeks] : isWideNextMonth ? this._nextMonthWeeks : this._weeks).flat();
1338
+ const weeksForSelectMultipleWeekDays = (isWideNextMonth ? this._nextMonthWeeks : this._weeks).flat();
1339
+ return html`
1340
+ <table
1341
+ class="sbb-calendar__table"
1342
+ @focusout=${(event) => this._handleTableBlur(event.relatedTarget)}
1343
+ @animationend=${(e) => this._tableAnimationEnd(e)}
1344
+ >
1345
+ <thead class="sbb-calendar__table-header">
1346
+ <tr>
1347
+ ${this.weekNumbers ? html`<th class="sbb-calendar__table-header-cell"></th>` : nothing}
1348
+ ${this._weekdays.map((weekDay, index) => html`
1349
+ <th class="sbb-calendar__table-header-cell">
1350
+ ${this.multiple ? html`
1351
+ <sbb-calendar-weekday
1352
+ .value=${weekDay}
1353
+ @click=${() => {
1354
+ const days = weeksForSelectMultipleWeekDays.filter((day) => day.weekDayValue === (index + 1) % 7);
1355
+ this._selectMultipleDates(days);
1356
+ }}
1357
+ ></sbb-calendar-weekday>
1358
+ ` : html`
1359
+ <sbb-screen-reader-only>${weekDay.long}</sbb-screen-reader-only>
1360
+ <span aria-hidden="true">${weekDay.narrow}</span>
1361
+ `}
1362
+ </th>
1363
+ `)}
1364
+ </tr>
1365
+ </thead>
1366
+ <tbody class="sbb-calendar__table-body">
1367
+ ${weeks.map((week, rowIndex) => {
1368
+ const firstRowOffset = DAYS_PER_ROW - week.length;
1369
+ if (rowIndex === 0 && firstRowOffset) return html`
1370
+ <tr>
1371
+ ${this.weekNumbers ? html`
1372
+ <td class="sbb-calendar__table-header-cell-vertical">
1373
+ ${this.multiple ? html`
1374
+ <sbb-calendar-weeknumber
1375
+ .value=${weekNumbers[0]}
1376
+ @click=${() => {
1377
+ const days = weeksForSelectMultipleWeekNumbers.filter((day) => day.weekValue === weekNumbers[0]);
1378
+ this._selectMultipleDates(days);
1379
+ }}
1380
+ ></sbb-calendar-weeknumber>
1381
+ ` : html`
1382
+ <sbb-screen-reader-only
1383
+ >${`${i18nCalendarWeekNumber[this._language.current]} ${weekNumbers[0]}`}</sbb-screen-reader-only
1384
+ >
1385
+ <span aria-hidden="true">${weekNumbers[0]}</span>
1386
+ `}
1387
+ </td>
1388
+ ` : nothing}
1389
+ ${[...Array(firstRowOffset).keys()].map(() => html`<td class="sbb-calendar__table-data"></td>`)}
1390
+ ${this._createDayCells(week)}
1391
+ </tr>
1392
+ `;
1393
+ return html`
1394
+ <tr>
1395
+ ${this.weekNumbers ? html`
1396
+ <td class="sbb-calendar__table-header-cell-vertical">
1397
+ ${this.multiple ? html`
1398
+ <sbb-calendar-weeknumber
1399
+ .value=${weekNumbers[rowIndex]}
1400
+ @click=${() => {
1401
+ const days = weeksForSelectMultipleWeekNumbers.filter((day) => day.weekValue === weekNumbers[rowIndex]);
1402
+ this._selectMultipleDates(days);
1403
+ }}
1404
+ ></sbb-calendar-weeknumber>
1405
+ ` : html`
1406
+ <sbb-screen-reader-only
1407
+ >${`${i18nCalendarWeekNumber[this._language.current]} ${weekNumbers[rowIndex]}`}</sbb-screen-reader-only
1408
+ >
1409
+ <span aria-hidden="true">${weekNumbers[rowIndex]}</span>
1410
+ `}
1411
+ </td>
1412
+ ` : nothing}
1413
+ ${this._createDayCells(week)}
1414
+ </tr>
1415
+ `;
1416
+ })}
1417
+ </tbody>
1418
+ </table>
1419
+ `;
1420
+ }
1421
+ /** Creates the table in orientation='vertical'. */
1422
+ _createDayTableVertical(weeks, weekNumbers, nextMonthActiveDate) {
1423
+ const weekOffset = this._dateAdapter.getFirstWeekOffset(nextMonthActiveDate ?? this._activeDate);
1424
+ const weeksForSelectMultipleWeekNumbers = (this._wide ? [...this._weeks, ...this._nextMonthWeeks] : nextMonthActiveDate ? this._nextMonthWeeks : this._weeks).flat();
1425
+ return html`
1426
+ <table
1427
+ class="sbb-calendar__table"
1428
+ @focusout=${(event) => this._handleTableBlur(event.relatedTarget)}
1429
+ @animationend=${(e) => this._tableAnimationEnd(e)}
1430
+ >
1431
+ ${this.weekNumbers ? html`
1432
+ <thead class="sbb-calendar__table-header">
1433
+ <tr>
1434
+ ${nextMonthActiveDate ? nothing : html`<th class="sbb-calendar__table-data"></th>`}
1435
+ ${weekNumbers.map((weekNumber) => html`
1436
+ <th class="sbb-calendar__table-header-cell">
1437
+ ${this.multiple ? html`
1438
+ <sbb-calendar-weeknumber
1439
+ .value=${weekNumber}
1440
+ @click=${() => {
1441
+ const days = weeksForSelectMultipleWeekNumbers.filter((day) => day.weekValue === weekNumber);
1442
+ this._selectMultipleDates(days);
1443
+ }}
1444
+ ></sbb-calendar-weeknumber>
1445
+ ` : html`
1446
+ <sbb-screen-reader-only
1447
+ >${`${i18nCalendarWeekNumber[this._language.current]} ${weekNumber}`}</sbb-screen-reader-only
1448
+ >
1449
+ <span aria-hidden="true">${weekNumber}</span>
1450
+ `}
1451
+ </th>
1452
+ `)}
1453
+ </tr>
1454
+ </thead>
1455
+ ` : nothing}
1456
+ <tbody class="sbb-calendar__table-body">
1457
+ ${weeks.map((week, rowIndex) => {
1458
+ const weekday = this._weekdays[rowIndex];
1459
+ const selectableDays = this._wide ? [...week, ...this._nextMonthWeeks[rowIndex]] : week;
1460
+ return html`
1461
+ <tr>
1462
+ ${nextMonthActiveDate ? nothing : html`
1463
+ <td class="sbb-calendar__table-header-cell-vertical">
1464
+ ${this.multiple ? html`
1465
+ <sbb-calendar-weekday
1466
+ .value=${weekday}
1467
+ @click=${() => this._selectMultipleDates(selectableDays)}
1468
+ >
1469
+ ${weekday.narrow}
1470
+ </sbb-calendar-weekday>
1471
+ ` : html`
1472
+ <sbb-screen-reader-only>${weekday.long}</sbb-screen-reader-only>
1473
+ <span aria-hidden="true">${weekday.narrow}</span>
1474
+ `}
1475
+ </td>
1476
+ `}
1477
+ ${rowIndex < weekOffset ? html`<td class="sbb-calendar__table-data"></td>` : nothing}
1478
+ ${this._createDayCells(week)}
1479
+ </tr>
1480
+ `;
1481
+ })}
1482
+ </tbody>
1483
+ </table>
1484
+ `;
1485
+ }
1486
+ /** Creates the cells for the daily view. */
1487
+ _createDayCells(week) {
1488
+ return week.map((day) => {
1489
+ return html`
1490
+ <td class="sbb-calendar__table-data sbb-calendar__day-cell">
1491
+ <slot name=${day.value}>
1492
+ <sbb-calendar-day
1493
+ slot=${day.value}
1494
+ @click=${() => this._selectDate(day.dateValue)}
1495
+ @keydown=${(evt) => this._handleKeyboardEvent(evt, day)}
1496
+ ></sbb-calendar-day>
1497
+ </slot>
1498
+ </td>
1499
+ `;
1500
+ });
1501
+ }
1502
+ /** Render the view for the month selection. */
1503
+ _renderMonthView() {
1504
+ return html`
1505
+ <div class="sbb-calendar__controls">
1506
+ ${this._getArrow("left", () => this._goToDifferentYear(-1), i18nPreviousYear[this._language.current], this._previousYearDisabled())}
1507
+ <div class="sbb-calendar__controls-month">${this._createLabelForMonthView()}</div>
1508
+ ${this._getArrow("right", () => this._goToDifferentYear(1), i18nNextYear[this._language.current], this._nextYearDisabled())}
1509
+ </div>
1510
+ <div class="sbb-calendar__table-overflow-break">
1511
+ <div class="sbb-calendar__table-container sbb-calendar__table-month-view">
1512
+ ${this._createMonthTable(this._months, this._chosenYear)}
1513
+ ${this._wide ? this._createMonthTable(this._months, this._chosenYear + 1) : nothing}
1514
+ </div>
1515
+ </div>
1516
+ `;
1517
+ }
1518
+ /** Creates the label with the year for the monthly view. */
1519
+ _createLabelForMonthView() {
1520
+ return html` <button
1521
+ type="button"
1522
+ id="sbb-calendar__month-selection"
1523
+ class="sbb-calendar__controls-change-date"
1524
+ aria-label=${`${i18nCalendarDateSelection[this._language.current]} ${this._chosenYear}`}
1525
+ @click=${() => this._resetCalendarViewAndEmitMonthChange(true)}
1526
+ >
1527
+ ${this._chosenYear} ${this._wide ? ` - ${this._chosenYear + 1}` : nothing}
1528
+ <sbb-icon name="chevron-small-up-small"></sbb-icon>
1529
+ </button>
1530
+ <sbb-screen-reader-only role="status"> ${this._chosenYear} </sbb-screen-reader-only>`;
1531
+ }
1532
+ /** Creates the table for the month selection view. */
1533
+ _createMonthTable(months, year) {
1534
+ return html`
1535
+ <table
1536
+ class="sbb-calendar__table"
1537
+ @animationend=${(e) => this._tableAnimationEnd(e)}
1538
+ >
1539
+ ${this._wide ? html`<thead class="sbb-calendar__table-header" aria-hidden="true">
1540
+ <tr>
1541
+ <th class="sbb-calendar__table-header-cell" colspan=${MONTHS_PER_ROW}>${year}</th>
1542
+ </tr>
1543
+ </thead>` : nothing}
1544
+ <tbody class="sbb-calendar__table-body">
1545
+ ${months.map((row) => html`
1546
+ <tr>
1547
+ ${row.map((month) => {
1548
+ return html`
1549
+ <td class="sbb-calendar__table-data">
1550
+ <sbb-calendar-month
1551
+ .value="${year}-${month.value}"
1552
+ @click=${() => this._onMonthSelection(month.monthValue, year)}
1553
+ @keydown=${(evt) => this._handleKeyboardEvent(evt)}
1554
+ >
1555
+ </sbb-calendar-month>
1556
+ </td>
1557
+ `;
1558
+ })}
1559
+ </tr>
1560
+ `)}
1561
+ </tbody>
1562
+ </table>
1563
+ `;
1564
+ }
1565
+ /** Select the month and change the view to day selection. */
1566
+ _onMonthSelection(month, year) {
1567
+ this._chosenMonth = month;
1568
+ this._nextCalendarView = "day";
1569
+ this._init(this._dateAdapter.createDate(year, this._chosenMonth, this._dateAdapter.getDate(this._activeDate)));
1570
+ this._startTableTransition();
1571
+ this._emitMonthChange();
1572
+ }
1573
+ /** Render the view for the year selection. */
1574
+ _renderYearView() {
1575
+ return html`
1576
+ <div class="sbb-calendar__controls">
1577
+ ${this._getArrow("left", () => this._goToDifferentYearRange(-YEARS_PER_PAGE), i18nPreviousYearRange(YEARS_PER_PAGE)[this._language.current], this._previousYearRangeDisabled())}
1578
+ <div class="sbb-calendar__controls-month">${this._createLabelForYearView()}</div>
1579
+ ${this._getArrow("right", () => this._goToDifferentYearRange(YEARS_PER_PAGE), i18nNextYearRange(YEARS_PER_PAGE)[this._language.current], this._nextYearRangeDisabled())}
1580
+ </div>
1581
+ <div class="sbb-calendar__table-overflow-break">
1582
+ <div class="sbb-calendar__table-container sbb-calendar__table-year-view">
1583
+ ${this._createYearTable(this._years)}
1584
+ ${this._wide ? this._createYearTable(this._nextMonthYears, true) : nothing}
1585
+ </div>
1586
+ </div>
1587
+ `;
1588
+ }
1589
+ /** Creates the button arrow for all the views. */
1590
+ _getArrow(direction, click, ariaLabel, disabled) {
1591
+ return html`<sbb-secondary-button
1592
+ size="m"
1593
+ icon-name="chevron-small-${direction}-small"
1594
+ aria-label=${ariaLabel}
1595
+ @click=${click}
1596
+ ?disabled=${disabled}
1597
+ id="sbb-calendar__controls-${direction === "left" ? "previous" : "next"}"
1598
+ ></sbb-secondary-button>`;
1599
+ }
1600
+ /** Creates the label with the year range for the yearly view. */
1601
+ _createLabelForYearView() {
1602
+ const firstYear = this._years.flat()[0];
1603
+ const lastYearArray = (this._wide ? this._nextMonthYears : this._years).flat();
1604
+ const yearLabel = `${firstYear} - ${lastYearArray[lastYearArray.length - 1]}`;
1605
+ return html`
1606
+ <button
1607
+ type="button"
1608
+ id="sbb-calendar__year-selection"
1609
+ class="sbb-calendar__controls-change-date"
1610
+ aria-label="${i18nCalendarDateSelection[this._language.current]} ${yearLabel}"
1611
+ @click=${() => this._resetCalendarViewAndEmitMonthChange(true)}
1612
+ >
1613
+ ${yearLabel}
1614
+ <sbb-icon name="chevron-small-up-small"></sbb-icon>
1615
+ </button>
1616
+ <sbb-screen-reader-only role="status"> ${yearLabel} </sbb-screen-reader-only>
1617
+ `;
1618
+ }
1619
+ /** Creates the table for the year selection view. */
1620
+ _createYearTable(years, shiftRight = false) {
1621
+ return html` <table
1622
+ class="sbb-calendar__table"
1623
+ @animationend=${(e) => this._tableAnimationEnd(e)}
1624
+ >
1625
+ <tbody class="sbb-calendar__table-body">
1626
+ ${years.map((row) => html` <tr>
1627
+ ${row.map((year) => {
1628
+ return html`
1629
+ <td class="sbb-calendar__table-data">
1630
+ <sbb-calendar-year
1631
+ .value=${String(year)}
1632
+ @keydown=${(evt) => this._handleKeyboardEvent(evt)}
1633
+ @click=${() => this._onYearSelection(year, shiftRight)}
1634
+ >
1635
+ </sbb-calendar-year>
1636
+ </td>
1637
+ `;
1638
+ })}
1639
+ </tr>`)}
1640
+ </tbody>
1641
+ </table>`;
1642
+ }
1643
+ /** Select the year and change the view to month selection. */
1644
+ _onYearSelection(year, rightSide) {
1645
+ this._chosenYear = rightSide ? year - 1 : year;
1646
+ this._nextCalendarView = "month";
1647
+ this._assignActiveDate(this._dateAdapter.createDate(this._chosenYear, this._dateAdapter.getMonth(this._activeDate), this._dateAdapter.getDate(this._activeDate)));
1648
+ this._startTableTransition();
1649
+ }
1650
+ _getView() {
1651
+ if (isServer || this.hydrationRequired) return html`${nothing}`;
1652
+ switch (this._calendarView) {
1653
+ case "year": return this._renderYearView();
1654
+ case "month": return this._renderMonthView();
1655
+ default: return this._renderDayView();
1656
+ }
1657
+ }
1658
+ _tableAnimationEnd(event) {
1659
+ const table = event.target;
1660
+ if (event.animationName === "hide") {
1661
+ table.classList.remove("sbb-calendar__table-hide");
1662
+ if (this._containingFocus) this._resetFocus = true;
1663
+ this._calendarView = this._nextCalendarView;
1664
+ } else if (event.animationName === "show") this.internals.states.delete("transition");
1665
+ }
1666
+ _startTableTransition() {
1667
+ this.internals.states.add("transition");
1668
+ this.shadowRoot?.querySelectorAll("table")?.forEach((e) => e.classList.toggle("sbb-calendar__table-hide"));
1669
+ }
1670
+ render() {
1671
+ return html`<div class="sbb-calendar__wrapper">${this._getView()}</div>`;
1672
+ }
1673
+ };
1674
+ })();
1675
+ //#endregion
1676
+ export { SbbMonthChangeEvent as n, SbbCalendarElement as t };
1677
+
1678
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY2FsZW5kYXIuY29tcG9uZW50LUNzQWw5RUFnLmpzIiwibmFtZXMiOltdLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3NyYy9lbGVtZW50cy9jYWxlbmRhci9jYWxlbmRhci9jYWxlbmRhci5zY3NzP2xpdCZpbmxpbmUiLCIuLi8uLi8uLi9zcmMvZWxlbWVudHMvY2FsZW5kYXIvY2FsZW5kYXIvY2FsZW5kYXIuY29tcG9uZW50LnRzIl0sInNvdXJjZXNDb250ZW50IjpbIkB1c2UgJy4uLy4uL2NvcmUvc3R5bGVzJyBhcyBzYmI7XG5cbjpob3N0IHtcbiAgZGlzcGxheTogYmxvY2s7XG5cbiAgLy8gV2UgYWRkIHdpZHRoIGRlZmluaXRpb24gdG8gaG9zdCwgdG8gbWFrZSBvdmVyd3JpdGluZyBlYXN5IGZvciBjb25zdW1lcnMuXG4gIHdpZHRoOiBtYXgtY29udGVudDtcblxuICAtLXNiYi1jYWxlbmRhci1jZWxsLXNpemU6ICN7c2JiLnB4LXRvLXJlbS1idWlsZCg0NCl9O1xuXG4gIC8vIE1hdGNoIEZpZ21hIGRlc2lnblxuICAtLXNiYi1jYWxlbmRhci1jZWxsLWJvcmRlci1yYWRpdXM6IGNhbGMoXG4gICAgdmFyKC0tc2JiLWJvcmRlci1yYWRpdXMtNHgpICsgdmFyKC0tc2JiLWNhbGVuZGFyLWNlbGwtYm9yZGVyLXdpZHRoKVxuICApO1xuICAtLXNiYi1jYWxlbmRhci1jZWxsLXRyYW5zaXRpb24tZHVyYXRpb246IHZhcihcbiAgICAtLXNiYi1kaXNhYmxlLWFuaW1hdGlvbi1kdXJhdGlvbixcbiAgICB2YXIoLS1zYmItYW5pbWF0aW9uLWR1cmF0aW9uLTJ4KVxuICApO1xuXG4gIEBpbmNsdWRlIHNiYi5tcSgkZnJvbTogc21hbGwpIHtcbiAgICAtLXNiYi1jYWxlbmRhci1jb250cm9sLXZpZXctY2hhbmdlLWhlaWdodDogI3tzYmIucHgtdG8tcmVtLWJ1aWxkKDQ4KX07XG4gIH1cbn1cblxuOmhvc3QoOnN0YXRlKGVuaGFuY2VkKSkge1xuICAtLXNiYi1jYWxlbmRhci1jZWxsLXBhZGRpbmctYmxvY2stZW5kOiB2YXIoLS1zYmItc3BhY2luZy1maXhlZC0zeCk7XG59XG5cbi5zYmItY2FsZW5kYXJfX3dyYXBwZXIge1xuICB3aWR0aDogMTAwJTtcbiAgZGlzcGxheTogYmxvY2s7XG4gIHRyYW5zaXRpb24tZHVyYXRpb246IHZhcigtLXNiYi1jYWxlbmRhci1jZWxsLXRyYW5zaXRpb24tZHVyYXRpb24pO1xufVxuXG4uc2JiLWNhbGVuZGFyX19jb250cm9scyB7XG4gIHdpZHRoOiAxMDAlO1xuICBkaXNwbGF5OiBpbmxpbmUtZmxleDtcbiAgYWxpZ24taXRlbXM6IGNlbnRlcjtcbiAgZ2FwOiB2YXIoLS1zYmItY2FsZW5kYXItY29udHJvbC1nYXApO1xuICBtYXJnaW4tYmxvY2stZW5kOiB2YXIoLS1zYmItY2FsZW5kYXItY29udHJvbC1tYXJnaW4tYmxvY2stZW5kKTtcbn1cblxuLnNiYi1jYWxlbmRhcl9fY29udHJvbHMtbW9udGgge1xuICB3aWR0aDogMTAwJTtcbiAgZGlzcGxheTogZmxleDtcbiAgZ2FwOiB2YXIoLS1zYmItY2FsZW5kYXItdGFibGVzLWdhcCk7XG59XG5cbiNzYmItY2FsZW5kYXJfX2NvbnRyb2xzLXByZXZpb3VzLFxuI3NiYi1jYWxlbmRhcl9fY29udHJvbHMtbmV4dCB7XG4gIC13ZWJraXQtdGFwLWhpZ2hsaWdodC1jb2xvcjogdHJhbnNwYXJlbnQ7XG59XG5cbi5zYmItY2FsZW5kYXJfX2NvbnRyb2xzLWNoYW5nZS1kYXRlIHtcbiAgQGluY2x1ZGUgc2JiLmJ1dHRvbi1yZXNldDtcblxuICBkaXNwbGF5OiBmbGV4O1xuICBhbGlnbi1pdGVtczogY2VudGVyO1xuICBtYXJnaW46IGF1dG87XG4gIGhlaWdodDogdmFyKC0tc2JiLWNhbGVuZGFyLWNvbnRyb2wtdmlldy1jaGFuZ2UtaGVpZ2h0KTtcbiAgZm9udC1zaXplOiB2YXIoLS1zYmItdGV4dC1mb250LXNpemUtcyk7XG4gIGxldHRlci1zcGFjaW5nOiB2YXIoLS1zYmItdHlwby1sZXR0ZXItc3BhY2luZy10ZXh0KTtcbiAgdGV4dC10cmFuc2Zvcm06IGNhcGl0YWxpemU7XG4gIGN1cnNvcjogdmFyKC0tc2JiLWN1cnNvci1wb2ludGVyKTtcbiAgcGFkZGluZy1pbmxpbmU6IHZhcigtLXNiYi1jYWxlbmRhci1jb250cm9sLXZpZXctY2hhbmdlLXBhZGRpbmctaW5saW5lKTtcbiAgYm9yZGVyLXJhZGl1czogdmFyKC0tc2JiLWJvcmRlci1yYWRpdXMtaW5maW5pdHkpO1xuICBiYWNrZ3JvdW5kLWNvbG9yOiB2YXIoLS1zYmItY2FsZW5kYXItY29udHJvbC12aWV3LWNoYW5nZS1iYWNrZ3JvdW5kKTtcbiAgY29sb3I6IHZhcigtLXNiYi1jYWxlbmRhci1jb250cm9sLXZpZXctY2hhbmdlLWNvbG9yKTtcbiAgdHJhbnNpdGlvbi1kdXJhdGlvbjogdmFyKC0tc2JiLWNhbGVuZGFyLWNlbGwtdHJhbnNpdGlvbi1kdXJhdGlvbik7XG4gIHRyYW5zaXRpb24tdGltaW5nLWZ1bmN0aW9uOiB2YXIoLS1zYmItY2FsZW5kYXItY2VsbC10cmFuc2l0aW9uLWVhc2luZy1mdW5jdGlvbik7XG4gIHRyYW5zaXRpb24tcHJvcGVydHk6IGJhY2tncm91bmQtY29sb3IsIHBhZGRpbmctYmxvY2stZW5kO1xuXG4gICY6ZGlzYWJsZWQge1xuICAgIC0tc2JiLWNhbGVuZGFyLWNvbnRyb2wtdmlldy1jaGFuZ2UtYmFja2dyb3VuZDogbGlnaHQtZGFyayhcbiAgICAgIHZhcigtLXNiYi1jb2xvci1taWxrKSxcbiAgICAgIHZhcigtLXNiYi1jb2xvci1hbnRocmFjaXRlKVxuICAgICk7XG4gICAgLS1zYmItY2FsZW5kYXItY29udHJvbC12aWV3LWNoYW5nZS1jb2xvcjogbGlnaHQtZGFyayhcbiAgICAgIHZhcigtLXNiYi1jb2xvci1ncmFuaXRlKSxcbiAgICAgIHZhcigtLXNiYi1jb2xvci1hbHVtaW5pdW0pXG4gICAgKTtcblxuICAgIGN1cnNvcjogdW5zZXQ7XG4gIH1cblxuICAmOmZvY3VzLXZpc2libGUge1xuICAgIEBpbmNsdWRlIHNiYi5mb2N1cy1vdXRsaW5lO1xuXG4gICAgb3V0bGluZS1vZmZzZXQ6IHZhcigtLXNiYi1zcGFjaW5nLWZpeGVkLTF4KTtcbiAgfVxuXG4gIEBpbmNsdWRlIHNiYi5ob3Zlci1tcSB7XG4gICAgJjpub3QoOmFjdGl2ZSwgOmRpc2FibGVkKTpob3ZlciB7XG4gICAgICBwYWRkaW5nLWJsb2NrLWVuZDogdmFyKC0tc2JiLWNhbGVuZGFyLWNlbGwtaG92ZXItc2hpZnQpO1xuICAgIH1cbiAgfVxuXG4gICY6bm90KDpkaXNhYmxlZCk6YWN0aXZlIHtcbiAgICAtLXNiYi1jYWxlbmRhci1jb250cm9sLXZpZXctY2hhbmdlLWJhY2tncm91bmQ6IHZhcigtLXNiYi1iYWNrZ3JvdW5kLWNvbG9yLTMpO1xuICB9XG59XG5cbi5zYmItY2FsZW5kYXJfX3RhYmxlLW1vbnRoLXZpZXcsXG4uc2JiLWNhbGVuZGFyX190YWJsZS15ZWFyLXZpZXcge1xuICAtLXNiYi1jYWxlbmRhci10YWJsZS1jb2x1bW4tc3BhY2VzOiA2O1xufVxuXG4uc2JiLWNhbGVuZGFyX190YWJsZS1vdmVyZmxvdy1icmVhayB7XG4gIC8vIFdlIGhhdmUgdG8gY3V0IHRoZSBuZWdhdGl2ZSBtYXJnaW4gd2hpY2ggaXMgYnVpbHQgYnkgdGhlIGNvbnRhaW5pbmcgc2JiLWNhbGVuZGFyX190YWJsZS1jb250YWluZXIuXG4gIC8vIE92ZXJmbG93OiBoaWRkZW4gd291bGQgY3V0IHRoZSBmb2N1cyBvdXRsaW5lLiBUaGVyZWZvcmUgd2UgdGFrZSBgY29udGFpbmAgaGVyZS5cbiAgY29udGFpbjogbGF5b3V0O1xufVxuXG4uc2JiLWNhbGVuZGFyX190YWJsZS1jb250YWluZXIge1xuICBkaXNwbGF5OiBmbGV4O1xuICBnYXA6IHZhcigtLXNiYi1jYWxlbmRhci10YWJsZXMtZ2FwKTtcbiAgbWFyZ2luLWlubGluZTogdmFyKC0tc2JiLWNhbGVuZGFyLW1hcmdpbik7XG5cbiAgLy8gVGhlIHBhZGRpbmcgb2YgdGhlIGZpcnN0IGFuZCBsYXN0IGNvbHVtbiBzaG91bGQgbm90IGJlIHZpc2libGUgaWYgY2FsZW5kYXIgaXMgc3RyZXRjaGVkLlxuICAvLyBUaGVyZWZvcmUgd2UgbmVlZCBhIG5lZ2F0aXZlIGlubGluZSBtYXJnaW4uXG4gIC8vIEFzIHdlIGRvbid0IHdhbnQgdG8gc3F1ZWV6ZSwgdGhlIG1hcmdpbiBzaG91bGQgbmV2ZXIgYmUgZ3JlYXRlciB0aGFuIHplcm8uXG5cbiAgLy8gTWluIHdpZHRoIGlzIGVxdWFscyB0byB0aGUgbm9ybWFsIHdpZHRoIG9mIHRoZSBjYWxlbmRhclxuICAtLXNiYi1jYWxlbmRhci1taW4td2lkdGg6IGNhbGMoNyAqIHZhcigtLXNiYi1jYWxlbmRhci1jZWxsLXNpemUpKTtcblxuICAvLyBUaGUgb3ZlcmZsb3cgdmFyaWFibGUgaXMgZXF1YWxzIHRoZSBkaWZmZXJlbmNlIGJldHdlZW4gdGhlIGFjdHVhbCB3aWR0aCBhbmQgdGhlIG1pbiB3aWR0aC5cbiAgLS1zYmItY2FsZW5kYXItb3ZlcmZsb3c6IGNhbGMoMTAwJSAtIHZhcigtLXNiYi1jYWxlbmRhci1taW4td2lkdGgpKTtcblxuICAvLyBUaGUgc3RhcnQgb2Zmc2V0IGlzIG5lZ2F0aXZlIG1hcmdpbiB3aGljaCBzaG91bGQgb3ZlcmxhcCB0aGUgcGFyZW50IGNvbnRhaW5lci4gU2hvdWxkIG5ldmVyIGJlIGEgcG9zaXRpdmUgdmFsdWUuXG4gIC0tc2JiLWNhbGVuZGFyLXN0YXJ0LW9mZnNldDogbWluKFxuICAgIDBweCxcbiAgICAtMSAqICh2YXIoLS1zYmItY2FsZW5kYXItb3ZlcmZsb3cpIC8gdmFyKC0tc2JiLWNhbGVuZGFyLXRhYmxlLWNvbHVtbi1zcGFjZXMpKVxuICApO1xuICAtLXNiYi1jYWxlbmRhci1tYXJnaW46IHZhcigtLXNiYi1jYWxlbmRhci1zdGFydC1vZmZzZXQpO1xuXG4gIDpob3N0KDpzdGF0ZSh3aWRlKSkgJiB7XG4gICAgLS1zYmItY2FsZW5kYXItbWluLXdpZHRoOiBjYWxjKFxuICAgICAgMiAqIDcgKiB2YXIoLS1zYmItY2FsZW5kYXItY2VsbC1zaXplKSArIHZhcigtLXNiYi1jYWxlbmRhci10YWJsZXMtZ2FwKVxuICAgICk7XG4gICAgLS1zYmItY2FsZW5kYXItbWFyZ2luOiBjYWxjKDAuNSAqIHZhcigtLXNiYi1jYWxlbmRhci1zdGFydC1vZmZzZXQpKTtcbiAgfVxuXG4gIDpob3N0KFtvcmllbnRhdGlvbj0naG9yaXpvbnRhbCddW3dlZWstbnVtYmVyc10pICYge1xuICAgIC0tc2JiLWNhbGVuZGFyLW1pbi13aWR0aDogY2FsYyg4ICogdmFyKC0tc2JiLWNhbGVuZGFyLWNlbGwtc2l6ZSkpO1xuICB9XG5cbiAgOmhvc3QoW29yaWVudGF0aW9uPSdob3Jpem9udGFsJ11bd2Vlay1udW1iZXJzXTpzdGF0ZSh3aWRlKSkgJiB7XG4gICAgLS1zYmItY2FsZW5kYXItbWluLXdpZHRoOiBjYWxjKFxuICAgICAgMiAqIDggKiB2YXIoLS1zYmItY2FsZW5kYXItY2VsbC1zaXplKSArIHZhcigtLXNiYi1jYWxlbmRhci10YWJsZXMtZ2FwKVxuICAgICk7XG4gIH1cblxuICAvLyBUaGUgY29udGFpbmVyJ3MgbWluLXdpZHRoIGlzIHNldCBpbiB2ZXJ0aWNhbCBtb2RlO1xuICAvLyB0aGUgdmFsdWUgYGNhbGMoNyAqIHZhcigtLXNiYi1jYWxlbmRhci1jZWxsLXNpemUpKWAgaXMgZmluZVxuICAvLyBjb25zaWRlcmluZyB0aGF0IHRoZSBtYXhpbXVtIG51bWJlciBvZiB3ZWVrcyBpbiBhIG1vbnRoIGlzIDYsIHBsdXMgMSBmb3IgdGhlIGhlYWRlciBhdCB0aGUgdGFibGUncyBsZWZ0IHNpZGUuXG4gIDpob3N0KFtvcmllbnRhdGlvbj0ndmVydGljYWwnXSkgJiB7XG4gICAgbWluLXdpZHRoOiB2YXIoLS1zYmItY2FsZW5kYXItbWluLXdpZHRoKTtcblxuICAgIC8vIFRoZSB2YXJpYWJsZSB0aGF0IGRlZmluZXMgdGhlIG1hcmdpbiBpcyBzZXQgdG8gMCB0byBjb3JyZWN0bHkgZGlzcGxheSB0aGUgdGFibGUgaWYgY29uc3VtZXJzIHNldCBhIGN1c3RvbSB3aWR0aC5cbiAgICAtLXNiYi1jYWxlbmRhci1zdGFydC1vZmZzZXQ6IDA7XG4gIH1cblxuICAvLyBUaGUgbWluLXdpZHRoIGluIHdpZGUgaXMgY2FsY3VsYXRlZCBhcyAobWF4IG51bWJlciBvZiB3ZWVrcyBpbiBhIG1vbnRoKSAqIChkaXNwbGF5ZWQgbW9udGhzKSArIChoZWFkZXIgY29sdW1uKSA9IDYgKiAyICsgMSA9IDEzLlxuICA6aG9zdChbb3JpZW50YXRpb249J3ZlcnRpY2FsJ106c3RhdGUod2lkZSkpICYge1xuICAgIC0tc2JiLWNhbGVuZGFyLW1pbi13aWR0aDogY2FsYyhcbiAgICAgIDEzICogdmFyKC0tc2JiLWNhbGVuZGFyLWNlbGwtc2l6ZSkgKyB2YXIoLS1zYmItY2FsZW5kYXItdGFibGVzLWdhcClcbiAgICApO1xuICB9XG59XG5cbi5zYmItY2FsZW5kYXJfX3RhYmxlIHtcbiAgd2lkdGg6IDEwMCU7XG4gIGJvcmRlci1jb2xsYXBzZTogY29sbGFwc2U7XG4gIGhlaWdodDogbWF4LWNvbnRlbnQ7XG5cbiAgYW5pbWF0aW9uOiB7XG4gICAgbmFtZTogc2hvdztcbiAgICBkdXJhdGlvbjogdmFyKC0tc2JiLWNhbGVuZGFyLXRhYmxlLWFuaW1hdGlvbi1kdXJhdGlvbik7XG4gIH1cblxuICAmLnNiYi1jYWxlbmRhcl9fdGFibGUtaGlkZSB7XG4gICAgLS1zYmItY2FsZW5kYXItY2VsbC10cmFuc2l0aW9uLWR1cmF0aW9uOiAwbXM7XG5cbiAgICBhbmltYXRpb246IHtcbiAgICAgIG5hbWU6IGhpZGU7XG4gICAgICBkdXJhdGlvbjogdmFyKC0tc2JiLWNhbGVuZGFyLXRhYmxlLWFuaW1hdGlvbi1kdXJhdGlvbik7XG4gICAgfVxuICB9XG5cbiAgOmhvc3QoOm5vdCg6c3RhdGUod2lkZSkpKSAmIHtcbiAgICAvLyBEdWUgdG8gYSBTYWZhcmkgaU9TIHJlbmRlcmluZyBidWcgd2UgbmVlZCB0byBkZWZpbmUgbWluLXdpZHRoIGFzIHdlbGwuXG4gICAgLy8gT3RoZXJ3aXNlLCBhZnRlciBvcmllbnRhdGlvbiBjaGFuZ2UsIHRoZXJlIGlzIGEgd3Jvbmcgd2lkdGggaWYgcGxhY2VkIGluIGFuIHNiYi1kaWFsb2cuXG4gICAgbWluLXdpZHRoOiAxMDAlO1xuICB9XG59XG5cbi5zYmItY2FsZW5kYXJfX3RhYmxlLWhlYWRlciB7XG4gIC8vIERlcGVuZGluZyB3aGVyZSBpbiB0aGUgRE9NIGl0J3MgdXNlZCwgaXQncyBwb3NzaWJsZSB0aGF0IGJyb3dzZXIgZGVmYXVsdHMgc2V0IHRleHQtYWxpZ24gdG8gbGVmdC5cbiAgdGV4dC1hbGlnbjogY2VudGVyO1xufVxuXG4uc2JiLWNhbGVuZGFyX190YWJsZS1ib2R5IHtcbiAgLy8gRGVwZW5kaW5nIHdoZXJlIGluIHRoZSBET00gaXQncyB1c2VkLCBpdCdzIHBvc3NpYmxlIHRoYXQgYnJvd3NlciBkZWZhdWx0cyBzZXQgdGV4dC1hbGlnbiB0byBsZWZ0LlxuICB0ZXh0LWFsaWduOiBjZW50ZXI7XG59XG5cbi5zYmItY2FsZW5kYXJfX3RhYmxlLWhlYWRlci1jZWxsLFxuLnNiYi1jYWxlbmRhcl9fdGFibGUtaGVhZGVyLWNlbGwtdmVydGljYWwge1xuICB3aWR0aDogdmFyKC0tc2JiLWNhbGVuZGFyLWNlbGwtc2l6ZSk7XG4gIGNvbG9yOiB2YXIoLS1zYmItY2FsZW5kYXItaGVhZGVyLWNvbG9yKTtcbiAgcGFkZGluZzogMDtcbiAgZm9udC1zaXplOiB2YXIoLS1zYmItdGV4dC1mb250LXNpemUteHMpO1xuICBmb250LXdlaWdodDogbm9ybWFsO1xuICBsZXR0ZXItc3BhY2luZzogdmFyKC0tc2JiLXR5cG8tbGV0dGVyLXNwYWNpbmctdGV4dCk7XG5cbiAgOmhvc3QoOm5vdChbbXVsdGlwbGVdKSkgJiB7XG4gICAgaGVpZ2h0OiB2YXIoLS1zYmItY2FsZW5kYXItY2VsbC1zaXplKTtcbiAgfVxufVxuXG4uc2JiLWNhbGVuZGFyX190YWJsZS1oZWFkZXItY2VsbC12ZXJ0aWNhbCB7XG4gIHBhZGRpbmctYmxvY2stZW5kOiB2YXIoLS1zYmItY2FsZW5kYXItY2VsbC1wYWRkaW5nLWJsb2NrLWVuZCk7XG59XG5cbi5zYmItY2FsZW5kYXJfX3RhYmxlLWRhdGEge1xuICBwb3NpdGlvbjogcmVsYXRpdmU7XG4gIHBhZGRpbmc6IDA7XG4gIHRleHQtYWxpZ246IGNlbnRlcjtcbn1cblxuLnNiYi1jYWxlbmRhcl9fZGF5LWNlbGwge1xuICBwYWRkaW5nLWJsb2NrLWVuZDogdmFyKC0tc2JiLWNhbGVuZGFyLWNlbGwtcGFkZGluZy1ibG9jay1lbmQpO1xufVxuXG5zYmItY2FsZW5kYXItZGF5IHtcbiAgOmhvc3QoOm5vdCg6c3RhdGUoZW5oYW5jZWQpKSkgJiB7XG4gICAgLS1zYmItY2FsZW5kYXItY2VsbC1qdXN0aWZ5LWNvbnRlbnQ6IGNlbnRlcjtcbiAgICAtLXNiYi1jYWxlbmRhci1kYXktaGVpZ2h0OiAje3NiYi5weC10by1yZW0tYnVpbGQoNDQpfTtcbiAgICAtLXNiYi1jYWxlbmRhci1kYXktZXh0cmEtZGlzcGxheTogbm9uZTtcbiAgICAtLXNiYi1jYWxlbmRhci1kYXktdmFsdWUtaGVpZ2h0OiB1bnNldDtcbiAgICAtLXNiYi1jYWxlbmRhci1kYXktY3Jvc3NlZC1vdXQtdG9wOiA1MCU7XG4gICAgLS1zYmItY2FsZW5kYXItZGF5LWNyb3NzZWQtb3V0LXRyYW5zbGF0ZTogdHJhbnNsYXRlKC01MCUsIC01MCUpIHJvdGF0ZSgtNDVkZWcpO1xuICB9XG59XG5cbkBrZXlmcmFtZXMgc2hvdyB7XG4gIGZyb20ge1xuICAgIG9wYWNpdHk6IDA7XG4gICAgdHJhbnNmb3JtOiB0cmFuc2xhdGVZKHZhcigtLXNiYi1jYWxlbmRhci10YWJsZS1hbmltYXRpb24tc2hpZnQpKTtcbiAgfVxuXG4gIHRvIHtcbiAgICBvcGFjaXR5OiAxO1xuICAgIHRyYW5zZm9ybTogdHJhbnNsYXRlWSgwJSk7XG4gIH1cbn1cblxuQGtleWZyYW1lcyBoaWRlIHtcbiAgZnJvbSB7XG4gICAgb3BhY2l0eTogMTtcbiAgICB0cmFuc2Zvcm06IHRyYW5zbGF0ZVkoMCUpO1xuICB9XG5cbiAgdG8ge1xuICAgIG9wYWNpdHk6IDA7XG4gICAgdHJhbnNmb3JtOiB0cmFuc2xhdGVZKHZhcigtLXNiYi1jYWxlbmRhci10YWJsZS1hbmltYXRpb24tc2hpZnQpKTtcbiAgfVxufVxuIiwiaW1wb3J0IHtcbiAgdHlwZSBDU1NSZXN1bHRHcm91cCxcbiAgaHRtbCxcbiAgaXNTZXJ2ZXIsXG4gIG5vdGhpbmcsXG4gIHR5cGUgUHJvcGVydHlWYWx1ZXMsXG4gIHR5cGUgVGVtcGxhdGVSZXN1bHQsXG59IGZyb20gJ2xpdCc7XG5pbXBvcnQgeyBwcm9wZXJ0eSwgc3RhdGUgfSBmcm9tICdsaXQvZGVjb3JhdG9ycy5qcyc7XG5cbmltcG9ydCB7IFNiYlNlY29uZGFyeUJ1dHRvbkVsZW1lbnQgfSBmcm9tICcuLi8uLi9idXR0b24ucHVyZS50cyc7XG5pbXBvcnQgeyBpc0Fycm93S2V5T3JQYWdlS2V5c1ByZXNzZWQgfSBmcm9tICcuLi8uLi9jb3JlL2ExMXkudHMnO1xuaW1wb3J0IHsgU2JiRWxlbWVudCB9IGZyb20gJy4uLy4uL2NvcmUvYmFzZS1lbGVtZW50cy50cyc7XG5pbXBvcnQgeyByZWFkQ29uZmlnIH0gZnJvbSAnLi4vLi4vY29yZS9jb25maWcudHMnO1xuaW1wb3J0IHtcbiAgU2JiTGFuZ3VhZ2VDb250cm9sbGVyLFxuICBTYmJNZWRpYU1hdGNoZXJDb250cm9sbGVyLFxuICBTYmJNZWRpYVF1ZXJ5QnJlYWtwb2ludExhcmdlQW5kQWJvdmUsXG59IGZyb20gJy4uLy4uL2NvcmUvY29udHJvbGxlcnMudHMnO1xuaW1wb3J0IHR5cGUgeyBEYXRlQWRhcHRlciB9IGZyb20gJy4uLy4uL2NvcmUvZGF0ZXRpbWUudHMnO1xuaW1wb3J0IHtcbiAgREFZU19QRVJfUk9XLFxuICBkZWZhdWx0RGF0ZUFkYXB0ZXIsXG4gIE1PTkRBWSxcbiAgTU9OVEhTX1BFUl9QQUdFLFxuICBNT05USFNfUEVSX1JPVyxcbiAgVEhVUlNEQVksXG4gIFRVRVNEQVksXG4gIFdFRE5FU0RBWSxcbiAgWUVBUlNfUEVSX1BBR0UsXG4gIFlFQVJTX1BFUl9ST1csXG59IGZyb20gJy4uLy4uL2NvcmUvZGF0ZXRpbWUudHMnO1xuaW1wb3J0IHsgZm9yY2VUeXBlLCBoYW5kbGVEaXN0aW5jdENoYW5nZSwgcGxhaW5EYXRlIH0gZnJvbSAnLi4vLi4vY29yZS9kZWNvcmF0b3JzLnRzJztcbmltcG9ydCB7XG4gIGkxOG5DYWxlbmRhckRhdGVTZWxlY3Rpb24sXG4gIGkxOG5DYWxlbmRhcldlZWtOdW1iZXIsXG4gIGkxOG5OZXh0TW9udGgsXG4gIGkxOG5OZXh0WWVhcixcbiAgaTE4bk5leHRZZWFyUmFuZ2UsXG4gIGkxOG5QcmV2aW91c01vbnRoLFxuICBpMThuUHJldmlvdXNZZWFyLFxuICBpMThuUHJldmlvdXNZZWFyUmFuZ2UsXG4gIGkxOG5ZZWFyTW9udGhTZWxlY3Rpb24sXG59IGZyb20gJy4uLy4uL2NvcmUvaTE4bi50cyc7XG5pbXBvcnQgdHlwZSB7IFNiYk9yaWVudGF0aW9uIH0gZnJvbSAnLi4vLi4vY29yZS9pbnRlcmZhY2VzLnRzJztcbmltcG9ydCB7IGJveFNpemluZ1N0eWxlcyB9IGZyb20gJy4uLy4uL2NvcmUvc3R5bGVzLnRzJztcbmltcG9ydCB7IFNiYkNhbGVuZGFyRGF5RWxlbWVudCB9IGZyb20gJy4uL2NhbGVuZGFyLWRheS9jYWxlbmRhci1kYXkuY29tcG9uZW50LnRzJztcbmltcG9ydCB7IFNiYkNhbGVuZGFyTW9udGhFbGVtZW50IH0gZnJvbSAnLi4vY2FsZW5kYXItbW9udGgvY2FsZW5kYXItbW9udGguY29tcG9uZW50LnRzJztcbmltcG9ydCB7IFNiYkNhbGVuZGFyV2Vla2RheUVsZW1lbnQgfSBmcm9tICcuLi9jYWxlbmRhci13ZWVrZGF5L2NhbGVuZGFyLXdlZWtkYXkuY29tcG9uZW50LnRzJztcbmltcG9ydCB7IFNiYkNhbGVuZGFyV2Vla251bWJlckVsZW1lbnQgfSBmcm9tICcuLi9jYWxlbmRhci13ZWVrbnVtYmVyL2NhbGVuZGFyLXdlZWtudW1iZXIuY29tcG9uZW50LnRzJztcbmltcG9ydCB7IFNiYkNhbGVuZGFyWWVhckVsZW1lbnQgfSBmcm9tICcuLi9jYWxlbmRhci15ZWFyL2NhbGVuZGFyLXllYXIuY29tcG9uZW50LnRzJztcbmltcG9ydCB0eXBlIHsgU2JiQ2FsZW5kYXJDZWxsQmFzZUVsZW1lbnQgfSBmcm9tICcuLi9jb21tb24udHMnO1xuXG5pbXBvcnQgc3R5bGUgZnJvbSAnLi9jYWxlbmRhci5zY3NzP2xpdCZpbmxpbmUnO1xuXG5pbXBvcnQgJy4uLy4uL2ljb24udHMnO1xuaW1wb3J0ICcuLi8uLi9zY3JlZW4tcmVhZGVyLW9ubHkudHMnO1xuXG4vLyBUT0RPKGJyZWFraW5nLWNoYW5nZSk6IFJlbW92ZSBjYWxsIHRvIGRlZmluZS5cblNiYlNlY29uZGFyeUJ1dHRvbkVsZW1lbnQuZGVmaW5lKCk7XG5cblNiYkNhbGVuZGFyRGF5RWxlbWVudC5kZWZpbmUoKTtcblNiYkNhbGVuZGFyTW9udGhFbGVtZW50LmRlZmluZSgpO1xuU2JiQ2FsZW5kYXJZZWFyRWxlbWVudC5kZWZpbmUoKTtcblNiYkNhbGVuZGFyV2Vla2RheUVsZW1lbnQuZGVmaW5lKCk7XG5TYmJDYWxlbmRhcldlZWtudW1iZXJFbGVtZW50LmRlZmluZSgpO1xuXG5leHBvcnQgY2xhc3MgU2JiTW9udGhDaGFuZ2VFdmVudCBleHRlbmRzIEV2ZW50IHtcbiAgcHJpdmF0ZSByZWFkb25seSBfcmFuZ2U6IHJlYWRvbmx5IERheVtdO1xuXG4gIHB1YmxpYyBnZXQgcmFuZ2UoKTogcmVhZG9ubHkgRGF5W10ge1xuICAgIHJldHVybiB0aGlzLl9yYW5nZTtcbiAgfVxuXG4gIHB1YmxpYyBjb25zdHJ1Y3RvcihyYW5nZTogcmVhZG9ubHkgRGF5W10pIHtcbiAgICBzdXBlcignbW9udGhjaGFuZ2UnLCB7IGJ1YmJsZXM6IHRydWUsIGNvbXBvc2VkOiB0cnVlIH0pO1xuICAgIHRoaXMuX3JhbmdlID0gT2JqZWN0LmZyZWV6ZShyYW5nZSB8fCBbXSk7XG4gIH1cbn1cblxuLyoqXG4gKiBQYXJhbWV0ZXJzIG5lZWRlZCBpbiB5ZWFyIGFuZCBtb250aCB2aWV3cyB0byBjb3JyZWN0bHkgY2FsY3VsYXRlIHRoZSBuZXh0IGVsZW1lbnQgaW4ga2V5Ym9hcmQgbmF2aWdhdGlvbi5cbiAqXG4gKiBUaGUgY2VsbCdzIGluZGV4IGFuZCB0aGUgZWxlbWVudCdzIGluZGV4IGluIGl0cyBtb250aCAvIHllYXIgYmF0Y2ggbXVzdCBiZSBkaXN0aW5ndWlzaGVkOlxuICogdGhlIGZpcnN0IGlzIHRoZSBpbmRleCBvZiB0aGUgZWxlbWVudCBpbiB0aGUgYXJyYXkgb2YgYWxsIHRoZSByZW5kZXJlZCBjZWxscywgd2hpbGUgdGhlIHNlY29uZCBpcyB0aGUgaW5kZXggb2YgdGhlIGVsZW1lbnQgcmVsYXRpdmUgdG8gaXRzIHRhYmxlLlxuICogSW4gbm9uLXdpZGUgbW9kZSwgdGhlIHd0byBhcmUgdGhlIHNhbWUsIHdoaWxlIGluIHdpZGUgbW9kZSB0aGUgY2VsbCdzIGluZGV4IGNhbiBnbyBmcm9tIDAgdG8gNDcgZm9yIHllYXJzIGFuZCBmcm9tIDAgdG8gMjMgZm9yIG1vbnRocyxcbiAqIHdoaWxlIHRoZSBlbGVtZW50IGluZGV4IGdvZXMgZnJvbSAwIHRvLCByZXNwZWN0aXZlbHksIDIzIGFuZCAxMS5cbiAqL1xuaW50ZXJmYWNlIENhbGVuZGFyS2V5Ym9hcmROYXZpZ2F0aW9uTW9udGhZZWFyVmlld3NQYXJhbWV0ZXJzIHtcbiAgLyoqIFRoZSBlbGVtZW50IGluZGV4IHdpdGhpbiBpdHMgeWVhciBvciBtb250aCByYW5nZS4gKi9cbiAgZWxlbWVudEluZGV4Rm9yV2lkZU1vZGU6IG51bWJlcjtcbiAgLyoqIEluIHdpZGUgbW9kZSwgdGhlIGluZGV4IG9mIHRoZSBmaXJzdCBlbGVtZW50IGluIHRoZSBzZWNvbmQgcGFuZWwsIG9yLCBhbHRlcm5hdGl2ZWx5LCB0aGUgbnVtYmVyIG9mIGVsZW1lbnRzIGluIHRoZSBmaXJzdCBwYW5lbC4gKi9cbiAgb2Zmc2V0Rm9yV2lkZU1vZGU6IG51bWJlcjtcbiAgLyoqIFRoZSBpbmRleCBvZiB0aGUgbGFzdCBlbGVtZW50IHdpdGhpbiB0aGUgZWxlbWVudCdzIG1vbnRoIChvciB5ZWFyIHJhbmdlKS4gKi9cbiAgbGFzdEVsZW1lbnRJbmRleEZvcldpZGVNb2RlOiBudW1iZXI7XG4gIC8qKiBUaGUgbnVtYmVyIG9mIGNlbGxzIGRpc3BsYXllZCBpbiBhIHNpbmdsZSByb3csIGRlcGVuZGluZyBvbiB0aGUgcmVuZGVyZWQgdmlldy4gKi9cbiAgdmVydGljYWxPZmZzZXQ6IG51bWJlcjtcbn1cblxuLyoqXG4gKiBQYXJhbWV0ZXJzIG5lZWRlZCBpbiBkYXkgdmlldyB0byBjb3JyZWN0bHkgY2FsY3VsYXRlIHRoZSBuZXh0IGVsZW1lbnQgaW4ga2V5Ym9hcmQgbmF2aWdhdGlvbi5cbiAqXG4gKiBJbiBvcmllbnRhdGlvbj0ndmVydGljYWwnLCBpdCdzIG5vdCBwb3NzaWJsZSB0byByZWx5IG9uIGFueSBhcnJheS9pbmRleCB0byBjYWxjdWxhdGUgdGhlIGVsZW1lbnQgdG8gbmF2aWdhdGUgdG8sXG4gKiBzbyBjYWxjdWxhdGlvbnMgb24gZGF0ZXMgbXVzdCBiZSBkb25lLCB3aGljaCBzaG91bGQgY29uc2lkZXIgdmlldyBib3VuZGFyaWVzLCBvZmZzZXRzIGFuZCBtb250aCdzIGxlbmd0aC5cbiAqL1xuaW50ZXJmYWNlIENhbGVuZGFyS2V5Ym9hcmROYXZpZ2F0aW9uRGF5Vmlld1BhcmFtZXRlcnMge1xuICAvKiogVGhlIGZpcnN0IGRheSByZW5kZXJlZC4gKi9cbiAgZmlyc3REYXlJblZpZXc6IHN0cmluZyB8IG51bGw7XG4gIC8qKiBUaGUgbGFzdCBkYXkgcmVuZGVyZWQuIEl0IGRlcGVuZHMgb24gdGhlICd3aWRlJyB2YWx1ZS4gKi9cbiAgbGFzdERheUluVmlldzogc3RyaW5nIHwgbnVsbDtcbiAgLyoqIFRoZSBvZmZzZXQgZnJvbSB0aGUgZmlyc3QgZGF5IG9mIHRoZSB3ZWVrIChNb25kYXkpIG9mIHRoZSBmaXJzdCByZW5kZXJlZCBtb250aC4gKi9cbiAgZmlyc3RNb250aE9mZnNldDogbnVtYmVyO1xuICAvKiogVGhlIG51bWJlciBvZiBkYXlzIGluIHRoZSBmaXJzdCByZW5kZXJlZCBtb250aC4gKi9cbiAgZmlyc3RNb250aExlbmd0aDogbnVtYmVyO1xuICAvKiogVGhlIG9mZnNldCBmcm9tIHRoZSBmaXJzdCBkYXkgb2YgdGhlIHdlZWsgKE1vbmRheSkgb2YgdGhlIHNlY29uZCByZW5kZXJlZCBtb250aC4gSWYgd2lkZSBpcyBmYWxzZSwgaXQncyBlcXVhbCB0byB6ZXJvLiAqL1xuICBzZWNvbmRNb250aE9mZnNldDogbnVtYmVyO1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIERheTxUID0gRGF0ZT4ge1xuICAvKiogRGF0ZSBhcyBJU08gc3RyaW5nLiAqL1xuICB2YWx1ZTogc3RyaW5nO1xuICBkYXlWYWx1ZTogc3RyaW5nO1xuICBtb250aFZhbHVlOiBzdHJpbmc7XG4gIHllYXJWYWx1ZTogc3RyaW5nO1xuICBkYXRlVmFsdWU6IFQ7XG4gIHdlZWtWYWx1ZTogbnVtYmVyO1xuICB3ZWVrRGF5VmFsdWU6IG51bWJlcjtcbn1cblxuLyoqIEBkZXByZWNhdGVkIHVzZSBNb250aENlbGwgKi9cbmV4cG9ydCBpbnRlcmZhY2UgTW9udGgge1xuICB2YWx1ZTogc3RyaW5nO1xuICBsb25nVmFsdWU6IHN0cmluZztcbiAgbW9udGhWYWx1ZTogbnVtYmVyO1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIE1vbnRoQ2VsbCB7XG4gIHZhbHVlOiBzdHJpbmc7XG4gIG1vbnRoVmFsdWU6IG51bWJlcjtcbn1cblxuZXhwb3J0IGludGVyZmFjZSBXZWVrZGF5IHtcbiAgbG9uZzogc3RyaW5nO1xuICBuYXJyb3c6IHN0cmluZztcbn1cblxuZXhwb3J0IHR5cGUgQ2FsZW5kYXJWaWV3ID0gJ2RheScgfCAnbW9udGgnIHwgJ3llYXInO1xuXG4vKipcbiAqIEl0IGRpc3BsYXlzIGEgY2FsZW5kYXIgd2hpY2ggYWxsb3dzIGNob29zaW5nIGEgZGF0ZS5cbiAqXG4gKiBAc2xvdCAtIFVzZSB0aGUgdW5uYW1lZCBzbG90IHRvIGFkZCBjdXN0b21pemVkIGBzYmItY2FsZW5kYXItZGF5YCBlbGVtZW50cy5cbiAqL1xuZXhwb3J0IGNsYXNzIFNiYkNhbGVuZGFyRWxlbWVudDxUID0gRGF0ZT4gZXh0ZW5kcyBTYmJFbGVtZW50IHtcbiAgcHVibGljIHN0YXRpYyBvdmVycmlkZSByZWFkb25seSBlbGVtZW50TmFtZTogc3RyaW5nID0gJ3NiYi1jYWxlbmRhcic7XG4gIHB1YmxpYyBzdGF0aWMgb3ZlcnJpZGUgc3R5bGVzOiBDU1NSZXN1bHRHcm91cCA9IFtib3hTaXppbmdTdHlsZXMsIHN0eWxlXTtcbiAgcHVibGljIHN0YXRpYyByZWFkb25seSBldmVudHMgPSB7XG4gICAgZGF0ZXNlbGVjdGVkOiAnZGF0ZXNlbGVjdGVkJyxcbiAgICBtb250aGNoYW5nZTogJ21vbnRoY2hhbmdlJyxcbiAgfSBhcyBjb25zdDtcblxuICAvKiogSWYgc2V0IHRvIHRydWUsIHR3byBtb250aHMgYXJlIGRpc3BsYXllZCAqL1xuICBAZm9yY2VUeXBlKClcbiAgQHByb3BlcnR5KHsgdHlwZTogQm9vbGVhbiB9KVxuICBwdWJsaWMgYWNjZXNzb3Igd2lkZTogYm9vbGVhbiA9IGZhbHNlO1xuXG4gIC8qKiBUaGUgaW5pdGlhbCB2aWV3IG9mIHRoZSBjYWxlbmRhciB3aGljaCBzaG91bGQgYmUgZGlzcGxheWVkIG9uIG9wZW5pbmcuICovXG4gIEBwcm9wZXJ0eSgpIHB1YmxpYyBhY2Nlc3NvciB2aWV3OiBDYWxlbmRhclZpZXcgPSAnZGF5JztcblxuICAvKipcbiAgICogVGhlIG1pbmltdW0gdmFsaWQgZGF0ZS4gQWNjZXB0cyBhIGRhdGUgb2JqZWN0IG9yIG51bGwuXG4gICAqIEFjY2VwdHMgYW4gSVNPODYwMSBmb3JtYXR0ZWQgc3RyaW5nIChlLmcuIDIwMjQtMTItMjQpIGFzIGF0dHJpYnV0ZS5cbiAgICovXG4gIEBwbGFpbkRhdGUoKVxuICBAcHJvcGVydHkoKVxuICBwdWJsaWMgYWNjZXNzb3IgbWluOiBUIHwgbnVsbCA9IG51bGw7XG5cbiAgLyoqXG4gICAqIFRoZSBtYXhpbXVtIHZhbGlkIGRhdGUuIEFjY2VwdHMgYSBkYXRlIG9iamVjdCBvciBudWxsLlxuICAgKiBBY2NlcHRzIGFuIElTTzg2MDEgZm9ybWF0dGVkIHN0cmluZyAoZS5nLiAyMDI0LTEyLTI0KSBhcyBhdHRyaWJ1dGUuXG4gICAqL1xuICBAcGxhaW5EYXRlKClcbiAgQHByb3BlcnR5KClcbiAgcHVibGljIGFjY2Vzc29yIG1heDogVCB8IG51bGwgPSBudWxsO1xuXG4gIC8qKiBXaGV0aGVyIHRoZSBjYWxlbmRhciBhbGxvd3MgZm9yIG11bHRpcGxlIGRhdGUgc2VsZWN0aW9uLiAqL1xuICBAZm9yY2VUeXBlKClcbiAgQGhhbmRsZURpc3RpbmN0Q2hhbmdlKChlOiBTYmJDYWxlbmRhckVsZW1lbnQ8VD4sIG5ld1ZhbHVlOiBib29sZWFuKSA9PlxuICAgIGUuX29uTXVsdGlwbGVDaGFuZ2VkKG5ld1ZhbHVlKSxcbiAgKVxuICBAcHJvcGVydHkoeyB0eXBlOiBCb29sZWFuIH0pXG4gIHB1YmxpYyBhY2Nlc3NvciBtdWx0aXBsZTogYm9vbGVhbiA9IGZhbHNlO1xuXG4gIC8qKlxuICAgKiBUaGUgc2VsZWN0ZWQgZGF0ZTogYWNjZXB0cyBhIGRhdGUgb2JqZWN0LCBvciwgaWYgYG11bHRpcGxlYCwgYW4gYXJyYXkgb2YgZGF0ZXMuXG4gICAqL1xuICBAcHJvcGVydHkoKVxuICBwdWJsaWMgc2V0IHNlbGVjdGVkKHZhbHVlOiBUIHwgVFtdIHwgbnVsbCkge1xuICAgIGlmIChBcnJheS5pc0FycmF5KHZhbHVlKSkge1xuICAgICAgdGhpcy5fc2VsZWN0ZWQgPSB2YWx1ZVxuICAgICAgICAubWFwKChkYXRlTGlrZTogVCkgPT5cbiAgICAgICAgICB0aGlzLl9kYXRlQWRhcHRlci5nZXRWYWxpZERhdGVPck51bGwodGhpcy5fZGF0ZUFkYXB0ZXIuZGVzZXJpYWxpemUoZGF0ZUxpa2UpKSxcbiAgICAgICAgKVxuICAgICAgICAuZmlsdGVyKChkYXRlOiBUIHwgbnVsbCk6IGRhdGUgaXMgVCA9PiBkYXRlICE9PSBudWxsKVxuICAgICAgICAuZmlsdGVyKFxuICAgICAgICAgIChkYXRlOiBUKSA9PlxuICAgICAgICAgICAgIXRoaXMuX2lzRGF5SW5SYW5nZSh0aGlzLl9kYXRlQWRhcHRlci50b0lzbzg2MDEoZGF0ZSkpIHx8IHRoaXMuX2RhdGVGaWx0ZXIoZGF0ZSksXG4gICAgICAgICk7XG4gICAgfSBlbHNlIHtcbiAgICAgIGNvbnN0IHNlbGVjdGVkRGF0ZSA9IHRoaXMuX2RhdGVBZGFwdGVyLmdldFZhbGlkRGF0ZU9yTnVsbChcbiAgICAgICAgdGhpcy5fZGF0ZUFkYXB0ZXIuZGVzZXJpYWxpemUodmFsdWUpLFxuICAgICAgKTtcbiAgICAgIGlmIChcbiAgICAgICAgISFzZWxlY3RlZERhdGUgJiZcbiAgICAgICAgKCF0aGlzLl9pc0RheUluUmFuZ2UodGhpcy5fZGF0ZUFkYXB0ZXIudG9Jc284NjAxKHNlbGVjdGVkRGF0ZSkpIHx8XG4gICAgICAgICAgdGhpcy5fZGF0ZUZpbHRlcihzZWxlY3RlZERhdGUpKVxuICAgICAgKSB7XG4gICAgICAgIHRoaXMuX3NlbGVjdGVkID0gc2VsZWN0ZWREYXRlO1xuICAgICAgfSBlbHNlIHtcbiAgICAgICAgdGhpcy5fc2VsZWN0ZWQgPSBudWxsO1xuICAgICAgfVxuICAgIH1cbiAgfVxuICBwdWJsaWMgZ2V0IHNlbGVjdGVkKCk6IFQgfCBUW10gfCBudWxsIHtcbiAgICByZXR1cm4gdGhpcy5fc2VsZWN0ZWQ7XG4gIH1cbiAgQHN0YXRlKCkgcHJpdmF0ZSBhY2Nlc3NvciBfc2VsZWN0ZWQ6IFQgfCBUW10gfCBudWxsID0gbnVsbDtcblxuICAvKiogQSBmdW5jdGlvbiB1c2VkIHRvIGZpbHRlciBvdXQgZGF0ZXMuICovXG4gIEBwcm9wZXJ0eSh7IGF0dHJpYnV0ZTogJ2RhdGUtZmlsdGVyJyB9KVxuICBwdWJsaWMgYWNjZXNzb3IgZGF0ZUZpbHRlcjogKChkYXRlOiBUIHwgbnVsbCkgPT4gYm9vbGVhbikgfCBudWxsID0gbnVsbDtcblxuICAvKiogVGhlIG9yaWVudGF0aW9uIG9mIGRheXMgaW4gdGhlIGNhbGVuZGFyLiAqL1xuICBAcHJvcGVydHkoeyByZWZsZWN0OiB0cnVlIH0pIHB1YmxpYyBhY2Nlc3NvciBvcmllbnRhdGlvbjogU2JiT3JpZW50YXRpb24gPSAnaG9yaXpvbnRhbCc7XG5cbiAgLyoqIFdoZXRoZXIgaXQgaGFzIHRvIGRpc3BsYXkgdGhlIHdlZWsgbnVtYmVycyBpbiBhZGRpdGlvbiB0byB3ZWVrIGRheXMuICovXG4gIEBmb3JjZVR5cGUoKVxuICBAcHJvcGVydHkoeyBhdHRyaWJ1dGU6ICd3ZWVrLW51bWJlcnMnLCB0eXBlOiBCb29sZWFuIH0pXG4gIHB1YmxpYyBhY2Nlc3NvciB3ZWVrTnVtYmVyczogYm9vbGVhbiA9IGZhbHNlO1xuXG4gIHByaXZhdGUgX2RhdGVBZGFwdGVyOiBEYXRlQWRhcHRlcjxUPiA9IHJlYWRDb25maWcoKS5kYXRldGltZT8uZGF0ZUFkYXB0ZXIgPz8gZGVmYXVsdERhdGVBZGFwdGVyO1xuXG4gIC8qKiBUaGUgY3VycmVudGx5IGFjdGl2ZSBkYXRlLiAqL1xuICBAc3RhdGUoKSBwcml2YXRlIGFjY2Vzc29yIF9hY3RpdmVEYXRlOiBUID0gdGhpcy5fZGF0ZUFkYXB0ZXIudG9kYXkoKTtcblxuICAvKiogVGhlIGN1cnJlbnQgd2lkZSBwcm9wZXJ0eSBjb25zaWRlcmluZyBwcm9wZXJ0eSB2YWx1ZSBhbmQgYnJlYWtwb2ludHMuIEZyb20gemVybyB0byBzbWFsbCBgd2lkZWAgaGFzIGFsd2F5cyB0byBiZSBmYWxzZS4gKi9cbiAgQHN0YXRlKClcbiAgcHJpdmF0ZSBzZXQgX3dpZGUod2lkZTogYm9vbGVhbikge1xuICAgIHRoaXMudG9nZ2xlU3RhdGUoJ3dpZGUnLCB3aWRlKTtcbiAgICB0aGlzLl93aWRlSW50ZXJuYWwgPSB3aWRlO1xuICB9XG4gIHByaXZhdGUgZ2V0IF93aWRlKCk6IGJvb2xlYW4ge1xuICAgIHJldHVybiB0aGlzLl93aWRlSW50ZXJuYWw7XG4gIH1cbiAgLy8gV2Uga2VlcCB0aGUgc3RhdGUgaW4gYSBmaWVsZCBiZWNhdXNlIG9mIHRoZSBXZWJLaXQgYnVnIGh0dHBzOi8vYnVncy53ZWJraXQub3JnL3Nob3dfYnVnLmNnaT9pZD0zMDM0NjcuXG4gIC8vIFRPRE86IHJlLWNoZWNrIHdoZXRoZXIgZmllbGQgaXMgbmVlZGVkXG4gIHByaXZhdGUgX3dpZGVJbnRlcm5hbDogYm9vbGVhbiA9IGZhbHNlO1xuXG4gIEBzdGF0ZSgpIHByaXZhdGUgYWNjZXNzb3IgX2NhbGVuZGFyVmlldzogQ2FsZW5kYXJWaWV3ID0gJ2RheSc7XG5cbiAgcHJpdmF0ZSBfbmV4dENhbGVuZGFyVmlldzogQ2FsZW5kYXJWaWV3ID0gJ2RheSc7XG5cbiAgLyoqIEluZm9ybWF0aW9uIGFib3V0IHRoZSByZW5kZXJlZCBkYXkgdmlldzsgdXNlZCBpbiBrZXlib2FyZCBuYXZpZ2F0aW9uLiAqL1xuICBwcml2YXRlIF9rZXlib2FyZE5hdmlnYXRpb25EYXlWaWV3UGFyYW1ldGVyczogQ2FsZW5kYXJLZXlib2FyZE5hdmlnYXRpb25EYXlWaWV3UGFyYW1ldGVycyA9IHtcbiAgICBmaXJzdERheUluVmlldzogbnVsbCxcbiAgICBsYXN0RGF5SW5WaWV3OiBudWxsLFxuICAgIGZpcnN0TW9udGhPZmZzZXQ6IDAsXG4gICAgZmlyc3RNb250aExlbmd0aDogMCxcbiAgICBzZWNvbmRNb250aE9mZnNldDogMCxcbiAgfTtcblxuICAvKiogQSBsaXN0IG9mIGRheXMsIGluIHR3byBmb3JtYXRzIChsb25nIGFuZCBzaW5nbGUgY2hhcikuICovXG4gIHByaXZhdGUgX3dlZWtkYXlzITogV2Vla2RheVtdO1xuXG4gIC8qKiBHcmlkIG9mIGNhbGVuZGFyIGNlbGxzIHJlcHJlc2VudGluZyB0aGUgZGF0ZXMgb2YgdGhlIG1vbnRoLiAqL1xuICBwcml2YXRlIF93ZWVrczogRGF5PFQ+W11bXSA9IFtdO1xuXG4gIC8qKiBHcmlkIG9mIGNhbGVuZGFyIGNlbGxzIHJlcHJlc2VudGluZyBtb250aHMuICovXG4gIHByaXZhdGUgX21vbnRocyE6IE1vbnRoQ2VsbFtdW107XG5cbiAgLyoqIEdyaWQgb2YgY2FsZW5kYXIgY2VsbHMgcmVwcmVzZW50aW5nIHllYXJzLiAqL1xuICBwcml2YXRlIF95ZWFycyE6IG51bWJlcltdW107XG5cbiAgLyoqIEdyaWQgb2YgY2FsZW5kYXIgY2VsbHMgcmVwcmVzZW50aW5nIHllYXJzIGZvciB0aGUgd2lkZSB2aWV3LiAqL1xuICBwcml2YXRlIF9uZXh0TW9udGhZZWFycyE6IG51bWJlcltdW107XG5cbiAgLyoqIEdyaWQgb2YgY2FsZW5kYXIgY2VsbHMgcmVwcmVzZW50aW5nIHRoZSBkYXRlcyBvZiB0aGUgbmV4dCBtb250aC4gKi9cbiAgcHJpdmF0ZSBfbmV4dE1vbnRoV2Vla3MhOiBEYXk8VD5bXVtdO1xuXG4gIC8qKiBBbiBhcnJheSBjb250YWluaW5nIGFsbCB0aGUgbW9udGggbmFtZXMgaW4gdGhlIGN1cnJlbnQgbGFuZ3VhZ2UuICovXG4gIHByaXZhdGUgX21vbnRoTmFtZXM6IHN0cmluZ1tdID0gdGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0TW9udGhOYW1lcygnbG9uZycpO1xuXG4gIC8qKiBBbiBhcnJheSBjb250YWluaW5nIHRoZSB3ZWVrcycgbnVtYmVycyBmb3IgdGhlIGN1cnJlbnQgbW9udGguICovXG4gIHByaXZhdGUgX3dlZWtOdW1iZXJzITogbnVtYmVyW107XG5cbiAgLyoqIEFuIGFycmF5IGNvbnRhaW5pbmcgdGhlIHdlZWtzJyBudW1iZXJzIGZvciB0aGUgbmV4dCBtb250aCBpbiB3aWRlIG1vZGUuICovXG4gIHByaXZhdGUgX25leHRNb250aFdlZWtOdW1iZXJzITogbnVtYmVyW107XG5cbiAgcHJpdmF0ZSBfZW5oYW5jZWRWYXJpYW50OiBib29sZWFuID0gZmFsc2U7XG5cbiAgLyoqIEEgbGlzdCBvZiBjYWxlbmRhcidzIGNlbGxzIGNvcnJlc3BvbmRpbmcgdG8gZGF5cywgbW9udGhzIG9yIHllYXJzIGRlcGVuZGluZyBvbiB0aGUgdmlldy4gKi9cbiAgcHJpdmF0ZSBnZXQgX2NlbGxzKCk6IFNiYkNhbGVuZGFyQ2VsbEJhc2VFbGVtZW50W10ge1xuICAgIHJldHVybiBBcnJheS5mcm9tPFNiYkNhbGVuZGFyQ2VsbEJhc2VFbGVtZW50PihcbiAgICAgICh0aGlzLl9jYWxlbmRhclZpZXcgPT09ICdkYXknXG4gICAgICAgID8gKEFycmF5LmZyb20odGhpcy5zaGFkb3dSb290IS5xdWVyeVNlbGVjdG9yQWxsKCdzbG90JykpLmZsYXRNYXAoKGU6IEhUTUxTbG90RWxlbWVudCkgPT5cbiAgICAgICAgICAgIGUuYXNzaWduZWRFbGVtZW50cyh7IGZsYXR0ZW46IHRydWUgfSksXG4gICAgICAgICAgKSBhcyBTYmJDYWxlbmRhckRheUVsZW1lbnRbXSlcbiAgICAgICAgOiB0aGlzLnNoYWRvd1Jvb3Q/LnF1ZXJ5U2VsZWN0b3JBbGwoYHNiYi1jYWxlbmRhci0ke3RoaXMuX2NhbGVuZGFyVmlld31gKSkgPz8gW10sXG4gICAgKTtcbiAgfVxuXG4gIC8qKiBUaGUgY2hvc2VuIHllYXIgaW4gdGhlIHllYXIgc2VsZWN0aW9uIHZpZXcuICovXG4gIHByaXZhdGUgX2Nob3NlblllYXI/OiBudW1iZXI7XG5cbiAgLyoqIFRoZSBjaG9zZW4gbW9udGggaW4gdGhlIHllYXIgc2VsZWN0aW9uIHZpZXcuICovXG4gIHByaXZhdGUgX2Nob3Nlbk1vbnRoPzogbnVtYmVyO1xuXG4gIC8qKiBXaGV0aGVyIHRoZSBmb2N1cyBzaG91bGQgYmUgcmVzZXQgb24gZm9jdXNDZWxsLiAqL1xuICBwcml2YXRlIF9yZXNldEZvY3VzID0gZmFsc2U7XG5cbiAgLyoqIFdoZXRoZXIgYW4gZWxlbWVudCBpbnNpZGUgdGhlIGNhbGVuZGFyIGlzIGN1cnJlbnRseSBmb2N1c2VkLiAqL1xuICBwcml2YXRlIF9jb250YWluaW5nRm9jdXMgPSBmYWxzZTtcblxuICBAc3RhdGUoKVxuICBwcml2YXRlIGFjY2Vzc29yIF9pbml0aWFsaXplZCA9IGZhbHNlO1xuXG4gIHByaXZhdGUgX2xhbmd1YWdlID0gbmV3IFNiYkxhbmd1YWdlQ29udHJvbGxlcih0aGlzKS53aXRoSGFuZGxlcigoKSA9PiB7XG4gICAgdGhpcy5fbW9udGhOYW1lcyA9IHRoaXMuX2RhdGVBZGFwdGVyLmdldE1vbnRoTmFtZXMoJ2xvbmcnKTtcbiAgICB0aGlzLl9jcmVhdGVNb250aFJvd3MoKTtcbiAgfSk7XG4gIHByaXZhdGUgX21lZGlhTWF0Y2hlciA9IG5ldyBTYmJNZWRpYU1hdGNoZXJDb250cm9sbGVyKHRoaXMsIHtcbiAgICBbU2JiTWVkaWFRdWVyeUJyZWFrcG9pbnRMYXJnZUFuZEFib3ZlXTogKCkgPT4gdGhpcy5faW5pdCgpLFxuICB9KTtcblxuICBwdWJsaWMgY29uc3RydWN0b3IoKSB7XG4gICAgc3VwZXIoKTtcbiAgICB0aGlzLl9jcmVhdGVNb250aFJvd3MoKTtcbiAgICB0aGlzLl9zZXRXZWVrZGF5cygpO1xuXG4gICAgLy8gV2UgbmVlZCB0byB0cmFjayB0aGUgZm9jdXMgYXMgd2Ugc2hvdWxkIG9ubHkgdGFrZSBmb2N1cyBpbnRvIHRoZSBjYWxlbmRhciwgd2hlbiB0aGVcbiAgICAvLyBmb2N1cyB3YXMgb25jZSBzZXQgaW50byB0aGUgY2FsZW5kYXIuXG4gICAgLy8gRm9yIHNoYWRvdyBET00gY29tcGF0aWJpbGl0eSB3ZSBuZWVkIHRvIHRyYWNrIHRoaXMgcHJvZ3JhbW1hdGljYWxseS5cbiAgICB0aGlzLmFkZEV2ZW50TGlzdGVuZXIoJ2ZvY3VzaW4nLCAoKSA9PiAodGhpcy5fY29udGFpbmluZ0ZvY3VzID0gdHJ1ZSkpO1xuICAgIHRoaXMuYWRkRXZlbnRMaXN0ZW5lcignZm9jdXNvdXQnLCAoKSA9PiAodGhpcy5fY29udGFpbmluZ0ZvY3VzID0gZmFsc2UpKTtcbiAgICB0aGlzLmFkZEV2ZW50TGlzdGVuZXIoJ2NsaWNrJywgKGUpID0+IHtcbiAgICAgIGNvbnN0IGRheSA9IChlLnRhcmdldCBhcyBIVE1MRWxlbWVudCkuY2xvc2VzdDxTYmJDYWxlbmRhckRheUVsZW1lbnQ8VD4+KCdzYmItY2FsZW5kYXItZGF5Jyk7XG4gICAgICBpZiAoZGF5KSB7XG4gICAgICAgIHRoaXMuX3NlbGVjdERhdGUoZGF5LnZhbHVlISk7XG4gICAgICB9XG4gICAgfSk7XG4gICAgdGhpcy5hZGRFdmVudExpc3RlbmVyKCdrZXlkb3duJywgKGUpID0+IHtcbiAgICAgIGlmICgoZS50YXJnZXQgYXMgSFRNTEVsZW1lbnQpLmxvY2FsTmFtZSA9PT0gJ3NiYi1jYWxlbmRhci1kYXknKSB7XG4gICAgICAgIHRoaXMuX2hhbmRsZUtleWJvYXJkRXZlbnQoXG4gICAgICAgICAgZSxcbiAgICAgICAgICB0aGlzLl9tYXBEYXRlVG9EYXkoKGUudGFyZ2V0IGFzIFNiYkNhbGVuZGFyRGF5RWxlbWVudCkudmFsdWUhIGFzIFQpLFxuICAgICAgICApO1xuICAgICAgfVxuICAgIH0pO1xuICB9XG5cbiAgcHJpdmF0ZSBfZGF0ZUZpbHRlcihkYXRlOiBUKTogYm9vbGVhbiB7XG4gICAgcmV0dXJuIHRoaXMuZGF0ZUZpbHRlcj8uKGRhdGUpID8/IHRydWU7XG4gIH1cblxuICAvKiogUmVzZXRzIHRoZSBhY3RpdmUgbW9udGggYWNjb3JkaW5nIHRvIHRoZSBuZXcgc3RhdGUgb2YgdGhlIGNhbGVuZGFyLiAqL1xuICBwdWJsaWMgcmVzZXRQb3NpdGlvbigpOiB2b2lkIHtcbiAgICB0aGlzLl9yZXNldENhbGVuZGFyVmlldygpO1xuICAgIHRoaXMuX2luaXQoKTtcbiAgfVxuXG4gIHB1YmxpYyBvdmVycmlkZSBjb25uZWN0ZWRDYWxsYmFjaygpOiB2b2lkIHtcbiAgICBzdXBlci5jb25uZWN0ZWRDYWxsYmFjaygpO1xuICAgIHRoaXMucmVzZXRQb3NpdGlvbigpO1xuICAgIHRoaXMuc2hhZG93Um9vdD8uYWRkRXZlbnRMaXN0ZW5lcignc2xvdGNoYW5nZScsIHRoaXMuX29uU2xvdENoYW5nZSwgeyBjYXB0dXJlOiB0cnVlIH0pO1xuICB9XG5cbiAgcHVibGljIG92ZXJyaWRlIGRpc2Nvbm5lY3RlZENhbGxiYWNrKCk6IHZvaWQge1xuICAgIHN1cGVyLmRpc2Nvbm5lY3RlZENhbGxiYWNrKCk7XG4gICAgdGhpcy5zaGFkb3dSb290Py5yZW1vdmVFdmVudExpc3RlbmVyKCdzbG90Y2hhbmdlJywgdGhpcy5fb25TbG90Q2hhbmdlLCB7IGNhcHR1cmU6IHRydWUgfSk7XG4gIH1cblxuICAvKiogQGludGVybmFsICovXG4gIHB1YmxpYyBvdmVycmlkZSBmb2N1cygpOiB2b2lkIHtcbiAgICB0aGlzLl9yZXNldEZvY3VzID0gdHJ1ZTtcbiAgICB0aGlzLl9mb2N1c0NlbGwoKTtcbiAgfVxuXG4gIHByb3RlY3RlZCBvdmVycmlkZSB3aWxsVXBkYXRlKGNoYW5nZWRQcm9wZXJ0aWVzOiBQcm9wZXJ0eVZhbHVlczx0aGlzPik6IHZvaWQge1xuICAgIHN1cGVyLndpbGxVcGRhdGUoY2hhbmdlZFByb3BlcnRpZXMpO1xuXG4gICAgaWYgKCF0aGlzLl9pbml0aWFsaXplZCkge1xuICAgICAgcmV0dXJuO1xuICAgIH1cblxuICAgIGlmIChjaGFuZ2VkUHJvcGVydGllcy5oYXMoJ3dpZGUnKSB8fCBjaGFuZ2VkUHJvcGVydGllcy5oYXMoJ29yaWVudGF0aW9uJykpIHtcbiAgICAgIHRoaXMucmVzZXRQb3NpdGlvbigpO1xuICAgIH1cblxuICAgIGlmIChjaGFuZ2VkUHJvcGVydGllcy5oYXMoJ3ZpZXcnKSkge1xuICAgICAgdGhpcy5fc2V0Q2hvc2VuWWVhcigpO1xuICAgICAgdGhpcy5fY2hvc2VuTW9udGggPSB1bmRlZmluZWQ7XG4gICAgICB0aGlzLl9uZXh0Q2FsZW5kYXJWaWV3ID0gdGhpcy5fY2FsZW5kYXJWaWV3ID0gdGhpcy52aWV3O1xuICAgIH1cbiAgfVxuXG4gIHByb3RlY3RlZCBvdmVycmlkZSB1cGRhdGVkKGNoYW5nZWRQcm9wZXJ0aWVzOiBQcm9wZXJ0eVZhbHVlczx0aGlzPik6IHZvaWQge1xuICAgIHN1cGVyLnVwZGF0ZWQoY2hhbmdlZFByb3BlcnRpZXMpO1xuICAgIC8vIFRoZSBjYWxlbmRhciBuZWVkcyB0byBjYWxjdWxhdGUgdGFiLWluZGV4ZXMgb24gZmlyc3QgcmVuZGVyLFxuICAgIC8vIGFuZCBldmVyeSB0aW1lIGEgZGF0ZSBpcyBzZWxlY3RlZCBvciB0aGUgbW9udGggdmlldyBjaGFuZ2VzLlxuICAgIHRoaXMuX3NldFRhYkluZGV4KCk7XG5cbiAgICAvLyBXaGVuIGNoYW5naW5nIHZpZXcgdG8geWVhci9tb250aCwgdGhlIHRhYmluZGV4IGlzIGNoYW5nZWQsIGJ1dCB0aGUgZm9jdXNlZCBlbGVtZW50IGlzIGdldHRpbmcgbG9zdC5cbiAgICAvLyBXZSBuZWVkIHRvIGNhbGwgYF9mb2N1c0NlbGwoKWAgbWV0aG9kIGV4cGxpY2l0bHkgdG8gY29ycmVjdGx5IHNldCB0aGUgZm9jdXMuXG4gICAgdGhpcy5fZm9jdXNDZWxsKCk7XG4gIH1cblxuICBwcml2YXRlIF9vblNsb3RDaGFuZ2UgPSAoKTogdm9pZCA9PiB7XG4gICAgdGhpcy5fZW5oYW5jZWRWYXJpYW50ID0gQXJyYXkuZnJvbSh0aGlzLmNoaWxkcmVuKS5zb21lKFxuICAgICAgKGMpID0+IGMubG9jYWxOYW1lID09PSAnc2JiLWNhbGVuZGFyLWRheScsXG4gICAgKTtcbiAgICB0aGlzLnRvZ2dsZVN0YXRlKCdlbmhhbmNlZCcsIHRoaXMuX2VuaGFuY2VkVmFyaWFudCk7XG4gICAgdGhpcy5fc2V0VGFiSW5kZXgoKTtcbiAgfTtcblxuICAvKipcbiAgICogVGhlIGBfc2VsZWN0ZWRgIHN0YXRlIHNob3VsZCBiZSBhZGFwdGVkIHdoZW4gdGhlIGBtdWx0aXBsZWAgcHJvcGVydHkgY2hhbmdlczpcbiAgICogICAtIGlmIGl0IGNoYW5nZXMgdG8gdHJ1ZSwgdGhlICdfc2VsZWN0ZWQnIGlzIHNldCB0byBhbiBhcnJheTtcbiAgICogICAtIGlmIGl0IGNoYW5nZXMgdG8gZmFsc2UsIHRoZSBmaXJzdCBhdmFpbGFibGUgb3B0aW9uIGlzIHNldCBhcyAndmFsdWUnIG90aGVyd2lzZSBpdCdzIHNldCB0byBudWxsLlxuICAgKi9cbiAgcHJpdmF0ZSBfb25NdWx0aXBsZUNoYW5nZWQoaXNNdWx0aXBsZTogYm9vbGVhbik6IHZvaWQge1xuICAgIGlmIChpc011bHRpcGxlICYmICFBcnJheS5pc0FycmF5KHRoaXMuX3NlbGVjdGVkKSkge1xuICAgICAgdGhpcy5fc2VsZWN0ZWQgPSB0aGlzLl9zZWxlY3RlZCA/IFt0aGlzLl9zZWxlY3RlZCBhcyBUXSA6IFtdO1xuICAgIH1cbiAgICBpZiAoIWlzTXVsdGlwbGUgJiYgQXJyYXkuaXNBcnJheSh0aGlzLl9zZWxlY3RlZCkpIHtcbiAgICAgIHRoaXMuX3NlbGVjdGVkID0gKHRoaXMuX3NlbGVjdGVkIGFzIFRbXSkubGVuZ3RoID8gKHRoaXMuX3NlbGVjdGVkIGFzIFRbXSlbMF0gOiBudWxsO1xuICAgIH1cbiAgfVxuXG4gIC8qKiBJbml0aWFsaXplcyB0aGUgY29tcG9uZW50LiAqL1xuICBwcml2YXRlIF9pbml0KGFjdGl2ZURhdGU/OiBUKTogdm9pZCB7XG4gICAgLy8gRHVlIHRvIGl0cyBjb21wbGV4aXR5LCB0aGUgY2FsZW5kYXIgaXMgb25seSBpbml0aWFsaXplZCBvbiBjbGllbnQgc2lkZVxuICAgIGlmIChpc1NlcnZlcikge1xuICAgICAgcmV0dXJuO1xuICAgIH0gZWxzZSBpZiAodGhpcy5oeWRyYXRpb25SZXF1aXJlZCkge1xuICAgICAgdGhpcy5oeWRyYXRpb25Db21wbGV0ZS50aGVuKCgpID0+IHRoaXMuX2luaXQoKSk7XG4gICAgICByZXR1cm47XG4gICAgfVxuXG4gICAgaWYgKGFjdGl2ZURhdGUpIHtcbiAgICAgIHRoaXMuX2Fzc2lnbkFjdGl2ZURhdGUoYWN0aXZlRGF0ZSk7XG4gICAgfVxuICAgIHRoaXMuX3dpZGUgPVxuICAgICAgKHRoaXMuX21lZGlhTWF0Y2hlci5tYXRjaGVzKFNiYk1lZGlhUXVlcnlCcmVha3BvaW50TGFyZ2VBbmRBYm92ZSkgPz8gZmFsc2UpICYmIHRoaXMud2lkZTtcbiAgICB0aGlzLl93ZWVrcyA9IHRoaXMuX2NyZWF0ZVdlZWtSb3dzKHRoaXMuX2FjdGl2ZURhdGUpO1xuICAgIHRoaXMuX3llYXJzID0gdGhpcy5fY3JlYXRlWWVhclJvd3MoKTtcbiAgICB0aGlzLl93ZWVrTnVtYmVycyA9IHRoaXMuX3dlZWtzXG4gICAgICAuZmxhdCgpXG4gICAgICAuc29ydCgoYSwgYikgPT4gYS52YWx1ZS5sb2NhbGVDb21wYXJlKGIudmFsdWUpKVxuICAgICAgLm1hcCgoZGF5OiBEYXk8VD4pID0+IGRheS53ZWVrVmFsdWUpXG4gICAgICAuZmlsdGVyKCh2LCBpLCBhKSA9PiBhLmluZGV4T2YodikgPT09IGkpO1xuICAgIHRoaXMuX25leHRNb250aFdlZWtzID0gW1tdXTtcbiAgICB0aGlzLl9uZXh0TW9udGhZZWFycyA9IFtbXV07XG4gICAgaWYgKHRoaXMuX3dpZGUpIHtcbiAgICAgIGNvbnN0IG5leHRNb250aERhdGUgPSB0aGlzLl9kYXRlQWRhcHRlci5hZGRDYWxlbmRhck1vbnRocyh0aGlzLl9hY3RpdmVEYXRlLCAxKTtcbiAgICAgIHRoaXMuX25leHRNb250aFdlZWtzID0gdGhpcy5fY3JlYXRlV2Vla1Jvd3MobmV4dE1vbnRoRGF0ZSwgdHJ1ZSk7XG4gICAgICB0aGlzLl9uZXh0TW9udGhZZWFycyA9IHRoaXMuX2NyZWF0ZVllYXJSb3dzKFlFQVJTX1BFUl9QQUdFKTtcbiAgICAgIHRoaXMuX25leHRNb250aFdlZWtOdW1iZXJzID0gdGhpcy5fbmV4dE1vbnRoV2Vla3NcbiAgICAgICAgLmZsYXQoKVxuICAgICAgICAuc29ydCgoYSwgYikgPT4gYS52YWx1ZS5sb2NhbGVDb21wYXJlKGIudmFsdWUpKVxuICAgICAgICAubWFwKChkYXk6IERheTxUPikgPT4gZGF5LndlZWtWYWx1ZSlcbiAgICAgICAgLmZpbHRlcigodiwgaSwgYSkgPT4gYS5pbmRleE9mKHYpID09PSBpKTtcbiAgICB9XG4gICAgdGhpcy5faW5pdGlhbGl6ZWQgPSB0cnVlO1xuICB9XG5cbiAgLyoqIEZvY3VzZXMgb24gYSBkYXkgY2VsbCBwcmlvcml0aXppbmcgdGhlIHNlbGVjdGVkIGRheSwgdGhlIGN1cnJlbnQgZGF5LCBhbmQgbGFzdGx5LCB0aGUgZmlyc3Qgc2VsZWN0YWJsZSBkYXkuICovXG4gIHByaXZhdGUgX2ZvY3VzQ2VsbCgpOiB2b2lkIHtcbiAgICBpZiAodGhpcy5fcmVzZXRGb2N1cykge1xuICAgICAgdGhpcy5fZ2V0Rmlyc3RGb2N1c2FibGUoKT8uZm9jdXMoKTtcbiAgICAgIHRoaXMuX3Jlc2V0Rm9jdXMgPSBmYWxzZTtcbiAgICB9XG4gIH1cblxuICAvKiogQ3JlYXRlcyB0aGUgYXJyYXkgb2Ygd2Vla2RheXMuICovXG4gIHByaXZhdGUgX3NldFdlZWtkYXlzKCk6IHZvaWQge1xuICAgIGNvbnN0IG5hcnJvd1dlZWtkYXlzOiBzdHJpbmdbXSA9IHRoaXMuX2RhdGVBZGFwdGVyLmdldERheU9mV2Vla05hbWVzKCduYXJyb3cnKTtcbiAgICBjb25zdCBsb25nV2Vla2RheXM6IHN0cmluZ1tdID0gdGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0RGF5T2ZXZWVrTmFtZXMoJ2xvbmcnKTtcbiAgICBjb25zdCB3ZWVrZGF5czogV2Vla2RheVtdID0gbG9uZ1dlZWtkYXlzLm1hcCgobG9uZzogc3RyaW5nLCBpOiBudW1iZXIpID0+ICh7XG4gICAgICBsb25nLFxuICAgICAgbmFycm93OiBuYXJyb3dXZWVrZGF5c1tpXSxcbiAgICB9KSk7XG5cbiAgICAvLyBSb3RhdGVzIHRoZSBsYWJlbHMgZm9yIGRheXMgb2YgdGhlIHdlZWsgYmFzZWQgb24gdGhlIGNvbmZpZ3VyZWQgZmlyc3QgZGF5IG9mIHRoZSB3ZWVrLlxuICAgIGNvbnN0IGZpcnN0RGF5T2ZXZWVrOiBudW1iZXIgPSB0aGlzLl9kYXRlQWRhcHRlci5nZXRGaXJzdERheU9mV2VlaygpO1xuICAgIHRoaXMuX3dlZWtkYXlzID0gd2Vla2RheXMuc2xpY2UoZmlyc3REYXlPZldlZWspLmNvbmNhdCh3ZWVrZGF5cy5zbGljZSgwLCBmaXJzdERheU9mV2VlaykpO1xuICB9XG5cbiAgLyoqIENyZWF0ZXMgdGhlIHJvd3MgYWxvbmcgdGhlIGhvcml6b250YWwgZGlyZWN0aW9uIGFuZCBzZXRzIHRoZSBwYXJhbWV0ZXJzIHVzZWQgaW4ga2V5Ym9hcmQgbmF2aWdhdGlvbi4gKi9cbiAgcHJpdmF0ZSBfY3JlYXRlV2Vla1Jvd3ModmFsdWU6IFQsIGlzU2Vjb25kTW9udGhJblZpZXcgPSBmYWxzZSk6IERheTxUPltdW10ge1xuICAgIGNvbnN0IGRheXNJbk1vbnRoOiBudW1iZXIgPSB0aGlzLl9kYXRlQWRhcHRlci5nZXROdW1EYXlzSW5Nb250aCh2YWx1ZSk7XG4gICAgY29uc3Qgd2Vla09mZnNldDogbnVtYmVyID0gdGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0Rmlyc3RXZWVrT2Zmc2V0KHZhbHVlKTtcbiAgICBpZiAoIWlzU2Vjb25kTW9udGhJblZpZXcpIHtcbiAgICAgIHRoaXMuX2tleWJvYXJkTmF2aWdhdGlvbkRheVZpZXdQYXJhbWV0ZXJzLmZpcnN0TW9udGhMZW5ndGggPSBkYXlzSW5Nb250aDtcbiAgICAgIHRoaXMuX2tleWJvYXJkTmF2aWdhdGlvbkRheVZpZXdQYXJhbWV0ZXJzLmZpcnN0TW9udGhPZmZzZXQgPSB3ZWVrT2Zmc2V0O1xuICAgICAgdGhpcy5fa2V5Ym9hcmROYXZpZ2F0aW9uRGF5Vmlld1BhcmFtZXRlcnMuZmlyc3REYXlJblZpZXcgPSB0aGlzLl9kYXRlQWRhcHRlci50b0lzbzg2MDEoXG4gICAgICAgIHRoaXMuX2RhdGVBZGFwdGVyLmNyZWF0ZURhdGUoXG4gICAgICAgICAgdGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0WWVhcih2YWx1ZSksXG4gICAgICAgICAgdGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0TW9udGgodmFsdWUpLFxuICAgICAgICAgIDEsXG4gICAgICAgICksXG4gICAgICApO1xuICAgICAgdGhpcy5fa2V5Ym9hcmROYXZpZ2F0aW9uRGF5Vmlld1BhcmFtZXRlcnMubGFzdERheUluVmlldyA9IHRoaXMuX2RhdGVBZGFwdGVyLnRvSXNvODYwMShcbiAgICAgICAgdGhpcy5fZGF0ZUFkYXB0ZXIuY3JlYXRlRGF0ZShcbiAgICAgICAgICB0aGlzLl9kYXRlQWRhcHRlci5nZXRZZWFyKHZhbHVlKSxcbiAgICAgICAgICB0aGlzLl9kYXRlQWRhcHRlci5nZXRNb250aCh2YWx1ZSksXG4gICAgICAgICAgZGF5c0luTW9udGgsXG4gICAgICAgICksXG4gICAgICApO1xuICAgIH0gZWxzZSB7XG4gICAgICB0aGlzLl9rZXlib2FyZE5hdmlnYXRpb25EYXlWaWV3UGFyYW1ldGVycy5zZWNvbmRNb250aE9mZnNldCA9IHdlZWtPZmZzZXQ7XG4gICAgICB0aGlzLl9rZXlib2FyZE5hdmlnYXRpb25EYXlWaWV3UGFyYW1ldGVycy5sYXN0RGF5SW5WaWV3ID0gdGhpcy5fZGF0ZUFkYXB0ZXIudG9Jc284NjAxKFxuICAgICAgICB0aGlzLl9kYXRlQWRhcHRlci5jcmVhdGVEYXRlKFxuICAgICAgICAgIHRoaXMuX2RhdGVBZGFwdGVyLmdldFllYXIodmFsdWUpLFxuICAgICAgICAgIHRoaXMuX2RhdGVBZGFwdGVyLmdldE1vbnRoKHZhbHVlKSxcbiAgICAgICAgICBkYXlzSW5Nb250aCxcbiAgICAgICAgKSxcbiAgICAgICk7XG4gICAgfVxuICAgIHJldHVybiB0aGlzLm9yaWVudGF0aW9uID09PSAnaG9yaXpvbnRhbCdcbiAgICAgID8gdGhpcy5fY3JlYXRlV2Vla1Jvd3NIb3Jpem9udGFsKHZhbHVlLCBkYXlzSW5Nb250aCwgd2Vla09mZnNldClcbiAgICAgIDogdGhpcy5fY3JlYXRlV2Vla1Jvd3NWZXJ0aWNhbCh2YWx1ZSwgZGF5c0luTW9udGgsIHdlZWtPZmZzZXQpO1xuICB9XG5cbiAgLyoqXG4gICAqIENyZWF0ZXMgdGhlIHJvd3MgZm9yIGVhY2ggd2VlayBpbiBvcmllbnRhdGlvbj0naG9yaXpvbnRhbCcuXG4gICAqXG4gICAqIEl0ZXJhdGVzIHRocm91Z2ggdGhlIGRheXMgb2YgdGhlIG1vbnRocywgY3JlYXRlcyBhIERheSBvYmplY3QgZm9yIGVhY2ggYW5kIHB1c2hlcyBpdCBpbnRvIGFuZCBhcnJheS5cbiAgICogRWFjaCBzZXZlbiBkYXlzIChjb25zaWRlcmluZyB0aGUgb2Zmc2V0IGF0IHRoZSBiZWdpbm5pbmcgb2YgdGhlIG1vbnRoKSByZXN0YXJ0cyBmcm9tIGFuIGVtcHR5IGFycmF5LlxuICAgKlxuICAgKiBUaGUgcmVzdWx0IGlzIGEgbWF0cml4IGluIHdoaWNoIGV2ZXJ5IHJvdyBpcyBhIHdlZWsgKG9yIHBhcnQgb2YgaXQsIGNvbnNpZGVyaW5nIG9mZnNldCkuXG4gICAqL1xuICBwcml2YXRlIF9jcmVhdGVXZWVrUm93c0hvcml6b250YWwodmFsdWU6IFQsIGRheXNJbk1vbnRoOiBudW1iZXIsIHdlZWtPZmZzZXQ6IG51bWJlcik6IERheTxUPltdW10ge1xuICAgIGNvbnN0IHdlZWtzOiBEYXk8VD5bXVtdID0gW1tdXTtcbiAgICBmb3IgKGxldCBpID0gMCwgY2VsbCA9IHdlZWtPZmZzZXQ7IGkgPCBkYXlzSW5Nb250aDsgaSsrLCBjZWxsKyspIHtcbiAgICAgIGlmIChjZWxsID09PSBEQVlTX1BFUl9ST1cpIHtcbiAgICAgICAgd2Vla3MucHVzaChbXSk7XG4gICAgICAgIGNlbGwgPSAwO1xuICAgICAgfVxuICAgICAgY29uc3QgZGF0ZSA9IHRoaXMuX2RhdGVBZGFwdGVyLmNyZWF0ZURhdGUoXG4gICAgICAgIHRoaXMuX2RhdGVBZGFwdGVyLmdldFllYXIodmFsdWUpLFxuICAgICAgICB0aGlzLl9kYXRlQWRhcHRlci5nZXRNb250aCh2YWx1ZSksXG4gICAgICAgIGkgKyAxLFxuICAgICAgKSE7XG4gICAgICB3ZWVrc1t3ZWVrcy5sZW5ndGggLSAxXS5wdXNoKHRoaXMuX21hcERhdGVUb0RheShkYXRlKSk7XG4gICAgfVxuICAgIHJldHVybiB3ZWVrcztcbiAgfVxuXG4gIC8qKlxuICAgKiBDcmVhdGVzIHRoZSByb3dzIGZvciBlYWNoIHdlZWsgaW4gb3JpZW50YXRpb249J3ZlcnRpY2FsJy5cbiAgICpcbiAgICogQ3JlYXRlcyBhIG1hdHJpeCB3aXRoIHNldmVuIGVtcHR5IHJvd3MuXG4gICAqIEl0ZXJhdGVzIHRocm91Z2ggdGhlIGRheXMgb2YgdGhlIG1vbnRocywgY3JlYXRlcyBhIERheSBvYmplY3QgZm9yIGVhY2hcbiAgICogYW5kIHB1c2hlcyBpdCBpbnRvIHRoZSBjb3JyZWN0IGFycmF5IGNvbnNpZGVyaW5nIHRoZSBvZmZzZXQgYXQgdGhlIGJlZ2lubmluZyBvZiB0aGUgbW9udGguXG4gICAqIEVhY2ggc2V2ZW4gZGF5cyAoaW5jbHVkaW5nIG9mZnNldCkgcmVzdGFydHMgZnJvbSB0aGUgZmlyc3QuXG4gICAqXG4gICAqIFRoZSByZXN1bHQgaXMgYSBtYXRyaXggaW4gd2hpY2ggZXZlcnkgcm93IGlzIGEgc2V0IG9mIHdlZWtkYXlzLCBzbzpcbiAgICogLSByb3cgMDogYWxsIHRoZSBNb25kYXlzO1xuICAgKiAtIHJvdyAxOiBhbGwgdGhlIFR1ZXNkYXlzO1xuICAgKiAtIC4uLlxuICAgKiAtIHJvdyA3OiBhbGwgdGhlIFN1bmRheXMuXG4gICAqL1xuICBwcml2YXRlIF9jcmVhdGVXZWVrUm93c1ZlcnRpY2FsKHZhbHVlOiBULCBkYXlzSW5Nb250aDogbnVtYmVyLCB3ZWVrT2Zmc2V0OiBudW1iZXIpOiBEYXk8VD5bXVtdIHtcbiAgICBjb25zdCB3ZWVrczogRGF5PFQ+W11bXSA9IEFycmF5LmZyb20oeyBsZW5ndGg6IERBWVNfUEVSX1JPVyB9LCAoKSA9PiBbXSk7XG4gICAgZm9yIChsZXQgaSA9IDAsIGNlbGwgPSB3ZWVrT2Zmc2V0OyBpIDwgZGF5c0luTW9udGg7IGkrKywgY2VsbCsrKSB7XG4gICAgICBpZiAoY2VsbCA9PT0gREFZU19QRVJfUk9XKSB7XG4gICAgICAgIGNlbGwgPSAwO1xuICAgICAgfVxuICAgICAgY29uc3QgZGF0ZSA9IHRoaXMuX2RhdGVBZGFwdGVyLmNyZWF0ZURhdGUoXG4gICAgICAgIHRoaXMuX2RhdGVBZGFwdGVyLmdldFllYXIodmFsdWUpLFxuICAgICAgICB0aGlzLl9kYXRlQWRhcHRlci5nZXRNb250aCh2YWx1ZSksXG4gICAgICAgIGkgKyAxLFxuICAgICAgKSE7XG4gICAgICB3ZWVrc1tjZWxsXS5wdXNoKHRoaXMuX21hcERhdGVUb0RheShkYXRlKSk7XG4gICAgfVxuICAgIHJldHVybiB3ZWVrcztcbiAgfVxuXG4gIHByaXZhdGUgX21hcERhdGVUb0RheShkYXRlOiBUKTogRGF5PFQ+IHtcbiAgICBjb25zdCBpc29EYXRlID0gdGhpcy5fZGF0ZUFkYXB0ZXIudG9Jc284NjAxKGRhdGUpO1xuICAgIHJldHVybiB7XG4gICAgICB2YWx1ZTogaXNvRGF0ZSxcbiAgICAgIGRhdGVWYWx1ZTogZGF0ZSxcbiAgICAgIGRheVZhbHVlOiBTdHJpbmcodGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0RGF0ZShkYXRlKSksXG4gICAgICBtb250aFZhbHVlOiBTdHJpbmcodGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0TW9udGgoZGF0ZSkpLFxuICAgICAgeWVhclZhbHVlOiBTdHJpbmcodGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0WWVhcihkYXRlKSksXG4gICAgICAvLyBUT0RPOiBJbXByb3ZlIHBlcmZvcm1hbmNlIG9mIHRoaXMsIGJ5IGtlZXBpbmcgdHJhY2sgb2YgdGhlXG4gICAgICAvLyB3ZWVrIG51bWJlciB3aGlsZSBpdGVyYXRpbmcgdGhyb3VnaCB0aGUgZGF5cy5cbiAgICAgIHdlZWtWYWx1ZTogdGhpcy5fZ2V0V2VlayhkYXRlKSxcbiAgICAgIHdlZWtEYXlWYWx1ZTogdGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0RGF5T2ZXZWVrKGRhdGUpLFxuICAgIH07XG4gIH1cblxuICBwcml2YXRlIF9nZXRXZWVrKGRhdGU6IFQpOiBudW1iZXIge1xuICAgIGNvbnN0IGZpcnN0RGF5T2ZZZWFyID0gdGhpcy5fZGF0ZUFkYXB0ZXIuY3JlYXRlRGF0ZSh0aGlzLl9kYXRlQWRhcHRlci5nZXRZZWFyKGRhdGUpLCAxLCAxKTtcbiAgICBjb25zdCB3ZWVrZGF5ID0gdGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0RGF5T2ZXZWVrKGZpcnN0RGF5T2ZZZWFyKTtcblxuICAgIGxldCB3ZWVrSW5kZXggPSBbTU9OREFZLCBUVUVTREFZLCBXRURORVNEQVksIFRIVVJTREFZXS5pbmNsdWRlcyh3ZWVrZGF5KSA/IDEgOiAwO1xuICAgIGxldCB3ZWVrU3RhcnQgPSB0aGlzLl9kYXRlQWRhcHRlci5hZGRDYWxlbmRhckRheXMoXG4gICAgICBmaXJzdERheU9mWWVhcixcbiAgICAgIHRoaXMuX2RhdGVBZGFwdGVyLmdldEZpcnN0V2Vla09mZnNldChmaXJzdERheU9mWWVhcikgKiAtMSxcbiAgICApO1xuICAgIHdoaWxlICh0aGlzLl9kYXRlQWRhcHRlci5jb21wYXJlRGF0ZSh3ZWVrU3RhcnQsIGRhdGUpIDw9IDApIHtcbiAgICAgIGNvbnN0IHdlZWtFbmQgPSB0aGlzLl9kYXRlQWRhcHRlci5hZGRDYWxlbmRhckRheXMod2Vla1N0YXJ0LCA2KTtcbiAgICAgIGlmICh0aGlzLl9kYXRlQWRhcHRlci5jb21wYXJlRGF0ZShkYXRlLCB3ZWVrRW5kKSA8PSAwKSB7XG4gICAgICAgIHJldHVybiB3ZWVrSW5kZXggPiAwXG4gICAgICAgICAgPyB3ZWVrSW5kZXhcbiAgICAgICAgICA6IHRoaXMuX2dldFdlZWsodGhpcy5fZGF0ZUFkYXB0ZXIuYWRkQ2FsZW5kYXJEYXlzKGZpcnN0RGF5T2ZZZWFyLCAtMSkpO1xuICAgICAgfVxuICAgICAgd2Vla1N0YXJ0ID0gdGhpcy5fZGF0ZUFkYXB0ZXIuYWRkQ2FsZW5kYXJEYXlzKHdlZWtTdGFydCwgREFZU19QRVJfUk9XKTtcbiAgICAgIHdlZWtJbmRleCsrO1xuICAgIH1cblxuICAgIHRocm93IG5ldyBFcnJvcignVGhlIHByb3ZpZGVkIGRhdGUgaXMgaW52YWxpZCcpO1xuICB9XG5cbiAgLyoqIEZvcmNlIHRoZSBjb252ZXJzaW9uIHRvIElTTzg2MDEgZm9ybWF0dGVkIHN0cmluZy4gKi9cbiAgcHJpdmF0ZSBfbWFwVmFsdWVUb0lTT0RhdGUodmFsdWU6IHN0cmluZyB8IERhdGUpOiBzdHJpbmcge1xuICAgIHJldHVybiB0eXBlb2YgdmFsdWUgPT09ICdzdHJpbmcnID8gdmFsdWUgOiB0aGlzLl9kYXRlQWRhcHRlci50b0lzbzg2MDEodmFsdWUgYXMgVCk7XG4gIH1cblxuICAvKiogQ3JlYXRlcyB0aGUgcm93cyBmb3IgdGhlIG1vbnRoIHNlbGVjdGlvbiB2aWV3LiAqL1xuICBwcml2YXRlIF9jcmVhdGVNb250aFJvd3MoKTogdm9pZCB7XG4gICAgY29uc3QgbW9udGhzOiBNb250aENlbGxbXSA9IG5ldyBBcnJheSgxMikuZmlsbChudWxsKS5tYXAoXG4gICAgICAoXywgaTogbnVtYmVyKTogTW9udGhDZWxsID0+ICh7XG4gICAgICAgIHZhbHVlOiBTdHJpbmcoaSArIDEpLnBhZFN0YXJ0KDIsICcwJyksXG4gICAgICAgIG1vbnRoVmFsdWU6IGkgKyAxLFxuICAgICAgfSksXG4gICAgKTtcbiAgICBjb25zdCByb3dzOiBudW1iZXIgPSAxMiAvIE1PTlRIU19QRVJfUk9XO1xuICAgIGNvbnN0IG1vbnRoQXJyYXk6IE1vbnRoQ2VsbFtdW10gPSBbXTtcbiAgICBmb3IgKGxldCBpOiBudW1iZXIgPSAwOyBpIDwgcm93czsgaSsrKSB7XG4gICAgICBtb250aEFycmF5LnB1c2gobW9udGhzLnNsaWNlKE1PTlRIU19QRVJfUk9XICogaSwgTU9OVEhTX1BFUl9ST1cgKiAoaSArIDEpKSk7XG4gICAgfVxuICAgIHRoaXMuX21vbnRocyA9IG1vbnRoQXJyYXk7XG4gIH1cblxuICAvKiogQ3JlYXRlcyB0aGUgcm93cyBmb3IgdGhlIHllYXIgc2VsZWN0aW9uIHZpZXcuICovXG4gIHByaXZhdGUgX2NyZWF0ZVllYXJSb3dzKG9mZnNldDogbnVtYmVyID0gMCk6IG51bWJlcltdW10ge1xuICAgIGNvbnN0IHN0YXJ0VmFsdWVZZWFyVmlldzogbnVtYmVyID0gdGhpcy5fZ2V0U3RhcnRWYWx1ZVllYXJWaWV3KCk7XG4gICAgY29uc3QgYWxsWWVhcnM6IG51bWJlcltdID0gbmV3IEFycmF5KFlFQVJTX1BFUl9QQUdFKVxuICAgICAgLmZpbGwoMClcbiAgICAgIC5tYXAoKF8sIGk6IG51bWJlcikgPT4gc3RhcnRWYWx1ZVllYXJWaWV3ICsgb2Zmc2V0ICsgaSk7XG4gICAgY29uc3Qgcm93czogbnVtYmVyID0gWUVBUlNfUEVSX1BBR0UgLyBZRUFSU19QRVJfUk9XO1xuICAgIGNvbnN0IHllYXJBcnJheTogbnVtYmVyW11bXSA9IFtdO1xuICAgIGZvciAobGV0IGk6IG51bWJlciA9IDA7IGkgPCByb3dzOyBpKyspIHtcbiAgICAgIHllYXJBcnJheS5wdXNoKGFsbFllYXJzLnNsaWNlKFlFQVJTX1BFUl9ST1cgKiBpLCBZRUFSU19QRVJfUk9XICogKGkgKyAxKSkpO1xuICAgIH1cbiAgICByZXR1cm4geWVhckFycmF5O1xuICB9XG5cbiAgLyoqXG4gICAqIENhbGN1bGF0ZXMgdGhlIGZpcnN0IHllYXIgdGhhdCB3aWxsIGJlIHNob3duIGluIHRoZSB5ZWFyIHNlbGVjdGlvbiBwYW5lbC5cbiAgICogSWYgYG1pbkRhdGVgIGFuZCBgbWF4RGF0ZWAgYXJlIGJvdGggbnVsbCwgdGhlIHN0YXJ0aW5nIHllYXIgaXMgY2FsY3VsYXRlZCBhc1xuICAgKiB0aGUgbXVsdGlwbGUgb2YgWUVBUlNfUEVSX1BBR0UgY2xvc2VzdCB0byBhbmQgbGVzcyB0aGFuIGFjdGl2ZURhdGUsXG4gICAqIGUuZy4sIHdpdGggYFlFQVJTX1BFUl9QQUdFYCA9IDI0IGFuZCBgYWN0aXZlRGF0ZWAgPSAyMDIwLCB0aGUgZnVuY3Rpb24gd2lsbCByZXR1cm4gMjAxNiAoMjQgKiA4MyksXG4gICAqIHdoaWxlIHdpdGggYGFjdGl2ZURhdGVgID0gMjAwMCwgdGhlIGZ1bmN0aW9uIHdpbGwgcmV0dXJuIDE5OTIgKDI0ICogODIpLlxuICAgKiBJZiBgbWluRGF0ZWAgaXMgbm90IG51bGwsIGl0IHJldHVybnMgdGhlIGNvcnJlc3BvbmRpbmcgeWVhcjsgaWYgYG1heERhdGVgIGlzIG5vdCBudWxsLFxuICAgKiBpdCByZXR1cm5zIHRoZSBjb3JyZXNwb25kaW5nIHllYXIgbWludXMgYFlFQVJTX1BFUl9QQUdFYCwgc28gdGhhdCB0aGUgYG1heERhdGVgIGlzIHRoZSBsYXN0IHJlbmRlcmVkIHllYXIuXG4gICAqIElmIGJvdGggYXJlIG5vdCBudWxsLCBgbWF4RGF0ZWAgaGFzIHByaW9yaXR5IG92ZXIgYG1pbkRhdGVgLlxuICAgKi9cbiAgcHJpdmF0ZSBfZ2V0U3RhcnRWYWx1ZVllYXJWaWV3KCk6IG51bWJlciB7XG4gICAgbGV0IHN0YXJ0aW5nWWVhciA9IDA7XG4gICAgaWYgKHRoaXMubWF4KSB7XG4gICAgICBzdGFydGluZ1llYXIgPSB0aGlzLl9kYXRlQWRhcHRlci5nZXRZZWFyKHRoaXMubWF4KSAtIFlFQVJTX1BFUl9QQUdFICsgMTtcbiAgICB9IGVsc2UgaWYgKHRoaXMubWluKSB7XG4gICAgICBzdGFydGluZ1llYXIgPSB0aGlzLl9kYXRlQWRhcHRlci5nZXRZZWFyKHRoaXMubWluKTtcbiAgICB9XG4gICAgY29uc3QgYWN0aXZlWWVhciA9IHRoaXMuX2RhdGVBZGFwdGVyLmdldFllYXIodGhpcy5fYWN0aXZlRGF0ZSk7XG4gICAgcmV0dXJuIChcbiAgICAgIGFjdGl2ZVllYXIgLVxuICAgICAgKCgoKGFjdGl2ZVllYXIgLSBzdGFydGluZ1llYXIpICUgWUVBUlNfUEVSX1BBR0UpICsgWUVBUlNfUEVSX1BBR0UpICUgWUVBUlNfUEVSX1BBR0UpXG4gICAgKTtcbiAgfVxuXG4gIC8qKiBDaGVja3MgaWYgZGF0ZSBpcyB3aXRoaW4gdGhlIG1pbi1tYXggcmFuZ2UuICovXG4gIHByaXZhdGUgX2lzRGF5SW5SYW5nZShkYXRlU3RyaW5nOiBzdHJpbmcpOiBib29sZWFuIHtcbiAgICBpZiAoIXRoaXMubWluICYmICF0aGlzLm1heCkge1xuICAgICAgcmV0dXJuIHRydWU7XG4gICAgfVxuICAgIGNvbnN0IGRhdGUgPSB0aGlzLl9kYXRlQWRhcHRlci5kZXNlcmlhbGl6ZShkYXRlU3RyaW5nKSE7XG4gICAgcmV0dXJuIHRoaXMuX2RhdGVBZGFwdGVyLnNhbWVEYXRlKGRhdGUsIHRoaXMuX2RhdGVBZGFwdGVyLmNsYW1wRGF0ZShkYXRlLCB0aGlzLm1pbiwgdGhpcy5tYXgpKTtcbiAgfVxuXG4gIC8qKiBFbWl0cyB0aGUgc2VsZWN0ZWQgZGF0ZSBhbmQgc2V0cyBpdCBpbnRlcm5hbGx5LiAqL1xuICBwcml2YXRlIF9zZWxlY3REYXRlKGRheTogVCk6IHZvaWQge1xuICAgIHRoaXMuX2Nob3Nlbk1vbnRoID0gdW5kZWZpbmVkO1xuICAgIHRoaXMuX3NldENob3NlblllYXIoKTtcbiAgICBpZiAodGhpcy5tdWx0aXBsZSkge1xuICAgICAgLy8gQ2hlY2sgaWYgX3NlbGVjdGVkIGhhcyBlbGVtZW50c1xuICAgICAgaWYgKHRoaXMuX3NlbGVjdGVkICYmICh0aGlzLl9zZWxlY3RlZCBhcyBUW10pLmxlbmd0aCA+IDApIHtcbiAgICAgICAgY29uc3QgaW5kZXhPZlNlbGVjdGVkRGF5OiBudW1iZXIgPSAodGhpcy5fc2VsZWN0ZWQgYXMgVFtdKS5maW5kSW5kZXgoXG4gICAgICAgICAgKHNlbCkgPT4gdGhpcy5fZGF0ZUFkYXB0ZXIuY29tcGFyZURhdGUoc2VsLCBkYXkpID09PSAwLFxuICAgICAgICApO1xuICAgICAgICAvLyBJZiB0aGUgc2VsZWN0ZWQgZGF0ZSBpcyBhbHJlYWR5IGluIHRoZSBfc2VsZWN0ZWQgYXJyYXksIHJlbW92ZSBpdCwgb3RoZXJ3aXNlIGFkZCBpdFxuICAgICAgICBpZiAoaW5kZXhPZlNlbGVjdGVkRGF5ICE9PSAtMSkge1xuICAgICAgICAgIHRoaXMuX3NlbGVjdGVkID0gKHRoaXMuX3NlbGVjdGVkIGFzIFRbXSkuZmlsdGVyKChfLCBpKSA9PiBpICE9PSBpbmRleE9mU2VsZWN0ZWREYXkpO1xuICAgICAgICB9IGVsc2Uge1xuICAgICAgICAgIHRoaXMuX3NlbGVjdGVkID0gWy4uLih0aGlzLl9zZWxlY3RlZCBhcyBUW10pLCBkYXldO1xuICAgICAgICB9XG4gICAgICB9IGVsc2Uge1xuICAgICAgICAvLyBJZiBfc2VsZWN0ZWQgaXMgZW1wdHksIHNldCBpdFxuICAgICAgICB0aGlzLl9zZWxlY3RlZCA9IFtkYXldO1xuICAgICAgfVxuICAgICAgdGhpcy5fZW1pdERhdGVTZWxlY3RlZEV2ZW50KHRoaXMuX3NlbGVjdGVkLm1hcCgoZSkgPT4gdGhpcy5fZGF0ZUFkYXB0ZXIuZGVzZXJpYWxpemUoZSkhKSk7XG4gICAgfSBlbHNlIHtcbiAgICAgIC8vIEluIHNpbmdsZSBzZWxlY3Rpb24sIGNoZWNrIGlmIHRoZSBkYXkgaXMgYWxyZWFkeSBzZWxlY3RlZFxuICAgICAgaWYgKCF0aGlzLl9zZWxlY3RlZCB8fCB0aGlzLl9kYXRlQWRhcHRlci5jb21wYXJlRGF0ZSh0aGlzLl9zZWxlY3RlZCBhcyBULCBkYXkpICE9PSAwKSB7XG4gICAgICAgIHRoaXMuX3NlbGVjdGVkID0gZGF5O1xuICAgICAgICB0aGlzLl9lbWl0RGF0ZVNlbGVjdGVkRXZlbnQodGhpcy5fZGF0ZUFkYXB0ZXIuZGVzZXJpYWxpemUoZGF5KSEpO1xuICAgICAgfVxuICAgIH1cbiAgfVxuXG4gIC8qKlxuICAgKiBIYW5kbGUgbXVsdGlwbGUgZGF0ZXMgc2VsZWN0aW9uIHZpYSB3ZWVrTnVtYmVyIC8gd2Vla0RheSBidXR0b25zOlxuICAgKiAtIGlmIENtZCBvciBDdHJsIGFyZSBwcmVzc2VkLCBhZGQgdGhlIG5ldyBkYXRlIHRvIHRoZSBjdXJyZW50IG9uZXM7XG4gICAqIC0gaWYgbm90LFxuICAgKiAgICAgLSBpZiB0aGUgbmV3IGRhdGVzIGFyZSB0aGUgc2FtZSBvZiB0aGUgY3VycmVudCBvbmVzLCBpdCBtZWFucyB0aGF0IHRoZSBzYW1lIGJ1dHRvbiBoYXMgYmVlbiBjbGlja2VkIHR3aWNlLCBzbyBkbyBub3RoaW5nO1xuICAgKiAgICAgLSBpZiBub3QsIHRoZSBzZWxlY3RlZCBkYXRlcyBhcmUgdGhlIG5ldyBvbmVzLlxuICAgKi9cbiAgcHJpdmF0ZSBfc2VsZWN0TXVsdGlwbGVEYXRlcyhkYXlzOiBEYXk8VD5bXSk6IHZvaWQge1xuICAgIC8vIEZpbHRlciBkaXNhYmxlZCBkYXlzIGJ5IG1hdGNoaW5nIHRoZSBwcm92aWRlZCBgZGF5c2AgcGFyYW1ldGVyIGFnYWluc3QgdGhlIGVuYWJsZWQgY2VsbHMuXG4gICAgLy8gU2luY2UgdGhlIGJ1dHRvbnMnIHZhbHVlIGlzIHNldCB0byB0aGUgRGF5J3MgaW50ZXJmYWNlIHZhbHVlIChJU08gc3RyaW5nKSwgdGhlcmUncyBubyBuZWVkIHRvIGRlc2VyaWFsaXplIGl0LlxuICAgIGNvbnN0IGVuYWJsZWREYXlzOiBzdHJpbmdbXSA9ICh0aGlzLl9jZWxscyBhcyBTYmJDYWxlbmRhckRheUVsZW1lbnRbXSlcbiAgICAgIC5maWx0ZXIoKGUpID0+ICFlLmRpc2FibGVkKVxuICAgICAgLm1hcCgoZSkgPT4gdGhpcy5fbWFwVmFsdWVUb0lTT0RhdGUoZS52YWx1ZSEpKTtcbiAgICBjb25zdCBkYXlzVG9BZGQ6IHN0cmluZ1tdID0gZGF5c1xuICAgICAgLm1hcCgoZTogRGF5PFQ+KSA9PiBlLnZhbHVlKVxuICAgICAgLmZpbHRlcigoaXNvRGF0ZTogc3RyaW5nKSA9PiBlbmFibGVkRGF5cy5pbmNsdWRlcyhpc29EYXRlKSk7XG4gICAgY29uc3QgZGF5c1RvQWRkU2V0ID0gbmV3IFNldChkYXlzVG9BZGQpO1xuICAgIGNvbnN0IHNlbGVjdGVkU2V0ID0gbmV3IFNldCgodGhpcy5fc2VsZWN0ZWQgYXMgVFtdKS5tYXAoKHMpID0+IHRoaXMuX2RhdGVBZGFwdGVyLnRvSXNvODYwMShzKSkpO1xuICAgIGNvbnN0IHNlbFN0cmluZ3MgPSB0aGlzLl91cGRhdGVTZWxlY3RlZFdpdGhNdWx0aXBsZURhdGVzKGRheXNUb0FkZCwgZGF5c1RvQWRkU2V0LCBzZWxlY3RlZFNldCk7XG4gICAgdGhpcy5fc2VsZWN0ZWQgPSBzZWxTdHJpbmdzLm1hcCgocykgPT4gdGhpcy5fZGF0ZUFkYXB0ZXIuZGVzZXJpYWxpemUocykhKTtcblxuICAgIHRoaXMuX2VtaXREYXRlU2VsZWN0ZWRFdmVudCh0aGlzLl9zZWxlY3RlZC5tYXAoKGUpID0+IHRoaXMuX2RhdGVBZGFwdGVyLmRlc2VyaWFsaXplKGUpISkpO1xuICB9XG5cbiAgLyoqXG4gICAqIEVtaXRzIHRoZSBkYXRlc2VsZWN0ZWQgZXZlbnQgZ2l2ZW4gdGhlIGRldGFpbCAoYXMgVCBvciBUW10gYmFzZWQgb24gdGhlIHZhbHVlIG9mIHRoZSBtdWx0aXBsZSBmbGFnKS5cbiAgICovXG4gIHByaXZhdGUgX2VtaXREYXRlU2VsZWN0ZWRFdmVudChkZXRhaWw6IFQgfCBUW10pOiB2b2lkIHtcbiAgICAvKiogQHR5cGUge0N1c3RvbUV2ZW50PFQgfCBUW10+fSBFdmVudCBlbWl0dGVkIG9uIGRhdGUgc2VsZWN0aW9uLiAqL1xuICAgIHRoaXMuZGlzcGF0Y2hFdmVudChcbiAgICAgIG5ldyBDdXN0b21FdmVudDxUIHwgVFtdPignZGF0ZXNlbGVjdGVkJywge1xuICAgICAgICBkZXRhaWwsXG4gICAgICAgIGNvbXBvc2VkOiB0cnVlLFxuICAgICAgICBidWJibGVzOiB0cnVlLFxuICAgICAgfSksXG4gICAgKTtcbiAgfVxuXG4gIHByaXZhdGUgX2VtaXRNb250aENoYW5nZSgpOiB2b2lkIHtcbiAgICAvLyBGSVhNRTogdGhlIG5hbWUgb2YgdGhpcyB2YXJpYWJsZSBhcHBlYXJzIGFzIGV2ZW50IG5hbWUgaW4gdGhlIHJlYWRtZVxuICAgIC8vICBkdWUgdG8gYSBidWcgaW4gdGhlIGN1c3RvbS1lbGVtZW50cy1tYW5pZmVzdCBsaWJyYXJ5LlxuICAgIC8vICBodHRwczovL2dpdGh1Yi5jb20vb3Blbi13Yy9jdXN0b20tZWxlbWVudHMtbWFuaWZlc3QvaXNzdWVzLzE0OVxuICAgIGNvbnN0IG1vbnRoY2hhbmdlID0gKHRoaXMud2lkZSA/IFsuLi50aGlzLl93ZWVrcywgLi4udGhpcy5fbmV4dE1vbnRoV2Vla3NdIDogdGhpcy5fd2Vla3MpXG4gICAgICAuZmxhdCgpXG4gICAgICAuc29ydCgoYSwgYikgPT4gYS52YWx1ZS5sb2NhbGVDb21wYXJlKGIudmFsdWUpKSBhcyBEYXlbXTtcbiAgICAvKipcbiAgICAgKiBAdHlwZSB7U2JiTW9udGhDaGFuZ2VFdmVudH1cbiAgICAgKiBFbWl0cyB3aGVuIHRoZSBtb250aCBjaGFuZ2VzLlxuICAgICAqIFRoZSBgcmFuZ2VgIHByb3BlcnR5IGNvbnRhaW5zIHRoZSBkYXlzIGFycmF5IG9mIHRoZSBjaG9zZW4gbW9udGguXG4gICAgICovXG4gICAgdGhpcy5kaXNwYXRjaEV2ZW50KG5ldyBTYmJNb250aENoYW5nZUV2ZW50KG1vbnRoY2hhbmdlKSk7XG4gIH1cblxuICAvKipcbiAgICogSW4gY2FzZSBvZiBtdWx0aXBsZSBzZWxlY3Rpb24sIG5ld2x5IGFkZGVkIGRheXMgbXVzdCBiZSBhZGRlZCB0byB0aGUgZXhpc3Rpbmcgb25lcywgd2l0aG91dCBkdXBsaWNhdGlvbi5cbiAgICogSWYgdGhlIGRheXMgdG8gYWRkIGFyZSBleGFjdGx5IHRoZSBzYW1lIGFzIHRoZSBzZWxlY3RlZCBvbmVzLCB0aGUgc2V0IG11c3QgYmUgZW1wdGllZC5cbiAgICovXG4gIHByaXZhdGUgX3VwZGF0ZVNlbGVjdGVkV2l0aE11bHRpcGxlRGF0ZXMoXG4gICAgZGF5c1RvQWRkOiBzdHJpbmdbXSxcbiAgICBkYXlzVG9BZGRTZXQ6IFNldDxzdHJpbmc+LFxuICAgIHNlbGVjdGVkU2V0OiBTZXQ8c3RyaW5nPixcbiAgKTogc3RyaW5nW10ge1xuICAgIGlmIChkYXlzVG9BZGQuZXZlcnkoKGRheTogc3RyaW5nKSA9PiBzZWxlY3RlZFNldC5oYXMoZGF5KSkpIHtcbiAgICAgIGRheXNUb0FkZFNldC5mb3JFYWNoKChkYXk6IHN0cmluZykgPT4gc2VsZWN0ZWRTZXQuZGVsZXRlKGRheSkpO1xuICAgIH0gZWxzZSB7XG4gICAgICBkYXlzVG9BZGRTZXQuZm9yRWFjaCgoZGF5OiBzdHJpbmcpID0+IHNlbGVjdGVkU2V0LmFkZChkYXkpKTtcbiAgICB9XG4gICAgcmV0dXJuIEFycmF5LmZyb20oc2VsZWN0ZWRTZXQpO1xuICB9XG5cbiAgcHJpdmF0ZSBfc2V0Q2hvc2VuWWVhcigpOiB2b2lkIHtcbiAgICBpZiAodGhpcy52aWV3ID09PSAnbW9udGgnKSB7XG4gICAgICBsZXQgc2VsZWN0ZWREYXRlOiBUIHwgdW5kZWZpbmVkO1xuICAgICAgaWYgKHRoaXMubXVsdGlwbGUpIHtcbiAgICAgICAgc2VsZWN0ZWREYXRlID0gKHRoaXMuc2VsZWN0ZWQgYXMgVFtdKS5hdCgtMSk7XG4gICAgICB9IGVsc2Uge1xuICAgICAgICBzZWxlY3RlZERhdGUgPSB0aGlzLnNlbGVjdGVkIGFzIFQ7XG4gICAgICB9XG4gICAgICB0aGlzLl9jaG9zZW5ZZWFyID0gdGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0WWVhcihzZWxlY3RlZERhdGUgPz8gdGhpcy5fZGF0ZUFkYXB0ZXIudG9kYXkoKSk7XG4gICAgfSBlbHNlIHtcbiAgICAgIHRoaXMuX2Nob3NlblllYXIgPSB1bmRlZmluZWQ7XG4gICAgfVxuICB9XG5cbiAgcHJpdmF0ZSBfYXNzaWduQWN0aXZlRGF0ZShkYXRlOiBUKTogdm9pZCB7XG4gICAgaWYgKHRoaXMubWluICYmIHRoaXMuX2RhdGVBZGFwdGVyLmNvbXBhcmVEYXRlKHRoaXMubWluLCBkYXRlKSA+IDApIHtcbiAgICAgIHRoaXMuX2FjdGl2ZURhdGUgPSB0aGlzLm1pbjtcbiAgICAgIHJldHVybjtcbiAgICB9XG4gICAgaWYgKHRoaXMubWF4ICYmIHRoaXMuX2RhdGVBZGFwdGVyLmNvbXBhcmVEYXRlKHRoaXMubWF4LCBkYXRlKSA8IDApIHtcbiAgICAgIHRoaXMuX2FjdGl2ZURhdGUgPSB0aGlzLm1heDtcbiAgICAgIHJldHVybjtcbiAgICB9XG4gICAgdGhpcy5fYWN0aXZlRGF0ZSA9IGRhdGU7XG4gIH1cblxuICAvKiogR29lcyB0byB0aGUgbW9udGggaWRlbnRpZmllZCBieSB0aGUgc2hpZnQuICovXG4gIHByaXZhdGUgX2dvVG9EaWZmZXJlbnRNb250aChtb250aHM6IG51bWJlcik6IHZvaWQge1xuICAgIHRoaXMuX2luaXQodGhpcy5fZGF0ZUFkYXB0ZXIuYWRkQ2FsZW5kYXJNb250aHModGhpcy5fYWN0aXZlRGF0ZSwgbW9udGhzKSk7XG4gICAgdGhpcy5fZW1pdE1vbnRoQ2hhbmdlKCk7XG4gIH1cblxuICBwcml2YXRlIF9nb1RvRGlmZmVyZW50WWVhcih5ZWFyczogbnVtYmVyKTogdm9pZCB7XG4gICAgdGhpcy5fY2hvc2VuWWVhciEgKz0geWVhcnM7XG4gICAgLy8gQ2FuJ3QgdXNlIGBfYXNzaWduQWN0aXZlRGF0ZSguLi4pYCBoZXJlLCBiZWNhdXNlIGl0IHdpbGwgc2V0IGl0IHRvIG1pbi9tYXggdmFsdWUgaWYgYXJndW1lbnQgaXMgb3V0IG9mIHJhbmdlXG4gICAgdGhpcy5fYWN0aXZlRGF0ZSA9IHRoaXMuX2RhdGVBZGFwdGVyLmNyZWF0ZURhdGUoXG4gICAgICB0aGlzLl9jaG9zZW5ZZWFyISxcbiAgICAgIHRoaXMuX2RhdGVBZGFwdGVyLmdldE1vbnRoKHRoaXMuX2FjdGl2ZURhdGUpLFxuICAgICAgdGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0RGF0ZSh0aGlzLl9hY3RpdmVEYXRlKSxcbiAgICApO1xuICAgIHRoaXMuX2luaXQoKTtcbiAgfVxuXG4gIHByaXZhdGUgX2dvVG9EaWZmZXJlbnRZZWFyUmFuZ2UoeWVhcnM6IG51bWJlcik6IHZvaWQge1xuICAgIHRoaXMuX2luaXQodGhpcy5fZGF0ZUFkYXB0ZXIuYWRkQ2FsZW5kYXJZZWFycyh0aGlzLl9hY3RpdmVEYXRlLCB5ZWFycykpO1xuICB9XG5cbiAgcHJpdmF0ZSBfcHJldkRpc2FibGVkKHByZXZEYXRlOiBUKTogYm9vbGVhbiB7XG4gICAgaWYgKCF0aGlzLm1pbikge1xuICAgICAgcmV0dXJuIGZhbHNlO1xuICAgIH1cbiAgICByZXR1cm4gdGhpcy5fZGF0ZUFkYXB0ZXIuY29tcGFyZURhdGUocHJldkRhdGUsIHRoaXMubWluKSA8IDA7XG4gIH1cblxuICBwcml2YXRlIF9uZXh0RGlzYWJsZWQobmV4dERhdGU6IFQpOiBib29sZWFuIHtcbiAgICBpZiAoIXRoaXMubWF4KSB7XG4gICAgICByZXR1cm4gZmFsc2U7XG4gICAgfVxuICAgIHJldHVybiB0aGlzLl9kYXRlQWRhcHRlci5jb21wYXJlRGF0ZShuZXh0RGF0ZSwgdGhpcy5tYXgpID4gMDtcbiAgfVxuXG4gIC8qKiBDaGVja3MgaWYgdGhlIFwicHJldmlvdXMgbW9udGhcIiBidXR0b24gc2hvdWxkIGJlIGRpc2FibGVkLiAqL1xuICBwcml2YXRlIF9wcmV2aW91c01vbnRoRGlzYWJsZWQoKTogYm9vbGVhbiB7XG4gICAgY29uc3QgcHJldk1vbnRoID0gdGhpcy5fZGF0ZUFkYXB0ZXIuYWRkQ2FsZW5kYXJEYXlzKFxuICAgICAgdGhpcy5fYWN0aXZlRGF0ZSxcbiAgICAgIHRoaXMuX2RhdGVBZGFwdGVyLmdldERhdGUodGhpcy5fYWN0aXZlRGF0ZSkgKiAtMSxcbiAgICApO1xuICAgIHJldHVybiB0aGlzLl9wcmV2RGlzYWJsZWQocHJldk1vbnRoKTtcbiAgfVxuXG4gIC8qKiBDaGVja3MgaWYgdGhlIFwibmV4dCBtb250aFwiIGJ1dHRvbiBzaG91bGQgYmUgZGlzYWJsZWQuICovXG4gIHByaXZhdGUgX25leHRNb250aERpc2FibGVkKCk6IGJvb2xlYW4ge1xuICAgIGxldCBuZXh0TW9udGggPSB0aGlzLl9kYXRlQWRhcHRlci5hZGRDYWxlbmRhck1vbnRocyh0aGlzLl9hY3RpdmVEYXRlLCB0aGlzLl93aWRlID8gMiA6IDEpO1xuICAgIG5leHRNb250aCA9IHRoaXMuX2RhdGVBZGFwdGVyLmNyZWF0ZURhdGUoXG4gICAgICB0aGlzLl9kYXRlQWRhcHRlci5nZXRZZWFyKG5leHRNb250aCksXG4gICAgICB0aGlzLl9kYXRlQWRhcHRlci5nZXRNb250aChuZXh0TW9udGgpLFxuICAgICAgMSxcbiAgICApO1xuICAgIHJldHVybiB0aGlzLl9uZXh0RGlzYWJsZWQobmV4dE1vbnRoKTtcbiAgfVxuXG4gIC8qKiBDaGVja3MgaWYgdGhlIFwicHJldmlvdXMgeWVhclwiIGJ1dHRvbiBzaG91bGQgYmUgZGlzYWJsZWQuICovXG4gIHByaXZhdGUgX3ByZXZpb3VzWWVhckRpc2FibGVkKCk6IGJvb2xlYW4ge1xuICAgIGNvbnN0IHByZXZZZWFyID0gdGhpcy5fZGF0ZUFkYXB0ZXIuY3JlYXRlRGF0ZShcbiAgICAgIHRoaXMuX2RhdGVBZGFwdGVyLmdldFllYXIodGhpcy5fYWN0aXZlRGF0ZSkgLSAxLFxuICAgICAgMTIsXG4gICAgICAzMSxcbiAgICApO1xuICAgIHJldHVybiB0aGlzLl9wcmV2RGlzYWJsZWQocHJldlllYXIpO1xuICB9XG5cbiAgLyoqIENoZWNrcyBpZiB0aGUgXCJuZXh0IHllYXJcIiBidXR0b24gc2hvdWxkIGJlIGRpc2FibGVkLiAqL1xuICBwcml2YXRlIF9uZXh0WWVhckRpc2FibGVkKCk6IGJvb2xlYW4ge1xuICAgIGNvbnN0IG5leHRZZWFyID0gdGhpcy5fZGF0ZUFkYXB0ZXIuY3JlYXRlRGF0ZShcbiAgICAgIHRoaXMuX2RhdGVBZGFwdGVyLmdldFllYXIodGhpcy5fYWN0aXZlRGF0ZSkgKyAodGhpcy5fd2lkZSA/IDIgOiAxKSxcbiAgICAgIDEsXG4gICAgICAxLFxuICAgICk7XG4gICAgcmV0dXJuIHRoaXMuX25leHREaXNhYmxlZChuZXh0WWVhcik7XG4gIH1cblxuICAvKiogQ2hlY2tzIGlmIHRoZSBcInByZXZpb3VzIHllYXJcIiBidXR0b24gc2hvdWxkIGJlIGRpc2FibGVkIGluIHllYXIgdmlldy4gKi9cbiAgcHJpdmF0ZSBfcHJldmlvdXNZZWFyUmFuZ2VEaXNhYmxlZCgpOiBib29sZWFuIHtcbiAgICBjb25zdCBwcmV2WWVhciA9IHRoaXMuX2RhdGVBZGFwdGVyLmNyZWF0ZURhdGUodGhpcy5feWVhcnNbMF1bMF0gLSAxLCAxMiwgMzEpO1xuICAgIHJldHVybiB0aGlzLl9wcmV2RGlzYWJsZWQocHJldlllYXIpO1xuICB9XG5cbiAgLyoqIENoZWNrcyBpZiB0aGUgXCJuZXh0IHllYXJcIiBidXR0b24gc2hvdWxkIGJlIGRpc2FibGVkIGluIHllYXIgdmlldy4gKi9cbiAgcHJpdmF0ZSBfbmV4dFllYXJSYW5nZURpc2FibGVkKCk6IGJvb2xlYW4ge1xuICAgIGNvbnN0IHllYXJzID0gdGhpcy5fd2lkZSA/IHRoaXMuX25leHRNb250aFllYXJzIDogdGhpcy5feWVhcnM7XG4gICAgY29uc3QgbGFzdFllYXJSYW5nZSA9IHllYXJzW3llYXJzLmxlbmd0aCAtIDFdO1xuICAgIGNvbnN0IGxhc3RZZWFyID0gbGFzdFllYXJSYW5nZVtsYXN0WWVhclJhbmdlLmxlbmd0aCAtIDFdO1xuICAgIGNvbnN0IG5leHRZZWFyID0gdGhpcy5fZGF0ZUFkYXB0ZXIuY3JlYXRlRGF0ZShsYXN0WWVhciArIDEsIDEsIDEpO1xuICAgIHJldHVybiB0aGlzLl9uZXh0RGlzYWJsZWQobmV4dFllYXIpO1xuICB9XG5cbiAgcHJpdmF0ZSBfaGFuZGxlVGFibGVCbHVyKGV2ZW50VGFyZ2V0OiBIVE1MRWxlbWVudCk6IHZvaWQge1xuICAgIGlmIChldmVudFRhcmdldD8ubG9jYWxOYW1lICE9PSAnc2JiLWNhbGVuZGFyLWRheScpIHtcbiAgICAgIHRoaXMuX3NldFRhYkluZGV4KCk7XG4gICAgfVxuICB9XG5cbiAgcHJpdmF0ZSBfc2V0VGFiSW5kZXgoKTogdm9pZCB7XG4gICAgQXJyYXkuZnJvbSh0aGlzLl9jZWxscy5maWx0ZXIoKGUpID0+IGUudGFiSW5kZXggPT09IDApID8/IFtdKS5mb3JFYWNoKFxuICAgICAgKGRheSkgPT4gKGRheS50YWJJbmRleCA9IC0xKSxcbiAgICApO1xuICAgIGNvbnN0IGZpcnN0Rm9jdXNhYmxlID0gdGhpcy5fZ2V0Rmlyc3RGb2N1c2FibGUoKTtcbiAgICBpZiAoZmlyc3RGb2N1c2FibGUpIHtcbiAgICAgIGZpcnN0Rm9jdXNhYmxlLnRhYkluZGV4ID0gMDtcbiAgICB9XG4gIH1cblxuICAvKiogR2V0IHRoZSBlbGVtZW50IGluIHRoZSBjYWxlbmRhciB0byBhc3NpZ24gZm9jdXMuICovXG4gIHByaXZhdGUgX2dldEZpcnN0Rm9jdXNhYmxlKCk6IFNiYkNhbGVuZGFyQ2VsbEJhc2VFbGVtZW50IHwgbnVsbCB7XG4gICAgaWYgKHRoaXMuX2NhbGVuZGFyVmlldyA9PT0gJ2RheScpIHtcbiAgICAgIGNvbnN0IHNlbGVjdGVkT3JDdXJyZW50ID1cbiAgICAgICAgdGhpcy5fY2VsbHMuZmluZCgoZSkgPT4gZS5tYXRjaGVzKCc6c3RhdGUoc2VsZWN0ZWQpJykpID8/XG4gICAgICAgIHRoaXMuX2NlbGxzLmZpbmQoKGUpID0+IGUubWF0Y2hlcygnOnN0YXRlKGN1cnJlbnQpJykpO1xuICAgICAgcmV0dXJuIHNlbGVjdGVkT3JDdXJyZW50ICYmICFzZWxlY3RlZE9yQ3VycmVudC5kaXNhYmxlZFxuICAgICAgICA/IHNlbGVjdGVkT3JDdXJyZW50XG4gICAgICAgIDogdGhpcy5fZ2V0Rmlyc3RGb2N1c2FibGVEYXkoKTtcbiAgICB9IGVsc2Uge1xuICAgICAgY29uc3Qgc2VsZWN0ZWRPckN1cnJlbnQgPVxuICAgICAgICB0aGlzLnNoYWRvd1Jvb3Q/LnF1ZXJ5U2VsZWN0b3I8U2JiQ2FsZW5kYXJDZWxsQmFzZUVsZW1lbnQ+KCc6c3RhdGUoc2VsZWN0ZWQpJykgPz9cbiAgICAgICAgdGhpcy5zaGFkb3dSb290Py5xdWVyeVNlbGVjdG9yPFNiYkNhbGVuZGFyQ2VsbEJhc2VFbGVtZW50PignOnN0YXRlKGN1cnJlbnQpJyk7XG4gICAgICByZXR1cm4gc2VsZWN0ZWRPckN1cnJlbnQgJiYgIXNlbGVjdGVkT3JDdXJyZW50LmRpc2FibGVkXG4gICAgICAgID8gc2VsZWN0ZWRPckN1cnJlbnRcbiAgICAgICAgOiB0aGlzLnNoYWRvd1Jvb3QhLnF1ZXJ5U2VsZWN0b3I8U2JiQ2FsZW5kYXJDZWxsQmFzZUVsZW1lbnQ+KFxuICAgICAgICAgICAgYHNiYi1jYWxlbmRhci0ke3RoaXMuX2NhbGVuZGFyVmlld306bm90KFtkaXNhYmxlZF0pYCxcbiAgICAgICAgICApO1xuICAgIH1cbiAgfVxuXG4gIC8qKlxuICAgKiBJbiBgZGF5YCB2aWV3IGluIGB2ZXJ0aWNhbGAgb3JpZW50YXRpb24sXG4gICAqIGlmIHRoZSBmaXJzdCBvZiB0aGUgbW9udGggaXMgbm90IGEgTW9uZGF5LCBpdCBpcyBub3QgdGhlIGZpcnN0IHJlbmRlcmVkIGVsZW1lbnQgaW4gdGhlIHRhYmxlLFxuICAgKiBzbyBgdGhpcy5zaGFkb3dSb290IS5xdWVyeVNlbGVjdG9yKCdzYmItY2FsZW5kYXItZGF5Om5vdChbZGlzYWJsZWRdKScpYCB3aWxsIHJldHVybiBhIHdyb25nIHZhbHVlLlxuICAgKlxuICAgKiBUbyBzb2x2ZSB0aGlzLCB0aGUgZWxlbWVudCB3aXRoIHRoZSBsb3dlc3QgYHZhbHVlYCBpcyB0YWtlbiAoSVNPIFN0cmluZyBhcmUgb3JkZXJlZCkuXG4gICAqL1xuICBwcml2YXRlIF9nZXRGaXJzdEZvY3VzYWJsZURheSgpOiBTYmJDYWxlbmRhckRheUVsZW1lbnQgfCBudWxsIHtcbiAgICBjb25zdCBjZWxscyA9IHRoaXMuX2NlbGxzIGFzIFNiYkNhbGVuZGFyRGF5RWxlbWVudFtdO1xuICAgIGNvbnN0IGRheXNJblZpZXcgPSBjZWxscy5maWx0ZXIoKGUpID0+ICFlLmRpc2FibGVkKTtcbiAgICBpZiAoIWRheXNJblZpZXcgfHwgZGF5c0luVmlldy5sZW5ndGggPT09IDApIHtcbiAgICAgIHJldHVybiBudWxsO1xuICAgIH0gZWxzZSB7XG4gICAgICBjb25zdCBmaXJzdEVsZW1lbnQgPSBkYXlzSW5WaWV3XG4gICAgICAgIC5tYXAoKGUpOiBzdHJpbmcgPT4gdGhpcy5fZGF0ZUFkYXB0ZXIudG9Jc284NjAxKGUudmFsdWUhIGFzIFQpKVxuICAgICAgICAuc29ydCgpWzBdO1xuICAgICAgcmV0dXJuIGNlbGxzLmZpbmQoKGUpID0+IGUubWF0Y2hlcyhgW3Nsb3Q9XCIke2ZpcnN0RWxlbWVudH1cIl1gKSkhID8/IG51bGw7XG4gICAgfVxuICB9XG5cbiAgcHJpdmF0ZSBfaGFuZGxlS2V5Ym9hcmRFdmVudChldmVudDogS2V5Ym9hcmRFdmVudCwgZGF5PzogRGF5PFQ+KTogdm9pZCB7XG4gICAgaWYgKGlzQXJyb3dLZXlPclBhZ2VLZXlzUHJlc3NlZChldmVudCkpIHtcbiAgICAgIGV2ZW50LnByZXZlbnREZWZhdWx0KCk7XG4gICAgfVxuICAgIC8vIEdldHMgdGhlIGN1cnJlbnRseSByZW5kZXJlZCB0YWJsZSdzIGNlbGw7XG4gICAgLy8gdGhleSBjb3VsZCBiZSBkYXlzLCBtb250aHMgb3IgeWVhcnMgYmFzZWQgb24gdGhlIGN1cnJlbnQgc2VsZWN0aW9uIHZpZXcuXG4gICAgLy8gSWYgYHdpZGVgIGlzIHRydWUsIHllYXJzIGFyZSBkb3VibGVkIGluIG51bWJlciBhbmQgZGF5cyBhcmUgKHJvdWdobHkpIGRvdWJsZWQgdG9vLCBhZmZlY3RpbmcgdGhlIGBpbmRleGAgY2FsY3VsYXRpb24uXG4gICAgY29uc3QgY2VsbHMgPSB0aGlzLl9jZWxscztcbiAgICBjb25zdCBpbmRleDogbnVtYmVyID0gY2VsbHMuZmluZEluZGV4KChlKSA9PiBlID09PSBldmVudC50YXJnZXQpO1xuICAgIGxldCBuZXh0RWw6IFNiYkNhbGVuZGFyQ2VsbEJhc2VFbGVtZW50O1xuICAgIGlmIChkYXkpIHtcbiAgICAgIG5leHRFbCA9IHRoaXMuX25hdmlnYXRlQnlLZXlib2FyZERheVZpZXcoZXZlbnQsIGluZGV4LCBjZWxscyBhcyBTYmJDYWxlbmRhckRheUVsZW1lbnRbXSwgZGF5KTtcbiAgICB9IGVsc2Uge1xuICAgICAgbmV4dEVsID0gdGhpcy5fbmF2aWdhdGVCeUtleWJvYXJkKGV2ZW50LCBpbmRleCwgY2VsbHMpO1xuICAgIH1cbiAgICBjb25zdCBhY3RpdmVFbCA9ICh0aGlzLl9lbmhhbmNlZFZhcmlhbnQgPyBkb2N1bWVudCA6IHRoaXMuc2hhZG93Um9vdCEpXG4gICAgICAuYWN0aXZlRWxlbWVudCBhcyBTYmJDYWxlbmRhckNlbGxCYXNlRWxlbWVudDtcbiAgICBpZiAobmV4dEVsICE9PSBhY3RpdmVFbCkge1xuICAgICAgbmV4dEVsLnRhYkluZGV4ID0gMDtcbiAgICAgIG5leHRFbD8uZm9jdXMoKTtcbiAgICAgIGFjdGl2ZUVsLnRhYkluZGV4ID0gLTE7XG4gICAgfVxuICB9XG5cbiAgcHJpdmF0ZSBfbmF2aWdhdGVCeUtleWJvYXJkRGF5VmlldyhcbiAgICBldnQ6IEtleWJvYXJkRXZlbnQsXG4gICAgaW5kZXg6IG51bWJlcixcbiAgICBjZWxsczogU2JiQ2FsZW5kYXJEYXlFbGVtZW50W10sXG4gICAgZGF5OiBEYXk8VD4sXG4gICk6IFNiYkNhbGVuZGFyRGF5RWxlbWVudCB7XG4gICAgY29uc3QgYXJyb3dzT2Zmc2V0ID1cbiAgICAgIHRoaXMub3JpZW50YXRpb24gPT09ICdob3Jpem9udGFsJ1xuICAgICAgICA/IHsgbGVmdFJpZ2h0OiAxLCB1cERvd246IERBWVNfUEVSX1JPVyB9XG4gICAgICAgIDogeyBsZWZ0UmlnaHQ6IERBWVNfUEVSX1JPVywgdXBEb3duOiAxIH07XG4gICAgY29uc3Qgb2Zmc2V0Rm9yVmVydGljYWw6IG51bWJlciA9XG4gICAgICBpbmRleCA8IHRoaXMuX2tleWJvYXJkTmF2aWdhdGlvbkRheVZpZXdQYXJhbWV0ZXJzLmZpcnN0TW9udGhMZW5ndGhcbiAgICAgICAgPyB0aGlzLl9rZXlib2FyZE5hdmlnYXRpb25EYXlWaWV3UGFyYW1ldGVycy5maXJzdE1vbnRoT2Zmc2V0XG4gICAgICAgIDogdGhpcy5fa2V5Ym9hcmROYXZpZ2F0aW9uRGF5Vmlld1BhcmFtZXRlcnMuc2Vjb25kTW9udGhPZmZzZXQ7XG5cbiAgICBzd2l0Y2ggKGV2dC5rZXkpIHtcbiAgICAgIGNhc2UgJ0Fycm93VXAnOlxuICAgICAgICByZXR1cm4gdGhpcy5fZmluZERheUFycm93cyhjZWxscywgaW5kZXgsIGRheS5kYXRlVmFsdWUsIC1hcnJvd3NPZmZzZXQudXBEb3duKTtcbiAgICAgIGNhc2UgJ0Fycm93RG93bic6XG4gICAgICAgIHJldHVybiB0aGlzLl9maW5kRGF5QXJyb3dzKGNlbGxzLCBpbmRleCwgZGF5LmRhdGVWYWx1ZSwgYXJyb3dzT2Zmc2V0LnVwRG93bik7XG4gICAgICBjYXNlICdBcnJvd0xlZnQnOlxuICAgICAgICByZXR1cm4gdGhpcy5fZmluZERheUFycm93cyhjZWxscywgaW5kZXgsIGRheS5kYXRlVmFsdWUsIC1hcnJvd3NPZmZzZXQubGVmdFJpZ2h0KTtcbiAgICAgIGNhc2UgJ0Fycm93UmlnaHQnOlxuICAgICAgICByZXR1cm4gdGhpcy5fZmluZERheUFycm93cyhjZWxscywgaW5kZXgsIGRheS5kYXRlVmFsdWUsIGFycm93c09mZnNldC5sZWZ0UmlnaHQpO1xuICAgICAgY2FzZSAnUGFnZVVwJzoge1xuICAgICAgICBpZiAodGhpcy5vcmllbnRhdGlvbiA9PT0gJ2hvcml6b250YWwnKSB7XG4gICAgICAgICAgY29uc3QgZmlyc3RPZldlZWs6IG51bWJlciA9ICtkYXkuZGF5VmFsdWUgJSBEQVlTX1BFUl9ST1cgfHwgREFZU19QRVJfUk9XO1xuICAgICAgICAgIGNvbnN0IGRlbHRhOiBudW1iZXIgPSBmaXJzdE9mV2VlayAtICtkYXkuZGF5VmFsdWU7XG4gICAgICAgICAgcmV0dXJuIHRoaXMuX2ZpbmREYXlQYWdlVXBEb3duKGNlbGxzLCBpbmRleCwgZGF5LCBkZWx0YSwgYXJyb3dzT2Zmc2V0LnVwRG93bik7XG4gICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgY29uc3Qgd2Vla051bWJlcjogbnVtYmVyID0gTWF0aC5jZWlsKCgrZGF5LmRheVZhbHVlICsgb2Zmc2V0Rm9yVmVydGljYWwpIC8gREFZU19QRVJfUk9XKTtcbiAgICAgICAgICBjb25zdCBmaXJzdE9mV2VlazogbnVtYmVyID0gKHdlZWtOdW1iZXIgLSAxKSAqIERBWVNfUEVSX1JPVyAtIG9mZnNldEZvclZlcnRpY2FsICsgMTtcbiAgICAgICAgICBjb25zdCBkZWx0YTogbnVtYmVyID0gZmlyc3RPZldlZWsgLSArZGF5LmRheVZhbHVlO1xuICAgICAgICAgIHJldHVybiB0aGlzLl9maW5kRGF5UGFnZVVwRG93bihjZWxscywgaW5kZXgsIGRheSwgZGVsdGEsIGFycm93c09mZnNldC51cERvd24pO1xuICAgICAgICB9XG4gICAgICB9XG4gICAgICBjYXNlICdQYWdlRG93bic6IHtcbiAgICAgICAgaWYgKHRoaXMub3JpZW50YXRpb24gPT09ICdob3Jpem9udGFsJykge1xuICAgICAgICAgIGNvbnN0IG1vbnRoSW5Cb3VuZHMgPSArZGF5Lm1vbnRoVmFsdWUgKyAxID4gMTIgPyAxIDogK2RheS5tb250aFZhbHVlICsgMTtcbiAgICAgICAgICBjb25zdCB5ZWFySW5Cb3VuZHMgPSArZGF5Lm1vbnRoVmFsdWUgKyAxID4gMTIgPyArZGF5LnllYXJWYWx1ZSArIDEgOiArZGF5LnllYXJWYWx1ZTtcbiAgICAgICAgICBjb25zdCBmaXJzdE5leHRNb250aDogVCA9IHRoaXMuX2RhdGVBZGFwdGVyLmNyZWF0ZURhdGUoeWVhckluQm91bmRzLCBtb250aEluQm91bmRzLCAxKTtcbiAgICAgICAgICBjb25zdCBsYXN0T2ZNb250aDogbnVtYmVyID0gdGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0RGF0ZShcbiAgICAgICAgICAgIHRoaXMuX2RhdGVBZGFwdGVyLmFkZENhbGVuZGFyRGF5cyhmaXJzdE5leHRNb250aCwgLTEpLFxuICAgICAgICAgICk7XG4gICAgICAgICAgY29uc3QgZGVsdGE6IG51bWJlciA9XG4gICAgICAgICAgICBNYXRoLnRydW5jKChsYXN0T2ZNb250aCAtICtkYXkuZGF5VmFsdWUhKSAvIERBWVNfUEVSX1JPVykgKiBEQVlTX1BFUl9ST1c7XG4gICAgICAgICAgcmV0dXJuIHRoaXMuX2ZpbmREYXlQYWdlVXBEb3duKGNlbGxzLCBpbmRleCwgZGF5LCBkZWx0YSwgLWFycm93c09mZnNldC51cERvd24pO1xuICAgICAgICB9IGVsc2Uge1xuICAgICAgICAgIGNvbnN0IHdlZWtOdW1iZXI6IG51bWJlciA9IE1hdGguY2VpbCgoK2RheS5kYXlWYWx1ZSArIG9mZnNldEZvclZlcnRpY2FsKSAvIERBWVNfUEVSX1JPVyk7XG4gICAgICAgICAgY29uc3QgbGFzdE9mV2VlazogbnVtYmVyID0gd2Vla051bWJlciAqIERBWVNfUEVSX1JPVyAtIG9mZnNldEZvclZlcnRpY2FsO1xuICAgICAgICAgIGNvbnN0IGRlbHRhOiBudW1iZXIgPSBsYXN0T2ZXZWVrIC0gK2RheS5kYXlWYWx1ZTtcbiAgICAgICAgICByZXR1cm4gdGhpcy5fZmluZERheVBhZ2VVcERvd24oY2VsbHMsIGluZGV4LCBkYXksIGRlbHRhLCAtYXJyb3dzT2Zmc2V0LnVwRG93bik7XG4gICAgICAgIH1cbiAgICAgIH1cbiAgICAgIGNhc2UgJ0hvbWUnOiB7XG4gICAgICAgIHJldHVybiB0aGlzLl9maW5kRGF5Rmlyc3QoY2VsbHMsIGluZGV4LCBkYXksIDEpO1xuICAgICAgfVxuICAgICAgY2FzZSAnRW5kJzoge1xuICAgICAgICBjb25zdCBtb250aEluQm91bmRzID0gK2RheS5tb250aFZhbHVlICsgMSA+IDEyID8gMSA6ICtkYXkubW9udGhWYWx1ZSArIDE7XG4gICAgICAgIGNvbnN0IHllYXJJbkJvdW5kcyA9ICtkYXkubW9udGhWYWx1ZSArIDEgPiAxMiA/ICtkYXkueWVhclZhbHVlICsgMSA6ICtkYXkueWVhclZhbHVlO1xuICAgICAgICBjb25zdCBmaXJzdE5leHRNb250aDogVCA9IHRoaXMuX2RhdGVBZGFwdGVyLmNyZWF0ZURhdGUoeWVhckluQm91bmRzLCBtb250aEluQm91bmRzLCAxKTtcbiAgICAgICAgcmV0dXJuIHRoaXMuX2ZpbmREYXlMYXN0KGNlbGxzLCBpbmRleCwgZmlyc3ROZXh0TW9udGgpO1xuICAgICAgfVxuICAgICAgZGVmYXVsdDpcbiAgICAgICAgcmV0dXJuIGNlbGxzW2luZGV4XTtcbiAgICB9XG4gIH1cblxuICBwcml2YXRlIF9pc0RheU91dE9mVmlldyhkYXRlOiBzdHJpbmcpOiBib29sZWFuIHtcbiAgICByZXR1cm4gKFxuICAgICAgZGF0ZSA8IHRoaXMuX2tleWJvYXJkTmF2aWdhdGlvbkRheVZpZXdQYXJhbWV0ZXJzLmZpcnN0RGF5SW5WaWV3ISB8fFxuICAgICAgZGF0ZSA+IHRoaXMuX2tleWJvYXJkTmF2aWdhdGlvbkRheVZpZXdQYXJhbWV0ZXJzLmxhc3REYXlJblZpZXchXG4gICAgKTtcbiAgfVxuXG4gIHByaXZhdGUgX2ZpbmREYXlBcnJvd3MoXG4gICAgY2VsbHM6IFNiYkNhbGVuZGFyRGF5RWxlbWVudFtdLFxuICAgIGluZGV4OiBudW1iZXIsXG4gICAgZGF0ZTogVCxcbiAgICBkZWx0YTogbnVtYmVyLFxuICApOiBTYmJDYWxlbmRhckRheUVsZW1lbnQge1xuICAgIGNvbnN0IG5ld0RhdGVWYWx1ZSA9IHRoaXMuX2RhdGVBZGFwdGVyLnRvSXNvODYwMShcbiAgICAgIHRoaXMuX2RhdGVBZGFwdGVyLmFkZENhbGVuZGFyRGF5cyhkYXRlLCBkZWx0YSksXG4gICAgKTtcbiAgICBpZiAodGhpcy5faXNEYXlPdXRPZlZpZXcobmV3RGF0ZVZhbHVlKSkge1xuICAgICAgcmV0dXJuIGNlbGxzW2luZGV4XTtcbiAgICB9XG4gICAgY29uc3QgbmV4dENlbGwgPSBjZWxscy5maW5kKChlKSA9PiB0aGlzLl9tYXBWYWx1ZVRvSVNPRGF0ZShlLnZhbHVlISkgPT09IG5ld0RhdGVWYWx1ZSk7XG4gICAgaWYgKCFuZXh0Q2VsbCB8fCBuZXh0Q2VsbC5kaXNhYmxlZCkge1xuICAgICAgcmV0dXJuIHRoaXMuX2ZpbmREYXlBcnJvd3MoY2VsbHMsIGluZGV4LCB0aGlzLl9kYXRlQWRhcHRlci5kZXNlcmlhbGl6ZShuZXdEYXRlVmFsdWUpISwgZGVsdGEpO1xuICAgIH1cbiAgICByZXR1cm4gbmV4dENlbGw7XG4gIH1cblxuICBwcml2YXRlIF9maW5kRGF5UGFnZVVwRG93bihcbiAgICBjZWxsczogU2JiQ2FsZW5kYXJEYXlFbGVtZW50W10sXG4gICAgaW5kZXg6IG51bWJlcixcbiAgICBkYXk6IERheTxUPixcbiAgICBkZWx0YTogbnVtYmVyLFxuICAgIGRlbHRhSWZEaXNhYmxlZDogbnVtYmVyLFxuICApOiBTYmJDYWxlbmRhckRheUVsZW1lbnQge1xuICAgIGNvbnN0IG5ld0RhdGVWYWx1ZSA9IHRoaXMuX2RhdGVBZGFwdGVyLnRvSXNvODYwMShcbiAgICAgIHRoaXMuX2RhdGVBZGFwdGVyLmFkZENhbGVuZGFyRGF5cyhkYXkuZGF0ZVZhbHVlLCBkZWx0YSksXG4gICAgKTtcbiAgICBpZiAodGhpcy5faXNEYXlPdXRPZlZpZXcobmV3RGF0ZVZhbHVlKSkge1xuICAgICAgcmV0dXJuIGNlbGxzW2luZGV4XTtcbiAgICB9XG4gICAgY29uc3QgbmV4dENlbGwgPSBjZWxscy5maW5kKChlKSA9PiB0aGlzLl9tYXBWYWx1ZVRvSVNPRGF0ZShlLnZhbHVlISkgPT09IG5ld0RhdGVWYWx1ZSk7XG4gICAgaWYgKCFuZXh0Q2VsbCB8fCBuZXh0Q2VsbC5kaXNhYmxlZCkge1xuICAgICAgcmV0dXJuIHRoaXMuX2ZpbmREYXlQYWdlVXBEb3duKGNlbGxzLCBpbmRleCwgZGF5LCBkZWx0YSArIGRlbHRhSWZEaXNhYmxlZCwgZGVsdGFJZkRpc2FibGVkKTtcbiAgICB9XG4gICAgcmV0dXJuIG5leHRDZWxsO1xuICB9XG5cbiAgcHJpdmF0ZSBfZmluZERheUZpcnN0KFxuICAgIGNlbGxzOiBTYmJDYWxlbmRhckRheUVsZW1lbnRbXSxcbiAgICBpbmRleDogbnVtYmVyLFxuICAgIGRheTogRGF5PFQ+LFxuICAgIGRhdGU6IG51bWJlcixcbiAgKTogU2JiQ2FsZW5kYXJEYXlFbGVtZW50IHtcbiAgICBjb25zdCBuZXdEYXRlVmFsdWUgPSB0aGlzLl9kYXRlQWRhcHRlci50b0lzbzg2MDEoXG4gICAgICB0aGlzLl9kYXRlQWRhcHRlci5jcmVhdGVEYXRlKCtkYXkueWVhclZhbHVlLCArZGF5Lm1vbnRoVmFsdWUsIGRhdGUpLFxuICAgICk7XG4gICAgaWYgKHRoaXMuX2lzRGF5T3V0T2ZWaWV3KG5ld0RhdGVWYWx1ZSkpIHtcbiAgICAgIHJldHVybiBjZWxsc1tpbmRleF07XG4gICAgfVxuICAgIGNvbnN0IG5leHRDZWxsID0gY2VsbHMuZmluZCgoZSkgPT4gdGhpcy5fbWFwVmFsdWVUb0lTT0RhdGUoZS52YWx1ZSEpID09PSBuZXdEYXRlVmFsdWUpO1xuICAgIGlmICghbmV4dENlbGwgfHwgbmV4dENlbGwuZGlzYWJsZWQpIHtcbiAgICAgIHJldHVybiB0aGlzLl9maW5kRGF5Rmlyc3QoY2VsbHMsIGluZGV4LCBkYXksIGRhdGUgKyAxKTtcbiAgICB9XG4gICAgcmV0dXJuIG5leHRDZWxsO1xuICB9XG5cbiAgcHJpdmF0ZSBfZmluZERheUxhc3QoXG4gICAgY2VsbHM6IFNiYkNhbGVuZGFyRGF5RWxlbWVudFtdLFxuICAgIGluZGV4OiBudW1iZXIsXG4gICAgZmlyc3ROZXh0TW9udGg6IFQsXG4gICk6IFNiYkNhbGVuZGFyRGF5RWxlbWVudCB7XG4gICAgY29uc3QgbmV3RGF0ZVZhbHVlID0gdGhpcy5fZGF0ZUFkYXB0ZXIudG9Jc284NjAxKFxuICAgICAgdGhpcy5fZGF0ZUFkYXB0ZXIuYWRkQ2FsZW5kYXJEYXlzKGZpcnN0TmV4dE1vbnRoLCAtMSksXG4gICAgKTtcbiAgICBpZiAodGhpcy5faXNEYXlPdXRPZlZpZXcobmV3RGF0ZVZhbHVlKSkge1xuICAgICAgcmV0dXJuIGNlbGxzW2luZGV4XTtcbiAgICB9XG4gICAgY29uc3QgbmV4dENlbGwgPSBjZWxscy5maW5kKChlKSA9PiB0aGlzLl9tYXBWYWx1ZVRvSVNPRGF0ZShlLnZhbHVlISkgPT09IG5ld0RhdGVWYWx1ZSk7XG4gICAgaWYgKCFuZXh0Q2VsbCB8fCBuZXh0Q2VsbC5kaXNhYmxlZCkge1xuICAgICAgcmV0dXJuIHRoaXMuX2ZpbmREYXlMYXN0KGNlbGxzLCBpbmRleCwgdGhpcy5fZGF0ZUFkYXB0ZXIuZGVzZXJpYWxpemUobmV3RGF0ZVZhbHVlKSEpO1xuICAgIH1cbiAgICByZXR1cm4gbmV4dENlbGw7XG4gIH1cblxuICAvKipcbiAgICogR2V0cyB0aGUgaW5kZXggb2YgdGhlIGVsZW1lbnQgdG8gbW92ZSB0bywgYmFzZWQgb24gYSBsaXN0IG9mIGVsZW1lbnRzICh3aGljaCBjYW4gYmUgcG90ZW50aWFsbHkgZGlzYWJsZWQpLFxuICAgKiB0aGUga2V5Ym9hcmQgaW5wdXQgYW5kIHRoZSBwb3NpdGlvbiBvZiB0aGUgY3VycmVudCBlbGVtZW50IGluIHRoZSBsaXN0LlxuICAgKiBJbiB0aGUgZGF5IHZpZXcsIHRoZSBgZGF5PzogRGF5YCBwYXJhbWV0ZXIgaXMgbWFuZGF0b3J5IGZvciBjYWxjdWxhdGlvbixcbiAgICogd2hpbGUgaW4gbW9udGggYW5kIHllYXIgdmlldyBpdCdzIG5vdCBkdWUgdG8gdGhlIGZpeGVkIGFtb3VudCBvZiByZW5kZXJlZCBjZWxscy5cbiAgICovXG4gIHByaXZhdGUgX25hdmlnYXRlQnlLZXlib2FyZChcbiAgICBldnQ6IEtleWJvYXJkRXZlbnQsXG4gICAgaW5kZXg6IG51bWJlcixcbiAgICBjZWxsczogU2JiQ2FsZW5kYXJDZWxsQmFzZUVsZW1lbnRbXSxcbiAgKTogU2JiQ2FsZW5kYXJDZWxsQmFzZUVsZW1lbnQge1xuICAgIGNvbnN0IHtcbiAgICAgIGVsZW1lbnRJbmRleEZvcldpZGVNb2RlLFxuICAgICAgb2Zmc2V0Rm9yV2lkZU1vZGUsXG4gICAgICBsYXN0RWxlbWVudEluZGV4Rm9yV2lkZU1vZGUsXG4gICAgICB2ZXJ0aWNhbE9mZnNldCxcbiAgICB9OiBDYWxlbmRhcktleWJvYXJkTmF2aWdhdGlvbk1vbnRoWWVhclZpZXdzUGFyYW1ldGVycyA9XG4gICAgICB0aGlzLl9jYWxjdWxhdGVQYXJhbWV0ZXJzRm9yS2V5Ym9hcmROYXZpZ2F0aW9uKGluZGV4LCB0aGlzLl9jYWxlbmRhclZpZXcgPT09ICd5ZWFyJyk7XG5cbiAgICBzd2l0Y2ggKGV2dC5rZXkpIHtcbiAgICAgIGNhc2UgJ0Fycm93VXAnOlxuICAgICAgICByZXR1cm4gdGhpcy5fZmluZE5leHQoY2VsbHMsIGluZGV4LCAtdmVydGljYWxPZmZzZXQpO1xuICAgICAgY2FzZSAnQXJyb3dEb3duJzpcbiAgICAgICAgcmV0dXJuIHRoaXMuX2ZpbmROZXh0KGNlbGxzLCBpbmRleCwgdmVydGljYWxPZmZzZXQpO1xuICAgICAgY2FzZSAnQXJyb3dMZWZ0JzpcbiAgICAgICAgcmV0dXJuIHRoaXMuX2ZpbmROZXh0KGNlbGxzLCBpbmRleCwgLTEpO1xuICAgICAgY2FzZSAnQXJyb3dSaWdodCc6XG4gICAgICAgIHJldHVybiB0aGlzLl9maW5kTmV4dChjZWxscywgaW5kZXgsIDEpO1xuICAgICAgY2FzZSAnSG9tZSc6XG4gICAgICAgIHJldHVybiB0aGlzLl9maW5kRmlyc3QoY2VsbHMsIG9mZnNldEZvcldpZGVNb2RlKTtcbiAgICAgIGNhc2UgJ1BhZ2VVcCc6XG4gICAgICAgIHJldHVybiB0aGlzLl9maW5kRmlyc3RPbkNvbHVtbihcbiAgICAgICAgICBjZWxscyxcbiAgICAgICAgICBlbGVtZW50SW5kZXhGb3JXaWRlTW9kZSxcbiAgICAgICAgICBvZmZzZXRGb3JXaWRlTW9kZSxcbiAgICAgICAgICB2ZXJ0aWNhbE9mZnNldCxcbiAgICAgICAgKTtcbiAgICAgIGNhc2UgJ1BhZ2VEb3duJzpcbiAgICAgICAgcmV0dXJuIHRoaXMuX2ZpbmRMYXN0T25Db2x1bW4oY2VsbHMsIGluZGV4LCBsYXN0RWxlbWVudEluZGV4Rm9yV2lkZU1vZGUsIHZlcnRpY2FsT2Zmc2V0KTtcbiAgICAgIGNhc2UgJ0VuZCc6XG4gICAgICAgIHJldHVybiB0aGlzLl9maW5kTGFzdChjZWxscywgbGFzdEVsZW1lbnRJbmRleEZvcldpZGVNb2RlIC0gMSk7XG4gICAgICBkZWZhdWx0OlxuICAgICAgICByZXR1cm4gY2VsbHNbaW5kZXhdO1xuICAgIH1cbiAgfVxuXG4gIC8qKlxuICAgKiBDYWxjdWxhdGVzIHRoZSBwYXJhbWV0ZXJzIG5lZWRlZCBpbiBrZXlib2FyZCBuYXZpZ2F0aW9uIGluIHllYXIgYW5kIG1vbnRoIHZpZXcuXG4gICAqIEBwYXJhbSBpbmRleCBUaGUgc3RhcnRpbmcgZWxlbWVudCdzIGluZGV4IGluIHRoZSBjZWxsIGFycmF5LlxuICAgKiBAcGFyYW0gaXNZZWFyVmlldyBXaGV0aGVyIHRoZSBkaXNwbGF5ZWQgYHZpZXdgIGlzIHRoZSB5ZWFyIG9uZS5cbiAgICovXG4gIHByaXZhdGUgX2NhbGN1bGF0ZVBhcmFtZXRlcnNGb3JLZXlib2FyZE5hdmlnYXRpb24oXG4gICAgaW5kZXg6IG51bWJlcixcbiAgICBpc1llYXJWaWV3OiBib29sZWFuLFxuICApOiBDYWxlbmRhcktleWJvYXJkTmF2aWdhdGlvbk1vbnRoWWVhclZpZXdzUGFyYW1ldGVycyB7XG4gICAgY29uc3QgZWxlbWVudHNQZXJQYWdlID0gaXNZZWFyVmlldyA/IFlFQVJTX1BFUl9QQUdFIDogTU9OVEhTX1BFUl9QQUdFO1xuICAgIGNvbnN0IG9mZnNldDogbnVtYmVyID0gTWF0aC50cnVuYyhpbmRleCAvIGVsZW1lbnRzUGVyUGFnZSkgKiBlbGVtZW50c1BlclBhZ2U7XG4gICAgY29uc3QgaW5kZXhJblZpZXc6IG51bWJlciA9IG9mZnNldCA9PT0gMCA/IGluZGV4IDogaW5kZXggLSBlbGVtZW50c1BlclBhZ2U7XG4gICAgcmV0dXJuIHtcbiAgICAgIHZlcnRpY2FsT2Zmc2V0OiBpc1llYXJWaWV3ID8gWUVBUlNfUEVSX1JPVyA6IE1PTlRIU19QRVJfUk9XLFxuICAgICAgZWxlbWVudEluZGV4Rm9yV2lkZU1vZGU6IGluZGV4SW5WaWV3LFxuICAgICAgb2Zmc2V0Rm9yV2lkZU1vZGU6IGluZGV4IC0gaW5kZXhJblZpZXcsXG4gICAgICBsYXN0RWxlbWVudEluZGV4Rm9yV2lkZU1vZGU6IG9mZnNldCA9PT0gMCA/IGVsZW1lbnRzUGVyUGFnZSA6IGVsZW1lbnRzUGVyUGFnZSAqIDIsXG4gICAgfTtcbiAgfVxuXG4gIC8qKlxuICAgKiBHZXRzIHRoZSBuZXh0IGVsZW1lbnQgb2YgdGhlIHByb3ZpZGVkIGFycmF5IHN0YXJ0aW5nIGZyb20gYGluZGV4YCBieSBhZGRpbmcgYGRlbHRhYC5cbiAgICogSWYgdGhlIGZvdW5kIGVsZW1lbnQgaXMgZGlzYWJsZWQsIGl0IGNvbnRpbnVlcyBhZGRpbmcgYGRlbHRhYCB1bnRpbCBpdCBmaW5kcyBhbiBlbmFibGVkIG9uZSBpbiB0aGUgYXJyYXkgYm91bmRzLlxuICAgKi9cbiAgcHJpdmF0ZSBfZmluZE5leHQoXG4gICAgZGF5czogU2JiQ2FsZW5kYXJDZWxsQmFzZUVsZW1lbnRbXSxcbiAgICBpbmRleDogbnVtYmVyLFxuICAgIGRlbHRhOiBudW1iZXIsXG4gICk6IFNiYkNhbGVuZGFyQ2VsbEJhc2VFbGVtZW50IHtcbiAgICBsZXQgbmV4dEluZGV4ID0gaW5kZXggKyBkZWx0YTtcbiAgICB3aGlsZSAobmV4dEluZGV4IDwgZGF5cy5sZW5ndGggJiYgZGF5c1tuZXh0SW5kZXhdPy5kaXNhYmxlZCkge1xuICAgICAgbmV4dEluZGV4ICs9IGRlbHRhO1xuICAgIH1cbiAgICByZXR1cm4gZGF5c1tuZXh0SW5kZXhdID8/IGRheXNbaW5kZXhdO1xuICB9XG5cbiAgLyoqIEZpbmQgdGhlIGZpcnN0IGVuYWJsZWQgZWxlbWVudCBpbiB0aGUgcHJvdmlkZWQgYXJyYXkuICovXG4gIHByaXZhdGUgX2ZpbmRGaXJzdChcbiAgICBkYXlzOiBTYmJDYWxlbmRhckNlbGxCYXNlRWxlbWVudFtdLFxuICAgIGZpcnN0T2ZDdXJyZW50TW9udGg6IG51bWJlcixcbiAgKTogU2JiQ2FsZW5kYXJDZWxsQmFzZUVsZW1lbnQge1xuICAgIHJldHVybiAhZGF5c1tmaXJzdE9mQ3VycmVudE1vbnRoXS5kaXNhYmxlZFxuICAgICAgPyBkYXlzW2ZpcnN0T2ZDdXJyZW50TW9udGhdXG4gICAgICA6IHRoaXMuX2ZpbmROZXh0KGRheXMsIGZpcnN0T2ZDdXJyZW50TW9udGgsIDEpO1xuICB9XG5cbiAgLyoqIEZpbmQgdGhlIGxhc3QgZW5hYmxlZCBlbGVtZW50IGluIHRoZSBwcm92aWRlZCBhcnJheS4gKi9cbiAgcHJpdmF0ZSBfZmluZExhc3QoXG4gICAgZGF5czogU2JiQ2FsZW5kYXJDZWxsQmFzZUVsZW1lbnRbXSxcbiAgICBsYXN0T2ZDdXJyZW50TW9udGg6IG51bWJlcixcbiAgKTogU2JiQ2FsZW5kYXJDZWxsQmFzZUVsZW1lbnQge1xuICAgIHJldHVybiAhZGF5c1tsYXN0T2ZDdXJyZW50TW9udGhdLmRpc2FibGVkXG4gICAgICA/IGRheXNbbGFzdE9mQ3VycmVudE1vbnRoXVxuICAgICAgOiB0aGlzLl9maW5kTmV4dChkYXlzLCBsYXN0T2ZDdXJyZW50TW9udGgsIC0xKTtcbiAgfVxuXG4gIC8qKiBGaW5kIHRoZSBmaXJzdCBlbmFibGVkIGVsZW1lbnQgaW4gdGhlIHNhbWUgY29sdW1uIG9mIHRoZSBwcm92aWRlZCBhcnJheS4gKi9cbiAgcHJpdmF0ZSBfZmluZEZpcnN0T25Db2x1bW4oXG4gICAgZGF5czogU2JiQ2FsZW5kYXJDZWxsQmFzZUVsZW1lbnRbXSxcbiAgICBpbmRleDogbnVtYmVyLFxuICAgIG9mZnNldDogbnVtYmVyLFxuICAgIHZlcnRpY2FsT2Zmc2V0OiBudW1iZXIsXG4gICk6IFNiYkNhbGVuZGFyQ2VsbEJhc2VFbGVtZW50IHtcbiAgICBjb25zdCBuZXh0SW5kZXggPSAoaW5kZXggJSB2ZXJ0aWNhbE9mZnNldCkgKyBvZmZzZXQ7XG4gICAgcmV0dXJuICFkYXlzW25leHRJbmRleF0uZGlzYWJsZWRcbiAgICAgID8gZGF5c1tuZXh0SW5kZXhdXG4gICAgICA6IHRoaXMuX2ZpbmROZXh0KGRheXMsIG5leHRJbmRleCwgdmVydGljYWxPZmZzZXQpO1xuICB9XG5cbiAgLyoqIEZpbmQgdGhlIGxhc3QgZW5hYmxlZCBlbGVtZW50IGluIHRoZSBzYW1lIGNvbHVtbiBvZiB0aGUgcHJvdmlkZWQgYXJyYXkuICovXG4gIHByaXZhdGUgX2ZpbmRMYXN0T25Db2x1bW4oXG4gICAgZGF5czogU2JiQ2FsZW5kYXJDZWxsQmFzZUVsZW1lbnRbXSxcbiAgICBpbmRleDogbnVtYmVyLFxuICAgIG9mZnNldDogbnVtYmVyLFxuICAgIHZlcnRpY2FsT2Zmc2V0OiBudW1iZXIsXG4gICk6IFNiYkNhbGVuZGFyQ2VsbEJhc2VFbGVtZW50IHtcbiAgICBjb25zdCBuZXh0SW5kZXggPSBpbmRleCArIE1hdGgudHJ1bmMoKG9mZnNldCAtIGluZGV4IC0gMSkgLyB2ZXJ0aWNhbE9mZnNldCkgKiB2ZXJ0aWNhbE9mZnNldDtcbiAgICByZXR1cm4gIWRheXNbbmV4dEluZGV4XS5kaXNhYmxlZFxuICAgICAgPyBkYXlzW25leHRJbmRleF1cbiAgICAgIDogdGhpcy5fZmluZE5leHQoZGF5cywgbmV4dEluZGV4LCAtdmVydGljYWxPZmZzZXQpO1xuICB9XG5cbiAgcHJpdmF0ZSBfcmVzZXRDYWxlbmRhclZpZXdBbmRFbWl0TW9udGhDaGFuZ2UoaW5pdFRyYW5zaXRpb24gPSBmYWxzZSk6IHZvaWQge1xuICAgIHRoaXMuX3Jlc2V0Q2FsZW5kYXJWaWV3KGluaXRUcmFuc2l0aW9uKTtcbiAgICB0aGlzLl9lbWl0TW9udGhDaGFuZ2UoKTtcbiAgfVxuXG4gIHByaXZhdGUgX3Jlc2V0Q2FsZW5kYXJWaWV3KGluaXRUcmFuc2l0aW9uID0gZmFsc2UpOiB2b2lkIHtcbiAgICBpZiAodGhpcy5fY29udGFpbmluZ0ZvY3VzKSB7XG4gICAgICB0aGlzLl9yZXNldEZvY3VzID0gdHJ1ZTtcbiAgICB9XG4gICAgdGhpcy5fYWN0aXZlRGF0ZSA9XG4gICAgICAodGhpcy5tdWx0aXBsZSA/ICh0aGlzLl9zZWxlY3RlZCBhcyBUW10pLmF0KC0xKSA6ICh0aGlzLl9zZWxlY3RlZCBhcyBUKSkgPz9cbiAgICAgIHRoaXMuX2RhdGVBZGFwdGVyLnRvZGF5KCk7XG4gICAgdGhpcy5fc2V0Q2hvc2VuWWVhcigpO1xuICAgIHRoaXMuX2Nob3Nlbk1vbnRoID0gdW5kZWZpbmVkO1xuICAgIHRoaXMuX2luaXQoKTtcbiAgICB0aGlzLl9uZXh0Q2FsZW5kYXJWaWV3ID0gdGhpcy5fY2FsZW5kYXJWaWV3ID0gdGhpcy52aWV3O1xuXG4gICAgaWYgKGluaXRUcmFuc2l0aW9uKSB7XG4gICAgICB0aGlzLl9zdGFydFRhYmxlVHJhbnNpdGlvbigpO1xuICAgIH1cbiAgfVxuXG4gIC8qKiBSZW5kZXIgdGhlIHZpZXcgZm9yIHRoZSBkYXkgc2VsZWN0aW9uLiAqL1xuICBwcml2YXRlIF9yZW5kZXJEYXlWaWV3KCk6IFRlbXBsYXRlUmVzdWx0IHtcbiAgICBjb25zdCBuZXh0TW9udGhBY3RpdmVEYXRlID0gdGhpcy5fd2lkZVxuICAgICAgPyB0aGlzLl9kYXRlQWRhcHRlci5hZGRDYWxlbmRhck1vbnRocyh0aGlzLl9hY3RpdmVEYXRlLCAxKVxuICAgICAgOiB1bmRlZmluZWQ7XG4gICAgcmV0dXJuIGh0bWxgXG4gICAgICA8ZGl2IGNsYXNzPVwic2JiLWNhbGVuZGFyX19jb250cm9sc1wiPlxuICAgICAgICAke3RoaXMuX2dldEFycm93KFxuICAgICAgICAgICdsZWZ0JyxcbiAgICAgICAgICAoKSA9PiB0aGlzLl9nb1RvRGlmZmVyZW50TW9udGgoLTEpLFxuICAgICAgICAgIGkxOG5QcmV2aW91c01vbnRoW3RoaXMuX2xhbmd1YWdlLmN1cnJlbnRdLFxuICAgICAgICAgIHRoaXMuX3ByZXZpb3VzTW9udGhEaXNhYmxlZCgpLFxuICAgICAgICApfVxuICAgICAgICA8ZGl2IGNsYXNzPVwic2JiLWNhbGVuZGFyX19jb250cm9scy1tb250aFwiPlxuICAgICAgICAgICR7dGhpcy5fY3JlYXRlTGFiZWxGb3JEYXlWaWV3KHRoaXMuX2FjdGl2ZURhdGUpfVxuICAgICAgICAgICR7dGhpcy5fd2lkZSA/IHRoaXMuX2NyZWF0ZUxhYmVsRm9yRGF5VmlldyhuZXh0TW9udGhBY3RpdmVEYXRlISkgOiBub3RoaW5nfVxuICAgICAgICAgIDxzYmItc2NyZWVuLXJlYWRlci1vbmx5IHJvbGU9XCJzdGF0dXNcIj5cbiAgICAgICAgICAgICR7dGhpcy5fY3JlYXRlQXJpYUxhYmVsRm9yRGF5Vmlldyh0aGlzLl9hY3RpdmVEYXRlLCBuZXh0TW9udGhBY3RpdmVEYXRlISl9XG4gICAgICAgICAgPC9zYmItc2NyZWVuLXJlYWRlci1vbmx5PlxuICAgICAgICA8L2Rpdj5cbiAgICAgICAgJHt0aGlzLl9nZXRBcnJvdyhcbiAgICAgICAgICAncmlnaHQnLFxuICAgICAgICAgICgpID0+IHRoaXMuX2dvVG9EaWZmZXJlbnRNb250aCgxKSxcbiAgICAgICAgICBpMThuTmV4dE1vbnRoW3RoaXMuX2xhbmd1YWdlLmN1cnJlbnRdLFxuICAgICAgICAgIHRoaXMuX25leHRNb250aERpc2FibGVkKCksXG4gICAgICAgICl9XG4gICAgICA8L2Rpdj5cbiAgICAgIDxkaXYgY2xhc3M9XCJzYmItY2FsZW5kYXJfX3RhYmxlLW92ZXJmbG93LWJyZWFrXCI+XG4gICAgICAgIDxkaXYgY2xhc3M9XCJzYmItY2FsZW5kYXJfX3RhYmxlLWNvbnRhaW5lciBzYmItY2FsZW5kYXJfX3RhYmxlLWRheS12aWV3XCI+XG4gICAgICAgICAgJHt0aGlzLm9yaWVudGF0aW9uID09PSAnaG9yaXpvbnRhbCdcbiAgICAgICAgICAgID8gaHRtbGBcbiAgICAgICAgICAgICAgICAke3RoaXMuX2NyZWF0ZURheVRhYmxlKHRoaXMuX3dlZWtzLCB0aGlzLl93ZWVrTnVtYmVycyl9XG4gICAgICAgICAgICAgICAgJHt0aGlzLl93aWRlXG4gICAgICAgICAgICAgICAgICA/IHRoaXMuX2NyZWF0ZURheVRhYmxlKHRoaXMuX25leHRNb250aFdlZWtzLCB0aGlzLl9uZXh0TW9udGhXZWVrTnVtYmVycywgdHJ1ZSlcbiAgICAgICAgICAgICAgICAgIDogbm90aGluZ31cbiAgICAgICAgICAgICAgYFxuICAgICAgICAgICAgOiBodG1sYFxuICAgICAgICAgICAgICAgICR7dGhpcy5fY3JlYXRlRGF5VGFibGVWZXJ0aWNhbCh0aGlzLl93ZWVrcywgdGhpcy5fd2Vla051bWJlcnMpfVxuICAgICAgICAgICAgICAgICR7dGhpcy5fd2lkZVxuICAgICAgICAgICAgICAgICAgPyB0aGlzLl9jcmVhdGVEYXlUYWJsZVZlcnRpY2FsKFxuICAgICAgICAgICAgICAgICAgICAgIHRoaXMuX25leHRNb250aFdlZWtzLFxuICAgICAgICAgICAgICAgICAgICAgIHRoaXMuX25leHRNb250aFdlZWtOdW1iZXJzLFxuICAgICAgICAgICAgICAgICAgICAgIG5leHRNb250aEFjdGl2ZURhdGUsXG4gICAgICAgICAgICAgICAgICAgIClcbiAgICAgICAgICAgICAgICAgIDogbm90aGluZ31cbiAgICAgICAgICAgICAgYH1cbiAgICAgICAgPC9kaXY+XG4gICAgICA8L2Rpdj5cbiAgICBgO1xuICB9XG5cbiAgLyoqIENyZWF0ZXMgdGhlIGxhYmVsIHdpdGggdGhlIG1vbnRoIGZvciB0aGUgZGFpbHkgdmlldy4gKi9cbiAgcHJpdmF0ZSBfY3JlYXRlTGFiZWxGb3JEYXlWaWV3KGQ6IFQpOiBUZW1wbGF0ZVJlc3VsdCB7XG4gICAgY29uc3QgbW9udGhMYWJlbCA9IGAke1xuICAgICAgdGhpcy5fbW9udGhOYW1lc1t0aGlzLl9kYXRlQWRhcHRlci5nZXRNb250aChkKSAtIDFdXG4gICAgfSAke3RoaXMuX2RhdGVBZGFwdGVyLmdldFllYXIoZCl9YDtcbiAgICByZXR1cm4gaHRtbGBcbiAgICAgIDxidXR0b25cbiAgICAgICAgdHlwZT1cImJ1dHRvblwiXG4gICAgICAgIGNsYXNzPVwic2JiLWNhbGVuZGFyX19kYXRlLXNlbGVjdGlvbiBzYmItY2FsZW5kYXJfX2NvbnRyb2xzLWNoYW5nZS1kYXRlXCJcbiAgICAgICAgYXJpYS1sYWJlbD1cIiR7aTE4blllYXJNb250aFNlbGVjdGlvblt0aGlzLl9sYW5ndWFnZS5jdXJyZW50XX0gJHttb250aExhYmVsfVwiXG4gICAgICAgIEBjbGljaz0keygpID0+IHtcbiAgICAgICAgICB0aGlzLl9yZXNldEZvY3VzID0gdHJ1ZTtcbiAgICAgICAgICB0aGlzLl9uZXh0Q2FsZW5kYXJWaWV3ID0gJ3llYXInO1xuICAgICAgICAgIHRoaXMuX3N0YXJ0VGFibGVUcmFuc2l0aW9uKCk7XG4gICAgICAgIH19XG4gICAgICA+XG4gICAgICAgICR7bW9udGhMYWJlbH1cbiAgICAgICAgPHNiYi1pY29uIG5hbWU9XCJjaGV2cm9uLXNtYWxsLWRvd24tc21hbGxcIj48L3NiYi1pY29uPlxuICAgICAgPC9idXR0b24+XG4gICAgYDtcbiAgfVxuXG4gIC8qKiBDcmVhdGVzIHRoZSBhcmlhLWxhYmVsIGZvciB0aGUgZGFpbHkgdmlldy4gKi9cbiAgcHJpdmF0ZSBfY3JlYXRlQXJpYUxhYmVsRm9yRGF5VmlldyguLi5kYXRlczogVFtdKTogc3RyaW5nIHtcbiAgICBsZXQgbW9udGhMYWJlbCA9ICcnO1xuICAgIGZvciAoY29uc3QgZCBvZiBkYXRlcykge1xuICAgICAgaWYgKGQpIHtcbiAgICAgICAgbW9udGhMYWJlbCArPSBgJHtcbiAgICAgICAgICB0aGlzLl9tb250aE5hbWVzW3RoaXMuX2RhdGVBZGFwdGVyLmdldE1vbnRoKGQpIC0gMV1cbiAgICAgICAgfSAke3RoaXMuX2RhdGVBZGFwdGVyLmdldFllYXIoZCl9IGA7XG4gICAgICB9XG4gICAgfVxuICAgIHJldHVybiBtb250aExhYmVsO1xuICB9XG5cbiAgLyoqIENyZWF0ZXMgdGhlIGNhbGVuZGFyIHRhYmxlIGZvciB0aGUgZGFpbHkgdmlldy4gKi9cbiAgcHJpdmF0ZSBfY3JlYXRlRGF5VGFibGUoXG4gICAgd2Vla3M6IERheTxUPltdW10sXG4gICAgd2Vla051bWJlcnM6IG51bWJlcltdLFxuICAgIGlzV2lkZU5leHRNb250aDogYm9vbGVhbiA9IGZhbHNlLFxuICApOiBUZW1wbGF0ZVJlc3VsdCB7XG4gICAgY29uc3Qgd2Vla3NGb3JTZWxlY3RNdWx0aXBsZVdlZWtOdW1iZXJzOiBEYXk8VD5bXSA9IChcbiAgICAgIHRoaXMuX3dpZGVcbiAgICAgICAgPyBbLi4udGhpcy5fd2Vla3MsIC4uLnRoaXMuX25leHRNb250aFdlZWtzXVxuICAgICAgICA6IGlzV2lkZU5leHRNb250aFxuICAgICAgICAgID8gdGhpcy5fbmV4dE1vbnRoV2Vla3NcbiAgICAgICAgICA6IHRoaXMuX3dlZWtzXG4gICAgKS5mbGF0KCk7XG4gICAgY29uc3Qgd2Vla3NGb3JTZWxlY3RNdWx0aXBsZVdlZWtEYXlzOiBEYXk8VD5bXSA9IChcbiAgICAgIGlzV2lkZU5leHRNb250aCA/IHRoaXMuX25leHRNb250aFdlZWtzIDogdGhpcy5fd2Vla3NcbiAgICApLmZsYXQoKTtcbiAgICByZXR1cm4gaHRtbGBcbiAgICAgIDx0YWJsZVxuICAgICAgICBjbGFzcz1cInNiYi1jYWxlbmRhcl9fdGFibGVcIlxuICAgICAgICBAZm9jdXNvdXQ9JHsoZXZlbnQ6IEZvY3VzRXZlbnQpID0+XG4gICAgICAgICAgdGhpcy5faGFuZGxlVGFibGVCbHVyKGV2ZW50LnJlbGF0ZWRUYXJnZXQgYXMgSFRNTEVsZW1lbnQpfVxuICAgICAgICBAYW5pbWF0aW9uZW5kPSR7KGU6IEFuaW1hdGlvbkV2ZW50KSA9PiB0aGlzLl90YWJsZUFuaW1hdGlvbkVuZChlKX1cbiAgICAgID5cbiAgICAgICAgPHRoZWFkIGNsYXNzPVwic2JiLWNhbGVuZGFyX190YWJsZS1oZWFkZXJcIj5cbiAgICAgICAgICA8dHI+XG4gICAgICAgICAgICAke3RoaXMud2Vla051bWJlcnMgPyBodG1sYDx0aCBjbGFzcz1cInNiYi1jYWxlbmRhcl9fdGFibGUtaGVhZGVyLWNlbGxcIj48L3RoPmAgOiBub3RoaW5nfVxuICAgICAgICAgICAgJHt0aGlzLl93ZWVrZGF5cy5tYXAoXG4gICAgICAgICAgICAgICh3ZWVrRGF5OiBXZWVrZGF5LCBpbmRleDogbnVtYmVyKSA9PiBodG1sYFxuICAgICAgICAgICAgICAgIDx0aCBjbGFzcz1cInNiYi1jYWxlbmRhcl9fdGFibGUtaGVhZGVyLWNlbGxcIj5cbiAgICAgICAgICAgICAgICAgICR7dGhpcy5tdWx0aXBsZVxuICAgICAgICAgICAgICAgICAgICA/IGh0bWxgXG4gICAgICAgICAgICAgICAgICAgICAgICA8c2JiLWNhbGVuZGFyLXdlZWtkYXlcbiAgICAgICAgICAgICAgICAgICAgICAgICAgLnZhbHVlPSR7d2Vla0RheX1cbiAgICAgICAgICAgICAgICAgICAgICAgICAgQGNsaWNrPSR7KCkgPT4ge1xuICAgICAgICAgICAgICAgICAgICAgICAgICAgIC8vIE5PVEU6IFN1bmRheXMgaGF2ZSBpbmRleCA3LCB3aGlsZSB0aGVpciB3ZWVrRGF5VmFsdWUgaXMgMFxuICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNvbnN0IGRheXM6IERheTxUPltdID0gd2Vla3NGb3JTZWxlY3RNdWx0aXBsZVdlZWtEYXlzLmZpbHRlcihcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIChkYXk6IERheTxUPikgPT4gZGF5LndlZWtEYXlWYWx1ZSA9PT0gKGluZGV4ICsgMSkgJSA3LFxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICkhO1xuICAgICAgICAgICAgICAgICAgICAgICAgICAgIHRoaXMuX3NlbGVjdE11bHRpcGxlRGF0ZXMoZGF5cyk7XG4gICAgICAgICAgICAgICAgICAgICAgICAgIH19XG4gICAgICAgICAgICAgICAgICAgICAgICA+PC9zYmItY2FsZW5kYXItd2Vla2RheT5cbiAgICAgICAgICAgICAgICAgICAgICBgXG4gICAgICAgICAgICAgICAgICAgIDogaHRtbGBcbiAgICAgICAgICAgICAgICAgICAgICAgIDxzYmItc2NyZWVuLXJlYWRlci1vbmx5PiR7d2Vla0RheS5sb25nfTwvc2JiLXNjcmVlbi1yZWFkZXItb25seT5cbiAgICAgICAgICAgICAgICAgICAgICAgIDxzcGFuIGFyaWEtaGlkZGVuPVwidHJ1ZVwiPiR7d2Vla0RheS5uYXJyb3d9PC9zcGFuPlxuICAgICAgICAgICAgICAgICAgICAgIGB9XG4gICAgICAgICAgICAgICAgPC90aD5cbiAgICAgICAgICAgICAgYCxcbiAgICAgICAgICAgICl9XG4gICAgICAgICAgPC90cj5cbiAgICAgICAgPC90aGVhZD5cbiAgICAgICAgPHRib2R5IGNsYXNzPVwic2JiLWNhbGVuZGFyX190YWJsZS1ib2R5XCI+XG4gICAgICAgICAgJHt3ZWVrcy5tYXAoKHdlZWs6IERheTxUPltdLCByb3dJbmRleDogbnVtYmVyKSA9PiB7XG4gICAgICAgICAgICBjb25zdCBmaXJzdFJvd09mZnNldDogbnVtYmVyID0gREFZU19QRVJfUk9XIC0gd2Vlay5sZW5ndGg7XG4gICAgICAgICAgICBpZiAocm93SW5kZXggPT09IDAgJiYgZmlyc3RSb3dPZmZzZXQpIHtcbiAgICAgICAgICAgICAgcmV0dXJuIGh0bWxgXG4gICAgICAgICAgICAgICAgPHRyPlxuICAgICAgICAgICAgICAgICAgJHt0aGlzLndlZWtOdW1iZXJzXG4gICAgICAgICAgICAgICAgICAgID8gaHRtbGBcbiAgICAgICAgICAgICAgICAgICAgICAgIDx0ZCBjbGFzcz1cInNiYi1jYWxlbmRhcl9fdGFibGUtaGVhZGVyLWNlbGwtdmVydGljYWxcIj5cbiAgICAgICAgICAgICAgICAgICAgICAgICAgJHt0aGlzLm11bHRpcGxlXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgPyBodG1sYFxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICA8c2JiLWNhbGVuZGFyLXdlZWtudW1iZXJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAudmFsdWU9JHt3ZWVrTnVtYmVyc1swXX1cbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBAY2xpY2s9JHsoKSA9PiB7XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBjb25zdCBkYXlzOiBEYXk8VD5bXSA9IHdlZWtzRm9yU2VsZWN0TXVsdGlwbGVXZWVrTnVtYmVycy5maWx0ZXIoXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIChkYXk6IERheTxUPikgPT4gZGF5LndlZWtWYWx1ZSA9PT0gd2Vla051bWJlcnNbMF0sXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICApITtcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHRoaXMuX3NlbGVjdE11bHRpcGxlRGF0ZXMoZGF5cyk7XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgfX1cbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPjwvc2JiLWNhbGVuZGFyLXdlZWtudW1iZXI+XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICBgXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgOiBodG1sYFxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICA8c2JiLXNjcmVlbi1yZWFkZXItb25seVxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID4ke2Ake2kxOG5DYWxlbmRhcldlZWtOdW1iZXJbdGhpcy5fbGFuZ3VhZ2UuY3VycmVudF19ICR7d2Vla051bWJlcnNbMF19YH08L3NiYi1zY3JlZW4tcmVhZGVyLW9ubHlcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPlxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICA8c3BhbiBhcmlhLWhpZGRlbj1cInRydWVcIj4ke3dlZWtOdW1iZXJzWzBdfTwvc3Bhbj5cbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGB9XG4gICAgICAgICAgICAgICAgICAgICAgICA8L3RkPlxuICAgICAgICAgICAgICAgICAgICAgIGBcbiAgICAgICAgICAgICAgICAgICAgOiBub3RoaW5nfVxuICAgICAgICAgICAgICAgICAgJHtbLi4uQXJyYXkoZmlyc3RSb3dPZmZzZXQpLmtleXMoKV0ubWFwKFxuICAgICAgICAgICAgICAgICAgICAoKSA9PiBodG1sYDx0ZCBjbGFzcz1cInNiYi1jYWxlbmRhcl9fdGFibGUtZGF0YVwiPjwvdGQ+YCxcbiAgICAgICAgICAgICAgICAgICl9XG4gICAgICAgICAgICAgICAgICAke3RoaXMuX2NyZWF0ZURheUNlbGxzKHdlZWspfVxuICAgICAgICAgICAgICAgIDwvdHI+XG4gICAgICAgICAgICAgIGA7XG4gICAgICAgICAgICB9XG4gICAgICAgICAgICByZXR1cm4gaHRtbGBcbiAgICAgICAgICAgICAgPHRyPlxuICAgICAgICAgICAgICAgICR7dGhpcy53ZWVrTnVtYmVyc1xuICAgICAgICAgICAgICAgICAgPyBodG1sYFxuICAgICAgICAgICAgICAgICAgICAgIDx0ZCBjbGFzcz1cInNiYi1jYWxlbmRhcl9fdGFibGUtaGVhZGVyLWNlbGwtdmVydGljYWxcIj5cbiAgICAgICAgICAgICAgICAgICAgICAgICR7dGhpcy5tdWx0aXBsZVxuICAgICAgICAgICAgICAgICAgICAgICAgICA/IGh0bWxgXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICA8c2JiLWNhbGVuZGFyLXdlZWtudW1iZXJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgLnZhbHVlPSR7d2Vla051bWJlcnNbcm93SW5kZXhdfVxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBAY2xpY2s9JHsoKSA9PiB7XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgY29uc3QgZGF5czogRGF5PFQ+W10gPSB3ZWVrc0ZvclNlbGVjdE11bHRpcGxlV2Vla051bWJlcnMuZmlsdGVyKFxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgKGRheTogRGF5PFQ+KSA9PiBkYXkud2Vla1ZhbHVlID09PSB3ZWVrTnVtYmVyc1tyb3dJbmRleF0sXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgKSE7XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgdGhpcy5fc2VsZWN0TXVsdGlwbGVEYXRlcyhkYXlzKTtcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgfX1cbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID48L3NiYi1jYWxlbmRhci13ZWVrbnVtYmVyPlxuICAgICAgICAgICAgICAgICAgICAgICAgICAgIGBcbiAgICAgICAgICAgICAgICAgICAgICAgICAgOiBodG1sYFxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPHNiYi1zY3JlZW4tcmVhZGVyLW9ubHlcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPiR7YCR7aTE4bkNhbGVuZGFyV2Vla051bWJlclt0aGlzLl9sYW5ndWFnZS5jdXJyZW50XX0gJHt3ZWVrTnVtYmVyc1tyb3dJbmRleF19YH08L3NiYi1zY3JlZW4tcmVhZGVyLW9ubHlcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID5cbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIDxzcGFuIGFyaWEtaGlkZGVuPVwidHJ1ZVwiPiR7d2Vla051bWJlcnNbcm93SW5kZXhdfTwvc3Bhbj5cbiAgICAgICAgICAgICAgICAgICAgICAgICAgICBgfVxuICAgICAgICAgICAgICAgICAgICAgIDwvdGQ+XG4gICAgICAgICAgICAgICAgICAgIGBcbiAgICAgICAgICAgICAgICAgIDogbm90aGluZ31cbiAgICAgICAgICAgICAgICAke3RoaXMuX2NyZWF0ZURheUNlbGxzKHdlZWspfVxuICAgICAgICAgICAgICA8L3RyPlxuICAgICAgICAgICAgYDtcbiAgICAgICAgICB9KX1cbiAgICAgICAgPC90Ym9keT5cbiAgICAgIDwvdGFibGU+XG4gICAgYDtcbiAgfVxuXG4gIC8qKiBDcmVhdGVzIHRoZSB0YWJsZSBpbiBvcmllbnRhdGlvbj0ndmVydGljYWwnLiAqL1xuICBwcml2YXRlIF9jcmVhdGVEYXlUYWJsZVZlcnRpY2FsKFxuICAgIHdlZWtzOiBEYXk8VD5bXVtdLFxuICAgIHdlZWtOdW1iZXJzOiBudW1iZXJbXSxcbiAgICBuZXh0TW9udGhBY3RpdmVEYXRlPzogVCxcbiAgKTogVGVtcGxhdGVSZXN1bHQge1xuICAgIGNvbnN0IHdlZWtPZmZzZXQgPSB0aGlzLl9kYXRlQWRhcHRlci5nZXRGaXJzdFdlZWtPZmZzZXQoXG4gICAgICBuZXh0TW9udGhBY3RpdmVEYXRlID8/IHRoaXMuX2FjdGl2ZURhdGUsXG4gICAgKTtcbiAgICBjb25zdCB3ZWVrc0ZvclNlbGVjdE11bHRpcGxlV2Vla051bWJlcnM6IERheTxUPltdID0gKFxuICAgICAgdGhpcy5fd2lkZVxuICAgICAgICA/IFsuLi50aGlzLl93ZWVrcywgLi4udGhpcy5fbmV4dE1vbnRoV2Vla3NdXG4gICAgICAgIDogbmV4dE1vbnRoQWN0aXZlRGF0ZVxuICAgICAgICAgID8gdGhpcy5fbmV4dE1vbnRoV2Vla3NcbiAgICAgICAgICA6IHRoaXMuX3dlZWtzXG4gICAgKS5mbGF0KCk7XG4gICAgcmV0dXJuIGh0bWxgXG4gICAgICA8dGFibGVcbiAgICAgICAgY2xhc3M9XCJzYmItY2FsZW5kYXJfX3RhYmxlXCJcbiAgICAgICAgQGZvY3Vzb3V0PSR7KGV2ZW50OiBGb2N1c0V2ZW50KSA9PlxuICAgICAgICAgIHRoaXMuX2hhbmRsZVRhYmxlQmx1cihldmVudC5yZWxhdGVkVGFyZ2V0IGFzIEhUTUxFbGVtZW50KX1cbiAgICAgICAgQGFuaW1hdGlvbmVuZD0keyhlOiBBbmltYXRpb25FdmVudCkgPT4gdGhpcy5fdGFibGVBbmltYXRpb25FbmQoZSl9XG4gICAgICA+XG4gICAgICAgICR7dGhpcy53ZWVrTnVtYmVyc1xuICAgICAgICAgID8gaHRtbGBcbiAgICAgICAgICAgICAgPHRoZWFkIGNsYXNzPVwic2JiLWNhbGVuZGFyX190YWJsZS1oZWFkZXJcIj5cbiAgICAgICAgICAgICAgICA8dHI+XG4gICAgICAgICAgICAgICAgICAke25leHRNb250aEFjdGl2ZURhdGVcbiAgICAgICAgICAgICAgICAgICAgPyBub3RoaW5nXG4gICAgICAgICAgICAgICAgICAgIDogaHRtbGA8dGggY2xhc3M9XCJzYmItY2FsZW5kYXJfX3RhYmxlLWRhdGFcIj48L3RoPmB9XG4gICAgICAgICAgICAgICAgICAke3dlZWtOdW1iZXJzLm1hcChcbiAgICAgICAgICAgICAgICAgICAgKHdlZWtOdW1iZXI6IG51bWJlcikgPT4gaHRtbGBcbiAgICAgICAgICAgICAgICAgICAgICA8dGggY2xhc3M9XCJzYmItY2FsZW5kYXJfX3RhYmxlLWhlYWRlci1jZWxsXCI+XG4gICAgICAgICAgICAgICAgICAgICAgICAke3RoaXMubXVsdGlwbGVcbiAgICAgICAgICAgICAgICAgICAgICAgICAgPyBodG1sYFxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPHNiYi1jYWxlbmRhci13ZWVrbnVtYmVyXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIC52YWx1ZT0ke3dlZWtOdW1iZXJ9XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIEBjbGljaz0keygpID0+IHtcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBjb25zdCBkYXlzOiBEYXk8VD5bXSA9IHdlZWtzRm9yU2VsZWN0TXVsdGlwbGVXZWVrTnVtYmVycy5maWx0ZXIoXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAoZGF5OiBEYXk8VD4pID0+IGRheS53ZWVrVmFsdWUgPT09IHdlZWtOdW1iZXIsXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgKSE7XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgdGhpcy5fc2VsZWN0TXVsdGlwbGVEYXRlcyhkYXlzKTtcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgfX1cbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID48L3NiYi1jYWxlbmRhci13ZWVrbnVtYmVyPlxuICAgICAgICAgICAgICAgICAgICAgICAgICAgIGBcbiAgICAgICAgICAgICAgICAgICAgICAgICAgOiBodG1sYFxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPHNiYi1zY3JlZW4tcmVhZGVyLW9ubHlcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPiR7YCR7aTE4bkNhbGVuZGFyV2Vla051bWJlclt0aGlzLl9sYW5ndWFnZS5jdXJyZW50XX0gJHt3ZWVrTnVtYmVyfWB9PC9zYmItc2NyZWVuLXJlYWRlci1vbmx5XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICA+XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICA8c3BhbiBhcmlhLWhpZGRlbj1cInRydWVcIj4ke3dlZWtOdW1iZXJ9PC9zcGFuPlxuICAgICAgICAgICAgICAgICAgICAgICAgICAgIGB9XG4gICAgICAgICAgICAgICAgICAgICAgPC90aD5cbiAgICAgICAgICAgICAgICAgICAgYCxcbiAgICAgICAgICAgICAgICAgICl9XG4gICAgICAgICAgICAgICAgPC90cj5cbiAgICAgICAgICAgICAgPC90aGVhZD5cbiAgICAgICAgICAgIGBcbiAgICAgICAgICA6IG5vdGhpbmd9XG4gICAgICAgIDx0Ym9keSBjbGFzcz1cInNiYi1jYWxlbmRhcl9fdGFibGUtYm9keVwiPlxuICAgICAgICAgICR7d2Vla3MubWFwKCh3ZWVrOiBEYXk8VD5bXSwgcm93SW5kZXg6IG51bWJlcikgPT4ge1xuICAgICAgICAgICAgY29uc3Qgd2Vla2RheSA9IHRoaXMuX3dlZWtkYXlzW3Jvd0luZGV4XTtcbiAgICAgICAgICAgIGNvbnN0IHNlbGVjdGFibGVEYXlzID0gdGhpcy5fd2lkZSA/IFsuLi53ZWVrLCAuLi50aGlzLl9uZXh0TW9udGhXZWVrc1tyb3dJbmRleF1dIDogd2VlaztcbiAgICAgICAgICAgIHJldHVybiBodG1sYFxuICAgICAgICAgICAgICA8dHI+XG4gICAgICAgICAgICAgICAgJHtuZXh0TW9udGhBY3RpdmVEYXRlXG4gICAgICAgICAgICAgICAgICA/IG5vdGhpbmdcbiAgICAgICAgICAgICAgICAgIDogaHRtbGBcbiAgICAgICAgICAgICAgICAgICAgICA8dGQgY2xhc3M9XCJzYmItY2FsZW5kYXJfX3RhYmxlLWhlYWRlci1jZWxsLXZlcnRpY2FsXCI+XG4gICAgICAgICAgICAgICAgICAgICAgICAke3RoaXMubXVsdGlwbGVcbiAgICAgICAgICAgICAgICAgICAgICAgICAgPyBodG1sYFxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPHNiYi1jYWxlbmRhci13ZWVrZGF5XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIC52YWx1ZT0ke3dlZWtkYXl9XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIEBjbGljaz0keygpID0+IHRoaXMuX3NlbGVjdE11bHRpcGxlRGF0ZXMoc2VsZWN0YWJsZURheXMpfVxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPlxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAke3dlZWtkYXkubmFycm93fVxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPC9zYmItY2FsZW5kYXItd2Vla2RheT5cbiAgICAgICAgICAgICAgICAgICAgICAgICAgICBgXG4gICAgICAgICAgICAgICAgICAgICAgICAgIDogaHRtbGBcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIDxzYmItc2NyZWVuLXJlYWRlci1vbmx5PiR7d2Vla2RheS5sb25nfTwvc2JiLXNjcmVlbi1yZWFkZXItb25seT5cbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIDxzcGFuIGFyaWEtaGlkZGVuPVwidHJ1ZVwiPiR7d2Vla2RheS5uYXJyb3d9PC9zcGFuPlxuICAgICAgICAgICAgICAgICAgICAgICAgICAgIGB9XG4gICAgICAgICAgICAgICAgICAgICAgPC90ZD5cbiAgICAgICAgICAgICAgICAgICAgYH1cbiAgICAgICAgICAgICAgICAke3Jvd0luZGV4IDwgd2Vla09mZnNldFxuICAgICAgICAgICAgICAgICAgPyBodG1sYDx0ZCBjbGFzcz1cInNiYi1jYWxlbmRhcl9fdGFibGUtZGF0YVwiPjwvdGQ+YFxuICAgICAgICAgICAgICAgICAgOiBub3RoaW5nfVxuICAgICAgICAgICAgICAgICR7dGhpcy5fY3JlYXRlRGF5Q2VsbHMod2Vlayl9XG4gICAgICAgICAgICAgIDwvdHI+XG4gICAgICAgICAgICBgO1xuICAgICAgICAgIH0pfVxuICAgICAgICA8L3Rib2R5PlxuICAgICAgPC90YWJsZT5cbiAgICBgO1xuICB9XG5cbiAgLyoqIENyZWF0ZXMgdGhlIGNlbGxzIGZvciB0aGUgZGFpbHkgdmlldy4gKi9cbiAgcHJpdmF0ZSBfY3JlYXRlRGF5Q2VsbHMod2VlazogRGF5PFQ+W10pOiBUZW1wbGF0ZVJlc3VsdFtdIHtcbiAgICByZXR1cm4gd2Vlay5tYXAoKGRheTogRGF5PFQ+KSA9PiB7XG4gICAgICByZXR1cm4gaHRtbGBcbiAgICAgICAgPHRkIGNsYXNzPVwic2JiLWNhbGVuZGFyX190YWJsZS1kYXRhIHNiYi1jYWxlbmRhcl9fZGF5LWNlbGxcIj5cbiAgICAgICAgICA8c2xvdCBuYW1lPSR7ZGF5LnZhbHVlfT5cbiAgICAgICAgICAgIDxzYmItY2FsZW5kYXItZGF5XG4gICAgICAgICAgICAgIHNsb3Q9JHtkYXkudmFsdWV9XG4gICAgICAgICAgICAgIEBjbGljaz0keygpID0+IHRoaXMuX3NlbGVjdERhdGUoZGF5LmRhdGVWYWx1ZSl9XG4gICAgICAgICAgICAgIEBrZXlkb3duPSR7KGV2dDogS2V5Ym9hcmRFdmVudCkgPT4gdGhpcy5faGFuZGxlS2V5Ym9hcmRFdmVudChldnQsIGRheSl9XG4gICAgICAgICAgICA+PC9zYmItY2FsZW5kYXItZGF5PlxuICAgICAgICAgIDwvc2xvdD5cbiAgICAgICAgPC90ZD5cbiAgICAgIGA7XG4gICAgfSk7XG4gIH1cblxuICAvKiogUmVuZGVyIHRoZSB2aWV3IGZvciB0aGUgbW9udGggc2VsZWN0aW9uLiAqL1xuICBwcml2YXRlIF9yZW5kZXJNb250aFZpZXcoKTogVGVtcGxhdGVSZXN1bHQge1xuICAgIHJldHVybiBodG1sYFxuICAgICAgPGRpdiBjbGFzcz1cInNiYi1jYWxlbmRhcl9fY29udHJvbHNcIj5cbiAgICAgICAgJHt0aGlzLl9nZXRBcnJvdyhcbiAgICAgICAgICAnbGVmdCcsXG4gICAgICAgICAgKCkgPT4gdGhpcy5fZ29Ub0RpZmZlcmVudFllYXIoLTEpLFxuICAgICAgICAgIGkxOG5QcmV2aW91c1llYXJbdGhpcy5fbGFuZ3VhZ2UuY3VycmVudF0sXG4gICAgICAgICAgdGhpcy5fcHJldmlvdXNZZWFyRGlzYWJsZWQoKSxcbiAgICAgICAgKX1cbiAgICAgICAgPGRpdiBjbGFzcz1cInNiYi1jYWxlbmRhcl9fY29udHJvbHMtbW9udGhcIj4ke3RoaXMuX2NyZWF0ZUxhYmVsRm9yTW9udGhWaWV3KCl9PC9kaXY+XG4gICAgICAgICR7dGhpcy5fZ2V0QXJyb3coXG4gICAgICAgICAgJ3JpZ2h0JyxcbiAgICAgICAgICAoKSA9PiB0aGlzLl9nb1RvRGlmZmVyZW50WWVhcigxKSxcbiAgICAgICAgICBpMThuTmV4dFllYXJbdGhpcy5fbGFuZ3VhZ2UuY3VycmVudF0sXG4gICAgICAgICAgdGhpcy5fbmV4dFllYXJEaXNhYmxlZCgpLFxuICAgICAgICApfVxuICAgICAgPC9kaXY+XG4gICAgICA8ZGl2IGNsYXNzPVwic2JiLWNhbGVuZGFyX190YWJsZS1vdmVyZmxvdy1icmVha1wiPlxuICAgICAgICA8ZGl2IGNsYXNzPVwic2JiLWNhbGVuZGFyX190YWJsZS1jb250YWluZXIgc2JiLWNhbGVuZGFyX190YWJsZS1tb250aC12aWV3XCI+XG4gICAgICAgICAgJHt0aGlzLl9jcmVhdGVNb250aFRhYmxlKHRoaXMuX21vbnRocywgdGhpcy5fY2hvc2VuWWVhciEpfVxuICAgICAgICAgICR7dGhpcy5fd2lkZSA/IHRoaXMuX2NyZWF0ZU1vbnRoVGFibGUodGhpcy5fbW9udGhzLCB0aGlzLl9jaG9zZW5ZZWFyISArIDEpIDogbm90aGluZ31cbiAgICAgICAgPC9kaXY+XG4gICAgICA8L2Rpdj5cbiAgICBgO1xuICB9XG5cbiAgLyoqIENyZWF0ZXMgdGhlIGxhYmVsIHdpdGggdGhlIHllYXIgZm9yIHRoZSBtb250aGx5IHZpZXcuICovXG4gIHByaXZhdGUgX2NyZWF0ZUxhYmVsRm9yTW9udGhWaWV3KCk6IFRlbXBsYXRlUmVzdWx0IHtcbiAgICByZXR1cm4gaHRtbGAgPGJ1dHRvblxuICAgICAgICB0eXBlPVwiYnV0dG9uXCJcbiAgICAgICAgaWQ9XCJzYmItY2FsZW5kYXJfX21vbnRoLXNlbGVjdGlvblwiXG4gICAgICAgIGNsYXNzPVwic2JiLWNhbGVuZGFyX19jb250cm9scy1jaGFuZ2UtZGF0ZVwiXG4gICAgICAgIGFyaWEtbGFiZWw9JHtgJHtpMThuQ2FsZW5kYXJEYXRlU2VsZWN0aW9uW3RoaXMuX2xhbmd1YWdlLmN1cnJlbnRdfSAke3RoaXMuX2Nob3NlblllYXJ9YH1cbiAgICAgICAgQGNsaWNrPSR7KCkgPT4gdGhpcy5fcmVzZXRDYWxlbmRhclZpZXdBbmRFbWl0TW9udGhDaGFuZ2UodHJ1ZSl9XG4gICAgICA+XG4gICAgICAgICR7dGhpcy5fY2hvc2VuWWVhcn0gJHt0aGlzLl93aWRlID8gYCAtICR7dGhpcy5fY2hvc2VuWWVhciEgKyAxfWAgOiBub3RoaW5nfVxuICAgICAgICA8c2JiLWljb24gbmFtZT1cImNoZXZyb24tc21hbGwtdXAtc21hbGxcIj48L3NiYi1pY29uPlxuICAgICAgPC9idXR0b24+XG4gICAgICA8c2JiLXNjcmVlbi1yZWFkZXItb25seSByb2xlPVwic3RhdHVzXCI+ICR7dGhpcy5fY2hvc2VuWWVhcn0gPC9zYmItc2NyZWVuLXJlYWRlci1vbmx5PmA7XG4gIH1cblxuICAvKiogQ3JlYXRlcyB0aGUgdGFibGUgZm9yIHRoZSBtb250aCBzZWxlY3Rpb24gdmlldy4gKi9cbiAgcHJpdmF0ZSBfY3JlYXRlTW9udGhUYWJsZShtb250aHM6IE1vbnRoQ2VsbFtdW10sIHllYXI6IG51bWJlcik6IFRlbXBsYXRlUmVzdWx0IHtcbiAgICByZXR1cm4gaHRtbGBcbiAgICAgIDx0YWJsZVxuICAgICAgICBjbGFzcz1cInNiYi1jYWxlbmRhcl9fdGFibGVcIlxuICAgICAgICBAYW5pbWF0aW9uZW5kPSR7KGU6IEFuaW1hdGlvbkV2ZW50KSA9PiB0aGlzLl90YWJsZUFuaW1hdGlvbkVuZChlKX1cbiAgICAgID5cbiAgICAgICAgJHt0aGlzLl93aWRlXG4gICAgICAgICAgPyBodG1sYDx0aGVhZCBjbGFzcz1cInNiYi1jYWxlbmRhcl9fdGFibGUtaGVhZGVyXCIgYXJpYS1oaWRkZW49XCJ0cnVlXCI+XG4gICAgICAgICAgICAgIDx0cj5cbiAgICAgICAgICAgICAgICA8dGggY2xhc3M9XCJzYmItY2FsZW5kYXJfX3RhYmxlLWhlYWRlci1jZWxsXCIgY29sc3Bhbj0ke01PTlRIU19QRVJfUk9XfT4ke3llYXJ9PC90aD5cbiAgICAgICAgICAgICAgPC90cj5cbiAgICAgICAgICAgIDwvdGhlYWQ+YFxuICAgICAgICAgIDogbm90aGluZ31cbiAgICAgICAgPHRib2R5IGNsYXNzPVwic2JiLWNhbGVuZGFyX190YWJsZS1ib2R5XCI+XG4gICAgICAgICAgJHttb250aHMubWFwKFxuICAgICAgICAgICAgKHJvdzogTW9udGhDZWxsW10pID0+IGh0bWxgXG4gICAgICAgICAgICAgIDx0cj5cbiAgICAgICAgICAgICAgICAke3Jvdy5tYXAoKG1vbnRoOiBNb250aENlbGwpID0+IHtcbiAgICAgICAgICAgICAgICAgIHJldHVybiBodG1sYFxuICAgICAgICAgICAgICAgICAgICA8dGQgY2xhc3M9XCJzYmItY2FsZW5kYXJfX3RhYmxlLWRhdGFcIj5cbiAgICAgICAgICAgICAgICAgICAgICA8c2JiLWNhbGVuZGFyLW1vbnRoXG4gICAgICAgICAgICAgICAgICAgICAgICAudmFsdWU9XCIke3llYXJ9LSR7bW9udGgudmFsdWV9XCJcbiAgICAgICAgICAgICAgICAgICAgICAgIEBjbGljaz0keygpID0+IHRoaXMuX29uTW9udGhTZWxlY3Rpb24obW9udGgubW9udGhWYWx1ZSwgeWVhcil9XG4gICAgICAgICAgICAgICAgICAgICAgICBAa2V5ZG93bj0keyhldnQ6IEtleWJvYXJkRXZlbnQpID0+IHRoaXMuX2hhbmRsZUtleWJvYXJkRXZlbnQoZXZ0KX1cbiAgICAgICAgICAgICAgICAgICAgICA+XG4gICAgICAgICAgICAgICAgICAgICAgPC9zYmItY2FsZW5kYXItbW9udGg+XG4gICAgICAgICAgICAgICAgICAgIDwvdGQ+XG4gICAgICAgICAgICAgICAgICBgO1xuICAgICAgICAgICAgICAgIH0pfVxuICAgICAgICAgICAgICA8L3RyPlxuICAgICAgICAgICAgYCxcbiAgICAgICAgICApfVxuICAgICAgICA8L3Rib2R5PlxuICAgICAgPC90YWJsZT5cbiAgICBgO1xuICB9XG5cbiAgLyoqIFNlbGVjdCB0aGUgbW9udGggYW5kIGNoYW5nZSB0aGUgdmlldyB0byBkYXkgc2VsZWN0aW9uLiAqL1xuICBwcml2YXRlIF9vbk1vbnRoU2VsZWN0aW9uKG1vbnRoOiBudW1iZXIsIHllYXI6IG51bWJlcik6IHZvaWQge1xuICAgIHRoaXMuX2Nob3Nlbk1vbnRoID0gbW9udGg7XG4gICAgdGhpcy5fbmV4dENhbGVuZGFyVmlldyA9ICdkYXknO1xuICAgIHRoaXMuX2luaXQoXG4gICAgICB0aGlzLl9kYXRlQWRhcHRlci5jcmVhdGVEYXRlKFxuICAgICAgICB5ZWFyLFxuICAgICAgICB0aGlzLl9jaG9zZW5Nb250aCxcbiAgICAgICAgdGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0RGF0ZSh0aGlzLl9hY3RpdmVEYXRlKSxcbiAgICAgICksXG4gICAgKTtcbiAgICB0aGlzLl9zdGFydFRhYmxlVHJhbnNpdGlvbigpO1xuICAgIHRoaXMuX2VtaXRNb250aENoYW5nZSgpO1xuICB9XG5cbiAgLyoqIFJlbmRlciB0aGUgdmlldyBmb3IgdGhlIHllYXIgc2VsZWN0aW9uLiAqL1xuICBwcml2YXRlIF9yZW5kZXJZZWFyVmlldygpOiBUZW1wbGF0ZVJlc3VsdCB7XG4gICAgcmV0dXJuIGh0bWxgXG4gICAgICA8ZGl2IGNsYXNzPVwic2JiLWNhbGVuZGFyX19jb250cm9sc1wiPlxuICAgICAgICAke3RoaXMuX2dldEFycm93KFxuICAgICAgICAgICdsZWZ0JyxcbiAgICAgICAgICAoKSA9PiB0aGlzLl9nb1RvRGlmZmVyZW50WWVhclJhbmdlKC1ZRUFSU19QRVJfUEFHRSksXG4gICAgICAgICAgaTE4blByZXZpb3VzWWVhclJhbmdlKFlFQVJTX1BFUl9QQUdFKVt0aGlzLl9sYW5ndWFnZS5jdXJyZW50XSxcbiAgICAgICAgICB0aGlzLl9wcmV2aW91c1llYXJSYW5nZURpc2FibGVkKCksXG4gICAgICAgICl9XG4gICAgICAgIDxkaXYgY2xhc3M9XCJzYmItY2FsZW5kYXJfX2NvbnRyb2xzLW1vbnRoXCI+JHt0aGlzLl9jcmVhdGVMYWJlbEZvclllYXJWaWV3KCl9PC9kaXY+XG4gICAgICAgICR7dGhpcy5fZ2V0QXJyb3coXG4gICAgICAgICAgJ3JpZ2h0JyxcbiAgICAgICAgICAoKSA9PiB0aGlzLl9nb1RvRGlmZmVyZW50WWVhclJhbmdlKFlFQVJTX1BFUl9QQUdFKSxcbiAgICAgICAgICBpMThuTmV4dFllYXJSYW5nZShZRUFSU19QRVJfUEFHRSlbdGhpcy5fbGFuZ3VhZ2UuY3VycmVudF0sXG4gICAgICAgICAgdGhpcy5fbmV4dFllYXJSYW5nZURpc2FibGVkKCksXG4gICAgICAgICl9XG4gICAgICA8L2Rpdj5cbiAgICAgIDxkaXYgY2xhc3M9XCJzYmItY2FsZW5kYXJfX3RhYmxlLW92ZXJmbG93LWJyZWFrXCI+XG4gICAgICAgIDxkaXYgY2xhc3M9XCJzYmItY2FsZW5kYXJfX3RhYmxlLWNvbnRhaW5lciBzYmItY2FsZW5kYXJfX3RhYmxlLXllYXItdmlld1wiPlxuICAgICAgICAgICR7dGhpcy5fY3JlYXRlWWVhclRhYmxlKHRoaXMuX3llYXJzKX1cbiAgICAgICAgICAke3RoaXMuX3dpZGUgPyB0aGlzLl9jcmVhdGVZZWFyVGFibGUodGhpcy5fbmV4dE1vbnRoWWVhcnMsIHRydWUpIDogbm90aGluZ31cbiAgICAgICAgPC9kaXY+XG4gICAgICA8L2Rpdj5cbiAgICBgO1xuICB9XG5cbiAgLyoqIENyZWF0ZXMgdGhlIGJ1dHRvbiBhcnJvdyBmb3IgYWxsIHRoZSB2aWV3cy4gKi9cbiAgcHJpdmF0ZSBfZ2V0QXJyb3coXG4gICAgZGlyZWN0aW9uOiAnbGVmdCcgfCAncmlnaHQnLFxuICAgIGNsaWNrOiAoKSA9PiB2b2lkLFxuICAgIGFyaWFMYWJlbDogc3RyaW5nLFxuICAgIGRpc2FibGVkOiBib29sZWFuLFxuICApOiBUZW1wbGF0ZVJlc3VsdCB7XG4gICAgcmV0dXJuIGh0bWxgPHNiYi1zZWNvbmRhcnktYnV0dG9uXG4gICAgICBzaXplPVwibVwiXG4gICAgICBpY29uLW5hbWU9XCJjaGV2cm9uLXNtYWxsLSR7ZGlyZWN0aW9ufS1zbWFsbFwiXG4gICAgICBhcmlhLWxhYmVsPSR7YXJpYUxhYmVsfVxuICAgICAgQGNsaWNrPSR7Y2xpY2t9XG4gICAgICA/ZGlzYWJsZWQ9JHtkaXNhYmxlZH1cbiAgICAgIGlkPVwic2JiLWNhbGVuZGFyX19jb250cm9scy0ke2RpcmVjdGlvbiA9PT0gJ2xlZnQnID8gJ3ByZXZpb3VzJyA6ICduZXh0J31cIlxuICAgID48L3NiYi1zZWNvbmRhcnktYnV0dG9uPmA7XG4gIH1cblxuICAvKiogQ3JlYXRlcyB0aGUgbGFiZWwgd2l0aCB0aGUgeWVhciByYW5nZSBmb3IgdGhlIHllYXJseSB2aWV3LiAqL1xuICBwcml2YXRlIF9jcmVhdGVMYWJlbEZvclllYXJWaWV3KCk6IFRlbXBsYXRlUmVzdWx0IHtcbiAgICBjb25zdCBmaXJzdFllYXI6IG51bWJlciA9IHRoaXMuX3llYXJzLmZsYXQoKVswXTtcbiAgICBjb25zdCBsYXN0WWVhckFycmF5OiBudW1iZXJbXSA9ICh0aGlzLl93aWRlID8gdGhpcy5fbmV4dE1vbnRoWWVhcnMgOiB0aGlzLl95ZWFycykuZmxhdCgpO1xuICAgIGNvbnN0IGxhc3RZZWFyOiBudW1iZXIgPSBsYXN0WWVhckFycmF5W2xhc3RZZWFyQXJyYXkubGVuZ3RoIC0gMV07XG4gICAgY29uc3QgeWVhckxhYmVsID0gYCR7Zmlyc3RZZWFyfSAtICR7bGFzdFllYXJ9YDtcbiAgICByZXR1cm4gaHRtbGBcbiAgICAgIDxidXR0b25cbiAgICAgICAgdHlwZT1cImJ1dHRvblwiXG4gICAgICAgIGlkPVwic2JiLWNhbGVuZGFyX195ZWFyLXNlbGVjdGlvblwiXG4gICAgICAgIGNsYXNzPVwic2JiLWNhbGVuZGFyX19jb250cm9scy1jaGFuZ2UtZGF0ZVwiXG4gICAgICAgIGFyaWEtbGFiZWw9XCIke2kxOG5DYWxlbmRhckRhdGVTZWxlY3Rpb25bdGhpcy5fbGFuZ3VhZ2UuY3VycmVudF19ICR7eWVhckxhYmVsfVwiXG4gICAgICAgIEBjbGljaz0keygpID0+IHRoaXMuX3Jlc2V0Q2FsZW5kYXJWaWV3QW5kRW1pdE1vbnRoQ2hhbmdlKHRydWUpfVxuICAgICAgPlxuICAgICAgICAke3llYXJMYWJlbH1cbiAgICAgICAgPHNiYi1pY29uIG5hbWU9XCJjaGV2cm9uLXNtYWxsLXVwLXNtYWxsXCI+PC9zYmItaWNvbj5cbiAgICAgIDwvYnV0dG9uPlxuICAgICAgPHNiYi1zY3JlZW4tcmVhZGVyLW9ubHkgcm9sZT1cInN0YXR1c1wiPiAke3llYXJMYWJlbH0gPC9zYmItc2NyZWVuLXJlYWRlci1vbmx5PlxuICAgIGA7XG4gIH1cblxuICAvKiogQ3JlYXRlcyB0aGUgdGFibGUgZm9yIHRoZSB5ZWFyIHNlbGVjdGlvbiB2aWV3LiAqL1xuICBwcml2YXRlIF9jcmVhdGVZZWFyVGFibGUoeWVhcnM6IG51bWJlcltdW10sIHNoaWZ0UmlnaHQgPSBmYWxzZSk6IFRlbXBsYXRlUmVzdWx0IHtcbiAgICByZXR1cm4gaHRtbGAgPHRhYmxlXG4gICAgICBjbGFzcz1cInNiYi1jYWxlbmRhcl9fdGFibGVcIlxuICAgICAgQGFuaW1hdGlvbmVuZD0keyhlOiBBbmltYXRpb25FdmVudCkgPT4gdGhpcy5fdGFibGVBbmltYXRpb25FbmQoZSl9XG4gICAgPlxuICAgICAgPHRib2R5IGNsYXNzPVwic2JiLWNhbGVuZGFyX190YWJsZS1ib2R5XCI+XG4gICAgICAgICR7eWVhcnMubWFwKFxuICAgICAgICAgIChyb3c6IG51bWJlcltdKSA9PlxuICAgICAgICAgICAgaHRtbGAgPHRyPlxuICAgICAgICAgICAgICAke3Jvdy5tYXAoKHllYXI6IG51bWJlcikgPT4ge1xuICAgICAgICAgICAgICAgIHJldHVybiBodG1sYFxuICAgICAgICAgICAgICAgICAgPHRkIGNsYXNzPVwic2JiLWNhbGVuZGFyX190YWJsZS1kYXRhXCI+XG4gICAgICAgICAgICAgICAgICAgIDxzYmItY2FsZW5kYXIteWVhclxuICAgICAgICAgICAgICAgICAgICAgIC52YWx1ZT0ke1N0cmluZyh5ZWFyKX1cbiAgICAgICAgICAgICAgICAgICAgICBAa2V5ZG93bj0keyhldnQ6IEtleWJvYXJkRXZlbnQpID0+IHRoaXMuX2hhbmRsZUtleWJvYXJkRXZlbnQoZXZ0KX1cbiAgICAgICAgICAgICAgICAgICAgICBAY2xpY2s9JHsoKSA9PiB0aGlzLl9vblllYXJTZWxlY3Rpb24oeWVhciwgc2hpZnRSaWdodCl9XG4gICAgICAgICAgICAgICAgICAgID5cbiAgICAgICAgICAgICAgICAgICAgPC9zYmItY2FsZW5kYXIteWVhcj5cbiAgICAgICAgICAgICAgICAgIDwvdGQ+XG4gICAgICAgICAgICAgICAgYDtcbiAgICAgICAgICAgICAgfSl9XG4gICAgICAgICAgICA8L3RyPmAsXG4gICAgICAgICl9XG4gICAgICA8L3Rib2R5PlxuICAgIDwvdGFibGU+YDtcbiAgfVxuXG4gIC8qKiBTZWxlY3QgdGhlIHllYXIgYW5kIGNoYW5nZSB0aGUgdmlldyB0byBtb250aCBzZWxlY3Rpb24uICovXG4gIHByaXZhdGUgX29uWWVhclNlbGVjdGlvbih5ZWFyOiBudW1iZXIsIHJpZ2h0U2lkZTogYm9vbGVhbik6IHZvaWQge1xuICAgIHRoaXMuX2Nob3NlblllYXIgPSByaWdodFNpZGUgPyB5ZWFyIC0gMSA6IHllYXI7XG4gICAgdGhpcy5fbmV4dENhbGVuZGFyVmlldyA9ICdtb250aCc7XG4gICAgdGhpcy5fYXNzaWduQWN0aXZlRGF0ZShcbiAgICAgIHRoaXMuX2RhdGVBZGFwdGVyLmNyZWF0ZURhdGUoXG4gICAgICAgIHRoaXMuX2Nob3NlblllYXIsXG4gICAgICAgIHRoaXMuX2RhdGVBZGFwdGVyLmdldE1vbnRoKHRoaXMuX2FjdGl2ZURhdGUpLFxuICAgICAgICB0aGlzLl9kYXRlQWRhcHRlci5nZXREYXRlKHRoaXMuX2FjdGl2ZURhdGUpLFxuICAgICAgKSxcbiAgICApO1xuICAgIHRoaXMuX3N0YXJ0VGFibGVUcmFuc2l0aW9uKCk7XG4gIH1cblxuICBwcml2YXRlIF9nZXRWaWV3KCk6IFRlbXBsYXRlUmVzdWx0IHtcbiAgICBpZiAoaXNTZXJ2ZXIgfHwgdGhpcy5oeWRyYXRpb25SZXF1aXJlZCkge1xuICAgICAgLy8gVE9ETzogV2UgZGlzYWJsZSBTU1IgZm9yIGNhbGVuZGFyIGZvciBub3cuIEZpZ3VyZSBvdXQsIGlmIHRoZXJlIGlzIGEgd2F5XG4gICAgICAvLyB0byBlbmFibGUgaXQsIHdoaWxlIGNvbnNpZGVyaW5nIGkxOG4gYW5kIGRhdGUgaW5mb3JtYXRpb24uXG4gICAgICByZXR1cm4gaHRtbGAke25vdGhpbmd9YDtcbiAgICB9XG4gICAgc3dpdGNoICh0aGlzLl9jYWxlbmRhclZpZXcpIHtcbiAgICAgIGNhc2UgJ3llYXInOlxuICAgICAgICByZXR1cm4gdGhpcy5fcmVuZGVyWWVhclZpZXcoKTtcbiAgICAgIGNhc2UgJ21vbnRoJzpcbiAgICAgICAgcmV0dXJuIHRoaXMuX3JlbmRlck1vbnRoVmlldygpO1xuICAgICAgY2FzZSAnZGF5JzpcbiAgICAgIGRlZmF1bHQ6XG4gICAgICAgIHJldHVybiB0aGlzLl9yZW5kZXJEYXlWaWV3KCk7XG4gICAgfVxuICB9XG5cbiAgcHJpdmF0ZSBfdGFibGVBbmltYXRpb25FbmQoZXZlbnQ6IEFuaW1hdGlvbkV2ZW50KTogdm9pZCB7XG4gICAgY29uc3QgdGFibGUgPSBldmVudC50YXJnZXQgYXMgSFRNTEVsZW1lbnQ7XG4gICAgaWYgKGV2ZW50LmFuaW1hdGlvbk5hbWUgPT09ICdoaWRlJykge1xuICAgICAgdGFibGUuY2xhc3NMaXN0LnJlbW92ZSgnc2JiLWNhbGVuZGFyX190YWJsZS1oaWRlJyk7XG4gICAgICBpZiAodGhpcy5fY29udGFpbmluZ0ZvY3VzKSB7XG4gICAgICAgIHRoaXMuX3Jlc2V0Rm9jdXMgPSB0cnVlO1xuICAgICAgfVxuICAgICAgdGhpcy5fY2FsZW5kYXJWaWV3ID0gdGhpcy5fbmV4dENhbGVuZGFyVmlldztcbiAgICB9IGVsc2UgaWYgKGV2ZW50LmFuaW1hdGlvbk5hbWUgPT09ICdzaG93Jykge1xuICAgICAgdGhpcy5pbnRlcm5hbHMuc3RhdGVzLmRlbGV0ZSgndHJhbnNpdGlvbicpO1xuICAgIH1cbiAgfVxuXG4gIHByaXZhdGUgX3N0YXJ0VGFibGVUcmFuc2l0aW9uKCk6IHZvaWQge1xuICAgIHRoaXMuaW50ZXJuYWxzLnN0YXRlcy5hZGQoJ3RyYW5zaXRpb24nKTtcbiAgICB0aGlzLnNoYWRvd1Jvb3RcbiAgICAgID8ucXVlcnlTZWxlY3RvckFsbCgndGFibGUnKVxuICAgICAgPy5mb3JFYWNoKChlKSA9PiBlLmNsYXNzTGlzdC50b2dnbGUoJ3NiYi1jYWxlbmRhcl9fdGFibGUtaGlkZScpKTtcbiAgfVxuXG4gIHByb3RlY3RlZCBvdmVycmlkZSByZW5kZXIoKTogVGVtcGxhdGVSZXN1bHQge1xuICAgIHJldHVybiBodG1sYDxkaXYgY2xhc3M9XCJzYmItY2FsZW5kYXJfX3dyYXBwZXJcIj4ke3RoaXMuX2dldFZpZXcoKX08L2Rpdj5gO1xuICB9XG59XG5cbmRlY2xhcmUgZ2xvYmFsIHtcbiAgaW50ZXJmYWNlIEhUTUxFbGVtZW50VGFnTmFtZU1hcCB7XG4gICAgLy8gZXNsaW50LWRpc2FibGUtbmV4dC1saW5lIEB0eXBlc2NyaXB0LWVzbGludC9uYW1pbmctY29udmVudGlvblxuICAgICdzYmItY2FsZW5kYXInOiBTYmJDYWxlbmRhckVsZW1lbnQ7XG4gIH1cbiAgaW50ZXJmYWNlIEhUTUxFbGVtZW50RXZlbnRNYXAge1xuICAgIG1vbnRoY2hhbmdlOiBTYmJNb250aENoYW5nZUV2ZW50O1xuICB9XG59XG4iXSwibWFwcGluZ3MiOiI7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7OztBQzJEQSwwQkFBMEIsUUFBUTtBQUVsQyxzQkFBc0IsUUFBUTtBQUM5Qix3QkFBd0IsUUFBUTtBQUNoQyx1QkFBdUIsUUFBUTtBQUMvQiwwQkFBMEIsUUFBUTtBQUNsQyw2QkFBNkIsUUFBUTtBQUVyQyxJQUFhLHNCQUFiLGNBQXlDLE1BQUs7Q0FHNUMsSUFBVyxRQUFLO0FBQ2QsU0FBTyxLQUFLOztD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