ng2-datepicker
Version:
ng2-datepicker is simple and minimal Angular datepicker component. It is fully customizable.
253 lines • 41.6 kB
JavaScript
import { Component, Input, HostListener, ElementRef, EventEmitter, Inject, ChangeDetectorRef } from '@angular/core';
import { DOCUMENT } from '@angular/common';
import { NG_VALUE_ACCESSOR } from '@angular/forms';
import { SlimScrollOptions, SlimScrollEvent } from 'ngx-slimscroll';
import { mergeDatepickerOptions, defaultOptions } from './datepicker-options.interface';
import { eachDayOfInterval, startOfMonth, endOfMonth, getDate, getMonth, getYear, isToday, isSameDay, isSameMonth, isSameYear, isBefore, isAfter, getDay, subDays, setDay, format, addMonths, subMonths, setYear, addYears, subYears } from 'date-fns';
import { fromEvent, Subscription } from 'rxjs';
import { filter } from 'rxjs/operators';
export class DatepickerComponent {
constructor(elementRef, ref, document) {
this.elementRef = elementRef;
this.ref = ref;
this.options = Object.assign({}, defaultOptions);
this.scrollOptions = new SlimScrollOptions(this.scrollBarOptions);
this.isOpened = false;
this.innerValue = new Date();
this.displayValue = '';
this.view = 'days';
this.date = new Date();
this.years = [];
this.days = [];
this.dayNames = [];
this.scrollEvents = new EventEmitter();
this.sub = new Subscription();
this.onTouchedCallback = () => { };
this.onChangeCallback = () => { };
this.doc = document;
}
get value() {
return this.innerValue;
}
set value(val) {
this.innerValue = val;
this.displayValue = format(this.innerValue, this.options.format, { locale: this.options.locale });
this.onChangeCallback(this.innerValue);
}
get title() {
return format(this.date, this.options.formatTitle);
}
get scrollBarOptions() {
return {
barBackground: (this.options && this.options.scrollBarColor) || '#dfe3e9',
gridBackground: 'transparent',
barBorderRadius: '3',
gridBorderRadius: '3',
barWidth: '6',
gridWidth: '6',
barMargin: '0',
gridMargin: '0'
};
}
ngOnInit() {
this.view = 'days';
this.date = new Date();
this.init();
}
ngOnChanges(changes) {
if ('options' in changes) {
this.options = mergeDatepickerOptions(this.options);
this.scrollOptions = new SlimScrollOptions(this.scrollBarOptions);
if (this.sub) {
this.sub.unsubscribe();
}
if (this.options.enableKeyboard) {
this.sub = fromEvent(this.doc || document, 'keyup')
.pipe(filter(() => this.isOpened))
.subscribe(e => {
e.preventDefault();
e.stopPropagation();
switch (e.key) {
case 'Down':
case 'ArrowDown':
this.prevYear();
break;
case 'Up':
case 'ArrowUp':
this.nextYear();
break;
case 'Left':
case 'ArrowLeft':
this.prevMonth();
break;
case 'Right':
case 'ArrowRight':
this.nextMonth();
break;
case 'Esc':
case 'Escape':
case 'Enter':
this.isOpened = false;
break;
default:
return;
}
});
}
}
}
ngOnDestroy() {
this.sub.unsubscribe();
}
toggle() {
this.isOpened = !this.isOpened;
if (this.isOpened) {
this.view = 'days';
this.date = this.value;
this.initDays();
}
}
toggleView() {
this.view = this.view === 'days' ? 'years' : 'days';
if (this.view === 'years') {
this.ref.detectChanges();
this.scrollToYear();
}
}
nextMonth() {
this.date = addMonths(this.date, 1);
this.initDays();
}
prevMonth() {
this.date = subMonths(this.date, 1);
this.initDays();
}
nextYear() {
this.date = addYears(this.date, 1);
this.initDays();
}
prevYear() {
this.date = subYears(this.date, 1);
this.initDays();
}
setDate(i) {
this.date = this.days[i].date;
this.value = this.date;
this.initDays();
this.isOpened = false;
}
setYear(i) {
this.date = setYear(this.date, this.years[i].year);
this.initDays();
this.initYears();
this.view = 'days';
}
scrollToYear() {
const parent = this.elementRef.nativeElement.querySelector('.main-calendar-years');
const el = this.elementRef.nativeElement.querySelector('.year-unit.is-selected');
const y = el.offsetTop - parent.clientHeight / 2 + el.clientHeight / 2;
const event = new SlimScrollEvent({ type: 'scrollTo', y, duration: 100 });
this.scrollEvents.emit(event);
}
init() {
this.initDayNames();
this.initDays();
this.initYears();
}
initDays() {
const date = this.date || new Date();
const [start, end] = [startOfMonth(date), endOfMonth(date)];
this.days = eachDayOfInterval({ start, end }).map((d) => this.generateDay(d));
const tmp = getDay(start) - this.options.firstCalendarDay;
const prevDays = tmp < 0 ? 7 - this.options.firstCalendarDay : tmp;
for (let i = 1; i <= prevDays; i++) {
const d = subDays(start, i);
this.days.unshift(this.generateDay(d, false));
}
}
initYears() {
const range = this.options.maxYear - this.options.minYear + 1;
this.years = Array.from(new Array(range), (_, i) => i + this.options.minYear).map(year => {
return { year, isThisYear: year === getYear(this.date) };
});
}
initDayNames() {
this.dayNames = [];
const start = this.options.firstCalendarDay;
for (let i = start; i <= 6 + start; i++) {
const date = setDay(new Date(), i);
this.dayNames.push(format(date, this.options.formatDays, { locale: this.options.locale }));
}
}
generateDay(date, inThisMonth = true) {
return {
date,
day: getDate(date),
month: getMonth(date),
year: getYear(date),
inThisMonth,
isToday: isToday(date),
isSelected: isSameDay(date, this.innerValue) && isSameMonth(date, this.innerValue) && isSameYear(date, this.innerValue),
isSelectable: this.isDateSelectable(date)
};
}
isDateSelectable(date) {
if (this.options.minDate && isBefore(date, this.options.minDate)) {
return false;
}
if (this.options.maxDate && isAfter(date, this.options.maxDate)) {
return false;
}
return true;
}
writeValue(val) {
if (!val) {
return;
}
this.innerValue = val;
this.displayValue = format(this.innerValue, this.options.format, { locale: this.options.locale });
this.init();
}
registerOnChange(fn) {
this.onChangeCallback = fn;
}
registerOnTouched(fn) {
this.onTouchedCallback = fn;
}
onBlur(e) {
if (!this.isOpened) {
return;
}
const input = this.elementRef.nativeElement.querySelector('.datepicker-container > input');
if (!input || e.target === input || input.contains(e.target)) {
return;
}
const container = this.elementRef.nativeElement.querySelector('.datepicker-container > .calendar-container');
if (container &&
container !== e.target &&
!container.contains(e.target) &&
!e.target.classList.contains('year-unit')) {
this.isOpened = false;
}
}
}
DatepickerComponent.decorators = [
{ type: Component, args: [{
selector: 'ngx-datepicker',
template: "<div class=\"datepicker-container\" [class]=\"options.calendarClass\">\n <input\n type=\"text\"\n [(ngModel)]=\"displayValue\"\n [class]=\"options.inputClass\"\n [placeholder]=\"options.placeholder\"\n readonly\n (click)=\"toggle()\"\n />\n\n <div class=\"calendar-container\" *ngIf=\"isOpened\">\n <div class=\"top-container\">\n <div class=\"month-year-container\">\n <span class=\"month-year-text\">\n <span (click)=\"toggleView()\">{{ title }}</span>\n </span>\n </div>\n <div class=\"controls\">\n <svg\n xmlns=\"http://www.w3.org/2000/svg\"\n class=\"control prev-month\"\n width=\"8\"\n height=\"13\"\n (click)=\"prevMonth()\"\n *ngIf=\"view === 'days'\"\n >\n <path\n d=\"M7.575 1.131a1.063 1.063 0 00-1.502 0l-4.93 4.93c-.42.42-.42 1.099 0 1.518l4.93 4.93a1.063 1.063 0 001.503-1.503L3.388 6.82l4.186-4.186a1.063 1.063 0 000-1.503z\"\n />\n </svg>\n <svg\n xmlns=\"http://www.w3.org/2000/svg\"\n class=\"control next-month\"\n width=\"8\"\n height=\"13\"\n (click)=\"nextMonth()\"\n *ngIf=\"view === 'days'\"\n >\n <path\n d=\"M1.14 1.131a1.063 1.063 0 011.502 0l4.93 4.93c.42.42.42 1.099 0 1.518l-4.93 4.93a1.063 1.063 0 01-1.503-1.503L5.327 6.82 1.14 2.634a1.063 1.063 0 010-1.503z\"\n />\n </svg>\n </div>\n </div>\n <div class=\"main-calendar-container is-days\" *ngIf=\"view === 'days'\">\n <div class=\"main-calendar-day-names\">\n <span class=\"day-name-unit\" *ngFor=\"let day of dayNames\">{{ day }}</span>\n </div>\n <div class=\"main-calendar-days\">\n <span\n class=\"day-unit\"\n *ngFor=\"let day of days; let i = index\"\n [ngClass]=\"{\n 'is-prev-month': !day.inThisMonth,\n 'is-today': day.isToday,\n 'is-selected': day.isSelected,\n 'is-disabled': !day.isSelectable\n }\"\n (click)=\"day.isSelectable && setDate(i)\"\n >{{ day.day }}</span\n >\n </div>\n </div>\n <div class=\"main-calendar-container is-years\" *ngIf=\"view === 'years'\">\n <div class=\"main-calendar-years\" slimScroll [options]=\"scrollOptions\" [scrollEvents]=\"scrollEvents\">\n <span\n class=\"year-unit\"\n *ngFor=\"let year of years; let i = index\"\n [ngClass]=\"{ 'is-selected': year.isThisYear }\"\n (click)=\"setYear(i)\"\n >{{ year.year }}</span\n >\n </div>\n </div>\n </div>\n</div>\n",
providers: [{ provide: NG_VALUE_ACCESSOR, useExisting: DatepickerComponent, multi: true }],
styles: [".datepicker-container{position:relative;display:inline-block;box-sizing:border-box}.datepicker-container *{box-sizing:inherit}.datepicker-container input{-webkit-touch-callout:none;-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none;user-select:none;cursor:pointer}.datepicker-container .is-hidden{display:none!important}.datepicker-container .calendar-container{position:absolute;width:320px;border-radius:4px;top:35px;left:0;z-index:10}.datepicker-container .calendar-container .top-container{-webkit-touch-callout:none;-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none;user-select:none;width:100%;padding:20px;display:flex;justify-content:space-between;align-items:center}.datepicker-container .calendar-container .top-container .month-year-container{height:100%;display:flex;align-items:center;cursor:pointer;position:relative}.datepicker-container .calendar-container .top-container .month-year-container .month-year-text{font-size:16px;width:100%}.datepicker-container .calendar-container .top-container .controls{height:100%;display:flex;justify-content:flex-end;align-items:center}.datepicker-container .calendar-container .top-container .controls .control{cursor:pointer;margin-left:20px}.datepicker-container .calendar-container .main-calendar-container{width:100%;height:100%;font-size:12px;display:block}.datepicker-container .calendar-container .main-calendar-container .main-calendar-day-names{-webkit-touch-callout:none;-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none;user-select:none;display:flex;align-items:center;height:35px;padding:0 10px}.datepicker-container .calendar-container .main-calendar-container .main-calendar-day-names .day-name-unit{width:calc(100% / 7);text-transform:uppercase;text-align:center}.datepicker-container .calendar-container .main-calendar-container .main-calendar-days,.datepicker-container .calendar-container .main-calendar-container .main-calendar-years{-webkit-touch-callout:none;-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none;user-select:none;padding:10px;width:100%;overflow:hidden;text-align:left;display:inline-block}.datepicker-container .calendar-container .main-calendar-container .main-calendar-days.is-hidden,.datepicker-container .calendar-container .main-calendar-container .main-calendar-years.is-hidden{display:none}.datepicker-container .calendar-container .main-calendar-container .main-calendar-days .day-unit,.datepicker-container .calendar-container .main-calendar-container .main-calendar-days .year-unit,.datepicker-container .calendar-container .main-calendar-container .main-calendar-years .day-unit,.datepicker-container .calendar-container .main-calendar-container .main-calendar-years .year-unit{width:calc(100% / 7);display:inline-flex;align-items:center;justify-content:center;cursor:pointer;-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none;user-select:none;font-size:13px;border-radius:50%}.datepicker-container .calendar-container .main-calendar-container .main-calendar-days .day-unit:before,.datepicker-container .calendar-container .main-calendar-container .main-calendar-days .year-unit:before,.datepicker-container .calendar-container .main-calendar-container .main-calendar-years .day-unit:before,.datepicker-container .calendar-container .main-calendar-container .main-calendar-years .year-unit:before{content:\"\";float:left;padding-top:calc(100% - 5px)}.datepicker-container .calendar-container .main-calendar-container .main-calendar-days .day-unit.is-disabled,.datepicker-container .calendar-container .main-calendar-container .main-calendar-days .year-unit.is-disabled,.datepicker-container .calendar-container .main-calendar-container .main-calendar-years .day-unit.is-disabled,.datepicker-container .calendar-container .main-calendar-container .main-calendar-years .year-unit.is-disabled{cursor:not-allowed}.datepicker-container .calendar-container .main-calendar-container .main-calendar-days .day-unit.is-disabled:hover,.datepicker-container .calendar-container .main-calendar-container .main-calendar-days .year-unit.is-disabled:hover,.datepicker-container .calendar-container .main-calendar-container .main-calendar-years .day-unit.is-disabled:hover,.datepicker-container .calendar-container .main-calendar-container .main-calendar-years .year-unit.is-disabled:hover{background:transparent}.datepicker-container .calendar-container .main-calendar-container .main-calendar-years{display:block;padding:10px;height:275px}.datepicker-container .calendar-container .main-calendar-container .main-calendar-years .year-unit{width:calc(100% / 4);height:35px;border-radius:20px;margin:8px 0}.datepicker-container .calendar-container .main-calendar-container .main-calendar-years .year-unit:before{padding-top:0}.datepicker-default .calendar-container{background:#fff;border:1px solid #eaedf3;box-shadow:0 4px 12px rgba(0,0,0,.05);top:35px;left:0;font-weight:600}.datepicker-default .month-year-text{color:#010001}.datepicker-default .control path{fill:#aaa8ab}.datepicker-default .control:hover path{fill:#010001}.datepicker-default .main-calendar-day-names{border-bottom:1px solid #eaedf3}.datepicker-default .day-name-unit{color:#aaa8ab}.datepicker-default .day-unit,.datepicker-default .year-unit{color:#010001}.datepicker-default .day-unit.is-prev-month,.datepicker-default .year-unit.is-prev-month{color:#bbbabe}.datepicker-default .day-unit.is-today,.datepicker-default .year-unit.is-today{background:#edeef2}.datepicker-default .day-unit.is-selected,.datepicker-default .day-unit:hover,.datepicker-default .year-unit.is-selected,.datepicker-default .year-unit:hover{background:#010001;color:#fff}.datepicker-default .day-unit.is-disabled,.datepicker-default .year-unit.is-disabled{color:#aaa8ab}.datepicker-default .day-unit.is-disabled:hover,.datepicker-default .year-unit.is-disabled:hover{background:transparent}"]
},] }
];
DatepickerComponent.ctorParameters = () => [
{ type: ElementRef },
{ type: ChangeDetectorRef },
{ type: undefined, decorators: [{ type: Inject, args: [DOCUMENT,] }] }
];
DatepickerComponent.propDecorators = {
options: [{ type: Input }],
scrollOptions: [{ type: Input }],
isOpened: [{ type: Input }],
onBlur: [{ type: HostListener, args: ['document:click', ['$event'],] }]
};
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"datepicker.component.js","sourceRoot":"","sources":["../../../../projects/datepicker/src/lib/datepicker.component.ts"],"names":[],"mappings":"AAAA,OAAO,EACL,SAAS,EAET,KAAK,EAGL,YAAY,EACZ,UAAU,EACV,YAAY,EAEZ,MAAM,EACN,iBAAiB,EAClB,MAAM,eAAe,CAAC;AACvB,OAAO,EAAE,QAAQ,EAAE,MAAM,iBAAiB,CAAC;AAC3C,OAAO,EAAE,iBAAiB,EAAwB,MAAM,gBAAgB,CAAC;AACzE,OAAO,EAAwC,iBAAiB,EAAE,eAAe,EAAE,MAAM,gBAAgB,CAAC;AAC1G,OAAO,EAAqB,sBAAsB,EAAE,cAAc,EAAE,MAAM,gCAAgC,CAAC;AAC3G,OAAO,EACL,iBAAiB,EACjB,YAAY,EACZ,UAAU,EACV,OAAO,EACP,QAAQ,EACR,OAAO,EACP,OAAO,EACP,SAAS,EACT,WAAW,EACX,UAAU,EACV,QAAQ,EACR,OAAO,EACP,MAAM,EACN,OAAO,EACP,MAAM,EACN,MAAM,EACN,SAAS,EACT,SAAS,EACT,OAAO,EACP,QAAQ,EACR,QAAQ,EACT,MAAM,UAAU,CAAC;AAClB,OAAO,EAAE,SAAS,EAAE,YAAY,EAAE,MAAM,MAAM,CAAC;AAC/C,OAAO,EAAE,MAAM,EAAE,MAAM,gBAAgB,CAAC;AAmBxC,MAAM,OAAO,mBAAmB;IA0B9B,YAAmB,UAAsB,EAAU,GAAsB,EAAoB,QAAc;QAAxF,eAAU,GAAV,UAAU,CAAY;QAAU,QAAG,GAAH,GAAG,CAAmB;QAzBhE,YAAO,qBAA2B,cAAc,EAAG;QACnD,kBAAa,GAAsB,IAAI,iBAAiB,CAAC,IAAI,CAAC,gBAAgB,CAAC,CAAC;QAChF,aAAQ,GAAG,KAAK,CAAC;QAE1B,eAAU,GAAS,IAAI,IAAI,EAAE,CAAC;QAC9B,iBAAY,GAAG,EAAE,CAAC;QAClB,SAAI,GAAqB,MAAM,CAAC;QAChC,SAAI,GAAS,IAAI,IAAI,EAAE,CAAC;QACxB,UAAK,GAA4C,EAAE,CAAC;QACpD,SAAI,GAAU,EAAE,CAAC;QACjB,aAAQ,GAAa,EAAE,CAAC;QACxB,iBAAY,GAAG,IAAI,YAAY,EAAoB,CAAC;QACpD,QAAG,GAAiB,IAAI,YAAY,EAAE,CAAC;QAoO/B,sBAAiB,GAAe,GAAG,EAAE,GAAE,CAAC,CAAC;QACzC,qBAAgB,GAAqB,GAAG,EAAE,GAAE,CAAC,CAAC;QAvNpD,IAAI,CAAC,GAAG,GAAG,QAAoB,CAAC;IAClC,CAAC;IAZD,IAAI,KAAK;QACP,OAAO,IAAI,CAAC,UAAU,CAAC;IACzB,CAAC;IAED,IAAI,KAAK,CAAC,GAAS;QACjB,IAAI,CAAC,UAAU,GAAG,GAAG,CAAC;QACtB,IAAI,CAAC,YAAY,GAAG,MAAM,CAAC,IAAI,CAAC,UAAU,EAAE,IAAI,CAAC,OAAO,CAAC,MAAgB,EAAE,EAAE,MAAM,EAAE,IAAI,CAAC,OAAO,CAAC,MAAM,EAAE,CAAC,CAAC;QAC5G,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC;IACzC,CAAC;IAMD,IAAI,KAAK;QACP,OAAO,MAAM,CAAC,IAAI,CAAC,IAAI,EAAE,IAAI,CAAC,OAAO,CAAC,WAAqB,CAAC,CAAC;IAC/D,CAAC;IAED,IAAY,gBAAgB;QAC1B,OAAO;YACL,aAAa,EAAE,CAAC,IAAI,CAAC,OAAO,IAAI,IAAI,CAAC,OAAO,CAAC,cAAc,CAAC,IAAI,SAAS;YACzE,cAAc,EAAE,aAAa;YAC7B,eAAe,EAAE,GAAG;YACpB,gBAAgB,EAAE,GAAG;YACrB,QAAQ,EAAE,GAAG;YACb,SAAS,EAAE,GAAG;YACd,SAAS,EAAE,GAAG;YACd,UAAU,EAAE,GAAG;SAChB,CAAC;IACJ,CAAC;IAED,QAAQ;QACN,IAAI,CAAC,IAAI,GAAG,MAAM,CAAC;QACnB,IAAI,CAAC,IAAI,GAAG,IAAI,IAAI,EAAE,CAAC;QACvB,IAAI,CAAC,IAAI,EAAE,CAAC;IACd,CAAC;IAED,WAAW,CAAC,OAAsB;QAChC,IAAI,SAAS,IAAI,OAAO,EAAE;YACxB,IAAI,CAAC,OAAO,GAAG,sBAAsB,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;YACpD,IAAI,CAAC,aAAa,GAAG,IAAI,iBAAiB,CAAC,IAAI,CAAC,gBAAgB,CAAC,CAAC;YAElE,IAAI,IAAI,CAAC,GAAG,EAAE;gBACZ,IAAI,CAAC,GAAG,CAAC,WAAW,EAAE,CAAC;aACxB;YAED,IAAI,IAAI,CAAC,OAAO,CAAC,cAAc,EAAE;gBAC/B,IAAI,CAAC,GAAG,GAAG,SAAS,CAAgB,IAAI,CAAC,GAAG,IAAI,QAAQ,EAAE,OAAO,CAAC;qBAC/D,IAAI,CAAC,MAAM,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;qBACjC,SAAS,CAAC,CAAC,CAAC,EAAE;oBACb,CAAC,CAAC,cAAc,EAAE,CAAC;oBACnB,CAAC,CAAC,eAAe,EAAE,CAAC;oBAEpB,QAAQ,CAAC,CAAC,GAAG,EAAE;wBACb,KAAK,MAAM,CAAC;wBACZ,KAAK,WAAW;4BACd,IAAI,CAAC,QAAQ,EAAE,CAAC;4BAChB,MAAM;wBACR,KAAK,IAAI,CAAC;wBACV,KAAK,SAAS;4BACZ,IAAI,CAAC,QAAQ,EAAE,CAAC;4BAChB,MAAM;wBACR,KAAK,MAAM,CAAC;wBACZ,KAAK,WAAW;4BACd,IAAI,CAAC,SAAS,EAAE,CAAC;4BACjB,MAAM;wBACR,KAAK,OAAO,CAAC;wBACb,KAAK,YAAY;4BACf,IAAI,CAAC,SAAS,EAAE,CAAC;4BACjB,MAAM;wBACR,KAAK,KAAK,CAAC;wBACX,KAAK,QAAQ,CAAC;wBACd,KAAK,OAAO;4BACV,IAAI,CAAC,QAAQ,GAAG,KAAK,CAAC;4BACtB,MAAM;wBACR;4BACE,OAAO;qBACV;gBACH,CAAC,CAAC,CAAC;aACN;SACF;IACH,CAAC;IAED,WAAW;QACT,IAAI,CAAC,GAAG,CAAC,WAAW,EAAE,CAAC;IACzB,CAAC;IAED,MAAM;QACJ,IAAI,CAAC,QAAQ,GAAG,CAAC,IAAI,CAAC,QAAQ,CAAC;QAC/B,IAAI,IAAI,CAAC,QAAQ,EAAE;YACjB,IAAI,CAAC,IAAI,GAAG,MAAM,CAAC;YACnB,IAAI,CAAC,IAAI,GAAG,IAAI,CAAC,KAAK,CAAC;YACvB,IAAI,CAAC,QAAQ,EAAE,CAAC;SACjB;IACH,CAAC;IAED,UAAU;QACR,IAAI,CAAC,IAAI,GAAG,IAAI,CAAC,IAAI,KAAK,MAAM,CAAC,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,MAAM,CAAC;QACpD,IAAI,IAAI,CAAC,IAAI,KAAK,OAAO,EAAE;YACzB,IAAI,CAAC,GAAG,CAAC,aAAa,EAAE,CAAC;YACzB,IAAI,CAAC,YAAY,EAAE,CAAC;SACrB;IACH,CAAC;IAED,SAAS;QACP,IAAI,CAAC,IAAI,GAAG,SAAS,CAAC,IAAI,CAAC,IAAI,EAAE,CAAC,CAAC,CAAC;QACpC,IAAI,CAAC,QAAQ,EAAE,CAAC;IAClB,CAAC;IAED,SAAS;QACP,IAAI,CAAC,IAAI,GAAG,SAAS,CAAC,IAAI,CAAC,IAAI,EAAE,CAAC,CAAC,CAAC;QACpC,IAAI,CAAC,QAAQ,EAAE,CAAC;IAClB,CAAC;IAED,QAAQ;QACN,IAAI,CAAC,IAAI,GAAG,QAAQ,CAAC,IAAI,CAAC,IAAI,EAAE,CAAC,CAAC,CAAC;QACnC,IAAI,CAAC,QAAQ,EAAE,CAAC;IAClB,CAAC;IAED,QAAQ;QACN,IAAI,CAAC,IAAI,GAAG,QAAQ,CAAC,IAAI,CAAC,IAAI,EAAE,CAAC,CAAC,CAAC;QACnC,IAAI,CAAC,QAAQ,EAAE,CAAC;IAClB,CAAC;IAED,OAAO,CAAC,CAAS;QACf,IAAI,CAAC,IAAI,GAAG,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC;QAC9B,IAAI,CAAC,KAAK,GAAG,IAAI,CAAC,IAAI,CAAC;QACvB,IAAI,CAAC,QAAQ,EAAE,CAAC;QAChB,IAAI,CAAC,QAAQ,GAAG,KAAK,CAAC;IACxB,CAAC;IAED,OAAO,CAAC,CAAS;QACf,IAAI,CAAC,IAAI,GAAG,OAAO,CAAC,IAAI,CAAC,IAAI,EAAE,IAAI,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC;QACnD,IAAI,CAAC,QAAQ,EAAE,CAAC;QAChB,IAAI,CAAC,SAAS,EAAE,CAAC;QACjB,IAAI,CAAC,IAAI,GAAG,MAAM,CAAC;IACrB,CAAC;IAEO,YAAY;QAClB,MAAM,MAAM,GAAG,IAAI,CAAC,UAAU,CAAC,aAAa,CAAC,aAAa,CAAC,sBAAsB,CAAC,CAAC;QACnF,MAAM,EAAE,GAAG,IAAI,CAAC,UAAU,CAAC,aAAa,CAAC,aAAa,CAAC,wBAAwB,CAAC,CAAC;QACjF,MAAM,CAAC,GAAG,EAAE,CAAC,SAAS,GAAG,MAAM,CAAC,YAAY,GAAG,CAAC,GAAG,EAAE,CAAC,YAAY,GAAG,CAAC,CAAC;QACvE,MAAM,KAAK,GAAG,IAAI,eAAe,CAAC,EAAE,IAAI,EAAE,UAAU,EAAE,CAAC,EAAE,QAAQ,EAAE,GAAG,EAAE,CAAC,CAAC;QAC1E,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;IAChC,CAAC;IAEO,IAAI;QACV,IAAI,CAAC,YAAY,EAAE,CAAC;QACpB,IAAI,CAAC,QAAQ,EAAE,CAAC;QAChB,IAAI,CAAC,SAAS,EAAE,CAAC;IACnB,CAAC;IAEO,QAAQ;QACd,MAAM,IAAI,GAAG,IAAI,CAAC,IAAI,IAAI,IAAI,IAAI,EAAE,CAAC;QACrC,MAAM,CAAC,KAAK,EAAE,GAAG,CAAC,GAAG,CAAC,YAAY,CAAC,IAAI,CAAC,EAAE,UAAU,CAAC,IAAI,CAAC,CAAC,CAAC;QAE5D,IAAI,CAAC,IAAI,GAAG,iBAAiB,CAAC,EAAE,KAAK,EAAE,GAAG,EAAE,CAAC,CAAC,GAAG,CAAC,CAAC,CAAO,EAAE,EAAE,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC,CAAC,CAAC,CAAC;QAEpF,MAAM,GAAG,GAAG,MAAM,CAAC,KAAK,CAAC,GAAI,IAAI,CAAC,OAAO,CAAC,gBAA2B,CAAC;QACtE,MAAM,QAAQ,GAAG,GAAG,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC,GAAI,IAAI,CAAC,OAAO,CAAC,gBAA2B,CAAC,CAAC,CAAC,GAAG,CAAC;QAC/E,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,IAAI,QAAQ,EAAE,CAAC,EAAE,EAAE;YAClC,MAAM,CAAC,GAAG,OAAO,CAAC,KAAK,EAAE,CAAC,CAAC,CAAC;YAC5B,IAAI,CAAC,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC,EAAE,KAAK,CAAC,CAAC,CAAC;SAC/C;IACH,CAAC;IAEO,SAAS;QACf,MAAM,KAAK,GAAI,IAAI,CAAC,OAAO,CAAC,OAAkB,GAAI,IAAI,CAAC,OAAO,CAAC,OAAkB,GAAG,CAAC,CAAC;QACtF,IAAI,CAAC,KAAK,GAAG,KAAK,CAAC,IAAI,CAAC,IAAI,KAAK,CAAC,KAAK,CAAC,EAAE,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC,CAAC,GAAI,IAAI,CAAC,OAAO,CAAC,OAAkB,CAAC,CAAC,GAAG,CAAC,IAAI,CAAC,EAAE;YACnG,OAAO,EAAE,IAAI,EAAE,UAAU,EAAE,IAAI,KAAK,OAAO,CAAC,IAAI,CAAC,IAAI,CAAC,EAAE,CAAC;QAC3D,CAAC,CAAC,CAAC;IACL,CAAC;IAEO,YAAY;QAClB,IAAI,CAAC,QAAQ,GAAG,EAAE,CAAC;QACnB,MAAM,KAAK,GAAG,IAAI,CAAC,OAAO,CAAC,gBAA0B,CAAC;QACtD,KAAK,IAAI,CAAC,GAAG,KAAK,EAAE,CAAC,IAAI,CAAC,GAAG,KAAK,EAAE,CAAC,EAAE,EAAE;YACvC,MAAM,IAAI,GAAG,MAAM,CAAC,IAAI,IAAI,EAAE,EAAE,CAAC,CAAC,CAAC;YACnC,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,MAAM,CAAC,IAAI,EAAE,IAAI,CAAC,OAAO,CAAC,UAAoB,EAAE,EAAE,MAAM,EAAE,IAAI,CAAC,OAAO,CAAC,MAAM,EAAE,CAAC,CAAC,CAAC;SACtG;IACH,CAAC;IAEO,WAAW,CAAC,IAAU,EAAE,cAAuB,IAAI;QACzD,OAAO;YACL,IAAI;YACJ,GAAG,EAAE,OAAO,CAAC,IAAI,CAAC;YAClB,KAAK,EAAE,QAAQ,CAAC,IAAI,CAAC;YACrB,IAAI,EAAE,OAAO,CAAC,IAAI,CAAC;YACnB,WAAW;YACX,OAAO,EAAE,OAAO,CAAC,IAAI,CAAC;YACtB,UAAU,EACR,SAAS,CAAC,IAAI,EAAE,IAAI,CAAC,UAAU,CAAC,IAAI,WAAW,CAAC,IAAI,EAAE,IAAI,CAAC,UAAU,CAAC,IAAI,UAAU,CAAC,IAAI,EAAE,IAAI,CAAC,UAAU,CAAC;YAC7G,YAAY,EAAE,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC;SAC1C,CAAC;IACJ,CAAC;IAEO,gBAAgB,CAAC,IAAU;QACjC,IAAI,IAAI,CAAC,OAAO,CAAC,OAAO,IAAI,QAAQ,CAAC,IAAI,EAAE,IAAI,CAAC,OAAO,CAAC,OAAO,CAAC,EAAE;YAChE,OAAO,KAAK,CAAC;SACd;QAED,IAAI,IAAI,CAAC,OAAO,CAAC,OAAO,IAAI,OAAO,CAAC,IAAI,EAAE,IAAI,CAAC,OAAO,CAAC,OAAO,CAAC,EAAE;YAC/D,OAAO,KAAK,CAAC;SACd;QAED,OAAO,IAAI,CAAC;IACd,CAAC;IAED,UAAU,CAAC,GAAS;QAClB,IAAI,CAAC,GAAG,EAAE;YACR,OAAO;SACR;QACD,IAAI,CAAC,UAAU,GAAG,GAAG,CAAC;QACtB,IAAI,CAAC,YAAY,GAAG,MAAM,CAAC,IAAI,CAAC,UAAU,EAAE,IAAI,CAAC,OAAO,CAAC,MAAgB,EAAE,EAAE,MAAM,EAAE,IAAI,CAAC,OAAO,CAAC,MAAM,EAAE,CAAC,CAAC;QAC5G,IAAI,CAAC,IAAI,EAAE,CAAC;IACd,CAAC;IAED,gBAAgB,CAAC,EAAO;QACtB,IAAI,CAAC,gBAAgB,GAAG,EAAE,CAAC;IAC7B,CAAC;IAED,iBAAiB,CAAC,EAAO;QACvB,IAAI,CAAC,iBAAiB,GAAG,EAAE,CAAC;IAC9B,CAAC;IAK2C,MAAM,CAAC,CAAa;QAC9D,IAAI,CAAC,IAAI,CAAC,QAAQ,EAAE;YAClB,OAAO;SACR;QAED,MAAM,KAAK,GAAG,IAAI,CAAC,UAAU,CAAC,aAAa,CAAC,aAAa,CAAC,+BAA+B,CAAC,CAAC;QAC3F,IAAI,CAAC,KAAK,IAAI,CAAC,CAAC,MAAM,KAAK,KAAK,IAAI,KAAK,CAAC,QAAQ,CAAC,CAAC,CAAC,MAAM,CAAC,EAAE;YAC5D,OAAO;SACR;QAED,MAAM,SAAS,GAAG,IAAI,CAAC,UAAU,CAAC,aAAa,CAAC,aAAa,CAAC,6CAA6C,CAAC,CAAC;QAC7G,IACE,SAAS;YACT,SAAS,KAAK,CAAC,CAAC,MAAM;YACtB,CAAC,SAAS,CAAC,QAAQ,CAAC,CAAC,CAAC,MAAM,CAAC;YAC7B,CAAE,CAAC,CAAC,MAAsB,CAAC,SAAS,CAAC,QAAQ,CAAC,WAAW,CAAC,EAC1D;YACA,IAAI,CAAC,QAAQ,GAAG,KAAK,CAAC;SACvB;IACH,CAAC;;;YA7QF,SAAS,SAAC;gBACT,QAAQ,EAAE,gBAAgB;gBAC1B,+oFAA0C;gBAE1C,SAAS,EAAE,CAAC,EAAE,OAAO,EAAE,iBAAiB,EAAE,WAAW,EAAE,mBAAmB,EAAE,KAAK,EAAE,IAAI,EAAE,CAAC;;aAC3F;;;YApDC,UAAU;YAIV,iBAAiB;4CA2E2D,MAAM,SAAC,QAAQ;;;sBAzB1F,KAAK;4BACL,KAAK;uBACL,KAAK;qBAiPL,YAAY,SAAC,gBAAgB,EAAE,CAAC,QAAQ,CAAC","sourcesContent":["import {\n  Component,\n  OnInit,\n  Input,\n  OnChanges,\n  SimpleChanges,\n  HostListener,\n  ElementRef,\n  EventEmitter,\n  OnDestroy,\n  Inject,\n  ChangeDetectorRef\n} from '@angular/core';\nimport { DOCUMENT } from '@angular/common';\nimport { NG_VALUE_ACCESSOR, ControlValueAccessor } from '@angular/forms';\nimport { ISlimScrollEvent, ISlimScrollOptions, SlimScrollOptions, SlimScrollEvent } from 'ngx-slimscroll';\nimport { DatepickerOptions, mergeDatepickerOptions, defaultOptions } from './datepicker-options.interface';\nimport {\n  eachDayOfInterval,\n  startOfMonth,\n  endOfMonth,\n  getDate,\n  getMonth,\n  getYear,\n  isToday,\n  isSameDay,\n  isSameMonth,\n  isSameYear,\n  isBefore,\n  isAfter,\n  getDay,\n  subDays,\n  setDay,\n  format,\n  addMonths,\n  subMonths,\n  setYear,\n  addYears,\n  subYears\n} from 'date-fns';\nimport { fromEvent, Subscription } from 'rxjs';\nimport { filter } from 'rxjs/operators';\n\ninterface Day {\n  date: Date;\n  day: number;\n  month: number;\n  year: number;\n  inThisMonth: boolean;\n  isToday: boolean;\n  isSelected: boolean;\n  isSelectable: boolean;\n}\n\n@Component({\n  selector: 'ngx-datepicker',\n  templateUrl: './datepicker.component.html',\n  styleUrls: ['./datepicker.component.sass'],\n  providers: [{ provide: NG_VALUE_ACCESSOR, useExisting: DatepickerComponent, multi: true }]\n})\nexport class DatepickerComponent implements ControlValueAccessor, OnInit, OnChanges, OnDestroy {\n  @Input() options: DatepickerOptions = { ...defaultOptions };\n  @Input() scrollOptions: SlimScrollOptions = new SlimScrollOptions(this.scrollBarOptions);\n  @Input() isOpened = false;\n\n  innerValue: Date = new Date();\n  displayValue = '';\n  view: 'days' | 'years' = 'days';\n  date: Date = new Date();\n  years: { year: number; isThisYear: boolean }[] = [];\n  days: Day[] = [];\n  dayNames: string[] = [];\n  scrollEvents = new EventEmitter<ISlimScrollEvent>();\n  sub: Subscription = new Subscription();\n  private doc?: Document;\n\n  get value(): Date {\n    return this.innerValue;\n  }\n\n  set value(val: Date) {\n    this.innerValue = val;\n    this.displayValue = format(this.innerValue, this.options.format as string, { locale: this.options.locale });\n    this.onChangeCallback(this.innerValue);\n  }\n\n  constructor(public elementRef: ElementRef, private ref: ChangeDetectorRef, @Inject(DOCUMENT) document?: any) {\n    this.doc = document as Document;\n  }\n\n  get title(): string {\n    return format(this.date, this.options.formatTitle as string);\n  }\n\n  private get scrollBarOptions(): ISlimScrollOptions {\n    return {\n      barBackground: (this.options && this.options.scrollBarColor) || '#dfe3e9',\n      gridBackground: 'transparent',\n      barBorderRadius: '3',\n      gridBorderRadius: '3',\n      barWidth: '6',\n      gridWidth: '6',\n      barMargin: '0',\n      gridMargin: '0'\n    };\n  }\n\n  ngOnInit(): void {\n    this.view = 'days';\n    this.date = new Date();\n    this.init();\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if ('options' in changes) {\n      this.options = mergeDatepickerOptions(this.options);\n      this.scrollOptions = new SlimScrollOptions(this.scrollBarOptions);\n\n      if (this.sub) {\n        this.sub.unsubscribe();\n      }\n\n      if (this.options.enableKeyboard) {\n        this.sub = fromEvent<KeyboardEvent>(this.doc || document, 'keyup')\n          .pipe(filter(() => this.isOpened))\n          .subscribe(e => {\n            e.preventDefault();\n            e.stopPropagation();\n\n            switch (e.key) {\n              case 'Down':\n              case 'ArrowDown':\n                this.prevYear();\n                break;\n              case 'Up':\n              case 'ArrowUp':\n                this.nextYear();\n                break;\n              case 'Left':\n              case 'ArrowLeft':\n                this.prevMonth();\n                break;\n              case 'Right':\n              case 'ArrowRight':\n                this.nextMonth();\n                break;\n              case 'Esc':\n              case 'Escape':\n              case 'Enter':\n                this.isOpened = false;\n                break;\n              default:\n                return;\n            }\n          });\n      }\n    }\n  }\n\n  ngOnDestroy(): void {\n    this.sub.unsubscribe();\n  }\n\n  toggle(): void {\n    this.isOpened = !this.isOpened;\n    if (this.isOpened) {\n      this.view = 'days';\n      this.date = this.value;\n      this.initDays();\n    }\n  }\n\n  toggleView(): void {\n    this.view = this.view === 'days' ? 'years' : 'days';\n    if (this.view === 'years') {\n      this.ref.detectChanges();\n      this.scrollToYear();\n    }\n  }\n\n  nextMonth(): void {\n    this.date = addMonths(this.date, 1);\n    this.initDays();\n  }\n\n  prevMonth(): void {\n    this.date = subMonths(this.date, 1);\n    this.initDays();\n  }\n\n  nextYear(): void {\n    this.date = addYears(this.date, 1);\n    this.initDays();\n  }\n\n  prevYear(): void {\n    this.date = subYears(this.date, 1);\n    this.initDays();\n  }\n\n  setDate(i: number): void {\n    this.date = this.days[i].date;\n    this.value = this.date;\n    this.initDays();\n    this.isOpened = false;\n  }\n\n  setYear(i: number): void {\n    this.date = setYear(this.date, this.years[i].year);\n    this.initDays();\n    this.initYears();\n    this.view = 'days';\n  }\n\n  private scrollToYear(): void {\n    const parent = this.elementRef.nativeElement.querySelector('.main-calendar-years');\n    const el = this.elementRef.nativeElement.querySelector('.year-unit.is-selected');\n    const y = el.offsetTop - parent.clientHeight / 2 + el.clientHeight / 2;\n    const event = new SlimScrollEvent({ type: 'scrollTo', y, duration: 100 });\n    this.scrollEvents.emit(event);\n  }\n\n  private init(): void {\n    this.initDayNames();\n    this.initDays();\n    this.initYears();\n  }\n\n  private initDays(): void {\n    const date = this.date || new Date();\n    const [start, end] = [startOfMonth(date), endOfMonth(date)];\n\n    this.days = eachDayOfInterval({ start, end }).map((d: Date) => this.generateDay(d));\n\n    const tmp = getDay(start) - (this.options.firstCalendarDay as number);\n    const prevDays = tmp < 0 ? 7 - (this.options.firstCalendarDay as number) : tmp;\n    for (let i = 1; i <= prevDays; i++) {\n      const d = subDays(start, i);\n      this.days.unshift(this.generateDay(d, false));\n    }\n  }\n\n  private initYears(): void {\n    const range = (this.options.maxYear as number) - (this.options.minYear as number) + 1;\n    this.years = Array.from(new Array(range), (_, i) => i + (this.options.minYear as number)).map(year => {\n      return { year, isThisYear: year === getYear(this.date) };\n    });\n  }\n\n  private initDayNames(): void {\n    this.dayNames = [];\n    const start = this.options.firstCalendarDay as number;\n    for (let i = start; i <= 6 + start; i++) {\n      const date = setDay(new Date(), i);\n      this.dayNames.push(format(date, this.options.formatDays as string, { locale: this.options.locale }));\n    }\n  }\n\n  private generateDay(date: Date, inThisMonth: boolean = true): Day {\n    return {\n      date,\n      day: getDate(date),\n      month: getMonth(date),\n      year: getYear(date),\n      inThisMonth,\n      isToday: isToday(date),\n      isSelected:\n        isSameDay(date, this.innerValue) && isSameMonth(date, this.innerValue) && isSameYear(date, this.innerValue),\n      isSelectable: this.isDateSelectable(date)\n    };\n  }\n\n  private isDateSelectable(date: Date): boolean {\n    if (this.options.minDate && isBefore(date, this.options.minDate)) {\n      return false;\n    }\n\n    if (this.options.maxDate && isAfter(date, this.options.maxDate)) {\n      return false;\n    }\n\n    return true;\n  }\n\n  writeValue(val: Date): void {\n    if (!val) {\n      return;\n    }\n    this.innerValue = val;\n    this.displayValue = format(this.innerValue, this.options.format as string, { locale: this.options.locale });\n    this.init();\n  }\n\n  registerOnChange(fn: any): void {\n    this.onChangeCallback = fn;\n  }\n\n  registerOnTouched(fn: any): void {\n    this.onTouchedCallback = fn;\n  }\n\n  private onTouchedCallback: () => void = () => {};\n  private onChangeCallback: (_: any) => void = () => {};\n\n  @HostListener('document:click', ['$event']) onBlur(e: MouseEvent): void {\n    if (!this.isOpened) {\n      return;\n    }\n\n    const input = this.elementRef.nativeElement.querySelector('.datepicker-container > input');\n    if (!input || e.target === input || input.contains(e.target)) {\n      return;\n    }\n\n    const container = this.elementRef.nativeElement.querySelector('.datepicker-container > .calendar-container');\n    if (\n      container &&\n      container !== e.target &&\n      !container.contains(e.target) &&\n      !(e.target as HTMLElement).classList.contains('year-unit')\n    ) {\n      this.isOpened = false;\n    }\n  }\n}\n"]}