@angular/cdk 14.0.0-next.11 → 14.0.0-next.12

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 (227) hide show
  1. package/a11y/focus-monitor/focus-monitor.d.ts +1 -1
  2. package/a11y/focus-trap/focus-trap.d.ts +1 -1
  3. package/a11y/live-announcer/live-announcer.d.ts +1 -1
  4. package/accordion/accordion-item.d.ts +1 -1
  5. package/accordion/accordion.d.ts +1 -1
  6. package/bidi/dir.d.ts +1 -1
  7. package/clipboard/copy-to-clipboard.d.ts +1 -1
  8. package/dialog/dialog-config.d.ts +115 -0
  9. package/dialog/dialog-container.d.ts +95 -0
  10. package/dialog/dialog-injectors.d.ts +24 -0
  11. package/dialog/dialog-module.d.ts +10 -0
  12. package/dialog/dialog-ref.d.ts +64 -0
  13. package/dialog/dialog.d.ts +103 -0
  14. package/dialog/dialog_public_index.d.ts +4 -0
  15. package/dialog/index.d.ts +8 -0
  16. package/dialog/package.json +9 -0
  17. package/dialog/public-api.d.ts +13 -0
  18. package/drag-drop/directives/drag-handle.d.ts +1 -1
  19. package/drag-drop/directives/drag-placeholder.d.ts +1 -1
  20. package/drag-drop/directives/drag-preview.d.ts +1 -1
  21. package/drag-drop/directives/drag.d.ts +8 -9
  22. package/drag-drop/directives/drop-list-group.d.ts +1 -1
  23. package/drag-drop/directives/drop-list.d.ts +1 -1
  24. package/esm2020/a11y/a11y-module.mjs +4 -4
  25. package/esm2020/a11y/aria-describer/aria-describer.mjs +3 -3
  26. package/esm2020/a11y/focus-monitor/focus-monitor.mjs +6 -6
  27. package/esm2020/a11y/focus-trap/configurable-focus-trap-factory.mjs +3 -3
  28. package/esm2020/a11y/focus-trap/focus-trap-manager.mjs +3 -3
  29. package/esm2020/a11y/focus-trap/focus-trap.mjs +6 -6
  30. package/esm2020/a11y/high-contrast-mode/high-contrast-mode-detector.mjs +3 -3
  31. package/esm2020/a11y/input-modality/input-modality-detector.mjs +3 -3
  32. package/esm2020/a11y/interactivity-checker/interactivity-checker.mjs +3 -3
  33. package/esm2020/a11y/live-announcer/live-announcer.mjs +6 -6
  34. package/esm2020/accordion/accordion-item.mjs +3 -3
  35. package/esm2020/accordion/accordion-module.mjs +4 -4
  36. package/esm2020/accordion/accordion.mjs +3 -3
  37. package/esm2020/bidi/bidi-module.mjs +4 -4
  38. package/esm2020/bidi/dir.mjs +3 -3
  39. package/esm2020/bidi/directionality.mjs +3 -3
  40. package/esm2020/clipboard/clipboard-module.mjs +4 -4
  41. package/esm2020/clipboard/clipboard.mjs +3 -3
  42. package/esm2020/clipboard/copy-to-clipboard.mjs +3 -3
  43. package/esm2020/collections/unique-selection-dispatcher.mjs +3 -3
  44. package/esm2020/dialog/dialog-config.mjs +58 -0
  45. package/esm2020/dialog/dialog-container.mjs +265 -0
  46. package/esm2020/dialog/dialog-injectors.mjs +26 -0
  47. package/esm2020/dialog/dialog-module.mjs +42 -0
  48. package/esm2020/dialog/dialog-ref.mjs +76 -0
  49. package/esm2020/dialog/dialog.mjs +285 -0
  50. package/esm2020/dialog/dialog_public_index.mjs +5 -0
  51. package/esm2020/dialog/index.mjs +9 -0
  52. package/esm2020/dialog/public-api.mjs +14 -0
  53. package/esm2020/drag-drop/directives/drag-handle.mjs +3 -3
  54. package/esm2020/drag-drop/directives/drag-placeholder.mjs +3 -3
  55. package/esm2020/drag-drop/directives/drag-preview.mjs +3 -3
  56. package/esm2020/drag-drop/directives/drag.mjs +11 -4
  57. package/esm2020/drag-drop/directives/drop-list-group.mjs +3 -3
  58. package/esm2020/drag-drop/directives/drop-list.mjs +3 -3
  59. package/esm2020/drag-drop/drag-drop-module.mjs +4 -4
  60. package/esm2020/drag-drop/drag-drop-registry.mjs +3 -3
  61. package/esm2020/drag-drop/drag-drop.mjs +3 -3
  62. package/esm2020/layout/breakpoints-observer.mjs +3 -3
  63. package/esm2020/layout/layout-module.mjs +4 -4
  64. package/esm2020/layout/media-matcher.mjs +3 -3
  65. package/esm2020/menu/context-menu-trigger.mjs +227 -0
  66. package/esm2020/menu/index.mjs +9 -0
  67. package/esm2020/menu/menu-aim.mjs +204 -0
  68. package/esm2020/menu/menu-bar.mjs +159 -0
  69. package/esm2020/menu/menu-base.mjs +203 -0
  70. package/esm2020/menu/menu-errors.mjs +22 -0
  71. package/esm2020/menu/menu-group.mjs +30 -0
  72. package/esm2020/menu/menu-interface.mjs +11 -0
  73. package/esm2020/menu/menu-item-checkbox.mjs +48 -0
  74. package/esm2020/menu/menu-item-radio.mjs +114 -0
  75. package/esm2020/menu/menu-item-selectable.mjs +42 -0
  76. package/esm2020/menu/menu-item.mjs +289 -0
  77. package/esm2020/menu/menu-module.mjs +62 -0
  78. package/esm2020/menu/menu-stack.mjs +156 -0
  79. package/esm2020/menu/menu-trigger-base.mjs +114 -0
  80. package/esm2020/menu/menu-trigger.mjs +302 -0
  81. package/esm2020/menu/menu.mjs +168 -0
  82. package/esm2020/menu/menu_public_index.mjs +5 -0
  83. package/esm2020/menu/pointer-focus-tracker.mjs +51 -0
  84. package/esm2020/menu/public-api.mjs +24 -0
  85. package/esm2020/observers/observe-content.mjs +13 -13
  86. package/esm2020/overlay/dispatchers/base-overlay-dispatcher.mjs +3 -3
  87. package/esm2020/overlay/dispatchers/overlay-keyboard-dispatcher.mjs +3 -3
  88. package/esm2020/overlay/dispatchers/overlay-outside-click-dispatcher.mjs +3 -3
  89. package/esm2020/overlay/fullscreen-overlay-container.mjs +3 -3
  90. package/esm2020/overlay/overlay-container.mjs +3 -3
  91. package/esm2020/overlay/overlay-directives.mjs +6 -6
  92. package/esm2020/overlay/overlay-module.mjs +4 -4
  93. package/esm2020/overlay/overlay-ref.mjs +4 -3
  94. package/esm2020/overlay/overlay.mjs +3 -3
  95. package/esm2020/overlay/position/overlay-position-builder.mjs +3 -3
  96. package/esm2020/overlay/scroll/scroll-strategy-options.mjs +3 -3
  97. package/esm2020/platform/platform-module.mjs +4 -4
  98. package/esm2020/platform/platform.mjs +3 -3
  99. package/esm2020/portal/portal-directives.mjs +16 -16
  100. package/esm2020/scrolling/fixed-size-virtual-scroll.mjs +3 -3
  101. package/esm2020/scrolling/scroll-dispatcher.mjs +3 -3
  102. package/esm2020/scrolling/scrollable.mjs +3 -3
  103. package/esm2020/scrolling/scrolling-module.mjs +8 -8
  104. package/esm2020/scrolling/viewport-ruler.mjs +3 -3
  105. package/esm2020/scrolling/virtual-for-of.mjs +3 -3
  106. package/esm2020/scrolling/virtual-scroll-viewport.mjs +3 -3
  107. package/esm2020/stepper/step-header.mjs +3 -3
  108. package/esm2020/stepper/step-label.mjs +3 -3
  109. package/esm2020/stepper/stepper-button.mjs +6 -6
  110. package/esm2020/stepper/stepper-module.mjs +4 -4
  111. package/esm2020/stepper/stepper.mjs +6 -6
  112. package/esm2020/table/cell.mjs +21 -21
  113. package/esm2020/table/coalesced-style-scheduler.mjs +3 -3
  114. package/esm2020/table/row.mjs +28 -28
  115. package/esm2020/table/table-module.mjs +4 -4
  116. package/esm2020/table/table.mjs +20 -20
  117. package/esm2020/table/text-column.mjs +4 -4
  118. package/esm2020/text-field/autofill.mjs +6 -6
  119. package/esm2020/text-field/autosize.mjs +3 -3
  120. package/esm2020/text-field/text-field-module.mjs +4 -4
  121. package/esm2020/tree/nested-node.mjs +3 -3
  122. package/esm2020/tree/node.mjs +3 -3
  123. package/esm2020/tree/outlet.mjs +3 -3
  124. package/esm2020/tree/padding.mjs +3 -3
  125. package/esm2020/tree/toggle.mjs +3 -3
  126. package/esm2020/tree/tree-module.mjs +4 -4
  127. package/esm2020/tree/tree.mjs +6 -6
  128. package/esm2020/version.mjs +1 -1
  129. package/fesm2015/a11y.mjs +40 -40
  130. package/fesm2015/accordion.mjs +10 -10
  131. package/fesm2015/bidi.mjs +10 -10
  132. package/fesm2015/cdk.mjs +1 -1
  133. package/fesm2015/cdk.mjs.map +1 -1
  134. package/fesm2015/clipboard.mjs +10 -10
  135. package/fesm2015/collections.mjs +3 -3
  136. package/fesm2015/dialog.mjs +749 -0
  137. package/fesm2015/dialog.mjs.map +1 -0
  138. package/fesm2015/drag-drop.mjs +35 -28
  139. package/fesm2015/drag-drop.mjs.map +1 -1
  140. package/fesm2015/layout.mjs +10 -10
  141. package/fesm2015/menu.mjs +2131 -0
  142. package/fesm2015/menu.mjs.map +1 -0
  143. package/fesm2015/observers.mjs +13 -13
  144. package/fesm2015/overlay.mjs +37 -36
  145. package/fesm2015/overlay.mjs.map +1 -1
  146. package/fesm2015/platform.mjs +7 -7
  147. package/fesm2015/portal.mjs +16 -16
  148. package/fesm2015/scrolling.mjs +26 -26
  149. package/fesm2015/stepper.mjs +22 -22
  150. package/fesm2015/table.mjs +78 -78
  151. package/fesm2015/table.mjs.map +1 -1
  152. package/fesm2015/text-field.mjs +13 -13
  153. package/fesm2015/tree.mjs +25 -25
  154. package/fesm2015/tree.mjs.map +1 -1
  155. package/fesm2020/a11y.mjs +40 -40
  156. package/fesm2020/accordion.mjs +10 -10
  157. package/fesm2020/bidi.mjs +10 -10
  158. package/fesm2020/cdk.mjs +1 -1
  159. package/fesm2020/cdk.mjs.map +1 -1
  160. package/fesm2020/clipboard.mjs +10 -10
  161. package/fesm2020/collections.mjs +3 -3
  162. package/fesm2020/dialog.mjs +753 -0
  163. package/fesm2020/dialog.mjs.map +1 -0
  164. package/fesm2020/drag-drop.mjs +35 -28
  165. package/fesm2020/drag-drop.mjs.map +1 -1
  166. package/fesm2020/layout.mjs +10 -10
  167. package/fesm2020/menu.mjs +2104 -0
  168. package/fesm2020/menu.mjs.map +1 -0
  169. package/fesm2020/observers.mjs +13 -13
  170. package/fesm2020/overlay.mjs +37 -36
  171. package/fesm2020/overlay.mjs.map +1 -1
  172. package/fesm2020/platform.mjs +7 -7
  173. package/fesm2020/portal.mjs +16 -16
  174. package/fesm2020/scrolling.mjs +26 -26
  175. package/fesm2020/stepper.mjs +22 -22
  176. package/fesm2020/table.mjs +78 -78
  177. package/fesm2020/table.mjs.map +1 -1
  178. package/fesm2020/text-field.mjs +13 -13
  179. package/fesm2020/tree.mjs +25 -25
  180. package/fesm2020/tree.mjs.map +1 -1
  181. package/menu/context-menu-trigger.d.ts +98 -0
  182. package/menu/index.d.ts +8 -0
  183. package/menu/menu-aim.d.ts +107 -0
  184. package/menu/menu-bar.d.ts +52 -0
  185. package/menu/menu-base.d.ts +104 -0
  186. package/menu/menu-errors.d.ts +17 -0
  187. package/menu/menu-group.d.ts +8 -0
  188. package/menu/menu-interface.d.ts +25 -0
  189. package/menu/menu-item-checkbox.d.ts +18 -0
  190. package/menu/menu-item-radio.d.ts +59 -0
  191. package/menu/menu-item-selectable.d.ts +21 -0
  192. package/menu/menu-item.d.ts +140 -0
  193. package/menu/menu-module.d.ts +17 -0
  194. package/menu/menu-stack.d.ts +122 -0
  195. package/menu/menu-trigger-base.d.ts +77 -0
  196. package/menu/menu-trigger.d.ts +104 -0
  197. package/menu/menu.d.ts +60 -0
  198. package/menu/menu_public_index.d.ts +4 -0
  199. package/menu/package.json +9 -0
  200. package/menu/pointer-focus-tracker.d.ts +47 -0
  201. package/menu/public-api.d.ts +23 -0
  202. package/observers/observe-content.d.ts +1 -1
  203. package/overlay/overlay-directives.d.ts +2 -2
  204. package/package.json +17 -1
  205. package/portal/portal-directives.d.ts +4 -4
  206. package/schematics/ng-add/index.js +1 -1
  207. package/schematics/ng-add/index.mjs +1 -1
  208. package/scrolling/fixed-size-virtual-scroll.d.ts +1 -1
  209. package/scrolling/scrollable.d.ts +1 -1
  210. package/scrolling/virtual-for-of.d.ts +1 -1
  211. package/scrolling/virtual-scroll-viewport.d.ts +1 -1
  212. package/stepper/step-header.d.ts +1 -1
  213. package/stepper/step-label.d.ts +1 -1
  214. package/stepper/stepper-button.d.ts +2 -2
  215. package/stepper/stepper.d.ts +2 -2
  216. package/table/cell.d.ts +7 -7
  217. package/table/row.d.ts +9 -9
  218. package/table/table.d.ts +6 -6
  219. package/table/text-column.d.ts +1 -1
  220. package/text-field/autofill.d.ts +1 -1
  221. package/text-field/autosize.d.ts +1 -1
  222. package/tree/nested-node.d.ts +1 -1
  223. package/tree/node.d.ts +1 -1
  224. package/tree/outlet.d.ts +1 -1
  225. package/tree/padding.d.ts +1 -1
  226. package/tree/toggle.d.ts +1 -1
  227. package/tree/tree.d.ts +2 -2
@@ -0,0 +1,227 @@
1
+ /**
2
+ * @license
3
+ * Copyright Google LLC All Rights Reserved.
4
+ *
5
+ * Use of this source code is governed by an MIT-style license that can be
6
+ * found in the LICENSE file at https://angular.io/license
7
+ */
8
+ import { Directive, Inject, Injectable, Injector, Input, Optional, ViewContainerRef, } from '@angular/core';
9
+ import { Directionality } from '@angular/cdk/bidi';
10
+ import { Overlay, OverlayConfig, STANDARD_DROPDOWN_BELOW_POSITIONS, } from '@angular/cdk/overlay';
11
+ import { coerceBooleanProperty } from '@angular/cdk/coercion';
12
+ import { merge, partition } from 'rxjs';
13
+ import { skip, takeUntil } from 'rxjs/operators';
14
+ import { MENU_STACK, MenuStack } from './menu-stack';
15
+ import { CdkMenuTriggerBase, MENU_TRIGGER } from './menu-trigger-base';
16
+ import * as i0 from "@angular/core";
17
+ import * as i1 from "@angular/cdk/overlay";
18
+ import * as i2 from "@angular/cdk/bidi";
19
+ import * as i3 from "./menu-stack";
20
+ /** The preferred menu positions for the context menu. */
21
+ const CONTEXT_MENU_POSITIONS = STANDARD_DROPDOWN_BELOW_POSITIONS.map(position => {
22
+ // In cases where the first menu item in the context menu is a trigger the submenu opens on a
23
+ // hover event. We offset the context menu 2px by default to prevent this from occurring.
24
+ const offsetX = position.overlayX === 'start' ? 2 : -2;
25
+ const offsetY = position.overlayY === 'top' ? 2 : -2;
26
+ return { ...position, offsetX, offsetY };
27
+ });
28
+ /** Tracks the last open context menu trigger across the entire application. */
29
+ export class ContextMenuTracker {
30
+ /**
31
+ * Close the previous open context menu and set the given one as being open.
32
+ * @param trigger The trigger for the currently open Context Menu.
33
+ */
34
+ update(trigger) {
35
+ if (ContextMenuTracker._openContextMenuTrigger !== trigger) {
36
+ ContextMenuTracker._openContextMenuTrigger?.close();
37
+ ContextMenuTracker._openContextMenuTrigger = trigger;
38
+ }
39
+ }
40
+ }
41
+ ContextMenuTracker.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.0.0-next.14", ngImport: i0, type: ContextMenuTracker, deps: [], target: i0.ɵɵFactoryTarget.Injectable });
42
+ ContextMenuTracker.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "14.0.0-next.14", ngImport: i0, type: ContextMenuTracker, providedIn: 'root' });
43
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.0.0-next.14", ngImport: i0, type: ContextMenuTracker, decorators: [{
44
+ type: Injectable,
45
+ args: [{ providedIn: 'root' }]
46
+ }] });
47
+ /**
48
+ * A directive that opens a menu when a user right-clicks within its host element.
49
+ * It is aware of nested context menus and will trigger only the lowest level non-disabled context menu.
50
+ */
51
+ export class CdkContextMenuTrigger extends CdkMenuTriggerBase {
52
+ constructor(
53
+ /** The DI injector for this component */
54
+ injector,
55
+ /** The view container ref for this component */
56
+ viewContainerRef,
57
+ /** The CDK overlay service */
58
+ _overlay,
59
+ /** The app's context menu tracking registry */
60
+ _contextMenuTracker,
61
+ /** The menu stack this menu is part of. */
62
+ menuStack,
63
+ /** The directionality of the current page */
64
+ _directionality) {
65
+ super(injector, viewContainerRef, menuStack);
66
+ this._overlay = _overlay;
67
+ this._contextMenuTracker = _contextMenuTracker;
68
+ this._directionality = _directionality;
69
+ this._disabled = false;
70
+ this._setMenuStackCloseListener();
71
+ }
72
+ /** Whether the context menu is disabled. */
73
+ get disabled() {
74
+ return this._disabled;
75
+ }
76
+ set disabled(value) {
77
+ this._disabled = coerceBooleanProperty(value);
78
+ }
79
+ /**
80
+ * Open the attached menu at the specified location.
81
+ * @param coordinates where to open the context menu
82
+ */
83
+ open(coordinates) {
84
+ this._open(coordinates, false);
85
+ }
86
+ /** Close the currently opened context menu. */
87
+ close() {
88
+ this.menuStack.closeAll();
89
+ }
90
+ /**
91
+ * Open the context menu and closes any previously open menus.
92
+ * @param event the mouse event which opens the context menu.
93
+ */
94
+ _openOnContextMenu(event) {
95
+ if (!this.disabled) {
96
+ // Prevent the native context menu from opening because we're opening a custom one.
97
+ event.preventDefault();
98
+ // Stop event propagation to ensure that only the closest enabled context menu opens.
99
+ // Otherwise, any context menus attached to containing elements would *also* open,
100
+ // resulting in multiple stacked context menus being displayed.
101
+ event.stopPropagation();
102
+ this._contextMenuTracker.update(this);
103
+ this._open({ x: event.clientX, y: event.clientY }, true);
104
+ // A context menu can be triggered via a mouse right click or a keyboard shortcut.
105
+ if (event.button === 2) {
106
+ this.childMenu?.focusFirstItem('mouse');
107
+ }
108
+ else if (event.button === 0) {
109
+ this.childMenu?.focusFirstItem('keyboard');
110
+ }
111
+ else {
112
+ this.childMenu?.focusFirstItem('program');
113
+ }
114
+ }
115
+ }
116
+ /**
117
+ * Get the configuration object used to create the overlay.
118
+ * @param coordinates the location to place the opened menu
119
+ */
120
+ _getOverlayConfig(coordinates) {
121
+ return new OverlayConfig({
122
+ positionStrategy: this._getOverlayPositionStrategy(coordinates),
123
+ scrollStrategy: this._overlay.scrollStrategies.block(),
124
+ direction: this._directionality,
125
+ });
126
+ }
127
+ /**
128
+ * Get the position strategy for the overlay which specifies where to place the menu.
129
+ * @param coordinates the location to place the opened menu
130
+ */
131
+ _getOverlayPositionStrategy(coordinates) {
132
+ return this._overlay
133
+ .position()
134
+ .flexibleConnectedTo(coordinates)
135
+ .withPositions(this.menuPosition ?? CONTEXT_MENU_POSITIONS);
136
+ }
137
+ /** Subscribe to the menu stack close events and close this menu when requested. */
138
+ _setMenuStackCloseListener() {
139
+ this.menuStack.closed.pipe(takeUntil(this.destroyed)).subscribe(({ item }) => {
140
+ if (item === this.childMenu && this.isOpen()) {
141
+ this.closed.next();
142
+ this.overlayRef.detach();
143
+ }
144
+ });
145
+ }
146
+ /**
147
+ * Subscribe to the overlays outside pointer events stream and handle closing out the stack if a
148
+ * click occurs outside the menus.
149
+ * @param ignoreFirstAuxClick Whether to ignore the first auxclick event outside the menu.
150
+ */
151
+ _subscribeToOutsideClicks(ignoreFirstAuxClick) {
152
+ if (this.overlayRef) {
153
+ let outsideClicks = this.overlayRef.outsidePointerEvents();
154
+ // If the menu was triggered by the `contextmenu` event, skip the first `auxclick` event
155
+ // because it fires when the mouse is released on the same click that opened the menu.
156
+ if (ignoreFirstAuxClick) {
157
+ const [auxClicks, nonAuxClicks] = partition(outsideClicks, ({ type }) => type === 'auxclick');
158
+ outsideClicks = merge(nonAuxClicks, auxClicks.pipe(skip(1)));
159
+ }
160
+ outsideClicks.pipe(takeUntil(this.stopOutsideClicksListener)).subscribe(event => {
161
+ if (!this.isElementInsideMenuStack(event.target)) {
162
+ this.menuStack.closeAll();
163
+ }
164
+ });
165
+ }
166
+ }
167
+ /**
168
+ * Open the attached menu at the specified location.
169
+ * @param coordinates where to open the context menu
170
+ * @param ignoreFirstOutsideAuxClick Whether to ignore the first auxclick outside the menu after opening.
171
+ */
172
+ _open(coordinates, ignoreFirstOutsideAuxClick) {
173
+ if (this.disabled) {
174
+ return;
175
+ }
176
+ if (this.isOpen()) {
177
+ // since we're moving this menu we need to close any submenus first otherwise they end up
178
+ // disconnected from this one.
179
+ this.menuStack.closeSubMenuOf(this.childMenu);
180
+ this.overlayRef.getConfig().positionStrategy.setOrigin(coordinates);
181
+ this.overlayRef.updatePosition();
182
+ }
183
+ else {
184
+ this.opened.next();
185
+ if (this.overlayRef) {
186
+ this.overlayRef.getConfig().positionStrategy.setOrigin(coordinates);
187
+ this.overlayRef.updatePosition();
188
+ }
189
+ else {
190
+ this.overlayRef = this._overlay.create(this._getOverlayConfig(coordinates));
191
+ }
192
+ this.overlayRef.attach(this.getMenuContentPortal());
193
+ this._subscribeToOutsideClicks(ignoreFirstOutsideAuxClick);
194
+ }
195
+ }
196
+ }
197
+ CdkContextMenuTrigger.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.0.0-next.14", ngImport: i0, type: CdkContextMenuTrigger, deps: [{ token: i0.Injector }, { token: i0.ViewContainerRef }, { token: i1.Overlay }, { token: ContextMenuTracker }, { token: MENU_STACK }, { token: i2.Directionality, optional: true }], target: i0.ɵɵFactoryTarget.Directive });
198
+ CdkContextMenuTrigger.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "12.0.0", version: "14.0.0-next.14", type: CdkContextMenuTrigger, selector: "[cdkContextMenuTriggerFor]", inputs: { menuTemplateRef: ["cdkContextMenuTriggerFor", "menuTemplateRef"], menuPosition: ["cdkContextMenuPosition", "menuPosition"], disabled: ["cdkContextMenuDisabled", "disabled"] }, outputs: { opened: "cdkContextMenuOpened", closed: "cdkContextMenuClosed" }, host: { listeners: { "contextmenu": "_openOnContextMenu($event)" }, properties: { "attr.data-cdk-menu-stack-id": "null" } }, providers: [
199
+ { provide: MENU_TRIGGER, useExisting: CdkContextMenuTrigger },
200
+ { provide: MENU_STACK, useClass: MenuStack },
201
+ ], exportAs: ["cdkContextMenuTriggerFor"], usesInheritance: true, ngImport: i0 });
202
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.0.0-next.14", ngImport: i0, type: CdkContextMenuTrigger, decorators: [{
203
+ type: Directive,
204
+ args: [{
205
+ selector: '[cdkContextMenuTriggerFor]',
206
+ exportAs: 'cdkContextMenuTriggerFor',
207
+ host: {
208
+ '[attr.data-cdk-menu-stack-id]': 'null',
209
+ '(contextmenu)': '_openOnContextMenu($event)',
210
+ },
211
+ inputs: ['menuTemplateRef: cdkContextMenuTriggerFor', 'menuPosition: cdkContextMenuPosition'],
212
+ outputs: ['opened: cdkContextMenuOpened', 'closed: cdkContextMenuClosed'],
213
+ providers: [
214
+ { provide: MENU_TRIGGER, useExisting: CdkContextMenuTrigger },
215
+ { provide: MENU_STACK, useClass: MenuStack },
216
+ ],
217
+ }]
218
+ }], ctorParameters: function () { return [{ type: i0.Injector }, { type: i0.ViewContainerRef }, { type: i1.Overlay }, { type: ContextMenuTracker }, { type: i3.MenuStack, decorators: [{
219
+ type: Inject,
220
+ args: [MENU_STACK]
221
+ }] }, { type: i2.Directionality, decorators: [{
222
+ type: Optional
223
+ }] }]; }, propDecorators: { disabled: [{
224
+ type: Input,
225
+ args: ['cdkContextMenuDisabled']
226
+ }] } });
227
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"context-menu-trigger.js","sourceRoot":"","sources":["../../../../../../src/cdk/menu/context-menu-trigger.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * @license\n * Copyright Google LLC All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.io/license\n */\n\nimport {\n  Directive,\n  Inject,\n  Injectable,\n  Injector,\n  Input,\n  OnDestroy,\n  Optional,\n  ViewContainerRef,\n} from '@angular/core';\nimport {Directionality} from '@angular/cdk/bidi';\nimport {\n  FlexibleConnectedPositionStrategy,\n  Overlay,\n  OverlayConfig,\n  STANDARD_DROPDOWN_BELOW_POSITIONS,\n} from '@angular/cdk/overlay';\nimport {BooleanInput, coerceBooleanProperty} from '@angular/cdk/coercion';\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\n/** The preferred menu positions for the context menu. */\nconst CONTEXT_MENU_POSITIONS = STANDARD_DROPDOWN_BELOW_POSITIONS.map(position => {\n  // In cases where the first menu item in the context menu is a trigger the submenu opens on a\n  // hover event. We offset the context menu 2px by default to prevent this from occurring.\n  const offsetX = position.overlayX === 'start' ? 2 : -2;\n  const offsetY = position.overlayY === 'top' ? 2 : -2;\n  return {...position, offsetX, offsetY};\n});\n\n/** Tracks the last open context menu trigger across the entire application. */\n@Injectable({providedIn: 'root'})\nexport class ContextMenuTracker {\n  /** The last open context menu trigger. */\n  private static _openContextMenuTrigger?: CdkContextMenuTrigger;\n\n  /**\n   * Close the previous open context menu and set the given one as being open.\n   * @param trigger The trigger for the currently open Context Menu.\n   */\n  update(trigger: CdkContextMenuTrigger) {\n    if (ContextMenuTracker._openContextMenuTrigger !== trigger) {\n      ContextMenuTracker._openContextMenuTrigger?.close();\n      ContextMenuTracker._openContextMenuTrigger = trigger;\n    }\n  }\n}\n\n/** The coordinates where the context menu should open. */\nexport type ContextMenuCoordinates = {x: number; y: number};\n\n/**\n * A directive that opens a menu when a user right-clicks within its host element.\n * It is aware of nested context menus and will trigger only the lowest level non-disabled context menu.\n */\n@Directive({\n  selector: '[cdkContextMenuTriggerFor]',\n  exportAs: 'cdkContextMenuTriggerFor',\n  host: {\n    '[attr.data-cdk-menu-stack-id]': 'null',\n    '(contextmenu)': '_openOnContextMenu($event)',\n  },\n  inputs: ['menuTemplateRef: cdkContextMenuTriggerFor', 'menuPosition: cdkContextMenuPosition'],\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  /** Whether the context menu is disabled. */\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    /** The DI injector for this component */\n    injector: Injector,\n    /** The view container ref for this component */\n    viewContainerRef: ViewContainerRef,\n    /** The CDK overlay service */\n    private readonly _overlay: Overlay,\n    /** The app's context menu tracking registry */\n    private readonly _contextMenuTracker: ContextMenuTracker,\n    /** The menu stack this menu is part of. */\n    @Inject(MENU_STACK) menuStack: MenuStack,\n    /** The directionality of the current page */\n    @Optional() private readonly _directionality?: Directionality,\n  ) {\n    super(injector, viewContainerRef, menuStack);\n    this._setMenuStackCloseListener();\n  }\n\n  /**\n   * Open the attached menu at the specified location.\n   * @param coordinates where to open the context menu\n   */\n  open(coordinates: ContextMenuCoordinates) {\n    this._open(coordinates, false);\n  }\n\n  /** Close the currently opened context menu. */\n  close() {\n    this.menuStack.closeAll();\n  }\n\n  /**\n   * Open the context menu and closes any previously open menus.\n   * @param event the mouse event which opens the context menu.\n   */\n  _openOnContextMenu(event: MouseEvent) {\n    if (!this.disabled) {\n      // Prevent the native context menu from opening because we're opening a custom one.\n      event.preventDefault();\n\n      // Stop event propagation to ensure that only the closest enabled context menu opens.\n      // Otherwise, any context menus attached to containing elements would *also* open,\n      // resulting in multiple stacked context menus being displayed.\n      event.stopPropagation();\n\n      this._contextMenuTracker.update(this);\n      this._open({x: event.clientX, y: event.clientY}, true);\n\n      // A context menu can be triggered via a mouse right click or a keyboard shortcut.\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  /**\n   * Get the configuration object used to create the overlay.\n   * @param coordinates the location to place the opened menu\n   */\n  private _getOverlayConfig(coordinates: ContextMenuCoordinates) {\n    return new OverlayConfig({\n      positionStrategy: this._getOverlayPositionStrategy(coordinates),\n      scrollStrategy: this._overlay.scrollStrategies.block(),\n      direction: this._directionality,\n    });\n  }\n\n  /**\n   * Get the position strategy for the overlay which specifies where to place the menu.\n   * @param coordinates the location to place the opened menu\n   */\n  private _getOverlayPositionStrategy(\n    coordinates: ContextMenuCoordinates,\n  ): FlexibleConnectedPositionStrategy {\n    return this._overlay\n      .position()\n      .flexibleConnectedTo(coordinates)\n      .withPositions(this.menuPosition ?? CONTEXT_MENU_POSITIONS);\n  }\n\n  /** Subscribe to the menu stack close events and close this menu when requested. */\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  /**\n   * Subscribe to the overlays outside pointer events stream and handle closing out the stack if a\n   * click occurs outside the menus.\n   * @param ignoreFirstAuxClick Whether to ignore the first auxclick event outside the menu.\n   */\n  private _subscribeToOutsideClicks(ignoreFirstAuxClick: boolean) {\n    if (this.overlayRef) {\n      let outsideClicks = this.overlayRef.outsidePointerEvents();\n      // If the menu was triggered by the `contextmenu` event, skip the first `auxclick` event\n      // because it fires when the mouse is released on the same click that opened the menu.\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(event.target as Element)) {\n          this.menuStack.closeAll();\n        }\n      });\n    }\n  }\n\n  /**\n   * Open the attached menu at the specified location.\n   * @param coordinates where to open the context menu\n   * @param ignoreFirstOutsideAuxClick Whether to ignore the first auxclick outside the menu after opening.\n   */\n  private _open(coordinates: ContextMenuCoordinates, ignoreFirstOutsideAuxClick: boolean) {\n    if (this.disabled) {\n      return;\n    }\n    if (this.isOpen()) {\n      // since we're moving this menu we need to close any submenus first otherwise they end up\n      // disconnected from this one.\n      this.menuStack.closeSubMenuOf(this.childMenu!);\n\n      (\n        this.overlayRef!.getConfig().positionStrategy as FlexibleConnectedPositionStrategy\n      ).setOrigin(coordinates);\n      this.overlayRef!.updatePosition();\n    } else {\n      this.opened.next();\n\n      if (this.overlayRef) {\n        (\n          this.overlayRef.getConfig().positionStrategy as FlexibleConnectedPositionStrategy\n        ).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,9 @@
1
+ /**
2
+ * @license
3
+ * Copyright Google LLC All Rights Reserved.
4
+ *
5
+ * Use of this source code is governed by an MIT-style license that can be
6
+ * found in the LICENSE file at https://angular.io/license
7
+ */
8
+ export * from './public-api';
9
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9zcmMvY2RrL21lbnUvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUE7Ozs7OztHQU1HO0FBRUgsY0FBYyxjQUFjLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIEBsaWNlbnNlXG4gKiBDb3B5cmlnaHQgR29vZ2xlIExMQyBBbGwgUmlnaHRzIFJlc2VydmVkLlxuICpcbiAqIFVzZSBvZiB0aGlzIHNvdXJjZSBjb2RlIGlzIGdvdmVybmVkIGJ5IGFuIE1JVC1zdHlsZSBsaWNlbnNlIHRoYXQgY2FuIGJlXG4gKiBmb3VuZCBpbiB0aGUgTElDRU5TRSBmaWxlIGF0IGh0dHBzOi8vYW5ndWxhci5pby9saWNlbnNlXG4gKi9cblxuZXhwb3J0ICogZnJvbSAnLi9wdWJsaWMtYXBpJztcbiJdfQ==
@@ -0,0 +1,204 @@
1
+ /**
2
+ * @license
3
+ * Copyright Google LLC All Rights Reserved.
4
+ *
5
+ * Use of this source code is governed by an MIT-style license that can be
6
+ * found in the LICENSE file at https://angular.io/license
7
+ */
8
+ import { Injectable, NgZone, InjectionToken, Directive } from '@angular/core';
9
+ import { fromEvent, Subject } from 'rxjs';
10
+ import { takeUntil, filter } from 'rxjs/operators';
11
+ import { throwMissingPointerFocusTracker, throwMissingMenuReference } from './menu-errors';
12
+ import * as i0 from "@angular/core";
13
+ /** Injection token used for an implementation of MenuAim. */
14
+ export const MENU_AIM = new InjectionToken('cdk-menu-aim');
15
+ /** Capture every nth mouse move event. */
16
+ const MOUSE_MOVE_SAMPLE_FREQUENCY = 3;
17
+ /** The number of mouse move events to track. */
18
+ const NUM_POINTS = 5;
19
+ /**
20
+ * How long to wait before closing a sibling menu if a user stops short of the submenu they were
21
+ * predicted to go into.
22
+ */
23
+ const CLOSE_DELAY = 300;
24
+ /** Calculate the slope between point a and b. */
25
+ function getSlope(a, b) {
26
+ return (b.y - a.y) / (b.x - a.x);
27
+ }
28
+ /** Calculate the y intercept for the given point and slope. */
29
+ function getYIntercept(point, slope) {
30
+ return point.y - slope * point.x;
31
+ }
32
+ /**
33
+ * Whether the given mouse trajectory line defined by the slope and y intercept falls within the
34
+ * submenu as defined by `submenuPoints`
35
+ * @param submenuPoints the submenu DOMRect points.
36
+ * @param m the slope of the trajectory line.
37
+ * @param b the y intercept of the trajectory line.
38
+ * @return true if any point on the line falls within the submenu.
39
+ */
40
+ function isWithinSubmenu(submenuPoints, m, b) {
41
+ const { left, right, top, bottom } = submenuPoints;
42
+ // Check for intersection with each edge of the submenu (left, right, top, bottom)
43
+ // by fixing one coordinate to that edge's coordinate (either x or y) and checking if the
44
+ // other coordinate is within bounds.
45
+ return ((m * left + b >= top && m * left + b <= bottom) ||
46
+ (m * right + b >= top && m * right + b <= bottom) ||
47
+ ((top - b) / m >= left && (top - b) / m <= right) ||
48
+ ((bottom - b) / m >= left && (bottom - b) / m <= right));
49
+ }
50
+ /**
51
+ * TargetMenuAim predicts if a user is moving into a submenu. It calculates the
52
+ * trajectory of the user's mouse movement in the current menu to determine if the
53
+ * mouse is moving towards an open submenu.
54
+ *
55
+ * The determination is made by calculating the slope of the users last NUM_POINTS moves where each
56
+ * pair of points determines if the trajectory line points into the submenu. It uses consensus
57
+ * approach by checking if at least NUM_POINTS / 2 pairs determine that the user is moving towards
58
+ * to submenu.
59
+ */
60
+ export class TargetMenuAim {
61
+ constructor(
62
+ /** The Angular zone. */
63
+ _ngZone) {
64
+ this._ngZone = _ngZone;
65
+ /** The last NUM_POINTS mouse move events. */
66
+ this._points = [];
67
+ /** Emits when this service is destroyed. */
68
+ this._destroyed = new Subject();
69
+ }
70
+ ngOnDestroy() {
71
+ this._destroyed.next();
72
+ this._destroyed.complete();
73
+ }
74
+ /**
75
+ * Set the Menu and its PointerFocusTracker.
76
+ * @param menu The menu that this menu aim service controls.
77
+ * @param pointerTracker The `PointerFocusTracker` for the given menu.
78
+ */
79
+ initialize(menu, pointerTracker) {
80
+ this._menu = menu;
81
+ this._pointerTracker = pointerTracker;
82
+ this._subscribeToMouseMoves();
83
+ }
84
+ /**
85
+ * Calls the `doToggle` callback when it is deemed that the user is not moving towards
86
+ * the submenu.
87
+ * @param doToggle the function called when the user is not moving towards the submenu.
88
+ */
89
+ toggle(doToggle) {
90
+ // If the menu is horizontal the sub-menus open below and there is no risk of premature
91
+ // closing of any sub-menus therefore we automatically resolve the callback.
92
+ if (this._menu.orientation === 'horizontal') {
93
+ doToggle();
94
+ }
95
+ this._checkConfigured();
96
+ const siblingItemIsWaiting = !!this._timeoutId;
97
+ const hasPoints = this._points.length > 1;
98
+ if (hasPoints && !siblingItemIsWaiting) {
99
+ if (this._isMovingToSubmenu()) {
100
+ this._startTimeout(doToggle);
101
+ }
102
+ else {
103
+ doToggle();
104
+ }
105
+ }
106
+ else if (!siblingItemIsWaiting) {
107
+ doToggle();
108
+ }
109
+ }
110
+ /**
111
+ * Start the delayed toggle handler if one isn't running already.
112
+ *
113
+ * The delayed toggle handler executes the `doToggle` callback after some period of time iff the
114
+ * users mouse is on an item in the current menu.
115
+ *
116
+ * @param doToggle the function called when the user is not moving towards the submenu.
117
+ */
118
+ _startTimeout(doToggle) {
119
+ // If the users mouse is moving towards a submenu we don't want to immediately resolve.
120
+ // Wait for some period of time before determining if the previous menu should close in
121
+ // cases where the user may have moved towards the submenu but stopped on a sibling menu
122
+ // item intentionally.
123
+ const timeoutId = setTimeout(() => {
124
+ // Resolve if the user is currently moused over some element in the root menu
125
+ if (this._pointerTracker.activeElement && timeoutId === this._timeoutId) {
126
+ doToggle();
127
+ }
128
+ this._timeoutId = null;
129
+ }, CLOSE_DELAY);
130
+ this._timeoutId = timeoutId;
131
+ }
132
+ /** Whether the user is heading towards the open submenu. */
133
+ _isMovingToSubmenu() {
134
+ const submenuPoints = this._getSubmenuBounds();
135
+ if (!submenuPoints) {
136
+ return false;
137
+ }
138
+ let numMoving = 0;
139
+ const currPoint = this._points[this._points.length - 1];
140
+ // start from the second last point and calculate the slope between each point and the last
141
+ // point.
142
+ for (let i = this._points.length - 2; i >= 0; i--) {
143
+ const previous = this._points[i];
144
+ const slope = getSlope(currPoint, previous);
145
+ if (isWithinSubmenu(submenuPoints, slope, getYIntercept(currPoint, slope))) {
146
+ numMoving++;
147
+ }
148
+ }
149
+ return numMoving >= Math.floor(NUM_POINTS / 2);
150
+ }
151
+ /** Get the bounding DOMRect for the open submenu. */
152
+ _getSubmenuBounds() {
153
+ return this._pointerTracker?.previousElement?.getMenu()?.nativeElement.getBoundingClientRect();
154
+ }
155
+ /**
156
+ * Check if a reference to the PointerFocusTracker and menu element is provided.
157
+ * @throws an error if neither reference is provided.
158
+ */
159
+ _checkConfigured() {
160
+ if (typeof ngDevMode === 'undefined' || ngDevMode) {
161
+ if (!this._pointerTracker) {
162
+ throwMissingPointerFocusTracker();
163
+ }
164
+ if (!this._menu) {
165
+ throwMissingMenuReference();
166
+ }
167
+ }
168
+ }
169
+ /** Subscribe to the root menus mouse move events and update the tracked mouse points. */
170
+ _subscribeToMouseMoves() {
171
+ this._ngZone.runOutsideAngular(() => {
172
+ fromEvent(this._menu.nativeElement, 'mousemove')
173
+ .pipe(filter((_, index) => index % MOUSE_MOVE_SAMPLE_FREQUENCY === 0), takeUntil(this._destroyed))
174
+ .subscribe((event) => {
175
+ this._points.push({ x: event.clientX, y: event.clientY });
176
+ if (this._points.length > NUM_POINTS) {
177
+ this._points.shift();
178
+ }
179
+ });
180
+ });
181
+ }
182
+ }
183
+ TargetMenuAim.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.0.0-next.14", ngImport: i0, type: TargetMenuAim, deps: [{ token: i0.NgZone }], target: i0.ɵɵFactoryTarget.Injectable });
184
+ TargetMenuAim.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "14.0.0-next.14", ngImport: i0, type: TargetMenuAim });
185
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.0.0-next.14", ngImport: i0, type: TargetMenuAim, decorators: [{
186
+ type: Injectable
187
+ }], ctorParameters: function () { return [{ type: i0.NgZone }]; } });
188
+ /**
189
+ * CdkTargetMenuAim is a provider for the TargetMenuAim service. It can be added to an
190
+ * element with either the `cdkMenu` or `cdkMenuBar` directive and child menu items.
191
+ */
192
+ export class CdkTargetMenuAim {
193
+ }
194
+ CdkTargetMenuAim.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.0.0-next.14", ngImport: i0, type: CdkTargetMenuAim, deps: [], target: i0.ɵɵFactoryTarget.Directive });
195
+ CdkTargetMenuAim.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "12.0.0", version: "14.0.0-next.14", type: CdkTargetMenuAim, selector: "[cdkTargetMenuAim]", providers: [{ provide: MENU_AIM, useClass: TargetMenuAim }], exportAs: ["cdkTargetMenuAim"], ngImport: i0 });
196
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.0.0-next.14", ngImport: i0, type: CdkTargetMenuAim, decorators: [{
197
+ type: Directive,
198
+ args: [{
199
+ selector: '[cdkTargetMenuAim]',
200
+ exportAs: 'cdkTargetMenuAim',
201
+ providers: [{ provide: MENU_AIM, useClass: TargetMenuAim }],
202
+ }]
203
+ }] });
204
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"menu-aim.js","sourceRoot":"","sources":["../../../../../../src/cdk/menu/menu-aim.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * @license\n * Copyright Google LLC All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.io/license\n */\n\nimport {Injectable, NgZone, OnDestroy, InjectionToken, Directive} from '@angular/core';\nimport {fromEvent, Subject} from 'rxjs';\nimport {takeUntil, filter} from 'rxjs/operators';\nimport {PointerFocusTracker, FocusableElement} from './pointer-focus-tracker';\nimport {Menu} from './menu-interface';\nimport {throwMissingPointerFocusTracker, throwMissingMenuReference} from './menu-errors';\n\n/**\n * MenuAim is responsible for determining if a sibling menuitem's menu should be closed when a\n * Toggler item is hovered into. It is up to the hovered in item to call the MenuAim service in\n * order to determine if it may perform its close actions.\n */\nexport interface MenuAim {\n  /**\n   * Set the Menu and its PointerFocusTracker.\n   * @param menu The menu that this menu aim service controls.\n   * @param pointerTracker The `PointerFocusTracker` for the given menu.\n   */\n  initialize(menu: Menu, pointerTracker: PointerFocusTracker<FocusableElement & Toggler>): void;\n\n  /**\n   * Calls the `doToggle` callback when it is deemed that the user is not moving towards\n   * the submenu.\n   * @param doToggle the function called when the user is not moving towards the submenu.\n   */\n  toggle(doToggle: () => void): void;\n}\n\n/** Injection token used for an implementation of MenuAim. */\nexport const MENU_AIM = new InjectionToken<MenuAim>('cdk-menu-aim');\n\n/** Capture every nth mouse move event. */\nconst MOUSE_MOVE_SAMPLE_FREQUENCY = 3;\n\n/** The number of mouse move events to track. */\nconst NUM_POINTS = 5;\n\n/**\n * How long to wait before closing a sibling menu if a user stops short of the submenu they were\n * predicted to go into.\n */\nconst CLOSE_DELAY = 300;\n\n/** An element which when hovered over may open or close a menu. */\nexport interface Toggler {\n  /** Gets the open menu, or undefined if no menu is open. */\n  getMenu(): Menu | undefined;\n}\n\n/** Calculate the slope between point a and b. */\nfunction getSlope(a: Point, b: Point) {\n  return (b.y - a.y) / (b.x - a.x);\n}\n\n/** Calculate the y intercept for the given point and slope. */\nfunction getYIntercept(point: Point, slope: number) {\n  return point.y - slope * point.x;\n}\n\n/** Represents a coordinate of mouse travel. */\ntype Point = {x: number; y: number};\n\n/**\n * Whether the given mouse trajectory line defined by the slope and y intercept falls within the\n * submenu as defined by `submenuPoints`\n * @param submenuPoints the submenu DOMRect points.\n * @param m the slope of the trajectory line.\n * @param b the y intercept of the trajectory line.\n * @return true if any point on the line falls within the submenu.\n */\nfunction isWithinSubmenu(submenuPoints: DOMRect, m: number, b: number) {\n  const {left, right, top, bottom} = submenuPoints;\n\n  // Check for intersection with each edge of the submenu (left, right, top, bottom)\n  // by fixing one coordinate to that edge's coordinate (either x or y) and checking if the\n  // other coordinate is within bounds.\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/**\n * TargetMenuAim predicts if a user is moving into a submenu. It calculates the\n * trajectory of the user's mouse movement in the current menu to determine if the\n * mouse is moving towards an open submenu.\n *\n * The determination is made by calculating the slope of the users last NUM_POINTS moves where each\n * pair of points determines if the trajectory line points into the submenu. It uses consensus\n * approach by checking if at least NUM_POINTS / 2 pairs determine that the user is moving towards\n * to submenu.\n */\n@Injectable()\nexport class TargetMenuAim implements MenuAim, OnDestroy {\n  /** The last NUM_POINTS mouse move events. */\n  private readonly _points: Point[] = [];\n\n  /** Reference to the root menu in which we are tracking mouse moves. */\n  private _menu: Menu;\n\n  /** Reference to the root menu's mouse manager. */\n  private _pointerTracker: PointerFocusTracker<Toggler & FocusableElement>;\n\n  /** The id associated with the current timeout call waiting to resolve. */\n  private _timeoutId: number | null;\n\n  /** Emits when this service is destroyed. */\n  private readonly _destroyed: Subject<void> = new Subject();\n\n  constructor(\n    /** The Angular zone. */\n    private readonly _ngZone: NgZone,\n  ) {}\n\n  ngOnDestroy() {\n    this._destroyed.next();\n    this._destroyed.complete();\n  }\n\n  /**\n   * Set the Menu and its PointerFocusTracker.\n   * @param menu The menu that this menu aim service controls.\n   * @param pointerTracker The `PointerFocusTracker` for the given menu.\n   */\n  initialize(menu: Menu, pointerTracker: PointerFocusTracker<FocusableElement & Toggler>) {\n    this._menu = menu;\n    this._pointerTracker = pointerTracker;\n    this._subscribeToMouseMoves();\n  }\n\n  /**\n   * Calls the `doToggle` callback when it is deemed that the user is not moving towards\n   * the submenu.\n   * @param doToggle the function called when the user is not moving towards the submenu.\n   */\n  toggle(doToggle: () => void) {\n    // If the menu is horizontal the sub-menus open below and there is no risk of premature\n    // closing of any sub-menus therefore we automatically resolve the callback.\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  /**\n   * Start the delayed toggle handler if one isn't running already.\n   *\n   * The delayed toggle handler executes the `doToggle` callback after some period of time iff the\n   * users mouse is on an item in the current menu.\n   *\n   * @param doToggle the function called when the user is not moving towards the submenu.\n   */\n  private _startTimeout(doToggle: () => void) {\n    // If the users mouse is moving towards a submenu we don't want to immediately resolve.\n    // Wait for some period of time before determining if the previous menu should close in\n    // cases where the user may have moved towards the submenu but stopped on a sibling menu\n    // item intentionally.\n    const timeoutId = setTimeout(() => {\n      // Resolve if the user is currently moused over some element in the root menu\n      if (this._pointerTracker!.activeElement && timeoutId === this._timeoutId) {\n        doToggle();\n      }\n      this._timeoutId = null;\n    }, CLOSE_DELAY) as any as number;\n\n    this._timeoutId = timeoutId;\n  }\n\n  /** Whether the user is heading towards the open submenu. */\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    // start from the second last point and calculate the slope between each point and the last\n    // point.\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  /** Get the bounding DOMRect for the open submenu. */\n  private _getSubmenuBounds(): DOMRect | undefined {\n    return this._pointerTracker?.previousElement?.getMenu()?.nativeElement.getBoundingClientRect();\n  }\n\n  /**\n   * Check if a reference to the PointerFocusTracker and menu element is provided.\n   * @throws an error if neither reference is provided.\n   */\n  private _checkConfigured() {\n    if (typeof ngDevMode === 'undefined' || ngDevMode) {\n      if (!this._pointerTracker) {\n        throwMissingPointerFocusTracker();\n      }\n      if (!this._menu) {\n        throwMissingMenuReference();\n      }\n    }\n  }\n\n  /** Subscribe to the root menus mouse move events and update the tracked mouse points. */\n  private _subscribeToMouseMoves() {\n    this._ngZone.runOutsideAngular(() => {\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/**\n * CdkTargetMenuAim is a provider for the TargetMenuAim service. It can be added to an\n * element with either the `cdkMenu` or `cdkMenuBar` directive and child menu items.\n */\n@Directive({\n  selector: '[cdkTargetMenuAim]',\n  exportAs: 'cdkTargetMenuAim',\n  providers: [{provide: MENU_AIM, useClass: TargetMenuAim}],\n})\nexport class CdkTargetMenuAim {}\n"]}