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.
Files changed (56) hide show
  1. package/README.md +25 -0
  2. package/esm2020/barsa-calendar.mjs +5 -0
  3. package/esm2020/lib/barsa-calendar.module.mjs +106 -0
  4. package/esm2020/lib/calendar-container/calendar-container.component.mjs +190 -0
  5. package/esm2020/lib/calendar-header/calendar-header.component.mjs +61 -0
  6. package/esm2020/lib/calendar-list/calendar-list.component.mjs +19 -0
  7. package/esm2020/lib/calendar-list-container/calendar-list-container.component.mjs +30 -0
  8. package/esm2020/lib/calendar-list-item/calendar-list-item.component.mjs +14 -0
  9. package/esm2020/lib/calendar-month/calendar-month.component.mjs +244 -0
  10. package/esm2020/lib/calendar-selection-days/calendar-selection-days.component.mjs +193 -0
  11. package/esm2020/lib/calendar-week/calendar-week.component.mjs +182 -0
  12. package/esm2020/lib/constants.mjs +4 -0
  13. package/esm2020/lib/date-time-picker/date-time-picker.component.mjs +324 -0
  14. package/esm2020/lib/day-event-list/day-event-list.component.mjs +53 -0
  15. package/esm2020/lib/day-number-box/day-number-box.component.mjs +105 -0
  16. package/esm2020/lib/days-in-week.pipe.mjs +17 -0
  17. package/esm2020/lib/equal-date.pipe.mjs +23 -0
  18. package/esm2020/lib/event-button/event-button.component.mjs +70 -0
  19. package/esm2020/lib/event-date.pipe.mjs +24 -0
  20. package/esm2020/lib/event-preview/event-preview.component.mjs +60 -0
  21. package/esm2020/lib/event-time.pipe.mjs +22 -0
  22. package/esm2020/lib/from-to-time.pipe.mjs +23 -0
  23. package/esm2020/lib/models.mjs +14 -0
  24. package/esm2020/lib/month-days/month-days.component.mjs +220 -0
  25. package/esm2020/lib/services/calendar.service.mjs +610 -0
  26. package/esm2020/public-api.mjs +21 -0
  27. package/fesm2015/barsa-calendar.mjs +2447 -0
  28. package/fesm2015/barsa-calendar.mjs.map +1 -0
  29. package/fesm2020/barsa-calendar.mjs +2455 -0
  30. package/fesm2020/barsa-calendar.mjs.map +1 -0
  31. package/index.d.ts +5 -0
  32. package/lib/barsa-calendar.module.d.ts +33 -0
  33. package/lib/calendar-container/calendar-container.component.d.ts +73 -0
  34. package/lib/calendar-header/calendar-header.component.d.ts +29 -0
  35. package/lib/calendar-list/calendar-list.component.d.ts +8 -0
  36. package/lib/calendar-list-container/calendar-list-container.component.d.ts +14 -0
  37. package/lib/calendar-list-item/calendar-list-item.component.d.ts +7 -0
  38. package/lib/calendar-month/calendar-month.component.d.ts +70 -0
  39. package/lib/calendar-selection-days/calendar-selection-days.component.d.ts +47 -0
  40. package/lib/calendar-week/calendar-week.component.d.ts +57 -0
  41. package/lib/constants.d.ts +1 -0
  42. package/lib/date-time-picker/date-time-picker.component.d.ts +98 -0
  43. package/lib/day-event-list/day-event-list.component.d.ts +20 -0
  44. package/lib/day-number-box/day-number-box.component.d.ts +30 -0
  45. package/lib/days-in-week.pipe.d.ts +9 -0
  46. package/lib/equal-date.pipe.d.ts +8 -0
  47. package/lib/event-button/event-button.component.d.ts +26 -0
  48. package/lib/event-date.pipe.d.ts +9 -0
  49. package/lib/event-preview/event-preview.component.d.ts +19 -0
  50. package/lib/event-time.pipe.d.ts +9 -0
  51. package/lib/from-to-time.pipe.d.ts +9 -0
  52. package/lib/models.d.ts +107 -0
  53. package/lib/month-days/month-days.component.d.ts +69 -0
  54. package/lib/services/calendar.service.d.ts +70 -0
  55. package/package.json +31 -0
  56. 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,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY2FsZW5kYXIuc2VydmljZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL2JhcnNhLWNhbGVuZGFyL3NyYy9saWIvc2VydmljZXMvY2FsZW5kYXIuc2VydmljZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsTUFBTSxFQUFFLFVBQVUsRUFBRSxRQUFRLEVBQUUsTUFBTSxlQUFlLENBQUM7QUFDN0QsT0FBTyxFQUFjLGVBQWUsRUFBRSxNQUFNLE1BQU0sQ0FBQztBQUNuRCxPQUFPLEVBQUUsU0FBUyxFQUFFLE1BQU0sZ0JBQWdCLENBQUM7QUFFM0MsT0FBTyxFQUVILGFBQWEsRUFDYixXQUFXLEVBQ1gsY0FBYyxFQUtqQixNQUFNLHNCQUFzQixDQUFDO0FBQzlCLE9BQU8sRUFBZ0IsaUJBQWlCLEVBQU8sU0FBUyxFQUFFLE1BQU0sV0FBVyxDQUFDOzs7QUFHNUUsTUFBTSxPQUFPLGVBQWdCLFNBQVEsYUFBYTtJQW1COUMsWUFBb0IsV0FBdUIsRUFBc0MsT0FBZ0I7UUFDN0YsS0FBSyxFQUFFLENBQUM7UUFEUSxnQkFBVyxHQUFYLFdBQVcsQ0FBWTtRQVBuQyxxQkFBZ0IsR0FBRyxJQUFJLGVBQWUsQ0FBc0IsRUFBRSxDQUFDLENBQUM7UUFFaEUsZ0JBQVcsR0FBRyxJQUFJLGVBQWUsQ0FBZSxPQUFPLENBQUMsQ0FBQztRQUN6RCw2QkFBd0IsR0FBRyxJQUFJLGVBQWUsQ0FBUyxDQUFDLENBQUMsQ0FBQyxDQUFDO1FBQzNELHdCQUFtQixHQUFHLElBQUksZUFBZSxDQUFRLEVBQUUsQ0FBQyxDQUFDO1FBQ3JELGdCQUFXLEdBQUcsSUFBSSxlQUFlLENBQVEsRUFBRSxDQUFDLENBQUM7UUFJakQsSUFBSSxDQUFDLFlBQVksR0FBRyxjQUFjLENBQUMsV0FBVyxFQUFFLE9BQU8sQ0FBQyxDQUFDO1FBQ3pELElBQUksQ0FBQyxLQUFLLEdBQUcsSUFBSSxDQUFDLFdBQVcsQ0FBQyxZQUFZLEVBQUUsQ0FBQztRQUM3QyxJQUFJLENBQUMsS0FBSyxHQUFHLElBQUksQ0FBQyxXQUFXLENBQUMsWUFBWSxFQUFFLENBQUM7UUFDN0MsSUFBSSxDQUFDLGFBQWEsR0FBRyxJQUFJLENBQUMsbUJBQW1CLENBQUMsWUFBWSxFQUFFLENBQUM7UUFDN0QsSUFBSSxDQUFDLFVBQVUsR0FBRyxJQUFJLENBQUMsZ0JBQWdCLENBQUMsWUFBWSxFQUFFLENBQUM7UUFDdkQsSUFBSSxDQUFDLGtCQUFrQixHQUFHLElBQUksQ0FBQyx3QkFBd0IsQ0FBQyxZQUFZLEVBQUUsQ0FBQztRQUN2RSxJQUFJLENBQUMsVUFBVSxFQUFFLENBQUM7SUFDdEIsQ0FBQztJQUNELElBQUksY0FBYztRQUNkLE9BQU8sSUFBSSxDQUFDLGVBQWUsQ0FBQztJQUNoQyxDQUFDO0lBQ0QsSUFBSSxhQUFhO1FBQ2IsT0FBTyxJQUFJLENBQUMsZUFBZSxDQUFDLFFBQVEsRUFBRSxDQUFDO0lBQzNDLENBQUM7SUFDRCxJQUFJLFNBQVM7UUFDVCxPQUFPLElBQUksQ0FBQyxnQkFBZ0IsQ0FBQyxRQUFRLEVBQUUsQ0FBQztJQUM1QyxDQUFDO0lBQ0QsSUFBSSxVQUFVO1FBQ1YsT0FBTyxJQUFJLENBQUMsV0FBVyxDQUFDLFFBQVEsRUFBRSxDQUFDLE1BQU0sR0FBRyxDQUFDLENBQUM7SUFDbEQsQ0FBQztJQUNELElBQUksSUFBSTtRQUNKLE9BQU8sSUFBSSxDQUFDLFdBQVcsQ0FBQyxRQUFRLEVBQUUsQ0FBQztJQUN2QyxDQUFDO0lBQ0QsSUFBSSxNQUFNO1FBQ04sT0FBTyxJQUFJLENBQUMsWUFBWSxDQUFDLFVBQVUsRUFBRSxDQUFDO0lBQzFDLENBQUM7SUFFRCxlQUFlLENBQUMsS0FBVztRQUN2QixNQUFNLFNBQVMsR0FBRyxJQUFJLENBQUMsWUFBWSxDQUFDLFdBQVcsQ0FBQyxLQUFLLENBQUMsQ0FBQztRQUN2RCxJQUFJLENBQUMsU0FBUyxDQUFDLFNBQVMsQ0FBQyxDQUFDO0lBQzlCLENBQUM7SUFDRCxjQUFjLENBQUMsbUJBQTJCO1FBQ3RDLElBQUksV0FBVyxHQUFHLElBQUksQ0FBQyxZQUFZLENBQUM7UUFDcEMsUUFBUSxtQkFBbUIsRUFBRTtZQUN6QixLQUFLLFNBQVM7Z0JBQ1YsV0FBVyxHQUFHLGNBQWMsQ0FBQyxJQUFJLENBQUMsV0FBVyxFQUFFLE9BQU8sQ0FBQyxDQUFDO2dCQUN4RCxNQUFNO1lBQ1YsS0FBSyxXQUFXO2dCQUNaLFdBQVcsR0FBRyxjQUFjLENBQUMsSUFBSSxDQUFDLFdBQVcsRUFBRSxPQUFPLENBQUMsQ0FBQztnQkFDeEQsTUFBTTtZQUNWLEtBQUssUUFBUTtnQkFDVCxXQUFXLEdBQUcsY0FBYyxDQUFDLElBQUksQ0FBQyxXQUFXLEVBQUUsT0FBTyxDQUFDLENBQUM7Z0JBQ3hELE1BQU07U0FDYjtRQUNELElBQUksQ0FBQyxZQUFZLEdBQUcsV0FBVyxDQUFDO1FBQ2hDLElBQUksQ0FBQyxVQUFVLEVBQUUsQ0FBQztJQUN0QixDQUFDO0lBQ0QsT0FBTyxDQUFDLElBQVc7UUFDZixJQUFJLENBQUMsV0FBVyxDQUFDLElBQUksQ0FBQyxJQUFJLENBQUMsQ0FBQztJQUNoQyxDQUFDO0lBRUQsV0FBVyxDQUFDLFNBQTRCLEVBQUUsVUFBa0IsRUFBRSxTQUFpQjtRQUMzRSxNQUFNLFNBQVMsR0FBRyxTQUFTLENBQUMsSUFBSSxDQUFDLEtBQUssQ0FBQyxVQUFVLEVBQUUsVUFBVSxHQUFHLFNBQVMsQ0FBQyxDQUFDO1FBQzNFLElBQUksQ0FBQyxtQkFBbUIsQ0FBQyxJQUFJLENBQUMsU0FBUyxDQUFDLENBQUM7SUFDN0MsQ0FBQztJQUNELFdBQVcsQ0FBQyxRQUEyQjtRQUNuQyxJQUFJLENBQUMsZUFBZSxDQUFDLElBQUksQ0FBQyxRQUFRLENBQUMsQ0FBQztJQUN4QyxDQUFDO0lBRUQsV0FBVztRQUNQLE9BQU8sSUFBSSxDQUFDLFlBQVksQ0FBQyxXQUFXLEVBQUUsQ0FBQztJQUMzQyxDQUFDO0lBQ0QsUUFBUTtRQUNKLE9BQU8sSUFBSSxDQUFDLFlBQVksQ0FBQyxRQUFRLEVBQUUsQ0FBQztJQUN4QyxDQUFDO0lBQ0QsU0FBUztRQUNMLElBQUksQ0FBQyxTQUFTLENBQUMsSUFBSSxDQUFDLGVBQWUsQ0FBQyxRQUFRLEVBQUUsQ0FBQyxDQUFDLENBQUM7UUFDakQsSUFBSSxDQUFDLE9BQU8sQ0FDUixJQUFJLENBQUMsV0FBVyxDQUFDLFFBQVEsRUFBRSxFQUMzQixJQUFJLENBQUMsZUFBZSxDQUFDLElBQUksQ0FBQyxJQUFJLENBQUMsQ0FBQyxDQUFDLEVBQUUsRUFBRSxDQUFDLENBQUMsQ0FBQyxPQUFPLENBQUMsRUFDaEQsSUFBSSxDQUFDLG1CQUFtQixDQUFDLFFBQVEsRUFBRSxDQUFDLE1BQU0sQ0FDN0MsQ0FBQztJQUNOLENBQUM7SUFDRCxXQUFXLENBQUMsUUFBYSxFQUFFLE1BQVcsRUFBRSxTQUE0QjtRQUNoRSxNQUFNLElBQUksR0FBRyxNQUFNLENBQUMsS0FBSyxHQUFHLFFBQVEsQ0FBQyxLQUFLLENBQUM7UUFDM0MsSUFBSSxJQUFJLEdBQUcsQ0FBQyxFQUFFO1lBQ1YsSUFBSSxDQUFDLE9BQU8sQ0FBQyxNQUFNLEVBQUUsUUFBUSxFQUFFLElBQUksR0FBRyxDQUFDLENBQUMsQ0FBQztTQUM1QzthQUFNO1lBQ0gsTUFBTSxLQUFLLEdBQUcsSUFBSSxHQUFHLENBQUMsQ0FBQztZQUN2QixJQUFJLENBQUMsT0FBTyxDQUFDLGFBQWEsQ0FBQyxDQUFDO1lBQzVCLElBQUksQ0FBQyxRQUFRLENBQUMsU0FBUyxFQUFFLFFBQVEsRUFBRSxLQUFLLENBQUMsQ0FBQztTQUM3QztJQUNMLENBQUM7SUFDRCxRQUFRLENBQUMsU0FBNEIsRUFBRSxRQUFjLEVBQUUsT0FBZSxDQUFDO1FBQ25FLElBQUksUUFBUSxJQUFJLElBQUksR0FBRyxDQUFDLEVBQUU7WUFDdEIsTUFBTSxVQUFVLEdBQUcsUUFBUSxDQUFDLEtBQUssQ0FBQztZQUNsQyxJQUFJLENBQUMscUJBQXFCLENBQUMsU0FBUyxDQUFDLENBQUM7WUFDdEMsSUFBSSxDQUFDLE9BQU8sQ0FBQyxTQUFTLENBQUMsSUFBSSxDQUFDLEtBQUssQ0FBQyxVQUFVLEVBQUUsVUFBVSxHQUFHLElBQUksQ0FBQyxDQUFDLENBQUM7U0FDckU7YUFBTTtZQUNILElBQUksQ0FBQyxxQkFBcUIsQ0FBQyxTQUFTLENBQUMsQ0FBQztZQUN0QyxJQUFJLENBQUMsT0FBTyxDQUFDLFNBQVMsQ0FBQyxJQUFJLENBQUMsQ0FBQztZQUM3QiwrQkFBK0I7WUFDL0IsOEJBQThCO1lBQzlCLHdEQUF3RDtZQUN4RCxJQUFJO1NBQ1A7SUFDTCxDQUFDO0lBQ0QsT0FBTyxDQUFDLElBQWtCLEVBQUUsUUFBYyxFQUFFLFNBQWtCO1FBQzFELE1BQU0sT0FBTyxHQUFHLElBQUksQ0FBQyxXQUFXLENBQUMsUUFBUSxFQUFFLENBQUM7UUFDNUMsSUFBSSxDQUFDLFdBQVcsQ0FBQyxJQUFJLENBQUMsSUFBSSxDQUFDLENBQUM7UUFDNUIsSUFBSSxJQUFJLEtBQUssT0FBTyxFQUFFO1lBQ2xCLElBQUksQ0FBQyxRQUFRLENBQUMsSUFBSSxDQUFDLGFBQWEsRUFBRSxRQUFRLENBQUMsQ0FBQztZQUM1QyxPQUFPO1NBQ1Y7YUFBTSxJQUFJLElBQUksS0FBSyxhQUFhLEVBQUU7WUFDL0IsSUFBSSxDQUFDLFFBQVEsQ0FBQyxJQUFJLENBQUMsYUFBYSxFQUFFLFFBQVEsRUFBRSxJQUFJLENBQUMsSUFBSSxDQUFDLE1BQU0sQ0FBQyxDQUFDO1lBQzlELE9BQU87U0FDVjtRQUNELE1BQU0sVUFBVSxHQUFHLElBQUksQ0FBQyxtQkFBbUIsQ0FBQyxRQUFRLEVBQUUsQ0FBQztRQUN2RCxTQUFTLEdBQUcsU0FBUyxJQUFJLENBQUMsSUFBSSxLQUFLLE1BQU0sQ0FBQyxDQUFDLENBQUMsQ0FBQyxDQUFDLENBQUMsQ0FBQyxDQUFDLENBQUMsQ0FBQztRQUNuRCxJQUFJLFVBQVUsR0FBRyxDQUFDLENBQUM7UUFDbkIsSUFBSSxDQUFDLFFBQVEsRUFBRTtZQUNYLElBQUksSUFBSSxLQUFLLE1BQU0sSUFBSSxJQUFJLEtBQUssS0FBSyxFQUFFO2dCQUNuQyxJQUFJLE9BQU8sS0FBSyxPQUFPLEVBQUU7b0JBQ3JCLFFBQVEsR0FBRyxJQUFJLENBQUMsYUFBYSxDQUFDLElBQUksQ0FBQyxJQUFJLENBQUMsQ0FBQyxDQUFDLEVBQUUsRUFBRSxDQUFDLENBQUMsQ0FBQyxjQUFjLENBQUMsQ0FBQztpQkFDcEU7cUJBQU0sSUFBSSxPQUFPLEtBQUssYUFBYSxFQUFFO29CQUNsQyxRQUFRLEdBQUcsSUFBSSxDQUFDLElBQUksQ0FBQyxDQUFDLENBQUMsQ0FBQztpQkFDM0I7cUJBQU07b0JBQ0gsUUFBUSxHQUFHLFVBQVUsQ0FBQyxDQUFDLENBQUMsQ0FBQztpQkFDNUI7YUFDSjtTQUNKO1FBRUQsSUFBSSxRQUFRLEVBQUU7WUFDVixJQUNJLFFBQVEsQ0FBQyxhQUFhLENBQUMsS0FBSyxLQUFLLElBQUksQ0FBQyxhQUFhLENBQUMsUUFBUSxDQUFDLEtBQUs7Z0JBQ2xFLFFBQVEsQ0FBQyxhQUFhLENBQUMsSUFBSSxLQUFLLElBQUksQ0FBQyxhQUFhLENBQUMsUUFBUSxDQUFDLElBQUksRUFDbEU7Z0JBQ0UsSUFBSSxDQUFDLFNBQVMsQ0FBQyxRQUFRLENBQUMsYUFBYSxDQUFDLENBQUM7YUFDMUM7WUFDRCxVQUFVLEdBQUcsSUFBSSxDQUFDLGFBQWEsQ0FBQyxJQUFJLENBQUMsU0FBUyxDQUMxQyxDQUFDLENBQUMsRUFBRSxFQUFFLENBQUMsQ0FBQyxDQUFDLEtBQUssS0FBSyxRQUFRLEVBQUUsS0FBSyxJQUFJLENBQUMsQ0FBQyxVQUFVLEtBQUssUUFBUSxDQUFDLFVBQVUsQ0FDN0UsQ0FBQztTQUNMO1FBQ0QsSUFBSSxDQUFDLFdBQVcsQ0FBQyxJQUFJLENBQUMsZUFBZSxDQUFDLFFBQVEsRUFBRSxFQUFFLFVBQVUsRUFBRSxTQUFTLENBQUMsQ0FBQztJQUM3RSxDQUFDO0lBRUQsUUFBUSxDQUFDLElBQThCLEVBQUUsU0FBUyxHQUFHLENBQUM7UUFDbEQsTUFBTSxHQUFHLEdBQVUsTUFBTSxDQUFDLE1BQU0sQ0FBQyxJQUFJLENBQUMsQ0FBQztRQUN2QyxHQUFHLENBQUMsT0FBTyxDQUNQLENBQ0ksT0FNRyxFQUNMLEVBQUU7WUFDQSxLQUFLLE1BQU0sRUFBRSxJQUFJLE9BQU8sRUFBRTtnQkFDdEIsTUFBTSxTQUFTLEdBQUcsRUFBRSxDQUFDLEdBQUcsQ0FBQztnQkFDekIsSUFBSSxTQUFTLElBQUksSUFBSSxDQUFDLFlBQVksQ0FBQyxTQUFTLENBQUMsU0FBUyxDQUFDLENBQUMsUUFBUSxDQUFDLElBQUksQ0FBQyxLQUFLLEVBQUUsS0FBSyxDQUFDLEVBQUU7b0JBQ2pGLEVBQUUsQ0FBQyxPQUFPLEdBQUcsR0FBRyxDQUFDO2lCQUNwQjthQUNKO1lBQ0QsS0FBSyxJQUFJLENBQUMsR0FBRyxPQUFPLENBQUMsTUFBTSxFQUFFLENBQUMsR0FBRyxTQUFTLEVBQUUsQ0FBQyxFQUFFLEVBQUU7Z0JBQzdDLE9BQU8sQ0FBQyxJQUFJLENBQUMsRUFBRSxPQUFPLEVBQUUsQ0FBQyxFQUFFLENBQUMsQ0FBQzthQUNoQztRQUNMLENBQUMsQ0FDSixDQUFDO0lBQ04sQ0FBQztJQUNELFNBQVMsQ0FBQyxhQUFvQyxFQUFFLGNBQThCO1FBQzFFLE1BQU0sY0FBYyxHQUFHLGNBQWMsQ0FBQyxTQUFTLENBQUM7UUFDaEQsTUFBTSxZQUFZLEdBQUcsY0FBYyxDQUFDLE9BQU8sQ0FBQztRQUM1QyxhQUFhLENBQUMsSUFBSSxDQUFDLENBQUMsS0FBSyxFQUFFLEtBQUssRUFBRSxFQUFFLENBQUMsQ0FBQyxLQUFLLENBQUMsY0FBYyxDQUFDLEdBQUcsS0FBSyxDQUFDLGNBQWMsQ0FBQyxDQUFDLENBQUMsQ0FBQyxDQUFDLENBQUMsQ0FBQyxDQUFDLENBQUMsQ0FBQyxDQUFDLENBQUMsQ0FBQztRQUMvRixhQUFhLENBQUMsSUFBSSxDQUFDLENBQUMsS0FBSyxFQUFFLEtBQUssRUFBRSxFQUFFO1lBQ2hDLE1BQU0sY0FBYyxHQUFHLEtBQUssQ0FBQyxjQUFjLENBQUMsQ0FBQztZQUM3QyxNQUFNLGNBQWMsR0FBRyxLQUFLLENBQUMsY0FBYyxDQUFDLENBQUM7WUFFN0MsTUFBTSxvQkFBb0IsR0FBRyxJQUFJLENBQUMsWUFBWSxDQUFDLFNBQVMsQ0FBQyxjQUFjLENBQUMsQ0FBQztZQUN6RSxNQUFNLG9CQUFvQixHQUFHLElBQUksQ0FBQyxZQUFZLENBQUMsU0FBUyxDQUFDLGNBQWMsQ0FBQyxDQUFDO1lBQ3pFLElBQUksb0JBQW9CLENBQUMsTUFBTSxDQUFDLG9CQUFvQixFQUFFLEtBQUssQ0FBQyxFQUFFO2dCQUMxRCxNQUFNLFlBQVksR0FBRyxLQUFLLENBQUMsWUFBWSxDQUFDLENBQUM7Z0JBQ3pDLE1BQU0sWUFBWSxHQUFHLEtBQUssQ0FBQyxZQUFZLENBQUMsQ0FBQztnQkFDekMsTUFBTSxRQUFRLEdBQUcsb0JBQW9CLENBQUMsSUFBSSxDQUFDLFlBQVksRUFBRSxLQUFLLENBQUMsQ0FBQztnQkFDaEUsTUFBTSxRQUFRLEdBQUcsb0JBQW9CLENBQUMsSUFBSSxDQUFDLFlBQVksRUFBRSxLQUFLLENBQUMsQ0FBQztnQkFDaEUsSUFBSSxRQUFRLEdBQUcsUUFBUSxFQUFFO29CQUNyQixPQUFPLENBQUMsQ0FBQztpQkFDWjtnQkFDRCxJQUFJLFFBQVEsR0FBRyxRQUFRLEVBQUU7b0JBQ3JCLE9BQU8sQ0FBQyxDQUFDLENBQUM7aUJBQ2I7Z0JBQ0QsT0FBTyxDQUFDLENBQUM7YUFDWjtZQUNEO2dCQUNJLE9BQU8sQ0FBQyxDQUFDO2FBQ1o7UUFDTCxDQUFDLENBQUMsQ0FBQztRQUNILE9BQU8sYUFBYSxDQUFDO0lBQ3pCLENBQUM7SUFDRCxjQUFjLENBQ1YsSUFBVyxFQUNYLFVBQWlDLEVBQ2pDLGNBQThCLEVBQzlCLFVBQWtCO1FBRWxCLE1BQU0sSUFBSSxHQUE2QixFQUFFLENBQUM7UUFDMUMsSUFBSSxDQUFDLE9BQU8sQ0FBQyxDQUFDLEdBQUcsRUFBRSxRQUFRLEVBQUUsRUFBRTtZQUMzQixJQUFJLE9BQU8sR0FBRyxDQUFDLENBQUM7WUFDaEIsVUFBVSxDQUFDLE9BQU8sQ0FBQyxDQUFDLEtBQUssRUFBRSxFQUFFO2dCQUN6QixNQUFNLFlBQVksR0FBRyxPQUFPLENBQUMsUUFBUSxFQUFFLENBQUM7Z0JBQ3hDLElBQUksQ0FBQyxJQUFJLENBQUMsWUFBWSxDQUFDLEVBQUU7b0JBQ3JCLElBQUksQ0FBQyxZQUFZLENBQUMsR0FBRyxFQUFFLENBQUM7aUJBQzNCO3FCQUFNLElBQUksSUFBSSxDQUFDLFlBQVksQ0FBQyxDQUFDLE1BQU0sR0FBRyxRQUFRLEVBQUU7b0JBQzdDLE9BQU8sRUFBRSxDQUFDO29CQUNWLE9BQU87aUJBQ1Y7Z0JBQ0QsS0FBSyxJQUFJLENBQUMsR0FBRyxJQUFJLENBQUMsWUFBWSxDQUFDLENBQUMsTUFBTSxFQUFFLENBQUMsR0FBRyxRQUFRLEVBQUUsQ0FBQyxFQUFFLEVBQUU7b0JBQ3ZELElBQUksQ0FBQyxZQUFZLENBQUMsQ0FBQyxJQUFJLENBQUMsRUFBRSxPQUFPLEVBQUUsQ0FBQyxFQUFFLENBQUMsQ0FBQztpQkFDM0M7Z0JBRUQsb0RBQW9EO2dCQUNwRCxNQUFNLE9BQU8sR0FBUSxFQUFFLE9BQU8sRUFBRSxDQUFDLEVBQUUsQ0FBQztnQkFDcEMsTUFBTSxTQUFTLEdBQUcsSUFBSSxDQUFDLFlBQVksQ0FBQyxTQUFTLENBQUMsR0FBRyxDQUFDLElBQUksQ0FBQyxDQUFDO2dCQUN4RCxNQUFNLGtCQUFrQixHQUFHLElBQUksQ0FBQyxZQUFZLENBQUMsU0FBUyxDQUFDLEtBQUssQ0FBQyxjQUFjLENBQUMsU0FBUyxDQUFDLENBQUMsQ0FBQztnQkFDeEYsTUFBTSxPQUFPLEdBQUcsS0FBSyxDQUFDLGNBQWMsQ0FBQyxPQUFPLENBQUMsQ0FBQztnQkFDOUMsTUFBTSxnQkFBZ0IsR0FBRyxJQUFJLENBQUMsWUFBWSxDQUFDLFNBQVMsQ0FBQyxPQUFPLENBQUMsQ0FBQztnQkFDOUQsTUFBTSxJQUFJLEdBQUcsSUFBSSxDQUFDLEdBQUcsQ0FBQyxrQkFBa0IsQ0FBQyxJQUFJLENBQUMsZ0JBQWdCLEVBQUUsS0FBSyxDQUFDLENBQUMsQ0FBQztnQkFDeEUsSUFBSSxJQUFJLEtBQUssQ0FBQyxJQUFJLFNBQVMsQ0FBQyxNQUFNLENBQUMsa0JBQWtCLEVBQUUsS0FBSyxDQUFDLEVBQUU7b0JBQzNELE9BQU8sQ0FBQyxPQUFPLEdBQUcsQ0FBQyxDQUFDO29CQUNwQixNQUFNLGNBQWMsR0FBRyxDQUFDLEtBQUssQ0FBQyxjQUFjLENBQUMsT0FBTyxHQUFHLFVBQVUsQ0FBQyxHQUFHLEVBQUUsQ0FBQztvQkFDeEUsT0FBTyxDQUFDLEdBQUcsR0FBRyxLQUFLLENBQUMsY0FBYyxDQUFDLElBQUksR0FBRyxVQUFVLEdBQUcsY0FBYyxDQUFDO29CQUN0RSxNQUFNLFlBQVksR0FBRyxDQUFDLEtBQUssQ0FBQyxZQUFZLENBQUMsT0FBTyxHQUFHLFVBQVUsQ0FBQyxHQUFHLEVBQUUsQ0FBQztvQkFDcEUsT0FBTyxDQUFDLE1BQU07d0JBQ1YsQ0FBQyxLQUFLLENBQUMsWUFBWSxDQUFDLElBQUksR0FBRyxLQUFLLENBQUMsY0FBYyxDQUFDLElBQUksQ0FBQyxHQUFHLFVBQVU7NEJBQ2xFLFlBQVk7NEJBQ1osY0FBYyxDQUFDO29CQUNuQixPQUFPLENBQUMsS0FBSyxHQUFHLEtBQUssQ0FBQztvQkFDdEIsT0FBTyxDQUFDLGFBQWEsR0FBRyxJQUFJLENBQUM7b0JBQzdCLE9BQU8sQ0FBQyxlQUFlLEdBQUcsSUFBSSxDQUFDO29CQUMvQixPQUFPLENBQUMsVUFBVSxHQUFHLElBQUksQ0FBQztvQkFDMUIsT0FBTyxDQUFDLFFBQVEsR0FBRyxJQUFJLENBQUM7b0JBQ3hCLE9BQU8sQ0FBQyxLQUFLLEdBQUcsT0FBTyxDQUFDLEdBQUcsR0FBRyxHQUFHLENBQUMsSUFBSSxDQUFDO29CQUN2QyxPQUFPLENBQUMsS0FBSzt3QkFDVCxLQUFLLENBQUMsY0FBYyxDQUFDLElBQUksR0FBRyxDQUFDOzRCQUM3QixLQUFLLENBQUMsY0FBYyxDQUFDLE9BQU8sR0FBRyxDQUFDOzRCQUNoQyxLQUFLLENBQUMsWUFBWSxDQUFDLElBQUksR0FBRyxDQUFDOzRCQUMzQixLQUFLLENBQUMsWUFBWSxDQUFDLE9BQU8sR0FBRyxDQUFDLENBQUM7b0JBQ25DLElBQUksQ0FBQyxZQUFZLENBQUMsQ0FBQyxNQUFNLENBQUMsUUFBUSxFQUFFLENBQUMsRUFBRSxPQUFPLENBQUMsQ0FBQztvQkFDaEQsR0FBRyxDQUFDLEtBQUssQ0FBQyxJQUFJLENBQUMsT0FBTyxDQUFDLENBQUM7b0JBQ3hCLE9BQU8sRUFBRSxDQUFDO2lCQUNiO3FCQUFNO29CQUNILElBQUksa0JBQWtCLEdBQUcsQ0FBQyxDQUFDO29CQUMzQixJQUFJLG9CQUFvQixHQUFHLENBQUMsQ0FBQztvQkFDN0IsTUFBTSxTQUFTLEdBQ1gsU0FBUyxDQUFDLGFBQWEsQ0FBQyxrQkFBa0IsRUFBRSxLQUFLLENBQUM7d0JBQ2xELFNBQVMsQ0FBQyxjQUFjLENBQUMsZ0JBQWdCLEVBQUUsS0FBSyxDQUFDLENBQUM7b0JBQ3RELElBQUksU0FBUyxFQUFFO3dCQUNYLGtCQUFrQixHQUFHLENBQUMsQ0FBQzt3QkFDdkIsSUFBSSxJQUFJLEtBQUssQ0FBQyxFQUFFOzRCQUNaLE9BQU8sQ0FBQyxPQUFPLEdBQUcsQ0FBQyxDQUFDOzRCQUNwQixPQUFPLENBQUMsS0FBSyxHQUFHLEtBQUssQ0FBQzt5QkFDekI7NkJBQU07NEJBQ0gsa0JBQWtCLEdBQUcsSUFBSSxDQUFDLEdBQUcsQ0FBQyxTQUFTLENBQUMsSUFBSSxDQUFDLGdCQUFnQixFQUFFLEtBQUssQ0FBQyxDQUFDLENBQUM7NEJBQ3ZFLG9CQUFvQixHQUFHLElBQUksQ0FBQyxHQUFHLENBQUMsU0FBUyxDQUFDLElBQUksQ0FBQyxrQkFBa0IsRUFBRSxLQUFLLENBQUMsQ0FBQyxDQUFDOzRCQUMzRSxNQUFNLFVBQVUsR0FBRyxTQUFTLENBQUMsTUFBTSxDQUFDLGtCQUFrQixFQUFFLEtBQUssQ0FBQyxDQUFDOzRCQUMvRCxNQUFNLFFBQVEsR0FBRyxTQUFTLENBQUMsTUFBTSxDQUFDLGdCQUFnQixFQUFFLEtBQUssQ0FBQyxDQUFDOzRCQUMzRCxJQUFJLE9BQU8sR0FBRyxJQUFJLENBQUM7NEJBQ25CLElBQUksQ0FBQyxVQUFVLEVBQUU7Z0NBQ2IsT0FBTyxHQUFHLGtCQUFrQixDQUFDOzZCQUNoQztpQ0FBTTtnQ0FDSCxPQUFPLEdBQUcsSUFBSSxHQUFHLENBQUMsQ0FBQyxDQUFDLG9CQUFvQjs2QkFDM0M7NEJBQ0QsT0FBTyxDQUFDLE9BQU8sR0FBRyxJQUFJLENBQUMsWUFBWSxDQUFDLE9BQU8sRUFBRSxRQUFRLEVBQUUsSUFBSSxDQUFDLE1BQU0sQ0FBQyxDQUFDOzRCQUNwRSxPQUFPLENBQUMsVUFBVSxHQUFHLFVBQVUsQ0FBQzs0QkFDaEMsT0FBTyxDQUFDLFFBQVEsR0FBRyxRQUFRLENBQUM7NEJBQzVCLE9BQU8sQ0FBQyxRQUFRLEdBQUcsQ0FBQyxVQUFVLElBQUksQ0FBQyxRQUFRLENBQUM7NEJBQzVDLE9BQU8sQ0FBQyxLQUFLLEdBQUcsS0FBSyxDQUFDO3lCQUN6Qjt3QkFDRCxJQUFJLE9BQU8sQ0FBQyxVQUFVLElBQUksUUFBUSxLQUFLLENBQUMsRUFBRTs0QkFDdEMsS0FBSyxJQUFJLElBQUksR0FBRyxDQUFDLEVBQUUsSUFBSSxHQUFHLE9BQU8sQ0FBQyxPQUFPLEVBQUUsSUFBSSxFQUFFLEVBQUU7Z0NBQy9DLE1BQU0sU0FBUyxHQUFHLFFBQVEsR0FBRyxJQUFJLENBQUM7Z0NBQ2xDLElBQUksU0FBUyxHQUFHLElBQUksQ0FBQyxNQUFNLEdBQUcsQ0FBQyxFQUFFO29DQUM3QixNQUFNO2lDQUNUO2dDQUNELE1BQU0sV0FBVyxHQUFHLElBQUksQ0FBQyxTQUFTLENBQUMsQ0FBQztnQ0FFcEMsTUFBTSxRQUFRLEdBQUc7b0NBQ2IsR0FBRyxPQUFPO29DQUNWLEtBQUssRUFBRSxXQUFXLENBQUMsSUFBSTtvQ0FDdkIsR0FBRyxFQUFFLE9BQU87b0NBQ1osVUFBVSxFQUFFLElBQUksS0FBSyxDQUFDLENBQUMsQ0FBQyxDQUFDLE9BQU8sQ0FBQyxVQUFVLENBQUMsQ0FBQyxDQUFDLElBQUk7b0NBQ2xELFNBQVMsRUFBRSxLQUFLLENBQUMsYUFBYTtvQ0FDOUIsT0FBTyxFQUFFLEtBQUssQ0FBQyxXQUFXO29DQUMxQixlQUFlLEVBQUUsUUFBUSxHQUFHLG9CQUFvQixHQUFHLENBQUMsQ0FBQztvQ0FDckQsYUFBYSxFQUFFLENBQUMsQ0FBQyxJQUFJLENBQUMsa0JBQWtCLENBQUM7b0NBQ3pDLFFBQVEsRUFDSixJQUFJLEtBQUssQ0FBQzt3Q0FDTixDQUFDLENBQUMsT0FBTyxDQUFDLFFBQVE7d0NBQ2xCLENBQUMsQ0FBQyxJQUFJLENBQUMsWUFBWTs2Q0FDWixTQUFTLENBQUMsV0FBVyxDQUFDLElBQUksQ0FBQzs2Q0FDM0IsU0FBUyxDQUFDLGtCQUFrQixFQUFFLGdCQUFnQixFQUFFLEtBQUssQ0FBQztvQ0FDckUsUUFBUSxFQUNKLElBQUksS0FBSyxDQUFDO3dDQUNOLENBQUMsQ0FBQyxPQUFPLENBQUMsUUFBUTt3Q0FDbEIsQ0FBQyxDQUFDLElBQUksQ0FBQyxHQUFHLENBQ0osSUFBSSxDQUFDLFlBQVk7NkNBQ1osU0FBUyxDQUFDLFdBQVcsQ0FBQyxJQUFJLENBQUM7NkNBQzNCLElBQUksQ0FBQyxnQkFBZ0IsRUFBRSxLQUFLLENBQUMsQ0FDckMsSUFBSSxDQUFDO29DQUNoQixPQUFPLEVBQUUsSUFBSSxLQUFLLENBQUMsQ0FBQyxDQUFDLENBQUMsT0FBTyxDQUFDLE9BQU8sQ0FBQyxDQUFDLENBQUMsSUFBSTtpQ0FDL0MsQ0FBQztnQ0FDRixJQUFJLENBQUMsT0FBTyxDQUFDLFFBQVEsRUFBRSxDQUFDLENBQUMsTUFBTSxDQUFDLFNBQVMsRUFBRSxDQUFDLEVBQUUsUUFBUSxDQUFDLENBQUM7Z0NBQ3hELFdBQVcsQ0FBQyxLQUFLLENBQUMsSUFBSSxDQUFDLFFBQVEsQ0FBQyxDQUFDOzZCQUNwQzs0QkFDRCxPQUFPLEVBQUUsQ0FBQzt5QkFDYjtxQkFDSjtpQkFDSjtZQUNMLENBQUMsQ0FBQyxDQUFDO1FBQ1AsQ0FBQyxDQUFDLENBQUM7UUFDSCxPQUFPLElBQUksQ0FBQztJQUNoQixDQUFDO0lBRUQsUUFBUTtRQUNKLE1BQU0sSUFBSSxHQUFHLElBQUksQ0FBQyxXQUFXLENBQUMsUUFBUSxFQUFFLENBQUM7UUFDekMsTUFBTSxhQUFhLEdBQUcsSUFBSSxDQUFDLGVBQWUsQ0FBQyxRQUFRLEVBQUUsQ0FBQztRQUN0RCxRQUFRLElBQUksRUFBRTtZQUNWLEtBQUssYUFBYSxDQUFDO1lBQ25CLEtBQUssT0FBTztnQkFDUjtvQkFDSSxNQUFNLElBQUksR0FBRyxJQUFJLENBQUMsSUFBSSxDQUFDO29CQUN2QixNQUFNLE9BQU8sR0FBRyxJQUFJLENBQUMsY0FBYyxDQUFDLElBQUksRUFBRSxhQUFhLENBQUMsQ0FBQztvQkFDekQsSUFBSSxDQUFDLE9BQU8sQ0FBQyxPQUFPLENBQUMsQ0FBQztpQkFDekI7Z0JBRUQsTUFBTTtZQUNWLEtBQUssTUFBTSxDQUFDO1lBQ1osS0FBSyxLQUFLO2dCQUNOO29CQUNJLE1BQU0sSUFBSSxHQUFHLElBQUksQ0FBQyxtQkFBbUIsQ0FBQyxRQUFRLEVBQUUsQ0FBQztvQkFDakQsTUFBTSxPQUFPLEdBQUcsSUFBSSxDQUFDLGFBQWEsQ0FBQyxJQUFJLEVBQUUsYUFBYSxDQUFDLENBQUM7b0JBQ3hELElBQUksQ0FBQyxtQkFBbUIsQ0FBQyxJQUFJLENBQUMsT0FBTyxDQUFDLENBQUM7aUJBQzFDO2dCQUNELE1BQU07Z0JBQ04sTUFBTTtZQUNWLEtBQUssTUFBTTtnQkFDUCxNQUFNO1NBQ2I7SUFDTCxDQUFDO0lBQ0QsU0FBUyxDQUFDLFNBQWlCLEVBQUUsVUFBa0I7UUFDM0MsTUFBTSxhQUFhLEdBQUcsSUFBSSxDQUFDLGFBQWEsQ0FBQyxRQUFRLENBQUMsS0FBSyxDQUFDO1FBQ3hELE1BQU0sR0FBRyxHQUFHLFVBQVUsR0FBRyxhQUFhLENBQUM7UUFDdkMsSUFBSSxDQUFDLFNBQVMsQ0FBQyxJQUFJLENBQUMsYUFBYSxDQUFDLFFBQVEsRUFBRSxHQUFHLENBQUMsQ0FBQztJQUNyRCxDQUFDO0lBQ0QsUUFBUSxDQUFDLElBQVk7UUFDakIsTUFBTSxZQUFZLEdBQUcsSUFBSSxDQUFDLGFBQWEsQ0FBQyxRQUFRLENBQUMsSUFBSSxDQUFDO1FBQ3RELE1BQU0sR0FBRyxHQUFHLElBQUksR0FBRyxZQUFZLENBQUM7UUFDaEMsSUFBSSxDQUFDLFlBQVksQ0FBQyxHQUFHLENBQUMsQ0FBQztJQUMzQixDQUFDO0lBQ0QsWUFBWTtRQUNSLElBQUksQ0FBQyxZQUFZLENBQUMsQ0FBQyxDQUFDLENBQUM7SUFDekIsQ0FBQztJQUNELFlBQVk7UUFDUixJQUFJLENBQUMsWUFBWSxDQUFDLENBQUMsQ0FBQyxDQUFDLENBQUM7SUFDMUIsQ0FBQztJQUNELFlBQVksQ0FBQyxLQUFhO1FBQ3RCLE1BQU0sWUFBWSxHQUFHLElBQUksQ0FBQyxhQUFhLENBQUMsUUFBUSxDQUFDLE1BQU0sQ0FBQyxNQUFNLEVBQUUsQ0FBQztRQUNqRSxZQUFZLENBQUMsV0FBVyxDQUFDLFlBQVksQ0FBQyxXQUFXLEVBQUUsR0FBRyxLQUFLLENBQUMsQ0FBQyxDQUFDLFlBQVk7UUFDMUUsSUFBSSxDQUFDLGVBQWUsQ0FBQyxZQUFZLENBQUMsQ0FBQztJQUN2QyxDQUFDO0lBQ0QsWUFBWTtRQUNSLE1BQU0sSUFBSSxHQUFHLElBQUksQ0FBQyxXQUFXLENBQUMsUUFBUSxFQUFFLENBQUM7UUFDekMsTUFBTSxhQUFhLEdBQUcsSUFBSSxDQUFDLGVBQWUsQ0FBQyxRQUFRLEVBQUUsQ0FBQztRQUN0RCxRQUFRLElBQUksRUFBRTtZQUNWLEtBQUssYUFBYSxDQUFDO1lBQ25CLEtBQUssT0FBTztnQkFDUjtvQkFDSSxNQUFNLElBQUksR0FBRyxJQUFJLENBQUMsSUFBSSxDQUFDO29CQUN2QixNQUFNLE9BQU8sR0FBRyxJQUFJLENBQUMsY0FBYyxDQUFDLElBQUksRUFBRSxhQUFhLENBQUMsQ0FBQztvQkFDekQsSUFBSSxDQUFDLE9BQU8sQ0FBQyxPQUFPLENBQUMsQ0FBQztpQkFDekI7Z0JBQ0QsTUFBTTtZQUNWLEtBQUssTUFBTSxDQUFDO1lBQ1osS0FBSyxLQUFLO2dCQUNOO29CQUNJLE1BQU0sSUFBSSxHQUFHLElBQUksQ0FBQyxtQkFBbUIsQ0FBQyxRQUFRLEVBQUUsQ0FBQztvQkFDakQsTUFBTSxPQUFPLEdBQUcsSUFBSSxDQUFDLGFBQWEsQ0FBQyxJQUFJLEVBQUUsYUFBYSxDQUFDLENBQUM7b0JBQ3hELElBQUksQ0FBQyxtQkFBbUIsQ0FBQyxJQUFJLENBQUMsT0FBTyxDQUFDLENBQUM7aUJBQzFDO2dCQUNELE1BQU07Z0JBQ04sTUFBTTtZQUNWLEtBQUssTUFBTTtnQkFDUCxNQUFNO1NBQ2I7SUFDTCxDQUFDO0lBQ0QsU0FBUyxDQUFDLFFBQWtCLEVBQUUsV0FBbUIsQ0FBQztRQUM5QyxJQUFJLGVBQWUsR0FBRyxJQUFJLENBQUMsWUFBWSxDQUFDLFNBQVMsQ0FBQyxRQUFRLENBQUMsSUFBSSxDQUFDLENBQUM7UUFDakUsSUFBSSxRQUFRLEdBQUcsQ0FBQyxFQUFFO1lBQ2QsZUFBZSxHQUFHLElBQUksQ0FBQyxZQUFZLENBQUMsUUFBUSxDQUFDLGVBQWUsRUFBRSxRQUFRLENBQUMsQ0FBQztTQUMzRTtRQUNELElBQUksUUFBUSxHQUFHLENBQUMsRUFBRTtZQUNkLGVBQWUsR0FBRyxJQUFJLENBQUMsWUFBWSxDQUFDLGFBQWEsQ0FBQyxlQUFlLEVBQUUsSUFBSSxDQUFDLEdBQUcsQ0FBQyxRQUFRLENBQUMsQ0FBQyxDQUFDO1NBQzFGO1FBQ0QsSUFBSSx1QkFBdUIsR0FBRyxJQUFJLENBQUMsU0FBUyxDQUFDLElBQUksQ0FBQyxDQUFDLENBQUMsRUFBRSxFQUFFLENBQUMsQ0FBQyxDQUFDLFFBQVEsQ0FBQyxNQUFNLENBQUMsTUFBTSxDQUFDLGVBQWUsRUFBRSxLQUFLLENBQUMsQ0FBQyxDQUFDO1FBQzNHLElBQUksQ0FBQyx1QkFBdUIsRUFBRTtZQUMxQix1QkFBdUIsR0FBRyxJQUFJLENBQUMsWUFBWSxDQUFDLGVBQWUsRUFBRSxJQUFJLENBQUMsS0FBSyxDQUFDLENBQUM7U0FDNUU7UUFDRCxJQUFJLENBQUMscUJBQXFCLENBQUMsdUJBQXVCLENBQUMsQ0FBQztRQUNwRCxPQUFPLHVCQUF1QixDQUFDO0lBQ25DLENBQUM7SUFFRCxxQkFBcUIsQ0FBQyx1QkFBMEM7UUFDNUQsTUFBTSxhQUFhLEdBQUcsQ0FBQyxHQUFHLElBQUksQ0FBQyxTQUFTLENBQUMsQ0FBQztRQUMxQyxNQUFNLE1BQU0sR0FBRyxhQUFhLENBQUMsSUFBSSxDQUM3QixDQUFDLENBQUMsRUFBRSxFQUFFLENBQ0YsQ0FBQyxDQUFDLFFBQVEsQ0FBQyxJQUFJLEtBQUssdUJBQXVCLENBQUMsUUFBUSxDQUFDLElBQUk7WUFDekQsQ0FBQyxDQUFDLFFBQVEsQ0FBQyxLQUFLLEtBQUssdUJBQXVCLENBQUMsUUFBUSxDQUFDLEtBQUssQ0FDbEUsQ0FBQztRQUNGLElBQUksQ0FBQyxNQUFNLEVBQUU7WUFDVCxhQUFhLENBQUMsSUFBSSxDQUFDLHVCQUF1QixDQUFDLENBQUM7WUFDNUMsSUFBSSxDQUFDLGdCQUFnQixDQUFDLElBQUksQ0FBQyxhQUFhLENBQUMsQ0FBQztTQUM3QztRQUNELElBQUksQ0FBQyxXQUFXLENBQUMsdUJBQXVCLENBQUMsQ0FBQztJQUM5QyxDQUFDO0lBRUQsb0JBQW9CLENBQUMsaUJBQW9DO1FBQ3JELE1BQU0sT0FBTyxHQUFHLElBQUksQ0FBQyxTQUFTLENBQUM7UUFDL0IsT0FBTyxDQUFDLElBQUksQ0FBQyxpQkFBaUIsQ0FBQyxDQUFDO1FBQ2hDLElBQUksQ0FBQyxnQkFBZ0IsQ0FBQyxJQUFJLENBQUMsT0FBTyxDQUFDLENBQUM7SUFDeEMsQ0FBQztJQUVELFNBQVMsQ0FBQyxLQUFLO1FBQ1gsTUFBTSxJQUFJLEdBQUcsQ0FBQyxHQUFHLEVBQUUsR0FBRyxFQUFFLEdBQUcsRUFBRSxHQUFHLEVBQUUsR0FBRyxFQUFFLEdBQUcsRUFBRSxHQUFHLEVBQUUsR0FBRyxFQUFFLEdBQUcsRUFBRSxHQUFHLENBQUMsQ0FBQztRQUNoRSxNQUFNLE9BQU8sR0FBRyxDQUFDLEdBQUcsRUFBRSxHQUFHLEVBQUUsR0FBRyxFQUFFLEdBQUcsRUFBRSxHQUFHLEVBQUUsR0FBRyxFQUFFLEdBQUcsRUFBRSxHQUFHLEVBQUUsR0FBRyxFQUFFLEdBQUcsQ0FBQyxDQUFDO1FBQ25FLElBQUksYUFBYSxHQUFHLEtBQUssQ0FBQztRQUMxQixJQUFJLEtBQUssQ0FBQztRQUNWLEtBQUssSUFBSSxDQUFDLEdBQUcsQ0FBQyxFQUFFLENBQUMsR0FBRyxJQUFJLENBQUMsTUFBTSxFQUFFLENBQUMsRUFBRSxFQUFFO1lBQ2xDLEtBQUssR0FBRyxJQUFJLE1BQU0sQ0FBQyxJQUFJLENBQUMsQ0FBQyxDQUFDLEVBQUUsR0FBRyxDQUFDLENBQUM7WUFDakMsYUFBYSxHQUFHLGFBQWEsQ0FBQyxPQUFPLENBQUMsS0FBSyxFQUFFLE9BQU8sQ0FBQyxDQUFDLENBQUMsQ0FBQyxDQUFDO1NBQzVEO1FBQ0QsT0FBTyxNQUFNLENBQUMsYUFBYSxDQUFDLENBQUM7SUFDakMsQ0FBQztJQUNPLFVBQVU7UUFDZCxJQUFJLENBQUMsS0FBSyxHQUFHLElBQUksQ0FBQyxZQUFZLENBQUMsU0FBUyxDQUFDLElBQUksSUFBSSxFQUFFLENBQUMsQ0FBQztRQUNyRCxNQUFNLGNBQWMsR0FBRyxJQUFJLENBQUMsb0JBQW9CLENBQUMsSUFBSSxJQUFJLEVBQUUsQ0FBQyxDQUFDO1FBQzdELElBQUksQ0FBQyxRQUFRLEdBQUcsY0FBYyxDQUFDLElBQUksQ0FBQyxJQUFJLENBQUMsQ0FBQyxDQUFDLEVBQUUsRUFBRSxDQUFDLENBQUMsQ0FBQyxPQUFPLENBQUMsQ0FBQztRQUMzRCxJQUFJLENBQUMsZUFBZSxHQUFHLElBQUksZUFBZSxDQUFvQixjQUFjLENBQUMsQ0FBQztRQUM5RSxJQUFJLENBQUMsU0FBUyxHQUFHLElBQUksQ0FBQyxlQUFlLENBQUMsWUFBWSxFQUFFLENBQUM7UUFDckQsSUFBSSxDQUFDLGdCQUFnQixDQUFDLElBQUksQ0FBQyxDQUFDLGNBQWMsQ0FBQyxDQUFDLENBQUM7UUFDN0MsSUFBSSxDQUFDLE9BQU8sQ0FBQyxjQUFjLENBQUMsSUFBSSxDQUFDLENBQUM7UUFDbEMsSUFBSSxDQUFDLFdBQVcsQ0FBQyxjQUFjLENBQUMsQ0FBQztRQUNqQyxJQUFJLENBQUMsd0JBQXdCLENBQUMsSUFBSSxDQUFDLGNBQWMsQ0FBQyxRQUFRLENBQUMsS0FBSyxDQUFDLENBQUM7UUFDbEUsSUFBSSxDQUFDLGVBQWUsR0FBRyxjQUFjLENBQUM7UUFDdEMsSUFBSSxDQUFDLFVBQVUsQ0FBQyxJQUFJLENBQUMsU0FBUyxDQUFDLElBQUksQ0FBQyxXQUFXLENBQUMsQ0FBQyxDQUFDLFNBQVMsRUFBRSxDQUFDO0lBQ2xFLENBQUM7SUFDTyxZQUFZLENBQUMsSUFBWSxFQUFFLFFBQWdCLEVBQUUsVUFBa0I7UUFDbkUsSUFBSSxJQUFJLEtBQUssQ0FBQyxFQUFFO1lBQ1osT0FBTyxDQUFDLENBQUM7U0FDWjtRQUNELElBQUksSUFBSSxDQUFDLEtBQUssQ0FBQyxJQUFJLEdBQUcsVUFBVSxDQUFDLEdBQUcsQ0FBQyxFQUFFO1lBQ25DLE9BQU8sVUFBVSxDQUFDO1NBQ3JCO1FBRUQsSUFBSSxRQUFRLEdBQUcsSUFBSSxHQUFHLFVBQVUsRUFBRTtZQUM5QixPQUFPLFVBQVUsR0FBRyxRQUFRLENBQUM7U0FDaEM7UUFDRCxPQUFPLElBQUksQ0FBQztJQUNoQixDQUFDO0lBRU8sY0FBYyxDQUFDLElBQVcsRUFBRSxTQUE0QjtRQUM1RCxNQUFNLFNBQVMsR0FBRyxJQUFJLENBQUMsTUFBTSxDQUFDO1FBQzlCLElBQUksT0FBTyxHQUFHLElBQUksQ0FBQyxTQUFTLEdBQUcsQ0FBQyxDQUFDLENBQUM7UUFDbEMsTUFBTSxZQUFZLEdBQUcsU0FBUyxDQUFDLElBQUksQ0FBQyxTQUFTLENBQ3pDLENBQUMsQ0FBQyxFQUFFLEVBQUUsQ0FBQyxDQUFDLENBQUMsVUFBVSxLQUFLLE9BQU8sQ0FBQyxVQUFVLElBQUksQ0FBQyxDQUFDLEtBQUssS0FBSyxPQUFPLENBQUMsS0FBSyxDQUMxRSxDQUFDO1FBQ0YsSUFBSSxPQUFPLEdBQUcsWUFBWSxHQUFHLENBQUMsQ0FBQyxDQUFDLENBQUMsQ0FBQyxTQUFTLENBQUMsSUFBSSxDQUFDLEtBQUssQ0FBQyxZQUFZLEdBQUcsQ0FBQyxFQUFFLFlBQVksR0FBRyxDQUFDLEdBQUcsU0FBUyxDQUFDLENBQUMsQ0FBQyxDQUFDLEVBQUUsQ0FBQztRQUM1RyxJQUFJLE9BQU8sQ0FBQyxNQUFNLEdBQUcsU0FBUyxFQUFFO1lBQzVCLHdCQUF3QjtZQUN4QixJQUFJLE9BQU8sQ0FBQyxNQUFNLEdBQUcsQ0FBQyxFQUFFO2dCQUNwQixPQUFPLEdBQUcsT0FBTyxDQUFDLE9BQU8sQ0FBQyxNQUFNLEdBQUcsQ0FBQyxDQUFDLENBQUM7YUFDekM7WUFDRCxNQUFNLFlBQVksR0FBRyxTQUFTLEdBQUcsT0FBTyxDQUFDLE1BQU0sQ0FBQztZQUNoRCxNQUFNLFNBQVMsR0FBRyxJQUFJLENBQUMsU0FBUyxDQUFDLFNBQVMsQ0FBQyxRQUFRLEVBQUUsQ0FBQyxDQUFDLENBQUM7WUFDeEQsSUFBSSxhQUFhLEdBQUcsQ0FBQyxDQUFDO1lBQ3RCLElBQUksWUFBWSxHQUFHLEVBQUUsRUFBRTtnQkFDbkIsYUFBYSxHQUFHLFNBQVMsQ0FBQyxJQUFJLENBQUMsU0FBUyxDQUNwQyxDQUFDLENBQUMsRUFBRSxFQUFFLENBQUMsQ0FBQyxDQUFDLFVBQVUsS0FBSyxPQUFPLENBQUMsVUFBVSxJQUFJLENBQUMsQ0FBQyxLQUFLLEtBQUssT0FBTyxDQUFDLEtBQUssQ0FDMUUsQ0FBQztnQkFDRixJQUFJLGFBQWEsS0FBSyxDQUFDLENBQUMsRUFBRTtvQkFDdEIsYUFBYSxHQUFHLENBQUMsQ0FBQztpQkFDckI7cUJBQU07b0JBQ0gsYUFBYSxFQUFFLENBQUM7aUJBQ25CO2FBQ0o7WUFDRCxNQUFNLFFBQVEsR0FBRyxTQUFTLENBQUMsSUFBSSxDQUFDLEtBQUssQ0FBQyxhQUFhLEVBQUUsYUFBYSxHQUFHLFlBQVksQ0FBQyxDQUFDO1lBQ25GLE9BQU8sR0FBRyxDQUFDLEdBQUcsT0FBTyxFQUFFLEdBQUcsUUFBUSxDQUFDLENBQUM7U0FDdkM7UUFDRCxPQUFPLE9BQU8sQ0FBQztJQUNuQixDQUFDO0lBQ08sYUFBYSxDQUFDLElBQVcsRUFBRSxTQUE0QjtRQUMzRCxNQUFNLFNBQVMsR0FBRyxJQUFJLENBQUMsTUFBTSxDQUFDO1FBQzlCLE1BQU0sT0FBTyxHQUFHLElBQUksQ0FBQyxTQUFTLEdBQUcsQ0FBQyxDQUFDLENBQUM7UUFDcEMsTUFBTSxZQUFZLEdBQUcsU0FBUyxDQUFDLElBQUksQ0FBQyxTQUFTLENBQ3pDLENBQUMsQ0FBQyxFQUFFLEVBQUUsQ0FBQyxDQUFDLENBQUMsVUFBVSxLQUFLLE9BQU8sQ0FBQyxVQUFVLElBQUksQ0FBQyxDQUFDLEtBQUssS0FBSyxPQUFPLENBQUMsS0FBSyxDQUMxRSxDQUFDO1FBQ0YsSUFBSSxPQUFPLEdBQUcsU0FBUyxDQUFDLElBQUksQ0FBQyxLQUFLLENBQUMsWUFBWSxHQUFHLENBQUMsRUFBRSxZQUFZLEdBQUcsQ0FBQyxHQUFHLFNBQVMsQ0FBQyxDQUFDO1FBQ25GLElBQUksT0FBTyxDQUFDLE1BQU0sR0FBRyxTQUFTLEVBQUU7WUFDNUIsd0JBQXdCO1lBQ3hCLElBQUksVUFBVSxHQUFHLE9BQU8sQ0FBQyxPQUFPLENBQUMsTUFBTSxHQUFHLENBQUMsQ0FBQyxDQUFDO1lBQzdDLElBQUksQ0FBQyxVQUFVLEVBQUU7Z0JBQ2IsVUFBVSxHQUFHLE9BQU8sQ0FBQzthQUN4QjtZQUNELE1BQU0sU0FBUyxHQUFHLElBQUksQ0FBQyxTQUFTLENBQUMsU0FBUyxDQUFDLFFBQVEsRUFBRSxDQUFDLENBQUMsQ0FBQztZQUN4RCxNQUFNLGFBQWEsR0FBRyxTQUFTLENBQUMsSUFBSSxDQUFDLFNBQVMsQ0FDMUMsQ0FBQyxDQUFDLEVBQUUsRUFBRSxDQUFDLENBQUMsQ0FBQyxVQUFVLEtBQUssVUFBVSxDQUFDLFVBQVUsSUFBSSxDQUFDLENBQUMsS0FBSyxLQUFLLFVBQVUsQ0FBQyxLQUFLLENBQ2hGLENBQUM7WUFDRixPQUFPLEdBQUcsT0FBTyxDQUFDLE1BQU0sQ0FDcEIsU0FBUyxDQUFDLElBQUksQ0FBQyxLQUFLLENBQUMsYUFBYSxHQUFHLENBQUMsRUFBRSxhQUFhLEdBQUcsQ0FBQyxHQUFHLENBQUMsU0FBUyxHQUFHLE9BQU8sQ0FBQyxNQUFNLENBQUMsQ0FBQyxDQUM1RixDQUFDO1NBQ0w7UUFDRCxPQUFPLE9BQU8sQ0FBQztJQUNuQixDQUFDO0lBRU8sY0FBYyxDQUFDLElBQVcsRUFBRSxTQUE0QjtRQUM1RCxNQUFNLFNBQVMsR0FBRyxJQUFJLENBQUMsTUFBTSxDQUFDO1FBQzlCLElBQUksUUFBUSxHQUFHLElBQUksQ0FBQyxDQUFDLENBQUMsQ0FBQztRQUN2QixNQUFNLGFBQWEsR0FBRyxTQUFTLENBQUMsSUFBSSxDQUFDLFNBQVMsQ0FDMUMsQ0FBQyxDQUFDLEVBQUUsRUFBRSxDQUFDLENBQUMsQ0FBQyxVQUFVLEtBQUssUUFBUSxDQUFDLFVBQVUsSUFBSSxDQUFDLENBQUMsS0FBSyxLQUFLLFFBQVEsQ0FBQyxLQUFLLENBQzVFLENBQUM7UUFDRixJQUFJLE9BQU8sR0FBVSxTQUFTLENBQUMsSUFBSSxDQUFDLEtBQUssQ0FDckMsYUFBYSxHQUFHLFNBQVMsR0FBRyxDQUFDLENBQUMsQ0FBQyxDQUFDLGFBQWEsR0FBRyxTQUFTLENBQUMsQ0FBQyxDQUFDLENBQUMsRUFDN0QsYUFBYSxDQUNoQixDQUFDO1FBQ0YsSUFBSSxPQUFPLENBQUMsTUFBTSxHQUFHLFNBQVMsRUFBRTtZQUM1Qix3QkFBd0I7WUFDeEIsSUFBSSxPQUFPLENBQUMsTUFBTSxHQUFHLENBQUMsRUFBRTtnQkFDcEIsUUFBUSxHQUFHLE9BQU8sQ0FBQyxDQUFDLENBQUMsQ0FBQzthQUN6QjtZQUNELE1BQU0sWUFBWSxHQUFHLFNBQVMsR0FBRyxPQUFPLENBQUMsTUFBTSxDQUFDO1lBQ2hELE1BQU0sU0FBUyxHQUFHLElBQUksQ0FBQyxTQUFTLENBQUMsU0FBUyxDQUFDLFFBQVEsRUFBRSxDQUFDLENBQUMsQ0FBQyxDQUFDO1lBQ3pELElBQUksV0FBVyxHQUFHLFNBQVMsQ0FBQyxJQUFJLENBQUMsTUFBTSxDQUFDO1lBQ3hDLElBQUksWUFBWSxHQUFHLEVBQUUsRUFBRTtnQkFDbkIsV0FBVyxHQUFHLFNBQVMsQ0FBQyxJQUFJLENBQUMsU0FBUyxDQUNsQyxDQUFDLENBQUMsRUFBRSxFQUFFLENBQUMsQ0FBQyxDQUFDLFVBQVUsS0FBSyxRQUFRLENBQUMsVUFBVSxJQUFJLENBQUMsQ0FBQyxLQUFLLEtBQUssUUFBUSxDQUFDLEtBQUssQ0FDNUUsQ0FBQztnQkFDRixJQUFJLFdBQVcsS0FBSyxDQUFDLENBQUMsRUFBRTtvQkFDcEIsV0FBVyxHQUFHLFNBQVMsQ0FBQyxJQUFJLENBQUMsTUFBTSxDQUFDO2lCQUN2QzthQUNKO1lBQ0QsTUFBTSxRQUFRLEdBQUcsU0FBUyxDQUFDLElBQUksQ0FBQyxLQUFLLENBQUMsV0FBVyxHQUFHLFlBQVksRUFBRSxXQUFXLENBQUMsQ0FBQztZQUMvRSxPQUFPLEdBQUcsQ0FBQyxHQUFHLFFBQVEsRUFBRSxHQUFHLE9BQU8sQ0FBQyxDQUFDO1NBQ3ZDO1FBQ0QsT0FBTyxPQUFPLENBQUM7SUFDbkIsQ0FBQztJQUNPLGFBQWEsQ0FBQyxJQUFXLEVBQUUsU0FBNEI7UUFDM0QsTUFBTSxTQUFTLEdBQUcsSUFBSSxDQUFDLE1BQU0sQ0FBQztRQUM5QixNQUFNLFFBQVEsR0FBRyxJQUFJLENBQUMsQ0FBQyxDQUFDLENBQUM7UUFDekIsTUFBTSxhQUFhLEdBQUcsU0FBUyxDQUFDLElBQUksQ0FBQyxTQUFTLENBQzFDLENBQUMsQ0FBQyxFQUFFLEVBQUUsQ0FBQyxDQUFDLENBQUMsVUFBVSxLQUFLLFFBQVEsQ0FBQyxVQUFVLElBQUksQ0FBQyxDQUFDLEtBQUssS0FBSyxRQUFRLENBQUMsS0FBSyxDQUM1RSxDQUFDO1FBQ0YsTUFBTSxhQUFhLEdBQUcsYUFBYSxHQUFHLFNBQVMsQ0FBQztRQUNoRCxJQUFJLE9BQU8sR0FBVSxFQUFFLENBQUM7UUFDeEIsSUFBSSxhQUFhLEdBQUcsQ0FBQyxFQUFFO1lBQ25CLE9BQU8sR0FBRyxhQUFhLEdBQUcsQ0FBQyxJQUFJLENBQUMsQ0FBQyxDQUFDLENBQUMsU0FBUyxDQUFDLElBQUksQ0FBQyxLQUFLLENBQUMsQ0FBQyxFQUFFLGFBQWEsR0FBRyxDQUFDLENBQUMsQ0FBQyxDQUFDLENBQUMsRUFBRSxDQUFDO1NBQ3RGO2FBQU07WUFDSCxPQUFPLEdBQUcsU0FBUyxDQUFDLElBQUksQ0FBQyxLQUFLLENBQUMsYUFBYSxHQUFHLFNBQVMsRUFBRSxhQUFhLENBQUMsQ0FBQztTQUM1RTtRQUNELElBQUksT0FBTyxDQUFDLE1BQU0sR0FBRyxTQUFTLEVBQUU7WUFDNUIsd0JBQXdCO1lBQ3hCLElBQUksV0FBVyxHQUFHLE9BQU8sQ0FBQyxDQUFDLENBQUMsQ0FBQztZQUM3QixJQUFJLENBQUMsV0FBVyxFQUFFO2dCQUNkLFdBQVcsR0FBRyxRQUFRLENBQUM7YUFDMUI7WUFDRCxNQUFNLFNBQVMsR0FBRyxJQUFJLENBQUMsU0FBUyxDQUFDLFNBQVMsQ0FBQyxRQUFRLEVBQUUsQ0FBQyxDQUFDLENBQUMsQ0FBQztZQUN6RCxNQUFNLGNBQWMsR0FBRyxTQUFTLENBQUMsSUFBSSxDQUFDLFNBQVMsQ0FDM0MsQ0FBQyxDQUFDLEVBQUUsRUFBRSxDQUFDLENBQUMsQ0FBQyxVQUFVLEtBQUssV0FBVyxDQUFDLFVBQVUsSUFBSSxDQUFDLENBQUMsS0FBSyxLQUFLLFdBQVcsQ0FBQyxLQUFLLENBQ2xGLENBQUM7WUFDRixPQUFPLEdBQUcsT0FBTyxDQUFDLE1BQU0sQ0FDcEIsU0FBUyxDQUFDLElBQUksQ0FBQyxLQUFLLENBQUMsY0FBYyxHQUFHLENBQUMsU0FBUyxHQUFHLE9BQU8sQ0FBQyxNQUFNLENBQUMsRUFBRSxjQUFjLENBQUMsQ0FDdEYsQ0FBQztTQUNMO1FBQ0QsT0FBTyxPQUFPLENBQUM7SUFDbkIsQ0FBQztJQUVPLG9CQUFvQixDQUFDLElBQUk7UUFDN0IsTUFBTSxZQUFZLEdBQUcsSUFBSSxDQUFDLFlBQVksQ0FBQyxZQUFZLENBQUMsSUFBSSxDQUFDLENBQUM7UUFDMUQsTUFBTSxVQUFVLEdBQUcsSUFBSSxDQUFDLFlBQVksQ0FBQyxTQUFTLENBQUMsSUFBSSxDQUFDLENBQUM7UUFDckQsT0FBTyxJQUFJLENBQUMsWUFBWSxDQUFDLFlBQVksRUFBRSxVQUFVLENBQUMsQ0FBQztJQUN2RCxDQUFDO0lBQ08sWUFBWSxDQUFDLElBQUksRUFBRSxLQUFLO1FBQzVCLE1BQU0sSUFBSSxHQUFVLEVBQUUsQ0FBQztRQUN2QixNQUFNLFVBQVUsR0FBRyxJQUFJLENBQUMsWUFBWSxDQUFDLFNBQVMsQ0FBQyxJQUFJLENBQUMsQ0FBQztRQUNyRCxNQUFNLFlBQVksR0FBUyxJQUFJLENBQUMsWUFBWSxDQUFDLFlBQVksQ0FBQyxVQUFVLENBQUMsQ0FBQztRQUN0RSxNQUFNLFVBQVUsR0FBUyxJQUFJLENBQUMsWUFBWSxDQUFDLFVBQVUsQ0FBQyxVQUFVLENBQUMsQ0FBQztRQUNsRSxNQUFNLFFBQVEsR0FBRyxJQUFJLENBQUMsWUFBWSxDQUFDLFdBQVcsQ0FBQyxVQUFVLENBQUMsQ0FBQztRQUMzRCxNQUFNLFNBQVMsR0FBRyxJQUFJLENBQUMsWUFBWSxDQUFDLEdBQUcsQ0FBQyxZQUFZLENBQUMsR0FBRyxDQUFDLENBQUMsQ0FBQywwREFBMEQ7UUFFckgsSUFBSSxXQUFXLEdBQUcsQ0FBQyxDQUFDO1FBQ3BCLElBQUksVUFBVSxHQUFHLENBQUMsQ0FBQztRQUNuQixLQUFLLElBQUksQ0FBQyxHQUFHLENBQUMsRUFBRSxDQUFDLEdBQUcsRUFBRSxFQUFFLENBQUMsRUFBRSxFQUFFO1lBQ3pCLE1BQU0sa0JBQWtCLEdBQVMsSUFBSSxJQUFJLENBQUMsWUFBWSxDQUFDLENBQUM7WUFDeEQsa0JBQWtCLENBQUMsT0FBTyxDQUFDLGtCQUFrQixDQUFDLE9BQU8sRUFBRSxHQUFHLENBQUMsR0FBRyxTQUFTLENBQUMsQ0FBQztZQUN6RSxNQUFNLEVBQUUsR0FBRyxFQUFFLFNBQVMsRUFBRSxHQUFpQyxJQUFJLENBQUMsT0FBTyxDQUNqRSxrQkFBa0IsRUFDbEIsVUFBVSxFQUNWLEtBQUssRUFDTCxVQUFVLEVBQ1YsWUFBWSxFQUNaLFFBQVEsRUFDUixDQUFDLENBQ0osQ0FBQztZQUNGLElBQUksQ0FBQyxJQUFJLENBQUMsR0FBRyxDQUFDLENBQUM7WUFDZixVQUFVLEVBQUUsQ0FBQztZQUNiLElBQUksU0FBUyxJQUFJLFVBQVUsR0FBRyxXQUFXLEdBQUcsQ0FBQyxFQUFFO2dCQUMzQyxXQUFXLEVBQUUsQ0FBQzthQUNqQjtTQUNKO1FBQ0QsaUVBQWlFO1FBQ2pFLG1FQUFtRTtRQUNuRSwwQ0FBMEM7UUFDMUMsSUFBSTtRQUNKLE1BQU0sWUFBWSxHQUFHLElBQUksaUJBQWlCLEVBQUUsQ0FBQztRQUM3QyxZQUFZLENBQUMsRUFBRSxHQUFHLFdBQVcsQ0FBQyxDQUFDLENBQUMsQ0FBQztRQUNqQyxZQUFZLENBQUMsUUFBUSxHQUFHLFFBQVEsQ0FBQztRQUNqQyxZQUFZLENBQUMsSUFBSSxHQUFHLElBQUksQ0FBQztRQUN6QixZQUFZLENBQUMsVUFBVSxHQUFHLFdBQVcsQ0FBQztRQUN0Qyw0QkFBNEI7UUFDNUIsT0FBTyxZQUFZLENBQUM7SUFDeEIsQ0FBQztJQUNPLE9BQU8sQ0FDWCxJQUFVLEVBQ1YsVUFBZSxFQUNmLEtBQVUsRUFDVixVQUFnQixFQUNoQixZQUFrQixFQUNsQixRQUFrQixFQUNsQixDQUFTO1FBRVQsTUFBTSxTQUFTLEdBQUcsSUFBSSxDQUFDLFlBQVksQ0FBQyxNQUFNLENBQUMsSUFBSSxFQUFFLFVBQVUsRUFBRSxPQUFPLENBQUMsQ0FBQztRQUN0RSxNQUFNLFlBQVksR0FBRyxJQUFJLENBQUMsWUFBWSxDQUFDLFNBQVMsQ0FBQyxJQUFJLENBQUMsQ0FBQztRQUN2RCxNQUFNLE9BQU8sR0FBRyxZQUFZLENBQUMsTUFBTSxDQUFDLEtBQUssRUFBRSxLQUFLLENBQUMsQ0FBQztRQUNsRCxNQUFNLFVBQVUsR0FBRyxJQUFJLENBQUMsWUFBWSxDQUFDLE9BQU8sQ0FBQyxJQUFJLENBQUMsQ0FBQztRQUNuRCxNQUFNLFNBQVMsR0FBRyxJQUFJLENBQUMsWUFBWSxDQUFDLEdBQUcsQ0FBQyxJQUFJLENBQUMsQ0FBQztRQUM5QyxNQUFNLFlBQVksR0FBRyxZQUFZLENBQUMsTUFBTSxDQUFDLFVBQVUsRUFBRSxLQUFLLENBQUMsQ0FBQztRQUM1RCxNQUFNLGNBQWMsR0FBRyxZQUFZLENBQUMsTUFBTSxDQUFDLFlBQVksRUFBRSxLQUFLLENBQUMsQ0FBQztRQUNoRSxNQUFNLGFBQWEsR0FBRyxTQUFTLENBQUMsQ0FBQyxDQUFDLFFBQVEsQ0FBQyxDQUFDLENBQUMsSUFBSSxDQUFDLFlBQVksQ0FBQyxXQUFXLENBQUMsSUFBSSxDQUFDLENBQUM7UUFDakYsTUFBTSxHQUFHLEdBQVE7WUFDYixLQUFLLEVBQUUsQ0FBQztZQUNSLElBQUk7WUFDSixTQUFTO1lBQ1QsTUFBTSxFQUFFLFNBQVMsQ0FBQyxJQUFJO1lBQ3RCLFVBQVU7WUFDVixXQUFXLEVBQUUsSUFBSSxDQUFDLFNBQVMsQ0FBQyxVQUFVLENBQUM7WUFDdkMsT0FBTztZQUNQLFNBQVM7WUFDVCxZQUFZO1lBQ1osY0FBYztZQUNkLFlBQVk7WUFDWixLQUFLLEVBQUUsYUFBYSxDQUFDLEtBQUs7WUFDMUIsSUFBSSxFQUFFLGFBQWEsQ0FBQyxJQUFJO1lBQ3hCLFNBQVMsRUFBRSxhQUFhLENBQUMsU0FBUztZQUNsQyxPQUFPLEVBQUUsWUFBWSxDQUFDLE1BQU0sQ0FBQyxNQUFNLENBQUM7WUFDcEMsS0FBSyxFQUFFLEVBQUU7WUFDVCxhQUFhO1lBQ2IsR0FBRyxFQUFFLElBQUksQ0FBQyxLQUFLLENBQUMsQ0FBQyxHQUFHLENBQUMsR0FBRyxDQUFDLENBQUM7WUFDMUIsR0FBRyxFQUFFLENBQUMsR0FBRyxDQUFDLENBQUMsR0FBRyxDQUFDLENBQUM7U0FDbkIsQ0FBQztRQUNGLE9BQU8sRUFBRSxHQUFHLEVBQUUsU0FBUyxFQUFFLENBQUM7SUFDOUIsQ0FBQzs7NEdBdnFCUSxlQUFlLDRDQW1CeUMsY0FBYztnSEFuQnRFLGVBQWU7MkZBQWYsZUFBZTtrQkFEM0IsVUFBVTs7MEJBb0J1QyxRQUFROzswQkFBSSxNQUFNOzJCQUFDLGNBQWMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBJbmplY3QsIEluamVjdGFibGUsIE9wdGlvbmFsIH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQgeyBPYnNlcnZhYmxlLCBCZWhhdmlvclN1YmplY3QgfSBmcm9tICdyeGpzJztcbmltcG9ydCB7IHRha2VVbnRpbCB9IGZyb20gJ3J4anMvb3BlcmF0b3JzJztcblxuaW1wb3J0IHtcbiAgICBEYXRlU2VydmljZSxcbiAgICBCYXNlQ29tcG9uZW50LFxuICAgIGdldFVuaXF1ZUlkLFxuICAgIGdldERhdGVTZXJ2aWNlLFxuICAgIExvZ1NlcnZpY2UsXG4gICAgRGF0ZUluZm8sXG4gICAgTWV0YW9iamVjdERhdGFNb2RlbCxcbiAgICBDYWxlbmRhckZpZWxkc1xufSBmcm9tICdiYXJzYS1ub3Zpbi1yYXktY29yZSc7XG5pbXBvcnQgeyBDYWxlbmRhck1vZGUsIENhbGVuZGFyTW9udGhJbmZvLCBEYXksIERheVN0YXR1cyB9IGZyb20gJy4uL21vZGVscyc7XG5cbkBJbmplY3RhYmxlKClcbmV4cG9ydCBjbGFzcyBDYWxlbmRhclNlcnZpY2UgZXh0ZW5kcyBCYXNlQ29tcG9uZW50IHtcbiAgICBzZWxlY3RlZCQ6IE9ic2VydmFibGU8Q2FsZW5kYXJNb250aEluZm8+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=