@sbb-esta/lyne-elements 3.8.0 → 3.9.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/core/base-elements/open-close-base-element.d.ts +2 -0
- package/core/base-elements/open-close-base-element.d.ts.map +1 -1
- package/core/base-elements/open-close-base-element.js +4 -0
- package/core/controllers/escapable-overlay-controller.js +3 -3
- package/core/controllers/inert-controller.d.ts +8 -1
- package/core/controllers/inert-controller.d.ts.map +1 -1
- package/core/controllers/inert-controller.js +25 -13
- package/core/overlay/position.d.ts +20 -0
- package/core/overlay/position.d.ts.map +1 -1
- package/core/overlay/position.js +34 -23
- package/core/overlay.js +11 -10
- package/core/styles/core.scss +7 -0
- package/core.css +6 -0
- package/custom-elements.json +612 -138
- package/development/core/base-elements/open-close-base-element.d.ts +2 -0
- package/development/core/base-elements/open-close-base-element.d.ts.map +1 -1
- package/development/core/base-elements/open-close-base-element.js +5 -1
- package/development/core/controllers/escapable-overlay-controller.js +2 -2
- package/development/core/controllers/inert-controller.d.ts +8 -1
- package/development/core/controllers/inert-controller.d.ts.map +1 -1
- package/development/core/controllers/inert-controller.js +49 -30
- package/development/core/overlay/position.d.ts +20 -0
- package/development/core/overlay/position.d.ts.map +1 -1
- package/development/core/overlay/position.js +41 -1
- package/development/core/overlay.js +2 -1
- package/development/menu/common/menu-action-common.d.ts.map +1 -1
- package/development/menu/common/menu-action-common.js +20 -3
- package/development/menu/menu/menu.component.d.ts +21 -10
- package/development/menu/menu/menu.component.d.ts.map +1 -1
- package/development/menu/menu/menu.component.js +203 -61
- package/development/option/option/option.component.js +4 -2
- package/development/tabs/tab/tab.component.d.ts +10 -4
- package/development/tabs/tab/tab.component.d.ts.map +1 -1
- package/development/tabs/tab/tab.component.js +15 -16
- package/development/tabs/tab-group/tab-group.component.d.ts +15 -14
- package/development/tabs/tab-group/tab-group.component.d.ts.map +1 -1
- package/development/tabs/tab-group/tab-group.component.js +46 -175
- package/development/tabs/tab-label/tab-label.component.d.ts +21 -2
- package/development/tabs/tab-label/tab-label.component.d.ts.map +1 -1
- package/development/tabs/tab-label/tab-label.component.js +91 -6
- package/development/tooltip/tooltip.component.d.ts +6 -6
- package/development/tooltip/tooltip.component.d.ts.map +1 -1
- package/development/tooltip/tooltip.component.js +14 -7
- package/menu/common/menu-action-common.d.ts.map +1 -1
- package/menu/common/menu-action-common.js +15 -12
- package/menu/menu/menu.component.d.ts +21 -10
- package/menu/menu/menu.component.d.ts.map +1 -1
- package/menu/menu/menu.component.js +144 -80
- package/off-brand-theme.css +6 -0
- package/option/option/option.component.js +1 -1
- package/package.json +1 -1
- package/safety-theme.css +6 -0
- package/standard-theme.css +6 -0
- package/tabs/tab/tab.component.d.ts +10 -4
- package/tabs/tab/tab.component.d.ts.map +1 -1
- package/tabs/tab/tab.component.js +22 -24
- package/tabs/tab-group/tab-group.component.d.ts +15 -14
- package/tabs/tab-group/tab-group.component.d.ts.map +1 -1
- package/tabs/tab-group/tab-group.component.js +68 -122
- package/tabs/tab-label/tab-label.component.d.ts +21 -2
- package/tabs/tab-label/tab-label.component.d.ts.map +1 -1
- package/tabs/tab-label/tab-label.component.js +88 -46
- package/tooltip/tooltip.component.d.ts +6 -6
- package/tooltip/tooltip.component.d.ts.map +1 -1
- package/tooltip/tooltip.component.js +59 -54
|
@@ -11,12 +11,15 @@ import { customElement, property } from "lit/decorators.js";
|
|
|
11
11
|
import { ref } from "lit/directives/ref.js";
|
|
12
12
|
import { SbbFocusTrapController, isArrowKeyOrPageKeysPressed, interactivityChecker, getNextElementIndex } from "../../core/a11y.js";
|
|
13
13
|
import { SbbOpenCloseBaseElement } from "../../core/base-elements.js";
|
|
14
|
-
import { SbbEscapableOverlayController, SbbInertController,
|
|
14
|
+
import { SbbEscapableOverlayController, SbbInertController, SbbMediaQueryBreakpointSmallAndBelow, SbbMediaMatcherController, SbbDarkModeController, SbbLanguageController } from "../../core/controllers.js";
|
|
15
15
|
import { idReference, forceType } from "../../core/decorators.js";
|
|
16
16
|
import { SbbScrollHandler, isZeroAnimationDuration } from "../../core/dom.js";
|
|
17
17
|
import { forwardEvent } from "../../core/eventing.js";
|
|
18
|
+
import { i18nGoBack } from "../../core/i18n/i18n.js";
|
|
18
19
|
import { SbbNamedSlotListMixin } from "../../core/mixins.js";
|
|
19
|
-
import { isEventOnElement, removeAriaOverlayTriggerAttributes, setAriaOverlayTriggerAttributes, getElementPosition } from "../../core/overlay.js";
|
|
20
|
+
import { isEventOnElement, removeAriaOverlayTriggerAttributes, setAriaOverlayTriggerAttributes, getElementPosition, getElementPositionHorizontal } from "../../core/overlay.js";
|
|
21
|
+
import "../../divider.js";
|
|
22
|
+
import "../menu-button.js";
|
|
20
23
|
const style = css`*,
|
|
21
24
|
::before,
|
|
22
25
|
::after {
|
|
@@ -32,7 +35,8 @@ const style = css`*,
|
|
|
32
35
|
var(--sbb-animation-duration-6x)
|
|
33
36
|
);
|
|
34
37
|
--sbb-menu-animation-easing: ease;
|
|
35
|
-
--sbb-menu-transform:
|
|
38
|
+
--sbb-menu-transform-y: 100%;
|
|
39
|
+
--sbb-menu-transform-x: 0%;
|
|
36
40
|
--sbb-menu-max-width: 100%;
|
|
37
41
|
--sbb-menu-min-width: 100%;
|
|
38
42
|
--sbb-menu-inset: 0 auto auto 0;
|
|
@@ -54,9 +58,10 @@ const style = css`*,
|
|
|
54
58
|
}
|
|
55
59
|
@media (min-width: calc(52.5rem)) {
|
|
56
60
|
:host {
|
|
57
|
-
--sbb-menu-transform:
|
|
61
|
+
--sbb-menu-transform-y: var(--sbb-spacing-fixed-2x);
|
|
58
62
|
--sbb-menu-max-width: 20rem;
|
|
59
63
|
--sbb-menu-min-width: 11.25rem;
|
|
64
|
+
--sbb-menu-back-button-display: none;
|
|
60
65
|
}
|
|
61
66
|
}
|
|
62
67
|
|
|
@@ -70,6 +75,10 @@ const style = css`*,
|
|
|
70
75
|
}
|
|
71
76
|
}
|
|
72
77
|
|
|
78
|
+
:host(:not(:is(:state(nested),[state--nested]))) {
|
|
79
|
+
--sbb-menu-back-button-display: none;
|
|
80
|
+
}
|
|
81
|
+
|
|
73
82
|
:host(:not([data-state=closed])) {
|
|
74
83
|
--sbb-menu-inset: 0;
|
|
75
84
|
}
|
|
@@ -80,16 +89,36 @@ const style = css`*,
|
|
|
80
89
|
--sbb-scrollbar-color-hover: color-mix(in srgb, var(--sbb-color-black) 60%, transparent);
|
|
81
90
|
}
|
|
82
91
|
|
|
92
|
+
@media (max-width: calc(52.4375rem)) {
|
|
93
|
+
:host(:is(:state(skip-animation),[state--skip-animation])) {
|
|
94
|
+
--sbb-menu-animation-duration: 0ms;
|
|
95
|
+
}
|
|
96
|
+
:host(:is(:state(nested-child),[state--nested-child])) {
|
|
97
|
+
--sbb-menu-transform-x: -100%;
|
|
98
|
+
}
|
|
99
|
+
:host(:is(:state(nested),[state--nested])[data-state]:not([data-state=closed])) {
|
|
100
|
+
--sbb-menu-open-animation-name: open-sideways;
|
|
101
|
+
}
|
|
102
|
+
:host([data-state][data-state=closing]:is(:state(nested),[state--nested]):not(:is(:state(close-all),[state--close-all]))) {
|
|
103
|
+
--sbb-menu-close-animation-name: close-sideways;
|
|
104
|
+
}
|
|
105
|
+
}
|
|
83
106
|
::slotted(:not(sbb-menu-button, sbb-menu-link, sbb-divider)) {
|
|
84
107
|
display: block;
|
|
85
108
|
padding-inline: var(--sbb-spacing-fixed-5x);
|
|
86
109
|
}
|
|
87
110
|
|
|
111
|
+
sbb-divider,
|
|
88
112
|
::slotted(sbb-divider) {
|
|
89
113
|
--sbb-divider-color: var(--sbb-background-color-4-inverted);
|
|
90
114
|
margin-block: var(--sbb-spacing-fixed-2x);
|
|
91
115
|
}
|
|
92
116
|
|
|
117
|
+
sbb-divider,
|
|
118
|
+
#sbb-menu__back-button {
|
|
119
|
+
display: var(--sbb-menu-back-button-display, block);
|
|
120
|
+
}
|
|
121
|
+
|
|
93
122
|
.sbb-menu__container {
|
|
94
123
|
position: fixed;
|
|
95
124
|
pointer-events: none;
|
|
@@ -109,6 +138,9 @@ const style = css`*,
|
|
|
109
138
|
transition-timing-function: var(--sbb-menu-animation-easing);
|
|
110
139
|
transition-property: background-color, visibility;
|
|
111
140
|
}
|
|
141
|
+
:host(:is(:state(nested),[state--nested])) .sbb-menu__container::before {
|
|
142
|
+
display: none;
|
|
143
|
+
}
|
|
112
144
|
|
|
113
145
|
.sbb-menu {
|
|
114
146
|
display: none;
|
|
@@ -128,18 +160,20 @@ const style = css`*,
|
|
|
128
160
|
background-color: var(--sbb-menu-background-color);
|
|
129
161
|
padding: 0;
|
|
130
162
|
overflow: hidden;
|
|
163
|
+
translate: var(--sbb-menu-transform-x) 0;
|
|
164
|
+
transition: translate var(--sbb-menu-animation-duration);
|
|
131
165
|
}
|
|
132
166
|
:host([data-state]:not([data-state=closed])) .sbb-menu {
|
|
133
167
|
display: block;
|
|
134
168
|
opacity: 1;
|
|
135
169
|
pointer-events: all;
|
|
136
|
-
animation-name: open;
|
|
170
|
+
animation-name: var(--sbb-menu-open-animation-name, open);
|
|
137
171
|
animation-duration: var(--sbb-menu-animation-duration);
|
|
138
172
|
animation-timing-function: var(--sbb-menu-animation-easing);
|
|
139
173
|
}
|
|
140
174
|
:host([data-state][data-state=closing]) .sbb-menu {
|
|
141
175
|
pointer-events: none;
|
|
142
|
-
animation-name: close;
|
|
176
|
+
animation-name: var(--sbb-menu-close-animation-name, close);
|
|
143
177
|
}
|
|
144
178
|
@media (forced-colors: active) {
|
|
145
179
|
.sbb-menu {
|
|
@@ -231,24 +265,41 @@ const style = css`*,
|
|
|
231
265
|
@keyframes open {
|
|
232
266
|
from {
|
|
233
267
|
opacity: 0;
|
|
234
|
-
|
|
268
|
+
translate: 0 var(--sbb-menu-transform-y);
|
|
235
269
|
}
|
|
236
270
|
to {
|
|
237
271
|
opacity: 1;
|
|
238
|
-
|
|
272
|
+
translate: 0 0;
|
|
239
273
|
}
|
|
240
274
|
}
|
|
241
275
|
@keyframes close {
|
|
242
276
|
from {
|
|
243
277
|
opacity: 1;
|
|
244
|
-
|
|
278
|
+
translate: 0 0;
|
|
245
279
|
}
|
|
246
280
|
to {
|
|
247
281
|
opacity: 0;
|
|
248
|
-
|
|
282
|
+
translate: 0 var(--sbb-menu-transform-y);
|
|
283
|
+
}
|
|
284
|
+
}
|
|
285
|
+
@keyframes open-sideways {
|
|
286
|
+
from {
|
|
287
|
+
translate: 100% 0;
|
|
288
|
+
}
|
|
289
|
+
to {
|
|
290
|
+
translate: 0 0;
|
|
291
|
+
}
|
|
292
|
+
}
|
|
293
|
+
@keyframes close-sideways {
|
|
294
|
+
from {
|
|
295
|
+
translate: 0 0;
|
|
296
|
+
}
|
|
297
|
+
to {
|
|
298
|
+
translate: 100% 0;
|
|
249
299
|
}
|
|
250
300
|
}`;
|
|
251
301
|
const MENU_OFFSET = 8;
|
|
302
|
+
const NESTED_MENU_OFFSET = -4;
|
|
252
303
|
const INTERACTIVE_ELEMENTS = [
|
|
253
304
|
"A",
|
|
254
305
|
"BUTTON",
|
|
@@ -257,7 +308,9 @@ const INTERACTIVE_ELEMENTS = [
|
|
|
257
308
|
"SBB-LINK",
|
|
258
309
|
"SBB-BLOCK-LINK",
|
|
259
310
|
"SBB-LINK-BUTTON",
|
|
260
|
-
"SBB-BLOCK-LINK-BUTTON"
|
|
311
|
+
"SBB-BLOCK-LINK-BUTTON",
|
|
312
|
+
"SBB-MENU-BUTTON",
|
|
313
|
+
"SBB-MENU-LINK"
|
|
261
314
|
];
|
|
262
315
|
let nextId = 0;
|
|
263
316
|
let SbbMenuElement = (() => {
|
|
@@ -288,8 +341,9 @@ let SbbMenuElement = (() => {
|
|
|
288
341
|
this._focusTrapController = new SbbFocusTrapController(this);
|
|
289
342
|
this._scrollHandler = new SbbScrollHandler();
|
|
290
343
|
this._inertController = new SbbInertController(this);
|
|
344
|
+
this._mobileBreakpoint = SbbMediaQueryBreakpointSmallAndBelow;
|
|
291
345
|
this._mediaMatcher = new SbbMediaMatcherController(this, {
|
|
292
|
-
[
|
|
346
|
+
[this._mobileBreakpoint]: (matches) => {
|
|
293
347
|
if (matches && (this.state === "opening" || this.state === "opened")) {
|
|
294
348
|
this._scrollHandler.disableScroll();
|
|
295
349
|
} else {
|
|
@@ -298,15 +352,18 @@ let SbbMenuElement = (() => {
|
|
|
298
352
|
}
|
|
299
353
|
});
|
|
300
354
|
this._darkModeController = new SbbDarkModeController(this, () => this._syncNegative());
|
|
355
|
+
this._language = new SbbLanguageController(this);
|
|
356
|
+
this._nestedMenu = null;
|
|
301
357
|
this._pointerDownListener = (event) => {
|
|
302
|
-
|
|
358
|
+
const menu = event.target.closest("sbb-menu");
|
|
359
|
+
this._isPointerDownEventOnMenu = isEventOnElement(this._menu, event) || this._nestedMenus().some((el) => menu === el);
|
|
303
360
|
};
|
|
304
361
|
this._closeOnBackdropClick = (event) => {
|
|
305
|
-
|
|
306
|
-
|
|
362
|
+
const target = event.target;
|
|
363
|
+
if (!this._isPointerDownEventOnMenu && !isEventOnElement(this._menu, event) && !this._nestedMenus().some((el) => el === target)) {
|
|
364
|
+
this.closeAll();
|
|
307
365
|
}
|
|
308
366
|
};
|
|
309
|
-
this.addEventListener?.("click", (e) => this._onClick(e));
|
|
310
367
|
this.addEventListener?.("keydown", (e) => this._handleKeyDown(e));
|
|
311
368
|
}
|
|
312
369
|
/**
|
|
@@ -330,34 +387,60 @@ let SbbMenuElement = (() => {
|
|
|
330
387
|
set listAccessibilityLabel(value) {
|
|
331
388
|
__privateSet(this, _listAccessibilityLabel_accessor_storage, value);
|
|
332
389
|
}
|
|
390
|
+
firstUpdated(changedProperties) {
|
|
391
|
+
super.firstUpdated(changedProperties);
|
|
392
|
+
this._configureTrigger();
|
|
393
|
+
}
|
|
394
|
+
escapeStrategy() {
|
|
395
|
+
this.closeAll();
|
|
396
|
+
}
|
|
333
397
|
/**
|
|
334
398
|
* Opens the menu on trigger click.
|
|
335
399
|
*/
|
|
336
400
|
open() {
|
|
337
|
-
if (this.state === "closing" || !this._menu) {
|
|
401
|
+
if (this.state === "closing" || this.state === "opened" || !this._menu || !this.dispatchBeforeOpenEvent()) {
|
|
338
402
|
return;
|
|
339
403
|
}
|
|
340
|
-
if (
|
|
341
|
-
|
|
404
|
+
if (this._isNested()) {
|
|
405
|
+
const parentMenu = this._parentMenu();
|
|
406
|
+
parentMenu.toggleState("nested-child", true);
|
|
407
|
+
if (parentMenu._nestedMenu !== this) {
|
|
408
|
+
parentMenu._nestedMenu?.close();
|
|
409
|
+
}
|
|
410
|
+
parentMenu._nestedMenu = this;
|
|
342
411
|
}
|
|
343
412
|
this.showPopover?.();
|
|
344
413
|
this.state = "opening";
|
|
345
414
|
this._setMenuPosition();
|
|
346
415
|
this._triggerElement?.setAttribute("aria-expanded", "true");
|
|
347
|
-
if (this.
|
|
416
|
+
if (this._isMobile()) {
|
|
348
417
|
this._scrollHandler.disableScroll();
|
|
349
418
|
}
|
|
350
419
|
if (this._isZeroAnimationDuration()) {
|
|
351
420
|
this._handleOpening();
|
|
352
421
|
}
|
|
353
422
|
}
|
|
354
|
-
/**
|
|
355
|
-
* Closes the menu.
|
|
356
|
-
*/
|
|
423
|
+
/** Closes the menu and all its nested menus. */
|
|
357
424
|
close() {
|
|
358
|
-
|
|
425
|
+
this._close();
|
|
426
|
+
}
|
|
427
|
+
/** Closes the menu and all related menus nested and parent menus). */
|
|
428
|
+
closeAll() {
|
|
429
|
+
this._mainMenu()._close(true);
|
|
430
|
+
}
|
|
431
|
+
/** @param [closeAll='false'] - If true, it ensures animations are correct by toggling some states when closing all related menus at once. */
|
|
432
|
+
_close(closeAll = false) {
|
|
433
|
+
if (this.state === "opening" && !this._isNested() || !this.dispatchBeforeCloseEvent()) {
|
|
359
434
|
return;
|
|
360
435
|
}
|
|
436
|
+
this._nestedMenu?._close(closeAll);
|
|
437
|
+
if (this._isNested()) {
|
|
438
|
+
const parentMenu = this._parentMenu();
|
|
439
|
+
this.toggleState("close-all", closeAll);
|
|
440
|
+
parentMenu.toggleState("skip-animation", closeAll);
|
|
441
|
+
parentMenu.toggleState("nested-child", false);
|
|
442
|
+
parentMenu._nestedMenu = null;
|
|
443
|
+
}
|
|
361
444
|
this.state = "closing";
|
|
362
445
|
this._triggerElement?.setAttribute("aria-expanded", "false");
|
|
363
446
|
if (this._isZeroAnimationDuration()) {
|
|
@@ -369,7 +452,11 @@ let SbbMenuElement = (() => {
|
|
|
369
452
|
}
|
|
370
453
|
_handleOpening() {
|
|
371
454
|
this.state = "opened";
|
|
372
|
-
this.
|
|
455
|
+
if (!this._isNested()) {
|
|
456
|
+
this._inertController.activate();
|
|
457
|
+
} else {
|
|
458
|
+
this._updateNestedInert();
|
|
459
|
+
}
|
|
373
460
|
this._escapableOverlayController.connect();
|
|
374
461
|
this._focusTrapController.focusInitialElement();
|
|
375
462
|
this._focusTrapController.enabled = true;
|
|
@@ -378,12 +465,18 @@ let SbbMenuElement = (() => {
|
|
|
378
465
|
}
|
|
379
466
|
_handleClosing() {
|
|
380
467
|
this.state = "closed";
|
|
468
|
+
this.toggleState("skip-animation", false);
|
|
469
|
+
this.toggleState("close-all", false);
|
|
381
470
|
this.hidePopover?.();
|
|
382
471
|
this._menu?.firstElementChild?.scrollTo(0, 0);
|
|
383
|
-
this.
|
|
472
|
+
if (!this._isNested()) {
|
|
473
|
+
this._inertController.deactivate();
|
|
474
|
+
} else {
|
|
475
|
+
this._updateNestedInert();
|
|
476
|
+
}
|
|
384
477
|
this._triggerElement?.focus({
|
|
385
478
|
// When inside the sbb-header, we prevent the scroll to avoid the snapping to the top of the page
|
|
386
|
-
preventScroll:
|
|
479
|
+
preventScroll: ["sbb-header-button", "sbb-header-link"].includes(this._triggerElement.localName)
|
|
387
480
|
});
|
|
388
481
|
this._escapableOverlayController.disconnect();
|
|
389
482
|
this.dispatchCloseEvent();
|
|
@@ -391,29 +484,28 @@ let SbbMenuElement = (() => {
|
|
|
391
484
|
this._focusTrapController.enabled = false;
|
|
392
485
|
this._scrollHandler.enableScroll();
|
|
393
486
|
}
|
|
394
|
-
/**
|
|
395
|
-
* Handles click and checks if its target is a sbb-menu-button/sbb-menu-link.
|
|
396
|
-
*/
|
|
397
|
-
_onClick(event) {
|
|
398
|
-
const target = event.target;
|
|
399
|
-
if (target?.localName === "sbb-menu-button" || target?.localName === "sbb-menu-link") {
|
|
400
|
-
this.close();
|
|
401
|
-
}
|
|
402
|
-
}
|
|
403
487
|
_handleKeyDown(evt) {
|
|
404
488
|
if (!isArrowKeyOrPageKeysPressed(evt)) {
|
|
405
489
|
return;
|
|
406
490
|
}
|
|
407
491
|
evt.preventDefault();
|
|
408
|
-
const enabledActions = Array.from(this.querySelectorAll("sbb-menu-button, sbb-menu-link")).filter((el) => (!el.disabled || el.disabledInteractive) && interactivityChecker.isVisible(el));
|
|
492
|
+
const enabledActions = Array.from(this.querySelectorAll("sbb-menu-button, sbb-menu-link")).concat(this.shadowRoot.querySelector("sbb-menu-button")).filter((el) => (!el.disabled || el.disabledInteractive) && interactivityChecker.isVisible(el));
|
|
409
493
|
const current = enabledActions.findIndex((e) => e === evt.target);
|
|
410
494
|
let nextIndex;
|
|
411
495
|
switch (evt.key) {
|
|
412
496
|
case "ArrowUp":
|
|
413
497
|
case "ArrowDown":
|
|
498
|
+
nextIndex = getNextElementIndex(evt, current, enabledActions.length);
|
|
499
|
+
break;
|
|
414
500
|
case "ArrowLeft":
|
|
501
|
+
if (this._isNested()) {
|
|
502
|
+
this.close();
|
|
503
|
+
}
|
|
504
|
+
break;
|
|
415
505
|
case "ArrowRight":
|
|
416
|
-
|
|
506
|
+
if (evt.target.hasAttribute("data-sbb-menu-trigger")) {
|
|
507
|
+
evt.target.click();
|
|
508
|
+
}
|
|
417
509
|
break;
|
|
418
510
|
case "PageUp":
|
|
419
511
|
case "Home":
|
|
@@ -423,11 +515,10 @@ let SbbMenuElement = (() => {
|
|
|
423
515
|
case "PageDown":
|
|
424
516
|
nextIndex = enabledActions.length - 1;
|
|
425
517
|
break;
|
|
426
|
-
// this should never happen since all the case allowed by `isArrowKeyOrPageKeysPressed` should be covered
|
|
427
|
-
default:
|
|
428
|
-
nextIndex = 0;
|
|
429
518
|
}
|
|
430
|
-
|
|
519
|
+
if (nextIndex !== void 0) {
|
|
520
|
+
enabledActions[nextIndex].focus();
|
|
521
|
+
}
|
|
431
522
|
}
|
|
432
523
|
// Removes trigger click listener on trigger change.
|
|
433
524
|
createRenderRoot() {
|
|
@@ -461,10 +552,6 @@ let SbbMenuElement = (() => {
|
|
|
461
552
|
this._configureTrigger();
|
|
462
553
|
}
|
|
463
554
|
}
|
|
464
|
-
firstUpdated(changedProperties) {
|
|
465
|
-
super.firstUpdated(changedProperties);
|
|
466
|
-
this._configureTrigger();
|
|
467
|
-
}
|
|
468
555
|
_checkListCase(event) {
|
|
469
556
|
if (this.children?.length && Array.from(this.children ?? []).every((c) => c.localName === "sbb-menu-button" || c.localName === "sbb-menu-link")) {
|
|
470
557
|
return;
|
|
@@ -491,6 +578,8 @@ let SbbMenuElement = (() => {
|
|
|
491
578
|
this._triggerElement.addEventListener("click", () => this.open(), {
|
|
492
579
|
signal: this._triggerAbortController.signal
|
|
493
580
|
});
|
|
581
|
+
this.toggleState("nested", ["sbb-menu-button", "sbb-menu-link"].includes(this._triggerElement.localName));
|
|
582
|
+
this._triggerElement.toggleAttribute("data-sbb-menu-trigger", true);
|
|
494
583
|
}
|
|
495
584
|
_attachWindowEvents() {
|
|
496
585
|
this._windowEventsController = new AbortController();
|
|
@@ -505,18 +594,55 @@ let SbbMenuElement = (() => {
|
|
|
505
594
|
passive: true,
|
|
506
595
|
signal: this._windowEventsController.signal
|
|
507
596
|
});
|
|
508
|
-
|
|
509
|
-
|
|
510
|
-
|
|
511
|
-
|
|
512
|
-
|
|
513
|
-
|
|
597
|
+
if (!this._isNested()) {
|
|
598
|
+
window.addEventListener("pointerdown", this._pointerDownListener, {
|
|
599
|
+
signal: this._windowEventsController.signal
|
|
600
|
+
});
|
|
601
|
+
window.addEventListener("pointerup", this._closeOnBackdropClick, {
|
|
602
|
+
signal: this._windowEventsController.signal
|
|
603
|
+
});
|
|
604
|
+
}
|
|
514
605
|
}
|
|
515
606
|
// Close menu at any click on an interactive element inside the <sbb-menu> that bubbles to the container.
|
|
516
|
-
|
|
607
|
+
_interactiveElementClick(event) {
|
|
517
608
|
const target = event.target;
|
|
518
|
-
if (INTERACTIVE_ELEMENTS.includes(target.nodeName) && !target.hasAttribute("disabled")) {
|
|
519
|
-
this.
|
|
609
|
+
if (INTERACTIVE_ELEMENTS.includes(target.nodeName) && !target.hasAttribute("disabled") && !target.hasAttribute("data-sbb-menu-trigger") && target.id !== "sbb-menu__back-button") {
|
|
610
|
+
this.closeAll();
|
|
611
|
+
}
|
|
612
|
+
}
|
|
613
|
+
/** Converts the linked list into an array of SbbMenuElement. */
|
|
614
|
+
_nestedMenus() {
|
|
615
|
+
const menus = [];
|
|
616
|
+
let current = this._nestedMenu;
|
|
617
|
+
while (current) {
|
|
618
|
+
menus.push(current);
|
|
619
|
+
current = current._nestedMenu;
|
|
620
|
+
}
|
|
621
|
+
return menus;
|
|
622
|
+
}
|
|
623
|
+
_parentMenu() {
|
|
624
|
+
return this._triggerElement?.closest("sbb-menu") ?? null;
|
|
625
|
+
}
|
|
626
|
+
/** The outermost menu. */
|
|
627
|
+
_mainMenu() {
|
|
628
|
+
return this._isNested() ? this._parentMenu()?._mainMenu() ?? this : this;
|
|
629
|
+
}
|
|
630
|
+
_isNested() {
|
|
631
|
+
return !!this._parentMenu();
|
|
632
|
+
}
|
|
633
|
+
_updateNestedInert() {
|
|
634
|
+
this._inertController.restoreAllExempted();
|
|
635
|
+
this._mainMenu()._nestedMenus().forEach((menu) => this._inertController.exempt(menu));
|
|
636
|
+
}
|
|
637
|
+
// Check if nested menu should be closed.
|
|
638
|
+
_handleMouseOver(event) {
|
|
639
|
+
const element = event.target;
|
|
640
|
+
const isMobile = this._isMobile();
|
|
641
|
+
if (!isMobile && this._nestedMenu && !element.classList.contains("sbb-menu__content") && !(element.getAttribute("aria-expanded") === "true")) {
|
|
642
|
+
this._nestedMenu.close();
|
|
643
|
+
}
|
|
644
|
+
if (element.hasAttribute("data-sbb-menu-trigger") && !isMobile) {
|
|
645
|
+
element.click();
|
|
520
646
|
}
|
|
521
647
|
}
|
|
522
648
|
// Set menu position (x, y) to '0' once the menu is closed and the transition ended to prevent the
|
|
@@ -524,19 +650,23 @@ let SbbMenuElement = (() => {
|
|
|
524
650
|
// In rare cases it can be that the animationEnd event is triggered twice.
|
|
525
651
|
// To avoid entering a corrupt state, exit when state is not expected.
|
|
526
652
|
_onMenuAnimationEnd(event) {
|
|
527
|
-
if (event.animationName === "open" && this.state === "opening") {
|
|
653
|
+
if ((event.animationName === "open" || event.animationName === "open-sideways") && this.state === "opening") {
|
|
528
654
|
this._handleOpening();
|
|
529
|
-
} else if (event.animationName === "close" && this.state === "closing") {
|
|
655
|
+
} else if ((event.animationName === "close" || event.animationName === "close-sideways") && this.state === "closing") {
|
|
530
656
|
this._handleClosing();
|
|
531
657
|
}
|
|
532
658
|
}
|
|
533
659
|
// Set menu position and max height if the breakpoint is medium-ultra.
|
|
534
660
|
_setMenuPosition() {
|
|
535
|
-
if (
|
|
661
|
+
if (this._isMobile() || !this._menu || !this._triggerElement || this.state === "closing") {
|
|
536
662
|
return;
|
|
537
663
|
}
|
|
538
|
-
const menuPosition = getElementPosition(this.shadowRoot.querySelector(".sbb-menu__content"), this._triggerElement, this.shadowRoot.querySelector(".sbb-menu__container"), {
|
|
664
|
+
const menuPosition = !this._isNested() ? getElementPosition(this.shadowRoot.querySelector(".sbb-menu__content"), this._triggerElement, this.shadowRoot.querySelector(".sbb-menu__container"), {
|
|
539
665
|
verticalOffset: MENU_OFFSET
|
|
666
|
+
}) : getElementPositionHorizontal(this.shadowRoot.querySelector(".sbb-menu__content"), this._triggerElement, this.shadowRoot.querySelector(".sbb-menu__container"), {
|
|
667
|
+
horizontalOffset: MENU_OFFSET,
|
|
668
|
+
verticalOffset: NESTED_MENU_OFFSET,
|
|
669
|
+
contentSelector: ".sbb-menu__content"
|
|
540
670
|
});
|
|
541
671
|
this.style.setProperty("--sbb-menu-position-x", `${menuPosition.left}px`);
|
|
542
672
|
this.style.setProperty("--sbb-menu-position-y", `${menuPosition.top}px`);
|
|
@@ -548,20 +678,32 @@ let SbbMenuElement = (() => {
|
|
|
548
678
|
el.negative = !this._darkModeController.matches();
|
|
549
679
|
});
|
|
550
680
|
}
|
|
681
|
+
_isMobile() {
|
|
682
|
+
return this._mediaMatcher.matches(this._mobileBreakpoint) ?? true;
|
|
683
|
+
}
|
|
551
684
|
render() {
|
|
552
685
|
return html`
|
|
553
686
|
<div class="sbb-menu__container">
|
|
554
687
|
<div
|
|
555
688
|
@animationend=${this._onMenuAnimationEnd}
|
|
689
|
+
@mouseover=${(e) => this._handleMouseOver(e)}
|
|
556
690
|
class="sbb-menu"
|
|
557
691
|
${ref((el) => this._menu = el)}
|
|
558
692
|
>
|
|
559
693
|
<div
|
|
560
|
-
@click=${(event) => this.
|
|
694
|
+
@click=${(event) => this._interactiveElementClick(event)}
|
|
561
695
|
@scroll=${(e) => forwardEvent(e, document)}
|
|
562
696
|
class="sbb-menu__content"
|
|
563
697
|
>
|
|
564
698
|
${this.listChildren.length ? this.renderList({ class: "sbb-menu-list", ariaLabel: this.listAccessibilityLabel }) : html`<slot></slot>`}
|
|
699
|
+
<sbb-divider></sbb-divider>
|
|
700
|
+
<sbb-menu-button
|
|
701
|
+
id="sbb-menu__back-button"
|
|
702
|
+
@click=${() => this.close()}
|
|
703
|
+
icon-name="chevron-small-left-small"
|
|
704
|
+
>
|
|
705
|
+
${i18nGoBack[this._language.current]}
|
|
706
|
+
</sbb-menu-button>
|
|
565
707
|
</div>
|
|
566
708
|
</div>
|
|
567
709
|
</div>
|
|
@@ -586,4 +728,4 @@ let SbbMenuElement = (() => {
|
|
|
586
728
|
export {
|
|
587
729
|
SbbMenuElement
|
|
588
730
|
};
|
|
589
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"menu.component.js","sources":["../../../../../src/elements/menu/menu/menu.component.ts"],"sourcesContent":["import {\n  type CSSResultGroup,\n  html,\n  isServer,\n  type PropertyDeclaration,\n  type PropertyValues,\n  type TemplateResult,\n} from 'lit';\nimport { customElement, property } from 'lit/decorators.js';\nimport { ref } from 'lit/directives/ref.js';\n\nimport {\n  getNextElementIndex,\n  interactivityChecker,\n  isArrowKeyOrPageKeysPressed,\n  SbbFocusTrapController,\n} from '../../core/a11y.js';\nimport { SbbOpenCloseBaseElement } from '../../core/base-elements.js';\nimport {\n  SbbDarkModeController,\n  SbbEscapableOverlayController,\n  SbbInertController,\n  SbbMediaMatcherController,\n  SbbMediaQueryBreakpointSmallAndBelow,\n} from '../../core/controllers.js';\nimport { forceType, idReference } from '../../core/decorators.js';\nimport { isZeroAnimationDuration, SbbScrollHandler } from '../../core/dom.js';\nimport { forwardEvent } from '../../core/eventing.js';\nimport { SbbNamedSlotListMixin, type SbbNegativeMixinType } from '../../core/mixins.js';\nimport {\n  getElementPosition,\n  isEventOnElement,\n  removeAriaOverlayTriggerAttributes,\n  setAriaOverlayTriggerAttributes,\n} from '../../core/overlay.js';\nimport type { SbbMenuButtonElement } from '../menu-button.js';\nimport type { SbbMenuLinkElement } from '../menu-link.js';\n\nimport style from './menu.scss?lit&inline';\n\nconst MENU_OFFSET = 8;\nconst INTERACTIVE_ELEMENTS = [\n  'A',\n  'BUTTON',\n  'SBB-BUTTON',\n  'SBB-BUTTON-LINK',\n  'SBB-LINK',\n  'SBB-BLOCK-LINK',\n  'SBB-LINK-BUTTON',\n  'SBB-BLOCK-LINK-BUTTON',\n];\n\nlet nextId = 0;\n\n/**\n * It displays a contextual menu with one or more action element.\n *\n * @slot - Use the unnamed slot to add `sbb-menu-button`/`sbb-menu-link` or other elements to the menu.\n * @cssprop [--sbb-menu-z-index=var(--sbb-overlay-default-z-index)] - To specify a custom stack order,\n * the `z-index` can be overridden by defining this CSS variable. The default `z-index` of the\n * component is set to `var(--sbb-overlay-default-z-index)` with a value of `1000`.\n */\nexport\n@customElement('sbb-menu')\nclass SbbMenuElement extends SbbNamedSlotListMixin<\n  SbbMenuButtonElement | SbbMenuLinkElement,\n  typeof SbbOpenCloseBaseElement\n>(SbbOpenCloseBaseElement) {\n  public static override styles: CSSResultGroup = style;\n  public static override readonly role = 'menu';\n  protected override readonly listChildLocalNames = ['sbb-menu-button', 'sbb-menu-link'];\n\n  /**\n   * The element that will trigger the menu overlay.\n   *\n   * For attribute usage, provide an id reference.\n   */\n  @idReference()\n  @property()\n  public accessor trigger: HTMLElement | null = null;\n\n  /**\n   * This will be forwarded as aria-label to the inner list.\n   * Used only if the menu automatically renders the actions inside as a list.\n   */\n  @forceType()\n  @property({ attribute: 'list-accessibility-label' })\n  public accessor listAccessibilityLabel: string = '';\n\n  private _menu!: HTMLDivElement;\n  private _triggerElement: HTMLElement | null = null;\n  private _triggerAbortController!: AbortController;\n  private _isPointerDownEventOnMenu: boolean = false;\n  private _windowEventsController!: AbortController;\n  private _escapableOverlayController = new SbbEscapableOverlayController(this);\n  private _focusTrapController = new SbbFocusTrapController(this);\n  private _scrollHandler = new SbbScrollHandler();\n  private _inertController = new SbbInertController(this);\n  private _mediaMatcher = new SbbMediaMatcherController(this, {\n    [SbbMediaQueryBreakpointSmallAndBelow]: (matches) => {\n      if (matches && (this.state === 'opening' || this.state === 'opened')) {\n        this._scrollHandler.disableScroll();\n      } else {\n        this._scrollHandler.enableScroll();\n      }\n    },\n  });\n  private _darkModeController = new SbbDarkModeController(this, () => this._syncNegative());\n\n  public constructor() {\n    super();\n    this.addEventListener?.('click', (e) => this._onClick(e));\n    this.addEventListener?.('keydown', (e) => this._handleKeyDown(e));\n  }\n\n  /**\n   * Opens the menu on trigger click.\n   */\n  public open(): void {\n    if (this.state === 'closing' || !this._menu) {\n      return;\n    }\n    if (!this.dispatchBeforeOpenEvent()) {\n      return;\n    }\n\n    this.showPopover?.();\n    this.state = 'opening';\n    this._setMenuPosition();\n    this._triggerElement?.setAttribute('aria-expanded', 'true');\n\n    // From zero to medium, disable scroll\n    if (this._mediaMatcher.matches(SbbMediaQueryBreakpointSmallAndBelow)) {\n      this._scrollHandler.disableScroll();\n    }\n\n    // If the animation duration is zero, the animationend event is not always fired reliably.\n    // In this case we directly set the `opened` state.\n    if (this._isZeroAnimationDuration()) {\n      this._handleOpening();\n    }\n  }\n\n  /**\n   * Closes the menu.\n   */\n  public close(): void {\n    if (this.state === 'opening' || !this.dispatchBeforeCloseEvent()) {\n      return;\n    }\n\n    this.state = 'closing';\n    this._triggerElement?.setAttribute('aria-expanded', 'false');\n\n    // If the animation duration is zero, the animationend event is not always fired reliably.\n    // In this case we directly set the `closed` state.\n    if (this._isZeroAnimationDuration()) {\n      this._handleClosing();\n    }\n  }\n\n  private _isZeroAnimationDuration(): boolean {\n    return isZeroAnimationDuration(this, '--sbb-menu-animation-duration');\n  }\n\n  private _handleOpening(): void {\n    this.state = 'opened';\n    this._inertController.activate();\n    this._escapableOverlayController.connect();\n    this._focusTrapController.focusInitialElement();\n    this._focusTrapController.enabled = true;\n    this._attachWindowEvents();\n    this.dispatchOpenEvent();\n  }\n\n  private _handleClosing(): void {\n    this.state = 'closed';\n    this.hidePopover?.();\n\n    this._menu?.firstElementChild?.scrollTo(0, 0);\n    this._inertController.deactivate();\n    // Manually focus last focused element\n    this._triggerElement?.focus({\n      // When inside the sbb-header, we prevent the scroll to avoid the snapping to the top of the page\n      preventScroll:\n        this._triggerElement.localName === 'sbb-header-button' ||\n        this._triggerElement.localName === 'sbb-header-link',\n    });\n    this._escapableOverlayController.disconnect();\n    this.dispatchCloseEvent();\n    this._windowEventsController?.abort();\n    this._focusTrapController.enabled = false;\n\n    // Starting from breakpoint medium, enable scroll\n    this._scrollHandler.enableScroll();\n  }\n\n  /**\n   * Handles click and checks if its target is a sbb-menu-button/sbb-menu-link.\n   */\n  private _onClick(event: Event): void {\n    const target = event.target as HTMLElement | undefined;\n    if (target?.localName === 'sbb-menu-button' || target?.localName === 'sbb-menu-link') {\n      this.close();\n    }\n  }\n\n  private _handleKeyDown(evt: KeyboardEvent): void {\n    if (!isArrowKeyOrPageKeysPressed(evt)) {\n      return;\n    }\n    evt.preventDefault();\n\n    const enabledActions: Element[] = Array.from(\n      this.querySelectorAll<SbbMenuButtonElement | SbbMenuLinkElement>(\n        'sbb-menu-button, sbb-menu-link',\n      ),\n    ).filter(\n      (el) => (!el.disabled || el.disabledInteractive) && interactivityChecker.isVisible(el),\n    );\n    const current = enabledActions.findIndex((e: Element) => e === evt.target);\n\n    let nextIndex;\n    switch (evt.key) {\n      case 'ArrowUp':\n      case 'ArrowDown':\n      case 'ArrowLeft':\n      case 'ArrowRight':\n        nextIndex = getNextElementIndex(evt, current, enabledActions.length);\n        break;\n\n      case 'PageUp':\n      case 'Home':\n        nextIndex = 0;\n        break;\n\n      case 'End':\n      case 'PageDown':\n        nextIndex = enabledActions.length - 1;\n        break;\n\n      // this should never happen since all the case allowed by `isArrowKeyOrPageKeysPressed` should be covered\n      default:\n        nextIndex = 0;\n    }\n\n    (enabledActions[nextIndex] as HTMLElement).focus();\n  }\n\n  // Removes trigger click listener on trigger change.\n  protected override createRenderRoot(): HTMLElement | DocumentFragment {\n    const renderRoot = super.createRenderRoot();\n    // Due to the fact that menu can both be a list and just a container, we need to check its\n    // state before the SbbNamedSlotListMixin handles the slotchange event, in order to avoid\n    // it interpreting the non list case as a list.\n    this.shadowRoot?.addEventListener(\n      'slotchange',\n      (e) => {\n        this._syncNegative();\n        this._checkListCase(e);\n      },\n      {\n        capture: true,\n      },\n    );\n    return renderRoot;\n  }\n\n  public override connectedCallback(): void {\n    this.popover = 'manual';\n    super.connectedCallback();\n    this.id ||= `sbb-menu-${nextId++}`;\n    if (this.hasUpdated) {\n      this._configureTrigger();\n    }\n  }\n\n  public override disconnectedCallback(): void {\n    super.disconnectedCallback();\n    this._triggerElement = null;\n    this._triggerAbortController?.abort();\n    this._windowEventsController?.abort();\n    this._scrollHandler.enableScroll();\n  }\n\n  public override requestUpdate(\n    name?: PropertyKey,\n    oldValue?: unknown,\n    options?: PropertyDeclaration,\n  ): void {\n    super.requestUpdate(name, oldValue, options);\n\n    if (!isServer && (!name || name === 'trigger') && this.hasUpdated) {\n      this._configureTrigger();\n    }\n  }\n\n  protected override firstUpdated(changedProperties: PropertyValues<this>): void {\n    super.firstUpdated(changedProperties);\n    this._configureTrigger();\n  }\n\n  private _checkListCase(event: Event): void {\n    // If all children are sbb-menu-button/menu-link instances, we render them as a list.\n    if (\n      this.children?.length &&\n      Array.from(this.children ?? []).every(\n        (c) => c.localName === 'sbb-menu-button' || c.localName === 'sbb-menu-link',\n      )\n    ) {\n      return;\n    }\n\n    event.stopImmediatePropagation();\n    if (this.listChildren.length) {\n      this.listChildren.forEach((c) => c.removeAttribute('slot'));\n      this.listChildren = [];\n    }\n  }\n\n  // Check if the trigger is valid and attach click event listeners.\n  private _configureTrigger(): void {\n    if (this.trigger === this._triggerElement) {\n      return;\n    }\n\n    this._triggerAbortController?.abort();\n    removeAriaOverlayTriggerAttributes(this._triggerElement);\n    this._triggerElement = this.trigger;\n\n    if (!this._triggerElement) {\n      return;\n    }\n\n    setAriaOverlayTriggerAttributes(this._triggerElement, 'menu', this.id, this.state);\n    this._triggerAbortController = new AbortController();\n    this._triggerElement.addEventListener('click', () => this.open(), {\n      signal: this._triggerAbortController.signal,\n    });\n  }\n\n  private _attachWindowEvents(): void {\n    this._windowEventsController = new AbortController();\n    document.addEventListener('scroll', () => this._setMenuPosition(), {\n      passive: true,\n      signal: this._windowEventsController.signal,\n      // Without capture, other scroll contexts would not bubble to this event listener.\n      // Capture allows us to react to all scroll contexts in this DOM.\n      capture: true,\n    });\n    window.addEventListener('resize', () => this._setMenuPosition(), {\n      passive: true,\n      signal: this._windowEventsController.signal,\n    });\n    // Close menu on backdrop click\n    window.addEventListener('pointerdown', this._pointerDownListener, {\n      signal: this._windowEventsController.signal,\n    });\n    window.addEventListener('pointerup', this._closeOnBackdropClick, {\n      signal: this._windowEventsController.signal,\n    });\n  }\n\n  // Close menu at any click on an interactive element inside the <sbb-menu> that bubbles to the container.\n  private _closeOnInteractiveElementClick(event: Event): void {\n    const target = event.target as HTMLElement;\n    if (INTERACTIVE_ELEMENTS.includes(target.nodeName) && !target.hasAttribute('disabled')) {\n      this.close();\n    }\n  }\n\n  // Check if the pointerdown event target is triggered on the menu.\n  private _pointerDownListener = (event: PointerEvent): void => {\n    this._isPointerDownEventOnMenu = isEventOnElement(this._menu, event);\n  };\n\n  // Close menu on backdrop click.\n  private _closeOnBackdropClick = (event: PointerEvent): void => {\n    if (!this._isPointerDownEventOnMenu && !isEventOnElement(this._menu, event)) {\n      this.close();\n    }\n  };\n\n  // Set menu position (x, y) to '0' once the menu is closed and the transition ended to prevent the\n  // viewport from overflowing. And set the focus to the first focusable element once the menu is open.\n  // In rare cases it can be that the animationEnd event is triggered twice.\n  // To avoid entering a corrupt state, exit when state is not expected.\n  private _onMenuAnimationEnd(event: AnimationEvent): void {\n    if (event.animationName === 'open' && this.state === 'opening') {\n      this._handleOpening();\n    } else if (event.animationName === 'close' && this.state === 'closing') {\n      this._handleClosing();\n    }\n  }\n\n  // Set menu position and max height if the breakpoint is medium-ultra.\n  private _setMenuPosition(): void {\n    // Starting from breakpoint medium\n    if (\n      (this._mediaMatcher.matches(SbbMediaQueryBreakpointSmallAndBelow) ?? true) ||\n      !this._menu ||\n      !this._triggerElement ||\n      this.state === 'closing'\n    ) {\n      return;\n    }\n\n    const menuPosition = getElementPosition(\n      this.shadowRoot!.querySelector('.sbb-menu__content')!,\n      this._triggerElement,\n      this.shadowRoot!.querySelector('.sbb-menu__container')!,\n      {\n        verticalOffset: MENU_OFFSET,\n      },\n    );\n\n    this.style.setProperty('--sbb-menu-position-x', `${menuPosition.left}px`);\n    this.style.setProperty('--sbb-menu-position-y', `${menuPosition.top}px`);\n    this.style.setProperty('--sbb-menu-max-height', menuPosition.maxHeight);\n  }\n\n  private _syncNegative(): void {\n    // Links and buttons are the most expected contents which have a negative property\n    this.querySelectorAll('[data-sbb-link], [data-sbb-button]')?.forEach((el: Element) => {\n      customElements.upgrade(el);\n      (el as Element & SbbNegativeMixinType).negative = !this._darkModeController.matches();\n    });\n  }\n\n  protected override render(): TemplateResult {\n    // TODO: Handle case with other elements than sbb-menu-button/sbb-menu-link.\n    return html`\n      <div class=\"sbb-menu__container\">\n        <div\n          @animationend=${this._onMenuAnimationEnd}\n          class=\"sbb-menu\"\n          ${ref((el?: Element) => (this._menu = el as HTMLDivElement))}\n        >\n          <div\n            @click=${(event: Event) => this._closeOnInteractiveElementClick(event)}\n            @scroll=${(e: Event) => forwardEvent(e, document)}\n            class=\"sbb-menu__content\"\n          >\n            ${this.listChildren.length\n              ? this.renderList({ class: 'sbb-menu-list', ariaLabel: this.listAccessibilityLabel })\n              : html`<slot></slot>`}\n          </div>\n        </div>\n      </div>\n    `;\n  }\n}\n\ndeclare global {\n  interface HTMLElementTagNameMap {\n    // eslint-disable-next-line @typescript-eslint/naming-convention\n    'sbb-menu': SbbMenuElement;\n  }\n}\n"],"names":[],"mappings":"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"}
|
|
731
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"menu.component.js","sources":["../../../../../src/elements/menu/menu/menu.component.ts"],"sourcesContent":["import {\n  type CSSResultGroup,\n  html,\n  isServer,\n  type PropertyDeclaration,\n  type PropertyValues,\n  type TemplateResult,\n} from 'lit';\nimport { customElement, property } from 'lit/decorators.js';\nimport { ref } from 'lit/directives/ref.js';\n\nimport {\n  getNextElementIndex,\n  interactivityChecker,\n  isArrowKeyOrPageKeysPressed,\n  SbbFocusTrapController,\n} from '../../core/a11y.js';\nimport { SbbOpenCloseBaseElement } from '../../core/base-elements.js';\nimport {\n  SbbDarkModeController,\n  SbbEscapableOverlayController,\n  SbbInertController,\n  SbbLanguageController,\n  SbbMediaMatcherController,\n  SbbMediaQueryBreakpointSmallAndBelow,\n} from '../../core/controllers.js';\nimport { forceType, idReference } from '../../core/decorators.js';\nimport { isZeroAnimationDuration, SbbScrollHandler } from '../../core/dom.js';\nimport { forwardEvent } from '../../core/eventing.js';\nimport { i18nGoBack } from '../../core/i18n/i18n.js';\nimport { SbbNamedSlotListMixin, type SbbNegativeMixinType } from '../../core/mixins.js';\nimport {\n  getElementPosition,\n  getElementPositionHorizontal,\n  isEventOnElement,\n  removeAriaOverlayTriggerAttributes,\n  setAriaOverlayTriggerAttributes,\n} from '../../core/overlay.js';\nimport type { SbbMenuButtonElement } from '../menu-button.js';\nimport type { SbbMenuLinkElement } from '../menu-link/menu-link.component.js';\n\nimport style from './menu.scss?lit&inline';\n\nimport '../../divider.js';\nimport '../menu-button.js';\n\nconst MENU_OFFSET = 8;\nconst NESTED_MENU_OFFSET = -4;\nconst INTERACTIVE_ELEMENTS = [\n  'A',\n  'BUTTON',\n  'SBB-BUTTON',\n  'SBB-BUTTON-LINK',\n  'SBB-LINK',\n  'SBB-BLOCK-LINK',\n  'SBB-LINK-BUTTON',\n  'SBB-BLOCK-LINK-BUTTON',\n  'SBB-MENU-BUTTON',\n  'SBB-MENU-LINK',\n];\n\nlet nextId = 0;\n\n/**\n * It displays a contextual menu with one or more action element.\n *\n * @slot - Use the unnamed slot to add `sbb-menu-button`/`sbb-menu-link` or other elements to the menu.\n * @cssprop [--sbb-menu-z-index=var(--sbb-overlay-default-z-index)] - To specify a custom stack order,\n * the `z-index` can be overridden by defining this CSS variable. The default `z-index` of the\n * component is set to `var(--sbb-overlay-default-z-index)` with a value of `1000`.\n */\nexport\n@customElement('sbb-menu')\nclass SbbMenuElement extends SbbNamedSlotListMixin<\n  SbbMenuButtonElement | SbbMenuLinkElement,\n  typeof SbbOpenCloseBaseElement\n>(SbbOpenCloseBaseElement) {\n  public static override styles: CSSResultGroup = style;\n  public static override readonly role = 'menu';\n  protected override readonly listChildLocalNames = ['sbb-menu-button', 'sbb-menu-link'];\n\n  /**\n   * The element that will trigger the menu overlay.\n   *\n   * For attribute usage, provide an id reference.\n   */\n  @idReference()\n  @property()\n  public accessor trigger: HTMLElement | null = null;\n\n  /**\n   * This will be forwarded as aria-label to the inner list.\n   * Used only if the menu automatically renders the actions inside as a list.\n   */\n  @forceType()\n  @property({ attribute: 'list-accessibility-label' })\n  public accessor listAccessibilityLabel: string = '';\n\n  private _menu!: HTMLDivElement;\n  private _triggerElement: HTMLElement | null = null;\n  private _triggerAbortController!: AbortController;\n  private _isPointerDownEventOnMenu: boolean = false;\n  private _windowEventsController!: AbortController;\n  private _escapableOverlayController = new SbbEscapableOverlayController(this);\n  private _focusTrapController = new SbbFocusTrapController(this);\n  private _scrollHandler = new SbbScrollHandler();\n  private _inertController = new SbbInertController(this);\n  private _mobileBreakpoint = SbbMediaQueryBreakpointSmallAndBelow;\n  private _mediaMatcher = new SbbMediaMatcherController(this, {\n    [this._mobileBreakpoint]: (matches) => {\n      if (matches && (this.state === 'opening' || this.state === 'opened')) {\n        this._scrollHandler.disableScroll();\n      } else {\n        this._scrollHandler.enableScroll();\n      }\n    },\n  });\n  private _darkModeController = new SbbDarkModeController(this, () => this._syncNegative());\n  private _language = new SbbLanguageController(this);\n  private _nestedMenu: SbbMenuElement | null = null;\n\n  public constructor() {\n    super();\n    this.addEventListener?.('keydown', (e) => this._handleKeyDown(e));\n  }\n\n  protected override firstUpdated(changedProperties: PropertyValues<this>): void {\n    super.firstUpdated(changedProperties);\n\n    this._configureTrigger();\n  }\n\n  public override escapeStrategy(): void {\n    this.closeAll();\n  }\n\n  /**\n   * Opens the menu on trigger click.\n   */\n  public open(): void {\n    if (\n      this.state === 'closing' ||\n      this.state === 'opened' ||\n      !this._menu ||\n      !this.dispatchBeforeOpenEvent()\n    ) {\n      return;\n    }\n\n    if (this._isNested()) {\n      const parentMenu = this._parentMenu()!;\n      parentMenu.toggleState('nested-child', true);\n\n      // In case we change between arrow key navigation and mouse navigation, it can be that another\n      // nested parent menu is still open. We have to close it.\n      if (parentMenu._nestedMenu !== this) {\n        parentMenu._nestedMenu?.close();\n      }\n      parentMenu._nestedMenu = this;\n    }\n\n    this.showPopover?.();\n    this.state = 'opening';\n    this._setMenuPosition();\n    this._triggerElement?.setAttribute('aria-expanded', 'true');\n\n    // From zero to medium, disable scroll\n    if (this._isMobile()) {\n      this._scrollHandler.disableScroll();\n    }\n\n    // If the animation duration is zero, the animationend event is not always fired reliably.\n    // In this case we directly set the `opened` state.\n    if (this._isZeroAnimationDuration()) {\n      this._handleOpening();\n    }\n  }\n\n  /** Closes the menu and all its nested menus. */\n  public close(): void {\n    this._close();\n  }\n\n  /** Closes the menu and all related menus  nested and parent menus). */\n  public closeAll(): void {\n    this._mainMenu()._close(true);\n  }\n\n  /** @param [closeAll='false'] - If true, it ensures animations are correct by toggling some states when closing all related menus at once. */\n  private _close(closeAll = false): void {\n    if ((this.state === 'opening' && !this._isNested()) || !this.dispatchBeforeCloseEvent()) {\n      return;\n    }\n\n    // Close nested menus first\n    this._nestedMenu?._close(closeAll);\n\n    if (this._isNested()) {\n      const parentMenu = this._parentMenu()!;\n      this.toggleState('close-all', closeAll);\n      parentMenu.toggleState('skip-animation', closeAll);\n      parentMenu.toggleState('nested-child', false);\n      parentMenu._nestedMenu = null;\n    }\n\n    this.state = 'closing';\n    this._triggerElement?.setAttribute('aria-expanded', 'false');\n\n    // If the animation duration is zero, the animationend event is not always fired reliably.\n    // In this case we directly set the `closed` state.\n    if (this._isZeroAnimationDuration()) {\n      this._handleClosing();\n    }\n  }\n\n  private _isZeroAnimationDuration(): boolean {\n    return isZeroAnimationDuration(this, '--sbb-menu-animation-duration');\n  }\n\n  private _handleOpening(): void {\n    this.state = 'opened';\n\n    if (!this._isNested()) {\n      this._inertController.activate();\n    } else {\n      this._updateNestedInert();\n    }\n    this._escapableOverlayController.connect();\n    this._focusTrapController.focusInitialElement();\n    this._focusTrapController.enabled = true;\n    this._attachWindowEvents();\n    this.dispatchOpenEvent();\n  }\n\n  private _handleClosing(): void {\n    this.state = 'closed';\n    this.toggleState('skip-animation', false);\n    this.toggleState('close-all', false);\n    this.hidePopover?.();\n\n    this._menu?.firstElementChild?.scrollTo(0, 0);\n    if (!this._isNested()) {\n      this._inertController.deactivate();\n    } else {\n      this._updateNestedInert();\n    }\n    // Manually focus last focused element\n    this._triggerElement?.focus({\n      // When inside the sbb-header, we prevent the scroll to avoid the snapping to the top of the page\n      preventScroll: ['sbb-header-button', 'sbb-header-link'].includes(\n        this._triggerElement.localName,\n      ),\n    });\n    this._escapableOverlayController.disconnect();\n    this.dispatchCloseEvent();\n    this._windowEventsController?.abort();\n    this._focusTrapController.enabled = false;\n\n    // Starting from breakpoint medium, enable scroll\n    this._scrollHandler.enableScroll();\n  }\n\n  private _handleKeyDown(evt: KeyboardEvent): void {\n    if (!isArrowKeyOrPageKeysPressed(evt)) {\n      return;\n    }\n    evt.preventDefault();\n\n    const enabledActions: Element[] = Array.from(\n      this.querySelectorAll<SbbMenuButtonElement | SbbMenuLinkElement>(\n        'sbb-menu-button, sbb-menu-link',\n      ),\n    )\n      .concat(this.shadowRoot!.querySelector('sbb-menu-button')!)\n      .filter(\n        (el) => (!el.disabled || el.disabledInteractive) && interactivityChecker.isVisible(el),\n      );\n    const current = enabledActions.findIndex((e: Element) => e === evt.target);\n\n    let nextIndex;\n    switch (evt.key) {\n      case 'ArrowUp':\n      case 'ArrowDown':\n        nextIndex = getNextElementIndex(evt, current, enabledActions.length);\n        break;\n\n      case 'ArrowLeft':\n        if (this._isNested()) {\n          this.close();\n        }\n        break;\n\n      case 'ArrowRight':\n        if ((evt.target as HTMLElement).hasAttribute('data-sbb-menu-trigger')) {\n          (evt.target as HTMLElement).click();\n        }\n        break;\n\n      case 'PageUp':\n      case 'Home':\n        nextIndex = 0;\n        break;\n\n      case 'End':\n      case 'PageDown':\n        nextIndex = enabledActions.length - 1;\n        break;\n    }\n\n    if (nextIndex !== undefined) {\n      (enabledActions[nextIndex] as HTMLElement).focus();\n    }\n  }\n\n  // Removes trigger click listener on trigger change.\n  protected override createRenderRoot(): HTMLElement | DocumentFragment {\n    const renderRoot = super.createRenderRoot();\n    // Due to the fact that menu can both be a list and just a container, we need to check its\n    // state before the SbbNamedSlotListMixin handles the slotchange event, in order to avoid\n    // it interpreting the non list case as a list.\n    this.shadowRoot?.addEventListener(\n      'slotchange',\n      (e) => {\n        this._syncNegative();\n        this._checkListCase(e);\n      },\n      {\n        capture: true,\n      },\n    );\n    return renderRoot;\n  }\n\n  public override connectedCallback(): void {\n    this.popover = 'manual';\n    super.connectedCallback();\n    this.id ||= `sbb-menu-${nextId++}`;\n    if (this.hasUpdated) {\n      this._configureTrigger();\n    }\n  }\n\n  public override disconnectedCallback(): void {\n    super.disconnectedCallback();\n    this._triggerElement = null;\n    this._triggerAbortController?.abort();\n    this._windowEventsController?.abort();\n    this._scrollHandler.enableScroll();\n  }\n\n  public override requestUpdate(\n    name?: PropertyKey,\n    oldValue?: unknown,\n    options?: PropertyDeclaration,\n  ): void {\n    super.requestUpdate(name, oldValue, options);\n\n    if (!isServer && (!name || name === 'trigger') && this.hasUpdated) {\n      this._configureTrigger();\n    }\n  }\n\n  private _checkListCase(event: Event): void {\n    // If all children are sbb-menu-button/menu-link instances, we render them as a list.\n    if (\n      this.children?.length &&\n      Array.from(this.children ?? []).every(\n        (c) => c.localName === 'sbb-menu-button' || c.localName === 'sbb-menu-link',\n      )\n    ) {\n      return;\n    }\n\n    event.stopImmediatePropagation();\n    if (this.listChildren.length) {\n      this.listChildren.forEach((c) => c.removeAttribute('slot'));\n      this.listChildren = [];\n    }\n  }\n\n  // Check if the trigger is valid and attach click event listeners.\n  private _configureTrigger(): void {\n    if (this.trigger === this._triggerElement) {\n      return;\n    }\n\n    this._triggerAbortController?.abort();\n    removeAriaOverlayTriggerAttributes(this._triggerElement);\n    this._triggerElement = this.trigger;\n\n    if (!this._triggerElement) {\n      return;\n    }\n\n    setAriaOverlayTriggerAttributes(this._triggerElement, 'menu', this.id, this.state);\n    this._triggerAbortController = new AbortController();\n    this._triggerElement.addEventListener('click', () => this.open(), {\n      signal: this._triggerAbortController.signal,\n    });\n\n    // Consider the menu as nested if the trigger is a menu button or menu link.\n    this.toggleState(\n      'nested',\n      ['sbb-menu-button', 'sbb-menu-link'].includes(this._triggerElement.localName),\n    );\n    this._triggerElement.toggleAttribute('data-sbb-menu-trigger', true);\n  }\n\n  private _attachWindowEvents(): void {\n    this._windowEventsController = new AbortController();\n    document.addEventListener('scroll', () => this._setMenuPosition(), {\n      passive: true,\n      signal: this._windowEventsController.signal,\n      // Without capture, other scroll contexts would not bubble to this event listener.\n      // Capture allows us to react to all scroll contexts in this DOM.\n      capture: true,\n    });\n    window.addEventListener('resize', () => this._setMenuPosition(), {\n      passive: true,\n      signal: this._windowEventsController.signal,\n    });\n\n    // Only the outermost menu needs to listen to the backdrop clicks\n    if (!this._isNested()) {\n      // Close menu on backdrop click\n      window.addEventListener('pointerdown', this._pointerDownListener, {\n        signal: this._windowEventsController.signal,\n      });\n      window.addEventListener('pointerup', this._closeOnBackdropClick, {\n        signal: this._windowEventsController.signal,\n      });\n    }\n  }\n\n  // Close menu at any click on an interactive element inside the <sbb-menu> that bubbles to the container.\n  private _interactiveElementClick(event: Event): void {\n    const target = event.target as HTMLElement;\n\n    if (\n      INTERACTIVE_ELEMENTS.includes(target.nodeName) &&\n      !target.hasAttribute('disabled') &&\n      !target.hasAttribute('data-sbb-menu-trigger') &&\n      target.id !== 'sbb-menu__back-button'\n    ) {\n      this.closeAll();\n    }\n  }\n\n  // Check if the pointerdown event target is triggered on the menu.\n  private _pointerDownListener = (event: PointerEvent): void => {\n    const menu = (event.target as HTMLElement).closest('sbb-menu');\n\n    // The pointer down is on the menu or one of its nested menus.\n    this._isPointerDownEventOnMenu =\n      isEventOnElement(this._menu, event) || this._nestedMenus().some((el) => menu === el);\n  };\n\n  // Close menu on backdrop click.\n  private _closeOnBackdropClick = (event: PointerEvent): void => {\n    const target = event.target as HTMLElement;\n\n    // The backdrop is only listened on the main menu (outermost menu).\n    // To close.\n    // - The pointer down and up need to be outside the menu.\n    // - The target should be not on a nested menu\n    if (\n      !this._isPointerDownEventOnMenu &&\n      !isEventOnElement(this._menu, event) &&\n      !this._nestedMenus().some((el) => el === target)\n    ) {\n      this.closeAll();\n    }\n  };\n\n  /** Converts the linked list into an array of SbbMenuElement. */\n  private _nestedMenus(): SbbMenuElement[] {\n    const menus: SbbMenuElement[] = [];\n    let current = this._nestedMenu;\n\n    while (current) {\n      menus.push(current);\n      current = current._nestedMenu;\n    }\n\n    return menus;\n  }\n\n  private _parentMenu(): SbbMenuElement | null {\n    return this._triggerElement?.closest('sbb-menu') ?? null;\n  }\n\n  /** The outermost menu. */\n  private _mainMenu(): SbbMenuElement {\n    return this._isNested() ? (this._parentMenu()?._mainMenu() ?? this) : this;\n  }\n\n  private _isNested(): boolean {\n    return !!this._parentMenu();\n  }\n\n  private _updateNestedInert(): void {\n    this._inertController.restoreAllExempted();\n    this._mainMenu()\n      ._nestedMenus()\n      .forEach((menu) => this._inertController.exempt(menu));\n  }\n\n  // Check if nested menu should be closed.\n  private _handleMouseOver(event: MouseEvent): void {\n    const element = event.target as HTMLElement;\n    const isMobile = this._isMobile();\n\n    // All nested menus should close in desktop mode if the cursor landed on\n    // anything other than the container, the container's scrollbar or the trigger itself\n    if (\n      !isMobile &&\n      this._nestedMenu &&\n      !element.classList.contains('sbb-menu__content') &&\n      !(element.getAttribute('aria-expanded') === 'true')\n    ) {\n      this._nestedMenu.close();\n    }\n\n    if (element.hasAttribute('data-sbb-menu-trigger') && !isMobile) {\n      element.click();\n    }\n  }\n\n  // Set menu position (x, y) to '0' once the menu is closed and the transition ended to prevent the\n  // viewport from overflowing. And set the focus to the first focusable element once the menu is open.\n  // In rare cases it can be that the animationEnd event is triggered twice.\n  // To avoid entering a corrupt state, exit when state is not expected.\n  private _onMenuAnimationEnd(event: AnimationEvent): void {\n    if (\n      (event.animationName === 'open' || event.animationName === 'open-sideways') &&\n      this.state === 'opening'\n    ) {\n      this._handleOpening();\n    } else if (\n      (event.animationName === 'close' || event.animationName === 'close-sideways') &&\n      this.state === 'closing'\n    ) {\n      this._handleClosing();\n    }\n  }\n\n  // Set menu position and max height if the breakpoint is medium-ultra.\n  private _setMenuPosition(): void {\n    // Starting from breakpoint medium\n    if (this._isMobile() || !this._menu || !this._triggerElement || this.state === 'closing') {\n      return;\n    }\n\n    const menuPosition = !this._isNested()\n      ? getElementPosition(\n          this.shadowRoot!.querySelector('.sbb-menu__content')!,\n          this._triggerElement,\n          this.shadowRoot!.querySelector('.sbb-menu__container')!,\n          {\n            verticalOffset: MENU_OFFSET,\n          },\n        )\n      : getElementPositionHorizontal(\n          this.shadowRoot!.querySelector('.sbb-menu__content')!,\n          this._triggerElement,\n          this.shadowRoot!.querySelector('.sbb-menu__container')!,\n          {\n            horizontalOffset: MENU_OFFSET,\n            verticalOffset: NESTED_MENU_OFFSET,\n            contentSelector: '.sbb-menu__content',\n          },\n        );\n\n    this.style.setProperty('--sbb-menu-position-x', `${menuPosition.left}px`);\n    this.style.setProperty('--sbb-menu-position-y', `${menuPosition.top}px`);\n    this.style.setProperty('--sbb-menu-max-height', menuPosition.maxHeight);\n  }\n\n  private _syncNegative(): void {\n    // Links and buttons are the most expected contents which have a negative property\n    this.querySelectorAll('[data-sbb-link], [data-sbb-button]')?.forEach((el: Element) => {\n      customElements.upgrade(el);\n      (el as Element & SbbNegativeMixinType).negative = !this._darkModeController.matches();\n    });\n  }\n\n  private _isMobile(): boolean {\n    return this._mediaMatcher.matches(this._mobileBreakpoint) ?? true;\n  }\n\n  protected override render(): TemplateResult {\n    // TODO: Handle case with other elements than sbb-menu-button/sbb-menu-link.\n    return html`\n      <div class=\"sbb-menu__container\">\n        <div\n          @animationend=${this._onMenuAnimationEnd}\n          @mouseover=${(e: MouseEvent) => this._handleMouseOver(e)}\n          class=\"sbb-menu\"\n          ${ref((el?: Element) => (this._menu = el as HTMLDivElement))}\n        >\n          <div\n            @click=${(event: Event) => this._interactiveElementClick(event)}\n            @scroll=${(e: Event) => forwardEvent(e, document)}\n            class=\"sbb-menu__content\"\n          >\n            ${this.listChildren.length\n              ? this.renderList({ class: 'sbb-menu-list', ariaLabel: this.listAccessibilityLabel })\n              : html`<slot></slot>`}\n            <sbb-divider></sbb-divider>\n            <sbb-menu-button\n              id=\"sbb-menu__back-button\"\n              @click=${() => this.close()}\n              icon-name=\"chevron-small-left-small\"\n            >\n              ${i18nGoBack[this._language.current]}\n            </sbb-menu-button>\n          </div>\n        </div>\n      </div>\n    `;\n  }\n}\n\ndeclare global {\n  interface HTMLElementTagNameMap {\n    // eslint-disable-next-line @typescript-eslint/naming-convention\n    'sbb-menu': SbbMenuElement;\n  }\n}\n"],"names":[],"mappings":"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"}
|