6
6
* found in the LICENSE file at https://angular.dev/license
7
7
*/
8
8
9
- import { AnimationEvent } from '@angular/animations' ;
10
9
import { _IdGenerator , CdkTrapFocus } from '@angular/cdk/a11y' ;
11
10
import { Directionality } from '@angular/cdk/bidi' ;
12
11
import { coerceStringArray } from '@angular/cdk/coercion' ;
@@ -34,6 +33,7 @@ import {DOCUMENT} from '@angular/common';
34
33
import {
35
34
afterNextRender ,
36
35
AfterViewInit ,
36
+ ANIMATION_MODULE_TYPE ,
37
37
booleanAttribute ,
38
38
ChangeDetectionStrategy ,
39
39
ChangeDetectorRef ,
@@ -46,10 +46,11 @@ import {
46
46
InjectionToken ,
47
47
Injector ,
48
48
Input ,
49
+ NgZone ,
49
50
OnChanges ,
50
51
OnDestroy ,
51
- OnInit ,
52
52
Output ,
53
+ Renderer2 ,
53
54
SimpleChanges ,
54
55
ViewChild ,
55
56
ViewContainerRef ,
@@ -70,7 +71,6 @@ import {
70
71
ExtractDateTypeFromSelection ,
71
72
MatDateSelectionModel ,
72
73
} from './date-selection-model' ;
73
- import { matDatepickerAnimations } from './datepicker-animations' ;
74
74
import { createMissingDateImplError } from './datepicker-errors' ;
75
75
import { DateFilterFn } from './datepicker-input-base' ;
76
76
import { MatDatepickerIntl } from './datepicker-intl' ;
@@ -120,31 +120,34 @@ export const MAT_DATEPICKER_SCROLL_STRATEGY_FACTORY_PROVIDER = {
120
120
host : {
121
121
'class' : 'mat-datepicker-content' ,
122
122
'[class]' : 'color ? "mat-" + color : ""' ,
123
- '[@transformPanel]' : '_animationState' ,
124
- '(@transformPanel.start)' : '_handleAnimationEvent($event)' ,
125
- '(@transformPanel.done)' : '_handleAnimationEvent($event)' ,
126
123
'[class.mat-datepicker-content-touch]' : 'datepicker.touchUi' ,
124
+ '[class.mat-datepicker-content-animations-enabled]' : '!_animationsDisabled' ,
127
125
} ,
128
- animations : [ matDatepickerAnimations . transformPanel , matDatepickerAnimations . fadeInCalendar ] ,
129
126
exportAs : 'matDatepickerContent' ,
130
127
encapsulation : ViewEncapsulation . None ,
131
128
changeDetection : ChangeDetectionStrategy . OnPush ,
132
129
imports : [ CdkTrapFocus , MatCalendar , CdkPortalOutlet , MatButton ] ,
133
130
} )
134
131
export class MatDatepickerContent < S , D = ExtractDateTypeFromSelection < S > >
135
- implements OnInit , AfterViewInit , OnDestroy
132
+ implements AfterViewInit , OnDestroy
136
133
{
137
- protected _elementRef = inject ( ElementRef ) ;
134
+ protected _elementRef = inject < ElementRef < HTMLElement > > ( ElementRef ) ;
135
+ protected _animationsDisabled =
136
+ inject ( ANIMATION_MODULE_TYPE , { optional : true } ) === 'NoopAnimations' ;
138
137
private _changeDetectorRef = inject ( ChangeDetectorRef ) ;
139
138
private _globalModel = inject < MatDateSelectionModel < S , D > > ( MatDateSelectionModel ) ;
140
139
private _dateAdapter = inject < DateAdapter < D > > ( DateAdapter ) ! ;
140
+ private _ngZone = inject ( NgZone ) ;
141
141
private _rangeSelectionStrategy = inject < MatDateRangeSelectionStrategy < D > > (
142
142
MAT_DATE_RANGE_SELECTION_STRATEGY ,
143
143
{ optional : true } ,
144
144
) ;
145
145
146
- private _subscriptions = new Subscription ( ) ;
146
+ private _stateChanges : Subscription | undefined ;
147
147
private _model : MatDateSelectionModel < S , D > ;
148
+ private _eventCleanups : ( ( ) => void ) [ ] | undefined ;
149
+ private _animationFallback : ReturnType < typeof setTimeout > | undefined ;
150
+
148
151
/** Reference to the internal calendar component. */
149
152
@ViewChild ( MatCalendar ) _calendar : MatCalendar < D > ;
150
153
@@ -175,9 +178,6 @@ export class MatDatepickerContent<S, D = ExtractDateTypeFromSelection<S>>
175
178
/** Whether the datepicker is above or below the input. */
176
179
_isAbove : boolean ;
177
180
178
- /** Current state of the animation. */
179
- _animationState : 'enter-dropdown' | 'enter-dialog' | 'void' ;
180
-
181
181
/** Emits when an animation has finished. */
182
182
readonly _animationDone = new Subject < void > ( ) ;
183
183
@@ -200,26 +200,31 @@ export class MatDatepickerContent<S, D = ExtractDateTypeFromSelection<S>>
200
200
201
201
constructor ( ) {
202
202
inject ( _CdkPrivateStyleLoader ) . load ( _VisuallyHiddenLoader ) ;
203
- const intl = inject ( MatDatepickerIntl ) ;
203
+ this . _closeButtonText = inject ( MatDatepickerIntl ) . closeCalendarLabel ;
204
204
205
- this . _closeButtonText = intl . closeCalendarLabel ;
206
- }
205
+ if ( ! this . _animationsDisabled ) {
206
+ const element = this . _elementRef . nativeElement ;
207
+ const renderer = inject ( Renderer2 ) ;
207
208
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
+ }
210
215
}
211
216
212
217
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
+ } ) ;
218
221
this . _calendar . focusActiveCell ( ) ;
219
222
}
220
223
221
224
ngOnDestroy ( ) {
222
- this . _subscriptions . unsubscribe ( ) ;
225
+ clearTimeout ( this . _animationFallback ) ;
226
+ this . _eventCleanups ?. forEach ( cleanup => cleanup ( ) ) ;
227
+ this . _stateChanges ?. unsubscribe ( ) ;
223
228
this . _animationDone . complete ( ) ;
224
229
}
225
230
@@ -258,17 +263,38 @@ export class MatDatepickerContent<S, D = ExtractDateTypeFromSelection<S>>
258
263
}
259
264
260
265
_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
+ }
263
281
}
264
282
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 ) ;
267
293
268
294
if ( ! this . _isAnimating ) {
269
295
this . _animationDone . next ( ) ;
270
296
}
271
- }
297
+ } ;
272
298
273
299
_getSelected ( ) {
274
300
return this . _model . selection as unknown as D | DateRange < D > | null ;
@@ -672,7 +698,6 @@ export abstract class MatDatepickerBase<
672
698
673
699
if ( this . _componentRef ) {
674
700
const { instance, location} = this . _componentRef ;
675
- instance . _startExitAnimation ( ) ;
676
701
instance . _animationDone . pipe ( take ( 1 ) ) . subscribe ( ( ) => {
677
702
const activeElement = this . _document . activeElement ;
678
703
@@ -690,6 +715,7 @@ export abstract class MatDatepickerBase<
690
715
this . _focusedElementBeforeOpen = null ;
691
716
this . _destroyOverlay ( ) ;
692
717
} ) ;
718
+ instance . _startExitAnimation ( ) ;
693
719
}
694
720
695
721
if ( canRestoreFocus ) {
0 commit comments