UNPKG

@blox/material

Version:

Material Components for Angular

370 lines 49.1 kB
import { ContentChildren, Directive, ElementRef, EventEmitter, HostBinding, Input, Output, Renderer2, Self, HostListener } from '@angular/core'; import { cssClasses as listCssClasses } from '@material/list'; import { MDCMenuFoundation, cssClasses, strings, DefaultFocusState } from '@material/menu'; import { MdcMenuSurfaceDirective } from '../menu-surface/mdc.menu-surface.directive'; import { MdcListDirective, MdcListFunction } from '../list/mdc.list.directive'; import { Subject } from 'rxjs'; import { takeUntil } from 'rxjs/operators'; // attributes on list-items that we maintain ourselves, so should be ignored // in the adapter: const ANGULAR_ITEM_ATTRIBUTES = [ strings.ARIA_CHECKED_ATTR, strings.ARIA_DISABLED_ATTR ]; // classes on list-items that we maintain ourselves, so should be ignored // in the adapter: const ANGULAR_ITEM_CLASSES = [ listCssClasses.LIST_ITEM_DISABLED_CLASS, cssClasses.MENU_SELECTED_LIST_ITEM ]; export var FocusOnOpen; (function (FocusOnOpen) { FocusOnOpen[FocusOnOpen["first"] = 0] = "first"; FocusOnOpen[FocusOnOpen["last"] = 1] = "last"; FocusOnOpen[FocusOnOpen["root"] = -1] = "root"; })(FocusOnOpen || (FocusOnOpen = {})); ; let nextId = 1; /** * Directive for a spec aligned material design Menu. * This directive should wrap an `mdcList`. The `mdcList` contains the menu items (and possible separators). * * An `mdcMenu` element will also match with the selector of the menu surface directive, documented * <a href="/components/menu-surface#mdcMenuSurface">here: mdcMenuSurface</a>. The * <a href="/components/menu-surface#mdcMenuAnchor">mdcMenuAnchor API</a> is documented on the same page. * * # Accessibility * * * For `role` and `aria-*` attributes on the list, see documentation for `mdcList`. * * The best way to open the menu by user interaction is to use the `mdcMenuTrigger` directive * on the interaction element (e.g. button). This takes care of following ARIA recommended practices * for focusing the correct element, and maintaining proper `aria-*` and `role` attributes on the * interaction element, menu, and list. * * When opening the `mdcMenuSurface` programmatic, the program is responsible for all of this. * (including focusing an element of the menu or the menu itself). * * The `mdcList` will be made focusable by setting a `"tabindex"="-1"` attribute. * * The `mdcList` will get an `aria-orientation=vertical` attribute. * * The `mdcList` will get an `aria-hidden=true` attribute when the menu surface is closed. */ export class MdcMenuDirective { constructor(_elm, rndr, surface) { this._elm = _elm; this.rndr = rndr; this.surface = surface; this.onDestroy$ = new Subject(); this.onListChange$ = new Subject(); /** @internal */ this.itemsChanged = new EventEmitter(); /** @internal */ this.itemValuesChanged = new EventEmitter(); /** @internal */ this._cls = true; this._id = null; this.cachedId = null; this._function = MdcListFunction.menu; this._lastList = null; /** * Event emitted when the user selects a value. The passed object contains a value * (set to the <code>value</code> of the selected list item), and an index * (set to the index of the selected list item). */ this.pick = new EventEmitter(); this.mdcAdapter = { addClassToElementAtIndex: (index, className) => { var _a, _b; // ignore classes we maintain ourselves if (!ANGULAR_ITEM_CLASSES.find(c => c === className)) { const elm = (_b = (_a = this._list) === null || _a === void 0 ? void 0 : _a.getItem(index)) === null || _b === void 0 ? void 0 : _b._elm.nativeElement; if (elm) this.rndr.addClass(elm, className); } }, removeClassFromElementAtIndex: (index, className) => { var _a, _b; // ignore classes we maintain ourselves if (!ANGULAR_ITEM_CLASSES.find(c => c === className)) { const elm = (_b = (_a = this._list) === null || _a === void 0 ? void 0 : _a.getItem(index)) === null || _b === void 0 ? void 0 : _b._elm.nativeElement; if (elm) this.rndr.addClass(elm, className); } }, addAttributeToElementAtIndex: (index, attr, value) => { var _a, _b; // ignore attributes we maintain ourselves if (!ANGULAR_ITEM_ATTRIBUTES.find(a => a === attr)) { const elm = (_b = (_a = this._list) === null || _a === void 0 ? void 0 : _a.getItem(index)) === null || _b === void 0 ? void 0 : _b._elm.nativeElement; if (elm) this.rndr.setAttribute(elm, attr, value); } }, removeAttributeFromElementAtIndex: (index, attr) => { var _a, _b; // ignore attributes we maintain ourselves if (!ANGULAR_ITEM_ATTRIBUTES.find(a => a === attr)) { const elm = (_b = (_a = this._list) === null || _a === void 0 ? void 0 : _a.getItem(index)) === null || _b === void 0 ? void 0 : _b._elm.nativeElement; if (elm) this.rndr.removeAttribute(elm, attr); } }, elementContainsClass: (element, className) => element.classList.contains(className), closeSurface: (skipRestoreFocus) => { if (skipRestoreFocus) this.surface.closeWithoutFocusRestore(); else this.surface.open = false; }, getElementIndex: (element) => { var _a; return (_a = this._list) === null || _a === void 0 ? void 0 : _a._items.toArray().findIndex(i => i._elm.nativeElement === element); }, notifySelected: (evtData) => { this.pick.emit({ index: evtData.index, value: this._list._items.toArray()[evtData.index].value }); }, getMenuItemCount: () => { var _a; return ((_a = this._list) === null || _a === void 0 ? void 0 : _a._items.length) || 0; }, focusItemAtIndex: (index) => { var _a; return (_a = this._list.getItem(index)) === null || _a === void 0 ? void 0 : _a._elm.nativeElement.focus(); }, focusListRoot: () => { var _a; return (_a = this._list) === null || _a === void 0 ? void 0 : _a._elm.nativeElement.focus(); }, getSelectedSiblingOfItemAtIndex: () => -1, isSelectableItemAtIndex: () => false // menuSelectionGroup not yet supported }; this.foundation = null; } ngOnInit() { // Force setter to be called in case id was not specified. this.id = this.id; } ngAfterContentInit() { this._lastList = this._listQuery.first; this._listQuery.changes.subscribe(() => { var _a; if (this._lastList !== this._listQuery.first) { this.onListChange$.next(); (_a = this._lastList) === null || _a === void 0 ? void 0 : _a._setFunction(MdcListFunction.plain); this._lastList = this._listQuery.first; this.destroyFoundation(); if (this._lastList) this.initAll(); } }); this.surface.afterOpened.pipe(takeUntil(this.onDestroy$)).subscribe(() => { var _a, _b; (_a = this.foundation) === null || _a === void 0 ? void 0 : _a.handleMenuSurfaceOpened(); // reset default focus state for programmatic opening of menu; // interactive opening sets the default when the open is triggered // (see openAndFocus) (_b = this.foundation) === null || _b === void 0 ? void 0 : _b.setDefaultFocusState(DefaultFocusState.NONE); }); this.surface.openChange.pipe(takeUntil(this.onDestroy$)).subscribe(() => { if (this._list) this._list._hidden = !this.surface.open; }); if (this._lastList) this.initAll(); } ngOnDestroy() { this.onListChange$.next(); this.onListChange$.complete(); this.onDestroy$.next(); this.onDestroy$.complete(); this.destroyFoundation(); } initAll() { var _a, _b; Promise.resolve().then(() => this._lastList._setFunction(this._function)); this.initFoundation(); this.subscribeItemActions(); (_a = this._lastList) === null || _a === void 0 ? void 0 : _a.itemsChanged.pipe(takeUntil(this.onListChange$)).subscribe(() => this.itemsChanged.emit()); (_b = this._lastList) === null || _b === void 0 ? void 0 : _b.itemValuesChanged.pipe(takeUntil(this.onListChange$)).subscribe(() => this.itemValuesChanged.emit()); } initFoundation() { this.foundation = new MDCMenuFoundation(this.mdcAdapter); this.foundation.init(); // suitable for programmatic opening, program can focus whatever element it wants: this.foundation.setDefaultFocusState(DefaultFocusState.NONE); if (this._list) this._list._hidden = !this.surface.open; } destroyFoundation() { if (this.foundation) { this.foundation.destroy(); this.foundation = null; } } subscribeItemActions() { var _a; (_a = this._lastList) === null || _a === void 0 ? void 0 : _a.itemAction.pipe(takeUntil(this.onListChange$)).subscribe(data => { var _a; (_a = this.foundation) === null || _a === void 0 ? void 0 : _a.handleItemAction(this._list.getItem(data.index)._elm.nativeElement); }); } /** @docs-private */ get id() { return this._id; } set id(value) { this._id = value || this._newId(); } /** @internal */ _newId() { this.cachedId = this.cachedId || `mdc-menu-${nextId++}`; return this.cachedId; } /** @docs-private */ get open() { return this.surface.open; } /** @docs-private */ openAndFocus(focus) { var _a, _b, _c; switch (focus) { case FocusOnOpen.first: (_a = this.foundation) === null || _a === void 0 ? void 0 : _a.setDefaultFocusState(DefaultFocusState.FIRST_ITEM); break; case FocusOnOpen.last: (_b = this.foundation) === null || _b === void 0 ? void 0 : _b.setDefaultFocusState(DefaultFocusState.LAST_ITEM); break; case FocusOnOpen.root: default: (_c = this.foundation) === null || _c === void 0 ? void 0 : _c.setDefaultFocusState(DefaultFocusState.LIST_ROOT); } this.surface.open = true; } /** @internal */ doClose() { this.surface.open = false; } /** @internal */ set _listFunction(val) { this._function = val; if (this._lastList) // otherwise this will happen in ngAfterContentInit this._list._setFunction(val); } /** @internal */ get _list() { return this._listQuery.first; } /** @internal */ _onKeydown(event) { var _a; (_a = this.foundation) === null || _a === void 0 ? void 0 : _a.handleKeydown(event); } } MdcMenuDirective.decorators = [ { type: Directive, args: [{ selector: '[mdcMenu],[mdcSelectMenu]', exportAs: 'mdcMenu' },] } ]; MdcMenuDirective.ctorParameters = () => [ { type: ElementRef }, { type: Renderer2 }, { type: MdcMenuSurfaceDirective, decorators: [{ type: Self }] } ]; MdcMenuDirective.propDecorators = { itemsChanged: [{ type: Output }], itemValuesChanged: [{ type: Output }], _cls: [{ type: HostBinding, args: ['class.mdc-menu',] }], pick: [{ type: Output }], _listQuery: [{ type: ContentChildren, args: [MdcListDirective,] }], id: [{ type: HostBinding }, { type: Input }], _onKeydown: [{ type: HostListener, args: ['keydown', ['$event'],] }] }; /** * * # Accessibility * * * `Enter`, `Space`, and `Down Arrow` keys open the menu and place focus on the first item. * * `Up Arrow` opens the menu and places focus on the last item * * Click/Touch events set focus to the mdcList root element * * * Attribute `role=button` will be set if the element is not already a button element. * * Attribute `aria-haspopup=menu` will be set if an `mdcMenu` is attached. * * Attribute `aria-expanded` will be set while the attached menu is open * * Attribute `aria-controls` will be set to the id of the attached menu. (And a unique id will be generated, * if none was set on the menu). * * `Enter`, `Space`, and `Down-Arrow` will open the menu with the first menu item focused. * * `Up-Arrow` will open the menu with the last menu item focused. * * Mouse/Touch events will open the menu with the list root element focused. The list root element * will handle keyboard navigation once it receives focus. */ export class MdcMenuTriggerDirective { constructor(elm) { /** @internal */ this._role = 'button'; this._mdcMenuTrigger = null; this.down = { enter: false, space: false }; if (elm.nativeElement.nodeName.toUpperCase() === 'BUTTON') this._role = null; } /** @internal */ onClick() { var _a, _b; if (this.down.enter || this.down.space) (_a = this._mdcMenuTrigger) === null || _a === void 0 ? void 0 : _a.openAndFocus(FocusOnOpen.first); else (_b = this._mdcMenuTrigger) === null || _b === void 0 ? void 0 : _b.openAndFocus(FocusOnOpen.root); } /** @internal */ onKeydown(event) { var _a, _b; this.setDown(event, true); const { key, keyCode } = event; if (key === 'ArrowUp' || keyCode === 38) (_a = this._mdcMenuTrigger) === null || _a === void 0 ? void 0 : _a.openAndFocus(FocusOnOpen.last); else if (key === 'ArrowDown' || keyCode === 40) (_b = this._mdcMenuTrigger) === null || _b === void 0 ? void 0 : _b.openAndFocus(FocusOnOpen.first); } /** @internal */ onKeyup(event) { this.setDown(event, false); } /** @internal */ get _hasPopup() { return this._mdcMenuTrigger ? 'menu' : null; } /** @internal */ get _expanded() { var _a; return ((_a = this._mdcMenuTrigger) === null || _a === void 0 ? void 0 : _a.open) ? 'true' : null; } /** @internal */ get _ariaControls() { var _a; return (_a = this._mdcMenuTrigger) === null || _a === void 0 ? void 0 : _a.id; } get mdcMenuTrigger() { return this._mdcMenuTrigger; } set mdcMenuTrigger(value) { if (value && value.openAndFocus) this._mdcMenuTrigger = value; else this._mdcMenuTrigger = null; } setDown(event, isDown) { const { key, keyCode } = event; if (key === 'Enter' || keyCode === 13) this.down.enter = isDown; else if (key === 'Space' || keyCode === 32) this.down.space = isDown; } } MdcMenuTriggerDirective.decorators = [ { type: Directive, args: [{ selector: '[mdcMenuTrigger]', },] } ]; MdcMenuTriggerDirective.ctorParameters = () => [ { type: ElementRef } ]; MdcMenuTriggerDirective.propDecorators = { _role: [{ type: HostBinding, args: ['attr.role',] }], onClick: [{ type: HostListener, args: ['click',] }], onKeydown: [{ type: HostListener, args: ['keydown', ['$event'],] }], onKeyup: [{ type: HostListener, args: ['keyup', ['$event'],] }], _hasPopup: [{ type: HostBinding, args: ['attr.aria-haspopup',] }], _expanded: [{ type: HostBinding, args: ['attr.aria-expanded',] }], _ariaControls: [{ type: HostBinding, args: ['attr.aria-controls',] }], mdcMenuTrigger: [{ type: Input }] }; export const MENU_DIRECTIVES = [ MdcMenuDirective, MdcMenuTriggerDirective ]; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"mdc.menu.directive.js","sourceRoot":"","sources":["../../../../src/components/menu/mdc.menu.directive.ts"],"names":[],"mappings":"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","sourcesContent":["import { AfterContentInit, ContentChildren, Directive, ElementRef, EventEmitter, HostBinding,\n  Input, OnDestroy, Output, QueryList, Renderer2, Self, HostListener, OnInit } from '@angular/core';\nimport { cssClasses as listCssClasses } from '@material/list';\nimport { MDCMenuFoundation, MDCMenuAdapter, cssClasses, strings, DefaultFocusState } from '@material/menu';\nimport { MdcMenuSurfaceDirective } from '../menu-surface/mdc.menu-surface.directive';\nimport { MdcListDirective, MdcListFunction } from '../list/mdc.list.directive';\nimport { Subject } from 'rxjs';\nimport { takeUntil } from 'rxjs/operators';\n\n/**\n * Data send by the <code>pick</code> event of <code>MdcMenuDirective</code>.\n */\nexport interface MdcMenuSelection {\n    /**\n     * The <code>value</code> of the selected menu item (<code>MdcListItemDirective</code>).\n     */\n    value: any,\n    /**\n     * The index of the selected menu item (<code>MdcListItemDirective</code>).\n     */\n    index: number\n}\n\n// attributes on list-items that we maintain ourselves, so should be ignored\n// in the adapter:\nconst ANGULAR_ITEM_ATTRIBUTES = [\n    strings.ARIA_CHECKED_ATTR, strings.ARIA_DISABLED_ATTR\n];\n// classes on list-items that we maintain ourselves, so should be ignored\n// in the adapter:\nconst ANGULAR_ITEM_CLASSES = [\n    listCssClasses.LIST_ITEM_DISABLED_CLASS, cssClasses.MENU_SELECTED_LIST_ITEM\n];\n\nexport enum FocusOnOpen {first = 0, last = 1, root = -1};\nlet nextId = 1;\n\n/**\n * Directive for a spec aligned material design Menu.\n * This directive should wrap an `mdcList`. The `mdcList` contains the menu items (and possible separators).\n * \n * An `mdcMenu` element will also match with the selector of the menu surface directive, documented\n * <a href=\"/components/menu-surface#mdcMenuSurface\">here: mdcMenuSurface</a>. The\n * <a href=\"/components/menu-surface#mdcMenuAnchor\">mdcMenuAnchor API</a> is documented on the same page.\n * \n * # Accessibility\n * \n * * For `role` and `aria-*` attributes on the list, see documentation for `mdcList`.\n * * The best way to open the menu by user interaction is to use the `mdcMenuTrigger` directive\n *   on the interaction element (e.g. button). This takes care of following ARIA recommended practices\n *   for focusing the correct element, and maintaining proper `aria-*` and `role` attributes on the\n *   interaction element, menu, and list.\n * * When opening the `mdcMenuSurface` programmatic, the program is responsible for all of this.\n *   (including focusing an element of the menu or the menu itself).\n * * The `mdcList` will be made focusable by setting a `\"tabindex\"=\"-1\"` attribute.\n * * The `mdcList` will get an `aria-orientation=vertical` attribute.\n * * The `mdcList` will get an `aria-hidden=true` attribute when the menu surface is closed.\n */\n@Directive({\n    selector: '[mdcMenu],[mdcSelectMenu]',\n    exportAs: 'mdcMenu'\n})\nexport class MdcMenuDirective implements AfterContentInit, OnInit, OnDestroy {\n    private onDestroy$: Subject<any> = new Subject();\n    private onListChange$: Subject<any> = new Subject();\n    /** @internal */\n    @Output() readonly itemsChanged: EventEmitter<void> = new EventEmitter();\n    /** @internal */\n    @Output() readonly itemValuesChanged: EventEmitter<void> = new EventEmitter();\n    /** @internal */\n    @HostBinding('class.mdc-menu') readonly _cls = true;\n    private _id: string | null = null;\n    private cachedId: string | null = null;\n    private _function = MdcListFunction.menu;\n    private _lastList: MdcListDirective | null= null;\n\n    /**\n     * Event emitted when the user selects a value. The passed object contains a value\n     * (set to the <code>value</code> of the selected list item), and an index\n     * (set to the index of the selected list item).\n     */\n    @Output() readonly pick: EventEmitter<MdcMenuSelection> = new EventEmitter();\n    /** @internal */\n    @ContentChildren(MdcListDirective) _listQuery?: QueryList<MdcListDirective>;\n    private mdcAdapter: MDCMenuAdapter = {\n        addClassToElementAtIndex: (index, className) => {\n            // ignore classes we maintain ourselves\n            if (!ANGULAR_ITEM_CLASSES.find(c => c === className)) {\n                const elm = this._list?.getItem(index)?._elm.nativeElement;\n                if (elm)\n                    this.rndr.addClass(elm, className);\n            }\n        },\n        removeClassFromElementAtIndex: (index, className) => {\n            // ignore classes we maintain ourselves\n            if (!ANGULAR_ITEM_CLASSES.find(c => c === className)) {\n                const elm = this._list?.getItem(index)?._elm.nativeElement;\n                if (elm)\n                    this.rndr.addClass(elm, className);\n            }\n        },\n        addAttributeToElementAtIndex: (index, attr, value) => {\n            // ignore attributes we maintain ourselves\n            if (!ANGULAR_ITEM_ATTRIBUTES.find(a => a === attr)) {\n                const elm = this._list?.getItem(index)?._elm.nativeElement;\n                if (elm)\n                    this.rndr.setAttribute(elm, attr, value);\n            }\n        },\n        removeAttributeFromElementAtIndex: (index, attr) => {\n            // ignore attributes we maintain ourselves\n            if (!ANGULAR_ITEM_ATTRIBUTES.find(a => a === attr)) {\n                const elm = this._list?.getItem(index)?._elm.nativeElement;\n                if (elm)\n                    this.rndr.removeAttribute(elm, attr);\n            }\n        },\n        elementContainsClass: (element, className) => element.classList.contains(className),\n        closeSurface: (skipRestoreFocus) => {\n            if (skipRestoreFocus)\n                this.surface.closeWithoutFocusRestore();\n            else\n                this.surface.open = false;\n        },\n        getElementIndex: (element) => this._list?._items!.toArray().findIndex(i => i._elm.nativeElement === element),\n        notifySelected: (evtData) => {\n            this.pick.emit({index: evtData.index, value: this._list._items!.toArray()[evtData.index].value});\n        },\n        getMenuItemCount: () => this._list?._items!.length || 0,\n        focusItemAtIndex: (index) => this._list.getItem(index)?._elm.nativeElement.focus(),\n        focusListRoot: () => this._list?._elm.nativeElement.focus(),\n        getSelectedSiblingOfItemAtIndex: () => -1, // menuSelectionGroup not yet supported\n        isSelectableItemAtIndex: () => false // menuSelectionGroup not yet supported\n    };\n    private foundation: MDCMenuFoundation | null = null;\n\n    constructor(public _elm: ElementRef, private rndr: Renderer2, @Self() private surface: MdcMenuSurfaceDirective) {\n    }\n\n    ngOnInit() {\n        // Force setter to be called in case id was not specified.\n        this.id = this.id;\n    }\n\n    ngAfterContentInit() {\n        this._lastList = this._listQuery!.first;\n        this._listQuery!.changes.subscribe(() => {\n            if (this._lastList !== this._listQuery!.first) {\n                this.onListChange$.next();\n                this._lastList?._setFunction(MdcListFunction.plain);\n                this._lastList = this._listQuery!.first;\n                this.destroyFoundation();\n                if (this._lastList)\n                    this.initAll();\n            }\n        });\n        this.surface.afterOpened.pipe(takeUntil(this.onDestroy$)).subscribe(() => {\n            this.foundation?.handleMenuSurfaceOpened();\n            // reset default focus state for programmatic opening of menu;\n            // interactive opening sets the default when the open is triggered\n            // (see openAndFocus)\n            this.foundation?.setDefaultFocusState(DefaultFocusState.NONE);\n        });\n        this.surface.openChange.pipe(takeUntil(this.onDestroy$)).subscribe(() => {\n            if (this._list)\n                this._list._hidden = !this.surface.open;\n        });\n        if (this._lastList)\n            this.initAll();\n    }\n\n    ngOnDestroy() {\n        this.onListChange$.next(); this.onListChange$.complete();\n        this.onDestroy$.next(); this.onDestroy$.complete();\n        this.destroyFoundation();\n    }\n\n    private initAll() {\n        Promise.resolve().then(() => this._lastList!._setFunction(this._function));\n        this.initFoundation();\n        this.subscribeItemActions();\n        this._lastList?.itemsChanged.pipe(takeUntil(this.onListChange$)).subscribe(() => this.itemsChanged.emit());\n        this._lastList?.itemValuesChanged.pipe(takeUntil(this.onListChange$)).subscribe(() => this.itemValuesChanged.emit());\n    }\n\n    private initFoundation() {\n        this.foundation = new MDCMenuFoundation(this.mdcAdapter);\n        this.foundation.init();\n        // suitable for programmatic opening, program can focus whatever element it wants:\n        this.foundation.setDefaultFocusState(DefaultFocusState.NONE);\n        if (this._list)\n            this._list._hidden = !this.surface.open;\n    }\n\n    private destroyFoundation() {\n        if (this.foundation) {\n            this.foundation.destroy();\n            this.foundation = null;\n        }\n    }\n\n    private subscribeItemActions() {\n        this._lastList?.itemAction.pipe(takeUntil(this.onListChange$)).subscribe(data => {\n            this.foundation?.handleItemAction(this._list.getItem(data.index)!._elm.nativeElement);\n        });\n    }\n\n    /** @docs-private */\n    @HostBinding()\n    @Input() get id() {\n        return this._id;\n    }\n  \n    set id(value: string | null) {\n        this._id = value || this._newId();\n    }\n\n    /** @internal */\n    _newId(): string {\n        this.cachedId = this.cachedId || `mdc-menu-${nextId++}`;\n        return this.cachedId;\n    }\n\n    /** @docs-private */\n    get open() {\n        return this.surface.open;\n    }\n\n    /** @docs-private */\n    openAndFocus(focus: FocusOnOpen) {\n        switch (focus) {\n            case FocusOnOpen.first:\n                this.foundation?.setDefaultFocusState(DefaultFocusState.FIRST_ITEM);\n                break;\n            case FocusOnOpen.last:\n                this.foundation?.setDefaultFocusState(DefaultFocusState.LAST_ITEM);\n                break;\n            case FocusOnOpen.root:\n            default:\n                this.foundation?.setDefaultFocusState(DefaultFocusState.LIST_ROOT);\n        }\n        this.surface.open = true;\n    }\n\n    /** @internal */\n    doClose() {\n        this.surface.open = false;\n    }\n\n    /** @internal */\n    set _listFunction(val: MdcListFunction) {\n        this._function = val;\n        if (this._lastList) // otherwise this will happen in ngAfterContentInit\n            this._list._setFunction(val);\n    }\n\n    /** @internal */\n    get _list(): MdcListDirective {\n        return this._listQuery!.first;\n    }\n\n    /** @internal */\n    @HostListener('keydown', ['$event']) _onKeydown(event: KeyboardEvent) {\n        this.foundation?.handleKeydown(event);\n    }\n}\n\n/**\n * \n * # Accessibility\n * \n * * `Enter`, `Space`, and `Down Arrow` keys open the menu and place focus on the first item.\n * * `Up Arrow` opens the menu and places focus on the last item\n * * Click/Touch events set focus to the mdcList root element\n * \n * * Attribute `role=button` will be set if the element is not already a button element.\n * * Attribute `aria-haspopup=menu` will be set if an `mdcMenu` is attached.\n * * Attribute `aria-expanded` will be set while the attached menu is open\n * * Attribute `aria-controls` will be set to the id of the attached menu. (And a unique id will be generated,\n *   if none was set on the menu).\n * * `Enter`, `Space`, and `Down-Arrow` will open the menu with the first menu item focused.\n * * `Up-Arrow` will open the menu with the last menu item focused.\n * * Mouse/Touch events will open the menu with the list root element focused. The list root element\n *   will handle keyboard navigation once it receives focus.\n */\n@Directive({\n    selector: '[mdcMenuTrigger]',\n})\nexport class MdcMenuTriggerDirective {\n    /** @internal */\n    @HostBinding('attr.role') _role: string | null = 'button';\n    private _mdcMenuTrigger: MdcMenuDirective | null = null;\n    private down = {\n        enter: false,\n        space: false\n    }\n\n    constructor(elm: ElementRef) {\n        if (elm.nativeElement.nodeName.toUpperCase() === 'BUTTON')\n            this._role = null;\n    }\n\n    /** @internal */\n    @HostListener('click') onClick() {\n        if (this.down.enter || this.down.space)\n            this._mdcMenuTrigger?.openAndFocus(FocusOnOpen.first);\n        else\n            this._mdcMenuTrigger?.openAndFocus(FocusOnOpen.root);\n    }\n\n    /** @internal */\n    @HostListener('keydown', ['$event']) onKeydown(event: KeyboardEvent) {\n        this.setDown(event, true);\n        const {key, keyCode} = event;\n        if (key === 'ArrowUp' || keyCode === 38)\n            this._mdcMenuTrigger?.openAndFocus(FocusOnOpen.last);\n        else if (key === 'ArrowDown' || keyCode === 40)\n            this._mdcMenuTrigger?.openAndFocus(FocusOnOpen.first);\n    }\n\n    /** @internal */\n    @HostListener('keyup', ['$event']) onKeyup(event: KeyboardEvent) {\n        this.setDown(event, false);\n    }\n\n    /** @internal */\n    @HostBinding('attr.aria-haspopup') get _hasPopup() {\n        return this._mdcMenuTrigger ? 'menu' : null;\n    }\n\n    /** @internal */\n    @HostBinding('attr.aria-expanded') get _expanded() {\n        return this._mdcMenuTrigger?.open ? 'true' : null;\n    }\n\n    /** @internal */\n    @HostBinding('attr.aria-controls') get _ariaControls() {\n        return this._mdcMenuTrigger?.id;\n    }\n\n    @Input() get mdcMenuTrigger() {\n        return this._mdcMenuTrigger;\n    }\n\n    set mdcMenuTrigger(value: MdcMenuDirective | null) {\n        if (value && value.openAndFocus)\n            this._mdcMenuTrigger = value;\n        else\n            this._mdcMenuTrigger = null;\n    }\n\n    private setDown(event: KeyboardEvent, isDown: boolean) {\n        const {key, keyCode} = event;\n        if (key === 'Enter' || keyCode === 13)\n            this.down.enter = isDown;\n        else if (key === 'Space' || keyCode === 32)\n            this.down.space = isDown;\n    }\n}\n\nexport const MENU_DIRECTIVES = [\n    MdcMenuDirective, MdcMenuTriggerDirective\n];\n"]}