Skip to content

Commit de5e57a

Browse files
committedNov 5, 2024·
fix(material/menu): use static elevation (#29968)
Currently the menu's elevation is set using elevation classes, because earlier versions of the Material spec called for the elevation to increase for each level in a nested menu. That's no longer part of the spec and in v19 we won't include the elevation classes by default anymore. These changes remove the code that was handling the dynamic elevation and set the elevation using a token instead. (cherry picked from commit b072047)
1 parent 5ea076a commit de5e57a

File tree

9 files changed

+24
-204
lines changed

9 files changed

+24
-204
lines changed
 

‎src/material/core/tokens/m2/mat/_menu.scss

+5-3
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,7 @@
11
@use '../../token-definition';
22
@use '../../../theming/inspection';
33
@use '../../../style/sass-utils';
4+
@use '../../../style/elevation';
45

56
// The prefix used to generate the fully qualified name for tokens in this file.
67
$prefix: (mat, menu);
@@ -18,9 +19,10 @@ $prefix: (mat, menu);
1819
item-trailing-spacing: 16px,
1920
item-with-icon-leading-spacing: 16px,
2021
item-with-icon-trailing-spacing: 16px,
21-
// Note that this uses a value, rather than a computed box-shadow, because we use
22-
// the value at runtime to determine which shadow to set based on the menu's depth.
23-
base-elevation-level: 8,
22+
container-elevation-shadow: elevation.get-box-shadow(8),
23+
24+
// Unused
25+
base-elevation-level: null,
2426
);
2527
}
2628

‎src/material/core/tokens/m3/mat/_menu.scss

+6-3
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,7 @@
11
@use 'sass:map';
22
@use '../../../style/sass-utils';
33
@use '../../token-definition';
4+
@use '../../../style/elevation';
45

56
// The prefix used to generate the fully qualified name for tokens in this file.
67
$prefix: (mat, menu);
@@ -35,9 +36,11 @@ $prefix: (mat, menu);
3536
item-with-icon-leading-spacing: token-definition.hardcode(12px, $exclude-hardcoded),
3637
item-with-icon-trailing-spacing: token-definition.hardcode(12px, $exclude-hardcoded),
3738
container-color: map.get($systems, md-sys-color, surface-container),
38-
// Note that this uses a value, rather than a computed box-shadow, because we use
39-
// the value at runtime to determine which shadow to set based on the menu's depth.
40-
base-elevation-level: token-definition.hardcode(2, $exclude-hardcoded),
39+
container-elevation-shadow: token-definition.hardcode(
40+
elevation.get-box-shadow(2), $exclude-hardcoded),
41+
42+
// Unused
43+
base-elevation-level: null,
4144
)
4245
);
4346

‎src/material/menu/menu-panel.ts

+5
Original file line numberDiff line numberDiff line change
@@ -33,6 +33,11 @@ export interface MatMenuPanel<T = any> {
3333
focusFirstItem: (origin?: FocusOrigin) => void;
3434
resetActiveItem: () => void;
3535
setPositionClasses?: (x: MenuPositionX, y: MenuPositionY) => void;
36+
37+
/**
38+
* @deprecated No longer used and will be removed.
39+
* @breaking-change 21.0.0
40+
*/
3641
setElevation?(depth: number): void;
3742
lazyContent?: MatMenuContent;
3843
backdropClass?: string;

‎src/material/menu/menu-trigger.ts

-16
Original file line numberDiff line numberDiff line change
@@ -382,26 +382,10 @@ export class MatMenuTrigger implements AfterContentInit, OnDestroy {
382382
private _initMenu(menu: MatMenuPanel): void {
383383
menu.parentMenu = this.triggersSubmenu() ? this._parentMaterialMenu : undefined;
384384
menu.direction = this.dir;
385-
this._setMenuElevation(menu);
386385
menu.focusFirstItem(this._openedBy || 'program');
387386
this._setIsMenuOpen(true);
388387
}
389388

390-
/** Updates the menu elevation based on the amount of parent menus that it has. */
391-
private _setMenuElevation(menu: MatMenuPanel): void {
392-
if (menu.setElevation) {
393-
let depth = 0;
394-
let parentMenu = menu.parentMenu;
395-
396-
while (parentMenu) {
397-
depth++;
398-
parentMenu = parentMenu.parentMenu;
399-
}
400-
401-
menu.setElevation(depth);
402-
}
403-
}
404-
405389
// set state rather than toggle to support triggers sharing a menu
406390
private _setIsMenuOpen(isOpen: boolean): void {
407391
if (isOpen !== this._menuOpen) {

‎src/material/menu/menu.html

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
<ng-template>
22
<div
3-
class="mat-mdc-menu-panel mat-mdc-elevation-specific"
3+
class="mat-mdc-menu-panel"
44
[id]="panelId"
55
[class]="_classList"
66
(keydown)="_handleKeydown($event)"

‎src/material/menu/menu.scss

+1
Original file line numberDiff line numberDiff line change
@@ -39,6 +39,7 @@ mat-menu {
3939
@include token-utils.use-tokens(tokens-mat-menu.$prefix, tokens-mat-menu.get-token-slots()) {
4040
@include token-utils.create-token-slot(border-radius, container-shape);
4141
@include token-utils.create-token-slot(background-color, container-color);
42+
@include token-utils.create-token-slot(box-shadow, container-elevation-shadow);
4243
}
4344

4445
// TODO(crisbeto): we don't need this for anything, but it was there when

‎src/material/menu/menu.spec.ts

-142
Original file line numberDiff line numberDiff line change
@@ -594,32 +594,6 @@ describe('MatMenu', () => {
594594
expect(panel.classList).toContain('custom-two');
595595
}));
596596

597-
it('should not remove mat-elevation class from overlay when panelClass is changed', fakeAsync(() => {
598-
const fixture = createComponent(SimpleMenu, [], [FakeIcon]);
599-
600-
fixture.componentInstance.panelClass = 'custom-one';
601-
fixture.detectChanges();
602-
fixture.componentInstance.trigger.openMenu();
603-
fixture.detectChanges();
604-
tick(500);
605-
606-
const panel = overlayContainerElement.querySelector('.mat-mdc-menu-panel')!;
607-
608-
expect(panel.classList).toContain('custom-one');
609-
expect(panel.classList).toContain('mat-elevation-z2');
610-
611-
fixture.componentInstance.panelClass = 'custom-two';
612-
fixture.changeDetectorRef.markForCheck();
613-
fixture.detectChanges();
614-
615-
expect(panel.classList).not.toContain('custom-one');
616-
expect(panel.classList).toContain('custom-two');
617-
expect(panel.classList).toContain('mat-mdc-elevation-specific');
618-
expect(panel.classList)
619-
.withContext('Expected mat-elevation-z2 not to be removed')
620-
.toContain('mat-elevation-z2');
621-
}));
622-
623597
it('should set the "menu" role on the overlay panel', fakeAsync(() => {
624598
const fixture = createComponent(SimpleMenu, [], [FakeIcon]);
625599
fixture.detectChanges();
@@ -2350,79 +2324,6 @@ describe('MatMenu', () => {
23502324
expect(menuItems[0].querySelector('.mat-mdc-menu-submenu-icon')).toBeFalsy();
23512325
}));
23522326

2353-
it('should increase the sub-menu elevation based on its depth', fakeAsync(() => {
2354-
compileTestComponent();
2355-
instance.rootTrigger.openMenu();
2356-
fixture.detectChanges();
2357-
tick(500);
2358-
2359-
instance.levelOneTrigger.openMenu();
2360-
fixture.detectChanges();
2361-
tick(500);
2362-
2363-
instance.levelTwoTrigger.openMenu();
2364-
fixture.detectChanges();
2365-
tick(500);
2366-
2367-
const menus = overlay.querySelectorAll('.mat-mdc-menu-panel');
2368-
2369-
expect(menus[0].classList).toContain('mat-mdc-elevation-specific');
2370-
expect(menus[0].classList)
2371-
.withContext('Expected root menu to have base elevation.')
2372-
.toContain('mat-elevation-z2');
2373-
2374-
expect(menus[1].classList).toContain('mat-mdc-elevation-specific');
2375-
expect(menus[1].classList)
2376-
.withContext('Expected first sub-menu to have base elevation + 1.')
2377-
.toContain('mat-elevation-z3');
2378-
2379-
expect(menus[2].classList).toContain('mat-mdc-elevation-specific');
2380-
expect(menus[2].classList)
2381-
.withContext('Expected second sub-menu to have base elevation + 2.')
2382-
.toContain('mat-elevation-z4');
2383-
}));
2384-
2385-
it('should update the elevation when the same menu is opened at a different depth', fakeAsync(() => {
2386-
compileTestComponent();
2387-
instance.rootTrigger.openMenu();
2388-
fixture.detectChanges();
2389-
2390-
instance.levelOneTrigger.openMenu();
2391-
fixture.detectChanges();
2392-
2393-
instance.levelTwoTrigger.openMenu();
2394-
fixture.detectChanges();
2395-
2396-
let lastMenu = overlay.querySelectorAll('.mat-mdc-menu-panel')[2];
2397-
2398-
expect(lastMenu.classList).toContain('mat-mdc-elevation-specific');
2399-
expect(lastMenu.classList)
2400-
.withContext('Expected menu to have the base elevation plus two.')
2401-
.toContain('mat-elevation-z4');
2402-
2403-
(overlay.querySelector('.cdk-overlay-backdrop')! as HTMLElement).click();
2404-
fixture.detectChanges();
2405-
tick(500);
2406-
2407-
expect(overlay.querySelectorAll('.mat-mdc-menu-panel').length)
2408-
.withContext('Expected no open menus')
2409-
.toBe(0);
2410-
2411-
instance.alternateTrigger.openMenu();
2412-
fixture.detectChanges();
2413-
tick(500);
2414-
2415-
lastMenu = overlay.querySelector('.mat-mdc-menu-panel') as HTMLElement;
2416-
2417-
expect(lastMenu.classList).toContain('mat-mdc-elevation-specific');
2418-
expect(lastMenu.classList)
2419-
.not.withContext('Expected menu not to maintain old elevation.')
2420-
.toContain('mat-elevation-z4');
2421-
expect(lastMenu.classList)
2422-
.withContext('Expected menu to have the proper updated elevation.')
2423-
.toContain('mat-elevation-z2');
2424-
}));
2425-
24262327
it('should not change focus origin if origin not specified for trigger', fakeAsync(() => {
24272328
compileTestComponent();
24282329

@@ -2442,28 +2343,6 @@ describe('MatMenu', () => {
24422343
expect(levelTwoTrigger.classList).toContain('cdk-mouse-focused');
24432344
}));
24442345

2445-
it('should not increase the elevation if the user specified a custom one', fakeAsync(() => {
2446-
const elevationFixture = createComponent(NestedMenuCustomElevation);
2447-
2448-
elevationFixture.detectChanges();
2449-
elevationFixture.componentInstance.rootTrigger.openMenu();
2450-
elevationFixture.detectChanges();
2451-
tick(500);
2452-
2453-
elevationFixture.componentInstance.levelOneTrigger.openMenu();
2454-
elevationFixture.detectChanges();
2455-
tick(500);
2456-
2457-
const menuClasses =
2458-
overlayContainerElement.querySelectorAll('.mat-mdc-menu-panel')[1].classList;
2459-
2460-
expect(menuClasses).toContain('mat-mdc-elevation-specific');
2461-
expect(menuClasses)
2462-
.withContext('Expected user elevation to be maintained')
2463-
.toContain('mat-elevation-z24');
2464-
expect(menuClasses).not.toContain('mat-elevation-z2', 'Expected no stacked elevation.');
2465-
}));
2466-
24672346
it('should close all of the menus when the root is closed programmatically', fakeAsync(() => {
24682347
compileTestComponent();
24692348
instance.rootTrigger.openMenu();
@@ -2934,27 +2813,6 @@ class NestedMenu {
29342813
showLazy = false;
29352814
}
29362815

2937-
@Component({
2938-
template: `
2939-
<button [matMenuTriggerFor]="root" #rootTrigger="matMenuTrigger">Toggle menu</button>
2940-
2941-
<mat-menu #root="matMenu">
2942-
<button mat-menu-item
2943-
[matMenuTriggerFor]="levelOne"
2944-
#levelOneTrigger="matMenuTrigger">One</button>
2945-
</mat-menu>
2946-
2947-
<mat-menu #levelOne="matMenu" class="mat-elevation-z24">
2948-
<button mat-menu-item>Two</button>
2949-
</mat-menu>
2950-
`,
2951-
standalone: false,
2952-
})
2953-
class NestedMenuCustomElevation {
2954-
@ViewChild('rootTrigger') rootTrigger: MatMenuTrigger;
2955-
@ViewChild('levelOneTrigger') levelOneTrigger: MatMenuTrigger;
2956-
}
2957-
29582816
@Component({
29592817
template: `
29602818
<button [matMenuTriggerFor]="root" #rootTriggerEl>Toggle menu</button>

‎src/material/menu/menu.ts

+3-37
Original file line numberDiff line numberDiff line change
@@ -116,9 +116,6 @@ export class MatMenu implements AfterContentInit, MatMenuPanel<MatMenuItem>, OnI
116116
private _xPosition: MenuPositionX;
117117
private _yPosition: MenuPositionY;
118118
private _firstItemFocusRef?: AfterRenderRef;
119-
private _previousElevation: string;
120-
private _elevationPrefix = 'mat-elevation-z';
121-
private _baseElevation: number | null = null;
122119

123120
/** All items inside the menu. Includes items nested inside another menu. */
124121
@ContentChildren(MatMenuItem, {descendants: true}) _allItems: QueryList<MatMenuItem>;
@@ -439,41 +436,10 @@ export class MatMenu implements AfterContentInit, MatMenuPanel<MatMenuItem>, OnI
439436
}
440437

441438
/**
442-
* Sets the menu panel elevation.
443-
* @param depth Number of parent menus that come before the menu.
439+
* @deprecated No longer used and will be removed.
440+
* @breaking-change 21.0.0
444441
*/
445-
setElevation(depth: number): void {
446-
// The base elevation depends on which version of the spec
447-
// we're running so we have to resolve it at runtime.
448-
if (this._baseElevation === null) {
449-
const styles =
450-
typeof getComputedStyle === 'function'
451-
? getComputedStyle(this._elementRef.nativeElement)
452-
: null;
453-
const value = styles?.getPropertyValue('--mat-menu-base-elevation-level') || '8';
454-
this._baseElevation = parseInt(value);
455-
}
456-
457-
// The elevation starts at the base and increases by one for each level.
458-
// Capped at 24 because that's the maximum elevation defined in the Material design spec.
459-
const elevation = Math.min(this._baseElevation + depth, 24);
460-
const newElevation = `${this._elevationPrefix}${elevation}`;
461-
const customElevation = Object.keys(this._classList).find(className => {
462-
return className.startsWith(this._elevationPrefix);
463-
});
464-
465-
if (!customElevation || customElevation === this._previousElevation) {
466-
const newClassList = {...this._classList};
467-
468-
if (this._previousElevation) {
469-
newClassList[this._previousElevation] = false;
470-
}
471-
472-
newClassList[newElevation] = true;
473-
this._previousElevation = newElevation;
474-
this._classList = newClassList;
475-
}
476-
}
442+
setElevation(_depth: number): void {}
477443

478444
/**
479445
* Adds classes to the menu panel based on its position. Can be used by

‎tools/public_api_guard/material/menu.md

+3-2
Original file line numberDiff line numberDiff line change
@@ -102,7 +102,8 @@ export class MatMenu implements AfterContentInit, MatMenuPanel<MatMenuItem>, OnI
102102
removeItem(_item: MatMenuItem): void;
103103
resetActiveItem(): void;
104104
_resetAnimation(): void;
105-
setElevation(depth: number): void;
105+
// @deprecated (undocumented)
106+
setElevation(_depth: number): void;
106107
setPositionClasses(posX?: MenuPositionX, posY?: MenuPositionY): void;
107108
_startAnimation(): void;
108109
templateRef: TemplateRef<any>;
@@ -222,7 +223,7 @@ export interface MatMenuPanel<T = any> {
222223
removeItem?: (item: T) => void;
223224
// (undocumented)
224225
resetActiveItem: () => void;
225-
// (undocumented)
226+
// @deprecated (undocumented)
226227
setElevation?(depth: number): void;
227228
// (undocumented)
228229
setPositionClasses?: (x: MenuPositionX, y: MenuPositionY) => void;

0 commit comments

Comments
 (0)
Please sign in to comment.