@ethlete/cdk 2.8.1 → 2.9.0
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/CHANGELOG.md +10 -0
- package/esm2022/lib/components/overlay/components/menu/context-menu-trigger.mjs +163 -0
- package/esm2022/lib/components/overlay/components/menu/menu-aim.mjs +140 -0
- package/esm2022/lib/components/overlay/components/menu/menu-bar.mjs +113 -0
- package/esm2022/lib/components/overlay/components/menu/menu-base.mjs +154 -0
- package/esm2022/lib/components/overlay/components/menu/menu-errors.mjs +7 -0
- package/esm2022/lib/components/overlay/components/menu/menu-group.mjs +26 -0
- package/esm2022/lib/components/overlay/components/menu/menu-interface.mjs +3 -0
- package/esm2022/lib/components/overlay/components/menu/menu-item-checkbox.mjs +39 -0
- package/esm2022/lib/components/overlay/components/menu/menu-item-radio.mjs +56 -0
- package/esm2022/lib/components/overlay/components/menu/menu-item-selectable.mjs +32 -0
- package/esm2022/lib/components/overlay/components/menu/menu-item.mjs +210 -0
- package/esm2022/lib/components/overlay/components/menu/menu-module.mjs +52 -0
- package/esm2022/lib/components/overlay/components/menu/menu-stack.mjs +101 -0
- package/esm2022/lib/components/overlay/components/menu/menu-trigger-base.mjs +77 -0
- package/esm2022/lib/components/overlay/components/menu/menu-trigger.mjs +240 -0
- package/esm2022/lib/components/overlay/components/menu/menu.mjs +116 -0
- package/esm2022/lib/components/overlay/components/menu/pointer-focus-tracker.mjs +26 -0
- package/esm2022/lib/components/overlay/components/menu/public-api.mjs +17 -0
- package/esm2022/lib/components/overlay/components/public-api.mjs +2 -1
- package/esm2022/lib/components/scrollable/components/scrollable/scrollable.component.mjs +3 -3
- package/fesm2022/ethlete-cdk.mjs +1561 -157
- package/fesm2022/ethlete-cdk.mjs.map +1 -1
- package/lib/components/overlay/components/menu/context-menu-trigger.d.ts +33 -0
- package/lib/components/overlay/components/menu/menu-aim.d.ts +34 -0
- package/lib/components/overlay/components/menu/menu-bar.d.ts +13 -0
- package/lib/components/overlay/components/menu/menu-base.d.ts +41 -0
- package/lib/components/overlay/components/menu/menu-errors.d.ts +2 -0
- package/lib/components/overlay/components/menu/menu-group.d.ts +5 -0
- package/lib/components/overlay/components/menu/menu-interface.d.ts +11 -0
- package/lib/components/overlay/components/menu/menu-item-checkbox.d.ts +9 -0
- package/lib/components/overlay/components/menu/menu-item-radio.d.ts +16 -0
- package/lib/components/overlay/components/menu/menu-item-selectable.d.ts +11 -0
- package/lib/components/overlay/components/menu/menu-item.d.ts +52 -0
- package/lib/components/overlay/components/menu/menu-module.d.ts +16 -0
- package/lib/components/overlay/components/menu/menu-stack.d.ts +54 -0
- package/lib/components/overlay/components/menu/menu-trigger-base.d.ts +33 -0
- package/lib/components/overlay/components/menu/menu-trigger.d.ts +34 -0
- package/lib/components/overlay/components/menu/menu.d.ts +17 -0
- package/lib/components/overlay/components/menu/pointer-focus-tracker.d.ts +17 -0
- package/lib/components/overlay/components/menu/public-api.d.ts +16 -0
- package/lib/components/overlay/components/public-api.d.ts +1 -0
- package/package.json +2 -2
package/CHANGELOG.md
CHANGED
|
@@ -1,5 +1,15 @@
|
|
|
1
1
|
# @ethlete/cdk
|
|
2
2
|
|
|
3
|
+
## 2.9.0
|
|
4
|
+
|
|
5
|
+
### Minor Changes
|
|
6
|
+
|
|
7
|
+
- [`e7fa66f`](https://github.com/ethlete-io/ethdk/commit/e7fa66f861a95a837daec1f566d02f328d0d5752) Thanks [@TomTomB](https://github.com/TomTomB)! - Add experimental menu component
|
|
8
|
+
|
|
9
|
+
### Patch Changes
|
|
10
|
+
|
|
11
|
+
- [`bd59856`](https://github.com/ethlete-io/ethdk/commit/bd5985620bcb4b0de63d8bc982907a45983a9e67) Thanks [@TomTomB](https://github.com/TomTomB)! - Update scrollable intersection states if children change
|
|
12
|
+
|
|
3
13
|
## 2.8.1
|
|
4
14
|
|
|
5
15
|
### Patch Changes
|
|
@@ -0,0 +1,163 @@
|
|
|
1
|
+
/* eslint-disable @angular-eslint/directive-class-suffix */
|
|
2
|
+
/* eslint-disable @angular-eslint/no-output-rename */
|
|
3
|
+
/* eslint-disable @angular-eslint/no-outputs-metadata-property */
|
|
4
|
+
/* eslint-disable @angular-eslint/no-input-rename */
|
|
5
|
+
/* eslint-disable @angular-eslint/no-inputs-metadata-property */
|
|
6
|
+
import { Directionality } from '@angular/cdk/bidi';
|
|
7
|
+
import { coerceBooleanProperty } from '@angular/cdk/coercion';
|
|
8
|
+
import { Overlay, OverlayConfig, STANDARD_DROPDOWN_BELOW_POSITIONS, } from '@angular/cdk/overlay';
|
|
9
|
+
import { _getEventTarget } from '@angular/cdk/platform';
|
|
10
|
+
import { Directive, Injectable, Input, inject } from '@angular/core';
|
|
11
|
+
import { merge, partition } from 'rxjs';
|
|
12
|
+
import { skip, takeUntil } from 'rxjs/operators';
|
|
13
|
+
import { MENU_STACK, MenuStack } from './menu-stack';
|
|
14
|
+
import { CdkMenuTriggerBase, MENU_TRIGGER } from './menu-trigger-base';
|
|
15
|
+
import * as i0 from "@angular/core";
|
|
16
|
+
const CONTEXT_MENU_POSITIONS = STANDARD_DROPDOWN_BELOW_POSITIONS.map((position) => {
|
|
17
|
+
const offsetX = position.overlayX === 'start' ? 2 : -2;
|
|
18
|
+
const offsetY = position.overlayY === 'top' ? 2 : -2;
|
|
19
|
+
return { ...position, offsetX, offsetY };
|
|
20
|
+
});
|
|
21
|
+
export class ContextMenuTracker {
|
|
22
|
+
update(trigger) {
|
|
23
|
+
if (ContextMenuTracker._openContextMenuTrigger !== trigger) {
|
|
24
|
+
ContextMenuTracker._openContextMenuTrigger?.close();
|
|
25
|
+
ContextMenuTracker._openContextMenuTrigger = trigger;
|
|
26
|
+
}
|
|
27
|
+
}
|
|
28
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.1.1", ngImport: i0, type: ContextMenuTracker, deps: [], target: i0.ɵɵFactoryTarget.Injectable }); }
|
|
29
|
+
static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "16.1.1", ngImport: i0, type: ContextMenuTracker, providedIn: 'root' }); }
|
|
30
|
+
}
|
|
31
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.1.1", ngImport: i0, type: ContextMenuTracker, decorators: [{
|
|
32
|
+
type: Injectable,
|
|
33
|
+
args: [{ providedIn: 'root' }]
|
|
34
|
+
}] });
|
|
35
|
+
export class CdkContextMenuTrigger extends CdkMenuTriggerBase {
|
|
36
|
+
get disabled() {
|
|
37
|
+
return this._disabled;
|
|
38
|
+
}
|
|
39
|
+
set disabled(value) {
|
|
40
|
+
this._disabled = coerceBooleanProperty(value);
|
|
41
|
+
}
|
|
42
|
+
constructor() {
|
|
43
|
+
super();
|
|
44
|
+
this._overlay = inject(Overlay);
|
|
45
|
+
this._directionality = inject(Directionality, { optional: true });
|
|
46
|
+
this._contextMenuTracker = inject(ContextMenuTracker);
|
|
47
|
+
this._disabled = false;
|
|
48
|
+
this._setMenuStackCloseListener();
|
|
49
|
+
}
|
|
50
|
+
open(coordinates) {
|
|
51
|
+
this._open(coordinates, false);
|
|
52
|
+
}
|
|
53
|
+
close() {
|
|
54
|
+
this.menuStack.closeAll();
|
|
55
|
+
}
|
|
56
|
+
_openOnContextMenu(event) {
|
|
57
|
+
if (!this.disabled) {
|
|
58
|
+
event.preventDefault();
|
|
59
|
+
event.stopPropagation();
|
|
60
|
+
this._contextMenuTracker.update(this);
|
|
61
|
+
this._open({ x: event.clientX, y: event.clientY }, true);
|
|
62
|
+
if (event.button === 2) {
|
|
63
|
+
this.childMenu?.focusFirstItem('mouse');
|
|
64
|
+
}
|
|
65
|
+
else if (event.button === 0) {
|
|
66
|
+
this.childMenu?.focusFirstItem('keyboard');
|
|
67
|
+
}
|
|
68
|
+
else {
|
|
69
|
+
this.childMenu?.focusFirstItem('program');
|
|
70
|
+
}
|
|
71
|
+
}
|
|
72
|
+
}
|
|
73
|
+
_getOverlayConfig(coordinates) {
|
|
74
|
+
return new OverlayConfig({
|
|
75
|
+
positionStrategy: this._getOverlayPositionStrategy(coordinates),
|
|
76
|
+
scrollStrategy: this._overlay.scrollStrategies.reposition(),
|
|
77
|
+
direction: this._directionality || undefined,
|
|
78
|
+
});
|
|
79
|
+
}
|
|
80
|
+
_getOverlayPositionStrategy(coordinates) {
|
|
81
|
+
return this._overlay
|
|
82
|
+
.position()
|
|
83
|
+
.flexibleConnectedTo(coordinates)
|
|
84
|
+
.withLockedPosition()
|
|
85
|
+
.withGrowAfterOpen()
|
|
86
|
+
.withPositions(this.menuPosition ?? CONTEXT_MENU_POSITIONS);
|
|
87
|
+
}
|
|
88
|
+
_setMenuStackCloseListener() {
|
|
89
|
+
this.menuStack.closed.pipe(takeUntil(this.destroyed)).subscribe(({ item }) => {
|
|
90
|
+
if (item === this.childMenu && this.isOpen()) {
|
|
91
|
+
this.closed.next();
|
|
92
|
+
this.overlayRef.detach();
|
|
93
|
+
}
|
|
94
|
+
});
|
|
95
|
+
}
|
|
96
|
+
_subscribeToOutsideClicks(ignoreFirstAuxClick) {
|
|
97
|
+
if (this.overlayRef) {
|
|
98
|
+
let outsideClicks = this.overlayRef.outsidePointerEvents();
|
|
99
|
+
if (ignoreFirstAuxClick) {
|
|
100
|
+
const [auxClicks, nonAuxClicks] = partition(outsideClicks, ({ type }) => type === 'auxclick');
|
|
101
|
+
outsideClicks = merge(nonAuxClicks, auxClicks.pipe(skip(1)));
|
|
102
|
+
}
|
|
103
|
+
outsideClicks.pipe(takeUntil(this.stopOutsideClicksListener)).subscribe((event) => {
|
|
104
|
+
if (!this.isElementInsideMenuStack(_getEventTarget(event))) {
|
|
105
|
+
this.menuStack.closeAll();
|
|
106
|
+
}
|
|
107
|
+
});
|
|
108
|
+
}
|
|
109
|
+
}
|
|
110
|
+
_open(coordinates, ignoreFirstOutsideAuxClick) {
|
|
111
|
+
if (this.disabled) {
|
|
112
|
+
return;
|
|
113
|
+
}
|
|
114
|
+
if (this.isOpen()) {
|
|
115
|
+
this.menuStack.closeSubMenuOf(this.childMenu);
|
|
116
|
+
this.overlayRef.getConfig().positionStrategy.setOrigin(coordinates);
|
|
117
|
+
this.overlayRef.updatePosition();
|
|
118
|
+
}
|
|
119
|
+
else {
|
|
120
|
+
this.opened.next();
|
|
121
|
+
if (this.overlayRef) {
|
|
122
|
+
this.overlayRef.getConfig().positionStrategy.setOrigin(coordinates);
|
|
123
|
+
this.overlayRef.updatePosition();
|
|
124
|
+
}
|
|
125
|
+
else {
|
|
126
|
+
this.overlayRef = this._overlay.create(this._getOverlayConfig(coordinates));
|
|
127
|
+
}
|
|
128
|
+
this.overlayRef.attach(this.getMenuContentPortal());
|
|
129
|
+
this._subscribeToOutsideClicks(ignoreFirstOutsideAuxClick);
|
|
130
|
+
}
|
|
131
|
+
}
|
|
132
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.1.1", ngImport: i0, type: CdkContextMenuTrigger, deps: [], target: i0.ɵɵFactoryTarget.Directive }); }
|
|
133
|
+
static { this.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "16.1.1", type: CdkContextMenuTrigger, isStandalone: true, selector: "[cdkContextMenuTriggerFor]", inputs: { menuTemplateRef: ["cdkContextMenuTriggerFor", "menuTemplateRef"], menuPosition: ["cdkContextMenuPosition", "menuPosition"], menuData: ["cdkContextMenuTriggerData", "menuData"], disabled: ["cdkContextMenuDisabled", "disabled"] }, outputs: { opened: "cdkContextMenuOpened", closed: "cdkContextMenuClosed" }, host: { listeners: { "contextmenu": "_openOnContextMenu($event)" }, properties: { "attr.data-cdk-menu-stack-id": "null" } }, providers: [
|
|
134
|
+
{ provide: MENU_TRIGGER, useExisting: CdkContextMenuTrigger },
|
|
135
|
+
{ provide: MENU_STACK, useClass: MenuStack },
|
|
136
|
+
], exportAs: ["cdkContextMenuTriggerFor"], usesInheritance: true, ngImport: i0 }); }
|
|
137
|
+
}
|
|
138
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.1.1", ngImport: i0, type: CdkContextMenuTrigger, decorators: [{
|
|
139
|
+
type: Directive,
|
|
140
|
+
args: [{
|
|
141
|
+
selector: '[cdkContextMenuTriggerFor]',
|
|
142
|
+
exportAs: 'cdkContextMenuTriggerFor',
|
|
143
|
+
standalone: true,
|
|
144
|
+
host: {
|
|
145
|
+
'[attr.data-cdk-menu-stack-id]': 'null',
|
|
146
|
+
'(contextmenu)': '_openOnContextMenu($event)',
|
|
147
|
+
},
|
|
148
|
+
inputs: [
|
|
149
|
+
'menuTemplateRef: cdkContextMenuTriggerFor',
|
|
150
|
+
'menuPosition: cdkContextMenuPosition',
|
|
151
|
+
'menuData: cdkContextMenuTriggerData',
|
|
152
|
+
],
|
|
153
|
+
outputs: ['opened: cdkContextMenuOpened', 'closed: cdkContextMenuClosed'],
|
|
154
|
+
providers: [
|
|
155
|
+
{ provide: MENU_TRIGGER, useExisting: CdkContextMenuTrigger },
|
|
156
|
+
{ provide: MENU_STACK, useClass: MenuStack },
|
|
157
|
+
],
|
|
158
|
+
}]
|
|
159
|
+
}], ctorParameters: function () { return []; }, propDecorators: { disabled: [{
|
|
160
|
+
type: Input,
|
|
161
|
+
args: ['cdkContextMenuDisabled']
|
|
162
|
+
}] } });
|
|
163
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"context-menu-trigger.js","sourceRoot":"","sources":["../../../../../../../../../libs/cdk/src/lib/components/overlay/components/menu/context-menu-trigger.ts"],"names":[],"mappings":"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","sourcesContent":["/* eslint-disable @angular-eslint/directive-class-suffix */\n/* eslint-disable @angular-eslint/no-output-rename */\n/* eslint-disable @angular-eslint/no-outputs-metadata-property */\n/* eslint-disable @angular-eslint/no-input-rename */\n/* eslint-disable @angular-eslint/no-inputs-metadata-property */\n\nimport { Directionality } from '@angular/cdk/bidi';\nimport { BooleanInput, coerceBooleanProperty } from '@angular/cdk/coercion';\nimport {\n  FlexibleConnectedPositionStrategy,\n  Overlay,\n  OverlayConfig,\n  STANDARD_DROPDOWN_BELOW_POSITIONS,\n} from '@angular/cdk/overlay';\nimport { _getEventTarget } from '@angular/cdk/platform';\nimport { Directive, Injectable, Input, OnDestroy, inject } from '@angular/core';\nimport { merge, partition } from 'rxjs';\nimport { skip, takeUntil } from 'rxjs/operators';\nimport { MENU_STACK, MenuStack } from './menu-stack';\nimport { CdkMenuTriggerBase, MENU_TRIGGER } from './menu-trigger-base';\n\nconst CONTEXT_MENU_POSITIONS = STANDARD_DROPDOWN_BELOW_POSITIONS.map((position) => {\n  const offsetX = position.overlayX === 'start' ? 2 : -2;\n  const offsetY = position.overlayY === 'top' ? 2 : -2;\n  return { ...position, offsetX, offsetY };\n});\n\n@Injectable({ providedIn: 'root' })\nexport class ContextMenuTracker {\n  private static _openContextMenuTrigger?: CdkContextMenuTrigger;\n\n  update(trigger: CdkContextMenuTrigger) {\n    if (ContextMenuTracker._openContextMenuTrigger !== trigger) {\n      ContextMenuTracker._openContextMenuTrigger?.close();\n      ContextMenuTracker._openContextMenuTrigger = trigger;\n    }\n  }\n}\n\nexport type ContextMenuCoordinates = { x: number; y: number };\n\n@Directive({\n  selector: '[cdkContextMenuTriggerFor]',\n  exportAs: 'cdkContextMenuTriggerFor',\n  standalone: true,\n  host: {\n    '[attr.data-cdk-menu-stack-id]': 'null',\n    '(contextmenu)': '_openOnContextMenu($event)',\n  },\n  inputs: [\n    'menuTemplateRef: cdkContextMenuTriggerFor',\n    'menuPosition: cdkContextMenuPosition',\n    'menuData: cdkContextMenuTriggerData',\n  ],\n  outputs: ['opened: cdkContextMenuOpened', 'closed: cdkContextMenuClosed'],\n  providers: [\n    { provide: MENU_TRIGGER, useExisting: CdkContextMenuTrigger },\n    { provide: MENU_STACK, useClass: MenuStack },\n  ],\n})\nexport class CdkContextMenuTrigger extends CdkMenuTriggerBase implements OnDestroy {\n  private readonly _overlay = inject(Overlay);\n\n  private readonly _directionality = inject(Directionality, { optional: true });\n\n  private readonly _contextMenuTracker = inject(ContextMenuTracker);\n\n  @Input('cdkContextMenuDisabled')\n  get disabled(): boolean {\n    return this._disabled;\n  }\n  set disabled(value: BooleanInput) {\n    this._disabled = coerceBooleanProperty(value);\n  }\n  private _disabled = false;\n\n  constructor() {\n    super();\n    this._setMenuStackCloseListener();\n  }\n\n  open(coordinates: ContextMenuCoordinates) {\n    this._open(coordinates, false);\n  }\n\n  close() {\n    this.menuStack.closeAll();\n  }\n\n  _openOnContextMenu(event: MouseEvent) {\n    if (!this.disabled) {\n      event.preventDefault();\n\n      event.stopPropagation();\n\n      this._contextMenuTracker.update(this);\n      this._open({ x: event.clientX, y: event.clientY }, true);\n\n      if (event.button === 2) {\n        this.childMenu?.focusFirstItem('mouse');\n      } else if (event.button === 0) {\n        this.childMenu?.focusFirstItem('keyboard');\n      } else {\n        this.childMenu?.focusFirstItem('program');\n      }\n    }\n  }\n\n  private _getOverlayConfig(coordinates: ContextMenuCoordinates) {\n    return new OverlayConfig({\n      positionStrategy: this._getOverlayPositionStrategy(coordinates),\n      scrollStrategy: this._overlay.scrollStrategies.reposition(),\n      direction: this._directionality || undefined,\n    });\n  }\n\n  private _getOverlayPositionStrategy(coordinates: ContextMenuCoordinates): FlexibleConnectedPositionStrategy {\n    return this._overlay\n      .position()\n      .flexibleConnectedTo(coordinates)\n      .withLockedPosition()\n      .withGrowAfterOpen()\n      .withPositions(this.menuPosition ?? CONTEXT_MENU_POSITIONS);\n  }\n\n  private _setMenuStackCloseListener() {\n    this.menuStack.closed.pipe(takeUntil(this.destroyed)).subscribe(({ item }) => {\n      if (item === this.childMenu && this.isOpen()) {\n        this.closed.next();\n        this.overlayRef!.detach();\n      }\n    });\n  }\n\n  private _subscribeToOutsideClicks(ignoreFirstAuxClick: boolean) {\n    if (this.overlayRef) {\n      let outsideClicks = this.overlayRef.outsidePointerEvents();\n\n      if (ignoreFirstAuxClick) {\n        const [auxClicks, nonAuxClicks] = partition(outsideClicks, ({ type }) => type === 'auxclick');\n        outsideClicks = merge(nonAuxClicks, auxClicks.pipe(skip(1)));\n      }\n      outsideClicks.pipe(takeUntil(this.stopOutsideClicksListener)).subscribe((event) => {\n        if (!this.isElementInsideMenuStack(_getEventTarget(event)!)) {\n          this.menuStack.closeAll();\n        }\n      });\n    }\n  }\n\n  private _open(coordinates: ContextMenuCoordinates, ignoreFirstOutsideAuxClick: boolean) {\n    if (this.disabled) {\n      return;\n    }\n    if (this.isOpen()) {\n      this.menuStack.closeSubMenuOf(this.childMenu!);\n\n      (this.overlayRef!.getConfig().positionStrategy as FlexibleConnectedPositionStrategy).setOrigin(coordinates);\n      this.overlayRef!.updatePosition();\n    } else {\n      this.opened.next();\n\n      if (this.overlayRef) {\n        (this.overlayRef.getConfig().positionStrategy as FlexibleConnectedPositionStrategy).setOrigin(coordinates);\n        this.overlayRef.updatePosition();\n      } else {\n        this.overlayRef = this._overlay.create(this._getOverlayConfig(coordinates));\n      }\n\n      this.overlayRef.attach(this.getMenuContentPortal());\n      this._subscribeToOutsideClicks(ignoreFirstOutsideAuxClick);\n    }\n  }\n}\n"]}
|
|
@@ -0,0 +1,140 @@
|
|
|
1
|
+
/* eslint-disable @angular-eslint/directive-class-suffix */
|
|
2
|
+
/* eslint-disable @angular-eslint/no-output-rename */
|
|
3
|
+
/* eslint-disable @angular-eslint/no-outputs-metadata-property */
|
|
4
|
+
/* eslint-disable @angular-eslint/no-input-rename */
|
|
5
|
+
/* eslint-disable @angular-eslint/no-inputs-metadata-property */
|
|
6
|
+
import { Directive, inject, Injectable, InjectionToken, isDevMode, NgZone } from '@angular/core';
|
|
7
|
+
import { fromEvent, Subject } from 'rxjs';
|
|
8
|
+
import { filter, takeUntil } from 'rxjs/operators';
|
|
9
|
+
import { throwMissingMenuReference, throwMissingPointerFocusTracker } from './menu-errors';
|
|
10
|
+
import * as i0 from "@angular/core";
|
|
11
|
+
export const MENU_AIM = new InjectionToken('cdk-menu-aim');
|
|
12
|
+
const MOUSE_MOVE_SAMPLE_FREQUENCY = 3;
|
|
13
|
+
const NUM_POINTS = 5;
|
|
14
|
+
const CLOSE_DELAY = 300;
|
|
15
|
+
function getSlope(a, b) {
|
|
16
|
+
return (b.y - a.y) / (b.x - a.x);
|
|
17
|
+
}
|
|
18
|
+
function getYIntercept(point, slope) {
|
|
19
|
+
return point.y - slope * point.x;
|
|
20
|
+
}
|
|
21
|
+
function isWithinSubmenu(submenuPoints, m, b) {
|
|
22
|
+
const { left, right, top, bottom } = submenuPoints;
|
|
23
|
+
return ((m * left + b >= top && m * left + b <= bottom) ||
|
|
24
|
+
(m * right + b >= top && m * right + b <= bottom) ||
|
|
25
|
+
((top - b) / m >= left && (top - b) / m <= right) ||
|
|
26
|
+
((bottom - b) / m >= left && (bottom - b) / m <= right));
|
|
27
|
+
}
|
|
28
|
+
export class TargetMenuAim {
|
|
29
|
+
constructor() {
|
|
30
|
+
this._ngZone = inject(NgZone);
|
|
31
|
+
this._points = [];
|
|
32
|
+
this._menu = null;
|
|
33
|
+
this._pointerTracker = null;
|
|
34
|
+
this._timeoutId = null;
|
|
35
|
+
this._destroyed = new Subject();
|
|
36
|
+
}
|
|
37
|
+
ngOnDestroy() {
|
|
38
|
+
this._destroyed.next();
|
|
39
|
+
this._destroyed.complete();
|
|
40
|
+
}
|
|
41
|
+
initialize(menu, pointerTracker) {
|
|
42
|
+
this._menu = menu;
|
|
43
|
+
this._pointerTracker = pointerTracker;
|
|
44
|
+
this._subscribeToMouseMoves();
|
|
45
|
+
}
|
|
46
|
+
toggle(doToggle) {
|
|
47
|
+
if (!this._menu)
|
|
48
|
+
return;
|
|
49
|
+
if (this._menu.orientation === 'horizontal') {
|
|
50
|
+
doToggle();
|
|
51
|
+
}
|
|
52
|
+
this._checkConfigured();
|
|
53
|
+
const siblingItemIsWaiting = !!this._timeoutId;
|
|
54
|
+
const hasPoints = this._points.length > 1;
|
|
55
|
+
if (hasPoints && !siblingItemIsWaiting) {
|
|
56
|
+
if (this._isMovingToSubmenu()) {
|
|
57
|
+
this._startTimeout(doToggle);
|
|
58
|
+
}
|
|
59
|
+
else {
|
|
60
|
+
doToggle();
|
|
61
|
+
}
|
|
62
|
+
}
|
|
63
|
+
else if (!siblingItemIsWaiting) {
|
|
64
|
+
doToggle();
|
|
65
|
+
}
|
|
66
|
+
}
|
|
67
|
+
_startTimeout(doToggle) {
|
|
68
|
+
const timeoutId = window.setTimeout(() => {
|
|
69
|
+
if (!this._pointerTracker)
|
|
70
|
+
return;
|
|
71
|
+
if (this._pointerTracker.activeElement && timeoutId === this._timeoutId) {
|
|
72
|
+
doToggle();
|
|
73
|
+
}
|
|
74
|
+
this._timeoutId = null;
|
|
75
|
+
}, CLOSE_DELAY);
|
|
76
|
+
this._timeoutId = timeoutId;
|
|
77
|
+
}
|
|
78
|
+
_isMovingToSubmenu() {
|
|
79
|
+
const submenuPoints = this._getSubmenuBounds();
|
|
80
|
+
if (!submenuPoints) {
|
|
81
|
+
return false;
|
|
82
|
+
}
|
|
83
|
+
let numMoving = 0;
|
|
84
|
+
const currPoint = this._points[this._points.length - 1];
|
|
85
|
+
for (let i = this._points.length - 2; i >= 0; i--) {
|
|
86
|
+
const previous = this._points[i];
|
|
87
|
+
const slope = getSlope(currPoint, previous);
|
|
88
|
+
if (isWithinSubmenu(submenuPoints, slope, getYIntercept(currPoint, slope))) {
|
|
89
|
+
numMoving++;
|
|
90
|
+
}
|
|
91
|
+
}
|
|
92
|
+
return numMoving >= Math.floor(NUM_POINTS / 2);
|
|
93
|
+
}
|
|
94
|
+
_getSubmenuBounds() {
|
|
95
|
+
return this._pointerTracker?.previousElement?.getMenu()?.nativeElement.getBoundingClientRect();
|
|
96
|
+
}
|
|
97
|
+
_checkConfigured() {
|
|
98
|
+
if (isDevMode()) {
|
|
99
|
+
if (!this._pointerTracker) {
|
|
100
|
+
throwMissingPointerFocusTracker();
|
|
101
|
+
}
|
|
102
|
+
if (!this._menu) {
|
|
103
|
+
throwMissingMenuReference();
|
|
104
|
+
}
|
|
105
|
+
}
|
|
106
|
+
}
|
|
107
|
+
_subscribeToMouseMoves() {
|
|
108
|
+
this._ngZone.runOutsideAngular(() => {
|
|
109
|
+
if (!this._menu)
|
|
110
|
+
return;
|
|
111
|
+
fromEvent(this._menu.nativeElement, 'mousemove')
|
|
112
|
+
.pipe(filter((_, index) => index % MOUSE_MOVE_SAMPLE_FREQUENCY === 0), takeUntil(this._destroyed))
|
|
113
|
+
.subscribe((event) => {
|
|
114
|
+
this._points.push({ x: event.clientX, y: event.clientY });
|
|
115
|
+
if (this._points.length > NUM_POINTS) {
|
|
116
|
+
this._points.shift();
|
|
117
|
+
}
|
|
118
|
+
});
|
|
119
|
+
});
|
|
120
|
+
}
|
|
121
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.1.1", ngImport: i0, type: TargetMenuAim, deps: [], target: i0.ɵɵFactoryTarget.Injectable }); }
|
|
122
|
+
static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "16.1.1", ngImport: i0, type: TargetMenuAim }); }
|
|
123
|
+
}
|
|
124
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.1.1", ngImport: i0, type: TargetMenuAim, decorators: [{
|
|
125
|
+
type: Injectable
|
|
126
|
+
}] });
|
|
127
|
+
export class CdkTargetMenuAim {
|
|
128
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.1.1", ngImport: i0, type: CdkTargetMenuAim, deps: [], target: i0.ɵɵFactoryTarget.Directive }); }
|
|
129
|
+
static { this.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "16.1.1", type: CdkTargetMenuAim, isStandalone: true, selector: "[cdkTargetMenuAim]", providers: [{ provide: MENU_AIM, useClass: TargetMenuAim }], exportAs: ["cdkTargetMenuAim"], ngImport: i0 }); }
|
|
130
|
+
}
|
|
131
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.1.1", ngImport: i0, type: CdkTargetMenuAim, decorators: [{
|
|
132
|
+
type: Directive,
|
|
133
|
+
args: [{
|
|
134
|
+
selector: '[cdkTargetMenuAim]',
|
|
135
|
+
exportAs: 'cdkTargetMenuAim',
|
|
136
|
+
standalone: true,
|
|
137
|
+
providers: [{ provide: MENU_AIM, useClass: TargetMenuAim }],
|
|
138
|
+
}]
|
|
139
|
+
}] });
|
|
140
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"menu-aim.js","sourceRoot":"","sources":["../../../../../../../../../libs/cdk/src/lib/components/overlay/components/menu/menu-aim.ts"],"names":[],"mappings":"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","sourcesContent":["/* eslint-disable @angular-eslint/directive-class-suffix */\n/* eslint-disable @angular-eslint/no-output-rename */\n/* eslint-disable @angular-eslint/no-outputs-metadata-property */\n/* eslint-disable @angular-eslint/no-input-rename */\n/* eslint-disable @angular-eslint/no-inputs-metadata-property */\n\nimport { Directive, inject, Injectable, InjectionToken, isDevMode, NgZone, OnDestroy } from '@angular/core';\nimport { fromEvent, Subject } from 'rxjs';\nimport { filter, takeUntil } from 'rxjs/operators';\nimport { throwMissingMenuReference, throwMissingPointerFocusTracker } from './menu-errors';\nimport { Menu } from './menu-interface';\nimport { FocusableElement, PointerFocusTracker } from './pointer-focus-tracker';\n\nexport interface MenuAim {\n  initialize(menu: Menu, pointerTracker: PointerFocusTracker<FocusableElement & Toggler>): void;\n\n  toggle(doToggle: () => void): void;\n}\n\nexport const MENU_AIM = new InjectionToken<MenuAim>('cdk-menu-aim');\n\nconst MOUSE_MOVE_SAMPLE_FREQUENCY = 3;\n\nconst NUM_POINTS = 5;\n\nconst CLOSE_DELAY = 300;\n\nexport interface Toggler {\n  getMenu(): Menu | undefined;\n}\n\nfunction getSlope(a: Point, b: Point) {\n  return (b.y - a.y) / (b.x - a.x);\n}\n\nfunction getYIntercept(point: Point, slope: number) {\n  return point.y - slope * point.x;\n}\n\ntype Point = { x: number; y: number };\n\nfunction isWithinSubmenu(submenuPoints: DOMRect, m: number, b: number) {\n  const { left, right, top, bottom } = submenuPoints;\n\n  return (\n    (m * left + b >= top && m * left + b <= bottom) ||\n    (m * right + b >= top && m * right + b <= bottom) ||\n    ((top - b) / m >= left && (top - b) / m <= right) ||\n    ((bottom - b) / m >= left && (bottom - b) / m <= right)\n  );\n}\n\n@Injectable()\nexport class TargetMenuAim implements MenuAim, OnDestroy {\n  private readonly _ngZone = inject(NgZone);\n\n  private readonly _points: Point[] = [];\n\n  private _menu: Menu | null = null;\n\n  private _pointerTracker: PointerFocusTracker<Toggler & FocusableElement> | null = null;\n\n  private _timeoutId: number | null = null;\n\n  private readonly _destroyed: Subject<void> = new Subject();\n\n  ngOnDestroy() {\n    this._destroyed.next();\n    this._destroyed.complete();\n  }\n\n  initialize(menu: Menu, pointerTracker: PointerFocusTracker<FocusableElement & Toggler>) {\n    this._menu = menu;\n    this._pointerTracker = pointerTracker;\n    this._subscribeToMouseMoves();\n  }\n\n  toggle(doToggle: () => void) {\n    if (!this._menu) return;\n\n    if (this._menu.orientation === 'horizontal') {\n      doToggle();\n    }\n\n    this._checkConfigured();\n\n    const siblingItemIsWaiting = !!this._timeoutId;\n    const hasPoints = this._points.length > 1;\n\n    if (hasPoints && !siblingItemIsWaiting) {\n      if (this._isMovingToSubmenu()) {\n        this._startTimeout(doToggle);\n      } else {\n        doToggle();\n      }\n    } else if (!siblingItemIsWaiting) {\n      doToggle();\n    }\n  }\n\n  private _startTimeout(doToggle: () => void) {\n    const timeoutId = window.setTimeout(() => {\n      if (!this._pointerTracker) return;\n\n      if (this._pointerTracker.activeElement && timeoutId === this._timeoutId) {\n        doToggle();\n      }\n      this._timeoutId = null;\n    }, CLOSE_DELAY);\n\n    this._timeoutId = timeoutId;\n  }\n\n  private _isMovingToSubmenu() {\n    const submenuPoints = this._getSubmenuBounds();\n    if (!submenuPoints) {\n      return false;\n    }\n\n    let numMoving = 0;\n    const currPoint = this._points[this._points.length - 1];\n\n    for (let i = this._points.length - 2; i >= 0; i--) {\n      const previous = this._points[i];\n      const slope = getSlope(currPoint, previous);\n      if (isWithinSubmenu(submenuPoints, slope, getYIntercept(currPoint, slope))) {\n        numMoving++;\n      }\n    }\n    return numMoving >= Math.floor(NUM_POINTS / 2);\n  }\n\n  private _getSubmenuBounds(): DOMRect | undefined {\n    return this._pointerTracker?.previousElement?.getMenu()?.nativeElement.getBoundingClientRect();\n  }\n\n  private _checkConfigured() {\n    if (isDevMode()) {\n      if (!this._pointerTracker) {\n        throwMissingPointerFocusTracker();\n      }\n      if (!this._menu) {\n        throwMissingMenuReference();\n      }\n    }\n  }\n\n  private _subscribeToMouseMoves() {\n    this._ngZone.runOutsideAngular(() => {\n      if (!this._menu) return;\n\n      fromEvent<MouseEvent>(this._menu.nativeElement, 'mousemove')\n        .pipe(\n          filter((_: MouseEvent, index: number) => index % MOUSE_MOVE_SAMPLE_FREQUENCY === 0),\n          takeUntil(this._destroyed),\n        )\n        .subscribe((event: MouseEvent) => {\n          this._points.push({ x: event.clientX, y: event.clientY });\n          if (this._points.length > NUM_POINTS) {\n            this._points.shift();\n          }\n        });\n    });\n  }\n}\n\n@Directive({\n  selector: '[cdkTargetMenuAim]',\n  exportAs: 'cdkTargetMenuAim',\n  standalone: true,\n  providers: [{ provide: MENU_AIM, useClass: TargetMenuAim }],\n})\nexport class CdkTargetMenuAim {}\n"]}
|
|
@@ -0,0 +1,113 @@
|
|
|
1
|
+
/* eslint-disable @angular-eslint/directive-class-suffix */
|
|
2
|
+
/* eslint-disable @angular-eslint/no-output-rename */
|
|
3
|
+
/* eslint-disable @angular-eslint/no-outputs-metadata-property */
|
|
4
|
+
/* eslint-disable @angular-eslint/no-input-rename */
|
|
5
|
+
/* eslint-disable @angular-eslint/no-inputs-metadata-property */
|
|
6
|
+
import { DOWN_ARROW, ESCAPE, LEFT_ARROW, RIGHT_ARROW, TAB, UP_ARROW, hasModifierKey } from '@angular/cdk/keycodes';
|
|
7
|
+
import { Directive } from '@angular/core';
|
|
8
|
+
import { takeUntil } from 'rxjs/operators';
|
|
9
|
+
import { CdkMenuBase } from './menu-base';
|
|
10
|
+
import { CdkMenuGroup } from './menu-group';
|
|
11
|
+
import { CDK_MENU } from './menu-interface';
|
|
12
|
+
import { MENU_STACK, MenuStack } from './menu-stack';
|
|
13
|
+
import * as i0 from "@angular/core";
|
|
14
|
+
export class CdkMenuBar extends CdkMenuBase {
|
|
15
|
+
constructor() {
|
|
16
|
+
super(...arguments);
|
|
17
|
+
this.orientation = 'horizontal';
|
|
18
|
+
this.isInline = true;
|
|
19
|
+
}
|
|
20
|
+
ngAfterContentInit() {
|
|
21
|
+
super.ngAfterContentInit();
|
|
22
|
+
this._subscribeToMenuStackEmptied();
|
|
23
|
+
}
|
|
24
|
+
_handleKeyEvent(event) {
|
|
25
|
+
const keyManager = this.keyManager;
|
|
26
|
+
if (!keyManager)
|
|
27
|
+
return;
|
|
28
|
+
switch (event.keyCode) {
|
|
29
|
+
case UP_ARROW:
|
|
30
|
+
case DOWN_ARROW:
|
|
31
|
+
case LEFT_ARROW:
|
|
32
|
+
case RIGHT_ARROW:
|
|
33
|
+
if (!hasModifierKey(event)) {
|
|
34
|
+
const horizontalArrows = event.keyCode === LEFT_ARROW || event.keyCode === RIGHT_ARROW;
|
|
35
|
+
if (horizontalArrows) {
|
|
36
|
+
event.preventDefault();
|
|
37
|
+
const prevIsOpen = keyManager.activeItem?.isMenuOpen();
|
|
38
|
+
keyManager.activeItem?.getMenuTrigger()?.close();
|
|
39
|
+
keyManager.setFocusOrigin('keyboard');
|
|
40
|
+
keyManager.onKeydown(event);
|
|
41
|
+
if (prevIsOpen) {
|
|
42
|
+
keyManager.activeItem?.getMenuTrigger()?.open();
|
|
43
|
+
}
|
|
44
|
+
}
|
|
45
|
+
}
|
|
46
|
+
break;
|
|
47
|
+
case ESCAPE:
|
|
48
|
+
if (!hasModifierKey(event)) {
|
|
49
|
+
event.preventDefault();
|
|
50
|
+
keyManager.activeItem?.getMenuTrigger()?.close();
|
|
51
|
+
}
|
|
52
|
+
break;
|
|
53
|
+
case TAB:
|
|
54
|
+
if (!hasModifierKey(event, 'altKey', 'metaKey', 'ctrlKey')) {
|
|
55
|
+
keyManager.activeItem?.getMenuTrigger()?.close();
|
|
56
|
+
}
|
|
57
|
+
break;
|
|
58
|
+
default:
|
|
59
|
+
keyManager.onKeydown(event);
|
|
60
|
+
}
|
|
61
|
+
}
|
|
62
|
+
_toggleOpenMenu(focusNext) {
|
|
63
|
+
const keyManager = this.keyManager;
|
|
64
|
+
if (!keyManager)
|
|
65
|
+
return;
|
|
66
|
+
switch (focusNext) {
|
|
67
|
+
case 0 /* FocusNext.nextItem */:
|
|
68
|
+
keyManager.setFocusOrigin('keyboard');
|
|
69
|
+
keyManager.setNextItemActive();
|
|
70
|
+
keyManager.activeItem?.getMenuTrigger()?.open();
|
|
71
|
+
break;
|
|
72
|
+
case 1 /* FocusNext.previousItem */:
|
|
73
|
+
keyManager.setFocusOrigin('keyboard');
|
|
74
|
+
keyManager.setPreviousItemActive();
|
|
75
|
+
keyManager.activeItem?.getMenuTrigger()?.open();
|
|
76
|
+
break;
|
|
77
|
+
case 2 /* FocusNext.currentItem */:
|
|
78
|
+
if (keyManager.activeItem) {
|
|
79
|
+
keyManager.setFocusOrigin('keyboard');
|
|
80
|
+
keyManager.setActiveItem(keyManager.activeItem);
|
|
81
|
+
}
|
|
82
|
+
break;
|
|
83
|
+
}
|
|
84
|
+
}
|
|
85
|
+
_subscribeToMenuStackEmptied() {
|
|
86
|
+
this.menuStack?.emptied.pipe(takeUntil(this.destroyed)).subscribe((event) => this._toggleOpenMenu(event));
|
|
87
|
+
}
|
|
88
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.1.1", ngImport: i0, type: CdkMenuBar, deps: null, target: i0.ɵɵFactoryTarget.Directive }); }
|
|
89
|
+
static { this.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "16.1.1", type: CdkMenuBar, isStandalone: true, selector: "[cdkMenuBar]", host: { attributes: { "role": "menubar" }, listeners: { "keydown": "_handleKeyEvent($event)" }, classAttribute: "cdk-menu-bar" }, providers: [
|
|
90
|
+
{ provide: CdkMenuGroup, useExisting: CdkMenuBar },
|
|
91
|
+
{ provide: CDK_MENU, useExisting: CdkMenuBar },
|
|
92
|
+
{ provide: MENU_STACK, useFactory: () => MenuStack.inline('horizontal') },
|
|
93
|
+
], exportAs: ["cdkMenuBar"], usesInheritance: true, ngImport: i0 }); }
|
|
94
|
+
}
|
|
95
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.1.1", ngImport: i0, type: CdkMenuBar, decorators: [{
|
|
96
|
+
type: Directive,
|
|
97
|
+
args: [{
|
|
98
|
+
selector: '[cdkMenuBar]',
|
|
99
|
+
exportAs: 'cdkMenuBar',
|
|
100
|
+
standalone: true,
|
|
101
|
+
host: {
|
|
102
|
+
role: 'menubar',
|
|
103
|
+
class: 'cdk-menu-bar',
|
|
104
|
+
'(keydown)': '_handleKeyEvent($event)',
|
|
105
|
+
},
|
|
106
|
+
providers: [
|
|
107
|
+
{ provide: CdkMenuGroup, useExisting: CdkMenuBar },
|
|
108
|
+
{ provide: CDK_MENU, useExisting: CdkMenuBar },
|
|
109
|
+
{ provide: MENU_STACK, useFactory: () => MenuStack.inline('horizontal') },
|
|
110
|
+
],
|
|
111
|
+
}]
|
|
112
|
+
}] });
|
|
113
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"menu-bar.js","sourceRoot":"","sources":["../../../../../../../../../libs/cdk/src/lib/components/overlay/components/menu/menu-bar.ts"],"names":[],"mappings":"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","sourcesContent":["/* eslint-disable @angular-eslint/directive-class-suffix */\n/* eslint-disable @angular-eslint/no-output-rename */\n/* eslint-disable @angular-eslint/no-outputs-metadata-property */\n/* eslint-disable @angular-eslint/no-input-rename */\n/* eslint-disable @angular-eslint/no-inputs-metadata-property */\n\nimport { DOWN_ARROW, ESCAPE, LEFT_ARROW, RIGHT_ARROW, TAB, UP_ARROW, hasModifierKey } from '@angular/cdk/keycodes';\nimport { AfterContentInit, Directive } from '@angular/core';\nimport { takeUntil } from 'rxjs/operators';\nimport { CdkMenuBase } from './menu-base';\nimport { CdkMenuGroup } from './menu-group';\nimport { CDK_MENU } from './menu-interface';\nimport { FocusNext, MENU_STACK, MenuStack } from './menu-stack';\n\n@Directive({\n  selector: '[cdkMenuBar]',\n  exportAs: 'cdkMenuBar',\n  standalone: true,\n  host: {\n    role: 'menubar',\n    class: 'cdk-menu-bar',\n    '(keydown)': '_handleKeyEvent($event)',\n  },\n  providers: [\n    { provide: CdkMenuGroup, useExisting: CdkMenuBar },\n    { provide: CDK_MENU, useExisting: CdkMenuBar },\n    { provide: MENU_STACK, useFactory: () => MenuStack.inline('horizontal') },\n  ],\n})\nexport class CdkMenuBar extends CdkMenuBase implements AfterContentInit {\n  override readonly orientation = 'horizontal';\n\n  override readonly isInline = true;\n\n  override ngAfterContentInit() {\n    super.ngAfterContentInit();\n    this._subscribeToMenuStackEmptied();\n  }\n\n  _handleKeyEvent(event: KeyboardEvent) {\n    const keyManager = this.keyManager;\n\n    if (!keyManager) return;\n\n    switch (event.keyCode) {\n      case UP_ARROW:\n      case DOWN_ARROW:\n      case LEFT_ARROW:\n      case RIGHT_ARROW:\n        if (!hasModifierKey(event)) {\n          const horizontalArrows = event.keyCode === LEFT_ARROW || event.keyCode === RIGHT_ARROW;\n\n          if (horizontalArrows) {\n            event.preventDefault();\n\n            const prevIsOpen = keyManager.activeItem?.isMenuOpen();\n            keyManager.activeItem?.getMenuTrigger()?.close();\n\n            keyManager.setFocusOrigin('keyboard');\n            keyManager.onKeydown(event);\n            if (prevIsOpen) {\n              keyManager.activeItem?.getMenuTrigger()?.open();\n            }\n          }\n        }\n        break;\n\n      case ESCAPE:\n        if (!hasModifierKey(event)) {\n          event.preventDefault();\n          keyManager.activeItem?.getMenuTrigger()?.close();\n        }\n        break;\n\n      case TAB:\n        if (!hasModifierKey(event, 'altKey', 'metaKey', 'ctrlKey')) {\n          keyManager.activeItem?.getMenuTrigger()?.close();\n        }\n        break;\n\n      default:\n        keyManager.onKeydown(event);\n    }\n  }\n\n  private _toggleOpenMenu(focusNext: FocusNext | undefined) {\n    const keyManager = this.keyManager;\n\n    if (!keyManager) return;\n\n    switch (focusNext) {\n      case FocusNext.nextItem:\n        keyManager.setFocusOrigin('keyboard');\n        keyManager.setNextItemActive();\n        keyManager.activeItem?.getMenuTrigger()?.open();\n        break;\n\n      case FocusNext.previousItem:\n        keyManager.setFocusOrigin('keyboard');\n        keyManager.setPreviousItemActive();\n        keyManager.activeItem?.getMenuTrigger()?.open();\n        break;\n\n      case FocusNext.currentItem:\n        if (keyManager.activeItem) {\n          keyManager.setFocusOrigin('keyboard');\n          keyManager.setActiveItem(keyManager.activeItem);\n        }\n        break;\n    }\n  }\n\n  private _subscribeToMenuStackEmptied() {\n    this.menuStack?.emptied.pipe(takeUntil(this.destroyed)).subscribe((event) => this._toggleOpenMenu(event));\n  }\n}\n"]}
|