@sbb-esta/lyne-elements-dev 4.8.1-dev.1773933819 → 4.8.1-dev.1774003987

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,446 +0,0 @@
1
- import { __esDecorate, __runInitializers } from "tslib";
2
- import { css, html, isServer } from "lit";
3
- import { property, state } from "lit/decorators.js";
4
- import { SbbElement } from "./core/base-elements.js";
5
- import { forceType, idReference } from "./core/decorators.js";
6
- import { isLean } from "./core/dom.js";
7
- import { boxSizingStyles } from "./core/styles.js";
8
- import { SbbFocusVisibleWithinController } from "./core/a11y.js";
9
- //#region src/elements/header/header/header.scss?lit&inline
10
- var header_default = css`:host {
11
- --sbb-logo-height: 1rem;
12
- --sbb-signet-height: 1rem;
13
- --sbb-header-transition-duration: var(
14
- --sbb-disable-animation-duration,
15
- var(--sbb-animation-duration-6x)
16
- );
17
- }
18
- @media (min-width: calc(64rem)) {
19
- :host {
20
- --sbb-logo-height: 1.25rem;
21
- --sbb-signet-height: 1.25rem;
22
- }
23
- }
24
- :host {
25
- display: block;
26
- height: var(--sbb-header-height);
27
- }
28
-
29
- :host([size=s]) ::slotted(:is(sbb-header-button, sbb-header-link)) {
30
- --sbb-header-action-min-height: var(--sbb-size-element-xs);
31
- --sbb-header-action-padding-inline: var(--sbb-spacing-fixed-4x);
32
- }
33
-
34
- :host([hide-on-scroll]) {
35
- --sbb-header-position: absolute;
36
- }
37
-
38
- :host([hide-on-scroll]:is(:state(fixed),[state--fixed])) {
39
- --sbb-header-position: fixed;
40
- --sbb-header-animation-name: hide;
41
- --sbb-header-transform: translate3d(0, -100%, 0);
42
- --sbb-header-inset-inline-end: var(--sbb-scrollbar-width, 0);
43
- }
44
-
45
- :host([hide-on-scroll]:is(:state(fixed),[state--fixed]):is(:state(animated),[state--animated])) {
46
- --sbb-header-transition-timing: cubic-bezier(0.4, 0, 1, 1);
47
- }
48
-
49
- :host([hide-on-scroll]:is(:state(fixed),[state--fixed]):is(:state(visible),[state--visible])) {
50
- --sbb-header-animation-name: show;
51
- --sbb-header-transform: translate3d(0, 0, 0);
52
- --sbb-header-transition-timing: cubic-bezier(0, 0, 0.2, 1);
53
- }
54
-
55
- :host([hide-on-scroll]:is(:state(fixed),[state--fixed]):is(:state(has-visible-focus-within),[state--has-visible-focus-within])) {
56
- --sbb-header-transition-duration: 0;
57
- --sbb-header-transform: translate3d(0, 0, 0);
58
- }
59
-
60
- .sbb-header {
61
- position: var(--sbb-header-position);
62
- inset: 0 var(--sbb-header-inset-inline-end) auto 0;
63
- background: var(--sbb-header-background);
64
- z-index: var(--sbb-header-z-index, 10);
65
- transform: var(--sbb-header-transform);
66
- transition-property: box-shadow;
67
- transition-duration: var(--sbb-header-transition-duration);
68
- transition-timing-function: var(--sbb-header-transition-timing);
69
- animation-name: var(--sbb-header-animation-name);
70
- animation-duration: var(--sbb-header-transition-duration);
71
- animation-timing-function: var(--sbb-header-transition-timing);
72
- }
73
- :host(:is(:is(:state(shadow),[state--shadow]), :is(:state(has-visible-focus-within),[state--has-visible-focus-within]):is(:state(fixed),[state--fixed]))) .sbb-header {
74
- box-shadow: var(--sbb-header-box-shadow);
75
- }
76
- @media (forced-colors: active) {
77
- .sbb-header {
78
- border-block-end: var(--sbb-border-width-1x) solid CanvasText;
79
- }
80
- }
81
-
82
- .sbb-header__wrapper {
83
- display: flex;
84
- align-items: center;
85
- justify-content: flex-start;
86
- height: var(--sbb-header-height);
87
- }
88
- :host(:not([expanded])) .sbb-header__wrapper {
89
- padding-inline: var(--sbb-page-spacing-padding, var(--sbb-layout-base-offset-responsive));
90
- margin-inline: auto;
91
- width: 100%;
92
- }
93
- @media (min-width: calc(90rem)) {
94
- :host(:not([expanded])) .sbb-header__wrapper {
95
- max-width: var(--sbb-spacing-max-width, calc(var(--sbb-layout-base-page-max-width) + 2 * var(--sbb-layout-base-offset-responsive)));
96
- }
97
- }
98
- :host([expanded]) .sbb-header__wrapper {
99
- padding-inline: var(--sbb-page-spacing-padding, var(--sbb-spacing-responsive-xxs));
100
- }
101
-
102
- ::slotted(:is(sbb-header-button, sbb-header-link):first-child) {
103
- margin-inline-start: var(--sbb-header-first-item-margin-inline-start);
104
- }
105
-
106
- ::slotted(*) {
107
- flex: 0 0 auto;
108
- }
109
-
110
- ::slotted(.sbb-header-shrinkable) {
111
- flex-shrink: 1;
112
- min-width: 0;
113
- }
114
-
115
- ::slotted(.sbb-header-spacer) {
116
- flex-grow: 1;
117
- }
118
-
119
- ::slotted(.sbb-header-logo) {
120
- margin-inline-start: var(--sbb-spacing-responsive-s);
121
- }
122
-
123
- ::slotted(a.sbb-header-logo) {
124
- display: block;
125
- padding-block: var(--sbb-spacing-fixed-3x);
126
- position: relative;
127
- outline: none !important;
128
- }
129
-
130
- ::slotted(a.sbb-header-logo)::before {
131
- content: "";
132
- display: block;
133
- position: absolute;
134
- inset: var(--sbb-spacing-fixed-3x) 0;
135
- }
136
-
137
- ::slotted(a.sbb-header-logo:focus-visible)::before {
138
- outline-offset: var(--sbb-focus-outline-offset);
139
- outline: var(--sbb-focus-outline-color) var(--sbb-focus-outline-style, solid) var(--sbb-focus-outline-width);
140
- outline-offset: var(--sbb-spacing-fixed-3x);
141
- border-radius: 1px;
142
- }
143
-
144
- @keyframes show {
145
- from {
146
- transform: translate3d(0, -100%, 0);
147
- }
148
- to {
149
- transform: translate3d(0, 0, 0);
150
- }
151
- }
152
- @keyframes hide {
153
- from {
154
- transform: translate3d(0, 0, 0);
155
- }
156
- to {
157
- transform: translate3d(0, -100%, 0);
158
- }
159
- }`;
160
- //#endregion
161
- //#region src/elements/header/header/header.component.ts
162
- var IS_MENU_OPENED_QUERY = "[aria-controls][aria-expanded='true']";
163
- /**
164
- * It displays a header section for the page.
165
- *
166
- * @slot - Use the unnamed slot to add actions, content and logo to the header.
167
- * @cssprop [--sbb-header-z-index=10] - Can be used to modify the z-index of the header.
168
- * @cssprop [--sbb-header-height=zero-small:var(--sbb-spacing-fixed-14x);large-ultra:var(--sbb-spacing-fixed-24x)] - Can be used to modify height of the header.
169
- */
170
- var SbbHeaderElement = (() => {
171
- let _classSuper = SbbElement;
172
- let _expanded_decorators;
173
- let _expanded_initializers = [];
174
- let _expanded_extraInitializers = [];
175
- let _scrollOrigin_decorators;
176
- let _scrollOrigin_initializers = [];
177
- let _scrollOrigin_extraInitializers = [];
178
- let _hideOnScroll_decorators;
179
- let _hideOnScroll_initializers = [];
180
- let _hideOnScroll_extraInitializers = [];
181
- let _size_decorators;
182
- let _size_initializers = [];
183
- let _size_extraInitializers = [];
184
- let __headerOnTop_decorators;
185
- let __headerOnTop_initializers = [];
186
- let __headerOnTop_extraInitializers = [];
187
- return class SbbHeaderElement extends _classSuper {
188
- static {
189
- const _metadata = typeof Symbol === "function" && Symbol.metadata ? Object.create(_classSuper[Symbol.metadata] ?? null) : void 0;
190
- _expanded_decorators = [forceType(), property({
191
- reflect: true,
192
- type: Boolean
193
- })];
194
- _scrollOrigin_decorators = [idReference(), property({ attribute: "scroll-origin" })];
195
- _hideOnScroll_decorators = [forceType(), property({
196
- attribute: "hide-on-scroll",
197
- reflect: true,
198
- type: Boolean
199
- })];
200
- _size_decorators = [property({ reflect: true })];
201
- __headerOnTop_decorators = [state()];
202
- __esDecorate(this, null, _expanded_decorators, {
203
- kind: "accessor",
204
- name: "expanded",
205
- static: false,
206
- private: false,
207
- access: {
208
- has: (obj) => "expanded" in obj,
209
- get: (obj) => obj.expanded,
210
- set: (obj, value) => {
211
- obj.expanded = value;
212
- }
213
- },
214
- metadata: _metadata
215
- }, _expanded_initializers, _expanded_extraInitializers);
216
- __esDecorate(this, null, _scrollOrigin_decorators, {
217
- kind: "accessor",
218
- name: "scrollOrigin",
219
- static: false,
220
- private: false,
221
- access: {
222
- has: (obj) => "scrollOrigin" in obj,
223
- get: (obj) => obj.scrollOrigin,
224
- set: (obj, value) => {
225
- obj.scrollOrigin = value;
226
- }
227
- },
228
- metadata: _metadata
229
- }, _scrollOrigin_initializers, _scrollOrigin_extraInitializers);
230
- __esDecorate(this, null, _hideOnScroll_decorators, {
231
- kind: "accessor",
232
- name: "hideOnScroll",
233
- static: false,
234
- private: false,
235
- access: {
236
- has: (obj) => "hideOnScroll" in obj,
237
- get: (obj) => obj.hideOnScroll,
238
- set: (obj, value) => {
239
- obj.hideOnScroll = value;
240
- }
241
- },
242
- metadata: _metadata
243
- }, _hideOnScroll_initializers, _hideOnScroll_extraInitializers);
244
- __esDecorate(this, null, _size_decorators, {
245
- kind: "accessor",
246
- name: "size",
247
- static: false,
248
- private: false,
249
- access: {
250
- has: (obj) => "size" in obj,
251
- get: (obj) => obj.size,
252
- set: (obj, value) => {
253
- obj.size = value;
254
- }
255
- },
256
- metadata: _metadata
257
- }, _size_initializers, _size_extraInitializers);
258
- __esDecorate(this, null, __headerOnTop_decorators, {
259
- kind: "accessor",
260
- name: "_headerOnTop",
261
- static: false,
262
- private: false,
263
- access: {
264
- has: (obj) => "_headerOnTop" in obj,
265
- get: (obj) => obj._headerOnTop,
266
- set: (obj, value) => {
267
- obj._headerOnTop = value;
268
- }
269
- },
270
- metadata: _metadata
271
- }, __headerOnTop_initializers, __headerOnTop_extraInitializers);
272
- if (_metadata) Object.defineProperty(this, Symbol.metadata, {
273
- enumerable: true,
274
- configurable: true,
275
- writable: true,
276
- value: _metadata
277
- });
278
- }
279
- static {
280
- this.elementName = "sbb-header";
281
- }
282
- static {
283
- this.styles = [boxSizingStyles, header_default];
284
- }
285
- #expanded_accessor_storage;
286
- /**
287
- * Whether to allow the header content to stretch to full width.
288
- * By default, the content has the appropriate page size.
289
- */
290
- get expanded() {
291
- return this.#expanded_accessor_storage;
292
- }
293
- set expanded(value) {
294
- this.#expanded_accessor_storage = value;
295
- }
296
- #scrollOrigin_accessor_storage;
297
- /**
298
- * The element's id or the element on which the scroll listener is attached.
299
- *
300
- * For attribute usage, provide an id reference.
301
- */
302
- get scrollOrigin() {
303
- return this.#scrollOrigin_accessor_storage;
304
- }
305
- set scrollOrigin(value) {
306
- this.#scrollOrigin_accessor_storage = value;
307
- }
308
- #hideOnScroll_accessor_storage;
309
- /** Whether the header should hide and show on scroll. */
310
- get hideOnScroll() {
311
- return this.#hideOnScroll_accessor_storage;
312
- }
313
- set hideOnScroll(value) {
314
- this.#hideOnScroll_accessor_storage = value;
315
- }
316
- #size_accessor_storage;
317
- /**
318
- * Size of the header, either m or s.
319
- * @default 'm' / 's' (lean)
320
- */
321
- get size() {
322
- return this.#size_accessor_storage;
323
- }
324
- set size(value) {
325
- this.#size_accessor_storage = value;
326
- }
327
- #_headerOnTop_accessor_storage;
328
- get _headerOnTop() {
329
- return this.#_headerOnTop_accessor_storage;
330
- }
331
- set _headerOnTop(value) {
332
- this.#_headerOnTop_accessor_storage = value;
333
- }
334
- constructor() {
335
- super();
336
- this.#expanded_accessor_storage = __runInitializers(this, _expanded_initializers, false);
337
- this.#scrollOrigin_accessor_storage = (__runInitializers(this, _expanded_extraInitializers), __runInitializers(this, _scrollOrigin_initializers, null));
338
- this.#hideOnScroll_accessor_storage = (__runInitializers(this, _scrollOrigin_extraInitializers), __runInitializers(this, _hideOnScroll_initializers, false));
339
- this.#size_accessor_storage = (__runInitializers(this, _hideOnScroll_extraInitializers), __runInitializers(this, _size_initializers, isLean() ? "s" : "m"));
340
- this.#_headerOnTop_accessor_storage = (__runInitializers(this, _size_extraInitializers), __runInitializers(this, __headerOnTop_initializers, true));
341
- this._scrollElement = __runInitializers(this, __headerOnTop_extraInitializers);
342
- this._lastScroll = 0;
343
- this.addController(new SbbFocusVisibleWithinController(this));
344
- }
345
- /** If `hideOnScroll` is set, checks the element to hook the listener on, and possibly add it.*/
346
- connectedCallback() {
347
- super.connectedCallback();
348
- if (this.hasUpdated) this._updateScrollListener();
349
- }
350
- /** Removes the scroll listener, if previously attached. */
351
- disconnectedCallback() {
352
- super.disconnectedCallback();
353
- this._scrollElement = null;
354
- this._scrollEventsController?.abort();
355
- }
356
- requestUpdate(name, oldValue, options) {
357
- super.requestUpdate(name, oldValue, options);
358
- if (!isServer && (!name || name === "scrollOrigin") && this.hasUpdated) this._updateScrollListener();
359
- }
360
- firstUpdated(changedProperties) {
361
- super.firstUpdated(changedProperties);
362
- this._updateScrollListener();
363
- }
364
- _updateScrollListener() {
365
- const scrollElement = this.scrollOrigin ?? document;
366
- if (scrollElement === this._scrollElement) return;
367
- this._scrollEventsController?.abort();
368
- this._scrollElement = scrollElement;
369
- if (!this._scrollElement) return;
370
- this._scrollEventsController = new AbortController();
371
- this._scrollFunction = this._getScrollFunction.bind(this);
372
- this._scrollElement.addEventListener("scroll", this._scrollFunction, {
373
- passive: true,
374
- signal: this._scrollEventsController.signal
375
- });
376
- const currentScroll = this._getCurrentScrollProperty("scrollTop");
377
- this._lastScroll = currentScroll <= 0 ? 0 : currentScroll;
378
- }
379
- /** Returns the correct function to attach on scroll. */
380
- _getScrollFunction() {
381
- return this.hideOnScroll ? this._scrollListener() : this._scrollShadowListener();
382
- }
383
- /** Returns the requested property of the scrollContext. */
384
- _getCurrentScrollProperty(property) {
385
- if (this._scrollElement instanceof Document) return this._scrollElement.documentElement[property] || this._scrollElement.body[property];
386
- return this._scrollElement?.[property] || 0;
387
- }
388
- /**
389
- * Sets the correct value for `scrollTop`, then:
390
- * - apply the shadow if the element/document has been scrolled down;
391
- * - hides the header, remove the shadow and possibly close any open menu on the header if it is not visible anymore;
392
- * - shows the header and re-apply the shadow if the element/document has been scrolled up.
393
- */
394
- _scrollListener() {
395
- const currentScroll = this._getCurrentScrollProperty("scrollTop");
396
- if (this._getCurrentScrollProperty("scrollHeight") - window.innerHeight - currentScroll <= 0) return;
397
- this.toggleState("shadow", currentScroll !== 0);
398
- if (currentScroll > this.offsetHeight && currentScroll > 0 && this._lastScroll < currentScroll) this._closeOpenOverlays();
399
- if (currentScroll > this.offsetHeight * 2) {
400
- this._headerOnTop = false;
401
- if (currentScroll > 0 && this._lastScroll < currentScroll) ["shadow", "visible"].forEach((name) => this.internals.states.delete(name));
402
- else [
403
- "fixed",
404
- "shadow",
405
- "animated",
406
- "visible"
407
- ].forEach((name) => this.internals.states.add(name));
408
- } else {
409
- if (currentScroll === 0) this._headerOnTop = true;
410
- if (this._headerOnTop) [
411
- "shadow",
412
- "animated",
413
- "fixed",
414
- "visible"
415
- ].forEach((name) => this.internals.states.delete(name));
416
- }
417
- this._lastScroll = currentScroll <= 0 ? 0 : currentScroll;
418
- }
419
- /** Apply the shadow if the element/document has been scrolled down. */
420
- _scrollShadowListener() {
421
- this.toggleState("shadow", this._getCurrentScrollProperty("scrollTop") !== 0);
422
- }
423
- _closeOpenOverlays() {
424
- if (this.matches(":is(:state(has-visible-focus-within),[state--has-visible-focus-within])")) return;
425
- const overlayTriggers = Array.from(this.querySelectorAll(IS_MENU_OPENED_QUERY));
426
- for (const overlayTrigger of overlayTriggers) {
427
- const overlayId = overlayTrigger.getAttribute("aria-controls");
428
- const overlay = document.getElementById(overlayId);
429
- if (typeof overlay?.close === "function") overlay.close();
430
- }
431
- }
432
- render() {
433
- return html`
434
- <header class="sbb-header">
435
- <div class="sbb-header__wrapper">
436
- <slot></slot>
437
- </div>
438
- </header>
439
- `;
440
- }
441
- };
442
- })();
443
- //#endregion
444
- export { SbbHeaderElement as t };
445
-
446
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"header.component-DEw_zC-n.js","names":[],"sources":["../../../src/elements/header/header/header.scss?lit&inline","../../../src/elements/header/header/header.component.ts"],"sourcesContent":["@use '../../core/styles' as sbb;\n\n:host {\n  --sbb-logo-height: #{sbb.px-to-rem-build(16)};\n  --sbb-signet-height: #{sbb.px-to-rem-build(16)};\n  --sbb-header-transition-duration: var(\n    --sbb-disable-animation-duration,\n    var(--sbb-animation-duration-6x)\n  );\n\n  @include sbb.mq($from: large) {\n    --sbb-logo-height: #{sbb.px-to-rem-build(20)};\n    --sbb-signet-height: #{sbb.px-to-rem-build(20)};\n  }\n\n  display: block;\n\n  // Setting the height here reserves the space for the header which will else be lost with fixed position.\n  height: var(--sbb-header-height);\n}\n\n:host([size='s']) {\n  ::slotted(:is(sbb-header-button, sbb-header-link)) {\n    --sbb-header-action-min-height: var(--sbb-size-element-xs);\n    --sbb-header-action-padding-inline: var(--sbb-spacing-fixed-4x);\n  }\n}\n\n:host([hide-on-scroll]) {\n  --sbb-header-position: absolute;\n}\n\n:host([hide-on-scroll]:state(fixed)) {\n  --sbb-header-position: fixed;\n  --sbb-header-animation-name: hide;\n  --sbb-header-transform: translate3d(0, -100%, 0);\n  --sbb-header-inset-inline-end: var(--sbb-scrollbar-width, 0);\n}\n\n:host([hide-on-scroll]:state(fixed):state(animated)) {\n  // Hide transition\n  --sbb-header-transition-timing: cubic-bezier(0.4, 0, 1, 1);\n}\n\n:host([hide-on-scroll]:state(fixed):state(visible)) {\n  --sbb-header-animation-name: show;\n  --sbb-header-transform: translate3d(0, 0, 0);\n\n  // Show transition\n  --sbb-header-transition-timing: cubic-bezier(0, 0, 0.2, 1);\n}\n\n:host([hide-on-scroll]:state(fixed):state(has-visible-focus-within)) {\n  --sbb-header-transition-duration: 0;\n  --sbb-header-transform: translate3d(0, 0, 0);\n}\n\n.sbb-header {\n  position: var(--sbb-header-position);\n  inset: 0 var(--sbb-header-inset-inline-end) auto 0;\n  background: var(--sbb-header-background);\n  z-index: var(--sbb-header-z-index, 10);\n  transform: var(--sbb-header-transform);\n  transition: {\n    property: box-shadow;\n    duration: var(--sbb-header-transition-duration);\n    timing-function: var(--sbb-header-transition-timing);\n  }\n  animation: {\n    name: var(--sbb-header-animation-name);\n    duration: var(--sbb-header-transition-duration);\n    timing-function: var(--sbb-header-transition-timing);\n  }\n\n  :host(:is(:state(shadow), :state(has-visible-focus-within):state(fixed))) & {\n    box-shadow: var(--sbb-header-box-shadow);\n  }\n\n  @include sbb.if-forced-colors {\n    border-block-end: var(--sbb-border-width-1x) solid CanvasText;\n  }\n}\n\n.sbb-header__wrapper {\n  display: flex;\n  align-items: center;\n  justify-content: flex-start;\n  height: var(--sbb-header-height);\n\n  :host(:not([expanded])) & {\n    @include sbb.page-spacing;\n  }\n\n  :host([expanded]) & {\n    @include sbb.page-spacing-expanded;\n  }\n}\n\n// Fix left offset if first element of the header is a sbb-header-button/sbb-header-link.\n// The value of the offset is calculated inside sbb-header-button/sbb-header-link styles.\n::slotted(:is(sbb-header-button, sbb-header-link):first-child) {\n  margin-inline-start: var(--sbb-header-first-item-margin-inline-start);\n}\n\n::slotted(*) {\n  flex: 0 0 auto;\n}\n\n::slotted(.sbb-header-shrinkable) {\n  flex-shrink: 1;\n  min-width: 0;\n}\n\n::slotted(.sbb-header-spacer) {\n  flex-grow: 1;\n}\n\n::slotted(.sbb-header-logo) {\n  margin-inline-start: var(--sbb-spacing-responsive-s);\n}\n\n// Apply padding and outline to possible slotted link in logo slot\n::slotted(a.sbb-header-logo) {\n  display: block;\n  padding-block: var(--sbb-spacing-fixed-3x);\n  position: relative;\n\n  // Use !important here to not interfere with Firefox focus ring definition\n  // which appears in normalize CSS of several frameworks.\n  outline: none !important;\n}\n\n// To show the correct outline, we need to span a hidden element filling the link but without the outline-offset.\n::slotted(a.sbb-header-logo)::before {\n  content: '';\n  display: block;\n  position: absolute;\n  inset: var(--sbb-spacing-fixed-3x) 0;\n}\n\n::slotted(a.sbb-header-logo:focus-visible)::before {\n  @include sbb.focus-outline;\n\n  outline-offset: var(--sbb-spacing-fixed-3x);\n\n  // As if the outline has an offset, the border radius increases, we set it to the smallest possible border-radius.\n  border-radius: 1px;\n}\n\n@keyframes show {\n  from {\n    transform: translate3d(0, -100%, 0);\n  }\n\n  to {\n    transform: translate3d(0, 0, 0);\n  }\n}\n\n@keyframes hide {\n  from {\n    transform: translate3d(0, 0, 0);\n  }\n\n  to {\n    transform: translate3d(0, -100%, 0);\n  }\n}\n","import {\n  type CSSResultGroup,\n  html,\n  isServer,\n  type PropertyDeclaration,\n  type PropertyValues,\n  type TemplateResult,\n} from 'lit';\nimport { property, state } from 'lit/decorators.js';\n\nimport { SbbFocusVisibleWithinController } from '../../core/a11y.ts';\nimport { SbbElement } from '../../core/base-elements.ts';\nimport { forceType, idReference } from '../../core/decorators.ts';\nimport { isLean } from '../../core/dom.ts';\nimport { boxSizingStyles } from '../../core/styles.ts';\n\nimport style from './header.scss?lit&inline';\n\nconst IS_MENU_OPENED_QUERY = \"[aria-controls][aria-expanded='true']\";\n\n/**\n * It displays a header section for the page.\n *\n * @slot - Use the unnamed slot to add actions, content and logo to the header.\n * @cssprop [--sbb-header-z-index=10] - Can be used to modify the z-index of the header.\n * @cssprop [--sbb-header-height=zero-small:var(--sbb-spacing-fixed-14x);large-ultra:var(--sbb-spacing-fixed-24x)] - Can be used to modify height of the header.\n */\nexport class SbbHeaderElement extends SbbElement {\n  public static override readonly elementName: string = 'sbb-header';\n  public static override styles: CSSResultGroup = [boxSizingStyles, style];\n\n  /**\n   * Whether to allow the header content to stretch to full width.\n   * By default, the content has the appropriate page size.\n   */\n  @forceType()\n  @property({ reflect: true, type: Boolean })\n  public accessor expanded: boolean = false;\n\n  /**\n   * The element's id or the element on which the scroll listener is attached.\n   *\n   * For attribute usage, provide an id reference.\n   */\n  @idReference()\n  @property({ attribute: 'scroll-origin' })\n  public accessor scrollOrigin: HTMLElement | null = null;\n\n  /** Whether the header should hide and show on scroll. */\n  @forceType()\n  @property({ attribute: 'hide-on-scroll', reflect: true, type: Boolean })\n  public accessor hideOnScroll: boolean = false;\n\n  /**\n   * Size of the header, either m or s.\n   * @default 'm' / 's' (lean)\n   */\n  @property({ reflect: true }) public accessor size: 'm' | 's' = isLean() ? 's' : 'm';\n\n  @state() private accessor _headerOnTop = true;\n\n  private _scrollElement: HTMLElement | Document | null | undefined;\n  private _scrollEventsController!: AbortController;\n  private _scrollFunction: (() => void) | undefined;\n  private _lastScroll = 0;\n\n  public constructor() {\n    super();\n    this.addController(new SbbFocusVisibleWithinController(this));\n  }\n\n  /** If `hideOnScroll` is set, checks the element to hook the listener on, and possibly add it.*/\n  public override connectedCallback(): void {\n    super.connectedCallback();\n    if (this.hasUpdated) {\n      this._updateScrollListener();\n    }\n  }\n\n  /** Removes the scroll listener, if previously attached. */\n  public override disconnectedCallback(): void {\n    super.disconnectedCallback();\n    this._scrollElement = null;\n    this._scrollEventsController?.abort();\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 === 'scrollOrigin') && this.hasUpdated) {\n      this._updateScrollListener();\n    }\n  }\n\n  protected override firstUpdated(changedProperties: PropertyValues<this>): void {\n    super.firstUpdated(changedProperties);\n    this._updateScrollListener();\n  }\n\n  private _updateScrollListener(): void {\n    const scrollElement = this.scrollOrigin ?? document;\n    if (scrollElement === this._scrollElement) {\n      return;\n    }\n\n    this._scrollEventsController?.abort();\n    this._scrollElement = scrollElement;\n    if (!this._scrollElement) {\n      return;\n    }\n    this._scrollEventsController = new AbortController();\n    this._scrollFunction = this._getScrollFunction.bind(this);\n    this._scrollElement.addEventListener('scroll', this._scrollFunction, {\n      passive: true,\n      signal: this._scrollEventsController.signal,\n    });\n\n    const currentScroll = this._getCurrentScrollProperty('scrollTop');\n    // `currentScroll` can be negative, e.g. on mobile; this is not allowed.\n    this._lastScroll = currentScroll <= 0 ? 0 : currentScroll;\n  }\n\n  /** Returns the correct function to attach on scroll. */\n  private _getScrollFunction(): void {\n    return this.hideOnScroll ? this._scrollListener() : this._scrollShadowListener();\n  }\n\n  /** Returns the requested property of the scrollContext. */\n  private _getCurrentScrollProperty(property: 'scrollTop' | 'scrollHeight'): number {\n    if (this._scrollElement instanceof Document) {\n      return this._scrollElement.documentElement[property] || this._scrollElement.body[property];\n    }\n    return this._scrollElement?.[property] || 0;\n  }\n\n  /**\n   * Sets the correct value for `scrollTop`, then:\n   * - apply the shadow if the element/document has been scrolled down;\n   * - hides the header, remove the shadow and possibly close any open menu on the header if it is not visible anymore;\n   * - shows the header and re-apply the shadow if the element/document has been scrolled up.\n   */\n  private _scrollListener(): void {\n    const currentScroll = this._getCurrentScrollProperty('scrollTop');\n\n    // Whether the scroll view is bouncing past the edge of content and back again.\n    if (this._getCurrentScrollProperty('scrollHeight') - window.innerHeight - currentScroll <= 0) {\n      return;\n    }\n\n    this.toggleState('shadow', currentScroll !== 0);\n\n    // Close open overlays when scrolling down if the header is scrolled out of sight.\n    if (\n      currentScroll > this.offsetHeight &&\n      currentScroll > 0 &&\n      this._lastScroll < currentScroll\n    ) {\n      this._closeOpenOverlays();\n    }\n    // Check if the header is scrolled out of sight, scroll position > header height * 2.\n    if (currentScroll > this.offsetHeight * 2) {\n      this._headerOnTop = false;\n      if (currentScroll > 0 && this._lastScroll < currentScroll) {\n        // Scrolling down\n        ['shadow', 'visible'].forEach((name) => this.internals.states.delete(name));\n      } else {\n        // Scrolling up\n        ['fixed', 'shadow', 'animated', 'visible'].forEach((name) =>\n          this.internals.states.add(name),\n        );\n      }\n    } else {\n      // Check if header in its original position, scroll position < header height.\n      // Reset header behavior when scroll hits top of the page, on scroll position = 0.\n      if (currentScroll === 0) {\n        this._headerOnTop = true;\n      }\n      if (this._headerOnTop) {\n        ['shadow', 'animated', 'fixed', 'visible'].forEach((name) =>\n          this.internals.states.delete(name),\n        );\n      }\n    }\n    // `currentScroll` can be negative, e.g. on mobile; this is not allowed.\n    this._lastScroll = currentScroll <= 0 ? 0 : currentScroll;\n  }\n\n  /** Apply the shadow if the element/document has been scrolled down. */\n  private _scrollShadowListener(): void {\n    this.toggleState('shadow', this._getCurrentScrollProperty('scrollTop') !== 0);\n  }\n\n  private _closeOpenOverlays(): void {\n    if (this.matches(':state(has-visible-focus-within)')) {\n      return;\n    }\n    const overlayTriggers: HTMLElement[] = Array.from(\n      this.querySelectorAll(IS_MENU_OPENED_QUERY) as NodeListOf<HTMLElement>,\n    );\n    for (const overlayTrigger of overlayTriggers) {\n      const overlayId: string = overlayTrigger.getAttribute('aria-controls')!;\n      const overlay = document.getElementById(overlayId) as HTMLElement & { close: () => void };\n      if (typeof overlay?.close === 'function') {\n        overlay.close();\n      }\n    }\n  }\n\n  protected override render(): TemplateResult {\n    return html`\n      <header class=\"sbb-header\">\n        <div class=\"sbb-header__wrapper\">\n          <slot></slot>\n        </div>\n      </header>\n    `;\n  }\n}\n\ndeclare global {\n  interface HTMLElementTagNameMap {\n    // eslint-disable-next-line @typescript-eslint/naming-convention\n    'sbb-header': SbbHeaderElement;\n  }\n}\n"],"mappings":"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"}