Skip to content

Commit 992aff1

Browse files
authoredJan 28, 2025··
fix(material/timepicker): switch away from animations module (#30404)
Reworks the timepicker to move it away from the animations module for the dropdown animation.
1 parent d84a4c3 commit 992aff1

File tree

4 files changed

+62
-20
lines changed

4 files changed

+62
-20
lines changed
 

‎src/material/timepicker/timepicker.html

+4-2
Original file line numberDiff line numberDiff line change
@@ -2,14 +2,16 @@
22
<div
33
role="listbox"
44
class="mat-timepicker-panel"
5+
[class.mat-timepicker-panel-animations-enabled]="!_animationsDisabled"
6+
[class.mat-timepicker-panel-exit]="!isOpen()"
57
[attr.aria-label]="ariaLabel() || null"
68
[attr.aria-labelledby]="_getAriaLabelledby()"
79
[id]="panelId"
8-
@panel>
10+
(animationend)="_handleAnimationEnd($event)">
911
@for (option of _timeOptions; track option.value) {
1012
<mat-option
1113
[value]="option.value"
12-
(onSelectionChange)="_selectValue(option.value)">{{option.label}}</mat-option>
14+
(onSelectionChange)="_selectValue($event.source)">{{option.label}}</mat-option>
1315
}
1416
</div>
1517
</ng-template>

‎src/material/timepicker/timepicker.scss

+30
Original file line numberDiff line numberDiff line change
@@ -2,6 +2,28 @@
22
@use '../core/tokens/token-utils';
33
@use '../core/tokens/m2/mat/timepicker' as tokens-mat-timepicker;
44

5+
@keyframes _mat-timepicker-enter {
6+
from {
7+
opacity: 0;
8+
transform: scaleY(0.8);
9+
}
10+
11+
to {
12+
opacity: 1;
13+
transform: none;
14+
}
15+
}
16+
17+
@keyframes _mat-timepicker-exit {
18+
from {
19+
opacity: 1;
20+
}
21+
22+
to {
23+
opacity: 0;
24+
}
25+
}
26+
527
mat-timepicker {
628
display: none;
729
}
@@ -38,6 +60,14 @@ mat-timepicker {
3860
}
3961
}
4062

63+
.mat-timepicker-panel-animations-enabled {
64+
animation: _mat-timepicker-enter 120ms cubic-bezier(0, 0, 0.2, 1);
65+
66+
&.mat-timepicker-panel-exit {
67+
animation: _mat-timepicker-exit 100ms linear;
68+
}
69+
}
70+
4171
.mat-timepicker-input[readonly] {
4272
cursor: pointer;
4373
}

‎src/material/timepicker/timepicker.ts

+24-17
Original file line numberDiff line numberDiff line change
@@ -9,6 +9,7 @@
99
import {
1010
afterNextRender,
1111
AfterRenderRef,
12+
ANIMATION_MODULE_TYPE,
1213
booleanAttribute,
1314
ChangeDetectionStrategy,
1415
Component,
@@ -32,7 +33,6 @@ import {
3233
ViewContainerRef,
3334
ViewEncapsulation,
3435
} from '@angular/core';
35-
import {animate, group, state, style, transition, trigger} from '@angular/animations';
3636
import {
3737
DateAdapter,
3838
MAT_DATE_FORMATS,
@@ -80,18 +80,6 @@ export interface MatTimepickerSelected<D> {
8080
useExisting: MatTimepicker,
8181
},
8282
],
83-
animations: [
84-
trigger('panel', [
85-
state('void', style({opacity: 0, transform: 'scaleY(0.8)'})),
86-
transition(':enter', [
87-
group([
88-
animate('0.03s linear', style({opacity: 1})),
89-
animate('0.12s cubic-bezier(0, 0, 0.2, 1)', style({transform: 'scaleY(1)'})),
90-
]),
91-
]),
92-
transition(':leave', [animate('0.075s linear', style({opacity: 0}))]),
93-
]),
94-
],
9583
})
9684
export class MatTimepicker<D> implements OnDestroy, MatOptionParentComponent {
9785
private _overlay = inject(Overlay);
@@ -101,6 +89,8 @@ export class MatTimepicker<D> implements OnDestroy, MatOptionParentComponent {
10189
private _defaultConfig = inject(MAT_TIMEPICKER_CONFIG, {optional: true});
10290
private _dateAdapter = inject<DateAdapter<D>>(DateAdapter, {optional: true})!;
10391
private _dateFormats = inject(MAT_DATE_FORMATS, {optional: true})!;
92+
protected _animationsDisabled =
93+
inject(ANIMATION_MODULE_TYPE, {optional: true}) === 'NoopAnimations';
10494

10595
private _isOpen = signal(false);
10696
private _activeDescendant = signal<string | null>(null);
@@ -246,8 +236,11 @@ export class MatTimepicker<D> implements OnDestroy, MatOptionParentComponent {
246236
close(): void {
247237
if (this._isOpen()) {
248238
this._isOpen.set(false);
249-
this._overlayRef?.detach();
250239
this.closed.emit();
240+
241+
if (this._animationsDisabled) {
242+
this._overlayRef?.detach();
243+
}
251244
}
252245
}
253246

@@ -270,9 +263,16 @@ export class MatTimepicker<D> implements OnDestroy, MatOptionParentComponent {
270263
}
271264

272265
/** Selects a specific time value. */
273-
protected _selectValue(value: D) {
266+
protected _selectValue(option: MatOption<D>) {
274267
this.close();
275-
this.selected.emit({value, source: this});
268+
this._keyManager.setActiveItem(option);
269+
this._options().forEach(current => {
270+
// This is primarily here so we don't show two selected options while animating away.
271+
if (current !== option) {
272+
current.deselect(false);
273+
}
274+
});
275+
this.selected.emit({value: option.value, source: this});
276276
this._input()?.focus();
277277
}
278278

@@ -284,6 +284,13 @@ export class MatTimepicker<D> implements OnDestroy, MatOptionParentComponent {
284284
return this.ariaLabelledby() || this._input()?._getLabelId() || null;
285285
}
286286

287+
/** Handles animation events coming from the panel. */
288+
protected _handleAnimationEnd(event: AnimationEvent) {
289+
if (event.animationName === '_mat-timepicker-exit') {
290+
this._overlayRef?.detach();
291+
}
292+
}
293+
287294
/** Creates an overlay reference for the timepicker panel. */
288295
private _getOverlayRef(): OverlayRef {
289296
if (this._overlayRef) {
@@ -409,7 +416,7 @@ export class MatTimepicker<D> implements OnDestroy, MatOptionParentComponent {
409416
event.preventDefault();
410417

411418
if (this._keyManager.activeItem) {
412-
this._selectValue(this._keyManager.activeItem.value);
419+
this._selectValue(this._keyManager.activeItem);
413420
} else {
414421
this.close();
415422
}

‎tools/public_api_guard/material/timepicker.md

+4-1
Original file line numberDiff line numberDiff line change
@@ -29,13 +29,16 @@ export const MAT_TIMEPICKER_CONFIG: InjectionToken<MatTimepickerConfig>;
2929
export class MatTimepicker<D> implements OnDestroy, MatOptionParentComponent {
3030
constructor();
3131
readonly activeDescendant: Signal<string | null>;
32+
// (undocumented)
33+
protected _animationsDisabled: boolean;
3234
readonly ariaLabel: InputSignal<string | null>;
3335
readonly ariaLabelledby: InputSignal<string | null>;
3436
close(): void;
3537
readonly closed: OutputEmitterRef<void>;
3638
readonly disabled: Signal<boolean>;
3739
readonly disableRipple: InputSignalWithTransform<boolean, unknown>;
3840
protected _getAriaLabelledby(): string | null;
41+
protected _handleAnimationEnd(event: AnimationEvent): void;
3942
readonly interval: InputSignalWithTransform<number | null, number | string | null>;
4043
readonly isOpen: Signal<boolean>;
4144
// (undocumented)
@@ -50,7 +53,7 @@ export class MatTimepicker<D> implements OnDestroy, MatOptionParentComponent {
5053
protected _panelTemplate: Signal<TemplateRef<unknown>>;
5154
registerInput(input: MatTimepickerInput<D>): void;
5255
readonly selected: OutputEmitterRef<MatTimepickerSelected<D>>;
53-
protected _selectValue(value: D): void;
56+
protected _selectValue(option: MatOption<D>): void;
5457
// (undocumented)
5558
protected _timeOptions: readonly MatTimepickerOption<D>[];
5659
// (undocumented)

0 commit comments

Comments
 (0)
Please sign in to comment.