UNPKG

ng2-datepicker

Version:

ng2-datepicker is simple and minimal Angular datepicker component. It is fully customizable.

253 lines 41.6 kB
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":"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","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"]}