UNPKG

@angular/material

Version:
1,302 lines (1,287 loc) 88.3 kB
/** * @license * Copyright Google LLC All Rights Reserved. * * Use of this source code is governed by an MIT-style license that can be * found in the LICENSE file at https://angular.io/license */ import { A11yModule } from '@angular/cdk/a11y'; import { Overlay, OverlayConfig, OverlayModule } from '@angular/cdk/overlay'; import { CommonModule, DOCUMENT } from '@angular/common'; import { ChangeDetectionStrategy, ChangeDetectorRef, Component, ContentChild, Directive, ElementRef, EventEmitter, Inject, Injectable, InjectionToken, Input, NgModule, NgZone, Optional, Output, ViewChild, ViewContainerRef, ViewEncapsulation, forwardRef } from '@angular/core'; import { MatButtonModule } from '@angular/material/button'; import { MatDialog, MatDialogModule } from '@angular/material/dialog'; import { DOWN_ARROW, END, ENTER, ESCAPE, HOME, LEFT_ARROW, PAGE_DOWN, PAGE_UP, RIGHT_ARROW, UP_ARROW } from '@angular/cdk/keycodes'; import { DateAdapter, MAT_DATE_FORMATS } from '@angular/material/core'; import { take } from 'rxjs/operators/take'; import { Subject } from 'rxjs/Subject'; import { Directionality } from '@angular/cdk/bidi'; import { coerceBooleanProperty } from '@angular/cdk/coercion'; import { ComponentPortal } from '@angular/cdk/portal'; import { filter } from 'rxjs/operators/filter'; import { Subscription } from 'rxjs/Subscription'; import { merge } from 'rxjs/observable/merge'; import { NG_VALIDATORS, NG_VALUE_ACCESSOR, Validators } from '@angular/forms'; import { MatFormField } from '@angular/material/form-field'; import { MAT_INPUT_VALUE_ACCESSOR } from '@angular/material/input'; import { of } from 'rxjs/observable/of'; /** * @fileoverview added by tsickle * @suppress {checkTypes} checked by tsc */ /** * \@docs-private * @param {?} provider * @return {?} */ function createMissingDateImplError(provider) { return Error(`MatDatepicker: No provider found for ${provider}. You must import one of the following ` + `modules at your application root: MatNativeDateModule, MatMomentDateModule, or provide a ` + `custom implementation.`); } /** * @fileoverview added by tsickle * @suppress {checkTypes} checked by tsc */ /** * Datepicker data that requires internationalization. */ class MatDatepickerIntl { constructor() { /** * Stream that emits whenever the labels here are changed. Use this to notify * components if the labels have changed after initialization. */ this.changes = new Subject(); /** * A label for the calendar popup (used by screen readers). */ this.calendarLabel = 'Calendar'; /** * A label for the button used to open the calendar popup (used by screen readers). */ this.openCalendarLabel = 'Open calendar'; /** * A label for the previous month button (used by screen readers). */ this.prevMonthLabel = 'Previous month'; /** * A label for the next month button (used by screen readers). */ this.nextMonthLabel = 'Next month'; /** * A label for the previous year button (used by screen readers). */ this.prevYearLabel = 'Previous year'; /** * A label for the next year button (used by screen readers). */ this.nextYearLabel = 'Next year'; /** * A label for the previous multi-year button (used by screen readers). */ this.prevMultiYearLabel = 'Previous 20 years'; /** * A label for the next multi-year button (used by screen readers). */ this.nextMultiYearLabel = 'Next 20 years'; /** * A label for the 'switch to month view' button (used by screen readers). */ this.switchToMonthViewLabel = 'Choose date'; /** * A label for the 'switch to year view' button (used by screen readers). */ this.switchToMultiYearViewLabel = 'Choose month and year'; } } MatDatepickerIntl.decorators = [ { type: Injectable }, ]; /** @nocollapse */ MatDatepickerIntl.ctorParameters = () => []; /** * @fileoverview added by tsickle * @suppress {checkTypes} checked by tsc */ /** * An internal class that represents the data corresponding to a single calendar cell. * \@docs-private */ class MatCalendarCell { /** * @param {?} value * @param {?} displayValue * @param {?} ariaLabel * @param {?} enabled */ constructor(value, displayValue, ariaLabel, enabled) { this.value = value; this.displayValue = displayValue; this.ariaLabel = ariaLabel; this.enabled = enabled; } } /** * An internal component used to display calendar data in a table. * \@docs-private */ class MatCalendarBody { constructor() { /** * The number of columns in the table. */ this.numCols = 7; /** * Whether to allow selection of disabled cells. */ this.allowDisabledSelection = false; /** * The cell number of the active cell in the table. */ this.activeCell = 0; /** * The aspect ratio (width / height) to use for the cells in the table. This aspect ratio will be * maintained even as the table resizes. */ this.cellAspectRatio = 1; /** * Emits when a new value is selected. */ this.selectedValueChange = new EventEmitter(); } /** * @param {?} cell * @return {?} */ _cellClicked(cell) { if (!this.allowDisabledSelection && !cell.enabled) { return; } this.selectedValueChange.emit(cell.value); } /** * The number of blank cells to put at the beginning for the first row. * @return {?} */ get _firstRowOffset() { return this.rows && this.rows.length && this.rows[0].length ? this.numCols - this.rows[0].length : 0; } /** * @param {?} rowIndex * @param {?} colIndex * @return {?} */ _isActiveCell(rowIndex, colIndex) { let /** @type {?} */ cellNumber = rowIndex * this.numCols + colIndex; // Account for the fact that the first row may not have as many cells. if (rowIndex) { cellNumber -= this._firstRowOffset; } return cellNumber == this.activeCell; } } MatCalendarBody.decorators = [ { type: Component, args: [{selector: '[mat-calendar-body]', template: "<tr *ngIf=\"_firstRowOffset < labelMinRequiredCells\" aria-hidden=\"true\"><td class=\"mat-calendar-body-label\" [attr.colspan]=\"numCols\" [style.paddingTop.%]=\"50 * cellAspectRatio / numCols\" [style.paddingBottom.%]=\"50 * cellAspectRatio / numCols\">{{label}}</td></tr><tr *ngFor=\"let row of rows; let rowIndex = index\" role=\"row\"><td *ngIf=\"rowIndex === 0 && _firstRowOffset\" aria-hidden=\"true\" class=\"mat-calendar-body-label\" [attr.colspan]=\"_firstRowOffset\" [style.paddingTop.%]=\"50 * cellAspectRatio / numCols\" [style.paddingBottom.%]=\"50 * cellAspectRatio / numCols\">{{_firstRowOffset >= labelMinRequiredCells ? label : ''}}</td><td *ngFor=\"let item of row; let colIndex = index\" role=\"gridcell\" class=\"mat-calendar-body-cell\" [tabindex]=\"_isActiveCell(rowIndex, colIndex) ? 0 : -1\" [class.mat-calendar-body-disabled]=\"!item.enabled\" [class.mat-calendar-body-active]=\"_isActiveCell(rowIndex, colIndex)\" [attr.aria-label]=\"item.ariaLabel\" [attr.aria-disabled]=\"!item.enabled || null\" (click)=\"_cellClicked(item)\" [style.width.%]=\"100 / numCols\" [style.paddingTop.%]=\"50 * cellAspectRatio / numCols\" [style.paddingBottom.%]=\"50 * cellAspectRatio / numCols\"><div class=\"mat-calendar-body-cell-content\" [class.mat-calendar-body-selected]=\"selectedValue === item.value\" [class.mat-calendar-body-today]=\"todayValue === item.value\">{{item.displayValue}}</div></td></tr>", styles: [".mat-calendar-body{min-width:224px}.mat-calendar-body-label{height:0;line-height:0;text-align:left;padding-left:4.71429%;padding-right:4.71429%}.mat-calendar-body-cell{position:relative;height:0;line-height:0;text-align:center;outline:0;cursor:pointer}.mat-calendar-body-disabled{cursor:default}.mat-calendar-body-cell-content{position:absolute;top:5%;left:5%;display:flex;align-items:center;justify-content:center;box-sizing:border-box;width:90%;height:90%;line-height:1;border-width:1px;border-style:solid;border-radius:999px}[dir=rtl] .mat-calendar-body-label{text-align:right}"], host: { 'class': 'mat-calendar-body', 'role': 'grid', 'attr.aria-readonly': 'true' }, exportAs: 'matCalendarBody', encapsulation: ViewEncapsulation.None, preserveWhitespaces: false, changeDetection: ChangeDetectionStrategy.OnPush, },] }, ]; /** @nocollapse */ MatCalendarBody.ctorParameters = () => []; MatCalendarBody.propDecorators = { "label": [{ type: Input },], "rows": [{ type: Input },], "todayValue": [{ type: Input },], "selectedValue": [{ type: Input },], "labelMinRequiredCells": [{ type: Input },], "numCols": [{ type: Input },], "allowDisabledSelection": [{ type: Input },], "activeCell": [{ type: Input },], "cellAspectRatio": [{ type: Input },], "selectedValueChange": [{ type: Output },], }; /** * @fileoverview added by tsickle * @suppress {checkTypes} checked by tsc */ const DAYS_PER_WEEK = 7; /** * An internal component used to display a single month in the datepicker. * \@docs-private */ class MatMonthView { /** * @param {?} _dateAdapter * @param {?} _dateFormats * @param {?} _changeDetectorRef */ constructor(_dateAdapter, _dateFormats, _changeDetectorRef) { this._dateAdapter = _dateAdapter; this._dateFormats = _dateFormats; this._changeDetectorRef = _changeDetectorRef; /** * Emits when a new date is selected. */ this.selectedChange = new EventEmitter(); /** * Emits when any date is selected. */ this._userSelection = new EventEmitter(); if (!this._dateAdapter) { throw createMissingDateImplError('DateAdapter'); } if (!this._dateFormats) { throw createMissingDateImplError('MAT_DATE_FORMATS'); } const /** @type {?} */ firstDayOfWeek = this._dateAdapter.getFirstDayOfWeek(); const /** @type {?} */ narrowWeekdays = this._dateAdapter.getDayOfWeekNames('narrow'); const /** @type {?} */ longWeekdays = this._dateAdapter.getDayOfWeekNames('long'); // Rotate the labels for days of the week based on the configured first day of the week. let /** @type {?} */ weekdays = longWeekdays.map((long, i) => { return { long, narrow: narrowWeekdays[i] }; }); this._weekdays = weekdays.slice(firstDayOfWeek).concat(weekdays.slice(0, firstDayOfWeek)); this._activeDate = this._dateAdapter.today(); } /** * The date to display in this month view (everything other than the month and year is ignored). * @return {?} */ get activeDate() { return this._activeDate; } /** * @param {?} value * @return {?} */ set activeDate(value) { let /** @type {?} */ oldActiveDate = this._activeDate; this._activeDate = this._getValidDateOrNull(this._dateAdapter.deserialize(value)) || this._dateAdapter.today(); if (!this._hasSameMonthAndYear(oldActiveDate, this._activeDate)) { this._init(); } } /** * The currently selected date. * @return {?} */ get selected() { return this._selected; } /** * @param {?} value * @return {?} */ set selected(value) { this._selected = this._getValidDateOrNull(this._dateAdapter.deserialize(value)); this._selectedDate = this._getDateInCurrentMonth(this._selected); } /** * The minimum selectable date. * @return {?} */ get minDate() { return this._minDate; } /** * @param {?} value * @return {?} */ set minDate(value) { this._minDate = this._getValidDateOrNull(this._dateAdapter.deserialize(value)); } /** * The maximum selectable date. * @return {?} */ get maxDate() { return this._maxDate; } /** * @param {?} value * @return {?} */ set maxDate(value) { this._maxDate = this._getValidDateOrNull(this._dateAdapter.deserialize(value)); } /** * @return {?} */ ngAfterContentInit() { this._init(); } /** * Handles when a new date is selected. * @param {?} date * @return {?} */ _dateSelected(date) { if (this._selectedDate != date) { const /** @type {?} */ selectedYear = this._dateAdapter.getYear(this.activeDate); const /** @type {?} */ selectedMonth = this._dateAdapter.getMonth(this.activeDate); const /** @type {?} */ selectedDate = this._dateAdapter.createDate(selectedYear, selectedMonth, date); this.selectedChange.emit(selectedDate); } this._userSelection.emit(); } /** * Initializes this month view. * @return {?} */ _init() { this._selectedDate = this._getDateInCurrentMonth(this.selected); this._todayDate = this._getDateInCurrentMonth(this._dateAdapter.today()); this._monthLabel = this._dateAdapter.getMonthNames('short')[this._dateAdapter.getMonth(this.activeDate)] .toLocaleUpperCase(); let /** @type {?} */ firstOfMonth = this._dateAdapter.createDate(this._dateAdapter.getYear(this.activeDate), this._dateAdapter.getMonth(this.activeDate), 1); this._firstWeekOffset = (DAYS_PER_WEEK + this._dateAdapter.getDayOfWeek(firstOfMonth) - this._dateAdapter.getFirstDayOfWeek()) % DAYS_PER_WEEK; this._createWeekCells(); this._changeDetectorRef.markForCheck(); } /** * Creates MatCalendarCells for the dates in this month. * @return {?} */ _createWeekCells() { const /** @type {?} */ daysInMonth = this._dateAdapter.getNumDaysInMonth(this.activeDate); const /** @type {?} */ dateNames = this._dateAdapter.getDateNames(); this._weeks = [[]]; for (let /** @type {?} */ i = 0, /** @type {?} */ cell = this._firstWeekOffset; i < daysInMonth; i++, cell++) { if (cell == DAYS_PER_WEEK) { this._weeks.push([]); cell = 0; } const /** @type {?} */ date = this._dateAdapter.createDate(this._dateAdapter.getYear(this.activeDate), this._dateAdapter.getMonth(this.activeDate), i + 1); const /** @type {?} */ enabled = this._shouldEnableDate(date); const /** @type {?} */ ariaLabel = this._dateAdapter.format(date, this._dateFormats.display.dateA11yLabel); this._weeks[this._weeks.length - 1] .push(new MatCalendarCell(i + 1, dateNames[i], ariaLabel, enabled)); } } /** * Date filter for the month * @param {?} date * @return {?} */ _shouldEnableDate(date) { return !!date && (!this.dateFilter || this.dateFilter(date)) && (!this.minDate || this._dateAdapter.compareDate(date, this.minDate) >= 0) && (!this.maxDate || this._dateAdapter.compareDate(date, this.maxDate) <= 0); } /** * Gets the date in this month that the given Date falls on. * Returns null if the given Date is in another month. * @param {?} date * @return {?} */ _getDateInCurrentMonth(date) { return date && this._hasSameMonthAndYear(date, this.activeDate) ? this._dateAdapter.getDate(date) : null; } /** * Checks whether the 2 dates are non-null and fall within the same month of the same year. * @param {?} d1 * @param {?} d2 * @return {?} */ _hasSameMonthAndYear(d1, d2) { return !!(d1 && d2 && this._dateAdapter.getMonth(d1) == this._dateAdapter.getMonth(d2) && this._dateAdapter.getYear(d1) == this._dateAdapter.getYear(d2)); } /** * @param {?} obj The object to check. * @return {?} The given object if it is both a date instance and valid, otherwise null. */ _getValidDateOrNull(obj) { return (this._dateAdapter.isDateInstance(obj) && this._dateAdapter.isValid(obj)) ? obj : null; } } MatMonthView.decorators = [ { type: Component, args: [{selector: 'mat-month-view', template: "<table class=\"mat-calendar-table\"><thead class=\"mat-calendar-table-header\"><tr><th *ngFor=\"let day of _weekdays\" [attr.aria-label]=\"day.long\">{{day.narrow}}</th></tr><tr><th class=\"mat-calendar-table-header-divider\" colspan=\"7\" aria-hidden=\"true\"></th></tr></thead><tbody mat-calendar-body [label]=\"_monthLabel\" [rows]=\"_weeks\" [todayValue]=\"_todayDate\" [selectedValue]=\"_selectedDate\" [labelMinRequiredCells]=\"3\" [activeCell]=\"_dateAdapter.getDate(activeDate) - 1\" (selectedValueChange)=\"_dateSelected($event)\"></tbody></table>", exportAs: 'matMonthView', encapsulation: ViewEncapsulation.None, preserveWhitespaces: false, changeDetection: ChangeDetectionStrategy.OnPush, },] }, ]; /** @nocollapse */ MatMonthView.ctorParameters = () => [ { type: DateAdapter, decorators: [{ type: Optional },] }, { type: undefined, decorators: [{ type: Optional }, { type: Inject, args: [MAT_DATE_FORMATS,] },] }, { type: ChangeDetectorRef, }, ]; MatMonthView.propDecorators = { "activeDate": [{ type: Input },], "selected": [{ type: Input },], "minDate": [{ type: Input },], "maxDate": [{ type: Input },], "dateFilter": [{ type: Input },], "selectedChange": [{ type: Output },], "_userSelection": [{ type: Output },], }; /** * @fileoverview added by tsickle * @suppress {checkTypes} checked by tsc */ const yearsPerPage = 24; const yearsPerRow = 4; /** * An internal component used to display a year selector in the datepicker. * \@docs-private */ class MatMultiYearView { /** * @param {?} _dateAdapter * @param {?} _changeDetectorRef */ constructor(_dateAdapter, _changeDetectorRef) { this._dateAdapter = _dateAdapter; this._changeDetectorRef = _changeDetectorRef; /** * Emits when a new month is selected. */ this.selectedChange = new EventEmitter(); if (!this._dateAdapter) { throw createMissingDateImplError('DateAdapter'); } this._activeDate = this._dateAdapter.today(); } /** * The date to display in this multi-year view (everything other than the year is ignored). * @return {?} */ get activeDate() { return this._activeDate; } /** * @param {?} value * @return {?} */ set activeDate(value) { let /** @type {?} */ oldActiveDate = this._activeDate; this._activeDate = this._getValidDateOrNull(this._dateAdapter.deserialize(value)) || this._dateAdapter.today(); if (Math.floor(this._dateAdapter.getYear(oldActiveDate) / yearsPerPage) != Math.floor(this._dateAdapter.getYear(this._activeDate) / yearsPerPage)) { this._init(); } } /** * The currently selected date. * @return {?} */ get selected() { return this._selected; } /** * @param {?} value * @return {?} */ set selected(value) { this._selected = this._getValidDateOrNull(this._dateAdapter.deserialize(value)); this._selectedYear = this._selected && this._dateAdapter.getYear(this._selected); } /** * The minimum selectable date. * @return {?} */ get minDate() { return this._minDate; } /** * @param {?} value * @return {?} */ set minDate(value) { this._minDate = this._getValidDateOrNull(this._dateAdapter.deserialize(value)); } /** * The maximum selectable date. * @return {?} */ get maxDate() { return this._maxDate; } /** * @param {?} value * @return {?} */ set maxDate(value) { this._maxDate = this._getValidDateOrNull(this._dateAdapter.deserialize(value)); } /** * @return {?} */ ngAfterContentInit() { this._init(); } /** * Initializes this multi-year view. * @return {?} */ _init() { this._todayYear = this._dateAdapter.getYear(this._dateAdapter.today()); let /** @type {?} */ activeYear = this._dateAdapter.getYear(this._activeDate); let /** @type {?} */ activeOffset = activeYear % yearsPerPage; this._years = []; for (let /** @type {?} */ i = 0, /** @type {?} */ row = []; i < yearsPerPage; i++) { row.push(activeYear - activeOffset + i); if (row.length == yearsPerRow) { this._years.push(row.map(year => this._createCellForYear(year))); row = []; } } this._changeDetectorRef.markForCheck(); } /** * Handles when a new year is selected. * @param {?} year * @return {?} */ _yearSelected(year) { let /** @type {?} */ month = this._dateAdapter.getMonth(this.activeDate); let /** @type {?} */ daysInMonth = this._dateAdapter.getNumDaysInMonth(this._dateAdapter.createDate(year, month, 1)); this.selectedChange.emit(this._dateAdapter.createDate(year, month, Math.min(this._dateAdapter.getDate(this.activeDate), daysInMonth))); } /** * @return {?} */ _getActiveCell() { return this._dateAdapter.getYear(this.activeDate) % yearsPerPage; } /** * Creates an MatCalendarCell for the given year. * @param {?} year * @return {?} */ _createCellForYear(year) { let /** @type {?} */ yearName = this._dateAdapter.getYearName(this._dateAdapter.createDate(year, 0, 1)); return new MatCalendarCell(year, yearName, yearName, this._shouldEnableYear(year)); } /** * Whether the given year is enabled. * @param {?} year * @return {?} */ _shouldEnableYear(year) { // disable if the year is greater than maxDate lower than minDate if (year === undefined || year === null || (this.maxDate && year > this._dateAdapter.getYear(this.maxDate)) || (this.minDate && year < this._dateAdapter.getYear(this.minDate))) { return false; } // enable if it reaches here and there's no filter defined if (!this.dateFilter) { return true; } const /** @type {?} */ firstOfYear = this._dateAdapter.createDate(year, 0, 1); // If any date in the year is enabled count the year as enabled. for (let /** @type {?} */ date = firstOfYear; this._dateAdapter.getYear(date) == year; date = this._dateAdapter.addCalendarDays(date, 1)) { if (this.dateFilter(date)) { return true; } } return false; } /** * @param {?} obj The object to check. * @return {?} The given object if it is both a date instance and valid, otherwise null. */ _getValidDateOrNull(obj) { return (this._dateAdapter.isDateInstance(obj) && this._dateAdapter.isValid(obj)) ? obj : null; } } MatMultiYearView.decorators = [ { type: Component, args: [{selector: 'mat-multi-year-view', template: "<table class=\"mat-calendar-table\"><thead class=\"mat-calendar-table-header\"><tr><th class=\"mat-calendar-table-header-divider\" colspan=\"4\"></th></tr></thead><tbody mat-calendar-body allowDisabledSelection=\"true\" [rows]=\"_years\" [todayValue]=\"_todayYear\" [selectedValue]=\"_selectedYear\" [numCols]=\"4\" [cellAspectRatio]=\"4 / 7\" [activeCell]=\"_getActiveCell()\" (selectedValueChange)=\"_yearSelected($event)\"></tbody></table>", exportAs: 'matMultiYearView', encapsulation: ViewEncapsulation.None, preserveWhitespaces: false, changeDetection: ChangeDetectionStrategy.OnPush, },] }, ]; /** @nocollapse */ MatMultiYearView.ctorParameters = () => [ { type: DateAdapter, decorators: [{ type: Optional },] }, { type: ChangeDetectorRef, }, ]; MatMultiYearView.propDecorators = { "activeDate": [{ type: Input },], "selected": [{ type: Input },], "minDate": [{ type: Input },], "maxDate": [{ type: Input },], "dateFilter": [{ type: Input },], "selectedChange": [{ type: Output },], }; /** * @fileoverview added by tsickle * @suppress {checkTypes} checked by tsc */ /** * An internal component used to display a single year in the datepicker. * \@docs-private */ class MatYearView { /** * @param {?} _dateAdapter * @param {?} _dateFormats * @param {?} _changeDetectorRef */ constructor(_dateAdapter, _dateFormats, _changeDetectorRef) { this._dateAdapter = _dateAdapter; this._dateFormats = _dateFormats; this._changeDetectorRef = _changeDetectorRef; /** * Emits when a new month is selected. */ this.selectedChange = new EventEmitter(); if (!this._dateAdapter) { throw createMissingDateImplError('DateAdapter'); } if (!this._dateFormats) { throw createMissingDateImplError('MAT_DATE_FORMATS'); } this._activeDate = this._dateAdapter.today(); } /** * The date to display in this year view (everything other than the year is ignored). * @return {?} */ get activeDate() { return this._activeDate; } /** * @param {?} value * @return {?} */ set activeDate(value) { let /** @type {?} */ oldActiveDate = this._activeDate; this._activeDate = this._getValidDateOrNull(this._dateAdapter.deserialize(value)) || this._dateAdapter.today(); if (this._dateAdapter.getYear(oldActiveDate) != this._dateAdapter.getYear(this._activeDate)) { this._init(); } } /** * The currently selected date. * @return {?} */ get selected() { return this._selected; } /** * @param {?} value * @return {?} */ set selected(value) { this._selected = this._getValidDateOrNull(this._dateAdapter.deserialize(value)); this._selectedMonth = this._getMonthInCurrentYear(this._selected); } /** * The minimum selectable date. * @return {?} */ get minDate() { return this._minDate; } /** * @param {?} value * @return {?} */ set minDate(value) { this._minDate = this._getValidDateOrNull(this._dateAdapter.deserialize(value)); } /** * The maximum selectable date. * @return {?} */ get maxDate() { return this._maxDate; } /** * @param {?} value * @return {?} */ set maxDate(value) { this._maxDate = this._getValidDateOrNull(this._dateAdapter.deserialize(value)); } /** * @return {?} */ ngAfterContentInit() { this._init(); } /** * Handles when a new month is selected. * @param {?} month * @return {?} */ _monthSelected(month) { let /** @type {?} */ daysInMonth = this._dateAdapter.getNumDaysInMonth(this._dateAdapter.createDate(this._dateAdapter.getYear(this.activeDate), month, 1)); this.selectedChange.emit(this._dateAdapter.createDate(this._dateAdapter.getYear(this.activeDate), month, Math.min(this._dateAdapter.getDate(this.activeDate), daysInMonth))); } /** * Initializes this year view. * @return {?} */ _init() { this._selectedMonth = this._getMonthInCurrentYear(this.selected); this._todayMonth = this._getMonthInCurrentYear(this._dateAdapter.today()); this._yearLabel = this._dateAdapter.getYearName(this.activeDate); let /** @type {?} */ monthNames = this._dateAdapter.getMonthNames('short'); // First row of months only contains 5 elements so we can fit the year label on the same row. this._months = [[0, 1, 2, 3], [4, 5, 6, 7], [8, 9, 10, 11]].map(row => row.map(month => this._createCellForMonth(month, monthNames[month]))); this._changeDetectorRef.markForCheck(); } /** * Gets the month in this year that the given Date falls on. * Returns null if the given Date is in another year. * @param {?} date * @return {?} */ _getMonthInCurrentYear(date) { return date && this._dateAdapter.getYear(date) == this._dateAdapter.getYear(this.activeDate) ? this._dateAdapter.getMonth(date) : null; } /** * Creates an MatCalendarCell for the given month. * @param {?} month * @param {?} monthName * @return {?} */ _createCellForMonth(month, monthName) { let /** @type {?} */ ariaLabel = this._dateAdapter.format(this._dateAdapter.createDate(this._dateAdapter.getYear(this.activeDate), month, 1), this._dateFormats.display.monthYearA11yLabel); return new MatCalendarCell(month, monthName.toLocaleUpperCase(), ariaLabel, this._shouldEnableMonth(month)); } /** * Whether the given month is enabled. * @param {?} month * @return {?} */ _shouldEnableMonth(month) { const /** @type {?} */ activeYear = this._dateAdapter.getYear(this.activeDate); if (month === undefined || month === null || this._isYearAndMonthAfterMaxDate(activeYear, month) || this._isYearAndMonthBeforeMinDate(activeYear, month)) { return false; } if (!this.dateFilter) { return true; } const /** @type {?} */ firstOfMonth = this._dateAdapter.createDate(activeYear, month, 1); // If any date in the month is enabled count the month as enabled. for (let /** @type {?} */ date = firstOfMonth; this._dateAdapter.getMonth(date) == month; date = this._dateAdapter.addCalendarDays(date, 1)) { if (this.dateFilter(date)) { return true; } } return false; } /** * Tests whether the combination month/year is after this.maxDate, considering * just the month and year of this.maxDate * @param {?} year * @param {?} month * @return {?} */ _isYearAndMonthAfterMaxDate(year, month) { if (this.maxDate) { const /** @type {?} */ maxYear = this._dateAdapter.getYear(this.maxDate); const /** @type {?} */ maxMonth = this._dateAdapter.getMonth(this.maxDate); return year > maxYear || (year === maxYear && month > maxMonth); } return false; } /** * Tests whether the combination month/year is before this.minDate, considering * just the month and year of this.minDate * @param {?} year * @param {?} month * @return {?} */ _isYearAndMonthBeforeMinDate(year, month) { if (this.minDate) { const /** @type {?} */ minYear = this._dateAdapter.getYear(this.minDate); const /** @type {?} */ minMonth = this._dateAdapter.getMonth(this.minDate); return year < minYear || (year === minYear && month < minMonth); } return false; } /** * @param {?} obj The object to check. * @return {?} The given object if it is both a date instance and valid, otherwise null. */ _getValidDateOrNull(obj) { return (this._dateAdapter.isDateInstance(obj) && this._dateAdapter.isValid(obj)) ? obj : null; } } MatYearView.decorators = [ { type: Component, args: [{selector: 'mat-year-view', template: "<table class=\"mat-calendar-table\"><thead class=\"mat-calendar-table-header\"><tr><th class=\"mat-calendar-table-header-divider\" colspan=\"4\"></th></tr></thead><tbody mat-calendar-body allowDisabledSelection=\"true\" [label]=\"_yearLabel\" [rows]=\"_months\" [todayValue]=\"_todayMonth\" [selectedValue]=\"_selectedMonth\" [labelMinRequiredCells]=\"2\" [numCols]=\"4\" [cellAspectRatio]=\"4 / 7\" [activeCell]=\"_dateAdapter.getMonth(activeDate)\" (selectedValueChange)=\"_monthSelected($event)\"></tbody></table>", exportAs: 'matYearView', encapsulation: ViewEncapsulation.None, preserveWhitespaces: false, changeDetection: ChangeDetectionStrategy.OnPush, },] }, ]; /** @nocollapse */ MatYearView.ctorParameters = () => [ { type: DateAdapter, decorators: [{ type: Optional },] }, { type: undefined, decorators: [{ type: Optional }, { type: Inject, args: [MAT_DATE_FORMATS,] },] }, { type: ChangeDetectorRef, }, ]; MatYearView.propDecorators = { "activeDate": [{ type: Input },], "selected": [{ type: Input },], "minDate": [{ type: Input },], "maxDate": [{ type: Input },], "dateFilter": [{ type: Input },], "selectedChange": [{ type: Output },], }; /** * @fileoverview added by tsickle * @suppress {checkTypes} checked by tsc */ /** * A calendar that is used as part of the datepicker. * \@docs-private */ class MatCalendar { /** * @param {?} _elementRef * @param {?} _intl * @param {?} _ngZone * @param {?} _dateAdapter * @param {?} _dateFormats * @param {?} changeDetectorRef * @param {?=} _dir */ constructor(_elementRef, _intl, _ngZone, _dateAdapter, _dateFormats, changeDetectorRef, _dir) { this._elementRef = _elementRef; this._intl = _intl; this._ngZone = _ngZone; this._dateAdapter = _dateAdapter; this._dateFormats = _dateFormats; this._dir = _dir; /** * Whether the calendar should be started in month or year view. */ this.startView = 'month'; /** * Emits when the currently selected date changes. */ this.selectedChange = new EventEmitter(); /** * Emits when any date is selected. */ this._userSelection = new EventEmitter(); if (!this._dateAdapter) { throw createMissingDateImplError('DateAdapter'); } if (!this._dateFormats) { throw createMissingDateImplError('MAT_DATE_FORMATS'); } this._intlChanges = _intl.changes.subscribe(() => changeDetectorRef.markForCheck()); } /** * A date representing the period (month or year) to start the calendar in. * @return {?} */ get startAt() { return this._startAt; } /** * @param {?} value * @return {?} */ set startAt(value) { this._startAt = this._getValidDateOrNull(this._dateAdapter.deserialize(value)); } /** * The currently selected date. * @return {?} */ get selected() { return this._selected; } /** * @param {?} value * @return {?} */ set selected(value) { this._selected = this._getValidDateOrNull(this._dateAdapter.deserialize(value)); } /** * The minimum selectable date. * @return {?} */ get minDate() { return this._minDate; } /** * @param {?} value * @return {?} */ set minDate(value) { this._minDate = this._getValidDateOrNull(this._dateAdapter.deserialize(value)); } /** * The maximum selectable date. * @return {?} */ get maxDate() { return this._maxDate; } /** * @param {?} value * @return {?} */ set maxDate(value) { this._maxDate = this._getValidDateOrNull(this._dateAdapter.deserialize(value)); } /** * The current active date. This determines which time period is shown and which date is * highlighted when using keyboard navigation. * @return {?} */ get _activeDate() { return this._clampedActiveDate; } /** * @param {?} value * @return {?} */ set _activeDate(value) { this._clampedActiveDate = this._dateAdapter.clampDate(value, this.minDate, this.maxDate); } /** * The label for the current calendar view. * @return {?} */ get _periodButtonText() { if (this._currentView == 'month') { return this._dateAdapter.format(this._activeDate, this._dateFormats.display.monthYearLabel) .toLocaleUpperCase(); } if (this._currentView == 'year') { return this._dateAdapter.getYearName(this._activeDate); } const /** @type {?} */ activeYear = this._dateAdapter.getYear(this._activeDate); const /** @type {?} */ firstYearInView = this._dateAdapter.getYearName(this._dateAdapter.createDate(activeYear - activeYear % 24, 0, 1)); const /** @type {?} */ lastYearInView = this._dateAdapter.getYearName(this._dateAdapter.createDate(activeYear + yearsPerPage - 1 - activeYear % 24, 0, 1)); return `${firstYearInView} \u2013 ${lastYearInView}`; } /** * @return {?} */ get _periodButtonLabel() { return this._currentView == 'month' ? this._intl.switchToMultiYearViewLabel : this._intl.switchToMonthViewLabel; } /** * The label for the the previous button. * @return {?} */ get _prevButtonLabel() { return { 'month': this._intl.prevMonthLabel, 'year': this._intl.prevYearLabel, 'multi-year': this._intl.prevMultiYearLabel }[this._currentView]; } /** * The label for the the next button. * @return {?} */ get _nextButtonLabel() { return { 'month': this._intl.nextMonthLabel, 'year': this._intl.nextYearLabel, 'multi-year': this._intl.nextMultiYearLabel }[this._currentView]; } /** * @return {?} */ ngAfterContentInit() { this._activeDate = this.startAt || this._dateAdapter.today(); this._focusActiveCell(); this._currentView = this.startView; } /** * @return {?} */ ngOnDestroy() { this._intlChanges.unsubscribe(); } /** * @param {?} changes * @return {?} */ ngOnChanges(changes) { const /** @type {?} */ change = changes["minDate"] || changes["maxDate"] || changes["dateFilter"]; if (change && !change.firstChange) { const /** @type {?} */ view = this.monthView || this.yearView || this.multiYearView; if (view) { view._init(); } } } /** * Handles date selection in the month view. * @param {?} date * @return {?} */ _dateSelected(date) { if (!this._dateAdapter.sameDate(date, this.selected)) { this.selectedChange.emit(date); } } /** * @return {?} */ _userSelected() { this._userSelection.emit(); } /** * Handles month selection in the multi-year view. * @param {?} date * @param {?} view * @return {?} */ _goToDateInView(date, view) { this._activeDate = date; this._currentView = view; } /** * Handles user clicks on the period label. * @return {?} */ _currentPeriodClicked() { this._currentView = this._currentView == 'month' ? 'multi-year' : 'month'; } /** * Handles user clicks on the previous button. * @return {?} */ _previousClicked() { this._activeDate = this._currentView == 'month' ? this._dateAdapter.addCalendarMonths(this._activeDate, -1) : this._dateAdapter.addCalendarYears(this._activeDate, this._currentView == 'year' ? -1 : -yearsPerPage); } /** * Handles user clicks on the next button. * @return {?} */ _nextClicked() { this._activeDate = this._currentView == 'month' ? this._dateAdapter.addCalendarMonths(this._activeDate, 1) : this._dateAdapter.addCalendarYears(this._activeDate, this._currentView == 'year' ? 1 : yearsPerPage); } /** * Whether the previous period button is enabled. * @return {?} */ _previousEnabled() { if (!this.minDate) { return true; } return !this.minDate || !this._isSameView(this._activeDate, this.minDate); } /** * Whether the next period button is enabled. * @return {?} */ _nextEnabled() { return !this.maxDate || !this._isSameView(this._activeDate, this.maxDate); } /** * Handles keydown events on the calendar body. * @param {?} event * @return {?} */ _handleCalendarBodyKeydown(event) { // TODO(mmalerba): We currently allow keyboard navigation to disabled dates, but just prevent // disabled ones from being selected. This may not be ideal, we should look into whether // navigation should skip over disabled dates, and if so, how to implement that efficiently. if (this._currentView == 'month') { this._handleCalendarBodyKeydownInMonthView(event); } else if (this._currentView == 'year') { this._handleCalendarBodyKeydownInYearView(event); } else { this._handleCalendarBodyKeydownInMultiYearView(event); } } /** * Focuses the active cell after the microtask queue is empty. * @return {?} */ _focusActiveCell() { this._ngZone.runOutsideAngular(() => { this._ngZone.onStable.asObservable().pipe(take(1)).subscribe(() => { this._elementRef.nativeElement.querySelector('.mat-calendar-body-active').focus(); }); }); } /** * Whether the two dates represent the same view in the current view mode (month or year). * @param {?} date1 * @param {?} date2 * @return {?} */ _isSameView(date1, date2) { if (this._currentView == 'month') { return this._dateAdapter.getYear(date1) == this._dateAdapter.getYear(date2) && this._dateAdapter.getMonth(date1) == this._dateAdapter.getMonth(date2); } if (this._currentView == 'year') { return this._dateAdapter.getYear(date1) == this._dateAdapter.getYear(date2); } // Otherwise we are in 'multi-year' view. return Math.floor(this._dateAdapter.getYear(date1) / yearsPerPage) == Math.floor(this._dateAdapter.getYear(date2) / yearsPerPage); } /** * Handles keydown events on the calendar body when calendar is in month view. * @param {?} event * @return {?} */ _handleCalendarBodyKeydownInMonthView(event) { const /** @type {?} */ isRtl = this._isRtl(); switch (event.keyCode) { case LEFT_ARROW: this._activeDate = this._dateAdapter.addCalendarDays(this._activeDate, isRtl ? 1 : -1); break; case RIGHT_ARROW: this._activeDate = this._dateAdapter.addCalendarDays(this._activeDate, isRtl ? -1 : 1); break; case UP_ARROW: this._activeDate = this._dateAdapter.addCalendarDays(this._activeDate, -7); break; case DOWN_ARROW: this._activeDate = this._dateAdapter.addCalendarDays(this._activeDate, 7); break; case HOME: this._activeDate = this._dateAdapter.addCalendarDays(this._activeDate, 1 - this._dateAdapter.getDate(this._activeDate)); break; case END: this._activeDate = this._dateAdapter.addCalendarDays(this._activeDate, (this._dateAdapter.getNumDaysInMonth(this._activeDate) - this._dateAdapter.getDate(this._activeDate))); break; case PAGE_UP: this._activeDate = event.altKey ? this._dateAdapter.addCalendarYears(this._activeDate, -1) : this._dateAdapter.addCalendarMonths(this._activeDate, -1); break; case PAGE_DOWN: this._activeDate = event.altKey ? this._dateAdapter.addCalendarYears(this._activeDate, 1) : this._dateAdapter.addCalendarMonths(this._activeDate, 1); break; case ENTER: if (!this.dateFilter || this.dateFilter(this._activeDate)) { this._dateSelected(this._activeDate); this._userSelected(); // Prevent unexpected default actions such as form submission. event.preventDefault(); } return; default: // Don't prevent default or focus active cell on keys that we don't explicitly handle. return; } this._focusActiveCell(); // Prevent unexpected default actions such as form submission. event.preventDefault(); } /** * Handles keydown events on the calendar body when calendar is in year view. * @param {?} event * @return {?} */ _handleCalendarBodyKeydownInYearView(event) { const /** @type {?} */ isRtl = this._isRtl(); switch (event.keyCode) { case LEFT_ARROW: this._activeDate = this._dateAdapter.addCalendarMonths(this._activeDate, isRtl ? 1 : -1); break; case RIGHT_ARROW: this._activeDate = this._dateAdapter.addCalendarMonths(this._activeDate, isRtl ? -1 : 1); break; case UP_ARROW: this._activeDate = this._dateAdapter.addCalendarMonths(this._activeDate, -4); break; case DOWN_ARROW: this._activeDate = this._dateAdapter.addCalendarMonths(this._activeDate, 4); break; case HOME: this._activeDate = this._dateAdapter.addCalendarMonths(this._activeDate, -this._dateAdapter.getMonth(this._activeDate)); break; case END: this._activeDate = this._dateAdapter.addCalendarMonths(this._activeDate, 11 - this._dateAdapter.getMonth(this._activeDate)); break; case PAGE_UP: this._activeDate = this._dateAdapter.addCalendarYears(this._activeDate, event.altKey ? -10 : -1); break; case PAGE_DOWN: this._activeDate = this._dateAdapter.addCalendarYears(this._activeDate, event.altKey ? 10 : 1); break; case ENTER: this._goToDateInView(this._activeDate, 'month'); break; default: // Don't prevent default or focus active cell on keys that we don't explicitly handle. return; } this._focusActiveCell(); // Prevent unexpected default actions such as form submission. event.preventDefault(); } /** * Handles keydown events on the calendar body when calendar is in multi-year view. * @param {?} event * @return {?} */ _handleCalendarBodyKeydownInMultiYearView(event) { switch (event.keyCode) { case LEFT_ARROW: this._activeDate = this._dateAdapter.addCalendarYears(this._activeDate, -1); break; case RIGHT_ARROW: this._activeDate = this._dateAdapter.addCalendarYears(this._activeDate, 1); break; case UP_ARROW: this._activeDate = this._dateAdapter.addCalendarYears(this._activeDate, -yearsPerRow); break; case DOWN_ARROW: this._activeDate = this._dateAdapter.addCalendarYears(this._activeDate, yearsPerRow); break; case HOME: this._activeDate = this._dateAdapter.addCalendarYears(this._activeDate, -this._dateAdapter.getYear(this._activeDate) % yearsPerPage); break; case END: this._activeDate = this._dateAdapter.addCalendarYears(this._activeDate, yearsPerPage - this._dateAdapter.getYear(this._activeDate) % yearsPerPage - 1); break; case PAGE_UP: this._activeDate = this._dateAdapter.addCalendarYears(this._activeDate, event.altKey ? -yearsPerPage * 10 : -yearsPerPage); break; case PAGE_DOWN: this._activeDate = this._dateAdapter.addCalendarYears(this._activeDate, event.altKey ? yearsPerPage * 10 : yearsPerPage); break; case ENTER: this._goToDateInView(this._activeDate, 'year'); break; default: // Don't prevent default or focus active cell on keys that we don't explicitly handle. return; } this._focusActiveCell(); // Prevent unexpected default actions such as form submission. event.preventDefault(); } /** * @param {?} obj The object to check. * @return {?} The given object if it is both a date instance and valid, otherwise null. */ _getValidDateOrNull(obj) { return (this._dateAdapter.isDateInstance(obj) && this._dateAdapter.isValid(obj)) ? obj : null; } /** * Determines whether the user has the RTL layout direction. * @return {?} */ _isRtl() { return this._dir && this._dir.value === 'rtl'; } } MatCalendar.decorators = [ { type: Component, args: [{selector: 'mat-calendar',