@sbb-esta/lyne-elements-dev 4.7.0-dev.1773755649 → 4.7.0-dev.1773817331

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.
@@ -1,2 +1,2 @@
1
- import { t as e } from "../../breadcrumb-group.component-BA8KK3Om.js";
1
+ import { t as e } from "../../breadcrumb-group.component-CfUyO1sI.js";
2
2
  export { e as SbbBreadcrumbGroupElement };
@@ -1,4 +1,4 @@
1
- import { t as e } from "../breadcrumb-group.component-BA8KK3Om.js";
1
+ import { t as e } from "../breadcrumb-group.component-CfUyO1sI.js";
2
2
  e.define(), console.warn("The entrypoint '@sbb-esta/elements/breadcrumb/breadcrumb-group.js' has been deprecated.\nUse either '@sbb-esta/elements/breadcrumb.js' or '@sbb-esta/elements/breadcrumb.pure.js' instead.");
3
3
  //#endregion
4
4
  export { e as SbbBreadcrumbGroupElement };
@@ -10,7 +10,7 @@ import "./icon.js";
10
10
  import { ResizeController as d } from "@lit-labs/observers/resize-controller.js";
11
11
  import { getNextElementIndex as f, isArrowKeyPressed as p, sbbInputModalityDetector as m } from "./core/a11y.js";
12
12
  //#region src/elements/breadcrumb/breadcrumb-group/breadcrumb-group.scss?lit&inline
13
- var h = n`:host{display:block}:host(:is(:state(loaded),[state--loaded])){--sbb-breadcrumb-group-visibility:visible}:host(:is(:is(:state(state-collapsed),[state--state-collapsed]),:is(:state(state-manually-expanded),[state--state-manually-expanded]))){--sbb-breadcrumb-group-wrap:wrap}.sbb-breadcrumb-group{font-size:inherit;flex-wrap:var(--sbb-breadcrumb-group-wrap);column-gap:var(--sbb-spacing-fixed-1x);visibility:var(--sbb-breadcrumb-group-visibility);margin:0;padding:0;list-style:none;display:flex}.sbb-breadcrumb-group__item{column-gap:var(--sbb-spacing-fixed-1x);flex:none;display:flex}.sbb-breadcrumb-group__divider-icon{color:var(--sbb-breadcrumb-group-ellipsis-color)}#sbb-breadcrumb-ellipsis{appearance:none;box-sizing:border-box;-webkit-tap-highlight-color:transparent;color:var(--sbb-breadcrumb-group-ellipsis-color);--sbb-text-font-size:var(--sbb-text-font-size-xxs);font-family:var(--sbb-typo-font-family);font-weight:400;line-height:var(--sbb-typo-line-height-text);letter-spacing:var(--sbb-typo-letter-spacing-text);font-size:var(--sbb-text-font-size);letter-spacing:.01em;line-height:0;font-size:var(--sbb-text-font-size-xxs);width:var(--sbb-size-icon-ui-small);height:var(--sbb-size-icon-ui-small);border:none;border:var(--sbb-breadcrumb-group-ellipsis-border-width) solid var(--sbb-breadcrumb-group-ellipsis-border-color);padding:0;cursor:var(--sbb-cursor-pointer);background:0 0;background-color:var(--sbb-breadcrumb-group-ellipsis-background-color);border-radius:50%;outline:none;margin:0;padding-block-end:.5em;overflow:hidden}@media (forced-colors:active){#sbb-breadcrumb-ellipsis{--sbb-breadcrumb-group-ellipsis-border-width:var(--sbb-border-width-2x);--sbb-breadcrumb-group-ellipsis-border-color:CanvasText}}@media (any-hover:hover){#sbb-breadcrumb-ellipsis:hover{--sbb-breadcrumb-group-ellipsis-color:var(--sbb-color-2);--sbb-breadcrumb-group-ellipsis-background-color:var(--sbb-background-color-3)}}@media (any-hover:hover) and (forced-colors:active){#sbb-breadcrumb-ellipsis:hover{--sbb-breadcrumb-group-ellipsis-border-color:Highlight}}#sbb-breadcrumb-ellipsis:focus-visible{outline-offset:var(--sbb-focus-outline-offset);outline:var(--sbb-focus-outline-color) var(--sbb-focus-outline-style,solid) var(--sbb-focus-outline-width)}`, g = 3, _ = (() => {
13
+ var h = n`:host{min-width:0;display:block}:host(:is(:state(loaded),[state--loaded])){--sbb-breadcrumb-group-visibility:visible}:host(:is(:is(:state(state-collapsed),[state--state-collapsed]),:is(:state(state-manually-expanded),[state--state-manually-expanded]))){--sbb-breadcrumb-group-wrap:wrap}.sbb-breadcrumb-group{font-size:inherit;flex-wrap:var(--sbb-breadcrumb-group-wrap);column-gap:var(--sbb-spacing-fixed-1x);visibility:var(--sbb-breadcrumb-group-visibility);margin:0;padding:0;list-style:none;display:flex}.sbb-breadcrumb-group__item{column-gap:var(--sbb-spacing-fixed-1x);flex:none;display:flex}.sbb-breadcrumb-group__divider-icon{color:var(--sbb-breadcrumb-group-ellipsis-color)}#sbb-breadcrumb-ellipsis{appearance:none;box-sizing:border-box;-webkit-tap-highlight-color:transparent;color:var(--sbb-breadcrumb-group-ellipsis-color);--sbb-text-font-size:var(--sbb-text-font-size-xxs);font-family:var(--sbb-typo-font-family);font-weight:400;line-height:var(--sbb-typo-line-height-text);letter-spacing:var(--sbb-typo-letter-spacing-text);font-size:var(--sbb-text-font-size);letter-spacing:.01em;line-height:0;font-size:var(--sbb-text-font-size-xxs);width:var(--sbb-size-icon-ui-small);height:var(--sbb-size-icon-ui-small);border:none;border:var(--sbb-breadcrumb-group-ellipsis-border-width) solid var(--sbb-breadcrumb-group-ellipsis-border-color);padding:0;cursor:var(--sbb-cursor-pointer);background:0 0;background-color:var(--sbb-breadcrumb-group-ellipsis-background-color);border-radius:50%;outline:none;margin:0;padding-block-end:.5em;overflow:hidden}@media (forced-colors:active){#sbb-breadcrumb-ellipsis{--sbb-breadcrumb-group-ellipsis-border-width:var(--sbb-border-width-2x);--sbb-breadcrumb-group-ellipsis-border-color:CanvasText}}@media (any-hover:hover){#sbb-breadcrumb-ellipsis:hover{--sbb-breadcrumb-group-ellipsis-color:var(--sbb-color-2);--sbb-breadcrumb-group-ellipsis-background-color:var(--sbb-background-color-3)}}@media (any-hover:hover) and (forced-colors:active){#sbb-breadcrumb-ellipsis:hover{--sbb-breadcrumb-group-ellipsis-border-color:Highlight}}#sbb-breadcrumb-ellipsis:focus-visible{outline-offset:var(--sbb-focus-outline-offset);outline:var(--sbb-focus-outline-color) var(--sbb-focus-outline-style,solid) var(--sbb-focus-outline-width)}`, g = 3, _ = (() => {
14
14
  let n = s(o), _ = [], v;
15
15
  return class extends n {
16
16
  static {
package/breadcrumb.js CHANGED
@@ -1,4 +1,4 @@
1
- import { t as e } from "./breadcrumb-group.component-BA8KK3Om.js";
1
+ import { t as e } from "./breadcrumb-group.component-CfUyO1sI.js";
2
2
  import { t } from "./breadcrumb.component-DjkZzGJD.js";
3
3
  import "./breadcrumb.pure.js";
4
4
  t.define(), e.define();
@@ -1,3 +1,3 @@
1
- import { t as e } from "./breadcrumb-group.component-BA8KK3Om.js";
1
+ import { t as e } from "./breadcrumb-group.component-CfUyO1sI.js";
2
2
  import { t } from "./breadcrumb.component-DjkZzGJD.js";
3
3
  export { t as SbbBreadcrumbElement, e as SbbBreadcrumbGroupElement };
@@ -1,2 +1,2 @@
1
- import { t as SbbBreadcrumbGroupElement } from "../../breadcrumb-group.component-DjsoMTPI.js";
1
+ import { t as SbbBreadcrumbGroupElement } from "../../breadcrumb-group.component-zU06d9ni.js";
2
2
  export { SbbBreadcrumbGroupElement };
@@ -1,4 +1,4 @@
1
- import { t as SbbBreadcrumbGroupElement } from "../breadcrumb-group.component-DjsoMTPI.js";
1
+ import { t as SbbBreadcrumbGroupElement } from "../breadcrumb-group.component-zU06d9ni.js";
2
2
  //#region src/elements/breadcrumb/breadcrumb-group.ts
3
3
  /** @entrypoint */
4
4
  SbbBreadcrumbGroupElement.define();
@@ -0,0 +1,279 @@
1
+ import { __esDecorate, __runInitializers } from "tslib";
2
+ import { css, html, nothing } from "lit";
3
+ import { state } from "lit/decorators.js";
4
+ import { SbbElement } from "./core/base-elements.js";
5
+ import { SbbNamedSlotListMixin } from "./core/mixins.js";
6
+ import { boxSizingStyles } from "./core/styles.js";
7
+ import { SbbLanguageController } from "./core/controllers.js";
8
+ import { i18nBreadcrumbEllipsisButtonLabel } from "./core/i18n.js";
9
+ import "./icon.js";
10
+ import { ResizeController } from "@lit-labs/observers/resize-controller.js";
11
+ import { getNextElementIndex, isArrowKeyPressed, sbbInputModalityDetector } from "./core/a11y.js";
12
+ //#region src/elements/breadcrumb/breadcrumb-group/breadcrumb-group.scss?lit&inline
13
+ var breadcrumb_group_default = css`:host {
14
+ display: block;
15
+ min-width: 0;
16
+ }
17
+
18
+ :host(:is(:state(loaded),[state--loaded])) {
19
+ --sbb-breadcrumb-group-visibility: visible;
20
+ }
21
+
22
+ :host(:is(:is(:state(state-collapsed),[state--state-collapsed]), :is(:state(state-manually-expanded),[state--state-manually-expanded]))) {
23
+ --sbb-breadcrumb-group-wrap: wrap;
24
+ }
25
+
26
+ .sbb-breadcrumb-group {
27
+ list-style: none;
28
+ margin: 0;
29
+ padding: 0;
30
+ font-size: inherit;
31
+ display: flex;
32
+ flex-wrap: var(--sbb-breadcrumb-group-wrap);
33
+ column-gap: var(--sbb-spacing-fixed-1x);
34
+ visibility: var(--sbb-breadcrumb-group-visibility);
35
+ }
36
+
37
+ .sbb-breadcrumb-group__item {
38
+ flex: 0 0 auto;
39
+ display: flex;
40
+ column-gap: var(--sbb-spacing-fixed-1x);
41
+ }
42
+
43
+ .sbb-breadcrumb-group__divider-icon {
44
+ color: var(--sbb-breadcrumb-group-ellipsis-color);
45
+ }
46
+
47
+ #sbb-breadcrumb-ellipsis {
48
+ appearance: none;
49
+ box-sizing: border-box;
50
+ margin: 0;
51
+ outline: none;
52
+ border: none;
53
+ border-radius: 0;
54
+ padding: 0;
55
+ background: none;
56
+ -webkit-tap-highlight-color: transparent;
57
+ color: inherit;
58
+ --sbb-text-font-size: var(--sbb-text-font-size-xxs);
59
+ font-family: var(--sbb-typo-font-family);
60
+ font-weight: normal;
61
+ line-height: var(--sbb-typo-line-height-text);
62
+ letter-spacing: var(--sbb-typo-letter-spacing-text);
63
+ font-size: var(--sbb-text-font-size);
64
+ line-height: 0;
65
+ letter-spacing: 0.01em;
66
+ font-size: var(--sbb-text-font-size-xxs);
67
+ width: var(--sbb-size-icon-ui-small);
68
+ height: var(--sbb-size-icon-ui-small);
69
+ border: var(--sbb-breadcrumb-group-ellipsis-border-width) solid var(--sbb-breadcrumb-group-ellipsis-border-color);
70
+ border-radius: 50%;
71
+ padding-block-end: 0.5em;
72
+ cursor: var(--sbb-cursor-pointer);
73
+ color: var(--sbb-breadcrumb-group-ellipsis-color);
74
+ background-color: var(--sbb-breadcrumb-group-ellipsis-background-color);
75
+ overflow: hidden;
76
+ }
77
+ @media (forced-colors: active) {
78
+ #sbb-breadcrumb-ellipsis {
79
+ --sbb-breadcrumb-group-ellipsis-border-width: var(--sbb-border-width-2x);
80
+ --sbb-breadcrumb-group-ellipsis-border-color: CanvasText;
81
+ }
82
+ }
83
+ @media (any-hover: hover) {
84
+ #sbb-breadcrumb-ellipsis:hover {
85
+ --sbb-breadcrumb-group-ellipsis-color: var(--sbb-color-2);
86
+ --sbb-breadcrumb-group-ellipsis-background-color: var(--sbb-background-color-3);
87
+ }
88
+ }
89
+ @media (any-hover: hover) and (forced-colors: active) {
90
+ #sbb-breadcrumb-ellipsis:hover {
91
+ --sbb-breadcrumb-group-ellipsis-border-color: Highlight;
92
+ }
93
+ }
94
+ #sbb-breadcrumb-ellipsis:focus-visible {
95
+ outline-offset: var(--sbb-focus-outline-offset);
96
+ outline: var(--sbb-focus-outline-color) var(--sbb-focus-outline-style, solid) var(--sbb-focus-outline-width);
97
+ }`;
98
+ //#endregion
99
+ //#region src/elements/breadcrumb/breadcrumb-group/breadcrumb-group.component.ts
100
+ var MIN_BREADCRUMBS_TO_COLLAPSE = 3;
101
+ /**
102
+ * It can be used as a container for one or more `sbb-breadcrumb` component.
103
+ *
104
+ * @slot - Use the unnamed slot to add `sbb-breadcrumb` elements.
105
+ */
106
+ var SbbBreadcrumbGroupElement = (() => {
107
+ let _classSuper = SbbNamedSlotListMixin(SbbElement);
108
+ let _instanceExtraInitializers = [];
109
+ let _set__state_decorators;
110
+ return class SbbBreadcrumbGroupElement extends _classSuper {
111
+ static {
112
+ const _metadata = typeof Symbol === "function" && Symbol.metadata ? Object.create(_classSuper[Symbol.metadata] ?? null) : void 0;
113
+ _set__state_decorators = [state()];
114
+ __esDecorate(this, null, _set__state_decorators, {
115
+ kind: "setter",
116
+ name: "_state",
117
+ static: false,
118
+ private: false,
119
+ access: {
120
+ has: (obj) => "_state" in obj,
121
+ set: (obj, value) => {
122
+ obj._state = value;
123
+ }
124
+ },
125
+ metadata: _metadata
126
+ }, null, _instanceExtraInitializers);
127
+ if (_metadata) Object.defineProperty(this, Symbol.metadata, {
128
+ enumerable: true,
129
+ configurable: true,
130
+ writable: true,
131
+ value: _metadata
132
+ });
133
+ }
134
+ static {
135
+ this.elementName = "sbb-breadcrumb-group";
136
+ }
137
+ static {
138
+ this.role = "navigation";
139
+ }
140
+ static {
141
+ this.styles = [boxSizingStyles, breadcrumb_group_default];
142
+ }
143
+ /** The state of the breadcrumb group. */
144
+ set _state(state) {
145
+ if (this._stateInternal) this.internals.states.delete(`state-${this._stateInternal}`);
146
+ this._stateInternal = state;
147
+ if (this._stateInternal) this.internals.states.add(`state-${this._stateInternal}`);
148
+ }
149
+ get _state() {
150
+ return this._stateInternal;
151
+ }
152
+ constructor() {
153
+ super();
154
+ this.listChildLocalNames = (__runInitializers(this, _instanceExtraInitializers), ["sbb-breadcrumb"]);
155
+ this._stateInternal = null;
156
+ this._resizeObserver = new ResizeController(this, {
157
+ target: null,
158
+ skipInitial: true,
159
+ callback: () => this._evaluateCollapsedState()
160
+ });
161
+ this._language = new SbbLanguageController(this);
162
+ this._markForFocus = false;
163
+ this.addEventListener?.("keydown", (e) => this._handleKeyDown(e));
164
+ }
165
+ _handleKeyDown(evt) {
166
+ if (!this.listChildren.length || evt.target !== this && evt.target.parentElement !== this) return;
167
+ if (isArrowKeyPressed(evt)) {
168
+ if (this._state === "collapsed") return this._focusNextCollapsed(evt);
169
+ this._focusNext(evt);
170
+ }
171
+ }
172
+ firstUpdated(changedProperties) {
173
+ super.firstUpdated(changedProperties);
174
+ this._resizeObserver.observe(this);
175
+ this.internals.states.add("loaded");
176
+ }
177
+ willUpdate(changedProperties) {
178
+ super.willUpdate(changedProperties);
179
+ if (changedProperties.has("listChildren")) this._syncBreadcrumbs();
180
+ }
181
+ updated(changedProperties) {
182
+ super.updated(changedProperties);
183
+ if (changedProperties.has("listChildren")) Promise.resolve().then(() => this._evaluateCollapsedState());
184
+ if (this._markForFocus && sbbInputModalityDetector.mostRecentModality === "keyboard") {
185
+ this.listChildren[1]?.focus();
186
+ this._markForFocus = false;
187
+ }
188
+ }
189
+ /** Apply the aria-current attribute to the last sbb-breadcrumb element. */
190
+ _syncBreadcrumbs() {
191
+ this.listChildren.slice(0, -1).filter((c) => c.hasAttribute("accessibility-current")).forEach((c) => c.removeAttribute("accessibility-current"));
192
+ this.listChildren[this.listChildren.length - 1]?.setAttribute("accessibility-current", "page");
193
+ if (this.listChildren.length < MIN_BREADCRUMBS_TO_COLLAPSE) this._state = null;
194
+ }
195
+ /**
196
+ * Sets the focus on the correct element when the ellipsis breadcrumb is displayed and the user is navigating with keyboard's arrows.
197
+ */
198
+ _focusNextCollapsed(evt) {
199
+ const arrayCollapsed = [
200
+ this.listChildren[0],
201
+ this.shadowRoot.querySelector("#sbb-breadcrumb-ellipsis"),
202
+ this.listChildren[this.listChildren.length - 1]
203
+ ];
204
+ this._focusNext(evt, arrayCollapsed);
205
+ }
206
+ _focusNext(evt, breadcrumbs = this.listChildren) {
207
+ breadcrumbs[getNextElementIndex(evt, breadcrumbs.findIndex((e) => e === document.activeElement || e === this.shadowRoot.activeElement), breadcrumbs.length)]?.focus();
208
+ evt.preventDefault();
209
+ }
210
+ /**
211
+ * Note: due to @State annotation on _state, this method triggers a new render; as a consequence, the focus is moved
212
+ * to the `body`, so if the ellipsis element has focus, it's asynchronously forced to the first element.
213
+ */
214
+ _expandBreadcrumbs() {
215
+ this._state = "manually-expanded";
216
+ this._markForFocus = true;
217
+ }
218
+ /** Evaluate if the expanded breadcrumb element fits in page width, otherwise it needs ellipsis */
219
+ _evaluateCollapsedState() {
220
+ if (!this._state && this.scrollWidth > this.offsetWidth && this.listChildren.length >= MIN_BREADCRUMBS_TO_COLLAPSE) {
221
+ this._state = "collapsed";
222
+ this._resizeObserver.hostDisconnected();
223
+ this.removeController(this._resizeObserver);
224
+ }
225
+ }
226
+ _renderCollapsed() {
227
+ return html`
228
+ <li class="sbb-breadcrumb-group__item">
229
+ <slot name="li-0"></slot>
230
+ </li>
231
+ <li class="sbb-breadcrumb-group__item" id="sbb-breadcrumb-group-ellipsis">
232
+ <sbb-icon
233
+ name="chevron-small-right-small"
234
+ class="sbb-breadcrumb-group__divider-icon"
235
+ ></sbb-icon>
236
+ <button
237
+ type="button"
238
+ id="sbb-breadcrumb-ellipsis"
239
+ aria-label=${i18nBreadcrumbEllipsisButtonLabel[this._language.current]}
240
+ aria-expanded="false"
241
+ @click=${() => this._expandBreadcrumbs()}
242
+ >
243
+ ...
244
+ </button>
245
+ </li>
246
+ <li class="sbb-breadcrumb-group__item">
247
+ <sbb-icon
248
+ name="chevron-small-right-small"
249
+ class="sbb-breadcrumb-group__divider-icon"
250
+ ></sbb-icon>
251
+ <slot name=${`li-${this.listChildren.length - 1}`}></slot>
252
+ </li>
253
+ `;
254
+ }
255
+ _renderExpanded() {
256
+ return this.listSlotEntries().map((slot, index, array) => html`
257
+ <li class="sbb-breadcrumb-group__item">
258
+ <slot name=${slot.name}></slot>
259
+ ${index !== array.length - 1 ? html`<sbb-icon
260
+ name="chevron-small-right-small"
261
+ class="sbb-breadcrumb-group__divider-icon"
262
+ ></sbb-icon>` : nothing}
263
+ </li>
264
+ `);
265
+ }
266
+ render() {
267
+ return html`
268
+ <ol class="sbb-breadcrumb-group">
269
+ ${this._state === "collapsed" ? this._renderCollapsed() : this._renderExpanded()}
270
+ </ol>
271
+ ${this.renderHiddenSlot()}
272
+ `;
273
+ }
274
+ };
275
+ })();
276
+ //#endregion
277
+ export { SbbBreadcrumbGroupElement as t };
278
+
279
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"breadcrumb-group.component-zU06d9ni.js","names":[],"sources":["../../../src/elements/breadcrumb/breadcrumb-group/breadcrumb-group.scss?lit&inline","../../../src/elements/breadcrumb/breadcrumb-group/breadcrumb-group.component.ts"],"sourcesContent":["@use '../../core/styles' as sbb;\n\n:host {\n  display: block;\n\n  // Allows size calculation even if in grid or flex context.\n  min-width: 0;\n}\n\n:host(:state(loaded)) {\n  --sbb-breadcrumb-group-visibility: visible;\n}\n\n:host(:is(:state(state-collapsed), :state(state-manually-expanded))) {\n  --sbb-breadcrumb-group-wrap: wrap;\n}\n\n.sbb-breadcrumb-group {\n  @include sbb.list-reset;\n\n  display: flex;\n  flex-wrap: var(--sbb-breadcrumb-group-wrap);\n  column-gap: var(--sbb-spacing-fixed-1x);\n  visibility: var(--sbb-breadcrumb-group-visibility);\n}\n\n.sbb-breadcrumb-group__item {\n  flex: 0 0 auto;\n  display: flex;\n  column-gap: var(--sbb-spacing-fixed-1x);\n}\n\n.sbb-breadcrumb-group__divider-icon {\n  color: var(--sbb-breadcrumb-group-ellipsis-color);\n}\n\n#sbb-breadcrumb-ellipsis {\n  @include sbb.button-reset;\n  @include sbb.text-xxs--regular;\n\n  // line height and letter-spacing needed to match squares drawn in Figma.\n  line-height: 0;\n  letter-spacing: 0.01em;\n  font-size: var(--sbb-text-font-size-xxs);\n  width: var(--sbb-size-icon-ui-small);\n  height: var(--sbb-size-icon-ui-small);\n  border: var(--sbb-breadcrumb-group-ellipsis-border-width) solid\n    var(--sbb-breadcrumb-group-ellipsis-border-color);\n  border-radius: 50%;\n\n  // In order to vertically center ..., we use padding with half the font size.\n  padding-block-end: 0.5em;\n  cursor: var(--sbb-cursor-pointer);\n  color: var(--sbb-breadcrumb-group-ellipsis-color);\n  background-color: var(--sbb-breadcrumb-group-ellipsis-background-color);\n  overflow: hidden;\n\n  @include sbb.if-forced-colors {\n    --sbb-breadcrumb-group-ellipsis-border-width: var(--sbb-border-width-2x);\n    --sbb-breadcrumb-group-ellipsis-border-color: CanvasText;\n  }\n\n  @include sbb.hover-mq($hover: true) {\n    &:hover {\n      --sbb-breadcrumb-group-ellipsis-color: var(--sbb-color-2);\n      --sbb-breadcrumb-group-ellipsis-background-color: var(--sbb-background-color-3);\n\n      @include sbb.if-forced-colors {\n        --sbb-breadcrumb-group-ellipsis-border-color: Highlight;\n      }\n    }\n  }\n\n  &:focus-visible {\n    @include sbb.focus-outline;\n  }\n}\n","import { ResizeController } from '@lit-labs/observers/resize-controller.js';\nimport { type CSSResultGroup, html, nothing, type PropertyValues, type TemplateResult } from 'lit';\nimport { state } from 'lit/decorators.js';\n\nimport {\n  getNextElementIndex,\n  isArrowKeyPressed,\n  sbbInputModalityDetector,\n} from '../../core/a11y.ts';\nimport { SbbElement } from '../../core/base-elements.ts';\nimport { SbbLanguageController } from '../../core/controllers.ts';\nimport { i18nBreadcrumbEllipsisButtonLabel } from '../../core/i18n.ts';\nimport { SbbNamedSlotListMixin, type WithListChildren } from '../../core/mixins.ts';\nimport { boxSizingStyles } from '../../core/styles.ts';\nimport type { SbbBreadcrumbElement } from '../breadcrumb/breadcrumb.component.ts';\n\nimport style from './breadcrumb-group.scss?lit&inline';\n\nimport '../../icon.ts';\n\nconst MIN_BREADCRUMBS_TO_COLLAPSE = 3;\n\n/**\n * It can be used as a container for one or more `sbb-breadcrumb` component.\n *\n * @slot - Use the unnamed slot to add `sbb-breadcrumb` elements.\n */\nexport class SbbBreadcrumbGroupElement extends SbbNamedSlotListMixin<\n  SbbBreadcrumbElement,\n  typeof SbbElement\n>(SbbElement) {\n  public static override readonly elementName: string = 'sbb-breadcrumb-group';\n  public static override readonly role = 'navigation';\n  public static override styles: CSSResultGroup = [boxSizingStyles, style];\n  protected override readonly listChildLocalNames = ['sbb-breadcrumb'];\n\n  /** The state of the breadcrumb group. */\n  @state()\n  private set _state(state: 'collapsed' | 'manually-expanded' | null) {\n    if (this._stateInternal) {\n      this.internals.states.delete(`state-${this._stateInternal}`);\n    }\n    this._stateInternal = state;\n    if (this._stateInternal) {\n      this.internals.states.add(`state-${this._stateInternal}`);\n    }\n  }\n  private get _state(): 'collapsed' | 'manually-expanded' | null {\n    return this._stateInternal;\n  }\n  private _stateInternal: 'collapsed' | 'manually-expanded' | null = null;\n\n  private _resizeObserver = new ResizeController(this, {\n    target: null,\n    skipInitial: true,\n    callback: () => this._evaluateCollapsedState(),\n  });\n  private _language = new SbbLanguageController(this);\n  private _markForFocus = false;\n\n  public constructor() {\n    super();\n    this.addEventListener?.('keydown', (e) => this._handleKeyDown(e));\n  }\n\n  private _handleKeyDown(evt: KeyboardEvent): void {\n    if (\n      !this.listChildren.length ||\n      // don't trap nested handling\n      ((evt.target as HTMLElement) !== this && (evt.target as HTMLElement).parentElement !== this)\n    ) {\n      return;\n    }\n\n    if (isArrowKeyPressed(evt)) {\n      if (this._state === 'collapsed') {\n        return this._focusNextCollapsed(evt);\n      }\n      this._focusNext(evt);\n    }\n  }\n\n  protected override firstUpdated(changedProperties: PropertyValues<this>): void {\n    super.firstUpdated(changedProperties);\n\n    this._resizeObserver.observe(this);\n    this.internals.states.add('loaded');\n  }\n\n  protected override willUpdate(changedProperties: PropertyValues<WithListChildren<this>>): void {\n    super.willUpdate(changedProperties);\n\n    if (changedProperties.has('listChildren')) {\n      this._syncBreadcrumbs();\n    }\n  }\n\n  protected override updated(changedProperties: PropertyValues<WithListChildren<this>>): void {\n    super.updated(changedProperties);\n    if (changedProperties.has('listChildren')) {\n      Promise.resolve().then(() => this._evaluateCollapsedState());\n    }\n    if (this._markForFocus && sbbInputModalityDetector.mostRecentModality === 'keyboard') {\n      this.listChildren[1]?.focus();\n\n      // Reset mark for focus\n      this._markForFocus = false;\n    }\n  }\n\n  /** Apply the aria-current attribute to the last sbb-breadcrumb element. */\n  private _syncBreadcrumbs(): void {\n    this.listChildren\n      .slice(0, -1)\n      .filter((c) => c.hasAttribute('accessibility-current'))\n      .forEach((c) => c.removeAttribute('accessibility-current'));\n    this.listChildren[this.listChildren.length - 1]?.setAttribute('accessibility-current', 'page');\n\n    // If it is not expandable, reset state\n    if (this.listChildren.length < MIN_BREADCRUMBS_TO_COLLAPSE) {\n      this._state = null;\n    }\n  }\n\n  /**\n   * Sets the focus on the correct element when the ellipsis breadcrumb is displayed and the user is navigating with keyboard's arrows.\n   */\n  private _focusNextCollapsed(evt: KeyboardEvent): void {\n    const arrayCollapsed: SbbBreadcrumbElement[] = [\n      this.listChildren[0],\n      this.shadowRoot!.querySelector('#sbb-breadcrumb-ellipsis') as SbbBreadcrumbElement,\n      this.listChildren[this.listChildren.length - 1],\n    ];\n    this._focusNext(evt, arrayCollapsed);\n  }\n\n  private _focusNext(\n    evt: KeyboardEvent,\n    breadcrumbs: SbbBreadcrumbElement[] = this.listChildren,\n  ): void {\n    const current: number = breadcrumbs.findIndex(\n      (e) => e === document.activeElement || e === this.shadowRoot!.activeElement,\n    );\n    const nextIndex: number = getNextElementIndex(evt, current, breadcrumbs.length);\n    breadcrumbs[nextIndex]?.focus();\n    evt.preventDefault();\n  }\n\n  /**\n   * Note: due to @State annotation on _state, this method triggers a new render; as a consequence, the focus is moved\n   * to the `body`, so if the ellipsis element has focus, it's asynchronously forced to the first element.\n   */\n  private _expandBreadcrumbs(): void {\n    this._state = 'manually-expanded';\n    this._markForFocus = true;\n  }\n\n  /** Evaluate if the expanded breadcrumb element fits in page width, otherwise it needs ellipsis */\n  private _evaluateCollapsedState(): void {\n    if (\n      !this._state &&\n      this.scrollWidth > this.offsetWidth &&\n      this.listChildren.length >= MIN_BREADCRUMBS_TO_COLLAPSE\n    ) {\n      this._state = 'collapsed';\n      this._resizeObserver.hostDisconnected();\n      this.removeController(this._resizeObserver);\n    }\n  }\n\n  private _renderCollapsed(): TemplateResult {\n    return html`\n      <li class=\"sbb-breadcrumb-group__item\">\n        <slot name=\"li-0\"></slot>\n      </li>\n      <li class=\"sbb-breadcrumb-group__item\" id=\"sbb-breadcrumb-group-ellipsis\">\n        <sbb-icon\n          name=\"chevron-small-right-small\"\n          class=\"sbb-breadcrumb-group__divider-icon\"\n        ></sbb-icon>\n        <button\n          type=\"button\"\n          id=\"sbb-breadcrumb-ellipsis\"\n          aria-label=${i18nBreadcrumbEllipsisButtonLabel[this._language.current]}\n          aria-expanded=\"false\"\n          @click=${() => this._expandBreadcrumbs()}\n        >\n          ...\n        </button>\n      </li>\n      <li class=\"sbb-breadcrumb-group__item\">\n        <sbb-icon\n          name=\"chevron-small-right-small\"\n          class=\"sbb-breadcrumb-group__divider-icon\"\n        ></sbb-icon>\n        <slot name=${`li-${this.listChildren.length - 1}`}></slot>\n      </li>\n    `;\n  }\n\n  private _renderExpanded(): TemplateResult[] {\n    return this.listSlotEntries().map(\n      (slot, index, array) => html`\n        <li class=\"sbb-breadcrumb-group__item\">\n          <slot name=${slot.name}></slot>\n          ${index !== array.length - 1\n            ? html`<sbb-icon\n                name=\"chevron-small-right-small\"\n                class=\"sbb-breadcrumb-group__divider-icon\"\n              ></sbb-icon>`\n            : nothing}\n        </li>\n      `,\n    );\n  }\n\n  protected override render(): TemplateResult {\n    return html`\n      <ol class=\"sbb-breadcrumb-group\">\n        ${this._state === 'collapsed' ? this._renderCollapsed() : this._renderExpanded()}\n      </ol>\n      ${this.renderHiddenSlot()}\n    `;\n  }\n}\n\ndeclare global {\n  interface HTMLElementTagNameMap {\n    // eslint-disable-next-line @typescript-eslint/naming-convention\n    'sbb-breadcrumb-group': SbbBreadcrumbGroupElement;\n  }\n}\n"],"mappings":"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"}
@@ -1,4 +1,4 @@
1
- import { t as SbbBreadcrumbGroupElement } from "./breadcrumb-group.component-DjsoMTPI.js";
1
+ import { t as SbbBreadcrumbGroupElement } from "./breadcrumb-group.component-zU06d9ni.js";
2
2
  import { t as SbbBreadcrumbElement } from "./breadcrumb.component-DvOf-PiO.js";
3
3
  import "./breadcrumb.pure.js";
4
4
  //#region src/elements/breadcrumb.ts
@@ -1,3 +1,3 @@
1
- import { t as SbbBreadcrumbGroupElement } from "./breadcrumb-group.component-DjsoMTPI.js";
1
+ import { t as SbbBreadcrumbGroupElement } from "./breadcrumb-group.component-zU06d9ni.js";
2
2
  import { t as SbbBreadcrumbElement } from "./breadcrumb.component-DvOf-PiO.js";
3
3
  export { SbbBreadcrumbElement, SbbBreadcrumbGroupElement };
package/package.json CHANGED
@@ -1,13 +1,13 @@
1
1
  {
2
2
  "name": "@sbb-esta/lyne-elements-dev",
3
- "version": "4.7.0-dev.1773755649",
3
+ "version": "4.7.0-dev.1773817331",
4
4
  "description": "Lyne Design System",
5
5
  "keywords": [
6
6
  "design system",
7
7
  "web components",
8
8
  "lit",
9
9
  "storybook",
10
- "https://github.com/sbb-design-systems/lyne-components/commit/937ad5f5766d215196d1dd0988719f23051f100b"
10
+ "https://github.com/sbb-design-systems/lyne-components/commit/47decd8f58ebb0f714ba4f2d99c7d7a727cdbb05"
11
11
  ],
12
12
  "type": "module",
13
13
  "exports": {
@@ -1,278 +0,0 @@
1
- import { __esDecorate, __runInitializers } from "tslib";
2
- import { css, html, nothing } from "lit";
3
- import { state } from "lit/decorators.js";
4
- import { SbbElement } from "./core/base-elements.js";
5
- import { SbbNamedSlotListMixin } from "./core/mixins.js";
6
- import { boxSizingStyles } from "./core/styles.js";
7
- import { SbbLanguageController } from "./core/controllers.js";
8
- import { i18nBreadcrumbEllipsisButtonLabel } from "./core/i18n.js";
9
- import "./icon.js";
10
- import { ResizeController } from "@lit-labs/observers/resize-controller.js";
11
- import { getNextElementIndex, isArrowKeyPressed, sbbInputModalityDetector } from "./core/a11y.js";
12
- //#region src/elements/breadcrumb/breadcrumb-group/breadcrumb-group.scss?lit&inline
13
- var breadcrumb_group_default = css`:host {
14
- display: block;
15
- }
16
-
17
- :host(:is(:state(loaded),[state--loaded])) {
18
- --sbb-breadcrumb-group-visibility: visible;
19
- }
20
-
21
- :host(:is(:is(:state(state-collapsed),[state--state-collapsed]), :is(:state(state-manually-expanded),[state--state-manually-expanded]))) {
22
- --sbb-breadcrumb-group-wrap: wrap;
23
- }
24
-
25
- .sbb-breadcrumb-group {
26
- list-style: none;
27
- margin: 0;
28
- padding: 0;
29
- font-size: inherit;
30
- display: flex;
31
- flex-wrap: var(--sbb-breadcrumb-group-wrap);
32
- column-gap: var(--sbb-spacing-fixed-1x);
33
- visibility: var(--sbb-breadcrumb-group-visibility);
34
- }
35
-
36
- .sbb-breadcrumb-group__item {
37
- flex: 0 0 auto;
38
- display: flex;
39
- column-gap: var(--sbb-spacing-fixed-1x);
40
- }
41
-
42
- .sbb-breadcrumb-group__divider-icon {
43
- color: var(--sbb-breadcrumb-group-ellipsis-color);
44
- }
45
-
46
- #sbb-breadcrumb-ellipsis {
47
- appearance: none;
48
- box-sizing: border-box;
49
- margin: 0;
50
- outline: none;
51
- border: none;
52
- border-radius: 0;
53
- padding: 0;
54
- background: none;
55
- -webkit-tap-highlight-color: transparent;
56
- color: inherit;
57
- --sbb-text-font-size: var(--sbb-text-font-size-xxs);
58
- font-family: var(--sbb-typo-font-family);
59
- font-weight: normal;
60
- line-height: var(--sbb-typo-line-height-text);
61
- letter-spacing: var(--sbb-typo-letter-spacing-text);
62
- font-size: var(--sbb-text-font-size);
63
- line-height: 0;
64
- letter-spacing: 0.01em;
65
- font-size: var(--sbb-text-font-size-xxs);
66
- width: var(--sbb-size-icon-ui-small);
67
- height: var(--sbb-size-icon-ui-small);
68
- border: var(--sbb-breadcrumb-group-ellipsis-border-width) solid var(--sbb-breadcrumb-group-ellipsis-border-color);
69
- border-radius: 50%;
70
- padding-block-end: 0.5em;
71
- cursor: var(--sbb-cursor-pointer);
72
- color: var(--sbb-breadcrumb-group-ellipsis-color);
73
- background-color: var(--sbb-breadcrumb-group-ellipsis-background-color);
74
- overflow: hidden;
75
- }
76
- @media (forced-colors: active) {
77
- #sbb-breadcrumb-ellipsis {
78
- --sbb-breadcrumb-group-ellipsis-border-width: var(--sbb-border-width-2x);
79
- --sbb-breadcrumb-group-ellipsis-border-color: CanvasText;
80
- }
81
- }
82
- @media (any-hover: hover) {
83
- #sbb-breadcrumb-ellipsis:hover {
84
- --sbb-breadcrumb-group-ellipsis-color: var(--sbb-color-2);
85
- --sbb-breadcrumb-group-ellipsis-background-color: var(--sbb-background-color-3);
86
- }
87
- }
88
- @media (any-hover: hover) and (forced-colors: active) {
89
- #sbb-breadcrumb-ellipsis:hover {
90
- --sbb-breadcrumb-group-ellipsis-border-color: Highlight;
91
- }
92
- }
93
- #sbb-breadcrumb-ellipsis:focus-visible {
94
- outline-offset: var(--sbb-focus-outline-offset);
95
- outline: var(--sbb-focus-outline-color) var(--sbb-focus-outline-style, solid) var(--sbb-focus-outline-width);
96
- }`;
97
- //#endregion
98
- //#region src/elements/breadcrumb/breadcrumb-group/breadcrumb-group.component.ts
99
- var MIN_BREADCRUMBS_TO_COLLAPSE = 3;
100
- /**
101
- * It can be used as a container for one or more `sbb-breadcrumb` component.
102
- *
103
- * @slot - Use the unnamed slot to add `sbb-breadcrumb` elements.
104
- */
105
- var SbbBreadcrumbGroupElement = (() => {
106
- let _classSuper = SbbNamedSlotListMixin(SbbElement);
107
- let _instanceExtraInitializers = [];
108
- let _set__state_decorators;
109
- return class SbbBreadcrumbGroupElement extends _classSuper {
110
- static {
111
- const _metadata = typeof Symbol === "function" && Symbol.metadata ? Object.create(_classSuper[Symbol.metadata] ?? null) : void 0;
112
- _set__state_decorators = [state()];
113
- __esDecorate(this, null, _set__state_decorators, {
114
- kind: "setter",
115
- name: "_state",
116
- static: false,
117
- private: false,
118
- access: {
119
- has: (obj) => "_state" in obj,
120
- set: (obj, value) => {
121
- obj._state = value;
122
- }
123
- },
124
- metadata: _metadata
125
- }, null, _instanceExtraInitializers);
126
- if (_metadata) Object.defineProperty(this, Symbol.metadata, {
127
- enumerable: true,
128
- configurable: true,
129
- writable: true,
130
- value: _metadata
131
- });
132
- }
133
- static {
134
- this.elementName = "sbb-breadcrumb-group";
135
- }
136
- static {
137
- this.role = "navigation";
138
- }
139
- static {
140
- this.styles = [boxSizingStyles, breadcrumb_group_default];
141
- }
142
- /** The state of the breadcrumb group. */
143
- set _state(state) {
144
- if (this._stateInternal) this.internals.states.delete(`state-${this._stateInternal}`);
145
- this._stateInternal = state;
146
- if (this._stateInternal) this.internals.states.add(`state-${this._stateInternal}`);
147
- }
148
- get _state() {
149
- return this._stateInternal;
150
- }
151
- constructor() {
152
- super();
153
- this.listChildLocalNames = (__runInitializers(this, _instanceExtraInitializers), ["sbb-breadcrumb"]);
154
- this._stateInternal = null;
155
- this._resizeObserver = new ResizeController(this, {
156
- target: null,
157
- skipInitial: true,
158
- callback: () => this._evaluateCollapsedState()
159
- });
160
- this._language = new SbbLanguageController(this);
161
- this._markForFocus = false;
162
- this.addEventListener?.("keydown", (e) => this._handleKeyDown(e));
163
- }
164
- _handleKeyDown(evt) {
165
- if (!this.listChildren.length || evt.target !== this && evt.target.parentElement !== this) return;
166
- if (isArrowKeyPressed(evt)) {
167
- if (this._state === "collapsed") return this._focusNextCollapsed(evt);
168
- this._focusNext(evt);
169
- }
170
- }
171
- firstUpdated(changedProperties) {
172
- super.firstUpdated(changedProperties);
173
- this._resizeObserver.observe(this);
174
- this.internals.states.add("loaded");
175
- }
176
- willUpdate(changedProperties) {
177
- super.willUpdate(changedProperties);
178
- if (changedProperties.has("listChildren")) this._syncBreadcrumbs();
179
- }
180
- updated(changedProperties) {
181
- super.updated(changedProperties);
182
- if (changedProperties.has("listChildren")) Promise.resolve().then(() => this._evaluateCollapsedState());
183
- if (this._markForFocus && sbbInputModalityDetector.mostRecentModality === "keyboard") {
184
- this.listChildren[1]?.focus();
185
- this._markForFocus = false;
186
- }
187
- }
188
- /** Apply the aria-current attribute to the last sbb-breadcrumb element. */
189
- _syncBreadcrumbs() {
190
- this.listChildren.slice(0, -1).filter((c) => c.hasAttribute("accessibility-current")).forEach((c) => c.removeAttribute("accessibility-current"));
191
- this.listChildren[this.listChildren.length - 1]?.setAttribute("accessibility-current", "page");
192
- if (this.listChildren.length < MIN_BREADCRUMBS_TO_COLLAPSE) this._state = null;
193
- }
194
- /**
195
- * Sets the focus on the correct element when the ellipsis breadcrumb is displayed and the user is navigating with keyboard's arrows.
196
- */
197
- _focusNextCollapsed(evt) {
198
- const arrayCollapsed = [
199
- this.listChildren[0],
200
- this.shadowRoot.querySelector("#sbb-breadcrumb-ellipsis"),
201
- this.listChildren[this.listChildren.length - 1]
202
- ];
203
- this._focusNext(evt, arrayCollapsed);
204
- }
205
- _focusNext(evt, breadcrumbs = this.listChildren) {
206
- breadcrumbs[getNextElementIndex(evt, breadcrumbs.findIndex((e) => e === document.activeElement || e === this.shadowRoot.activeElement), breadcrumbs.length)]?.focus();
207
- evt.preventDefault();
208
- }
209
- /**
210
- * Note: due to @State annotation on _state, this method triggers a new render; as a consequence, the focus is moved
211
- * to the `body`, so if the ellipsis element has focus, it's asynchronously forced to the first element.
212
- */
213
- _expandBreadcrumbs() {
214
- this._state = "manually-expanded";
215
- this._markForFocus = true;
216
- }
217
- /** Evaluate if the expanded breadcrumb element fits in page width, otherwise it needs ellipsis */
218
- _evaluateCollapsedState() {
219
- if (!this._state && this.scrollWidth > this.offsetWidth && this.listChildren.length >= MIN_BREADCRUMBS_TO_COLLAPSE) {
220
- this._state = "collapsed";
221
- this._resizeObserver.hostDisconnected();
222
- this.removeController(this._resizeObserver);
223
- }
224
- }
225
- _renderCollapsed() {
226
- return html`
227
- <li class="sbb-breadcrumb-group__item">
228
- <slot name="li-0"></slot>
229
- </li>
230
- <li class="sbb-breadcrumb-group__item" id="sbb-breadcrumb-group-ellipsis">
231
- <sbb-icon
232
- name="chevron-small-right-small"
233
- class="sbb-breadcrumb-group__divider-icon"
234
- ></sbb-icon>
235
- <button
236
- type="button"
237
- id="sbb-breadcrumb-ellipsis"
238
- aria-label=${i18nBreadcrumbEllipsisButtonLabel[this._language.current]}
239
- aria-expanded="false"
240
- @click=${() => this._expandBreadcrumbs()}
241
- >
242
- ...
243
- </button>
244
- </li>
245
- <li class="sbb-breadcrumb-group__item">
246
- <sbb-icon
247
- name="chevron-small-right-small"
248
- class="sbb-breadcrumb-group__divider-icon"
249
- ></sbb-icon>
250
- <slot name=${`li-${this.listChildren.length - 1}`}></slot>
251
- </li>
252
- `;
253
- }
254
- _renderExpanded() {
255
- return this.listSlotEntries().map((slot, index, array) => html`
256
- <li class="sbb-breadcrumb-group__item">
257
- <slot name=${slot.name}></slot>
258
- ${index !== array.length - 1 ? html`<sbb-icon
259
- name="chevron-small-right-small"
260
- class="sbb-breadcrumb-group__divider-icon"
261
- ></sbb-icon>` : nothing}
262
- </li>
263
- `);
264
- }
265
- render() {
266
- return html`
267
- <ol class="sbb-breadcrumb-group">
268
- ${this._state === "collapsed" ? this._renderCollapsed() : this._renderExpanded()}
269
- </ol>
270
- ${this.renderHiddenSlot()}
271
- `;
272
- }
273
- };
274
- })();
275
- //#endregion
276
- export { SbbBreadcrumbGroupElement as t };
277
-
278
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"breadcrumb-group.component-DjsoMTPI.js","names":[],"sources":["../../../src/elements/breadcrumb/breadcrumb-group/breadcrumb-group.scss?lit&inline","../../../src/elements/breadcrumb/breadcrumb-group/breadcrumb-group.component.ts"],"sourcesContent":["@use '../../core/styles' as sbb;\n\n:host {\n  display: block;\n}\n\n:host(:state(loaded)) {\n  --sbb-breadcrumb-group-visibility: visible;\n}\n\n:host(:is(:state(state-collapsed), :state(state-manually-expanded))) {\n  --sbb-breadcrumb-group-wrap: wrap;\n}\n\n.sbb-breadcrumb-group {\n  @include sbb.list-reset;\n\n  display: flex;\n  flex-wrap: var(--sbb-breadcrumb-group-wrap);\n  column-gap: var(--sbb-spacing-fixed-1x);\n  visibility: var(--sbb-breadcrumb-group-visibility);\n}\n\n.sbb-breadcrumb-group__item {\n  flex: 0 0 auto;\n  display: flex;\n  column-gap: var(--sbb-spacing-fixed-1x);\n}\n\n.sbb-breadcrumb-group__divider-icon {\n  color: var(--sbb-breadcrumb-group-ellipsis-color);\n}\n\n#sbb-breadcrumb-ellipsis {\n  @include sbb.button-reset;\n  @include sbb.text-xxs--regular;\n\n  // line height and letter-spacing needed to match squares drawn in Figma.\n  line-height: 0;\n  letter-spacing: 0.01em;\n  font-size: var(--sbb-text-font-size-xxs);\n  width: var(--sbb-size-icon-ui-small);\n  height: var(--sbb-size-icon-ui-small);\n  border: var(--sbb-breadcrumb-group-ellipsis-border-width) solid\n    var(--sbb-breadcrumb-group-ellipsis-border-color);\n  border-radius: 50%;\n\n  // In order to vertically center ..., we use padding with half the font size.\n  padding-block-end: 0.5em;\n  cursor: var(--sbb-cursor-pointer);\n  color: var(--sbb-breadcrumb-group-ellipsis-color);\n  background-color: var(--sbb-breadcrumb-group-ellipsis-background-color);\n  overflow: hidden;\n\n  @include sbb.if-forced-colors {\n    --sbb-breadcrumb-group-ellipsis-border-width: var(--sbb-border-width-2x);\n    --sbb-breadcrumb-group-ellipsis-border-color: CanvasText;\n  }\n\n  @include sbb.hover-mq($hover: true) {\n    &:hover {\n      --sbb-breadcrumb-group-ellipsis-color: var(--sbb-color-2);\n      --sbb-breadcrumb-group-ellipsis-background-color: var(--sbb-background-color-3);\n\n      @include sbb.if-forced-colors {\n        --sbb-breadcrumb-group-ellipsis-border-color: Highlight;\n      }\n    }\n  }\n\n  &:focus-visible {\n    @include sbb.focus-outline;\n  }\n}\n","import { ResizeController } from '@lit-labs/observers/resize-controller.js';\nimport { type CSSResultGroup, html, nothing, type PropertyValues, type TemplateResult } from 'lit';\nimport { state } from 'lit/decorators.js';\n\nimport {\n  getNextElementIndex,\n  isArrowKeyPressed,\n  sbbInputModalityDetector,\n} from '../../core/a11y.ts';\nimport { SbbElement } from '../../core/base-elements.ts';\nimport { SbbLanguageController } from '../../core/controllers.ts';\nimport { i18nBreadcrumbEllipsisButtonLabel } from '../../core/i18n.ts';\nimport { SbbNamedSlotListMixin, type WithListChildren } from '../../core/mixins.ts';\nimport { boxSizingStyles } from '../../core/styles.ts';\nimport type { SbbBreadcrumbElement } from '../breadcrumb/breadcrumb.component.ts';\n\nimport style from './breadcrumb-group.scss?lit&inline';\n\nimport '../../icon.ts';\n\nconst MIN_BREADCRUMBS_TO_COLLAPSE = 3;\n\n/**\n * It can be used as a container for one or more `sbb-breadcrumb` component.\n *\n * @slot - Use the unnamed slot to add `sbb-breadcrumb` elements.\n */\nexport class SbbBreadcrumbGroupElement extends SbbNamedSlotListMixin<\n  SbbBreadcrumbElement,\n  typeof SbbElement\n>(SbbElement) {\n  public static override readonly elementName: string = 'sbb-breadcrumb-group';\n  public static override readonly role = 'navigation';\n  public static override styles: CSSResultGroup = [boxSizingStyles, style];\n  protected override readonly listChildLocalNames = ['sbb-breadcrumb'];\n\n  /** The state of the breadcrumb group. */\n  @state()\n  private set _state(state: 'collapsed' | 'manually-expanded' | null) {\n    if (this._stateInternal) {\n      this.internals.states.delete(`state-${this._stateInternal}`);\n    }\n    this._stateInternal = state;\n    if (this._stateInternal) {\n      this.internals.states.add(`state-${this._stateInternal}`);\n    }\n  }\n  private get _state(): 'collapsed' | 'manually-expanded' | null {\n    return this._stateInternal;\n  }\n  private _stateInternal: 'collapsed' | 'manually-expanded' | null = null;\n\n  private _resizeObserver = new ResizeController(this, {\n    target: null,\n    skipInitial: true,\n    callback: () => this._evaluateCollapsedState(),\n  });\n  private _language = new SbbLanguageController(this);\n  private _markForFocus = false;\n\n  public constructor() {\n    super();\n    this.addEventListener?.('keydown', (e) => this._handleKeyDown(e));\n  }\n\n  private _handleKeyDown(evt: KeyboardEvent): void {\n    if (\n      !this.listChildren.length ||\n      // don't trap nested handling\n      ((evt.target as HTMLElement) !== this && (evt.target as HTMLElement).parentElement !== this)\n    ) {\n      return;\n    }\n\n    if (isArrowKeyPressed(evt)) {\n      if (this._state === 'collapsed') {\n        return this._focusNextCollapsed(evt);\n      }\n      this._focusNext(evt);\n    }\n  }\n\n  protected override firstUpdated(changedProperties: PropertyValues<this>): void {\n    super.firstUpdated(changedProperties);\n\n    this._resizeObserver.observe(this);\n    this.internals.states.add('loaded');\n  }\n\n  protected override willUpdate(changedProperties: PropertyValues<WithListChildren<this>>): void {\n    super.willUpdate(changedProperties);\n\n    if (changedProperties.has('listChildren')) {\n      this._syncBreadcrumbs();\n    }\n  }\n\n  protected override updated(changedProperties: PropertyValues<WithListChildren<this>>): void {\n    super.updated(changedProperties);\n    if (changedProperties.has('listChildren')) {\n      Promise.resolve().then(() => this._evaluateCollapsedState());\n    }\n    if (this._markForFocus && sbbInputModalityDetector.mostRecentModality === 'keyboard') {\n      this.listChildren[1]?.focus();\n\n      // Reset mark for focus\n      this._markForFocus = false;\n    }\n  }\n\n  /** Apply the aria-current attribute to the last sbb-breadcrumb element. */\n  private _syncBreadcrumbs(): void {\n    this.listChildren\n      .slice(0, -1)\n      .filter((c) => c.hasAttribute('accessibility-current'))\n      .forEach((c) => c.removeAttribute('accessibility-current'));\n    this.listChildren[this.listChildren.length - 1]?.setAttribute('accessibility-current', 'page');\n\n    // If it is not expandable, reset state\n    if (this.listChildren.length < MIN_BREADCRUMBS_TO_COLLAPSE) {\n      this._state = null;\n    }\n  }\n\n  /**\n   * Sets the focus on the correct element when the ellipsis breadcrumb is displayed and the user is navigating with keyboard's arrows.\n   */\n  private _focusNextCollapsed(evt: KeyboardEvent): void {\n    const arrayCollapsed: SbbBreadcrumbElement[] = [\n      this.listChildren[0],\n      this.shadowRoot!.querySelector('#sbb-breadcrumb-ellipsis') as SbbBreadcrumbElement,\n      this.listChildren[this.listChildren.length - 1],\n    ];\n    this._focusNext(evt, arrayCollapsed);\n  }\n\n  private _focusNext(\n    evt: KeyboardEvent,\n    breadcrumbs: SbbBreadcrumbElement[] = this.listChildren,\n  ): void {\n    const current: number = breadcrumbs.findIndex(\n      (e) => e === document.activeElement || e === this.shadowRoot!.activeElement,\n    );\n    const nextIndex: number = getNextElementIndex(evt, current, breadcrumbs.length);\n    breadcrumbs[nextIndex]?.focus();\n    evt.preventDefault();\n  }\n\n  /**\n   * Note: due to @State annotation on _state, this method triggers a new render; as a consequence, the focus is moved\n   * to the `body`, so if the ellipsis element has focus, it's asynchronously forced to the first element.\n   */\n  private _expandBreadcrumbs(): void {\n    this._state = 'manually-expanded';\n    this._markForFocus = true;\n  }\n\n  /** Evaluate if the expanded breadcrumb element fits in page width, otherwise it needs ellipsis */\n  private _evaluateCollapsedState(): void {\n    if (\n      !this._state &&\n      this.scrollWidth > this.offsetWidth &&\n      this.listChildren.length >= MIN_BREADCRUMBS_TO_COLLAPSE\n    ) {\n      this._state = 'collapsed';\n      this._resizeObserver.hostDisconnected();\n      this.removeController(this._resizeObserver);\n    }\n  }\n\n  private _renderCollapsed(): TemplateResult {\n    return html`\n      <li class=\"sbb-breadcrumb-group__item\">\n        <slot name=\"li-0\"></slot>\n      </li>\n      <li class=\"sbb-breadcrumb-group__item\" id=\"sbb-breadcrumb-group-ellipsis\">\n        <sbb-icon\n          name=\"chevron-small-right-small\"\n          class=\"sbb-breadcrumb-group__divider-icon\"\n        ></sbb-icon>\n        <button\n          type=\"button\"\n          id=\"sbb-breadcrumb-ellipsis\"\n          aria-label=${i18nBreadcrumbEllipsisButtonLabel[this._language.current]}\n          aria-expanded=\"false\"\n          @click=${() => this._expandBreadcrumbs()}\n        >\n          ...\n        </button>\n      </li>\n      <li class=\"sbb-breadcrumb-group__item\">\n        <sbb-icon\n          name=\"chevron-small-right-small\"\n          class=\"sbb-breadcrumb-group__divider-icon\"\n        ></sbb-icon>\n        <slot name=${`li-${this.listChildren.length - 1}`}></slot>\n      </li>\n    `;\n  }\n\n  private _renderExpanded(): TemplateResult[] {\n    return this.listSlotEntries().map(\n      (slot, index, array) => html`\n        <li class=\"sbb-breadcrumb-group__item\">\n          <slot name=${slot.name}></slot>\n          ${index !== array.length - 1\n            ? html`<sbb-icon\n                name=\"chevron-small-right-small\"\n                class=\"sbb-breadcrumb-group__divider-icon\"\n              ></sbb-icon>`\n            : nothing}\n        </li>\n      `,\n    );\n  }\n\n  protected override render(): TemplateResult {\n    return html`\n      <ol class=\"sbb-breadcrumb-group\">\n        ${this._state === 'collapsed' ? this._renderCollapsed() : this._renderExpanded()}\n      </ol>\n      ${this.renderHiddenSlot()}\n    `;\n  }\n}\n\ndeclare global {\n  interface HTMLElementTagNameMap {\n    // eslint-disable-next-line @typescript-eslint/naming-convention\n    'sbb-breadcrumb-group': SbbBreadcrumbGroupElement;\n  }\n}\n"],"mappings":"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"}