1 | /**
|
---|
2 | * @license
|
---|
3 | * Copyright Google LLC All Rights Reserved.
|
---|
4 | *
|
---|
5 | * Use of this source code is governed by an MIT-style license that can be
|
---|
6 | * found in the LICENSE file at https://angular.io/license
|
---|
7 | */
|
---|
8 | import { ComponentPortal } from '@angular/cdk/portal';
|
---|
9 | import { ChangeDetectionStrategy, ChangeDetectorRef, Component, EventEmitter, forwardRef, Inject, Input, Optional, Output, ViewChild, ViewEncapsulation, } from '@angular/core';
|
---|
10 | import { DateAdapter, MAT_DATE_FORMATS, } from '@angular/material/core';
|
---|
11 | import { Subject } from 'rxjs';
|
---|
12 | import { createMissingDateImplError } from './datepicker-errors';
|
---|
13 | import { MatDatepickerIntl } from './datepicker-intl';
|
---|
14 | import { MatMonthView } from './month-view';
|
---|
15 | import { getActiveOffset, isSameMultiYearView, MatMultiYearView, yearsPerPage } from './multi-year-view';
|
---|
16 | import { MatYearView } from './year-view';
|
---|
17 | import { MAT_SINGLE_DATE_SELECTION_MODEL_PROVIDER, DateRange } from './date-selection-model';
|
---|
18 | /** Counter used to generate unique IDs. */
|
---|
19 | let uniqueId = 0;
|
---|
20 | /** Default header for MatCalendar */
|
---|
21 | export class MatCalendarHeader {
|
---|
22 | constructor(_intl, calendar, _dateAdapter, _dateFormats, changeDetectorRef) {
|
---|
23 | this._intl = _intl;
|
---|
24 | this.calendar = calendar;
|
---|
25 | this._dateAdapter = _dateAdapter;
|
---|
26 | this._dateFormats = _dateFormats;
|
---|
27 | this._buttonDescriptionId = `mat-calendar-button-${uniqueId++}`;
|
---|
28 | this.calendar.stateChanges.subscribe(() => changeDetectorRef.markForCheck());
|
---|
29 | }
|
---|
30 | /** The label for the current calendar view. */
|
---|
31 | get periodButtonText() {
|
---|
32 | if (this.calendar.currentView == 'month') {
|
---|
33 | return this._dateAdapter
|
---|
34 | .format(this.calendar.activeDate, this._dateFormats.display.monthYearLabel)
|
---|
35 | .toLocaleUpperCase();
|
---|
36 | }
|
---|
37 | if (this.calendar.currentView == 'year') {
|
---|
38 | return this._dateAdapter.getYearName(this.calendar.activeDate);
|
---|
39 | }
|
---|
40 | // The offset from the active year to the "slot" for the starting year is the
|
---|
41 | // *actual* first rendered year in the multi-year view, and the last year is
|
---|
42 | // just yearsPerPage - 1 away.
|
---|
43 | const activeYear = this._dateAdapter.getYear(this.calendar.activeDate);
|
---|
44 | const minYearOfPage = activeYear - getActiveOffset(this._dateAdapter, this.calendar.activeDate, this.calendar.minDate, this.calendar.maxDate);
|
---|
45 | const maxYearOfPage = minYearOfPage + yearsPerPage - 1;
|
---|
46 | const minYearName = this._dateAdapter.getYearName(this._dateAdapter.createDate(minYearOfPage, 0, 1));
|
---|
47 | const maxYearName = this._dateAdapter.getYearName(this._dateAdapter.createDate(maxYearOfPage, 0, 1));
|
---|
48 | return this._intl.formatYearRange(minYearName, maxYearName);
|
---|
49 | }
|
---|
50 | get periodButtonLabel() {
|
---|
51 | return this.calendar.currentView == 'month' ?
|
---|
52 | this._intl.switchToMultiYearViewLabel : this._intl.switchToMonthViewLabel;
|
---|
53 | }
|
---|
54 | /** The label for the previous button. */
|
---|
55 | get prevButtonLabel() {
|
---|
56 | return {
|
---|
57 | 'month': this._intl.prevMonthLabel,
|
---|
58 | 'year': this._intl.prevYearLabel,
|
---|
59 | 'multi-year': this._intl.prevMultiYearLabel
|
---|
60 | }[this.calendar.currentView];
|
---|
61 | }
|
---|
62 | /** The label for the next button. */
|
---|
63 | get nextButtonLabel() {
|
---|
64 | return {
|
---|
65 | 'month': this._intl.nextMonthLabel,
|
---|
66 | 'year': this._intl.nextYearLabel,
|
---|
67 | 'multi-year': this._intl.nextMultiYearLabel
|
---|
68 | }[this.calendar.currentView];
|
---|
69 | }
|
---|
70 | /** Handles user clicks on the period label. */
|
---|
71 | currentPeriodClicked() {
|
---|
72 | this.calendar.currentView = this.calendar.currentView == 'month' ? 'multi-year' : 'month';
|
---|
73 | }
|
---|
74 | /** Handles user clicks on the previous button. */
|
---|
75 | previousClicked() {
|
---|
76 | this.calendar.activeDate = this.calendar.currentView == 'month' ?
|
---|
77 | this._dateAdapter.addCalendarMonths(this.calendar.activeDate, -1) :
|
---|
78 | this._dateAdapter.addCalendarYears(this.calendar.activeDate, this.calendar.currentView == 'year' ? -1 : -yearsPerPage);
|
---|
79 | }
|
---|
80 | /** Handles user clicks on the next button. */
|
---|
81 | nextClicked() {
|
---|
82 | this.calendar.activeDate = this.calendar.currentView == 'month' ?
|
---|
83 | this._dateAdapter.addCalendarMonths(this.calendar.activeDate, 1) :
|
---|
84 | this._dateAdapter.addCalendarYears(this.calendar.activeDate, this.calendar.currentView == 'year' ? 1 : yearsPerPage);
|
---|
85 | }
|
---|
86 | /** Whether the previous period button is enabled. */
|
---|
87 | previousEnabled() {
|
---|
88 | if (!this.calendar.minDate) {
|
---|
89 | return true;
|
---|
90 | }
|
---|
91 | return !this.calendar.minDate ||
|
---|
92 | !this._isSameView(this.calendar.activeDate, this.calendar.minDate);
|
---|
93 | }
|
---|
94 | /** Whether the next period button is enabled. */
|
---|
95 | nextEnabled() {
|
---|
96 | return !this.calendar.maxDate ||
|
---|
97 | !this._isSameView(this.calendar.activeDate, this.calendar.maxDate);
|
---|
98 | }
|
---|
99 | /** Whether the two dates represent the same view in the current view mode (month or year). */
|
---|
100 | _isSameView(date1, date2) {
|
---|
101 | if (this.calendar.currentView == 'month') {
|
---|
102 | return this._dateAdapter.getYear(date1) == this._dateAdapter.getYear(date2) &&
|
---|
103 | this._dateAdapter.getMonth(date1) == this._dateAdapter.getMonth(date2);
|
---|
104 | }
|
---|
105 | if (this.calendar.currentView == 'year') {
|
---|
106 | return this._dateAdapter.getYear(date1) == this._dateAdapter.getYear(date2);
|
---|
107 | }
|
---|
108 | // Otherwise we are in 'multi-year' view.
|
---|
109 | return isSameMultiYearView(this._dateAdapter, date1, date2, this.calendar.minDate, this.calendar.maxDate);
|
---|
110 | }
|
---|
111 | }
|
---|
112 | MatCalendarHeader.decorators = [
|
---|
113 | { type: Component, args: [{
|
---|
114 | selector: 'mat-calendar-header',
|
---|
115 | template: "<div class=\"mat-calendar-header\">\n <div class=\"mat-calendar-controls\">\n <button mat-button type=\"button\" class=\"mat-calendar-period-button\"\n (click)=\"currentPeriodClicked()\" [attr.aria-label]=\"periodButtonLabel\"\n [attr.aria-describedby]=\"_buttonDescriptionId\"\n cdkAriaLive=\"polite\">\n <span [attr.id]=\"_buttonDescriptionId\">{{periodButtonText}}</span>\n <div class=\"mat-calendar-arrow\"\n [class.mat-calendar-invert]=\"calendar.currentView !== 'month'\"></div>\n </button>\n\n <div class=\"mat-calendar-spacer\"></div>\n\n <ng-content></ng-content>\n\n <button mat-icon-button type=\"button\" class=\"mat-calendar-previous-button\"\n [disabled]=\"!previousEnabled()\" (click)=\"previousClicked()\"\n [attr.aria-label]=\"prevButtonLabel\">\n </button>\n\n <button mat-icon-button type=\"button\" class=\"mat-calendar-next-button\"\n [disabled]=\"!nextEnabled()\" (click)=\"nextClicked()\"\n [attr.aria-label]=\"nextButtonLabel\">\n </button>\n </div>\n</div>\n",
|
---|
116 | exportAs: 'matCalendarHeader',
|
---|
117 | encapsulation: ViewEncapsulation.None,
|
---|
118 | changeDetection: ChangeDetectionStrategy.OnPush
|
---|
119 | },] }
|
---|
120 | ];
|
---|
121 | MatCalendarHeader.ctorParameters = () => [
|
---|
122 | { type: MatDatepickerIntl },
|
---|
123 | { type: MatCalendar, decorators: [{ type: Inject, args: [forwardRef(() => MatCalendar),] }] },
|
---|
124 | { type: DateAdapter, decorators: [{ type: Optional }] },
|
---|
125 | { type: undefined, decorators: [{ type: Optional }, { type: Inject, args: [MAT_DATE_FORMATS,] }] },
|
---|
126 | { type: ChangeDetectorRef }
|
---|
127 | ];
|
---|
128 | /** A calendar that is used as part of the datepicker. */
|
---|
129 | export class MatCalendar {
|
---|
130 | constructor(_intl, _dateAdapter, _dateFormats, _changeDetectorRef) {
|
---|
131 | this._dateAdapter = _dateAdapter;
|
---|
132 | this._dateFormats = _dateFormats;
|
---|
133 | this._changeDetectorRef = _changeDetectorRef;
|
---|
134 | /**
|
---|
135 | * Used for scheduling that focus should be moved to the active cell on the next tick.
|
---|
136 | * We need to schedule it, rather than do it immediately, because we have to wait
|
---|
137 | * for Angular to re-evaluate the view children.
|
---|
138 | */
|
---|
139 | this._moveFocusOnNextTick = false;
|
---|
140 | /** Whether the calendar should be started in month or year view. */
|
---|
141 | this.startView = 'month';
|
---|
142 | /** Emits when the currently selected date changes. */
|
---|
143 | this.selectedChange = new EventEmitter();
|
---|
144 | /**
|
---|
145 | * Emits the year chosen in multiyear view.
|
---|
146 | * This doesn't imply a change on the selected date.
|
---|
147 | */
|
---|
148 | this.yearSelected = new EventEmitter();
|
---|
149 | /**
|
---|
150 | * Emits the month chosen in year view.
|
---|
151 | * This doesn't imply a change on the selected date.
|
---|
152 | */
|
---|
153 | this.monthSelected = new EventEmitter();
|
---|
154 | /**
|
---|
155 | * Emits when the current view changes.
|
---|
156 | */
|
---|
157 | this.viewChanged = new EventEmitter(true);
|
---|
158 | /** Emits when any date is selected. */
|
---|
159 | this._userSelection = new EventEmitter();
|
---|
160 | /**
|
---|
161 | * Emits whenever there is a state change that the header may need to respond to.
|
---|
162 | */
|
---|
163 | this.stateChanges = new Subject();
|
---|
164 | if (typeof ngDevMode === 'undefined' || ngDevMode) {
|
---|
165 | if (!this._dateAdapter) {
|
---|
166 | throw createMissingDateImplError('DateAdapter');
|
---|
167 | }
|
---|
168 | if (!this._dateFormats) {
|
---|
169 | throw createMissingDateImplError('MAT_DATE_FORMATS');
|
---|
170 | }
|
---|
171 | }
|
---|
172 | this._intlChanges = _intl.changes.subscribe(() => {
|
---|
173 | _changeDetectorRef.markForCheck();
|
---|
174 | this.stateChanges.next();
|
---|
175 | });
|
---|
176 | }
|
---|
177 | /** A date representing the period (month or year) to start the calendar in. */
|
---|
178 | get startAt() { return this._startAt; }
|
---|
179 | set startAt(value) {
|
---|
180 | this._startAt = this._dateAdapter.getValidDateOrNull(this._dateAdapter.deserialize(value));
|
---|
181 | }
|
---|
182 | /** The currently selected date. */
|
---|
183 | get selected() { return this._selected; }
|
---|
184 | set selected(value) {
|
---|
185 | if (value instanceof DateRange) {
|
---|
186 | this._selected = value;
|
---|
187 | }
|
---|
188 | else {
|
---|
189 | this._selected = this._dateAdapter.getValidDateOrNull(this._dateAdapter.deserialize(value));
|
---|
190 | }
|
---|
191 | }
|
---|
192 | /** The minimum selectable date. */
|
---|
193 | get minDate() { return this._minDate; }
|
---|
194 | set minDate(value) {
|
---|
195 | this._minDate = this._dateAdapter.getValidDateOrNull(this._dateAdapter.deserialize(value));
|
---|
196 | }
|
---|
197 | /** The maximum selectable date. */
|
---|
198 | get maxDate() { return this._maxDate; }
|
---|
199 | set maxDate(value) {
|
---|
200 | this._maxDate = this._dateAdapter.getValidDateOrNull(this._dateAdapter.deserialize(value));
|
---|
201 | }
|
---|
202 | /**
|
---|
203 | * The current active date. This determines which time period is shown and which date is
|
---|
204 | * highlighted when using keyboard navigation.
|
---|
205 | */
|
---|
206 | get activeDate() { return this._clampedActiveDate; }
|
---|
207 | set activeDate(value) {
|
---|
208 | this._clampedActiveDate = this._dateAdapter.clampDate(value, this.minDate, this.maxDate);
|
---|
209 | this.stateChanges.next();
|
---|
210 | this._changeDetectorRef.markForCheck();
|
---|
211 | }
|
---|
212 | /** Whether the calendar is in month view. */
|
---|
213 | get currentView() { return this._currentView; }
|
---|
214 | set currentView(value) {
|
---|
215 | const viewChangedResult = this._currentView !== value ? value : null;
|
---|
216 | this._currentView = value;
|
---|
217 | this._moveFocusOnNextTick = true;
|
---|
218 | this._changeDetectorRef.markForCheck();
|
---|
219 | if (viewChangedResult) {
|
---|
220 | this.viewChanged.emit(viewChangedResult);
|
---|
221 | }
|
---|
222 | }
|
---|
223 | ngAfterContentInit() {
|
---|
224 | this._calendarHeaderPortal = new ComponentPortal(this.headerComponent || MatCalendarHeader);
|
---|
225 | this.activeDate = this.startAt || this._dateAdapter.today();
|
---|
226 | // Assign to the private property since we don't want to move focus on init.
|
---|
227 | this._currentView = this.startView;
|
---|
228 | }
|
---|
229 | ngAfterViewChecked() {
|
---|
230 | if (this._moveFocusOnNextTick) {
|
---|
231 | this._moveFocusOnNextTick = false;
|
---|
232 | this.focusActiveCell();
|
---|
233 | }
|
---|
234 | }
|
---|
235 | ngOnDestroy() {
|
---|
236 | this._intlChanges.unsubscribe();
|
---|
237 | this.stateChanges.complete();
|
---|
238 | }
|
---|
239 | ngOnChanges(changes) {
|
---|
240 | const change = changes['minDate'] || changes['maxDate'] || changes['dateFilter'];
|
---|
241 | if (change && !change.firstChange) {
|
---|
242 | const view = this._getCurrentViewComponent();
|
---|
243 | if (view) {
|
---|
244 | // We need to `detectChanges` manually here, because the `minDate`, `maxDate` etc. are
|
---|
245 | // passed down to the view via data bindings which won't be up-to-date when we call `_init`.
|
---|
246 | this._changeDetectorRef.detectChanges();
|
---|
247 | view._init();
|
---|
248 | }
|
---|
249 | }
|
---|
250 | this.stateChanges.next();
|
---|
251 | }
|
---|
252 | /** Focuses the active date. */
|
---|
253 | focusActiveCell() {
|
---|
254 | this._getCurrentViewComponent()._focusActiveCell(false);
|
---|
255 | }
|
---|
256 | /** Updates today's date after an update of the active date */
|
---|
257 | updateTodaysDate() {
|
---|
258 | this._getCurrentViewComponent()._init();
|
---|
259 | }
|
---|
260 | /** Handles date selection in the month view. */
|
---|
261 | _dateSelected(event) {
|
---|
262 | const date = event.value;
|
---|
263 | if (this.selected instanceof DateRange ||
|
---|
264 | (date && !this._dateAdapter.sameDate(date, this.selected))) {
|
---|
265 | this.selectedChange.emit(date);
|
---|
266 | }
|
---|
267 | this._userSelection.emit(event);
|
---|
268 | }
|
---|
269 | /** Handles year selection in the multiyear view. */
|
---|
270 | _yearSelectedInMultiYearView(normalizedYear) {
|
---|
271 | this.yearSelected.emit(normalizedYear);
|
---|
272 | }
|
---|
273 | /** Handles month selection in the year view. */
|
---|
274 | _monthSelectedInYearView(normalizedMonth) {
|
---|
275 | this.monthSelected.emit(normalizedMonth);
|
---|
276 | }
|
---|
277 | /** Handles year/month selection in the multi-year/year views. */
|
---|
278 | _goToDateInView(date, view) {
|
---|
279 | this.activeDate = date;
|
---|
280 | this.currentView = view;
|
---|
281 | }
|
---|
282 | /** Returns the component instance that corresponds to the current calendar view. */
|
---|
283 | _getCurrentViewComponent() {
|
---|
284 | // The return type is explicitly written as a union to ensure that the Closure compiler does
|
---|
285 | // not optimize calls to _init(). Without the explict return type, TypeScript narrows it to
|
---|
286 | // only the first component type. See https://github.com/angular/components/issues/22996.
|
---|
287 | return this.monthView || this.yearView || this.multiYearView;
|
---|
288 | }
|
---|
289 | }
|
---|
290 | MatCalendar.decorators = [
|
---|
291 | { type: Component, args: [{
|
---|
292 | selector: 'mat-calendar',
|
---|
293 | template: "<ng-template [cdkPortalOutlet]=\"_calendarHeaderPortal\"></ng-template>\n\n<div class=\"mat-calendar-content\" [ngSwitch]=\"currentView\" cdkMonitorSubtreeFocus tabindex=\"-1\">\n <mat-month-view\n *ngSwitchCase=\"'month'\"\n [(activeDate)]=\"activeDate\"\n [selected]=\"selected\"\n [dateFilter]=\"dateFilter\"\n [maxDate]=\"maxDate\"\n [minDate]=\"minDate\"\n [dateClass]=\"dateClass\"\n [comparisonStart]=\"comparisonStart\"\n [comparisonEnd]=\"comparisonEnd\"\n (_userSelection)=\"_dateSelected($event)\">\n </mat-month-view>\n\n <mat-year-view\n *ngSwitchCase=\"'year'\"\n [(activeDate)]=\"activeDate\"\n [selected]=\"selected\"\n [dateFilter]=\"dateFilter\"\n [maxDate]=\"maxDate\"\n [minDate]=\"minDate\"\n [dateClass]=\"dateClass\"\n (monthSelected)=\"_monthSelectedInYearView($event)\"\n (selectedChange)=\"_goToDateInView($event, 'month')\">\n </mat-year-view>\n\n <mat-multi-year-view\n *ngSwitchCase=\"'multi-year'\"\n [(activeDate)]=\"activeDate\"\n [selected]=\"selected\"\n [dateFilter]=\"dateFilter\"\n [maxDate]=\"maxDate\"\n [minDate]=\"minDate\"\n [dateClass]=\"dateClass\"\n (yearSelected)=\"_yearSelectedInMultiYearView($event)\"\n (selectedChange)=\"_goToDateInView($event, 'year')\">\n </mat-multi-year-view>\n</div>\n",
|
---|
294 | host: {
|
---|
295 | 'class': 'mat-calendar',
|
---|
296 | },
|
---|
297 | exportAs: 'matCalendar',
|
---|
298 | encapsulation: ViewEncapsulation.None,
|
---|
299 | changeDetection: ChangeDetectionStrategy.OnPush,
|
---|
300 | providers: [MAT_SINGLE_DATE_SELECTION_MODEL_PROVIDER],
|
---|
301 | styles: [".mat-calendar{display:block}.mat-calendar-header{padding:8px 8px 0 8px}.mat-calendar-content{padding:0 8px 8px 8px;outline:none}.mat-calendar-controls{display:flex;margin:5% calc(33% / 7 - 16px)}.mat-calendar-controls .mat-icon-button:hover .mat-button-focus-overlay{opacity:.04}.mat-calendar-spacer{flex:1 1 auto}.mat-calendar-period-button{min-width:0}.mat-calendar-arrow{display:inline-block;width:0;height:0;border-left:5px solid transparent;border-right:5px solid transparent;border-top-width:5px;border-top-style:solid;margin:0 0 0 5px;vertical-align:middle}.mat-calendar-arrow.mat-calendar-invert{transform:rotate(180deg)}[dir=rtl] .mat-calendar-arrow{margin:0 5px 0 0}.mat-calendar-previous-button,.mat-calendar-next-button{position:relative}.mat-calendar-previous-button::after,.mat-calendar-next-button::after{top:0;left:0;right:0;bottom:0;position:absolute;content:\"\";margin:15.5px;border:0 solid currentColor;border-top-width:2px}[dir=rtl] .mat-calendar-previous-button,[dir=rtl] .mat-calendar-next-button{transform:rotate(180deg)}.mat-calendar-previous-button::after{border-left-width:2px;transform:translateX(2px) rotate(-45deg)}.mat-calendar-next-button::after{border-right-width:2px;transform:translateX(-2px) rotate(45deg)}.mat-calendar-table{border-spacing:0;border-collapse:collapse;width:100%}.mat-calendar-table-header th{text-align:center;padding:0 0 8px 0}.mat-calendar-table-header-divider{position:relative;height:1px}.mat-calendar-table-header-divider::after{content:\"\";position:absolute;top:0;left:-8px;right:-8px;height:1px}.mat-calendar-abbr{text-decoration:none}\n"]
|
---|
302 | },] }
|
---|
303 | ];
|
---|
304 | MatCalendar.ctorParameters = () => [
|
---|
305 | { type: MatDatepickerIntl },
|
---|
306 | { type: DateAdapter, decorators: [{ type: Optional }] },
|
---|
307 | { type: undefined, decorators: [{ type: Optional }, { type: Inject, args: [MAT_DATE_FORMATS,] }] },
|
---|
308 | { type: ChangeDetectorRef }
|
---|
309 | ];
|
---|
310 | MatCalendar.propDecorators = {
|
---|
311 | headerComponent: [{ type: Input }],
|
---|
312 | startAt: [{ type: Input }],
|
---|
313 | startView: [{ type: Input }],
|
---|
314 | selected: [{ type: Input }],
|
---|
315 | minDate: [{ type: Input }],
|
---|
316 | maxDate: [{ type: Input }],
|
---|
317 | dateFilter: [{ type: Input }],
|
---|
318 | dateClass: [{ type: Input }],
|
---|
319 | comparisonStart: [{ type: Input }],
|
---|
320 | comparisonEnd: [{ type: Input }],
|
---|
321 | selectedChange: [{ type: Output }],
|
---|
322 | yearSelected: [{ type: Output }],
|
---|
323 | monthSelected: [{ type: Output }],
|
---|
324 | viewChanged: [{ type: Output }],
|
---|
325 | _userSelection: [{ type: Output }],
|
---|
326 | monthView: [{ type: ViewChild, args: [MatMonthView,] }],
|
---|
327 | yearView: [{ type: ViewChild, args: [MatYearView,] }],
|
---|
328 | multiYearView: [{ type: ViewChild, args: [MatMultiYearView,] }]
|
---|
329 | };
|
---|
330 | //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"calendar.js","sourceRoot":"","sources":["../../../../../../src/material/datepicker/calendar.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * @license\n * Copyright Google LLC All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.io/license\n */\n\nimport {ComponentPortal, ComponentType, Portal} from '@angular/cdk/portal';\nimport {\n  AfterContentInit,\n  AfterViewChecked,\n  ChangeDetectionStrategy,\n  ChangeDetectorRef,\n  Component,\n  EventEmitter,\n  forwardRef,\n  Inject,\n  Input,\n  OnChanges,\n  OnDestroy,\n  Optional,\n  Output,\n  SimpleChanges,\n  ViewChild,\n  ViewEncapsulation,\n} from '@angular/core';\nimport {\n  DateAdapter,\n  MAT_DATE_FORMATS,\n  MatDateFormats,\n} from '@angular/material/core';\nimport {Subject, Subscription} from 'rxjs';\nimport {MatCalendarUserEvent, MatCalendarCellClassFunction} from './calendar-body';\nimport {createMissingDateImplError} from './datepicker-errors';\nimport {MatDatepickerIntl} from './datepicker-intl';\nimport {MatMonthView} from './month-view';\nimport {\n  getActiveOffset,\n  isSameMultiYearView,\n  MatMultiYearView,\n  yearsPerPage\n} from './multi-year-view';\nimport {MatYearView} from './year-view';\nimport {MAT_SINGLE_DATE_SELECTION_MODEL_PROVIDER, DateRange} from './date-selection-model';\n\n/**\n * Possible views for the calendar.\n * @docs-private\n */\nexport type MatCalendarView = 'month' | 'year' | 'multi-year';\n\n/** Counter used to generate unique IDs. */\nlet uniqueId = 0;\n\n/** Default header for MatCalendar */\n@Component({\n  selector: 'mat-calendar-header',\n  templateUrl: 'calendar-header.html',\n  exportAs: 'matCalendarHeader',\n  encapsulation: ViewEncapsulation.None,\n  changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class MatCalendarHeader<D> {\n  _buttonDescriptionId = `mat-calendar-button-${uniqueId++}`;\n\n  constructor(private _intl: MatDatepickerIntl,\n              @Inject(forwardRef(() => MatCalendar)) public calendar: MatCalendar<D>,\n              @Optional() private _dateAdapter: DateAdapter<D>,\n              @Optional() @Inject(MAT_DATE_FORMATS) private _dateFormats: MatDateFormats,\n              changeDetectorRef: ChangeDetectorRef) {\n\n    this.calendar.stateChanges.subscribe(() => changeDetectorRef.markForCheck());\n  }\n\n  /** The label for the current calendar view. */\n  get periodButtonText(): string {\n    if (this.calendar.currentView == 'month') {\n      return this._dateAdapter\n          .format(this.calendar.activeDate, this._dateFormats.display.monthYearLabel)\n              .toLocaleUpperCase();\n    }\n    if (this.calendar.currentView == 'year') {\n      return this._dateAdapter.getYearName(this.calendar.activeDate);\n    }\n\n    // The offset from the active year to the \"slot\" for the starting year is the\n    // *actual* first rendered year in the multi-year view, and the last year is\n    // just yearsPerPage - 1 away.\n    const activeYear = this._dateAdapter.getYear(this.calendar.activeDate);\n    const minYearOfPage = activeYear - getActiveOffset(\n      this._dateAdapter, this.calendar.activeDate, this.calendar.minDate, this.calendar.maxDate);\n    const maxYearOfPage = minYearOfPage + yearsPerPage - 1;\n    const minYearName =\n      this._dateAdapter.getYearName(this._dateAdapter.createDate(minYearOfPage, 0, 1));\n    const maxYearName =\n      this._dateAdapter.getYearName(this._dateAdapter.createDate(maxYearOfPage, 0, 1));\n    return this._intl.formatYearRange(minYearName, maxYearName);\n  }\n\n  get periodButtonLabel(): string {\n    return this.calendar.currentView == 'month' ?\n        this._intl.switchToMultiYearViewLabel : this._intl.switchToMonthViewLabel;\n  }\n\n  /** The label for the previous button. */\n  get prevButtonLabel(): string {\n    return {\n      'month': this._intl.prevMonthLabel,\n      'year': this._intl.prevYearLabel,\n      'multi-year': this._intl.prevMultiYearLabel\n    }[this.calendar.currentView];\n  }\n\n  /** The label for the next button. */\n  get nextButtonLabel(): string {\n    return {\n      'month': this._intl.nextMonthLabel,\n      'year': this._intl.nextYearLabel,\n      'multi-year': this._intl.nextMultiYearLabel\n    }[this.calendar.currentView];\n  }\n\n  /** Handles user clicks on the period label. */\n  currentPeriodClicked(): void {\n    this.calendar.currentView = this.calendar.currentView == 'month' ? 'multi-year' : 'month';\n  }\n\n  /** Handles user clicks on the previous button. */\n  previousClicked(): void {\n    this.calendar.activeDate = this.calendar.currentView == 'month' ?\n        this._dateAdapter.addCalendarMonths(this.calendar.activeDate, -1) :\n            this._dateAdapter.addCalendarYears(\n                this.calendar.activeDate, this.calendar.currentView == 'year' ? -1 : -yearsPerPage\n            );\n  }\n\n  /** Handles user clicks on the next button. */\n  nextClicked(): void {\n    this.calendar.activeDate = this.calendar.currentView == 'month' ?\n        this._dateAdapter.addCalendarMonths(this.calendar.activeDate, 1) :\n            this._dateAdapter.addCalendarYears(\n                this.calendar.activeDate,\n                    this.calendar.currentView == 'year' ? 1 : yearsPerPage\n            );\n  }\n\n  /** Whether the previous period button is enabled. */\n  previousEnabled(): boolean {\n    if (!this.calendar.minDate) {\n      return true;\n    }\n    return !this.calendar.minDate ||\n        !this._isSameView(this.calendar.activeDate, this.calendar.minDate);\n  }\n\n  /** Whether the next period button is enabled. */\n  nextEnabled(): boolean {\n    return !this.calendar.maxDate ||\n        !this._isSameView(this.calendar.activeDate, this.calendar.maxDate);\n  }\n\n  /** Whether the two dates represent the same view in the current view mode (month or year). */\n  private _isSameView(date1: D, date2: D): boolean {\n    if (this.calendar.currentView == 'month') {\n      return this._dateAdapter.getYear(date1) == this._dateAdapter.getYear(date2) &&\n          this._dateAdapter.getMonth(date1) == this._dateAdapter.getMonth(date2);\n    }\n    if (this.calendar.currentView == 'year') {\n      return this._dateAdapter.getYear(date1) == this._dateAdapter.getYear(date2);\n    }\n    // Otherwise we are in 'multi-year' view.\n    return isSameMultiYearView(\n      this._dateAdapter, date1, date2, this.calendar.minDate, this.calendar.maxDate);\n  }\n}\n\n/** A calendar that is used as part of the datepicker. */\n@Component({\n  selector: 'mat-calendar',\n  templateUrl: 'calendar.html',\n  styleUrls: ['calendar.css'],\n  host: {\n    'class': 'mat-calendar',\n  },\n  exportAs: 'matCalendar',\n  encapsulation: ViewEncapsulation.None,\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  providers: [MAT_SINGLE_DATE_SELECTION_MODEL_PROVIDER]\n})\nexport class MatCalendar<D> implements AfterContentInit, AfterViewChecked, OnDestroy, OnChanges {\n  /** An input indicating the type of the header component, if set. */\n  @Input() headerComponent: ComponentType<any>;\n\n  /** A portal containing the header component type for this calendar. */\n  _calendarHeaderPortal: Portal<any>;\n\n  private _intlChanges: Subscription;\n\n  /**\n   * Used for scheduling that focus should be moved to the active cell on the next tick.\n   * We need to schedule it, rather than do it immediately, because we have to wait\n   * for Angular to re-evaluate the view children.\n   */\n  private _moveFocusOnNextTick = false;\n\n  /** A date representing the period (month or year) to start the calendar in. */\n  @Input()\n  get startAt(): D | null { return this._startAt; }\n  set startAt(value: D | null) {\n    this._startAt = this._dateAdapter.getValidDateOrNull(this._dateAdapter.deserialize(value));\n  }\n  private _startAt: D | null;\n\n  /** Whether the calendar should be started in month or year view. */\n  @Input() startView: MatCalendarView = 'month';\n\n  /** The currently selected date. */\n  @Input()\n  get selected(): DateRange<D> | D | null { return this._selected; }\n  set selected(value: DateRange<D> | D | null) {\n    if (value instanceof DateRange) {\n      this._selected = value;\n    } else {\n      this._selected = this._dateAdapter.getValidDateOrNull(this._dateAdapter.deserialize(value));\n    }\n  }\n  private _selected: DateRange<D> | D | null;\n\n  /** The minimum selectable date. */\n  @Input()\n  get minDate(): D | null { return this._minDate; }\n  set minDate(value: D | null) {\n    this._minDate = this._dateAdapter.getValidDateOrNull(this._dateAdapter.deserialize(value));\n  }\n  private _minDate: D | null;\n\n  /** The maximum selectable date. */\n  @Input()\n  get maxDate(): D | null { return this._maxDate; }\n  set maxDate(value: D | null) {\n    this._maxDate = this._dateAdapter.getValidDateOrNull(this._dateAdapter.deserialize(value));\n  }\n  private _maxDate: D | null;\n\n  /** Function used to filter which dates are selectable. */\n  @Input() dateFilter: (date: D) => boolean;\n\n  /** Function that can be used to add custom CSS classes to dates. */\n  @Input() dateClass: MatCalendarCellClassFunction<D>;\n\n  /** Start of the comparison range. */\n  @Input() comparisonStart: D | null;\n\n  /** End of the comparison range. */\n  @Input() comparisonEnd: D | null;\n\n  /** Emits when the currently selected date changes. */\n  @Output() readonly selectedChange: EventEmitter<D | null> = new EventEmitter<D | null>();\n\n  /**\n   * Emits the year chosen in multiyear view.\n   * This doesn't imply a change on the selected date.\n   */\n  @Output() readonly yearSelected: EventEmitter<D> = new EventEmitter<D>();\n\n  /**\n   * Emits the month chosen in year view.\n   * This doesn't imply a change on the selected date.\n   */\n  @Output() readonly monthSelected: EventEmitter<D> = new EventEmitter<D>();\n\n  /**\n   * Emits when the current view changes.\n   */\n  @Output() readonly viewChanged: EventEmitter<MatCalendarView> =\n    new EventEmitter<MatCalendarView>(true);\n\n  /** Emits when any date is selected. */\n  @Output() readonly _userSelection: EventEmitter<MatCalendarUserEvent<D | null>> =\n      new EventEmitter<MatCalendarUserEvent<D | null>>();\n\n  /** Reference to the current month view component. */\n  @ViewChild(MatMonthView) monthView: MatMonthView<D>;\n\n  /** Reference to the current year view component. */\n  @ViewChild(MatYearView) yearView: MatYearView<D>;\n\n  /** Reference to the current multi-year view component. */\n  @ViewChild(MatMultiYearView) multiYearView: MatMultiYearView<D>;\n\n  /**\n   * The current active date. This determines which time period is shown and which date is\n   * highlighted when using keyboard navigation.\n   */\n  get activeDate(): D { return this._clampedActiveDate; }\n  set activeDate(value: D) {\n    this._clampedActiveDate = this._dateAdapter.clampDate(value, this.minDate, this.maxDate);\n    this.stateChanges.next();\n    this._changeDetectorRef.markForCheck();\n  }\n  private _clampedActiveDate: D;\n\n  /** Whether the calendar is in month view. */\n  get currentView(): MatCalendarView { return this._currentView; }\n  set currentView(value: MatCalendarView) {\n    const viewChangedResult = this._currentView !== value ? value : null;\n    this._currentView = value;\n    this._moveFocusOnNextTick = true;\n    this._changeDetectorRef.markForCheck();\n    if (viewChangedResult) {\n      this.viewChanged.emit(viewChangedResult);\n    }\n  }\n  private _currentView: MatCalendarView;\n\n  /**\n   * Emits whenever there is a state change that the header may need to respond to.\n   */\n  readonly stateChanges = new Subject<void>();\n\n  constructor(_intl: MatDatepickerIntl,\n              @Optional() private _dateAdapter: DateAdapter<D>,\n              @Optional() @Inject(MAT_DATE_FORMATS) private _dateFormats: MatDateFormats,\n              private _changeDetectorRef: ChangeDetectorRef) {\n\n    if (typeof ngDevMode === 'undefined' || ngDevMode) {\n      if (!this._dateAdapter) {\n        throw createMissingDateImplError('DateAdapter');\n      }\n\n      if (!this._dateFormats) {\n        throw createMissingDateImplError('MAT_DATE_FORMATS');\n      }\n    }\n\n    this._intlChanges = _intl.changes.subscribe(() => {\n      _changeDetectorRef.markForCheck();\n      this.stateChanges.next();\n    });\n  }\n\n  ngAfterContentInit() {\n    this._calendarHeaderPortal = new ComponentPortal(this.headerComponent || MatCalendarHeader);\n    this.activeDate = this.startAt || this._dateAdapter.today();\n\n    // Assign to the private property since we don't want to move focus on init.\n    this._currentView = this.startView;\n  }\n\n  ngAfterViewChecked() {\n    if (this._moveFocusOnNextTick) {\n      this._moveFocusOnNextTick = false;\n      this.focusActiveCell();\n    }\n  }\n\n  ngOnDestroy() {\n    this._intlChanges.unsubscribe();\n    this.stateChanges.complete();\n  }\n\n  ngOnChanges(changes: SimpleChanges) {\n    const change =\n        changes['minDate'] || changes['maxDate'] || changes['dateFilter'];\n\n    if (change && !change.firstChange) {\n      const view = this._getCurrentViewComponent();\n\n      if (view) {\n        // We need to `detectChanges` manually here, because the `minDate`, `maxDate` etc. are\n        // passed down to the view via data bindings which won't be up-to-date when we call `_init`.\n        this._changeDetectorRef.detectChanges();\n        view._init();\n      }\n    }\n\n    this.stateChanges.next();\n  }\n\n  /** Focuses the active date. */\n  focusActiveCell() {\n    this._getCurrentViewComponent()._focusActiveCell(false);\n  }\n\n  /** Updates today's date after an update of the active date */\n  updateTodaysDate() {\n    this._getCurrentViewComponent()._init();\n  }\n\n  /** Handles date selection in the month view. */\n  _dateSelected(event: MatCalendarUserEvent<D | null>): void {\n    const date = event.value;\n\n    if (this.selected instanceof DateRange ||\n        (date && !this._dateAdapter.sameDate(date, this.selected))) {\n      this.selectedChange.emit(date);\n    }\n\n    this._userSelection.emit(event);\n  }\n\n  /** Handles year selection in the multiyear view. */\n  _yearSelectedInMultiYearView(normalizedYear: D) {\n    this.yearSelected.emit(normalizedYear);\n  }\n\n  /** Handles month selection in the year view. */\n  _monthSelectedInYearView(normalizedMonth: D) {\n    this.monthSelected.emit(normalizedMonth);\n  }\n\n  /** Handles year/month selection in the multi-year/year views. */\n  _goToDateInView(date: D, view: 'month' | 'year' | 'multi-year'): void {\n    this.activeDate = date;\n    this.currentView = view;\n  }\n\n  /** Returns the component instance that corresponds to the current calendar view. */\n  private _getCurrentViewComponent(): MatMonthView<D> | MatYearView<D> | MatMultiYearView<D> {\n    // The return type is explicitly written as a union to ensure that the Closure compiler does\n    // not optimize calls to _init(). Without the explict return type, TypeScript narrows it to\n    // only the first component type. See https://github.com/angular/components/issues/22996.\n    return this.monthView || this.yearView || this.multiYearView;\n  }\n}\n"]} |
---|