@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.
@@ -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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