@spectrum-web-components/menu 0.31.0 → 0.31.1-overlay.29

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/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@spectrum-web-components/menu",
3
- "version": "0.31.0",
3
+ "version": "0.31.1-overlay.29+93d3f11dd",
4
4
  "publishConfig": {
5
5
  "access": "public"
6
6
  },
@@ -85,13 +85,13 @@
85
85
  ],
86
86
  "dependencies": {
87
87
  "@lit-labs/observers": "^2.0.0",
88
- "@spectrum-web-components/action-button": "^0.31.0",
89
- "@spectrum-web-components/base": "^0.31.0",
90
- "@spectrum-web-components/divider": "^0.31.0",
91
- "@spectrum-web-components/icon": "^0.31.0",
92
- "@spectrum-web-components/icons-ui": "^0.31.0",
93
- "@spectrum-web-components/overlay": "^0.31.0",
94
- "@spectrum-web-components/shared": "^0.31.0"
88
+ "@spectrum-web-components/action-button": "^0.31.1-overlay.29+93d3f11dd",
89
+ "@spectrum-web-components/base": "^0.31.1-overlay.29+93d3f11dd",
90
+ "@spectrum-web-components/divider": "^0.31.1-overlay.29+93d3f11dd",
91
+ "@spectrum-web-components/icon": "^0.31.1-overlay.29+93d3f11dd",
92
+ "@spectrum-web-components/icons-ui": "^0.31.1-overlay.29+93d3f11dd",
93
+ "@spectrum-web-components/overlay": "^0.31.1-overlay.29+93d3f11dd",
94
+ "@spectrum-web-components/shared": "^0.31.1-overlay.29+93d3f11dd"
95
95
  },
96
96
  "devDependencies": {
97
97
  "@spectrum-css/menu": "^4.0.26"
@@ -102,5 +102,5 @@
102
102
  "./sp-*.js",
103
103
  "./**/*.dev.js"
104
104
  ],
105
- "gitHead": "a30eeed27948a9a372b792b751f968c54337ab9e"
105
+ "gitHead": "93d3f11dde0f578bd1ddcf7fbb78b02916fdfbe0"
106
106
  }
package/src/Menu.d.ts CHANGED
@@ -22,7 +22,7 @@ export interface MenuChildItem {
22
22
  */
23
23
  export declare class Menu extends SpectrumElement {
24
24
  static get styles(): CSSResultArray;
25
- isSubmenu: boolean;
25
+ private get isSubmenu();
26
26
  label: string;
27
27
  selects: undefined | 'inherit' | 'single' | 'multiple';
28
28
  value: string;
@@ -73,17 +73,24 @@ export declare class Menu extends SpectrumElement {
73
73
  startListeningToKeyboard(): void;
74
74
  handleFocusout(event: FocusEvent): void;
75
75
  stopListeningToKeyboard(): void;
76
+ private descendentOverlays;
77
+ protected handleDescendentOverlayOpened(event: Event): void;
78
+ handleSubmenuOpened: (event: Event) => void;
76
79
  selectOrToggleItem(targetItem: MenuItem): Promise<void>;
77
80
  protected navigateWithinMenu(event: KeyboardEvent): void;
78
- protected navigateBetweenRelatedMenus(code: string): void;
81
+ protected navigateBetweenRelatedMenus(event: KeyboardEvent): void;
79
82
  handleKeydown(event: KeyboardEvent): void;
80
83
  focusMenuItemByOffset(offset: number): MenuItem;
81
84
  private prepareToCleanUp;
85
+ private _hasUpdatedSelectedItemIndex;
82
86
  updateSelectedItemIndex(): void;
83
- private _willUpdateItems;
87
+ private _willUpdateItems?;
84
88
  private handleItemsChanged;
85
89
  private updateItemFocus;
90
+ closeDescendentOverlays(): void;
86
91
  private forwardFocusVisibleToItem;
92
+ private handleSlotchange;
93
+ protected renderMenuItemSlot(): TemplateResult;
87
94
  render(): TemplateResult;
88
95
  private _notFirstUpdated;
89
96
  protected firstUpdated(changed: PropertyValues): void;
@@ -92,5 +99,6 @@ export declare class Menu extends SpectrumElement {
92
99
  connectedCallback(): void;
93
100
  protected childItemsUpdated: Promise<unknown[]>;
94
101
  protected cacheUpdated: Promise<void>;
102
+ protected resolveCacheUpdated: () => void;
95
103
  protected getUpdateComplete(): Promise<boolean>;
96
104
  }
package/src/Menu.dev.js CHANGED
@@ -26,7 +26,6 @@ function elementIsOrContains(el, isOrContains) {
26
26
  export class Menu extends SpectrumElement {
27
27
  constructor() {
28
28
  super();
29
- this.isSubmenu = false;
30
29
  this.label = "";
31
30
  this.value = "";
32
31
  this.valueSeparator = ",";
@@ -36,9 +35,42 @@ export class Menu extends SpectrumElement {
36
35
  this.focusedItemIndex = 0;
37
36
  this.focusInItemIndex = 0;
38
37
  this.selectedItemsMap = /* @__PURE__ */ new Map();
39
- this._willUpdateItems = false;
38
+ this.removeChildItem = async (event) => {
39
+ this.childItemSet.delete(event.item);
40
+ this.cachedChildItems = void 0;
41
+ if (event.item.focused) {
42
+ this.handleItemsChanged();
43
+ await this.updateComplete;
44
+ this.focus();
45
+ }
46
+ };
47
+ this.descendentOverlays = /* @__PURE__ */ new Map();
48
+ this.handleSubmenuOpened = (event) => {
49
+ event.stopPropagation();
50
+ const target = event.composedPath()[0];
51
+ target.dispatchEvent(
52
+ new Event("sp-menu-submenu-opened", {
53
+ bubbles: true,
54
+ composed: true
55
+ })
56
+ );
57
+ const focusedItem = this.childItems[this.focusedItemIndex];
58
+ if (focusedItem) {
59
+ focusedItem.focused = false;
60
+ }
61
+ const openedItem = event.composedPath().find((el) => this.childItemSet.has(el));
62
+ if (!openedItem)
63
+ return;
64
+ const openedItemIndex = this.childItems.indexOf(openedItem);
65
+ this.focusedItemIndex = openedItemIndex;
66
+ this.focusInItemIndex = openedItemIndex;
67
+ };
68
+ this._hasUpdatedSelectedItemIndex = false;
40
69
  this._notFirstUpdated = false;
41
70
  this.cacheUpdated = Promise.resolve();
71
+ this.resolveCacheUpdated = () => {
72
+ return;
73
+ };
42
74
  this.addEventListener(
43
75
  "sp-menu-item-added-or-updated",
44
76
  this.onSelectableItemAddedOrUpdated
@@ -53,10 +85,15 @@ export class Menu extends SpectrumElement {
53
85
  this.addEventListener("sp-menu-item-removed", this.removeChildItem);
54
86
  this.addEventListener("click", this.onClick);
55
87
  this.addEventListener("focusin", this.handleFocusin);
88
+ this.addEventListener("focusout", this.handleFocusout);
89
+ this.addEventListener("sp-opened", this.handleSubmenuOpened);
56
90
  }
57
91
  static get styles() {
58
92
  return [menuStyles];
59
93
  }
94
+ get isSubmenu() {
95
+ return this.slot === "submenu";
96
+ }
60
97
  get childItems() {
61
98
  if (!this.cachedChildItems) {
62
99
  this.cachedChildItems = this.updateCachedMenuItems();
@@ -107,34 +144,44 @@ export class Menu extends SpectrumElement {
107
144
  * @param event
108
145
  */
109
146
  onFocusableItemAddedOrUpdated(event) {
147
+ event.menuCascade.set(this, {
148
+ hadFocusRoot: !!event.item.menuData.focusRoot,
149
+ ancestorWithSelects: event.currentAncestorWithSelects
150
+ });
151
+ if (this.selects) {
152
+ event.currentAncestorWithSelects = this;
153
+ }
154
+ event.focusRoot = this;
155
+ }
156
+ /**
157
+ * When a descendant `<sp-menu-item>` element is added or updated it will dispatch
158
+ * this event to announce its presence in the DOM. During the bubble phase the first
159
+ * Menu based element that the event encounters that does not inherit selection will
160
+ * manage the selection state of the dispatching `<sp-menu-item>` element.
161
+ * @param event
162
+ */
163
+ onSelectableItemAddedOrUpdated(event) {
110
164
  var _a;
111
- if (event.item.menuData.focusRoot) {
165
+ const cascadeData = event.menuCascade.get(this);
166
+ if (!cascadeData)
167
+ return;
168
+ event.parentMenu = this;
169
+ if (cascadeData.hadFocusRoot) {
112
170
  this.tabIndex = -1;
113
171
  }
114
- event.focusRoot = this;
115
172
  this.addChildItem(event.item);
116
173
  if (this.selects === "inherit") {
117
174
  this.resolvedSelects = "inherit";
118
- this.resolvedRole = ((_a = event.currentAncestorWithSelects) == null ? void 0 : _a.getAttribute(
175
+ this.resolvedRole = ((_a = cascadeData.ancestorWithSelects) == null ? void 0 : _a.getAttribute(
119
176
  "role"
120
177
  )) || this.getAttribute("role") || void 0;
121
178
  } else if (this.selects) {
122
179
  this.resolvedRole = this.getAttribute("role") || void 0;
123
180
  this.resolvedSelects = this.selects;
124
- event.currentAncestorWithSelects = this;
125
181
  } else {
126
182
  this.resolvedRole = this.getAttribute("role") || void 0;
127
183
  this.resolvedSelects = this.resolvedRole === "none" ? "ignore" : "none";
128
184
  }
129
- }
130
- /**
131
- * When a descendant `<sp-menu-item>` element is added or updated it will dispatch
132
- * this event to announce its presence in the DOM. During the bubble phase the first
133
- * Menu based element that the event encounters that does not inherit selection will
134
- * manage the selection state of the dispatching `<sp-menu-item>` element.
135
- * @param event
136
- */
137
- onSelectableItemAddedOrUpdated(event) {
138
185
  const selects = this.resolvedSelects === "single" || this.resolvedSelects === "multiple";
139
186
  if ((selects || !this.selects && this.resolvedSelects !== "ignore") && !event.item.menuData.selectionRoot) {
140
187
  event.item.setRole(this.childRole);
@@ -145,15 +192,6 @@ export class Menu extends SpectrumElement {
145
192
  this.childItemSet.add(item);
146
193
  this.handleItemsChanged();
147
194
  }
148
- async removeChildItem(event) {
149
- this.childItemSet.delete(event.item);
150
- this.cachedChildItems = void 0;
151
- if (event.item.focused) {
152
- this.handleItemsChanged();
153
- await this.updateComplete;
154
- this.focus();
155
- }
156
- }
157
195
  focus({ preventScroll } = {}) {
158
196
  if (!this.childItems.length || this.childItems.every((childItem) => childItem.disabled)) {
159
197
  return;
@@ -166,7 +204,7 @@ export class Menu extends SpectrumElement {
166
204
  }
167
205
  this.focusMenuItemByOffset(0);
168
206
  super.focus({ preventScroll });
169
- const selectedItem = this.querySelector("[selected]");
207
+ const selectedItem = this.selectedItems[0];
170
208
  if (selectedItem && !preventScroll) {
171
209
  selectedItem.scrollIntoView({ block: "nearest" });
172
210
  }
@@ -203,18 +241,18 @@ export class Menu extends SpectrumElement {
203
241
  }
204
242
  handleFocusin(event) {
205
243
  var _a;
206
- const isOrContainsRelatedTarget = elementIsOrContains(
244
+ const wasOrContainedRelatedTarget = elementIsOrContains(
207
245
  this,
208
246
  event.relatedTarget
209
247
  );
210
- if (isOrContainsRelatedTarget || this.childItems.some(
248
+ if (this.childItems.some(
211
249
  (childItem) => childItem.menuData.focusRoot !== this
212
250
  )) {
213
251
  return;
214
252
  }
215
253
  const activeElement = this.getRootNode().activeElement;
216
254
  const selectionRoot = ((_a = this.childItems[this.focusedItemIndex]) == null ? void 0 : _a.menuData.selectionRoot) || this;
217
- if (activeElement !== selectionRoot || !isOrContainsRelatedTarget) {
255
+ if (activeElement !== selectionRoot || !wasOrContainedRelatedTarget && event.target !== this) {
218
256
  selectionRoot.focus({ preventScroll: true });
219
257
  if (activeElement && this.focusedItemIndex === 0) {
220
258
  const offset = this.childItems.findIndex(
@@ -229,27 +267,26 @@ export class Menu extends SpectrumElement {
229
267
  }
230
268
  startListeningToKeyboard() {
231
269
  this.addEventListener("keydown", this.handleKeydown);
232
- this.addEventListener("focusout", this.handleFocusout);
233
270
  }
234
271
  handleFocusout(event) {
235
272
  if (elementIsOrContains(this, event.relatedTarget)) {
236
- event.composedPath()[0].focused = false;
237
273
  return;
238
274
  }
239
275
  this.stopListeningToKeyboard();
240
- if (event.target === this && this.childItems.some(
241
- (childItem) => childItem.menuData.focusRoot === this
242
- )) {
243
- const focusedItem = this.childItems[this.focusedItemIndex];
244
- if (focusedItem) {
245
- focusedItem.focused = false;
246
- }
247
- }
276
+ this.childItems.forEach((child) => child.focused = false);
248
277
  this.removeAttribute("aria-activedescendant");
249
278
  }
250
279
  stopListeningToKeyboard() {
251
280
  this.removeEventListener("keydown", this.handleKeydown);
252
- this.removeEventListener("focusout", this.handleFocusout);
281
+ }
282
+ handleDescendentOverlayOpened(event) {
283
+ const target = event.composedPath()[0];
284
+ if (!target.overlayElement)
285
+ return;
286
+ this.descendentOverlays.set(
287
+ target.overlayElement,
288
+ target.overlayElement
289
+ );
253
290
  }
254
291
  async selectOrToggleItem(targetItem) {
255
292
  const resolvedSelects = this.resolvedSelects;
@@ -258,6 +295,7 @@ export class Menu extends SpectrumElement {
258
295
  const oldSelectedItems = this.selectedItems.slice();
259
296
  const oldValue = this.value;
260
297
  this.childItems[this.focusedItemIndex].focused = false;
298
+ this.childItems[this.focusedItemIndex].active = false;
261
299
  this.focusedItemIndex = this.childItems.indexOf(targetItem);
262
300
  this.forwardFocusVisibleToItem(targetItem);
263
301
  if (resolvedSelects === "multiple") {
@@ -321,23 +359,30 @@ export class Menu extends SpectrumElement {
321
359
  return;
322
360
  }
323
361
  event.preventDefault();
362
+ event.stopPropagation();
324
363
  itemToFocus.scrollIntoView({ block: "nearest" });
325
364
  }
326
- navigateBetweenRelatedMenus(code) {
365
+ navigateBetweenRelatedMenus(event) {
366
+ const { code } = event;
327
367
  const shouldOpenSubmenu = this.isLTR && code === "ArrowRight" || !this.isLTR && code === "ArrowLeft";
328
368
  const shouldCloseSelfAsSubmenu = this.isLTR && code === "ArrowLeft" || !this.isLTR && code === "ArrowRight";
329
369
  if (shouldOpenSubmenu) {
370
+ event.stopPropagation();
330
371
  const lastFocusedItem = this.childItems[this.focusedItemIndex];
331
372
  if (lastFocusedItem == null ? void 0 : lastFocusedItem.hasSubmenu) {
332
- this.blur();
333
373
  lastFocusedItem.openOverlay();
334
374
  }
335
375
  } else if (shouldCloseSelfAsSubmenu && this.isSubmenu) {
376
+ event.stopPropagation();
336
377
  this.dispatchEvent(new Event("close", { bubbles: true }));
378
+ this.updateSelectedItemIndex();
337
379
  }
338
380
  }
339
381
  handleKeydown(event) {
340
382
  var _a;
383
+ if (event.target !== this && this !== event.target.parentElement) {
384
+ return;
385
+ }
341
386
  const { code } = event;
342
387
  if (code === "Tab") {
343
388
  this.prepareToCleanUp();
@@ -346,7 +391,6 @@ export class Menu extends SpectrumElement {
346
391
  if (code === "Space") {
347
392
  const lastFocusedItem = this.childItems[this.focusedItemIndex];
348
393
  if (lastFocusedItem == null ? void 0 : lastFocusedItem.hasSubmenu) {
349
- this.blur();
350
394
  lastFocusedItem.openOverlay();
351
395
  return;
352
396
  }
@@ -359,12 +403,15 @@ export class Menu extends SpectrumElement {
359
403
  this.navigateWithinMenu(event);
360
404
  return;
361
405
  }
362
- this.navigateBetweenRelatedMenus(code);
406
+ this.navigateBetweenRelatedMenus(event);
363
407
  }
364
408
  focusMenuItemByOffset(offset) {
365
409
  const step = offset || 1;
366
410
  const focusedItem = this.childItems[this.focusedItemIndex];
367
- focusedItem.focused = false;
411
+ if (focusedItem) {
412
+ focusedItem.focused = false;
413
+ focusedItem.active = false;
414
+ }
368
415
  this.focusedItemIndex = (this.childItems.length + this.focusedItemIndex + offset) % this.childItems.length;
369
416
  let itemToFocus = this.childItems[this.focusedItemIndex];
370
417
  let availableItems = this.childItems.length;
@@ -403,7 +450,7 @@ export class Menu extends SpectrumElement {
403
450
  itemIndex -= 1;
404
451
  const childItem = this.childItems[itemIndex];
405
452
  if (childItem.menuData.selectionRoot === this) {
406
- if (childItem.selected) {
453
+ if (childItem.selected || !this._hasUpdatedSelectedItemIndex && this.selected.includes(childItem.value)) {
407
454
  firstOrFirstSelectedIndex = itemIndex;
408
455
  selectedItemsMap.set(childItem, true);
409
456
  selected.unshift(childItem.value);
@@ -429,20 +476,22 @@ export class Menu extends SpectrumElement {
429
476
  handleItemsChanged() {
430
477
  this.cachedChildItems = void 0;
431
478
  if (!this._willUpdateItems) {
432
- let resolve = () => {
433
- return;
434
- };
435
- this.cacheUpdated = new Promise((res) => resolve = res);
436
- this._willUpdateItems = true;
437
- window.requestAnimationFrame(() => {
438
- if (this.cachedChildItems === void 0) {
439
- this.updateSelectedItemIndex();
440
- this.updateItemFocus();
479
+ this.cacheUpdated = new Promise(
480
+ (res) => this.resolveCacheUpdated = () => {
481
+ res();
482
+ this._willUpdateItems = void 0;
441
483
  }
442
- this._willUpdateItems = false;
443
- resolve();
444
- });
484
+ );
485
+ } else {
486
+ cancelAnimationFrame(this._willUpdateItems);
445
487
  }
488
+ this._willUpdateItems = requestAnimationFrame(() => {
489
+ if (this._willUpdateItems || this.cachedChildItems === void 0) {
490
+ this.updateSelectedItemIndex();
491
+ this.updateItemFocus();
492
+ }
493
+ this.resolveCacheUpdated();
494
+ });
446
495
  }
447
496
  updateItemFocus() {
448
497
  if (this.childItems.length == 0) {
@@ -453,21 +502,51 @@ export class Menu extends SpectrumElement {
453
502
  this.forwardFocusVisibleToItem(focusInItem);
454
503
  }
455
504
  }
505
+ closeDescendentOverlays() {
506
+ this.descendentOverlays.forEach((overlay) => {
507
+ overlay.open = false;
508
+ });
509
+ this.descendentOverlays = /* @__PURE__ */ new Map();
510
+ }
456
511
  forwardFocusVisibleToItem(item) {
457
512
  if (item.menuData.focusRoot !== this) {
458
513
  return;
459
514
  }
460
- item.focused = this.hasVisibleFocusInTree();
515
+ this.closeDescendentOverlays();
516
+ const focused = this.hasVisibleFocusInTree() || !!this.childItems.find((child) => {
517
+ return child.hasVisibleFocusInTree();
518
+ });
519
+ item.focused = focused;
461
520
  this.setAttribute("aria-activedescendant", item.id);
462
521
  if (item.menuData.selectionRoot && item.menuData.selectionRoot !== this) {
463
522
  item.menuData.selectionRoot.focus();
464
523
  }
465
524
  }
466
- render() {
525
+ handleSlotchange({
526
+ target
527
+ }) {
528
+ const assignedElement = target.assignedElements({
529
+ flatten: true
530
+ });
531
+ if (this.childItems.length !== assignedElement.length) {
532
+ assignedElement.forEach((item) => {
533
+ if (typeof item.triggerUpdate !== "undefined") {
534
+ item.triggerUpdate();
535
+ }
536
+ });
537
+ }
538
+ }
539
+ renderMenuItemSlot() {
467
540
  return html`
468
- <slot></slot>
541
+ <slot
542
+ @sp-menu-submenu-opened=${this.handleDescendentOverlayOpened}
543
+ @slotchange=${this.handleSlotchange}
544
+ ></slot>
469
545
  `;
470
546
  }
547
+ render() {
548
+ return this.renderMenuItemSlot();
549
+ }
471
550
  firstUpdated(changed) {
472
551
  super.firstUpdated(changed);
473
552
  if (!this.hasAttribute("tabindex")) {
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "version": 3,
3
3
  "sources": ["Menu.ts"],
4
- "sourcesContent": ["/*\nCopyright 2020 Adobe. All rights reserved.\nThis file is licensed to you under the Apache License, Version 2.0 (the \"License\");\nyou may not use this file except in compliance with the License. You may obtain a copy\nof the License at http://www.apache.org/licenses/LICENSE-2.0\n\nUnless required by applicable law or agreed to in writing, software distributed under\nthe License is distributed on an \"AS IS\" BASIS, WITHOUT WARRANTIES OR REPRESENTATIONS\nOF ANY KIND, either express or implied. See the License for the specific language\ngoverning permissions and limitations under the License.\n*/\n\nimport {\n CSSResultArray,\n html,\n PropertyValues,\n SpectrumElement,\n TemplateResult,\n} from '@spectrum-web-components/base';\nimport {\n property,\n query,\n} from '@spectrum-web-components/base/src/decorators.js';\n\nimport { MenuItem } from './MenuItem.dev.js'\nimport type {\n MenuItemAddedOrUpdatedEvent,\n MenuItemRemovedEvent,\n} from './MenuItem.dev.js'\nimport menuStyles from './menu.css.js';\n\nexport interface MenuChildItem {\n menuItem: MenuItem;\n managed: boolean;\n active: boolean;\n focusable: boolean;\n focusRoot: Menu;\n}\n\ntype SelectsType = 'none' | 'ignore' | 'inherit' | 'multiple' | 'single';\ntype RoleType = 'group' | 'menu' | 'listbox' | 'none';\n\nfunction elementIsOrContains(\n el: Node,\n isOrContains: Node | undefined | null\n): boolean {\n return !!isOrContains && (el === isOrContains || el.contains(isOrContains));\n}\n\n/**\n * Spectrum Menu Component\n * @element sp-menu\n *\n * @slot - menu items to be listed in the menu\n * @fires change - Announces that the `value` of the element has changed\n * @attr selects - whether the element has a specific selection algorithm that it applies\n * to its item descendants. `single` allows only one descendent to be selected at a time.\n * `multiple` allows many descendants to be selected. `inherit` will be applied dynamically\n * when an ancestor of this element is actively managing the selection of its descendents.\n * When the `selects` attribute is not present a `value` will not be maintained and the Menu\n * Item children of this Menu will not have their `selected` state managed.\n */\nexport class Menu extends SpectrumElement {\n public static override get styles(): CSSResultArray {\n return [menuStyles];\n }\n\n public isSubmenu = false;\n\n @property({ type: String, reflect: true })\n public label = '';\n\n @property({ type: String, reflect: true })\n public selects: undefined | 'inherit' | 'single' | 'multiple';\n\n @property({ type: String })\n public value = '';\n\n // For the multiple select case, we'll join the value strings together\n // for the value property with this separator\n @property({ type: String, attribute: 'value-separator' })\n public valueSeparator = ',';\n\n // TODO: which of these to keep?\n // TODO: allow setting this in the API to change the values\n @property({ attribute: false })\n public selected = [] as string[];\n\n @property({ attribute: false })\n public selectedItems = [] as MenuItem[];\n\n @query('slot:not([name])')\n public menuSlot!: HTMLSlotElement;\n\n private childItemSet = new Set<MenuItem>();\n public focusedItemIndex = 0;\n public focusInItemIndex = 0;\n\n private selectedItemsMap = new Map() as Map<MenuItem, boolean>;\n\n public get childItems(): MenuItem[] {\n if (!this.cachedChildItems) {\n this.cachedChildItems = this.updateCachedMenuItems();\n }\n return this.cachedChildItems;\n }\n\n private cachedChildItems: MenuItem[] | undefined;\n\n private updateCachedMenuItems(): MenuItem[] {\n this.cachedChildItems = [];\n\n const slotElements = this.menuSlot\n ? this.menuSlot.assignedElements({ flatten: true })\n : [];\n for (const slotElement of slotElements) {\n const childMenuItems: MenuItem[] =\n slotElement instanceof MenuItem\n ? [slotElement as MenuItem]\n : ([...slotElement.querySelectorAll(`*`)] as MenuItem[]);\n\n for (const childMenuItem of childMenuItems) {\n if (this.childItemSet.has(childMenuItem)) {\n this.cachedChildItems.push(childMenuItem);\n }\n }\n }\n\n return this.cachedChildItems;\n }\n\n /**\n * Hide this getter from web-component-analyzer until\n * https://github.com/runem/web-component-analyzer/issues/131\n * has been addressed.\n *\n * @private\n */\n public get childRole(): string {\n if (this.resolvedRole === 'listbox') {\n return 'option';\n }\n switch (this.resolvedSelects) {\n case 'single':\n return 'menuitemradio';\n case 'multiple':\n return 'menuitemcheckbox';\n default:\n return 'menuitem';\n }\n }\n\n protected get ownRole(): string {\n return 'menu';\n }\n\n private resolvedSelects?: SelectsType;\n private resolvedRole?: RoleType;\n\n /**\n * When a descendant `<sp-menu-item>` element is added or updated it will dispatch\n * this event to announce its presence in the DOM. During the capture phase the first\n * Menu based element that the event encounters will manage the focus state of the\n * dispatching `<sp-menu-item>` element.\n * @param event\n */\n private onFocusableItemAddedOrUpdated(\n event: MenuItemAddedOrUpdatedEvent\n ): void {\n if (event.item.menuData.focusRoot) {\n // Only have one tab stop per Menu tree\n this.tabIndex = -1;\n }\n event.focusRoot = this;\n this.addChildItem(event.item);\n\n if (this.selects === 'inherit') {\n this.resolvedSelects = 'inherit';\n this.resolvedRole = (event.currentAncestorWithSelects?.getAttribute(\n 'role'\n ) ||\n this.getAttribute('role') ||\n undefined) as RoleType;\n } else if (this.selects) {\n this.resolvedRole = (this.getAttribute('role') ||\n undefined) as RoleType;\n this.resolvedSelects = this.selects;\n event.currentAncestorWithSelects = this;\n } else {\n this.resolvedRole = (this.getAttribute('role') ||\n undefined) as RoleType;\n this.resolvedSelects =\n this.resolvedRole === 'none' ? 'ignore' : 'none';\n }\n }\n\n /**\n * When a descendant `<sp-menu-item>` element is added or updated it will dispatch\n * this event to announce its presence in the DOM. During the bubble phase the first\n * Menu based element that the event encounters that does not inherit selection will\n * manage the selection state of the dispatching `<sp-menu-item>` element.\n * @param event\n */\n private onSelectableItemAddedOrUpdated(\n event: MenuItemAddedOrUpdatedEvent\n ): void {\n const selects =\n this.resolvedSelects === 'single' ||\n this.resolvedSelects === 'multiple';\n if (\n (selects || (!this.selects && this.resolvedSelects !== 'ignore')) &&\n !event.item.menuData.selectionRoot\n ) {\n event.item.setRole(this.childRole);\n event.selectionRoot = this;\n }\n }\n\n private addChildItem(item: MenuItem): void {\n this.childItemSet.add(item);\n this.handleItemsChanged();\n }\n\n private async removeChildItem(event: MenuItemRemovedEvent): Promise<void> {\n this.childItemSet.delete(event.item);\n this.cachedChildItems = undefined;\n if (event.item.focused) {\n this.handleItemsChanged();\n await this.updateComplete;\n this.focus();\n }\n }\n\n public constructor() {\n super();\n\n this.addEventListener(\n 'sp-menu-item-added-or-updated',\n this.onSelectableItemAddedOrUpdated\n );\n this.addEventListener(\n 'sp-menu-item-added-or-updated',\n this.onFocusableItemAddedOrUpdated,\n {\n capture: true,\n }\n );\n\n this.addEventListener('sp-menu-item-removed', this.removeChildItem);\n this.addEventListener('click', this.onClick);\n this.addEventListener('focusin', this.handleFocusin);\n }\n\n public override focus({ preventScroll }: FocusOptions = {}): void {\n if (\n !this.childItems.length ||\n this.childItems.every((childItem) => childItem.disabled)\n ) {\n return;\n }\n if (\n this.childItems.some(\n (childItem) => childItem.menuData.focusRoot !== this\n )\n ) {\n super.focus({ preventScroll });\n return;\n }\n this.focusMenuItemByOffset(0);\n super.focus({ preventScroll });\n const selectedItem = this.querySelector('[selected]');\n if (selectedItem && !preventScroll) {\n selectedItem.scrollIntoView({ block: 'nearest' });\n }\n }\n\n private onClick(event: Event): void {\n if (event.defaultPrevented) {\n return;\n }\n const path = event.composedPath();\n const target = path.find((el) => {\n /* c8 ignore next 3 */\n if (!(el instanceof Element)) {\n return false;\n }\n return el.getAttribute('role') === this.childRole;\n }) as MenuItem;\n if (target?.href && target.href.length) {\n // This event will NOT ALLOW CANCELATION as link action\n // cancelation should occur on the `<sp-menu-item>` itself.\n this.dispatchEvent(\n new Event('change', {\n bubbles: true,\n composed: true,\n })\n );\n return;\n } else if (\n target?.menuData.selectionRoot === this &&\n this.childItems.length\n ) {\n event.preventDefault();\n if (target.hasSubmenu || target.open) {\n return;\n }\n this.selectOrToggleItem(target);\n } else {\n return;\n }\n this.prepareToCleanUp();\n }\n\n public handleFocusin(event: FocusEvent): void {\n const isOrContainsRelatedTarget = elementIsOrContains(\n this,\n event.relatedTarget as Node\n );\n if (\n isOrContainsRelatedTarget ||\n this.childItems.some(\n (childItem) => childItem.menuData.focusRoot !== this\n )\n ) {\n return;\n }\n const activeElement = (this.getRootNode() as Document).activeElement as\n | MenuItem\n | Menu;\n const selectionRoot =\n this.childItems[this.focusedItemIndex]?.menuData.selectionRoot ||\n this;\n if (activeElement !== selectionRoot || !isOrContainsRelatedTarget) {\n selectionRoot.focus({ preventScroll: true });\n if (activeElement && this.focusedItemIndex === 0) {\n const offset = this.childItems.findIndex(\n (childItem) => childItem === activeElement\n );\n if (offset > 0) {\n this.focusMenuItemByOffset(offset);\n }\n }\n }\n this.startListeningToKeyboard();\n }\n\n public startListeningToKeyboard(): void {\n this.addEventListener('keydown', this.handleKeydown);\n this.addEventListener('focusout', this.handleFocusout);\n }\n\n public handleFocusout(event: FocusEvent): void {\n if (elementIsOrContains(this, event.relatedTarget as Node)) {\n (event.composedPath()[0] as MenuItem).focused = false;\n return;\n }\n this.stopListeningToKeyboard();\n if (\n event.target === this &&\n this.childItems.some(\n (childItem) => childItem.menuData.focusRoot === this\n )\n ) {\n const focusedItem = this.childItems[this.focusedItemIndex];\n if (focusedItem) {\n focusedItem.focused = false;\n }\n }\n this.removeAttribute('aria-activedescendant');\n }\n\n public stopListeningToKeyboard(): void {\n this.removeEventListener('keydown', this.handleKeydown);\n this.removeEventListener('focusout', this.handleFocusout);\n }\n\n public async selectOrToggleItem(targetItem: MenuItem): Promise<void> {\n const resolvedSelects = this.resolvedSelects;\n const oldSelectedItemsMap = new Map(this.selectedItemsMap);\n const oldSelected = this.selected.slice();\n const oldSelectedItems = this.selectedItems.slice();\n const oldValue = this.value;\n this.childItems[this.focusedItemIndex].focused = false;\n this.focusedItemIndex = this.childItems.indexOf(targetItem);\n this.forwardFocusVisibleToItem(targetItem);\n\n if (resolvedSelects === 'multiple') {\n if (this.selectedItemsMap.has(targetItem)) {\n this.selectedItemsMap.delete(targetItem);\n } else {\n this.selectedItemsMap.set(targetItem, true);\n }\n\n // Match HTML select and set the first selected\n // item as the value. Also set the selected array\n // in the order of the menu items.\n const selected: string[] = [];\n const selectedItems: MenuItem[] = [];\n\n this.childItemSet.forEach((childItem) => {\n if (childItem.menuData.selectionRoot !== this) return;\n\n if (this.selectedItemsMap.has(childItem)) {\n selected.push(childItem.value);\n selectedItems.push(childItem);\n }\n });\n this.selected = selected;\n this.selectedItems = selectedItems;\n this.value = this.selected.join(this.valueSeparator);\n } else {\n this.selectedItemsMap.clear();\n this.selectedItemsMap.set(targetItem, true);\n this.value = targetItem.value;\n this.selected = [targetItem.value];\n this.selectedItems = [targetItem];\n }\n\n await this.updateComplete;\n const applyDefault = this.dispatchEvent(\n new Event('change', {\n cancelable: true,\n bubbles: true,\n composed: true,\n })\n );\n if (!applyDefault) {\n // Cancel the event & don't apply the selection\n this.selected = oldSelected;\n this.selectedItems = oldSelectedItems;\n this.selectedItemsMap = oldSelectedItemsMap;\n this.value = oldValue;\n return;\n }\n // Apply the selection changes to the menu items\n if (resolvedSelects === 'single') {\n for (const oldItem of oldSelectedItemsMap.keys()) {\n if (oldItem !== targetItem) {\n oldItem.selected = false;\n }\n }\n targetItem.selected = true;\n } else if (resolvedSelects === 'multiple') {\n targetItem.selected = !targetItem.selected;\n }\n }\n\n protected navigateWithinMenu(event: KeyboardEvent): void {\n const { code } = event;\n const lastFocusedItem = this.childItems[this.focusedItemIndex];\n const direction = code === 'ArrowDown' ? 1 : -1;\n const itemToFocus = this.focusMenuItemByOffset(direction);\n if (itemToFocus === lastFocusedItem) {\n return;\n }\n event.preventDefault();\n itemToFocus.scrollIntoView({ block: 'nearest' });\n }\n\n protected navigateBetweenRelatedMenus(code: string): void {\n const shouldOpenSubmenu =\n (this.isLTR && code === 'ArrowRight') ||\n (!this.isLTR && code === 'ArrowLeft');\n const shouldCloseSelfAsSubmenu =\n (this.isLTR && code === 'ArrowLeft') ||\n (!this.isLTR && code === 'ArrowRight');\n if (shouldOpenSubmenu) {\n const lastFocusedItem = this.childItems[this.focusedItemIndex];\n if (lastFocusedItem?.hasSubmenu) {\n // Remove focus while opening overlay from keyboard or the visible focus\n // will slip back to the first item in the menu.\n this.blur();\n lastFocusedItem.openOverlay();\n }\n } else if (shouldCloseSelfAsSubmenu && this.isSubmenu) {\n this.dispatchEvent(new Event('close', { bubbles: true }));\n }\n }\n\n public handleKeydown(event: KeyboardEvent): void {\n const { code } = event;\n if (code === 'Tab') {\n this.prepareToCleanUp();\n return;\n }\n if (code === 'Space') {\n const lastFocusedItem = this.childItems[this.focusedItemIndex];\n if (lastFocusedItem?.hasSubmenu) {\n // Remove focus while opening overlay from keyboard or the visible focus\n // will slip back to the first item in the menu.\n this.blur();\n lastFocusedItem.openOverlay();\n return;\n }\n }\n if (code === 'Space' || code === 'Enter') {\n this.childItems[this.focusedItemIndex]?.click();\n return;\n }\n if (code === 'ArrowDown' || code === 'ArrowUp') {\n this.navigateWithinMenu(event);\n return;\n }\n this.navigateBetweenRelatedMenus(code);\n }\n\n public focusMenuItemByOffset(offset: number): MenuItem {\n const step = offset || 1;\n const focusedItem = this.childItems[this.focusedItemIndex];\n focusedItem.focused = false;\n this.focusedItemIndex =\n (this.childItems.length + this.focusedItemIndex + offset) %\n this.childItems.length;\n let itemToFocus = this.childItems[this.focusedItemIndex];\n let availableItems = this.childItems.length;\n // cycle through the available items in the directions of the offset to find the next non-disabled item\n while (itemToFocus.disabled && availableItems) {\n availableItems -= 1;\n this.focusedItemIndex =\n (this.childItems.length + this.focusedItemIndex + step) %\n this.childItems.length;\n itemToFocus = this.childItems[this.focusedItemIndex];\n }\n // if there are no non-disabled items, skip the work to focus a child\n if (!itemToFocus?.disabled) {\n this.forwardFocusVisibleToItem(itemToFocus);\n }\n return itemToFocus;\n }\n\n private prepareToCleanUp(): void {\n document.addEventListener(\n 'focusout',\n () => {\n requestAnimationFrame(() => {\n const focusedItem = this.childItems[this.focusedItemIndex];\n if (focusedItem) {\n focusedItem.focused = false;\n this.updateSelectedItemIndex();\n }\n });\n },\n { once: true }\n );\n }\n\n public updateSelectedItemIndex(): void {\n let firstOrFirstSelectedIndex = 0;\n const selectedItemsMap = new Map<MenuItem, boolean>();\n const selected: string[] = [];\n const selectedItems: MenuItem[] = [];\n let itemIndex = this.childItems.length;\n while (itemIndex) {\n itemIndex -= 1;\n const childItem = this.childItems[itemIndex];\n if (childItem.menuData.selectionRoot === this) {\n if (childItem.selected) {\n firstOrFirstSelectedIndex = itemIndex;\n selectedItemsMap.set(childItem, true);\n selected.unshift(childItem.value);\n selectedItems.unshift(childItem);\n }\n // Remove \"focused\" from non-\"selected\" items ONLY\n // Preserve \"focused\" on index===0 when no selection\n if (itemIndex !== firstOrFirstSelectedIndex) {\n childItem.focused = false;\n }\n }\n }\n selectedItems.map((item, i) => {\n // When there is more than one \"selected\" item,\n // ensure only the first one can be \"focused\"\n if (i > 0) {\n item.focused = false;\n }\n });\n this.selectedItemsMap = selectedItemsMap;\n this.selected = selected;\n this.selectedItems = selectedItems;\n this.value = this.selected.join(this.valueSeparator);\n this.focusedItemIndex = firstOrFirstSelectedIndex;\n this.focusInItemIndex = firstOrFirstSelectedIndex;\n }\n\n private _willUpdateItems = false;\n\n private handleItemsChanged(): void {\n this.cachedChildItems = undefined;\n if (!this._willUpdateItems) {\n /* c8 ignore next 3 */\n let resolve = (): void => {\n return;\n };\n this.cacheUpdated = new Promise((res) => (resolve = res));\n this._willUpdateItems = true;\n // Debounce the update so we only update once\n // if multiple items have changed\n window.requestAnimationFrame(() => {\n if (this.cachedChildItems === undefined) {\n this.updateSelectedItemIndex();\n this.updateItemFocus();\n }\n this._willUpdateItems = false;\n resolve();\n });\n }\n }\n\n private updateItemFocus(): void {\n if (this.childItems.length == 0) {\n return;\n }\n const focusInItem = this.childItems[this.focusInItemIndex];\n if (\n (this.getRootNode() as Document).activeElement ===\n focusInItem.menuData.focusRoot\n ) {\n this.forwardFocusVisibleToItem(focusInItem);\n }\n }\n\n private forwardFocusVisibleToItem(item: MenuItem): void {\n if (item.menuData.focusRoot !== this) {\n return;\n }\n item.focused = this.hasVisibleFocusInTree();\n this.setAttribute('aria-activedescendant', item.id);\n if (\n item.menuData.selectionRoot &&\n item.menuData.selectionRoot !== this\n ) {\n item.menuData.selectionRoot.focus();\n }\n }\n\n public override render(): TemplateResult {\n return html`\n <slot></slot>\n `;\n }\n\n private _notFirstUpdated = false;\n\n protected override firstUpdated(changed: PropertyValues): void {\n super.firstUpdated(changed);\n if (!this.hasAttribute('tabindex')) {\n const role = this.getAttribute('role');\n if (role === 'group') {\n this.tabIndex = -1;\n } else if (role !== 'none') {\n this.tabIndex = 0;\n }\n }\n const updates: Promise<unknown>[] = [\n new Promise((res) => requestAnimationFrame(() => res(true))),\n ];\n [...this.children].forEach((item) => {\n if ((item as MenuItem).localName === 'sp-menu-item') {\n updates.push((item as MenuItem).updateComplete);\n }\n });\n this.childItemsUpdated = Promise.all(updates);\n }\n\n protected override updated(changes: PropertyValues<this>): void {\n super.updated(changes);\n if (changes.has('selects') && this._notFirstUpdated) {\n this.selectsChanged();\n }\n if (changes.has('label')) {\n if (this.label) {\n this.setAttribute('aria-label', this.label);\n } else {\n this.removeAttribute('aria-label');\n }\n }\n this._notFirstUpdated = true;\n }\n\n protected selectsChanged(): void {\n const updates: Promise<unknown>[] = [\n new Promise((res) => requestAnimationFrame(() => res(true))),\n ];\n this.childItemSet.forEach((childItem) => {\n updates.push(childItem.triggerUpdate());\n });\n this.childItemsUpdated = Promise.all(updates);\n }\n\n public override connectedCallback(): void {\n super.connectedCallback();\n if (!this.hasAttribute('role')) {\n this.setAttribute('role', this.ownRole);\n }\n this.updateComplete.then(() => this.updateItemFocus());\n }\n\n protected childItemsUpdated!: Promise<unknown[]>;\n protected cacheUpdated = Promise.resolve();\n\n protected override async getUpdateComplete(): Promise<boolean> {\n const complete = (await super.getUpdateComplete()) as boolean;\n await this.childItemsUpdated;\n await this.cacheUpdated;\n return complete;\n }\n}\n"],
5
- "mappings": "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",
4
+ "sourcesContent": ["/*\nCopyright 2020 Adobe. All rights reserved.\nThis file is licensed to you under the Apache License, Version 2.0 (the \"License\");\nyou may not use this file except in compliance with the License. You may obtain a copy\nof the License at http://www.apache.org/licenses/LICENSE-2.0\n\nUnless required by applicable law or agreed to in writing, software distributed under\nthe License is distributed on an \"AS IS\" BASIS, WITHOUT WARRANTIES OR REPRESENTATIONS\nOF ANY KIND, either express or implied. See the License for the specific language\ngoverning permissions and limitations under the License.\n*/\n\nimport {\n CSSResultArray,\n html,\n PropertyValues,\n SpectrumElement,\n TemplateResult,\n} from '@spectrum-web-components/base';\nimport {\n property,\n query,\n} from '@spectrum-web-components/base/src/decorators.js';\n\nimport { MenuItem } from './MenuItem.dev.js'\nimport type {\n MenuItemAddedOrUpdatedEvent,\n MenuItemRemovedEvent,\n} from './MenuItem.dev.js'\nimport { OverlayBase } from '@spectrum-web-components/overlay/src/OverlayBase.js';\nimport menuStyles from './menu.css.js';\n\nexport interface MenuChildItem {\n menuItem: MenuItem;\n managed: boolean;\n active: boolean;\n focusable: boolean;\n focusRoot: Menu;\n}\n\ntype SelectsType = 'none' | 'ignore' | 'inherit' | 'multiple' | 'single';\ntype RoleType = 'group' | 'menu' | 'listbox' | 'none';\n\nfunction elementIsOrContains(\n el: Node,\n isOrContains: Node | undefined | null\n): boolean {\n return !!isOrContains && (el === isOrContains || el.contains(isOrContains));\n}\n\n/**\n * Spectrum Menu Component\n * @element sp-menu\n *\n * @slot - menu items to be listed in the menu\n * @fires change - Announces that the `value` of the element has changed\n * @attr selects - whether the element has a specific selection algorithm that it applies\n * to its item descendants. `single` allows only one descendent to be selected at a time.\n * `multiple` allows many descendants to be selected. `inherit` will be applied dynamically\n * when an ancestor of this element is actively managing the selection of its descendents.\n * When the `selects` attribute is not present a `value` will not be maintained and the Menu\n * Item children of this Menu will not have their `selected` state managed.\n */\nexport class Menu extends SpectrumElement {\n public static override get styles(): CSSResultArray {\n return [menuStyles];\n }\n\n private get isSubmenu(): boolean {\n return this.slot === 'submenu';\n }\n\n @property({ type: String, reflect: true })\n public label = '';\n\n @property({ type: String, reflect: true })\n public selects: undefined | 'inherit' | 'single' | 'multiple';\n\n @property({ type: String })\n public value = '';\n\n // For the multiple select case, we'll join the value strings together\n // for the value property with this separator\n @property({ type: String, attribute: 'value-separator' })\n public valueSeparator = ',';\n\n // TODO: which of these to keep?\n // TODO: allow setting this in the API to change the values\n @property({ attribute: false })\n public selected = [] as string[];\n\n @property({ attribute: false })\n public selectedItems = [] as MenuItem[];\n\n @query('slot:not([name])')\n public menuSlot!: HTMLSlotElement;\n\n private childItemSet = new Set<MenuItem>();\n public focusedItemIndex = 0;\n public focusInItemIndex = 0;\n\n private selectedItemsMap = new Map() as Map<MenuItem, boolean>;\n\n public get childItems(): MenuItem[] {\n if (!this.cachedChildItems) {\n this.cachedChildItems = this.updateCachedMenuItems();\n }\n return this.cachedChildItems;\n }\n\n private cachedChildItems: MenuItem[] | undefined;\n\n private updateCachedMenuItems(): MenuItem[] {\n this.cachedChildItems = [];\n\n const slotElements = this.menuSlot\n ? this.menuSlot.assignedElements({ flatten: true })\n : [];\n for (const slotElement of slotElements) {\n const childMenuItems: MenuItem[] =\n slotElement instanceof MenuItem\n ? [slotElement as MenuItem]\n : ([...slotElement.querySelectorAll(`*`)] as MenuItem[]);\n\n for (const childMenuItem of childMenuItems) {\n if (this.childItemSet.has(childMenuItem)) {\n this.cachedChildItems.push(childMenuItem);\n }\n }\n }\n\n return this.cachedChildItems;\n }\n\n /**\n * Hide this getter from web-component-analyzer until\n * https://github.com/runem/web-component-analyzer/issues/131\n * has been addressed.\n *\n * @private\n */\n public get childRole(): string {\n if (this.resolvedRole === 'listbox') {\n return 'option';\n }\n switch (this.resolvedSelects) {\n case 'single':\n return 'menuitemradio';\n case 'multiple':\n return 'menuitemcheckbox';\n default:\n return 'menuitem';\n }\n }\n\n protected get ownRole(): string {\n return 'menu';\n }\n\n private resolvedSelects?: SelectsType;\n private resolvedRole?: RoleType;\n\n /**\n * When a descendant `<sp-menu-item>` element is added or updated it will dispatch\n * this event to announce its presence in the DOM. During the capture phase the first\n * Menu based element that the event encounters will manage the focus state of the\n * dispatching `<sp-menu-item>` element.\n * @param event\n */\n private onFocusableItemAddedOrUpdated(\n event: MenuItemAddedOrUpdatedEvent\n ): void {\n event.menuCascade.set(this, {\n hadFocusRoot: !!event.item.menuData.focusRoot,\n ancestorWithSelects: event.currentAncestorWithSelects,\n });\n if (this.selects) {\n event.currentAncestorWithSelects = this;\n }\n event.focusRoot = this;\n }\n\n /**\n * When a descendant `<sp-menu-item>` element is added or updated it will dispatch\n * this event to announce its presence in the DOM. During the bubble phase the first\n * Menu based element that the event encounters that does not inherit selection will\n * manage the selection state of the dispatching `<sp-menu-item>` element.\n * @param event\n */\n private onSelectableItemAddedOrUpdated(\n event: MenuItemAddedOrUpdatedEvent\n ): void {\n const cascadeData = event.menuCascade.get(this);\n if (!cascadeData) return;\n\n event.parentMenu = this;\n if (cascadeData.hadFocusRoot) {\n // Only have one tab stop per Menu tree\n this.tabIndex = -1;\n }\n this.addChildItem(event.item);\n\n if (this.selects === 'inherit') {\n this.resolvedSelects = 'inherit';\n this.resolvedRole = (cascadeData.ancestorWithSelects?.getAttribute(\n 'role'\n ) ||\n this.getAttribute('role') ||\n undefined) as RoleType;\n } else if (this.selects) {\n this.resolvedRole = (this.getAttribute('role') ||\n undefined) as RoleType;\n this.resolvedSelects = this.selects;\n } else {\n this.resolvedRole = (this.getAttribute('role') ||\n undefined) as RoleType;\n // `none` predominately occurs when leveraging `sp-menu-group` as the internal\n // `sp-menu` should be non-functional to the selection or focus interactions.\n this.resolvedSelects =\n this.resolvedRole === 'none' ? 'ignore' : 'none';\n }\n\n const selects =\n this.resolvedSelects === 'single' ||\n this.resolvedSelects === 'multiple';\n if (\n (selects || (!this.selects && this.resolvedSelects !== 'ignore')) &&\n !event.item.menuData.selectionRoot\n ) {\n event.item.setRole(this.childRole);\n event.selectionRoot = this;\n }\n }\n\n private addChildItem(item: MenuItem): void {\n this.childItemSet.add(item);\n this.handleItemsChanged();\n }\n\n private removeChildItem = async (\n event: MenuItemRemovedEvent\n ): Promise<void> => {\n this.childItemSet.delete(event.item);\n this.cachedChildItems = undefined;\n if (event.item.focused) {\n this.handleItemsChanged();\n await this.updateComplete;\n this.focus();\n }\n };\n\n public constructor() {\n super();\n\n this.addEventListener(\n 'sp-menu-item-added-or-updated',\n this.onSelectableItemAddedOrUpdated\n );\n this.addEventListener(\n 'sp-menu-item-added-or-updated',\n this.onFocusableItemAddedOrUpdated,\n {\n capture: true,\n }\n );\n\n this.addEventListener('sp-menu-item-removed', this.removeChildItem);\n this.addEventListener('click', this.onClick);\n this.addEventListener('focusin', this.handleFocusin);\n this.addEventListener('focusout', this.handleFocusout);\n this.addEventListener('sp-opened', this.handleSubmenuOpened);\n }\n\n public override focus({ preventScroll }: FocusOptions = {}): void {\n if (\n !this.childItems.length ||\n this.childItems.every((childItem) => childItem.disabled)\n ) {\n return;\n }\n if (\n this.childItems.some(\n (childItem) => childItem.menuData.focusRoot !== this\n )\n ) {\n super.focus({ preventScroll });\n return;\n }\n this.focusMenuItemByOffset(0);\n super.focus({ preventScroll });\n const selectedItem = this.selectedItems[0];\n if (selectedItem && !preventScroll) {\n selectedItem.scrollIntoView({ block: 'nearest' });\n }\n }\n\n private onClick(event: Event): void {\n if (event.defaultPrevented) {\n return;\n }\n const path = event.composedPath();\n const target = path.find((el) => {\n /* c8 ignore next 3 */\n if (!(el instanceof Element)) {\n return false;\n }\n return el.getAttribute('role') === this.childRole;\n }) as MenuItem;\n if (target?.href && target.href.length) {\n // This event will NOT ALLOW CANCELATION as link action\n // cancelation should occur on the `<sp-menu-item>` itself.\n this.dispatchEvent(\n new Event('change', {\n bubbles: true,\n composed: true,\n })\n );\n return;\n } else if (\n target?.menuData.selectionRoot === this &&\n this.childItems.length\n ) {\n event.preventDefault();\n if (target.hasSubmenu || target.open) {\n return;\n }\n this.selectOrToggleItem(target);\n } else {\n return;\n }\n this.prepareToCleanUp();\n }\n\n public handleFocusin(event: FocusEvent): void {\n const wasOrContainedRelatedTarget = elementIsOrContains(\n this,\n event.relatedTarget as Node\n );\n if (\n this.childItems.some(\n (childItem) => childItem.menuData.focusRoot !== this\n )\n ) {\n return;\n }\n const activeElement = (this.getRootNode() as Document).activeElement as\n | MenuItem\n | Menu;\n const selectionRoot =\n this.childItems[this.focusedItemIndex]?.menuData.selectionRoot ||\n this;\n if (\n activeElement !== selectionRoot ||\n (!wasOrContainedRelatedTarget && event.target !== this)\n ) {\n selectionRoot.focus({ preventScroll: true });\n if (activeElement && this.focusedItemIndex === 0) {\n const offset = this.childItems.findIndex(\n (childItem) => childItem === activeElement\n );\n if (offset > 0) {\n this.focusMenuItemByOffset(offset);\n }\n }\n }\n this.startListeningToKeyboard();\n }\n\n public startListeningToKeyboard(): void {\n this.addEventListener('keydown', this.handleKeydown);\n }\n\n public handleFocusout(event: FocusEvent): void {\n if (elementIsOrContains(this, event.relatedTarget as Node)) {\n return;\n }\n this.stopListeningToKeyboard();\n this.childItems.forEach((child) => (child.focused = false));\n this.removeAttribute('aria-activedescendant');\n }\n\n public stopListeningToKeyboard(): void {\n this.removeEventListener('keydown', this.handleKeydown);\n }\n\n private descendentOverlays = new Map<OverlayBase, OverlayBase>();\n\n protected handleDescendentOverlayOpened(event: Event): void {\n const target = event.composedPath()[0] as MenuItem;\n if (!target.overlayElement) return;\n this.descendentOverlays.set(\n target.overlayElement,\n target.overlayElement\n );\n }\n\n public handleSubmenuOpened = (event: Event): void => {\n event.stopPropagation();\n const target = event.composedPath()[0] as OverlayBase;\n target.dispatchEvent(\n new Event('sp-menu-submenu-opened', {\n bubbles: true,\n composed: true,\n })\n );\n const focusedItem = this.childItems[this.focusedItemIndex];\n if (focusedItem) {\n focusedItem.focused = false;\n }\n const openedItem = event\n .composedPath()\n .find((el) => this.childItemSet.has(el as MenuItem));\n if (!openedItem) return;\n const openedItemIndex = this.childItems.indexOf(openedItem as MenuItem);\n this.focusedItemIndex = openedItemIndex;\n this.focusInItemIndex = openedItemIndex;\n };\n\n public async selectOrToggleItem(targetItem: MenuItem): Promise<void> {\n const resolvedSelects = this.resolvedSelects;\n const oldSelectedItemsMap = new Map(this.selectedItemsMap);\n const oldSelected = this.selected.slice();\n const oldSelectedItems = this.selectedItems.slice();\n const oldValue = this.value;\n this.childItems[this.focusedItemIndex].focused = false;\n this.childItems[this.focusedItemIndex].active = false;\n this.focusedItemIndex = this.childItems.indexOf(targetItem);\n this.forwardFocusVisibleToItem(targetItem);\n\n if (resolvedSelects === 'multiple') {\n if (this.selectedItemsMap.has(targetItem)) {\n this.selectedItemsMap.delete(targetItem);\n } else {\n this.selectedItemsMap.set(targetItem, true);\n }\n\n // Match HTML select and set the first selected\n // item as the value. Also set the selected array\n // in the order of the menu items.\n const selected: string[] = [];\n const selectedItems: MenuItem[] = [];\n\n this.childItemSet.forEach((childItem) => {\n if (childItem.menuData.selectionRoot !== this) return;\n\n if (this.selectedItemsMap.has(childItem)) {\n selected.push(childItem.value);\n selectedItems.push(childItem);\n }\n });\n this.selected = selected;\n this.selectedItems = selectedItems;\n this.value = this.selected.join(this.valueSeparator);\n } else {\n this.selectedItemsMap.clear();\n this.selectedItemsMap.set(targetItem, true);\n this.value = targetItem.value;\n this.selected = [targetItem.value];\n this.selectedItems = [targetItem];\n }\n\n await this.updateComplete;\n const applyDefault = this.dispatchEvent(\n new Event('change', {\n cancelable: true,\n bubbles: true,\n composed: true,\n })\n );\n if (!applyDefault) {\n // Cancel the event & don't apply the selection\n this.selected = oldSelected;\n this.selectedItems = oldSelectedItems;\n this.selectedItemsMap = oldSelectedItemsMap;\n this.value = oldValue;\n return;\n }\n // Apply the selection changes to the menu items\n if (resolvedSelects === 'single') {\n for (const oldItem of oldSelectedItemsMap.keys()) {\n if (oldItem !== targetItem) {\n oldItem.selected = false;\n }\n }\n targetItem.selected = true;\n } else if (resolvedSelects === 'multiple') {\n targetItem.selected = !targetItem.selected;\n }\n }\n\n protected navigateWithinMenu(event: KeyboardEvent): void {\n const { code } = event;\n const lastFocusedItem = this.childItems[this.focusedItemIndex];\n const direction = code === 'ArrowDown' ? 1 : -1;\n const itemToFocus = this.focusMenuItemByOffset(direction);\n if (itemToFocus === lastFocusedItem) {\n return;\n }\n event.preventDefault();\n event.stopPropagation();\n itemToFocus.scrollIntoView({ block: 'nearest' });\n }\n\n protected navigateBetweenRelatedMenus(event: KeyboardEvent): void {\n const { code } = event;\n const shouldOpenSubmenu =\n (this.isLTR && code === 'ArrowRight') ||\n (!this.isLTR && code === 'ArrowLeft');\n const shouldCloseSelfAsSubmenu =\n (this.isLTR && code === 'ArrowLeft') ||\n (!this.isLTR && code === 'ArrowRight');\n if (shouldOpenSubmenu) {\n event.stopPropagation();\n const lastFocusedItem = this.childItems[this.focusedItemIndex];\n if (lastFocusedItem?.hasSubmenu) {\n // Remove focus while opening overlay from keyboard or the visible focus\n // will slip back to the first item in the menu.\n // this.blur();\n lastFocusedItem.openOverlay();\n }\n } else if (shouldCloseSelfAsSubmenu && this.isSubmenu) {\n event.stopPropagation();\n this.dispatchEvent(new Event('close', { bubbles: true }));\n this.updateSelectedItemIndex();\n }\n }\n\n public handleKeydown(event: KeyboardEvent): void {\n if (\n event.target !== this &&\n this !== (event.target as HTMLElement).parentElement\n ) {\n return;\n }\n const { code } = event;\n if (code === 'Tab') {\n this.prepareToCleanUp();\n return;\n }\n if (code === 'Space') {\n const lastFocusedItem = this.childItems[this.focusedItemIndex];\n if (lastFocusedItem?.hasSubmenu) {\n // Remove focus while opening overlay from keyboard or the visible focus\n // will slip back to the first item in the menu.\n // this.blur();\n lastFocusedItem.openOverlay();\n return;\n }\n }\n if (code === 'Space' || code === 'Enter') {\n this.childItems[this.focusedItemIndex]?.click();\n return;\n }\n if (code === 'ArrowDown' || code === 'ArrowUp') {\n this.navigateWithinMenu(event);\n return;\n }\n this.navigateBetweenRelatedMenus(event);\n }\n\n public focusMenuItemByOffset(offset: number): MenuItem {\n const step = offset || 1;\n const focusedItem = this.childItems[this.focusedItemIndex];\n if (focusedItem) {\n focusedItem.focused = false;\n focusedItem.active = false;\n }\n this.focusedItemIndex =\n (this.childItems.length + this.focusedItemIndex + offset) %\n this.childItems.length;\n let itemToFocus = this.childItems[this.focusedItemIndex];\n let availableItems = this.childItems.length;\n // cycle through the available items in the directions of the offset to find the next non-disabled item\n while (itemToFocus.disabled && availableItems) {\n availableItems -= 1;\n this.focusedItemIndex =\n (this.childItems.length + this.focusedItemIndex + step) %\n this.childItems.length;\n itemToFocus = this.childItems[this.focusedItemIndex];\n }\n // if there are no non-disabled items, skip the work to focus a child\n if (!itemToFocus?.disabled) {\n this.forwardFocusVisibleToItem(itemToFocus);\n }\n return itemToFocus;\n }\n\n private prepareToCleanUp(): void {\n document.addEventListener(\n 'focusout',\n () => {\n requestAnimationFrame(() => {\n const focusedItem = this.childItems[this.focusedItemIndex];\n if (focusedItem) {\n focusedItem.focused = false;\n this.updateSelectedItemIndex();\n }\n });\n },\n { once: true }\n );\n }\n\n private _hasUpdatedSelectedItemIndex = false;\n\n public updateSelectedItemIndex(): void {\n let firstOrFirstSelectedIndex = 0;\n const selectedItemsMap = new Map<MenuItem, boolean>();\n const selected: string[] = [];\n const selectedItems: MenuItem[] = [];\n let itemIndex = this.childItems.length;\n while (itemIndex) {\n itemIndex -= 1;\n const childItem = this.childItems[itemIndex];\n if (childItem.menuData.selectionRoot === this) {\n if (\n childItem.selected ||\n (!this._hasUpdatedSelectedItemIndex &&\n this.selected.includes(childItem.value))\n ) {\n firstOrFirstSelectedIndex = itemIndex;\n selectedItemsMap.set(childItem, true);\n selected.unshift(childItem.value);\n selectedItems.unshift(childItem);\n }\n // Remove \"focused\" from non-\"selected\" items ONLY\n // Preserve \"focused\" on index===0 when no selection\n if (itemIndex !== firstOrFirstSelectedIndex) {\n childItem.focused = false;\n }\n }\n }\n selectedItems.map((item, i) => {\n // When there is more than one \"selected\" item,\n // ensure only the first one can be \"focused\"\n if (i > 0) {\n item.focused = false;\n }\n });\n this.selectedItemsMap = selectedItemsMap;\n this.selected = selected;\n this.selectedItems = selectedItems;\n this.value = this.selected.join(this.valueSeparator);\n this.focusedItemIndex = firstOrFirstSelectedIndex;\n this.focusInItemIndex = firstOrFirstSelectedIndex;\n }\n\n private _willUpdateItems?: number;\n\n private handleItemsChanged(): void {\n this.cachedChildItems = undefined;\n if (!this._willUpdateItems) {\n // collect ONE proise for all of the item updates in a batch\n this.cacheUpdated = new Promise(\n (res) =>\n (this.resolveCacheUpdated = (): void => {\n res();\n this._willUpdateItems = undefined;\n })\n );\n } else {\n // reset the animation frame when subsequent updates are received for the same batch\n cancelAnimationFrame(this._willUpdateItems);\n }\n this._willUpdateItems = requestAnimationFrame(() => {\n if (this._willUpdateItems || this.cachedChildItems === undefined) {\n this.updateSelectedItemIndex();\n this.updateItemFocus();\n }\n this.resolveCacheUpdated();\n });\n }\n\n private updateItemFocus(): void {\n if (this.childItems.length == 0) {\n return;\n }\n const focusInItem = this.childItems[this.focusInItemIndex];\n if (\n (this.getRootNode() as Document).activeElement ===\n focusInItem.menuData.focusRoot\n ) {\n this.forwardFocusVisibleToItem(focusInItem);\n }\n }\n\n public closeDescendentOverlays(): void {\n this.descendentOverlays.forEach((overlay) => {\n overlay.open = false;\n });\n this.descendentOverlays = new Map<OverlayBase, OverlayBase>();\n }\n\n private forwardFocusVisibleToItem(item: MenuItem): void {\n if (item.menuData.focusRoot !== this) {\n return;\n }\n this.closeDescendentOverlays();\n const focused =\n this.hasVisibleFocusInTree() ||\n !!this.childItems.find((child) => {\n return child.hasVisibleFocusInTree();\n });\n item.focused = focused;\n this.setAttribute('aria-activedescendant', item.id);\n if (\n item.menuData.selectionRoot &&\n item.menuData.selectionRoot !== this\n ) {\n item.menuData.selectionRoot.focus();\n }\n }\n\n private handleSlotchange({\n target,\n }: Event & { target: HTMLSlotElement }): void {\n const assignedElement = target.assignedElements({\n flatten: true,\n }) as MenuItem[];\n if (this.childItems.length !== assignedElement.length) {\n assignedElement.forEach((item) => {\n if (typeof item.triggerUpdate !== 'undefined') {\n item.triggerUpdate();\n }\n });\n }\n }\n\n protected renderMenuItemSlot(): TemplateResult {\n return html`\n <slot\n @sp-menu-submenu-opened=${this.handleDescendentOverlayOpened}\n @slotchange=${this.handleSlotchange}\n ></slot>\n `;\n }\n\n public override render(): TemplateResult {\n return this.renderMenuItemSlot();\n }\n\n private _notFirstUpdated = false;\n\n protected override firstUpdated(changed: PropertyValues): void {\n super.firstUpdated(changed);\n if (!this.hasAttribute('tabindex')) {\n const role = this.getAttribute('role');\n if (role === 'group') {\n this.tabIndex = -1;\n } else if (role !== 'none') {\n this.tabIndex = 0;\n }\n }\n const updates: Promise<unknown>[] = [\n new Promise((res) => requestAnimationFrame(() => res(true))),\n ];\n [...this.children].forEach((item) => {\n if ((item as MenuItem).localName === 'sp-menu-item') {\n updates.push((item as MenuItem).updateComplete);\n }\n });\n this.childItemsUpdated = Promise.all(updates);\n }\n\n protected override updated(changes: PropertyValues<this>): void {\n super.updated(changes);\n if (changes.has('selects') && this._notFirstUpdated) {\n this.selectsChanged();\n }\n if (changes.has('label')) {\n if (this.label) {\n this.setAttribute('aria-label', this.label);\n } else {\n this.removeAttribute('aria-label');\n }\n }\n this._notFirstUpdated = true;\n }\n\n protected selectsChanged(): void {\n const updates: Promise<unknown>[] = [\n new Promise((res) => requestAnimationFrame(() => res(true))),\n ];\n this.childItemSet.forEach((childItem) => {\n updates.push(childItem.triggerUpdate());\n });\n this.childItemsUpdated = Promise.all(updates);\n }\n\n public override connectedCallback(): void {\n super.connectedCallback();\n if (!this.hasAttribute('role')) {\n this.setAttribute('role', this.ownRole);\n }\n this.updateComplete.then(() => this.updateItemFocus());\n }\n\n protected childItemsUpdated!: Promise<unknown[]>;\n protected cacheUpdated = Promise.resolve();\n protected resolveCacheUpdated = (): void => {\n return;\n };\n\n protected override async getUpdateComplete(): Promise<boolean> {\n const complete = (await super.getUpdateComplete()) as boolean;\n await this.childItemsUpdated;\n await this.cacheUpdated;\n return complete;\n }\n}\n"],
5
+ "mappings": "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",
6
6
  "names": []
7
7
  }