@sbb-esta/lyne-elements-dev 5.0.0-next.3-dev.1779242419 → 5.0.0-next.3-dev.1779265349

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