@sbb-esta/lyne-elements-dev 4.7.0-dev.1773737228 → 4.7.0-dev.1773741165

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (116) hide show
  1. package/calendar/calendar/calendar.component.js +1 -1
  2. package/calendar/calendar-day/calendar-day.component.js +1 -1
  3. package/calendar/calendar-day.js +1 -1
  4. package/calendar/calendar-month/calendar-month.component.js +84 -0
  5. package/calendar/calendar-month.js +4 -0
  6. package/calendar/calendar-weekday/calendar-weekday.component.js +54 -0
  7. package/calendar/calendar-weekday.js +4 -0
  8. package/calendar/calendar-weeknumber/calendar-weeknumber.component.js +56 -0
  9. package/calendar/calendar-weeknumber.js +4 -0
  10. package/calendar/calendar-year/calendar-year.component.js +80 -0
  11. package/calendar/calendar-year.js +4 -0
  12. package/calendar/calendar.js +1 -1
  13. package/calendar/common/calendar-cell-base-element.js +23 -0
  14. package/calendar/common.js +6 -0
  15. package/calendar-day.component-Iewo-vBR.js +109 -0
  16. package/{calendar.component-BgZDy-rX.js → calendar.component-Cz-OGQu6.js} +140 -224
  17. package/calendar.js +8 -4
  18. package/calendar.pure.js +7 -3
  19. package/carousel/carousel/carousel.component.js +1 -1
  20. package/carousel/carousel.js +1 -1
  21. package/carousel.js +1 -1
  22. package/carousel.pure.js +1 -1
  23. package/core/styles/core.scss +8 -0
  24. package/core.css +40 -0
  25. package/custom-elements.json +4273 -375
  26. package/development/calendar/calendar/calendar.component.d.ts +7 -9
  27. package/development/calendar/calendar/calendar.component.d.ts.map +1 -1
  28. package/development/calendar/calendar/calendar.component.js +1 -1
  29. package/development/calendar/calendar-day/calendar-day.component.d.ts +7 -21
  30. package/development/calendar/calendar-day/calendar-day.component.d.ts.map +1 -1
  31. package/development/calendar/calendar-day/calendar-day.component.js +1 -1
  32. package/development/calendar/calendar-day.js +1 -1
  33. package/development/calendar/calendar-month/calendar-month.component.d.ts +33 -0
  34. package/development/calendar/calendar-month/calendar-month.component.d.ts.map +1 -0
  35. package/development/calendar/calendar-month/calendar-month.component.js +112 -0
  36. package/development/calendar/calendar-month.d.ts +2 -0
  37. package/development/calendar/calendar-month.d.ts.map +1 -0
  38. package/development/calendar/calendar-month.js +10 -0
  39. package/development/calendar/calendar-weekday/calendar-weekday.component.d.ts +23 -0
  40. package/development/calendar/calendar-weekday/calendar-weekday.component.d.ts.map +1 -0
  41. package/development/calendar/calendar-weekday/calendar-weekday.component.js +67 -0
  42. package/development/calendar/calendar-weekday.d.ts +2 -0
  43. package/development/calendar/calendar-weekday.d.ts.map +1 -0
  44. package/development/calendar/calendar-weekday.js +10 -0
  45. package/development/calendar/calendar-weeknumber/calendar-weeknumber.component.d.ts +23 -0
  46. package/development/calendar/calendar-weeknumber/calendar-weeknumber.component.d.ts.map +1 -0
  47. package/development/calendar/calendar-weeknumber/calendar-weeknumber.component.js +70 -0
  48. package/development/calendar/calendar-weeknumber.d.ts +2 -0
  49. package/development/calendar/calendar-weeknumber.d.ts.map +1 -0
  50. package/development/calendar/calendar-weeknumber.js +10 -0
  51. package/development/calendar/calendar-year/calendar-year.component.d.ts +28 -0
  52. package/development/calendar/calendar-year/calendar-year.component.d.ts.map +1 -0
  53. package/development/calendar/calendar-year/calendar-year.component.js +103 -0
  54. package/development/calendar/calendar-year.d.ts +2 -0
  55. package/development/calendar/calendar-year.d.ts.map +1 -0
  56. package/development/calendar/calendar-year.js +10 -0
  57. package/development/calendar/calendar.js +1 -1
  58. package/development/calendar/common/calendar-cell-base-element.d.ts +24 -0
  59. package/development/calendar/common/calendar-cell-base-element.d.ts.map +1 -0
  60. package/development/calendar/common/calendar-cell-base-element.js +35 -0
  61. package/development/calendar/common.d.ts +3 -0
  62. package/development/calendar/common.d.ts.map +1 -0
  63. package/development/calendar/common.js +120 -0
  64. package/development/calendar-day.component-Brz0Wyai.js +178 -0
  65. package/development/calendar.component-0Pl7qEXL.js +2222 -0
  66. package/development/calendar.d.ts.map +1 -1
  67. package/development/calendar.js +12 -4
  68. package/development/calendar.pure.d.ts +4 -0
  69. package/development/calendar.pure.d.ts.map +1 -1
  70. package/development/calendar.pure.js +7 -3
  71. package/development/carousel/carousel/carousel.component.js +1 -1
  72. package/development/carousel/carousel.js +1 -1
  73. package/development/{carousel.component-b4C-hxkj.js → carousel.component-Cs0gWroH.js} +2 -2
  74. package/development/carousel.js +1 -1
  75. package/development/carousel.pure.js +1 -1
  76. package/development/mini-calendar/mini-calendar/mini-calendar.component.js +1 -1
  77. package/development/mini-calendar/mini-calendar-day/mini-calendar-day.component.js +1 -1
  78. package/development/mini-calendar/mini-calendar-day.js +1 -1
  79. package/development/mini-calendar/mini-calendar-month/mini-calendar-month.component.js +1 -1
  80. package/development/mini-calendar/mini-calendar-month.js +1 -1
  81. package/development/mini-calendar/mini-calendar.js +1 -1
  82. package/development/{mini-calendar-day.component-DLP7Pa6R.js → mini-calendar-day.component-DdjlUG0k.js} +2 -2
  83. package/development/{mini-calendar-month.component-C1jsGlJ2.js → mini-calendar-month.component-BjxmoYz6.js} +2 -2
  84. package/development/{mini-calendar.component-BzqjMchx.js → mini-calendar.component-B38J84Gg.js} +2 -2
  85. package/development/mini-calendar.js +3 -3
  86. package/development/mini-calendar.pure.js +3 -3
  87. package/development/popover/popover.component.d.ts.map +1 -1
  88. package/development/popover/popover.component.js +1 -1
  89. package/development/{popover.component-BTfbUEzF.js → popover.component-BpYi6btR.js} +2 -2
  90. package/development/popover.js +1 -1
  91. package/development/popover.pure.js +1 -1
  92. package/index.d.ts +5 -1
  93. package/index.js +5 -1
  94. package/mini-calendar/mini-calendar/mini-calendar.component.js +1 -1
  95. package/mini-calendar/mini-calendar-day/mini-calendar-day.component.js +1 -1
  96. package/mini-calendar/mini-calendar-day.js +1 -1
  97. package/mini-calendar/mini-calendar-month/mini-calendar-month.component.js +1 -1
  98. package/mini-calendar/mini-calendar-month.js +1 -1
  99. package/mini-calendar/mini-calendar.js +1 -1
  100. package/{mini-calendar.component-DpPMb47Y.js → mini-calendar.component-_YJbFi4P.js} +4 -4
  101. package/mini-calendar.js +3 -3
  102. package/mini-calendar.pure.js +3 -3
  103. package/off-brand-theme.css +40 -0
  104. package/package.json +27 -2
  105. package/popover/popover.component.js +1 -1
  106. package/{popover.component-Dh9vbuDG.js → popover.component-D_nJwaqJ.js} +1 -1
  107. package/popover.js +1 -1
  108. package/popover.pure.js +1 -1
  109. package/safety-theme.css +40 -0
  110. package/standard-theme.css +40 -0
  111. package/calendar-day.component-DqAXvoP3.js +0 -113
  112. package/development/calendar-day.component-De9Xn3-q.js +0 -283
  113. package/development/calendar.component-PHpgLc0z.js +0 -2458
  114. package/{carousel.component-Bd0iGUbf.js → carousel.component-BA0V2JHI.js} +1 -1
  115. package/{mini-calendar-day.component-cAawa9Gr.js → mini-calendar-day.component-D5HlCTH0.js} +1 -1
  116. package/{mini-calendar-month.component-DiYSVb-Y.js → mini-calendar-month.component-CGAOxDtz.js} +1 -1
@@ -0,0 +1,2222 @@
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 { SbbLanguageController, SbbMediaMatcherController, SbbMediaQueryBreakpointLargeAndAbove } from "./core/controllers.js";
8
+ import { i18nCalendarDateSelection, i18nCalendarWeekNumber, i18nNextMonth, i18nNextYear, i18nNextYearRange, i18nPreviousMonth, i18nPreviousYear, i18nPreviousYearRange, i18nYearMonthSelection } from "./core/i18n.js";
9
+ import "./icon.js";
10
+ import { isArrowKeyOrPageKeysPressed } from "./core/a11y.js";
11
+ import { readConfig } from "./core/config.js";
12
+ import { DAYS_PER_ROW, MONTHS_PER_PAGE, MONTHS_PER_ROW, YEARS_PER_PAGE, YEARS_PER_ROW, defaultDateAdapter } from "./core/datetime.js";
13
+ import { SbbCalendarDayElement } from "./calendar/calendar-day/calendar-day.component.js";
14
+ import { SbbCalendarMonthElement } from "./calendar/calendar-month/calendar-month.component.js";
15
+ import { SbbCalendarWeekdayElement } from "./calendar/calendar-weekday/calendar-weekday.component.js";
16
+ import { SbbCalendarWeeknumberElement } from "./calendar/calendar-weeknumber/calendar-weeknumber.component.js";
17
+ import { SbbCalendarYearElement } from "./calendar/calendar-year/calendar-year.component.js";
18
+ import "./button/secondary-button.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
+ SbbCalendarDayElement.define();
790
+ SbbCalendarMonthElement.define();
791
+ SbbCalendarYearElement.define();
792
+ SbbCalendarWeekdayElement.define();
793
+ SbbCalendarWeeknumberElement.define();
794
+ var SbbMonthChangeEvent = class extends Event {
795
+ get range() {
796
+ return this._range;
797
+ }
798
+ constructor(range) {
799
+ super("monthchange", {
800
+ bubbles: true,
801
+ composed: true
802
+ });
803
+ this._range = Object.freeze(range || []);
804
+ }
805
+ };
806
+ /**
807
+ * It displays a calendar which allows choosing a date.
808
+ *
809
+ * @slot - Use the unnamed slot to add customized `sbb-calendar-day` elements.
810
+ */
811
+ var SbbCalendarElement = (() => {
812
+ let _classSuper = SbbElement;
813
+ let _instanceExtraInitializers = [];
814
+ let _wide_decorators;
815
+ let _wide_initializers = [];
816
+ let _wide_extraInitializers = [];
817
+ let _view_decorators;
818
+ let _view_initializers = [];
819
+ let _view_extraInitializers = [];
820
+ let _min_decorators;
821
+ let _min_initializers = [];
822
+ let _min_extraInitializers = [];
823
+ let _max_decorators;
824
+ let _max_initializers = [];
825
+ let _max_extraInitializers = [];
826
+ let _multiple_decorators;
827
+ let _multiple_initializers = [];
828
+ let _multiple_extraInitializers = [];
829
+ let _set_selected_decorators;
830
+ let __selected_decorators;
831
+ let __selected_initializers = [];
832
+ let __selected_extraInitializers = [];
833
+ let _dateFilter_decorators;
834
+ let _dateFilter_initializers = [];
835
+ let _dateFilter_extraInitializers = [];
836
+ let _orientation_decorators;
837
+ let _orientation_initializers = [];
838
+ let _orientation_extraInitializers = [];
839
+ let _weekNumbers_decorators;
840
+ let _weekNumbers_initializers = [];
841
+ let _weekNumbers_extraInitializers = [];
842
+ let __activeDate_decorators;
843
+ let __activeDate_initializers = [];
844
+ let __activeDate_extraInitializers = [];
845
+ let _set__wide_decorators;
846
+ let __calendarView_decorators;
847
+ let __calendarView_initializers = [];
848
+ let __calendarView_extraInitializers = [];
849
+ let __initialized_decorators;
850
+ let __initialized_initializers = [];
851
+ let __initialized_extraInitializers = [];
852
+ return class SbbCalendarElement extends _classSuper {
853
+ static {
854
+ const _metadata = typeof Symbol === "function" && Symbol.metadata ? Object.create(_classSuper[Symbol.metadata] ?? null) : void 0;
855
+ _wide_decorators = [forceType(), property({ type: Boolean })];
856
+ _view_decorators = [property()];
857
+ _min_decorators = [plainDate(), property()];
858
+ _max_decorators = [plainDate(), property()];
859
+ _multiple_decorators = [
860
+ forceType(),
861
+ handleDistinctChange((e, newValue) => e._onMultipleChanged(newValue)),
862
+ property({ type: Boolean })
863
+ ];
864
+ _set_selected_decorators = [property()];
865
+ __selected_decorators = [state()];
866
+ _dateFilter_decorators = [property({ attribute: "date-filter" })];
867
+ _orientation_decorators = [property({ reflect: true })];
868
+ _weekNumbers_decorators = [forceType(), property({
869
+ attribute: "week-numbers",
870
+ type: Boolean
871
+ })];
872
+ __activeDate_decorators = [state()];
873
+ _set__wide_decorators = [state()];
874
+ __calendarView_decorators = [state()];
875
+ __initialized_decorators = [state()];
876
+ __esDecorate(this, null, _wide_decorators, {
877
+ kind: "accessor",
878
+ name: "wide",
879
+ static: false,
880
+ private: false,
881
+ access: {
882
+ has: (obj) => "wide" in obj,
883
+ get: (obj) => obj.wide,
884
+ set: (obj, value) => {
885
+ obj.wide = value;
886
+ }
887
+ },
888
+ metadata: _metadata
889
+ }, _wide_initializers, _wide_extraInitializers);
890
+ __esDecorate(this, null, _view_decorators, {
891
+ kind: "accessor",
892
+ name: "view",
893
+ static: false,
894
+ private: false,
895
+ access: {
896
+ has: (obj) => "view" in obj,
897
+ get: (obj) => obj.view,
898
+ set: (obj, value) => {
899
+ obj.view = value;
900
+ }
901
+ },
902
+ metadata: _metadata
903
+ }, _view_initializers, _view_extraInitializers);
904
+ __esDecorate(this, null, _min_decorators, {
905
+ kind: "accessor",
906
+ name: "min",
907
+ static: false,
908
+ private: false,
909
+ access: {
910
+ has: (obj) => "min" in obj,
911
+ get: (obj) => obj.min,
912
+ set: (obj, value) => {
913
+ obj.min = value;
914
+ }
915
+ },
916
+ metadata: _metadata
917
+ }, _min_initializers, _min_extraInitializers);
918
+ __esDecorate(this, null, _max_decorators, {
919
+ kind: "accessor",
920
+ name: "max",
921
+ static: false,
922
+ private: false,
923
+ access: {
924
+ has: (obj) => "max" in obj,
925
+ get: (obj) => obj.max,
926
+ set: (obj, value) => {
927
+ obj.max = value;
928
+ }
929
+ },
930
+ metadata: _metadata
931
+ }, _max_initializers, _max_extraInitializers);
932
+ __esDecorate(this, null, _multiple_decorators, {
933
+ kind: "accessor",
934
+ name: "multiple",
935
+ static: false,
936
+ private: false,
937
+ access: {
938
+ has: (obj) => "multiple" in obj,
939
+ get: (obj) => obj.multiple,
940
+ set: (obj, value) => {
941
+ obj.multiple = value;
942
+ }
943
+ },
944
+ metadata: _metadata
945
+ }, _multiple_initializers, _multiple_extraInitializers);
946
+ __esDecorate(this, null, _set_selected_decorators, {
947
+ kind: "setter",
948
+ name: "selected",
949
+ static: false,
950
+ private: false,
951
+ access: {
952
+ has: (obj) => "selected" in obj,
953
+ set: (obj, value) => {
954
+ obj.selected = value;
955
+ }
956
+ },
957
+ metadata: _metadata
958
+ }, null, _instanceExtraInitializers);
959
+ __esDecorate(this, null, __selected_decorators, {
960
+ kind: "accessor",
961
+ name: "_selected",
962
+ static: false,
963
+ private: false,
964
+ access: {
965
+ has: (obj) => "_selected" in obj,
966
+ get: (obj) => obj._selected,
967
+ set: (obj, value) => {
968
+ obj._selected = value;
969
+ }
970
+ },
971
+ metadata: _metadata
972
+ }, __selected_initializers, __selected_extraInitializers);
973
+ __esDecorate(this, null, _dateFilter_decorators, {
974
+ kind: "accessor",
975
+ name: "dateFilter",
976
+ static: false,
977
+ private: false,
978
+ access: {
979
+ has: (obj) => "dateFilter" in obj,
980
+ get: (obj) => obj.dateFilter,
981
+ set: (obj, value) => {
982
+ obj.dateFilter = value;
983
+ }
984
+ },
985
+ metadata: _metadata
986
+ }, _dateFilter_initializers, _dateFilter_extraInitializers);
987
+ __esDecorate(this, null, _orientation_decorators, {
988
+ kind: "accessor",
989
+ name: "orientation",
990
+ static: false,
991
+ private: false,
992
+ access: {
993
+ has: (obj) => "orientation" in obj,
994
+ get: (obj) => obj.orientation,
995
+ set: (obj, value) => {
996
+ obj.orientation = value;
997
+ }
998
+ },
999
+ metadata: _metadata
1000
+ }, _orientation_initializers, _orientation_extraInitializers);
1001
+ __esDecorate(this, null, _weekNumbers_decorators, {
1002
+ kind: "accessor",
1003
+ name: "weekNumbers",
1004
+ static: false,
1005
+ private: false,
1006
+ access: {
1007
+ has: (obj) => "weekNumbers" in obj,
1008
+ get: (obj) => obj.weekNumbers,
1009
+ set: (obj, value) => {
1010
+ obj.weekNumbers = value;
1011
+ }
1012
+ },
1013
+ metadata: _metadata
1014
+ }, _weekNumbers_initializers, _weekNumbers_extraInitializers);
1015
+ __esDecorate(this, null, __activeDate_decorators, {
1016
+ kind: "accessor",
1017
+ name: "_activeDate",
1018
+ static: false,
1019
+ private: false,
1020
+ access: {
1021
+ has: (obj) => "_activeDate" in obj,
1022
+ get: (obj) => obj._activeDate,
1023
+ set: (obj, value) => {
1024
+ obj._activeDate = value;
1025
+ }
1026
+ },
1027
+ metadata: _metadata
1028
+ }, __activeDate_initializers, __activeDate_extraInitializers);
1029
+ __esDecorate(this, null, _set__wide_decorators, {
1030
+ kind: "setter",
1031
+ name: "_wide",
1032
+ static: false,
1033
+ private: false,
1034
+ access: {
1035
+ has: (obj) => "_wide" in obj,
1036
+ set: (obj, value) => {
1037
+ obj._wide = value;
1038
+ }
1039
+ },
1040
+ metadata: _metadata
1041
+ }, null, _instanceExtraInitializers);
1042
+ __esDecorate(this, null, __calendarView_decorators, {
1043
+ kind: "accessor",
1044
+ name: "_calendarView",
1045
+ static: false,
1046
+ private: false,
1047
+ access: {
1048
+ has: (obj) => "_calendarView" in obj,
1049
+ get: (obj) => obj._calendarView,
1050
+ set: (obj, value) => {
1051
+ obj._calendarView = value;
1052
+ }
1053
+ },
1054
+ metadata: _metadata
1055
+ }, __calendarView_initializers, __calendarView_extraInitializers);
1056
+ __esDecorate(this, null, __initialized_decorators, {
1057
+ kind: "accessor",
1058
+ name: "_initialized",
1059
+ static: false,
1060
+ private: false,
1061
+ access: {
1062
+ has: (obj) => "_initialized" in obj,
1063
+ get: (obj) => obj._initialized,
1064
+ set: (obj, value) => {
1065
+ obj._initialized = value;
1066
+ }
1067
+ },
1068
+ metadata: _metadata
1069
+ }, __initialized_initializers, __initialized_extraInitializers);
1070
+ if (_metadata) Object.defineProperty(this, Symbol.metadata, {
1071
+ enumerable: true,
1072
+ configurable: true,
1073
+ writable: true,
1074
+ value: _metadata
1075
+ });
1076
+ }
1077
+ static {
1078
+ this.elementName = "sbb-calendar";
1079
+ }
1080
+ static {
1081
+ this.styles = [boxSizingStyles, calendar_default];
1082
+ }
1083
+ static {
1084
+ this.events = {
1085
+ dateselected: "dateselected",
1086
+ monthchange: "monthchange"
1087
+ };
1088
+ }
1089
+ #wide_accessor_storage;
1090
+ /** If set to true, two months are displayed */
1091
+ get wide() {
1092
+ return this.#wide_accessor_storage;
1093
+ }
1094
+ set wide(value) {
1095
+ this.#wide_accessor_storage = value;
1096
+ }
1097
+ #view_accessor_storage;
1098
+ /** The initial view of the calendar which should be displayed on opening. */
1099
+ get view() {
1100
+ return this.#view_accessor_storage;
1101
+ }
1102
+ set view(value) {
1103
+ this.#view_accessor_storage = value;
1104
+ }
1105
+ #min_accessor_storage;
1106
+ /**
1107
+ * The minimum valid date. Accepts a date object or null.
1108
+ * Accepts an ISO8601 formatted string (e.g. 2024-12-24) as attribute.
1109
+ */
1110
+ get min() {
1111
+ return this.#min_accessor_storage;
1112
+ }
1113
+ set min(value) {
1114
+ this.#min_accessor_storage = value;
1115
+ }
1116
+ #max_accessor_storage;
1117
+ /**
1118
+ * The maximum valid date. Accepts a date object or null.
1119
+ * Accepts an ISO8601 formatted string (e.g. 2024-12-24) as attribute.
1120
+ */
1121
+ get max() {
1122
+ return this.#max_accessor_storage;
1123
+ }
1124
+ set max(value) {
1125
+ this.#max_accessor_storage = value;
1126
+ }
1127
+ #multiple_accessor_storage;
1128
+ /** Whether the calendar allows for multiple date selection. */
1129
+ get multiple() {
1130
+ return this.#multiple_accessor_storage;
1131
+ }
1132
+ set multiple(value) {
1133
+ this.#multiple_accessor_storage = value;
1134
+ }
1135
+ /**
1136
+ * The selected date: accepts a date object, or, if `multiple`, an array of dates.
1137
+ */
1138
+ set selected(value) {
1139
+ 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));
1140
+ else {
1141
+ const selectedDate = this._dateAdapter.getValidDateOrNull(this._dateAdapter.deserialize(value));
1142
+ if (!!selectedDate && (!this._isDayInRange(this._dateAdapter.toIso8601(selectedDate)) || this._dateFilter(selectedDate))) this._selected = selectedDate;
1143
+ else this._selected = null;
1144
+ }
1145
+ }
1146
+ get selected() {
1147
+ return this._selected;
1148
+ }
1149
+ #_selected_accessor_storage;
1150
+ get _selected() {
1151
+ return this.#_selected_accessor_storage;
1152
+ }
1153
+ set _selected(value) {
1154
+ this.#_selected_accessor_storage = value;
1155
+ }
1156
+ #dateFilter_accessor_storage;
1157
+ /** A function used to filter out dates. */
1158
+ get dateFilter() {
1159
+ return this.#dateFilter_accessor_storage;
1160
+ }
1161
+ set dateFilter(value) {
1162
+ this.#dateFilter_accessor_storage = value;
1163
+ }
1164
+ #orientation_accessor_storage;
1165
+ /** The orientation of days in the calendar. */
1166
+ get orientation() {
1167
+ return this.#orientation_accessor_storage;
1168
+ }
1169
+ set orientation(value) {
1170
+ this.#orientation_accessor_storage = value;
1171
+ }
1172
+ #weekNumbers_accessor_storage;
1173
+ /** Whether it has to display the week numbers in addition to week days. */
1174
+ get weekNumbers() {
1175
+ return this.#weekNumbers_accessor_storage;
1176
+ }
1177
+ set weekNumbers(value) {
1178
+ this.#weekNumbers_accessor_storage = value;
1179
+ }
1180
+ #_activeDate_accessor_storage;
1181
+ /** The currently active date. */
1182
+ get _activeDate() {
1183
+ return this.#_activeDate_accessor_storage;
1184
+ }
1185
+ set _activeDate(value) {
1186
+ this.#_activeDate_accessor_storage = value;
1187
+ }
1188
+ /** The current wide property considering property value and breakpoints. From zero to small `wide` has always to be false. */
1189
+ set _wide(wide) {
1190
+ this.toggleState("wide", wide);
1191
+ this._wideInternal = wide;
1192
+ }
1193
+ get _wide() {
1194
+ return this._wideInternal;
1195
+ }
1196
+ #_calendarView_accessor_storage;
1197
+ get _calendarView() {
1198
+ return this.#_calendarView_accessor_storage;
1199
+ }
1200
+ set _calendarView(value) {
1201
+ this.#_calendarView_accessor_storage = value;
1202
+ }
1203
+ /** A list of calendar's cells corresponding to days, months or years depending on the view. */
1204
+ get _cells() {
1205
+ 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}`)) ?? []);
1206
+ }
1207
+ #_initialized_accessor_storage;
1208
+ get _initialized() {
1209
+ return this.#_initialized_accessor_storage;
1210
+ }
1211
+ set _initialized(value) {
1212
+ this.#_initialized_accessor_storage = value;
1213
+ }
1214
+ constructor() {
1215
+ super();
1216
+ this.#wide_accessor_storage = (__runInitializers(this, _instanceExtraInitializers), __runInitializers(this, _wide_initializers, false));
1217
+ this.#view_accessor_storage = (__runInitializers(this, _wide_extraInitializers), __runInitializers(this, _view_initializers, "day"));
1218
+ this.#min_accessor_storage = (__runInitializers(this, _view_extraInitializers), __runInitializers(this, _min_initializers, null));
1219
+ this.#max_accessor_storage = (__runInitializers(this, _min_extraInitializers), __runInitializers(this, _max_initializers, null));
1220
+ this.#multiple_accessor_storage = (__runInitializers(this, _max_extraInitializers), __runInitializers(this, _multiple_initializers, false));
1221
+ this.#_selected_accessor_storage = (__runInitializers(this, _multiple_extraInitializers), __runInitializers(this, __selected_initializers, null));
1222
+ this.#dateFilter_accessor_storage = (__runInitializers(this, __selected_extraInitializers), __runInitializers(this, _dateFilter_initializers, null));
1223
+ this.#orientation_accessor_storage = (__runInitializers(this, _dateFilter_extraInitializers), __runInitializers(this, _orientation_initializers, "horizontal"));
1224
+ this.#weekNumbers_accessor_storage = (__runInitializers(this, _orientation_extraInitializers), __runInitializers(this, _weekNumbers_initializers, false));
1225
+ this._dateAdapter = (__runInitializers(this, _weekNumbers_extraInitializers), readConfig().datetime?.dateAdapter ?? defaultDateAdapter);
1226
+ this.#_activeDate_accessor_storage = __runInitializers(this, __activeDate_initializers, this._dateAdapter.today());
1227
+ this._wideInternal = (__runInitializers(this, __activeDate_extraInitializers), false);
1228
+ this.#_calendarView_accessor_storage = __runInitializers(this, __calendarView_initializers, "day");
1229
+ this._nextCalendarView = (__runInitializers(this, __calendarView_extraInitializers), "day");
1230
+ /** Information about the rendered day view; used in keyboard navigation. */
1231
+ this._keyboardNavigationDayViewParameters = {
1232
+ firstDayInView: null,
1233
+ lastDayInView: null,
1234
+ firstMonthOffset: 0,
1235
+ firstMonthLength: 0,
1236
+ secondMonthOffset: 0
1237
+ };
1238
+ /** Grid of calendar cells representing the dates of the month. */
1239
+ this._weeks = [];
1240
+ /** An array containing all the month names in the current language. */
1241
+ this._monthNames = this._dateAdapter.getMonthNames("long");
1242
+ this._enhancedVariant = false;
1243
+ /** Whether the focus should be reset on focusCell. */
1244
+ this._resetFocus = false;
1245
+ /** Whether an element inside the calendar is currently focused. */
1246
+ this._containingFocus = false;
1247
+ this.#_initialized_accessor_storage = __runInitializers(this, __initialized_initializers, false);
1248
+ this._language = (__runInitializers(this, __initialized_extraInitializers), new SbbLanguageController(this).withHandler(() => {
1249
+ this._monthNames = this._dateAdapter.getMonthNames("long");
1250
+ this._createMonthRows();
1251
+ }));
1252
+ this._mediaMatcher = new SbbMediaMatcherController(this, { [SbbMediaQueryBreakpointLargeAndAbove]: () => this._init() });
1253
+ this._onSlotChange = () => {
1254
+ this._enhancedVariant = Array.from(this.children).some((c) => c.localName === "sbb-calendar-day");
1255
+ this.toggleState("enhanced", this._enhancedVariant);
1256
+ this._setTabIndex();
1257
+ };
1258
+ this._createMonthRows();
1259
+ this._setWeekdays();
1260
+ this.addEventListener("focusin", () => this._containingFocus = true);
1261
+ this.addEventListener("focusout", () => this._containingFocus = false);
1262
+ this.addEventListener("click", (e) => {
1263
+ const day = e.target.closest("sbb-calendar-day");
1264
+ if (day) this._selectDate(day.value);
1265
+ });
1266
+ this.addEventListener("keydown", (e) => {
1267
+ if (e.target.localName === "sbb-calendar-day") this._handleKeyboardEvent(e, this._mapDateToDay(e.target.value));
1268
+ });
1269
+ }
1270
+ _dateFilter(date) {
1271
+ return this.dateFilter?.(date) ?? true;
1272
+ }
1273
+ /** Resets the active month according to the new state of the calendar. */
1274
+ resetPosition() {
1275
+ this._resetCalendarView();
1276
+ this._init();
1277
+ }
1278
+ connectedCallback() {
1279
+ super.connectedCallback();
1280
+ this.resetPosition();
1281
+ this.shadowRoot?.addEventListener("slotchange", this._onSlotChange, { capture: true });
1282
+ }
1283
+ disconnectedCallback() {
1284
+ super.disconnectedCallback();
1285
+ this.shadowRoot?.removeEventListener("slotchange", this._onSlotChange, { capture: true });
1286
+ }
1287
+ /** @internal */
1288
+ focus() {
1289
+ this._resetFocus = true;
1290
+ this._focusCell();
1291
+ }
1292
+ willUpdate(changedProperties) {
1293
+ super.willUpdate(changedProperties);
1294
+ if (!this._initialized) return;
1295
+ if (changedProperties.has("wide") || changedProperties.has("orientation")) this.resetPosition();
1296
+ if (changedProperties.has("view")) {
1297
+ this._setChosenYear();
1298
+ this._chosenMonth = void 0;
1299
+ this._nextCalendarView = this._calendarView = this.view;
1300
+ }
1301
+ }
1302
+ updated(changedProperties) {
1303
+ super.updated(changedProperties);
1304
+ this._setTabIndex();
1305
+ this._focusCell();
1306
+ }
1307
+ /**
1308
+ * The `_selected` state should be adapted when the `multiple` property changes:
1309
+ * - if it changes to true, the '_selected' is set to an array;
1310
+ * - if it changes to false, the first available option is set as 'value' otherwise it's set to null.
1311
+ */
1312
+ _onMultipleChanged(isMultiple) {
1313
+ if (isMultiple && !Array.isArray(this._selected)) this._selected = this._selected ? [this._selected] : [];
1314
+ if (!isMultiple && Array.isArray(this._selected)) this._selected = this._selected.length ? this._selected[0] : null;
1315
+ }
1316
+ /** Initializes the component. */
1317
+ _init(activeDate) {
1318
+ if (isServer) return;
1319
+ else if (this.hydrationRequired) {
1320
+ this.hydrationComplete.then(() => this._init());
1321
+ return;
1322
+ }
1323
+ if (activeDate) this._assignActiveDate(activeDate);
1324
+ this._wide = (this._mediaMatcher.matches(SbbMediaQueryBreakpointLargeAndAbove) ?? false) && this.wide;
1325
+ this._weeks = this._createWeekRows(this._activeDate);
1326
+ this._years = this._createYearRows();
1327
+ this._weekNumbers = this._createWeekNumbers(this._activeDate);
1328
+ this._nextMonthWeeks = [[]];
1329
+ this._nextMonthYears = [[]];
1330
+ if (this._wide) {
1331
+ const nextMonthDate = this._dateAdapter.addCalendarMonths(this._activeDate, 1);
1332
+ this._nextMonthWeeks = this._createWeekRows(nextMonthDate, true);
1333
+ this._nextMonthYears = this._createYearRows(YEARS_PER_PAGE);
1334
+ this._nextMonthWeekNumbers = this._createWeekNumbers(nextMonthDate);
1335
+ }
1336
+ this._initialized = true;
1337
+ }
1338
+ /** Focuses on a day cell prioritizing the selected day, the current day, and lastly, the first selectable day. */
1339
+ _focusCell() {
1340
+ if (this._resetFocus) {
1341
+ this._getFirstFocusable()?.focus();
1342
+ this._resetFocus = false;
1343
+ }
1344
+ }
1345
+ /** Creates the array of weekdays. */
1346
+ _setWeekdays() {
1347
+ const narrowWeekdays = this._dateAdapter.getDayOfWeekNames("narrow");
1348
+ const weekdays = this._dateAdapter.getDayOfWeekNames("long").map((long, i) => ({
1349
+ long,
1350
+ narrow: narrowWeekdays[i]
1351
+ }));
1352
+ const firstDayOfWeek = this._dateAdapter.getFirstDayOfWeek();
1353
+ this._weekdays = weekdays.slice(firstDayOfWeek).concat(weekdays.slice(0, firstDayOfWeek));
1354
+ }
1355
+ /**
1356
+ * Given a date, it returns the week numbers for the month the date belongs to.
1357
+ * TODO: check if date-fns can be replaced with custom logic.
1358
+ *
1359
+ * Since the calculation is not simple (see https://en.wikipedia.org/wiki/Week#Numbering),
1360
+ * the date-fns library has been used this way:
1361
+ * the first and the last day of the month are calculated and then passed to the `eachWeekOfInterval` function,
1362
+ * which returns an array containing the starting day of every ISO week of the month,
1363
+ * considering Monday as the first day.
1364
+ * Then, this array is mapped via the `getWeek` function, which returns the ISO week number for that date.
1365
+ */
1366
+ _createWeekNumbers(date) {
1367
+ return eachWeekOfInterval({
1368
+ start: startOfMonth(date),
1369
+ end: endOfMonth(date)
1370
+ }, { weekStartsOn: 1 }).map((firstDayOfWeek) => getWeek(firstDayOfWeek, {
1371
+ weekStartsOn: 1,
1372
+ firstWeekContainsDate: 4
1373
+ }));
1374
+ }
1375
+ /** Creates the rows along the horizontal direction and sets the parameters used in keyboard navigation. */
1376
+ _createWeekRows(value, isSecondMonthInView = false) {
1377
+ const daysInMonth = this._dateAdapter.getNumDaysInMonth(value);
1378
+ const weekOffset = this._dateAdapter.getFirstWeekOffset(value);
1379
+ if (!isSecondMonthInView) {
1380
+ this._keyboardNavigationDayViewParameters.firstMonthLength = daysInMonth;
1381
+ this._keyboardNavigationDayViewParameters.firstMonthOffset = weekOffset;
1382
+ this._keyboardNavigationDayViewParameters.firstDayInView = this._dateAdapter.toIso8601(this._dateAdapter.createDate(this._dateAdapter.getYear(value), this._dateAdapter.getMonth(value), 1));
1383
+ this._keyboardNavigationDayViewParameters.lastDayInView = this._dateAdapter.toIso8601(this._dateAdapter.createDate(this._dateAdapter.getYear(value), this._dateAdapter.getMonth(value), daysInMonth));
1384
+ } else {
1385
+ this._keyboardNavigationDayViewParameters.secondMonthOffset = weekOffset;
1386
+ this._keyboardNavigationDayViewParameters.lastDayInView = this._dateAdapter.toIso8601(this._dateAdapter.createDate(this._dateAdapter.getYear(value), this._dateAdapter.getMonth(value), daysInMonth));
1387
+ }
1388
+ return this.orientation === "horizontal" ? this._createWeekRowsHorizontal(value, daysInMonth, weekOffset) : this._createWeekRowsVertical(value, daysInMonth, weekOffset);
1389
+ }
1390
+ /**
1391
+ * Creates the rows for each week in orientation='horizontal'.
1392
+ *
1393
+ * Iterates through the days of the months, creates a Day object for each and pushes it into and array.
1394
+ * Each seven days (considering the offset at the beginning of the month) restarts from an empty array.
1395
+ *
1396
+ * The result is a matrix in which every row is a week (or part of it, considering offset).
1397
+ */
1398
+ _createWeekRowsHorizontal(value, daysInMonth, weekOffset) {
1399
+ const weeks = [[]];
1400
+ for (let i = 0, cell = weekOffset; i < daysInMonth; i++, cell++) {
1401
+ if (cell === DAYS_PER_ROW) {
1402
+ weeks.push([]);
1403
+ cell = 0;
1404
+ }
1405
+ const date = this._dateAdapter.createDate(this._dateAdapter.getYear(value), this._dateAdapter.getMonth(value), i + 1);
1406
+ weeks[weeks.length - 1].push(this._mapDateToDay(date));
1407
+ }
1408
+ return weeks;
1409
+ }
1410
+ /**
1411
+ * Creates the rows for each week in orientation='vertical'.
1412
+ *
1413
+ * Creates a matrix with seven empty rows.
1414
+ * Iterates through the days of the months, creates a Day object for each
1415
+ * and pushes it into the correct array considering the offset at the beginning of the month.
1416
+ * Each seven days (including offset) restarts from the first.
1417
+ *
1418
+ * The result is a matrix in which every row is a set of weekdays, so:
1419
+ * - row 0: all the Mondays;
1420
+ * - row 1: all the Tuesdays;
1421
+ * - ...
1422
+ * - row 7: all the Sundays.
1423
+ */
1424
+ _createWeekRowsVertical(value, daysInMonth, weekOffset) {
1425
+ const weeks = Array.from({ length: DAYS_PER_ROW }, () => []);
1426
+ for (let i = 0, cell = weekOffset; i < daysInMonth; i++, cell++) {
1427
+ if (cell === DAYS_PER_ROW) cell = 0;
1428
+ const date = this._dateAdapter.createDate(this._dateAdapter.getYear(value), this._dateAdapter.getMonth(value), i + 1);
1429
+ weeks[cell].push(this._mapDateToDay(date));
1430
+ }
1431
+ return weeks;
1432
+ }
1433
+ _mapDateToDay(date) {
1434
+ const isoDate = this._dateAdapter.toIso8601(date);
1435
+ return {
1436
+ value: isoDate,
1437
+ dateValue: date,
1438
+ dayValue: String(this._dateAdapter.getDate(date)),
1439
+ monthValue: String(this._dateAdapter.getMonth(date)),
1440
+ yearValue: String(this._dateAdapter.getYear(date)),
1441
+ weekValue: getWeek(isoDate, {
1442
+ weekStartsOn: 1,
1443
+ firstWeekContainsDate: 4
1444
+ }),
1445
+ weekDayValue: this._dateAdapter.getDayOfWeek(date)
1446
+ };
1447
+ }
1448
+ /** Force the conversion to ISO8601 formatted string. */
1449
+ _mapValueToISODate(value) {
1450
+ return typeof value === "string" ? value : this._dateAdapter.toIso8601(value);
1451
+ }
1452
+ /** Creates the rows for the month selection view. */
1453
+ _createMonthRows() {
1454
+ const months = new Array(12).fill(null).map((_, i) => ({
1455
+ value: String(i + 1).padStart(2, "0"),
1456
+ monthValue: i + 1
1457
+ }));
1458
+ const rows = 12 / MONTHS_PER_ROW;
1459
+ const monthArray = [];
1460
+ for (let i = 0; i < rows; i++) monthArray.push(months.slice(MONTHS_PER_ROW * i, MONTHS_PER_ROW * (i + 1)));
1461
+ this._months = monthArray;
1462
+ }
1463
+ /** Creates the rows for the year selection view. */
1464
+ _createYearRows(offset = 0) {
1465
+ const startValueYearView = this._getStartValueYearView();
1466
+ const allYears = new Array(YEARS_PER_PAGE).fill(0).map((_, i) => startValueYearView + offset + i);
1467
+ const rows = YEARS_PER_PAGE / YEARS_PER_ROW;
1468
+ const yearArray = [];
1469
+ for (let i = 0; i < rows; i++) yearArray.push(allYears.slice(YEARS_PER_ROW * i, YEARS_PER_ROW * (i + 1)));
1470
+ return yearArray;
1471
+ }
1472
+ /**
1473
+ * Calculates the first year that will be shown in the year selection panel.
1474
+ * If `minDate` and `maxDate` are both null, the starting year is calculated as
1475
+ * the multiple of YEARS_PER_PAGE closest to and less than activeDate,
1476
+ * e.g., with `YEARS_PER_PAGE` = 24 and `activeDate` = 2020, the function will return 2016 (24 * 83),
1477
+ * while with `activeDate` = 2000, the function will return 1992 (24 * 82).
1478
+ * If `minDate` is not null, it returns the corresponding year; if `maxDate` is not null,
1479
+ * it returns the corresponding year minus `YEARS_PER_PAGE`, so that the `maxDate` is the last rendered year.
1480
+ * If both are not null, `maxDate` has priority over `minDate`.
1481
+ */
1482
+ _getStartValueYearView() {
1483
+ let startingYear = 0;
1484
+ if (this.max) startingYear = this._dateAdapter.getYear(this.max) - YEARS_PER_PAGE + 1;
1485
+ else if (this.min) startingYear = this._dateAdapter.getYear(this.min);
1486
+ const activeYear = this._dateAdapter.getYear(this._activeDate);
1487
+ return activeYear - ((activeYear - startingYear) % YEARS_PER_PAGE + YEARS_PER_PAGE) % YEARS_PER_PAGE;
1488
+ }
1489
+ /** Checks if date is within the min-max range. */
1490
+ _isDayInRange(dateString) {
1491
+ if (!this.min && !this.max) return true;
1492
+ const date = this._dateAdapter.deserialize(dateString);
1493
+ return this._dateAdapter.sameDate(date, this._dateAdapter.clampDate(date, this.min, this.max));
1494
+ }
1495
+ /** Emits the selected date and sets it internally. */
1496
+ _selectDate(day) {
1497
+ this._chosenMonth = void 0;
1498
+ this._setChosenYear();
1499
+ if (this.multiple) {
1500
+ if (this._selected && this._selected.length > 0) {
1501
+ const indexOfSelectedDay = this._selected.findIndex((sel) => this._dateAdapter.compareDate(sel, day) === 0);
1502
+ if (indexOfSelectedDay !== -1) this._selected = this._selected.filter((_, i) => i !== indexOfSelectedDay);
1503
+ else this._selected = [...this._selected, day];
1504
+ } else this._selected = [day];
1505
+ this._emitDateSelectedEvent(this._selected.map((e) => this._dateAdapter.deserialize(e)));
1506
+ } else if (!this._selected || this._dateAdapter.compareDate(this._selected, day) !== 0) {
1507
+ this._selected = day;
1508
+ this._emitDateSelectedEvent(this._dateAdapter.deserialize(day));
1509
+ }
1510
+ }
1511
+ /**
1512
+ * Handle multiple dates selection via weekNumber / weekDay buttons:
1513
+ * - if Cmd or Ctrl are pressed, add the new date to the current ones;
1514
+ * - if not,
1515
+ * - if the new dates are the same of the current ones, it means that the same button has been clicked twice, so do nothing;
1516
+ * - if not, the selected dates are the new ones.
1517
+ */
1518
+ _selectMultipleDates(days) {
1519
+ const enabledDays = this._cells.filter((e) => !e.disabled).map((e) => this._mapValueToISODate(e.value));
1520
+ const daysToAdd = days.map((e) => e.value).filter((isoDate) => enabledDays.includes(isoDate));
1521
+ const daysToAddSet = new Set(daysToAdd);
1522
+ const selectedSet = new Set(this._selected.map((s) => this._dateAdapter.toIso8601(s)));
1523
+ this._selected = this._updateSelectedWithMultipleDates(daysToAdd, daysToAddSet, selectedSet).map((s) => this._dateAdapter.deserialize(s));
1524
+ this._emitDateSelectedEvent(this._selected.map((e) => this._dateAdapter.deserialize(e)));
1525
+ }
1526
+ /**
1527
+ * Emits the dateselected event given the detail (as T or T[] based on the value of the multiple flag).
1528
+ */
1529
+ _emitDateSelectedEvent(detail) {
1530
+ /** @type {CustomEvent<T | T[]>} Event emitted on date selection. */
1531
+ this.dispatchEvent(new CustomEvent("dateselected", {
1532
+ detail,
1533
+ composed: true,
1534
+ bubbles: true
1535
+ }));
1536
+ }
1537
+ _emitMonthChange() {
1538
+ const monthchange = (this.wide ? [...this._weeks, ...this._nextMonthWeeks] : this._weeks).flat().sort((a, b) => a.value.localeCompare(b.value));
1539
+ /**
1540
+ * @type {SbbMonthChangeEvent}
1541
+ * Emits when the month changes.
1542
+ * The `range` property contains the days array of the chosen month.
1543
+ */
1544
+ this.dispatchEvent(new SbbMonthChangeEvent(monthchange));
1545
+ }
1546
+ /**
1547
+ * In case of multiple selection, newly added days must be added to the existing ones, without duplication.
1548
+ * If the days to add are exactly the same as the selected ones, the set must be emptied.
1549
+ */
1550
+ _updateSelectedWithMultipleDates(daysToAdd, daysToAddSet, selectedSet) {
1551
+ if (daysToAdd.every((day) => selectedSet.has(day))) daysToAddSet.forEach((day) => selectedSet.delete(day));
1552
+ else daysToAddSet.forEach((day) => selectedSet.add(day));
1553
+ return Array.from(selectedSet);
1554
+ }
1555
+ _setChosenYear() {
1556
+ if (this.view === "month") {
1557
+ let selectedDate;
1558
+ if (this.multiple) selectedDate = this.selected.at(-1);
1559
+ else selectedDate = this.selected;
1560
+ this._chosenYear = this._dateAdapter.getYear(selectedDate ?? this._dateAdapter.today());
1561
+ } else this._chosenYear = void 0;
1562
+ }
1563
+ _assignActiveDate(date) {
1564
+ if (this.min && this._dateAdapter.compareDate(this.min, date) > 0) {
1565
+ this._activeDate = this.min;
1566
+ return;
1567
+ }
1568
+ if (this.max && this._dateAdapter.compareDate(this.max, date) < 0) {
1569
+ this._activeDate = this.max;
1570
+ return;
1571
+ }
1572
+ this._activeDate = date;
1573
+ }
1574
+ /** Goes to the month identified by the shift. */
1575
+ _goToDifferentMonth(months) {
1576
+ this._init(this._dateAdapter.addCalendarMonths(this._activeDate, months));
1577
+ this._emitMonthChange();
1578
+ }
1579
+ _goToDifferentYear(years) {
1580
+ this._chosenYear += years;
1581
+ this._activeDate = this._dateAdapter.createDate(this._chosenYear, this._dateAdapter.getMonth(this._activeDate), this._dateAdapter.getDate(this._activeDate));
1582
+ this._init();
1583
+ }
1584
+ _goToDifferentYearRange(years) {
1585
+ this._init(this._dateAdapter.addCalendarYears(this._activeDate, years));
1586
+ }
1587
+ _prevDisabled(prevDate) {
1588
+ if (!this.min) return false;
1589
+ return this._dateAdapter.compareDate(prevDate, this.min) < 0;
1590
+ }
1591
+ _nextDisabled(nextDate) {
1592
+ if (!this.max) return false;
1593
+ return this._dateAdapter.compareDate(nextDate, this.max) > 0;
1594
+ }
1595
+ /** Checks if the "previous month" button should be disabled. */
1596
+ _previousMonthDisabled() {
1597
+ const prevMonth = this._dateAdapter.addCalendarDays(this._activeDate, this._dateAdapter.getDate(this._activeDate) * -1);
1598
+ return this._prevDisabled(prevMonth);
1599
+ }
1600
+ /** Checks if the "next month" button should be disabled. */
1601
+ _nextMonthDisabled() {
1602
+ let nextMonth = this._dateAdapter.addCalendarMonths(this._activeDate, this._wide ? 2 : 1);
1603
+ nextMonth = this._dateAdapter.createDate(this._dateAdapter.getYear(nextMonth), this._dateAdapter.getMonth(nextMonth), 1);
1604
+ return this._nextDisabled(nextMonth);
1605
+ }
1606
+ /** Checks if the "previous year" button should be disabled. */
1607
+ _previousYearDisabled() {
1608
+ const prevYear = this._dateAdapter.createDate(this._dateAdapter.getYear(this._activeDate) - 1, 12, 31);
1609
+ return this._prevDisabled(prevYear);
1610
+ }
1611
+ /** Checks if the "next year" button should be disabled. */
1612
+ _nextYearDisabled() {
1613
+ const nextYear = this._dateAdapter.createDate(this._dateAdapter.getYear(this._activeDate) + (this._wide ? 2 : 1), 1, 1);
1614
+ return this._nextDisabled(nextYear);
1615
+ }
1616
+ /** Checks if the "previous year" button should be disabled in year view. */
1617
+ _previousYearRangeDisabled() {
1618
+ const prevYear = this._dateAdapter.createDate(this._years[0][0] - 1, 12, 31);
1619
+ return this._prevDisabled(prevYear);
1620
+ }
1621
+ /** Checks if the "next year" button should be disabled in year view. */
1622
+ _nextYearRangeDisabled() {
1623
+ const years = this._wide ? this._nextMonthYears : this._years;
1624
+ const lastYearRange = years[years.length - 1];
1625
+ const lastYear = lastYearRange[lastYearRange.length - 1];
1626
+ const nextYear = this._dateAdapter.createDate(lastYear + 1, 1, 1);
1627
+ return this._nextDisabled(nextYear);
1628
+ }
1629
+ _handleTableBlur(eventTarget) {
1630
+ if (eventTarget?.localName !== "sbb-calendar-day") this._setTabIndex();
1631
+ }
1632
+ _setTabIndex() {
1633
+ Array.from(this._cells.filter((e) => e.tabIndex === 0) ?? []).forEach((day) => day.tabIndex = -1);
1634
+ const firstFocusable = this._getFirstFocusable();
1635
+ if (firstFocusable) firstFocusable.tabIndex = 0;
1636
+ }
1637
+ /** Get the element in the calendar to assign focus. */
1638
+ _getFirstFocusable() {
1639
+ if (this._calendarView === "day") {
1640
+ const selectedOrCurrent = this._cells.find((e) => e.matches(":is(:state(selected),[state--selected])")) ?? this._cells.find((e) => e.matches(":is(:state(current),[state--current])"));
1641
+ return selectedOrCurrent && !selectedOrCurrent.disabled ? selectedOrCurrent : this._getFirstFocusableDay();
1642
+ } else {
1643
+ const selectedOrCurrent = this.shadowRoot?.querySelector(":is(:state(selected),[state--selected])") ?? this.shadowRoot?.querySelector(":is(:state(current),[state--current])");
1644
+ return selectedOrCurrent && !selectedOrCurrent.disabled ? selectedOrCurrent : this.shadowRoot.querySelector(`sbb-calendar-${this._calendarView}:not([disabled])`);
1645
+ }
1646
+ }
1647
+ /**
1648
+ * In `day` view in `vertical` orientation,
1649
+ * if the first of the month is not a Monday, it is not the first rendered element in the table,
1650
+ * so `this.shadowRoot!.querySelector('sbb-calendar-day:not([disabled])')` will return a wrong value.
1651
+ *
1652
+ * To solve this, the element with the lowest `value` is taken (ISO String are ordered).
1653
+ */
1654
+ _getFirstFocusableDay() {
1655
+ const cells = this._cells;
1656
+ const daysInView = cells.filter((e) => !e.disabled);
1657
+ if (!daysInView || daysInView.length === 0) return null;
1658
+ else {
1659
+ const firstElement = daysInView.map((e) => this._dateAdapter.toIso8601(e.value)).sort()[0];
1660
+ return cells.find((e) => e.matches(`[slot="${firstElement}"]`)) ?? null;
1661
+ }
1662
+ }
1663
+ _handleKeyboardEvent(event, day) {
1664
+ if (isArrowKeyOrPageKeysPressed(event)) event.preventDefault();
1665
+ const cells = this._cells;
1666
+ const index = cells.findIndex((e) => e === event.target);
1667
+ let nextEl;
1668
+ if (day) nextEl = this._navigateByKeyboardDayView(event, index, cells, day);
1669
+ else nextEl = this._navigateByKeyboard(event, index, cells);
1670
+ const activeEl = (this._enhancedVariant ? document : this.shadowRoot).activeElement;
1671
+ if (nextEl !== activeEl) {
1672
+ nextEl.tabIndex = 0;
1673
+ nextEl?.focus();
1674
+ activeEl.tabIndex = -1;
1675
+ }
1676
+ }
1677
+ _navigateByKeyboardDayView(evt, index, cells, day) {
1678
+ const arrowsOffset = this.orientation === "horizontal" ? {
1679
+ leftRight: 1,
1680
+ upDown: DAYS_PER_ROW
1681
+ } : {
1682
+ leftRight: DAYS_PER_ROW,
1683
+ upDown: 1
1684
+ };
1685
+ const offsetForVertical = index < this._keyboardNavigationDayViewParameters.firstMonthLength ? this._keyboardNavigationDayViewParameters.firstMonthOffset : this._keyboardNavigationDayViewParameters.secondMonthOffset;
1686
+ switch (evt.key) {
1687
+ case "ArrowUp": return this._findDayArrows(cells, index, day.dateValue, -arrowsOffset.upDown);
1688
+ case "ArrowDown": return this._findDayArrows(cells, index, day.dateValue, arrowsOffset.upDown);
1689
+ case "ArrowLeft": return this._findDayArrows(cells, index, day.dateValue, -arrowsOffset.leftRight);
1690
+ case "ArrowRight": return this._findDayArrows(cells, index, day.dateValue, arrowsOffset.leftRight);
1691
+ case "PageUp": if (this.orientation === "horizontal") {
1692
+ const delta = (+day.dayValue % DAYS_PER_ROW || DAYS_PER_ROW) - +day.dayValue;
1693
+ return this._findDayPageUpDown(cells, index, day, delta, arrowsOffset.upDown);
1694
+ } else {
1695
+ const delta = (Math.ceil((+day.dayValue + offsetForVertical) / DAYS_PER_ROW) - 1) * DAYS_PER_ROW - offsetForVertical + 1 - +day.dayValue;
1696
+ return this._findDayPageUpDown(cells, index, day, delta, arrowsOffset.upDown);
1697
+ }
1698
+ case "PageDown": if (this.orientation === "horizontal") {
1699
+ const monthInBounds = +day.monthValue + 1 > 12 ? 1 : +day.monthValue + 1;
1700
+ const yearInBounds = +day.monthValue + 1 > 12 ? +day.yearValue + 1 : +day.yearValue;
1701
+ const firstNextMonth = this._dateAdapter.createDate(yearInBounds, monthInBounds, 1);
1702
+ const lastOfMonth = this._dateAdapter.getDate(this._dateAdapter.addCalendarDays(firstNextMonth, -1));
1703
+ const delta = Math.trunc((lastOfMonth - +day.dayValue) / DAYS_PER_ROW) * DAYS_PER_ROW;
1704
+ return this._findDayPageUpDown(cells, index, day, delta, -arrowsOffset.upDown);
1705
+ } else {
1706
+ const delta = Math.ceil((+day.dayValue + offsetForVertical) / DAYS_PER_ROW) * DAYS_PER_ROW - offsetForVertical - +day.dayValue;
1707
+ return this._findDayPageUpDown(cells, index, day, delta, -arrowsOffset.upDown);
1708
+ }
1709
+ case "Home": return this._findDayFirst(cells, index, day, 1);
1710
+ case "End": {
1711
+ const monthInBounds = +day.monthValue + 1 > 12 ? 1 : +day.monthValue + 1;
1712
+ const yearInBounds = +day.monthValue + 1 > 12 ? +day.yearValue + 1 : +day.yearValue;
1713
+ const firstNextMonth = this._dateAdapter.createDate(yearInBounds, monthInBounds, 1);
1714
+ return this._findDayLast(cells, index, firstNextMonth);
1715
+ }
1716
+ default: return cells[index];
1717
+ }
1718
+ }
1719
+ _isDayOutOfView(date) {
1720
+ return date < this._keyboardNavigationDayViewParameters.firstDayInView || date > this._keyboardNavigationDayViewParameters.lastDayInView;
1721
+ }
1722
+ _findDayArrows(cells, index, date, delta) {
1723
+ const newDateValue = this._dateAdapter.toIso8601(this._dateAdapter.addCalendarDays(date, delta));
1724
+ if (this._isDayOutOfView(newDateValue)) return cells[index];
1725
+ const nextCell = cells.find((e) => this._mapValueToISODate(e.value) === newDateValue);
1726
+ if (!nextCell || nextCell.disabled) return this._findDayArrows(cells, index, this._dateAdapter.deserialize(newDateValue), delta);
1727
+ return nextCell;
1728
+ }
1729
+ _findDayPageUpDown(cells, index, day, delta, deltaIfDisabled) {
1730
+ const newDateValue = this._dateAdapter.toIso8601(this._dateAdapter.addCalendarDays(day.dateValue, delta));
1731
+ if (this._isDayOutOfView(newDateValue)) return cells[index];
1732
+ const nextCell = cells.find((e) => this._mapValueToISODate(e.value) === newDateValue);
1733
+ if (!nextCell || nextCell.disabled) return this._findDayPageUpDown(cells, index, day, delta + deltaIfDisabled, deltaIfDisabled);
1734
+ return nextCell;
1735
+ }
1736
+ _findDayFirst(cells, index, day, date) {
1737
+ const newDateValue = this._dateAdapter.toIso8601(this._dateAdapter.createDate(+day.yearValue, +day.monthValue, date));
1738
+ if (this._isDayOutOfView(newDateValue)) return cells[index];
1739
+ const nextCell = cells.find((e) => this._mapValueToISODate(e.value) === newDateValue);
1740
+ if (!nextCell || nextCell.disabled) return this._findDayFirst(cells, index, day, date + 1);
1741
+ return nextCell;
1742
+ }
1743
+ _findDayLast(cells, index, firstNextMonth) {
1744
+ const newDateValue = this._dateAdapter.toIso8601(this._dateAdapter.addCalendarDays(firstNextMonth, -1));
1745
+ if (this._isDayOutOfView(newDateValue)) return cells[index];
1746
+ const nextCell = cells.find((e) => this._mapValueToISODate(e.value) === newDateValue);
1747
+ if (!nextCell || nextCell.disabled) return this._findDayLast(cells, index, this._dateAdapter.deserialize(newDateValue));
1748
+ return nextCell;
1749
+ }
1750
+ /**
1751
+ * Gets the index of the element to move to, based on a list of elements (which can be potentially disabled),
1752
+ * the keyboard input and the position of the current element in the list.
1753
+ * In the day view, the `day?: Day` parameter is mandatory for calculation,
1754
+ * while in month and year view it's not due to the fixed amount of rendered cells.
1755
+ */
1756
+ _navigateByKeyboard(evt, index, cells) {
1757
+ const { elementIndexForWideMode, offsetForWideMode, lastElementIndexForWideMode, verticalOffset } = this._calculateParametersForKeyboardNavigation(index, this._calendarView === "year");
1758
+ switch (evt.key) {
1759
+ case "ArrowUp": return this._findNext(cells, index, -verticalOffset);
1760
+ case "ArrowDown": return this._findNext(cells, index, verticalOffset);
1761
+ case "ArrowLeft": return this._findNext(cells, index, -1);
1762
+ case "ArrowRight": return this._findNext(cells, index, 1);
1763
+ case "Home": return this._findFirst(cells, offsetForWideMode);
1764
+ case "PageUp": return this._findFirstOnColumn(cells, elementIndexForWideMode, offsetForWideMode, verticalOffset);
1765
+ case "PageDown": return this._findLastOnColumn(cells, index, lastElementIndexForWideMode, verticalOffset);
1766
+ case "End": return this._findLast(cells, lastElementIndexForWideMode - 1);
1767
+ default: return cells[index];
1768
+ }
1769
+ }
1770
+ /**
1771
+ * Calculates the parameters needed in keyboard navigation in year and month view.
1772
+ * @param index The starting element's index in the cell array.
1773
+ * @param isYearView Whether the displayed `view` is the year one.
1774
+ */
1775
+ _calculateParametersForKeyboardNavigation(index, isYearView) {
1776
+ const elementsPerPage = isYearView ? YEARS_PER_PAGE : MONTHS_PER_PAGE;
1777
+ const offset = Math.trunc(index / elementsPerPage) * elementsPerPage;
1778
+ const indexInView = offset === 0 ? index : index - elementsPerPage;
1779
+ return {
1780
+ verticalOffset: isYearView ? YEARS_PER_ROW : MONTHS_PER_ROW,
1781
+ elementIndexForWideMode: indexInView,
1782
+ offsetForWideMode: index - indexInView,
1783
+ lastElementIndexForWideMode: offset === 0 ? elementsPerPage : elementsPerPage * 2
1784
+ };
1785
+ }
1786
+ /**
1787
+ * Gets the next element of the provided array starting from `index` by adding `delta`.
1788
+ * If the found element is disabled, it continues adding `delta` until it finds an enabled one in the array bounds.
1789
+ */
1790
+ _findNext(days, index, delta) {
1791
+ let nextIndex = index + delta;
1792
+ while (nextIndex < days.length && days[nextIndex]?.disabled) nextIndex += delta;
1793
+ return days[nextIndex] ?? days[index];
1794
+ }
1795
+ /** Find the first enabled element in the provided array. */
1796
+ _findFirst(days, firstOfCurrentMonth) {
1797
+ return !days[firstOfCurrentMonth].disabled ? days[firstOfCurrentMonth] : this._findNext(days, firstOfCurrentMonth, 1);
1798
+ }
1799
+ /** Find the last enabled element in the provided array. */
1800
+ _findLast(days, lastOfCurrentMonth) {
1801
+ return !days[lastOfCurrentMonth].disabled ? days[lastOfCurrentMonth] : this._findNext(days, lastOfCurrentMonth, -1);
1802
+ }
1803
+ /** Find the first enabled element in the same column of the provided array. */
1804
+ _findFirstOnColumn(days, index, offset, verticalOffset) {
1805
+ const nextIndex = index % verticalOffset + offset;
1806
+ return !days[nextIndex].disabled ? days[nextIndex] : this._findNext(days, nextIndex, verticalOffset);
1807
+ }
1808
+ /** Find the last enabled element in the same column of the provided array. */
1809
+ _findLastOnColumn(days, index, offset, verticalOffset) {
1810
+ const nextIndex = index + Math.trunc((offset - index - 1) / verticalOffset) * verticalOffset;
1811
+ return !days[nextIndex].disabled ? days[nextIndex] : this._findNext(days, nextIndex, -verticalOffset);
1812
+ }
1813
+ _resetCalendarViewAndEmitMonthChange(initTransition = false) {
1814
+ this._resetCalendarView(initTransition);
1815
+ this._emitMonthChange();
1816
+ }
1817
+ _resetCalendarView(initTransition = false) {
1818
+ if (this._containingFocus) this._resetFocus = true;
1819
+ this._activeDate = (this.multiple ? this._selected.at(-1) : this._selected) ?? this._dateAdapter.today();
1820
+ this._setChosenYear();
1821
+ this._chosenMonth = void 0;
1822
+ this._init();
1823
+ this._nextCalendarView = this._calendarView = this.view;
1824
+ if (initTransition) this._startTableTransition();
1825
+ }
1826
+ /** Render the view for the day selection. */
1827
+ _renderDayView() {
1828
+ const nextMonthActiveDate = this._wide ? this._dateAdapter.addCalendarMonths(this._activeDate, 1) : void 0;
1829
+ return html`
1830
+ <div class="sbb-calendar__controls">
1831
+ ${this._getArrow("left", () => this._goToDifferentMonth(-1), i18nPreviousMonth[this._language.current], this._previousMonthDisabled())}
1832
+ <div class="sbb-calendar__controls-month">
1833
+ ${this._createLabelForDayView(this._activeDate)}
1834
+ ${this._wide ? this._createLabelForDayView(nextMonthActiveDate) : nothing}
1835
+ <sbb-screen-reader-only role="status">
1836
+ ${this._createAriaLabelForDayView(this._activeDate, nextMonthActiveDate)}
1837
+ </sbb-screen-reader-only>
1838
+ </div>
1839
+ ${this._getArrow("right", () => this._goToDifferentMonth(1), i18nNextMonth[this._language.current], this._nextMonthDisabled())}
1840
+ </div>
1841
+ <div class="sbb-calendar__table-overflow-break">
1842
+ <div class="sbb-calendar__table-container sbb-calendar__table-day-view">
1843
+ ${this.orientation === "horizontal" ? html`
1844
+ ${this._createDayTable(this._weeks, this._weekNumbers)}
1845
+ ${this._wide ? this._createDayTable(this._nextMonthWeeks, this._nextMonthWeekNumbers, true) : nothing}
1846
+ ` : html`
1847
+ ${this._createDayTableVertical(this._weeks, this._weekNumbers)}
1848
+ ${this._wide ? this._createDayTableVertical(this._nextMonthWeeks, this._nextMonthWeekNumbers, nextMonthActiveDate) : nothing}
1849
+ `}
1850
+ </div>
1851
+ </div>
1852
+ `;
1853
+ }
1854
+ /** Creates the label with the month for the daily view. */
1855
+ _createLabelForDayView(d) {
1856
+ const monthLabel = `${this._monthNames[this._dateAdapter.getMonth(d) - 1]} ${this._dateAdapter.getYear(d)}`;
1857
+ return html`
1858
+ <button
1859
+ type="button"
1860
+ class="sbb-calendar__date-selection sbb-calendar__controls-change-date"
1861
+ aria-label="${i18nYearMonthSelection[this._language.current]} ${monthLabel}"
1862
+ @click=${() => {
1863
+ this._resetFocus = true;
1864
+ this._nextCalendarView = "year";
1865
+ this._startTableTransition();
1866
+ }}
1867
+ >
1868
+ ${monthLabel}
1869
+ <sbb-icon name="chevron-small-down-small"></sbb-icon>
1870
+ </button>
1871
+ `;
1872
+ }
1873
+ /** Creates the aria-label for the daily view. */
1874
+ _createAriaLabelForDayView(...dates) {
1875
+ let monthLabel = "";
1876
+ for (const d of dates) if (d) monthLabel += `${this._monthNames[this._dateAdapter.getMonth(d) - 1]} ${this._dateAdapter.getYear(d)} `;
1877
+ return monthLabel;
1878
+ }
1879
+ /** Creates the calendar table for the daily view. */
1880
+ _createDayTable(weeks, weekNumbers, isWideNextMonth = false) {
1881
+ const weeksForSelectMultipleWeekNumbers = (this._wide ? [...this._weeks, ...this._nextMonthWeeks] : isWideNextMonth ? this._nextMonthWeeks : this._weeks).flat();
1882
+ const weeksForSelectMultipleWeekDays = (isWideNextMonth ? this._nextMonthWeeks : this._weeks).flat();
1883
+ return html`
1884
+ <table
1885
+ class="sbb-calendar__table"
1886
+ @focusout=${(event) => this._handleTableBlur(event.relatedTarget)}
1887
+ @animationend=${(e) => this._tableAnimationEnd(e)}
1888
+ >
1889
+ <thead class="sbb-calendar__table-header">
1890
+ <tr>
1891
+ ${this.weekNumbers ? html`<th class="sbb-calendar__table-header-cell"></th>` : nothing}
1892
+ ${this._weekdays.map((weekDay, index) => html`
1893
+ <th class="sbb-calendar__table-header-cell">
1894
+ ${this.multiple ? html`
1895
+ <sbb-calendar-weekday
1896
+ .value=${weekDay}
1897
+ @click=${() => {
1898
+ const days = weeksForSelectMultipleWeekDays.filter((day) => day.weekDayValue === (index + 1) % 7);
1899
+ this._selectMultipleDates(days);
1900
+ }}
1901
+ ></sbb-calendar-weekday>
1902
+ ` : html`
1903
+ <sbb-screen-reader-only>${weekDay.long}</sbb-screen-reader-only>
1904
+ <span aria-hidden="true">${weekDay.narrow}</span>
1905
+ `}
1906
+ </th>
1907
+ `)}
1908
+ </tr>
1909
+ </thead>
1910
+ <tbody class="sbb-calendar__table-body">
1911
+ ${weeks.map((week, rowIndex) => {
1912
+ const firstRowOffset = DAYS_PER_ROW - week.length;
1913
+ if (rowIndex === 0 && firstRowOffset) return html`
1914
+ <tr>
1915
+ ${this.weekNumbers ? html`
1916
+ <td class="sbb-calendar__table-header-cell-vertical">
1917
+ ${this.multiple ? html`
1918
+ <sbb-calendar-weeknumber
1919
+ .value=${weekNumbers[0]}
1920
+ @click=${() => {
1921
+ const days = weeksForSelectMultipleWeekNumbers.filter((day) => day.weekValue === weekNumbers[0]);
1922
+ this._selectMultipleDates(days);
1923
+ }}
1924
+ ></sbb-calendar-weeknumber>
1925
+ ` : html`
1926
+ <sbb-screen-reader-only
1927
+ >${`${i18nCalendarWeekNumber[this._language.current]} ${weekNumbers[0]}`}</sbb-screen-reader-only
1928
+ >
1929
+ <span aria-hidden="true">${weekNumbers[0]}</span>
1930
+ `}
1931
+ </td>
1932
+ ` : nothing}
1933
+ ${[...Array(firstRowOffset).keys()].map(() => html`<td class="sbb-calendar__table-data"></td>`)}
1934
+ ${this._createDayCells(week)}
1935
+ </tr>
1936
+ `;
1937
+ return html`
1938
+ <tr>
1939
+ ${this.weekNumbers ? html`
1940
+ <td class="sbb-calendar__table-header-cell-vertical">
1941
+ ${this.multiple ? html`
1942
+ <sbb-calendar-weeknumber
1943
+ .value=${weekNumbers[rowIndex]}
1944
+ @click=${() => {
1945
+ const days = weeksForSelectMultipleWeekNumbers.filter((day) => day.weekValue === weekNumbers[rowIndex]);
1946
+ this._selectMultipleDates(days);
1947
+ }}
1948
+ ></sbb-calendar-weeknumber>
1949
+ ` : html`
1950
+ <sbb-screen-reader-only
1951
+ >${`${i18nCalendarWeekNumber[this._language.current]} ${weekNumbers[rowIndex]}`}</sbb-screen-reader-only
1952
+ >
1953
+ <span aria-hidden="true">${weekNumbers[rowIndex]}</span>
1954
+ `}
1955
+ </td>
1956
+ ` : nothing}
1957
+ ${this._createDayCells(week)}
1958
+ </tr>
1959
+ `;
1960
+ })}
1961
+ </tbody>
1962
+ </table>
1963
+ `;
1964
+ }
1965
+ /** Creates the table in orientation='vertical'. */
1966
+ _createDayTableVertical(weeks, weekNumbers, nextMonthActiveDate) {
1967
+ const weekOffset = this._dateAdapter.getFirstWeekOffset(nextMonthActiveDate ?? this._activeDate);
1968
+ const weeksForSelectMultipleWeekNumbers = (this._wide ? [...this._weeks, ...this._nextMonthWeeks] : nextMonthActiveDate ? this._nextMonthWeeks : this._weeks).flat();
1969
+ return html`
1970
+ <table
1971
+ class="sbb-calendar__table"
1972
+ @focusout=${(event) => this._handleTableBlur(event.relatedTarget)}
1973
+ @animationend=${(e) => this._tableAnimationEnd(e)}
1974
+ >
1975
+ ${this.weekNumbers ? html`
1976
+ <thead class="sbb-calendar__table-header">
1977
+ <tr>
1978
+ ${nextMonthActiveDate ? nothing : html`<th class="sbb-calendar__table-data"></th>`}
1979
+ ${weekNumbers.map((weekNumber) => html`
1980
+ <th class="sbb-calendar__table-header-cell">
1981
+ ${this.multiple ? html`
1982
+ <sbb-calendar-weeknumber
1983
+ .value=${weekNumber}
1984
+ @click=${() => {
1985
+ const days = weeksForSelectMultipleWeekNumbers.filter((day) => day.weekValue === weekNumber);
1986
+ this._selectMultipleDates(days);
1987
+ }}
1988
+ ></sbb-calendar-weeknumber>
1989
+ ` : html`
1990
+ <sbb-screen-reader-only
1991
+ >${`${i18nCalendarWeekNumber[this._language.current]} ${weekNumber}`}</sbb-screen-reader-only
1992
+ >
1993
+ <span aria-hidden="true">${weekNumber}</span>
1994
+ `}
1995
+ </th>
1996
+ `)}
1997
+ </tr>
1998
+ </thead>
1999
+ ` : nothing}
2000
+ <tbody class="sbb-calendar__table-body">
2001
+ ${weeks.map((week, rowIndex) => {
2002
+ const weekday = this._weekdays[rowIndex];
2003
+ const selectableDays = this._wide ? [...week, ...this._nextMonthWeeks[rowIndex]] : week;
2004
+ return html`
2005
+ <tr>
2006
+ ${nextMonthActiveDate ? nothing : html`
2007
+ <td class="sbb-calendar__table-header-cell-vertical">
2008
+ ${this.multiple ? html`
2009
+ <sbb-calendar-weekday
2010
+ .value=${weekday}
2011
+ @click=${() => this._selectMultipleDates(selectableDays)}
2012
+ >
2013
+ ${weekday.narrow}
2014
+ </sbb-calendar-weekday>
2015
+ ` : html`
2016
+ <sbb-screen-reader-only>${weekday.long}</sbb-screen-reader-only>
2017
+ <span aria-hidden="true">${weekday.narrow}</span>
2018
+ `}
2019
+ </td>
2020
+ `}
2021
+ ${rowIndex < weekOffset ? html`<td class="sbb-calendar__table-data"></td>` : nothing}
2022
+ ${this._createDayCells(week)}
2023
+ </tr>
2024
+ `;
2025
+ })}
2026
+ </tbody>
2027
+ </table>
2028
+ `;
2029
+ }
2030
+ /** Creates the cells for the daily view. */
2031
+ _createDayCells(week) {
2032
+ return week.map((day) => {
2033
+ return html`
2034
+ <td class="sbb-calendar__table-data sbb-calendar__day-cell">
2035
+ <slot name=${day.value}>
2036
+ <sbb-calendar-day
2037
+ slot=${day.value}
2038
+ @click=${() => this._selectDate(day.dateValue)}
2039
+ @keydown=${(evt) => this._handleKeyboardEvent(evt, day)}
2040
+ ></sbb-calendar-day>
2041
+ </slot>
2042
+ </td>
2043
+ `;
2044
+ });
2045
+ }
2046
+ /** Render the view for the month selection. */
2047
+ _renderMonthView() {
2048
+ return html`
2049
+ <div class="sbb-calendar__controls">
2050
+ ${this._getArrow("left", () => this._goToDifferentYear(-1), i18nPreviousYear[this._language.current], this._previousYearDisabled())}
2051
+ <div class="sbb-calendar__controls-month">${this._createLabelForMonthView()}</div>
2052
+ ${this._getArrow("right", () => this._goToDifferentYear(1), i18nNextYear[this._language.current], this._nextYearDisabled())}
2053
+ </div>
2054
+ <div class="sbb-calendar__table-overflow-break">
2055
+ <div class="sbb-calendar__table-container sbb-calendar__table-month-view">
2056
+ ${this._createMonthTable(this._months, this._chosenYear)}
2057
+ ${this._wide ? this._createMonthTable(this._months, this._chosenYear + 1) : nothing}
2058
+ </div>
2059
+ </div>
2060
+ `;
2061
+ }
2062
+ /** Creates the label with the year for the monthly view. */
2063
+ _createLabelForMonthView() {
2064
+ return html` <button
2065
+ type="button"
2066
+ id="sbb-calendar__month-selection"
2067
+ class="sbb-calendar__controls-change-date"
2068
+ aria-label=${`${i18nCalendarDateSelection[this._language.current]} ${this._chosenYear}`}
2069
+ @click=${() => this._resetCalendarViewAndEmitMonthChange(true)}
2070
+ >
2071
+ ${this._chosenYear} ${this._wide ? ` - ${this._chosenYear + 1}` : nothing}
2072
+ <sbb-icon name="chevron-small-up-small"></sbb-icon>
2073
+ </button>
2074
+ <sbb-screen-reader-only role="status"> ${this._chosenYear} </sbb-screen-reader-only>`;
2075
+ }
2076
+ /** Creates the table for the month selection view. */
2077
+ _createMonthTable(months, year) {
2078
+ return html`
2079
+ <table
2080
+ class="sbb-calendar__table"
2081
+ @animationend=${(e) => this._tableAnimationEnd(e)}
2082
+ >
2083
+ ${this._wide ? html`<thead class="sbb-calendar__table-header" aria-hidden="true">
2084
+ <tr>
2085
+ <th class="sbb-calendar__table-header-cell" colspan=${MONTHS_PER_ROW}>${year}</th>
2086
+ </tr>
2087
+ </thead>` : nothing}
2088
+ <tbody class="sbb-calendar__table-body">
2089
+ ${months.map((row) => html`
2090
+ <tr>
2091
+ ${row.map((month) => {
2092
+ return html`
2093
+ <td class="sbb-calendar__table-data">
2094
+ <sbb-calendar-month
2095
+ .value="${year}-${month.value}"
2096
+ @click=${() => this._onMonthSelection(month.monthValue, year)}
2097
+ @keydown=${(evt) => this._handleKeyboardEvent(evt)}
2098
+ >
2099
+ </sbb-calendar-month>
2100
+ </td>
2101
+ `;
2102
+ })}
2103
+ </tr>
2104
+ `)}
2105
+ </tbody>
2106
+ </table>
2107
+ `;
2108
+ }
2109
+ /** Select the month and change the view to day selection. */
2110
+ _onMonthSelection(month, year) {
2111
+ this._chosenMonth = month;
2112
+ this._nextCalendarView = "day";
2113
+ this._init(this._dateAdapter.createDate(year, this._chosenMonth, this._dateAdapter.getDate(this._activeDate)));
2114
+ this._startTableTransition();
2115
+ this._emitMonthChange();
2116
+ }
2117
+ /** Render the view for the year selection. */
2118
+ _renderYearView() {
2119
+ return html`
2120
+ <div class="sbb-calendar__controls">
2121
+ ${this._getArrow("left", () => this._goToDifferentYearRange(-YEARS_PER_PAGE), i18nPreviousYearRange(YEARS_PER_PAGE)[this._language.current], this._previousYearRangeDisabled())}
2122
+ <div class="sbb-calendar__controls-month">${this._createLabelForYearView()}</div>
2123
+ ${this._getArrow("right", () => this._goToDifferentYearRange(YEARS_PER_PAGE), i18nNextYearRange(YEARS_PER_PAGE)[this._language.current], this._nextYearRangeDisabled())}
2124
+ </div>
2125
+ <div class="sbb-calendar__table-overflow-break">
2126
+ <div class="sbb-calendar__table-container sbb-calendar__table-year-view">
2127
+ ${this._createYearTable(this._years)}
2128
+ ${this._wide ? this._createYearTable(this._nextMonthYears, true) : nothing}
2129
+ </div>
2130
+ </div>
2131
+ `;
2132
+ }
2133
+ /** Creates the button arrow for all the views. */
2134
+ _getArrow(direction, click, ariaLabel, disabled) {
2135
+ return html`<sbb-secondary-button
2136
+ size="m"
2137
+ icon-name="chevron-small-${direction}-small"
2138
+ aria-label=${ariaLabel}
2139
+ @click=${click}
2140
+ ?disabled=${disabled}
2141
+ id="sbb-calendar__controls-${direction === "left" ? "previous" : "next"}"
2142
+ ></sbb-secondary-button>`;
2143
+ }
2144
+ /** Creates the label with the year range for the yearly view. */
2145
+ _createLabelForYearView() {
2146
+ const firstYear = this._years.flat()[0];
2147
+ const lastYearArray = (this._wide ? this._nextMonthYears : this._years).flat();
2148
+ const yearLabel = `${firstYear} - ${lastYearArray[lastYearArray.length - 1]}`;
2149
+ return html`
2150
+ <button
2151
+ type="button"
2152
+ id="sbb-calendar__year-selection"
2153
+ class="sbb-calendar__controls-change-date"
2154
+ aria-label="${i18nCalendarDateSelection[this._language.current]} ${yearLabel}"
2155
+ @click=${() => this._resetCalendarViewAndEmitMonthChange(true)}
2156
+ >
2157
+ ${yearLabel}
2158
+ <sbb-icon name="chevron-small-up-small"></sbb-icon>
2159
+ </button>
2160
+ <sbb-screen-reader-only role="status"> ${yearLabel} </sbb-screen-reader-only>
2161
+ `;
2162
+ }
2163
+ /** Creates the table for the year selection view. */
2164
+ _createYearTable(years, shiftRight = false) {
2165
+ return html` <table
2166
+ class="sbb-calendar__table"
2167
+ @animationend=${(e) => this._tableAnimationEnd(e)}
2168
+ >
2169
+ <tbody class="sbb-calendar__table-body">
2170
+ ${years.map((row) => html` <tr>
2171
+ ${row.map((year) => {
2172
+ return html`
2173
+ <td class="sbb-calendar__table-data">
2174
+ <sbb-calendar-year
2175
+ .value=${String(year)}
2176
+ @keydown=${(evt) => this._handleKeyboardEvent(evt)}
2177
+ @click=${() => this._onYearSelection(year, shiftRight)}
2178
+ >
2179
+ </sbb-calendar-year>
2180
+ </td>
2181
+ `;
2182
+ })}
2183
+ </tr>`)}
2184
+ </tbody>
2185
+ </table>`;
2186
+ }
2187
+ /** Select the year and change the view to month selection. */
2188
+ _onYearSelection(year, rightSide) {
2189
+ this._chosenYear = rightSide ? year - 1 : year;
2190
+ this._nextCalendarView = "month";
2191
+ this._assignActiveDate(this._dateAdapter.createDate(this._chosenYear, this._dateAdapter.getMonth(this._activeDate), this._dateAdapter.getDate(this._activeDate)));
2192
+ this._startTableTransition();
2193
+ }
2194
+ _getView() {
2195
+ if (isServer || this.hydrationRequired) return html`${nothing}`;
2196
+ switch (this._calendarView) {
2197
+ case "year": return this._renderYearView();
2198
+ case "month": return this._renderMonthView();
2199
+ default: return this._renderDayView();
2200
+ }
2201
+ }
2202
+ _tableAnimationEnd(event) {
2203
+ const table = event.target;
2204
+ if (event.animationName === "hide") {
2205
+ table.classList.remove("sbb-calendar__table-hide");
2206
+ if (this._containingFocus) this._resetFocus = true;
2207
+ this._calendarView = this._nextCalendarView;
2208
+ } else if (event.animationName === "show") this.internals.states.delete("transition");
2209
+ }
2210
+ _startTableTransition() {
2211
+ this.internals.states.add("transition");
2212
+ this.shadowRoot?.querySelectorAll("table")?.forEach((e) => e.classList.toggle("sbb-calendar__table-hide"));
2213
+ }
2214
+ render() {
2215
+ return html`<div class="sbb-calendar__wrapper">${this._getView()}</div>`;
2216
+ }
2217
+ };
2218
+ })();
2219
+ //#endregion
2220
+ export { SbbMonthChangeEvent as n, SbbCalendarElement as t };
2221
+
2222
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY2FsZW5kYXIuY29tcG9uZW50LTBQbDdxRVhMLmpzIiwibmFtZXMiOltdLCJzb3VyY2VzIjpbIi4uLy4uLy4uL25vZGVfbW9kdWxlcy9kYXRlLWZucy9jb25zdGFudHMuanMiLCIuLi8uLi8uLi9ub2RlX21vZHVsZXMvZGF0ZS1mbnMvY29uc3RydWN0RnJvbS5qcyIsIi4uLy4uLy4uL25vZGVfbW9kdWxlcy9kYXRlLWZucy90b0RhdGUuanMiLCIuLi8uLi8uLi9ub2RlX21vZHVsZXMvZGF0ZS1mbnMvYWRkRGF5cy5qcyIsIi4uLy4uLy4uL25vZGVfbW9kdWxlcy9kYXRlLWZucy9fbGliL2RlZmF1bHRPcHRpb25zLmpzIiwiLi4vLi4vLi4vbm9kZV9tb2R1bGVzL2RhdGUtZm5zL3N0YXJ0T2ZXZWVrLmpzIiwiLi4vLi4vLi4vbm9kZV9tb2R1bGVzL2RhdGUtZm5zL19saWIvbm9ybWFsaXplRGF0ZXMuanMiLCIuLi8uLi8uLi9ub2RlX21vZHVsZXMvZGF0ZS1mbnMvYWRkV2Vla3MuanMiLCIuLi8uLi8uLi9ub2RlX21vZHVsZXMvZGF0ZS1mbnMvZW5kT2ZNb250aC5qcyIsIi4uLy4uLy4uL25vZGVfbW9kdWxlcy9kYXRlLWZucy9fbGliL25vcm1hbGl6ZUludGVydmFsLmpzIiwiLi4vLi4vLi4vbm9kZV9tb2R1bGVzL2RhdGUtZm5zL2VhY2hXZWVrT2ZJbnRlcnZhbC5qcyIsIi4uLy4uLy4uL25vZGVfbW9kdWxlcy9kYXRlLWZucy9zdGFydE9mTW9udGguanMiLCIuLi8uLi8uLi9ub2RlX21vZHVsZXMvZGF0ZS1mbnMvZ2V0V2Vla1llYXIuanMiLCIuLi8uLi8uLi9ub2RlX21vZHVsZXMvZGF0ZS1mbnMvc3RhcnRPZldlZWtZZWFyLmpzIiwiLi4vLi4vLi4vbm9kZV9tb2R1bGVzL2RhdGUtZm5zL2dldFdlZWsuanMiLCIuLi8uLi8uLi9zcmMvZWxlbWVudHMvY2FsZW5kYXIvY2FsZW5kYXIvY2FsZW5kYXIuc2Nzcz9saXQmaW5saW5lIiwiLi4vLi4vLi4vc3JjL2VsZW1lbnRzL2NhbGVuZGFyL2NhbGVuZGFyL2NhbGVuZGFyLmNvbXBvbmVudC50cyJdLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIEBtb2R1bGUgY29uc3RhbnRzXG4gKiBAc3VtbWFyeSBVc2VmdWwgY29uc3RhbnRzXG4gKiBAZGVzY3JpcHRpb25cbiAqIENvbGxlY3Rpb24gb2YgdXNlZnVsIGRhdGUgY29uc3RhbnRzLlxuICpcbiAqIFRoZSBjb25zdGFudHMgY291bGQgYmUgaW1wb3J0ZWQgZnJvbSBgZGF0ZS1mbnMvY29uc3RhbnRzYDpcbiAqXG4gKiBgYGB0c1xuICogaW1wb3J0IHsgbWF4VGltZSwgbWluVGltZSB9IGZyb20gXCIuL2NvbnN0YW50cy9kYXRlLWZucy9jb25zdGFudHNcIjtcbiAqXG4gKiBmdW5jdGlvbiBpc0FsbG93ZWRUaW1lKHRpbWUpIHtcbiAqICAgcmV0dXJuIHRpbWUgPD0gbWF4VGltZSAmJiB0aW1lID49IG1pblRpbWU7XG4gKiB9XG4gKiBgYGBcbiAqL1xuXG4vKipcbiAqIEBjb25zdGFudFxuICogQG5hbWUgZGF5c0luV2Vla1xuICogQHN1bW1hcnkgRGF5cyBpbiAxIHdlZWsuXG4gKi9cbmV4cG9ydCBjb25zdCBkYXlzSW5XZWVrID0gNztcblxuLyoqXG4gKiBAY29uc3RhbnRcbiAqIEBuYW1lIGRheXNJblllYXJcbiAqIEBzdW1tYXJ5IERheXMgaW4gMSB5ZWFyLlxuICpcbiAqIEBkZXNjcmlwdGlvblxuICogSG93IG1hbnkgZGF5cyBpbiBhIHllYXIuXG4gKlxuICogT25lIHllYXJzIGVxdWFscyAzNjUuMjQyNSBkYXlzIGFjY29yZGluZyB0byB0aGUgZm9ybXVsYTpcbiAqXG4gKiA+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/LmluKTtcbiAgY29uc3QgZGlmZiA9ICtzdGFydE9mV2VlayhfZGF0ZSwgb3B0aW9ucykgLSArc3RhcnRPZldlZWtZZWFyKF9kYXRlLCBvcHRpb25zKTtcblxuICAvLyBSb3VuZCB0aGUgbnVtYmVyIG9mIHdlZWtzIHRvIHRoZSBuZWFyZXN0IGludGVnZXIgYmVjYXVzZSB0aGUgbnVtYmVyIG9mXG4gIC8vIG1pbGxpc2Vjb25kcyBpbiBhIHdlZWsgaXMgbm90IGNvbnN0YW50IChlLmcuIGl0J3MgZGlmZmVyZW50IGluIHRoZSB3ZWVrIG9mXG4gIC8vIHRoZSBkYXlsaWdodCBzYXZpbmcgdGltZSBjbG9jayBzaGlmdCkuXG4gIHJldHVybiBNYXRoLnJvdW5kKGRpZmYgLyBtaWxsaXNlY29uZHNJbldlZWspICsgMTtcbn1cblxuLy8gRmFsbGJhY2sgZm9yIG1vZHVsYXJpemVkIGltcG9ydHM6XG5leHBvcnQgZGVmYXVsdCBnZXRXZWVrO1xuIiwiQHVzZSAnLi4vLi4vY29yZS9zdHlsZXMnIGFzIHNiYjtcblxuOmhvc3Qge1xuICBkaXNwbGF5OiBibG9jaztcblxuICAvLyBXZSBhZGQgd2lkdGggZGVmaW5pdGlvbiB0byBob3N0LCB0byBtYWtlIG92ZXJ3cml0aW5nIGVhc3kgZm9yIGNvbnN1bWVycy5cbiAgd2lkdGg6IG1heC1jb250ZW50O1xuXG4gIC0tc2JiLWNhbGVuZGFyLWNlbGwtc2l6ZTogI3tzYmIucHgtdG8tcmVtLWJ1aWxkKDQ0KX07XG5cbiAgLy8gTWF0Y2ggRmlnbWEgZGVzaWduXG4gIC0tc2JiLWNhbGVuZGFyLWNlbGwtYm9yZGVyLXJhZGl1czogY2FsYyhcbiAgICB2YXIoLS1zYmItYm9yZGVyLXJhZGl1cy00eCkgKyB2YXIoLS1zYmItY2FsZW5kYXItY2VsbC1ib3JkZXItd2lkdGgpXG4gICk7XG4gIC0tc2JiLWNhbGVuZGFyLWNlbGwtdHJhbnNpdGlvbi1kdXJhdGlvbjogdmFyKFxuICAgIC0tc2JiLWRpc2FibGUtYW5pbWF0aW9uLWR1cmF0aW9uLFxuICAgIHZhcigtLXNiYi1hbmltYXRpb24tZHVyYXRpb24tMngpXG4gICk7XG5cbiAgQGluY2x1ZGUgc2JiLm1xKCRmcm9tOiBzbWFsbCkge1xuICAgIC0tc2JiLWNhbGVuZGFyLWNvbnRyb2wtdmlldy1jaGFuZ2UtaGVpZ2h0OiAje3NiYi5weC10by1yZW0tYnVpbGQoNDgpfTtcbiAgfVxufVxuXG46aG9zdCg6c3RhdGUoZW5oYW5jZWQpKSB7XG4gIC0tc2JiLWNhbGVuZGFyLWNlbGwtcGFkZGluZy1ibG9jay1lbmQ6IHZhcigtLXNiYi1zcGFjaW5nLWZpeGVkLTN4KTtcbn1cblxuLnNiYi1jYWxlbmRhcl9fd3JhcHBlciB7XG4gIHdpZHRoOiAxMDAlO1xuICBkaXNwbGF5OiBibG9jaztcbiAgdHJhbnNpdGlvbi1kdXJhdGlvbjogdmFyKC0tc2JiLWNhbGVuZGFyLWNlbGwtdHJhbnNpdGlvbi1kdXJhdGlvbik7XG59XG5cbi5zYmItY2FsZW5kYXJfX2NvbnRyb2xzIHtcbiAgd2lkdGg6IDEwMCU7XG4gIGRpc3BsYXk6IGlubGluZS1mbGV4O1xuICBhbGlnbi1pdGVtczogY2VudGVyO1xuICBnYXA6IHZhcigtLXNiYi1jYWxlbmRhci1jb250cm9sLWdhcCk7XG4gIG1hcmdpbi1ibG9jay1lbmQ6IHZhcigtLXNiYi1jYWxlbmRhci1jb250cm9sLW1hcmdpbi1ibG9jay1lbmQpO1xufVxuXG4uc2JiLWNhbGVuZGFyX19jb250cm9scy1tb250aCB7XG4gIHdpZHRoOiAxMDAlO1xuICBkaXNwbGF5OiBmbGV4O1xuICBnYXA6IHZhcigtLXNiYi1jYWxlbmRhci10YWJsZXMtZ2FwKTtcbn1cblxuI3NiYi1jYWxlbmRhcl9fY29udHJvbHMtcHJldmlvdXMsXG4jc2JiLWNhbGVuZGFyX19jb250cm9scy1uZXh0IHtcbiAgLXdlYmtpdC10YXAtaGlnaGxpZ2h0LWNvbG9yOiB0cmFuc3BhcmVudDtcbn1cblxuLnNiYi1jYWxlbmRhcl9fY29udHJvbHMtY2hhbmdlLWRhdGUge1xuICBAaW5jbHVkZSBzYmIuYnV0dG9uLXJlc2V0O1xuXG4gIGRpc3BsYXk6IGZsZXg7XG4gIGFsaWduLWl0ZW1zOiBjZW50ZXI7XG4gIG1hcmdpbjogYXV0bztcbiAgaGVpZ2h0OiB2YXIoLS1zYmItY2FsZW5kYXItY29udHJvbC12aWV3LWNoYW5nZS1oZWlnaHQpO1xuICBmb250LXNpemU6IHZhcigtLXNiYi10ZXh0LWZvbnQtc2l6ZS1zKTtcbiAgbGV0dGVyLXNwYWNpbmc6IHZhcigtLXNiYi10eXBvLWxldHRlci1zcGFjaW5nLXRleHQpO1xuICB0ZXh0LXRyYW5zZm9ybTogY2FwaXRhbGl6ZTtcbiAgY3Vyc29yOiB2YXIoLS1zYmItY3Vyc29yLXBvaW50ZXIpO1xuICBwYWRkaW5nLWlubGluZTogdmFyKC0tc2JiLWNhbGVuZGFyLWNvbnRyb2wtdmlldy1jaGFuZ2UtcGFkZGluZy1pbmxpbmUpO1xuICBib3JkZXItcmFkaXVzOiB2YXIoLS1zYmItYm9yZGVyLXJhZGl1cy1pbmZpbml0eSk7XG4gIGJhY2tncm91bmQtY29sb3I6IHZhcigtLXNiYi1jYWxlbmRhci1jb250cm9sLXZpZXctY2hhbmdlLWJhY2tncm91bmQpO1xuICBjb2xvcjogdmFyKC0tc2JiLWNhbGVuZGFyLWNvbnRyb2wtdmlldy1jaGFuZ2UtY29sb3IpO1xuICB0cmFuc2l0aW9uLWR1cmF0aW9uOiB2YXIoLS1zYmItY2FsZW5kYXItY2VsbC10cmFuc2l0aW9uLWR1cmF0aW9uKTtcbiAgdHJhbnNpdGlvbi10aW1pbmctZnVuY3Rpb246IHZhcigtLXNiYi1jYWxlbmRhci1jZWxsLXRyYW5zaXRpb24tZWFzaW5nLWZ1bmN0aW9uKTtcbiAgdHJhbnNpdGlvbi1wcm9wZXJ0eTogYmFja2dyb3VuZC1jb2xvciwgcGFkZGluZy1ibG9jay1lbmQ7XG5cbiAgJjpkaXNhYmxlZCB7XG4gICAgLS1zYmItY2FsZW5kYXItY29udHJvbC12aWV3LWNoYW5nZS1iYWNrZ3JvdW5kOiBsaWdodC1kYXJrKFxuICAgICAgdmFyKC0tc2JiLWNvbG9yLW1pbGspLFxuICAgICAgdmFyKC0tc2JiLWNvbG9yLWFudGhyYWNpdGUpXG4gICAgKTtcbiAgICAtLXNiYi1jYWxlbmRhci1jb250cm9sLXZpZXctY2hhbmdlLWNvbG9yOiBsaWdodC1kYXJrKFxuICAgICAgdmFyKC0tc2JiLWNvbG9yLWdyYW5pdGUpLFxuICAgICAgdmFyKC0tc2JiLWNvbG9yLWFsdW1pbml1bSlcbiAgICApO1xuXG4gICAgY3Vyc29yOiB1bnNldDtcbiAgfVxuXG4gICY6Zm9jdXMtdmlzaWJsZSB7XG4gICAgQGluY2x1ZGUgc2JiLmZvY3VzLW91dGxpbmU7XG5cbiAgICBvdXRsaW5lLW9mZnNldDogdmFyKC0tc2JiLXNwYWNpbmctZml4ZWQtMXgpO1xuICB9XG5cbiAgQGluY2x1ZGUgc2JiLmhvdmVyLW1xIHtcbiAgICAmOm5vdCg6YWN0aXZlLCA6ZGlzYWJsZWQpOmhvdmVyIHtcbiAgICAgIHBhZGRpbmctYmxvY2stZW5kOiB2YXIoLS1zYmItY2FsZW5kYXItY2VsbC1ob3Zlci1zaGlmdCk7XG4gICAgfVxuICB9XG5cbiAgJjpub3QoOmRpc2FibGVkKTphY3RpdmUge1xuICAgIC0tc2JiLWNhbGVuZGFyLWNvbnRyb2wtdmlldy1jaGFuZ2UtYmFja2dyb3VuZDogdmFyKC0tc2JiLWJhY2tncm91bmQtY29sb3ItMyk7XG4gIH1cbn1cblxuLnNiYi1jYWxlbmRhcl9fdGFibGUtbW9udGgtdmlldyxcbi5zYmItY2FsZW5kYXJfX3RhYmxlLXllYXItdmlldyB7XG4gIC0tc2JiLWNhbGVuZGFyLXRhYmxlLWNvbHVtbi1zcGFjZXM6IDY7XG59XG5cbi5zYmItY2FsZW5kYXJfX3RhYmxlLW92ZXJmbG93LWJyZWFrIHtcbiAgLy8gV2UgaGF2ZSB0byBjdXQgdGhlIG5lZ2F0aXZlIG1hcmdpbiB3aGljaCBpcyBidWlsdCBieSB0aGUgY29udGFpbmluZyBzYmItY2FsZW5kYXJfX3RhYmxlLWNvbnRhaW5lci5cbiAgLy8gT3ZlcmZsb3c6IGhpZGRlbiB3b3VsZCBjdXQgdGhlIGZvY3VzIG91dGxpbmUuIFRoZXJlZm9yZSB3ZSB0YWtlIGBjb250YWluYCBoZXJlLlxuICBjb250YWluOiBsYXlvdXQ7XG59XG5cbi5zYmItY2FsZW5kYXJfX3RhYmxlLWNvbnRhaW5lciB7XG4gIGRpc3BsYXk6IGZsZXg7XG4gIGdhcDogdmFyKC0tc2JiLWNhbGVuZGFyLXRhYmxlcy1nYXApO1xuICBtYXJnaW4taW5saW5lOiB2YXIoLS1zYmItY2FsZW5kYXItbWFyZ2luKTtcblxuICAvLyBUaGUgcGFkZGluZyBvZiB0aGUgZmlyc3QgYW5kIGxhc3QgY29sdW1uIHNob3VsZCBub3QgYmUgdmlzaWJsZSBpZiBjYWxlbmRhciBpcyBzdHJldGNoZWQuXG4gIC8vIFRoZXJlZm9yZSB3ZSBuZWVkIGEgbmVnYXRpdmUgaW5saW5lIG1hcmdpbi5cbiAgLy8gQXMgd2UgZG9uJ3Qgd2FudCB0byBzcXVlZXplLCB0aGUgbWFyZ2luIHNob3VsZCBuZXZlciBiZSBncmVhdGVyIHRoYW4gemVyby5cblxuICAvLyBNaW4gd2lkdGggaXMgZXF1YWxzIHRvIHRoZSBub3JtYWwgd2lkdGggb2YgdGhlIGNhbGVuZGFyXG4gIC0tc2JiLWNhbGVuZGFyLW1pbi13aWR0aDogY2FsYyg3ICogdmFyKC0tc2JiLWNhbGVuZGFyLWNlbGwtc2l6ZSkpO1xuXG4gIC8vIFRoZSBvdmVyZmxvdyB2YXJpYWJsZSBpcyBlcXVhbHMgdGhlIGRpZmZlcmVuY2UgYmV0d2VlbiB0aGUgYWN0dWFsIHdpZHRoIGFuZCB0aGUgbWluIHdpZHRoLlxuICAtLXNiYi1jYWxlbmRhci1vdmVyZmxvdzogY2FsYygxMDAlIC0gdmFyKC0tc2JiLWNhbGVuZGFyLW1pbi13aWR0aCkpO1xuXG4gIC8vIFRoZSBzdGFydCBvZmZzZXQgaXMgbmVnYXRpdmUgbWFyZ2luIHdoaWNoIHNob3VsZCBvdmVybGFwIHRoZSBwYXJlbnQgY29udGFpbmVyLiBTaG91bGQgbmV2ZXIgYmUgYSBwb3NpdGl2ZSB2YWx1ZS5cbiAgLS1zYmItY2FsZW5kYXItc3RhcnQtb2Zmc2V0OiBtaW4oXG4gICAgMHB4LFxuICAgIC0xICogKHZhcigtLXNiYi1jYWxlbmRhci1vdmVyZmxvdykgLyB2YXIoLS1zYmItY2FsZW5kYXItdGFibGUtY29sdW1uLXNwYWNlcykpXG4gICk7XG4gIC0tc2JiLWNhbGVuZGFyLW1hcmdpbjogdmFyKC0tc2JiLWNhbGVuZGFyLXN0YXJ0LW9mZnNldCk7XG5cbiAgOmhvc3QoOnN0YXRlKHdpZGUpKSAmIHtcbiAgICAtLXNiYi1jYWxlbmRhci1taW4td2lkdGg6IGNhbGMoXG4gICAgICAyICogNyAqIHZhcigtLXNiYi1jYWxlbmRhci1jZWxsLXNpemUpICsgdmFyKC0tc2JiLWNhbGVuZGFyLXRhYmxlcy1nYXApXG4gICAgKTtcbiAgICAtLXNiYi1jYWxlbmRhci1tYXJnaW46IGNhbGMoMC41ICogdmFyKC0tc2JiLWNhbGVuZGFyLXN0YXJ0LW9mZnNldCkpO1xuICB9XG5cbiAgOmhvc3QoW29yaWVudGF0aW9uPSdob3Jpem9udGFsJ11bd2Vlay1udW1iZXJzXSkgJiB7XG4gICAgLS1zYmItY2FsZW5kYXItbWluLXdpZHRoOiBjYWxjKDggKiB2YXIoLS1zYmItY2FsZW5kYXItY2VsbC1zaXplKSk7XG4gIH1cblxuICA6aG9zdChbb3JpZW50YXRpb249J2hvcml6b250YWwnXVt3ZWVrLW51bWJlcnNdOnN0YXRlKHdpZGUpKSAmIHtcbiAgICAtLXNiYi1jYWxlbmRhci1taW4td2lkdGg6IGNhbGMoXG4gICAgICAyICogOCAqIHZhcigtLXNiYi1jYWxlbmRhci1jZWxsLXNpemUpICsgdmFyKC0tc2JiLWNhbGVuZGFyLXRhYmxlcy1nYXApXG4gICAgKTtcbiAgfVxuXG4gIC8vIFRoZSBjb250YWluZXIncyBtaW4td2lkdGggaXMgc2V0IGluIHZlcnRpY2FsIG1vZGU7XG4gIC8vIHRoZSB2YWx1ZSBgY2FsYyg3ICogdmFyKC0tc2JiLWNhbGVuZGFyLWNlbGwtc2l6ZSkpYCBpcyBmaW5lXG4gIC8vIGNvbnNpZGVyaW5nIHRoYXQgdGhlIG1heGltdW0gbnVtYmVyIG9mIHdlZWtzIGluIGEgbW9udGggaXMgNiwgcGx1cyAxIGZvciB0aGUgaGVhZGVyIGF0IHRoZSB0YWJsZSdzIGxlZnQgc2lkZS5cbiAgOmhvc3QoW29yaWVudGF0aW9uPSd2ZXJ0aWNhbCddKSAmIHtcbiAgICBtaW4td2lkdGg6IHZhcigtLXNiYi1jYWxlbmRhci1taW4td2lkdGgpO1xuXG4gICAgLy8gVGhlIHZhcmlhYmxlIHRoYXQgZGVmaW5lcyB0aGUgbWFyZ2luIGlzIHNldCB0byAwIHRvIGNvcnJlY3RseSBkaXNwbGF5IHRoZSB0YWJsZSBpZiBjb25zdW1lcnMgc2V0IGEgY3VzdG9tIHdpZHRoLlxuICAgIC0tc2JiLWNhbGVuZGFyLXN0YXJ0LW9mZnNldDogMDtcbiAgfVxuXG4gIC8vIFRoZSBtaW4td2lkdGggaW4gd2lkZSBpcyBjYWxjdWxhdGVkIGFzIChtYXggbnVtYmVyIG9mIHdlZWtzIGluIGEgbW9udGgpICogKGRpc3BsYXllZCBtb250aHMpICsgKGhlYWRlciBjb2x1bW4pID0gNiAqIDIgKyAxID0gMTMuXG4gIDpob3N0KFtvcmllbnRhdGlvbj0ndmVydGljYWwnXTpzdGF0ZSh3aWRlKSkgJiB7XG4gICAgLS1zYmItY2FsZW5kYXItbWluLXdpZHRoOiBjYWxjKFxuICAgICAgMTMgKiB2YXIoLS1zYmItY2FsZW5kYXItY2VsbC1zaXplKSArIHZhcigtLXNiYi1jYWxlbmRhci10YWJsZXMtZ2FwKVxuICAgICk7XG4gIH1cbn1cblxuLnNiYi1jYWxlbmRhcl9fdGFibGUge1xuICB3aWR0aDogMTAwJTtcbiAgYm9yZGVyLWNvbGxhcHNlOiBjb2xsYXBzZTtcbiAgaGVpZ2h0OiBtYXgtY29udGVudDtcblxuICBhbmltYXRpb246IHtcbiAgICBuYW1lOiBzaG93O1xuICAgIGR1cmF0aW9uOiB2YXIoLS1zYmItY2FsZW5kYXItdGFibGUtYW5pbWF0aW9uLWR1cmF0aW9uKTtcbiAgfVxuXG4gICYuc2JiLWNhbGVuZGFyX190YWJsZS1oaWRlIHtcbiAgICAtLXNiYi1jYWxlbmRhci1jZWxsLXRyYW5zaXRpb24tZHVyYXRpb246IDBtcztcblxuICAgIGFuaW1hdGlvbjoge1xuICAgICAgbmFtZTogaGlkZTtcbiAgICAgIGR1cmF0aW9uOiB2YXIoLS1zYmItY2FsZW5kYXItdGFibGUtYW5pbWF0aW9uLWR1cmF0aW9uKTtcbiAgICB9XG4gIH1cblxuICA6aG9zdCg6bm90KDpzdGF0ZSh3aWRlKSkpICYge1xuICAgIC8vIER1ZSB0byBhIFNhZmFyaSBpT1MgcmVuZGVyaW5nIGJ1ZyB3ZSBuZWVkIHRvIGRlZmluZSBtaW4td2lkdGggYXMgd2VsbC5cbiAgICAvLyBPdGhlcndpc2UsIGFmdGVyIG9yaWVudGF0aW9uIGNoYW5nZSwgdGhlcmUgaXMgYSB3cm9uZyB3aWR0aCBpZiBwbGFjZWQgaW4gYW4gc2JiLWRpYWxvZy5cbiAgICBtaW4td2lkdGg6IDEwMCU7XG4gIH1cbn1cblxuLnNiYi1jYWxlbmRhcl9fdGFibGUtaGVhZGVyIHtcbiAgLy8gRGVwZW5kaW5nIHdoZXJlIGluIHRoZSBET00gaXQncyB1c2VkLCBpdCdzIHBvc3NpYmxlIHRoYXQgYnJvd3NlciBkZWZhdWx0cyBzZXQgdGV4dC1hbGlnbiB0byBsZWZ0LlxuICB0ZXh0LWFsaWduOiBjZW50ZXI7XG59XG5cbi5zYmItY2FsZW5kYXJfX3RhYmxlLWJvZHkge1xuICAvLyBEZXBlbmRpbmcgd2hlcmUgaW4gdGhlIERPTSBpdCdzIHVzZWQsIGl0J3MgcG9zc2libGUgdGhhdCBicm93c2VyIGRlZmF1bHRzIHNldCB0ZXh0LWFsaWduIHRvIGxlZnQuXG4gIHRleHQtYWxpZ246IGNlbnRlcjtcbn1cblxuLnNiYi1jYWxlbmRhcl9fdGFibGUtaGVhZGVyLWNlbGwsXG4uc2JiLWNhbGVuZGFyX190YWJsZS1oZWFkZXItY2VsbC12ZXJ0aWNhbCB7XG4gIHdpZHRoOiB2YXIoLS1zYmItY2FsZW5kYXItY2VsbC1zaXplKTtcbiAgY29sb3I6IHZhcigtLXNiYi1jYWxlbmRhci1oZWFkZXItY29sb3IpO1xuICBwYWRkaW5nOiAwO1xuICBmb250LXNpemU6IHZhcigtLXNiYi10ZXh0LWZvbnQtc2l6ZS14cyk7XG4gIGZvbnQtd2VpZ2h0OiBub3JtYWw7XG4gIGxldHRlci1zcGFjaW5nOiB2YXIoLS1zYmItdHlwby1sZXR0ZXItc3BhY2luZy10ZXh0KTtcblxuICA6aG9zdCg6bm90KFttdWx0aXBsZV0pKSAmIHtcbiAgICBoZWlnaHQ6IHZhcigtLXNiYi1jYWxlbmRhci1jZWxsLXNpemUpO1xuICB9XG59XG5cbi5zYmItY2FsZW5kYXJfX3RhYmxlLWhlYWRlci1jZWxsLXZlcnRpY2FsIHtcbiAgcGFkZGluZy1ibG9jay1lbmQ6IHZhcigtLXNiYi1jYWxlbmRhci1jZWxsLXBhZGRpbmctYmxvY2stZW5kKTtcbn1cblxuLnNiYi1jYWxlbmRhcl9fdGFibGUtZGF0YSB7XG4gIHBvc2l0aW9uOiByZWxhdGl2ZTtcbiAgcGFkZGluZzogMDtcbiAgdGV4dC1hbGlnbjogY2VudGVyO1xufVxuXG4uc2JiLWNhbGVuZGFyX19kYXktY2VsbCB7XG4gIHBhZGRpbmctYmxvY2stZW5kOiB2YXIoLS1zYmItY2FsZW5kYXItY2VsbC1wYWRkaW5nLWJsb2NrLWVuZCk7XG59XG5cbnNiYi1jYWxlbmRhci1kYXkge1xuICA6aG9zdCg6bm90KDpzdGF0ZShlbmhhbmNlZCkpKSAmIHtcbiAgICAtLXNiYi1jYWxlbmRhci1jZWxsLWp1c3RpZnktY29udGVudDogY2VudGVyO1xuICAgIC0tc2JiLWNhbGVuZGFyLWRheS1oZWlnaHQ6ICN7c2JiLnB4LXRvLXJlbS1idWlsZCg0NCl9O1xuICAgIC0tc2JiLWNhbGVuZGFyLWRheS1leHRyYS1kaXNwbGF5OiBub25lO1xuICAgIC0tc2JiLWNhbGVuZGFyLWRheS12YWx1ZS1oZWlnaHQ6IHVuc2V0O1xuICAgIC0tc2JiLWNhbGVuZGFyLWRheS1jcm9zc2VkLW91dC10b3A6IDUwJTtcbiAgICAtLXNiYi1jYWxlbmRhci1kYXktY3Jvc3NlZC1vdXQtdHJhbnNsYXRlOiB0cmFuc2xhdGUoLTUwJSwgLTUwJSkgcm90YXRlKC00NWRlZyk7XG4gIH1cbn1cblxuQGtleWZyYW1lcyBzaG93IHtcbiAgZnJvbSB7XG4gICAgb3BhY2l0eTogMDtcbiAgICB0cmFuc2Zvcm06IHRyYW5zbGF0ZVkodmFyKC0tc2JiLWNhbGVuZGFyLXRhYmxlLWFuaW1hdGlvbi1zaGlmdCkpO1xuICB9XG5cbiAgdG8ge1xuICAgIG9wYWNpdHk6IDE7XG4gICAgdHJhbnNmb3JtOiB0cmFuc2xhdGVZKDAlKTtcbiAgfVxufVxuXG5Aa2V5ZnJhbWVzIGhpZGUge1xuICBmcm9tIHtcbiAgICBvcGFjaXR5OiAxO1xuICAgIHRyYW5zZm9ybTogdHJhbnNsYXRlWSgwJSk7XG4gIH1cblxuICB0byB7XG4gICAgb3BhY2l0eTogMDtcbiAgICB0cmFuc2Zvcm06IHRyYW5zbGF0ZVkodmFyKC0tc2JiLWNhbGVuZGFyLXRhYmxlLWFuaW1hdGlvbi1zaGlmdCkpO1xuICB9XG59XG4iLCJpbXBvcnQgeyBlYWNoV2Vla09mSW50ZXJ2YWwsIGVuZE9mTW9udGgsIGdldFdlZWssIHN0YXJ0T2ZNb250aCB9IGZyb20gJ2RhdGUtZm5zJztcbmltcG9ydCB7XG4gIHR5cGUgQ1NTUmVzdWx0R3JvdXAsXG4gIGh0bWwsXG4gIGlzU2VydmVyLFxuICBub3RoaW5nLFxuICB0eXBlIFByb3BlcnR5VmFsdWVzLFxuICB0eXBlIFRlbXBsYXRlUmVzdWx0LFxufSBmcm9tICdsaXQnO1xuaW1wb3J0IHsgcHJvcGVydHksIHN0YXRlIH0gZnJvbSAnbGl0L2RlY29yYXRvcnMuanMnO1xuXG5pbXBvcnQgeyBpc0Fycm93S2V5T3JQYWdlS2V5c1ByZXNzZWQgfSBmcm9tICcuLi8uLi9jb3JlL2ExMXkudHMnO1xuaW1wb3J0IHsgU2JiRWxlbWVudCB9IGZyb20gJy4uLy4uL2NvcmUvYmFzZS1lbGVtZW50cy50cyc7XG5pbXBvcnQgeyByZWFkQ29uZmlnIH0gZnJvbSAnLi4vLi4vY29yZS9jb25maWcudHMnO1xuaW1wb3J0IHtcbiAgU2JiTGFuZ3VhZ2VDb250cm9sbGVyLFxuICBTYmJNZWRpYU1hdGNoZXJDb250cm9sbGVyLFxuICBTYmJNZWRpYVF1ZXJ5QnJlYWtwb2ludExhcmdlQW5kQWJvdmUsXG59IGZyb20gJy4uLy4uL2NvcmUvY29udHJvbGxlcnMudHMnO1xuaW1wb3J0IHR5cGUgeyBEYXRlQWRhcHRlciB9IGZyb20gJy4uLy4uL2NvcmUvZGF0ZXRpbWUudHMnO1xuaW1wb3J0IHtcbiAgREFZU19QRVJfUk9XLFxuICBkZWZhdWx0RGF0ZUFkYXB0ZXIsXG4gIE1PTlRIU19QRVJfUEFHRSxcbiAgTU9OVEhTX1BFUl9ST1csXG4gIFlFQVJTX1BFUl9QQUdFLFxuICBZRUFSU19QRVJfUk9XLFxufSBmcm9tICcuLi8uLi9jb3JlL2RhdGV0aW1lLnRzJztcbmltcG9ydCB7IGZvcmNlVHlwZSwgaGFuZGxlRGlzdGluY3RDaGFuZ2UsIHBsYWluRGF0ZSB9IGZyb20gJy4uLy4uL2NvcmUvZGVjb3JhdG9ycy50cyc7XG5pbXBvcnQge1xuICBpMThuQ2FsZW5kYXJEYXRlU2VsZWN0aW9uLFxuICBpMThuQ2FsZW5kYXJXZWVrTnVtYmVyLFxuICBpMThuTmV4dE1vbnRoLFxuICBpMThuTmV4dFllYXIsXG4gIGkxOG5OZXh0WWVhclJhbmdlLFxuICBpMThuUHJldmlvdXNNb250aCxcbiAgaTE4blByZXZpb3VzWWVhcixcbiAgaTE4blByZXZpb3VzWWVhclJhbmdlLFxuICBpMThuWWVhck1vbnRoU2VsZWN0aW9uLFxufSBmcm9tICcuLi8uLi9jb3JlL2kxOG4udHMnO1xuaW1wb3J0IHR5cGUgeyBTYmJPcmllbnRhdGlvbiB9IGZyb20gJy4uLy4uL2NvcmUvaW50ZXJmYWNlcy50cyc7XG5pbXBvcnQgeyBib3hTaXppbmdTdHlsZXMgfSBmcm9tICcuLi8uLi9jb3JlL3N0eWxlcy50cyc7XG5pbXBvcnQgeyBTYmJDYWxlbmRhckRheUVsZW1lbnQgfSBmcm9tICcuLi9jYWxlbmRhci1kYXkvY2FsZW5kYXItZGF5LmNvbXBvbmVudC50cyc7XG5pbXBvcnQgeyBTYmJDYWxlbmRhck1vbnRoRWxlbWVudCB9IGZyb20gJy4uL2NhbGVuZGFyLW1vbnRoL2NhbGVuZGFyLW1vbnRoLmNvbXBvbmVudC50cyc7XG5pbXBvcnQgeyBTYmJDYWxlbmRhcldlZWtkYXlFbGVtZW50IH0gZnJvbSAnLi4vY2FsZW5kYXItd2Vla2RheS9jYWxlbmRhci13ZWVrZGF5LmNvbXBvbmVudC50cyc7XG5pbXBvcnQgeyBTYmJDYWxlbmRhcldlZWtudW1iZXJFbGVtZW50IH0gZnJvbSAnLi4vY2FsZW5kYXItd2Vla251bWJlci9jYWxlbmRhci13ZWVrbnVtYmVyLmNvbXBvbmVudC50cyc7XG5pbXBvcnQgeyBTYmJDYWxlbmRhclllYXJFbGVtZW50IH0gZnJvbSAnLi4vY2FsZW5kYXIteWVhci9jYWxlbmRhci15ZWFyLmNvbXBvbmVudC50cyc7XG5pbXBvcnQgdHlwZSB7IFNiYkNhbGVuZGFyQ2VsbEJhc2VFbGVtZW50IH0gZnJvbSAnLi4vY29tbW9uLnRzJztcblxuaW1wb3J0IHN0eWxlIGZyb20gJy4vY2FsZW5kYXIuc2Nzcz9saXQmaW5saW5lJztcblxuaW1wb3J0ICcuLi8uLi9idXR0b24vc2Vjb25kYXJ5LWJ1dHRvbi50cyc7XG5pbXBvcnQgJy4uLy4uL2ljb24udHMnO1xuaW1wb3J0ICcuLi8uLi9zY3JlZW4tcmVhZGVyLW9ubHkudHMnO1xuXG5TYmJDYWxlbmRhckRheUVsZW1lbnQuZGVmaW5lKCk7XG5TYmJDYWxlbmRhck1vbnRoRWxlbWVudC5kZWZpbmUoKTtcblNiYkNhbGVuZGFyWWVhckVsZW1lbnQuZGVmaW5lKCk7XG5TYmJDYWxlbmRhcldlZWtkYXlFbGVtZW50LmRlZmluZSgpO1xuU2JiQ2FsZW5kYXJXZWVrbnVtYmVyRWxlbWVudC5kZWZpbmUoKTtcblxuZXhwb3J0IGNsYXNzIFNiYk1vbnRoQ2hhbmdlRXZlbnQgZXh0ZW5kcyBFdmVudCB7XG4gIHByaXZhdGUgcmVhZG9ubHkgX3JhbmdlOiByZWFkb25seSBEYXlbXTtcblxuICBwdWJsaWMgZ2V0IHJhbmdlKCk6IHJlYWRvbmx5IERheVtdIHtcbiAgICByZXR1cm4gdGhpcy5fcmFuZ2U7XG4gIH1cblxuICBwdWJsaWMgY29uc3RydWN0b3IocmFuZ2U6IHJlYWRvbmx5IERheVtdKSB7XG4gICAgc3VwZXIoJ21vbnRoY2hhbmdlJywgeyBidWJibGVzOiB0cnVlLCBjb21wb3NlZDogdHJ1ZSB9KTtcbiAgICB0aGlzLl9yYW5nZSA9IE9iamVjdC5mcmVlemUocmFuZ2UgfHwgW10pO1xuICB9XG59XG5cbi8qKlxuICogUGFyYW1ldGVycyBuZWVkZWQgaW4geWVhciBhbmQgbW9udGggdmlld3MgdG8gY29ycmVjdGx5IGNhbGN1bGF0ZSB0aGUgbmV4dCBlbGVtZW50IGluIGtleWJvYXJkIG5hdmlnYXRpb24uXG4gKlxuICogVGhlIGNlbGwncyBpbmRleCBhbmQgdGhlIGVsZW1lbnQncyBpbmRleCBpbiBpdHMgbW9udGggLyB5ZWFyIGJhdGNoIG11c3QgYmUgZGlzdGluZ3Vpc2hlZDpcbiAqIHRoZSBmaXJzdCBpcyB0aGUgaW5kZXggb2YgdGhlIGVsZW1lbnQgaW4gdGhlIGFycmF5IG9mIGFsbCB0aGUgcmVuZGVyZWQgY2VsbHMsIHdoaWxlIHRoZSBzZWNvbmQgaXMgdGhlIGluZGV4IG9mIHRoZSBlbGVtZW50IHJlbGF0aXZlIHRvIGl0cyB0YWJsZS5cbiAqIEluIG5vbi13aWRlIG1vZGUsIHRoZSB3dG8gYXJlIHRoZSBzYW1lLCB3aGlsZSBpbiB3aWRlIG1vZGUgdGhlIGNlbGwncyBpbmRleCBjYW4gZ28gZnJvbSAwIHRvIDQ3IGZvciB5ZWFycyBhbmQgZnJvbSAwIHRvIDIzIGZvciBtb250aHMsXG4gKiB3aGlsZSB0aGUgZWxlbWVudCBpbmRleCBnb2VzIGZyb20gMCB0bywgcmVzcGVjdGl2ZWx5LCAyMyBhbmQgMTEuXG4gKi9cbmludGVyZmFjZSBDYWxlbmRhcktleWJvYXJkTmF2aWdhdGlvbk1vbnRoWWVhclZpZXdzUGFyYW1ldGVycyB7XG4gIC8qKiBUaGUgZWxlbWVudCBpbmRleCB3aXRoaW4gaXRzIHllYXIgb3IgbW9udGggcmFuZ2UuICovXG4gIGVsZW1lbnRJbmRleEZvcldpZGVNb2RlOiBudW1iZXI7XG4gIC8qKiBJbiB3aWRlIG1vZGUsIHRoZSBpbmRleCBvZiB0aGUgZmlyc3QgZWxlbWVudCBpbiB0aGUgc2Vjb25kIHBhbmVsLCBvciwgYWx0ZXJuYXRpdmVseSwgdGhlIG51bWJlciBvZiBlbGVtZW50cyBpbiB0aGUgZmlyc3QgcGFuZWwuICovXG4gIG9mZnNldEZvcldpZGVNb2RlOiBudW1iZXI7XG4gIC8qKiBUaGUgaW5kZXggb2YgdGhlIGxhc3QgZWxlbWVudCB3aXRoaW4gdGhlIGVsZW1lbnQncyBtb250aCAob3IgeWVhciByYW5nZSkuICovXG4gIGxhc3RFbGVtZW50SW5kZXhGb3JXaWRlTW9kZTogbnVtYmVyO1xuICAvKiogVGhlIG51bWJlciBvZiBjZWxscyBkaXNwbGF5ZWQgaW4gYSBzaW5nbGUgcm93LCBkZXBlbmRpbmcgb24gdGhlIHJlbmRlcmVkIHZpZXcuICovXG4gIHZlcnRpY2FsT2Zmc2V0OiBudW1iZXI7XG59XG5cbi8qKlxuICogUGFyYW1ldGVycyBuZWVkZWQgaW4gZGF5IHZpZXcgdG8gY29ycmVjdGx5IGNhbGN1bGF0ZSB0aGUgbmV4dCBlbGVtZW50IGluIGtleWJvYXJkIG5hdmlnYXRpb24uXG4gKlxuICogSW4gb3JpZW50YXRpb249J3ZlcnRpY2FsJywgaXQncyBub3QgcG9zc2libGUgdG8gcmVseSBvbiBhbnkgYXJyYXkvaW5kZXggdG8gY2FsY3VsYXRlIHRoZSBlbGVtZW50IHRvIG5hdmlnYXRlIHRvLFxuICogc28gY2FsY3VsYXRpb25zIG9uIGRhdGVzIG11c3QgYmUgZG9uZSwgd2hpY2ggc2hvdWxkIGNvbnNpZGVyIHZpZXcgYm91bmRhcmllcywgb2Zmc2V0cyBhbmQgbW9udGgncyBsZW5ndGguXG4gKi9cbmludGVyZmFjZSBDYWxlbmRhcktleWJvYXJkTmF2aWdhdGlvbkRheVZpZXdQYXJhbWV0ZXJzIHtcbiAgLyoqIFRoZSBmaXJzdCBkYXkgcmVuZGVyZWQuICovXG4gIGZpcnN0RGF5SW5WaWV3OiBzdHJpbmcgfCBudWxsO1xuICAvKiogVGhlIGxhc3QgZGF5IHJlbmRlcmVkLiBJdCBkZXBlbmRzIG9uIHRoZSAnd2lkZScgdmFsdWUuICovXG4gIGxhc3REYXlJblZpZXc6IHN0cmluZyB8IG51bGw7XG4gIC8qKiBUaGUgb2Zmc2V0IGZyb20gdGhlIGZpcnN0IGRheSBvZiB0aGUgd2VlayAoTW9uZGF5KSBvZiB0aGUgZmlyc3QgcmVuZGVyZWQgbW9udGguICovXG4gIGZpcnN0TW9udGhPZmZzZXQ6IG51bWJlcjtcbiAgLyoqIFRoZSBudW1iZXIgb2YgZGF5cyBpbiB0aGUgZmlyc3QgcmVuZGVyZWQgbW9udGguICovXG4gIGZpcnN0TW9udGhMZW5ndGg6IG51bWJlcjtcbiAgLyoqIFRoZSBvZmZzZXQgZnJvbSB0aGUgZmlyc3QgZGF5IG9mIHRoZSB3ZWVrIChNb25kYXkpIG9mIHRoZSBzZWNvbmQgcmVuZGVyZWQgbW9udGguIElmIHdpZGUgaXMgZmFsc2UsIGl0J3MgZXF1YWwgdG8gemVyby4gKi9cbiAgc2Vjb25kTW9udGhPZmZzZXQ6IG51bWJlcjtcbn1cblxuZXhwb3J0IGludGVyZmFjZSBEYXk8VCA9IERhdGU+IHtcbiAgLyoqIERhdGUgYXMgSVNPIHN0cmluZy4gKi9cbiAgdmFsdWU6IHN0cmluZztcbiAgZGF5VmFsdWU6IHN0cmluZztcbiAgbW9udGhWYWx1ZTogc3RyaW5nO1xuICB5ZWFyVmFsdWU6IHN0cmluZztcbiAgZGF0ZVZhbHVlOiBUO1xuICB3ZWVrVmFsdWU6IG51bWJlcjtcbiAgd2Vla0RheVZhbHVlOiBudW1iZXI7XG59XG5cbi8qKiBAZGVwcmVjYXRlZCB1c2UgTW9udGhDZWxsICovXG5leHBvcnQgaW50ZXJmYWNlIE1vbnRoIHtcbiAgdmFsdWU6IHN0cmluZztcbiAgbG9uZ1ZhbHVlOiBzdHJpbmc7XG4gIG1vbnRoVmFsdWU6IG51bWJlcjtcbn1cblxuZXhwb3J0IGludGVyZmFjZSBNb250aENlbGwge1xuICB2YWx1ZTogc3RyaW5nO1xuICBtb250aFZhbHVlOiBudW1iZXI7XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgV2Vla2RheSB7XG4gIGxvbmc6IHN0cmluZztcbiAgbmFycm93OiBzdHJpbmc7XG59XG5cbmV4cG9ydCB0eXBlIENhbGVuZGFyVmlldyA9ICdkYXknIHwgJ21vbnRoJyB8ICd5ZWFyJztcblxuLyoqXG4gKiBJdCBkaXNwbGF5cyBhIGNhbGVuZGFyIHdoaWNoIGFsbG93cyBjaG9vc2luZyBhIGRhdGUuXG4gKlxuICogQHNsb3QgLSBVc2UgdGhlIHVubmFtZWQgc2xvdCB0byBhZGQgY3VzdG9taXplZCBgc2JiLWNhbGVuZGFyLWRheWAgZWxlbWVudHMuXG4gKi9cbmV4cG9ydCBjbGFzcyBTYmJDYWxlbmRhckVsZW1lbnQ8VCA9IERhdGU+IGV4dGVuZHMgU2JiRWxlbWVudCB7XG4gIHB1YmxpYyBzdGF0aWMgb3ZlcnJpZGUgcmVhZG9ubHkgZWxlbWVudE5hbWU6IHN0cmluZyA9ICdzYmItY2FsZW5kYXInO1xuICBwdWJsaWMgc3RhdGljIG92ZXJyaWRlIHN0eWxlczogQ1NTUmVzdWx0R3JvdXAgPSBbYm94U2l6aW5nU3R5bGVzLCBzdHlsZV07XG4gIHB1YmxpYyBzdGF0aWMgcmVhZG9ubHkgZXZlbnRzID0ge1xuICAgIGRhdGVzZWxlY3RlZDogJ2RhdGVzZWxlY3RlZCcsXG4gICAgbW9udGhjaGFuZ2U6ICdtb250aGNoYW5nZScsXG4gIH0gYXMgY29uc3Q7XG5cbiAgLyoqIElmIHNldCB0byB0cnVlLCB0d28gbW9udGhzIGFyZSBkaXNwbGF5ZWQgKi9cbiAgQGZvcmNlVHlwZSgpXG4gIEBwcm9wZXJ0eSh7IHR5cGU6IEJvb2xlYW4gfSlcbiAgcHVibGljIGFjY2Vzc29yIHdpZGU6IGJvb2xlYW4gPSBmYWxzZTtcblxuICAvKiogVGhlIGluaXRpYWwgdmlldyBvZiB0aGUgY2FsZW5kYXIgd2hpY2ggc2hvdWxkIGJlIGRpc3BsYXllZCBvbiBvcGVuaW5nLiAqL1xuICBAcHJvcGVydHkoKSBwdWJsaWMgYWNjZXNzb3IgdmlldzogQ2FsZW5kYXJWaWV3ID0gJ2RheSc7XG5cbiAgLyoqXG4gICAqIFRoZSBtaW5pbXVtIHZhbGlkIGRhdGUuIEFjY2VwdHMgYSBkYXRlIG9iamVjdCBvciBudWxsLlxuICAgKiBBY2NlcHRzIGFuIElTTzg2MDEgZm9ybWF0dGVkIHN0cmluZyAoZS5nLiAyMDI0LTEyLTI0KSBhcyBhdHRyaWJ1dGUuXG4gICAqL1xuICBAcGxhaW5EYXRlKClcbiAgQHByb3BlcnR5KClcbiAgcHVibGljIGFjY2Vzc29yIG1pbjogVCB8IG51bGwgPSBudWxsO1xuXG4gIC8qKlxuICAgKiBUaGUgbWF4aW11bSB2YWxpZCBkYXRlLiBBY2NlcHRzIGEgZGF0ZSBvYmplY3Qgb3IgbnVsbC5cbiAgICogQWNjZXB0cyBhbiBJU084NjAxIGZvcm1hdHRlZCBzdHJpbmcgKGUuZy4gMjAyNC0xMi0yNCkgYXMgYXR0cmlidXRlLlxuICAgKi9cbiAgQHBsYWluRGF0ZSgpXG4gIEBwcm9wZXJ0eSgpXG4gIHB1YmxpYyBhY2Nlc3NvciBtYXg6IFQgfCBudWxsID0gbnVsbDtcblxuICAvKiogV2hldGhlciB0aGUgY2FsZW5kYXIgYWxsb3dzIGZvciBtdWx0aXBsZSBkYXRlIHNlbGVjdGlvbi4gKi9cbiAgQGZvcmNlVHlwZSgpXG4gIEBoYW5kbGVEaXN0aW5jdENoYW5nZSgoZTogU2JiQ2FsZW5kYXJFbGVtZW50PFQ+LCBuZXdWYWx1ZTogYm9vbGVhbikgPT5cbiAgICBlLl9vbk11bHRpcGxlQ2hhbmdlZChuZXdWYWx1ZSksXG4gIClcbiAgQHByb3BlcnR5KHsgdHlwZTogQm9vbGVhbiB9KVxuICBwdWJsaWMgYWNjZXNzb3IgbXVsdGlwbGU6IGJvb2xlYW4gPSBmYWxzZTtcblxuICAvKipcbiAgICogVGhlIHNlbGVjdGVkIGRhdGU6IGFjY2VwdHMgYSBkYXRlIG9iamVjdCwgb3IsIGlmIGBtdWx0aXBsZWAsIGFuIGFycmF5IG9mIGRhdGVzLlxuICAgKi9cbiAgQHByb3BlcnR5KClcbiAgcHVibGljIHNldCBzZWxlY3RlZCh2YWx1ZTogVCB8IFRbXSB8IG51bGwpIHtcbiAgICBpZiAoQXJyYXkuaXNBcnJheSh2YWx1ZSkpIHtcbiAgICAgIHRoaXMuX3NlbGVjdGVkID0gdmFsdWVcbiAgICAgICAgLm1hcCgoZGF0ZUxpa2U6IFQpID0+XG4gICAgICAgICAgdGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0VmFsaWREYXRlT3JOdWxsKHRoaXMuX2RhdGVBZGFwdGVyLmRlc2VyaWFsaXplKGRhdGVMaWtlKSksXG4gICAgICAgIClcbiAgICAgICAgLmZpbHRlcigoZGF0ZTogVCB8IG51bGwpOiBkYXRlIGlzIFQgPT4gZGF0ZSAhPT0gbnVsbClcbiAgICAgICAgLmZpbHRlcihcbiAgICAgICAgICAoZGF0ZTogVCkgPT5cbiAgICAgICAgICAgICF0aGlzLl9pc0RheUluUmFuZ2UodGhpcy5fZGF0ZUFkYXB0ZXIudG9Jc284NjAxKGRhdGUpKSB8fCB0aGlzLl9kYXRlRmlsdGVyKGRhdGUpLFxuICAgICAgICApO1xuICAgIH0gZWxzZSB7XG4gICAgICBjb25zdCBzZWxlY3RlZERhdGUgPSB0aGlzLl9kYXRlQWRhcHRlci5nZXRWYWxpZERhdGVPck51bGwoXG4gICAgICAgIHRoaXMuX2RhdGVBZGFwdGVyLmRlc2VyaWFsaXplKHZhbHVlKSxcbiAgICAgICk7XG4gICAgICBpZiAoXG4gICAgICAgICEhc2VsZWN0ZWREYXRlICYmXG4gICAgICAgICghdGhpcy5faXNEYXlJblJhbmdlKHRoaXMuX2RhdGVBZGFwdGVyLnRvSXNvODYwMShzZWxlY3RlZERhdGUpKSB8fFxuICAgICAgICAgIHRoaXMuX2RhdGVGaWx0ZXIoc2VsZWN0ZWREYXRlKSlcbiAgICAgICkge1xuICAgICAgICB0aGlzLl9zZWxlY3RlZCA9IHNlbGVjdGVkRGF0ZTtcbiAgICAgIH0gZWxzZSB7XG4gICAgICAgIHRoaXMuX3NlbGVjdGVkID0gbnVsbDtcbiAgICAgIH1cbiAgICB9XG4gIH1cbiAgcHVibGljIGdldCBzZWxlY3RlZCgpOiBUIHwgVFtdIHwgbnVsbCB7XG4gICAgcmV0dXJuIHRoaXMuX3NlbGVjdGVkO1xuICB9XG4gIEBzdGF0ZSgpIHByaXZhdGUgYWNjZXNzb3IgX3NlbGVjdGVkOiBUIHwgVFtdIHwgbnVsbCA9IG51bGw7XG5cbiAgLyoqIEEgZnVuY3Rpb24gdXNlZCB0byBmaWx0ZXIgb3V0IGRhdGVzLiAqL1xuICBAcHJvcGVydHkoeyBhdHRyaWJ1dGU6ICdkYXRlLWZpbHRlcicgfSlcbiAgcHVibGljIGFjY2Vzc29yIGRhdGVGaWx0ZXI6ICgoZGF0ZTogVCB8IG51bGwpID0+IGJvb2xlYW4pIHwgbnVsbCA9IG51bGw7XG5cbiAgLyoqIFRoZSBvcmllbnRhdGlvbiBvZiBkYXlzIGluIHRoZSBjYWxlbmRhci4gKi9cbiAgQHByb3BlcnR5KHsgcmVmbGVjdDogdHJ1ZSB9KSBwdWJsaWMgYWNjZXNzb3Igb3JpZW50YXRpb246IFNiYk9yaWVudGF0aW9uID0gJ2hvcml6b250YWwnO1xuXG4gIC8qKiBXaGV0aGVyIGl0IGhhcyB0byBkaXNwbGF5IHRoZSB3ZWVrIG51bWJlcnMgaW4gYWRkaXRpb24gdG8gd2VlayBkYXlzLiAqL1xuICBAZm9yY2VUeXBlKClcbiAgQHByb3BlcnR5KHsgYXR0cmlidXRlOiAnd2Vlay1udW1iZXJzJywgdHlwZTogQm9vbGVhbiB9KVxuICBwdWJsaWMgYWNjZXNzb3Igd2Vla051bWJlcnM6IGJvb2xlYW4gPSBmYWxzZTtcblxuICBwcml2YXRlIF9kYXRlQWRhcHRlcjogRGF0ZUFkYXB0ZXI8VD4gPSByZWFkQ29uZmlnKCkuZGF0ZXRpbWU/LmRhdGVBZGFwdGVyID8/IGRlZmF1bHREYXRlQWRhcHRlcjtcblxuICAvKiogVGhlIGN1cnJlbnRseSBhY3RpdmUgZGF0ZS4gKi9cbiAgQHN0YXRlKCkgcHJpdmF0ZSBhY2Nlc3NvciBfYWN0aXZlRGF0ZTogVCA9IHRoaXMuX2RhdGVBZGFwdGVyLnRvZGF5KCk7XG5cbiAgLyoqIFRoZSBjdXJyZW50IHdpZGUgcHJvcGVydHkgY29uc2lkZXJpbmcgcHJvcGVydHkgdmFsdWUgYW5kIGJyZWFrcG9pbnRzLiBGcm9tIHplcm8gdG8gc21hbGwgYHdpZGVgIGhhcyBhbHdheXMgdG8gYmUgZmFsc2UuICovXG4gIEBzdGF0ZSgpXG4gIHByaXZhdGUgc2V0IF93aWRlKHdpZGU6IGJvb2xlYW4pIHtcbiAgICB0aGlzLnRvZ2dsZVN0YXRlKCd3aWRlJywgd2lkZSk7XG4gICAgdGhpcy5fd2lkZUludGVybmFsID0gd2lkZTtcbiAgfVxuICBwcml2YXRlIGdldCBfd2lkZSgpOiBib29sZWFuIHtcbiAgICByZXR1cm4gdGhpcy5fd2lkZUludGVybmFsO1xuICB9XG4gIC8vIFdlIGtlZXAgdGhlIHN0YXRlIGluIGEgZmllbGQgYmVjYXVzZSBvZiB0aGUgV2ViS2l0IGJ1ZyBodHRwczovL2J1Z3Mud2Via2l0Lm9yZy9zaG93X2J1Zy5jZ2k/aWQ9MzAzNDY3LlxuICAvLyBUT0RPOiByZS1jaGVjayB3aGV0aGVyIGZpZWxkIGlzIG5lZWRlZFxuICBwcml2YXRlIF93aWRlSW50ZXJuYWw6IGJvb2xlYW4gPSBmYWxzZTtcblxuICBAc3RhdGUoKSBwcml2YXRlIGFjY2Vzc29yIF9jYWxlbmRhclZpZXc6IENhbGVuZGFyVmlldyA9ICdkYXknO1xuXG4gIHByaXZhdGUgX25leHRDYWxlbmRhclZpZXc6IENhbGVuZGFyVmlldyA9ICdkYXknO1xuXG4gIC8qKiBJbmZvcm1hdGlvbiBhYm91dCB0aGUgcmVuZGVyZWQgZGF5IHZpZXc7IHVzZWQgaW4ga2V5Ym9hcmQgbmF2aWdhdGlvbi4gKi9cbiAgcHJpdmF0ZSBfa2V5Ym9hcmROYXZpZ2F0aW9uRGF5Vmlld1BhcmFtZXRlcnM6IENhbGVuZGFyS2V5Ym9hcmROYXZpZ2F0aW9uRGF5Vmlld1BhcmFtZXRlcnMgPSB7XG4gICAgZmlyc3REYXlJblZpZXc6IG51bGwsXG4gICAgbGFzdERheUluVmlldzogbnVsbCxcbiAgICBmaXJzdE1vbnRoT2Zmc2V0OiAwLFxuICAgIGZpcnN0TW9udGhMZW5ndGg6IDAsXG4gICAgc2Vjb25kTW9udGhPZmZzZXQ6IDAsXG4gIH07XG5cbiAgLyoqIEEgbGlzdCBvZiBkYXlzLCBpbiB0d28gZm9ybWF0cyAobG9uZyBhbmQgc2luZ2xlIGNoYXIpLiAqL1xuICBwcml2YXRlIF93ZWVrZGF5cyE6IFdlZWtkYXlbXTtcblxuICAvKiogR3JpZCBvZiBjYWxlbmRhciBjZWxscyByZXByZXNlbnRpbmcgdGhlIGRhdGVzIG9mIHRoZSBtb250aC4gKi9cbiAgcHJpdmF0ZSBfd2Vla3M6IERheTxUPltdW10gPSBbXTtcblxuICAvKiogR3JpZCBvZiBjYWxlbmRhciBjZWxscyByZXByZXNlbnRpbmcgbW9udGhzLiAqL1xuICBwcml2YXRlIF9tb250aHMhOiBNb250aENlbGxbXVtdO1xuXG4gIC8qKiBHcmlkIG9mIGNhbGVuZGFyIGNlbGxzIHJlcHJlc2VudGluZyB5ZWFycy4gKi9cbiAgcHJpdmF0ZSBfeWVhcnMhOiBudW1iZXJbXVtdO1xuXG4gIC8qKiBHcmlkIG9mIGNhbGVuZGFyIGNlbGxzIHJlcHJlc2VudGluZyB5ZWFycyBmb3IgdGhlIHdpZGUgdmlldy4gKi9cbiAgcHJpdmF0ZSBfbmV4dE1vbnRoWWVhcnMhOiBudW1iZXJbXVtdO1xuXG4gIC8qKiBHcmlkIG9mIGNhbGVuZGFyIGNlbGxzIHJlcHJlc2VudGluZyB0aGUgZGF0ZXMgb2YgdGhlIG5leHQgbW9udGguICovXG4gIHByaXZhdGUgX25leHRNb250aFdlZWtzITogRGF5PFQ+W11bXTtcblxuICAvKiogQW4gYXJyYXkgY29udGFpbmluZyBhbGwgdGhlIG1vbnRoIG5hbWVzIGluIHRoZSBjdXJyZW50IGxhbmd1YWdlLiAqL1xuICBwcml2YXRlIF9tb250aE5hbWVzOiBzdHJpbmdbXSA9IHRoaXMuX2RhdGVBZGFwdGVyLmdldE1vbnRoTmFtZXMoJ2xvbmcnKTtcblxuICAvKiogQW4gYXJyYXkgY29udGFpbmluZyB0aGUgd2Vla3MnIG51bWJlcnMgZm9yIHRoZSBjdXJyZW50IG1vbnRoLiAqL1xuICBwcml2YXRlIF93ZWVrTnVtYmVycyE6IG51bWJlcltdO1xuXG4gIC8qKiBBbiBhcnJheSBjb250YWluaW5nIHRoZSB3ZWVrcycgbnVtYmVycyBmb3IgdGhlIG5leHQgbW9udGggaW4gd2lkZSBtb2RlLiAqL1xuICBwcml2YXRlIF9uZXh0TW9udGhXZWVrTnVtYmVycyE6IG51bWJlcltdO1xuXG4gIHByaXZhdGUgX2VuaGFuY2VkVmFyaWFudDogYm9vbGVhbiA9IGZhbHNlO1xuXG4gIC8qKiBBIGxpc3Qgb2YgY2FsZW5kYXIncyBjZWxscyBjb3JyZXNwb25kaW5nIHRvIGRheXMsIG1vbnRocyBvciB5ZWFycyBkZXBlbmRpbmcgb24gdGhlIHZpZXcuICovXG4gIHByaXZhdGUgZ2V0IF9jZWxscygpOiBTYmJDYWxlbmRhckNlbGxCYXNlRWxlbWVudFtdIHtcbiAgICByZXR1cm4gQXJyYXkuZnJvbTxTYmJDYWxlbmRhckNlbGxCYXNlRWxlbWVudD4oXG4gICAgICAodGhpcy5fY2FsZW5kYXJWaWV3ID09PSAnZGF5J1xuICAgICAgICA/IChBcnJheS5mcm9tKHRoaXMuc2hhZG93Um9vdCEucXVlcnlTZWxlY3RvckFsbCgnc2xvdCcpKS5mbGF0TWFwKChlOiBIVE1MU2xvdEVsZW1lbnQpID0+XG4gICAgICAgICAgICBlLmFzc2lnbmVkRWxlbWVudHMoeyBmbGF0dGVuOiB0cnVlIH0pLFxuICAgICAgICAgICkgYXMgU2JiQ2FsZW5kYXJEYXlFbGVtZW50W10pXG4gICAgICAgIDogdGhpcy5zaGFkb3dSb290Py5xdWVyeVNlbGVjdG9yQWxsKGBzYmItY2FsZW5kYXItJHt0aGlzLl9jYWxlbmRhclZpZXd9YCkpID8/IFtdLFxuICAgICk7XG4gIH1cblxuICAvKiogVGhlIGNob3NlbiB5ZWFyIGluIHRoZSB5ZWFyIHNlbGVjdGlvbiB2aWV3LiAqL1xuICBwcml2YXRlIF9jaG9zZW5ZZWFyPzogbnVtYmVyO1xuXG4gIC8qKiBUaGUgY2hvc2VuIG1vbnRoIGluIHRoZSB5ZWFyIHNlbGVjdGlvbiB2aWV3LiAqL1xuICBwcml2YXRlIF9jaG9zZW5Nb250aD86IG51bWJlcjtcblxuICAvKiogV2hldGhlciB0aGUgZm9jdXMgc2hvdWxkIGJlIHJlc2V0IG9uIGZvY3VzQ2VsbC4gKi9cbiAgcHJpdmF0ZSBfcmVzZXRGb2N1cyA9IGZhbHNlO1xuXG4gIC8qKiBXaGV0aGVyIGFuIGVsZW1lbnQgaW5zaWRlIHRoZSBjYWxlbmRhciBpcyBjdXJyZW50bHkgZm9jdXNlZC4gKi9cbiAgcHJpdmF0ZSBfY29udGFpbmluZ0ZvY3VzID0gZmFsc2U7XG5cbiAgQHN0YXRlKClcbiAgcHJpdmF0ZSBhY2Nlc3NvciBfaW5pdGlhbGl6ZWQgPSBmYWxzZTtcblxuICBwcml2YXRlIF9sYW5ndWFnZSA9IG5ldyBTYmJMYW5ndWFnZUNvbnRyb2xsZXIodGhpcykud2l0aEhhbmRsZXIoKCkgPT4ge1xuICAgIHRoaXMuX21vbnRoTmFtZXMgPSB0aGlzLl9kYXRlQWRhcHRlci5nZXRNb250aE5hbWVzKCdsb25nJyk7XG4gICAgdGhpcy5fY3JlYXRlTW9udGhSb3dzKCk7XG4gIH0pO1xuICBwcml2YXRlIF9tZWRpYU1hdGNoZXIgPSBuZXcgU2JiTWVkaWFNYXRjaGVyQ29udHJvbGxlcih0aGlzLCB7XG4gICAgW1NiYk1lZGlhUXVlcnlCcmVha3BvaW50TGFyZ2VBbmRBYm92ZV06ICgpID0+IHRoaXMuX2luaXQoKSxcbiAgfSk7XG5cbiAgcHVibGljIGNvbnN0cnVjdG9yKCkge1xuICAgIHN1cGVyKCk7XG4gICAgdGhpcy5fY3JlYXRlTW9udGhSb3dzKCk7XG4gICAgdGhpcy5fc2V0V2Vla2RheXMoKTtcblxuICAgIC8vIFdlIG5lZWQgdG8gdHJhY2sgdGhlIGZvY3VzIGFzIHdlIHNob3VsZCBvbmx5IHRha2UgZm9jdXMgaW50byB0aGUgY2FsZW5kYXIsIHdoZW4gdGhlXG4gICAgLy8gZm9jdXMgd2FzIG9uY2Ugc2V0IGludG8gdGhlIGNhbGVuZGFyLlxuICAgIC8vIEZvciBzaGFkb3cgRE9NIGNvbXBhdGliaWxpdHkgd2UgbmVlZCB0byB0cmFjayB0aGlzIHByb2dyYW1tYXRpY2FsbHkuXG4gICAgdGhpcy5hZGRFdmVudExpc3RlbmVyKCdmb2N1c2luJywgKCkgPT4gKHRoaXMuX2NvbnRhaW5pbmdGb2N1cyA9IHRydWUpKTtcbiAgICB0aGlzLmFkZEV2ZW50TGlzdGVuZXIoJ2ZvY3Vzb3V0JywgKCkgPT4gKHRoaXMuX2NvbnRhaW5pbmdGb2N1cyA9IGZhbHNlKSk7XG4gICAgdGhpcy5hZGRFdmVudExpc3RlbmVyKCdjbGljaycsIChlKSA9PiB7XG4gICAgICBjb25zdCBkYXkgPSAoZS50YXJnZXQgYXMgSFRNTEVsZW1lbnQpLmNsb3Nlc3Q8U2JiQ2FsZW5kYXJEYXlFbGVtZW50PFQ+Pignc2JiLWNhbGVuZGFyLWRheScpO1xuICAgICAgaWYgKGRheSkge1xuICAgICAgICB0aGlzLl9zZWxlY3REYXRlKGRheS52YWx1ZSEpO1xuICAgICAgfVxuICAgIH0pO1xuICAgIHRoaXMuYWRkRXZlbnRMaXN0ZW5lcigna2V5ZG93bicsIChlKSA9PiB7XG4gICAgICBpZiAoKGUudGFyZ2V0IGFzIEhUTUxFbGVtZW50KS5sb2NhbE5hbWUgPT09ICdzYmItY2FsZW5kYXItZGF5Jykge1xuICAgICAgICB0aGlzLl9oYW5kbGVLZXlib2FyZEV2ZW50KFxuICAgICAgICAgIGUsXG4gICAgICAgICAgdGhpcy5fbWFwRGF0ZVRvRGF5KChlLnRhcmdldCBhcyBTYmJDYWxlbmRhckRheUVsZW1lbnQpLnZhbHVlISBhcyBUKSxcbiAgICAgICAgKTtcbiAgICAgIH1cbiAgICB9KTtcbiAgfVxuXG4gIHByaXZhdGUgX2RhdGVGaWx0ZXIoZGF0ZTogVCk6IGJvb2xlYW4ge1xuICAgIHJldHVybiB0aGlzLmRhdGVGaWx0ZXI/LihkYXRlKSA/PyB0cnVlO1xuICB9XG5cbiAgLyoqIFJlc2V0cyB0aGUgYWN0aXZlIG1vbnRoIGFjY29yZGluZyB0byB0aGUgbmV3IHN0YXRlIG9mIHRoZSBjYWxlbmRhci4gKi9cbiAgcHVibGljIHJlc2V0UG9zaXRpb24oKTogdm9pZCB7XG4gICAgdGhpcy5fcmVzZXRDYWxlbmRhclZpZXcoKTtcbiAgICB0aGlzLl9pbml0KCk7XG4gIH1cblxuICBwdWJsaWMgb3ZlcnJpZGUgY29ubmVjdGVkQ2FsbGJhY2soKTogdm9pZCB7XG4gICAgc3VwZXIuY29ubmVjdGVkQ2FsbGJhY2soKTtcbiAgICB0aGlzLnJlc2V0UG9zaXRpb24oKTtcbiAgICB0aGlzLnNoYWRvd1Jvb3Q/LmFkZEV2ZW50TGlzdGVuZXIoJ3Nsb3RjaGFuZ2UnLCB0aGlzLl9vblNsb3RDaGFuZ2UsIHsgY2FwdHVyZTogdHJ1ZSB9KTtcbiAgfVxuXG4gIHB1YmxpYyBvdmVycmlkZSBkaXNjb25uZWN0ZWRDYWxsYmFjaygpOiB2b2lkIHtcbiAgICBzdXBlci5kaXNjb25uZWN0ZWRDYWxsYmFjaygpO1xuICAgIHRoaXMuc2hhZG93Um9vdD8ucmVtb3ZlRXZlbnRMaXN0ZW5lcignc2xvdGNoYW5nZScsIHRoaXMuX29uU2xvdENoYW5nZSwgeyBjYXB0dXJlOiB0cnVlIH0pO1xuICB9XG5cbiAgLyoqIEBpbnRlcm5hbCAqL1xuICBwdWJsaWMgb3ZlcnJpZGUgZm9jdXMoKTogdm9pZCB7XG4gICAgdGhpcy5fcmVzZXRGb2N1cyA9IHRydWU7XG4gICAgdGhpcy5fZm9jdXNDZWxsKCk7XG4gIH1cblxuICBwcm90ZWN0ZWQgb3ZlcnJpZGUgd2lsbFVwZGF0ZShjaGFuZ2VkUHJvcGVydGllczogUHJvcGVydHlWYWx1ZXM8dGhpcz4pOiB2b2lkIHtcbiAgICBzdXBlci53aWxsVXBkYXRlKGNoYW5nZWRQcm9wZXJ0aWVzKTtcblxuICAgIGlmICghdGhpcy5faW5pdGlhbGl6ZWQpIHtcbiAgICAgIHJldHVybjtcbiAgICB9XG5cbiAgICBpZiAoY2hhbmdlZFByb3BlcnRpZXMuaGFzKCd3aWRlJykgfHwgY2hhbmdlZFByb3BlcnRpZXMuaGFzKCdvcmllbnRhdGlvbicpKSB7XG4gICAgICB0aGlzLnJlc2V0UG9zaXRpb24oKTtcbiAgICB9XG5cbiAgICBpZiAoY2hhbmdlZFByb3BlcnRpZXMuaGFzKCd2aWV3JykpIHtcbiAgICAgIHRoaXMuX3NldENob3NlblllYXIoKTtcbiAgICAgIHRoaXMuX2Nob3Nlbk1vbnRoID0gdW5kZWZpbmVkO1xuICAgICAgdGhpcy5fbmV4dENhbGVuZGFyVmlldyA9IHRoaXMuX2NhbGVuZGFyVmlldyA9IHRoaXMudmlldztcbiAgICB9XG4gIH1cblxuICBwcm90ZWN0ZWQgb3ZlcnJpZGUgdXBkYXRlZChjaGFuZ2VkUHJvcGVydGllczogUHJvcGVydHlWYWx1ZXM8dGhpcz4pOiB2b2lkIHtcbiAgICBzdXBlci51cGRhdGVkKGNoYW5nZWRQcm9wZXJ0aWVzKTtcbiAgICAvLyBUaGUgY2FsZW5kYXIgbmVlZHMgdG8gY2FsY3VsYXRlIHRhYi1pbmRleGVzIG9uIGZpcnN0IHJlbmRlcixcbiAgICAvLyBhbmQgZXZlcnkgdGltZSBhIGRhdGUgaXMgc2VsZWN0ZWQgb3IgdGhlIG1vbnRoIHZpZXcgY2hhbmdlcy5cbiAgICB0aGlzLl9zZXRUYWJJbmRleCgpO1xuXG4gICAgLy8gV2hlbiBjaGFuZ2luZyB2aWV3IHRvIHllYXIvbW9udGgsIHRoZSB0YWJpbmRleCBpcyBjaGFuZ2VkLCBidXQgdGhlIGZvY3VzZWQgZWxlbWVudCBpcyBnZXR0aW5nIGxvc3QuXG4gICAgLy8gV2UgbmVlZCB0byBjYWxsIGBfZm9jdXNDZWxsKClgIG1ldGhvZCBleHBsaWNpdGx5IHRvIGNvcnJlY3RseSBzZXQgdGhlIGZvY3VzLlxuICAgIHRoaXMuX2ZvY3VzQ2VsbCgpO1xuICB9XG5cbiAgcHJpdmF0ZSBfb25TbG90Q2hhbmdlID0gKCk6IHZvaWQgPT4ge1xuICAgIHRoaXMuX2VuaGFuY2VkVmFyaWFudCA9IEFycmF5LmZyb20odGhpcy5jaGlsZHJlbikuc29tZShcbiAgICAgIChjKSA9PiBjLmxvY2FsTmFtZSA9PT0gJ3NiYi1jYWxlbmRhci1kYXknLFxuICAgICk7XG4gICAgdGhpcy50b2dnbGVTdGF0ZSgnZW5oYW5jZWQnLCB0aGlzLl9lbmhhbmNlZFZhcmlhbnQpO1xuICAgIHRoaXMuX3NldFRhYkluZGV4KCk7XG4gIH07XG5cbiAgLyoqXG4gICAqIFRoZSBgX3NlbGVjdGVkYCBzdGF0ZSBzaG91bGQgYmUgYWRhcHRlZCB3aGVuIHRoZSBgbXVsdGlwbGVgIHByb3BlcnR5IGNoYW5nZXM6XG4gICAqICAgLSBpZiBpdCBjaGFuZ2VzIHRvIHRydWUsIHRoZSAnX3NlbGVjdGVkJyBpcyBzZXQgdG8gYW4gYXJyYXk7XG4gICAqICAgLSBpZiBpdCBjaGFuZ2VzIHRvIGZhbHNlLCB0aGUgZmlyc3QgYXZhaWxhYmxlIG9wdGlvbiBpcyBzZXQgYXMgJ3ZhbHVlJyBvdGhlcndpc2UgaXQncyBzZXQgdG8gbnVsbC5cbiAgICovXG4gIHByaXZhdGUgX29uTXVsdGlwbGVDaGFuZ2VkKGlzTXVsdGlwbGU6IGJvb2xlYW4pOiB2b2lkIHtcbiAgICBpZiAoaXNNdWx0aXBsZSAmJiAhQXJyYXkuaXNBcnJheSh0aGlzLl9zZWxlY3RlZCkpIHtcbiAgICAgIHRoaXMuX3NlbGVjdGVkID0gdGhpcy5fc2VsZWN0ZWQgPyBbdGhpcy5fc2VsZWN0ZWQgYXMgVF0gOiBbXTtcbiAgICB9XG4gICAgaWYgKCFpc011bHRpcGxlICYmIEFycmF5LmlzQXJyYXkodGhpcy5fc2VsZWN0ZWQpKSB7XG4gICAgICB0aGlzLl9zZWxlY3RlZCA9ICh0aGlzLl9zZWxlY3RlZCBhcyBUW10pLmxlbmd0aCA/ICh0aGlzLl9zZWxlY3RlZCBhcyBUW10pWzBdIDogbnVsbDtcbiAgICB9XG4gIH1cblxuICAvKiogSW5pdGlhbGl6ZXMgdGhlIGNvbXBvbmVudC4gKi9cbiAgcHJpdmF0ZSBfaW5pdChhY3RpdmVEYXRlPzogVCk6IHZvaWQge1xuICAgIC8vIER1ZSB0byBpdHMgY29tcGxleGl0eSwgdGhlIGNhbGVuZGFyIGlzIG9ubHkgaW5pdGlhbGl6ZWQgb24gY2xpZW50IHNpZGVcbiAgICBpZiAoaXNTZXJ2ZXIpIHtcbiAgICAgIHJldHVybjtcbiAgICB9IGVsc2UgaWYgKHRoaXMuaHlkcmF0aW9uUmVxdWlyZWQpIHtcbiAgICAgIHRoaXMuaHlkcmF0aW9uQ29tcGxldGUudGhlbigoKSA9PiB0aGlzLl9pbml0KCkpO1xuICAgICAgcmV0dXJuO1xuICAgIH1cblxuICAgIGlmIChhY3RpdmVEYXRlKSB7XG4gICAgICB0aGlzLl9hc3NpZ25BY3RpdmVEYXRlKGFjdGl2ZURhdGUpO1xuICAgIH1cbiAgICB0aGlzLl93aWRlID1cbiAgICAgICh0aGlzLl9tZWRpYU1hdGNoZXIubWF0Y2hlcyhTYmJNZWRpYVF1ZXJ5QnJlYWtwb2ludExhcmdlQW5kQWJvdmUpID8/IGZhbHNlKSAmJiB0aGlzLndpZGU7XG4gICAgdGhpcy5fd2Vla3MgPSB0aGlzLl9jcmVhdGVXZWVrUm93cyh0aGlzLl9hY3RpdmVEYXRlKTtcbiAgICB0aGlzLl95ZWFycyA9IHRoaXMuX2NyZWF0ZVllYXJSb3dzKCk7XG4gICAgdGhpcy5fd2Vla051bWJlcnMgPSB0aGlzLl9jcmVhdGVXZWVrTnVtYmVycyh0aGlzLl9hY3RpdmVEYXRlKTtcbiAgICB0aGlzLl9uZXh0TW9udGhXZWVrcyA9IFtbXV07XG4gICAgdGhpcy5fbmV4dE1vbnRoWWVhcnMgPSBbW11dO1xuICAgIGlmICh0aGlzLl93aWRlKSB7XG4gICAgICBjb25zdCBuZXh0TW9udGhEYXRlID0gdGhpcy5fZGF0ZUFkYXB0ZXIuYWRkQ2FsZW5kYXJNb250aHModGhpcy5fYWN0aXZlRGF0ZSwgMSk7XG4gICAgICB0aGlzLl9uZXh0TW9udGhXZWVrcyA9IHRoaXMuX2NyZWF0ZVdlZWtSb3dzKG5leHRNb250aERhdGUsIHRydWUpO1xuICAgICAgdGhpcy5fbmV4dE1vbnRoWWVhcnMgPSB0aGlzLl9jcmVhdGVZZWFyUm93cyhZRUFSU19QRVJfUEFHRSk7XG4gICAgICB0aGlzLl9uZXh0TW9udGhXZWVrTnVtYmVycyA9IHRoaXMuX2NyZWF0ZVdlZWtOdW1iZXJzKG5leHRNb250aERhdGUpO1xuICAgIH1cbiAgICB0aGlzLl9pbml0aWFsaXplZCA9IHRydWU7XG4gIH1cblxuICAvKiogRm9jdXNlcyBvbiBhIGRheSBjZWxsIHByaW9yaXRpemluZyB0aGUgc2VsZWN0ZWQgZGF5LCB0aGUgY3VycmVudCBkYXksIGFuZCBsYXN0bHksIHRoZSBmaXJzdCBzZWxlY3RhYmxlIGRheS4gKi9cbiAgcHJpdmF0ZSBfZm9jdXNDZWxsKCk6IHZvaWQge1xuICAgIGlmICh0aGlzLl9yZXNldEZvY3VzKSB7XG4gICAgICB0aGlzLl9nZXRGaXJzdEZvY3VzYWJsZSgpPy5mb2N1cygpO1xuICAgICAgdGhpcy5fcmVzZXRGb2N1cyA9IGZhbHNlO1xuICAgIH1cbiAgfVxuXG4gIC8qKiBDcmVhdGVzIHRoZSBhcnJheSBvZiB3ZWVrZGF5cy4gKi9cbiAgcHJpdmF0ZSBfc2V0V2Vla2RheXMoKTogdm9pZCB7XG4gICAgY29uc3QgbmFycm93V2Vla2RheXM6IHN0cmluZ1tdID0gdGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0RGF5T2ZXZWVrTmFtZXMoJ25hcnJvdycpO1xuICAgIGNvbnN0IGxvbmdXZWVrZGF5czogc3RyaW5nW10gPSB0aGlzLl9kYXRlQWRhcHRlci5nZXREYXlPZldlZWtOYW1lcygnbG9uZycpO1xuICAgIGNvbnN0IHdlZWtkYXlzOiBXZWVrZGF5W10gPSBsb25nV2Vla2RheXMubWFwKChsb25nOiBzdHJpbmcsIGk6IG51bWJlcikgPT4gKHtcbiAgICAgIGxvbmcsXG4gICAgICBuYXJyb3c6IG5hcnJvd1dlZWtkYXlzW2ldLFxuICAgIH0pKTtcblxuICAgIC8vIFJvdGF0ZXMgdGhlIGxhYmVscyBmb3IgZGF5cyBvZiB0aGUgd2VlayBiYXNlZCBvbiB0aGUgY29uZmlndXJlZCBmaXJzdCBkYXkgb2YgdGhlIHdlZWsuXG4gICAgY29uc3QgZmlyc3REYXlPZldlZWs6IG51bWJlciA9IHRoaXMuX2RhdGVBZGFwdGVyLmdldEZpcnN0RGF5T2ZXZWVrKCk7XG4gICAgdGhpcy5fd2Vla2RheXMgPSB3ZWVrZGF5cy5zbGljZShmaXJzdERheU9mV2VlaykuY29uY2F0KHdlZWtkYXlzLnNsaWNlKDAsIGZpcnN0RGF5T2ZXZWVrKSk7XG4gIH1cblxuICAvKipcbiAgICogR2l2ZW4gYSBkYXRlLCBpdCByZXR1cm5zIHRoZSB3ZWVrIG51bWJlcnMgZm9yIHRoZSBtb250aCB0aGUgZGF0ZSBiZWxvbmdzIHRvLlxuICAgKiBUT0RPOiBjaGVjayBpZiBkYXRlLWZucyBjYW4gYmUgcmVwbGFjZWQgd2l0aCBjdXN0b20gbG9naWMuXG4gICAqXG4gICAqIFNpbmNlIHRoZSBjYWxjdWxhdGlvbiBpcyBub3Qgc2ltcGxlIChzZWUgaHR0cHM6Ly9lbi53aWtpcGVkaWEub3JnL3dpa2kvV2VlayNOdW1iZXJpbmcpLFxuICAgKiB0aGUgZGF0ZS1mbnMgbGlicmFyeSBoYXMgYmVlbiB1c2VkIHRoaXMgd2F5OlxuICAgKiB0aGUgZmlyc3QgYW5kIHRoZSBsYXN0IGRheSBvZiB0aGUgbW9udGggYXJlIGNhbGN1bGF0ZWQgYW5kIHRoZW4gcGFzc2VkIHRvIHRoZSBgZWFjaFdlZWtPZkludGVydmFsYCBmdW5jdGlvbixcbiAgICogd2hpY2ggcmV0dXJucyBhbiBhcnJheSBjb250YWluaW5nIHRoZSBzdGFydGluZyBkYXkgb2YgZXZlcnkgSVNPIHdlZWsgb2YgdGhlIG1vbnRoLFxuICAgKiBjb25zaWRlcmluZyBNb25kYXkgYXMgdGhlIGZpcnN0IGRheS5cbiAgICogVGhlbiwgdGhpcyBhcnJheSBpcyBtYXBwZWQgdmlhIHRoZSBgZ2V0V2Vla2AgZnVuY3Rpb24sIHdoaWNoIHJldHVybnMgdGhlIElTTyB3ZWVrIG51bWJlciBmb3IgdGhhdCBkYXRlLlxuICAgKi9cbiAgcHJpdmF0ZSBfY3JlYXRlV2Vla051bWJlcnMoZGF0ZTogVCk6IG51bWJlcltdIHtcbiAgICByZXR1cm4gZWFjaFdlZWtPZkludGVydmFsKFxuICAgICAgeyBzdGFydDogc3RhcnRPZk1vbnRoKGRhdGUgYXMgRGF0ZSksIGVuZDogZW5kT2ZNb250aChkYXRlIGFzIERhdGUpIH0sXG4gICAgICB7IHdlZWtTdGFydHNPbjogMSB9LFxuICAgICkubWFwKChmaXJzdERheU9mV2VlazogRGF0ZSkgPT5cbiAgICAgIGdldFdlZWsoZmlyc3REYXlPZldlZWssIHsgd2Vla1N0YXJ0c09uOiAxLCBmaXJzdFdlZWtDb250YWluc0RhdGU6IDQgfSksXG4gICAgKTtcbiAgfVxuXG4gIC8qKiBDcmVhdGVzIHRoZSByb3dzIGFsb25nIHRoZSBob3Jpem9udGFsIGRpcmVjdGlvbiBhbmQgc2V0cyB0aGUgcGFyYW1ldGVycyB1c2VkIGluIGtleWJvYXJkIG5hdmlnYXRpb24uICovXG4gIHByaXZhdGUgX2NyZWF0ZVdlZWtSb3dzKHZhbHVlOiBULCBpc1NlY29uZE1vbnRoSW5WaWV3ID0gZmFsc2UpOiBEYXk8VD5bXVtdIHtcbiAgICBjb25zdCBkYXlzSW5Nb250aDogbnVtYmVyID0gdGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0TnVtRGF5c0luTW9udGgodmFsdWUpO1xuICAgIGNvbnN0IHdlZWtPZmZzZXQ6IG51bWJlciA9IHRoaXMuX2RhdGVBZGFwdGVyLmdldEZpcnN0V2Vla09mZnNldCh2YWx1ZSk7XG4gICAgaWYgKCFpc1NlY29uZE1vbnRoSW5WaWV3KSB7XG4gICAgICB0aGlzLl9rZXlib2FyZE5hdmlnYXRpb25EYXlWaWV3UGFyYW1ldGVycy5maXJzdE1vbnRoTGVuZ3RoID0gZGF5c0luTW9udGg7XG4gICAgICB0aGlzLl9rZXlib2FyZE5hdmlnYXRpb25EYXlWaWV3UGFyYW1ldGVycy5maXJzdE1vbnRoT2Zmc2V0ID0gd2Vla09mZnNldDtcbiAgICAgIHRoaXMuX2tleWJvYXJkTmF2aWdhdGlvbkRheVZpZXdQYXJhbWV0ZXJzLmZpcnN0RGF5SW5WaWV3ID0gdGhpcy5fZGF0ZUFkYXB0ZXIudG9Jc284NjAxKFxuICAgICAgICB0aGlzLl9kYXRlQWRhcHRlci5jcmVhdGVEYXRlKFxuICAgICAgICAgIHRoaXMuX2RhdGVBZGFwdGVyLmdldFllYXIodmFsdWUpLFxuICAgICAgICAgIHRoaXMuX2RhdGVBZGFwdGVyLmdldE1vbnRoKHZhbHVlKSxcbiAgICAgICAgICAxLFxuICAgICAgICApLFxuICAgICAgKTtcbiAgICAgIHRoaXMuX2tleWJvYXJkTmF2aWdhdGlvbkRheVZpZXdQYXJhbWV0ZXJzLmxhc3REYXlJblZpZXcgPSB0aGlzLl9kYXRlQWRhcHRlci50b0lzbzg2MDEoXG4gICAgICAgIHRoaXMuX2RhdGVBZGFwdGVyLmNyZWF0ZURhdGUoXG4gICAgICAgICAgdGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0WWVhcih2YWx1ZSksXG4gICAgICAgICAgdGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0TW9udGgodmFsdWUpLFxuICAgICAgICAgIGRheXNJbk1vbnRoLFxuICAgICAgICApLFxuICAgICAgKTtcbiAgICB9IGVsc2Uge1xuICAgICAgdGhpcy5fa2V5Ym9hcmROYXZpZ2F0aW9uRGF5Vmlld1BhcmFtZXRlcnMuc2Vjb25kTW9udGhPZmZzZXQgPSB3ZWVrT2Zmc2V0O1xuICAgICAgdGhpcy5fa2V5Ym9hcmROYXZpZ2F0aW9uRGF5Vmlld1BhcmFtZXRlcnMubGFzdERheUluVmlldyA9IHRoaXMuX2RhdGVBZGFwdGVyLnRvSXNvODYwMShcbiAgICAgICAgdGhpcy5fZGF0ZUFkYXB0ZXIuY3JlYXRlRGF0ZShcbiAgICAgICAgICB0aGlzLl9kYXRlQWRhcHRlci5nZXRZZWFyKHZhbHVlKSxcbiAgICAgICAgICB0aGlzLl9kYXRlQWRhcHRlci5nZXRNb250aCh2YWx1ZSksXG4gICAgICAgICAgZGF5c0luTW9udGgsXG4gICAgICAgICksXG4gICAgICApO1xuICAgIH1cbiAgICByZXR1cm4gdGhpcy5vcmllbnRhdGlvbiA9PT0gJ2hvcml6b250YWwnXG4gICAgICA/IHRoaXMuX2NyZWF0ZVdlZWtSb3dzSG9yaXpvbnRhbCh2YWx1ZSwgZGF5c0luTW9udGgsIHdlZWtPZmZzZXQpXG4gICAgICA6IHRoaXMuX2NyZWF0ZVdlZWtSb3dzVmVydGljYWwodmFsdWUsIGRheXNJbk1vbnRoLCB3ZWVrT2Zmc2V0KTtcbiAgfVxuXG4gIC8qKlxuICAgKiBDcmVhdGVzIHRoZSByb3dzIGZvciBlYWNoIHdlZWsgaW4gb3JpZW50YXRpb249J2hvcml6b250YWwnLlxuICAgKlxuICAgKiBJdGVyYXRlcyB0aHJvdWdoIHRoZSBkYXlzIG9mIHRoZSBtb250aHMsIGNyZWF0ZXMgYSBEYXkgb2JqZWN0IGZvciBlYWNoIGFuZCBwdXNoZXMgaXQgaW50byBhbmQgYXJyYXkuXG4gICAqIEVhY2ggc2V2ZW4gZGF5cyAoY29uc2lkZXJpbmcgdGhlIG9mZnNldCBhdCB0aGUgYmVnaW5uaW5nIG9mIHRoZSBtb250aCkgcmVzdGFydHMgZnJvbSBhbiBlbXB0eSBhcnJheS5cbiAgICpcbiAgICogVGhlIHJlc3VsdCBpcyBhIG1hdHJpeCBpbiB3aGljaCBldmVyeSByb3cgaXMgYSB3ZWVrIChvciBwYXJ0IG9mIGl0LCBjb25zaWRlcmluZyBvZmZzZXQpLlxuICAgKi9cbiAgcHJpdmF0ZSBfY3JlYXRlV2Vla1Jvd3NIb3Jpem9udGFsKHZhbHVlOiBULCBkYXlzSW5Nb250aDogbnVtYmVyLCB3ZWVrT2Zmc2V0OiBudW1iZXIpOiBEYXk8VD5bXVtdIHtcbiAgICBjb25zdCB3ZWVrczogRGF5PFQ+W11bXSA9IFtbXV07XG4gICAgZm9yIChsZXQgaSA9IDAsIGNlbGwgPSB3ZWVrT2Zmc2V0OyBpIDwgZGF5c0luTW9udGg7IGkrKywgY2VsbCsrKSB7XG4gICAgICBpZiAoY2VsbCA9PT0gREFZU19QRVJfUk9XKSB7XG4gICAgICAgIHdlZWtzLnB1c2goW10pO1xuICAgICAgICBjZWxsID0gMDtcbiAgICAgIH1cbiAgICAgIGNvbnN0IGRhdGUgPSB0aGlzLl9kYXRlQWRhcHRlci5jcmVhdGVEYXRlKFxuICAgICAgICB0aGlzLl9kYXRlQWRhcHRlci5nZXRZZWFyKHZhbHVlKSxcbiAgICAgICAgdGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0TW9udGgodmFsdWUpLFxuICAgICAgICBpICsgMSxcbiAgICAgICkhO1xuICAgICAgd2Vla3Nbd2Vla3MubGVuZ3RoIC0gMV0ucHVzaCh0aGlzLl9tYXBEYXRlVG9EYXkoZGF0ZSkpO1xuICAgIH1cbiAgICByZXR1cm4gd2Vla3M7XG4gIH1cblxuICAvKipcbiAgICogQ3JlYXRlcyB0aGUgcm93cyBmb3IgZWFjaCB3ZWVrIGluIG9yaWVudGF0aW9uPSd2ZXJ0aWNhbCcuXG4gICAqXG4gICAqIENyZWF0ZXMgYSBtYXRyaXggd2l0aCBzZXZlbiBlbXB0eSByb3dzLlxuICAgKiBJdGVyYXRlcyB0aHJvdWdoIHRoZSBkYXlzIG9mIHRoZSBtb250aHMsIGNyZWF0ZXMgYSBEYXkgb2JqZWN0IGZvciBlYWNoXG4gICAqIGFuZCBwdXNoZXMgaXQgaW50byB0aGUgY29ycmVjdCBhcnJheSBjb25zaWRlcmluZyB0aGUgb2Zmc2V0IGF0IHRoZSBiZWdpbm5pbmcgb2YgdGhlIG1vbnRoLlxuICAgKiBFYWNoIHNldmVuIGRheXMgKGluY2x1ZGluZyBvZmZzZXQpIHJlc3RhcnRzIGZyb20gdGhlIGZpcnN0LlxuICAgKlxuICAgKiBUaGUgcmVzdWx0IGlzIGEgbWF0cml4IGluIHdoaWNoIGV2ZXJ5IHJvdyBpcyBhIHNldCBvZiB3ZWVrZGF5cywgc286XG4gICAqIC0gcm93IDA6IGFsbCB0aGUgTW9uZGF5cztcbiAgICogLSByb3cgMTogYWxsIHRoZSBUdWVzZGF5cztcbiAgICogLSAuLi5cbiAgICogLSByb3cgNzogYWxsIHRoZSBTdW5kYXlzLlxuICAgKi9cbiAgcHJpdmF0ZSBfY3JlYXRlV2Vla1Jvd3NWZXJ0aWNhbCh2YWx1ZTogVCwgZGF5c0luTW9udGg6IG51bWJlciwgd2Vla09mZnNldDogbnVtYmVyKTogRGF5PFQ+W11bXSB7XG4gICAgY29uc3Qgd2Vla3M6IERheTxUPltdW10gPSBBcnJheS5mcm9tKHsgbGVuZ3RoOiBEQVlTX1BFUl9ST1cgfSwgKCkgPT4gW10pO1xuICAgIGZvciAobGV0IGkgPSAwLCBjZWxsID0gd2Vla09mZnNldDsgaSA8IGRheXNJbk1vbnRoOyBpKyssIGNlbGwrKykge1xuICAgICAgaWYgKGNlbGwgPT09IERBWVNfUEVSX1JPVykge1xuICAgICAgICBjZWxsID0gMDtcbiAgICAgIH1cbiAgICAgIGNvbnN0IGRhdGUgPSB0aGlzLl9kYXRlQWRhcHRlci5jcmVhdGVEYXRlKFxuICAgICAgICB0aGlzLl9kYXRlQWRhcHRlci5nZXRZZWFyKHZhbHVlKSxcbiAgICAgICAgdGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0TW9udGgodmFsdWUpLFxuICAgICAgICBpICsgMSxcbiAgICAgICkhO1xuICAgICAgd2Vla3NbY2VsbF0ucHVzaCh0aGlzLl9tYXBEYXRlVG9EYXkoZGF0ZSkpO1xuICAgIH1cbiAgICByZXR1cm4gd2Vla3M7XG4gIH1cblxuICBwcml2YXRlIF9tYXBEYXRlVG9EYXkoZGF0ZTogVCk6IERheTxUPiB7XG4gICAgY29uc3QgaXNvRGF0ZSA9IHRoaXMuX2RhdGVBZGFwdGVyLnRvSXNvODYwMShkYXRlKTtcbiAgICByZXR1cm4ge1xuICAgICAgdmFsdWU6IGlzb0RhdGUsXG4gICAgICBkYXRlVmFsdWU6IGRhdGUsXG4gICAgICBkYXlWYWx1ZTogU3RyaW5nKHRoaXMuX2RhdGVBZGFwdGVyLmdldERhdGUoZGF0ZSkpLFxuICAgICAgbW9udGhWYWx1ZTogU3RyaW5nKHRoaXMuX2RhdGVBZGFwdGVyLmdldE1vbnRoKGRhdGUpKSxcbiAgICAgIHllYXJWYWx1ZTogU3RyaW5nKHRoaXMuX2RhdGVBZGFwdGVyLmdldFllYXIoZGF0ZSkpLFxuICAgICAgd2Vla1ZhbHVlOiBnZXRXZWVrKGlzb0RhdGUsIHsgd2Vla1N0YXJ0c09uOiAxLCBmaXJzdFdlZWtDb250YWluc0RhdGU6IDQgfSksXG4gICAgICB3ZWVrRGF5VmFsdWU6IHRoaXMuX2RhdGVBZGFwdGVyLmdldERheU9mV2VlayhkYXRlKSxcbiAgICB9O1xuICB9XG5cbiAgLyoqIEZvcmNlIHRoZSBjb252ZXJzaW9uIHRvIElTTzg2MDEgZm9ybWF0dGVkIHN0cmluZy4gKi9cbiAgcHJpdmF0ZSBfbWFwVmFsdWVUb0lTT0RhdGUodmFsdWU6IHN0cmluZyB8IERhdGUpOiBzdHJpbmcge1xuICAgIHJldHVybiB0eXBlb2YgdmFsdWUgPT09ICdzdHJpbmcnID8gdmFsdWUgOiB0aGlzLl9kYXRlQWRhcHRlci50b0lzbzg2MDEodmFsdWUgYXMgVCk7XG4gIH1cblxuICAvKiogQ3JlYXRlcyB0aGUgcm93cyBmb3IgdGhlIG1vbnRoIHNlbGVjdGlvbiB2aWV3LiAqL1xuICBwcml2YXRlIF9jcmVhdGVNb250aFJvd3MoKTogdm9pZCB7XG4gICAgY29uc3QgbW9udGhzOiBNb250aENlbGxbXSA9IG5ldyBBcnJheSgxMikuZmlsbChudWxsKS5tYXAoXG4gICAgICAoXywgaTogbnVtYmVyKTogTW9udGhDZWxsID0+ICh7XG4gICAgICAgIHZhbHVlOiBTdHJpbmcoaSArIDEpLnBhZFN0YXJ0KDIsICcwJyksXG4gICAgICAgIG1vbnRoVmFsdWU6IGkgKyAxLFxuICAgICAgfSksXG4gICAgKTtcbiAgICBjb25zdCByb3dzOiBudW1iZXIgPSAxMiAvIE1PTlRIU19QRVJfUk9XO1xuICAgIGNvbnN0IG1vbnRoQXJyYXk6IE1vbnRoQ2VsbFtdW10gPSBbXTtcbiAgICBmb3IgKGxldCBpOiBudW1iZXIgPSAwOyBpIDwgcm93czsgaSsrKSB7XG4gICAgICBtb250aEFycmF5LnB1c2gobW9udGhzLnNsaWNlKE1PTlRIU19QRVJfUk9XICogaSwgTU9OVEhTX1BFUl9ST1cgKiAoaSArIDEpKSk7XG4gICAgfVxuICAgIHRoaXMuX21vbnRocyA9IG1vbnRoQXJyYXk7XG4gIH1cblxuICAvKiogQ3JlYXRlcyB0aGUgcm93cyBmb3IgdGhlIHllYXIgc2VsZWN0aW9uIHZpZXcuICovXG4gIHByaXZhdGUgX2NyZWF0ZVllYXJSb3dzKG9mZnNldDogbnVtYmVyID0gMCk6IG51bWJlcltdW10ge1xuICAgIGNvbnN0IHN0YXJ0VmFsdWVZZWFyVmlldzogbnVtYmVyID0gdGhpcy5fZ2V0U3RhcnRWYWx1ZVllYXJWaWV3KCk7XG4gICAgY29uc3QgYWxsWWVhcnM6IG51bWJlcltdID0gbmV3IEFycmF5KFlFQVJTX1BFUl9QQUdFKVxuICAgICAgLmZpbGwoMClcbiAgICAgIC5tYXAoKF8sIGk6IG51bWJlcikgPT4gc3RhcnRWYWx1ZVllYXJWaWV3ICsgb2Zmc2V0ICsgaSk7XG4gICAgY29uc3Qgcm93czogbnVtYmVyID0gWUVBUlNfUEVSX1BBR0UgLyBZRUFSU19QRVJfUk9XO1xuICAgIGNvbnN0IHllYXJBcnJheTogbnVtYmVyW11bXSA9IFtdO1xuICAgIGZvciAobGV0IGk6IG51bWJlciA9IDA7IGkgPCByb3dzOyBpKyspIHtcbiAgICAgIHllYXJBcnJheS5wdXNoKGFsbFllYXJzLnNsaWNlKFlFQVJTX1BFUl9ST1cgKiBpLCBZRUFSU19QRVJfUk9XICogKGkgKyAxKSkpO1xuICAgIH1cbiAgICByZXR1cm4geWVhckFycmF5O1xuICB9XG5cbiAgLyoqXG4gICAqIENhbGN1bGF0ZXMgdGhlIGZpcnN0IHllYXIgdGhhdCB3aWxsIGJlIHNob3duIGluIHRoZSB5ZWFyIHNlbGVjdGlvbiBwYW5lbC5cbiAgICogSWYgYG1pbkRhdGVgIGFuZCBgbWF4RGF0ZWAgYXJlIGJvdGggbnVsbCwgdGhlIHN0YXJ0aW5nIHllYXIgaXMgY2FsY3VsYXRlZCBhc1xuICAgKiB0aGUgbXVsdGlwbGUgb2YgWUVBUlNfUEVSX1BBR0UgY2xvc2VzdCB0byBhbmQgbGVzcyB0aGFuIGFjdGl2ZURhdGUsXG4gICAqIGUuZy4sIHdpdGggYFlFQVJTX1BFUl9QQUdFYCA9IDI0IGFuZCBgYWN0aXZlRGF0ZWAgPSAyMDIwLCB0aGUgZnVuY3Rpb24gd2lsbCByZXR1cm4gMjAxNiAoMjQgKiA4MyksXG4gICAqIHdoaWxlIHdpdGggYGFjdGl2ZURhdGVgID0gMjAwMCwgdGhlIGZ1bmN0aW9uIHdpbGwgcmV0dXJuIDE5OTIgKDI0ICogODIpLlxuICAgKiBJZiBgbWluRGF0ZWAgaXMgbm90IG51bGwsIGl0IHJldHVybnMgdGhlIGNvcnJlc3BvbmRpbmcgeWVhcjsgaWYgYG1heERhdGVgIGlzIG5vdCBudWxsLFxuICAgKiBpdCByZXR1cm5zIHRoZSBjb3JyZXNwb25kaW5nIHllYXIgbWludXMgYFlFQVJTX1BFUl9QQUdFYCwgc28gdGhhdCB0aGUgYG1heERhdGVgIGlzIHRoZSBsYXN0IHJlbmRlcmVkIHllYXIuXG4gICAqIElmIGJvdGggYXJlIG5vdCBudWxsLCBgbWF4RGF0ZWAgaGFzIHByaW9yaXR5IG92ZXIgYG1pbkRhdGVgLlxuICAgKi9cbiAgcHJpdmF0ZSBfZ2V0U3RhcnRWYWx1ZVllYXJWaWV3KCk6IG51bWJlciB7XG4gICAgbGV0IHN0YXJ0aW5nWWVhciA9IDA7XG4gICAgaWYgKHRoaXMubWF4KSB7XG4gICAgICBzdGFydGluZ1llYXIgPSB0aGlzLl9kYXRlQWRhcHRlci5nZXRZZWFyKHRoaXMubWF4KSAtIFlFQVJTX1BFUl9QQUdFICsgMTtcbiAgICB9IGVsc2UgaWYgKHRoaXMubWluKSB7XG4gICAgICBzdGFydGluZ1llYXIgPSB0aGlzLl9kYXRlQWRhcHRlci5nZXRZZWFyKHRoaXMubWluKTtcbiAgICB9XG4gICAgY29uc3QgYWN0aXZlWWVhciA9IHRoaXMuX2RhdGVBZGFwdGVyLmdldFllYXIodGhpcy5fYWN0aXZlRGF0ZSk7XG4gICAgcmV0dXJuIChcbiAgICAgIGFjdGl2ZVllYXIgLVxuICAgICAgKCgoKGFjdGl2ZVllYXIgLSBzdGFydGluZ1llYXIpICUgWUVBUlNfUEVSX1BBR0UpICsgWUVBUlNfUEVSX1BBR0UpICUgWUVBUlNfUEVSX1BBR0UpXG4gICAgKTtcbiAgfVxuXG4gIC8qKiBDaGVja3MgaWYgZGF0ZSBpcyB3aXRoaW4gdGhlIG1pbi1tYXggcmFuZ2UuICovXG4gIHByaXZhdGUgX2lzRGF5SW5SYW5nZShkYXRlU3RyaW5nOiBzdHJpbmcpOiBib29sZWFuIHtcbiAgICBpZiAoIXRoaXMubWluICYmICF0aGlzLm1heCkge1xuICAgICAgcmV0dXJuIHRydWU7XG4gICAgfVxuICAgIGNvbnN0IGRhdGUgPSB0aGlzLl9kYXRlQWRhcHRlci5kZXNlcmlhbGl6ZShkYXRlU3RyaW5nKSE7XG4gICAgcmV0dXJuIHRoaXMuX2RhdGVBZGFwdGVyLnNhbWVEYXRlKGRhdGUsIHRoaXMuX2RhdGVBZGFwdGVyLmNsYW1wRGF0ZShkYXRlLCB0aGlzLm1pbiwgdGhpcy5tYXgpKTtcbiAgfVxuXG4gIC8qKiBFbWl0cyB0aGUgc2VsZWN0ZWQgZGF0ZSBhbmQgc2V0cyBpdCBpbnRlcm5hbGx5LiAqL1xuICBwcml2YXRlIF9zZWxlY3REYXRlKGRheTogVCk6IHZvaWQge1xuICAgIHRoaXMuX2Nob3Nlbk1vbnRoID0gdW5kZWZpbmVkO1xuICAgIHRoaXMuX3NldENob3NlblllYXIoKTtcbiAgICBpZiAodGhpcy5tdWx0aXBsZSkge1xuICAgICAgLy8gQ2hlY2sgaWYgX3NlbGVjdGVkIGhhcyBlbGVtZW50c1xuICAgICAgaWYgKHRoaXMuX3NlbGVjdGVkICYmICh0aGlzLl9zZWxlY3RlZCBhcyBUW10pLmxlbmd0aCA+IDApIHtcbiAgICAgICAgY29uc3QgaW5kZXhPZlNlbGVjdGVkRGF5OiBudW1iZXIgPSAodGhpcy5fc2VsZWN0ZWQgYXMgVFtdKS5maW5kSW5kZXgoXG4gICAgICAgICAgKHNlbCkgPT4gdGhpcy5fZGF0ZUFkYXB0ZXIuY29tcGFyZURhdGUoc2VsLCBkYXkpID09PSAwLFxuICAgICAgICApO1xuICAgICAgICAvLyBJZiB0aGUgc2VsZWN0ZWQgZGF0ZSBpcyBhbHJlYWR5IGluIHRoZSBfc2VsZWN0ZWQgYXJyYXksIHJlbW92ZSBpdCwgb3RoZXJ3aXNlIGFkZCBpdFxuICAgICAgICBpZiAoaW5kZXhPZlNlbGVjdGVkRGF5ICE9PSAtMSkge1xuICAgICAgICAgIHRoaXMuX3NlbGVjdGVkID0gKHRoaXMuX3NlbGVjdGVkIGFzIFRbXSkuZmlsdGVyKChfLCBpKSA9PiBpICE9PSBpbmRleE9mU2VsZWN0ZWREYXkpO1xuICAgICAgICB9IGVsc2Uge1xuICAgICAgICAgIHRoaXMuX3NlbGVjdGVkID0gWy4uLih0aGlzLl9zZWxlY3RlZCBhcyBUW10pLCBkYXldO1xuICAgICAgICB9XG4gICAgICB9IGVsc2Uge1xuICAgICAgICAvLyBJZiBfc2VsZWN0ZWQgaXMgZW1wdHksIHNldCBpdFxuICAgICAgICB0aGlzLl9zZWxlY3RlZCA9IFtkYXldO1xuICAgICAgfVxuICAgICAgdGhpcy5fZW1pdERhdGVTZWxlY3RlZEV2ZW50KHRoaXMuX3NlbGVjdGVkLm1hcCgoZSkgPT4gdGhpcy5fZGF0ZUFkYXB0ZXIuZGVzZXJpYWxpemUoZSkhKSk7XG4gICAgfSBlbHNlIHtcbiAgICAgIC8vIEluIHNpbmdsZSBzZWxlY3Rpb24sIGNoZWNrIGlmIHRoZSBkYXkgaXMgYWxyZWFkeSBzZWxlY3RlZFxuICAgICAgaWYgKCF0aGlzLl9zZWxlY3RlZCB8fCB0aGlzLl9kYXRlQWRhcHRlci5jb21wYXJlRGF0ZSh0aGlzLl9zZWxlY3RlZCBhcyBULCBkYXkpICE9PSAwKSB7XG4gICAgICAgIHRoaXMuX3NlbGVjdGVkID0gZGF5O1xuICAgICAgICB0aGlzLl9lbWl0RGF0ZVNlbGVjdGVkRXZlbnQodGhpcy5fZGF0ZUFkYXB0ZXIuZGVzZXJpYWxpemUoZGF5KSEpO1xuICAgICAgfVxuICAgIH1cbiAgfVxuXG4gIC8qKlxuICAgKiBIYW5kbGUgbXVsdGlwbGUgZGF0ZXMgc2VsZWN0aW9uIHZpYSB3ZWVrTnVtYmVyIC8gd2Vla0RheSBidXR0b25zOlxuICAgKiAtIGlmIENtZCBvciBDdHJsIGFyZSBwcmVzc2VkLCBhZGQgdGhlIG5ldyBkYXRlIHRvIHRoZSBjdXJyZW50IG9uZXM7XG4gICAqIC0gaWYgbm90LFxuICAgKiAgICAgLSBpZiB0aGUgbmV3IGRhdGVzIGFyZSB0aGUgc2FtZSBvZiB0aGUgY3VycmVudCBvbmVzLCBpdCBtZWFucyB0aGF0IHRoZSBzYW1lIGJ1dHRvbiBoYXMgYmVlbiBjbGlja2VkIHR3aWNlLCBzbyBkbyBub3RoaW5nO1xuICAgKiAgICAgLSBpZiBub3QsIHRoZSBzZWxlY3RlZCBkYXRlcyBhcmUgdGhlIG5ldyBvbmVzLlxuICAgKi9cbiAgcHJpdmF0ZSBfc2VsZWN0TXVsdGlwbGVEYXRlcyhkYXlzOiBEYXk8VD5bXSk6IHZvaWQge1xuICAgIC8vIEZpbHRlciBkaXNhYmxlZCBkYXlzIGJ5IG1hdGNoaW5nIHRoZSBwcm92aWRlZCBgZGF5c2AgcGFyYW1ldGVyIGFnYWluc3QgdGhlIGVuYWJsZWQgY2VsbHMuXG4gICAgLy8gU2luY2UgdGhlIGJ1dHRvbnMnIHZhbHVlIGlzIHNldCB0byB0aGUgRGF5J3MgaW50ZXJmYWNlIHZhbHVlIChJU08gc3RyaW5nKSwgdGhlcmUncyBubyBuZWVkIHRvIGRlc2VyaWFsaXplIGl0LlxuICAgIGNvbnN0IGVuYWJsZWREYXlzOiBzdHJpbmdbXSA9ICh0aGlzLl9jZWxscyBhcyBTYmJDYWxlbmRhckRheUVsZW1lbnRbXSlcbiAgICAgIC5maWx0ZXIoKGUpID0+ICFlLmRpc2FibGVkKVxuICAgICAgLm1hcCgoZSkgPT4gdGhpcy5fbWFwVmFsdWVUb0lTT0RhdGUoZS52YWx1ZSEpKTtcbiAgICBjb25zdCBkYXlzVG9BZGQ6IHN0cmluZ1tdID0gZGF5c1xuICAgICAgLm1hcCgoZTogRGF5PFQ+KSA9PiBlLnZhbHVlKVxuICAgICAgLmZpbHRlcigoaXNvRGF0ZTogc3RyaW5nKSA9PiBlbmFibGVkRGF5cy5pbmNsdWRlcyhpc29EYXRlKSk7XG4gICAgY29uc3QgZGF5c1RvQWRkU2V0ID0gbmV3IFNldChkYXlzVG9BZGQpO1xuICAgIGNvbnN0IHNlbGVjdGVkU2V0ID0gbmV3IFNldCgodGhpcy5fc2VsZWN0ZWQgYXMgVFtdKS5tYXAoKHMpID0+IHRoaXMuX2RhdGVBZGFwdGVyLnRvSXNvODYwMShzKSkpO1xuICAgIGNvbnN0IHNlbFN0cmluZ3MgPSB0aGlzLl91cGRhdGVTZWxlY3RlZFdpdGhNdWx0aXBsZURhdGVzKGRheXNUb0FkZCwgZGF5c1RvQWRkU2V0LCBzZWxlY3RlZFNldCk7XG4gICAgdGhpcy5fc2VsZWN0ZWQgPSBzZWxTdHJpbmdzLm1hcCgocykgPT4gdGhpcy5fZGF0ZUFkYXB0ZXIuZGVzZXJpYWxpemUocykhKTtcblxuICAgIHRoaXMuX2VtaXREYXRlU2VsZWN0ZWRFdmVudCh0aGlzLl9zZWxlY3RlZC5tYXAoKGUpID0+IHRoaXMuX2RhdGVBZGFwdGVyLmRlc2VyaWFsaXplKGUpISkpO1xuICB9XG5cbiAgLyoqXG4gICAqIEVtaXRzIHRoZSBkYXRlc2VsZWN0ZWQgZXZlbnQgZ2l2ZW4gdGhlIGRldGFpbCAoYXMgVCBvciBUW10gYmFzZWQgb24gdGhlIHZhbHVlIG9mIHRoZSBtdWx0aXBsZSBmbGFnKS5cbiAgICovXG4gIHByaXZhdGUgX2VtaXREYXRlU2VsZWN0ZWRFdmVudChkZXRhaWw6IFQgfCBUW10pOiB2b2lkIHtcbiAgICAvKiogQHR5cGUge0N1c3RvbUV2ZW50PFQgfCBUW10+fSBFdmVudCBlbWl0dGVkIG9uIGRhdGUgc2VsZWN0aW9uLiAqL1xuICAgIHRoaXMuZGlzcGF0Y2hFdmVudChcbiAgICAgIG5ldyBDdXN0b21FdmVudDxUIHwgVFtdPignZGF0ZXNlbGVjdGVkJywge1xuICAgICAgICBkZXRhaWwsXG4gICAgICAgIGNvbXBvc2VkOiB0cnVlLFxuICAgICAgICBidWJibGVzOiB0cnVlLFxuICAgICAgfSksXG4gICAgKTtcbiAgfVxuXG4gIHByaXZhdGUgX2VtaXRNb250aENoYW5nZSgpOiB2b2lkIHtcbiAgICAvLyBGSVhNRTogdGhlIG5hbWUgb2YgdGhpcyB2YXJpYWJsZSBhcHBlYXJzIGFzIGV2ZW50IG5hbWUgaW4gdGhlIHJlYWRtZVxuICAgIC8vICBkdWUgdG8gYSBidWcgaW4gdGhlIGN1c3RvbS1lbGVtZW50cy1tYW5pZmVzdCBsaWJyYXJ5LlxuICAgIC8vICBodHRwczovL2dpdGh1Yi5jb20vb3Blbi13Yy9jdXN0b20tZWxlbWVudHMtbWFuaWZlc3QvaXNzdWVzLzE0OVxuICAgIGNvbnN0IG1vbnRoY2hhbmdlID0gKHRoaXMud2lkZSA/IFsuLi50aGlzLl93ZWVrcywgLi4udGhpcy5fbmV4dE1vbnRoV2Vla3NdIDogdGhpcy5fd2Vla3MpXG4gICAgICAuZmxhdCgpXG4gICAgICAuc29ydCgoYSwgYikgPT4gYS52YWx1ZS5sb2NhbGVDb21wYXJlKGIudmFsdWUpKSBhcyBEYXlbXTtcbiAgICAvKipcbiAgICAgKiBAdHlwZSB7U2JiTW9udGhDaGFuZ2VFdmVudH1cbiAgICAgKiBFbWl0cyB3aGVuIHRoZSBtb250aCBjaGFuZ2VzLlxuICAgICAqIFRoZSBgcmFuZ2VgIHByb3BlcnR5IGNvbnRhaW5zIHRoZSBkYXlzIGFycmF5IG9mIHRoZSBjaG9zZW4gbW9udGguXG4gICAgICovXG4gICAgdGhpcy5kaXNwYXRjaEV2ZW50KG5ldyBTYmJNb250aENoYW5nZUV2ZW50KG1vbnRoY2hhbmdlKSk7XG4gIH1cblxuICAvKipcbiAgICogSW4gY2FzZSBvZiBtdWx0aXBsZSBzZWxlY3Rpb24sIG5ld2x5IGFkZGVkIGRheXMgbXVzdCBiZSBhZGRlZCB0byB0aGUgZXhpc3Rpbmcgb25lcywgd2l0aG91dCBkdXBsaWNhdGlvbi5cbiAgICogSWYgdGhlIGRheXMgdG8gYWRkIGFyZSBleGFjdGx5IHRoZSBzYW1lIGFzIHRoZSBzZWxlY3RlZCBvbmVzLCB0aGUgc2V0IG11c3QgYmUgZW1wdGllZC5cbiAgICovXG4gIHByaXZhdGUgX3VwZGF0ZVNlbGVjdGVkV2l0aE11bHRpcGxlRGF0ZXMoXG4gICAgZGF5c1RvQWRkOiBzdHJpbmdbXSxcbiAgICBkYXlzVG9BZGRTZXQ6IFNldDxzdHJpbmc+LFxuICAgIHNlbGVjdGVkU2V0OiBTZXQ8c3RyaW5nPixcbiAgKTogc3RyaW5nW10ge1xuICAgIGlmIChkYXlzVG9BZGQuZXZlcnkoKGRheTogc3RyaW5nKSA9PiBzZWxlY3RlZFNldC5oYXMoZGF5KSkpIHtcbiAgICAgIGRheXNUb0FkZFNldC5mb3JFYWNoKChkYXk6IHN0cmluZykgPT4gc2VsZWN0ZWRTZXQuZGVsZXRlKGRheSkpO1xuICAgIH0gZWxzZSB7XG4gICAgICBkYXlzVG9BZGRTZXQuZm9yRWFjaCgoZGF5OiBzdHJpbmcpID0+IHNlbGVjdGVkU2V0LmFkZChkYXkpKTtcbiAgICB9XG4gICAgcmV0dXJuIEFycmF5LmZyb20oc2VsZWN0ZWRTZXQpO1xuICB9XG5cbiAgcHJpdmF0ZSBfc2V0Q2hvc2VuWWVhcigpOiB2b2lkIHtcbiAgICBpZiAodGhpcy52aWV3ID09PSAnbW9udGgnKSB7XG4gICAgICBsZXQgc2VsZWN0ZWREYXRlOiBUIHwgdW5kZWZpbmVkO1xuICAgICAgaWYgKHRoaXMubXVsdGlwbGUpIHtcbiAgICAgICAgc2VsZWN0ZWREYXRlID0gKHRoaXMuc2VsZWN0ZWQgYXMgVFtdKS5hdCgtMSk7XG4gICAgICB9IGVsc2Uge1xuICAgICAgICBzZWxlY3RlZERhdGUgPSB0aGlzLnNlbGVjdGVkIGFzIFQ7XG4gICAgICB9XG4gICAgICB0aGlzLl9jaG9zZW5ZZWFyID0gdGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0WWVhcihzZWxlY3RlZERhdGUgPz8gdGhpcy5fZGF0ZUFkYXB0ZXIudG9kYXkoKSk7XG4gICAgfSBlbHNlIHtcbiAgICAgIHRoaXMuX2Nob3NlblllYXIgPSB1bmRlZmluZWQ7XG4gICAgfVxuICB9XG5cbiAgcHJpdmF0ZSBfYXNzaWduQWN0aXZlRGF0ZShkYXRlOiBUKTogdm9pZCB7XG4gICAgaWYgKHRoaXMubWluICYmIHRoaXMuX2RhdGVBZGFwdGVyLmNvbXBhcmVEYXRlKHRoaXMubWluLCBkYXRlKSA+IDApIHtcbiAgICAgIHRoaXMuX2FjdGl2ZURhdGUgPSB0aGlzLm1pbjtcbiAgICAgIHJldHVybjtcbiAgICB9XG4gICAgaWYgKHRoaXMubWF4ICYmIHRoaXMuX2RhdGVBZGFwdGVyLmNvbXBhcmVEYXRlKHRoaXMubWF4LCBkYXRlKSA8IDApIHtcbiAgICAgIHRoaXMuX2FjdGl2ZURhdGUgPSB0aGlzLm1heDtcbiAgICAgIHJldHVybjtcbiAgICB9XG4gICAgdGhpcy5fYWN0aXZlRGF0ZSA9IGRhdGU7XG4gIH1cblxuICAvKiogR29lcyB0byB0aGUgbW9udGggaWRlbnRpZmllZCBieSB0aGUgc2hpZnQuICovXG4gIHByaXZhdGUgX2dvVG9EaWZmZXJlbnRNb250aChtb250aHM6IG51bWJlcik6IHZvaWQge1xuICAgIHRoaXMuX2luaXQodGhpcy5fZGF0ZUFkYXB0ZXIuYWRkQ2FsZW5kYXJNb250aHModGhpcy5fYWN0aXZlRGF0ZSwgbW9udGhzKSk7XG4gICAgdGhpcy5fZW1pdE1vbnRoQ2hhbmdlKCk7XG4gIH1cblxuICBwcml2YXRlIF9nb1RvRGlmZmVyZW50WWVhcih5ZWFyczogbnVtYmVyKTogdm9pZCB7XG4gICAgdGhpcy5fY2hvc2VuWWVhciEgKz0geWVhcnM7XG4gICAgLy8gQ2FuJ3QgdXNlIGBfYXNzaWduQWN0aXZlRGF0ZSguLi4pYCBoZXJlLCBiZWNhdXNlIGl0IHdpbGwgc2V0IGl0IHRvIG1pbi9tYXggdmFsdWUgaWYgYXJndW1lbnQgaXMgb3V0IG9mIHJhbmdlXG4gICAgdGhpcy5fYWN0aXZlRGF0ZSA9IHRoaXMuX2RhdGVBZGFwdGVyLmNyZWF0ZURhdGUoXG4gICAgICB0aGlzLl9jaG9zZW5ZZWFyISxcbiAgICAgIHRoaXMuX2RhdGVBZGFwdGVyLmdldE1vbnRoKHRoaXMuX2FjdGl2ZURhdGUpLFxuICAgICAgdGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0RGF0ZSh0aGlzLl9hY3RpdmVEYXRlKSxcbiAgICApO1xuICAgIHRoaXMuX2luaXQoKTtcbiAgfVxuXG4gIHByaXZhdGUgX2dvVG9EaWZmZXJlbnRZZWFyUmFuZ2UoeWVhcnM6IG51bWJlcik6IHZvaWQge1xuICAgIHRoaXMuX2luaXQodGhpcy5fZGF0ZUFkYXB0ZXIuYWRkQ2FsZW5kYXJZZWFycyh0aGlzLl9hY3RpdmVEYXRlLCB5ZWFycykpO1xuICB9XG5cbiAgcHJpdmF0ZSBfcHJldkRpc2FibGVkKHByZXZEYXRlOiBUKTogYm9vbGVhbiB7XG4gICAgaWYgKCF0aGlzLm1pbikge1xuICAgICAgcmV0dXJuIGZhbHNlO1xuICAgIH1cbiAgICByZXR1cm4gdGhpcy5fZGF0ZUFkYXB0ZXIuY29tcGFyZURhdGUocHJldkRhdGUsIHRoaXMubWluKSA8IDA7XG4gIH1cblxuICBwcml2YXRlIF9uZXh0RGlzYWJsZWQobmV4dERhdGU6IFQpOiBib29sZWFuIHtcbiAgICBpZiAoIXRoaXMubWF4KSB7XG4gICAgICByZXR1cm4gZmFsc2U7XG4gICAgfVxuICAgIHJldHVybiB0aGlzLl9kYXRlQWRhcHRlci5jb21wYXJlRGF0ZShuZXh0RGF0ZSwgdGhpcy5tYXgpID4gMDtcbiAgfVxuXG4gIC8qKiBDaGVja3MgaWYgdGhlIFwicHJldmlvdXMgbW9udGhcIiBidXR0b24gc2hvdWxkIGJlIGRpc2FibGVkLiAqL1xuICBwcml2YXRlIF9wcmV2aW91c01vbnRoRGlzYWJsZWQoKTogYm9vbGVhbiB7XG4gICAgY29uc3QgcHJldk1vbnRoID0gdGhpcy5fZGF0ZUFkYXB0ZXIuYWRkQ2FsZW5kYXJEYXlzKFxuICAgICAgdGhpcy5fYWN0aXZlRGF0ZSxcbiAgICAgIHRoaXMuX2RhdGVBZGFwdGVyLmdldERhdGUodGhpcy5fYWN0aXZlRGF0ZSkgKiAtMSxcbiAgICApO1xuICAgIHJldHVybiB0aGlzLl9wcmV2RGlzYWJsZWQocHJldk1vbnRoKTtcbiAgfVxuXG4gIC8qKiBDaGVja3MgaWYgdGhlIFwibmV4dCBtb250aFwiIGJ1dHRvbiBzaG91bGQgYmUgZGlzYWJsZWQuICovXG4gIHByaXZhdGUgX25leHRNb250aERpc2FibGVkKCk6IGJvb2xlYW4ge1xuICAgIGxldCBuZXh0TW9udGggPSB0aGlzLl9kYXRlQWRhcHRlci5hZGRDYWxlbmRhck1vbnRocyh0aGlzLl9hY3RpdmVEYXRlLCB0aGlzLl93aWRlID8gMiA6IDEpO1xuICAgIG5leHRNb250aCA9IHRoaXMuX2RhdGVBZGFwdGVyLmNyZWF0ZURhdGUoXG4gICAgICB0aGlzLl9kYXRlQWRhcHRlci5nZXRZZWFyKG5leHRNb250aCksXG4gICAgICB0aGlzLl9kYXRlQWRhcHRlci5nZXRNb250aChuZXh0TW9udGgpLFxuICAgICAgMSxcbiAgICApO1xuICAgIHJldHVybiB0aGlzLl9uZXh0RGlzYWJsZWQobmV4dE1vbnRoKTtcbiAgfVxuXG4gIC8qKiBDaGVja3MgaWYgdGhlIFwicHJldmlvdXMgeWVhclwiIGJ1dHRvbiBzaG91bGQgYmUgZGlzYWJsZWQuICovXG4gIHByaXZhdGUgX3ByZXZpb3VzWWVhckRpc2FibGVkKCk6IGJvb2xlYW4ge1xuICAgIGNvbnN0IHByZXZZZWFyID0gdGhpcy5fZGF0ZUFkYXB0ZXIuY3JlYXRlRGF0ZShcbiAgICAgIHRoaXMuX2RhdGVBZGFwdGVyLmdldFllYXIodGhpcy5fYWN0aXZlRGF0ZSkgLSAxLFxuICAgICAgMTIsXG4gICAgICAzMSxcbiAgICApO1xuICAgIHJldHVybiB0aGlzLl9wcmV2RGlzYWJsZWQocHJldlllYXIpO1xuICB9XG5cbiAgLyoqIENoZWNrcyBpZiB0aGUgXCJuZXh0IHllYXJcIiBidXR0b24gc2hvdWxkIGJlIGRpc2FibGVkLiAqL1xuICBwcml2YXRlIF9uZXh0WWVhckRpc2FibGVkKCk6IGJvb2xlYW4ge1xuICAgIGNvbnN0IG5leHRZZWFyID0gdGhpcy5fZGF0ZUFkYXB0ZXIuY3JlYXRlRGF0ZShcbiAgICAgIHRoaXMuX2RhdGVBZGFwdGVyLmdldFllYXIodGhpcy5fYWN0aXZlRGF0ZSkgKyAodGhpcy5fd2lkZSA/IDIgOiAxKSxcbiAgICAgIDEsXG4gICAgICAxLFxuICAgICk7XG4gICAgcmV0dXJuIHRoaXMuX25leHREaXNhYmxlZChuZXh0WWVhcik7XG4gIH1cblxuICAvKiogQ2hlY2tzIGlmIHRoZSBcInByZXZpb3VzIHllYXJcIiBidXR0b24gc2hvdWxkIGJlIGRpc2FibGVkIGluIHllYXIgdmlldy4gKi9cbiAgcHJpdmF0ZSBfcHJldmlvdXNZZWFyUmFuZ2VEaXNhYmxlZCgpOiBib29sZWFuIHtcbiAgICBjb25zdCBwcmV2WWVhciA9IHRoaXMuX2RhdGVBZGFwdGVyLmNyZWF0ZURhdGUodGhpcy5feWVhcnNbMF1bMF0gLSAxLCAxMiwgMzEpO1xuICAgIHJldHVybiB0aGlzLl9wcmV2RGlzYWJsZWQocHJldlllYXIpO1xuICB9XG5cbiAgLyoqIENoZWNrcyBpZiB0aGUgXCJuZXh0IHllYXJcIiBidXR0b24gc2hvdWxkIGJlIGRpc2FibGVkIGluIHllYXIgdmlldy4gKi9cbiAgcHJpdmF0ZSBfbmV4dFllYXJSYW5nZURpc2FibGVkKCk6IGJvb2xlYW4ge1xuICAgIGNvbnN0IHllYXJzID0gdGhpcy5fd2lkZSA/IHRoaXMuX25leHRNb250aFllYXJzIDogdGhpcy5feWVhcnM7XG4gICAgY29uc3QgbGFzdFllYXJSYW5nZSA9IHllYXJzW3llYXJzLmxlbmd0aCAtIDFdO1xuICAgIGNvbnN0IGxhc3RZZWFyID0gbGFzdFllYXJSYW5nZVtsYXN0WWVhclJhbmdlLmxlbmd0aCAtIDFdO1xuICAgIGNvbnN0IG5leHRZZWFyID0gdGhpcy5fZGF0ZUFkYXB0ZXIuY3JlYXRlRGF0ZShsYXN0WWVhciArIDEsIDEsIDEpO1xuICAgIHJldHVybiB0aGlzLl9uZXh0RGlzYWJsZWQobmV4dFllYXIpO1xuICB9XG5cbiAgcHJpdmF0ZSBfaGFuZGxlVGFibGVCbHVyKGV2ZW50VGFyZ2V0OiBIVE1MRWxlbWVudCk6IHZvaWQge1xuICAgIGlmIChldmVudFRhcmdldD8ubG9jYWxOYW1lICE9PSAnc2JiLWNhbGVuZGFyLWRheScpIHtcbiAgICAgIHRoaXMuX3NldFRhYkluZGV4KCk7XG4gICAgfVxuICB9XG5cbiAgcHJpdmF0ZSBfc2V0VGFiSW5kZXgoKTogdm9pZCB7XG4gICAgQXJyYXkuZnJvbSh0aGlzLl9jZWxscy5maWx0ZXIoKGUpID0+IGUudGFiSW5kZXggPT09IDApID8/IFtdKS5mb3JFYWNoKFxuICAgICAgKGRheSkgPT4gKGRheS50YWJJbmRleCA9IC0xKSxcbiAgICApO1xuICAgIGNvbnN0IGZpcnN0Rm9jdXNhYmxlID0gdGhpcy5fZ2V0Rmlyc3RGb2N1c2FibGUoKTtcbiAgICBpZiAoZmlyc3RGb2N1c2FibGUpIHtcbiAgICAgIGZpcnN0Rm9jdXNhYmxlLnRhYkluZGV4ID0gMDtcbiAgICB9XG4gIH1cblxuICAvKiogR2V0IHRoZSBlbGVtZW50IGluIHRoZSBjYWxlbmRhciB0byBhc3NpZ24gZm9jdXMuICovXG4gIHByaXZhdGUgX2dldEZpcnN0Rm9jdXNhYmxlKCk6IFNiYkNhbGVuZGFyQ2VsbEJhc2VFbGVtZW50IHwgbnVsbCB7XG4gICAgaWYgKHRoaXMuX2NhbGVuZGFyVmlldyA9PT0gJ2RheScpIHtcbiAgICAgIGNvbnN0IHNlbGVjdGVkT3JDdXJyZW50ID1cbiAgICAgICAgdGhpcy5fY2VsbHMuZmluZCgoZSkgPT4gZS5tYXRjaGVzKCc6c3RhdGUoc2VsZWN0ZWQpJykpID8/XG4gICAgICAgIHRoaXMuX2NlbGxzLmZpbmQoKGUpID0+IGUubWF0Y2hlcygnOnN0YXRlKGN1cnJlbnQpJykpO1xuICAgICAgcmV0dXJuIHNlbGVjdGVkT3JDdXJyZW50ICYmICFzZWxlY3RlZE9yQ3VycmVudC5kaXNhYmxlZFxuICAgICAgICA/IHNlbGVjdGVkT3JDdXJyZW50XG4gICAgICAgIDogdGhpcy5fZ2V0Rmlyc3RGb2N1c2FibGVEYXkoKTtcbiAgICB9IGVsc2Uge1xuICAgICAgY29uc3Qgc2VsZWN0ZWRPckN1cnJlbnQgPVxuICAgICAgICB0aGlzLnNoYWRvd1Jvb3Q/LnF1ZXJ5U2VsZWN0b3I8U2JiQ2FsZW5kYXJDZWxsQmFzZUVsZW1lbnQ+KCc6c3RhdGUoc2VsZWN0ZWQpJykgPz9cbiAgICAgICAgdGhpcy5zaGFkb3dSb290Py5xdWVyeVNlbGVjdG9yPFNiYkNhbGVuZGFyQ2VsbEJhc2VFbGVtZW50PignOnN0YXRlKGN1cnJlbnQpJyk7XG4gICAgICByZXR1cm4gc2VsZWN0ZWRPckN1cnJlbnQgJiYgIXNlbGVjdGVkT3JDdXJyZW50LmRpc2FibGVkXG4gICAgICAgID8gc2VsZWN0ZWRPckN1cnJlbnRcbiAgICAgICAgOiB0aGlzLnNoYWRvd1Jvb3QhLnF1ZXJ5U2VsZWN0b3I8U2JiQ2FsZW5kYXJDZWxsQmFzZUVsZW1lbnQ+KFxuICAgICAgICAgICAgYHNiYi1jYWxlbmRhci0ke3RoaXMuX2NhbGVuZGFyVmlld306bm90KFtkaXNhYmxlZF0pYCxcbiAgICAgICAgICApO1xuICAgIH1cbiAgfVxuXG4gIC8qKlxuICAgKiBJbiBgZGF5YCB2aWV3IGluIGB2ZXJ0aWNhbGAgb3JpZW50YXRpb24sXG4gICAqIGlmIHRoZSBmaXJzdCBvZiB0aGUgbW9udGggaXMgbm90IGEgTW9uZGF5LCBpdCBpcyBub3QgdGhlIGZpcnN0IHJlbmRlcmVkIGVsZW1lbnQgaW4gdGhlIHRhYmxlLFxuICAgKiBzbyBgdGhpcy5zaGFkb3dSb290IS5xdWVyeVNlbGVjdG9yKCdzYmItY2FsZW5kYXItZGF5Om5vdChbZGlzYWJsZWRdKScpYCB3aWxsIHJldHVybiBhIHdyb25nIHZhbHVlLlxuICAgKlxuICAgKiBUbyBzb2x2ZSB0aGlzLCB0aGUgZWxlbWVudCB3aXRoIHRoZSBsb3dlc3QgYHZhbHVlYCBpcyB0YWtlbiAoSVNPIFN0cmluZyBhcmUgb3JkZXJlZCkuXG4gICAqL1xuICBwcml2YXRlIF9nZXRGaXJzdEZvY3VzYWJsZURheSgpOiBTYmJDYWxlbmRhckRheUVsZW1lbnQgfCBudWxsIHtcbiAgICBjb25zdCBjZWxscyA9IHRoaXMuX2NlbGxzIGFzIFNiYkNhbGVuZGFyRGF5RWxlbWVudFtdO1xuICAgIGNvbnN0IGRheXNJblZpZXcgPSBjZWxscy5maWx0ZXIoKGUpID0+ICFlLmRpc2FibGVkKTtcbiAgICBpZiAoIWRheXNJblZpZXcgfHwgZGF5c0luVmlldy5sZW5ndGggPT09IDApIHtcbiAgICAgIHJldHVybiBudWxsO1xuICAgIH0gZWxzZSB7XG4gICAgICBjb25zdCBmaXJzdEVsZW1lbnQgPSBkYXlzSW5WaWV3XG4gICAgICAgIC5tYXAoKGUpOiBzdHJpbmcgPT4gdGhpcy5fZGF0ZUFkYXB0ZXIudG9Jc284NjAxKGUudmFsdWUhIGFzIFQpKVxuICAgICAgICAuc29ydCgpWzBdO1xuICAgICAgcmV0dXJuIGNlbGxzLmZpbmQoKGUpID0+IGUubWF0Y2hlcyhgW3Nsb3Q9XCIke2ZpcnN0RWxlbWVudH1cIl1gKSkhID8/IG51bGw7XG4gICAgfVxuICB9XG5cbiAgcHJpdmF0ZSBfaGFuZGxlS2V5Ym9hcmRFdmVudChldmVudDogS2V5Ym9hcmRFdmVudCwgZGF5PzogRGF5PFQ+KTogdm9pZCB7XG4gICAgaWYgKGlzQXJyb3dLZXlPclBhZ2VLZXlzUHJlc3NlZChldmVudCkpIHtcbiAgICAgIGV2ZW50LnByZXZlbnREZWZhdWx0KCk7XG4gICAgfVxuICAgIC8vIEdldHMgdGhlIGN1cnJlbnRseSByZW5kZXJlZCB0YWJsZSdzIGNlbGw7XG4gICAgLy8gdGhleSBjb3VsZCBiZSBkYXlzLCBtb250aHMgb3IgeWVhcnMgYmFzZWQgb24gdGhlIGN1cnJlbnQgc2VsZWN0aW9uIHZpZXcuXG4gICAgLy8gSWYgYHdpZGVgIGlzIHRydWUsIHllYXJzIGFyZSBkb3VibGVkIGluIG51bWJlciBhbmQgZGF5cyBhcmUgKHJvdWdobHkpIGRvdWJsZWQgdG9vLCBhZmZlY3RpbmcgdGhlIGBpbmRleGAgY2FsY3VsYXRpb24uXG4gICAgY29uc3QgY2VsbHMgPSB0aGlzLl9jZWxscztcbiAgICBjb25zdCBpbmRleDogbnVtYmVyID0gY2VsbHMuZmluZEluZGV4KChlKSA9PiBlID09PSBldmVudC50YXJnZXQpO1xuICAgIGxldCBuZXh0RWw6IFNiYkNhbGVuZGFyQ2VsbEJhc2VFbGVtZW50O1xuICAgIGlmIChkYXkpIHtcbiAgICAgIG5leHRFbCA9IHRoaXMuX25hdmlnYXRlQnlLZXlib2FyZERheVZpZXcoZXZlbnQsIGluZGV4LCBjZWxscyBhcyBTYmJDYWxlbmRhckRheUVsZW1lbnRbXSwgZGF5KTtcbiAgICB9IGVsc2Uge1xuICAgICAgbmV4dEVsID0gdGhpcy5fbmF2aWdhdGVCeUtleWJvYXJkKGV2ZW50LCBpbmRleCwgY2VsbHMpO1xuICAgIH1cbiAgICBjb25zdCBhY3RpdmVFbCA9ICh0aGlzLl9lbmhhbmNlZFZhcmlhbnQgPyBkb2N1bWVudCA6IHRoaXMuc2hhZG93Um9vdCEpXG4gICAgICAuYWN0aXZlRWxlbWVudCBhcyBTYmJDYWxlbmRhckNlbGxCYXNlRWxlbWVudDtcbiAgICBpZiAobmV4dEVsICE9PSBhY3RpdmVFbCkge1xuICAgICAgbmV4dEVsLnRhYkluZGV4ID0gMDtcbiAgICAgIG5leHRFbD8uZm9jdXMoKTtcbiAgICAgIGFjdGl2ZUVsLnRhYkluZGV4ID0gLTE7XG4gICAgfVxuICB9XG5cbiAgcHJpdmF0ZSBfbmF2aWdhdGVCeUtleWJvYXJkRGF5VmlldyhcbiAgICBldnQ6IEtleWJvYXJkRXZlbnQsXG4gICAgaW5kZXg6IG51bWJlcixcbiAgICBjZWxsczogU2JiQ2FsZW5kYXJEYXlFbGVtZW50W10sXG4gICAgZGF5OiBEYXk8VD4sXG4gICk6IFNiYkNhbGVuZGFyRGF5RWxlbWVudCB7XG4gICAgY29uc3QgYXJyb3dzT2Zmc2V0ID1cbiAgICAgIHRoaXMub3JpZW50YXRpb24gPT09ICdob3Jpem9udGFsJ1xuICAgICAgICA/IHsgbGVmdFJpZ2h0OiAxLCB1cERvd246IERBWVNfUEVSX1JPVyB9XG4gICAgICAgIDogeyBsZWZ0UmlnaHQ6IERBWVNfUEVSX1JPVywgdXBEb3duOiAxIH07XG4gICAgY29uc3Qgb2Zmc2V0Rm9yVmVydGljYWw6IG51bWJlciA9XG4gICAgICBpbmRleCA8IHRoaXMuX2tleWJvYXJkTmF2aWdhdGlvbkRheVZpZXdQYXJhbWV0ZXJzLmZpcnN0TW9udGhMZW5ndGhcbiAgICAgICAgPyB0aGlzLl9rZXlib2FyZE5hdmlnYXRpb25EYXlWaWV3UGFyYW1ldGVycy5maXJzdE1vbnRoT2Zmc2V0XG4gICAgICAgIDogdGhpcy5fa2V5Ym9hcmROYXZpZ2F0aW9uRGF5Vmlld1BhcmFtZXRlcnMuc2Vjb25kTW9udGhPZmZzZXQ7XG5cbiAgICBzd2l0Y2ggKGV2dC5rZXkpIHtcbiAgICAgIGNhc2UgJ0Fycm93VXAnOlxuICAgICAgICByZXR1cm4gdGhpcy5fZmluZERheUFycm93cyhjZWxscywgaW5kZXgsIGRheS5kYXRlVmFsdWUsIC1hcnJvd3NPZmZzZXQudXBEb3duKTtcbiAgICAgIGNhc2UgJ0Fycm93RG93bic6XG4gICAgICAgIHJldHVybiB0aGlzLl9maW5kRGF5QXJyb3dzKGNlbGxzLCBpbmRleCwgZGF5LmRhdGVWYWx1ZSwgYXJyb3dzT2Zmc2V0LnVwRG93bik7XG4gICAgICBjYXNlICdBcnJvd0xlZnQnOlxuICAgICAgICByZXR1cm4gdGhpcy5fZmluZERheUFycm93cyhjZWxscywgaW5kZXgsIGRheS5kYXRlVmFsdWUsIC1hcnJvd3NPZmZzZXQubGVmdFJpZ2h0KTtcbiAgICAgIGNhc2UgJ0Fycm93UmlnaHQnOlxuICAgICAgICByZXR1cm4gdGhpcy5fZmluZERheUFycm93cyhjZWxscywgaW5kZXgsIGRheS5kYXRlVmFsdWUsIGFycm93c09mZnNldC5sZWZ0UmlnaHQpO1xuICAgICAgY2FzZSAnUGFnZVVwJzoge1xuICAgICAgICBpZiAodGhpcy5vcmllbnRhdGlvbiA9PT0gJ2hvcml6b250YWwnKSB7XG4gICAgICAgICAgY29uc3QgZmlyc3RPZldlZWs6IG51bWJlciA9ICtkYXkuZGF5VmFsdWUgJSBEQVlTX1BFUl9ST1cgfHwgREFZU19QRVJfUk9XO1xuICAgICAgICAgIGNvbnN0IGRlbHRhOiBudW1iZXIgPSBmaXJzdE9mV2VlayAtICtkYXkuZGF5VmFsdWU7XG4gICAgICAgICAgcmV0dXJuIHRoaXMuX2ZpbmREYXlQYWdlVXBEb3duKGNlbGxzLCBpbmRleCwgZGF5LCBkZWx0YSwgYXJyb3dzT2Zmc2V0LnVwRG93bik7XG4gICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgY29uc3Qgd2Vla051bWJlcjogbnVtYmVyID0gTWF0aC5jZWlsKCgrZGF5LmRheVZhbHVlICsgb2Zmc2V0Rm9yVmVydGljYWwpIC8gREFZU19QRVJfUk9XKTtcbiAgICAgICAgICBjb25zdCBmaXJzdE9mV2VlazogbnVtYmVyID0gKHdlZWtOdW1iZXIgLSAxKSAqIERBWVNfUEVSX1JPVyAtIG9mZnNldEZvclZlcnRpY2FsICsgMTtcbiAgICAgICAgICBjb25zdCBkZWx0YTogbnVtYmVyID0gZmlyc3RPZldlZWsgLSArZGF5LmRheVZhbHVlO1xuICAgICAgICAgIHJldHVybiB0aGlzLl9maW5kRGF5UGFnZVVwRG93bihjZWxscywgaW5kZXgsIGRheSwgZGVsdGEsIGFycm93c09mZnNldC51cERvd24pO1xuICAgICAgICB9XG4gICAgICB9XG4gICAgICBjYXNlICdQYWdlRG93bic6IHtcbiAgICAgICAgaWYgKHRoaXMub3JpZW50YXRpb24gPT09ICdob3Jpem9udGFsJykge1xuICAgICAgICAgIGNvbnN0IG1vbnRoSW5Cb3VuZHMgPSArZGF5Lm1vbnRoVmFsdWUgKyAxID4gMTIgPyAxIDogK2RheS5tb250aFZhbHVlICsgMTtcbiAgICAgICAgICBjb25zdCB5ZWFySW5Cb3VuZHMgPSArZGF5Lm1vbnRoVmFsdWUgKyAxID4gMTIgPyArZGF5LnllYXJWYWx1ZSArIDEgOiArZGF5LnllYXJWYWx1ZTtcbiAgICAgICAgICBjb25zdCBmaXJzdE5leHRNb250aDogVCA9IHRoaXMuX2RhdGVBZGFwdGVyLmNyZWF0ZURhdGUoeWVhckluQm91bmRzLCBtb250aEluQm91bmRzLCAxKTtcbiAgICAgICAgICBjb25zdCBsYXN0T2ZNb250aDogbnVtYmVyID0gdGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0RGF0ZShcbiAgICAgICAgICAgIHRoaXMuX2RhdGVBZGFwdGVyLmFkZENhbGVuZGFyRGF5cyhmaXJzdE5leHRNb250aCwgLTEpLFxuICAgICAgICAgICk7XG4gICAgICAgICAgY29uc3QgZGVsdGE6IG51bWJlciA9XG4gICAgICAgICAgICBNYXRoLnRydW5jKChsYXN0T2ZNb250aCAtICtkYXkuZGF5VmFsdWUhKSAvIERBWVNfUEVSX1JPVykgKiBEQVlTX1BFUl9ST1c7XG4gICAgICAgICAgcmV0dXJuIHRoaXMuX2ZpbmREYXlQYWdlVXBEb3duKGNlbGxzLCBpbmRleCwgZGF5LCBkZWx0YSwgLWFycm93c09mZnNldC51cERvd24pO1xuICAgICAgICB9IGVsc2Uge1xuICAgICAgICAgIGNvbnN0IHdlZWtOdW1iZXI6IG51bWJlciA9IE1hdGguY2VpbCgoK2RheS5kYXlWYWx1ZSArIG9mZnNldEZvclZlcnRpY2FsKSAvIERBWVNfUEVSX1JPVyk7XG4gICAgICAgICAgY29uc3QgbGFzdE9mV2VlazogbnVtYmVyID0gd2Vla051bWJlciAqIERBWVNfUEVSX1JPVyAtIG9mZnNldEZvclZlcnRpY2FsO1xuICAgICAgICAgIGNvbnN0IGRlbHRhOiBudW1iZXIgPSBsYXN0T2ZXZWVrIC0gK2RheS5kYXlWYWx1ZTtcbiAgICAgICAgICByZXR1cm4gdGhpcy5fZmluZERheVBhZ2VVcERvd24oY2VsbHMsIGluZGV4LCBkYXksIGRlbHRhLCAtYXJyb3dzT2Zmc2V0LnVwRG93bik7XG4gICAgICAgIH1cbiAgICAgIH1cbiAgICAgIGNhc2UgJ0hvbWUnOiB7XG4gICAgICAgIHJldHVybiB0aGlzLl9maW5kRGF5Rmlyc3QoY2VsbHMsIGluZGV4LCBkYXksIDEpO1xuICAgICAgfVxuICAgICAgY2FzZSAnRW5kJzoge1xuICAgICAgICBjb25zdCBtb250aEluQm91bmRzID0gK2RheS5tb250aFZhbHVlICsgMSA+IDEyID8gMSA6ICtkYXkubW9udGhWYWx1ZSArIDE7XG4gICAgICAgIGNvbnN0IHllYXJJbkJvdW5kcyA9ICtkYXkubW9udGhWYWx1ZSArIDEgPiAxMiA/ICtkYXkueWVhclZhbHVlICsgMSA6ICtkYXkueWVhclZhbHVlO1xuICAgICAgICBjb25zdCBmaXJzdE5leHRNb250aDogVCA9IHRoaXMuX2RhdGVBZGFwdGVyLmNyZWF0ZURhdGUoeWVhckluQm91bmRzLCBtb250aEluQm91bmRzLCAxKTtcbiAgICAgICAgcmV0dXJuIHRoaXMuX2ZpbmREYXlMYXN0KGNlbGxzLCBpbmRleCwgZmlyc3ROZXh0TW9udGgpO1xuICAgICAgfVxuICAgICAgZGVmYXVsdDpcbiAgICAgICAgcmV0dXJuIGNlbGxzW2luZGV4XTtcbiAgICB9XG4gIH1cblxuICBwcml2YXRlIF9pc0RheU91dE9mVmlldyhkYXRlOiBzdHJpbmcpOiBib29sZWFuIHtcbiAgICByZXR1cm4gKFxuICAgICAgZGF0ZSA8IHRoaXMuX2tleWJvYXJkTmF2aWdhdGlvbkRheVZpZXdQYXJhbWV0ZXJzLmZpcnN0RGF5SW5WaWV3ISB8fFxuICAgICAgZGF0ZSA+IHRoaXMuX2tleWJvYXJkTmF2aWdhdGlvbkRheVZpZXdQYXJhbWV0ZXJzLmxhc3REYXlJblZpZXchXG4gICAgKTtcbiAgfVxuXG4gIHByaXZhdGUgX2ZpbmREYXlBcnJvd3MoXG4gICAgY2VsbHM6IFNiYkNhbGVuZGFyRGF5RWxlbWVudFtdLFxuICAgIGluZGV4OiBudW1iZXIsXG4gICAgZGF0ZTogVCxcbiAgICBkZWx0YTogbnVtYmVyLFxuICApOiBTYmJDYWxlbmRhckRheUVsZW1lbnQge1xuICAgIGNvbnN0IG5ld0RhdGVWYWx1ZSA9IHRoaXMuX2RhdGVBZGFwdGVyLnRvSXNvODYwMShcbiAgICAgIHRoaXMuX2RhdGVBZGFwdGVyLmFkZENhbGVuZGFyRGF5cyhkYXRlLCBkZWx0YSksXG4gICAgKTtcbiAgICBpZiAodGhpcy5faXNEYXlPdXRPZlZpZXcobmV3RGF0ZVZhbHVlKSkge1xuICAgICAgcmV0dXJuIGNlbGxzW2luZGV4XTtcbiAgICB9XG4gICAgY29uc3QgbmV4dENlbGwgPSBjZWxscy5maW5kKChlKSA9PiB0aGlzLl9tYXBWYWx1ZVRvSVNPRGF0ZShlLnZhbHVlISkgPT09IG5ld0RhdGVWYWx1ZSk7XG4gICAgaWYgKCFuZXh0Q2VsbCB8fCBuZXh0Q2VsbC5kaXNhYmxlZCkge1xuICAgICAgcmV0dXJuIHRoaXMuX2ZpbmREYXlBcnJvd3MoY2VsbHMsIGluZGV4LCB0aGlzLl9kYXRlQWRhcHRlci5kZXNlcmlhbGl6ZShuZXdEYXRlVmFsdWUpISwgZGVsdGEpO1xuICAgIH1cbiAgICByZXR1cm4gbmV4dENlbGw7XG4gIH1cblxuICBwcml2YXRlIF9maW5kRGF5UGFnZVVwRG93bihcbiAgICBjZWxsczogU2JiQ2FsZW5kYXJEYXlFbGVtZW50W10sXG4gICAgaW5kZXg6IG51bWJlcixcbiAgICBkYXk6IERheTxUPixcbiAgICBkZWx0YTogbnVtYmVyLFxuICAgIGRlbHRhSWZEaXNhYmxlZDogbnVtYmVyLFxuICApOiBTYmJDYWxlbmRhckRheUVsZW1lbnQge1xuICAgIGNvbnN0IG5ld0RhdGVWYWx1ZSA9IHRoaXMuX2RhdGVBZGFwdGVyLnRvSXNvODYwMShcbiAgICAgIHRoaXMuX2RhdGVBZGFwdGVyLmFkZENhbGVuZGFyRGF5cyhkYXkuZGF0ZVZhbHVlLCBkZWx0YSksXG4gICAgKTtcbiAgICBpZiAodGhpcy5faXNEYXlPdXRPZlZpZXcobmV3RGF0ZVZhbHVlKSkge1xuICAgICAgcmV0dXJuIGNlbGxzW2luZGV4XTtcbiAgICB9XG4gICAgY29uc3QgbmV4dENlbGwgPSBjZWxscy5maW5kKChlKSA9PiB0aGlzLl9tYXBWYWx1ZVRvSVNPRGF0ZShlLnZhbHVlISkgPT09IG5ld0RhdGVWYWx1ZSk7XG4gICAgaWYgKCFuZXh0Q2VsbCB8fCBuZXh0Q2VsbC5kaXNhYmxlZCkge1xuICAgICAgcmV0dXJuIHRoaXMuX2ZpbmREYXlQYWdlVXBEb3duKGNlbGxzLCBpbmRleCwgZGF5LCBkZWx0YSArIGRlbHRhSWZEaXNhYmxlZCwgZGVsdGFJZkRpc2FibGVkKTtcbiAgICB9XG4gICAgcmV0dXJuIG5leHRDZWxsO1xuICB9XG5cbiAgcHJpdmF0ZSBfZmluZERheUZpcnN0KFxuICAgIGNlbGxzOiBTYmJDYWxlbmRhckRheUVsZW1lbnRbXSxcbiAgICBpbmRleDogbnVtYmVyLFxuICAgIGRheTogRGF5PFQ+LFxuICAgIGRhdGU6IG51bWJlcixcbiAgKTogU2JiQ2FsZW5kYXJEYXlFbGVtZW50IHtcbiAgICBjb25zdCBuZXdEYXRlVmFsdWUgPSB0aGlzLl9kYXRlQWRhcHRlci50b0lzbzg2MDEoXG4gICAgICB0aGlzLl9kYXRlQWRhcHRlci5jcmVhdGVEYXRlKCtkYXkueWVhclZhbHVlLCArZGF5Lm1vbnRoVmFsdWUsIGRhdGUpLFxuICAgICk7XG4gICAgaWYgKHRoaXMuX2lzRGF5T3V0T2ZWaWV3KG5ld0RhdGVWYWx1ZSkpIHtcbiAgICAgIHJldHVybiBjZWxsc1tpbmRleF07XG4gICAgfVxuICAgIGNvbnN0IG5leHRDZWxsID0gY2VsbHMuZmluZCgoZSkgPT4gdGhpcy5fbWFwVmFsdWVUb0lTT0RhdGUoZS52YWx1ZSEpID09PSBuZXdEYXRlVmFsdWUpO1xuICAgIGlmICghbmV4dENlbGwgfHwgbmV4dENlbGwuZGlzYWJsZWQpIHtcbiAgICAgIHJldHVybiB0aGlzLl9maW5kRGF5Rmlyc3QoY2VsbHMsIGluZGV4LCBkYXksIGRhdGUgKyAxKTtcbiAgICB9XG4gICAgcmV0dXJuIG5leHRDZWxsO1xuICB9XG5cbiAgcHJpdmF0ZSBfZmluZERheUxhc3QoXG4gICAgY2VsbHM6IFNiYkNhbGVuZGFyRGF5RWxlbWVudFtdLFxuICAgIGluZGV4OiBudW1iZXIsXG4gICAgZmlyc3ROZXh0TW9udGg6IFQsXG4gICk6IFNiYkNhbGVuZGFyRGF5RWxlbWVudCB7XG4gICAgY29uc3QgbmV3RGF0ZVZhbHVlID0gdGhpcy5fZGF0ZUFkYXB0ZXIudG9Jc284NjAxKFxuICAgICAgdGhpcy5fZGF0ZUFkYXB0ZXIuYWRkQ2FsZW5kYXJEYXlzKGZpcnN0TmV4dE1vbnRoLCAtMSksXG4gICAgKTtcbiAgICBpZiAodGhpcy5faXNEYXlPdXRPZlZpZXcobmV3RGF0ZVZhbHVlKSkge1xuICAgICAgcmV0dXJuIGNlbGxzW2luZGV4XTtcbiAgICB9XG4gICAgY29uc3QgbmV4dENlbGwgPSBjZWxscy5maW5kKChlKSA9PiB0aGlzLl9tYXBWYWx1ZVRvSVNPRGF0ZShlLnZhbHVlISkgPT09IG5ld0RhdGVWYWx1ZSk7XG4gICAgaWYgKCFuZXh0Q2VsbCB8fCBuZXh0Q2VsbC5kaXNhYmxlZCkge1xuICAgICAgcmV0dXJuIHRoaXMuX2ZpbmREYXlMYXN0KGNlbGxzLCBpbmRleCwgdGhpcy5fZGF0ZUFkYXB0ZXIuZGVzZXJpYWxpemUobmV3RGF0ZVZhbHVlKSEpO1xuICAgIH1cbiAgICByZXR1cm4gbmV4dENlbGw7XG4gIH1cblxuICAvKipcbiAgICogR2V0cyB0aGUgaW5kZXggb2YgdGhlIGVsZW1lbnQgdG8gbW92ZSB0bywgYmFzZWQgb24gYSBsaXN0IG9mIGVsZW1lbnRzICh3aGljaCBjYW4gYmUgcG90ZW50aWFsbHkgZGlzYWJsZWQpLFxuICAgKiB0aGUga2V5Ym9hcmQgaW5wdXQgYW5kIHRoZSBwb3NpdGlvbiBvZiB0aGUgY3VycmVudCBlbGVtZW50IGluIHRoZSBsaXN0LlxuICAgKiBJbiB0aGUgZGF5IHZpZXcsIHRoZSBgZGF5PzogRGF5YCBwYXJhbWV0ZXIgaXMgbWFuZGF0b3J5IGZvciBjYWxjdWxhdGlvbixcbiAgICogd2hpbGUgaW4gbW9udGggYW5kIHllYXIgdmlldyBpdCdzIG5vdCBkdWUgdG8gdGhlIGZpeGVkIGFtb3VudCBvZiByZW5kZXJlZCBjZWxscy5cbiAgICovXG4gIHByaXZhdGUgX25hdmlnYXRlQnlLZXlib2FyZChcbiAgICBldnQ6IEtleWJvYXJkRXZlbnQsXG4gICAgaW5kZXg6IG51bWJlcixcbiAgICBjZWxsczogU2JiQ2FsZW5kYXJDZWxsQmFzZUVsZW1lbnRbXSxcbiAgKTogU2JiQ2FsZW5kYXJDZWxsQmFzZUVsZW1lbnQge1xuICAgIGNvbnN0IHtcbiAgICAgIGVsZW1lbnRJbmRleEZvcldpZGVNb2RlLFxuICAgICAgb2Zmc2V0Rm9yV2lkZU1vZGUsXG4gICAgICBsYXN0RWxlbWVudEluZGV4Rm9yV2lkZU1vZGUsXG4gICAgICB2ZXJ0aWNhbE9mZnNldCxcbiAgICB9OiBDYWxlbmRhcktleWJvYXJkTmF2aWdhdGlvbk1vbnRoWWVhclZpZXdzUGFyYW1ldGVycyA9XG4gICAgICB0aGlzLl9jYWxjdWxhdGVQYXJhbWV0ZXJzRm9yS2V5Ym9hcmROYXZpZ2F0aW9uKGluZGV4LCB0aGlzLl9jYWxlbmRhclZpZXcgPT09ICd5ZWFyJyk7XG5cbiAgICBzd2l0Y2ggKGV2dC5rZXkpIHtcbiAgICAgIGNhc2UgJ0Fycm93VXAnOlxuICAgICAgICByZXR1cm4gdGhpcy5fZmluZE5leHQoY2VsbHMsIGluZGV4LCAtdmVydGljYWxPZmZzZXQpO1xuICAgICAgY2FzZSAnQXJyb3dEb3duJzpcbiAgICAgICAgcmV0dXJuIHRoaXMuX2ZpbmROZXh0KGNlbGxzLCBpbmRleCwgdmVydGljYWxPZmZzZXQpO1xuICAgICAgY2FzZSAnQXJyb3dMZWZ0JzpcbiAgICAgICAgcmV0dXJuIHRoaXMuX2ZpbmROZXh0KGNlbGxzLCBpbmRleCwgLTEpO1xuICAgICAgY2FzZSAnQXJyb3dSaWdodCc6XG4gICAgICAgIHJldHVybiB0aGlzLl9maW5kTmV4dChjZWxscywgaW5kZXgsIDEpO1xuICAgICAgY2FzZSAnSG9tZSc6XG4gICAgICAgIHJldHVybiB0aGlzLl9maW5kRmlyc3QoY2VsbHMsIG9mZnNldEZvcldpZGVNb2RlKTtcbiAgICAgIGNhc2UgJ1BhZ2VVcCc6XG4gICAgICAgIHJldHVybiB0aGlzLl9maW5kRmlyc3RPbkNvbHVtbihcbiAgICAgICAgICBjZWxscyxcbiAgICAgICAgICBlbGVtZW50SW5kZXhGb3JXaWRlTW9kZSxcbiAgICAgICAgICBvZmZzZXRGb3JXaWRlTW9kZSxcbiAgICAgICAgICB2ZXJ0aWNhbE9mZnNldCxcbiAgICAgICAgKTtcbiAgICAgIGNhc2UgJ1BhZ2VEb3duJzpcbiAgICAgICAgcmV0dXJuIHRoaXMuX2ZpbmRMYXN0T25Db2x1bW4oY2VsbHMsIGluZGV4LCBsYXN0RWxlbWVudEluZGV4Rm9yV2lkZU1vZGUsIHZlcnRpY2FsT2Zmc2V0KTtcbiAgICAgIGNhc2UgJ0VuZCc6XG4gICAgICAgIHJldHVybiB0aGlzLl9maW5kTGFzdChjZWxscywgbGFzdEVsZW1lbnRJbmRleEZvcldpZGVNb2RlIC0gMSk7XG4gICAgICBkZWZhdWx0OlxuICAgICAgICByZXR1cm4gY2VsbHNbaW5kZXhdO1xuICAgIH1cbiAgfVxuXG4gIC8qKlxuICAgKiBDYWxjdWxhdGVzIHRoZSBwYXJhbWV0ZXJzIG5lZWRlZCBpbiBrZXlib2FyZCBuYXZpZ2F0aW9uIGluIHllYXIgYW5kIG1vbnRoIHZpZXcuXG4gICAqIEBwYXJhbSBpbmRleCBUaGUgc3RhcnRpbmcgZWxlbWVudCdzIGluZGV4IGluIHRoZSBjZWxsIGFycmF5LlxuICAgKiBAcGFyYW0gaXNZZWFyVmlldyBXaGV0aGVyIHRoZSBkaXNwbGF5ZWQgYHZpZXdgIGlzIHRoZSB5ZWFyIG9uZS5cbiAgICovXG4gIHByaXZhdGUgX2NhbGN1bGF0ZVBhcmFtZXRlcnNGb3JLZXlib2FyZE5hdmlnYXRpb24oXG4gICAgaW5kZXg6IG51bWJlcixcbiAgICBpc1llYXJWaWV3OiBib29sZWFuLFxuICApOiBDYWxlbmRhcktleWJvYXJkTmF2aWdhdGlvbk1vbnRoWWVhclZpZXdzUGFyYW1ldGVycyB7XG4gICAgY29uc3QgZWxlbWVudHNQZXJQYWdlID0gaXNZZWFyVmlldyA/IFlFQVJTX1BFUl9QQUdFIDogTU9OVEhTX1BFUl9QQUdFO1xuICAgIGNvbnN0IG9mZnNldDogbnVtYmVyID0gTWF0aC50cnVuYyhpbmRleCAvIGVsZW1lbnRzUGVyUGFnZSkgKiBlbGVtZW50c1BlclBhZ2U7XG4gICAgY29uc3QgaW5kZXhJblZpZXc6IG51bWJlciA9IG9mZnNldCA9PT0gMCA/IGluZGV4IDogaW5kZXggLSBlbGVtZW50c1BlclBhZ2U7XG4gICAgcmV0dXJuIHtcbiAgICAgIHZlcnRpY2FsT2Zmc2V0OiBpc1llYXJWaWV3ID8gWUVBUlNfUEVSX1JPVyA6IE1PTlRIU19QRVJfUk9XLFxuICAgICAgZWxlbWVudEluZGV4Rm9yV2lkZU1vZGU6IGluZGV4SW5WaWV3LFxuICAgICAgb2Zmc2V0Rm9yV2lkZU1vZGU6IGluZGV4IC0gaW5kZXhJblZpZXcsXG4gICAgICBsYXN0RWxlbWVudEluZGV4Rm9yV2lkZU1vZGU6IG9mZnNldCA9PT0gMCA/IGVsZW1lbnRzUGVyUGFnZSA6IGVsZW1lbnRzUGVyUGFnZSAqIDIsXG4gICAgfTtcbiAgfVxuXG4gIC8qKlxuICAgKiBHZXRzIHRoZSBuZXh0IGVsZW1lbnQgb2YgdGhlIHByb3ZpZGVkIGFycmF5IHN0YXJ0aW5nIGZyb20gYGluZGV4YCBieSBhZGRpbmcgYGRlbHRhYC5cbiAgICogSWYgdGhlIGZvdW5kIGVsZW1lbnQgaXMgZGlzYWJsZWQsIGl0IGNvbnRpbnVlcyBhZGRpbmcgYGRlbHRhYCB1bnRpbCBpdCBmaW5kcyBhbiBlbmFibGVkIG9uZSBpbiB0aGUgYXJyYXkgYm91bmRzLlxuICAgKi9cbiAgcHJpdmF0ZSBfZmluZE5leHQoXG4gICAgZGF5czogU2JiQ2FsZW5kYXJDZWxsQmFzZUVsZW1lbnRbXSxcbiAgICBpbmRleDogbnVtYmVyLFxuICAgIGRlbHRhOiBudW1iZXIsXG4gICk6IFNiYkNhbGVuZGFyQ2VsbEJhc2VFbGVtZW50IHtcbiAgICBsZXQgbmV4dEluZGV4ID0gaW5kZXggKyBkZWx0YTtcbiAgICB3aGlsZSAobmV4dEluZGV4IDwgZGF5cy5sZW5ndGggJiYgZGF5c1tuZXh0SW5kZXhdPy5kaXNhYmxlZCkge1xuICAgICAgbmV4dEluZGV4ICs9IGRlbHRhO1xuICAgIH1cbiAgICByZXR1cm4gZGF5c1tuZXh0SW5kZXhdID8/IGRheXNbaW5kZXhdO1xuICB9XG5cbiAgLyoqIEZpbmQgdGhlIGZpcnN0IGVuYWJsZWQgZWxlbWVudCBpbiB0aGUgcHJvdmlkZWQgYXJyYXkuICovXG4gIHByaXZhdGUgX2ZpbmRGaXJzdChcbiAgICBkYXlzOiBTYmJDYWxlbmRhckNlbGxCYXNlRWxlbWVudFtdLFxuICAgIGZpcnN0T2ZDdXJyZW50TW9udGg6IG51bWJlcixcbiAgKTogU2JiQ2FsZW5kYXJDZWxsQmFzZUVsZW1lbnQge1xuICAgIHJldHVybiAhZGF5c1tmaXJzdE9mQ3VycmVudE1vbnRoXS5kaXNhYmxlZFxuICAgICAgPyBkYXlzW2ZpcnN0T2ZDdXJyZW50TW9udGhdXG4gICAgICA6IHRoaXMuX2ZpbmROZXh0KGRheXMsIGZpcnN0T2ZDdXJyZW50TW9udGgsIDEpO1xuICB9XG5cbiAgLyoqIEZpbmQgdGhlIGxhc3QgZW5hYmxlZCBlbGVtZW50IGluIHRoZSBwcm92aWRlZCBhcnJheS4gKi9cbiAgcHJpdmF0ZSBfZmluZExhc3QoXG4gICAgZGF5czogU2JiQ2FsZW5kYXJDZWxsQmFzZUVsZW1lbnRbXSxcbiAgICBsYXN0T2ZDdXJyZW50TW9udGg6IG51bWJlcixcbiAgKTogU2JiQ2FsZW5kYXJDZWxsQmFzZUVsZW1lbnQge1xuICAgIHJldHVybiAhZGF5c1tsYXN0T2ZDdXJyZW50TW9udGhdLmRpc2FibGVkXG4gICAgICA/IGRheXNbbGFzdE9mQ3VycmVudE1vbnRoXVxuICAgICAgOiB0aGlzLl9maW5kTmV4dChkYXlzLCBsYXN0T2ZDdXJyZW50TW9udGgsIC0xKTtcbiAgfVxuXG4gIC8qKiBGaW5kIHRoZSBmaXJzdCBlbmFibGVkIGVsZW1lbnQgaW4gdGhlIHNhbWUgY29sdW1uIG9mIHRoZSBwcm92aWRlZCBhcnJheS4gKi9cbiAgcHJpdmF0ZSBfZmluZEZpcnN0T25Db2x1bW4oXG4gICAgZGF5czogU2JiQ2FsZW5kYXJDZWxsQmFzZUVsZW1lbnRbXSxcbiAgICBpbmRleDogbnVtYmVyLFxuICAgIG9mZnNldDogbnVtYmVyLFxuICAgIHZlcnRpY2FsT2Zmc2V0OiBudW1iZXIsXG4gICk6IFNiYkNhbGVuZGFyQ2VsbEJhc2VFbGVtZW50IHtcbiAgICBjb25zdCBuZXh0SW5kZXggPSAoaW5kZXggJSB2ZXJ0aWNhbE9mZnNldCkgKyBvZmZzZXQ7XG4gICAgcmV0dXJuICFkYXlzW25leHRJbmRleF0uZGlzYWJsZWRcbiAgICAgID8gZGF5c1tuZXh0SW5kZXhdXG4gICAgICA6IHRoaXMuX2ZpbmROZXh0KGRheXMsIG5leHRJbmRleCwgdmVydGljYWxPZmZzZXQpO1xuICB9XG5cbiAgLyoqIEZpbmQgdGhlIGxhc3QgZW5hYmxlZCBlbGVtZW50IGluIHRoZSBzYW1lIGNvbHVtbiBvZiB0aGUgcHJvdmlkZWQgYXJyYXkuICovXG4gIHByaXZhdGUgX2ZpbmRMYXN0T25Db2x1bW4oXG4gICAgZGF5czogU2JiQ2FsZW5kYXJDZWxsQmFzZUVsZW1lbnRbXSxcbiAgICBpbmRleDogbnVtYmVyLFxuICAgIG9mZnNldDogbnVtYmVyLFxuICAgIHZlcnRpY2FsT2Zmc2V0OiBudW1iZXIsXG4gICk6IFNiYkNhbGVuZGFyQ2VsbEJhc2VFbGVtZW50IHtcbiAgICBjb25zdCBuZXh0SW5kZXggPSBpbmRleCArIE1hdGgudHJ1bmMoKG9mZnNldCAtIGluZGV4IC0gMSkgLyB2ZXJ0aWNhbE9mZnNldCkgKiB2ZXJ0aWNhbE9mZnNldDtcbiAgICByZXR1cm4gIWRheXNbbmV4dEluZGV4XS5kaXNhYmxlZFxuICAgICAgPyBkYXlzW25leHRJbmRleF1cbiAgICAgIDogdGhpcy5fZmluZE5leHQoZGF5cywgbmV4dEluZGV4LCAtdmVydGljYWxPZmZzZXQpO1xuICB9XG5cbiAgcHJpdmF0ZSBfcmVzZXRDYWxlbmRhclZpZXdBbmRFbWl0TW9udGhDaGFuZ2UoaW5pdFRyYW5zaXRpb24gPSBmYWxzZSk6IHZvaWQge1xuICAgIHRoaXMuX3Jlc2V0Q2FsZW5kYXJWaWV3KGluaXRUcmFuc2l0aW9uKTtcbiAgICB0aGlzLl9lbWl0TW9udGhDaGFuZ2UoKTtcbiAgfVxuXG4gIHByaXZhdGUgX3Jlc2V0Q2FsZW5kYXJWaWV3KGluaXRUcmFuc2l0aW9uID0gZmFsc2UpOiB2b2lkIHtcbiAgICBpZiAodGhpcy5fY29udGFpbmluZ0ZvY3VzKSB7XG4gICAgICB0aGlzLl9yZXNldEZvY3VzID0gdHJ1ZTtcbiAgICB9XG4gICAgdGhpcy5fYWN0aXZlRGF0ZSA9XG4gICAgICAodGhpcy5tdWx0aXBsZSA/ICh0aGlzLl9zZWxlY3RlZCBhcyBUW10pLmF0KC0xKSA6ICh0aGlzLl9zZWxlY3RlZCBhcyBUKSkgPz9cbiAgICAgIHRoaXMuX2RhdGVBZGFwdGVyLnRvZGF5KCk7XG4gICAgdGhpcy5fc2V0Q2hvc2VuWWVhcigpO1xuICAgIHRoaXMuX2Nob3Nlbk1vbnRoID0gdW5kZWZpbmVkO1xuICAgIHRoaXMuX2luaXQoKTtcbiAgICB0aGlzLl9uZXh0Q2FsZW5kYXJWaWV3ID0gdGhpcy5fY2FsZW5kYXJWaWV3ID0gdGhpcy52aWV3O1xuXG4gICAgaWYgKGluaXRUcmFuc2l0aW9uKSB7XG4gICAgICB0aGlzLl9zdGFydFRhYmxlVHJhbnNpdGlvbigpO1xuICAgIH1cbiAgfVxuXG4gIC8qKiBSZW5kZXIgdGhlIHZpZXcgZm9yIHRoZSBkYXkgc2VsZWN0aW9uLiAqL1xuICBwcml2YXRlIF9yZW5kZXJEYXlWaWV3KCk6IFRlbXBsYXRlUmVzdWx0IHtcbiAgICBjb25zdCBuZXh0TW9udGhBY3RpdmVEYXRlID0gdGhpcy5fd2lkZVxuICAgICAgPyB0aGlzLl9kYXRlQWRhcHRlci5hZGRDYWxlbmRhck1vbnRocyh0aGlzLl9hY3RpdmVEYXRlLCAxKVxuICAgICAgOiB1bmRlZmluZWQ7XG4gICAgcmV0dXJuIGh0bWxgXG4gICAgICA8ZGl2IGNsYXNzPVwic2JiLWNhbGVuZGFyX19jb250cm9sc1wiPlxuICAgICAgICAke3RoaXMuX2dldEFycm93KFxuICAgICAgICAgICdsZWZ0JyxcbiAgICAgICAgICAoKSA9PiB0aGlzLl9nb1RvRGlmZmVyZW50TW9udGgoLTEpLFxuICAgICAgICAgIGkxOG5QcmV2aW91c01vbnRoW3RoaXMuX2xhbmd1YWdlLmN1cnJlbnRdLFxuICAgICAgICAgIHRoaXMuX3ByZXZpb3VzTW9udGhEaXNhYmxlZCgpLFxuICAgICAgICApfVxuICAgICAgICA8ZGl2IGNsYXNzPVwic2JiLWNhbGVuZGFyX19jb250cm9scy1tb250aFwiPlxuICAgICAgICAgICR7dGhpcy5fY3JlYXRlTGFiZWxGb3JEYXlWaWV3KHRoaXMuX2FjdGl2ZURhdGUpfVxuICAgICAgICAgICR7dGhpcy5fd2lkZSA/IHRoaXMuX2NyZWF0ZUxhYmVsRm9yRGF5VmlldyhuZXh0TW9udGhBY3RpdmVEYXRlISkgOiBub3RoaW5nfVxuICAgICAgICAgIDxzYmItc2NyZWVuLXJlYWRlci1vbmx5IHJvbGU9XCJzdGF0dXNcIj5cbiAgICAgICAgICAgICR7dGhpcy5fY3JlYXRlQXJpYUxhYmVsRm9yRGF5Vmlldyh0aGlzLl9hY3RpdmVEYXRlLCBuZXh0TW9udGhBY3RpdmVEYXRlISl9XG4gICAgICAgICAgPC9zYmItc2NyZWVuLXJlYWRlci1vbmx5PlxuICAgICAgICA8L2Rpdj5cbiAgICAgICAgJHt0aGlzLl9nZXRBcnJvdyhcbiAgICAgICAgICAncmlnaHQnLFxuICAgICAgICAgICgpID0+IHRoaXMuX2dvVG9EaWZmZXJlbnRNb250aCgxKSxcbiAgICAgICAgICBpMThuTmV4dE1vbnRoW3RoaXMuX2xhbmd1YWdlLmN1cnJlbnRdLFxuICAgICAgICAgIHRoaXMuX25leHRNb250aERpc2FibGVkKCksXG4gICAgICAgICl9XG4gICAgICA8L2Rpdj5cbiAgICAgIDxkaXYgY2xhc3M9XCJzYmItY2FsZW5kYXJfX3RhYmxlLW92ZXJmbG93LWJyZWFrXCI+XG4gICAgICAgIDxkaXYgY2xhc3M9XCJzYmItY2FsZW5kYXJfX3RhYmxlLWNvbnRhaW5lciBzYmItY2FsZW5kYXJfX3RhYmxlLWRheS12aWV3XCI+XG4gICAgICAgICAgJHt0aGlzLm9yaWVudGF0aW9uID09PSAnaG9yaXpvbnRhbCdcbiAgICAgICAgICAgID8gaHRtbGBcbiAgICAgICAgICAgICAgICAke3RoaXMuX2NyZWF0ZURheVRhYmxlKHRoaXMuX3dlZWtzLCB0aGlzLl93ZWVrTnVtYmVycyl9XG4gICAgICAgICAgICAgICAgJHt0aGlzLl93aWRlXG4gICAgICAgICAgICAgICAgICA/IHRoaXMuX2NyZWF0ZURheVRhYmxlKHRoaXMuX25leHRNb250aFdlZWtzLCB0aGlzLl9uZXh0TW9udGhXZWVrTnVtYmVycywgdHJ1ZSlcbiAgICAgICAgICAgICAgICAgIDogbm90aGluZ31cbiAgICAgICAgICAgICAgYFxuICAgICAgICAgICAgOiBodG1sYFxuICAgICAgICAgICAgICAgICR7dGhpcy5fY3JlYXRlRGF5VGFibGVWZXJ0aWNhbCh0aGlzLl93ZWVrcywgdGhpcy5fd2Vla051bWJlcnMpfVxuICAgICAgICAgICAgICAgICR7dGhpcy5fd2lkZVxuICAgICAgICAgICAgICAgICAgPyB0aGlzLl9jcmVhdGVEYXlUYWJsZVZlcnRpY2FsKFxuICAgICAgICAgICAgICAgICAgICAgIHRoaXMuX25leHRNb250aFdlZWtzLFxuICAgICAgICAgICAgICAgICAgICAgIHRoaXMuX25leHRNb250aFdlZWtOdW1iZXJzLFxuICAgICAgICAgICAgICAgICAgICAgIG5leHRNb250aEFjdGl2ZURhdGUsXG4gICAgICAgICAgICAgICAgICAgIClcbiAgICAgICAgICAgICAgICAgIDogbm90aGluZ31cbiAgICAgICAgICAgICAgYH1cbiAgICAgICAgPC9kaXY+XG4gICAgICA8L2Rpdj5cbiAgICBgO1xuICB9XG5cbiAgLyoqIENyZWF0ZXMgdGhlIGxhYmVsIHdpdGggdGhlIG1vbnRoIGZvciB0aGUgZGFpbHkgdmlldy4gKi9cbiAgcHJpdmF0ZSBfY3JlYXRlTGFiZWxGb3JEYXlWaWV3KGQ6IFQpOiBUZW1wbGF0ZVJlc3VsdCB7XG4gICAgY29uc3QgbW9udGhMYWJlbCA9IGAke1xuICAgICAgdGhpcy5fbW9udGhOYW1lc1t0aGlzLl9kYXRlQWRhcHRlci5nZXRNb250aChkKSAtIDFdXG4gICAgfSAke3RoaXMuX2RhdGVBZGFwdGVyLmdldFllYXIoZCl9YDtcbiAgICByZXR1cm4gaHRtbGBcbiAgICAgIDxidXR0b25cbiAgICAgICAgdHlwZT1cImJ1dHRvblwiXG4gICAgICAgIGNsYXNzPVwic2JiLWNhbGVuZGFyX19kYXRlLXNlbGVjdGlvbiBzYmItY2FsZW5kYXJfX2NvbnRyb2xzLWNoYW5nZS1kYXRlXCJcbiAgICAgICAgYXJpYS1sYWJlbD1cIiR7aTE4blllYXJNb250aFNlbGVjdGlvblt0aGlzLl9sYW5ndWFnZS5jdXJyZW50XX0gJHttb250aExhYmVsfVwiXG4gICAgICAgIEBjbGljaz0keygpID0+IHtcbiAgICAgICAgICB0aGlzLl9yZXNldEZvY3VzID0gdHJ1ZTtcbiAgICAgICAgICB0aGlzLl9uZXh0Q2FsZW5kYXJWaWV3ID0gJ3llYXInO1xuICAgICAgICAgIHRoaXMuX3N0YXJ0VGFibGVUcmFuc2l0aW9uKCk7XG4gICAgICAgIH19XG4gICAgICA+XG4gICAgICAgICR7bW9udGhMYWJlbH1cbiAgICAgICAgPHNiYi1pY29uIG5hbWU9XCJjaGV2cm9uLXNtYWxsLWRvd24tc21hbGxcIj48L3NiYi1pY29uPlxuICAgICAgPC9idXR0b24+XG4gICAgYDtcbiAgfVxuXG4gIC8qKiBDcmVhdGVzIHRoZSBhcmlhLWxhYmVsIGZvciB0aGUgZGFpbHkgdmlldy4gKi9cbiAgcHJpdmF0ZSBfY3JlYXRlQXJpYUxhYmVsRm9yRGF5VmlldyguLi5kYXRlczogVFtdKTogc3RyaW5nIHtcbiAgICBsZXQgbW9udGhMYWJlbCA9ICcnO1xuICAgIGZvciAoY29uc3QgZCBvZiBkYXRlcykge1xuICAgICAgaWYgKGQpIHtcbiAgICAgICAgbW9udGhMYWJlbCArPSBgJHtcbiAgICAgICAgICB0aGlzLl9tb250aE5hbWVzW3RoaXMuX2RhdGVBZGFwdGVyLmdldE1vbnRoKGQpIC0gMV1cbiAgICAgICAgfSAke3RoaXMuX2RhdGVBZGFwdGVyLmdldFllYXIoZCl9IGA7XG4gICAgICB9XG4gICAgfVxuICAgIHJldHVybiBtb250aExhYmVsO1xuICB9XG5cbiAgLyoqIENyZWF0ZXMgdGhlIGNhbGVuZGFyIHRhYmxlIGZvciB0aGUgZGFpbHkgdmlldy4gKi9cbiAgcHJpdmF0ZSBfY3JlYXRlRGF5VGFibGUoXG4gICAgd2Vla3M6IERheTxUPltdW10sXG4gICAgd2Vla051bWJlcnM6IG51bWJlcltdLFxuICAgIGlzV2lkZU5leHRNb250aDogYm9vbGVhbiA9IGZhbHNlLFxuICApOiBUZW1wbGF0ZVJlc3VsdCB7XG4gICAgY29uc3Qgd2Vla3NGb3JTZWxlY3RNdWx0aXBsZVdlZWtOdW1iZXJzOiBEYXk8VD5bXSA9IChcbiAgICAgIHRoaXMuX3dpZGVcbiAgICAgICAgPyBbLi4udGhpcy5fd2Vla3MsIC4uLnRoaXMuX25leHRNb250aFdlZWtzXVxuICAgICAgICA6IGlzV2lkZU5leHRNb250aFxuICAgICAgICAgID8gdGhpcy5fbmV4dE1vbnRoV2Vla3NcbiAgICAgICAgICA6IHRoaXMuX3dlZWtzXG4gICAgKS5mbGF0KCk7XG4gICAgY29uc3Qgd2Vla3NGb3JTZWxlY3RNdWx0aXBsZVdlZWtEYXlzOiBEYXk8VD5bXSA9IChcbiAgICAgIGlzV2lkZU5leHRNb250aCA/IHRoaXMuX25leHRNb250aFdlZWtzIDogdGhpcy5fd2Vla3NcbiAgICApLmZsYXQoKTtcbiAgICByZXR1cm4gaHRtbGBcbiAgICAgIDx0YWJsZVxuICAgICAgICBjbGFzcz1cInNiYi1jYWxlbmRhcl9fdGFibGVcIlxuICAgICAgICBAZm9jdXNvdXQ9JHsoZXZlbnQ6IEZvY3VzRXZlbnQpID0+XG4gICAgICAgICAgdGhpcy5faGFuZGxlVGFibGVCbHVyKGV2ZW50LnJlbGF0ZWRUYXJnZXQgYXMgSFRNTEVsZW1lbnQpfVxuICAgICAgICBAYW5pbWF0aW9uZW5kPSR7KGU6IEFuaW1hdGlvbkV2ZW50KSA9PiB0aGlzLl90YWJsZUFuaW1hdGlvbkVuZChlKX1cbiAgICAgID5cbiAgICAgICAgPHRoZWFkIGNsYXNzPVwic2JiLWNhbGVuZGFyX190YWJsZS1oZWFkZXJcIj5cbiAgICAgICAgICA8dHI+XG4gICAgICAgICAgICAke3RoaXMud2Vla051bWJlcnMgPyBodG1sYDx0aCBjbGFzcz1cInNiYi1jYWxlbmRhcl9fdGFibGUtaGVhZGVyLWNlbGxcIj48L3RoPmAgOiBub3RoaW5nfVxuICAgICAgICAgICAgJHt0aGlzLl93ZWVrZGF5cy5tYXAoXG4gICAgICAgICAgICAgICh3ZWVrRGF5OiBXZWVrZGF5LCBpbmRleDogbnVtYmVyKSA9PiBodG1sYFxuICAgICAgICAgICAgICAgIDx0aCBjbGFzcz1cInNiYi1jYWxlbmRhcl9fdGFibGUtaGVhZGVyLWNlbGxcIj5cbiAgICAgICAgICAgICAgICAgICR7dGhpcy5tdWx0aXBsZVxuICAgICAgICAgICAgICAgICAgICA/IGh0bWxgXG4gICAgICAgICAgICAgICAgICAgICAgICA8c2JiLWNhbGVuZGFyLXdlZWtkYXlcbiAgICAgICAgICAgICAgICAgICAgICAgICAgLnZhbHVlPSR7d2Vla0RheX1cbiAgICAgICAgICAgICAgICAgICAgICAgICAgQGNsaWNrPSR7KCkgPT4ge1xuICAgICAgICAgICAgICAgICAgICAgICAgICAgIC8vIE5PVEU6IFN1bmRheXMgaGF2ZSBpbmRleCA3LCB3aGlsZSB0aGVpciB3ZWVrRGF5VmFsdWUgaXMgMFxuICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNvbnN0IGRheXM6IERheTxUPltdID0gd2Vla3NGb3JTZWxlY3RNdWx0aXBsZVdlZWtEYXlzLmZpbHRlcihcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIChkYXk6IERheTxUPikgPT4gZGF5LndlZWtEYXlWYWx1ZSA9PT0gKGluZGV4ICsgMSkgJSA3LFxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICkhO1xuICAgICAgICAgICAgICAgICAgICAgICAgICAgIHRoaXMuX3NlbGVjdE11bHRpcGxlRGF0ZXMoZGF5cyk7XG4gICAgICAgICAgICAgICAgICAgICAgICAgIH19XG4gICAgICAgICAgICAgICAgICAgICAgICA+PC9zYmItY2FsZW5kYXItd2Vla2RheT5cbiAgICAgICAgICAgICAgICAgICAgICBgXG4gICAgICAgICAgICAgICAgICAgIDogaHRtbGBcbiAgICAgICAgICAgICAgICAgICAgICAgIDxzYmItc2NyZWVuLXJlYWRlci1vbmx5PiR7d2Vla0RheS5sb25nfTwvc2JiLXNjcmVlbi1yZWFkZXItb25seT5cbiAgICAgICAgICAgICAgICAgICAgICAgIDxzcGFuIGFyaWEtaGlkZGVuPVwidHJ1ZVwiPiR7d2Vla0RheS5uYXJyb3d9PC9zcGFuPlxuICAgICAgICAgICAgICAgICAgICAgIGB9XG4gICAgICAgICAgICAgICAgPC90aD5cbiAgICAgICAgICAgICAgYCxcbiAgICAgICAgICAgICl9XG4gICAgICAgICAgPC90cj5cbiAgICAgICAgPC90aGVhZD5cbiAgICAgICAgPHRib2R5IGNsYXNzPVwic2JiLWNhbGVuZGFyX190YWJsZS1ib2R5XCI+XG4gICAgICAgICAgJHt3ZWVrcy5tYXAoKHdlZWs6IERheTxUPltdLCByb3dJbmRleDogbnVtYmVyKSA9PiB7XG4gICAgICAgICAgICBjb25zdCBmaXJzdFJvd09mZnNldDogbnVtYmVyID0gREFZU19QRVJfUk9XIC0gd2Vlay5sZW5ndGg7XG4gICAgICAgICAgICBpZiAocm93SW5kZXggPT09IDAgJiYgZmlyc3RSb3dPZmZzZXQpIHtcbiAgICAgICAgICAgICAgcmV0dXJuIGh0bWxgXG4gICAgICAgICAgICAgICAgPHRyPlxuICAgICAgICAgICAgICAgICAgJHt0aGlzLndlZWtOdW1iZXJzXG4gICAgICAgICAgICAgICAgICAgID8gaHRtbGBcbiAgICAgICAgICAgICAgICAgICAgICAgIDx0ZCBjbGFzcz1cInNiYi1jYWxlbmRhcl9fdGFibGUtaGVhZGVyLWNlbGwtdmVydGljYWxcIj5cbiAgICAgICAgICAgICAgICAgICAgICAgICAgJHt0aGlzLm11bHRpcGxlXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgPyBodG1sYFxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICA8c2JiLWNhbGVuZGFyLXdlZWtudW1iZXJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAudmFsdWU9JHt3ZWVrTnVtYmVyc1swXX1cbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBAY2xpY2s9JHsoKSA9PiB7XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBjb25zdCBkYXlzOiBEYXk8VD5bXSA9IHdlZWtzRm9yU2VsZWN0TXVsdGlwbGVXZWVrTnVtYmVycy5maWx0ZXIoXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIChkYXk6IERheTxUPikgPT4gZGF5LndlZWtWYWx1ZSA9PT0gd2Vla051bWJlcnNbMF0sXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICApITtcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHRoaXMuX3NlbGVjdE11bHRpcGxlRGF0ZXMoZGF5cyk7XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgfX1cbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPjwvc2JiLWNhbGVuZGFyLXdlZWtudW1iZXI+XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICBgXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgOiBodG1sYFxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICA8c2JiLXNjcmVlbi1yZWFkZXItb25seVxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID4ke2Ake2kxOG5DYWxlbmRhcldlZWtOdW1iZXJbdGhpcy5fbGFuZ3VhZ2UuY3VycmVudF19ICR7d2Vla051bWJlcnNbMF19YH08L3NiYi1zY3JlZW4tcmVhZGVyLW9ubHlcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPlxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICA8c3BhbiBhcmlhLWhpZGRlbj1cInRydWVcIj4ke3dlZWtOdW1iZXJzWzBdfTwvc3Bhbj5cbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGB9XG4gICAgICAgICAgICAgICAgICAgICAgICA8L3RkPlxuICAgICAgICAgICAgICAgICAgICAgIGBcbiAgICAgICAgICAgICAgICAgICAgOiBub3RoaW5nfVxuICAgICAgICAgICAgICAgICAgJHtbLi4uQXJyYXkoZmlyc3RSb3dPZmZzZXQpLmtleXMoKV0ubWFwKFxuICAgICAgICAgICAgICAgICAgICAoKSA9PiBodG1sYDx0ZCBjbGFzcz1cInNiYi1jYWxlbmRhcl9fdGFibGUtZGF0YVwiPjwvdGQ+YCxcbiAgICAgICAgICAgICAgICAgICl9XG4gICAgICAgICAgICAgICAgICAke3RoaXMuX2NyZWF0ZURheUNlbGxzKHdlZWspfVxuICAgICAgICAgICAgICAgIDwvdHI+XG4gICAgICAgICAgICAgIGA7XG4gICAgICAgICAgICB9XG4gICAgICAgICAgICByZXR1cm4gaHRtbGBcbiAgICAgICAgICAgICAgPHRyPlxuICAgICAgICAgICAgICAgICR7dGhpcy53ZWVrTnVtYmVyc1xuICAgICAgICAgICAgICAgICAgPyBodG1sYFxuICAgICAgICAgICAgICAgICAgICAgIDx0ZCBjbGFzcz1cInNiYi1jYWxlbmRhcl9fdGFibGUtaGVhZGVyLWNlbGwtdmVydGljYWxcIj5cbiAgICAgICAgICAgICAgICAgICAgICAgICR7dGhpcy5tdWx0aXBsZVxuICAgICAgICAgICAgICAgICAgICAgICAgICA/IGh0bWxgXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICA8c2JiLWNhbGVuZGFyLXdlZWtudW1iZXJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgLnZhbHVlPSR7d2Vla051bWJlcnNbcm93SW5kZXhdfVxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBAY2xpY2s9JHsoKSA9PiB7XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgY29uc3QgZGF5czogRGF5PFQ+W10gPSB3ZWVrc0ZvclNlbGVjdE11bHRpcGxlV2Vla051bWJlcnMuZmlsdGVyKFxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgKGRheTogRGF5PFQ+KSA9PiBkYXkud2Vla1ZhbHVlID09PSB3ZWVrTnVtYmVyc1tyb3dJbmRleF0sXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgKSE7XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgdGhpcy5fc2VsZWN0TXVsdGlwbGVEYXRlcyhkYXlzKTtcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgfX1cbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID48L3NiYi1jYWxlbmRhci13ZWVrbnVtYmVyPlxuICAgICAgICAgICAgICAgICAgICAgICAgICAgIGBcbiAgICAgICAgICAgICAgICAgICAgICAgICAgOiBodG1sYFxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPHNiYi1zY3JlZW4tcmVhZGVyLW9ubHlcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPiR7YCR7aTE4bkNhbGVuZGFyV2Vla051bWJlclt0aGlzLl9sYW5ndWFnZS5jdXJyZW50XX0gJHt3ZWVrTnVtYmVyc1tyb3dJbmRleF19YH08L3NiYi1zY3JlZW4tcmVhZGVyLW9ubHlcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID5cbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIDxzcGFuIGFyaWEtaGlkZGVuPVwidHJ1ZVwiPiR7d2Vla051bWJlcnNbcm93SW5kZXhdfTwvc3Bhbj5cbiAgICAgICAgICAgICAgICAgICAgICAgICAgICBgfVxuICAgICAgICAgICAgICAgICAgICAgIDwvdGQ+XG4gICAgICAgICAgICAgICAgICAgIGBcbiAgICAgICAgICAgICAgICAgIDogbm90aGluZ31cbiAgICAgICAgICAgICAgICAke3RoaXMuX2NyZWF0ZURheUNlbGxzKHdlZWspfVxuICAgICAgICAgICAgICA8L3RyPlxuICAgICAgICAgICAgYDtcbiAgICAgICAgICB9KX1cbiAgICAgICAgPC90Ym9keT5cbiAgICAgIDwvdGFibGU+XG4gICAgYDtcbiAgfVxuXG4gIC8qKiBDcmVhdGVzIHRoZSB0YWJsZSBpbiBvcmllbnRhdGlvbj0ndmVydGljYWwnLiAqL1xuICBwcml2YXRlIF9jcmVhdGVEYXlUYWJsZVZlcnRpY2FsKFxuICAgIHdlZWtzOiBEYXk8VD5bXVtdLFxuICAgIHdlZWtOdW1iZXJzOiBudW1iZXJbXSxcbiAgICBuZXh0TW9udGhBY3RpdmVEYXRlPzogVCxcbiAgKTogVGVtcGxhdGVSZXN1bHQge1xuICAgIGNvbnN0IHdlZWtPZmZzZXQgPSB0aGlzLl9kYXRlQWRhcHRlci5nZXRGaXJzdFdlZWtPZmZzZXQoXG4gICAgICBuZXh0TW9udGhBY3RpdmVEYXRlID8/IHRoaXMuX2FjdGl2ZURhdGUsXG4gICAgKTtcbiAgICBjb25zdCB3ZWVrc0ZvclNlbGVjdE11bHRpcGxlV2Vla051bWJlcnM6IERheTxUPltdID0gKFxuICAgICAgdGhpcy5fd2lkZVxuICAgICAgICA/IFsuLi50aGlzLl93ZWVrcywgLi4udGhpcy5fbmV4dE1vbnRoV2Vla3NdXG4gICAgICAgIDogbmV4dE1vbnRoQWN0aXZlRGF0ZVxuICAgICAgICAgID8gdGhpcy5fbmV4dE1vbnRoV2Vla3NcbiAgICAgICAgICA6IHRoaXMuX3dlZWtzXG4gICAgKS5mbGF0KCk7XG4gICAgcmV0dXJuIGh0bWxgXG4gICAgICA8dGFibGVcbiAgICAgICAgY2xhc3M9XCJzYmItY2FsZW5kYXJfX3RhYmxlXCJcbiAgICAgICAgQGZvY3Vzb3V0PSR7KGV2ZW50OiBGb2N1c0V2ZW50KSA9PlxuICAgICAgICAgIHRoaXMuX2hhbmRsZVRhYmxlQmx1cihldmVudC5yZWxhdGVkVGFyZ2V0IGFzIEhUTUxFbGVtZW50KX1cbiAgICAgICAgQGFuaW1hdGlvbmVuZD0keyhlOiBBbmltYXRpb25FdmVudCkgPT4gdGhpcy5fdGFibGVBbmltYXRpb25FbmQoZSl9XG4gICAgICA+XG4gICAgICAgICR7dGhpcy53ZWVrTnVtYmVyc1xuICAgICAgICAgID8gaHRtbGBcbiAgICAgICAgICAgICAgPHRoZWFkIGNsYXNzPVwic2JiLWNhbGVuZGFyX190YWJsZS1oZWFkZXJcIj5cbiAgICAgICAgICAgICAgICA8dHI+XG4gICAgICAgICAgICAgICAgICAke25leHRNb250aEFjdGl2ZURhdGVcbiAgICAgICAgICAgICAgICAgICAgPyBub3RoaW5nXG4gICAgICAgICAgICAgICAgICAgIDogaHRtbGA8dGggY2xhc3M9XCJzYmItY2FsZW5kYXJfX3RhYmxlLWRhdGFcIj48L3RoPmB9XG4gICAgICAgICAgICAgICAgICAke3dlZWtOdW1iZXJzLm1hcChcbiAgICAgICAgICAgICAgICAgICAgKHdlZWtOdW1iZXI6IG51bWJlcikgPT4gaHRtbGBcbiAgICAgICAgICAgICAgICAgICAgICA8dGggY2xhc3M9XCJzYmItY2FsZW5kYXJfX3RhYmxlLWhlYWRlci1jZWxsXCI+XG4gICAgICAgICAgICAgICAgICAgICAgICAke3RoaXMubXVsdGlwbGVcbiAgICAgICAgICAgICAgICAgICAgICAgICAgPyBodG1sYFxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPHNiYi1jYWxlbmRhci13ZWVrbnVtYmVyXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIC52YWx1ZT0ke3dlZWtOdW1iZXJ9XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIEBjbGljaz0keygpID0+IHtcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBjb25zdCBkYXlzOiBEYXk8VD5bXSA9IHdlZWtzRm9yU2VsZWN0TXVsdGlwbGVXZWVrTnVtYmVycy5maWx0ZXIoXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAoZGF5OiBEYXk8VD4pID0+IGRheS53ZWVrVmFsdWUgPT09IHdlZWtOdW1iZXIsXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgKSE7XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgdGhpcy5fc2VsZWN0TXVsdGlwbGVEYXRlcyhkYXlzKTtcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgfX1cbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID48L3NiYi1jYWxlbmRhci13ZWVrbnVtYmVyPlxuICAgICAgICAgICAgICAgICAgICAgICAgICAgIGBcbiAgICAgICAgICAgICAgICAgICAgICAgICAgOiBodG1sYFxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPHNiYi1zY3JlZW4tcmVhZGVyLW9ubHlcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPiR7YCR7aTE4bkNhbGVuZGFyV2Vla051bWJlclt0aGlzLl9sYW5ndWFnZS5jdXJyZW50XX0gJHt3ZWVrTnVtYmVyfWB9PC9zYmItc2NyZWVuLXJlYWRlci1vbmx5XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICA+XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICA8c3BhbiBhcmlhLWhpZGRlbj1cInRydWVcIj4ke3dlZWtOdW1iZXJ9PC9zcGFuPlxuICAgICAgICAgICAgICAgICAgICAgICAgICAgIGB9XG4gICAgICAgICAgICAgICAgICAgICAgPC90aD5cbiAgICAgICAgICAgICAgICAgICAgYCxcbiAgICAgICAgICAgICAgICAgICl9XG4gICAgICAgICAgICAgICAgPC90cj5cbiAgICAgICAgICAgICAgPC90aGVhZD5cbiAgICAgICAgICAgIGBcbiAgICAgICAgICA6IG5vdGhpbmd9XG4gICAgICAgIDx0Ym9keSBjbGFzcz1cInNiYi1jYWxlbmRhcl9fdGFibGUtYm9keVwiPlxuICAgICAgICAgICR7d2Vla3MubWFwKCh3ZWVrOiBEYXk8VD5bXSwgcm93SW5kZXg6IG51bWJlcikgPT4ge1xuICAgICAgICAgICAgY29uc3Qgd2Vla2RheSA9IHRoaXMuX3dlZWtkYXlzW3Jvd0luZGV4XTtcbiAgICAgICAgICAgIGNvbnN0IHNlbGVjdGFibGVEYXlzID0gdGhpcy5fd2lkZSA/IFsuLi53ZWVrLCAuLi50aGlzLl9uZXh0TW9udGhXZWVrc1tyb3dJbmRleF1dIDogd2VlaztcbiAgICAgICAgICAgIHJldHVybiBodG1sYFxuICAgICAgICAgICAgICA8dHI+XG4gICAgICAgICAgICAgICAgJHtuZXh0TW9udGhBY3RpdmVEYXRlXG4gICAgICAgICAgICAgICAgICA/IG5vdGhpbmdcbiAgICAgICAgICAgICAgICAgIDogaHRtbGBcbiAgICAgICAgICAgICAgICAgICAgICA8dGQgY2xhc3M9XCJzYmItY2FsZW5kYXJfX3RhYmxlLWhlYWRlci1jZWxsLXZlcnRpY2FsXCI+XG4gICAgICAgICAgICAgICAgICAgICAgICAke3RoaXMubXVsdGlwbGVcbiAgICAgICAgICAgICAgICAgICAgICAgICAgPyBodG1sYFxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPHNiYi1jYWxlbmRhci13ZWVrZGF5XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIC52YWx1ZT0ke3dlZWtkYXl9XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIEBjbGljaz0keygpID0+IHRoaXMuX3NlbGVjdE11bHRpcGxlRGF0ZXMoc2VsZWN0YWJsZURheXMpfVxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPlxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAke3dlZWtkYXkubmFycm93fVxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPC9zYmItY2FsZW5kYXItd2Vla2RheT5cbiAgICAgICAgICAgICAgICAgICAgICAgICAgICBgXG4gICAgICAgICAgICAgICAgICAgICAgICAgIDogaHRtbGBcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIDxzYmItc2NyZWVuLXJlYWRlci1vbmx5PiR7d2Vla2RheS5sb25nfTwvc2JiLXNjcmVlbi1yZWFkZXItb25seT5cbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIDxzcGFuIGFyaWEtaGlkZGVuPVwidHJ1ZVwiPiR7d2Vla2RheS5uYXJyb3d9PC9zcGFuPlxuICAgICAgICAgICAgICAgICAgICAgICAgICAgIGB9XG4gICAgICAgICAgICAgICAgICAgICAgPC90ZD5cbiAgICAgICAgICAgICAgICAgICAgYH1cbiAgICAgICAgICAgICAgICAke3Jvd0luZGV4IDwgd2Vla09mZnNldFxuICAgICAgICAgICAgICAgICAgPyBodG1sYDx0ZCBjbGFzcz1cInNiYi1jYWxlbmRhcl9fdGFibGUtZGF0YVwiPjwvdGQ+YFxuICAgICAgICAgICAgICAgICAgOiBub3RoaW5nfVxuICAgICAgICAgICAgICAgICR7dGhpcy5fY3JlYXRlRGF5Q2VsbHMod2Vlayl9XG4gICAgICAgICAgICAgIDwvdHI+XG4gICAgICAgICAgICBgO1xuICAgICAgICAgIH0pfVxuICAgICAgICA8L3Rib2R5PlxuICAgICAgPC90YWJsZT5cbiAgICBgO1xuICB9XG5cbiAgLyoqIENyZWF0ZXMgdGhlIGNlbGxzIGZvciB0aGUgZGFpbHkgdmlldy4gKi9cbiAgcHJpdmF0ZSBfY3JlYXRlRGF5Q2VsbHMod2VlazogRGF5PFQ+W10pOiBUZW1wbGF0ZVJlc3VsdFtdIHtcbiAgICByZXR1cm4gd2Vlay5tYXAoKGRheTogRGF5PFQ+KSA9PiB7XG4gICAgICByZXR1cm4gaHRtbGBcbiAgICAgICAgPHRkIGNsYXNzPVwic2JiLWNhbGVuZGFyX190YWJsZS1kYXRhIHNiYi1jYWxlbmRhcl9fZGF5LWNlbGxcIj5cbiAgICAgICAgICA8c2xvdCBuYW1lPSR7ZGF5LnZhbHVlfT5cbiAgICAgICAgICAgIDxzYmItY2FsZW5kYXItZGF5XG4gICAgICAgICAgICAgIHNsb3Q9JHtkYXkudmFsdWV9XG4gICAgICAgICAgICAgIEBjbGljaz0keygpID0+IHRoaXMuX3NlbGVjdERhdGUoZGF5LmRhdGVWYWx1ZSl9XG4gICAgICAgICAgICAgIEBrZXlkb3duPSR7KGV2dDogS2V5Ym9hcmRFdmVudCkgPT4gdGhpcy5faGFuZGxlS2V5Ym9hcmRFdmVudChldnQsIGRheSl9XG4gICAgICAgICAgICA+PC9zYmItY2FsZW5kYXItZGF5PlxuICAgICAgICAgIDwvc2xvdD5cbiAgICAgICAgPC90ZD5cbiAgICAgIGA7XG4gICAgfSk7XG4gIH1cblxuICAvKiogUmVuZGVyIHRoZSB2aWV3IGZvciB0aGUgbW9udGggc2VsZWN0aW9uLiAqL1xuICBwcml2YXRlIF9yZW5kZXJNb250aFZpZXcoKTogVGVtcGxhdGVSZXN1bHQge1xuICAgIHJldHVybiBodG1sYFxuICAgICAgPGRpdiBjbGFzcz1cInNiYi1jYWxlbmRhcl9fY29udHJvbHNcIj5cbiAgICAgICAgJHt0aGlzLl9nZXRBcnJvdyhcbiAgICAgICAgICAnbGVmdCcsXG4gICAgICAgICAgKCkgPT4gdGhpcy5fZ29Ub0RpZmZlcmVudFllYXIoLTEpLFxuICAgICAgICAgIGkxOG5QcmV2aW91c1llYXJbdGhpcy5fbGFuZ3VhZ2UuY3VycmVudF0sXG4gICAgICAgICAgdGhpcy5fcHJldmlvdXNZZWFyRGlzYWJsZWQoKSxcbiAgICAgICAgKX1cbiAgICAgICAgPGRpdiBjbGFzcz1cInNiYi1jYWxlbmRhcl9fY29udHJvbHMtbW9udGhcIj4ke3RoaXMuX2NyZWF0ZUxhYmVsRm9yTW9udGhWaWV3KCl9PC9kaXY+XG4gICAgICAgICR7dGhpcy5fZ2V0QXJyb3coXG4gICAgICAgICAgJ3JpZ2h0JyxcbiAgICAgICAgICAoKSA9PiB0aGlzLl9nb1RvRGlmZmVyZW50WWVhcigxKSxcbiAgICAgICAgICBpMThuTmV4dFllYXJbdGhpcy5fbGFuZ3VhZ2UuY3VycmVudF0sXG4gICAgICAgICAgdGhpcy5fbmV4dFllYXJEaXNhYmxlZCgpLFxuICAgICAgICApfVxuICAgICAgPC9kaXY+XG4gICAgICA8ZGl2IGNsYXNzPVwic2JiLWNhbGVuZGFyX190YWJsZS1vdmVyZmxvdy1icmVha1wiPlxuICAgICAgICA8ZGl2IGNsYXNzPVwic2JiLWNhbGVuZGFyX190YWJsZS1jb250YWluZXIgc2JiLWNhbGVuZGFyX190YWJsZS1tb250aC12aWV3XCI+XG4gICAgICAgICAgJHt0aGlzLl9jcmVhdGVNb250aFRhYmxlKHRoaXMuX21vbnRocywgdGhpcy5fY2hvc2VuWWVhciEpfVxuICAgICAgICAgICR7dGhpcy5fd2lkZSA/IHRoaXMuX2NyZWF0ZU1vbnRoVGFibGUodGhpcy5fbW9udGhzLCB0aGlzLl9jaG9zZW5ZZWFyISArIDEpIDogbm90aGluZ31cbiAgICAgICAgPC9kaXY+XG4gICAgICA8L2Rpdj5cbiAgICBgO1xuICB9XG5cbiAgLyoqIENyZWF0ZXMgdGhlIGxhYmVsIHdpdGggdGhlIHllYXIgZm9yIHRoZSBtb250aGx5IHZpZXcuICovXG4gIHByaXZhdGUgX2NyZWF0ZUxhYmVsRm9yTW9udGhWaWV3KCk6IFRlbXBsYXRlUmVzdWx0IHtcbiAgICByZXR1cm4gaHRtbGAgPGJ1dHRvblxuICAgICAgICB0eXBlPVwiYnV0dG9uXCJcbiAgICAgICAgaWQ9XCJzYmItY2FsZW5kYXJfX21vbnRoLXNlbGVjdGlvblwiXG4gICAgICAgIGNsYXNzPVwic2JiLWNhbGVuZGFyX19jb250cm9scy1jaGFuZ2UtZGF0ZVwiXG4gICAgICAgIGFyaWEtbGFiZWw9JHtgJHtpMThuQ2FsZW5kYXJEYXRlU2VsZWN0aW9uW3RoaXMuX2xhbmd1YWdlLmN1cnJlbnRdfSAke3RoaXMuX2Nob3NlblllYXJ9YH1cbiAgICAgICAgQGNsaWNrPSR7KCkgPT4gdGhpcy5fcmVzZXRDYWxlbmRhclZpZXdBbmRFbWl0TW9udGhDaGFuZ2UodHJ1ZSl9XG4gICAgICA+XG4gICAgICAgICR7dGhpcy5fY2hvc2VuWWVhcn0gJHt0aGlzLl93aWRlID8gYCAtICR7dGhpcy5fY2hvc2VuWWVhciEgKyAxfWAgOiBub3RoaW5nfVxuICAgICAgICA8c2JiLWljb24gbmFtZT1cImNoZXZyb24tc21hbGwtdXAtc21hbGxcIj48L3NiYi1pY29uPlxuICAgICAgPC9idXR0b24+XG4gICAgICA8c2JiLXNjcmVlbi1yZWFkZXItb25seSByb2xlPVwic3RhdHVzXCI+ICR7dGhpcy5fY2hvc2VuWWVhcn0gPC9zYmItc2NyZWVuLXJlYWRlci1vbmx5PmA7XG4gIH1cblxuICAvKiogQ3JlYXRlcyB0aGUgdGFibGUgZm9yIHRoZSBtb250aCBzZWxlY3Rpb24gdmlldy4gKi9cbiAgcHJpdmF0ZSBfY3JlYXRlTW9udGhUYWJsZShtb250aHM6IE1vbnRoQ2VsbFtdW10sIHllYXI6IG51bWJlcik6IFRlbXBsYXRlUmVzdWx0IHtcbiAgICByZXR1cm4gaHRtbGBcbiAgICAgIDx0YWJsZVxuICAgICAgICBjbGFzcz1cInNiYi1jYWxlbmRhcl9fdGFibGVcIlxuICAgICAgICBAYW5pbWF0aW9uZW5kPSR7KGU6IEFuaW1hdGlvbkV2ZW50KSA9PiB0aGlzLl90YWJsZUFuaW1hdGlvbkVuZChlKX1cbiAgICAgID5cbiAgICAgICAgJHt0aGlzLl93aWRlXG4gICAgICAgICAgPyBodG1sYDx0aGVhZCBjbGFzcz1cInNiYi1jYWxlbmRhcl9fdGFibGUtaGVhZGVyXCIgYXJpYS1oaWRkZW49XCJ0cnVlXCI+XG4gICAgICAgICAgICAgIDx0cj5cbiAgICAgICAgICAgICAgICA8dGggY2xhc3M9XCJzYmItY2FsZW5kYXJfX3RhYmxlLWhlYWRlci1jZWxsXCIgY29sc3Bhbj0ke01PTlRIU19QRVJfUk9XfT4ke3llYXJ9PC90aD5cbiAgICAgICAgICAgICAgPC90cj5cbiAgICAgICAgICAgIDwvdGhlYWQ+YFxuICAgICAgICAgIDogbm90aGluZ31cbiAgICAgICAgPHRib2R5IGNsYXNzPVwic2JiLWNhbGVuZGFyX190YWJsZS1ib2R5XCI+XG4gICAgICAgICAgJHttb250aHMubWFwKFxuICAgICAgICAgICAgKHJvdzogTW9udGhDZWxsW10pID0+IGh0bWxgXG4gICAgICAgICAgICAgIDx0cj5cbiAgICAgICAgICAgICAgICAke3Jvdy5tYXAoKG1vbnRoOiBNb250aENlbGwpID0+IHtcbiAgICAgICAgICAgICAgICAgIHJldHVybiBodG1sYFxuICAgICAgICAgICAgICAgICAgICA8dGQgY2xhc3M9XCJzYmItY2FsZW5kYXJfX3RhYmxlLWRhdGFcIj5cbiAgICAgICAgICAgICAgICAgICAgICA8c2JiLWNhbGVuZGFyLW1vbnRoXG4gICAgICAgICAgICAgICAgICAgICAgICAudmFsdWU9XCIke3llYXJ9LSR7bW9udGgudmFsdWV9XCJcbiAgICAgICAgICAgICAgICAgICAgICAgIEBjbGljaz0keygpID0+IHRoaXMuX29uTW9udGhTZWxlY3Rpb24obW9udGgubW9udGhWYWx1ZSwgeWVhcil9XG4gICAgICAgICAgICAgICAgICAgICAgICBAa2V5ZG93bj0keyhldnQ6IEtleWJvYXJkRXZlbnQpID0+IHRoaXMuX2hhbmRsZUtleWJvYXJkRXZlbnQoZXZ0KX1cbiAgICAgICAgICAgICAgICAgICAgICA+XG4gICAgICAgICAgICAgICAgICAgICAgPC9zYmItY2FsZW5kYXItbW9udGg+XG4gICAgICAgICAgICAgICAgICAgIDwvdGQ+XG4gICAgICAgICAgICAgICAgICBgO1xuICAgICAgICAgICAgICAgIH0pfVxuICAgICAgICAgICAgICA8L3RyPlxuICAgICAgICAgICAgYCxcbiAgICAgICAgICApfVxuICAgICAgICA8L3Rib2R5PlxuICAgICAgPC90YWJsZT5cbiAgICBgO1xuICB9XG5cbiAgLyoqIFNlbGVjdCB0aGUgbW9udGggYW5kIGNoYW5nZSB0aGUgdmlldyB0byBkYXkgc2VsZWN0aW9uLiAqL1xuICBwcml2YXRlIF9vbk1vbnRoU2VsZWN0aW9uKG1vbnRoOiBudW1iZXIsIHllYXI6IG51bWJlcik6IHZvaWQge1xuICAgIHRoaXMuX2Nob3Nlbk1vbnRoID0gbW9udGg7XG4gICAgdGhpcy5fbmV4dENhbGVuZGFyVmlldyA9ICdkYXknO1xuICAgIHRoaXMuX2luaXQoXG4gICAgICB0aGlzLl9kYXRlQWRhcHRlci5jcmVhdGVEYXRlKFxuICAgICAgICB5ZWFyLFxuICAgICAgICB0aGlzLl9jaG9zZW5Nb250aCxcbiAgICAgICAgdGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0RGF0ZSh0aGlzLl9hY3RpdmVEYXRlKSxcbiAgICAgICksXG4gICAgKTtcbiAgICB0aGlzLl9zdGFydFRhYmxlVHJhbnNpdGlvbigpO1xuICAgIHRoaXMuX2VtaXRNb250aENoYW5nZSgpO1xuICB9XG5cbiAgLyoqIFJlbmRlciB0aGUgdmlldyBmb3IgdGhlIHllYXIgc2VsZWN0aW9uLiAqL1xuICBwcml2YXRlIF9yZW5kZXJZZWFyVmlldygpOiBUZW1wbGF0ZVJlc3VsdCB7XG4gICAgcmV0dXJuIGh0bWxgXG4gICAgICA8ZGl2IGNsYXNzPVwic2JiLWNhbGVuZGFyX19jb250cm9sc1wiPlxuICAgICAgICAke3RoaXMuX2dldEFycm93KFxuICAgICAgICAgICdsZWZ0JyxcbiAgICAgICAgICAoKSA9PiB0aGlzLl9nb1RvRGlmZmVyZW50WWVhclJhbmdlKC1ZRUFSU19QRVJfUEFHRSksXG4gICAgICAgICAgaTE4blByZXZpb3VzWWVhclJhbmdlKFlFQVJTX1BFUl9QQUdFKVt0aGlzLl9sYW5ndWFnZS5jdXJyZW50XSxcbiAgICAgICAgICB0aGlzLl9wcmV2aW91c1llYXJSYW5nZURpc2FibGVkKCksXG4gICAgICAgICl9XG4gICAgICAgIDxkaXYgY2xhc3M9XCJzYmItY2FsZW5kYXJfX2NvbnRyb2xzLW1vbnRoXCI+JHt0aGlzLl9jcmVhdGVMYWJlbEZvclllYXJWaWV3KCl9PC9kaXY+XG4gICAgICAgICR7dGhpcy5fZ2V0QXJyb3coXG4gICAgICAgICAgJ3JpZ2h0JyxcbiAgICAgICAgICAoKSA9PiB0aGlzLl9nb1RvRGlmZmVyZW50WWVhclJhbmdlKFlFQVJTX1BFUl9QQUdFKSxcbiAgICAgICAgICBpMThuTmV4dFllYXJSYW5nZShZRUFSU19QRVJfUEFHRSlbdGhpcy5fbGFuZ3VhZ2UuY3VycmVudF0sXG4gICAgICAgICAgdGhpcy5fbmV4dFllYXJSYW5nZURpc2FibGVkKCksXG4gICAgICAgICl9XG4gICAgICA8L2Rpdj5cbiAgICAgIDxkaXYgY2xhc3M9XCJzYmItY2FsZW5kYXJfX3RhYmxlLW92ZXJmbG93LWJyZWFrXCI+XG4gICAgICAgIDxkaXYgY2xhc3M9XCJzYmItY2FsZW5kYXJfX3RhYmxlLWNvbnRhaW5lciBzYmItY2FsZW5kYXJfX3RhYmxlLXllYXItdmlld1wiPlxuICAgICAgICAgICR7dGhpcy5fY3JlYXRlWWVhclRhYmxlKHRoaXMuX3llYXJzKX1cbiAgICAgICAgICAke3RoaXMuX3dpZGUgPyB0aGlzLl9jcmVhdGVZZWFyVGFibGUodGhpcy5fbmV4dE1vbnRoWWVhcnMsIHRydWUpIDogbm90aGluZ31cbiAgICAgICAgPC9kaXY+XG4gICAgICA8L2Rpdj5cbiAgICBgO1xuICB9XG5cbiAgLyoqIENyZWF0ZXMgdGhlIGJ1dHRvbiBhcnJvdyBmb3IgYWxsIHRoZSB2aWV3cy4gKi9cbiAgcHJpdmF0ZSBfZ2V0QXJyb3coXG4gICAgZGlyZWN0aW9uOiAnbGVmdCcgfCAncmlnaHQnLFxuICAgIGNsaWNrOiAoKSA9PiB2b2lkLFxuICAgIGFyaWFMYWJlbDogc3RyaW5nLFxuICAgIGRpc2FibGVkOiBib29sZWFuLFxuICApOiBUZW1wbGF0ZVJlc3VsdCB7XG4gICAgcmV0dXJuIGh0bWxgPHNiYi1zZWNvbmRhcnktYnV0dG9uXG4gICAgICBzaXplPVwibVwiXG4gICAgICBpY29uLW5hbWU9XCJjaGV2cm9uLXNtYWxsLSR7ZGlyZWN0aW9ufS1zbWFsbFwiXG4gICAgICBhcmlhLWxhYmVsPSR7YXJpYUxhYmVsfVxuICAgICAgQGNsaWNrPSR7Y2xpY2t9XG4gICAgICA/ZGlzYWJsZWQ9JHtkaXNhYmxlZH1cbiAgICAgIGlkPVwic2JiLWNhbGVuZGFyX19jb250cm9scy0ke2RpcmVjdGlvbiA9PT0gJ2xlZnQnID8gJ3ByZXZpb3VzJyA6ICduZXh0J31cIlxuICAgID48L3NiYi1zZWNvbmRhcnktYnV0dG9uPmA7XG4gIH1cblxuICAvKiogQ3JlYXRlcyB0aGUgbGFiZWwgd2l0aCB0aGUgeWVhciByYW5nZSBmb3IgdGhlIHllYXJseSB2aWV3LiAqL1xuICBwcml2YXRlIF9jcmVhdGVMYWJlbEZvclllYXJWaWV3KCk6IFRlbXBsYXRlUmVzdWx0IHtcbiAgICBjb25zdCBmaXJzdFllYXI6IG51bWJlciA9IHRoaXMuX3llYXJzLmZsYXQoKVswXTtcbiAgICBjb25zdCBsYXN0WWVhckFycmF5OiBudW1iZXJbXSA9ICh0aGlzLl93aWRlID8gdGhpcy5fbmV4dE1vbnRoWWVhcnMgOiB0aGlzLl95ZWFycykuZmxhdCgpO1xuICAgIGNvbnN0IGxhc3RZZWFyOiBudW1iZXIgPSBsYXN0WWVhckFycmF5W2xhc3RZZWFyQXJyYXkubGVuZ3RoIC0gMV07XG4gICAgY29uc3QgeWVhckxhYmVsID0gYCR7Zmlyc3RZZWFyfSAtICR7bGFzdFllYXJ9YDtcbiAgICByZXR1cm4gaHRtbGBcbiAgICAgIDxidXR0b25cbiAgICAgICAgdHlwZT1cImJ1dHRvblwiXG4gICAgICAgIGlkPVwic2JiLWNhbGVuZGFyX195ZWFyLXNlbGVjdGlvblwiXG4gICAgICAgIGNsYXNzPVwic2JiLWNhbGVuZGFyX19jb250cm9scy1jaGFuZ2UtZGF0ZVwiXG4gICAgICAgIGFyaWEtbGFiZWw9XCIke2kxOG5DYWxlbmRhckRhdGVTZWxlY3Rpb25bdGhpcy5fbGFuZ3VhZ2UuY3VycmVudF19ICR7eWVhckxhYmVsfVwiXG4gICAgICAgIEBjbGljaz0keygpID0+IHRoaXMuX3Jlc2V0Q2FsZW5kYXJWaWV3QW5kRW1pdE1vbnRoQ2hhbmdlKHRydWUpfVxuICAgICAgPlxuICAgICAgICAke3llYXJMYWJlbH1cbiAgICAgICAgPHNiYi1pY29uIG5hbWU9XCJjaGV2cm9uLXNtYWxsLXVwLXNtYWxsXCI+PC9zYmItaWNvbj5cbiAgICAgIDwvYnV0dG9uPlxuICAgICAgPHNiYi1zY3JlZW4tcmVhZGVyLW9ubHkgcm9sZT1cInN0YXR1c1wiPiAke3llYXJMYWJlbH0gPC9zYmItc2NyZWVuLXJlYWRlci1vbmx5PlxuICAgIGA7XG4gIH1cblxuICAvKiogQ3JlYXRlcyB0aGUgdGFibGUgZm9yIHRoZSB5ZWFyIHNlbGVjdGlvbiB2aWV3LiAqL1xuICBwcml2YXRlIF9jcmVhdGVZZWFyVGFibGUoeWVhcnM6IG51bWJlcltdW10sIHNoaWZ0UmlnaHQgPSBmYWxzZSk6IFRlbXBsYXRlUmVzdWx0IHtcbiAgICByZXR1cm4gaHRtbGAgPHRhYmxlXG4gICAgICBjbGFzcz1cInNiYi1jYWxlbmRhcl9fdGFibGVcIlxuICAgICAgQGFuaW1hdGlvbmVuZD0keyhlOiBBbmltYXRpb25FdmVudCkgPT4gdGhpcy5fdGFibGVBbmltYXRpb25FbmQoZSl9XG4gICAgPlxuICAgICAgPHRib2R5IGNsYXNzPVwic2JiLWNhbGVuZGFyX190YWJsZS1ib2R5XCI+XG4gICAgICAgICR7eWVhcnMubWFwKFxuICAgICAgICAgIChyb3c6IG51bWJlcltdKSA9PlxuICAgICAgICAgICAgaHRtbGAgPHRyPlxuICAgICAgICAgICAgICAke3Jvdy5tYXAoKHllYXI6IG51bWJlcikgPT4ge1xuICAgICAgICAgICAgICAgIHJldHVybiBodG1sYFxuICAgICAgICAgICAgICAgICAgPHRkIGNsYXNzPVwic2JiLWNhbGVuZGFyX190YWJsZS1kYXRhXCI+XG4gICAgICAgICAgICAgICAgICAgIDxzYmItY2FsZW5kYXIteWVhclxuICAgICAgICAgICAgICAgICAgICAgIC52YWx1ZT0ke1N0cmluZyh5ZWFyKX1cbiAgICAgICAgICAgICAgICAgICAgICBAa2V5ZG93bj0keyhldnQ6IEtleWJvYXJkRXZlbnQpID0+IHRoaXMuX2hhbmRsZUtleWJvYXJkRXZlbnQoZXZ0KX1cbiAgICAgICAgICAgICAgICAgICAgICBAY2xpY2s9JHsoKSA9PiB0aGlzLl9vblllYXJTZWxlY3Rpb24oeWVhciwgc2hpZnRSaWdodCl9XG4gICAgICAgICAgICAgICAgICAgID5cbiAgICAgICAgICAgICAgICAgICAgPC9zYmItY2FsZW5kYXIteWVhcj5cbiAgICAgICAgICAgICAgICAgIDwvdGQ+XG4gICAgICAgICAgICAgICAgYDtcbiAgICAgICAgICAgICAgfSl9XG4gICAgICAgICAgICA8L3RyPmAsXG4gICAgICAgICl9XG4gICAgICA8L3Rib2R5PlxuICAgIDwvdGFibGU+YDtcbiAgfVxuXG4gIC8qKiBTZWxlY3QgdGhlIHllYXIgYW5kIGNoYW5nZSB0aGUgdmlldyB0byBtb250aCBzZWxlY3Rpb24uICovXG4gIHByaXZhdGUgX29uWWVhclNlbGVjdGlvbih5ZWFyOiBudW1iZXIsIHJpZ2h0U2lkZTogYm9vbGVhbik6IHZvaWQge1xuICAgIHRoaXMuX2Nob3NlblllYXIgPSByaWdodFNpZGUgPyB5ZWFyIC0gMSA6IHllYXI7XG4gICAgdGhpcy5fbmV4dENhbGVuZGFyVmlldyA9ICdtb250aCc7XG4gICAgdGhpcy5fYXNzaWduQWN0aXZlRGF0ZShcbiAgICAgIHRoaXMuX2RhdGVBZGFwdGVyLmNyZWF0ZURhdGUoXG4gICAgICAgIHRoaXMuX2Nob3NlblllYXIsXG4gICAgICAgIHRoaXMuX2RhdGVBZGFwdGVyLmdldE1vbnRoKHRoaXMuX2FjdGl2ZURhdGUpLFxuICAgICAgICB0aGlzLl9kYXRlQWRhcHRlci5nZXREYXRlKHRoaXMuX2FjdGl2ZURhdGUpLFxuICAgICAgKSxcbiAgICApO1xuICAgIHRoaXMuX3N0YXJ0VGFibGVUcmFuc2l0aW9uKCk7XG4gIH1cblxuICBwcml2YXRlIF9nZXRWaWV3KCk6IFRlbXBsYXRlUmVzdWx0IHtcbiAgICBpZiAoaXNTZXJ2ZXIgfHwgdGhpcy5oeWRyYXRpb25SZXF1aXJlZCkge1xuICAgICAgLy8gVE9ETzogV2UgZGlzYWJsZSBTU1IgZm9yIGNhbGVuZGFyIGZvciBub3cuIEZpZ3VyZSBvdXQsIGlmIHRoZXJlIGlzIGEgd2F5XG4gICAgICAvLyB0byBlbmFibGUgaXQsIHdoaWxlIGNvbnNpZGVyaW5nIGkxOG4gYW5kIGRhdGUgaW5mb3JtYXRpb24uXG4gICAgICByZXR1cm4gaHRtbGAke25vdGhpbmd9YDtcbiAgICB9XG4gICAgc3dpdGNoICh0aGlzLl9jYWxlbmRhclZpZXcpIHtcbiAgICAgIGNhc2UgJ3llYXInOlxuICAgICAgICByZXR1cm4gdGhpcy5fcmVuZGVyWWVhclZpZXcoKTtcbiAgICAgIGNhc2UgJ21vbnRoJzpcbiAgICAgICAgcmV0dXJuIHRoaXMuX3JlbmRlck1vbnRoVmlldygpO1xuICAgICAgY2FzZSAnZGF5JzpcbiAgICAgIGRlZmF1bHQ6XG4gICAgICAgIHJldHVybiB0aGlzLl9yZW5kZXJEYXlWaWV3KCk7XG4gICAgfVxuICB9XG5cbiAgcHJpdmF0ZSBfdGFibGVBbmltYXRpb25FbmQoZXZlbnQ6IEFuaW1hdGlvbkV2ZW50KTogdm9pZCB7XG4gICAgY29uc3QgdGFibGUgPSBldmVudC50YXJnZXQgYXMgSFRNTEVsZW1lbnQ7XG4gICAgaWYgKGV2ZW50LmFuaW1hdGlvbk5hbWUgPT09ICdoaWRlJykge1xuICAgICAgdGFibGUuY2xhc3NMaXN0LnJlbW92ZSgnc2JiLWNhbGVuZGFyX190YWJsZS1oaWRlJyk7XG4gICAgICBpZiAodGhpcy5fY29udGFpbmluZ0ZvY3VzKSB7XG4gICAgICAgIHRoaXMuX3Jlc2V0Rm9jdXMgPSB0cnVlO1xuICAgICAgfVxuICAgICAgdGhpcy5fY2FsZW5kYXJWaWV3ID0gdGhpcy5fbmV4dENhbGVuZGFyVmlldztcbiAgICB9IGVsc2UgaWYgKGV2ZW50LmFuaW1hdGlvbk5hbWUgPT09ICdzaG93Jykge1xuICAgICAgdGhpcy5pbnRlcm5hbHMuc3RhdGVzLmRlbGV0ZSgndHJhbnNpdGlvbicpO1xuICAgIH1cbiAgfVxuXG4gIHByaXZhdGUgX3N0YXJ0VGFibGVUcmFuc2l0aW9uKCk6IHZvaWQge1xuICAgIHRoaXMuaW50ZXJuYWxzLnN0YXRlcy5hZGQoJ3RyYW5zaXRpb24nKTtcbiAgICB0aGlzLnNoYWRvd1Jvb3RcbiAgICAgID8ucXVlcnlTZWxlY3RvckFsbCgndGFibGUnKVxuICAgICAgPy5mb3JFYWNoKChlKSA9PiBlLmNsYXNzTGlzdC50b2dnbGUoJ3NiYi1jYWxlbmRhcl9fdGFibGUtaGlkZScpKTtcbiAgfVxuXG4gIHByb3RlY3RlZCBvdmVycmlkZSByZW5kZXIoKTogVGVtcGxhdGVSZXN1bHQge1xuICAgIHJldHVybiBodG1sYDxkaXYgY2xhc3M9XCJzYmItY2FsZW5kYXJfX3dyYXBwZXJcIj4ke3RoaXMuX2dldFZpZXcoKX08L2Rpdj5gO1xuICB9XG59XG5cbmRlY2xhcmUgZ2xvYmFsIHtcbiAgaW50ZXJmYWNlIEhUTUxFbGVtZW50VGFnTmFtZU1hcCB7XG4gICAgLy8gZXNsaW50LWRpc2FibGUtbmV4dC1saW5lIEB0eXBlc2NyaXB0LWVzbGludC9uYW1pbmctY29udmVudGlvblxuICAgICdzYmItY2FsZW5kYXInOiBTYmJDYWxlbmRhckVsZW1lbnQ7XG4gIH1cbiAgaW50ZXJmYWNlIEhUTUxFbGVtZW50RXZlbnRNYXAge1xuICAgIG1vbnRoY2hhbmdlOiBTYmJNb250aENoYW5nZUV2ZW50O1xuICB9XG59XG4iXSwieF9nb29nbGVfaWdub3JlTGlzdCI6WzAsMSwyLDMsNCw1LDYsNyw4LDksMTAsMTEsMTIsMTMsMTRdLCJtYXBwaW5ncyI6Ijs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7O0FBcUNBLElBQWEsYUFBYTtBQWdCSCxLQUFLLElBQUksSUFBSSxFQUFFLEdBQUcsS0FBSyxLQUFLLEtBQUs7Ozs7OztBQXVCeEQsSUFBYSxxQkFBcUI7Ozs7OztBQWtHbEMsSUFBYSxlQWRnQixPQWNlO0FBT2YsZUFBZTtBQU9mLGVBQWUsYUFPRSxLQU9HOzs7Ozs7Ozs7Ozs7QUFhakQsSUFBYSxzQkFBc0IsT0FBTyxJQUFJLG9CQUFvQjs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7QUNsTGxFLFNBQWdCLGNBQWMsTUFBTSxPQUFPO0FBQ3pDLEtBQUksT0FBTyxTQUFTLFdBQVksUUFBTyxLQUFLLE1BQU07QUFFbEQsS0FBSSxRQUFRLE9BQU8sU0FBUyxZQUFZLHVCQUF1QixLQUM3RCxRQUFPLEtBQUsscUJBQXFCLE1BQU07QUFFekMsS0FBSSxnQkFBZ0IsS0FBTSxRQUFPLElBQUksS0FBSyxZQUFZLE1BQU07QUFFNUQsUUFBTyxJQUFJLEtBQUssTUFBTTs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7O0FDTHhCLFNBQWdCLE9BQU8sVUFBVSxTQUFTO0FBRXhDLFFBQU8sY0FBYyxXQUFXLFVBQVUsU0FBUzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7QUNickQsU0FBZ0IsUUFBUSxNQUFNLFFBQVEsU0FBUztDQUM3QyxNQUFNLFFBQVEsT0FBTyxNQUFNLFNBQVMsR0FBRztBQUN2QyxLQUFJ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