@ngrdt/menu 0.0.11 → 0.0.14
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +157 -3
- package/esm2022/index.mjs +2 -2
- package/esm2022/lib/menu/rdt-menu.component.mjs +57 -32
- package/esm2022/lib/menu-bar/rdt-menu-bar.component.mjs +13 -6
- package/esm2022/lib/menu-base/rdt-menu-base.component.mjs +106 -31
- package/esm2022/lib/menu-overlay/rdt-menu-overlay.component.mjs +36 -33
- package/esm2022/lib/private-models.mjs +8 -1
- package/esm2022/lib/rdt-menu.module.mjs +5 -5
- package/fesm2022/ngrdt-menu.mjs +309 -194
- package/fesm2022/ngrdt-menu.mjs.map +1 -1
- package/index.d.ts +1 -1
- package/lib/menu/rdt-menu.component.d.ts +15 -9
- package/lib/menu-bar/rdt-menu-bar.component.d.ts +3 -0
- package/lib/menu-base/rdt-menu-base.component.d.ts +24 -6
- package/lib/menu-overlay/rdt-menu-overlay.component.d.ts +9 -6
- package/lib/private-models.d.ts +17 -7
- package/package.json +2 -2
package/README.md
CHANGED
|
@@ -1,7 +1,161 @@
|
|
|
1
1
|
# @ngrdt/menu
|
|
2
2
|
|
|
3
|
-
This
|
|
3
|
+
This package provides two desktop first components - nested menu `rdt-menu` which is activated by your own `rdt-button` and `rdt-menu-bar` which uses custom buttons.
|
|
4
4
|
|
|
5
|
-
|
|
5
|
+
The point of this library is to provide fully customizable and accessible menu compliant with ARIA standards, compatible with custom shortcuts as well as tab key, arrows etc according to [Mozilla guidelines](https://developer.mozilla.org/en-US/docs/Web/Accessibility/ARIA/Reference/Roles/menubar_role).
|
|
6
6
|
|
|
7
|
-
|
|
7
|
+
## General usage
|
|
8
|
+
|
|
9
|
+
For menu to work you need to define and provide `rdt-button`. See component `ngrdt-demo/src/app/test-menu-button` for basic instructions on how to do that. This only applies to menu, not menu bar.
|
|
10
|
+
|
|
11
|
+
|
|
12
|
+
## Styling
|
|
13
|
+
|
|
14
|
+
Setting `background` and `color` properties directly is not recommended. Instead set CSS variables. You may use custom styles for menu bar or menu overlay for extra spacing, `border-radius`, icons etc.
|
|
15
|
+
|
|
16
|
+
For ergonomics you should always use different colors or borders for different states. Use semi transparent versions of one color if you don't know which colors to pick.
|
|
17
|
+
|
|
18
|
+
### State order
|
|
19
|
+
|
|
20
|
+
Menu items can be in various overlapping states. Here is order of importance:
|
|
21
|
+
|
|
22
|
+
1. focus - User navigates to button using arrows or `tab` key.
|
|
23
|
+
2. hover
|
|
24
|
+
3. expanded - Submenu opened by this item is currently active and visible.
|
|
25
|
+
4. route-active - Either menu itself or one of its submenus contains link to current page.
|
|
26
|
+
|
|
27
|
+
### Leveraging overriden CSS variables
|
|
28
|
+
|
|
29
|
+
Component styles are designed in a way which lets you use variables for multiple purposes.
|
|
30
|
+
|
|
31
|
+
```css
|
|
32
|
+
.menu-bar-item .menu-bar-item-content:hover {
|
|
33
|
+
--rdt-menu-bar-item-background: var(--rdt-menu-bar-item-hover-background);
|
|
34
|
+
--rdt-menu-bar-item-text-color: var(--rdt-menu-bar-item-hover-text-color);
|
|
35
|
+
--rdt-menu-bar-item-border: var(--rdt-menu-bar-item-hover-border);
|
|
36
|
+
}
|
|
37
|
+
```
|
|
38
|
+
|
|
39
|
+
For instance the default icons are made purely in CSS and use this pattern to always have the same color as the button label.
|
|
40
|
+
|
|
41
|
+
```css
|
|
42
|
+
.menu-bar-item-icon.rdt-menu-icon-down {
|
|
43
|
+
width: 0;
|
|
44
|
+
height: 0;
|
|
45
|
+
border-left: var(--rdt-menu-default-icon-size) solid transparent;
|
|
46
|
+
border-right: var(--rdt-menu-default-icon-size) solid transparent;
|
|
47
|
+
border-top: var(--rdt-menu-default-icon-size) solid
|
|
48
|
+
var(--rdt-menu-bar-item-text-color);
|
|
49
|
+
}
|
|
50
|
+
```
|
|
51
|
+
|
|
52
|
+
### Playground theme
|
|
53
|
+
|
|
54
|
+
I recommend using this theme to understand all different variables.
|
|
55
|
+
|
|
56
|
+
```css
|
|
57
|
+
:root {
|
|
58
|
+
/*************************************************/
|
|
59
|
+
/************** Menu overlay styles **************/
|
|
60
|
+
/*************************************************/
|
|
61
|
+
|
|
62
|
+
// Minimum width of menu overlay
|
|
63
|
+
--rdt-menu-min-width: 12rem;
|
|
64
|
+
|
|
65
|
+
// Border radius of menu overlay
|
|
66
|
+
--rdt-menu-border-radius: 5px;
|
|
67
|
+
|
|
68
|
+
// Background color of menu overlay - mainly color of horizontal padding.
|
|
69
|
+
--rdt-menu-background: black;
|
|
70
|
+
|
|
71
|
+
// Horizontal padding of menu overlay (above and below the menu items)
|
|
72
|
+
--rdt-menu-padding-horizontal-padding: 0.75rem;
|
|
73
|
+
|
|
74
|
+
// Box shadow of menu overlay
|
|
75
|
+
--rdt-menu-box-shadow: 0px 5px 6px -3px rgba(0, 0, 0, 0.2),
|
|
76
|
+
0px 9px 12px 1px rgba(0, 0, 0, 0.14), 0px 3px 16px 2px rgba(0, 0, 0, 0.12);
|
|
77
|
+
|
|
78
|
+
// Margin of menu overlay (distance from bottom of menu (bar) button to top of menu overlay)
|
|
79
|
+
// If you set higher value than 10, then adjust [hitboxMargin] accordingly.
|
|
80
|
+
--rdt-menu-overlay-margin-x: 0px;
|
|
81
|
+
--rdt-menu-overlay-margin-y: 5px;
|
|
82
|
+
|
|
83
|
+
// Size of default arrows
|
|
84
|
+
--rdt-menu-default-icon-size: 0.3rem;
|
|
85
|
+
|
|
86
|
+
/**********************************************/
|
|
87
|
+
/************** Menu item styles **************/
|
|
88
|
+
/**********************************************/
|
|
89
|
+
// These only change styles of items within popup. NOT the menu bar items.
|
|
90
|
+
|
|
91
|
+
// Default menu item style
|
|
92
|
+
--rdt-menu-item-background: white;
|
|
93
|
+
--rdt-menu-item-text-color: black;
|
|
94
|
+
--rdt-menu-item-padding: 0.5rem 1rem;
|
|
95
|
+
--rdt-menu-item-border: none;
|
|
96
|
+
--rdt-menu-item-outline: none;
|
|
97
|
+
|
|
98
|
+
// Hover styles
|
|
99
|
+
--rdt-menu-item-hover-background: green;
|
|
100
|
+
--rdt-menu-item-hover-text-color: green;
|
|
101
|
+
--rdt-menu-item-hover-border: none;
|
|
102
|
+
--rdt-menu-item-hover-outline: none;
|
|
103
|
+
|
|
104
|
+
// Expanded styles
|
|
105
|
+
--rdt-menu-item-expanded-background: cyan;
|
|
106
|
+
--rdt-menu-item-expanded-text-color: cyan;
|
|
107
|
+
--rdt-menu-item-expanded-border: none;
|
|
108
|
+
--rdt-menu-item-expanded-outline: none;
|
|
109
|
+
|
|
110
|
+
// Focus styles
|
|
111
|
+
--rdt-menu-item-focus-background: red;
|
|
112
|
+
--rdt-menu-item-focus-text-color: red;
|
|
113
|
+
--rdt-menu-item-focus-border: none;
|
|
114
|
+
--rdt-menu-item-focus-outline: none;
|
|
115
|
+
|
|
116
|
+
// Route active styles
|
|
117
|
+
--rdt-menu-item-route-active-background: lightblue;
|
|
118
|
+
--rdt-menu-item-route-active-text-color: lightblue;
|
|
119
|
+
--rdt-menu-item-route-active-border: none;
|
|
120
|
+
--rdt-menu-item-route-active-outline: none;
|
|
121
|
+
|
|
122
|
+
/**********************************************/
|
|
123
|
+
/************ Menu bar item styles ************/
|
|
124
|
+
/**********************************************/
|
|
125
|
+
|
|
126
|
+
--rdt-menu-bar-padding: 0 0.5rem;
|
|
127
|
+
|
|
128
|
+
// Default menu bar item style
|
|
129
|
+
--rdt-menu-bar-item-background: transparent;
|
|
130
|
+
--rdt-menu-bar-item-text-color: black;
|
|
131
|
+
--rdt-menu-bar-item-padding: 0.5rem;
|
|
132
|
+
--rdt-menu-bar-item-border-radius: 5px;
|
|
133
|
+
--rdt-menu-bar-item-border: none;
|
|
134
|
+
--rdt-menu-bar-item-outline: none;
|
|
135
|
+
--rdt-menu-bar-item-margin: 0 0.5rem;
|
|
136
|
+
|
|
137
|
+
// Hover styles
|
|
138
|
+
--rdt-menu-bar-item-hover-background: lightgreen;
|
|
139
|
+
--rdt-menu-bar-item-hover-text-color: lightgreen;
|
|
140
|
+
--rdt-menu-bar-item-hover-border: none;
|
|
141
|
+
--rdt-menu-bar-item-hover-outline: none;
|
|
142
|
+
|
|
143
|
+
// Expanded styles
|
|
144
|
+
--rdt-menu-bar-item-expanded-background: blue;
|
|
145
|
+
--rdt-menu-bar-item-expanded-text-color: blue;
|
|
146
|
+
--rdt-menu-bar-item-expanded-border: none;
|
|
147
|
+
--rdt-menu-bar-item-expanded-outline: none;
|
|
148
|
+
|
|
149
|
+
// Focus styles
|
|
150
|
+
--rdt-menu-bar-item-focus-background: orange;
|
|
151
|
+
--rdt-menu-bar-item-focus-text-color: orange;
|
|
152
|
+
--rdt-menu-bar-item-focus-border: none;
|
|
153
|
+
--rdt-menu-bar-item-focus-outline: none;
|
|
154
|
+
|
|
155
|
+
// Route active styles
|
|
156
|
+
--rdt-menu-bar-item-route-active-background: pink;
|
|
157
|
+
--rdt-menu-bar-item-route-active-text-color: pink;
|
|
158
|
+
--rdt-menu-bar-item-route-active-border: none;
|
|
159
|
+
--rdt-menu-bar-item-route-active-outline: none;
|
|
160
|
+
}
|
|
161
|
+
```
|
package/esm2022/index.mjs
CHANGED
|
@@ -1,5 +1,5 @@
|
|
|
1
|
-
export * from './lib/menu/rdt-menu.component';
|
|
2
1
|
export * from './lib/menu-bar/rdt-menu-bar.component';
|
|
2
|
+
export * from './lib/menu/rdt-menu.component';
|
|
3
3
|
export * from './lib/models';
|
|
4
4
|
export * from './lib/rdt-menu.module';
|
|
5
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
5
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9AbmdyZHQvbWVudS9zcmMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsY0FBYyx1Q0FBdUMsQ0FBQztBQUN0RCxjQUFjLCtCQUErQixDQUFDO0FBQzlDLGNBQWMsY0FBYyxDQUFDO0FBQzdCLGNBQWMsdUJBQXVCLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgKiBmcm9tICcuL2xpYi9tZW51LWJhci9yZHQtbWVudS1iYXIuY29tcG9uZW50JztcbmV4cG9ydCAqIGZyb20gJy4vbGliL21lbnUvcmR0LW1lbnUuY29tcG9uZW50JztcbmV4cG9ydCAqIGZyb20gJy4vbGliL21vZGVscyc7XG5leHBvcnQgKiBmcm9tICcuL2xpYi9yZHQtbWVudS5tb2R1bGUnO1xuIl19
|
|
@@ -1,52 +1,77 @@
|
|
|
1
|
-
import { ChangeDetectionStrategy, Component,
|
|
2
|
-
import { RDT_BUTTON_BASE_PROVIDER
|
|
1
|
+
import { ChangeDetectionStrategy, Component, inject, Input, QueryList, ViewEncapsulation, } from '@angular/core';
|
|
2
|
+
import { RDT_BUTTON_BASE_PROVIDER } from '@ngrdt/button';
|
|
3
3
|
import { RdtMenuBaseComponent } from '../menu-base/rdt-menu-base.component';
|
|
4
4
|
import { parseMenuItems } from '../private-models';
|
|
5
5
|
import * as i0 from "@angular/core";
|
|
6
6
|
import * as i1 from "@angular/common";
|
|
7
|
-
import * as i2 from "
|
|
8
|
-
import * as i3 from "../menu-overlay/rdt-menu-overlay.component";
|
|
7
|
+
import * as i2 from "../menu-overlay/rdt-menu-overlay.component";
|
|
9
8
|
export class RdtMenuComponent extends RdtMenuBaseComponent {
|
|
10
9
|
buttonClass = inject(RDT_BUTTON_BASE_PROVIDER);
|
|
10
|
+
trigger;
|
|
11
11
|
dataTestId = '';
|
|
12
|
-
get
|
|
13
|
-
return this.
|
|
12
|
+
get items() {
|
|
13
|
+
return this._items;
|
|
14
14
|
}
|
|
15
|
-
set
|
|
16
|
-
this.
|
|
17
|
-
|
|
15
|
+
set items(value) {
|
|
16
|
+
this._items = value;
|
|
17
|
+
const wrapper = {
|
|
18
|
+
label: '',
|
|
19
|
+
icon: '',
|
|
20
|
+
items: value,
|
|
21
|
+
};
|
|
22
|
+
this.allParsedItems = parseMenuItems([wrapper], this.injector);
|
|
18
23
|
this.filterItems();
|
|
19
24
|
}
|
|
20
|
-
|
|
21
|
-
|
|
22
|
-
|
|
25
|
+
_items;
|
|
26
|
+
focusableElements = new QueryList();
|
|
27
|
+
get buttonContainer() {
|
|
28
|
+
return this.anchorElement ?? undefined;
|
|
23
29
|
}
|
|
24
30
|
get parsedItem() {
|
|
25
31
|
return this.parsedItems[0];
|
|
26
32
|
}
|
|
27
|
-
get
|
|
28
|
-
return
|
|
29
|
-
|
|
30
|
-
|
|
31
|
-
|
|
32
|
-
|
|
33
|
-
|
|
33
|
+
get anchorElement() {
|
|
34
|
+
return this.trigger.anchorElement;
|
|
35
|
+
}
|
|
36
|
+
ngOnInit() {
|
|
37
|
+
super.ngOnInit();
|
|
38
|
+
this.trigger.tabIndex = 0;
|
|
39
|
+
this.trigger.dataTestId = this.dataTestId;
|
|
40
|
+
this.trigger.click$.subscribe(() => this.toggle());
|
|
41
|
+
this.listenPointerOver();
|
|
42
|
+
if (this.anchorElement) {
|
|
43
|
+
console.log('Anchor element:', this.anchorElement);
|
|
44
|
+
this.focusableElements.reset([{ nativeElement: this.anchorElement }]);
|
|
45
|
+
}
|
|
46
|
+
}
|
|
47
|
+
ngDoCheck() {
|
|
48
|
+
if (this.trigger) {
|
|
49
|
+
this.trigger.aria = {
|
|
34
50
|
role: 'menuitem',
|
|
35
|
-
'aria-haspopup': '
|
|
36
|
-
'aria-expanded': this.
|
|
37
|
-
}
|
|
38
|
-
}
|
|
51
|
+
'aria-haspopup': 'true',
|
|
52
|
+
'aria-expanded': this.expandedChild?.item === this.parsedItem,
|
|
53
|
+
};
|
|
54
|
+
}
|
|
39
55
|
}
|
|
40
56
|
toggle() {
|
|
41
57
|
this.onItemClick(this.parsedItem);
|
|
58
|
+
this.cd.markForCheck();
|
|
59
|
+
}
|
|
60
|
+
listenPointerOver() {
|
|
61
|
+
const target = this.anchorElement;
|
|
62
|
+
const listener = this.renderer.listen(target, 'pointerover', (event) => {
|
|
63
|
+
this.onItemPointerEnter(this.parsedItem);
|
|
64
|
+
this.cd.markForCheck();
|
|
65
|
+
});
|
|
66
|
+
this.destroyRef.onDestroy(() => listener());
|
|
42
67
|
}
|
|
43
68
|
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.7", ngImport: i0, type: RdtMenuComponent, deps: null, target: i0.ɵɵFactoryTarget.Component });
|
|
44
|
-
static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.7", type: RdtMenuComponent, selector: "rdt-menu", inputs: {
|
|
69
|
+
static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.7", type: RdtMenuComponent, selector: "rdt-menu", inputs: { trigger: "trigger", dataTestId: "dataTestId", items: "items" }, providers: [
|
|
45
70
|
{
|
|
46
71
|
provide: RdtMenuBaseComponent,
|
|
47
72
|
useExisting: RdtMenuComponent,
|
|
48
73
|
},
|
|
49
|
-
], usesInheritance: true, ngImport: i0, template: "<
|
|
74
|
+
], usesInheritance: true, ngImport: i0, template: "<rdt-menu-overlay\r\n *ngIf=\"parsedItem\"\r\n [anchorElement]=\"anchorElement!\"\r\n [item]=\"$any(parsedItem)\"\r\n [expanded]=\"expandedChild?.item === parsedItem\"\r\n [autofocusItem]=\"autofocusSubmenuItem\"\r\n [preferredHorizontalDir]=\"preferredHorizontalDir\"\r\n [preferredVerticalDir]=\"preferredVerticalDir\"\r\n [level]=\"0\"\r\n/>\r\n", styles: ["a.menu-bar-item-content,a.menu-item-content,button.menu-bar-item-content,button.menu-item-content{white-space:nowrap;appearance:none;text-decoration:none;box-sizing:border-box}.menu-item-right-content,.menu-bar-item-right-content{margin-left:auto;display:flex;align-items:center}.menu-item-shortcut,.menu-bar-item-shortcut{margin-left:2rem}.menu-item-icon,.menu-bar-item-icon{margin-left:.5rem;font-size:1.2rem}.dp3-menu-base ul{margin-bottom:0}.rdt-menu-root{margin-left:0}.rdt-menu-root.rdt-overlay-down{margin-top:var(--rdt-menu-overlay-margin-y)}.rdt-menu-root.rdt-overlay-up{margin-top:calc(-1 * var(--rdt-menu-overlay-margin-y))}.rdt-menu-sub.rdt-overlay-left{margin-left:calc(-1 * var(--rdt-menu-overlay-margin-x))}.rdt-menu-sub.rdt-overlay-right{margin-left:var(--rdt-menu-overlay-margin-x)}rdt-menu{display:block;position:relative}\n"], dependencies: [{ kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: i2.RdtMenuOverlayComponent, selector: "rdt-menu-overlay", inputs: ["item", "level", "preferredHorizontalDir", "preferredVerticalDir", "expanded", "autofocusItem", "anchorElement"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush, encapsulation: i0.ViewEncapsulation.None });
|
|
50
75
|
}
|
|
51
76
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.7", ngImport: i0, type: RdtMenuComponent, decorators: [{
|
|
52
77
|
type: Component,
|
|
@@ -55,14 +80,14 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.7", ngImpor
|
|
|
55
80
|
provide: RdtMenuBaseComponent,
|
|
56
81
|
useExisting: RdtMenuComponent,
|
|
57
82
|
},
|
|
58
|
-
], template: "<
|
|
59
|
-
}], propDecorators: {
|
|
83
|
+
], template: "<rdt-menu-overlay\r\n *ngIf=\"parsedItem\"\r\n [anchorElement]=\"anchorElement!\"\r\n [item]=\"$any(parsedItem)\"\r\n [expanded]=\"expandedChild?.item === parsedItem\"\r\n [autofocusItem]=\"autofocusSubmenuItem\"\r\n [preferredHorizontalDir]=\"preferredHorizontalDir\"\r\n [preferredVerticalDir]=\"preferredVerticalDir\"\r\n [level]=\"0\"\r\n/>\r\n", styles: ["a.menu-bar-item-content,a.menu-item-content,button.menu-bar-item-content,button.menu-item-content{white-space:nowrap;appearance:none;text-decoration:none;box-sizing:border-box}.menu-item-right-content,.menu-bar-item-right-content{margin-left:auto;display:flex;align-items:center}.menu-item-shortcut,.menu-bar-item-shortcut{margin-left:2rem}.menu-item-icon,.menu-bar-item-icon{margin-left:.5rem;font-size:1.2rem}.dp3-menu-base ul{margin-bottom:0}.rdt-menu-root{margin-left:0}.rdt-menu-root.rdt-overlay-down{margin-top:var(--rdt-menu-overlay-margin-y)}.rdt-menu-root.rdt-overlay-up{margin-top:calc(-1 * var(--rdt-menu-overlay-margin-y))}.rdt-menu-sub.rdt-overlay-left{margin-left:calc(-1 * var(--rdt-menu-overlay-margin-x))}.rdt-menu-sub.rdt-overlay-right{margin-left:var(--rdt-menu-overlay-margin-x)}rdt-menu{display:block;position:relative}\n"] }]
|
|
84
|
+
}], propDecorators: { trigger: [{
|
|
85
|
+
type: Input,
|
|
86
|
+
args: [{ required: true }]
|
|
87
|
+
}], dataTestId: [{
|
|
60
88
|
type: Input
|
|
61
|
-
}],
|
|
89
|
+
}], items: [{
|
|
62
90
|
type: Input,
|
|
63
91
|
args: [{ required: true }]
|
|
64
|
-
}], onPointerLeave: [{
|
|
65
|
-
type: HostListener,
|
|
66
|
-
args: ['pointerleave']
|
|
67
92
|
}] } });
|
|
68
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
93
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"rdt-menu.component.js","sourceRoot":"","sources":["../../../../../../@ngrdt/menu/src/lib/menu/rdt-menu.component.ts","../../../../../../@ngrdt/menu/src/lib/menu/rdt-menu.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\r\n  ChangeDetectionStrategy,\r\n  Component,\r\n  DoCheck,\r\n  ElementRef,\r\n  inject,\r\n  Input,\r\n  OnInit,\r\n  QueryList,\r\n  ViewEncapsulation,\r\n} from '@angular/core';\r\nimport { RDT_BUTTON_BASE_PROVIDER, RdtButtonBaseInt } from '@ngrdt/button';\r\nimport { RdtMenuBaseComponent } from '../menu-base/rdt-menu-base.component';\r\nimport { RdtMenuItem } from '../models';\r\nimport { parseMenuItems } from '../private-models';\r\n\r\n@Component({\r\n  selector: 'rdt-menu',\r\n  templateUrl: './rdt-menu.component.html',\r\n  styleUrl: './rdt-menu.component.scss',\r\n  changeDetection: ChangeDetectionStrategy.OnPush,\r\n  encapsulation: ViewEncapsulation.None,\r\n  providers: [\r\n    {\r\n      provide: RdtMenuBaseComponent,\r\n      useExisting: RdtMenuComponent,\r\n    },\r\n  ],\r\n})\r\nexport class RdtMenuComponent\r\n  extends RdtMenuBaseComponent\r\n  implements OnInit, DoCheck\r\n{\r\n  protected readonly buttonClass = inject(RDT_BUTTON_BASE_PROVIDER);\r\n\r\n  @Input({ required: true })\r\n  trigger!: RdtButtonBaseInt;\r\n\r\n  @Input()\r\n  dataTestId = '';\r\n\r\n  @Input({ required: true })\r\n  get items() {\r\n    return this._items;\r\n  }\r\n  set items(value: RdtMenuItem[]) {\r\n    this._items = value;\r\n    const wrapper: RdtMenuItem = {\r\n      label: '',\r\n      icon: '',\r\n      items: value,\r\n    };\r\n    this.allParsedItems = parseMenuItems([wrapper], this.injector);\r\n    this.filterItems();\r\n  }\r\n  private _items!: RdtMenuItem[];\r\n\r\n  protected override readonly focusableElements = new QueryList<\r\n    ElementRef<HTMLElement>\r\n  >();\r\n\r\n  override get buttonContainer(): HTMLElement | undefined {\r\n    return this.anchorElement ?? undefined;\r\n  }\r\n\r\n  get parsedItem() {\r\n    return this.parsedItems[0];\r\n  }\r\n\r\n  get anchorElement() {\r\n    return this.trigger.anchorElement;\r\n  }\r\n\r\n  override ngOnInit(): void {\r\n    super.ngOnInit();\r\n    this.trigger.tabIndex = 0;\r\n    this.trigger.dataTestId = this.dataTestId;\r\n    this.trigger.click$.subscribe(() => this.toggle());\r\n    this.listenPointerOver();\r\n    if (this.anchorElement) {\r\n      console.log('Anchor element:', this.anchorElement);\r\n      this.focusableElements.reset([{ nativeElement: this.anchorElement }]);\r\n    }\r\n  }\r\n\r\n  ngDoCheck(): void {\r\n    if (this.trigger) {\r\n      this.trigger.aria = {\r\n        role: 'menuitem',\r\n        'aria-haspopup': 'true',\r\n        'aria-expanded': this.expandedChild?.item === this.parsedItem,\r\n      };\r\n    }\r\n  }\r\n\r\n  toggle() {\r\n    this.onItemClick(this.parsedItem);\r\n    this.cd.markForCheck();\r\n  }\r\n\r\n  private listenPointerOver() {\r\n    const target = this.anchorElement;\r\n    const listener = this.renderer.listen(target, 'pointerover', (event) => {\r\n      this.onItemPointerEnter(this.parsedItem);\r\n      this.cd.markForCheck();\r\n    });\r\n\r\n    this.destroyRef.onDestroy(() => listener());\r\n  }\r\n}\r\n","<rdt-menu-overlay\r\n  *ngIf=\"parsedItem\"\r\n  [anchorElement]=\"anchorElement!\"\r\n  [item]=\"$any(parsedItem)\"\r\n  [expanded]=\"expandedChild?.item === parsedItem\"\r\n  [autofocusItem]=\"autofocusSubmenuItem\"\r\n  [preferredHorizontalDir]=\"preferredHorizontalDir\"\r\n  [preferredVerticalDir]=\"preferredVerticalDir\"\r\n  [level]=\"0\"\r\n/>\r\n"]}
|
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
import { ChangeDetectionStrategy, Component, HostBinding, Input, numberAttribute, ViewEncapsulation, } from '@angular/core';
|
|
1
|
+
import { ChangeDetectionStrategy, Component, ElementRef, HostBinding, Input, numberAttribute, ViewChild, ViewEncapsulation, } from '@angular/core';
|
|
2
2
|
import { RdtMenuBaseComponent } from '../menu-base/rdt-menu-base.component';
|
|
3
3
|
import { parseMenuItems } from '../private-models';
|
|
4
4
|
import * as i0 from "@angular/core";
|
|
@@ -8,6 +8,10 @@ import * as i3 from "@ngrdt/router";
|
|
|
8
8
|
import * as i4 from "@ngrdt/shortcuts";
|
|
9
9
|
import * as i5 from "../menu-overlay/rdt-menu-overlay.component";
|
|
10
10
|
export class RdtMenuBarComponent extends RdtMenuBaseComponent {
|
|
11
|
+
buttonContainerRef;
|
|
12
|
+
get buttonContainer() {
|
|
13
|
+
return this.buttonContainerRef?.nativeElement;
|
|
14
|
+
}
|
|
11
15
|
get items() {
|
|
12
16
|
return this._items;
|
|
13
17
|
}
|
|
@@ -21,12 +25,12 @@ export class RdtMenuBarComponent extends RdtMenuBaseComponent {
|
|
|
21
25
|
footerHeight = 0;
|
|
22
26
|
roleAttr = 'menubar';
|
|
23
27
|
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.7", ngImport: i0, type: RdtMenuBarComponent, deps: null, target: i0.ɵɵFactoryTarget.Component });
|
|
24
|
-
static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "
|
|
28
|
+
static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.7", type: RdtMenuBarComponent, selector: "rdt-menu-bar", inputs: { items: "items", headerHeight: ["headerHeight", "headerHeight", numberAttribute], footerHeight: ["footerHeight", "footerHeight", numberAttribute] }, host: { properties: { "attr.role": "this.roleAttr" } }, providers: [
|
|
25
29
|
{
|
|
26
30
|
provide: RdtMenuBaseComponent,
|
|
27
31
|
useExisting: RdtMenuBarComponent,
|
|
28
32
|
},
|
|
29
|
-
], usesInheritance: true, ngImport: i0, template: "<ul class=\"menu-bar-item-container\" role=\"presentation\">\r\n
|
|
33
|
+
], viewQueries: [{ propertyName: "buttonContainerRef", first: true, predicate: ["buttonContainer"], descendants: true, static: true }], usesInheritance: true, ngImport: i0, template: "<ul class=\"menu-bar-item-container\" role=\"presentation\" #buttonContainer>\r\n @for(item of parsedItems; track item; let i = $index) {\r\n <li\r\n class=\"menu-bar-item\"\r\n role=\"presentation\"\r\n rdtAnyRouteActive=\"menu-bar-item-route-active\"\r\n [watchedRoutes]=\"getChildRoutes(item)\"\r\n (rdtKeyListener)=\"onKeyDown(i, $event)\"\r\n #anchorEl\r\n >\r\n @if(!item.routerLink && !item.externalLink) {\r\n <button\r\n #focusableItem\r\n class=\"menu-bar-item-content\"\r\n role=\"menuitem\"\r\n (click)=\"onItemClick(item)\"\r\n (pointerenter)=\"onItemPointerEnter(item)\"\r\n [attr.aria-haspopup]=\"item.items ? 'menu' : null\"\r\n [attr.aria-expanded]=\"item === expandedChild?.item\"\r\n [attr.tabindex]=\"0\"\r\n [attr.data-testid]=\"item.dataTestId\"\r\n >\r\n {{ item.label }}\r\n\r\n <div class=\"menu-bar-item-right-content\">\r\n @if(item.shortcut) {\r\n <span class=\"menu-item-shortcut\">\r\n {{ item.shortcut.label }}\r\n </span>\r\n }\r\n <!-- TODO\r\n <rdt-icon\r\n *ngIf=\"item.items\"\r\n name=\"expand_more\"\r\n class=\"menu-bar-item-icon\"\r\n inverted\r\n />\r\n <rdt-icon\r\n *ngIf=\"item.icon\"\r\n [name]=\"item.icon\"\r\n class=\"menu-bar-item-icon\"\r\n inverted\r\n />\r\n -->\r\n @if(item.items) {\r\n <span class=\"menu-bar-item-icon rdt-menu-icon-down\"></span>\r\n }\r\n </div>\r\n </button>\r\n }\r\n\r\n <ng-template #linkBody let-item>\r\n {{ item.label }}\r\n\r\n <div class=\"menu-bar-item-right-content\">\r\n @if(item.shortcut) {\r\n <span class=\"menu-item-shortcut\">\r\n {{ item.shortcut.label }}\r\n </span>\r\n }\r\n <!--\r\n <rdt-icon\r\n *ngIf=\"item.icon\"\r\n [name]=\"item.icon\"\r\n class=\"menu-bar-item-icon\"\r\n inverted\r\n />\r\n --></div>\r\n </ng-template>\r\n\r\n @if(item.routerLink) {\r\n <a\r\n #focusableItem\r\n class=\"menu-bar-item-content\"\r\n role=\"menuitem\"\r\n (click)=\"onItemClick(item)\"\r\n [routerLink]=\"item.routerLink\"\r\n [target]=\"item.target!\"\r\n [attr.tabindex]=\"0\"\r\n [attr.data-testid]=\"item.dataTestId\"\r\n >\r\n <ng-container\r\n *ngTemplateOutlet=\"linkBody; context: { $implicit: item }\"\r\n />\r\n </a>\r\n } @if(item.externalLink) {\r\n <a\r\n #focusableItem\r\n class=\"menu-bar-item-content\"\r\n role=\"menuitem\"\r\n (click)=\"onItemClick(item)\"\r\n [href]=\"item.externalLink\"\r\n [target]=\"item.target\"\r\n [attr.tabindex]=\"0\"\r\n [attr.data-testid]=\"item.dataTestId\"\r\n >\r\n <ng-container\r\n *ngTemplateOutlet=\"linkBody; context: { $implicit: item }\"\r\n />\r\n </a>\r\n } @if (item.items) {\r\n <rdt-menu-overlay\r\n [item]=\"$any(item)\"\r\n [expanded]=\"item === expandedChild?.item\"\r\n [autofocusItem]=\"autofocusSubmenuItem\"\r\n [preferredHorizontalDir]=\"preferredHorizontalDir\"\r\n [preferredVerticalDir]=\"preferredVerticalDir\"\r\n [level]=\"0\"\r\n [anchorElement]=\"anchorEl\"\r\n />\r\n }\r\n </li>\r\n }\r\n</ul>\r\n", styles: ["a.menu-bar-item-content,a.menu-item-content,button.menu-bar-item-content,button.menu-item-content{white-space:nowrap;appearance:none;text-decoration:none;box-sizing:border-box}.menu-item-right-content,.menu-bar-item-right-content{margin-left:auto;display:flex;align-items:center}.menu-item-shortcut,.menu-bar-item-shortcut{margin-left:2rem}.menu-item-icon,.menu-bar-item-icon{margin-left:.5rem;font-size:1.2rem}.dp3-menu-base ul{margin-bottom:0}.rdt-menu-root{margin-left:0}.rdt-menu-root.rdt-overlay-down{margin-top:var(--rdt-menu-overlay-margin-y)}.rdt-menu-root.rdt-overlay-up{margin-top:calc(-1 * var(--rdt-menu-overlay-margin-y))}.rdt-menu-sub.rdt-overlay-left{margin-left:calc(-1 * var(--rdt-menu-overlay-margin-x))}.rdt-menu-sub.rdt-overlay-right{margin-left:var(--rdt-menu-overlay-margin-x)}rdt-menu-bar{-webkit-user-select:none;user-select:none;pointer-events:none;padding:var(--rdt-menu-bar-padding)}.menu-bar-item-container{display:flex;list-style-type:none;margin-bottom:0;margin-top:0;margin-block-start:0;margin-block-end:0;padding-left:0}.menu-bar-item{pointer-events:all;margin:var(--rdt-menu-bar-item-margin);position:relative;display:flex;align-items:center;justify-content:center}.menu-bar-item.menu-bar-item-route-active .menu-bar-item-content{--rdt-menu-bar-item-background: var( --rdt-menu-bar-item-route-active-background );--rdt-menu-bar-item-text-color: var( --rdt-menu-bar-item-route-active-text-color );--rdt-menu-bar-item-border: var(--rdt-menu-bar-item-route-active-border);--rdt-menu-bar-item-outline: var(--rdt-menu-bar-item-route-active-outline)}.menu-bar-item [aria-expanded=true].menu-bar-item-content{--rdt-menu-bar-item-background: var( --rdt-menu-bar-item-expanded-background );--rdt-menu-bar-item-text-color: var( --rdt-menu-bar-item-expanded-text-color );--rdt-menu-bar-item-border: var(--rdt-menu-bar-item-expanded-border);--rdt-menu-bar-item-outline: var(--rdt-menu-bar-item-expanded-outline)}.menu-bar-item-content{cursor:pointer;display:flex;align-items:center;overflow:hidden;padding:var(--rdt-menu-bar-item-padding);border-radius:var(--rdt-menu-bar-item-border-radius);border:var(--rdt-menu-bar-item-border);background-color:var(--rdt-menu-bar-item-background);color:var(--rdt-menu-bar-item-text-color);outline:var(--rdt-menu-bar-item-outline);font-size:inherit}.menu-bar-item-content:hover{--rdt-menu-bar-item-background: var(--rdt-menu-bar-item-hover-background);--rdt-menu-bar-item-text-color: var(--rdt-menu-bar-item-hover-text-color);--rdt-menu-bar-item-border: var(--rdt-menu-bar-item-hover-border);--rdt-menu-bar-item-outline: var(--rdt-menu-bar-item-hover-outline)}.menu-bar-item .menu-bar-item-content.focus-visible:focus,.menu-bar-item .menu-bar-item-content:focus-visible{--rdt-menu-bar-item-background: var(--rdt-menu-bar-item-focus-background);--rdt-menu-bar-item-text-color: var(--rdt-menu-bar-item-focus-text-color);--rdt-menu-bar-item-border: var(--rdt-menu-bar-item-focus-border);--rdt-menu-bar-item-outline: var(--rdt-menu-bar-item-focus-outline)}.menu-bar-item-icon.rdt-menu-icon-down{width:0;height:0;border-left:var(--rdt-menu-default-icon-size) solid transparent;border-right:var(--rdt-menu-default-icon-size) solid transparent;border-top:var(--rdt-menu-default-icon-size) solid var(--rdt-menu-bar-item-text-color)}\n"], dependencies: [{ kind: "directive", type: i1.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "directive", type: i2.RouterLink, selector: "[routerLink]", inputs: ["target", "queryParams", "fragment", "queryParamsHandling", "state", "info", "relativeTo", "preserveFragment", "skipLocationChange", "replaceUrl", "routerLink"] }, { kind: "directive", type: i3.RdtAnyRouteActiveDirective, selector: "[rdtAnyRouteActive]", inputs: ["rdtAnyRouteActive", "watchedRoutes", "anyRouteActiveOptions", "ariaCurrentWhenActive"] }, { kind: "directive", type: i4.RdtKeyListenerDirective, selector: "[rdtKeyListener]", outputs: ["rdtKeyListener"] }, { kind: "component", type: i5.RdtMenuOverlayComponent, selector: "rdt-menu-overlay", inputs: ["item", "level", "preferredHorizontalDir", "preferredVerticalDir", "expanded", "autofocusItem", "anchorElement"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush, encapsulation: i0.ViewEncapsulation.None });
|
|
30
34
|
}
|
|
31
35
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.7", ngImport: i0, type: RdtMenuBarComponent, decorators: [{
|
|
32
36
|
type: Component,
|
|
@@ -35,8 +39,11 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.7", ngImpor
|
|
|
35
39
|
provide: RdtMenuBaseComponent,
|
|
36
40
|
useExisting: RdtMenuBarComponent,
|
|
37
41
|
},
|
|
38
|
-
], template: "<ul class=\"menu-bar-item-container\" role=\"presentation\">\r\n
|
|
39
|
-
}], propDecorators: {
|
|
42
|
+
], template: "<ul class=\"menu-bar-item-container\" role=\"presentation\" #buttonContainer>\r\n @for(item of parsedItems; track item; let i = $index) {\r\n <li\r\n class=\"menu-bar-item\"\r\n role=\"presentation\"\r\n rdtAnyRouteActive=\"menu-bar-item-route-active\"\r\n [watchedRoutes]=\"getChildRoutes(item)\"\r\n (rdtKeyListener)=\"onKeyDown(i, $event)\"\r\n #anchorEl\r\n >\r\n @if(!item.routerLink && !item.externalLink) {\r\n <button\r\n #focusableItem\r\n class=\"menu-bar-item-content\"\r\n role=\"menuitem\"\r\n (click)=\"onItemClick(item)\"\r\n (pointerenter)=\"onItemPointerEnter(item)\"\r\n [attr.aria-haspopup]=\"item.items ? 'menu' : null\"\r\n [attr.aria-expanded]=\"item === expandedChild?.item\"\r\n [attr.tabindex]=\"0\"\r\n [attr.data-testid]=\"item.dataTestId\"\r\n >\r\n {{ item.label }}\r\n\r\n <div class=\"menu-bar-item-right-content\">\r\n @if(item.shortcut) {\r\n <span class=\"menu-item-shortcut\">\r\n {{ item.shortcut.label }}\r\n </span>\r\n }\r\n <!-- TODO\r\n <rdt-icon\r\n *ngIf=\"item.items\"\r\n name=\"expand_more\"\r\n class=\"menu-bar-item-icon\"\r\n inverted\r\n />\r\n <rdt-icon\r\n *ngIf=\"item.icon\"\r\n [name]=\"item.icon\"\r\n class=\"menu-bar-item-icon\"\r\n inverted\r\n />\r\n -->\r\n @if(item.items) {\r\n <span class=\"menu-bar-item-icon rdt-menu-icon-down\"></span>\r\n }\r\n </div>\r\n </button>\r\n }\r\n\r\n <ng-template #linkBody let-item>\r\n {{ item.label }}\r\n\r\n <div class=\"menu-bar-item-right-content\">\r\n @if(item.shortcut) {\r\n <span class=\"menu-item-shortcut\">\r\n {{ item.shortcut.label }}\r\n </span>\r\n }\r\n <!--\r\n <rdt-icon\r\n *ngIf=\"item.icon\"\r\n [name]=\"item.icon\"\r\n class=\"menu-bar-item-icon\"\r\n inverted\r\n />\r\n --></div>\r\n </ng-template>\r\n\r\n @if(item.routerLink) {\r\n <a\r\n #focusableItem\r\n class=\"menu-bar-item-content\"\r\n role=\"menuitem\"\r\n (click)=\"onItemClick(item)\"\r\n [routerLink]=\"item.routerLink\"\r\n [target]=\"item.target!\"\r\n [attr.tabindex]=\"0\"\r\n [attr.data-testid]=\"item.dataTestId\"\r\n >\r\n <ng-container\r\n *ngTemplateOutlet=\"linkBody; context: { $implicit: item }\"\r\n />\r\n </a>\r\n } @if(item.externalLink) {\r\n <a\r\n #focusableItem\r\n class=\"menu-bar-item-content\"\r\n role=\"menuitem\"\r\n (click)=\"onItemClick(item)\"\r\n [href]=\"item.externalLink\"\r\n [target]=\"item.target\"\r\n [attr.tabindex]=\"0\"\r\n [attr.data-testid]=\"item.dataTestId\"\r\n >\r\n <ng-container\r\n *ngTemplateOutlet=\"linkBody; context: { $implicit: item }\"\r\n />\r\n </a>\r\n } @if (item.items) {\r\n <rdt-menu-overlay\r\n [item]=\"$any(item)\"\r\n [expanded]=\"item === expandedChild?.item\"\r\n [autofocusItem]=\"autofocusSubmenuItem\"\r\n [preferredHorizontalDir]=\"preferredHorizontalDir\"\r\n [preferredVerticalDir]=\"preferredVerticalDir\"\r\n [level]=\"0\"\r\n [anchorElement]=\"anchorEl\"\r\n />\r\n }\r\n </li>\r\n }\r\n</ul>\r\n", styles: ["a.menu-bar-item-content,a.menu-item-content,button.menu-bar-item-content,button.menu-item-content{white-space:nowrap;appearance:none;text-decoration:none;box-sizing:border-box}.menu-item-right-content,.menu-bar-item-right-content{margin-left:auto;display:flex;align-items:center}.menu-item-shortcut,.menu-bar-item-shortcut{margin-left:2rem}.menu-item-icon,.menu-bar-item-icon{margin-left:.5rem;font-size:1.2rem}.dp3-menu-base ul{margin-bottom:0}.rdt-menu-root{margin-left:0}.rdt-menu-root.rdt-overlay-down{margin-top:var(--rdt-menu-overlay-margin-y)}.rdt-menu-root.rdt-overlay-up{margin-top:calc(-1 * var(--rdt-menu-overlay-margin-y))}.rdt-menu-sub.rdt-overlay-left{margin-left:calc(-1 * var(--rdt-menu-overlay-margin-x))}.rdt-menu-sub.rdt-overlay-right{margin-left:var(--rdt-menu-overlay-margin-x)}rdt-menu-bar{-webkit-user-select:none;user-select:none;pointer-events:none;padding:var(--rdt-menu-bar-padding)}.menu-bar-item-container{display:flex;list-style-type:none;margin-bottom:0;margin-top:0;margin-block-start:0;margin-block-end:0;padding-left:0}.menu-bar-item{pointer-events:all;margin:var(--rdt-menu-bar-item-margin);position:relative;display:flex;align-items:center;justify-content:center}.menu-bar-item.menu-bar-item-route-active .menu-bar-item-content{--rdt-menu-bar-item-background: var( --rdt-menu-bar-item-route-active-background );--rdt-menu-bar-item-text-color: var( --rdt-menu-bar-item-route-active-text-color );--rdt-menu-bar-item-border: var(--rdt-menu-bar-item-route-active-border);--rdt-menu-bar-item-outline: var(--rdt-menu-bar-item-route-active-outline)}.menu-bar-item [aria-expanded=true].menu-bar-item-content{--rdt-menu-bar-item-background: var( --rdt-menu-bar-item-expanded-background );--rdt-menu-bar-item-text-color: var( --rdt-menu-bar-item-expanded-text-color );--rdt-menu-bar-item-border: var(--rdt-menu-bar-item-expanded-border);--rdt-menu-bar-item-outline: var(--rdt-menu-bar-item-expanded-outline)}.menu-bar-item-content{cursor:pointer;display:flex;align-items:center;overflow:hidden;padding:var(--rdt-menu-bar-item-padding);border-radius:var(--rdt-menu-bar-item-border-radius);border:var(--rdt-menu-bar-item-border);background-color:var(--rdt-menu-bar-item-background);color:var(--rdt-menu-bar-item-text-color);outline:var(--rdt-menu-bar-item-outline);font-size:inherit}.menu-bar-item-content:hover{--rdt-menu-bar-item-background: var(--rdt-menu-bar-item-hover-background);--rdt-menu-bar-item-text-color: var(--rdt-menu-bar-item-hover-text-color);--rdt-menu-bar-item-border: var(--rdt-menu-bar-item-hover-border);--rdt-menu-bar-item-outline: var(--rdt-menu-bar-item-hover-outline)}.menu-bar-item .menu-bar-item-content.focus-visible:focus,.menu-bar-item .menu-bar-item-content:focus-visible{--rdt-menu-bar-item-background: var(--rdt-menu-bar-item-focus-background);--rdt-menu-bar-item-text-color: var(--rdt-menu-bar-item-focus-text-color);--rdt-menu-bar-item-border: var(--rdt-menu-bar-item-focus-border);--rdt-menu-bar-item-outline: var(--rdt-menu-bar-item-focus-outline)}.menu-bar-item-icon.rdt-menu-icon-down{width:0;height:0;border-left:var(--rdt-menu-default-icon-size) solid transparent;border-right:var(--rdt-menu-default-icon-size) solid transparent;border-top:var(--rdt-menu-default-icon-size) solid var(--rdt-menu-bar-item-text-color)}\n"] }]
|
|
43
|
+
}], propDecorators: { buttonContainerRef: [{
|
|
44
|
+
type: ViewChild,
|
|
45
|
+
args: ['buttonContainer', { static: true }]
|
|
46
|
+
}], items: [{
|
|
40
47
|
type: Input
|
|
41
48
|
}], headerHeight: [{
|
|
42
49
|
type: Input,
|
|
@@ -48,4 +55,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.7", ngImpor
|
|
|
48
55
|
type: HostBinding,
|
|
49
56
|
args: ['attr.role']
|
|
50
57
|
}] } });
|
|
51
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
58
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"rdt-menu-bar.component.js","sourceRoot":"","sources":["../../../../../../@ngrdt/menu/src/lib/menu-bar/rdt-menu-bar.component.ts","../../../../../../@ngrdt/menu/src/lib/menu-bar/rdt-menu-bar.component.html"],"names":[],"mappings":"AAAA,OAAO,EACL,uBAAuB,EACvB,SAAS,EACT,UAAU,EACV,WAAW,EACX,KAAK,EACL,eAAe,EACf,SAAS,EACT,iBAAiB,GAClB,MAAM,eAAe,CAAC;AACvB,OAAO,EAAE,oBAAoB,EAAE,MAAM,sCAAsC,CAAC;AAE5E,OAAO,EAAE,cAAc,EAAE,MAAM,mBAAmB,CAAC;;;;;;;AAenD,MAAM,OAAO,mBAAoB,SAAQ,oBAAoB;IAE3D,kBAAkB,CAA2B;IAE7C,IAAI,eAAe;QACjB,OAAO,IAAI,CAAC,kBAAkB,EAAE,aAAa,CAAC;IAChD,CAAC;IAED,IACI,KAAK;QACP,OAAO,IAAI,CAAC,MAAM,CAAC;IACrB,CAAC;IACD,IAAI,KAAK,CAAC,KAAoB;QAC5B,IAAI,CAAC,MAAM,GAAG,KAAK,CAAC;QACpB,IAAI,CAAC,cAAc,GAAG,cAAc,CAAC,KAAK,EAAE,IAAI,CAAC,QAAQ,CAAC,CAAC;QAC3D,IAAI,CAAC,WAAW,EAAE,CAAC;IACrB,CAAC;IACO,MAAM,CAAiB;IAG/B,YAAY,GAAG,CAAC,CAAC;IAGjB,YAAY,GAAG,CAAC,CAAC;IAGT,QAAQ,GAAG,SAAS,CAAC;uGA1BlB,mBAAmB;2FAAnB,mBAAmB,qGAmBV,eAAe,kDAGf,eAAe,wEA7BxB;YACT;gBACE,OAAO,EAAE,oBAAoB;gBAC7B,WAAW,EAAE,mBAAmB;aACjC;SACF,sLCzBH,s4GAkHA;;2FDvFa,mBAAmB;kBAb/B,SAAS;+BACE,cAAc,mBAGP,uBAAuB,CAAC,MAAM,iBAChC,iBAAiB,CAAC,IAAI,aAC1B;wBACT;4BACE,OAAO,EAAE,oBAAoB;4BAC7B,WAAW,qBAAqB;yBACjC;qBACF;8BAID,kBAAkB;sBADjB,SAAS;uBAAC,iBAAiB,EAAE,EAAE,MAAM,EAAE,IAAI,EAAE;gBAQ1C,KAAK;sBADR,KAAK;gBAYN,YAAY;sBADX,KAAK;uBAAC,EAAE,SAAS,EAAE,eAAe,EAAE;gBAIrC,YAAY;sBADX,KAAK;uBAAC,EAAE,SAAS,EAAE,eAAe,EAAE;gBAI7B,QAAQ;sBADf,WAAW;uBAAC,WAAW","sourcesContent":["import {\r\n  ChangeDetectionStrategy,\r\n  Component,\r\n  ElementRef,\r\n  HostBinding,\r\n  Input,\r\n  numberAttribute,\r\n  ViewChild,\r\n  ViewEncapsulation,\r\n} from '@angular/core';\r\nimport { RdtMenuBaseComponent } from '../menu-base/rdt-menu-base.component';\r\nimport { RdtMenuItem } from '../models';\r\nimport { parseMenuItems } from '../private-models';\r\n\r\n@Component({\r\n  selector: 'rdt-menu-bar',\r\n  templateUrl: './rdt-menu-bar.component.html',\r\n  styleUrl: './rdt-menu-bar.component.scss',\r\n  changeDetection: ChangeDetectionStrategy.OnPush,\r\n  encapsulation: ViewEncapsulation.None,\r\n  providers: [\r\n    {\r\n      provide: RdtMenuBaseComponent,\r\n      useExisting: RdtMenuBarComponent,\r\n    },\r\n  ],\r\n})\r\nexport class RdtMenuBarComponent extends RdtMenuBaseComponent {\r\n  @ViewChild('buttonContainer', { static: true })\r\n  buttonContainerRef?: ElementRef<HTMLElement>;\r\n\r\n  get buttonContainer() {\r\n    return this.buttonContainerRef?.nativeElement;\r\n  }\r\n\r\n  @Input()\r\n  get items() {\r\n    return this._items;\r\n  }\r\n  set items(value: RdtMenuItem[]) {\r\n    this._items = value;\r\n    this.allParsedItems = parseMenuItems(value, this.injector);\r\n    this.filterItems();\r\n  }\r\n  private _items!: RdtMenuItem[];\r\n\r\n  @Input({ transform: numberAttribute })\r\n  headerHeight = 0;\r\n\r\n  @Input({ transform: numberAttribute })\r\n  footerHeight = 0;\r\n\r\n  @HostBinding('attr.role')\r\n  private roleAttr = 'menubar';\r\n}\r\n","<ul class=\"menu-bar-item-container\" role=\"presentation\" #buttonContainer>\r\n  @for(item of parsedItems; track item; let i = $index) {\r\n  <li\r\n    class=\"menu-bar-item\"\r\n    role=\"presentation\"\r\n    rdtAnyRouteActive=\"menu-bar-item-route-active\"\r\n    [watchedRoutes]=\"getChildRoutes(item)\"\r\n    (rdtKeyListener)=\"onKeyDown(i, $event)\"\r\n    #anchorEl\r\n  >\r\n    @if(!item.routerLink && !item.externalLink) {\r\n    <button\r\n      #focusableItem\r\n      class=\"menu-bar-item-content\"\r\n      role=\"menuitem\"\r\n      (click)=\"onItemClick(item)\"\r\n      (pointerenter)=\"onItemPointerEnter(item)\"\r\n      [attr.aria-haspopup]=\"item.items ? 'menu' : null\"\r\n      [attr.aria-expanded]=\"item === expandedChild?.item\"\r\n      [attr.tabindex]=\"0\"\r\n      [attr.data-testid]=\"item.dataTestId\"\r\n    >\r\n      {{ item.label }}\r\n\r\n      <div class=\"menu-bar-item-right-content\">\r\n        @if(item.shortcut) {\r\n        <span class=\"menu-item-shortcut\">\r\n          {{ item.shortcut.label }}\r\n        </span>\r\n        }\r\n        <!-- TODO\r\n          <rdt-icon\r\n            *ngIf=\"item.items\"\r\n            name=\"expand_more\"\r\n            class=\"menu-bar-item-icon\"\r\n            inverted\r\n          />\r\n          <rdt-icon\r\n            *ngIf=\"item.icon\"\r\n            [name]=\"item.icon\"\r\n            class=\"menu-bar-item-icon\"\r\n            inverted\r\n          />\r\n        -->\r\n        @if(item.items) {\r\n        <span class=\"menu-bar-item-icon rdt-menu-icon-down\"></span>\r\n        }\r\n      </div>\r\n    </button>\r\n    }\r\n\r\n    <ng-template #linkBody let-item>\r\n      {{ item.label }}\r\n\r\n      <div class=\"menu-bar-item-right-content\">\r\n        @if(item.shortcut) {\r\n        <span class=\"menu-item-shortcut\">\r\n          {{ item.shortcut.label }}\r\n        </span>\r\n        }\r\n        <!--\r\n        <rdt-icon\r\n          *ngIf=\"item.icon\"\r\n          [name]=\"item.icon\"\r\n          class=\"menu-bar-item-icon\"\r\n          inverted\r\n        />\r\n      --></div>\r\n    </ng-template>\r\n\r\n    @if(item.routerLink) {\r\n    <a\r\n      #focusableItem\r\n      class=\"menu-bar-item-content\"\r\n      role=\"menuitem\"\r\n      (click)=\"onItemClick(item)\"\r\n      [routerLink]=\"item.routerLink\"\r\n      [target]=\"item.target!\"\r\n      [attr.tabindex]=\"0\"\r\n      [attr.data-testid]=\"item.dataTestId\"\r\n    >\r\n      <ng-container\r\n        *ngTemplateOutlet=\"linkBody; context: { $implicit: item }\"\r\n      />\r\n    </a>\r\n    } @if(item.externalLink) {\r\n    <a\r\n      #focusableItem\r\n      class=\"menu-bar-item-content\"\r\n      role=\"menuitem\"\r\n      (click)=\"onItemClick(item)\"\r\n      [href]=\"item.externalLink\"\r\n      [target]=\"item.target\"\r\n      [attr.tabindex]=\"0\"\r\n      [attr.data-testid]=\"item.dataTestId\"\r\n    >\r\n      <ng-container\r\n        *ngTemplateOutlet=\"linkBody; context: { $implicit: item }\"\r\n      />\r\n    </a>\r\n    } @if (item.items) {\r\n    <rdt-menu-overlay\r\n      [item]=\"$any(item)\"\r\n      [expanded]=\"item === expandedChild?.item\"\r\n      [autofocusItem]=\"autofocusSubmenuItem\"\r\n      [preferredHorizontalDir]=\"preferredHorizontalDir\"\r\n      [preferredVerticalDir]=\"preferredVerticalDir\"\r\n      [level]=\"0\"\r\n      [anchorElement]=\"anchorEl\"\r\n    />\r\n    }\r\n  </li>\r\n  }\r\n</ul>\r\n"]}
|