Skip to content

Commit 066c740

Browse files
authoredJan 21, 2025··
fix(material/datepicker): switch away from animations module (#30360)
Reworks the datepicker so it no longer depends on the animations module.
1 parent 013fe04 commit 066c740

File tree

6 files changed

+109
-46
lines changed

6 files changed

+109
-46
lines changed
 

‎src/material/datepicker/datepicker-animations.ts

+2
Original file line numberDiff line numberDiff line change
@@ -18,6 +18,8 @@ import {
1818
/**
1919
* Animations used by the Material datepicker.
2020
* @docs-private
21+
* @deprecated No longer used, will be removed.
22+
* @breaking-change 21.0.0
2123
*/
2224
export const matDatepickerAnimations: {
2325
readonly transformPanel: AnimationTriggerMetadata;

‎src/material/datepicker/datepicker-base.ts

+56-30
Original file line numberDiff line numberDiff line change
@@ -6,7 +6,6 @@
66
* found in the LICENSE file at https://angular.dev/license
77
*/
88

9-
import {AnimationEvent} from '@angular/animations';
109
import {_IdGenerator, CdkTrapFocus} from '@angular/cdk/a11y';
1110
import {Directionality} from '@angular/cdk/bidi';
1211
import {coerceStringArray} from '@angular/cdk/coercion';
@@ -34,6 +33,7 @@ import {DOCUMENT} from '@angular/common';
3433
import {
3534
afterNextRender,
3635
AfterViewInit,
36+
ANIMATION_MODULE_TYPE,
3737
booleanAttribute,
3838
ChangeDetectionStrategy,
3939
ChangeDetectorRef,
@@ -46,10 +46,11 @@ import {
4646
InjectionToken,
4747
Injector,
4848
Input,
49+
NgZone,
4950
OnChanges,
5051
OnDestroy,
51-
OnInit,
5252
Output,
53+
Renderer2,
5354
SimpleChanges,
5455
ViewChild,
5556
ViewContainerRef,
@@ -70,7 +71,6 @@ import {
7071
ExtractDateTypeFromSelection,
7172
MatDateSelectionModel,
7273
} from './date-selection-model';
73-
import {matDatepickerAnimations} from './datepicker-animations';
7474
import {createMissingDateImplError} from './datepicker-errors';
7575
import {DateFilterFn} from './datepicker-input-base';
7676
import {MatDatepickerIntl} from './datepicker-intl';
@@ -120,31 +120,34 @@ export const MAT_DATEPICKER_SCROLL_STRATEGY_FACTORY_PROVIDER = {
120120
host: {
121121
'class': 'mat-datepicker-content',
122122
'[class]': 'color ? "mat-" + color : ""',
123-
'[@transformPanel]': '_animationState',
124-
'(@transformPanel.start)': '_handleAnimationEvent($event)',
125-
'(@transformPanel.done)': '_handleAnimationEvent($event)',
126123
'[class.mat-datepicker-content-touch]': 'datepicker.touchUi',
124+
'[class.mat-datepicker-content-animations-enabled]': '!_animationsDisabled',
127125
},
128-
animations: [matDatepickerAnimations.transformPanel, matDatepickerAnimations.fadeInCalendar],
129126
exportAs: 'matDatepickerContent',
130127
encapsulation: ViewEncapsulation.None,
131128
changeDetection: ChangeDetectionStrategy.OnPush,
132129
imports: [CdkTrapFocus, MatCalendar, CdkPortalOutlet, MatButton],
133130
})
134131
export class MatDatepickerContent<S, D = ExtractDateTypeFromSelection<S>>
135-
implements OnInit, AfterViewInit, OnDestroy
132+
implements AfterViewInit, OnDestroy
136133
{
137-
protected _elementRef = inject(ElementRef);
134+
protected _elementRef = inject<ElementRef<HTMLElement>>(ElementRef);
135+
protected _animationsDisabled =
136+
inject(ANIMATION_MODULE_TYPE, {optional: true}) === 'NoopAnimations';
138137
private _changeDetectorRef = inject(ChangeDetectorRef);
139138
private _globalModel = inject<MatDateSelectionModel<S, D>>(MatDateSelectionModel);
140139
private _dateAdapter = inject<DateAdapter<D>>(DateAdapter)!;
140+
private _ngZone = inject(NgZone);
141141
private _rangeSelectionStrategy = inject<MatDateRangeSelectionStrategy<D>>(
142142
MAT_DATE_RANGE_SELECTION_STRATEGY,
143143
{optional: true},
144144
);
145145

146-
private _subscriptions = new Subscription();
146+
private _stateChanges: Subscription | undefined;
147147
private _model: MatDateSelectionModel<S, D>;
148+
private _eventCleanups: (() => void)[] | undefined;
149+
private _animationFallback: ReturnType<typeof setTimeout> | undefined;
150+
148151
/** Reference to the internal calendar component. */
149152
@ViewChild(MatCalendar) _calendar: MatCalendar<D>;
150153

@@ -175,9 +178,6 @@ export class MatDatepickerContent<S, D = ExtractDateTypeFromSelection<S>>
175178
/** Whether the datepicker is above or below the input. */
176179
_isAbove: boolean;
177180

178-
/** Current state of the animation. */
179-
_animationState: 'enter-dropdown' | 'enter-dialog' | 'void';
180-
181181
/** Emits when an animation has finished. */
182182
readonly _animationDone = new Subject<void>();
183183

@@ -200,26 +200,31 @@ export class MatDatepickerContent<S, D = ExtractDateTypeFromSelection<S>>
200200

201201
constructor() {
202202
inject(_CdkPrivateStyleLoader).load(_VisuallyHiddenLoader);
203-
const intl = inject(MatDatepickerIntl);
203+
this._closeButtonText = inject(MatDatepickerIntl).closeCalendarLabel;
204204

205-
this._closeButtonText = intl.closeCalendarLabel;
206-
}
205+
if (!this._animationsDisabled) {
206+
const element = this._elementRef.nativeElement;
207+
const renderer = inject(Renderer2);
207208

208-
ngOnInit() {
209-
this._animationState = this.datepicker.touchUi ? 'enter-dialog' : 'enter-dropdown';
209+
this._eventCleanups = this._ngZone.runOutsideAngular(() => [
210+
renderer.listen(element, 'animationstart', this._handleAnimationEvent),
211+
renderer.listen(element, 'animationend', this._handleAnimationEvent),
212+
renderer.listen(element, 'animationcancel', this._handleAnimationEvent),
213+
]);
214+
}
210215
}
211216

212217
ngAfterViewInit() {
213-
this._subscriptions.add(
214-
this.datepicker.stateChanges.subscribe(() => {
215-
this._changeDetectorRef.markForCheck();
216-
}),
217-
);
218+
this._stateChanges = this.datepicker.stateChanges.subscribe(() => {
219+
this._changeDetectorRef.markForCheck();
220+
});
218221
this._calendar.focusActiveCell();
219222
}
220223

221224
ngOnDestroy() {
222-
this._subscriptions.unsubscribe();
225+
clearTimeout(this._animationFallback);
226+
this._eventCleanups?.forEach(cleanup => cleanup());
227+
this._stateChanges?.unsubscribe();
223228
this._animationDone.complete();
224229
}
225230

@@ -258,17 +263,38 @@ export class MatDatepickerContent<S, D = ExtractDateTypeFromSelection<S>>
258263
}
259264

260265
_startExitAnimation() {
261-
this._animationState = 'void';
262-
this._changeDetectorRef.markForCheck();
266+
this._elementRef.nativeElement.classList.add('mat-datepicker-content-exit');
267+
268+
if (this._animationsDisabled) {
269+
this._animationDone.next();
270+
} else {
271+
// Some internal apps disable animations in tests using `* {animation: none !important}`.
272+
// If that happens, the animation events won't fire and we'll never clean up the overlay.
273+
// Add a fallback that will fire if the animation doesn't run in a certain amount of time.
274+
clearTimeout(this._animationFallback);
275+
this._animationFallback = setTimeout(() => {
276+
if (!this._isAnimating) {
277+
this._animationDone.next();
278+
}
279+
}, 200);
280+
}
263281
}
264282

265-
_handleAnimationEvent(event: AnimationEvent) {
266-
this._isAnimating = event.phaseName === 'start';
283+
private _handleAnimationEvent = (event: AnimationEvent) => {
284+
const element = this._elementRef.nativeElement;
285+
286+
if (event.target !== element || !event.animationName.startsWith('_mat-datepicker-content')) {
287+
return;
288+
}
289+
290+
clearTimeout(this._animationFallback);
291+
this._isAnimating = event.type === 'animationstart';
292+
element.classList.toggle('mat-datepicker-content-animating', this._isAnimating);
267293

268294
if (!this._isAnimating) {
269295
this._animationDone.next();
270296
}
271-
}
297+
};
272298

273299
_getSelected() {
274300
return this._model.selection as unknown as D | DateRange<D> | null;
@@ -672,7 +698,6 @@ export abstract class MatDatepickerBase<
672698

673699
if (this._componentRef) {
674700
const {instance, location} = this._componentRef;
675-
instance._startExitAnimation();
676701
instance._animationDone.pipe(take(1)).subscribe(() => {
677702
const activeElement = this._document.activeElement;
678703

@@ -690,6 +715,7 @@ export abstract class MatDatepickerBase<
690715
this._focusedElementBeforeOpen = null;
691716
this._destroyOverlay();
692717
});
718+
instance._startExitAnimation();
693719
}
694720

695721
if (canRestoreFocus) {

‎src/material/datepicker/datepicker-content.html

-1
Original file line numberDiff line numberDiff line change
@@ -19,7 +19,6 @@
1919
[dateClass]="datepicker.dateClass"
2020
[comparisonStart]="comparisonStart"
2121
[comparisonEnd]="comparisonEnd"
22-
[@fadeInCalendar]="'enter'"
2322
[startDateAccessibleName]="startDateAccessibleName"
2423
[endDateAccessibleName]="endDateAccessibleName"
2524
(yearSelected)="datepicker._selectYear($event)"

‎src/material/datepicker/datepicker-content.scss

+46-1
Original file line numberDiff line numberDiff line change
@@ -24,6 +24,39 @@ $touch-min-height: 312px;
2424
$touch-max-width: 750px;
2525
$touch-max-height: 788px;
2626

27+
@keyframes _mat-datepicker-content-dropdown-enter {
28+
from {
29+
opacity: 0;
30+
transform: scaleY(0.8);
31+
}
32+
33+
to {
34+
opacity: 1;
35+
transform: none;
36+
}
37+
}
38+
39+
@keyframes _mat-datepicker-content-dialog-enter {
40+
from {
41+
opacity: 0;
42+
transform: scale(0.8);
43+
}
44+
45+
to {
46+
opacity: 1;
47+
transform: none;
48+
}
49+
}
50+
51+
@keyframes _mat-datepicker-content-exit {
52+
from {
53+
opacity: 1;
54+
}
55+
56+
to {
57+
opacity: 0;
58+
}
59+
}
2760

2861
.mat-datepicker-content {
2962
display: block;
@@ -37,6 +70,10 @@ $touch-max-height: 788px;
3770
@include token-utils.create-token-slot(border-radius, calendar-container-shape);
3871
}
3972

73+
&.mat-datepicker-content-animations-enabled {
74+
animation: _mat-datepicker-content-dropdown-enter 120ms cubic-bezier(0, 0, 0.2, 1);
75+
}
76+
4077
.mat-calendar {
4178
width: $non-touch-calendar-width;
4279
height: $non-touch-calendar-height;
@@ -59,7 +96,7 @@ $touch-max-height: 788px;
5996

6097
// Hide the button while the overlay is animating, because it's rendered
6198
// outside of it and it seems to cause scrollbars in some cases (see #21493).
62-
.ng-animating & {
99+
.mat-datepicker-content-animating & {
63100
display: none;
64101
}
65102
}
@@ -89,6 +126,10 @@ $touch-max-height: 788px;
89126
// Prevents the content from jumping around on Windows while the animation is running.
90127
overflow: visible;
91128

129+
&.mat-datepicker-content-animations-enabled {
130+
animation: _mat-datepicker-content-dialog-enter 150ms cubic-bezier(0, 0, 0.2, 1);
131+
}
132+
92133
.mat-datepicker-content-container {
93134
min-height: $touch-min-height;
94135
max-height: $touch-max-height;
@@ -102,6 +143,10 @@ $touch-max-height: 788px;
102143
}
103144
}
104145

146+
.mat-datepicker-content-exit.mat-datepicker-content-animations-enabled {
147+
animation: _mat-datepicker-content-exit 100ms linear;
148+
}
149+
105150
@media all and (orientation: landscape) {
106151
.mat-datepicker-content-touch .mat-datepicker-content-container {
107152
width: $touch-landscape-width;

‎src/material/datepicker/datepicker.spec.ts

-5
Original file line numberDiff line numberDiff line change
@@ -481,25 +481,20 @@ describe('MatDatepicker', () => {
481481
for (let i = 0; i < 3; i++) {
482482
testComponent.datepicker.open();
483483
fixture.detectChanges();
484-
tick();
485484

486485
testComponent.datepicker.close();
487486
fixture.detectChanges();
488-
tick();
489487
}
490488

491489
testComponent.datepicker.open();
492490
fixture.detectChanges();
493-
tick();
494-
flush();
495491

496492
const spy = jasmine.createSpy('close event spy');
497493
const subscription = testComponent.datepicker.closedStream.subscribe(spy);
498494
const backdrop = document.querySelector('.cdk-overlay-backdrop')! as HTMLElement;
499495

500496
backdrop.click();
501497
fixture.detectChanges();
502-
flush();
503498

504499
expect(spy).toHaveBeenCalledTimes(1);
505500
expect(testComponent.datepicker.opened).toBe(false);

‎tools/public_api_guard/material/datepicker.md

+5-9
Original file line numberDiff line numberDiff line change
@@ -8,7 +8,6 @@ import { AbstractControl } from '@angular/forms';
88
import { AfterContentInit } from '@angular/core';
99
import { AfterViewChecked } from '@angular/core';
1010
import { AfterViewInit } from '@angular/core';
11-
import { AnimationEvent as AnimationEvent_2 } from '@angular/animations';
1211
import { AnimationTriggerMetadata } from '@angular/animations';
1312
import { ChangeDetectorRef } from '@angular/core';
1413
import { ComponentType } from '@angular/cdk/portal';
@@ -338,7 +337,7 @@ export class MatDatepickerActions implements AfterViewInit, OnDestroy {
338337
static ɵfac: i0.ɵɵFactoryDeclaration<MatDatepickerActions, never>;
339338
}
340339

341-
// @public
340+
// @public @deprecated
342341
export const matDatepickerAnimations: {
343342
readonly transformPanel: AnimationTriggerMetadata;
344343
readonly fadeInCalendar: AnimationTriggerMetadata;
@@ -367,11 +366,12 @@ export class MatDatepickerCancel {
367366
}
368367

369368
// @public
370-
export class MatDatepickerContent<S, D = ExtractDateTypeFromSelection<S>> implements OnInit, AfterViewInit, OnDestroy {
369+
export class MatDatepickerContent<S, D = ExtractDateTypeFromSelection<S>> implements AfterViewInit, OnDestroy {
371370
constructor(...args: unknown[]);
372371
_actionsPortal: TemplatePortal | null;
373372
readonly _animationDone: Subject<void>;
374-
_animationState: 'enter-dropdown' | 'enter-dialog' | 'void';
373+
// (undocumented)
374+
protected _animationsDisabled: boolean;
375375
_applyPendingSelection(): void;
376376
_assignActions(portal: TemplatePortal<any> | null, forceRerender: boolean): void;
377377
_calendar: MatCalendar<D>;
@@ -383,13 +383,11 @@ export class MatDatepickerContent<S, D = ExtractDateTypeFromSelection<S>> implem
383383
datepicker: MatDatepickerBase<any, S, D>;
384384
_dialogLabelId: string | null;
385385
// (undocumented)
386-
protected _elementRef: ElementRef<any>;
386+
protected _elementRef: ElementRef<HTMLElement>;
387387
endDateAccessibleName: string | null;
388388
// (undocumented)
389389
_getSelected(): D | DateRange<D> | null;
390390
// (undocumented)
391-
_handleAnimationEvent(event: AnimationEvent_2): void;
392-
// (undocumented)
393391
_handleUserDragDrop(event: MatCalendarUserEvent<DateRange<D>>): void;
394392
// (undocumented)
395393
_handleUserSelection(event: MatCalendarUserEvent<D | null>): void;
@@ -399,8 +397,6 @@ export class MatDatepickerContent<S, D = ExtractDateTypeFromSelection<S>> implem
399397
ngAfterViewInit(): void;
400398
// (undocumented)
401399
ngOnDestroy(): void;
402-
// (undocumented)
403-
ngOnInit(): void;
404400
startDateAccessibleName: string | null;
405401
// (undocumented)
406402
_startExitAnimation(): void;

0 commit comments

Comments
 (0)
Please sign in to comment.