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 { Directionality } from '@angular/cdk/bidi';
|
---|
9 | import { coerceBooleanProperty, coerceStringArray } from '@angular/cdk/coercion';
|
---|
10 | import { ESCAPE, hasModifierKey, UP_ARROW } from '@angular/cdk/keycodes';
|
---|
11 | import { Overlay, OverlayConfig, FlexibleConnectedPositionStrategy, } from '@angular/cdk/overlay';
|
---|
12 | import { ComponentPortal } from '@angular/cdk/portal';
|
---|
13 | import { DOCUMENT } from '@angular/common';
|
---|
14 | import { ChangeDetectionStrategy, Component, ElementRef, EventEmitter, Inject, InjectionToken, Input, NgZone, Optional, Output, ViewChild, ViewContainerRef, ViewEncapsulation, ChangeDetectorRef, Directive, } from '@angular/core';
|
---|
15 | import { DateAdapter, mixinColor, } from '@angular/material/core';
|
---|
16 | import { merge, Subject, Subscription } from 'rxjs';
|
---|
17 | import { filter, take } from 'rxjs/operators';
|
---|
18 | import { _getFocusedElementPierceShadowDom } from '@angular/cdk/platform';
|
---|
19 | import { MatCalendar } from './calendar';
|
---|
20 | import { matDatepickerAnimations } from './datepicker-animations';
|
---|
21 | import { createMissingDateImplError } from './datepicker-errors';
|
---|
22 | import { MatDateSelectionModel, DateRange, } from './date-selection-model';
|
---|
23 | import { MAT_DATE_RANGE_SELECTION_STRATEGY, } from './date-range-selection-strategy';
|
---|
24 | import { MatDatepickerIntl } from './datepicker-intl';
|
---|
25 | /** Used to generate a unique ID for each datepicker instance. */
|
---|
26 | let datepickerUid = 0;
|
---|
27 | /** Injection token that determines the scroll handling while the calendar is open. */
|
---|
28 | export const MAT_DATEPICKER_SCROLL_STRATEGY = new InjectionToken('mat-datepicker-scroll-strategy');
|
---|
29 | /** @docs-private */
|
---|
30 | export function MAT_DATEPICKER_SCROLL_STRATEGY_FACTORY(overlay) {
|
---|
31 | return () => overlay.scrollStrategies.reposition();
|
---|
32 | }
|
---|
33 | /** @docs-private */
|
---|
34 | export const MAT_DATEPICKER_SCROLL_STRATEGY_FACTORY_PROVIDER = {
|
---|
35 | provide: MAT_DATEPICKER_SCROLL_STRATEGY,
|
---|
36 | deps: [Overlay],
|
---|
37 | useFactory: MAT_DATEPICKER_SCROLL_STRATEGY_FACTORY,
|
---|
38 | };
|
---|
39 | // Boilerplate for applying mixins to MatDatepickerContent.
|
---|
40 | /** @docs-private */
|
---|
41 | const _MatDatepickerContentBase = mixinColor(class {
|
---|
42 | constructor(_elementRef) {
|
---|
43 | this._elementRef = _elementRef;
|
---|
44 | }
|
---|
45 | });
|
---|
46 | /**
|
---|
47 | * Component used as the content for the datepicker overlay. We use this instead of using
|
---|
48 | * MatCalendar directly as the content so we can control the initial focus. This also gives us a
|
---|
49 | * place to put additional features of the overlay that are not part of the calendar itself in the
|
---|
50 | * future. (e.g. confirmation buttons).
|
---|
51 | * @docs-private
|
---|
52 | */
|
---|
53 | export class MatDatepickerContent extends _MatDatepickerContentBase {
|
---|
54 | constructor(elementRef, _changeDetectorRef, _globalModel, _dateAdapter, _rangeSelectionStrategy, intl) {
|
---|
55 | super(elementRef);
|
---|
56 | this._changeDetectorRef = _changeDetectorRef;
|
---|
57 | this._globalModel = _globalModel;
|
---|
58 | this._dateAdapter = _dateAdapter;
|
---|
59 | this._rangeSelectionStrategy = _rangeSelectionStrategy;
|
---|
60 | this._subscriptions = new Subscription();
|
---|
61 | /** Emits when an animation has finished. */
|
---|
62 | this._animationDone = new Subject();
|
---|
63 | /** Portal with projected action buttons. */
|
---|
64 | this._actionsPortal = null;
|
---|
65 | this._closeButtonText = intl.closeCalendarLabel;
|
---|
66 | }
|
---|
67 | ngOnInit() {
|
---|
68 | // If we have actions, clone the model so that we have the ability to cancel the selection,
|
---|
69 | // otherwise update the global model directly. Note that we want to assign this as soon as
|
---|
70 | // possible, but `_actionsPortal` isn't available in the constructor so we do it in `ngOnInit`.
|
---|
71 | this._model = this._actionsPortal ? this._globalModel.clone() : this._globalModel;
|
---|
72 | this._animationState = this.datepicker.touchUi ? 'enter-dialog' : 'enter-dropdown';
|
---|
73 | }
|
---|
74 | ngAfterViewInit() {
|
---|
75 | this._subscriptions.add(this.datepicker.stateChanges.subscribe(() => {
|
---|
76 | this._changeDetectorRef.markForCheck();
|
---|
77 | }));
|
---|
78 | this._calendar.focusActiveCell();
|
---|
79 | }
|
---|
80 | ngOnDestroy() {
|
---|
81 | this._subscriptions.unsubscribe();
|
---|
82 | this._animationDone.complete();
|
---|
83 | }
|
---|
84 | _handleUserSelection(event) {
|
---|
85 | const selection = this._model.selection;
|
---|
86 | const value = event.value;
|
---|
87 | const isRange = selection instanceof DateRange;
|
---|
88 | // If we're selecting a range and we have a selection strategy, always pass the value through
|
---|
89 | // there. Otherwise don't assign null values to the model, unless we're selecting a range.
|
---|
90 | // A null value when picking a range means that the user cancelled the selection (e.g. by
|
---|
91 | // pressing escape), whereas when selecting a single value it means that the value didn't
|
---|
92 | // change. This isn't very intuitive, but it's here for backwards-compatibility.
|
---|
93 | if (isRange && this._rangeSelectionStrategy) {
|
---|
94 | const newSelection = this._rangeSelectionStrategy.selectionFinished(value, selection, event.event);
|
---|
95 | this._model.updateSelection(newSelection, this);
|
---|
96 | }
|
---|
97 | else if (value && (isRange ||
|
---|
98 | !this._dateAdapter.sameDate(value, selection))) {
|
---|
99 | this._model.add(value);
|
---|
100 | }
|
---|
101 | // Delegate closing the overlay to the actions.
|
---|
102 | if ((!this._model || this._model.isComplete()) && !this._actionsPortal) {
|
---|
103 | this.datepicker.close();
|
---|
104 | }
|
---|
105 | }
|
---|
106 | _startExitAnimation() {
|
---|
107 | this._animationState = 'void';
|
---|
108 | this._changeDetectorRef.markForCheck();
|
---|
109 | }
|
---|
110 | _getSelected() {
|
---|
111 | return this._model.selection;
|
---|
112 | }
|
---|
113 | /** Applies the current pending selection to the global model. */
|
---|
114 | _applyPendingSelection() {
|
---|
115 | if (this._model !== this._globalModel) {
|
---|
116 | this._globalModel.updateSelection(this._model.selection, this);
|
---|
117 | }
|
---|
118 | }
|
---|
119 | }
|
---|
120 | MatDatepickerContent.decorators = [
|
---|
121 | { type: Component, args: [{
|
---|
122 | selector: 'mat-datepicker-content',
|
---|
123 | template: "<div\n cdkTrapFocus\n class=\"mat-datepicker-content-container\"\n [class.mat-datepicker-content-container-with-actions]=\"_actionsPortal\">\n <mat-calendar\n [id]=\"datepicker.id\"\n [ngClass]=\"datepicker.panelClass\"\n [startAt]=\"datepicker.startAt\"\n [startView]=\"datepicker.startView\"\n [minDate]=\"datepicker._getMinDate()\"\n [maxDate]=\"datepicker._getMaxDate()\"\n [dateFilter]=\"datepicker._getDateFilter()\"\n [headerComponent]=\"datepicker.calendarHeaderComponent\"\n [selected]=\"_getSelected()\"\n [dateClass]=\"datepicker.dateClass\"\n [comparisonStart]=\"comparisonStart\"\n [comparisonEnd]=\"comparisonEnd\"\n [@fadeInCalendar]=\"'enter'\"\n (yearSelected)=\"datepicker._selectYear($event)\"\n (monthSelected)=\"datepicker._selectMonth($event)\"\n (viewChanged)=\"datepicker._viewChanged($event)\"\n (_userSelection)=\"_handleUserSelection($event)\"></mat-calendar>\n\n <ng-template [cdkPortalOutlet]=\"_actionsPortal\"></ng-template>\n\n <!-- Invisible close button for screen reader users. -->\n <button\n type=\"button\"\n mat-raised-button\n [color]=\"color || 'primary'\"\n class=\"mat-datepicker-close-button\"\n [class.cdk-visually-hidden]=\"!_closeButtonFocused\"\n (focus)=\"_closeButtonFocused = true\"\n (blur)=\"_closeButtonFocused = false\"\n (click)=\"datepicker.close()\">{{ _closeButtonText }}</button>\n</div>\n",
|
---|
124 | host: {
|
---|
125 | 'class': 'mat-datepicker-content',
|
---|
126 | '[@transformPanel]': '_animationState',
|
---|
127 | '(@transformPanel.done)': '_animationDone.next()',
|
---|
128 | '[class.mat-datepicker-content-touch]': 'datepicker.touchUi',
|
---|
129 | },
|
---|
130 | animations: [
|
---|
131 | matDatepickerAnimations.transformPanel,
|
---|
132 | matDatepickerAnimations.fadeInCalendar,
|
---|
133 | ],
|
---|
134 | exportAs: 'matDatepickerContent',
|
---|
135 | encapsulation: ViewEncapsulation.None,
|
---|
136 | changeDetection: ChangeDetectionStrategy.OnPush,
|
---|
137 | inputs: ['color'],
|
---|
138 | styles: [".mat-datepicker-content{display:block;border-radius:4px}.mat-datepicker-content .mat-calendar{width:296px;height:354px}.mat-datepicker-content .mat-datepicker-close-button{position:absolute;top:100%;left:0;margin-top:8px}.ng-animating .mat-datepicker-content .mat-datepicker-close-button{display:none}.mat-datepicker-content-container{display:flex;flex-direction:column;justify-content:space-between}.mat-datepicker-content-touch{display:block;max-height:80vh;position:relative;overflow:visible}.mat-datepicker-content-touch .mat-datepicker-content-container{min-height:312px;max-height:788px;min-width:250px;max-width:750px}.mat-datepicker-content-touch .mat-calendar{width:100%;height:auto}@media all and (orientation: landscape){.mat-datepicker-content-touch .mat-datepicker-content-container{width:64vh;height:80vh}}@media all and (orientation: portrait){.mat-datepicker-content-touch .mat-datepicker-content-container{width:80vw;height:100vw}.mat-datepicker-content-touch .mat-datepicker-content-container-with-actions{height:115vw}}\n"]
|
---|
139 | },] }
|
---|
140 | ];
|
---|
141 | MatDatepickerContent.ctorParameters = () => [
|
---|
142 | { type: ElementRef },
|
---|
143 | { type: ChangeDetectorRef },
|
---|
144 | { type: MatDateSelectionModel },
|
---|
145 | { type: DateAdapter },
|
---|
146 | { type: undefined, decorators: [{ type: Optional }, { type: Inject, args: [MAT_DATE_RANGE_SELECTION_STRATEGY,] }] },
|
---|
147 | { type: MatDatepickerIntl }
|
---|
148 | ];
|
---|
149 | MatDatepickerContent.propDecorators = {
|
---|
150 | _calendar: [{ type: ViewChild, args: [MatCalendar,] }]
|
---|
151 | };
|
---|
152 | /** Base class for a datepicker. */
|
---|
153 | export class MatDatepickerBase {
|
---|
154 | constructor(
|
---|
155 | /**
|
---|
156 | * @deprecated `_dialog` parameter is no longer being used and it will be removed.
|
---|
157 | * @breaking-change 13.0.0
|
---|
158 | */
|
---|
159 | _dialog, _overlay, _ngZone, _viewContainerRef, scrollStrategy, _dateAdapter, _dir,
|
---|
160 | /**
|
---|
161 | * @deprecated No longer being used. To be removed.
|
---|
162 | * @breaking-change 13.0.0
|
---|
163 | */
|
---|
164 | _document, _model) {
|
---|
165 | this._overlay = _overlay;
|
---|
166 | this._ngZone = _ngZone;
|
---|
167 | this._viewContainerRef = _viewContainerRef;
|
---|
168 | this._dateAdapter = _dateAdapter;
|
---|
169 | this._dir = _dir;
|
---|
170 | this._model = _model;
|
---|
171 | this._inputStateChanges = Subscription.EMPTY;
|
---|
172 | /** The view that the calendar should start in. */
|
---|
173 | this.startView = 'month';
|
---|
174 | this._touchUi = false;
|
---|
175 | /** Preferred position of the datepicker in the X axis. */
|
---|
176 | this.xPosition = 'start';
|
---|
177 | /** Preferred position of the datepicker in the Y axis. */
|
---|
178 | this.yPosition = 'below';
|
---|
179 | this._restoreFocus = true;
|
---|
180 | /**
|
---|
181 | * Emits selected year in multiyear view.
|
---|
182 | * This doesn't imply a change on the selected date.
|
---|
183 | */
|
---|
184 | this.yearSelected = new EventEmitter();
|
---|
185 | /**
|
---|
186 | * Emits selected month in year view.
|
---|
187 | * This doesn't imply a change on the selected date.
|
---|
188 | */
|
---|
189 | this.monthSelected = new EventEmitter();
|
---|
190 | /**
|
---|
191 | * Emits when the current view changes.
|
---|
192 | */
|
---|
193 | this.viewChanged = new EventEmitter(true);
|
---|
194 | /** Emits when the datepicker has been opened. */
|
---|
195 | this.openedStream = new EventEmitter();
|
---|
196 | /** Emits when the datepicker has been closed. */
|
---|
197 | this.closedStream = new EventEmitter();
|
---|
198 | this._opened = false;
|
---|
199 | /** The id for the datepicker calendar. */
|
---|
200 | this.id = `mat-datepicker-${datepickerUid++}`;
|
---|
201 | /** The element that was focused before the datepicker was opened. */
|
---|
202 | this._focusedElementBeforeOpen = null;
|
---|
203 | /** Unique class that will be added to the backdrop so that the test harnesses can look it up. */
|
---|
204 | this._backdropHarnessClass = `${this.id}-backdrop`;
|
---|
205 | /** Emits when the datepicker's state changes. */
|
---|
206 | this.stateChanges = new Subject();
|
---|
207 | if (!this._dateAdapter && (typeof ngDevMode === 'undefined' || ngDevMode)) {
|
---|
208 | throw createMissingDateImplError('DateAdapter');
|
---|
209 | }
|
---|
210 | this._scrollStrategy = scrollStrategy;
|
---|
211 | }
|
---|
212 | /** The date to open the calendar to initially. */
|
---|
213 | get startAt() {
|
---|
214 | // If an explicit startAt is set we start there, otherwise we start at whatever the currently
|
---|
215 | // selected value is.
|
---|
216 | return this._startAt || (this.datepickerInput ? this.datepickerInput.getStartValue() : null);
|
---|
217 | }
|
---|
218 | set startAt(value) {
|
---|
219 | this._startAt = this._dateAdapter.getValidDateOrNull(this._dateAdapter.deserialize(value));
|
---|
220 | }
|
---|
221 | /** Color palette to use on the datepicker's calendar. */
|
---|
222 | get color() {
|
---|
223 | return this._color ||
|
---|
224 | (this.datepickerInput ? this.datepickerInput.getThemePalette() : undefined);
|
---|
225 | }
|
---|
226 | set color(value) {
|
---|
227 | this._color = value;
|
---|
228 | }
|
---|
229 | /**
|
---|
230 | * Whether the calendar UI is in touch mode. In touch mode the calendar opens in a dialog rather
|
---|
231 | * than a dropdown and elements have more padding to allow for bigger touch targets.
|
---|
232 | */
|
---|
233 | get touchUi() { return this._touchUi; }
|
---|
234 | set touchUi(value) {
|
---|
235 | this._touchUi = coerceBooleanProperty(value);
|
---|
236 | }
|
---|
237 | /** Whether the datepicker pop-up should be disabled. */
|
---|
238 | get disabled() {
|
---|
239 | return this._disabled === undefined && this.datepickerInput ?
|
---|
240 | this.datepickerInput.disabled : !!this._disabled;
|
---|
241 | }
|
---|
242 | set disabled(value) {
|
---|
243 | const newValue = coerceBooleanProperty(value);
|
---|
244 | if (newValue !== this._disabled) {
|
---|
245 | this._disabled = newValue;
|
---|
246 | this.stateChanges.next(undefined);
|
---|
247 | }
|
---|
248 | }
|
---|
249 | /**
|
---|
250 | * Whether to restore focus to the previously-focused element when the calendar is closed.
|
---|
251 | * Note that automatic focus restoration is an accessibility feature and it is recommended that
|
---|
252 | * you provide your own equivalent, if you decide to turn it off.
|
---|
253 | */
|
---|
254 | get restoreFocus() { return this._restoreFocus; }
|
---|
255 | set restoreFocus(value) {
|
---|
256 | this._restoreFocus = coerceBooleanProperty(value);
|
---|
257 | }
|
---|
258 | /**
|
---|
259 | * Classes to be passed to the date picker panel.
|
---|
260 | * Supports string and string array values, similar to `ngClass`.
|
---|
261 | */
|
---|
262 | get panelClass() { return this._panelClass; }
|
---|
263 | set panelClass(value) {
|
---|
264 | this._panelClass = coerceStringArray(value);
|
---|
265 | }
|
---|
266 | /** Whether the calendar is open. */
|
---|
267 | get opened() { return this._opened; }
|
---|
268 | set opened(value) {
|
---|
269 | coerceBooleanProperty(value) ? this.open() : this.close();
|
---|
270 | }
|
---|
271 | /** The minimum selectable date. */
|
---|
272 | _getMinDate() {
|
---|
273 | return this.datepickerInput && this.datepickerInput.min;
|
---|
274 | }
|
---|
275 | /** The maximum selectable date. */
|
---|
276 | _getMaxDate() {
|
---|
277 | return this.datepickerInput && this.datepickerInput.max;
|
---|
278 | }
|
---|
279 | _getDateFilter() {
|
---|
280 | return this.datepickerInput && this.datepickerInput.dateFilter;
|
---|
281 | }
|
---|
282 | ngOnChanges(changes) {
|
---|
283 | const positionChange = changes['xPosition'] || changes['yPosition'];
|
---|
284 | if (positionChange && !positionChange.firstChange && this._overlayRef) {
|
---|
285 | const positionStrategy = this._overlayRef.getConfig().positionStrategy;
|
---|
286 | if (positionStrategy instanceof FlexibleConnectedPositionStrategy) {
|
---|
287 | this._setConnectedPositions(positionStrategy);
|
---|
288 | if (this.opened) {
|
---|
289 | this._overlayRef.updatePosition();
|
---|
290 | }
|
---|
291 | }
|
---|
292 | }
|
---|
293 | this.stateChanges.next(undefined);
|
---|
294 | }
|
---|
295 | ngOnDestroy() {
|
---|
296 | this._destroyOverlay();
|
---|
297 | this.close();
|
---|
298 | this._inputStateChanges.unsubscribe();
|
---|
299 | this.stateChanges.complete();
|
---|
300 | }
|
---|
301 | /** Selects the given date */
|
---|
302 | select(date) {
|
---|
303 | this._model.add(date);
|
---|
304 | }
|
---|
305 | /** Emits the selected year in multiyear view */
|
---|
306 | _selectYear(normalizedYear) {
|
---|
307 | this.yearSelected.emit(normalizedYear);
|
---|
308 | }
|
---|
309 | /** Emits selected month in year view */
|
---|
310 | _selectMonth(normalizedMonth) {
|
---|
311 | this.monthSelected.emit(normalizedMonth);
|
---|
312 | }
|
---|
313 | /** Emits changed view */
|
---|
314 | _viewChanged(view) {
|
---|
315 | this.viewChanged.emit(view);
|
---|
316 | }
|
---|
317 | /**
|
---|
318 | * Register an input with this datepicker.
|
---|
319 | * @param input The datepicker input to register with this datepicker.
|
---|
320 | * @returns Selection model that the input should hook itself up to.
|
---|
321 | */
|
---|
322 | registerInput(input) {
|
---|
323 | if (this.datepickerInput && (typeof ngDevMode === 'undefined' || ngDevMode)) {
|
---|
324 | throw Error('A MatDatepicker can only be associated with a single input.');
|
---|
325 | }
|
---|
326 | this._inputStateChanges.unsubscribe();
|
---|
327 | this.datepickerInput = input;
|
---|
328 | this._inputStateChanges =
|
---|
329 | input.stateChanges.subscribe(() => this.stateChanges.next(undefined));
|
---|
330 | return this._model;
|
---|
331 | }
|
---|
332 | /**
|
---|
333 | * Registers a portal containing action buttons with the datepicker.
|
---|
334 | * @param portal Portal to be registered.
|
---|
335 | */
|
---|
336 | registerActions(portal) {
|
---|
337 | if (this._actionsPortal && (typeof ngDevMode === 'undefined' || ngDevMode)) {
|
---|
338 | throw Error('A MatDatepicker can only be associated with a single actions row.');
|
---|
339 | }
|
---|
340 | this._actionsPortal = portal;
|
---|
341 | }
|
---|
342 | /**
|
---|
343 | * Removes a portal containing action buttons from the datepicker.
|
---|
344 | * @param portal Portal to be removed.
|
---|
345 | */
|
---|
346 | removeActions(portal) {
|
---|
347 | if (portal === this._actionsPortal) {
|
---|
348 | this._actionsPortal = null;
|
---|
349 | }
|
---|
350 | }
|
---|
351 | /** Open the calendar. */
|
---|
352 | open() {
|
---|
353 | if (this._opened || this.disabled) {
|
---|
354 | return;
|
---|
355 | }
|
---|
356 | if (!this.datepickerInput && (typeof ngDevMode === 'undefined' || ngDevMode)) {
|
---|
357 | throw Error('Attempted to open an MatDatepicker with no associated input.');
|
---|
358 | }
|
---|
359 | this._focusedElementBeforeOpen = _getFocusedElementPierceShadowDom();
|
---|
360 | this._openOverlay();
|
---|
361 | this._opened = true;
|
---|
362 | this.openedStream.emit();
|
---|
363 | }
|
---|
364 | /** Close the calendar. */
|
---|
365 | close() {
|
---|
366 | if (!this._opened) {
|
---|
367 | return;
|
---|
368 | }
|
---|
369 | if (this._componentRef) {
|
---|
370 | const instance = this._componentRef.instance;
|
---|
371 | instance._startExitAnimation();
|
---|
372 | instance._animationDone.pipe(take(1)).subscribe(() => this._destroyOverlay());
|
---|
373 | }
|
---|
374 | const completeClose = () => {
|
---|
375 | // The `_opened` could've been reset already if
|
---|
376 | // we got two events in quick succession.
|
---|
377 | if (this._opened) {
|
---|
378 | this._opened = false;
|
---|
379 | this.closedStream.emit();
|
---|
380 | this._focusedElementBeforeOpen = null;
|
---|
381 | }
|
---|
382 | };
|
---|
383 | if (this._restoreFocus && this._focusedElementBeforeOpen &&
|
---|
384 | typeof this._focusedElementBeforeOpen.focus === 'function') {
|
---|
385 | // Because IE moves focus asynchronously, we can't count on it being restored before we've
|
---|
386 | // marked the datepicker as closed. If the event fires out of sequence and the element that
|
---|
387 | // we're refocusing opens the datepicker on focus, the user could be stuck with not being
|
---|
388 | // able to close the calendar at all. We work around it by making the logic, that marks
|
---|
389 | // the datepicker as closed, async as well.
|
---|
390 | this._focusedElementBeforeOpen.focus();
|
---|
391 | setTimeout(completeClose);
|
---|
392 | }
|
---|
393 | else {
|
---|
394 | completeClose();
|
---|
395 | }
|
---|
396 | }
|
---|
397 | /** Applies the current pending selection on the overlay to the model. */
|
---|
398 | _applyPendingSelection() {
|
---|
399 | var _a, _b;
|
---|
400 | (_b = (_a = this._componentRef) === null || _a === void 0 ? void 0 : _a.instance) === null || _b === void 0 ? void 0 : _b._applyPendingSelection();
|
---|
401 | }
|
---|
402 | /** Forwards relevant values from the datepicker to the datepicker content inside the overlay. */
|
---|
403 | _forwardContentValues(instance) {
|
---|
404 | instance.datepicker = this;
|
---|
405 | instance.color = this.color;
|
---|
406 | instance._actionsPortal = this._actionsPortal;
|
---|
407 | }
|
---|
408 | /** Opens the overlay with the calendar. */
|
---|
409 | _openOverlay() {
|
---|
410 | this._destroyOverlay();
|
---|
411 | const isDialog = this.touchUi;
|
---|
412 | const labelId = this.datepickerInput.getOverlayLabelId();
|
---|
413 | const portal = new ComponentPortal(MatDatepickerContent, this._viewContainerRef);
|
---|
414 | const overlayRef = this._overlayRef = this._overlay.create(new OverlayConfig({
|
---|
415 | positionStrategy: isDialog ? this._getDialogStrategy() : this._getDropdownStrategy(),
|
---|
416 | hasBackdrop: true,
|
---|
417 | backdropClass: [
|
---|
418 | isDialog ? 'cdk-overlay-dark-backdrop' : 'mat-overlay-transparent-backdrop',
|
---|
419 | this._backdropHarnessClass
|
---|
420 | ],
|
---|
421 | direction: this._dir,
|
---|
422 | scrollStrategy: isDialog ? this._overlay.scrollStrategies.block() : this._scrollStrategy(),
|
---|
423 | panelClass: `mat-datepicker-${isDialog ? 'dialog' : 'popup'}`,
|
---|
424 | }));
|
---|
425 | const overlayElement = overlayRef.overlayElement;
|
---|
426 | overlayElement.setAttribute('role', 'dialog');
|
---|
427 | if (labelId) {
|
---|
428 | overlayElement.setAttribute('aria-labelledby', labelId);
|
---|
429 | }
|
---|
430 | if (isDialog) {
|
---|
431 | overlayElement.setAttribute('aria-modal', 'true');
|
---|
432 | }
|
---|
433 | this._getCloseStream(overlayRef).subscribe(event => {
|
---|
434 | if (event) {
|
---|
435 | event.preventDefault();
|
---|
436 | }
|
---|
437 | this.close();
|
---|
438 | });
|
---|
439 | this._componentRef = overlayRef.attach(portal);
|
---|
440 | this._forwardContentValues(this._componentRef.instance);
|
---|
441 | // Update the position once the calendar has rendered. Only relevant in dropdown mode.
|
---|
442 | if (!isDialog) {
|
---|
443 | this._ngZone.onStable.pipe(take(1)).subscribe(() => overlayRef.updatePosition());
|
---|
444 | }
|
---|
445 | }
|
---|
446 | /** Destroys the current overlay. */
|
---|
447 | _destroyOverlay() {
|
---|
448 | if (this._overlayRef) {
|
---|
449 | this._overlayRef.dispose();
|
---|
450 | this._overlayRef = this._componentRef = null;
|
---|
451 | }
|
---|
452 | }
|
---|
453 | /** Gets a position strategy that will open the calendar as a dropdown. */
|
---|
454 | _getDialogStrategy() {
|
---|
455 | return this._overlay.position().global().centerHorizontally().centerVertically();
|
---|
456 | }
|
---|
457 | /** Gets a position strategy that will open the calendar as a dropdown. */
|
---|
458 | _getDropdownStrategy() {
|
---|
459 | const strategy = this._overlay.position()
|
---|
460 | .flexibleConnectedTo(this.datepickerInput.getConnectedOverlayOrigin())
|
---|
461 | .withTransformOriginOn('.mat-datepicker-content')
|
---|
462 | .withFlexibleDimensions(false)
|
---|
463 | .withViewportMargin(8)
|
---|
464 | .withLockedPosition();
|
---|
465 | return this._setConnectedPositions(strategy);
|
---|
466 | }
|
---|
467 | /** Sets the positions of the datepicker in dropdown mode based on the current configuration. */
|
---|
468 | _setConnectedPositions(strategy) {
|
---|
469 | const primaryX = this.xPosition === 'end' ? 'end' : 'start';
|
---|
470 | const secondaryX = primaryX === 'start' ? 'end' : 'start';
|
---|
471 | const primaryY = this.yPosition === 'above' ? 'bottom' : 'top';
|
---|
472 | const secondaryY = primaryY === 'top' ? 'bottom' : 'top';
|
---|
473 | return strategy.withPositions([
|
---|
474 | {
|
---|
475 | originX: primaryX,
|
---|
476 | originY: secondaryY,
|
---|
477 | overlayX: primaryX,
|
---|
478 | overlayY: primaryY
|
---|
479 | },
|
---|
480 | {
|
---|
481 | originX: primaryX,
|
---|
482 | originY: primaryY,
|
---|
483 | overlayX: primaryX,
|
---|
484 | overlayY: secondaryY
|
---|
485 | },
|
---|
486 | {
|
---|
487 | originX: secondaryX,
|
---|
488 | originY: secondaryY,
|
---|
489 | overlayX: secondaryX,
|
---|
490 | overlayY: primaryY
|
---|
491 | },
|
---|
492 | {
|
---|
493 | originX: secondaryX,
|
---|
494 | originY: primaryY,
|
---|
495 | overlayX: secondaryX,
|
---|
496 | overlayY: secondaryY
|
---|
497 | }
|
---|
498 | ]);
|
---|
499 | }
|
---|
500 | /** Gets an observable that will emit when the overlay is supposed to be closed. */
|
---|
501 | _getCloseStream(overlayRef) {
|
---|
502 | return merge(overlayRef.backdropClick(), overlayRef.detachments(), overlayRef.keydownEvents().pipe(filter(event => {
|
---|
503 | // Closing on alt + up is only valid when there's an input associated with the datepicker.
|
---|
504 | return (event.keyCode === ESCAPE && !hasModifierKey(event)) || (this.datepickerInput &&
|
---|
505 | hasModifierKey(event, 'altKey') && event.keyCode === UP_ARROW);
|
---|
506 | })));
|
---|
507 | }
|
---|
508 | }
|
---|
509 | MatDatepickerBase.decorators = [
|
---|
510 | { type: Directive }
|
---|
511 | ];
|
---|
512 | MatDatepickerBase.ctorParameters = () => [
|
---|
513 | { type: undefined, decorators: [{ type: Inject, args: [ElementRef,] }] },
|
---|
514 | { type: Overlay },
|
---|
515 | { type: NgZone },
|
---|
516 | { type: ViewContainerRef },
|
---|
517 | { type: undefined, decorators: [{ type: Inject, args: [MAT_DATEPICKER_SCROLL_STRATEGY,] }] },
|
---|
518 | { type: DateAdapter, decorators: [{ type: Optional }] },
|
---|
519 | { type: Directionality, decorators: [{ type: Optional }] },
|
---|
520 | { type: undefined, decorators: [{ type: Optional }, { type: Inject, args: [DOCUMENT,] }] },
|
---|
521 | { type: MatDateSelectionModel }
|
---|
522 | ];
|
---|
523 | MatDatepickerBase.propDecorators = {
|
---|
524 | calendarHeaderComponent: [{ type: Input }],
|
---|
525 | startAt: [{ type: Input }],
|
---|
526 | startView: [{ type: Input }],
|
---|
527 | color: [{ type: Input }],
|
---|
528 | touchUi: [{ type: Input }],
|
---|
529 | disabled: [{ type: Input }],
|
---|
530 | xPosition: [{ type: Input }],
|
---|
531 | yPosition: [{ type: Input }],
|
---|
532 | restoreFocus: [{ type: Input }],
|
---|
533 | yearSelected: [{ type: Output }],
|
---|
534 | monthSelected: [{ type: Output }],
|
---|
535 | viewChanged: [{ type: Output }],
|
---|
536 | dateClass: [{ type: Input }],
|
---|
537 | openedStream: [{ type: Output, args: ['opened',] }],
|
---|
538 | closedStream: [{ type: Output, args: ['closed',] }],
|
---|
539 | panelClass: [{ type: Input }],
|
---|
540 | opened: [{ type: Input }]
|
---|
541 | };
|
---|
542 | //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"datepicker-base.js","sourceRoot":"","sources":["../../../../../../src/material/datepicker/datepicker-base.ts"],"names":[],"mappings":"AAAA;;;;;;GAMG;AAEH,OAAO,EAAC,cAAc,EAAC,MAAM,mBAAmB,CAAC;AACjD,OAAO,EAAe,qBAAqB,EAAE,iBAAiB,EAAC,MAAM,uBAAuB,CAAC;AAC7F,OAAO,EAAC,MAAM,EAAE,cAAc,EAAE,QAAQ,EAAC,MAAM,uBAAuB,CAAC;AACvE,OAAO,EACL,OAAO,EACP,aAAa,EAGb,iCAAiC,GAClC,MAAM,sBAAsB,CAAC;AAC9B,OAAO,EAAC,eAAe,EAAgC,MAAM,qBAAqB,CAAC;AACnF,OAAO,EAAC,QAAQ,EAAC,MAAM,iBAAiB,CAAC;AACzC,OAAO,EAEL,uBAAuB,EACvB,SAAS,EAET,UAAU,EACV,YAAY,EACZ,MAAM,EACN,cAAc,EACd,KAAK,EACL,MAAM,EAEN,QAAQ,EACR,MAAM,EACN,SAAS,EACT,gBAAgB,EAChB,iBAAiB,EACjB,iBAAiB,EACjB,SAAS,GAIV,MAAM,eAAe,CAAC;AACvB,OAAO,EAEL,WAAW,EACX,UAAU,GAEX,MAAM,wBAAwB,CAAC;AAChC,OAAO,EAAC,KAAK,EAAE,OAAO,EAAc,YAAY,EAAC,MAAM,MAAM,CAAC;AAC9D,OAAO,EAAC,MAAM,EAAE,IAAI,EAAC,MAAM,gBAAgB,CAAC;AAC5C,OAAO,EAAC,iCAAiC,EAAC,MAAM,uBAAuB,CAAC;AACxE,OAAO,EAAC,WAAW,EAAkB,MAAM,YAAY,CAAC;AACxD,OAAO,EAAC,uBAAuB,EAAC,MAAM,yBAAyB,CAAC;AAChE,OAAO,EAAC,0BAA0B,EAAC,MAAM,qBAAqB,CAAC;AAG/D,OAAO,EAEL,qBAAqB,EACrB,SAAS,GACV,MAAM,wBAAwB,CAAC;AAChC,OAAO,EACL,iCAAiC,GAElC,MAAM,iCAAiC,CAAC;AACzC,OAAO,EAAC,iBAAiB,EAAC,MAAM,mBAAmB,CAAC;AAEpD,iEAAiE;AACjE,IAAI,aAAa,GAAG,CAAC,CAAC;AAEtB,sFAAsF;AACtF,MAAM,CAAC,MAAM,8BAA8B,GACvC,IAAI,cAAc,CAAuB,gCAAgC,CAAC,CAAC;AAE/E,oBAAoB;AACpB,MAAM,UAAU,sCAAsC,CAAC,OAAgB;IACrE,OAAO,GAAG,EAAE,CAAC,OAAO,CAAC,gBAAgB,CAAC,UAAU,EAAE,CAAC;AACrD,CAAC;AAQD,oBAAoB;AACpB,MAAM,CAAC,MAAM,+CAA+C,GAAG;IAC7D,OAAO,EAAE,8BAA8B;IACvC,IAAI,EAAE,CAAC,OAAO,CAAC;IACf,UAAU,EAAE,sCAAsC;CACnD,CAAC;AAEF,2DAA2D;AAC3D,oBAAoB;AACpB,MAAM,yBAAyB,GAAG,UAAU,CAAC;IAC3C,YAAmB,WAAuB;QAAvB,gBAAW,GAAX,WAAW,CAAY;IAAG,CAAC;CAC/C,CAAC,CAAC;AAEH;;;;;;GAMG;AAoBH,MAAM,OAAO,oBACX,SAAQ,yBAAyB;IAkCjC,YACE,UAAsB,EACd,kBAAqC,EACrC,YAAyC,EACzC,YAA4B,EAExB,uBAAyD,EACrE,IAAuB;QACvB,KAAK,CAAC,UAAU,CAAC,CAAC;QANV,uBAAkB,GAAlB,kBAAkB,CAAmB;QACrC,iBAAY,GAAZ,YAAY,CAA6B;QACzC,iBAAY,GAAZ,YAAY,CAAgB;QAExB,4BAAuB,GAAvB,uBAAuB,CAAkC;QAvC/D,mBAAc,GAAG,IAAI,YAAY,EAAE,CAAC;QAqB5C,4CAA4C;QACnC,mBAAc,GAAG,IAAI,OAAO,EAAQ,CAAC;QAQ9C,4CAA4C;QAC5C,mBAAc,GAA0B,IAAI,CAAC;QAW3C,IAAI,CAAC,gBAAgB,GAAG,IAAI,CAAC,kBAAkB,CAAC;IAClD,CAAC;IAED,QAAQ;QACN,2FAA2F;QAC3F,0FAA0F;QAC1F,+FAA+F;QAC/F,IAAI,CAAC,MAAM,GAAG,IAAI,CAAC,cAAc,CAAC,CAAC,CAAC,IAAI,CAAC,YAAY,CAAC,KAAK,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,YAAY,CAAC;QAClF,IAAI,CAAC,eAAe,GAAG,IAAI,CAAC,UAAU,CAAC,OAAO,CAAC,CAAC,CAAC,cAAc,CAAC,CAAC,CAAC,gBAAgB,CAAC;IACrF,CAAC;IAED,eAAe;QACb,IAAI,CAAC,cAAc,CAAC,GAAG,CAAC,IAAI,CAAC,UAAU,CAAC,YAAY,CAAC,SAAS,CAAC,GAAG,EAAE;YAClE,IAAI,CAAC,kBAAkB,CAAC,YAAY,EAAE,CAAC;QACzC,CAAC,CAAC,CAAC,CAAC;QACJ,IAAI,CAAC,SAAS,CAAC,eAAe,EAAE,CAAC;IACnC,CAAC;IAED,WAAW;QACT,IAAI,CAAC,cAAc,CAAC,WAAW,EAAE,CAAC;QAClC,IAAI,CAAC,cAAc,CAAC,QAAQ,EAAE,CAAC;IACjC,CAAC;IAED,oBAAoB,CAAC,KAAqC;QACxD,MAAM,SAAS,GAAG,IAAI,CAAC,MAAM,CAAC,SAAS,CAAC;QACxC,MAAM,KAAK,GAAG,KAAK,CAAC,KAAK,CAAC;QAC1B,MAAM,OAAO,GAAG,SAAS,YAAY,SAAS,CAAC;QAE/C,6FAA6F;QAC7F,0FAA0F;QAC1F,yFAAyF;QACzF,yFAAyF;QACzF,gFAAgF;QAChF,IAAI,OAAO,IAAI,IAAI,CAAC,uBAAuB,EAAE;YAC3C,MAAM,YAAY,GAAG,IAAI,CAAC,uBAAuB,CAAC,iBAAiB,CAAC,KAAK,EACrE,SAAoC,EAAE,KAAK,CAAC,KAAK,CAAC,CAAC;YACvD,IAAI,CAAC,MAAM,CAAC,eAAe,CAAC,YAA4B,EAAE,IAAI,CAAC,CAAC;SACjE;aAAM,IAAI,KAAK,IAAI,CAAC,OAAO;YAClB,CAAC,IAAI,CAAC,YAAY,CAAC,QAAQ,CAAC,KAAK,EAAE,SAAyB,CAAC,CAAC,EAAE;YACxE,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC;SACxB;QAED,+CAA+C;QAC/C,IAAI,CAAC,CAAC,IAAI,CAAC,MAAM,IAAI,IAAI,CAAC,MAAM,CAAC,UAAU,EAAE,CAAC,IAAI,CAAC,IAAI,CAAC,cAAc,EAAE;YACtE,IAAI,CAAC,UAAU,CAAC,KAAK,EAAE,CAAC;SACzB;IACH,CAAC;IAED,mBAAmB;QACjB,IAAI,CAAC,eAAe,GAAG,MAAM,CAAC;QAC9B,IAAI,CAAC,kBAAkB,CAAC,YAAY,EAAE,CAAC;IACzC,CAAC;IAED,YAAY;QACV,OAAO,IAAI,CAAC,MAAM,CAAC,SAA+C,CAAC;IACrE,CAAC;IAED,iEAAiE;IACjE,sBAAsB;QACpB,IAAI,IAAI,CAAC,MAAM,KAAK,IAAI,CAAC,YAAY,EAAE;YACrC,IAAI,CAAC,YAAY,CAAC,eAAe,CAAC,IAAI,CAAC,MAAM,CAAC,SAAS,EAAE,IAAI,CAAC,CAAC;SAChE;IACH,CAAC;;;YA7HF,SAAS,SAAC;gBACT,QAAQ,EAAE,wBAAwB;gBAClC,m6CAAsC;gBAEtC,IAAI,EAAE;oBACJ,OAAO,EAAE,wBAAwB;oBACjC,mBAAmB,EAAE,iBAAiB;oBACtC,wBAAwB,EAAE,uBAAuB;oBACjD,sCAAsC,EAAE,oBAAoB;iBAC7D;gBACD,UAAU,EAAE;oBACV,uBAAuB,CAAC,cAAc;oBACtC,uBAAuB,CAAC,cAAc;iBACvC;gBACD,QAAQ,EAAE,sBAAsB;gBAChC,aAAa,EAAE,iBAAiB,CAAC,IAAI;gBACrC,eAAe,EAAE,uBAAuB,CAAC,MAAM;gBAC/C,MAAM,EAAE,CAAC,OAAO,CAAC;;aAClB;;;YAnGC,UAAU;YAYV,iBAAiB;YAsBjB,qBAAqB;YAdrB,WAAW;4CAwHR,QAAQ,YAAI,MAAM,SAAC,iCAAiC;YAnGjD,iBAAiB;;;wBAiEtB,SAAS,SAAC,WAAW;;AA6IxB,mCAAmC;AAEnC,MAAM,OAAgB,iBAAiB;IAsKrC;IACE;;;OAGG;IACiB,OAAY,EACxB,QAAiB,EACjB,OAAe,EACf,iBAAmC,EACH,cAAmB,EACvC,YAA4B,EAC5B,IAAoB;IACxC;;;OAGG;IAC2B,SAAc,EACpC,MAAmC;QAXnC,aAAQ,GAAR,QAAQ,CAAS;QACjB,YAAO,GAAP,OAAO,CAAQ;QACf,sBAAiB,GAAjB,iBAAiB,CAAkB;QAEvB,iBAAY,GAAZ,YAAY,CAAgB;QAC5B,SAAI,GAAJ,IAAI,CAAgB;QAMhC,WAAM,GAAN,MAAM,CAA6B;QAnLrC,uBAAkB,GAAG,YAAY,CAAC,KAAK,CAAC;QAiBhD,kDAAkD;QACzC,cAAS,GAAoC,OAAO,CAAC;QAsBtD,aAAQ,GAAG,KAAK,CAAC;QAkBzB,0DAA0D;QAE1D,cAAS,GAAgC,OAAO,CAAC;QAEjD,0DAA0D;QAE1D,cAAS,GAAgC,OAAO,CAAC;QAYzC,kBAAa,GAAG,IAAI,CAAC;QAE7B;;;WAGG;QACgB,iBAAY,GAAoB,IAAI,YAAY,EAAK,CAAC;QAEzE;;;WAGG;QACgB,kBAAa,GAAoB,IAAI,YAAY,EAAK,CAAC;QAE1E;;WAEG;QACgB,gBAAW,GAC5B,IAAI,YAAY,CAAkB,IAAI,CAAC,CAAC;QAK1C,iDAAiD;QACtB,iBAAY,GAAG,IAAI,YAAY,EAAQ,CAAC;QAEnE,iDAAiD;QACtB,iBAAY,GAAG,IAAI,YAAY,EAAQ,CAAC;QAmB3D,YAAO,GAAG,KAAK,CAAC;QAExB,0CAA0C;QAC1C,OAAE,GAAW,kBAAkB,aAAa,EAAE,EAAE,CAAC;QAsBjD,qEAAqE;QAC7D,8BAAyB,GAAuB,IAAI,CAAC;QAE7D,iGAAiG;QACzF,0BAAqB,GAAG,GAAG,IAAI,CAAC,EAAE,WAAW,CAAC;QAQtD,iDAAiD;QACxC,iBAAY,GAAG,IAAI,OAAO,EAAQ,CAAC;QAoB1C,IAAI,CAAC,IAAI,CAAC,YAAY,IAAI,CAAC,OAAO,SAAS,KAAK,WAAW,IAAI,SAAS,CAAC,EAAE;YACzE,MAAM,0BAA0B,CAAC,aAAa,CAAC,CAAC;SACjD;QAED,IAAI,CAAC,eAAe,GAAG,cAAc,CAAC;IACxC,CAAC;IApLD,kDAAkD;IAClD,IACI,OAAO;QACT,6FAA6F;QAC7F,qBAAqB;QACrB,OAAO,IAAI,CAAC,QAAQ,IAAI,CAAC,IAAI,CAAC,eAAe,CAAC,CAAC,CAAC,IAAI,CAAC,eAAe,CAAC,aAAa,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC;IAC/F,CAAC;IACD,IAAI,OAAO,CAAC,KAAe;QACzB,IAAI,CAAC,QAAQ,GAAG,IAAI,CAAC,YAAY,CAAC,kBAAkB,CAAC,IAAI,CAAC,YAAY,CAAC,WAAW,CAAC,KAAK,CAAC,CAAC,CAAC;IAC7F,CAAC;IAMD,yDAAyD;IACzD,IACI,KAAK;QACP,OAAO,IAAI,CAAC,MAAM;YACd,CAAC,IAAI,CAAC,eAAe,CAAC,CAAC,CAAC,IAAI,CAAC,eAAe,CAAC,eAAe,EAAE,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC;IAClF,CAAC;IACD,IAAI,KAAK,CAAC,KAAmB;QAC3B,IAAI,CAAC,MAAM,GAAG,KAAK,CAAC;IACtB,CAAC;IAGD;;;OAGG;IACH,IACI,OAAO,KAAc,OAAO,IAAI,CAAC,QAAQ,CAAC,CAAC,CAAC;IAChD,IAAI,OAAO,CAAC,KAAc;QACxB,IAAI,CAAC,QAAQ,GAAG,qBAAqB,CAAC,KAAK,CAAC,CAAC;IAC/C,CAAC;IAGD,wDAAwD;IACxD,IACI,QAAQ;QACV,OAAO,IAAI,CAAC,SAAS,KAAK,SAAS,IAAI,IAAI,CAAC,eAAe,CAAC,CAAC;YACzD,IAAI,CAAC,eAAe,CAAC,QAAQ,CAAC,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,SAAS,CAAC;IACvD,CAAC;IACD,IAAI,QAAQ,CAAC,KAAc;QACzB,MAAM,QAAQ,GAAG,qBAAqB,CAAC,KAAK,CAAC,CAAC;QAE9C,IAAI,QAAQ,KAAK,IAAI,CAAC,SAAS,EAAE;YAC/B,IAAI,CAAC,SAAS,GAAG,QAAQ,CAAC;YAC1B,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;SACnC;IACH,CAAC;IAWD;;;;OAIG;IACH,IACI,YAAY,KAAc,OAAO,IAAI,CAAC,aAAa,CAAC,CAAC,CAAC;IAC1D,IAAI,YAAY,CAAC,KAAc;QAC7B,IAAI,CAAC,aAAa,GAAG,qBAAqB,CAAC,KAAK,CAAC,CAAC;IACpD,CAAC;IA8BD;;;OAGG;IACH,IACI,UAAU,KAAwB,OAAO,IAAI,CAAC,WAAW,CAAC,CAAC,CAAC;IAChE,IAAI,UAAU,CAAC,KAAwB;QACrC,IAAI,CAAC,WAAW,GAAG,iBAAiB,CAAC,KAAK,CAAC,CAAC;IAC9C,CAAC;IAGD,oCAAoC;IACpC,IACI,MAAM,KAAc,OAAO,IAAI,CAAC,OAAO,CAAC,CAAC,CAAC;IAC9C,IAAI,MAAM,CAAC,KAAc;QACvB,qBAAqB,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,IAAI,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,KAAK,EAAE,CAAC;IAC5D,CAAC;IAMD,mCAAmC;IACnC,WAAW;QACT,OAAO,IAAI,CAAC,eAAe,IAAI,IAAI,CAAC,eAAe,CAAC,GAAG,CAAC;IAC1D,CAAC;IAED,mCAAmC;IACnC,WAAW;QACT,OAAO,IAAI,CAAC,eAAe,IAAI,IAAI,CAAC,eAAe,CAAC,GAAG,CAAC;IAC1D,CAAC;IAED,cAAc;QACZ,OAAO,IAAI,CAAC,eAAe,IAAI,IAAI,CAAC,eAAe,CAAC,UAAU,CAAC;IACjE,CAAC;IAgDD,WAAW,CAAC,OAAsB;QAChC,MAAM,cAAc,GAAG,OAAO,CAAC,WAAW,CAAC,IAAI,OAAO,CAAC,WAAW,CAAC,CAAC;QAEpE,IAAI,cAAc,IAAI,CAAC,cAAc,CAAC,WAAW,IAAI,IAAI,CAAC,WAAW,EAAE;YACrE,MAAM,gBAAgB,GAAG,IAAI,CAAC,WAAW,CAAC,SAAS,EAAE,CAAC,gBAAgB,CAAC;YAEvE,IAAI,gBAAgB,YAAY,iCAAiC,EAAE;gBACjE,IAAI,CAAC,sBAAsB,CAAC,gBAAgB,CAAC,CAAC;gBAE9C,IAAI,IAAI,CAAC,MAAM,EAAE;oBACf,IAAI,CAAC,WAAW,CAAC,cAAc,EAAE,CAAC;iBACnC;aACF;SACF;QAED,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;IACpC,CAAC;IAED,WAAW;QACT,IAAI,CAAC,eAAe,EAAE,CAAC;QACvB,IAAI,CAAC,KAAK,EAAE,CAAC;QACb,IAAI,CAAC,kBAAkB,CAAC,WAAW,EAAE,CAAC;QACtC,IAAI,CAAC,YAAY,CAAC,QAAQ,EAAE,CAAC;IAC/B,CAAC;IAED,6BAA6B;IAC7B,MAAM,CAAC,IAAO;QACZ,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;IACxB,CAAC;IAED,gDAAgD;IAChD,WAAW,CAAC,cAAiB;QAC3B,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC,cAAc,CAAC,CAAC;IACzC,CAAC;IAED,wCAAwC;IACxC,YAAY,CAAC,eAAkB;QAC7B,IAAI,CAAC,aAAa,CAAC,IAAI,CAAC,eAAe,CAAC,CAAC;IAC3C,CAAC;IAED,yBAAyB;IACzB,YAAY,CAAC,IAAqB;QAChC,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;IAC9B,CAAC;IAED;;;;OAIG;IACH,aAAa,CAAC,KAAQ;QACpB,IAAI,IAAI,CAAC,eAAe,IAAI,CAAC,OAAO,SAAS,KAAK,WAAW,IAAI,SAAS,CAAC,EAAE;YAC3E,MAAM,KAAK,CAAC,6DAA6D,CAAC,CAAC;SAC5E;QACD,IAAI,CAAC,kBAAkB,CAAC,WAAW,EAAE,CAAC;QACtC,IAAI,CAAC,eAAe,GAAG,KAAK,CAAC;QAC7B,IAAI,CAAC,kBAAkB;YACnB,KAAK,CAAC,YAAY,CAAC,SAAS,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC,CAAC;QAC1E,OAAO,IAAI,CAAC,MAAM,CAAC;IACrB,CAAC;IAED;;;OAGG;IACH,eAAe,CAAC,MAAsB;QACpC,IAAI,IAAI,CAAC,cAAc,IAAI,CAAC,OAAO,SAAS,KAAK,WAAW,IAAI,SAAS,CAAC,EAAE;YAC1E,MAAM,KAAK,CAAC,mEAAmE,CAAC,CAAC;SAClF;QACD,IAAI,CAAC,cAAc,GAAG,MAAM,CAAC;IAC/B,CAAC;IAED;;;OAGG;IACH,aAAa,CAAC,MAAsB;QAClC,IAAI,MAAM,KAAK,IAAI,CAAC,cAAc,EAAE;YAClC,IAAI,CAAC,cAAc,GAAG,IAAI,CAAC;SAC5B;IACH,CAAC;IAED,yBAAyB;IACzB,IAAI;QACF,IAAI,IAAI,CAAC,OAAO,IAAI,IAAI,CAAC,QAAQ,EAAE;YACjC,OAAO;SACR;QAED,IAAI,CAAC,IAAI,CAAC,eAAe,IAAI,CAAC,OAAO,SAAS,KAAK,WAAW,IAAI,SAAS,CAAC,EAAE;YAC5E,MAAM,KAAK,CAAC,8DAA8D,CAAC,CAAC;SAC7E;QAED,IAAI,CAAC,yBAAyB,GAAG,iCAAiC,EAAE,CAAC;QACrE,IAAI,CAAC,YAAY,EAAE,CAAC;QACpB,IAAI,CAAC,OAAO,GAAG,IAAI,CAAC;QACpB,IAAI,CAAC,YAAY,CAAC,IAAI,EAAE,CAAC;IAC3B,CAAC;IAED,0BAA0B;IAC1B,KAAK;QACH,IAAI,CAAC,IAAI,CAAC,OAAO,EAAE;YACjB,OAAO;SACR;QAED,IAAI,IAAI,CAAC,aAAa,EAAE;YACtB,MAAM,QAAQ,GAAG,IAAI,CAAC,aAAa,CAAC,QAAQ,CAAC;YAC7C,QAAQ,CAAC,mBAAmB,EAAE,CAAC;YAC/B,QAAQ,CAAC,cAAc,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,CAAC,SAAS,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,eAAe,EAAE,CAAC,CAAC;SAC/E;QAED,MAAM,aAAa,GAAG,GAAG,EAAE;YACzB,+CAA+C;YAC/C,yCAAyC;YACzC,IAAI,IAAI,CAAC,OAAO,EAAE;gBAChB,IAAI,CAAC,OAAO,GAAG,KAAK,CAAC;gBACrB,IAAI,CAAC,YAAY,CAAC,IAAI,EAAE,CAAC;gBACzB,IAAI,CAAC,yBAAyB,GAAG,IAAI,CAAC;aACvC;QACH,CAAC,CAAC;QAEF,IAAI,IAAI,CAAC,aAAa,IAAI,IAAI,CAAC,yBAAyB;YACtD,OAAO,IAAI,CAAC,yBAAyB,CAAC,KAAK,KAAK,UAAU,EAAE;YAC5D,0FAA0F;YAC1F,2FAA2F;YAC3F,yFAAyF;YACzF,uFAAuF;YACvF,2CAA2C;YAC3C,IAAI,CAAC,yBAAyB,CAAC,KAAK,EAAE,CAAC;YACvC,UAAU,CAAC,aAAa,CAAC,CAAC;SAC3B;aAAM;YACL,aAAa,EAAE,CAAC;SACjB;IACH,CAAC;IAED,yEAAyE;IACzE,sBAAsB;;QACpB,MAAA,MAAA,IAAI,CAAC,aAAa,0CAAE,QAAQ,0CAAE,sBAAsB,EAAE,CAAC;IACzD,CAAC;IAED,iGAAiG;IACvF,qBAAqB,CAAC,QAAoC;QAClE,QAAQ,CAAC,UAAU,GAAG,IAAI,CAAC;QAC3B,QAAQ,CAAC,KAAK,GAAG,IAAI,CAAC,KAAK,CAAC;QAC5B,QAAQ,CAAC,cAAc,GAAG,IAAI,CAAC,cAAc,CAAC;IAChD,CAAC;IAED,2CAA2C;IACnC,YAAY;QAClB,IAAI,CAAC,eAAe,EAAE,CAAC;QAEvB,MAAM,QAAQ,GAAG,IAAI,CAAC,OAAO,CAAC;QAC9B,MAAM,OAAO,GAAG,IAAI,CAAC,eAAe,CAAC,iBAAiB,EAAE,CAAC;QACzD,MAAM,MAAM,GAAG,IAAI,eAAe,CAA6B,oBAAoB,EACjF,IAAI,CAAC,iBAAiB,CAAC,CAAC;QAC1B,MAAM,UAAU,GAAG,IAAI,CAAC,WAAW,GAAG,IAAI,CAAC,QAAQ,CAAC,MAAM,CAAC,IAAI,aAAa,CAAC;YAC3E,gBAAgB,EAAE,QAAQ,CAAC,CAAC,CAAC,IAAI,CAAC,kBAAkB,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,oBAAoB,EAAE;YACpF,WAAW,EAAE,IAAI;YACjB,aAAa,EAAE;gBACb,QAAQ,CAAC,CAAC,CAAC,2BAA2B,CAAC,CAAC,CAAC,kCAAkC;gBAC3E,IAAI,CAAC,qBAAqB;aAC3B;YACD,SAAS,EAAE,IAAI,CAAC,IAAI;YACpB,cAAc,EAAE,QAAQ,CAAC,CAAC,CAAC,IAAI,CAAC,QAAQ,CAAC,gBAAgB,CAAC,KAAK,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,eAAe,EAAE;YAC1F,UAAU,EAAE,kBAAkB,QAAQ,CAAC,CAAC,CAAC,QAAQ,CAAC,CAAC,CAAC,OAAO,EAAE;SAC9D,CAAC,CAAC,CAAC;QACJ,MAAM,cAAc,GAAG,UAAU,CAAC,cAAc,CAAC;QACjD,cAAc,CAAC,YAAY,CAAC,MAAM,EAAE,QAAQ,CAAC,CAAC;QAE9C,IAAI,OAAO,EAAE;YACX,cAAc,CAAC,YAAY,CAAC,iBAAiB,EAAE,OAAO,CAAC,CAAC;SACzD;QAED,IAAI,QAAQ,EAAE;YACZ,cAAc,CAAC,YAAY,CAAC,YAAY,EAAE,MAAM,CAAC,CAAC;SACnD;QAED,IAAI,CAAC,eAAe,CAAC,UAAU,CAAC,CAAC,SAAS,CAAC,KAAK,CAAC,EAAE;YACjD,IAAI,KAAK,EAAE;gBACT,KAAK,CAAC,cAAc,EAAE,CAAC;aACxB;YACD,IAAI,CAAC,KAAK,EAAE,CAAC;QACf,CAAC,CAAC,CAAC;QAEH,IAAI,CAAC,aAAa,GAAG,UAAU,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC;QAC/C,IAAI,CAAC,qBAAqB,CAAC,IAAI,CAAC,aAAa,CAAC,QAAQ,CAAC,CAAC;QAExD,sFAAsF;QACtF,IAAI,CAAC,QAAQ,EAAE;YACb,IAAI,CAAC,OAAO,CAAC,QAAQ,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,CAAC,SAAS,CAAC,GAAG,EAAE,CAAC,UAAU,CAAC,cAAc,EAAE,CAAC,CAAC;SAClF;IACH,CAAC;IAED,oCAAoC;IAC5B,eAAe;QACrB,IAAI,IAAI,CAAC,WAAW,EAAE;YACpB,IAAI,CAAC,WAAW,CAAC,OAAO,EAAE,CAAC;YAC3B,IAAI,CAAC,WAAW,GAAG,IAAI,CAAC,aAAa,GAAG,IAAI,CAAC;SAC9C;IACH,CAAC;IAED,0EAA0E;IAClE,kBAAkB;QACxB,OAAO,IAAI,CAAC,QAAQ,CAAC,QAAQ,EAAE,CAAC,MAAM,EAAE,CAAC,kBAAkB,EAAE,CAAC,gBAAgB,EAAE,CAAC;IACnF,CAAC;IAED,0EAA0E;IAClE,oBAAoB;QAC1B,MAAM,QAAQ,GAAG,IAAI,CAAC,QAAQ,CAAC,QAAQ,EAAE;aACtC,mBAAmB,CAAC,IAAI,CAAC,eAAe,CAAC,yBAAyB,EAAE,CAAC;aACrE,qBAAqB,CAAC,yBAAyB,CAAC;aAChD,sBAAsB,CAAC,KAAK,CAAC;aAC7B,kBAAkB,CAAC,CAAC,CAAC;aACrB,kBAAkB,EAAE,CAAC;QAExB,OAAO,IAAI,CAAC,sBAAsB,CAAC,QAAQ,CAAC,CAAC;IAC/C,CAAC;IAED,gGAAgG;IACxF,sBAAsB,CAAC,QAA2C;QACxE,MAAM,QAAQ,GAAG,IAAI,CAAC,SAAS,KAAK,KAAK,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC,OAAO,CAAC;QAC5D,MAAM,UAAU,GAAG,QAAQ,KAAK,OAAO,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC,OAAO,CAAC;QAC1D,MAAM,QAAQ,GAAG,IAAI,CAAC,SAAS,KAAK,OAAO,CAAC,CAAC,CAAC,QAAQ,CAAC,CAAC,CAAC,KAAK,CAAC;QAC/D,MAAM,UAAU,GAAG,QAAQ,KAAK,KAAK,CAAC,CAAC,CAAC,QAAQ,CAAC,CAAC,CAAC,KAAK,CAAC;QAEzD,OAAO,QAAQ,CAAC,aAAa,CAAC;YAC5B;gBACE,OAAO,EAAE,QAAQ;gBACjB,OAAO,EAAE,UAAU;gBACnB,QAAQ,EAAE,QAAQ;gBAClB,QAAQ,EAAE,QAAQ;aACnB;YACD;gBACE,OAAO,EAAE,QAAQ;gBACjB,OAAO,EAAE,QAAQ;gBACjB,QAAQ,EAAE,QAAQ;gBAClB,QAAQ,EAAE,UAAU;aACrB;YACD;gBACE,OAAO,EAAE,UAAU;gBACnB,OAAO,EAAE,UAAU;gBACnB,QAAQ,EAAE,UAAU;gBACpB,QAAQ,EAAE,QAAQ;aACnB;YACD;gBACE,OAAO,EAAE,UAAU;gBACnB,OAAO,EAAE,QAAQ;gBACjB,QAAQ,EAAE,UAAU;gBACpB,QAAQ,EAAE,UAAU;aACrB;SACF,CAAC,CAAC;IACL,CAAC;IAED,mFAAmF;IAC3E,eAAe,CAAC,UAAsB;QAC5C,OAAO,KAAK,CACV,UAAU,CAAC,aAAa,EAAE,EAC1B,UAAU,CAAC,WAAW,EAAE,EACxB,UAAU,CAAC,aAAa,EAAE,CAAC,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,EAAE;YAC7C,0FAA0F;YAC1F,OAAO,CAAC,KAAK,CAAC,OAAO,KAAK,MAAM,IAAI,CAAC,cAAc,CAAC,KAAK,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,eAAe;gBAC5E,cAAc,CAAC,KAAK,EAAE,QAAQ,CAAC,IAAI,KAAK,CAAC,OAAO,KAAK,QAAQ,CAAC,CAAC;QACzE,CAAC,CAAC,CAAC,CACJ,CAAC;IACJ,CAAC;;;YAvcF,SAAS;;;4CA4KL,MAAM,SAAC,UAAU;YAjbpB,OAAO;YAkBP,MAAM;YAKN,gBAAgB;4CA8Zb,MAAM,SAAC,8BAA8B;YApZxC,WAAW,uBAqZR,QAAQ;YA1bL,cAAc,uBA2bjB,QAAQ;4CAKR,QAAQ,YAAI,MAAM,SAAC,QAAQ;YA7Y9B,qBAAqB;;;sCA8NpB,KAAK;sBAGL,KAAK;wBAYL,KAAK;oBAGL,KAAK;sBAcL,KAAK;uBAQL,KAAK;wBAgBL,KAAK;wBAIL,KAAK;2BAQL,KAAK;2BAWL,MAAM;4BAMN,MAAM;0BAKN,MAAM;wBAIN,KAAK;2BAGL,MAAM,SAAC,QAAQ;2BAGf,MAAM,SAAC,QAAQ;yBAMf,KAAK;qBAQL,KAAK","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 {Directionality} from '@angular/cdk/bidi';\nimport {BooleanInput, coerceBooleanProperty, coerceStringArray} from '@angular/cdk/coercion';\nimport {ESCAPE, hasModifierKey, UP_ARROW} from '@angular/cdk/keycodes';\nimport {\n  Overlay,\n  OverlayConfig,\n  OverlayRef,\n  ScrollStrategy,\n  FlexibleConnectedPositionStrategy,\n} from '@angular/cdk/overlay';\nimport {ComponentPortal, ComponentType, TemplatePortal} from '@angular/cdk/portal';\nimport {DOCUMENT} from '@angular/common';\nimport {\n  AfterViewInit,\n  ChangeDetectionStrategy,\n  Component,\n  ComponentRef,\n  ElementRef,\n  EventEmitter,\n  Inject,\n  InjectionToken,\n  Input,\n  NgZone,\n  OnDestroy,\n  Optional,\n  Output,\n  ViewChild,\n  ViewContainerRef,\n  ViewEncapsulation,\n  ChangeDetectorRef,\n  Directive,\n  OnChanges,\n  SimpleChanges,\n  OnInit,\n} from '@angular/core';\nimport {\n  CanColor,\n  DateAdapter,\n  mixinColor,\n  ThemePalette,\n} from '@angular/material/core';\nimport {merge, Subject, Observable, Subscription} from 'rxjs';\nimport {filter, take} from 'rxjs/operators';\nimport {_getFocusedElementPierceShadowDom} from '@angular/cdk/platform';\nimport {MatCalendar, MatCalendarView} from './calendar';\nimport {matDatepickerAnimations} from './datepicker-animations';\nimport {createMissingDateImplError} from './datepicker-errors';\nimport {MatCalendarUserEvent, MatCalendarCellClassFunction} from './calendar-body';\nimport {DateFilterFn} from './datepicker-input-base';\nimport {\n  ExtractDateTypeFromSelection,\n  MatDateSelectionModel,\n  DateRange,\n} from './date-selection-model';\nimport {\n  MAT_DATE_RANGE_SELECTION_STRATEGY,\n  MatDateRangeSelectionStrategy,\n} from './date-range-selection-strategy';\nimport {MatDatepickerIntl} from './datepicker-intl';\n\n/** Used to generate a unique ID for each datepicker instance. */\nlet datepickerUid = 0;\n\n/** Injection token that determines the scroll handling while the calendar is open. */\nexport const MAT_DATEPICKER_SCROLL_STRATEGY =\n    new InjectionToken<() => ScrollStrategy>('mat-datepicker-scroll-strategy');\n\n/** @docs-private */\nexport function MAT_DATEPICKER_SCROLL_STRATEGY_FACTORY(overlay: Overlay): () => ScrollStrategy {\n  return () => overlay.scrollStrategies.reposition();\n}\n\n/** Possible positions for the datepicker dropdown along the X axis. */\nexport type DatepickerDropdownPositionX = 'start' | 'end';\n\n/** Possible positions for the datepicker dropdown along the Y axis. */\nexport type DatepickerDropdownPositionY = 'above' | 'below';\n\n/** @docs-private */\nexport const MAT_DATEPICKER_SCROLL_STRATEGY_FACTORY_PROVIDER = {\n  provide: MAT_DATEPICKER_SCROLL_STRATEGY,\n  deps: [Overlay],\n  useFactory: MAT_DATEPICKER_SCROLL_STRATEGY_FACTORY,\n};\n\n// Boilerplate for applying mixins to MatDatepickerContent.\n/** @docs-private */\nconst _MatDatepickerContentBase = mixinColor(class {\n  constructor(public _elementRef: ElementRef) {}\n});\n\n/**\n * Component used as the content for the datepicker overlay. We use this instead of using\n * MatCalendar directly as the content so we can control the initial focus. This also gives us a\n * place to put additional features of the overlay that are not part of the calendar itself in the\n * future. (e.g. confirmation buttons).\n * @docs-private\n */\n@Component({\n  selector: 'mat-datepicker-content',\n  templateUrl: 'datepicker-content.html',\n  styleUrls: ['datepicker-content.css'],\n  host: {\n    'class': 'mat-datepicker-content',\n    '[@transformPanel]': '_animationState',\n    '(@transformPanel.done)': '_animationDone.next()',\n    '[class.mat-datepicker-content-touch]': 'datepicker.touchUi',\n  },\n  animations: [\n    matDatepickerAnimations.transformPanel,\n    matDatepickerAnimations.fadeInCalendar,\n  ],\n  exportAs: 'matDatepickerContent',\n  encapsulation: ViewEncapsulation.None,\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  inputs: ['color'],\n})\nexport class MatDatepickerContent<S, D = ExtractDateTypeFromSelection<S>>\n  extends _MatDatepickerContentBase implements OnInit, AfterViewInit, OnDestroy, CanColor {\n  private _subscriptions = new Subscription();\n  private _model: MatDateSelectionModel<S, D>;\n\n  /** Reference to the internal calendar component. */\n  @ViewChild(MatCalendar) _calendar: MatCalendar<D>;\n\n  /** Reference to the datepicker that created the overlay. */\n  datepicker: MatDatepickerBase<any, S, D>;\n\n  /** Start of the comparison range. */\n  comparisonStart: D | null;\n\n  /** End of the comparison range. */\n  comparisonEnd: D | null;\n\n  /** Whether the datepicker is above or below the input. */\n  _isAbove: boolean;\n\n  /** Current state of the animation. */\n  _animationState: 'enter-dropdown' | 'enter-dialog' | 'void';\n\n  /** Emits when an animation has finished. */\n  readonly _animationDone = new Subject<void>();\n\n  /** Text for the close button. */\n  _closeButtonText: string;\n\n  /** Whether the close button currently has focus. */\n  _closeButtonFocused: boolean;\n\n  /** Portal with projected action buttons. */\n  _actionsPortal: TemplatePortal | null = null;\n\n  constructor(\n    elementRef: ElementRef,\n    private _changeDetectorRef: ChangeDetectorRef,\n    private _globalModel: MatDateSelectionModel<S, D>,\n    private _dateAdapter: DateAdapter<D>,\n    @Optional() @Inject(MAT_DATE_RANGE_SELECTION_STRATEGY)\n        private _rangeSelectionStrategy: MatDateRangeSelectionStrategy<D>,\n    intl: MatDatepickerIntl) {\n    super(elementRef);\n    this._closeButtonText = intl.closeCalendarLabel;\n  }\n\n  ngOnInit() {\n    // If we have actions, clone the model so that we have the ability to cancel the selection,\n    // otherwise update the global model directly. Note that we want to assign this as soon as\n    // possible, but `_actionsPortal` isn't available in the constructor so we do it in `ngOnInit`.\n    this._model = this._actionsPortal ? this._globalModel.clone() : this._globalModel;\n    this._animationState = this.datepicker.touchUi ? 'enter-dialog' : 'enter-dropdown';\n  }\n\n  ngAfterViewInit() {\n    this._subscriptions.add(this.datepicker.stateChanges.subscribe(() => {\n      this._changeDetectorRef.markForCheck();\n    }));\n    this._calendar.focusActiveCell();\n  }\n\n  ngOnDestroy() {\n    this._subscriptions.unsubscribe();\n    this._animationDone.complete();\n  }\n\n  _handleUserSelection(event: MatCalendarUserEvent<D | null>) {\n    const selection = this._model.selection;\n    const value = event.value;\n    const isRange = selection instanceof DateRange;\n\n    // If we're selecting a range and we have a selection strategy, always pass the value through\n    // there. Otherwise don't assign null values to the model, unless we're selecting a range.\n    // A null value when picking a range means that the user cancelled the selection (e.g. by\n    // pressing escape), whereas when selecting a single value it means that the value didn't\n    // change. This isn't very intuitive, but it's here for backwards-compatibility.\n    if (isRange && this._rangeSelectionStrategy) {\n      const newSelection = this._rangeSelectionStrategy.selectionFinished(value,\n          selection as unknown as DateRange<D>, event.event);\n      this._model.updateSelection(newSelection as unknown as S, this);\n    } else if (value && (isRange ||\n              !this._dateAdapter.sameDate(value, selection as unknown as D))) {\n      this._model.add(value);\n    }\n\n    // Delegate closing the overlay to the actions.\n    if ((!this._model || this._model.isComplete()) && !this._actionsPortal) {\n      this.datepicker.close();\n    }\n  }\n\n  _startExitAnimation() {\n    this._animationState = 'void';\n    this._changeDetectorRef.markForCheck();\n  }\n\n  _getSelected() {\n    return this._model.selection as unknown as D | DateRange<D> | null;\n  }\n\n  /** Applies the current pending selection to the global model. */\n  _applyPendingSelection() {\n    if (this._model !== this._globalModel) {\n      this._globalModel.updateSelection(this._model.selection, this);\n    }\n  }\n}\n\n/** Form control that can be associated with a datepicker. */\nexport interface MatDatepickerControl<D> {\n  getStartValue(): D | null;\n  getThemePalette(): ThemePalette;\n  min: D | null;\n  max: D | null;\n  disabled: boolean;\n  dateFilter: DateFilterFn<D>;\n  getConnectedOverlayOrigin(): ElementRef;\n  getOverlayLabelId(): string | null;\n  stateChanges: Observable<void>;\n}\n\n/** A datepicker that can be attached to a {@link MatDatepickerControl}. */\nexport interface MatDatepickerPanel<C extends MatDatepickerControl<D>, S,\n    D = ExtractDateTypeFromSelection<S>> {\n  /** Stream that emits whenever the date picker is closed. */\n  closedStream: EventEmitter<void>;\n  /** Color palette to use on the datepicker's calendar. */\n  color: ThemePalette;\n  /** The input element the datepicker is associated with. */\n  datepickerInput: C;\n  /** Whether the datepicker pop-up should be disabled. */\n  disabled: boolean;\n  /** The id for the datepicker's calendar. */\n  id: string;\n  /** Whether the datepicker is open. */\n  opened: boolean;\n  /** Stream that emits whenever the date picker is opened. */\n  openedStream: EventEmitter<void>;\n  /** Emits when the datepicker's state changes. */\n  stateChanges: Subject<void>;\n  /** Opens the datepicker. */\n  open(): void;\n  /** Register an input with the datepicker. */\n  registerInput(input: C): MatDateSelectionModel<S, D>;\n}\n\n/** Base class for a datepicker. */\n@Directive()\nexport abstract class MatDatepickerBase<C extends MatDatepickerControl<D>, S,\n  D = ExtractDateTypeFromSelection<S>> implements MatDatepickerPanel<C, S, D>, OnDestroy,\n    OnChanges {\n  private _scrollStrategy: () => ScrollStrategy;\n  private _inputStateChanges = Subscription.EMPTY;\n\n  /** An input indicating the type of the custom header component for the calendar, if set. */\n  @Input() calendarHeaderComponent: ComponentType<any>;\n\n  /** The date to open the calendar to initially. */\n  @Input()\n  get startAt(): D | null {\n    // If an explicit startAt is set we start there, otherwise we start at whatever the currently\n    // selected value is.\n    return this._startAt || (this.datepickerInput ? this.datepickerInput.getStartValue() : null);\n  }\n  set startAt(value: D | null) {\n    this._startAt = this._dateAdapter.getValidDateOrNull(this._dateAdapter.deserialize(value));\n  }\n  private _startAt: D | null;\n\n  /** The view that the calendar should start in. */\n  @Input() startView: 'month' | 'year' | 'multi-year' = 'month';\n\n  /** Color palette to use on the datepicker's calendar. */\n  @Input()\n  get color(): ThemePalette {\n    return this._color ||\n        (this.datepickerInput ? this.datepickerInput.getThemePalette() : undefined);\n  }\n  set color(value: ThemePalette) {\n    this._color = value;\n  }\n  _color: ThemePalette;\n\n  /**\n   * Whether the calendar UI is in touch mode. In touch mode the calendar opens in a dialog rather\n   * than a dropdown and elements have more padding to allow for bigger touch targets.\n   */\n  @Input()\n  get touchUi(): boolean { return this._touchUi; }\n  set touchUi(value: boolean) {\n    this._touchUi = coerceBooleanProperty(value);\n  }\n  private _touchUi = false;\n\n  /** Whether the datepicker pop-up should be disabled. */\n  @Input()\n  get disabled(): boolean {\n    return this._disabled === undefined && this.datepickerInput ?\n        this.datepickerInput.disabled : !!this._disabled;\n  }\n  set disabled(value: boolean) {\n    const newValue = coerceBooleanProperty(value);\n\n    if (newValue !== this._disabled) {\n      this._disabled = newValue;\n      this.stateChanges.next(undefined);\n    }\n  }\n  private _disabled: boolean;\n\n  /** Preferred position of the datepicker in the X axis. */\n  @Input()\n  xPosition: DatepickerDropdownPositionX = 'start';\n\n  /** Preferred position of the datepicker in the Y axis. */\n  @Input()\n  yPosition: DatepickerDropdownPositionY = 'below';\n\n  /**\n   * Whether to restore focus to the previously-focused element when the calendar is closed.\n   * Note that automatic focus restoration is an accessibility feature and it is recommended that\n   * you provide your own equivalent, if you decide to turn it off.\n   */\n  @Input()\n  get restoreFocus(): boolean { return this._restoreFocus; }\n  set restoreFocus(value: boolean) {\n    this._restoreFocus = coerceBooleanProperty(value);\n  }\n  private _restoreFocus = true;\n\n  /**\n   * Emits selected year 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 selected month 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  /** Function that can be used to add custom CSS classes to dates. */\n  @Input() dateClass: MatCalendarCellClassFunction<D>;\n\n  /** Emits when the datepicker has been opened. */\n  @Output('opened') readonly openedStream = new EventEmitter<void>();\n\n  /** Emits when the datepicker has been closed. */\n  @Output('closed') readonly closedStream = new EventEmitter<void>();\n\n  /**\n   * Classes to be passed to the date picker panel.\n   * Supports string and string array values, similar to `ngClass`.\n   */\n  @Input()\n  get panelClass(): string | string[] { return this._panelClass; }\n  set panelClass(value: string | string[]) {\n    this._panelClass = coerceStringArray(value);\n  }\n  private _panelClass: string[];\n\n  /** Whether the calendar is open. */\n  @Input()\n  get opened(): boolean { return this._opened; }\n  set opened(value: boolean) {\n    coerceBooleanProperty(value) ? this.open() : this.close();\n  }\n  private _opened = false;\n\n  /** The id for the datepicker calendar. */\n  id: string = `mat-datepicker-${datepickerUid++}`;\n\n  /** The minimum selectable date. */\n  _getMinDate(): D | null {\n    return this.datepickerInput && this.datepickerInput.min;\n  }\n\n  /** The maximum selectable date. */\n  _getMaxDate(): D | null {\n    return this.datepickerInput && this.datepickerInput.max;\n  }\n\n  _getDateFilter(): DateFilterFn<D> {\n    return this.datepickerInput && this.datepickerInput.dateFilter;\n  }\n\n  /** A reference to the overlay into which we've rendered the calendar. */\n  private _overlayRef: OverlayRef | null;\n\n  /** Reference to the component instance rendered in the overlay. */\n  private _componentRef: ComponentRef<MatDatepickerContent<S, D>> | null;\n\n  /** The element that was focused before the datepicker was opened. */\n  private _focusedElementBeforeOpen: HTMLElement | null = null;\n\n  /** Unique class that will be added to the backdrop so that the test harnesses can look it up. */\n  private _backdropHarnessClass = `${this.id}-backdrop`;\n\n  /** Currently-registered actions portal. */\n  private _actionsPortal: TemplatePortal | null;\n\n  /** The input element this datepicker is associated with. */\n  datepickerInput: C;\n\n  /** Emits when the datepicker's state changes. */\n  readonly stateChanges = new Subject<void>();\n\n  constructor(\n    /**\n     * @deprecated `_dialog` parameter is no longer being used and it will be removed.\n     * @breaking-change 13.0.0\n     */\n    @Inject(ElementRef) _dialog: any,\n    private _overlay: Overlay,\n    private _ngZone: NgZone,\n    private _viewContainerRef: ViewContainerRef,\n    @Inject(MAT_DATEPICKER_SCROLL_STRATEGY) scrollStrategy: any,\n    @Optional() private _dateAdapter: DateAdapter<D>,\n    @Optional() private _dir: Directionality,\n    /**\n     * @deprecated No longer being used. To be removed.\n     * @breaking-change 13.0.0\n     */\n    @Optional() @Inject(DOCUMENT) _document: any,\n    private _model: MatDateSelectionModel<S, D>) {\n    if (!this._dateAdapter && (typeof ngDevMode === 'undefined' || ngDevMode)) {\n      throw createMissingDateImplError('DateAdapter');\n    }\n\n    this._scrollStrategy = scrollStrategy;\n  }\n\n  ngOnChanges(changes: SimpleChanges) {\n    const positionChange = changes['xPosition'] || changes['yPosition'];\n\n    if (positionChange && !positionChange.firstChange && this._overlayRef) {\n      const positionStrategy = this._overlayRef.getConfig().positionStrategy;\n\n      if (positionStrategy instanceof FlexibleConnectedPositionStrategy) {\n        this._setConnectedPositions(positionStrategy);\n\n        if (this.opened) {\n          this._overlayRef.updatePosition();\n        }\n      }\n    }\n\n    this.stateChanges.next(undefined);\n  }\n\n  ngOnDestroy() {\n    this._destroyOverlay();\n    this.close();\n    this._inputStateChanges.unsubscribe();\n    this.stateChanges.complete();\n  }\n\n  /** Selects the given date */\n  select(date: D): void {\n    this._model.add(date);\n  }\n\n  /** Emits the selected year in multiyear view */\n  _selectYear(normalizedYear: D): void {\n    this.yearSelected.emit(normalizedYear);\n  }\n\n  /** Emits selected month in year view */\n  _selectMonth(normalizedMonth: D): void {\n    this.monthSelected.emit(normalizedMonth);\n  }\n\n  /** Emits changed view */\n  _viewChanged(view: MatCalendarView): void {\n    this.viewChanged.emit(view);\n  }\n\n  /**\n   * Register an input with this datepicker.\n   * @param input The datepicker input to register with this datepicker.\n   * @returns Selection model that the input should hook itself up to.\n   */\n  registerInput(input: C): MatDateSelectionModel<S, D> {\n    if (this.datepickerInput && (typeof ngDevMode === 'undefined' || ngDevMode)) {\n      throw Error('A MatDatepicker can only be associated with a single input.');\n    }\n    this._inputStateChanges.unsubscribe();\n    this.datepickerInput = input;\n    this._inputStateChanges =\n        input.stateChanges.subscribe(() => this.stateChanges.next(undefined));\n    return this._model;\n  }\n\n  /**\n   * Registers a portal containing action buttons with the datepicker.\n   * @param portal Portal to be registered.\n   */\n  registerActions(portal: TemplatePortal): void {\n    if (this._actionsPortal && (typeof ngDevMode === 'undefined' || ngDevMode)) {\n      throw Error('A MatDatepicker can only be associated with a single actions row.');\n    }\n    this._actionsPortal = portal;\n  }\n\n  /**\n   * Removes a portal containing action buttons from the datepicker.\n   * @param portal Portal to be removed.\n   */\n  removeActions(portal: TemplatePortal): void {\n    if (portal === this._actionsPortal) {\n      this._actionsPortal = null;\n    }\n  }\n\n  /** Open the calendar. */\n  open(): void {\n    if (this._opened || this.disabled) {\n      return;\n    }\n\n    if (!this.datepickerInput && (typeof ngDevMode === 'undefined' || ngDevMode)) {\n      throw Error('Attempted to open an MatDatepicker with no associated input.');\n    }\n\n    this._focusedElementBeforeOpen = _getFocusedElementPierceShadowDom();\n    this._openOverlay();\n    this._opened = true;\n    this.openedStream.emit();\n  }\n\n  /** Close the calendar. */\n  close(): void {\n    if (!this._opened) {\n      return;\n    }\n\n    if (this._componentRef) {\n      const instance = this._componentRef.instance;\n      instance._startExitAnimation();\n      instance._animationDone.pipe(take(1)).subscribe(() => this._destroyOverlay());\n    }\n\n    const completeClose = () => {\n      // The `_opened` could've been reset already if\n      // we got two events in quick succession.\n      if (this._opened) {\n        this._opened = false;\n        this.closedStream.emit();\n        this._focusedElementBeforeOpen = null;\n      }\n    };\n\n    if (this._restoreFocus && this._focusedElementBeforeOpen &&\n      typeof this._focusedElementBeforeOpen.focus === 'function') {\n      // Because IE moves focus asynchronously, we can't count on it being restored before we've\n      // marked the datepicker as closed. If the event fires out of sequence and the element that\n      // we're refocusing opens the datepicker on focus, the user could be stuck with not being\n      // able to close the calendar at all. We work around it by making the logic, that marks\n      // the datepicker as closed, async as well.\n      this._focusedElementBeforeOpen.focus();\n      setTimeout(completeClose);\n    } else {\n      completeClose();\n    }\n  }\n\n  /** Applies the current pending selection on the overlay to the model. */\n  _applyPendingSelection() {\n    this._componentRef?.instance?._applyPendingSelection();\n  }\n\n  /** Forwards relevant values from the datepicker to the datepicker content inside the overlay. */\n  protected _forwardContentValues(instance: MatDatepickerContent<S, D>) {\n    instance.datepicker = this;\n    instance.color = this.color;\n    instance._actionsPortal = this._actionsPortal;\n  }\n\n  /** Opens the overlay with the calendar. */\n  private _openOverlay(): void {\n    this._destroyOverlay();\n\n    const isDialog = this.touchUi;\n    const labelId = this.datepickerInput.getOverlayLabelId();\n    const portal = new ComponentPortal<MatDatepickerContent<S, D>>(MatDatepickerContent,\n      this._viewContainerRef);\n    const overlayRef = this._overlayRef = this._overlay.create(new OverlayConfig({\n      positionStrategy: isDialog ? this._getDialogStrategy() : this._getDropdownStrategy(),\n      hasBackdrop: true,\n      backdropClass: [\n        isDialog ? 'cdk-overlay-dark-backdrop' : 'mat-overlay-transparent-backdrop',\n        this._backdropHarnessClass\n      ],\n      direction: this._dir,\n      scrollStrategy: isDialog ? this._overlay.scrollStrategies.block() : this._scrollStrategy(),\n      panelClass: `mat-datepicker-${isDialog ? 'dialog' : 'popup'}`,\n    }));\n    const overlayElement = overlayRef.overlayElement;\n    overlayElement.setAttribute('role', 'dialog');\n\n    if (labelId) {\n      overlayElement.setAttribute('aria-labelledby', labelId);\n    }\n\n    if (isDialog) {\n      overlayElement.setAttribute('aria-modal', 'true');\n    }\n\n    this._getCloseStream(overlayRef).subscribe(event => {\n      if (event) {\n        event.preventDefault();\n      }\n      this.close();\n    });\n\n    this._componentRef = overlayRef.attach(portal);\n    this._forwardContentValues(this._componentRef.instance);\n\n    // Update the position once the calendar has rendered. Only relevant in dropdown mode.\n    if (!isDialog) {\n      this._ngZone.onStable.pipe(take(1)).subscribe(() => overlayRef.updatePosition());\n    }\n  }\n\n  /** Destroys the current overlay. */\n  private _destroyOverlay() {\n    if (this._overlayRef) {\n      this._overlayRef.dispose();\n      this._overlayRef = this._componentRef = null;\n    }\n  }\n\n  /** Gets a position strategy that will open the calendar as a dropdown. */\n  private _getDialogStrategy() {\n    return this._overlay.position().global().centerHorizontally().centerVertically();\n  }\n\n  /** Gets a position strategy that will open the calendar as a dropdown. */\n  private _getDropdownStrategy() {\n    const strategy = this._overlay.position()\n      .flexibleConnectedTo(this.datepickerInput.getConnectedOverlayOrigin())\n      .withTransformOriginOn('.mat-datepicker-content')\n      .withFlexibleDimensions(false)\n      .withViewportMargin(8)\n      .withLockedPosition();\n\n    return this._setConnectedPositions(strategy);\n  }\n\n  /** Sets the positions of the datepicker in dropdown mode based on the current configuration. */\n  private _setConnectedPositions(strategy: FlexibleConnectedPositionStrategy) {\n    const primaryX = this.xPosition === 'end' ? 'end' : 'start';\n    const secondaryX = primaryX === 'start' ? 'end' : 'start';\n    const primaryY = this.yPosition === 'above' ? 'bottom' : 'top';\n    const secondaryY = primaryY === 'top' ? 'bottom' : 'top';\n\n    return strategy.withPositions([\n      {\n        originX: primaryX,\n        originY: secondaryY,\n        overlayX: primaryX,\n        overlayY: primaryY\n      },\n      {\n        originX: primaryX,\n        originY: primaryY,\n        overlayX: primaryX,\n        overlayY: secondaryY\n      },\n      {\n        originX: secondaryX,\n        originY: secondaryY,\n        overlayX: secondaryX,\n        overlayY: primaryY\n      },\n      {\n        originX: secondaryX,\n        originY: primaryY,\n        overlayX: secondaryX,\n        overlayY: secondaryY\n      }\n    ]);\n  }\n\n  /** Gets an observable that will emit when the overlay is supposed to be closed. */\n  private _getCloseStream(overlayRef: OverlayRef) {\n    return merge(\n      overlayRef.backdropClick(),\n      overlayRef.detachments(),\n      overlayRef.keydownEvents().pipe(filter(event => {\n        // Closing on alt + up is only valid when there's an input associated with the datepicker.\n        return (event.keyCode === ESCAPE && !hasModifierKey(event)) || (this.datepickerInput &&\n                hasModifierKey(event, 'altKey') && event.keyCode === UP_ARROW);\n      }))\n    );\n  }\n\n  static ngAcceptInputType_disabled: BooleanInput;\n  static ngAcceptInputType_opened: BooleanInput;\n  static ngAcceptInputType_touchUi: BooleanInput;\n  static ngAcceptInputType_restoreFocus: BooleanInput;\n}\n"]} |
---|