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