@sbb-esta/lyne-elements-dev 4.9.0-dev.1774875578 → 4.9.0-dev.1774887348
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/calendar/calendar/calendar.component.js +1 -1
- package/calendar/calendar.js +1 -1
- package/{calendar.component-Dk8Xun-b.js → calendar.component-BRIW9HVu.js} +88 -186
- package/calendar.js +1 -1
- package/calendar.pure.js +1 -1
- package/core/datetime/date-adapter.js +13 -13
- package/core/datetime.js +4 -4
- package/core.js +47 -47
- package/custom-elements.json +1050 -939
- package/development/calendar/calendar/calendar.component.d.ts +1 -12
- package/development/calendar/calendar/calendar.component.d.ts.map +1 -1
- package/development/calendar/calendar/calendar.component.js +1 -1
- package/development/calendar/calendar.js +1 -1
- package/development/calendar.component-CsAl9EAg.js +1678 -0
- package/development/calendar.js +1 -1
- package/development/calendar.pure.js +1 -1
- package/development/core/datetime/date-adapter.d.ts +9 -2
- package/development/core/datetime/date-adapter.d.ts.map +1 -1
- package/development/core/datetime/date-adapter.js +11 -4
- package/development/core/datetime.js +2 -2
- package/development/core.js +2 -2
- package/package.json +2 -2
- package/development/calendar.component-CRvTYw1K.js +0 -2223
|
@@ -0,0 +1,1678 @@
|
|
|
1
|
+
import { __esDecorate, __runInitializers } from "tslib";
|
|
2
|
+
import { css, html, isServer, nothing } from "lit";
|
|
3
|
+
import { property, state } from "lit/decorators.js";
|
|
4
|
+
import { SbbElement } from "./core/base-elements.js";
|
|
5
|
+
import { forceType, handleDistinctChange, plainDate } from "./core/decorators.js";
|
|
6
|
+
import { boxSizingStyles } from "./core/styles.js";
|
|
7
|
+
import { SbbSecondaryButtonElement } from "./button.pure.js";
|
|
8
|
+
import { SbbLanguageController, SbbMediaMatcherController, SbbMediaQueryBreakpointLargeAndAbove } from "./core/controllers.js";
|
|
9
|
+
import { i18nCalendarDateSelection, i18nCalendarWeekNumber, i18nNextMonth, i18nNextYear, i18nNextYearRange, i18nPreviousMonth, i18nPreviousYear, i18nPreviousYearRange, i18nYearMonthSelection } from "./core/i18n.js";
|
|
10
|
+
import "./icon.js";
|
|
11
|
+
import { isArrowKeyOrPageKeysPressed } from "./core/a11y.js";
|
|
12
|
+
import { readConfig } from "./core/config.js";
|
|
13
|
+
import { DAYS_PER_ROW, MONDAY, MONTHS_PER_PAGE, MONTHS_PER_ROW, THURSDAY, TUESDAY, WEDNESDAY, YEARS_PER_PAGE, YEARS_PER_ROW, defaultDateAdapter } from "./core/datetime.js";
|
|
14
|
+
import { SbbCalendarDayElement } from "./calendar/calendar-day/calendar-day.component.js";
|
|
15
|
+
import { SbbCalendarMonthElement } from "./calendar/calendar-month/calendar-month.component.js";
|
|
16
|
+
import { SbbCalendarWeekdayElement } from "./calendar/calendar-weekday/calendar-weekday.component.js";
|
|
17
|
+
import { SbbCalendarWeeknumberElement } from "./calendar/calendar-weeknumber/calendar-weeknumber.component.js";
|
|
18
|
+
import { SbbCalendarYearElement } from "./calendar/calendar-year/calendar-year.component.js";
|
|
19
|
+
import "./screen-reader-only.js";
|
|
20
|
+
//#region src/elements/calendar/calendar/calendar.scss?lit&inline
|
|
21
|
+
var calendar_default = css`:host {
|
|
22
|
+
display: block;
|
|
23
|
+
width: max-content;
|
|
24
|
+
--sbb-calendar-cell-size: 2.75rem;
|
|
25
|
+
--sbb-calendar-cell-border-radius: calc(
|
|
26
|
+
var(--sbb-border-radius-4x) + var(--sbb-calendar-cell-border-width)
|
|
27
|
+
);
|
|
28
|
+
--sbb-calendar-cell-transition-duration: var(
|
|
29
|
+
--sbb-disable-animation-duration,
|
|
30
|
+
var(--sbb-animation-duration-2x)
|
|
31
|
+
);
|
|
32
|
+
}
|
|
33
|
+
@media (min-width: calc(37.5rem)) {
|
|
34
|
+
:host {
|
|
35
|
+
--sbb-calendar-control-view-change-height: 3rem;
|
|
36
|
+
}
|
|
37
|
+
}
|
|
38
|
+
|
|
39
|
+
:host(:is(:state(enhanced),[state--enhanced])) {
|
|
40
|
+
--sbb-calendar-cell-padding-block-end: var(--sbb-spacing-fixed-3x);
|
|
41
|
+
}
|
|
42
|
+
|
|
43
|
+
.sbb-calendar__wrapper {
|
|
44
|
+
width: 100%;
|
|
45
|
+
display: block;
|
|
46
|
+
transition-duration: var(--sbb-calendar-cell-transition-duration);
|
|
47
|
+
}
|
|
48
|
+
|
|
49
|
+
.sbb-calendar__controls {
|
|
50
|
+
width: 100%;
|
|
51
|
+
display: inline-flex;
|
|
52
|
+
align-items: center;
|
|
53
|
+
gap: var(--sbb-calendar-control-gap);
|
|
54
|
+
margin-block-end: var(--sbb-calendar-control-margin-block-end);
|
|
55
|
+
}
|
|
56
|
+
|
|
57
|
+
.sbb-calendar__controls-month {
|
|
58
|
+
width: 100%;
|
|
59
|
+
display: flex;
|
|
60
|
+
gap: var(--sbb-calendar-tables-gap);
|
|
61
|
+
}
|
|
62
|
+
|
|
63
|
+
#sbb-calendar__controls-previous,
|
|
64
|
+
#sbb-calendar__controls-next {
|
|
65
|
+
-webkit-tap-highlight-color: transparent;
|
|
66
|
+
}
|
|
67
|
+
|
|
68
|
+
.sbb-calendar__controls-change-date {
|
|
69
|
+
appearance: none;
|
|
70
|
+
box-sizing: border-box;
|
|
71
|
+
margin: 0;
|
|
72
|
+
outline: none;
|
|
73
|
+
border: none;
|
|
74
|
+
border-radius: 0;
|
|
75
|
+
padding: 0;
|
|
76
|
+
background: none;
|
|
77
|
+
-webkit-tap-highlight-color: transparent;
|
|
78
|
+
color: inherit;
|
|
79
|
+
display: flex;
|
|
80
|
+
align-items: center;
|
|
81
|
+
margin: auto;
|
|
82
|
+
height: var(--sbb-calendar-control-view-change-height);
|
|
83
|
+
font-size: var(--sbb-text-font-size-s);
|
|
84
|
+
letter-spacing: var(--sbb-typo-letter-spacing-text);
|
|
85
|
+
text-transform: capitalize;
|
|
86
|
+
cursor: var(--sbb-cursor-pointer);
|
|
87
|
+
padding-inline: var(--sbb-calendar-control-view-change-padding-inline);
|
|
88
|
+
border-radius: var(--sbb-border-radius-infinity);
|
|
89
|
+
background-color: var(--sbb-calendar-control-view-change-background);
|
|
90
|
+
color: var(--sbb-calendar-control-view-change-color);
|
|
91
|
+
transition-duration: var(--sbb-calendar-cell-transition-duration);
|
|
92
|
+
transition-timing-function: var(--sbb-calendar-cell-transition-easing-function);
|
|
93
|
+
transition-property: background-color, padding-block-end;
|
|
94
|
+
}
|
|
95
|
+
.sbb-calendar__controls-change-date:disabled {
|
|
96
|
+
--sbb-calendar-control-view-change-background: var(--sbb-color-milk);
|
|
97
|
+
--sbb-calendar-control-view-change-background: light-dark(
|
|
98
|
+
var(--sbb-color-milk),
|
|
99
|
+
var(--sbb-color-anthracite)
|
|
100
|
+
);
|
|
101
|
+
--sbb-calendar-control-view-change-color: var(--sbb-color-granite);
|
|
102
|
+
--sbb-calendar-control-view-change-color: light-dark(
|
|
103
|
+
var(--sbb-color-granite),
|
|
104
|
+
var(--sbb-color-aluminium)
|
|
105
|
+
);
|
|
106
|
+
cursor: unset;
|
|
107
|
+
}
|
|
108
|
+
.sbb-calendar__controls-change-date:focus-visible {
|
|
109
|
+
outline-offset: var(--sbb-focus-outline-offset);
|
|
110
|
+
outline: var(--sbb-focus-outline-color) var(--sbb-focus-outline-style, solid) var(--sbb-focus-outline-width);
|
|
111
|
+
outline-offset: var(--sbb-spacing-fixed-1x);
|
|
112
|
+
}
|
|
113
|
+
@media (any-hover: hover) {
|
|
114
|
+
.sbb-calendar__controls-change-date:not(:active, :disabled):hover {
|
|
115
|
+
padding-block-end: var(--sbb-calendar-cell-hover-shift);
|
|
116
|
+
}
|
|
117
|
+
}
|
|
118
|
+
.sbb-calendar__controls-change-date:not(:disabled):active {
|
|
119
|
+
--sbb-calendar-control-view-change-background: var(--sbb-background-color-3);
|
|
120
|
+
}
|
|
121
|
+
|
|
122
|
+
.sbb-calendar__table-month-view,
|
|
123
|
+
.sbb-calendar__table-year-view {
|
|
124
|
+
--sbb-calendar-table-column-spaces: 6;
|
|
125
|
+
}
|
|
126
|
+
|
|
127
|
+
.sbb-calendar__table-overflow-break {
|
|
128
|
+
contain: layout;
|
|
129
|
+
}
|
|
130
|
+
|
|
131
|
+
.sbb-calendar__table-container {
|
|
132
|
+
display: flex;
|
|
133
|
+
gap: var(--sbb-calendar-tables-gap);
|
|
134
|
+
margin-inline: var(--sbb-calendar-margin);
|
|
135
|
+
--sbb-calendar-min-width: calc(7 * var(--sbb-calendar-cell-size));
|
|
136
|
+
--sbb-calendar-overflow: calc(100% - var(--sbb-calendar-min-width));
|
|
137
|
+
--sbb-calendar-start-offset: min(
|
|
138
|
+
0px,
|
|
139
|
+
-1 * (var(--sbb-calendar-overflow) / var(--sbb-calendar-table-column-spaces))
|
|
140
|
+
);
|
|
141
|
+
--sbb-calendar-margin: var(--sbb-calendar-start-offset);
|
|
142
|
+
}
|
|
143
|
+
:host(:is(:state(wide),[state--wide])) .sbb-calendar__table-container {
|
|
144
|
+
--sbb-calendar-min-width: calc(
|
|
145
|
+
2 * 7 * var(--sbb-calendar-cell-size) + var(--sbb-calendar-tables-gap)
|
|
146
|
+
);
|
|
147
|
+
--sbb-calendar-margin: calc(0.5 * var(--sbb-calendar-start-offset));
|
|
148
|
+
}
|
|
149
|
+
:host([orientation=horizontal][week-numbers]) .sbb-calendar__table-container {
|
|
150
|
+
--sbb-calendar-min-width: calc(8 * var(--sbb-calendar-cell-size));
|
|
151
|
+
}
|
|
152
|
+
:host([orientation=horizontal][week-numbers]:is(:state(wide),[state--wide])) .sbb-calendar__table-container {
|
|
153
|
+
--sbb-calendar-min-width: calc(
|
|
154
|
+
2 * 8 * var(--sbb-calendar-cell-size) + var(--sbb-calendar-tables-gap)
|
|
155
|
+
);
|
|
156
|
+
}
|
|
157
|
+
:host([orientation=vertical]) .sbb-calendar__table-container {
|
|
158
|
+
min-width: var(--sbb-calendar-min-width);
|
|
159
|
+
--sbb-calendar-start-offset: 0;
|
|
160
|
+
}
|
|
161
|
+
:host([orientation=vertical]:is(:state(wide),[state--wide])) .sbb-calendar__table-container {
|
|
162
|
+
--sbb-calendar-min-width: calc(
|
|
163
|
+
13 * var(--sbb-calendar-cell-size) + var(--sbb-calendar-tables-gap)
|
|
164
|
+
);
|
|
165
|
+
}
|
|
166
|
+
|
|
167
|
+
.sbb-calendar__table {
|
|
168
|
+
width: 100%;
|
|
169
|
+
border-collapse: collapse;
|
|
170
|
+
height: max-content;
|
|
171
|
+
animation-name: show;
|
|
172
|
+
animation-duration: var(--sbb-calendar-table-animation-duration);
|
|
173
|
+
}
|
|
174
|
+
.sbb-calendar__table.sbb-calendar__table-hide {
|
|
175
|
+
--sbb-calendar-cell-transition-duration: 0ms;
|
|
176
|
+
animation-name: hide;
|
|
177
|
+
animation-duration: var(--sbb-calendar-table-animation-duration);
|
|
178
|
+
}
|
|
179
|
+
:host(:not(:is(:state(wide),[state--wide]))) .sbb-calendar__table {
|
|
180
|
+
min-width: 100%;
|
|
181
|
+
}
|
|
182
|
+
|
|
183
|
+
.sbb-calendar__table-header {
|
|
184
|
+
text-align: center;
|
|
185
|
+
}
|
|
186
|
+
|
|
187
|
+
.sbb-calendar__table-body {
|
|
188
|
+
text-align: center;
|
|
189
|
+
}
|
|
190
|
+
|
|
191
|
+
.sbb-calendar__table-header-cell,
|
|
192
|
+
.sbb-calendar__table-header-cell-vertical {
|
|
193
|
+
width: var(--sbb-calendar-cell-size);
|
|
194
|
+
color: var(--sbb-calendar-header-color);
|
|
195
|
+
padding: 0;
|
|
196
|
+
font-size: var(--sbb-text-font-size-xs);
|
|
197
|
+
font-weight: normal;
|
|
198
|
+
letter-spacing: var(--sbb-typo-letter-spacing-text);
|
|
199
|
+
}
|
|
200
|
+
:host(:not([multiple])) .sbb-calendar__table-header-cell,
|
|
201
|
+
:host(:not([multiple])) .sbb-calendar__table-header-cell-vertical {
|
|
202
|
+
height: var(--sbb-calendar-cell-size);
|
|
203
|
+
}
|
|
204
|
+
|
|
205
|
+
.sbb-calendar__table-header-cell-vertical {
|
|
206
|
+
padding-block-end: var(--sbb-calendar-cell-padding-block-end);
|
|
207
|
+
}
|
|
208
|
+
|
|
209
|
+
.sbb-calendar__table-data {
|
|
210
|
+
position: relative;
|
|
211
|
+
padding: 0;
|
|
212
|
+
text-align: center;
|
|
213
|
+
}
|
|
214
|
+
|
|
215
|
+
.sbb-calendar__day-cell {
|
|
216
|
+
padding-block-end: var(--sbb-calendar-cell-padding-block-end);
|
|
217
|
+
}
|
|
218
|
+
|
|
219
|
+
:host(:not(:is(:state(enhanced),[state--enhanced]))) sbb-calendar-day {
|
|
220
|
+
--sbb-calendar-cell-justify-content: center;
|
|
221
|
+
--sbb-calendar-day-height: 2.75rem;
|
|
222
|
+
--sbb-calendar-day-extra-display: none;
|
|
223
|
+
--sbb-calendar-day-value-height: unset;
|
|
224
|
+
--sbb-calendar-day-crossed-out-top: 50%;
|
|
225
|
+
--sbb-calendar-day-crossed-out-translate: translate(-50%, -50%) rotate(-45deg);
|
|
226
|
+
}
|
|
227
|
+
|
|
228
|
+
@keyframes show {
|
|
229
|
+
from {
|
|
230
|
+
opacity: 0;
|
|
231
|
+
transform: translateY(var(--sbb-calendar-table-animation-shift));
|
|
232
|
+
}
|
|
233
|
+
to {
|
|
234
|
+
opacity: 1;
|
|
235
|
+
transform: translateY(0%);
|
|
236
|
+
}
|
|
237
|
+
}
|
|
238
|
+
@keyframes hide {
|
|
239
|
+
from {
|
|
240
|
+
opacity: 1;
|
|
241
|
+
transform: translateY(0%);
|
|
242
|
+
}
|
|
243
|
+
to {
|
|
244
|
+
opacity: 0;
|
|
245
|
+
transform: translateY(var(--sbb-calendar-table-animation-shift));
|
|
246
|
+
}
|
|
247
|
+
}`;
|
|
248
|
+
//#endregion
|
|
249
|
+
//#region src/elements/calendar/calendar/calendar.component.ts
|
|
250
|
+
SbbSecondaryButtonElement.define();
|
|
251
|
+
SbbCalendarDayElement.define();
|
|
252
|
+
SbbCalendarMonthElement.define();
|
|
253
|
+
SbbCalendarYearElement.define();
|
|
254
|
+
SbbCalendarWeekdayElement.define();
|
|
255
|
+
SbbCalendarWeeknumberElement.define();
|
|
256
|
+
var SbbMonthChangeEvent = class extends Event {
|
|
257
|
+
get range() {
|
|
258
|
+
return this._range;
|
|
259
|
+
}
|
|
260
|
+
constructor(range) {
|
|
261
|
+
super("monthchange", {
|
|
262
|
+
bubbles: true,
|
|
263
|
+
composed: true
|
|
264
|
+
});
|
|
265
|
+
this._range = Object.freeze(range || []);
|
|
266
|
+
}
|
|
267
|
+
};
|
|
268
|
+
/**
|
|
269
|
+
* It displays a calendar which allows choosing a date.
|
|
270
|
+
*
|
|
271
|
+
* @slot - Use the unnamed slot to add customized `sbb-calendar-day` elements.
|
|
272
|
+
*/
|
|
273
|
+
var SbbCalendarElement = (() => {
|
|
274
|
+
let _classSuper = SbbElement;
|
|
275
|
+
let _instanceExtraInitializers = [];
|
|
276
|
+
let _wide_decorators;
|
|
277
|
+
let _wide_initializers = [];
|
|
278
|
+
let _wide_extraInitializers = [];
|
|
279
|
+
let _view_decorators;
|
|
280
|
+
let _view_initializers = [];
|
|
281
|
+
let _view_extraInitializers = [];
|
|
282
|
+
let _min_decorators;
|
|
283
|
+
let _min_initializers = [];
|
|
284
|
+
let _min_extraInitializers = [];
|
|
285
|
+
let _max_decorators;
|
|
286
|
+
let _max_initializers = [];
|
|
287
|
+
let _max_extraInitializers = [];
|
|
288
|
+
let _multiple_decorators;
|
|
289
|
+
let _multiple_initializers = [];
|
|
290
|
+
let _multiple_extraInitializers = [];
|
|
291
|
+
let _set_selected_decorators;
|
|
292
|
+
let __selected_decorators;
|
|
293
|
+
let __selected_initializers = [];
|
|
294
|
+
let __selected_extraInitializers = [];
|
|
295
|
+
let _dateFilter_decorators;
|
|
296
|
+
let _dateFilter_initializers = [];
|
|
297
|
+
let _dateFilter_extraInitializers = [];
|
|
298
|
+
let _orientation_decorators;
|
|
299
|
+
let _orientation_initializers = [];
|
|
300
|
+
let _orientation_extraInitializers = [];
|
|
301
|
+
let _weekNumbers_decorators;
|
|
302
|
+
let _weekNumbers_initializers = [];
|
|
303
|
+
let _weekNumbers_extraInitializers = [];
|
|
304
|
+
let __activeDate_decorators;
|
|
305
|
+
let __activeDate_initializers = [];
|
|
306
|
+
let __activeDate_extraInitializers = [];
|
|
307
|
+
let _set__wide_decorators;
|
|
308
|
+
let __calendarView_decorators;
|
|
309
|
+
let __calendarView_initializers = [];
|
|
310
|
+
let __calendarView_extraInitializers = [];
|
|
311
|
+
let __initialized_decorators;
|
|
312
|
+
let __initialized_initializers = [];
|
|
313
|
+
let __initialized_extraInitializers = [];
|
|
314
|
+
return class SbbCalendarElement extends _classSuper {
|
|
315
|
+
static {
|
|
316
|
+
const _metadata = typeof Symbol === "function" && Symbol.metadata ? Object.create(_classSuper[Symbol.metadata] ?? null) : void 0;
|
|
317
|
+
_wide_decorators = [forceType(), property({ type: Boolean })];
|
|
318
|
+
_view_decorators = [property()];
|
|
319
|
+
_min_decorators = [plainDate(), property()];
|
|
320
|
+
_max_decorators = [plainDate(), property()];
|
|
321
|
+
_multiple_decorators = [
|
|
322
|
+
forceType(),
|
|
323
|
+
handleDistinctChange((e, newValue) => e._onMultipleChanged(newValue)),
|
|
324
|
+
property({ type: Boolean })
|
|
325
|
+
];
|
|
326
|
+
_set_selected_decorators = [property()];
|
|
327
|
+
__selected_decorators = [state()];
|
|
328
|
+
_dateFilter_decorators = [property({ attribute: "date-filter" })];
|
|
329
|
+
_orientation_decorators = [property({ reflect: true })];
|
|
330
|
+
_weekNumbers_decorators = [forceType(), property({
|
|
331
|
+
attribute: "week-numbers",
|
|
332
|
+
type: Boolean
|
|
333
|
+
})];
|
|
334
|
+
__activeDate_decorators = [state()];
|
|
335
|
+
_set__wide_decorators = [state()];
|
|
336
|
+
__calendarView_decorators = [state()];
|
|
337
|
+
__initialized_decorators = [state()];
|
|
338
|
+
__esDecorate(this, null, _wide_decorators, {
|
|
339
|
+
kind: "accessor",
|
|
340
|
+
name: "wide",
|
|
341
|
+
static: false,
|
|
342
|
+
private: false,
|
|
343
|
+
access: {
|
|
344
|
+
has: (obj) => "wide" in obj,
|
|
345
|
+
get: (obj) => obj.wide,
|
|
346
|
+
set: (obj, value) => {
|
|
347
|
+
obj.wide = value;
|
|
348
|
+
}
|
|
349
|
+
},
|
|
350
|
+
metadata: _metadata
|
|
351
|
+
}, _wide_initializers, _wide_extraInitializers);
|
|
352
|
+
__esDecorate(this, null, _view_decorators, {
|
|
353
|
+
kind: "accessor",
|
|
354
|
+
name: "view",
|
|
355
|
+
static: false,
|
|
356
|
+
private: false,
|
|
357
|
+
access: {
|
|
358
|
+
has: (obj) => "view" in obj,
|
|
359
|
+
get: (obj) => obj.view,
|
|
360
|
+
set: (obj, value) => {
|
|
361
|
+
obj.view = value;
|
|
362
|
+
}
|
|
363
|
+
},
|
|
364
|
+
metadata: _metadata
|
|
365
|
+
}, _view_initializers, _view_extraInitializers);
|
|
366
|
+
__esDecorate(this, null, _min_decorators, {
|
|
367
|
+
kind: "accessor",
|
|
368
|
+
name: "min",
|
|
369
|
+
static: false,
|
|
370
|
+
private: false,
|
|
371
|
+
access: {
|
|
372
|
+
has: (obj) => "min" in obj,
|
|
373
|
+
get: (obj) => obj.min,
|
|
374
|
+
set: (obj, value) => {
|
|
375
|
+
obj.min = value;
|
|
376
|
+
}
|
|
377
|
+
},
|
|
378
|
+
metadata: _metadata
|
|
379
|
+
}, _min_initializers, _min_extraInitializers);
|
|
380
|
+
__esDecorate(this, null, _max_decorators, {
|
|
381
|
+
kind: "accessor",
|
|
382
|
+
name: "max",
|
|
383
|
+
static: false,
|
|
384
|
+
private: false,
|
|
385
|
+
access: {
|
|
386
|
+
has: (obj) => "max" in obj,
|
|
387
|
+
get: (obj) => obj.max,
|
|
388
|
+
set: (obj, value) => {
|
|
389
|
+
obj.max = value;
|
|
390
|
+
}
|
|
391
|
+
},
|
|
392
|
+
metadata: _metadata
|
|
393
|
+
}, _max_initializers, _max_extraInitializers);
|
|
394
|
+
__esDecorate(this, null, _multiple_decorators, {
|
|
395
|
+
kind: "accessor",
|
|
396
|
+
name: "multiple",
|
|
397
|
+
static: false,
|
|
398
|
+
private: false,
|
|
399
|
+
access: {
|
|
400
|
+
has: (obj) => "multiple" in obj,
|
|
401
|
+
get: (obj) => obj.multiple,
|
|
402
|
+
set: (obj, value) => {
|
|
403
|
+
obj.multiple = value;
|
|
404
|
+
}
|
|
405
|
+
},
|
|
406
|
+
metadata: _metadata
|
|
407
|
+
}, _multiple_initializers, _multiple_extraInitializers);
|
|
408
|
+
__esDecorate(this, null, _set_selected_decorators, {
|
|
409
|
+
kind: "setter",
|
|
410
|
+
name: "selected",
|
|
411
|
+
static: false,
|
|
412
|
+
private: false,
|
|
413
|
+
access: {
|
|
414
|
+
has: (obj) => "selected" in obj,
|
|
415
|
+
set: (obj, value) => {
|
|
416
|
+
obj.selected = value;
|
|
417
|
+
}
|
|
418
|
+
},
|
|
419
|
+
metadata: _metadata
|
|
420
|
+
}, null, _instanceExtraInitializers);
|
|
421
|
+
__esDecorate(this, null, __selected_decorators, {
|
|
422
|
+
kind: "accessor",
|
|
423
|
+
name: "_selected",
|
|
424
|
+
static: false,
|
|
425
|
+
private: false,
|
|
426
|
+
access: {
|
|
427
|
+
has: (obj) => "_selected" in obj,
|
|
428
|
+
get: (obj) => obj._selected,
|
|
429
|
+
set: (obj, value) => {
|
|
430
|
+
obj._selected = value;
|
|
431
|
+
}
|
|
432
|
+
},
|
|
433
|
+
metadata: _metadata
|
|
434
|
+
}, __selected_initializers, __selected_extraInitializers);
|
|
435
|
+
__esDecorate(this, null, _dateFilter_decorators, {
|
|
436
|
+
kind: "accessor",
|
|
437
|
+
name: "dateFilter",
|
|
438
|
+
static: false,
|
|
439
|
+
private: false,
|
|
440
|
+
access: {
|
|
441
|
+
has: (obj) => "dateFilter" in obj,
|
|
442
|
+
get: (obj) => obj.dateFilter,
|
|
443
|
+
set: (obj, value) => {
|
|
444
|
+
obj.dateFilter = value;
|
|
445
|
+
}
|
|
446
|
+
},
|
|
447
|
+
metadata: _metadata
|
|
448
|
+
}, _dateFilter_initializers, _dateFilter_extraInitializers);
|
|
449
|
+
__esDecorate(this, null, _orientation_decorators, {
|
|
450
|
+
kind: "accessor",
|
|
451
|
+
name: "orientation",
|
|
452
|
+
static: false,
|
|
453
|
+
private: false,
|
|
454
|
+
access: {
|
|
455
|
+
has: (obj) => "orientation" in obj,
|
|
456
|
+
get: (obj) => obj.orientation,
|
|
457
|
+
set: (obj, value) => {
|
|
458
|
+
obj.orientation = value;
|
|
459
|
+
}
|
|
460
|
+
},
|
|
461
|
+
metadata: _metadata
|
|
462
|
+
}, _orientation_initializers, _orientation_extraInitializers);
|
|
463
|
+
__esDecorate(this, null, _weekNumbers_decorators, {
|
|
464
|
+
kind: "accessor",
|
|
465
|
+
name: "weekNumbers",
|
|
466
|
+
static: false,
|
|
467
|
+
private: false,
|
|
468
|
+
access: {
|
|
469
|
+
has: (obj) => "weekNumbers" in obj,
|
|
470
|
+
get: (obj) => obj.weekNumbers,
|
|
471
|
+
set: (obj, value) => {
|
|
472
|
+
obj.weekNumbers = value;
|
|
473
|
+
}
|
|
474
|
+
},
|
|
475
|
+
metadata: _metadata
|
|
476
|
+
}, _weekNumbers_initializers, _weekNumbers_extraInitializers);
|
|
477
|
+
__esDecorate(this, null, __activeDate_decorators, {
|
|
478
|
+
kind: "accessor",
|
|
479
|
+
name: "_activeDate",
|
|
480
|
+
static: false,
|
|
481
|
+
private: false,
|
|
482
|
+
access: {
|
|
483
|
+
has: (obj) => "_activeDate" in obj,
|
|
484
|
+
get: (obj) => obj._activeDate,
|
|
485
|
+
set: (obj, value) => {
|
|
486
|
+
obj._activeDate = value;
|
|
487
|
+
}
|
|
488
|
+
},
|
|
489
|
+
metadata: _metadata
|
|
490
|
+
}, __activeDate_initializers, __activeDate_extraInitializers);
|
|
491
|
+
__esDecorate(this, null, _set__wide_decorators, {
|
|
492
|
+
kind: "setter",
|
|
493
|
+
name: "_wide",
|
|
494
|
+
static: false,
|
|
495
|
+
private: false,
|
|
496
|
+
access: {
|
|
497
|
+
has: (obj) => "_wide" in obj,
|
|
498
|
+
set: (obj, value) => {
|
|
499
|
+
obj._wide = value;
|
|
500
|
+
}
|
|
501
|
+
},
|
|
502
|
+
metadata: _metadata
|
|
503
|
+
}, null, _instanceExtraInitializers);
|
|
504
|
+
__esDecorate(this, null, __calendarView_decorators, {
|
|
505
|
+
kind: "accessor",
|
|
506
|
+
name: "_calendarView",
|
|
507
|
+
static: false,
|
|
508
|
+
private: false,
|
|
509
|
+
access: {
|
|
510
|
+
has: (obj) => "_calendarView" in obj,
|
|
511
|
+
get: (obj) => obj._calendarView,
|
|
512
|
+
set: (obj, value) => {
|
|
513
|
+
obj._calendarView = value;
|
|
514
|
+
}
|
|
515
|
+
},
|
|
516
|
+
metadata: _metadata
|
|
517
|
+
}, __calendarView_initializers, __calendarView_extraInitializers);
|
|
518
|
+
__esDecorate(this, null, __initialized_decorators, {
|
|
519
|
+
kind: "accessor",
|
|
520
|
+
name: "_initialized",
|
|
521
|
+
static: false,
|
|
522
|
+
private: false,
|
|
523
|
+
access: {
|
|
524
|
+
has: (obj) => "_initialized" in obj,
|
|
525
|
+
get: (obj) => obj._initialized,
|
|
526
|
+
set: (obj, value) => {
|
|
527
|
+
obj._initialized = value;
|
|
528
|
+
}
|
|
529
|
+
},
|
|
530
|
+
metadata: _metadata
|
|
531
|
+
}, __initialized_initializers, __initialized_extraInitializers);
|
|
532
|
+
if (_metadata) Object.defineProperty(this, Symbol.metadata, {
|
|
533
|
+
enumerable: true,
|
|
534
|
+
configurable: true,
|
|
535
|
+
writable: true,
|
|
536
|
+
value: _metadata
|
|
537
|
+
});
|
|
538
|
+
}
|
|
539
|
+
static {
|
|
540
|
+
this.elementName = "sbb-calendar";
|
|
541
|
+
}
|
|
542
|
+
static {
|
|
543
|
+
this.styles = [boxSizingStyles, calendar_default];
|
|
544
|
+
}
|
|
545
|
+
static {
|
|
546
|
+
this.events = {
|
|
547
|
+
dateselected: "dateselected",
|
|
548
|
+
monthchange: "monthchange"
|
|
549
|
+
};
|
|
550
|
+
}
|
|
551
|
+
#wide_accessor_storage;
|
|
552
|
+
/** If set to true, two months are displayed */
|
|
553
|
+
get wide() {
|
|
554
|
+
return this.#wide_accessor_storage;
|
|
555
|
+
}
|
|
556
|
+
set wide(value) {
|
|
557
|
+
this.#wide_accessor_storage = value;
|
|
558
|
+
}
|
|
559
|
+
#view_accessor_storage;
|
|
560
|
+
/** The initial view of the calendar which should be displayed on opening. */
|
|
561
|
+
get view() {
|
|
562
|
+
return this.#view_accessor_storage;
|
|
563
|
+
}
|
|
564
|
+
set view(value) {
|
|
565
|
+
this.#view_accessor_storage = value;
|
|
566
|
+
}
|
|
567
|
+
#min_accessor_storage;
|
|
568
|
+
/**
|
|
569
|
+
* The minimum valid date. Accepts a date object or null.
|
|
570
|
+
* Accepts an ISO8601 formatted string (e.g. 2024-12-24) as attribute.
|
|
571
|
+
*/
|
|
572
|
+
get min() {
|
|
573
|
+
return this.#min_accessor_storage;
|
|
574
|
+
}
|
|
575
|
+
set min(value) {
|
|
576
|
+
this.#min_accessor_storage = value;
|
|
577
|
+
}
|
|
578
|
+
#max_accessor_storage;
|
|
579
|
+
/**
|
|
580
|
+
* The maximum valid date. Accepts a date object or null.
|
|
581
|
+
* Accepts an ISO8601 formatted string (e.g. 2024-12-24) as attribute.
|
|
582
|
+
*/
|
|
583
|
+
get max() {
|
|
584
|
+
return this.#max_accessor_storage;
|
|
585
|
+
}
|
|
586
|
+
set max(value) {
|
|
587
|
+
this.#max_accessor_storage = value;
|
|
588
|
+
}
|
|
589
|
+
#multiple_accessor_storage;
|
|
590
|
+
/** Whether the calendar allows for multiple date selection. */
|
|
591
|
+
get multiple() {
|
|
592
|
+
return this.#multiple_accessor_storage;
|
|
593
|
+
}
|
|
594
|
+
set multiple(value) {
|
|
595
|
+
this.#multiple_accessor_storage = value;
|
|
596
|
+
}
|
|
597
|
+
/**
|
|
598
|
+
* The selected date: accepts a date object, or, if `multiple`, an array of dates.
|
|
599
|
+
*/
|
|
600
|
+
set selected(value) {
|
|
601
|
+
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));
|
|
602
|
+
else {
|
|
603
|
+
const selectedDate = this._dateAdapter.getValidDateOrNull(this._dateAdapter.deserialize(value));
|
|
604
|
+
if (!!selectedDate && (!this._isDayInRange(this._dateAdapter.toIso8601(selectedDate)) || this._dateFilter(selectedDate))) this._selected = selectedDate;
|
|
605
|
+
else this._selected = null;
|
|
606
|
+
}
|
|
607
|
+
}
|
|
608
|
+
get selected() {
|
|
609
|
+
return this._selected;
|
|
610
|
+
}
|
|
611
|
+
#_selected_accessor_storage;
|
|
612
|
+
get _selected() {
|
|
613
|
+
return this.#_selected_accessor_storage;
|
|
614
|
+
}
|
|
615
|
+
set _selected(value) {
|
|
616
|
+
this.#_selected_accessor_storage = value;
|
|
617
|
+
}
|
|
618
|
+
#dateFilter_accessor_storage;
|
|
619
|
+
/** A function used to filter out dates. */
|
|
620
|
+
get dateFilter() {
|
|
621
|
+
return this.#dateFilter_accessor_storage;
|
|
622
|
+
}
|
|
623
|
+
set dateFilter(value) {
|
|
624
|
+
this.#dateFilter_accessor_storage = value;
|
|
625
|
+
}
|
|
626
|
+
#orientation_accessor_storage;
|
|
627
|
+
/** The orientation of days in the calendar. */
|
|
628
|
+
get orientation() {
|
|
629
|
+
return this.#orientation_accessor_storage;
|
|
630
|
+
}
|
|
631
|
+
set orientation(value) {
|
|
632
|
+
this.#orientation_accessor_storage = value;
|
|
633
|
+
}
|
|
634
|
+
#weekNumbers_accessor_storage;
|
|
635
|
+
/** Whether it has to display the week numbers in addition to week days. */
|
|
636
|
+
get weekNumbers() {
|
|
637
|
+
return this.#weekNumbers_accessor_storage;
|
|
638
|
+
}
|
|
639
|
+
set weekNumbers(value) {
|
|
640
|
+
this.#weekNumbers_accessor_storage = value;
|
|
641
|
+
}
|
|
642
|
+
#_activeDate_accessor_storage;
|
|
643
|
+
/** The currently active date. */
|
|
644
|
+
get _activeDate() {
|
|
645
|
+
return this.#_activeDate_accessor_storage;
|
|
646
|
+
}
|
|
647
|
+
set _activeDate(value) {
|
|
648
|
+
this.#_activeDate_accessor_storage = value;
|
|
649
|
+
}
|
|
650
|
+
/** The current wide property considering property value and breakpoints. From zero to small `wide` has always to be false. */
|
|
651
|
+
set _wide(wide) {
|
|
652
|
+
this.toggleState("wide", wide);
|
|
653
|
+
this._wideInternal = wide;
|
|
654
|
+
}
|
|
655
|
+
get _wide() {
|
|
656
|
+
return this._wideInternal;
|
|
657
|
+
}
|
|
658
|
+
#_calendarView_accessor_storage;
|
|
659
|
+
get _calendarView() {
|
|
660
|
+
return this.#_calendarView_accessor_storage;
|
|
661
|
+
}
|
|
662
|
+
set _calendarView(value) {
|
|
663
|
+
this.#_calendarView_accessor_storage = value;
|
|
664
|
+
}
|
|
665
|
+
/** A list of calendar's cells corresponding to days, months or years depending on the view. */
|
|
666
|
+
get _cells() {
|
|
667
|
+
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}`)) ?? []);
|
|
668
|
+
}
|
|
669
|
+
#_initialized_accessor_storage;
|
|
670
|
+
get _initialized() {
|
|
671
|
+
return this.#_initialized_accessor_storage;
|
|
672
|
+
}
|
|
673
|
+
set _initialized(value) {
|
|
674
|
+
this.#_initialized_accessor_storage = value;
|
|
675
|
+
}
|
|
676
|
+
constructor() {
|
|
677
|
+
super();
|
|
678
|
+
this.#wide_accessor_storage = (__runInitializers(this, _instanceExtraInitializers), __runInitializers(this, _wide_initializers, false));
|
|
679
|
+
this.#view_accessor_storage = (__runInitializers(this, _wide_extraInitializers), __runInitializers(this, _view_initializers, "day"));
|
|
680
|
+
this.#min_accessor_storage = (__runInitializers(this, _view_extraInitializers), __runInitializers(this, _min_initializers, null));
|
|
681
|
+
this.#max_accessor_storage = (__runInitializers(this, _min_extraInitializers), __runInitializers(this, _max_initializers, null));
|
|
682
|
+
this.#multiple_accessor_storage = (__runInitializers(this, _max_extraInitializers), __runInitializers(this, _multiple_initializers, false));
|
|
683
|
+
this.#_selected_accessor_storage = (__runInitializers(this, _multiple_extraInitializers), __runInitializers(this, __selected_initializers, null));
|
|
684
|
+
this.#dateFilter_accessor_storage = (__runInitializers(this, __selected_extraInitializers), __runInitializers(this, _dateFilter_initializers, null));
|
|
685
|
+
this.#orientation_accessor_storage = (__runInitializers(this, _dateFilter_extraInitializers), __runInitializers(this, _orientation_initializers, "horizontal"));
|
|
686
|
+
this.#weekNumbers_accessor_storage = (__runInitializers(this, _orientation_extraInitializers), __runInitializers(this, _weekNumbers_initializers, false));
|
|
687
|
+
this._dateAdapter = (__runInitializers(this, _weekNumbers_extraInitializers), readConfig().datetime?.dateAdapter ?? defaultDateAdapter);
|
|
688
|
+
this.#_activeDate_accessor_storage = __runInitializers(this, __activeDate_initializers, this._dateAdapter.today());
|
|
689
|
+
this._wideInternal = (__runInitializers(this, __activeDate_extraInitializers), false);
|
|
690
|
+
this.#_calendarView_accessor_storage = __runInitializers(this, __calendarView_initializers, "day");
|
|
691
|
+
this._nextCalendarView = (__runInitializers(this, __calendarView_extraInitializers), "day");
|
|
692
|
+
/** Information about the rendered day view; used in keyboard navigation. */
|
|
693
|
+
this._keyboardNavigationDayViewParameters = {
|
|
694
|
+
firstDayInView: null,
|
|
695
|
+
lastDayInView: null,
|
|
696
|
+
firstMonthOffset: 0,
|
|
697
|
+
firstMonthLength: 0,
|
|
698
|
+
secondMonthOffset: 0
|
|
699
|
+
};
|
|
700
|
+
/** Grid of calendar cells representing the dates of the month. */
|
|
701
|
+
this._weeks = [];
|
|
702
|
+
/** An array containing all the month names in the current language. */
|
|
703
|
+
this._monthNames = this._dateAdapter.getMonthNames("long");
|
|
704
|
+
this._enhancedVariant = false;
|
|
705
|
+
/** Whether the focus should be reset on focusCell. */
|
|
706
|
+
this._resetFocus = false;
|
|
707
|
+
/** Whether an element inside the calendar is currently focused. */
|
|
708
|
+
this._containingFocus = false;
|
|
709
|
+
this.#_initialized_accessor_storage = __runInitializers(this, __initialized_initializers, false);
|
|
710
|
+
this._language = (__runInitializers(this, __initialized_extraInitializers), new SbbLanguageController(this).withHandler(() => {
|
|
711
|
+
this._monthNames = this._dateAdapter.getMonthNames("long");
|
|
712
|
+
this._createMonthRows();
|
|
713
|
+
}));
|
|
714
|
+
this._mediaMatcher = new SbbMediaMatcherController(this, { [SbbMediaQueryBreakpointLargeAndAbove]: () => this._init() });
|
|
715
|
+
this._onSlotChange = () => {
|
|
716
|
+
this._enhancedVariant = Array.from(this.children).some((c) => c.localName === "sbb-calendar-day");
|
|
717
|
+
this.toggleState("enhanced", this._enhancedVariant);
|
|
718
|
+
this._setTabIndex();
|
|
719
|
+
};
|
|
720
|
+
this._createMonthRows();
|
|
721
|
+
this._setWeekdays();
|
|
722
|
+
this.addEventListener("focusin", () => this._containingFocus = true);
|
|
723
|
+
this.addEventListener("focusout", () => this._containingFocus = false);
|
|
724
|
+
this.addEventListener("click", (e) => {
|
|
725
|
+
const day = e.target.closest("sbb-calendar-day");
|
|
726
|
+
if (day) this._selectDate(day.value);
|
|
727
|
+
});
|
|
728
|
+
this.addEventListener("keydown", (e) => {
|
|
729
|
+
if (e.target.localName === "sbb-calendar-day") this._handleKeyboardEvent(e, this._mapDateToDay(e.target.value));
|
|
730
|
+
});
|
|
731
|
+
}
|
|
732
|
+
_dateFilter(date) {
|
|
733
|
+
return this.dateFilter?.(date) ?? true;
|
|
734
|
+
}
|
|
735
|
+
/** Resets the active month according to the new state of the calendar. */
|
|
736
|
+
resetPosition() {
|
|
737
|
+
this._resetCalendarView();
|
|
738
|
+
this._init();
|
|
739
|
+
}
|
|
740
|
+
connectedCallback() {
|
|
741
|
+
super.connectedCallback();
|
|
742
|
+
this.resetPosition();
|
|
743
|
+
this.shadowRoot?.addEventListener("slotchange", this._onSlotChange, { capture: true });
|
|
744
|
+
}
|
|
745
|
+
disconnectedCallback() {
|
|
746
|
+
super.disconnectedCallback();
|
|
747
|
+
this.shadowRoot?.removeEventListener("slotchange", this._onSlotChange, { capture: true });
|
|
748
|
+
}
|
|
749
|
+
/** @internal */
|
|
750
|
+
focus() {
|
|
751
|
+
this._resetFocus = true;
|
|
752
|
+
this._focusCell();
|
|
753
|
+
}
|
|
754
|
+
willUpdate(changedProperties) {
|
|
755
|
+
super.willUpdate(changedProperties);
|
|
756
|
+
if (!this._initialized) return;
|
|
757
|
+
if (changedProperties.has("wide") || changedProperties.has("orientation")) this.resetPosition();
|
|
758
|
+
if (changedProperties.has("view")) {
|
|
759
|
+
this._setChosenYear();
|
|
760
|
+
this._chosenMonth = void 0;
|
|
761
|
+
this._nextCalendarView = this._calendarView = this.view;
|
|
762
|
+
}
|
|
763
|
+
}
|
|
764
|
+
updated(changedProperties) {
|
|
765
|
+
super.updated(changedProperties);
|
|
766
|
+
this._setTabIndex();
|
|
767
|
+
this._focusCell();
|
|
768
|
+
}
|
|
769
|
+
/**
|
|
770
|
+
* The `_selected` state should be adapted when the `multiple` property changes:
|
|
771
|
+
* - if it changes to true, the '_selected' is set to an array;
|
|
772
|
+
* - if it changes to false, the first available option is set as 'value' otherwise it's set to null.
|
|
773
|
+
*/
|
|
774
|
+
_onMultipleChanged(isMultiple) {
|
|
775
|
+
if (isMultiple && !Array.isArray(this._selected)) this._selected = this._selected ? [this._selected] : [];
|
|
776
|
+
if (!isMultiple && Array.isArray(this._selected)) this._selected = this._selected.length ? this._selected[0] : null;
|
|
777
|
+
}
|
|
778
|
+
/** Initializes the component. */
|
|
779
|
+
_init(activeDate) {
|
|
780
|
+
if (isServer) return;
|
|
781
|
+
else if (this.hydrationRequired) {
|
|
782
|
+
this.hydrationComplete.then(() => this._init());
|
|
783
|
+
return;
|
|
784
|
+
}
|
|
785
|
+
if (activeDate) this._assignActiveDate(activeDate);
|
|
786
|
+
this._wide = (this._mediaMatcher.matches(SbbMediaQueryBreakpointLargeAndAbove) ?? false) && this.wide;
|
|
787
|
+
this._weeks = this._createWeekRows(this._activeDate);
|
|
788
|
+
this._years = this._createYearRows();
|
|
789
|
+
this._weekNumbers = this._weeks.flat().sort((a, b) => a.value.localeCompare(b.value)).map((day) => day.weekValue).filter((v, i, a) => a.indexOf(v) === i);
|
|
790
|
+
this._nextMonthWeeks = [[]];
|
|
791
|
+
this._nextMonthYears = [[]];
|
|
792
|
+
if (this._wide) {
|
|
793
|
+
const nextMonthDate = this._dateAdapter.addCalendarMonths(this._activeDate, 1);
|
|
794
|
+
this._nextMonthWeeks = this._createWeekRows(nextMonthDate, true);
|
|
795
|
+
this._nextMonthYears = this._createYearRows(YEARS_PER_PAGE);
|
|
796
|
+
this._nextMonthWeekNumbers = this._nextMonthWeeks.flat().sort((a, b) => a.value.localeCompare(b.value)).map((day) => day.weekValue).filter((v, i, a) => a.indexOf(v) === i);
|
|
797
|
+
}
|
|
798
|
+
this._initialized = true;
|
|
799
|
+
}
|
|
800
|
+
/** Focuses on a day cell prioritizing the selected day, the current day, and lastly, the first selectable day. */
|
|
801
|
+
_focusCell() {
|
|
802
|
+
if (this._resetFocus) {
|
|
803
|
+
this._getFirstFocusable()?.focus();
|
|
804
|
+
this._resetFocus = false;
|
|
805
|
+
}
|
|
806
|
+
}
|
|
807
|
+
/** Creates the array of weekdays. */
|
|
808
|
+
_setWeekdays() {
|
|
809
|
+
const narrowWeekdays = this._dateAdapter.getDayOfWeekNames("narrow");
|
|
810
|
+
const weekdays = this._dateAdapter.getDayOfWeekNames("long").map((long, i) => ({
|
|
811
|
+
long,
|
|
812
|
+
narrow: narrowWeekdays[i]
|
|
813
|
+
}));
|
|
814
|
+
const firstDayOfWeek = this._dateAdapter.getFirstDayOfWeek();
|
|
815
|
+
this._weekdays = weekdays.slice(firstDayOfWeek).concat(weekdays.slice(0, firstDayOfWeek));
|
|
816
|
+
}
|
|
817
|
+
/** Creates the rows along the horizontal direction and sets the parameters used in keyboard navigation. */
|
|
818
|
+
_createWeekRows(value, isSecondMonthInView = false) {
|
|
819
|
+
const daysInMonth = this._dateAdapter.getNumDaysInMonth(value);
|
|
820
|
+
const weekOffset = this._dateAdapter.getFirstWeekOffset(value);
|
|
821
|
+
if (!isSecondMonthInView) {
|
|
822
|
+
this._keyboardNavigationDayViewParameters.firstMonthLength = daysInMonth;
|
|
823
|
+
this._keyboardNavigationDayViewParameters.firstMonthOffset = weekOffset;
|
|
824
|
+
this._keyboardNavigationDayViewParameters.firstDayInView = this._dateAdapter.toIso8601(this._dateAdapter.createDate(this._dateAdapter.getYear(value), this._dateAdapter.getMonth(value), 1));
|
|
825
|
+
this._keyboardNavigationDayViewParameters.lastDayInView = this._dateAdapter.toIso8601(this._dateAdapter.createDate(this._dateAdapter.getYear(value), this._dateAdapter.getMonth(value), daysInMonth));
|
|
826
|
+
} else {
|
|
827
|
+
this._keyboardNavigationDayViewParameters.secondMonthOffset = weekOffset;
|
|
828
|
+
this._keyboardNavigationDayViewParameters.lastDayInView = this._dateAdapter.toIso8601(this._dateAdapter.createDate(this._dateAdapter.getYear(value), this._dateAdapter.getMonth(value), daysInMonth));
|
|
829
|
+
}
|
|
830
|
+
return this.orientation === "horizontal" ? this._createWeekRowsHorizontal(value, daysInMonth, weekOffset) : this._createWeekRowsVertical(value, daysInMonth, weekOffset);
|
|
831
|
+
}
|
|
832
|
+
/**
|
|
833
|
+
* Creates the rows for each week in orientation='horizontal'.
|
|
834
|
+
*
|
|
835
|
+
* Iterates through the days of the months, creates a Day object for each and pushes it into and array.
|
|
836
|
+
* Each seven days (considering the offset at the beginning of the month) restarts from an empty array.
|
|
837
|
+
*
|
|
838
|
+
* The result is a matrix in which every row is a week (or part of it, considering offset).
|
|
839
|
+
*/
|
|
840
|
+
_createWeekRowsHorizontal(value, daysInMonth, weekOffset) {
|
|
841
|
+
const weeks = [[]];
|
|
842
|
+
for (let i = 0, cell = weekOffset; i < daysInMonth; i++, cell++) {
|
|
843
|
+
if (cell === DAYS_PER_ROW) {
|
|
844
|
+
weeks.push([]);
|
|
845
|
+
cell = 0;
|
|
846
|
+
}
|
|
847
|
+
const date = this._dateAdapter.createDate(this._dateAdapter.getYear(value), this._dateAdapter.getMonth(value), i + 1);
|
|
848
|
+
weeks[weeks.length - 1].push(this._mapDateToDay(date));
|
|
849
|
+
}
|
|
850
|
+
return weeks;
|
|
851
|
+
}
|
|
852
|
+
/**
|
|
853
|
+
* Creates the rows for each week in orientation='vertical'.
|
|
854
|
+
*
|
|
855
|
+
* Creates a matrix with seven empty rows.
|
|
856
|
+
* Iterates through the days of the months, creates a Day object for each
|
|
857
|
+
* and pushes it into the correct array considering the offset at the beginning of the month.
|
|
858
|
+
* Each seven days (including offset) restarts from the first.
|
|
859
|
+
*
|
|
860
|
+
* The result is a matrix in which every row is a set of weekdays, so:
|
|
861
|
+
* - row 0: all the Mondays;
|
|
862
|
+
* - row 1: all the Tuesdays;
|
|
863
|
+
* - ...
|
|
864
|
+
* - row 7: all the Sundays.
|
|
865
|
+
*/
|
|
866
|
+
_createWeekRowsVertical(value, daysInMonth, weekOffset) {
|
|
867
|
+
const weeks = Array.from({ length: DAYS_PER_ROW }, () => []);
|
|
868
|
+
for (let i = 0, cell = weekOffset; i < daysInMonth; i++, cell++) {
|
|
869
|
+
if (cell === DAYS_PER_ROW) cell = 0;
|
|
870
|
+
const date = this._dateAdapter.createDate(this._dateAdapter.getYear(value), this._dateAdapter.getMonth(value), i + 1);
|
|
871
|
+
weeks[cell].push(this._mapDateToDay(date));
|
|
872
|
+
}
|
|
873
|
+
return weeks;
|
|
874
|
+
}
|
|
875
|
+
_mapDateToDay(date) {
|
|
876
|
+
return {
|
|
877
|
+
value: this._dateAdapter.toIso8601(date),
|
|
878
|
+
dateValue: date,
|
|
879
|
+
dayValue: String(this._dateAdapter.getDate(date)),
|
|
880
|
+
monthValue: String(this._dateAdapter.getMonth(date)),
|
|
881
|
+
yearValue: String(this._dateAdapter.getYear(date)),
|
|
882
|
+
weekValue: this._getWeek(date),
|
|
883
|
+
weekDayValue: this._dateAdapter.getDayOfWeek(date)
|
|
884
|
+
};
|
|
885
|
+
}
|
|
886
|
+
_getWeek(date) {
|
|
887
|
+
const firstDayOfYear = this._dateAdapter.createDate(this._dateAdapter.getYear(date), 1, 1);
|
|
888
|
+
const weekday = this._dateAdapter.getDayOfWeek(firstDayOfYear);
|
|
889
|
+
let weekIndex = [
|
|
890
|
+
MONDAY,
|
|
891
|
+
TUESDAY,
|
|
892
|
+
WEDNESDAY,
|
|
893
|
+
THURSDAY
|
|
894
|
+
].includes(weekday) ? 1 : 0;
|
|
895
|
+
let weekStart = this._dateAdapter.addCalendarDays(firstDayOfYear, this._dateAdapter.getFirstWeekOffset(firstDayOfYear) * -1);
|
|
896
|
+
while (this._dateAdapter.compareDate(weekStart, date) <= 0) {
|
|
897
|
+
const weekEnd = this._dateAdapter.addCalendarDays(weekStart, 6);
|
|
898
|
+
if (this._dateAdapter.compareDate(date, weekEnd) <= 0) return weekIndex > 0 ? weekIndex : this._getWeek(this._dateAdapter.addCalendarDays(firstDayOfYear, -1));
|
|
899
|
+
weekStart = this._dateAdapter.addCalendarDays(weekStart, DAYS_PER_ROW);
|
|
900
|
+
weekIndex++;
|
|
901
|
+
}
|
|
902
|
+
throw new Error("The provided date is invalid");
|
|
903
|
+
}
|
|
904
|
+
/** Force the conversion to ISO8601 formatted string. */
|
|
905
|
+
_mapValueToISODate(value) {
|
|
906
|
+
return typeof value === "string" ? value : this._dateAdapter.toIso8601(value);
|
|
907
|
+
}
|
|
908
|
+
/** Creates the rows for the month selection view. */
|
|
909
|
+
_createMonthRows() {
|
|
910
|
+
const months = new Array(12).fill(null).map((_, i) => ({
|
|
911
|
+
value: String(i + 1).padStart(2, "0"),
|
|
912
|
+
monthValue: i + 1
|
|
913
|
+
}));
|
|
914
|
+
const rows = 12 / MONTHS_PER_ROW;
|
|
915
|
+
const monthArray = [];
|
|
916
|
+
for (let i = 0; i < rows; i++) monthArray.push(months.slice(MONTHS_PER_ROW * i, MONTHS_PER_ROW * (i + 1)));
|
|
917
|
+
this._months = monthArray;
|
|
918
|
+
}
|
|
919
|
+
/** Creates the rows for the year selection view. */
|
|
920
|
+
_createYearRows(offset = 0) {
|
|
921
|
+
const startValueYearView = this._getStartValueYearView();
|
|
922
|
+
const allYears = new Array(YEARS_PER_PAGE).fill(0).map((_, i) => startValueYearView + offset + i);
|
|
923
|
+
const rows = YEARS_PER_PAGE / YEARS_PER_ROW;
|
|
924
|
+
const yearArray = [];
|
|
925
|
+
for (let i = 0; i < rows; i++) yearArray.push(allYears.slice(YEARS_PER_ROW * i, YEARS_PER_ROW * (i + 1)));
|
|
926
|
+
return yearArray;
|
|
927
|
+
}
|
|
928
|
+
/**
|
|
929
|
+
* Calculates the first year that will be shown in the year selection panel.
|
|
930
|
+
* If `minDate` and `maxDate` are both null, the starting year is calculated as
|
|
931
|
+
* the multiple of YEARS_PER_PAGE closest to and less than activeDate,
|
|
932
|
+
* e.g., with `YEARS_PER_PAGE` = 24 and `activeDate` = 2020, the function will return 2016 (24 * 83),
|
|
933
|
+
* while with `activeDate` = 2000, the function will return 1992 (24 * 82).
|
|
934
|
+
* If `minDate` is not null, it returns the corresponding year; if `maxDate` is not null,
|
|
935
|
+
* it returns the corresponding year minus `YEARS_PER_PAGE`, so that the `maxDate` is the last rendered year.
|
|
936
|
+
* If both are not null, `maxDate` has priority over `minDate`.
|
|
937
|
+
*/
|
|
938
|
+
_getStartValueYearView() {
|
|
939
|
+
let startingYear = 0;
|
|
940
|
+
if (this.max) startingYear = this._dateAdapter.getYear(this.max) - YEARS_PER_PAGE + 1;
|
|
941
|
+
else if (this.min) startingYear = this._dateAdapter.getYear(this.min);
|
|
942
|
+
const activeYear = this._dateAdapter.getYear(this._activeDate);
|
|
943
|
+
return activeYear - ((activeYear - startingYear) % YEARS_PER_PAGE + YEARS_PER_PAGE) % YEARS_PER_PAGE;
|
|
944
|
+
}
|
|
945
|
+
/** Checks if date is within the min-max range. */
|
|
946
|
+
_isDayInRange(dateString) {
|
|
947
|
+
if (!this.min && !this.max) return true;
|
|
948
|
+
const date = this._dateAdapter.deserialize(dateString);
|
|
949
|
+
return this._dateAdapter.sameDate(date, this._dateAdapter.clampDate(date, this.min, this.max));
|
|
950
|
+
}
|
|
951
|
+
/** Emits the selected date and sets it internally. */
|
|
952
|
+
_selectDate(day) {
|
|
953
|
+
this._chosenMonth = void 0;
|
|
954
|
+
this._setChosenYear();
|
|
955
|
+
if (this.multiple) {
|
|
956
|
+
if (this._selected && this._selected.length > 0) {
|
|
957
|
+
const indexOfSelectedDay = this._selected.findIndex((sel) => this._dateAdapter.compareDate(sel, day) === 0);
|
|
958
|
+
if (indexOfSelectedDay !== -1) this._selected = this._selected.filter((_, i) => i !== indexOfSelectedDay);
|
|
959
|
+
else this._selected = [...this._selected, day];
|
|
960
|
+
} else this._selected = [day];
|
|
961
|
+
this._emitDateSelectedEvent(this._selected.map((e) => this._dateAdapter.deserialize(e)));
|
|
962
|
+
} else if (!this._selected || this._dateAdapter.compareDate(this._selected, day) !== 0) {
|
|
963
|
+
this._selected = day;
|
|
964
|
+
this._emitDateSelectedEvent(this._dateAdapter.deserialize(day));
|
|
965
|
+
}
|
|
966
|
+
}
|
|
967
|
+
/**
|
|
968
|
+
* Handle multiple dates selection via weekNumber / weekDay buttons:
|
|
969
|
+
* - if Cmd or Ctrl are pressed, add the new date to the current ones;
|
|
970
|
+
* - if not,
|
|
971
|
+
* - if the new dates are the same of the current ones, it means that the same button has been clicked twice, so do nothing;
|
|
972
|
+
* - if not, the selected dates are the new ones.
|
|
973
|
+
*/
|
|
974
|
+
_selectMultipleDates(days) {
|
|
975
|
+
const enabledDays = this._cells.filter((e) => !e.disabled).map((e) => this._mapValueToISODate(e.value));
|
|
976
|
+
const daysToAdd = days.map((e) => e.value).filter((isoDate) => enabledDays.includes(isoDate));
|
|
977
|
+
const daysToAddSet = new Set(daysToAdd);
|
|
978
|
+
const selectedSet = new Set(this._selected.map((s) => this._dateAdapter.toIso8601(s)));
|
|
979
|
+
this._selected = this._updateSelectedWithMultipleDates(daysToAdd, daysToAddSet, selectedSet).map((s) => this._dateAdapter.deserialize(s));
|
|
980
|
+
this._emitDateSelectedEvent(this._selected.map((e) => this._dateAdapter.deserialize(e)));
|
|
981
|
+
}
|
|
982
|
+
/**
|
|
983
|
+
* Emits the dateselected event given the detail (as T or T[] based on the value of the multiple flag).
|
|
984
|
+
*/
|
|
985
|
+
_emitDateSelectedEvent(detail) {
|
|
986
|
+
/** @type {CustomEvent<T | T[]>} Event emitted on date selection. */
|
|
987
|
+
this.dispatchEvent(new CustomEvent("dateselected", {
|
|
988
|
+
detail,
|
|
989
|
+
composed: true,
|
|
990
|
+
bubbles: true
|
|
991
|
+
}));
|
|
992
|
+
}
|
|
993
|
+
_emitMonthChange() {
|
|
994
|
+
const monthchange = (this.wide ? [...this._weeks, ...this._nextMonthWeeks] : this._weeks).flat().sort((a, b) => a.value.localeCompare(b.value));
|
|
995
|
+
/**
|
|
996
|
+
* @type {SbbMonthChangeEvent}
|
|
997
|
+
* Emits when the month changes.
|
|
998
|
+
* The `range` property contains the days array of the chosen month.
|
|
999
|
+
*/
|
|
1000
|
+
this.dispatchEvent(new SbbMonthChangeEvent(monthchange));
|
|
1001
|
+
}
|
|
1002
|
+
/**
|
|
1003
|
+
* In case of multiple selection, newly added days must be added to the existing ones, without duplication.
|
|
1004
|
+
* If the days to add are exactly the same as the selected ones, the set must be emptied.
|
|
1005
|
+
*/
|
|
1006
|
+
_updateSelectedWithMultipleDates(daysToAdd, daysToAddSet, selectedSet) {
|
|
1007
|
+
if (daysToAdd.every((day) => selectedSet.has(day))) daysToAddSet.forEach((day) => selectedSet.delete(day));
|
|
1008
|
+
else daysToAddSet.forEach((day) => selectedSet.add(day));
|
|
1009
|
+
return Array.from(selectedSet);
|
|
1010
|
+
}
|
|
1011
|
+
_setChosenYear() {
|
|
1012
|
+
if (this.view === "month") {
|
|
1013
|
+
let selectedDate;
|
|
1014
|
+
if (this.multiple) selectedDate = this.selected.at(-1);
|
|
1015
|
+
else selectedDate = this.selected;
|
|
1016
|
+
this._chosenYear = this._dateAdapter.getYear(selectedDate ?? this._dateAdapter.today());
|
|
1017
|
+
} else this._chosenYear = void 0;
|
|
1018
|
+
}
|
|
1019
|
+
_assignActiveDate(date) {
|
|
1020
|
+
if (this.min && this._dateAdapter.compareDate(this.min, date) > 0) {
|
|
1021
|
+
this._activeDate = this.min;
|
|
1022
|
+
return;
|
|
1023
|
+
}
|
|
1024
|
+
if (this.max && this._dateAdapter.compareDate(this.max, date) < 0) {
|
|
1025
|
+
this._activeDate = this.max;
|
|
1026
|
+
return;
|
|
1027
|
+
}
|
|
1028
|
+
this._activeDate = date;
|
|
1029
|
+
}
|
|
1030
|
+
/** Goes to the month identified by the shift. */
|
|
1031
|
+
_goToDifferentMonth(months) {
|
|
1032
|
+
this._init(this._dateAdapter.addCalendarMonths(this._activeDate, months));
|
|
1033
|
+
this._emitMonthChange();
|
|
1034
|
+
}
|
|
1035
|
+
_goToDifferentYear(years) {
|
|
1036
|
+
this._chosenYear += years;
|
|
1037
|
+
this._activeDate = this._dateAdapter.createDate(this._chosenYear, this._dateAdapter.getMonth(this._activeDate), this._dateAdapter.getDate(this._activeDate));
|
|
1038
|
+
this._init();
|
|
1039
|
+
}
|
|
1040
|
+
_goToDifferentYearRange(years) {
|
|
1041
|
+
this._init(this._dateAdapter.addCalendarYears(this._activeDate, years));
|
|
1042
|
+
}
|
|
1043
|
+
_prevDisabled(prevDate) {
|
|
1044
|
+
if (!this.min) return false;
|
|
1045
|
+
return this._dateAdapter.compareDate(prevDate, this.min) < 0;
|
|
1046
|
+
}
|
|
1047
|
+
_nextDisabled(nextDate) {
|
|
1048
|
+
if (!this.max) return false;
|
|
1049
|
+
return this._dateAdapter.compareDate(nextDate, this.max) > 0;
|
|
1050
|
+
}
|
|
1051
|
+
/** Checks if the "previous month" button should be disabled. */
|
|
1052
|
+
_previousMonthDisabled() {
|
|
1053
|
+
const prevMonth = this._dateAdapter.addCalendarDays(this._activeDate, this._dateAdapter.getDate(this._activeDate) * -1);
|
|
1054
|
+
return this._prevDisabled(prevMonth);
|
|
1055
|
+
}
|
|
1056
|
+
/** Checks if the "next month" button should be disabled. */
|
|
1057
|
+
_nextMonthDisabled() {
|
|
1058
|
+
let nextMonth = this._dateAdapter.addCalendarMonths(this._activeDate, this._wide ? 2 : 1);
|
|
1059
|
+
nextMonth = this._dateAdapter.createDate(this._dateAdapter.getYear(nextMonth), this._dateAdapter.getMonth(nextMonth), 1);
|
|
1060
|
+
return this._nextDisabled(nextMonth);
|
|
1061
|
+
}
|
|
1062
|
+
/** Checks if the "previous year" button should be disabled. */
|
|
1063
|
+
_previousYearDisabled() {
|
|
1064
|
+
const prevYear = this._dateAdapter.createDate(this._dateAdapter.getYear(this._activeDate) - 1, 12, 31);
|
|
1065
|
+
return this._prevDisabled(prevYear);
|
|
1066
|
+
}
|
|
1067
|
+
/** Checks if the "next year" button should be disabled. */
|
|
1068
|
+
_nextYearDisabled() {
|
|
1069
|
+
const nextYear = this._dateAdapter.createDate(this._dateAdapter.getYear(this._activeDate) + (this._wide ? 2 : 1), 1, 1);
|
|
1070
|
+
return this._nextDisabled(nextYear);
|
|
1071
|
+
}
|
|
1072
|
+
/** Checks if the "previous year" button should be disabled in year view. */
|
|
1073
|
+
_previousYearRangeDisabled() {
|
|
1074
|
+
const prevYear = this._dateAdapter.createDate(this._years[0][0] - 1, 12, 31);
|
|
1075
|
+
return this._prevDisabled(prevYear);
|
|
1076
|
+
}
|
|
1077
|
+
/** Checks if the "next year" button should be disabled in year view. */
|
|
1078
|
+
_nextYearRangeDisabled() {
|
|
1079
|
+
const years = this._wide ? this._nextMonthYears : this._years;
|
|
1080
|
+
const lastYearRange = years[years.length - 1];
|
|
1081
|
+
const lastYear = lastYearRange[lastYearRange.length - 1];
|
|
1082
|
+
const nextYear = this._dateAdapter.createDate(lastYear + 1, 1, 1);
|
|
1083
|
+
return this._nextDisabled(nextYear);
|
|
1084
|
+
}
|
|
1085
|
+
_handleTableBlur(eventTarget) {
|
|
1086
|
+
if (eventTarget?.localName !== "sbb-calendar-day") this._setTabIndex();
|
|
1087
|
+
}
|
|
1088
|
+
_setTabIndex() {
|
|
1089
|
+
Array.from(this._cells.filter((e) => e.tabIndex === 0) ?? []).forEach((day) => day.tabIndex = -1);
|
|
1090
|
+
const firstFocusable = this._getFirstFocusable();
|
|
1091
|
+
if (firstFocusable) firstFocusable.tabIndex = 0;
|
|
1092
|
+
}
|
|
1093
|
+
/** Get the element in the calendar to assign focus. */
|
|
1094
|
+
_getFirstFocusable() {
|
|
1095
|
+
if (this._calendarView === "day") {
|
|
1096
|
+
const selectedOrCurrent = this._cells.find((e) => e.matches(":is(:state(selected),[state--selected])")) ?? this._cells.find((e) => e.matches(":is(:state(current),[state--current])"));
|
|
1097
|
+
return selectedOrCurrent && !selectedOrCurrent.disabled ? selectedOrCurrent : this._getFirstFocusableDay();
|
|
1098
|
+
} else {
|
|
1099
|
+
const selectedOrCurrent = this.shadowRoot?.querySelector(":is(:state(selected),[state--selected])") ?? this.shadowRoot?.querySelector(":is(:state(current),[state--current])");
|
|
1100
|
+
return selectedOrCurrent && !selectedOrCurrent.disabled ? selectedOrCurrent : this.shadowRoot.querySelector(`sbb-calendar-${this._calendarView}:not([disabled])`);
|
|
1101
|
+
}
|
|
1102
|
+
}
|
|
1103
|
+
/**
|
|
1104
|
+
* In `day` view in `vertical` orientation,
|
|
1105
|
+
* if the first of the month is not a Monday, it is not the first rendered element in the table,
|
|
1106
|
+
* so `this.shadowRoot!.querySelector('sbb-calendar-day:not([disabled])')` will return a wrong value.
|
|
1107
|
+
*
|
|
1108
|
+
* To solve this, the element with the lowest `value` is taken (ISO String are ordered).
|
|
1109
|
+
*/
|
|
1110
|
+
_getFirstFocusableDay() {
|
|
1111
|
+
const cells = this._cells;
|
|
1112
|
+
const daysInView = cells.filter((e) => !e.disabled);
|
|
1113
|
+
if (!daysInView || daysInView.length === 0) return null;
|
|
1114
|
+
else {
|
|
1115
|
+
const firstElement = daysInView.map((e) => this._dateAdapter.toIso8601(e.value)).sort()[0];
|
|
1116
|
+
return cells.find((e) => e.matches(`[slot="${firstElement}"]`)) ?? null;
|
|
1117
|
+
}
|
|
1118
|
+
}
|
|
1119
|
+
_handleKeyboardEvent(event, day) {
|
|
1120
|
+
if (isArrowKeyOrPageKeysPressed(event)) event.preventDefault();
|
|
1121
|
+
const cells = this._cells;
|
|
1122
|
+
const index = cells.findIndex((e) => e === event.target);
|
|
1123
|
+
let nextEl;
|
|
1124
|
+
if (day) nextEl = this._navigateByKeyboardDayView(event, index, cells, day);
|
|
1125
|
+
else nextEl = this._navigateByKeyboard(event, index, cells);
|
|
1126
|
+
const activeEl = (this._enhancedVariant ? document : this.shadowRoot).activeElement;
|
|
1127
|
+
if (nextEl !== activeEl) {
|
|
1128
|
+
nextEl.tabIndex = 0;
|
|
1129
|
+
nextEl?.focus();
|
|
1130
|
+
activeEl.tabIndex = -1;
|
|
1131
|
+
}
|
|
1132
|
+
}
|
|
1133
|
+
_navigateByKeyboardDayView(evt, index, cells, day) {
|
|
1134
|
+
const arrowsOffset = this.orientation === "horizontal" ? {
|
|
1135
|
+
leftRight: 1,
|
|
1136
|
+
upDown: DAYS_PER_ROW
|
|
1137
|
+
} : {
|
|
1138
|
+
leftRight: DAYS_PER_ROW,
|
|
1139
|
+
upDown: 1
|
|
1140
|
+
};
|
|
1141
|
+
const offsetForVertical = index < this._keyboardNavigationDayViewParameters.firstMonthLength ? this._keyboardNavigationDayViewParameters.firstMonthOffset : this._keyboardNavigationDayViewParameters.secondMonthOffset;
|
|
1142
|
+
switch (evt.key) {
|
|
1143
|
+
case "ArrowUp": return this._findDayArrows(cells, index, day.dateValue, -arrowsOffset.upDown);
|
|
1144
|
+
case "ArrowDown": return this._findDayArrows(cells, index, day.dateValue, arrowsOffset.upDown);
|
|
1145
|
+
case "ArrowLeft": return this._findDayArrows(cells, index, day.dateValue, -arrowsOffset.leftRight);
|
|
1146
|
+
case "ArrowRight": return this._findDayArrows(cells, index, day.dateValue, arrowsOffset.leftRight);
|
|
1147
|
+
case "PageUp": if (this.orientation === "horizontal") {
|
|
1148
|
+
const delta = (+day.dayValue % DAYS_PER_ROW || DAYS_PER_ROW) - +day.dayValue;
|
|
1149
|
+
return this._findDayPageUpDown(cells, index, day, delta, arrowsOffset.upDown);
|
|
1150
|
+
} else {
|
|
1151
|
+
const delta = (Math.ceil((+day.dayValue + offsetForVertical) / DAYS_PER_ROW) - 1) * DAYS_PER_ROW - offsetForVertical + 1 - +day.dayValue;
|
|
1152
|
+
return this._findDayPageUpDown(cells, index, day, delta, arrowsOffset.upDown);
|
|
1153
|
+
}
|
|
1154
|
+
case "PageDown": if (this.orientation === "horizontal") {
|
|
1155
|
+
const monthInBounds = +day.monthValue + 1 > 12 ? 1 : +day.monthValue + 1;
|
|
1156
|
+
const yearInBounds = +day.monthValue + 1 > 12 ? +day.yearValue + 1 : +day.yearValue;
|
|
1157
|
+
const firstNextMonth = this._dateAdapter.createDate(yearInBounds, monthInBounds, 1);
|
|
1158
|
+
const lastOfMonth = this._dateAdapter.getDate(this._dateAdapter.addCalendarDays(firstNextMonth, -1));
|
|
1159
|
+
const delta = Math.trunc((lastOfMonth - +day.dayValue) / DAYS_PER_ROW) * DAYS_PER_ROW;
|
|
1160
|
+
return this._findDayPageUpDown(cells, index, day, delta, -arrowsOffset.upDown);
|
|
1161
|
+
} else {
|
|
1162
|
+
const delta = Math.ceil((+day.dayValue + offsetForVertical) / DAYS_PER_ROW) * DAYS_PER_ROW - offsetForVertical - +day.dayValue;
|
|
1163
|
+
return this._findDayPageUpDown(cells, index, day, delta, -arrowsOffset.upDown);
|
|
1164
|
+
}
|
|
1165
|
+
case "Home": return this._findDayFirst(cells, index, day, 1);
|
|
1166
|
+
case "End": {
|
|
1167
|
+
const monthInBounds = +day.monthValue + 1 > 12 ? 1 : +day.monthValue + 1;
|
|
1168
|
+
const yearInBounds = +day.monthValue + 1 > 12 ? +day.yearValue + 1 : +day.yearValue;
|
|
1169
|
+
const firstNextMonth = this._dateAdapter.createDate(yearInBounds, monthInBounds, 1);
|
|
1170
|
+
return this._findDayLast(cells, index, firstNextMonth);
|
|
1171
|
+
}
|
|
1172
|
+
default: return cells[index];
|
|
1173
|
+
}
|
|
1174
|
+
}
|
|
1175
|
+
_isDayOutOfView(date) {
|
|
1176
|
+
return date < this._keyboardNavigationDayViewParameters.firstDayInView || date > this._keyboardNavigationDayViewParameters.lastDayInView;
|
|
1177
|
+
}
|
|
1178
|
+
_findDayArrows(cells, index, date, delta) {
|
|
1179
|
+
const newDateValue = this._dateAdapter.toIso8601(this._dateAdapter.addCalendarDays(date, delta));
|
|
1180
|
+
if (this._isDayOutOfView(newDateValue)) return cells[index];
|
|
1181
|
+
const nextCell = cells.find((e) => this._mapValueToISODate(e.value) === newDateValue);
|
|
1182
|
+
if (!nextCell || nextCell.disabled) return this._findDayArrows(cells, index, this._dateAdapter.deserialize(newDateValue), delta);
|
|
1183
|
+
return nextCell;
|
|
1184
|
+
}
|
|
1185
|
+
_findDayPageUpDown(cells, index, day, delta, deltaIfDisabled) {
|
|
1186
|
+
const newDateValue = this._dateAdapter.toIso8601(this._dateAdapter.addCalendarDays(day.dateValue, delta));
|
|
1187
|
+
if (this._isDayOutOfView(newDateValue)) return cells[index];
|
|
1188
|
+
const nextCell = cells.find((e) => this._mapValueToISODate(e.value) === newDateValue);
|
|
1189
|
+
if (!nextCell || nextCell.disabled) return this._findDayPageUpDown(cells, index, day, delta + deltaIfDisabled, deltaIfDisabled);
|
|
1190
|
+
return nextCell;
|
|
1191
|
+
}
|
|
1192
|
+
_findDayFirst(cells, index, day, date) {
|
|
1193
|
+
const newDateValue = this._dateAdapter.toIso8601(this._dateAdapter.createDate(+day.yearValue, +day.monthValue, date));
|
|
1194
|
+
if (this._isDayOutOfView(newDateValue)) return cells[index];
|
|
1195
|
+
const nextCell = cells.find((e) => this._mapValueToISODate(e.value) === newDateValue);
|
|
1196
|
+
if (!nextCell || nextCell.disabled) return this._findDayFirst(cells, index, day, date + 1);
|
|
1197
|
+
return nextCell;
|
|
1198
|
+
}
|
|
1199
|
+
_findDayLast(cells, index, firstNextMonth) {
|
|
1200
|
+
const newDateValue = this._dateAdapter.toIso8601(this._dateAdapter.addCalendarDays(firstNextMonth, -1));
|
|
1201
|
+
if (this._isDayOutOfView(newDateValue)) return cells[index];
|
|
1202
|
+
const nextCell = cells.find((e) => this._mapValueToISODate(e.value) === newDateValue);
|
|
1203
|
+
if (!nextCell || nextCell.disabled) return this._findDayLast(cells, index, this._dateAdapter.deserialize(newDateValue));
|
|
1204
|
+
return nextCell;
|
|
1205
|
+
}
|
|
1206
|
+
/**
|
|
1207
|
+
* Gets the index of the element to move to, based on a list of elements (which can be potentially disabled),
|
|
1208
|
+
* the keyboard input and the position of the current element in the list.
|
|
1209
|
+
* In the day view, the `day?: Day` parameter is mandatory for calculation,
|
|
1210
|
+
* while in month and year view it's not due to the fixed amount of rendered cells.
|
|
1211
|
+
*/
|
|
1212
|
+
_navigateByKeyboard(evt, index, cells) {
|
|
1213
|
+
const { elementIndexForWideMode, offsetForWideMode, lastElementIndexForWideMode, verticalOffset } = this._calculateParametersForKeyboardNavigation(index, this._calendarView === "year");
|
|
1214
|
+
switch (evt.key) {
|
|
1215
|
+
case "ArrowUp": return this._findNext(cells, index, -verticalOffset);
|
|
1216
|
+
case "ArrowDown": return this._findNext(cells, index, verticalOffset);
|
|
1217
|
+
case "ArrowLeft": return this._findNext(cells, index, -1);
|
|
1218
|
+
case "ArrowRight": return this._findNext(cells, index, 1);
|
|
1219
|
+
case "Home": return this._findFirst(cells, offsetForWideMode);
|
|
1220
|
+
case "PageUp": return this._findFirstOnColumn(cells, elementIndexForWideMode, offsetForWideMode, verticalOffset);
|
|
1221
|
+
case "PageDown": return this._findLastOnColumn(cells, index, lastElementIndexForWideMode, verticalOffset);
|
|
1222
|
+
case "End": return this._findLast(cells, lastElementIndexForWideMode - 1);
|
|
1223
|
+
default: return cells[index];
|
|
1224
|
+
}
|
|
1225
|
+
}
|
|
1226
|
+
/**
|
|
1227
|
+
* Calculates the parameters needed in keyboard navigation in year and month view.
|
|
1228
|
+
* @param index The starting element's index in the cell array.
|
|
1229
|
+
* @param isYearView Whether the displayed `view` is the year one.
|
|
1230
|
+
*/
|
|
1231
|
+
_calculateParametersForKeyboardNavigation(index, isYearView) {
|
|
1232
|
+
const elementsPerPage = isYearView ? YEARS_PER_PAGE : MONTHS_PER_PAGE;
|
|
1233
|
+
const offset = Math.trunc(index / elementsPerPage) * elementsPerPage;
|
|
1234
|
+
const indexInView = offset === 0 ? index : index - elementsPerPage;
|
|
1235
|
+
return {
|
|
1236
|
+
verticalOffset: isYearView ? YEARS_PER_ROW : MONTHS_PER_ROW,
|
|
1237
|
+
elementIndexForWideMode: indexInView,
|
|
1238
|
+
offsetForWideMode: index - indexInView,
|
|
1239
|
+
lastElementIndexForWideMode: offset === 0 ? elementsPerPage : elementsPerPage * 2
|
|
1240
|
+
};
|
|
1241
|
+
}
|
|
1242
|
+
/**
|
|
1243
|
+
* Gets the next element of the provided array starting from `index` by adding `delta`.
|
|
1244
|
+
* If the found element is disabled, it continues adding `delta` until it finds an enabled one in the array bounds.
|
|
1245
|
+
*/
|
|
1246
|
+
_findNext(days, index, delta) {
|
|
1247
|
+
let nextIndex = index + delta;
|
|
1248
|
+
while (nextIndex < days.length && days[nextIndex]?.disabled) nextIndex += delta;
|
|
1249
|
+
return days[nextIndex] ?? days[index];
|
|
1250
|
+
}
|
|
1251
|
+
/** Find the first enabled element in the provided array. */
|
|
1252
|
+
_findFirst(days, firstOfCurrentMonth) {
|
|
1253
|
+
return !days[firstOfCurrentMonth].disabled ? days[firstOfCurrentMonth] : this._findNext(days, firstOfCurrentMonth, 1);
|
|
1254
|
+
}
|
|
1255
|
+
/** Find the last enabled element in the provided array. */
|
|
1256
|
+
_findLast(days, lastOfCurrentMonth) {
|
|
1257
|
+
return !days[lastOfCurrentMonth].disabled ? days[lastOfCurrentMonth] : this._findNext(days, lastOfCurrentMonth, -1);
|
|
1258
|
+
}
|
|
1259
|
+
/** Find the first enabled element in the same column of the provided array. */
|
|
1260
|
+
_findFirstOnColumn(days, index, offset, verticalOffset) {
|
|
1261
|
+
const nextIndex = index % verticalOffset + offset;
|
|
1262
|
+
return !days[nextIndex].disabled ? days[nextIndex] : this._findNext(days, nextIndex, verticalOffset);
|
|
1263
|
+
}
|
|
1264
|
+
/** Find the last enabled element in the same column of the provided array. */
|
|
1265
|
+
_findLastOnColumn(days, index, offset, verticalOffset) {
|
|
1266
|
+
const nextIndex = index + Math.trunc((offset - index - 1) / verticalOffset) * verticalOffset;
|
|
1267
|
+
return !days[nextIndex].disabled ? days[nextIndex] : this._findNext(days, nextIndex, -verticalOffset);
|
|
1268
|
+
}
|
|
1269
|
+
_resetCalendarViewAndEmitMonthChange(initTransition = false) {
|
|
1270
|
+
this._resetCalendarView(initTransition);
|
|
1271
|
+
this._emitMonthChange();
|
|
1272
|
+
}
|
|
1273
|
+
_resetCalendarView(initTransition = false) {
|
|
1274
|
+
if (this._containingFocus) this._resetFocus = true;
|
|
1275
|
+
this._activeDate = (this.multiple ? this._selected.at(-1) : this._selected) ?? this._dateAdapter.today();
|
|
1276
|
+
this._setChosenYear();
|
|
1277
|
+
this._chosenMonth = void 0;
|
|
1278
|
+
this._init();
|
|
1279
|
+
this._nextCalendarView = this._calendarView = this.view;
|
|
1280
|
+
if (initTransition) this._startTableTransition();
|
|
1281
|
+
}
|
|
1282
|
+
/** Render the view for the day selection. */
|
|
1283
|
+
_renderDayView() {
|
|
1284
|
+
const nextMonthActiveDate = this._wide ? this._dateAdapter.addCalendarMonths(this._activeDate, 1) : void 0;
|
|
1285
|
+
return html`
|
|
1286
|
+
<div class="sbb-calendar__controls">
|
|
1287
|
+
${this._getArrow("left", () => this._goToDifferentMonth(-1), i18nPreviousMonth[this._language.current], this._previousMonthDisabled())}
|
|
1288
|
+
<div class="sbb-calendar__controls-month">
|
|
1289
|
+
${this._createLabelForDayView(this._activeDate)}
|
|
1290
|
+
${this._wide ? this._createLabelForDayView(nextMonthActiveDate) : nothing}
|
|
1291
|
+
<sbb-screen-reader-only role="status">
|
|
1292
|
+
${this._createAriaLabelForDayView(this._activeDate, nextMonthActiveDate)}
|
|
1293
|
+
</sbb-screen-reader-only>
|
|
1294
|
+
</div>
|
|
1295
|
+
${this._getArrow("right", () => this._goToDifferentMonth(1), i18nNextMonth[this._language.current], this._nextMonthDisabled())}
|
|
1296
|
+
</div>
|
|
1297
|
+
<div class="sbb-calendar__table-overflow-break">
|
|
1298
|
+
<div class="sbb-calendar__table-container sbb-calendar__table-day-view">
|
|
1299
|
+
${this.orientation === "horizontal" ? html`
|
|
1300
|
+
${this._createDayTable(this._weeks, this._weekNumbers)}
|
|
1301
|
+
${this._wide ? this._createDayTable(this._nextMonthWeeks, this._nextMonthWeekNumbers, true) : nothing}
|
|
1302
|
+
` : html`
|
|
1303
|
+
${this._createDayTableVertical(this._weeks, this._weekNumbers)}
|
|
1304
|
+
${this._wide ? this._createDayTableVertical(this._nextMonthWeeks, this._nextMonthWeekNumbers, nextMonthActiveDate) : nothing}
|
|
1305
|
+
`}
|
|
1306
|
+
</div>
|
|
1307
|
+
</div>
|
|
1308
|
+
`;
|
|
1309
|
+
}
|
|
1310
|
+
/** Creates the label with the month for the daily view. */
|
|
1311
|
+
_createLabelForDayView(d) {
|
|
1312
|
+
const monthLabel = `${this._monthNames[this._dateAdapter.getMonth(d) - 1]} ${this._dateAdapter.getYear(d)}`;
|
|
1313
|
+
return html`
|
|
1314
|
+
<button
|
|
1315
|
+
type="button"
|
|
1316
|
+
class="sbb-calendar__date-selection sbb-calendar__controls-change-date"
|
|
1317
|
+
aria-label="${i18nYearMonthSelection[this._language.current]} ${monthLabel}"
|
|
1318
|
+
@click=${() => {
|
|
1319
|
+
this._resetFocus = true;
|
|
1320
|
+
this._nextCalendarView = "year";
|
|
1321
|
+
this._startTableTransition();
|
|
1322
|
+
}}
|
|
1323
|
+
>
|
|
1324
|
+
${monthLabel}
|
|
1325
|
+
<sbb-icon name="chevron-small-down-small"></sbb-icon>
|
|
1326
|
+
</button>
|
|
1327
|
+
`;
|
|
1328
|
+
}
|
|
1329
|
+
/** Creates the aria-label for the daily view. */
|
|
1330
|
+
_createAriaLabelForDayView(...dates) {
|
|
1331
|
+
let monthLabel = "";
|
|
1332
|
+
for (const d of dates) if (d) monthLabel += `${this._monthNames[this._dateAdapter.getMonth(d) - 1]} ${this._dateAdapter.getYear(d)} `;
|
|
1333
|
+
return monthLabel;
|
|
1334
|
+
}
|
|
1335
|
+
/** Creates the calendar table for the daily view. */
|
|
1336
|
+
_createDayTable(weeks, weekNumbers, isWideNextMonth = false) {
|
|
1337
|
+
const weeksForSelectMultipleWeekNumbers = (this._wide ? [...this._weeks, ...this._nextMonthWeeks] : isWideNextMonth ? this._nextMonthWeeks : this._weeks).flat();
|
|
1338
|
+
const weeksForSelectMultipleWeekDays = (isWideNextMonth ? this._nextMonthWeeks : this._weeks).flat();
|
|
1339
|
+
return html`
|
|
1340
|
+
<table
|
|
1341
|
+
class="sbb-calendar__table"
|
|
1342
|
+
@focusout=${(event) => this._handleTableBlur(event.relatedTarget)}
|
|
1343
|
+
@animationend=${(e) => this._tableAnimationEnd(e)}
|
|
1344
|
+
>
|
|
1345
|
+
<thead class="sbb-calendar__table-header">
|
|
1346
|
+
<tr>
|
|
1347
|
+
${this.weekNumbers ? html`<th class="sbb-calendar__table-header-cell"></th>` : nothing}
|
|
1348
|
+
${this._weekdays.map((weekDay, index) => html`
|
|
1349
|
+
<th class="sbb-calendar__table-header-cell">
|
|
1350
|
+
${this.multiple ? html`
|
|
1351
|
+
<sbb-calendar-weekday
|
|
1352
|
+
.value=${weekDay}
|
|
1353
|
+
@click=${() => {
|
|
1354
|
+
const days = weeksForSelectMultipleWeekDays.filter((day) => day.weekDayValue === (index + 1) % 7);
|
|
1355
|
+
this._selectMultipleDates(days);
|
|
1356
|
+
}}
|
|
1357
|
+
></sbb-calendar-weekday>
|
|
1358
|
+
` : html`
|
|
1359
|
+
<sbb-screen-reader-only>${weekDay.long}</sbb-screen-reader-only>
|
|
1360
|
+
<span aria-hidden="true">${weekDay.narrow}</span>
|
|
1361
|
+
`}
|
|
1362
|
+
</th>
|
|
1363
|
+
`)}
|
|
1364
|
+
</tr>
|
|
1365
|
+
</thead>
|
|
1366
|
+
<tbody class="sbb-calendar__table-body">
|
|
1367
|
+
${weeks.map((week, rowIndex) => {
|
|
1368
|
+
const firstRowOffset = DAYS_PER_ROW - week.length;
|
|
1369
|
+
if (rowIndex === 0 && firstRowOffset) return html`
|
|
1370
|
+
<tr>
|
|
1371
|
+
${this.weekNumbers ? html`
|
|
1372
|
+
<td class="sbb-calendar__table-header-cell-vertical">
|
|
1373
|
+
${this.multiple ? html`
|
|
1374
|
+
<sbb-calendar-weeknumber
|
|
1375
|
+
.value=${weekNumbers[0]}
|
|
1376
|
+
@click=${() => {
|
|
1377
|
+
const days = weeksForSelectMultipleWeekNumbers.filter((day) => day.weekValue === weekNumbers[0]);
|
|
1378
|
+
this._selectMultipleDates(days);
|
|
1379
|
+
}}
|
|
1380
|
+
></sbb-calendar-weeknumber>
|
|
1381
|
+
` : html`
|
|
1382
|
+
<sbb-screen-reader-only
|
|
1383
|
+
>${`${i18nCalendarWeekNumber[this._language.current]} ${weekNumbers[0]}`}</sbb-screen-reader-only
|
|
1384
|
+
>
|
|
1385
|
+
<span aria-hidden="true">${weekNumbers[0]}</span>
|
|
1386
|
+
`}
|
|
1387
|
+
</td>
|
|
1388
|
+
` : nothing}
|
|
1389
|
+
${[...Array(firstRowOffset).keys()].map(() => html`<td class="sbb-calendar__table-data"></td>`)}
|
|
1390
|
+
${this._createDayCells(week)}
|
|
1391
|
+
</tr>
|
|
1392
|
+
`;
|
|
1393
|
+
return html`
|
|
1394
|
+
<tr>
|
|
1395
|
+
${this.weekNumbers ? html`
|
|
1396
|
+
<td class="sbb-calendar__table-header-cell-vertical">
|
|
1397
|
+
${this.multiple ? html`
|
|
1398
|
+
<sbb-calendar-weeknumber
|
|
1399
|
+
.value=${weekNumbers[rowIndex]}
|
|
1400
|
+
@click=${() => {
|
|
1401
|
+
const days = weeksForSelectMultipleWeekNumbers.filter((day) => day.weekValue === weekNumbers[rowIndex]);
|
|
1402
|
+
this._selectMultipleDates(days);
|
|
1403
|
+
}}
|
|
1404
|
+
></sbb-calendar-weeknumber>
|
|
1405
|
+
` : html`
|
|
1406
|
+
<sbb-screen-reader-only
|
|
1407
|
+
>${`${i18nCalendarWeekNumber[this._language.current]} ${weekNumbers[rowIndex]}`}</sbb-screen-reader-only
|
|
1408
|
+
>
|
|
1409
|
+
<span aria-hidden="true">${weekNumbers[rowIndex]}</span>
|
|
1410
|
+
`}
|
|
1411
|
+
</td>
|
|
1412
|
+
` : nothing}
|
|
1413
|
+
${this._createDayCells(week)}
|
|
1414
|
+
</tr>
|
|
1415
|
+
`;
|
|
1416
|
+
})}
|
|
1417
|
+
</tbody>
|
|
1418
|
+
</table>
|
|
1419
|
+
`;
|
|
1420
|
+
}
|
|
1421
|
+
/** Creates the table in orientation='vertical'. */
|
|
1422
|
+
_createDayTableVertical(weeks, weekNumbers, nextMonthActiveDate) {
|
|
1423
|
+
const weekOffset = this._dateAdapter.getFirstWeekOffset(nextMonthActiveDate ?? this._activeDate);
|
|
1424
|
+
const weeksForSelectMultipleWeekNumbers = (this._wide ? [...this._weeks, ...this._nextMonthWeeks] : nextMonthActiveDate ? this._nextMonthWeeks : this._weeks).flat();
|
|
1425
|
+
return html`
|
|
1426
|
+
<table
|
|
1427
|
+
class="sbb-calendar__table"
|
|
1428
|
+
@focusout=${(event) => this._handleTableBlur(event.relatedTarget)}
|
|
1429
|
+
@animationend=${(e) => this._tableAnimationEnd(e)}
|
|
1430
|
+
>
|
|
1431
|
+
${this.weekNumbers ? html`
|
|
1432
|
+
<thead class="sbb-calendar__table-header">
|
|
1433
|
+
<tr>
|
|
1434
|
+
${nextMonthActiveDate ? nothing : html`<th class="sbb-calendar__table-data"></th>`}
|
|
1435
|
+
${weekNumbers.map((weekNumber) => html`
|
|
1436
|
+
<th class="sbb-calendar__table-header-cell">
|
|
1437
|
+
${this.multiple ? html`
|
|
1438
|
+
<sbb-calendar-weeknumber
|
|
1439
|
+
.value=${weekNumber}
|
|
1440
|
+
@click=${() => {
|
|
1441
|
+
const days = weeksForSelectMultipleWeekNumbers.filter((day) => day.weekValue === weekNumber);
|
|
1442
|
+
this._selectMultipleDates(days);
|
|
1443
|
+
}}
|
|
1444
|
+
></sbb-calendar-weeknumber>
|
|
1445
|
+
` : html`
|
|
1446
|
+
<sbb-screen-reader-only
|
|
1447
|
+
>${`${i18nCalendarWeekNumber[this._language.current]} ${weekNumber}`}</sbb-screen-reader-only
|
|
1448
|
+
>
|
|
1449
|
+
<span aria-hidden="true">${weekNumber}</span>
|
|
1450
|
+
`}
|
|
1451
|
+
</th>
|
|
1452
|
+
`)}
|
|
1453
|
+
</tr>
|
|
1454
|
+
</thead>
|
|
1455
|
+
` : nothing}
|
|
1456
|
+
<tbody class="sbb-calendar__table-body">
|
|
1457
|
+
${weeks.map((week, rowIndex) => {
|
|
1458
|
+
const weekday = this._weekdays[rowIndex];
|
|
1459
|
+
const selectableDays = this._wide ? [...week, ...this._nextMonthWeeks[rowIndex]] : week;
|
|
1460
|
+
return html`
|
|
1461
|
+
<tr>
|
|
1462
|
+
${nextMonthActiveDate ? nothing : html`
|
|
1463
|
+
<td class="sbb-calendar__table-header-cell-vertical">
|
|
1464
|
+
${this.multiple ? html`
|
|
1465
|
+
<sbb-calendar-weekday
|
|
1466
|
+
.value=${weekday}
|
|
1467
|
+
@click=${() => this._selectMultipleDates(selectableDays)}
|
|
1468
|
+
>
|
|
1469
|
+
${weekday.narrow}
|
|
1470
|
+
</sbb-calendar-weekday>
|
|
1471
|
+
` : html`
|
|
1472
|
+
<sbb-screen-reader-only>${weekday.long}</sbb-screen-reader-only>
|
|
1473
|
+
<span aria-hidden="true">${weekday.narrow}</span>
|
|
1474
|
+
`}
|
|
1475
|
+
</td>
|
|
1476
|
+
`}
|
|
1477
|
+
${rowIndex < weekOffset ? html`<td class="sbb-calendar__table-data"></td>` : nothing}
|
|
1478
|
+
${this._createDayCells(week)}
|
|
1479
|
+
</tr>
|
|
1480
|
+
`;
|
|
1481
|
+
})}
|
|
1482
|
+
</tbody>
|
|
1483
|
+
</table>
|
|
1484
|
+
`;
|
|
1485
|
+
}
|
|
1486
|
+
/** Creates the cells for the daily view. */
|
|
1487
|
+
_createDayCells(week) {
|
|
1488
|
+
return week.map((day) => {
|
|
1489
|
+
return html`
|
|
1490
|
+
<td class="sbb-calendar__table-data sbb-calendar__day-cell">
|
|
1491
|
+
<slot name=${day.value}>
|
|
1492
|
+
<sbb-calendar-day
|
|
1493
|
+
slot=${day.value}
|
|
1494
|
+
@click=${() => this._selectDate(day.dateValue)}
|
|
1495
|
+
@keydown=${(evt) => this._handleKeyboardEvent(evt, day)}
|
|
1496
|
+
></sbb-calendar-day>
|
|
1497
|
+
</slot>
|
|
1498
|
+
</td>
|
|
1499
|
+
`;
|
|
1500
|
+
});
|
|
1501
|
+
}
|
|
1502
|
+
/** Render the view for the month selection. */
|
|
1503
|
+
_renderMonthView() {
|
|
1504
|
+
return html`
|
|
1505
|
+
<div class="sbb-calendar__controls">
|
|
1506
|
+
${this._getArrow("left", () => this._goToDifferentYear(-1), i18nPreviousYear[this._language.current], this._previousYearDisabled())}
|
|
1507
|
+
<div class="sbb-calendar__controls-month">${this._createLabelForMonthView()}</div>
|
|
1508
|
+
${this._getArrow("right", () => this._goToDifferentYear(1), i18nNextYear[this._language.current], this._nextYearDisabled())}
|
|
1509
|
+
</div>
|
|
1510
|
+
<div class="sbb-calendar__table-overflow-break">
|
|
1511
|
+
<div class="sbb-calendar__table-container sbb-calendar__table-month-view">
|
|
1512
|
+
${this._createMonthTable(this._months, this._chosenYear)}
|
|
1513
|
+
${this._wide ? this._createMonthTable(this._months, this._chosenYear + 1) : nothing}
|
|
1514
|
+
</div>
|
|
1515
|
+
</div>
|
|
1516
|
+
`;
|
|
1517
|
+
}
|
|
1518
|
+
/** Creates the label with the year for the monthly view. */
|
|
1519
|
+
_createLabelForMonthView() {
|
|
1520
|
+
return html` <button
|
|
1521
|
+
type="button"
|
|
1522
|
+
id="sbb-calendar__month-selection"
|
|
1523
|
+
class="sbb-calendar__controls-change-date"
|
|
1524
|
+
aria-label=${`${i18nCalendarDateSelection[this._language.current]} ${this._chosenYear}`}
|
|
1525
|
+
@click=${() => this._resetCalendarViewAndEmitMonthChange(true)}
|
|
1526
|
+
>
|
|
1527
|
+
${this._chosenYear} ${this._wide ? ` - ${this._chosenYear + 1}` : nothing}
|
|
1528
|
+
<sbb-icon name="chevron-small-up-small"></sbb-icon>
|
|
1529
|
+
</button>
|
|
1530
|
+
<sbb-screen-reader-only role="status"> ${this._chosenYear} </sbb-screen-reader-only>`;
|
|
1531
|
+
}
|
|
1532
|
+
/** Creates the table for the month selection view. */
|
|
1533
|
+
_createMonthTable(months, year) {
|
|
1534
|
+
return html`
|
|
1535
|
+
<table
|
|
1536
|
+
class="sbb-calendar__table"
|
|
1537
|
+
@animationend=${(e) => this._tableAnimationEnd(e)}
|
|
1538
|
+
>
|
|
1539
|
+
${this._wide ? html`<thead class="sbb-calendar__table-header" aria-hidden="true">
|
|
1540
|
+
<tr>
|
|
1541
|
+
<th class="sbb-calendar__table-header-cell" colspan=${MONTHS_PER_ROW}>${year}</th>
|
|
1542
|
+
</tr>
|
|
1543
|
+
</thead>` : nothing}
|
|
1544
|
+
<tbody class="sbb-calendar__table-body">
|
|
1545
|
+
${months.map((row) => html`
|
|
1546
|
+
<tr>
|
|
1547
|
+
${row.map((month) => {
|
|
1548
|
+
return html`
|
|
1549
|
+
<td class="sbb-calendar__table-data">
|
|
1550
|
+
<sbb-calendar-month
|
|
1551
|
+
.value="${year}-${month.value}"
|
|
1552
|
+
@click=${() => this._onMonthSelection(month.monthValue, year)}
|
|
1553
|
+
@keydown=${(evt) => this._handleKeyboardEvent(evt)}
|
|
1554
|
+
>
|
|
1555
|
+
</sbb-calendar-month>
|
|
1556
|
+
</td>
|
|
1557
|
+
`;
|
|
1558
|
+
})}
|
|
1559
|
+
</tr>
|
|
1560
|
+
`)}
|
|
1561
|
+
</tbody>
|
|
1562
|
+
</table>
|
|
1563
|
+
`;
|
|
1564
|
+
}
|
|
1565
|
+
/** Select the month and change the view to day selection. */
|
|
1566
|
+
_onMonthSelection(month, year) {
|
|
1567
|
+
this._chosenMonth = month;
|
|
1568
|
+
this._nextCalendarView = "day";
|
|
1569
|
+
this._init(this._dateAdapter.createDate(year, this._chosenMonth, this._dateAdapter.getDate(this._activeDate)));
|
|
1570
|
+
this._startTableTransition();
|
|
1571
|
+
this._emitMonthChange();
|
|
1572
|
+
}
|
|
1573
|
+
/** Render the view for the year selection. */
|
|
1574
|
+
_renderYearView() {
|
|
1575
|
+
return html`
|
|
1576
|
+
<div class="sbb-calendar__controls">
|
|
1577
|
+
${this._getArrow("left", () => this._goToDifferentYearRange(-YEARS_PER_PAGE), i18nPreviousYearRange(YEARS_PER_PAGE)[this._language.current], this._previousYearRangeDisabled())}
|
|
1578
|
+
<div class="sbb-calendar__controls-month">${this._createLabelForYearView()}</div>
|
|
1579
|
+
${this._getArrow("right", () => this._goToDifferentYearRange(YEARS_PER_PAGE), i18nNextYearRange(YEARS_PER_PAGE)[this._language.current], this._nextYearRangeDisabled())}
|
|
1580
|
+
</div>
|
|
1581
|
+
<div class="sbb-calendar__table-overflow-break">
|
|
1582
|
+
<div class="sbb-calendar__table-container sbb-calendar__table-year-view">
|
|
1583
|
+
${this._createYearTable(this._years)}
|
|
1584
|
+
${this._wide ? this._createYearTable(this._nextMonthYears, true) : nothing}
|
|
1585
|
+
</div>
|
|
1586
|
+
</div>
|
|
1587
|
+
`;
|
|
1588
|
+
}
|
|
1589
|
+
/** Creates the button arrow for all the views. */
|
|
1590
|
+
_getArrow(direction, click, ariaLabel, disabled) {
|
|
1591
|
+
return html`<sbb-secondary-button
|
|
1592
|
+
size="m"
|
|
1593
|
+
icon-name="chevron-small-${direction}-small"
|
|
1594
|
+
aria-label=${ariaLabel}
|
|
1595
|
+
@click=${click}
|
|
1596
|
+
?disabled=${disabled}
|
|
1597
|
+
id="sbb-calendar__controls-${direction === "left" ? "previous" : "next"}"
|
|
1598
|
+
></sbb-secondary-button>`;
|
|
1599
|
+
}
|
|
1600
|
+
/** Creates the label with the year range for the yearly view. */
|
|
1601
|
+
_createLabelForYearView() {
|
|
1602
|
+
const firstYear = this._years.flat()[0];
|
|
1603
|
+
const lastYearArray = (this._wide ? this._nextMonthYears : this._years).flat();
|
|
1604
|
+
const yearLabel = `${firstYear} - ${lastYearArray[lastYearArray.length - 1]}`;
|
|
1605
|
+
return html`
|
|
1606
|
+
<button
|
|
1607
|
+
type="button"
|
|
1608
|
+
id="sbb-calendar__year-selection"
|
|
1609
|
+
class="sbb-calendar__controls-change-date"
|
|
1610
|
+
aria-label="${i18nCalendarDateSelection[this._language.current]} ${yearLabel}"
|
|
1611
|
+
@click=${() => this._resetCalendarViewAndEmitMonthChange(true)}
|
|
1612
|
+
>
|
|
1613
|
+
${yearLabel}
|
|
1614
|
+
<sbb-icon name="chevron-small-up-small"></sbb-icon>
|
|
1615
|
+
</button>
|
|
1616
|
+
<sbb-screen-reader-only role="status"> ${yearLabel} </sbb-screen-reader-only>
|
|
1617
|
+
`;
|
|
1618
|
+
}
|
|
1619
|
+
/** Creates the table for the year selection view. */
|
|
1620
|
+
_createYearTable(years, shiftRight = false) {
|
|
1621
|
+
return html` <table
|
|
1622
|
+
class="sbb-calendar__table"
|
|
1623
|
+
@animationend=${(e) => this._tableAnimationEnd(e)}
|
|
1624
|
+
>
|
|
1625
|
+
<tbody class="sbb-calendar__table-body">
|
|
1626
|
+
${years.map((row) => html` <tr>
|
|
1627
|
+
${row.map((year) => {
|
|
1628
|
+
return html`
|
|
1629
|
+
<td class="sbb-calendar__table-data">
|
|
1630
|
+
<sbb-calendar-year
|
|
1631
|
+
.value=${String(year)}
|
|
1632
|
+
@keydown=${(evt) => this._handleKeyboardEvent(evt)}
|
|
1633
|
+
@click=${() => this._onYearSelection(year, shiftRight)}
|
|
1634
|
+
>
|
|
1635
|
+
</sbb-calendar-year>
|
|
1636
|
+
</td>
|
|
1637
|
+
`;
|
|
1638
|
+
})}
|
|
1639
|
+
</tr>`)}
|
|
1640
|
+
</tbody>
|
|
1641
|
+
</table>`;
|
|
1642
|
+
}
|
|
1643
|
+
/** Select the year and change the view to month selection. */
|
|
1644
|
+
_onYearSelection(year, rightSide) {
|
|
1645
|
+
this._chosenYear = rightSide ? year - 1 : year;
|
|
1646
|
+
this._nextCalendarView = "month";
|
|
1647
|
+
this._assignActiveDate(this._dateAdapter.createDate(this._chosenYear, this._dateAdapter.getMonth(this._activeDate), this._dateAdapter.getDate(this._activeDate)));
|
|
1648
|
+
this._startTableTransition();
|
|
1649
|
+
}
|
|
1650
|
+
_getView() {
|
|
1651
|
+
if (isServer || this.hydrationRequired) return html`${nothing}`;
|
|
1652
|
+
switch (this._calendarView) {
|
|
1653
|
+
case "year": return this._renderYearView();
|
|
1654
|
+
case "month": return this._renderMonthView();
|
|
1655
|
+
default: return this._renderDayView();
|
|
1656
|
+
}
|
|
1657
|
+
}
|
|
1658
|
+
_tableAnimationEnd(event) {
|
|
1659
|
+
const table = event.target;
|
|
1660
|
+
if (event.animationName === "hide") {
|
|
1661
|
+
table.classList.remove("sbb-calendar__table-hide");
|
|
1662
|
+
if (this._containingFocus) this._resetFocus = true;
|
|
1663
|
+
this._calendarView = this._nextCalendarView;
|
|
1664
|
+
} else if (event.animationName === "show") this.internals.states.delete("transition");
|
|
1665
|
+
}
|
|
1666
|
+
_startTableTransition() {
|
|
1667
|
+
this.internals.states.add("transition");
|
|
1668
|
+
this.shadowRoot?.querySelectorAll("table")?.forEach((e) => e.classList.toggle("sbb-calendar__table-hide"));
|
|
1669
|
+
}
|
|
1670
|
+
render() {
|
|
1671
|
+
return html`<div class="sbb-calendar__wrapper">${this._getView()}</div>`;
|
|
1672
|
+
}
|
|
1673
|
+
};
|
|
1674
|
+
})();
|
|
1675
|
+
//#endregion
|
|
1676
|
+
export { SbbMonthChangeEvent as n, SbbCalendarElement as t };
|
|
1677
|
+
|
|
1678
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY2FsZW5kYXIuY29tcG9uZW50LUNzQWw5RUFnLmpzIiwibmFtZXMiOltdLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3NyYy9lbGVtZW50cy9jYWxlbmRhci9jYWxlbmRhci9jYWxlbmRhci5zY3NzP2xpdCZpbmxpbmUiLCIuLi8uLi8uLi9zcmMvZWxlbWVudHMvY2FsZW5kYXIvY2FsZW5kYXIvY2FsZW5kYXIuY29tcG9uZW50LnRzIl0sInNvdXJjZXNDb250ZW50IjpbIkB1c2UgJy4uLy4uL2NvcmUvc3R5bGVzJyBhcyBzYmI7XG5cbjpob3N0IHtcbiAgZGlzcGxheTogYmxvY2s7XG5cbiAgLy8gV2UgYWRkIHdpZHRoIGRlZmluaXRpb24gdG8gaG9zdCwgdG8gbWFrZSBvdmVyd3JpdGluZyBlYXN5IGZvciBjb25zdW1lcnMuXG4gIHdpZHRoOiBtYXgtY29udGVudDtcblxuICAtLXNiYi1jYWxlbmRhci1jZWxsLXNpemU6ICN7c2JiLnB4LXRvLXJlbS1idWlsZCg0NCl9O1xuXG4gIC8vIE1hdGNoIEZpZ21hIGRlc2lnblxuICAtLXNiYi1jYWxlbmRhci1jZWxsLWJvcmRlci1yYWRpdXM6IGNhbGMoXG4gICAgdmFyKC0tc2JiLWJvcmRlci1yYWRpdXMtNHgpICsgdmFyKC0tc2JiLWNhbGVuZGFyLWNlbGwtYm9yZGVyLXdpZHRoKVxuICApO1xuICAtLXNiYi1jYWxlbmRhci1jZWxsLXRyYW5zaXRpb24tZHVyYXRpb246IHZhcihcbiAgICAtLXNiYi1kaXNhYmxlLWFuaW1hdGlvbi1kdXJhdGlvbixcbiAgICB2YXIoLS1zYmItYW5pbWF0aW9uLWR1cmF0aW9uLTJ4KVxuICApO1xuXG4gIEBpbmNsdWRlIHNiYi5tcSgkZnJvbTogc21hbGwpIHtcbiAgICAtLXNiYi1jYWxlbmRhci1jb250cm9sLXZpZXctY2hhbmdlLWhlaWdodDogI3tzYmIucHgtdG8tcmVtLWJ1aWxkKDQ4KX07XG4gIH1cbn1cblxuOmhvc3QoOnN0YXRlKGVuaGFuY2VkKSkge1xuICAtLXNiYi1jYWxlbmRhci1jZWxsLXBhZGRpbmctYmxvY2stZW5kOiB2YXIoLS1zYmItc3BhY2luZy1maXhlZC0zeCk7XG59XG5cbi5zYmItY2FsZW5kYXJfX3dyYXBwZXIge1xuICB3aWR0aDogMTAwJTtcbiAgZGlzcGxheTogYmxvY2s7XG4gIHRyYW5zaXRpb24tZHVyYXRpb246IHZhcigtLXNiYi1jYWxlbmRhci1jZWxsLXRyYW5zaXRpb24tZHVyYXRpb24pO1xufVxuXG4uc2JiLWNhbGVuZGFyX19jb250cm9scyB7XG4gIHdpZHRoOiAxMDAlO1xuICBkaXNwbGF5OiBpbmxpbmUtZmxleDtcbiAgYWxpZ24taXRlbXM6IGNlbnRlcjtcbiAgZ2FwOiB2YXIoLS1zYmItY2FsZW5kYXItY29udHJvbC1nYXApO1xuICBtYXJnaW4tYmxvY2stZW5kOiB2YXIoLS1zYmItY2FsZW5kYXItY29udHJvbC1tYXJnaW4tYmxvY2stZW5kKTtcbn1cblxuLnNiYi1jYWxlbmRhcl9fY29udHJvbHMtbW9udGgge1xuICB3aWR0aDogMTAwJTtcbiAgZGlzcGxheTogZmxleDtcbiAgZ2FwOiB2YXIoLS1zYmItY2FsZW5kYXItdGFibGVzLWdhcCk7XG59XG5cbiNzYmItY2FsZW5kYXJfX2NvbnRyb2xzLXByZXZpb3VzLFxuI3NiYi1jYWxlbmRhcl9fY29udHJvbHMtbmV4dCB7XG4gIC13ZWJraXQtdGFwLWhpZ2hsaWdodC1jb2xvcjogdHJhbnNwYXJlbnQ7XG59XG5cbi5zYmItY2FsZW5kYXJfX2NvbnRyb2xzLWNoYW5nZS1kYXRlIHtcbiAgQGluY2x1ZGUgc2JiLmJ1dHRvbi1yZXNldDtcblxuICBkaXNwbGF5OiBmbGV4O1xuICBhbGlnbi1pdGVtczogY2VudGVyO1xuICBtYXJnaW46IGF1dG87XG4gIGhlaWdodDogdmFyKC0tc2JiLWNhbGVuZGFyLWNvbnRyb2wtdmlldy1jaGFuZ2UtaGVpZ2h0KTtcbiAgZm9udC1zaXplOiB2YXIoLS1zYmItdGV4dC1mb250LXNpemUtcyk7XG4gIGxldHRlci1zcGFjaW5nOiB2YXIoLS1zYmItdHlwby1sZXR0ZXItc3BhY2luZy10ZXh0KTtcbiAgdGV4dC10cmFuc2Zvcm06IGNhcGl0YWxpemU7XG4gIGN1cnNvcjogdmFyKC0tc2JiLWN1cnNvci1wb2ludGVyKTtcbiAgcGFkZGluZy1pbmxpbmU6IHZhcigtLXNiYi1jYWxlbmRhci1jb250cm9sLXZpZXctY2hhbmdlLXBhZGRpbmctaW5saW5lKTtcbiAgYm9yZGVyLXJhZGl1czogdmFyKC0tc2JiLWJvcmRlci1yYWRpdXMtaW5maW5pdHkpO1xuICBiYWNrZ3JvdW5kLWNvbG9yOiB2YXIoLS1zYmItY2FsZW5kYXItY29udHJvbC12aWV3LWNoYW5nZS1iYWNrZ3JvdW5kKTtcbiAgY29sb3I6IHZhcigtLXNiYi1jYWxlbmRhci1jb250cm9sLXZpZXctY2hhbmdlLWNvbG9yKTtcbiAgdHJhbnNpdGlvbi1kdXJhdGlvbjogdmFyKC0tc2JiLWNhbGVuZGFyLWNlbGwtdHJhbnNpdGlvbi1kdXJhdGlvbik7XG4gIHRyYW5zaXRpb24tdGltaW5nLWZ1bmN0aW9uOiB2YXIoLS1zYmItY2FsZW5kYXItY2VsbC10cmFuc2l0aW9uLWVhc2luZy1mdW5jdGlvbik7XG4gIHRyYW5zaXRpb24tcHJvcGVydHk6IGJhY2tncm91bmQtY29sb3IsIHBhZGRpbmctYmxvY2stZW5kO1xuXG4gICY6ZGlzYWJsZWQge1xuICAgIC0tc2JiLWNhbGVuZGFyLWNvbnRyb2wtdmlldy1jaGFuZ2UtYmFja2dyb3VuZDogbGlnaHQtZGFyayhcbiAgICAgIHZhcigtLXNiYi1jb2xvci1taWxrKSxcbiAgICAgIHZhcigtLXNiYi1jb2xvci1hbnRocmFjaXRlKVxuICAgICk7XG4gICAgLS1zYmItY2FsZW5kYXItY29udHJvbC12aWV3LWNoYW5nZS1jb2xvcjogbGlnaHQtZGFyayhcbiAgICAgIHZhcigtLXNiYi1jb2xvci1ncmFuaXRlKSxcbiAgICAgIHZhcigtLXNiYi1jb2xvci1hbHVtaW5pdW0pXG4gICAgKTtcblxuICAgIGN1cnNvcjogdW5zZXQ7XG4gIH1cblxuICAmOmZvY3VzLXZpc2libGUge1xuICAgIEBpbmNsdWRlIHNiYi5mb2N1cy1vdXRsaW5lO1xuXG4gICAgb3V0bGluZS1vZmZzZXQ6IHZhcigtLXNiYi1zcGFjaW5nLWZpeGVkLTF4KTtcbiAgfVxuXG4gIEBpbmNsdWRlIHNiYi5ob3Zlci1tcSB7XG4gICAgJjpub3QoOmFjdGl2ZSwgOmRpc2FibGVkKTpob3ZlciB7XG4gICAgICBwYWRkaW5nLWJsb2NrLWVuZDogdmFyKC0tc2JiLWNhbGVuZGFyLWNlbGwtaG92ZXItc2hpZnQpO1xuICAgIH1cbiAgfVxuXG4gICY6bm90KDpkaXNhYmxlZCk6YWN0aXZlIHtcbiAgICAtLXNiYi1jYWxlbmRhci1jb250cm9sLXZpZXctY2hhbmdlLWJhY2tncm91bmQ6IHZhcigtLXNiYi1iYWNrZ3JvdW5kLWNvbG9yLTMpO1xuICB9XG59XG5cbi5zYmItY2FsZW5kYXJfX3RhYmxlLW1vbnRoLXZpZXcsXG4uc2JiLWNhbGVuZGFyX190YWJsZS15ZWFyLXZpZXcge1xuICAtLXNiYi1jYWxlbmRhci10YWJsZS1jb2x1bW4tc3BhY2VzOiA2O1xufVxuXG4uc2JiLWNhbGVuZGFyX190YWJsZS1vdmVyZmxvdy1icmVhayB7XG4gIC8vIFdlIGhhdmUgdG8gY3V0IHRoZSBuZWdhdGl2ZSBtYXJnaW4gd2hpY2ggaXMgYnVpbHQgYnkgdGhlIGNvbnRhaW5pbmcgc2JiLWNhbGVuZGFyX190YWJsZS1jb250YWluZXIuXG4gIC8vIE92ZXJmbG93OiBoaWRkZW4gd291bGQgY3V0IHRoZSBmb2N1cyBvdXRsaW5lLiBUaGVyZWZvcmUgd2UgdGFrZSBgY29udGFpbmAgaGVyZS5cbiAgY29udGFpbjogbGF5b3V0O1xufVxuXG4uc2JiLWNhbGVuZGFyX190YWJsZS1jb250YWluZXIge1xuICBkaXNwbGF5OiBmbGV4O1xuICBnYXA6IHZhcigtLXNiYi1jYWxlbmRhci10YWJsZXMtZ2FwKTtcbiAgbWFyZ2luLWlubGluZTogdmFyKC0tc2JiLWNhbGVuZGFyLW1hcmdpbik7XG5cbiAgLy8gVGhlIHBhZGRpbmcgb2YgdGhlIGZpcnN0IGFuZCBsYXN0IGNvbHVtbiBzaG91bGQgbm90IGJlIHZpc2libGUgaWYgY2FsZW5kYXIgaXMgc3RyZXRjaGVkLlxuICAvLyBUaGVyZWZvcmUgd2UgbmVlZCBhIG5lZ2F0aXZlIGlubGluZSBtYXJnaW4uXG4gIC8vIEFzIHdlIGRvbid0IHdhbnQgdG8gc3F1ZWV6ZSwgdGhlIG1hcmdpbiBzaG91bGQgbmV2ZXIgYmUgZ3JlYXRlciB0aGFuIHplcm8uXG5cbiAgLy8gTWluIHdpZHRoIGlzIGVxdWFscyB0byB0aGUgbm9ybWFsIHdpZHRoIG9mIHRoZSBjYWxlbmRhclxuICAtLXNiYi1jYWxlbmRhci1taW4td2lkdGg6IGNhbGMoNyAqIHZhcigtLXNiYi1jYWxlbmRhci1jZWxsLXNpemUpKTtcblxuICAvLyBUaGUgb3ZlcmZsb3cgdmFyaWFibGUgaXMgZXF1YWxzIHRoZSBkaWZmZXJlbmNlIGJldHdlZW4gdGhlIGFjdHVhbCB3aWR0aCBhbmQgdGhlIG1pbiB3aWR0aC5cbiAgLS1zYmItY2FsZW5kYXItb3ZlcmZsb3c6IGNhbGMoMTAwJSAtIHZhcigtLXNiYi1jYWxlbmRhci1taW4td2lkdGgpKTtcblxuICAvLyBUaGUgc3RhcnQgb2Zmc2V0IGlzIG5lZ2F0aXZlIG1hcmdpbiB3aGljaCBzaG91bGQgb3ZlcmxhcCB0aGUgcGFyZW50IGNvbnRhaW5lci4gU2hvdWxkIG5ldmVyIGJlIGEgcG9zaXRpdmUgdmFsdWUuXG4gIC0tc2JiLWNhbGVuZGFyLXN0YXJ0LW9mZnNldDogbWluKFxuICAgIDBweCxcbiAgICAtMSAqICh2YXIoLS1zYmItY2FsZW5kYXItb3ZlcmZsb3cpIC8gdmFyKC0tc2JiLWNhbGVuZGFyLXRhYmxlLWNvbHVtbi1zcGFjZXMpKVxuICApO1xuICAtLXNiYi1jYWxlbmRhci1tYXJnaW46IHZhcigtLXNiYi1jYWxlbmRhci1zdGFydC1vZmZzZXQpO1xuXG4gIDpob3N0KDpzdGF0ZSh3aWRlKSkgJiB7XG4gICAgLS1zYmItY2FsZW5kYXItbWluLXdpZHRoOiBjYWxjKFxuICAgICAgMiAqIDcgKiB2YXIoLS1zYmItY2FsZW5kYXItY2VsbC1zaXplKSArIHZhcigtLXNiYi1jYWxlbmRhci10YWJsZXMtZ2FwKVxuICAgICk7XG4gICAgLS1zYmItY2FsZW5kYXItbWFyZ2luOiBjYWxjKDAuNSAqIHZhcigtLXNiYi1jYWxlbmRhci1zdGFydC1vZmZzZXQpKTtcbiAgfVxuXG4gIDpob3N0KFtvcmllbnRhdGlvbj0naG9yaXpvbnRhbCddW3dlZWstbnVtYmVyc10pICYge1xuICAgIC0tc2JiLWNhbGVuZGFyLW1pbi13aWR0aDogY2FsYyg4ICogdmFyKC0tc2JiLWNhbGVuZGFyLWNlbGwtc2l6ZSkpO1xuICB9XG5cbiAgOmhvc3QoW29yaWVudGF0aW9uPSdob3Jpem9udGFsJ11bd2Vlay1udW1iZXJzXTpzdGF0ZSh3aWRlKSkgJiB7XG4gICAgLS1zYmItY2FsZW5kYXItbWluLXdpZHRoOiBjYWxjKFxuICAgICAgMiAqIDggKiB2YXIoLS1zYmItY2FsZW5kYXItY2VsbC1zaXplKSArIHZhcigtLXNiYi1jYWxlbmRhci10YWJsZXMtZ2FwKVxuICAgICk7XG4gIH1cblxuICAvLyBUaGUgY29udGFpbmVyJ3MgbWluLXdpZHRoIGlzIHNldCBpbiB2ZXJ0aWNhbCBtb2RlO1xuICAvLyB0aGUgdmFsdWUgYGNhbGMoNyAqIHZhcigtLXNiYi1jYWxlbmRhci1jZWxsLXNpemUpKWAgaXMgZmluZVxuICAvLyBjb25zaWRlcmluZyB0aGF0IHRoZSBtYXhpbXVtIG51bWJlciBvZiB3ZWVrcyBpbiBhIG1vbnRoIGlzIDYsIHBsdXMgMSBmb3IgdGhlIGhlYWRlciBhdCB0aGUgdGFibGUncyBsZWZ0IHNpZGUuXG4gIDpob3N0KFtvcmllbnRhdGlvbj0ndmVydGljYWwnXSkgJiB7XG4gICAgbWluLXdpZHRoOiB2YXIoLS1zYmItY2FsZW5kYXItbWluLXdpZHRoKTtcblxuICAgIC8vIFRoZSB2YXJpYWJsZSB0aGF0IGRlZmluZXMgdGhlIG1hcmdpbiBpcyBzZXQgdG8gMCB0byBjb3JyZWN0bHkgZGlzcGxheSB0aGUgdGFibGUgaWYgY29uc3VtZXJzIHNldCBhIGN1c3RvbSB3aWR0aC5cbiAgICAtLXNiYi1jYWxlbmRhci1zdGFydC1vZmZzZXQ6IDA7XG4gIH1cblxuICAvLyBUaGUgbWluLXdpZHRoIGluIHdpZGUgaXMgY2FsY3VsYXRlZCBhcyAobWF4IG51bWJlciBvZiB3ZWVrcyBpbiBhIG1vbnRoKSAqIChkaXNwbGF5ZWQgbW9udGhzKSArIChoZWFkZXIgY29sdW1uKSA9IDYgKiAyICsgMSA9IDEzLlxuICA6aG9zdChbb3JpZW50YXRpb249J3ZlcnRpY2FsJ106c3RhdGUod2lkZSkpICYge1xuICAgIC0tc2JiLWNhbGVuZGFyLW1pbi13aWR0aDogY2FsYyhcbiAgICAgIDEzICogdmFyKC0tc2JiLWNhbGVuZGFyLWNlbGwtc2l6ZSkgKyB2YXIoLS1zYmItY2FsZW5kYXItdGFibGVzLWdhcClcbiAgICApO1xuICB9XG59XG5cbi5zYmItY2FsZW5kYXJfX3RhYmxlIHtcbiAgd2lkdGg6IDEwMCU7XG4gIGJvcmRlci1jb2xsYXBzZTogY29sbGFwc2U7XG4gIGhlaWdodDogbWF4LWNvbnRlbnQ7XG5cbiAgYW5pbWF0aW9uOiB7XG4gICAgbmFtZTogc2hvdztcbiAgICBkdXJhdGlvbjogdmFyKC0tc2JiLWNhbGVuZGFyLXRhYmxlLWFuaW1hdGlvbi1kdXJhdGlvbik7XG4gIH1cblxuICAmLnNiYi1jYWxlbmRhcl9fdGFibGUtaGlkZSB7XG4gICAgLS1zYmItY2FsZW5kYXItY2VsbC10cmFuc2l0aW9uLWR1cmF0aW9uOiAwbXM7XG5cbiAgICBhbmltYXRpb246IHtcbiAgICAgIG5hbWU6IGhpZGU7XG4gICAgICBkdXJhdGlvbjogdmFyKC0tc2JiLWNhbGVuZGFyLXRhYmxlLWFuaW1hdGlvbi1kdXJhdGlvbik7XG4gICAgfVxuICB9XG5cbiAgOmhvc3QoOm5vdCg6c3RhdGUod2lkZSkpKSAmIHtcbiAgICAvLyBEdWUgdG8gYSBTYWZhcmkgaU9TIHJlbmRlcmluZyBidWcgd2UgbmVlZCB0byBkZWZpbmUgbWluLXdpZHRoIGFzIHdlbGwuXG4gICAgLy8gT3RoZXJ3aXNlLCBhZnRlciBvcmllbnRhdGlvbiBjaGFuZ2UsIHRoZXJlIGlzIGEgd3Jvbmcgd2lkdGggaWYgcGxhY2VkIGluIGFuIHNiYi1kaWFsb2cuXG4gICAgbWluLXdpZHRoOiAxMDAlO1xuICB9XG59XG5cbi5zYmItY2FsZW5kYXJfX3RhYmxlLWhlYWRlciB7XG4gIC8vIERlcGVuZGluZyB3aGVyZSBpbiB0aGUgRE9NIGl0J3MgdXNlZCwgaXQncyBwb3NzaWJsZSB0aGF0IGJyb3dzZXIgZGVmYXVsdHMgc2V0IHRleHQtYWxpZ24gdG8gbGVmdC5cbiAgdGV4dC1hbGlnbjogY2VudGVyO1xufVxuXG4uc2JiLWNhbGVuZGFyX190YWJsZS1ib2R5IHtcbiAgLy8gRGVwZW5kaW5nIHdoZXJlIGluIHRoZSBET00gaXQncyB1c2VkLCBpdCdzIHBvc3NpYmxlIHRoYXQgYnJvd3NlciBkZWZhdWx0cyBzZXQgdGV4dC1hbGlnbiB0byBsZWZ0LlxuICB0ZXh0LWFsaWduOiBjZW50ZXI7XG59XG5cbi5zYmItY2FsZW5kYXJfX3RhYmxlLWhlYWRlci1jZWxsLFxuLnNiYi1jYWxlbmRhcl9fdGFibGUtaGVhZGVyLWNlbGwtdmVydGljYWwge1xuICB3aWR0aDogdmFyKC0tc2JiLWNhbGVuZGFyLWNlbGwtc2l6ZSk7XG4gIGNvbG9yOiB2YXIoLS1zYmItY2FsZW5kYXItaGVhZGVyLWNvbG9yKTtcbiAgcGFkZGluZzogMDtcbiAgZm9udC1zaXplOiB2YXIoLS1zYmItdGV4dC1mb250LXNpemUteHMpO1xuICBmb250LXdlaWdodDogbm9ybWFsO1xuICBsZXR0ZXItc3BhY2luZzogdmFyKC0tc2JiLXR5cG8tbGV0dGVyLXNwYWNpbmctdGV4dCk7XG5cbiAgOmhvc3QoOm5vdChbbXVsdGlwbGVdKSkgJiB7XG4gICAgaGVpZ2h0OiB2YXIoLS1zYmItY2FsZW5kYXItY2VsbC1zaXplKTtcbiAgfVxufVxuXG4uc2JiLWNhbGVuZGFyX190YWJsZS1oZWFkZXItY2VsbC12ZXJ0aWNhbCB7XG4gIHBhZGRpbmctYmxvY2stZW5kOiB2YXIoLS1zYmItY2FsZW5kYXItY2VsbC1wYWRkaW5nLWJsb2NrLWVuZCk7XG59XG5cbi5zYmItY2FsZW5kYXJfX3RhYmxlLWRhdGEge1xuICBwb3NpdGlvbjogcmVsYXRpdmU7XG4gIHBhZGRpbmc6IDA7XG4gIHRleHQtYWxpZ246IGNlbnRlcjtcbn1cblxuLnNiYi1jYWxlbmRhcl9fZGF5LWNlbGwge1xuICBwYWRkaW5nLWJsb2NrLWVuZDogdmFyKC0tc2JiLWNhbGVuZGFyLWNlbGwtcGFkZGluZy1ibG9jay1lbmQpO1xufVxuXG5zYmItY2FsZW5kYXItZGF5IHtcbiAgOmhvc3QoOm5vdCg6c3RhdGUoZW5oYW5jZWQpKSkgJiB7XG4gICAgLS1zYmItY2FsZW5kYXItY2VsbC1qdXN0aWZ5LWNvbnRlbnQ6IGNlbnRlcjtcbiAgICAtLXNiYi1jYWxlbmRhci1kYXktaGVpZ2h0OiAje3NiYi5weC10by1yZW0tYnVpbGQoNDQpfTtcbiAgICAtLXNiYi1jYWxlbmRhci1kYXktZXh0cmEtZGlzcGxheTogbm9uZTtcbiAgICAtLXNiYi1jYWxlbmRhci1kYXktdmFsdWUtaGVpZ2h0OiB1bnNldDtcbiAgICAtLXNiYi1jYWxlbmRhci1kYXktY3Jvc3NlZC1vdXQtdG9wOiA1MCU7XG4gICAgLS1zYmItY2FsZW5kYXItZGF5LWNyb3NzZWQtb3V0LXRyYW5zbGF0ZTogdHJhbnNsYXRlKC01MCUsIC01MCUpIHJvdGF0ZSgtNDVkZWcpO1xuICB9XG59XG5cbkBrZXlmcmFtZXMgc2hvdyB7XG4gIGZyb20ge1xuICAgIG9wYWNpdHk6IDA7XG4gICAgdHJhbnNmb3JtOiB0cmFuc2xhdGVZKHZhcigtLXNiYi1jYWxlbmRhci10YWJsZS1hbmltYXRpb24tc2hpZnQpKTtcbiAgfVxuXG4gIHRvIHtcbiAgICBvcGFjaXR5OiAxO1xuICAgIHRyYW5zZm9ybTogdHJhbnNsYXRlWSgwJSk7XG4gIH1cbn1cblxuQGtleWZyYW1lcyBoaWRlIHtcbiAgZnJvbSB7XG4gICAgb3BhY2l0eTogMTtcbiAgICB0cmFuc2Zvcm06IHRyYW5zbGF0ZVkoMCUpO1xuICB9XG5cbiAgdG8ge1xuICAgIG9wYWNpdHk6IDA7XG4gICAgdHJhbnNmb3JtOiB0cmFuc2xhdGVZKHZhcigtLXNiYi1jYWxlbmRhci10YWJsZS1hbmltYXRpb24tc2hpZnQpKTtcbiAgfVxufVxuIiwiaW1wb3J0IHtcbiAgdHlwZSBDU1NSZXN1bHRHcm91cCxcbiAgaHRtbCxcbiAgaXNTZXJ2ZXIsXG4gIG5vdGhpbmcsXG4gIHR5cGUgUHJvcGVydHlWYWx1ZXMsXG4gIHR5cGUgVGVtcGxhdGVSZXN1bHQsXG59IGZyb20gJ2xpdCc7XG5pbXBvcnQgeyBwcm9wZXJ0eSwgc3RhdGUgfSBmcm9tICdsaXQvZGVjb3JhdG9ycy5qcyc7XG5cbmltcG9ydCB7IFNiYlNlY29uZGFyeUJ1dHRvbkVsZW1lbnQgfSBmcm9tICcuLi8uLi9idXR0b24ucHVyZS50cyc7XG5pbXBvcnQgeyBpc0Fycm93S2V5T3JQYWdlS2V5c1ByZXNzZWQgfSBmcm9tICcuLi8uLi9jb3JlL2ExMXkudHMnO1xuaW1wb3J0IHsgU2JiRWxlbWVudCB9IGZyb20gJy4uLy4uL2NvcmUvYmFzZS1lbGVtZW50cy50cyc7XG5pbXBvcnQgeyByZWFkQ29uZmlnIH0gZnJvbSAnLi4vLi4vY29yZS9jb25maWcudHMnO1xuaW1wb3J0IHtcbiAgU2JiTGFuZ3VhZ2VDb250cm9sbGVyLFxuICBTYmJNZWRpYU1hdGNoZXJDb250cm9sbGVyLFxuICBTYmJNZWRpYVF1ZXJ5QnJlYWtwb2ludExhcmdlQW5kQWJvdmUsXG59IGZyb20gJy4uLy4uL2NvcmUvY29udHJvbGxlcnMudHMnO1xuaW1wb3J0IHR5cGUgeyBEYXRlQWRhcHRlciB9IGZyb20gJy4uLy4uL2NvcmUvZGF0ZXRpbWUudHMnO1xuaW1wb3J0IHtcbiAgREFZU19QRVJfUk9XLFxuICBkZWZhdWx0RGF0ZUFkYXB0ZXIsXG4gIE1PTkRBWSxcbiAgTU9OVEhTX1BFUl9QQUdFLFxuICBNT05USFNfUEVSX1JPVyxcbiAgVEhVUlNEQVksXG4gIFRVRVNEQVksXG4gIFdFRE5FU0RBWSxcbiAgWUVBUlNfUEVSX1BBR0UsXG4gIFlFQVJTX1BFUl9ST1csXG59IGZyb20gJy4uLy4uL2NvcmUvZGF0ZXRpbWUudHMnO1xuaW1wb3J0IHsgZm9yY2VUeXBlLCBoYW5kbGVEaXN0aW5jdENoYW5nZSwgcGxhaW5EYXRlIH0gZnJvbSAnLi4vLi4vY29yZS9kZWNvcmF0b3JzLnRzJztcbmltcG9ydCB7XG4gIGkxOG5DYWxlbmRhckRhdGVTZWxlY3Rpb24sXG4gIGkxOG5DYWxlbmRhcldlZWtOdW1iZXIsXG4gIGkxOG5OZXh0TW9udGgsXG4gIGkxOG5OZXh0WWVhcixcbiAgaTE4bk5leHRZZWFyUmFuZ2UsXG4gIGkxOG5QcmV2aW91c01vbnRoLFxuICBpMThuUHJldmlvdXNZZWFyLFxuICBpMThuUHJldmlvdXNZZWFyUmFuZ2UsXG4gIGkxOG5ZZWFyTW9udGhTZWxlY3Rpb24sXG59IGZyb20gJy4uLy4uL2NvcmUvaTE4bi50cyc7XG5pbXBvcnQgdHlwZSB7IFNiYk9yaWVudGF0aW9uIH0gZnJvbSAnLi4vLi4vY29yZS9pbnRlcmZhY2VzLnRzJztcbmltcG9ydCB7IGJveFNpemluZ1N0eWxlcyB9IGZyb20gJy4uLy4uL2NvcmUvc3R5bGVzLnRzJztcbmltcG9ydCB7IFNiYkNhbGVuZGFyRGF5RWxlbWVudCB9IGZyb20gJy4uL2NhbGVuZGFyLWRheS9jYWxlbmRhci1kYXkuY29tcG9uZW50LnRzJztcbmltcG9ydCB7IFNiYkNhbGVuZGFyTW9udGhFbGVtZW50IH0gZnJvbSAnLi4vY2FsZW5kYXItbW9udGgvY2FsZW5kYXItbW9udGguY29tcG9uZW50LnRzJztcbmltcG9ydCB7IFNiYkNhbGVuZGFyV2Vla2RheUVsZW1lbnQgfSBmcm9tICcuLi9jYWxlbmRhci13ZWVrZGF5L2NhbGVuZGFyLXdlZWtkYXkuY29tcG9uZW50LnRzJztcbmltcG9ydCB7IFNiYkNhbGVuZGFyV2Vla251bWJlckVsZW1lbnQgfSBmcm9tICcuLi9jYWxlbmRhci13ZWVrbnVtYmVyL2NhbGVuZGFyLXdlZWtudW1iZXIuY29tcG9uZW50LnRzJztcbmltcG9ydCB7IFNiYkNhbGVuZGFyWWVhckVsZW1lbnQgfSBmcm9tICcuLi9jYWxlbmRhci15ZWFyL2NhbGVuZGFyLXllYXIuY29tcG9uZW50LnRzJztcbmltcG9ydCB0eXBlIHsgU2JiQ2FsZW5kYXJDZWxsQmFzZUVsZW1lbnQgfSBmcm9tICcuLi9jb21tb24udHMnO1xuXG5pbXBvcnQgc3R5bGUgZnJvbSAnLi9jYWxlbmRhci5zY3NzP2xpdCZpbmxpbmUnO1xuXG5pbXBvcnQgJy4uLy4uL2ljb24udHMnO1xuaW1wb3J0ICcuLi8uLi9zY3JlZW4tcmVhZGVyLW9ubHkudHMnO1xuXG4vLyBUT0RPKGJyZWFraW5nLWNoYW5nZSk6IFJlbW92ZSBjYWxsIHRvIGRlZmluZS5cblNiYlNlY29uZGFyeUJ1dHRvbkVsZW1lbnQuZGVmaW5lKCk7XG5cblNiYkNhbGVuZGFyRGF5RWxlbWVudC5kZWZpbmUoKTtcblNiYkNhbGVuZGFyTW9udGhFbGVtZW50LmRlZmluZSgpO1xuU2JiQ2FsZW5kYXJZZWFyRWxlbWVudC5kZWZpbmUoKTtcblNiYkNhbGVuZGFyV2Vla2RheUVsZW1lbnQuZGVmaW5lKCk7XG5TYmJDYWxlbmRhcldlZWtudW1iZXJFbGVtZW50LmRlZmluZSgpO1xuXG5leHBvcnQgY2xhc3MgU2JiTW9udGhDaGFuZ2VFdmVudCBleHRlbmRzIEV2ZW50IHtcbiAgcHJpdmF0ZSByZWFkb25seSBfcmFuZ2U6IHJlYWRvbmx5IERheVtdO1xuXG4gIHB1YmxpYyBnZXQgcmFuZ2UoKTogcmVhZG9ubHkgRGF5W10ge1xuICAgIHJldHVybiB0aGlzLl9yYW5nZTtcbiAgfVxuXG4gIHB1YmxpYyBjb25zdHJ1Y3RvcihyYW5nZTogcmVhZG9ubHkgRGF5W10pIHtcbiAgICBzdXBlcignbW9udGhjaGFuZ2UnLCB7IGJ1YmJsZXM6IHRydWUsIGNvbXBvc2VkOiB0cnVlIH0pO1xuICAgIHRoaXMuX3JhbmdlID0gT2JqZWN0LmZyZWV6ZShyYW5nZSB8fCBbXSk7XG4gIH1cbn1cblxuLyoqXG4gKiBQYXJhbWV0ZXJzIG5lZWRlZCBpbiB5ZWFyIGFuZCBtb250aCB2aWV3cyB0byBjb3JyZWN0bHkgY2FsY3VsYXRlIHRoZSBuZXh0IGVsZW1lbnQgaW4ga2V5Ym9hcmQgbmF2aWdhdGlvbi5cbiAqXG4gKiBUaGUgY2VsbCdzIGluZGV4IGFuZCB0aGUgZWxlbWVudCdzIGluZGV4IGluIGl0cyBtb250aCAvIHllYXIgYmF0Y2ggbXVzdCBiZSBkaXN0aW5ndWlzaGVkOlxuICogdGhlIGZpcnN0IGlzIHRoZSBpbmRleCBvZiB0aGUgZWxlbWVudCBpbiB0aGUgYXJyYXkgb2YgYWxsIHRoZSByZW5kZXJlZCBjZWxscywgd2hpbGUgdGhlIHNlY29uZCBpcyB0aGUgaW5kZXggb2YgdGhlIGVsZW1lbnQgcmVsYXRpdmUgdG8gaXRzIHRhYmxlLlxuICogSW4gbm9uLXdpZGUgbW9kZSwgdGhlIHd0byBhcmUgdGhlIHNhbWUsIHdoaWxlIGluIHdpZGUgbW9kZSB0aGUgY2VsbCdzIGluZGV4IGNhbiBnbyBmcm9tIDAgdG8gNDcgZm9yIHllYXJzIGFuZCBmcm9tIDAgdG8gMjMgZm9yIG1vbnRocyxcbiAqIHdoaWxlIHRoZSBlbGVtZW50IGluZGV4IGdvZXMgZnJvbSAwIHRvLCByZXNwZWN0aXZlbHksIDIzIGFuZCAxMS5cbiAqL1xuaW50ZXJmYWNlIENhbGVuZGFyS2V5Ym9hcmROYXZpZ2F0aW9uTW9udGhZZWFyVmlld3NQYXJhbWV0ZXJzIHtcbiAgLyoqIFRoZSBlbGVtZW50IGluZGV4IHdpdGhpbiBpdHMgeWVhciBvciBtb250aCByYW5nZS4gKi9cbiAgZWxlbWVudEluZGV4Rm9yV2lkZU1vZGU6IG51bWJlcjtcbiAgLyoqIEluIHdpZGUgbW9kZSwgdGhlIGluZGV4IG9mIHRoZSBmaXJzdCBlbGVtZW50IGluIHRoZSBzZWNvbmQgcGFuZWwsIG9yLCBhbHRlcm5hdGl2ZWx5LCB0aGUgbnVtYmVyIG9mIGVsZW1lbnRzIGluIHRoZSBmaXJzdCBwYW5lbC4gKi9cbiAgb2Zmc2V0Rm9yV2lkZU1vZGU6IG51bWJlcjtcbiAgLyoqIFRoZSBpbmRleCBvZiB0aGUgbGFzdCBlbGVtZW50IHdpdGhpbiB0aGUgZWxlbWVudCdzIG1vbnRoIChvciB5ZWFyIHJhbmdlKS4gKi9cbiAgbGFzdEVsZW1lbnRJbmRleEZvcldpZGVNb2RlOiBudW1iZXI7XG4gIC8qKiBUaGUgbnVtYmVyIG9mIGNlbGxzIGRpc3BsYXllZCBpbiBhIHNpbmdsZSByb3csIGRlcGVuZGluZyBvbiB0aGUgcmVuZGVyZWQgdmlldy4gKi9cbiAgdmVydGljYWxPZmZzZXQ6IG51bWJlcjtcbn1cblxuLyoqXG4gKiBQYXJhbWV0ZXJzIG5lZWRlZCBpbiBkYXkgdmlldyB0byBjb3JyZWN0bHkgY2FsY3VsYXRlIHRoZSBuZXh0IGVsZW1lbnQgaW4ga2V5Ym9hcmQgbmF2aWdhdGlvbi5cbiAqXG4gKiBJbiBvcmllbnRhdGlvbj0ndmVydGljYWwnLCBpdCdzIG5vdCBwb3NzaWJsZSB0byByZWx5IG9uIGFueSBhcnJheS9pbmRleCB0byBjYWxjdWxhdGUgdGhlIGVsZW1lbnQgdG8gbmF2aWdhdGUgdG8sXG4gKiBzbyBjYWxjdWxhdGlvbnMgb24gZGF0ZXMgbXVzdCBiZSBkb25lLCB3aGljaCBzaG91bGQgY29uc2lkZXIgdmlldyBib3VuZGFyaWVzLCBvZmZzZXRzIGFuZCBtb250aCdzIGxlbmd0aC5cbiAqL1xuaW50ZXJmYWNlIENhbGVuZGFyS2V5Ym9hcmROYXZpZ2F0aW9uRGF5Vmlld1BhcmFtZXRlcnMge1xuICAvKiogVGhlIGZpcnN0IGRheSByZW5kZXJlZC4gKi9cbiAgZmlyc3REYXlJblZpZXc6IHN0cmluZyB8IG51bGw7XG4gIC8qKiBUaGUgbGFzdCBkYXkgcmVuZGVyZWQuIEl0IGRlcGVuZHMgb24gdGhlICd3aWRlJyB2YWx1ZS4gKi9cbiAgbGFzdERheUluVmlldzogc3RyaW5nIHwgbnVsbDtcbiAgLyoqIFRoZSBvZmZzZXQgZnJvbSB0aGUgZmlyc3QgZGF5IG9mIHRoZSB3ZWVrIChNb25kYXkpIG9mIHRoZSBmaXJzdCByZW5kZXJlZCBtb250aC4gKi9cbiAgZmlyc3RNb250aE9mZnNldDogbnVtYmVyO1xuICAvKiogVGhlIG51bWJlciBvZiBkYXlzIGluIHRoZSBmaXJzdCByZW5kZXJlZCBtb250aC4gKi9cbiAgZmlyc3RNb250aExlbmd0aDogbnVtYmVyO1xuICAvKiogVGhlIG9mZnNldCBmcm9tIHRoZSBmaXJzdCBkYXkgb2YgdGhlIHdlZWsgKE1vbmRheSkgb2YgdGhlIHNlY29uZCByZW5kZXJlZCBtb250aC4gSWYgd2lkZSBpcyBmYWxzZSwgaXQncyBlcXVhbCB0byB6ZXJvLiAqL1xuICBzZWNvbmRNb250aE9mZnNldDogbnVtYmVyO1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIERheTxUID0gRGF0ZT4ge1xuICAvKiogRGF0ZSBhcyBJU08gc3RyaW5nLiAqL1xuICB2YWx1ZTogc3RyaW5nO1xuICBkYXlWYWx1ZTogc3RyaW5nO1xuICBtb250aFZhbHVlOiBzdHJpbmc7XG4gIHllYXJWYWx1ZTogc3RyaW5nO1xuICBkYXRlVmFsdWU6IFQ7XG4gIHdlZWtWYWx1ZTogbnVtYmVyO1xuICB3ZWVrRGF5VmFsdWU6IG51bWJlcjtcbn1cblxuLyoqIEBkZXByZWNhdGVkIHVzZSBNb250aENlbGwgKi9cbmV4cG9ydCBpbnRlcmZhY2UgTW9udGgge1xuICB2YWx1ZTogc3RyaW5nO1xuICBsb25nVmFsdWU6IHN0cmluZztcbiAgbW9udGhWYWx1ZTogbnVtYmVyO1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIE1vbnRoQ2VsbCB7XG4gIHZhbHVlOiBzdHJpbmc7XG4gIG1vbnRoVmFsdWU6IG51bWJlcjtcbn1cblxuZXhwb3J0IGludGVyZmFjZSBXZWVrZGF5IHtcbiAgbG9uZzogc3RyaW5nO1xuICBuYXJyb3c6IHN0cmluZztcbn1cblxuZXhwb3J0IHR5cGUgQ2FsZW5kYXJWaWV3ID0gJ2RheScgfCAnbW9udGgnIHwgJ3llYXInO1xuXG4vKipcbiAqIEl0IGRpc3BsYXlzIGEgY2FsZW5kYXIgd2hpY2ggYWxsb3dzIGNob29zaW5nIGEgZGF0ZS5cbiAqXG4gKiBAc2xvdCAtIFVzZSB0aGUgdW5uYW1lZCBzbG90IHRvIGFkZCBjdXN0b21pemVkIGBzYmItY2FsZW5kYXItZGF5YCBlbGVtZW50cy5cbiAqL1xuZXhwb3J0IGNsYXNzIFNiYkNhbGVuZGFyRWxlbWVudDxUID0gRGF0ZT4gZXh0ZW5kcyBTYmJFbGVtZW50IHtcbiAgcHVibGljIHN0YXRpYyBvdmVycmlkZSByZWFkb25seSBlbGVtZW50TmFtZTogc3RyaW5nID0gJ3NiYi1jYWxlbmRhcic7XG4gIHB1YmxpYyBzdGF0aWMgb3ZlcnJpZGUgc3R5bGVzOiBDU1NSZXN1bHRHcm91cCA9IFtib3hTaXppbmdTdHlsZXMsIHN0eWxlXTtcbiAgcHVibGljIHN0YXRpYyByZWFkb25seSBldmVudHMgPSB7XG4gICAgZGF0ZXNlbGVjdGVkOiAnZGF0ZXNlbGVjdGVkJyxcbiAgICBtb250aGNoYW5nZTogJ21vbnRoY2hhbmdlJyxcbiAgfSBhcyBjb25zdDtcblxuICAvKiogSWYgc2V0IHRvIHRydWUsIHR3byBtb250aHMgYXJlIGRpc3BsYXllZCAqL1xuICBAZm9yY2VUeXBlKClcbiAgQHByb3BlcnR5KHsgdHlwZTogQm9vbGVhbiB9KVxuICBwdWJsaWMgYWNjZXNzb3Igd2lkZTogYm9vbGVhbiA9IGZhbHNlO1xuXG4gIC8qKiBUaGUgaW5pdGlhbCB2aWV3IG9mIHRoZSBjYWxlbmRhciB3aGljaCBzaG91bGQgYmUgZGlzcGxheWVkIG9uIG9wZW5pbmcuICovXG4gIEBwcm9wZXJ0eSgpIHB1YmxpYyBhY2Nlc3NvciB2aWV3OiBDYWxlbmRhclZpZXcgPSAnZGF5JztcblxuICAvKipcbiAgICogVGhlIG1pbmltdW0gdmFsaWQgZGF0ZS4gQWNjZXB0cyBhIGRhdGUgb2JqZWN0IG9yIG51bGwuXG4gICAqIEFjY2VwdHMgYW4gSVNPODYwMSBmb3JtYXR0ZWQgc3RyaW5nIChlLmcuIDIwMjQtMTItMjQpIGFzIGF0dHJpYnV0ZS5cbiAgICovXG4gIEBwbGFpbkRhdGUoKVxuICBAcHJvcGVydHkoKVxuICBwdWJsaWMgYWNjZXNzb3IgbWluOiBUIHwgbnVsbCA9IG51bGw7XG5cbiAgLyoqXG4gICAqIFRoZSBtYXhpbXVtIHZhbGlkIGRhdGUuIEFjY2VwdHMgYSBkYXRlIG9iamVjdCBvciBudWxsLlxuICAgKiBBY2NlcHRzIGFuIElTTzg2MDEgZm9ybWF0dGVkIHN0cmluZyAoZS5nLiAyMDI0LTEyLTI0KSBhcyBhdHRyaWJ1dGUuXG4gICAqL1xuICBAcGxhaW5EYXRlKClcbiAgQHByb3BlcnR5KClcbiAgcHVibGljIGFjY2Vzc29yIG1heDogVCB8IG51bGwgPSBudWxsO1xuXG4gIC8qKiBXaGV0aGVyIHRoZSBjYWxlbmRhciBhbGxvd3MgZm9yIG11bHRpcGxlIGRhdGUgc2VsZWN0aW9uLiAqL1xuICBAZm9yY2VUeXBlKClcbiAgQGhhbmRsZURpc3RpbmN0Q2hhbmdlKChlOiBTYmJDYWxlbmRhckVsZW1lbnQ8VD4sIG5ld1ZhbHVlOiBib29sZWFuKSA9PlxuICAgIGUuX29uTXVsdGlwbGVDaGFuZ2VkKG5ld1ZhbHVlKSxcbiAgKVxuICBAcHJvcGVydHkoeyB0eXBlOiBCb29sZWFuIH0pXG4gIHB1YmxpYyBhY2Nlc3NvciBtdWx0aXBsZTogYm9vbGVhbiA9IGZhbHNlO1xuXG4gIC8qKlxuICAgKiBUaGUgc2VsZWN0ZWQgZGF0ZTogYWNjZXB0cyBhIGRhdGUgb2JqZWN0LCBvciwgaWYgYG11bHRpcGxlYCwgYW4gYXJyYXkgb2YgZGF0ZXMuXG4gICAqL1xuICBAcHJvcGVydHkoKVxuICBwdWJsaWMgc2V0IHNlbGVjdGVkKHZhbHVlOiBUIHwgVFtdIHwgbnVsbCkge1xuICAgIGlmIChBcnJheS5pc0FycmF5KHZhbHVlKSkge1xuICAgICAgdGhpcy5fc2VsZWN0ZWQgPSB2YWx1ZVxuICAgICAgICAubWFwKChkYXRlTGlrZTogVCkgPT5cbiAgICAgICAgICB0aGlzLl9kYXRlQWRhcHRlci5nZXRWYWxpZERhdGVPck51bGwodGhpcy5fZGF0ZUFkYXB0ZXIuZGVzZXJpYWxpemUoZGF0ZUxpa2UpKSxcbiAgICAgICAgKVxuICAgICAgICAuZmlsdGVyKChkYXRlOiBUIHwgbnVsbCk6IGRhdGUgaXMgVCA9PiBkYXRlICE9PSBudWxsKVxuICAgICAgICAuZmlsdGVyKFxuICAgICAgICAgIChkYXRlOiBUKSA9PlxuICAgICAgICAgICAgIXRoaXMuX2lzRGF5SW5SYW5nZSh0aGlzLl9kYXRlQWRhcHRlci50b0lzbzg2MDEoZGF0ZSkpIHx8IHRoaXMuX2RhdGVGaWx0ZXIoZGF0ZSksXG4gICAgICAgICk7XG4gICAgfSBlbHNlIHtcbiAgICAgIGNvbnN0IHNlbGVjdGVkRGF0ZSA9IHRoaXMuX2RhdGVBZGFwdGVyLmdldFZhbGlkRGF0ZU9yTnVsbChcbiAgICAgICAgdGhpcy5fZGF0ZUFkYXB0ZXIuZGVzZXJpYWxpemUodmFsdWUpLFxuICAgICAgKTtcbiAgICAgIGlmIChcbiAgICAgICAgISFzZWxlY3RlZERhdGUgJiZcbiAgICAgICAgKCF0aGlzLl9pc0RheUluUmFuZ2UodGhpcy5fZGF0ZUFkYXB0ZXIudG9Jc284NjAxKHNlbGVjdGVkRGF0ZSkpIHx8XG4gICAgICAgICAgdGhpcy5fZGF0ZUZpbHRlcihzZWxlY3RlZERhdGUpKVxuICAgICAgKSB7XG4gICAgICAgIHRoaXMuX3NlbGVjdGVkID0gc2VsZWN0ZWREYXRlO1xuICAgICAgfSBlbHNlIHtcbiAgICAgICAgdGhpcy5fc2VsZWN0ZWQgPSBudWxsO1xuICAgICAgfVxuICAgIH1cbiAgfVxuICBwdWJsaWMgZ2V0IHNlbGVjdGVkKCk6IFQgfCBUW10gfCBudWxsIHtcbiAgICByZXR1cm4gdGhpcy5fc2VsZWN0ZWQ7XG4gIH1cbiAgQHN0YXRlKCkgcHJpdmF0ZSBhY2Nlc3NvciBfc2VsZWN0ZWQ6IFQgfCBUW10gfCBudWxsID0gbnVsbDtcblxuICAvKiogQSBmdW5jdGlvbiB1c2VkIHRvIGZpbHRlciBvdXQgZGF0ZXMuICovXG4gIEBwcm9wZXJ0eSh7IGF0dHJpYnV0ZTogJ2RhdGUtZmlsdGVyJyB9KVxuICBwdWJsaWMgYWNjZXNzb3IgZGF0ZUZpbHRlcjogKChkYXRlOiBUIHwgbnVsbCkgPT4gYm9vbGVhbikgfCBudWxsID0gbnVsbDtcblxuICAvKiogVGhlIG9yaWVudGF0aW9uIG9mIGRheXMgaW4gdGhlIGNhbGVuZGFyLiAqL1xuICBAcHJvcGVydHkoeyByZWZsZWN0OiB0cnVlIH0pIHB1YmxpYyBhY2Nlc3NvciBvcmllbnRhdGlvbjogU2JiT3JpZW50YXRpb24gPSAnaG9yaXpvbnRhbCc7XG5cbiAgLyoqIFdoZXRoZXIgaXQgaGFzIHRvIGRpc3BsYXkgdGhlIHdlZWsgbnVtYmVycyBpbiBhZGRpdGlvbiB0byB3ZWVrIGRheXMuICovXG4gIEBmb3JjZVR5cGUoKVxuICBAcHJvcGVydHkoeyBhdHRyaWJ1dGU6ICd3ZWVrLW51bWJlcnMnLCB0eXBlOiBCb29sZWFuIH0pXG4gIHB1YmxpYyBhY2Nlc3NvciB3ZWVrTnVtYmVyczogYm9vbGVhbiA9IGZhbHNlO1xuXG4gIHByaXZhdGUgX2RhdGVBZGFwdGVyOiBEYXRlQWRhcHRlcjxUPiA9IHJlYWRDb25maWcoKS5kYXRldGltZT8uZGF0ZUFkYXB0ZXIgPz8gZGVmYXVsdERhdGVBZGFwdGVyO1xuXG4gIC8qKiBUaGUgY3VycmVudGx5IGFjdGl2ZSBkYXRlLiAqL1xuICBAc3RhdGUoKSBwcml2YXRlIGFjY2Vzc29yIF9hY3RpdmVEYXRlOiBUID0gdGhpcy5fZGF0ZUFkYXB0ZXIudG9kYXkoKTtcblxuICAvKiogVGhlIGN1cnJlbnQgd2lkZSBwcm9wZXJ0eSBjb25zaWRlcmluZyBwcm9wZXJ0eSB2YWx1ZSBhbmQgYnJlYWtwb2ludHMuIEZyb20gemVybyB0byBzbWFsbCBgd2lkZWAgaGFzIGFsd2F5cyB0byBiZSBmYWxzZS4gKi9cbiAgQHN0YXRlKClcbiAgcHJpdmF0ZSBzZXQgX3dpZGUod2lkZTogYm9vbGVhbikge1xuICAgIHRoaXMudG9nZ2xlU3RhdGUoJ3dpZGUnLCB3aWRlKTtcbiAgICB0aGlzLl93aWRlSW50ZXJuYWwgPSB3aWRlO1xuICB9XG4gIHByaXZhdGUgZ2V0IF93aWRlKCk6IGJvb2xlYW4ge1xuICAgIHJldHVybiB0aGlzLl93aWRlSW50ZXJuYWw7XG4gIH1cbiAgLy8gV2Uga2VlcCB0aGUgc3RhdGUgaW4gYSBmaWVsZCBiZWNhdXNlIG9mIHRoZSBXZWJLaXQgYnVnIGh0dHBzOi8vYnVncy53ZWJraXQub3JnL3Nob3dfYnVnLmNnaT9pZD0zMDM0NjcuXG4gIC8vIFRPRE86IHJlLWNoZWNrIHdoZXRoZXIgZmllbGQgaXMgbmVlZGVkXG4gIHByaXZhdGUgX3dpZGVJbnRlcm5hbDogYm9vbGVhbiA9IGZhbHNlO1xuXG4gIEBzdGF0ZSgpIHByaXZhdGUgYWNjZXNzb3IgX2NhbGVuZGFyVmlldzogQ2FsZW5kYXJWaWV3ID0gJ2RheSc7XG5cbiAgcHJpdmF0ZSBfbmV4dENhbGVuZGFyVmlldzogQ2FsZW5kYXJWaWV3ID0gJ2RheSc7XG5cbiAgLyoqIEluZm9ybWF0aW9uIGFib3V0IHRoZSByZW5kZXJlZCBkYXkgdmlldzsgdXNlZCBpbiBrZXlib2FyZCBuYXZpZ2F0aW9uLiAqL1xuICBwcml2YXRlIF9rZXlib2FyZE5hdmlnYXRpb25EYXlWaWV3UGFyYW1ldGVyczogQ2FsZW5kYXJLZXlib2FyZE5hdmlnYXRpb25EYXlWaWV3UGFyYW1ldGVycyA9IHtcbiAgICBmaXJzdERheUluVmlldzogbnVsbCxcbiAgICBsYXN0RGF5SW5WaWV3OiBudWxsLFxuICAgIGZpcnN0TW9udGhPZmZzZXQ6IDAsXG4gICAgZmlyc3RNb250aExlbmd0aDogMCxcbiAgICBzZWNvbmRNb250aE9mZnNldDogMCxcbiAgfTtcblxuICAvKiogQSBsaXN0IG9mIGRheXMsIGluIHR3byBmb3JtYXRzIChsb25nIGFuZCBzaW5nbGUgY2hhcikuICovXG4gIHByaXZhdGUgX3dlZWtkYXlzITogV2Vla2RheVtdO1xuXG4gIC8qKiBHcmlkIG9mIGNhbGVuZGFyIGNlbGxzIHJlcHJlc2VudGluZyB0aGUgZGF0ZXMgb2YgdGhlIG1vbnRoLiAqL1xuICBwcml2YXRlIF93ZWVrczogRGF5PFQ+W11bXSA9IFtdO1xuXG4gIC8qKiBHcmlkIG9mIGNhbGVuZGFyIGNlbGxzIHJlcHJlc2VudGluZyBtb250aHMuICovXG4gIHByaXZhdGUgX21vbnRocyE6IE1vbnRoQ2VsbFtdW107XG5cbiAgLyoqIEdyaWQgb2YgY2FsZW5kYXIgY2VsbHMgcmVwcmVzZW50aW5nIHllYXJzLiAqL1xuICBwcml2YXRlIF95ZWFycyE6IG51bWJlcltdW107XG5cbiAgLyoqIEdyaWQgb2YgY2FsZW5kYXIgY2VsbHMgcmVwcmVzZW50aW5nIHllYXJzIGZvciB0aGUgd2lkZSB2aWV3LiAqL1xuICBwcml2YXRlIF9uZXh0TW9udGhZZWFycyE6IG51bWJlcltdW107XG5cbiAgLyoqIEdyaWQgb2YgY2FsZW5kYXIgY2VsbHMgcmVwcmVzZW50aW5nIHRoZSBkYXRlcyBvZiB0aGUgbmV4dCBtb250aC4gKi9cbiAgcHJpdmF0ZSBfbmV4dE1vbnRoV2Vla3MhOiBEYXk8VD5bXVtdO1xuXG4gIC8qKiBBbiBhcnJheSBjb250YWluaW5nIGFsbCB0aGUgbW9udGggbmFtZXMgaW4gdGhlIGN1cnJlbnQgbGFuZ3VhZ2UuICovXG4gIHByaXZhdGUgX21vbnRoTmFtZXM6IHN0cmluZ1tdID0gdGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0TW9udGhOYW1lcygnbG9uZycpO1xuXG4gIC8qKiBBbiBhcnJheSBjb250YWluaW5nIHRoZSB3ZWVrcycgbnVtYmVycyBmb3IgdGhlIGN1cnJlbnQgbW9udGguICovXG4gIHByaXZhdGUgX3dlZWtOdW1iZXJzITogbnVtYmVyW107XG5cbiAgLyoqIEFuIGFycmF5IGNvbnRhaW5pbmcgdGhlIHdlZWtzJyBudW1iZXJzIGZvciB0aGUgbmV4dCBtb250aCBpbiB3aWRlIG1vZGUuICovXG4gIHByaXZhdGUgX25leHRNb250aFdlZWtOdW1iZXJzITogbnVtYmVyW107XG5cbiAgcHJpdmF0ZSBfZW5oYW5jZWRWYXJpYW50OiBib29sZWFuID0gZmFsc2U7XG5cbiAgLyoqIEEgbGlzdCBvZiBjYWxlbmRhcidzIGNlbGxzIGNvcnJlc3BvbmRpbmcgdG8gZGF5cywgbW9udGhzIG9yIHllYXJzIGRlcGVuZGluZyBvbiB0aGUgdmlldy4gKi9cbiAgcHJpdmF0ZSBnZXQgX2NlbGxzKCk6IFNiYkNhbGVuZGFyQ2VsbEJhc2VFbGVtZW50W10ge1xuICAgIHJldHVybiBBcnJheS5mcm9tPFNiYkNhbGVuZGFyQ2VsbEJhc2VFbGVtZW50PihcbiAgICAgICh0aGlzLl9jYWxlbmRhclZpZXcgPT09ICdkYXknXG4gICAgICAgID8gKEFycmF5LmZyb20odGhpcy5zaGFkb3dSb290IS5xdWVyeVNlbGVjdG9yQWxsKCdzbG90JykpLmZsYXRNYXAoKGU6IEhUTUxTbG90RWxlbWVudCkgPT5cbiAgICAgICAgICAgIGUuYXNzaWduZWRFbGVtZW50cyh7IGZsYXR0ZW46IHRydWUgfSksXG4gICAgICAgICAgKSBhcyBTYmJDYWxlbmRhckRheUVsZW1lbnRbXSlcbiAgICAgICAgOiB0aGlzLnNoYWRvd1Jvb3Q/LnF1ZXJ5U2VsZWN0b3JBbGwoYHNiYi1jYWxlbmRhci0ke3RoaXMuX2NhbGVuZGFyVmlld31gKSkgPz8gW10sXG4gICAgKTtcbiAgfVxuXG4gIC8qKiBUaGUgY2hvc2VuIHllYXIgaW4gdGhlIHllYXIgc2VsZWN0aW9uIHZpZXcuICovXG4gIHByaXZhdGUgX2Nob3NlblllYXI/OiBudW1iZXI7XG5cbiAgLyoqIFRoZSBjaG9zZW4gbW9udGggaW4gdGhlIHllYXIgc2VsZWN0aW9uIHZpZXcuICovXG4gIHByaXZhdGUgX2Nob3Nlbk1vbnRoPzogbnVtYmVyO1xuXG4gIC8qKiBXaGV0aGVyIHRoZSBmb2N1cyBzaG91bGQgYmUgcmVzZXQgb24gZm9jdXNDZWxsLiAqL1xuICBwcml2YXRlIF9yZXNldEZvY3VzID0gZmFsc2U7XG5cbiAgLyoqIFdoZXRoZXIgYW4gZWxlbWVudCBpbnNpZGUgdGhlIGNhbGVuZGFyIGlzIGN1cnJlbnRseSBmb2N1c2VkLiAqL1xuICBwcml2YXRlIF9jb250YWluaW5nRm9jdXMgPSBmYWxzZTtcblxuICBAc3RhdGUoKVxuICBwcml2YXRlIGFjY2Vzc29yIF9pbml0aWFsaXplZCA9IGZhbHNlO1xuXG4gIHByaXZhdGUgX2xhbmd1YWdlID0gbmV3IFNiYkxhbmd1YWdlQ29udHJvbGxlcih0aGlzKS53aXRoSGFuZGxlcigoKSA9PiB7XG4gICAgdGhpcy5fbW9udGhOYW1lcyA9IHRoaXMuX2RhdGVBZGFwdGVyLmdldE1vbnRoTmFtZXMoJ2xvbmcnKTtcbiAgICB0aGlzLl9jcmVhdGVNb250aFJvd3MoKTtcbiAgfSk7XG4gIHByaXZhdGUgX21lZGlhTWF0Y2hlciA9IG5ldyBTYmJNZWRpYU1hdGNoZXJDb250cm9sbGVyKHRoaXMsIHtcbiAgICBbU2JiTWVkaWFRdWVyeUJyZWFrcG9pbnRMYXJnZUFuZEFib3ZlXTogKCkgPT4gdGhpcy5faW5pdCgpLFxuICB9KTtcblxuICBwdWJsaWMgY29uc3RydWN0b3IoKSB7XG4gICAgc3VwZXIoKTtcbiAgICB0aGlzLl9jcmVhdGVNb250aFJvd3MoKTtcbiAgICB0aGlzLl9zZXRXZWVrZGF5cygpO1xuXG4gICAgLy8gV2UgbmVlZCB0byB0cmFjayB0aGUgZm9jdXMgYXMgd2Ugc2hvdWxkIG9ubHkgdGFrZSBmb2N1cyBpbnRvIHRoZSBjYWxlbmRhciwgd2hlbiB0aGVcbiAgICAvLyBmb2N1cyB3YXMgb25jZSBzZXQgaW50byB0aGUgY2FsZW5kYXIuXG4gICAgLy8gRm9yIHNoYWRvdyBET00gY29tcGF0aWJpbGl0eSB3ZSBuZWVkIHRvIHRyYWNrIHRoaXMgcHJvZ3JhbW1hdGljYWxseS5cbiAgICB0aGlzLmFkZEV2ZW50TGlzdGVuZXIoJ2ZvY3VzaW4nLCAoKSA9PiAodGhpcy5fY29udGFpbmluZ0ZvY3VzID0gdHJ1ZSkpO1xuICAgIHRoaXMuYWRkRXZlbnRMaXN0ZW5lcignZm9jdXNvdXQnLCAoKSA9PiAodGhpcy5fY29udGFpbmluZ0ZvY3VzID0gZmFsc2UpKTtcbiAgICB0aGlzLmFkZEV2ZW50TGlzdGVuZXIoJ2NsaWNrJywgKGUpID0+IHtcbiAgICAgIGNvbnN0IGRheSA9IChlLnRhcmdldCBhcyBIVE1MRWxlbWVudCkuY2xvc2VzdDxTYmJDYWxlbmRhckRheUVsZW1lbnQ8VD4+KCdzYmItY2FsZW5kYXItZGF5Jyk7XG4gICAgICBpZiAoZGF5KSB7XG4gICAgICAgIHRoaXMuX3NlbGVjdERhdGUoZGF5LnZhbHVlISk7XG4gICAgICB9XG4gICAgfSk7XG4gICAgdGhpcy5hZGRFdmVudExpc3RlbmVyKCdrZXlkb3duJywgKGUpID0+IHtcbiAgICAgIGlmICgoZS50YXJnZXQgYXMgSFRNTEVsZW1lbnQpLmxvY2FsTmFtZSA9PT0gJ3NiYi1jYWxlbmRhci1kYXknKSB7XG4gICAgICAgIHRoaXMuX2hhbmRsZUtleWJvYXJkRXZlbnQoXG4gICAgICAgICAgZSxcbiAgICAgICAgICB0aGlzLl9tYXBEYXRlVG9EYXkoKGUudGFyZ2V0IGFzIFNiYkNhbGVuZGFyRGF5RWxlbWVudCkudmFsdWUhIGFzIFQpLFxuICAgICAgICApO1xuICAgICAgfVxuICAgIH0pO1xuICB9XG5cbiAgcHJpdmF0ZSBfZGF0ZUZpbHRlcihkYXRlOiBUKTogYm9vbGVhbiB7XG4gICAgcmV0dXJuIHRoaXMuZGF0ZUZpbHRlcj8uKGRhdGUpID8/IHRydWU7XG4gIH1cblxuICAvKiogUmVzZXRzIHRoZSBhY3RpdmUgbW9udGggYWNjb3JkaW5nIHRvIHRoZSBuZXcgc3RhdGUgb2YgdGhlIGNhbGVuZGFyLiAqL1xuICBwdWJsaWMgcmVzZXRQb3NpdGlvbigpOiB2b2lkIHtcbiAgICB0aGlzLl9yZXNldENhbGVuZGFyVmlldygpO1xuICAgIHRoaXMuX2luaXQoKTtcbiAgfVxuXG4gIHB1YmxpYyBvdmVycmlkZSBjb25uZWN0ZWRDYWxsYmFjaygpOiB2b2lkIHtcbiAgICBzdXBlci5jb25uZWN0ZWRDYWxsYmFjaygpO1xuICAgIHRoaXMucmVzZXRQb3NpdGlvbigpO1xuICAgIHRoaXMuc2hhZG93Um9vdD8uYWRkRXZlbnRMaXN0ZW5lcignc2xvdGNoYW5nZScsIHRoaXMuX29uU2xvdENoYW5nZSwgeyBjYXB0dXJlOiB0cnVlIH0pO1xuICB9XG5cbiAgcHVibGljIG92ZXJyaWRlIGRpc2Nvbm5lY3RlZENhbGxiYWNrKCk6IHZvaWQge1xuICAgIHN1cGVyLmRpc2Nvbm5lY3RlZENhbGxiYWNrKCk7XG4gICAgdGhpcy5zaGFkb3dSb290Py5yZW1vdmVFdmVudExpc3RlbmVyKCdzbG90Y2hhbmdlJywgdGhpcy5fb25TbG90Q2hhbmdlLCB7IGNhcHR1cmU6IHRydWUgfSk7XG4gIH1cblxuICAvKiogQGludGVybmFsICovXG4gIHB1YmxpYyBvdmVycmlkZSBmb2N1cygpOiB2b2lkIHtcbiAgICB0aGlzLl9yZXNldEZvY3VzID0gdHJ1ZTtcbiAgICB0aGlzLl9mb2N1c0NlbGwoKTtcbiAgfVxuXG4gIHByb3RlY3RlZCBvdmVycmlkZSB3aWxsVXBkYXRlKGNoYW5nZWRQcm9wZXJ0aWVzOiBQcm9wZXJ0eVZhbHVlczx0aGlzPik6IHZvaWQge1xuICAgIHN1cGVyLndpbGxVcGRhdGUoY2hhbmdlZFByb3BlcnRpZXMpO1xuXG4gICAgaWYgKCF0aGlzLl9pbml0aWFsaXplZCkge1xuICAgICAgcmV0dXJuO1xuICAgIH1cblxuICAgIGlmIChjaGFuZ2VkUHJvcGVydGllcy5oYXMoJ3dpZGUnKSB8fCBjaGFuZ2VkUHJvcGVydGllcy5oYXMoJ29yaWVudGF0aW9uJykpIHtcbiAgICAgIHRoaXMucmVzZXRQb3NpdGlvbigpO1xuICAgIH1cblxuICAgIGlmIChjaGFuZ2VkUHJvcGVydGllcy5oYXMoJ3ZpZXcnKSkge1xuICAgICAgdGhpcy5fc2V0Q2hvc2VuWWVhcigpO1xuICAgICAgdGhpcy5fY2hvc2VuTW9udGggPSB1bmRlZmluZWQ7XG4gICAgICB0aGlzLl9uZXh0Q2FsZW5kYXJWaWV3ID0gdGhpcy5fY2FsZW5kYXJWaWV3ID0gdGhpcy52aWV3O1xuICAgIH1cbiAgfVxuXG4gIHByb3RlY3RlZCBvdmVycmlkZSB1cGRhdGVkKGNoYW5nZWRQcm9wZXJ0aWVzOiBQcm9wZXJ0eVZhbHVlczx0aGlzPik6IHZvaWQge1xuICAgIHN1cGVyLnVwZGF0ZWQoY2hhbmdlZFByb3BlcnRpZXMpO1xuICAgIC8vIFRoZSBjYWxlbmRhciBuZWVkcyB0byBjYWxjdWxhdGUgdGFiLWluZGV4ZXMgb24gZmlyc3QgcmVuZGVyLFxuICAgIC8vIGFuZCBldmVyeSB0aW1lIGEgZGF0ZSBpcyBzZWxlY3RlZCBvciB0aGUgbW9udGggdmlldyBjaGFuZ2VzLlxuICAgIHRoaXMuX3NldFRhYkluZGV4KCk7XG5cbiAgICAvLyBXaGVuIGNoYW5naW5nIHZpZXcgdG8geWVhci9tb250aCwgdGhlIHRhYmluZGV4IGlzIGNoYW5nZWQsIGJ1dCB0aGUgZm9jdXNlZCBlbGVtZW50IGlzIGdldHRpbmcgbG9zdC5cbiAgICAvLyBXZSBuZWVkIHRvIGNhbGwgYF9mb2N1c0NlbGwoKWAgbWV0aG9kIGV4cGxpY2l0bHkgdG8gY29ycmVjdGx5IHNldCB0aGUgZm9jdXMuXG4gICAgdGhpcy5fZm9jdXNDZWxsKCk7XG4gIH1cblxuICBwcml2YXRlIF9vblNsb3RDaGFuZ2UgPSAoKTogdm9pZCA9PiB7XG4gICAgdGhpcy5fZW5oYW5jZWRWYXJpYW50ID0gQXJyYXkuZnJvbSh0aGlzLmNoaWxkcmVuKS5zb21lKFxuICAgICAgKGMpID0+IGMubG9jYWxOYW1lID09PSAnc2JiLWNhbGVuZGFyLWRheScsXG4gICAgKTtcbiAgICB0aGlzLnRvZ2dsZVN0YXRlKCdlbmhhbmNlZCcsIHRoaXMuX2VuaGFuY2VkVmFyaWFudCk7XG4gICAgdGhpcy5fc2V0VGFiSW5kZXgoKTtcbiAgfTtcblxuICAvKipcbiAgICogVGhlIGBfc2VsZWN0ZWRgIHN0YXRlIHNob3VsZCBiZSBhZGFwdGVkIHdoZW4gdGhlIGBtdWx0aXBsZWAgcHJvcGVydHkgY2hhbmdlczpcbiAgICogICAtIGlmIGl0IGNoYW5nZXMgdG8gdHJ1ZSwgdGhlICdfc2VsZWN0ZWQnIGlzIHNldCB0byBhbiBhcnJheTtcbiAgICogICAtIGlmIGl0IGNoYW5nZXMgdG8gZmFsc2UsIHRoZSBmaXJzdCBhdmFpbGFibGUgb3B0aW9uIGlzIHNldCBhcyAndmFsdWUnIG90aGVyd2lzZSBpdCdzIHNldCB0byBudWxsLlxuICAgKi9cbiAgcHJpdmF0ZSBfb25NdWx0aXBsZUNoYW5nZWQoaXNNdWx0aXBsZTogYm9vbGVhbik6IHZvaWQge1xuICAgIGlmIChpc011bHRpcGxlICYmICFBcnJheS5pc0FycmF5KHRoaXMuX3NlbGVjdGVkKSkge1xuICAgICAgdGhpcy5fc2VsZWN0ZWQgPSB0aGlzLl9zZWxlY3RlZCA/IFt0aGlzLl9zZWxlY3RlZCBhcyBUXSA6IFtdO1xuICAgIH1cbiAgICBpZiAoIWlzTXVsdGlwbGUgJiYgQXJyYXkuaXNBcnJheSh0aGlzLl9zZWxlY3RlZCkpIHtcbiAgICAgIHRoaXMuX3NlbGVjdGVkID0gKHRoaXMuX3NlbGVjdGVkIGFzIFRbXSkubGVuZ3RoID8gKHRoaXMuX3NlbGVjdGVkIGFzIFRbXSlbMF0gOiBudWxsO1xuICAgIH1cbiAgfVxuXG4gIC8qKiBJbml0aWFsaXplcyB0aGUgY29tcG9uZW50LiAqL1xuICBwcml2YXRlIF9pbml0KGFjdGl2ZURhdGU/OiBUKTogdm9pZCB7XG4gICAgLy8gRHVlIHRvIGl0cyBjb21wbGV4aXR5LCB0aGUgY2FsZW5kYXIgaXMgb25seSBpbml0aWFsaXplZCBvbiBjbGllbnQgc2lkZVxuICAgIGlmIChpc1NlcnZlcikge1xuICAgICAgcmV0dXJuO1xuICAgIH0gZWxzZSBpZiAodGhpcy5oeWRyYXRpb25SZXF1aXJlZCkge1xuICAgICAgdGhpcy5oeWRyYXRpb25Db21wbGV0ZS50aGVuKCgpID0+IHRoaXMuX2luaXQoKSk7XG4gICAgICByZXR1cm47XG4gICAgfVxuXG4gICAgaWYgKGFjdGl2ZURhdGUpIHtcbiAgICAgIHRoaXMuX2Fzc2lnbkFjdGl2ZURhdGUoYWN0aXZlRGF0ZSk7XG4gICAgfVxuICAgIHRoaXMuX3dpZGUgPVxuICAgICAgKHRoaXMuX21lZGlhTWF0Y2hlci5tYXRjaGVzKFNiYk1lZGlhUXVlcnlCcmVha3BvaW50TGFyZ2VBbmRBYm92ZSkgPz8gZmFsc2UpICYmIHRoaXMud2lkZTtcbiAgICB0aGlzLl93ZWVrcyA9IHRoaXMuX2NyZWF0ZVdlZWtSb3dzKHRoaXMuX2FjdGl2ZURhdGUpO1xuICAgIHRoaXMuX3llYXJzID0gdGhpcy5fY3JlYXRlWWVhclJvd3MoKTtcbiAgICB0aGlzLl93ZWVrTnVtYmVycyA9IHRoaXMuX3dlZWtzXG4gICAgICAuZmxhdCgpXG4gICAgICAuc29ydCgoYSwgYikgPT4gYS52YWx1ZS5sb2NhbGVDb21wYXJlKGIudmFsdWUpKVxuICAgICAgLm1hcCgoZGF5OiBEYXk8VD4pID0+IGRheS53ZWVrVmFsdWUpXG4gICAgICAuZmlsdGVyKCh2LCBpLCBhKSA9PiBhLmluZGV4T2YodikgPT09IGkpO1xuICAgIHRoaXMuX25leHRNb250aFdlZWtzID0gW1tdXTtcbiAgICB0aGlzLl9uZXh0TW9udGhZZWFycyA9IFtbXV07XG4gICAgaWYgKHRoaXMuX3dpZGUpIHtcbiAgICAgIGNvbnN0IG5leHRNb250aERhdGUgPSB0aGlzLl9kYXRlQWRhcHRlci5hZGRDYWxlbmRhck1vbnRocyh0aGlzLl9hY3RpdmVEYXRlLCAxKTtcbiAgICAgIHRoaXMuX25leHRNb250aFdlZWtzID0gdGhpcy5fY3JlYXRlV2Vla1Jvd3MobmV4dE1vbnRoRGF0ZSwgdHJ1ZSk7XG4gICAgICB0aGlzLl9uZXh0TW9udGhZZWFycyA9IHRoaXMuX2NyZWF0ZVllYXJSb3dzKFlFQVJTX1BFUl9QQUdFKTtcbiAgICAgIHRoaXMuX25leHRNb250aFdlZWtOdW1iZXJzID0gdGhpcy5fbmV4dE1vbnRoV2Vla3NcbiAgICAgICAgLmZsYXQoKVxuICAgICAgICAuc29ydCgoYSwgYikgPT4gYS52YWx1ZS5sb2NhbGVDb21wYXJlKGIudmFsdWUpKVxuICAgICAgICAubWFwKChkYXk6IERheTxUPikgPT4gZGF5LndlZWtWYWx1ZSlcbiAgICAgICAgLmZpbHRlcigodiwgaSwgYSkgPT4gYS5pbmRleE9mKHYpID09PSBpKTtcbiAgICB9XG4gICAgdGhpcy5faW5pdGlhbGl6ZWQgPSB0cnVlO1xuICB9XG5cbiAgLyoqIEZvY3VzZXMgb24gYSBkYXkgY2VsbCBwcmlvcml0aXppbmcgdGhlIHNlbGVjdGVkIGRheSwgdGhlIGN1cnJlbnQgZGF5LCBhbmQgbGFzdGx5LCB0aGUgZmlyc3Qgc2VsZWN0YWJsZSBkYXkuICovXG4gIHByaXZhdGUgX2ZvY3VzQ2VsbCgpOiB2b2lkIHtcbiAgICBpZiAodGhpcy5fcmVzZXRGb2N1cykge1xuICAgICAgdGhpcy5fZ2V0Rmlyc3RGb2N1c2FibGUoKT8uZm9jdXMoKTtcbiAgICAgIHRoaXMuX3Jlc2V0Rm9jdXMgPSBmYWxzZTtcbiAgICB9XG4gIH1cblxuICAvKiogQ3JlYXRlcyB0aGUgYXJyYXkgb2Ygd2Vla2RheXMuICovXG4gIHByaXZhdGUgX3NldFdlZWtkYXlzKCk6IHZvaWQge1xuICAgIGNvbnN0IG5hcnJvd1dlZWtkYXlzOiBzdHJpbmdbXSA9IHRoaXMuX2RhdGVBZGFwdGVyLmdldERheU9mV2Vla05hbWVzKCduYXJyb3cnKTtcbiAgICBjb25zdCBsb25nV2Vla2RheXM6IHN0cmluZ1tdID0gdGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0RGF5T2ZXZWVrTmFtZXMoJ2xvbmcnKTtcbiAgICBjb25zdCB3ZWVrZGF5czogV2Vla2RheVtdID0gbG9uZ1dlZWtkYXlzLm1hcCgobG9uZzogc3RyaW5nLCBpOiBudW1iZXIpID0+ICh7XG4gICAgICBsb25nLFxuICAgICAgbmFycm93OiBuYXJyb3dXZWVrZGF5c1tpXSxcbiAgICB9KSk7XG5cbiAgICAvLyBSb3RhdGVzIHRoZSBsYWJlbHMgZm9yIGRheXMgb2YgdGhlIHdlZWsgYmFzZWQgb24gdGhlIGNvbmZpZ3VyZWQgZmlyc3QgZGF5IG9mIHRoZSB3ZWVrLlxuICAgIGNvbnN0IGZpcnN0RGF5T2ZXZWVrOiBudW1iZXIgPSB0aGlzLl9kYXRlQWRhcHRlci5nZXRGaXJzdERheU9mV2VlaygpO1xuICAgIHRoaXMuX3dlZWtkYXlzID0gd2Vla2RheXMuc2xpY2UoZmlyc3REYXlPZldlZWspLmNvbmNhdCh3ZWVrZGF5cy5zbGljZSgwLCBmaXJzdERheU9mV2VlaykpO1xuICB9XG5cbiAgLyoqIENyZWF0ZXMgdGhlIHJvd3MgYWxvbmcgdGhlIGhvcml6b250YWwgZGlyZWN0aW9uIGFuZCBzZXRzIHRoZSBwYXJhbWV0ZXJzIHVzZWQgaW4ga2V5Ym9hcmQgbmF2aWdhdGlvbi4gKi9cbiAgcHJpdmF0ZSBfY3JlYXRlV2Vla1Jvd3ModmFsdWU6IFQsIGlzU2Vjb25kTW9udGhJblZpZXcgPSBmYWxzZSk6IERheTxUPltdW10ge1xuICAgIGNvbnN0IGRheXNJbk1vbnRoOiBudW1iZXIgPSB0aGlzLl9kYXRlQWRhcHRlci5nZXROdW1EYXlzSW5Nb250aCh2YWx1ZSk7XG4gICAgY29uc3Qgd2Vla09mZnNldDogbnVtYmVyID0gdGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0Rmlyc3RXZWVrT2Zmc2V0KHZhbHVlKTtcbiAgICBpZiAoIWlzU2Vjb25kTW9udGhJblZpZXcpIHtcbiAgICAgIHRoaXMuX2tleWJvYXJkTmF2aWdhdGlvbkRheVZpZXdQYXJhbWV0ZXJzLmZpcnN0TW9udGhMZW5ndGggPSBkYXlzSW5Nb250aDtcbiAgICAgIHRoaXMuX2tleWJvYXJkTmF2aWdhdGlvbkRheVZpZXdQYXJhbWV0ZXJzLmZpcnN0TW9udGhPZmZzZXQgPSB3ZWVrT2Zmc2V0O1xuICAgICAgdGhpcy5fa2V5Ym9hcmROYXZpZ2F0aW9uRGF5Vmlld1BhcmFtZXRlcnMuZmlyc3REYXlJblZpZXcgPSB0aGlzLl9kYXRlQWRhcHRlci50b0lzbzg2MDEoXG4gICAgICAgIHRoaXMuX2RhdGVBZGFwdGVyLmNyZWF0ZURhdGUoXG4gICAgICAgICAgdGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0WWVhcih2YWx1ZSksXG4gICAgICAgICAgdGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0TW9udGgodmFsdWUpLFxuICAgICAgICAgIDEsXG4gICAgICAgICksXG4gICAgICApO1xuICAgICAgdGhpcy5fa2V5Ym9hcmROYXZpZ2F0aW9uRGF5Vmlld1BhcmFtZXRlcnMubGFzdERheUluVmlldyA9IHRoaXMuX2RhdGVBZGFwdGVyLnRvSXNvODYwMShcbiAgICAgICAgdGhpcy5fZGF0ZUFkYXB0ZXIuY3JlYXRlRGF0ZShcbiAgICAgICAgICB0aGlzLl9kYXRlQWRhcHRlci5nZXRZZWFyKHZhbHVlKSxcbiAgICAgICAgICB0aGlzLl9kYXRlQWRhcHRlci5nZXRNb250aCh2YWx1ZSksXG4gICAgICAgICAgZGF5c0luTW9udGgsXG4gICAgICAgICksXG4gICAgICApO1xuICAgIH0gZWxzZSB7XG4gICAgICB0aGlzLl9rZXlib2FyZE5hdmlnYXRpb25EYXlWaWV3UGFyYW1ldGVycy5zZWNvbmRNb250aE9mZnNldCA9IHdlZWtPZmZzZXQ7XG4gICAgICB0aGlzLl9rZXlib2FyZE5hdmlnYXRpb25EYXlWaWV3UGFyYW1ldGVycy5sYXN0RGF5SW5WaWV3ID0gdGhpcy5fZGF0ZUFkYXB0ZXIudG9Jc284NjAxKFxuICAgICAgICB0aGlzLl9kYXRlQWRhcHRlci5jcmVhdGVEYXRlKFxuICAgICAgICAgIHRoaXMuX2RhdGVBZGFwdGVyLmdldFllYXIodmFsdWUpLFxuICAgICAgICAgIHRoaXMuX2RhdGVBZGFwdGVyLmdldE1vbnRoKHZhbHVlKSxcbiAgICAgICAgICBkYXlzSW5Nb250aCxcbiAgICAgICAgKSxcbiAgICAgICk7XG4gICAgfVxuICAgIHJldHVybiB0aGlzLm9yaWVudGF0aW9uID09PSAnaG9yaXpvbnRhbCdcbiAgICAgID8gdGhpcy5fY3JlYXRlV2Vla1Jvd3NIb3Jpem9udGFsKHZhbHVlLCBkYXlzSW5Nb250aCwgd2Vla09mZnNldClcbiAgICAgIDogdGhpcy5fY3JlYXRlV2Vla1Jvd3NWZXJ0aWNhbCh2YWx1ZSwgZGF5c0luTW9udGgsIHdlZWtPZmZzZXQpO1xuICB9XG5cbiAgLyoqXG4gICAqIENyZWF0ZXMgdGhlIHJvd3MgZm9yIGVhY2ggd2VlayBpbiBvcmllbnRhdGlvbj0naG9yaXpvbnRhbCcuXG4gICAqXG4gICAqIEl0ZXJhdGVzIHRocm91Z2ggdGhlIGRheXMgb2YgdGhlIG1vbnRocywgY3JlYXRlcyBhIERheSBvYmplY3QgZm9yIGVhY2ggYW5kIHB1c2hlcyBpdCBpbnRvIGFuZCBhcnJheS5cbiAgICogRWFjaCBzZXZlbiBkYXlzIChjb25zaWRlcmluZyB0aGUgb2Zmc2V0IGF0IHRoZSBiZWdpbm5pbmcgb2YgdGhlIG1vbnRoKSByZXN0YXJ0cyBmcm9tIGFuIGVtcHR5IGFycmF5LlxuICAgKlxuICAgKiBUaGUgcmVzdWx0IGlzIGEgbWF0cml4IGluIHdoaWNoIGV2ZXJ5IHJvdyBpcyBhIHdlZWsgKG9yIHBhcnQgb2YgaXQsIGNvbnNpZGVyaW5nIG9mZnNldCkuXG4gICAqL1xuICBwcml2YXRlIF9jcmVhdGVXZWVrUm93c0hvcml6b250YWwodmFsdWU6IFQsIGRheXNJbk1vbnRoOiBudW1iZXIsIHdlZWtPZmZzZXQ6IG51bWJlcik6IERheTxUPltdW10ge1xuICAgIGNvbnN0IHdlZWtzOiBEYXk8VD5bXVtdID0gW1tdXTtcbiAgICBmb3IgKGxldCBpID0gMCwgY2VsbCA9IHdlZWtPZmZzZXQ7IGkgPCBkYXlzSW5Nb250aDsgaSsrLCBjZWxsKyspIHtcbiAgICAgIGlmIChjZWxsID09PSBEQVlTX1BFUl9ST1cpIHtcbiAgICAgICAgd2Vla3MucHVzaChbXSk7XG4gICAgICAgIGNlbGwgPSAwO1xuICAgICAgfVxuICAgICAgY29uc3QgZGF0ZSA9IHRoaXMuX2RhdGVBZGFwdGVyLmNyZWF0ZURhdGUoXG4gICAgICAgIHRoaXMuX2RhdGVBZGFwdGVyLmdldFllYXIodmFsdWUpLFxuICAgICAgICB0aGlzLl9kYXRlQWRhcHRlci5nZXRNb250aCh2YWx1ZSksXG4gICAgICAgIGkgKyAxLFxuICAgICAgKSE7XG4gICAgICB3ZWVrc1t3ZWVrcy5sZW5ndGggLSAxXS5wdXNoKHRoaXMuX21hcERhdGVUb0RheShkYXRlKSk7XG4gICAgfVxuICAgIHJldHVybiB3ZWVrcztcbiAgfVxuXG4gIC8qKlxuICAgKiBDcmVhdGVzIHRoZSByb3dzIGZvciBlYWNoIHdlZWsgaW4gb3JpZW50YXRpb249J3ZlcnRpY2FsJy5cbiAgICpcbiAgICogQ3JlYXRlcyBhIG1hdHJpeCB3aXRoIHNldmVuIGVtcHR5IHJvd3MuXG4gICAqIEl0ZXJhdGVzIHRocm91Z2ggdGhlIGRheXMgb2YgdGhlIG1vbnRocywgY3JlYXRlcyBhIERheSBvYmplY3QgZm9yIGVhY2hcbiAgICogYW5kIHB1c2hlcyBpdCBpbnRvIHRoZSBjb3JyZWN0IGFycmF5IGNvbnNpZGVyaW5nIHRoZSBvZmZzZXQgYXQgdGhlIGJlZ2lubmluZyBvZiB0aGUgbW9udGguXG4gICAqIEVhY2ggc2V2ZW4gZGF5cyAoaW5jbHVkaW5nIG9mZnNldCkgcmVzdGFydHMgZnJvbSB0aGUgZmlyc3QuXG4gICAqXG4gICAqIFRoZSByZXN1bHQgaXMgYSBtYXRyaXggaW4gd2hpY2ggZXZlcnkgcm93IGlzIGEgc2V0IG9mIHdlZWtkYXlzLCBzbzpcbiAgICogLSByb3cgMDogYWxsIHRoZSBNb25kYXlzO1xuICAgKiAtIHJvdyAxOiBhbGwgdGhlIFR1ZXNkYXlzO1xuICAgKiAtIC4uLlxuICAgKiAtIHJvdyA3OiBhbGwgdGhlIFN1bmRheXMuXG4gICAqL1xuICBwcml2YXRlIF9jcmVhdGVXZWVrUm93c1ZlcnRpY2FsKHZhbHVlOiBULCBkYXlzSW5Nb250aDogbnVtYmVyLCB3ZWVrT2Zmc2V0OiBudW1iZXIpOiBEYXk8VD5bXVtdIHtcbiAgICBjb25zdCB3ZWVrczogRGF5PFQ+W11bXSA9IEFycmF5LmZyb20oeyBsZW5ndGg6IERBWVNfUEVSX1JPVyB9LCAoKSA9PiBbXSk7XG4gICAgZm9yIChsZXQgaSA9IDAsIGNlbGwgPSB3ZWVrT2Zmc2V0OyBpIDwgZGF5c0luTW9udGg7IGkrKywgY2VsbCsrKSB7XG4gICAgICBpZiAoY2VsbCA9PT0gREFZU19QRVJfUk9XKSB7XG4gICAgICAgIGNlbGwgPSAwO1xuICAgICAgfVxuICAgICAgY29uc3QgZGF0ZSA9IHRoaXMuX2RhdGVBZGFwdGVyLmNyZWF0ZURhdGUoXG4gICAgICAgIHRoaXMuX2RhdGVBZGFwdGVyLmdldFllYXIodmFsdWUpLFxuICAgICAgICB0aGlzLl9kYXRlQWRhcHRlci5nZXRNb250aCh2YWx1ZSksXG4gICAgICAgIGkgKyAxLFxuICAgICAgKSE7XG4gICAgICB3ZWVrc1tjZWxsXS5wdXNoKHRoaXMuX21hcERhdGVUb0RheShkYXRlKSk7XG4gICAgfVxuICAgIHJldHVybiB3ZWVrcztcbiAgfVxuXG4gIHByaXZhdGUgX21hcERhdGVUb0RheShkYXRlOiBUKTogRGF5PFQ+IHtcbiAgICBjb25zdCBpc29EYXRlID0gdGhpcy5fZGF0ZUFkYXB0ZXIudG9Jc284NjAxKGRhdGUpO1xuICAgIHJldHVybiB7XG4gICAgICB2YWx1ZTogaXNvRGF0ZSxcbiAgICAgIGRhdGVWYWx1ZTogZGF0ZSxcbiAgICAgIGRheVZhbHVlOiBTdHJpbmcodGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0RGF0ZShkYXRlKSksXG4gICAgICBtb250aFZhbHVlOiBTdHJpbmcodGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0TW9udGgoZGF0ZSkpLFxuICAgICAgeWVhclZhbHVlOiBTdHJpbmcodGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0WWVhcihkYXRlKSksXG4gICAgICAvLyBUT0RPOiBJbXByb3ZlIHBlcmZvcm1hbmNlIG9mIHRoaXMsIGJ5IGtlZXBpbmcgdHJhY2sgb2YgdGhlXG4gICAgICAvLyB3ZWVrIG51bWJlciB3aGlsZSBpdGVyYXRpbmcgdGhyb3VnaCB0aGUgZGF5cy5cbiAgICAgIHdlZWtWYWx1ZTogdGhpcy5fZ2V0V2VlayhkYXRlKSxcbiAgICAgIHdlZWtEYXlWYWx1ZTogdGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0RGF5T2ZXZWVrKGRhdGUpLFxuICAgIH07XG4gIH1cblxuICBwcml2YXRlIF9nZXRXZWVrKGRhdGU6IFQpOiBudW1iZXIge1xuICAgIGNvbnN0IGZpcnN0RGF5T2ZZZWFyID0gdGhpcy5fZGF0ZUFkYXB0ZXIuY3JlYXRlRGF0ZSh0aGlzLl9kYXRlQWRhcHRlci5nZXRZZWFyKGRhdGUpLCAxLCAxKTtcbiAgICBjb25zdCB3ZWVrZGF5ID0gdGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0RGF5T2ZXZWVrKGZpcnN0RGF5T2ZZZWFyKTtcblxuICAgIGxldCB3ZWVrSW5kZXggPSBbTU9OREFZLCBUVUVTREFZLCBXRURORVNEQVksIFRIVVJTREFZXS5pbmNsdWRlcyh3ZWVrZGF5KSA/IDEgOiAwO1xuICAgIGxldCB3ZWVrU3RhcnQgPSB0aGlzLl9kYXRlQWRhcHRlci5hZGRDYWxlbmRhckRheXMoXG4gICAgICBmaXJzdERheU9mWWVhcixcbiAgICAgIHRoaXMuX2RhdGVBZGFwdGVyLmdldEZpcnN0V2Vla09mZnNldChmaXJzdERheU9mWWVhcikgKiAtMSxcbiAgICApO1xuICAgIHdoaWxlICh0aGlzLl9kYXRlQWRhcHRlci5jb21wYXJlRGF0ZSh3ZWVrU3RhcnQsIGRhdGUpIDw9IDApIHtcbiAgICAgIGNvbnN0IHdlZWtFbmQgPSB0aGlzLl9kYXRlQWRhcHRlci5hZGRDYWxlbmRhckRheXMod2Vla1N0YXJ0LCA2KTtcbiAgICAgIGlmICh0aGlzLl9kYXRlQWRhcHRlci5jb21wYXJlRGF0ZShkYXRlLCB3ZWVrRW5kKSA8PSAwKSB7XG4gICAgICAgIHJldHVybiB3ZWVrSW5kZXggPiAwXG4gICAgICAgICAgPyB3ZWVrSW5kZXhcbiAgICAgICAgICA6IHRoaXMuX2dldFdlZWsodGhpcy5fZGF0ZUFkYXB0ZXIuYWRkQ2FsZW5kYXJEYXlzKGZpcnN0RGF5T2ZZZWFyLCAtMSkpO1xuICAgICAgfVxuICAgICAgd2Vla1N0YXJ0ID0gdGhpcy5fZGF0ZUFkYXB0ZXIuYWRkQ2FsZW5kYXJEYXlzKHdlZWtTdGFydCwgREFZU19QRVJfUk9XKTtcbiAgICAgIHdlZWtJbmRleCsrO1xuICAgIH1cblxuICAgIHRocm93IG5ldyBFcnJvcignVGhlIHByb3ZpZGVkIGRhdGUgaXMgaW52YWxpZCcpO1xuICB9XG5cbiAgLyoqIEZvcmNlIHRoZSBjb252ZXJzaW9uIHRvIElTTzg2MDEgZm9ybWF0dGVkIHN0cmluZy4gKi9cbiAgcHJpdmF0ZSBfbWFwVmFsdWVUb0lTT0RhdGUodmFsdWU6IHN0cmluZyB8IERhdGUpOiBzdHJpbmcge1xuICAgIHJldHVybiB0eXBlb2YgdmFsdWUgPT09ICdzdHJpbmcnID8gdmFsdWUgOiB0aGlzLl9kYXRlQWRhcHRlci50b0lzbzg2MDEodmFsdWUgYXMgVCk7XG4gIH1cblxuICAvKiogQ3JlYXRlcyB0aGUgcm93cyBmb3IgdGhlIG1vbnRoIHNlbGVjdGlvbiB2aWV3LiAqL1xuICBwcml2YXRlIF9jcmVhdGVNb250aFJvd3MoKTogdm9pZCB7XG4gICAgY29uc3QgbW9udGhzOiBNb250aENlbGxbXSA9IG5ldyBBcnJheSgxMikuZmlsbChudWxsKS5tYXAoXG4gICAgICAoXywgaTogbnVtYmVyKTogTW9udGhDZWxsID0+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+YDtcbiAgfVxuXG4gIC8qKiBTZWxlY3QgdGhlIHllYXIgYW5kIGNoYW5nZSB0aGUgdmlldyB0byBtb250aCBzZWxlY3Rpb24uICovXG4gIHByaXZhdGUgX29uWWVhclNlbGVjdGlvbih5ZWFyOiBudW1iZXIsIHJpZ2h0U2lkZTogYm9vbGVhbik6IHZvaWQge1xuICAgIHRoaXMuX2Nob3NlblllYXIgPSByaWdodFNpZGUgPyB5ZWFyIC0gMSA6IHllYXI7XG4gICAgdGhpcy5fbmV4dENhbGVuZGFyVmlldyA9ICdtb250aCc7XG4gICAgdGhpcy5fYXNzaWduQWN0aXZlRGF0ZShcbiAgICAgIHRoaXMuX2RhdGVBZGFwdGVyLmNyZWF0ZURhdGUoXG4gICAgICAgIHRoaXMuX2Nob3NlblllYXIsXG4gICAgICAgIHRoaXMuX2RhdGVBZGFwdGVyLmdldE1vbnRoKHRoaXMuX2FjdGl2ZURhdGUpLFxuICAgICAgICB0aGlzLl9kYXRlQWRhcHRlci5nZXREYXRlKHRoaXMuX2FjdGl2ZURhdGUpLFxuICAgICAgKSxcbiAgICApO1xuICAgIHRoaXMuX3N0YXJ0VGFibGVUcmFuc2l0aW9uKCk7XG4gIH1cblxuICBwcml2YXRlIF9nZXRWaWV3KCk6IFRlbXBsYXRlUmVzdWx0IHtcbiAgICBpZiAoaXNTZXJ2ZXIgfHwgdGhpcy5oeWRyYXRpb25SZXF1aXJlZCkge1xuICAgICAgLy8gVE9ETzogV2UgZGlzYWJsZSBTU1IgZm9yIGNhbGVuZGFyIGZvciBub3cuIEZpZ3VyZSBvdXQsIGlmIHRoZXJlIGlzIGEgd2F5XG4gICAgICAvLyB0byBlbmFibGUgaXQsIHdoaWxlIGNvbnNpZGVyaW5nIGkxOG4gYW5kIGRhdGUgaW5mb3JtYXRpb24uXG4gICAgICByZXR1cm4gaHRtbGAke25vdGhpbmd9YDtcbiAgICB9XG4gICAgc3dpdGNoICh0aGlzLl9jYWxlbmRhclZpZXcpIHtcbiAgICAgIGNhc2UgJ3llYXInOlxuICAgICAgICByZXR1cm4gdGhpcy5fcmVuZGVyWWVhclZpZXcoKTtcbiAgICAgIGNhc2UgJ21vbnRoJzpcbiAgICAgICAgcmV0dXJuIHRoaXMuX3JlbmRlck1vbnRoVmlldygpO1xuICAgICAgY2FzZSAnZGF5JzpcbiAgICAgIGRlZmF1bHQ6XG4gICAgICAgIHJldHVybiB0aGlzLl9yZW5kZXJEYXlWaWV3KCk7XG4gICAgfVxuICB9XG5cbiAgcHJpdmF0ZSBfdGFibGVBbmltYXRpb25FbmQoZXZlbnQ6IEFuaW1hdGlvbkV2ZW50KTogdm9pZCB7XG4gICAgY29uc3QgdGFibGUgPSBldmVudC50YXJnZXQgYXMgSFRNTEVsZW1lbnQ7XG4gICAgaWYgKGV2ZW50LmFuaW1hdGlvbk5hbWUgPT09ICdoaWRlJykge1xuICAgICAgdGFibGUuY2xhc3NMaXN0LnJlbW92ZSgnc2JiLWNhbGVuZGFyX190YWJsZS1oaWRlJyk7XG4gICAgICBpZiAodGhpcy5fY29udGFpbmluZ0ZvY3VzKSB7XG4gICAgICAgIHRoaXMuX3Jlc2V0Rm9jdXMgPSB0cnVlO1xuICAgICAgfVxuICAgICAgdGhpcy5fY2FsZW5kYXJWaWV3ID0gdGhpcy5fbmV4dENhbGVuZGFyVmlldztcbiAgICB9IGVsc2UgaWYgKGV2ZW50LmFuaW1hdGlvbk5hbWUgPT09ICdzaG93Jykge1xuICAgICAgdGhpcy5pbnRlcm5hbHMuc3RhdGVzLmRlbGV0ZSgndHJhbnNpdGlvbicpO1xuICAgIH1cbiAgfVxuXG4gIHByaXZhdGUgX3N0YXJ0VGFibGVUcmFuc2l0aW9uKCk6IHZvaWQge1xuICAgIHRoaXMuaW50ZXJuYWxzLnN0YXRlcy5hZGQoJ3RyYW5zaXRpb24nKTtcbiAgICB0aGlzLnNoYWRvd1Jvb3RcbiAgICAgID8ucXVlcnlTZWxlY3RvckFsbCgndGFibGUnKVxuICAgICAgPy5mb3JFYWNoKChlKSA9PiBlLmNsYXNzTGlzdC50b2dnbGUoJ3NiYi1jYWxlbmRhcl9fdGFibGUtaGlkZScpKTtcbiAgfVxuXG4gIHByb3RlY3RlZCBvdmVycmlkZSByZW5kZXIoKTogVGVtcGxhdGVSZXN1bHQge1xuICAgIHJldHVybiBodG1sYDxkaXYgY2xhc3M9XCJzYmItY2FsZW5kYXJfX3dyYXBwZXJcIj4ke3RoaXMuX2dldFZpZXcoKX08L2Rpdj5gO1xuICB9XG59XG5cbmRlY2xhcmUgZ2xvYmFsIHtcbiAgaW50ZXJmYWNlIEhUTUxFbGVtZW50VGFnTmFtZU1hcCB7XG4gICAgLy8gZXNsaW50LWRpc2FibGUtbmV4dC1saW5lIEB0eXBlc2NyaXB0LWVzbGludC9uYW1pbmctY29udmVudGlvblxuICAgICdzYmItY2FsZW5kYXInOiBTYmJDYWxlbmRhckVsZW1lbnQ7XG4gIH1cbiAgaW50ZXJmYWNlIEhUTUxFbGVtZW50RXZlbnRNYXAge1xuICAgIG1vbnRoY2hhbmdlOiBTYmJNb250aENoYW5nZUV2ZW50O1xuICB9XG59XG4iXSwibWFwcGluZ3MiOiI7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7OztBQzJEQSwwQkFBMEIsUUFBUTtBQUVsQyxzQkFBc0IsUUFBUTtBQUM5Qix3QkFBd0IsUUFBUTtBQUNoQyx1QkFBdUIsUUFBUTtBQUMvQiwwQkFBMEIsUUFBUTtBQUNsQyw2QkFBNkIsUUFBUTtBQUVyQyxJQUFhLHNCQUFiLGNBQXlDLE1BQUs7Q0FHNUMsSUFBVyxRQUFLO0FBQ2QsU0FBTyxLQUFLOztD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
|