@sbb-esta/lyne-elements 4.5.1 → 4.7.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 (88) hide show
  1. package/autocomplete/autocomplete-base-element.js +34 -34
  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/chip-group/chip-group.component.js +69 -49
  8. package/core/styles/core.scss +46 -73
  9. package/core/styles/mixins/overlay.scss +25 -15
  10. package/core/styles/mixins/pearl-chain-bullet.scss +12 -6
  11. package/core.css +114 -49
  12. package/custom-elements.json +18902 -17898
  13. package/datepicker/common/datepicker-button.js +18 -14
  14. package/datepicker/datepicker/datepicker.component.js +1 -1
  15. package/datepicker/datepicker-next-day/datepicker-next-day.component.js +18 -26
  16. package/datepicker/datepicker-previous-day/datepicker-previous-day.component.js +20 -28
  17. package/development/autocomplete/autocomplete-base-element.d.ts +2 -0
  18. package/development/autocomplete/autocomplete-base-element.d.ts.map +1 -1
  19. package/development/autocomplete/autocomplete-base-element.js +37 -34
  20. package/development/calendar/{calendar.component.d.ts → calendar/calendar.component.d.ts} +22 -2
  21. package/development/calendar/calendar/calendar.component.d.ts.map +1 -0
  22. package/development/calendar/calendar/calendar.component.js +1990 -0
  23. package/development/calendar/calendar-day/calendar-day.component.d.ts +49 -0
  24. package/development/calendar/calendar-day/calendar-day.component.d.ts.map +1 -0
  25. package/development/calendar/calendar-day/calendar-day.component.js +255 -0
  26. package/development/calendar/calendar-day.d.ts +5 -0
  27. package/development/calendar/calendar-day.d.ts.map +1 -0
  28. package/development/calendar/calendar-day.js +5 -0
  29. package/development/calendar/calendar.d.ts +5 -0
  30. package/development/calendar/calendar.d.ts.map +1 -0
  31. package/development/calendar/calendar.js +6 -0
  32. package/development/calendar.d.ts +2 -1
  33. package/development/calendar.d.ts.map +1 -1
  34. package/development/calendar.js +6 -3
  35. package/development/chip/chip-group/chip-group.component.d.ts +6 -0
  36. package/development/chip/chip-group/chip-group.component.d.ts.map +1 -1
  37. package/development/chip/chip-group/chip-group.component.js +33 -4
  38. package/development/container/container/container.component.d.ts +2 -0
  39. package/development/container/container/container.component.d.ts.map +1 -1
  40. package/development/container/container/container.component.js +1 -1
  41. package/development/datepicker/common/datepicker-button.d.ts +3 -1
  42. package/development/datepicker/common/datepicker-button.d.ts.map +1 -1
  43. package/development/datepicker/common/datepicker-button.js +8 -4
  44. package/development/datepicker/datepicker/datepicker.component.d.ts.map +1 -1
  45. package/development/datepicker/datepicker/datepicker.component.js +2 -1
  46. package/development/datepicker/datepicker-next-day/datepicker-next-day.component.d.ts +1 -1
  47. package/development/datepicker/datepicker-next-day/datepicker-next-day.component.d.ts.map +1 -1
  48. package/development/datepicker/datepicker-next-day/datepicker-next-day.component.js +3 -12
  49. package/development/datepicker/datepicker-previous-day/datepicker-previous-day.component.d.ts +1 -1
  50. package/development/datepicker/datepicker-previous-day/datepicker-previous-day.component.d.ts.map +1 -1
  51. package/development/datepicker/datepicker-previous-day/datepicker-previous-day.component.js +3 -12
  52. package/development/dialog/dialog/dialog.component.js +24 -37
  53. package/development/dialog/dialog-actions/dialog-actions.component.js +2 -2
  54. package/development/dialog/dialog-close-button/dialog-close-button.component.js +2 -2
  55. package/development/footer/footer.component.js +4 -19
  56. package/development/form-field/form-field/form-field.component.js +8 -5
  57. package/development/mini-calendar/mini-calendar-month/mini-calendar-month.component.d.ts +1 -1
  58. package/development/mini-calendar/mini-calendar-month/mini-calendar-month.component.d.ts.map +1 -1
  59. package/development/mini-calendar/mini-calendar-month/mini-calendar-month.component.js +2 -2
  60. package/development/option/optgroup/optgroup-base-element.js +4 -13
  61. package/development/option/option/option-base-element.d.ts.map +1 -1
  62. package/development/option/option/option-base-element.js +12 -14
  63. package/development/option/option/option.component.js +4 -28
  64. package/development/option/option-hint/option-hint.component.d.ts.map +1 -1
  65. package/development/option/option-hint/option-hint.component.js +11 -26
  66. package/development/select/select.component.js +11 -16
  67. package/development/status/status.component.d.ts.map +1 -1
  68. package/development/status/status.component.js +20 -26
  69. package/dialog/dialog/dialog.component.js +19 -19
  70. package/dialog/dialog-actions/dialog-actions.component.js +6 -6
  71. package/dialog/dialog-close-button/dialog-close-button.component.js +9 -9
  72. package/footer/footer.component.js +22 -22
  73. package/form-field/form-field/form-field.component.js +11 -11
  74. package/index.d.ts +3 -1
  75. package/index.js +3 -1
  76. package/mini-calendar/mini-calendar-month/mini-calendar-month.component.js +1 -1
  77. package/off-brand-theme.css +114 -49
  78. package/option/optgroup/optgroup-base-element.js +22 -22
  79. package/option/option/option-base-element.js +11 -13
  80. package/option/option/option.component.js +15 -15
  81. package/option/option-hint/option-hint.component.js +16 -18
  82. package/package.json +11 -1
  83. package/safety-theme.css +114 -49
  84. package/select/select.component.js +18 -18
  85. package/standard-theme.css +114 -49
  86. package/status/status.component.js +27 -29
  87. package/development/calendar/calendar.component.d.ts.map +0 -1
  88. 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/LmluKTtcbiAgY29uc3QgZGlmZiA9ICtzdGFydE9mV2VlayhfZGF0ZSwgb3B0aW9ucykgLSArc3RhcnRPZldlZWtZZWFyKF9kYXRlLCBvcHRpb25zKTtcblxuICAvLyBSb3VuZCB0aGUgbnVtYmVyIG9mIHdlZWtzIHRvIHRoZSBuZWFyZXN0IGludGVnZXIgYmVjYXVzZSB0aGUgbnVtYmVyIG9mXG4gIC8vIG1pbGxpc2Vjb25kcyBpbiBhIHdlZWsgaXMgbm90IGNvbnN0YW50IChlLmcuIGl0J3MgZGlmZmVyZW50IGluIHRoZSB3ZWVrIG9mXG4gIC8vIHRoZSBkYXlsaWdodCBzYXZpbmcgdGltZSBjbG9jayBzaGlmdCkuXG4gIHJldHVybiBNYXRoLnJvdW5kKGRpZmYgLyBtaWxsaXNlY29uZHNJbldlZWspICsgMTtcbn1cblxuLy8gRmFsbGJhY2sgZm9yIG1vZHVsYXJpemVkIGltcG9ydHM6XG5leHBvcnQgZGVmYXVsdCBnZXRXZWVrO1xuIiwiaW1wb3J0IHsgZWFjaFdlZWtPZkludGVydmFsLCBlbmRPZk1vbnRoLCBnZXRXZWVrLCBzdGFydE9mTW9udGggfSBmcm9tICdkYXRlLWZucyc7XG5pbXBvcnQge1xuICB0eXBlIENTU1Jlc3VsdEdyb3VwLFxuICBodG1sLFxuICBpc1NlcnZlcixcbiAgTGl0RWxlbWVudCxcbiAgbm90aGluZyxcbiAgdHlwZSBQcm9wZXJ0eVZhbHVlcyxcbiAgdHlwZSBUZW1wbGF0ZVJlc3VsdCxcbn0gZnJvbSAnbGl0JztcbmltcG9ydCB7IGN1c3RvbUVsZW1lbnQsIHByb3BlcnR5LCBzdGF0ZSB9IGZyb20gJ2xpdC9kZWNvcmF0b3JzLmpzJztcbmltcG9ydCB7IGNsYXNzTWFwIH0gZnJvbSAnbGl0L2RpcmVjdGl2ZXMvY2xhc3MtbWFwLmpzJztcblxuaW1wb3J0IHsgaXNBcnJvd0tleU9yUGFnZUtleXNQcmVzc2VkIH0gZnJvbSAnLi4vLi4vY29yZS9hMTF5LnRzJztcbmltcG9ydCB7IHJlYWRDb25maWcgfSBmcm9tICcuLi8uLi9jb3JlL2NvbmZpZy50cyc7XG5pbXBvcnQge1xuICBTYmJMYW5ndWFnZUNvbnRyb2xsZXIsXG4gIFNiYk1lZGlhTWF0Y2hlckNvbnRyb2xsZXIsXG4gIFNiYk1lZGlhUXVlcnlCcmVha3BvaW50TGFyZ2VBbmRBYm92ZSxcbn0gZnJvbSAnLi4vLi4vY29yZS9jb250cm9sbGVycy50cyc7XG5pbXBvcnQgdHlwZSB7IERhdGVBZGFwdGVyIH0gZnJvbSAnLi4vLi4vY29yZS9kYXRldGltZS50cyc7XG5pbXBvcnQge1xuICBEQVlTX1BFUl9ST1csXG4gIGRlZmF1bHREYXRlQWRhcHRlcixcbiAgTU9OVEhTX1BFUl9QQUdFLFxuICBNT05USFNfUEVSX1JPVyxcbiAgWUVBUlNfUEVSX1BBR0UsXG4gIFlFQVJTX1BFUl9ST1csXG59IGZyb20gJy4uLy4uL2NvcmUvZGF0ZXRpbWUudHMnO1xuaW1wb3J0IHsgZm9yY2VUeXBlLCBoYW5kbGVEaXN0aW5jdENoYW5nZSwgcGxhaW5EYXRlIH0gZnJvbSAnLi4vLi4vY29yZS9kZWNvcmF0b3JzLnRzJztcbmltcG9ydCB7XG4gIGkxOG5DYWxlbmRhckRhdGVTZWxlY3Rpb24sXG4gIGkxOG5DYWxlbmRhcldlZWtOdW1iZXIsXG4gIGkxOG5OZXh0TW9udGgsXG4gIGkxOG5OZXh0WWVhcixcbiAgaTE4bk5leHRZZWFyUmFuZ2UsXG4gIGkxOG5QcmV2aW91c01vbnRoLFxuICBpMThuUHJldmlvdXNZZWFyLFxuICBpMThuUHJldmlvdXNZZWFyUmFuZ2UsXG4gIGkxOG5ZZWFyTW9udGhTZWxlY3Rpb24sXG59IGZyb20gJy4uLy4uL2NvcmUvaTE4bi50cyc7XG5pbXBvcnQgdHlwZSB7IFNiYk9yaWVudGF0aW9uIH0gZnJvbSAnLi4vLi4vY29yZS9pbnRlcmZhY2VzLnRzJztcbmltcG9ydCB7IFNiYkVsZW1lbnRJbnRlcm5hbHNNaXhpbiwgU2JiSHlkcmF0aW9uTWl4aW4gfSBmcm9tICcuLi8uLi9jb3JlL21peGlucy50cyc7XG5pbXBvcnQgeyBib3hTaXppbmdTdHlsZXMgfSBmcm9tICcuLi8uLi9jb3JlL3N0eWxlcy50cyc7XG5pbXBvcnQgdHlwZSB7IFNiYkNhbGVuZGFyRGF5RWxlbWVudCB9IGZyb20gJy4uL2NhbGVuZGFyLWRheS9jYWxlbmRhci1kYXkuY29tcG9uZW50LnRzJztcblxuaW1wb3J0IHN0eWxlIGZyb20gJy4vY2FsZW5kYXIuc2Nzcz9saXQmaW5saW5lJztcblxuaW1wb3J0ICcuLi8uLi9idXR0b24vc2Vjb25kYXJ5LWJ1dHRvbi50cyc7XG5pbXBvcnQgJy4uLy4uL2ljb24udHMnO1xuaW1wb3J0ICcuLi8uLi9zY3JlZW4tcmVhZGVyLW9ubHkudHMnO1xuaW1wb3J0ICcuLi9jYWxlbmRhci1kYXkvY2FsZW5kYXItZGF5LmNvbXBvbmVudC50cyc7XG5cbmV4cG9ydCBjbGFzcyBTYmJNb250aENoYW5nZUV2ZW50IGV4dGVuZHMgRXZlbnQge1xuICBwcml2YXRlIHJlYWRvbmx5IF9yYW5nZTogcmVhZG9ubHkgRGF5W107XG5cbiAgcHVibGljIGdldCByYW5nZSgpOiByZWFkb25seSBEYXlbXSB7XG4gICAgcmV0dXJuIHRoaXMuX3JhbmdlO1xuICB9XG5cbiAgcHVibGljIGNvbnN0cnVjdG9yKHJhbmdlOiByZWFkb25seSBEYXlbXSkge1xuICAgIHN1cGVyKCdtb250aGNoYW5nZScsIHsgYnViYmxlczogdHJ1ZSwgY29tcG9zZWQ6IHRydWUgfSk7XG4gICAgdGhpcy5fcmFuZ2UgPSBPYmplY3QuZnJlZXplKHJhbmdlIHx8IFtdKTtcbiAgfVxufVxuXG4vKipcbiAqIFBhcmFtZXRlcnMgbmVlZGVkIGluIHllYXIgYW5kIG1vbnRoIHZpZXdzIHRvIGNvcnJlY3RseSBjYWxjdWxhdGUgdGhlIG5leHQgZWxlbWVudCBpbiBrZXlib2FyZCBuYXZpZ2F0aW9uLlxuICpcbiAqIFRoZSBjZWxsJ3MgaW5kZXggYW5kIHRoZSBlbGVtZW50J3MgaW5kZXggaW4gaXRzIG1vbnRoIC8geWVhciBiYXRjaCBtdXN0IGJlIGRpc3Rpbmd1aXNoZWQ6XG4gKiB0aGUgZmlyc3QgaXMgdGhlIGluZGV4IG9mIHRoZSBlbGVtZW50IGluIHRoZSBhcnJheSBvZiBhbGwgdGhlIHJlbmRlcmVkIGNlbGxzLCB3aGlsZSB0aGUgc2Vjb25kIGlzIHRoZSBpbmRleCBvZiB0aGUgZWxlbWVudCByZWxhdGl2ZSB0byBpdHMgdGFibGUuXG4gKiBJbiBub24td2lkZSBtb2RlLCB0aGUgd3RvIGFyZSB0aGUgc2FtZSwgd2hpbGUgaW4gd2lkZSBtb2RlIHRoZSBjZWxsJ3MgaW5kZXggY2FuIGdvIGZyb20gMCB0byA0NyBmb3IgeWVhcnMgYW5kIGZyb20gMCB0byAyMyBmb3IgbW9udGhzLFxuICogd2hpbGUgdGhlIGVsZW1lbnQgaW5kZXggZ29lcyBmcm9tIDAgdG8sIHJlc3BlY3RpdmVseSwgMjMgYW5kIDExLlxuICovXG5pbnRlcmZhY2UgQ2FsZW5kYXJLZXlib2FyZE5hdmlnYXRpb25Nb250aFllYXJWaWV3c1BhcmFtZXRlcnMge1xuICAvKiogVGhlIGVsZW1lbnQgaW5kZXggd2l0aGluIGl0cyB5ZWFyIG9yIG1vbnRoIHJhbmdlLiAqL1xuICBlbGVtZW50SW5kZXhGb3JXaWRlTW9kZTogbnVtYmVyO1xuICAvKiogSW4gd2lkZSBtb2RlLCB0aGUgaW5kZXggb2YgdGhlIGZpcnN0IGVsZW1lbnQgaW4gdGhlIHNlY29uZCBwYW5lbCwgb3IsIGFsdGVybmF0aXZlbHksIHRoZSBudW1iZXIgb2YgZWxlbWVudHMgaW4gdGhlIGZpcnN0IHBhbmVsLiAqL1xuICBvZmZzZXRGb3JXaWRlTW9kZTogbnVtYmVyO1xuICAvKiogVGhlIGluZGV4IG9mIHRoZSBsYXN0IGVsZW1lbnQgd2l0aGluIHRoZSBlbGVtZW50J3MgbW9udGggKG9yIHllYXIgcmFuZ2UpLiAqL1xuICBsYXN0RWxlbWVudEluZGV4Rm9yV2lkZU1vZGU6IG51bWJlcjtcbiAgLyoqIFRoZSBudW1iZXIgb2YgY2VsbHMgZGlzcGxheWVkIGluIGEgc2luZ2xlIHJvdywgZGVwZW5kaW5nIG9uIHRoZSByZW5kZXJlZCB2aWV3LiAqL1xuICB2ZXJ0aWNhbE9mZnNldDogbnVtYmVyO1xufVxuXG4vKipcbiAqIFBhcmFtZXRlcnMgbmVlZGVkIGluIGRheSB2aWV3IHRvIGNvcnJlY3RseSBjYWxjdWxhdGUgdGhlIG5leHQgZWxlbWVudCBpbiBrZXlib2FyZCBuYXZpZ2F0aW9uLlxuICpcbiAqIEluIG9yaWVudGF0aW9uPSd2ZXJ0aWNhbCcsIGl0J3Mgbm90IHBvc3NpYmxlIHRvIHJlbHkgb24gYW55IGFycmF5L2luZGV4IHRvIGNhbGN1bGF0ZSB0aGUgZWxlbWVudCB0byBuYXZpZ2F0ZSB0byxcbiAqIHNvIGNhbGN1bGF0aW9ucyBvbiBkYXRlcyBtdXN0IGJlIGRvbmUsIHdoaWNoIHNob3VsZCBjb25zaWRlciB2aWV3IGJvdW5kYXJpZXMsIG9mZnNldHMgYW5kIG1vbnRoJ3MgbGVuZ3RoLlxuICovXG5pbnRlcmZhY2UgQ2FsZW5kYXJLZXlib2FyZE5hdmlnYXRpb25EYXlWaWV3UGFyYW1ldGVycyB7XG4gIC8qKiBUaGUgZmlyc3QgZGF5IHJlbmRlcmVkLiAqL1xuICBmaXJzdERheUluVmlldzogc3RyaW5nIHwgbnVsbDtcbiAgLyoqIFRoZSBsYXN0IGRheSByZW5kZXJlZC4gSXQgZGVwZW5kcyBvbiB0aGUgJ3dpZGUnIHZhbHVlLiAqL1xuICBsYXN0RGF5SW5WaWV3OiBzdHJpbmcgfCBudWxsO1xuICAvKiogVGhlIG9mZnNldCBmcm9tIHRoZSBmaXJzdCBkYXkgb2YgdGhlIHdlZWsgKE1vbmRheSkgb2YgdGhlIGZpcnN0IHJlbmRlcmVkIG1vbnRoLiAqL1xuICBmaXJzdE1vbnRoT2Zmc2V0OiBudW1iZXI7XG4gIC8qKiBUaGUgbnVtYmVyIG9mIGRheXMgaW4gdGhlIGZpcnN0IHJlbmRlcmVkIG1vbnRoLiAqL1xuICBmaXJzdE1vbnRoTGVuZ3RoOiBudW1iZXI7XG4gIC8qKiBUaGUgb2Zmc2V0IGZyb20gdGhlIGZpcnN0IGRheSBvZiB0aGUgd2VlayAoTW9uZGF5KSBvZiB0aGUgc2Vjb25kIHJlbmRlcmVkIG1vbnRoLiBJZiB3aWRlIGlzIGZhbHNlLCBpdCdzIGVxdWFsIHRvIHplcm8uICovXG4gIHNlY29uZE1vbnRoT2Zmc2V0OiBudW1iZXI7XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgRGF5PFQgPSBEYXRlPiB7XG4gIC8qKiBEYXRlIGFzIElTTyBzdHJpbmcuICovXG4gIHZhbHVlOiBzdHJpbmc7XG4gIGRheVZhbHVlOiBzdHJpbmc7XG4gIG1vbnRoVmFsdWU6IHN0cmluZztcbiAgeWVhclZhbHVlOiBzdHJpbmc7XG4gIGRhdGVWYWx1ZTogVDtcbiAgd2Vla1ZhbHVlOiBudW1iZXI7XG4gIHdlZWtEYXlWYWx1ZTogbnVtYmVyO1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIE1vbnRoIHtcbiAgdmFsdWU6IHN0cmluZztcbiAgbG9uZ1ZhbHVlOiBzdHJpbmc7XG4gIG1vbnRoVmFsdWU6IG51bWJlcjtcbn1cblxuZXhwb3J0IGludGVyZmFjZSBXZWVrZGF5IHtcbiAgbG9uZzogc3RyaW5nO1xuICBuYXJyb3c6IHN0cmluZztcbn1cblxuZXhwb3J0IHR5cGUgQ2FsZW5kYXJWaWV3ID0gJ2RheScgfCAnbW9udGgnIHwgJ3llYXInO1xuXG4vKipcbiAqIEl0IGRpc3BsYXlzIGEgY2FsZW5kYXIgd2hpY2ggYWxsb3dzIGNob29zaW5nIGEgZGF0ZS5cbiAqXG4gKiBAc2xvdCAtIFVzZSB0aGUgdW5uYW1lZCBzbG90IHRvIGFkZCBjdXN0b21pemVkIGBzYmItY2FsZW5kYXItZGF5YCBlbGVtZW50cy5cbiAqL1xuZXhwb3J0XG5AY3VzdG9tRWxlbWVudCgnc2JiLWNhbGVuZGFyJylcbmNsYXNzIFNiYkNhbGVuZGFyRWxlbWVudDxUID0gRGF0ZT4gZXh0ZW5kcyBTYmJIeWRyYXRpb25NaXhpbihTYmJFbGVtZW50SW50ZXJuYWxzTWl4aW4oTGl0RWxlbWVudCkpIHtcbiAgcHVibGljIHN0YXRpYyBvdmVycmlkZSBzdHlsZXM6IENTU1Jlc3VsdEdyb3VwID0gW2JveFNpemluZ1N0eWxlcywgc3R5bGVdO1xuICBwdWJsaWMgc3RhdGljIHJlYWRvbmx5IGV2ZW50cyA9IHtcbiAgICBkYXRlc2VsZWN0ZWQ6ICdkYXRlc2VsZWN0ZWQnLFxuICAgIG1vbnRoY2hhbmdlOiAnbW9udGhjaGFuZ2UnLFxuICB9IGFzIGNvbnN0O1xuXG4gIC8qKiBJZiBzZXQgdG8gdHJ1ZSwgdHdvIG1vbnRocyBhcmUgZGlzcGxheWVkICovXG4gIEBmb3JjZVR5cGUoKVxuICBAcHJvcGVydHkoeyB0eXBlOiBCb29sZWFuIH0pXG4gIHB1YmxpYyBhY2Nlc3NvciB3aWRlOiBib29sZWFuID0gZmFsc2U7XG5cbiAgLyoqIFRoZSBpbml0aWFsIHZpZXcgb2YgdGhlIGNhbGVuZGFyIHdoaWNoIHNob3VsZCBiZSBkaXNwbGF5ZWQgb24gb3BlbmluZy4gKi9cbiAgQHByb3BlcnR5KCkgcHVibGljIGFjY2Vzc29yIHZpZXc6IENhbGVuZGFyVmlldyA9ICdkYXknO1xuXG4gIC8qKlxuICAgKiBUaGUgbWluaW11bSB2YWxpZCBkYXRlLiBBY2NlcHRzIGEgZGF0ZSBvYmplY3Qgb3IgbnVsbC5cbiAgICogQWNjZXB0cyBhbiBJU084NjAxIGZvcm1hdHRlZCBzdHJpbmcgKGUuZy4gMjAyNC0xMi0yNCkgYXMgYXR0cmlidXRlLlxuICAgKi9cbiAgQHBsYWluRGF0ZSgpXG4gIEBwcm9wZXJ0eSgpXG4gIHB1YmxpYyBhY2Nlc3NvciBtaW46IFQgfCBudWxsID0gbnVsbDtcblxuICAvKipcbiAgICogVGhlIG1heGltdW0gdmFsaWQgZGF0ZS4gQWNjZXB0cyBhIGRhdGUgb2JqZWN0IG9yIG51bGwuXG4gICAqIEFjY2VwdHMgYW4gSVNPODYwMSBmb3JtYXR0ZWQgc3RyaW5nIChlLmcuIDIwMjQtMTItMjQpIGFzIGF0dHJpYnV0ZS5cbiAgICovXG4gIEBwbGFpbkRhdGUoKVxuICBAcHJvcGVydHkoKVxuICBwdWJsaWMgYWNjZXNzb3IgbWF4OiBUIHwgbnVsbCA9IG51bGw7XG5cbiAgLyoqIFdoZXRoZXIgdGhlIGNhbGVuZGFyIGFsbG93cyBmb3IgbXVsdGlwbGUgZGF0ZSBzZWxlY3Rpb24uICovXG4gIEBmb3JjZVR5cGUoKVxuICBAaGFuZGxlRGlzdGluY3RDaGFuZ2UoKGU6IFNiYkNhbGVuZGFyRWxlbWVudDxUPiwgbmV3VmFsdWU6IGJvb2xlYW4pID0+XG4gICAgZS5fb25NdWx0aXBsZUNoYW5nZWQobmV3VmFsdWUpLFxuICApXG4gIEBwcm9wZXJ0eSh7IHR5cGU6IEJvb2xlYW4gfSlcbiAgcHVibGljIGFjY2Vzc29yIG11bHRpcGxlOiBib29sZWFuID0gZmFsc2U7XG5cbiAgLyoqXG4gICAqIFRoZSBzZWxlY3RlZCBkYXRlOiBhY2NlcHRzIGEgZGF0ZSBvYmplY3QsIG9yLCBpZiBgbXVsdGlwbGVgLCBhbiBhcnJheSBvZiBkYXRlcy5cbiAgICovXG4gIEBwcm9wZXJ0eSgpXG4gIHB1YmxpYyBzZXQgc2VsZWN0ZWQodmFsdWU6IFQgfCBUW10gfCBudWxsKSB7XG4gICAgaWYgKEFycmF5LmlzQXJyYXkodmFsdWUpKSB7XG4gICAgICB0aGlzLl9zZWxlY3RlZCA9IHZhbHVlXG4gICAgICAgIC5tYXAoKGRhdGVMaWtlOiBUKSA9PlxuICAgICAgICAgIHRoaXMuX2RhdGVBZGFwdGVyLmdldFZhbGlkRGF0ZU9yTnVsbCh0aGlzLl9kYXRlQWRhcHRlci5kZXNlcmlhbGl6ZShkYXRlTGlrZSkpLFxuICAgICAgICApXG4gICAgICAgIC5maWx0ZXIoKGRhdGU6IFQgfCBudWxsKTogZGF0ZSBpcyBUID0+IGRhdGUgIT09IG51bGwpXG4gICAgICAgIC5maWx0ZXIoXG4gICAgICAgICAgKGRhdGU6IFQpID0+XG4gICAgICAgICAgICAhdGhpcy5faXNEYXlJblJhbmdlKHRoaXMuX2RhdGVBZGFwdGVyLnRvSXNvODYwMShkYXRlKSkgfHwgdGhpcy5fZGF0ZUZpbHRlcihkYXRlKSxcbiAgICAgICAgKTtcbiAgICB9IGVsc2Uge1xuICAgICAgY29uc3Qgc2VsZWN0ZWREYXRlID0gdGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0VmFsaWREYXRlT3JOdWxsKFxuICAgICAgICB0aGlzLl9kYXRlQWRhcHRlci5kZXNlcmlhbGl6ZSh2YWx1ZSksXG4gICAgICApO1xuICAgICAgaWYgKFxuICAgICAgICAhIXNlbGVjdGVkRGF0ZSAmJlxuICAgICAgICAoIXRoaXMuX2lzRGF5SW5SYW5nZSh0aGlzLl9kYXRlQWRhcHRlci50b0lzbzg2MDEoc2VsZWN0ZWREYXRlKSkgfHxcbiAgICAgICAgICB0aGlzLl9kYXRlRmlsdGVyKHNlbGVjdGVkRGF0ZSkpXG4gICAgICApIHtcbiAgICAgICAgdGhpcy5fc2VsZWN0ZWQgPSBzZWxlY3RlZERhdGU7XG4gICAgICB9IGVsc2Uge1xuICAgICAgICB0aGlzLl9zZWxlY3RlZCA9IG51bGw7XG4gICAgICB9XG4gICAgfVxuICB9XG4gIHB1YmxpYyBnZXQgc2VsZWN0ZWQoKTogVCB8IFRbXSB8IG51bGwge1xuICAgIHJldHVybiB0aGlzLl9zZWxlY3RlZDtcbiAgfVxuICBAc3RhdGUoKSBwcml2YXRlIGFjY2Vzc29yIF9zZWxlY3RlZDogVCB8IFRbXSB8IG51bGwgPSBudWxsO1xuXG4gIC8qKiBBIGZ1bmN0aW9uIHVzZWQgdG8gZmlsdGVyIG91dCBkYXRlcy4gKi9cbiAgQHByb3BlcnR5KHsgYXR0cmlidXRlOiAnZGF0ZS1maWx0ZXInIH0pXG4gIHB1YmxpYyBhY2Nlc3NvciBkYXRlRmlsdGVyOiAoKGRhdGU6IFQgfCBudWxsKSA9PiBib29sZWFuKSB8IG51bGwgPSBudWxsO1xuXG4gIC8qKiBUaGUgb3JpZW50YXRpb24gb2YgZGF5cyBpbiB0aGUgY2FsZW5kYXIuICovXG4gIEBwcm9wZXJ0eSh7IHJlZmxlY3Q6IHRydWUgfSkgcHVibGljIGFjY2Vzc29yIG9yaWVudGF0aW9uOiBTYmJPcmllbnRhdGlvbiA9ICdob3Jpem9udGFsJztcblxuICAvKiogV2hldGhlciBpdCBoYXMgdG8gZGlzcGxheSB0aGUgd2VlayBudW1iZXJzIGluIGFkZGl0aW9uIHRvIHdlZWsgZGF5cy4gKi9cbiAgQGZvcmNlVHlwZSgpXG4gIEBwcm9wZXJ0eSh7IGF0dHJpYnV0ZTogJ3dlZWstbnVtYmVycycsIHR5cGU6IEJvb2xlYW4gfSlcbiAgcHVibGljIGFjY2Vzc29yIHdlZWtOdW1iZXJzOiBib29sZWFuID0gZmFsc2U7XG5cbiAgcHJpdmF0ZSBfZGF0ZUFkYXB0ZXI6IERhdGVBZGFwdGVyPFQ+ID0gcmVhZENvbmZpZygpLmRhdGV0aW1lPy5kYXRlQWRhcHRlciA/PyBkZWZhdWx0RGF0ZUFkYXB0ZXI7XG5cbiAgLyoqIFRoZSBjdXJyZW50bHkgYWN0aXZlIGRhdGUuICovXG4gIEBzdGF0ZSgpIHByaXZhdGUgYWNjZXNzb3IgX2FjdGl2ZURhdGU6IFQgPSB0aGlzLl9kYXRlQWRhcHRlci50b2RheSgpO1xuXG4gIC8qKiBUaGUgY3VycmVudCB3aWRlIHByb3BlcnR5IGNvbnNpZGVyaW5nIHByb3BlcnR5IHZhbHVlIGFuZCBicmVha3BvaW50cy4gRnJvbSB6ZXJvIHRvIHNtYWxsIGB3aWRlYCBoYXMgYWx3YXlzIHRvIGJlIGZhbHNlLiAqL1xuICBAc3RhdGUoKVxuICBwcml2YXRlIHNldCBfd2lkZSh3aWRlOiBib29sZWFuKSB7XG4gICAgdGhpcy50b2dnbGVTdGF0ZSgnd2lkZScsIHdpZGUpO1xuICAgIHRoaXMuX3dpZGVJbnRlcm5hbCA9IHdpZGU7XG4gIH1cbiAgcHJpdmF0ZSBnZXQgX3dpZGUoKTogYm9vbGVhbiB7XG4gICAgcmV0dXJuIHRoaXMuX3dpZGVJbnRlcm5hbDtcbiAgfVxuICAvLyBXZSBrZWVwIHRoZSBzdGF0ZSBpbiBhIGZpZWxkIGJlY2F1c2Ugb2YgdGhlIFdlYktpdCBidWcgaHR0cHM6Ly9idWdzLndlYmtpdC5vcmcvc2hvd19idWcuY2dpP2lkPTMwMzQ2Ny5cbiAgLy8gVE9ETzogcmUtY2hlY2sgd2hldGhlciBmaWVsZCBpcyBuZWVkZWRcbiAgcHJpdmF0ZSBfd2lkZUludGVybmFsOiBib29sZWFuID0gZmFsc2U7XG5cbiAgQHN0YXRlKCkgcHJpdmF0ZSBhY2Nlc3NvciBfY2FsZW5kYXJWaWV3OiBDYWxlbmRhclZpZXcgPSAnZGF5JztcblxuICBwcml2YXRlIF9uZXh0Q2FsZW5kYXJWaWV3OiBDYWxlbmRhclZpZXcgPSAnZGF5JztcblxuICAvKiogSW5mb3JtYXRpb24gYWJvdXQgdGhlIHJlbmRlcmVkIGRheSB2aWV3OyB1c2VkIGluIGtleWJvYXJkIG5hdmlnYXRpb24uICovXG4gIHByaXZhdGUgX2tleWJvYXJkTmF2aWdhdGlvbkRheVZpZXdQYXJhbWV0ZXJzOiBDYWxlbmRhcktleWJvYXJkTmF2aWdhdGlvbkRheVZpZXdQYXJhbWV0ZXJzID0ge1xuICAgIGZpcnN0RGF5SW5WaWV3OiBudWxsLFxuICAgIGxhc3REYXlJblZpZXc6IG51bGwsXG4gICAgZmlyc3RNb250aE9mZnNldDogMCxcbiAgICBmaXJzdE1vbnRoTGVuZ3RoOiAwLFxuICAgIHNlY29uZE1vbnRoT2Zmc2V0OiAwLFxuICB9O1xuXG4gIC8qKiBBIGxpc3Qgb2YgZGF5cywgaW4gdHdvIGZvcm1hdHMgKGxvbmcgYW5kIHNpbmdsZSBjaGFyKS4gKi9cbiAgcHJpdmF0ZSBfd2Vla2RheXMhOiBXZWVrZGF5W107XG5cbiAgLyoqIEdyaWQgb2YgY2FsZW5kYXIgY2VsbHMgcmVwcmVzZW50aW5nIHRoZSBkYXRlcyBvZiB0aGUgbW9udGguICovXG4gIHByaXZhdGUgX3dlZWtzOiBEYXk8VD5bXVtdID0gW107XG5cbiAgLyoqIEdyaWQgb2YgY2FsZW5kYXIgY2VsbHMgcmVwcmVzZW50aW5nIG1vbnRocy4gKi9cbiAgcHJpdmF0ZSBfbW9udGhzITogTW9udGhbXVtdO1xuXG4gIC8qKiBHcmlkIG9mIGNhbGVuZGFyIGNlbGxzIHJlcHJlc2VudGluZyB5ZWFycy4gKi9cbiAgcHJpdmF0ZSBfeWVhcnMhOiBudW1iZXJbXVtdO1xuXG4gIC8qKiBHcmlkIG9mIGNhbGVuZGFyIGNlbGxzIHJlcHJlc2VudGluZyB5ZWFycyBmb3IgdGhlIHdpZGUgdmlldy4gKi9cbiAgcHJpdmF0ZSBfbmV4dE1vbnRoWWVhcnMhOiBudW1iZXJbXVtdO1xuXG4gIC8qKiBHcmlkIG9mIGNhbGVuZGFyIGNlbGxzIHJlcHJlc2VudGluZyB0aGUgZGF0ZXMgb2YgdGhlIG5leHQgbW9udGguICovXG4gIHByaXZhdGUgX25leHRNb250aFdlZWtzITogRGF5PFQ+W11bXTtcblxuICAvKiogQW4gYXJyYXkgY29udGFpbmluZyBhbGwgdGhlIG1vbnRoIG5hbWVzIGluIHRoZSBjdXJyZW50IGxhbmd1YWdlLiAqL1xuICBwcml2YXRlIF9tb250aE5hbWVzOiBzdHJpbmdbXSA9IHRoaXMuX2RhdGVBZGFwdGVyLmdldE1vbnRoTmFtZXMoJ2xvbmcnKTtcblxuICAvKiogQW4gYXJyYXkgY29udGFpbmluZyB0aGUgd2Vla3MnIG51bWJlcnMgZm9yIHRoZSBjdXJyZW50IG1vbnRoLiAqL1xuICBwcml2YXRlIF93ZWVrTnVtYmVycyE6IG51bWJlcltdO1xuXG4gIC8qKiBBbiBhcnJheSBjb250YWluaW5nIHRoZSB3ZWVrcycgbnVtYmVycyBmb3IgdGhlIG5leHQgbW9udGggaW4gd2lkZSBtb2RlLiAqL1xuICBwcml2YXRlIF9uZXh0TW9udGhXZWVrTnVtYmVycyE6IG51bWJlcltdO1xuXG4gIHByaXZhdGUgX2VuaGFuY2VkVmFyaWFudDogYm9vbGVhbiA9IGZhbHNlO1xuXG4gIC8qKiBBIGxpc3Qgb2YgYnV0dG9ucyBjb3JyZXNwb25kaW5nIHRvIGRheXMsIG1vbnRocyBvciB5ZWFycyBkZXBlbmRpbmcgb24gdGhlIHZpZXcuICovXG4gIHByaXZhdGUgZ2V0IF9jZWxscygpOiAoSFRNTEJ1dHRvbkVsZW1lbnQgfCBTYmJDYWxlbmRhckRheUVsZW1lbnQpW10ge1xuICAgIHJldHVybiBBcnJheS5mcm9tPEhUTUxCdXR0b25FbGVtZW50IHwgU2JiQ2FsZW5kYXJEYXlFbGVtZW50PihcbiAgICAgICh0aGlzLl9jYWxlbmRhclZpZXcgPT09ICdkYXknXG4gICAgICAgID8gdGhpcy5fZ2V0Um9vdEZvclF1ZXJ5U2VsZWN0b3IoKT8ucXVlcnlTZWxlY3RvckFsbCgnc2JiLWNhbGVuZGFyLWRheScpXG4gICAgICAgIDogdGhpcy5zaGFkb3dSb290Py5xdWVyeVNlbGVjdG9yQWxsKCcuc2JiLWNhbGVuZGFyX19jZWxsJykpID8/IFtdLFxuICAgICk7XG4gIH1cblxuICAvKiogVGhlIGNob3NlbiB5ZWFyIGluIHRoZSB5ZWFyIHNlbGVjdGlvbiB2aWV3LiAqL1xuICBwcml2YXRlIF9jaG9zZW5ZZWFyPzogbnVtYmVyO1xuXG4gIC8qKiBUaGUgY2hvc2VuIG1vbnRoIGluIHRoZSB5ZWFyIHNlbGVjdGlvbiB2aWV3LiAqL1xuICBwcml2YXRlIF9jaG9zZW5Nb250aD86IG51bWJlcjtcblxuICAvKiogV2hldGhlciB0aGUgZm9jdXMgc2hvdWxkIGJlIHJlc2V0IG9uIGZvY3VzQ2VsbC4gKi9cbiAgcHJpdmF0ZSBfcmVzZXRGb2N1cyA9IGZhbHNlO1xuXG4gIC8qKiBXaGV0aGVyIGFuIGVsZW1lbnQgaW5zaWRlIHRoZSBjYWxlbmRhciBpcyBjdXJyZW50bHkgZm9jdXNlZC4gKi9cbiAgcHJpdmF0ZSBfY29udGFpbmluZ0ZvY3VzID0gZmFsc2U7XG5cbiAgQHN0YXRlKClcbiAgcHJpdmF0ZSBhY2Nlc3NvciBfaW5pdGlhbGl6ZWQgPSBmYWxzZTtcblxuICBwcml2YXRlIF9sYW5ndWFnZSA9IG5ldyBTYmJMYW5ndWFnZUNvbnRyb2xsZXIodGhpcykud2l0aEhhbmRsZXIoKCkgPT4ge1xuICAgIHRoaXMuX21vbnRoTmFtZXMgPSB0aGlzLl9kYXRlQWRhcHRlci5nZXRNb250aE5hbWVzKCdsb25nJyk7XG4gICAgdGhpcy5fY3JlYXRlTW9udGhSb3dzKCk7XG4gIH0pO1xuICBwcml2YXRlIF9tZWRpYU1hdGNoZXIgPSBuZXcgU2JiTWVkaWFNYXRjaGVyQ29udHJvbGxlcih0aGlzLCB7XG4gICAgW1NiYk1lZGlhUXVlcnlCcmVha3BvaW50TGFyZ2VBbmRBYm92ZV06ICgpID0+IHRoaXMuX2luaXQoKSxcbiAgfSk7XG5cbiAgcHVibGljIGNvbnN0cnVjdG9yKCkge1xuICAgIHN1cGVyKCk7XG4gICAgdGhpcy5fY3JlYXRlTW9udGhSb3dzKCk7XG4gICAgdGhpcy5fc2V0V2Vla2RheXMoKTtcblxuICAgIC8vIFdlIG5lZWQgdG8gdHJhY2sgdGhlIGZvY3VzIGFzIHdlIHNob3VsZCBvbmx5IHRha2UgZm9jdXMgaW50byB0aGUgY2FsZW5kYXIsIHdoZW4gdGhlXG4gICAgLy8gZm9jdXMgd2FzIG9uY2Ugc2V0IGludG8gdGhlIGNhbGVuZGFyLlxuICAgIC8vIEZvciBzaGFkb3cgRE9NIGNvbXBhdGliaWxpdHkgd2UgbmVlZCB0byB0cmFjayB0aGlzIHByb2dyYW1tYXRpY2FsbHkuXG4gICAgdGhpcy5hZGRFdmVudExpc3RlbmVyKCdmb2N1c2luJywgKCkgPT4gKHRoaXMuX2NvbnRhaW5pbmdGb2N1cyA9IHRydWUpKTtcbiAgICB0aGlzLmFkZEV2ZW50TGlzdGVuZXIoJ2ZvY3Vzb3V0JywgKCkgPT4gKHRoaXMuX2NvbnRhaW5pbmdGb2N1cyA9IGZhbHNlKSk7XG4gICAgdGhpcy5hZGRFdmVudExpc3RlbmVyKCdjbGljaycsIChlKSA9PiB7XG4gICAgICBjb25zdCBkYXkgPSAoZS50YXJnZXQgYXMgSFRNTEVsZW1lbnQpLmNsb3Nlc3Q8U2JiQ2FsZW5kYXJEYXlFbGVtZW50PFQ+Pignc2JiLWNhbGVuZGFyLWRheScpO1xuICAgICAgaWYgKGRheSkge1xuICAgICAgICB0aGlzLl9zZWxlY3REYXRlKGRheS52YWx1ZSEpO1xuICAgICAgfVxuICAgIH0pO1xuICAgIHRoaXMuYWRkRXZlbnRMaXN0ZW5lcigna2V5ZG93bicsIChlKSA9PiB7XG4gICAgICBpZiAoKGUudGFyZ2V0IGFzIEhUTUxFbGVtZW50KS5sb2NhbE5hbWUgPT09ICdzYmItY2FsZW5kYXItZGF5Jykge1xuICAgICAgICB0aGlzLl9oYW5kbGVLZXlib2FyZEV2ZW50KFxuICAgICAgICAgIGUsXG4gICAgICAgICAgdGhpcy5fbWFwRGF0ZVRvRGF5KChlLnRhcmdldCBhcyBTYmJDYWxlbmRhckRheUVsZW1lbnQpLnZhbHVlISBhcyBUKSxcbiAgICAgICAgKTtcbiAgICAgIH1cbiAgICB9KTtcbiAgfVxuXG4gIHByaXZhdGUgX2dldFJvb3RGb3JRdWVyeVNlbGVjdG9yKCk6IHRoaXMgfCBTaGFkb3dSb290IHtcbiAgICByZXR1cm4gdGhpcy5fZW5oYW5jZWRWYXJpYW50ID8gdGhpcyA6IHRoaXMuc2hhZG93Um9vdCE7XG4gIH1cblxuICBwcml2YXRlIF9kYXRlRmlsdGVyKGRhdGU6IFQpOiBib29sZWFuIHtcbiAgICByZXR1cm4gdGhpcy5kYXRlRmlsdGVyPy4oZGF0ZSkgPz8gdHJ1ZTtcbiAgfVxuXG4gIC8qKiBSZXNldHMgdGhlIGFjdGl2ZSBtb250aCBhY2NvcmRpbmcgdG8gdGhlIG5ldyBzdGF0ZSBvZiB0aGUgY2FsZW5kYXIuICovXG4gIHB1YmxpYyByZXNldFBvc2l0aW9uKCk6IHZvaWQge1xuICAgIHRoaXMuX3Jlc2V0Q2FsZW5kYXJWaWV3KCk7XG4gICAgdGhpcy5faW5pdCgpO1xuICB9XG5cbiAgcHVibGljIG92ZXJyaWRlIGNvbm5lY3RlZENhbGxiYWNrKCk6IHZvaWQge1xuICAgIHN1cGVyLmNvbm5lY3RlZENhbGxiYWNrKCk7XG4gICAgdGhpcy5yZXNldFBvc2l0aW9uKCk7XG4gICAgdGhpcy5zaGFkb3dSb290Py5hZGRFdmVudExpc3RlbmVyKCdzbG90Y2hhbmdlJywgdGhpcy5fb25TbG90Q2hhbmdlLCB7IGNhcHR1cmU6IHRydWUgfSk7XG4gIH1cblxuICBwdWJsaWMgb3ZlcnJpZGUgZGlzY29ubmVjdGVkQ2FsbGJhY2soKTogdm9pZCB7XG4gICAgc3VwZXIuZGlzY29ubmVjdGVkQ2FsbGJhY2soKTtcbiAgICB0aGlzLnNoYWRvd1Jvb3Q/LnJlbW92ZUV2ZW50TGlzdGVuZXIoJ3Nsb3RjaGFuZ2UnLCB0aGlzLl9vblNsb3RDaGFuZ2UsIHsgY2FwdHVyZTogdHJ1ZSB9KTtcbiAgfVxuXG4gIC8qKiBAaW50ZXJuYWwgKi9cbiAgcHVibGljIG92ZXJyaWRlIGZvY3VzKCk6IHZvaWQge1xuICAgIHRoaXMuX3Jlc2V0Rm9jdXMgPSB0cnVlO1xuICAgIHRoaXMuX2ZvY3VzQ2VsbCgpO1xuICB9XG5cbiAgcHJvdGVjdGVkIG92ZXJyaWRlIHdpbGxVcGRhdGUoY2hhbmdlZFByb3BlcnRpZXM6IFByb3BlcnR5VmFsdWVzPHRoaXM+KTogdm9pZCB7XG4gICAgc3VwZXIud2lsbFVwZGF0ZShjaGFuZ2VkUHJvcGVydGllcyk7XG5cbiAgICBpZiAoIXRoaXMuX2luaXRpYWxpemVkKSB7XG4gICAgICByZXR1cm47XG4gICAgfVxuXG4gICAgaWYgKGNoYW5nZWRQcm9wZXJ0aWVzLmhhcygnd2lkZScpIHx8IGNoYW5nZWRQcm9wZXJ0aWVzLmhhcygnb3JpZW50YXRpb24nKSkge1xuICAgICAgdGhpcy5yZXNldFBvc2l0aW9uKCk7XG4gICAgfVxuXG4gICAgaWYgKGNoYW5nZWRQcm9wZXJ0aWVzLmhhcygndmlldycpKSB7XG4gICAgICB0aGlzLl9zZXRDaG9zZW5ZZWFyKCk7XG4gICAgICB0aGlzLl9jaG9zZW5Nb250aCA9IHVuZGVmaW5lZDtcbiAgICAgIHRoaXMuX25leHRDYWxlbmRhclZpZXcgPSB0aGlzLl9jYWxlbmRhclZpZXcgPSB0aGlzLnZpZXc7XG4gICAgfVxuICB9XG5cbiAgcHJvdGVjdGVkIG92ZXJyaWRlIHVwZGF0ZWQoY2hhbmdlZFByb3BlcnRpZXM6IFByb3BlcnR5VmFsdWVzPHRoaXM+KTogdm9pZCB7XG4gICAgc3VwZXIudXBkYXRlZChjaGFuZ2VkUHJvcGVydGllcyk7XG4gICAgLy8gVGhlIGNhbGVuZGFyIG5lZWRzIHRvIGNhbGN1bGF0ZSB0YWItaW5kZXhlcyBvbiBmaXJzdCByZW5kZXIsXG4gICAgLy8gYW5kIGV2ZXJ5IHRpbWUgYSBkYXRlIGlzIHNlbGVjdGVkIG9yIHRoZSBtb250aCB2aWV3IGNoYW5nZXMuXG4gICAgdGhpcy5fc2V0VGFiSW5kZXgoKTtcblxuICAgIC8vIFdoZW4gY2hhbmdpbmcgdmlldyB0byB5ZWFyL21vbnRoLCB0aGUgdGFiaW5kZXggaXMgY2hhbmdlZCwgYnV0IHRoZSBmb2N1c2VkIGVsZW1lbnQgaXMgZ2V0dGluZyBsb3N0LlxuICAgIC8vIFdlIG5lZWQgdG8gY2FsbCBgX2ZvY3VzQ2VsbCgpYCBtZXRob2QgZXhwbGljaXRseSB0byBjb3JyZWN0bHkgc2V0IHRoZSBmb2N1cy5cbiAgICB0aGlzLl9mb2N1c0NlbGwoKTtcbiAgfVxuXG4gIHByaXZhdGUgX29uU2xvdENoYW5nZSA9ICgpOiB2b2lkID0+IHtcbiAgICB0aGlzLl9lbmhhbmNlZFZhcmlhbnQgPSBBcnJheS5mcm9tKHRoaXMuY2hpbGRyZW4pLnNvbWUoXG4gICAgICAoYykgPT4gYy5sb2NhbE5hbWUgPT09ICdzYmItY2FsZW5kYXItZGF5JyxcbiAgICApO1xuICAgIHRoaXMudG9nZ2xlU3RhdGUoJ2VuaGFuY2VkJywgdGhpcy5fZW5oYW5jZWRWYXJpYW50KTtcbiAgICB0aGlzLl9zZXRUYWJJbmRleCgpO1xuICB9O1xuXG4gIC8qKlxuICAgKiBUaGUgYF9zZWxlY3RlZGAgc3RhdGUgc2hvdWxkIGJlIGFkYXB0ZWQgd2hlbiB0aGUgYG11bHRpcGxlYCBwcm9wZXJ0eSBjaGFuZ2VzOlxuICAgKiAgIC0gaWYgaXQgY2hhbmdlcyB0byB0cnVlLCB0aGUgJ19zZWxlY3RlZCcgaXMgc2V0IHRvIGFuIGFycmF5O1xuICAgKiAgIC0gaWYgaXQgY2hhbmdlcyB0byBmYWxzZSwgdGhlIGZpcnN0IGF2YWlsYWJsZSBvcHRpb24gaXMgc2V0IGFzICd2YWx1ZScgb3RoZXJ3aXNlIGl0J3Mgc2V0IHRvIG51bGwuXG4gICAqL1xuICBwcml2YXRlIF9vbk11bHRpcGxlQ2hhbmdlZChpc011bHRpcGxlOiBib29sZWFuKTogdm9pZCB7XG4gICAgaWYgKGlzTXVsdGlwbGUgJiYgIUFycmF5LmlzQXJyYXkodGhpcy5fc2VsZWN0ZWQpKSB7XG4gICAgICB0aGlzLl9zZWxlY3RlZCA9IHRoaXMuX3NlbGVjdGVkID8gW3RoaXMuX3NlbGVjdGVkIGFzIFRdIDogW107XG4gICAgfVxuICAgIGlmICghaXNNdWx0aXBsZSAmJiBBcnJheS5pc0FycmF5KHRoaXMuX3NlbGVjdGVkKSkge1xuICAgICAgdGhpcy5fc2VsZWN0ZWQgPSAodGhpcy5fc2VsZWN0ZWQgYXMgVFtdKS5sZW5ndGggPyAodGhpcy5fc2VsZWN0ZWQgYXMgVFtdKVswXSA6IG51bGw7XG4gICAgfVxuICB9XG5cbiAgLyoqIEluaXRpYWxpemVzIHRoZSBjb21wb25lbnQuICovXG4gIHByaXZhdGUgX2luaXQoYWN0aXZlRGF0ZT86IFQpOiB2b2lkIHtcbiAgICAvLyBEdWUgdG8gaXRzIGNvbXBsZXhpdHksIHRoZSBjYWxlbmRhciBpcyBvbmx5IGluaXRpYWxpemVkIG9uIGNsaWVudCBzaWRlXG4gICAgaWYgKGlzU2VydmVyKSB7XG4gICAgICByZXR1cm47XG4gICAgfSBlbHNlIGlmICh0aGlzLmh5ZHJhdGlvblJlcXVpcmVkKSB7XG4gICAgICB0aGlzLmh5ZHJhdGlvbkNvbXBsZXRlLnRoZW4oKCkgPT4gdGhpcy5faW5pdCgpKTtcbiAgICAgIHJldHVybjtcbiAgICB9XG5cbiAgICBpZiAoYWN0aXZlRGF0ZSkge1xuICAgICAgdGhpcy5fYXNzaWduQWN0aXZlRGF0ZShhY3RpdmVEYXRlKTtcbiAgICB9XG4gICAgdGhpcy5fd2lkZSA9XG4gICAgICAodGhpcy5fbWVkaWFNYXRjaGVyLm1hdGNoZXMoU2JiTWVkaWFRdWVyeUJyZWFrcG9pbnRMYXJnZUFuZEFib3ZlKSA/PyBmYWxzZSkgJiYgdGhpcy53aWRlO1xuICAgIHRoaXMuX3dlZWtzID0gdGhpcy5fY3JlYXRlV2Vla1Jvd3ModGhpcy5fYWN0aXZlRGF0ZSk7XG4gICAgdGhpcy5feWVhcnMgPSB0aGlzLl9jcmVhdGVZZWFyUm93cygpO1xuICAgIHRoaXMuX3dlZWtOdW1iZXJzID0gdGhpcy5fY3JlYXRlV2Vla051bWJlcnModGhpcy5fYWN0aXZlRGF0ZSk7XG4gICAgdGhpcy5fbmV4dE1vbnRoV2Vla3MgPSBbW11dO1xuICAgIHRoaXMuX25leHRNb250aFllYXJzID0gW1tdXTtcbiAgICBpZiAodGhpcy5fd2lkZSkge1xuICAgICAgY29uc3QgbmV4dE1vbnRoRGF0ZSA9IHRoaXMuX2RhdGVBZGFwdGVyLmFkZENhbGVuZGFyTW9udGhzKHRoaXMuX2FjdGl2ZURhdGUsIDEpO1xuICAgICAgdGhpcy5fbmV4dE1vbnRoV2Vla3MgPSB0aGlzLl9jcmVhdGVXZWVrUm93cyhuZXh0TW9udGhEYXRlLCB0cnVlKTtcbiAgICAgIHRoaXMuX25leHRNb250aFllYXJzID0gdGhpcy5fY3JlYXRlWWVhclJvd3MoWUVBUlNfUEVSX1BBR0UpO1xuICAgICAgdGhpcy5fbmV4dE1vbnRoV2Vla051bWJlcnMgPSB0aGlzLl9jcmVhdGVXZWVrTnVtYmVycyhuZXh0TW9udGhEYXRlKTtcbiAgICB9XG4gICAgdGhpcy5faW5pdGlhbGl6ZWQgPSB0cnVlO1xuICB9XG5cbiAgLyoqIEZvY3VzZXMgb24gYSBkYXkgY2VsbCBwcmlvcml0aXppbmcgdGhlIHNlbGVjdGVkIGRheSwgdGhlIGN1cnJlbnQgZGF5LCBhbmQgbGFzdGx5LCB0aGUgZmlyc3Qgc2VsZWN0YWJsZSBkYXkuICovXG4gIHByaXZhdGUgX2ZvY3VzQ2VsbCgpOiB2b2lkIHtcbiAgICBpZiAodGhpcy5fcmVzZXRGb2N1cykge1xuICAgICAgdGhpcy5fZ2V0Rmlyc3RGb2N1c2FibGUoKT8uZm9jdXMoKTtcbiAgICAgIHRoaXMuX3Jlc2V0Rm9jdXMgPSBmYWxzZTtcbiAgICB9XG4gIH1cblxuICAvKiogQ3JlYXRlcyB0aGUgYXJyYXkgb2Ygd2Vla2RheXMuICovXG4gIHByaXZhdGUgX3NldFdlZWtkYXlzKCk6IHZvaWQge1xuICAgIGNvbnN0IG5hcnJvd1dlZWtkYXlzOiBzdHJpbmdbXSA9IHRoaXMuX2RhdGVBZGFwdGVyLmdldERheU9mV2Vla05hbWVzKCduYXJyb3cnKTtcbiAgICBjb25zdCBsb25nV2Vla2RheXM6IHN0cmluZ1tdID0gdGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0RGF5T2ZXZWVrTmFtZXMoJ2xvbmcnKTtcbiAgICBjb25zdCB3ZWVrZGF5czogV2Vla2RheVtdID0gbG9uZ1dlZWtkYXlzLm1hcCgobG9uZzogc3RyaW5nLCBpOiBudW1iZXIpID0+ICh7XG4gICAgICBsb25nLFxuICAgICAgbmFycm93OiBuYXJyb3dXZWVrZGF5c1tpXSxcbiAgICB9KSk7XG5cbiAgICAvLyBSb3RhdGVzIHRoZSBsYWJlbHMgZm9yIGRheXMgb2YgdGhlIHdlZWsgYmFzZWQgb24gdGhlIGNvbmZpZ3VyZWQgZmlyc3QgZGF5IG9mIHRoZSB3ZWVrLlxuICAgIGNvbnN0IGZpcnN0RGF5T2ZXZWVrOiBudW1iZXIgPSB0aGlzLl9kYXRlQWRhcHRlci5nZXRGaXJzdERheU9mV2VlaygpO1xuICAgIHRoaXMuX3dlZWtkYXlzID0gd2Vla2RheXMuc2xpY2UoZmlyc3REYXlPZldlZWspLmNvbmNhdCh3ZWVrZGF5cy5zbGljZSgwLCBmaXJzdERheU9mV2VlaykpO1xuICB9XG5cbiAgLyoqXG4gICAqIEdpdmVuIGEgZGF0ZSwgaXQgcmV0dXJucyB0aGUgd2VlayBudW1iZXJzIGZvciB0aGUgbW9udGggdGhlIGRhdGUgYmVsb25ncyB0by5cbiAgICogVE9ETzogY2hlY2sgaWYgZGF0ZS1mbnMgY2FuIGJlIHJlcGxhY2VkIHdpdGggY3VzdG9tIGxvZ2ljLlxuICAgKlxuICAgKiBTaW5jZSB0aGUgY2FsY3VsYXRpb24gaXMgbm90IHNpbXBsZSAoc2VlIGh0dHBzOi8vZW4ud2lraXBlZGlhLm9yZy93aWtpL1dlZWsjTnVtYmVyaW5nKSxcbiAgICogdGhlIGRhdGUtZm5zIGxpYnJhcnkgaGFzIGJlZW4gdXNlZCB0aGlzIHdheTpcbiAgICogdGhlIGZpcnN0IGFuZCB0aGUgbGFzdCBkYXkgb2YgdGhlIG1vbnRoIGFyZSBjYWxjdWxhdGVkIGFuZCB0aGVuIHBhc3NlZCB0byB0aGUgYGVhY2hXZWVrT2ZJbnRlcnZhbGAgZnVuY3Rpb24sXG4gICAqIHdoaWNoIHJldHVybnMgYW4gYXJyYXkgY29udGFpbmluZyB0aGUgc3RhcnRpbmcgZGF5IG9mIGV2ZXJ5IElTTyB3ZWVrIG9mIHRoZSBtb250aCxcbiAgICogY29uc2lkZXJpbmcgTW9uZGF5IGFzIHRoZSBmaXJzdCBkYXkuXG4gICAqIFRoZW4sIHRoaXMgYXJyYXkgaXMgbWFwcGVkIHZpYSB0aGUgYGdldFdlZWtgIGZ1bmN0aW9uLCB3aGljaCByZXR1cm5zIHRoZSBJU08gd2VlayBudW1iZXIgZm9yIHRoYXQgZGF0ZS5cbiAgICovXG4gIHByaXZhdGUgX2NyZWF0ZVdlZWtOdW1iZXJzKGRhdGU6IFQpOiBudW1iZXJbXSB7XG4gICAgcmV0dXJuIGVhY2hXZWVrT2ZJbnRlcnZhbChcbiAgICAgIHsgc3RhcnQ6IHN0YXJ0T2ZNb250aChkYXRlIGFzIERhdGUpLCBlbmQ6IGVuZE9mTW9udGgoZGF0ZSBhcyBEYXRlKSB9LFxuICAgICAgeyB3ZWVrU3RhcnRzT246IDEgfSxcbiAgICApLm1hcCgoZmlyc3REYXlPZldlZWs6IERhdGUpID0+XG4gICAgICBnZXRXZWVrKGZpcnN0RGF5T2ZXZWVrLCB7IHdlZWtTdGFydHNPbjogMSwgZmlyc3RXZWVrQ29udGFpbnNEYXRlOiA0IH0pLFxuICAgICk7XG4gIH1cblxuICAvKiogQ3JlYXRlcyB0aGUgcm93cyBhbG9uZyB0aGUgaG9yaXpvbnRhbCBkaXJlY3Rpb24gYW5kIHNldHMgdGhlIHBhcmFtZXRlcnMgdXNlZCBpbiBrZXlib2FyZCBuYXZpZ2F0aW9uLiAqL1xuICBwcml2YXRlIF9jcmVhdGVXZWVrUm93cyh2YWx1ZTogVCwgaXNTZWNvbmRNb250aEluVmlldyA9IGZhbHNlKTogRGF5PFQ+W11bXSB7XG4gICAgY29uc3QgZGF5c0luTW9udGg6IG51bWJlciA9IHRoaXMuX2RhdGVBZGFwdGVyLmdldE51bURheXNJbk1vbnRoKHZhbHVlKTtcbiAgICBjb25zdCB3ZWVrT2Zmc2V0OiBudW1iZXIgPSB0aGlzLl9kYXRlQWRhcHRlci5nZXRGaXJzdFdlZWtPZmZzZXQodmFsdWUpO1xuICAgIGlmICghaXNTZWNvbmRNb250aEluVmlldykge1xuICAgICAgdGhpcy5fa2V5Ym9hcmROYXZpZ2F0aW9uRGF5Vmlld1BhcmFtZXRlcnMuZmlyc3RNb250aExlbmd0aCA9IGRheXNJbk1vbnRoO1xuICAgICAgdGhpcy5fa2V5Ym9hcmROYXZpZ2F0aW9uRGF5Vmlld1BhcmFtZXRlcnMuZmlyc3RNb250aE9mZnNldCA9IHdlZWtPZmZzZXQ7XG4gICAgICB0aGlzLl9rZXlib2FyZE5hdmlnYXRpb25EYXlWaWV3UGFyYW1ldGVycy5maXJzdERheUluVmlldyA9IHRoaXMuX2RhdGVBZGFwdGVyLnRvSXNvODYwMShcbiAgICAgICAgdGhpcy5fZGF0ZUFkYXB0ZXIuY3JlYXRlRGF0ZShcbiAgICAgICAgICB0aGlzLl9kYXRlQWRhcHRlci5nZXRZZWFyKHZhbHVlKSxcbiAgICAgICAgICB0aGlzLl9kYXRlQWRhcHRlci5nZXRNb250aCh2YWx1ZSksXG4gICAgICAgICAgMSxcbiAgICAgICAgKSxcbiAgICAgICk7XG4gICAgICB0aGlzLl9rZXlib2FyZE5hdmlnYXRpb25EYXlWaWV3UGFyYW1ldGVycy5sYXN0RGF5SW5WaWV3ID0gdGhpcy5fZGF0ZUFkYXB0ZXIudG9Jc284NjAxKFxuICAgICAgICB0aGlzLl9kYXRlQWRhcHRlci5jcmVhdGVEYXRlKFxuICAgICAgICAgIHRoaXMuX2RhdGVBZGFwdGVyLmdldFllYXIodmFsdWUpLFxuICAgICAgICAgIHRoaXMuX2RhdGVBZGFwdGVyLmdldE1vbnRoKHZhbHVlKSxcbiAgICAgICAgICBkYXlzSW5Nb250aCxcbiAgICAgICAgKSxcbiAgICAgICk7XG4gICAgfSBlbHNlIHtcbiAgICAgIHRoaXMuX2tleWJvYXJkTmF2aWdhdGlvbkRheVZpZXdQYXJhbWV0ZXJzLnNlY29uZE1vbnRoT2Zmc2V0ID0gd2Vla09mZnNldDtcbiAgICAgIHRoaXMuX2tleWJvYXJkTmF2aWdhdGlvbkRheVZpZXdQYXJhbWV0ZXJzLmxhc3REYXlJblZpZXcgPSB0aGlzLl9kYXRlQWRhcHRlci50b0lzbzg2MDEoXG4gICAgICAgIHRoaXMuX2RhdGVBZGFwdGVyLmNyZWF0ZURhdGUoXG4gICAgICAgICAgdGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0WWVhcih2YWx1ZSksXG4gICAgICAgICAgdGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0TW9udGgodmFsdWUpLFxuICAgICAgICAgIGRheXNJbk1vbnRoLFxuICAgICAgICApLFxuICAgICAgKTtcbiAgICB9XG4gICAgcmV0dXJuIHRoaXMub3JpZW50YXRpb24gPT09ICdob3Jpem9udGFsJ1xuICAgICAgPyB0aGlzLl9jcmVhdGVXZWVrUm93c0hvcml6b250YWwodmFsdWUsIGRheXNJbk1vbnRoLCB3ZWVrT2Zmc2V0KVxuICAgICAgOiB0aGlzLl9jcmVhdGVXZWVrUm93c1ZlcnRpY2FsKHZhbHVlLCBkYXlzSW5Nb250aCwgd2Vla09mZnNldCk7XG4gIH1cblxuICAvKipcbiAgICogQ3JlYXRlcyB0aGUgcm93cyBmb3IgZWFjaCB3ZWVrIGluIG9yaWVudGF0aW9uPSdob3Jpem9udGFsJy5cbiAgICpcbiAgICogSXRlcmF0ZXMgdGhyb3VnaCB0aGUgZGF5cyBvZiB0aGUgbW9udGhzLCBjcmVhdGVzIGEgRGF5IG9iamVjdCBmb3IgZWFjaCBhbmQgcHVzaGVzIGl0IGludG8gYW5kIGFycmF5LlxuICAgKiBFYWNoIHNldmVuIGRheXMgKGNvbnNpZGVyaW5nIHRoZSBvZmZzZXQgYXQgdGhlIGJlZ2lubmluZyBvZiB0aGUgbW9udGgpIHJlc3RhcnRzIGZyb20gYW4gZW1wdHkgYXJyYXkuXG4gICAqXG4gICAqIFRoZSByZXN1bHQgaXMgYSBtYXRyaXggaW4gd2hpY2ggZXZlcnkgcm93IGlzIGEgd2VlayAob3IgcGFydCBvZiBpdCwgY29uc2lkZXJpbmcgb2Zmc2V0KS5cbiAgICovXG4gIHByaXZhdGUgX2NyZWF0ZVdlZWtSb3dzSG9yaXpvbnRhbCh2YWx1ZTogVCwgZGF5c0luTW9udGg6IG51bWJlciwgd2Vla09mZnNldDogbnVtYmVyKTogRGF5PFQ+W11bXSB7XG4gICAgY29uc3Qgd2Vla3M6IERheTxUPltdW10gPSBbW11dO1xuICAgIGZvciAobGV0IGkgPSAwLCBjZWxsID0gd2Vla09mZnNldDsgaSA8IGRheXNJbk1vbnRoOyBpKyssIGNlbGwrKykge1xuICAgICAgaWYgKGNlbGwgPT09IERBWVNfUEVSX1JPVykge1xuICAgICAgICB3ZWVrcy5wdXNoKFtdKTtcbiAgICAgICAgY2VsbCA9IDA7XG4gICAgICB9XG4gICAgICBjb25zdCBkYXRlID0gdGhpcy5fZGF0ZUFkYXB0ZXIuY3JlYXRlRGF0ZShcbiAgICAgICAgdGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0WWVhcih2YWx1ZSksXG4gICAgICAgIHRoaXMuX2RhdGVBZGFwdGVyLmdldE1vbnRoKHZhbHVlKSxcbiAgICAgICAgaSArIDEsXG4gICAgICApITtcbiAgICAgIHdlZWtzW3dlZWtzLmxlbmd0aCAtIDFdLnB1c2godGhpcy5fbWFwRGF0ZVRvRGF5KGRhdGUpKTtcbiAgICB9XG4gICAgcmV0dXJuIHdlZWtzO1xuICB9XG5cbiAgLyoqXG4gICAqIENyZWF0ZXMgdGhlIHJvd3MgZm9yIGVhY2ggd2VlayBpbiBvcmllbnRhdGlvbj0ndmVydGljYWwnLlxuICAgKlxuICAgKiBDcmVhdGVzIGEgbWF0cml4IHdpdGggc2V2ZW4gZW1wdHkgcm93cy5cbiAgICogSXRlcmF0ZXMgdGhyb3VnaCB0aGUgZGF5cyBvZiB0aGUgbW9udGhzLCBjcmVhdGVzIGEgRGF5IG9iamVjdCBmb3IgZWFjaFxuICAgKiBhbmQgcHVzaGVzIGl0IGludG8gdGhlIGNvcnJlY3QgYXJyYXkgY29uc2lkZXJpbmcgdGhlIG9mZnNldCBhdCB0aGUgYmVnaW5uaW5nIG9mIHRoZSBtb250aC5cbiAgICogRWFjaCBzZXZlbiBkYXlzIChpbmNsdWRpbmcgb2Zmc2V0KSByZXN0YXJ0cyBmcm9tIHRoZSBmaXJzdC5cbiAgICpcbiAgICogVGhlIHJlc3VsdCBpcyBhIG1hdHJpeCBpbiB3aGljaCBldmVyeSByb3cgaXMgYSBzZXQgb2Ygd2Vla2RheXMsIHNvOlxuICAgKiAtIHJvdyAwOiBhbGwgdGhlIE1vbmRheXM7XG4gICAqIC0gcm93IDE6IGFsbCB0aGUgVHVlc2RheXM7XG4gICAqIC0gLi4uXG4gICAqIC0gcm93IDc6IGFsbCB0aGUgU3VuZGF5cy5cbiAgICovXG4gIHByaXZhdGUgX2NyZWF0ZVdlZWtSb3dzVmVydGljYWwodmFsdWU6IFQsIGRheXNJbk1vbnRoOiBudW1iZXIsIHdlZWtPZmZzZXQ6IG51bWJlcik6IERheTxUPltdW10ge1xuICAgIGNvbnN0IHdlZWtzOiBEYXk8VD5bXVtdID0gQXJyYXkuZnJvbSh7IGxlbmd0aDogREFZU19QRVJfUk9XIH0sICgpID0+IFtdKTtcbiAgICBmb3IgKGxldCBpID0gMCwgY2VsbCA9IHdlZWtPZmZzZXQ7IGkgPCBkYXlzSW5Nb250aDsgaSsrLCBjZWxsKyspIHtcbiAgICAgIGlmIChjZWxsID09PSBEQVlTX1BFUl9ST1cpIHtcbiAgICAgICAgY2VsbCA9IDA7XG4gICAgICB9XG4gICAgICBjb25zdCBkYXRlID0gdGhpcy5fZGF0ZUFkYXB0ZXIuY3JlYXRlRGF0ZShcbiAgICAgICAgdGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0WWVhcih2YWx1ZSksXG4gICAgICAgIHRoaXMuX2RhdGVBZGFwdGVyLmdldE1vbnRoKHZhbHVlKSxcbiAgICAgICAgaSArIDEsXG4gICAgICApITtcbiAgICAgIHdlZWtzW2NlbGxdLnB1c2godGhpcy5fbWFwRGF0ZVRvRGF5KGRhdGUpKTtcbiAgICB9XG4gICAgcmV0dXJuIHdlZWtzO1xuICB9XG5cbiAgcHJpdmF0ZSBfbWFwRGF0ZVRvRGF5KGRhdGU6IFQpOiBEYXk8VD4ge1xuICAgIGNvbnN0IGlzb0RhdGUgPSB0aGlzLl9kYXRlQWRhcHRlci50b0lzbzg2MDEoZGF0ZSk7XG4gICAgcmV0dXJuIHtcbiAgICAgIHZhbHVlOiBpc29EYXRlLFxuICAgICAgZGF0ZVZhbHVlOiBkYXRlLFxuICAgICAgZGF5VmFsdWU6IFN0cmluZyh0aGlzLl9kYXRlQWRhcHRlci5nZXREYXRlKGRhdGUpKSxcbiAgICAgIG1vbnRoVmFsdWU6IFN0cmluZyh0aGlzLl9kYXRlQWRhcHRlci5nZXRNb250aChkYXRlKSksXG4gICAgICB5ZWFyVmFsdWU6IFN0cmluZyh0aGlzLl9kYXRlQWRhcHRlci5nZXRZZWFyKGRhdGUpKSxcbiAgICAgIHdlZWtWYWx1ZTogZ2V0V2Vlayhpc29EYXRlLCB7IHdlZWtTdGFydHNPbjogMSwgZmlyc3RXZWVrQ29udGFpbnNEYXRlOiA0IH0pLFxuICAgICAgd2Vla0RheVZhbHVlOiB0aGlzLl9kYXRlQWRhcHRlci5nZXREYXlPZldlZWsoZGF0ZSksXG4gICAgfTtcbiAgfVxuXG4gIC8qKiBGb3JjZSB0aGUgY29udmVyc2lvbiB0byBJU084NjAxIGZvcm1hdHRlZCBzdHJpbmcuICovXG4gIHByaXZhdGUgX21hcFZhbHVlVG9JU09EYXRlKHZhbHVlOiBzdHJpbmcgfCBEYXRlKTogc3RyaW5nIHtcbiAgICByZXR1cm4gdHlwZW9mIHZhbHVlID09PSAnc3RyaW5nJyA/IHZhbHVlIDogdGhpcy5fZGF0ZUFkYXB0ZXIudG9Jc284NjAxKHZhbHVlIGFzIFQpO1xuICB9XG5cbiAgLyoqIENyZWF0ZXMgdGhlIHJvd3MgZm9yIHRoZSBtb250aCBzZWxlY3Rpb24gdmlldy4gKi9cbiAgcHJpdmF0ZSBfY3JlYXRlTW9udGhSb3dzKCk6IHZvaWQge1xuICAgIGNvbnN0IHNob3J0TmFtZXM6IHN0cmluZ1tdID0gdGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0TW9udGhOYW1lcygnc2hvcnQnKTtcbiAgICBjb25zdCBtb250aHM6IE1vbnRoW10gPSBuZXcgQXJyYXkoMTIpLmZpbGwobnVsbCkubWFwKFxuICAgICAgKF8sIGk6IG51bWJlcik6IE1vbnRoID0+ICh7XG4gICAgICAgIHZhbHVlOiBzaG9ydE5hbWVzW2ldLFxuICAgICAgICBsb25nVmFsdWU6IHRoaXMuX21vbnRoTmFtZXNbaV0sXG4gICAgICAgIG1vbnRoVmFsdWU6IGkgKyAxLFxuICAgICAgfSksXG4gICAgKTtcbiAgICBjb25zdCByb3dzOiBudW1iZXIgPSAxMiAvIE1PTlRIU19QRVJfUk9XO1xuICAgIGNvbnN0IG1vbnRoQXJyYXk6IE1vbnRoW11bXSA9IFtdO1xuICAgIGZvciAobGV0IGk6IG51bWJlciA9IDA7IGkgPCByb3dzOyBpKyspIHtcbiAgICAgIG1vbnRoQXJyYXkucHVzaChtb250aHMuc2xpY2UoTU9OVEhTX1BFUl9ST1cgKiBpLCBNT05USFNfUEVSX1JPVyAqIChpICsgMSkpKTtcbiAgICB9XG4gICAgdGhpcy5fbW9udGhzID0gbW9udGhBcnJheTtcbiAgfVxuXG4gIC8qKiBDcmVhdGVzIHRoZSByb3dzIGZvciB0aGUgeWVhciBzZWxlY3Rpb24gdmlldy4gKi9cbiAgcHJpdmF0ZSBfY3JlYXRlWWVhclJvd3Mob2Zmc2V0OiBudW1iZXIgPSAwKTogbnVtYmVyW11bXSB7XG4gICAgY29uc3Qgc3RhcnRWYWx1ZVllYXJWaWV3OiBudW1iZXIgPSB0aGlzLl9nZXRTdGFydFZhbHVlWWVhclZpZXcoKTtcbiAgICBjb25zdCBhbGxZZWFyczogbnVtYmVyW10gPSBuZXcgQXJyYXkoWUVBUlNfUEVSX1BBR0UpXG4gICAgICAuZmlsbCgwKVxuICAgICAgLm1hcCgoXywgaTogbnVtYmVyKSA9PiBzdGFydFZhbHVlWWVhclZpZXcgKyBvZmZzZXQgKyBpKTtcbiAgICBjb25zdCByb3dzOiBudW1iZXIgPSBZRUFSU19QRVJfUEFHRSAvIFlFQVJTX1BFUl9ST1c7XG4gICAgY29uc3QgeWVhckFycmF5OiBudW1iZXJbXVtdID0gW107XG4gICAgZm9yIChsZXQgaTogbnVtYmVyID0gMDsgaSA8IHJvd3M7IGkrKykge1xuICAgICAgeWVhckFycmF5LnB1c2goYWxsWWVhcnMuc2xpY2UoWUVBUlNfUEVSX1JPVyAqIGksIFlFQVJTX1BFUl9ST1cgKiAoaSArIDEpKSk7XG4gICAgfVxuICAgIHJldHVybiB5ZWFyQXJyYXk7XG4gIH1cblxuICAvKipcbiAgICogQ2FsY3VsYXRlcyB0aGUgZmlyc3QgeWVhciB0aGF0IHdpbGwgYmUgc2hvd24gaW4gdGhlIHllYXIgc2VsZWN0aW9uIHBhbmVsLlxuICAgKiBJZiBgbWluRGF0ZWAgYW5kIGBtYXhEYXRlYCBhcmUgYm90aCBudWxsLCB0aGUgc3RhcnRpbmcgeWVhciBpcyBjYWxjdWxhdGVkIGFzXG4gICAqIHRoZSBtdWx0aXBsZSBvZiBZRUFSU19QRVJfUEFHRSBjbG9zZXN0IHRvIGFuZCBsZXNzIHRoYW4gYWN0aXZlRGF0ZSxcbiAgICogZS5nLiwgd2l0aCBgWUVBUlNfUEVSX1BBR0VgID0gMjQgYW5kIGBhY3RpdmVEYXRlYCA9IDIwMjAsIHRoZSBmdW5jdGlvbiB3aWxsIHJldHVybiAyMDE2ICgyNCAqIDgzKSxcbiAgICogd2hpbGUgd2l0aCBgYWN0aXZlRGF0ZWAgPSAyMDAwLCB0aGUgZnVuY3Rpb24gd2lsbCByZXR1cm4gMTk5MiAoMjQgKiA4MikuXG4gICAqIElmIGBtaW5EYXRlYCBpcyBub3QgbnVsbCwgaXQgcmV0dXJucyB0aGUgY29ycmVzcG9uZGluZyB5ZWFyOyBpZiBgbWF4RGF0ZWAgaXMgbm90IG51bGwsXG4gICAqIGl0IHJldHVybnMgdGhlIGNvcnJlc3BvbmRpbmcgeWVhciBtaW51cyBgWUVBUlNfUEVSX1BBR0VgLCBzbyB0aGF0IHRoZSBgbWF4RGF0ZWAgaXMgdGhlIGxhc3QgcmVuZGVyZWQgeWVhci5cbiAgICogSWYgYm90aCBhcmUgbm90IG51bGwsIGBtYXhEYXRlYCBoYXMgcHJpb3JpdHkgb3ZlciBgbWluRGF0ZWAuXG4gICAqL1xuICBwcml2YXRlIF9nZXRTdGFydFZhbHVlWWVhclZpZXcoKTogbnVtYmVyIHtcbiAgICBsZXQgc3RhcnRpbmdZZWFyID0gMDtcbiAgICBpZiAodGhpcy5tYXgpIHtcbiAgICAgIHN0YXJ0aW5nWWVhciA9IHRoaXMuX2RhdGVBZGFwdGVyLmdldFllYXIodGhpcy5tYXgpIC0gWUVBUlNfUEVSX1BBR0UgKyAxO1xuICAgIH0gZWxzZSBpZiAodGhpcy5taW4pIHtcbiAgICAgIHN0YXJ0aW5nWWVhciA9IHRoaXMuX2RhdGVBZGFwdGVyLmdldFllYXIodGhpcy5taW4pO1xuICAgIH1cbiAgICBjb25zdCBhY3RpdmVZZWFyID0gdGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0WWVhcih0aGlzLl9hY3RpdmVEYXRlKTtcbiAgICByZXR1cm4gKFxuICAgICAgYWN0aXZlWWVhciAtXG4gICAgICAoKCgoYWN0aXZlWWVhciAtIHN0YXJ0aW5nWWVhcikgJSBZRUFSU19QRVJfUEFHRSkgKyBZRUFSU19QRVJfUEFHRSkgJSBZRUFSU19QRVJfUEFHRSlcbiAgICApO1xuICB9XG5cbiAgLyoqIENoZWNrcyBpZiBkYXRlIGlzIHdpdGhpbiB0aGUgbWluLW1heCByYW5nZS4gKi9cbiAgcHJpdmF0ZSBfaXNEYXlJblJhbmdlKGRhdGVTdHJpbmc6IHN0cmluZyk6IGJvb2xlYW4ge1xuICAgIGlmICghdGhpcy5taW4gJiYgIXRoaXMubWF4KSB7XG4gICAgICByZXR1cm4gdHJ1ZTtcbiAgICB9XG4gICAgY29uc3QgZGF0ZSA9IHRoaXMuX2RhdGVBZGFwdGVyLmRlc2VyaWFsaXplKGRhdGVTdHJpbmcpITtcbiAgICByZXR1cm4gdGhpcy5fZGF0ZUFkYXB0ZXIuc2FtZURhdGUoZGF0ZSwgdGhpcy5fZGF0ZUFkYXB0ZXIuY2xhbXBEYXRlKGRhdGUsIHRoaXMubWluLCB0aGlzLm1heCkpO1xuICB9XG5cbiAgLyoqIENoZWNrcyBpZiBkYXRlIGlzIHdpdGhpbiB0aGUgbWluLW1heCByYW5nZSBpbiBtb250aCB2aWV3LiAqL1xuICBwcml2YXRlIF9pc01vbnRoSW5SYW5nZShtb250aDogbnVtYmVyLCB5ZWFyOiBudW1iZXIpOiBib29sZWFuIHtcbiAgICBpZiAoIXRoaXMubWluICYmICF0aGlzLm1heCkge1xuICAgICAgcmV0dXJuIHRydWU7XG4gICAgfVxuXG4gICAgY29uc3QgaXNCZWZvcmVNaW46IGJvb2xlYW4gPVxuICAgICAgdGhpcy5fZGF0ZUFkYXB0ZXIuaXNWYWxpZCh0aGlzLm1pbikgJiZcbiAgICAgICh5ZWFyIDwgdGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0WWVhcih0aGlzLm1pbiEpIHx8XG4gICAgICAgICh5ZWFyID09PSB0aGlzLl9kYXRlQWRhcHRlci5nZXRZZWFyKHRoaXMubWluISkgJiZcbiAgICAgICAgICBtb250aCA8IHRoaXMuX2RhdGVBZGFwdGVyLmdldE1vbnRoKHRoaXMubWluISkpKTtcblxuICAgIGNvbnN0IGlzQWZ0ZXJNYXg6IGJvb2xlYW4gPVxuICAgICAgdGhpcy5fZGF0ZUFkYXB0ZXIuaXNWYWxpZCh0aGlzLm1heCkgJiZcbiAgICAgICh5ZWFyID4gdGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0WWVhcih0aGlzLm1heCEpIHx8XG4gICAgICAgICh5ZWFyID09PSB0aGlzLl9kYXRlQWRhcHRlci5nZXRZZWFyKHRoaXMubWF4ISkgJiZcbiAgICAgICAgICBtb250aCA+IHRoaXMuX2RhdGVBZGFwdGVyLmdldE1vbnRoKHRoaXMubWF4ISkpKTtcblxuICAgIHJldHVybiAhKGlzQmVmb3JlTWluIHx8IGlzQWZ0ZXJNYXgpO1xuICB9XG5cbiAgLyoqIENoZWNrcyBpZiBkYXRlIGlzIHdpdGhpbiB0aGUgbWluLW1heCByYW5nZSBpbiB5ZWFyIHZpZXcuICovXG4gIHByaXZhdGUgX2lzWWVhckluUmFuZ2UoeWVhcjogbnVtYmVyKTogYm9vbGVhbiB7XG4gICAgaWYgKCF0aGlzLm1pbiAmJiAhdGhpcy5tYXgpIHtcbiAgICAgIHJldHVybiB0cnVlO1xuICAgIH1cbiAgICBjb25zdCBpc0JlZm9yZU1pbjogYm9vbGVhbiA9XG4gICAgICB0aGlzLl9kYXRlQWRhcHRlci5pc1ZhbGlkKHRoaXMubWluKSAmJiB0aGlzLl9kYXRlQWRhcHRlci5nZXRZZWFyKHRoaXMubWluISkgPiB5ZWFyO1xuICAgIGNvbnN0IGlzQWZ0ZXJNYXg6IGJvb2xlYW4gPVxuICAgICAgdGhpcy5fZGF0ZUFkYXB0ZXIuaXNWYWxpZCh0aGlzLm1heCkgJiYgdGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0WWVhcih0aGlzLm1heCEpIDwgeWVhcjtcbiAgICByZXR1cm4gIShpc0JlZm9yZU1pbiB8fCBpc0FmdGVyTWF4KTtcbiAgfVxuXG4gIC8vIEltcGxlbWVudGF0aW9uIGFkYXB0ZWQgZnJvbSBodHRwczovL2dpdGh1Yi5jb20vYW5ndWxhci9jb21wb25lbnRzL2Jsb2IvbWFpbi9zcmMvbWF0ZXJpYWwvZGF0ZXBpY2tlci95ZWFyLXZpZXcudHMjTDM2NlxuICBwcml2YXRlIF9pc01vbnRoRmlsdGVyZWRPdXQobW9udGg6IG51bWJlciwgeWVhcjogbnVtYmVyKTogYm9vbGVhbiB7XG4gICAgaWYgKCF0aGlzLmRhdGVGaWx0ZXIpIHtcbiAgICAgIHJldHVybiB0cnVlO1xuICAgIH1cblxuICAgIGNvbnN0IGZpcnN0T2ZNb250aCA9IHRoaXMuX2RhdGVBZGFwdGVyLmNyZWF0ZURhdGUoeWVhciwgbW9udGgsIDEpITtcbiAgICBmb3IgKFxuICAgICAgbGV0IGRhdGU6IFQgPSBmaXJzdE9mTW9udGg7XG4gICAgICB0aGlzLl9kYXRlQWRhcHRlci5nZXRNb250aChkYXRlKSA9PSBtb250aDtcbiAgICAgIGRhdGUgPSB0aGlzLl9kYXRlQWRhcHRlci5hZGRDYWxlbmRhckRheXMoZGF0ZSwgMSlcbiAgICApIHtcbiAgICAgIGlmICh0aGlzLmRhdGVGaWx0ZXIoZGF0ZSkpIHtcbiAgICAgICAgcmV0dXJuIHRydWU7XG4gICAgICB9XG4gICAgfVxuXG4gICAgcmV0dXJuIGZhbHNlO1xuICB9XG5cbiAgLy8gSW1wbGVtZW50YXRpb24gYWRhcHRlZCBmcm9tIGh0dHBzOi8vZ2l0aHViLmNvbS9hbmd1bGFyL2NvbXBvbmVudHMvYmxvYi9tYWluL3NyYy9tYXRlcmlhbC9kYXRlcGlja2VyL211bHRpLXllYXItdmlldy50cyNMMzUxXG4gIHByaXZhdGUgX2lzWWVhckZpbHRlcmVkT3V0KHllYXI6IG51bWJlcik6IGJvb2xlYW4ge1xuICAgIGlmICghdGhpcy5kYXRlRmlsdGVyKSB7XG4gICAgICByZXR1cm4gdHJ1ZTtcbiAgICB9XG5cbiAgICBjb25zdCBmaXJzdE9mWWVhciA9IHRoaXMuX2RhdGVBZGFwdGVyLmNyZWF0ZURhdGUoeWVhciwgMSwgMSkhO1xuICAgIGZvciAoXG4gICAgICBsZXQgZGF0ZTogVCA9IGZpcnN0T2ZZZWFyO1xuICAgICAgdGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0WWVhcihkYXRlKSA9PSB5ZWFyO1xuICAgICAgZGF0ZSA9IHRoaXMuX2RhdGVBZGFwdGVyLmFkZENhbGVuZGFyRGF5cyhkYXRlLCAxKVxuICAgICkge1xuICAgICAgaWYgKHRoaXMuZGF0ZUZpbHRlcihkYXRlKSkge1xuICAgICAgICByZXR1cm4gdHJ1ZTtcbiAgICAgIH1cbiAgICB9XG5cbiAgICByZXR1cm4gZmFsc2U7XG4gIH1cblxuICAvKiogRW1pdHMgdGhlIHNlbGVjdGVkIGRhdGUgYW5kIHNldHMgaXQgaW50ZXJuYWxseS4gKi9cbiAgcHJpdmF0ZSBfc2VsZWN0RGF0ZShkYXk6IFQpOiB2b2lkIHtcbiAgICB0aGlzLl9jaG9zZW5Nb250aCA9IHVuZGVmaW5lZDtcbiAgICB0aGlzLl9zZXRDaG9zZW5ZZWFyKCk7XG4gICAgaWYgKHRoaXMubXVsdGlwbGUpIHtcbiAgICAgIC8vIENoZWNrIGlmIF9zZWxlY3RlZCBoYXMgZWxlbWVudHNcbiAgICAgIGlmICh0aGlzLl9zZWxlY3RlZCAmJiAodGhpcy5fc2VsZWN0ZWQgYXMgVFtdKS5sZW5ndGggPiAwKSB7XG4gICAgICAgIGNvbnN0IGluZGV4T2ZTZWxlY3RlZERheTogbnVtYmVyID0gKHRoaXMuX3NlbGVjdGVkIGFzIFRbXSkuZmluZEluZGV4KFxuICAgICAgICAgIChzZWwpID0+IHRoaXMuX2RhdGVBZGFwdGVyLmNvbXBhcmVEYXRlKHNlbCwgZGF5KSA9PT0gMCxcbiAgICAgICAgKTtcbiAgICAgICAgLy8gSWYgdGhlIHNlbGVjdGVkIGRhdGUgaXMgYWxyZWFkeSBpbiB0aGUgX3NlbGVjdGVkIGFycmF5LCByZW1vdmUgaXQsIG90aGVyd2lzZSBhZGQgaXRcbiAgICAgICAgaWYgKGluZGV4T2ZTZWxlY3RlZERheSAhPT0gLTEpIHtcbiAgICAgICAgICB0aGlzLl9zZWxlY3RlZCA9ICh0aGlzLl9zZWxlY3RlZCBhcyBUW10pLmZpbHRlcigoXywgaSkgPT4gaSAhPT0gaW5kZXhPZlNlbGVjdGVkRGF5KTtcbiAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICB0aGlzLl9zZWxlY3RlZCA9IFsuLi4odGhpcy5fc2VsZWN0ZWQgYXMgVFtdKSwgZGF5XTtcbiAgICAgICAgfVxuICAgICAgfSBlbHNlIHtcbiAgICAgICAgLy8gSWYgX3NlbGVjdGVkIGlzIGVtcHR5LCBzZXQgaXRcbiAgICAgICAgdGhpcy5fc2VsZWN0ZWQgPSBbZGF5XTtcbiAgICAgIH1cbiAgICAgIHRoaXMuX2VtaXREYXRlU2VsZWN0ZWRFdmVudCh0aGlzLl9zZWxlY3RlZC5tYXAoKGUpID0+IHRoaXMuX2RhdGVBZGFwdGVyLmRlc2VyaWFsaXplKGUpISkpO1xuICAgIH0gZWxzZSB7XG4gICAgICAvLyBJbiBzaW5nbGUgc2VsZWN0aW9uLCBjaGVjayBpZiB0aGUgZGF5IGlzIGFscmVhZHkgc2VsZWN0ZWRcbiAgICAgIGlmICghdGhpcy5fc2VsZWN0ZWQgfHwgdGhpcy5fZGF0ZUFkYXB0ZXIuY29tcGFyZURhdGUodGhpcy5fc2VsZWN0ZWQgYXMgVCwgZGF5KSAhPT0gMCkge1xuICAgICAgICB0aGlzLl9zZWxlY3RlZCA9IGRheTtcbiAgICAgICAgdGhpcy5fZW1pdERhdGVTZWxlY3RlZEV2ZW50KHRoaXMuX2RhdGVBZGFwdGVyLmRlc2VyaWFsaXplKGRheSkhKTtcbiAgICAgIH1cbiAgICB9XG4gIH1cblxuICAvKipcbiAgICogSGFuZGxlIG11bHRpcGxlIGRhdGVzIHNlbGVjdGlvbiB2aWEgd2Vla051bWJlciAvIHdlZWtEYXkgYnV0dG9uczpcbiAgICogLSBpZiBDbWQgb3IgQ3RybCBhcmUgcHJlc3NlZCwgYWRkIHRoZSBuZXcgZGF0ZSB0byB0aGUgY3VycmVudCBvbmVzO1xuICAgKiAtIGlmIG5vdCxcbiAgICogICAgIC0gaWYgdGhlIG5ldyBkYXRlcyBhcmUgdGhlIHNhbWUgb2YgdGhlIGN1cnJlbnQgb25lcywgaXQgbWVhbnMgdGhhdCB0aGUgc2FtZSBidXR0b24gaGFzIGJlZW4gY2xpY2tlZCB0d2ljZSwgc28gZG8gbm90aGluZztcbiAgICogICAgIC0gaWYgbm90LCB0aGUgc2VsZWN0ZWQgZGF0ZXMgYXJlIHRoZSBuZXcgb25lcy5cbiAgICovXG4gIHByaXZhdGUgX3NlbGVjdE11bHRpcGxlRGF0ZXMoZGF5czogRGF5PFQ+W10pOiB2b2lkIHtcbiAgICAvLyBGaWx0ZXIgZGlzYWJsZWQgZGF5cyBieSBtYXRjaGluZyB0aGUgcHJvdmlkZWQgYGRheXNgIHBhcmFtZXRlciBhZ2FpbnN0IHRoZSBlbmFibGVkIGNlbGxzLlxuICAgIC8vIFNpbmNlIHRoZSBidXR0b25zJyB2YWx1ZSBpcyBzZXQgdG8gdGhlIERheSdzIGludGVyZmFjZSB2YWx1ZSAoSVNPIHN0cmluZyksIHRoZXJlJ3Mgbm8gbmVlZCB0byBkZXNlcmlhbGl6ZSBpdC5cbiAgICBjb25zdCBlbmFibGVkRGF5czogc3RyaW5nW10gPSB0aGlzLl9jZWxsc1xuICAgICAgLmZpbHRlcigoZSkgPT4gIWUuZGlzYWJsZWQpXG4gICAgICAubWFwKChlKSA9PiB0aGlzLl9tYXBWYWx1ZVRvSVNPRGF0ZShlLnZhbHVlISkpO1xuICAgIGNvbnN0IGRheXNUb0FkZDogc3RyaW5nW10gPSBkYXlzXG4gICAgICAubWFwKChlOiBEYXk8VD4pID0+IGUudmFsdWUpXG4gICAgICAuZmlsdGVyKChpc29EYXRlOiBzdHJpbmcpID0+IGVuYWJsZWREYXlzLmluY2x1ZGVzKGlzb0RhdGUpKTtcbiAgICBjb25zdCBkYXlzVG9BZGRTZXQgPSBuZXcgU2V0KGRheXNUb0FkZCk7XG4gICAgY29uc3Qgc2VsZWN0ZWRTZXQgPSBuZXcgU2V0KCh0aGlzLl9zZWxlY3RlZCBhcyBUW10pLm1hcCgocykgPT4gdGhpcy5fZGF0ZUFkYXB0ZXIudG9Jc284NjAxKHMpKSk7XG4gICAgY29uc3Qgc2VsU3RyaW5ncyA9IHRoaXMuX3VwZGF0ZVNlbGVjdGVkV2l0aE11bHRpcGxlRGF0ZXMoZGF5c1RvQWRkLCBkYXlzVG9BZGRTZXQsIHNlbGVjdGVkU2V0KTtcbiAgICB0aGlzLl9zZWxlY3RlZCA9IHNlbFN0cmluZ3MubWFwKChzKSA9PiB0aGlzLl9kYXRlQWRhcHRlci5kZXNlcmlhbGl6ZShzKSEpO1xuXG4gICAgdGhpcy5fZW1pdERhdGVTZWxlY3RlZEV2ZW50KHRoaXMuX3NlbGVjdGVkLm1hcCgoZSkgPT4gdGhpcy5fZGF0ZUFkYXB0ZXIuZGVzZXJpYWxpemUoZSkhKSk7XG4gIH1cblxuICAvKipcbiAgICogRW1pdHMgdGhlIGRhdGVzZWxlY3RlZCBldmVudCBnaXZlbiB0aGUgZGV0YWlsIChhcyBUIG9yIFRbXSBiYXNlZCBvbiB0aGUgdmFsdWUgb2YgdGhlIG11bHRpcGxlIGZsYWcpLlxuICAgKi9cbiAgcHJpdmF0ZSBfZW1pdERhdGVTZWxlY3RlZEV2ZW50KGRldGFpbDogVCB8IFRbXSk6IHZvaWQge1xuICAgIC8qKiBAdHlwZSB7Q3VzdG9tRXZlbnQ8VCB8IFRbXT59IEV2ZW50IGVtaXR0ZWQgb24gZGF0ZSBzZWxlY3Rpb24uICovXG4gICAgdGhpcy5kaXNwYXRjaEV2ZW50KFxuICAgICAgbmV3IEN1c3RvbUV2ZW50PFQgfCBUW10+KCdkYXRlc2VsZWN0ZWQnLCB7XG4gICAgICAgIGRldGFpbCxcbiAgICAgICAgY29tcG9zZWQ6IHRydWUsXG4gICAgICAgIGJ1YmJsZXM6IHRydWUsXG4gICAgICB9KSxcbiAgICApO1xuICB9XG5cbiAgcHJpdmF0ZSBfZW1pdE1vbnRoQ2hhbmdlKCk6IHZvaWQge1xuICAgIC8vIEZJWE1FOiB0aGUgbmFtZSBvZiB0aGlzIHZhcmlhYmxlIGFwcGVhcnMgYXMgZXZlbnQgbmFtZSBpbiB0aGUgcmVhZG1lXG4gICAgLy8gIGR1ZSB0byBhIGJ1ZyBpbiB0aGUgY3VzdG9tLWVsZW1lbnRzLW1hbmlmZXN0IGxpYnJhcnkuXG4gICAgLy8gIGh0dHBzOi8vZ2l0aHViLmNvbS9vcGVuLXdjL2N1c3RvbS1lbGVtZW50cy1tYW5pZmVzdC9pc3N1ZXMvMTQ5XG4gICAgY29uc3QgbW9udGhjaGFuZ2UgPSAodGhpcy53aWRlID8gWy4uLnRoaXMuX3dlZWtzLCAuLi50aGlzLl9uZXh0TW9udGhXZWVrc10gOiB0aGlzLl93ZWVrcylcbiAgICAgIC5mbGF0KClcbiAgICAgIC5zb3J0KChhLCBiKSA9PiBhLnZhbHVlLmxvY2FsZUNvbXBhcmUoYi52YWx1ZSkpIGFzIERheVtdO1xuICAgIC8qKlxuICAgICAqIEB0eXBlIHtTYmJNb250aENoYW5nZUV2ZW50fVxuICAgICAqIEVtaXRzIHdoZW4gdGhlIG1vbnRoIGNoYW5nZXMuXG4gICAgICogVGhlIGByYW5nZWAgcHJvcGVydHkgY29udGFpbnMgdGhlIGRheXMgYXJyYXkgb2YgdGhlIGNob3NlbiBtb250aC5cbiAgICAgKi9cbiAgICB0aGlzLmRpc3BhdGNoRXZlbnQobmV3IFNiYk1vbnRoQ2hhbmdlRXZlbnQobW9udGhjaGFuZ2UpKTtcbiAgfVxuXG4gIC8qKlxuICAgKiBJbiBjYXNlIG9mIG11bHRpcGxlIHNlbGVjdGlvbiwgbmV3bHkgYWRkZWQgZGF5cyBtdXN0IGJlIGFkZGVkIHRvIHRoZSBleGlzdGluZyBvbmVzLCB3aXRob3V0IGR1cGxpY2F0aW9uLlxuICAgKiBJZiB0aGUgZGF5cyB0byBhZGQgYXJlIGV4YWN0bHkgdGhlIHNhbWUgYXMgdGhlIHNlbGVjdGVkIG9uZXMsIHRoZSBzZXQgbXVzdCBiZSBlbXB0aWVkLlxuICAgKi9cbiAgcHJpdmF0ZSBfdXBkYXRlU2VsZWN0ZWRXaXRoTXVsdGlwbGVEYXRlcyhcbiAgICBkYXlzVG9BZGQ6IHN0cmluZ1tdLFxuICAgIGRheXNUb0FkZFNldDogU2V0PHN0cmluZz4sXG4gICAgc2VsZWN0ZWRTZXQ6IFNldDxzdHJpbmc+LFxuICApOiBzdHJpbmdbXSB7XG4gICAgaWYgKGRheXNUb0FkZC5ldmVyeSgoZGF5OiBzdHJpbmcpID0+IHNlbGVjdGVkU2V0LmhhcyhkYXkpKSkge1xuICAgICAgZGF5c1RvQWRkU2V0LmZvckVhY2goKGRheTogc3RyaW5nKSA9PiBzZWxlY3RlZFNldC5kZWxldGUoZGF5KSk7XG4gICAgfSBlbHNlIHtcbiAgICAgIGRheXNUb0FkZFNldC5mb3JFYWNoKChkYXk6IHN0cmluZykgPT4gc2VsZWN0ZWRTZXQuYWRkKGRheSkpO1xuICAgIH1cbiAgICByZXR1cm4gQXJyYXkuZnJvbShzZWxlY3RlZFNldCk7XG4gIH1cblxuICBwcml2YXRlIF9zZXRDaG9zZW5ZZWFyKCk6IHZvaWQge1xuICAgIGlmICh0aGlzLnZpZXcgPT09ICdtb250aCcpIHtcbiAgICAgIGxldCBzZWxlY3RlZERhdGU6IFQgfCB1bmRlZmluZWQ7XG4gICAgICBpZiAodGhpcy5tdWx0aXBsZSkge1xuICAgICAgICBzZWxlY3RlZERhdGUgPSAodGhpcy5zZWxlY3RlZCBhcyBUW10pLmF0KC0xKTtcbiAgICAgIH0gZWxzZSB7XG4gICAgICAgIHNlbGVjdGVkRGF0ZSA9IHRoaXMuc2VsZWN0ZWQgYXMgVDtcbiAgICAgIH1cbiAgICAgIHRoaXMuX2Nob3NlblllYXIgPSB0aGlzLl9kYXRlQWRhcHRlci5nZXRZZWFyKHNlbGVjdGVkRGF0ZSA/PyB0aGlzLl9kYXRlQWRhcHRlci50b2RheSgpKTtcbiAgICB9IGVsc2Uge1xuICAgICAgdGhpcy5fY2hvc2VuWWVhciA9IHVuZGVmaW5lZDtcbiAgICB9XG4gIH1cblxuICBwcml2YXRlIF9hc3NpZ25BY3RpdmVEYXRlKGRhdGU6IFQpOiB2b2lkIHtcbiAgICBpZiAodGhpcy5taW4gJiYgdGhpcy5fZGF0ZUFkYXB0ZXIuY29tcGFyZURhdGUodGhpcy5taW4sIGRhdGUpID4gMCkge1xuICAgICAgdGhpcy5fYWN0aXZlRGF0ZSA9IHRoaXMubWluO1xuICAgICAgcmV0dXJuO1xuICAgIH1cbiAgICBpZiAodGhpcy5tYXggJiYgdGhpcy5fZGF0ZUFkYXB0ZXIuY29tcGFyZURhdGUodGhpcy5tYXgsIGRhdGUpIDwgMCkge1xuICAgICAgdGhpcy5fYWN0aXZlRGF0ZSA9IHRoaXMubWF4O1xuICAgICAgcmV0dXJuO1xuICAgIH1cbiAgICB0aGlzLl9hY3RpdmVEYXRlID0gZGF0ZTtcbiAgfVxuXG4gIC8qKiBHb2VzIHRvIHRoZSBtb250aCBpZGVudGlmaWVkIGJ5IHRoZSBzaGlmdC4gKi9cbiAgcHJpdmF0ZSBfZ29Ub0RpZmZlcmVudE1vbnRoKG1vbnRoczogbnVtYmVyKTogdm9pZCB7XG4gICAgdGhpcy5faW5pdCh0aGlzLl9kYXRlQWRhcHRlci5hZGRDYWxlbmRhck1vbnRocyh0aGlzLl9hY3RpdmVEYXRlLCBtb250aHMpKTtcbiAgICB0aGlzLl9lbWl0TW9udGhDaGFuZ2UoKTtcbiAgfVxuXG4gIHByaXZhdGUgX2dvVG9EaWZmZXJlbnRZZWFyKHllYXJzOiBudW1iZXIpOiB2b2lkIHtcbiAgICB0aGlzLl9jaG9zZW5ZZWFyISArPSB5ZWFycztcbiAgICAvLyBDYW4ndCB1c2UgYF9hc3NpZ25BY3RpdmVEYXRlKC4uLilgIGhlcmUsIGJlY2F1c2UgaXQgd2lsbCBzZXQgaXQgdG8gbWluL21heCB2YWx1ZSBpZiBhcmd1bWVudCBpcyBvdXQgb2YgcmFuZ2VcbiAgICB0aGlzLl9hY3RpdmVEYXRlID0gdGhpcy5fZGF0ZUFkYXB0ZXIuY3JlYXRlRGF0ZShcbiAgICAgIHRoaXMuX2Nob3NlblllYXIhLFxuICAgICAgdGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0TW9udGgodGhpcy5fYWN0aXZlRGF0ZSksXG4gICAgICB0aGlzLl9kYXRlQWRhcHRlci5nZXREYXRlKHRoaXMuX2FjdGl2ZURhdGUpLFxuICAgICk7XG4gICAgdGhpcy5faW5pdCgpO1xuICB9XG5cbiAgcHJpdmF0ZSBfZ29Ub0RpZmZlcmVudFllYXJSYW5nZSh5ZWFyczogbnVtYmVyKTogdm9pZCB7XG4gICAgdGhpcy5faW5pdCh0aGlzLl9kYXRlQWRhcHRlci5hZGRDYWxlbmRhclllYXJzKHRoaXMuX2FjdGl2ZURhdGUsIHllYXJzKSk7XG4gIH1cblxuICBwcml2YXRlIF9wcmV2RGlzYWJsZWQocHJldkRhdGU6IFQpOiBib29sZWFuIHtcbiAgICBpZiAoIXRoaXMubWluKSB7XG4gICAgICByZXR1cm4gZmFsc2U7XG4gICAgfVxuICAgIHJldHVybiB0aGlzLl9kYXRlQWRhcHRlci5jb21wYXJlRGF0ZShwcmV2RGF0ZSwgdGhpcy5taW4pIDwgMDtcbiAgfVxuXG4gIHByaXZhdGUgX25leHREaXNhYmxlZChuZXh0RGF0ZTogVCk6IGJvb2xlYW4ge1xuICAgIGlmICghdGhpcy5tYXgpIHtcbiAgICAgIHJldHVybiBmYWxzZTtcbiAgICB9XG4gICAgcmV0dXJuIHRoaXMuX2RhdGVBZGFwdGVyLmNvbXBhcmVEYXRlKG5leHREYXRlLCB0aGlzLm1heCkgPiAwO1xuICB9XG5cbiAgLyoqIENoZWNrcyBpZiB0aGUgXCJwcmV2aW91cyBtb250aFwiIGJ1dHRvbiBzaG91bGQgYmUgZGlzYWJsZWQuICovXG4gIHByaXZhdGUgX3ByZXZpb3VzTW9udGhEaXNhYmxlZCgpOiBib29sZWFuIHtcbiAgICBjb25zdCBwcmV2TW9udGggPSB0aGlzLl9kYXRlQWRhcHRlci5hZGRDYWxlbmRhckRheXMoXG4gICAgICB0aGlzLl9hY3RpdmVEYXRlLFxuICAgICAgdGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0RGF0ZSh0aGlzLl9hY3RpdmVEYXRlKSAqIC0xLFxuICAgICk7XG4gICAgcmV0dXJuIHRoaXMuX3ByZXZEaXNhYmxlZChwcmV2TW9udGgpO1xuICB9XG5cbiAgLyoqIENoZWNrcyBpZiB0aGUgXCJuZXh0IG1vbnRoXCIgYnV0dG9uIHNob3VsZCBiZSBkaXNhYmxlZC4gKi9cbiAgcHJpdmF0ZSBfbmV4dE1vbnRoRGlzYWJsZWQoKTogYm9vbGVhbiB7XG4gICAgbGV0IG5leHRNb250aCA9IHRoaXMuX2RhdGVBZGFwdGVyLmFkZENhbGVuZGFyTW9udGhzKHRoaXMuX2FjdGl2ZURhdGUsIHRoaXMuX3dpZGUgPyAyIDogMSk7XG4gICAgbmV4dE1vbnRoID0gdGhpcy5fZGF0ZUFkYXB0ZXIuY3JlYXRlRGF0ZShcbiAgICAgIHRoaXMuX2RhdGVBZGFwdGVyLmdldFllYXIobmV4dE1vbnRoKSxcbiAgICAgIHRoaXMuX2RhdGVBZGFwdGVyLmdldE1vbnRoKG5leHRNb250aCksXG4gICAgICAxLFxuICAgICk7XG4gICAgcmV0dXJuIHRoaXMuX25leHREaXNhYmxlZChuZXh0TW9udGgpO1xuICB9XG5cbiAgLyoqIENoZWNrcyBpZiB0aGUgXCJwcmV2aW91cyB5ZWFyXCIgYnV0dG9uIHNob3VsZCBiZSBkaXNhYmxlZC4gKi9cbiAgcHJpdmF0ZSBfcHJldmlvdXNZZWFyRGlzYWJsZWQoKTogYm9vbGVhbiB7XG4gICAgY29uc3QgcHJldlllYXIgPSB0aGlzLl9kYXRlQWRhcHRlci5jcmVhdGVEYXRlKFxuICAgICAgdGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0WWVhcih0aGlzLl9hY3RpdmVEYXRlKSAtIDEsXG4gICAgICAxMixcbiAgICAgIDMxLFxuICAgICk7XG4gICAgcmV0dXJuIHRoaXMuX3ByZXZEaXNhYmxlZChwcmV2WWVhcik7XG4gIH1cblxuICAvKiogQ2hlY2tzIGlmIHRoZSBcIm5leHQgeWVhclwiIGJ1dHRvbiBzaG91bGQgYmUgZGlzYWJsZWQuICovXG4gIHByaXZhdGUgX25leHRZZWFyRGlzYWJsZWQoKTogYm9vbGVhbiB7XG4gICAgY29uc3QgbmV4dFllYXIgPSB0aGlzLl9kYXRlQWRhcHRlci5jcmVhdGVEYXRlKFxuICAgICAgdGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0WWVhcih0aGlzLl9hY3RpdmVEYXRlKSArICh0aGlzLl93aWRlID8gMiA6IDEpLFxuICAgICAgMSxcbiAgICAgIDEsXG4gICAgKTtcbiAgICByZXR1cm4gdGhpcy5fbmV4dERpc2FibGVkKG5leHRZZWFyKTtcbiAgfVxuXG4gIC8qKiBDaGVja3MgaWYgdGhlIFwicHJldmlvdXMgeWVhclwiIGJ1dHRvbiBzaG91bGQgYmUgZGlzYWJsZWQgaW4geWVhciB2aWV3LiAqL1xuICBwcml2YXRlIF9wcmV2aW91c1llYXJSYW5nZURpc2FibGVkKCk6IGJvb2xlYW4ge1xuICAgIGNvbnN0IHByZXZZZWFyID0gdGhpcy5fZGF0ZUFkYXB0ZXIuY3JlYXRlRGF0ZSh0aGlzLl95ZWFyc1swXVswXSAtIDEsIDEyLCAzMSk7XG4gICAgcmV0dXJuIHRoaXMuX3ByZXZEaXNhYmxlZChwcmV2WWVhcik7XG4gIH1cblxuICAvKiogQ2hlY2tzIGlmIHRoZSBcIm5leHQgeWVhclwiIGJ1dHRvbiBzaG91bGQgYmUgZGlzYWJsZWQgaW4geWVhciB2aWV3LiAqL1xuICBwcml2YXRlIF9uZXh0WWVhclJhbmdlRGlzYWJsZWQoKTogYm9vbGVhbiB7XG4gICAgY29uc3QgeWVhcnMgPSB0aGlzLl93aWRlID8gdGhpcy5fbmV4dE1vbnRoWWVhcnMgOiB0aGlzLl95ZWFycztcbiAgICBjb25zdCBsYXN0WWVhclJhbmdlID0geWVhcnNbeWVhcnMubGVuZ3RoIC0gMV07XG4gICAgY29uc3QgbGFzdFllYXIgPSBsYXN0WWVhclJhbmdlW2xhc3RZZWFyUmFuZ2UubGVuZ3RoIC0gMV07XG4gICAgY29uc3QgbmV4dFllYXIgPSB0aGlzLl9kYXRlQWRhcHRlci5jcmVhdGVEYXRlKGxhc3RZZWFyICsgMSwgMSwgMSk7XG4gICAgcmV0dXJuIHRoaXMuX25leHREaXNhYmxlZChuZXh0WWVhcik7XG4gIH1cblxuICBwcml2YXRlIF9oYW5kbGVUYWJsZUJsdXIoZXZlbnRUYXJnZXQ6IEhUTUxFbGVtZW50KTogdm9pZCB7XG4gICAgaWYgKGV2ZW50VGFyZ2V0Py5sb2NhbE5hbWUgIT09ICdzYmItY2FsZW5kYXItZGF5Jykge1xuICAgICAgdGhpcy5fc2V0VGFiSW5kZXgoKTtcbiAgICB9XG4gIH1cblxuICBwcml2YXRlIF9zZXRUYWJJbmRleCgpOiB2b2lkIHtcbiAgICBjb25zdCBxdWVyeSA9IHRoaXMuX2NhbGVuZGFyVmlldyA9PT0gJ2RheScgPyAnc2JiLWNhbGVuZGFyLWRheScgOiAnLnNiYi1jYWxlbmRhcl9fY2VsbCc7XG4gICAgQXJyYXkuZnJvbShcbiAgICAgIHRoaXMuX2dldFJvb3RGb3JRdWVyeVNlbGVjdG9yKCkucXVlcnlTZWxlY3RvckFsbChgJHtxdWVyeX1bdGFiaW5kZXg9XCIwXCJdYCkgPz8gW10sXG4gICAgKS5mb3JFYWNoKChkYXkpID0+ICgoZGF5IGFzIEhUTUxCdXR0b25FbGVtZW50IHwgU2JiQ2FsZW5kYXJEYXlFbGVtZW50KS50YWJJbmRleCA9IC0xKSk7XG4gICAgY29uc3QgZmlyc3RGb2N1c2FibGUgPSB0aGlzLl9nZXRGaXJzdEZvY3VzYWJsZSgpO1xuICAgIGlmIChmaXJzdEZvY3VzYWJsZSkge1xuICAgICAgZmlyc3RGb2N1c2FibGUudGFiSW5kZXggPSAwO1xuICAgIH1cbiAgfVxuXG4gIC8qKiBHZXQgdGhlIGVsZW1lbnQgaW4gdGhlIGNhbGVuZGFyIHRvIGFzc2lnbiBmb2N1cy4gKi9cbiAgcHJpdmF0ZSBfZ2V0Rmlyc3RGb2N1c2FibGUoKTogU2JiQ2FsZW5kYXJEYXlFbGVtZW50IHwgSFRNTEJ1dHRvbkVsZW1lbnQgfCBudWxsIHtcbiAgICBjb25zdCByb290ID0gdGhpcy5fZ2V0Um9vdEZvclF1ZXJ5U2VsZWN0b3IoKTtcbiAgICBpZiAodGhpcy5fY2FsZW5kYXJWaWV3ID09PSAnZGF5Jykge1xuICAgICAgY29uc3Qgc2VsZWN0ZWRPckN1cnJlbnQgPVxuICAgICAgICByb290LnF1ZXJ5U2VsZWN0b3I8U2JiQ2FsZW5kYXJEYXlFbGVtZW50PignOnN0YXRlKHNlbGVjdGVkKScpID8/XG4gICAgICAgIHJvb3QucXVlcnlTZWxlY3RvcjxTYmJDYWxlbmRhckRheUVsZW1lbnQ+KCc6c3RhdGUoY3VycmVudCknKTtcbiAgICAgIHJldHVybiBzZWxlY3RlZE9yQ3VycmVudCAmJiAhc2VsZWN0ZWRPckN1cnJlbnQuZGlzYWJsZWRcbiAgICAgICAgPyBzZWxlY3RlZE9yQ3VycmVudFxuICAgICAgICA6IHRoaXMuX2dldEZpcnN0Rm9jdXNhYmxlRGF5KCk7XG4gICAgfSBlbHNlIHtcbiAgICAgIGNvbnN0IHNlbGVjdGVkT3JDdXJyZW50ID0gdGhpcy5zaGFkb3dSb290Py5xdWVyeVNlbGVjdG9yPEhUTUxCdXR0b25FbGVtZW50PihcbiAgICAgICAgJy5zYmItY2FsZW5kYXJfX2NlbGwtY3VycmVudCcsXG4gICAgICApO1xuICAgICAgcmV0dXJuIHNlbGVjdGVkT3JDdXJyZW50ICYmICFzZWxlY3RlZE9yQ3VycmVudC5kaXNhYmxlZFxuICAgICAgICA/IHNlbGVjdGVkT3JDdXJyZW50XG4gICAgICAgIDogdGhpcy5zaGFkb3dSb290IS5xdWVyeVNlbGVjdG9yPEhUTUxCdXR0b25FbGVtZW50PignLnNiYi1jYWxlbmRhcl9fY2VsbDpub3QoW2Rpc2FibGVkXSknKTtcbiAgICB9XG4gIH1cblxuICAvKipcbiAgICogSW4gYGRheWAgdmlldyBpbiBgdmVydGljYWxgIG9yaWVudGF0aW9uLFxuICAgKiBpZiB0aGUgZmlyc3Qgb2YgdGhlIG1vbnRoIGlzIG5vdCBhIE1vbmRheSwgaXQgaXMgbm90IHRoZSBmaXJzdCByZW5kZXJlZCBlbGVtZW50IGluIHRoZSB0YWJsZSxcbiAgICogc28gYHRoaXMuc2hhZG93Um9vdCEucXVlcnlTZWxlY3RvcignLnNiYi1jYWxlbmRhcl9fY2VsbDpub3QoW2Rpc2FibGVkXSknKWAgd2lsbCByZXR1cm4gYSB3cm9uZyB2YWx1ZS5cbiAgICpcbiAgICogVG8gc29sdmUgdGhpcywgdGhlIGVsZW1lbnQgd2l0aCB0aGUgbG93ZXN0IGB2YWx1ZWAgaXMgdGFrZW4gKElTTyBTdHJpbmcgYXJlIG9yZGVyZWQpLlxuICAgKi9cbiAgcHJpdmF0ZSBfZ2V0Rmlyc3RGb2N1c2FibGVEYXkoKTogU2JiQ2FsZW5kYXJEYXlFbGVtZW50IHwgbnVsbCB7XG4gICAgY29uc3QgZGF5c0luVmlldzogU2JiQ2FsZW5kYXJEYXlFbGVtZW50W10gPSBBcnJheS5mcm9tKFxuICAgICAgdGhpcy5fZ2V0Um9vdEZvclF1ZXJ5U2VsZWN0b3IoKS5xdWVyeVNlbGVjdG9yQWxsKCdzYmItY2FsZW5kYXItZGF5Om5vdChbZGlzYWJsZWRdKScpLFxuICAgICk7XG4gICAgaWYgKCFkYXlzSW5WaWV3IHx8IGRheXNJblZpZXcubGVuZ3RoID09PSAwKSB7XG4gICAgICByZXR1cm4gbnVsbDtcbiAgICB9IGVsc2Uge1xuICAgICAgY29uc3QgZmlyc3RFbGVtZW50ID0gZGF5c0luVmlld1xuICAgICAgICAubWFwKChlOiBTYmJDYWxlbmRhckRheUVsZW1lbnQpOiBzdHJpbmcgPT4gdGhpcy5fZGF0ZUFkYXB0ZXIudG9Jc284NjAxKGUudmFsdWUhIGFzIFQpKVxuICAgICAgICAuc29ydCgpWzBdO1xuICAgICAgcmV0dXJuIHRoaXMuX2dldFJvb3RGb3JRdWVyeVNlbGVjdG9yKCkucXVlcnlTZWxlY3RvcihcbiAgICAgICAgYHNiYi1jYWxlbmRhci1kYXlbc2xvdD1cIiR7Zmlyc3RFbGVtZW50fVwiXWAsXG4gICAgICApO1xuICAgIH1cbiAgfVxuXG4gIHByaXZhdGUgX2hhbmRsZUtleWJvYXJkRXZlbnQoZXZlbnQ6IEtleWJvYXJkRXZlbnQsIGRheT86IERheTxUPik6IHZvaWQge1xuICAgIGlmIChpc0Fycm93S2V5T3JQYWdlS2V5c1ByZXNzZWQoZXZlbnQpKSB7XG4gICAgICBldmVudC5wcmV2ZW50RGVmYXVsdCgpO1xuICAgIH1cbiAgICAvLyBHZXRzIHRoZSBjdXJyZW50bHkgcmVuZGVyZWQgdGFibGUncyBjZWxsO1xuICAgIC8vIHRoZXkgY291bGQgYmUgZGF5cywgbW9udGhzIG9yIHllYXJzIGJhc2VkIG9uIHRoZSBjdXJyZW50IHNlbGVjdGlvbiB2aWV3LlxuICAgIC8vIElmIGB3aWRlYCBpcyB0cnVlLCB5ZWFycyBhcmUgZG91YmxlZCBpbiBudW1iZXIgYW5kIGRheXMgYXJlIChyb3VnaGx5KSBkb3VibGVkIHRvbywgYWZmZWN0aW5nIHRoZSBgaW5kZXhgIGNhbGN1bGF0aW9uLlxuICAgIGNvbnN0IGNlbGxzID0gdGhpcy5fY2VsbHM7XG4gICAgY29uc3QgaW5kZXg6IG51bWJlciA9IGNlbGxzLmZpbmRJbmRleCgoZSkgPT4gZSA9PT0gZXZlbnQudGFyZ2V0KTtcbiAgICBsZXQgbmV4dEVsOiBIVE1MQnV0dG9uRWxlbWVudCB8IFNiYkNhbGVuZGFyRGF5RWxlbWVudDtcbiAgICBpZiAoZGF5KSB7XG4gICAgICBuZXh0RWwgPSB0aGlzLl9uYXZpZ2F0ZUJ5S2V5Ym9hcmREYXlWaWV3KGV2ZW50LCBpbmRleCwgY2VsbHMsIGRheSk7XG4gICAgfSBlbHNlIHtcbiAgICAgIG5leHRFbCA9IHRoaXMuX25hdmlnYXRlQnlLZXlib2FyZChldmVudCwgaW5kZXgsIGNlbGxzIGFzIEhUTUxCdXR0b25FbGVtZW50W10pO1xuICAgIH1cbiAgICBjb25zdCBhY3RpdmVFbCA9ICh0aGlzLl9lbmhhbmNlZFZhcmlhbnQgPyBkb2N1bWVudCA6IHRoaXMuc2hhZG93Um9vdCEpLmFjdGl2ZUVsZW1lbnQgYXNcbiAgICAgIHwgSFRNTEJ1dHRvbkVsZW1lbnRcbiAgICAgIHwgU2JiQ2FsZW5kYXJEYXlFbGVtZW50O1xuICAgIGlmIChuZXh0RWwgIT09IGFjdGl2ZUVsKSB7XG4gICAgICBuZXh0RWwudGFiSW5kZXggPSAwO1xuICAgICAgbmV4dEVsPy5mb2N1cygpO1xuICAgICAgYWN0aXZlRWwudGFiSW5kZXggPSAtMTtcbiAgICB9XG4gIH1cblxuICBwcml2YXRlIF9uYXZpZ2F0ZUJ5S2V5Ym9hcmREYXlWaWV3KFxuICAgIGV2dDogS2V5Ym9hcmRFdmVudCxcbiAgICBpbmRleDogbnVtYmVyLFxuICAgIGNlbGxzOiAoSFRNTEJ1dHRvbkVsZW1lbnQgfCBTYmJDYWxlbmRhckRheUVsZW1lbnQpW10sXG4gICAgZGF5OiBEYXk8VD4sXG4gICk6IEhUTUxCdXR0b25FbGVtZW50IHwgU2JiQ2FsZW5kYXJEYXlFbGVtZW50IHtcbiAgICBjb25zdCBhcnJvd3NPZmZzZXQgPVxuICAgICAgdGhpcy5vcmllbnRhdGlvbiA9PT0gJ2hvcml6b250YWwnXG4gICAgICAgID8geyBsZWZ0UmlnaHQ6IDEsIHVwRG93bjogREFZU19QRVJfUk9XIH1cbiAgICAgICAgOiB7IGxlZnRSaWdodDogREFZU19QRVJfUk9XLCB1cERvd246IDEgfTtcbiAgICBjb25zdCBvZmZzZXRGb3JWZXJ0aWNhbDogbnVtYmVyID1cbiAgICAgIGluZGV4IDwgdGhpcy5fa2V5Ym9hcmROYXZpZ2F0aW9uRGF5Vmlld1BhcmFtZXRlcnMuZmlyc3RNb250aExlbmd0aFxuICAgICAgICA/IHRoaXMuX2tleWJvYXJkTmF2aWdhdGlvbkRheVZpZXdQYXJhbWV0ZXJzLmZpcnN0TW9udGhPZmZzZXRcbiAgICAgICAgOiB0aGlzLl9rZXlib2FyZE5hdmlnYXRpb25EYXlWaWV3UGFyYW1ldGVycy5zZWNvbmRNb250aE9mZnNldDtcblxuICAgIHN3aXRjaCAoZXZ0LmtleSkge1xuICAgICAgY2FzZSAnQXJyb3dVcCc6XG4gICAgICAgIHJldHVybiB0aGlzLl9maW5kRGF5QXJyb3dzKGNlbGxzLCBpbmRleCwgZGF5LmRhdGVWYWx1ZSwgLWFycm93c09mZnNldC51cERvd24pO1xuICAgICAgY2FzZSAnQXJyb3dEb3duJzpcbiAgICAgICAgcmV0dXJuIHRoaXMuX2ZpbmREYXlBcnJvd3MoY2VsbHMsIGluZGV4LCBkYXkuZGF0ZVZhbHVlLCBhcnJvd3NPZmZzZXQudXBEb3duKTtcbiAgICAgIGNhc2UgJ0Fycm93TGVmdCc6XG4gICAgICAgIHJldHVybiB0aGlzLl9maW5kRGF5QXJyb3dzKGNlbGxzLCBpbmRleCwgZGF5LmRhdGVWYWx1ZSwgLWFycm93c09mZnNldC5sZWZ0UmlnaHQpO1xuICAgICAgY2FzZSAnQXJyb3dSaWdodCc6XG4gICAgICAgIHJldHVybiB0aGlzLl9maW5kRGF5QXJyb3dzKGNlbGxzLCBpbmRleCwgZGF5LmRhdGVWYWx1ZSwgYXJyb3dzT2Zmc2V0LmxlZnRSaWdodCk7XG4gICAgICBjYXNlICdQYWdlVXAnOiB7XG4gICAgICAgIGlmICh0aGlzLm9yaWVudGF0aW9uID09PSAnaG9yaXpvbnRhbCcpIHtcbiAgICAgICAgICBjb25zdCBmaXJzdE9mV2VlazogbnVtYmVyID0gK2RheS5kYXlWYWx1ZSAlIERBWVNfUEVSX1JPVyB8fCBEQVlTX1BFUl9ST1c7XG4gICAgICAgICAgY29uc3QgZGVsdGE6IG51bWJlciA9IGZpcnN0T2ZXZWVrIC0gK2RheS5kYXlWYWx1ZTtcbiAgICAgICAgICByZXR1cm4gdGhpcy5fZmluZERheVBhZ2VVcERvd24oY2VsbHMsIGluZGV4LCBkYXksIGRlbHRhLCBhcnJvd3NPZmZzZXQudXBEb3duKTtcbiAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICBjb25zdCB3ZWVrTnVtYmVyOiBudW1iZXIgPSBNYXRoLmNlaWwoKCtkYXkuZGF5VmFsdWUgKyBvZmZzZXRGb3JWZXJ0aWNhbCkgLyBEQVlTX1BFUl9ST1cpO1xuICAgICAgICAgIGNvbnN0IGZpcnN0T2ZXZWVrOiBudW1iZXIgPSAod2Vla051bWJlciAtIDEpICogREFZU19QRVJfUk9XIC0gb2Zmc2V0Rm9yVmVydGljYWwgKyAxO1xuICAgICAgICAgIGNvbnN0IGRlbHRhOiBudW1iZXIgPSBmaXJzdE9mV2VlayAtICtkYXkuZGF5VmFsdWU7XG4gICAgICAgICAgcmV0dXJuIHRoaXMuX2ZpbmREYXlQYWdlVXBEb3duKGNlbGxzLCBpbmRleCwgZGF5LCBkZWx0YSwgYXJyb3dzT2Zmc2V0LnVwRG93bik7XG4gICAgICAgIH1cbiAgICAgIH1cbiAgICAgIGNhc2UgJ1BhZ2VEb3duJzoge1xuICAgICAgICBpZiAodGhpcy5vcmllbnRhdGlvbiA9PT0gJ2hvcml6b250YWwnKSB7XG4gICAgICAgICAgY29uc3QgbW9udGhJbkJvdW5kcyA9ICtkYXkubW9udGhWYWx1ZSArIDEgPiAxMiA/IDEgOiArZGF5Lm1vbnRoVmFsdWUgKyAxO1xuICAgICAgICAgIGNvbnN0IHllYXJJbkJvdW5kcyA9ICtkYXkubW9udGhWYWx1ZSArIDEgPiAxMiA/ICtkYXkueWVhclZhbHVlICsgMSA6ICtkYXkueWVhclZhbHVlO1xuICAgICAgICAgIGNvbnN0IGZpcnN0TmV4dE1vbnRoOiBUID0gdGhpcy5fZGF0ZUFkYXB0ZXIuY3JlYXRlRGF0ZSh5ZWFySW5Cb3VuZHMsIG1vbnRoSW5Cb3VuZHMsIDEpO1xuICAgICAgICAgIGNvbnN0IGxhc3RPZk1vbnRoOiBudW1iZXIgPSB0aGlzLl9kYXRlQWRhcHRlci5nZXREYXRlKFxuICAgICAgICAgICAgdGhpcy5fZGF0ZUFkYXB0ZXIuYWRkQ2FsZW5kYXJEYXlzKGZpcnN0TmV4dE1vbnRoLCAtMSksXG4gICAgICAgICAgKTtcbiAgICAgICAgICBjb25zdCBkZWx0YTogbnVtYmVyID1cbiAgICAgICAgICAgIE1hdGgudHJ1bmMoKGxhc3RPZk1vbnRoIC0gK2RheS5kYXlWYWx1ZSEpIC8gREFZU19QRVJfUk9XKSAqIERBWVNfUEVSX1JPVztcbiAgICAgICAgICByZXR1cm4gdGhpcy5fZmluZERheVBhZ2VVcERvd24oY2VsbHMsIGluZGV4LCBkYXksIGRlbHRhLCAtYXJyb3dzT2Zmc2V0LnVwRG93bik7XG4gICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgY29uc3Qgd2Vla051bWJlcjogbnVtYmVyID0gTWF0aC5jZWlsKCgrZGF5LmRheVZhbHVlICsgb2Zmc2V0Rm9yVmVydGljYWwpIC8gREFZU19QRVJfUk9XKTtcbiAgICAgICAgICBjb25zdCBsYXN0T2ZXZWVrOiBudW1iZXIgPSB3ZWVrTnVtYmVyICogREFZU19QRVJfUk9XIC0gb2Zmc2V0Rm9yVmVydGljYWw7XG4gICAgICAgICAgY29uc3QgZGVsdGE6IG51bWJlciA9IGxhc3RPZldlZWsgLSArZGF5LmRheVZhbHVlO1xuICAgICAgICAgIHJldHVybiB0aGlzLl9maW5kRGF5UGFnZVVwRG93bihjZWxscywgaW5kZXgsIGRheSwgZGVsdGEsIC1hcnJvd3NPZmZzZXQudXBEb3duKTtcbiAgICAgICAgfVxuICAgICAgfVxuICAgICAgY2FzZSAnSG9tZSc6IHtcbiAgICAgICAgcmV0dXJuIHRoaXMuX2ZpbmREYXlGaXJzdChjZWxscywgaW5kZXgsIGRheSwgMSk7XG4gICAgICB9XG4gICAgICBjYXNlICdFbmQnOiB7XG4gICAgICAgIGNvbnN0IG1vbnRoSW5Cb3VuZHMgPSArZGF5Lm1vbnRoVmFsdWUgKyAxID4gMTIgPyAxIDogK2RheS5tb250aFZhbHVlICsgMTtcbiAgICAgICAgY29uc3QgeWVhckluQm91bmRzID0gK2RheS5tb250aFZhbHVlICsgMSA+IDEyID8gK2RheS55ZWFyVmFsdWUgKyAxIDogK2RheS55ZWFyVmFsdWU7XG4gICAgICAgIGNvbnN0IGZpcnN0TmV4dE1vbnRoOiBUID0gdGhpcy5fZGF0ZUFkYXB0ZXIuY3JlYXRlRGF0ZSh5ZWFySW5Cb3VuZHMsIG1vbnRoSW5Cb3VuZHMsIDEpO1xuICAgICAgICByZXR1cm4gdGhpcy5fZmluZERheUxhc3QoY2VsbHMsIGluZGV4LCBmaXJzdE5leHRNb250aCk7XG4gICAgICB9XG4gICAgICBkZWZhdWx0OlxuICAgICAgICByZXR1cm4gY2VsbHNbaW5kZXhdO1xuICAgIH1cbiAgfVxuXG4gIHByaXZhdGUgX2lzRGF5T3V0T2ZWaWV3KGRhdGU6IHN0cmluZyk6IGJvb2xlYW4ge1xuICAgIHJldHVybiAoXG4gICAgICBkYXRlIDwgdGhpcy5fa2V5Ym9hcmROYXZpZ2F0aW9uRGF5Vmlld1BhcmFtZXRlcnMuZmlyc3REYXlJblZpZXchIHx8XG4gICAgICBkYXRlID4gdGhpcy5fa2V5Ym9hcmROYXZpZ2F0aW9uRGF5Vmlld1BhcmFtZXRlcnMubGFzdERheUluVmlldyFcbiAgICApO1xuICB9XG5cbiAgcHJpdmF0ZSBfZmluZERheUFycm93cyhcbiAgICBjZWxsczogKEhUTUxCdXR0b25FbGVtZW50IHwgU2JiQ2FsZW5kYXJEYXlFbGVtZW50KVtdLFxuICAgIGluZGV4OiBudW1iZXIsXG4gICAgZGF0ZTogVCxcbiAgICBkZWx0YTogbnVtYmVyLFxuICApOiBIVE1MQnV0dG9uRWxlbWVudCB8IFNiYkNhbGVuZGFyRGF5RWxlbWVudCB7XG4gICAgY29uc3QgbmV3RGF0ZVZhbHVlID0gdGhpcy5fZGF0ZUFkYXB0ZXIudG9Jc284NjAxKFxuICAgICAgdGhpcy5fZGF0ZUFkYXB0ZXIuYWRkQ2FsZW5kYXJEYXlzKGRhdGUsIGRlbHRhKSxcbiAgICApO1xuICAgIGlmICh0aGlzLl9pc0RheU91dE9mVmlldyhuZXdEYXRlVmFsdWUpKSB7XG4gICAgICByZXR1cm4gY2VsbHNbaW5kZXhdO1xuICAgIH1cbiAgICBjb25zdCBuZXh0Q2VsbCA9IGNlbGxzLmZpbmQoKGUpID0+IHRoaXMuX21hcFZhbHVlVG9JU09EYXRlKGUudmFsdWUhKSA9PT0gbmV3RGF0ZVZhbHVlKTtcbiAgICBpZiAoIW5leHRDZWxsIHx8IG5leHRDZWxsLmRpc2FibGVkKSB7XG4gICAgICByZXR1cm4gdGhpcy5fZmluZERheUFycm93cyhjZWxscywgaW5kZXgsIHRoaXMuX2RhdGVBZGFwdGVyLmRlc2VyaWFsaXplKG5ld0RhdGVWYWx1ZSkhLCBkZWx0YSk7XG4gICAgfVxuICAgIHJldHVybiBuZXh0Q2VsbDtcbiAgfVxuXG4gIHByaXZhdGUgX2ZpbmREYXlQYWdlVXBEb3duKFxuICAgIGNlbGxzOiAoSFRNTEJ1dHRvbkVsZW1lbnQgfCBTYmJDYWxlbmRhckRheUVsZW1lbnQpW10sXG4gICAgaW5kZXg6IG51bWJlcixcbiAgICBkYXk6IERheTxUPixcbiAgICBkZWx0YTogbnVtYmVyLFxuICAgIGRlbHRhSWZEaXNhYmxlZDogbnVtYmVyLFxuICApOiBIVE1MQnV0dG9uRWxlbWVudCB8IFNiYkNhbGVuZGFyRGF5RWxlbWVudCB7XG4gICAgY29uc3QgbmV3RGF0ZVZhbHVlID0gdGhpcy5fZGF0ZUFkYXB0ZXIudG9Jc284NjAxKFxuICAgICAgdGhpcy5fZGF0ZUFkYXB0ZXIuYWRkQ2FsZW5kYXJEYXlzKGRheS5kYXRlVmFsdWUsIGRlbHRhKSxcbiAgICApO1xuICAgIGlmICh0aGlzLl9pc0RheU91dE9mVmlldyhuZXdEYXRlVmFsdWUpKSB7XG4gICAgICByZXR1cm4gY2VsbHNbaW5kZXhdO1xuICAgIH1cbiAgICBjb25zdCBuZXh0Q2VsbCA9IGNlbGxzLmZpbmQoKGUpID0+IHRoaXMuX21hcFZhbHVlVG9JU09EYXRlKGUudmFsdWUhKSA9PT0gbmV3RGF0ZVZhbHVlKTtcbiAgICBpZiAoIW5leHRDZWxsIHx8IG5leHRDZWxsLmRpc2FibGVkKSB7XG4gICAgICByZXR1cm4gdGhpcy5fZmluZERheVBhZ2VVcERvd24oY2VsbHMsIGluZGV4LCBkYXksIGRlbHRhICsgZGVsdGFJZkRpc2FibGVkLCBkZWx0YUlmRGlzYWJsZWQpO1xuICAgIH1cbiAgICByZXR1cm4gbmV4dENlbGw7XG4gIH1cblxuICBwcml2YXRlIF9maW5kRGF5Rmlyc3QoXG4gICAgY2VsbHM6IChIVE1MQnV0dG9uRWxlbWVudCB8IFNiYkNhbGVuZGFyRGF5RWxlbWVudClbXSxcbiAgICBpbmRleDogbnVtYmVyLFxuICAgIGRheTogRGF5PFQ+LFxuICAgIGRhdGU6IG51bWJlcixcbiAgKTogSFRNTEJ1dHRvbkVsZW1lbnQgfCBTYmJDYWxlbmRhckRheUVsZW1lbnQge1xuICAgIGNvbnN0IG5ld0RhdGVWYWx1ZSA9IHRoaXMuX2RhdGVBZGFwdGVyLnRvSXNvODYwMShcbiAgICAgIHRoaXMuX2RhdGVBZGFwdGVyLmNyZWF0ZURhdGUoK2RheS55ZWFyVmFsdWUsICtkYXkubW9udGhWYWx1ZSwgZGF0ZSksXG4gICAgKTtcbiAgICBpZiAodGhpcy5faXNEYXlPdXRPZlZpZXcobmV3RGF0ZVZhbHVlKSkge1xuICAgICAgcmV0dXJuIGNlbGxzW2luZGV4XTtcbiAgICB9XG4gICAgY29uc3QgbmV4dENlbGwgPSBjZWxscy5maW5kKChlKSA9PiB0aGlzLl9tYXBWYWx1ZVRvSVNPRGF0ZShlLnZhbHVlISkgPT09IG5ld0RhdGVWYWx1ZSk7XG4gICAgaWYgKCFuZXh0Q2VsbCB8fCBuZXh0Q2VsbC5kaXNhYmxlZCkge1xuICAgICAgcmV0dXJuIHRoaXMuX2ZpbmREYXlGaXJzdChjZWxscywgaW5kZXgsIGRheSwgZGF0ZSArIDEpO1xuICAgIH1cbiAgICByZXR1cm4gbmV4dENlbGw7XG4gIH1cblxuICBwcml2YXRlIF9maW5kRGF5TGFzdChcbiAgICBjZWxsczogKEhUTUxCdXR0b25FbGVtZW50IHwgU2JiQ2FsZW5kYXJEYXlFbGVtZW50KVtdLFxuICAgIGluZGV4OiBudW1iZXIsXG4gICAgZmlyc3ROZXh0TW9udGg6IFQsXG4gICk6IEhUTUxCdXR0b25FbGVtZW50IHwgU2JiQ2FsZW5kYXJEYXlFbGVtZW50IHtcbiAgICBjb25zdCBuZXdEYXRlVmFsdWUgPSB0aGlzLl9kYXRlQWRhcHRlci50b0lzbzg2MDEoXG4gICAgICB0aGlzLl9kYXRlQWRhcHRlci5hZGRDYWxlbmRhckRheXMoZmlyc3ROZXh0TW9udGgsIC0xKSxcbiAgICApO1xuICAgIGlmICh0aGlzLl9pc0RheU91dE9mVmlldyhuZXdEYXRlVmFsdWUpKSB7XG4gICAgICByZXR1cm4gY2VsbHNbaW5kZXhdO1xuICAgIH1cbiAgICBjb25zdCBuZXh0Q2VsbCA9IGNlbGxzLmZpbmQoKGUpID0+IHRoaXMuX21hcFZhbHVlVG9JU09EYXRlKGUudmFsdWUhKSA9PT0gbmV3RGF0ZVZhbHVlKTtcbiAgICBpZiAoIW5leHRDZWxsIHx8IG5leHRDZWxsLmRpc2FibGVkKSB7XG4gICAgICByZXR1cm4gdGhpcy5fZmluZERheUxhc3QoY2VsbHMsIGluZGV4LCB0aGlzLl9kYXRlQWRhcHRlci5kZXNlcmlhbGl6ZShuZXdEYXRlVmFsdWUpISk7XG4gICAgfVxuICAgIHJldHVybiBuZXh0Q2VsbDtcbiAgfVxuXG4gIC8qKlxuICAgKiBHZXRzIHRoZSBpbmRleCBvZiB0aGUgZWxlbWVudCB0byBtb3ZlIHRvLCBiYXNlZCBvbiBhIGxpc3Qgb2YgZWxlbWVudHMgKHdoaWNoIGNhbiBiZSBwb3RlbnRpYWxseSBkaXNhYmxlZCksXG4gICAqIHRoZSBrZXlib2FyZCBpbnB1dCBhbmQgdGhlIHBvc2l0aW9uIG9mIHRoZSBjdXJyZW50IGVsZW1lbnQgaW4gdGhlIGxpc3QuXG4gICAqIEluIHRoZSBkYXkgdmlldywgdGhlIGBkYXk/OiBEYXlgIHBhcmFtZXRlciBpcyBtYW5kYXRvcnkgZm9yIGNhbGN1bGF0aW9uLFxuICAgKiB3aGlsZSBpbiBtb250aCBhbmQgeWVhciB2aWV3IGl0J3Mgbm90IGR1ZSB0byB0aGUgZml4ZWQgYW1vdW50IG9mIHJlbmRlcmVkIGNlbGxzLlxuICAgKi9cbiAgcHJpdmF0ZSBfbmF2aWdhdGVCeUtleWJvYXJkKFxuICAgIGV2dDogS2V5Ym9hcmRFdmVudCxcbiAgICBpbmRleDogbnVtYmVyLFxuICAgIGNlbGxzOiBIVE1MQnV0dG9uRWxlbWVudFtdLFxuICApOiBIVE1MQnV0dG9uRWxlbWVudCB7XG4gICAgY29uc3Qge1xuICAgICAgZWxlbWVudEluZGV4Rm9yV2lkZU1vZGUsXG4gICAgICBvZmZzZXRGb3JXaWRlTW9kZSxcbiAgICAgIGxhc3RFbGVtZW50SW5kZXhGb3JXaWRlTW9kZSxcbiAgICAgIHZlcnRpY2FsT2Zmc2V0LFxuICAgIH06IENhbGVuZGFyS2V5Ym9hcmROYXZpZ2F0aW9uTW9udGhZZWFyVmlld3NQYXJhbWV0ZXJzID1cbiAgICAgIHRoaXMuX2NhbGN1bGF0ZVBhcmFtZXRlcnNGb3JLZXlib2FyZE5hdmlnYXRpb24oaW5kZXgsIHRoaXMuX2NhbGVuZGFyVmlldyA9PT0gJ3llYXInKTtcblxuICAgIHN3aXRjaCAoZXZ0LmtleSkge1xuICAgICAgY2FzZSAnQXJyb3dVcCc6XG4gICAgICAgIHJldHVybiB0aGlzLl9maW5kTmV4dChjZWxscywgaW5kZXgsIC12ZXJ0aWNhbE9mZnNldCk7XG4gICAgICBjYXNlICdBcnJvd0Rvd24nOlxuICAgICAgICByZXR1cm4gdGhpcy5fZmluZE5leHQoY2VsbHMsIGluZGV4LCB2ZXJ0aWNhbE9mZnNldCk7XG4gICAgICBjYXNlICdBcnJvd0xlZnQnOlxuICAgICAgICByZXR1cm4gdGhpcy5fZmluZE5leHQoY2VsbHMsIGluZGV4LCAtMSk7XG4gICAgICBjYXNlICdBcnJvd1JpZ2h0JzpcbiAgICAgICAgcmV0dXJuIHRoaXMuX2ZpbmROZXh0KGNlbGxzLCBpbmRleCwgMSk7XG4gICAgICBjYXNlICdIb21lJzpcbiAgICAgICAgcmV0dXJuIHRoaXMuX2ZpbmRGaXJzdChjZWxscywgb2Zmc2V0Rm9yV2lkZU1vZGUpO1xuICAgICAgY2FzZSAnUGFnZVVwJzpcbiAgICAgICAgcmV0dXJuIHRoaXMuX2ZpbmRGaXJzdE9uQ29sdW1uKFxuICAgICAgICAgIGNlbGxzLFxuICAgICAgICAgIGVsZW1lbnRJbmRleEZvcldpZGVNb2RlLFxuICAgICAgICAgIG9mZnNldEZvcldpZGVNb2RlLFxuICAgICAgICAgIHZlcnRpY2FsT2Zmc2V0LFxuICAgICAgICApO1xuICAgICAgY2FzZSAnUGFnZURvd24nOlxuICAgICAgICByZXR1cm4gdGhpcy5fZmluZExhc3RPbkNvbHVtbihjZWxscywgaW5kZXgsIGxhc3RFbGVtZW50SW5kZXhGb3JXaWRlTW9kZSwgdmVydGljYWxPZmZzZXQpO1xuICAgICAgY2FzZSAnRW5kJzpcbiAgICAgICAgcmV0dXJuIHRoaXMuX2ZpbmRMYXN0KGNlbGxzLCBsYXN0RWxlbWVudEluZGV4Rm9yV2lkZU1vZGUgLSAxKTtcbiAgICAgIGRlZmF1bHQ6XG4gICAgICAgIHJldHVybiBjZWxsc1tpbmRleF07XG4gICAgfVxuICB9XG5cbiAgLyoqXG4gICAqIENhbGN1bGF0ZXMgdGhlIHBhcmFtZXRlcnMgbmVlZGVkIGluIGtleWJvYXJkIG5hdmlnYXRpb24gaW4geWVhciBhbmQgbW9udGggdmlldy5cbiAgICogQHBhcmFtIGluZGV4IFRoZSBzdGFydGluZyBlbGVtZW50J3MgaW5kZXggaW4gdGhlIGNlbGwgYXJyYXkuXG4gICAqIEBwYXJhbSBpc1llYXJWaWV3IFdoZXRoZXIgdGhlIGRpc3BsYXllZCBgdmlld2AgaXMgdGhlIHllYXIgb25lLlxuICAgKi9cbiAgcHJpdmF0ZSBfY2FsY3VsYXRlUGFyYW1ldGVyc0ZvcktleWJvYXJkTmF2aWdhdGlvbihcbiAgICBpbmRleDogbnVtYmVyLFxuICAgIGlzWWVhclZpZXc6IGJvb2xlYW4sXG4gICk6IENhbGVuZGFyS2V5Ym9hcmROYXZpZ2F0aW9uTW9udGhZZWFyVmlld3NQYXJhbWV0ZXJzIHtcbiAgICBjb25zdCBlbGVtZW50c1BlclBhZ2UgPSBpc1llYXJWaWV3ID8gWUVBUlNfUEVSX1BBR0UgOiBNT05USFNfUEVSX1BBR0U7XG4gICAgY29uc3Qgb2Zmc2V0OiBudW1iZXIgPSBNYXRoLnRydW5jKGluZGV4IC8gZWxlbWVudHNQZXJQYWdlKSAqIGVsZW1lbnRzUGVyUGFnZTtcbiAgICBjb25zdCBpbmRleEluVmlldzogbnVtYmVyID0gb2Zmc2V0ID09PSAwID8gaW5kZXggOiBpbmRleCAtIGVsZW1lbnRzUGVyUGFnZTtcbiAgICByZXR1cm4ge1xuICAgICAgdmVydGljYWxPZmZzZXQ6IGlzWWVhclZpZXcgPyBZRUFSU19QRVJfUk9XIDogTU9OVEhTX1BFUl9ST1csXG4gICAgICBlbGVtZW50SW5kZXhGb3JXaWRlTW9kZTogaW5kZXhJblZpZXcsXG4gICAgICBvZmZzZXRGb3JXaWRlTW9kZTogaW5kZXggLSBpbmRleEluVmlldyxcbiAgICAgIGxhc3RFbGVtZW50SW5kZXhGb3JXaWRlTW9kZTogb2Zmc2V0ID09PSAwID8gZWxlbWVudHNQZXJQYWdlIDogZWxlbWVudHNQZXJQYWdlICogMixcbiAgICB9O1xuICB9XG5cbiAgLyoqXG4gICAqIEdldHMgdGhlIG5leHQgZWxlbWVudCBvZiB0aGUgcHJvdmlkZWQgYXJyYXkgc3RhcnRpbmcgZnJvbSBgaW5kZXhgIGJ5IGFkZGluZyBgZGVsdGFgLlxuICAgKiBJZiB0aGUgZm91bmQgZWxlbWVudCBpcyBkaXNhYmxlZCwgaXQgY29udGludWVzIGFkZGluZyBgZGVsdGFgIHVudGlsIGl0IGZpbmRzIGFuIGVuYWJsZWQgb25lIGluIHRoZSBhcnJheSBib3VuZHMuXG4gICAqL1xuICBwcml2YXRlIF9maW5kTmV4dChkYXlzOiBIVE1MQnV0dG9uRWxlbWVudFtdLCBpbmRleDogbnVtYmVyLCBkZWx0YTogbnVtYmVyKTogSFRNTEJ1dHRvbkVsZW1lbnQge1xuICAgIGxldCBuZXh0SW5kZXggPSBpbmRleCArIGRlbHRhO1xuICAgIHdoaWxlIChuZXh0SW5kZXggPCBkYXlzLmxlbmd0aCAmJiBkYXlzW25leHRJbmRleF0/LmRpc2FibGVkKSB7XG4gICAgICBuZXh0SW5kZXggKz0gZGVsdGE7XG4gICAgfVxuICAgIHJldHVybiBkYXlzW25leHRJbmRleF0gPz8gZGF5c1tpbmRleF07XG4gIH1cblxuICAvKiogRmluZCB0aGUgZmlyc3QgZW5hYmxlZCBlbGVtZW50IGluIHRoZSBwcm92aWRlZCBhcnJheS4gKi9cbiAgcHJpdmF0ZSBfZmluZEZpcnN0KGRheXM6IEhUTUxCdXR0b25FbGVtZW50W10sIGZpcnN0T2ZDdXJyZW50TW9udGg6IG51bWJlcik6IEhUTUxCdXR0b25FbGVtZW50IHtcbiAgICByZXR1cm4gIWRheXNbZmlyc3RPZkN1cnJlbnRNb250aF0uZGlzYWJsZWRcbiAgICAgID8gZGF5c1tmaXJzdE9mQ3VycmVudE1vbnRoXVxuICAgICAgOiB0aGlzLl9maW5kTmV4dChkYXlzLCBmaXJzdE9mQ3VycmVudE1vbnRoLCAxKTtcbiAgfVxuXG4gIC8qKiBGaW5kIHRoZSBsYXN0IGVuYWJsZWQgZWxlbWVudCBpbiB0aGUgcHJvdmlkZWQgYXJyYXkuICovXG4gIHByaXZhdGUgX2ZpbmRMYXN0KGRheXM6IEhUTUxCdXR0b25FbGVtZW50W10sIGxhc3RPZkN1cnJlbnRNb250aDogbnVtYmVyKTogSFRNTEJ1dHRvbkVsZW1lbnQge1xuICAgIHJldHVybiAhZGF5c1tsYXN0T2ZDdXJyZW50TW9udGhdLmRpc2FibGVkXG4gICAgICA/IGRheXNbbGFzdE9mQ3VycmVudE1vbnRoXVxuICAgICAgOiB0aGlzLl9maW5kTmV4dChkYXlzLCBsYXN0T2ZDdXJyZW50TW9udGgsIC0xKTtcbiAgfVxuXG4gIC8qKiBGaW5kIHRoZSBmaXJzdCBlbmFibGVkIGVsZW1lbnQgaW4gdGhlIHNhbWUgY29sdW1uIG9mIHRoZSBwcm92aWRlZCBhcnJheS4gKi9cbiAgcHJpdmF0ZSBfZmluZEZpcnN0T25Db2x1bW4oXG4gICAgZGF5czogSFRNTEJ1dHRvbkVsZW1lbnRbXSxcbiAgICBpbmRleDogbnVtYmVyLFxuICAgIG9mZnNldDogbnVtYmVyLFxuICAgIHZlcnRpY2FsT2Zmc2V0OiBudW1iZXIsXG4gICk6IEhUTUxCdXR0b25FbGVtZW50IHtcbiAgICBjb25zdCBuZXh0SW5kZXggPSAoaW5kZXggJSB2ZXJ0aWNhbE9mZnNldCkgKyBvZmZzZXQ7XG4gICAgcmV0dXJuICFkYXlzW25leHRJbmRleF0uZGlzYWJsZWRcbiAgICAgID8gZGF5c1tuZXh0SW5kZXhdXG4gICAgICA6IHRoaXMuX2ZpbmROZXh0KGRheXMsIG5leHRJbmRleCwgdmVydGljYWxPZmZzZXQpO1xuICB9XG5cbiAgLyoqIEZpbmQgdGhlIGxhc3QgZW5hYmxlZCBlbGVtZW50IGluIHRoZSBzYW1lIGNvbHVtbiBvZiB0aGUgcHJvdmlkZWQgYXJyYXkuICovXG4gIHByaXZhdGUgX2ZpbmRMYXN0T25Db2x1bW4oXG4gICAgZGF5czogSFRNTEJ1dHRvbkVsZW1lbnRbXSxcbiAgICBpbmRleDogbnVtYmVyLFxuICAgIG9mZnNldDogbnVtYmVyLFxuICAgIHZlcnRpY2FsT2Zmc2V0OiBudW1iZXIsXG4gICk6IEhUTUxCdXR0b25FbGVtZW50IHtcbiAgICBjb25zdCBuZXh0SW5kZXggPSBpbmRleCArIE1hdGgudHJ1bmMoKG9mZnNldCAtIGluZGV4IC0gMSkgLyB2ZXJ0aWNhbE9mZnNldCkgKiB2ZXJ0aWNhbE9mZnNldDtcbiAgICByZXR1cm4gIWRheXNbbmV4dEluZGV4XS5kaXNhYmxlZFxuICAgICAgPyBkYXlzW25leHRJbmRleF1cbiAgICAgIDogdGhpcy5fZmluZE5leHQoZGF5cywgbmV4dEluZGV4LCAtdmVydGljYWxPZmZzZXQpO1xuICB9XG5cbiAgcHJpdmF0ZSBfcmVzZXRDYWxlbmRhclZpZXdBbmRFbWl0TW9udGhDaGFuZ2UoaW5pdFRyYW5zaXRpb24gPSBmYWxzZSk6IHZvaWQge1xuICAgIHRoaXMuX3Jlc2V0Q2FsZW5kYXJWaWV3KGluaXRUcmFuc2l0aW9uKTtcbiAgICB0aGlzLl9lbWl0TW9udGhDaGFuZ2UoKTtcbiAgfVxuXG4gIHByaXZhdGUgX3Jlc2V0Q2FsZW5kYXJWaWV3KGluaXRUcmFuc2l0aW9uID0gZmFsc2UpOiB2b2lkIHtcbiAgICBpZiAodGhpcy5fY29udGFpbmluZ0ZvY3VzKSB7XG4gICAgICB0aGlzLl9yZXNldEZvY3VzID0gdHJ1ZTtcbiAgICB9XG4gICAgdGhpcy5fYWN0aXZlRGF0ZSA9XG4gICAgICAodGhpcy5tdWx0aXBsZSA/ICh0aGlzLl9zZWxlY3RlZCBhcyBUW10pLmF0KC0xKSA6ICh0aGlzLl9zZWxlY3RlZCBhcyBUKSkgPz9cbiAgICAgIHRoaXMuX2RhdGVBZGFwdGVyLnRvZGF5KCk7XG4gICAgdGhpcy5fc2V0Q2hvc2VuWWVhcigpO1xuICAgIHRoaXMuX2Nob3Nlbk1vbnRoID0gdW5kZWZpbmVkO1xuICAgIHRoaXMuX2luaXQoKTtcbiAgICB0aGlzLl9uZXh0Q2FsZW5kYXJWaWV3ID0gdGhpcy5fY2FsZW5kYXJWaWV3ID0gdGhpcy52aWV3O1xuXG4gICAgaWYgKGluaXRUcmFuc2l0aW9uKSB7XG4gICAgICB0aGlzLl9zdGFydFRhYmxlVHJhbnNpdGlvbigpO1xuICAgIH1cbiAgfVxuXG4gIC8qKiBSZW5kZXIgdGhlIHZpZXcgZm9yIHRoZSBkYXkgc2VsZWN0aW9uLiAqL1xuICBwcml2YXRlIF9yZW5kZXJEYXlWaWV3KCk6IFRlbXBsYXRlUmVzdWx0IHtcbiAgICBjb25zdCBuZXh0TW9udGhBY3RpdmVEYXRlID0gdGhpcy5fd2lkZVxuICAgICAgPyB0aGlzLl9kYXRlQWRhcHRlci5hZGRDYWxlbmRhck1vbnRocyh0aGlzLl9hY3RpdmVEYXRlLCAxKVxuICAgICAgOiB1bmRlZmluZWQ7XG4gICAgcmV0dXJuIGh0bWxgXG4gICAgICA8ZGl2IGNsYXNzPVwic2JiLWNhbGVuZGFyX19jb250cm9sc1wiPlxuICAgICAgICAke3RoaXMuX2dldEFycm93KFxuICAgICAgICAgICdsZWZ0JyxcbiAgICAgICAgICAoKSA9PiB0aGlzLl9nb1RvRGlmZmVyZW50TW9udGgoLTEpLFxuICAgICAgICAgIGkxOG5QcmV2aW91c01vbnRoW3RoaXMuX2xhbmd1YWdlLmN1cnJlbnRdLFxuICAgICAgICAgIHRoaXMuX3ByZXZpb3VzTW9udGhEaXNhYmxlZCgpLFxuICAgICAgICApfVxuICAgICAgICA8ZGl2IGNsYXNzPVwic2JiLWNhbGVuZGFyX19jb250cm9scy1tb250aFwiPlxuICAgICAgICAgICR7dGhpcy5fY3JlYXRlTGFiZWxGb3JEYXlWaWV3KHRoaXMuX2FjdGl2ZURhdGUpfVxuICAgICAgICAgICR7dGhpcy5fd2lkZSA/IHRoaXMuX2NyZWF0ZUxhYmVsRm9yRGF5VmlldyhuZXh0TW9udGhBY3RpdmVEYXRlISkgOiBub3RoaW5nfVxuICAgICAgICAgIDxzYmItc2NyZWVuLXJlYWRlci1vbmx5IHJvbGU9XCJzdGF0dXNcIj5cbiAgICAgICAgICAgICR7dGhpcy5fY3JlYXRlQXJpYUxhYmVsRm9yRGF5Vmlldyh0aGlzLl9hY3RpdmVEYXRlLCBuZXh0TW9udGhBY3RpdmVEYXRlISl9XG4gICAgICAgICAgPC9zYmItc2NyZWVuLXJlYWRlci1vbmx5PlxuICAgICAgICA8L2Rpdj5cbiAgICAgICAgJHt0aGlzLl9nZXRBcnJvdyhcbiAgICAgICAgICAncmlnaHQnLFxuICAgICAgICAgICgpID0+IHRoaXMuX2dvVG9EaWZmZXJlbnRNb250aCgxKSxcbiAgICAgICAgICBpMThuTmV4dE1vbnRoW3RoaXMuX2xhbmd1YWdlLmN1cnJlbnRdLFxuICAgICAgICAgIHRoaXMuX25leHRNb250aERpc2FibGVkKCksXG4gICAgICAgICl9XG4gICAgICA8L2Rpdj5cbiAgICAgIDxkaXYgY2xhc3M9XCJzYmItY2FsZW5kYXJfX3RhYmxlLW92ZXJmbG93LWJyZWFrXCI+XG4gICAgICAgIDxkaXYgY2xhc3M9XCJzYmItY2FsZW5kYXJfX3RhYmxlLWNvbnRhaW5lciBzYmItY2FsZW5kYXJfX3RhYmxlLWRheS12aWV3XCI+XG4gICAgICAgICAgJHt0aGlzLm9yaWVudGF0aW9uID09PSAnaG9yaXpvbnRhbCdcbiAgICAgICAgICAgID8gaHRtbGBcbiAgICAgICAgICAgICAgICAke3RoaXMuX2NyZWF0ZURheVRhYmxlKHRoaXMuX3dlZWtzLCB0aGlzLl93ZWVrTnVtYmVycyl9XG4gICAgICAgICAgICAgICAgJHt0aGlzLl93aWRlXG4gICAgICAgICAgICAgICAgICA/IHRoaXMuX2NyZWF0ZURheVRhYmxlKHRoaXMuX25leHRNb250aFdlZWtzLCB0aGlzLl9uZXh0TW9udGhXZWVrTnVtYmVycywgdHJ1ZSlcbiAgICAgICAgICAgICAgICAgIDogbm90aGluZ31cbiAgICAgICAgICAgICAgYFxuICAgICAgICAgICAgOiBodG1sYFxuICAgICAgICAgICAgICAgICR7dGhpcy5fY3JlYXRlRGF5VGFibGVWZXJ0aWNhbCh0aGlzLl93ZWVrcywgdGhpcy5fd2Vla051bWJlcnMpfVxuICAgICAgICAgICAgICAgICR7dGhpcy5fd2lkZVxuICAgICAgICAgICAgICAgICAgPyB0aGlzLl9jcmVhdGVEYXlUYWJsZVZlcnRpY2FsKFxuICAgICAgICAgICAgICAgICAgICAgIHRoaXMuX25leHRNb250aFdlZWtzLFxuICAgICAgICAgICAgICAgICAgICAgIHRoaXMuX25leHRNb250aFdlZWtOdW1iZXJzLFxuICAgICAgICAgICAgICAgICAgICAgIG5leHRNb250aEFjdGl2ZURhdGUsXG4gICAgICAgICAgICAgICAgICAgIClcbiAgICAgICAgICAgICAgICAgIDogbm90aGluZ31cbiAgICAgICAgICAgICAgYH1cbiAgICAgICAgPC9kaXY+XG4gICAgICA8L2Rpdj5cbiAgICBgO1xuICB9XG5cbiAgLyoqIENyZWF0ZXMgdGhlIGxhYmVsIHdpdGggdGhlIG1vbnRoIGZvciB0aGUgZGFpbHkgdmlldy4gKi9cbiAgcHJpdmF0ZSBfY3JlYXRlTGFiZWxGb3JEYXlWaWV3KGQ6IFQpOiBUZW1wbGF0ZVJlc3VsdCB7XG4gICAgY29uc3QgbW9udGhMYWJlbCA9IGAke1xuICAgICAgdGhpcy5fbW9udGhOYW1lc1t0aGlzLl9kYXRlQWRhcHRlci5nZXRNb250aChkKSAtIDFdXG4gICAgfSAke3RoaXMuX2RhdGVBZGFwdGVyLmdldFllYXIoZCl9YDtcbiAgICByZXR1cm4gaHRtbGBcbiAgICAgIDxidXR0b25cbiAgICAgICAgdHlwZT1cImJ1dHRvblwiXG4gICAgICAgIGNsYXNzPVwic2JiLWNhbGVuZGFyX19kYXRlLXNlbGVjdGlvbiBzYmItY2FsZW5kYXJfX2NvbnRyb2xzLWNoYW5nZS1kYXRlXCJcbiAgICAgICAgYXJpYS1sYWJlbD1cIiR7aTE4blllYXJNb250aFNlbGVjdGlvblt0aGlzLl9sYW5ndWFnZS5jdXJyZW50XX0gJHttb250aExhYmVsfVwiXG4gICAgICAgIEBjbGljaz0keygpID0+IHtcbiAgICAgICAgICB0aGlzLl9yZXNldEZvY3VzID0gdHJ1ZTtcbiAgICAgICAgICB0aGlzLl9uZXh0Q2FsZW5kYXJWaWV3ID0gJ3llYXInO1xuICAgICAgICAgIHRoaXMuX3N0YXJ0VGFibGVUcmFuc2l0aW9uKCk7XG4gICAgICAgIH19XG4gICAgICA+XG4gICAgICAgICR7bW9udGhMYWJlbH1cbiAgICAgICAgPHNiYi1pY29uIG5hbWU9XCJjaGV2cm9uLXNtYWxsLWRvd24tc21hbGxcIj48L3NiYi1pY29uPlxuICAgICAgPC9idXR0b24+XG4gICAgYDtcbiAgfVxuXG4gIC8qKiBDcmVhdGVzIHRoZSBhcmlhLWxhYmVsIGZvciB0aGUgZGFpbHkgdmlldy4gKi9cbiAgcHJpdmF0ZSBfY3JlYXRlQXJpYUxhYmVsRm9yRGF5VmlldyguLi5kYXRlczogVFtdKTogc3RyaW5nIHtcbiAgICBsZXQgbW9udGhMYWJlbCA9ICcnO1xuICAgIGZvciAoY29uc3QgZCBvZiBkYXRlcykge1xuICAgICAgaWYgKGQpIHtcbiAgICAgICAgbW9udGhMYWJlbCArPSBgJHtcbiAgICAgICAgICB0aGlzLl9tb250aE5hbWVzW3RoaXMuX2RhdGVBZGFwdGVyLmdldE1vbnRoKGQpIC0gMV1cbiAgICAgICAgfSAke3RoaXMuX2RhdGVBZGFwdGVyLmdldFllYXIoZCl9IGA7XG4gICAgICB9XG4gICAgfVxuICAgIHJldHVybiBtb250aExhYmVsO1xuICB9XG5cbiAgLyoqIENyZWF0ZXMgdGhlIGNhbGVuZGFyIHRhYmxlIGZvciB0aGUgZGFpbHkgdmlldy4gKi9cbiAgcHJpdmF0ZSBfY3JlYXRlRGF5VGFibGUoXG4gICAgd2Vla3M6IERheTxUPltdW10sXG4gICAgd2Vla051bWJlcnM6IG51bWJlcltdLFxuICAgIGlzV2lkZU5leHRNb250aDogYm9vbGVhbiA9IGZhbHNlLFxuICApOiBUZW1wbGF0ZVJlc3VsdCB7XG4gICAgY29uc3Qgd2Vla3NGb3JTZWxlY3RNdWx0aXBsZVdlZWtOdW1iZXJzOiBEYXk8VD5bXSA9IChcbiAgICAgIHRoaXMuX3dpZGVcbiAgICAgICAgPyBbLi4udGhpcy5fd2Vla3MsIC4uLnRoaXMuX25leHRNb250aFdlZWtzXVxuICAgICAgICA6IGlzV2lkZU5leHRNb250aFxuICAgICAgICAgID8gdGhpcy5fbmV4dE1vbnRoV2Vla3NcbiAgICAgICAgICA6IHRoaXMuX3dlZWtzXG4gICAgKS5mbGF0KCk7XG4gICAgY29uc3Qgd2Vla3NGb3JTZWxlY3RNdWx0aXBsZVdlZWtEYXlzOiBEYXk8VD5bXSA9IChcbiAgICAgIGlzV2lkZU5leHRNb250aCA/IHRoaXMuX25leHRNb250aFdlZWtzIDogdGhpcy5fd2Vla3NcbiAgICApLmZsYXQoKTtcbiAgICByZXR1cm4gaHRtbGBcbiAgICAgIDx0YWJsZVxuICAgICAgICBjbGFzcz1cInNiYi1jYWxlbmRhcl9fdGFibGVcIlxuICAgICAgICBAZm9jdXNvdXQ9JHsoZXZlbnQ6IEZvY3VzRXZlbnQpID0+XG4gICAgICAgICAgdGhpcy5faGFuZGxlVGFibGVCbHVyKGV2ZW50LnJlbGF0ZWRUYXJnZXQgYXMgSFRNTEVsZW1lbnQpfVxuICAgICAgICBAYW5pbWF0aW9uZW5kPSR7KGU6IEFuaW1hdGlvbkV2ZW50KSA9PiB0aGlzLl90YWJsZUFuaW1hdGlvbkVuZChlKX1cbiAgICAgID5cbiAgICAgICAgPHRoZWFkIGNsYXNzPVwic2JiLWNhbGVuZGFyX190YWJsZS1oZWFkZXJcIj5cbiAgICAgICAgICA8dHI+XG4gICAgICAgICAgICAke3RoaXMud2Vla051bWJlcnMgPyBodG1sYDx0aCBjbGFzcz1cInNiYi1jYWxlbmRhcl9fdGFibGUtaGVhZGVyLWNlbGxcIj48L3RoPmAgOiBub3RoaW5nfVxuICAgICAgICAgICAgJHt0aGlzLl93ZWVrZGF5cy5tYXAoXG4gICAgICAgICAgICAgIChkYXk6IFdlZWtkYXksIGluZGV4OiBudW1iZXIpID0+IGh0bWxgXG4gICAgICAgICAgICAgICAgPHRoIGNsYXNzPVwic2JiLWNhbGVuZGFyX190YWJsZS1oZWFkZXItY2VsbFwiPlxuICAgICAgICAgICAgICAgICAgJHt0aGlzLm11bHRpcGxlXG4gICAgICAgICAgICAgICAgICAgID8gaHRtbGBcbiAgICAgICAgICAgICAgICAgICAgICAgIDxidXR0b25cbiAgICAgICAgICAgICAgICAgICAgICAgICAgY2xhc3M9XCJzYmItY2FsZW5kYXJfX2hlYWRlci1jZWxsIHNiYi1jYWxlbmRhcl9fd2Vla2RheVwiXG4gICAgICAgICAgICAgICAgICAgICAgICAgIGFyaWEtbGFiZWw9JHtkYXkubG9uZ31cbiAgICAgICAgICAgICAgICAgICAgICAgICAgQGNsaWNrPSR7KCkgPT4ge1xuICAgICAgICAgICAgICAgICAgICAgICAgICAgIC8vIE5PVEU6IFN1bmRheXMgaGF2ZSBpbmRleCA3LCB3aGlsZSB0aGVpciB3ZWVrRGF5VmFsdWUgaXMgMFxuICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNvbnN0IGRheXM6IERheTxUPltdID0gd2Vla3NGb3JTZWxlY3RNdWx0aXBsZVdlZWtEYXlzLmZpbHRlcihcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIChkYXk6IERheTxUPikgPT4gZGF5LndlZWtEYXlWYWx1ZSA9PT0gKGluZGV4ICsgMSkgJSA3LFxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICkhO1xuICAgICAgICAgICAgICAgICAgICAgICAgICAgIHRoaXMuX3NlbGVjdE11bHRpcGxlRGF0ZXMoZGF5cyk7XG4gICAgICAgICAgICAgICAgICAgICAgICAgIH19XG4gICAgICAgICAgICAgICAgICAgICAgICA+XG4gICAgICAgICAgICAgICAgICAgICAgICAgICR7ZGF5Lm5hcnJvd31cbiAgICAgICAgICAgICAgICAgICAgICAgIDwvYnV0dG9uPlxuICAgICAgICAgICAgICAgICAgICAgIGBcbiAgICAgICAgICAgICAgICAgICAgOiBodG1sYFxuICAgICAgICAgICAgICAgICAgICAgICAgPHNiYi1zY3JlZW4tcmVhZGVyLW9ubHk+JHtkYXkubG9uZ308L3NiYi1zY3JlZW4tcmVhZGVyLW9ubHk+XG4gICAgICAgICAgICAgICAgICAgICAgICA8c3BhbiBhcmlhLWhpZGRlbj1cInRydWVcIj4ke2RheS5uYXJyb3d9PC9zcGFuPlxuICAgICAgICAgICAgICAgICAgICAgIGB9XG4gICAgICAgICAgICAgICAgPC90aD5cbiAgICAgICAgICAgICAgYCxcbiAgICAgICAgICAgICl9XG4gICAgICAgICAgPC90cj5cbiAgICAgICAgPC90aGVhZD5cbiAgICAgICAgPHRib2R5IGNsYXNzPVwic2JiLWNhbGVuZGFyX190YWJsZS1ib2R5XCI+XG4gICAgICAgICAgJHt3ZWVrcy5tYXAoKHdlZWs6IERheTxUPltdLCByb3dJbmRleDogbnVtYmVyKSA9PiB7XG4gICAgICAgICAgICBjb25zdCBmaXJzdFJvd09mZnNldDogbnVtYmVyID0gREFZU19QRVJfUk9XIC0gd2Vlay5sZW5ndGg7XG4gICAgICAgICAgICBpZiAocm93SW5kZXggPT09IDAgJiYgZmlyc3RSb3dPZmZzZXQpIHtcbiAgICAgICAgICAgICAgcmV0dXJuIGh0bWxgXG4gICAgICAgICAgICAgICAgPHRyPlxuICAgICAgICAgICAgICAgICAgJHt0aGlzLndlZWtOdW1iZXJzXG4gICAgICAgICAgICAgICAgICAgID8gaHRtbGBcbiAgICAgICAgICAgICAgICAgICAgICAgIDx0ZCBjbGFzcz1cInNiYi1jYWxlbmRhcl9fdGFibGUtaGVhZGVyLWNlbGwtdmVydGljYWxcIj5cbiAgICAgICAgICAgICAgICAgICAgICAgICAgJHt0aGlzLm11bHRpcGxlXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgPyBodG1sYFxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICA8YnV0dG9uXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgY2xhc3M9XCJzYmItY2FsZW5kYXJfX2hlYWRlci1jZWxsIHNiYi1jYWxlbmRhcl9fd2Vla2RheVwiXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgYXJpYS1sYWJlbD0ke2Ake2kxOG5DYWxlbmRhcldlZWtOdW1iZXJbdGhpcy5fbGFuZ3VhZ2UuY3VycmVudF19ICR7d2Vla051bWJlcnNbMF19YH1cbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBAY2xpY2s9JHsoKSA9PiB7XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBjb25zdCBkYXlzOiBEYXk8VD5bXSA9IHdlZWtzRm9yU2VsZWN0TXVsdGlwbGVXZWVrTnVtYmVycy5maWx0ZXIoXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIChkYXk6IERheTxUPikgPT4gZGF5LndlZWtWYWx1ZSA9PT0gd2Vla051bWJlcnNbMF0sXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICApITtcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHRoaXMuX3NlbGVjdE11bHRpcGxlRGF0ZXMoZGF5cyk7XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgfX1cbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPlxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICR7d2Vla051bWJlcnNbMF19XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIDwvYnV0dG9uPlxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgYFxuICAgICAgICAgICAgICAgICAgICAgICAgICAgIDogaHRtbGBcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPHNiYi1zY3JlZW4tcmVhZGVyLW9ubHlcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICA+JHtgJHtpMThuQ2FsZW5kYXJXZWVrTnVtYmVyW3RoaXMuX2xhbmd1YWdlLmN1cnJlbnRdfSAke3dlZWtOdW1iZXJzWzBdfWB9PC9zYmItc2NyZWVuLXJlYWRlci1vbmx5XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID5cbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPHNwYW4gYXJpYS1oaWRkZW49XCJ0cnVlXCI+JHt3ZWVrTnVtYmVyc1swXX08L3NwYW4+XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICBgfVxuICAgICAgICAgICAgICAgICAgICAgICAgPC90ZD5cbiAgICAgICAgICAgICAgICAgICAgICBgXG4gICAgICAgICAgICAgICAgICAgIDogbm90aGluZ31cbiAgICAgICAgICAgICAgICAgICR7Wy4uLkFycmF5KGZpcnN0Um93T2Zmc2V0KS5rZXlzKCldLm1hcChcbiAgICAgICAgICAgICAgICAgICAgKCkgPT4gaHRtbGA8dGQgY2xhc3M9XCJzYmItY2FsZW5kYXJfX3RhYmxlLWRhdGFcIj48L3RkPmAsXG4gICAgICAgICAgICAgICAgICApfVxuICAgICAgICAgICAgICAgICAgJHt0aGlzLl9jcmVhdGVEYXlDZWxscyh3ZWVrKX1cbiAgICAgICAgICAgICAgICA8L3RyPlxuICAgICAgICAgICAgICBgO1xuICAgICAgICAgICAgfVxuICAgICAgICAgICAgcmV0dXJuIGh0bWxgXG4gICAgICAgICAgICAgIDx0cj5cbiAgICAgICAgICAgICAgICAke3RoaXMud2Vla051bWJlcnNcbiAgICAgICAgICAgICAgICAgID8gaHRtbGBcbiAgICAgICAgICAgICAgICAgICAgICA8dGQgY2xhc3M9XCJzYmItY2FsZW5kYXJfX3RhYmxlLWhlYWRlci1jZWxsLXZlcnRpY2FsXCI+XG4gICAgICAgICAgICAgICAgICAgICAgICAke3RoaXMubXVsdGlwbGVcbiAgICAgICAgICAgICAgICAgICAgICAgICAgPyBodG1sYFxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPGJ1dHRvblxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBjbGFzcz1cInNiYi1jYWxlbmRhcl9faGVhZGVyLWNlbGwgc2JiLWNhbGVuZGFyX193ZWVrZGF5XCJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgYXJpYS1sYWJlbD0ke2Ake2kxOG5DYWxlbmRhcldlZWtOdW1iZXJbdGhpcy5fbGFuZ3VhZ2UuY3VycmVudF19ICR7d2Vla051bWJlcnNbcm93SW5kZXhdfWB9XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIEBjbGljaz0keygpID0+IHtcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBjb25zdCBkYXlzOiBEYXk8VD5bXSA9IHdlZWtzRm9yU2VsZWN0TXVsdGlwbGVXZWVrTnVtYmVycy5maWx0ZXIoXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAoZGF5OiBEYXk8VD4pID0+IGRheS53ZWVrVmFsdWUgPT09IHdlZWtOdW1iZXJzW3Jvd0luZGV4XSxcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICApITtcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB0aGlzLl9zZWxlY3RNdWx0aXBsZURhdGVzKGRheXMpO1xuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB9fVxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPlxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAke3dlZWtOdW1iZXJzW3Jvd0luZGV4XX1cbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIDwvYnV0dG9uPlxuICAgICAgICAgICAgICAgICAgICAgICAgICAgIGBcbiAgICAgICAgICAgICAgICAgICAgICAgICAgOiBodG1sYFxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPHNiYi1zY3JlZW4tcmVhZGVyLW9ubHlcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPiR7YCR7aTE4bkNhbGVuZGFyV2Vla051bWJlclt0aGlzLl9sYW5ndWFnZS5jdXJyZW50XX0gJHt3ZWVrTnVtYmVyc1tyb3dJbmRleF19YH08L3NiYi1zY3JlZW4tcmVhZGVyLW9ubHlcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID5cbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIDxzcGFuIGFyaWEtaGlkZGVuPVwidHJ1ZVwiPiR7d2Vla051bWJlcnNbcm93SW5kZXhdfTwvc3Bhbj5cbiAgICAgICAgICAgICAgICAgICAgICAgICAgICBgfVxuICAgICAgICAgICAgICAgICAgICAgIDwvdGQ+XG4gICAgICAgICAgICAgICAgICAgIGBcbiAgICAgICAgICAgICAgICAgIDogbm90aGluZ31cbiAgICAgICAgICAgICAgICAke3RoaXMuX2NyZWF0ZURheUNlbGxzKHdlZWspfVxuICAgICAgICAgICAgICA8L3RyPlxuICAgICAgICAgICAgYDtcbiAgICAgICAgICB9KX1cbiAgICAgICAgPC90Ym9keT5cbiAgICAgIDwvdGFibGU+XG4gICAgYDtcbiAgfVxuXG4gIC8qKiBDcmVhdGVzIHRoZSB0YWJsZSBpbiBvcmllbnRhdGlvbj0ndmVydGljYWwnLiAqL1xuICBwcml2YXRlIF9jcmVhdGVEYXlUYWJsZVZlcnRpY2FsKFxuICAgIHdlZWtzOiBEYXk8VD5bXVtdLFxuICAgIHdlZWtOdW1iZXJzOiBudW1iZXJbXSxcbiAgICBuZXh0TW9udGhBY3RpdmVEYXRlPzogVCxcbiAgKTogVGVtcGxhdGVSZXN1bHQge1xuICAgIGNvbnN0IHdlZWtPZmZzZXQgPSB0aGlzLl9kYXRlQWRhcHRlci5nZXRGaXJzdFdlZWtPZmZzZXQoXG4gICAgICBuZXh0TW9udGhBY3RpdmVEYXRlID8/IHRoaXMuX2FjdGl2ZURhdGUsXG4gICAgKTtcbiAgICBjb25zdCB3ZWVrc0ZvclNlbGVjdE11bHRpcGxlV2Vla051bWJlcnM6IERheTxUPltdID0gKFxuICAgICAgdGhpcy5fd2lkZVxuICAgICAgICA/IFsuLi50aGlzLl93ZWVrcywgLi4udGhpcy5fbmV4dE1vbnRoV2Vla3NdXG4gICAgICAgIDogbmV4dE1vbnRoQWN0aXZlRGF0ZVxuICAgICAgICAgID8gdGhpcy5fbmV4dE1vbnRoV2Vla3NcbiAgICAgICAgICA6IHRoaXMuX3dlZWtzXG4gICAgKS5mbGF0KCk7XG4gICAgcmV0dXJuIGh0bWxgXG4gICAgICA8dGFibGVcbiAgICAgICAgY2xhc3M9XCJzYmItY2FsZW5kYXJfX3RhYmxlXCJcbiAgICAgICAgQGZvY3Vzb3V0PSR7KGV2ZW50OiBGb2N1c0V2ZW50KSA9PlxuICAgICAgICAgIHRoaXMuX2hhbmRsZVRhYmxlQmx1cihldmVudC5yZWxhdGVkVGFyZ2V0IGFzIEhUTUxFbGVtZW50KX1cbiAgICAgICAgQGFuaW1hdGlvbmVuZD0keyhlOiBBbmltYXRpb25FdmVudCkgPT4gdGhpcy5fdGFibGVBbmltYXRpb25FbmQoZSl9XG4gICAgICA+XG4gICAgICAgICR7dGhpcy53ZWVrTnVtYmVyc1xuICAgICAgICAgID8gaHRtbGBcbiAgICAgICAgICAgICAgPHRoZWFkIGNsYXNzPVwic2JiLWNhbGVuZGFyX190YWJsZS1oZWFkZXJcIj5cbiAgICAgICAgICAgICAgICA8dHI+XG4gICAgICAgICAgICAgICAgICAke25leHRNb250aEFjdGl2ZURhdGVcbiAgICAgICAgICAgICAgICAgICAgPyBub3RoaW5nXG4gICAgICAgICAgICAgICAgICAgIDogaHRtbGA8dGggY2xhc3M9XCJzYmItY2FsZW5kYXJfX3RhYmxlLWRhdGFcIj48L3RoPmB9XG4gICAgICAgICAgICAgICAgICAke3dlZWtOdW1iZXJzLm1hcChcbiAgICAgICAgICAgICAgICAgICAgKHdlZWtOdW1iZXI6IG51bWJlcikgPT4gaHRtbGBcbiAgICAgICAgICAgICAgICAgICAgICA8dGggY2xhc3M9XCJzYmItY2FsZW5kYXJfX3RhYmxlLWhlYWRlci1jZWxsXCI+XG4gICAgICAgICAgICAgICAgICAgICAgICAke3RoaXMubXVsdGlwbGVcbiAgICAgICAgICAgICAgICAgICAgICAgICAgPyBodG1sYFxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPGJ1dHRvblxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBjbGFzcz1cInNiYi1jYWxlbmRhcl9faGVhZGVyLWNlbGwgc2JiLWNhbGVuZGFyX193ZWVrZGF5XCJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgYXJpYS1sYWJlbD0ke2Ake2kxOG5DYWxlbmRhcldlZWtOdW1iZXJbdGhpcy5fbGFuZ3VhZ2UuY3VycmVudF19ICR7d2Vla051bWJlcn1gfVxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBAY2xpY2s9JHsoKSA9PiB7XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgY29uc3QgZGF5czogRGF5PFQ+W10gPSB3ZWVrc0ZvclNlbGVjdE11bHRpcGxlV2Vla051bWJlcnMuZmlsdGVyKFxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgKGRheTogRGF5PFQ+KSA9PiBkYXkud2Vla1ZhbHVlID09PSB3ZWVrTnVtYmVyLFxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICkhO1xuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHRoaXMuX3NlbGVjdE11bHRpcGxlRGF0ZXMoZGF5cyk7XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIH19XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICA+XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICR7d2Vla051bWJlcn1cbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIDwvYnV0dG9uPlxuICAgICAgICAgICAgICAgICAgICAgICAgICAgIGBcbiAgICAgICAgICAgICAgICAgICAgICAgICAgOiBodG1sYFxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPHNiYi1zY3JlZW4tcmVhZGVyLW9ubHlcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPiR7YCR7aTE4bkNhbGVuZGFyV2Vla051bWJlclt0aGlzLl9sYW5ndWFnZS5jdXJyZW50XX0gJHt3ZWVrTnVtYmVyfWB9PC9zYmItc2NyZWVuLXJlYWRlci1vbmx5XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICA+XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICA8c3BhbiBhcmlhLWhpZGRlbj1cInRydWVcIj4ke3dlZWtOdW1iZXJ9PC9zcGFuPlxuICAgICAgICAgICAgICAgICAgICAgICAgICAgIGB9XG4gICAgICAgICAgICAgICAgICAgICAgPC90aD5cbiAgICAgICAgICAgICAgICAgICAgYCxcbiAgICAgICAgICAgICAgICAgICl9XG4gICAgICAgICAgICAgICAgPC90cj5cbiAgICAgICAgICAgICAgPC90aGVhZD5cbiAgICAgICAgICAgIGBcbiAgICAgICAgICA6IG5vdGhpbmd9XG4gICAgICAgIDx0Ym9keSBjbGFzcz1cInNiYi1jYWxlbmRhcl9fdGFibGUtYm9keVwiPlxuICAgICAgICAgICR7d2Vla3MubWFwKCh3ZWVrOiBEYXk8VD5bXSwgcm93SW5kZXg6IG51bWJlcikgPT4ge1xuICAgICAgICAgICAgY29uc3Qgd2Vla2RheSA9IHRoaXMuX3dlZWtkYXlzW3Jvd0luZGV4XTtcbiAgICAgICAgICAgIGNvbnN0IHNlbGVjdGFibGVEYXlzID0gdGhpcy5fd2lkZSA/IFsuLi53ZWVrLCAuLi50aGlzLl9uZXh0TW9udGhXZWVrc1tyb3dJbmRleF1dIDogd2VlaztcbiAgICAgICAgICAgIHJldHVybiBodG1sYFxuICAgICAgICAgICAgICA8dHI+XG4gICAgICAgICAgICAgICAgJHtuZXh0TW9udGhBY3RpdmVEYXRlXG4gICAgICAgICAgICAgICAgICA/IG5vdGhpbmdcbiAgICAgICAgICAgICAgICAgIDogaHRtbGBcbiAgICAgICAgICAgICAgICAgICAgICA8dGQgY2xhc3M9XCJzYmItY2FsZW5kYXJfX3RhYmxlLWhlYWRlci1jZWxsLXZlcnRpY2FsXCI+XG4gICAgICAgICAgICAgICAgICAgICAgICAke3RoaXMubXVsdGlwbGVcbiAgICAgICAgICAgICAgICAgICAgICAgICAgPyBodG1sYFxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPGJ1dHRvblxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBjbGFzcz1cInNiYi1jYWxlbmRhcl9faGVhZGVyLWNlbGwgc2JiLWNhbGVuZGFyX193ZWVrZGF5XCJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgYXJpYS1sYWJlbD0ke3dlZWtkYXkubG9uZ31cbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgQGNsaWNrPSR7KCkgPT4gdGhpcy5fc2VsZWN0TXVsdGlwbGVEYXRlcyhzZWxlY3RhYmxlRGF5cyl9XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICA+XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICR7d2Vla2RheS5uYXJyb3d9XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICA8L2J1dHRvbj5cbiAgICAgICAgICAgICAgICAgICAgICAgICAgICBgXG4gICAgICAgICAgICAgICAgICAgICAgICAgIDogaHRtbGBcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIDxzYmItc2NyZWVuLXJlYWRlci1vbmx5PiR7d2Vla2RheS5sb25nfTwvc2JiLXNjcmVlbi1yZWFkZXItb25seT5cbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIDxzcGFuIGFyaWEtaGlkZGVuPVwidHJ1ZVwiPiR7d2Vla2RheS5uYXJyb3d9PC9zcGFuPlxuICAgICAgICAgICAgICAgICAgICAgICAgICAgIGB9XG4gICAgICAgICAgICAgICAgICAgICAgPC90ZD5cbiAgICAgICAgICAgICAgICAgICAgYH1cbiAgICAgICAgICAgICAgICAke3Jvd0luZGV4IDwgd2Vla09mZnNldFxuICAgICAgICAgICAgICAgICAgPyBodG1sYDx0ZCBjbGFzcz1cInNiYi1jYWxlbmRhcl9fdGFibGUtZGF0YVwiPjwvdGQ+YFxuICAgICAgICAgICAgICAgICAgOiBub3RoaW5nfVxuICAgICAgICAgICAgICAgICR7dGhpcy5fY3JlYXRlRGF5Q2VsbHMod2Vlayl9XG4gICAgICAgICAgICAgIDwvdHI+XG4gICAgICAgICAgICBgO1xuICAgICAgICAgIH0pfVxuICAgICAgICA8L3Rib2R5PlxuICAgICAgPC90YWJsZT5cbiAgICBgO1xuICB9XG5cbiAgLyoqIENyZWF0ZXMgdGhlIGNlbGxzIGZvciB0aGUgZGFpbHkgdmlldy4gKi9cbiAgcHJpdmF0ZSBfY3JlYXRlRGF5Q2VsbHMod2VlazogRGF5PFQ+W10pOiBUZW1wbGF0ZVJlc3VsdFtdIHtcbiAgICByZXR1cm4gd2Vlay5tYXAoKGRheTogRGF5PFQ+KSA9PiB7XG4gICAgICByZXR1cm4gaHRtbGBcbiAgICAgICAgPHRkIGNsYXNzPVwiIHNiYi1jYWxlbmRhcl9fdGFibGUtZGF0YSBzYmItY2FsZW5kYXJfX2RheS1jZWxsXCI+XG4gICAgICAgICAgPHNsb3QgbmFtZT0ke2RheS52YWx1ZX0+XG4gICAgICAgICAgICA8c2JiLWNhbGVuZGFyLWRheVxuICAgICAgICAgICAgICBzbG90PSR7ZGF5LnZhbHVlfVxuICAgICAgICAgICAgICBAY2xpY2s9JHsoKSA9PiB0aGlzLl9zZWxlY3REYXRlKGRheS5kYXRlVmFsdWUpfVxuICAgICAgICAgICAgICBAa2V5ZG93bj0keyhldnQ6IEtleWJvYXJkRXZlbnQpID0+IHRoaXMuX2hhbmRsZUtleWJvYXJkRXZlbnQoZXZ0LCBkYXkpfVxuICAgICAgICAgICAgPjwvc2JiLWNhbGVuZGFyLWRheT5cbiAgICAgICAgICA8L3Nsb3Q+XG4gICAgICAgIDwvdGQ+XG4gICAgICBgO1xuICAgIH0pO1xuICB9XG5cbiAgLyoqIFJlbmRlciB0aGUgdmlldyBmb3IgdGhlIG1vbnRoIHNlbGVjdGlvbi4gKi9cbiAgcHJpdmF0ZSBfcmVuZGVyTW9udGhWaWV3KCk6IFRlbXBsYXRlUmVzdWx0IHtcbiAgICByZXR1cm4gaHRtbGBcbiAgICAgIDxkaXYgY2xhc3M9XCJzYmItY2FsZW5kYXJfX2NvbnRyb2xzXCI+XG4gICAgICAgICR7dGhpcy5fZ2V0QXJyb3coXG4gICAgICAgICAgJ2xlZnQnLFxuICAgICAgICAgICgpID0+IHRoaXMuX2dvVG9EaWZmZXJlbnRZZWFyKC0xKSxcbiAgICAgICAgICBpMThuUHJldmlvdXNZZWFyW3RoaXMuX2xhbmd1YWdlLmN1cnJlbnRdLFxuICAgICAgICAgIHRoaXMuX3ByZXZpb3VzWWVhckRpc2FibGVkKCksXG4gICAgICAgICl9XG4gICAgICAgIDxkaXYgY2xhc3M9XCJzYmItY2FsZW5kYXJfX2NvbnRyb2xzLW1vbnRoXCI+JHt0aGlzLl9jcmVhdGVMYWJlbEZvck1vbnRoVmlldygpfTwvZGl2PlxuICAgICAgICAke3RoaXMuX2dldEFycm93KFxuICAgICAgICAgICdyaWdodCcsXG4gICAgICAgICAgKCkgPT4gdGhpcy5fZ29Ub0RpZmZlcmVudFllYXIoMSksXG4gICAgICAgICAgaTE4bk5leHRZZWFyW3RoaXMuX2xhbmd1YWdlLmN1cnJlbnRdLFxuICAgICAgICAgIHRoaXMuX25leHRZZWFyRGlzYWJsZWQoKSxcbiAgICAgICAgKX1cbiAgICAgIDwvZGl2PlxuICAgICAgPGRpdiBjbGFzcz1cInNiYi1jYWxlbmRhcl9fdGFibGUtb3ZlcmZsb3ctYnJlYWtcIj5cbiAgICAgICAgPGRpdiBjbGFzcz1cInNiYi1jYWxlbmRhcl9fdGFibGUtY29udGFpbmVyIHNiYi1jYWxlbmRhcl9fdGFibGUtbW9udGgtdmlld1wiPlxuICAgICAgICAgICR7dGhpcy5fY3JlYXRlTW9udGhUYWJsZSh0aGlzLl9tb250aHMsIHRoaXMuX2Nob3NlblllYXIhKX1cbiAgICAgICAgICAke3RoaXMuX3dpZGUgPyB0aGlzLl9jcmVhdGVNb250aFRhYmxlKHRoaXMuX21vbnRocywgdGhpcy5fY2hvc2VuWWVhciEgKyAxKSA6IG5vdGhpbmd9XG4gICAgICAgIDwvZGl2PlxuICAgICAgPC9kaXY+XG4gICAgYDtcbiAgfVxuXG4gIC8qKiBDcmVhdGVzIHRoZSBsYWJlbCB3aXRoIHRoZSB5ZWFyIGZvciB0aGUgbW9udGhseSB2aWV3LiAqL1xuICBwcml2YXRlIF9jcmVhdGVMYWJlbEZvck1vbnRoVmlldygpOiBUZW1wbGF0ZVJlc3VsdCB7XG4gICAgcmV0dXJuIGh0bWxgIDxidXR0b25cbiAgICAgICAgdHlwZT1cImJ1dHRvblwiXG4gICAgICAgIGlkPVwic2JiLWNhbGVuZGFyX19tb250aC1zZWxlY3Rpb25cIlxuICAgICAgICBjbGFzcz1cInNiYi1jYWxlbmRhcl9fY29udHJvbHMtY2hhbmdlLWRhdGVcIlxuICAgICAgICBhcmlhLWxhYmVsPSR7YCR7aTE4bkNhbGVuZGFyRGF0ZVNlbGVjdGlvblt0aGlzLl9sYW5ndWFnZS5jdXJyZW50XX0gJHt0aGlzLl9jaG9zZW5ZZWFyfWB9XG4gICAgICAgIEBjbGljaz0keygpID0+IHRoaXMuX3Jlc2V0Q2FsZW5kYXJWaWV3QW5kRW1pdE1vbnRoQ2hhbmdlKHRydWUpfVxuICAgICAgPlxuICAgICAgICAke3RoaXMuX2Nob3NlblllYXJ9ICR7dGhpcy5fd2lkZSA/IGAgLSAke3RoaXMuX2Nob3NlblllYXIhICsgMX1gIDogbm90aGluZ31cbiAgICAgICAgPHNiYi1pY29uIG5hbWU9XCJjaGV2cm9uLXNtYWxsLXVwLXNtYWxsXCI+PC9zYmItaWNvbj5cbiAgICAgIDwvYnV0dG9uPlxuICAgICAgPHNiYi1zY3JlZW4tcmVhZGVyLW9ubHkgcm9sZT1cInN0YXR1c1wiPiAke3RoaXMuX2Nob3NlblllYXJ9IDwvc2JiLXNjcmVlbi1yZWFkZXItb25seT5gO1xuICB9XG5cbiAgLyoqIENyZWF0ZXMgdGhlIHRhYmxlIGZvciB0aGUgbW9udGggc2VsZWN0aW9uIHZpZXcuICovXG4gIHByaXZhdGUgX2NyZWF0ZU1vbnRoVGFibGUobW9udGhzOiBNb250aFtdW10sIHllYXI6IG51bWJlcik6IFRlbXBsYXRlUmVzdWx0IHtcbiAgICByZXR1cm4gaHRtbGBcbiAgICAgIDx0YWJsZVxuICAgICAgICBjbGFzcz1cInNiYi1jYWxlbmRhcl9fdGFibGVcIlxuICAgICAgICBAYW5pbWF0aW9uZW5kPSR7KGU6IEFuaW1hdGlvbkV2ZW50KSA9PiB0aGlzLl90YWJsZUFuaW1hdGlvbkVuZChlKX1cbiAgICAgID5cbiAgICAgICAgJHt0aGlzLl93aWRlXG4gICAgICAgICAgPyBodG1sYDx0aGVhZCBjbGFzcz1cInNiYi1jYWxlbmRhcl9fdGFibGUtaGVhZGVyXCIgYXJpYS1oaWRkZW49XCJ0cnVlXCI+XG4gICAgICAgICAgICAgIDx0cj5cbiAgICAgICAgICAgICAgICA8dGggY2xhc3M9XCJzYmItY2FsZW5kYXJfX3RhYmxlLWhlYWRlci1jZWxsXCIgY29sc3Bhbj0ke01PTlRIU19QRVJfUk9XfT4ke3llYXJ9PC90aD5cbiAgICAgICAgICAgICAgPC90cj5cbiAgICAgICAgICAgIDwvdGhlYWQ+YFxuICAgICAgICAgIDogbm90aGluZ31cbiAgICAgICAgPHRib2R5IGNsYXNzPVwic2JiLWNhbGVuZGFyX190YWJsZS1ib2R5XCI+XG4gICAgICAgICAgJHttb250aHMubWFwKFxuICAgICAgICAgICAgKHJvdzogTW9udGhbXSkgPT4gaHRtbGBcbiAgICAgICAgICAgICAgPHRyPlxuICAgICAgICAgICAgICAgICR7cm93Lm1hcCgobW9udGg6IE1vbnRoKSA9PiB7XG4gICAgICAgICAgICAgICAgICBsZXQgc2VsZWN0ZWQ6IGJvb2xlYW47XG4gICAgICAgICAgICAgICAgICBpZiAodGhpcy5tdWx0aXBsZSkge1xuICAgICAgICAgICAgICAgICAgICBzZWxlY3RlZCA9XG4gICAgICAgICAgICAgICAgICAgICAgKHRoaXMuX3NlbGVjdGVkIGFzIFRbXSkuZmluZChcbiAgICAgICAgICAgICAgICAgICAgICAgIChkYXRlOiBUKSA9PlxuICAgICAgICAgICAgICAgICAgICAgICAgICB5ZWFyID09PSB0aGlzLl9kYXRlQWRhcHRlci5nZXRZZWFyKGRhdGUpICYmXG4gICAgICAgICAgICAgICAgICAgICAgICAgIG1vbnRoLm1vbnRoVmFsdWUgPT09IHRoaXMuX2RhdGVBZGFwdGVyLmdldE1vbnRoKGRhdGUpLFxuICAgICAgICAgICAgICAgICAgICAgICkgIT09IHVuZGVmaW5lZDtcbiAgICAgICAgICAgICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgICAgICAgICAgIGNvbnN0IHNlbGVjdGVkTW9udGggPSB0aGlzLl9zZWxlY3RlZFxuICAgICAgICAgICAgICAgICAgICAgID8gdGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0TW9udGgodGhpcy5fc2VsZWN0ZWQgYXMgVClcbiAgICAgICAgICAgICAgICAgICAgICA6IHVuZGVmaW5lZDtcbiAgICAgICAgICAgICAgICAgICAgY29uc3Qgc2VsZWN0ZWRZZWFyID0gdGhpcy5fc2VsZWN0ZWRcbiAgICAgICAgICAgICAgICAgICAgICA/IHRoaXMuX2RhdGVBZGFwdGVyLmdldFllYXIodGhpcy5fc2VsZWN0ZWQgYXMgVClcbiAgICAgICAgICAgICAgICAgICAgICA6IHVuZGVmaW5lZDtcbiAgICAgICAgICAgICAgICAgICAgc2VsZWN0ZWQgPVxuICAgICAgICAgICAgICAgICAgICAgICEhdGhpcy5fc2VsZWN0ZWQgJiZcbiAgICAgICAgICAgICAgICAgICAgICB5ZWFyID09PSBzZWxlY3RlZFllYXIgJiZcbiAgICAgICAgICAgICAgICAgICAgICBtb250aC5tb250aFZhbHVlID09PSBzZWxlY3RlZE1vbnRoO1xuICAgICAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgICAgICAgY29uc3QgaXNPdXRPZlJhbmdlID0gIXRoaXMuX2lzTW9udGhJblJhbmdlKG1vbnRoLm1vbnRoVmFsdWUsIHllYXIpO1xuICAgICAgICAgICAgICAgICAgY29uc3QgaXNGaWx0ZXJlZE91dCA9ICF0aGlzLl9pc01vbnRoRmlsdGVyZWRPdXQobW9udGgubW9udGhWYWx1ZSwgeWVhcik7XG4gICAgICAgICAgICAgICAgICBjb25zdCBpc0N1cnJlbnRNb250aCA9XG4gICAgICAgICAgICAgICAgICAgIHllYXIgPT09IHRoaXMuX2RhdGVBZGFwdGVyLmdldFllYXIodGhpcy5fZGF0ZUFkYXB0ZXIudG9kYXkoKSkgJiZcbiAgICAgICAgICAgICAgICAgICAgdGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0TW9udGgodGhpcy5fZGF0ZUFkYXB0ZXIudG9kYXkoKSkgPT09IG1vbnRoLm1vbnRoVmFsdWU7XG5cbiAgICAgICAgICAgICAgICAgIHJldHVybiBodG1sYCA8dGRcbiAgICAgICAgICAgICAgICAgICAgY2xhc3M9JHtjbGFzc01hcCh7XG4gICAgICAgICAgICAgICAgICAgICAgJ3NiYi1jYWxlbmRhcl9fdGFibGUtZGF0YSc6IHRydWUsXG4gICAgICAgICAgICAgICAgICAgICAgJ3NiYi1jYWxlbmRhcl9fdGFibGUtbW9udGgnOiB0cnVlLFxuICAgICAgICAgICAgICAgICAgICB9KX1cbiAgICAgICAgICAgICAgICAgID5cbiAgICAgICAgICAgICAgICAgICAgPGJ1dHRvblxuICAgICAgICAgICAgICAgICAgICAgIGNsYXNzPSR7Y2xhc3NNYXAoe1xuICAgICAgICAgICAgICAgICAgICAgICAgJ3NiYi1jYWxlbmRhcl9fY2VsbCc6IHRydWUsXG4gICAgICAgICAgICAgICAgICAgICAgICAnc2JiLWNhbGVuZGFyX19jZWxsLWN1cnJlbnQnOiBpc0N1cnJlbnRNb250aCxcbiAgICAgICAgICAgICAgICAgICAgICAgICdzYmItY2FsZW5kYXJfX2Nyb3NzZWQtb3V0JzogIWlzT3V0T2ZSYW5nZSAmJiBpc0ZpbHRlcmVkT3V0LFxuICAgICAgICAgICAgICAgICAgICAgICAgJ3NiYi1jYWxlbmRhcl9fc2VsZWN0ZWQnOiBzZWxlY3RlZCxcbiAgICAgICAgICAgICAgICAgICAgICB9KX1cbiAgICAgICAgICAgICAgICAgICAgICBAY2xpY2s9JHsoKSA9PiB0aGlzLl9vbk1vbnRoU2VsZWN0aW9uKG1vbnRoLm1vbnRoVmFsdWUsIHllYXIpfVxuICAgICAgICAgICAgICAgICAgICAgID9kaXNhYmxlZD0ke2lzT3V0T2ZSYW5nZSB8fCBpc0ZpbHRlcmVkT3V0fVxuICAgICAgICAgICAgICAgICAgICAgIGFyaWEtbGFiZWw9JHtgJHttb250aC5sb25nVmFsdWV9ICR7eWVhcn1gfVxuICAgICAgICAgICAgICAgICAgICAgIGFyaWEtcHJlc3NlZD0ke3NlbGVjdGVkfVxuICAgICAgICAgICAgICAgICAgICAgIGFyaWEtZGlzYWJsZWQ9JHtTdHJpbmcoaXNPdXRPZlJhbmdlIHx8IGlzRmlsdGVyZWRPdXQpfVxuICAgICAgICAgICAgICAgICAgICAgIHRhYmluZGV4PVwiLTFcIlxuICAgICAgICAgICAgICAgICAgICAgIGRhdGEtbW9udGg9JHttb250aC5tb250aFZhbHVlIHx8IG5vdGhpbmd9XG4gICAgICAgICAgICAgICAgICAgICAgQGtleWRvd249JHsoZXZ0OiBLZXlib2FyZEV2ZW50KSA9PiB0aGlzLl9oYW5kbGVLZXlib2FyZEV2ZW50KGV2dCl9XG4gICAgICAgICAgICAgICAgICAgID5cbiAgICAgICAgICAgICAgICAgICAgICAke21vbnRoLnZhbHVlfVxuICAgICAgICAgICAgICAgICAgICA8L2J1dHRvbj5cbiAgICAgICAgICAgICAgICAgIDwvdGQ+YDtcbiAgICAgICAgICAgICAgICB9KX1cbiAgICAgICAgICAgICAgPC90cj5cbiAgICAgICAgICAgIGAsXG4gICAgICAgICAgKX1cbiAgICAgICAgPC90Ym9keT5cbiAgICAgIDwvdGFibGU+XG4gICAgYDtcbiAgfVxuXG4gIC8qKiBTZWxlY3QgdGhlIG1vbnRoIGFuZCBjaGFuZ2UgdGhlIHZpZXcgdG8gZGF5IHNlbGVjdGlvbi4gKi9cbiAgcHJpdmF0ZSBfb25Nb250aFNlbGVjdGlvbihtb250aDogbnVtYmVyLCB5ZWFyOiBudW1iZXIpOiB2b2lkIHtcbiAgICB0aGlzLl9jaG9zZW5Nb250aCA9IG1vbnRoO1xuICAgIHRoaXMuX25leHRDYWxlbmRhclZpZXcgPSAnZGF5JztcbiAgICB0aGlzLl9pbml0KFxuICAgICAgdGhpcy5fZGF0ZUFkYXB0ZXIuY3JlYXRlRGF0ZShcbiAgICAgICAgeWVhcixcbiAgICAgICAgdGhpcy5fY2hvc2VuTW9udGgsXG4gICAgICAgIHRoaXMuX2RhdGVBZGFwdGVyLmdldERhdGUodGhpcy5fYWN0aXZlRGF0ZSksXG4gICAgICApLFxuICAgICk7XG4gICAgdGhpcy5fc3RhcnRUYWJsZVRyYW5zaXRpb24oKTtcbiAgICB0aGlzLl9lbWl0TW9udGhDaGFuZ2UoKTtcbiAgfVxuXG4gIC8qKiBSZW5kZXIgdGhlIHZpZXcgZm9yIHRoZSB5ZWFyIHNlbGVjdGlvbi4gKi9cbiAgcHJpdmF0ZSBfcmVuZGVyWWVhclZpZXcoKTogVGVtcGxhdGVSZXN1bHQge1xuICAgIHJldHVybiBodG1sYFxuICAgICAgPGRpdiBjbGFzcz1cInNiYi1jYWxlbmRhcl9fY29udHJvbHNcIj5cbiAgICAgICAgJHt0aGlzLl9nZXRBcnJvdyhcbiAgICAgICAgICAnbGVmdCcsXG4gICAgICAgICAgKCkgPT4gdGhpcy5fZ29Ub0RpZmZlcmVudFllYXJSYW5nZSgtWUVBUlNfUEVSX1BBR0UpLFxuICAgICAgICAgIGkxOG5QcmV2aW91c1llYXJSYW5nZShZRUFSU19QRVJfUEFHRSlbdGhpcy5fbGFuZ3VhZ2UuY3VycmVudF0sXG4gICAgICAgICAgdGhpcy5fcHJldmlvdXNZZWFyUmFuZ2VEaXNhYmxlZCgpLFxuICAgICAgICApfVxuICAgICAgICA8ZGl2IGNsYXNzPVwic2JiLWNhbGVuZGFyX19jb250cm9scy1tb250aFwiPiR7dGhpcy5fY3JlYXRlTGFiZWxGb3JZZWFyVmlldygpfTwvZGl2PlxuICAgICAgICAke3RoaXMuX2dldEFycm93KFxuICAgICAgICAgICdyaWdodCcsXG4gICAgICAgICAgKCkgPT4gdGhpcy5fZ29Ub0RpZmZlcmVudFllYXJSYW5nZShZRUFSU19QRVJfUEFHRSksXG4gICAgICAgICAgaTE4bk5leHRZZWFyUmFuZ2UoWUVBUlNfUEVSX1BBR0UpW3RoaXMuX2xhbmd1YWdlLmN1cnJlbnRdLFxuICAgICAgICAgIHRoaXMuX25leHRZZWFyUmFuZ2VEaXNhYmxlZCgpLFxuICAgICAgICApfVxuICAgICAgPC9kaXY+XG4gICAgICA8ZGl2IGNsYXNzPVwic2JiLWNhbGVuZGFyX190YWJsZS1vdmVyZmxvdy1icmVha1wiPlxuICAgICAgICA8ZGl2IGNsYXNzPVwic2JiLWNhbGVuZGFyX190YWJsZS1jb250YWluZXIgc2JiLWNhbGVuZGFyX190YWJsZS15ZWFyLXZpZXdcIj5cbiAgICAgICAgICAke3RoaXMuX2NyZWF0ZVllYXJUYWJsZSh0aGlzLl95ZWFycyl9XG4gICAgICAgICAgJHt0aGlzLl93aWRlID8gdGhpcy5fY3JlYXRlWWVhclRhYmxlKHRoaXMuX25leHRNb250aFllYXJzLCB0cnVlKSA6IG5vdGhpbmd9XG4gICAgICAgIDwvZGl2PlxuICAgICAgPC9kaXY+XG4gICAgYDtcbiAgfVxuXG4gIC8qKiBDcmVhdGVzIHRoZSBidXR0b24gYXJyb3cgZm9yIGFsbCB0aGUgdmlld3MuICovXG4gIHByaXZhdGUgX2dldEFycm93KFxuICAgIGRpcmVjdGlvbjogJ2xlZnQnIHwgJ3JpZ2h0JyxcbiAgICBjbGljazogKCkgPT4gdm9pZCxcbiAgICBhcmlhTGFiZWw6IHN0cmluZyxcbiAgICBkaXNhYmxlZDogYm9vbGVhbixcbiAgKTogVGVtcGxhdGVSZXN1bHQge1xuICAgIHJldHVybiBodG1sYDxzYmItc2Vjb25kYXJ5LWJ1dHRvblxuICAgICAgc2l6ZT1cIm1cIlxuICAgICAgaWNvbi1uYW1lPVwiY2hldnJvbi1zbWFsbC0ke2RpcmVjdGlvbn0tc21hbGxcIlxuICAgICAgYXJpYS1sYWJlbD0ke2FyaWFMYWJlbH1cbiAgICAgIEBjbGljaz0ke2NsaWNrfVxuICAgICAgP2Rpc2FibGVkPSR7ZGlzYWJsZWR9XG4gICAgICBpZD1cInNiYi1jYWxlbmRhcl9fY29udHJvbHMtJHtkaXJlY3Rpb24gPT09ICdsZWZ0JyA/ICdwcmV2aW91cycgOiAnbmV4dCd9XCJcbiAgICA+PC9zYmItc2Vjb25kYXJ5LWJ1dHRvbj5gO1xuICB9XG5cbiAgLyoqIENyZWF0ZXMgdGhlIGxhYmVsIHdpdGggdGhlIHllYXIgcmFuZ2UgZm9yIHRoZSB5ZWFybHkgdmlldy4gKi9cbiAgcHJpdmF0ZSBfY3JlYXRlTGFiZWxGb3JZZWFyVmlldygpOiBUZW1wbGF0ZVJlc3VsdCB7XG4gICAgY29uc3QgZmlyc3RZZWFyOiBudW1iZXIgPSB0aGlzLl95ZWFycy5mbGF0KClbMF07XG4gICAgY29uc3QgbGFzdFllYXJBcnJheTogbnVtYmVyW10gPSAodGhpcy5fd2lkZSA/IHRoaXMuX25leHRNb250aFllYXJzIDogdGhpcy5feWVhcnMpLmZsYXQoKTtcbiAgICBjb25zdCBsYXN0WWVhcjogbnVtYmVyID0gbGFzdFllYXJBcnJheVtsYXN0WWVhckFycmF5Lmxlbmd0aCAtIDFdO1xuICAgIGNvbnN0IHllYXJMYWJlbCA9IGAke2ZpcnN0WWVhcn0gLSAke2xhc3RZZWFyfWA7XG4gICAgcmV0dXJuIGh0bWxgXG4gICAgICA8YnV0dG9uXG4gICAgICAgIHR5cGU9XCJidXR0b25cIlxuICAgICAgICBpZD1cInNiYi1jYWxlbmRhcl9feWVhci1zZWxlY3Rpb25cIlxuICAgICAgICBjbGFzcz1cInNiYi1jYWxlbmRhcl9fY29udHJvbHMtY2hhbmdlLWRhdGVcIlxuICAgICAgICBhcmlhLWxhYmVsPVwiJHtpMThuQ2FsZW5kYXJEYXRlU2VsZWN0aW9uW3RoaXMuX2xhbmd1YWdlLmN1cnJlbnRdfSAke3llYXJMYWJlbH1cIlxuICAgICAgICBAY2xpY2s9JHsoKSA9PiB0aGlzLl9yZXNldENhbGVuZGFyVmlld0FuZEVtaXRNb250aENoYW5nZSh0cnVlKX1cbiAgICAgID5cbiAgICAgICAgJHt5ZWFyTGFiZWx9XG4gICAgICAgIDxzYmItaWNvbiBuYW1lPVwiY2hldnJvbi1zbWFsbC11cC1zbWFsbFwiPjwvc2JiLWljb24+XG4gICAgICA8L2J1dHRvbj5cbiAgICAgIDxzYmItc2NyZWVuLXJlYWRlci1vbmx5IHJvbGU9XCJzdGF0dXNcIj4gJHt5ZWFyTGFiZWx9IDwvc2JiLXNjcmVlbi1yZWFkZXItb25seT5cbiAgICBgO1xuICB9XG5cbiAgLyoqIENyZWF0ZXMgdGhlIHRhYmxlIGZvciB0aGUgeWVhciBzZWxlY3Rpb24gdmlldy4gKi9cbiAgcHJpdmF0ZSBfY3JlYXRlWWVhclRhYmxlKHllYXJzOiBudW1iZXJbXVtdLCBzaGlmdFJpZ2h0ID0gZmFsc2UpOiBUZW1wbGF0ZVJlc3VsdCB7XG4gICAgY29uc3Qgbm93ID0gdGhpcy5fZGF0ZUFkYXB0ZXIudG9kYXkoKTtcbiAgICByZXR1cm4gaHRtbGAgPHRhYmxlXG4gICAgICBjbGFzcz1cInNiYi1jYWxlbmRhcl9fdGFibGVcIlxuICAgICAgQGFuaW1hdGlvbmVuZD0keyhlOiBBbmltYXRpb25FdmVudCkgPT4gdGhpcy5fdGFibGVBbmltYXRpb25FbmQoZSl9XG4gICAgPlxuICAgICAgPHRib2R5IGNsYXNzPVwic2JiLWNhbGVuZGFyX190YWJsZS1ib2R5XCI+XG4gICAgICAgICR7eWVhcnMubWFwKFxuICAgICAgICAgIChyb3c6IG51bWJlcltdKSA9PlxuICAgICAgICAgICAgaHRtbGAgPHRyPlxuICAgICAgICAgICAgICAke3Jvdy5tYXAoKHllYXI6IG51bWJlcikgPT4ge1xuICAgICAgICAgICAgICAgIGxldCBzZWxlY3RlZDogYm9vbGVhbjtcbiAgICAgICAgICAgICAgICBpZiAodGhpcy5tdWx0aXBsZSkge1xuICAgICAgICAgICAgICAgICAgc2VsZWN0ZWQgPVxuICAgICAgICAgICAgICAgICAgICAodGhpcy5fc2VsZWN0ZWQgYXMgVFtdKS5maW5kKFxuICAgICAgICAgICAgICAgICAgICAgIChkYXRlOiBUKSA9PiB5ZWFyID09PSB0aGlzLl9kYXRlQWRhcHRlci5nZXRZZWFyKGRhdGUpLFxuICAgICAgICAgICAgICAgICAgICApICE9PSB1bmRlZmluZWQ7XG4gICAgICAgICAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICAgICAgICAgIGNvbnN0IHNlbGVjdGVkWWVhciA9IHRoaXMuX3NlbGVjdGVkXG4gICAgICAgICAgICAgICAgICAgID8gdGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0WWVhcih0aGlzLl9zZWxlY3RlZCBhcyBUKVxuICAgICAgICAgICAgICAgICAgICA6IHVuZGVmaW5lZDtcbiAgICAgICAgICAgICAgICAgIHNlbGVjdGVkID0gISF0aGlzLl9zZWxlY3RlZCAmJiB5ZWFyID09PSBzZWxlY3RlZFllYXI7XG4gICAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgICAgIGNvbnN0IGlzT3V0T2ZSYW5nZSA9ICF0aGlzLl9pc1llYXJJblJhbmdlKHllYXIpO1xuICAgICAgICAgICAgICAgIGNvbnN0IGlzRmlsdGVyZWRPdXQgPSAhdGhpcy5faXNZZWFyRmlsdGVyZWRPdXQoeWVhcik7XG4gICAgICAgICAgICAgICAgY29uc3QgaXNDdXJyZW50WWVhciA9IHRoaXMuX2RhdGVBZGFwdGVyLmdldFllYXIobm93KSA9PT0geWVhcjtcbiAgICAgICAgICAgICAgICByZXR1cm4gaHRtbGAgPHRkIGNsYXNzPVwic2JiLWNhbGVuZGFyX190YWJsZS1kYXRhIHNiYi1jYWxlbmRhcl9fdGFibGUteWVhclwiPlxuICAgICAgICAgICAgICAgICAgPGJ1dHRvblxuICAgICAgICAgICAgICAgICAgICBjbGFzcz0ke2NsYXNzTWFwKHtcbiAgICAgICAgICAgICAgICAgICAgICAnc2JiLWNhbGVuZGFyX19jZWxsJzogdHJ1ZSxcbiAgICAgICAgICAgICAgICAgICAgICAnc2JiLWNhbGVuZGFyX19jZWxsLWN1cnJlbnQnOiBpc0N1cnJlbnRZZWFyLFxuICAgICAgICAgICAgICAgICAgICAgICdzYmItY2FsZW5kYXJfX2Nyb3NzZWQtb3V0JzogIWlzT3V0T2ZSYW5nZSAmJiBpc0ZpbHRlcmVkT3V0LFxuICAgICAgICAgICAgICAgICAgICAgICdzYmItY2FsZW5kYXJfX3NlbGVjdGVkJzogc2VsZWN0ZWQsXG4gICAgICAgICAgICAgICAgICAgIH0pfVxuICAgICAgICAgICAgICAgICAgICBAY2xpY2s9JHsoKSA9PiB0aGlzLl9vblllYXJTZWxlY3Rpb24oeWVhciwgc2hpZnRSaWdodCl9XG4gICAgICAgICAgICAgICAgICAgID9kaXNhYmxlZD0ke2lzT3V0T2ZSYW5nZSB8fCBpc0ZpbHRlcmVkT3V0fVxuICAgICAgICAgICAgICAgICAgICBhcmlhLWxhYmVsPSR7eWVhcn1cbiAgICAgICAgICAgICAgICAgICAgYXJpYS1wcmVzc2VkPSR7c2VsZWN0ZWR9XG4gICAgICAgICAgICAgICAgICAgIGFyaWEtZGlzYWJsZWQ9JHtTdHJpbmcoaXNPdXRPZlJhbmdlIHx8IGlzRmlsdGVyZWRPdXQpfVxuICAgICAgICAgICAgICAgICAgICB0YWJpbmRleD1cIi0xXCJcbiAgICAgICAgICAgICAgICAgICAgZGF0YS15ZWFyPSR7eWVhciB8fCBub3RoaW5nfVxuICAgICAgICAgICAgICAgICAgICBAa2V5ZG93bj0keyhldnQ6IEtleWJvYXJkRXZlbnQpID0+IHRoaXMuX2hhbmRsZUtleWJvYXJkRXZlbnQoZXZ0KX1cbiAgICAgICAgICAgICAgICAgID5cbiAgICAgICAgICAgICAgICAgICAgJHt5ZWFyfVxuICAgICAgICAgICAgICAgICAgPC9idXR0b24+XG4gICAgICAgICAgICAgICAgPC90ZD5gO1xuICAgICAgICAgICAgICB9KX1cbiAgICAgICAgICAgIDwvdHI+YCxcbiAgICAgICAgKX1cbiAgICAgIDwvdGJvZHk+XG4gICAgPC90YWJsZT5gO1xuICB9XG5cbiAgLyoqIFNlbGVjdCB0aGUgeWVhciBhbmQgY2hhbmdlIHRoZSB2aWV3IHRvIG1vbnRoIHNlbGVjdGlvbi4gKi9cbiAgcHJpdmF0ZSBfb25ZZWFyU2VsZWN0aW9uKHllYXI6IG51bWJlciwgcmlnaHRTaWRlOiBib29sZWFuKTogdm9pZCB7XG4gICAgdGhpcy5fY2hvc2VuWWVhciA9IHJpZ2h0U2lkZSA/IHllYXIgLSAxIDogeWVhcjtcbiAgICB0aGlzLl9uZXh0Q2FsZW5kYXJWaWV3ID0gJ21vbnRoJztcbiAgICB0aGlzLl9hc3NpZ25BY3RpdmVEYXRlKFxuICAgICAgdGhpcy5fZGF0ZUFkYXB0ZXIuY3JlYXRlRGF0ZShcbiAgICAgICAgdGhpcy5fY2hvc2VuWWVhcixcbiAgICAgICAgdGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0TW9udGgodGhpcy5fYWN0aXZlRGF0ZSksXG4gICAgICAgIHRoaXMuX2RhdGVBZGFwdGVyLmdldERhdGUodGhpcy5fYWN0aXZlRGF0ZSksXG4gICAgICApLFxuICAgICk7XG4gICAgdGhpcy5fc3RhcnRUYWJsZVRyYW5zaXRpb24oKTtcbiAgfVxuXG4gIHByaXZhdGUgX2dldFZpZXcoKTogVGVtcGxhdGVSZXN1bHQge1xuICAgIGlmIChpc1NlcnZlciB8fCB0aGlzLmh5ZHJhdGlvblJlcXVpcmVkKSB7XG4gICAgICAvLyBUT0RPOiBXZSBkaXNhYmxlIFNTUiBmb3IgY2FsZW5kYXIgZm9yIG5vdy4gRmlndXJlIG91dCwgaWYgdGhlcmUgaXMgYSB3YXlcbiAgICAgIC8vIHRvIGVuYWJsZSBpdCwgd2hpbGUgY29uc2lkZXJpbmcgaTE4biBhbmQgZGF0ZSBpbmZvcm1hdGlvbi5cbiAgICAgIHJldHVybiBodG1sYCR7bm90aGluZ31gO1xuICAgIH1cbiAgICBzd2l0Y2ggKHRoaXMuX2NhbGVuZGFyVmlldykge1xuICAgICAgY2FzZSAneWVhcic6XG4gICAgICAgIHJldHVybiB0aGlzLl9yZW5kZXJZZWFyVmlldygpO1xuICAgICAgY2FzZSAnbW9udGgnOlxuICAgICAgICByZXR1cm4gdGhpcy5fcmVuZGVyTW9udGhWaWV3KCk7XG4gICAgICBjYXNlICdkYXknOlxuICAgICAgZGVmYXVsdDpcbiAgICAgICAgcmV0dXJuIHRoaXMuX3JlbmRlckRheVZpZXcoKTtcbiAgICB9XG4gIH1cblxuICBwcml2YXRlIF90YWJsZUFuaW1hdGlvbkVuZChldmVudDogQW5pbWF0aW9uRXZlbnQpOiB2b2lkIHtcbiAgICBjb25zdCB0YWJsZSA9IGV2ZW50LnRhcmdldCBhcyBIVE1MRWxlbWVudDtcbiAgICBpZiAoZXZlbnQuYW5pbWF0aW9uTmFtZSA9PT0gJ2hpZGUnKSB7XG4gICAgICB0YWJsZS5jbGFzc0xpc3QucmVtb3ZlKCdzYmItY2FsZW5kYXJfX3RhYmxlLWhpZGUnKTtcbiAgICAgIGlmICh0aGlzLl9jb250YWluaW5nRm9jdXMpIHtcbiAgICAgICAgdGhpcy5fcmVzZXRGb2N1cyA9IHRydWU7XG4gICAgICB9XG4gICAgICB0aGlzLl9jYWxlbmRhclZpZXcgPSB0aGlzLl9uZXh0Q2FsZW5kYXJWaWV3O1xuICAgIH0gZWxzZSBpZiAoZXZlbnQuYW5pbWF0aW9uTmFtZSA9PT0gJ3Nob3cnKSB7XG4gICAgICB0aGlzLmludGVybmFscy5zdGF0ZXMuZGVsZXRlKCd0cmFuc2l0aW9uJyk7XG4gICAgfVxuICB9XG5cbiAgcHJpdmF0ZSBfc3RhcnRUYWJsZVRyYW5zaXRpb24oKTogdm9pZCB7XG4gICAgdGhpcy5pbnRlcm5hbHMuc3RhdGVzLmFkZCgndHJhbnNpdGlvbicpO1xuICAgIHRoaXMuc2hhZG93Um9vdFxuICAgICAgPy5xdWVyeVNlbGVjdG9yQWxsKCd0YWJsZScpXG4gICAgICA/LmZvckVhY2goKGUpID0+IGUuY2xhc3NMaXN0LnRvZ2dsZSgnc2JiLWNhbGVuZGFyX190YWJsZS1oaWRlJykpO1xuICB9XG5cbiAgcHJvdGVjdGVkIG92ZXJyaWRlIHJlbmRlcigpOiBUZW1wbGF0ZVJlc3VsdCB7XG4gICAgcmV0dXJuIGh0bWxgPGRpdiBjbGFzcz1cInNiYi1jYWxlbmRhcl9fd3JhcHBlclwiPiR7dGhpcy5fZ2V0VmlldygpfTwvZGl2PmA7XG4gIH1cbn1cblxuZGVjbGFyZSBnbG9iYWwge1xuICBpbnRlcmZhY2UgSFRNTEVsZW1lbnRUYWdOYW1lTWFwIHtcbiAgICAvLyBlc2xpbnQtZGlzYWJsZS1uZXh0LWxpbmUgQHR5cGVzY3JpcHQtZXNsaW50L25hbWluZy1jb252ZW50aW9uXG4gICAgJ3NiYi1jYWxlbmRhcic6IFNiYkNhbGVuZGFyRWxlbWVudDtcbiAgfVxuICBpbnRlcmZhY2UgSFRNTEVsZW1lbnRFdmVudE1hcCB7XG4gICAgbW9udGhjaGFuZ2U6IFNiYk1vbnRoQ2hhbmdlRXZlbnQ7XG4gIH1cbn1cbiJdLCJuYW1lcyI6WyJkZWZhdWx0T3B0aW9ucyIsImRheSJdLCJtYXBwaW5ncyI6Ijs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7QUE0RU8sTUFBTSxxQkFBcUI7QUEySTNCLE1BQU0sc0JBQXNCLHVCQUFPLElBQUksbUJBQW1CO0FDbEwxRCxTQUFTLGNBQWMsTUFBTSxPQUFPO0FBQ3pDLE1BQUksT0FBTyxTQUFTLFdBQVksUUFBTyxLQUFLLEtBQUs7QUFFakQsTUFBSSxRQUFRLE9BQU8sU0FBUyxZQUFZLHVCQUF1QjtBQUM3RCxXQUFPLEtBQUssbUJBQW1CLEVBQUUsS0FBSztBQUV4QyxNQUFJLGdCQUFnQixLQUFNLFFBQU8sSUFBSSxLQUFLLFlBQVksS0FBSztBQUUzRCxTQUFPLElBQUksS0FBSyxLQUFLO0FBQ3ZCO0FDTk8sU0FBUyxPQUFPLFVBQVUsU0FBUztBQUV4QyxTQUFPLGNBQWMsV0FB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RXZDLGlCQUFBLElBQUEsTUFBQSwwQkFBQSxFQUFBLE1BQUEsWUFBQSxNQUFBLGdCQUFBLFFBQUEsT0FBQSxTQUFBLE9BQUEsUUFBQSxFQUFBLEtBQUEsQ0FBQSxRQUFBLGtCQUFBLEtBQUEsS0FBQSxDQUFBLFFBQUEsSUFBaUIsY0FBWSxLQUFBLENBQUEsS0FBQSxVQUFBO0FBQUEsVUFBWixlQUFZO0FBQUEsSUFBQSxLQUFBLFVBQUEsVUFBQSxHQUFBLDRCQUFBLCtCQUFBO0FBeEsvQixpQkFBQSxNQUFBLG1CQUFBLEVBQUEsT0FBQSxXQUFBLEdBQUEsa0JBQUEsRUFBQSxNQUFBLFNBQUEsTUFBQSxXQUFBLE1BQUEsVUFBQSxVQUFBLEdBQUEsTUFBQSx1QkFBQTs7O1FBQ3lCLEdBQUEsU0FBeUIsQ0FBQyxpQkFBaUIsS0FBSyxHQUNoRCxHQUFBLFNBQVM7QUFBQSxJQUM5QixjQUFjO0FBQUEsSUFDZCxhQUFhO0FBQUEsRUFBQSxHQUpYLGtCQUFBLFlBQUEsdUJBQUEsR0FBNkI7OzsiLCJ4X2dvb2dsZV9pZ25vcmVMaXN0IjpbMCwxLDIsMyw0LDUsNiw3LDgsOSwxMCwxMSwxMiwxMywxNF19