@luminela/menu
Version:
A simple menubar component for Angular.
133 lines • 20.1 kB
JavaScript
import { Component, ContentChildren, ElementRef, HostBinding, HostListener, Input } from "@angular/core";
import { MenuComponent } from "../menu/menu.component";
import { ActiveDescendantKeyManager } from "@angular/cdk/a11y";
export class MenuBarComponent {
constructor(hostElementRef) {
this.hostElementRef = hostElementRef;
this.menuThemeClass = "theme-light";
this.menuTheme = "light";
this.menuClass = "";
}
set theme(theme) {
this.menuTheme = theme;
this.menuThemeClass = theme === "dark" ? "theme-dark" : "theme-light";
}
get lt() {
return this.menuThemeClass === "theme-light";
}
get dt() {
return this.menuThemeClass === "theme-dark";
}
ngOnInit() {
}
ngAfterContentInit() {
console.log(this.menuList);
}
ngAfterViewInit() {
this.keyManager = new ActiveDescendantKeyManager(this.menuList.toArray()).withWrap(true).skipPredicate(m => m.disabled);
}
onMenuChange(data) {
this.menuChangeData = data;
}
onMenuClose(data) {
this.keyManager.setActiveItem(null);
this.previousMenuData = null;
if (!!this.currentMenu) {
this.currentMenu.focused = false;
this.currentMenu = null;
}
this.menuChangeData = null;
}
onMenuClick(event, menu) {
var _a;
if (this.previousMenuElement === event.target.closest(`[data-uid='${menu.uid}']`)) {
menu.focused = !menu.focused;
if (!menu.focused) {
(_a = this.previousMenuData) === null || _a === void 0 ? void 0 : _a.contextMenuRef.popupRef.close();
}
return;
}
this.previousMenuElement = event.target.closest("li").firstChild;
this.keyManager.setActiveItem(this.menuList.find(m => m.uid === menu.uid));
this.currentMenu = menu;
}
onMenuMouseEnter(event, menu) {
var _a;
if (this.previousMenuData) {
if (this.previousMenuElement === event.target.closest(`[data-uid='${menu.uid}']`)) {
return;
}
(_a = this.previousMenuData) === null || _a === void 0 ? void 0 : _a.contextMenuRef.popupRef.close();
event.target.dispatchEvent(new MouseEvent("click", Object.assign({}, event)));
this.keyManager.setActiveItem(this.menuList.find(m => m.uid === menu.uid));
}
this.previousMenuElement = event.target.closest("li").firstChild;
this.currentMenu = menu;
}
onMenuMouseLeave(event, menu) {
// this.currentMenu = null;
// menu.focused = false;
// this.keyManager.setActiveItem(null);
}
onMenuOpen(data) {
if (data.depth > 0) {
return;
}
this.previousMenuData = data;
}
onKeydown(event) {
var _a, _b, _c;
if (!this.previousMenuData) {
return;
}
switch (event.key) {
case "Enter":
case "NumEnter":
case " ":
this.keyManager.setActiveItem(null);
return;
case "ArrowUp":
case "ArrowDown":
console.log("Ignore " + event.key);
return;
case "ArrowLeft":
if (((_a = this.menuChangeData) === null || _a === void 0 ? void 0 : _a.depth) > 1) {
return;
}
this.keyManager.setPreviousItemActive();
break;
case "ArrowRight":
if (((_c = (_b = this.menuChangeData) === null || _b === void 0 ? void 0 : _b.item.menuItems) === null || _c === void 0 ? void 0 : _c.length) > 0) {
return;
}
this.keyManager.setNextItemActive();
break;
default:
this.keyManager.onKeydown(event);
break;
}
if (this.keyManager.activeItem) {
this.currentMenu = this.keyManager.activeItem;
document.querySelector(`[data-uid='${this.currentMenu.uid}']`).dispatchEvent(new MouseEvent("mouseenter"));
}
}
}
MenuBarComponent.decorators = [
{ type: Component, args: [{
selector: "yui-menu-bar",
template: "<div class=\"wrapper\">\n <ul class=\"menubar-list\" *ngIf=\"menuList?.length > 0\">\n <li *ngFor=\"let item of menuList\" #listElement (mouseenter)=\"onMenuMouseEnter($event, item)\"\n (mouseleave)=\"onMenuMouseLeave($event, item)\" (click)=\"onMenuClick($event, item)\"\n [attr.data-uid]=\"item.uid\"\n [ngClass]=\"{'active': item.focused && !item.disabled, 'disabled': item.disabled}\">\n <span *ngIf=\"!item.textTemplate\">{{item.text}}</span>\n <ng-container [ngTemplateOutlet]=\"item.textTemplate\" *ngIf=\"!!item.textTemplate\"></ng-container>\n <yui-contextmenu [target]=\"listElement\" [menuItems]=\"item.menuItems\" [theme]=\"menuTheme\" [menuClass]=\"menuClass\"\n [precise]=\"false\" trigger=\"click\" (menuOpen)=\"onMenuOpen($event)\"\n (menuClose)=\"onMenuClose($event)\" (menuChange)=\"onMenuChange($event)\"></yui-contextmenu>\n </li>\n </ul>\n</div>\n",
styles: ["div.wrapper{height:25px;width:100%}div.wrapper,div.wrapper *{-moz-user-select:none;-ms-user-select:none;-webkit-user-select:none;box-sizing:border-box;user-select:none}ul.menubar-list{justify-content:flex-start;width:100%}ul.menubar-list,ul.menubar-list>li{align-items:center;display:flex;height:100%}ul.menubar-list>li{border-left:1px solid transparent;border-right:1px solid transparent;font-size:13px;font-weight:600;justify-content:center;padding:0 10px}ul.menubar-list>li.disabled{pointer-events:none}:host-context(.theme-light) ul.menubar-list{background:#e8e8e8;border-bottom:1px solid #d1d1d1}:host-context(.theme-light) ul.menubar-list>li{color:#383848}:host-context(.theme-light) ul.menubar-list>li.active,:host-context(.theme-light) ul.menubar-list>li:hover:not(.disabled){background:#f2f2f4}:host-context(.theme-light) ul.menubar-list>li.active:not(:first-child),:host-context(.theme-light) ul.menubar-list>li:hover:not(.disabled):not(:first-child){border-left:1px solid #d1d1d1}:host-context(.theme-light) ul.menubar-list>li.disabled{color:#a7a7af}:host-context(.theme-dark) ul.menubar-list{background:#383838;border-bottom:1px solid #585858}:host-context(.theme-dark) ul.menubar-list>li{color:#c0c0c6}:host-context(.theme-dark) ul.menubar-list>li.active,:host-context(.theme-dark) ul.menubar-list>li:hover:not(.disabled){background:#47474f}:host-context(.theme-dark) ul.menubar-list>li.active:not(:first-child),:host-context(.theme-dark) ul.menubar-list>li:hover:not(.disabled):not(:first-child){border-left:1px solid #585858}:host-context(.theme-dark) ul.menubar-list>li.disabled{color:#6a6a70}"]
},] }
];
MenuBarComponent.ctorParameters = () => [
{ type: ElementRef }
];
MenuBarComponent.propDecorators = {
menuList: [{ type: ContentChildren, args: [MenuComponent,] }],
menuClass: [{ type: Input }],
theme: [{ type: Input }],
lt: [{ type: HostBinding, args: ["class.theme-light",] }],
dt: [{ type: HostBinding, args: ["class.theme-dark",] }],
onKeydown: [{ type: HostListener, args: ["window:keydown", ["$event"],] }]
};
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"menu-bar.component.js","sourceRoot":"/home/runner/work/yui-menu/yui-menu/projects/yui-menu/src/","sources":["lib/components/menu-bar/menu-bar.component.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n    AfterContentInit,\n    AfterViewInit,\n    Component,\n    ContentChildren,\n    ElementRef,\n    HostBinding,\n    HostListener,\n    Input,\n    OnInit,\n    QueryList\n} from \"@angular/core\";\nimport {MenuComponent} from \"../menu/menu.component\";\nimport {IContextMenuData, IMenuChangeEvent, IMenuCloseEvent, IMenuOpenEvent} from \"@luminela/contextmenu\";\nimport {ActiveDescendantKeyManager} from \"@angular/cdk/a11y\";\nimport {MenuTheme, MenuThemeClass} from \"../../interfaces/MenuTheme\";\n\n@Component({\n    selector: \"yui-menu-bar\",\n    templateUrl: \"./menu-bar.component.html\",\n    styleUrls: [\"./menu-bar.component.scss\"]\n})\nexport class MenuBarComponent implements OnInit, AfterContentInit, AfterViewInit {\n\n    private keyManager: ActiveDescendantKeyManager<MenuComponent>;\n    private menuThemeClass: MenuThemeClass = \"theme-light\";\n    public currentMenu: MenuComponent;\n    public menuChangeData: IMenuChangeEvent;\n    public menuTheme: MenuTheme = \"light\";\n    public previousMenuData: IContextMenuData;\n    public previousMenuElement: HTMLLIElement;\n    @ContentChildren(MenuComponent) menuList: QueryList<MenuComponent>;\n    @Input() menuClass: string = \"\";\n\n    @Input() set theme(theme: MenuTheme) {\n        this.menuTheme = theme;\n        this.menuThemeClass = theme === \"dark\" ? \"theme-dark\" : \"theme-light\";\n    }\n\n    @HostBinding(\"class.theme-light\") get lt() {\n        return this.menuThemeClass === \"theme-light\";\n    }\n\n    @HostBinding(\"class.theme-dark\") get dt() {\n        return this.menuThemeClass === \"theme-dark\";\n    }\n\n    constructor(\n        private readonly hostElementRef: ElementRef\n    ) {\n    }\n\n    ngOnInit(): void {\n    }\n\n    ngAfterContentInit() {\n        console.log(this.menuList);\n    }\n\n    ngAfterViewInit() {\n        this.keyManager = new ActiveDescendantKeyManager<MenuComponent>(this.menuList.toArray()).withWrap(true).skipPredicate(m => m.disabled);\n    }\n\n    public onMenuChange(data: IMenuChangeEvent): void {\n        this.menuChangeData = data;\n    }\n\n    public onMenuClose(data: IMenuCloseEvent): void {\n        this.keyManager.setActiveItem(null);\n        this.previousMenuData = null;\n        if (!!this.currentMenu) {\n            this.currentMenu.focused = false;\n            this.currentMenu = null;\n        }\n        this.menuChangeData = null;\n    }\n\n    public onMenuClick(event: MouseEvent, menu: MenuComponent): void {\n        if (this.previousMenuElement === (event.target as HTMLElement).closest(`[data-uid='${menu.uid}']`)) {\n            menu.focused = !menu.focused;\n            if (!menu.focused) {\n                this.previousMenuData?.contextMenuRef.popupRef.close();\n            }\n            return;\n        }\n        this.previousMenuElement = (event.target as HTMLElement).closest(\"li\").firstChild as HTMLLIElement;\n        this.keyManager.setActiveItem(this.menuList.find(m => m.uid === menu.uid));\n        this.currentMenu = menu;\n    }\n\n    public onMenuMouseEnter(event: MouseEvent, menu: MenuComponent): void {\n        if (this.previousMenuData) {\n            if (this.previousMenuElement === (event.target as HTMLElement).closest(`[data-uid='${menu.uid}']`)) {\n                return;\n            }\n            this.previousMenuData?.contextMenuRef.popupRef.close();\n            event.target.dispatchEvent(new MouseEvent(\"click\", {...event}));\n            this.keyManager.setActiveItem(this.menuList.find(m => m.uid === menu.uid));\n        }\n        this.previousMenuElement = (event.target as HTMLElement).closest(\"li\").firstChild as HTMLLIElement;\n        this.currentMenu = menu;\n\n    }\n\n    public onMenuMouseLeave(event: MouseEvent, menu: MenuComponent): void {\n        // this.currentMenu = null;\n        // menu.focused = false;\n        // this.keyManager.setActiveItem(null);\n    }\n\n    public onMenuOpen(data: IMenuOpenEvent): void {\n        if (data.depth > 0) {\n            return;\n        }\n        this.previousMenuData = data;\n    }\n\n    @HostListener(\"window:keydown\", [\"$event\"])\n    public onKeydown(event: KeyboardEvent): void {\n        if (!this.previousMenuData) {\n            return;\n        }\n        switch (event.key) {\n            case \"Enter\":\n            case \"NumEnter\":\n            case \" \":\n                this.keyManager.setActiveItem(null);\n                return;\n            case \"ArrowUp\":\n            case \"ArrowDown\":\n                console.log(\"Ignore \" + event.key);\n                return;\n            case \"ArrowLeft\":\n                if (this.menuChangeData?.depth > 1) {\n                    return;\n                }\n                this.keyManager.setPreviousItemActive();\n                break;\n            case \"ArrowRight\":\n                if (this.menuChangeData?.item.menuItems?.length > 0) {\n                    return;\n                }\n                this.keyManager.setNextItemActive();\n                break;\n            default:\n                this.keyManager.onKeydown(event);\n                break;\n        }\n        if (this.keyManager.activeItem) {\n            this.currentMenu = this.keyManager.activeItem;\n            document.querySelector(`[data-uid='${this.currentMenu.uid}']`).dispatchEvent(new MouseEvent(\"mouseenter\"));\n        }\n    }\n}\n"]}