You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
226 lines
26 KiB
226 lines
26 KiB
/**
|
|
* @license
|
|
* Copyright Google LLC All Rights Reserved.
|
|
*
|
|
* Use of this source code is governed by an MIT-style license that can be
|
|
* found in the LICENSE file at https://angular.io/license
|
|
*/
|
|
import { FocusMonitor } from '@angular/cdk/a11y';
|
|
import { ChangeDetectionStrategy, Component, ElementRef, ViewEncapsulation, Inject, Optional, Input, HostListener, ChangeDetectorRef, } from '@angular/core';
|
|
import { mixinDisabled, mixinDisableRipple, } from '@angular/material/core';
|
|
import { Subject } from 'rxjs';
|
|
import { DOCUMENT } from '@angular/common';
|
|
import { MAT_MENU_PANEL } from './menu-panel';
|
|
// Boilerplate for applying mixins to MatMenuItem.
|
|
/** @docs-private */
|
|
import * as ɵngcc0 from '@angular/core';
|
|
import * as ɵngcc1 from '@angular/cdk/a11y';
|
|
import * as ɵngcc2 from '@angular/material/core';
|
|
import * as ɵngcc3 from '@angular/common';
|
|
|
|
const _c0 = ["mat-menu-item", ""];
|
|
function MatMenuItem__svg_svg_2_Template(rf, ctx) { if (rf & 1) {
|
|
ɵngcc0.ɵɵnamespaceSVG();
|
|
ɵngcc0.ɵɵelementStart(0, "svg", 2);
|
|
ɵngcc0.ɵɵelement(1, "polygon", 3);
|
|
ɵngcc0.ɵɵelementEnd();
|
|
} }
|
|
const _c1 = ["*"];
|
|
const _MatMenuItemBase = mixinDisableRipple(mixinDisabled(class {
|
|
}));
|
|
/**
|
|
* Single item inside of a `mat-menu`. Provides the menu item styling and accessibility treatment.
|
|
*/
|
|
export class MatMenuItem extends _MatMenuItemBase {
|
|
constructor(_elementRef,
|
|
/**
|
|
* @deprecated `_document` parameter is no longer being used and will be removed.
|
|
* @breaking-change 12.0.0
|
|
*/
|
|
_document, _focusMonitor, _parentMenu,
|
|
/**
|
|
* @deprecated `_changeDetectorRef` to become a required parameter.
|
|
* @breaking-change 14.0.0
|
|
*/
|
|
_changeDetectorRef) {
|
|
// @breaking-change 8.0.0 make `_focusMonitor` and `document` required params.
|
|
super();
|
|
this._elementRef = _elementRef;
|
|
this._focusMonitor = _focusMonitor;
|
|
this._parentMenu = _parentMenu;
|
|
this._changeDetectorRef = _changeDetectorRef;
|
|
/** ARIA role for the menu item. */
|
|
this.role = 'menuitem';
|
|
/** Stream that emits when the menu item is hovered. */
|
|
this._hovered = new Subject();
|
|
/** Stream that emits when the menu item is focused. */
|
|
this._focused = new Subject();
|
|
/** Whether the menu item is highlighted. */
|
|
this._highlighted = false;
|
|
/** Whether the menu item acts as a trigger for a sub-menu. */
|
|
this._triggersSubmenu = false;
|
|
if (_parentMenu && _parentMenu.addItem) {
|
|
_parentMenu.addItem(this);
|
|
}
|
|
}
|
|
/** Focuses the menu item. */
|
|
focus(origin, options) {
|
|
if (this._focusMonitor && origin) {
|
|
this._focusMonitor.focusVia(this._getHostElement(), origin, options);
|
|
}
|
|
else {
|
|
this._getHostElement().focus(options);
|
|
}
|
|
this._focused.next(this);
|
|
}
|
|
ngAfterViewInit() {
|
|
if (this._focusMonitor) {
|
|
// Start monitoring the element so it gets the appropriate focused classes. We want
|
|
// to show the focus style for menu items only when the focus was not caused by a
|
|
// mouse or touch interaction.
|
|
this._focusMonitor.monitor(this._elementRef, false);
|
|
}
|
|
}
|
|
ngOnDestroy() {
|
|
if (this._focusMonitor) {
|
|
this._focusMonitor.stopMonitoring(this._elementRef);
|
|
}
|
|
if (this._parentMenu && this._parentMenu.removeItem) {
|
|
this._parentMenu.removeItem(this);
|
|
}
|
|
this._hovered.complete();
|
|
this._focused.complete();
|
|
}
|
|
/** Used to set the `tabindex`. */
|
|
_getTabIndex() {
|
|
return this.disabled ? '-1' : '0';
|
|
}
|
|
/** Returns the host DOM element. */
|
|
_getHostElement() {
|
|
return this._elementRef.nativeElement;
|
|
}
|
|
/** Prevents the default element actions if it is disabled. */
|
|
// We have to use a `HostListener` here in order to support both Ivy and ViewEngine.
|
|
// In Ivy the `host` bindings will be merged when this class is extended, whereas in
|
|
// ViewEngine they're overwritten.
|
|
// TODO(crisbeto): we move this back into `host` once Ivy is turned on by default.
|
|
// tslint:disable-next-line:no-host-decorator-in-concrete
|
|
_checkDisabled(event) {
|
|
if (this.disabled) {
|
|
event.preventDefault();
|
|
event.stopPropagation();
|
|
}
|
|
}
|
|
/** Emits to the hover stream. */
|
|
// We have to use a `HostListener` here in order to support both Ivy and ViewEngine.
|
|
// In Ivy the `host` bindings will be merged when this class is extended, whereas in
|
|
// ViewEngine they're overwritten.
|
|
// TODO(crisbeto): we move this back into `host` once Ivy is turned on by default.
|
|
// tslint:disable-next-line:no-host-decorator-in-concrete
|
|
_handleMouseEnter() {
|
|
this._hovered.next(this);
|
|
}
|
|
/** Gets the label to be used when determining whether the option should be focused. */
|
|
getLabel() {
|
|
var _a, _b;
|
|
const clone = this._elementRef.nativeElement.cloneNode(true);
|
|
const icons = clone.querySelectorAll('mat-icon, .material-icons');
|
|
// Strip away icons so they don't show up in the text.
|
|
for (let i = 0; i < icons.length; i++) {
|
|
const icon = icons[i];
|
|
(_a = icon.parentNode) === null || _a === void 0 ? void 0 : _a.removeChild(icon);
|
|
}
|
|
return ((_b = clone.textContent) === null || _b === void 0 ? void 0 : _b.trim()) || '';
|
|
}
|
|
_setHighlighted(isHighlighted) {
|
|
var _a;
|
|
// We need to mark this for check for the case where the content is coming from a
|
|
// `matMenuContent` whose change detection tree is at the declaration position,
|
|
// not the insertion position. See #23175.
|
|
// @breaking-change 14.0.0 Remove null check for `_changeDetectorRef`.
|
|
this._highlighted = isHighlighted;
|
|
(_a = this._changeDetectorRef) === null || _a === void 0 ? void 0 : _a.markForCheck();
|
|
}
|
|
}
|
|
MatMenuItem.ɵfac = function MatMenuItem_Factory(t) { return new (t || MatMenuItem)(ɵngcc0.ɵɵdirectiveInject(ɵngcc0.ElementRef), ɵngcc0.ɵɵdirectiveInject(DOCUMENT), ɵngcc0.ɵɵdirectiveInject(ɵngcc1.FocusMonitor), ɵngcc0.ɵɵdirectiveInject(MAT_MENU_PANEL, 8), ɵngcc0.ɵɵdirectiveInject(ɵngcc0.ChangeDetectorRef)); };
|
|
MatMenuItem.ɵcmp = /*@__PURE__*/ ɵngcc0.ɵɵdefineComponent({ type: MatMenuItem, selectors: [["", "mat-menu-item", ""]], hostAttrs: [1, "mat-focus-indicator"], hostVars: 10, hostBindings: function MatMenuItem_HostBindings(rf, ctx) { if (rf & 1) {
|
|
ɵngcc0.ɵɵlistener("click", function MatMenuItem_click_HostBindingHandler($event) { return ctx._checkDisabled($event); })("mouseenter", function MatMenuItem_mouseenter_HostBindingHandler() { return ctx._handleMouseEnter(); });
|
|
} if (rf & 2) {
|
|
ɵngcc0.ɵɵattribute("role", ctx.role)("tabindex", ctx._getTabIndex())("aria-disabled", ctx.disabled.toString())("disabled", ctx.disabled || null);
|
|
ɵngcc0.ɵɵclassProp("mat-menu-item", true)("mat-menu-item-highlighted", ctx._highlighted)("mat-menu-item-submenu-trigger", ctx._triggersSubmenu);
|
|
} }, inputs: { disabled: "disabled", disableRipple: "disableRipple", role: "role" }, exportAs: ["matMenuItem"], features: [ɵngcc0.ɵɵInheritDefinitionFeature], attrs: _c0, ngContentSelectors: _c1, decls: 3, vars: 3, consts: [["matRipple", "", 1, "mat-menu-ripple", 3, "matRippleDisabled", "matRippleTrigger"], ["class", "mat-menu-submenu-icon", "viewBox", "0 0 5 10", "focusable", "false", 4, "ngIf"], ["viewBox", "0 0 5 10", "focusable", "false", 1, "mat-menu-submenu-icon"], ["points", "0,0 5,5 0,10"]], template: function MatMenuItem_Template(rf, ctx) { if (rf & 1) {
|
|
ɵngcc0.ɵɵprojectionDef();
|
|
ɵngcc0.ɵɵprojection(0);
|
|
ɵngcc0.ɵɵelement(1, "div", 0);
|
|
ɵngcc0.ɵɵtemplate(2, MatMenuItem__svg_svg_2_Template, 2, 0, "svg", 1);
|
|
} if (rf & 2) {
|
|
ɵngcc0.ɵɵadvance(1);
|
|
ɵngcc0.ɵɵproperty("matRippleDisabled", ctx.disableRipple || ctx.disabled)("matRippleTrigger", ctx._getHostElement());
|
|
ɵngcc0.ɵɵadvance(1);
|
|
ɵngcc0.ɵɵproperty("ngIf", ctx._triggersSubmenu);
|
|
} }, directives: [ɵngcc2.MatRipple, ɵngcc3.NgIf], encapsulation: 2, changeDetection: 0 });
|
|
MatMenuItem.ctorParameters = () => [
|
|
{ type: ElementRef },
|
|
{ type: undefined, decorators: [{ type: Inject, args: [DOCUMENT,] }] },
|
|
{ type: FocusMonitor },
|
|
{ type: undefined, decorators: [{ type: Inject, args: [MAT_MENU_PANEL,] }, { type: Optional }] },
|
|
{ type: ChangeDetectorRef }
|
|
];
|
|
MatMenuItem.propDecorators = {
|
|
role: [{ type: Input }],
|
|
_checkDisabled: [{ type: HostListener, args: ['click', ['$event'],] }],
|
|
_handleMouseEnter: [{ type: HostListener, args: ['mouseenter',] }]
|
|
};
|
|
(function () { (typeof ngDevMode === "undefined" || ngDevMode) && ɵngcc0.ɵsetClassMetadata(MatMenuItem, [{
|
|
type: Component,
|
|
args: [{
|
|
selector: '[mat-menu-item]',
|
|
exportAs: 'matMenuItem',
|
|
inputs: ['disabled', 'disableRipple'],
|
|
host: {
|
|
'[attr.role]': 'role',
|
|
'[class.mat-menu-item]': 'true',
|
|
'[class.mat-menu-item-highlighted]': '_highlighted',
|
|
'[class.mat-menu-item-submenu-trigger]': '_triggersSubmenu',
|
|
'[attr.tabindex]': '_getTabIndex()',
|
|
'[attr.aria-disabled]': 'disabled.toString()',
|
|
'[attr.disabled]': 'disabled || null',
|
|
'class': 'mat-focus-indicator'
|
|
},
|
|
changeDetection: ChangeDetectionStrategy.OnPush,
|
|
encapsulation: ViewEncapsulation.None,
|
|
template: "<ng-content></ng-content>\n<div class=\"mat-menu-ripple\" matRipple\n [matRippleDisabled]=\"disableRipple || disabled\"\n [matRippleTrigger]=\"_getHostElement()\">\n</div>\n\n<svg\n *ngIf=\"_triggersSubmenu\"\n class=\"mat-menu-submenu-icon\"\n viewBox=\"0 0 5 10\"\n focusable=\"false\"><polygon points=\"0,0 5,5 0,10\"/></svg>\n"
|
|
}]
|
|
}], function () { return [{ type: ɵngcc0.ElementRef }, { type: undefined, decorators: [{
|
|
type: Inject,
|
|
args: [DOCUMENT]
|
|
}] }, { type: ɵngcc1.FocusMonitor }, { type: undefined, decorators: [{
|
|
type: Inject,
|
|
args: [MAT_MENU_PANEL]
|
|
}, {
|
|
type: Optional
|
|
}] }, { type: ɵngcc0.ChangeDetectorRef }]; }, { role: [{
|
|
type: Input
|
|
}],
|
|
/** Prevents the default element actions if it is disabled. */
|
|
// We have to use a `HostListener` here in order to support both Ivy and ViewEngine.
|
|
// In Ivy the `host` bindings will be merged when this class is extended, whereas in
|
|
// ViewEngine they're overwritten.
|
|
// TODO(crisbeto): we move this back into `host` once Ivy is turned on by default.
|
|
// tslint:disable-next-line:no-host-decorator-in-concrete
|
|
_checkDisabled: [{
|
|
type: HostListener,
|
|
args: ['click', ['$event']]
|
|
}],
|
|
/** Emits to the hover stream. */
|
|
// We have to use a `HostListener` here in order to support both Ivy and ViewEngine.
|
|
// In Ivy the `host` bindings will be merged when this class is extended, whereas in
|
|
// ViewEngine they're overwritten.
|
|
// TODO(crisbeto): we move this back into `host` once Ivy is turned on by default.
|
|
// tslint:disable-next-line:no-host-decorator-in-concrete
|
|
_handleMouseEnter: [{
|
|
type: HostListener,
|
|
args: ['mouseenter']
|
|
}] }); })();
|
|
|
|
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"menu-item.js","sources":["../../../../../../src/material/menu/menu-item.ts"],"names":[],"mappings":"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","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 {FocusableOption, FocusMonitor, FocusOrigin} from '@angular/cdk/a11y';\nimport {BooleanInput} from '@angular/cdk/coercion';\nimport {\n  ChangeDetectionStrategy,\n  Component,\n  ElementRef,\n  OnDestroy,\n  ViewEncapsulation,\n  Inject,\n  Optional,\n  Input,\n  HostListener,\n  AfterViewInit,\n  ChangeDetectorRef,\n} from '@angular/core';\nimport {\n  CanDisable,\n  CanDisableRipple,\n  mixinDisabled,\n  mixinDisableRipple,\n} from '@angular/material/core';\nimport {Subject} from 'rxjs';\nimport {DOCUMENT} from '@angular/common';\nimport {MAT_MENU_PANEL, MatMenuPanel} from './menu-panel';\n\n// Boilerplate for applying mixins to MatMenuItem.\n/** @docs-private */\nconst _MatMenuItemBase = mixinDisableRipple(mixinDisabled(class {}));\n\n/**\n * Single item inside of a `mat-menu`. Provides the menu item styling and accessibility treatment.\n */\n@Component({\n  selector: '[mat-menu-item]',\n  exportAs: 'matMenuItem',\n  inputs: ['disabled', 'disableRipple'],\n  host: {\n    '[attr.role]': 'role',\n    '[class.mat-menu-item]': 'true',\n    '[class.mat-menu-item-highlighted]': '_highlighted',\n    '[class.mat-menu-item-submenu-trigger]': '_triggersSubmenu',\n    '[attr.tabindex]': '_getTabIndex()',\n    '[attr.aria-disabled]': 'disabled.toString()',\n    '[attr.disabled]': 'disabled || null',\n    'class': 'mat-focus-indicator',\n  },\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  encapsulation: ViewEncapsulation.None,\n  templateUrl: 'menu-item.html',\n})\nexport class MatMenuItem extends _MatMenuItemBase\n    implements FocusableOption, CanDisable, CanDisableRipple, AfterViewInit, OnDestroy {\n\n  /** ARIA role for the menu item. */\n  @Input() role: 'menuitem' | 'menuitemradio' | 'menuitemcheckbox' = 'menuitem';\n\n  /** Stream that emits when the menu item is hovered. */\n  readonly _hovered: Subject<MatMenuItem> = new Subject<MatMenuItem>();\n\n  /** Stream that emits when the menu item is focused. */\n  readonly _focused = new Subject<MatMenuItem>();\n\n  /** Whether the menu item is highlighted. */\n  _highlighted: boolean = false;\n\n  /** Whether the menu item acts as a trigger for a sub-menu. */\n  _triggersSubmenu: boolean = false;\n\n  constructor(\n    private _elementRef: ElementRef<HTMLElement>,\n    /**\n     * @deprecated `_document` parameter is no longer being used and will be removed.\n     * @breaking-change 12.0.0\n     */\n    @Inject(DOCUMENT) _document?: any,\n    private _focusMonitor?: FocusMonitor,\n    @Inject(MAT_MENU_PANEL) @Optional() public _parentMenu?: MatMenuPanel<MatMenuItem>,\n    /**\n     * @deprecated `_changeDetectorRef` to become a required parameter.\n     * @breaking-change 14.0.0\n     */\n    private _changeDetectorRef?: ChangeDetectorRef) {\n\n    // @breaking-change 8.0.0 make `_focusMonitor` and `document` required params.\n    super();\n\n    if (_parentMenu && _parentMenu.addItem) {\n      _parentMenu.addItem(this);\n    }\n  }\n\n  /** Focuses the menu item. */\n  focus(origin?: FocusOrigin, options?: FocusOptions): void {\n    if (this._focusMonitor && origin) {\n      this._focusMonitor.focusVia(this._getHostElement(), origin, options);\n    } else {\n      this._getHostElement().focus(options);\n    }\n\n    this._focused.next(this);\n  }\n\n  ngAfterViewInit() {\n    if (this._focusMonitor) {\n      // Start monitoring the element so it gets the appropriate focused classes. We want\n      // to show the focus style for menu items only when the focus was not caused by a\n      // mouse or touch interaction.\n      this._focusMonitor.monitor(this._elementRef, false);\n    }\n  }\n\n  ngOnDestroy() {\n    if (this._focusMonitor) {\n      this._focusMonitor.stopMonitoring(this._elementRef);\n    }\n\n    if (this._parentMenu && this._parentMenu.removeItem) {\n      this._parentMenu.removeItem(this);\n    }\n\n    this._hovered.complete();\n    this._focused.complete();\n  }\n\n  /** Used to set the `tabindex`. */\n  _getTabIndex(): string {\n    return this.disabled ? '-1' : '0';\n  }\n\n  /** Returns the host DOM element. */\n  _getHostElement(): HTMLElement {\n    return this._elementRef.nativeElement;\n  }\n\n  /** Prevents the default element actions if it is disabled. */\n  // We have to use a `HostListener` here in order to support both Ivy and ViewEngine.\n  // In Ivy the `host` bindings will be merged when this class is extended, whereas in\n  // ViewEngine they're overwritten.\n  // TODO(crisbeto): we move this back into `host` once Ivy is turned on by default.\n  // tslint:disable-next-line:no-host-decorator-in-concrete\n  @HostListener('click', ['$event'])\n  _checkDisabled(event: Event): void {\n    if (this.disabled) {\n      event.preventDefault();\n      event.stopPropagation();\n    }\n  }\n\n  /** Emits to the hover stream. */\n  // We have to use a `HostListener` here in order to support both Ivy and ViewEngine.\n  // In Ivy the `host` bindings will be merged when this class is extended, whereas in\n  // ViewEngine they're overwritten.\n  // TODO(crisbeto): we move this back into `host` once Ivy is turned on by default.\n  // tslint:disable-next-line:no-host-decorator-in-concrete\n  @HostListener('mouseenter')\n  _handleMouseEnter() {\n    this._hovered.next(this);\n  }\n\n  /** Gets the label to be used when determining whether the option should be focused. */\n  getLabel(): string {\n    const clone = this._elementRef.nativeElement.cloneNode(true) as HTMLElement;\n    const icons = clone.querySelectorAll('mat-icon, .material-icons');\n\n    // Strip away icons so they don't show up in the text.\n    for (let i = 0; i < icons.length; i++) {\n      const icon = icons[i];\n      icon.parentNode?.removeChild(icon);\n    }\n\n    return clone.textContent?.trim() || '';\n  }\n\n  _setHighlighted(isHighlighted: boolean) {\n    // We need to mark this for check for the case where the content is coming from a\n    // `matMenuContent` whose change detection tree is at the declaration position,\n    // not the insertion position. See #23175.\n    // @breaking-change 14.0.0 Remove null check for `_changeDetectorRef`.\n    this._highlighted = isHighlighted;\n    this._changeDetectorRef?.markForCheck();\n  }\n\n  static ngAcceptInputType_disabled: BooleanInput;\n  static ngAcceptInputType_disableRipple: BooleanInput;\n}\n"]}
|