ngx-animating-datepicker
Version:
An Animating Datepicker for Angular 2+, for some smooth date picking :).
584 lines • 58.1 kB
JavaScript
/**
* @fileoverview added by tsickle
* @suppress {checkTypes,extraRequire,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
*/
import * as tslib_1 from "tslib";
import { Component, ElementRef, HostBinding, Input, ViewChild, } from '@angular/core';
import { DatepickerService } from '../../services/datepicker.service';
import { UtilitiesService } from '../../services/utilities.service';
import { DatepickerComponent } from '../datepicker/datepicker.component';
var AnimatepickerComponent = /** @class */ (function (_super) {
tslib_1.__extends(AnimatepickerComponent, _super);
function AnimatepickerComponent(elementRef, utilities) {
var _this = _super.call(this, utilities, elementRef) || this;
_this.elementRef = elementRef;
_this.utilities = utilities;
/* ==============================================
* Internal Properties
* ============================================== */
_this.animate = true;
_this.isAnimating = false;
_this.leftInnerPosition = 0;
_this.currentYearMonth = null;
_this.initialised = false;
/* ==============================================
* External Properties
* ============================================== */
/**
* Number of months: the number of months displayed
*/
_this._numberOfMonths = new Array(1);
return _this;
}
Object.defineProperty(AnimatepickerComponent.prototype, "numberOfMonths", {
get: /**
* @return {?}
*/
function () {
return this._numberOfMonths;
},
set: /**
* @param {?} value
* @return {?}
*/
function (value) {
if (value === undefined || value === this._numberOfMonths.length) {
return;
}
this._numberOfMonths = new Array(value);
this.setDatePickerDimension();
this.goToDate(this.date);
},
enumerable: true,
configurable: true
});
/**
* @return {?}
*/
AnimatepickerComponent.prototype.ngOnInit = /**
* @return {?}
*/
function () {
// Get the computed width from the calendar. Set the initial width
/** @type {?} */
var computedWidth = window
.getComputedStyle(this.elementRef.nativeElement, null)
.getPropertyValue('width');
this.initialWidth = parseInt(computedWidth, 10);
this.initialised = true;
// Set the current year and month object
if (!this.month && !this.year) {
this.goToDate(this.options.currentDate);
}
};
/**
* @return {?}
*/
AnimatepickerComponent.prototype.ngAfterViewInit = /**
* @return {?}
*/
function () {
var _this = this;
setTimeout(function () {
_this.setDatePickerDimension();
_this.setDatepickerHeight(true);
});
};
/**
* Set the height and the width properties
*/
/**
* Set the height and the width properties
* @return {?}
*/
AnimatepickerComponent.prototype.setDatePickerDimension = /**
* Set the height and the width properties
* @return {?}
*/
function () {
this.datepickerHeight =
this.calendarContainer.nativeElement.offsetHeight +
this.calendarTopContainer.nativeElement.offsetHeight +
this.footer.nativeElement.offsetHeight;
this.calendarHeight = this.calendarContainer.nativeElement.offsetHeight;
this.datepickerWidth = this.initialWidth * this._numberOfMonths.length;
};
/**
* Go to a specific month
*
* @param date - optional
*/
/**
* Go to a specific month
*
* @param {?=} date - optional
* @return {?}
*/
AnimatepickerComponent.prototype.goToDate = /**
* Go to a specific month
*
* @param {?=} date - optional
* @return {?}
*/
function (date) {
if (date) {
this.currentYearMonth = this.getNextYearMonthArray(date.getFullYear(), date.getMonth());
}
this.calendarWidth = 50 / this._numberOfMonths.length;
this.months = this.getNextMonthArray(this.currentYearMonth, true);
this.resetStyle();
};
/**
* Create an array of the next year and months
*
* @param year
* @param month
*/
/**
* Create an array of the next year and months
*
* @param {?} year
* @param {?} month
* @return {?}
*/
AnimatepickerComponent.prototype.getNextYearMonthArray = /**
* Create an array of the next year and months
*
* @param {?} year
* @param {?} month
* @return {?}
*/
function (year, month) {
/** @type {?} */
var array = [];
for (var index = 0; index < this._numberOfMonths.length; index++) {
array.push({ year: year, month: month });
year = DatepickerService.getYearOfNextMonth(year, month);
month = DatepickerService.getNextMonth(month);
}
return array;
};
/**
* Create an array of the previous year and months
*
* @param year
* @param month
*/
/**
* Create an array of the previous year and months
*
* @param {?} year
* @param {?} month
* @return {?}
*/
AnimatepickerComponent.prototype.getPreviousYearMonthArray = /**
* Create an array of the previous year and months
*
* @param {?} year
* @param {?} month
* @return {?}
*/
function (year, month) {
/** @type {?} */
var array = [];
for (var index = 0; index < this._numberOfMonths.length; index++) {
array.unshift({ year: year, month: month });
year = DatepickerService.getYearOfPreviousMonth(year, month);
month = DatepickerService.getPreviousMonth(month);
}
return array;
};
/**
* Set the datepicker height, used when animating
*
* @param directionRight - Set optional when sliding to the right
*/
/**
* Set the datepicker height, used when animating
*
* @param {?=} directionRight - Set optional when sliding to the right
* @return {?}
*/
AnimatepickerComponent.prototype.setDatepickerHeight = /**
* Set the datepicker height, used when animating
*
* @param {?=} directionRight - Set optional when sliding to the right
* @return {?}
*/
function (directionRight) {
/** @type {?} */
var indexArray;
// TODO: Seperate this logic for readability purpose
if (this._numberOfMonths.length > 1) {
/** @type {?} */
var start = directionRight ? 0 : this._numberOfMonths.length;
/** @type {?} */
var end = directionRight
? this._numberOfMonths.length - 1
: this._numberOfMonths.length + this._numberOfMonths.length - 1;
indexArray = this.utilities.createArray(start, end);
}
else {
indexArray = directionRight ? [0] : [1];
}
/** @type {?} */
var that = this;
setTimeout(function () {
/** @type {?} */
var calendarArray = that.elementRef.nativeElement.querySelectorAll('.datepicker__calendar-container');
/** @type {?} */
var offsetHeight = 0;
indexArray.forEach(function (el) {
if (calendarArray[el].offsetHeight > offsetHeight) {
offsetHeight = calendarArray[el].offsetHeight;
}
});
// TODO: Merge with setHeight function.
that.datepickerHeight =
offsetHeight + that.calendarTopContainer.nativeElement.offsetHeight + that.footer.nativeElement.offsetHeight;
that.calendarHeight = offsetHeight;
});
};
/**
* Get next month array, gets multiple months.
* Used when the options animate is set or multiple months are visable
*
* @return Month[]
*/
/**
* Get next month array, gets multiple months.
* Used when the options animate is set or multiple months are visable
*
* @param {?} currentYearMonth
* @param {?=} keepDate
* @param {?=} nextMonthsYearMonthArray
* @return {?} Month[]
*/
AnimatepickerComponent.prototype.getNextMonthArray = /**
* Get next month array, gets multiple months.
* Used when the options animate is set or multiple months are visable
*
* @param {?} currentYearMonth
* @param {?=} keepDate
* @param {?=} nextMonthsYearMonthArray
* @return {?} Month[]
*/
function (currentYearMonth, keepDate, nextMonthsYearMonthArray) {
var _this = this;
if (keepDate === void 0) { keepDate = false; }
// Get the last index, used for selecting the right year month object
/** @type {?} */
var lastIndex = this._numberOfMonths.length - 1;
// Get next year and month in an Object
/** @type {?} */
var nextMonths = nextMonthsYearMonthArray ||
this.getNextYearMonthArray(DatepickerService.getYearOfNextMonth(currentYearMonth[lastIndex].year, currentYearMonth[lastIndex].month), DatepickerService.getNextMonth(currentYearMonth[lastIndex].month));
// Concatenates the two objects to create a total year and month object
this.totalYearMonth = currentYearMonth.concat(nextMonths);
// Create the calendar array using the total year and month Object
/** @type {?} */
var monthArray = [];
this.totalYearMonth.forEach(function (e) {
return monthArray.push(_this.createCalendarArray(e.year, e.month));
});
// Set the new current year and month object.
if (!keepDate) {
this.currentYearMonth = nextMonths;
}
return [].concat.apply([], monthArray);
};
/**
* Gets an array of previous months.
* Used for animation and when more months are displayed
*
* @param currentYearMonth
* @param keepDate
*/
/**
* Gets an array of previous months.
* Used for animation and when more months are displayed
*
* @param {?} currentYearMonth
* @param {?=} keepDate
* @return {?}
*/
AnimatepickerComponent.prototype.getPreviousMonthArray = /**
* Gets an array of previous months.
* Used for animation and when more months are displayed
*
* @param {?} currentYearMonth
* @param {?=} keepDate
* @return {?}
*/
function (currentYearMonth, keepDate) {
var _this = this;
if (keepDate === void 0) { keepDate = false; }
// Get previous year and month in an Object
/** @type {?} */
var previousMonths = this.getPreviousYearMonthArray(DatepickerService.getYearOfPreviousMonth(currentYearMonth[0].year, currentYearMonth[0].month), DatepickerService.getPreviousMonth(currentYearMonth[0].month));
// Concatenates the two objects to create a total year and month object
this.totalYearMonth = previousMonths.concat(currentYearMonth);
// Create the calendar array using the total year and month Object
/** @type {?} */
var monthArray = [];
this.totalYearMonth.forEach(function (e) {
monthArray.push(_this.createCalendarArray(e['year'], e['month']));
});
// Set the new current year and month object.
if (!keepDate) {
this.currentYearMonth = previousMonths;
}
return [].concat.apply([], monthArray);
};
/**
* Update value is being triggered
*
* @param date
*/
/**
* Update value is being triggered
*
* @param {?} date
* @return {?}
*/
AnimatepickerComponent.prototype.updateValue = /**
* Update value is being triggered
*
* @param {?} date
* @return {?}
*/
function (date) {
if (this.options.range) {
this.selectRange(date);
}
else if (!this.isSelected(date)) {
if (this.options.selectMultiple) {
this.selectDate(date);
}
else {
this.toggleDate(date);
}
if (this.options.closeOnSelect) {
this.close(true);
}
}
else {
this.deselectDate(date);
if (this.options.closeOnSelect) {
this.close(true);
}
}
this.months = this.getNextMonthArray(this.currentYearMonth, true);
this.resetStyle();
};
/**
* Go to the next month
*/
/**
* Go to the next month
* @return {?}
*/
AnimatepickerComponent.prototype.goToNextMonth = /**
* Go to the next month
* @return {?}
*/
function () {
if (this.isAnimating) {
return;
}
this.months = this.getNextMonthArray(this.currentYearMonth);
this.resetStyle();
this.setDatepickerHeight();
this.slideLeft();
};
/**
* Go to the previous month
*/
/**
* Go to the previous month
* @return {?}
*/
AnimatepickerComponent.prototype.goToPreviousMonth = /**
* Go to the previous month
* @return {?}
*/
function () {
if (this.isAnimating) {
return;
}
this.months = this.getPreviousMonthArray(this.currentYearMonth);
this.resetStyle(true);
this.setDatepickerHeight(true);
this.slideRight();
};
/**
* Go to a specific month
* TODO: WIP Check if date is in current range, or if it is later or earlier
*/
/**
* Go to a specific month
* TODO: WIP Check if date is in current range, or if it is later or earlier
* @param {?} date
* @return {?}
*/
AnimatepickerComponent.prototype.goToMonth = /**
* Go to a specific month
* TODO: WIP Check if date is in current range, or if it is later or earlier
* @param {?} date
* @return {?}
*/
function (date) {
/** @type {?} */
var nextMonths = this.getNextYearMonthArray(date.getFullYear(), date.getMonth());
this.months = this.getNextMonthArray(this.totalYearMonth, false, nextMonths);
this.resetStyle();
this.setDatepickerHeight();
this.slideRight();
};
/**
* Slide to the right
*/
/**
* Slide to the right
* @return {?}
*/
AnimatepickerComponent.prototype.slideRight = /**
* Slide to the right
* @return {?}
*/
function () {
var _this = this;
this.setIsAnimating();
setTimeout(function () {
_this.transition =
'transform ' + _this.options.animationSpeed + 'ms ' + _this.options.easing;
_this.translateX = 50;
});
};
/**
* Slide to the left (criss cross)
*/
/**
* Slide to the left (criss cross)
* @return {?}
*/
AnimatepickerComponent.prototype.slideLeft = /**
* Slide to the left (criss cross)
* @return {?}
*/
function () {
var _this = this;
this.setIsAnimating();
setTimeout(function () {
_this.transition =
'transform ' + _this.options.animationSpeed + 'ms ' + _this.options.easing;
_this.translateX = -50;
});
};
/**
* Set animating state
*/
/**
* Set animating state
* @return {?}
*/
AnimatepickerComponent.prototype.setIsAnimating = /**
* Set animating state
* @return {?}
*/
function () {
var _this = this;
this.isAnimating = true;
setTimeout(function () {
_this.isAnimating = false;
}, this.options.animationSpeed);
};
/**
* Reset Style
*
* @param resetForPrevious - Optional
*/
/**
* Reset Style
*
* @param {?=} resetForPrevious - Optional
* @return {?}
*/
AnimatepickerComponent.prototype.resetStyle = /**
* Reset Style
*
* @param {?=} resetForPrevious - Optional
* @return {?}
*/
function (resetForPrevious) {
this.transition = 'transform 0ms ease-in';
this.translateX = 0;
this.leftInnerPosition = resetForPrevious ? -100 : 0;
};
AnimatepickerComponent.decorators = [
{ type: Component, args: [{
selector: 'aa-animatepicker',
template: "<div class=\"datepicker__wrapper\" [ngStyle]=\"datepickerPosition\">\n\t<div #calendarTopContainer>\n\t\t<div class=\"datepicker__header\" #header>\n\t\t\t<ng-content select=\"header\"></ng-content>\n\t\t</div>\n\t\t<aa-navigation\n\t\t\t\t(previousClick)=\"goToPreviousMonth()\"\n\t\t\t\t(nextClick)=\"goToNextMonth()\"\n\t\t\t\t(subNavigationClick)=\"goToDate($event)\"\n\t\t\t\t[language]=\"language\"\n\t\t\t\t[animate]=\"animate\"\n\t\t\t\t[translateX]=\"translateX\"\n\t\t\t\t[transition]=\"transition\"\n\t\t\t\t[leftPosition]=\"leftInnerPosition\"\n\t\t\t\t[hideNavigation]=\"options.hideNavigation\"\n\t\t\t\t[totalYearMonth]=\"totalYearMonth\"\n\t\t></aa-navigation>\n\n\t\t<div class=\"datepicker__weekdays-wrapper\">\n\t\t\t<div \t*ngFor=\"let month of numberOfMonths\"\n\t\t\t\t\t[ngStyle]=\"{\n\t\t\t\t'width.%': (100 / numberOfMonths.length)\n\t\t\t\t}\"\n\t\t\t\t\tclass=\"datepicker__weekdays-container\">\n\t\t\t\t<table class=\"datepicker__weekdays\">\n\t\t\t\t\t<thead>\n\t\t\t\t\t<td class=\"datepicker__weekday\" *ngFor=\"let weekday of weekdays; index as i\">{{weekday}}</td>\n\t\t\t\t\t</thead>\n\t\t\t\t</table>\n\t\t\t</div>\n\t\t</div>\n\t</div>\n\t<div class=\"datepicker__main\" [ngStyle]=\"{\n\t\t'height.px': calendarHeight}\">\n\t\t<div class=\"datepicker__calendar-wrapper\" #calendarContainer [ngStyle]=\"{\n\t\t\t\t\t'transform': 'translateX(' + translateX + '%)',\n\t\t\t\t\t'transition': transition,\n\t\t\t\t\t'left.%': leftInnerPosition\n\t\t\t\t\t}\"\n\t\t>\n\t\t\t<div\n\t\t\t\t\t*ngFor=\"let month of months;\"\n\t\t\t\t\tclass=\"datepicker__calendar-container\"\n\t\t\t\t\t[ngStyle]=\"{'width.%': calendarWidth}\"\n\t\t\t>\n\t\t\t\t<table class=\"datepicker__calendar\">\n\t\t\t\t\t<tbody>\n\t\t\t\t\t<tr *ngFor=\"let week of month.weeks; index as i\" class=\"datepicker__week\">\n\t\t\t\t\t\t<td class=\"datepicker__day\" *ngFor=\"let day of week.days; index as i\"\n\t\t\t\t\t\t\t[ngClass]=\"{\n\t\t\t\t\t\t'is-first': day.isFirst,\n\t\t\t\t\t\t'is-last': day.isLast,\n\t\t\t\t\t\t'is-hidden': day.isHidden,\n\t\t\t\t\t\t'is-disabled': day.isDisabled,\n\t\t\t\t\t\t'is-today': day.isToday,\n\t\t\t\t\t\t'is-selected': day.isSelected,\n\t\t\t\t\t\t'is-in-range': day.isInRange,\n\t\t\t\t\t\t'is-start-date': day.isStartDate,\n\t\t\t\t\t\t'is-end-date': day.isEndDate,\n\t\t\t\t\t\t'is-rest': day.isRest\n\t\t\t\t\t\t}\">\n\t\t\t\t\t\t\t<button class=\"datepicker__button\" [disabled]=\"day.isDisabled || day.isHidden\"\n\t\t\t\t\t\t\t\t\t(click)=\"updateValue(day.date)\">{{day.dayNumber}}\n\t\t\t\t\t\t\t</button>\n\t\t\t\t\t\t</td>\n\t\t\t\t\t</tr>\n\t\t\t\t\t</tbody>\n\t\t\t\t</table>\n\t\t\t</div>\n\t\t</div>\n\t</div>\n\t<div class=\"datepicker__footer\" #footer>\n\t\t<ng-content select=\"footer\"></ng-content>\n\t</div>\n</div>\n\t\t",
styles: [":host{font-family:Arial,Helvetica,sans-serif;border:1px solid #d9d9d8;width:300px;position:relative;display:inline-block;z-index:2;border-radius:4px;box-shadow:0 1px 5px rgba(0,0,0,.15);overflow:hidden;background-color:#fff;box-sizing:border-box;visibility:hidden}:host *{box-sizing:border-box}:host .datepicker__calendar-container{padding:0 10px 10px}:host .datepicker__footer{position:relative;z-index:1}:host table{width:100%;table-layout:fixed;border-spacing:0;border-collapse:collapse}:host td{padding:0}:host .datepicker__weekdays-wrapper::after,:host .datepicker__weekdays-wrapper::before{content:' ';display:table}:host .datepicker__weekdays-wrapper::after{clear:both}:host .datepicker__weekdays-container{padding:10px 10px 0;float:left}:host .datepicker__weekdays{table-layout:fixed;width:100%}:host .datepicker__weekday{color:grey;font-size:12px;height:20px;text-align:center}:host .datepicker__day{position:relative;text-align:center;height:40px;width:auto;border:1px solid #d9d9d8}:host .datepicker__day.is-rest{border:none}:host .datepicker__button{padding:0;background-color:transparent;border:none;outline:0;font-style:inherit;cursor:pointer;color:#8e8d8a;width:100%;height:100%}:host .datepicker__button:hover{border:1px solid transparent;background-color:#f2f2f2;color:#8e8d8a}:host .is-hidden{opacity:0;display:table-cell}:host .is-rest{border:none}:host .is-rest .datepicker__button{color:#c0c0be}:host .is-today .datepicker__button{background-color:#eae7dc}:host .is-in-range .datepicker__button{background-color:#e98074;color:#fff}:host .is-in-range .datepicker__button:hover{background-color:#e66c5e}:host .is-selected .datepicker__button{background-color:#e85a4f;color:#fff;font-weight:700}:host .is-selected .datepicker__button:hover{background-color:#e23022}:host .is-start-date .datepicker__button{background-color:#e85a4f;color:#fff}:host .is-end-date .datepicker__button{background-color:#e85a4f;color:#fff}:host .is-disabled .datepicker__button{color:#d9d9d8;cursor:default}:host .is-disabled .datepicker__button:hover{background-color:transparent}:host.is-directive{visibility:hidden;position:absolute}:host.is-open{visibility:visible}:host.is-animate{transition:height .2s ease-in;width:300px}:host.is-animate .datepicker__main{transition:height .2s ease-in}:host.is-animate .datepicker__calendar-wrapper{position:absolute;width:200%;left:0}:host.is-animate .datepicker__calendar{float:left;width:100%}:host.is-animate .datepicker__calendar-container{float:left}"]
},] },
];
AnimatepickerComponent.ctorParameters = function () { return [
{ type: ElementRef },
{ type: UtilitiesService }
]; };
AnimatepickerComponent.propDecorators = {
numberOfMonths: [{ type: Input }],
calendarContainer: [{ type: ViewChild, args: ['calendarContainer',] }],
calendarTopContainer: [{ type: ViewChild, args: ['calendarTopContainer',] }],
footer: [{ type: ViewChild, args: ['footer',] }],
datepickerWidth: [{ type: HostBinding, args: ['style.width.px',] }],
datepickerHeight: [{ type: HostBinding, args: ['style.height.px',] }]
};
return AnimatepickerComponent;
}(DatepickerComponent));
export { AnimatepickerComponent };
if (false) {
/** @type {?} */
AnimatepickerComponent.prototype.animate;
/** @type {?} */
AnimatepickerComponent.prototype.initialWidth;
/** @type {?} */
AnimatepickerComponent.prototype.calendarWidth;
/** @type {?} */
AnimatepickerComponent.prototype.isAnimating;
/** @type {?} */
AnimatepickerComponent.prototype.leftInnerPosition;
/** @type {?} */
AnimatepickerComponent.prototype.transition;
/** @type {?} */
AnimatepickerComponent.prototype.translateX;
/** @type {?} */
AnimatepickerComponent.prototype.currentYearMonth;
/** @type {?} */
AnimatepickerComponent.prototype.datepickerPosition;
/** @type {?} */
AnimatepickerComponent.prototype.initialised;
/** @type {?} */
AnimatepickerComponent.prototype.calendarHeight;
/**
* Number of months: the number of months displayed
* @type {?}
* @private
*/
AnimatepickerComponent.prototype._numberOfMonths;
/** @type {?} */
AnimatepickerComponent.prototype.calendarContainer;
/** @type {?} */
AnimatepickerComponent.prototype.calendarTopContainer;
/** @type {?} */
AnimatepickerComponent.prototype.footer;
/** @type {?} */
AnimatepickerComponent.prototype.datepickerWidth;
/** @type {?} */
AnimatepickerComponent.prototype.datepickerHeight;
/** @type {?} */
AnimatepickerComponent.prototype.elementRef;
/** @type {?} */
AnimatepickerComponent.prototype.utilities;
}
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYW5pbWF0ZXBpY2tlci5jb21wb25lbnQuanMiLCJzb3VyY2VSb290Ijoibmc6Ly9uZ3gtYW5pbWF0aW5nLWRhdGVwaWNrZXIvIiwic291cmNlcyI6WyJsaWIvY29tcG9uZW50cy9hbmltYXRlcGlja2VyL2FuaW1hdGVwaWNrZXIuY29tcG9uZW50LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7Ozs7O0FBQUEsT0FBTyxFQUVOLFNBQVMsRUFDVCxVQUFVLEVBQ1YsV0FBVyxFQUNYLEtBQUssRUFFTCxTQUFTLEdBQ1QsTUFBTSxlQUFlLENBQUM7QUFFdkIsT0FBTyxFQUFFLGlCQUFpQixFQUFFLE1BQU0sbUNBQW1DLENBQUM7QUFDdEUsT0FBTyxFQUFFLGdCQUFnQixFQUFFLE1BQU0sa0NBQWtDLENBQUM7QUFDcEUsT0FBTyxFQUFFLG1CQUFtQixFQUFFLE1BQU0sb0NBQW9DLENBQUM7QUFFekU7SUFnRjRDLGtEQUFtQjtJQWtEOUQsZ0NBQW1CLFVBQXNCLEVBQVMsU0FBMkI7UUFBN0UsWUFDQyxrQkFBTSxTQUFTLEVBQUUsVUFBVSxDQUFDLFNBQzVCO1FBRmtCLGdCQUFVLEdBQVYsVUFBVSxDQUFZO1FBQVMsZUFBUyxHQUFULFNBQVMsQ0FBa0I7UUFqRDdFOzs0REFFb0Q7UUFFN0MsYUFBTyxHQUFHLElBQUksQ0FBQztRQUdmLGlCQUFXLEdBQUcsS0FBSyxDQUFDO1FBQ3BCLHVCQUFpQixHQUFHLENBQUMsQ0FBQztRQUd0QixzQkFBZ0IsR0FBVyxJQUFJLENBQUM7UUFFaEMsaUJBQVcsR0FBRyxLQUFLLENBQUM7UUFHM0I7OzREQUVvRDtRQUVwRDs7V0FFRztRQUNLLHFCQUFlLEdBQVEsSUFBSSxLQUFLLENBQUMsQ0FBQyxDQUFDLENBQUM7O0lBNEI1QyxDQUFDO0lBM0JELHNCQUNJLGtEQUFjOzs7O1FBVWxCO1lBQ0MsT0FBTyxJQUFJLENBQUMsZUFBZSxDQUFDO1FBQzdCLENBQUM7Ozs7O1FBYkQsVUFDbUIsS0FBSztZQUN2QixJQUFJLEtBQUssS0FBSyxT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bnQsXG5cdEVsZW1lbnRSZWYsXG5cdEhvc3RCaW5kaW5nLFxuXHRJbnB1dCxcblx0T25Jbml0LFxuXHRWaWV3Q2hpbGQsXG59IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuaW1wb3J0IHsgWWVhck1vbnRoLCBNb250aCB9IGZyb20gJy4uLy4uL21vZGVscy9kYXRlcGlja2VyLm1vZGVsJztcbmltcG9ydCB7IERhdGVwaWNrZXJTZXJ2aWNlIH0gZnJvbSAnLi4vLi4vc2VydmljZXMvZGF0ZXBpY2tlci5zZXJ2aWNlJztcbmltcG9ydCB7IFV0aWxpdGllc1NlcnZpY2UgfSBmcm9tICcuLi8uLi9zZXJ2aWNlcy91dGlsaXRpZXMuc2VydmljZSc7XG5pbXBvcnQgeyBEYXRlcGlja2VyQ29tcG9uZW50IH0gZnJvbSAnLi4vZGF0ZXBpY2tlci9kYXRlcGlja2VyLmNvbXBvbmVudCc7XG5cbkBDb21wb25lbnQoe1xuXHRzZWxlY3RvcjogJ2FhLWFuaW1hdGVwaWNrZXInLFxuXHR0ZW1wbGF0ZTogYDxkaXYgY2xhc3M9XCJkYXRlcGlja2VyX193cmFwcGVyXCIgW25nU3R5bGVdPVwiZGF0ZXBpY2tlclBvc2l0aW9uXCI+XG5cdDxkaXYgI2NhbGVuZGFyVG9wQ29udGFpbmVyPlxuXHRcdDxkaXYgY2xhc3M9XCJkYXRlcGlja2VyX19oZWFkZXJcIiAjaGVhZGVyPlxuXHRcdFx0PG5nLWNvbnRlbnQgc2VsZWN0PVwiaGVhZGVyXCI+PC9uZy1jb250ZW50PlxuXHRcdDwvZGl2PlxuXHRcdDxhYS1uYXZpZ2F0aW9uXG5cdFx0XHRcdChwcmV2aW91c0NsaWNrKT1cImdvVG9QcmV2aW91c01vbnRoKClcIlxuXHRcdFx0XHQobmV4dENsaWNrKT1cImdvVG9OZXh0TW9udGgoKVwiXG5cdFx0XHRcdChzdWJOYXZpZ2F0aW9uQ2xpY2spPVwiZ29Ub0RhdGUoJGV2ZW50KVwiXG5cdFx0XHRcdFtsYW5ndWFnZV09XCJsYW5ndWFnZVwiXG5cdFx0XHRcdFthbmltYXRlXT1cImFuaW1hdGVcIlxuXHRcdFx0XHRbdHJhbnNsYXRlWF09XCJ0cmFuc2xhdGVYXCJcblx0XHRcdFx0W3RyYW5zaXRpb25dPVwidHJhbnNpdGlvblwiXG5cdFx0XHRcdFtsZWZ0UG9zaXRpb25dPVwibGVmdElubmVyUG9zaXRpb25cIlxuXHRcdFx0XHRbaGlkZU5hdmlnYXRpb25dPVwib3B0aW9ucy5oaWRlTmF2aWdhdGlvblwiXG5cdFx0XHRcdFt0b3RhbFllYXJNb250aF09XCJ0b3RhbFllYXJNb250aFwiXG5cdFx0PjwvYWEtbmF2aWdhdGlvbj5cblxuXHRcdDxkaXYgY2xhc3M9XCJkYXRlcGlja2VyX193ZWVrZGF5cy13cmFwcGVyXCI+XG5cdFx0XHQ8ZGl2IFx0Km5nRm9yPVwibGV0IG1vbnRoIG9mIG51bWJlck9mTW9udGhzXCJcblx0XHRcdFx0XHRbbmdTdHlsZV09XCJ7XG5cdFx0XHRcdCd3aWR0aC4lJzogKDEwMCAvIG51bWJlck9mTW9udGhzLmxlbmd0aClcblx0XHRcdFx0fVwiXG5cdFx0XHRcdFx0Y2xhc3M9XCJkYXRlcGlja2VyX193ZWVrZGF5cy1jb250YWluZXJcIj5cblx0XHRcdFx0PHRhYmxlIGNsYXNzPVwiZGF0ZXBpY2tlcl9fd2Vla2RheXNcIj5cblx0XHRcdFx0XHQ8dGhlYWQ+XG5cdFx0XHRcdFx0PHRkIGNsYXNzPVwiZGF0ZXBpY2tlcl9fd2Vla2RheVwiICpuZ0Zvcj1cImxldCB3ZWVrZGF5IG9mIHdlZWtkYXlzOyBpbmRleCBhcyBpXCI+e3t3ZWVrZGF5fX08L3RkPlxuXHRcdFx0XHRcdDwvdGhlYWQ+XG5cdFx0XHRcdDwvdGFibGU+XG5cdFx0XHQ8L2Rpdj5cblx0XHQ8L2Rpdj5cblx0PC9kaXY+XG5cdDxkaXYgY2xhc3M9XCJkYXRlcGlja2VyX19tYWluXCIgW25nU3R5bGVdPVwie1xuXHRcdCdoZWlnaHQucHgnOiBjYWxlbmRhckhlaWdodH1cIj5cblx0XHQ8ZGl2IGNsYXNzPVwiZGF0ZXBpY2tlcl9fY2FsZW5kYXItd3JhcHBlclwiICNjYWxlbmRhckNvbnRhaW5lciBbbmdTdHlsZV09XCJ7XG5cdFx0XHRcdFx0J3RyYW5zZm9ybSc6ICd0cmFuc2xhdGVYKCcgKyB0cmFuc2xhdGVYICsgJyUpJyxcblx0XHRcdFx0XHQndHJhbnNpdGlvbic6IHRyYW5zaXRpb24sXG5cdFx0XHRcdFx0J2xlZnQuJSc6IGxlZnRJbm5lclBvc2l0aW9uXG5cdFx0XHRcdFx0fVwiXG5cdFx0PlxuXHRcdFx0PGRpdlxuXHRcdFx0XHRcdCpuZ0Zvcj1cImxldCBtb250aCBvZiBtb250aHM7XCJcblx0XHRcdFx0XHRjbGFzcz1cImRhdGVwaWNrZXJfX2NhbGVuZGFyLWNvbnRhaW5lclwiXG5cdFx0XHRcdFx0W25nU3R5bGVdPVwieyd3aWR0aC4lJzogY2FsZW5kYXJXaWR0aH1cIlxuXHRcdFx0PlxuXHRcdFx0XHQ8dGFibGUgY2xhc3M9XCJkYXRlcGlja2VyX19jYWxlbmRhclwiPlxuXHRcdFx0XHRcdDx0Ym9keT5cblx0XHRcdFx0XHQ8dHIgKm5nRm9yPVwibGV0IHdlZWsgb2YgbW9udGgud2Vla3M7IGluZGV4IGFzIGlcIiBjbGFzcz1cImRhdGVwaWNrZXJfX3dlZWtcIj5cblx0XHRcdFx0XHRcdDx0ZCBjbGFzcz1cImRhdGVwaWNrZXJfX2RheVwiICpuZ0Zvcj1cImxldCBkYXkgb2Ygd2Vlay5kYXlzOyBpbmRleCBhcyBpXCJcblx0XHRcdFx0XHRcdFx0W25nQ2xhc3NdPVwie1xuXHRcdFx0XHRcdFx0J2lzLWZpcnN0JzogZGF5LmlzRmlyc3QsXG5cdFx0XHRcdFx0XHQnaXMtbGFzdCc6IGRheS5pc0xhc3QsXG5cdFx0XHRcdFx0XHQnaXMtaGlkZGVuJzogZGF5LmlzSGlkZGVuLFxuXHRcdFx0XHRcdFx0J2lzLWRpc2FibGVkJzogZGF5LmlzRGlzYWJsZWQsXG5cdFx0XHRcdFx0XHQnaXMtdG9kYXknOiBkYXkuaXNUb2RheSxcblx0XHRcdFx0XHRcdCdpcy1zZWxlY3RlZCc6IGRheS5pc1NlbGVjdGVkLFxuXHRcdFx0XHRcdFx0J2lzLWluLXJhbmdlJzogZGF5LmlzSW5SYW5nZSxcblx0XHRcdFx0XHRcdCdpcy1zdGFydC1kYXRlJzogZGF5LmlzU3RhcnREYXRlLFxuXHRcdFx0XHRcdFx0J2lzLWVuZC1kYXRlJzogZGF5LmlzRW5kRGF0ZSxcblx0XHRcdFx0XHRcdCdpcy1yZXN0JzogZGF5LmlzUmVzdFxuXHRcdFx0XHRcdFx0fVwiPlxuXHRcdFx0XHRcdFx0XHQ8YnV0dG9uIGNsYXNzPVwiZGF0ZXBpY2tlcl9fYnV0dG9uXCIgW2Rpc2FibGVkXT1cImRheS5pc0Rpc2FibGVkIHx8IGRheS5pc0hpZGRlblwiXG5cdFx0XHRcdFx0XHRcdFx0XHQoY2xpY2spPVwidXBkYXRlVmFsdWUoZGF5LmRhdGUpXCI+e3tkYXkuZGF5TnVtYmVyfX1cblx0XHRcdFx0XHRcdFx0PC9idXR0b24+XG5cdFx0XHRcdFx0XHQ8L3RkPlxuXHRcdFx0XHRcdDwvdHI+XG5cdFx0XHRcdFx0PC90Ym9keT5cblx0XHRcdFx0PC90YWJsZT5cblx0XHRcdDwvZGl2PlxuXHRcdDwvZGl2PlxuXHQ8L2Rpdj5cblx0PGRpdiBjbGFzcz1cImRhdGVwaWNrZXJfX2Zvb3RlclwiICNmb290ZXI+XG5cdFx0PG5nLWNvbnRlbnQgc2VsZWN0PVwiZm9vdGVyXCI+PC9uZy1jb250ZW50PlxuXHQ8L2Rpdj5cbjwvZGl2PlxuXHRcdGAsXG5cdHN0eWxlczogW2A6aG9zdHtmb250LWZhbWlseTpBcmlhbCxIZWx2ZXRpY2Esc2Fucy1zZXJpZjtib3JkZXI6MXB4IHNvbGlkICNkOWQ5ZDg7d2lkdGg6MzAwcHg7cG9zaXRpb246cmVsYXRpdmU7ZGlzcGxheTppbmxpbmUtYmxvY2s7ei1pbmRleDoyO2JvcmRlci1yYWRpdXM6NHB4O2JveC1zaGFkb3c6MCAxcHggNXB4IHJnYmEoMCwwLDAsLjE1KTtvdmVyZmxvdzpoaWRkZW47YmFja2dyb3VuZC1jb2xvcjojZmZmO2JveC1zaXppbmc6Ym9yZGVyLWJveDt2aXNpYmlsaXR5OmhpZGRlbn06aG9zdCAqe2JveC1zaXppbmc6Ym9yZGVyLWJveH06aG9zdCAuZGF0ZXBpY2tlcl9fY2FsZW5kYXItY29udGFpbmVye3BhZGRpbmc6MCAxMHB4IDEwcHh9Omhvc3QgLmRhdGVwaWNrZXJfX2Zvb3Rlcntwb3NpdGlvbjpyZWxhdGl2ZTt6LWluZGV4OjF9Omhvc3QgdGFibGV7d2lkdGg6MTAwJTt0YWJsZS1sYXlvdXQ6Zml4ZWQ7Ym9yZGVyLXNwYWNpbmc6MDtib3JkZXItY29sbGFwc2U6Y29sbGFwc2V9Omhvc3QgdGR7cGFkZGluZzowfTpob3N0IC5kYXRlcGlja2VyX193ZWVrZGF5cy13cmFwcGVyOjphZnRlciw6aG9zdCAuZGF0ZXBpY2tlcl9fd2Vla2RheXMtd3JhcHBlcjo6YmVmb3Jle2NvbnRlbnQ6JyAnO2Rpc3BsYXk6dGFibGV9Omhvc3QgLmRhdGVwaWNrZXJfX3dlZWtkYXlzLXdyYXBwZXI6OmFmdGVye2NsZWFyOmJvdGh9Omhvc3QgLmRhdGVwaWNrZXJfX3dlZWtkYXlzLWNvbnRhaW5lcntwYWRkaW5nOjEwcHggMTBweCAwO2Zsb2F0OmxlZnR9Omhvc3QgLmRhdGVwaWNrZXJfX3dlZWtkYXlze3RhYmxlLWxheW91dDpmaXhlZDt3aWR0aDoxMDAlfTpob3N0IC5kYXRlcGlja2VyX193ZWVrZGF5e2NvbG9yOmdyZXk7Zm9udC1zaXplOjEycHg7aGVpZ2h0OjIwcHg7dGV4dC1hbGlnbjpjZW50ZXJ9Omhvc3QgLmRhdGVwaWNrZXJfX2RheXtwb3NpdGlvbjpyZWxhdGl2ZTt0ZXh0LWFsaWduOmNlbnRlcjtoZWlnaHQ6NDBweDt3aWR0aDphdXRvO2JvcmRlcjoxcHggc29saWQgI2Q5ZDlkOH06aG9zdCAuZGF0ZXBpY2tlcl9fZGF5LmlzLXJlc3R7Ym9yZGVyOm5vbmV9Omhvc3QgLmRhdGVwaWNrZXJfX2J1dHRvbntwYWRkaW5nOjA7YmFja2dyb3VuZC1jb2xvcjp0cmFuc3BhcmVudDtib3JkZXI6bm9uZTtvdXRsaW5lOjA7Zm9udC1zdHlsZTppbmhlcml0O2N1cnNvcjpwb2ludGVyO2NvbG9yOiM4ZThkOGE7d2lkdGg6MTAwJTtoZWlnaHQ6MTAwJX06aG9zdCAuZGF0ZXBpY2tlcl9fYnV0dG9uOmhvdmVye2JvcmRlcjoxcHggc29saWQgdHJhbnNwYXJlbnQ7YmFja2dyb3VuZC1jb2xvcjojZjJmMmYyO2NvbG9yOiM4ZThkOGF9Omhvc3QgLmlzLWhpZGRlbntvcGFjaXR5OjA7ZGlzcGxheTp0YWJsZS1jZWxsfTpob3N0IC5pcy1yZXN0e2JvcmRlcjpub25lfTpob3N0IC5pcy1yZXN0IC5kYXRlcGlja2VyX19idXR0b257Y29sb3I6I2MwYzBiZX06aG9zdCAuaXMtdG9kYXkgLmRhdGVwaWNrZXJfX2J1dHRvbntiYWNrZ3JvdW5kLWNvbG9yOiNlYWU3ZGN9Omhvc3QgLmlzLWluLXJhbmdlIC5kYXRlcGlja2VyX19idXR0b257YmFja2dyb3VuZC1jb2xvcjojZTk4MDc0O2NvbG9yOiNmZmZ9Omhvc3QgLmlzLWluLXJhbmdlIC5kYXRlcGlja2VyX19idXR0b246aG92ZXJ7YmFja2dyb3VuZC1jb2xvcjojZTY2YzVlfTpob3N0IC5pcy1zZWxlY3RlZCAuZGF0ZXBpY2tlcl9fYnV0dG9ue2JhY2tncm91bmQtY29sb3I6I2U4NWE0Zjtjb2xvcjojZmZmO2ZvbnQtd2VpZ2h0OjcwMH06aG9zdCAuaXMtc2VsZWN0ZWQgLmRhdGVwaWNrZXJfX2J1dHRvbjpob3ZlcntiYWNrZ3JvdW5kLWNvbG9yOiNlMjMwMjJ9Omhvc3QgLmlzLXN0YXJ0LWRhdGUgLmRhdGVwaWNrZXJfX2J1dHRvbntiYWNrZ3JvdW5kLWNvbG9yOiNlODVhNGY7Y29sb3I6I2ZmZn06aG9zdCAuaXMtZW5kLWRhdGUgLmRhdGVwaWNrZXJfX2J1dHRvbntiYWNrZ3JvdW5kLWNvbG9yOiNlODVhNGY7Y29sb3I6I2ZmZn06aG9zdCAuaXMtZGlzYWJsZWQgLmRhdGVwaWNrZXJfX2J1dHRvbntjb2xvcjojZDlkOWQ4O2N1cnNvcjpkZWZhdWx0fTpob3N0IC5pcy1kaXNhYmxlZCAuZGF0ZXBpY2tlcl9fYnV0dG9uOmhvdmVye2JhY2tncm91bmQtY29sb3I6dHJhbnNwYXJlbnR9Omhvc3QuaXMtZGlyZWN0aXZle3Zpc2liaWxpdHk6aGlkZGVuO3Bvc2l0aW9uOmFic29sdXRlfTpob3N0LmlzLW9wZW57dmlzaWJpbGl0eTp2aXNpYmxlfTpob3N0LmlzLWFuaW1hdGV7dHJhbnNpdGlvbjpoZWlnaHQgLjJzIGVhc2UtaW47d2lkdGg6MzAwcHh9Omhvc3QuaXMtYW5pbWF0ZSAuZGF0ZXBpY2tlcl9fbWFpbnt0cmFuc2l0aW9uOmhlaWdodCAuMnMgZWFzZS1pbn06aG9zdC5pcy1hbmltYXRlIC5kYXRlcGlja2VyX19jYWxlbmRhci13cmFwcGVye3Bvc2l0aW9uOmFic29sdXRlO3dpZHRoOjIwMCU7bGVmdDowfTpob3N0LmlzLWFuaW1hdGUgLmRhdGVwaWNrZXJfX2NhbGVuZGFye2Zsb2F0OmxlZnQ7d2lkdGg6MTAwJX06aG9zdC5pcy1hbmltYXRlIC5kYXRlcGlja2VyX19jYWxlbmRhci1jb250YWluZXJ7ZmxvYXQ6bGVmdH1gXVxufSlcbmV4cG9ydCBjbGFzcyBBbmltYXRlcGlja2VyQ29tcG9uZW50IGV4dGVuZHMgRGF0ZXBpY2tlckNvbXBvbmVudCBpbXBsZW1lbnRzIE9uSW5pdCwgQWZ0ZXJWaWV3SW5pdCB7XG5cdC8qID09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT1cblx0ICogSW50ZXJuYWwgUHJvcGVydGllc1xuXHQgKiA9PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09ICovXG5cblx0cHVibGljIGFuaW1hdGUgPSB0cnVlO1xuXHRwdWJsaWMgaW5pdGlhbFdpZHRoOiBudW1iZXI7XG5cdHB1YmxpYyBjYWxlbmRhcldpZHRoOiBudW1iZXI7XG5cdHB1YmxpYyBpc0FuaW1hdGluZyA9IGZhbHNlO1xuXHRwdWJsaWMgbGVmdElubmVyUG9zaXRpb24gPSAwO1xuXHRwdWJsaWMgdHJhbnNpdGlvbjogc3RyaW5nO1xuXHRwdWJsaWMgdHJhbnNsYXRlWDogbnVtYmVyO1xuXHRwdWJsaWMgY3VycmVudFllYXJNb250aDogb2JqZWN0ID0gbnVsbDtcblx0cHVibGljIGRhdGVwaWNrZXJQb3NpdGlvbjogb2JqZWN0O1xuXHRwdWJsaWMgaW5pdGlhbGlzZWQgPSBmYWxzZTtcblx0cHVibGljIGNhbGVuZGFySGVpZ2h0OiBudW1iZXJcblxuXHQvKiA9PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09XG5cdCAqIEV4dGVybmFsIFByb3BlcnRpZXNcblx0ICogPT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PSAqL1xuXG5cdC8qKlxuXHQgKiBOdW1iZXIgb2YgbW9udGhzOiB0aGUgbnVtYmVyIG9mIG1vbnRocyBkaXNwbGF5ZWRcblx0ICovXG5cdHByaXZhdGUgX251bWJlck9mTW9udGhzOiBhbnkgPSBuZXcgQXJyYXkoMSk7XG5cdEBJbnB1dCgpXG5cdHNldCBudW1iZXJPZk1vbnRocyh2YWx1ZSkge1xuXHRcdGlmICh2YWx1ZSA9PT0gdW5kZWZpbmVkIHx8IHZhbHVlID09PSB0aGlzLl9udW1iZXJPZk1vbnRocy5sZW5ndGgpIHtcblx0XHRcdHJldHVybjtcblx0XHR9XG5cdFx0dGhpcy5fbnVtYmVyT2ZNb250aHMgPSBuZXcgQXJyYXkodmFsdWUpO1xuXHRcdHRoaXMuc2V0RGF0ZVBpY2tlckRpbWVuc2lvbigpO1xuXG5cdFx0dGhpcy5nb1RvRGF0ZSh0aGlzLmRhdGUpO1xuXHR9XG5cblx0Z2V0IG51bWJlck9mTW9udGhzKCk6IE51bWJlcltdIHtcblx0XHRyZXR1cm4gdGhpcy5fbnVtYmVyT2ZNb250aHM7XG5cdH1cblxuXHQvKiA9PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09XG5cdCAqIEJpbmRpbmdzIGFuZCBDaGlsZHJlblxuXHQgKiA9PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09ICovXG5cblx0QFZpZXdDaGlsZCgnY2FsZW5kYXJDb250YWluZXInKSBwdWJsaWMgY2FsZW5kYXJDb250YWluZXI6IEVsZW1lbnRSZWY7XG5cdEBWaWV3Q2hpbGQoJ2NhbGVuZGFyVG9wQ29udGFpbmVyJykgcHVibGljIGNhbGVuZGFyVG9wQ29udGFpbmVyOiBFbGVtZW50UmVmO1xuXHRAVmlld0NoaWxkKCdmb290ZXInKSBwdWJsaWMgZm9vdGVyOiBFbGVtZW50UmVmO1xuXHRASG9zdEJpbmRpbmcoJ3N0eWxlLndpZHRoLnB4JykgcHVibGljIGRhdGVwaWNrZXJXaWR0aDogbnVtYmVyO1xuXHRASG9zdEJpbmRpbmcoJ3N0eWxlLmhlaWdodC5weCcpIHB1YmxpYyBkYXRlcGlja2VySGVpZ2h0OiBudW1iZXI7XG5cblx0Y29uc3RydWN0b3IocHVibGljIGVsZW1lbnRSZWY6IEVsZW1lbnRSZWYsIHB1YmxpYyB1dGlsaXRpZXM6IFV0aWxpdGllc1NlcnZpY2UpIHtcblx0XHRzdXBlcih1dGlsaXRpZXMsIGVsZW1lbnRSZWYpO1xuXHR9XG5cblx0bmdPbkluaXQoKSB7XG5cdFx0Ly8gR2V0IHRoZSBjb21wdXRlZCB3aWR0aCBmcm9tIHRoZSBjYWxlbmRhci4gU2V0IHRoZSBpbml0aWFsIHdpZHRoXG5cdFx0Y29uc3QgY29tcHV0ZWRXaWR0aCA9IHdpbmRvd1xuXHRcdFx0LmdldENvbXB1dGVkU3R5bGUodGhpcy5lbGVtZW50UmVmLm5hdGl2ZUVsZW1lbnQsIG51bGwpXG5cdFx0XHQuZ2V0UHJvcGVydHlWYWx1ZSgnd2lkdGgnKTtcblx0XHR0aGlzLmluaXRpYWxXaWR0aCA9IHBhcnNlSW50KGNvbXB1dGVkV2lkdGgsIDEwKTtcblx0XHR0aGlzLmluaXRpYWxpc2VkID0gdHJ1ZTtcblxuXHRcdC8vIFNldCB0aGUgY3VycmVudCB5ZWFyIGFuZCBtb250aCBvYmplY3Rcblx0XHRpZiAoIXRoaXMubW9udGggJiYgIXRoaXMueWVhcikge1xuXHRcdFx0dGhpcy5nb1RvRGF0ZSh0aGlzLm9wdGlvbnMuY3VycmVudERhdGUpO1xuXHRcdH1cblx0fVxuXG5cdG5nQWZ0ZXJWaWV3SW5pdCgpIHtcblx0XHRzZXRUaW1lb3V0KCgpID0+IHtcblx0XHRcdHRoaXMuc2V0RGF0ZVBpY2tlckRpbWVuc2lvbigpO1xuXHRcdFx0dGhpcy5zZXREYXRlcGlja2VySGVpZ2h0KHRydWUpO1xuXHRcdH0pO1xuXHR9XG5cblx0LyoqXG5cdCAqIFNldCB0aGUgaGVpZ2h0IGFuZCB0aGUgd2lkdGggcHJvcGVydGllc1xuXHQgKi9cblx0c2V0RGF0ZVBpY2tlckRpbWVuc2lvbigpOiB2b2lkIHtcblx0XHR0aGlzLmRhdGVwaWNrZXJIZWlnaHQgPVxuXHRcdFx0dGhpcy5jYWxlbmRhckNvbnRhaW5lci5uYXRpdmVFbGVtZW50Lm9mZnNldEhlaWdodCArXG5cdFx0XHR0aGlzLmNhbGVuZGFyVG9wQ29udGFpbmVyLm5hdGl2ZUVsZW1lbnQub2Zmc2V0SGVpZ2h0ICsgXG5cdFx0XHR0aGlzLmZvb3Rlci5uYXRpdmVFbGVtZW50Lm9mZnNldEhlaWdodDtcblx0XHR0aGlzLmNhbGVuZGFySGVpZ2h0ID0gdGhpcy5jYWxlbmRhckNvbnRhaW5lci5uYXRpdmVFbGVtZW50Lm9mZnNldEhlaWdodDtcblx0XHR0aGlzLmRhdGVwaWNrZXJXaWR0aCA9IHRoaXMuaW5pdGlhbFdpZHRoICogdGhpcy5fbnVtYmVyT2ZNb250aHMubGVuZ3RoO1xuXHR9XG5cblx0LyoqXG5cdCAqIEdvIHRvIGEgc3BlY2lmaWMgbW9udGhcblx0ICpcblx0ICogQHBhcmFtIGRhdGUgLSBvcHRpb25hbFxuXHQgKi9cblx0Z29Ub0RhdGUoZGF0ZT86IERhdGUpOiB2b2lkIHtcblx0XHRpZiAoZGF0ZSkge1xuXHRcdFx0dGhpcy5jdXJyZW50WWVhck1vbnRoID0gdGhpcy5nZXROZXh0WWVhck1vbnRoQXJyYXkoZGF0ZS5nZXRGdWxsWWVhcigpLCBkYXRlLmdldE1vbnRoKCkpO1xuXHRcdH1cblx0XHR0aGlzLmNhbGVuZGFyV2lkdGggPSA1MCAvIHRoaXMuX251bWJlck9mTW9udGhzLmxlbmd0aDtcblx0XHR0aGlzLm1vbnRocyA9IHRoaXMuZ2V0TmV4dE1vbnRoQXJyYXkodGhpcy5jdXJyZW50WWVhck1vbnRoLCB0cnVlKTtcblx0XHR0aGlzLnJlc2V0U3R5bGUoKTtcblx0fVxuXG5cdC8qKlxuXHQgKiBDcmVhdGUgYW4gYXJyYXkgb2YgdGhlIG5leHQgeWVhciBhbmQgbW9udGhzXG5cdCAqXG5cdCAqIEBwYXJhbSB5ZWFyXG5cdCAqIEBwYXJhbSBtb250aFxuXHQgKi9cblx0Z2V0TmV4dFllYXJNb250aEFycmF5KHllYXI6IG51bWJlciwgbW9udGg6IG51bWJlcik6IFllYXJNb250aFtdIHtcblx0XHRjb25zdCBhcnJheSA9IFtdO1xuXHRcdGZvciAobGV0IGluZGV4ID0gMDsgaW5kZXggPCB0aGlzLl9udW1iZXJPZk1vbnRocy5sZW5ndGg7IGluZGV4KyspIHtcblx0XHRcdGFycmF5LnB1c2goeyB5ZWFyOiB5ZWFyLCBtb250aDogbW9udGggfSk7XG5cdFx0XHR5ZWFyID0gRGF0ZXBpY2tlclNlcnZpY2UuZ2V0WWVhck9mTmV4dE1vbnRoKHllYXIsIG1vbnRoKTtcblx0XHRcdG1vbnRoID0gRGF0ZXBpY2tlclNlcnZpY2UuZ2V0TmV4dE1vbnRoKG1vbnRoKTtcblx0XHR9XG5cdFx0cmV0dXJuIGFycmF5O1xuXHR9XG5cblx0LyoqXG5cdCAqIENyZWF0ZSBhbiBhcnJheSBvZiB0aGUgcHJldmlvdXMgeWVhciBhbmQgbW9udGhzXG5cdCAqXG5cdCAqIEBwYXJhbSB5ZWFyXG5cdCAqIEBwYXJhbSBtb250aFxuXHQgKi9cblx0Z2V0UHJldmlvdXNZZWFyTW9udGhBcnJheSh5ZWFyOiBudW1iZXIsIG1vbnRoOiBudW1iZXIpOiBZZWFyTW9udGhbXSB7XG5cdFx0Y29uc3QgYXJyYXkgPSBbXTtcblx0XHRmb3IgKGxldCBpbmRleCA9IDA7IGluZGV4IDwgdGhpcy5fbnVtYmVyT2ZNb250aHMubGVuZ3RoOyBpbmRleCsrKSB7XG5cdFx0XHRhcnJheS51bnNoaWZ0KHsgeWVhcjogeWVhciwgbW9udGg6IG1vbnRoIH0pO1xuXHRcdFx0eWVhciA9IERhdGVwaWNrZXJTZXJ2aWNlLmdldFllYXJPZlByZXZpb3VzTW9udGgoeWVhciwgbW9udGgpO1xuXHRcdFx0bW9udGggPSBEYXRlcGlja2VyU2VydmljZS5nZXRQcmV2aW91c01vbnRoKG1vbnRoKTtcblx0XHR9XG5cdFx0cmV0dXJuIGFycmF5O1xuXHR9XG5cblx0LyoqXG5cdCAqIFNldCB0aGUgZGF0ZXBpY2tlciBoZWlnaHQsIHVzZWQgd2hlbiBhbmltYXRpbmdcblx0ICpcblx0ICogQHBhcmFtIGRpcmVjdGlvblJpZ2h0IC0gU2V0IG9wdGlvbmFsIHdoZW4gc2xpZGluZyB0byB0aGUgcmlnaHRcblx0ICovXG5cdHNldERhdGVwaWNrZXJIZWlnaHQoZGlyZWN0aW9uUmlnaHQ/OiBib29sZWFuKTogdm9pZCB7XG5cdFx0bGV0IGluZGV4QXJyYXk6IG51bWJlcltdO1xuXG5cdFx0Ly8gVE9ETzogU2VwZXJhdGUgdGhpcyBsb2dpYyBmb3IgcmVhZGFiaWxpdHkgcHVycG9zZVxuXHRcdGlmICh0aGlzLl9udW1iZXJPZk1vbnRocy5sZW5ndGggPiAxKSB7XG