@sbb-esta/lyne-elements-dev 4.9.0-dev.1774875578 → 4.9.0-dev.1774887348
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/calendar/calendar/calendar.component.js +1 -1
- package/calendar/calendar.js +1 -1
- package/{calendar.component-Dk8Xun-b.js → calendar.component-BRIW9HVu.js} +88 -186
- package/calendar.js +1 -1
- package/calendar.pure.js +1 -1
- package/core/datetime/date-adapter.js +13 -13
- package/core/datetime.js +4 -4
- package/core.js +47 -47
- package/custom-elements.json +1050 -939
- package/development/calendar/calendar/calendar.component.d.ts +1 -12
- package/development/calendar/calendar/calendar.component.d.ts.map +1 -1
- package/development/calendar/calendar/calendar.component.js +1 -1
- package/development/calendar/calendar.js +1 -1
- package/development/calendar.component-CsAl9EAg.js +1678 -0
- package/development/calendar.js +1 -1
- package/development/calendar.pure.js +1 -1
- package/development/core/datetime/date-adapter.d.ts +9 -2
- package/development/core/datetime/date-adapter.d.ts.map +1 -1
- package/development/core/datetime/date-adapter.js +11 -4
- package/development/core/datetime.js +2 -2
- package/development/core.js +2 -2
- package/package.json +2 -2
- package/development/calendar.component-CRvTYw1K.js +0 -2223
|
@@ -1,2223 +0,0 @@
|
|
|
1
|
-
import { __esDecorate, __runInitializers } from "tslib";
|
|
2
|
-
import { css, html, isServer, nothing } from "lit";
|
|
3
|
-
import { property, state } from "lit/decorators.js";
|
|
4
|
-
import { SbbElement } from "./core/base-elements.js";
|
|
5
|
-
import { forceType, handleDistinctChange, plainDate } from "./core/decorators.js";
|
|
6
|
-
import { boxSizingStyles } from "./core/styles.js";
|
|
7
|
-
import { SbbSecondaryButtonElement } from "./button.pure.js";
|
|
8
|
-
import { SbbLanguageController, SbbMediaMatcherController, SbbMediaQueryBreakpointLargeAndAbove } from "./core/controllers.js";
|
|
9
|
-
import { i18nCalendarDateSelection, i18nCalendarWeekNumber, i18nNextMonth, i18nNextYear, i18nNextYearRange, i18nPreviousMonth, i18nPreviousYear, i18nPreviousYearRange, i18nYearMonthSelection } from "./core/i18n.js";
|
|
10
|
-
import "./icon.js";
|
|
11
|
-
import { isArrowKeyOrPageKeysPressed } from "./core/a11y.js";
|
|
12
|
-
import { readConfig } from "./core/config.js";
|
|
13
|
-
import { DAYS_PER_ROW, MONTHS_PER_PAGE, MONTHS_PER_ROW, YEARS_PER_PAGE, YEARS_PER_ROW, defaultDateAdapter } from "./core/datetime.js";
|
|
14
|
-
import { SbbCalendarDayElement } from "./calendar/calendar-day/calendar-day.component.js";
|
|
15
|
-
import { SbbCalendarMonthElement } from "./calendar/calendar-month/calendar-month.component.js";
|
|
16
|
-
import { SbbCalendarWeekdayElement } from "./calendar/calendar-weekday/calendar-weekday.component.js";
|
|
17
|
-
import { SbbCalendarWeeknumberElement } from "./calendar/calendar-weeknumber/calendar-weeknumber.component.js";
|
|
18
|
-
import { SbbCalendarYearElement } from "./calendar/calendar-year/calendar-year.component.js";
|
|
19
|
-
import "./screen-reader-only.js";
|
|
20
|
-
//#region node_modules/date-fns/constants.js
|
|
21
|
-
/**
|
|
22
|
-
* @constant
|
|
23
|
-
* @name daysInYear
|
|
24
|
-
* @summary Days in 1 year.
|
|
25
|
-
*
|
|
26
|
-
* @description
|
|
27
|
-
* How many days in a year.
|
|
28
|
-
*
|
|
29
|
-
* One years equals 365.2425 days according to the formula:
|
|
30
|
-
*
|
|
31
|
-
* > Leap year occurs every 4 years, except for years that are divisible by 100 and not divisible by 400.
|
|
32
|
-
* > 1 mean year = (365+1/4-1/100+1/400) days = 365.2425 days
|
|
33
|
-
*/
|
|
34
|
-
var daysInYear = 365.2425;
|
|
35
|
-
-(Math.pow(10, 8) * 24 * 60 * 60 * 1e3);
|
|
36
|
-
/**
|
|
37
|
-
* @constant
|
|
38
|
-
* @name millisecondsInWeek
|
|
39
|
-
* @summary Milliseconds in 1 week.
|
|
40
|
-
*/
|
|
41
|
-
var millisecondsInWeek = 6048e5;
|
|
42
|
-
/**
|
|
43
|
-
* @constant
|
|
44
|
-
* @name secondsInDay
|
|
45
|
-
* @summary Seconds in 1 day.
|
|
46
|
-
*/
|
|
47
|
-
var secondsInDay = 3600 * 24;
|
|
48
|
-
secondsInDay * 7;
|
|
49
|
-
secondsInDay * daysInYear / 12 * 3;
|
|
50
|
-
/**
|
|
51
|
-
* @constant
|
|
52
|
-
* @name constructFromSymbol
|
|
53
|
-
* @summary Symbol enabling Date extensions to inherit properties from the reference date.
|
|
54
|
-
*
|
|
55
|
-
* The symbol is used to enable the `constructFrom` function to construct a date
|
|
56
|
-
* using a reference date and a value. It allows to transfer extra properties
|
|
57
|
-
* from the reference date to the new date. It's useful for extensions like
|
|
58
|
-
* [`TZDate`](https://github.com/date-fns/tz) that accept a time zone as
|
|
59
|
-
* a constructor argument.
|
|
60
|
-
*/
|
|
61
|
-
var constructFromSymbol = Symbol.for("constructDateFrom");
|
|
62
|
-
//#endregion
|
|
63
|
-
//#region node_modules/date-fns/constructFrom.js
|
|
64
|
-
/**
|
|
65
|
-
* @name constructFrom
|
|
66
|
-
* @category Generic Helpers
|
|
67
|
-
* @summary Constructs a date using the reference date and the value
|
|
68
|
-
*
|
|
69
|
-
* @description
|
|
70
|
-
* The function constructs a new date using the constructor from the reference
|
|
71
|
-
* date and the given value. It helps to build generic functions that accept
|
|
72
|
-
* date extensions.
|
|
73
|
-
*
|
|
74
|
-
* It defaults to `Date` if the passed reference date is a number or a string.
|
|
75
|
-
*
|
|
76
|
-
* Starting from v3.7.0, it allows to construct a date using `[Symbol.for("constructDateFrom")]`
|
|
77
|
-
* enabling to transfer extra properties from the reference date to the new date.
|
|
78
|
-
* It's useful for extensions like [`TZDate`](https://github.com/date-fns/tz)
|
|
79
|
-
* that accept a time zone as a constructor argument.
|
|
80
|
-
*
|
|
81
|
-
* @typeParam DateType - The `Date` type, the function operates on. Gets inferred from passed arguments. Allows to use extensions like [`UTCDate`](https://github.com/date-fns/utc).
|
|
82
|
-
*
|
|
83
|
-
* @param date - The reference date to take constructor from
|
|
84
|
-
* @param value - The value to create the date
|
|
85
|
-
*
|
|
86
|
-
* @returns Date initialized using the given date and value
|
|
87
|
-
*
|
|
88
|
-
* @example
|
|
89
|
-
* import { constructFrom } from "./constructFrom/date-fns";
|
|
90
|
-
*
|
|
91
|
-
* // A function that clones a date preserving the original type
|
|
92
|
-
* function cloneDate<DateType extends Date>(date: DateType): DateType {
|
|
93
|
-
* return constructFrom(
|
|
94
|
-
* date, // Use constructor from the given date
|
|
95
|
-
* date.getTime() // Use the date value to create a new date
|
|
96
|
-
* );
|
|
97
|
-
* }
|
|
98
|
-
*/
|
|
99
|
-
function constructFrom(date, value) {
|
|
100
|
-
if (typeof date === "function") return date(value);
|
|
101
|
-
if (date && typeof date === "object" && constructFromSymbol in date) return date[constructFromSymbol](value);
|
|
102
|
-
if (date instanceof Date) return new date.constructor(value);
|
|
103
|
-
return new Date(value);
|
|
104
|
-
}
|
|
105
|
-
//#endregion
|
|
106
|
-
//#region node_modules/date-fns/toDate.js
|
|
107
|
-
/**
|
|
108
|
-
* @name toDate
|
|
109
|
-
* @category Common Helpers
|
|
110
|
-
* @summary Convert the given argument to an instance of Date.
|
|
111
|
-
*
|
|
112
|
-
* @description
|
|
113
|
-
* Convert the given argument to an instance of Date.
|
|
114
|
-
*
|
|
115
|
-
* If the argument is an instance of Date, the function returns its clone.
|
|
116
|
-
*
|
|
117
|
-
* If the argument is a number, it is treated as a timestamp.
|
|
118
|
-
*
|
|
119
|
-
* If the argument is none of the above, the function returns Invalid Date.
|
|
120
|
-
*
|
|
121
|
-
* Starting from v3.7.0, it clones a date using `[Symbol.for("constructDateFrom")]`
|
|
122
|
-
* enabling to transfer extra properties from the reference date to the new date.
|
|
123
|
-
* It's useful for extensions like [`TZDate`](https://github.com/date-fns/tz)
|
|
124
|
-
* that accept a time zone as a constructor argument.
|
|
125
|
-
*
|
|
126
|
-
* **Note**: *all* Date arguments passed to any *date-fns* function is processed by `toDate`.
|
|
127
|
-
*
|
|
128
|
-
* @typeParam DateType - The `Date` type, the function operates on. Gets inferred from passed arguments. Allows to use extensions like [`UTCDate`](https://github.com/date-fns/utc).
|
|
129
|
-
* @typeParam ResultDate - The result `Date` type, it is the type returned from the context function if it is passed, or inferred from the arguments.
|
|
130
|
-
*
|
|
131
|
-
* @param argument - The value to convert
|
|
132
|
-
*
|
|
133
|
-
* @returns The parsed date in the local time zone
|
|
134
|
-
*
|
|
135
|
-
* @example
|
|
136
|
-
* // Clone the date:
|
|
137
|
-
* const result = toDate(new Date(2014, 1, 11, 11, 30, 30))
|
|
138
|
-
* //=> Tue Feb 11 2014 11:30:30
|
|
139
|
-
*
|
|
140
|
-
* @example
|
|
141
|
-
* // Convert the timestamp to date:
|
|
142
|
-
* const result = toDate(1392098430000)
|
|
143
|
-
* //=> Tue Feb 11 2014 11:30:30
|
|
144
|
-
*/
|
|
145
|
-
function toDate(argument, context) {
|
|
146
|
-
return constructFrom(context || argument, argument);
|
|
147
|
-
}
|
|
148
|
-
//#endregion
|
|
149
|
-
//#region node_modules/date-fns/addDays.js
|
|
150
|
-
/**
|
|
151
|
-
* The {@link addDays} function options.
|
|
152
|
-
*/
|
|
153
|
-
/**
|
|
154
|
-
* @name addDays
|
|
155
|
-
* @category Day Helpers
|
|
156
|
-
* @summary Add the specified number of days to the given date.
|
|
157
|
-
*
|
|
158
|
-
* @description
|
|
159
|
-
* Add the specified number of days to the given date.
|
|
160
|
-
*
|
|
161
|
-
* @typeParam DateType - The `Date` type, the function operates on. Gets inferred from passed arguments. Allows to use extensions like [`UTCDate`](https://github.com/date-fns/utc).
|
|
162
|
-
* @typeParam ResultDate - The result `Date` type, it is the type returned from the context function if it is passed, or inferred from the arguments.
|
|
163
|
-
*
|
|
164
|
-
* @param date - The date to be changed
|
|
165
|
-
* @param amount - The amount of days to be added.
|
|
166
|
-
* @param options - An object with options
|
|
167
|
-
*
|
|
168
|
-
* @returns The new date with the days added
|
|
169
|
-
*
|
|
170
|
-
* @example
|
|
171
|
-
* // Add 10 days to 1 September 2014:
|
|
172
|
-
* const result = addDays(new Date(2014, 8, 1), 10)
|
|
173
|
-
* //=> Thu Sep 11 2014 00:00:00
|
|
174
|
-
*/
|
|
175
|
-
function addDays(date, amount, options) {
|
|
176
|
-
const _date = toDate(date, options?.in);
|
|
177
|
-
if (isNaN(amount)) return constructFrom(options?.in || date, NaN);
|
|
178
|
-
if (!amount) return _date;
|
|
179
|
-
_date.setDate(_date.getDate() + amount);
|
|
180
|
-
return _date;
|
|
181
|
-
}
|
|
182
|
-
//#endregion
|
|
183
|
-
//#region node_modules/date-fns/_lib/defaultOptions.js
|
|
184
|
-
var defaultOptions = {};
|
|
185
|
-
function getDefaultOptions() {
|
|
186
|
-
return defaultOptions;
|
|
187
|
-
}
|
|
188
|
-
//#endregion
|
|
189
|
-
//#region node_modules/date-fns/startOfWeek.js
|
|
190
|
-
/**
|
|
191
|
-
* The {@link startOfWeek} function options.
|
|
192
|
-
*/
|
|
193
|
-
/**
|
|
194
|
-
* @name startOfWeek
|
|
195
|
-
* @category Week Helpers
|
|
196
|
-
* @summary Return the start of a week for the given date.
|
|
197
|
-
*
|
|
198
|
-
* @description
|
|
199
|
-
* Return the start of a week for the given date.
|
|
200
|
-
* The result will be in the local timezone.
|
|
201
|
-
*
|
|
202
|
-
* @typeParam DateType - The `Date` type, the function operates on. Gets inferred from passed arguments. Allows to use extensions like [`UTCDate`](https://github.com/date-fns/utc).
|
|
203
|
-
* @typeParam ResultDate - The result `Date` type, it is the type returned from the context function if it is passed, or inferred from the arguments.
|
|
204
|
-
*
|
|
205
|
-
* @param date - The original date
|
|
206
|
-
* @param options - An object with options
|
|
207
|
-
*
|
|
208
|
-
* @returns The start of a week
|
|
209
|
-
*
|
|
210
|
-
* @example
|
|
211
|
-
* // The start of a week for 2 September 2014 11:55:00:
|
|
212
|
-
* const result = startOfWeek(new Date(2014, 8, 2, 11, 55, 0))
|
|
213
|
-
* //=> Sun Aug 31 2014 00:00:00
|
|
214
|
-
*
|
|
215
|
-
* @example
|
|
216
|
-
* // If the week starts on Monday, the start of the week for 2 September 2014 11:55:00:
|
|
217
|
-
* const result = startOfWeek(new Date(2014, 8, 2, 11, 55, 0), { weekStartsOn: 1 })
|
|
218
|
-
* //=> Mon Sep 01 2014 00:00:00
|
|
219
|
-
*/
|
|
220
|
-
function startOfWeek(date, options) {
|
|
221
|
-
const defaultOptions = getDefaultOptions();
|
|
222
|
-
const weekStartsOn = options?.weekStartsOn ?? options?.locale?.options?.weekStartsOn ?? defaultOptions.weekStartsOn ?? defaultOptions.locale?.options?.weekStartsOn ?? 0;
|
|
223
|
-
const _date = toDate(date, options?.in);
|
|
224
|
-
const day = _date.getDay();
|
|
225
|
-
const diff = (day < weekStartsOn ? 7 : 0) + day - weekStartsOn;
|
|
226
|
-
_date.setDate(_date.getDate() - diff);
|
|
227
|
-
_date.setHours(0, 0, 0, 0);
|
|
228
|
-
return _date;
|
|
229
|
-
}
|
|
230
|
-
//#endregion
|
|
231
|
-
//#region node_modules/date-fns/_lib/normalizeDates.js
|
|
232
|
-
function normalizeDates(context, ...dates) {
|
|
233
|
-
const normalize = constructFrom.bind(null, context || dates.find((date) => typeof date === "object"));
|
|
234
|
-
return dates.map(normalize);
|
|
235
|
-
}
|
|
236
|
-
//#endregion
|
|
237
|
-
//#region node_modules/date-fns/addWeeks.js
|
|
238
|
-
/**
|
|
239
|
-
* The {@link addWeeks} function options.
|
|
240
|
-
*/
|
|
241
|
-
/**
|
|
242
|
-
* @name addWeeks
|
|
243
|
-
* @category Week Helpers
|
|
244
|
-
* @summary Add the specified number of weeks to the given date.
|
|
245
|
-
*
|
|
246
|
-
* @description
|
|
247
|
-
* Add the specified number of weeks to the given date.
|
|
248
|
-
*
|
|
249
|
-
* @typeParam DateType - The `Date` type, the function operates on. Gets inferred from passed arguments. Allows to use extensions like [`UTCDate`](https://github.com/date-fns/utc).
|
|
250
|
-
* @typeParam ResultDate - The result `Date` type, it is the type returned from the context function if it is passed, or inferred from the arguments.
|
|
251
|
-
*
|
|
252
|
-
* @param date - The date to be changed
|
|
253
|
-
* @param amount - The amount of weeks to be added.
|
|
254
|
-
* @param options - An object with options
|
|
255
|
-
*
|
|
256
|
-
* @returns The new date with the weeks added
|
|
257
|
-
*
|
|
258
|
-
* @example
|
|
259
|
-
* // Add 4 weeks to 1 September 2014:
|
|
260
|
-
* const result = addWeeks(new Date(2014, 8, 1), 4)
|
|
261
|
-
* //=> Mon Sep 29 2014 00:00:00
|
|
262
|
-
*/
|
|
263
|
-
function addWeeks(date, amount, options) {
|
|
264
|
-
return addDays(date, amount * 7, options);
|
|
265
|
-
}
|
|
266
|
-
//#endregion
|
|
267
|
-
//#region node_modules/date-fns/endOfMonth.js
|
|
268
|
-
/**
|
|
269
|
-
* The {@link endOfMonth} function options.
|
|
270
|
-
*/
|
|
271
|
-
/**
|
|
272
|
-
* @name endOfMonth
|
|
273
|
-
* @category Month Helpers
|
|
274
|
-
* @summary Return the end of a month for the given date.
|
|
275
|
-
*
|
|
276
|
-
* @description
|
|
277
|
-
* Return the end of a month for the given date.
|
|
278
|
-
* The result will be in the local timezone.
|
|
279
|
-
*
|
|
280
|
-
* @typeParam DateType - The `Date` type, the function operates on. Gets inferred from passed arguments. Allows to use extensions like [`UTCDate`](https://github.com/date-fns/utc).
|
|
281
|
-
* @typeParam ResultDate - The result `Date` type, it is the type returned from the context function if it is passed, or inferred from the arguments.
|
|
282
|
-
*
|
|
283
|
-
* @param date - The original date
|
|
284
|
-
* @param options - An object with options
|
|
285
|
-
*
|
|
286
|
-
* @returns The end of a month
|
|
287
|
-
*
|
|
288
|
-
* @example
|
|
289
|
-
* // The end of a month for 2 September 2014 11:55:00:
|
|
290
|
-
* const result = endOfMonth(new Date(2014, 8, 2, 11, 55, 0))
|
|
291
|
-
* //=> Tue Sep 30 2014 23:59:59.999
|
|
292
|
-
*/
|
|
293
|
-
function endOfMonth(date, options) {
|
|
294
|
-
const _date = toDate(date, options?.in);
|
|
295
|
-
const month = _date.getMonth();
|
|
296
|
-
_date.setFullYear(_date.getFullYear(), month + 1, 0);
|
|
297
|
-
_date.setHours(23, 59, 59, 999);
|
|
298
|
-
return _date;
|
|
299
|
-
}
|
|
300
|
-
//#endregion
|
|
301
|
-
//#region node_modules/date-fns/_lib/normalizeInterval.js
|
|
302
|
-
function normalizeInterval(context, interval) {
|
|
303
|
-
const [start, end] = normalizeDates(context, interval.start, interval.end);
|
|
304
|
-
return {
|
|
305
|
-
start,
|
|
306
|
-
end
|
|
307
|
-
};
|
|
308
|
-
}
|
|
309
|
-
//#endregion
|
|
310
|
-
//#region node_modules/date-fns/eachWeekOfInterval.js
|
|
311
|
-
/**
|
|
312
|
-
* The {@link eachWeekOfInterval} function options.
|
|
313
|
-
*/
|
|
314
|
-
/**
|
|
315
|
-
* The {@link eachWeekOfInterval} function result type. It resolves the proper data type.
|
|
316
|
-
* It uses the first argument date object type, starting from the interval start date,
|
|
317
|
-
* then the end interval date. If a context function is passed, it uses the context function return type.
|
|
318
|
-
*/
|
|
319
|
-
/**
|
|
320
|
-
* @name eachWeekOfInterval
|
|
321
|
-
* @category Interval Helpers
|
|
322
|
-
* @summary Return the array of weeks within the specified time interval.
|
|
323
|
-
*
|
|
324
|
-
* @description
|
|
325
|
-
* Return the array of weeks within the specified time interval.
|
|
326
|
-
*
|
|
327
|
-
* @param interval - The interval.
|
|
328
|
-
* @param options - An object with options.
|
|
329
|
-
*
|
|
330
|
-
* @returns The array with starts of weeks from the week of the interval start to the week of the interval end
|
|
331
|
-
*
|
|
332
|
-
* @example
|
|
333
|
-
* // Each week within interval 6 October 2014 - 23 November 2014:
|
|
334
|
-
* const result = eachWeekOfInterval({
|
|
335
|
-
* start: new Date(2014, 9, 6),
|
|
336
|
-
* end: new Date(2014, 10, 23)
|
|
337
|
-
* })
|
|
338
|
-
* //=> [
|
|
339
|
-
* // Sun Oct 05 2014 00:00:00,
|
|
340
|
-
* // Sun Oct 12 2014 00:00:00,
|
|
341
|
-
* // Sun Oct 19 2014 00:00:00,
|
|
342
|
-
* // Sun Oct 26 2014 00:00:00,
|
|
343
|
-
* // Sun Nov 02 2014 00:00:00,
|
|
344
|
-
* // Sun Nov 09 2014 00:00:00,
|
|
345
|
-
* // Sun Nov 16 2014 00:00:00,
|
|
346
|
-
* // Sun Nov 23 2014 00:00:00
|
|
347
|
-
* // ]
|
|
348
|
-
*/
|
|
349
|
-
function eachWeekOfInterval(interval, options) {
|
|
350
|
-
const { start, end } = normalizeInterval(options?.in, interval);
|
|
351
|
-
let reversed = +start > +end;
|
|
352
|
-
const startDateWeek = reversed ? startOfWeek(end, options) : startOfWeek(start, options);
|
|
353
|
-
const endDateWeek = reversed ? startOfWeek(start, options) : startOfWeek(end, options);
|
|
354
|
-
startDateWeek.setHours(15);
|
|
355
|
-
endDateWeek.setHours(15);
|
|
356
|
-
const endTime = +endDateWeek.getTime();
|
|
357
|
-
let currentDate = startDateWeek;
|
|
358
|
-
let step = options?.step ?? 1;
|
|
359
|
-
if (!step) return [];
|
|
360
|
-
if (step < 0) {
|
|
361
|
-
step = -step;
|
|
362
|
-
reversed = !reversed;
|
|
363
|
-
}
|
|
364
|
-
const dates = [];
|
|
365
|
-
while (+currentDate <= endTime) {
|
|
366
|
-
currentDate.setHours(0);
|
|
367
|
-
dates.push(constructFrom(start, currentDate));
|
|
368
|
-
currentDate = addWeeks(currentDate, step);
|
|
369
|
-
currentDate.setHours(15);
|
|
370
|
-
}
|
|
371
|
-
return reversed ? dates.reverse() : dates;
|
|
372
|
-
}
|
|
373
|
-
//#endregion
|
|
374
|
-
//#region node_modules/date-fns/startOfMonth.js
|
|
375
|
-
/**
|
|
376
|
-
* The {@link startOfMonth} function options.
|
|
377
|
-
*/
|
|
378
|
-
/**
|
|
379
|
-
* @name startOfMonth
|
|
380
|
-
* @category Month Helpers
|
|
381
|
-
* @summary Return the start of a month for the given date.
|
|
382
|
-
*
|
|
383
|
-
* @description
|
|
384
|
-
* Return the start of a month for the given date. The result will be in the local timezone.
|
|
385
|
-
*
|
|
386
|
-
* @typeParam DateType - The `Date` type, the function operates on. Gets inferred from passed arguments.
|
|
387
|
-
* Allows to use extensions like [`UTCDate`](https://github.com/date-fns/utc).
|
|
388
|
-
* @typeParam ResultDate - The result `Date` type, it is the type returned from the context function if it is passed,
|
|
389
|
-
* or inferred from the arguments.
|
|
390
|
-
*
|
|
391
|
-
* @param date - The original date
|
|
392
|
-
* @param options - An object with options
|
|
393
|
-
*
|
|
394
|
-
* @returns The start of a month
|
|
395
|
-
*
|
|
396
|
-
* @example
|
|
397
|
-
* // The start of a month for 2 September 2014 11:55:00:
|
|
398
|
-
* const result = startOfMonth(new Date(2014, 8, 2, 11, 55, 0))
|
|
399
|
-
* //=> Mon Sep 01 2014 00:00:00
|
|
400
|
-
*/
|
|
401
|
-
function startOfMonth(date, options) {
|
|
402
|
-
const _date = toDate(date, options?.in);
|
|
403
|
-
_date.setDate(1);
|
|
404
|
-
_date.setHours(0, 0, 0, 0);
|
|
405
|
-
return _date;
|
|
406
|
-
}
|
|
407
|
-
//#endregion
|
|
408
|
-
//#region node_modules/date-fns/getWeekYear.js
|
|
409
|
-
/**
|
|
410
|
-
* The {@link getWeekYear} function options.
|
|
411
|
-
*/
|
|
412
|
-
/**
|
|
413
|
-
* @name getWeekYear
|
|
414
|
-
* @category Week-Numbering Year Helpers
|
|
415
|
-
* @summary Get the local week-numbering year of the given date.
|
|
416
|
-
*
|
|
417
|
-
* @description
|
|
418
|
-
* Get the local week-numbering year of the given date.
|
|
419
|
-
* The exact calculation depends on the values of
|
|
420
|
-
* `options.weekStartsOn` (which is the index of the first day of the week)
|
|
421
|
-
* and `options.firstWeekContainsDate` (which is the day of January, which is always in
|
|
422
|
-
* the first week of the week-numbering year)
|
|
423
|
-
*
|
|
424
|
-
* Week numbering: https://en.wikipedia.org/wiki/Week#The_ISO_week_date_system
|
|
425
|
-
*
|
|
426
|
-
* @param date - The given date
|
|
427
|
-
* @param options - An object with options.
|
|
428
|
-
*
|
|
429
|
-
* @returns The local week-numbering year
|
|
430
|
-
*
|
|
431
|
-
* @example
|
|
432
|
-
* // Which week numbering year is 26 December 2004 with the default settings?
|
|
433
|
-
* const result = getWeekYear(new Date(2004, 11, 26))
|
|
434
|
-
* //=> 2005
|
|
435
|
-
*
|
|
436
|
-
* @example
|
|
437
|
-
* // Which week numbering year is 26 December 2004 if week starts on Saturday?
|
|
438
|
-
* const result = getWeekYear(new Date(2004, 11, 26), { weekStartsOn: 6 })
|
|
439
|
-
* //=> 2004
|
|
440
|
-
*
|
|
441
|
-
* @example
|
|
442
|
-
* // Which week numbering year is 26 December 2004 if the first week contains 4 January?
|
|
443
|
-
* const result = getWeekYear(new Date(2004, 11, 26), { firstWeekContainsDate: 4 })
|
|
444
|
-
* //=> 2004
|
|
445
|
-
*/
|
|
446
|
-
function getWeekYear(date, options) {
|
|
447
|
-
const _date = toDate(date, options?.in);
|
|
448
|
-
const year = _date.getFullYear();
|
|
449
|
-
const defaultOptions = getDefaultOptions();
|
|
450
|
-
const firstWeekContainsDate = options?.firstWeekContainsDate ?? options?.locale?.options?.firstWeekContainsDate ?? defaultOptions.firstWeekContainsDate ?? defaultOptions.locale?.options?.firstWeekContainsDate ?? 1;
|
|
451
|
-
const firstWeekOfNextYear = constructFrom(options?.in || date, 0);
|
|
452
|
-
firstWeekOfNextYear.setFullYear(year + 1, 0, firstWeekContainsDate);
|
|
453
|
-
firstWeekOfNextYear.setHours(0, 0, 0, 0);
|
|
454
|
-
const startOfNextYear = startOfWeek(firstWeekOfNextYear, options);
|
|
455
|
-
const firstWeekOfThisYear = constructFrom(options?.in || date, 0);
|
|
456
|
-
firstWeekOfThisYear.setFullYear(year, 0, firstWeekContainsDate);
|
|
457
|
-
firstWeekOfThisYear.setHours(0, 0, 0, 0);
|
|
458
|
-
const startOfThisYear = startOfWeek(firstWeekOfThisYear, options);
|
|
459
|
-
if (+_date >= +startOfNextYear) return year + 1;
|
|
460
|
-
else if (+_date >= +startOfThisYear) return year;
|
|
461
|
-
else return year - 1;
|
|
462
|
-
}
|
|
463
|
-
//#endregion
|
|
464
|
-
//#region node_modules/date-fns/startOfWeekYear.js
|
|
465
|
-
/**
|
|
466
|
-
* The {@link startOfWeekYear} function options.
|
|
467
|
-
*/
|
|
468
|
-
/**
|
|
469
|
-
* @name startOfWeekYear
|
|
470
|
-
* @category Week-Numbering Year Helpers
|
|
471
|
-
* @summary Return the start of a local week-numbering year for the given date.
|
|
472
|
-
*
|
|
473
|
-
* @description
|
|
474
|
-
* Return the start of a local week-numbering year.
|
|
475
|
-
* The exact calculation depends on the values of
|
|
476
|
-
* `options.weekStartsOn` (which is the index of the first day of the week)
|
|
477
|
-
* and `options.firstWeekContainsDate` (which is the day of January, which is always in
|
|
478
|
-
* the first week of the week-numbering year)
|
|
479
|
-
*
|
|
480
|
-
* Week numbering: https://en.wikipedia.org/wiki/Week#The_ISO_week_date_system
|
|
481
|
-
*
|
|
482
|
-
* @typeParam DateType - The `Date` type, the function operates on. Gets inferred from passed arguments. Allows to use extensions like [`UTCDate`](https://github.com/date-fns/utc).
|
|
483
|
-
* @typeParam ResultDate - The result `Date` type.
|
|
484
|
-
*
|
|
485
|
-
* @param date - The original date
|
|
486
|
-
* @param options - An object with options
|
|
487
|
-
*
|
|
488
|
-
* @returns The start of a week-numbering year
|
|
489
|
-
*
|
|
490
|
-
* @example
|
|
491
|
-
* // The start of an a week-numbering year for 2 July 2005 with default settings:
|
|
492
|
-
* const result = startOfWeekYear(new Date(2005, 6, 2))
|
|
493
|
-
* //=> Sun Dec 26 2004 00:00:00
|
|
494
|
-
*
|
|
495
|
-
* @example
|
|
496
|
-
* // The start of a week-numbering year for 2 July 2005
|
|
497
|
-
* // if Monday is the first day of week
|
|
498
|
-
* // and 4 January is always in the first week of the year:
|
|
499
|
-
* const result = startOfWeekYear(new Date(2005, 6, 2), {
|
|
500
|
-
* weekStartsOn: 1,
|
|
501
|
-
* firstWeekContainsDate: 4
|
|
502
|
-
* })
|
|
503
|
-
* //=> Mon Jan 03 2005 00:00:00
|
|
504
|
-
*/
|
|
505
|
-
function startOfWeekYear(date, options) {
|
|
506
|
-
const defaultOptions = getDefaultOptions();
|
|
507
|
-
const firstWeekContainsDate = options?.firstWeekContainsDate ?? options?.locale?.options?.firstWeekContainsDate ?? defaultOptions.firstWeekContainsDate ?? defaultOptions.locale?.options?.firstWeekContainsDate ?? 1;
|
|
508
|
-
const year = getWeekYear(date, options);
|
|
509
|
-
const firstWeek = constructFrom(options?.in || date, 0);
|
|
510
|
-
firstWeek.setFullYear(year, 0, firstWeekContainsDate);
|
|
511
|
-
firstWeek.setHours(0, 0, 0, 0);
|
|
512
|
-
return startOfWeek(firstWeek, options);
|
|
513
|
-
}
|
|
514
|
-
//#endregion
|
|
515
|
-
//#region node_modules/date-fns/getWeek.js
|
|
516
|
-
/**
|
|
517
|
-
* The {@link getWeek} function options.
|
|
518
|
-
*/
|
|
519
|
-
/**
|
|
520
|
-
* @name getWeek
|
|
521
|
-
* @category Week Helpers
|
|
522
|
-
* @summary Get the local week index of the given date.
|
|
523
|
-
*
|
|
524
|
-
* @description
|
|
525
|
-
* Get the local week index of the given date.
|
|
526
|
-
* The exact calculation depends on the values of
|
|
527
|
-
* `options.weekStartsOn` (which is the index of the first day of the week)
|
|
528
|
-
* and `options.firstWeekContainsDate` (which is the day of January, which is always in
|
|
529
|
-
* the first week of the week-numbering year)
|
|
530
|
-
*
|
|
531
|
-
* Week numbering: https://en.wikipedia.org/wiki/Week#The_ISO_week_date_system
|
|
532
|
-
*
|
|
533
|
-
* @param date - The given date
|
|
534
|
-
* @param options - An object with options
|
|
535
|
-
*
|
|
536
|
-
* @returns The week
|
|
537
|
-
*
|
|
538
|
-
* @example
|
|
539
|
-
* // Which week of the local week numbering year is 2 January 2005 with default options?
|
|
540
|
-
* const result = getWeek(new Date(2005, 0, 2))
|
|
541
|
-
* //=> 2
|
|
542
|
-
*
|
|
543
|
-
* @example
|
|
544
|
-
* // Which week of the local week numbering year is 2 January 2005,
|
|
545
|
-
* // if Monday is the first day of the week,
|
|
546
|
-
* // and the first week of the year always contains 4 January?
|
|
547
|
-
* const result = getWeek(new Date(2005, 0, 2), {
|
|
548
|
-
* weekStartsOn: 1,
|
|
549
|
-
* firstWeekContainsDate: 4
|
|
550
|
-
* })
|
|
551
|
-
* //=> 53
|
|
552
|
-
*/
|
|
553
|
-
function getWeek(date, options) {
|
|
554
|
-
const _date = toDate(date, options?.in);
|
|
555
|
-
const diff = +startOfWeek(_date, options) - +startOfWeekYear(_date, options);
|
|
556
|
-
return Math.round(diff / millisecondsInWeek) + 1;
|
|
557
|
-
}
|
|
558
|
-
//#endregion
|
|
559
|
-
//#region src/elements/calendar/calendar/calendar.scss?lit&inline
|
|
560
|
-
var calendar_default = css`:host {
|
|
561
|
-
display: block;
|
|
562
|
-
width: max-content;
|
|
563
|
-
--sbb-calendar-cell-size: 2.75rem;
|
|
564
|
-
--sbb-calendar-cell-border-radius: calc(
|
|
565
|
-
var(--sbb-border-radius-4x) + var(--sbb-calendar-cell-border-width)
|
|
566
|
-
);
|
|
567
|
-
--sbb-calendar-cell-transition-duration: var(
|
|
568
|
-
--sbb-disable-animation-duration,
|
|
569
|
-
var(--sbb-animation-duration-2x)
|
|
570
|
-
);
|
|
571
|
-
}
|
|
572
|
-
@media (min-width: calc(37.5rem)) {
|
|
573
|
-
:host {
|
|
574
|
-
--sbb-calendar-control-view-change-height: 3rem;
|
|
575
|
-
}
|
|
576
|
-
}
|
|
577
|
-
|
|
578
|
-
:host(:is(:state(enhanced),[state--enhanced])) {
|
|
579
|
-
--sbb-calendar-cell-padding-block-end: var(--sbb-spacing-fixed-3x);
|
|
580
|
-
}
|
|
581
|
-
|
|
582
|
-
.sbb-calendar__wrapper {
|
|
583
|
-
width: 100%;
|
|
584
|
-
display: block;
|
|
585
|
-
transition-duration: var(--sbb-calendar-cell-transition-duration);
|
|
586
|
-
}
|
|
587
|
-
|
|
588
|
-
.sbb-calendar__controls {
|
|
589
|
-
width: 100%;
|
|
590
|
-
display: inline-flex;
|
|
591
|
-
align-items: center;
|
|
592
|
-
gap: var(--sbb-calendar-control-gap);
|
|
593
|
-
margin-block-end: var(--sbb-calendar-control-margin-block-end);
|
|
594
|
-
}
|
|
595
|
-
|
|
596
|
-
.sbb-calendar__controls-month {
|
|
597
|
-
width: 100%;
|
|
598
|
-
display: flex;
|
|
599
|
-
gap: var(--sbb-calendar-tables-gap);
|
|
600
|
-
}
|
|
601
|
-
|
|
602
|
-
#sbb-calendar__controls-previous,
|
|
603
|
-
#sbb-calendar__controls-next {
|
|
604
|
-
-webkit-tap-highlight-color: transparent;
|
|
605
|
-
}
|
|
606
|
-
|
|
607
|
-
.sbb-calendar__controls-change-date {
|
|
608
|
-
appearance: none;
|
|
609
|
-
box-sizing: border-box;
|
|
610
|
-
margin: 0;
|
|
611
|
-
outline: none;
|
|
612
|
-
border: none;
|
|
613
|
-
border-radius: 0;
|
|
614
|
-
padding: 0;
|
|
615
|
-
background: none;
|
|
616
|
-
-webkit-tap-highlight-color: transparent;
|
|
617
|
-
color: inherit;
|
|
618
|
-
display: flex;
|
|
619
|
-
align-items: center;
|
|
620
|
-
margin: auto;
|
|
621
|
-
height: var(--sbb-calendar-control-view-change-height);
|
|
622
|
-
font-size: var(--sbb-text-font-size-s);
|
|
623
|
-
letter-spacing: var(--sbb-typo-letter-spacing-text);
|
|
624
|
-
text-transform: capitalize;
|
|
625
|
-
cursor: var(--sbb-cursor-pointer);
|
|
626
|
-
padding-inline: var(--sbb-calendar-control-view-change-padding-inline);
|
|
627
|
-
border-radius: var(--sbb-border-radius-infinity);
|
|
628
|
-
background-color: var(--sbb-calendar-control-view-change-background);
|
|
629
|
-
color: var(--sbb-calendar-control-view-change-color);
|
|
630
|
-
transition-duration: var(--sbb-calendar-cell-transition-duration);
|
|
631
|
-
transition-timing-function: var(--sbb-calendar-cell-transition-easing-function);
|
|
632
|
-
transition-property: background-color, padding-block-end;
|
|
633
|
-
}
|
|
634
|
-
.sbb-calendar__controls-change-date:disabled {
|
|
635
|
-
--sbb-calendar-control-view-change-background: var(--sbb-color-milk);
|
|
636
|
-
--sbb-calendar-control-view-change-background: light-dark(
|
|
637
|
-
var(--sbb-color-milk),
|
|
638
|
-
var(--sbb-color-anthracite)
|
|
639
|
-
);
|
|
640
|
-
--sbb-calendar-control-view-change-color: var(--sbb-color-granite);
|
|
641
|
-
--sbb-calendar-control-view-change-color: light-dark(
|
|
642
|
-
var(--sbb-color-granite),
|
|
643
|
-
var(--sbb-color-aluminium)
|
|
644
|
-
);
|
|
645
|
-
cursor: unset;
|
|
646
|
-
}
|
|
647
|
-
.sbb-calendar__controls-change-date:focus-visible {
|
|
648
|
-
outline-offset: var(--sbb-focus-outline-offset);
|
|
649
|
-
outline: var(--sbb-focus-outline-color) var(--sbb-focus-outline-style, solid) var(--sbb-focus-outline-width);
|
|
650
|
-
outline-offset: var(--sbb-spacing-fixed-1x);
|
|
651
|
-
}
|
|
652
|
-
@media (any-hover: hover) {
|
|
653
|
-
.sbb-calendar__controls-change-date:not(:active, :disabled):hover {
|
|
654
|
-
padding-block-end: var(--sbb-calendar-cell-hover-shift);
|
|
655
|
-
}
|
|
656
|
-
}
|
|
657
|
-
.sbb-calendar__controls-change-date:not(:disabled):active {
|
|
658
|
-
--sbb-calendar-control-view-change-background: var(--sbb-background-color-3);
|
|
659
|
-
}
|
|
660
|
-
|
|
661
|
-
.sbb-calendar__table-month-view,
|
|
662
|
-
.sbb-calendar__table-year-view {
|
|
663
|
-
--sbb-calendar-table-column-spaces: 6;
|
|
664
|
-
}
|
|
665
|
-
|
|
666
|
-
.sbb-calendar__table-overflow-break {
|
|
667
|
-
contain: layout;
|
|
668
|
-
}
|
|
669
|
-
|
|
670
|
-
.sbb-calendar__table-container {
|
|
671
|
-
display: flex;
|
|
672
|
-
gap: var(--sbb-calendar-tables-gap);
|
|
673
|
-
margin-inline: var(--sbb-calendar-margin);
|
|
674
|
-
--sbb-calendar-min-width: calc(7 * var(--sbb-calendar-cell-size));
|
|
675
|
-
--sbb-calendar-overflow: calc(100% - var(--sbb-calendar-min-width));
|
|
676
|
-
--sbb-calendar-start-offset: min(
|
|
677
|
-
0px,
|
|
678
|
-
-1 * (var(--sbb-calendar-overflow) / var(--sbb-calendar-table-column-spaces))
|
|
679
|
-
);
|
|
680
|
-
--sbb-calendar-margin: var(--sbb-calendar-start-offset);
|
|
681
|
-
}
|
|
682
|
-
:host(:is(:state(wide),[state--wide])) .sbb-calendar__table-container {
|
|
683
|
-
--sbb-calendar-min-width: calc(
|
|
684
|
-
2 * 7 * var(--sbb-calendar-cell-size) + var(--sbb-calendar-tables-gap)
|
|
685
|
-
);
|
|
686
|
-
--sbb-calendar-margin: calc(0.5 * var(--sbb-calendar-start-offset));
|
|
687
|
-
}
|
|
688
|
-
:host([orientation=horizontal][week-numbers]) .sbb-calendar__table-container {
|
|
689
|
-
--sbb-calendar-min-width: calc(8 * var(--sbb-calendar-cell-size));
|
|
690
|
-
}
|
|
691
|
-
:host([orientation=horizontal][week-numbers]:is(:state(wide),[state--wide])) .sbb-calendar__table-container {
|
|
692
|
-
--sbb-calendar-min-width: calc(
|
|
693
|
-
2 * 8 * var(--sbb-calendar-cell-size) + var(--sbb-calendar-tables-gap)
|
|
694
|
-
);
|
|
695
|
-
}
|
|
696
|
-
:host([orientation=vertical]) .sbb-calendar__table-container {
|
|
697
|
-
min-width: var(--sbb-calendar-min-width);
|
|
698
|
-
--sbb-calendar-start-offset: 0;
|
|
699
|
-
}
|
|
700
|
-
:host([orientation=vertical]:is(:state(wide),[state--wide])) .sbb-calendar__table-container {
|
|
701
|
-
--sbb-calendar-min-width: calc(
|
|
702
|
-
13 * var(--sbb-calendar-cell-size) + var(--sbb-calendar-tables-gap)
|
|
703
|
-
);
|
|
704
|
-
}
|
|
705
|
-
|
|
706
|
-
.sbb-calendar__table {
|
|
707
|
-
width: 100%;
|
|
708
|
-
border-collapse: collapse;
|
|
709
|
-
height: max-content;
|
|
710
|
-
animation-name: show;
|
|
711
|
-
animation-duration: var(--sbb-calendar-table-animation-duration);
|
|
712
|
-
}
|
|
713
|
-
.sbb-calendar__table.sbb-calendar__table-hide {
|
|
714
|
-
--sbb-calendar-cell-transition-duration: 0ms;
|
|
715
|
-
animation-name: hide;
|
|
716
|
-
animation-duration: var(--sbb-calendar-table-animation-duration);
|
|
717
|
-
}
|
|
718
|
-
:host(:not(:is(:state(wide),[state--wide]))) .sbb-calendar__table {
|
|
719
|
-
min-width: 100%;
|
|
720
|
-
}
|
|
721
|
-
|
|
722
|
-
.sbb-calendar__table-header {
|
|
723
|
-
text-align: center;
|
|
724
|
-
}
|
|
725
|
-
|
|
726
|
-
.sbb-calendar__table-body {
|
|
727
|
-
text-align: center;
|
|
728
|
-
}
|
|
729
|
-
|
|
730
|
-
.sbb-calendar__table-header-cell,
|
|
731
|
-
.sbb-calendar__table-header-cell-vertical {
|
|
732
|
-
width: var(--sbb-calendar-cell-size);
|
|
733
|
-
color: var(--sbb-calendar-header-color);
|
|
734
|
-
padding: 0;
|
|
735
|
-
font-size: var(--sbb-text-font-size-xs);
|
|
736
|
-
font-weight: normal;
|
|
737
|
-
letter-spacing: var(--sbb-typo-letter-spacing-text);
|
|
738
|
-
}
|
|
739
|
-
:host(:not([multiple])) .sbb-calendar__table-header-cell,
|
|
740
|
-
:host(:not([multiple])) .sbb-calendar__table-header-cell-vertical {
|
|
741
|
-
height: var(--sbb-calendar-cell-size);
|
|
742
|
-
}
|
|
743
|
-
|
|
744
|
-
.sbb-calendar__table-header-cell-vertical {
|
|
745
|
-
padding-block-end: var(--sbb-calendar-cell-padding-block-end);
|
|
746
|
-
}
|
|
747
|
-
|
|
748
|
-
.sbb-calendar__table-data {
|
|
749
|
-
position: relative;
|
|
750
|
-
padding: 0;
|
|
751
|
-
text-align: center;
|
|
752
|
-
}
|
|
753
|
-
|
|
754
|
-
.sbb-calendar__day-cell {
|
|
755
|
-
padding-block-end: var(--sbb-calendar-cell-padding-block-end);
|
|
756
|
-
}
|
|
757
|
-
|
|
758
|
-
:host(:not(:is(:state(enhanced),[state--enhanced]))) sbb-calendar-day {
|
|
759
|
-
--sbb-calendar-cell-justify-content: center;
|
|
760
|
-
--sbb-calendar-day-height: 2.75rem;
|
|
761
|
-
--sbb-calendar-day-extra-display: none;
|
|
762
|
-
--sbb-calendar-day-value-height: unset;
|
|
763
|
-
--sbb-calendar-day-crossed-out-top: 50%;
|
|
764
|
-
--sbb-calendar-day-crossed-out-translate: translate(-50%, -50%) rotate(-45deg);
|
|
765
|
-
}
|
|
766
|
-
|
|
767
|
-
@keyframes show {
|
|
768
|
-
from {
|
|
769
|
-
opacity: 0;
|
|
770
|
-
transform: translateY(var(--sbb-calendar-table-animation-shift));
|
|
771
|
-
}
|
|
772
|
-
to {
|
|
773
|
-
opacity: 1;
|
|
774
|
-
transform: translateY(0%);
|
|
775
|
-
}
|
|
776
|
-
}
|
|
777
|
-
@keyframes hide {
|
|
778
|
-
from {
|
|
779
|
-
opacity: 1;
|
|
780
|
-
transform: translateY(0%);
|
|
781
|
-
}
|
|
782
|
-
to {
|
|
783
|
-
opacity: 0;
|
|
784
|
-
transform: translateY(var(--sbb-calendar-table-animation-shift));
|
|
785
|
-
}
|
|
786
|
-
}`;
|
|
787
|
-
//#endregion
|
|
788
|
-
//#region src/elements/calendar/calendar/calendar.component.ts
|
|
789
|
-
SbbSecondaryButtonElement.define();
|
|
790
|
-
SbbCalendarDayElement.define();
|
|
791
|
-
SbbCalendarMonthElement.define();
|
|
792
|
-
SbbCalendarYearElement.define();
|
|
793
|
-
SbbCalendarWeekdayElement.define();
|
|
794
|
-
SbbCalendarWeeknumberElement.define();
|
|
795
|
-
var SbbMonthChangeEvent = class extends Event {
|
|
796
|
-
get range() {
|
|
797
|
-
return this._range;
|
|
798
|
-
}
|
|
799
|
-
constructor(range) {
|
|
800
|
-
super("monthchange", {
|
|
801
|
-
bubbles: true,
|
|
802
|
-
composed: true
|
|
803
|
-
});
|
|
804
|
-
this._range = Object.freeze(range || []);
|
|
805
|
-
}
|
|
806
|
-
};
|
|
807
|
-
/**
|
|
808
|
-
* It displays a calendar which allows choosing a date.
|
|
809
|
-
*
|
|
810
|
-
* @slot - Use the unnamed slot to add customized `sbb-calendar-day` elements.
|
|
811
|
-
*/
|
|
812
|
-
var SbbCalendarElement = (() => {
|
|
813
|
-
let _classSuper = SbbElement;
|
|
814
|
-
let _instanceExtraInitializers = [];
|
|
815
|
-
let _wide_decorators;
|
|
816
|
-
let _wide_initializers = [];
|
|
817
|
-
let _wide_extraInitializers = [];
|
|
818
|
-
let _view_decorators;
|
|
819
|
-
let _view_initializers = [];
|
|
820
|
-
let _view_extraInitializers = [];
|
|
821
|
-
let _min_decorators;
|
|
822
|
-
let _min_initializers = [];
|
|
823
|
-
let _min_extraInitializers = [];
|
|
824
|
-
let _max_decorators;
|
|
825
|
-
let _max_initializers = [];
|
|
826
|
-
let _max_extraInitializers = [];
|
|
827
|
-
let _multiple_decorators;
|
|
828
|
-
let _multiple_initializers = [];
|
|
829
|
-
let _multiple_extraInitializers = [];
|
|
830
|
-
let _set_selected_decorators;
|
|
831
|
-
let __selected_decorators;
|
|
832
|
-
let __selected_initializers = [];
|
|
833
|
-
let __selected_extraInitializers = [];
|
|
834
|
-
let _dateFilter_decorators;
|
|
835
|
-
let _dateFilter_initializers = [];
|
|
836
|
-
let _dateFilter_extraInitializers = [];
|
|
837
|
-
let _orientation_decorators;
|
|
838
|
-
let _orientation_initializers = [];
|
|
839
|
-
let _orientation_extraInitializers = [];
|
|
840
|
-
let _weekNumbers_decorators;
|
|
841
|
-
let _weekNumbers_initializers = [];
|
|
842
|
-
let _weekNumbers_extraInitializers = [];
|
|
843
|
-
let __activeDate_decorators;
|
|
844
|
-
let __activeDate_initializers = [];
|
|
845
|
-
let __activeDate_extraInitializers = [];
|
|
846
|
-
let _set__wide_decorators;
|
|
847
|
-
let __calendarView_decorators;
|
|
848
|
-
let __calendarView_initializers = [];
|
|
849
|
-
let __calendarView_extraInitializers = [];
|
|
850
|
-
let __initialized_decorators;
|
|
851
|
-
let __initialized_initializers = [];
|
|
852
|
-
let __initialized_extraInitializers = [];
|
|
853
|
-
return class SbbCalendarElement extends _classSuper {
|
|
854
|
-
static {
|
|
855
|
-
const _metadata = typeof Symbol === "function" && Symbol.metadata ? Object.create(_classSuper[Symbol.metadata] ?? null) : void 0;
|
|
856
|
-
_wide_decorators = [forceType(), property({ type: Boolean })];
|
|
857
|
-
_view_decorators = [property()];
|
|
858
|
-
_min_decorators = [plainDate(), property()];
|
|
859
|
-
_max_decorators = [plainDate(), property()];
|
|
860
|
-
_multiple_decorators = [
|
|
861
|
-
forceType(),
|
|
862
|
-
handleDistinctChange((e, newValue) => e._onMultipleChanged(newValue)),
|
|
863
|
-
property({ type: Boolean })
|
|
864
|
-
];
|
|
865
|
-
_set_selected_decorators = [property()];
|
|
866
|
-
__selected_decorators = [state()];
|
|
867
|
-
_dateFilter_decorators = [property({ attribute: "date-filter" })];
|
|
868
|
-
_orientation_decorators = [property({ reflect: true })];
|
|
869
|
-
_weekNumbers_decorators = [forceType(), property({
|
|
870
|
-
attribute: "week-numbers",
|
|
871
|
-
type: Boolean
|
|
872
|
-
})];
|
|
873
|
-
__activeDate_decorators = [state()];
|
|
874
|
-
_set__wide_decorators = [state()];
|
|
875
|
-
__calendarView_decorators = [state()];
|
|
876
|
-
__initialized_decorators = [state()];
|
|
877
|
-
__esDecorate(this, null, _wide_decorators, {
|
|
878
|
-
kind: "accessor",
|
|
879
|
-
name: "wide",
|
|
880
|
-
static: false,
|
|
881
|
-
private: false,
|
|
882
|
-
access: {
|
|
883
|
-
has: (obj) => "wide" in obj,
|
|
884
|
-
get: (obj) => obj.wide,
|
|
885
|
-
set: (obj, value) => {
|
|
886
|
-
obj.wide = value;
|
|
887
|
-
}
|
|
888
|
-
},
|
|
889
|
-
metadata: _metadata
|
|
890
|
-
}, _wide_initializers, _wide_extraInitializers);
|
|
891
|
-
__esDecorate(this, null, _view_decorators, {
|
|
892
|
-
kind: "accessor",
|
|
893
|
-
name: "view",
|
|
894
|
-
static: false,
|
|
895
|
-
private: false,
|
|
896
|
-
access: {
|
|
897
|
-
has: (obj) => "view" in obj,
|
|
898
|
-
get: (obj) => obj.view,
|
|
899
|
-
set: (obj, value) => {
|
|
900
|
-
obj.view = value;
|
|
901
|
-
}
|
|
902
|
-
},
|
|
903
|
-
metadata: _metadata
|
|
904
|
-
}, _view_initializers, _view_extraInitializers);
|
|
905
|
-
__esDecorate(this, null, _min_decorators, {
|
|
906
|
-
kind: "accessor",
|
|
907
|
-
name: "min",
|
|
908
|
-
static: false,
|
|
909
|
-
private: false,
|
|
910
|
-
access: {
|
|
911
|
-
has: (obj) => "min" in obj,
|
|
912
|
-
get: (obj) => obj.min,
|
|
913
|
-
set: (obj, value) => {
|
|
914
|
-
obj.min = value;
|
|
915
|
-
}
|
|
916
|
-
},
|
|
917
|
-
metadata: _metadata
|
|
918
|
-
}, _min_initializers, _min_extraInitializers);
|
|
919
|
-
__esDecorate(this, null, _max_decorators, {
|
|
920
|
-
kind: "accessor",
|
|
921
|
-
name: "max",
|
|
922
|
-
static: false,
|
|
923
|
-
private: false,
|
|
924
|
-
access: {
|
|
925
|
-
has: (obj) => "max" in obj,
|
|
926
|
-
get: (obj) => obj.max,
|
|
927
|
-
set: (obj, value) => {
|
|
928
|
-
obj.max = value;
|
|
929
|
-
}
|
|
930
|
-
},
|
|
931
|
-
metadata: _metadata
|
|
932
|
-
}, _max_initializers, _max_extraInitializers);
|
|
933
|
-
__esDecorate(this, null, _multiple_decorators, {
|
|
934
|
-
kind: "accessor",
|
|
935
|
-
name: "multiple",
|
|
936
|
-
static: false,
|
|
937
|
-
private: false,
|
|
938
|
-
access: {
|
|
939
|
-
has: (obj) => "multiple" in obj,
|
|
940
|
-
get: (obj) => obj.multiple,
|
|
941
|
-
set: (obj, value) => {
|
|
942
|
-
obj.multiple = value;
|
|
943
|
-
}
|
|
944
|
-
},
|
|
945
|
-
metadata: _metadata
|
|
946
|
-
}, _multiple_initializers, _multiple_extraInitializers);
|
|
947
|
-
__esDecorate(this, null, _set_selected_decorators, {
|
|
948
|
-
kind: "setter",
|
|
949
|
-
name: "selected",
|
|
950
|
-
static: false,
|
|
951
|
-
private: false,
|
|
952
|
-
access: {
|
|
953
|
-
has: (obj) => "selected" in obj,
|
|
954
|
-
set: (obj, value) => {
|
|
955
|
-
obj.selected = value;
|
|
956
|
-
}
|
|
957
|
-
},
|
|
958
|
-
metadata: _metadata
|
|
959
|
-
}, null, _instanceExtraInitializers);
|
|
960
|
-
__esDecorate(this, null, __selected_decorators, {
|
|
961
|
-
kind: "accessor",
|
|
962
|
-
name: "_selected",
|
|
963
|
-
static: false,
|
|
964
|
-
private: false,
|
|
965
|
-
access: {
|
|
966
|
-
has: (obj) => "_selected" in obj,
|
|
967
|
-
get: (obj) => obj._selected,
|
|
968
|
-
set: (obj, value) => {
|
|
969
|
-
obj._selected = value;
|
|
970
|
-
}
|
|
971
|
-
},
|
|
972
|
-
metadata: _metadata
|
|
973
|
-
}, __selected_initializers, __selected_extraInitializers);
|
|
974
|
-
__esDecorate(this, null, _dateFilter_decorators, {
|
|
975
|
-
kind: "accessor",
|
|
976
|
-
name: "dateFilter",
|
|
977
|
-
static: false,
|
|
978
|
-
private: false,
|
|
979
|
-
access: {
|
|
980
|
-
has: (obj) => "dateFilter" in obj,
|
|
981
|
-
get: (obj) => obj.dateFilter,
|
|
982
|
-
set: (obj, value) => {
|
|
983
|
-
obj.dateFilter = value;
|
|
984
|
-
}
|
|
985
|
-
},
|
|
986
|
-
metadata: _metadata
|
|
987
|
-
}, _dateFilter_initializers, _dateFilter_extraInitializers);
|
|
988
|
-
__esDecorate(this, null, _orientation_decorators, {
|
|
989
|
-
kind: "accessor",
|
|
990
|
-
name: "orientation",
|
|
991
|
-
static: false,
|
|
992
|
-
private: false,
|
|
993
|
-
access: {
|
|
994
|
-
has: (obj) => "orientation" in obj,
|
|
995
|
-
get: (obj) => obj.orientation,
|
|
996
|
-
set: (obj, value) => {
|
|
997
|
-
obj.orientation = value;
|
|
998
|
-
}
|
|
999
|
-
},
|
|
1000
|
-
metadata: _metadata
|
|
1001
|
-
}, _orientation_initializers, _orientation_extraInitializers);
|
|
1002
|
-
__esDecorate(this, null, _weekNumbers_decorators, {
|
|
1003
|
-
kind: "accessor",
|
|
1004
|
-
name: "weekNumbers",
|
|
1005
|
-
static: false,
|
|
1006
|
-
private: false,
|
|
1007
|
-
access: {
|
|
1008
|
-
has: (obj) => "weekNumbers" in obj,
|
|
1009
|
-
get: (obj) => obj.weekNumbers,
|
|
1010
|
-
set: (obj, value) => {
|
|
1011
|
-
obj.weekNumbers = value;
|
|
1012
|
-
}
|
|
1013
|
-
},
|
|
1014
|
-
metadata: _metadata
|
|
1015
|
-
}, _weekNumbers_initializers, _weekNumbers_extraInitializers);
|
|
1016
|
-
__esDecorate(this, null, __activeDate_decorators, {
|
|
1017
|
-
kind: "accessor",
|
|
1018
|
-
name: "_activeDate",
|
|
1019
|
-
static: false,
|
|
1020
|
-
private: false,
|
|
1021
|
-
access: {
|
|
1022
|
-
has: (obj) => "_activeDate" in obj,
|
|
1023
|
-
get: (obj) => obj._activeDate,
|
|
1024
|
-
set: (obj, value) => {
|
|
1025
|
-
obj._activeDate = value;
|
|
1026
|
-
}
|
|
1027
|
-
},
|
|
1028
|
-
metadata: _metadata
|
|
1029
|
-
}, __activeDate_initializers, __activeDate_extraInitializers);
|
|
1030
|
-
__esDecorate(this, null, _set__wide_decorators, {
|
|
1031
|
-
kind: "setter",
|
|
1032
|
-
name: "_wide",
|
|
1033
|
-
static: false,
|
|
1034
|
-
private: false,
|
|
1035
|
-
access: {
|
|
1036
|
-
has: (obj) => "_wide" in obj,
|
|
1037
|
-
set: (obj, value) => {
|
|
1038
|
-
obj._wide = value;
|
|
1039
|
-
}
|
|
1040
|
-
},
|
|
1041
|
-
metadata: _metadata
|
|
1042
|
-
}, null, _instanceExtraInitializers);
|
|
1043
|
-
__esDecorate(this, null, __calendarView_decorators, {
|
|
1044
|
-
kind: "accessor",
|
|
1045
|
-
name: "_calendarView",
|
|
1046
|
-
static: false,
|
|
1047
|
-
private: false,
|
|
1048
|
-
access: {
|
|
1049
|
-
has: (obj) => "_calendarView" in obj,
|
|
1050
|
-
get: (obj) => obj._calendarView,
|
|
1051
|
-
set: (obj, value) => {
|
|
1052
|
-
obj._calendarView = value;
|
|
1053
|
-
}
|
|
1054
|
-
},
|
|
1055
|
-
metadata: _metadata
|
|
1056
|
-
}, __calendarView_initializers, __calendarView_extraInitializers);
|
|
1057
|
-
__esDecorate(this, null, __initialized_decorators, {
|
|
1058
|
-
kind: "accessor",
|
|
1059
|
-
name: "_initialized",
|
|
1060
|
-
static: false,
|
|
1061
|
-
private: false,
|
|
1062
|
-
access: {
|
|
1063
|
-
has: (obj) => "_initialized" in obj,
|
|
1064
|
-
get: (obj) => obj._initialized,
|
|
1065
|
-
set: (obj, value) => {
|
|
1066
|
-
obj._initialized = value;
|
|
1067
|
-
}
|
|
1068
|
-
},
|
|
1069
|
-
metadata: _metadata
|
|
1070
|
-
}, __initialized_initializers, __initialized_extraInitializers);
|
|
1071
|
-
if (_metadata) Object.defineProperty(this, Symbol.metadata, {
|
|
1072
|
-
enumerable: true,
|
|
1073
|
-
configurable: true,
|
|
1074
|
-
writable: true,
|
|
1075
|
-
value: _metadata
|
|
1076
|
-
});
|
|
1077
|
-
}
|
|
1078
|
-
static {
|
|
1079
|
-
this.elementName = "sbb-calendar";
|
|
1080
|
-
}
|
|
1081
|
-
static {
|
|
1082
|
-
this.styles = [boxSizingStyles, calendar_default];
|
|
1083
|
-
}
|
|
1084
|
-
static {
|
|
1085
|
-
this.events = {
|
|
1086
|
-
dateselected: "dateselected",
|
|
1087
|
-
monthchange: "monthchange"
|
|
1088
|
-
};
|
|
1089
|
-
}
|
|
1090
|
-
#wide_accessor_storage;
|
|
1091
|
-
/** If set to true, two months are displayed */
|
|
1092
|
-
get wide() {
|
|
1093
|
-
return this.#wide_accessor_storage;
|
|
1094
|
-
}
|
|
1095
|
-
set wide(value) {
|
|
1096
|
-
this.#wide_accessor_storage = value;
|
|
1097
|
-
}
|
|
1098
|
-
#view_accessor_storage;
|
|
1099
|
-
/** The initial view of the calendar which should be displayed on opening. */
|
|
1100
|
-
get view() {
|
|
1101
|
-
return this.#view_accessor_storage;
|
|
1102
|
-
}
|
|
1103
|
-
set view(value) {
|
|
1104
|
-
this.#view_accessor_storage = value;
|
|
1105
|
-
}
|
|
1106
|
-
#min_accessor_storage;
|
|
1107
|
-
/**
|
|
1108
|
-
* The minimum valid date. Accepts a date object or null.
|
|
1109
|
-
* Accepts an ISO8601 formatted string (e.g. 2024-12-24) as attribute.
|
|
1110
|
-
*/
|
|
1111
|
-
get min() {
|
|
1112
|
-
return this.#min_accessor_storage;
|
|
1113
|
-
}
|
|
1114
|
-
set min(value) {
|
|
1115
|
-
this.#min_accessor_storage = value;
|
|
1116
|
-
}
|
|
1117
|
-
#max_accessor_storage;
|
|
1118
|
-
/**
|
|
1119
|
-
* The maximum valid date. Accepts a date object or null.
|
|
1120
|
-
* Accepts an ISO8601 formatted string (e.g. 2024-12-24) as attribute.
|
|
1121
|
-
*/
|
|
1122
|
-
get max() {
|
|
1123
|
-
return this.#max_accessor_storage;
|
|
1124
|
-
}
|
|
1125
|
-
set max(value) {
|
|
1126
|
-
this.#max_accessor_storage = value;
|
|
1127
|
-
}
|
|
1128
|
-
#multiple_accessor_storage;
|
|
1129
|
-
/** Whether the calendar allows for multiple date selection. */
|
|
1130
|
-
get multiple() {
|
|
1131
|
-
return this.#multiple_accessor_storage;
|
|
1132
|
-
}
|
|
1133
|
-
set multiple(value) {
|
|
1134
|
-
this.#multiple_accessor_storage = value;
|
|
1135
|
-
}
|
|
1136
|
-
/**
|
|
1137
|
-
* The selected date: accepts a date object, or, if `multiple`, an array of dates.
|
|
1138
|
-
*/
|
|
1139
|
-
set selected(value) {
|
|
1140
|
-
if (Array.isArray(value)) this._selected = value.map((dateLike) => this._dateAdapter.getValidDateOrNull(this._dateAdapter.deserialize(dateLike))).filter((date) => date !== null).filter((date) => !this._isDayInRange(this._dateAdapter.toIso8601(date)) || this._dateFilter(date));
|
|
1141
|
-
else {
|
|
1142
|
-
const selectedDate = this._dateAdapter.getValidDateOrNull(this._dateAdapter.deserialize(value));
|
|
1143
|
-
if (!!selectedDate && (!this._isDayInRange(this._dateAdapter.toIso8601(selectedDate)) || this._dateFilter(selectedDate))) this._selected = selectedDate;
|
|
1144
|
-
else this._selected = null;
|
|
1145
|
-
}
|
|
1146
|
-
}
|
|
1147
|
-
get selected() {
|
|
1148
|
-
return this._selected;
|
|
1149
|
-
}
|
|
1150
|
-
#_selected_accessor_storage;
|
|
1151
|
-
get _selected() {
|
|
1152
|
-
return this.#_selected_accessor_storage;
|
|
1153
|
-
}
|
|
1154
|
-
set _selected(value) {
|
|
1155
|
-
this.#_selected_accessor_storage = value;
|
|
1156
|
-
}
|
|
1157
|
-
#dateFilter_accessor_storage;
|
|
1158
|
-
/** A function used to filter out dates. */
|
|
1159
|
-
get dateFilter() {
|
|
1160
|
-
return this.#dateFilter_accessor_storage;
|
|
1161
|
-
}
|
|
1162
|
-
set dateFilter(value) {
|
|
1163
|
-
this.#dateFilter_accessor_storage = value;
|
|
1164
|
-
}
|
|
1165
|
-
#orientation_accessor_storage;
|
|
1166
|
-
/** The orientation of days in the calendar. */
|
|
1167
|
-
get orientation() {
|
|
1168
|
-
return this.#orientation_accessor_storage;
|
|
1169
|
-
}
|
|
1170
|
-
set orientation(value) {
|
|
1171
|
-
this.#orientation_accessor_storage = value;
|
|
1172
|
-
}
|
|
1173
|
-
#weekNumbers_accessor_storage;
|
|
1174
|
-
/** Whether it has to display the week numbers in addition to week days. */
|
|
1175
|
-
get weekNumbers() {
|
|
1176
|
-
return this.#weekNumbers_accessor_storage;
|
|
1177
|
-
}
|
|
1178
|
-
set weekNumbers(value) {
|
|
1179
|
-
this.#weekNumbers_accessor_storage = value;
|
|
1180
|
-
}
|
|
1181
|
-
#_activeDate_accessor_storage;
|
|
1182
|
-
/** The currently active date. */
|
|
1183
|
-
get _activeDate() {
|
|
1184
|
-
return this.#_activeDate_accessor_storage;
|
|
1185
|
-
}
|
|
1186
|
-
set _activeDate(value) {
|
|
1187
|
-
this.#_activeDate_accessor_storage = value;
|
|
1188
|
-
}
|
|
1189
|
-
/** The current wide property considering property value and breakpoints. From zero to small `wide` has always to be false. */
|
|
1190
|
-
set _wide(wide) {
|
|
1191
|
-
this.toggleState("wide", wide);
|
|
1192
|
-
this._wideInternal = wide;
|
|
1193
|
-
}
|
|
1194
|
-
get _wide() {
|
|
1195
|
-
return this._wideInternal;
|
|
1196
|
-
}
|
|
1197
|
-
#_calendarView_accessor_storage;
|
|
1198
|
-
get _calendarView() {
|
|
1199
|
-
return this.#_calendarView_accessor_storage;
|
|
1200
|
-
}
|
|
1201
|
-
set _calendarView(value) {
|
|
1202
|
-
this.#_calendarView_accessor_storage = value;
|
|
1203
|
-
}
|
|
1204
|
-
/** A list of calendar's cells corresponding to days, months or years depending on the view. */
|
|
1205
|
-
get _cells() {
|
|
1206
|
-
return Array.from((this._calendarView === "day" ? Array.from(this.shadowRoot.querySelectorAll("slot")).flatMap((e) => e.assignedElements({ flatten: true })) : this.shadowRoot?.querySelectorAll(`sbb-calendar-${this._calendarView}`)) ?? []);
|
|
1207
|
-
}
|
|
1208
|
-
#_initialized_accessor_storage;
|
|
1209
|
-
get _initialized() {
|
|
1210
|
-
return this.#_initialized_accessor_storage;
|
|
1211
|
-
}
|
|
1212
|
-
set _initialized(value) {
|
|
1213
|
-
this.#_initialized_accessor_storage = value;
|
|
1214
|
-
}
|
|
1215
|
-
constructor() {
|
|
1216
|
-
super();
|
|
1217
|
-
this.#wide_accessor_storage = (__runInitializers(this, _instanceExtraInitializers), __runInitializers(this, _wide_initializers, false));
|
|
1218
|
-
this.#view_accessor_storage = (__runInitializers(this, _wide_extraInitializers), __runInitializers(this, _view_initializers, "day"));
|
|
1219
|
-
this.#min_accessor_storage = (__runInitializers(this, _view_extraInitializers), __runInitializers(this, _min_initializers, null));
|
|
1220
|
-
this.#max_accessor_storage = (__runInitializers(this, _min_extraInitializers), __runInitializers(this, _max_initializers, null));
|
|
1221
|
-
this.#multiple_accessor_storage = (__runInitializers(this, _max_extraInitializers), __runInitializers(this, _multiple_initializers, false));
|
|
1222
|
-
this.#_selected_accessor_storage = (__runInitializers(this, _multiple_extraInitializers), __runInitializers(this, __selected_initializers, null));
|
|
1223
|
-
this.#dateFilter_accessor_storage = (__runInitializers(this, __selected_extraInitializers), __runInitializers(this, _dateFilter_initializers, null));
|
|
1224
|
-
this.#orientation_accessor_storage = (__runInitializers(this, _dateFilter_extraInitializers), __runInitializers(this, _orientation_initializers, "horizontal"));
|
|
1225
|
-
this.#weekNumbers_accessor_storage = (__runInitializers(this, _orientation_extraInitializers), __runInitializers(this, _weekNumbers_initializers, false));
|
|
1226
|
-
this._dateAdapter = (__runInitializers(this, _weekNumbers_extraInitializers), readConfig().datetime?.dateAdapter ?? defaultDateAdapter);
|
|
1227
|
-
this.#_activeDate_accessor_storage = __runInitializers(this, __activeDate_initializers, this._dateAdapter.today());
|
|
1228
|
-
this._wideInternal = (__runInitializers(this, __activeDate_extraInitializers), false);
|
|
1229
|
-
this.#_calendarView_accessor_storage = __runInitializers(this, __calendarView_initializers, "day");
|
|
1230
|
-
this._nextCalendarView = (__runInitializers(this, __calendarView_extraInitializers), "day");
|
|
1231
|
-
/** Information about the rendered day view; used in keyboard navigation. */
|
|
1232
|
-
this._keyboardNavigationDayViewParameters = {
|
|
1233
|
-
firstDayInView: null,
|
|
1234
|
-
lastDayInView: null,
|
|
1235
|
-
firstMonthOffset: 0,
|
|
1236
|
-
firstMonthLength: 0,
|
|
1237
|
-
secondMonthOffset: 0
|
|
1238
|
-
};
|
|
1239
|
-
/** Grid of calendar cells representing the dates of the month. */
|
|
1240
|
-
this._weeks = [];
|
|
1241
|
-
/** An array containing all the month names in the current language. */
|
|
1242
|
-
this._monthNames = this._dateAdapter.getMonthNames("long");
|
|
1243
|
-
this._enhancedVariant = false;
|
|
1244
|
-
/** Whether the focus should be reset on focusCell. */
|
|
1245
|
-
this._resetFocus = false;
|
|
1246
|
-
/** Whether an element inside the calendar is currently focused. */
|
|
1247
|
-
this._containingFocus = false;
|
|
1248
|
-
this.#_initialized_accessor_storage = __runInitializers(this, __initialized_initializers, false);
|
|
1249
|
-
this._language = (__runInitializers(this, __initialized_extraInitializers), new SbbLanguageController(this).withHandler(() => {
|
|
1250
|
-
this._monthNames = this._dateAdapter.getMonthNames("long");
|
|
1251
|
-
this._createMonthRows();
|
|
1252
|
-
}));
|
|
1253
|
-
this._mediaMatcher = new SbbMediaMatcherController(this, { [SbbMediaQueryBreakpointLargeAndAbove]: () => this._init() });
|
|
1254
|
-
this._onSlotChange = () => {
|
|
1255
|
-
this._enhancedVariant = Array.from(this.children).some((c) => c.localName === "sbb-calendar-day");
|
|
1256
|
-
this.toggleState("enhanced", this._enhancedVariant);
|
|
1257
|
-
this._setTabIndex();
|
|
1258
|
-
};
|
|
1259
|
-
this._createMonthRows();
|
|
1260
|
-
this._setWeekdays();
|
|
1261
|
-
this.addEventListener("focusin", () => this._containingFocus = true);
|
|
1262
|
-
this.addEventListener("focusout", () => this._containingFocus = false);
|
|
1263
|
-
this.addEventListener("click", (e) => {
|
|
1264
|
-
const day = e.target.closest("sbb-calendar-day");
|
|
1265
|
-
if (day) this._selectDate(day.value);
|
|
1266
|
-
});
|
|
1267
|
-
this.addEventListener("keydown", (e) => {
|
|
1268
|
-
if (e.target.localName === "sbb-calendar-day") this._handleKeyboardEvent(e, this._mapDateToDay(e.target.value));
|
|
1269
|
-
});
|
|
1270
|
-
}
|
|
1271
|
-
_dateFilter(date) {
|
|
1272
|
-
return this.dateFilter?.(date) ?? true;
|
|
1273
|
-
}
|
|
1274
|
-
/** Resets the active month according to the new state of the calendar. */
|
|
1275
|
-
resetPosition() {
|
|
1276
|
-
this._resetCalendarView();
|
|
1277
|
-
this._init();
|
|
1278
|
-
}
|
|
1279
|
-
connectedCallback() {
|
|
1280
|
-
super.connectedCallback();
|
|
1281
|
-
this.resetPosition();
|
|
1282
|
-
this.shadowRoot?.addEventListener("slotchange", this._onSlotChange, { capture: true });
|
|
1283
|
-
}
|
|
1284
|
-
disconnectedCallback() {
|
|
1285
|
-
super.disconnectedCallback();
|
|
1286
|
-
this.shadowRoot?.removeEventListener("slotchange", this._onSlotChange, { capture: true });
|
|
1287
|
-
}
|
|
1288
|
-
/** @internal */
|
|
1289
|
-
focus() {
|
|
1290
|
-
this._resetFocus = true;
|
|
1291
|
-
this._focusCell();
|
|
1292
|
-
}
|
|
1293
|
-
willUpdate(changedProperties) {
|
|
1294
|
-
super.willUpdate(changedProperties);
|
|
1295
|
-
if (!this._initialized) return;
|
|
1296
|
-
if (changedProperties.has("wide") || changedProperties.has("orientation")) this.resetPosition();
|
|
1297
|
-
if (changedProperties.has("view")) {
|
|
1298
|
-
this._setChosenYear();
|
|
1299
|
-
this._chosenMonth = void 0;
|
|
1300
|
-
this._nextCalendarView = this._calendarView = this.view;
|
|
1301
|
-
}
|
|
1302
|
-
}
|
|
1303
|
-
updated(changedProperties) {
|
|
1304
|
-
super.updated(changedProperties);
|
|
1305
|
-
this._setTabIndex();
|
|
1306
|
-
this._focusCell();
|
|
1307
|
-
}
|
|
1308
|
-
/**
|
|
1309
|
-
* The `_selected` state should be adapted when the `multiple` property changes:
|
|
1310
|
-
* - if it changes to true, the '_selected' is set to an array;
|
|
1311
|
-
* - if it changes to false, the first available option is set as 'value' otherwise it's set to null.
|
|
1312
|
-
*/
|
|
1313
|
-
_onMultipleChanged(isMultiple) {
|
|
1314
|
-
if (isMultiple && !Array.isArray(this._selected)) this._selected = this._selected ? [this._selected] : [];
|
|
1315
|
-
if (!isMultiple && Array.isArray(this._selected)) this._selected = this._selected.length ? this._selected[0] : null;
|
|
1316
|
-
}
|
|
1317
|
-
/** Initializes the component. */
|
|
1318
|
-
_init(activeDate) {
|
|
1319
|
-
if (isServer) return;
|
|
1320
|
-
else if (this.hydrationRequired) {
|
|
1321
|
-
this.hydrationComplete.then(() => this._init());
|
|
1322
|
-
return;
|
|
1323
|
-
}
|
|
1324
|
-
if (activeDate) this._assignActiveDate(activeDate);
|
|
1325
|
-
this._wide = (this._mediaMatcher.matches(SbbMediaQueryBreakpointLargeAndAbove) ?? false) && this.wide;
|
|
1326
|
-
this._weeks = this._createWeekRows(this._activeDate);
|
|
1327
|
-
this._years = this._createYearRows();
|
|
1328
|
-
this._weekNumbers = this._createWeekNumbers(this._activeDate);
|
|
1329
|
-
this._nextMonthWeeks = [[]];
|
|
1330
|
-
this._nextMonthYears = [[]];
|
|
1331
|
-
if (this._wide) {
|
|
1332
|
-
const nextMonthDate = this._dateAdapter.addCalendarMonths(this._activeDate, 1);
|
|
1333
|
-
this._nextMonthWeeks = this._createWeekRows(nextMonthDate, true);
|
|
1334
|
-
this._nextMonthYears = this._createYearRows(YEARS_PER_PAGE);
|
|
1335
|
-
this._nextMonthWeekNumbers = this._createWeekNumbers(nextMonthDate);
|
|
1336
|
-
}
|
|
1337
|
-
this._initialized = true;
|
|
1338
|
-
}
|
|
1339
|
-
/** Focuses on a day cell prioritizing the selected day, the current day, and lastly, the first selectable day. */
|
|
1340
|
-
_focusCell() {
|
|
1341
|
-
if (this._resetFocus) {
|
|
1342
|
-
this._getFirstFocusable()?.focus();
|
|
1343
|
-
this._resetFocus = false;
|
|
1344
|
-
}
|
|
1345
|
-
}
|
|
1346
|
-
/** Creates the array of weekdays. */
|
|
1347
|
-
_setWeekdays() {
|
|
1348
|
-
const narrowWeekdays = this._dateAdapter.getDayOfWeekNames("narrow");
|
|
1349
|
-
const weekdays = this._dateAdapter.getDayOfWeekNames("long").map((long, i) => ({
|
|
1350
|
-
long,
|
|
1351
|
-
narrow: narrowWeekdays[i]
|
|
1352
|
-
}));
|
|
1353
|
-
const firstDayOfWeek = this._dateAdapter.getFirstDayOfWeek();
|
|
1354
|
-
this._weekdays = weekdays.slice(firstDayOfWeek).concat(weekdays.slice(0, firstDayOfWeek));
|
|
1355
|
-
}
|
|
1356
|
-
/**
|
|
1357
|
-
* Given a date, it returns the week numbers for the month the date belongs to.
|
|
1358
|
-
* TODO: check if date-fns can be replaced with custom logic.
|
|
1359
|
-
*
|
|
1360
|
-
* Since the calculation is not simple (see https://en.wikipedia.org/wiki/Week#Numbering),
|
|
1361
|
-
* the date-fns library has been used this way:
|
|
1362
|
-
* the first and the last day of the month are calculated and then passed to the `eachWeekOfInterval` function,
|
|
1363
|
-
* which returns an array containing the starting day of every ISO week of the month,
|
|
1364
|
-
* considering Monday as the first day.
|
|
1365
|
-
* Then, this array is mapped via the `getWeek` function, which returns the ISO week number for that date.
|
|
1366
|
-
*/
|
|
1367
|
-
_createWeekNumbers(date) {
|
|
1368
|
-
return eachWeekOfInterval({
|
|
1369
|
-
start: startOfMonth(date),
|
|
1370
|
-
end: endOfMonth(date)
|
|
1371
|
-
}, { weekStartsOn: 1 }).map((firstDayOfWeek) => getWeek(firstDayOfWeek, {
|
|
1372
|
-
weekStartsOn: 1,
|
|
1373
|
-
firstWeekContainsDate: 4
|
|
1374
|
-
}));
|
|
1375
|
-
}
|
|
1376
|
-
/** Creates the rows along the horizontal direction and sets the parameters used in keyboard navigation. */
|
|
1377
|
-
_createWeekRows(value, isSecondMonthInView = false) {
|
|
1378
|
-
const daysInMonth = this._dateAdapter.getNumDaysInMonth(value);
|
|
1379
|
-
const weekOffset = this._dateAdapter.getFirstWeekOffset(value);
|
|
1380
|
-
if (!isSecondMonthInView) {
|
|
1381
|
-
this._keyboardNavigationDayViewParameters.firstMonthLength = daysInMonth;
|
|
1382
|
-
this._keyboardNavigationDayViewParameters.firstMonthOffset = weekOffset;
|
|
1383
|
-
this._keyboardNavigationDayViewParameters.firstDayInView = this._dateAdapter.toIso8601(this._dateAdapter.createDate(this._dateAdapter.getYear(value), this._dateAdapter.getMonth(value), 1));
|
|
1384
|
-
this._keyboardNavigationDayViewParameters.lastDayInView = this._dateAdapter.toIso8601(this._dateAdapter.createDate(this._dateAdapter.getYear(value), this._dateAdapter.getMonth(value), daysInMonth));
|
|
1385
|
-
} else {
|
|
1386
|
-
this._keyboardNavigationDayViewParameters.secondMonthOffset = weekOffset;
|
|
1387
|
-
this._keyboardNavigationDayViewParameters.lastDayInView = this._dateAdapter.toIso8601(this._dateAdapter.createDate(this._dateAdapter.getYear(value), this._dateAdapter.getMonth(value), daysInMonth));
|
|
1388
|
-
}
|
|
1389
|
-
return this.orientation === "horizontal" ? this._createWeekRowsHorizontal(value, daysInMonth, weekOffset) : this._createWeekRowsVertical(value, daysInMonth, weekOffset);
|
|
1390
|
-
}
|
|
1391
|
-
/**
|
|
1392
|
-
* Creates the rows for each week in orientation='horizontal'.
|
|
1393
|
-
*
|
|
1394
|
-
* Iterates through the days of the months, creates a Day object for each and pushes it into and array.
|
|
1395
|
-
* Each seven days (considering the offset at the beginning of the month) restarts from an empty array.
|
|
1396
|
-
*
|
|
1397
|
-
* The result is a matrix in which every row is a week (or part of it, considering offset).
|
|
1398
|
-
*/
|
|
1399
|
-
_createWeekRowsHorizontal(value, daysInMonth, weekOffset) {
|
|
1400
|
-
const weeks = [[]];
|
|
1401
|
-
for (let i = 0, cell = weekOffset; i < daysInMonth; i++, cell++) {
|
|
1402
|
-
if (cell === DAYS_PER_ROW) {
|
|
1403
|
-
weeks.push([]);
|
|
1404
|
-
cell = 0;
|
|
1405
|
-
}
|
|
1406
|
-
const date = this._dateAdapter.createDate(this._dateAdapter.getYear(value), this._dateAdapter.getMonth(value), i + 1);
|
|
1407
|
-
weeks[weeks.length - 1].push(this._mapDateToDay(date));
|
|
1408
|
-
}
|
|
1409
|
-
return weeks;
|
|
1410
|
-
}
|
|
1411
|
-
/**
|
|
1412
|
-
* Creates the rows for each week in orientation='vertical'.
|
|
1413
|
-
*
|
|
1414
|
-
* Creates a matrix with seven empty rows.
|
|
1415
|
-
* Iterates through the days of the months, creates a Day object for each
|
|
1416
|
-
* and pushes it into the correct array considering the offset at the beginning of the month.
|
|
1417
|
-
* Each seven days (including offset) restarts from the first.
|
|
1418
|
-
*
|
|
1419
|
-
* The result is a matrix in which every row is a set of weekdays, so:
|
|
1420
|
-
* - row 0: all the Mondays;
|
|
1421
|
-
* - row 1: all the Tuesdays;
|
|
1422
|
-
* - ...
|
|
1423
|
-
* - row 7: all the Sundays.
|
|
1424
|
-
*/
|
|
1425
|
-
_createWeekRowsVertical(value, daysInMonth, weekOffset) {
|
|
1426
|
-
const weeks = Array.from({ length: DAYS_PER_ROW }, () => []);
|
|
1427
|
-
for (let i = 0, cell = weekOffset; i < daysInMonth; i++, cell++) {
|
|
1428
|
-
if (cell === DAYS_PER_ROW) cell = 0;
|
|
1429
|
-
const date = this._dateAdapter.createDate(this._dateAdapter.getYear(value), this._dateAdapter.getMonth(value), i + 1);
|
|
1430
|
-
weeks[cell].push(this._mapDateToDay(date));
|
|
1431
|
-
}
|
|
1432
|
-
return weeks;
|
|
1433
|
-
}
|
|
1434
|
-
_mapDateToDay(date) {
|
|
1435
|
-
const isoDate = this._dateAdapter.toIso8601(date);
|
|
1436
|
-
return {
|
|
1437
|
-
value: isoDate,
|
|
1438
|
-
dateValue: date,
|
|
1439
|
-
dayValue: String(this._dateAdapter.getDate(date)),
|
|
1440
|
-
monthValue: String(this._dateAdapter.getMonth(date)),
|
|
1441
|
-
yearValue: String(this._dateAdapter.getYear(date)),
|
|
1442
|
-
weekValue: getWeek(isoDate, {
|
|
1443
|
-
weekStartsOn: 1,
|
|
1444
|
-
firstWeekContainsDate: 4
|
|
1445
|
-
}),
|
|
1446
|
-
weekDayValue: this._dateAdapter.getDayOfWeek(date)
|
|
1447
|
-
};
|
|
1448
|
-
}
|
|
1449
|
-
/** Force the conversion to ISO8601 formatted string. */
|
|
1450
|
-
_mapValueToISODate(value) {
|
|
1451
|
-
return typeof value === "string" ? value : this._dateAdapter.toIso8601(value);
|
|
1452
|
-
}
|
|
1453
|
-
/** Creates the rows for the month selection view. */
|
|
1454
|
-
_createMonthRows() {
|
|
1455
|
-
const months = new Array(12).fill(null).map((_, i) => ({
|
|
1456
|
-
value: String(i + 1).padStart(2, "0"),
|
|
1457
|
-
monthValue: i + 1
|
|
1458
|
-
}));
|
|
1459
|
-
const rows = 12 / MONTHS_PER_ROW;
|
|
1460
|
-
const monthArray = [];
|
|
1461
|
-
for (let i = 0; i < rows; i++) monthArray.push(months.slice(MONTHS_PER_ROW * i, MONTHS_PER_ROW * (i + 1)));
|
|
1462
|
-
this._months = monthArray;
|
|
1463
|
-
}
|
|
1464
|
-
/** Creates the rows for the year selection view. */
|
|
1465
|
-
_createYearRows(offset = 0) {
|
|
1466
|
-
const startValueYearView = this._getStartValueYearView();
|
|
1467
|
-
const allYears = new Array(YEARS_PER_PAGE).fill(0).map((_, i) => startValueYearView + offset + i);
|
|
1468
|
-
const rows = YEARS_PER_PAGE / YEARS_PER_ROW;
|
|
1469
|
-
const yearArray = [];
|
|
1470
|
-
for (let i = 0; i < rows; i++) yearArray.push(allYears.slice(YEARS_PER_ROW * i, YEARS_PER_ROW * (i + 1)));
|
|
1471
|
-
return yearArray;
|
|
1472
|
-
}
|
|
1473
|
-
/**
|
|
1474
|
-
* Calculates the first year that will be shown in the year selection panel.
|
|
1475
|
-
* If `minDate` and `maxDate` are both null, the starting year is calculated as
|
|
1476
|
-
* the multiple of YEARS_PER_PAGE closest to and less than activeDate,
|
|
1477
|
-
* e.g., with `YEARS_PER_PAGE` = 24 and `activeDate` = 2020, the function will return 2016 (24 * 83),
|
|
1478
|
-
* while with `activeDate` = 2000, the function will return 1992 (24 * 82).
|
|
1479
|
-
* If `minDate` is not null, it returns the corresponding year; if `maxDate` is not null,
|
|
1480
|
-
* it returns the corresponding year minus `YEARS_PER_PAGE`, so that the `maxDate` is the last rendered year.
|
|
1481
|
-
* If both are not null, `maxDate` has priority over `minDate`.
|
|
1482
|
-
*/
|
|
1483
|
-
_getStartValueYearView() {
|
|
1484
|
-
let startingYear = 0;
|
|
1485
|
-
if (this.max) startingYear = this._dateAdapter.getYear(this.max) - YEARS_PER_PAGE + 1;
|
|
1486
|
-
else if (this.min) startingYear = this._dateAdapter.getYear(this.min);
|
|
1487
|
-
const activeYear = this._dateAdapter.getYear(this._activeDate);
|
|
1488
|
-
return activeYear - ((activeYear - startingYear) % YEARS_PER_PAGE + YEARS_PER_PAGE) % YEARS_PER_PAGE;
|
|
1489
|
-
}
|
|
1490
|
-
/** Checks if date is within the min-max range. */
|
|
1491
|
-
_isDayInRange(dateString) {
|
|
1492
|
-
if (!this.min && !this.max) return true;
|
|
1493
|
-
const date = this._dateAdapter.deserialize(dateString);
|
|
1494
|
-
return this._dateAdapter.sameDate(date, this._dateAdapter.clampDate(date, this.min, this.max));
|
|
1495
|
-
}
|
|
1496
|
-
/** Emits the selected date and sets it internally. */
|
|
1497
|
-
_selectDate(day) {
|
|
1498
|
-
this._chosenMonth = void 0;
|
|
1499
|
-
this._setChosenYear();
|
|
1500
|
-
if (this.multiple) {
|
|
1501
|
-
if (this._selected && this._selected.length > 0) {
|
|
1502
|
-
const indexOfSelectedDay = this._selected.findIndex((sel) => this._dateAdapter.compareDate(sel, day) === 0);
|
|
1503
|
-
if (indexOfSelectedDay !== -1) this._selected = this._selected.filter((_, i) => i !== indexOfSelectedDay);
|
|
1504
|
-
else this._selected = [...this._selected, day];
|
|
1505
|
-
} else this._selected = [day];
|
|
1506
|
-
this._emitDateSelectedEvent(this._selected.map((e) => this._dateAdapter.deserialize(e)));
|
|
1507
|
-
} else if (!this._selected || this._dateAdapter.compareDate(this._selected, day) !== 0) {
|
|
1508
|
-
this._selected = day;
|
|
1509
|
-
this._emitDateSelectedEvent(this._dateAdapter.deserialize(day));
|
|
1510
|
-
}
|
|
1511
|
-
}
|
|
1512
|
-
/**
|
|
1513
|
-
* Handle multiple dates selection via weekNumber / weekDay buttons:
|
|
1514
|
-
* - if Cmd or Ctrl are pressed, add the new date to the current ones;
|
|
1515
|
-
* - if not,
|
|
1516
|
-
* - if the new dates are the same of the current ones, it means that the same button has been clicked twice, so do nothing;
|
|
1517
|
-
* - if not, the selected dates are the new ones.
|
|
1518
|
-
*/
|
|
1519
|
-
_selectMultipleDates(days) {
|
|
1520
|
-
const enabledDays = this._cells.filter((e) => !e.disabled).map((e) => this._mapValueToISODate(e.value));
|
|
1521
|
-
const daysToAdd = days.map((e) => e.value).filter((isoDate) => enabledDays.includes(isoDate));
|
|
1522
|
-
const daysToAddSet = new Set(daysToAdd);
|
|
1523
|
-
const selectedSet = new Set(this._selected.map((s) => this._dateAdapter.toIso8601(s)));
|
|
1524
|
-
this._selected = this._updateSelectedWithMultipleDates(daysToAdd, daysToAddSet, selectedSet).map((s) => this._dateAdapter.deserialize(s));
|
|
1525
|
-
this._emitDateSelectedEvent(this._selected.map((e) => this._dateAdapter.deserialize(e)));
|
|
1526
|
-
}
|
|
1527
|
-
/**
|
|
1528
|
-
* Emits the dateselected event given the detail (as T or T[] based on the value of the multiple flag).
|
|
1529
|
-
*/
|
|
1530
|
-
_emitDateSelectedEvent(detail) {
|
|
1531
|
-
/** @type {CustomEvent<T | T[]>} Event emitted on date selection. */
|
|
1532
|
-
this.dispatchEvent(new CustomEvent("dateselected", {
|
|
1533
|
-
detail,
|
|
1534
|
-
composed: true,
|
|
1535
|
-
bubbles: true
|
|
1536
|
-
}));
|
|
1537
|
-
}
|
|
1538
|
-
_emitMonthChange() {
|
|
1539
|
-
const monthchange = (this.wide ? [...this._weeks, ...this._nextMonthWeeks] : this._weeks).flat().sort((a, b) => a.value.localeCompare(b.value));
|
|
1540
|
-
/**
|
|
1541
|
-
* @type {SbbMonthChangeEvent}
|
|
1542
|
-
* Emits when the month changes.
|
|
1543
|
-
* The `range` property contains the days array of the chosen month.
|
|
1544
|
-
*/
|
|
1545
|
-
this.dispatchEvent(new SbbMonthChangeEvent(monthchange));
|
|
1546
|
-
}
|
|
1547
|
-
/**
|
|
1548
|
-
* In case of multiple selection, newly added days must be added to the existing ones, without duplication.
|
|
1549
|
-
* If the days to add are exactly the same as the selected ones, the set must be emptied.
|
|
1550
|
-
*/
|
|
1551
|
-
_updateSelectedWithMultipleDates(daysToAdd, daysToAddSet, selectedSet) {
|
|
1552
|
-
if (daysToAdd.every((day) => selectedSet.has(day))) daysToAddSet.forEach((day) => selectedSet.delete(day));
|
|
1553
|
-
else daysToAddSet.forEach((day) => selectedSet.add(day));
|
|
1554
|
-
return Array.from(selectedSet);
|
|
1555
|
-
}
|
|
1556
|
-
_setChosenYear() {
|
|
1557
|
-
if (this.view === "month") {
|
|
1558
|
-
let selectedDate;
|
|
1559
|
-
if (this.multiple) selectedDate = this.selected.at(-1);
|
|
1560
|
-
else selectedDate = this.selected;
|
|
1561
|
-
this._chosenYear = this._dateAdapter.getYear(selectedDate ?? this._dateAdapter.today());
|
|
1562
|
-
} else this._chosenYear = void 0;
|
|
1563
|
-
}
|
|
1564
|
-
_assignActiveDate(date) {
|
|
1565
|
-
if (this.min && this._dateAdapter.compareDate(this.min, date) > 0) {
|
|
1566
|
-
this._activeDate = this.min;
|
|
1567
|
-
return;
|
|
1568
|
-
}
|
|
1569
|
-
if (this.max && this._dateAdapter.compareDate(this.max, date) < 0) {
|
|
1570
|
-
this._activeDate = this.max;
|
|
1571
|
-
return;
|
|
1572
|
-
}
|
|
1573
|
-
this._activeDate = date;
|
|
1574
|
-
}
|
|
1575
|
-
/** Goes to the month identified by the shift. */
|
|
1576
|
-
_goToDifferentMonth(months) {
|
|
1577
|
-
this._init(this._dateAdapter.addCalendarMonths(this._activeDate, months));
|
|
1578
|
-
this._emitMonthChange();
|
|
1579
|
-
}
|
|
1580
|
-
_goToDifferentYear(years) {
|
|
1581
|
-
this._chosenYear += years;
|
|
1582
|
-
this._activeDate = this._dateAdapter.createDate(this._chosenYear, this._dateAdapter.getMonth(this._activeDate), this._dateAdapter.getDate(this._activeDate));
|
|
1583
|
-
this._init();
|
|
1584
|
-
}
|
|
1585
|
-
_goToDifferentYearRange(years) {
|
|
1586
|
-
this._init(this._dateAdapter.addCalendarYears(this._activeDate, years));
|
|
1587
|
-
}
|
|
1588
|
-
_prevDisabled(prevDate) {
|
|
1589
|
-
if (!this.min) return false;
|
|
1590
|
-
return this._dateAdapter.compareDate(prevDate, this.min) < 0;
|
|
1591
|
-
}
|
|
1592
|
-
_nextDisabled(nextDate) {
|
|
1593
|
-
if (!this.max) return false;
|
|
1594
|
-
return this._dateAdapter.compareDate(nextDate, this.max) > 0;
|
|
1595
|
-
}
|
|
1596
|
-
/** Checks if the "previous month" button should be disabled. */
|
|
1597
|
-
_previousMonthDisabled() {
|
|
1598
|
-
const prevMonth = this._dateAdapter.addCalendarDays(this._activeDate, this._dateAdapter.getDate(this._activeDate) * -1);
|
|
1599
|
-
return this._prevDisabled(prevMonth);
|
|
1600
|
-
}
|
|
1601
|
-
/** Checks if the "next month" button should be disabled. */
|
|
1602
|
-
_nextMonthDisabled() {
|
|
1603
|
-
let nextMonth = this._dateAdapter.addCalendarMonths(this._activeDate, this._wide ? 2 : 1);
|
|
1604
|
-
nextMonth = this._dateAdapter.createDate(this._dateAdapter.getYear(nextMonth), this._dateAdapter.getMonth(nextMonth), 1);
|
|
1605
|
-
return this._nextDisabled(nextMonth);
|
|
1606
|
-
}
|
|
1607
|
-
/** Checks if the "previous year" button should be disabled. */
|
|
1608
|
-
_previousYearDisabled() {
|
|
1609
|
-
const prevYear = this._dateAdapter.createDate(this._dateAdapter.getYear(this._activeDate) - 1, 12, 31);
|
|
1610
|
-
return this._prevDisabled(prevYear);
|
|
1611
|
-
}
|
|
1612
|
-
/** Checks if the "next year" button should be disabled. */
|
|
1613
|
-
_nextYearDisabled() {
|
|
1614
|
-
const nextYear = this._dateAdapter.createDate(this._dateAdapter.getYear(this._activeDate) + (this._wide ? 2 : 1), 1, 1);
|
|
1615
|
-
return this._nextDisabled(nextYear);
|
|
1616
|
-
}
|
|
1617
|
-
/** Checks if the "previous year" button should be disabled in year view. */
|
|
1618
|
-
_previousYearRangeDisabled() {
|
|
1619
|
-
const prevYear = this._dateAdapter.createDate(this._years[0][0] - 1, 12, 31);
|
|
1620
|
-
return this._prevDisabled(prevYear);
|
|
1621
|
-
}
|
|
1622
|
-
/** Checks if the "next year" button should be disabled in year view. */
|
|
1623
|
-
_nextYearRangeDisabled() {
|
|
1624
|
-
const years = this._wide ? this._nextMonthYears : this._years;
|
|
1625
|
-
const lastYearRange = years[years.length - 1];
|
|
1626
|
-
const lastYear = lastYearRange[lastYearRange.length - 1];
|
|
1627
|
-
const nextYear = this._dateAdapter.createDate(lastYear + 1, 1, 1);
|
|
1628
|
-
return this._nextDisabled(nextYear);
|
|
1629
|
-
}
|
|
1630
|
-
_handleTableBlur(eventTarget) {
|
|
1631
|
-
if (eventTarget?.localName !== "sbb-calendar-day") this._setTabIndex();
|
|
1632
|
-
}
|
|
1633
|
-
_setTabIndex() {
|
|
1634
|
-
Array.from(this._cells.filter((e) => e.tabIndex === 0) ?? []).forEach((day) => day.tabIndex = -1);
|
|
1635
|
-
const firstFocusable = this._getFirstFocusable();
|
|
1636
|
-
if (firstFocusable) firstFocusable.tabIndex = 0;
|
|
1637
|
-
}
|
|
1638
|
-
/** Get the element in the calendar to assign focus. */
|
|
1639
|
-
_getFirstFocusable() {
|
|
1640
|
-
if (this._calendarView === "day") {
|
|
1641
|
-
const selectedOrCurrent = this._cells.find((e) => e.matches(":is(:state(selected),[state--selected])")) ?? this._cells.find((e) => e.matches(":is(:state(current),[state--current])"));
|
|
1642
|
-
return selectedOrCurrent && !selectedOrCurrent.disabled ? selectedOrCurrent : this._getFirstFocusableDay();
|
|
1643
|
-
} else {
|
|
1644
|
-
const selectedOrCurrent = this.shadowRoot?.querySelector(":is(:state(selected),[state--selected])") ?? this.shadowRoot?.querySelector(":is(:state(current),[state--current])");
|
|
1645
|
-
return selectedOrCurrent && !selectedOrCurrent.disabled ? selectedOrCurrent : this.shadowRoot.querySelector(`sbb-calendar-${this._calendarView}:not([disabled])`);
|
|
1646
|
-
}
|
|
1647
|
-
}
|
|
1648
|
-
/**
|
|
1649
|
-
* In `day` view in `vertical` orientation,
|
|
1650
|
-
* if the first of the month is not a Monday, it is not the first rendered element in the table,
|
|
1651
|
-
* so `this.shadowRoot!.querySelector('sbb-calendar-day:not([disabled])')` will return a wrong value.
|
|
1652
|
-
*
|
|
1653
|
-
* To solve this, the element with the lowest `value` is taken (ISO String are ordered).
|
|
1654
|
-
*/
|
|
1655
|
-
_getFirstFocusableDay() {
|
|
1656
|
-
const cells = this._cells;
|
|
1657
|
-
const daysInView = cells.filter((e) => !e.disabled);
|
|
1658
|
-
if (!daysInView || daysInView.length === 0) return null;
|
|
1659
|
-
else {
|
|
1660
|
-
const firstElement = daysInView.map((e) => this._dateAdapter.toIso8601(e.value)).sort()[0];
|
|
1661
|
-
return cells.find((e) => e.matches(`[slot="${firstElement}"]`)) ?? null;
|
|
1662
|
-
}
|
|
1663
|
-
}
|
|
1664
|
-
_handleKeyboardEvent(event, day) {
|
|
1665
|
-
if (isArrowKeyOrPageKeysPressed(event)) event.preventDefault();
|
|
1666
|
-
const cells = this._cells;
|
|
1667
|
-
const index = cells.findIndex((e) => e === event.target);
|
|
1668
|
-
let nextEl;
|
|
1669
|
-
if (day) nextEl = this._navigateByKeyboardDayView(event, index, cells, day);
|
|
1670
|
-
else nextEl = this._navigateByKeyboard(event, index, cells);
|
|
1671
|
-
const activeEl = (this._enhancedVariant ? document : this.shadowRoot).activeElement;
|
|
1672
|
-
if (nextEl !== activeEl) {
|
|
1673
|
-
nextEl.tabIndex = 0;
|
|
1674
|
-
nextEl?.focus();
|
|
1675
|
-
activeEl.tabIndex = -1;
|
|
1676
|
-
}
|
|
1677
|
-
}
|
|
1678
|
-
_navigateByKeyboardDayView(evt, index, cells, day) {
|
|
1679
|
-
const arrowsOffset = this.orientation === "horizontal" ? {
|
|
1680
|
-
leftRight: 1,
|
|
1681
|
-
upDown: DAYS_PER_ROW
|
|
1682
|
-
} : {
|
|
1683
|
-
leftRight: DAYS_PER_ROW,
|
|
1684
|
-
upDown: 1
|
|
1685
|
-
};
|
|
1686
|
-
const offsetForVertical = index < this._keyboardNavigationDayViewParameters.firstMonthLength ? this._keyboardNavigationDayViewParameters.firstMonthOffset : this._keyboardNavigationDayViewParameters.secondMonthOffset;
|
|
1687
|
-
switch (evt.key) {
|
|
1688
|
-
case "ArrowUp": return this._findDayArrows(cells, index, day.dateValue, -arrowsOffset.upDown);
|
|
1689
|
-
case "ArrowDown": return this._findDayArrows(cells, index, day.dateValue, arrowsOffset.upDown);
|
|
1690
|
-
case "ArrowLeft": return this._findDayArrows(cells, index, day.dateValue, -arrowsOffset.leftRight);
|
|
1691
|
-
case "ArrowRight": return this._findDayArrows(cells, index, day.dateValue, arrowsOffset.leftRight);
|
|
1692
|
-
case "PageUp": if (this.orientation === "horizontal") {
|
|
1693
|
-
const delta = (+day.dayValue % DAYS_PER_ROW || DAYS_PER_ROW) - +day.dayValue;
|
|
1694
|
-
return this._findDayPageUpDown(cells, index, day, delta, arrowsOffset.upDown);
|
|
1695
|
-
} else {
|
|
1696
|
-
const delta = (Math.ceil((+day.dayValue + offsetForVertical) / DAYS_PER_ROW) - 1) * DAYS_PER_ROW - offsetForVertical + 1 - +day.dayValue;
|
|
1697
|
-
return this._findDayPageUpDown(cells, index, day, delta, arrowsOffset.upDown);
|
|
1698
|
-
}
|
|
1699
|
-
case "PageDown": if (this.orientation === "horizontal") {
|
|
1700
|
-
const monthInBounds = +day.monthValue + 1 > 12 ? 1 : +day.monthValue + 1;
|
|
1701
|
-
const yearInBounds = +day.monthValue + 1 > 12 ? +day.yearValue + 1 : +day.yearValue;
|
|
1702
|
-
const firstNextMonth = this._dateAdapter.createDate(yearInBounds, monthInBounds, 1);
|
|
1703
|
-
const lastOfMonth = this._dateAdapter.getDate(this._dateAdapter.addCalendarDays(firstNextMonth, -1));
|
|
1704
|
-
const delta = Math.trunc((lastOfMonth - +day.dayValue) / DAYS_PER_ROW) * DAYS_PER_ROW;
|
|
1705
|
-
return this._findDayPageUpDown(cells, index, day, delta, -arrowsOffset.upDown);
|
|
1706
|
-
} else {
|
|
1707
|
-
const delta = Math.ceil((+day.dayValue + offsetForVertical) / DAYS_PER_ROW) * DAYS_PER_ROW - offsetForVertical - +day.dayValue;
|
|
1708
|
-
return this._findDayPageUpDown(cells, index, day, delta, -arrowsOffset.upDown);
|
|
1709
|
-
}
|
|
1710
|
-
case "Home": return this._findDayFirst(cells, index, day, 1);
|
|
1711
|
-
case "End": {
|
|
1712
|
-
const monthInBounds = +day.monthValue + 1 > 12 ? 1 : +day.monthValue + 1;
|
|
1713
|
-
const yearInBounds = +day.monthValue + 1 > 12 ? +day.yearValue + 1 : +day.yearValue;
|
|
1714
|
-
const firstNextMonth = this._dateAdapter.createDate(yearInBounds, monthInBounds, 1);
|
|
1715
|
-
return this._findDayLast(cells, index, firstNextMonth);
|
|
1716
|
-
}
|
|
1717
|
-
default: return cells[index];
|
|
1718
|
-
}
|
|
1719
|
-
}
|
|
1720
|
-
_isDayOutOfView(date) {
|
|
1721
|
-
return date < this._keyboardNavigationDayViewParameters.firstDayInView || date > this._keyboardNavigationDayViewParameters.lastDayInView;
|
|
1722
|
-
}
|
|
1723
|
-
_findDayArrows(cells, index, date, delta) {
|
|
1724
|
-
const newDateValue = this._dateAdapter.toIso8601(this._dateAdapter.addCalendarDays(date, delta));
|
|
1725
|
-
if (this._isDayOutOfView(newDateValue)) return cells[index];
|
|
1726
|
-
const nextCell = cells.find((e) => this._mapValueToISODate(e.value) === newDateValue);
|
|
1727
|
-
if (!nextCell || nextCell.disabled) return this._findDayArrows(cells, index, this._dateAdapter.deserialize(newDateValue), delta);
|
|
1728
|
-
return nextCell;
|
|
1729
|
-
}
|
|
1730
|
-
_findDayPageUpDown(cells, index, day, delta, deltaIfDisabled) {
|
|
1731
|
-
const newDateValue = this._dateAdapter.toIso8601(this._dateAdapter.addCalendarDays(day.dateValue, delta));
|
|
1732
|
-
if (this._isDayOutOfView(newDateValue)) return cells[index];
|
|
1733
|
-
const nextCell = cells.find((e) => this._mapValueToISODate(e.value) === newDateValue);
|
|
1734
|
-
if (!nextCell || nextCell.disabled) return this._findDayPageUpDown(cells, index, day, delta + deltaIfDisabled, deltaIfDisabled);
|
|
1735
|
-
return nextCell;
|
|
1736
|
-
}
|
|
1737
|
-
_findDayFirst(cells, index, day, date) {
|
|
1738
|
-
const newDateValue = this._dateAdapter.toIso8601(this._dateAdapter.createDate(+day.yearValue, +day.monthValue, date));
|
|
1739
|
-
if (this._isDayOutOfView(newDateValue)) return cells[index];
|
|
1740
|
-
const nextCell = cells.find((e) => this._mapValueToISODate(e.value) === newDateValue);
|
|
1741
|
-
if (!nextCell || nextCell.disabled) return this._findDayFirst(cells, index, day, date + 1);
|
|
1742
|
-
return nextCell;
|
|
1743
|
-
}
|
|
1744
|
-
_findDayLast(cells, index, firstNextMonth) {
|
|
1745
|
-
const newDateValue = this._dateAdapter.toIso8601(this._dateAdapter.addCalendarDays(firstNextMonth, -1));
|
|
1746
|
-
if (this._isDayOutOfView(newDateValue)) return cells[index];
|
|
1747
|
-
const nextCell = cells.find((e) => this._mapValueToISODate(e.value) === newDateValue);
|
|
1748
|
-
if (!nextCell || nextCell.disabled) return this._findDayLast(cells, index, this._dateAdapter.deserialize(newDateValue));
|
|
1749
|
-
return nextCell;
|
|
1750
|
-
}
|
|
1751
|
-
/**
|
|
1752
|
-
* Gets the index of the element to move to, based on a list of elements (which can be potentially disabled),
|
|
1753
|
-
* the keyboard input and the position of the current element in the list.
|
|
1754
|
-
* In the day view, the `day?: Day` parameter is mandatory for calculation,
|
|
1755
|
-
* while in month and year view it's not due to the fixed amount of rendered cells.
|
|
1756
|
-
*/
|
|
1757
|
-
_navigateByKeyboard(evt, index, cells) {
|
|
1758
|
-
const { elementIndexForWideMode, offsetForWideMode, lastElementIndexForWideMode, verticalOffset } = this._calculateParametersForKeyboardNavigation(index, this._calendarView === "year");
|
|
1759
|
-
switch (evt.key) {
|
|
1760
|
-
case "ArrowUp": return this._findNext(cells, index, -verticalOffset);
|
|
1761
|
-
case "ArrowDown": return this._findNext(cells, index, verticalOffset);
|
|
1762
|
-
case "ArrowLeft": return this._findNext(cells, index, -1);
|
|
1763
|
-
case "ArrowRight": return this._findNext(cells, index, 1);
|
|
1764
|
-
case "Home": return this._findFirst(cells, offsetForWideMode);
|
|
1765
|
-
case "PageUp": return this._findFirstOnColumn(cells, elementIndexForWideMode, offsetForWideMode, verticalOffset);
|
|
1766
|
-
case "PageDown": return this._findLastOnColumn(cells, index, lastElementIndexForWideMode, verticalOffset);
|
|
1767
|
-
case "End": return this._findLast(cells, lastElementIndexForWideMode - 1);
|
|
1768
|
-
default: return cells[index];
|
|
1769
|
-
}
|
|
1770
|
-
}
|
|
1771
|
-
/**
|
|
1772
|
-
* Calculates the parameters needed in keyboard navigation in year and month view.
|
|
1773
|
-
* @param index The starting element's index in the cell array.
|
|
1774
|
-
* @param isYearView Whether the displayed `view` is the year one.
|
|
1775
|
-
*/
|
|
1776
|
-
_calculateParametersForKeyboardNavigation(index, isYearView) {
|
|
1777
|
-
const elementsPerPage = isYearView ? YEARS_PER_PAGE : MONTHS_PER_PAGE;
|
|
1778
|
-
const offset = Math.trunc(index / elementsPerPage) * elementsPerPage;
|
|
1779
|
-
const indexInView = offset === 0 ? index : index - elementsPerPage;
|
|
1780
|
-
return {
|
|
1781
|
-
verticalOffset: isYearView ? YEARS_PER_ROW : MONTHS_PER_ROW,
|
|
1782
|
-
elementIndexForWideMode: indexInView,
|
|
1783
|
-
offsetForWideMode: index - indexInView,
|
|
1784
|
-
lastElementIndexForWideMode: offset === 0 ? elementsPerPage : elementsPerPage * 2
|
|
1785
|
-
};
|
|
1786
|
-
}
|
|
1787
|
-
/**
|
|
1788
|
-
* Gets the next element of the provided array starting from `index` by adding `delta`.
|
|
1789
|
-
* If the found element is disabled, it continues adding `delta` until it finds an enabled one in the array bounds.
|
|
1790
|
-
*/
|
|
1791
|
-
_findNext(days, index, delta) {
|
|
1792
|
-
let nextIndex = index + delta;
|
|
1793
|
-
while (nextIndex < days.length && days[nextIndex]?.disabled) nextIndex += delta;
|
|
1794
|
-
return days[nextIndex] ?? days[index];
|
|
1795
|
-
}
|
|
1796
|
-
/** Find the first enabled element in the provided array. */
|
|
1797
|
-
_findFirst(days, firstOfCurrentMonth) {
|
|
1798
|
-
return !days[firstOfCurrentMonth].disabled ? days[firstOfCurrentMonth] : this._findNext(days, firstOfCurrentMonth, 1);
|
|
1799
|
-
}
|
|
1800
|
-
/** Find the last enabled element in the provided array. */
|
|
1801
|
-
_findLast(days, lastOfCurrentMonth) {
|
|
1802
|
-
return !days[lastOfCurrentMonth].disabled ? days[lastOfCurrentMonth] : this._findNext(days, lastOfCurrentMonth, -1);
|
|
1803
|
-
}
|
|
1804
|
-
/** Find the first enabled element in the same column of the provided array. */
|
|
1805
|
-
_findFirstOnColumn(days, index, offset, verticalOffset) {
|
|
1806
|
-
const nextIndex = index % verticalOffset + offset;
|
|
1807
|
-
return !days[nextIndex].disabled ? days[nextIndex] : this._findNext(days, nextIndex, verticalOffset);
|
|
1808
|
-
}
|
|
1809
|
-
/** Find the last enabled element in the same column of the provided array. */
|
|
1810
|
-
_findLastOnColumn(days, index, offset, verticalOffset) {
|
|
1811
|
-
const nextIndex = index + Math.trunc((offset - index - 1) / verticalOffset) * verticalOffset;
|
|
1812
|
-
return !days[nextIndex].disabled ? days[nextIndex] : this._findNext(days, nextIndex, -verticalOffset);
|
|
1813
|
-
}
|
|
1814
|
-
_resetCalendarViewAndEmitMonthChange(initTransition = false) {
|
|
1815
|
-
this._resetCalendarView(initTransition);
|
|
1816
|
-
this._emitMonthChange();
|
|
1817
|
-
}
|
|
1818
|
-
_resetCalendarView(initTransition = false) {
|
|
1819
|
-
if (this._containingFocus) this._resetFocus = true;
|
|
1820
|
-
this._activeDate = (this.multiple ? this._selected.at(-1) : this._selected) ?? this._dateAdapter.today();
|
|
1821
|
-
this._setChosenYear();
|
|
1822
|
-
this._chosenMonth = void 0;
|
|
1823
|
-
this._init();
|
|
1824
|
-
this._nextCalendarView = this._calendarView = this.view;
|
|
1825
|
-
if (initTransition) this._startTableTransition();
|
|
1826
|
-
}
|
|
1827
|
-
/** Render the view for the day selection. */
|
|
1828
|
-
_renderDayView() {
|
|
1829
|
-
const nextMonthActiveDate = this._wide ? this._dateAdapter.addCalendarMonths(this._activeDate, 1) : void 0;
|
|
1830
|
-
return html`
|
|
1831
|
-
<div class="sbb-calendar__controls">
|
|
1832
|
-
${this._getArrow("left", () => this._goToDifferentMonth(-1), i18nPreviousMonth[this._language.current], this._previousMonthDisabled())}
|
|
1833
|
-
<div class="sbb-calendar__controls-month">
|
|
1834
|
-
${this._createLabelForDayView(this._activeDate)}
|
|
1835
|
-
${this._wide ? this._createLabelForDayView(nextMonthActiveDate) : nothing}
|
|
1836
|
-
<sbb-screen-reader-only role="status">
|
|
1837
|
-
${this._createAriaLabelForDayView(this._activeDate, nextMonthActiveDate)}
|
|
1838
|
-
</sbb-screen-reader-only>
|
|
1839
|
-
</div>
|
|
1840
|
-
${this._getArrow("right", () => this._goToDifferentMonth(1), i18nNextMonth[this._language.current], this._nextMonthDisabled())}
|
|
1841
|
-
</div>
|
|
1842
|
-
<div class="sbb-calendar__table-overflow-break">
|
|
1843
|
-
<div class="sbb-calendar__table-container sbb-calendar__table-day-view">
|
|
1844
|
-
${this.orientation === "horizontal" ? html`
|
|
1845
|
-
${this._createDayTable(this._weeks, this._weekNumbers)}
|
|
1846
|
-
${this._wide ? this._createDayTable(this._nextMonthWeeks, this._nextMonthWeekNumbers, true) : nothing}
|
|
1847
|
-
` : html`
|
|
1848
|
-
${this._createDayTableVertical(this._weeks, this._weekNumbers)}
|
|
1849
|
-
${this._wide ? this._createDayTableVertical(this._nextMonthWeeks, this._nextMonthWeekNumbers, nextMonthActiveDate) : nothing}
|
|
1850
|
-
`}
|
|
1851
|
-
</div>
|
|
1852
|
-
</div>
|
|
1853
|
-
`;
|
|
1854
|
-
}
|
|
1855
|
-
/** Creates the label with the month for the daily view. */
|
|
1856
|
-
_createLabelForDayView(d) {
|
|
1857
|
-
const monthLabel = `${this._monthNames[this._dateAdapter.getMonth(d) - 1]} ${this._dateAdapter.getYear(d)}`;
|
|
1858
|
-
return html`
|
|
1859
|
-
<button
|
|
1860
|
-
type="button"
|
|
1861
|
-
class="sbb-calendar__date-selection sbb-calendar__controls-change-date"
|
|
1862
|
-
aria-label="${i18nYearMonthSelection[this._language.current]} ${monthLabel}"
|
|
1863
|
-
@click=${() => {
|
|
1864
|
-
this._resetFocus = true;
|
|
1865
|
-
this._nextCalendarView = "year";
|
|
1866
|
-
this._startTableTransition();
|
|
1867
|
-
}}
|
|
1868
|
-
>
|
|
1869
|
-
${monthLabel}
|
|
1870
|
-
<sbb-icon name="chevron-small-down-small"></sbb-icon>
|
|
1871
|
-
</button>
|
|
1872
|
-
`;
|
|
1873
|
-
}
|
|
1874
|
-
/** Creates the aria-label for the daily view. */
|
|
1875
|
-
_createAriaLabelForDayView(...dates) {
|
|
1876
|
-
let monthLabel = "";
|
|
1877
|
-
for (const d of dates) if (d) monthLabel += `${this._monthNames[this._dateAdapter.getMonth(d) - 1]} ${this._dateAdapter.getYear(d)} `;
|
|
1878
|
-
return monthLabel;
|
|
1879
|
-
}
|
|
1880
|
-
/** Creates the calendar table for the daily view. */
|
|
1881
|
-
_createDayTable(weeks, weekNumbers, isWideNextMonth = false) {
|
|
1882
|
-
const weeksForSelectMultipleWeekNumbers = (this._wide ? [...this._weeks, ...this._nextMonthWeeks] : isWideNextMonth ? this._nextMonthWeeks : this._weeks).flat();
|
|
1883
|
-
const weeksForSelectMultipleWeekDays = (isWideNextMonth ? this._nextMonthWeeks : this._weeks).flat();
|
|
1884
|
-
return html`
|
|
1885
|
-
<table
|
|
1886
|
-
class="sbb-calendar__table"
|
|
1887
|
-
@focusout=${(event) => this._handleTableBlur(event.relatedTarget)}
|
|
1888
|
-
@animationend=${(e) => this._tableAnimationEnd(e)}
|
|
1889
|
-
>
|
|
1890
|
-
<thead class="sbb-calendar__table-header">
|
|
1891
|
-
<tr>
|
|
1892
|
-
${this.weekNumbers ? html`<th class="sbb-calendar__table-header-cell"></th>` : nothing}
|
|
1893
|
-
${this._weekdays.map((weekDay, index) => html`
|
|
1894
|
-
<th class="sbb-calendar__table-header-cell">
|
|
1895
|
-
${this.multiple ? html`
|
|
1896
|
-
<sbb-calendar-weekday
|
|
1897
|
-
.value=${weekDay}
|
|
1898
|
-
@click=${() => {
|
|
1899
|
-
const days = weeksForSelectMultipleWeekDays.filter((day) => day.weekDayValue === (index + 1) % 7);
|
|
1900
|
-
this._selectMultipleDates(days);
|
|
1901
|
-
}}
|
|
1902
|
-
></sbb-calendar-weekday>
|
|
1903
|
-
` : html`
|
|
1904
|
-
<sbb-screen-reader-only>${weekDay.long}</sbb-screen-reader-only>
|
|
1905
|
-
<span aria-hidden="true">${weekDay.narrow}</span>
|
|
1906
|
-
`}
|
|
1907
|
-
</th>
|
|
1908
|
-
`)}
|
|
1909
|
-
</tr>
|
|
1910
|
-
</thead>
|
|
1911
|
-
<tbody class="sbb-calendar__table-body">
|
|
1912
|
-
${weeks.map((week, rowIndex) => {
|
|
1913
|
-
const firstRowOffset = DAYS_PER_ROW - week.length;
|
|
1914
|
-
if (rowIndex === 0 && firstRowOffset) return html`
|
|
1915
|
-
<tr>
|
|
1916
|
-
${this.weekNumbers ? html`
|
|
1917
|
-
<td class="sbb-calendar__table-header-cell-vertical">
|
|
1918
|
-
${this.multiple ? html`
|
|
1919
|
-
<sbb-calendar-weeknumber
|
|
1920
|
-
.value=${weekNumbers[0]}
|
|
1921
|
-
@click=${() => {
|
|
1922
|
-
const days = weeksForSelectMultipleWeekNumbers.filter((day) => day.weekValue === weekNumbers[0]);
|
|
1923
|
-
this._selectMultipleDates(days);
|
|
1924
|
-
}}
|
|
1925
|
-
></sbb-calendar-weeknumber>
|
|
1926
|
-
` : html`
|
|
1927
|
-
<sbb-screen-reader-only
|
|
1928
|
-
>${`${i18nCalendarWeekNumber[this._language.current]} ${weekNumbers[0]}`}</sbb-screen-reader-only
|
|
1929
|
-
>
|
|
1930
|
-
<span aria-hidden="true">${weekNumbers[0]}</span>
|
|
1931
|
-
`}
|
|
1932
|
-
</td>
|
|
1933
|
-
` : nothing}
|
|
1934
|
-
${[...Array(firstRowOffset).keys()].map(() => html`<td class="sbb-calendar__table-data"></td>`)}
|
|
1935
|
-
${this._createDayCells(week)}
|
|
1936
|
-
</tr>
|
|
1937
|
-
`;
|
|
1938
|
-
return html`
|
|
1939
|
-
<tr>
|
|
1940
|
-
${this.weekNumbers ? html`
|
|
1941
|
-
<td class="sbb-calendar__table-header-cell-vertical">
|
|
1942
|
-
${this.multiple ? html`
|
|
1943
|
-
<sbb-calendar-weeknumber
|
|
1944
|
-
.value=${weekNumbers[rowIndex]}
|
|
1945
|
-
@click=${() => {
|
|
1946
|
-
const days = weeksForSelectMultipleWeekNumbers.filter((day) => day.weekValue === weekNumbers[rowIndex]);
|
|
1947
|
-
this._selectMultipleDates(days);
|
|
1948
|
-
}}
|
|
1949
|
-
></sbb-calendar-weeknumber>
|
|
1950
|
-
` : html`
|
|
1951
|
-
<sbb-screen-reader-only
|
|
1952
|
-
>${`${i18nCalendarWeekNumber[this._language.current]} ${weekNumbers[rowIndex]}`}</sbb-screen-reader-only
|
|
1953
|
-
>
|
|
1954
|
-
<span aria-hidden="true">${weekNumbers[rowIndex]}</span>
|
|
1955
|
-
`}
|
|
1956
|
-
</td>
|
|
1957
|
-
` : nothing}
|
|
1958
|
-
${this._createDayCells(week)}
|
|
1959
|
-
</tr>
|
|
1960
|
-
`;
|
|
1961
|
-
})}
|
|
1962
|
-
</tbody>
|
|
1963
|
-
</table>
|
|
1964
|
-
`;
|
|
1965
|
-
}
|
|
1966
|
-
/** Creates the table in orientation='vertical'. */
|
|
1967
|
-
_createDayTableVertical(weeks, weekNumbers, nextMonthActiveDate) {
|
|
1968
|
-
const weekOffset = this._dateAdapter.getFirstWeekOffset(nextMonthActiveDate ?? this._activeDate);
|
|
1969
|
-
const weeksForSelectMultipleWeekNumbers = (this._wide ? [...this._weeks, ...this._nextMonthWeeks] : nextMonthActiveDate ? this._nextMonthWeeks : this._weeks).flat();
|
|
1970
|
-
return html`
|
|
1971
|
-
<table
|
|
1972
|
-
class="sbb-calendar__table"
|
|
1973
|
-
@focusout=${(event) => this._handleTableBlur(event.relatedTarget)}
|
|
1974
|
-
@animationend=${(e) => this._tableAnimationEnd(e)}
|
|
1975
|
-
>
|
|
1976
|
-
${this.weekNumbers ? html`
|
|
1977
|
-
<thead class="sbb-calendar__table-header">
|
|
1978
|
-
<tr>
|
|
1979
|
-
${nextMonthActiveDate ? nothing : html`<th class="sbb-calendar__table-data"></th>`}
|
|
1980
|
-
${weekNumbers.map((weekNumber) => html`
|
|
1981
|
-
<th class="sbb-calendar__table-header-cell">
|
|
1982
|
-
${this.multiple ? html`
|
|
1983
|
-
<sbb-calendar-weeknumber
|
|
1984
|
-
.value=${weekNumber}
|
|
1985
|
-
@click=${() => {
|
|
1986
|
-
const days = weeksForSelectMultipleWeekNumbers.filter((day) => day.weekValue === weekNumber);
|
|
1987
|
-
this._selectMultipleDates(days);
|
|
1988
|
-
}}
|
|
1989
|
-
></sbb-calendar-weeknumber>
|
|
1990
|
-
` : html`
|
|
1991
|
-
<sbb-screen-reader-only
|
|
1992
|
-
>${`${i18nCalendarWeekNumber[this._language.current]} ${weekNumber}`}</sbb-screen-reader-only
|
|
1993
|
-
>
|
|
1994
|
-
<span aria-hidden="true">${weekNumber}</span>
|
|
1995
|
-
`}
|
|
1996
|
-
</th>
|
|
1997
|
-
`)}
|
|
1998
|
-
</tr>
|
|
1999
|
-
</thead>
|
|
2000
|
-
` : nothing}
|
|
2001
|
-
<tbody class="sbb-calendar__table-body">
|
|
2002
|
-
${weeks.map((week, rowIndex) => {
|
|
2003
|
-
const weekday = this._weekdays[rowIndex];
|
|
2004
|
-
const selectableDays = this._wide ? [...week, ...this._nextMonthWeeks[rowIndex]] : week;
|
|
2005
|
-
return html`
|
|
2006
|
-
<tr>
|
|
2007
|
-
${nextMonthActiveDate ? nothing : html`
|
|
2008
|
-
<td class="sbb-calendar__table-header-cell-vertical">
|
|
2009
|
-
${this.multiple ? html`
|
|
2010
|
-
<sbb-calendar-weekday
|
|
2011
|
-
.value=${weekday}
|
|
2012
|
-
@click=${() => this._selectMultipleDates(selectableDays)}
|
|
2013
|
-
>
|
|
2014
|
-
${weekday.narrow}
|
|
2015
|
-
</sbb-calendar-weekday>
|
|
2016
|
-
` : html`
|
|
2017
|
-
<sbb-screen-reader-only>${weekday.long}</sbb-screen-reader-only>
|
|
2018
|
-
<span aria-hidden="true">${weekday.narrow}</span>
|
|
2019
|
-
`}
|
|
2020
|
-
</td>
|
|
2021
|
-
`}
|
|
2022
|
-
${rowIndex < weekOffset ? html`<td class="sbb-calendar__table-data"></td>` : nothing}
|
|
2023
|
-
${this._createDayCells(week)}
|
|
2024
|
-
</tr>
|
|
2025
|
-
`;
|
|
2026
|
-
})}
|
|
2027
|
-
</tbody>
|
|
2028
|
-
</table>
|
|
2029
|
-
`;
|
|
2030
|
-
}
|
|
2031
|
-
/** Creates the cells for the daily view. */
|
|
2032
|
-
_createDayCells(week) {
|
|
2033
|
-
return week.map((day) => {
|
|
2034
|
-
return html`
|
|
2035
|
-
<td class="sbb-calendar__table-data sbb-calendar__day-cell">
|
|
2036
|
-
<slot name=${day.value}>
|
|
2037
|
-
<sbb-calendar-day
|
|
2038
|
-
slot=${day.value}
|
|
2039
|
-
@click=${() => this._selectDate(day.dateValue)}
|
|
2040
|
-
@keydown=${(evt) => this._handleKeyboardEvent(evt, day)}
|
|
2041
|
-
></sbb-calendar-day>
|
|
2042
|
-
</slot>
|
|
2043
|
-
</td>
|
|
2044
|
-
`;
|
|
2045
|
-
});
|
|
2046
|
-
}
|
|
2047
|
-
/** Render the view for the month selection. */
|
|
2048
|
-
_renderMonthView() {
|
|
2049
|
-
return html`
|
|
2050
|
-
<div class="sbb-calendar__controls">
|
|
2051
|
-
${this._getArrow("left", () => this._goToDifferentYear(-1), i18nPreviousYear[this._language.current], this._previousYearDisabled())}
|
|
2052
|
-
<div class="sbb-calendar__controls-month">${this._createLabelForMonthView()}</div>
|
|
2053
|
-
${this._getArrow("right", () => this._goToDifferentYear(1), i18nNextYear[this._language.current], this._nextYearDisabled())}
|
|
2054
|
-
</div>
|
|
2055
|
-
<div class="sbb-calendar__table-overflow-break">
|
|
2056
|
-
<div class="sbb-calendar__table-container sbb-calendar__table-month-view">
|
|
2057
|
-
${this._createMonthTable(this._months, this._chosenYear)}
|
|
2058
|
-
${this._wide ? this._createMonthTable(this._months, this._chosenYear + 1) : nothing}
|
|
2059
|
-
</div>
|
|
2060
|
-
</div>
|
|
2061
|
-
`;
|
|
2062
|
-
}
|
|
2063
|
-
/** Creates the label with the year for the monthly view. */
|
|
2064
|
-
_createLabelForMonthView() {
|
|
2065
|
-
return html` <button
|
|
2066
|
-
type="button"
|
|
2067
|
-
id="sbb-calendar__month-selection"
|
|
2068
|
-
class="sbb-calendar__controls-change-date"
|
|
2069
|
-
aria-label=${`${i18nCalendarDateSelection[this._language.current]} ${this._chosenYear}`}
|
|
2070
|
-
@click=${() => this._resetCalendarViewAndEmitMonthChange(true)}
|
|
2071
|
-
>
|
|
2072
|
-
${this._chosenYear} ${this._wide ? ` - ${this._chosenYear + 1}` : nothing}
|
|
2073
|
-
<sbb-icon name="chevron-small-up-small"></sbb-icon>
|
|
2074
|
-
</button>
|
|
2075
|
-
<sbb-screen-reader-only role="status"> ${this._chosenYear} </sbb-screen-reader-only>`;
|
|
2076
|
-
}
|
|
2077
|
-
/** Creates the table for the month selection view. */
|
|
2078
|
-
_createMonthTable(months, year) {
|
|
2079
|
-
return html`
|
|
2080
|
-
<table
|
|
2081
|
-
class="sbb-calendar__table"
|
|
2082
|
-
@animationend=${(e) => this._tableAnimationEnd(e)}
|
|
2083
|
-
>
|
|
2084
|
-
${this._wide ? html`<thead class="sbb-calendar__table-header" aria-hidden="true">
|
|
2085
|
-
<tr>
|
|
2086
|
-
<th class="sbb-calendar__table-header-cell" colspan=${MONTHS_PER_ROW}>${year}</th>
|
|
2087
|
-
</tr>
|
|
2088
|
-
</thead>` : nothing}
|
|
2089
|
-
<tbody class="sbb-calendar__table-body">
|
|
2090
|
-
${months.map((row) => html`
|
|
2091
|
-
<tr>
|
|
2092
|
-
${row.map((month) => {
|
|
2093
|
-
return html`
|
|
2094
|
-
<td class="sbb-calendar__table-data">
|
|
2095
|
-
<sbb-calendar-month
|
|
2096
|
-
.value="${year}-${month.value}"
|
|
2097
|
-
@click=${() => this._onMonthSelection(month.monthValue, year)}
|
|
2098
|
-
@keydown=${(evt) => this._handleKeyboardEvent(evt)}
|
|
2099
|
-
>
|
|
2100
|
-
</sbb-calendar-month>
|
|
2101
|
-
</td>
|
|
2102
|
-
`;
|
|
2103
|
-
})}
|
|
2104
|
-
</tr>
|
|
2105
|
-
`)}
|
|
2106
|
-
</tbody>
|
|
2107
|
-
</table>
|
|
2108
|
-
`;
|
|
2109
|
-
}
|
|
2110
|
-
/** Select the month and change the view to day selection. */
|
|
2111
|
-
_onMonthSelection(month, year) {
|
|
2112
|
-
this._chosenMonth = month;
|
|
2113
|
-
this._nextCalendarView = "day";
|
|
2114
|
-
this._init(this._dateAdapter.createDate(year, this._chosenMonth, this._dateAdapter.getDate(this._activeDate)));
|
|
2115
|
-
this._startTableTransition();
|
|
2116
|
-
this._emitMonthChange();
|
|
2117
|
-
}
|
|
2118
|
-
/** Render the view for the year selection. */
|
|
2119
|
-
_renderYearView() {
|
|
2120
|
-
return html`
|
|
2121
|
-
<div class="sbb-calendar__controls">
|
|
2122
|
-
${this._getArrow("left", () => this._goToDifferentYearRange(-YEARS_PER_PAGE), i18nPreviousYearRange(YEARS_PER_PAGE)[this._language.current], this._previousYearRangeDisabled())}
|
|
2123
|
-
<div class="sbb-calendar__controls-month">${this._createLabelForYearView()}</div>
|
|
2124
|
-
${this._getArrow("right", () => this._goToDifferentYearRange(YEARS_PER_PAGE), i18nNextYearRange(YEARS_PER_PAGE)[this._language.current], this._nextYearRangeDisabled())}
|
|
2125
|
-
</div>
|
|
2126
|
-
<div class="sbb-calendar__table-overflow-break">
|
|
2127
|
-
<div class="sbb-calendar__table-container sbb-calendar__table-year-view">
|
|
2128
|
-
${this._createYearTable(this._years)}
|
|
2129
|
-
${this._wide ? this._createYearTable(this._nextMonthYears, true) : nothing}
|
|
2130
|
-
</div>
|
|
2131
|
-
</div>
|
|
2132
|
-
`;
|
|
2133
|
-
}
|
|
2134
|
-
/** Creates the button arrow for all the views. */
|
|
2135
|
-
_getArrow(direction, click, ariaLabel, disabled) {
|
|
2136
|
-
return html`<sbb-secondary-button
|
|
2137
|
-
size="m"
|
|
2138
|
-
icon-name="chevron-small-${direction}-small"
|
|
2139
|
-
aria-label=${ariaLabel}
|
|
2140
|
-
@click=${click}
|
|
2141
|
-
?disabled=${disabled}
|
|
2142
|
-
id="sbb-calendar__controls-${direction === "left" ? "previous" : "next"}"
|
|
2143
|
-
></sbb-secondary-button>`;
|
|
2144
|
-
}
|
|
2145
|
-
/** Creates the label with the year range for the yearly view. */
|
|
2146
|
-
_createLabelForYearView() {
|
|
2147
|
-
const firstYear = this._years.flat()[0];
|
|
2148
|
-
const lastYearArray = (this._wide ? this._nextMonthYears : this._years).flat();
|
|
2149
|
-
const yearLabel = `${firstYear} - ${lastYearArray[lastYearArray.length - 1]}`;
|
|
2150
|
-
return html`
|
|
2151
|
-
<button
|
|
2152
|
-
type="button"
|
|
2153
|
-
id="sbb-calendar__year-selection"
|
|
2154
|
-
class="sbb-calendar__controls-change-date"
|
|
2155
|
-
aria-label="${i18nCalendarDateSelection[this._language.current]} ${yearLabel}"
|
|
2156
|
-
@click=${() => this._resetCalendarViewAndEmitMonthChange(true)}
|
|
2157
|
-
>
|
|
2158
|
-
${yearLabel}
|
|
2159
|
-
<sbb-icon name="chevron-small-up-small"></sbb-icon>
|
|
2160
|
-
</button>
|
|
2161
|
-
<sbb-screen-reader-only role="status"> ${yearLabel} </sbb-screen-reader-only>
|
|
2162
|
-
`;
|
|
2163
|
-
}
|
|
2164
|
-
/** Creates the table for the year selection view. */
|
|
2165
|
-
_createYearTable(years, shiftRight = false) {
|
|
2166
|
-
return html` <table
|
|
2167
|
-
class="sbb-calendar__table"
|
|
2168
|
-
@animationend=${(e) => this._tableAnimationEnd(e)}
|
|
2169
|
-
>
|
|
2170
|
-
<tbody class="sbb-calendar__table-body">
|
|
2171
|
-
${years.map((row) => html` <tr>
|
|
2172
|
-
${row.map((year) => {
|
|
2173
|
-
return html`
|
|
2174
|
-
<td class="sbb-calendar__table-data">
|
|
2175
|
-
<sbb-calendar-year
|
|
2176
|
-
.value=${String(year)}
|
|
2177
|
-
@keydown=${(evt) => this._handleKeyboardEvent(evt)}
|
|
2178
|
-
@click=${() => this._onYearSelection(year, shiftRight)}
|
|
2179
|
-
>
|
|
2180
|
-
</sbb-calendar-year>
|
|
2181
|
-
</td>
|
|
2182
|
-
`;
|
|
2183
|
-
})}
|
|
2184
|
-
</tr>`)}
|
|
2185
|
-
</tbody>
|
|
2186
|
-
</table>`;
|
|
2187
|
-
}
|
|
2188
|
-
/** Select the year and change the view to month selection. */
|
|
2189
|
-
_onYearSelection(year, rightSide) {
|
|
2190
|
-
this._chosenYear = rightSide ? year - 1 : year;
|
|
2191
|
-
this._nextCalendarView = "month";
|
|
2192
|
-
this._assignActiveDate(this._dateAdapter.createDate(this._chosenYear, this._dateAdapter.getMonth(this._activeDate), this._dateAdapter.getDate(this._activeDate)));
|
|
2193
|
-
this._startTableTransition();
|
|
2194
|
-
}
|
|
2195
|
-
_getView() {
|
|
2196
|
-
if (isServer || this.hydrationRequired) return html`${nothing}`;
|
|
2197
|
-
switch (this._calendarView) {
|
|
2198
|
-
case "year": return this._renderYearView();
|
|
2199
|
-
case "month": return this._renderMonthView();
|
|
2200
|
-
default: return this._renderDayView();
|
|
2201
|
-
}
|
|
2202
|
-
}
|
|
2203
|
-
_tableAnimationEnd(event) {
|
|
2204
|
-
const table = event.target;
|
|
2205
|
-
if (event.animationName === "hide") {
|
|
2206
|
-
table.classList.remove("sbb-calendar__table-hide");
|
|
2207
|
-
if (this._containingFocus) this._resetFocus = true;
|
|
2208
|
-
this._calendarView = this._nextCalendarView;
|
|
2209
|
-
} else if (event.animationName === "show") this.internals.states.delete("transition");
|
|
2210
|
-
}
|
|
2211
|
-
_startTableTransition() {
|
|
2212
|
-
this.internals.states.add("transition");
|
|
2213
|
-
this.shadowRoot?.querySelectorAll("table")?.forEach((e) => e.classList.toggle("sbb-calendar__table-hide"));
|
|
2214
|
-
}
|
|
2215
|
-
render() {
|
|
2216
|
-
return html`<div class="sbb-calendar__wrapper">${this._getView()}</div>`;
|
|
2217
|
-
}
|
|
2218
|
-
};
|
|
2219
|
-
})();
|
|
2220
|
-
//#endregion
|
|
2221
|
-
export { SbbMonthChangeEvent as n, SbbCalendarElement as t };
|
|
2222
|
-
|
|
2223
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY2FsZW5kYXIuY29tcG9uZW50LUNSdlRZdzFLLmpzIiwibmFtZXMiOltdLCJzb3VyY2VzIjpbIi4uLy4uLy4uL25vZGVfbW9kdWxlcy9kYXRlLWZucy9jb25zdGFudHMuanMiLCIuLi8uLi8uLi9ub2RlX21vZHVsZXMvZGF0ZS1mbnMvY29uc3RydWN0RnJvbS5qcyIsIi4uLy4uLy4uL25vZGVfbW9kdWxlcy9kYXRlLWZucy90b0RhdGUuanMiLCIuLi8uLi8uLi9ub2RlX21vZHVsZXMvZGF0ZS1mbnMvYWRkRGF5cy5qcyIsIi4uLy4uLy4uL25vZGVfbW9kdWxlcy9kYXRlLWZucy9fbGliL2RlZmF1bHRPcHRpb25zLmpzIiwiLi4vLi4vLi4vbm9kZV9tb2R1bGVzL2RhdGUtZm5zL3N0YXJ0T2ZXZWVrLmpzIiwiLi4vLi4vLi4vbm9kZV9tb2R1bGVzL2RhdGUtZm5zL19saWIvbm9ybWFsaXplRGF0ZXMuanMiLCIuLi8uLi8uLi9ub2RlX21vZHVsZXMvZGF0ZS1mbnMvYWRkV2Vla3MuanMiLCIuLi8uLi8uLi9ub2RlX21vZHVsZXMvZGF0ZS1mbnMvZW5kT2ZNb250aC5qcyIsIi4uLy4uLy4uL25vZGVfbW9kdWxlcy9kYXRlLWZucy9fbGliL25vcm1hbGl6ZUludGVydmFsLmpzIiwiLi4vLi4vLi4vbm9kZV9tb2R1bGVzL2RhdGUtZm5zL2VhY2hXZWVrT2ZJbnRlcnZhbC5qcyIsIi4uLy4uLy4uL25vZGVfbW9kdWxlcy9kYXRlLWZucy9zdGFydE9mTW9udGguanMiLCIuLi8uLi8uLi9ub2RlX21vZHVsZXMvZGF0ZS1mbnMvZ2V0V2Vla1llYXIuanMiLCIuLi8uLi8uLi9ub2RlX21vZHVsZXMvZGF0ZS1mbnMvc3RhcnRPZldlZWtZZWFyLmpzIiwiLi4vLi4vLi4vbm9kZV9tb2R1bGVzL2RhdGUtZm5zL2dldFdlZWsuanMiLCIuLi8uLi8uLi9zcmMvZWxlbWVudHMvY2FsZW5kYXIvY2FsZW5kYXIvY2FsZW5kYXIuc2Nzcz9saXQmaW5saW5lIiwiLi4vLi4vLi4vc3JjL2VsZW1lbnRzL2NhbGVuZGFyL2NhbGVuZGFyL2NhbGVuZGFyLmNvbXBvbmVudC50cyJdLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIEBtb2R1bGUgY29uc3RhbnRzXG4gKiBAc3VtbWFyeSBVc2VmdWwgY29uc3RhbnRzXG4gKiBAZGVzY3JpcHRpb25cbiAqIENvbGxlY3Rpb24gb2YgdXNlZnVsIGRhdGUgY29uc3RhbnRzLlxuICpcbiAqIFRoZSBjb25zdGFudHMgY291bGQgYmUgaW1wb3J0ZWQgZnJvbSBgZGF0ZS1mbnMvY29uc3RhbnRzYDpcbiAqXG4gKiBgYGB0c1xuICogaW1wb3J0IHsgbWF4VGltZSwgbWluVGltZSB9IGZyb20gXCIuL2NvbnN0YW50cy9kYXRlLWZucy9jb25zdGFudHNcIjtcbiAqXG4gKiBmdW5jdGlvbiBpc0FsbG93ZWRUaW1lKHRpbWUpIHtcbiAqICAgcmV0dXJuIHRpbWUgPD0gbWF4VGltZSAmJiB0aW1lID49IG1pblRpbWU7XG4gKiB9XG4gKiBgYGBcbiAqL1xuXG4vKipcbiAqIEBjb25zdGFudFxuICogQG5hbWUgZGF5c0luV2Vla1xuICogQHN1bW1hcnkgRGF5cyBpbiAxIHdlZWsuXG4gKi9cbmV4cG9ydCBjb25zdCBkYXlzSW5XZWVrID0gNztcblxuLyoqXG4gKiBAY29uc3RhbnRcbiAqIEBuYW1lIGRheXNJblllYXJcbiAqIEBzdW1tYXJ5IERheXMgaW4gMSB5ZWFyLlxuICpcbiAqIEBkZXNjcmlwdGlvblxuICogSG93IG1hbnkgZGF5cyBpbiBhIHllYXIuXG4gKlxuICogT25lIHllYXJzIGVxdWFscyAzNjUuMjQyNSBkYXlzIGFjY29yZGluZyB0byB0aGUgZm9ybXVsYTpcbiAqXG4gKiA+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/LmluKTtcbiAgY29uc3QgZGlmZiA9ICtzdGFydE9mV2VlayhfZGF0ZSwgb3B0aW9ucykgLSArc3RhcnRPZldlZWtZZWFyKF9kYXRlLCBvcHRpb25zKTtcblxuICAvLyBSb3VuZCB0aGUgbnVtYmVyIG9mIHdlZWtzIHRvIHRoZSBuZWFyZXN0IGludGVnZXIgYmVjYXVzZSB0aGUgbnVtYmVyIG9mXG4gIC8vIG1pbGxpc2Vjb25kcyBpbiBhIHdlZWsgaXMgbm90IGNvbnN0YW50IChlLmcuIGl0J3MgZGlmZmVyZW50IGluIHRoZSB3ZWVrIG9mXG4gIC8vIHRoZSBkYXlsaWdodCBzYXZpbmcgdGltZSBjbG9jayBzaGlmdCkuXG4gIHJldHVybiBNYXRoLnJvdW5kKGRpZmYgLyBtaWxsaXNlY29uZHNJbldlZWspICsgMTtcbn1cblxuLy8gRmFsbGJhY2sgZm9yIG1vZHVsYXJpemVkIGltcG9ydHM6XG5leHBvcnQgZGVmYXVsdCBnZXRXZWVrO1xuIiwiQHVzZSAnLi4vLi4vY29yZS9zdHlsZXMnIGFzIHNiYjtcblxuOmhvc3Qge1xuICBkaXNwbGF5OiBibG9jaztcblxuICAvLyBXZSBhZGQgd2lkdGggZGVmaW5pdGlvbiB0byBob3N0LCB0byBtYWtlIG92ZXJ3cml0aW5nIGVhc3kgZm9yIGNvbnN1bWVycy5cbiAgd2lkdGg6IG1heC1jb250ZW50O1xuXG4gIC0tc2JiLWNhbGVuZGFyLWNlbGwtc2l6ZTogI3tzYmIucHgtdG8tcmVtLWJ1aWxkKDQ0KX07XG5cbiAgLy8gTWF0Y2ggRmlnbWEgZGVzaWduXG4gIC0tc2JiLWNhbGVuZGFyLWNlbGwtYm9yZGVyLXJhZGl1czogY2FsYyhcbiAgICB2YXIoLS1zYmItYm9yZGVyLXJhZGl1cy00eCkgKyB2YXIoLS1zYmItY2FsZW5kYXItY2VsbC1ib3JkZXItd2lkdGgpXG4gICk7XG4gIC0tc2JiLWNhbGVuZGFyLWNlbGwtdHJhbnNpdGlvbi1kdXJhdGlvbjogdmFyKFxuICAgIC0tc2JiLWRpc2FibGUtYW5pbWF0aW9uLWR1cmF0aW9uLFxuICAgIHZhcigtLXNiYi1hbmltYXRpb24tZHVyYXRpb24tMngpXG4gICk7XG5cbiAgQGluY2x1ZGUgc2JiLm1xKCRmcm9tOiBzbWFsbCkge1xuICAgIC0tc2JiLWNhbGVuZGFyLWNvbnRyb2wtdmlldy1jaGFuZ2UtaGVpZ2h0OiAje3NiYi5weC10by1yZW0tYnVpbGQoNDgpfTtcbiAgfVxufVxuXG46aG9zdCg6c3RhdGUoZW5oYW5jZWQpKSB7XG4gIC0tc2JiLWNhbGVuZGFyLWNlbGwtcGFkZGluZy1ibG9jay1lbmQ6IHZhcigtLXNiYi1zcGFjaW5nLWZpeGVkLTN4KTtcbn1cblxuLnNiYi1jYWxlbmRhcl9fd3JhcHBlciB7XG4gIHdpZHRoOiAxMDAlO1xuICBkaXNwbGF5OiBibG9jaztcbiAgdHJhbnNpdGlvbi1kdXJhdGlvbjogdmFyKC0tc2JiLWNhbGVuZGFyLWNlbGwtdHJhbnNpdGlvbi1kdXJhdGlvbik7XG59XG5cbi5zYmItY2FsZW5kYXJfX2NvbnRyb2xzIHtcbiAgd2lkdGg6IDEwMCU7XG4gIGRpc3BsYXk6IGlubGluZS1mbGV4O1xuICBhbGlnbi1pdGVtczogY2VudGVyO1xuICBnYXA6IHZhcigtLXNiYi1jYWxlbmRhci1jb250cm9sLWdhcCk7XG4gIG1hcmdpbi1ibG9jay1lbmQ6IHZhcigtLXNiYi1jYWxlbmRhci1jb250cm9sLW1hcmdpbi1ibG9jay1lbmQpO1xufVxuXG4uc2JiLWNhbGVuZGFyX19jb250cm9scy1tb250aCB7XG4gIHdpZHRoOiAxMDAlO1xuICBkaXNwbGF5OiBmbGV4O1xuICBnYXA6IHZhcigtLXNiYi1jYWxlbmRhci10YWJsZXMtZ2FwKTtcbn1cblxuI3NiYi1jYWxlbmRhcl9fY29udHJvbHMtcHJldmlvdXMsXG4jc2JiLWNhbGVuZGFyX19jb250cm9scy1uZXh0IHtcbiAgLXdlYmtpdC10YXAtaGlnaGxpZ2h0LWNvbG9yOiB0cmFuc3BhcmVudDtcbn1cblxuLnNiYi1jYWxlbmRhcl9fY29udHJvbHMtY2hhbmdlLWRhdGUge1xuICBAaW5jbHVkZSBzYmIuYnV0dG9uLXJlc2V0O1xuXG4gIGRpc3BsYXk6IGZsZXg7XG4gIGFsaWduLWl0ZW1zOiBjZW50ZXI7XG4gIG1hcmdpbjogYXV0bztcbiAgaGVpZ2h0OiB2YXIoLS1zYmItY2FsZW5kYXItY29udHJvbC12aWV3LWNoYW5nZS1oZWlnaHQpO1xuICBmb250LXNpemU6IHZhcigtLXNiYi10ZXh0LWZvbnQtc2l6ZS1zKTtcbiAgbGV0dGVyLXNwYWNpbmc6IHZhcigtLXNiYi10eXBvLWxldHRlci1zcGFjaW5nLXRleHQpO1xuICB0ZXh0LXRyYW5zZm9ybTogY2FwaXRhbGl6ZTtcbiAgY3Vyc29yOiB2YXIoLS1zYmItY3Vyc29yLXBvaW50ZXIpO1xuICBwYWRkaW5nLWlubGluZTogdmFyKC0tc2JiLWNhbGVuZGFyLWNvbnRyb2wtdmlldy1jaGFuZ2UtcGFkZGluZy1pbmxpbmUpO1xuICBib3JkZXItcmFkaXVzOiB2YXIoLS1zYmItYm9yZGVyLXJhZGl1cy1pbmZpbml0eSk7XG4gIGJhY2tncm91bmQtY29sb3I6IHZhcigtLXNiYi1jYWxlbmRhci1jb250cm9sLXZpZXctY2hhbmdlLWJhY2tncm91bmQpO1xuICBjb2xvcjogdmFyKC0tc2JiLWNhbGVuZGFyLWNvbnRyb2wtdmlldy1jaGFuZ2UtY29sb3IpO1xuICB0cmFuc2l0aW9uLWR1cmF0aW9uOiB2YXIoLS1zYmItY2FsZW5kYXItY2VsbC10cmFuc2l0aW9uLWR1cmF0aW9uKTtcbiAgdHJhbnNpdGlvbi10aW1pbmctZnVuY3Rpb246IHZhcigtLXNiYi1jYWxlbmRhci1jZWxsLXRyYW5zaXRpb24tZWFzaW5nLWZ1bmN0aW9uKTtcbiAgdHJhbnNpdGlvbi1wcm9wZXJ0eTogYmFja2dyb3VuZC1jb2xvciwgcGFkZGluZy1ibG9jay1lbmQ7XG5cbiAgJjpkaXNhYmxlZCB7XG4gICAgLS1zYmItY2FsZW5kYXItY29udHJvbC12aWV3LWNoYW5nZS1iYWNrZ3JvdW5kOiBsaWdodC1kYXJrKFxuICAgICAgdmFyKC0tc2JiLWNvbG9yLW1pbGspLFxuICAgICAgdmFyKC0tc2JiLWNvbG9yLWFudGhyYWNpdGUpXG4gICAgKTtcbiAgICAtLXNiYi1jYWxlbmRhci1jb250cm9sLXZpZXctY2hhbmdlLWNvbG9yOiBsaWdodC1kYXJrKFxuICAgICAgdmFyKC0tc2JiLWNvbG9yLWdyYW5pdGUpLFxuICAgICAgdmFyKC0tc2JiLWNvbG9yLWFsdW1pbml1bSlcbiAgICApO1xuXG4gICAgY3Vyc29yOiB1bnNldDtcbiAgfVxuXG4gICY6Zm9jdXMtdmlzaWJsZSB7XG4gICAgQGluY2x1ZGUgc2JiLmZvY3VzLW91dGxpbmU7XG5cbiAgICBvdXRsaW5lLW9mZnNldDogdmFyKC0tc2JiLXNwYWNpbmctZml4ZWQtMXgpO1xuICB9XG5cbiAgQGluY2x1ZGUgc2JiLmhvdmVyLW1xIHtcbiAgICAmOm5vdCg6YWN0aXZlLCA6ZGlzYWJsZWQpOmhvdmVyIHtcbiAgICAgIHBhZGRpbmctYmxvY2stZW5kOiB2YXIoLS1zYmItY2FsZW5kYXItY2VsbC1ob3Zlci1zaGlmdCk7XG4gICAgfVxuICB9XG5cbiAgJjpub3QoOmRpc2FibGVkKTphY3RpdmUge1xuICAgIC0tc2JiLWNhbGVuZGFyLWNvbnRyb2wtdmlldy1jaGFuZ2UtYmFja2dyb3VuZDogdmFyKC0tc2JiLWJhY2tncm91bmQtY29sb3ItMyk7XG4gIH1cbn1cblxuLnNiYi1jYWxlbmRhcl9fdGFibGUtbW9udGgtdmlldyxcbi5zYmItY2FsZW5kYXJfX3RhYmxlLXllYXItdmlldyB7XG4gIC0tc2JiLWNhbGVuZGFyLXRhYmxlLWNvbHVtbi1zcGFjZXM6IDY7XG59XG5cbi5zYmItY2FsZW5kYXJfX3RhYmxlLW92ZXJmbG93LWJyZWFrIHtcbiAgLy8gV2UgaGF2ZSB0byBjdXQgdGhlIG5lZ2F0aXZlIG1hcmdpbiB3aGljaCBpcyBidWlsdCBieSB0aGUgY29udGFpbmluZyBzYmItY2FsZW5kYXJfX3RhYmxlLWNvbnRhaW5lci5cbiAgLy8gT3ZlcmZsb3c6IGhpZGRlbiB3b3VsZCBjdXQgdGhlIGZvY3VzIG91dGxpbmUuIFRoZXJlZm9yZSB3ZSB0YWtlIGBjb250YWluYCBoZXJlLlxuICBjb250YWluOiBsYXlvdXQ7XG59XG5cbi5zYmItY2FsZW5kYXJfX3RhYmxlLWNvbnRhaW5lciB7XG4gIGRpc3BsYXk6IGZsZXg7XG4gIGdhcDogdmFyKC0tc2JiLWNhbGVuZGFyLXRhYmxlcy1nYXApO1xuICBtYXJnaW4taW5saW5lOiB2YXIoLS1zYmItY2FsZW5kYXItbWFyZ2luKTtcblxuICAvLyBUaGUgcGFkZGluZyBvZiB0aGUgZmlyc3QgYW5kIGxhc3QgY29sdW1uIHNob3VsZCBub3QgYmUgdmlzaWJsZSBpZiBjYWxlbmRhciBpcyBzdHJldGNoZWQuXG4gIC8vIFRoZXJlZm9yZSB3ZSBuZWVkIGEgbmVnYXRpdmUgaW5saW5lIG1hcmdpbi5cbiAgLy8gQXMgd2UgZG9uJ3Qgd2FudCB0byBzcXVlZXplLCB0aGUgbWFyZ2luIHNob3VsZCBuZXZlciBiZSBncmVhdGVyIHRoYW4gemVyby5cblxuICAvLyBNaW4gd2lkdGggaXMgZXF1YWxzIHRvIHRoZSBub3JtYWwgd2lkdGggb2YgdGhlIGNhbGVuZGFyXG4gIC0tc2JiLWNhbGVuZGFyLW1pbi13aWR0aDogY2FsYyg3ICogdmFyKC0tc2JiLWNhbGVuZGFyLWNlbGwtc2l6ZSkpO1xuXG4gIC8vIFRoZSBvdmVyZmxvdyB2YXJpYWJsZSBpcyBlcXVhbHMgdGhlIGRpZmZlcmVuY2UgYmV0d2VlbiB0aGUgYWN0dWFsIHdpZHRoIGFuZCB0aGUgbWluIHdpZHRoLlxuICAtLXNiYi1jYWxlbmRhci1vdmVyZmxvdzogY2FsYygxMDAlIC0gdmFyKC0tc2JiLWNhbGVuZGFyLW1pbi13aWR0aCkpO1xuXG4gIC8vIFRoZSBzdGFydCBvZmZzZXQgaXMgbmVnYXRpdmUgbWFyZ2luIHdoaWNoIHNob3VsZCBvdmVybGFwIHRoZSBwYXJlbnQgY29udGFpbmVyLiBTaG91bGQgbmV2ZXIgYmUgYSBwb3NpdGl2ZSB2YWx1ZS5cbiAgLS1zYmItY2FsZW5kYXItc3RhcnQtb2Zmc2V0OiBtaW4oXG4gICAgMHB4LFxuICAgIC0xICogKHZhcigtLXNiYi1jYWxlbmRhci1vdmVyZmxvdykgLyB2YXIoLS1zYmItY2FsZW5kYXItdGFibGUtY29sdW1uLXNwYWNlcykpXG4gICk7XG4gIC0tc2JiLWNhbGVuZGFyLW1hcmdpbjogdmFyKC0tc2JiLWNhbGVuZGFyLXN0YXJ0LW9mZnNldCk7XG5cbiAgOmhvc3QoOnN0YXRlKHdpZGUpKSAmIHtcbiAgICAtLXNiYi1jYWxlbmRhci1taW4td2lkdGg6IGNhbGMoXG4gICAgICAyICogNyAqIHZhcigtLXNiYi1jYWxlbmRhci1jZWxsLXNpemUpICsgdmFyKC0tc2JiLWNhbGVuZGFyLXRhYmxlcy1nYXApXG4gICAgKTtcbiAgICAtLXNiYi1jYWxlbmRhci1tYXJnaW46IGNhbGMoMC41ICogdmFyKC0tc2JiLWNhbGVuZGFyLXN0YXJ0LW9mZnNldCkpO1xuICB9XG5cbiAgOmhvc3QoW29yaWVudGF0aW9uPSdob3Jpem9udGFsJ11bd2Vlay1udW1iZXJzXSkgJiB7XG4gICAgLS1zYmItY2FsZW5kYXItbWluLXdpZHRoOiBjYWxjKDggKiB2YXIoLS1zYmItY2FsZW5kYXItY2VsbC1zaXplKSk7XG4gIH1cblxuICA6aG9zdChbb3JpZW50YXRpb249J2hvcml6b250YWwnXVt3ZWVrLW51bWJlcnNdOnN0YXRlKHdpZGUpKSAmIHtcbiAgICAtLXNiYi1jYWxlbmRhci1taW4td2lkdGg6IGNhbGMoXG4gICAgICAyICogOCAqIHZhcigtLXNiYi1jYWxlbmRhci1jZWxsLXNpemUpICsgdmFyKC0tc2JiLWNhbGVuZGFyLXRhYmxlcy1nYXApXG4gICAgKTtcbiAgfVxuXG4gIC8vIFRoZSBjb250YWluZXIncyBtaW4td2lkdGggaXMgc2V0IGluIHZlcnRpY2FsIG1vZGU7XG4gIC8vIHRoZSB2YWx1ZSBgY2FsYyg3ICogdmFyKC0tc2JiLWNhbGVuZGFyLWNlbGwtc2l6ZSkpYCBpcyBmaW5lXG4gIC8vIGNvbnNpZGVyaW5nIHRoYXQgdGhlIG1heGltdW0gbnVtYmVyIG9mIHdlZWtzIGluIGEgbW9udGggaXMgNiwgcGx1cyAxIGZvciB0aGUgaGVhZGVyIGF0IHRoZSB0YWJsZSdzIGxlZnQgc2lkZS5cbiAgOmhvc3QoW29yaWVudGF0aW9uPSd2ZXJ0aWNhbCddKSAmIHtcbiAgICBtaW4td2lkdGg6IHZhcigtLXNiYi1jYWxlbmRhci1taW4td2lkdGgpO1xuXG4gICAgLy8gVGhlIHZhcmlhYmxlIHRoYXQgZGVmaW5lcyB0aGUgbWFyZ2luIGlzIHNldCB0byAwIHRvIGNvcnJlY3RseSBkaXNwbGF5IHRoZSB0YWJsZSBpZiBjb25zdW1lcnMgc2V0IGEgY3VzdG9tIHdpZHRoLlxuICAgIC0tc2JiLWNhbGVuZGFyLXN0YXJ0LW9mZnNldDogMDtcbiAgfVxuXG4gIC8vIFRoZSBtaW4td2lkdGggaW4gd2lkZSBpcyBjYWxjdWxhdGVkIGFzIChtYXggbnVtYmVyIG9mIHdlZWtzIGluIGEgbW9udGgpICogKGRpc3BsYXllZCBtb250aHMpICsgKGhlYWRlciBjb2x1bW4pID0gNiAqIDIgKyAxID0gMTMuXG4gIDpob3N0KFtvcmllbnRhdGlvbj0ndmVydGljYWwnXTpzdGF0ZSh3aWRlKSkgJiB7XG4gICAgLS1zYmItY2FsZW5kYXItbWluLXdpZHRoOiBjYWxjKFxuICAgICAgMTMgKiB2YXIoLS1zYmItY2FsZW5kYXItY2VsbC1zaXplKSArIHZhcigtLXNiYi1jYWxlbmRhci10YWJsZXMtZ2FwKVxuICAgICk7XG4gIH1cbn1cblxuLnNiYi1jYWxlbmRhcl9fdGFibGUge1xuICB3aWR0aDogMTAwJTtcbiAgYm9yZGVyLWNvbGxhcHNlOiBjb2xsYXBzZTtcbiAgaGVpZ2h0OiBtYXgtY29udGVudDtcblxuICBhbmltYXRpb246IHtcbiAgICBuYW1lOiBzaG93O1xuICAgIGR1cmF0aW9uOiB2YXIoLS1zYmItY2FsZW5kYXItdGFibGUtYW5pbWF0aW9uLWR1cmF0aW9uKTtcbiAgfVxuXG4gICYuc2JiLWNhbGVuZGFyX190YWJsZS1oaWRlIHtcbiAgICAtLXNiYi1jYWxlbmRhci1jZWxsLXRyYW5zaXRpb24tZHVyYXRpb246IDBtcztcblxuICAgIGFuaW1hdGlvbjoge1xuICAgICAgbmFtZTogaGlkZTtcbiAgICAgIGR1cmF0aW9uOiB2YXIoLS1zYmItY2FsZW5kYXItdGFibGUtYW5pbWF0aW9uLWR1cmF0aW9uKTtcbiAgICB9XG4gIH1cblxuICA6aG9zdCg6bm90KDpzdGF0ZSh3aWRlKSkpICYge1xuICAgIC8vIER1ZSB0byBhIFNhZmFyaSBpT1MgcmVuZGVyaW5nIGJ1ZyB3ZSBuZWVkIHRvIGRlZmluZSBtaW4td2lkdGggYXMgd2VsbC5cbiAgICAvLyBPdGhlcndpc2UsIGFmdGVyIG9yaWVudGF0aW9uIGNoYW5nZSwgdGhlcmUgaXMgYSB3cm9uZyB3aWR0aCBpZiBwbGFjZWQgaW4gYW4gc2JiLWRpYWxvZy5cbiAgICBtaW4td2lkdGg6IDEwMCU7XG4gIH1cbn1cblxuLnNiYi1jYWxlbmRhcl9fdGFibGUtaGVhZGVyIHtcbiAgLy8gRGVwZW5kaW5nIHdoZXJlIGluIHRoZSBET00gaXQncyB1c2VkLCBpdCdzIHBvc3NpYmxlIHRoYXQgYnJvd3NlciBkZWZhdWx0cyBzZXQgdGV4dC1hbGlnbiB0byBsZWZ0LlxuICB0ZXh0LWFsaWduOiBjZW50ZXI7XG59XG5cbi5zYmItY2FsZW5kYXJfX3RhYmxlLWJvZHkge1xuICAvLyBEZXBlbmRpbmcgd2hlcmUgaW4gdGhlIERPTSBpdCdzIHVzZWQsIGl0J3MgcG9zc2libGUgdGhhdCBicm93c2VyIGRlZmF1bHRzIHNldCB0ZXh0LWFsaWduIHRvIGxlZnQuXG4gIHRleHQtYWxpZ246IGNlbnRlcjtcbn1cblxuLnNiYi1jYWxlbmRhcl9fdGFibGUtaGVhZGVyLWNlbGwsXG4uc2JiLWNhbGVuZGFyX190YWJsZS1oZWFkZXItY2VsbC12ZXJ0aWNhbCB7XG4gIHdpZHRoOiB2YXIoLS1zYmItY2FsZW5kYXItY2VsbC1zaXplKTtcbiAgY29sb3I6IHZhcigtLXNiYi1jYWxlbmRhci1oZWFkZXItY29sb3IpO1xuICBwYWRkaW5nOiAwO1xuICBmb250LXNpemU6IHZhcigtLXNiYi10ZXh0LWZvbnQtc2l6ZS14cyk7XG4gIGZvbnQtd2VpZ2h0OiBub3JtYWw7XG4gIGxldHRlci1zcGFjaW5nOiB2YXIoLS1zYmItdHlwby1sZXR0ZXItc3BhY2luZy10ZXh0KTtcblxuICA6aG9zdCg6bm90KFttdWx0aXBsZV0pKSAmIHtcbiAgICBoZWlnaHQ6IHZhcigtLXNiYi1jYWxlbmRhci1jZWxsLXNpemUpO1xuICB9XG59XG5cbi5zYmItY2FsZW5kYXJfX3RhYmxlLWhlYWRlci1jZWxsLXZlcnRpY2FsIHtcbiAgcGFkZGluZy1ibG9jay1lbmQ6IHZhcigtLXNiYi1jYWxlbmRhci1jZWxsLXBhZGRpbmctYmxvY2stZW5kKTtcbn1cblxuLnNiYi1jYWxlbmRhcl9fdGFibGUtZGF0YSB7XG4gIHBvc2l0aW9uOiByZWxhdGl2ZTtcbiAgcGFkZGluZzogMDtcbiAgdGV4dC1hbGlnbjogY2VudGVyO1xufVxuXG4uc2JiLWNhbGVuZGFyX19kYXktY2VsbCB7XG4gIHBhZGRpbmctYmxvY2stZW5kOiB2YXIoLS1zYmItY2FsZW5kYXItY2VsbC1wYWRkaW5nLWJsb2NrLWVuZCk7XG59XG5cbnNiYi1jYWxlbmRhci1kYXkge1xuICA6aG9zdCg6bm90KDpzdGF0ZShlbmhhbmNlZCkpKSAmIHtcbiAgICAtLXNiYi1jYWxlbmRhci1jZWxsLWp1c3RpZnktY29udGVudDogY2VudGVyO1xuICAgIC0tc2JiLWNhbGVuZGFyLWRheS1oZWlnaHQ6ICN7c2JiLnB4LXRvLXJlbS1idWlsZCg0NCl9O1xuICAgIC0tc2JiLWNhbGVuZGFyLWRheS1leHRyYS1kaXNwbGF5OiBub25lO1xuICAgIC0tc2JiLWNhbGVuZGFyLWRheS12YWx1ZS1oZWlnaHQ6IHVuc2V0O1xuICAgIC0tc2JiLWNhbGVuZGFyLWRheS1jcm9zc2VkLW91dC10b3A6IDUwJTtcbiAgICAtLXNiYi1jYWxlbmRhci1kYXktY3Jvc3NlZC1vdXQtdHJhbnNsYXRlOiB0cmFuc2xhdGUoLTUwJSwgLTUwJSkgcm90YXRlKC00NWRlZyk7XG4gIH1cbn1cblxuQGtleWZyYW1lcyBzaG93IHtcbiAgZnJvbSB7XG4gICAgb3BhY2l0eTogMDtcbiAgICB0cmFuc2Zvcm06IHRyYW5zbGF0ZVkodmFyKC0tc2JiLWNhbGVuZGFyLXRhYmxlLWFuaW1hdGlvbi1zaGlmdCkpO1xuICB9XG5cbiAgdG8ge1xuICAgIG9wYWNpdHk6IDE7XG4gICAgdHJhbnNmb3JtOiB0cmFuc2xhdGVZKDAlKTtcbiAgfVxufVxuXG5Aa2V5ZnJhbWVzIGhpZGUge1xuICBmcm9tIHtcbiAgICBvcGFjaXR5OiAxO1xuICAgIHRyYW5zZm9ybTogdHJhbnNsYXRlWSgwJSk7XG4gIH1cblxuICB0byB7XG4gICAgb3BhY2l0eTogMDtcbiAgICB0cmFuc2Zvcm06IHRyYW5zbGF0ZVkodmFyKC0tc2JiLWNhbGVuZGFyLXRhYmxlLWFuaW1hdGlvbi1zaGlmdCkpO1xuICB9XG59XG4iLCJpbXBvcnQgeyBlYWNoV2Vla09mSW50ZXJ2YWwsIGVuZE9mTW9udGgsIGdldFdlZWssIHN0YXJ0T2ZNb250aCB9IGZyb20gJ2RhdGUtZm5zJztcbmltcG9ydCB7XG4gIHR5cGUgQ1NTUmVzdWx0R3JvdXAsXG4gIGh0bWwsXG4gIGlzU2VydmVyLFxuICBub3RoaW5nLFxuICB0eXBlIFByb3BlcnR5VmFsdWVzLFxuICB0eXBlIFRlbXBsYXRlUmVzdWx0LFxufSBmcm9tICdsaXQnO1xuaW1wb3J0IHsgcHJvcGVydHksIHN0YXRlIH0gZnJvbSAnbGl0L2RlY29yYXRvcnMuanMnO1xuXG5pbXBvcnQgeyBTYmJTZWNvbmRhcnlCdXR0b25FbGVtZW50IH0gZnJvbSAnLi4vLi4vYnV0dG9uLnB1cmUudHMnO1xuaW1wb3J0IHsgaXNBcnJvd0tleU9yUGFnZUtleXNQcmVzc2VkIH0gZnJvbSAnLi4vLi4vY29yZS9hMTF5LnRzJztcbmltcG9ydCB7IFNiYkVsZW1lbnQgfSBmcm9tICcuLi8uLi9jb3JlL2Jhc2UtZWxlbWVudHMudHMnO1xuaW1wb3J0IHsgcmVhZENvbmZpZyB9IGZyb20gJy4uLy4uL2NvcmUvY29uZmlnLnRzJztcbmltcG9ydCB7XG4gIFNiYkxhbmd1YWdlQ29udHJvbGxlcixcbiAgU2JiTWVkaWFNYXRjaGVyQ29udHJvbGxlcixcbiAgU2JiTWVkaWFRdWVyeUJyZWFrcG9pbnRMYXJnZUFuZEFib3ZlLFxufSBmcm9tICcuLi8uLi9jb3JlL2NvbnRyb2xsZXJzLnRzJztcbmltcG9ydCB0eXBlIHsgRGF0ZUFkYXB0ZXIgfSBmcm9tICcuLi8uLi9jb3JlL2RhdGV0aW1lLnRzJztcbmltcG9ydCB7XG4gIERBWVNfUEVSX1JPVyxcbiAgZGVmYXVsdERhdGVBZGFwdGVyLFxuICBNT05USFNfUEVSX1BBR0UsXG4gIE1PTlRIU19QRVJfUk9XLFxuICBZRUFSU19QRVJfUEFHRSxcbiAgWUVBUlNfUEVSX1JPVyxcbn0gZnJvbSAnLi4vLi4vY29yZS9kYXRldGltZS50cyc7XG5pbXBvcnQgeyBmb3JjZVR5cGUsIGhhbmRsZURpc3RpbmN0Q2hhbmdlLCBwbGFpbkRhdGUgfSBmcm9tICcuLi8uLi9jb3JlL2RlY29yYXRvcnMudHMnO1xuaW1wb3J0IHtcbiAgaTE4bkNhbGVuZGFyRGF0ZVNlbGVjdGlvbixcbiAgaTE4bkNhbGVuZGFyV2Vla051bWJlcixcbiAgaTE4bk5leHRNb250aCxcbiAgaTE4bk5leHRZZWFyLFxuICBpMThuTmV4dFllYXJSYW5nZSxcbiAgaTE4blByZXZpb3VzTW9udGgsXG4gIGkxOG5QcmV2aW91c1llYXIsXG4gIGkxOG5QcmV2aW91c1llYXJSYW5nZSxcbiAgaTE4blllYXJNb250aFNlbGVjdGlvbixcbn0gZnJvbSAnLi4vLi4vY29yZS9pMThuLnRzJztcbmltcG9ydCB0eXBlIHsgU2JiT3JpZW50YXRpb24gfSBmcm9tICcuLi8uLi9jb3JlL2ludGVyZmFjZXMudHMnO1xuaW1wb3J0IHsgYm94U2l6aW5nU3R5bGVzIH0gZnJvbSAnLi4vLi4vY29yZS9zdHlsZXMudHMnO1xuaW1wb3J0IHsgU2JiQ2FsZW5kYXJEYXlFbGVtZW50IH0gZnJvbSAnLi4vY2FsZW5kYXItZGF5L2NhbGVuZGFyLWRheS5jb21wb25lbnQudHMnO1xuaW1wb3J0IHsgU2JiQ2FsZW5kYXJNb250aEVsZW1lbnQgfSBmcm9tICcuLi9jYWxlbmRhci1tb250aC9jYWxlbmRhci1tb250aC5jb21wb25lbnQudHMnO1xuaW1wb3J0IHsgU2JiQ2FsZW5kYXJXZWVrZGF5RWxlbWVudCB9IGZyb20gJy4uL2NhbGVuZGFyLXdlZWtkYXkvY2FsZW5kYXItd2Vla2RheS5jb21wb25lbnQudHMnO1xuaW1wb3J0IHsgU2JiQ2FsZW5kYXJXZWVrbnVtYmVyRWxlbWVudCB9IGZyb20gJy4uL2NhbGVuZGFyLXdlZWtudW1iZXIvY2FsZW5kYXItd2Vla251bWJlci5jb21wb25lbnQudHMnO1xuaW1wb3J0IHsgU2JiQ2FsZW5kYXJZZWFyRWxlbWVudCB9IGZyb20gJy4uL2NhbGVuZGFyLXllYXIvY2FsZW5kYXIteWVhci5jb21wb25lbnQudHMnO1xuaW1wb3J0IHR5cGUgeyBTYmJDYWxlbmRhckNlbGxCYXNlRWxlbWVudCB9IGZyb20gJy4uL2NvbW1vbi50cyc7XG5cbmltcG9ydCBzdHlsZSBmcm9tICcuL2NhbGVuZGFyLnNjc3M/bGl0JmlubGluZSc7XG5cbmltcG9ydCAnLi4vLi4vaWNvbi50cyc7XG5pbXBvcnQgJy4uLy4uL3NjcmVlbi1yZWFkZXItb25seS50cyc7XG5cbi8vIFRPRE8oYnJlYWtpbmctY2hhbmdlKTogUmVtb3ZlIGNhbGwgdG8gZGVmaW5lLlxuU2JiU2Vjb25kYXJ5QnV0dG9uRWxlbWVudC5kZWZpbmUoKTtcblxuU2JiQ2FsZW5kYXJEYXlFbGVtZW50LmRlZmluZSgpO1xuU2JiQ2FsZW5kYXJNb250aEVsZW1lbnQuZGVmaW5lKCk7XG5TYmJDYWxlbmRhclllYXJFbGVtZW50LmRlZmluZSgpO1xuU2JiQ2FsZW5kYXJXZWVrZGF5RWxlbWVudC5kZWZpbmUoKTtcblNiYkNhbGVuZGFyV2Vla251bWJlckVsZW1lbnQuZGVmaW5lKCk7XG5cbmV4cG9ydCBjbGFzcyBTYmJNb250aENoYW5nZUV2ZW50IGV4dGVuZHMgRXZlbnQge1xuICBwcml2YXRlIHJlYWRvbmx5IF9yYW5nZTogcmVhZG9ubHkgRGF5W107XG5cbiAgcHVibGljIGdldCByYW5nZSgpOiByZWFkb25seSBEYXlbXSB7XG4gICAgcmV0dXJuIHRoaXMuX3JhbmdlO1xuICB9XG5cbiAgcHVibGljIGNvbnN0cnVjdG9yKHJhbmdlOiByZWFkb25seSBEYXlbXSkge1xuICAgIHN1cGVyKCdtb250aGNoYW5nZScsIHsgYnViYmxlczogdHJ1ZSwgY29tcG9zZWQ6IHRydWUgfSk7XG4gICAgdGhpcy5fcmFuZ2UgPSBPYmplY3QuZnJlZXplKHJhbmdlIHx8IFtdKTtcbiAgfVxufVxuXG4vKipcbiAqIFBhcmFtZXRlcnMgbmVlZGVkIGluIHllYXIgYW5kIG1vbnRoIHZpZXdzIHRvIGNvcnJlY3RseSBjYWxjdWxhdGUgdGhlIG5leHQgZWxlbWVudCBpbiBrZXlib2FyZCBuYXZpZ2F0aW9uLlxuICpcbiAqIFRoZSBjZWxsJ3MgaW5kZXggYW5kIHRoZSBlbGVtZW50J3MgaW5kZXggaW4gaXRzIG1vbnRoIC8geWVhciBiYXRjaCBtdXN0IGJlIGRpc3Rpbmd1aXNoZWQ6XG4gKiB0aGUgZmlyc3QgaXMgdGhlIGluZGV4IG9mIHRoZSBlbGVtZW50IGluIHRoZSBhcnJheSBvZiBhbGwgdGhlIHJlbmRlcmVkIGNlbGxzLCB3aGlsZSB0aGUgc2Vjb25kIGlzIHRoZSBpbmRleCBvZiB0aGUgZWxlbWVudCByZWxhdGl2ZSB0byBpdHMgdGFibGUuXG4gKiBJbiBub24td2lkZSBtb2RlLCB0aGUgd3RvIGFyZSB0aGUgc2FtZSwgd2hpbGUgaW4gd2lkZSBtb2RlIHRoZSBjZWxsJ3MgaW5kZXggY2FuIGdvIGZyb20gMCB0byA0NyBmb3IgeWVhcnMgYW5kIGZyb20gMCB0byAyMyBmb3IgbW9udGhzLFxuICogd2hpbGUgdGhlIGVsZW1lbnQgaW5kZXggZ29lcyBmcm9tIDAgdG8sIHJlc3BlY3RpdmVseSwgMjMgYW5kIDExLlxuICovXG5pbnRlcmZhY2UgQ2FsZW5kYXJLZXlib2FyZE5hdmlnYXRpb25Nb250aFllYXJWaWV3c1BhcmFtZXRlcnMge1xuICAvKiogVGhlIGVsZW1lbnQgaW5kZXggd2l0aGluIGl0cyB5ZWFyIG9yIG1vbnRoIHJhbmdlLiAqL1xuICBlbGVtZW50SW5kZXhGb3JXaWRlTW9kZTogbnVtYmVyO1xuICAvKiogSW4gd2lkZSBtb2RlLCB0aGUgaW5kZXggb2YgdGhlIGZpcnN0IGVsZW1lbnQgaW4gdGhlIHNlY29uZCBwYW5lbCwgb3IsIGFsdGVybmF0aXZlbHksIHRoZSBudW1iZXIgb2YgZWxlbWVudHMgaW4gdGhlIGZpcnN0IHBhbmVsLiAqL1xuICBvZmZzZXRGb3JXaWRlTW9kZTogbnVtYmVyO1xuICAvKiogVGhlIGluZGV4IG9mIHRoZSBsYXN0IGVsZW1lbnQgd2l0aGluIHRoZSBlbGVtZW50J3MgbW9udGggKG9yIHllYXIgcmFuZ2UpLiAqL1xuICBsYXN0RWxlbWVudEluZGV4Rm9yV2lkZU1vZGU6IG51bWJlcjtcbiAgLyoqIFRoZSBudW1iZXIgb2YgY2VsbHMgZGlzcGxheWVkIGluIGEgc2luZ2xlIHJvdywgZGVwZW5kaW5nIG9uIHRoZSByZW5kZXJlZCB2aWV3LiAqL1xuICB2ZXJ0aWNhbE9mZnNldDogbnVtYmVyO1xufVxuXG4vKipcbiAqIFBhcmFtZXRlcnMgbmVlZGVkIGluIGRheSB2aWV3IHRvIGNvcnJlY3RseSBjYWxjdWxhdGUgdGhlIG5leHQgZWxlbWVudCBpbiBrZXlib2FyZCBuYXZpZ2F0aW9uLlxuICpcbiAqIEluIG9yaWVudGF0aW9uPSd2ZXJ0aWNhbCcsIGl0J3Mgbm90IHBvc3NpYmxlIHRvIHJlbHkgb24gYW55IGFycmF5L2luZGV4IHRvIGNhbGN1bGF0ZSB0aGUgZWxlbWVudCB0byBuYXZpZ2F0ZSB0byxcbiAqIHNvIGNhbGN1bGF0aW9ucyBvbiBkYXRlcyBtdXN0IGJlIGRvbmUsIHdoaWNoIHNob3VsZCBjb25zaWRlciB2aWV3IGJvdW5kYXJpZXMsIG9mZnNldHMgYW5kIG1vbnRoJ3MgbGVuZ3RoLlxuICovXG5pbnRlcmZhY2UgQ2FsZW5kYXJLZXlib2FyZE5hdmlnYXRpb25EYXlWaWV3UGFyYW1ldGVycyB7XG4gIC8qKiBUaGUgZmlyc3QgZGF5IHJlbmRlcmVkLiAqL1xuICBmaXJzdERheUluVmlldzogc3RyaW5nIHwgbnVsbDtcbiAgLyoqIFRoZSBsYXN0IGRheSByZW5kZXJlZC4gSXQgZGVwZW5kcyBvbiB0aGUgJ3dpZGUnIHZhbHVlLiAqL1xuICBsYXN0RGF5SW5WaWV3OiBzdHJpbmcgfCBudWxsO1xuICAvKiogVGhlIG9mZnNldCBmcm9tIHRoZSBmaXJzdCBkYXkgb2YgdGhlIHdlZWsgKE1vbmRheSkgb2YgdGhlIGZpcnN0IHJlbmRlcmVkIG1vbnRoLiAqL1xuICBmaXJzdE1vbnRoT2Zmc2V0OiBudW1iZXI7XG4gIC8qKiBUaGUgbnVtYmVyIG9mIGRheXMgaW4gdGhlIGZpcnN0IHJlbmRlcmVkIG1vbnRoLiAqL1xuICBmaXJzdE1vbnRoTGVuZ3RoOiBudW1iZXI7XG4gIC8qKiBUaGUgb2Zmc2V0IGZyb20gdGhlIGZpcnN0IGRheSBvZiB0aGUgd2VlayAoTW9uZGF5KSBvZiB0aGUgc2Vjb25kIHJlbmRlcmVkIG1vbnRoLiBJZiB3aWRlIGlzIGZhbHNlLCBpdCdzIGVxdWFsIHRvIHplcm8uICovXG4gIHNlY29uZE1vbnRoT2Zmc2V0OiBudW1iZXI7XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgRGF5PFQgPSBEYXRlPiB7XG4gIC8qKiBEYXRlIGFzIElTTyBzdHJpbmcuICovXG4gIHZhbHVlOiBzdHJpbmc7XG4gIGRheVZhbHVlOiBzdHJpbmc7XG4gIG1vbnRoVmFsdWU6IHN0cmluZztcbiAgeWVhclZhbHVlOiBzdHJpbmc7XG4gIGRhdGVWYWx1ZTogVDtcbiAgd2Vla1ZhbHVlOiBudW1iZXI7XG4gIHdlZWtEYXlWYWx1ZTogbnVtYmVyO1xufVxuXG4vKiogQGRlcHJlY2F0ZWQgdXNlIE1vbnRoQ2VsbCAqL1xuZXhwb3J0IGludGVyZmFjZSBNb250aCB7XG4gIHZhbHVlOiBzdHJpbmc7XG4gIGxvbmdWYWx1ZTogc3RyaW5nO1xuICBtb250aFZhbHVlOiBudW1iZXI7XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgTW9udGhDZWxsIHtcbiAgdmFsdWU6IHN0cmluZztcbiAgbW9udGhWYWx1ZTogbnVtYmVyO1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIFdlZWtkYXkge1xuICBsb25nOiBzdHJpbmc7XG4gIG5hcnJvdzogc3RyaW5nO1xufVxuXG5leHBvcnQgdHlwZSBDYWxlbmRhclZpZXcgPSAnZGF5JyB8ICdtb250aCcgfCAneWVhcic7XG5cbi8qKlxuICogSXQgZGlzcGxheXMgYSBjYWxlbmRhciB3aGljaCBhbGxvd3MgY2hvb3NpbmcgYSBkYXRlLlxuICpcbiAqIEBzbG90IC0gVXNlIHRoZSB1bm5hbWVkIHNsb3QgdG8gYWRkIGN1c3RvbWl6ZWQgYHNiYi1jYWxlbmRhci1kYXlgIGVsZW1lbnRzLlxuICovXG5leHBvcnQgY2xhc3MgU2JiQ2FsZW5kYXJFbGVtZW50PFQgPSBEYXRlPiBleHRlbmRzIFNiYkVsZW1lbnQge1xuICBwdWJsaWMgc3RhdGljIG92ZXJyaWRlIHJlYWRvbmx5IGVsZW1lbnROYW1lOiBzdHJpbmcgPSAnc2JiLWNhbGVuZGFyJztcbiAgcHVibGljIHN0YXRpYyBvdmVycmlkZSBzdHlsZXM6IENTU1Jlc3VsdEdyb3VwID0gW2JveFNpemluZ1N0eWxlcywgc3R5bGVdO1xuICBwdWJsaWMgc3RhdGljIHJlYWRvbmx5IGV2ZW50cyA9IHtcbiAgICBkYXRlc2VsZWN0ZWQ6ICdkYXRlc2VsZWN0ZWQnLFxuICAgIG1vbnRoY2hhbmdlOiAnbW9udGhjaGFuZ2UnLFxuICB9IGFzIGNvbnN0O1xuXG4gIC8qKiBJZiBzZXQgdG8gdHJ1ZSwgdHdvIG1vbnRocyBhcmUgZGlzcGxheWVkICovXG4gIEBmb3JjZVR5cGUoKVxuICBAcHJvcGVydHkoeyB0eXBlOiBCb29sZWFuIH0pXG4gIHB1YmxpYyBhY2Nlc3NvciB3aWRlOiBib29sZWFuID0gZmFsc2U7XG5cbiAgLyoqIFRoZSBpbml0aWFsIHZpZXcgb2YgdGhlIGNhbGVuZGFyIHdoaWNoIHNob3VsZCBiZSBkaXNwbGF5ZWQgb24gb3BlbmluZy4gKi9cbiAgQHByb3BlcnR5KCkgcHVibGljIGFjY2Vzc29yIHZpZXc6IENhbGVuZGFyVmlldyA9ICdkYXknO1xuXG4gIC8qKlxuICAgKiBUaGUgbWluaW11bSB2YWxpZCBkYXRlLiBBY2NlcHRzIGEgZGF0ZSBvYmplY3Qgb3IgbnVsbC5cbiAgICogQWNjZXB0cyBhbiBJU084NjAxIGZvcm1hdHRlZCBzdHJpbmcgKGUuZy4gMjAyNC0xMi0yNCkgYXMgYXR0cmlidXRlLlxuICAgKi9cbiAgQHBsYWluRGF0ZSgpXG4gIEBwcm9wZXJ0eSgpXG4gIHB1YmxpYyBhY2Nlc3NvciBtaW46IFQgfCBudWxsID0gbnVsbDtcblxuICAvKipcbiAgICogVGhlIG1heGltdW0gdmFsaWQgZGF0ZS4gQWNjZXB0cyBhIGRhdGUgb2JqZWN0IG9yIG51bGwuXG4gICAqIEFjY2VwdHMgYW4gSVNPODYwMSBmb3JtYXR0ZWQgc3RyaW5nIChlLmcuIDIwMjQtMTItMjQpIGFzIGF0dHJpYnV0ZS5cbiAgICovXG4gIEBwbGFpbkRhdGUoKVxuICBAcHJvcGVydHkoKVxuICBwdWJsaWMgYWNjZXNzb3IgbWF4OiBUIHwgbnVsbCA9IG51bGw7XG5cbiAgLyoqIFdoZXRoZXIgdGhlIGNhbGVuZGFyIGFsbG93cyBmb3IgbXVsdGlwbGUgZGF0ZSBzZWxlY3Rpb24uICovXG4gIEBmb3JjZVR5cGUoKVxuICBAaGFuZGxlRGlzdGluY3RDaGFuZ2UoKGU6IFNiYkNhbGVuZGFyRWxlbWVudDxUPiwgbmV3VmFsdWU6IGJvb2xlYW4pID0+XG4gICAgZS5fb25NdWx0aXBsZUNoYW5nZWQobmV3VmFsdWUpLFxuICApXG4gIEBwcm9wZXJ0eSh7IHR5cGU6IEJvb2xlYW4gfSlcbiAgcHVibGljIGFjY2Vzc29yIG11bHRpcGxlOiBib29sZWFuID0gZmFsc2U7XG5cbiAgLyoqXG4gICAqIFRoZSBzZWxlY3RlZCBkYXRlOiBhY2NlcHRzIGEgZGF0ZSBvYmplY3QsIG9yLCBpZiBgbXVsdGlwbGVgLCBhbiBhcnJheSBvZiBkYXRlcy5cbiAgICovXG4gIEBwcm9wZXJ0eSgpXG4gIHB1YmxpYyBzZXQgc2VsZWN0ZWQodmFsdWU6IFQgfCBUW10gfCBudWxsKSB7XG4gICAgaWYgKEFycmF5LmlzQXJyYXkodmFsdWUpKSB7XG4gICAgICB0aGlzLl9zZWxlY3RlZCA9IHZhbHVlXG4gICAgICAgIC5tYXAoKGRhdGVMaWtlOiBUKSA9PlxuICAgICAgICAgIHRoaXMuX2RhdGVBZGFwdGVyLmdldFZhbGlkRGF0ZU9yTnVsbCh0aGlzLl9kYXRlQWRhcHRlci5kZXNlcmlhbGl6ZShkYXRlTGlrZSkpLFxuICAgICAgICApXG4gICAgICAgIC5maWx0ZXIoKGRhdGU6IFQgfCBudWxsKTogZGF0ZSBpcyBUID0+IGRhdGUgIT09IG51bGwpXG4gICAgICAgIC5maWx0ZXIoXG4gICAgICAgICAgKGRhdGU6IFQpID0+XG4gICAgICAgICAgICAhdGhpcy5faXNEYXlJblJhbmdlKHRoaXMuX2RhdGVBZGFwdGVyLnRvSXNvODYwMShkYXRlKSkgfHwgdGhpcy5fZGF0ZUZpbHRlcihkYXRlKSxcbiAgICAgICAgKTtcbiAgICB9IGVsc2Uge1xuICAgICAgY29uc3Qgc2VsZWN0ZWREYXRlID0gdGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0VmFsaWREYXRlT3JOdWxsKFxuICAgICAgICB0aGlzLl9kYXRlQWRhcHRlci5kZXNlcmlhbGl6ZSh2YWx1ZSksXG4gICAgICApO1xuICAgICAgaWYgKFxuICAgICAgICAhIXNlbGVjdGVkRGF0ZSAmJlxuICAgICAgICAoIXRoaXMuX2lzRGF5SW5SYW5nZSh0aGlzLl9kYXRlQWRhcHRlci50b0lzbzg2MDEoc2VsZWN0ZWREYXRlKSkgfHxcbiAgICAgICAgICB0aGlzLl9kYXRlRmlsdGVyKHNlbGVjdGVkRGF0ZSkpXG4gICAgICApIHtcbiAgICAgICAgdGhpcy5fc2VsZWN0ZWQgPSBzZWxlY3RlZERhdGU7XG4gICAgICB9IGVsc2Uge1xuICAgICAgICB0aGlzLl9zZWxlY3RlZCA9IG51bGw7XG4gICAgICB9XG4gICAgfVxuICB9XG4gIHB1YmxpYyBnZXQgc2VsZWN0ZWQoKTogVCB8IFRbXSB8IG51bGwge1xuICAgIHJldHVybiB0aGlzLl9zZWxlY3RlZDtcbiAgfVxuICBAc3RhdGUoKSBwcml2YXRlIGFjY2Vzc29yIF9zZWxlY3RlZDogVCB8IFRbXSB8IG51bGwgPSBudWxsO1xuXG4gIC8qKiBBIGZ1bmN0aW9uIHVzZWQgdG8gZmlsdGVyIG91dCBkYXRlcy4gKi9cbiAgQHByb3BlcnR5KHsgYXR0cmlidXRlOiAnZGF0ZS1maWx0ZXInIH0pXG4gIHB1YmxpYyBhY2Nlc3NvciBkYXRlRmlsdGVyOiAoKGRhdGU6IFQgfCBudWxsKSA9PiBib29sZWFuKSB8IG51bGwgPSBudWxsO1xuXG4gIC8qKiBUaGUgb3JpZW50YXRpb24gb2YgZGF5cyBpbiB0aGUgY2FsZW5kYXIuICovXG4gIEBwcm9wZXJ0eSh7IHJlZmxlY3Q6IHRydWUgfSkgcHVibGljIGFjY2Vzc29yIG9yaWVudGF0aW9uOiBTYmJPcmllbnRhdGlvbiA9ICdob3Jpem9udGFsJztcblxuICAvKiogV2hldGhlciBpdCBoYXMgdG8gZGlzcGxheSB0aGUgd2VlayBudW1iZXJzIGluIGFkZGl0aW9uIHRvIHdlZWsgZGF5cy4gKi9cbiAgQGZvcmNlVHlwZSgpXG4gIEBwcm9wZXJ0eSh7IGF0dHJpYnV0ZTogJ3dlZWstbnVtYmVycycsIHR5cGU6IEJvb2xlYW4gfSlcbiAgcHVibGljIGFjY2Vzc29yIHdlZWtOdW1iZXJzOiBib29sZWFuID0gZmFsc2U7XG5cbiAgcHJpdmF0ZSBfZGF0ZUFkYXB0ZXI6IERhdGVBZGFwdGVyPFQ+ID0gcmVhZENvbmZpZygpLmRhdGV0aW1lPy5kYXRlQWRhcHRlciA/PyBkZWZhdWx0RGF0ZUFkYXB0ZXI7XG5cbiAgLyoqIFRoZSBjdXJyZW50bHkgYWN0aXZlIGRhdGUuICovXG4gIEBzdGF0ZSgpIHByaXZhdGUgYWNjZXNzb3IgX2FjdGl2ZURhdGU6IFQgPSB0aGlzLl9kYXRlQWRhcHRlci50b2RheSgpO1xuXG4gIC8qKiBUaGUgY3VycmVudCB3aWRlIHByb3BlcnR5IGNvbnNpZGVyaW5nIHByb3BlcnR5IHZhbHVlIGFuZCBicmVha3BvaW50cy4gRnJvbSB6ZXJvIHRvIHNtYWxsIGB3aWRlYCBoYXMgYWx3YXlzIHRvIGJlIGZhbHNlLiAqL1xuICBAc3RhdGUoKVxuICBwcml2YXRlIHNldCBfd2lkZSh3aWRlOiBib29sZWFuKSB7XG4gICAgdGhpcy50b2dnbGVTdGF0ZSgnd2lkZScsIHdpZGUpO1xuICAgIHRoaXMuX3dpZGVJbnRlcm5hbCA9IHdpZGU7XG4gIH1cbiAgcHJpdmF0ZSBnZXQgX3dpZGUoKTogYm9vbGVhbiB7XG4gICAgcmV0dXJuIHRoaXMuX3dpZGVJbnRlcm5hbDtcbiAgfVxuICAvLyBXZSBrZWVwIHRoZSBzdGF0ZSBpbiBhIGZpZWxkIGJlY2F1c2Ugb2YgdGhlIFdlYktpdCBidWcgaHR0cHM6Ly9idWdzLndlYmtpdC5vcmcvc2hvd19idWcuY2dpP2lkPTMwMzQ2Ny5cbiAgLy8gVE9ETzogcmUtY2hlY2sgd2hldGhlciBmaWVsZCBpcyBuZWVkZWRcbiAgcHJpdmF0ZSBfd2lkZUludGVybmFsOiBib29sZWFuID0gZmFsc2U7XG5cbiAgQHN0YXRlKCkgcHJpdmF0ZSBhY2Nlc3NvciBfY2FsZW5kYXJWaWV3OiBDYWxlbmRhclZpZXcgPSAnZGF5JztcblxuICBwcml2YXRlIF9uZXh0Q2FsZW5kYXJWaWV3OiBDYWxlbmRhclZpZXcgPSAnZGF5JztcblxuICAvKiogSW5mb3JtYXRpb24gYWJvdXQgdGhlIHJlbmRlcmVkIGRheSB2aWV3OyB1c2VkIGluIGtleWJvYXJkIG5hdmlnYXRpb24uICovXG4gIHByaXZhdGUgX2tleWJvYXJkTmF2aWdhdGlvbkRheVZpZXdQYXJhbWV0ZXJzOiBDYWxlbmRhcktleWJvYXJkTmF2aWdhdGlvbkRheVZpZXdQYXJhbWV0ZXJzID0ge1xuICAgIGZpcnN0RGF5SW5WaWV3OiBudWxsLFxuICAgIGxhc3REYXlJblZpZXc6IG51bGwsXG4gICAgZmlyc3RNb250aE9mZnNldDogMCxcbiAgICBmaXJzdE1vbnRoTGVuZ3RoOiAwLFxuICAgIHNlY29uZE1vbnRoT2Zmc2V0OiAwLFxuICB9O1xuXG4gIC8qKiBBIGxpc3Qgb2YgZGF5cywgaW4gdHdvIGZvcm1hdHMgKGxvbmcgYW5kIHNpbmdsZSBjaGFyKS4gKi9cbiAgcHJpdmF0ZSBfd2Vla2RheXMhOiBXZWVrZGF5W107XG5cbiAgLyoqIEdyaWQgb2YgY2FsZW5kYXIgY2VsbHMgcmVwcmVzZW50aW5nIHRoZSBkYXRlcyBvZiB0aGUgbW9udGguICovXG4gIHByaXZhdGUgX3dlZWtzOiBEYXk8VD5bXVtdID0gW107XG5cbiAgLyoqIEdyaWQgb2YgY2FsZW5kYXIgY2VsbHMgcmVwcmVzZW50aW5nIG1vbnRocy4gKi9cbiAgcHJpdmF0ZSBfbW9udGhzITogTW9udGhDZWxsW11bXTtcblxuICAvKiogR3JpZCBvZiBjYWxlbmRhciBjZWxscyByZXByZXNlbnRpbmcgeWVhcnMuICovXG4gIHByaXZhdGUgX3llYXJzITogbnVtYmVyW11bXTtcblxuICAvKiogR3JpZCBvZiBjYWxlbmRhciBjZWxscyByZXByZXNlbnRpbmcgeWVhcnMgZm9yIHRoZSB3aWRlIHZpZXcuICovXG4gIHByaXZhdGUgX25leHRNb250aFllYXJzITogbnVtYmVyW11bXTtcblxuICAvKiogR3JpZCBvZiBjYWxlbmRhciBjZWxscyByZXByZXNlbnRpbmcgdGhlIGRhdGVzIG9mIHRoZSBuZXh0IG1vbnRoLiAqL1xuICBwcml2YXRlIF9uZXh0TW9udGhXZWVrcyE6IERheTxUPltdW107XG5cbiAgLyoqIEFuIGFycmF5IGNvbnRhaW5pbmcgYWxsIHRoZSBtb250aCBuYW1lcyBpbiB0aGUgY3VycmVudCBsYW5ndWFnZS4gKi9cbiAgcHJpdmF0ZSBfbW9udGhOYW1lczogc3RyaW5nW10gPSB0aGlzLl9kYXRlQWRhcHRlci5nZXRNb250aE5hbWVzKCdsb25nJyk7XG5cbiAgLyoqIEFuIGFycmF5IGNvbnRhaW5pbmcgdGhlIHdlZWtzJyBudW1iZXJzIGZvciB0aGUgY3VycmVudCBtb250aC4gKi9cbiAgcHJpdmF0ZSBfd2Vla051bWJlcnMhOiBudW1iZXJbXTtcblxuICAvKiogQW4gYXJyYXkgY29udGFpbmluZyB0aGUgd2Vla3MnIG51bWJlcnMgZm9yIHRoZSBuZXh0IG1vbnRoIGluIHdpZGUgbW9kZS4gKi9cbiAgcHJpdmF0ZSBfbmV4dE1vbnRoV2Vla051bWJlcnMhOiBudW1iZXJbXTtcblxuICBwcml2YXRlIF9lbmhhbmNlZFZhcmlhbnQ6IGJvb2xlYW4gPSBmYWxzZTtcblxuICAvKiogQSBsaXN0IG9mIGNhbGVuZGFyJ3MgY2VsbHMgY29ycmVzcG9uZGluZyB0byBkYXlzLCBtb250aHMgb3IgeWVhcnMgZGVwZW5kaW5nIG9uIHRoZSB2aWV3LiAqL1xuICBwcml2YXRlIGdldCBfY2VsbHMoKTogU2JiQ2FsZW5kYXJDZWxsQmFzZUVsZW1lbnRbXSB7XG4gICAgcmV0dXJuIEFycmF5LmZyb208U2JiQ2FsZW5kYXJDZWxsQmFzZUVsZW1lbnQ+KFxuICAgICAgKHRoaXMuX2NhbGVuZGFyVmlldyA9PT0gJ2RheSdcbiAgICAgICAgPyAoQXJyYXkuZnJvbSh0aGlzLnNoYWRvd1Jvb3QhLnF1ZXJ5U2VsZWN0b3JBbGwoJ3Nsb3QnKSkuZmxhdE1hcCgoZTogSFRNTFNsb3RFbGVtZW50KSA9PlxuICAgICAgICAgICAgZS5hc3NpZ25lZEVsZW1lbnRzKHsgZmxhdHRlbjogdHJ1ZSB9KSxcbiAgICAgICAgICApIGFzIFNiYkNhbGVuZGFyRGF5RWxlbWVudFtdKVxuICAgICAgICA6IHRoaXMuc2hhZG93Um9vdD8ucXVlcnlTZWxlY3RvckFsbChgc2JiLWNhbGVuZGFyLSR7dGhpcy5fY2FsZW5kYXJWaWV3fWApKSA/PyBbXSxcbiAgICApO1xuICB9XG5cbiAgLyoqIFRoZSBjaG9zZW4geWVhciBpbiB0aGUgeWVhciBzZWxlY3Rpb24gdmlldy4gKi9cbiAgcHJpdmF0ZSBfY2hvc2VuWWVhcj86IG51bWJlcjtcblxuICAvKiogVGhlIGNob3NlbiBtb250aCBpbiB0aGUgeWVhciBzZWxlY3Rpb24gdmlldy4gKi9cbiAgcHJpdmF0ZSBfY2hvc2VuTW9udGg/OiBudW1iZXI7XG5cbiAgLyoqIFdoZXRoZXIgdGhlIGZvY3VzIHNob3VsZCBiZSByZXNldCBvbiBmb2N1c0NlbGwuICovXG4gIHByaXZhdGUgX3Jlc2V0Rm9jdXMgPSBmYWxzZTtcblxuICAvKiogV2hldGhlciBhbiBlbGVtZW50IGluc2lkZSB0aGUgY2FsZW5kYXIgaXMgY3VycmVudGx5IGZvY3VzZWQuICovXG4gIHByaXZhdGUgX2NvbnRhaW5pbmdGb2N1cyA9IGZhbHNlO1xuXG4gIEBzdGF0ZSgpXG4gIHByaXZhdGUgYWNjZXNzb3IgX2luaXRpYWxpemVkID0gZmFsc2U7XG5cbiAgcHJpdmF0ZSBfbGFuZ3VhZ2UgPSBuZXcgU2JiTGFuZ3VhZ2VDb250cm9sbGVyKHRoaXMpLndpdGhIYW5kbGVyKCgpID0+IHtcbiAgICB0aGlzLl9tb250aE5hbWVzID0gdGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0TW9udGhOYW1lcygnbG9uZycpO1xuICAgIHRoaXMuX2NyZWF0ZU1vbnRoUm93cygpO1xuICB9KTtcbiAgcHJpdmF0ZSBfbWVkaWFNYXRjaGVyID0gbmV3IFNiYk1lZGlhTWF0Y2hlckNvbnRyb2xsZXIodGhpcywge1xuICAgIFtTYmJNZWRpYVF1ZXJ5QnJlYWtwb2ludExhcmdlQW5kQWJvdmVdOiAoKSA9PiB0aGlzLl9pbml0KCksXG4gIH0pO1xuXG4gIHB1YmxpYyBjb25zdHJ1Y3RvcigpIHtcbiAgICBzdXBlcigpO1xuICAgIHRoaXMuX2NyZWF0ZU1vbnRoUm93cygpO1xuICAgIHRoaXMuX3NldFdlZWtkYXlzKCk7XG5cbiAgICAvLyBXZSBuZWVkIHRvIHRyYWNrIHRoZSBmb2N1cyBhcyB3ZSBzaG91bGQgb25seSB0YWtlIGZvY3VzIGludG8gdGhlIGNhbGVuZGFyLCB3aGVuIHRoZVxuICAgIC8vIGZvY3VzIHdhcyBvbmNlIHNldCBpbnRvIHRoZSBjYWxlbmRhci5cbiAgICAvLyBGb3Igc2hhZG93IERPTSBjb21wYXRpYmlsaXR5IHdlIG5lZWQgdG8gdHJhY2sgdGhpcyBwcm9ncmFtbWF0aWNhbGx5LlxuICAgIHRoaXMuYWRkRXZlbnRMaXN0ZW5lcignZm9jdXNpbicsICgpID0+ICh0aGlzLl9jb250YWluaW5nRm9jdXMgPSB0cnVlKSk7XG4gICAgdGhpcy5hZGRFdmVudExpc3RlbmVyKCdmb2N1c291dCcsICgpID0+ICh0aGlzLl9jb250YWluaW5nRm9jdXMgPSBmYWxzZSkpO1xuICAgIHRoaXMuYWRkRXZlbnRMaXN0ZW5lcignY2xpY2snLCAoZSkgPT4ge1xuICAgICAgY29uc3QgZGF5ID0gKGUudGFyZ2V0IGFzIEhUTUxFbGVtZW50KS5jbG9zZXN0PFNiYkNhbGVuZGFyRGF5RWxlbWVudDxUPj4oJ3NiYi1jYWxlbmRhci1kYXknKTtcbiAgICAgIGlmIChkYXkpIHtcbiAgICAgICAgdGhpcy5fc2VsZWN0RGF0ZShkYXkudmFsdWUhKTtcbiAgICAgIH1cbiAgICB9KTtcbiAgICB0aGlzLmFkZEV2ZW50TGlzdGVuZXIoJ2tleWRvd24nLCAoZSkgPT4ge1xuICAgICAgaWYgKChlLnRhcmdldCBhcyBIVE1MRWxlbWVudCkubG9jYWxOYW1lID09PSAnc2JiLWNhbGVuZGFyLWRheScpIHtcbiAgICAgICAgdGhpcy5faGFuZGxlS2V5Ym9hcmRFdmVudChcbiAgICAgICAgICBlLFxuICAgICAgICAgIHRoaXMuX21hcERhdGVUb0RheSgoZS50YXJnZXQgYXMgU2JiQ2FsZW5kYXJEYXlFbGVtZW50KS52YWx1ZSEgYXMgVCksXG4gICAgICAgICk7XG4gICAgICB9XG4gICAgfSk7XG4gIH1cblxuICBwcml2YXRlIF9kYXRlRmlsdGVyKGRhdGU6IFQpOiBib29sZWFuIHtcbiAgICByZXR1cm4gdGhpcy5kYXRlRmlsdGVyPy4oZGF0ZSkgPz8gdHJ1ZTtcbiAgfVxuXG4gIC8qKiBSZXNldHMgdGhlIGFjdGl2ZSBtb250aCBhY2NvcmRpbmcgdG8gdGhlIG5ldyBzdGF0ZSBvZiB0aGUgY2FsZW5kYXIuICovXG4gIHB1YmxpYyByZXNldFBvc2l0aW9uKCk6IHZvaWQge1xuICAgIHRoaXMuX3Jlc2V0Q2FsZW5kYXJWaWV3KCk7XG4gICAgdGhpcy5faW5pdCgpO1xuICB9XG5cbiAgcHVibGljIG92ZXJyaWRlIGNvbm5lY3RlZENhbGxiYWNrKCk6IHZvaWQge1xuICAgIHN1cGVyLmNvbm5lY3RlZENhbGxiYWNrKCk7XG4gICAgdGhpcy5yZXNldFBvc2l0aW9uKCk7XG4gICAgdGhpcy5zaGFkb3dSb290Py5hZGRFdmVudExpc3RlbmVyKCdzbG90Y2hhbmdlJywgdGhpcy5fb25TbG90Q2hhbmdlLCB7IGNhcHR1cmU6IHRydWUgfSk7XG4gIH1cblxuICBwdWJsaWMgb3ZlcnJpZGUgZGlzY29ubmVjdGVkQ2FsbGJhY2soKTogdm9pZCB7XG4gICAgc3VwZXIuZGlzY29ubmVjdGVkQ2FsbGJhY2soKTtcbiAgICB0aGlzLnNoYWRvd1Jvb3Q/LnJlbW92ZUV2ZW50TGlzdGVuZXIoJ3Nsb3RjaGFuZ2UnLCB0aGlzLl9vblNsb3RDaGFuZ2UsIHsgY2FwdHVyZTogdHJ1ZSB9KTtcbiAgfVxuXG4gIC8qKiBAaW50ZXJuYWwgKi9cbiAgcHVibGljIG92ZXJyaWRlIGZvY3VzKCk6IHZvaWQge1xuICAgIHRoaXMuX3Jlc2V0Rm9jdXMgPSB0cnVlO1xuICAgIHRoaXMuX2ZvY3VzQ2VsbCgpO1xuICB9XG5cbiAgcHJvdGVjdGVkIG92ZXJyaWRlIHdpbGxVcGRhdGUoY2hhbmdlZFByb3BlcnRpZXM6IFByb3BlcnR5VmFsdWVzPHRoaXM+KTogdm9pZCB7XG4gICAgc3VwZXIud2lsbFVwZGF0ZShjaGFuZ2VkUHJvcGVydGllcyk7XG5cbiAgICBpZiAoIXRoaXMuX2luaXRpYWxpemVkKSB7XG4gICAgICByZXR1cm47XG4gICAgfVxuXG4gICAgaWYgKGNoYW5nZWRQcm9wZXJ0aWVzLmhhcygnd2lkZScpIHx8IGNoYW5nZWRQcm9wZXJ0aWVzLmhhcygnb3JpZW50YXRpb24nKSkge1xuICAgICAgdGhpcy5yZXNldFBvc2l0aW9uKCk7XG4gICAgfVxuXG4gICAgaWYgKGNoYW5nZWRQcm9wZXJ0aWVzLmhhcygndmlldycpKSB7XG4gICAgICB0aGlzLl9zZXRDaG9zZW5ZZWFyKCk7XG4gICAgICB0aGlzLl9jaG9zZW5Nb250aCA9IHVuZGVmaW5lZDtcbiAgICAgIHRoaXMuX25leHRDYWxlbmRhclZpZXcgPSB0aGlzLl9jYWxlbmRhclZpZXcgPSB0aGlzLnZpZXc7XG4gICAgfVxuICB9XG5cbiAgcHJvdGVjdGVkIG92ZXJyaWRlIHVwZGF0ZWQoY2hhbmdlZFByb3BlcnRpZXM6IFByb3BlcnR5VmFsdWVzPHRoaXM+KTogdm9pZCB7XG4gICAgc3VwZXIudXBkYXRlZChjaGFuZ2VkUHJvcGVydGllcyk7XG4gICAgLy8gVGhlIGNhbGVuZGFyIG5lZWRzIHRvIGNhbGN1bGF0ZSB0YWItaW5kZXhlcyBvbiBmaXJzdCByZW5kZXIsXG4gICAgLy8gYW5kIGV2ZXJ5IHRpbWUgYSBkYXRlIGlzIHNlbGVjdGVkIG9yIHRoZSBtb250aCB2aWV3IGNoYW5nZXMuXG4gICAgdGhpcy5fc2V0VGFiSW5kZXgoKTtcblxuICAgIC8vIFdoZW4gY2hhbmdpbmcgdmlldyB0byB5ZWFyL21vbnRoLCB0aGUgdGFiaW5kZXggaXMgY2hhbmdlZCwgYnV0IHRoZSBmb2N1c2VkIGVsZW1lbnQgaXMgZ2V0dGluZyBsb3N0LlxuICAgIC8vIFdlIG5lZWQgdG8gY2FsbCBgX2ZvY3VzQ2VsbCgpYCBtZXRob2QgZXhwbGljaXRseSB0byBjb3JyZWN0bHkgc2V0IHRoZSBmb2N1cy5cbiAgICB0aGlzLl9mb2N1c0NlbGwoKTtcbiAgfVxuXG4gIHByaXZhdGUgX29uU2xvdENoYW5nZSA9ICgpOiB2b2lkID0+IHtcbiAgICB0aGlzLl9lbmhhbmNlZFZhcmlhbnQgPSBBcnJheS5mcm9tKHRoaXMuY2hpbGRyZW4pLnNvbWUoXG4gICAgICAoYykgPT4gYy5sb2NhbE5hbWUgPT09ICdzYmItY2FsZW5kYXItZGF5JyxcbiAgICApO1xuICAgIHRoaXMudG9nZ2xlU3RhdGUoJ2VuaGFuY2VkJywgdGhpcy5fZW5oYW5jZWRWYXJpYW50KTtcbiAgICB0aGlzLl9zZXRUYWJJbmRleCgpO1xuICB9O1xuXG4gIC8qKlxuICAgKiBUaGUgYF9zZWxlY3RlZGAgc3RhdGUgc2hvdWxkIGJlIGFkYXB0ZWQgd2hlbiB0aGUgYG11bHRpcGxlYCBwcm9wZXJ0eSBjaGFuZ2VzOlxuICAgKiAgIC0gaWYgaXQgY2hhbmdlcyB0byB0cnVlLCB0aGUgJ19zZWxlY3RlZCcgaXMgc2V0IHRvIGFuIGFycmF5O1xuICAgKiAgIC0gaWYgaXQgY2hhbmdlcyB0byBmYWxzZSwgdGhlIGZpcnN0IGF2YWlsYWJsZSBvcHRpb24gaXMgc2V0IGFzICd2YWx1ZScgb3RoZXJ3aXNlIGl0J3Mgc2V0IHRvIG51bGwuXG4gICAqL1xuICBwcml2YXRlIF9vbk11bHRpcGxlQ2hhbmdlZChpc011bHRpcGxlOiBib29sZWFuKTogdm9pZCB7XG4gICAgaWYgKGlzTXVsdGlwbGUgJiYgIUFycmF5LmlzQXJyYXkodGhpcy5fc2VsZWN0ZWQpKSB7XG4gICAgICB0aGlzLl9zZWxlY3RlZCA9IHRoaXMuX3NlbGVjdGVkID8gW3RoaXMuX3NlbGVjdGVkIGFzIFRdIDogW107XG4gICAgfVxuICAgIGlmICghaXNNdWx0aXBsZSAmJiBBcnJheS5pc0FycmF5KHRoaXMuX3NlbGVjdGVkKSkge1xuICAgICAgdGhpcy5fc2VsZWN0ZWQgPSAodGhpcy5fc2VsZWN0ZWQgYXMgVFtdKS5sZW5ndGggPyAodGhpcy5fc2VsZWN0ZWQgYXMgVFtdKVswXSA6IG51bGw7XG4gICAgfVxuICB9XG5cbiAgLyoqIEluaXRpYWxpemVzIHRoZSBjb21wb25lbnQuICovXG4gIHByaXZhdGUgX2luaXQoYWN0aXZlRGF0ZT86IFQpOiB2b2lkIHtcbiAgICAvLyBEdWUgdG8gaXRzIGNvbXBsZXhpdHksIHRoZSBjYWxlbmRhciBpcyBvbmx5IGluaXRpYWxpemVkIG9uIGNsaWVudCBzaWRlXG4gICAgaWYgKGlzU2VydmVyKSB7XG4gICAgICByZXR1cm47XG4gICAgfSBlbHNlIGlmICh0aGlzLmh5ZHJhdGlvblJlcXVpcmVkKSB7XG4gICAgICB0aGlzLmh5ZHJhdGlvbkNvbXBsZXRlLnRoZW4oKCkgPT4gdGhpcy5faW5pdCgpKTtcbiAgICAgIHJldHVybjtcbiAgICB9XG5cbiAgICBpZiAoYWN0aXZlRGF0ZSkge1xuICAgICAgdGhpcy5fYXNzaWduQWN0aXZlRGF0ZShhY3RpdmVEYXRlKTtcbiAgICB9XG4gICAgdGhpcy5fd2lkZSA9XG4gICAgICAodGhpcy5fbWVkaWFNYXRjaGVyLm1hdGNoZXMoU2JiTWVkaWFRdWVyeUJyZWFrcG9pbnRMYXJnZUFuZEFib3ZlKSA/PyBmYWxzZSkgJiYgdGhpcy53aWRlO1xuICAgIHRoaXMuX3dlZWtzID0gdGhpcy5fY3JlYXRlV2Vla1Jvd3ModGhpcy5fYWN0aXZlRGF0ZSk7XG4gICAgdGhpcy5feWVhcnMgPSB0aGlzLl9jcmVhdGVZZWFyUm93cygpO1xuICAgIHRoaXMuX3dlZWtOdW1iZXJzID0gdGhpcy5fY3JlYXRlV2Vla051bWJlcnModGhpcy5fYWN0aXZlRGF0ZSk7XG4gICAgdGhpcy5fbmV4dE1vbnRoV2Vla3MgPSBbW11dO1xuICAgIHRoaXMuX25leHRNb250aFllYXJzID0gW1tdXTtcbiAgICBpZiAodGhpcy5fd2lkZSkge1xuICAgICAgY29uc3QgbmV4dE1vbnRoRGF0ZSA9IHRoaXMuX2RhdGVBZGFwdGVyLmFkZENhbGVuZGFyTW9udGhzKHRoaXMuX2FjdGl2ZURhdGUsIDEpO1xuICAgICAgdGhpcy5fbmV4dE1vbnRoV2Vla3MgPSB0aGlzLl9jcmVhdGVXZWVrUm93cyhuZXh0TW9udGhEYXRlLCB0cnVlKTtcbiAgICAgIHRoaXMuX25leHRNb250aFllYXJzID0gdGhpcy5fY3JlYXRlWWVhclJvd3MoWUVBUlNfUEVSX1BBR0UpO1xuICAgICAgdGhpcy5fbmV4dE1vbnRoV2Vla051bWJlcnMgPSB0aGlzLl9jcmVhdGVXZWVrTnVtYmVycyhuZXh0TW9udGhEYXRlKTtcbiAgICB9XG4gICAgdGhpcy5faW5pdGlhbGl6ZWQgPSB0cnVlO1xuICB9XG5cbiAgLyoqIEZvY3VzZXMgb24gYSBkYXkgY2VsbCBwcmlvcml0aXppbmcgdGhlIHNlbGVjdGVkIGRheSwgdGhlIGN1cnJlbnQgZGF5LCBhbmQgbGFzdGx5LCB0aGUgZmlyc3Qgc2VsZWN0YWJsZSBkYXkuICovXG4gIHByaXZhdGUgX2ZvY3VzQ2VsbCgpOiB2b2lkIHtcbiAgICBpZiAodGhpcy5fcmVzZXRGb2N1cykge1xuICAgICAgdGhpcy5fZ2V0Rmlyc3RGb2N1c2FibGUoKT8uZm9jdXMoKTtcbiAgICAgIHRoaXMuX3Jlc2V0Rm9jdXMgPSBmYWxzZTtcbiAgICB9XG4gIH1cblxuICAvKiogQ3JlYXRlcyB0aGUgYXJyYXkgb2Ygd2Vla2RheXMuICovXG4gIHByaXZhdGUgX3NldFdlZWtkYXlzKCk6IHZvaWQge1xuICAgIGNvbnN0IG5hcnJvd1dlZWtkYXlzOiBzdHJpbmdbXSA9IHRoaXMuX2RhdGVBZGFwdGVyLmdldERheU9mV2Vla05hbWVzKCduYXJyb3cnKTtcbiAgICBjb25zdCBsb25nV2Vla2RheXM6IHN0cmluZ1tdID0gdGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0RGF5T2ZXZWVrTmFtZXMoJ2xvbmcnKTtcbiAgICBjb25zdCB3ZWVrZGF5czogV2Vla2RheVtdID0gbG9uZ1dlZWtkYXlzLm1hcCgobG9uZzogc3RyaW5nLCBpOiBudW1iZXIpID0+ICh7XG4gICAgICBsb25nLFxuICAgICAgbmFycm93OiBuYXJyb3dXZWVrZGF5c1tpXSxcbiAgICB9KSk7XG5cbiAgICAvLyBSb3RhdGVzIHRoZSBsYWJlbHMgZm9yIGRheXMgb2YgdGhlIHdlZWsgYmFzZWQgb24gdGhlIGNvbmZpZ3VyZWQgZmlyc3QgZGF5IG9mIHRoZSB3ZWVrLlxuICAgIGNvbnN0IGZpcnN0RGF5T2ZXZWVrOiBudW1iZXIgPSB0aGlzLl9kYXRlQWRhcHRlci5nZXRGaXJzdERheU9mV2VlaygpO1xuICAgIHRoaXMuX3dlZWtkYXlzID0gd2Vla2RheXMuc2xpY2UoZmlyc3REYXlPZldlZWspLmNvbmNhdCh3ZWVrZGF5cy5zbGljZSgwLCBmaXJzdERheU9mV2VlaykpO1xuICB9XG5cbiAgLyoqXG4gICAqIEdpdmVuIGEgZGF0ZSwgaXQgcmV0dXJucyB0aGUgd2VlayBudW1iZXJzIGZvciB0aGUgbW9udGggdGhlIGRhdGUgYmVsb25ncyB0by5cbiAgICogVE9ETzogY2hlY2sgaWYgZGF0ZS1mbnMgY2FuIGJlIHJlcGxhY2VkIHdpdGggY3VzdG9tIGxvZ2ljLlxuICAgKlxuICAgKiBTaW5jZSB0aGUgY2FsY3VsYXRpb24gaXMgbm90IHNpbXBsZSAoc2VlIGh0dHBzOi8vZW4ud2lraXBlZGlhLm9yZy93aWtpL1dlZWsjTnVtYmVyaW5nKSxcbiAgICogdGhlIGRhdGUtZm5zIGxpYnJhcnkgaGFzIGJlZW4gdXNlZCB0aGlzIHdheTpcbiAgICogdGhlIGZpcnN0IGFuZCB0aGUgbGFzdCBkYXkgb2YgdGhlIG1vbnRoIGFyZSBjYWxjdWxhdGVkIGFuZCB0aGVuIHBhc3NlZCB0byB0aGUgYGVhY2hXZWVrT2ZJbnRlcnZhbGAgZnVuY3Rpb24sXG4gICAqIHdoaWNoIHJldHVybnMgYW4gYXJyYXkgY29udGFpbmluZyB0aGUgc3RhcnRpbmcgZGF5IG9mIGV2ZXJ5IElTTyB3ZWVrIG9mIHRoZSBtb250aCxcbiAgICogY29uc2lkZXJpbmcgTW9uZGF5IGFzIHRoZSBmaXJzdCBkYXkuXG4gICAqIFRoZW4sIHRoaXMgYXJyYXkgaXMgbWFwcGVkIHZpYSB0aGUgYGdldFdlZWtgIGZ1bmN0aW9uLCB3aGljaCByZXR1cm5zIHRoZSBJU08gd2VlayBudW1iZXIgZm9yIHRoYXQgZGF0ZS5cbiAgICovXG4gIHByaXZhdGUgX2NyZWF0ZVdlZWtOdW1iZXJzKGRhdGU6IFQpOiBudW1iZXJbXSB7XG4gICAgcmV0dXJuIGVhY2hXZWVrT2ZJbnRlcnZhbChcbiAgICAgIHsgc3RhcnQ6IHN0YXJ0T2ZNb250aChkYXRlIGFzIERhdGUpLCBlbmQ6IGVuZE9mTW9udGgoZGF0ZSBhcyBEYXRlKSB9LFxuICAgICAgeyB3ZWVrU3RhcnRzT246IDEgfSxcbiAgICApLm1hcCgoZmlyc3REYXlPZldlZWs6IERhdGUpID0+XG4gICAgICBnZXRXZWVrKGZpcnN0RGF5T2ZXZWVrLCB7IHdlZWtTdGFydHNPbjogMSwgZmlyc3RXZWVrQ29udGFpbnNEYXRlOiA0IH0pLFxuICAgICk7XG4gIH1cblxuICAvKiogQ3JlYXRlcyB0aGUgcm93cyBhbG9uZyB0aGUgaG9yaXpvbnRhbCBkaXJlY3Rpb24gYW5kIHNldHMgdGhlIHBhcmFtZXRlcnMgdXNlZCBpbiBrZXlib2FyZCBuYXZpZ2F0aW9uLiAqL1xuICBwcml2YXRlIF9jcmVhdGVXZWVrUm93cyh2YWx1ZTogVCwgaXNTZWNvbmRNb250aEluVmlldyA9IGZhbHNlKTogRGF5PFQ+W11bXSB7XG4gICAgY29uc3QgZGF5c0luTW9udGg6IG51bWJlciA9IHRoaXMuX2RhdGVBZGFwdGVyLmdldE51bURheXNJbk1vbnRoKHZhbHVlKTtcbiAgICBjb25zdCB3ZWVrT2Zmc2V0OiBudW1iZXIgPSB0aGlzLl9kYXRlQWRhcHRlci5nZXRGaXJzdFdlZWtPZmZzZXQodmFsdWUpO1xuICAgIGlmICghaXNTZWNvbmRNb250aEluVmlldykge1xuICAgICAgdGhpcy5fa2V5Ym9hcmROYXZpZ2F0aW9uRGF5Vmlld1BhcmFtZXRlcnMuZmlyc3RNb250aExlbmd0aCA9IGRheXNJbk1vbnRoO1xuICAgICAgdGhpcy5fa2V5Ym9hcmROYXZpZ2F0aW9uRGF5Vmlld1BhcmFtZXRlcnMuZmlyc3RNb250aE9mZnNldCA9IHdlZWtPZmZzZXQ7XG4gICAgICB0aGlzLl9rZXlib2FyZE5hdmlnYXRpb25EYXlWaWV3UGFyYW1ldGVycy5maXJzdERheUluVmlldyA9IHRoaXMuX2RhdGVBZGFwdGVyLnRvSXNvODYwMShcbiAgICAgICAgdGhpcy5fZGF0ZUFkYXB0ZXIuY3JlYXRlRGF0ZShcbiAgICAgICAgICB0aGlzLl9kYXRlQWRhcHRlci5nZXRZZWFyKHZhbHVlKSxcbiAgICAgICAgICB0aGlzLl9kYXRlQWRhcHRlci5nZXRNb250aCh2YWx1ZSksXG4gICAgICAgICAgMSxcbiAgICAgICAgKSxcbiAgICAgICk7XG4gICAgICB0aGlzLl9rZXlib2FyZE5hdmlnYXRpb25EYXlWaWV3UGFyYW1ldGVycy5sYXN0RGF5SW5WaWV3ID0gdGhpcy5fZGF0ZUFkYXB0ZXIudG9Jc284NjAxKFxuICAgICAgICB0aGlzLl9kYXRlQWRhcHRlci5jcmVhdGVEYXRlKFxuICAgICAgICAgIHRoaXMuX2RhdGVBZGFwdGVyLmdldFllYXIodmFsdWUpLFxuICAgICAgICAgIHRoaXMuX2RhdGVBZGFwdGVyLmdldE1vbnRoKHZhbHVlKSxcbiAgICAgICAgICBkYXlzSW5Nb250aCxcbiAgICAgICAgKSxcbiAgICAgICk7XG4gICAgfSBlbHNlIHtcbiAgICAgIHRoaXMuX2tleWJvYXJkTmF2aWdhdGlvbkRheVZpZXdQYXJhbWV0ZXJzLnNlY29uZE1vbnRoT2Zmc2V0ID0gd2Vla09mZnNldDtcbiAgICAgIHRoaXMuX2tleWJvYXJkTmF2aWdhdGlvbkRheVZpZXdQYXJhbWV0ZXJzLmxhc3REYXlJblZpZXcgPSB0aGlzLl9kYXRlQWRhcHRlci50b0lzbzg2MDEoXG4gICAgICAgIHRoaXMuX2RhdGVBZGFwdGVyLmNyZWF0ZURhdGUoXG4gICAgICAgICAgdGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0WWVhcih2YWx1ZSksXG4gICAgICAgICAgdGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0TW9udGgodmFsdWUpLFxuICAgICAgICAgIGRheXNJbk1vbnRoLFxuICAgICAgICApLFxuICAgICAgKTtcbiAgICB9XG4gICAgcmV0dXJuIHRoaXMub3JpZW50YXRpb24gPT09ICdob3Jpem9udGFsJ1xuICAgICAgPyB0aGlzLl9jcmVhdGVXZWVrUm93c0hvcml6b250YWwodmFsdWUsIGRheXNJbk1vbnRoLCB3ZWVrT2Zmc2V0KVxuICAgICAgOiB0aGlzLl9jcmVhdGVXZWVrUm93c1ZlcnRpY2FsKHZhbHVlLCBkYXlzSW5Nb250aCwgd2Vla09mZnNldCk7XG4gIH1cblxuICAvKipcbiAgICogQ3JlYXRlcyB0aGUgcm93cyBmb3IgZWFjaCB3ZWVrIGluIG9yaWVudGF0aW9uPSdob3Jpem9udGFsJy5cbiAgICpcbiAgICogSXRlcmF0ZXMgdGhyb3VnaCB0aGUgZGF5cyBvZiB0aGUgbW9udGhzLCBjcmVhdGVzIGEgRGF5IG9iamVjdCBmb3IgZWFjaCBhbmQgcHVzaGVzIGl0IGludG8gYW5kIGFycmF5LlxuICAgKiBFYWNoIHNldmVuIGRheXMgKGNvbnNpZGVyaW5nIHRoZSBvZmZzZXQgYXQgdGhlIGJlZ2lubmluZyBvZiB0aGUgbW9udGgpIHJlc3RhcnRzIGZyb20gYW4gZW1wdHkgYXJyYXkuXG4gICAqXG4gICAqIFRoZSByZXN1bHQgaXMgYSBtYXRyaXggaW4gd2hpY2ggZXZlcnkgcm93IGlzIGEgd2VlayAob3IgcGFydCBvZiBpdCwgY29uc2lkZXJpbmcgb2Zmc2V0KS5cbiAgICovXG4gIHByaXZhdGUgX2NyZWF0ZVdlZWtSb3dzSG9yaXpvbnRhbCh2YWx1ZTogVCwgZGF5c0luTW9udGg6IG51bWJlciwgd2Vla09mZnNldDogbnVtYmVyKTogRGF5PFQ+W11bXSB7XG4gICAgY29uc3Qgd2Vla3M6IERheTxUPltdW10gPSBbW11dO1xuICAgIGZvciAobGV0IGkgPSAwLCBjZWxsID0gd2Vla09mZnNldDsgaSA8IGRheXNJbk1vbnRoOyBpKyssIGNlbGwrKykge1xuICAgICAgaWYgKGNlbGwgPT09IERBWVNfUEVSX1JPVykge1xuICAgICAgICB3ZWVrcy5wdXNoKFtdKTtcbiAgICAgICAgY2VsbCA9IDA7XG4gICAgICB9XG4gICAgICBjb25zdCBkYXRlID0gdGhpcy5fZGF0ZUFkYXB0ZXIuY3JlYXRlRGF0ZShcbiAgICAgICAgdGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0WWVhcih2YWx1ZSksXG4gICAgICAgIHRoaXMuX2RhdGVBZGFwdGVyLmdldE1vbnRoKHZhbHVlKSxcbiAgICAgICAgaSArIDEsXG4gICAgICApITtcbiAgICAgIHdlZWtzW3dlZWtzLmxlbmd0aCAtIDFdLnB1c2godGhpcy5fbWFwRGF0ZVRvRGF5KGRhdGUpKTtcbiAgICB9XG4gICAgcmV0dXJuIHdlZWtzO1xuICB9XG5cbiAgLyoqXG4gICAqIENyZWF0ZXMgdGhlIHJvd3MgZm9yIGVhY2ggd2VlayBpbiBvcmllbnRhdGlvbj0ndmVydGljYWwnLlxuICAgKlxuICAgKiBDcmVhdGVzIGEgbWF0cml4IHdpdGggc2V2ZW4gZW1wdHkgcm93cy5cbiAgICogSXRlcmF0ZXMgdGhyb3VnaCB0aGUgZGF5cyBvZiB0aGUgbW9udGhzLCBjcmVhdGVzIGEgRGF5IG9iamVjdCBmb3IgZWFjaFxuICAgKiBhbmQgcHVzaGVzIGl0IGludG8gdGhlIGNvcnJlY3QgYXJyYXkgY29uc2lkZXJpbmcgdGhlIG9mZnNldCBhdCB0aGUgYmVnaW5uaW5nIG9mIHRoZSBtb250aC5cbiAgICogRWFjaCBzZXZlbiBkYXlzIChpbmNsdWRpbmcgb2Zmc2V0KSByZXN0YXJ0cyBmcm9tIHRoZSBmaXJzdC5cbiAgICpcbiAgICogVGhlIHJlc3VsdCBpcyBhIG1hdHJpeCBpbiB3aGljaCBldmVyeSByb3cgaXMgYSBzZXQgb2Ygd2Vla2RheXMsIHNvOlxuICAgKiAtIHJvdyAwOiBhbGwgdGhlIE1vbmRheXM7XG4gICAqIC0gcm93IDE6IGFsbCB0aGUgVHVlc2RheXM7XG4gICAqIC0gLi4uXG4gICAqIC0gcm93IDc6IGFsbCB0aGUgU3VuZGF5cy5cbiAgICovXG4gIHByaXZhdGUgX2NyZWF0ZVdlZWtSb3dzVmVydGljYWwodmFsdWU6IFQsIGRheXNJbk1vbnRoOiBudW1iZXIsIHdlZWtPZmZzZXQ6IG51bWJlcik6IERheTxUPltdW10ge1xuICAgIGNvbnN0IHdlZWtzOiBEYXk8VD5bXVtdID0gQXJyYXkuZnJvbSh7IGxlbmd0aDogREFZU19QRVJfUk9XIH0sICgpID0+IFtdKTtcbiAgICBmb3IgKGxldCBpID0gMCwgY2VsbCA9IHdlZWtPZmZzZXQ7IGkgPCBkYXlzSW5Nb250aDsgaSsrLCBjZWxsKyspIHtcbiAgICAgIGlmIChjZWxsID09PSBEQVlTX1BFUl9ST1cpIHtcbiAgICAgICAgY2VsbCA9IDA7XG4gICAgICB9XG4gICAgICBjb25zdCBkYXRlID0gdGhpcy5fZGF0ZUFkYXB0ZXIuY3JlYXRlRGF0ZShcbiAgICAgICAgdGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0WWVhcih2YWx1ZSksXG4gICAgICAgIHRoaXMuX2RhdGVBZGFwdGVyLmdldE1vbnRoKHZhbHVlKSxcbiAgICAgICAgaSArIDEsXG4gICAgICApITtcbiAgICAgIHdlZWtzW2NlbGxdLnB1c2godGhpcy5fbWFwRGF0ZVRvRGF5KGRhdGUpKTtcbiAgICB9XG4gICAgcmV0dXJuIHdlZWtzO1xuICB9XG5cbiAgcHJpdmF0ZSBfbWFwRGF0ZVRvRGF5KGRhdGU6IFQpOiBEYXk8VD4ge1xuICAgIGNvbnN0IGlzb0RhdGUgPSB0aGlzLl9kYXRlQWRhcHRlci50b0lzbzg2MDEoZGF0ZSk7XG4gICAgcmV0dXJuIHtcbiAgICAgIHZhbHVlOiBpc29EYXRlLFxuICAgICAgZGF0ZVZhbHVlOiBkYXRlLFxuICAgICAgZGF5VmFsdWU6IFN0cmluZyh0aGlzLl9kYXRlQWRhcHRlci5nZXREYXRlKGRhdGUpKSxcbiAgICAgIG1vbnRoVmFsdWU6IFN0cmluZyh0aGlzLl9kYXRlQWRhcHRlci5nZXRNb250aChkYXRlKSksXG4gICAgICB5ZWFyVmFsdWU6IFN0cmluZyh0aGlzLl9kYXRlQWRhcHRlci5nZXRZZWFyKGRhdGUpKSxcbiAgICAgIHdlZWtWYWx1ZTogZ2V0V2Vlayhpc29EYXRlLCB7IHdlZWtTdGFydHNPbjogMSwgZmlyc3RXZWVrQ29udGFpbnNEYXRlOiA0IH0pLFxuICAgICAgd2Vla0RheVZhbHVlOiB0aGlzLl9kYXRlQWRhcHRlci5nZXREYXlPZldlZWsoZGF0ZSksXG4gICAgfTtcbiAgfVxuXG4gIC8qKiBGb3JjZSB0aGUgY29udmVyc2lvbiB0byBJU084NjAxIGZvcm1hdHRlZCBzdHJpbmcuICovXG4gIHByaXZhdGUgX21hcFZhbHVlVG9JU09EYXRlKHZhbHVlOiBzdHJpbmcgfCBEYXRlKTogc3RyaW5nIHtcbiAgICByZXR1cm4gdHlwZW9mIHZhbHVlID09PSAnc3RyaW5nJyA/IHZhbHVlIDogdGhpcy5fZGF0ZUFkYXB0ZXIudG9Jc284NjAxKHZhbHVlIGFzIFQpO1xuICB9XG5cbiAgLyoqIENyZWF0ZXMgdGhlIHJvd3MgZm9yIHRoZSBtb250aCBzZWxlY3Rpb24gdmlldy4gKi9cbiAgcHJpdmF0ZSBfY3JlYXRlTW9udGhSb3dzKCk6IHZvaWQge1xuICAgIGNvbnN0IG1vbnRoczogTW9udGhDZWxsW10gPSBuZXcgQXJyYXkoMTIpLmZpbGwobnVsbCkubWFwKFxuICAgICAgKF8sIGk6IG51bWJlcik6IE1vbnRoQ2VsbCA9PiAoe1xuICAgICAgICB2YWx1ZTogU3RyaW5nKGkgKyAxKS5wYWRTdGFydCgyLCAnMCcpLFxuICAgICAgICBtb250aFZhbHVlOiBpICsgMSxcbiAgICAgIH0pLFxuICAgICk7XG4gICAgY29uc3Qgcm93czogbnVtYmVyID0gMTIgLyBNT05USFNfUEVSX1JPVztcbiAgICBjb25zdCBtb250aEFycmF5OiBNb250aENlbGxbXVtdID0gW107XG4gICAgZm9yIChsZXQgaTogbnVtYmVyID0gMDsgaSA8IHJvd3M7IGkrKykge1xuICAgICAgbW9udGhBcnJheS5wdXNoKG1vbnRocy5zbGljZShNT05USFNfUEVSX1JPVyAqIGksIE1PTlRIU19QRVJfUk9XICogKGkgKyAxKSkpO1xuICAgIH1cbiAgICB0aGlzLl9tb250aHMgPSBtb250aEFycmF5O1xuICB9XG5cbiAgLyoqIENyZWF0ZXMgdGhlIHJvd3MgZm9yIHRoZSB5ZWFyIHNlbGVjdGlvbiB2aWV3LiAqL1xuICBwcml2YXRlIF9jcmVhdGVZZWFyUm93cyhvZmZzZXQ6IG51bWJlciA9IDApOiBudW1iZXJbXVtdIHtcbiAgICBjb25zdCBzdGFydFZhbHVlWWVhclZpZXc6IG51bWJlciA9IHRoaXMuX2dldFN0YXJ0VmFsdWVZZWFyVmlldygpO1xuICAgIGNvbnN0IGFsbFllYXJzOiBudW1iZXJbXSA9IG5ldyBBcnJheShZRUFSU19QRVJfUEFHRSlcbiAgICAgIC5maWxsKDApXG4gICAgICAubWFwKChfLCBpOiBudW1iZXIpID0+IHN0YXJ0VmFsdWVZZWFyVmlldyArIG9mZnNldCArIGkpO1xuICAgIGNvbnN0IHJvd3M6IG51bWJlciA9IFlFQVJTX1BFUl9QQUdFIC8gWUVBUlNfUEVSX1JPVztcbiAgICBjb25zdCB5ZWFyQXJyYXk6IG51bWJlcltdW10gPSBbXTtcbiAgICBmb3IgKGxldCBpOiBudW1iZXIgPSAwOyBpIDwgcm93czsgaSsrKSB7XG4gICAgICB5ZWFyQXJyYXkucHVzaChhbGxZZWFycy5zbGljZShZRUFSU19QRVJfUk9XICogaSwgWUVBUlNfUEVSX1JPVyAqIChpICsgMSkpKTtcbiAgICB9XG4gICAgcmV0dXJuIHllYXJBcnJheTtcbiAgfVxuXG4gIC8qKlxuICAgKiBDYWxjdWxhdGVzIHRoZSBmaXJzdCB5ZWFyIHRoYXQgd2lsbCBiZSBzaG93biBpbiB0aGUgeWVhciBzZWxlY3Rpb24gcGFuZWwuXG4gICAqIElmIGBtaW5EYXRlYCBhbmQgYG1heERhdGVgIGFyZSBib3RoIG51bGwsIHRoZSBzdGFydGluZyB5ZWFyIGlzIGNhbGN1bGF0ZWQgYXNcbiAgICogdGhlIG11bHRpcGxlIG9mIFlFQVJTX1BFUl9QQUdFIGNsb3Nlc3QgdG8gYW5kIGxlc3MgdGhhbiBhY3RpdmVEYXRlLFxuICAgKiBlLmcuLCB3aXRoIGBZRUFSU19QRVJfUEFHRWAgPSAyNCBhbmQgYGFjdGl2ZURhdGVgID0gMjAyMCwgdGhlIGZ1bmN0aW9uIHdpbGwgcmV0dXJuIDIwMTYgKDI0ICogODMpLFxuICAgKiB3aGlsZSB3aXRoIGBhY3RpdmVEYXRlYCA9IDIwMDAsIHRoZSBmdW5jdGlvbiB3aWxsIHJldHVybiAxOTkyICgyNCAqIDgyKS5cbiAgICogSWYgYG1pbkRhdGVgIGlzIG5vdCBudWxsLCBpdCByZXR1cm5zIHRoZSBjb3JyZXNwb25kaW5nIHllYXI7IGlmIGBtYXhEYXRlYCBpcyBub3QgbnVsbCxcbiAgICogaXQgcmV0dXJucyB0aGUgY29ycmVzcG9uZGluZyB5ZWFyIG1pbnVzIGBZRUFSU19QRVJfUEFHRWAsIHNvIHRoYXQgdGhlIGBtYXhEYXRlYCBpcyB0aGUgbGFzdCByZW5kZXJlZCB5ZWFyLlxuICAgKiBJZiBib3RoIGFyZSBub3QgbnVsbCwgYG1heERhdGVgIGhhcyBwcmlvcml0eSBvdmVyIGBtaW5EYXRlYC5cbiAgICovXG4gIHByaXZhdGUgX2dldFN0YXJ0VmFsdWVZZWFyVmlldygpOiBudW1iZXIge1xuICAgIGxldCBzdGFydGluZ1llYXIgPSAwO1xuICAgIGlmICh0aGlzLm1heCkge1xuICAgICAgc3RhcnRpbmdZZWFyID0gdGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0WWVhcih0aGlzLm1heCkgLSBZRUFSU19QRVJfUEFHRSArIDE7XG4gICAgfSBlbHNlIGlmICh0aGlzLm1pbikge1xuICAgICAgc3RhcnRpbmdZZWFyID0gdGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0WWVhcih0aGlzLm1pbik7XG4gICAgfVxuICAgIGNvbnN0IGFjdGl2ZVllYXIgPSB0aGlzLl9kYXRlQWRhcHRlci5nZXRZZWFyKHRoaXMuX2FjdGl2ZURhdGUpO1xuICAgIHJldHVybiAoXG4gICAgICBhY3RpdmVZZWFyIC1cbiAgICAgICgoKChhY3RpdmVZZWFyIC0gc3RhcnRpbmdZZWFyKSAlIFlFQVJTX1BFUl9QQUdFKSArIFlFQVJTX1BFUl9QQUdFKSAlIFlFQVJTX1BFUl9QQUdFKVxuICAgICk7XG4gIH1cblxuICAvKiogQ2hlY2tzIGlmIGRhdGUgaXMgd2l0aGluIHRoZSBtaW4tbWF4IHJhbmdlLiAqL1xuICBwcml2YXRlIF9pc0RheUluUmFuZ2UoZGF0ZVN0cmluZzogc3RyaW5nKTogYm9vbGVhbiB7XG4gICAgaWYgKCF0aGlzLm1pbiAmJiAhdGhpcy5tYXgpIHtcbiAgICAgIHJldHVybiB0cnVlO1xuICAgIH1cbiAgICBjb25zdCBkYXRlID0gdGhpcy5fZGF0ZUFkYXB0ZXIuZGVzZXJpYWxpemUoZGF0ZVN0cmluZykhO1xuICAgIHJldHVybiB0aGlzLl9kYXRlQWRhcHRlci5zYW1lRGF0ZShkYXRlLCB0aGlzLl9kYXRlQWRhcHRlci5jbGFtcERhdGUoZGF0ZSwgdGhpcy5taW4sIHRoaXMubWF4KSk7XG4gIH1cblxuICAvKiogRW1pdHMgdGhlIHNlbGVjdGVkIGRhdGUgYW5kIHNldHMgaXQgaW50ZXJuYWxseS4gKi9cbiAgcHJpdmF0ZSBfc2VsZWN0RGF0ZShkYXk6IFQpOiB2b2lkIHtcbiAgICB0aGlzLl9jaG9zZW5Nb250aCA9IHVuZGVmaW5lZDtcbiAgICB0aGlzLl9zZXRDaG9zZW5ZZWFyKCk7XG4gICAgaWYgKHRoaXMubXVsdGlwbGUpIHtcbiAgICAgIC8vIENoZWNrIGlmIF9zZWxlY3RlZCBoYXMgZWxlbWVudHNcbiAgICAgIGlmICh0aGlzLl9zZWxlY3RlZCAmJiAodGhpcy5fc2VsZWN0ZWQgYXMgVFtdKS5sZW5ndGggPiAwKSB7XG4gICAgICAgIGNvbnN0IGluZGV4T2ZTZWxlY3RlZERheTogbnVtYmVyID0gKHRoaXMuX3NlbGVjdGVkIGFzIFRbXSkuZmluZEluZGV4KFxuICAgICAgICAgIChzZWwpID0+IHRoaXMuX2RhdGVBZGFwdGVyLmNvbXBhcmVEYXRlKHNlbCwgZGF5KSA9PT0gMCxcbiAgICAgICAgKTtcbiAgICAgICAgLy8gSWYgdGhlIHNlbGVjdGVkIGRhdGUgaXMgYWxyZWFkeSBpbiB0aGUgX3NlbGVjdGVkIGFycmF5LCByZW1vdmUgaXQsIG90aGVyd2lzZSBhZGQgaXRcbiAgICAgICAgaWYgKGluZGV4T2ZTZWxlY3RlZERheSAhPT0gLTEpIHtcbiAgICAgICAgICB0aGlzLl9zZWxlY3RlZCA9ICh0aGlzLl9zZWxlY3RlZCBhcyBUW10pLmZpbHRlcigoXywgaSkgPT4gaSAhPT0gaW5kZXhPZlNlbGVjdGVkRGF5KTtcbiAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICB0aGlzLl9zZWxlY3RlZCA9IFsuLi4odGhpcy5fc2VsZWN0ZWQgYXMgVFtdKSwgZGF5XTtcbiAgICAgICAgfVxuICAgICAgfSBlbHNlIHtcbiAgICAgICAgLy8gSWYgX3NlbGVjdGVkIGlzIGVtcHR5LCBzZXQgaXRcbiAgICAgICAgdGhpcy5fc2VsZWN0ZWQgPSBbZGF5XTtcbiAgICAgIH1cbiAgICAgIHRoaXMuX2VtaXREYXRlU2VsZWN0ZWRFdmVudCh0aGlzLl9zZWxlY3RlZC5tYXAoKGUpID0+IHRoaXMuX2RhdGVBZGFwdGVyLmRlc2VyaWFsaXplKGUpISkpO1xuICAgIH0gZWxzZSB7XG4gICAgICAvLyBJbiBzaW5nbGUgc2VsZWN0aW9uLCBjaGVjayBpZiB0aGUgZGF5IGlzIGFscmVhZHkgc2VsZWN0ZWRcbiAgICAgIGlmICghdGhpcy5fc2VsZWN0ZWQgfHwgdGhpcy5fZGF0ZUFkYXB0ZXIuY29tcGFyZURhdGUodGhpcy5fc2VsZWN0ZWQgYXMgVCwgZGF5KSAhPT0gMCkge1xuICAgICAgICB0aGlzLl9zZWxlY3RlZCA9IGRheTtcbiAgICAgICAgdGhpcy5fZW1pdERhdGVTZWxlY3RlZEV2ZW50KHRoaXMuX2RhdGVBZGFwdGVyLmRlc2VyaWFsaXplKGRheSkhKTtcbiAgICAgIH1cbiAgICB9XG4gIH1cblxuICAvKipcbiAgICogSGFuZGxlIG11bHRpcGxlIGRhdGVzIHNlbGVjdGlvbiB2aWEgd2Vla051bWJlciAvIHdlZWtEYXkgYnV0dG9uczpcbiAgICogLSBpZiBDbWQgb3IgQ3RybCBhcmUgcHJlc3NlZCwgYWRkIHRoZSBuZXcgZGF0ZSB0byB0aGUgY3VycmVudCBvbmVzO1xuICAgKiAtIGlmIG5vdCxcbiAgICogICAgIC0gaWYgdGhlIG5ldyBkYXRlcyBhcmUgdGhlIHNhbWUgb2YgdGhlIGN1cnJlbnQgb25lcywgaXQgbWVhbnMgdGhhdCB0aGUgc2FtZSBidXR0b24gaGFzIGJlZW4gY2xpY2tlZCB0d2ljZSwgc28gZG8gbm90aGluZztcbiAgICogICAgIC0gaWYgbm90LCB0aGUgc2VsZWN0ZWQgZGF0ZXMgYXJlIHRoZSBuZXcgb25lcy5cbiAgICovXG4gIHByaXZhdGUgX3NlbGVjdE11bHRpcGxlRGF0ZXMoZGF5czogRGF5PFQ+W10pOiB2b2lkIHtcbiAgICAvLyBGaWx0ZXIgZGlzYWJsZWQgZGF5cyBieSBtYXRjaGluZyB0aGUgcHJvdmlkZWQgYGRheXNgIHBhcmFtZXRlciBhZ2FpbnN0IHRoZSBlbmFibGVkIGNlbGxzLlxuICAgIC8vIFNpbmNlIHRoZSBidXR0b25zJyB2YWx1ZSBpcyBzZXQgdG8gdGhlIERheSdzIGludGVyZmFjZSB2YWx1ZSAoSVNPIHN0cmluZyksIHRoZXJlJ3Mgbm8gbmVlZCB0byBkZXNlcmlhbGl6ZSBpdC5cbiAgICBjb25zdCBlbmFibGVkRGF5czogc3RyaW5nW10gPSAodGhpcy5fY2VsbHMgYXMgU2JiQ2FsZW5kYXJEYXlFbGVtZW50W10pXG4gICAgICAuZmlsdGVyKChlKSA9PiAhZS5kaXNhYmxlZClcbiAgICAgIC5tYXAoKGUpID0+IHRoaXMuX21hcFZhbHVlVG9JU09EYXRlKGUudmFsdWUhKSk7XG4gICAgY29uc3QgZGF5c1RvQWRkOiBzdHJpbmdbXSA9IGRheXNcbiAgICAgIC5tYXAoKGU6IERheTxUPikgPT4gZS52YWx1ZSlcbiAgICAgIC5maWx0ZXIoKGlzb0RhdGU6IHN0cmluZykgPT4gZW5hYmxlZERheXMuaW5jbHVkZXMoaXNvRGF0ZSkpO1xuICAgIGNvbnN0IGRheXNUb0FkZFNldCA9IG5ldyBTZXQoZGF5c1RvQWRkKTtcbiAgICBjb25zdCBzZWxlY3RlZFNldCA9IG5ldyBTZXQoKHRoaXMuX3NlbGVjdGVkIGFzIFRbXSkubWFwKChzKSA9PiB0aGlzLl9kYXRlQWRhcHRlci50b0lzbzg2MDEocykpKTtcbiAgICBjb25zdCBzZWxTdHJpbmdzID0gdGhpcy5fdXBkYXRlU2VsZWN0ZWRXaXRoTXVsdGlwbGVEYXRlcyhkYXlzVG9BZGQsIGRheXNUb0FkZFNldCwgc2VsZWN0ZWRTZXQpO1xuICAgIHRoaXMuX3NlbGVjdGVkID0gc2VsU3RyaW5ncy5tYXAoKHMpID0+IHRoaXMuX2RhdGVBZGFwdGVyLmRlc2VyaWFsaXplKHMpISk7XG5cbiAgICB0aGlzLl9lbWl0RGF0ZVNlbGVjdGVkRXZlbnQodGhpcy5fc2VsZWN0ZWQubWFwKChlKSA9PiB0aGlzLl9kYXRlQWRhcHRlci5kZXNlcmlhbGl6ZShlKSEpKTtcbiAgfVxuXG4gIC8qKlxuICAgKiBFbWl0cyB0aGUgZGF0ZXNlbGVjdGVkIGV2ZW50IGdpdmVuIHRoZSBkZXRhaWwgKGFzIFQgb3IgVFtdIGJhc2VkIG9uIHRoZSB2YWx1ZSBvZiB0aGUgbXVsdGlwbGUgZmxhZykuXG4gICAqL1xuICBwcml2YXRlIF9lbWl0RGF0ZVNlbGVjdGVkRXZlbnQoZGV0YWlsOiBUIHwgVFtdKTogdm9pZCB7XG4gICAgLyoqIEB0eXBlIHtDdXN0b21FdmVudDxUIHwgVFtdPn0gRXZlbnQgZW1pdHRlZCBvbiBkYXRlIHNlbGVjdGlvbi4gKi9cbiAgICB0aGlzLmRpc3BhdGNoRXZlbnQoXG4gICAgICBuZXcgQ3VzdG9tRXZlbnQ8VCB8IFRbXT4oJ2RhdGVzZWxlY3RlZCcsIHtcbiAgICAgICAgZGV0YWlsLFxuICAgICAgICBjb21wb3NlZDogdHJ1ZSxcbiAgICAgICAgYnViYmxlczogdHJ1ZSxcbiAgICAgIH0pLFxuICAgICk7XG4gIH1cblxuICBwcml2YXRlIF9lbWl0TW9udGhDaGFuZ2UoKTogdm9pZCB7XG4gICAgLy8gRklYTUU6IHRoZSBuYW1lIG9mIHRoaXMgdmFyaWFibGUgYXBwZWFycyBhcyBldmVudCBuYW1lIGluIHRoZSByZWFkbWVcbiAgICAvLyAgZHVlIHRvIGEgYnVnIGluIHRoZSBjdXN0b20tZWxlbWVudHMtbWFuaWZlc3QgbGlicmFyeS5cbiAgICAvLyAgaHR0cHM6Ly9naXRodWIuY29tL29wZW4td2MvY3VzdG9tLWVsZW1lbnRzLW1hbmlmZXN0L2lzc3Vlcy8xNDlcbiAgICBjb25zdCBtb250aGNoYW5nZSA9ICh0aGlzLndpZGUgPyBbLi4udGhpcy5fd2Vla3MsIC4uLnRoaXMuX25leHRNb250aFdlZWtzXSA6IHRoaXMuX3dlZWtzKVxuICAgICAgLmZsYXQoKVxuICAgICAgLnNvcnQoKGEsIGIpID0+IGEudmFsdWUubG9jYWxlQ29tcGFyZShiLnZhbHVlKSkgYXMgRGF5W107XG4gICAgLyoqXG4gICAgICogQHR5cGUge1NiYk1vbnRoQ2hhbmdlRXZlbnR9XG4gICAgICogRW1pdHMgd2hlbiB0aGUgbW9udGggY2hhbmdlcy5cbiAgICAgKiBUaGUgYHJhbmdlYCBwcm9wZXJ0eSBjb250YWlucyB0aGUgZGF5cyBhcnJheSBvZiB0aGUgY2hvc2VuIG1vbnRoLlxuICAgICAqL1xuICAgIHRoaXMuZGlzcGF0Y2hFdmVudChuZXcgU2JiTW9udGhDaGFuZ2VFdmVudChtb250aGNoYW5nZSkpO1xuICB9XG5cbiAgLyoqXG4gICAqIEluIGNhc2Ugb2YgbXVsdGlwbGUgc2VsZWN0aW9uLCBuZXdseSBhZGRlZCBkYXlzIG11c3QgYmUgYWRkZWQgdG8gdGhlIGV4aXN0aW5nIG9uZXMsIHdpdGhvdXQgZHVwbGljYXRpb24uXG4gICAqIElmIHRoZSBkYXlzIHRvIGFkZCBhcmUgZXhhY3RseSB0aGUgc2FtZSBhcyB0aGUgc2VsZWN0ZWQgb25lcywgdGhlIHNldCBtdXN0IGJlIGVtcHRpZWQuXG4gICAqL1xuICBwcml2YXRlIF91cGRhdGVTZWxlY3RlZFdpdGhNdWx0aXBsZURhdGVzKFxuICAgIGRheXNUb0FkZDogc3RyaW5nW10sXG4gICAgZGF5c1RvQWRkU2V0OiBTZXQ8c3RyaW5nPixcbiAgICBzZWxlY3RlZFNldDogU2V0PHN0cmluZz4sXG4gICk6IHN0cmluZ1tdIHtcbiAgICBpZiAoZGF5c1RvQWRkLmV2ZXJ5KChkYXk6IHN0cmluZykgPT4gc2VsZWN0ZWRTZXQuaGFzKGRheSkpKSB7XG4gICAgICBkYXlzVG9BZGRTZXQuZm9yRWFjaCgoZGF5OiBzdHJpbmcpID0+IHNlbGVjdGVkU2V0LmRlbGV0ZShkYXkpKTtcbiAgICB9IGVsc2Uge1xuICAgICAgZGF5c1RvQWRkU2V0LmZvckVhY2goKGRheTogc3RyaW5nKSA9PiBzZWxlY3RlZFNldC5hZGQoZGF5KSk7XG4gICAgfVxuICAgIHJldHVybiBBcnJheS5mcm9tKHNlbGVjdGVkU2V0KTtcbiAgfVxuXG4gIHByaXZhdGUgX3NldENob3NlblllYXIoKTogdm9pZCB7XG4gICAgaWYgKHRoaXMudmlldyA9PT0gJ21vbnRoJykge1xuICAgICAgbGV0IHNlbGVjdGVkRGF0ZTogVCB8IHVuZGVmaW5lZDtcbiAgICAgIGlmICh0aGlzLm11bHRpcGxlKSB7XG4gICAgICAgIHNlbGVjdGVkRGF0ZSA9ICh0aGlzLnNlbGVjdGVkIGFzIFRbXSkuYXQoLTEpO1xuICAgICAgfSBlbHNlIHtcbiAgICAgICAgc2VsZWN0ZWREYXRlID0gdGhpcy5zZWxlY3RlZCBhcyBUO1xuICAgICAgfVxuICAgICAgdGhpcy5fY2hvc2VuWWVhciA9IHRoaXMuX2RhdGVBZGFwdGVyLmdldFllYXIoc2VsZWN0ZWREYXRlID8/IHRoaXMuX2RhdGVBZGFwdGVyLnRvZGF5KCkpO1xuICAgIH0gZWxzZSB7XG4gICAgICB0aGlzLl9jaG9zZW5ZZWFyID0gdW5kZWZpbmVkO1xuICAgIH1cbiAgfVxuXG4gIHByaXZhdGUgX2Fzc2lnbkFjdGl2ZURhdGUoZGF0ZTogVCk6IHZvaWQge1xuICAgIGlmICh0aGlzLm1pbiAmJiB0aGlzLl9kYXRlQWRhcHRlci5jb21wYXJlRGF0ZSh0aGlzLm1pbiwgZGF0ZSkgPiAwKSB7XG4gICAgICB0aGlzLl9hY3RpdmVEYXRlID0gdGhpcy5taW47XG4gICAgICByZXR1cm47XG4gICAgfVxuICAgIGlmICh0aGlzLm1heCAmJiB0aGlzLl9kYXRlQWRhcHRlci5jb21wYXJlRGF0ZSh0aGlzLm1heCwgZGF0ZSkgPCAwKSB7XG4gICAgICB0aGlzLl9hY3RpdmVEYXRlID0gdGhpcy5tYXg7XG4gICAgICByZXR1cm47XG4gICAgfVxuICAgIHRoaXMuX2FjdGl2ZURhdGUgPSBkYXRlO1xuICB9XG5cbiAgLyoqIEdvZXMgdG8gdGhlIG1vbnRoIGlkZW50aWZpZWQgYnkgdGhlIHNoaWZ0LiAqL1xuICBwcml2YXRlIF9nb1RvRGlmZmVyZW50TW9udGgobW9udGhzOiBudW1iZXIpOiB2b2lkIHtcbiAgICB0aGlzLl9pbml0KHRoaXMuX2RhdGVBZGFwdGVyLmFkZENhbGVuZGFyTW9udGhzKHRoaXMuX2FjdGl2ZURhdGUsIG1vbnRocykpO1xuICAgIHRoaXMuX2VtaXRNb250aENoYW5nZSgpO1xuICB9XG5cbiAgcHJpdmF0ZSBfZ29Ub0RpZmZlcmVudFllYXIoeWVhcnM6IG51bWJlcik6IHZvaWQge1xuICAgIHRoaXMuX2Nob3NlblllYXIhICs9IHllYXJzO1xuICAgIC8vIENhbid0IHVzZSBgX2Fzc2lnbkFjdGl2ZURhdGUoLi4uKWAgaGVyZSwgYmVjYXVzZSBpdCB3aWxsIHNldCBpdCB0byBtaW4vbWF4IHZhbHVlIGlmIGFyZ3VtZW50IGlzIG91dCBvZiByYW5nZVxuICAgIHRoaXMuX2FjdGl2ZURhdGUgPSB0aGlzLl9kYXRlQWRhcHRlci5jcmVhdGVEYXRlKFxuICAgICAgdGhpcy5fY2hvc2VuWWVhciEsXG4gICAgICB0aGlzLl9kYXRlQWRhcHRlci5nZXRNb250aCh0aGlzLl9hY3RpdmVEYXRlKSxcbiAgICAgIHRoaXMuX2RhdGVBZGFwdGVyLmdldERhdGUodGhpcy5fYWN0aXZlRGF0ZSksXG4gICAgKTtcbiAgICB0aGlzLl9pbml0KCk7XG4gIH1cblxuICBwcml2YXRlIF9nb1RvRGlmZmVyZW50WWVhclJhbmdlKHllYXJzOiBudW1iZXIpOiB2b2lkIHtcbiAgICB0aGlzLl9pbml0KHRoaXMuX2RhdGVBZGFwdGVyLmFkZENhbGVuZGFyWWVhcnModGhpcy5fYWN0aXZlRGF0ZSwgeWVhcnMpKTtcbiAgfVxuXG4gIHByaXZhdGUgX3ByZXZEaXNhYmxlZChwcmV2RGF0ZTogVCk6IGJvb2xlYW4ge1xuICAgIGlmICghdGhpcy5taW4pIHtcbiAgICAgIHJldHVybiBmYWxzZTtcbiAgICB9XG4gICAgcmV0dXJuIHRoaXMuX2RhdGVBZGFwdGVyLmNvbXBhcmVEYXRlKHByZXZEYXRlLCB0aGlzLm1pbikgPCAwO1xuICB9XG5cbiAgcHJpdmF0ZSBfbmV4dERpc2FibGVkKG5leHREYXRlOiBUKTogYm9vbGVhbiB7XG4gICAgaWYgKCF0aGlzLm1heCkge1xuICAgICAgcmV0dXJuIGZhbHNlO1xuICAgIH1cbiAgICByZXR1cm4gdGhpcy5fZGF0ZUFkYXB0ZXIuY29tcGFyZURhdGUobmV4dERhdGUsIHRoaXMubWF4KSA+IDA7XG4gIH1cblxuICAvKiogQ2hlY2tzIGlmIHRoZSBcInByZXZpb3VzIG1vbnRoXCIgYnV0dG9uIHNob3VsZCBiZSBkaXNhYmxlZC4gKi9cbiAgcHJpdmF0ZSBfcHJldmlvdXNNb250aERpc2FibGVkKCk6IGJvb2xlYW4ge1xuICAgIGNvbnN0IHByZXZNb250aCA9IHRoaXMuX2RhdGVBZGFwdGVyLmFkZENhbGVuZGFyRGF5cyhcbiAgICAgIHRoaXMuX2FjdGl2ZURhdGUsXG4gICAgICB0aGlzLl9kYXRlQWRhcHRlci5nZXREYXRlKHRoaXMuX2FjdGl2ZURhdGUpICogLTEsXG4gICAgKTtcbiAgICByZXR1cm4gdGhpcy5fcHJldkRpc2FibGVkKHByZXZNb250aCk7XG4gIH1cblxuICAvKiogQ2hlY2tzIGlmIHRoZSBcIm5leHQgbW9udGhcIiBidXR0b24gc2hvdWxkIGJlIGRpc2FibGVkLiAqL1xuICBwcml2YXRlIF9uZXh0TW9udGhEaXNhYmxlZCgpOiBib29sZWFuIHtcbiAgICBsZXQgbmV4dE1vbnRoID0gdGhpcy5fZGF0ZUFkYXB0ZXIuYWRkQ2FsZW5kYXJNb250aHModGhpcy5fYWN0aXZlRGF0ZSwgdGhpcy5fd2lkZSA/IDIgOiAxKTtcbiAgICBuZXh0TW9udGggPSB0aGlzLl9kYXRlQWRhcHRlci5jcmVhdGVEYXRlKFxuICAgICAgdGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0WWVhcihuZXh0TW9udGgpLFxuICAgICAgdGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0TW9udGgobmV4dE1vbnRoKSxcbiAgICAgIDEsXG4gICAgKTtcbiAgICByZXR1cm4gdGhpcy5fbmV4dERpc2FibGVkKG5leHRNb250aCk7XG4gIH1cblxuICAvKiogQ2hlY2tzIGlmIHRoZSBcInByZXZpb3VzIHllYXJcIiBidXR0b24gc2hvdWxkIGJlIGRpc2FibGVkLiAqL1xuICBwcml2YXRlIF9wcmV2aW91c1llYXJEaXNhYmxlZCgpOiBib29sZWFuIHtcbiAgICBjb25zdCBwcmV2WWVhciA9IHRoaXMuX2RhdGVBZGFwdGVyLmNyZWF0ZURhdGUoXG4gICAgICB0aGlzLl9kYXRlQWRhcHRlci5nZXRZZWFyKHRoaXMuX2FjdGl2ZURhdGUpIC0gMSxcbiAgICAgIDEyLFxuICAgICAgMzEsXG4gICAgKTtcbiAgICByZXR1cm4gdGhpcy5fcHJldkRpc2FibGVkKHByZXZZZWFyKTtcbiAgfVxuXG4gIC8qKiBDaGVja3MgaWYgdGhlIFwibmV4dCB5ZWFyXCIgYnV0dG9uIHNob3VsZCBiZSBkaXNhYmxlZC4gKi9cbiAgcHJpdmF0ZSBfbmV4dFllYXJEaXNhYmxlZCgpOiBib29sZWFuIHtcbiAgICBjb25zdCBuZXh0WWVhciA9IHRoaXMuX2RhdGVBZGFwdGVyLmNyZWF0ZURhdGUoXG4gICAgICB0aGlzLl9kYXRlQWRhcHRlci5nZXRZZWFyKHRoaXMuX2FjdGl2ZURhdGUpICsgKHRoaXMuX3dpZGUgPyAyIDogMSksXG4gICAgICAxLFxuICAgICAgMSxcbiAgICApO1xuICAgIHJldHVybiB0aGlzLl9uZXh0RGlzYWJsZWQobmV4dFllYXIpO1xuICB9XG5cbiAgLyoqIENoZWNrcyBpZiB0aGUgXCJwcmV2aW91cyB5ZWFyXCIgYnV0dG9uIHNob3VsZCBiZSBkaXNhYmxlZCBpbiB5ZWFyIHZpZXcuICovXG4gIHByaXZhdGUgX3ByZXZpb3VzWWVhclJhbmdlRGlzYWJsZWQoKTogYm9vbGVhbiB7XG4gICAgY29uc3QgcHJldlllYXIgPSB0aGlzLl9kYXRlQWRhcHRlci5jcmVhdGVEYXRlKHRoaXMuX3llYXJzWzBdWzBdIC0gMSwgMTIsIDMxKTtcbiAgICByZXR1cm4gdGhpcy5fcHJldkRpc2FibGVkKHByZXZZZWFyKTtcbiAgfVxuXG4gIC8qKiBDaGVja3MgaWYgdGhlIFwibmV4dCB5ZWFyXCIgYnV0dG9uIHNob3VsZCBiZSBkaXNhYmxlZCBpbiB5ZWFyIHZpZXcuICovXG4gIHByaXZhdGUgX25leHRZZWFyUmFuZ2VEaXNhYmxlZCgpOiBib29sZWFuIHtcbiAgICBjb25zdCB5ZWFycyA9IHRoaXMuX3dpZGUgPyB0aGlzLl9uZXh0TW9udGhZZWFycyA6IHRoaXMuX3llYXJzO1xuICAgIGNvbnN0IGxhc3RZZWFyUmFuZ2UgPSB5ZWFyc1t5ZWFycy5sZW5ndGggLSAxXTtcbiAgICBjb25zdCBsYXN0WWVhciA9IGxhc3RZZWFyUmFuZ2VbbGFzdFllYXJSYW5nZS5sZW5ndGggLSAxXTtcbiAgICBjb25zdCBuZXh0WWVhciA9IHRoaXMuX2RhdGVBZGFwdGVyLmNyZWF0ZURhdGUobGFzdFllYXIgKyAxLCAxLCAxKTtcbiAgICByZXR1cm4gdGhpcy5fbmV4dERpc2FibGVkKG5leHRZZWFyKTtcbiAgfVxuXG4gIHByaXZhdGUgX2hhbmRsZVRhYmxlQmx1cihldmVudFRhcmdldDogSFRNTEVsZW1lbnQpOiB2b2lkIHtcbiAgICBpZiAoZXZlbnRUYXJnZXQ/LmxvY2FsTmFtZSAhPT0gJ3NiYi1jYWxlbmRhci1kYXknKSB7XG4gICAgICB0aGlzLl9zZXRUYWJJbmRleCgpO1xuICAgIH1cbiAgfVxuXG4gIHByaXZhdGUgX3NldFRhYkluZGV4KCk6IHZvaWQge1xuICAgIEFycmF5LmZyb20odGhpcy5fY2VsbHMuZmlsdGVyKChlKSA9PiBlLnRhYkluZGV4ID09PSAwKSA/PyBbXSkuZm9yRWFjaChcbiAgICAgIChkYXkpID0+IChkYXkudGFiSW5kZXggPSAtMSksXG4gICAgKTtcbiAgICBjb25zdCBmaXJzdEZvY3VzYWJsZSA9IHRoaXMuX2dldEZpcnN0Rm9jdXNhYmxlKCk7XG4gICAgaWYgKGZpcnN0Rm9jdXNhYmxlKSB7XG4gICAgICBmaXJzdEZvY3VzYWJsZS50YWJJbmRleCA9IDA7XG4gICAgfVxuICB9XG5cbiAgLyoqIEdldCB0aGUgZWxlbWVudCBpbiB0aGUgY2FsZW5kYXIgdG8gYXNzaWduIGZvY3VzLiAqL1xuICBwcml2YXRlIF9nZXRGaXJzdEZvY3VzYWJsZSgpOiBTYmJDYWxlbmRhckNlbGxCYXNlRWxlbWVudCB8IG51bGwge1xuICAgIGlmICh0aGlzLl9jYWxlbmRhclZpZXcgPT09ICdkYXknKSB7XG4gICAgICBjb25zdCBzZWxlY3RlZE9yQ3VycmVudCA9XG4gICAgICAgIHRoaXMuX2NlbGxzLmZpbmQoKGUpID0+IGUubWF0Y2hlcygnOnN0YXRlKHNlbGVjdGVkKScpKSA/P1xuICAgICAgICB0aGlzLl9jZWxscy5maW5kKChlKSA9PiBlLm1hdGNoZXMoJzpzdGF0ZShjdXJyZW50KScpKTtcbiAgICAgIHJldHVybiBzZWxlY3RlZE9yQ3VycmVudCAmJiAhc2VsZWN0ZWRPckN1cnJlbnQuZGlzYWJsZWRcbiAgICAgICAgPyBzZWxlY3RlZE9yQ3VycmVudFxuICAgICAgICA6IHRoaXMuX2dldEZpcnN0Rm9jdXNhYmxlRGF5KCk7XG4gICAgfSBlbHNlIHtcbiAgICAgIGNvbnN0IHNlbGVjdGVkT3JDdXJyZW50ID1cbiAgICAgICAgdGhpcy5zaGFkb3dSb290Py5xdWVyeVNlbGVjdG9yPFNiYkNhbGVuZGFyQ2VsbEJhc2VFbGVtZW50PignOnN0YXRlKHNlbGVjdGVkKScpID8/XG4gICAgICAgIHRoaXMuc2hhZG93Um9vdD8ucXVlcnlTZWxlY3RvcjxTYmJDYWxlbmRhckNlbGxCYXNlRWxlbWVudD4oJzpzdGF0ZShjdXJyZW50KScpO1xuICAgICAgcmV0dXJuIHNlbGVjdGVkT3JDdXJyZW50ICYmICFzZWxlY3RlZE9yQ3VycmVudC5kaXNhYmxlZFxuICAgICAgICA/IHNlbGVjdGVkT3JDdXJyZW50XG4gICAgICAgIDogdGhpcy5zaGFkb3dSb290IS5xdWVyeVNlbGVjdG9yPFNiYkNhbGVuZGFyQ2VsbEJhc2VFbGVtZW50PihcbiAgICAgICAgICAgIGBzYmItY2FsZW5kYXItJHt0aGlzLl9jYWxlbmRhclZpZXd9Om5vdChbZGlzYWJsZWRdKWAsXG4gICAgICAgICAgKTtcbiAgICB9XG4gIH1cblxuICAvKipcbiAgICogSW4gYGRheWAgdmlldyBpbiBgdmVydGljYWxgIG9yaWVudGF0aW9uLFxuICAgKiBpZiB0aGUgZmlyc3Qgb2YgdGhlIG1vbnRoIGlzIG5vdCBhIE1vbmRheSwgaXQgaXMgbm90IHRoZSBmaXJzdCByZW5kZXJlZCBlbGVtZW50IGluIHRoZSB0YWJsZSxcbiAgICogc28gYHRoaXMuc2hhZG93Um9vdCEucXVlcnlTZWxlY3Rvcignc2JiLWNhbGVuZGFyLWRheTpub3QoW2Rpc2FibGVkXSknKWAgd2lsbCByZXR1cm4gYSB3cm9uZyB2YWx1ZS5cbiAgICpcbiAgICogVG8gc29sdmUgdGhpcywgdGhlIGVsZW1lbnQgd2l0aCB0aGUgbG93ZXN0IGB2YWx1ZWAgaXMgdGFrZW4gKElTTyBTdHJpbmcgYXJlIG9yZGVyZWQpLlxuICAgKi9cbiAgcHJpdmF0ZSBfZ2V0Rmlyc3RGb2N1c2FibGVEYXkoKTogU2JiQ2FsZW5kYXJEYXlFbGVtZW50IHwgbnVsbCB7XG4gICAgY29uc3QgY2VsbHMgPSB0aGlzLl9jZWxscyBhcyBTYmJDYWxlbmRhckRheUVsZW1lbnRbXTtcbiAgICBjb25zdCBkYXlzSW5WaWV3ID0gY2VsbHMuZmlsdGVyKChlKSA9PiAhZS5kaXNhYmxlZCk7XG4gICAgaWYgKCFkYXlzSW5WaWV3IHx8IGRheXNJblZpZXcubGVuZ3RoID09PSAwKSB7XG4gICAgICByZXR1cm4gbnVsbDtcbiAgICB9IGVsc2Uge1xuICAgICAgY29uc3QgZmlyc3RFbGVtZW50ID0gZGF5c0luVmlld1xuICAgICAgICAubWFwKChlKTogc3RyaW5nID0+IHRoaXMuX2RhdGVBZGFwdGVyLnRvSXNvODYwMShlLnZhbHVlISBhcyBUKSlcbiAgICAgICAgLnNvcnQoKVswXTtcbiAgICAgIHJldHVybiBjZWxscy5maW5kKChlKSA9PiBlLm1hdGNoZXMoYFtzbG90PVwiJHtmaXJzdEVsZW1lbnR9XCJdYCkpISA/PyBudWxsO1xuICAgIH1cbiAgfVxuXG4gIHByaXZhdGUgX2hhbmRsZUtleWJvYXJkRXZlbnQoZXZlbnQ6IEtleWJvYXJkRXZlbnQsIGRheT86IERheTxUPik6IHZvaWQge1xuICAgIGlmIChpc0Fycm93S2V5T3JQYWdlS2V5c1ByZXNzZWQoZXZlbnQpKSB7XG4gICAgICBldmVudC5wcmV2ZW50RGVmYXVsdCgpO1xuICAgIH1cbiAgICAvLyBHZXRzIHRoZSBjdXJyZW50bHkgcmVuZGVyZWQgdGFibGUncyBjZWxsO1xuICAgIC8vIHRoZXkgY291bGQgYmUgZGF5cywgbW9udGhzIG9yIHllYXJzIGJhc2VkIG9uIHRoZSBjdXJyZW50IHNlbGVjdGlvbiB2aWV3LlxuICAgIC8vIElmIGB3aWRlYCBpcyB0cnVlLCB5ZWFycyBhcmUgZG91YmxlZCBpbiBudW1iZXIgYW5kIGRheXMgYXJlIChyb3VnaGx5KSBkb3VibGVkIHRvbywgYWZmZWN0aW5nIHRoZSBgaW5kZXhgIGNhbGN1bGF0aW9uLlxuICAgIGNvbnN0IGNlbGxzID0gdGhpcy5fY2VsbHM7XG4gICAgY29uc3QgaW5kZXg6IG51bWJlciA9IGNlbGxzLmZpbmRJbmRleCgoZSkgPT4gZSA9PT0gZXZlbnQudGFyZ2V0KTtcbiAgICBsZXQgbmV4dEVsOiBTYmJDYWxlbmRhckNlbGxCYXNlRWxlbWVudDtcbiAgICBpZiAoZGF5KSB7XG4gICAgICBuZXh0RWwgPSB0aGlzLl9uYXZpZ2F0ZUJ5S2V5Ym9hcmREYXlWaWV3KGV2ZW50LCBpbmRleCwgY2VsbHMgYXMgU2JiQ2FsZW5kYXJEYXlFbGVtZW50W10sIGRheSk7XG4gICAgfSBlbHNlIHtcbiAgICAgIG5leHRFbCA9IHRoaXMuX25hdmlnYXRlQnlLZXlib2FyZChldmVudCwgaW5kZXgsIGNlbGxzKTtcbiAgICB9XG4gICAgY29uc3QgYWN0aXZlRWwgPSAodGhpcy5fZW5oYW5jZWRWYXJpYW50ID8gZG9jdW1lbnQgOiB0aGlzLnNoYWRvd1Jvb3QhKVxuICAgICAgLmFjdGl2ZUVsZW1lbnQgYXMgU2JiQ2FsZW5kYXJDZWxsQmFzZUVsZW1lbnQ7XG4gICAgaWYgKG5leHRFbCAhPT0gYWN0aXZlRWwpIHtcbiAgICAgIG5leHRFbC50YWJJbmRleCA9IDA7XG4gICAgICBuZXh0RWw/LmZvY3VzKCk7XG4gICAgICBhY3RpdmVFbC50YWJJbmRleCA9IC0xO1xuICAgIH1cbiAgfVxuXG4gIHByaXZhdGUgX25hdmlnYXRlQnlLZXlib2FyZERheVZpZXcoXG4gICAgZXZ0OiBLZXlib2FyZEV2ZW50LFxuICAgIGluZGV4OiBudW1iZXIsXG4gICAgY2VsbHM6IFNiYkNhbGVuZGFyRGF5RWxlbWVudFtdLFxuICAgIGRheTogRGF5PFQ+LFxuICApOiBTYmJDYWxlbmRhckRheUVsZW1lbnQge1xuICAgIGNvbnN0IGFycm93c09mZnNldCA9XG4gICAgICB0aGlzLm9yaWVudGF0aW9uID09PSAnaG9yaXpvbnRhbCdcbiAgICAgICAgPyB7IGxlZnRSaWdodDogMSwgdXBEb3duOiBEQVlTX1BFUl9ST1cgfVxuICAgICAgICA6IHsgbGVmdFJpZ2h0OiBEQVlTX1BFUl9ST1csIHVwRG93bjogMSB9O1xuICAgIGNvbnN0IG9mZnNldEZvclZlcnRpY2FsOiBudW1iZXIgPVxuICAgICAgaW5kZXggPCB0aGlzLl9rZXlib2FyZE5hdmlnYXRpb25EYXlWaWV3UGFyYW1ldGVycy5maXJzdE1vbnRoTGVuZ3RoXG4gICAgICAgID8gdGhpcy5fa2V5Ym9hcmROYXZpZ2F0aW9uRGF5Vmlld1BhcmFtZXRlcnMuZmlyc3RNb250aE9mZnNldFxuICAgICAgICA6IHRoaXMuX2tleWJvYXJkTmF2aWdhdGlvbkRheVZpZXdQYXJhbWV0ZXJzLnNlY29uZE1vbnRoT2Zmc2V0O1xuXG4gICAgc3dpdGNoIChldnQua2V5KSB7XG4gICAgICBjYXNlICdBcnJvd1VwJzpcbiAgICAgICAgcmV0dXJuIHRoaXMuX2ZpbmREYXlBcnJvd3MoY2VsbHMsIGluZGV4LCBkYXkuZGF0ZVZhbHVlLCAtYXJyb3dzT2Zmc2V0LnVwRG93bik7XG4gICAgICBjYXNlICdBcnJvd0Rvd24nOlxuICAgICAgICByZXR1cm4gdGhpcy5fZmluZERheUFycm93cyhjZWxscywgaW5kZXgsIGRheS5kYXRlVmFsdWUsIGFycm93c09mZnNldC51cERvd24pO1xuICAgICAgY2FzZSAnQXJyb3dMZWZ0JzpcbiAgICAgICAgcmV0dXJuIHRoaXMuX2ZpbmREYXlBcnJvd3MoY2VsbHMsIGluZGV4LCBkYXkuZGF0ZVZhbHVlLCAtYXJyb3dzT2Zmc2V0LmxlZnRSaWdodCk7XG4gICAgICBjYXNlICdBcnJvd1JpZ2h0JzpcbiAgICAgICAgcmV0dXJuIHRoaXMuX2ZpbmREYXlBcnJvd3MoY2VsbHMsIGluZGV4LCBkYXkuZGF0ZVZhbHVlLCBhcnJvd3NPZmZzZXQubGVmdFJpZ2h0KTtcbiAgICAgIGNhc2UgJ1BhZ2VVcCc6IHtcbiAgICAgICAgaWYgKHRoaXMub3JpZW50YXRpb24gPT09ICdob3Jpem9udGFsJykge1xuICAgICAgICAgIGNvbnN0IGZpcnN0T2ZXZWVrOiBudW1iZXIgPSArZGF5LmRheVZhbHVlICUgREFZU19QRVJfUk9XIHx8IERBWVNfUEVSX1JPVztcbiAgICAgICAgICBjb25zdCBkZWx0YTogbnVtYmVyID0gZmlyc3RPZldlZWsgLSArZGF5LmRheVZhbHVlO1xuICAgICAgICAgIHJldHVybiB0aGlzLl9maW5kRGF5UGFnZVVwRG93bihjZWxscywgaW5kZXgsIGRheSwgZGVsdGEsIGFycm93c09mZnNldC51cERvd24pO1xuICAgICAgICB9IGVsc2Uge1xuICAgICAgICAgIGNvbnN0IHdlZWtOdW1iZXI6IG51bWJlciA9IE1hdGguY2VpbCgoK2RheS5kYXlWYWx1ZSArIG9mZnNldEZvclZlcnRpY2FsKSAvIERBWVNfUEVSX1JPVyk7XG4gICAgICAgICAgY29uc3QgZmlyc3RPZldlZWs6IG51bWJlciA9ICh3ZWVrTnVtYmVyIC0gMSkgKiBEQVlTX1BFUl9ST1cgLSBvZmZzZXRGb3JWZXJ0aWNhbCArIDE7XG4gICAgICAgICAgY29uc3QgZGVsdGE6IG51bWJlciA9IGZpcnN0T2ZXZWVrIC0gK2RheS5kYXlWYWx1ZTtcbiAgICAgICAgICByZXR1cm4gdGhpcy5fZmluZERheVBhZ2VVcERvd24oY2VsbHMsIGluZGV4LCBkYXksIGRlbHRhLCBhcnJvd3NPZmZzZXQudXBEb3duKTtcbiAgICAgICAgfVxuICAgICAgfVxuICAgICAgY2FzZSAnUGFnZURvd24nOiB7XG4gICAgICAgIGlmICh0aGlzLm9yaWVudGF0aW9uID09PSAnaG9yaXpvbnRhbCcpIHtcbiAgICAgICAgICBjb25zdCBtb250aEluQm91bmRzID0gK2RheS5tb250aFZhbHVlICsgMSA+IDEyID8gMSA6ICtkYXkubW9udGhWYWx1ZSArIDE7XG4gICAgICAgICAgY29uc3QgeWVhckluQm91bmRzID0gK2RheS5tb250aFZhbHVlICsgMSA+IDEyID8gK2RheS55ZWFyVmFsdWUgKyAxIDogK2RheS55ZWFyVmFsdWU7XG4gICAgICAgICAgY29uc3QgZmlyc3ROZXh0TW9udGg6IFQgPSB0aGlzLl9kYXRlQWRhcHRlci5jcmVhdGVEYXRlKHllYXJJbkJvdW5kcywgbW9udGhJbkJvdW5kcywgMSk7XG4gICAgICAgICAgY29uc3QgbGFzdE9mTW9udGg6IG51bWJlciA9IHRoaXMuX2RhdGVBZGFwdGVyLmdldERhdGUoXG4gICAgICAgICAgICB0aGlzLl9kYXRlQWRhcHRlci5hZGRDYWxlbmRhckRheXMoZmlyc3ROZXh0TW9udGgsIC0xKSxcbiAgICAgICAgICApO1xuICAgICAgICAgIGNvbnN0IGRlbHRhOiBudW1iZXIgPVxuICAgICAgICAgICAgTWF0aC50cnVuYygobGFzdE9mTW9udGggLSArZGF5LmRheVZhbHVlISkgLyBEQVlTX1BFUl9ST1cpICogREFZU19QRVJfUk9XO1xuICAgICAgICAgIHJldHVybiB0aGlzLl9maW5kRGF5UGFnZVVwRG93bihjZWxscywgaW5kZXgsIGRheSwgZGVsdGEsIC1hcnJvd3NPZmZzZXQudXBEb3duKTtcbiAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICBjb25zdCB3ZWVrTnVtYmVyOiBudW1iZXIgPSBNYXRoLmNlaWwoKCtkYXkuZGF5VmFsdWUgKyBvZmZzZXRGb3JWZXJ0aWNhbCkgLyBEQVlTX1BFUl9ST1cpO1xuICAgICAgICAgIGNvbnN0IGxhc3RPZldlZWs6IG51bWJlciA9IHdlZWtOdW1iZXIgKiBEQVlTX1BFUl9ST1cgLSBvZmZzZXRGb3JWZXJ0aWNhbDtcbiAgICAgICAgICBjb25zdCBkZWx0YTogbnVtYmVyID0gbGFzdE9mV2VlayAtICtkYXkuZGF5VmFsdWU7XG4gICAgICAgICAgcmV0dXJuIHRoaXMuX2ZpbmREYXlQYWdlVXBEb3duKGNlbGxzLCBpbmRleCwgZGF5LCBkZWx0YSwgLWFycm93c09mZnNldC51cERvd24pO1xuICAgICAgICB9XG4gICAgICB9XG4gICAgICBjYXNlICdIb21lJzoge1xuICAgICAgICByZXR1cm4gdGhpcy5fZmluZERheUZpcnN0KGNlbGxzLCBpbmRleCwgZGF5LCAxKTtcbiAgICAgIH1cbiAgICAgIGNhc2UgJ0VuZCc6IHtcbiAgICAgICAgY29uc3QgbW9udGhJbkJvdW5kcyA9ICtkYXkubW9udGhWYWx1ZSArIDEgPiAxMiA/IDEgOiArZGF5Lm1vbnRoVmFsdWUgKyAxO1xuICAgICAgICBjb25zdCB5ZWFySW5Cb3VuZHMgPSArZGF5Lm1vbnRoVmFsdWUgKyAxID4gMTIgPyArZGF5LnllYXJWYWx1ZSArIDEgOiArZGF5LnllYXJWYWx1ZTtcbiAgICAgICAgY29uc3QgZmlyc3ROZXh0TW9udGg6IFQgPSB0aGlzLl9kYXRlQWRhcHRlci5jcmVhdGVEYXRlKHllYXJJbkJvdW5kcywgbW9udGhJbkJvdW5kcywgMSk7XG4gICAgICAgIHJldHVybiB0aGlzLl9maW5kRGF5TGFzdChjZWxscywgaW5kZXgsIGZpcnN0TmV4dE1vbnRoKTtcbiAgICAgIH1cbiAgICAgIGRlZmF1bHQ6XG4gICAgICAgIHJldHVybiBjZWxsc1tpbmRleF07XG4gICAgfVxuICB9XG5cbiAgcHJpdmF0ZSBfaXNEYXlPdXRPZlZpZXcoZGF0ZTogc3RyaW5nKTogYm9vbGVhbiB7XG4gICAgcmV0dXJuIChcbiAgICAgIGRhdGUgPCB0aGlzLl9rZXlib2FyZE5hdmlnYXRpb25EYXlWaWV3UGFyYW1ldGVycy5maXJzdERheUluVmlldyEgfHxcbiAgICAgIGRhdGUgPiB0aGlzLl9rZXlib2FyZE5hdmlnYXRpb25EYXlWaWV3UGFyYW1ldGVycy5sYXN0RGF5SW5WaWV3IVxuICAgICk7XG4gIH1cblxuICBwcml2YXRlIF9maW5kRGF5QXJyb3dzKFxuICAgIGNlbGxzOiBTYmJDYWxlbmRhckRheUVsZW1lbnRbXSxcbiAgICBpbmRleDogbnVtYmVyLFxuICAgIGRhdGU6IFQsXG4gICAgZGVsdGE6IG51bWJlcixcbiAgKTogU2JiQ2FsZW5kYXJEYXlFbGVtZW50IHtcbiAgICBjb25zdCBuZXdEYXRlVmFsdWUgPSB0aGlzLl9kYXRlQWRhcHRlci50b0lzbzg2MDEoXG4gICAgICB0aGlzLl9kYXRlQWRhcHRlci5hZGRDYWxlbmRhckRheXMoZGF0ZSwgZGVsdGEpLFxuICAgICk7XG4gICAgaWYgKHRoaXMuX2lzRGF5T3V0T2ZWaWV3KG5ld0RhdGVWYWx1ZSkpIHtcbiAgICAgIHJldHVybiBjZWxsc1tpbmRleF07XG4gICAgfVxuICAgIGNvbnN0IG5leHRDZWxsID0gY2VsbHMuZmluZCgoZSkgPT4gdGhpcy5fbWFwVmFsdWVUb0lTT0RhdGUoZS52YWx1ZSEpID09PSBuZXdEYXRlVmFsdWUpO1xuICAgIGlmICghbmV4dENlbGwgfHwgbmV4dENlbGwuZGlzYWJsZWQpIHtcbiAgICAgIHJldHVybiB0aGlzLl9maW5kRGF5QXJyb3dzKGNlbGxzLCBpbmRleCwgdGhpcy5fZGF0ZUFkYXB0ZXIuZGVzZXJpYWxpemUobmV3RGF0ZVZhbHVlKSEsIGRlbHRhKTtcbiAgICB9XG4gICAgcmV0dXJuIG5leHRDZWxsO1xuICB9XG5cbiAgcHJpdmF0ZSBfZmluZERheVBhZ2VVcERvd24oXG4gICAgY2VsbHM6IFNiYkNhbGVuZGFyRGF5RWxlbWVudFtdLFxuICAgIGluZGV4OiBudW1iZXIsXG4gICAgZGF5OiBEYXk8VD4sXG4gICAgZGVsdGE6IG51bWJlcixcbiAgICBkZWx0YUlmRGlzYWJsZWQ6IG51bWJlcixcbiAgKTogU2JiQ2FsZW5kYXJEYXlFbGVtZW50IHtcbiAgICBjb25zdCBuZXdEYXRlVmFsdWUgPSB0aGlzLl9kYXRlQWRhcHRlci50b0lzbzg2MDEoXG4gICAgICB0aGlzLl9kYXRlQWRhcHRlci5hZGRDYWxlbmRhckRheXMoZGF5LmRhdGVWYWx1ZSwgZGVsdGEpLFxuICAgICk7XG4gICAgaWYgKHRoaXMuX2lzRGF5T3V0T2ZWaWV3KG5ld0RhdGVWYWx1ZSkpIHtcbiAgICAgIHJldHVybiBjZWxsc1tpbmRleF07XG4gICAgfVxuICAgIGNvbnN0IG5leHRDZWxsID0gY2VsbHMuZmluZCgoZSkgPT4gdGhpcy5fbWFwVmFsdWVUb0lTT0RhdGUoZS52YWx1ZSEpID09PSBuZXdEYXRlVmFsdWUpO1xuICAgIGlmICghbmV4dENlbGwgfHwgbmV4dENlbGwuZGlzYWJsZWQpIHtcbiAgICAgIHJldHVybiB0aGlzLl9maW5kRGF5UGFnZVVwRG93bihjZWxscywgaW5kZXgsIGRheSwgZGVsdGEgKyBkZWx0YUlmRGlzYWJsZWQsIGRlbHRhSWZEaXNhYmxlZCk7XG4gICAgfVxuICAgIHJldHVybiBuZXh0Q2VsbDtcbiAgfVxuXG4gIHByaXZhdGUgX2ZpbmREYXlGaXJzdChcbiAgICBjZWxsczogU2JiQ2FsZW5kYXJEYXlFbGVtZW50W10sXG4gICAgaW5kZXg6IG51bWJlcixcbiAgICBkYXk6IERheTxUPixcbiAgICBkYXRlOiBudW1iZXIsXG4gICk6IFNiYkNhbGVuZGFyRGF5RWxlbWVudCB7XG4gICAgY29uc3QgbmV3RGF0ZVZhbHVlID0gdGhpcy5fZGF0ZUFkYXB0ZXIudG9Jc284NjAxKFxuICAgICAgdGhpcy5fZGF0ZUFkYXB0ZXIuY3JlYXRlRGF0ZSgrZGF5LnllYXJWYWx1ZSwgK2RheS5tb250aFZhbHVlLCBkYXRlKSxcbiAgICApO1xuICAgIGlmICh0aGlzLl9pc0RheU91dE9mVmlldyhuZXdEYXRlVmFsdWUpKSB7XG4gICAgICByZXR1cm4gY2VsbHNbaW5kZXhdO1xuICAgIH1cbiAgICBjb25zdCBuZXh0Q2VsbCA9IGNlbGxzLmZpbmQoKGUpID0+IHRoaXMuX21hcFZhbHVlVG9JU09EYXRlKGUudmFsdWUhKSA9PT0gbmV3RGF0ZVZhbHVlKTtcbiAgICBpZiAoIW5leHRDZWxsIHx8IG5leHRDZWxsLmRpc2FibGVkKSB7XG4gICAgICByZXR1cm4gdGhpcy5fZmluZERheUZpcnN0KGNlbGxzLCBpbmRleCwgZGF5LCBkYXRlICsgMSk7XG4gICAgfVxuICAgIHJldHVybiBuZXh0Q2VsbDtcbiAgfVxuXG4gIHByaXZhdGUgX2ZpbmREYXlMYXN0KFxuICAgIGNlbGxzOiBTYmJDYWxlbmRhckRheUVsZW1lbnRbXSxcbiAgICBpbmRleDogbnVtYmVyLFxuICAgIGZpcnN0TmV4dE1vbnRoOiBULFxuICApOiBTYmJDYWxlbmRhckRheUVsZW1lbnQge1xuICAgIGNvbnN0IG5ld0RhdGVWYWx1ZSA9IHRoaXMuX2RhdGVBZGFwdGVyLnRvSXNvODYwMShcbiAgICAgIHRoaXMuX2RhdGVBZGFwdGVyLmFkZENhbGVuZGFyRGF5cyhmaXJzdE5leHRNb250aCwgLTEpLFxuICAgICk7XG4gICAgaWYgKHRoaXMuX2lzRGF5T3V0T2ZWaWV3KG5ld0RhdGVWYWx1ZSkpIHtcbiAgICAgIHJldHVybiBjZWxsc1tpbmRleF07XG4gICAgfVxuICAgIGNvbnN0IG5leHRDZWxsID0gY2VsbHMuZmluZCgoZSkgPT4gdGhpcy5fbWFwVmFsdWVUb0lTT0RhdGUoZS52YWx1ZSEpID09PSBuZXdEYXRlVmFsdWUpO1xuICAgIGlmICghbmV4dENlbGwgfHwgbmV4dENlbGwuZGlzYWJsZWQpIHtcbiAgICAgIHJldHVybiB0aGlzLl9maW5kRGF5TGFzdChjZWxscywgaW5kZXgsIHRoaXMuX2RhdGVBZGFwdGVyLmRlc2VyaWFsaXplKG5ld0RhdGVWYWx1ZSkhKTtcbiAgICB9XG4gICAgcmV0dXJuIG5leHRDZWxsO1xuICB9XG5cbiAgLyoqXG4gICAqIEdldHMgdGhlIGluZGV4IG9mIHRoZSBlbGVtZW50IHRvIG1vdmUgdG8sIGJhc2VkIG9uIGEgbGlzdCBvZiBlbGVtZW50cyAod2hpY2ggY2FuIGJlIHBvdGVudGlhbGx5IGRpc2FibGVkKSxcbiAgICogdGhlIGtleWJvYXJkIGlucHV0IGFuZCB0aGUgcG9zaXRpb24gb2YgdGhlIGN1cnJlbnQgZWxlbWVudCBpbiB0aGUgbGlzdC5cbiAgICogSW4gdGhlIGRheSB2aWV3LCB0aGUgYGRheT86IERheWAgcGFyYW1ldGVyIGlzIG1hbmRhdG9yeSBmb3IgY2FsY3VsYXRpb24sXG4gICAqIHdoaWxlIGluIG1vbnRoIGFuZCB5ZWFyIHZpZXcgaXQncyBub3QgZHVlIHRvIHRoZSBmaXhlZCBhbW91bnQgb2YgcmVuZGVyZWQgY2VsbHMuXG4gICAqL1xuICBwcml2YXRlIF9uYXZpZ2F0ZUJ5S2V5Ym9hcmQoXG4gICAgZXZ0OiBLZXlib2FyZEV2ZW50LFxuICAgIGluZGV4OiBudW1iZXIsXG4gICAgY2VsbHM6IFNiYkNhbGVuZGFyQ2VsbEJhc2VFbGVtZW50W10sXG4gICk6IFNiYkNhbGVuZGFyQ2VsbEJhc2VFbGVtZW50IHtcbiAgICBjb25zdCB7XG4gICAgICBlbGVtZW50SW5kZXhGb3JXaWRlTW9kZSxcbiAgICAgIG9mZnNldEZvcldpZGVNb2RlLFxuICAgICAgbGFzdEVsZW1lbnRJbmRleEZvcldpZGVNb2RlLFxuICAgICAgdmVydGljYWxPZmZzZXQsXG4gICAgfTogQ2FsZW5kYXJLZXlib2FyZE5hdmlnYXRpb25Nb250aFllYXJWaWV3c1BhcmFtZXRlcnMgPVxuICAgICAgdGhpcy5fY2FsY3VsYXRlUGFyYW1ldGVyc0ZvcktleWJvYXJkTmF2aWdhdGlvbihpbmRleCwgdGhpcy5fY2FsZW5kYXJWaWV3ID09PSAneWVhcicpO1xuXG4gICAgc3dpdGNoIChldnQua2V5KSB7XG4gICAgICBjYXNlICdBcnJvd1VwJzpcbiAgICAgICAgcmV0dXJuIHRoaXMuX2ZpbmROZXh0KGNlbGxzLCBpbmRleCwgLXZlcnRpY2FsT2Zmc2V0KTtcbiAgICAgIGNhc2UgJ0Fycm93RG93bic6XG4gICAgICAgIHJldHVybiB0aGlzLl9maW5kTmV4dChjZWxscywgaW5kZXgsIHZlcnRpY2FsT2Zmc2V0KTtcbiAgICAgIGNhc2UgJ0Fycm93TGVmdCc6XG4gICAgICAgIHJldHVybiB0aGlzLl9maW5kTmV4dChjZWxscywgaW5kZXgsIC0xKTtcbiAgICAgIGNhc2UgJ0Fycm93UmlnaHQnOlxuICAgICAgICByZXR1cm4gdGhpcy5fZmluZE5leHQoY2VsbHMsIGluZGV4LCAxKTtcbiAgICAgIGNhc2UgJ0hvbWUnOlxuICAgICAgICByZXR1cm4gdGhpcy5fZmluZEZpcnN0KGNlbGxzLCBvZmZzZXRGb3JXaWRlTW9kZSk7XG4gICAgICBjYXNlICdQYWdlVXAnOlxuICAgICAgICByZXR1cm4gdGhpcy5fZmluZEZpcnN0T25Db2x1bW4oXG4gICAgICAgICAgY2VsbHMsXG4gICAgICAgICAgZWxlbWVudEluZGV4Rm9yV2lkZU1vZGUsXG4gICAgICAgICAgb2Zmc2V0Rm9yV2lkZU1vZGUsXG4gICAgICAgICAgdmVydGljYWxPZmZzZXQsXG4gICAgICAgICk7XG4gICAgICBjYXNlICdQYWdlRG93bic6XG4gICAgICAgIHJldHVybiB0aGlzLl9maW5kTGFzdE9uQ29sdW1uKGNlbGxzLCBpbmRleCwgbGFzdEVsZW1lbnRJbmRleEZvcldpZGVNb2RlLCB2ZXJ0aWNhbE9mZnNldCk7XG4gICAgICBjYXNlICdFbmQnOlxuICAgICAgICByZXR1cm4gdGhpcy5fZmluZExhc3QoY2VsbHMsIGxhc3RFbGVtZW50SW5kZXhGb3JXaWRlTW9kZSAtIDEpO1xuICAgICAgZGVmYXVsdDpcbiAgICAgICAgcmV0dXJuIGNlbGxzW2luZGV4XTtcbiAgICB9XG4gIH1cblxuICAvKipcbiAgICogQ2FsY3VsYXRlcyB0aGUgcGFyYW1ldGVycyBuZWVkZWQgaW4ga2V5Ym9hcmQgbmF2aWdhdGlvbiBpbiB5ZWFyIGFuZCBtb250aCB2aWV3LlxuICAgKiBAcGFyYW0gaW5kZXggVGhlIHN0YXJ0aW5nIGVsZW1lbnQncyBpbmRleCBpbiB0aGUgY2VsbCBhcnJheS5cbiAgICogQHBhcmFtIGlzWWVhclZpZXcgV2hldGhlciB0aGUgZGlzcGxheWVkIGB2aWV3YCBpcyB0aGUgeWVhciBvbmUuXG4gICAqL1xuICBwcml2YXRlIF9jYWxjdWxhdGVQYXJhbWV0ZXJzRm9yS2V5Ym9hcmROYXZpZ2F0aW9uKFxuICAgIGluZGV4OiBudW1iZXIsXG4gICAgaXNZZWFyVmlldzogYm9vbGVhbixcbiAgKTogQ2FsZW5kYXJLZXlib2FyZE5hdmlnYXRpb25Nb250aFllYXJWaWV3c1BhcmFtZXRlcnMge1xuICAgIGNvbnN0IGVsZW1lbnRzUGVyUGFnZSA9IGlzWWVhclZpZXcgPyBZRUFSU19QRVJfUEFHRSA6IE1PTlRIU19QRVJfUEFHRTtcbiAgICBjb25zdCBvZmZzZXQ6IG51bWJlciA9IE1hdGgudHJ1bmMoaW5kZXggLyBlbGVtZW50c1BlclBhZ2UpICogZWxlbWVudHNQZXJQYWdlO1xuICAgIGNvbnN0IGluZGV4SW5WaWV3OiBudW1iZXIgPSBvZmZzZXQgPT09IDAgPyBpbmRleCA6IGluZGV4IC0gZWxlbWVudHNQZXJQYWdlO1xuICAgIHJldHVybiB7XG4gICAgICB2ZXJ0aWNhbE9mZnNldDogaXNZZWFyVmlldyA/IFlFQVJTX1BFUl9ST1cgOiBNT05USFNfUEVSX1JPVyxcbiAgICAgIGVsZW1lbnRJbmRleEZvcldpZGVNb2RlOiBpbmRleEluVmlldyxcbiAgICAgIG9mZnNldEZvcldpZGVNb2RlOiBpbmRleCAtIGluZGV4SW5WaWV3LFxuICAgICAgbGFzdEVsZW1lbnRJbmRleEZvcldpZGVNb2RlOiBvZmZzZXQgPT09IDAgPyBlbGVtZW50c1BlclBhZ2UgOiBlbGVtZW50c1BlclBhZ2UgKiAyLFxuICAgIH07XG4gIH1cblxuICAvKipcbiAgICogR2V0cyB0aGUgbmV4dCBlbGVtZW50IG9mIHRoZSBwcm92aWRlZCBhcnJheSBzdGFydGluZyBmcm9tIGBpbmRleGAgYnkgYWRkaW5nIGBkZWx0YWAuXG4gICAqIElmIHRoZSBmb3VuZCBlbGVtZW50IGlzIGRpc2FibGVkLCBpdCBjb250aW51ZXMgYWRkaW5nIGBkZWx0YWAgdW50aWwgaXQgZmluZHMgYW4gZW5hYmxlZCBvbmUgaW4gdGhlIGFycmF5IGJvdW5kcy5cbiAgICovXG4gIHByaXZhdGUgX2ZpbmROZXh0KFxuICAgIGRheXM6IFNiYkNhbGVuZGFyQ2VsbEJhc2VFbGVtZW50W10sXG4gICAgaW5kZXg6IG51bWJlcixcbiAgICBkZWx0YTogbnVtYmVyLFxuICApOiBTYmJDYWxlbmRhckNlbGxCYXNlRWxlbWVudCB7XG4gICAgbGV0IG5leHRJbmRleCA9IGluZGV4ICsgZGVsdGE7XG4gICAgd2hpbGUgKG5leHRJbmRleCA8IGRheXMubGVuZ3RoICYmIGRheXNbbmV4dEluZGV4XT8uZGlzYWJsZWQpIHtcbiAgICAgIG5leHRJbmRleCArPSBkZWx0YTtcbiAgICB9XG4gICAgcmV0dXJuIGRheXNbbmV4dEluZGV4XSA/PyBkYXlzW2luZGV4XTtcbiAgfVxuXG4gIC8qKiBGaW5kIHRoZSBmaXJzdCBlbmFibGVkIGVsZW1lbnQgaW4gdGhlIHByb3ZpZGVkIGFycmF5LiAqL1xuICBwcml2YXRlIF9maW5kRmlyc3QoXG4gICAgZGF5czogU2JiQ2FsZW5kYXJDZWxsQmFzZUVsZW1lbnRbXSxcbiAgICBmaXJzdE9mQ3VycmVudE1vbnRoOiBudW1iZXIsXG4gICk6IFNiYkNhbGVuZGFyQ2VsbEJhc2VFbGVtZW50IHtcbiAgICByZXR1cm4gIWRheXNbZmlyc3RPZkN1cnJlbnRNb250aF0uZGlzYWJsZWRcbiAgICAgID8gZGF5c1tmaXJzdE9mQ3VycmVudE1vbnRoXVxuICAgICAgOiB0aGlzLl9maW5kTmV4dChkYXlzLCBmaXJzdE9mQ3VycmVudE1vbnRoLCAxKTtcbiAgfVxuXG4gIC8qKiBGaW5kIHRoZSBsYXN0IGVuYWJsZWQgZWxlbWVudCBpbiB0aGUgcHJvdmlkZWQgYXJyYXkuICovXG4gIHByaXZhdGUgX2ZpbmRMYXN0KFxuICAgIGRheXM6IFNiYkNhbGVuZGFyQ2VsbEJhc2VFbGVtZW50W10sXG4gICAgbGFzdE9mQ3VycmVudE1vbnRoOiBudW1iZXIsXG4gICk6IFNiYkNhbGVuZGFyQ2VsbEJhc2VFbGVtZW50IHtcbiAgICByZXR1cm4gIWRheXNbbGFzdE9mQ3VycmVudE1vbnRoXS5kaXNhYmxlZFxuICAgICAgPyBkYXlzW2xhc3RPZkN1cnJlbnRNb250aF1cbiAgICAgIDogdGhpcy5fZmluZE5leHQoZGF5cywgbGFzdE9mQ3VycmVudE1vbnRoLCAtMSk7XG4gIH1cblxuICAvKiogRmluZCB0aGUgZmlyc3QgZW5hYmxlZCBlbGVtZW50IGluIHRoZSBzYW1lIGNvbHVtbiBvZiB0aGUgcHJvdmlkZWQgYXJyYXkuICovXG4gIHByaXZhdGUgX2ZpbmRGaXJzdE9uQ29sdW1uKFxuICAgIGRheXM6IFNiYkNhbGVuZGFyQ2VsbEJhc2VFbGVtZW50W10sXG4gICAgaW5kZXg6IG51bWJlcixcbiAgICBvZmZzZXQ6IG51bWJlcixcbiAgICB2ZXJ0aWNhbE9mZnNldDogbnVtYmVyLFxuICApOiBTYmJDYWxlbmRhckNlbGxCYXNlRWxlbWVudCB7XG4gICAgY29uc3QgbmV4dEluZGV4ID0gKGluZGV4ICUgdmVydGljYWxPZmZzZXQpICsgb2Zmc2V0O1xuICAgIHJldHVybiAhZGF5c1tuZXh0SW5kZXhdLmRpc2FibGVkXG4gICAgICA/IGRheXNbbmV4dEluZGV4XVxuICAgICAgOiB0aGlzLl9maW5kTmV4dChkYXlzLCBuZXh0SW5kZXgsIHZlcnRpY2FsT2Zmc2V0KTtcbiAgfVxuXG4gIC8qKiBGaW5kIHRoZSBsYXN0IGVuYWJsZWQgZWxlbWVudCBpbiB0aGUgc2FtZSBjb2x1bW4gb2YgdGhlIHByb3ZpZGVkIGFycmF5LiAqL1xuICBwcml2YXRlIF9maW5kTGFzdE9uQ29sdW1uKFxuICAgIGRheXM6IFNiYkNhbGVuZGFyQ2VsbEJhc2VFbGVtZW50W10sXG4gICAgaW5kZXg6IG51bWJlcixcbiAgICBvZmZzZXQ6IG51bWJlcixcbiAgICB2ZXJ0aWNhbE9mZnNldDogbnVtYmVyLFxuICApOiBTYmJDYWxlbmRhckNlbGxCYXNlRWxlbWVudCB7XG4gICAgY29uc3QgbmV4dEluZGV4ID0gaW5kZXggKyBNYXRoLnRydW5jKChvZmZzZXQgLSBpbmRleCAtIDEpIC8gdmVydGljYWxPZmZzZXQpICogdmVydGljYWxPZmZzZXQ7XG4gICAgcmV0dXJuICFkYXlzW25leHRJbmRleF0uZGlzYWJsZWRcbiAgICAgID8gZGF5c1tuZXh0SW5kZXhdXG4gICAgICA6IHRoaXMuX2ZpbmROZXh0KGRheXMsIG5leHRJbmRleCwgLXZlcnRpY2FsT2Zmc2V0KTtcbiAgfVxuXG4gIHByaXZhdGUgX3Jlc2V0Q2FsZW5kYXJWaWV3QW5kRW1pdE1vbnRoQ2hhbmdlKGluaXRUcmFuc2l0aW9uID0gZmFsc2UpOiB2b2lkIHtcbiAgICB0aGlzLl9yZXNldENhbGVuZGFyVmlldyhpbml0VHJhbnNpdGlvbik7XG4gICAgdGhpcy5fZW1pdE1vbnRoQ2hhbmdlKCk7XG4gIH1cblxuICBwcml2YXRlIF9yZXNldENhbGVuZGFyVmlldyhpbml0VHJhbnNpdGlvbiA9IGZhbHNlKTogdm9pZCB7XG4gICAgaWYgKHRoaXMuX2NvbnRhaW5pbmdGb2N1cykge1xuICAgICAgdGhpcy5fcmVzZXRGb2N1cyA9IHRydWU7XG4gICAgfVxuICAgIHRoaXMuX2FjdGl2ZURhdGUgPVxuICAgICAgKHRoaXMubXVsdGlwbGUgPyAodGhpcy5fc2VsZWN0ZWQgYXMgVFtdKS5hdCgtMSkgOiAodGhpcy5fc2VsZWN0ZWQgYXMgVCkpID8/XG4gICAgICB0aGlzLl9kYXRlQWRhcHRlci50b2RheSgpO1xuICAgIHRoaXMuX3NldENob3NlblllYXIoKTtcbiAgICB0aGlzLl9jaG9zZW5Nb250aCA9IHVuZGVmaW5lZDtcbiAgICB0aGlzLl9pbml0KCk7XG4gICAgdGhpcy5fbmV4dENhbGVuZGFyVmlldyA9IHRoaXMuX2NhbGVuZGFyVmlldyA9IHRoaXMudmlldztcblxuICAgIGlmIChpbml0VHJhbnNpdGlvbikge1xuICAgICAgdGhpcy5fc3RhcnRUYWJsZVRyYW5zaXRpb24oKTtcbiAgICB9XG4gIH1cblxuICAvKiogUmVuZGVyIHRoZSB2aWV3IGZvciB0aGUgZGF5IHNlbGVjdGlvbi4gKi9cbiAgcHJpdmF0ZSBfcmVuZGVyRGF5VmlldygpOiBUZW1wbGF0ZVJlc3VsdCB7XG4gICAgY29uc3QgbmV4dE1vbnRoQWN0aXZlRGF0ZSA9IHRoaXMuX3dpZGVcbiAgICAgID8gdGhpcy5fZGF0ZUFkYXB0ZXIuYWRkQ2FsZW5kYXJNb250aHModGhpcy5fYWN0aXZlRGF0ZSwgMSlcbiAgICAgIDogdW5kZWZpbmVkO1xuICAgIHJldHVybiBodG1sYFxuICAgICAgPGRpdiBjbGFzcz1cInNiYi1jYWxlbmRhcl9fY29udHJvbHNcIj5cbiAgICAgICAgJHt0aGlzLl9nZXRBcnJvdyhcbiAgICAgICAgICAnbGVmdCcsXG4gICAgICAgICAgKCkgPT4gdGhpcy5fZ29Ub0RpZmZlcmVudE1vbnRoKC0xKSxcbiAgICAgICAgICBpMThuUHJldmlvdXNNb250aFt0aGlzLl9sYW5ndWFnZS5jdXJyZW50XSxcbiAgICAgICAgICB0aGlzLl9wcmV2aW91c01vbnRoRGlzYWJsZWQoKSxcbiAgICAgICAgKX1cbiAgICAgICAgPGRpdiBjbGFzcz1cInNiYi1jYWxlbmRhcl9fY29udHJvbHMtbW9udGhcIj5cbiAgICAgICAgICAke3RoaXMuX2NyZWF0ZUxhYmVsRm9yRGF5Vmlldyh0aGlzLl9hY3RpdmVEYXRlKX1cbiAgICAgICAgICAke3RoaXMuX3dpZGUgPyB0aGlzLl9jcmVhdGVMYWJlbEZvckRheVZpZXcobmV4dE1vbnRoQWN0aXZlRGF0ZSEpIDogbm90aGluZ31cbiAgICAgICAgICA8c2JiLXNjcmVlbi1yZWFkZXItb25seSByb2xlPVwic3RhdHVzXCI+XG4gICAgICAgICAgICAke3RoaXMuX2NyZWF0ZUFyaWFMYWJlbEZvckRheVZpZXcodGhpcy5fYWN0aXZlRGF0ZSwgbmV4dE1vbnRoQWN0aXZlRGF0ZSEpfVxuICAgICAgICAgIDwvc2JiLXNjcmVlbi1yZWFkZXItb25seT5cbiAgICAgICAgPC9kaXY+XG4gICAgICAgICR7dGhpcy5fZ2V0QXJyb3coXG4gICAgICAgICAgJ3JpZ2h0JyxcbiAgICAgICAgICAoKSA9PiB0aGlzLl9nb1RvRGlmZmVyZW50TW9udGgoMSksXG4gICAgICAgICAgaTE4bk5leHRNb250aFt0aGlzLl9sYW5ndWFnZS5jdXJyZW50XSxcbiAgICAgICAgICB0aGlzLl9uZXh0TW9udGhEaXNhYmxlZCgpLFxuICAgICAgICApfVxuICAgICAgPC9kaXY+XG4gICAgICA8ZGl2IGNsYXNzPVwic2JiLWNhbGVuZGFyX190YWJsZS1vdmVyZmxvdy1icmVha1wiPlxuICAgICAgICA8ZGl2IGNsYXNzPVwic2JiLWNhbGVuZGFyX190YWJsZS1jb250YWluZXIgc2JiLWNhbGVuZGFyX190YWJsZS1kYXktdmlld1wiPlxuICAgICAgICAgICR7dGhpcy5vcmllbnRhdGlvbiA9PT0gJ2hvcml6b250YWwnXG4gICAgICAgICAgICA/IGh0bWxgXG4gICAgICAgICAgICAgICAgJHt0aGlzLl9jcmVhdGVEYXlUYWJsZSh0aGlzLl93ZWVrcywgdGhpcy5fd2Vla051bWJlcnMpfVxuICAgICAgICAgICAgICAgICR7dGhpcy5fd2lkZVxuICAgICAgICAgICAgICAgICAgPyB0aGlzLl9jcmVhdGVEYXlUYWJsZSh0aGlzLl9uZXh0TW9udGhXZWVrcywgdGhpcy5fbmV4dE1vbnRoV2Vla051bWJlcnMsIHRydWUpXG4gICAgICAgICAgICAgICAgICA6IG5vdGhpbmd9XG4gICAgICAgICAgICAgIGBcbiAgICAgICAgICAgIDogaHRtbGBcbiAgICAgICAgICAgICAgICAke3RoaXMuX2NyZWF0ZURheVRhYmxlVmVydGljYWwodGhpcy5fd2Vla3MsIHRoaXMuX3dlZWtOdW1iZXJzKX1cbiAgICAgICAgICAgICAgICAke3RoaXMuX3dpZGVcbiAgICAgICAgICAgICAgICAgID8gdGhpcy5fY3JlYXRlRGF5VGFibGVWZXJ0aWNhbChcbiAgICAgICAgICAgICAgICAgICAgICB0aGlzLl9uZXh0TW9udGhXZWVrcyxcbiAgICAgICAgICAgICAgICAgICAgICB0aGlzLl9uZXh0TW9udGhXZWVrTnVtYmVycyxcbiAgICAgICAgICAgICAgICAgICAgICBuZXh0TW9udGhBY3RpdmVEYXRlLFxuICAgICAgICAgICAgICAgICAgICApXG4gICAgICAgICAgICAgICAgICA6IG5vdGhpbmd9XG4gICAgICAgICAgICAgIGB9XG4gICAgICAgIDwvZGl2PlxuICAgICAgPC9kaXY+XG4gICAgYDtcbiAgfVxuXG4gIC8qKiBDcmVhdGVzIHRoZSBsYWJlbCB3aXRoIHRoZSBtb250aCBmb3IgdGhlIGRhaWx5IHZpZXcuICovXG4gIHByaXZhdGUgX2NyZWF0ZUxhYmVsRm9yRGF5VmlldyhkOiBUKTogVGVtcGxhdGVSZXN1bHQge1xuICAgIGNvbnN0IG1vbnRoTGFiZWwgPSBgJHtcbiAgICAgIHRoaXMuX21vbnRoTmFtZXNbdGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0TW9udGgoZCkgLSAxXVxuICAgIH0gJHt0aGlzLl9kYXRlQWRhcHRlci5nZXRZZWFyKGQpfWA7XG4gICAgcmV0dXJuIGh0bWxgXG4gICAgICA8YnV0dG9uXG4gICAgICAgIHR5cGU9XCJidXR0b25cIlxuICAgICAgICBjbGFzcz1cInNiYi1jYWxlbmRhcl9fZGF0ZS1zZWxlY3Rpb24gc2JiLWNhbGVuZGFyX19jb250cm9scy1jaGFuZ2UtZGF0ZVwiXG4gICAgICAgIGFyaWEtbGFiZWw9XCIke2kxOG5ZZWFyTW9udGhTZWxlY3Rpb25bdGhpcy5fbGFuZ3VhZ2UuY3VycmVudF19ICR7bW9udGhMYWJlbH1cIlxuICAgICAgICBAY2xpY2s9JHsoKSA9PiB7XG4gICAgICAgICAgdGhpcy5fcmVzZXRGb2N1cyA9IHRydWU7XG4gICAgICAgICAgdGhpcy5fbmV4dENhbGVuZGFyVmlldyA9ICd5ZWFyJztcbiAgICAgICAgICB0aGlzLl9zdGFydFRhYmxlVHJhbnNpdGlvbigpO1xuICAgICAgICB9fVxuICAgICAgPlxuICAgICAgICAke21vbnRoTGFiZWx9XG4gICAgICAgIDxzYmItaWNvbiBuYW1lPVwiY2hldnJvbi1zbWFsbC1kb3duLXNtYWxsXCI+PC9zYmItaWNvbj5cbiAgICAgIDwvYnV0dG9uPlxuICAgIGA7XG4gIH1cblxuICAvKiogQ3JlYXRlcyB0aGUgYXJpYS1sYWJlbCBmb3IgdGhlIGRhaWx5IHZpZXcuICovXG4gIHByaXZhdGUgX2NyZWF0ZUFyaWFMYWJlbEZvckRheVZpZXcoLi4uZGF0ZXM6IFRbXSk6IHN0cmluZyB7XG4gICAgbGV0IG1vbnRoTGFiZWwgPSAnJztcbiAgICBmb3IgKGNvbnN0IGQgb2YgZGF0ZXMpIHtcbiAgICAgIGlmIChkKSB7XG4gICAgICAgIG1vbnRoTGFiZWwgKz0gYCR7XG4gICAgICAgICAgdGhpcy5fbW9udGhOYW1lc1t0aGlzLl9kYXRlQWRhcHRlci5nZXRNb250aChkKSAtIDFdXG4gICAgICAgIH0gJHt0aGlzLl9kYXRlQWRhcHRlci5nZXRZZWFyKGQpfSBgO1xuICAgICAgfVxuICAgIH1cbiAgICByZXR1cm4gbW9udGhMYWJlbDtcbiAgfVxuXG4gIC8qKiBDcmVhdGVzIHRoZSBjYWxlbmRhciB0YWJsZSBmb3IgdGhlIGRhaWx5IHZpZXcuICovXG4gIHByaXZhdGUgX2NyZWF0ZURheVRhYmxlKFxuICAgIHdlZWtzOiBEYXk8VD5bXVtdLFxuICAgIHdlZWtOdW1iZXJzOiBudW1iZXJbXSxcbiAgICBpc1dpZGVOZXh0TW9udGg6IGJvb2xlYW4gPSBmYWxzZSxcbiAgKTogVGVtcGxhdGVSZXN1bHQge1xuICAgIGNvbnN0IHdlZWtzRm9yU2VsZWN0TXVsdGlwbGVXZWVrTnVtYmVyczogRGF5PFQ+W10gPSAoXG4gICAgICB0aGlzLl93aWRlXG4gICAgICAgID8gWy4uLnRoaXMuX3dlZWtzLCAuLi50aGlzLl9uZXh0TW9udGhXZWVrc11cbiAgICAgICAgOiBpc1dpZGVOZXh0TW9udGhcbiAgICAgICAgICA/IHRoaXMuX25leHRNb250aFdlZWtzXG4gICAgICAgICAgOiB0aGlzLl93ZWVrc1xuICAgICkuZmxhdCgpO1xuICAgIGNvbnN0IHdlZWtzRm9yU2VsZWN0TXVsdGlwbGVXZWVrRGF5czogRGF5PFQ+W10gPSAoXG4gICAgICBpc1dpZGVOZXh0TW9udGggPyB0aGlzLl9uZXh0TW9udGhXZWVrcyA6IHRoaXMuX3dlZWtzXG4gICAgKS5mbGF0KCk7XG4gICAgcmV0dXJuIGh0bWxgXG4gICAgICA8dGFibGVcbiAgICAgICAgY2xhc3M9XCJzYmItY2FsZW5kYXJfX3RhYmxlXCJcbiAgICAgICAgQGZvY3Vzb3V0PSR7KGV2ZW50OiBGb2N1c0V2ZW50KSA9PlxuICAgICAgICAgIHRoaXMuX2hhbmRsZVRhYmxlQmx1cihldmVudC5yZWxhdGVkVGFyZ2V0IGFzIEhUTUxFbGVtZW50KX1cbiAgICAgICAgQGFuaW1hdGlvbmVuZD0keyhlOiBBbmltYXRpb25FdmVudCkgPT4gdGhpcy5fdGFibGVBbmltYXRpb25FbmQoZSl9XG4gICAgICA+XG4gICAgICAgIDx0aGVhZCBjbGFzcz1cInNiYi1jYWxlbmRhcl9fdGFibGUtaGVhZGVyXCI+XG4gICAgICAgICAgPHRyPlxuICAgICAgICAgICAgJHt0aGlzLndlZWtOdW1iZXJzID8gaHRtbGA8dGggY2xhc3M9XCJzYmItY2FsZW5kYXJfX3RhYmxlLWhlYWRlci1jZWxsXCI+PC90aD5gIDogbm90aGluZ31cbiAgICAgICAgICAgICR7dGhpcy5fd2Vla2RheXMubWFwKFxuICAgICAgICAgICAgICAod2Vla0RheTogV2Vla2RheSwgaW5kZXg6IG51bWJlcikgPT4gaHRtbGBcbiAgICAgICAgICAgICAgICA8dGggY2xhc3M9XCJzYmItY2FsZW5kYXJfX3RhYmxlLWhlYWRlci1jZWxsXCI+XG4gICAgICAgICAgICAgICAgICAke3RoaXMubXVsdGlwbGVcbiAgICAgICAgICAgICAgICAgICAgPyBodG1sYFxuICAgICAgICAgICAgICAgICAgICAgICAgPHNiYi1jYWxlbmRhci13ZWVrZGF5XG4gICAgICAgICAgICAgICAgICAgICAgICAgIC52YWx1ZT0ke3dlZWtEYXl9XG4gICAgICAgICAgICAgICAgICAgICAgICAgIEBjbGljaz0keygpID0+IHtcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAvLyBOT1RFOiBTdW5kYXlzIGhhdmUgaW5kZXggNywgd2hpbGUgdGhlaXIgd2Vla0RheVZhbHVlIGlzIDBcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICBjb25zdCBkYXlzOiBEYXk8VD5bXSA9IHdlZWtzRm9yU2VsZWN0TXVsdGlwbGVXZWVrRGF5cy5maWx0ZXIoXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAoZGF5OiBEYXk8VD4pID0+IGRheS53ZWVrRGF5VmFsdWUgPT09IChpbmRleCArIDEpICUgNyxcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICApITtcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICB0aGlzLl9zZWxlY3RNdWx0aXBsZURhdGVzKGRheXMpO1xuICAgICAgICAgICAgICAgICAgICAgICAgICB9fVxuICAgICAgICAgICAgICAgICAgICAgICAgPjwvc2JiLWNhbGVuZGFyLXdlZWtkYXk+XG4gICAgICAgICAgICAgICAgICAgICAgYFxuICAgICAgICAgICAgICAgICAgICA6IGh0bWxgXG4gICAgICAgICAgICAgICAgICAgICAgICA8c2JiLXNjcmVlbi1yZWFkZXItb25seT4ke3dlZWtEYXkubG9uZ308L3NiYi1zY3JlZW4tcmVhZGVyLW9ubHk+XG4gICAgICAgICAgICAgICAgICAgICAgICA8c3BhbiBhcmlhLWhpZGRlbj1cInRydWVcIj4ke3dlZWtEYXkubmFycm93fTwvc3Bhbj5cbiAgICAgICAgICAgICAgICAgICAgICBgfVxuICAgICAgICAgICAgICAgIDwvdGg+XG4gICAgICAgICAgICAgIGAsXG4gICAgICAgICAgICApfVxuICAgICAgICAgIDwvdHI+XG4gICAgICAgIDwvdGhlYWQ+XG4gICAgICAgIDx0Ym9keSBjbGFzcz1cInNiYi1jYWxlbmRhcl9fdGFibGUtYm9keVwiPlxuICAgICAgICAgICR7d2Vla3MubWFwKCh3ZWVrOiBEYXk8VD5bXSwgcm93SW5kZXg6IG51bWJlcikgPT4ge1xuICAgICAgICAgICAgY29uc3QgZmlyc3RSb3dPZmZzZXQ6IG51bWJlciA9IERBWVNfUEVSX1JPVyAtIHdlZWsubGVuZ3RoO1xuICAgICAgICAgICAgaWYgKHJvd0luZGV4ID09PSAwICYmIGZpcnN0Um93T2Zmc2V0KSB7XG4gICAgICAgICAgICAgIHJldHVybiBodG1sYFxuICAgICAgICAgICAgICAgIDx0cj5cbiAgICAgICAgICAgICAgICAgICR7dGhpcy53ZWVrTnVtYmVyc1xuICAgICAgICAgICAgICAgICAgICA/IGh0bWxgXG4gICAgICAgICAgICAgICAgICAgICAgICA8dGQgY2xhc3M9XCJzYmItY2FsZW5kYXJfX3RhYmxlLWhlYWRlci1jZWxsLXZlcnRpY2FsXCI+XG4gICAgICAgICAgICAgICAgICAgICAgICAgICR7dGhpcy5tdWx0aXBsZVxuICAgICAgICAgICAgICAgICAgICAgICAgICAgID8gaHRtbGBcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPHNiYi1jYWxlbmRhci13ZWVrbnVtYmVyXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgLnZhbHVlPSR7d2Vla051bWJlcnNbMF19XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgQGNsaWNrPSR7KCkgPT4ge1xuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgY29uc3QgZGF5czogRGF5PFQ+W10gPSB3ZWVrc0ZvclNlbGVjdE11bHRpcGxlV2Vla051bWJlcnMuZmlsdGVyKFxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAoZGF5OiBEYXk8VD4pID0+IGRheS53ZWVrVmFsdWUgPT09IHdlZWtOdW1iZXJzWzBdLFxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgKSE7XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB0aGlzLl9zZWxlY3RNdWx0aXBsZURhdGVzKGRheXMpO1xuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIH19XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID48L3NiYi1jYWxlbmRhci13ZWVrbnVtYmVyPlxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgYFxuICAgICAgICAgICAgICAgICAgICAgICAgICAgIDogaHRtbGBcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPHNiYi1zY3JlZW4tcmVhZGVyLW9ubHlcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICA+JHtgJHtpMThuQ2FsZW5kYXJXZWVrTnVtYmVyW3RoaXMuX2xhbmd1YWdlLmN1cnJlbnRdfSAke3dlZWtOdW1iZXJzWzBdfWB9PC9zYmItc2NyZWVuLXJlYWRlci1vbmx5XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID5cbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPHNwYW4gYXJpYS1oaWRkZW49XCJ0cnVlXCI+JHt3ZWVrTnVtYmVyc1swXX08L3NwYW4+XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICBgfVxuICAgICAgICAgICAgICAgICAgICAgICAgPC90ZD5cbiAgICAgICAgICAgICAgICAgICAgICBgXG4gICAgICAgICAgICAgICAgICAgIDogbm90aGluZ31cbiAgICAgICAgICAgICAgICAgICR7Wy4uLkFycmF5KGZpcnN0Um93T2Zmc2V0KS5rZXlzKCldLm1hcChcbiAgICAgICAgICAgICAgICAgICAgKCkgPT4gaHRtbGA8dGQgY2xhc3M9XCJzYmItY2FsZW5kYXJfX3RhYmxlLWRhdGFcIj48L3RkPmAsXG4gICAgICAgICAgICAgICAgICApfVxuICAgICAgICAgICAgICAgICAgJHt0aGlzLl9jcmVhdGVEYXlDZWxscyh3ZWVrKX1cbiAgICAgICAgICAgICAgICA8L3RyPlxuICAgICAgICAgICAgICBgO1xuICAgICAgICAgICAgfVxuICAgICAgICAgICAgcmV0dXJuIGh0bWxgXG4gICAgICAgICAgICAgIDx0cj5cbiAgICAgICAgICAgICAgICAke3RoaXMud2Vla051bWJlcnNcbiAgICAgICAgICAgICAgICAgID8gaHRtbGBcbiAgICAgICAgICAgICAgICAgICAgICA8dGQgY2xhc3M9XCJzYmItY2FsZW5kYXJfX3RhYmxlLWhlYWRlci1jZWxsLXZlcnRpY2FsXCI+XG4gICAgICAgICAgICAgICAgICAgICAgICAke3RoaXMubXVsdGlwbGVcbiAgICAgICAgICAgICAgICAgICAgICAgICAgPyBodG1sYFxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPHNiYi1jYWxlbmRhci13ZWVrbnVtYmVyXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIC52YWx1ZT0ke3dlZWtOdW1iZXJzW3Jvd0luZGV4XX1cbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgQGNsaWNrPSR7KCkgPT4ge1xuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNvbnN0IGRheXM6IERheTxUPltdID0gd2Vla3NGb3JTZWxlY3RNdWx0aXBsZVdlZWtOdW1iZXJzLmZpbHRlcihcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIChkYXk6IERheTxUPikgPT4gZGF5LndlZWtWYWx1ZSA9PT0gd2Vla051bWJlcnNbcm93SW5kZXhdLFxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICkhO1xuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHRoaXMuX3NlbGVjdE11bHRpcGxlRGF0ZXMoZGF5cyk7XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIH19XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICA+PC9zYmItY2FsZW5kYXItd2Vla251bWJlcj5cbiAgICAgICAgICAgICAgICAgICAgICAgICAgICBgXG4gICAgICAgICAgICAgICAgICAgICAgICAgIDogaHRtbGBcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIDxzYmItc2NyZWVuLXJlYWRlci1vbmx5XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID4ke2Ake2kxOG5DYWxlbmRhcldlZWtOdW1iZXJbdGhpcy5fbGFuZ3VhZ2UuY3VycmVudF19ICR7d2Vla051bWJlcnNbcm93SW5kZXhdfWB9PC9zYmItc2NyZWVuLXJlYWRlci1vbmx5XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICA+XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICA8c3BhbiBhcmlhLWhpZGRlbj1cInRydWVcIj4ke3dlZWtOdW1iZXJzW3Jvd0luZGV4XX08L3NwYW4+XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgYH1cbiAgICAgICAgICAgICAgICAgICAgICA8L3RkPlxuICAgICAgICAgICAgICAgICAgICBgXG4gICAgICAgICAgICAgICAgICA6IG5vdGhpbmd9XG4gICAgICAgICAgICAgICAgJHt0aGlzLl9jcmVhdGVEYXlDZWxscyh3ZWVrKX1cbiAgICAgICAgICAgICAgPC90cj5cbiAgICAgICAgICAgIGA7XG4gICAgICAgICAgfSl9XG4gICAgICAgIDwvdGJvZHk+XG4gICAgICA8L3RhYmxlPlxuICAgIGA7XG4gIH1cblxuICAvKiogQ3JlYXRlcyB0aGUgdGFibGUgaW4gb3JpZW50YXRpb249J3ZlcnRpY2FsJy4gKi9cbiAgcHJpdmF0ZSBfY3JlYXRlRGF5VGFibGVWZXJ0aWNhbChcbiAgICB3ZWVrczogRGF5PFQ+W11bXSxcbiAgICB3ZWVrTnVtYmVyczogbnVtYmVyW10sXG4gICAgbmV4dE1vbnRoQWN0aXZlRGF0ZT86IFQsXG4gICk6IFRlbXBsYXRlUmVzdWx0IHtcbiAgICBjb25zdCB3ZWVrT2Zmc2V0ID0gdGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0Rmlyc3RXZWVrT2Zmc2V0KFxuICAgICAgbmV4dE1vbnRoQWN0aXZlRGF0ZSA/PyB0aGlzLl9hY3RpdmVEYXRlLFxuICAgICk7XG4gICAgY29uc3Qgd2Vla3NGb3JTZWxlY3RNdWx0aXBsZVdlZWtOdW1iZXJzOiBEYXk8VD5bXSA9IChcbiAgICAgIHRoaXMuX3dpZGVcbiAgICAgICAgPyBbLi4udGhpcy5fd2Vla3MsIC4uLnRoaXMuX25leHRNb250aFdlZWtzXVxuICAgICAgICA6IG5leHRNb250aEFjdGl2ZURhdGVcbiAgICAgICAgICA/IHRoaXMuX25leHRNb250aFdlZWtzXG4gICAgICAgICAgOiB0aGlzLl93ZWVrc1xuICAgICkuZmxhdCgpO1xuICAgIHJldHVybiBodG1sYFxuICAgICAgPHRhYmxlXG4gICAgICAgIGNsYXNzPVwic2JiLWNhbGVuZGFyX190YWJsZVwiXG4gICAgICAgIEBmb2N1c291dD0keyhldmVudDogRm9jdXNFdmVudCkgPT5cbiAgICAgICAgICB0aGlzLl9oYW5kbGVUYWJsZUJsdXIoZXZlbnQucmVsYXRlZFRhcmdldCBhcyBIVE1MRWxlbWVudCl9XG4gICAgICAgIEBhbmltYXRpb25lbmQ9JHsoZTogQW5pbWF0aW9uRXZlbnQpID0+IHRoaXMuX3RhYmxlQW5pbWF0aW9uRW5kKGUpfVxuICAgICAgPlxuICAgICAgICAke3RoaXMud2Vla051bWJlcnNcbiAgICAgICAgICA/IGh0bWxgXG4gICAgICAgICAgICAgIDx0aGVhZCBjbGFzcz1cInNiYi1jYWxlbmRhcl9fdGFibGUtaGVhZGVyXCI+XG4gICAgICAgICAgICAgICAgPHRyPlxuICAgICAgICAgICAgICAgICAgJHtuZXh0TW9udGhBY3RpdmVEYXRlXG4gICAgICAgICAgICAgICAgICAgID8gbm90aGluZ1xuICAgICAgICAgICAgICAgICAgICA6IGh0bWxgPHRoIGNsYXNzPVwic2JiLWNhbGVuZGFyX190YWJsZS1kYXRhXCI+PC90aD5gfVxuICAgICAgICAgICAgICAgICAgJHt3ZWVrTnVtYmVycy5tYXAoXG4gICAgICAgICAgICAgICAgICAgICh3ZWVrTnVtYmVyOiBudW1iZXIpID0+IGh0bWxgXG4gICAgICAgICAgICAgICAgICAgICAgPHRoIGNsYXNzPVwic2JiLWNhbGVuZGFyX190YWJsZS1oZWFkZXItY2VsbFwiPlxuICAgICAgICAgICAgICAgICAgICAgICAgJHt0aGlzLm11bHRpcGxlXG4gICAgICAgICAgICAgICAgICAgICAgICAgID8gaHRtbGBcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIDxzYmItY2FsZW5kYXItd2Vla251bWJlclxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAudmFsdWU9JHt3ZWVrTnVtYmVyfVxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBAY2xpY2s9JHsoKSA9PiB7XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgY29uc3QgZGF5czogRGF5PFQ+W10gPSB3ZWVrc0ZvclNlbGVjdE11bHRpcGxlV2Vla051bWJlcnMuZmlsdGVyKFxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgKGRheTogRGF5PFQ+KSA9PiBkYXkud2Vla1ZhbHVlID09PSB3ZWVrTnVtYmVyLFxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICkhO1xuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHRoaXMuX3NlbGVjdE11bHRpcGxlRGF0ZXMoZGF5cyk7XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIH19XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICA+PC9zYmItY2FsZW5kYXItd2Vla251bWJlcj5cbiAgICAgICAgICAgICAgICAgICAgICAgICAgICBgXG4gICAgICAgICAgICAgICAgICAgICAgICAgIDogaHRtbGBcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIDxzYmItc2NyZWVuLXJlYWRlci1vbmx5XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID4ke2Ake2kxOG5DYWxlbmRhcldlZWtOdW1iZXJbdGhpcy5fbGFuZ3VhZ2UuY3VycmVudF19ICR7d2Vla051bWJlcn1gfTwvc2JiLXNjcmVlbi1yZWFkZXItb25seVxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPlxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPHNwYW4gYXJpYS1oaWRkZW49XCJ0cnVlXCI+JHt3ZWVrTnVtYmVyfTwvc3Bhbj5cbiAgICAgICAgICAgICAgICAgICAgICAgICAgICBgfVxuICAgICAgICAgICAgICAgICAgICAgIDwvdGg+XG4gICAgICAgICAgICAgICAgICAgIGAsXG4gICAgICAgICAgICAgICAgICApfVxuICAgICAgICAgICAgICAgIDwvdHI+XG4gICAgICAgICAgICAgIDwvdGhlYWQ+XG4gICAgICAgICAgICBgXG4gICAgICAgICAgOiBub3RoaW5nfVxuICAgICAgICA8dGJvZHkgY2xhc3M9XCJzYmItY2FsZW5kYXJfX3RhYmxlLWJvZHlcIj5cbiAgICAgICAgICAke3dlZWtzLm1hcCgod2VlazogRGF5PFQ+W10sIHJvd0luZGV4OiBudW1iZXIpID0+IHtcbiAgICAgICAgICAgIGNvbnN0IHdlZWtkYXkgPSB0aGlzLl93ZWVrZGF5c1tyb3dJbmRleF07XG4gICAgICAgICAgICBjb25zdCBzZWxlY3RhYmxlRGF5cyA9IHRoaXMuX3dpZGUgPyBbLi4ud2VlaywgLi4udGhpcy5fbmV4dE1vbnRoV2Vla3Nbcm93SW5kZXhdXSA6IHdlZWs7XG4gICAgICAgICAgICByZXR1cm4gaHRtbGBcbiAgICAgICAgICAgICAgPHRyPlxuICAgICAgICAgICAgICAgICR7bmV4dE1vbnRoQWN0aXZlRGF0ZVxuICAgICAgICAgICAgICAgICAgPyBub3RoaW5nXG4gICAgICAgICAgICAgICAgICA6IGh0bWxgXG4gICAgICAgICAgICAgICAgICAgICAgPHRkIGNsYXNzPVwic2JiLWNhbGVuZGFyX190YWJsZS1oZWFkZXItY2VsbC12ZXJ0aWNhbFwiPlxuICAgICAgICAgICAgICAgICAgICAgICAgJHt0aGlzLm11bHRpcGxlXG4gICAgICAgICAgICAgICAgICAgICAgICAgID8gaHRtbGBcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIDxzYmItY2FsZW5kYXItd2Vla2RheVxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAudmFsdWU9JHt3ZWVrZGF5fVxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBAY2xpY2s9JHsoKSA9PiB0aGlzLl9zZWxlY3RNdWx0aXBsZURhdGVzKHNlbGVjdGFibGVEYXlzKX1cbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID5cbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgJHt3ZWVrZGF5Lm5hcnJvd31cbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIDwvc2JiLWNhbGVuZGFyLXdlZWtkYXk+XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgYFxuICAgICAgICAgICAgICAgICAgICAgICAgICA6IGh0bWxgXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICA8c2JiLXNjcmVlbi1yZWFkZXItb25seT4ke3dlZWtkYXkubG9uZ308L3NiYi1zY3JlZW4tcmVhZGVyLW9ubHk+XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICA8c3BhbiBhcmlhLWhpZGRlbj1cInRydWVcIj4ke3dlZWtkYXkubmFycm93fTwvc3Bhbj5cbiAgICAgICAgICAgICAgICAgICAgICAgICAgICBgfVxuICAgICAgICAgICAgICAgICAgICAgIDwvdGQ+XG4gICAgICAgICAgICAgICAgICAgIGB9XG4gICAgICAgICAgICAgICAgJHtyb3dJbmRleCA8IHdlZWtPZmZzZXRcbiAgICAgICAgICAgICAgICAgID8gaHRtbGA8dGQgY2xhc3M9XCJzYmItY2FsZW5kYXJfX3RhYmxlLWRhdGFcIj48L3RkPmBcbiAgICAgICAgICAgICAgICAgIDogbm90aGluZ31cbiAgICAgICAgICAgICAgICAke3RoaXMuX2NyZWF0ZURheUNlbGxzKHdlZWspfVxuICAgICAgICAgICAgICA8L3RyPlxuICAgICAgICAgICAgYDtcbiAgICAgICAgICB9KX1cbiAgICAgICAgPC90Ym9keT5cbiAgICAgIDwvdGFibGU+XG4gICAgYDtcbiAgfVxuXG4gIC8qKiBDcmVhdGVzIHRoZSBjZWxscyBmb3IgdGhlIGRhaWx5IHZpZXcuICovXG4gIHByaXZhdGUgX2NyZWF0ZURheUNlbGxzKHdlZWs6IERheTxUPltdKTogVGVtcGxhdGVSZXN1bHRbXSB7XG4gICAgcmV0dXJuIHdlZWsubWFwKChkYXk6IERheTxUPikgPT4ge1xuICAgICAgcmV0dXJuIGh0bWxgXG4gICAgICAgIDx0ZCBjbGFzcz1cInNiYi1jYWxlbmRhcl9fdGFibGUtZGF0YSBzYmItY2FsZW5kYXJfX2RheS1jZWxsXCI+XG4gICAgICAgICAgPHNsb3QgbmFtZT0ke2RheS52YWx1ZX0+XG4gICAgICAgICAgICA8c2JiLWNhbGVuZGFyLWRheVxuICAgICAgICAgICAgICBzbG90PSR7ZGF5LnZhbHVlfVxuICAgICAgICAgICAgICBAY2xpY2s9JHsoKSA9PiB0aGlzLl9zZWxlY3REYXRlKGRheS5kYXRlVmFsdWUpfVxuICAgICAgICAgICAgICBAa2V5ZG93bj0keyhldnQ6IEtleWJvYXJkRXZlbnQpID0+IHRoaXMuX2hhbmRsZUtleWJvYXJkRXZlbnQoZXZ0LCBkYXkpfVxuICAgICAgICAgICAgPjwvc2JiLWNhbGVuZGFyLWRheT5cbiAgICAgICAgICA8L3Nsb3Q+XG4gICAgICAgIDwvdGQ+XG4gICAgICBgO1xuICAgIH0pO1xuICB9XG5cbiAgLyoqIFJlbmRlciB0aGUgdmlldyBmb3IgdGhlIG1vbnRoIHNlbGVjdGlvbi4gKi9cbiAgcHJpdmF0ZSBfcmVuZGVyTW9udGhWaWV3KCk6IFRlbXBsYXRlUmVzdWx0IHtcbiAgICByZXR1cm4gaHRtbGBcbiAgICAgIDxkaXYgY2xhc3M9XCJzYmItY2FsZW5kYXJfX2NvbnRyb2xzXCI+XG4gICAgICAgICR7dGhpcy5fZ2V0QXJyb3coXG4gICAgICAgICAgJ2xlZnQnLFxuICAgICAgICAgICgpID0+IHRoaXMuX2dvVG9EaWZmZXJlbnRZZWFyKC0xKSxcbiAgICAgICAgICBpMThuUHJldmlvdXNZZWFyW3RoaXMuX2xhbmd1YWdlLmN1cnJlbnRdLFxuICAgICAgICAgIHRoaXMuX3ByZXZpb3VzWWVhckRpc2FibGVkKCksXG4gICAgICAgICl9XG4gICAgICAgIDxkaXYgY2xhc3M9XCJzYmItY2FsZW5kYXJfX2NvbnRyb2xzLW1vbnRoXCI+JHt0aGlzLl9jcmVhdGVMYWJlbEZvck1vbnRoVmlldygpfTwvZGl2PlxuICAgICAgICAke3RoaXMuX2dldEFycm93KFxuICAgICAgICAgICdyaWdodCcsXG4gICAgICAgICAgKCkgPT4gdGhpcy5fZ29Ub0RpZmZlcmVudFllYXIoMSksXG4gICAgICAgICAgaTE4bk5leHRZZWFyW3RoaXMuX2xhbmd1YWdlLmN1cnJlbnRdLFxuICAgICAgICAgIHRoaXMuX25leHRZZWFyRGlzYWJsZWQoKSxcbiAgICAgICAgKX1cbiAgICAgIDwvZGl2PlxuICAgICAgPGRpdiBjbGFzcz1cInNiYi1jYWxlbmRhcl9fdGFibGUtb3ZlcmZsb3ctYnJlYWtcIj5cbiAgICAgICAgPGRpdiBjbGFzcz1cInNiYi1jYWxlbmRhcl9fdGFibGUtY29udGFpbmVyIHNiYi1jYWxlbmRhcl9fdGFibGUtbW9udGgtdmlld1wiPlxuICAgICAgICAgICR7dGhpcy5fY3JlYXRlTW9udGhUYWJsZSh0aGlzLl9tb250aHMsIHRoaXMuX2Nob3NlblllYXIhKX1cbiAgICAgICAgICAke3RoaXMuX3dpZGUgPyB0aGlzLl9jcmVhdGVNb250aFRhYmxlKHRoaXMuX21vbnRocywgdGhpcy5fY2hvc2VuWWVhciEgKyAxKSA6IG5vdGhpbmd9XG4gICAgICAgIDwvZGl2PlxuICAgICAgPC9kaXY+XG4gICAgYDtcbiAgfVxuXG4gIC8qKiBDcmVhdGVzIHRoZSBsYWJlbCB3aXRoIHRoZSB5ZWFyIGZvciB0aGUgbW9udGhseSB2aWV3LiAqL1xuICBwcml2YXRlIF9jcmVhdGVMYWJlbEZvck1vbnRoVmlldygpOiBUZW1wbGF0ZVJlc3VsdCB7XG4gICAgcmV0dXJuIGh0bWxgIDxidXR0b25cbiAgICAgICAgdHlwZT1cImJ1dHRvblwiXG4gICAgICAgIGlkPVwic2JiLWNhbGVuZGFyX19tb250aC1zZWxlY3Rpb25cIlxuICAgICAgICBjbGFzcz1cInNiYi1jYWxlbmRhcl9fY29udHJvbHMtY2hhbmdlLWRhdGVcIlxuICAgICAgICBhcmlhLWxhYmVsPSR7YCR7aTE4bkNhbGVuZGFyRGF0ZVNlbGVjdGlvblt0aGlzLl9sYW5ndWFnZS5jdXJyZW50XX0gJHt0aGlzLl9jaG9zZW5ZZWFyfWB9XG4gICAgICAgIEBjbGljaz0keygpID0+IHRoaXMuX3Jlc2V0Q2FsZW5kYXJWaWV3QW5kRW1pdE1vbnRoQ2hhbmdlKHRydWUpfVxuICAgICAgPlxuICAgICAgICAke3RoaXMuX2Nob3NlblllYXJ9ICR7dGhpcy5fd2lkZSA/IGAgLSAke3RoaXMuX2Nob3NlblllYXIhICsgMX1gIDogbm90aGluZ31cbiAgICAgICAgPHNiYi1pY29uIG5hbWU9XCJjaGV2cm9uLXNtYWxsLXVwLXNtYWxsXCI+PC9zYmItaWNvbj5cbiAgICAgIDwvYnV0dG9uPlxuICAgICAgPHNiYi1zY3JlZW4tcmVhZGVyLW9ubHkgcm9sZT1cInN0YXR1c1wiPiAke3RoaXMuX2Nob3NlblllYXJ9IDwvc2JiLXNjcmVlbi1yZWFkZXItb25seT5gO1xuICB9XG5cbiAgLyoqIENyZWF0ZXMgdGhlIHRhYmxlIGZvciB0aGUgbW9udGggc2VsZWN0aW9uIHZpZXcuICovXG4gIHByaXZhdGUgX2NyZWF0ZU1vbnRoVGFibGUobW9udGhzOiBNb250aENlbGxbXVtdLCB5ZWFyOiBudW1iZXIpOiBUZW1wbGF0ZVJlc3VsdCB7XG4gICAgcmV0dXJuIGh0bWxgXG4gICAgICA8dGFibGVcbiAgICAgICAgY2xhc3M9XCJzYmItY2FsZW5kYXJfX3RhYmxlXCJcbiAgICAgICAgQGFuaW1hdGlvbmVuZD0keyhlOiBBbmltYXRpb25FdmVudCkgPT4gdGhpcy5fdGFibGVBbmltYXRpb25FbmQoZSl9XG4gICAgICA+XG4gICAgICAgICR7dGhpcy5fd2lkZVxuICAgICAgICAgID8gaHRtbGA8dGhlYWQgY2xhc3M9XCJzYmItY2FsZW5kYXJfX3RhYmxlLWhlYWRlclwiIGFyaWEtaGlkZGVuPVwidHJ1ZVwiPlxuICAgICAgICAgICAgICA8dHI+XG4gICAgICAgICAgICAgICAgPHRoIGNsYXNzPVwic2JiLWNhbGVuZGFyX190YWJsZS1oZWFkZXItY2VsbFwiIGNvbHNwYW49JHtNT05USFNfUEVSX1JPV30+JHt5ZWFyfTwvdGg+XG4gICAgICAgICAgICAgIDwvdHI+XG4gICAgICAgICAgICA8L3RoZWFkPmBcbiAgICAgICAgICA6IG5vdGhpbmd9XG4gICAgICAgIDx0Ym9keSBjbGFzcz1cInNiYi1jYWxlbmRhcl9fdGFibGUtYm9keVwiPlxuICAgICAgICAgICR7bW9udGhzLm1hcChcbiAgICAgICAgICAgIChyb3c6IE1vbnRoQ2VsbFtdKSA9PiBodG1sYFxuICAgICAgICAgICAgICA8dHI+XG4gICAgICAgICAgICAgICAgJHtyb3cubWFwKChtb250aDogTW9udGhDZWxsKSA9PiB7XG4gICAgICAgICAgICAgICAgICByZXR1cm4gaHRtbGBcbiAgICAgICAgICAgICAgICAgICAgPHRkIGNsYXNzPVwic2JiLWNhbGVuZGFyX190YWJsZS1kYXRhXCI+XG4gICAgICAgICAgICAgICAgICAgICAgPHNiYi1jYWxlbmRhci1tb250aFxuICAgICAgICAgICAgICAgICAgICAgICAgLnZhbHVlPVwiJHt5ZWFyfS0ke21vbnRoLnZhbHVlfVwiXG4gICAgICAgICAgICAgICAgICAgICAgICBAY2xpY2s9JHsoKSA9PiB0aGlzLl9vbk1vbnRoU2VsZWN0aW9uKG1vbnRoLm1vbnRoVmFsdWUsIHllYXIpfVxuICAgICAgICAgICAgICAgICAgICAgICAgQGtleWRvd249JHsoZXZ0OiBLZXlib2FyZEV2ZW50KSA9PiB0aGlzLl9oYW5kbGVLZXlib2FyZEV2ZW50KGV2dCl9XG4gICAgICAgICAgICAgICAgICAgICAgPlxuICAgICAgICAgICAgICAgICAgICAgIDwvc2JiLWNhbGVuZGFyLW1vbnRoPlxuICAgICAgICAgICAgICAgICAgICA8L3RkPlxuICAgICAgICAgICAgICAgICAgYDtcbiAgICAgICAgICAgICAgICB9KX1cbiAgICAgICAgICAgICAgPC90cj5cbiAgICAgICAgICAgIGAsXG4gICAgICAgICAgKX1cbiAgICAgICAgPC90Ym9keT5cbiAgICAgIDwvdGFibGU+XG4gICAgYDtcbiAgfVxuXG4gIC8qKiBTZWxlY3QgdGhlIG1vbnRoIGFuZCBjaGFuZ2UgdGhlIHZpZXcgdG8gZGF5IHNlbGVjdGlvbi4gKi9cbiAgcHJpdmF0ZSBfb25Nb250aFNlbGVjdGlvbihtb250aDogbnVtYmVyLCB5ZWFyOiBudW1iZXIpOiB2b2lkIHtcbiAgICB0aGlzLl9jaG9zZW5Nb250aCA9IG1vbnRoO1xuICAgIHRoaXMuX25leHRDYWxlbmRhclZpZXcgPSAnZGF5JztcbiAgICB0aGlzLl9pbml0KFxuICAgICAgdGhpcy5fZGF0ZUFkYXB0ZXIuY3JlYXRlRGF0ZShcbiAgICAgICAgeWVhcixcbiAgICAgICAgdGhpcy5fY2hvc2VuTW9udGgsXG4gICAgICAgIHRoaXMuX2RhdGVBZGFwdGVyLmdldERhdGUodGhpcy5fYWN0aXZlRGF0ZSksXG4gICAgICApLFxuICAgICk7XG4gICAgdGhpcy5fc3RhcnRUYWJsZVRyYW5zaXRpb24oKTtcbiAgICB0aGlzLl9lbWl0TW9udGhDaGFuZ2UoKTtcbiAgfVxuXG4gIC8qKiBSZW5kZXIgdGhlIHZpZXcgZm9yIHRoZSB5ZWFyIHNlbGVjdGlvbi4gKi9cbiAgcHJpdmF0ZSBfcmVuZGVyWWVhclZpZXcoKTogVGVtcGxhdGVSZXN1bHQge1xuICAgIHJldHVybiBodG1sYFxuICAgICAgPGRpdiBjbGFzcz1cInNiYi1jYWxlbmRhcl9fY29udHJvbHNcIj5cbiAgICAgICAgJHt0aGlzLl9nZXRBcnJvdyhcbiAgICAgICAgICAnbGVmdCcsXG4gICAgICAgICAgKCkgPT4gdGhpcy5fZ29Ub0RpZmZlcmVudFllYXJSYW5nZSgtWUVBUlNfUEVSX1BBR0UpLFxuICAgICAgICAgIGkxOG5QcmV2aW91c1llYXJSYW5nZShZRUFSU19QRVJfUEFHRSlbdGhpcy5fbGFuZ3VhZ2UuY3VycmVudF0sXG4gICAgICAgICAgdGhpcy5fcHJldmlvdXNZZWFyUmFuZ2VEaXNhYmxlZCgpLFxuICAgICAgICApfVxuICAgICAgICA8ZGl2IGNsYXNzPVwic2JiLWNhbGVuZGFyX19jb250cm9scy1tb250aFwiPiR7dGhpcy5fY3JlYXRlTGFiZWxGb3JZZWFyVmlldygpfTwvZGl2PlxuICAgICAgICAke3RoaXMuX2dldEFycm93KFxuICAgICAgICAgICdyaWdodCcsXG4gICAgICAgICAgKCkgPT4gdGhpcy5fZ29Ub0RpZmZlcmVudFllYXJSYW5nZShZRUFSU19QRVJfUEFHRSksXG4gICAgICAgICAgaTE4bk5leHRZZWFyUmFuZ2UoWUVBUlNfUEVSX1BBR0UpW3RoaXMuX2xhbmd1YWdlLmN1cnJlbnRdLFxuICAgICAgICAgIHRoaXMuX25leHRZZWFyUmFuZ2VEaXNhYmxlZCgpLFxuICAgICAgICApfVxuICAgICAgPC9kaXY+XG4gICAgICA8ZGl2IGNsYXNzPVwic2JiLWNhbGVuZGFyX190YWJsZS1vdmVyZmxvdy1icmVha1wiPlxuICAgICAgICA8ZGl2IGNsYXNzPVwic2JiLWNhbGVuZGFyX190YWJsZS1jb250YWluZXIgc2JiLWNhbGVuZGFyX190YWJsZS15ZWFyLXZpZXdcIj5cbiAgICAgICAgICAke3RoaXMuX2NyZWF0ZVllYXJUYWJsZSh0aGlzLl95ZWFycyl9XG4gICAgICAgICAgJHt0aGlzLl93aWRlID8gdGhpcy5fY3JlYXRlWWVhclRhYmxlKHRoaXMuX25leHRNb250aFllYXJzLCB0cnVlKSA6IG5vdGhpbmd9XG4gICAgICAgIDwvZGl2PlxuICAgICAgPC9kaXY+XG4gICAgYDtcbiAgfVxuXG4gIC8qKiBDcmVhdGVzIHRoZSBidXR0b24gYXJyb3cgZm9yIGFsbCB0aGUgdmlld3MuICovXG4gIHByaXZhdGUgX2dldEFycm93KFxuICAgIGRpcmVjdGlvbjogJ2xlZnQnIHwgJ3JpZ2h0JyxcbiAgICBjbGljazogKCkgPT4gdm9pZCxcbiAgICBhcmlhTGFiZWw6IHN0cmluZyxcbiAgICBkaXNhYmxlZDogYm9vbGVhbixcbiAgKTogVGVtcGxhdGVSZXN1bHQge1xuICAgIHJldHVybiBodG1sYDxzYmItc2Vjb25kYXJ5LWJ1dHRvblxuICAgICAgc2l6ZT1cIm1cIlxuICAgICAgaWNvbi1uYW1lPVwiY2hldnJvbi1zbWFsbC0ke2RpcmVjdGlvbn0tc21hbGxcIlxuICAgICAgYXJpYS1sYWJlbD0ke2FyaWFMYWJlbH1cbiAgICAgIEBjbGljaz0ke2NsaWNrfVxuICAgICAgP2Rpc2FibGVkPSR7ZGlzYWJsZWR9XG4gICAgICBpZD1cInNiYi1jYWxlbmRhcl9fY29udHJvbHMtJHtkaXJlY3Rpb24gPT09ICdsZWZ0JyA/ICdwcmV2aW91cycgOiAnbmV4dCd9XCJcbiAgICA+PC9zYmItc2Vjb25kYXJ5LWJ1dHRvbj5gO1xuICB9XG5cbiAgLyoqIENyZWF0ZXMgdGhlIGxhYmVsIHdpdGggdGhlIHllYXIgcmFuZ2UgZm9yIHRoZSB5ZWFybHkgdmlldy4gKi9cbiAgcHJpdmF0ZSBfY3JlYXRlTGFiZWxGb3JZZWFyVmlldygpOiBUZW1wbGF0ZVJlc3VsdCB7XG4gICAgY29uc3QgZmlyc3RZZWFyOiBudW1iZXIgPSB0aGlzLl95ZWFycy5mbGF0KClbMF07XG4gICAgY29uc3QgbGFzdFllYXJBcnJheTogbnVtYmVyW10gPSAodGhpcy5fd2lkZSA/IHRoaXMuX25leHRNb250aFllYXJzIDogdGhpcy5feWVhcnMpLmZsYXQoKTtcbiAgICBjb25zdCBsYXN0WWVhcjogbnVtYmVyID0gbGFzdFllYXJBcnJheVtsYXN0WWVhckFycmF5Lmxlbmd0aCAtIDFdO1xuICAgIGNvbnN0IHllYXJMYWJlbCA9IGAke2ZpcnN0WWVhcn0gLSAke2xhc3RZZWFyfWA7XG4gICAgcmV0dXJuIGh0bWxgXG4gICAgICA8YnV0dG9uXG4gICAgICAgIHR5cGU9XCJidXR0b25cIlxuICAgICAgICBpZD1cInNiYi1jYWxlbmRhcl9feWVhci1zZWxlY3Rpb25cIlxuICAgICAgICBjbGFzcz1cInNiYi1jYWxlbmRhcl9fY29udHJvbHMtY2hhbmdlLWRhdGVcIlxuICAgICAgICBhcmlhLWxhYmVsPVwiJHtpMThuQ2FsZW5kYXJEYXRlU2VsZWN0aW9uW3RoaXMuX2xhbmd1YWdlLmN1cnJlbnRdfSAke3llYXJMYWJlbH1cIlxuICAgICAgICBAY2xpY2s9JHsoKSA9PiB0aGlzLl9yZXNldENhbGVuZGFyVmlld0FuZEVtaXRNb250aENoYW5nZSh0cnVlKX1cbiAgICAgID5cbiAgICAgICAgJHt5ZWFyTGFiZWx9XG4gICAgICAgIDxzYmItaWNvbiBuYW1lPVwiY2hldnJvbi1zbWFsbC11cC1zbWFsbFwiPjwvc2JiLWljb24+XG4gICAgICA8L2J1dHRvbj5cbiAgICAgIDxzYmItc2NyZWVuLXJlYWRlci1vbmx5IHJvbGU9XCJzdGF0dXNcIj4gJHt5ZWFyTGFiZWx9IDwvc2JiLXNjcmVlbi1yZWFkZXItb25seT5cbiAgICBgO1xuICB9XG5cbiAgLyoqIENyZWF0ZXMgdGhlIHRhYmxlIGZvciB0aGUgeWVhciBzZWxlY3Rpb24gdmlldy4gKi9cbiAgcHJpdmF0ZSBfY3JlYXRlWWVhclRhYmxlKHllYXJzOiBudW1iZXJbXVtdLCBzaGlmdFJpZ2h0ID0gZmFsc2UpOiBUZW1wbGF0ZVJlc3VsdCB7XG4gICAgcmV0dXJuIGh0bWxgIDx0YWJsZVxuICAgICAgY2xhc3M9XCJzYmItY2FsZW5kYXJfX3RhYmxlXCJcbiAgICAgIEBhbmltYXRpb25lbmQ9JHsoZTogQW5pbWF0aW9uRXZlbnQpID0+IHRoaXMuX3RhYmxlQW5pbWF0aW9uRW5kKGUpfVxuICAgID5cbiAgICAgIDx0Ym9keSBjbGFzcz1cInNiYi1jYWxlbmRhcl9fdGFibGUtYm9keVwiPlxuICAgICAgICAke3llYXJzLm1hcChcbiAgICAgICAgICAocm93OiBudW1iZXJbXSkgPT5cbiAgICAgICAgICAgIGh0bWxgIDx0cj5cbiAgICAgICAgICAgICAgJHtyb3cubWFwKCh5ZWFyOiBudW1iZXIpID0+IHtcbiAgICAgICAgICAgICAgICByZXR1cm4gaHRtbGBcbiAgICAgICAgICAgICAgICAgIDx0ZCBjbGFzcz1cInNiYi1jYWxlbmRhcl9fdGFibGUtZGF0YVwiPlxuICAgICAgICAgICAgICAgICAgICA8c2JiLWNhbGVuZGFyLXllYXJcbiAgICAgICAgICAgICAgICAgICAgICAudmFsdWU9JHtTdHJpbmcoeWVhcil9XG4gICAgICAgICAgICAgICAgICAgICAgQGtleWRvd249JHsoZXZ0OiBLZXlib2FyZEV2ZW50KSA9PiB0aGlzLl9oYW5kbGVLZXlib2FyZEV2ZW50KGV2dCl9XG4gICAgICAgICAgICAgICAgICAgICAgQGNsaWNrPSR7KCkgPT4gdGhpcy5fb25ZZWFyU2VsZWN0aW9uKHllYXIsIHNoaWZ0UmlnaHQpfVxuICAgICAgICAgICAgICAgICAgICA+XG4gICAgICAgICAgICAgICAgICAgIDwvc2JiLWNhbGVuZGFyLXllYXI+XG4gICAgICAgICAgICAgICAgICA8L3RkPlxuICAgICAgICAgICAgICAgIGA7XG4gICAgICAgICAgICAgIH0pfVxuICAgICAgICAgICAgPC90cj5gLFxuICAgICAgICApfVxuICAgICAgPC90Ym9keT5cbiAgICA8L3RhYmxlPmA7XG4gIH1cblxuICAvKiogU2VsZWN0IHRoZSB5ZWFyIGFuZCBjaGFuZ2UgdGhlIHZpZXcgdG8gbW9udGggc2VsZWN0aW9uLiAqL1xuICBwcml2YXRlIF9vblllYXJTZWxlY3Rpb24oeWVhcjogbnVtYmVyLCByaWdodFNpZGU6IGJvb2xlYW4pOiB2b2lkIHtcbiAgICB0aGlzLl9jaG9zZW5ZZWFyID0gcmlnaHRTaWRlID8geWVhciAtIDEgOiB5ZWFyO1xuICAgIHRoaXMuX25leHRDYWxlbmRhclZpZXcgPSAnbW9udGgnO1xuICAgIHRoaXMuX2Fzc2lnbkFjdGl2ZURhdGUoXG4gICAgICB0aGlzLl9kYXRlQWRhcHRlci5jcmVhdGVEYXRlKFxuICAgICAgICB0aGlzLl9jaG9zZW5ZZWFyLFxuICAgICAgICB0aGlzLl9kYXRlQWRhcHRlci5nZXRNb250aCh0aGlzLl9hY3RpdmVEYXRlKSxcbiAgICAgICAgdGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0RGF0ZSh0aGlzLl9hY3RpdmVEYXRlKSxcbiAgICAgICksXG4gICAgKTtcbiAgICB0aGlzLl9zdGFydFRhYmxlVHJhbnNpdGlvbigpO1xuICB9XG5cbiAgcHJpdmF0ZSBfZ2V0VmlldygpOiBUZW1wbGF0ZVJlc3VsdCB7XG4gICAgaWYgKGlzU2VydmVyIHx8IHRoaXMuaHlkcmF0aW9uUmVxdWlyZWQpIHtcbiAgICAgIC8vIFRPRE86IFdlIGRpc2FibGUgU1NSIGZvciBjYWxlbmRhciBmb3Igbm93LiBGaWd1cmUgb3V0LCBpZiB0aGVyZSBpcyBhIHdheVxuICAgICAgLy8gdG8gZW5hYmxlIGl0LCB3aGlsZSBjb25zaWRlcmluZyBpMThuIGFuZCBkYXRlIGluZm9ybWF0aW9uLlxuICAgICAgcmV0dXJuIGh0bWxgJHtub3RoaW5nfWA7XG4gICAgfVxuICAgIHN3aXRjaCAodGhpcy5fY2FsZW5kYXJWaWV3KSB7XG4gICAgICBjYXNlICd5ZWFyJzpcbiAgICAgICAgcmV0dXJuIHRoaXMuX3JlbmRlclllYXJWaWV3KCk7XG4gICAgICBjYXNlICdtb250aCc6XG4gICAgICAgIHJldHVybiB0aGlzLl9yZW5kZXJNb250aFZpZXcoKTtcbiAgICAgIGNhc2UgJ2RheSc6XG4gICAgICBkZWZhdWx0OlxuICAgICAgICByZXR1cm4gdGhpcy5fcmVuZGVyRGF5VmlldygpO1xuICAgIH1cbiAgfVxuXG4gIHByaXZhdGUgX3RhYmxlQW5pbWF0aW9uRW5kKGV2ZW50OiBBbmltYXRpb25FdmVudCk6IHZvaWQge1xuICAgIGNvbnN0IHRhYmxlID0gZXZlbnQudGFyZ2V0IGFzIEhUTUxFbGVtZW50O1xuICAgIGlmIChldmVudC5hbmltYXRpb25OYW1lID09PSAnaGlkZScpIHtcbiAgICAgIHRhYmxlLmNsYXNzTGlzdC5yZW1vdmUoJ3NiYi1jYWxlbmRhcl9fdGFibGUtaGlkZScpO1xuICAgICAgaWYgKHRoaXMuX2NvbnRhaW5pbmdGb2N1cykge1xuICAgICAgICB0aGlzLl9yZXNldEZvY3VzID0gdHJ1ZTtcbiAgICAgIH1cbiAgICAgIHRoaXMuX2NhbGVuZGFyVmlldyA9IHRoaXMuX25leHRDYWxlbmRhclZpZXc7XG4gICAgfSBlbHNlIGlmIChldmVudC5hbmltYXRpb25OYW1lID09PSAnc2hvdycpIHtcbiAgICAgIHRoaXMuaW50ZXJuYWxzLnN0YXRlcy5kZWxldGUoJ3RyYW5zaXRpb24nKTtcbiAgICB9XG4gIH1cblxuICBwcml2YXRlIF9zdGFydFRhYmxlVHJhbnNpdGlvbigpOiB2b2lkIHtcbiAgICB0aGlzLmludGVybmFscy5zdGF0ZXMuYWRkKCd0cmFuc2l0aW9uJyk7XG4gICAgdGhpcy5zaGFkb3dSb290XG4gICAgICA/LnF1ZXJ5U2VsZWN0b3JBbGwoJ3RhYmxlJylcbiAgICAgID8uZm9yRWFjaCgoZSkgPT4gZS5jbGFzc0xpc3QudG9nZ2xlKCdzYmItY2FsZW5kYXJfX3RhYmxlLWhpZGUnKSk7XG4gIH1cblxuICBwcm90ZWN0ZWQgb3ZlcnJpZGUgcmVuZGVyKCk6IFRlbXBsYXRlUmVzdWx0IHtcbiAgICByZXR1cm4gaHRtbGA8ZGl2IGNsYXNzPVwic2JiLWNhbGVuZGFyX193cmFwcGVyXCI+JHt0aGlzLl9nZXRWaWV3KCl9PC9kaXY+YDtcbiAgfVxufVxuXG5kZWNsYXJlIGdsb2JhbCB7XG4gIGludGVyZmFjZSBIVE1MRWxlbWVudFRhZ05hbWVNYXAge1xuICAgIC8vIGVzbGludC1kaXNhYmxlLW5leHQtbGluZSBAdHlwZXNjcmlwdC1lc2xpbnQvbmFtaW5nLWNvbnZlbnRpb25cbiAgICAnc2JiLWNhbGVuZGFyJzogU2JiQ2FsZW5kYXJFbGVtZW50O1xuICB9XG4gIGludGVyZmFjZSBIVE1MRWxlbWVudEV2ZW50TWFwIHtcbiAgICBtb250aGNoYW5nZTogU2JiTW9udGhDaGFuZ2VFdmVudDtcbiAgfVxufVxuIl0sInhfZ29vZ2xlX2lnbm9yZUxpc3QiOlswLDEsMiwzLDQsNSw2LDcsOCw5LDEwLDExLDEyLDEzLDE0XSwibWFwcGluZ3MiOiI7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7OztBQXFDQSxJQUFhLGFBQWE7QUFnQ0gsRUFoQkEsS0FBSyxJQUFJLElBQUksRUFBRSxHQUFHLEtBQUssS0FBSyxLQUFLOzs7Ozs7QUF1QnhELElBQWEscUJBQXFCOzs7Ozs7QUFrR2xDLElBQWEsZUFkZ0IsT0FjZTtBQU9mLGVBQWU7QUFPZixlQUFlLGFBT0UsS0FPRzs7Ozs7Ozs7Ozs7O0FBYWpELElBQWEsc0JBQXNCLE9BQU8sSUFBSSxvQkFBb0I7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7O0FDbExsRSxTQUFnQixjQUFjLE1BQU0sT0FBTztBQUN6QyxLQUFJLE9BQU8sU0FBUyxXQUFZLFFBQU8sS0FBSyxNQUFNO0FBRWxELEtBQUksUUFBUSxPQUFPLFNBQVMsWUFBWSx1QkFBdUIsS0FDN0QsUUFBTyxLQUFLLHFCQUFxQixNQUFNO0FBRXpDLEtBQUksZ0JBQWdCLEtBQU0sUUFBTyxJQUFJLEtBQUssWUFBWSxNQUFNO0FBRTVELFFBQU8sSUFBSSxLQUFLLE1BQU07Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7OztBQ0x4QixTQUFnQixPQUFPLFVBQVUsU0FBUztBQUV4QyxRQUFPLGNBQWMsV0FB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
|