@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.
Files changed (43) hide show
  1. package/CHANGELOG.md +10 -0
  2. package/esm2022/lib/components/overlay/components/menu/context-menu-trigger.mjs +163 -0
  3. package/esm2022/lib/components/overlay/components/menu/menu-aim.mjs +140 -0
  4. package/esm2022/lib/components/overlay/components/menu/menu-bar.mjs +113 -0
  5. package/esm2022/lib/components/overlay/components/menu/menu-base.mjs +154 -0
  6. package/esm2022/lib/components/overlay/components/menu/menu-errors.mjs +7 -0
  7. package/esm2022/lib/components/overlay/components/menu/menu-group.mjs +26 -0
  8. package/esm2022/lib/components/overlay/components/menu/menu-interface.mjs +3 -0
  9. package/esm2022/lib/components/overlay/components/menu/menu-item-checkbox.mjs +39 -0
  10. package/esm2022/lib/components/overlay/components/menu/menu-item-radio.mjs +56 -0
  11. package/esm2022/lib/components/overlay/components/menu/menu-item-selectable.mjs +32 -0
  12. package/esm2022/lib/components/overlay/components/menu/menu-item.mjs +210 -0
  13. package/esm2022/lib/components/overlay/components/menu/menu-module.mjs +52 -0
  14. package/esm2022/lib/components/overlay/components/menu/menu-stack.mjs +101 -0
  15. package/esm2022/lib/components/overlay/components/menu/menu-trigger-base.mjs +77 -0
  16. package/esm2022/lib/components/overlay/components/menu/menu-trigger.mjs +240 -0
  17. package/esm2022/lib/components/overlay/components/menu/menu.mjs +116 -0
  18. package/esm2022/lib/components/overlay/components/menu/pointer-focus-tracker.mjs +26 -0
  19. package/esm2022/lib/components/overlay/components/menu/public-api.mjs +17 -0
  20. package/esm2022/lib/components/overlay/components/public-api.mjs +2 -1
  21. package/esm2022/lib/components/scrollable/components/scrollable/scrollable.component.mjs +3 -3
  22. package/fesm2022/ethlete-cdk.mjs +1561 -157
  23. package/fesm2022/ethlete-cdk.mjs.map +1 -1
  24. package/lib/components/overlay/components/menu/context-menu-trigger.d.ts +33 -0
  25. package/lib/components/overlay/components/menu/menu-aim.d.ts +34 -0
  26. package/lib/components/overlay/components/menu/menu-bar.d.ts +13 -0
  27. package/lib/components/overlay/components/menu/menu-base.d.ts +41 -0
  28. package/lib/components/overlay/components/menu/menu-errors.d.ts +2 -0
  29. package/lib/components/overlay/components/menu/menu-group.d.ts +5 -0
  30. package/lib/components/overlay/components/menu/menu-interface.d.ts +11 -0
  31. package/lib/components/overlay/components/menu/menu-item-checkbox.d.ts +9 -0
  32. package/lib/components/overlay/components/menu/menu-item-radio.d.ts +16 -0
  33. package/lib/components/overlay/components/menu/menu-item-selectable.d.ts +11 -0
  34. package/lib/components/overlay/components/menu/menu-item.d.ts +52 -0
  35. package/lib/components/overlay/components/menu/menu-module.d.ts +16 -0
  36. package/lib/components/overlay/components/menu/menu-stack.d.ts +54 -0
  37. package/lib/components/overlay/components/menu/menu-trigger-base.d.ts +33 -0
  38. package/lib/components/overlay/components/menu/menu-trigger.d.ts +34 -0
  39. package/lib/components/overlay/components/menu/menu.d.ts +17 -0
  40. package/lib/components/overlay/components/menu/pointer-focus-tracker.d.ts +17 -0
  41. package/lib/components/overlay/components/menu/public-api.d.ts +16 -0
  42. package/lib/components/overlay/components/public-api.d.ts +1 -0
  43. 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"]}