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