@sbb-esta/lyne-elements 4.5.0 → 4.6.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (65) hide show
  1. package/autocomplete/autocomplete-base-element.js +5 -5
  2. package/calendar/{calendar.component.js → calendar/calendar.component.js} +413 -401
  3. package/calendar/calendar-day/calendar-day.component.js +98 -0
  4. package/calendar/calendar-day.js +4 -0
  5. package/calendar/calendar.js +5 -0
  6. package/calendar.js +5 -2
  7. package/chip-label/chip-label.component.js +13 -13
  8. package/container/container/container.component.js +9 -9
  9. package/container/sticky-bar/sticky-bar.component.js +12 -12
  10. package/core/datetime/temporal-date-adapter.js +12 -6
  11. package/core/datetime.js +12 -13
  12. package/core/styles/core.scss +14 -12
  13. package/core/styles/mixins/chip.scss +6 -6
  14. package/core.css +22 -8
  15. package/custom-elements.json +18264 -17381
  16. package/datepicker/datepicker/datepicker.component.js +1 -1
  17. package/development/autocomplete/autocomplete-base-element.d.ts.map +1 -1
  18. package/development/autocomplete/autocomplete-base-element.js +8 -6
  19. package/development/calendar/{calendar.component.d.ts → calendar/calendar.component.d.ts} +23 -3
  20. package/development/calendar/calendar/calendar.component.d.ts.map +1 -0
  21. package/development/calendar/calendar/calendar.component.js +1990 -0
  22. package/development/calendar/calendar-day/calendar-day.component.d.ts +49 -0
  23. package/development/calendar/calendar-day/calendar-day.component.d.ts.map +1 -0
  24. package/development/calendar/calendar-day/calendar-day.component.js +255 -0
  25. package/development/calendar/calendar-day.d.ts +5 -0
  26. package/development/calendar/calendar-day.d.ts.map +1 -0
  27. package/development/calendar/calendar-day.js +5 -0
  28. package/development/calendar/calendar.d.ts +5 -0
  29. package/development/calendar/calendar.d.ts.map +1 -0
  30. package/development/calendar/calendar.js +6 -0
  31. package/development/calendar.d.ts +2 -1
  32. package/development/calendar.d.ts.map +1 -1
  33. package/development/calendar.js +6 -3
  34. package/development/chip-label/chip-label.component.js +6 -15
  35. package/development/container/container/container.component.js +1 -3
  36. package/development/container/sticky-bar/sticky-bar.component.js +9 -14
  37. package/development/core/datetime/native-date-adapter.js +1 -1
  38. package/development/core/datetime/temporal-date-adapter.d.ts +7 -1
  39. package/development/core/datetime/temporal-date-adapter.d.ts.map +1 -1
  40. package/development/core/datetime/temporal-date-adapter.js +14 -7
  41. package/development/core/datetime.js +2 -3
  42. package/development/datepicker/datepicker/datepicker.component.d.ts.map +1 -1
  43. package/development/datepicker/datepicker/datepicker.component.js +2 -1
  44. package/development/dialog/dialog-title/dialog-title.component.js +2 -1
  45. package/development/expansion-panel/expansion-panel/expansion-panel.component.js +6 -5
  46. package/development/form-field/form-field/form-field.component.js +7 -5
  47. package/development/radio-button/radio-button-group/radio-button-group.component.js +2 -2
  48. package/development/tabs/tab/tab.component.js +5 -4
  49. package/development/tabs/tab-group/tab-group.component.d.ts +1 -0
  50. package/development/tabs/tab-group/tab-group.component.d.ts.map +1 -1
  51. package/development/tabs/tab-group/tab-group.component.js +7 -1
  52. package/dialog/dialog-title/dialog-title.component.js +7 -7
  53. package/expansion-panel/expansion-panel/expansion-panel.component.js +9 -9
  54. package/form-field/form-field/form-field.component.js +11 -11
  55. package/index.d.ts +3 -1
  56. package/index.js +3 -1
  57. package/off-brand-theme.css +22 -8
  58. package/package.json +11 -1
  59. package/radio-button/radio-button-group/radio-button-group.component.js +1 -1
  60. package/safety-theme.css +22 -8
  61. package/standard-theme.css +22 -8
  62. package/tabs/tab/tab.component.js +1 -1
  63. package/tabs/tab-group/tab-group.component.js +21 -18
  64. package/development/calendar/calendar.component.d.ts.map +0 -1
  65. package/development/calendar/calendar.component.js +0 -1957
@@ -0,0 +1,1990 @@
1
+ var __typeError = (msg) => {
2
+ throw TypeError(msg);
3
+ };
4
+ var __accessCheck = (obj, member, msg) => member.has(obj) || __typeError("Cannot " + msg);
5
+ var __privateGet = (obj, member, getter) => (__accessCheck(obj, member, "read from private field"), getter ? getter.call(obj) : member.get(obj));
6
+ var __privateAdd = (obj, member, value) => member.has(obj) ? __typeError("Cannot add the same private member more than once") : member instanceof WeakSet ? member.add(obj) : member.set(obj, value);
7
+ var __privateSet = (obj, member, value, setter) => (__accessCheck(obj, member, "write to private field"), setter ? setter.call(obj, value) : member.set(obj, value), value);
8
+ import { __esDecorate, __runInitializers } from "tslib";
9
+ import { css, LitElement, isServer, nothing, html } from "lit";
10
+ import { customElement, property, state } from "lit/decorators.js";
11
+ import { classMap } from "lit/directives/class-map.js";
12
+ import { isArrowKeyOrPageKeysPressed } from "../../core/a11y.js";
13
+ import { readConfig } from "../../core/config.js";
14
+ import { SbbLanguageController, SbbMediaMatcherController, SbbMediaQueryBreakpointLargeAndAbove } from "../../core/controllers.js";
15
+ import { defaultDateAdapter, YEARS_PER_PAGE, DAYS_PER_ROW, MONTHS_PER_ROW, YEARS_PER_ROW, MONTHS_PER_PAGE } from "../../core/datetime.js";
16
+ import { forceType, plainDate, handleDistinctChange } from "../../core/decorators.js";
17
+ import { i18nPreviousMonth, i18nNextMonth, i18nYearMonthSelection, i18nCalendarWeekNumber, i18nPreviousYear, i18nNextYear, i18nCalendarDateSelection, i18nPreviousYearRange, i18nNextYearRange } from "../../core/i18n.js";
18
+ import { SbbHydrationMixin, SbbElementInternalsMixin } from "../../core/mixins.js";
19
+ import { boxSizingStyles } from "../../core/styles.js";
20
+ import "../../button/secondary-button.js";
21
+ import "../../icon.js";
22
+ import "../../screen-reader-only.js";
23
+ import "../calendar-day/calendar-day.component.js";
24
+ const millisecondsInWeek = 6048e5;
25
+ const constructFromSymbol = /* @__PURE__ */ Symbol.for("constructDateFrom");
26
+ function constructFrom(date, value) {
27
+ if (typeof date === "function") return date(value);
28
+ if (date && typeof date === "object" && constructFromSymbol in date)
29
+ return date[constructFromSymbol](value);
30
+ if (date instanceof Date) return new date.constructor(value);
31
+ return new Date(value);
32
+ }
33
+ function toDate(argument, context) {
34
+ return constructFrom(context || argument, argument);
35
+ }
36
+ function addDays(date, amount, options) {
37
+ const _date = toDate(date, options?.in);
38
+ if (isNaN(amount)) return constructFrom(date, NaN);
39
+ if (!amount) return _date;
40
+ _date.setDate(_date.getDate() + amount);
41
+ return _date;
42
+ }
43
+ let defaultOptions = {};
44
+ function getDefaultOptions() {
45
+ return defaultOptions;
46
+ }
47
+ function startOfWeek(date, options) {
48
+ const defaultOptions2 = getDefaultOptions();
49
+ const weekStartsOn = options?.weekStartsOn ?? options?.locale?.options?.weekStartsOn ?? defaultOptions2.weekStartsOn ?? defaultOptions2.locale?.options?.weekStartsOn ?? 0;
50
+ const _date = toDate(date, options?.in);
51
+ const day = _date.getDay();
52
+ const diff = (day < weekStartsOn ? 7 : 0) + day - weekStartsOn;
53
+ _date.setDate(_date.getDate() - diff);
54
+ _date.setHours(0, 0, 0, 0);
55
+ return _date;
56
+ }
57
+ function normalizeDates(context, ...dates) {
58
+ const normalize = constructFrom.bind(
59
+ null,
60
+ context || dates.find((date) => typeof date === "object")
61
+ );
62
+ return dates.map(normalize);
63
+ }
64
+ function addWeeks(date, amount, options) {
65
+ return addDays(date, amount * 7, options);
66
+ }
67
+ function endOfMonth(date, options) {
68
+ const _date = toDate(date, options?.in);
69
+ const month = _date.getMonth();
70
+ _date.setFullYear(_date.getFullYear(), month + 1, 0);
71
+ _date.setHours(23, 59, 59, 999);
72
+ return _date;
73
+ }
74
+ function normalizeInterval(context, interval) {
75
+ const [start, end] = normalizeDates(context, interval.start, interval.end);
76
+ return { start, end };
77
+ }
78
+ function eachWeekOfInterval(interval, options) {
79
+ const { start, end } = normalizeInterval(options?.in, interval);
80
+ let reversed = +start > +end;
81
+ const startDateWeek = reversed ? startOfWeek(end, options) : startOfWeek(start, options);
82
+ const endDateWeek = reversed ? startOfWeek(start, options) : startOfWeek(end, options);
83
+ startDateWeek.setHours(15);
84
+ endDateWeek.setHours(15);
85
+ const endTime = +endDateWeek.getTime();
86
+ let currentDate = startDateWeek;
87
+ let step = options?.step ?? 1;
88
+ if (!step) return [];
89
+ if (step < 0) {
90
+ step = -step;
91
+ reversed = !reversed;
92
+ }
93
+ const dates = [];
94
+ while (+currentDate <= endTime) {
95
+ currentDate.setHours(0);
96
+ dates.push(constructFrom(start, currentDate));
97
+ currentDate = addWeeks(currentDate, step);
98
+ currentDate.setHours(15);
99
+ }
100
+ return reversed ? dates.reverse() : dates;
101
+ }
102
+ function startOfMonth(date, options) {
103
+ const _date = toDate(date, options?.in);
104
+ _date.setDate(1);
105
+ _date.setHours(0, 0, 0, 0);
106
+ return _date;
107
+ }
108
+ function getWeekYear(date, options) {
109
+ const _date = toDate(date, options?.in);
110
+ const year = _date.getFullYear();
111
+ const defaultOptions2 = getDefaultOptions();
112
+ const firstWeekContainsDate = options?.firstWeekContainsDate ?? options?.locale?.options?.firstWeekContainsDate ?? defaultOptions2.firstWeekContainsDate ?? defaultOptions2.locale?.options?.firstWeekContainsDate ?? 1;
113
+ const firstWeekOfNextYear = constructFrom(options?.in || date, 0);
114
+ firstWeekOfNextYear.setFullYear(year + 1, 0, firstWeekContainsDate);
115
+ firstWeekOfNextYear.setHours(0, 0, 0, 0);
116
+ const startOfNextYear = startOfWeek(firstWeekOfNextYear, options);
117
+ const firstWeekOfThisYear = constructFrom(options?.in || date, 0);
118
+ firstWeekOfThisYear.setFullYear(year, 0, firstWeekContainsDate);
119
+ firstWeekOfThisYear.setHours(0, 0, 0, 0);
120
+ const startOfThisYear = startOfWeek(firstWeekOfThisYear, options);
121
+ if (+_date >= +startOfNextYear) {
122
+ return year + 1;
123
+ } else if (+_date >= +startOfThisYear) {
124
+ return year;
125
+ } else {
126
+ return year - 1;
127
+ }
128
+ }
129
+ function startOfWeekYear(date, options) {
130
+ const defaultOptions2 = getDefaultOptions();
131
+ const firstWeekContainsDate = options?.firstWeekContainsDate ?? options?.locale?.options?.firstWeekContainsDate ?? defaultOptions2.firstWeekContainsDate ?? defaultOptions2.locale?.options?.firstWeekContainsDate ?? 1;
132
+ const year = getWeekYear(date, options);
133
+ const firstWeek = constructFrom(options?.in || date, 0);
134
+ firstWeek.setFullYear(year, 0, firstWeekContainsDate);
135
+ firstWeek.setHours(0, 0, 0, 0);
136
+ const _date = startOfWeek(firstWeek, options);
137
+ return _date;
138
+ }
139
+ function getWeek(date, options) {
140
+ const _date = toDate(date, options?.in);
141
+ const diff = +startOfWeek(_date, options) - +startOfWeekYear(_date, options);
142
+ return Math.round(diff / millisecondsInWeek) + 1;
143
+ }
144
+ const style = css`:host {
145
+ display: block;
146
+ width: max-content;
147
+ --sbb-calendar-cell-size: 2.5rem;
148
+ --sbb-calendar-hover-shift: 0.0625rem;
149
+ --sbb-calendar-wide-cell-size: 4.375rem;
150
+ --sbb-calendar-header-color: var(--sbb-color-granite);
151
+ --sbb-calendar-header-color: light-dark(var(--sbb-color-granite), var(--sbb-color-smoke));
152
+ --sbb-calendar-cell-background-color: transparent;
153
+ --sbb-calendar-cell-background-color-hover: var(--sbb-background-color-3);
154
+ --sbb-calendar-cell-background-color-active: var(--sbb-background-color-4);
155
+ --sbb-calendar-cell-padding: 0.125rem;
156
+ --sbb-calendar-cell-color: var(--sbb-color-2);
157
+ --sbb-calendar-cell-border-radius: var(--sbb-border-radius-4x);
158
+ --sbb-calendar-cell-selected-border: var(--sbb-border-width-2x) solid var(--sbb-border-color-2);
159
+ --sbb-calendar-cell-selected-background-color: var(--sbb-background-color-2-inverted);
160
+ --sbb-calendar-cell-disabled-height: 0.09375rem;
161
+ --sbb-calendar-cell-disabled-width: 1.59375rem;
162
+ --sbb-calendar-cell-disabled-color: var(--sbb-color-granite);
163
+ --sbb-calendar-cell-disabled-color: light-dark(var(--sbb-color-granite), var(--sbb-color-smoke));
164
+ --sbb-calendar-cell-transition-duration: var(
165
+ --sbb-disable-animation-duration,
166
+ var(--sbb-animation-duration-2x)
167
+ );
168
+ --sbb-calendar-cell-transition-easing-function: var(--sbb-animation-easing);
169
+ --sbb-calendar-tables-gap: var(--sbb-spacing-fixed-10x);
170
+ --sbb-calendar-table-animation-shift: 0.00625rem;
171
+ --sbb-calendar-table-animation-duration: 0.1ms;
172
+ --sbb-calendar-table-column-spaces: 12;
173
+ --sbb-calendar-control-view-change-height: 2.75rem;
174
+ --sbb-calendar-control-view-change-color: var(--sbb-color-3);
175
+ --sbb-calendar-control-view-change-background: var(--sbb-background-color-1);
176
+ --sbb-calendar-cell-year-month-width: 4.8125rem;
177
+ --sbb-calendar-cell-year-month-height: 2.75rem;
178
+ }
179
+ @media (min-width: calc(37.5rem)) {
180
+ :host {
181
+ --sbb-calendar-cell-size: 2.75rem;
182
+ --sbb-calendar-wide-cell-size: 4.8125rem;
183
+ --sbb-calendar-control-view-change-height: 3rem;
184
+ }
185
+ }
186
+
187
+ :host(:is(:state(enhanced),[state--enhanced])) {
188
+ --sbb-calendar-cell-padding-block-end: var(--sbb-spacing-fixed-3x);
189
+ }
190
+
191
+ .sbb-calendar__wrapper {
192
+ width: 100%;
193
+ display: block;
194
+ transition-duration: var(--sbb-calendar-cell-transition-duration);
195
+ }
196
+
197
+ .sbb-calendar__controls {
198
+ width: 100%;
199
+ display: inline-flex;
200
+ align-items: center;
201
+ gap: var(--sbb-spacing-fixed-2x);
202
+ margin-block-end: var(--sbb-spacing-fixed-4x);
203
+ }
204
+
205
+ .sbb-calendar__controls-month {
206
+ width: 100%;
207
+ display: flex;
208
+ gap: var(--sbb-calendar-tables-gap);
209
+ }
210
+
211
+ #sbb-calendar__controls-previous,
212
+ #sbb-calendar__controls-next {
213
+ -webkit-tap-highlight-color: transparent;
214
+ }
215
+
216
+ .sbb-calendar__controls-change-date {
217
+ appearance: none;
218
+ box-sizing: border-box;
219
+ margin: 0;
220
+ outline: none;
221
+ border: none;
222
+ border-radius: 0;
223
+ padding: 0;
224
+ background: none;
225
+ -webkit-tap-highlight-color: transparent;
226
+ color: inherit;
227
+ --sbb-text-font-size: var(--sbb-text-font-size-s);
228
+ font-family: var(--sbb-typo-font-family);
229
+ font-weight: normal;
230
+ line-height: var(--sbb-typo-line-height-text);
231
+ letter-spacing: var(--sbb-typo-letter-spacing-text);
232
+ font-size: var(--sbb-text-font-size);
233
+ display: flex;
234
+ align-items: center;
235
+ margin: auto;
236
+ height: var(--sbb-calendar-control-view-change-height);
237
+ text-transform: capitalize;
238
+ cursor: var(--sbb-cursor-pointer);
239
+ padding-inline: var(--sbb-spacing-fixed-5x) var(--sbb-spacing-fixed-2x);
240
+ border-radius: var(--sbb-border-radius-infinity);
241
+ background-color: var(--sbb-calendar-control-view-change-background);
242
+ color: var(--sbb-calendar-control-view-change-color);
243
+ transition-duration: var(--sbb-calendar-cell-transition-duration);
244
+ transition-timing-function: var(--sbb-calendar-cell-transition-easing-function);
245
+ transition-property: background-color, padding-block-end;
246
+ }
247
+ .sbb-calendar__controls-change-date:disabled {
248
+ --sbb-calendar-control-view-change-background: var(--sbb-color-milk);
249
+ --sbb-calendar-control-view-change-background: light-dark(
250
+ var(--sbb-color-milk),
251
+ var(--sbb-color-anthracite)
252
+ );
253
+ --sbb-calendar-control-view-change-color: var(--sbb-color-granite);
254
+ --sbb-calendar-control-view-change-color: light-dark(
255
+ var(--sbb-color-granite),
256
+ var(--sbb-color-aluminium)
257
+ );
258
+ cursor: unset;
259
+ }
260
+ .sbb-calendar__controls-change-date:focus-visible {
261
+ outline-offset: var(--sbb-focus-outline-offset);
262
+ outline: var(--sbb-focus-outline-color) var(--sbb-focus-outline-style, solid) var(--sbb-focus-outline-width);
263
+ outline-offset: var(--sbb-spacing-fixed-1x);
264
+ }
265
+ @media (any-hover: hover) {
266
+ .sbb-calendar__controls-change-date:not(:active, :disabled):hover {
267
+ padding-block-end: var(--sbb-calendar-hover-shift);
268
+ }
269
+ }
270
+ .sbb-calendar__controls-change-date:not(:disabled):active {
271
+ --sbb-calendar-control-view-change-background: var(--sbb-background-color-3);
272
+ }
273
+
274
+ .sbb-calendar__table-month-view,
275
+ .sbb-calendar__table-year-view {
276
+ --sbb-calendar-table-column-spaces: 6;
277
+ }
278
+
279
+ .sbb-calendar__table-overflow-break {
280
+ contain: layout;
281
+ }
282
+
283
+ .sbb-calendar__table-container {
284
+ display: flex;
285
+ gap: var(--sbb-calendar-tables-gap);
286
+ margin-inline: var(--sbb-calendar-margin);
287
+ --sbb-calendar-min-width: calc(7 * var(--sbb-calendar-cell-size));
288
+ --sbb-calendar-overflow: calc(100% - var(--sbb-calendar-min-width));
289
+ --sbb-calendar-start-offset: min(
290
+ 0px,
291
+ -1 * (var(--sbb-calendar-overflow) / var(--sbb-calendar-table-column-spaces))
292
+ );
293
+ --sbb-calendar-margin: var(--sbb-calendar-start-offset);
294
+ }
295
+ :host(:is(:state(wide),[state--wide])) .sbb-calendar__table-container {
296
+ --sbb-calendar-min-width: calc(
297
+ 2 * 7 * var(--sbb-calendar-cell-size) + var(--sbb-calendar-tables-gap)
298
+ );
299
+ --sbb-calendar-margin: calc(0.5 * var(--sbb-calendar-start-offset));
300
+ }
301
+ :host([orientation=horizontal][week-numbers]) .sbb-calendar__table-container {
302
+ --sbb-calendar-min-width: calc(8 * var(--sbb-calendar-cell-size));
303
+ }
304
+ :host([orientation=horizontal][week-numbers]:is(:state(wide),[state--wide])) .sbb-calendar__table-container {
305
+ --sbb-calendar-min-width: calc(
306
+ 2 * 8 * var(--sbb-calendar-cell-size) + var(--sbb-calendar-tables-gap)
307
+ );
308
+ }
309
+ :host([orientation=vertical]) .sbb-calendar__table-container {
310
+ min-width: var(--sbb-calendar-min-width);
311
+ --sbb-calendar-start-offset: 0;
312
+ }
313
+ :host([orientation=vertical]:is(:state(wide),[state--wide])) .sbb-calendar__table-container {
314
+ --sbb-calendar-min-width: calc(
315
+ 13 * var(--sbb-calendar-cell-size) + var(--sbb-calendar-tables-gap)
316
+ );
317
+ }
318
+
319
+ .sbb-calendar__table {
320
+ width: 100%;
321
+ border-collapse: collapse;
322
+ height: max-content;
323
+ animation-name: show;
324
+ animation-duration: var(--sbb-calendar-table-animation-duration);
325
+ }
326
+ .sbb-calendar__table.sbb-calendar__table-hide {
327
+ --sbb-calendar-cell-transition-duration: 0ms;
328
+ animation-name: hide;
329
+ animation-duration: var(--sbb-calendar-table-animation-duration);
330
+ }
331
+ :host(:not(:is(:state(wide),[state--wide]))) .sbb-calendar__table {
332
+ min-width: 100%;
333
+ }
334
+
335
+ .sbb-calendar__table-header {
336
+ text-align: center;
337
+ }
338
+
339
+ .sbb-calendar__table-body {
340
+ text-align: center;
341
+ }
342
+
343
+ .sbb-calendar__table-header-cell,
344
+ .sbb-calendar__table-header-cell-vertical {
345
+ --sbb-text-font-size: var(--sbb-text-font-size-xs);
346
+ font-family: var(--sbb-typo-font-family);
347
+ font-weight: normal;
348
+ line-height: var(--sbb-typo-line-height-text);
349
+ letter-spacing: var(--sbb-typo-letter-spacing-text);
350
+ font-size: var(--sbb-text-font-size);
351
+ width: var(--sbb-calendar-cell-size);
352
+ color: var(--sbb-calendar-header-color);
353
+ padding: 0;
354
+ }
355
+ :host(:not([multiple])) .sbb-calendar__table-header-cell,
356
+ :host(:not([multiple])) .sbb-calendar__table-header-cell-vertical {
357
+ height: var(--sbb-calendar-cell-size);
358
+ }
359
+
360
+ .sbb-calendar__table-header-cell-vertical {
361
+ padding-block-end: var(--sbb-calendar-cell-padding-block-end);
362
+ }
363
+
364
+ .sbb-calendar__table-data {
365
+ position: relative;
366
+ padding: 0;
367
+ text-align: center;
368
+ }
369
+
370
+ .sbb-calendar__header-cell,
371
+ .sbb-calendar__cell {
372
+ appearance: none;
373
+ box-sizing: border-box;
374
+ margin: 0;
375
+ outline: none;
376
+ border: none;
377
+ border-radius: 0;
378
+ padding: 0;
379
+ background: none;
380
+ -webkit-tap-highlight-color: transparent;
381
+ color: inherit;
382
+ --sbb-text-font-size: var(--sbb-text-font-size-s);
383
+ font-family: var(--sbb-typo-font-family);
384
+ font-weight: normal;
385
+ line-height: var(--sbb-typo-line-height-text);
386
+ letter-spacing: var(--sbb-typo-letter-spacing-text);
387
+ font-size: var(--sbb-text-font-size);
388
+ height: var(--sbb-calendar-cell-size);
389
+ color: var(--sbb-calendar-cell-color);
390
+ cursor: var(--sbb-cursor-pointer);
391
+ position: relative;
392
+ z-index: 0;
393
+ }
394
+ .sbb-calendar__header-cell::before,
395
+ .sbb-calendar__cell::before {
396
+ content: "";
397
+ position: absolute;
398
+ inset: var(--sbb-calendar-cell-padding);
399
+ background-color: var(--sbb-calendar-cell-background-color);
400
+ border-radius: var(--sbb-calendar-cell-border-radius);
401
+ z-index: -1;
402
+ transition-duration: var(--sbb-calendar-cell-transition-duration);
403
+ transition-timing-function: var(--sbb-calendar-cell-transition-easing-function);
404
+ transition-property: background-color;
405
+ }
406
+ @media (any-hover: hover) {
407
+ .sbb-calendar__header-cell:not(:active, :disabled):hover,
408
+ .sbb-calendar__cell:not(:active, :disabled):hover {
409
+ --sbb-calendar-cell-background-color: var(--sbb-calendar-cell-background-color-hover);
410
+ padding-block-end: var(--sbb-calendar-hover-shift);
411
+ }
412
+ }
413
+ @media (any-hover: hover) and (forced-colors: active) {
414
+ .sbb-calendar__header-cell:not(:active, :disabled):hover::before,
415
+ .sbb-calendar__cell:not(:active, :disabled):hover::before {
416
+ outline-offset: var(--sbb-focus-outline-offset);
417
+ outline: var(--sbb-focus-outline-color) var(--sbb-focus-outline-style, solid) var(--sbb-focus-outline-width);
418
+ --sbb-focus-outline-style: initial;
419
+ }
420
+ }
421
+ .sbb-calendar__header-cell:disabled,
422
+ .sbb-calendar__cell:disabled {
423
+ --sbb-calendar-cell-color: var(--sbb-calendar-cell-disabled-color);
424
+ cursor: unset;
425
+ }
426
+ .sbb-calendar__header-cell:focus-visible::before,
427
+ .sbb-calendar__cell:focus-visible::before {
428
+ outline-offset: var(--sbb-focus-outline-offset);
429
+ outline: var(--sbb-focus-outline-color) var(--sbb-focus-outline-style, solid) var(--sbb-focus-outline-width);
430
+ }
431
+ .sbb-calendar__header-cell:not(:disabled):active,
432
+ .sbb-calendar__cell:not(:disabled):active {
433
+ --sbb-calendar-cell-background-color: var(--sbb-calendar-cell-background-color-active);
434
+ }
435
+ @media (forced-colors: active) {
436
+ .sbb-calendar__header-cell:not(:disabled):active::before,
437
+ .sbb-calendar__cell:not(:disabled):active::before {
438
+ outline-offset: var(--sbb-focus-outline-offset);
439
+ outline: var(--sbb-focus-outline-color) var(--sbb-focus-outline-style, solid) var(--sbb-focus-outline-width);
440
+ --sbb-focus-outline-style: initial;
441
+ }
442
+ }
443
+
444
+ .sbb-calendar__header-cell {
445
+ --sbb-text-font-size: var(--sbb-text-font-size-xs);
446
+ font-family: var(--sbb-typo-font-family);
447
+ font-weight: normal;
448
+ line-height: var(--sbb-typo-line-height-text);
449
+ letter-spacing: var(--sbb-typo-letter-spacing-text);
450
+ font-size: var(--sbb-text-font-size);
451
+ color: var(--sbb-calendar-header-color);
452
+ }
453
+
454
+ .sbb-calendar__cell {
455
+ height: var(--sbb-calendar-cell-year-month-height);
456
+ width: var(--sbb-calendar-cell-year-month-width);
457
+ }
458
+
459
+ .sbb-calendar__weekday {
460
+ width: var(--sbb-calendar-cell-size);
461
+ }
462
+
463
+ .sbb-calendar__day-cell {
464
+ padding-block-end: var(--sbb-calendar-cell-padding-block-end);
465
+ }
466
+
467
+ .sbb-calendar__crossed-out::after {
468
+ content: "";
469
+ height: var(--sbb-calendar-cell-disabled-height);
470
+ width: var(--sbb-calendar-cell-disabled-width);
471
+ position: absolute;
472
+ background-color: var(--sbb-calendar-cell-disabled-color);
473
+ top: 50%;
474
+ left: 50%;
475
+ transform: translate(-50%, -50%) rotate(-45deg);
476
+ }
477
+
478
+ .sbb-calendar__cell-current {
479
+ --sbb-text-font-size: var(--sbb-text-font-size-s);
480
+ font-family: var(--sbb-typo-font-family);
481
+ font-weight: normal;
482
+ line-height: var(--sbb-typo-line-height-text);
483
+ letter-spacing: var(--sbb-typo-letter-spacing-text);
484
+ font-size: var(--sbb-text-font-size);
485
+ font-weight: bold;
486
+ }
487
+
488
+ .sbb-calendar__selected::before {
489
+ border: var(--sbb-calendar-cell-selected-border);
490
+ border-radius: var(--sbb-calendar-cell-border-radius);
491
+ }
492
+
493
+ sbb-calendar-day {
494
+ --sbb-calendar-day-flex-direction: row;
495
+ --sbb-calendar-day-justify-content: center;
496
+ --sbb-calendar-day-align-items: center;
497
+ --sbb-calendar-day-extra-display: none;
498
+ --sbb-calendar-day-crossed-out-top: 50%;
499
+ --sbb-calendar-day-crossed-out-translate: translate(-50%, -50%) rotate(-45deg);
500
+ --sbb-calendar-day-height: 2.75rem;
501
+ --sbb-calendar-day-value-height: unset;
502
+ }
503
+
504
+ @keyframes show {
505
+ from {
506
+ opacity: 0;
507
+ transform: translateY(var(--sbb-calendar-table-animation-shift));
508
+ }
509
+ to {
510
+ opacity: 1;
511
+ transform: translateY(0%);
512
+ }
513
+ }
514
+ @keyframes hide {
515
+ from {
516
+ opacity: 1;
517
+ transform: translateY(0%);
518
+ }
519
+ to {
520
+ opacity: 0;
521
+ transform: translateY(var(--sbb-calendar-table-animation-shift));
522
+ }
523
+ }`;
524
+ class SbbMonthChangeEvent extends Event {
525
+ get range() {
526
+ return this._range;
527
+ }
528
+ constructor(range) {
529
+ super("monthchange", { bubbles: true, composed: true });
530
+ this._range = Object.freeze(range || []);
531
+ }
532
+ }
533
+ let SbbCalendarElement = (() => {
534
+ var _wide_accessor_storage, _view_accessor_storage, _min_accessor_storage, _max_accessor_storage, _multiple_accessor_storage, __selected_accessor_storage, _dateFilter_accessor_storage, _orientation_accessor_storage, _weekNumbers_accessor_storage, __activeDate_accessor_storage, __calendarView_accessor_storage, __initialized_accessor_storage, _a;
535
+ let _classDecorators = [customElement("sbb-calendar")];
536
+ let _classDescriptor;
537
+ let _classExtraInitializers = [];
538
+ let _classThis;
539
+ let _classSuper = SbbHydrationMixin(SbbElementInternalsMixin(LitElement));
540
+ let _instanceExtraInitializers = [];
541
+ let _wide_decorators;
542
+ let _wide_initializers = [];
543
+ let _wide_extraInitializers = [];
544
+ let _view_decorators;
545
+ let _view_initializers = [];
546
+ let _view_extraInitializers = [];
547
+ let _min_decorators;
548
+ let _min_initializers = [];
549
+ let _min_extraInitializers = [];
550
+ let _max_decorators;
551
+ let _max_initializers = [];
552
+ let _max_extraInitializers = [];
553
+ let _multiple_decorators;
554
+ let _multiple_initializers = [];
555
+ let _multiple_extraInitializers = [];
556
+ let _set_selected_decorators;
557
+ let __selected_decorators;
558
+ let __selected_initializers = [];
559
+ let __selected_extraInitializers = [];
560
+ let _dateFilter_decorators;
561
+ let _dateFilter_initializers = [];
562
+ let _dateFilter_extraInitializers = [];
563
+ let _orientation_decorators;
564
+ let _orientation_initializers = [];
565
+ let _orientation_extraInitializers = [];
566
+ let _weekNumbers_decorators;
567
+ let _weekNumbers_initializers = [];
568
+ let _weekNumbers_extraInitializers = [];
569
+ let __activeDate_decorators;
570
+ let __activeDate_initializers = [];
571
+ let __activeDate_extraInitializers = [];
572
+ let _set__wide_decorators;
573
+ let __calendarView_decorators;
574
+ let __calendarView_initializers = [];
575
+ let __calendarView_extraInitializers = [];
576
+ let __initialized_decorators;
577
+ let __initialized_initializers = [];
578
+ let __initialized_extraInitializers = [];
579
+ _a = class extends _classSuper {
580
+ constructor() {
581
+ super();
582
+ __privateAdd(this, _wide_accessor_storage);
583
+ __privateAdd(this, _view_accessor_storage);
584
+ __privateAdd(this, _min_accessor_storage);
585
+ __privateAdd(this, _max_accessor_storage);
586
+ __privateAdd(this, _multiple_accessor_storage);
587
+ __privateAdd(this, __selected_accessor_storage);
588
+ __privateAdd(this, _dateFilter_accessor_storage);
589
+ __privateAdd(this, _orientation_accessor_storage);
590
+ __privateAdd(this, _weekNumbers_accessor_storage);
591
+ __privateAdd(this, __activeDate_accessor_storage);
592
+ __privateAdd(this, __calendarView_accessor_storage);
593
+ __privateAdd(this, __initialized_accessor_storage);
594
+ __privateSet(this, _wide_accessor_storage, (__runInitializers(this, _instanceExtraInitializers), __runInitializers(this, _wide_initializers, false)));
595
+ __privateSet(this, _view_accessor_storage, (__runInitializers(this, _wide_extraInitializers), __runInitializers(this, _view_initializers, "day")));
596
+ __privateSet(this, _min_accessor_storage, (__runInitializers(this, _view_extraInitializers), __runInitializers(this, _min_initializers, null)));
597
+ __privateSet(this, _max_accessor_storage, (__runInitializers(this, _min_extraInitializers), __runInitializers(this, _max_initializers, null)));
598
+ __privateSet(this, _multiple_accessor_storage, (__runInitializers(this, _max_extraInitializers), __runInitializers(this, _multiple_initializers, false)));
599
+ __privateSet(this, __selected_accessor_storage, (__runInitializers(this, _multiple_extraInitializers), __runInitializers(this, __selected_initializers, null)));
600
+ __privateSet(this, _dateFilter_accessor_storage, (__runInitializers(this, __selected_extraInitializers), __runInitializers(this, _dateFilter_initializers, null)));
601
+ __privateSet(this, _orientation_accessor_storage, (__runInitializers(this, _dateFilter_extraInitializers), __runInitializers(this, _orientation_initializers, "horizontal")));
602
+ __privateSet(this, _weekNumbers_accessor_storage, (__runInitializers(this, _orientation_extraInitializers), __runInitializers(this, _weekNumbers_initializers, false)));
603
+ this._dateAdapter = (__runInitializers(this, _weekNumbers_extraInitializers), readConfig().datetime?.dateAdapter ?? defaultDateAdapter);
604
+ __privateSet(this, __activeDate_accessor_storage, __runInitializers(this, __activeDate_initializers, this._dateAdapter.today()));
605
+ this._wideInternal = (__runInitializers(this, __activeDate_extraInitializers), false);
606
+ __privateSet(this, __calendarView_accessor_storage, __runInitializers(this, __calendarView_initializers, "day"));
607
+ this._nextCalendarView = (__runInitializers(this, __calendarView_extraInitializers), "day");
608
+ this._keyboardNavigationDayViewParameters = {
609
+ firstDayInView: null,
610
+ lastDayInView: null,
611
+ firstMonthOffset: 0,
612
+ firstMonthLength: 0,
613
+ secondMonthOffset: 0
614
+ };
615
+ this._weeks = [];
616
+ this._monthNames = this._dateAdapter.getMonthNames("long");
617
+ this._enhancedVariant = false;
618
+ this._resetFocus = false;
619
+ this._containingFocus = false;
620
+ __privateSet(this, __initialized_accessor_storage, __runInitializers(this, __initialized_initializers, false));
621
+ this._language = (__runInitializers(this, __initialized_extraInitializers), new SbbLanguageController(this).withHandler(() => {
622
+ this._monthNames = this._dateAdapter.getMonthNames("long");
623
+ this._createMonthRows();
624
+ }));
625
+ this._mediaMatcher = new SbbMediaMatcherController(this, {
626
+ [SbbMediaQueryBreakpointLargeAndAbove]: () => this._init()
627
+ });
628
+ this._onSlotChange = () => {
629
+ this._enhancedVariant = Array.from(this.children).some((c) => c.localName === "sbb-calendar-day");
630
+ this.toggleState("enhanced", this._enhancedVariant);
631
+ this._setTabIndex();
632
+ };
633
+ this._createMonthRows();
634
+ this._setWeekdays();
635
+ this.addEventListener("focusin", () => this._containingFocus = true);
636
+ this.addEventListener("focusout", () => this._containingFocus = false);
637
+ this.addEventListener("click", (e) => {
638
+ const day = e.target.closest("sbb-calendar-day");
639
+ if (day) {
640
+ this._selectDate(day.value);
641
+ }
642
+ });
643
+ this.addEventListener("keydown", (e) => {
644
+ if (e.target.localName === "sbb-calendar-day") {
645
+ this._handleKeyboardEvent(e, this._mapDateToDay(e.target.value));
646
+ }
647
+ });
648
+ }
649
+ /** If set to true, two months are displayed */
650
+ get wide() {
651
+ return __privateGet(this, _wide_accessor_storage);
652
+ }
653
+ set wide(value) {
654
+ __privateSet(this, _wide_accessor_storage, value);
655
+ }
656
+ /** The initial view of the calendar which should be displayed on opening. */
657
+ get view() {
658
+ return __privateGet(this, _view_accessor_storage);
659
+ }
660
+ set view(value) {
661
+ __privateSet(this, _view_accessor_storage, value);
662
+ }
663
+ /**
664
+ * The minimum valid date. Accepts a date object or null.
665
+ * Accepts an ISO8601 formatted string (e.g. 2024-12-24) as attribute.
666
+ */
667
+ get min() {
668
+ return __privateGet(this, _min_accessor_storage);
669
+ }
670
+ set min(value) {
671
+ __privateSet(this, _min_accessor_storage, value);
672
+ }
673
+ /**
674
+ * The maximum valid date. Accepts a date object or null.
675
+ * Accepts an ISO8601 formatted string (e.g. 2024-12-24) as attribute.
676
+ */
677
+ get max() {
678
+ return __privateGet(this, _max_accessor_storage);
679
+ }
680
+ set max(value) {
681
+ __privateSet(this, _max_accessor_storage, value);
682
+ }
683
+ /** Whether the calendar allows for multiple date selection. */
684
+ get multiple() {
685
+ return __privateGet(this, _multiple_accessor_storage);
686
+ }
687
+ set multiple(value) {
688
+ __privateSet(this, _multiple_accessor_storage, value);
689
+ }
690
+ /**
691
+ * The selected date: accepts a date object, or, if `multiple`, an array of dates.
692
+ */
693
+ set selected(value) {
694
+ if (Array.isArray(value)) {
695
+ 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));
696
+ } else {
697
+ const selectedDate = this._dateAdapter.getValidDateOrNull(this._dateAdapter.deserialize(value));
698
+ if (!!selectedDate && (!this._isDayInRange(this._dateAdapter.toIso8601(selectedDate)) || this._dateFilter(selectedDate))) {
699
+ this._selected = selectedDate;
700
+ } else {
701
+ this._selected = null;
702
+ }
703
+ }
704
+ }
705
+ get selected() {
706
+ return this._selected;
707
+ }
708
+ get _selected() {
709
+ return __privateGet(this, __selected_accessor_storage);
710
+ }
711
+ set _selected(value) {
712
+ __privateSet(this, __selected_accessor_storage, value);
713
+ }
714
+ /** A function used to filter out dates. */
715
+ get dateFilter() {
716
+ return __privateGet(this, _dateFilter_accessor_storage);
717
+ }
718
+ set dateFilter(value) {
719
+ __privateSet(this, _dateFilter_accessor_storage, value);
720
+ }
721
+ /** The orientation of days in the calendar. */
722
+ get orientation() {
723
+ return __privateGet(this, _orientation_accessor_storage);
724
+ }
725
+ set orientation(value) {
726
+ __privateSet(this, _orientation_accessor_storage, value);
727
+ }
728
+ /** Whether it has to display the week numbers in addition to week days. */
729
+ get weekNumbers() {
730
+ return __privateGet(this, _weekNumbers_accessor_storage);
731
+ }
732
+ set weekNumbers(value) {
733
+ __privateSet(this, _weekNumbers_accessor_storage, value);
734
+ }
735
+ /** The currently active date. */
736
+ get _activeDate() {
737
+ return __privateGet(this, __activeDate_accessor_storage);
738
+ }
739
+ set _activeDate(value) {
740
+ __privateSet(this, __activeDate_accessor_storage, value);
741
+ }
742
+ /** The current wide property considering property value and breakpoints. From zero to small `wide` has always to be false. */
743
+ set _wide(wide) {
744
+ this.toggleState("wide", wide);
745
+ this._wideInternal = wide;
746
+ }
747
+ get _wide() {
748
+ return this._wideInternal;
749
+ }
750
+ get _calendarView() {
751
+ return __privateGet(this, __calendarView_accessor_storage);
752
+ }
753
+ set _calendarView(value) {
754
+ __privateSet(this, __calendarView_accessor_storage, value);
755
+ }
756
+ /** A list of buttons corresponding to days, months or years depending on the view. */
757
+ get _cells() {
758
+ return Array.from((this._calendarView === "day" ? this._getRootForQuerySelector()?.querySelectorAll("sbb-calendar-day") : this.shadowRoot?.querySelectorAll(".sbb-calendar__cell")) ?? []);
759
+ }
760
+ get _initialized() {
761
+ return __privateGet(this, __initialized_accessor_storage);
762
+ }
763
+ set _initialized(value) {
764
+ __privateSet(this, __initialized_accessor_storage, value);
765
+ }
766
+ _getRootForQuerySelector() {
767
+ return this._enhancedVariant ? this : this.shadowRoot;
768
+ }
769
+ _dateFilter(date) {
770
+ return this.dateFilter?.(date) ?? true;
771
+ }
772
+ /** Resets the active month according to the new state of the calendar. */
773
+ resetPosition() {
774
+ this._resetCalendarView();
775
+ this._init();
776
+ }
777
+ connectedCallback() {
778
+ super.connectedCallback();
779
+ this.resetPosition();
780
+ this.shadowRoot?.addEventListener("slotchange", this._onSlotChange, { capture: true });
781
+ }
782
+ disconnectedCallback() {
783
+ super.disconnectedCallback();
784
+ this.shadowRoot?.removeEventListener("slotchange", this._onSlotChange, { capture: true });
785
+ }
786
+ /** @internal */
787
+ focus() {
788
+ this._resetFocus = true;
789
+ this._focusCell();
790
+ }
791
+ willUpdate(changedProperties) {
792
+ super.willUpdate(changedProperties);
793
+ if (!this._initialized) {
794
+ return;
795
+ }
796
+ if (changedProperties.has("wide") || changedProperties.has("orientation")) {
797
+ this.resetPosition();
798
+ }
799
+ if (changedProperties.has("view")) {
800
+ this._setChosenYear();
801
+ this._chosenMonth = void 0;
802
+ this._nextCalendarView = this._calendarView = this.view;
803
+ }
804
+ }
805
+ updated(changedProperties) {
806
+ super.updated(changedProperties);
807
+ this._setTabIndex();
808
+ this._focusCell();
809
+ }
810
+ /**
811
+ * The `_selected` state should be adapted when the `multiple` property changes:
812
+ * - if it changes to true, the '_selected' is set to an array;
813
+ * - if it changes to false, the first available option is set as 'value' otherwise it's set to null.
814
+ */
815
+ _onMultipleChanged(isMultiple) {
816
+ if (isMultiple && !Array.isArray(this._selected)) {
817
+ this._selected = this._selected ? [this._selected] : [];
818
+ }
819
+ if (!isMultiple && Array.isArray(this._selected)) {
820
+ this._selected = this._selected.length ? this._selected[0] : null;
821
+ }
822
+ }
823
+ /** Initializes the component. */
824
+ _init(activeDate) {
825
+ if (isServer) {
826
+ return;
827
+ } else if (this.hydrationRequired) {
828
+ this.hydrationComplete.then(() => this._init());
829
+ return;
830
+ }
831
+ if (activeDate) {
832
+ this._assignActiveDate(activeDate);
833
+ }
834
+ this._wide = (this._mediaMatcher.matches(SbbMediaQueryBreakpointLargeAndAbove) ?? false) && this.wide;
835
+ this._weeks = this._createWeekRows(this._activeDate);
836
+ this._years = this._createYearRows();
837
+ this._weekNumbers = this._createWeekNumbers(this._activeDate);
838
+ this._nextMonthWeeks = [[]];
839
+ this._nextMonthYears = [[]];
840
+ if (this._wide) {
841
+ const nextMonthDate = this._dateAdapter.addCalendarMonths(this._activeDate, 1);
842
+ this._nextMonthWeeks = this._createWeekRows(nextMonthDate, true);
843
+ this._nextMonthYears = this._createYearRows(YEARS_PER_PAGE);
844
+ this._nextMonthWeekNumbers = this._createWeekNumbers(nextMonthDate);
845
+ }
846
+ this._initialized = true;
847
+ }
848
+ /** Focuses on a day cell prioritizing the selected day, the current day, and lastly, the first selectable day. */
849
+ _focusCell() {
850
+ if (this._resetFocus) {
851
+ this._getFirstFocusable()?.focus();
852
+ this._resetFocus = false;
853
+ }
854
+ }
855
+ /** Creates the array of weekdays. */
856
+ _setWeekdays() {
857
+ const narrowWeekdays = this._dateAdapter.getDayOfWeekNames("narrow");
858
+ const longWeekdays = this._dateAdapter.getDayOfWeekNames("long");
859
+ const weekdays = longWeekdays.map((long, i) => ({
860
+ long,
861
+ narrow: narrowWeekdays[i]
862
+ }));
863
+ const firstDayOfWeek = this._dateAdapter.getFirstDayOfWeek();
864
+ this._weekdays = weekdays.slice(firstDayOfWeek).concat(weekdays.slice(0, firstDayOfWeek));
865
+ }
866
+ /**
867
+ * Given a date, it returns the week numbers for the month the date belongs to.
868
+ * TODO: check if date-fns can be replaced with custom logic.
869
+ *
870
+ * Since the calculation is not simple (see https://en.wikipedia.org/wiki/Week#Numbering),
871
+ * the date-fns library has been used this way:
872
+ * the first and the last day of the month are calculated and then passed to the `eachWeekOfInterval` function,
873
+ * which returns an array containing the starting day of every ISO week of the month,
874
+ * considering Monday as the first day.
875
+ * Then, this array is mapped via the `getWeek` function, which returns the ISO week number for that date.
876
+ */
877
+ _createWeekNumbers(date) {
878
+ return eachWeekOfInterval({ start: startOfMonth(date), end: endOfMonth(date) }, { weekStartsOn: 1 }).map((firstDayOfWeek) => getWeek(firstDayOfWeek, { weekStartsOn: 1, firstWeekContainsDate: 4 }));
879
+ }
880
+ /** Creates the rows along the horizontal direction and sets the parameters used in keyboard navigation. */
881
+ _createWeekRows(value, isSecondMonthInView = false) {
882
+ const daysInMonth = this._dateAdapter.getNumDaysInMonth(value);
883
+ const weekOffset = this._dateAdapter.getFirstWeekOffset(value);
884
+ if (!isSecondMonthInView) {
885
+ this._keyboardNavigationDayViewParameters.firstMonthLength = daysInMonth;
886
+ this._keyboardNavigationDayViewParameters.firstMonthOffset = weekOffset;
887
+ this._keyboardNavigationDayViewParameters.firstDayInView = this._dateAdapter.toIso8601(this._dateAdapter.createDate(this._dateAdapter.getYear(value), this._dateAdapter.getMonth(value), 1));
888
+ this._keyboardNavigationDayViewParameters.lastDayInView = this._dateAdapter.toIso8601(this._dateAdapter.createDate(this._dateAdapter.getYear(value), this._dateAdapter.getMonth(value), daysInMonth));
889
+ } else {
890
+ this._keyboardNavigationDayViewParameters.secondMonthOffset = weekOffset;
891
+ this._keyboardNavigationDayViewParameters.lastDayInView = this._dateAdapter.toIso8601(this._dateAdapter.createDate(this._dateAdapter.getYear(value), this._dateAdapter.getMonth(value), daysInMonth));
892
+ }
893
+ return this.orientation === "horizontal" ? this._createWeekRowsHorizontal(value, daysInMonth, weekOffset) : this._createWeekRowsVertical(value, daysInMonth, weekOffset);
894
+ }
895
+ /**
896
+ * Creates the rows for each week in orientation='horizontal'.
897
+ *
898
+ * Iterates through the days of the months, creates a Day object for each and pushes it into and array.
899
+ * Each seven days (considering the offset at the beginning of the month) restarts from an empty array.
900
+ *
901
+ * The result is a matrix in which every row is a week (or part of it, considering offset).
902
+ */
903
+ _createWeekRowsHorizontal(value, daysInMonth, weekOffset) {
904
+ const weeks = [[]];
905
+ for (let i = 0, cell = weekOffset; i < daysInMonth; i++, cell++) {
906
+ if (cell === DAYS_PER_ROW) {
907
+ weeks.push([]);
908
+ cell = 0;
909
+ }
910
+ const date = this._dateAdapter.createDate(this._dateAdapter.getYear(value), this._dateAdapter.getMonth(value), i + 1);
911
+ weeks[weeks.length - 1].push(this._mapDateToDay(date));
912
+ }
913
+ return weeks;
914
+ }
915
+ /**
916
+ * Creates the rows for each week in orientation='vertical'.
917
+ *
918
+ * Creates a matrix with seven empty rows.
919
+ * Iterates through the days of the months, creates a Day object for each
920
+ * and pushes it into the correct array considering the offset at the beginning of the month.
921
+ * Each seven days (including offset) restarts from the first.
922
+ *
923
+ * The result is a matrix in which every row is a set of weekdays, so:
924
+ * - row 0: all the Mondays;
925
+ * - row 1: all the Tuesdays;
926
+ * - ...
927
+ * - row 7: all the Sundays.
928
+ */
929
+ _createWeekRowsVertical(value, daysInMonth, weekOffset) {
930
+ const weeks = Array.from({ length: DAYS_PER_ROW }, () => []);
931
+ for (let i = 0, cell = weekOffset; i < daysInMonth; i++, cell++) {
932
+ if (cell === DAYS_PER_ROW) {
933
+ cell = 0;
934
+ }
935
+ const date = this._dateAdapter.createDate(this._dateAdapter.getYear(value), this._dateAdapter.getMonth(value), i + 1);
936
+ weeks[cell].push(this._mapDateToDay(date));
937
+ }
938
+ return weeks;
939
+ }
940
+ _mapDateToDay(date) {
941
+ const isoDate = this._dateAdapter.toIso8601(date);
942
+ return {
943
+ value: isoDate,
944
+ dateValue: date,
945
+ dayValue: String(this._dateAdapter.getDate(date)),
946
+ monthValue: String(this._dateAdapter.getMonth(date)),
947
+ yearValue: String(this._dateAdapter.getYear(date)),
948
+ weekValue: getWeek(isoDate, { weekStartsOn: 1, firstWeekContainsDate: 4 }),
949
+ weekDayValue: this._dateAdapter.getDayOfWeek(date)
950
+ };
951
+ }
952
+ /** Force the conversion to ISO8601 formatted string. */
953
+ _mapValueToISODate(value) {
954
+ return typeof value === "string" ? value : this._dateAdapter.toIso8601(value);
955
+ }
956
+ /** Creates the rows for the month selection view. */
957
+ _createMonthRows() {
958
+ const shortNames = this._dateAdapter.getMonthNames("short");
959
+ const months = new Array(12).fill(null).map((_, i) => ({
960
+ value: shortNames[i],
961
+ longValue: this._monthNames[i],
962
+ monthValue: i + 1
963
+ }));
964
+ const rows = 12 / MONTHS_PER_ROW;
965
+ const monthArray = [];
966
+ for (let i = 0; i < rows; i++) {
967
+ monthArray.push(months.slice(MONTHS_PER_ROW * i, MONTHS_PER_ROW * (i + 1)));
968
+ }
969
+ this._months = monthArray;
970
+ }
971
+ /** Creates the rows for the year selection view. */
972
+ _createYearRows(offset = 0) {
973
+ const startValueYearView = this._getStartValueYearView();
974
+ const allYears = new Array(YEARS_PER_PAGE).fill(0).map((_, i) => startValueYearView + offset + i);
975
+ const rows = YEARS_PER_PAGE / YEARS_PER_ROW;
976
+ const yearArray = [];
977
+ for (let i = 0; i < rows; i++) {
978
+ yearArray.push(allYears.slice(YEARS_PER_ROW * i, YEARS_PER_ROW * (i + 1)));
979
+ }
980
+ return yearArray;
981
+ }
982
+ /**
983
+ * Calculates the first year that will be shown in the year selection panel.
984
+ * If `minDate` and `maxDate` are both null, the starting year is calculated as
985
+ * the multiple of YEARS_PER_PAGE closest to and less than activeDate,
986
+ * e.g., with `YEARS_PER_PAGE` = 24 and `activeDate` = 2020, the function will return 2016 (24 * 83),
987
+ * while with `activeDate` = 2000, the function will return 1992 (24 * 82).
988
+ * If `minDate` is not null, it returns the corresponding year; if `maxDate` is not null,
989
+ * it returns the corresponding year minus `YEARS_PER_PAGE`, so that the `maxDate` is the last rendered year.
990
+ * If both are not null, `maxDate` has priority over `minDate`.
991
+ */
992
+ _getStartValueYearView() {
993
+ let startingYear = 0;
994
+ if (this.max) {
995
+ startingYear = this._dateAdapter.getYear(this.max) - YEARS_PER_PAGE + 1;
996
+ } else if (this.min) {
997
+ startingYear = this._dateAdapter.getYear(this.min);
998
+ }
999
+ const activeYear = this._dateAdapter.getYear(this._activeDate);
1000
+ return activeYear - ((activeYear - startingYear) % YEARS_PER_PAGE + YEARS_PER_PAGE) % YEARS_PER_PAGE;
1001
+ }
1002
+ /** Checks if date is within the min-max range. */
1003
+ _isDayInRange(dateString) {
1004
+ if (!this.min && !this.max) {
1005
+ return true;
1006
+ }
1007
+ const date = this._dateAdapter.deserialize(dateString);
1008
+ return this._dateAdapter.sameDate(date, this._dateAdapter.clampDate(date, this.min, this.max));
1009
+ }
1010
+ /** Checks if date is within the min-max range in month view. */
1011
+ _isMonthInRange(month, year) {
1012
+ if (!this.min && !this.max) {
1013
+ return true;
1014
+ }
1015
+ const isBeforeMin = this._dateAdapter.isValid(this.min) && (year < this._dateAdapter.getYear(this.min) || year === this._dateAdapter.getYear(this.min) && month < this._dateAdapter.getMonth(this.min));
1016
+ const isAfterMax = this._dateAdapter.isValid(this.max) && (year > this._dateAdapter.getYear(this.max) || year === this._dateAdapter.getYear(this.max) && month > this._dateAdapter.getMonth(this.max));
1017
+ return !(isBeforeMin || isAfterMax);
1018
+ }
1019
+ /** Checks if date is within the min-max range in year view. */
1020
+ _isYearInRange(year) {
1021
+ if (!this.min && !this.max) {
1022
+ return true;
1023
+ }
1024
+ const isBeforeMin = this._dateAdapter.isValid(this.min) && this._dateAdapter.getYear(this.min) > year;
1025
+ const isAfterMax = this._dateAdapter.isValid(this.max) && this._dateAdapter.getYear(this.max) < year;
1026
+ return !(isBeforeMin || isAfterMax);
1027
+ }
1028
+ // Implementation adapted from https://github.com/angular/components/blob/main/src/material/datepicker/year-view.ts#L366
1029
+ _isMonthFilteredOut(month, year) {
1030
+ if (!this.dateFilter) {
1031
+ return true;
1032
+ }
1033
+ const firstOfMonth = this._dateAdapter.createDate(year, month, 1);
1034
+ for (let date = firstOfMonth; this._dateAdapter.getMonth(date) == month; date = this._dateAdapter.addCalendarDays(date, 1)) {
1035
+ if (this.dateFilter(date)) {
1036
+ return true;
1037
+ }
1038
+ }
1039
+ return false;
1040
+ }
1041
+ // Implementation adapted from https://github.com/angular/components/blob/main/src/material/datepicker/multi-year-view.ts#L351
1042
+ _isYearFilteredOut(year) {
1043
+ if (!this.dateFilter) {
1044
+ return true;
1045
+ }
1046
+ const firstOfYear = this._dateAdapter.createDate(year, 1, 1);
1047
+ for (let date = firstOfYear; this._dateAdapter.getYear(date) == year; date = this._dateAdapter.addCalendarDays(date, 1)) {
1048
+ if (this.dateFilter(date)) {
1049
+ return true;
1050
+ }
1051
+ }
1052
+ return false;
1053
+ }
1054
+ /** Emits the selected date and sets it internally. */
1055
+ _selectDate(day) {
1056
+ this._chosenMonth = void 0;
1057
+ this._setChosenYear();
1058
+ if (this.multiple) {
1059
+ if (this._selected && this._selected.length > 0) {
1060
+ const indexOfSelectedDay = this._selected.findIndex((sel) => this._dateAdapter.compareDate(sel, day) === 0);
1061
+ if (indexOfSelectedDay !== -1) {
1062
+ this._selected = this._selected.filter((_, i) => i !== indexOfSelectedDay);
1063
+ } else {
1064
+ this._selected = [...this._selected, day];
1065
+ }
1066
+ } else {
1067
+ this._selected = [day];
1068
+ }
1069
+ this._emitDateSelectedEvent(this._selected.map((e) => this._dateAdapter.deserialize(e)));
1070
+ } else {
1071
+ if (!this._selected || this._dateAdapter.compareDate(this._selected, day) !== 0) {
1072
+ this._selected = day;
1073
+ this._emitDateSelectedEvent(this._dateAdapter.deserialize(day));
1074
+ }
1075
+ }
1076
+ }
1077
+ /**
1078
+ * Handle multiple dates selection via weekNumber / weekDay buttons:
1079
+ * - if Cmd or Ctrl are pressed, add the new date to the current ones;
1080
+ * - if not,
1081
+ * - if the new dates are the same of the current ones, it means that the same button has been clicked twice, so do nothing;
1082
+ * - if not, the selected dates are the new ones.
1083
+ */
1084
+ _selectMultipleDates(days) {
1085
+ const enabledDays = this._cells.filter((e) => !e.disabled).map((e) => this._mapValueToISODate(e.value));
1086
+ const daysToAdd = days.map((e) => e.value).filter((isoDate) => enabledDays.includes(isoDate));
1087
+ const daysToAddSet = new Set(daysToAdd);
1088
+ const selectedSet = new Set(this._selected.map((s) => this._dateAdapter.toIso8601(s)));
1089
+ const selStrings = this._updateSelectedWithMultipleDates(daysToAdd, daysToAddSet, selectedSet);
1090
+ this._selected = selStrings.map((s) => this._dateAdapter.deserialize(s));
1091
+ this._emitDateSelectedEvent(this._selected.map((e) => this._dateAdapter.deserialize(e)));
1092
+ }
1093
+ /**
1094
+ * Emits the dateselected event given the detail (as T or T[] based on the value of the multiple flag).
1095
+ */
1096
+ _emitDateSelectedEvent(detail) {
1097
+ this.dispatchEvent(new CustomEvent("dateselected", {
1098
+ detail,
1099
+ composed: true,
1100
+ bubbles: true
1101
+ }));
1102
+ }
1103
+ _emitMonthChange() {
1104
+ const monthchange = (this.wide ? [...this._weeks, ...this._nextMonthWeeks] : this._weeks).flat().sort((a, b) => a.value.localeCompare(b.value));
1105
+ this.dispatchEvent(new SbbMonthChangeEvent(monthchange));
1106
+ }
1107
+ /**
1108
+ * In case of multiple selection, newly added days must be added to the existing ones, without duplication.
1109
+ * If the days to add are exactly the same as the selected ones, the set must be emptied.
1110
+ */
1111
+ _updateSelectedWithMultipleDates(daysToAdd, daysToAddSet, selectedSet) {
1112
+ if (daysToAdd.every((day) => selectedSet.has(day))) {
1113
+ daysToAddSet.forEach((day) => selectedSet.delete(day));
1114
+ } else {
1115
+ daysToAddSet.forEach((day) => selectedSet.add(day));
1116
+ }
1117
+ return Array.from(selectedSet);
1118
+ }
1119
+ _setChosenYear() {
1120
+ if (this.view === "month") {
1121
+ let selectedDate;
1122
+ if (this.multiple) {
1123
+ selectedDate = this.selected.at(-1);
1124
+ } else {
1125
+ selectedDate = this.selected;
1126
+ }
1127
+ this._chosenYear = this._dateAdapter.getYear(selectedDate ?? this._dateAdapter.today());
1128
+ } else {
1129
+ this._chosenYear = void 0;
1130
+ }
1131
+ }
1132
+ _assignActiveDate(date) {
1133
+ if (this.min && this._dateAdapter.compareDate(this.min, date) > 0) {
1134
+ this._activeDate = this.min;
1135
+ return;
1136
+ }
1137
+ if (this.max && this._dateAdapter.compareDate(this.max, date) < 0) {
1138
+ this._activeDate = this.max;
1139
+ return;
1140
+ }
1141
+ this._activeDate = date;
1142
+ }
1143
+ /** Goes to the month identified by the shift. */
1144
+ _goToDifferentMonth(months) {
1145
+ this._init(this._dateAdapter.addCalendarMonths(this._activeDate, months));
1146
+ this._emitMonthChange();
1147
+ }
1148
+ _goToDifferentYear(years) {
1149
+ this._chosenYear += years;
1150
+ this._activeDate = this._dateAdapter.createDate(this._chosenYear, this._dateAdapter.getMonth(this._activeDate), this._dateAdapter.getDate(this._activeDate));
1151
+ this._init();
1152
+ }
1153
+ _goToDifferentYearRange(years) {
1154
+ this._init(this._dateAdapter.addCalendarYears(this._activeDate, years));
1155
+ }
1156
+ _prevDisabled(prevDate) {
1157
+ if (!this.min) {
1158
+ return false;
1159
+ }
1160
+ return this._dateAdapter.compareDate(prevDate, this.min) < 0;
1161
+ }
1162
+ _nextDisabled(nextDate) {
1163
+ if (!this.max) {
1164
+ return false;
1165
+ }
1166
+ return this._dateAdapter.compareDate(nextDate, this.max) > 0;
1167
+ }
1168
+ /** Checks if the "previous month" button should be disabled. */
1169
+ _previousMonthDisabled() {
1170
+ const prevMonth = this._dateAdapter.addCalendarDays(this._activeDate, this._dateAdapter.getDate(this._activeDate) * -1);
1171
+ return this._prevDisabled(prevMonth);
1172
+ }
1173
+ /** Checks if the "next month" button should be disabled. */
1174
+ _nextMonthDisabled() {
1175
+ let nextMonth = this._dateAdapter.addCalendarMonths(this._activeDate, this._wide ? 2 : 1);
1176
+ nextMonth = this._dateAdapter.createDate(this._dateAdapter.getYear(nextMonth), this._dateAdapter.getMonth(nextMonth), 1);
1177
+ return this._nextDisabled(nextMonth);
1178
+ }
1179
+ /** Checks if the "previous year" button should be disabled. */
1180
+ _previousYearDisabled() {
1181
+ const prevYear = this._dateAdapter.createDate(this._dateAdapter.getYear(this._activeDate) - 1, 12, 31);
1182
+ return this._prevDisabled(prevYear);
1183
+ }
1184
+ /** Checks if the "next year" button should be disabled. */
1185
+ _nextYearDisabled() {
1186
+ const nextYear = this._dateAdapter.createDate(this._dateAdapter.getYear(this._activeDate) + (this._wide ? 2 : 1), 1, 1);
1187
+ return this._nextDisabled(nextYear);
1188
+ }
1189
+ /** Checks if the "previous year" button should be disabled in year view. */
1190
+ _previousYearRangeDisabled() {
1191
+ const prevYear = this._dateAdapter.createDate(this._years[0][0] - 1, 12, 31);
1192
+ return this._prevDisabled(prevYear);
1193
+ }
1194
+ /** Checks if the "next year" button should be disabled in year view. */
1195
+ _nextYearRangeDisabled() {
1196
+ const years = this._wide ? this._nextMonthYears : this._years;
1197
+ const lastYearRange = years[years.length - 1];
1198
+ const lastYear = lastYearRange[lastYearRange.length - 1];
1199
+ const nextYear = this._dateAdapter.createDate(lastYear + 1, 1, 1);
1200
+ return this._nextDisabled(nextYear);
1201
+ }
1202
+ _handleTableBlur(eventTarget) {
1203
+ if (eventTarget?.localName !== "sbb-calendar-day") {
1204
+ this._setTabIndex();
1205
+ }
1206
+ }
1207
+ _setTabIndex() {
1208
+ const query = this._calendarView === "day" ? "sbb-calendar-day" : ".sbb-calendar__cell";
1209
+ Array.from(this._getRootForQuerySelector().querySelectorAll(`${query}[tabindex="0"]`) ?? []).forEach((day) => day.tabIndex = -1);
1210
+ const firstFocusable = this._getFirstFocusable();
1211
+ if (firstFocusable) {
1212
+ firstFocusable.tabIndex = 0;
1213
+ }
1214
+ }
1215
+ /** Get the element in the calendar to assign focus. */
1216
+ _getFirstFocusable() {
1217
+ const root = this._getRootForQuerySelector();
1218
+ if (this._calendarView === "day") {
1219
+ const selectedOrCurrent = root.querySelector(":is(:state(selected),[state--selected])") ?? root.querySelector(":is(:state(current),[state--current])");
1220
+ return selectedOrCurrent && !selectedOrCurrent.disabled ? selectedOrCurrent : this._getFirstFocusableDay();
1221
+ } else {
1222
+ const selectedOrCurrent = this.shadowRoot?.querySelector(".sbb-calendar__cell-current");
1223
+ return selectedOrCurrent && !selectedOrCurrent.disabled ? selectedOrCurrent : this.shadowRoot.querySelector(".sbb-calendar__cell:not([disabled])");
1224
+ }
1225
+ }
1226
+ /**
1227
+ * In `day` view in `vertical` orientation,
1228
+ * if the first of the month is not a Monday, it is not the first rendered element in the table,
1229
+ * so `this.shadowRoot!.querySelector('.sbb-calendar__cell:not([disabled])')` will return a wrong value.
1230
+ *
1231
+ * To solve this, the element with the lowest `value` is taken (ISO String are ordered).
1232
+ */
1233
+ _getFirstFocusableDay() {
1234
+ const daysInView = Array.from(this._getRootForQuerySelector().querySelectorAll("sbb-calendar-day:not([disabled])"));
1235
+ if (!daysInView || daysInView.length === 0) {
1236
+ return null;
1237
+ } else {
1238
+ const firstElement = daysInView.map((e) => this._dateAdapter.toIso8601(e.value)).sort()[0];
1239
+ return this._getRootForQuerySelector().querySelector(`sbb-calendar-day[slot="${firstElement}"]`);
1240
+ }
1241
+ }
1242
+ _handleKeyboardEvent(event, day) {
1243
+ if (isArrowKeyOrPageKeysPressed(event)) {
1244
+ event.preventDefault();
1245
+ }
1246
+ const cells = this._cells;
1247
+ const index = cells.findIndex((e) => e === event.target);
1248
+ let nextEl;
1249
+ if (day) {
1250
+ nextEl = this._navigateByKeyboardDayView(event, index, cells, day);
1251
+ } else {
1252
+ nextEl = this._navigateByKeyboard(event, index, cells);
1253
+ }
1254
+ const activeEl = (this._enhancedVariant ? document : this.shadowRoot).activeElement;
1255
+ if (nextEl !== activeEl) {
1256
+ nextEl.tabIndex = 0;
1257
+ nextEl?.focus();
1258
+ activeEl.tabIndex = -1;
1259
+ }
1260
+ }
1261
+ _navigateByKeyboardDayView(evt, index, cells, day) {
1262
+ const arrowsOffset = this.orientation === "horizontal" ? { leftRight: 1, upDown: DAYS_PER_ROW } : { leftRight: DAYS_PER_ROW, upDown: 1 };
1263
+ const offsetForVertical = index < this._keyboardNavigationDayViewParameters.firstMonthLength ? this._keyboardNavigationDayViewParameters.firstMonthOffset : this._keyboardNavigationDayViewParameters.secondMonthOffset;
1264
+ switch (evt.key) {
1265
+ case "ArrowUp":
1266
+ return this._findDayArrows(cells, index, day.dateValue, -arrowsOffset.upDown);
1267
+ case "ArrowDown":
1268
+ return this._findDayArrows(cells, index, day.dateValue, arrowsOffset.upDown);
1269
+ case "ArrowLeft":
1270
+ return this._findDayArrows(cells, index, day.dateValue, -arrowsOffset.leftRight);
1271
+ case "ArrowRight":
1272
+ return this._findDayArrows(cells, index, day.dateValue, arrowsOffset.leftRight);
1273
+ case "PageUp": {
1274
+ if (this.orientation === "horizontal") {
1275
+ const firstOfWeek = +day.dayValue % DAYS_PER_ROW || DAYS_PER_ROW;
1276
+ const delta = firstOfWeek - +day.dayValue;
1277
+ return this._findDayPageUpDown(cells, index, day, delta, arrowsOffset.upDown);
1278
+ } else {
1279
+ const weekNumber = Math.ceil((+day.dayValue + offsetForVertical) / DAYS_PER_ROW);
1280
+ const firstOfWeek = (weekNumber - 1) * DAYS_PER_ROW - offsetForVertical + 1;
1281
+ const delta = firstOfWeek - +day.dayValue;
1282
+ return this._findDayPageUpDown(cells, index, day, delta, arrowsOffset.upDown);
1283
+ }
1284
+ }
1285
+ case "PageDown": {
1286
+ if (this.orientation === "horizontal") {
1287
+ const monthInBounds = +day.monthValue + 1 > 12 ? 1 : +day.monthValue + 1;
1288
+ const yearInBounds = +day.monthValue + 1 > 12 ? +day.yearValue + 1 : +day.yearValue;
1289
+ const firstNextMonth = this._dateAdapter.createDate(yearInBounds, monthInBounds, 1);
1290
+ const lastOfMonth = this._dateAdapter.getDate(this._dateAdapter.addCalendarDays(firstNextMonth, -1));
1291
+ const delta = Math.trunc((lastOfMonth - +day.dayValue) / DAYS_PER_ROW) * DAYS_PER_ROW;
1292
+ return this._findDayPageUpDown(cells, index, day, delta, -arrowsOffset.upDown);
1293
+ } else {
1294
+ const weekNumber = Math.ceil((+day.dayValue + offsetForVertical) / DAYS_PER_ROW);
1295
+ const lastOfWeek = weekNumber * DAYS_PER_ROW - offsetForVertical;
1296
+ const delta = lastOfWeek - +day.dayValue;
1297
+ return this._findDayPageUpDown(cells, index, day, delta, -arrowsOffset.upDown);
1298
+ }
1299
+ }
1300
+ case "Home": {
1301
+ return this._findDayFirst(cells, index, day, 1);
1302
+ }
1303
+ case "End": {
1304
+ const monthInBounds = +day.monthValue + 1 > 12 ? 1 : +day.monthValue + 1;
1305
+ const yearInBounds = +day.monthValue + 1 > 12 ? +day.yearValue + 1 : +day.yearValue;
1306
+ const firstNextMonth = this._dateAdapter.createDate(yearInBounds, monthInBounds, 1);
1307
+ return this._findDayLast(cells, index, firstNextMonth);
1308
+ }
1309
+ default:
1310
+ return cells[index];
1311
+ }
1312
+ }
1313
+ _isDayOutOfView(date) {
1314
+ return date < this._keyboardNavigationDayViewParameters.firstDayInView || date > this._keyboardNavigationDayViewParameters.lastDayInView;
1315
+ }
1316
+ _findDayArrows(cells, index, date, delta) {
1317
+ const newDateValue = this._dateAdapter.toIso8601(this._dateAdapter.addCalendarDays(date, delta));
1318
+ if (this._isDayOutOfView(newDateValue)) {
1319
+ return cells[index];
1320
+ }
1321
+ const nextCell = cells.find((e) => this._mapValueToISODate(e.value) === newDateValue);
1322
+ if (!nextCell || nextCell.disabled) {
1323
+ return this._findDayArrows(cells, index, this._dateAdapter.deserialize(newDateValue), delta);
1324
+ }
1325
+ return nextCell;
1326
+ }
1327
+ _findDayPageUpDown(cells, index, day, delta, deltaIfDisabled) {
1328
+ const newDateValue = this._dateAdapter.toIso8601(this._dateAdapter.addCalendarDays(day.dateValue, delta));
1329
+ if (this._isDayOutOfView(newDateValue)) {
1330
+ return cells[index];
1331
+ }
1332
+ const nextCell = cells.find((e) => this._mapValueToISODate(e.value) === newDateValue);
1333
+ if (!nextCell || nextCell.disabled) {
1334
+ return this._findDayPageUpDown(cells, index, day, delta + deltaIfDisabled, deltaIfDisabled);
1335
+ }
1336
+ return nextCell;
1337
+ }
1338
+ _findDayFirst(cells, index, day, date) {
1339
+ const newDateValue = this._dateAdapter.toIso8601(this._dateAdapter.createDate(+day.yearValue, +day.monthValue, date));
1340
+ if (this._isDayOutOfView(newDateValue)) {
1341
+ return cells[index];
1342
+ }
1343
+ const nextCell = cells.find((e) => this._mapValueToISODate(e.value) === newDateValue);
1344
+ if (!nextCell || nextCell.disabled) {
1345
+ return this._findDayFirst(cells, index, day, date + 1);
1346
+ }
1347
+ return nextCell;
1348
+ }
1349
+ _findDayLast(cells, index, firstNextMonth) {
1350
+ const newDateValue = this._dateAdapter.toIso8601(this._dateAdapter.addCalendarDays(firstNextMonth, -1));
1351
+ if (this._isDayOutOfView(newDateValue)) {
1352
+ return cells[index];
1353
+ }
1354
+ const nextCell = cells.find((e) => this._mapValueToISODate(e.value) === newDateValue);
1355
+ if (!nextCell || nextCell.disabled) {
1356
+ return this._findDayLast(cells, index, this._dateAdapter.deserialize(newDateValue));
1357
+ }
1358
+ return nextCell;
1359
+ }
1360
+ /**
1361
+ * Gets the index of the element to move to, based on a list of elements (which can be potentially disabled),
1362
+ * the keyboard input and the position of the current element in the list.
1363
+ * In the day view, the `day?: Day` parameter is mandatory for calculation,
1364
+ * while in month and year view it's not due to the fixed amount of rendered cells.
1365
+ */
1366
+ _navigateByKeyboard(evt, index, cells) {
1367
+ const { elementIndexForWideMode, offsetForWideMode, lastElementIndexForWideMode, verticalOffset } = this._calculateParametersForKeyboardNavigation(index, this._calendarView === "year");
1368
+ switch (evt.key) {
1369
+ case "ArrowUp":
1370
+ return this._findNext(cells, index, -verticalOffset);
1371
+ case "ArrowDown":
1372
+ return this._findNext(cells, index, verticalOffset);
1373
+ case "ArrowLeft":
1374
+ return this._findNext(cells, index, -1);
1375
+ case "ArrowRight":
1376
+ return this._findNext(cells, index, 1);
1377
+ case "Home":
1378
+ return this._findFirst(cells, offsetForWideMode);
1379
+ case "PageUp":
1380
+ return this._findFirstOnColumn(cells, elementIndexForWideMode, offsetForWideMode, verticalOffset);
1381
+ case "PageDown":
1382
+ return this._findLastOnColumn(cells, index, lastElementIndexForWideMode, verticalOffset);
1383
+ case "End":
1384
+ return this._findLast(cells, lastElementIndexForWideMode - 1);
1385
+ default:
1386
+ return cells[index];
1387
+ }
1388
+ }
1389
+ /**
1390
+ * Calculates the parameters needed in keyboard navigation in year and month view.
1391
+ * @param index The starting element's index in the cell array.
1392
+ * @param isYearView Whether the displayed `view` is the year one.
1393
+ */
1394
+ _calculateParametersForKeyboardNavigation(index, isYearView) {
1395
+ const elementsPerPage = isYearView ? YEARS_PER_PAGE : MONTHS_PER_PAGE;
1396
+ const offset = Math.trunc(index / elementsPerPage) * elementsPerPage;
1397
+ const indexInView = offset === 0 ? index : index - elementsPerPage;
1398
+ return {
1399
+ verticalOffset: isYearView ? YEARS_PER_ROW : MONTHS_PER_ROW,
1400
+ elementIndexForWideMode: indexInView,
1401
+ offsetForWideMode: index - indexInView,
1402
+ lastElementIndexForWideMode: offset === 0 ? elementsPerPage : elementsPerPage * 2
1403
+ };
1404
+ }
1405
+ /**
1406
+ * Gets the next element of the provided array starting from `index` by adding `delta`.
1407
+ * If the found element is disabled, it continues adding `delta` until it finds an enabled one in the array bounds.
1408
+ */
1409
+ _findNext(days, index, delta) {
1410
+ let nextIndex = index + delta;
1411
+ while (nextIndex < days.length && days[nextIndex]?.disabled) {
1412
+ nextIndex += delta;
1413
+ }
1414
+ return days[nextIndex] ?? days[index];
1415
+ }
1416
+ /** Find the first enabled element in the provided array. */
1417
+ _findFirst(days, firstOfCurrentMonth) {
1418
+ return !days[firstOfCurrentMonth].disabled ? days[firstOfCurrentMonth] : this._findNext(days, firstOfCurrentMonth, 1);
1419
+ }
1420
+ /** Find the last enabled element in the provided array. */
1421
+ _findLast(days, lastOfCurrentMonth) {
1422
+ return !days[lastOfCurrentMonth].disabled ? days[lastOfCurrentMonth] : this._findNext(days, lastOfCurrentMonth, -1);
1423
+ }
1424
+ /** Find the first enabled element in the same column of the provided array. */
1425
+ _findFirstOnColumn(days, index, offset, verticalOffset) {
1426
+ const nextIndex = index % verticalOffset + offset;
1427
+ return !days[nextIndex].disabled ? days[nextIndex] : this._findNext(days, nextIndex, verticalOffset);
1428
+ }
1429
+ /** Find the last enabled element in the same column of the provided array. */
1430
+ _findLastOnColumn(days, index, offset, verticalOffset) {
1431
+ const nextIndex = index + Math.trunc((offset - index - 1) / verticalOffset) * verticalOffset;
1432
+ return !days[nextIndex].disabled ? days[nextIndex] : this._findNext(days, nextIndex, -verticalOffset);
1433
+ }
1434
+ _resetCalendarViewAndEmitMonthChange(initTransition = false) {
1435
+ this._resetCalendarView(initTransition);
1436
+ this._emitMonthChange();
1437
+ }
1438
+ _resetCalendarView(initTransition = false) {
1439
+ if (this._containingFocus) {
1440
+ this._resetFocus = true;
1441
+ }
1442
+ this._activeDate = (this.multiple ? this._selected.at(-1) : this._selected) ?? this._dateAdapter.today();
1443
+ this._setChosenYear();
1444
+ this._chosenMonth = void 0;
1445
+ this._init();
1446
+ this._nextCalendarView = this._calendarView = this.view;
1447
+ if (initTransition) {
1448
+ this._startTableTransition();
1449
+ }
1450
+ }
1451
+ /** Render the view for the day selection. */
1452
+ _renderDayView() {
1453
+ const nextMonthActiveDate = this._wide ? this._dateAdapter.addCalendarMonths(this._activeDate, 1) : void 0;
1454
+ return html`
1455
+ <div class="sbb-calendar__controls">
1456
+ ${this._getArrow("left", () => this._goToDifferentMonth(-1), i18nPreviousMonth[this._language.current], this._previousMonthDisabled())}
1457
+ <div class="sbb-calendar__controls-month">
1458
+ ${this._createLabelForDayView(this._activeDate)}
1459
+ ${this._wide ? this._createLabelForDayView(nextMonthActiveDate) : nothing}
1460
+ <sbb-screen-reader-only role="status">
1461
+ ${this._createAriaLabelForDayView(this._activeDate, nextMonthActiveDate)}
1462
+ </sbb-screen-reader-only>
1463
+ </div>
1464
+ ${this._getArrow("right", () => this._goToDifferentMonth(1), i18nNextMonth[this._language.current], this._nextMonthDisabled())}
1465
+ </div>
1466
+ <div class="sbb-calendar__table-overflow-break">
1467
+ <div class="sbb-calendar__table-container sbb-calendar__table-day-view">
1468
+ ${this.orientation === "horizontal" ? html`
1469
+ ${this._createDayTable(this._weeks, this._weekNumbers)}
1470
+ ${this._wide ? this._createDayTable(this._nextMonthWeeks, this._nextMonthWeekNumbers, true) : nothing}
1471
+ ` : html`
1472
+ ${this._createDayTableVertical(this._weeks, this._weekNumbers)}
1473
+ ${this._wide ? this._createDayTableVertical(this._nextMonthWeeks, this._nextMonthWeekNumbers, nextMonthActiveDate) : nothing}
1474
+ `}
1475
+ </div>
1476
+ </div>
1477
+ `;
1478
+ }
1479
+ /** Creates the label with the month for the daily view. */
1480
+ _createLabelForDayView(d) {
1481
+ const monthLabel = `${this._monthNames[this._dateAdapter.getMonth(d) - 1]} ${this._dateAdapter.getYear(d)}`;
1482
+ return html`
1483
+ <button
1484
+ type="button"
1485
+ class="sbb-calendar__date-selection sbb-calendar__controls-change-date"
1486
+ aria-label="${i18nYearMonthSelection[this._language.current]} ${monthLabel}"
1487
+ @click=${() => {
1488
+ this._resetFocus = true;
1489
+ this._nextCalendarView = "year";
1490
+ this._startTableTransition();
1491
+ }}
1492
+ >
1493
+ ${monthLabel}
1494
+ <sbb-icon name="chevron-small-down-small"></sbb-icon>
1495
+ </button>
1496
+ `;
1497
+ }
1498
+ /** Creates the aria-label for the daily view. */
1499
+ _createAriaLabelForDayView(...dates) {
1500
+ let monthLabel = "";
1501
+ for (const d of dates) {
1502
+ if (d) {
1503
+ monthLabel += `${this._monthNames[this._dateAdapter.getMonth(d) - 1]} ${this._dateAdapter.getYear(d)} `;
1504
+ }
1505
+ }
1506
+ return monthLabel;
1507
+ }
1508
+ /** Creates the calendar table for the daily view. */
1509
+ _createDayTable(weeks, weekNumbers, isWideNextMonth = false) {
1510
+ const weeksForSelectMultipleWeekNumbers = (this._wide ? [...this._weeks, ...this._nextMonthWeeks] : isWideNextMonth ? this._nextMonthWeeks : this._weeks).flat();
1511
+ const weeksForSelectMultipleWeekDays = (isWideNextMonth ? this._nextMonthWeeks : this._weeks).flat();
1512
+ return html`
1513
+ <table
1514
+ class="sbb-calendar__table"
1515
+ @focusout=${(event) => this._handleTableBlur(event.relatedTarget)}
1516
+ @animationend=${(e) => this._tableAnimationEnd(e)}
1517
+ >
1518
+ <thead class="sbb-calendar__table-header">
1519
+ <tr>
1520
+ ${this.weekNumbers ? html`<th class="sbb-calendar__table-header-cell"></th>` : nothing}
1521
+ ${this._weekdays.map((day, index) => html`
1522
+ <th class="sbb-calendar__table-header-cell">
1523
+ ${this.multiple ? html`
1524
+ <button
1525
+ class="sbb-calendar__header-cell sbb-calendar__weekday"
1526
+ aria-label=${day.long}
1527
+ @click=${() => {
1528
+ const days = weeksForSelectMultipleWeekDays.filter((day2) => day2.weekDayValue === (index + 1) % 7);
1529
+ this._selectMultipleDates(days);
1530
+ }}
1531
+ >
1532
+ ${day.narrow}
1533
+ </button>
1534
+ ` : html`
1535
+ <sbb-screen-reader-only>${day.long}</sbb-screen-reader-only>
1536
+ <span aria-hidden="true">${day.narrow}</span>
1537
+ `}
1538
+ </th>
1539
+ `)}
1540
+ </tr>
1541
+ </thead>
1542
+ <tbody class="sbb-calendar__table-body">
1543
+ ${weeks.map((week, rowIndex) => {
1544
+ const firstRowOffset = DAYS_PER_ROW - week.length;
1545
+ if (rowIndex === 0 && firstRowOffset) {
1546
+ return html`
1547
+ <tr>
1548
+ ${this.weekNumbers ? html`
1549
+ <td class="sbb-calendar__table-header-cell-vertical">
1550
+ ${this.multiple ? html`
1551
+ <button
1552
+ class="sbb-calendar__header-cell sbb-calendar__weekday"
1553
+ aria-label=${`${i18nCalendarWeekNumber[this._language.current]} ${weekNumbers[0]}`}
1554
+ @click=${() => {
1555
+ const days = weeksForSelectMultipleWeekNumbers.filter((day) => day.weekValue === weekNumbers[0]);
1556
+ this._selectMultipleDates(days);
1557
+ }}
1558
+ >
1559
+ ${weekNumbers[0]}
1560
+ </button>
1561
+ ` : html`
1562
+ <sbb-screen-reader-only
1563
+ >${`${i18nCalendarWeekNumber[this._language.current]} ${weekNumbers[0]}`}</sbb-screen-reader-only
1564
+ >
1565
+ <span aria-hidden="true">${weekNumbers[0]}</span>
1566
+ `}
1567
+ </td>
1568
+ ` : nothing}
1569
+ ${[...Array(firstRowOffset).keys()].map(() => html`<td class="sbb-calendar__table-data"></td>`)}
1570
+ ${this._createDayCells(week)}
1571
+ </tr>
1572
+ `;
1573
+ }
1574
+ return html`
1575
+ <tr>
1576
+ ${this.weekNumbers ? html`
1577
+ <td class="sbb-calendar__table-header-cell-vertical">
1578
+ ${this.multiple ? html`
1579
+ <button
1580
+ class="sbb-calendar__header-cell sbb-calendar__weekday"
1581
+ aria-label=${`${i18nCalendarWeekNumber[this._language.current]} ${weekNumbers[rowIndex]}`}
1582
+ @click=${() => {
1583
+ const days = weeksForSelectMultipleWeekNumbers.filter((day) => day.weekValue === weekNumbers[rowIndex]);
1584
+ this._selectMultipleDates(days);
1585
+ }}
1586
+ >
1587
+ ${weekNumbers[rowIndex]}
1588
+ </button>
1589
+ ` : html`
1590
+ <sbb-screen-reader-only
1591
+ >${`${i18nCalendarWeekNumber[this._language.current]} ${weekNumbers[rowIndex]}`}</sbb-screen-reader-only
1592
+ >
1593
+ <span aria-hidden="true">${weekNumbers[rowIndex]}</span>
1594
+ `}
1595
+ </td>
1596
+ ` : nothing}
1597
+ ${this._createDayCells(week)}
1598
+ </tr>
1599
+ `;
1600
+ })}
1601
+ </tbody>
1602
+ </table>
1603
+ `;
1604
+ }
1605
+ /** Creates the table in orientation='vertical'. */
1606
+ _createDayTableVertical(weeks, weekNumbers, nextMonthActiveDate) {
1607
+ const weekOffset = this._dateAdapter.getFirstWeekOffset(nextMonthActiveDate ?? this._activeDate);
1608
+ const weeksForSelectMultipleWeekNumbers = (this._wide ? [...this._weeks, ...this._nextMonthWeeks] : nextMonthActiveDate ? this._nextMonthWeeks : this._weeks).flat();
1609
+ return html`
1610
+ <table
1611
+ class="sbb-calendar__table"
1612
+ @focusout=${(event) => this._handleTableBlur(event.relatedTarget)}
1613
+ @animationend=${(e) => this._tableAnimationEnd(e)}
1614
+ >
1615
+ ${this.weekNumbers ? html`
1616
+ <thead class="sbb-calendar__table-header">
1617
+ <tr>
1618
+ ${nextMonthActiveDate ? nothing : html`<th class="sbb-calendar__table-data"></th>`}
1619
+ ${weekNumbers.map((weekNumber) => html`
1620
+ <th class="sbb-calendar__table-header-cell">
1621
+ ${this.multiple ? html`
1622
+ <button
1623
+ class="sbb-calendar__header-cell sbb-calendar__weekday"
1624
+ aria-label=${`${i18nCalendarWeekNumber[this._language.current]} ${weekNumber}`}
1625
+ @click=${() => {
1626
+ const days = weeksForSelectMultipleWeekNumbers.filter((day) => day.weekValue === weekNumber);
1627
+ this._selectMultipleDates(days);
1628
+ }}
1629
+ >
1630
+ ${weekNumber}
1631
+ </button>
1632
+ ` : html`
1633
+ <sbb-screen-reader-only
1634
+ >${`${i18nCalendarWeekNumber[this._language.current]} ${weekNumber}`}</sbb-screen-reader-only
1635
+ >
1636
+ <span aria-hidden="true">${weekNumber}</span>
1637
+ `}
1638
+ </th>
1639
+ `)}
1640
+ </tr>
1641
+ </thead>
1642
+ ` : nothing}
1643
+ <tbody class="sbb-calendar__table-body">
1644
+ ${weeks.map((week, rowIndex) => {
1645
+ const weekday = this._weekdays[rowIndex];
1646
+ const selectableDays = this._wide ? [...week, ...this._nextMonthWeeks[rowIndex]] : week;
1647
+ return html`
1648
+ <tr>
1649
+ ${nextMonthActiveDate ? nothing : html`
1650
+ <td class="sbb-calendar__table-header-cell-vertical">
1651
+ ${this.multiple ? html`
1652
+ <button
1653
+ class="sbb-calendar__header-cell sbb-calendar__weekday"
1654
+ aria-label=${weekday.long}
1655
+ @click=${() => this._selectMultipleDates(selectableDays)}
1656
+ >
1657
+ ${weekday.narrow}
1658
+ </button>
1659
+ ` : html`
1660
+ <sbb-screen-reader-only>${weekday.long}</sbb-screen-reader-only>
1661
+ <span aria-hidden="true">${weekday.narrow}</span>
1662
+ `}
1663
+ </td>
1664
+ `}
1665
+ ${rowIndex < weekOffset ? html`<td class="sbb-calendar__table-data"></td>` : nothing}
1666
+ ${this._createDayCells(week)}
1667
+ </tr>
1668
+ `;
1669
+ })}
1670
+ </tbody>
1671
+ </table>
1672
+ `;
1673
+ }
1674
+ /** Creates the cells for the daily view. */
1675
+ _createDayCells(week) {
1676
+ return week.map((day) => {
1677
+ return html`
1678
+ <td class=" sbb-calendar__table-data sbb-calendar__day-cell">
1679
+ <slot name=${day.value}>
1680
+ <sbb-calendar-day
1681
+ slot=${day.value}
1682
+ @click=${() => this._selectDate(day.dateValue)}
1683
+ @keydown=${(evt) => this._handleKeyboardEvent(evt, day)}
1684
+ ></sbb-calendar-day>
1685
+ </slot>
1686
+ </td>
1687
+ `;
1688
+ });
1689
+ }
1690
+ /** Render the view for the month selection. */
1691
+ _renderMonthView() {
1692
+ return html`
1693
+ <div class="sbb-calendar__controls">
1694
+ ${this._getArrow("left", () => this._goToDifferentYear(-1), i18nPreviousYear[this._language.current], this._previousYearDisabled())}
1695
+ <div class="sbb-calendar__controls-month">${this._createLabelForMonthView()}</div>
1696
+ ${this._getArrow("right", () => this._goToDifferentYear(1), i18nNextYear[this._language.current], this._nextYearDisabled())}
1697
+ </div>
1698
+ <div class="sbb-calendar__table-overflow-break">
1699
+ <div class="sbb-calendar__table-container sbb-calendar__table-month-view">
1700
+ ${this._createMonthTable(this._months, this._chosenYear)}
1701
+ ${this._wide ? this._createMonthTable(this._months, this._chosenYear + 1) : nothing}
1702
+ </div>
1703
+ </div>
1704
+ `;
1705
+ }
1706
+ /** Creates the label with the year for the monthly view. */
1707
+ _createLabelForMonthView() {
1708
+ return html` <button
1709
+ type="button"
1710
+ id="sbb-calendar__month-selection"
1711
+ class="sbb-calendar__controls-change-date"
1712
+ aria-label=${`${i18nCalendarDateSelection[this._language.current]} ${this._chosenYear}`}
1713
+ @click=${() => this._resetCalendarViewAndEmitMonthChange(true)}
1714
+ >
1715
+ ${this._chosenYear} ${this._wide ? ` - ${this._chosenYear + 1}` : nothing}
1716
+ <sbb-icon name="chevron-small-up-small"></sbb-icon>
1717
+ </button>
1718
+ <sbb-screen-reader-only role="status"> ${this._chosenYear} </sbb-screen-reader-only>`;
1719
+ }
1720
+ /** Creates the table for the month selection view. */
1721
+ _createMonthTable(months, year) {
1722
+ return html`
1723
+ <table
1724
+ class="sbb-calendar__table"
1725
+ @animationend=${(e) => this._tableAnimationEnd(e)}
1726
+ >
1727
+ ${this._wide ? html`<thead class="sbb-calendar__table-header" aria-hidden="true">
1728
+ <tr>
1729
+ <th class="sbb-calendar__table-header-cell" colspan=${MONTHS_PER_ROW}>${year}</th>
1730
+ </tr>
1731
+ </thead>` : nothing}
1732
+ <tbody class="sbb-calendar__table-body">
1733
+ ${months.map((row) => html`
1734
+ <tr>
1735
+ ${row.map((month) => {
1736
+ let selected;
1737
+ if (this.multiple) {
1738
+ selected = this._selected.find((date) => year === this._dateAdapter.getYear(date) && month.monthValue === this._dateAdapter.getMonth(date)) !== void 0;
1739
+ } else {
1740
+ const selectedMonth = this._selected ? this._dateAdapter.getMonth(this._selected) : void 0;
1741
+ const selectedYear = this._selected ? this._dateAdapter.getYear(this._selected) : void 0;
1742
+ selected = !!this._selected && year === selectedYear && month.monthValue === selectedMonth;
1743
+ }
1744
+ const isOutOfRange = !this._isMonthInRange(month.monthValue, year);
1745
+ const isFilteredOut = !this._isMonthFilteredOut(month.monthValue, year);
1746
+ const isCurrentMonth = year === this._dateAdapter.getYear(this._dateAdapter.today()) && this._dateAdapter.getMonth(this._dateAdapter.today()) === month.monthValue;
1747
+ return html` <td
1748
+ class=${classMap({
1749
+ "sbb-calendar__table-data": true,
1750
+ "sbb-calendar__table-month": true
1751
+ })}
1752
+ >
1753
+ <button
1754
+ class=${classMap({
1755
+ "sbb-calendar__cell": true,
1756
+ "sbb-calendar__cell-current": isCurrentMonth,
1757
+ "sbb-calendar__crossed-out": !isOutOfRange && isFilteredOut,
1758
+ "sbb-calendar__selected": selected
1759
+ })}
1760
+ @click=${() => this._onMonthSelection(month.monthValue, year)}
1761
+ ?disabled=${isOutOfRange || isFilteredOut}
1762
+ aria-label=${`${month.longValue} ${year}`}
1763
+ aria-pressed=${selected}
1764
+ aria-disabled=${String(isOutOfRange || isFilteredOut)}
1765
+ tabindex="-1"
1766
+ data-month=${month.monthValue || nothing}
1767
+ @keydown=${(evt) => this._handleKeyboardEvent(evt)}
1768
+ >
1769
+ ${month.value}
1770
+ </button>
1771
+ </td>`;
1772
+ })}
1773
+ </tr>
1774
+ `)}
1775
+ </tbody>
1776
+ </table>
1777
+ `;
1778
+ }
1779
+ /** Select the month and change the view to day selection. */
1780
+ _onMonthSelection(month, year) {
1781
+ this._chosenMonth = month;
1782
+ this._nextCalendarView = "day";
1783
+ this._init(this._dateAdapter.createDate(year, this._chosenMonth, this._dateAdapter.getDate(this._activeDate)));
1784
+ this._startTableTransition();
1785
+ this._emitMonthChange();
1786
+ }
1787
+ /** Render the view for the year selection. */
1788
+ _renderYearView() {
1789
+ return html`
1790
+ <div class="sbb-calendar__controls">
1791
+ ${this._getArrow("left", () => this._goToDifferentYearRange(-YEARS_PER_PAGE), i18nPreviousYearRange(YEARS_PER_PAGE)[this._language.current], this._previousYearRangeDisabled())}
1792
+ <div class="sbb-calendar__controls-month">${this._createLabelForYearView()}</div>
1793
+ ${this._getArrow("right", () => this._goToDifferentYearRange(YEARS_PER_PAGE), i18nNextYearRange(YEARS_PER_PAGE)[this._language.current], this._nextYearRangeDisabled())}
1794
+ </div>
1795
+ <div class="sbb-calendar__table-overflow-break">
1796
+ <div class="sbb-calendar__table-container sbb-calendar__table-year-view">
1797
+ ${this._createYearTable(this._years)}
1798
+ ${this._wide ? this._createYearTable(this._nextMonthYears, true) : nothing}
1799
+ </div>
1800
+ </div>
1801
+ `;
1802
+ }
1803
+ /** Creates the button arrow for all the views. */
1804
+ _getArrow(direction, click, ariaLabel, disabled) {
1805
+ return html`<sbb-secondary-button
1806
+ size="m"
1807
+ icon-name="chevron-small-${direction}-small"
1808
+ aria-label=${ariaLabel}
1809
+ @click=${click}
1810
+ ?disabled=${disabled}
1811
+ id="sbb-calendar__controls-${direction === "left" ? "previous" : "next"}"
1812
+ ></sbb-secondary-button>`;
1813
+ }
1814
+ /** Creates the label with the year range for the yearly view. */
1815
+ _createLabelForYearView() {
1816
+ const firstYear = this._years.flat()[0];
1817
+ const lastYearArray = (this._wide ? this._nextMonthYears : this._years).flat();
1818
+ const lastYear = lastYearArray[lastYearArray.length - 1];
1819
+ const yearLabel = `${firstYear} - ${lastYear}`;
1820
+ return html`
1821
+ <button
1822
+ type="button"
1823
+ id="sbb-calendar__year-selection"
1824
+ class="sbb-calendar__controls-change-date"
1825
+ aria-label="${i18nCalendarDateSelection[this._language.current]} ${yearLabel}"
1826
+ @click=${() => this._resetCalendarViewAndEmitMonthChange(true)}
1827
+ >
1828
+ ${yearLabel}
1829
+ <sbb-icon name="chevron-small-up-small"></sbb-icon>
1830
+ </button>
1831
+ <sbb-screen-reader-only role="status"> ${yearLabel} </sbb-screen-reader-only>
1832
+ `;
1833
+ }
1834
+ /** Creates the table for the year selection view. */
1835
+ _createYearTable(years, shiftRight = false) {
1836
+ const now = this._dateAdapter.today();
1837
+ return html` <table
1838
+ class="sbb-calendar__table"
1839
+ @animationend=${(e) => this._tableAnimationEnd(e)}
1840
+ >
1841
+ <tbody class="sbb-calendar__table-body">
1842
+ ${years.map((row) => html` <tr>
1843
+ ${row.map((year) => {
1844
+ let selected;
1845
+ if (this.multiple) {
1846
+ selected = this._selected.find((date) => year === this._dateAdapter.getYear(date)) !== void 0;
1847
+ } else {
1848
+ const selectedYear = this._selected ? this._dateAdapter.getYear(this._selected) : void 0;
1849
+ selected = !!this._selected && year === selectedYear;
1850
+ }
1851
+ const isOutOfRange = !this._isYearInRange(year);
1852
+ const isFilteredOut = !this._isYearFilteredOut(year);
1853
+ const isCurrentYear = this._dateAdapter.getYear(now) === year;
1854
+ return html` <td class="sbb-calendar__table-data sbb-calendar__table-year">
1855
+ <button
1856
+ class=${classMap({
1857
+ "sbb-calendar__cell": true,
1858
+ "sbb-calendar__cell-current": isCurrentYear,
1859
+ "sbb-calendar__crossed-out": !isOutOfRange && isFilteredOut,
1860
+ "sbb-calendar__selected": selected
1861
+ })}
1862
+ @click=${() => this._onYearSelection(year, shiftRight)}
1863
+ ?disabled=${isOutOfRange || isFilteredOut}
1864
+ aria-label=${year}
1865
+ aria-pressed=${selected}
1866
+ aria-disabled=${String(isOutOfRange || isFilteredOut)}
1867
+ tabindex="-1"
1868
+ data-year=${year || nothing}
1869
+ @keydown=${(evt) => this._handleKeyboardEvent(evt)}
1870
+ >
1871
+ ${year}
1872
+ </button>
1873
+ </td>`;
1874
+ })}
1875
+ </tr>`)}
1876
+ </tbody>
1877
+ </table>`;
1878
+ }
1879
+ /** Select the year and change the view to month selection. */
1880
+ _onYearSelection(year, rightSide) {
1881
+ this._chosenYear = rightSide ? year - 1 : year;
1882
+ this._nextCalendarView = "month";
1883
+ this._assignActiveDate(this._dateAdapter.createDate(this._chosenYear, this._dateAdapter.getMonth(this._activeDate), this._dateAdapter.getDate(this._activeDate)));
1884
+ this._startTableTransition();
1885
+ }
1886
+ _getView() {
1887
+ if (isServer || this.hydrationRequired) {
1888
+ return html`${nothing}`;
1889
+ }
1890
+ switch (this._calendarView) {
1891
+ case "year":
1892
+ return this._renderYearView();
1893
+ case "month":
1894
+ return this._renderMonthView();
1895
+ case "day":
1896
+ default:
1897
+ return this._renderDayView();
1898
+ }
1899
+ }
1900
+ _tableAnimationEnd(event) {
1901
+ const table = event.target;
1902
+ if (event.animationName === "hide") {
1903
+ table.classList.remove("sbb-calendar__table-hide");
1904
+ if (this._containingFocus) {
1905
+ this._resetFocus = true;
1906
+ }
1907
+ this._calendarView = this._nextCalendarView;
1908
+ } else if (event.animationName === "show") {
1909
+ this.internals.states.delete("transition");
1910
+ }
1911
+ }
1912
+ _startTableTransition() {
1913
+ this.internals.states.add("transition");
1914
+ this.shadowRoot?.querySelectorAll("table")?.forEach((e) => e.classList.toggle("sbb-calendar__table-hide"));
1915
+ }
1916
+ render() {
1917
+ return html`<div class="sbb-calendar__wrapper">${this._getView()}</div>`;
1918
+ }
1919
+ }, _wide_accessor_storage = new WeakMap(), _view_accessor_storage = new WeakMap(), _min_accessor_storage = new WeakMap(), _max_accessor_storage = new WeakMap(), _multiple_accessor_storage = new WeakMap(), __selected_accessor_storage = new WeakMap(), _dateFilter_accessor_storage = new WeakMap(), _orientation_accessor_storage = new WeakMap(), _weekNumbers_accessor_storage = new WeakMap(), __activeDate_accessor_storage = new WeakMap(), __calendarView_accessor_storage = new WeakMap(), __initialized_accessor_storage = new WeakMap(), _classThis = _a, (() => {
1920
+ const _metadata = typeof Symbol === "function" && Symbol.metadata ? Object.create(_classSuper[Symbol.metadata] ?? null) : void 0;
1921
+ _wide_decorators = [forceType(), property({ type: Boolean })];
1922
+ _view_decorators = [property()];
1923
+ _min_decorators = [plainDate(), property()];
1924
+ _max_decorators = [plainDate(), property()];
1925
+ _multiple_decorators = [forceType(), handleDistinctChange((e, newValue) => e._onMultipleChanged(newValue)), property({ type: Boolean })];
1926
+ _set_selected_decorators = [property()];
1927
+ __selected_decorators = [state()];
1928
+ _dateFilter_decorators = [property({ attribute: "date-filter" })];
1929
+ _orientation_decorators = [property({ reflect: true })];
1930
+ _weekNumbers_decorators = [forceType(), property({ attribute: "week-numbers", type: Boolean })];
1931
+ __activeDate_decorators = [state()];
1932
+ _set__wide_decorators = [state()];
1933
+ __calendarView_decorators = [state()];
1934
+ __initialized_decorators = [state()];
1935
+ __esDecorate(_a, null, _wide_decorators, { kind: "accessor", name: "wide", static: false, private: false, access: { has: (obj) => "wide" in obj, get: (obj) => obj.wide, set: (obj, value) => {
1936
+ obj.wide = value;
1937
+ } }, metadata: _metadata }, _wide_initializers, _wide_extraInitializers);
1938
+ __esDecorate(_a, null, _view_decorators, { kind: "accessor", name: "view", static: false, private: false, access: { has: (obj) => "view" in obj, get: (obj) => obj.view, set: (obj, value) => {
1939
+ obj.view = value;
1940
+ } }, metadata: _metadata }, _view_initializers, _view_extraInitializers);
1941
+ __esDecorate(_a, null, _min_decorators, { kind: "accessor", name: "min", static: false, private: false, access: { has: (obj) => "min" in obj, get: (obj) => obj.min, set: (obj, value) => {
1942
+ obj.min = value;
1943
+ } }, metadata: _metadata }, _min_initializers, _min_extraInitializers);
1944
+ __esDecorate(_a, null, _max_decorators, { kind: "accessor", name: "max", static: false, private: false, access: { has: (obj) => "max" in obj, get: (obj) => obj.max, set: (obj, value) => {
1945
+ obj.max = value;
1946
+ } }, metadata: _metadata }, _max_initializers, _max_extraInitializers);
1947
+ __esDecorate(_a, null, _multiple_decorators, { kind: "accessor", name: "multiple", static: false, private: false, access: { has: (obj) => "multiple" in obj, get: (obj) => obj.multiple, set: (obj, value) => {
1948
+ obj.multiple = value;
1949
+ } }, metadata: _metadata }, _multiple_initializers, _multiple_extraInitializers);
1950
+ __esDecorate(_a, null, _set_selected_decorators, { kind: "setter", name: "selected", static: false, private: false, access: { has: (obj) => "selected" in obj, set: (obj, value) => {
1951
+ obj.selected = value;
1952
+ } }, metadata: _metadata }, null, _instanceExtraInitializers);
1953
+ __esDecorate(_a, null, __selected_decorators, { kind: "accessor", name: "_selected", static: false, private: false, access: { has: (obj) => "_selected" in obj, get: (obj) => obj._selected, set: (obj, value) => {
1954
+ obj._selected = value;
1955
+ } }, metadata: _metadata }, __selected_initializers, __selected_extraInitializers);
1956
+ __esDecorate(_a, null, _dateFilter_decorators, { kind: "accessor", name: "dateFilter", static: false, private: false, access: { has: (obj) => "dateFilter" in obj, get: (obj) => obj.dateFilter, set: (obj, value) => {
1957
+ obj.dateFilter = value;
1958
+ } }, metadata: _metadata }, _dateFilter_initializers, _dateFilter_extraInitializers);
1959
+ __esDecorate(_a, null, _orientation_decorators, { kind: "accessor", name: "orientation", static: false, private: false, access: { has: (obj) => "orientation" in obj, get: (obj) => obj.orientation, set: (obj, value) => {
1960
+ obj.orientation = value;
1961
+ } }, metadata: _metadata }, _orientation_initializers, _orientation_extraInitializers);
1962
+ __esDecorate(_a, null, _weekNumbers_decorators, { kind: "accessor", name: "weekNumbers", static: false, private: false, access: { has: (obj) => "weekNumbers" in obj, get: (obj) => obj.weekNumbers, set: (obj, value) => {
1963
+ obj.weekNumbers = value;
1964
+ } }, metadata: _metadata }, _weekNumbers_initializers, _weekNumbers_extraInitializers);
1965
+ __esDecorate(_a, null, __activeDate_decorators, { kind: "accessor", name: "_activeDate", static: false, private: false, access: { has: (obj) => "_activeDate" in obj, get: (obj) => obj._activeDate, set: (obj, value) => {
1966
+ obj._activeDate = value;
1967
+ } }, metadata: _metadata }, __activeDate_initializers, __activeDate_extraInitializers);
1968
+ __esDecorate(_a, null, _set__wide_decorators, { kind: "setter", name: "_wide", static: false, private: false, access: { has: (obj) => "_wide" in obj, set: (obj, value) => {
1969
+ obj._wide = value;
1970
+ } }, metadata: _metadata }, null, _instanceExtraInitializers);
1971
+ __esDecorate(_a, null, __calendarView_decorators, { kind: "accessor", name: "_calendarView", static: false, private: false, access: { has: (obj) => "_calendarView" in obj, get: (obj) => obj._calendarView, set: (obj, value) => {
1972
+ obj._calendarView = value;
1973
+ } }, metadata: _metadata }, __calendarView_initializers, __calendarView_extraInitializers);
1974
+ __esDecorate(_a, null, __initialized_decorators, { kind: "accessor", name: "_initialized", static: false, private: false, access: { has: (obj) => "_initialized" in obj, get: (obj) => obj._initialized, set: (obj, value) => {
1975
+ obj._initialized = value;
1976
+ } }, metadata: _metadata }, __initialized_initializers, __initialized_extraInitializers);
1977
+ __esDecorate(null, _classDescriptor = { value: _classThis }, _classDecorators, { kind: "class", name: _classThis.name, metadata: _metadata }, null, _classExtraInitializers);
1978
+ _classThis = _classDescriptor.value;
1979
+ if (_metadata) Object.defineProperty(_classThis, Symbol.metadata, { enumerable: true, configurable: true, writable: true, value: _metadata });
1980
+ })(), _a.styles = [boxSizingStyles, style], _a.events = {
1981
+ dateselected: "dateselected",
1982
+ monthchange: "monthchange"
1983
+ }, __runInitializers(_classThis, _classExtraInitializers), _a;
1984
+ return _classThis;
1985
+ })();
1986
+ export {
1987
+ SbbCalendarElement,
1988
+ SbbMonthChangeEvent
1989
+ };
1990
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY2FsZW5kYXIuY29tcG9uZW50LmpzIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi9ub2RlX21vZHVsZXMvZGF0ZS1mbnMvY29uc3RhbnRzLmpzIiwiLi4vLi4vLi4vLi4vLi4vbm9kZV9tb2R1bGVzL2RhdGUtZm5zL2NvbnN0cnVjdEZyb20uanMiLCIuLi8uLi8uLi8uLi8uLi9ub2RlX21vZHVsZXMvZGF0ZS1mbnMvdG9EYXRlLmpzIiwiLi4vLi4vLi4vLi4vLi4vbm9kZV9tb2R1bGVzL2RhdGUtZm5zL2FkZERheXMuanMiLCIuLi8uLi8uLi8uLi8uLi9ub2RlX21vZHVsZXMvZGF0ZS1mbnMvX2xpYi9kZWZhdWx0T3B0aW9ucy5qcyIsIi4uLy4uLy4uLy4uLy4uL25vZGVfbW9kdWxlcy9kYXRlLWZucy9zdGFydE9mV2Vlay5qcyIsIi4uLy4uLy4uLy4uLy4uL25vZGVfbW9kdWxlcy9kYXRlLWZucy9fbGliL25vcm1hbGl6ZURhdGVzLmpzIiwiLi4vLi4vLi4vLi4vLi4vbm9kZV9tb2R1bGVzL2RhdGUtZm5zL2FkZFdlZWtzLmpzIiwiLi4vLi4vLi4vLi4vLi4vbm9kZV9tb2R1bGVzL2RhdGUtZm5zL2VuZE9mTW9udGguanMiLCIuLi8uLi8uLi8uLi8uLi9ub2RlX21vZHVsZXMvZGF0ZS1mbnMvX2xpYi9ub3JtYWxpemVJbnRlcnZhbC5qcyIsIi4uLy4uLy4uLy4uLy4uL25vZGVfbW9kdWxlcy9kYXRlLWZucy9lYWNoV2Vla09mSW50ZXJ2YWwuanMiLCIuLi8uLi8uLi8uLi8uLi9ub2RlX21vZHVsZXMvZGF0ZS1mbnMvc3RhcnRPZk1vbnRoLmpzIiwiLi4vLi4vLi4vLi4vLi4vbm9kZV9tb2R1bGVzL2RhdGUtZm5zL2dldFdlZWtZZWFyLmpzIiwiLi4vLi4vLi4vLi4vLi4vbm9kZV9tb2R1bGVzL2RhdGUtZm5zL3N0YXJ0T2ZXZWVrWWVhci5qcyIsIi4uLy4uLy4uLy4uLy4uL25vZGVfbW9kdWxlcy9kYXRlLWZucy9nZXRXZWVrLmpzIiwiLi4vLi4vLi4vLi4vLi4vc3JjL2VsZW1lbnRzL2NhbGVuZGFyL2NhbGVuZGFyL2NhbGVuZGFyLmNvbXBvbmVudC50cyJdLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIEBtb2R1bGUgY29uc3RhbnRzXG4gKiBAc3VtbWFyeSBVc2VmdWwgY29uc3RhbnRzXG4gKiBAZGVzY3JpcHRpb25cbiAqIENvbGxlY3Rpb24gb2YgdXNlZnVsIGRhdGUgY29uc3RhbnRzLlxuICpcbiAqIFRoZSBjb25zdGFudHMgY291bGQgYmUgaW1wb3J0ZWQgZnJvbSBgZGF0ZS1mbnMvY29uc3RhbnRzYDpcbiAqXG4gKiBgYGB0c1xuICogaW1wb3J0IHsgbWF4VGltZSwgbWluVGltZSB9IGZyb20gXCIuL2NvbnN0YW50cy9kYXRlLWZucy9jb25zdGFudHNcIjtcbiAqXG4gKiBmdW5jdGlvbiBpc0FsbG93ZWRUaW1lKHRpbWUpIHtcbiAqICAgcmV0dXJuIHRpbWUgPD0gbWF4VGltZSAmJiB0aW1lID49IG1pblRpbWU7XG4gKiB9XG4gKiBgYGBcbiAqL1xuXG4vKipcbiAqIEBjb25zdGFudFxuICogQG5hbWUgZGF5c0luV2Vla1xuICogQHN1bW1hcnkgRGF5cyBpbiAxIHdlZWsuXG4gKi9cbmV4cG9ydCBjb25zdCBkYXlzSW5XZWVrID0gNztcblxuLyoqXG4gKiBAY29uc3RhbnRcbiAqIEBuYW1lIGRheXNJblllYXJcbiAqIEBzdW1tYXJ5IERheXMgaW4gMSB5ZWFyLlxuICpcbiAqIEBkZXNjcmlwdGlvblxuICogSG93IG1hbnkgZGF5cyBpbiBhIHllYXIuXG4gKlxuICogT25lIHllYXJzIGVxdWFscyAzNjUuMjQyNSBkYXlzIGFjY29yZGluZyB0byB0aGUgZm9ybXVsYTpcbiAqXG4gKiA+IExlYXAgeWVhciBvY2N1cnMgZXZlcnkgNCB5ZWFycywgZXhjZXB0IGZvciB5ZWFycyB0aGF0IGFyZSBkaXZpc2libGUgYnkgMTAwIGFuZCBub3QgZGl2aXNpYmxlIGJ5IDQwMC5cbiAqID4gMSBtZWFuIHllYXIgPSAoMzY1KzEvNC0xLzEwMCsxLzQwMCkgZGF5cyA9IDM2NS4yNDI1IGRheXNcbiAqL1xuZXhwb3J0IGNvbnN0IGRheXNJblllYXIgPSAzNjUuMjQyNTtcblxuLyoqXG4gKiBAY29uc3RhbnRcbiAqIEBuYW1lIG1heFRpbWVcbiAqIEBzdW1tYXJ5IE1heGltdW0gYWxsb3dlZCB0aW1lLlxuICpcbiAqIEBleGFtcGxlXG4gKiBpbXBvcnQgeyBtYXhUaW1lIH0gZnJvbSBcIi4vY29uc3RhbnRzL2RhdGUtZm5zL2NvbnN0YW50c1wiO1xuICpcbiAqIGNvbnN0IGlzVmFsaWQgPSA4NjQwMDAwMDAwMDAwMDAxIDw9IG1heFRpbWU7XG4gKiAvLz0+IGZhbHNlXG4gKlxuICogbmV3IERhdGUoODY0MDAwMDAwMDAwMDAwMSk7XG4gKiAvLz0+IEludmFsaWQgRGF0ZVxuICovXG5leHBvcnQgY29uc3QgbWF4VGltZSA9IE1hdGgucG93KDEwLCA4KSAqIDI0ICogNjAgKiA2MCAqIDEwMDA7XG5cbi8qKlxuICogQGNvbnN0YW50XG4gKiBAbmFtZSBtaW5UaW1lXG4gKiBAc3VtbWFyeSBNaW5pbXVtIGFsbG93ZWQgdGltZS5cbiAqXG4gKiBAZXhhbXBsZVxuICogaW1wb3J0IHsgbWluVGltZSB9IGZyb20gXCIuL2NvbnN0YW50cy9kYXRlLWZucy9jb25zdGFudHNcIjtcbiAqXG4gKiBjb25zdCBpc1ZhbGlkID0gLTg2NDAwMDAwMDAwMDAwMDEgPj0gbWluVGltZTtcbiAqIC8vPT4gZmFsc2VcbiAqXG4gKiBuZXcgRGF0ZSgtODY0MDAwMDAwMDAwMDAwMSlcbiAqIC8vPT4gSW52YWxpZCBEYXRlXG4gKi9cbmV4cG9ydCBjb25zdCBtaW5UaW1lID0gLW1heFRpbWU7XG5cbi8qKlxuICogQGNvbnN0YW50XG4gKiBAbmFtZSBtaWxsaXNlY29uZHNJbldlZWtcbiAqIEBzdW1tYXJ5IE1pbGxpc2Vjb25kcyBpbiAxIHdlZWsuXG4gKi9cbmV4cG9ydCBjb25zdCBtaWxsaXNlY29uZHNJbldlZWsgPSA2MDQ4MDAwMDA7XG5cbi8qKlxuICogQGNvbnN0YW50XG4gKiBAbmFtZSBtaWxsaXNlY29uZHNJbkRheVxuICogQHN1bW1hcnkgTWlsbGlzZWNvbmRzIGluIDEgZGF5LlxuICovXG5leHBvcnQgY29uc3QgbWlsbGlzZWNvbmRzSW5EYXkgPSA4NjQwMDAwMDtcblxuLyoqXG4gKiBAY29uc3RhbnRcbiAqIEBuYW1lIG1pbGxpc2Vjb25kc0luTWludXRlXG4gKiBAc3VtbWFyeSBNaWxsaXNlY29uZHMgaW4gMSBtaW51dGVcbiAqL1xuZXhwb3J0IGNvbnN0IG1pbGxpc2Vjb25kc0luTWludXRlID0gNjAwMDA7XG5cbi8qKlxuICogQGNvbnN0YW50XG4gKiBAbmFtZSBtaWxsaXNlY29uZHNJbkhvdXJcbiAqIEBzdW1tYXJ5IE1pbGxpc2Vjb25kcyBpbiAxIGhvdXJcbiAqL1xuZXhwb3J0IGNvbnN0IG1pbGxpc2Vjb25kc0luSG91ciA9IDM2MDAwMDA7XG5cbi8qKlxuICogQGNvbnN0YW50XG4gKiBAbmFtZSBtaWxsaXNlY29uZHNJblNlY29uZFxuICogQHN1bW1hcnkgTWlsbGlzZWNvbmRzIGluIDEgc2Vjb25kXG4gKi9cbmV4cG9ydCBjb25zdCBtaWxsaXNlY29uZHNJblNlY29uZCA9IDEwMDA7XG5cbi8qKlxuICogQGNvbnN0YW50XG4gKiBAbmFtZSBtaW51dGVzSW5ZZWFyXG4gKiBAc3VtbWFyeSBNaW51dGVzIGluIDEgeWVhci5cbiAqL1xuZXhwb3J0IGNvbnN0IG1pbnV0ZXNJblllYXIgPSA1MjU2MDA7XG5cbi8qKlxuICogQGNvbnN0YW50XG4gKiBAbmFtZSBtaW51dGVzSW5Nb250aFxuICogQHN1bW1hcnkgTWludXRlcyBpbiAxIG1vbnRoLlxuICovXG5leHBvcnQgY29uc3QgbWludXRlc0luTW9udGggPSA0MzIwMDtcblxuLyoqXG4gKiBAY29uc3RhbnRcbiAqIEBuYW1lIG1pbnV0ZXNJbkRheVxuICogQHN1bW1hcnkgTWludXRlcyBpbiAxIGRheS5cbiAqL1xuZXhwb3J0IGNvbnN0IG1pbnV0ZXNJbkRheSA9IDE0NDA7XG5cbi8qKlxuICogQGNvbnN0YW50XG4gKiBAbmFtZSBtaW51dGVzSW5Ib3VyXG4gKiBAc3VtbWFyeSBNaW51dGVzIGluIDEgaG91ci5cbiAqL1xuZXhwb3J0IGNvbnN0IG1pbnV0ZXNJbkhvdXIgPSA2MDtcblxuLyoqXG4gKiBAY29uc3RhbnRcbiAqIEBuYW1lIG1vbnRoc0luUXVhcnRlclxuICogQHN1bW1hcnkgTW9udGhzIGluIDEgcXVhcnRlci5cbiAqL1xuZXhwb3J0IGNvbnN0IG1vbnRoc0luUXVhcnRlciA9IDM7XG5cbi8qKlxuICogQGNvbnN0YW50XG4gKiBAbmFtZSBtb250aHNJblllYXJcbiAqIEBzdW1tYXJ5IE1vbnRocyBpbiAxIHllYXIuXG4gKi9cbmV4cG9ydCBjb25zdCBtb250aHNJblllYXIgPSAxMjtcblxuLyoqXG4gKiBAY29uc3RhbnRcbiAqIEBuYW1lIHF1YXJ0ZXJzSW5ZZWFyXG4gKiBAc3VtbWFyeSBRdWFydGVycyBpbiAxIHllYXJcbiAqL1xuZXhwb3J0IGNvbnN0IHF1YXJ0ZXJzSW5ZZWFyID0gNDtcblxuLyoqXG4gKiBAY29uc3RhbnRcbiAqIEBuYW1lIHNlY29uZHNJbkhvdXJcbiAqIEBzdW1tYXJ5IFNlY29uZHMgaW4gMSBob3VyLlxuICovXG5leHBvcnQgY29uc3Qgc2Vjb25kc0luSG91ciA9IDM2MDA7XG5cbi8qKlxuICogQGNvbnN0YW50XG4gKiBAbmFtZSBzZWNvbmRzSW5NaW51dGVcbiAqIEBzdW1tYXJ5IFNlY29uZHMgaW4gMSBtaW51dGUuXG4gKi9cbmV4cG9ydCBjb25zdCBzZWNvbmRzSW5NaW51dGUgPSA2MDtcblxuLyoqXG4gKiBAY29uc3RhbnRcbiAqIEBuYW1lIHNlY29uZHNJbkRheVxuICogQHN1bW1hcnkgU2Vjb25kcyBpbiAxIGRheS5cbiAqL1xuZXhwb3J0IGNvbnN0IHNlY29uZHNJbkRheSA9IHNlY29uZHNJbkhvdXIgKiAyNDtcblxuLyoqXG4gKiBAY29uc3RhbnRcbiAqIEBuYW1lIHNlY29uZHNJbldlZWtcbiAqIEBzdW1tYXJ5IFNlY29uZHMgaW4gMSB3ZWVrLlxuICovXG5leHBvcnQgY29uc3Qgc2Vjb25kc0luV2VlayA9IHNlY29uZHNJbkRheSAqIDc7XG5cbi8qKlxuICogQGNvbnN0YW50XG4gKiBAbmFtZSBzZWNvbmRzSW5ZZWFyXG4gKiBAc3VtbWFyeSBTZWNvbmRzIGluIDEgeWVhci5cbiAqL1xuZXhwb3J0IGNvbnN0IHNlY29uZHNJblllYXIgPSBzZWNvbmRzSW5EYXkgKiBkYXlzSW5ZZWFyO1xuXG4vKipcbiAqIEBjb25zdGFudFxuICogQG5hbWUgc2Vjb25kc0luTW9udGhcbiAqIEBzdW1tYXJ5IFNlY29uZHMgaW4gMSBtb250aFxuICovXG5leHBvcnQgY29uc3Qgc2Vjb25kc0luTW9udGggPSBzZWNvbmRzSW5ZZWFyIC8gMTI7XG5cbi8qKlxuICogQGNvbnN0YW50XG4gKiBAbmFtZSBzZWNvbmRzSW5RdWFydGVyXG4gKiBAc3VtbWFyeSBTZWNvbmRzIGluIDEgcXVhcnRlci5cbiAqL1xuZXhwb3J0IGNvbnN0IHNlY29uZHNJblF1YXJ0ZXIgPSBzZWNvbmRzSW5Nb250aCAqIDM7XG5cbi8qKlxuICogQGNvbnN0YW50XG4gKiBAbmFtZSBjb25zdHJ1Y3RGcm9tU3ltYm9sXG4gKiBAc3VtbWFyeSBTeW1ib2wgZW5hYmxpbmcgRGF0ZSBleHRlbnNpb25zIHRvIGluaGVyaXQgcHJvcGVydGllcyBmcm9tIHRoZSByZWZlcmVuY2UgZGF0ZS5cbiAqXG4gKiBUaGUgc3ltYm9sIGlzIHVzZWQgdG8gZW5hYmxlIHRoZSBgY29uc3RydWN0RnJvbWAgZnVuY3Rpb24gdG8gY29uc3RydWN0IGEgZGF0ZVxuICogdXNpbmcgYSByZWZlcmVuY2UgZGF0ZSBhbmQgYSB2YWx1ZS4gSXQgYWxsb3dzIHRvIHRyYW5zZmVyIGV4dHJhIHByb3BlcnRpZXNcbiAqIGZyb20gdGhlIHJlZmVyZW5jZSBkYXRlIHRvIHRoZSBuZXcgZGF0ZS4gSXQncyB1c2VmdWwgZm9yIGV4dGVuc2lvbnMgbGlrZVxuICogW2BUWkRhdGVgXShodHRwczovL2dpdGh1Yi5jb20vZGF0ZS1mbnMvdHopIHRoYXQgYWNjZXB0IGEgdGltZSB6b25lIGFzXG4gKiBhIGNvbnN0cnVjdG9yIGFyZ3VtZW50LlxuICovXG5leHBvcnQgY29uc3QgY29uc3RydWN0RnJvbVN5bWJvbCA9IFN5bWJvbC5mb3IoXCJjb25zdHJ1Y3REYXRlRnJvbVwiKTtcbiIsImltcG9ydCB7IGNvbnN0cnVjdEZyb21TeW1ib2wgfSBmcm9tIFwiLi9jb25zdGFudHMuanNcIjtcblxuLyoqXG4gKiBAbmFtZSBjb25zdHJ1Y3RGcm9tXG4gKiBAY2F0ZWdvcnkgR2VuZXJpYyBIZWxwZXJzXG4gKiBAc3VtbWFyeSBDb25zdHJ1Y3RzIGEgZGF0ZSB1c2luZyB0aGUgcmVmZXJlbmNlIGRhdGUgYW5kIHRoZSB2YWx1ZVxuICpcbiAqIEBkZXNjcmlwdGlvblxuICogVGhlIGZ1bmN0aW9uIGNvbnN0cnVjdHMgYSBuZXcgZGF0ZSB1c2luZyB0aGUgY29uc3RydWN0b3IgZnJvbSB0aGUgcmVmZXJlbmNlXG4gKiBkYXRlIGFuZCB0aGUgZ2l2ZW4gdmFsdWUuIEl0IGhlbHBzIHRvIGJ1aWxkIGdlbmVyaWMgZnVuY3Rpb25zIHRoYXQgYWNjZXB0XG4gKiBkYXRlIGV4dGVuc2lvbnMuXG4gKlxuICogSXQgZGVmYXVsdHMgdG8gYERhdGVgIGlmIHRoZSBwYXNzZWQgcmVmZXJlbmNlIGRhdGUgaXMgYSBudW1iZXIgb3IgYSBzdHJpbmcuXG4gKlxuICogU3RhcnRpbmcgZnJvbSB2My43LjAsIGl0IGFsbG93cyB0byBjb25zdHJ1Y3QgYSBkYXRlIHVzaW5nIGBbU3ltYm9sLmZvcihcImNvbnN0cnVjdERhdGVGcm9tXCIpXWBcbiAqIGVuYWJsaW5nIHRvIHRyYW5zZmVyIGV4dHJhIHByb3BlcnRpZXMgZnJvbSB0aGUgcmVmZXJlbmNlIGRhdGUgdG8gdGhlIG5ldyBkYXRlLlxuICogSXQncyB1c2VmdWwgZm9yIGV4dGVuc2lvbnMgbGlrZSBbYFRaRGF0ZWBdKGh0dHBzOi8vZ2l0aHViLmNvbS9kYXRlLWZucy90eilcbiAqIHRoYXQgYWNjZXB0IGEgdGltZSB6b25lIGFzIGEgY29uc3RydWN0b3IgYXJndW1lbnQuXG4gKlxuICogQHR5cGVQYXJhbSBEYXRlVHlwZSAtIFRoZSBgRGF0ZWAgdHlwZSwgdGhlIGZ1bmN0aW9uIG9wZXJhdGVzIG9uLiBHZXRzIGluZmVycmVkIGZyb20gcGFzc2VkIGFyZ3VtZW50cy4gQWxsb3dzIHRvIHVzZSBleHRlbnNpb25zIGxpa2UgW2BVVENEYXRlYF0oaHR0cHM6Ly9naXRodWIuY29tL2RhdGUtZm5zL3V0YykuXG4gKlxuICogQHBhcmFtIGRhdGUgLSBUaGUgcmVmZXJlbmNlIGRhdGUgdG8gdGFrZSBjb25zdHJ1Y3RvciBmcm9tXG4gKiBAcGFyYW0gdmFsdWUgLSBUaGUgdmFsdWUgdG8gY3JlYXRlIHRoZSBkYXRlXG4gKlxuICogQHJldHVybnMgRGF0ZSBpbml0aWFsaXplZCB1c2luZyB0aGUgZ2l2ZW4gZGF0ZSBhbmQgdmFsdWVcbiAqXG4gKiBAZXhhbXBsZVxuICogaW1wb3J0IHsgY29uc3RydWN0RnJvbSB9IGZyb20gXCIuL2NvbnN0cnVjdEZyb20vZGF0ZS1mbnNcIjtcbiAqXG4gKiAvLyBBIGZ1bmN0aW9uIHRoYXQgY2xvbmVzIGEgZGF0ZSBwcmVzZXJ2aW5nIHRoZSBvcmlnaW5hbCB0eXBlXG4gKiBmdW5jdGlvbiBjbG9uZURhdGU8RGF0ZVR5cGUgZXh0ZW5kcyBEYXRlPihkYXRlOiBEYXRlVHlwZSk6IERhdGVUeXBlIHtcbiAqICAgcmV0dXJuIGNvbnN0cnVjdEZyb20oXG4gKiAgICAgZGF0ZSwgLy8gVXNlIGNvbnN0cnVjdG9yIGZyb20gdGhlIGdpdmVuIGRhdGVcbiAqICAgICBkYXRlLmdldFRpbWUoKSAvLyBVc2UgdGhlIGRhdGUgdmFsdWUgdG8gY3JlYXRlIGEgbmV3IGRhdGVcbiAqICAgKTtcbiAqIH1cbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIGNvbnN0cnVjdEZyb20oZGF0ZSwgdmFsdWUpIHtcbiAgaWYgKHR5cGVvZiBkYXRlID09PSBcImZ1bmN0aW9uXCIpIHJldHVybiBkYXRlKHZhbHVlKTtcblxuICBpZiAoZGF0ZSAmJiB0eXBlb2YgZGF0ZSA9PT0gXCJvYmplY3RcIiAmJiBjb25zdHJ1Y3RGcm9tU3ltYm9sIGluIGRhdGUpXG4gICAgcmV0dXJuIGRhdGVbY29uc3RydWN0RnJvbVN5bWJvbF0odmFsdWUpO1xuXG4gIGlmIChkYXRlIGluc3RhbmNlb2YgRGF0ZSkgcmV0dXJuIG5ldyBkYXRlLmNvbnN0cnVjdG9yKHZhbHVlKTtcblxuICByZXR1cm4gbmV3IERhdGUodmFsdWUpO1xufVxuXG4vLyBGYWxsYmFjayBmb3IgbW9kdWxhcml6ZWQgaW1wb3J0czpcbmV4cG9ydCBkZWZhdWx0IGNvbnN0cnVjdEZyb207XG4iLCJpbXBvcnQgeyBjb25zdHJ1Y3RGcm9tIH0gZnJvbSBcIi4vY29uc3RydWN0RnJvbS5qc1wiO1xuXG4vKipcbiAqIEBuYW1lIHRvRGF0ZVxuICogQGNhdGVnb3J5IENvbW1vbiBIZWxwZXJzXG4gKiBAc3VtbWFyeSBDb252ZXJ0IHRoZSBnaXZlbiBhcmd1bWVudCB0byBhbiBpbnN0YW5jZSBvZiBEYXRlLlxuICpcbiAqIEBkZXNjcmlwdGlvblxuICogQ29udmVydCB0aGUgZ2l2ZW4gYXJndW1lbnQgdG8gYW4gaW5zdGFuY2Ugb2YgRGF0ZS5cbiAqXG4gKiBJZiB0aGUgYXJndW1lbnQgaXMgYW4gaW5zdGFuY2Ugb2YgRGF0ZSwgdGhlIGZ1bmN0aW9uIHJldHVybnMgaXRzIGNsb25lLlxuICpcbiAqIElmIHRoZSBhcmd1bWVudCBpcyBhIG51bWJlciwgaXQgaXMgdHJlYXRlZCBhcyBhIHRpbWVzdGFtcC5cbiAqXG4gKiBJZiB0aGUgYXJndW1lbnQgaXMgbm9uZSBvZiB0aGUgYWJvdmUsIHRoZSBmdW5jdGlvbiByZXR1cm5zIEludmFsaWQgRGF0ZS5cbiAqXG4gKiBTdGFydGluZyBmcm9tIHYzLjcuMCwgaXQgY2xvbmVzIGEgZGF0ZSB1c2luZyBgW1N5bWJvbC5mb3IoXCJjb25zdHJ1Y3REYXRlRnJvbVwiKV1gXG4gKiBlbmFibGluZyB0byB0cmFuc2ZlciBleHRyYSBwcm9wZXJ0aWVzIGZyb20gdGhlIHJlZmVyZW5jZSBkYXRlIHRvIHRoZSBuZXcgZGF0ZS5cbiAqIEl0J3MgdXNlZnVsIGZvciBleHRlbnNpb25zIGxpa2UgW2BUWkRhdGVgXShodHRwczovL2dpdGh1Yi5jb20vZGF0ZS1mbnMvdHopXG4gKiB0aGF0IGFjY2VwdCBhIHRpbWUgem9uZSBhcyBhIGNvbnN0cnVjdG9yIGFyZ3VtZW50LlxuICpcbiAqICoqTm90ZSoqOiAqYWxsKiBEYXRlIGFyZ3VtZW50cyBwYXNzZWQgdG8gYW55ICpkYXRlLWZucyogZnVuY3Rpb24gaXMgcHJvY2Vzc2VkIGJ5IGB0b0RhdGVgLlxuICpcbiAqIEB0eXBlUGFyYW0gRGF0ZVR5cGUgLSBUaGUgYERhdGVgIHR5cGUsIHRoZSBmdW5jdGlvbiBvcGVyYXRlcyBvbi4gR2V0cyBpbmZlcnJlZCBmcm9tIHBhc3NlZCBhcmd1bWVudHMuIEFsbG93cyB0byB1c2UgZXh0ZW5zaW9ucyBsaWtlIFtgVVRDRGF0ZWBdKGh0dHBzOi8vZ2l0aHViLmNvbS9kYXRlLWZucy91dGMpLlxuICogQHR5cGVQYXJhbSBSZXN1bHREYXRlIC0gVGhlIHJlc3VsdCBgRGF0ZWAgdHlwZSwgaXQgaXMgdGhlIHR5cGUgcmV0dXJuZWQgZnJvbSB0aGUgY29udGV4dCBmdW5jdGlvbiBpZiBpdCBpcyBwYXNzZWQsIG9yIGluZmVycmVkIGZyb20gdGhlIGFyZ3VtZW50cy5cbiAqXG4gKiBAcGFyYW0gYXJndW1lbnQgLSBUaGUgdmFsdWUgdG8gY29udmVydFxuICpcbiAqIEByZXR1cm5zIFRoZSBwYXJzZWQgZGF0ZSBpbiB0aGUgbG9jYWwgdGltZSB6b25lXG4gKlxuICogQGV4YW1wbGVcbiAqIC8vIENsb25lIHRoZSBkYXRlOlxuICogY29uc3QgcmVzdWx0ID0gdG9EYXRlKG5ldyBEYXRlKDIwMTQsIDEsIDExLCAxMSwgMzAsIDMwKSlcbiAqIC8vPT4gVHVlIEZlYiAxMSAyMDE0IDExOjMwOjMwXG4gKlxuICogQGV4YW1wbGVcbiAqIC8vIENvbnZlcnQgdGhlIHRpbWVzdGFtcCB0byBkYXRlOlxuICogY29uc3QgcmVzdWx0ID0gdG9EYXRlKDEzOTIwOTg0MzAwMDApXG4gKiAvLz0+IFR1ZSBGZWIgMTEgMjAxNCAxMTozMDozMFxuICovXG5leHBvcnQgZnVuY3Rpb24gdG9EYXRlKGFyZ3VtZW50LCBjb250ZXh0KSB7XG4gIC8vIFtUT0RPXSBHZXQgcmlkIG9mIGB0b0RhdGVgIG9yIGBjb25zdHJ1Y3RGcm9tYD9cbiAgcmV0dXJuIGNvbnN0cnVjdEZyb20oY29udGV4dCB8fCBhcmd1bWVudCwgYXJndW1lbnQpO1xufVxuXG4vLyBGYWxsYmFjayBmb3IgbW9kdWxhcml6ZWQgaW1wb3J0czpcbmV4cG9ydCBkZWZhdWx0IHRvRGF0ZTtcbiIsImltcG9ydCB7IGNvbnN0cnVjdEZyb20gfSBmcm9tIFwiLi9jb25zdHJ1Y3RGcm9tLmpzXCI7XG5pbXBvcnQgeyB0b0RhdGUgfSBmcm9tIFwiLi90b0RhdGUuanNcIjtcblxuLyoqXG4gKiBUaGUge0BsaW5rIGFkZERheXN9IGZ1bmN0aW9uIG9wdGlvbnMuXG4gKi9cblxuLyoqXG4gKiBAbmFtZSBhZGREYXlzXG4gKiBAY2F0ZWdvcnkgRGF5IEhlbHBlcnNcbiAqIEBzdW1tYXJ5IEFkZCB0aGUgc3BlY2lmaWVkIG51bWJlciBvZiBkYXlzIHRvIHRoZSBnaXZlbiBkYXRlLlxuICpcbiAqIEBkZXNjcmlwdGlvblxuICogQWRkIHRoZSBzcGVjaWZpZWQgbnVtYmVyIG9mIGRheXMgdG8gdGhlIGdpdmVuIGRhdGUuXG4gKlxuICogQHR5cGVQYXJhbSBEYXRlVHlwZSAtIFRoZSBgRGF0ZWAgdHlwZSwgdGhlIGZ1bmN0aW9uIG9wZXJhdGVzIG9uLiBHZXRzIGluZmVycmVkIGZyb20gcGFzc2VkIGFyZ3VtZW50cy4gQWxsb3dzIHRvIHVzZSBleHRlbnNpb25zIGxpa2UgW2BVVENEYXRlYF0oaHR0cHM6Ly9naXRodWIuY29tL2RhdGUtZm5zL3V0YykuXG4gKiBAdHlwZVBhcmFtIFJlc3VsdERhdGUgLSBUaGUgcmVzdWx0IGBEYXRlYCB0eXBlLCBpdCBpcyB0aGUgdHlwZSByZXR1cm5lZCBmcm9tIHRoZSBjb250ZXh0IGZ1bmN0aW9uIGlmIGl0IGlzIHBhc3NlZCwgb3IgaW5mZXJyZWQgZnJvbSB0aGUgYXJndW1lbnRzLlxuICpcbiAqIEBwYXJhbSBkYXRlIC0gVGhlIGRhdGUgdG8gYmUgY2hhbmdlZFxuICogQHBhcmFtIGFtb3VudCAtIFRoZSBhbW91bnQgb2YgZGF5cyB0byBiZSBhZGRlZC5cbiAqIEBwYXJhbSBvcHRpb25zIC0gQW4gb2JqZWN0IHdpdGggb3B0aW9uc1xuICpcbiAqIEByZXR1cm5zIFRoZSBuZXcgZGF0ZSB3aXRoIHRoZSBkYXlzIGFkZGVkXG4gKlxuICogQGV4YW1wbGVcbiAqIC8vIEFkZCAxMCBkYXlzIHRvIDEgU2VwdGVtYmVyIDIwMTQ6XG4gKiBjb25zdCByZXN1bHQgPSBhZGREYXlzKG5ldyBEYXRlKDIwMTQsIDgsIDEpLCAxMClcbiAqIC8vPT4gVGh1IFNlcCAxMSAyMDE0IDAwOjAwOjAwXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiBhZGREYXlzKGRhdGUsIGFtb3VudCwgb3B0aW9ucykge1xuICBjb25zdCBfZGF0ZSA9IHRvRGF0ZShkYXRlLCBvcHRpb25zPy5pbik7XG4gIGlmIChpc05hTihhbW91bnQpKSByZXR1cm4gY29uc3RydWN0RnJvbShvcHRpb25zPy5pbiB8fCBkYXRlLCBOYU4pO1xuXG4gIC8vIElmIDAgZGF5cywgbm8tb3AgdG8gYXZvaWQgY2hhbmdpbmcgdGltZXMgaW4gdGhlIGhvdXIgYmVmb3JlIGVuZCBvZiBEU1RcbiAgaWYgKCFhbW91bnQpIHJldHVybiBfZGF0ZTtcblxuICBfZGF0ZS5zZXREYXRlKF9kYXRlLmdldERhdGUoKSArIGFtb3VudCk7XG4gIHJldHVybiBfZGF0ZTtcbn1cblxuLy8gRmFsbGJhY2sgZm9yIG1vZHVsYXJpemVkIGltcG9ydHM6XG5leHBvcnQgZGVmYXVsdCBhZGREYXlzO1xuIiwibGV0IGRlZmF1bHRPcHRpb25zID0ge307XG5cbmV4cG9ydCBmdW5jdGlvbiBnZXREZWZhdWx0T3B0aW9ucygpIHtcbiAgcmV0dXJuIGRlZmF1bHRPcHRpb25zO1xufVxuXG5leHBvcnQgZnVuY3Rpb24gc2V0RGVmYXVsdE9wdGlvbnMobmV3T3B0aW9ucykge1xuICBkZWZhdWx0T3B0aW9ucyA9IG5ld09wdGlvbnM7XG59XG4iLCJpbXBvcnQgeyBnZXREZWZhdWx0T3B0aW9ucyB9IGZyb20gXCIuL19saWIvZGVmYXVsdE9wdGlvbnMuanNcIjtcbmltcG9ydCB7IHRvRGF0ZSB9IGZyb20gXCIuL3RvRGF0ZS5qc1wiO1xuXG4vKipcbiAqIFRoZSB7QGxpbmsgc3RhcnRPZldlZWt9IGZ1bmN0aW9uIG9wdGlvbnMuXG4gKi9cblxuLyoqXG4gKiBAbmFtZSBzdGFydE9mV2Vla1xuICogQGNhdGVnb3J5IFdlZWsgSGVscGVyc1xuICogQHN1bW1hcnkgUmV0dXJuIHRoZSBzdGFydCBvZiBhIHdlZWsgZm9yIHRoZSBnaXZlbiBkYXRlLlxuICpcbiAqIEBkZXNjcmlwdGlvblxuICogUmV0dXJuIHRoZSBzdGFydCBvZiBhIHdlZWsgZm9yIHRoZSBnaXZlbiBkYXRlLlxuICogVGhlIHJlc3VsdCB3aWxsIGJlIGluIHRoZSBsb2NhbCB0aW1lem9uZS5cbiAqXG4gKiBAdHlwZVBhcmFtIERhdGVUeXBlIC0gVGhlIGBEYXRlYCB0eXBlLCB0aGUgZnVuY3Rpb24gb3BlcmF0ZXMgb24uIEdldHMgaW5mZXJyZWQgZnJvbSBwYXNzZWQgYXJndW1lbnRzLiBBbGxvd3MgdG8gdXNlIGV4dGVuc2lvbnMgbGlrZSBbYFVUQ0RhdGVgXShodHRwczovL2dpdGh1Yi5jb20vZGF0ZS1mbnMvdXRjKS5cbiAqIEB0eXBlUGFyYW0gUmVzdWx0RGF0ZSAtIFRoZSByZXN1bHQgYERhdGVgIHR5cGUsIGl0IGlzIHRoZSB0eXBlIHJldHVybmVkIGZyb20gdGhlIGNvbnRleHQgZnVuY3Rpb24gaWYgaXQgaXMgcGFzc2VkLCBvciBpbmZlcnJlZCBmcm9tIHRoZSBhcmd1bWVudHMuXG4gKlxuICogQHBhcmFtIGRhdGUgLSBUaGUgb3JpZ2luYWwgZGF0ZVxuICogQHBhcmFtIG9wdGlvbnMgLSBBbiBvYmplY3Qgd2l0aCBvcHRpb25zXG4gKlxuICogQHJldHVybnMgVGhlIHN0YXJ0IG9mIGEgd2Vla1xuICpcbiAqIEBleGFtcGxlXG4gKiAvLyBUaGUgc3RhcnQgb2YgYSB3ZWVrIGZvciAyIFNlcHRlbWJlciAyMDE0IDExOjU1OjAwOlxuICogY29uc3QgcmVzdWx0ID0gc3RhcnRPZldlZWsobmV3IERhdGUoMjAxNCwgOCwgMiwgMTEsIDU1LCAwKSlcbiAqIC8vPT4gU3VuIEF1ZyAzMSAyMDE0IDAwOjAwOjAwXG4gKlxuICogQGV4YW1wbGVcbiAqIC8vIElmIHRoZSB3ZWVrIHN0YXJ0cyBvbiBNb25kYXksIHRoZSBzdGFydCBvZiB0aGUgd2VlayBmb3IgMiBTZXB0ZW1iZXIgMjAxNCAxMTo1NTowMDpcbiAqIGNvbnN0IHJlc3VsdCA9IHN0YXJ0T2ZXZWVrKG5ldyBEYXRlKDIwMTQsIDgsIDIsIDExLCA1NSwgMCksIHsgd2Vla1N0YXJ0c09uOiAxIH0pXG4gKiAvLz0+IE1vbiBTZXAgMDEgMjAxNCAwMDowMDowMFxuICovXG5leHBvcnQgZnVuY3Rpb24gc3RhcnRPZldlZWsoZGF0ZSwgb3B0aW9ucykge1xuICBjb25zdCBkZWZhdWx0T3B0aW9ucyA9IGdldERlZmF1bHRPcHRpb25zKCk7XG4gIGNvbnN0IHdlZWtTdGFydHNPbiA9XG4gICAgb3B0aW9ucz8ud2Vla1N0YXJ0c09uID8/XG4gICAgb3B0aW9ucz8ubG9jYWxlPy5vcHRpb25zPy53ZWVrU3RhcnRzT24gPz9cbiAgICBkZWZhdWx0T3B0aW9ucy53ZWVrU3RhcnRzT24gPz9cbiAgICBkZWZhdWx0T3B0aW9ucy5sb2NhbGU/Lm9wdGlvbnM/LndlZWtTdGFydHNPbiA/P1xuICAgIDA7XG5cbiAgY29uc3QgX2RhdGUgPSB0b0RhdGUoZGF0ZSwgb3B0aW9ucz8uaW4pO1xuICBjb25zdCBkYXkgPSBfZGF0ZS5nZXREYXkoKTtcbiAgY29uc3QgZGlmZiA9IChkYXkgPCB3ZWVrU3RhcnRzT24gPyA3IDogMCkgKyBkYXkgLSB3ZWVrU3RhcnRzT247XG5cbiAgX2RhdGUuc2V0RGF0ZShfZGF0ZS5nZXREYXRlKCkgLSBkaWZmKTtcbiAgX2RhdGUuc2V0SG91cnMoMCwgMCwgMCwgMCk7XG4gIHJldHVybiBfZGF0ZTtcbn1cblxuLy8gRmFsbGJhY2sgZm9yIG1vZHVsYXJpemVkIGltcG9ydHM6XG5leHBvcnQgZGVmYXVsdCBzdGFydE9mV2VlaztcbiIsImltcG9ydCB7IGNvbnN0cnVjdEZyb20gfSBmcm9tIFwiLi4vY29uc3RydWN0RnJvbS5qc1wiO1xuXG5leHBvcnQgZnVuY3Rpb24gbm9ybWFsaXplRGF0ZXMoY29udGV4dCwgLi4uZGF0ZXMpIHtcbiAgY29uc3Qgbm9ybWFsaXplID0gY29uc3RydWN0RnJvbS5iaW5kKFxuICAgIG51bGwsXG4gICAgY29udGV4dCB8fCBkYXRlcy5maW5kKChkYXRlKSA9PiB0eXBlb2YgZGF0ZSA9PT0gXCJvYmplY3RcIiksXG4gICk7XG4gIHJldHVybiBkYXRlcy5tYXAobm9ybWFsaXplKTtcbn1cbiIsImltcG9ydCB7IGFkZERheXMgfSBmcm9tIFwiLi9hZGREYXlzLmpzXCI7XG5cbi8qKlxuICogVGhlIHtAbGluayBhZGRXZWVrc30gZnVuY3Rpb24gb3B0aW9ucy5cbiAqL1xuXG4vKipcbiAqIEBuYW1lIGFkZFdlZWtzXG4gKiBAY2F0ZWdvcnkgV2VlayBIZWxwZXJzXG4gKiBAc3VtbWFyeSBBZGQgdGhlIHNwZWNpZmllZCBudW1iZXIgb2Ygd2Vla3MgdG8gdGhlIGdpdmVuIGRhdGUuXG4gKlxuICogQGRlc2NyaXB0aW9uXG4gKiBBZGQgdGhlIHNwZWNpZmllZCBudW1iZXIgb2Ygd2Vla3MgdG8gdGhlIGdpdmVuIGRhdGUuXG4gKlxuICogQHR5cGVQYXJhbSBEYXRlVHlwZSAtIFRoZSBgRGF0ZWAgdHlwZSwgdGhlIGZ1bmN0aW9uIG9wZXJhdGVzIG9uLiBHZXRzIGluZmVycmVkIGZyb20gcGFzc2VkIGFyZ3VtZW50cy4gQWxsb3dzIHRvIHVzZSBleHRlbnNpb25zIGxpa2UgW2BVVENEYXRlYF0oaHR0cHM6Ly9naXRodWIuY29tL2RhdGUtZm5zL3V0YykuXG4gKiBAdHlwZVBhcmFtIFJlc3VsdERhdGUgLSBUaGUgcmVzdWx0IGBEYXRlYCB0eXBlLCBpdCBpcyB0aGUgdHlwZSByZXR1cm5lZCBmcm9tIHRoZSBjb250ZXh0IGZ1bmN0aW9uIGlmIGl0IGlzIHBhc3NlZCwgb3IgaW5mZXJyZWQgZnJvbSB0aGUgYXJndW1lbnRzLlxuICpcbiAqIEBwYXJhbSBkYXRlIC0gVGhlIGRhdGUgdG8gYmUgY2hhbmdlZFxuICogQHBhcmFtIGFtb3VudCAtIFRoZSBhbW91bnQgb2Ygd2Vla3MgdG8gYmUgYWRkZWQuXG4gKiBAcGFyYW0gb3B0aW9ucyAtIEFuIG9iamVjdCB3aXRoIG9wdGlvbnNcbiAqXG4gKiBAcmV0dXJucyBUaGUgbmV3IGRhdGUgd2l0aCB0aGUgd2Vla3MgYWRkZWRcbiAqXG4gKiBAZXhhbXBsZVxuICogLy8gQWRkIDQgd2Vla3MgdG8gMSBTZXB0ZW1iZXIgMjAxNDpcbiAqIGNvbnN0IHJlc3VsdCA9IGFkZFdlZWtzKG5ldyBEYXRlKDIwMTQsIDgsIDEpLCA0KVxuICogLy89PiBNb24gU2VwIDI5IDIwMTQgMDA6MDA6MDBcbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIGFkZFdlZWtzKGRhdGUsIGFtb3VudCwgb3B0aW9ucykge1xuICByZXR1cm4gYWRkRGF5cyhkYXRlLCBhbW91bnQgKiA3LCBvcHRpb25zKTtcbn1cblxuLy8gRmFsbGJhY2sgZm9yIG1vZHVsYXJpemVkIGltcG9ydHM6XG5leHBvcnQgZGVmYXVsdCBhZGRXZWVrcztcbiIsImltcG9ydCB7IHRvRGF0ZSB9IGZyb20gXCIuL3RvRGF0ZS5qc1wiO1xuXG4vKipcbiAqIFRoZSB7QGxpbmsgZW5kT2ZNb250aH0gZnVuY3Rpb24gb3B0aW9ucy5cbiAqL1xuXG4vKipcbiAqIEBuYW1lIGVuZE9mTW9udGhcbiAqIEBjYXRlZ29yeSBNb250aCBIZWxwZXJzXG4gKiBAc3VtbWFyeSBSZXR1cm4gdGhlIGVuZCBvZiBhIG1vbnRoIGZvciB0aGUgZ2l2ZW4gZGF0ZS5cbiAqXG4gKiBAZGVzY3JpcHRpb25cbiAqIFJldHVybiB0aGUgZW5kIG9mIGEgbW9udGggZm9yIHRoZSBnaXZlbiBkYXRlLlxuICogVGhlIHJlc3VsdCB3aWxsIGJlIGluIHRoZSBsb2NhbCB0aW1lem9uZS5cbiAqXG4gKiBAdHlwZVBhcmFtIERhdGVUeXBlIC0gVGhlIGBEYXRlYCB0eXBlLCB0aGUgZnVuY3Rpb24gb3BlcmF0ZXMgb24uIEdldHMgaW5mZXJyZWQgZnJvbSBwYXNzZWQgYXJndW1lbnRzLiBBbGxvd3MgdG8gdXNlIGV4dGVuc2lvbnMgbGlrZSBbYFVUQ0RhdGVgXShodHRwczovL2dpdGh1Yi5jb20vZGF0ZS1mbnMvdXRjKS5cbiAqIEB0eXBlUGFyYW0gUmVzdWx0RGF0ZSAtIFRoZSByZXN1bHQgYERhdGVgIHR5cGUsIGl0IGlzIHRoZSB0eXBlIHJldHVybmVkIGZyb20gdGhlIGNvbnRleHQgZnVuY3Rpb24gaWYgaXQgaXMgcGFzc2VkLCBvciBpbmZlcnJlZCBmcm9tIHRoZSBhcmd1bWVudHMuXG4gKlxuICogQHBhcmFtIGRhdGUgLSBUaGUgb3JpZ2luYWwgZGF0ZVxuICogQHBhcmFtIG9wdGlvbnMgLSBBbiBvYmplY3Qgd2l0aCBvcHRpb25zXG4gKlxuICogQHJldHVybnMgVGhlIGVuZCBvZiBhIG1vbnRoXG4gKlxuICogQGV4YW1wbGVcbiAqIC8vIFRoZSBlbmQgb2YgYSBtb250aCBmb3IgMiBTZXB0ZW1iZXIgMjAxNCAxMTo1NTowMDpcbiAqIGNvbnN0IHJlc3VsdCA9IGVuZE9mTW9udGgobmV3IERhdGUoMjAxNCwgOCwgMiwgMTEsIDU1LCAwKSlcbiAqIC8vPT4gVHVlIFNlcCAzMCAyMDE0IDIzOjU5OjU5Ljk5OVxuICovXG5leHBvcnQgZnVuY3Rpb24gZW5kT2ZNb250aChkYXRlLCBvcHRpb25zKSB7XG4gIGNvbnN0IF9kYXRlID0gdG9EYXRlKGRhdGUsIG9wdGlvbnM/LmluKTtcbiAgY29uc3QgbW9udGggPSBfZGF0ZS5nZXRNb250aCgpO1xuICBfZGF0ZS5zZXRGdWxsWWVhcihfZGF0ZS5nZXRGdWxsWWVhcigpLCBtb250aCArIDEsIDApO1xuICBfZGF0ZS5zZXRIb3VycygyMywgNTksIDU5LCA5OTkpO1xuICByZXR1cm4gX2RhdGU7XG59XG5cbi8vIEZhbGxiYWNrIGZvciBtb2R1bGFyaXplZCBpbXBvcnRzOlxuZXhwb3J0IGRlZmF1bHQgZW5kT2ZNb250aDtcbiIsImltcG9ydCB7IG5vcm1hbGl6ZURhdGVzIH0gZnJvbSBcIi4vbm9ybWFsaXplRGF0ZXMuanNcIjtcblxuZXhwb3J0IGZ1bmN0aW9uIG5vcm1hbGl6ZUludGVydmFsKGNvbnRleHQsIGludGVydmFsKSB7XG4gIGNvbnN0IFtzdGFydCwgZW5kXSA9IG5vcm1hbGl6ZURhdGVzKGNvbnRleHQsIGludGVydmFsLnN0YXJ0LCBpbnRlcnZhbC5lbmQpO1xuICByZXR1cm4geyBzdGFydCwgZW5kIH07XG59XG4iLCJpbXBvcnQgeyBub3JtYWxpemVJbnRlcnZhbCB9IGZyb20gXCIuL19saWIvbm9ybWFsaXplSW50ZXJ2YWwuanNcIjtcbmltcG9ydCB7IGFkZFdlZWtzIH0gZnJvbSBcIi4vYWRkV2Vla3MuanNcIjtcbmltcG9ydCB7IGNvbnN0cnVjdEZyb20gfSBmcm9tIFwiLi9jb25zdHJ1Y3RGcm9tLmpzXCI7XG5pbXBvcnQgeyBzdGFydE9mV2VlayB9IGZyb20gXCIuL3N0YXJ0T2ZXZWVrLmpzXCI7XG5cbi8qKlxuICogVGhlIHtAbGluayBlYWNoV2Vla09mSW50ZXJ2YWx9IGZ1bmN0aW9uIG9wdGlvbnMuXG4gKi9cblxuLyoqXG4gKiBUaGUge0BsaW5rIGVhY2hXZWVrT2ZJbnRlcnZhbH0gZnVuY3Rpb24gcmVzdWx0IHR5cGUuIEl0IHJlc29sdmVzIHRoZSBwcm9wZXIgZGF0YSB0eXBlLlxuICogSXQgdXNlcyB0aGUgZmlyc3QgYXJndW1lbnQgZGF0ZSBvYmplY3QgdHlwZSwgc3RhcnRpbmcgZnJvbSB0aGUgaW50ZXJ2YWwgc3RhcnQgZGF0ZSxcbiAqIHRoZW4gdGhlIGVuZCBpbnRlcnZhbCBkYXRlLiBJZiBhIGNvbnRleHQgZnVuY3Rpb24gaXMgcGFzc2VkLCBpdCB1c2VzIHRoZSBjb250ZXh0IGZ1bmN0aW9uIHJldHVybiB0eXBlLlxuICovXG5cbi8qKlxuICogQG5hbWUgZWFjaFdlZWtPZkludGVydmFsXG4gKiBAY2F0ZWdvcnkgSW50ZXJ2YWwgSGVscGVyc1xuICogQHN1bW1hcnkgUmV0dXJuIHRoZSBhcnJheSBvZiB3ZWVrcyB3aXRoaW4gdGhlIHNwZWNpZmllZCB0aW1lIGludGVydmFsLlxuICpcbiAqIEBkZXNjcmlwdGlvblxuICogUmV0dXJuIHRoZSBhcnJheSBvZiB3ZWVrcyB3aXRoaW4gdGhlIHNwZWNpZmllZCB0aW1lIGludGVydmFsLlxuICpcbiAqIEBwYXJhbSBpbnRlcnZhbCAtIFRoZSBpbnRlcnZhbC5cbiAqIEBwYXJhbSBvcHRpb25zIC0gQW4gb2JqZWN0IHdpdGggb3B0aW9ucy5cbiAqXG4gKiBAcmV0dXJucyBUaGUgYXJyYXkgd2l0aCBzdGFydHMgb2Ygd2Vla3MgZnJvbSB0aGUgd2VlayBvZiB0aGUgaW50ZXJ2YWwgc3RhcnQgdG8gdGhlIHdlZWsgb2YgdGhlIGludGVydmFsIGVuZFxuICpcbiAqIEBleGFtcGxlXG4gKiAvLyBFYWNoIHdlZWsgd2l0aGluIGludGVydmFsIDYgT2N0b2JlciAyMDE0IC0gMjMgTm92ZW1iZXIgMjAxNDpcbiAqIGNvbnN0IHJlc3VsdCA9IGVhY2hXZWVrT2ZJbnRlcnZhbCh7XG4gKiAgIHN0YXJ0OiBuZXcgRGF0ZSgyMDE0LCA5LCA2KSxcbiAqICAgZW5kOiBuZXcgRGF0ZSgyMDE0LCAxMCwgMjMpXG4gKiB9KVxuICogLy89PiBbXG4gKiAvLyAgIFN1biBPY3QgMDUgMjAxNCAwMDowMDowMCxcbiAqIC8vICAgU3VuIE9jdCAxMiAyMDE0IDAwOjAwOjAwLFxuICogLy8gICBTdW4gT2N0IDE5IDIwMTQgMDA6MDA6MDAsXG4gKiAvLyAgIFN1biBPY3QgMjYgMjAxNCAwMDowMDowMCxcbiAqIC8vICAgU3VuIE5vdiAwMiAyMDE0IDAwOjAwOjAwLFxuICogLy8gICBTdW4gTm92IDA5IDIwMTQgMDA6MDA6MDAsXG4gKiAvLyAgIFN1biBOb3YgMTYgMjAxNCAwMDowMDowMCxcbiAqIC8vICAgU3VuIE5vdiAyMyAyMDE0IDAwOjAwOjAwXG4gKiAvLyBdXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiBlYWNoV2Vla09mSW50ZXJ2YWwoaW50ZXJ2YWwsIG9wdGlvbnMpIHtcbiAgY29uc3QgeyBzdGFydCwgZW5kIH0gPSBub3JtYWxpemVJbnRlcnZhbChvcHRpb25zPy5pbiwgaW50ZXJ2YWwpO1xuXG4gIGxldCByZXZlcnNlZCA9ICtzdGFydCA+ICtlbmQ7XG4gIGNvbnN0IHN0YXJ0RGF0ZVdlZWsgPSByZXZlcnNlZFxuICAgID8gc3RhcnRPZldlZWsoZW5kLCBvcHRpb25zKVxuICAgIDogc3RhcnRPZldlZWsoc3RhcnQsIG9wdGlvbnMpO1xuICBjb25zdCBlbmREYXRlV2VlayA9IHJldmVyc2VkXG4gICAgPyBzdGFydE9mV2VlayhzdGFydCwgb3B0aW9ucylcbiAgICA6IHN0YXJ0T2ZXZWVrKGVuZCwgb3B0aW9ucyk7XG5cbiAgc3RhcnREYXRlV2Vlay5zZXRIb3VycygxNSk7XG4gIGVuZERhdGVXZWVrLnNldEhvdXJzKDE1KTtcblxuICBjb25zdCBlbmRUaW1lID0gK2VuZERhdGVXZWVrLmdldFRpbWUoKTtcbiAgbGV0IGN1cnJlbnREYXRlID0gc3RhcnREYXRlV2VlaztcblxuICBsZXQgc3RlcCA9IG9wdGlvbnM/LnN0ZXAgPz8gMTtcbiAgaWYgKCFzdGVwKSByZXR1cm4gW107XG4gIGlmIChzdGVwIDwgMCkge1xuICAgIHN0ZXAgPSAtc3RlcDtcbiAgICByZXZlcnNlZCA9ICFyZXZlcnNlZDtcbiAgfVxuXG4gIGNvbnN0IGRhdGVzID0gW107XG5cbiAgd2hpbGUgKCtjdXJyZW50RGF0ZSA8PSBlbmRUaW1lKSB7XG4gICAgY3VycmVudERhdGUuc2V0SG91cnMoMCk7XG4gICAgZGF0ZXMucHVzaChjb25zdHJ1Y3RGcm9tKHN0YXJ0LCBjdXJyZW50RGF0ZSkpO1xuICAgIGN1cnJlbnREYXRlID0gYWRkV2Vla3MoY3VycmVudERhdGUsIHN0ZXApO1xuICAgIGN1cnJlbnREYXRlLnNldEhvdXJzKDE1KTtcbiAgfVxuXG4gIHJldHVybiByZXZlcnNlZCA/IGRhdGVzLnJldmVyc2UoKSA6IGRhdGVzO1xufVxuXG4vLyBGYWxsYmFjayBmb3IgbW9kdWxhcml6ZWQgaW1wb3J0czpcbmV4cG9ydCBkZWZhdWx0IGVhY2hXZWVrT2ZJbnRlcnZhbDtcbiIsImltcG9ydCB7IHRvRGF0ZSB9IGZyb20gXCIuL3RvRGF0ZS5qc1wiO1xuXG4vKipcbiAqIFRoZSB7QGxpbmsgc3RhcnRPZk1vbnRofSBmdW5jdGlvbiBvcHRpb25zLlxuICovXG5cbi8qKlxuICogQG5hbWUgc3RhcnRPZk1vbnRoXG4gKiBAY2F0ZWdvcnkgTW9udGggSGVscGVyc1xuICogQHN1bW1hcnkgUmV0dXJuIHRoZSBzdGFydCBvZiBhIG1vbnRoIGZvciB0aGUgZ2l2ZW4gZGF0ZS5cbiAqXG4gKiBAZGVzY3JpcHRpb25cbiAqIFJldHVybiB0aGUgc3RhcnQgb2YgYSBtb250aCBmb3IgdGhlIGdpdmVuIGRhdGUuIFRoZSByZXN1bHQgd2lsbCBiZSBpbiB0aGUgbG9jYWwgdGltZXpvbmUuXG4gKlxuICogQHR5cGVQYXJhbSBEYXRlVHlwZSAtIFRoZSBgRGF0ZWAgdHlwZSwgdGhlIGZ1bmN0aW9uIG9wZXJhdGVzIG9uLiBHZXRzIGluZmVycmVkIGZyb20gcGFzc2VkIGFyZ3VtZW50cy5cbiAqIEFsbG93cyB0byB1c2UgZXh0ZW5zaW9ucyBsaWtlIFtgVVRDRGF0ZWBdKGh0dHBzOi8vZ2l0aHViLmNvbS9kYXRlLWZucy91dGMpLlxuICogQHR5cGVQYXJhbSBSZXN1bHREYXRlIC0gVGhlIHJlc3VsdCBgRGF0ZWAgdHlwZSwgaXQgaXMgdGhlIHR5cGUgcmV0dXJuZWQgZnJvbSB0aGUgY29udGV4dCBmdW5jdGlvbiBpZiBpdCBpcyBwYXNzZWQsXG4gKiBvciBpbmZlcnJlZCBmcm9tIHRoZSBhcmd1bWVudHMuXG4gKlxuICogQHBhcmFtIGRhdGUgLSBUaGUgb3JpZ2luYWwgZGF0ZVxuICogQHBhcmFtIG9wdGlvbnMgLSBBbiBvYmplY3Qgd2l0aCBvcHRpb25zXG4gKlxuICogQHJldHVybnMgVGhlIHN0YXJ0IG9mIGEgbW9udGhcbiAqXG4gKiBAZXhhbXBsZVxuICogLy8gVGhlIHN0YXJ0IG9mIGEgbW9udGggZm9yIDIgU2VwdGVtYmVyIDIwMTQgMTE6NTU6MDA6XG4gKiBjb25zdCByZXN1bHQgPSBzdGFydE9mTW9udGgobmV3IERhdGUoMjAxNCwgOCwgMiwgMTEsIDU1LCAwKSlcbiAqIC8vPT4gTW9uIFNlcCAwMSAyMDE0IDAwOjAwOjAwXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiBzdGFydE9mTW9udGgoZGF0ZSwgb3B0aW9ucykge1xuICBjb25zdCBfZGF0ZSA9IHRvRGF0ZShkYXRlLCBvcHRpb25zPy5pbik7XG4gIF9kYXRlLnNldERhdGUoMSk7XG4gIF9kYXRlLnNldEhvdXJzKDAsIDAsIDAsIDApO1xuICByZXR1cm4gX2RhdGU7XG59XG5cbi8vIEZhbGxiYWNrIGZvciBtb2R1bGFyaXplZCBpbXBvcnRzOlxuZXhwb3J0IGRlZmF1bHQgc3RhcnRPZk1vbnRoO1xuIiwiaW1wb3J0IHsgZ2V0RGVmYXVsdE9wdGlvbnMgfSBmcm9tIFwiLi9fbGliL2RlZmF1bHRPcHRpb25zLmpzXCI7XG5pbXBvcnQgeyBjb25zdHJ1Y3RGcm9tIH0gZnJvbSBcIi4vY29uc3RydWN0RnJvbS5qc1wiO1xuaW1wb3J0IHsgc3RhcnRPZldlZWsgfSBmcm9tIFwiLi9zdGFydE9mV2Vlay5qc1wiO1xuaW1wb3J0IHsgdG9EYXRlIH0gZnJvbSBcIi4vdG9EYXRlLmpzXCI7XG5cbi8qKlxuICogVGhlIHtAbGluayBnZXRXZWVrWWVhcn0gZnVuY3Rpb24gb3B0aW9ucy5cbiAqL1xuXG4vKipcbiAqIEBuYW1lIGdldFdlZWtZZWFyXG4gKiBAY2F0ZWdvcnkgV2Vlay1OdW1iZXJpbmcgWWVhciBIZWxwZXJzXG4gKiBAc3VtbWFyeSBHZXQgdGhlIGxvY2FsIHdlZWstbnVtYmVyaW5nIHllYXIgb2YgdGhlIGdpdmVuIGRhdGUuXG4gKlxuICogQGRlc2NyaXB0aW9uXG4gKiBHZXQgdGhlIGxvY2FsIHdlZWstbnVtYmVyaW5nIHllYXIgb2YgdGhlIGdpdmVuIGRhdGUuXG4gKiBUaGUgZXhhY3QgY2FsY3VsYXRpb24gZGVwZW5kcyBvbiB0aGUgdmFsdWVzIG9mXG4gKiBgb3B0aW9ucy53ZWVrU3RhcnRzT25gICh3aGljaCBpcyB0aGUgaW5kZXggb2YgdGhlIGZpcnN0IGRheSBvZiB0aGUgd2VlaylcbiAqIGFuZCBgb3B0aW9ucy5maXJzdFdlZWtDb250YWluc0RhdGVgICh3aGljaCBpcyB0aGUgZGF5IG9mIEphbnVhcnksIHdoaWNoIGlzIGFsd2F5cyBpblxuICogdGhlIGZpcnN0IHdlZWsgb2YgdGhlIHdlZWstbnVtYmVyaW5nIHllYXIpXG4gKlxuICogV2VlayBudW1iZXJpbmc6IGh0dHBzOi8vZW4ud2lraXBlZGlhLm9yZy93aWtpL1dlZWsjVGhlX0lTT193ZWVrX2RhdGVfc3lzdGVtXG4gKlxuICogQHBhcmFtIGRhdGUgLSBUaGUgZ2l2ZW4gZGF0ZVxuICogQHBhcmFtIG9wdGlvbnMgLSBBbiBvYmplY3Qgd2l0aCBvcHRpb25zLlxuICpcbiAqIEByZXR1cm5zIFRoZSBsb2NhbCB3ZWVrLW51bWJlcmluZyB5ZWFyXG4gKlxuICogQGV4YW1wbGVcbiAqIC8vIFdoaWNoIHdlZWsgbnVtYmVyaW5nIHllYXIgaXMgMjYgRGVjZW1iZXIgMjAwNCB3aXRoIHRoZSBkZWZhdWx0IHNldHRpbmdzP1xuICogY29uc3QgcmVzdWx0ID0gZ2V0V2Vla1llYXIobmV3IERhdGUoMjAwNCwgMTEsIDI2KSlcbiAqIC8vPT4gMjAwNVxuICpcbiAqIEBleGFtcGxlXG4gKiAvLyBXaGljaCB3ZWVrIG51bWJlcmluZyB5ZWFyIGlzIDI2IERlY2VtYmVyIDIwMDQgaWYgd2VlayBzdGFydHMgb24gU2F0dXJkYXk/XG4gKiBjb25zdCByZXN1bHQgPSBnZXRXZWVrWWVhcihuZXcgRGF0ZSgyMDA0LCAxMSwgMjYpLCB7IHdlZWtTdGFydHNPbjogNiB9KVxuICogLy89PiAyMDA0XG4gKlxuICogQGV4YW1wbGVcbiAqIC8vIFdoaWNoIHdlZWsgbnVtYmVyaW5nIHllYXIgaXMgMjYgRGVjZW1iZXIgMjAwNCBpZiB0aGUgZmlyc3Qgd2VlayBjb250YWlucyA0IEphbnVhcnk/XG4gKiBjb25zdCByZXN1bHQgPSBnZXRXZWVrWWVhcihuZXcgRGF0ZSgyMDA0LCAxMSwgMjYpLCB7IGZpcnN0V2Vla0NvbnRhaW5zRGF0ZTogNCB9KVxuICogLy89PiAyMDA0XG4gKi9cbmV4cG9ydCBmdW5jdGlvbiBnZXRXZWVrWWVhcihkYXRlLCBvcHRpb25zKSB7XG4gIGNvbnN0IF9kYXRlID0gdG9EYXRlKGRhdGUsIG9wdGlvbnM/LmluKTtcbiAgY29uc3QgeWVhciA9IF9kYXRlLmdldEZ1bGxZZWFyKCk7XG5cbiAgY29uc3QgZGVmYXVsdE9wdGlvbnMgPSBnZXREZWZhdWx0T3B0aW9ucygpO1xuICBjb25zdCBmaXJzdFdlZWtDb250YWluc0RhdGUgPVxuICAgIG9wdGlvbnM/LmZpcnN0V2Vla0NvbnRhaW5zRGF0ZSA/P1xuICAgIG9wdGlvbnM/LmxvY2FsZT8ub3B0aW9ucz8uZmlyc3RXZWVrQ29udGFpbnNEYXRlID8/XG4gICAgZGVmYXVsdE9wdGlvbnMuZmlyc3RXZWVrQ29udGFpbnNEYXRlID8/XG4gICAgZGVmYXVsdE9wdGlvbnMubG9jYWxlPy5vcHRpb25zPy5maXJzdFdlZWtDb250YWluc0RhdGUgPz9cbiAgICAxO1xuXG4gIGNvbnN0IGZpcnN0V2Vla09mTmV4dFllYXIgPSBjb25zdHJ1Y3RGcm9tKG9wdGlvbnM/LmluIHx8IGRhdGUsIDApO1xuICBmaXJzdFdlZWtPZk5leHRZZWFyLnNldEZ1bGxZZWFyKHllYXIgKyAxLCAwLCBmaXJzdFdlZWtDb250YWluc0RhdGUpO1xuICBmaXJzdFdlZWtPZk5leHRZZWFyLnNldEhvdXJzKDAsIDAsIDAsIDApO1xuICBjb25zdCBzdGFydE9mTmV4dFllYXIgPSBzdGFydE9mV2VlayhmaXJzdFdlZWtPZk5leHRZZWFyLCBvcHRpb25zKTtcblxuICBjb25zdCBmaXJzdFdlZWtPZlRoaXNZZWFyID0gY29uc3RydWN0RnJvbShvcHRpb25zPy5pbiB8fCBkYXRlLCAwKTtcbiAgZmlyc3RXZWVrT2ZUaGlzWWVhci5zZXRGdWxsWWVhcih5ZWFyLCAwLCBmaXJzdFdlZWtDb250YWluc0RhdGUpO1xuICBmaXJzdFdlZWtPZlRoaXNZZWFyLnNldEhvdXJzKDAsIDAsIDAsIDApO1xuICBjb25zdCBzdGFydE9mVGhpc1llYXIgPSBzdGFydE9mV2VlayhmaXJzdFdlZWtPZlRoaXNZZWFyLCBvcHRpb25zKTtcblxuICBpZiAoK19kYXRlID49ICtzdGFydE9mTmV4dFllYXIpIHtcbiAgICByZXR1cm4geWVhciArIDE7XG4gIH0gZWxzZSBpZiAoK19kYXRlID49ICtzdGFydE9mVGhpc1llYXIpIHtcbiAgICByZXR1cm4geWVhcjtcbiAgfSBlbHNlIHtcbiAgICByZXR1cm4geWVhciAtIDE7XG4gIH1cbn1cblxuLy8gRmFsbGJhY2sgZm9yIG1vZHVsYXJpemVkIGltcG9ydHM6XG5leHBvcnQgZGVmYXVsdCBnZXRXZWVrWWVhcjtcbiIsImltcG9ydCB7IGdldERlZmF1bHRPcHRpb25zIH0gZnJvbSBcIi4vX2xpYi9kZWZhdWx0T3B0aW9ucy5qc1wiO1xuaW1wb3J0IHsgY29uc3RydWN0RnJvbSB9IGZyb20gXCIuL2NvbnN0cnVjdEZyb20uanNcIjtcbmltcG9ydCB7IGdldFdlZWtZZWFyIH0gZnJvbSBcIi4vZ2V0V2Vla1llYXIuanNcIjtcbmltcG9ydCB7IHN0YXJ0T2ZXZWVrIH0gZnJvbSBcIi4vc3RhcnRPZldlZWsuanNcIjtcblxuLyoqXG4gKiBUaGUge0BsaW5rIHN0YXJ0T2ZXZWVrWWVhcn0gZnVuY3Rpb24gb3B0aW9ucy5cbiAqL1xuXG4vKipcbiAqIEBuYW1lIHN0YXJ0T2ZXZWVrWWVhclxuICogQGNhdGVnb3J5IFdlZWstTnVtYmVyaW5nIFllYXIgSGVscGVyc1xuICogQHN1bW1hcnkgUmV0dXJuIHRoZSBzdGFydCBvZiBhIGxvY2FsIHdlZWstbnVtYmVyaW5nIHllYXIgZm9yIHRoZSBnaXZlbiBkYXRlLlxuICpcbiAqIEBkZXNjcmlwdGlvblxuICogUmV0dXJuIHRoZSBzdGFydCBvZiBhIGxvY2FsIHdlZWstbnVtYmVyaW5nIHllYXIuXG4gKiBUaGUgZXhhY3QgY2FsY3VsYXRpb24gZGVwZW5kcyBvbiB0aGUgdmFsdWVzIG9mXG4gKiBgb3B0aW9ucy53ZWVrU3RhcnRzT25gICh3aGljaCBpcyB0aGUgaW5kZXggb2YgdGhlIGZpcnN0IGRheSBvZiB0aGUgd2VlaylcbiAqIGFuZCBgb3B0aW9ucy5maXJzdFdlZWtDb250YWluc0RhdGVgICh3aGljaCBpcyB0aGUgZGF5IG9mIEphbnVhcnksIHdoaWNoIGlzIGFsd2F5cyBpblxuICogdGhlIGZpcnN0IHdlZWsgb2YgdGhlIHdlZWstbnVtYmVyaW5nIHllYXIpXG4gKlxuICogV2VlayBudW1iZXJpbmc6IGh0dHBzOi8vZW4ud2lraXBlZGlhLm9yZy93aWtpL1dlZWsjVGhlX0lTT193ZWVrX2RhdGVfc3lzdGVtXG4gKlxuICogQHR5cGVQYXJhbSBEYXRlVHlwZSAtIFRoZSBgRGF0ZWAgdHlwZSwgdGhlIGZ1bmN0aW9uIG9wZXJhdGVzIG9uLiBHZXRzIGluZmVycmVkIGZyb20gcGFzc2VkIGFyZ3VtZW50cy4gQWxsb3dzIHRvIHVzZSBleHRlbnNpb25zIGxpa2UgW2BVVENEYXRlYF0oaHR0cHM6Ly9naXRodWIuY29tL2RhdGUtZm5zL3V0YykuXG4gKiBAdHlwZVBhcmFtIFJlc3VsdERhdGUgLSBUaGUgcmVzdWx0IGBEYXRlYCB0eXBlLlxuICpcbiAqIEBwYXJhbSBkYXRlIC0gVGhlIG9yaWdpbmFsIGRhdGVcbiAqIEBwYXJhbSBvcHRpb25zIC0gQW4gb2JqZWN0IHdpdGggb3B0aW9uc1xuICpcbiAqIEByZXR1cm5zIFRoZSBzdGFydCBvZiBhIHdlZWstbnVtYmVyaW5nIHllYXJcbiAqXG4gKiBAZXhhbXBsZVxuICogLy8gVGhlIHN0YXJ0IG9mIGFuIGEgd2Vlay1udW1iZXJpbmcgeWVhciBmb3IgMiBKdWx5IDIwMDUgd2l0aCBkZWZhdWx0IHNldHRpbmdzOlxuICogY29uc3QgcmVzdWx0ID0gc3RhcnRPZldlZWtZZWFyKG5ldyBEYXRlKDIwMDUsIDYsIDIpKVxuICogLy89PiBTdW4gRGVjIDI2IDIwMDQgMDA6MDA6MDBcbiAqXG4gKiBAZXhhbXBsZVxuICogLy8gVGhlIHN0YXJ0IG9mIGEgd2Vlay1udW1iZXJpbmcgeWVhciBmb3IgMiBKdWx5IDIwMDVcbiAqIC8vIGlmIE1vbmRheSBpcyB0aGUgZmlyc3QgZGF5IG9mIHdlZWtcbiAqIC8vIGFuZCA0IEphbnVhcnkgaXMgYWx3YXlzIGluIHRoZSBmaXJzdCB3ZWVrIG9mIHRoZSB5ZWFyOlxuICogY29uc3QgcmVzdWx0ID0gc3RhcnRPZldlZWtZZWFyKG5ldyBEYXRlKDIwMDUsIDYsIDIpLCB7XG4gKiAgIHdlZWtTdGFydHNPbjogMSxcbiAqICAgZmlyc3RXZWVrQ29udGFpbnNEYXRlOiA0XG4gKiB9KVxuICogLy89PiBNb24gSmFuIDAzIDIwMDUgMDA6MDA6MDBcbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIHN0YXJ0T2ZXZWVrWWVhcihkYXRlLCBvcHRpb25zKSB7XG4gIGNvbnN0IGRlZmF1bHRPcHRpb25zID0gZ2V0RGVmYXVsdE9wdGlvbnMoKTtcbiAgY29uc3QgZmlyc3RXZWVrQ29udGFpbnNEYXRlID1cbiAgICBvcHRpb25zPy5maXJzdFdlZWtDb250YWluc0RhdGUgPz9cbiAgICBvcHRpb25zPy5sb2NhbGU/Lm9wdGlvbnM/LmZpcnN0V2Vla0NvbnRhaW5zRGF0ZSA/P1xuICAgIGRlZmF1bHRPcHRpb25zLmZpcnN0V2Vla0NvbnRhaW5zRGF0ZSA/P1xuICAgIGRlZmF1bHRPcHRpb25zLmxvY2FsZT8ub3B0aW9ucz8uZmlyc3RXZWVrQ29udGFpbnNEYXRlID8/XG4gICAgMTtcblxuICBjb25zdCB5ZWFyID0gZ2V0V2Vla1llYXIoZGF0ZSwgb3B0aW9ucyk7XG4gIGNvbnN0IGZpcnN0V2VlayA9IGNvbnN0cnVjdEZyb20ob3B0aW9ucz8uaW4gfHwgZGF0ZSwgMCk7XG4gIGZpcnN0V2Vlay5zZXRGdWxsWWVhcih5ZWFyLCAwLCBmaXJzdFdlZWtDb250YWluc0RhdGUpO1xuICBmaXJzdFdlZWsuc2V0SG91cnMoMCwgMCwgMCwgMCk7XG4gIGNvbnN0IF9kYXRlID0gc3RhcnRPZldlZWsoZmlyc3RXZWVrLCBvcHRpb25zKTtcbiAgcmV0dXJuIF9kYXRlO1xufVxuXG4vLyBGYWxsYmFjayBmb3IgbW9kdWxhcml6ZWQgaW1wb3J0czpcbmV4cG9ydCBkZWZhdWx0IHN0YXJ0T2ZXZWVrWWVhcjtcbiIsImltcG9ydCB7IG1pbGxpc2Vjb25kc0luV2VlayB9IGZyb20gXCIuL2NvbnN0YW50cy5qc1wiO1xuaW1wb3J0IHsgc3RhcnRPZldlZWsgfSBmcm9tIFwiLi9zdGFydE9mV2Vlay5qc1wiO1xuaW1wb3J0IHsgc3RhcnRPZldlZWtZZWFyIH0gZnJvbSBcIi4vc3RhcnRPZldlZWtZZWFyLmpzXCI7XG5pbXBvcnQgeyB0b0RhdGUgfSBmcm9tIFwiLi90b0RhdGUuanNcIjtcblxuLyoqXG4gKiBUaGUge0BsaW5rIGdldFdlZWt9IGZ1bmN0aW9uIG9wdGlvbnMuXG4gKi9cblxuLyoqXG4gKiBAbmFtZSBnZXRXZWVrXG4gKiBAY2F0ZWdvcnkgV2VlayBIZWxwZXJzXG4gKiBAc3VtbWFyeSBHZXQgdGhlIGxvY2FsIHdlZWsgaW5kZXggb2YgdGhlIGdpdmVuIGRhdGUuXG4gKlxuICogQGRlc2NyaXB0aW9uXG4gKiBHZXQgdGhlIGxvY2FsIHdlZWsgaW5kZXggb2YgdGhlIGdpdmVuIGRhdGUuXG4gKiBUaGUgZXhhY3QgY2FsY3VsYXRpb24gZGVwZW5kcyBvbiB0aGUgdmFsdWVzIG9mXG4gKiBgb3B0aW9ucy53ZWVrU3RhcnRzT25gICh3aGljaCBpcyB0aGUgaW5kZXggb2YgdGhlIGZpcnN0IGRheSBvZiB0aGUgd2VlaylcbiAqIGFuZCBgb3B0aW9ucy5maXJzdFdlZWtDb250YWluc0RhdGVgICh3aGljaCBpcyB0aGUgZGF5IG9mIEphbnVhcnksIHdoaWNoIGlzIGFsd2F5cyBpblxuICogdGhlIGZpcnN0IHdlZWsgb2YgdGhlIHdlZWstbnVtYmVyaW5nIHllYXIpXG4gKlxuICogV2VlayBudW1iZXJpbmc6IGh0dHBzOi8vZW4ud2lraXBlZGlhLm9yZy93aWtpL1dlZWsjVGhlX0lTT193ZWVrX2RhdGVfc3lzdGVtXG4gKlxuICogQHBhcmFtIGRhdGUgLSBUaGUgZ2l2ZW4gZGF0ZVxuICogQHBhcmFtIG9wdGlvbnMgLSBBbiBvYmplY3Qgd2l0aCBvcHRpb25zXG4gKlxuICogQHJldHVybnMgVGhlIHdlZWtcbiAqXG4gKiBAZXhhbXBsZVxuICogLy8gV2hpY2ggd2VlayBvZiB0aGUgbG9jYWwgd2VlayBudW1iZXJpbmcgeWVhciBpcyAyIEphbnVhcnkgMjAwNSB3aXRoIGRlZmF1bHQgb3B0aW9ucz9cbiAqIGNvbnN0IHJlc3VsdCA9IGdldFdlZWsobmV3IERhdGUoMjAwNSwgMCwgMikpXG4gKiAvLz0+IDJcbiAqXG4gKiBAZXhhbXBsZVxuICogLy8gV2hpY2ggd2VlayBvZiB0aGUgbG9jYWwgd2VlayBudW1iZXJpbmcgeWVhciBpcyAyIEphbnVhcnkgMjAwNSxcbiAqIC8vIGlmIE1vbmRheSBpcyB0aGUgZmlyc3QgZGF5IG9mIHRoZSB3ZWVrLFxuICogLy8gYW5kIHRoZSBmaXJzdCB3ZWVrIG9mIHRoZSB5ZWFyIGFsd2F5cyBjb250YWlucyA0IEphbnVhcnk/XG4gKiBjb25zdCByZXN1bHQgPSBnZXRXZWVrKG5ldyBEYXRlKDIwMDUsIDAsIDIpLCB7XG4gKiAgIHdlZWtTdGFydHNPbjogMSxcbiAqICAgZmlyc3RXZWVrQ29udGFpbnNEYXRlOiA0XG4gKiB9KVxuICogLy89PiA1M1xuICovXG5leHBvcnQgZnVuY3Rpb24gZ2V0V2VlayhkYXRlLCBvcHRpb25zKSB7XG4gIGNvbnN0IF9kYXRlID0gdG9EYXRlKGRhdGUsIG9wdGlvbnM/LmluKTtcbiAgY29uc3QgZGlmZiA9ICtzdGFydE9mV2VlayhfZGF0ZSwgb3B0aW9ucykgLSArc3RhcnRPZldlZWtZZWFyKF9kYXRlLCBvcHRpb25zKTtcblxuICAvLyBSb3VuZCB0aGUgbnVtYmVyIG9mIHdlZWtzIHRvIHRoZSBuZWFyZXN0IGludGVnZXIgYmVjYXVzZSB0aGUgbnVtYmVyIG9mXG4gIC8vIG1pbGxpc2Vjb25kcyBpbiBhIHdlZWsgaXMgbm90IGNvbnN0YW50IChlLmcuIGl0J3MgZGlmZmVyZW50IGluIHRoZSB3ZWVrIG9mXG4gIC8vIHRoZSBkYXlsaWdodCBzYXZpbmcgdGltZSBjbG9jayBzaGlmdCkuXG4gIHJldHVybiBNYXRoLnJvdW5kKGRpZmYgLyBtaWxsaXNlY29uZHNJbldlZWspICsgMTtcbn1cblxuLy8gRmFsbGJhY2sgZm9yIG1vZHVsYXJpemVkIGltcG9ydHM6XG5leHBvcnQgZGVmYXVsdCBnZXRXZWVrO1xuIiwiaW1wb3J0IHsgZWFjaFdlZWtPZkludGVydmFsLCBlbmRPZk1vbnRoLCBnZXRXZWVrLCBzdGFydE9mTW9udGggfSBmcm9tICdkYXRlLWZucyc7XG5pbXBvcnQge1xuICB0eXBlIENTU1Jlc3VsdEdyb3VwLFxuICBodG1sLFxuICBpc1NlcnZlcixcbiAgTGl0RWxlbWVudCxcbiAgbm90aGluZyxcbiAgdHlwZSBQcm9wZXJ0eVZhbHVlcyxcbiAgdHlwZSBUZW1wbGF0ZVJlc3VsdCxcbn0gZnJvbSAnbGl0JztcbmltcG9ydCB7IGN1c3RvbUVsZW1lbnQsIHByb3BlcnR5LCBzdGF0ZSB9IGZyb20gJ2xpdC9kZWNvcmF0b3JzLmpzJztcbmltcG9ydCB7IGNsYXNzTWFwIH0gZnJvbSAnbGl0L2RpcmVjdGl2ZXMvY2xhc3MtbWFwLmpzJztcblxuaW1wb3J0IHsgaXNBcnJvd0tleU9yUGFnZUtleXNQcmVzc2VkIH0gZnJvbSAnLi4vLi4vY29yZS9hMTF5LnRzJztcbmltcG9ydCB7IHJlYWRDb25maWcgfSBmcm9tICcuLi8uLi9jb3JlL2NvbmZpZy50cyc7XG5pbXBvcnQge1xuICBTYmJMYW5ndWFnZUNvbnRyb2xsZXIsXG4gIFNiYk1lZGlhTWF0Y2hlckNvbnRyb2xsZXIsXG4gIFNiYk1lZGlhUXVlcnlCcmVha3BvaW50TGFyZ2VBbmRBYm92ZSxcbn0gZnJvbSAnLi4vLi4vY29yZS9jb250cm9sbGVycy50cyc7XG5pbXBvcnQgdHlwZSB7IERhdGVBZGFwdGVyIH0gZnJvbSAnLi4vLi4vY29yZS9kYXRldGltZS50cyc7XG5pbXBvcnQge1xuICBEQVlTX1BFUl9ST1csXG4gIGRlZmF1bHREYXRlQWRhcHRlcixcbiAgTU9OVEhTX1BFUl9QQUdFLFxuICBNT05USFNfUEVSX1JPVyxcbiAgWUVBUlNfUEVSX1BBR0UsXG4gIFlFQVJTX1BFUl9ST1csXG59IGZyb20gJy4uLy4uL2NvcmUvZGF0ZXRpbWUudHMnO1xuaW1wb3J0IHsgZm9yY2VUeXBlLCBoYW5kbGVEaXN0aW5jdENoYW5nZSwgcGxhaW5EYXRlIH0gZnJvbSAnLi4vLi4vY29yZS9kZWNvcmF0b3JzLnRzJztcbmltcG9ydCB7XG4gIGkxOG5DYWxlbmRhckRhdGVTZWxlY3Rpb24sXG4gIGkxOG5DYWxlbmRhcldlZWtOdW1iZXIsXG4gIGkxOG5OZXh0TW9udGgsXG4gIGkxOG5OZXh0WWVhcixcbiAgaTE4bk5leHRZZWFyUmFuZ2UsXG4gIGkxOG5QcmV2aW91c01vbnRoLFxuICBpMThuUHJldmlvdXNZZWFyLFxuICBpMThuUHJldmlvdXNZZWFyUmFuZ2UsXG4gIGkxOG5ZZWFyTW9udGhTZWxlY3Rpb24sXG59IGZyb20gJy4uLy4uL2NvcmUvaTE4bi50cyc7XG5pbXBvcnQgdHlwZSB7IFNiYk9yaWVudGF0aW9uIH0gZnJvbSAnLi4vLi4vY29yZS9pbnRlcmZhY2VzLnRzJztcbmltcG9ydCB7IFNiYkVsZW1lbnRJbnRlcm5hbHNNaXhpbiwgU2JiSHlkcmF0aW9uTWl4aW4gfSBmcm9tICcuLi8uLi9jb3JlL21peGlucy50cyc7XG5pbXBvcnQgeyBib3hTaXppbmdTdHlsZXMgfSBmcm9tICcuLi8uLi9jb3JlL3N0eWxlcy50cyc7XG5pbXBvcnQgdHlwZSB7IFNiYkNhbGVuZGFyRGF5RWxlbWVudCB9IGZyb20gJy4uL2NhbGVuZGFyLWRheS9jYWxlbmRhci1kYXkuY29tcG9uZW50LnRzJztcblxuaW1wb3J0IHN0eWxlIGZyb20gJy4vY2FsZW5kYXIuc2Nzcz9saXQmaW5saW5lJztcblxuaW1wb3J0ICcuLi8uLi9idXR0b24vc2Vjb25kYXJ5LWJ1dHRvbi50cyc7XG5pbXBvcnQgJy4uLy4uL2ljb24udHMnO1xuaW1wb3J0ICcuLi8uLi9zY3JlZW4tcmVhZGVyLW9ubHkudHMnO1xuaW1wb3J0ICcuLi9jYWxlbmRhci1kYXkvY2FsZW5kYXItZGF5LmNvbXBvbmVudC50cyc7XG5cbmV4cG9ydCBjbGFzcyBTYmJNb250aENoYW5nZUV2ZW50IGV4dGVuZHMgRXZlbnQge1xuICBwcml2YXRlIHJlYWRvbmx5IF9yYW5nZTogcmVhZG9ubHkgRGF5W107XG5cbiAgcHVibGljIGdldCByYW5nZSgpOiByZWFkb25seSBEYXlbXSB7XG4gICAgcmV0dXJuIHRoaXMuX3JhbmdlO1xuICB9XG5cbiAgcHVibGljIGNvbnN0cnVjdG9yKHJhbmdlOiByZWFkb25seSBEYXlbXSkge1xuICAgIHN1cGVyKCdtb250aGNoYW5nZScsIHsgYnViYmxlczogdHJ1ZSwgY29tcG9zZWQ6IHRydWUgfSk7XG4gICAgdGhpcy5fcmFuZ2UgPSBPYmplY3QuZnJlZXplKHJhbmdlIHx8IFtdKTtcbiAgfVxufVxuXG4vKipcbiAqIFBhcmFtZXRlcnMgbmVlZGVkIGluIHllYXIgYW5kIG1vbnRoIHZpZXdzIHRvIGNvcnJlY3RseSBjYWxjdWxhdGUgdGhlIG5leHQgZWxlbWVudCBpbiBrZXlib2FyZCBuYXZpZ2F0aW9uLlxuICpcbiAqIFRoZSBjZWxsJ3MgaW5kZXggYW5kIHRoZSBlbGVtZW50J3MgaW5kZXggaW4gaXRzIG1vbnRoIC8geWVhciBiYXRjaCBtdXN0IGJlIGRpc3Rpbmd1aXNoZWQ6XG4gKiB0aGUgZmlyc3QgaXMgdGhlIGluZGV4IG9mIHRoZSBlbGVtZW50IGluIHRoZSBhcnJheSBvZiBhbGwgdGhlIHJlbmRlcmVkIGNlbGxzLCB3aGlsZSB0aGUgc2Vjb25kIGlzIHRoZSBpbmRleCBvZiB0aGUgZWxlbWVudCByZWxhdGl2ZSB0byBpdHMgdGFibGUuXG4gKiBJbiBub24td2lkZSBtb2RlLCB0aGUgd3RvIGFyZSB0aGUgc2FtZSwgd2hpbGUgaW4gd2lkZSBtb2RlIHRoZSBjZWxsJ3MgaW5kZXggY2FuIGdvIGZyb20gMCB0byA0NyBmb3IgeWVhcnMgYW5kIGZyb20gMCB0byAyMyBmb3IgbW9udGhzLFxuICogd2hpbGUgdGhlIGVsZW1lbnQgaW5kZXggZ29lcyBmcm9tIDAgdG8sIHJlc3BlY3RpdmVseSwgMjMgYW5kIDExLlxuICovXG5pbnRlcmZhY2UgQ2FsZW5kYXJLZXlib2FyZE5hdmlnYXRpb25Nb250aFllYXJWaWV3c1BhcmFtZXRlcnMge1xuICAvKiogVGhlIGVsZW1lbnQgaW5kZXggd2l0aGluIGl0cyB5ZWFyIG9yIG1vbnRoIHJhbmdlLiAqL1xuICBlbGVtZW50SW5kZXhGb3JXaWRlTW9kZTogbnVtYmVyO1xuICAvKiogSW4gd2lkZSBtb2RlLCB0aGUgaW5kZXggb2YgdGhlIGZpcnN0IGVsZW1lbnQgaW4gdGhlIHNlY29uZCBwYW5lbCwgb3IsIGFsdGVybmF0aXZlbHksIHRoZSBudW1iZXIgb2YgZWxlbWVudHMgaW4gdGhlIGZpcnN0IHBhbmVsLiAqL1xuICBvZmZzZXRGb3JXaWRlTW9kZTogbnVtYmVyO1xuICAvKiogVGhlIGluZGV4IG9mIHRoZSBsYXN0IGVsZW1lbnQgd2l0aGluIHRoZSBlbGVtZW50J3MgbW9udGggKG9yIHllYXIgcmFuZ2UpLiAqL1xuICBsYXN0RWxlbWVudEluZGV4Rm9yV2lkZU1vZGU6IG51bWJlcjtcbiAgLyoqIFRoZSBudW1iZXIgb2YgY2VsbHMgZGlzcGxheWVkIGluIGEgc2luZ2xlIHJvdywgZGVwZW5kaW5nIG9uIHRoZSByZW5kZXJlZCB2aWV3LiAqL1xuICB2ZXJ0aWNhbE9mZnNldDogbnVtYmVyO1xufVxuXG4vKipcbiAqIFBhcmFtZXRlcnMgbmVlZGVkIGluIGRheSB2aWV3IHRvIGNvcnJlY3RseSBjYWxjdWxhdGUgdGhlIG5leHQgZWxlbWVudCBpbiBrZXlib2FyZCBuYXZpZ2F0aW9uLlxuICpcbiAqIEluIG9yaWVudGF0aW9uPSd2ZXJ0aWNhbCcsIGl0J3Mgbm90IHBvc3NpYmxlIHRvIHJlbHkgb24gYW55IGFycmF5L2luZGV4IHRvIGNhbGN1bGF0ZSB0aGUgZWxlbWVudCB0byBuYXZpZ2F0ZSB0byxcbiAqIHNvIGNhbGN1bGF0aW9ucyBvbiBkYXRlcyBtdXN0IGJlIGRvbmUsIHdoaWNoIHNob3VsZCBjb25zaWRlciB2aWV3IGJvdW5kYXJpZXMsIG9mZnNldHMgYW5kIG1vbnRoJ3MgbGVuZ3RoLlxuICovXG5pbnRlcmZhY2UgQ2FsZW5kYXJLZXlib2FyZE5hdmlnYXRpb25EYXlWaWV3UGFyYW1ldGVycyB7XG4gIC8qKiBUaGUgZmlyc3QgZGF5IHJlbmRlcmVkLiAqL1xuICBmaXJzdERheUluVmlldzogc3RyaW5nIHwgbnVsbDtcbiAgLyoqIFRoZSBsYXN0IGRheSByZW5kZXJlZC4gSXQgZGVwZW5kcyBvbiB0aGUgJ3dpZGUnIHZhbHVlLiAqL1xuICBsYXN0RGF5SW5WaWV3OiBzdHJpbmcgfCBudWxsO1xuICAvKiogVGhlIG9mZnNldCBmcm9tIHRoZSBmaXJzdCBkYXkgb2YgdGhlIHdlZWsgKE1vbmRheSkgb2YgdGhlIGZpcnN0IHJlbmRlcmVkIG1vbnRoLiAqL1xuICBmaXJzdE1vbnRoT2Zmc2V0OiBudW1iZXI7XG4gIC8qKiBUaGUgbnVtYmVyIG9mIGRheXMgaW4gdGhlIGZpcnN0IHJlbmRlcmVkIG1vbnRoLiAqL1xuICBmaXJzdE1vbnRoTGVuZ3RoOiBudW1iZXI7XG4gIC8qKiBUaGUgb2Zmc2V0IGZyb20gdGhlIGZpcnN0IGRheSBvZiB0aGUgd2VlayAoTW9uZGF5KSBvZiB0aGUgc2Vjb25kIHJlbmRlcmVkIG1vbnRoLiBJZiB3aWRlIGlzIGZhbHNlLCBpdCdzIGVxdWFsIHRvIHplcm8uICovXG4gIHNlY29uZE1vbnRoT2Zmc2V0OiBudW1iZXI7XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgRGF5PFQgPSBEYXRlPiB7XG4gIC8qKiBEYXRlIGFzIElTTyBzdHJpbmcuICovXG4gIHZhbHVlOiBzdHJpbmc7XG4gIGRheVZhbHVlOiBzdHJpbmc7XG4gIG1vbnRoVmFsdWU6IHN0cmluZztcbiAgeWVhclZhbHVlOiBzdHJpbmc7XG4gIGRhdGVWYWx1ZTogVDtcbiAgd2Vla1ZhbHVlOiBudW1iZXI7XG4gIHdlZWtEYXlWYWx1ZTogbnVtYmVyO1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIE1vbnRoIHtcbiAgdmFsdWU6IHN0cmluZztcbiAgbG9uZ1ZhbHVlOiBzdHJpbmc7XG4gIG1vbnRoVmFsdWU6IG51bWJlcjtcbn1cblxuZXhwb3J0IGludGVyZmFjZSBXZWVrZGF5IHtcbiAgbG9uZzogc3RyaW5nO1xuICBuYXJyb3c6IHN0cmluZztcbn1cblxuZXhwb3J0IHR5cGUgQ2FsZW5kYXJWaWV3ID0gJ2RheScgfCAnbW9udGgnIHwgJ3llYXInO1xuXG4vKipcbiAqIEl0IGRpc3BsYXlzIGEgY2FsZW5kYXIgd2hpY2ggYWxsb3dzIGNob29zaW5nIGEgZGF0ZS5cbiAqXG4gKiBAc2xvdCAtIFVzZSB0aGUgdW5uYW1lZCBzbG90IHRvIGFkZCBjdXN0b21pemVkIGBzYmItY2FsZW5kYXItZGF5YCBlbGVtZW50cy5cbiAqL1xuZXhwb3J0XG5AY3VzdG9tRWxlbWVudCgnc2JiLWNhbGVuZGFyJylcbmNsYXNzIFNiYkNhbGVuZGFyRWxlbWVudDxUIGV4dGVuZHMgRGF0ZSA9IERhdGU+IGV4dGVuZHMgU2JiSHlkcmF0aW9uTWl4aW4oXG4gIFNiYkVsZW1lbnRJbnRlcm5hbHNNaXhpbihMaXRFbGVtZW50KSxcbikge1xuICBwdWJsaWMgc3RhdGljIG92ZXJyaWRlIHN0eWxlczogQ1NTUmVzdWx0R3JvdXAgPSBbYm94U2l6aW5nU3R5bGVzLCBzdHlsZV07XG4gIHB1YmxpYyBzdGF0aWMgcmVhZG9ubHkgZXZlbnRzID0ge1xuICAgIGRhdGVzZWxlY3RlZDogJ2RhdGVzZWxlY3RlZCcsXG4gICAgbW9udGhjaGFuZ2U6ICdtb250aGNoYW5nZScsXG4gIH0gYXMgY29uc3Q7XG5cbiAgLyoqIElmIHNldCB0byB0cnVlLCB0d28gbW9udGhzIGFyZSBkaXNwbGF5ZWQgKi9cbiAgQGZvcmNlVHlwZSgpXG4gIEBwcm9wZXJ0eSh7IHR5cGU6IEJvb2xlYW4gfSlcbiAgcHVibGljIGFjY2Vzc29yIHdpZGU6IGJvb2xlYW4gPSBmYWxzZTtcblxuICAvKiogVGhlIGluaXRpYWwgdmlldyBvZiB0aGUgY2FsZW5kYXIgd2hpY2ggc2hvdWxkIGJlIGRpc3BsYXllZCBvbiBvcGVuaW5nLiAqL1xuICBAcHJvcGVydHkoKSBwdWJsaWMgYWNjZXNzb3IgdmlldzogQ2FsZW5kYXJWaWV3ID0gJ2RheSc7XG5cbiAgLyoqXG4gICAqIFRoZSBtaW5pbXVtIHZhbGlkIGRhdGUuIEFjY2VwdHMgYSBkYXRlIG9iamVjdCBvciBudWxsLlxuICAgKiBBY2NlcHRzIGFuIElTTzg2MDEgZm9ybWF0dGVkIHN0cmluZyAoZS5nLiAyMDI0LTEyLTI0KSBhcyBhdHRyaWJ1dGUuXG4gICAqL1xuICBAcGxhaW5EYXRlKClcbiAgQHByb3BlcnR5KClcbiAgcHVibGljIGFjY2Vzc29yIG1pbjogVCB8IG51bGwgPSBudWxsO1xuXG4gIC8qKlxuICAgKiBUaGUgbWF4aW11bSB2YWxpZCBkYXRlLiBBY2NlcHRzIGEgZGF0ZSBvYmplY3Qgb3IgbnVsbC5cbiAgICogQWNjZXB0cyBhbiBJU084NjAxIGZvcm1hdHRlZCBzdHJpbmcgKGUuZy4gMjAyNC0xMi0yNCkgYXMgYXR0cmlidXRlLlxuICAgKi9cbiAgQHBsYWluRGF0ZSgpXG4gIEBwcm9wZXJ0eSgpXG4gIHB1YmxpYyBhY2Nlc3NvciBtYXg6IFQgfCBudWxsID0gbnVsbDtcblxuICAvKiogV2hldGhlciB0aGUgY2FsZW5kYXIgYWxsb3dzIGZvciBtdWx0aXBsZSBkYXRlIHNlbGVjdGlvbi4gKi9cbiAgQGZvcmNlVHlwZSgpXG4gIEBoYW5kbGVEaXN0aW5jdENoYW5nZSgoZTogU2JiQ2FsZW5kYXJFbGVtZW50PFQ+LCBuZXdWYWx1ZTogYm9vbGVhbikgPT5cbiAgICBlLl9vbk11bHRpcGxlQ2hhbmdlZChuZXdWYWx1ZSksXG4gIClcbiAgQHByb3BlcnR5KHsgdHlwZTogQm9vbGVhbiB9KVxuICBwdWJsaWMgYWNjZXNzb3IgbXVsdGlwbGU6IGJvb2xlYW4gPSBmYWxzZTtcblxuICAvKipcbiAgICogVGhlIHNlbGVjdGVkIGRhdGU6IGFjY2VwdHMgYSBkYXRlIG9iamVjdCwgb3IsIGlmIGBtdWx0aXBsZWAsIGFuIGFycmF5IG9mIGRhdGVzLlxuICAgKi9cbiAgQHByb3BlcnR5KClcbiAgcHVibGljIHNldCBzZWxlY3RlZCh2YWx1ZTogVCB8IFRbXSB8IG51bGwpIHtcbiAgICBpZiAoQXJyYXkuaXNBcnJheSh2YWx1ZSkpIHtcbiAgICAgIHRoaXMuX3NlbGVjdGVkID0gdmFsdWVcbiAgICAgICAgLm1hcCgoZGF0ZUxpa2U6IFQpID0+XG4gICAgICAgICAgdGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0VmFsaWREYXRlT3JOdWxsKHRoaXMuX2RhdGVBZGFwdGVyLmRlc2VyaWFsaXplKGRhdGVMaWtlKSksXG4gICAgICAgIClcbiAgICAgICAgLmZpbHRlcigoZGF0ZTogVCB8IG51bGwpOiBkYXRlIGlzIFQgPT4gZGF0ZSAhPT0gbnVsbClcbiAgICAgICAgLmZpbHRlcihcbiAgICAgICAgICAoZGF0ZTogVCkgPT5cbiAgICAgICAgICAgICF0aGlzLl9pc0RheUluUmFuZ2UodGhpcy5fZGF0ZUFkYXB0ZXIudG9Jc284NjAxKGRhdGUpKSB8fCB0aGlzLl9kYXRlRmlsdGVyKGRhdGUpLFxuICAgICAgICApO1xuICAgIH0gZWxzZSB7XG4gICAgICBjb25zdCBzZWxlY3RlZERhdGUgPSB0aGlzLl9kYXRlQWRhcHRlci5nZXRWYWxpZERhdGVPck51bGwoXG4gICAgICAgIHRoaXMuX2RhdGVBZGFwdGVyLmRlc2VyaWFsaXplKHZhbHVlKSxcbiAgICAgICk7XG4gICAgICBpZiAoXG4gICAgICAgICEhc2VsZWN0ZWREYXRlICYmXG4gICAgICAgICghdGhpcy5faXNEYXlJblJhbmdlKHRoaXMuX2RhdGVBZGFwdGVyLnRvSXNvODYwMShzZWxlY3RlZERhdGUpKSB8fFxuICAgICAgICAgIHRoaXMuX2RhdGVGaWx0ZXIoc2VsZWN0ZWREYXRlKSlcbiAgICAgICkge1xuICAgICAgICB0aGlzLl9zZWxlY3RlZCA9IHNlbGVjdGVkRGF0ZTtcbiAgICAgIH0gZWxzZSB7XG4gICAgICAgIHRoaXMuX3NlbGVjdGVkID0gbnVsbDtcbiAgICAgIH1cbiAgICB9XG4gIH1cbiAgcHVibGljIGdldCBzZWxlY3RlZCgpOiBUIHwgVFtdIHwgbnVsbCB7XG4gICAgcmV0dXJuIHRoaXMuX3NlbGVjdGVkO1xuICB9XG4gIEBzdGF0ZSgpIHByaXZhdGUgYWNjZXNzb3IgX3NlbGVjdGVkOiBUIHwgVFtdIHwgbnVsbCA9IG51bGw7XG5cbiAgLyoqIEEgZnVuY3Rpb24gdXNlZCB0byBmaWx0ZXIgb3V0IGRhdGVzLiAqL1xuICBAcHJvcGVydHkoeyBhdHRyaWJ1dGU6ICdkYXRlLWZpbHRlcicgfSlcbiAgcHVibGljIGFjY2Vzc29yIGRhdGVGaWx0ZXI6ICgoZGF0ZTogVCB8IG51bGwpID0+IGJvb2xlYW4pIHwgbnVsbCA9IG51bGw7XG5cbiAgLyoqIFRoZSBvcmllbnRhdGlvbiBvZiBkYXlzIGluIHRoZSBjYWxlbmRhci4gKi9cbiAgQHByb3BlcnR5KHsgcmVmbGVjdDogdHJ1ZSB9KSBwdWJsaWMgYWNjZXNzb3Igb3JpZW50YXRpb246IFNiYk9yaWVudGF0aW9uID0gJ2hvcml6b250YWwnO1xuXG4gIC8qKiBXaGV0aGVyIGl0IGhhcyB0byBkaXNwbGF5IHRoZSB3ZWVrIG51bWJlcnMgaW4gYWRkaXRpb24gdG8gd2VlayBkYXlzLiAqL1xuICBAZm9yY2VUeXBlKClcbiAgQHByb3BlcnR5KHsgYXR0cmlidXRlOiAnd2Vlay1udW1iZXJzJywgdHlwZTogQm9vbGVhbiB9KVxuICBwdWJsaWMgYWNjZXNzb3Igd2Vla051bWJlcnM6IGJvb2xlYW4gPSBmYWxzZTtcblxuICBwcml2YXRlIF9kYXRlQWRhcHRlcjogRGF0ZUFkYXB0ZXI8VD4gPSByZWFkQ29uZmlnKCkuZGF0ZXRpbWU/LmRhdGVBZGFwdGVyID8/IGRlZmF1bHREYXRlQWRhcHRlcjtcblxuICAvKiogVGhlIGN1cnJlbnRseSBhY3RpdmUgZGF0ZS4gKi9cbiAgQHN0YXRlKCkgcHJpdmF0ZSBhY2Nlc3NvciBfYWN0aXZlRGF0ZTogVCA9IHRoaXMuX2RhdGVBZGFwdGVyLnRvZGF5KCk7XG5cbiAgLyoqIFRoZSBjdXJyZW50IHdpZGUgcHJvcGVydHkgY29uc2lkZXJpbmcgcHJvcGVydHkgdmFsdWUgYW5kIGJyZWFrcG9pbnRzLiBGcm9tIHplcm8gdG8gc21hbGwgYHdpZGVgIGhhcyBhbHdheXMgdG8gYmUgZmFsc2UuICovXG4gIEBzdGF0ZSgpXG4gIHByaXZhdGUgc2V0IF93aWRlKHdpZGU6IGJvb2xlYW4pIHtcbiAgICB0aGlzLnRvZ2dsZVN0YXRlKCd3aWRlJywgd2lkZSk7XG4gICAgdGhpcy5fd2lkZUludGVybmFsID0gd2lkZTtcbiAgfVxuICBwcml2YXRlIGdldCBfd2lkZSgpOiBib29sZWFuIHtcbiAgICByZXR1cm4gdGhpcy5fd2lkZUludGVybmFsO1xuICB9XG4gIC8vIFdlIGtlZXAgdGhlIHN0YXRlIGluIGEgZmllbGQgYmVjYXVzZSBvZiB0aGUgV2ViS2l0IGJ1ZyBodHRwczovL2J1Z3Mud2Via2l0Lm9yZy9zaG93X2J1Zy5jZ2k/aWQ9MzAzNDY3LlxuICAvLyBUT0RPOiByZS1jaGVjayB3aGV0aGVyIGZpZWxkIGlzIG5lZWRlZFxuICBwcml2YXRlIF93aWRlSW50ZXJuYWw6IGJvb2xlYW4gPSBmYWxzZTtcblxuICBAc3RhdGUoKSBwcml2YXRlIGFjY2Vzc29yIF9jYWxlbmRhclZpZXc6IENhbGVuZGFyVmlldyA9ICdkYXknO1xuXG4gIHByaXZhdGUgX25leHRDYWxlbmRhclZpZXc6IENhbGVuZGFyVmlldyA9ICdkYXknO1xuXG4gIC8qKiBJbmZvcm1hdGlvbiBhYm91dCB0aGUgcmVuZGVyZWQgZGF5IHZpZXc7IHVzZWQgaW4ga2V5Ym9hcmQgbmF2aWdhdGlvbi4gKi9cbiAgcHJpdmF0ZSBfa2V5Ym9hcmROYXZpZ2F0aW9uRGF5Vmlld1BhcmFtZXRlcnM6IENhbGVuZGFyS2V5Ym9hcmROYXZpZ2F0aW9uRGF5Vmlld1BhcmFtZXRlcnMgPSB7XG4gICAgZmlyc3REYXlJblZpZXc6IG51bGwsXG4gICAgbGFzdERheUluVmlldzogbnVsbCxcbiAgICBmaXJzdE1vbnRoT2Zmc2V0OiAwLFxuICAgIGZpcnN0TW9udGhMZW5ndGg6IDAsXG4gICAgc2Vjb25kTW9udGhPZmZzZXQ6IDAsXG4gIH07XG5cbiAgLyoqIEEgbGlzdCBvZiBkYXlzLCBpbiB0d28gZm9ybWF0cyAobG9uZyBhbmQgc2luZ2xlIGNoYXIpLiAqL1xuICBwcml2YXRlIF93ZWVrZGF5cyE6IFdlZWtkYXlbXTtcblxuICAvKiogR3JpZCBvZiBjYWxlbmRhciBjZWxscyByZXByZXNlbnRpbmcgdGhlIGRhdGVzIG9mIHRoZSBtb250aC4gKi9cbiAgcHJpdmF0ZSBfd2Vla3M6IERheTxUPltdW10gPSBbXTtcblxuICAvKiogR3JpZCBvZiBjYWxlbmRhciBjZWxscyByZXByZXNlbnRpbmcgbW9udGhzLiAqL1xuICBwcml2YXRlIF9tb250aHMhOiBNb250aFtdW107XG5cbiAgLyoqIEdyaWQgb2YgY2FsZW5kYXIgY2VsbHMgcmVwcmVzZW50aW5nIHllYXJzLiAqL1xuICBwcml2YXRlIF95ZWFycyE6IG51bWJlcltdW107XG5cbiAgLyoqIEdyaWQgb2YgY2FsZW5kYXIgY2VsbHMgcmVwcmVzZW50aW5nIHllYXJzIGZvciB0aGUgd2lkZSB2aWV3LiAqL1xuICBwcml2YXRlIF9uZXh0TW9udGhZZWFycyE6IG51bWJlcltdW107XG5cbiAgLyoqIEdyaWQgb2YgY2FsZW5kYXIgY2VsbHMgcmVwcmVzZW50aW5nIHRoZSBkYXRlcyBvZiB0aGUgbmV4dCBtb250aC4gKi9cbiAgcHJpdmF0ZSBfbmV4dE1vbnRoV2Vla3MhOiBEYXk8VD5bXVtdO1xuXG4gIC8qKiBBbiBhcnJheSBjb250YWluaW5nIGFsbCB0aGUgbW9udGggbmFtZXMgaW4gdGhlIGN1cnJlbnQgbGFuZ3VhZ2UuICovXG4gIHByaXZhdGUgX21vbnRoTmFtZXM6IHN0cmluZ1tdID0gdGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0TW9udGhOYW1lcygnbG9uZycpO1xuXG4gIC8qKiBBbiBhcnJheSBjb250YWluaW5nIHRoZSB3ZWVrcycgbnVtYmVycyBmb3IgdGhlIGN1cnJlbnQgbW9udGguICovXG4gIHByaXZhdGUgX3dlZWtOdW1iZXJzITogbnVtYmVyW107XG5cbiAgLyoqIEFuIGFycmF5IGNvbnRhaW5pbmcgdGhlIHdlZWtzJyBudW1iZXJzIGZvciB0aGUgbmV4dCBtb250aCBpbiB3aWRlIG1vZGUuICovXG4gIHByaXZhdGUgX25leHRNb250aFdlZWtOdW1iZXJzITogbnVtYmVyW107XG5cbiAgcHJpdmF0ZSBfZW5oYW5jZWRWYXJpYW50OiBib29sZWFuID0gZmFsc2U7XG5cbiAgLyoqIEEgbGlzdCBvZiBidXR0b25zIGNvcnJlc3BvbmRpbmcgdG8gZGF5cywgbW9udGhzIG9yIHllYXJzIGRlcGVuZGluZyBvbiB0aGUgdmlldy4gKi9cbiAgcHJpdmF0ZSBnZXQgX2NlbGxzKCk6IChIVE1MQnV0dG9uRWxlbWVudCB8IFNiYkNhbGVuZGFyRGF5RWxlbWVudClbXSB7XG4gICAgcmV0dXJuIEFycmF5LmZyb208SFRNTEJ1dHRvbkVsZW1lbnQgfCBTYmJDYWxlbmRhckRheUVsZW1lbnQ+KFxuICAgICAgKHRoaXMuX2NhbGVuZGFyVmlldyA9PT0gJ2RheSdcbiAgICAgICAgPyB0aGlzLl9nZXRSb290Rm9yUXVlcnlTZWxlY3RvcigpPy5xdWVyeVNlbGVjdG9yQWxsKCdzYmItY2FsZW5kYXItZGF5JylcbiAgICAgICAgOiB0aGlzLnNoYWRvd1Jvb3Q/LnF1ZXJ5U2VsZWN0b3JBbGwoJy5zYmItY2FsZW5kYXJfX2NlbGwnKSkgPz8gW10sXG4gICAgKTtcbiAgfVxuXG4gIC8qKiBUaGUgY2hvc2VuIHllYXIgaW4gdGhlIHllYXIgc2VsZWN0aW9uIHZpZXcuICovXG4gIHByaXZhdGUgX2Nob3NlblllYXI/OiBudW1iZXI7XG5cbiAgLyoqIFRoZSBjaG9zZW4gbW9udGggaW4gdGhlIHllYXIgc2VsZWN0aW9uIHZpZXcuICovXG4gIHByaXZhdGUgX2Nob3Nlbk1vbnRoPzogbnVtYmVyO1xuXG4gIC8qKiBXaGV0aGVyIHRoZSBmb2N1cyBzaG91bGQgYmUgcmVzZXQgb24gZm9jdXNDZWxsLiAqL1xuICBwcml2YXRlIF9yZXNldEZvY3VzID0gZmFsc2U7XG5cbiAgLyoqIFdoZXRoZXIgYW4gZWxlbWVudCBpbnNpZGUgdGhlIGNhbGVuZGFyIGlzIGN1cnJlbnRseSBmb2N1c2VkLiAqL1xuICBwcml2YXRlIF9jb250YWluaW5nRm9jdXMgPSBmYWxzZTtcblxuICBAc3RhdGUoKVxuICBwcml2YXRlIGFjY2Vzc29yIF9pbml0aWFsaXplZCA9IGZhbHNlO1xuXG4gIHByaXZhdGUgX2xhbmd1YWdlID0gbmV3IFNiYkxhbmd1YWdlQ29udHJvbGxlcih0aGlzKS53aXRoSGFuZGxlcigoKSA9PiB7XG4gICAgdGhpcy5fbW9udGhOYW1lcyA9IHRoaXMuX2RhdGVBZGFwdGVyLmdldE1vbnRoTmFtZXMoJ2xvbmcnKTtcbiAgICB0aGlzLl9jcmVhdGVNb250aFJvd3MoKTtcbiAgfSk7XG4gIHByaXZhdGUgX21lZGlhTWF0Y2hlciA9IG5ldyBTYmJNZWRpYU1hdGNoZXJDb250cm9sbGVyKHRoaXMsIHtcbiAgICBbU2JiTWVkaWFRdWVyeUJyZWFrcG9pbnRMYXJnZUFuZEFib3ZlXTogKCkgPT4gdGhpcy5faW5pdCgpLFxuICB9KTtcblxuICBwdWJsaWMgY29uc3RydWN0b3IoKSB7XG4gICAgc3VwZXIoKTtcbiAgICB0aGlzLl9jcmVhdGVNb250aFJvd3MoKTtcbiAgICB0aGlzLl9zZXRXZWVrZGF5cygpO1xuXG4gICAgLy8gV2UgbmVlZCB0byB0cmFjayB0aGUgZm9jdXMgYXMgd2Ugc2hvdWxkIG9ubHkgdGFrZSBmb2N1cyBpbnRvIHRoZSBjYWxlbmRhciwgd2hlbiB0aGVcbiAgICAvLyBmb2N1cyB3YXMgb25jZSBzZXQgaW50byB0aGUgY2FsZW5kYXIuXG4gICAgLy8gRm9yIHNoYWRvdyBET00gY29tcGF0aWJpbGl0eSB3ZSBuZWVkIHRvIHRyYWNrIHRoaXMgcHJvZ3JhbW1hdGljYWxseS5cbiAgICB0aGlzLmFkZEV2ZW50TGlzdGVuZXIoJ2ZvY3VzaW4nLCAoKSA9PiAodGhpcy5fY29udGFpbmluZ0ZvY3VzID0gdHJ1ZSkpO1xuICAgIHRoaXMuYWRkRXZlbnRMaXN0ZW5lcignZm9jdXNvdXQnLCAoKSA9PiAodGhpcy5fY29udGFpbmluZ0ZvY3VzID0gZmFsc2UpKTtcbiAgICB0aGlzLmFkZEV2ZW50TGlzdGVuZXIoJ2NsaWNrJywgKGUpID0+IHtcbiAgICAgIGNvbnN0IGRheSA9IChlLnRhcmdldCBhcyBIVE1MRWxlbWVudCkuY2xvc2VzdDxTYmJDYWxlbmRhckRheUVsZW1lbnQ8VD4+KCdzYmItY2FsZW5kYXItZGF5Jyk7XG4gICAgICBpZiAoZGF5KSB7XG4gICAgICAgIHRoaXMuX3NlbGVjdERhdGUoZGF5LnZhbHVlISk7XG4gICAgICB9XG4gICAgfSk7XG4gICAgdGhpcy5hZGRFdmVudExpc3RlbmVyKCdrZXlkb3duJywgKGUpID0+IHtcbiAgICAgIGlmICgoZS50YXJnZXQgYXMgSFRNTEVsZW1lbnQpLmxvY2FsTmFtZSA9PT0gJ3NiYi1jYWxlbmRhci1kYXknKSB7XG4gICAgICAgIHRoaXMuX2hhbmRsZUtleWJvYXJkRXZlbnQoXG4gICAgICAgICAgZSxcbiAgICAgICAgICB0aGlzLl9tYXBEYXRlVG9EYXkoKGUudGFyZ2V0IGFzIFNiYkNhbGVuZGFyRGF5RWxlbWVudCkudmFsdWUhIGFzIFQpLFxuICAgICAgICApO1xuICAgICAgfVxuICAgIH0pO1xuICB9XG5cbiAgcHJpdmF0ZSBfZ2V0Um9vdEZvclF1ZXJ5U2VsZWN0b3IoKTogdGhpcyB8IFNoYWRvd1Jvb3Qge1xuICAgIHJldHVybiB0aGlzLl9lbmhhbmNlZFZhcmlhbnQgPyB0aGlzIDogdGhpcy5zaGFkb3dSb290ITtcbiAgfVxuXG4gIHByaXZhdGUgX2RhdGVGaWx0ZXIoZGF0ZTogVCk6IGJvb2xlYW4ge1xuICAgIHJldHVybiB0aGlzLmRhdGVGaWx0ZXI/LihkYXRlKSA/PyB0cnVlO1xuICB9XG5cbiAgLyoqIFJlc2V0cyB0aGUgYWN0aXZlIG1vbnRoIGFjY29yZGluZyB0byB0aGUgbmV3IHN0YXRlIG9mIHRoZSBjYWxlbmRhci4gKi9cbiAgcHVibGljIHJlc2V0UG9zaXRpb24oKTogdm9pZCB7XG4gICAgdGhpcy5fcmVzZXRDYWxlbmRhclZpZXcoKTtcbiAgICB0aGlzLl9pbml0KCk7XG4gIH1cblxuICBwdWJsaWMgb3ZlcnJpZGUgY29ubmVjdGVkQ2FsbGJhY2soKTogdm9pZCB7XG4gICAgc3VwZXIuY29ubmVjdGVkQ2FsbGJhY2soKTtcbiAgICB0aGlzLnJlc2V0UG9zaXRpb24oKTtcbiAgICB0aGlzLnNoYWRvd1Jvb3Q/LmFkZEV2ZW50TGlzdGVuZXIoJ3Nsb3RjaGFuZ2UnLCB0aGlzLl9vblNsb3RDaGFuZ2UsIHsgY2FwdHVyZTogdHJ1ZSB9KTtcbiAgfVxuXG4gIHB1YmxpYyBvdmVycmlkZSBkaXNjb25uZWN0ZWRDYWxsYmFjaygpOiB2b2lkIHtcbiAgICBzdXBlci5kaXNjb25uZWN0ZWRDYWxsYmFjaygpO1xuICAgIHRoaXMuc2hhZG93Um9vdD8ucmVtb3ZlRXZlbnRMaXN0ZW5lcignc2xvdGNoYW5nZScsIHRoaXMuX29uU2xvdENoYW5nZSwgeyBjYXB0dXJlOiB0cnVlIH0pO1xuICB9XG5cbiAgLyoqIEBpbnRlcm5hbCAqL1xuICBwdWJsaWMgb3ZlcnJpZGUgZm9jdXMoKTogdm9pZCB7XG4gICAgdGhpcy5fcmVzZXRGb2N1cyA9IHRydWU7XG4gICAgdGhpcy5fZm9jdXNDZWxsKCk7XG4gIH1cblxuICBwcm90ZWN0ZWQgb3ZlcnJpZGUgd2lsbFVwZGF0ZShjaGFuZ2VkUHJvcGVydGllczogUHJvcGVydHlWYWx1ZXM8dGhpcz4pOiB2b2lkIHtcbiAgICBzdXBlci53aWxsVXBkYXRlKGNoYW5nZWRQcm9wZXJ0aWVzKTtcblxuICAgIGlmICghdGhpcy5faW5pdGlhbGl6ZWQpIHtcbiAgICAgIHJldHVybjtcbiAgICB9XG5cbiAgICBpZiAoY2hhbmdlZFByb3BlcnRpZXMuaGFzKCd3aWRlJykgfHwgY2hhbmdlZFByb3BlcnRpZXMuaGFzKCdvcmllbnRhdGlvbicpKSB7XG4gICAgICB0aGlzLnJlc2V0UG9zaXRpb24oKTtcbiAgICB9XG5cbiAgICBpZiAoY2hhbmdlZFByb3BlcnRpZXMuaGFzKCd2aWV3JykpIHtcbiAgICAgIHRoaXMuX3NldENob3NlblllYXIoKTtcbiAgICAgIHRoaXMuX2Nob3Nlbk1vbnRoID0gdW5kZWZpbmVkO1xuICAgICAgdGhpcy5fbmV4dENhbGVuZGFyVmlldyA9IHRoaXMuX2NhbGVuZGFyVmlldyA9IHRoaXMudmlldztcbiAgICB9XG4gIH1cblxuICBwcm90ZWN0ZWQgb3ZlcnJpZGUgdXBkYXRlZChjaGFuZ2VkUHJvcGVydGllczogUHJvcGVydHlWYWx1ZXM8dGhpcz4pOiB2b2lkIHtcbiAgICBzdXBlci51cGRhdGVkKGNoYW5nZWRQcm9wZXJ0aWVzKTtcbiAgICAvLyBUaGUgY2FsZW5kYXIgbmVlZHMgdG8gY2FsY3VsYXRlIHRhYi1pbmRleGVzIG9uIGZpcnN0IHJlbmRlcixcbiAgICAvLyBhbmQgZXZlcnkgdGltZSBhIGRhdGUgaXMgc2VsZWN0ZWQgb3IgdGhlIG1vbnRoIHZpZXcgY2hhbmdlcy5cbiAgICB0aGlzLl9zZXRUYWJJbmRleCgpO1xuXG4gICAgLy8gV2hlbiBjaGFuZ2luZyB2aWV3IHRvIHllYXIvbW9udGgsIHRoZSB0YWJpbmRleCBpcyBjaGFuZ2VkLCBidXQgdGhlIGZvY3VzZWQgZWxlbWVudCBpcyBnZXR0aW5nIGxvc3QuXG4gICAgLy8gV2UgbmVlZCB0byBjYWxsIGBfZm9jdXNDZWxsKClgIG1ldGhvZCBleHBsaWNpdGx5IHRvIGNvcnJlY3RseSBzZXQgdGhlIGZvY3VzLlxuICAgIHRoaXMuX2ZvY3VzQ2VsbCgpO1xuICB9XG5cbiAgcHJpdmF0ZSBfb25TbG90Q2hhbmdlID0gKCk6IHZvaWQgPT4ge1xuICAgIHRoaXMuX2VuaGFuY2VkVmFyaWFudCA9IEFycmF5LmZyb20odGhpcy5jaGlsZHJlbikuc29tZShcbiAgICAgIChjKSA9PiBjLmxvY2FsTmFtZSA9PT0gJ3NiYi1jYWxlbmRhci1kYXknLFxuICAgICk7XG4gICAgdGhpcy50b2dnbGVTdGF0ZSgnZW5oYW5jZWQnLCB0aGlzLl9lbmhhbmNlZFZhcmlhbnQpO1xuICAgIHRoaXMuX3NldFRhYkluZGV4KCk7XG4gIH07XG5cbiAgLyoqXG4gICAqIFRoZSBgX3NlbGVjdGVkYCBzdGF0ZSBzaG91bGQgYmUgYWRhcHRlZCB3aGVuIHRoZSBgbXVsdGlwbGVgIHByb3BlcnR5IGNoYW5nZXM6XG4gICAqICAgLSBpZiBpdCBjaGFuZ2VzIHRvIHRydWUsIHRoZSAnX3NlbGVjdGVkJyBpcyBzZXQgdG8gYW4gYXJyYXk7XG4gICAqICAgLSBpZiBpdCBjaGFuZ2VzIHRvIGZhbHNlLCB0aGUgZmlyc3QgYXZhaWxhYmxlIG9wdGlvbiBpcyBzZXQgYXMgJ3ZhbHVlJyBvdGhlcndpc2UgaXQncyBzZXQgdG8gbnVsbC5cbiAgICovXG4gIHByaXZhdGUgX29uTXVsdGlwbGVDaGFuZ2VkKGlzTXVsdGlwbGU6IGJvb2xlYW4pOiB2b2lkIHtcbiAgICBpZiAoaXNNdWx0aXBsZSAmJiAhQXJyYXkuaXNBcnJheSh0aGlzLl9zZWxlY3RlZCkpIHtcbiAgICAgIHRoaXMuX3NlbGVjdGVkID0gdGhpcy5fc2VsZWN0ZWQgPyBbdGhpcy5fc2VsZWN0ZWQgYXMgVF0gOiBbXTtcbiAgICB9XG4gICAgaWYgKCFpc011bHRpcGxlICYmIEFycmF5LmlzQXJyYXkodGhpcy5fc2VsZWN0ZWQpKSB7XG4gICAgICB0aGlzLl9zZWxlY3RlZCA9ICh0aGlzLl9zZWxlY3RlZCBhcyBUW10pLmxlbmd0aCA/ICh0aGlzLl9zZWxlY3RlZCBhcyBUW10pWzBdIDogbnVsbDtcbiAgICB9XG4gIH1cblxuICAvKiogSW5pdGlhbGl6ZXMgdGhlIGNvbXBvbmVudC4gKi9cbiAgcHJpdmF0ZSBfaW5pdChhY3RpdmVEYXRlPzogVCk6IHZvaWQge1xuICAgIC8vIER1ZSB0byBpdHMgY29tcGxleGl0eSwgdGhlIGNhbGVuZGFyIGlzIG9ubHkgaW5pdGlhbGl6ZWQgb24gY2xpZW50IHNpZGVcbiAgICBpZiAoaXNTZXJ2ZXIpIHtcbiAgICAgIHJldHVybjtcbiAgICB9IGVsc2UgaWYgKHRoaXMuaHlkcmF0aW9uUmVxdWlyZWQpIHtcbiAgICAgIHRoaXMuaHlkcmF0aW9uQ29tcGxldGUudGhlbigoKSA9PiB0aGlzLl9pbml0KCkpO1xuICAgICAgcmV0dXJuO1xuICAgIH1cblxuICAgIGlmIChhY3RpdmVEYXRlKSB7XG4gICAgICB0aGlzLl9hc3NpZ25BY3RpdmVEYXRlKGFjdGl2ZURhdGUpO1xuICAgIH1cbiAgICB0aGlzLl93aWRlID1cbiAgICAgICh0aGlzLl9tZWRpYU1hdGNoZXIubWF0Y2hlcyhTYmJNZWRpYVF1ZXJ5QnJlYWtwb2ludExhcmdlQW5kQWJvdmUpID8/IGZhbHNlKSAmJiB0aGlzLndpZGU7XG4gICAgdGhpcy5fd2Vla3MgPSB0aGlzLl9jcmVhdGVXZWVrUm93cyh0aGlzLl9hY3RpdmVEYXRlKTtcbiAgICB0aGlzLl95ZWFycyA9IHRoaXMuX2NyZWF0ZVllYXJSb3dzKCk7XG4gICAgdGhpcy5fd2Vla051bWJlcnMgPSB0aGlzLl9jcmVhdGVXZWVrTnVtYmVycyh0aGlzLl9hY3RpdmVEYXRlKTtcbiAgICB0aGlzLl9uZXh0TW9udGhXZWVrcyA9IFtbXV07XG4gICAgdGhpcy5fbmV4dE1vbnRoWWVhcnMgPSBbW11dO1xuICAgIGlmICh0aGlzLl93aWRlKSB7XG4gICAgICBjb25zdCBuZXh0TW9udGhEYXRlID0gdGhpcy5fZGF0ZUFkYXB0ZXIuYWRkQ2FsZW5kYXJNb250aHModGhpcy5fYWN0aXZlRGF0ZSwgMSk7XG4gICAgICB0aGlzLl9uZXh0TW9udGhXZWVrcyA9IHRoaXMuX2NyZWF0ZVdlZWtSb3dzKG5leHRNb250aERhdGUsIHRydWUpO1xuICAgICAgdGhpcy5fbmV4dE1vbnRoWWVhcnMgPSB0aGlzLl9jcmVhdGVZZWFyUm93cyhZRUFSU19QRVJfUEFHRSk7XG4gICAgICB0aGlzLl9uZXh0TW9udGhXZWVrTnVtYmVycyA9IHRoaXMuX2NyZWF0ZVdlZWtOdW1iZXJzKG5leHRNb250aERhdGUpO1xuICAgIH1cbiAgICB0aGlzLl9pbml0aWFsaXplZCA9IHRydWU7XG4gIH1cblxuICAvKiogRm9jdXNlcyBvbiBhIGRheSBjZWxsIHByaW9yaXRpemluZyB0aGUgc2VsZWN0ZWQgZGF5LCB0aGUgY3VycmVudCBkYXksIGFuZCBsYXN0bHksIHRoZSBmaXJzdCBzZWxlY3RhYmxlIGRheS4gKi9cbiAgcHJpdmF0ZSBfZm9jdXNDZWxsKCk6IHZvaWQge1xuICAgIGlmICh0aGlzLl9yZXNldEZvY3VzKSB7XG4gICAgICB0aGlzLl9nZXRGaXJzdEZvY3VzYWJsZSgpPy5mb2N1cygpO1xuICAgICAgdGhpcy5fcmVzZXRGb2N1cyA9IGZhbHNlO1xuICAgIH1cbiAgfVxuXG4gIC8qKiBDcmVhdGVzIHRoZSBhcnJheSBvZiB3ZWVrZGF5cy4gKi9cbiAgcHJpdmF0ZSBfc2V0V2Vla2RheXMoKTogdm9pZCB7XG4gICAgY29uc3QgbmFycm93V2Vla2RheXM6IHN0cmluZ1tdID0gdGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0RGF5T2ZXZWVrTmFtZXMoJ25hcnJvdycpO1xuICAgIGNvbnN0IGxvbmdXZWVrZGF5czogc3RyaW5nW10gPSB0aGlzLl9kYXRlQWRhcHRlci5nZXREYXlPZldlZWtOYW1lcygnbG9uZycpO1xuICAgIGNvbnN0IHdlZWtkYXlzOiBXZWVrZGF5W10gPSBsb25nV2Vla2RheXMubWFwKChsb25nOiBzdHJpbmcsIGk6IG51bWJlcikgPT4gKHtcbiAgICAgIGxvbmcsXG4gICAgICBuYXJyb3c6IG5hcnJvd1dlZWtkYXlzW2ldLFxuICAgIH0pKTtcblxuICAgIC8vIFJvdGF0ZXMgdGhlIGxhYmVscyBmb3IgZGF5cyBvZiB0aGUgd2VlayBiYXNlZCBvbiB0aGUgY29uZmlndXJlZCBmaXJzdCBkYXkgb2YgdGhlIHdlZWsuXG4gICAgY29uc3QgZmlyc3REYXlPZldlZWs6IG51bWJlciA9IHRoaXMuX2RhdGVBZGFwdGVyLmdldEZpcnN0RGF5T2ZXZWVrKCk7XG4gICAgdGhpcy5fd2Vla2RheXMgPSB3ZWVrZGF5cy5zbGljZShmaXJzdERheU9mV2VlaykuY29uY2F0KHdlZWtkYXlzLnNsaWNlKDAsIGZpcnN0RGF5T2ZXZWVrKSk7XG4gIH1cblxuICAvKipcbiAgICogR2l2ZW4gYSBkYXRlLCBpdCByZXR1cm5zIHRoZSB3ZWVrIG51bWJlcnMgZm9yIHRoZSBtb250aCB0aGUgZGF0ZSBiZWxvbmdzIHRvLlxuICAgKiBUT0RPOiBjaGVjayBpZiBkYXRlLWZucyBjYW4gYmUgcmVwbGFjZWQgd2l0aCBjdXN0b20gbG9naWMuXG4gICAqXG4gICAqIFNpbmNlIHRoZSBjYWxjdWxhdGlvbiBpcyBub3Qgc2ltcGxlIChzZWUgaHR0cHM6Ly9lbi53aWtpcGVkaWEub3JnL3dpa2kvV2VlayNOdW1iZXJpbmcpLFxuICAgKiB0aGUgZGF0ZS1mbnMgbGlicmFyeSBoYXMgYmVlbiB1c2VkIHRoaXMgd2F5OlxuICAgKiB0aGUgZmlyc3QgYW5kIHRoZSBsYXN0IGRheSBvZiB0aGUgbW9udGggYXJlIGNhbGN1bGF0ZWQgYW5kIHRoZW4gcGFzc2VkIHRvIHRoZSBgZWFjaFdlZWtPZkludGVydmFsYCBmdW5jdGlvbixcbiAgICogd2hpY2ggcmV0dXJucyBhbiBhcnJheSBjb250YWluaW5nIHRoZSBzdGFydGluZyBkYXkgb2YgZXZlcnkgSVNPIHdlZWsgb2YgdGhlIG1vbnRoLFxuICAgKiBjb25zaWRlcmluZyBNb25kYXkgYXMgdGhlIGZpcnN0IGRheS5cbiAgICogVGhlbiwgdGhpcyBhcnJheSBpcyBtYXBwZWQgdmlhIHRoZSBgZ2V0V2Vla2AgZnVuY3Rpb24sIHdoaWNoIHJldHVybnMgdGhlIElTTyB3ZWVrIG51bWJlciBmb3IgdGhhdCBkYXRlLlxuICAgKi9cbiAgcHJpdmF0ZSBfY3JlYXRlV2Vla051bWJlcnMoZGF0ZTogVCk6IG51bWJlcltdIHtcbiAgICByZXR1cm4gZWFjaFdlZWtPZkludGVydmFsKFxuICAgICAgeyBzdGFydDogc3RhcnRPZk1vbnRoKGRhdGUgYXMgRGF0ZSksIGVuZDogZW5kT2ZNb250aChkYXRlIGFzIERhdGUpIH0sXG4gICAgICB7IHdlZWtTdGFydHNPbjogMSB9LFxuICAgICkubWFwKChmaXJzdERheU9mV2VlazogRGF0ZSkgPT5cbiAgICAgIGdldFdlZWsoZmlyc3REYXlPZldlZWssIHsgd2Vla1N0YXJ0c09uOiAxLCBmaXJzdFdlZWtDb250YWluc0RhdGU6IDQgfSksXG4gICAgKTtcbiAgfVxuXG4gIC8qKiBDcmVhdGVzIHRoZSByb3dzIGFsb25nIHRoZSBob3Jpem9udGFsIGRpcmVjdGlvbiBhbmQgc2V0cyB0aGUgcGFyYW1ldGVycyB1c2VkIGluIGtleWJvYXJkIG5hdmlnYXRpb24uICovXG4gIHByaXZhdGUgX2NyZWF0ZVdlZWtSb3dzKHZhbHVlOiBULCBpc1NlY29uZE1vbnRoSW5WaWV3ID0gZmFsc2UpOiBEYXk8VD5bXVtdIHtcbiAgICBjb25zdCBkYXlzSW5Nb250aDogbnVtYmVyID0gdGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0TnVtRGF5c0luTW9udGgodmFsdWUpO1xuICAgIGNvbnN0IHdlZWtPZmZzZXQ6IG51bWJlciA9IHRoaXMuX2RhdGVBZGFwdGVyLmdldEZpcnN0V2Vla09mZnNldCh2YWx1ZSk7XG4gICAgaWYgKCFpc1NlY29uZE1vbnRoSW5WaWV3KSB7XG4gICAgICB0aGlzLl9rZXlib2FyZE5hdmlnYXRpb25EYXlWaWV3UGFyYW1ldGVycy5maXJzdE1vbnRoTGVuZ3RoID0gZGF5c0luTW9udGg7XG4gICAgICB0aGlzLl9rZXlib2FyZE5hdmlnYXRpb25EYXlWaWV3UGFyYW1ldGVycy5maXJzdE1vbnRoT2Zmc2V0ID0gd2Vla09mZnNldDtcbiAgICAgIHRoaXMuX2tleWJvYXJkTmF2aWdhdGlvbkRheVZpZXdQYXJhbWV0ZXJzLmZpcnN0RGF5SW5WaWV3ID0gdGhpcy5fZGF0ZUFkYXB0ZXIudG9Jc284NjAxKFxuICAgICAgICB0aGlzLl9kYXRlQWRhcHRlci5jcmVhdGVEYXRlKFxuICAgICAgICAgIHRoaXMuX2RhdGVBZGFwdGVyLmdldFllYXIodmFsdWUpLFxuICAgICAgICAgIHRoaXMuX2RhdGVBZGFwdGVyLmdldE1vbnRoKHZhbHVlKSxcbiAgICAgICAgICAxLFxuICAgICAgICApLFxuICAgICAgKTtcbiAgICAgIHRoaXMuX2tleWJvYXJkTmF2aWdhdGlvbkRheVZpZXdQYXJhbWV0ZXJzLmxhc3REYXlJblZpZXcgPSB0aGlzLl9kYXRlQWRhcHRlci50b0lzbzg2MDEoXG4gICAgICAgIHRoaXMuX2RhdGVBZGFwdGVyLmNyZWF0ZURhdGUoXG4gICAgICAgICAgdGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0WWVhcih2YWx1ZSksXG4gICAgICAgICAgdGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0TW9udGgodmFsdWUpLFxuICAgICAgICAgIGRheXNJbk1vbnRoLFxuICAgICAgICApLFxuICAgICAgKTtcbiAgICB9IGVsc2Uge1xuICAgICAgdGhpcy5fa2V5Ym9hcmROYXZpZ2F0aW9uRGF5Vmlld1BhcmFtZXRlcnMuc2Vjb25kTW9udGhPZmZzZXQgPSB3ZWVrT2Zmc2V0O1xuICAgICAgdGhpcy5fa2V5Ym9hcmROYXZpZ2F0aW9uRGF5Vmlld1BhcmFtZXRlcnMubGFzdERheUluVmlldyA9IHRoaXMuX2RhdGVBZGFwdGVyLnRvSXNvODYwMShcbiAgICAgICAgdGhpcy5fZGF0ZUFkYXB0ZXIuY3JlYXRlRGF0ZShcbiAgICAgICAgICB0aGlzLl9kYXRlQWRhcHRlci5nZXRZZWFyKHZhbHVlKSxcbiAgICAgICAgICB0aGlzLl9kYXRlQWRhcHRlci5nZXRNb250aCh2YWx1ZSksXG4gICAgICAgICAgZGF5c0luTW9udGgsXG4gICAgICAgICksXG4gICAgICApO1xuICAgIH1cbiAgICByZXR1cm4gdGhpcy5vcmllbnRhdGlvbiA9PT0gJ2hvcml6b250YWwnXG4gICAgICA/IHRoaXMuX2NyZWF0ZVdlZWtSb3dzSG9yaXpvbnRhbCh2YWx1ZSwgZGF5c0luTW9udGgsIHdlZWtPZmZzZXQpXG4gICAgICA6IHRoaXMuX2NyZWF0ZVdlZWtSb3dzVmVydGljYWwodmFsdWUsIGRheXNJbk1vbnRoLCB3ZWVrT2Zmc2V0KTtcbiAgfVxuXG4gIC8qKlxuICAgKiBDcmVhdGVzIHRoZSByb3dzIGZvciBlYWNoIHdlZWsgaW4gb3JpZW50YXRpb249J2hvcml6b250YWwnLlxuICAgKlxuICAgKiBJdGVyYXRlcyB0aHJvdWdoIHRoZSBkYXlzIG9mIHRoZSBtb250aHMsIGNyZWF0ZXMgYSBEYXkgb2JqZWN0IGZvciBlYWNoIGFuZCBwdXNoZXMgaXQgaW50byBhbmQgYXJyYXkuXG4gICAqIEVhY2ggc2V2ZW4gZGF5cyAoY29uc2lkZXJpbmcgdGhlIG9mZnNldCBhdCB0aGUgYmVnaW5uaW5nIG9mIHRoZSBtb250aCkgcmVzdGFydHMgZnJvbSBhbiBlbXB0eSBhcnJheS5cbiAgICpcbiAgICogVGhlIHJlc3VsdCBpcyBhIG1hdHJpeCBpbiB3aGljaCBldmVyeSByb3cgaXMgYSB3ZWVrIChvciBwYXJ0IG9mIGl0LCBjb25zaWRlcmluZyBvZmZzZXQpLlxuICAgKi9cbiAgcHJpdmF0ZSBfY3JlYXRlV2Vla1Jvd3NIb3Jpem9udGFsKHZhbHVlOiBULCBkYXlzSW5Nb250aDogbnVtYmVyLCB3ZWVrT2Zmc2V0OiBudW1iZXIpOiBEYXk8VD5bXVtdIHtcbiAgICBjb25zdCB3ZWVrczogRGF5PFQ+W11bXSA9IFtbXV07XG4gICAgZm9yIChsZXQgaSA9IDAsIGNlbGwgPSB3ZWVrT2Zmc2V0OyBpIDwgZGF5c0luTW9udGg7IGkrKywgY2VsbCsrKSB7XG4gICAgICBpZiAoY2VsbCA9PT0gREFZU19QRVJfUk9XKSB7XG4gICAgICAgIHdlZWtzLnB1c2goW10pO1xuICAgICAgICBjZWxsID0gMDtcbiAgICAgIH1cbiAgICAgIGNvbnN0IGRhdGUgPSB0aGlzLl9kYXRlQWRhcHRlci5jcmVhdGVEYXRlKFxuICAgICAgICB0aGlzLl9kYXRlQWRhcHRlci5nZXRZZWFyKHZhbHVlKSxcbiAgICAgICAgdGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0TW9udGgodmFsdWUpLFxuICAgICAgICBpICsgMSxcbiAgICAgICkhO1xuICAgICAgd2Vla3Nbd2Vla3MubGVuZ3RoIC0gMV0ucHVzaCh0aGlzLl9tYXBEYXRlVG9EYXkoZGF0ZSkpO1xuICAgIH1cbiAgICByZXR1cm4gd2Vla3M7XG4gIH1cblxuICAvKipcbiAgICogQ3JlYXRlcyB0aGUgcm93cyBmb3IgZWFjaCB3ZWVrIGluIG9yaWVudGF0aW9uPSd2ZXJ0aWNhbCcuXG4gICAqXG4gICAqIENyZWF0ZXMgYSBtYXRyaXggd2l0aCBzZXZlbiBlbXB0eSByb3dzLlxuICAgKiBJdGVyYXRlcyB0aHJvdWdoIHRoZSBkYXlzIG9mIHRoZSBtb250aHMsIGNyZWF0ZXMgYSBEYXkgb2JqZWN0IGZvciBlYWNoXG4gICAqIGFuZCBwdXNoZXMgaXQgaW50byB0aGUgY29ycmVjdCBhcnJheSBjb25zaWRlcmluZyB0aGUgb2Zmc2V0IGF0IHRoZSBiZWdpbm5pbmcgb2YgdGhlIG1vbnRoLlxuICAgKiBFYWNoIHNldmVuIGRheXMgKGluY2x1ZGluZyBvZmZzZXQpIHJlc3RhcnRzIGZyb20gdGhlIGZpcnN0LlxuICAgKlxuICAgKiBUaGUgcmVzdWx0IGlzIGEgbWF0cml4IGluIHdoaWNoIGV2ZXJ5IHJvdyBpcyBhIHNldCBvZiB3ZWVrZGF5cywgc286XG4gICAqIC0gcm93IDA6IGFsbCB0aGUgTW9uZGF5cztcbiAgICogLSByb3cgMTogYWxsIHRoZSBUdWVzZGF5cztcbiAgICogLSAuLi5cbiAgICogLSByb3cgNzogYWxsIHRoZSBTdW5kYXlzLlxuICAgKi9cbiAgcHJpdmF0ZSBfY3JlYXRlV2Vla1Jvd3NWZXJ0aWNhbCh2YWx1ZTogVCwgZGF5c0luTW9udGg6IG51bWJlciwgd2Vla09mZnNldDogbnVtYmVyKTogRGF5PFQ+W11bXSB7XG4gICAgY29uc3Qgd2Vla3M6IERheTxUPltdW10gPSBBcnJheS5mcm9tKHsgbGVuZ3RoOiBEQVlTX1BFUl9ST1cgfSwgKCkgPT4gW10pO1xuICAgIGZvciAobGV0IGkgPSAwLCBjZWxsID0gd2Vla09mZnNldDsgaSA8IGRheXNJbk1vbnRoOyBpKyssIGNlbGwrKykge1xuICAgICAgaWYgKGNlbGwgPT09IERBWVNfUEVSX1JPVykge1xuICAgICAgICBjZWxsID0gMDtcbiAgICAgIH1cbiAgICAgIGNvbnN0IGRhdGUgPSB0aGlzLl9kYXRlQWRhcHRlci5jcmVhdGVEYXRlKFxuICAgICAgICB0aGlzLl9kYXRlQWRhcHRlci5nZXRZZWFyKHZhbHVlKSxcbiAgICAgICAgdGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0TW9udGgodmFsdWUpLFxuICAgICAgICBpICsgMSxcbiAgICAgICkhO1xuICAgICAgd2Vla3NbY2VsbF0ucHVzaCh0aGlzLl9tYXBEYXRlVG9EYXkoZGF0ZSkpO1xuICAgIH1cbiAgICByZXR1cm4gd2Vla3M7XG4gIH1cblxuICBwcml2YXRlIF9tYXBEYXRlVG9EYXkoZGF0ZTogVCk6IERheTxUPiB7XG4gICAgY29uc3QgaXNvRGF0ZSA9IHRoaXMuX2RhdGVBZGFwdGVyLnRvSXNvODYwMShkYXRlKTtcbiAgICByZXR1cm4ge1xuICAgICAgdmFsdWU6IGlzb0RhdGUsXG4gICAgICBkYXRlVmFsdWU6IGRhdGUsXG4gICAgICBkYXlWYWx1ZTogU3RyaW5nKHRoaXMuX2RhdGVBZGFwdGVyLmdldERhdGUoZGF0ZSkpLFxuICAgICAgbW9udGhWYWx1ZTogU3RyaW5nKHRoaXMuX2RhdGVBZGFwdGVyLmdldE1vbnRoKGRhdGUpKSxcbiAgICAgIHllYXJWYWx1ZTogU3RyaW5nKHRoaXMuX2RhdGVBZGFwdGVyLmdldFllYXIoZGF0ZSkpLFxuICAgICAgd2Vla1ZhbHVlOiBnZXRXZWVrKGlzb0RhdGUsIHsgd2Vla1N0YXJ0c09uOiAxLCBmaXJzdFdlZWtDb250YWluc0RhdGU6IDQgfSksXG4gICAgICB3ZWVrRGF5VmFsdWU6IHRoaXMuX2RhdGVBZGFwdGVyLmdldERheU9mV2VlayhkYXRlKSxcbiAgICB9O1xuICB9XG5cbiAgLyoqIEZvcmNlIHRoZSBjb252ZXJzaW9uIHRvIElTTzg2MDEgZm9ybWF0dGVkIHN0cmluZy4gKi9cbiAgcHJpdmF0ZSBfbWFwVmFsdWVUb0lTT0RhdGUodmFsdWU6IHN0cmluZyB8IERhdGUpOiBzdHJpbmcge1xuICAgIHJldHVybiB0eXBlb2YgdmFsdWUgPT09ICdzdHJpbmcnID8gdmFsdWUgOiB0aGlzLl9kYXRlQWRhcHRlci50b0lzbzg2MDEodmFsdWUgYXMgVCk7XG4gIH1cblxuICAvKiogQ3JlYXRlcyB0aGUgcm93cyBmb3IgdGhlIG1vbnRoIHNlbGVjdGlvbiB2aWV3LiAqL1xuICBwcml2YXRlIF9jcmVhdGVNb250aFJvd3MoKTogdm9pZCB7XG4gICAgY29uc3Qgc2hvcnROYW1lczogc3RyaW5nW10gPSB0aGlzLl9kYXRlQWRhcHRlci5nZXRNb250aE5hbWVzKCdzaG9ydCcpO1xuICAgIGNvbnN0IG1vbnRoczogTW9udGhbXSA9IG5ldyBBcnJheSgxMikuZmlsbChudWxsKS5tYXAoXG4gICAgICAoXywgaTogbnVtYmVyKTogTW9udGggPT4gKHtcbiAgICAgICAgdmFsdWU6IHNob3J0TmFtZXNbaV0sXG4gICAgICAgIGxvbmdWYWx1ZTogdGhpcy5fbW9udGhOYW1lc1tpXSxcbiAgICAgICAgbW9udGhWYWx1ZTogaSArIDEsXG4gICAgICB9KSxcbiAgICApO1xuICAgIGNvbnN0IHJvd3M6IG51bWJlciA9IDEyIC8gTU9OVEhTX1BFUl9ST1c7XG4gICAgY29uc3QgbW9udGhBcnJheTogTW9udGhbXVtdID0gW107XG4gICAgZm9yIChsZXQgaTogbnVtYmVyID0gMDsgaSA8IHJvd3M7IGkrKykge1xuICAgICAgbW9udGhBcnJheS5wdXNoKG1vbnRocy5zbGljZShNT05USFNfUEVSX1JPVyAqIGksIE1PTlRIU19QRVJfUk9XICogKGkgKyAxKSkpO1xuICAgIH1cbiAgICB0aGlzLl9tb250aHMgPSBtb250aEFycmF5O1xuICB9XG5cbiAgLyoqIENyZWF0ZXMgdGhlIHJvd3MgZm9yIHRoZSB5ZWFyIHNlbGVjdGlvbiB2aWV3LiAqL1xuICBwcml2YXRlIF9jcmVhdGVZZWFyUm93cyhvZmZzZXQ6IG51bWJlciA9IDApOiBudW1iZXJbXVtdIHtcbiAgICBjb25zdCBzdGFydFZhbHVlWWVhclZpZXc6IG51bWJlciA9IHRoaXMuX2dldFN0YXJ0VmFsdWVZZWFyVmlldygpO1xuICAgIGNvbnN0IGFsbFllYXJzOiBudW1iZXJbXSA9IG5ldyBBcnJheShZRUFSU19QRVJfUEFHRSlcbiAgICAgIC5maWxsKDApXG4gICAgICAubWFwKChfLCBpOiBudW1iZXIpID0+IHN0YXJ0VmFsdWVZZWFyVmlldyArIG9mZnNldCArIGkpO1xuICAgIGNvbnN0IHJvd3M6IG51bWJlciA9IFlFQVJTX1BFUl9QQUdFIC8gWUVBUlNfUEVSX1JPVztcbiAgICBjb25zdCB5ZWFyQXJyYXk6IG51bWJlcltdW10gPSBbXTtcbiAgICBmb3IgKGxldCBpOiBudW1iZXIgPSAwOyBpIDwgcm93czsgaSsrKSB7XG4gICAgICB5ZWFyQXJyYXkucHVzaChhbGxZZWFycy5zbGljZShZRUFSU19QRVJfUk9XICogaSwgWUVBUlNfUEVSX1JPVyAqIChpICsgMSkpKTtcbiAgICB9XG4gICAgcmV0dXJuIHllYXJBcnJheTtcbiAgfVxuXG4gIC8qKlxuICAgKiBDYWxjdWxhdGVzIHRoZSBmaXJzdCB5ZWFyIHRoYXQgd2lsbCBiZSBzaG93biBpbiB0aGUgeWVhciBzZWxlY3Rpb24gcGFuZWwuXG4gICAqIElmIGBtaW5EYXRlYCBhbmQgYG1heERhdGVgIGFyZSBib3RoIG51bGwsIHRoZSBzdGFydGluZyB5ZWFyIGlzIGNhbGN1bGF0ZWQgYXNcbiAgICogdGhlIG11bHRpcGxlIG9mIFlFQVJTX1BFUl9QQUdFIGNsb3Nlc3QgdG8gYW5kIGxlc3MgdGhhbiBhY3RpdmVEYXRlLFxuICAgKiBlLmcuLCB3aXRoIGBZRUFSU19QRVJfUEFHRWAgPSAyNCBhbmQgYGFjdGl2ZURhdGVgID0gMjAyMCwgdGhlIGZ1bmN0aW9uIHdpbGwgcmV0dXJuIDIwMTYgKDI0ICogODMpLFxuICAgKiB3aGlsZSB3aXRoIGBhY3RpdmVEYXRlYCA9IDIwMDAsIHRoZSBmdW5jdGlvbiB3aWxsIHJldHVybiAxOTkyICgyNCAqIDgyKS5cbiAgICogSWYgYG1pbkRhdGVgIGlzIG5vdCBudWxsLCBpdCByZXR1cm5zIHRoZSBjb3JyZXNwb25kaW5nIHllYXI7IGlmIGBtYXhEYXRlYCBpcyBub3QgbnVsbCxcbiAgICogaXQgcmV0dXJucyB0aGUgY29ycmVzcG9uZGluZyB5ZWFyIG1pbnVzIGBZRUFSU19QRVJfUEFHRWAsIHNvIHRoYXQgdGhlIGBtYXhEYXRlYCBpcyB0aGUgbGFzdCByZW5kZXJlZCB5ZWFyLlxuICAgKiBJZiBib3RoIGFyZSBub3QgbnVsbCwgYG1heERhdGVgIGhhcyBwcmlvcml0eSBvdmVyIGBtaW5EYXRlYC5cbiAgICovXG4gIHByaXZhdGUgX2dldFN0YXJ0VmFsdWVZZWFyVmlldygpOiBudW1iZXIge1xuICAgIGxldCBzdGFydGluZ1llYXIgPSAwO1xuICAgIGlmICh0aGlzLm1heCkge1xuICAgICAgc3RhcnRpbmdZZWFyID0gdGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0WWVhcih0aGlzLm1heCkgLSBZRUFSU19QRVJfUEFHRSArIDE7XG4gICAgfSBlbHNlIGlmICh0aGlzLm1pbikge1xuICAgICAgc3RhcnRpbmdZZWFyID0gdGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0WWVhcih0aGlzLm1pbik7XG4gICAgfVxuICAgIGNvbnN0IGFjdGl2ZVllYXIgPSB0aGlzLl9kYXRlQWRhcHRlci5nZXRZZWFyKHRoaXMuX2FjdGl2ZURhdGUpO1xuICAgIHJldHVybiAoXG4gICAgICBhY3RpdmVZZWFyIC1cbiAgICAgICgoKChhY3RpdmVZZWFyIC0gc3RhcnRpbmdZZWFyKSAlIFlFQVJTX1BFUl9QQUdFKSArIFlFQVJTX1BFUl9QQUdFKSAlIFlFQVJTX1BFUl9QQUdFKVxuICAgICk7XG4gIH1cblxuICAvKiogQ2hlY2tzIGlmIGRhdGUgaXMgd2l0aGluIHRoZSBtaW4tbWF4IHJhbmdlLiAqL1xuICBwcml2YXRlIF9pc0RheUluUmFuZ2UoZGF0ZVN0cmluZzogc3RyaW5nKTogYm9vbGVhbiB7XG4gICAgaWYgKCF0aGlzLm1pbiAmJiAhdGhpcy5tYXgpIHtcbiAgICAgIHJldHVybiB0cnVlO1xuICAgIH1cbiAgICBjb25zdCBkYXRlID0gdGhpcy5fZGF0ZUFkYXB0ZXIuZGVzZXJpYWxpemUoZGF0ZVN0cmluZykhO1xuICAgIHJldHVybiB0aGlzLl9kYXRlQWRhcHRlci5zYW1lRGF0ZShkYXRlLCB0aGlzLl9kYXRlQWRhcHRlci5jbGFtcERhdGUoZGF0ZSwgdGhpcy5taW4sIHRoaXMubWF4KSk7XG4gIH1cblxuICAvKiogQ2hlY2tzIGlmIGRhdGUgaXMgd2l0aGluIHRoZSBtaW4tbWF4IHJhbmdlIGluIG1vbnRoIHZpZXcuICovXG4gIHByaXZhdGUgX2lzTW9udGhJblJhbmdlKG1vbnRoOiBudW1iZXIsIHllYXI6IG51bWJlcik6IGJvb2xlYW4ge1xuICAgIGlmICghdGhpcy5taW4gJiYgIXRoaXMubWF4KSB7XG4gICAgICByZXR1cm4gdHJ1ZTtcbiAgICB9XG5cbiAgICBjb25zdCBpc0JlZm9yZU1pbjogYm9vbGVhbiA9XG4gICAgICB0aGlzLl9kYXRlQWRhcHRlci5pc1ZhbGlkKHRoaXMubWluKSAmJlxuICAgICAgKHllYXIgPCB0aGlzLl9kYXRlQWRhcHRlci5nZXRZZWFyKHRoaXMubWluISkgfHxcbiAgICAgICAgKHllYXIgPT09IHRoaXMuX2RhdGVBZGFwdGVyLmdldFllYXIodGhpcy5taW4hKSAmJlxuICAgICAgICAgIG1vbnRoIDwgdGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0TW9udGgodGhpcy5taW4hKSkpO1xuXG4gICAgY29uc3QgaXNBZnRlck1heDogYm9vbGVhbiA9XG4gICAgICB0aGlzLl9kYXRlQWRhcHRlci5pc1ZhbGlkKHRoaXMubWF4KSAmJlxuICAgICAgKHllYXIgPiB0aGlzLl9kYXRlQWRhcHRlci5nZXRZZWFyKHRoaXMubWF4ISkgfHxcbiAgICAgICAgKHllYXIgPT09IHRoaXMuX2RhdGVBZGFwdGVyLmdldFllYXIodGhpcy5tYXghKSAmJlxuICAgICAgICAgIG1vbnRoID4gdGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0TW9udGgodGhpcy5tYXghKSkpO1xuXG4gICAgcmV0dXJuICEoaXNCZWZvcmVNaW4gfHwgaXNBZnRlck1heCk7XG4gIH1cblxuICAvKiogQ2hlY2tzIGlmIGRhdGUgaXMgd2l0aGluIHRoZSBtaW4tbWF4IHJhbmdlIGluIHllYXIgdmlldy4gKi9cbiAgcHJpdmF0ZSBfaXNZZWFySW5SYW5nZSh5ZWFyOiBudW1iZXIpOiBib29sZWFuIHtcbiAgICBpZiAoIXRoaXMubWluICYmICF0aGlzLm1heCkge1xuICAgICAgcmV0dXJuIHRydWU7XG4gICAgfVxuICAgIGNvbnN0IGlzQmVmb3JlTWluOiBib29sZWFuID1cbiAgICAgIHRoaXMuX2RhdGVBZGFwdGVyLmlzVmFsaWQodGhpcy5taW4pICYmIHRoaXMuX2RhdGVBZGFwdGVyLmdldFllYXIodGhpcy5taW4hKSA+IHllYXI7XG4gICAgY29uc3QgaXNBZnRlck1heDogYm9vbGVhbiA9XG4gICAgICB0aGlzLl9kYXRlQWRhcHRlci5pc1ZhbGlkKHRoaXMubWF4KSAmJiB0aGlzLl9kYXRlQWRhcHRlci5nZXRZZWFyKHRoaXMubWF4ISkgPCB5ZWFyO1xuICAgIHJldHVybiAhKGlzQmVmb3JlTWluIHx8IGlzQWZ0ZXJNYXgpO1xuICB9XG5cbiAgLy8gSW1wbGVtZW50YXRpb24gYWRhcHRlZCBmcm9tIGh0dHBzOi8vZ2l0aHViLmNvbS9hbmd1bGFyL2NvbXBvbmVudHMvYmxvYi9tYWluL3NyYy9tYXRlcmlhbC9kYXRlcGlja2VyL3llYXItdmlldy50cyNMMzY2XG4gIHByaXZhdGUgX2lzTW9udGhGaWx0ZXJlZE91dChtb250aDogbnVtYmVyLCB5ZWFyOiBudW1iZXIpOiBib29sZWFuIHtcbiAgICBpZiAoIXRoaXMuZGF0ZUZpbHRlcikge1xuICAgICAgcmV0dXJuIHRydWU7XG4gICAgfVxuXG4gICAgY29uc3QgZmlyc3RPZk1vbnRoID0gdGhpcy5fZGF0ZUFkYXB0ZXIuY3JlYXRlRGF0ZSh5ZWFyLCBtb250aCwgMSkhO1xuICAgIGZvciAoXG4gICAgICBsZXQgZGF0ZTogVCA9IGZpcnN0T2ZNb250aDtcbiAgICAgIHRoaXMuX2RhdGVBZGFwdGVyLmdldE1vbnRoKGRhdGUpID09IG1vbnRoO1xuICAgICAgZGF0ZSA9IHRoaXMuX2RhdGVBZGFwdGVyLmFkZENhbGVuZGFyRGF5cyhkYXRlLCAxKVxuICAgICkge1xuICAgICAgaWYgKHRoaXMuZGF0ZUZpbHRlcihkYXRlKSkge1xuICAgICAgICByZXR1cm4gdHJ1ZTtcbiAgICAgIH1cbiAgICB9XG5cbiAgICByZXR1cm4gZmFsc2U7XG4gIH1cblxuICAvLyBJbXBsZW1lbnRhdGlvbiBhZGFwdGVkIGZyb20gaHR0cHM6Ly9naXRodWIuY29tL2FuZ3VsYXIvY29tcG9uZW50cy9ibG9iL21haW4vc3JjL21hdGVyaWFsL2RhdGVwaWNrZXIvbXVsdGkteWVhci12aWV3LnRzI0wzNTFcbiAgcHJpdmF0ZSBfaXNZZWFyRmlsdGVyZWRPdXQoeWVhcjogbnVtYmVyKTogYm9vbGVhbiB7XG4gICAgaWYgKCF0aGlzLmRhdGVGaWx0ZXIpIHtcbiAgICAgIHJldHVybiB0cnVlO1xuICAgIH1cblxuICAgIGNvbnN0IGZpcnN0T2ZZZWFyID0gdGhpcy5fZGF0ZUFkYXB0ZXIuY3JlYXRlRGF0ZSh5ZWFyLCAxLCAxKSE7XG4gICAgZm9yIChcbiAgICAgIGxldCBkYXRlOiBUID0gZmlyc3RPZlllYXI7XG4gICAgICB0aGlzLl9kYXRlQWRhcHRlci5nZXRZZWFyKGRhdGUpID09IHllYXI7XG4gICAgICBkYXRlID0gdGhpcy5fZGF0ZUFkYXB0ZXIuYWRkQ2FsZW5kYXJEYXlzKGRhdGUsIDEpXG4gICAgKSB7XG4gICAgICBpZiAodGhpcy5kYXRlRmlsdGVyKGRhdGUpKSB7XG4gICAgICAgIHJldHVybiB0cnVlO1xuICAgICAgfVxuICAgIH1cblxuICAgIHJldHVybiBmYWxzZTtcbiAgfVxuXG4gIC8qKiBFbWl0cyB0aGUgc2VsZWN0ZWQgZGF0ZSBhbmQgc2V0cyBpdCBpbnRlcm5hbGx5LiAqL1xuICBwcml2YXRlIF9zZWxlY3REYXRlKGRheTogVCk6IHZvaWQge1xuICAgIHRoaXMuX2Nob3Nlbk1vbnRoID0gdW5kZWZpbmVkO1xuICAgIHRoaXMuX3NldENob3NlblllYXIoKTtcbiAgICBpZiAodGhpcy5tdWx0aXBsZSkge1xuICAgICAgLy8gQ2hlY2sgaWYgX3NlbGVjdGVkIGhhcyBlbGVtZW50c1xuICAgICAgaWYgKHRoaXMuX3NlbGVjdGVkICYmICh0aGlzLl9zZWxlY3RlZCBhcyBUW10pLmxlbmd0aCA+IDApIHtcbiAgICAgICAgY29uc3QgaW5kZXhPZlNlbGVjdGVkRGF5OiBudW1iZXIgPSAodGhpcy5fc2VsZWN0ZWQgYXMgVFtdKS5maW5kSW5kZXgoXG4gICAgICAgICAgKHNlbCkgPT4gdGhpcy5fZGF0ZUFkYXB0ZXIuY29tcGFyZURhdGUoc2VsLCBkYXkpID09PSAwLFxuICAgICAgICApO1xuICAgICAgICAvLyBJZiB0aGUgc2VsZWN0ZWQgZGF0ZSBpcyBhbHJlYWR5IGluIHRoZSBfc2VsZWN0ZWQgYXJyYXksIHJlbW92ZSBpdCwgb3RoZXJ3aXNlIGFkZCBpdFxuICAgICAgICBpZiAoaW5kZXhPZlNlbGVjdGVkRGF5ICE9PSAtMSkge1xuICAgICAgICAgIHRoaXMuX3NlbGVjdGVkID0gKHRoaXMuX3NlbGVjdGVkIGFzIFRbXSkuZmlsdGVyKChfLCBpKSA9PiBpICE9PSBpbmRleE9mU2VsZWN0ZWREYXkpO1xuICAgICAgICB9IGVsc2Uge1xuICAgICAgICAgIHRoaXMuX3NlbGVjdGVkID0gWy4uLih0aGlzLl9zZWxlY3RlZCBhcyBUW10pLCBkYXldO1xuICAgICAgICB9XG4gICAgICB9IGVsc2Uge1xuICAgICAgICAvLyBJZiBfc2VsZWN0ZWQgaXMgZW1wdHksIHNldCBpdFxuICAgICAgICB0aGlzLl9zZWxlY3RlZCA9IFtkYXldO1xuICAgICAgfVxuICAgICAgdGhpcy5fZW1pdERhdGVTZWxlY3RlZEV2ZW50KHRoaXMuX3NlbGVjdGVkLm1hcCgoZSkgPT4gdGhpcy5fZGF0ZUFkYXB0ZXIuZGVzZXJpYWxpemUoZSkhKSk7XG4gICAgfSBlbHNlIHtcbiAgICAgIC8vIEluIHNpbmdsZSBzZWxlY3Rpb24sIGNoZWNrIGlmIHRoZSBkYXkgaXMgYWxyZWFkeSBzZWxlY3RlZFxuICAgICAgaWYgKCF0aGlzLl9zZWxlY3RlZCB8fCB0aGlzLl9kYXRlQWRhcHRlci5jb21wYXJlRGF0ZSh0aGlzLl9zZWxlY3RlZCBhcyBULCBkYXkpICE9PSAwKSB7XG4gICAgICAgIHRoaXMuX3NlbGVjdGVkID0gZGF5O1xuICAgICAgICB0aGlzLl9lbWl0RGF0ZVNlbGVjdGVkRXZlbnQodGhpcy5fZGF0ZUFkYXB0ZXIuZGVzZXJpYWxpemUoZGF5KSEpO1xuICAgICAgfVxuICAgIH1cbiAgfVxuXG4gIC8qKlxuICAgKiBIYW5kbGUgbXVsdGlwbGUgZGF0ZXMgc2VsZWN0aW9uIHZpYSB3ZWVrTnVtYmVyIC8gd2Vla0RheSBidXR0b25zOlxuICAgKiAtIGlmIENtZCBvciBDdHJsIGFyZSBwcmVzc2VkLCBhZGQgdGhlIG5ldyBkYXRlIHRvIHRoZSBjdXJyZW50IG9uZXM7XG4gICAqIC0gaWYgbm90LFxuICAgKiAgICAgLSBpZiB0aGUgbmV3IGRhdGVzIGFyZSB0aGUgc2FtZSBvZiB0aGUgY3VycmVudCBvbmVzLCBpdCBtZWFucyB0aGF0IHRoZSBzYW1lIGJ1dHRvbiBoYXMgYmVlbiBjbGlja2VkIHR3aWNlLCBzbyBkbyBub3RoaW5nO1xuICAgKiAgICAgLSBpZiBub3QsIHRoZSBzZWxlY3RlZCBkYXRlcyBhcmUgdGhlIG5ldyBvbmVzLlxuICAgKi9cbiAgcHJpdmF0ZSBfc2VsZWN0TXVsdGlwbGVEYXRlcyhkYXlzOiBEYXk8VD5bXSk6IHZvaWQge1xuICAgIC8vIEZpbHRlciBkaXNhYmxlZCBkYXlzIGJ5IG1hdGNoaW5nIHRoZSBwcm92aWRlZCBgZGF5c2AgcGFyYW1ldGVyIGFnYWluc3QgdGhlIGVuYWJsZWQgY2VsbHMuXG4gICAgLy8gU2luY2UgdGhlIGJ1dHRvbnMnIHZhbHVlIGlzIHNldCB0byB0aGUgRGF5J3MgaW50ZXJmYWNlIHZhbHVlIChJU08gc3RyaW5nKSwgdGhlcmUncyBubyBuZWVkIHRvIGRlc2VyaWFsaXplIGl0LlxuICAgIGNvbnN0IGVuYWJsZWREYXlzOiBzdHJpbmdbXSA9IHRoaXMuX2NlbGxzXG4gICAgICAuZmlsdGVyKChlKSA9PiAhZS5kaXNhYmxlZClcbiAgICAgIC5tYXAoKGUpID0+IHRoaXMuX21hcFZhbHVlVG9JU09EYXRlKGUudmFsdWUhKSk7XG4gICAgY29uc3QgZGF5c1RvQWRkOiBzdHJpbmdbXSA9IGRheXNcbiAgICAgIC5tYXAoKGU6IERheTxUPikgPT4gZS52YWx1ZSlcbiAgICAgIC5maWx0ZXIoKGlzb0RhdGU6IHN0cmluZykgPT4gZW5hYmxlZERheXMuaW5jbHVkZXMoaXNvRGF0ZSkpO1xuICAgIGNvbnN0IGRheXNUb0FkZFNldCA9IG5ldyBTZXQoZGF5c1RvQWRkKTtcbiAgICBjb25zdCBzZWxlY3RlZFNldCA9IG5ldyBTZXQoKHRoaXMuX3NlbGVjdGVkIGFzIFRbXSkubWFwKChzKSA9PiB0aGlzLl9kYXRlQWRhcHRlci50b0lzbzg2MDEocykpKTtcbiAgICBjb25zdCBzZWxTdHJpbmdzID0gdGhpcy5fdXBkYXRlU2VsZWN0ZWRXaXRoTXVsdGlwbGVEYXRlcyhkYXlzVG9BZGQsIGRheXNUb0FkZFNldCwgc2VsZWN0ZWRTZXQpO1xuICAgIHRoaXMuX3NlbGVjdGVkID0gc2VsU3RyaW5ncy5tYXAoKHMpID0+IHRoaXMuX2RhdGVBZGFwdGVyLmRlc2VyaWFsaXplKHMpISk7XG5cbiAgICB0aGlzLl9lbWl0RGF0ZVNlbGVjdGVkRXZlbnQodGhpcy5fc2VsZWN0ZWQubWFwKChlKSA9PiB0aGlzLl9kYXRlQWRhcHRlci5kZXNlcmlhbGl6ZShlKSEpKTtcbiAgfVxuXG4gIC8qKlxuICAgKiBFbWl0cyB0aGUgZGF0ZXNlbGVjdGVkIGV2ZW50IGdpdmVuIHRoZSBkZXRhaWwgKGFzIFQgb3IgVFtdIGJhc2VkIG9uIHRoZSB2YWx1ZSBvZiB0aGUgbXVsdGlwbGUgZmxhZykuXG4gICAqL1xuICBwcml2YXRlIF9lbWl0RGF0ZVNlbGVjdGVkRXZlbnQoZGV0YWlsOiBUIHwgVFtdKTogdm9pZCB7XG4gICAgLyoqIEB0eXBlIHtDdXN0b21FdmVudDxUIHwgVFtdPn0gRXZlbnQgZW1pdHRlZCBvbiBkYXRlIHNlbGVjdGlvbi4gKi9cbiAgICB0aGlzLmRpc3BhdGNoRXZlbnQoXG4gICAgICBuZXcgQ3VzdG9tRXZlbnQ8VCB8IFRbXT4oJ2RhdGVzZWxlY3RlZCcsIHtcbiAgICAgICAgZGV0YWlsLFxuICAgICAgICBjb21wb3NlZDogdHJ1ZSxcbiAgICAgICAgYnViYmxlczogdHJ1ZSxcbiAgICAgIH0pLFxuICAgICk7XG4gIH1cblxuICBwcml2YXRlIF9lbWl0TW9udGhDaGFuZ2UoKTogdm9pZCB7XG4gICAgLy8gRklYTUU6IHRoZSBuYW1lIG9mIHRoaXMgdmFyaWFibGUgYXBwZWFycyBhcyBldmVudCBuYW1lIGluIHRoZSByZWFkbWVcbiAgICAvLyAgZHVlIHRvIGEgYnVnIGluIHRoZSBjdXN0b20tZWxlbWVudHMtbWFuaWZlc3QgbGlicmFyeS5cbiAgICAvLyAgaHR0cHM6Ly9naXRodWIuY29tL29wZW4td2MvY3VzdG9tLWVsZW1lbnRzLW1hbmlmZXN0L2lzc3Vlcy8xNDlcbiAgICBjb25zdCBtb250aGNoYW5nZSA9ICh0aGlzLndpZGUgPyBbLi4udGhpcy5fd2Vla3MsIC4uLnRoaXMuX25leHRNb250aFdlZWtzXSA6IHRoaXMuX3dlZWtzKVxuICAgICAgLmZsYXQoKVxuICAgICAgLnNvcnQoKGEsIGIpID0+IGEudmFsdWUubG9jYWxlQ29tcGFyZShiLnZhbHVlKSk7XG4gICAgLyoqXG4gICAgICogQHR5cGUge1NiYk1vbnRoQ2hhbmdlRXZlbnR9XG4gICAgICogRW1pdHMgd2hlbiB0aGUgbW9udGggY2hhbmdlcy5cbiAgICAgKiBUaGUgYHJhbmdlYCBwcm9wZXJ0eSBjb250YWlucyB0aGUgZGF5cyBhcnJheSBvZiB0aGUgY2hvc2VuIG1vbnRoLlxuICAgICAqL1xuICAgIHRoaXMuZGlzcGF0Y2hFdmVudChuZXcgU2JiTW9udGhDaGFuZ2VFdmVudChtb250aGNoYW5nZSkpO1xuICB9XG5cbiAgLyoqXG4gICAqIEluIGNhc2Ugb2YgbXVsdGlwbGUgc2VsZWN0aW9uLCBuZXdseSBhZGRlZCBkYXlzIG11c3QgYmUgYWRkZWQgdG8gdGhlIGV4aXN0aW5nIG9uZXMsIHdpdGhvdXQgZHVwbGljYXRpb24uXG4gICAqIElmIHRoZSBkYXlzIHRvIGFkZCBhcmUgZXhhY3RseSB0aGUgc2FtZSBhcyB0aGUgc2VsZWN0ZWQgb25lcywgdGhlIHNldCBtdXN0IGJlIGVtcHRpZWQuXG4gICAqL1xuICBwcml2YXRlIF91cGRhdGVTZWxlY3RlZFdpdGhNdWx0aXBsZURhdGVzKFxuICAgIGRheXNUb0FkZDogc3RyaW5nW10sXG4gICAgZGF5c1RvQWRkU2V0OiBTZXQ8c3RyaW5nPixcbiAgICBzZWxlY3RlZFNldDogU2V0PHN0cmluZz4sXG4gICk6IHN0cmluZ1tdIHtcbiAgICBpZiAoZGF5c1RvQWRkLmV2ZXJ5KChkYXk6IHN0cmluZykgPT4gc2VsZWN0ZWRTZXQuaGFzKGRheSkpKSB7XG4gICAgICBkYXlzVG9BZGRTZXQuZm9yRWFjaCgoZGF5OiBzdHJpbmcpID0+IHNlbGVjdGVkU2V0LmRlbGV0ZShkYXkpKTtcbiAgICB9IGVsc2Uge1xuICAgICAgZGF5c1RvQWRkU2V0LmZvckVhY2goKGRheTogc3RyaW5nKSA9PiBzZWxlY3RlZFNldC5hZGQoZGF5KSk7XG4gICAgfVxuICAgIHJldHVybiBBcnJheS5mcm9tKHNlbGVjdGVkU2V0KTtcbiAgfVxuXG4gIHByaXZhdGUgX3NldENob3NlblllYXIoKTogdm9pZCB7XG4gICAgaWYgKHRoaXMudmlldyA9PT0gJ21vbnRoJykge1xuICAgICAgbGV0IHNlbGVjdGVkRGF0ZTogVCB8IHVuZGVmaW5lZDtcbiAgICAgIGlmICh0aGlzLm11bHRpcGxlKSB7XG4gICAgICAgIHNlbGVjdGVkRGF0ZSA9ICh0aGlzLnNlbGVjdGVkIGFzIFRbXSkuYXQoLTEpO1xuICAgICAgfSBlbHNlIHtcbiAgICAgICAgc2VsZWN0ZWREYXRlID0gdGhpcy5zZWxlY3RlZCBhcyBUO1xuICAgICAgfVxuICAgICAgdGhpcy5fY2hvc2VuWWVhciA9IHRoaXMuX2RhdGVBZGFwdGVyLmdldFllYXIoc2VsZWN0ZWREYXRlID8/IHRoaXMuX2RhdGVBZGFwdGVyLnRvZGF5KCkpO1xuICAgIH0gZWxzZSB7XG4gICAgICB0aGlzLl9jaG9zZW5ZZWFyID0gdW5kZWZpbmVkO1xuICAgIH1cbiAgfVxuXG4gIHByaXZhdGUgX2Fzc2lnbkFjdGl2ZURhdGUoZGF0ZTogVCk6IHZvaWQge1xuICAgIGlmICh0aGlzLm1pbiAmJiB0aGlzLl9kYXRlQWRhcHRlci5jb21wYXJlRGF0ZSh0aGlzLm1pbiwgZGF0ZSkgPiAwKSB7XG4gICAgICB0aGlzLl9hY3RpdmVEYXRlID0gdGhpcy5taW47XG4gICAgICByZXR1cm47XG4gICAgfVxuICAgIGlmICh0aGlzLm1heCAmJiB0aGlzLl9kYXRlQWRhcHRlci5jb21wYXJlRGF0ZSh0aGlzLm1heCwgZGF0ZSkgPCAwKSB7XG4gICAgICB0aGlzLl9hY3RpdmVEYXRlID0gdGhpcy5tYXg7XG4gICAgICByZXR1cm47XG4gICAgfVxuICAgIHRoaXMuX2FjdGl2ZURhdGUgPSBkYXRlO1xuICB9XG5cbiAgLyoqIEdvZXMgdG8gdGhlIG1vbnRoIGlkZW50aWZpZWQgYnkgdGhlIHNoaWZ0LiAqL1xuICBwcml2YXRlIF9nb1RvRGlmZmVyZW50TW9udGgobW9udGhzOiBudW1iZXIpOiB2b2lkIHtcbiAgICB0aGlzLl9pbml0KHRoaXMuX2RhdGVBZGFwdGVyLmFkZENhbGVuZGFyTW9udGhzKHRoaXMuX2FjdGl2ZURhdGUsIG1vbnRocykpO1xuICAgIHRoaXMuX2VtaXRNb250aENoYW5nZSgpO1xuICB9XG5cbiAgcHJpdmF0ZSBfZ29Ub0RpZmZlcmVudFllYXIoeWVhcnM6IG51bWJlcik6IHZvaWQge1xuICAgIHRoaXMuX2Nob3NlblllYXIhICs9IHllYXJzO1xuICAgIC8vIENhbid0IHVzZSBgX2Fzc2lnbkFjdGl2ZURhdGUoLi4uKWAgaGVyZSwgYmVjYXVzZSBpdCB3aWxsIHNldCBpdCB0byBtaW4vbWF4IHZhbHVlIGlmIGFyZ3VtZW50IGlzIG91dCBvZiByYW5nZVxuICAgIHRoaXMuX2FjdGl2ZURhdGUgPSB0aGlzLl9kYXRlQWRhcHRlci5jcmVhdGVEYXRlKFxuICAgICAgdGhpcy5fY2hvc2VuWWVhciEsXG4gICAgICB0aGlzLl9kYXRlQWRhcHRlci5nZXRNb250aCh0aGlzLl9hY3RpdmVEYXRlKSxcbiAgICAgIHRoaXMuX2RhdGVBZGFwdGVyLmdldERhdGUodGhpcy5fYWN0aXZlRGF0ZSksXG4gICAgKTtcbiAgICB0aGlzLl9pbml0KCk7XG4gIH1cblxuICBwcml2YXRlIF9nb1RvRGlmZmVyZW50WWVhclJhbmdlKHllYXJzOiBudW1iZXIpOiB2b2lkIHtcbiAgICB0aGlzLl9pbml0KHRoaXMuX2RhdGVBZGFwdGVyLmFkZENhbGVuZGFyWWVhcnModGhpcy5fYWN0aXZlRGF0ZSwgeWVhcnMpKTtcbiAgfVxuXG4gIHByaXZhdGUgX3ByZXZEaXNhYmxlZChwcmV2RGF0ZTogVCk6IGJvb2xlYW4ge1xuICAgIGlmICghdGhpcy5taW4pIHtcbiAgICAgIHJldHVybiBmYWxzZTtcbiAgICB9XG4gICAgcmV0dXJuIHRoaXMuX2RhdGVBZGFwdGVyLmNvbXBhcmVEYXRlKHByZXZEYXRlLCB0aGlzLm1pbikgPCAwO1xuICB9XG5cbiAgcHJpdmF0ZSBfbmV4dERpc2FibGVkKG5leHREYXRlOiBUKTogYm9vbGVhbiB7XG4gICAgaWYgKCF0aGlzLm1heCkge1xuICAgICAgcmV0dXJuIGZhbHNlO1xuICAgIH1cbiAgICByZXR1cm4gdGhpcy5fZGF0ZUFkYXB0ZXIuY29tcGFyZURhdGUobmV4dERhdGUsIHRoaXMubWF4KSA+IDA7XG4gIH1cblxuICAvKiogQ2hlY2tzIGlmIHRoZSBcInByZXZpb3VzIG1vbnRoXCIgYnV0dG9uIHNob3VsZCBiZSBkaXNhYmxlZC4gKi9cbiAgcHJpdmF0ZSBfcHJldmlvdXNNb250aERpc2FibGVkKCk6IGJvb2xlYW4ge1xuICAgIGNvbnN0IHByZXZNb250aCA9IHRoaXMuX2RhdGVBZGFwdGVyLmFkZENhbGVuZGFyRGF5cyhcbiAgICAgIHRoaXMuX2FjdGl2ZURhdGUsXG4gICAgICB0aGlzLl9kYXRlQWRhcHRlci5nZXREYXRlKHRoaXMuX2FjdGl2ZURhdGUpICogLTEsXG4gICAgKTtcbiAgICByZXR1cm4gdGhpcy5fcHJldkRpc2FibGVkKHByZXZNb250aCk7XG4gIH1cblxuICAvKiogQ2hlY2tzIGlmIHRoZSBcIm5leHQgbW9udGhcIiBidXR0b24gc2hvdWxkIGJlIGRpc2FibGVkLiAqL1xuICBwcml2YXRlIF9uZXh0TW9udGhEaXNhYmxlZCgpOiBib29sZWFuIHtcbiAgICBsZXQgbmV4dE1vbnRoID0gdGhpcy5fZGF0ZUFkYXB0ZXIuYWRkQ2FsZW5kYXJNb250aHModGhpcy5fYWN0aXZlRGF0ZSwgdGhpcy5fd2lkZSA/IDIgOiAxKTtcbiAgICBuZXh0TW9udGggPSB0aGlzLl9kYXRlQWRhcHRlci5jcmVhdGVEYXRlKFxuICAgICAgdGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0WWVhcihuZXh0TW9udGgpLFxuICAgICAgdGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0TW9udGgobmV4dE1vbnRoKSxcbiAgICAgIDEsXG4gICAgKTtcbiAgICByZXR1cm4gdGhpcy5fbmV4dERpc2FibGVkKG5leHRNb250aCk7XG4gIH1cblxuICAvKiogQ2hlY2tzIGlmIHRoZSBcInByZXZpb3VzIHllYXJcIiBidXR0b24gc2hvdWxkIGJlIGRpc2FibGVkLiAqL1xuICBwcml2YXRlIF9wcmV2aW91c1llYXJEaXNhYmxlZCgpOiBib29sZWFuIHtcbiAgICBjb25zdCBwcmV2WWVhciA9IHRoaXMuX2RhdGVBZGFwdGVyLmNyZWF0ZURhdGUoXG4gICAgICB0aGlzLl9kYXRlQWRhcHRlci5nZXRZZWFyKHRoaXMuX2FjdGl2ZURhdGUpIC0gMSxcbiAgICAgIDEyLFxuICAgICAgMzEsXG4gICAgKTtcbiAgICByZXR1cm4gdGhpcy5fcHJldkRpc2FibGVkKHByZXZZZWFyKTtcbiAgfVxuXG4gIC8qKiBDaGVja3MgaWYgdGhlIFwibmV4dCB5ZWFyXCIgYnV0dG9uIHNob3VsZCBiZSBkaXNhYmxlZC4gKi9cbiAgcHJpdmF0ZSBfbmV4dFllYXJEaXNhYmxlZCgpOiBib29sZWFuIHtcbiAgICBjb25zdCBuZXh0WWVhciA9IHRoaXMuX2RhdGVBZGFwdGVyLmNyZWF0ZURhdGUoXG4gICAgICB0aGlzLl9kYXRlQWRhcHRlci5nZXRZZWFyKHRoaXMuX2FjdGl2ZURhdGUpICsgKHRoaXMuX3dpZGUgPyAyIDogMSksXG4gICAgICAxLFxuICAgICAgMSxcbiAgICApO1xuICAgIHJldHVybiB0aGlzLl9uZXh0RGlzYWJsZWQobmV4dFllYXIpO1xuICB9XG5cbiAgLyoqIENoZWNrcyBpZiB0aGUgXCJwcmV2aW91cyB5ZWFyXCIgYnV0dG9uIHNob3VsZCBiZSBkaXNhYmxlZCBpbiB5ZWFyIHZpZXcuICovXG4gIHByaXZhdGUgX3ByZXZpb3VzWWVhclJhbmdlRGlzYWJsZWQoKTogYm9vbGVhbiB7XG4gICAgY29uc3QgcHJldlllYXIgPSB0aGlzLl9kYXRlQWRhcHRlci5jcmVhdGVEYXRlKHRoaXMuX3llYXJzWzBdWzBdIC0gMSwgMTIsIDMxKTtcbiAgICByZXR1cm4gdGhpcy5fcHJldkRpc2FibGVkKHByZXZZZWFyKTtcbiAgfVxuXG4gIC8qKiBDaGVja3MgaWYgdGhlIFwibmV4dCB5ZWFyXCIgYnV0dG9uIHNob3VsZCBiZSBkaXNhYmxlZCBpbiB5ZWFyIHZpZXcuICovXG4gIHByaXZhdGUgX25leHRZZWFyUmFuZ2VEaXNhYmxlZCgpOiBib29sZWFuIHtcbiAgICBjb25zdCB5ZWFycyA9IHRoaXMuX3dpZGUgPyB0aGlzLl9uZXh0TW9udGhZZWFycyA6IHRoaXMuX3llYXJzO1xuICAgIGNvbnN0IGxhc3RZZWFyUmFuZ2UgPSB5ZWFyc1t5ZWFycy5sZW5ndGggLSAxXTtcbiAgICBjb25zdCBsYXN0WWVhciA9IGxhc3RZZWFyUmFuZ2VbbGFzdFllYXJSYW5nZS5sZW5ndGggLSAxXTtcbiAgICBjb25zdCBuZXh0WWVhciA9IHRoaXMuX2RhdGVBZGFwdGVyLmNyZWF0ZURhdGUobGFzdFllYXIgKyAxLCAxLCAxKTtcbiAgICByZXR1cm4gdGhpcy5fbmV4dERpc2FibGVkKG5leHRZZWFyKTtcbiAgfVxuXG4gIHByaXZhdGUgX2hhbmRsZVRhYmxlQmx1cihldmVudFRhcmdldDogSFRNTEVsZW1lbnQpOiB2b2lkIHtcbiAgICBpZiAoZXZlbnRUYXJnZXQ/LmxvY2FsTmFtZSAhPT0gJ3NiYi1jYWxlbmRhci1kYXknKSB7XG4gICAgICB0aGlzLl9zZXRUYWJJbmRleCgpO1xuICAgIH1cbiAgfVxuXG4gIHByaXZhdGUgX3NldFRhYkluZGV4KCk6IHZvaWQge1xuICAgIGNvbnN0IHF1ZXJ5ID0gdGhpcy5fY2FsZW5kYXJWaWV3ID09PSAnZGF5JyA/ICdzYmItY2FsZW5kYXItZGF5JyA6ICcuc2JiLWNhbGVuZGFyX19jZWxsJztcbiAgICBBcnJheS5mcm9tKFxuICAgICAgdGhpcy5fZ2V0Um9vdEZvclF1ZXJ5U2VsZWN0b3IoKS5xdWVyeVNlbGVjdG9yQWxsKGAke3F1ZXJ5fVt0YWJpbmRleD1cIjBcIl1gKSA/PyBbXSxcbiAgICApLmZvckVhY2goKGRheSkgPT4gKChkYXkgYXMgSFRNTEJ1dHRvbkVsZW1lbnQgfCBTYmJDYWxlbmRhckRheUVsZW1lbnQpLnRhYkluZGV4ID0gLTEpKTtcbiAgICBjb25zdCBmaXJzdEZvY3VzYWJsZSA9IHRoaXMuX2dldEZpcnN0Rm9jdXNhYmxlKCk7XG4gICAgaWYgKGZpcnN0Rm9jdXNhYmxlKSB7XG4gICAgICBmaXJzdEZvY3VzYWJsZS50YWJJbmRleCA9IDA7XG4gICAgfVxuICB9XG5cbiAgLyoqIEdldCB0aGUgZWxlbWVudCBpbiB0aGUgY2FsZW5kYXIgdG8gYXNzaWduIGZvY3VzLiAqL1xuICBwcml2YXRlIF9nZXRGaXJzdEZvY3VzYWJsZSgpOiBTYmJDYWxlbmRhckRheUVsZW1lbnQgfCBIVE1MQnV0dG9uRWxlbWVudCB8IG51bGwge1xuICAgIGNvbnN0IHJvb3QgPSB0aGlzLl9nZXRSb290Rm9yUXVlcnlTZWxlY3RvcigpO1xuICAgIGlmICh0aGlzLl9jYWxlbmRhclZpZXcgPT09ICdkYXknKSB7XG4gICAgICBjb25zdCBzZWxlY3RlZE9yQ3VycmVudCA9XG4gICAgICAgIHJvb3QucXVlcnlTZWxlY3RvcjxTYmJDYWxlbmRhckRheUVsZW1lbnQ+KCc6c3RhdGUoc2VsZWN0ZWQpJykgPz9cbiAgICAgICAgcm9vdC5xdWVyeVNlbGVjdG9yPFNiYkNhbGVuZGFyRGF5RWxlbWVudD4oJzpzdGF0ZShjdXJyZW50KScpO1xuICAgICAgcmV0dXJuIHNlbGVjdGVkT3JDdXJyZW50ICYmICFzZWxlY3RlZE9yQ3VycmVudC5kaXNhYmxlZFxuICAgICAgICA/IHNlbGVjdGVkT3JDdXJyZW50XG4gICAgICAgIDogdGhpcy5fZ2V0Rmlyc3RGb2N1c2FibGVEYXkoKTtcbiAgICB9IGVsc2Uge1xuICAgICAgY29uc3Qgc2VsZWN0ZWRPckN1cnJlbnQgPSB0aGlzLnNoYWRvd1Jvb3Q/LnF1ZXJ5U2VsZWN0b3I8SFRNTEJ1dHRvbkVsZW1lbnQ+KFxuICAgICAgICAnLnNiYi1jYWxlbmRhcl9fY2VsbC1jdXJyZW50JyxcbiAgICAgICk7XG4gICAgICByZXR1cm4gc2VsZWN0ZWRPckN1cnJlbnQgJiYgIXNlbGVjdGVkT3JDdXJyZW50LmRpc2FibGVkXG4gICAgICAgID8gc2VsZWN0ZWRPckN1cnJlbnRcbiAgICAgICAgOiB0aGlzLnNoYWRvd1Jvb3QhLnF1ZXJ5U2VsZWN0b3I8SFRNTEJ1dHRvbkVsZW1lbnQ+KCcuc2JiLWNhbGVuZGFyX19jZWxsOm5vdChbZGlzYWJsZWRdKScpO1xuICAgIH1cbiAgfVxuXG4gIC8qKlxuICAgKiBJbiBgZGF5YCB2aWV3IGluIGB2ZXJ0aWNhbGAgb3JpZW50YXRpb24sXG4gICAqIGlmIHRoZSBmaXJzdCBvZiB0aGUgbW9udGggaXMgbm90IGEgTW9uZGF5LCBpdCBpcyBub3QgdGhlIGZpcnN0IHJlbmRlcmVkIGVsZW1lbnQgaW4gdGhlIHRhYmxlLFxuICAgKiBzbyBgdGhpcy5zaGFkb3dSb290IS5xdWVyeVNlbGVjdG9yKCcuc2JiLWNhbGVuZGFyX19jZWxsOm5vdChbZGlzYWJsZWRdKScpYCB3aWxsIHJldHVybiBhIHdyb25nIHZhbHVlLlxuICAgKlxuICAgKiBUbyBzb2x2ZSB0aGlzLCB0aGUgZWxlbWVudCB3aXRoIHRoZSBsb3dlc3QgYHZhbHVlYCBpcyB0YWtlbiAoSVNPIFN0cmluZyBhcmUgb3JkZXJlZCkuXG4gICAqL1xuICBwcml2YXRlIF9nZXRGaXJzdEZvY3VzYWJsZURheSgpOiBTYmJDYWxlbmRhckRheUVsZW1lbnQgfCBudWxsIHtcbiAgICBjb25zdCBkYXlzSW5WaWV3OiBTYmJDYWxlbmRhckRheUVsZW1lbnRbXSA9IEFycmF5LmZyb20oXG4gICAgICB0aGlzLl9nZXRSb290Rm9yUXVlcnlTZWxlY3RvcigpLnF1ZXJ5U2VsZWN0b3JBbGwoJ3NiYi1jYWxlbmRhci1kYXk6bm90KFtkaXNhYmxlZF0pJyksXG4gICAgKTtcbiAgICBpZiAoIWRheXNJblZpZXcgfHwgZGF5c0luVmlldy5sZW5ndGggPT09IDApIHtcbiAgICAgIHJldHVybiBudWxsO1xuICAgIH0gZWxzZSB7XG4gICAgICBjb25zdCBmaXJzdEVsZW1lbnQgPSBkYXlzSW5WaWV3XG4gICAgICAgIC5tYXAoKGU6IFNiYkNhbGVuZGFyRGF5RWxlbWVudCk6IHN0cmluZyA9PiB0aGlzLl9kYXRlQWRhcHRlci50b0lzbzg2MDEoZS52YWx1ZSEgYXMgVCkpXG4gICAgICAgIC5zb3J0KClbMF07XG4gICAgICByZXR1cm4gdGhpcy5fZ2V0Um9vdEZvclF1ZXJ5U2VsZWN0b3IoKS5xdWVyeVNlbGVjdG9yKFxuICAgICAgICBgc2JiLWNhbGVuZGFyLWRheVtzbG90PVwiJHtmaXJzdEVsZW1lbnR9XCJdYCxcbiAgICAgICk7XG4gICAgfVxuICB9XG5cbiAgcHJpdmF0ZSBfaGFuZGxlS2V5Ym9hcmRFdmVudChldmVudDogS2V5Ym9hcmRFdmVudCwgZGF5PzogRGF5PFQ+KTogdm9pZCB7XG4gICAgaWYgKGlzQXJyb3dLZXlPclBhZ2VLZXlzUHJlc3NlZChldmVudCkpIHtcbiAgICAgIGV2ZW50LnByZXZlbnREZWZhdWx0KCk7XG4gICAgfVxuICAgIC8vIEdldHMgdGhlIGN1cnJlbnRseSByZW5kZXJlZCB0YWJsZSdzIGNlbGw7XG4gICAgLy8gdGhleSBjb3VsZCBiZSBkYXlzLCBtb250aHMgb3IgeWVhcnMgYmFzZWQgb24gdGhlIGN1cnJlbnQgc2VsZWN0aW9uIHZpZXcuXG4gICAgLy8gSWYgYHdpZGVgIGlzIHRydWUsIHllYXJzIGFyZSBkb3VibGVkIGluIG51bWJlciBhbmQgZGF5cyBhcmUgKHJvdWdobHkpIGRvdWJsZWQgdG9vLCBhZmZlY3RpbmcgdGhlIGBpbmRleGAgY2FsY3VsYXRpb24uXG4gICAgY29uc3QgY2VsbHMgPSB0aGlzLl9jZWxscztcbiAgICBjb25zdCBpbmRleDogbnVtYmVyID0gY2VsbHMuZmluZEluZGV4KChlKSA9PiBlID09PSBldmVudC50YXJnZXQpO1xuICAgIGxldCBuZXh0RWw6IEhUTUxCdXR0b25FbGVtZW50IHwgU2JiQ2FsZW5kYXJEYXlFbGVtZW50O1xuICAgIGlmIChkYXkpIHtcbiAgICAgIG5leHRFbCA9IHRoaXMuX25hdmlnYXRlQnlLZXlib2FyZERheVZpZXcoZXZlbnQsIGluZGV4LCBjZWxscywgZGF5KTtcbiAgICB9IGVsc2Uge1xuICAgICAgbmV4dEVsID0gdGhpcy5fbmF2aWdhdGVCeUtleWJvYXJkKGV2ZW50LCBpbmRleCwgY2VsbHMgYXMgSFRNTEJ1dHRvbkVsZW1lbnRbXSk7XG4gICAgfVxuICAgIGNvbnN0IGFjdGl2ZUVsID0gKHRoaXMuX2VuaGFuY2VkVmFyaWFudCA/IGRvY3VtZW50IDogdGhpcy5zaGFkb3dSb290ISkuYWN0aXZlRWxlbWVudCBhc1xuICAgICAgfCBIVE1MQnV0dG9uRWxlbWVudFxuICAgICAgfCBTYmJDYWxlbmRhckRheUVsZW1lbnQ7XG4gICAgaWYgKG5leHRFbCAhPT0gYWN0aXZlRWwpIHtcbiAgICAgIG5leHRFbC50YWJJbmRleCA9IDA7XG4gICAgICBuZXh0RWw/LmZvY3VzKCk7XG4gICAgICBhY3RpdmVFbC50YWJJbmRleCA9IC0xO1xuICAgIH1cbiAgfVxuXG4gIHByaXZhdGUgX25hdmlnYXRlQnlLZXlib2FyZERheVZpZXcoXG4gICAgZXZ0OiBLZXlib2FyZEV2ZW50LFxuICAgIGluZGV4OiBudW1iZXIsXG4gICAgY2VsbHM6IChIVE1MQnV0dG9uRWxlbWVudCB8IFNiYkNhbGVuZGFyRGF5RWxlbWVudClbXSxcbiAgICBkYXk6IERheTxUPixcbiAgKTogSFRNTEJ1dHRvbkVsZW1lbnQgfCBTYmJDYWxlbmRhckRheUVsZW1lbnQge1xuICAgIGNvbnN0IGFycm93c09mZnNldCA9XG4gICAgICB0aGlzLm9yaWVudGF0aW9uID09PSAnaG9yaXpvbnRhbCdcbiAgICAgICAgPyB7IGxlZnRSaWdodDogMSwgdXBEb3duOiBEQVlTX1BFUl9ST1cgfVxuICAgICAgICA6IHsgbGVmdFJpZ2h0OiBEQVlTX1BFUl9ST1csIHVwRG93bjogMSB9O1xuICAgIGNvbnN0IG9mZnNldEZvclZlcnRpY2FsOiBudW1iZXIgPVxuICAgICAgaW5kZXggPCB0aGlzLl9rZXlib2FyZE5hdmlnYXRpb25EYXlWaWV3UGFyYW1ldGVycy5maXJzdE1vbnRoTGVuZ3RoXG4gICAgICAgID8gdGhpcy5fa2V5Ym9hcmROYXZpZ2F0aW9uRGF5Vmlld1BhcmFtZXRlcnMuZmlyc3RNb250aE9mZnNldFxuICAgICAgICA6IHRoaXMuX2tleWJvYXJkTmF2aWdhdGlvbkRheVZpZXdQYXJhbWV0ZXJzLnNlY29uZE1vbnRoT2Zmc2V0O1xuXG4gICAgc3dpdGNoIChldnQua2V5KSB7XG4gICAgICBjYXNlICdBcnJvd1VwJzpcbiAgICAgICAgcmV0dXJuIHRoaXMuX2ZpbmREYXlBcnJvd3MoY2VsbHMsIGluZGV4LCBkYXkuZGF0ZVZhbHVlLCAtYXJyb3dzT2Zmc2V0LnVwRG93bik7XG4gICAgICBjYXNlICdBcnJvd0Rvd24nOlxuICAgICAgICByZXR1cm4gdGhpcy5fZmluZERheUFycm93cyhjZWxscywgaW5kZXgsIGRheS5kYXRlVmFsdWUsIGFycm93c09mZnNldC51cERvd24pO1xuICAgICAgY2FzZSAnQXJyb3dMZWZ0JzpcbiAgICAgICAgcmV0dXJuIHRoaXMuX2ZpbmREYXlBcnJvd3MoY2VsbHMsIGluZGV4LCBkYXkuZGF0ZVZhbHVlLCAtYXJyb3dzT2Zmc2V0LmxlZnRSaWdodCk7XG4gICAgICBjYXNlICdBcnJvd1JpZ2h0JzpcbiAgICAgICAgcmV0dXJuIHRoaXMuX2ZpbmREYXlBcnJvd3MoY2VsbHMsIGluZGV4LCBkYXkuZGF0ZVZhbHVlLCBhcnJvd3NPZmZzZXQubGVmdFJpZ2h0KTtcbiAgICAgIGNhc2UgJ1BhZ2VVcCc6IHtcbiAgICAgICAgaWYgKHRoaXMub3JpZW50YXRpb24gPT09ICdob3Jpem9udGFsJykge1xuICAgICAgICAgIGNvbnN0IGZpcnN0T2ZXZWVrOiBudW1iZXIgPSArZGF5LmRheVZhbHVlICUgREFZU19QRVJfUk9XIHx8IERBWVNfUEVSX1JPVztcbiAgICAgICAgICBjb25zdCBkZWx0YTogbnVtYmVyID0gZmlyc3RPZldlZWsgLSArZGF5LmRheVZhbHVlO1xuICAgICAgICAgIHJldHVybiB0aGlzLl9maW5kRGF5UGFnZVVwRG93bihjZWxscywgaW5kZXgsIGRheSwgZGVsdGEsIGFycm93c09mZnNldC51cERvd24pO1xuICAgICAgICB9IGVsc2Uge1xuICAgICAgICAgIGNvbnN0IHdlZWtOdW1iZXI6IG51bWJlciA9IE1hdGguY2VpbCgoK2RheS5kYXlWYWx1ZSArIG9mZnNldEZvclZlcnRpY2FsKSAvIERBWVNfUEVSX1JPVyk7XG4gICAgICAgICAgY29uc3QgZmlyc3RPZldlZWs6IG51bWJlciA9ICh3ZWVrTnVtYmVyIC0gMSkgKiBEQVlTX1BFUl9ST1cgLSBvZmZzZXRGb3JWZXJ0aWNhbCArIDE7XG4gICAgICAgICAgY29uc3QgZGVsdGE6IG51bWJlciA9IGZpcnN0T2ZXZWVrIC0gK2RheS5kYXlWYWx1ZTtcbiAgICAgICAgICByZXR1cm4gdGhpcy5fZmluZERheVBhZ2VVcERvd24oY2VsbHMsIGluZGV4LCBkYXksIGRlbHRhLCBhcnJvd3NPZmZzZXQudXBEb3duKTtcbiAgICAgICAgfVxuICAgICAgfVxuICAgICAgY2FzZSAnUGFnZURvd24nOiB7XG4gICAgICAgIGlmICh0aGlzLm9yaWVudGF0aW9uID09PSAnaG9yaXpvbnRhbCcpIHtcbiAgICAgICAgICBjb25zdCBtb250aEluQm91bmRzID0gK2RheS5tb250aFZhbHVlICsgMSA+IDEyID8gMSA6ICtkYXkubW9udGhWYWx1ZSArIDE7XG4gICAgICAgICAgY29uc3QgeWVhckluQm91bmRzID0gK2RheS5tb250aFZhbHVlICsgMSA+IDEyID8gK2RheS55ZWFyVmFsdWUgKyAxIDogK2RheS55ZWFyVmFsdWU7XG4gICAgICAgICAgY29uc3QgZmlyc3ROZXh0TW9udGg6IFQgPSB0aGlzLl9kYXRlQWRhcHRlci5jcmVhdGVEYXRlKHllYXJJbkJvdW5kcywgbW9udGhJbkJvdW5kcywgMSk7XG4gICAgICAgICAgY29uc3QgbGFzdE9mTW9udGg6IG51bWJlciA9IHRoaXMuX2RhdGVBZGFwdGVyLmdldERhdGUoXG4gICAgICAgICAgICB0aGlzLl9kYXRlQWRhcHRlci5hZGRDYWxlbmRhckRheXMoZmlyc3ROZXh0TW9udGgsIC0xKSxcbiAgICAgICAgICApO1xuICAgICAgICAgIGNvbnN0IGRlbHRhOiBudW1iZXIgPVxuICAgICAgICAgICAgTWF0aC50cnVuYygobGFzdE9mTW9udGggLSArZGF5LmRheVZhbHVlISkgLyBEQVlTX1BFUl9ST1cpICogREFZU19QRVJfUk9XO1xuICAgICAgICAgIHJldHVybiB0aGlzLl9maW5kRGF5UGFnZVVwRG93bihjZWxscywgaW5kZXgsIGRheSwgZGVsdGEsIC1hcnJvd3NPZmZzZXQudXBEb3duKTtcbiAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICBjb25zdCB3ZWVrTnVtYmVyOiBudW1iZXIgPSBNYXRoLmNlaWwoKCtkYXkuZGF5VmFsdWUgKyBvZmZzZXRGb3JWZXJ0aWNhbCkgLyBEQVlTX1BFUl9ST1cpO1xuICAgICAgICAgIGNvbnN0IGxhc3RPZldlZWs6IG51bWJlciA9IHdlZWtOdW1iZXIgKiBEQVlTX1BFUl9ST1cgLSBvZmZzZXRGb3JWZXJ0aWNhbDtcbiAgICAgICAgICBjb25zdCBkZWx0YTogbnVtYmVyID0gbGFzdE9mV2VlayAtICtkYXkuZGF5VmFsdWU7XG4gICAgICAgICAgcmV0dXJuIHRoaXMuX2ZpbmREYXlQYWdlVXBEb3duKGNlbGxzLCBpbmRleCwgZGF5LCBkZWx0YSwgLWFycm93c09mZnNldC51cERvd24pO1xuICAgICAgICB9XG4gICAgICB9XG4gICAgICBjYXNlICdIb21lJzoge1xuICAgICAgICByZXR1cm4gdGhpcy5fZmluZERheUZpcnN0KGNlbGxzLCBpbmRleCwgZGF5LCAxKTtcbiAgICAgIH1cbiAgICAgIGNhc2UgJ0VuZCc6IHtcbiAgICAgICAgY29uc3QgbW9udGhJbkJvdW5kcyA9ICtkYXkubW9udGhWYWx1ZSArIDEgPiAxMiA/IDEgOiArZGF5Lm1vbnRoVmFsdWUgKyAxO1xuICAgICAgICBjb25zdCB5ZWFySW5Cb3VuZHMgPSArZGF5Lm1vbnRoVmFsdWUgKyAxID4gMTIgPyArZGF5LnllYXJWYWx1ZSArIDEgOiArZGF5LnllYXJWYWx1ZTtcbiAgICAgICAgY29uc3QgZmlyc3ROZXh0TW9udGg6IFQgPSB0aGlzLl9kYXRlQWRhcHRlci5jcmVhdGVEYXRlKHllYXJJbkJvdW5kcywgbW9udGhJbkJvdW5kcywgMSk7XG4gICAgICAgIHJldHVybiB0aGlzLl9maW5kRGF5TGFzdChjZWxscywgaW5kZXgsIGZpcnN0TmV4dE1vbnRoKTtcbiAgICAgIH1cbiAgICAgIGRlZmF1bHQ6XG4gICAgICAgIHJldHVybiBjZWxsc1tpbmRleF07XG4gICAgfVxuICB9XG5cbiAgcHJpdmF0ZSBfaXNEYXlPdXRPZlZpZXcoZGF0ZTogc3RyaW5nKTogYm9vbGVhbiB7XG4gICAgcmV0dXJuIChcbiAgICAgIGRhdGUgPCB0aGlzLl9rZXlib2FyZE5hdmlnYXRpb25EYXlWaWV3UGFyYW1ldGVycy5maXJzdERheUluVmlldyEgfHxcbiAgICAgIGRhdGUgPiB0aGlzLl9rZXlib2FyZE5hdmlnYXRpb25EYXlWaWV3UGFyYW1ldGVycy5sYXN0RGF5SW5WaWV3IVxuICAgICk7XG4gIH1cblxuICBwcml2YXRlIF9maW5kRGF5QXJyb3dzKFxuICAgIGNlbGxzOiAoSFRNTEJ1dHRvbkVsZW1lbnQgfCBTYmJDYWxlbmRhckRheUVsZW1lbnQpW10sXG4gICAgaW5kZXg6IG51bWJlcixcbiAgICBkYXRlOiBULFxuICAgIGRlbHRhOiBudW1iZXIsXG4gICk6IEhUTUxCdXR0b25FbGVtZW50IHwgU2JiQ2FsZW5kYXJEYXlFbGVtZW50IHtcbiAgICBjb25zdCBuZXdEYXRlVmFsdWUgPSB0aGlzLl9kYXRlQWRhcHRlci50b0lzbzg2MDEoXG4gICAgICB0aGlzLl9kYXRlQWRhcHRlci5hZGRDYWxlbmRhckRheXMoZGF0ZSwgZGVsdGEpLFxuICAgICk7XG4gICAgaWYgKHRoaXMuX2lzRGF5T3V0T2ZWaWV3KG5ld0RhdGVWYWx1ZSkpIHtcbiAgICAgIHJldHVybiBjZWxsc1tpbmRleF07XG4gICAgfVxuICAgIGNvbnN0IG5leHRDZWxsID0gY2VsbHMuZmluZCgoZSkgPT4gdGhpcy5fbWFwVmFsdWVUb0lTT0RhdGUoZS52YWx1ZSEpID09PSBuZXdEYXRlVmFsdWUpO1xuICAgIGlmICghbmV4dENlbGwgfHwgbmV4dENlbGwuZGlzYWJsZWQpIHtcbiAgICAgIHJldHVybiB0aGlzLl9maW5kRGF5QXJyb3dzKGNlbGxzLCBpbmRleCwgdGhpcy5fZGF0ZUFkYXB0ZXIuZGVzZXJpYWxpemUobmV3RGF0ZVZhbHVlKSEsIGRlbHRhKTtcbiAgICB9XG4gICAgcmV0dXJuIG5leHRDZWxsO1xuICB9XG5cbiAgcHJpdmF0ZSBfZmluZERheVBhZ2VVcERvd24oXG4gICAgY2VsbHM6IChIVE1MQnV0dG9uRWxlbWVudCB8IFNiYkNhbGVuZGFyRGF5RWxlbWVudClbXSxcbiAgICBpbmRleDogbnVtYmVyLFxuICAgIGRheTogRGF5PFQ+LFxuICAgIGRlbHRhOiBudW1iZXIsXG4gICAgZGVsdGFJZkRpc2FibGVkOiBudW1iZXIsXG4gICk6IEhUTUxCdXR0b25FbGVtZW50IHwgU2JiQ2FsZW5kYXJEYXlFbGVtZW50IHtcbiAgICBjb25zdCBuZXdEYXRlVmFsdWUgPSB0aGlzLl9kYXRlQWRhcHRlci50b0lzbzg2MDEoXG4gICAgICB0aGlzLl9kYXRlQWRhcHRlci5hZGRDYWxlbmRhckRheXMoZGF5LmRhdGVWYWx1ZSwgZGVsdGEpLFxuICAgICk7XG4gICAgaWYgKHRoaXMuX2lzRGF5T3V0T2ZWaWV3KG5ld0RhdGVWYWx1ZSkpIHtcbiAgICAgIHJldHVybiBjZWxsc1tpbmRleF07XG4gICAgfVxuICAgIGNvbnN0IG5leHRDZWxsID0gY2VsbHMuZmluZCgoZSkgPT4gdGhpcy5fbWFwVmFsdWVUb0lTT0RhdGUoZS52YWx1ZSEpID09PSBuZXdEYXRlVmFsdWUpO1xuICAgIGlmICghbmV4dENlbGwgfHwgbmV4dENlbGwuZGlzYWJsZWQpIHtcbiAgICAgIHJldHVybiB0aGlzLl9maW5kRGF5UGFnZVVwRG93bihjZWxscywgaW5kZXgsIGRheSwgZGVsdGEgKyBkZWx0YUlmRGlzYWJsZWQsIGRlbHRhSWZEaXNhYmxlZCk7XG4gICAgfVxuICAgIHJldHVybiBuZXh0Q2VsbDtcbiAgfVxuXG4gIHByaXZhdGUgX2ZpbmREYXlGaXJzdChcbiAgICBjZWxsczogKEhUTUxCdXR0b25FbGVtZW50IHwgU2JiQ2FsZW5kYXJEYXlFbGVtZW50KVtdLFxuICAgIGluZGV4OiBudW1iZXIsXG4gICAgZGF5OiBEYXk8VD4sXG4gICAgZGF0ZTogbnVtYmVyLFxuICApOiBIVE1MQnV0dG9uRWxlbWVudCB8IFNiYkNhbGVuZGFyRGF5RWxlbWVudCB7XG4gICAgY29uc3QgbmV3RGF0ZVZhbHVlID0gdGhpcy5fZGF0ZUFkYXB0ZXIudG9Jc284NjAxKFxuICAgICAgdGhpcy5fZGF0ZUFkYXB0ZXIuY3JlYXRlRGF0ZSgrZGF5LnllYXJWYWx1ZSwgK2RheS5tb250aFZhbHVlLCBkYXRlKSxcbiAgICApO1xuICAgIGlmICh0aGlzLl9pc0RheU91dE9mVmlldyhuZXdEYXRlVmFsdWUpKSB7XG4gICAgICByZXR1cm4gY2VsbHNbaW5kZXhdO1xuICAgIH1cbiAgICBjb25zdCBuZXh0Q2VsbCA9IGNlbGxzLmZpbmQoKGUpID0+IHRoaXMuX21hcFZhbHVlVG9JU09EYXRlKGUudmFsdWUhKSA9PT0gbmV3RGF0ZVZhbHVlKTtcbiAgICBpZiAoIW5leHRDZWxsIHx8IG5leHRDZWxsLmRpc2FibGVkKSB7XG4gICAgICByZXR1cm4gdGhpcy5fZmluZERheUZpcnN0KGNlbGxzLCBpbmRleCwgZGF5LCBkYXRlICsgMSk7XG4gICAgfVxuICAgIHJldHVybiBuZXh0Q2VsbDtcbiAgfVxuXG4gIHByaXZhdGUgX2ZpbmREYXlMYXN0KFxuICAgIGNlbGxzOiAoSFRNTEJ1dHRvbkVsZW1lbnQgfCBTYmJDYWxlbmRhckRheUVsZW1lbnQpW10sXG4gICAgaW5kZXg6IG51bWJlcixcbiAgICBmaXJzdE5leHRNb250aDogVCxcbiAgKTogSFRNTEJ1dHRvbkVsZW1lbnQgfCBTYmJDYWxlbmRhckRheUVsZW1lbnQge1xuICAgIGNvbnN0IG5ld0RhdGVWYWx1ZSA9IHRoaXMuX2RhdGVBZGFwdGVyLnRvSXNvODYwMShcbiAgICAgIHRoaXMuX2RhdGVBZGFwdGVyLmFkZENhbGVuZGFyRGF5cyhmaXJzdE5leHRNb250aCwgLTEpLFxuICAgICk7XG4gICAgaWYgKHRoaXMuX2lzRGF5T3V0T2ZWaWV3KG5ld0RhdGVWYWx1ZSkpIHtcbiAgICAgIHJldHVybiBjZWxsc1tpbmRleF07XG4gICAgfVxuICAgIGNvbnN0IG5leHRDZWxsID0gY2VsbHMuZmluZCgoZSkgPT4gdGhpcy5fbWFwVmFsdWVUb0lTT0RhdGUoZS52YWx1ZSEpID09PSBuZXdEYXRlVmFsdWUpO1xuICAgIGlmICghbmV4dENlbGwgfHwgbmV4dENlbGwuZGlzYWJsZWQpIHtcbiAgICAgIHJldHVybiB0aGlzLl9maW5kRGF5TGFzdChjZWxscywgaW5kZXgsIHRoaXMuX2RhdGVBZGFwdGVyLmRlc2VyaWFsaXplKG5ld0RhdGVWYWx1ZSkhKTtcbiAgICB9XG4gICAgcmV0dXJuIG5leHRDZWxsO1xuICB9XG5cbiAgLyoqXG4gICAqIEdldHMgdGhlIGluZGV4IG9mIHRoZSBlbGVtZW50IHRvIG1vdmUgdG8sIGJhc2VkIG9uIGEgbGlzdCBvZiBlbGVtZW50cyAod2hpY2ggY2FuIGJlIHBvdGVudGlhbGx5IGRpc2FibGVkKSxcbiAgICogdGhlIGtleWJvYXJkIGlucHV0IGFuZCB0aGUgcG9zaXRpb24gb2YgdGhlIGN1cnJlbnQgZWxlbWVudCBpbiB0aGUgbGlzdC5cbiAgICogSW4gdGhlIGRheSB2aWV3LCB0aGUgYGRheT86IERheWAgcGFyYW1ldGVyIGlzIG1hbmRhdG9yeSBmb3IgY2FsY3VsYXRpb24sXG4gICAqIHdoaWxlIGluIG1vbnRoIGFuZCB5ZWFyIHZpZXcgaXQncyBub3QgZHVlIHRvIHRoZSBmaXhlZCBhbW91bnQgb2YgcmVuZGVyZWQgY2VsbHMuXG4gICAqL1xuICBwcml2YXRlIF9uYXZpZ2F0ZUJ5S2V5Ym9hcmQoXG4gICAgZXZ0OiBLZXlib2FyZEV2ZW50LFxuICAgIGluZGV4OiBudW1iZXIsXG4gICAgY2VsbHM6IEhUTUxCdXR0b25FbGVtZW50W10sXG4gICk6IEhUTUxCdXR0b25FbGVtZW50IHtcbiAgICBjb25zdCB7XG4gICAgICBlbGVtZW50SW5kZXhGb3JXaWRlTW9kZSxcbiAgICAgIG9mZnNldEZvcldpZGVNb2RlLFxuICAgICAgbGFzdEVsZW1lbnRJbmRleEZvcldpZGVNb2RlLFxuICAgICAgdmVydGljYWxPZmZzZXQsXG4gICAgfTogQ2FsZW5kYXJLZXlib2FyZE5hdmlnYXRpb25Nb250aFllYXJWaWV3c1BhcmFtZXRlcnMgPVxuICAgICAgdGhpcy5fY2FsY3VsYXRlUGFyYW1ldGVyc0ZvcktleWJvYXJkTmF2aWdhdGlvbihpbmRleCwgdGhpcy5fY2FsZW5kYXJWaWV3ID09PSAneWVhcicpO1xuXG4gICAgc3dpdGNoIChldnQua2V5KSB7XG4gICAgICBjYXNlICdBcnJvd1VwJzpcbiAgICAgICAgcmV0dXJuIHRoaXMuX2ZpbmROZXh0KGNlbGxzLCBpbmRleCwgLXZlcnRpY2FsT2Zmc2V0KTtcbiAgICAgIGNhc2UgJ0Fycm93RG93bic6XG4gICAgICAgIHJldHVybiB0aGlzLl9maW5kTmV4dChjZWxscywgaW5kZXgsIHZlcnRpY2FsT2Zmc2V0KTtcbiAgICAgIGNhc2UgJ0Fycm93TGVmdCc6XG4gICAgICAgIHJldHVybiB0aGlzLl9maW5kTmV4dChjZWxscywgaW5kZXgsIC0xKTtcbiAgICAgIGNhc2UgJ0Fycm93UmlnaHQnOlxuICAgICAgICByZXR1cm4gdGhpcy5fZmluZE5leHQoY2VsbHMsIGluZGV4LCAxKTtcbiAgICAgIGNhc2UgJ0hvbWUnOlxuICAgICAgICByZXR1cm4gdGhpcy5fZmluZEZpcnN0KGNlbGxzLCBvZmZzZXRGb3JXaWRlTW9kZSk7XG4gICAgICBjYXNlICdQYWdlVXAnOlxuICAgICAgICByZXR1cm4gdGhpcy5fZmluZEZpcnN0T25Db2x1bW4oXG4gICAgICAgICAgY2VsbHMsXG4gICAgICAgICAgZWxlbWVudEluZGV4Rm9yV2lkZU1vZGUsXG4gICAgICAgICAgb2Zmc2V0Rm9yV2lkZU1vZGUsXG4gICAgICAgICAgdmVydGljYWxPZmZzZXQsXG4gICAgICAgICk7XG4gICAgICBjYXNlICdQYWdlRG93bic6XG4gICAgICAgIHJldHVybiB0aGlzLl9maW5kTGFzdE9uQ29sdW1uKGNlbGxzLCBpbmRleCwgbGFzdEVsZW1lbnRJbmRleEZvcldpZGVNb2RlLCB2ZXJ0aWNhbE9mZnNldCk7XG4gICAgICBjYXNlICdFbmQnOlxuICAgICAgICByZXR1cm4gdGhpcy5fZmluZExhc3QoY2VsbHMsIGxhc3RFbGVtZW50SW5kZXhGb3JXaWRlTW9kZSAtIDEpO1xuICAgICAgZGVmYXVsdDpcbiAgICAgICAgcmV0dXJuIGNlbGxzW2luZGV4XTtcbiAgICB9XG4gIH1cblxuICAvKipcbiAgICogQ2FsY3VsYXRlcyB0aGUgcGFyYW1ldGVycyBuZWVkZWQgaW4ga2V5Ym9hcmQgbmF2aWdhdGlvbiBpbiB5ZWFyIGFuZCBtb250aCB2aWV3LlxuICAgKiBAcGFyYW0gaW5kZXggVGhlIHN0YXJ0aW5nIGVsZW1lbnQncyBpbmRleCBpbiB0aGUgY2VsbCBhcnJheS5cbiAgICogQHBhcmFtIGlzWWVhclZpZXcgV2hldGhlciB0aGUgZGlzcGxheWVkIGB2aWV3YCBpcyB0aGUgeWVhciBvbmUuXG4gICAqL1xuICBwcml2YXRlIF9jYWxjdWxhdGVQYXJhbWV0ZXJzRm9yS2V5Ym9hcmROYXZpZ2F0aW9uKFxuICAgIGluZGV4OiBudW1iZXIsXG4gICAgaXNZZWFyVmlldzogYm9vbGVhbixcbiAgKTogQ2FsZW5kYXJLZXlib2FyZE5hdmlnYXRpb25Nb250aFllYXJWaWV3c1BhcmFtZXRlcnMge1xuICAgIGNvbnN0IGVsZW1lbnRzUGVyUGFnZSA9IGlzWWVhclZpZXcgPyBZRUFSU19QRVJfUEFHRSA6IE1PTlRIU19QRVJfUEFHRTtcbiAgICBjb25zdCBvZmZzZXQ6IG51bWJlciA9IE1hdGgudHJ1bmMoaW5kZXggLyBlbGVtZW50c1BlclBhZ2UpICogZWxlbWVudHNQZXJQYWdlO1xuICAgIGNvbnN0IGluZGV4SW5WaWV3OiBudW1iZXIgPSBvZmZzZXQgPT09IDAgPyBpbmRleCA6IGluZGV4IC0gZWxlbWVudHNQZXJQYWdlO1xuICAgIHJldHVybiB7XG4gICAgICB2ZXJ0aWNhbE9mZnNldDogaXNZZWFyVmlldyA/IFlFQVJTX1BFUl9ST1cgOiBNT05USFNfUEVSX1JPVyxcbiAgICAgIGVsZW1lbnRJbmRleEZvcldpZGVNb2RlOiBpbmRleEluVmlldyxcbiAgICAgIG9mZnNldEZvcldpZGVNb2RlOiBpbmRleCAtIGluZGV4SW5WaWV3LFxuICAgICAgbGFzdEVsZW1lbnRJbmRleEZvcldpZGVNb2RlOiBvZmZzZXQgPT09IDAgPyBlbGVtZW50c1BlclBhZ2UgOiBlbGVtZW50c1BlclBhZ2UgKiAyLFxuICAgIH07XG4gIH1cblxuICAvKipcbiAgICogR2V0cyB0aGUgbmV4dCBlbGVtZW50IG9mIHRoZSBwcm92aWRlZCBhcnJheSBzdGFydGluZyBmcm9tIGBpbmRleGAgYnkgYWRkaW5nIGBkZWx0YWAuXG4gICAqIElmIHRoZSBmb3VuZCBlbGVtZW50IGlzIGRpc2FibGVkLCBpdCBjb250aW51ZXMgYWRkaW5nIGBkZWx0YWAgdW50aWwgaXQgZmluZHMgYW4gZW5hYmxlZCBvbmUgaW4gdGhlIGFycmF5IGJvdW5kcy5cbiAgICovXG4gIHByaXZhdGUgX2ZpbmROZXh0KGRheXM6IEhUTUxCdXR0b25FbGVtZW50W10sIGluZGV4OiBudW1iZXIsIGRlbHRhOiBudW1iZXIpOiBIVE1MQnV0dG9uRWxlbWVudCB7XG4gICAgbGV0IG5leHRJbmRleCA9IGluZGV4ICsgZGVsdGE7XG4gICAgd2hpbGUgKG5leHRJbmRleCA8IGRheXMubGVuZ3RoICYmIGRheXNbbmV4dEluZGV4XT8uZGlzYWJsZWQpIHtcbiAgICAgIG5leHRJbmRleCArPSBkZWx0YTtcbiAgICB9XG4gICAgcmV0dXJuIGRheXNbbmV4dEluZGV4XSA/PyBkYXlzW2luZGV4XTtcbiAgfVxuXG4gIC8qKiBGaW5kIHRoZSBmaXJzdCBlbmFibGVkIGVsZW1lbnQgaW4gdGhlIHByb3ZpZGVkIGFycmF5LiAqL1xuICBwcml2YXRlIF9maW5kRmlyc3QoZGF5czogSFRNTEJ1dHRvbkVsZW1lbnRbXSwgZmlyc3RPZkN1cnJlbnRNb250aDogbnVtYmVyKTogSFRNTEJ1dHRvbkVsZW1lbnQge1xuICAgIHJldHVybiAhZGF5c1tmaXJzdE9mQ3VycmVudE1vbnRoXS5kaXNhYmxlZFxuICAgICAgPyBkYXlzW2ZpcnN0T2ZDdXJyZW50TW9udGhdXG4gICAgICA6IHRoaXMuX2ZpbmROZXh0KGRheXMsIGZpcnN0T2ZDdXJyZW50TW9udGgsIDEpO1xuICB9XG5cbiAgLyoqIEZpbmQgdGhlIGxhc3QgZW5hYmxlZCBlbGVtZW50IGluIHRoZSBwcm92aWRlZCBhcnJheS4gKi9cbiAgcHJpdmF0ZSBfZmluZExhc3QoZGF5czogSFRNTEJ1dHRvbkVsZW1lbnRbXSwgbGFzdE9mQ3VycmVudE1vbnRoOiBudW1iZXIpOiBIVE1MQnV0dG9uRWxlbWVudCB7XG4gICAgcmV0dXJuICFkYXlzW2xhc3RPZkN1cnJlbnRNb250aF0uZGlzYWJsZWRcbiAgICAgID8gZGF5c1tsYXN0T2ZDdXJyZW50TW9udGhdXG4gICAgICA6IHRoaXMuX2ZpbmROZXh0KGRheXMsIGxhc3RPZkN1cnJlbnRNb250aCwgLTEpO1xuICB9XG5cbiAgLyoqIEZpbmQgdGhlIGZpcnN0IGVuYWJsZWQgZWxlbWVudCBpbiB0aGUgc2FtZSBjb2x1bW4gb2YgdGhlIHByb3ZpZGVkIGFycmF5LiAqL1xuICBwcml2YXRlIF9maW5kRmlyc3RPbkNvbHVtbihcbiAgICBkYXlzOiBIVE1MQnV0dG9uRWxlbWVudFtdLFxuICAgIGluZGV4OiBudW1iZXIsXG4gICAgb2Zmc2V0OiBudW1iZXIsXG4gICAgdmVydGljYWxPZmZzZXQ6IG51bWJlcixcbiAgKTogSFRNTEJ1dHRvbkVsZW1lbnQge1xuICAgIGNvbnN0IG5leHRJbmRleCA9IChpbmRleCAlIHZlcnRpY2FsT2Zmc2V0KSArIG9mZnNldDtcbiAgICByZXR1cm4gIWRheXNbbmV4dEluZGV4XS5kaXNhYmxlZFxuICAgICAgPyBkYXlzW25leHRJbmRleF1cbiAgICAgIDogdGhpcy5fZmluZE5leHQoZGF5cywgbmV4dEluZGV4LCB2ZXJ0aWNhbE9mZnNldCk7XG4gIH1cblxuICAvKiogRmluZCB0aGUgbGFzdCBlbmFibGVkIGVsZW1lbnQgaW4gdGhlIHNhbWUgY29sdW1uIG9mIHRoZSBwcm92aWRlZCBhcnJheS4gKi9cbiAgcHJpdmF0ZSBfZmluZExhc3RPbkNvbHVtbihcbiAgICBkYXlzOiBIVE1MQnV0dG9uRWxlbWVudFtdLFxuICAgIGluZGV4OiBudW1iZXIsXG4gICAgb2Zmc2V0OiBudW1iZXIsXG4gICAgdmVydGljYWxPZmZzZXQ6IG51bWJlcixcbiAgKTogSFRNTEJ1dHRvbkVsZW1lbnQge1xuICAgIGNvbnN0IG5leHRJbmRleCA9IGluZGV4ICsgTWF0aC50cnVuYygob2Zmc2V0IC0gaW5kZXggLSAxKSAvIHZlcnRpY2FsT2Zmc2V0KSAqIHZlcnRpY2FsT2Zmc2V0O1xuICAgIHJldHVybiAhZGF5c1tuZXh0SW5kZXhdLmRpc2FibGVkXG4gICAgICA/IGRheXNbbmV4dEluZGV4XVxuICAgICAgOiB0aGlzLl9maW5kTmV4dChkYXlzLCBuZXh0SW5kZXgsIC12ZXJ0aWNhbE9mZnNldCk7XG4gIH1cblxuICBwcml2YXRlIF9yZXNldENhbGVuZGFyVmlld0FuZEVtaXRNb250aENoYW5nZShpbml0VHJhbnNpdGlvbiA9IGZhbHNlKTogdm9pZCB7XG4gICAgdGhpcy5fcmVzZXRDYWxlbmRhclZpZXcoaW5pdFRyYW5zaXRpb24pO1xuICAgIHRoaXMuX2VtaXRNb250aENoYW5nZSgpO1xuICB9XG5cbiAgcHJpdmF0ZSBfcmVzZXRDYWxlbmRhclZpZXcoaW5pdFRyYW5zaXRpb24gPSBmYWxzZSk6IHZvaWQge1xuICAgIGlmICh0aGlzLl9jb250YWluaW5nRm9jdXMpIHtcbiAgICAgIHRoaXMuX3Jlc2V0Rm9jdXMgPSB0cnVlO1xuICAgIH1cbiAgICB0aGlzLl9hY3RpdmVEYXRlID1cbiAgICAgICh0aGlzLm11bHRpcGxlID8gKHRoaXMuX3NlbGVjdGVkIGFzIFRbXSkuYXQoLTEpIDogKHRoaXMuX3NlbGVjdGVkIGFzIFQpKSA/P1xuICAgICAgdGhpcy5fZGF0ZUFkYXB0ZXIudG9kYXkoKTtcbiAgICB0aGlzLl9zZXRDaG9zZW5ZZWFyKCk7XG4gICAgdGhpcy5fY2hvc2VuTW9udGggPSB1bmRlZmluZWQ7XG4gICAgdGhpcy5faW5pdCgpO1xuICAgIHRoaXMuX25leHRDYWxlbmRhclZpZXcgPSB0aGlzLl9jYWxlbmRhclZpZXcgPSB0aGlzLnZpZXc7XG5cbiAgICBpZiAoaW5pdFRyYW5zaXRpb24pIHtcbiAgICAgIHRoaXMuX3N0YXJ0VGFibGVUcmFuc2l0aW9uKCk7XG4gICAgfVxuICB9XG5cbiAgLyoqIFJlbmRlciB0aGUgdmlldyBmb3IgdGhlIGRheSBzZWxlY3Rpb24uICovXG4gIHByaXZhdGUgX3JlbmRlckRheVZpZXcoKTogVGVtcGxhdGVSZXN1bHQge1xuICAgIGNvbnN0IG5leHRNb250aEFjdGl2ZURhdGUgPSB0aGlzLl93aWRlXG4gICAgICA/IHRoaXMuX2RhdGVBZGFwdGVyLmFkZENhbGVuZGFyTW9udGhzKHRoaXMuX2FjdGl2ZURhdGUsIDEpXG4gICAgICA6IHVuZGVmaW5lZDtcbiAgICByZXR1cm4gaHRtbGBcbiAgICAgIDxkaXYgY2xhc3M9XCJzYmItY2FsZW5kYXJfX2NvbnRyb2xzXCI+XG4gICAgICAgICR7dGhpcy5fZ2V0QXJyb3coXG4gICAgICAgICAgJ2xlZnQnLFxuICAgICAgICAgICgpID0+IHRoaXMuX2dvVG9EaWZmZXJlbnRNb250aCgtMSksXG4gICAgICAgICAgaTE4blByZXZpb3VzTW9udGhbdGhpcy5fbGFuZ3VhZ2UuY3VycmVudF0sXG4gICAgICAgICAgdGhpcy5fcHJldmlvdXNNb250aERpc2FibGVkKCksXG4gICAgICAgICl9XG4gICAgICAgIDxkaXYgY2xhc3M9XCJzYmItY2FsZW5kYXJfX2NvbnRyb2xzLW1vbnRoXCI+XG4gICAgICAgICAgJHt0aGlzLl9jcmVhdGVMYWJlbEZvckRheVZpZXcodGhpcy5fYWN0aXZlRGF0ZSl9XG4gICAgICAgICAgJHt0aGlzLl93aWRlID8gdGhpcy5fY3JlYXRlTGFiZWxGb3JEYXlWaWV3KG5leHRNb250aEFjdGl2ZURhdGUhKSA6IG5vdGhpbmd9XG4gICAgICAgICAgPHNiYi1zY3JlZW4tcmVhZGVyLW9ubHkgcm9sZT1cInN0YXR1c1wiPlxuICAgICAgICAgICAgJHt0aGlzLl9jcmVhdGVBcmlhTGFiZWxGb3JEYXlWaWV3KHRoaXMuX2FjdGl2ZURhdGUsIG5leHRNb250aEFjdGl2ZURhdGUhKX1cbiAgICAgICAgICA8L3NiYi1zY3JlZW4tcmVhZGVyLW9ubHk+XG4gICAgICAgIDwvZGl2PlxuICAgICAgICAke3RoaXMuX2dldEFycm93KFxuICAgICAgICAgICdyaWdodCcsXG4gICAgICAgICAgKCkgPT4gdGhpcy5fZ29Ub0RpZmZlcmVudE1vbnRoKDEpLFxuICAgICAgICAgIGkxOG5OZXh0TW9udGhbdGhpcy5fbGFuZ3VhZ2UuY3VycmVudF0sXG4gICAgICAgICAgdGhpcy5fbmV4dE1vbnRoRGlzYWJsZWQoKSxcbiAgICAgICAgKX1cbiAgICAgIDwvZGl2PlxuICAgICAgPGRpdiBjbGFzcz1cInNiYi1jYWxlbmRhcl9fdGFibGUtb3ZlcmZsb3ctYnJlYWtcIj5cbiAgICAgICAgPGRpdiBjbGFzcz1cInNiYi1jYWxlbmRhcl9fdGFibGUtY29udGFpbmVyIHNiYi1jYWxlbmRhcl9fdGFibGUtZGF5LXZpZXdcIj5cbiAgICAgICAgICAke3RoaXMub3JpZW50YXRpb24gPT09ICdob3Jpem9udGFsJ1xuICAgICAgICAgICAgPyBodG1sYFxuICAgICAgICAgICAgICAgICR7dGhpcy5fY3JlYXRlRGF5VGFibGUodGhpcy5fd2Vla3MsIHRoaXMuX3dlZWtOdW1iZXJzKX1cbiAgICAgICAgICAgICAgICAke3RoaXMuX3dpZGVcbiAgICAgICAgICAgICAgICAgID8gdGhpcy5fY3JlYXRlRGF5VGFibGUodGhpcy5fbmV4dE1vbnRoV2Vla3MsIHRoaXMuX25leHRNb250aFdlZWtOdW1iZXJzLCB0cnVlKVxuICAgICAgICAgICAgICAgICAgOiBub3RoaW5nfVxuICAgICAgICAgICAgICBgXG4gICAgICAgICAgICA6IGh0bWxgXG4gICAgICAgICAgICAgICAgJHt0aGlzLl9jcmVhdGVEYXlUYWJsZVZlcnRpY2FsKHRoaXMuX3dlZWtzLCB0aGlzLl93ZWVrTnVtYmVycyl9XG4gICAgICAgICAgICAgICAgJHt0aGlzLl93aWRlXG4gICAgICAgICAgICAgICAgICA/IHRoaXMuX2NyZWF0ZURheVRhYmxlVmVydGljYWwoXG4gICAgICAgICAgICAgICAgICAgICAgdGhpcy5fbmV4dE1vbnRoV2Vla3MsXG4gICAgICAgICAgICAgICAgICAgICAgdGhpcy5fbmV4dE1vbnRoV2Vla051bWJlcnMsXG4gICAgICAgICAgICAgICAgICAgICAgbmV4dE1vbnRoQWN0aXZlRGF0ZSxcbiAgICAgICAgICAgICAgICAgICAgKVxuICAgICAgICAgICAgICAgICAgOiBub3RoaW5nfVxuICAgICAgICAgICAgICBgfVxuICAgICAgICA8L2Rpdj5cbiAgICAgIDwvZGl2PlxuICAgIGA7XG4gIH1cblxuICAvKiogQ3JlYXRlcyB0aGUgbGFiZWwgd2l0aCB0aGUgbW9udGggZm9yIHRoZSBkYWlseSB2aWV3LiAqL1xuICBwcml2YXRlIF9jcmVhdGVMYWJlbEZvckRheVZpZXcoZDogVCk6IFRlbXBsYXRlUmVzdWx0IHtcbiAgICBjb25zdCBtb250aExhYmVsID0gYCR7XG4gICAgICB0aGlzLl9tb250aE5hbWVzW3RoaXMuX2RhdGVBZGFwdGVyLmdldE1vbnRoKGQpIC0gMV1cbiAgICB9ICR7dGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0WWVhcihkKX1gO1xuICAgIHJldHVybiBodG1sYFxuICAgICAgPGJ1dHRvblxuICAgICAgICB0eXBlPVwiYnV0dG9uXCJcbiAgICAgICAgY2xhc3M9XCJzYmItY2FsZW5kYXJfX2RhdGUtc2VsZWN0aW9uIHNiYi1jYWxlbmRhcl9fY29udHJvbHMtY2hhbmdlLWRhdGVcIlxuICAgICAgICBhcmlhLWxhYmVsPVwiJHtpMThuWWVhck1vbnRoU2VsZWN0aW9uW3RoaXMuX2xhbmd1YWdlLmN1cnJlbnRdfSAke21vbnRoTGFiZWx9XCJcbiAgICAgICAgQGNsaWNrPSR7KCkgPT4ge1xuICAgICAgICAgIHRoaXMuX3Jlc2V0Rm9jdXMgPSB0cnVlO1xuICAgICAgICAgIHRoaXMuX25leHRDYWxlbmRhclZpZXcgPSAneWVhcic7XG4gICAgICAgICAgdGhpcy5fc3RhcnRUYWJsZVRyYW5zaXRpb24oKTtcbiAgICAgICAgfX1cbiAgICAgID5cbiAgICAgICAgJHttb250aExhYmVsfVxuICAgICAgICA8c2JiLWljb24gbmFtZT1cImNoZXZyb24tc21hbGwtZG93bi1zbWFsbFwiPjwvc2JiLWljb24+XG4gICAgICA8L2J1dHRvbj5cbiAgICBgO1xuICB9XG5cbiAgLyoqIENyZWF0ZXMgdGhlIGFyaWEtbGFiZWwgZm9yIHRoZSBkYWlseSB2aWV3LiAqL1xuICBwcml2YXRlIF9jcmVhdGVBcmlhTGFiZWxGb3JEYXlWaWV3KC4uLmRhdGVzOiBUW10pOiBzdHJpbmcge1xuICAgIGxldCBtb250aExhYmVsID0gJyc7XG4gICAgZm9yIChjb25zdCBkIG9mIGRhdGVzKSB7XG4gICAgICBpZiAoZCkge1xuICAgICAgICBtb250aExhYmVsICs9IGAke1xuICAgICAgICAgIHRoaXMuX21vbnRoTmFtZXNbdGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0TW9udGgoZCkgLSAxXVxuICAgICAgICB9ICR7dGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0WWVhcihkKX0gYDtcbiAgICAgIH1cbiAgICB9XG4gICAgcmV0dXJuIG1vbnRoTGFiZWw7XG4gIH1cblxuICAvKiogQ3JlYXRlcyB0aGUgY2FsZW5kYXIgdGFibGUgZm9yIHRoZSBkYWlseSB2aWV3LiAqL1xuICBwcml2YXRlIF9jcmVhdGVEYXlUYWJsZShcbiAgICB3ZWVrczogRGF5PFQ+W11bXSxcbiAgICB3ZWVrTnVtYmVyczogbnVtYmVyW10sXG4gICAgaXNXaWRlTmV4dE1vbnRoOiBib29sZWFuID0gZmFsc2UsXG4gICk6IFRlbXBsYXRlUmVzdWx0IHtcbiAgICBjb25zdCB3ZWVrc0ZvclNlbGVjdE11bHRpcGxlV2Vla051bWJlcnM6IERheTxUPltdID0gKFxuICAgICAgdGhpcy5fd2lkZVxuICAgICAgICA/IFsuLi50aGlzLl93ZWVrcywgLi4udGhpcy5fbmV4dE1vbnRoV2Vla3NdXG4gICAgICAgIDogaXNXaWRlTmV4dE1vbnRoXG4gICAgICAgICAgPyB0aGlzLl9uZXh0TW9udGhXZWVrc1xuICAgICAgICAgIDogdGhpcy5fd2Vla3NcbiAgICApLmZsYXQoKTtcbiAgICBjb25zdCB3ZWVrc0ZvclNlbGVjdE11bHRpcGxlV2Vla0RheXM6IERheTxUPltdID0gKFxuICAgICAgaXNXaWRlTmV4dE1vbnRoID8gdGhpcy5fbmV4dE1vbnRoV2Vla3MgOiB0aGlzLl93ZWVrc1xuICAgICkuZmxhdCgpO1xuICAgIHJldHVybiBodG1sYFxuICAgICAgPHRhYmxlXG4gICAgICAgIGNsYXNzPVwic2JiLWNhbGVuZGFyX190YWJsZVwiXG4gICAgICAgIEBmb2N1c291dD0keyhldmVudDogRm9jdXNFdmVudCkgPT5cbiAgICAgICAgICB0aGlzLl9oYW5kbGVUYWJsZUJsdXIoZXZlbnQucmVsYXRlZFRhcmdldCBhcyBIVE1MRWxlbWVudCl9XG4gICAgICAgIEBhbmltYXRpb25lbmQ9JHsoZTogQW5pbWF0aW9uRXZlbnQpID0+IHRoaXMuX3RhYmxlQW5pbWF0aW9uRW5kKGUpfVxuICAgICAgPlxuICAgICAgICA8dGhlYWQgY2xhc3M9XCJzYmItY2FsZW5kYXJfX3RhYmxlLWhlYWRlclwiPlxuICAgICAgICAgIDx0cj5cbiAgICAgICAgICAgICR7dGhpcy53ZWVrTnVtYmVycyA/IGh0bWxgPHRoIGNsYXNzPVwic2JiLWNhbGVuZGFyX190YWJsZS1oZWFkZXItY2VsbFwiPjwvdGg+YCA6IG5vdGhpbmd9XG4gICAgICAgICAgICAke3RoaXMuX3dlZWtkYXlzLm1hcChcbiAgICAgICAgICAgICAgKGRheTogV2Vla2RheSwgaW5kZXg6IG51bWJlcikgPT4gaHRtbGBcbiAgICAgICAgICAgICAgICA8dGggY2xhc3M9XCJzYmItY2FsZW5kYXJfX3RhYmxlLWhlYWRlci1jZWxsXCI+XG4gICAgICAgICAgICAgICAgICAke3RoaXMubXVsdGlwbGVcbiAgICAgICAgICAgICAgICAgICAgPyBodG1sYFxuICAgICAgICAgICAgICAgICAgICAgICAgPGJ1dHRvblxuICAgICAgICAgICAgICAgICAgICAgICAgICBjbGFzcz1cInNiYi1jYWxlbmRhcl9faGVhZGVyLWNlbGwgc2JiLWNhbGVuZGFyX193ZWVrZGF5XCJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgYXJpYS1sYWJlbD0ke2RheS5sb25nfVxuICAgICAgICAgICAgICAgICAgICAgICAgICBAY2xpY2s9JHsoKSA9PiB7XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgLy8gTk9URTogU3VuZGF5cyBoYXZlIGluZGV4IDcsIHdoaWxlIHRoZWlyIHdlZWtEYXlWYWx1ZSBpcyAwXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgY29uc3QgZGF5czogRGF5PFQ+W10gPSB3ZWVrc0ZvclNlbGVjdE11bHRpcGxlV2Vla0RheXMuZmlsdGVyKFxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgKGRheTogRGF5PFQ+KSA9PiBkYXkud2Vla0RheVZhbHVlID09PSAoaW5kZXggKyAxKSAlIDcsXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgKSE7XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgdGhpcy5fc2VsZWN0TXVsdGlwbGVEYXRlcyhkYXlzKTtcbiAgICAgICAgICAgICAgICAgICAgICAgICAgfX1cbiAgICAgICAgICAgICAgICAgICAgICAgID5cbiAgICAgICAgICAgICAgICAgICAgICAgICAgJHtkYXkubmFycm93fVxuICAgICAgICAgICAgICAgICAgICAgICAgPC9idXR0b24+XG4gICAgICAgICAgICAgICAgICAgICAgYFxuICAgICAgICAgICAgICAgICAgICA6IGh0bWxgXG4gICAgICAgICAgICAgICAgICAgICAgICA8c2JiLXNjcmVlbi1yZWFkZXItb25seT4ke2RheS5sb25nfTwvc2JiLXNjcmVlbi1yZWFkZXItb25seT5cbiAgICAgICAgICAgICAgICAgICAgICAgIDxzcGFuIGFyaWEtaGlkZGVuPVwidHJ1ZVwiPiR7ZGF5Lm5hcnJvd308L3NwYW4+XG4gICAgICAgICAgICAgICAgICAgICAgYH1cbiAgICAgICAgICAgICAgICA8L3RoPlxuICAgICAgICAgICAgICBgLFxuICAgICAgICAgICAgKX1cbiAgICAgICAgICA8L3RyPlxuICAgICAgICA8L3RoZWFkPlxuICAgICAgICA8dGJvZHkgY2xhc3M9XCJzYmItY2FsZW5kYXJfX3RhYmxlLWJvZHlcIj5cbiAgICAgICAgICAke3dlZWtzLm1hcCgod2VlazogRGF5PFQ+W10sIHJvd0luZGV4OiBudW1iZXIpID0+IHtcbiAgICAgICAgICAgIGNvbnN0IGZpcnN0Um93T2Zmc2V0OiBudW1iZXIgPSBEQVlTX1BFUl9ST1cgLSB3ZWVrLmxlbmd0aDtcbiAgICAgICAgICAgIGlmIChyb3dJbmRleCA9PT0gMCAmJiBmaXJzdFJvd09mZnNldCkge1xuICAgICAgICAgICAgICByZXR1cm4gaHRtbGBcbiAgICAgICAgICAgICAgICA8dHI+XG4gICAgICAgICAgICAgICAgICAke3RoaXMud2Vla051bWJlcnNcbiAgICAgICAgICAgICAgICAgICAgPyBodG1sYFxuICAgICAgICAgICAgICAgICAgICAgICAgPHRkIGNsYXNzPVwic2JiLWNhbGVuZGFyX190YWJsZS1oZWFkZXItY2VsbC12ZXJ0aWNhbFwiPlxuICAgICAgICAgICAgICAgICAgICAgICAgICAke3RoaXMubXVsdGlwbGVcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICA/IGh0bWxgXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIDxidXR0b25cbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBjbGFzcz1cInNiYi1jYWxlbmRhcl9faGVhZGVyLWNlbGwgc2JiLWNhbGVuZGFyX193ZWVrZGF5XCJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBhcmlhLWxhYmVsPSR7YCR7aTE4bkNhbGVuZGFyV2Vla051bWJlclt0aGlzLl9sYW5ndWFnZS5jdXJyZW50XX0gJHt3ZWVrTnVtYmVyc1swXX1gfVxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIEBjbGljaz0keygpID0+IHtcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNvbnN0IGRheXM6IERheTxUPltdID0gd2Vla3NGb3JTZWxlY3RNdWx0aXBsZVdlZWtOdW1iZXJzLmZpbHRlcihcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgKGRheTogRGF5PFQ+KSA9PiBkYXkud2Vla1ZhbHVlID09PSB3ZWVrTnVtYmVyc1swXSxcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICkhO1xuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgdGhpcy5fc2VsZWN0TXVsdGlwbGVEYXRlcyhkYXlzKTtcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB9fVxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICA+XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgJHt3ZWVrTnVtYmVyc1swXX1cbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPC9idXR0b24+XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICBgXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgOiBodG1sYFxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICA8c2JiLXNjcmVlbi1yZWFkZXItb25seVxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID4ke2Ake2kxOG5DYWxlbmRhcldlZWtOdW1iZXJbdGhpcy5fbGFuZ3VhZ2UuY3VycmVudF19ICR7d2Vla051bWJlcnNbMF19YH08L3NiYi1zY3JlZW4tcmVhZGVyLW9ubHlcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPlxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICA8c3BhbiBhcmlhLWhpZGRlbj1cInRydWVcIj4ke3dlZWtOdW1iZXJzWzBdfTwvc3Bhbj5cbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGB9XG4gICAgICAgICAgICAgICAgICAgICAgICA8L3RkPlxuICAgICAgICAgICAgICAgICAgICAgIGBcbiAgICAgICAgICAgICAgICAgICAgOiBub3RoaW5nfVxuICAgICAgICAgICAgICAgICAgJHtbLi4uQXJyYXkoZmlyc3RSb3dPZmZzZXQpLmtleXMoKV0ubWFwKFxuICAgICAgICAgICAgICAgICAgICAoKSA9PiBodG1sYDx0ZCBjbGFzcz1cInNiYi1jYWxlbmRhcl9fdGFibGUtZGF0YVwiPjwvdGQ+YCxcbiAgICAgICAgICAgICAgICAgICl9XG4gICAgICAgICAgICAgICAgICAke3RoaXMuX2NyZWF0ZURheUNlbGxzKHdlZWspfVxuICAgICAgICAgICAgICAgIDwvdHI+XG4gICAgICAgICAgICAgIGA7XG4gICAgICAgICAgICB9XG4gICAgICAgICAgICByZXR1cm4gaHRtbGBcbiAgICAgICAgICAgICAgPHRyPlxuICAgICAgICAgICAgICAgICR7dGhpcy53ZWVrTnVtYmVyc1xuICAgICAgICAgICAgICAgICAgPyBodG1sYFxuICAgICAgICAgICAgICAgICAgICAgIDx0ZCBjbGFzcz1cInNiYi1jYWxlbmRhcl9fdGFibGUtaGVhZGVyLWNlbGwtdmVydGljYWxcIj5cbiAgICAgICAgICAgICAgICAgICAgICAgICR7dGhpcy5tdWx0aXBsZVxuICAgICAgICAgICAgICAgICAgICAgICAgICA/IGh0bWxgXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICA8YnV0dG9uXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNsYXNzPVwic2JiLWNhbGVuZGFyX19oZWFkZXItY2VsbCBzYmItY2FsZW5kYXJfX3dlZWtkYXlcIlxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBhcmlhLWxhYmVsPSR7YCR7aTE4bkNhbGVuZGFyV2Vla051bWJlclt0aGlzLl9sYW5ndWFnZS5jdXJyZW50XX0gJHt3ZWVrTnVtYmVyc1tyb3dJbmRleF19YH1cbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgQGNsaWNrPSR7KCkgPT4ge1xuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNvbnN0IGRheXM6IERheTxUPltdID0gd2Vla3NGb3JTZWxlY3RNdWx0aXBsZVdlZWtOdW1iZXJzLmZpbHRlcihcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIChkYXk6IERheTxUPikgPT4gZGF5LndlZWtWYWx1ZSA9PT0gd2Vla051bWJlcnNbcm93SW5kZXhdLFxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICkhO1xuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHRoaXMuX3NlbGVjdE11bHRpcGxlRGF0ZXMoZGF5cyk7XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIH19XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICA+XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICR7d2Vla051bWJlcnNbcm93SW5kZXhdfVxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPC9idXR0b24+XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgYFxuICAgICAgICAgICAgICAgICAgICAgICAgICA6IGh0bWxgXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICA8c2JiLXNjcmVlbi1yZWFkZXItb25seVxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICA+JHtgJHtpMThuQ2FsZW5kYXJXZWVrTnVtYmVyW3RoaXMuX2xhbmd1YWdlLmN1cnJlbnRdfSAke3dlZWtOdW1iZXJzW3Jvd0luZGV4XX1gfTwvc2JiLXNjcmVlbi1yZWFkZXItb25seVxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPlxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPHNwYW4gYXJpYS1oaWRkZW49XCJ0cnVlXCI+JHt3ZWVrTnVtYmVyc1tyb3dJbmRleF19PC9zcGFuPlxuICAgICAgICAgICAgICAgICAgICAgICAgICAgIGB9XG4gICAgICAgICAgICAgICAgICAgICAgPC90ZD5cbiAgICAgICAgICAgICAgICAgICAgYFxuICAgICAgICAgICAgICAgICAgOiBub3RoaW5nfVxuICAgICAgICAgICAgICAgICR7dGhpcy5fY3JlYXRlRGF5Q2VsbHMod2Vlayl9XG4gICAgICAgICAgICAgIDwvdHI+XG4gICAgICAgICAgICBgO1xuICAgICAgICAgIH0pfVxuICAgICAgICA8L3Rib2R5PlxuICAgICAgPC90YWJsZT5cbiAgICBgO1xuICB9XG5cbiAgLyoqIENyZWF0ZXMgdGhlIHRhYmxlIGluIG9yaWVudGF0aW9uPSd2ZXJ0aWNhbCcuICovXG4gIHByaXZhdGUgX2NyZWF0ZURheVRhYmxlVmVydGljYWwoXG4gICAgd2Vla3M6IERheTxUPltdW10sXG4gICAgd2Vla051bWJlcnM6IG51bWJlcltdLFxuICAgIG5leHRNb250aEFjdGl2ZURhdGU/OiBULFxuICApOiBUZW1wbGF0ZVJlc3VsdCB7XG4gICAgY29uc3Qgd2Vla09mZnNldCA9IHRoaXMuX2RhdGVBZGFwdGVyLmdldEZpcnN0V2Vla09mZnNldChcbiAgICAgIG5leHRNb250aEFjdGl2ZURhdGUgPz8gdGhpcy5fYWN0aXZlRGF0ZSxcbiAgICApO1xuICAgIGNvbnN0IHdlZWtzRm9yU2VsZWN0TXVsdGlwbGVXZWVrTnVtYmVyczogRGF5PFQ+W10gPSAoXG4gICAgICB0aGlzLl93aWRlXG4gICAgICAgID8gWy4uLnRoaXMuX3dlZWtzLCAuLi50aGlzLl9uZXh0TW9udGhXZWVrc11cbiAgICAgICAgOiBuZXh0TW9udGhBY3RpdmVEYXRlXG4gICAgICAgICAgPyB0aGlzLl9uZXh0TW9udGhXZWVrc1xuICAgICAgICAgIDogdGhpcy5fd2Vla3NcbiAgICApLmZsYXQoKTtcbiAgICByZXR1cm4gaHRtbGBcbiAgICAgIDx0YWJsZVxuICAgICAgICBjbGFzcz1cInNiYi1jYWxlbmRhcl9fdGFibGVcIlxuICAgICAgICBAZm9jdXNvdXQ9JHsoZXZlbnQ6IEZvY3VzRXZlbnQpID0+XG4gICAgICAgICAgdGhpcy5faGFuZGxlVGFibGVCbHVyKGV2ZW50LnJlbGF0ZWRUYXJnZXQgYXMgSFRNTEVsZW1lbnQpfVxuICAgICAgICBAYW5pbWF0aW9uZW5kPSR7KGU6IEFuaW1hdGlvbkV2ZW50KSA9PiB0aGlzLl90YWJsZUFuaW1hdGlvbkVuZChlKX1cbiAgICAgID5cbiAgICAgICAgJHt0aGlzLndlZWtOdW1iZXJzXG4gICAgICAgICAgPyBodG1sYFxuICAgICAgICAgICAgICA8dGhlYWQgY2xhc3M9XCJzYmItY2FsZW5kYXJfX3RhYmxlLWhlYWRlclwiPlxuICAgICAgICAgICAgICAgIDx0cj5cbiAgICAgICAgICAgICAgICAgICR7bmV4dE1vbnRoQWN0aXZlRGF0ZVxuICAgICAgICAgICAgICAgICAgICA/IG5vdGhpbmdcbiAgICAgICAgICAgICAgICAgICAgOiBodG1sYDx0aCBjbGFzcz1cInNiYi1jYWxlbmRhcl9fdGFibGUtZGF0YVwiPjwvdGg+YH1cbiAgICAgICAgICAgICAgICAgICR7d2Vla051bWJlcnMubWFwKFxuICAgICAgICAgICAgICAgICAgICAod2Vla051bWJlcjogbnVtYmVyKSA9PiBodG1sYFxuICAgICAgICAgICAgICAgICAgICAgIDx0aCBjbGFzcz1cInNiYi1jYWxlbmRhcl9fdGFibGUtaGVhZGVyLWNlbGxcIj5cbiAgICAgICAgICAgICAgICAgICAgICAgICR7dGhpcy5tdWx0aXBsZVxuICAgICAgICAgICAgICAgICAgICAgICAgICA/IGh0bWxgXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICA8YnV0dG9uXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNsYXNzPVwic2JiLWNhbGVuZGFyX19oZWFkZXItY2VsbCBzYmItY2FsZW5kYXJfX3dlZWtkYXlcIlxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBhcmlhLWxhYmVsPSR7YCR7aTE4bkNhbGVuZGFyV2Vla051bWJlclt0aGlzLl9sYW5ndWFnZS5jdXJyZW50XX0gJHt3ZWVrTnVtYmVyfWB9XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIEBjbGljaz0keygpID0+IHtcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBjb25zdCBkYXlzOiBEYXk8VD5bXSA9IHdlZWtzRm9yU2VsZWN0TXVsdGlwbGVXZWVrTnVtYmVycy5maWx0ZXIoXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAoZGF5OiBEYXk8VD4pID0+IGRheS53ZWVrVmFsdWUgPT09IHdlZWtOdW1iZXIsXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgKSE7XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgdGhpcy5fc2VsZWN0TXVsdGlwbGVEYXRlcyhkYXlzKTtcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgfX1cbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID5cbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgJHt3ZWVrTnVtYmVyfVxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPC9idXR0b24+XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgYFxuICAgICAgICAgICAgICAgICAgICAgICAgICA6IGh0bWxgXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICA8c2JiLXNjcmVlbi1yZWFkZXItb25seVxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICA+JHtgJHtpMThuQ2FsZW5kYXJXZWVrTnVtYmVyW3RoaXMuX2xhbmd1YWdlLmN1cnJlbnRdfSAke3dlZWtOdW1iZXJ9YH08L3NiYi1zY3JlZW4tcmVhZGVyLW9ubHlcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID5cbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIDxzcGFuIGFyaWEtaGlkZGVuPVwidHJ1ZVwiPiR7d2Vla051bWJlcn08L3NwYW4+XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgYH1cbiAgICAgICAgICAgICAgICAgICAgICA8L3RoPlxuICAgICAgICAgICAgICAgICAgICBgLFxuICAgICAgICAgICAgICAgICAgKX1cbiAgICAgICAgICAgICAgICA8L3RyPlxuICAgICAgICAgICAgICA8L3RoZWFkPlxuICAgICAgICAgICAgYFxuICAgICAgICAgIDogbm90aGluZ31cbiAgICAgICAgPHRib2R5IGNsYXNzPVwic2JiLWNhbGVuZGFyX190YWJsZS1ib2R5XCI+XG4gICAgICAgICAgJHt3ZWVrcy5tYXAoKHdlZWs6IERheTxUPltdLCByb3dJbmRleDogbnVtYmVyKSA9PiB7XG4gICAgICAgICAgICBjb25zdCB3ZWVrZGF5ID0gdGhpcy5fd2Vla2RheXNbcm93SW5kZXhdO1xuICAgICAgICAgICAgY29uc3Qgc2VsZWN0YWJsZURheXMgPSB0aGlzLl93aWRlID8gWy4uLndlZWssIC4uLnRoaXMuX25leHRNb250aFdlZWtzW3Jvd0luZGV4XV0gOiB3ZWVrO1xuICAgICAgICAgICAgcmV0dXJuIGh0bWxgXG4gICAgICAgICAgICAgIDx0cj5cbiAgICAgICAgICAgICAgICAke25leHRNb250aEFjdGl2ZURhdGVcbiAgICAgICAgICAgICAgICAgID8gbm90aGluZ1xuICAgICAgICAgICAgICAgICAgOiBodG1sYFxuICAgICAgICAgICAgICAgICAgICAgIDx0ZCBjbGFzcz1cInNiYi1jYWxlbmRhcl9fdGFibGUtaGVhZGVyLWNlbGwtdmVydGljYWxcIj5cbiAgICAgICAgICAgICAgICAgICAgICAgICR7dGhpcy5tdWx0aXBsZVxuICAgICAgICAgICAgICAgICAgICAgICAgICA/IGh0bWxgXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICA8YnV0dG9uXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNsYXNzPVwic2JiLWNhbGVuZGFyX19oZWFkZXItY2VsbCBzYmItY2FsZW5kYXJfX3dlZWtkYXlcIlxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBhcmlhLWxhYmVsPSR7d2Vla2RheS5sb25nfVxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBAY2xpY2s9JHsoKSA9PiB0aGlzLl9zZWxlY3RNdWx0aXBsZURhdGVzKHNlbGVjdGFibGVEYXlzKX1cbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID5cbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgJHt3ZWVrZGF5Lm5hcnJvd31cbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIDwvYnV0dG9uPlxuICAgICAgICAgICAgICAgICAgICAgICAgICAgIGBcbiAgICAgICAgICAgICAgICAgICAgICAgICAgOiBodG1sYFxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPHNiYi1zY3JlZW4tcmVhZGVyLW9ubHk+JHt3ZWVrZGF5Lmxvbmd9PC9zYmItc2NyZWVuLXJlYWRlci1vbmx5PlxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPHNwYW4gYXJpYS1oaWRkZW49XCJ0cnVlXCI+JHt3ZWVrZGF5Lm5hcnJvd308L3NwYW4+XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgYH1cbiAgICAgICAgICAgICAgICAgICAgICA8L3RkPlxuICAgICAgICAgICAgICAgICAgICBgfVxuICAgICAgICAgICAgICAgICR7cm93SW5kZXggPCB3ZWVrT2Zmc2V0XG4gICAgICAgICAgICAgICAgICA/IGh0bWxgPHRkIGNsYXNzPVwic2JiLWNhbGVuZGFyX190YWJsZS1kYXRhXCI+PC90ZD5gXG4gICAgICAgICAgICAgICAgICA6IG5vdGhpbmd9XG4gICAgICAgICAgICAgICAgJHt0aGlzLl9jcmVhdGVEYXlDZWxscyh3ZWVrKX1cbiAgICAgICAgICAgICAgPC90cj5cbiAgICAgICAgICAgIGA7XG4gICAgICAgICAgfSl9XG4gICAgICAgIDwvdGJvZHk+XG4gICAgICA8L3RhYmxlPlxuICAgIGA7XG4gIH1cblxuICAvKiogQ3JlYXRlcyB0aGUgY2VsbHMgZm9yIHRoZSBkYWlseSB2aWV3LiAqL1xuICBwcml2YXRlIF9jcmVhdGVEYXlDZWxscyh3ZWVrOiBEYXk8VD5bXSk6IFRlbXBsYXRlUmVzdWx0W10ge1xuICAgIHJldHVybiB3ZWVrLm1hcCgoZGF5OiBEYXk8VD4pID0+IHtcbiAgICAgIHJldHVybiBodG1sYFxuICAgICAgICA8dGQgY2xhc3M9XCIgc2JiLWNhbGVuZGFyX190YWJsZS1kYXRhIHNiYi1jYWxlbmRhcl9fZGF5LWNlbGxcIj5cbiAgICAgICAgICA8c2xvdCBuYW1lPSR7ZGF5LnZhbHVlfT5cbiAgICAgICAgICAgIDxzYmItY2FsZW5kYXItZGF5XG4gICAgICAgICAgICAgIHNsb3Q9JHtkYXkudmFsdWV9XG4gICAgICAgICAgICAgIEBjbGljaz0keygpID0+IHRoaXMuX3NlbGVjdERhdGUoZGF5LmRhdGVWYWx1ZSl9XG4gICAgICAgICAgICAgIEBrZXlkb3duPSR7KGV2dDogS2V5Ym9hcmRFdmVudCkgPT4gdGhpcy5faGFuZGxlS2V5Ym9hcmRFdmVudChldnQsIGRheSl9XG4gICAgICAgICAgICA+PC9zYmItY2FsZW5kYXItZGF5PlxuICAgICAgICAgIDwvc2xvdD5cbiAgICAgICAgPC90ZD5cbiAgICAgIGA7XG4gICAgfSk7XG4gIH1cblxuICAvKiogUmVuZGVyIHRoZSB2aWV3IGZvciB0aGUgbW9udGggc2VsZWN0aW9uLiAqL1xuICBwcml2YXRlIF9yZW5kZXJNb250aFZpZXcoKTogVGVtcGxhdGVSZXN1bHQge1xuICAgIHJldHVybiBodG1sYFxuICAgICAgPGRpdiBjbGFzcz1cInNiYi1jYWxlbmRhcl9fY29udHJvbHNcIj5cbiAgICAgICAgJHt0aGlzLl9nZXRBcnJvdyhcbiAgICAgICAgICAnbGVmdCcsXG4gICAgICAgICAgKCkgPT4gdGhpcy5fZ29Ub0RpZmZlcmVudFllYXIoLTEpLFxuICAgICAgICAgIGkxOG5QcmV2aW91c1llYXJbdGhpcy5fbGFuZ3VhZ2UuY3VycmVudF0sXG4gICAgICAgICAgdGhpcy5fcHJldmlvdXNZZWFyRGlzYWJsZWQoKSxcbiAgICAgICAgKX1cbiAgICAgICAgPGRpdiBjbGFzcz1cInNiYi1jYWxlbmRhcl9fY29udHJvbHMtbW9udGhcIj4ke3RoaXMuX2NyZWF0ZUxhYmVsRm9yTW9udGhWaWV3KCl9PC9kaXY+XG4gICAgICAgICR7dGhpcy5fZ2V0QXJyb3coXG4gICAgICAgICAgJ3JpZ2h0JyxcbiAgICAgICAgICAoKSA9PiB0aGlzLl9nb1RvRGlmZmVyZW50WWVhcigxKSxcbiAgICAgICAgICBpMThuTmV4dFllYXJbdGhpcy5fbGFuZ3VhZ2UuY3VycmVudF0sXG4gICAgICAgICAgdGhpcy5fbmV4dFllYXJEaXNhYmxlZCgpLFxuICAgICAgICApfVxuICAgICAgPC9kaXY+XG4gICAgICA8ZGl2IGNsYXNzPVwic2JiLWNhbGVuZGFyX190YWJsZS1vdmVyZmxvdy1icmVha1wiPlxuICAgICAgICA8ZGl2IGNsYXNzPVwic2JiLWNhbGVuZGFyX190YWJsZS1jb250YWluZXIgc2JiLWNhbGVuZGFyX190YWJsZS1tb250aC12aWV3XCI+XG4gICAgICAgICAgJHt0aGlzLl9jcmVhdGVNb250aFRhYmxlKHRoaXMuX21vbnRocywgdGhpcy5fY2hvc2VuWWVhciEpfVxuICAgICAgICAgICR7dGhpcy5fd2lkZSA/IHRoaXMuX2NyZWF0ZU1vbnRoVGFibGUodGhpcy5fbW9udGhzLCB0aGlzLl9jaG9zZW5ZZWFyISArIDEpIDogbm90aGluZ31cbiAgICAgICAgPC9kaXY+XG4gICAgICA8L2Rpdj5cbiAgICBgO1xuICB9XG5cbiAgLyoqIENyZWF0ZXMgdGhlIGxhYmVsIHdpdGggdGhlIHllYXIgZm9yIHRoZSBtb250aGx5IHZpZXcuICovXG4gIHByaXZhdGUgX2NyZWF0ZUxhYmVsRm9yTW9udGhWaWV3KCk6IFRlbXBsYXRlUmVzdWx0IHtcbiAgICByZXR1cm4gaHRtbGAgPGJ1dHRvblxuICAgICAgICB0eXBlPVwiYnV0dG9uXCJcbiAgICAgICAgaWQ9XCJzYmItY2FsZW5kYXJfX21vbnRoLXNlbGVjdGlvblwiXG4gICAgICAgIGNsYXNzPVwic2JiLWNhbGVuZGFyX19jb250cm9scy1jaGFuZ2UtZGF0ZVwiXG4gICAgICAgIGFyaWEtbGFiZWw9JHtgJHtpMThuQ2FsZW5kYXJEYXRlU2VsZWN0aW9uW3RoaXMuX2xhbmd1YWdlLmN1cnJlbnRdfSAke3RoaXMuX2Nob3NlblllYXJ9YH1cbiAgICAgICAgQGNsaWNrPSR7KCkgPT4gdGhpcy5fcmVzZXRDYWxlbmRhclZpZXdBbmRFbWl0TW9udGhDaGFuZ2UodHJ1ZSl9XG4gICAgICA+XG4gICAgICAgICR7dGhpcy5fY2hvc2VuWWVhcn0gJHt0aGlzLl93aWRlID8gYCAtICR7dGhpcy5fY2hvc2VuWWVhciEgKyAxfWAgOiBub3RoaW5nfVxuICAgICAgICA8c2JiLWljb24gbmFtZT1cImNoZXZyb24tc21hbGwtdXAtc21hbGxcIj48L3NiYi1pY29uPlxuICAgICAgPC9idXR0b24+XG4gICAgICA8c2JiLXNjcmVlbi1yZWFkZXItb25seSByb2xlPVwic3RhdHVzXCI+ICR7dGhpcy5fY2hvc2VuWWVhcn0gPC9zYmItc2NyZWVuLXJlYWRlci1vbmx5PmA7XG4gIH1cblxuICAvKiogQ3JlYXRlcyB0aGUgdGFibGUgZm9yIHRoZSBtb250aCBzZWxlY3Rpb24gdmlldy4gKi9cbiAgcHJpdmF0ZSBfY3JlYXRlTW9udGhUYWJsZShtb250aHM6IE1vbnRoW11bXSwgeWVhcjogbnVtYmVyKTogVGVtcGxhdGVSZXN1bHQge1xuICAgIHJldHVybiBodG1sYFxuICAgICAgPHRhYmxlXG4gICAgICAgIGNsYXNzPVwic2JiLWNhbGVuZGFyX190YWJsZVwiXG4gICAgICAgIEBhbmltYXRpb25lbmQ9JHsoZTogQW5pbWF0aW9uRXZlbnQpID0+IHRoaXMuX3RhYmxlQW5pbWF0aW9uRW5kKGUpfVxuICAgICAgPlxuICAgICAgICAke3RoaXMuX3dpZGVcbiAgICAgICAgICA/IGh0bWxgPHRoZWFkIGNsYXNzPVwic2JiLWNhbGVuZGFyX190YWJsZS1oZWFkZXJcIiBhcmlhLWhpZGRlbj1cInRydWVcIj5cbiAgICAgICAgICAgICAgPHRyPlxuICAgICAgICAgICAgICAgIDx0aCBjbGFzcz1cInNiYi1jYWxlbmRhcl9fdGFibGUtaGVhZGVyLWNlbGxcIiBjb2xzcGFuPSR7TU9OVEhTX1BFUl9ST1d9PiR7eWVhcn08L3RoPlxuICAgICAgICAgICAgICA8L3RyPlxuICAgICAgICAgICAgPC90aGVhZD5gXG4gICAgICAgICAgOiBub3RoaW5nfVxuICAgICAgICA8dGJvZHkgY2xhc3M9XCJzYmItY2FsZW5kYXJfX3RhYmxlLWJvZHlcIj5cbiAgICAgICAgICAke21vbnRocy5tYXAoXG4gICAgICAgICAgICAocm93OiBNb250aFtdKSA9PiBodG1sYFxuICAgICAgICAgICAgICA8dHI+XG4gICAgICAgICAgICAgICAgJHtyb3cubWFwKChtb250aDogTW9udGgpID0+IHtcbiAgICAgICAgICAgICAgICAgIGxldCBzZWxlY3RlZDogYm9vbGVhbjtcbiAgICAgICAgICAgICAgICAgIGlmICh0aGlzLm11bHRpcGxlKSB7XG4gICAgICAgICAgICAgICAgICAgIHNlbGVjdGVkID1cbiAgICAgICAgICAgICAgICAgICAgICAodGhpcy5fc2VsZWN0ZWQgYXMgVFtdKS5maW5kKFxuICAgICAgICAgICAgICAgICAgICAgICAgKGRhdGU6IFQpID0+XG4gICAgICAgICAgICAgICAgICAgICAgICAgIHllYXIgPT09IHRoaXMuX2RhdGVBZGFwdGVyLmdldFllYXIoZGF0ZSkgJiZcbiAgICAgICAgICAgICAgICAgICAgICAgICAgbW9udGgubW9udGhWYWx1ZSA9PT0gdGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0TW9udGgoZGF0ZSksXG4gICAgICAgICAgICAgICAgICAgICAgKSAhPT0gdW5kZWZpbmVkO1xuICAgICAgICAgICAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICAgICAgICAgICAgY29uc3Qgc2VsZWN0ZWRNb250aCA9IHRoaXMuX3NlbGVjdGVkXG4gICAgICAgICAgICAgICAgICAgICAgPyB0aGlzLl9kYXRlQWRhcHRlci5nZXRNb250aCh0aGlzLl9zZWxlY3RlZCBhcyBUKVxuICAgICAgICAgICAgICAgICAgICAgIDogdW5kZWZpbmVkO1xuICAgICAgICAgICAgICAgICAgICBjb25zdCBzZWxlY3RlZFllYXIgPSB0aGlzLl9zZWxlY3RlZFxuICAgICAgICAgICAgICAgICAgICAgID8gdGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0WWVhcih0aGlzLl9zZWxlY3RlZCBhcyBUKVxuICAgICAgICAgICAgICAgICAgICAgIDogdW5kZWZpbmVkO1xuICAgICAgICAgICAgICAgICAgICBzZWxlY3RlZCA9XG4gICAgICAgICAgICAgICAgICAgICAgISF0aGlzLl9zZWxlY3RlZCAmJlxuICAgICAgICAgICAgICAgICAgICAgIHllYXIgPT09IHNlbGVjdGVkWWVhciAmJlxuICAgICAgICAgICAgICAgICAgICAgIG1vbnRoLm1vbnRoVmFsdWUgPT09IHNlbGVjdGVkTW9udGg7XG4gICAgICAgICAgICAgICAgICB9XG4gICAgICAgICAgICAgICAgICBjb25zdCBpc091dE9mUmFuZ2UgPSAhdGhpcy5faXNNb250aEluUmFuZ2UobW9udGgubW9udGhWYWx1ZSwgeWVhcik7XG4gICAgICAgICAgICAgICAgICBjb25zdCBpc0ZpbHRlcmVkT3V0ID0gIXRoaXMuX2lzTW9udGhGaWx0ZXJlZE91dChtb250aC5tb250aFZhbHVlLCB5ZWFyKTtcbiAgICAgICAgICAgICAgICAgIGNvbnN0IGlzQ3VycmVudE1vbnRoID1cbiAgICAgICAgICAgICAgICAgICAgeWVhciA9PT0gdGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0WWVhcih0aGlzLl9kYXRlQWRhcHRlci50b2RheSgpKSAmJlxuICAgICAgICAgICAgICAgICAgICB0aGlzLl9kYXRlQWRhcHRlci5nZXRNb250aCh0aGlzLl9kYXRlQWRhcHRlci50b2RheSgpKSA9PT0gbW9udGgubW9udGhWYWx1ZTtcblxuICAgICAgICAgICAgICAgICAgcmV0dXJuIGh0bWxgIDx0ZFxuICAgICAgICAgICAgICAgICAgICBjbGFzcz0ke2NsYXNzTWFwKHtcbiAgICAgICAgICAgICAgICAgICAgICAnc2JiLWNhbGVuZGFyX190YWJsZS1kYXRhJzogdHJ1ZSxcbiAgICAgICAgICAgICAgICAgICAgICAnc2JiLWNhbGVuZGFyX190YWJsZS1tb250aCc6IHRydWUsXG4gICAgICAgICAgICAgICAgICAgIH0pfVxuICAgICAgICAgICAgICAgICAgPlxuICAgICAgICAgICAgICAgICAgICA8YnV0dG9uXG4gICAgICAgICAgICAgICAgICAgICAgY2xhc3M9JHtjbGFzc01hcCh7XG4gICAgICAgICAgICAgICAgICAgICAgICAnc2JiLWNhbGVuZGFyX19jZWxsJzogdHJ1ZSxcbiAgICAgICAgICAgICAgICAgICAgICAgICdzYmItY2FsZW5kYXJfX2NlbGwtY3VycmVudCc6IGlzQ3VycmVudE1vbnRoLFxuICAgICAgICAgICAgICAgICAgICAgICAgJ3NiYi1jYWxlbmRhcl9fY3Jvc3NlZC1vdXQnOiAhaXNPdXRPZlJhbmdlICYmIGlzRmlsdGVyZWRPdXQsXG4gICAgICAgICAgICAgICAgICAgICAgICAnc2JiLWNhbGVuZGFyX19zZWxlY3RlZCc6IHNlbGVjdGVkLFxuICAgICAgICAgICAgICAgICAgICAgIH0pfVxuICAgICAgICAgICAgICAgICAgICAgIEBjbGljaz0keygpID0+IHRoaXMuX29uTW9udGhTZWxlY3Rpb24obW9udGgubW9udGhWYWx1ZSwgeWVhcil9XG4gICAgICAgICAgICAgICAgICAgICAgP2Rpc2FibGVkPSR7aXNPdXRPZlJhbmdlIHx8IGlzRmlsdGVyZWRPdXR9XG4gICAgICAgICAgICAgICAgICAgICAgYXJpYS1sYWJlbD0ke2Ake21vbnRoLmxvbmdWYWx1ZX0gJHt5ZWFyfWB9XG4gICAgICAgICAgICAgICAgICAgICAgYXJpYS1wcmVzc2VkPSR7c2VsZWN0ZWR9XG4gICAgICAgICAgICAgICAgICAgICAgYXJpYS1kaXNhYmxlZD0ke1N0cmluZyhpc091dE9mUmFuZ2UgfHwgaXNGaWx0ZXJlZE91dCl9XG4gICAgICAgICAgICAgICAgICAgICAgdGFiaW5kZXg9XCItMVwiXG4gICAgICAgICAgICAgICAgICAgICAgZGF0YS1tb250aD0ke21vbnRoLm1vbnRoVmFsdWUgfHwgbm90aGluZ31cbiAgICAgICAgICAgICAgICAgICAgICBAa2V5ZG93bj0keyhldnQ6IEtleWJvYXJkRXZlbnQpID0+IHRoaXMuX2hhbmRsZUtleWJvYXJkRXZlbnQoZXZ0KX1cbiAgICAgICAgICAgICAgICAgICAgPlxuICAgICAgICAgICAgICAgICAgICAgICR7bW9udGgudmFsdWV9XG4gICAgICAgICAgICAgICAgICAgIDwvYnV0dG9uPlxuICAgICAgICAgICAgICAgICAgPC90ZD5gO1xuICAgICAgICAgICAgICAgIH0pfVxuICAgICAgICAgICAgICA8L3RyPlxuICAgICAgICAgICAgYCxcbiAgICAgICAgICApfVxuICAgICAgICA8L3Rib2R5PlxuICAgICAgPC90YWJsZT5cbiAgICBgO1xuICB9XG5cbiAgLyoqIFNlbGVjdCB0aGUgbW9udGggYW5kIGNoYW5nZSB0aGUgdmlldyB0byBkYXkgc2VsZWN0aW9uLiAqL1xuICBwcml2YXRlIF9vbk1vbnRoU2VsZWN0aW9uKG1vbnRoOiBudW1iZXIsIHllYXI6IG51bWJlcik6IHZvaWQge1xuICAgIHRoaXMuX2Nob3Nlbk1vbnRoID0gbW9udGg7XG4gICAgdGhpcy5fbmV4dENhbGVuZGFyVmlldyA9ICdkYXknO1xuICAgIHRoaXMuX2luaXQoXG4gICAgICB0aGlzLl9kYXRlQWRhcHRlci5jcmVhdGVEYXRlKFxuICAgICAgICB5ZWFyLFxuICAgICAgICB0aGlzLl9jaG9zZW5Nb250aCxcbiAgICAgICAgdGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0RGF0ZSh0aGlzLl9hY3RpdmVEYXRlKSxcbiAgICAgICksXG4gICAgKTtcbiAgICB0aGlzLl9zdGFydFRhYmxlVHJhbnNpdGlvbigpO1xuICAgIHRoaXMuX2VtaXRNb250aENoYW5nZSgpO1xuICB9XG5cbiAgLyoqIFJlbmRlciB0aGUgdmlldyBmb3IgdGhlIHllYXIgc2VsZWN0aW9uLiAqL1xuICBwcml2YXRlIF9yZW5kZXJZZWFyVmlldygpOiBUZW1wbGF0ZVJlc3VsdCB7XG4gICAgcmV0dXJuIGh0bWxgXG4gICAgICA8ZGl2IGNsYXNzPVwic2JiLWNhbGVuZGFyX19jb250cm9sc1wiPlxuICAgICAgICAke3RoaXMuX2dldEFycm93KFxuICAgICAgICAgICdsZWZ0JyxcbiAgICAgICAgICAoKSA9PiB0aGlzLl9nb1RvRGlmZmVyZW50WWVhclJhbmdlKC1ZRUFSU19QRVJfUEFHRSksXG4gICAgICAgICAgaTE4blByZXZpb3VzWWVhclJhbmdlKFlFQVJTX1BFUl9QQUdFKVt0aGlzLl9sYW5ndWFnZS5jdXJyZW50XSxcbiAgICAgICAgICB0aGlzLl9wcmV2aW91c1llYXJSYW5nZURpc2FibGVkKCksXG4gICAgICAgICl9XG4gICAgICAgIDxkaXYgY2xhc3M9XCJzYmItY2FsZW5kYXJfX2NvbnRyb2xzLW1vbnRoXCI+JHt0aGlzLl9jcmVhdGVMYWJlbEZvclllYXJWaWV3KCl9PC9kaXY+XG4gICAgICAgICR7dGhpcy5fZ2V0QXJyb3coXG4gICAgICAgICAgJ3JpZ2h0JyxcbiAgICAgICAgICAoKSA9PiB0aGlzLl9nb1RvRGlmZmVyZW50WWVhclJhbmdlKFlFQVJTX1BFUl9QQUdFKSxcbiAgICAgICAgICBpMThuTmV4dFllYXJSYW5nZShZRUFSU19QRVJfUEFHRSlbdGhpcy5fbGFuZ3VhZ2UuY3VycmVudF0sXG4gICAgICAgICAgdGhpcy5fbmV4dFllYXJSYW5nZURpc2FibGVkKCksXG4gICAgICAgICl9XG4gICAgICA8L2Rpdj5cbiAgICAgIDxkaXYgY2xhc3M9XCJzYmItY2FsZW5kYXJfX3RhYmxlLW92ZXJmbG93LWJyZWFrXCI+XG4gICAgICAgIDxkaXYgY2xhc3M9XCJzYmItY2FsZW5kYXJfX3RhYmxlLWNvbnRhaW5lciBzYmItY2FsZW5kYXJfX3RhYmxlLXllYXItdmlld1wiPlxuICAgICAgICAgICR7dGhpcy5fY3JlYXRlWWVhclRhYmxlKHRoaXMuX3llYXJzKX1cbiAgICAgICAgICAke3RoaXMuX3dpZGUgPyB0aGlzLl9jcmVhdGVZZWFyVGFibGUodGhpcy5fbmV4dE1vbnRoWWVhcnMsIHRydWUpIDogbm90aGluZ31cbiAgICAgICAgPC9kaXY+XG4gICAgICA8L2Rpdj5cbiAgICBgO1xuICB9XG5cbiAgLyoqIENyZWF0ZXMgdGhlIGJ1dHRvbiBhcnJvdyBmb3IgYWxsIHRoZSB2aWV3cy4gKi9cbiAgcHJpdmF0ZSBfZ2V0QXJyb3coXG4gICAgZGlyZWN0aW9uOiAnbGVmdCcgfCAncmlnaHQnLFxuICAgIGNsaWNrOiAoKSA9PiB2b2lkLFxuICAgIGFyaWFMYWJlbDogc3RyaW5nLFxuICAgIGRpc2FibGVkOiBib29sZWFuLFxuICApOiBUZW1wbGF0ZVJlc3VsdCB7XG4gICAgcmV0dXJuIGh0bWxgPHNiYi1zZWNvbmRhcnktYnV0dG9uXG4gICAgICBzaXplPVwibVwiXG4gICAgICBpY29uLW5hbWU9XCJjaGV2cm9uLXNtYWxsLSR7ZGlyZWN0aW9ufS1zbWFsbFwiXG4gICAgICBhcmlhLWxhYmVsPSR7YXJpYUxhYmVsfVxuICAgICAgQGNsaWNrPSR7Y2xpY2t9XG4gICAgICA/ZGlzYWJsZWQ9JHtkaXNhYmxlZH1cbiAgICAgIGlkPVwic2JiLWNhbGVuZGFyX19jb250cm9scy0ke2RpcmVjdGlvbiA9PT0gJ2xlZnQnID8gJ3ByZXZpb3VzJyA6ICduZXh0J31cIlxuICAgID48L3NiYi1zZWNvbmRhcnktYnV0dG9uPmA7XG4gIH1cblxuICAvKiogQ3JlYXRlcyB0aGUgbGFiZWwgd2l0aCB0aGUgeWVhciByYW5nZSBmb3IgdGhlIHllYXJseSB2aWV3LiAqL1xuICBwcml2YXRlIF9jcmVhdGVMYWJlbEZvclllYXJWaWV3KCk6IFRlbXBsYXRlUmVzdWx0IHtcbiAgICBjb25zdCBmaXJzdFllYXI6IG51bWJlciA9IHRoaXMuX3llYXJzLmZsYXQoKVswXTtcbiAgICBjb25zdCBsYXN0WWVhckFycmF5OiBudW1iZXJbXSA9ICh0aGlzLl93aWRlID8gdGhpcy5fbmV4dE1vbnRoWWVhcnMgOiB0aGlzLl95ZWFycykuZmxhdCgpO1xuICAgIGNvbnN0IGxhc3RZZWFyOiBudW1iZXIgPSBsYXN0WWVhckFycmF5W2xhc3RZZWFyQXJyYXkubGVuZ3RoIC0gMV07XG4gICAgY29uc3QgeWVhckxhYmVsID0gYCR7Zmlyc3RZZWFyfSAtICR7bGFzdFllYXJ9YDtcbiAgICByZXR1cm4gaHRtbGBcbiAgICAgIDxidXR0b25cbiAgICAgICAgdHlwZT1cImJ1dHRvblwiXG4gICAgICAgIGlkPVwic2JiLWNhbGVuZGFyX195ZWFyLXNlbGVjdGlvblwiXG4gICAgICAgIGNsYXNzPVwic2JiLWNhbGVuZGFyX19jb250cm9scy1jaGFuZ2UtZGF0ZVwiXG4gICAgICAgIGFyaWEtbGFiZWw9XCIke2kxOG5DYWxlbmRhckRhdGVTZWxlY3Rpb25bdGhpcy5fbGFuZ3VhZ2UuY3VycmVudF19ICR7eWVhckxhYmVsfVwiXG4gICAgICAgIEBjbGljaz0keygpID0+IHRoaXMuX3Jlc2V0Q2FsZW5kYXJWaWV3QW5kRW1pdE1vbnRoQ2hhbmdlKHRydWUpfVxuICAgICAgPlxuICAgICAgICAke3llYXJMYWJlbH1cbiAgICAgICAgPHNiYi1pY29uIG5hbWU9XCJjaGV2cm9uLXNtYWxsLXVwLXNtYWxsXCI+PC9zYmItaWNvbj5cbiAgICAgIDwvYnV0dG9uPlxuICAgICAgPHNiYi1zY3JlZW4tcmVhZGVyLW9ubHkgcm9sZT1cInN0YXR1c1wiPiAke3llYXJMYWJlbH0gPC9zYmItc2NyZWVuLXJlYWRlci1vbmx5PlxuICAgIGA7XG4gIH1cblxuICAvKiogQ3JlYXRlcyB0aGUgdGFibGUgZm9yIHRoZSB5ZWFyIHNlbGVjdGlvbiB2aWV3LiAqL1xuICBwcml2YXRlIF9jcmVhdGVZZWFyVGFibGUoeWVhcnM6IG51bWJlcltdW10sIHNoaWZ0UmlnaHQgPSBmYWxzZSk6IFRlbXBsYXRlUmVzdWx0IHtcbiAgICBjb25zdCBub3cgPSB0aGlzLl9kYXRlQWRhcHRlci50b2RheSgpO1xuICAgIHJldHVybiBodG1sYCA8dGFibGVcbiAgICAgIGNsYXNzPVwic2JiLWNhbGVuZGFyX190YWJsZVwiXG4gICAgICBAYW5pbWF0aW9uZW5kPSR7KGU6IEFuaW1hdGlvbkV2ZW50KSA9PiB0aGlzLl90YWJsZUFuaW1hdGlvbkVuZChlKX1cbiAgICA+XG4gICAgICA8dGJvZHkgY2xhc3M9XCJzYmItY2FsZW5kYXJfX3RhYmxlLWJvZHlcIj5cbiAgICAgICAgJHt5ZWFycy5tYXAoXG4gICAgICAgICAgKHJvdzogbnVtYmVyW10pID0+XG4gICAgICAgICAgICBodG1sYCA8dHI+XG4gICAgICAgICAgICAgICR7cm93Lm1hcCgoeWVhcjogbnVtYmVyKSA9PiB7XG4gICAgICAgICAgICAgICAgbGV0IHNlbGVjdGVkOiBib29sZWFuO1xuICAgICAgICAgICAgICAgIGlmICh0aGlzLm11bHRpcGxlKSB7XG4gICAgICAgICAgICAgICAgICBzZWxlY3RlZCA9XG4gICAgICAgICAgICAgICAgICAgICh0aGlzLl9zZWxlY3RlZCBhcyBUW10pLmZpbmQoXG4gICAgICAgICAgICAgICAgICAgICAgKGRhdGU6IFQpID0+IHllYXIgPT09IHRoaXMuX2RhdGVBZGFwdGVyLmdldFllYXIoZGF0ZSksXG4gICAgICAgICAgICAgICAgICAgICkgIT09IHVuZGVmaW5lZDtcbiAgICAgICAgICAgICAgICB9IGVsc2Uge1xuICAgICAgICAgICAgICAgICAgY29uc3Qgc2VsZWN0ZWRZZWFyID0gdGhpcy5fc2VsZWN0ZWRcbiAgICAgICAgICAgICAgICAgICAgPyB0aGlzLl9kYXRlQWRhcHRlci5nZXRZZWFyKHRoaXMuX3NlbGVjdGVkIGFzIFQpXG4gICAgICAgICAgICAgICAgICAgIDogdW5kZWZpbmVkO1xuICAgICAgICAgICAgICAgICAgc2VsZWN0ZWQgPSAhIXRoaXMuX3NlbGVjdGVkICYmIHllYXIgPT09IHNlbGVjdGVkWWVhcjtcbiAgICAgICAgICAgICAgICB9XG4gICAgICAgICAgICAgICAgY29uc3QgaXNPdXRPZlJhbmdlID0gIXRoaXMuX2lzWWVhckluUmFuZ2UoeWVhcik7XG4gICAgICAgICAgICAgICAgY29uc3QgaXNGaWx0ZXJlZE91dCA9ICF0aGlzLl9pc1llYXJGaWx0ZXJlZE91dCh5ZWFyKTtcbiAgICAgICAgICAgICAgICBjb25zdCBpc0N1cnJlbnRZZWFyID0gdGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0WWVhcihub3cpID09PSB5ZWFyO1xuICAgICAgICAgICAgICAgIHJldHVybiBodG1sYCA8dGQgY2xhc3M9XCJzYmItY2FsZW5kYXJfX3RhYmxlLWRhdGEgc2JiLWNhbGVuZGFyX190YWJsZS15ZWFyXCI+XG4gICAgICAgICAgICAgICAgICA8YnV0dG9uXG4gICAgICAgICAgICAgICAgICAgIGNsYXNzPSR7Y2xhc3NNYXAoe1xuICAgICAgICAgICAgICAgICAgICAgICdzYmItY2FsZW5kYXJfX2NlbGwnOiB0cnVlLFxuICAgICAgICAgICAgICAgICAgICAgICdzYmItY2FsZW5kYXJfX2NlbGwtY3VycmVudCc6IGlzQ3VycmVudFllYXIsXG4gICAgICAgICAgICAgICAgICAgICAgJ3NiYi1jYWxlbmRhcl9fY3Jvc3NlZC1vdXQnOiAhaXNPdXRPZlJhbmdlICYmIGlzRmlsdGVyZWRPdXQsXG4gICAgICAgICAgICAgICAgICAgICAgJ3NiYi1jYWxlbmRhcl9fc2VsZWN0ZWQnOiBzZWxlY3RlZCxcbiAgICAgICAgICAgICAgICAgICAgfSl9XG4gICAgICAgICAgICAgICAgICAgIEBjbGljaz0keygpID0+IHRoaXMuX29uWWVhclNlbGVjdGlvbih5ZWFyLCBzaGlmdFJpZ2h0KX1cbiAgICAgICAgICAgICAgICAgICAgP2Rpc2FibGVkPSR7aXNPdXRPZlJhbmdlIHx8IGlzRmlsdGVyZWRPdXR9XG4gICAgICAgICAgICAgICAgICAgIGFyaWEtbGFiZWw9JHt5ZWFyfVxuICAgICAgICAgICAgICAgICAgICBhcmlhLXByZXNzZWQ9JHtzZWxlY3RlZH1cbiAgICAgICAgICAgICAgICAgICAgYXJpYS1kaXNhYmxlZD0ke1N0cmluZyhpc091dE9mUmFuZ2UgfHwgaXNGaWx0ZXJlZE91dCl9XG4gICAgICAgICAgICAgICAgICAgIHRhYmluZGV4PVwiLTFcIlxuICAgICAgICAgICAgICAgICAgICBkYXRhLXllYXI9JHt5ZWFyIHx8IG5vdGhpbmd9XG4gICAgICAgICAgICAgICAgICAgIEBrZXlkb3duPSR7KGV2dDogS2V5Ym9hcmRFdmVudCkgPT4gdGhpcy5faGFuZGxlS2V5Ym9hcmRFdmVudChldnQpfVxuICAgICAgICAgICAgICAgICAgPlxuICAgICAgICAgICAgICAgICAgICAke3llYXJ9XG4gICAgICAgICAgICAgICAgICA8L2J1dHRvbj5cbiAgICAgICAgICAgICAgICA8L3RkPmA7XG4gICAgICAgICAgICAgIH0pfVxuICAgICAgICAgICAgPC90cj5gLFxuICAgICAgICApfVxuICAgICAgPC90Ym9keT5cbiAgICA8L3RhYmxlPmA7XG4gIH1cblxuICAvKiogU2VsZWN0IHRoZSB5ZWFyIGFuZCBjaGFuZ2UgdGhlIHZpZXcgdG8gbW9udGggc2VsZWN0aW9uLiAqL1xuICBwcml2YXRlIF9vblllYXJTZWxlY3Rpb24oeWVhcjogbnVtYmVyLCByaWdodFNpZGU6IGJvb2xlYW4pOiB2b2lkIHtcbiAgICB0aGlzLl9jaG9zZW5ZZWFyID0gcmlnaHRTaWRlID8geWVhciAtIDEgOiB5ZWFyO1xuICAgIHRoaXMuX25leHRDYWxlbmRhclZpZXcgPSAnbW9udGgnO1xuICAgIHRoaXMuX2Fzc2lnbkFjdGl2ZURhdGUoXG4gICAgICB0aGlzLl9kYXRlQWRhcHRlci5jcmVhdGVEYXRlKFxuICAgICAgICB0aGlzLl9jaG9zZW5ZZWFyLFxuICAgICAgICB0aGlzLl9kYXRlQWRhcHRlci5nZXRNb250aCh0aGlzLl9hY3RpdmVEYXRlKSxcbiAgICAgICAgdGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0RGF0ZSh0aGlzLl9hY3RpdmVEYXRlKSxcbiAgICAgICksXG4gICAgKTtcbiAgICB0aGlzLl9zdGFydFRhYmxlVHJhbnNpdGlvbigpO1xuICB9XG5cbiAgcHJpdmF0ZSBfZ2V0VmlldygpOiBUZW1wbGF0ZVJlc3VsdCB7XG4gICAgaWYgKGlzU2VydmVyIHx8IHRoaXMuaHlkcmF0aW9uUmVxdWlyZWQpIHtcbiAgICAgIC8vIFRPRE86IFdlIGRpc2FibGUgU1NSIGZvciBjYWxlbmRhciBmb3Igbm93LiBGaWd1cmUgb3V0LCBpZiB0aGVyZSBpcyBhIHdheVxuICAgICAgLy8gdG8gZW5hYmxlIGl0LCB3aGlsZSBjb25zaWRlcmluZyBpMThuIGFuZCBkYXRlIGluZm9ybWF0aW9uLlxuICAgICAgcmV0dXJuIGh0bWxgJHtub3RoaW5nfWA7XG4gICAgfVxuICAgIHN3aXRjaCAodGhpcy5fY2FsZW5kYXJWaWV3KSB7XG4gICAgICBjYXNlICd5ZWFyJzpcbiAgICAgICAgcmV0dXJuIHRoaXMuX3JlbmRlclllYXJWaWV3KCk7XG4gICAgICBjYXNlICdtb250aCc6XG4gICAgICAgIHJldHVybiB0aGlzLl9yZW5kZXJNb250aFZpZXcoKTtcbiAgICAgIGNhc2UgJ2RheSc6XG4gICAgICBkZWZhdWx0OlxuICAgICAgICByZXR1cm4gdGhpcy5fcmVuZGVyRGF5VmlldygpO1xuICAgIH1cbiAgfVxuXG4gIHByaXZhdGUgX3RhYmxlQW5pbWF0aW9uRW5kKGV2ZW50OiBBbmltYXRpb25FdmVudCk6IHZvaWQge1xuICAgIGNvbnN0IHRhYmxlID0gZXZlbnQudGFyZ2V0IGFzIEhUTUxFbGVtZW50O1xuICAgIGlmIChldmVudC5hbmltYXRpb25OYW1lID09PSAnaGlkZScpIHtcbiAgICAgIHRhYmxlLmNsYXNzTGlzdC5yZW1vdmUoJ3NiYi1jYWxlbmRhcl9fdGFibGUtaGlkZScpO1xuICAgICAgaWYgKHRoaXMuX2NvbnRhaW5pbmdGb2N1cykge1xuICAgICAgICB0aGlzLl9yZXNldEZvY3VzID0gdHJ1ZTtcbiAgICAgIH1cbiAgICAgIHRoaXMuX2NhbGVuZGFyVmlldyA9IHRoaXMuX25leHRDYWxlbmRhclZpZXc7XG4gICAgfSBlbHNlIGlmIChldmVudC5hbmltYXRpb25OYW1lID09PSAnc2hvdycpIHtcbiAgICAgIHRoaXMuaW50ZXJuYWxzLnN0YXRlcy5kZWxldGUoJ3RyYW5zaXRpb24nKTtcbiAgICB9XG4gIH1cblxuICBwcml2YXRlIF9zdGFydFRhYmxlVHJhbnNpdGlvbigpOiB2b2lkIHtcbiAgICB0aGlzLmludGVybmFscy5zdGF0ZXMuYWRkKCd0cmFuc2l0aW9uJyk7XG4gICAgdGhpcy5zaGFkb3dSb290XG4gICAgICA/LnF1ZXJ5U2VsZWN0b3JBbGwoJ3RhYmxlJylcbiAgICAgID8uZm9yRWFjaCgoZSkgPT4gZS5jbGFzc0xpc3QudG9nZ2xlKCdzYmItY2FsZW5kYXJfX3RhYmxlLWhpZGUnKSk7XG4gIH1cblxuICBwcm90ZWN0ZWQgb3ZlcnJpZGUgcmVuZGVyKCk6IFRlbXBsYXRlUmVzdWx0IHtcbiAgICByZXR1cm4gaHRtbGA8ZGl2IGNsYXNzPVwic2JiLWNhbGVuZGFyX193cmFwcGVyXCI+JHt0aGlzLl9nZXRWaWV3KCl9PC9kaXY+YDtcbiAgfVxufVxuXG5kZWNsYXJlIGdsb2JhbCB7XG4gIGludGVyZmFjZSBIVE1MRWxlbWVudFRhZ05hbWVNYXAge1xuICAgIC8vIGVzbGludC1kaXNhYmxlLW5leHQtbGluZSBAdHlwZXNjcmlwdC1lc2xpbnQvbmFtaW5nLWNvbnZlbnRpb25cbiAgICAnc2JiLWNhbGVuZGFyJzogU2JiQ2FsZW5kYXJFbGVtZW50O1xuICB9XG4gIGludGVyZmFjZSBIVE1MRWxlbWVudEV2ZW50TWFwIHtcbiAgICBtb250aGNoYW5nZTogU2JiTW9udGhDaGFuZ2VFdmVudDtcbiAgfVxufVxuIl0sIm5hbWVzIjpbImRlZmF1bHRPcHRpb25zIiwiZGF5Il0sIm1hcHBpbmdzIjoiOzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7OztBQTRFTyxNQUFNLHFCQUFxQjtBQTJJM0IsTUFBTSxzQkFBc0IsdUJBQU8sSUFBSSxtQkFBbUI7QUNsTDFELFNBQVMsY0FBYyxNQUFNLE9BQU87QUFDekMsTUFBSSxPQUFPLFNBQVMsV0FBWSxRQUFPLEtBQUssS0FBSztBQUVqRCxNQUFJLFFBQVEsT0FBTyxTQUFTLFlBQVksdUJBQXVCO0FBQzdELFdBQU8sS0FBSyxtQkFBbUIsRUFBRSxLQUFLO0FBRXhDLE1BQUksZ0JBQWdCLEtBQU0sUUFBTyxJQUFJLEtBQUssWUFBWSxLQUFLO0FBRTNELFNBQU8sSUFBSSxLQUFLLEtBQUs7QUFDdkI7QUNOTyxTQUFTLE9BQU8sVUFBVSxTQUFTO0FBRXhDLFNB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LGdDQUFBO0FBZ0V2QyxpQkFBQSxJQUFBLE1BQUEsMEJBQUEsRUFBQSxNQUFBLFlBQUEsTUFBQSxnQkFBQSxRQUFBLE9BQUEsU0FBQSxPQUFBLFFBQUEsRUFBQSxLQUFBLENBQUEsUUFBQSxrQkFBQSxLQUFBLEtBQUEsQ0FBQSxRQUFBLElBQWlCLGNBQVksS0FBQSxDQUFBLEtBQUEsVUFBQTtBQUFBLFVBQVosZUFBWTtBQUFBLElBQUEsS0FBQSxVQUFBLFVBQUEsR0FBQSw0QkFBQSwrQkFBQTtBQTFLL0IsaUJBQUEsTUFBQSxtQkFBQSxFQUFBLE9BQUEsV0FBQSxHQUFBLGtCQUFBLEVBQUEsTUFBQSxTQUFBLE1BQUEsV0FBQSxNQUFBLFVBQUEsVUFBQSxHQUFBLE1BQUEsdUJBQUE7OztRQUd5QixHQUFBLFNBQXlCLENBQUMsaUJBQWlCLEtBQUssR0FDaEQsR0FBQSxTQUFTO0FBQUEsSUFDOUIsY0FBYztBQUFBLElBQ2QsYUFBYTtBQUFBLEVBQUEsR0FOWCxrQkFBQSxZQUFBLHVCQUFBLEdBQTBDOzs7IiwieF9nb29nbGVfaWdub3JlTGlzdCI6WzAsMSwyLDMsNCw1LDYsNyw4LDksMTAsMTEsMTIsMTMsMTRdfQ==