UNPKG

@hxui/angular

Version:

An Angular library based on the [HXUI design system](https://hxui.io).

194 lines 43.8 kB
import { Component, EventEmitter, HostBinding, Input, Output } from '@angular/core'; import { BehaviorSubject } from 'rxjs'; import { Visibility } from '../enums'; import { DatepickerViewModeEnum } from './datepicker.model'; import * as i0 from "@angular/core"; import * as i1 from "@angular/common"; export class DatepickerComponent { constructor() { this.update = new EventEmitter(); this.viewMode$ = new BehaviorSubject(DatepickerViewModeEnum.Days); this.DatepickerViewModeEnum = DatepickerViewModeEnum; this.visibilityEnum = Visibility; this.visibility = Visibility.Hidden; this.days = new Array(); this.week = [ 'Monday', 'Tuesday', 'Wednesday', 'Thursday', 'Friday', 'Saturday', 'Sunday' ]; this.months = [ 'Jan', 'Feb', 'Mar', 'Apr', 'May', 'Jun', 'Jul', 'Aug', 'Sep', 'Oct', 'Nov', 'Dec' ]; this.years = new Array(); this.cellCount = 41; this.yearCellCount = 20; } get classes() { return 'hxui-reset hx-card hxa-datepicker-calendar'; } ngOnInit() { const date = this.selectedDate ? this.selectedDate : new Date(); this.presentDate = new Date(date.getFullYear(), date.getMonth(), date.getDate()); this.viewDate = this.viewDate || new Date(date.getFullYear(), date.getMonth()); this.renderCalendar(); } ngOnChanges(changes) { // update view date if (changes.selectedDate && changes.selectedDate.currentValue) { this.viewDate = new Date(this.selectedDate.getFullYear(), this.selectedDate.getMonth()); } } /** update and emit selected date */ setSelectedDate(date) { if (!this.isInvalidDay(date)) { this.selectedDate = date; this.update.emit(date); } } // Populates the days array with the current month, and completes the view with partial dates from sibling months renderCalendar() { for (let i = 0; i <= this.cellCount; i++) { // date will be set to the first day of the month set in this.viewDate const date = new Date(this.viewDate.getFullYear(), this.viewDate.getMonth()); // Shifts the week to start from Monday, rather than Sunday, this causes the index to start at 1 const dayOffset = date.getDay() === 0 ? 7 : date.getDay(); this.days[i] = new Date(date.setDate(2 - dayOffset + i)); } } next() { if (this.viewMode$.value === DatepickerViewModeEnum.Days) { this.nextMonth(); } else if (this.viewMode$.value === DatepickerViewModeEnum.Years) { this.nextYear(); } } previous() { if (this.viewMode$.value === DatepickerViewModeEnum.Days) { this.previousMonth(); } else if (this.viewMode$.value === DatepickerViewModeEnum.Years) { this.previousYear(); } } previousMonth() { this.viewDate = new Date(this.viewDate.getFullYear(), this.viewDate.getMonth() - 1); this.renderCalendar(); } nextMonth() { this.viewDate = new Date(this.viewDate.getFullYear(), this.viewDate.getMonth() + 1); this.renderCalendar(); } isCurrentMonth(inputDate) { return inputDate.getMonth() === this.viewDate.getMonth(); } isCurrentDay(inputDate) { return inputDate.getTime() === this.presentDate.getTime(); } isSelectedDay(inputDate) { if (this.selectedDate) { return inputDate.getTime() === this.selectedDate.getTime(); } return false; } isInvalidDay(inputDate) { return this.validators .map(fn => fn(inputDate)) .reduce((prev, next) => prev || next, false); } isCurrentYear(year) { return year === this.presentDate.getFullYear(); } isSelectedYear(year) { return year === this.viewDate.getFullYear(); } isInvalidYear(year) { const newDate = new Date(new Date(this.viewDate.getTime()).setFullYear(year)); return this.validators .map(fn => fn(newDate)) .reduce((prev, next) => prev || next, false); } isCurrentMonthByIndex(month) { return month === this.presentDate.getMonth(); } isSelectedMonthByIndex(month) { return month === this.viewDate.getMonth(); } isInvalidMonthByIndex(month) { const newDate = new Date(new Date(this.viewDate.getTime()).setMonth(month)); return this.validators .map(fn => fn(newDate)) .reduce((prev, next) => prev || next, false); } previousYear() { this.getYearCollection(this.years[0] - this.yearCellCount); } nextYear() { this.getYearCollection(this.years[0] + this.yearCellCount); } setYear(year) { if (!this.isInvalidYear(year)) { this.viewDate.setFullYear(year); this.renderCalendar(); this.toggleYear(); } } setMonth(month) { if (!this.isInvalidMonthByIndex(month)) { this.viewDate.setMonth(month); this.renderCalendar(); this.viewMode$.next(DatepickerViewModeEnum.Days); } } toggleYear() { this.viewMode$.next(this.viewMode$.value === DatepickerViewModeEnum.Years ? DatepickerViewModeEnum.Months : DatepickerViewModeEnum.Years); if (this.viewMode$.value === DatepickerViewModeEnum.Years) { this.getYearCollection(); } } getYearCollection(startFrom = null) { const yearsBeforeActive = 7; const activeYear = startFrom ? startFrom : this.viewDate.getFullYear() - yearsBeforeActive; this.years = []; for (let i = 0; i < this.yearCellCount; i++) { this.years.push(activeYear + i); } } } DatepickerComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.3.11", ngImport: i0, type: DatepickerComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); DatepickerComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.3.11", type: DatepickerComponent, selector: "hxa-datepicker", inputs: { selectedDate: "selectedDate", validators: "validators" }, outputs: { update: "update" }, host: { properties: { "class": "this.classes" } }, usesOnChanges: true, ngImport: i0, template: "<ng-container *ngIf=\"{ viewMode$ : viewMode$ | async } as observables\">\r\n <div class=\"hx-card-header hxa-datepicker-calendar__header\">\r\n <div class=\"hxa-datepicker-calendar__month\">\r\n <button\r\n class=\"hxa-datepicker-calendar__icon hx-button is-transparent is-large\"\r\n *ngIf=\"observables.viewMode$ !== DatepickerViewModeEnum.Months\"\r\n title=\"Previous Month\"\r\n (click)=\"previous()\"\r\n >\r\n <span class=\"hx-icon-control\">\r\n <i class=\"hx-icon icon-angle-left is-medium\"></i>\r\n </span>\r\n </button>\r\n <div class=\"hxa-datepicker-calendar__month-title\">\r\n <button\r\n class=\"hx-button is-flat is-info is-large\"\r\n (click)=\"toggleYear()\"\r\n >\r\n {{ (observables.viewMode$ === DatepickerViewModeEnum.Years) ? years[0]\r\n + ' - ' + years[years.length-1] : viewDate?.toLocaleString(\"en-au\", {\r\n month: \"long\", year: \"numeric\"}) }}\r\n </button>\r\n </div>\r\n <button\r\n class=\"hxa-datepicker-calendar__icon hx-button is-transparent is-large\"\r\n *ngIf=\"observables.viewMode$ !== DatepickerViewModeEnum.Months\"\r\n title=\"Next Month\"\r\n (click)=\"next()\"\r\n >\r\n <span class=\"hx-icon-control\">\r\n <i class=\"hx-icon icon-angle-right is-medium\"></i>\r\n </span>\r\n </button>\r\n </div>\r\n <div\r\n class=\"hxa-datepicker-calendar__week\"\r\n *ngIf=\"observables.viewMode$ === DatepickerViewModeEnum.Days\"\r\n >\r\n <div\r\n class=\"hxa-datepicker-calendar__weekday\"\r\n *ngFor=\"let weekday of week\"\r\n >\r\n {{weekday | slice:0:3}}\r\n </div>\r\n </div>\r\n </div>\r\n <div class=\"hxa-datepicker-calendar__contents\">\r\n <ng-container *ngIf=\"observables.viewMode$ === DatepickerViewModeEnum.Days\">\r\n <button\r\n class=\"hxa-datepicker-calendar__day hx-button is-transparent\"\r\n *ngFor=\"let day of days\"\r\n [ngClass]=\"{\r\n 'hxa-datepicker-calendar__day-siblingmonth': !isCurrentMonth(day),\r\n 'hxa-datepicker-calendar__day-currentday': isCurrentDay(day),\r\n 'hxa-datepicker-calendar__day-selectedday': isSelectedDay(day),\r\n 'hxa-datepicker-calendar__day-invalidday': isInvalidDay(day)\r\n }\"\r\n (click)=\"setSelectedDate(day)\"\r\n >\r\n {{day.getDate()}}\r\n </button>\r\n </ng-container>\r\n\r\n <ng-container\r\n *ngIf=\"observables.viewMode$ === DatepickerViewModeEnum.Months\"\r\n >\r\n <button\r\n class=\"hxa-datepicker-calendar__months hx-button is-transparent\"\r\n *ngFor=\"let month of months; let indexOfMth=index;\"\r\n [ngClass]=\"{\r\n 'hxa-datepicker-calendar__day-currentday': isCurrentMonthByIndex(indexOfMth),\r\n 'hxa-datepicker-calendar__day-selectedday': isSelectedMonthByIndex(indexOfMth),\r\n 'hxa-datepicker-calendar__day-invalidday': isInvalidMonthByIndex(indexOfMth)\r\n }\"\r\n (click)=\"setMonth(indexOfMth)\"\r\n >\r\n {{ month }}\r\n </button>\r\n </ng-container>\r\n\r\n <ng-container\r\n *ngIf=\"observables.viewMode$ === DatepickerViewModeEnum.Years\"\r\n >\r\n <button\r\n *ngFor=\"let year of years\"\r\n class=\"hxa-datepicker-calendar__year hx-button is-transparent\"\r\n [ngClass]=\"{\r\n 'hxa-datepicker-calendar__day-currentday': isCurrentYear(year),\r\n 'hxa-datepicker-calendar__day-selectedday': isSelectedYear(year),\r\n 'hxa-datepicker-calendar__day-invalidday': isInvalidYear(year)\r\n }\"\r\n (click)=\"setYear(year)\"\r\n >\r\n {{ year }}\r\n </button>\r\n </ng-container>\r\n </div>\r\n</ng-container>\r\n", styles: [":host.hx-card{border:unset}:host.hx-card:hover,:host.hx-card.is-hovered{box-shadow:none}.hxa-datepicker-calendar:host{width:19rem;height:21rem;display:flex;flex-direction:column;font-size:1rem;border:unset}.hxa-datepicker-calendar__header{padding:.5rem .5rem 0;flex-direction:column;align-items:stretch}.hxa-datepicker-calendar__icon.hx-button.is-transparent:hover{color:#000}.hxa-datepicker-calendar__icon .hx-icon{color:#0d4d78}.hxa-datepicker-calendar__month{display:flex;align-items:center;justify-content:space-between;text-align:center;margin-bottom:.5rem}.hxa-datepicker-calendar__month-title{font-size:1.25em;font-weight:400;flex:3;display:flex;justify-content:center;align-items:center;color:#0d4d78}.hxa-datepicker-calendar__week{display:flex;width:100%;text-align:center}.hxa-datepicker-calendar__weekday{flex:1;color:#0d4d78;font-size:.85em}.hxa-datepicker-calendar__contents{padding:.5rem;background-color:#f6f6f980;display:flex;flex-flow:row wrap;flex:1;justify-content:space-around;align-content:space-around}.hxa-datepicker-calendar__year{margin:.5rem}.hxa-datepicker-calendar__year.hx-button{padding:.5rem}.hxa-datepicker-calendar__months{margin:.7rem}.hxa-datepicker-calendar__months.hx-button{padding:.5rem;font-weight:500}.hxa-datepicker-calendar__day,.hxa-datepicker-calendar__year,.hxa-datepicker-calendar__months{flex:1 1 14%;height:16.666%;display:flex;justify-content:center;align-items:center}.hxa-datepicker-calendar__day.hx-button.is-transparent,.hxa-datepicker-calendar__year.hx-button.is-transparent,.hxa-datepicker-calendar__months.hx-button.is-transparent{font-weight:400;padding:0;color:#41b987}.hxa-datepicker-calendar__day-siblingmonth.hx-button.is-transparent,.hxa-datepicker-calendar__year-siblingmonth.hx-button.is-transparent,.hxa-datepicker-calendar__months-siblingmonth.hx-button.is-transparent{color:#3b3b3b;font-weight:100}.hxa-datepicker-calendar__day-selectedday.hx-button.is-transparent,.hxa-datepicker-calendar__year-selectedday.hx-button.is-transparent,.hxa-datepicker-calendar__months-selectedday.hx-button.is-transparent{color:#fff;background:#41b987}.hxa-datepicker-calendar__day-invalidday.hx-button.is-transparent,.hxa-datepicker-calendar__year-invalidday.hx-button.is-transparent,.hxa-datepicker-calendar__months-invalidday.hx-button.is-transparent{color:#e0e0e1;pointer-events:none}.hxa-datepicker-calendar__day-currentday.hx-button.is-transparent,.hxa-datepicker-calendar__year-currentday.hx-button.is-transparent,.hxa-datepicker-calendar__months-currentday.hx-button.is-transparent{border:2px solid #41b987}\n"], directives: [{ type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { type: i1.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }], pipes: { "async": i1.AsyncPipe, "slice": i1.SlicePipe } }); i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.3.11", ngImport: i0, type: DatepickerComponent, decorators: [{ type: Component, args: [{ selector: 'hxa-datepicker', template: "<ng-container *ngIf=\"{ viewMode$ : viewMode$ | async } as observables\">\r\n <div class=\"hx-card-header hxa-datepicker-calendar__header\">\r\n <div class=\"hxa-datepicker-calendar__month\">\r\n <button\r\n class=\"hxa-datepicker-calendar__icon hx-button is-transparent is-large\"\r\n *ngIf=\"observables.viewMode$ !== DatepickerViewModeEnum.Months\"\r\n title=\"Previous Month\"\r\n (click)=\"previous()\"\r\n >\r\n <span class=\"hx-icon-control\">\r\n <i class=\"hx-icon icon-angle-left is-medium\"></i>\r\n </span>\r\n </button>\r\n <div class=\"hxa-datepicker-calendar__month-title\">\r\n <button\r\n class=\"hx-button is-flat is-info is-large\"\r\n (click)=\"toggleYear()\"\r\n >\r\n {{ (observables.viewMode$ === DatepickerViewModeEnum.Years) ? years[0]\r\n + ' - ' + years[years.length-1] : viewDate?.toLocaleString(\"en-au\", {\r\n month: \"long\", year: \"numeric\"}) }}\r\n </button>\r\n </div>\r\n <button\r\n class=\"hxa-datepicker-calendar__icon hx-button is-transparent is-large\"\r\n *ngIf=\"observables.viewMode$ !== DatepickerViewModeEnum.Months\"\r\n title=\"Next Month\"\r\n (click)=\"next()\"\r\n >\r\n <span class=\"hx-icon-control\">\r\n <i class=\"hx-icon icon-angle-right is-medium\"></i>\r\n </span>\r\n </button>\r\n </div>\r\n <div\r\n class=\"hxa-datepicker-calendar__week\"\r\n *ngIf=\"observables.viewMode$ === DatepickerViewModeEnum.Days\"\r\n >\r\n <div\r\n class=\"hxa-datepicker-calendar__weekday\"\r\n *ngFor=\"let weekday of week\"\r\n >\r\n {{weekday | slice:0:3}}\r\n </div>\r\n </div>\r\n </div>\r\n <div class=\"hxa-datepicker-calendar__contents\">\r\n <ng-container *ngIf=\"observables.viewMode$ === DatepickerViewModeEnum.Days\">\r\n <button\r\n class=\"hxa-datepicker-calendar__day hx-button is-transparent\"\r\n *ngFor=\"let day of days\"\r\n [ngClass]=\"{\r\n 'hxa-datepicker-calendar__day-siblingmonth': !isCurrentMonth(day),\r\n 'hxa-datepicker-calendar__day-currentday': isCurrentDay(day),\r\n 'hxa-datepicker-calendar__day-selectedday': isSelectedDay(day),\r\n 'hxa-datepicker-calendar__day-invalidday': isInvalidDay(day)\r\n }\"\r\n (click)=\"setSelectedDate(day)\"\r\n >\r\n {{day.getDate()}}\r\n </button>\r\n </ng-container>\r\n\r\n <ng-container\r\n *ngIf=\"observables.viewMode$ === DatepickerViewModeEnum.Months\"\r\n >\r\n <button\r\n class=\"hxa-datepicker-calendar__months hx-button is-transparent\"\r\n *ngFor=\"let month of months; let indexOfMth=index;\"\r\n [ngClass]=\"{\r\n 'hxa-datepicker-calendar__day-currentday': isCurrentMonthByIndex(indexOfMth),\r\n 'hxa-datepicker-calendar__day-selectedday': isSelectedMonthByIndex(indexOfMth),\r\n 'hxa-datepicker-calendar__day-invalidday': isInvalidMonthByIndex(indexOfMth)\r\n }\"\r\n (click)=\"setMonth(indexOfMth)\"\r\n >\r\n {{ month }}\r\n </button>\r\n </ng-container>\r\n\r\n <ng-container\r\n *ngIf=\"observables.viewMode$ === DatepickerViewModeEnum.Years\"\r\n >\r\n <button\r\n *ngFor=\"let year of years\"\r\n class=\"hxa-datepicker-calendar__year hx-button is-transparent\"\r\n [ngClass]=\"{\r\n 'hxa-datepicker-calendar__day-currentday': isCurrentYear(year),\r\n 'hxa-datepicker-calendar__day-selectedday': isSelectedYear(year),\r\n 'hxa-datepicker-calendar__day-invalidday': isInvalidYear(year)\r\n }\"\r\n (click)=\"setYear(year)\"\r\n >\r\n {{ year }}\r\n </button>\r\n </ng-container>\r\n </div>\r\n</ng-container>\r\n", styles: [":host.hx-card{border:unset}:host.hx-card:hover,:host.hx-card.is-hovered{box-shadow:none}.hxa-datepicker-calendar:host{width:19rem;height:21rem;display:flex;flex-direction:column;font-size:1rem;border:unset}.hxa-datepicker-calendar__header{padding:.5rem .5rem 0;flex-direction:column;align-items:stretch}.hxa-datepicker-calendar__icon.hx-button.is-transparent:hover{color:#000}.hxa-datepicker-calendar__icon .hx-icon{color:#0d4d78}.hxa-datepicker-calendar__month{display:flex;align-items:center;justify-content:space-between;text-align:center;margin-bottom:.5rem}.hxa-datepicker-calendar__month-title{font-size:1.25em;font-weight:400;flex:3;display:flex;justify-content:center;align-items:center;color:#0d4d78}.hxa-datepicker-calendar__week{display:flex;width:100%;text-align:center}.hxa-datepicker-calendar__weekday{flex:1;color:#0d4d78;font-size:.85em}.hxa-datepicker-calendar__contents{padding:.5rem;background-color:#f6f6f980;display:flex;flex-flow:row wrap;flex:1;justify-content:space-around;align-content:space-around}.hxa-datepicker-calendar__year{margin:.5rem}.hxa-datepicker-calendar__year.hx-button{padding:.5rem}.hxa-datepicker-calendar__months{margin:.7rem}.hxa-datepicker-calendar__months.hx-button{padding:.5rem;font-weight:500}.hxa-datepicker-calendar__day,.hxa-datepicker-calendar__year,.hxa-datepicker-calendar__months{flex:1 1 14%;height:16.666%;display:flex;justify-content:center;align-items:center}.hxa-datepicker-calendar__day.hx-button.is-transparent,.hxa-datepicker-calendar__year.hx-button.is-transparent,.hxa-datepicker-calendar__months.hx-button.is-transparent{font-weight:400;padding:0;color:#41b987}.hxa-datepicker-calendar__day-siblingmonth.hx-button.is-transparent,.hxa-datepicker-calendar__year-siblingmonth.hx-button.is-transparent,.hxa-datepicker-calendar__months-siblingmonth.hx-button.is-transparent{color:#3b3b3b;font-weight:100}.hxa-datepicker-calendar__day-selectedday.hx-button.is-transparent,.hxa-datepicker-calendar__year-selectedday.hx-button.is-transparent,.hxa-datepicker-calendar__months-selectedday.hx-button.is-transparent{color:#fff;background:#41b987}.hxa-datepicker-calendar__day-invalidday.hx-button.is-transparent,.hxa-datepicker-calendar__year-invalidday.hx-button.is-transparent,.hxa-datepicker-calendar__months-invalidday.hx-button.is-transparent{color:#e0e0e1;pointer-events:none}.hxa-datepicker-calendar__day-currentday.hx-button.is-transparent,.hxa-datepicker-calendar__year-currentday.hx-button.is-transparent,.hxa-datepicker-calendar__months-currentday.hx-button.is-transparent{border:2px solid #41b987}\n"] }] }], propDecorators: { classes: [{ type: HostBinding, args: ['class'] }], selectedDate: [{ type: Input }], validators: [{ type: Input }], update: [{ type: Output }] } }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"datepicker.component.js","sourceRoot":"","sources":["../../../../../projects/hx-ui/src/lib/datepicker/datepicker.component.ts","../../../../../projects/hx-ui/src/lib/datepicker/datepicker.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\r\n  Component,\r\n  EventEmitter,\r\n  HostBinding,\r\n  Input,\r\n  OnChanges,\r\n  OnInit,\r\n  Output,\r\n  SimpleChanges\r\n} from '@angular/core';\r\nimport { BehaviorSubject } from 'rxjs';\r\nimport { Visibility } from '../enums';\r\nimport { DatepickerViewModeEnum } from './datepicker.model';\r\n\r\n@Component({\r\n  selector: 'hxa-datepicker',\r\n  templateUrl: './datepicker.component.html',\r\n  styleUrls: ['./datepicker.component.scss']\r\n})\r\nexport class DatepickerComponent implements OnInit, OnChanges {\r\n  @HostBinding('class')\r\n  get classes() {\r\n    return 'hxui-reset hx-card hxa-datepicker-calendar';\r\n  }\r\n\r\n  @Input()\r\n  selectedDate: Date;\r\n\r\n  @Input()\r\n  validators: Array<(date: Date) => boolean>;\r\n\r\n  @Output()\r\n  update = new EventEmitter<Date>();\r\n\r\n  viewMode$ = new BehaviorSubject<DatepickerViewModeEnum>(\r\n    DatepickerViewModeEnum.Days\r\n  );\r\n  DatepickerViewModeEnum = DatepickerViewModeEnum;\r\n  visibilityEnum = Visibility;\r\n  visibility: Visibility = Visibility.Hidden;\r\n  viewDate: Date;\r\n  days: Array<Date> = new Array<Date>();\r\n  week: Array<string> = [\r\n    'Monday',\r\n    'Tuesday',\r\n    'Wednesday',\r\n    'Thursday',\r\n    'Friday',\r\n    'Saturday',\r\n    'Sunday'\r\n  ];\r\n  months: Array<string> = [\r\n    'Jan',\r\n    'Feb',\r\n    'Mar',\r\n    'Apr',\r\n    'May',\r\n    'Jun',\r\n    'Jul',\r\n    'Aug',\r\n    'Sep',\r\n    'Oct',\r\n    'Nov',\r\n    'Dec'\r\n  ];\r\n  years: Array<number> = new Array<number>();\r\n  private presentDate: Date;\r\n  private cellCount = 41;\r\n  private yearCellCount = 20;\r\n\r\n  ngOnInit(): void {\r\n    const date: Date = this.selectedDate ? this.selectedDate : new Date();\r\n    this.presentDate = new Date(\r\n      date.getFullYear(),\r\n      date.getMonth(),\r\n      date.getDate()\r\n    );\r\n    this.viewDate =\r\n      this.viewDate || new Date(date.getFullYear(), date.getMonth());\r\n    this.renderCalendar();\r\n  }\r\n\r\n  ngOnChanges(changes: SimpleChanges): void {\r\n    // update view date\r\n    if (changes.selectedDate && changes.selectedDate.currentValue) {\r\n      this.viewDate = new Date(\r\n        this.selectedDate.getFullYear(),\r\n        this.selectedDate.getMonth()\r\n      );\r\n    }\r\n  }\r\n\r\n  /** update and emit selected date  */\r\n  public setSelectedDate(date: Date): void {\r\n    if (!this.isInvalidDay(date)) {\r\n      this.selectedDate = date;\r\n      this.update.emit(date);\r\n    }\r\n  }\r\n\r\n  // Populates the days array with the current month, and completes the view with partial dates from sibling months\r\n  public renderCalendar(): void {\r\n    for (let i = 0; i <= this.cellCount; i++) {\r\n      // date will be set to the first day of the month set in this.viewDate\r\n      const date: Date = new Date(\r\n        this.viewDate.getFullYear(),\r\n        this.viewDate.getMonth()\r\n      );\r\n      // Shifts the week to start from Monday, rather than Sunday, this causes the index to start at 1\r\n      const dayOffset = date.getDay() === 0 ? 7 : date.getDay();\r\n      this.days[i] = new Date(date.setDate(2 - dayOffset + i));\r\n    }\r\n  }\r\n\r\n  public next() {\r\n    if (this.viewMode$.value === DatepickerViewModeEnum.Days) {\r\n      this.nextMonth();\r\n    } else if (this.viewMode$.value === DatepickerViewModeEnum.Years) {\r\n      this.nextYear();\r\n    }\r\n  }\r\n\r\n  public previous() {\r\n    if (this.viewMode$.value === DatepickerViewModeEnum.Days) {\r\n      this.previousMonth();\r\n    } else if (this.viewMode$.value === DatepickerViewModeEnum.Years) {\r\n      this.previousYear();\r\n    }\r\n  }\r\n\r\n  public previousMonth(): void {\r\n    this.viewDate = new Date(\r\n      this.viewDate.getFullYear(),\r\n      this.viewDate.getMonth() - 1\r\n    );\r\n    this.renderCalendar();\r\n  }\r\n\r\n  public nextMonth(): void {\r\n    this.viewDate = new Date(\r\n      this.viewDate.getFullYear(),\r\n      this.viewDate.getMonth() + 1\r\n    );\r\n    this.renderCalendar();\r\n  }\r\n\r\n  public isCurrentMonth(inputDate: Date): boolean {\r\n    return inputDate.getMonth() === this.viewDate.getMonth();\r\n  }\r\n\r\n  public isCurrentDay(inputDate: Date): boolean {\r\n    return inputDate.getTime() === this.presentDate.getTime();\r\n  }\r\n\r\n  public isSelectedDay(inputDate: Date): boolean {\r\n    if (this.selectedDate) {\r\n      return inputDate.getTime() === this.selectedDate.getTime();\r\n    }\r\n\r\n    return false;\r\n  }\r\n\r\n  public isInvalidDay(inputDate: Date): boolean {\r\n    return this.validators\r\n      .map(fn => fn(inputDate))\r\n      .reduce((prev, next) => prev || next, false);\r\n  }\r\n\r\n  public isCurrentYear(year: number): boolean {\r\n    return year === this.presentDate.getFullYear();\r\n  }\r\n\r\n  public isSelectedYear(year: number): boolean {\r\n    return year === this.viewDate.getFullYear();\r\n  }\r\n\r\n  public isInvalidYear(year: number): boolean {\r\n    const newDate = new Date(\r\n      new Date(this.viewDate.getTime()).setFullYear(year)\r\n    );\r\n    return this.validators\r\n      .map(fn => fn(newDate))\r\n      .reduce((prev, next) => prev || next, false);\r\n  }\r\n\r\n  public isCurrentMonthByIndex(month: number): boolean {\r\n    return month === this.presentDate.getMonth();\r\n  }\r\n\r\n  public isSelectedMonthByIndex(month: number): boolean {\r\n    return month === this.viewDate.getMonth();\r\n  }\r\n\r\n  public isInvalidMonthByIndex(month: number): boolean {\r\n    const newDate = new Date(new Date(this.viewDate.getTime()).setMonth(month));\r\n    return this.validators\r\n      .map(fn => fn(newDate))\r\n      .reduce((prev, next) => prev || next, false);\r\n  }\r\n\r\n  public previousYear(): void {\r\n    this.getYearCollection(this.years[0] - this.yearCellCount);\r\n  }\r\n\r\n  public nextYear(): void {\r\n    this.getYearCollection(this.years[0] + this.yearCellCount);\r\n  }\r\n\r\n  public setYear(year) {\r\n    if (!this.isInvalidYear(year)) {\r\n      this.viewDate.setFullYear(year);\r\n      this.renderCalendar();\r\n      this.toggleYear();\r\n    }\r\n  }\r\n\r\n  public setMonth(month) {\r\n    if (!this.isInvalidMonthByIndex(month)) {\r\n      this.viewDate.setMonth(month);\r\n      this.renderCalendar();\r\n      this.viewMode$.next(DatepickerViewModeEnum.Days);\r\n    }\r\n  }\r\n\r\n  public toggleYear() {\r\n    this.viewMode$.next(\r\n      this.viewMode$.value === DatepickerViewModeEnum.Years\r\n        ? DatepickerViewModeEnum.Months\r\n        : DatepickerViewModeEnum.Years\r\n    );\r\n    if (this.viewMode$.value === DatepickerViewModeEnum.Years) {\r\n      this.getYearCollection();\r\n    }\r\n  }\r\n\r\n  private getYearCollection(startFrom = null) {\r\n    const yearsBeforeActive = 7;\r\n    const activeYear = startFrom\r\n      ? startFrom\r\n      : this.viewDate.getFullYear() - yearsBeforeActive;\r\n    this.years = [];\r\n    for (let i = 0; i < this.yearCellCount; i++) {\r\n      this.years.push(activeYear + i);\r\n    }\r\n  }\r\n}\r\n","<ng-container *ngIf=\"{ viewMode$ : viewMode$ | async } as observables\">\r\n  <div class=\"hx-card-header hxa-datepicker-calendar__header\">\r\n    <div class=\"hxa-datepicker-calendar__month\">\r\n      <button\r\n        class=\"hxa-datepicker-calendar__icon hx-button is-transparent is-large\"\r\n        *ngIf=\"observables.viewMode$ !== DatepickerViewModeEnum.Months\"\r\n        title=\"Previous Month\"\r\n        (click)=\"previous()\"\r\n      >\r\n        <span class=\"hx-icon-control\">\r\n          <i class=\"hx-icon icon-angle-left is-medium\"></i>\r\n        </span>\r\n      </button>\r\n      <div class=\"hxa-datepicker-calendar__month-title\">\r\n        <button\r\n          class=\"hx-button is-flat is-info is-large\"\r\n          (click)=\"toggleYear()\"\r\n        >\r\n          {{ (observables.viewMode$ === DatepickerViewModeEnum.Years) ? years[0]\r\n          + ' - ' + years[years.length-1] : viewDate?.toLocaleString(\"en-au\", {\r\n          month: \"long\", year: \"numeric\"}) }}\r\n        </button>\r\n      </div>\r\n      <button\r\n        class=\"hxa-datepicker-calendar__icon hx-button is-transparent is-large\"\r\n        *ngIf=\"observables.viewMode$ !== DatepickerViewModeEnum.Months\"\r\n        title=\"Next Month\"\r\n        (click)=\"next()\"\r\n      >\r\n        <span class=\"hx-icon-control\">\r\n          <i class=\"hx-icon icon-angle-right is-medium\"></i>\r\n        </span>\r\n      </button>\r\n    </div>\r\n    <div\r\n      class=\"hxa-datepicker-calendar__week\"\r\n      *ngIf=\"observables.viewMode$ === DatepickerViewModeEnum.Days\"\r\n    >\r\n      <div\r\n        class=\"hxa-datepicker-calendar__weekday\"\r\n        *ngFor=\"let weekday of week\"\r\n      >\r\n        {{weekday | slice:0:3}}\r\n      </div>\r\n    </div>\r\n  </div>\r\n  <div class=\"hxa-datepicker-calendar__contents\">\r\n    <ng-container *ngIf=\"observables.viewMode$ === DatepickerViewModeEnum.Days\">\r\n      <button\r\n        class=\"hxa-datepicker-calendar__day hx-button is-transparent\"\r\n        *ngFor=\"let day of days\"\r\n        [ngClass]=\"{\r\n          'hxa-datepicker-calendar__day-siblingmonth': !isCurrentMonth(day),\r\n          'hxa-datepicker-calendar__day-currentday': isCurrentDay(day),\r\n          'hxa-datepicker-calendar__day-selectedday': isSelectedDay(day),\r\n          'hxa-datepicker-calendar__day-invalidday': isInvalidDay(day)\r\n        }\"\r\n        (click)=\"setSelectedDate(day)\"\r\n      >\r\n        {{day.getDate()}}\r\n      </button>\r\n    </ng-container>\r\n\r\n    <ng-container\r\n      *ngIf=\"observables.viewMode$ === DatepickerViewModeEnum.Months\"\r\n    >\r\n      <button\r\n        class=\"hxa-datepicker-calendar__months hx-button is-transparent\"\r\n        *ngFor=\"let month of months; let indexOfMth=index;\"\r\n        [ngClass]=\"{\r\n          'hxa-datepicker-calendar__day-currentday': isCurrentMonthByIndex(indexOfMth),\r\n          'hxa-datepicker-calendar__day-selectedday': isSelectedMonthByIndex(indexOfMth),\r\n          'hxa-datepicker-calendar__day-invalidday': isInvalidMonthByIndex(indexOfMth)\r\n        }\"\r\n        (click)=\"setMonth(indexOfMth)\"\r\n      >\r\n        {{ month }}\r\n      </button>\r\n    </ng-container>\r\n\r\n    <ng-container\r\n      *ngIf=\"observables.viewMode$ === DatepickerViewModeEnum.Years\"\r\n    >\r\n      <button\r\n        *ngFor=\"let year of years\"\r\n        class=\"hxa-datepicker-calendar__year hx-button is-transparent\"\r\n        [ngClass]=\"{\r\n          'hxa-datepicker-calendar__day-currentday': isCurrentYear(year),\r\n          'hxa-datepicker-calendar__day-selectedday': isSelectedYear(year),\r\n          'hxa-datepicker-calendar__day-invalidday': isInvalidYear(year)\r\n        }\"\r\n        (click)=\"setYear(year)\"\r\n      >\r\n        {{ year }}\r\n      </button>\r\n    </ng-container>\r\n  </div>\r\n</ng-container>\r\n"]}