barsa-calendar 0.0.2
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/README.md +25 -0
- package/esm2020/barsa-calendar.mjs +5 -0
- package/esm2020/lib/barsa-calendar.module.mjs +106 -0
- package/esm2020/lib/calendar-container/calendar-container.component.mjs +190 -0
- package/esm2020/lib/calendar-header/calendar-header.component.mjs +61 -0
- package/esm2020/lib/calendar-list/calendar-list.component.mjs +19 -0
- package/esm2020/lib/calendar-list-container/calendar-list-container.component.mjs +30 -0
- package/esm2020/lib/calendar-list-item/calendar-list-item.component.mjs +14 -0
- package/esm2020/lib/calendar-month/calendar-month.component.mjs +244 -0
- package/esm2020/lib/calendar-selection-days/calendar-selection-days.component.mjs +193 -0
- package/esm2020/lib/calendar-week/calendar-week.component.mjs +182 -0
- package/esm2020/lib/constants.mjs +4 -0
- package/esm2020/lib/date-time-picker/date-time-picker.component.mjs +324 -0
- package/esm2020/lib/day-event-list/day-event-list.component.mjs +53 -0
- package/esm2020/lib/day-number-box/day-number-box.component.mjs +105 -0
- package/esm2020/lib/days-in-week.pipe.mjs +17 -0
- package/esm2020/lib/equal-date.pipe.mjs +23 -0
- package/esm2020/lib/event-button/event-button.component.mjs +70 -0
- package/esm2020/lib/event-date.pipe.mjs +24 -0
- package/esm2020/lib/event-preview/event-preview.component.mjs +60 -0
- package/esm2020/lib/event-time.pipe.mjs +22 -0
- package/esm2020/lib/from-to-time.pipe.mjs +23 -0
- package/esm2020/lib/models.mjs +14 -0
- package/esm2020/lib/month-days/month-days.component.mjs +220 -0
- package/esm2020/lib/services/calendar.service.mjs +610 -0
- package/esm2020/public-api.mjs +21 -0
- package/fesm2015/barsa-calendar.mjs +2447 -0
- package/fesm2015/barsa-calendar.mjs.map +1 -0
- package/fesm2020/barsa-calendar.mjs +2455 -0
- package/fesm2020/barsa-calendar.mjs.map +1 -0
- package/index.d.ts +5 -0
- package/lib/barsa-calendar.module.d.ts +33 -0
- package/lib/calendar-container/calendar-container.component.d.ts +73 -0
- package/lib/calendar-header/calendar-header.component.d.ts +29 -0
- package/lib/calendar-list/calendar-list.component.d.ts +8 -0
- package/lib/calendar-list-container/calendar-list-container.component.d.ts +14 -0
- package/lib/calendar-list-item/calendar-list-item.component.d.ts +7 -0
- package/lib/calendar-month/calendar-month.component.d.ts +70 -0
- package/lib/calendar-selection-days/calendar-selection-days.component.d.ts +47 -0
- package/lib/calendar-week/calendar-week.component.d.ts +57 -0
- package/lib/constants.d.ts +1 -0
- package/lib/date-time-picker/date-time-picker.component.d.ts +98 -0
- package/lib/day-event-list/day-event-list.component.d.ts +20 -0
- package/lib/day-number-box/day-number-box.component.d.ts +30 -0
- package/lib/days-in-week.pipe.d.ts +9 -0
- package/lib/equal-date.pipe.d.ts +8 -0
- package/lib/event-button/event-button.component.d.ts +26 -0
- package/lib/event-date.pipe.d.ts +9 -0
- package/lib/event-preview/event-preview.component.d.ts +19 -0
- package/lib/event-time.pipe.d.ts +9 -0
- package/lib/from-to-time.pipe.d.ts +9 -0
- package/lib/models.d.ts +107 -0
- package/lib/month-days/month-days.component.d.ts +69 -0
- package/lib/services/calendar.service.d.ts +70 -0
- package/package.json +31 -0
- package/public-api.d.ts +17 -0
|
@@ -0,0 +1,610 @@
|
|
|
1
|
+
import { Inject, Injectable, Optional } from '@angular/core';
|
|
2
|
+
import { BehaviorSubject } from 'rxjs';
|
|
3
|
+
import { takeUntil } from 'rxjs/operators';
|
|
4
|
+
import { BaseComponent, getUniqueId, getDateService } from 'barsa-novin-ray-core';
|
|
5
|
+
import { CalendarMonthInfo, DayStatus } from '../models';
|
|
6
|
+
import * as i0 from "@angular/core";
|
|
7
|
+
import * as i1 from "barsa-novin-ray-core";
|
|
8
|
+
export class CalendarService extends BaseComponent {
|
|
9
|
+
constructor(_logService, culture) {
|
|
10
|
+
super();
|
|
11
|
+
this._logService = _logService;
|
|
12
|
+
this._calendarsSource = new BehaviorSubject([]);
|
|
13
|
+
this._modeSource = new BehaviorSubject('month');
|
|
14
|
+
this._currentMonthIndexSource = new BehaviorSubject(-1);
|
|
15
|
+
this._weekModeDaysSource = new BehaviorSubject([]);
|
|
16
|
+
this._daysSource = new BehaviorSubject([]);
|
|
17
|
+
this._dateService = getDateService(_logService, culture);
|
|
18
|
+
this.days$ = this._daysSource.asObservable();
|
|
19
|
+
this.mode$ = this._modeSource.asObservable();
|
|
20
|
+
this.weekModeDays$ = this._weekModeDaysSource.asObservable();
|
|
21
|
+
this.calendars$ = this._calendarsSource.asObservable();
|
|
22
|
+
this.currentMonthIndex$ = this._currentMonthIndexSource.asObservable();
|
|
23
|
+
this._initilize();
|
|
24
|
+
}
|
|
25
|
+
get todayMonthInfo() {
|
|
26
|
+
return this._todayMonthInfo;
|
|
27
|
+
}
|
|
28
|
+
get selectedMonth() {
|
|
29
|
+
return this._selectedSource.getValue();
|
|
30
|
+
}
|
|
31
|
+
get calendars() {
|
|
32
|
+
return this._calendarsSource.getValue();
|
|
33
|
+
}
|
|
34
|
+
get weeksCount() {
|
|
35
|
+
return this._daysSource.getValue().length / 7;
|
|
36
|
+
}
|
|
37
|
+
get days() {
|
|
38
|
+
return this._daysSource.getValue();
|
|
39
|
+
}
|
|
40
|
+
get Months() {
|
|
41
|
+
return this._dateService.monthNames();
|
|
42
|
+
}
|
|
43
|
+
loadMonthByDate(value) {
|
|
44
|
+
const monthInfo = this._dateService.getDateInfo(value);
|
|
45
|
+
this.loadMonth(monthInfo);
|
|
46
|
+
}
|
|
47
|
+
setDateService(primaryCalendarType) {
|
|
48
|
+
let dateService = this._dateService;
|
|
49
|
+
switch (primaryCalendarType) {
|
|
50
|
+
case 'Persian':
|
|
51
|
+
dateService = getDateService(this._logService, 'fa-IR');
|
|
52
|
+
break;
|
|
53
|
+
case 'Gregorian':
|
|
54
|
+
dateService = getDateService(this._logService, 'en-US');
|
|
55
|
+
break;
|
|
56
|
+
case 'Arabic':
|
|
57
|
+
dateService = getDateService(this._logService, 'ar-AE');
|
|
58
|
+
break;
|
|
59
|
+
}
|
|
60
|
+
this._dateService = dateService;
|
|
61
|
+
this._initilize();
|
|
62
|
+
}
|
|
63
|
+
setDays(days) {
|
|
64
|
+
this._daysSource.next(days);
|
|
65
|
+
}
|
|
66
|
+
prepareWeek(monthInfo, startIndex, daysCount) {
|
|
67
|
+
const startDays = monthInfo.days.slice(startIndex, startIndex + daysCount);
|
|
68
|
+
this._weekModeDaysSource.next(startDays);
|
|
69
|
+
}
|
|
70
|
+
setSelected(selected) {
|
|
71
|
+
this._selectedSource.next(selected);
|
|
72
|
+
}
|
|
73
|
+
weekDaysMin() {
|
|
74
|
+
return this._dateService.weekdaysMin();
|
|
75
|
+
}
|
|
76
|
+
weekDays() {
|
|
77
|
+
return this._dateService.weekDays();
|
|
78
|
+
}
|
|
79
|
+
gotoToday() {
|
|
80
|
+
this.loadMonth(this._todayMonthInfo.dateInfo, 0);
|
|
81
|
+
this.setMode(this._modeSource.getValue(), this._todayMonthInfo.days.find((c) => c.isToday), this._weekModeDaysSource.getValue().length);
|
|
82
|
+
}
|
|
83
|
+
rangeSelect(startDay, endDay, monthInfo) {
|
|
84
|
+
const diff = endDay.index - startDay.index;
|
|
85
|
+
if (diff < 7) {
|
|
86
|
+
this.setMode('week', startDay, diff + 1);
|
|
87
|
+
}
|
|
88
|
+
else {
|
|
89
|
+
const diff2 = diff + 1;
|
|
90
|
+
this.setMode('customMonth');
|
|
91
|
+
this.setMonth(monthInfo, startDay, diff2);
|
|
92
|
+
}
|
|
93
|
+
}
|
|
94
|
+
setMonth(monthInfo, startDay, diff = 0) {
|
|
95
|
+
if (startDay && diff > 0) {
|
|
96
|
+
const startIndex = startDay.index;
|
|
97
|
+
this.loadCalendarMonthInfo(monthInfo);
|
|
98
|
+
this.setDays(monthInfo.days.slice(startIndex, startIndex + diff));
|
|
99
|
+
}
|
|
100
|
+
else {
|
|
101
|
+
this.loadCalendarMonthInfo(monthInfo);
|
|
102
|
+
this.setDays(monthInfo.days);
|
|
103
|
+
// else if (mode === 'month') {
|
|
104
|
+
// startDay = daysLoaded[0];
|
|
105
|
+
// if (startDay) this.loadMonth(startDay.monthDateInfo);
|
|
106
|
+
// }
|
|
107
|
+
}
|
|
108
|
+
}
|
|
109
|
+
setMode(mode, startDay, daysCount) {
|
|
110
|
+
const oldMode = this._modeSource.getValue();
|
|
111
|
+
this._modeSource.next(mode);
|
|
112
|
+
if (mode === 'month') {
|
|
113
|
+
this.setMonth(this.selectedMonth, startDay);
|
|
114
|
+
return;
|
|
115
|
+
}
|
|
116
|
+
else if (mode === 'customMonth') {
|
|
117
|
+
this.setMonth(this.selectedMonth, startDay, this.days.length);
|
|
118
|
+
return;
|
|
119
|
+
}
|
|
120
|
+
const daysLoaded = this._weekModeDaysSource.getValue();
|
|
121
|
+
daysCount = daysCount ?? (mode === 'week' ? 7 : 1);
|
|
122
|
+
let startIndex = 0;
|
|
123
|
+
if (!startDay) {
|
|
124
|
+
if (mode === 'week' || mode === 'day') {
|
|
125
|
+
if (oldMode === 'month') {
|
|
126
|
+
startDay = this.selectedMonth.days.find((c) => c.isStartOfMonth);
|
|
127
|
+
}
|
|
128
|
+
else if (oldMode === 'customMonth') {
|
|
129
|
+
startDay = this.days[0];
|
|
130
|
+
}
|
|
131
|
+
else {
|
|
132
|
+
startDay = daysLoaded[0];
|
|
133
|
+
}
|
|
134
|
+
}
|
|
135
|
+
}
|
|
136
|
+
if (startDay) {
|
|
137
|
+
if (startDay.monthDateInfo.month !== this.selectedMonth.dateInfo.month ||
|
|
138
|
+
startDay.monthDateInfo.year !== this.selectedMonth.dateInfo.year) {
|
|
139
|
+
this.loadMonth(startDay.monthDateInfo);
|
|
140
|
+
}
|
|
141
|
+
startIndex = this.selectedMonth.days.findIndex((c) => c.month === startDay?.month && c.dayInMonth === startDay.dayInMonth);
|
|
142
|
+
}
|
|
143
|
+
this.prepareWeek(this._selectedSource.getValue(), startIndex, daysCount);
|
|
144
|
+
}
|
|
145
|
+
fillWeek(week, daysCount = 7) {
|
|
146
|
+
const arr = Object.values(week);
|
|
147
|
+
arr.forEach((element) => {
|
|
148
|
+
for (const el of element) {
|
|
149
|
+
const valueDate = el.end;
|
|
150
|
+
if (valueDate && this._dateService.getMoment(valueDate).isBefore(this.today, 'day')) {
|
|
151
|
+
el.opacity = 0.5;
|
|
152
|
+
}
|
|
153
|
+
}
|
|
154
|
+
for (let i = element.length; i < daysCount; i++) {
|
|
155
|
+
element.push({ colSpan: 1 });
|
|
156
|
+
}
|
|
157
|
+
});
|
|
158
|
+
}
|
|
159
|
+
sortTasks(calendarTasks, calendarFields) {
|
|
160
|
+
const startDateField = calendarFields.StartDate;
|
|
161
|
+
const endDateField = calendarFields.EndDate;
|
|
162
|
+
calendarTasks.sort((taskA, taskB) => (taskA[startDateField] < taskB[startDateField] ? -1 : 1));
|
|
163
|
+
calendarTasks.sort((taskA, taskB) => {
|
|
164
|
+
const taskAStartDate = taskA[startDateField];
|
|
165
|
+
const taskBStartDate = taskB[startDateField];
|
|
166
|
+
const momentTaskAStartDate = this._dateService.getMoment(taskAStartDate);
|
|
167
|
+
const momentTaskBStartDate = this._dateService.getMoment(taskBStartDate);
|
|
168
|
+
if (momentTaskAStartDate.isSame(momentTaskBStartDate, 'day')) {
|
|
169
|
+
const taskAEndDate = taskA[endDateField];
|
|
170
|
+
const taskBEndDate = taskB[endDateField];
|
|
171
|
+
const dayADiff = momentTaskAStartDate.diff(taskAEndDate, 'day');
|
|
172
|
+
const dayBDiff = momentTaskBStartDate.diff(taskBEndDate, 'day');
|
|
173
|
+
if (dayADiff > dayBDiff) {
|
|
174
|
+
return 1;
|
|
175
|
+
}
|
|
176
|
+
if (dayADiff < dayBDiff) {
|
|
177
|
+
return -1;
|
|
178
|
+
}
|
|
179
|
+
return 0;
|
|
180
|
+
}
|
|
181
|
+
{
|
|
182
|
+
return 0;
|
|
183
|
+
}
|
|
184
|
+
});
|
|
185
|
+
return calendarTasks;
|
|
186
|
+
}
|
|
187
|
+
createWeekTask(days, moDataList, calendarFields, taskHeight) {
|
|
188
|
+
const week = {};
|
|
189
|
+
days.forEach((day, dayIndex) => {
|
|
190
|
+
let counter = 0;
|
|
191
|
+
moDataList.forEach((event) => {
|
|
192
|
+
const counterIndex = counter.toString();
|
|
193
|
+
if (!week[counterIndex]) {
|
|
194
|
+
week[counterIndex] = [];
|
|
195
|
+
}
|
|
196
|
+
else if (week[counterIndex].length > dayIndex) {
|
|
197
|
+
counter++;
|
|
198
|
+
return;
|
|
199
|
+
}
|
|
200
|
+
for (let m = week[counterIndex].length; m < dayIndex; m++) {
|
|
201
|
+
week[counterIndex].push({ colSpan: 1 });
|
|
202
|
+
}
|
|
203
|
+
// agar week[counter] colspan ja dasht anjam gardad.
|
|
204
|
+
const newTask = { colSpan: 1 };
|
|
205
|
+
const momentDay = this._dateService.getMoment(day.date);
|
|
206
|
+
const momentOfStartEvent = this._dateService.getMoment(event[calendarFields.StartDate]);
|
|
207
|
+
const endDate = event[calendarFields.EndDate];
|
|
208
|
+
const momentOfEndEvent = this._dateService.getMoment(endDate);
|
|
209
|
+
const diff = Math.abs(momentOfStartEvent.diff(momentOfEndEvent, 'day'));
|
|
210
|
+
if (diff === 0 && momentDay.isSame(momentOfStartEvent, 'day')) {
|
|
211
|
+
newTask.colSpan = 1;
|
|
212
|
+
const startMinuteTop = (event.$StartDateInfo.minutes * taskHeight) / 60;
|
|
213
|
+
newTask.top = event.$StartDateInfo.hour * taskHeight + startMinuteTop;
|
|
214
|
+
const endMinuteTop = (event.$EndDateInfo.minutes * taskHeight) / 60;
|
|
215
|
+
newTask.height =
|
|
216
|
+
(event.$EndDateInfo.hour - event.$StartDateInfo.hour) * taskHeight +
|
|
217
|
+
endMinuteTop -
|
|
218
|
+
startMinuteTop;
|
|
219
|
+
newTask.event = event;
|
|
220
|
+
newTask.enddayIsInRow = true;
|
|
221
|
+
newTask.startdayIsInRow = true;
|
|
222
|
+
newTask.isStartDay = true;
|
|
223
|
+
newTask.isEndDay = true;
|
|
224
|
+
newTask.start = newTask.end = day.date;
|
|
225
|
+
newTask.inDay =
|
|
226
|
+
event.$StartDateInfo.hour > 0 ||
|
|
227
|
+
event.$StartDateInfo.minutes > 0 ||
|
|
228
|
+
event.$EndDateInfo.hour > 0 ||
|
|
229
|
+
event.$EndDateInfo.minutes > 0;
|
|
230
|
+
week[counterIndex].splice(dayIndex, 0, newTask);
|
|
231
|
+
day.tasks.push(newTask);
|
|
232
|
+
counter++;
|
|
233
|
+
}
|
|
234
|
+
else {
|
|
235
|
+
let diffFromEndOfEvent = 0;
|
|
236
|
+
let diffFromStartOfEvent = 0;
|
|
237
|
+
const isBetween = momentDay.isSameOrAfter(momentOfStartEvent, 'day') &&
|
|
238
|
+
momentDay.isSameOrBefore(momentOfEndEvent, 'day');
|
|
239
|
+
if (isBetween) {
|
|
240
|
+
diffFromEndOfEvent = 0;
|
|
241
|
+
if (diff === 0) {
|
|
242
|
+
newTask.colSpan = 1;
|
|
243
|
+
newTask.event = event;
|
|
244
|
+
}
|
|
245
|
+
else {
|
|
246
|
+
diffFromEndOfEvent = Math.abs(momentDay.diff(momentOfEndEvent, 'day'));
|
|
247
|
+
diffFromStartOfEvent = Math.abs(momentDay.diff(momentOfStartEvent, 'day'));
|
|
248
|
+
const isStartDay = momentDay.isSame(momentOfStartEvent, 'day');
|
|
249
|
+
const isEndDay = momentDay.isSame(momentOfEndEvent, 'day');
|
|
250
|
+
let colSpan = diff;
|
|
251
|
+
if (!isStartDay) {
|
|
252
|
+
colSpan = diffFromEndOfEvent;
|
|
253
|
+
}
|
|
254
|
+
else {
|
|
255
|
+
colSpan = diff + 1; // include the start
|
|
256
|
+
}
|
|
257
|
+
newTask.colSpan = this._calcColSpan(colSpan, dayIndex, days.length);
|
|
258
|
+
newTask.isStartDay = isStartDay;
|
|
259
|
+
newTask.isEndDay = isEndDay;
|
|
260
|
+
newTask.isMiddle = !isStartDay && !isEndDay;
|
|
261
|
+
newTask.event = event;
|
|
262
|
+
}
|
|
263
|
+
if (newTask.isStartDay || dayIndex === 0) {
|
|
264
|
+
for (let span = 0; span < newTask.colSpan; span++) {
|
|
265
|
+
const taskIndex = dayIndex + span;
|
|
266
|
+
if (taskIndex > days.length - 1) {
|
|
267
|
+
break;
|
|
268
|
+
}
|
|
269
|
+
const taskDayInfo = days[taskIndex];
|
|
270
|
+
const taskInfo = {
|
|
271
|
+
...newTask,
|
|
272
|
+
start: taskDayInfo.date,
|
|
273
|
+
end: endDate,
|
|
274
|
+
isStartDay: span === 0 ? newTask.isStartDay : null,
|
|
275
|
+
startTask: event.$StartDayInfo,
|
|
276
|
+
endTask: event.$EndDayInfo,
|
|
277
|
+
startdayIsInRow: dayIndex - diffFromStartOfEvent > -1,
|
|
278
|
+
enddayIsInRow: !!days[diffFromEndOfEvent],
|
|
279
|
+
isMiddle: span === 0
|
|
280
|
+
? newTask.isMiddle
|
|
281
|
+
: this._dateService
|
|
282
|
+
.getMoment(taskDayInfo.date)
|
|
283
|
+
.isBetween(momentOfStartEvent, momentOfEndEvent, 'day'),
|
|
284
|
+
isEndDay: span === 0
|
|
285
|
+
? newTask.isEndDay
|
|
286
|
+
: Math.abs(this._dateService
|
|
287
|
+
.getMoment(taskDayInfo.date)
|
|
288
|
+
.diff(momentOfEndEvent, 'day')) <= 1,
|
|
289
|
+
colSpan: span === 0 ? newTask.colSpan : null
|
|
290
|
+
};
|
|
291
|
+
week[counter.toString()].splice(taskIndex, 0, taskInfo);
|
|
292
|
+
taskDayInfo.tasks.push(taskInfo);
|
|
293
|
+
}
|
|
294
|
+
counter++;
|
|
295
|
+
}
|
|
296
|
+
}
|
|
297
|
+
}
|
|
298
|
+
});
|
|
299
|
+
});
|
|
300
|
+
return week;
|
|
301
|
+
}
|
|
302
|
+
loadNext() {
|
|
303
|
+
const mode = this._modeSource.getValue();
|
|
304
|
+
const selectedMonth = this._selectedSource.getValue();
|
|
305
|
+
switch (mode) {
|
|
306
|
+
case 'customMonth':
|
|
307
|
+
case 'month':
|
|
308
|
+
{
|
|
309
|
+
const days = this.days;
|
|
310
|
+
const newDays = this._loadNextMonth(days, selectedMonth);
|
|
311
|
+
this.setDays(newDays);
|
|
312
|
+
}
|
|
313
|
+
break;
|
|
314
|
+
case 'week':
|
|
315
|
+
case 'day':
|
|
316
|
+
{
|
|
317
|
+
const days = this._weekModeDaysSource.getValue();
|
|
318
|
+
const newDays = this._loadNextWeek(days, selectedMonth);
|
|
319
|
+
this._weekModeDaysSource.next(newDays);
|
|
320
|
+
}
|
|
321
|
+
break;
|
|
322
|
+
break;
|
|
323
|
+
case 'list':
|
|
324
|
+
break;
|
|
325
|
+
}
|
|
326
|
+
}
|
|
327
|
+
gotoMonth(monthName, monthIndex) {
|
|
328
|
+
const selectedMonth = this.selectedMonth.dateInfo.month;
|
|
329
|
+
const val = monthIndex - selectedMonth;
|
|
330
|
+
this.loadMonth(this.selectedMonth.dateInfo, val);
|
|
331
|
+
}
|
|
332
|
+
gotoYear(year) {
|
|
333
|
+
const selectedYear = this.selectedMonth.dateInfo.year;
|
|
334
|
+
const val = year - selectedYear;
|
|
335
|
+
this.increaseYear(val);
|
|
336
|
+
}
|
|
337
|
+
loadNextYear() {
|
|
338
|
+
this.increaseYear(1);
|
|
339
|
+
}
|
|
340
|
+
loadPrevYear() {
|
|
341
|
+
this.increaseYear(-1);
|
|
342
|
+
}
|
|
343
|
+
increaseYear(value) {
|
|
344
|
+
const selectedYear = this.selectedMonth.dateInfo.moment.toDate();
|
|
345
|
+
selectedYear.setFullYear(selectedYear.getFullYear() + value); // next year
|
|
346
|
+
this.loadMonthByDate(selectedYear);
|
|
347
|
+
}
|
|
348
|
+
loadPrevious() {
|
|
349
|
+
const mode = this._modeSource.getValue();
|
|
350
|
+
const selectedMonth = this._selectedSource.getValue();
|
|
351
|
+
switch (mode) {
|
|
352
|
+
case 'customMonth':
|
|
353
|
+
case 'month':
|
|
354
|
+
{
|
|
355
|
+
const days = this.days;
|
|
356
|
+
const newDays = this._loadPrevMonth(days, selectedMonth);
|
|
357
|
+
this.setDays(newDays);
|
|
358
|
+
}
|
|
359
|
+
break;
|
|
360
|
+
case 'week':
|
|
361
|
+
case 'day':
|
|
362
|
+
{
|
|
363
|
+
const days = this._weekModeDaysSource.getValue();
|
|
364
|
+
const newDays = this._loadPrevWeek(days, selectedMonth);
|
|
365
|
+
this._weekModeDaysSource.next(newDays);
|
|
366
|
+
}
|
|
367
|
+
break;
|
|
368
|
+
break;
|
|
369
|
+
case 'list':
|
|
370
|
+
break;
|
|
371
|
+
}
|
|
372
|
+
}
|
|
373
|
+
loadMonth(dateInfo, addMonth = 0) {
|
|
374
|
+
let loadedMonthDate = this._dateService.getMoment(dateInfo.date);
|
|
375
|
+
if (addMonth > 0) {
|
|
376
|
+
loadedMonthDate = this._dateService.addMonth(loadedMonthDate, addMonth);
|
|
377
|
+
}
|
|
378
|
+
if (addMonth < 0) {
|
|
379
|
+
loadedMonthDate = this._dateService.subtractMonth(loadedMonthDate, Math.abs(addMonth));
|
|
380
|
+
}
|
|
381
|
+
let loadedCalendarMonthInfo = this.calendars.find((c) => c.dateInfo.moment.isSame(loadedMonthDate, 'day'));
|
|
382
|
+
if (!loadedCalendarMonthInfo) {
|
|
383
|
+
loadedCalendarMonthInfo = this.getMonthInfo(loadedMonthDate, this.today);
|
|
384
|
+
}
|
|
385
|
+
this.loadCalendarMonthInfo(loadedCalendarMonthInfo);
|
|
386
|
+
return loadedCalendarMonthInfo;
|
|
387
|
+
}
|
|
388
|
+
loadCalendarMonthInfo(loadedCalendarMonthInfo) {
|
|
389
|
+
const calendarsInfo = [...this.calendars];
|
|
390
|
+
const exists = calendarsInfo.find((c) => c.dateInfo.year === loadedCalendarMonthInfo.dateInfo.year &&
|
|
391
|
+
c.dateInfo.month === loadedCalendarMonthInfo.dateInfo.month);
|
|
392
|
+
if (!exists) {
|
|
393
|
+
calendarsInfo.push(loadedCalendarMonthInfo);
|
|
394
|
+
this._calendarsSource.next(calendarsInfo);
|
|
395
|
+
}
|
|
396
|
+
this.setSelected(loadedCalendarMonthInfo);
|
|
397
|
+
}
|
|
398
|
+
addCalendarMonthInfo(calendarMonthInfo) {
|
|
399
|
+
const newList = this.calendars;
|
|
400
|
+
newList.push(calendarMonthInfo);
|
|
401
|
+
this._calendarsSource.next(newList);
|
|
402
|
+
}
|
|
403
|
+
getNumber(value) {
|
|
404
|
+
const find = ['۰', '۱', '۲', '۳', '۴', '۵', '۶', '۷', '۸', '۹'];
|
|
405
|
+
const replace = ['0', '1', '2', '3', '4', '5', '6', '7', '8', '9'];
|
|
406
|
+
let replaceString = value;
|
|
407
|
+
let regex;
|
|
408
|
+
for (let i = 0; i < find.length; i++) {
|
|
409
|
+
regex = new RegExp(find[i], 'g');
|
|
410
|
+
replaceString = replaceString.replace(regex, replace[i]);
|
|
411
|
+
}
|
|
412
|
+
return Number(replaceString);
|
|
413
|
+
}
|
|
414
|
+
_initilize() {
|
|
415
|
+
this.today = this._dateService.getMoment(new Date());
|
|
416
|
+
const todayMonthInfo = this.getCalendarMonthInfo(new Date());
|
|
417
|
+
this.todayDay = todayMonthInfo.days.find((c) => c.isToday);
|
|
418
|
+
this._selectedSource = new BehaviorSubject(todayMonthInfo);
|
|
419
|
+
this.selected$ = this._selectedSource.asObservable();
|
|
420
|
+
this._calendarsSource.next([todayMonthInfo]);
|
|
421
|
+
this.setDays(todayMonthInfo.days);
|
|
422
|
+
this.setSelected(todayMonthInfo);
|
|
423
|
+
this._currentMonthIndexSource.next(todayMonthInfo.dateInfo.month);
|
|
424
|
+
this._todayMonthInfo = todayMonthInfo;
|
|
425
|
+
this.calendars$.pipe(takeUntil(this._onDestroy$)).subscribe();
|
|
426
|
+
}
|
|
427
|
+
_calcColSpan(diff, dayIndex, maxColSpan) {
|
|
428
|
+
if (diff === 0) {
|
|
429
|
+
return 2;
|
|
430
|
+
}
|
|
431
|
+
if (Math.floor(diff / maxColSpan) > 0) {
|
|
432
|
+
return maxColSpan;
|
|
433
|
+
}
|
|
434
|
+
if (dayIndex + diff > maxColSpan) {
|
|
435
|
+
return maxColSpan - dayIndex;
|
|
436
|
+
}
|
|
437
|
+
return diff;
|
|
438
|
+
}
|
|
439
|
+
_loadNextMonth(days, monthInfo) {
|
|
440
|
+
const daysCount = days.length;
|
|
441
|
+
let lastDay = days[daysCount - 1];
|
|
442
|
+
const lastDayIndex = monthInfo.days.findIndex((c) => c.dayInMonth === lastDay.dayInMonth && c.month === lastDay.month);
|
|
443
|
+
let newDays = lastDayIndex > -1 ? monthInfo.days.slice(lastDayIndex + 1, lastDayIndex + 1 + daysCount) : [];
|
|
444
|
+
if (newDays.length < daysCount) {
|
|
445
|
+
// we move to next month
|
|
446
|
+
if (newDays.length > 0) {
|
|
447
|
+
lastDay = newDays[newDays.length - 1];
|
|
448
|
+
}
|
|
449
|
+
const needCountDay = daysCount - newDays.length;
|
|
450
|
+
const nextMonth = this.loadMonth(monthInfo.dateInfo, 1);
|
|
451
|
+
let startPosition = 0;
|
|
452
|
+
if (needCountDay < 42) {
|
|
453
|
+
startPosition = nextMonth.days.findIndex((c) => c.dayInMonth === lastDay.dayInMonth && c.month === lastDay.month);
|
|
454
|
+
if (startPosition === -1) {
|
|
455
|
+
startPosition = 0;
|
|
456
|
+
}
|
|
457
|
+
else {
|
|
458
|
+
startPosition++;
|
|
459
|
+
}
|
|
460
|
+
}
|
|
461
|
+
const nextDays = nextMonth.days.slice(startPosition, startPosition + needCountDay);
|
|
462
|
+
newDays = [...newDays, ...nextDays];
|
|
463
|
+
}
|
|
464
|
+
return newDays;
|
|
465
|
+
}
|
|
466
|
+
_loadNextWeek(days, monthInfo) {
|
|
467
|
+
const daysCount = days.length;
|
|
468
|
+
const lastDay = days[daysCount - 1];
|
|
469
|
+
const lastDayIndex = monthInfo.days.findIndex((c) => c.dayInMonth === lastDay.dayInMonth && c.month === lastDay.month);
|
|
470
|
+
let newDays = monthInfo.days.slice(lastDayIndex + 1, lastDayIndex + 1 + daysCount);
|
|
471
|
+
if (newDays.length < daysCount) {
|
|
472
|
+
// we move to next month
|
|
473
|
+
let lastNewDay = newDays[newDays.length - 1];
|
|
474
|
+
if (!lastNewDay) {
|
|
475
|
+
lastNewDay = lastDay;
|
|
476
|
+
}
|
|
477
|
+
const nextMonth = this.loadMonth(monthInfo.dateInfo, 1);
|
|
478
|
+
const lastDayIndex2 = nextMonth.days.findIndex((c) => c.dayInMonth === lastNewDay.dayInMonth && c.month === lastNewDay.month);
|
|
479
|
+
newDays = newDays.concat(nextMonth.days.slice(lastDayIndex2 + 1, lastDayIndex2 + 1 + (daysCount - newDays.length)));
|
|
480
|
+
}
|
|
481
|
+
return newDays;
|
|
482
|
+
}
|
|
483
|
+
_loadPrevMonth(days, monthInfo) {
|
|
484
|
+
const daysCount = days.length;
|
|
485
|
+
let firstDay = days[0];
|
|
486
|
+
const firstDayIndex = monthInfo.days.findIndex((c) => c.dayInMonth === firstDay.dayInMonth && c.month === firstDay.month);
|
|
487
|
+
let newDays = monthInfo.days.slice(firstDayIndex - daysCount > 0 ? firstDayIndex - daysCount : 0, firstDayIndex);
|
|
488
|
+
if (newDays.length < daysCount) {
|
|
489
|
+
// we move to prev month
|
|
490
|
+
if (newDays.length > 0) {
|
|
491
|
+
firstDay = newDays[0];
|
|
492
|
+
}
|
|
493
|
+
const needCountDay = daysCount - newDays.length;
|
|
494
|
+
const prevMonth = this.loadMonth(monthInfo.dateInfo, -1);
|
|
495
|
+
let endPosition = prevMonth.days.length;
|
|
496
|
+
if (needCountDay < 42) {
|
|
497
|
+
endPosition = prevMonth.days.findIndex((c) => c.dayInMonth === firstDay.dayInMonth && c.month === firstDay.month);
|
|
498
|
+
if (endPosition === -1) {
|
|
499
|
+
endPosition = prevMonth.days.length;
|
|
500
|
+
}
|
|
501
|
+
}
|
|
502
|
+
const prevDays = prevMonth.days.slice(endPosition - needCountDay, endPosition);
|
|
503
|
+
newDays = [...prevDays, ...newDays];
|
|
504
|
+
}
|
|
505
|
+
return newDays;
|
|
506
|
+
}
|
|
507
|
+
_loadPrevWeek(days, monthInfo) {
|
|
508
|
+
const daysCount = days.length;
|
|
509
|
+
const firstDay = days[0];
|
|
510
|
+
const firstDayIndex = monthInfo.days.findIndex((c) => c.dayInMonth === firstDay.dayInMonth && c.month === firstDay.month);
|
|
511
|
+
const partIndexPrev = firstDayIndex - daysCount;
|
|
512
|
+
let newDays = [];
|
|
513
|
+
if (partIndexPrev < 0) {
|
|
514
|
+
newDays = firstDayIndex - 1 >= 0 ? monthInfo.days.slice(0, firstDayIndex - 1) : [];
|
|
515
|
+
}
|
|
516
|
+
else {
|
|
517
|
+
newDays = monthInfo.days.slice(firstDayIndex - daysCount, firstDayIndex);
|
|
518
|
+
}
|
|
519
|
+
if (newDays.length < daysCount) {
|
|
520
|
+
// we move to prev month
|
|
521
|
+
let firstNewDay = newDays[0];
|
|
522
|
+
if (!firstNewDay) {
|
|
523
|
+
firstNewDay = firstDay;
|
|
524
|
+
}
|
|
525
|
+
const prevMonth = this.loadMonth(monthInfo.dateInfo, -1);
|
|
526
|
+
const firstDayIndex2 = prevMonth.days.findIndex((c) => c.dayInMonth === firstNewDay.dayInMonth && c.month === firstNewDay.month);
|
|
527
|
+
newDays = newDays.concat(prevMonth.days.slice(firstDayIndex2 - (daysCount - newDays.length), firstDayIndex2));
|
|
528
|
+
}
|
|
529
|
+
return newDays;
|
|
530
|
+
}
|
|
531
|
+
getCalendarMonthInfo(date) {
|
|
532
|
+
const startOfMonth = this._dateService.startOfMonth(date);
|
|
533
|
+
const momentDate = this._dateService.getMoment(date);
|
|
534
|
+
return this.getMonthInfo(startOfMonth, momentDate);
|
|
535
|
+
}
|
|
536
|
+
getMonthInfo(date, today) {
|
|
537
|
+
const days = [];
|
|
538
|
+
const activeDate = this._dateService.getMoment(date);
|
|
539
|
+
const startOfMonth = this._dateService.startOfMonth(activeDate);
|
|
540
|
+
const endOfMonth = this._dateService.endOfMonth(activeDate);
|
|
541
|
+
const dateInfo = this._dateService.getDateInfo(activeDate);
|
|
542
|
+
const dayInWeek = this._dateService.day(startOfMonth) + 1; // we add 1 to the dayInWeek because it's starts with zero
|
|
543
|
+
let weekCounter = 0;
|
|
544
|
+
let dayCounter = 0;
|
|
545
|
+
for (let i = 0; i < 42; i++) {
|
|
546
|
+
const dateOfStartOfMonth = new Date(startOfMonth);
|
|
547
|
+
dateOfStartOfMonth.setDate(dateOfStartOfMonth.getDate() + i - dayInWeek);
|
|
548
|
+
const { day, isInMonth } = this._getDay(dateOfStartOfMonth, activeDate, today, endOfMonth, startOfMonth, dateInfo, i);
|
|
549
|
+
days.push(day);
|
|
550
|
+
dayCounter++;
|
|
551
|
+
if (isInMonth && dayCounter > weekCounter * 7) {
|
|
552
|
+
weekCounter++;
|
|
553
|
+
}
|
|
554
|
+
}
|
|
555
|
+
// const endOfMonthIndex = days.findIndex((c) => c.isEndOfMonth);
|
|
556
|
+
// if (endOfMonthIndex > -1 && days.length - endOfMonthIndex > 7) {
|
|
557
|
+
// // days.splice(days.length - 7, 7);
|
|
558
|
+
// }
|
|
559
|
+
const calendarInfo = new CalendarMonthInfo();
|
|
560
|
+
calendarInfo.id = getUniqueId(3);
|
|
561
|
+
calendarInfo.dateInfo = dateInfo;
|
|
562
|
+
calendarInfo.days = days;
|
|
563
|
+
calendarInfo.weeksCount = weekCounter;
|
|
564
|
+
// this.cdr.detectChanges();
|
|
565
|
+
return calendarInfo;
|
|
566
|
+
}
|
|
567
|
+
_getDay(date, activeDate, today, endOfMonth, startOfMonth, dateInfo, i) {
|
|
568
|
+
const isInMonth = this._dateService.isSame(date, activeDate, 'month');
|
|
569
|
+
const momentOfDate = this._dateService.getMoment(date);
|
|
570
|
+
const isToday = momentOfDate.isSame(today, 'day');
|
|
571
|
+
const dayInMonth = this._dateService.strDate(date);
|
|
572
|
+
const dayInWeek = this._dateService.day(date);
|
|
573
|
+
const isEndOfMonth = momentOfDate.isSame(endOfMonth, 'day');
|
|
574
|
+
const isStartOfMonth = momentOfDate.isSame(startOfMonth, 'day');
|
|
575
|
+
const monthDateInfo = isInMonth ? dateInfo : this._dateService.getDateInfo(date);
|
|
576
|
+
const day = {
|
|
577
|
+
index: i,
|
|
578
|
+
date,
|
|
579
|
+
dayInWeek,
|
|
580
|
+
status: DayStatus.Open,
|
|
581
|
+
dayInMonth,
|
|
582
|
+
dayInMonthN: this.getNumber(dayInMonth),
|
|
583
|
+
isToday,
|
|
584
|
+
isInMonth,
|
|
585
|
+
isEndOfMonth,
|
|
586
|
+
isStartOfMonth,
|
|
587
|
+
momentOfDate,
|
|
588
|
+
month: monthDateInfo.month,
|
|
589
|
+
year: monthDateInfo.year,
|
|
590
|
+
monthName: monthDateInfo.monthName,
|
|
591
|
+
dayName: momentOfDate.format('dddd'),
|
|
592
|
+
tasks: [],
|
|
593
|
+
monthDateInfo,
|
|
594
|
+
row: Math.floor(i / 7 + 1),
|
|
595
|
+
col: 6 - (i % 7)
|
|
596
|
+
};
|
|
597
|
+
return { day, isInMonth };
|
|
598
|
+
}
|
|
599
|
+
}
|
|
600
|
+
CalendarService.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.2.7", ngImport: i0, type: CalendarService, deps: [{ token: i1.LogService }, { token: 'Date_Culture', optional: true }], target: i0.ɵɵFactoryTarget.Injectable });
|
|
601
|
+
CalendarService.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "14.2.7", ngImport: i0, type: CalendarService });
|
|
602
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.2.7", ngImport: i0, type: CalendarService, decorators: [{
|
|
603
|
+
type: Injectable
|
|
604
|
+
}], ctorParameters: function () { return [{ type: i1.LogService }, { type: undefined, decorators: [{
|
|
605
|
+
type: Optional
|
|
606
|
+
}, {
|
|
607
|
+
type: Inject,
|
|
608
|
+
args: ['Date_Culture']
|
|
609
|
+
}] }]; } });
|
|
610
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY2FsZW5kYXIuc2VydmljZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL2JhcnNhLWNhbGVuZGFyL3NyYy9saWIvc2VydmljZXMvY2FsZW5kYXIuc2VydmljZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsTUFBTSxFQUFFLFVBQVUsRUFBRSxRQUFRLEVBQUUsTUFBTSxlQUFlLENBQUM7QUFDN0QsT0FBTyxFQUFjLGVBQWUsRUFBRSxNQUFNLE1BQU0sQ0FBQztBQUNuRCxPQUFPLEVBQUUsU0FBUyxFQUFFLE1BQU0sZ0JBQWdCLENBQUM7QUFFM0MsT0FBTyxFQUVILGFBQWEsRUFDYixXQUFXLEVBQ1gsY0FBYyxFQUtqQixNQUFNLHNCQUFzQixDQUFDO0FBQzlCLE9BQU8sRUFBZ0IsaUJBQWlCLEVBQU8sU0FBUyxFQUFFLE1BQU0sV0FBVyxDQUFDOzs7QUFHNUUsTUFBTSxPQUFPLGVBQWdCLFNBQVEsYUFBYTtJQW1COUMsWUFBb0IsV0FBdUIsRUFBc0MsT0FBZ0I7UUFDN0YsS0FBSyxFQUFFLENBQUM7UUFEUSxnQkFBVyxHQUFYLFdBQVcsQ0FBWTtRQVBuQyxxQkFBZ0IsR0FBRyxJQUFJLGVBQWUsQ0FBc0IsRUFBRSxDQUFDLENBQUM7UUFFaEUsZ0JBQVcsR0FBRyxJQUFJLGVBQWUsQ0FBZSxPQUFPLENBQUMsQ0FBQztRQUN6RCw2QkFBd0IsR0FBRyxJQUFJLGVBQWUsQ0FBUyxDQUFDLENBQUMsQ0FBQyxDQUFDO1FBQzNELHdCQUFtQixHQUFHLElBQUksZUFBZSxDQUFRLEVBQUUsQ0FBQyxDQUFDO1FBQ3JELGdCQUFXLEdBQUcsSUFBSSxlQUFlLENBQVEsRUFBRSxDQUFDLENBQUM7UUFJakQsSUFBSSxDQUFDLFlBQVksR0FBRyxjQUFjLENBQUMsV0FBVyxFQUFFLE9BQU8sQ0FBQyxDQUFDO1FBQ3pELElBQUksQ0FBQyxLQUFLLEdBQUcsSUFBSSxDQUFDLFdBQVcsQ0FBQyxZQUFZ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Q0FBQztZQUNSLElBQUk7WUFDSixTQUFTO1lBQ1QsTUFBTSxFQUFFLFNBQVMsQ0FBQyxJQUFJO1lBQ3RCLFVBQVU7WUFDVixXQUFXLEVBQUUsSUFBSSxDQUFDLFNBQVMsQ0FBQyxVQUFVLENBQUM7WUFDdkMsT0FBTztZQUNQLFNBQVM7WUFDVCxZQUFZO1lBQ1osY0FBYztZQUNkLFlBQVk7WUFDWixLQUFLLEVBQUUsYUFBYSxDQUFDLEtBQUs7WUFDMUIsSUFBSSxFQUFFLGFBQWEsQ0FBQyxJQUFJO1lBQ3hCLFNBQVMsRUFBRSxhQUFhLENBQUMsU0FBUztZQUNsQyxPQUFPLEVBQUUsWUFBWSxDQUFDLE1BQU0sQ0FBQyxNQUFNLENBQUM7WUFDcEMsS0FBSyxFQUFFLEVBQUU7WUFDVCxhQUFhO1lBQ2IsR0FBRyxFQUFFLElBQUksQ0FBQyxLQUFLLENBQUMsQ0FBQyxHQUFHLENBQUMsR0FBRyxDQUFDLENBQUM7WUFDMUIsR0FBRyxFQUFFLENBQUMsR0FBRyxDQUFDLENBQUMsR0FBRyxDQUFDLENBQUM7U0FDbkIsQ0FBQztRQUNGLE9BQU8sRUFBRSxHQUFHLEVBQUUsU0FBUyxFQUFFLENBQUM7SUFDOUIsQ0FBQzs7NEdBdnFCUSxlQUFlLDRDQW1CeUMsY0FBYztnSEFuQnRFLGVBQWU7MkZBQWYsZUFBZTtrQkFEM0IsVUFBVTs7MEJBb0J1QyxRQUFROzswQkFBSSxNQUFNOzJCQUFDLGNBQWMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBJbmplY3QsIEluamVjdGFibGUsIE9wdGlvbmFsIH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQgeyBPYnNlcnZhYmxlLCBCZWhhdmlvclN1YmplY3QgfSBmcm9tICdyeGpzJztcbmltcG9ydCB7IHRha2VVbnRpbCB9IGZyb20gJ3J4anMvb3BlcmF0b3JzJztcblxuaW1wb3J0IHtcbiAgICBEYXRlU2VydmljZSxcbiAgICBCYXNlQ29tcG9uZW50LFxuICAgIGdldFVuaXF1ZUlkLFxuICAgIGdldERhdGVTZXJ2aWNlLFxuICAgIExvZ1NlcnZpY2UsXG4gICAgRGF0ZUluZm8sXG4gICAgTWV0YW9iamVjdERhdGFNb2RlbCxcbiAgICBDYWxlbmRhckZpZWxkc1xufSBmcm9tICdiYXJzYS1ub3Zpbi1yYXktY29yZSc7XG5pbXBvcnQgeyBDYWxlbmRhck1vZGUsIENhbGVuZGFyTW9udGhJbmZvLCBEYXksIERheVN0YXR1cyB9IGZyb20gJy4uL21vZGVscyc7XG5cbkBJbmplY3RhYmxlKClcbmV4cG9ydCBjbGFzcyBDYWxlbmRhclNlcnZpY2UgZXh0ZW5kcyBCYXNlQ29tcG9uZW50IHtcbiAgICBzZWxlY3RlZCQ6IE9ic2VydmFibGU8Q2FsZW5kYXJNb250aEluZm8+O1xuICAgIGRheXMkOiBPYnNlcnZhYmxlPERheVtdPjtcbiAgICBtb2RlJDogT2JzZXJ2YWJsZTxDYWxlbmRhck1vZGU+O1xuICAgIHdlZWtNb2RlRGF5cyQ6IE9ic2VydmFibGU8RGF5W10+O1xuICAgIGNhbGVuZGFycyQ6IE9ic2VydmFibGU8Q2FsZW5kYXJNb250aEluZm9bXT47XG4gICAgY3VycmVudE1vbnRoSW5kZXgkOiBPYnNlcnZhYmxlPG51bWJlcj47XG4gICAgdG9kYXk6IERhdGU7XG4gICAgdG9kYXlEYXk6IERheSB8IHVuZGVmaW5lZDtcbiAgICBfZGF0ZVNlcnZpY2U6IERhdGVTZXJ2aWNlO1xuXG4gICAgcHJpdmF0ZSBfdG9kYXlNb250aEluZm86IENhbGVuZGFyTW9udGhJbmZvO1xuICAgIHByaXZhdGUgX2NhbGVuZGFyc1NvdXJjZSA9IG5ldyBCZWhhdmlvclN1YmplY3Q8Q2FsZW5kYXJNb250aEluZm9bXT4oW10pO1xuICAgIHByaXZhdGUgX3NlbGVjdGVkU291cmNlOiBCZWhhdmlvclN1YmplY3Q8Q2FsZW5kYXJNb250aEluZm8+O1xuICAgIHByaXZhdGUgX21vZGVTb3VyY2UgPSBuZXcgQmVoYXZpb3JTdWJqZWN0PENhbGVuZGFyTW9kZT4oJ21vbnRoJyk7XG4gICAgcHJpdmF0ZSBfY3VycmVudE1vbnRoSW5kZXhTb3VyY2UgPSBuZXcgQmVoYXZpb3JTdWJqZWN0PG51bWJlcj4oLTEpO1xuICAgIHByaXZhdGUgX3dlZWtNb2RlRGF5c1NvdXJjZSA9IG5ldyBCZWhhdmlvclN1YmplY3Q8RGF5W10+KFtdKTtcbiAgICBwcml2YXRlIF9kYXlzU291cmNlID0gbmV3IEJlaGF2aW9yU3ViamVjdDxEYXlbXT4oW10pO1xuXG4gICAgY29uc3RydWN0b3IocHJpdmF0ZSBfbG9nU2VydmljZTogTG9nU2VydmljZSwgQE9wdGlvbmFsKCkgQEluamVjdCgnRGF0ZV9DdWx0dXJlJykgY3VsdHVyZT86IHN0cmluZykge1xuICAgICAgICBzdXBlcigpO1xuICAgICAgICB0aGlzLl9kYXRlU2VydmljZSA9IGdldERhdGVTZXJ2aWNlKF9sb2dTZXJ2aWNlLCBjdWx0dXJlKTtcbiAgICAgICAgdGhpcy5kYXlzJCA9IHRoaXMuX2RheXNTb3VyY2UuYXNPYnNlcnZhYmxlKCk7XG4gICAgICAgIHRoaXMubW9kZSQgPSB0aGlzLl9tb2RlU291cmNlLmFzT2JzZXJ2YWJsZSgpO1xuICAgICAgICB0aGlzLndlZWtNb2RlRGF5cyQgPSB0aGlzLl93ZWVrTW9kZURheXNTb3VyY2UuYXNPYnNlcnZhYmxlKCk7XG4gICAgICAgIHRoaXMuY2FsZW5kYXJzJCA9IHRoaXMuX2NhbGVuZGFyc1NvdXJjZS5hc09ic2VydmFibGUoKTtcbiAgICAgICAgdGhpcy5jdXJyZW50TW9udGhJbmRleCQgPSB0aGlzLl9jdXJyZW50TW9udGhJbmRleFNvdXJjZS5hc09ic2VydmFibGUoKTtcbiAgICAgICAgdGhpcy5faW5pdGlsaXplKCk7XG4gICAgfVxuICAgIGdldCB0b2RheU1vbnRoSW5mbygpOiBDYWxlbmRhck1vbnRoSW5mbyB7XG4gICAgICAgIHJldHVybiB0aGlzLl90b2RheU1vbnRoSW5mbztcbiAgICB9XG4gICAgZ2V0IHNlbGVjdGVkTW9udGgoKTogQ2FsZW5kYXJNb250aEluZm8ge1xuICAgICAgICByZXR1cm4gdGhpcy5fc2VsZWN0ZWRTb3VyY2UuZ2V0VmFsdWUoKTtcbiAgICB9XG4gICAgZ2V0IGNhbGVuZGFycygpOiBBcnJheTxDYWxlbmRhck1vbnRoSW5mbz4ge1xuICAgICAgICByZXR1cm4gdGhpcy5fY2FsZW5kYXJzU291cmNlLmdldFZhbHVlKCk7XG4gICAgfVxuICAgIGdldCB3ZWVrc0NvdW50KCk6IG51bWJlciB7XG4gICAgICAgIHJldHVybiB0aGlzLl9kYXlzU291cmNlLmdldFZhbHVlKCkubGVuZ3RoIC8gNztcbiAgICB9XG4gICAgZ2V0IGRheXMoKTogQXJyYXk8RGF5PiB7XG4gICAgICAgIHJldHVybiB0aGlzLl9kYXlzU291cmNlLmdldFZhbHVlKCk7XG4gICAgfVxuICAgIGdldCBNb250aHMoKTogc3RyaW5nW10ge1xuICAgICAgICByZXR1cm4gdGhpcy5fZGF0ZVNlcnZpY2UubW9udGhOYW1lcygpO1xuICAgIH1cblxuICAgIGxvYWRNb250aEJ5RGF0ZSh2YWx1ZTogRGF0ZSk6IHZvaWQge1xuICAgICAgICBjb25zdCBtb250aEluZm8gPSB0aGlzLl9kYXRlU2VydmljZS5nZXREYXRlSW5mbyh2YWx1ZSk7XG4gICAgICAgIHRoaXMubG9hZE1vbnRoKG1vbnRoSW5mbyk7XG4gICAgfVxuICAgIHNldERhdGVTZXJ2aWNlKHByaW1hcnlDYWxlbmRhclR5cGU6IHN0cmluZyk6IHZvaWQge1xuICAgICAgICBsZXQgZGF0ZVNlcnZpY2UgPSB0aGlzLl9kYXRlU2VydmljZTtcbiAgICAgICAgc3dpdGNoIChwcmltYXJ5Q2FsZW5kYXJUeXBlKSB7XG4gICAgICAgICAgICBjYXNlICdQZXJzaWFuJzpcbiAgICAgICAgICAgICAgICBkYXRlU2VydmljZSA9IGdldERhdGVTZXJ2aWNlKHRoaXMuX2xvZ1NlcnZpY2UsICdmYS1JUicpO1xuICAgICAgICAgICAgICAgIGJyZWFrO1xuICAgICAgICAgICAgY2FzZSAnR3JlZ29yaWFuJzpcbiAgICAgICAgICAgICAgICBkYXRlU2VydmljZSA9IGdldERhdGVTZXJ2aWNlKHRoaXMuX2xvZ1NlcnZpY2UsICdlbi1VUycpO1xuICAgICAgICAgICAgICAgIGJyZWFrO1xuICAgICAgICAgICAgY2FzZSAnQXJhYmljJzpcbiAgICAgICAgICAgICAgICBkYXRlU2VydmljZSA9IGdldERhdGVTZXJ2aWNlKHRoaXMuX2xvZ1NlcnZpY2UsICdhci1BRScpO1xuICAgICAgICAgICAgICAgIGJyZWFrO1xuICAgICAgICB9XG4gICAgICAgIHRoaXMuX2RhdGVTZXJ2aWNlID0gZGF0ZVNlcnZpY2U7XG4gICAgICAgIHRoaXMuX2luaXRpbGl6ZSgpO1xuICAgIH1cbiAgICBzZXREYXlzKGRheXM6IERheVtdKTogdm9pZCB7XG4gICAgICAgIHRoaXMuX2RheXNTb3VyY2UubmV4dChkYXlzKTtcbiAgICB9XG5cbiAgICBwcmVwYXJlV2Vlayhtb250aEluZm86IENhbGVuZGFyTW9udGhJbmZvLCBzdGFydEluZGV4OiBudW1iZXIsIGRheXNDb3VudDogbnVtYmVyKTogdm9pZCB7XG4gICAgICAgIGNvbnN0IHN0YXJ0RGF5cyA9IG1vbnRoSW5mby5kYXlzLnNsaWNlKHN0YXJ0SW5kZXgsIHN0YXJ0SW5kZXggKyBkYXlzQ291bnQpO1xuICAgICAgICB0aGlzLl93ZWVrTW9kZURheXNTb3VyY2UubmV4dChzdGFydERheXMpO1xuICAgIH1cbiAgICBzZXRTZWxlY3RlZChzZWxlY3RlZDogQ2FsZW5kYXJNb250aEluZm8pOiB2b2lkIHtcbiAgICAgICAgdGhpcy5fc2VsZWN0ZWRTb3VyY2UubmV4dChzZWxlY3RlZCk7XG4gICAgfVxuXG4gICAgd2Vla0RheXNNaW4oKTogQXJyYXk8c3RyaW5nPiB7XG4gICAgICAgIHJldHVybiB0aGlzLl9kYXRlU2VydmljZS53ZWVrZGF5c01pbigpO1xuICAgIH1cbiAgICB3ZWVrRGF5cygpOiBBcnJheTxzdHJpbmc+IHtcbiAgICAgICAgcmV0dXJuIHRoaXMuX2RhdGVTZXJ2aWNlLndlZWtEYXlzKCk7XG4gICAgfVxuICAgIGdvdG9Ub2RheSgpOiB2b2lkIHtcbiAgICAgICAgdGhpcy5sb2FkTW9udGgodGhpcy5fdG9kYXlNb250aEluZm8uZGF0ZUluZm8sIDApO1xuICAgICAgICB0aGlzLnNldE1vZGUoXG4gICAgICAgICAgICB0aGlzLl9tb2RlU291cmNlLmdldFZhbHVlKCksXG4gICAgICAgICAgICB0aGlzLl90b2RheU1vbnRoSW5mby5kYXlzLmZpbmQoKGMpID0+IGMuaXNUb2RheSksXG4gICAgICAgICAgICB0aGlzLl93ZWVrTW9kZURheXNTb3VyY2UuZ2V0VmFsdWUoKS5sZW5ndGhcbiAgICAgICAgKTtcbiAgICB9XG4gICAgcmFuZ2VTZWxlY3Qoc3RhcnREYXk6IERheSwgZW5kRGF5OiBEYXksIG1vbnRoSW5mbzogQ2FsZW5kYXJNb250aEluZm8pOiB2b2lkIHtcbiAgICAgICAgY29uc3QgZGlmZiA9IGVuZERheS5pbmRleCAtIHN0YXJ0RGF5LmluZGV4O1xuICAgICAgICBpZiAoZGlmZiA8IDcpIHtcbiAgICAgICAgICAgIHRoaXMuc2V0TW9kZSgnd2VlaycsIHN0YXJ0RGF5LCBkaWZmICsgMSk7XG4gICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgICBjb25zdCBkaWZmMiA9IGRpZmYgKyAxO1xuICAgICAgICAgICAgdGhpcy5zZXRNb2RlKCdjdXN0b21Nb250aCcpO1xuICAgICAgICAgICAgdGhpcy5zZXRNb250aChtb250aEluZm8sIHN0YXJ0RGF5LCBkaWZmMik7XG4gICAgICAgIH1cbiAgICB9XG4gICAgc2V0TW9udGgobW9udGhJbmZvOiBDYWxlbmRhck1vbnRoSW5mbywgc3RhcnREYXk/OiBEYXksIGRpZmY6IG51bWJlciA9IDApOiB2b2lkIHtcbiAgICAgICAgaWYgKHN0YXJ0RGF5ICYmIGRpZmYgPiAwKSB7XG4gICAgICAgICAgICBjb25zdCBzdGFydEluZGV4ID0gc3RhcnREYXkuaW5kZXg7XG4gICAgICAgICAgICB0aGlzLmxvYWRDYWxlbmRhck1vbnRoSW5mbyhtb250aEluZm8pO1xuICAgICAgICAgICAgdGhpcy5zZXREYXlzKG1vbnRoSW5mby5kYXlzLnNsaWNlKHN0YXJ0SW5kZXgsIHN0YXJ0SW5kZXggKyBkaWZmKSk7XG4gICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgICB0aGlzLmxvYWRDYWxlbmRhck1vbnRoSW5mbyhtb250aEluZm8pO1xuICAgICAgICAgICAgdGhpcy5zZXREYXlzKG1vbnRoSW5mby5kYXlzKTtcbiAgICAgICAgICAgIC8vIGVsc2UgaWYgKG1vZGUgPT09ICdtb250aCcpIHtcbiAgICAgICAgICAgIC8vICAgc3RhcnREYXkgPSBkYXlzTG9hZGVkWzBdO1xuICAgICAgICAgICAgLy8gaWYgKHN0YXJ0RGF5KSB0aGlzLmxvYWRNb250aChzdGFydERheS5tb250aERhdGVJbmZvKTtcbiAgICAgICAgICAgIC8vIH1cbiAgICAgICAgfVxuICAgIH1cbiAgICBzZXRNb2RlKG1vZGU6IENhbGVuZGFyTW9kZSwgc3RhcnREYXk/OiBEYXksIGRheXNDb3VudD86IG51bWJlcik6IHZvaWQge1xuICAgICAgICBjb25zdCBvbGRNb2RlID0gdGhpcy5fbW9kZVNvdXJjZS5nZXRWYWx1ZSgpO1xuICAgICAgICB0aGlzLl9tb2RlU291cmNlLm5leHQobW9kZSk7XG4gICAgICAgIGlmIChtb2RlID09PSAnbW9udGgnKSB7XG4gICAgICAgICAgICB0aGlzLnNldE1vbnRoKHRoaXMuc2VsZWN0ZWRNb250aCwgc3RhcnREYXkpO1xuICAgICAgICAgICAgcmV0dXJuO1xuICAgICAgICB9IGVsc2UgaWYgKG1vZGUgPT09ICdjdXN0b21Nb250aCcpIHtcbiAgICAgICAgICAgIHRoaXMuc2V0TW9udGgodGhpcy5zZWxlY3RlZE1vbnRoLCBzdGFydERheSwgdGhpcy5kYXlzLmxlbmd0aCk7XG4gICAgICAgICAgICByZXR1cm47XG4gICAgICAgIH1cbiAgICAgICAgY29uc3QgZGF5c0xvYWRlZCA9IHRoaXMuX3dlZWtNb2RlRGF5c1NvdXJjZS5nZXRWYWx1ZSgpO1xuICAgICAgICBkYXlzQ291bnQgPSBkYXlzQ291bnQgPz8gKG1vZGUgPT09ICd3ZWVrJyA/IDcgOiAxKTtcbiAgICAgICAgbGV0IHN0YXJ0SW5kZXggPSAwO1xuICAgICAgICBpZiAoIXN0YXJ0RGF5KSB7XG4gICAgICAgICAgICBpZiAobW9kZSA9PT0gJ3dlZWsnIHx8IG1vZGUgPT09ICdkYXknKSB7XG4gICAgICAgICAgICAgICAgaWYgKG9sZE1vZGUgPT09ICdtb250aCcpIHtcbiAgICAgICAgICAgICAgICAgICAgc3RhcnREYXkgPSB0aGlzLnNlbGVjdGVkTW9udGguZGF5cy5maW5kKChjKSA9PiBjLmlzU3RhcnRPZk1vbnRoKTtcbiAgICAgICAgICAgICAgICB9IGVsc2UgaWYgKG9sZE1vZGUgPT09ICdjdXN0b21Nb250aCcpIHtcbiAgICAgICAgICAgICAgICAgICAgc3RhcnREYXkgPSB0aGlzLmRheXNbMF07XG4gICAgICAgICAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICAgICAgICAgICAgc3RhcnREYXkgPSBkYXlzTG9hZGVkWzBdO1xuICAgICAgICAgICAgICAgIH1cbiAgICAgICAgICAgIH1cbiAgICAgICAgfVxuXG4gICAgICAgIGlmIChzdGFydERheSkge1xuICAgICAgICAgICAgaWYgKFxuICAgICAgICAgICAgICAgIHN0YXJ0RGF5Lm1vbnRoRGF0ZUluZm8ubW9udGggIT09IHRoaXMuc2VsZWN0ZWRNb250aC5kYXRlSW5mby5tb250aCB8fFxuICAgICAgICAgICAgICAgIHN0YXJ0RGF5Lm1vbnRoRGF0ZUluZm8ueWVhciAhPT0gdGhpcy5zZWxlY3RlZE1vbnRoLmRhdGVJbmZvLnllYXJcbiAgICAgICAgICAgICkge1xuICAgICAgICAgICAgICAgIHRoaXMubG9hZE1vbnRoKHN0YXJ0RGF5Lm1vbnRoRGF0ZUluZm8pO1xuICAgICAgICAgICAgfVxuICAgICAgICAgICAgc3RhcnRJbmRleCA9IHRoaXMuc2VsZWN0ZWRNb250aC5kYXlzLmZpbmRJbmRleChcbiAgICAgICAgICAgICAgICAoYykgPT4gYy5tb250aCA9PT0gc3RhcnREYXk/Lm1vbnRoICYmIGMuZGF5SW5Nb250aCA9PT0gc3RhcnREYXkuZGF5SW5Nb250aFxuICAgICAgICAgICAgKTtcbiAgICAgICAgfVxuICAgICAgICB0aGlzLnByZXBhcmVXZWVrKHRoaXMuX3NlbGVjdGVkU291cmNlLmdldFZhbHVlKCksIHN0YXJ0SW5kZXgsIGRheXNDb3VudCk7XG4gICAgfVxuXG4gICAgZmlsbFdlZWsod2VlazogeyBba2V5OiBzdHJpbmddOiBhbnlbXSB9LCBkYXlzQ291bnQgPSA3KTogdm9pZCB7XG4gICAgICAgIGNvbnN0IGFycjogYW55W10gPSBPYmplY3QudmFsdWVzKHdlZWspO1xuICAgICAgICBhcnIuZm9yRWFjaChcbiAgICAgICAgICAgIChcbiAgICAgICAgICAgICAgICBlbGVtZW50OiB7XG4gICAgICAgICAgICAgICAgICAgIGNvbFNwYW46IG51bWJlcjtcbiAgICAgICAgICAgICAgICAgICAgZXZlbnQ/OiBNZXRhb2JqZWN0RGF0YU1vZGVsO1xuICAgICAgICAgICAgICAgICAgICBvcGFjaXR5PzogbnVtYmVyO1xuICAgICAgICAgICAgICAgICAgICBzdGFydD86IERhdGU7XG4gICAgICAgICAgICAgICAgICAgIGVuZD86IERhdGU7XG4gICAgICAgICAgICAgICAgfVtdXG4gICAgICAgICAgICApID0+IHtcbiAgICAgICAgICAgICAgICBmb3IgKGNvbnN0IGVsIG9mIGVsZW1lbnQpIHtcbiAgICAgICAgICAgICAgICAgICAgY29uc3QgdmFsdWVEYXRlID0gZWwuZW5kO1xuICAgICAgICAgICAgICAgICAgICBpZiAodmFsdWVEYXRlICYmIHRoaXMuX2RhdGVTZXJ2aWNlLmdldE1vbWVudCh2YWx1ZURhdGUpLmlzQmVmb3JlKHRoaXMudG9kYXksICdkYXknKSkge1xuICAgICAgICAgICAgICAgICAgICAgICAgZWwub3BhY2l0eSA9IDAuNTtcbiAgICAgICAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgICAgIH1cbiAgICAgICAgICAgICAgICBmb3IgKGxldCBpID0gZWxlbWVudC5sZW5ndGg7IGkgPCBkYXlzQ291bnQ7IGkrKykge1xuICAgICAgICAgICAgICAgICAgICBlbGVtZW50LnB1c2goeyBjb2xTcGFuOiAxIH0pO1xuICAgICAgICAgICAgICAgIH1cbiAgICAgICAgICAgIH1cbiAgICAgICAgKTtcbiAgICB9XG4gICAgc29ydFRhc2tzKGNhbGVuZGFyVGFza3M6IE1ldGFvYmplY3REYXRhTW9kZWxbXSwgY2FsZW5kYXJGaWVsZHM6IENhbGVuZGFyRmllbGRzKTogQXJyYXk8TWV0YW9iamVjdERhdGFNb2RlbD4ge1xuICAgICAgICBjb25zdCBzdGFydERhdGVGaWVsZCA9IGNhbGVuZGFyRmllbGRzLlN0YXJ0RGF0ZTtcbiAgICAgICAgY29uc3QgZW5kRGF0ZUZpZWxkID0gY2FsZW5kYXJGaWVsZHMuRW5kRGF0ZTtcbiAgICAgICAgY2FsZW5kYXJUYXNrcy5zb3J0KCh0YXNrQSwgdGFza0IpID0+ICh0YXNrQVtzdGFydERhdGVGaWVsZF0gPCB0YXNrQltzdGFydERhdGVGaWVsZF0gPyAtMSA6IDEpKTtcbiAgICAgICAgY2FsZW5kYXJUYXNrcy5zb3J0KCh0YXNrQSwgdGFza0IpID0+IHtcbiAgICAgICAgICAgIGNvbnN0IHRhc2tBU3RhcnREYXRlID0gdGFza0Fbc3RhcnREYXRlRmllbGRdO1xuICAgICAgICAgICAgY29uc3QgdGFza0JTdGFydERhdGUgPSB0YXNrQltzdGFydERhdGVGaWVsZF07XG5cbiAgICAgICAgICAgIGNvbnN0IG1vbWVudFRhc2tBU3RhcnREYXRlID0gdGhpcy5fZGF0ZVNlcnZpY2UuZ2V0TW9tZW50KHRhc2tBU3RhcnREYXRlKTtcbiAgICAgICAgICAgIGNvbnN0IG1vbWVudFRhc2tCU3RhcnREYXRlID0gdGhpcy5fZGF0ZVNlcnZpY2UuZ2V0TW9tZW50KHRhc2tCU3RhcnREYXRlKTtcbiAgICAgICAgICAgIGlmIChtb21lbnRUYXNrQVN0YXJ0RGF0ZS5pc1NhbWUobW9tZW50VGFza0JTdGFydERhdGUsICdkYXknKSkge1xuICAgICAgICAgICAgICAgIGNvbnN0IHRhc2tBRW5kRGF0ZSA9IHRhc2tBW2VuZERhdGVGaWVsZF07XG4gICAgICAgICAgICAgICAgY29uc3QgdGFza0JFbmREYXRlID0gdGFza0JbZW5kRGF0ZUZpZWxkXTtcbiAgICAgICAgICAgICAgICBjb25zdCBkYXlBRGlmZiA9IG1vbWVudFRhc2tBU3RhcnREYXRlLmRpZmYodGFza0FFbmREYXRlLCAnZGF5Jyk7XG4gICAgICAgICAgICAgICAgY29uc3QgZGF5QkRpZmYgPSBtb21lbnRUYXNrQlN0YXJ0RGF0ZS5kaWZmKHRhc2tCRW5kRGF0ZSwgJ2RheScpO1xuICAgICAgICAgICAgICAgIGlmIChkYXlBRGlmZiA+IGRheUJEaWZmKSB7XG4gICAgICAgICAgICAgICAgICAgIHJldHVybiAxO1xuICAgICAgICAgICAgICAgIH1cbiAgICAgICAgICAgICAgICBpZiAoZGF5QURpZmYgPCBkYXlCRGlmZikge1xuICAgICAgICAgICAgICAgICAgICByZXR1cm4gLTE7XG4gICAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgICAgIHJldHVybiAwO1xuICAgICAgICAgICAgfVxuICAgICAgICAgICAge1xuICAgICAgICAgICAgICAgIHJldHVybiAwO1xuICAgICAgICAgICAgfVxuICAgICAgICB9KTtcbiAgICAgICAgcmV0dXJuIGNhbGVuZGFyVGFza3M7XG4gICAgfVxuICAgIGNyZWF0ZVdlZWtUYXNrKFxuICAgICAgICBkYXlzOiBEYXlbXSxcbiAgICAgICAgbW9EYXRhTGlzdDogTWV0YW9iamVjdERhdGFNb2RlbFtdLFxuICAgICAgICBjYWxlbmRhckZpZWxkczogQ2FsZW5kYXJGaWVsZHMsXG4gICAgICAgIHRhc2tIZWlnaHQ6IG51bWJlclxuICAgICk6IHsgW2tleTogc3RyaW5nXTogYW55W10gfSB7XG4gICAgICAgIGNvbnN0IHdlZWs6IHsgW2tleTogc3RyaW5nXTogYW55W10gfSA9IHt9O1xuICAgICAgICBkYXlzLmZvckVhY2goKGRheSwgZGF5SW5kZXgpID0+IHtcbiAgICAgICAgICAgIGxldCBjb3VudGVyID0gMDtcbiAgICAgICAgICAgIG1vRGF0YUxpc3QuZm9yRWFjaCgoZXZlbnQpID0+IHtcbiAgICAgICAgICAgICAgICBjb25zdCBjb3VudGVySW5kZXggPSBjb3VudGVyLnRvU3RyaW5nKCk7XG4gICAgICAgICAgICAgICAgaWYgKCF3ZWVrW2NvdW50ZXJJbmRleF0pIHtcbiAgICAgICAgICAgICAgICAgICAgd2Vla1tjb3VudGVySW5kZXhdID0gW107XG4gICAgICAgICAgICAgICAgfSBlbHNlIGlmICh3ZWVrW2NvdW50ZXJJbmRleF0ubGVuZ3RoID4gZGF5SW5kZXgpIHtcbiAgICAgICAgICAgICAgICAgICAgY291bnRlcisrO1xuICAgICAgICAgICAgICAgICAgICByZXR1cm47XG4gICAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgICAgIGZvciAobGV0IG0gPSB3ZWVrW2NvdW50ZXJJbmRleF0ubGVuZ3RoOyBtIDwgZGF5SW5kZXg7IG0rKykge1xuICAgICAgICAgICAgICAgICAgICB3ZWVrW2NvdW50ZXJJbmRleF0ucHVzaCh7IGNvbFNwYW46IDEgfSk7XG4gICAgICAgICAgICAgICAgfVxuXG4gICAgICAgICAgICAgICAgLy8gYWdhciB3ZWVrW2NvdW50ZXJdIGNvbHNwYW4gamEgZGFzaHQgYW5qYW0gZ2FyZGFkLlxuICAgICAgICAgICAgICAgIGNvbnN0IG5ld1Rhc2s6IGFueSA9IHsgY29sU3BhbjogMSB9O1xuICAgICAgICAgICAgICAgIGNvbnN0IG1vbWVudERheSA9IHRoaXMuX2RhdGVTZXJ2aWNlLmdldE1vbWVudChkYXkuZGF0ZSk7XG4gICAgICAgICAgICAgICAgY29uc3QgbW9tZW50T2ZTdGFydEV2ZW50ID0gdGhpcy5fZGF0ZVNlcnZpY2UuZ2V0TW9tZW50KGV2ZW50W2NhbGVuZGFyRmllbGRzLlN0YXJ0RGF0ZV0pO1xuICAgICAgICAgICAgICAgIGNvbnN0IGVuZERhdGUgPSBldmVudFtjYWxlbmRhckZpZWxkcy5FbmREYXRlXTtcbiAgICAgICAgICAgICAgICBjb25zdCBtb21lbnRPZkVuZEV2ZW50ID0gdGhpcy5fZGF0ZVNlcnZpY2UuZ2V0TW9tZW50KGVuZERhdGUpO1xuICAgICAgICAgICAgICAgIGNvbnN0IGRpZmYgPSBNYXRoLmFicyhtb21lbnRPZlN0YXJ0RXZlbnQuZGlmZihtb21lbnRPZkVuZEV2ZW50LCAnZGF5JykpO1xuICAgICAgICAgICAgICAgIGlmIChkaWZmID09PSAwICYmIG1vbWVudERheS5pc1NhbWUobW9tZW50T2ZTdGFydEV2ZW50LCAnZGF5JykpIHtcbiAgICAgICAgICAgICAgICAgICAgbmV3VGFzay5jb2xTcGFuID0gMTtcbiAgICAgICAgICAgICAgICAgICAgY29uc3Qgc3RhcnRNaW51dGVUb3AgPSAoZXZlbnQuJFN0YXJ0RGF0ZUluZm8ubWludXRlcyAqIHRhc2tIZWlnaHQpIC8gNjA7XG4gICAgICAgICAgICAgICAgICAgIG5ld1Rhc2sudG9wID0gZXZlbnQuJFN0YXJ0RGF0ZUluZm8uaG91ciAqIHRhc2tIZWlnaHQgKyBzdGFydE1pbnV0ZVRvcDtcbiAgICAgICAgICAgICAgICAgICAgY29uc3QgZW5kTWludXRlVG9wID0gKGV2ZW50LiRFbmREYXRlSW5mby5taW51dGVzICogdGFza0hlaWdodCkgLyA2MDtcbiAgICAgICAgICAgICAgICAgICAgbmV3VGFzay5oZWlnaHQgPVxuICAgICAgICAgICAgICAgICAgICAgICAgKGV2ZW50LiRFbmREYXRlSW5mby5ob3VyIC0gZXZlbnQuJFN0YXJ0RGF0ZUluZm8uaG91cikgKiB0YXNrSGVpZ2h0ICtcbiAgICAgICAgICAgICAgICAgICAgICAgIGVuZE1pbnV0ZVRvcCAtXG4gICAgICAgICAgICAgICAgICAgICAgICBzdGFydE1pbnV0ZVRvcDtcbiAgICAgICAgICAgICAgICAgICAgbmV3VGFzay5ldmVudCA9IGV2ZW50O1xuICAgICAgICAgICAgICAgICAgICBuZXdUYXNrLmVuZGRheUlzSW5Sb3cgPSB0cnVlO1xuICAgICAgICAgICAgICAgICAgICBuZXdUYXNrLnN0YXJ0ZGF5SXNJblJvdyA9IHRydWU7XG4gICAgICAgICAgICAgICAgICAgIG5ld1Rhc2suaXNTdGFydERheSA9IHRydWU7XG4gICAgICAgICAgICAgICAgICAgIG5ld1Rhc2suaXNFbmREYXkgPSB0cnVlO1xuICAgICAgICAgICAgICAgICAgICBuZXdUYXNrLnN0YXJ0ID0gbmV3VGFzay5lbmQgPSBkYXkuZGF0ZTtcbiAgICAgICAgICAgICAgICAgICAgbmV3VGFzay5pbkRheSA9XG4gICAgICAgICAgICAgICAgICAgICAgICBldmVudC4kU3RhcnREYXRlSW5mby5ob3VyID4gMCB8fFxuICAgICAgICAgICAgICAgICAgICAgICAgZXZlbnQuJFN0YXJ0RGF0ZUluZm8ubWludXRlcyA+IDAgfHxcbiAgICAgICAgICAgICAgICAgICAgICAgIGV2ZW50LiRFbmREYXRlSW5mby5ob3VyID4gMCB8fFxuICAgICAgICAgICAgICAgICAgICAgICAgZXZlbnQuJEVuZERhdGVJbmZvLm1pbnV0ZXMgPiAwO1xuICAgICAgICAgICAgICAgICAgICB3ZWVrW2NvdW50ZXJJbmRleF0uc3BsaWNlKGRheUluZGV4LCAwLCBuZXdUYXNrKTtcbiAgICAgICAgICAgICAgICAgICAgZGF5LnRhc2tzLnB1c2gobmV3VGFzayk7XG4gICAgICAgICAgICAgICAgICAgIGNvdW50ZXIrKztcbiAgICAgICAgICAgICAgICB9IGVsc2Uge1xuICAgICAgICAgICAgICAgICAgICBsZXQgZGlmZkZyb21FbmRPZkV2ZW50ID0gMDtcbiAgICAgICAgICAgICAgICAgICAgbGV0IGRpZmZGcm9tU3RhcnRPZkV2ZW50ID0gMDtcbiAgICAgICAgICAgICAgICAgICAgY29uc3QgaXNCZXR3ZWVuID1cbiAgICAgICAgICAgICAgICAgICAgICAgIG1vbWVudERheS5pc1NhbWVPckFmdGVyKG1vbWVudE9mU3RhcnRFdmVudCwgJ2RheScpICYmXG4gICAgICAgICAgICAgICAgICAgICAgICBtb21lbnREYXkuaXNTYW1lT3JCZWZvcmUobW9tZW50T2ZFbmRFdmVudCwgJ2RheScpO1xuICAgICAgICAgICAgICAgICAgICBpZiAoaXNCZXR3ZWVuKSB7XG4gICAgICAgICAgICAgICAgICAgICAgICBkaWZmRnJvbUVuZE9mRXZlbnQgPSAwO1xuICAgICAgICAgICAgICAgICAgICAgICAgaWYgKGRpZmYgPT09IDApIHtcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICBuZXdUYXNrLmNvbFNwYW4gPSAxO1xuICAgICAgICAgICAgICAgICAgICAgICAgICAgIG5ld1Rhc2suZXZlbnQgPSBldmVudDtcbiAgICAgICAgICAgICAgICAgICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgZGlmZkZyb21FbmRPZkV2ZW50ID0gTWF0aC5hYnMobW9tZW50RGF5LmRpZmYobW9tZW50T2ZFbmRFdmVudCwgJ2RheScpKTtcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICBkaWZmRnJvbVN0YXJ0T2ZFdmVudCA9IE1hdGguYWJzKG1vbWVudERheS5kaWZmKG1vbWVudE9mU3RhcnRFdmVudCwgJ2RheScpKTtcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICBjb25zdCBpc1N0YXJ0RGF5ID0gbW9tZW50RGF5LmlzU2FtZShtb21lbnRPZlN0YXJ0RXZlbnQsICdkYXknKTtcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICBjb25zdCBpc0VuZERheSA9IG1vbWVudERheS5pc1NhbWUobW9tZW50T2ZFbmRFdmVudCwgJ2RheScpO1xuICAgICAgICAgICAgICAgICAgICAgICAgICAgIGxldCBjb2xTcGFuID0gZGlmZjtcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICBpZiAoIWlzU3RhcnREYXkpIHtcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgY29sU3BhbiA9IGRpZmZGcm9tRW5kT2ZFdmVudDtcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICB9IGVsc2Uge1xuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBjb2xTcGFuID0gZGlmZiArIDE7IC8vIGluY2x1ZGUgdGhlIHN0YXJ0XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgICAgICAgICAgICAgICAgIG5ld1Rhc2suY29sU3BhbiA9IHRoaXMuX2NhbGNDb2xTcGFuKGNvbFNwYW4sIGRheUluZGV4LCBkYXlzLmxlbmd0aCk7XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgbmV3VGFzay5pc1N0YXJ0RGF5ID0gaXNTdGFydERheTtcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICBuZXdUYXNrLmlzRW5kRGF5ID0gaXNFbmREYXk7XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgbmV3VGFzay5pc01pZGRsZSA9ICFpc1N0YXJ0RGF5ICYmICFpc0VuZERheTtcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICBuZXdUYXNrLmV2ZW50ID0gZXZlbnQ7XG4gICAgICAgICAgICAgICAgICAgICAgICB9XG4gICAgICAgICAgICAgICAgICAgICAgICBpZiAobmV3VGFzay5pc1N0YXJ0RGF5IHx8IGRheUluZGV4ID09PSAwKSB7XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgZm9yIChsZXQgc3BhbiA9IDA7IHNwYW4gPCBuZXdUYXNrLmNvbFNwYW47IHNwYW4rKykge1xuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBjb25zdCB0YXNrSW5kZXggPSBkYXlJbmRleCArIHNwYW47XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGlmICh0YXNrSW5kZXggPiBkYXlzLmxlbmd0aCAtIDEpIHtcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGJyZWFrO1xuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB9XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNvbnN0IHRhc2tEYXlJbmZvID0gZGF5c1t0YXNrSW5kZXhdO1xuXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNvbnN0IHRhc2tJbmZvID0ge1xuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgLi4ubmV3VGFzayxcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHN0YXJ0OiB0YXNrRGF5SW5mby5kYXRlLFxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZW5kOiBlbmREYXRlLFxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgaXNTdGFydERheTogc3BhbiA9PT0gMCA/IG5ld1Rhc2suaXNTdGFydERheSA6IG51bGwsXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBzdGFydFRhc2s6IGV2ZW50LiRTdGFydERheUluZm8sXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBlbmRUYXNrOiBldmVudC4kRW5kRGF5SW5mbyxcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHN0YXJ0ZGF5SXNJblJvdzogZGF5SW5kZXggLSBkaWZmRnJvbVN0YXJ0T2ZFdmVudCA+IC0xLFxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZW5kZGF5SXNJblJvdzogISFkYXlzW2RpZmZGcm9tRW5kT2ZFdmVudF0sXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBpc01pZGRsZTpcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBzcGFuID09PSAwXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID8gbmV3VGFzay5pc01pZGRsZVxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICA6IHRoaXMuX2RhdGVTZXJ2aWNlXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIC5nZXRNb21lbnQodGFza0RheUluZm8uZGF0ZSlcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgLmlzQmV0d2Vlbihtb21lbnRPZlN0YXJ0RXZlbnQsIG1vbWVudE9mRW5kRXZlbnQsICdkYXknKSxcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGlzRW5kRGF5OlxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHNwYW4gPT09IDBcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPyBuZXdUYXNrLmlzRW5kRGF5XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIDogTWF0aC5hYnMoXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHRoaXMuX2RhdGVTZXJ2aWNlXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAuZ2V0TW9tZW50KHRhc2tEYXlJbmZvLmRhdGUpXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAuZGlmZihtb21lbnRPZkVuZEV2ZW50LCAnZGF5JylcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICApIDw9IDEsXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBjb2xTcGFuOiBzcGFuID09PSAwID8gbmV3VGFzay5jb2xTcGFuIDogbnVsbFxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB9O1xuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB3ZWVrW2NvdW50ZXIudG9TdHJpbmcoKV0uc3BsaWNlKHRhc2tJbmRleCwgMCwgdGFza0luZm8pO1xuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB0YXNrRGF5SW5mby50YXNrcy5wdXNoKHRhc2tJbmZvKTtcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICB9XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgY291bnRlcisrO1xuICAgICAgICAgICAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgICAgICAgICB9XG4gICAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgfSk7XG4gICAgICAgIH0pO1xuICAgICAgICByZXR1cm4gd2VlaztcbiAgICB9XG5cbiAgICBsb2FkTmV4dCgpOiB2b2lkIHtcbiAgICAgICAgY29uc3QgbW9kZSA9IHRoaXMuX21vZGVTb3VyY2UuZ2V0VmFsdWUoKTtcbiAgICAgICAgY29uc3Qgc2VsZWN0ZWRNb250aCA9IHRoaXMuX3NlbGVjdGVkU291cmNlLmdldFZhbHVlKCk7XG4gICAgICAgIHN3aXRjaCAobW9kZSkge1xuICAgICAgICAgICAgY2FzZSAnY3VzdG9tTW9udGgnOlxuICAgICAgICAgICAgY2FzZSAnbW9udGgnOlxuICAgICAgICAgICAgICAgIHtcbiAgICAgICAgICAgICAgICAgICAgY29uc3QgZGF5cyA9IHRoaXMuZGF5cztcbiAgICAgICAgICAgICAgICAgICAgY29uc3QgbmV3RGF5cyA9IHRoaXMuX2xvYWROZXh0TW9udGgoZGF5cywgc2VsZWN0ZWRNb250aCk7XG4gICAgICAgICAgICAgICAgICAgIHRoaXMuc2V0RGF5cyhuZXdEYXlzKTtcbiAgICAgICAgICAgICAgICB9XG5cbiAgICAgICAgICAgICAgICBicmVhaztcbiAgICAgICAgICAgIGNhc2UgJ3dlZWsnOlxuICAgICAgICAgICAgY2FzZSAnZGF5JzpcbiAgICAgICAgICAgICAgICB7XG4gICAgICAgICAgICAgICAgICAgIGNvbnN0IGRheXMgPSB0aGlzLl93ZWVrTW9kZURheXNTb3VyY2UuZ2V0VmFsdWUoKTtcbiAgICAgICAgICAgICAgICAgICAgY29uc3QgbmV3RGF5cyA9IHRoaXMuX2xvYWROZXh0V2VlayhkYXlzLCBzZWxlY3RlZE1vbnRoKTtcbiAgICAgICAgICAgICAgICAgICAgdGhpcy5fd2Vla01vZGVEYXlzU291cmNlLm5leHQobmV3RGF5cyk7XG4gICAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgICAgIGJyZWFrO1xuICAgICAgICAgICAgICAgIGJyZWFrO1xuICAgICAgICAgICAgY2FzZSAnbGlzdCc6XG4gICAgICAgICAgICAgICAgYnJlYWs7XG4gICAgICAgIH1cbiAgICB9XG4gICAgZ290b01vbnRoKG1vbnRoTmFtZTogc3RyaW5nLCBtb250aEluZGV4OiBudW1iZXIpOiB2b2lkIHtcbiAgICAgICAgY29uc3Qgc2VsZWN0ZWRNb250aCA9IHRoaXMuc2VsZWN0ZWRNb250aC5kYXRlSW5mby5tb250aDtcbiAgICAgICAgY29uc3QgdmFsID0gbW9udGhJbmRleCAtIHNlbGVjdGVkTW9udGg7XG4gICAgICAgIHRoaXMubG9hZE1vbnRoKHRoaXMuc2VsZWN0ZWRNb250aC5kYXRlSW5mbywgdmFsKTtcbiAgICB9XG4gICAgZ290b1llYXIoeWVhcjogbnVtYmVyKTogdm9pZCB7XG4gICAgICAgIGNvbnN0IHNlbGVjdGVkWWVhciA9IHRoaXMuc2VsZWN0ZWRNb250aC5kYXRlSW5mby55ZWFyO1xuICAgICAgICBjb25zdCB2YWwgPSB5ZWFyIC0gc2VsZWN0ZWRZZWFyO1xuICAgICAgICB0aGlzLmluY3JlYXNlWWVhcih2YWwpO1xuICAgIH1cbiAgICBsb2FkTmV4dFllYXIoKTogdm9pZCB7XG4gICAgICAgIHRoaXMuaW5jcmVhc2VZZWFyKDEpO1xuICAgIH1cbiAgICBsb2FkUHJldlllYXIoKTogdm9pZCB7XG4gICAgICAgIHRoaXMuaW5jcmVhc2VZZWFyKC0xKTtcbiAgICB9XG4gICAgaW5jcmVhc2VZZWFyKHZhbHVlOiBudW1iZXIpOiB2b2lkIHtcbiAgICAgICAgY29uc3Qgc2VsZWN0ZWRZZWFyID0gdGhpcy5zZWxlY3RlZE1vbnRoLmRhdGVJbmZvLm1vbWVudC50b0RhdGUoKTtcbiAgICAgICAgc2VsZWN0ZWRZZWFyLnNldEZ1bGxZZWFyKHNlbGVjdGVkWWVhci5nZXRGdWxsWWVhcigpICsgdmFsdWUpOyAvLyBuZXh0IHllYXJcbiAgICAgICAgdGhpcy5sb2FkTW9udGhCeURhdGUoc2VsZWN0ZWRZZWFyKTtcbiAgICB9XG4gICAgbG9hZFByZXZpb3VzKCk6IHZvaWQge1xuICAgICAgICBjb25zdCBtb2RlID0gdGhpcy5fbW9kZVNvdXJjZS5nZXRWYWx1ZSgpO1xuICAgICAgICBjb25zdCBzZWxlY3RlZE1vbnRoID0gdGhpcy5fc2VsZWN0ZWRTb3VyY2UuZ2V0VmFsdWUoKTtcbiAgICAgICAgc3dpdGNoIChtb2RlKSB7XG4gICAgICAgICAgICBjYXNlICdjdXN0b21Nb250aCc6XG4gICAgICAgICAgICBjYXNlICdtb250aCc6XG4gICAgICAgICAgICAgICAge1xuICAgICAgICAgICAgICAgICAgICBjb25zdCBkYXlzID0gdGhpcy5kYXlzO1xuICAgICAgICAgICAgICAgICAgICBjb25zdCBuZXdEYXlzID0gdGhpcy5fbG9hZFByZXZNb250aChkYXlzLCBzZWxlY3RlZE1vbnRoKTtcbiAgICAgICAgICAgICAgICAgICAgdGhpcy5zZXREYXlzKG5ld0RheXMpO1xuICAgICAgICAgICAgICAgIH1cbiAgICAgICAgICAgICAgICBicmVhaztcbiAgICAgICAgICAgIGNhc2UgJ3dlZWsnOlxuICAgICAgICAgICAgY2FzZSAnZGF5JzpcbiAgICAgICAgICAgICAgICB7XG4gICAgICAgICAgICAgICAgICAgIGNvbnN0IGRheXMgPSB0aGlzLl93ZWVrTW9kZURheXNTb3VyY2UuZ2V0VmFsdWUoKTtcbiAgICAgICAgICAgICAgICAgICAgY29uc3QgbmV3RGF5cyA9IHRoaXMuX2xvYWRQcmV2V2VlayhkYXlzLCBzZWxlY3RlZE1vbnRoKTtcbiAgICAgICAgICAgICAgICAgICAgdGhpcy5fd2Vla01vZGVEYXlzU291cmNlLm5leHQobmV3RGF5cyk7XG4gICAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgICAgIGJyZWFrO1xuICAgICAgICAgICAgICAgIGJyZWFrO1xuICAgICAgICAgICAgY2FzZSAnbGlzdCc6XG4gICAgICAgICAgICAgICAgYnJlYWs7XG4gICAgICAgIH1cbiAgICB9XG4gICAgbG9hZE1vbnRoKGRhdGVJbmZvOiBEYXRlSW5mbywgYWRkTW9udGg6IG51bWJlciA9IDApOiBDYWxlbmRhck1vbnRoSW5mbyB7XG4gICAgICAgIGxldCBsb2FkZWRNb250aERhdGUgPSB0aGlzLl9kYXRlU2VydmljZS5nZXRNb21lbnQoZGF0ZUluZm8uZGF0ZSk7XG4gICAgICAgIGlmIChhZGRNb250aCA+IDApIHtcbiAgICAgICAgICAgIGxvYWRlZE1vbnRoRGF0ZSA9IHRoaXMuX2RhdGVTZXJ2aWNlLmFkZE1vbnRoKGxvYWRlZE1vbnRoRGF0ZSwgYWRkTW9udGgpO1xuICAgICAgICB9XG4gICAgICAgIGlmIChhZGRNb250aCA8IDApIHtcbiAgICAgICAgICAgIGxvYWRlZE1vbnRoRGF0ZSA9IHRoaXMuX2RhdGVTZXJ2aWNlLnN1YnRyYWN0TW9udGgobG9hZGVkTW9udGhEYXRlLCBNYXRoLmFicyhhZGRNb250aCkpO1xuICAgICAgICB9XG4gICAgICAgIGxldCBsb2FkZWRDYWxlbmRhck1vbnRoSW5mbyA9IHRoaXMuY2FsZW5kYXJzLmZpbmQoKGMpID0+IGMuZGF0ZUluZm8ubW9tZW50LmlzU2FtZShsb2FkZWRNb250aERhdGUsICdkYXknKSk7XG4gICAgICAgIGlmICghbG9hZGVkQ2FsZW5kYXJNb250aEluZm8pIHtcbiAgICAgICAgICAgIGxvYWRlZENhbGVuZGFyTW9udGhJbmZvID0gdGhpcy5nZXRNb250aEluZm8obG9hZGVkTW9udGhEYXRlLCB0aGlzLnRvZGF5KTtcbiAgICAgICAgfVxuICAgICAgICB0aGlzLmxvYWRDYWxlbmRhck1vbnRoSW5mbyhsb2FkZWRDYWxlbmRhck1vbnRoSW5mbyk7XG4gICAgICAgIHJldHVybiBsb2FkZWRDYWxlbmRhck1vbnRoSW5mbztcbiAgICB9XG5cbiAgICBsb2FkQ2FsZW5kYXJNb250aEluZm8obG9hZGVkQ2FsZW5kYXJNb250aEluZm86IENhbGVuZGFyTW9udGhJbmZvKTogdm9pZCB7XG4gICAgICAgIGNvbnN0IGNhbGVuZGFyc0luZm8gPSBbLi4udGhpcy5jYWxlbmRhcnNdO1xuICAgICAgICBjb25zdCBleGlzdHMgPSBjYWxlbmRhcnNJbmZvLmZpbmQoXG4gICAgICAgICAgICAoYykgPT5cbiAgICAgICAgICAgICAgICBjLmRhdGVJbmZvLnllYXIgPT09IGxvYWRlZENhbGVuZGFyTW9udGhJbmZvLmRhdGVJbmZvLnllYXIgJiZcbiAgICAgICAgICAgICAgICBjLmRhdGVJbmZvLm1vbnRoID09PSBsb2FkZWRDYWxlbmRhck1vbnRoSW5mby5kYXRlSW5mby5tb250aFxuICAgICAgICApO1xuICAgICAgICBpZiAoIWV4aXN0cykge1xuICAgICAgICAgICAgY2FsZW5kYXJzSW5mby5wdXNoKGxvYWRlZENhbGVuZGFyTW9udGhJbmZvKTtcbiAgICAgICAgICAgIHRoaXMuX2NhbGVuZGFyc1NvdXJjZS5uZXh0KGNhbGVuZGFyc0luZm8pO1xuICAgICAgICB9XG4gICAgICAgIHRoaXMuc2V0U2VsZWN0ZWQobG9hZGVkQ2FsZW5kYXJNb250aEluZm8pO1xuICAgIH1cblxuICAgIGFkZENhbGVuZGFyTW9udGhJbmZvKGNhbGVuZGFyTW9udGhJbmZvOiBDYWxlbmRhck1vbnRoSW5mbyk6IHZvaWQge1xuICAgICAgICBjb25zdCBuZXdMaXN0ID0gdGhpcy5jYWxlbmRhcnM7XG4gICAgICAgIG5ld0xpc3QucHVzaChjYWxlbmRhck1vbnRoSW5mbyk7XG4gICAgICAgIHRoaXMuX2NhbGVuZGFyc1NvdXJjZS5uZXh0KG5ld0xpc3QpO1xuICAgIH1cblxuICAgIGdldE51bWJlcih2YWx1ZSk6IG51bWJlciB7XG4gICAgICAgIGNvbnN0IGZpbmQgPSBbJ9uwJywgJ9uxJywgJ9uyJywgJ9uzJywgJ9u0JywgJ9u1JywgJ9u2JywgJ9u3JywgJ9u4JywgJ9u5J107XG4gICAgICAgIGNvbnN0IHJlcGxhY2UgPSBbJzAnLCAnMScsICcyJywgJzMnLCAnNCcsICc1JywgJzYnLCAnNycsICc4JywgJzknXTtcbiAgICAgICAgbGV0IHJlcGxhY2VTdHJpbmcgPSB2YWx1ZTtcbiAgICAgICAgbGV0IHJlZ2V4O1xuICAgICAgICBmb3IgKGxldCBpID0gMDsgaSA8IGZpbmQubGVuZ3RoOyBpKyspIHtcbiAgICAgICAgICAgIHJlZ2V4ID0gbmV3IFJlZ0V4cChmaW5kW2ldLCAnZycpO1xuICAgICAgICAgICAgcmVwbGFjZVN0cmluZyA9IHJlcGxhY2VTdHJpbmcucmVwbGFjZShyZWdleCwgcmVwbGFjZVtpXSk7XG4gICAgICAgIH1cbiAgICAgICAgcmV0dXJuIE51bWJlcihyZXBsYWNlU3RyaW5nKTtcbiAgICB9XG4gICAgcHJpdmF0ZSBfaW5pdGlsaXplKCk6IHZvaWQge1xuICAgICAgICB0aGlzLnRvZGF5ID0gdGhpcy5fZGF0ZVNlcnZpY2UuZ2V0TW9tZW50KG5ldyBEYXRlKCkpO1xuICAgICAgICBjb25zdCB0b2RheU1vbnRoSW5mbyA9IHRoaXMuZ2V0Q2FsZW5kYXJNb250aEluZm8obmV3IERhdGUoKSk7XG4gICAgICAgIHRoaXMudG9kYXlEYXkgPSB0b2RheU1vbnRoSW5mby5kYXlzLmZpbmQoKGMpID0+IGMuaXNUb2RheSk7XG4gICAgICAgIHRoaXMuX3NlbGVjdGVkU291cmNlID0gbmV3IEJlaGF2aW9yU3ViamVjdDxDYWxlbmRhck1vbnRoSW5mbz4odG9kYXlNb250aEluZm8pO1xuICAgICAgICB0aGlzLnNlbGVjdGVkJCA9IHRoaXMuX3NlbGVjdGVkU291cmNlLmFzT2JzZXJ2YWJsZSgpO1xuICAgICAgICB0aGlzLl9jYWxlbmRhcnNTb3VyY2UubmV4dChbdG9kYXlNb250aEluZm9dKTtcbiAgICAgICAgdGhpcy5zZXREYXlzKHRvZGF5TW9udGhJbmZvLmRheXMpO1xuICAgICAgICB0aGlzLnNldFNlbGVjdGVkKHRvZGF5TW9udGhJbmZvKTtcbiAgICAgICAgdGhpcy5fY3VycmVudE1vbnRoSW5kZXhTb3VyY2UubmV4dCh0b2RheU1vbnRoSW5mby5kYXRlSW5mby5tb250aCk7XG4gICAgICAgIHRoaXMuX3RvZGF5TW9udGhJbmZvID0gdG9kYXlNb250aEluZm87XG4gICAgICAgIHRoaXMuY2FsZW5kYXJzJC5waXBlKHRha2VVbnRpbCh0aGlzLl9vbkRlc3Ryb3kkKSkuc3Vic2NyaWJlKCk7XG4gICAgfVxuICAgIHByaXZhdGUgX2NhbGNDb2xTcGFuKGRpZmY6IG51bWJlciwgZGF5SW5kZXg6IG51bWJlciwgbWF4Q29sU3BhbjogbnVtYmVyKTogbnVtYmVyIHtcbiAgICAgICAgaWYgKGRpZmYgPT09IDApIHtcbiAgICAgICAgICAgIHJldHVybiAyO1xuICAgICAgICB9XG4gICAgICAgIGlmIChNYXRoLmZsb29yKGRpZmYgLyBtYXhDb2xTcGFuKSA+IDApIHtcbiAgICAgICAgICAgIHJldHVybiBtYXhDb2xTcGFuO1xuICAgICAgICB9XG5cbiAgICAgICAgaWYgKGRheUluZGV4ICsgZGlmZiA+IG1heENvbFNwYW4pIHtcbiAgICAgICAgICAgIHJldHVybiBtYXhDb2xTcGFuIC0gZGF5SW5kZXg7XG4gICAgICAgIH1cbiAgICAgICAgcmV0dXJuIGRpZmY7XG4gICAgfVxuXG4gICAgcHJpdmF0ZSBfbG9hZE5leHRNb250aChkYXlzOiBEYXlbXSwgbW9udGhJbmZvOiBDYWxlbmRhck1vbnRoSW5mbyk6IEFycmF5PERheT4ge1xuICAgICAgICBjb25zdCBkYXlzQ291bnQgPSBkYXlzLmxlbmd0aDtcbiAgICAgICAgbGV0IGxhc3REYXkgPSBkYXlzW2RheXNDb3VudCAtIDFdO1xuICAgICAgICBjb25zdCBsYXN0RGF5SW5kZXggPSBtb250aEluZm8uZGF5cy5maW5kSW5kZXgoXG4gICAgICAgICAgICAoYykgPT4gYy5kYXlJbk1vbnRoID09PSBsYXN0RGF5LmRheUluTW9udGggJiYgYy5tb250aCA9PT0gbGFzdERheS5tb250aFxuICAgICAgICApO1xuICAgICAgICBsZXQgbmV3RGF5cyA9IGxhc3REYXlJbmRleCA+IC0xID8gbW9udGhJbmZvLmRheXMuc2xpY2UobGFzdERheUluZGV4ICsgMSwgbGFzdERheUluZGV4ICsgMSArIGRheXNDb3VudCkgOiBbXTtcbiAgICAgICAgaWYgKG5ld0RheXMubGVuZ3RoIDwgZGF5c0NvdW50KSB7XG4gICAgICAgICAgICAvLyB3ZSBtb3ZlIHRvIG5leHQgbW9udGhcbiAgICAgICAgICAgIGlmIChuZXdEYXlzLmxlbmd0aCA+IDApIHtcbiAgICAgICAgICAgICAgICBsYXN0RGF5ID0gbmV3RGF5c1tuZXdEYXlzLmxlbmd0aCAtIDFdO1xuICAgICAgICAgICAgfVxuICAgICAgICAgICAgY29uc3QgbmVlZENvdW50RGF5ID0gZGF5c0NvdW50IC0gbmV3RGF5cy5sZW5ndGg7XG4gICAgICAgICAgICBjb25zdCBuZXh0TW9udGggPSB0aGlzLmxvYWRNb250aChtb250aEluZm8uZGF0ZUluZm8sIDEpO1xuICAgICAgICAgICAgbGV0IHN0YXJ0UG9zaXRpb24gPSAwO1xuICAgICAgICAgICAgaWYgKG5lZWRDb3VudERheSA8IDQyKSB7XG4gICAgICAgICAgICAgICAgc3RhcnRQb3NpdGlvbiA9IG5leHRNb250aC5kYXlzLmZpbmRJbmRleChcbiAgICAgICAgICAgICAgICAgICAgKGMpID0+IGMuZGF5SW5Nb250aCA9PT0gbGFzdERheS5kYXlJbk1vbnRoICYmIGMubW9udGggPT09IGxhc3REYXkubW9udGhcbiAgICAgICAgICAgICAgICApO1xuICAgICAgICAgICAgICAgIGlmIChzdGFydFBvc2l0aW9uID09PSAtMSkge1xuICAgICAgICAgICAgICAgICAgICBzdGFydFBvc2l0aW9uID0gMDtcbiAgICAgICAgICAgICAgICB9IGVsc2Uge1xuICAgICAgICAgICAgICAgICAgICBzdGFydFBvc2l0aW9uKys7XG4gICAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgfVxuICAgICAgICAgICAgY29uc3QgbmV4dERheXMgPSBuZXh0TW9udGguZGF5cy5zbGljZShzdGFydFBvc2l0aW9uLCBzdGFydFBvc2l0aW9uICsgbmVlZENvdW50RGF5KTtcbiAgICAgICAgICAgIG5ld0RheXMgPSBbLi4ubmV3RGF5cywgLi4ubmV4dERheXNdO1xuICAgICAgICB9XG4gICAgICAgIHJldHVybiBuZXdEYXlzO1xuICAgIH1cbiAgICBwcml2YXRlIF9sb2FkTmV4dFdlZWsoZGF5czogRGF5W10sIG1vbnRoSW5mbzogQ2FsZW5kYXJNb250aEluZm8pOiBBcnJheTxEYXk+IHtcbiAgICAgICAgY29uc3QgZGF5c0NvdW50ID0gZGF5cy5sZW5ndGg7XG4gICAgICAgIGNvbnN0IGxhc3REYXkgPSBkYXlzW2RheXNDb3VudCAtIDFdO1xuICAgICAgICBjb25zdCBsYXN0RGF5SW5kZXggPSBtb250aEluZm8uZGF5cy5maW5kSW5kZXgoXG4gICAgICAgICAgICAoYykgPT4gYy5kYXlJbk1vbnRoID09PSBsYXN0RGF5LmRheUluTW9udGggJiYgYy5tb250aCA9PT0gbGFzdERheS5tb250aFxuICAgICAgICApO1xuICAgICAgICBsZXQgbmV3RGF5cyA9IG1vbnRoSW5mby5kYXlzLnNsaWNlKGxhc3REYXlJbmRleCArIDEsIGxhc3REYXlJbmRleCArIDEgKyBkYXlzQ291bnQpO1xuICAgICAgICBpZiAobmV3RGF5cy5sZW5ndGggPCBkYXlzQ291bnQpIHtcbiAgICAgICAgICAgIC8vIHdlIG1vdmUgdG8gbmV4dCBtb250aFxuICAgICAgICAgICAgbGV0IGxhc3ROZXdEYXkgPSBuZXdEYXlzW25ld0RheXMubGVuZ3RoIC0gMV07XG4gICAgICAgICAgICBpZiAoIWxhc3ROZXdEYXkpIHtcbiAgICAgICAgICAgICAgICBsYXN0TmV3RGF5ID0gbGFzdERheTtcbiAgICAgICAgICAgIH1cbiAgICAgICAgICAgIGNvbnN0IG5leHRNb250aCA9IHRoaXMubG9hZE1vbnRoKG1vbnRoSW5mby5kYXRlSW5mbywgMSk7XG4gICAgICAgICAgICBjb25zdCBsYXN0RGF5SW5kZXgyID0gbmV4dE1vbnRoLmRheXMuZmluZEluZGV4KFxuICAgICAgICAgICAgICAgIChjKSA9PiBjLmRheUluTW9udGggPT09IGxhc3ROZXdEYXkuZGF5SW5Nb250aCAmJiBjLm1vbnRoID09PSBsYXN0TmV3RGF5Lm1vbnRoXG4gICAgICAgICAgICApO1xuICAgICAgICAgICAgbmV3RGF5cyA9IG5ld0RheXMuY29uY2F0KFxuICAgICAgICAgICAgICAgIG5leHRNb250aC5kYXlzLnNsaWNlKGxhc3REYXlJbmRleDIgKyAxLCBsYXN0RGF5SW5kZXgyICsgMSArIChkYXlzQ291bnQgLSBuZXdEYXlzLmxlbmd0aCkpXG4gICAgICAgICAgICApO1xuICAgICAgICB9XG4gICAgICAgIHJldHVybiBuZXdEYXlzO1xuICAgIH1cblxuICAgIHByaXZhdGUgX2xvYWRQcmV2TW9udGgoZGF5czogRGF5W10sIG1vbnRoSW5mbzogQ2FsZW5kYXJNb250aEluZm8pOiBBcnJheTxEYXk+IHtcbiAgICAgICAgY29uc3QgZGF5c0NvdW50ID0gZGF5cy5sZW5ndGg7XG4gICAgICAgIGxldCBmaXJzdERheSA9IGRheXNbMF07XG4gICAgICAgIGNvbnN0IGZpcnN0RGF5SW5kZXggPSBtb250aEluZm8uZGF5cy5maW5kSW5kZXgoXG4gICAgICAgICAgICAoYykgPT4gYy5kYXlJbk1vbnRoID09PSBmaXJzdERheS5kYXlJbk1vbnRoICYmIGMubW9udGggPT09IGZpcnN0RGF5Lm1vbnRoXG4gICAgICAgICk7XG4gICAgICAgIGxldCBuZXdEYXlzOiBEYXlbXSA9IG1vbnRoSW5mby5kYXlzLnNsaWNlKFxuICAgICAgICAgICAgZmlyc3REYXlJbmRleCAtIGRheXNDb3VudCA+IDAgPyBmaXJzdERheUluZGV4IC0gZGF5c0NvdW50IDogMCxcbiAgICAgICAgICAgIGZpcnN0RGF5SW5kZXhcbiAgICAgICAgKTtcbiAgICAgICAgaWYgKG5ld0RheXMubGVuZ3RoIDwgZGF5c0NvdW50KSB7XG4gICAgICAgICAgICAvLyB3ZSBtb3ZlIHRvIHByZXYgbW9udGhcbiAgICAgICAgICAgIGlmIChuZXdEYXlzLmxlbmd0aCA+IDApIHtcbiAgICAgICAgICAgICAgICBmaXJzdERheSA9IG5ld0RheXNbMF07XG4gICAgICAgICAgICB9XG4gICAgICAgICAgICBjb25zdCBuZWVkQ291bnREYXkgPSBkYXlzQ291bnQgLSBuZXdEYXlzLmxlbmd0aDtcbiAgICAgICAgICAgIGNvbnN0IHByZXZNb250aCA9IHRoaXMubG9hZE1vbnRoKG1vbnRoSW5mby5kYXRlSW5mbywgLTEpO1xuICAgICAgICAgICAgbGV0IGVuZFBvc2l0aW9uID0gcHJldk1vbnRoLmRheXMubGVuZ3RoO1xuICAgICAgICAgICAgaWYgKG5lZWRDb3VudERheSA8IDQyKSB7XG4gICAgICAgICAgICAgICAgZW5kUG9zaXRpb24gPSBwcmV2TW9udGguZGF5cy5maW5kSW5kZXgoXG4gICAgICAgICAgICAgICAgICAgIChjKSA9PiBjLmRheUluTW9udGggPT09IGZpcnN0RGF5LmRheUluTW9udGggJiYgYy5tb250aCA9PT0gZmlyc3REYXkubW9udGhcbiAgICAgICAgICAgICAgICApO1xuICAgICAgICAgICAgICAgIGlmIChlbmRQb3NpdGlvbiA9PT0gLTEpIHtcbiAgICAgICAgICAgICAgICAgICAgZW5kUG9zaXRpb24gPSBwcmV2TW9udGguZGF5cy5sZW5ndGg7XG4gICAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgfVxuICAgICAgICAgICAgY29uc3QgcHJldkRheXMgPSBwcmV2TW9udGguZGF5cy5zbGljZShlbmRQb3NpdGlvbiAtIG5lZWRDb3VudERheSwgZW5kUG9zaXRpb24pO1xuICAgICAgICAgICAgbmV3RGF5cyA9IFsuLi5wcmV2RGF5cywgLi4ubmV3RGF5c107XG4gICAgICAgIH1cbiAgICAgICAgcmV0dXJuIG5ld0RheXM7XG4gICAgfVxuICAgIHByaXZhdGUgX2xvYWRQcmV2V2VlayhkYXlzOiBEYXlbXSwgbW9udGhJbmZvOiBDYWxlbmRhck1vbnRoSW5mbyk6IEFycmF5PERheT4ge1xuICAgICAgICBjb25zdCBkYXlzQ291bnQgPSBkYXlzLmxlbmd0aDtcbiAgICAgICAgY29uc3QgZmlyc3REYXkgPSBkYXlzWzBdO1xuICAgICAgICBjb25zdCBmaXJzdERheUluZGV4ID0gbW9udGhJbmZvLmRheXMuZmluZEluZGV4KFxuICAgICAgICAgICAgKGMpID0+IGMuZGF5SW5Nb250aCA9PT0gZmlyc3REYXkuZGF5SW5Nb250aCAmJiBjLm1vbnRoID09PSBmaXJzdERheS5tb250aFxuICAgICAgICApO1xuICAgICAgICBjb25zdCBwYXJ0SW5kZXhQcmV2ID0gZmlyc3REYXlJbmRleCAtIGRheXNDb3VudDtcbiAgICAgICAgbGV0IG5ld0RheXM6IERheVtdID0gW107XG4gICAgICAgIGlmIChwYXJ0SW5kZXhQcmV2IDwgMCkge1xuICAgICAgICAgICAgbmV3RGF5cyA9IGZpcnN0RGF5SW5kZXggLSAxID49IDAgPyBtb250aEluZm8uZGF5cy5zbGljZSgwLCBmaXJzdERheUluZGV4IC0gMSkgOiBbXTtcbiAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICAgIG5ld0RheXMgPSBtb250aEluZm8uZGF5cy5zbGljZShmaXJzdERheUluZGV4IC0gZGF5c0NvdW50LCBmaXJzdERheUluZGV4KTtcbiAgICAgICAgfVxuICAgICAgICBpZiAobmV3RGF5cy5sZW5ndGggPCBkYXlzQ291bnQpIHtcbiAgICAgICAgICAgIC8vIHdlIG1vdmUgdG8gcHJldiBtb250aFxuICAgICAgICAgICAgbGV0IGZpcnN0TmV3RGF5ID0gbmV3RGF5c1swXTtcbiAgICAgICAgICAgIGlmICghZmlyc3ROZXdEYXkpIHtcbiAgICAgICAgICAgICAgICBmaXJzdE5ld0RheSA9IGZpcnN0RGF5O1xuICAgICAgICAgICAgfVxuICAgICAgICAgICAgY29uc3QgcHJldk1vbnRoID0gdGhpcy5sb2FkTW9udGgobW9udGhJbmZvLmRhdGVJbmZvLCAtMSk7XG4gICAgICAgICAgICBjb25zdCBmaXJzdERheUluZGV4MiA9IHByZXZNb250aC5kYXlzLmZpbmRJbmRleChcbiAgICAgICAgICAgICAgICAoYykgPT4gYy5kYXlJbk1vbnRoID09PSBmaXJzdE5ld0RheS5kYXlJbk1vbnRoICYmIGMubW9udGggPT09IGZpcnN0TmV3RGF5Lm1vbnRoXG4gICAgICAgICAgICApO1xuICAgICAgICAgICAgbmV3RGF5cyA9IG5ld0RheXMuY29uY2F0KFxuICAgICAgICAgICAgICAgIHByZXZNb250aC5kYXlzLnNsaWNlKGZpcnN0RGF5SW5kZXgyIC0gKGRheXNDb3VudCAtIG5ld0RheXMubGVuZ3RoKSwgZmlyc3REYXlJbmRleDIpXG4gICAgICAgICAgICApO1xuICAgICAgICB9XG4gICAgICAgIHJldHVybiBuZXdEYXlzO1xuICAgIH1cblxuICAgIHByaXZhdGUgZ2V0Q2FsZW5kYXJNb250aEluZm8oZGF0ZSk6IENhbGVuZGFyTW9udGhJbmZvIHtcbiAgICAgICAgY29uc3Qgc3RhcnRPZk1vbnRoID0gdGhpcy5fZGF0ZVNlcnZpY2Uuc3RhcnRPZk1vbnRoKGRhdGUpO1xuICAgICAgICBjb25zdCBtb21lbnREYXRlID0gdGhpcy5fZGF0ZVNlcnZpY2UuZ2V0TW9tZW50KGRhdGUpO1xuICAgICAgICByZXR1cm4gdGhpcy5nZXRNb250aEluZm8oc3RhcnRPZk1vbnRoLCBtb21lbnREYXRlKTtcbiAgICB9XG4gICAgcHJpdmF0ZSBnZXRNb250aEluZm8oZGF0ZSwgdG9kYXkpOiBDYWxlbmRhck1vbnRoSW5mbyB7XG4gICAgICAgIGNvbnN0IGRheXM6IERheVtdID0gW107XG4gICAgICAgIGNvbnN0IGFjdGl2ZURhdGUgPSB0aGlzLl9kYXRlU2VydmljZS5nZXRNb21lbnQoZGF0ZSk7XG4gICAgICAgIGNvbnN0IHN0YXJ0T2ZNb250aDogRGF0ZSA9IHRoaXMuX2RhdGVTZXJ2aWNlLnN0YXJ0T2ZNb250aChhY3RpdmVEYXRlKTtcbiAgICAgICAgY29uc3QgZW5kT2ZNb250aDogRGF0ZSA9IHRoaXMuX2RhdGVTZXJ2aWNlLmVuZE9mTW9udGgoYWN0aXZlRGF0ZSk7XG4gICAgICAgIGNvbnN0IGRhdGVJbmZvID0gdGhpcy5fZGF0ZVNlcnZpY2UuZ2V0RGF0ZUluZm8oYWN0aXZlRGF0ZSk7XG4gICAgICAgIGNvbnN0IGRheUluV2VlayA9IHRoaXMuX2RhdGVTZXJ2aWNlLmRheShzdGFydE9mTW9udGgpICsgMTsgLy8gd2UgYWRkIDEgdG8gdGhlIGRheUluV2VlayBiZWNhdXNlIGl0J3Mgc3RhcnRzIHdpdGggemVyb1xuXG4gICAgICAgIGxldCB3ZWVrQ291bnRlciA9IDA7XG4gICAgICAgIGxldCBkYXlDb3VudGVyID0gMDtcbiAgICAgICAgZm9yIChsZXQgaSA9IDA7IGkgPCA0MjsgaSsrKSB7XG4gICAgICAgICAgICBjb25zdCBkYXRlT2ZTdGFydE9mTW9udGg6IERhdGUgPSBuZXcgRGF0ZShzdGFydE9mTW9udGgpO1xuICAgICAgICAgICAgZGF0ZU9mU3RhcnRPZk1vbnRoLnNldERhdGUoZGF0ZU9mU3RhcnRPZk1vbnRoLmdldERhdGUoKSArIGkgLSBkYXlJbldlZWspO1xuICAgICAgICAgICAgY29uc3QgeyBkYXksIGlzSW5Nb250aCB9OiB7IGRheTogRGF5OyBpc0luTW9udGg6IGFueSB9ID0gdGhpcy5fZ2V0RGF5KFxuICAgICAgICAgICAgICAgIGRhdGVPZlN0YXJ0T2ZNb250aCxcbiAgICAgICAgICAgICAgICBhY3RpdmVEYXRlLFxuICAgICAgICAgICAgICAgIHRvZGF5LFxuICAgICAgICAgICAgICAgIGVuZE9mTW9udGgsXG4gICAgICAgICAgICAgICAgc3RhcnRPZk1vbnRoLFxuICAgICAgICAgICAgICAgIGRhdGVJbmZvLFxuICAgICAgICAgICAgICAgIGlcbiAgICAgICAgICAgICk7XG4gICAgICAgICAgICBkYXlzLnB1c2goZGF5KTtcbiAgICAgICAgICAgIGRheUNvdW50ZXIrKztcbiAgICAgICAgICAgIGlmIChpc0luTW9udGggJiYgZGF5Q291bnRlciA+IHdlZWtDb3VudGVyICogNykge1xuICAgICAgICAgICAgICAgIHdlZWtDb3VudGVyKys7XG4gICAgICAgICAgICB9XG4gICAgICAgIH1cbiAgICAgICAgLy8gY29uc3QgZW5kT2ZNb250aEluZGV4ID0gZGF5cy5maW5kSW5kZXgoKGMpID0+IGMuaXNFbmRPZk1vbnRoKTtcbiAgICAgICAgLy8gaWYgKGVuZE9mTW9udGhJbmRleCA+IC0xICYmIGRheXMubGVuZ3RoIC0gZW5kT2ZNb250aEluZGV4ID4gNykge1xuICAgICAgICAvLyAgICAgLy8gZGF5cy5zcGxpY2UoZGF5cy5sZW5ndGggLSA3LCA3KTtcbiAgICAgICAgLy8gfVxuICAgICAgICBjb25zdCBjYWxlbmRhckluZm8gPSBuZXcgQ2FsZW5kYXJNb250aEluZm8oKTtcbiAgICAgICAgY2FsZW5kYXJJbmZvLmlkID0gZ2V0VW5pcXVlSWQoMyk7XG4gICAgICAgIGNhbGVuZGFySW5mby5kYXRlSW5mbyA9IGRhdGVJbmZvO1xuICAgICAgICBjYWxlbmRhckluZm8uZGF5cyA9IGRheXM7XG4gICAgICAgIGNhbGVuZGFySW5mby53ZWVrc0NvdW50ID0gd2Vla0NvdW50ZXI7XG4gICAgICAgIC8vIHRoaXMuY2RyLmRldGVjdENoYW5nZXMoKTtcbiAgICAgICAgcmV0dXJuIGNhbGVuZGFySW5mbztcbiAgICB9XG4gICAgcHJpdmF0ZSBfZ2V0RGF5KFxuICAgICAgICBkYXRlOiBEYXRlLFxuICAgICAgICBhY3RpdmVEYXRlOiBhbnksXG4gICAgICAgIHRvZGF5OiBhbnksXG4gICAgICAgIGVuZE9mTW9udGg6IERhdGUsXG4gICAgICAgIHN0YXJ0T2ZNb250aDogRGF0ZSxcbiAgICAgICAgZGF0ZUluZm86IERhdGVJbmZvLFxuICAgICAgICBpOiBudW1iZXJcbiAgICApOiB7IGRheTogRGF5OyBpc0luTW9udGg6IGJvb2xlYW4gfSB7XG4gICAgICAgIGNvbnN0IGlzSW5Nb250aCA9IHRoaXMuX2RhdGVTZXJ2aWNlLmlzU2FtZShkYXRlLCBhY3RpdmVEYXRlLCAnbW9udGgnKTtcbiAgICAgICAgY29uc3QgbW9tZW50T2ZEYXRlID0gdGhpcy5fZGF0ZVNlcnZpY2UuZ2V0TW9tZW50KGRhdGUpO1xuICAgICAgICBjb25zdCBpc1RvZGF5ID0gbW9tZW50T2ZEYXRlLmlzU2FtZSh0b2RheSwgJ2RheScpO1xuICAgICAgICBjb25zdCBkYXlJbk1vbnRoID0gdGhpcy5fZGF0ZVNlcnZpY2Uuc3RyRGF0ZShkYXRlKTtcbiAgICAgICAgY29uc3QgZGF5SW5XZWVrID0gdGhpcy5fZGF0ZVNlcnZpY2UuZGF5KGRhdGUpO1xuICAgICAgICBjb25zdCBpc0VuZE9mTW9udGggPSBtb21lbnRPZkRhdGUuaXNTYW1lKGVuZE9mTW9udGgsICdkYXknKTtcbiAgICAgICAgY29uc3QgaXNTdGFydE9mTW9udGggPSBtb21lbnRPZkRhdGUuaXNTYW1lKHN0YXJ0T2ZNb250aCwgJ2RheScpO1xuICAgICAgICBjb25zdCBtb250aERhdGVJbmZvID0gaXNJbk1vbnRoID8gZGF0ZUluZm8gOiB0aGlzLl9kYXRlU2VydmljZS5nZXREYXRlSW5mbyhkYXRlKTtcbiAgICAgICAgY29uc3QgZGF5OiBEYXkgPSB7XG4gICAgICAgICAgICBpbmRleDogaSxcbiAgICAgICAgICAgIGRhdGUsXG4gICAgICAgICAgICBkYXlJbldlZWssXG4gICAgICAgICAgICBzdGF0dXM6IERheVN0YXR1cy5PcGVuLFxuICAgICAgICAgICAgZGF5SW5Nb250aCxcbiAgICAgICAgICAgIGRheUluTW9udGhOOiB0aGlzLmdldE51bWJlcihkYXlJbk1vbnRoKSxcbiAgICAgICAgICAgIGlzVG9kYXksXG4gICAgICAgICAgICBpc0luTW9udGgsXG4gICAgICAgICAgICBpc0VuZE9mTW9udGgsXG4gICAgICAgICAgICBpc1N0YXJ0T2ZNb250aCxcbiAgICAgICAgICAgIG1vbWVudE9mRGF0ZSxcbiAgICAgICAgICAgIG1vbnRoOiBtb250aERhdGVJbmZvLm1vbnRoLFxuICAgICAgICAgICAgeWVhcjogbW9udGhEYXRlSW5mby55ZWFyLFxuICAgICAgICAgICAgbW9udGhOYW1lOiBtb250aERhdGVJbmZvLm1vbnRoTmFtZSxcbiAgICAgICAgICAgIGRheU5hbWU6IG1vbWVudE9mRGF0ZS5mb3JtYXQoJ2RkZGQnKSxcbiAgICAgICAgICAgIHRhc2tzOiBbXSxcbiAgICAgICAgICAgIG1vbnRoRGF0ZUluZm8sXG4gICAgICAgICAgICByb3c6IE1hdGguZmxvb3IoaSAvIDcgKyAxKSxcbiAgICAgICAgICAgIGNvbDogNiAtIChpICUgNylcbiAgICAgICAgfTtcbiAgICAgICAgcmV0dXJuIHsgZGF5LCBpc0luTW9udGggfTtcbiAgICB9XG59XG4iXX0=
|