@sbb-esta/lyne-elements 3.13.0 → 3.13.1
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/carousel/carousel/carousel.component.d.ts +2 -1
- package/carousel/carousel/carousel.component.d.ts.map +1 -1
- package/carousel/carousel/carousel.component.js +32 -27
- package/carousel/carousel-list/carousel-list.component.d.ts +1 -1
- package/carousel/carousel-list/carousel-list.component.d.ts.map +1 -1
- package/carousel/carousel-list/carousel-list.component.js +26 -31
- package/custom-elements.json +13 -11
- package/development/carousel/carousel/carousel.component.d.ts +2 -1
- package/development/carousel/carousel/carousel.component.d.ts.map +1 -1
- package/development/carousel/carousel/carousel.component.js +19 -8
- package/development/carousel/carousel-list/carousel-list.component.d.ts +1 -1
- package/development/carousel/carousel-list/carousel-list.component.d.ts.map +1 -1
- package/development/carousel/carousel-list/carousel-list.component.js +7 -14
- package/package.json +1 -1
|
@@ -21,7 +21,8 @@ export declare class SbbCarouselElement extends SbbCarouselElement_base {
|
|
|
21
21
|
firstUpdated(_changedProperties: PropertyValues): void;
|
|
22
22
|
disconnectedCallback(): void;
|
|
23
23
|
private _handleSlotchange;
|
|
24
|
-
private
|
|
24
|
+
private _setupPaginator;
|
|
25
|
+
private _scrollAtPaginatorChange;
|
|
25
26
|
protected render(): TemplateResult;
|
|
26
27
|
}
|
|
27
28
|
declare global {
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"carousel.component.d.ts","sourceRoot":"","sources":["../../../../src/elements/carousel/carousel/carousel.component.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,cAAc,EAAE,MAAM,uBAAuB,CAAC;AAC5D,OAAO,KAAK,EAAE,cAAc,EAAE,cAAc,EAAE,MAAM,KAAK,CAAC;AAC1D,OAAO,EAAQ,UAAU,EAAE,MAAM,KAAK,CAAC;AAmBvC,OAAO,6BAA6B,CAAC;;AAIrC;;;;GAIG;AACH,qBAEM,kBAAmB,SAAQ,uBAAoC;IACnE,OAAuB,MAAM,EAAE,cAAc,CAAS;IAEtD;;OAEG;IACH,SAEgB,MAAM,EAAE,OAAO,CAAS;IAExC,OAAO,CAAC,UAAU,CAA2C;IAC7D,OAAO,CAAC,gBAAgB,CAAgC;IACxD,OAAO,CAAC,SAAS,CAAmC;IACpD,OAAO,CAAC,8BAA8B,CAAM;;IAwB5B,iBAAiB,IAAI,IAAI;
|
|
1
|
+
{"version":3,"file":"carousel.component.d.ts","sourceRoot":"","sources":["../../../../src/elements/carousel/carousel/carousel.component.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,cAAc,EAAE,MAAM,uBAAuB,CAAC;AAC5D,OAAO,KAAK,EAAE,cAAc,EAAE,cAAc,EAAE,MAAM,KAAK,CAAC;AAC1D,OAAO,EAAQ,UAAU,EAAE,MAAM,KAAK,CAAC;AAmBvC,OAAO,6BAA6B,CAAC;;AAIrC;;;;GAIG;AACH,qBAEM,kBAAmB,SAAQ,uBAAoC;IACnE,OAAuB,MAAM,EAAE,cAAc,CAAS;IAEtD;;OAEG;IACH,SAEgB,MAAM,EAAE,OAAO,CAAS;IAExC,OAAO,CAAC,UAAU,CAA2C;IAC7D,OAAO,CAAC,gBAAgB,CAAgC;IACxD,OAAO,CAAC,SAAS,CAAmC;IACpD,OAAO,CAAC,8BAA8B,CAAM;;IAwB5B,iBAAiB,IAAI,IAAI;IASzB,YAAY,CAAC,kBAAkB,EAAE,cAAc,GAAG,IAAI;IAQtD,oBAAoB,IAAI,IAAI;IAK5C,OAAO,CAAC,iBAAiB;IAyBzB,OAAO,CAAC,eAAe;IAgBvB,OAAO,CAAC,wBAAwB;cAwBb,MAAM,IAAI,cAAc;CAU5C;AAED,OAAO,CAAC,MAAM,CAAC;IACb,UAAU,qBAAqB;QAE7B,cAAc,EAAE,kBAAkB,CAAC;KACpC;CACF"}
|
|
@@ -1,24 +1,24 @@
|
|
|
1
|
-
var
|
|
1
|
+
var p = (a) => {
|
|
2
2
|
throw TypeError(a);
|
|
3
3
|
};
|
|
4
|
-
var
|
|
5
|
-
var
|
|
4
|
+
var _ = (a, t, s) => t.has(a) || p("Cannot " + s);
|
|
5
|
+
var v = (a, t, s) => (_(a, t, "read from private field"), s ? s.call(a) : t.get(a)), m = (a, t, s) => t.has(a) ? p("Cannot add the same private member more than once") : t instanceof WeakSet ? t.add(a) : t.set(a, s), b = (a, t, s, r) => (_(a, t, "write to private field"), r ? r.call(a, s) : t.set(a, s), s);
|
|
6
6
|
import { __esDecorate as w, __runInitializers as c } from "tslib";
|
|
7
7
|
import { css as y, LitElement as x, html as P } from "lit";
|
|
8
8
|
import { customElement as S, property as C } from "lit/decorators.js";
|
|
9
9
|
import { SbbLanguageController as I } from "../../core/controllers/language-controller.js";
|
|
10
|
-
import { forceType as
|
|
11
|
-
import { i18nNextSlide as
|
|
10
|
+
import { forceType as k } from "../../core/decorators.js";
|
|
11
|
+
import { i18nNextSlide as L, i18nPreviousSlide as q, i18nSlide as A, i18nCarouselArrowsNavigationHint as E } from "../../core/i18n/i18n.js";
|
|
12
12
|
import { SbbElementInternalsMixin as B } from "../../core/mixins.js";
|
|
13
13
|
import "../../screen-reader-only.js";
|
|
14
14
|
const z = y`*,:before,:after{box-sizing:border-box}:host{--sbb-carousel-border-radius: var(--sbb-border-radius-4x);--sbb-carousel-background-color: var(--sbb-background-color-1);display:inline-block;overflow:hidden;border-radius:var(--sbb-carousel-border-radius) var(--sbb-carousel-border-radius) 0 0}:host([shadow]){box-shadow:var(--sbb-shadow-elevation-level-9-shadow-2-offset-x) var(--sbb-shadow-elevation-level-9-shadow-2-offset-y) var(--sbb-shadow-elevation-level-9-shadow-2-blur) var(--sbb-shadow-elevation-level-9-shadow-2-spread) var(--sbb-shadow-color-hard-2),var(--sbb-shadow-elevation-level-9-shadow-1-offset-x) var(--sbb-shadow-elevation-level-9-shadow-1-offset-y) var(--sbb-shadow-elevation-level-9-shadow-1-blur) var(--sbb-shadow-elevation-level-9-shadow-1-spread) var(--sbb-shadow-color-hard-1);border-radius:var(--sbb-carousel-border-radius)}:host(:not([shadow])) ::slotted(sbb-carousel-list){--sbb-carousel-inherit-border-radius: var(--sbb-carousel-border-radius)}.sbb-carousel{display:flex;flex-direction:column;justify-content:center;align-items:center;background-color:var(--sbb-carousel-background-color)}::slotted(sbb-compact-paginator){padding:var(--sbb-spacing-responsive-xs)}`;
|
|
15
15
|
let W = (() => {
|
|
16
|
-
var n,
|
|
17
|
-
let a = [S("sbb-carousel")],
|
|
18
|
-
return
|
|
16
|
+
var n, i;
|
|
17
|
+
let a = [S("sbb-carousel")], t, s = [], r, h = B(x), u, g = [], f = [];
|
|
18
|
+
return i = class extends h {
|
|
19
19
|
constructor() {
|
|
20
20
|
super();
|
|
21
|
-
|
|
21
|
+
m(this, n);
|
|
22
22
|
b(this, n, c(this, g, !1)), this._paginator = (c(this, f), null), this._abortController = null, this._language = new I(this), this._requestedPageIndexByPaginator = -1, this.addEventListener?.("show", (e) => {
|
|
23
23
|
this._requestedPageIndexByPaginator !== -1 && this._requestedPageIndexByPaginator !== e.detail.index || (this._paginator && e.detail.index !== this._paginator.pageIndex && (this._paginator.pageIndex = e.detail.index), this._requestedPageIndexByPaginator = -1);
|
|
24
24
|
});
|
|
@@ -27,13 +27,13 @@ let W = (() => {
|
|
|
27
27
|
* Used to display a box-shadow around the component.
|
|
28
28
|
*/
|
|
29
29
|
get shadow() {
|
|
30
|
-
return
|
|
30
|
+
return v(this, n);
|
|
31
31
|
}
|
|
32
32
|
set shadow(e) {
|
|
33
33
|
b(this, n, e);
|
|
34
34
|
}
|
|
35
35
|
connectedCallback() {
|
|
36
|
-
super.connectedCallback(), this.internals.role = "region", this.internals.ariaLabel = "carousel";
|
|
36
|
+
super.connectedCallback(), this.internals.role = "region", this.internals.ariaLabel = "carousel", this._setupPaginator();
|
|
37
37
|
}
|
|
38
38
|
firstUpdated(e) {
|
|
39
39
|
super.firstUpdated(e), this.internals.ariaDescribedByElements = [
|
|
@@ -44,20 +44,25 @@ let W = (() => {
|
|
|
44
44
|
super.disconnectedCallback(), this._abortController?.abort();
|
|
45
45
|
}
|
|
46
46
|
_handleSlotchange() {
|
|
47
|
-
const e = Array.from(this.children).find((
|
|
48
|
-
if (!e || !
|
|
47
|
+
const e = Array.from(this.children).find((d) => d.localName === "sbb-compact-paginator"), o = Array.from(this.children).find((d) => d.localName === "sbb-carousel-list");
|
|
48
|
+
if (!e || !o)
|
|
49
49
|
return;
|
|
50
|
-
const
|
|
51
|
-
|
|
50
|
+
const l = o.querySelectorAll("sbb-carousel-item");
|
|
51
|
+
l && l.length > 0 && (e.length = l.length, e.pageSize = 1), e.accessibilityNextPageLabel ||= L[this._language.current], e.accessibilityPreviousPageLabel ||= q[this._language.current], e.accessibilityPageLabel ||= A[this._language.current], e !== this._paginator && (this._paginator = e, this._setupPaginator());
|
|
52
|
+
}
|
|
53
|
+
_setupPaginator() {
|
|
54
|
+
this._abortController?.abort(), this._paginator && (this._abortController = new AbortController(), this._paginator.addEventListener("click", () => this._scrollAtPaginatorChange(), {
|
|
52
55
|
signal: this._abortController.signal
|
|
53
|
-
})
|
|
56
|
+
}));
|
|
54
57
|
}
|
|
55
|
-
|
|
56
|
-
this.
|
|
57
|
-
|
|
58
|
-
|
|
59
|
-
|
|
60
|
-
|
|
58
|
+
_scrollAtPaginatorChange() {
|
|
59
|
+
if (!this._paginator)
|
|
60
|
+
return;
|
|
61
|
+
this._requestedPageIndexByPaginator = this._paginator.pageIndex;
|
|
62
|
+
const e = this.querySelector("sbb-carousel-list");
|
|
63
|
+
if (e) {
|
|
64
|
+
const l = e.querySelectorAll("sbb-carousel-item")[this._paginator.pageIndex].offsetLeft - this.offsetLeft;
|
|
65
|
+
(e.clientWidth <= 100 || Math.abs(e.scrollLeft - l) > 50) && e.scrollTo({ left: l });
|
|
61
66
|
}
|
|
62
67
|
}
|
|
63
68
|
render() {
|
|
@@ -70,12 +75,12 @@ let W = (() => {
|
|
|
70
75
|
</div>
|
|
71
76
|
`;
|
|
72
77
|
}
|
|
73
|
-
}, n = new WeakMap(),
|
|
78
|
+
}, n = new WeakMap(), r = i, (() => {
|
|
74
79
|
const e = typeof Symbol == "function" && Symbol.metadata ? Object.create(h[Symbol.metadata] ?? null) : void 0;
|
|
75
|
-
u = [
|
|
76
|
-
|
|
77
|
-
} }, metadata: e }, g, f), w(null,
|
|
78
|
-
})(),
|
|
80
|
+
u = [k(), C({ reflect: !0, type: Boolean })], w(i, null, u, { kind: "accessor", name: "shadow", static: !1, private: !1, access: { has: (o) => "shadow" in o, get: (o) => o.shadow, set: (o, l) => {
|
|
81
|
+
o.shadow = l;
|
|
82
|
+
} }, metadata: e }, g, f), w(null, t = { value: r }, a, { kind: "class", name: r.name, metadata: e }, null, s), r = t.value, e && Object.defineProperty(r, Symbol.metadata, { enumerable: !0, configurable: !0, writable: !0, value: e });
|
|
83
|
+
})(), i.styles = z, c(r, s), r;
|
|
79
84
|
})();
|
|
80
85
|
export {
|
|
81
86
|
W as SbbCarouselElement
|
|
@@ -12,7 +12,7 @@ export declare class SbbCarouselListElement extends SbbCarouselListElement_base
|
|
|
12
12
|
private _observedCarouselItems;
|
|
13
13
|
private _beforeShowObserver;
|
|
14
14
|
private _showObserver;
|
|
15
|
-
private
|
|
15
|
+
private _resizeObserverController;
|
|
16
16
|
constructor();
|
|
17
17
|
/** Gets the slotted items. */
|
|
18
18
|
private _carouselItems;
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"carousel-list.component.d.ts","sourceRoot":"","sources":["../../../../src/elements/carousel/carousel-list/carousel-list.component.ts"],"names":[],"mappings":"
|
|
1
|
+
{"version":3,"file":"carousel-list.component.d.ts","sourceRoot":"","sources":["../../../../src/elements/carousel/carousel-list/carousel-list.component.ts"],"names":[],"mappings":"AAEA,OAAO,EAAE,KAAK,cAAc,EAAQ,UAAU,EAAE,KAAK,cAAc,EAAE,MAAM,KAAK,CAAC;;AAcjF;;;;GAIG;AACH,qBAEM,sBAAuB,SAAQ,2BAAoC;IACvE,OAAuB,MAAM,EAAE,cAAc,CAAS;IAEtD,OAAO,CAAC,aAAa,CAAK;IAC1B,OAAO,CAAC,SAAS,CAAmC;IACpD,OAAO,CAAC,sBAAsB,CAAgC;IAE9D,OAAO,CAAC,mBAAmB,CAgBxB;IAEH,OAAO,CAAC,aAAa,CAwBlB;IAEH,OAAO,CAAC,yBAAyB,CAG9B;;IAQH,8BAA8B;IAC9B,OAAO,CAAC,cAAc;IAItB,OAAO,CAAC,iBAAiB;IAoBzB;;;OAGG;IACH,OAAO,CAAC,eAAe;IAyBvB,OAAO,CAAC,UAAU;IAwBF,iBAAiB,IAAI,IAAI;cAOtB,MAAM,IAAI,cAAc;CAG5C;AAED,OAAO,CAAC,MAAM,CAAC;IACb,UAAU,qBAAqB;QAE7B,mBAAmB,EAAE,sBAAsB,CAAC;KAC7C;CACF"}
|
|
@@ -1,38 +1,39 @@
|
|
|
1
1
|
import { __esDecorate as b, __runInitializers as d } from "tslib";
|
|
2
|
-
import { IntersectionController as
|
|
3
|
-
import {
|
|
2
|
+
import { IntersectionController as c } from "@lit-labs/observers/intersection-controller.js";
|
|
3
|
+
import { ResizeController as f } from "@lit-labs/observers/resize-controller.js";
|
|
4
|
+
import { css as m, LitElement as _, html as v } from "lit";
|
|
4
5
|
import { customElement as g } from "lit/decorators.js";
|
|
5
6
|
import { isArrowKeyPressed as I } from "../../core/a11y.js";
|
|
6
|
-
import { SbbLanguageController as
|
|
7
|
-
import { i18nCarouselItemAriaLabel as
|
|
7
|
+
import { SbbLanguageController as p } from "../../core/controllers.js";
|
|
8
|
+
import { i18nCarouselItemAriaLabel as w } from "../../core/i18n.js";
|
|
8
9
|
import { SbbElementInternalsMixin as x } from "../../core/mixins.js";
|
|
9
|
-
const y =
|
|
10
|
+
const y = m`*,:before,:after{box-sizing:border-box}:host{--sbb-carousel-list-border-radius: var(--sbb-border-radius-4x);display:flex;height:var(--sbb-carousel-list-height);width:var(--sbb-carousel-list-width, 0);overflow-x:auto;scroll-snap-type:x mandatory;scroll-behavior:smooth;scrollbar-width:none;outline:none;border-radius:var(--sbb-carousel-inherit-border-radius)}:host(:focus-visible):before{outline-offset:var(--sbb-focus-outline-offset);outline:var(--sbb-focus-outline-color) var(--sbb-focus-outline-style, solid) var(--sbb-focus-outline-width);content:"";position:absolute;height:inherit;width:inherit;border-start-start-radius:var(--sbb-carousel-list-border-radius);border-start-end-radius:var(--sbb-carousel-list-border-radius);border-end-start-radius:var(--sbb-carousel-inherit-border-radius);border-end-end-radius:var(--sbb-carousel-inherit-border-radius)}`;
|
|
10
11
|
let H = (() => {
|
|
11
|
-
var
|
|
12
|
-
let u = [g("sbb-carousel-list")],
|
|
13
|
-
return
|
|
12
|
+
var o;
|
|
13
|
+
let u = [g("sbb-carousel-list")], n, a = [], s, l = x(_);
|
|
14
|
+
return o = class extends l {
|
|
14
15
|
constructor() {
|
|
15
|
-
super(), this._currentIndex = 0, this._language = new
|
|
16
|
+
super(), this._currentIndex = 0, this._language = new p(this), this._observedCarouselItems = [], this._beforeShowObserver = new c(this, {
|
|
16
17
|
target: null,
|
|
17
18
|
callback: (e) => {
|
|
18
19
|
e.filter((r) => r.isIntersecting && r.target !== this).forEach((r) => {
|
|
19
|
-
const
|
|
20
|
-
|
|
21
|
-
detail: { index: this._carouselItems().findIndex((h) => h ===
|
|
20
|
+
const i = r.target;
|
|
21
|
+
i.dispatchEvent(new CustomEvent("beforeshow", {
|
|
22
|
+
detail: { index: this._carouselItems().findIndex((h) => h === i) },
|
|
22
23
|
bubbles: !0,
|
|
23
24
|
composed: !0
|
|
24
25
|
}));
|
|
25
26
|
});
|
|
26
27
|
},
|
|
27
28
|
config: { threshold: 0.01, root: this, rootMargin: "100% 0% 100% 0%" }
|
|
28
|
-
}), this._showObserver = new
|
|
29
|
+
}), this._showObserver = new c(this, {
|
|
29
30
|
target: null,
|
|
30
31
|
callback: (e) => {
|
|
31
32
|
for (const t of e) {
|
|
32
33
|
if (t.target === this)
|
|
33
34
|
continue;
|
|
34
35
|
const r = t.target;
|
|
35
|
-
t.isIntersecting ? (r.ariaHidden = null, this._currentIndex = this._carouselItems().findIndex((
|
|
36
|
+
t.isIntersecting ? (r.ariaHidden = null, this._currentIndex = this._carouselItems().findIndex((i) => i === r), r.dispatchEvent(new CustomEvent("show", {
|
|
36
37
|
detail: { index: this._currentIndex },
|
|
37
38
|
bubbles: !0,
|
|
38
39
|
composed: !0
|
|
@@ -40,15 +41,9 @@ let H = (() => {
|
|
|
40
41
|
}
|
|
41
42
|
},
|
|
42
43
|
config: { threshold: 0.99, root: this, rootMargin: "100% 0% 100% 0%" }
|
|
43
|
-
}), this.
|
|
44
|
-
|
|
45
|
-
|
|
46
|
-
t.intersectionRatio > 0 && (this._readDimensions(), this._firstVisibleIntersectionController.unobserve(this));
|
|
47
|
-
});
|
|
48
|
-
},
|
|
49
|
-
config: {
|
|
50
|
-
root: _ ? null : document?.documentElement
|
|
51
|
-
}
|
|
44
|
+
}), this._resizeObserverController = new f(this, {
|
|
45
|
+
skipInitial: !0,
|
|
46
|
+
callback: () => this._readDimensions()
|
|
52
47
|
}), this.addEventListener?.("keydown", (e) => this._onKeyDown(e));
|
|
53
48
|
}
|
|
54
49
|
/** Gets the slotted items. */
|
|
@@ -61,7 +56,7 @@ let H = (() => {
|
|
|
61
56
|
});
|
|
62
57
|
const e = this._carouselItems();
|
|
63
58
|
e.forEach((t, r) => {
|
|
64
|
-
t.ariaLabel ||=
|
|
59
|
+
t.ariaLabel ||= w(r + 1, e.length)[this._language.current], t.ariaHidden = r === this._currentIndex ? null : "true";
|
|
65
60
|
}), this._readDimensions();
|
|
66
61
|
}
|
|
67
62
|
/**
|
|
@@ -75,15 +70,15 @@ let H = (() => {
|
|
|
75
70
|
const t = e[0];
|
|
76
71
|
t.clientHeight > 0 && this.style.setProperty("--sbb-carousel-list-height", `${t.clientHeight}px`), t.clientWidth > 0 && (this.style.setProperty("--sbb-carousel-list-width", `${t.clientWidth}px`), e.forEach((r) => {
|
|
77
72
|
this._beforeShowObserver.observe(r), this._showObserver.observe(r);
|
|
78
|
-
}), this._observedCarouselItems = e);
|
|
73
|
+
}), this._observedCarouselItems = e, this._resizeObserverController.unobserve(this));
|
|
79
74
|
}
|
|
80
75
|
_onKeyDown(e) {
|
|
81
76
|
if (!I(e))
|
|
82
77
|
return;
|
|
83
78
|
e.preventDefault();
|
|
84
79
|
let t = this._currentIndex;
|
|
85
|
-
const r = e.key === "ArrowUp" || e.key === "ArrowLeft",
|
|
86
|
-
r ? t = Math.max(0, this._currentIndex - 1) :
|
|
80
|
+
const r = e.key === "ArrowUp" || e.key === "ArrowLeft", i = e.key === "ArrowDown" || e.key === "ArrowRight";
|
|
81
|
+
r ? t = Math.max(0, this._currentIndex - 1) : i && (t = Math.min(this._carouselItems().length - 1, this._currentIndex + 1)), t !== this._currentIndex && (this._currentIndex = t, this.scrollTo({
|
|
87
82
|
left: this._carouselItems()[this._currentIndex].offsetLeft - this.offsetLeft
|
|
88
83
|
}));
|
|
89
84
|
}
|
|
@@ -93,10 +88,10 @@ let H = (() => {
|
|
|
93
88
|
render() {
|
|
94
89
|
return v`<slot @slotchange=${this._handleSlotchange}></slot>`;
|
|
95
90
|
}
|
|
96
|
-
}, s =
|
|
97
|
-
const e = typeof Symbol == "function" && Symbol.metadata ? Object.create(
|
|
98
|
-
b(null,
|
|
99
|
-
})(),
|
|
91
|
+
}, s = o, (() => {
|
|
92
|
+
const e = typeof Symbol == "function" && Symbol.metadata ? Object.create(l[Symbol.metadata] ?? null) : void 0;
|
|
93
|
+
b(null, n = { value: s }, u, { kind: "class", name: s.name, metadata: e }, null, a), s = n.value, e && Object.defineProperty(s, Symbol.metadata, { enumerable: !0, configurable: !0, writable: !0, value: e });
|
|
94
|
+
})(), o.styles = y, d(s, a), s;
|
|
100
95
|
})();
|
|
101
96
|
export {
|
|
102
97
|
H as SbbCarouselListElement
|
package/custom-elements.json
CHANGED
|
@@ -39229,21 +39229,23 @@
|
|
|
39229
39229
|
},
|
|
39230
39230
|
{
|
|
39231
39231
|
"kind": "method",
|
|
39232
|
-
"name": "
|
|
39232
|
+
"name": "_setupPaginator",
|
|
39233
39233
|
"privacy": "private",
|
|
39234
39234
|
"return": {
|
|
39235
39235
|
"type": {
|
|
39236
39236
|
"text": "void"
|
|
39237
39237
|
}
|
|
39238
|
-
}
|
|
39239
|
-
|
|
39240
|
-
|
|
39241
|
-
|
|
39242
|
-
|
|
39243
|
-
|
|
39244
|
-
|
|
39238
|
+
}
|
|
39239
|
+
},
|
|
39240
|
+
{
|
|
39241
|
+
"kind": "method",
|
|
39242
|
+
"name": "_scrollAtPaginatorChange",
|
|
39243
|
+
"privacy": "private",
|
|
39244
|
+
"return": {
|
|
39245
|
+
"type": {
|
|
39246
|
+
"text": "void"
|
|
39245
39247
|
}
|
|
39246
|
-
|
|
39248
|
+
}
|
|
39247
39249
|
},
|
|
39248
39250
|
{
|
|
39249
39251
|
"kind": "method",
|
|
@@ -39477,9 +39479,9 @@
|
|
|
39477
39479
|
},
|
|
39478
39480
|
{
|
|
39479
39481
|
"kind": "field",
|
|
39480
|
-
"name": "
|
|
39482
|
+
"name": "_resizeObserverController",
|
|
39481
39483
|
"privacy": "private",
|
|
39482
|
-
"default": "new
|
|
39484
|
+
"default": "new ResizeController(this, { skipInitial: true, callback: () => this._readDimensions(), })"
|
|
39483
39485
|
},
|
|
39484
39486
|
{
|
|
39485
39487
|
"kind": "method",
|
|
@@ -21,7 +21,8 @@ export declare class SbbCarouselElement extends SbbCarouselElement_base {
|
|
|
21
21
|
firstUpdated(_changedProperties: PropertyValues): void;
|
|
22
22
|
disconnectedCallback(): void;
|
|
23
23
|
private _handleSlotchange;
|
|
24
|
-
private
|
|
24
|
+
private _setupPaginator;
|
|
25
|
+
private _scrollAtPaginatorChange;
|
|
25
26
|
protected render(): TemplateResult;
|
|
26
27
|
}
|
|
27
28
|
declare global {
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"carousel.component.d.ts","sourceRoot":"","sources":["../../../../../src/elements/carousel/carousel/carousel.component.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,cAAc,EAAE,MAAM,uBAAuB,CAAC;AAC5D,OAAO,KAAK,EAAE,cAAc,EAAE,cAAc,EAAE,MAAM,KAAK,CAAC;AAC1D,OAAO,EAAQ,UAAU,EAAE,MAAM,KAAK,CAAC;AAmBvC,OAAO,6BAA6B,CAAC;;AAIrC;;;;GAIG;AACH,qBAEM,kBAAmB,SAAQ,uBAAoC;IACnE,OAAuB,MAAM,EAAE,cAAc,CAAS;IAEtD;;OAEG;IACH,SAEgB,MAAM,EAAE,OAAO,CAAS;IAExC,OAAO,CAAC,UAAU,CAA2C;IAC7D,OAAO,CAAC,gBAAgB,CAAgC;IACxD,OAAO,CAAC,SAAS,CAAmC;IACpD,OAAO,CAAC,8BAA8B,CAAM;;IAwB5B,iBAAiB,IAAI,IAAI;
|
|
1
|
+
{"version":3,"file":"carousel.component.d.ts","sourceRoot":"","sources":["../../../../../src/elements/carousel/carousel/carousel.component.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,cAAc,EAAE,MAAM,uBAAuB,CAAC;AAC5D,OAAO,KAAK,EAAE,cAAc,EAAE,cAAc,EAAE,MAAM,KAAK,CAAC;AAC1D,OAAO,EAAQ,UAAU,EAAE,MAAM,KAAK,CAAC;AAmBvC,OAAO,6BAA6B,CAAC;;AAIrC;;;;GAIG;AACH,qBAEM,kBAAmB,SAAQ,uBAAoC;IACnE,OAAuB,MAAM,EAAE,cAAc,CAAS;IAEtD;;OAEG;IACH,SAEgB,MAAM,EAAE,OAAO,CAAS;IAExC,OAAO,CAAC,UAAU,CAA2C;IAC7D,OAAO,CAAC,gBAAgB,CAAgC;IACxD,OAAO,CAAC,SAAS,CAAmC;IACpD,OAAO,CAAC,8BAA8B,CAAM;;IAwB5B,iBAAiB,IAAI,IAAI;IASzB,YAAY,CAAC,kBAAkB,EAAE,cAAc,GAAG,IAAI;IAQtD,oBAAoB,IAAI,IAAI;IAK5C,OAAO,CAAC,iBAAiB;IAyBzB,OAAO,CAAC,eAAe;IAgBvB,OAAO,CAAC,wBAAwB;cAwBb,MAAM,IAAI,cAAc;CAU5C;AAED,OAAO,CAAC,MAAM,CAAC;IACb,UAAU,qBAAqB;QAE7B,cAAc,EAAE,kBAAkB,CAAC;KACpC;CACF"}
|
|
@@ -91,6 +91,7 @@ let SbbCarouselElement = (() => {
|
|
|
91
91
|
super.connectedCallback();
|
|
92
92
|
this.internals.role = "region";
|
|
93
93
|
this.internals.ariaLabel = "carousel";
|
|
94
|
+
this._setupPaginator();
|
|
94
95
|
}
|
|
95
96
|
firstUpdated(_changedProperties) {
|
|
96
97
|
super.firstUpdated(_changedProperties);
|
|
@@ -117,19 +118,29 @@ let SbbCarouselElement = (() => {
|
|
|
117
118
|
paginator.accessibilityPreviousPageLabel ||= i18nPreviousSlide[this._language.current];
|
|
118
119
|
paginator.accessibilityPageLabel ||= i18nSlide[this._language.current];
|
|
119
120
|
if (paginator !== this._paginator) {
|
|
120
|
-
this._abortController = new AbortController();
|
|
121
|
-
paginator.addEventListener("page", (e) => this._scrollAtPageChange(e), {
|
|
122
|
-
signal: this._abortController.signal
|
|
123
|
-
});
|
|
124
121
|
this._paginator = paginator;
|
|
122
|
+
this._setupPaginator();
|
|
125
123
|
}
|
|
126
124
|
}
|
|
127
|
-
|
|
128
|
-
this.
|
|
125
|
+
_setupPaginator() {
|
|
126
|
+
this._abortController?.abort();
|
|
127
|
+
if (!this._paginator) {
|
|
128
|
+
return;
|
|
129
|
+
}
|
|
130
|
+
this._abortController = new AbortController();
|
|
131
|
+
this._paginator.addEventListener("click", () => this._scrollAtPaginatorChange(), {
|
|
132
|
+
signal: this._abortController.signal
|
|
133
|
+
});
|
|
134
|
+
}
|
|
135
|
+
_scrollAtPaginatorChange() {
|
|
136
|
+
if (!this._paginator) {
|
|
137
|
+
return;
|
|
138
|
+
}
|
|
139
|
+
this._requestedPageIndexByPaginator = this._paginator.pageIndex;
|
|
129
140
|
const list = this.querySelector("sbb-carousel-list");
|
|
130
141
|
if (list) {
|
|
131
142
|
const items = list.querySelectorAll("sbb-carousel-item");
|
|
132
|
-
const offsetLeft = items[
|
|
143
|
+
const offsetLeft = items[this._paginator.pageIndex].offsetLeft - this.offsetLeft;
|
|
133
144
|
if (list.clientWidth <= 100 || Math.abs(list.scrollLeft - offsetLeft) > 50) {
|
|
134
145
|
list.scrollTo({ left: offsetLeft });
|
|
135
146
|
}
|
|
@@ -160,4 +171,4 @@ let SbbCarouselElement = (() => {
|
|
|
160
171
|
export {
|
|
161
172
|
SbbCarouselElement
|
|
162
173
|
};
|
|
163
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"carousel.component.js","sources":["../../../../../src/elements/carousel/carousel/carousel.component.ts"],"sourcesContent":["import type { PropertyValues } from '@lit/reactive-element';\nimport type { CSSResultGroup, TemplateResult } from 'lit';\nimport { html, LitElement } from 'lit';\nimport { customElement, property } from 'lit/decorators.js';\n\nimport { SbbLanguageController } from '../../core/controllers/language-controller.js';\nimport { forceType } from '../../core/decorators.js';\nimport {\n  i18nCarouselArrowsNavigationHint,\n  i18nNextSlide,\n  i18nPreviousSlide,\n  i18nSlide,\n} from '../../core/i18n/i18n.js';\nimport type { SbbPaginatorPageEventDetails } from '../../core/interfaces.js';\nimport { SbbElementInternalsMixin } from '../../core/mixins.js';\nimport type { SbbCompactPaginatorElement } from '../../paginator/compact-paginator/compact-paginator.component.js';\nimport type {\n  SbbCarouselItemElement,\n  SbbCarouselItemEventDetail,\n} from '../carousel-item/carousel-item.component.js';\nimport type { SbbCarouselListElement } from '../carousel-list/carousel-list.component.js';\nimport '../../screen-reader-only.js';\n\nimport style from './carousel.scss?lit&inline';\n\n/**\n * It displays a carousel component.\n *\n * @slot - Use the unnamed slot to add the `sbb-carousel-list` for content and a `sbb-paginator` for controls.\n */\nexport\n@customElement('sbb-carousel')\nclass SbbCarouselElement extends SbbElementInternalsMixin(LitElement) {\n  public static override styles: CSSResultGroup = style;\n\n  /**\n   * Used to display a box-shadow around the component.\n   */\n  @forceType()\n  @property({ reflect: true, type: Boolean })\n  public accessor shadow: boolean = false;\n\n  private _paginator: SbbCompactPaginatorElement | null = null;\n  private _abortController: AbortController | null = null;\n  private _language = new SbbLanguageController(this);\n  private _requestedPageIndexByPaginator = -1;\n\n  public constructor() {\n    super();\n\n    // If the list is scrolled using mouse/keyboard, it keeps the paginator updated.\n    this.addEventListener?.('show', (e: CustomEvent<SbbCarouselItemEventDetail>) => {\n      // We have to give priority to the paginator for the case,\n      // if during an animation the next page is called from the paginator, the paginator is reset.\n      if (\n        this._requestedPageIndexByPaginator !== -1 &&\n        this._requestedPageIndexByPaginator !== e.detail.index\n      ) {\n        return;\n      }\n      if (this._paginator) {\n        if (e.detail.index !== this._paginator.pageIndex) {\n          this._paginator.pageIndex = e.detail.index;\n        }\n      }\n      this._requestedPageIndexByPaginator = -1;\n    });\n  }\n\n  public override connectedCallback(): void {\n    super.connectedCallback();\n\n    this.internals.role = 'region';\n    this.internals.ariaLabel = 'carousel';\n  }\n\n  public override firstUpdated(_changedProperties: PropertyValues): void {\n    super.firstUpdated(_changedProperties);\n\n    this.internals.ariaDescribedByElements = [\n      this.shadowRoot!.querySelector('#sbb-carousel-arrows-navigation-hint')!,\n    ];\n  }\n\n  public override disconnectedCallback(): void {\n    super.disconnectedCallback();\n    this._abortController?.abort();\n  }\n\n  private _handleSlotchange(): void {\n    const paginator: SbbCompactPaginatorElement = Array.from(this.children).find(\n      (el) => el.localName === 'sbb-compact-paginator',\n    ) as SbbCompactPaginatorElement;\n    const list: SbbCarouselListElement = Array.from(this.children).find(\n      (el) => el.localName === 'sbb-carousel-list',\n    ) as SbbCarouselListElement;\n    if (!paginator || !list) {\n      return;\n    }\n    const items = list.querySelectorAll<SbbCarouselItemElement>('sbb-carousel-item');\n    if (items && items.length > 0) {\n      paginator.length = items.length;\n      paginator.pageSize = 1;\n    }\n    paginator.accessibilityNextPageLabel ||= i18nNextSlide[this._language.current];\n    paginator.accessibilityPreviousPageLabel ||= i18nPreviousSlide[this._language.current];\n    paginator.accessibilityPageLabel ||= i18nSlide[this._language.current];\n\n    if (paginator !== this._paginator) {\n      this._abortController = new AbortController();\n      paginator.addEventListener('page', (e) => this._scrollAtPageChange(e), {\n        signal: this._abortController.signal,\n      });\n      this._paginator = paginator;\n    }\n  }\n\n  private _scrollAtPageChange(e: CustomEvent<SbbPaginatorPageEventDetails>): void {\n    this._requestedPageIndexByPaginator = e.detail.pageIndex;\n    const list = this.querySelector<SbbCarouselListElement>('sbb-carousel-list');\n\n    if (list) {\n      const items = list.querySelectorAll<SbbCarouselItemElement>('sbb-carousel-item');\n\n      // As the offsetLeft is always rendered from the viewport boundaries, we need to subtract the offsetLeft\n      // from the carousel to get the offset inside the scroll area.\n      const offsetLeft = items[e.detail.pageIndex].offsetLeft - this.offsetLeft;\n\n      // Prevents redundant scrolling when the paginator updates after a scroll event\n      // by checking the distance between the current and target scroll positions.\n      // (show event is triggered slightly before fully scrolled).\n      if (list.clientWidth <= 100 || Math.abs(list.scrollLeft - offsetLeft) > 50) {\n        list.scrollTo({ left: offsetLeft });\n      }\n    }\n  }\n\n  protected override render(): TemplateResult {\n    return html`\n      <div class=\"sbb-carousel\">\n        <sbb-screen-reader-only id=\"sbb-carousel-arrows-navigation-hint\"\n          >${i18nCarouselArrowsNavigationHint[this._language.current]}</sbb-screen-reader-only\n        >\n        <slot @slotchange=${this._handleSlotchange}></slot>\n      </div>\n    `;\n  }\n}\n\ndeclare global {\n  interface HTMLElementTagNameMap {\n    // eslint-disable-next-line @typescript-eslint/naming-convention\n    'sbb-carousel': SbbCarouselElement;\n  }\n}\n"],"names":[],"mappings":"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"}
|
|
174
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"carousel.component.js","sources":["../../../../../src/elements/carousel/carousel/carousel.component.ts"],"sourcesContent":["import type { PropertyValues } from '@lit/reactive-element';\nimport type { CSSResultGroup, TemplateResult } from 'lit';\nimport { html, LitElement } from 'lit';\nimport { customElement, property } from 'lit/decorators.js';\n\nimport { SbbLanguageController } from '../../core/controllers/language-controller.js';\nimport { forceType } from '../../core/decorators.js';\nimport {\n  i18nCarouselArrowsNavigationHint,\n  i18nNextSlide,\n  i18nPreviousSlide,\n  i18nSlide,\n} from '../../core/i18n/i18n.js';\nimport { SbbElementInternalsMixin } from '../../core/mixins.js';\nimport type { SbbCompactPaginatorElement } from '../../paginator/compact-paginator/compact-paginator.component.js';\nimport type {\n  SbbCarouselItemElement,\n  SbbCarouselItemEventDetail,\n} from '../carousel-item/carousel-item.component.js';\nimport type { SbbCarouselListElement } from '../carousel-list/carousel-list.component.js';\n\nimport '../../screen-reader-only.js';\n\nimport style from './carousel.scss?lit&inline';\n\n/**\n * It displays a carousel component.\n *\n * @slot - Use the unnamed slot to add the `sbb-carousel-list` for content and a `sbb-paginator` for controls.\n */\nexport\n@customElement('sbb-carousel')\nclass SbbCarouselElement extends SbbElementInternalsMixin(LitElement) {\n  public static override styles: CSSResultGroup = style;\n\n  /**\n   * Used to display a box-shadow around the component.\n   */\n  @forceType()\n  @property({ reflect: true, type: Boolean })\n  public accessor shadow: boolean = false;\n\n  private _paginator: SbbCompactPaginatorElement | null = null;\n  private _abortController: AbortController | null = null;\n  private _language = new SbbLanguageController(this);\n  private _requestedPageIndexByPaginator = -1;\n\n  public constructor() {\n    super();\n\n    // If the list is scrolled using mouse/keyboard, it keeps the paginator updated.\n    this.addEventListener?.('show', (e: CustomEvent<SbbCarouselItemEventDetail>) => {\n      // We have to give priority to the paginator for the case,\n      // if during an animation the next page is called from the paginator, the paginator is reset.\n      if (\n        this._requestedPageIndexByPaginator !== -1 &&\n        this._requestedPageIndexByPaginator !== e.detail.index\n      ) {\n        return;\n      }\n      if (this._paginator) {\n        if (e.detail.index !== this._paginator.pageIndex) {\n          this._paginator.pageIndex = e.detail.index;\n        }\n      }\n      this._requestedPageIndexByPaginator = -1;\n    });\n  }\n\n  public override connectedCallback(): void {\n    super.connectedCallback();\n\n    this.internals.role = 'region';\n    this.internals.ariaLabel = 'carousel';\n\n    this._setupPaginator();\n  }\n\n  public override firstUpdated(_changedProperties: PropertyValues): void {\n    super.firstUpdated(_changedProperties);\n\n    this.internals.ariaDescribedByElements = [\n      this.shadowRoot!.querySelector('#sbb-carousel-arrows-navigation-hint')!,\n    ];\n  }\n\n  public override disconnectedCallback(): void {\n    super.disconnectedCallback();\n    this._abortController?.abort();\n  }\n\n  private _handleSlotchange(): void {\n    const paginator: SbbCompactPaginatorElement = Array.from(this.children).find(\n      (el) => el.localName === 'sbb-compact-paginator',\n    ) as SbbCompactPaginatorElement;\n    const list: SbbCarouselListElement = Array.from(this.children).find(\n      (el) => el.localName === 'sbb-carousel-list',\n    ) as SbbCarouselListElement;\n    if (!paginator || !list) {\n      return;\n    }\n    const items = list.querySelectorAll<SbbCarouselItemElement>('sbb-carousel-item');\n    if (items && items.length > 0) {\n      paginator.length = items.length;\n      paginator.pageSize = 1;\n    }\n    paginator.accessibilityNextPageLabel ||= i18nNextSlide[this._language.current];\n    paginator.accessibilityPreviousPageLabel ||= i18nPreviousSlide[this._language.current];\n    paginator.accessibilityPageLabel ||= i18nSlide[this._language.current];\n\n    if (paginator !== this._paginator) {\n      this._paginator = paginator;\n      this._setupPaginator();\n    }\n  }\n\n  private _setupPaginator(): void {\n    this._abortController?.abort();\n\n    if (!this._paginator) {\n      return;\n    }\n    this._abortController = new AbortController();\n\n    // By listening to the click event on the paginator, we can ensure that the change was user triggered.\n    // If we instead use the page event, we would also catch programmatic changes to the paginator\n    // which can cause a loop and wrong timing as we also listen to the show event.\n    this._paginator.addEventListener('click', () => this._scrollAtPaginatorChange(), {\n      signal: this._abortController.signal,\n    });\n  }\n\n  private _scrollAtPaginatorChange(): void {\n    if (!this._paginator) {\n      return;\n    }\n    this._requestedPageIndexByPaginator = this._paginator.pageIndex;\n\n    const list = this.querySelector<SbbCarouselListElement>('sbb-carousel-list');\n\n    if (list) {\n      const items = list.querySelectorAll<SbbCarouselItemElement>('sbb-carousel-item');\n\n      // As the offsetLeft is always rendered from the viewport boundaries, we need to subtract the offsetLeft\n      // from the carousel to get the offset inside the scroll area.\n      const offsetLeft = items[this._paginator.pageIndex].offsetLeft - this.offsetLeft;\n\n      // Prevents redundant scrolling when the paginator updates after a scroll event\n      // by checking the distance between the current and target scroll positions.\n      // (show event is triggered slightly before fully scrolled).\n      if (list.clientWidth <= 100 || Math.abs(list.scrollLeft - offsetLeft) > 50) {\n        list.scrollTo({ left: offsetLeft });\n      }\n    }\n  }\n\n  protected override render(): TemplateResult {\n    return html`\n      <div class=\"sbb-carousel\">\n        <sbb-screen-reader-only id=\"sbb-carousel-arrows-navigation-hint\"\n          >${i18nCarouselArrowsNavigationHint[this._language.current]}</sbb-screen-reader-only\n        >\n        <slot @slotchange=${this._handleSlotchange}></slot>\n      </div>\n    `;\n  }\n}\n\ndeclare global {\n  interface HTMLElementTagNameMap {\n    // eslint-disable-next-line @typescript-eslint/naming-convention\n    'sbb-carousel': SbbCarouselElement;\n  }\n}\n"],"names":[],"mappings":"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"}
|
|
@@ -12,7 +12,7 @@ export declare class SbbCarouselListElement extends SbbCarouselListElement_base
|
|
|
12
12
|
private _observedCarouselItems;
|
|
13
13
|
private _beforeShowObserver;
|
|
14
14
|
private _showObserver;
|
|
15
|
-
private
|
|
15
|
+
private _resizeObserverController;
|
|
16
16
|
constructor();
|
|
17
17
|
/** Gets the slotted items. */
|
|
18
18
|
private _carouselItems;
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"carousel-list.component.d.ts","sourceRoot":"","sources":["../../../../../src/elements/carousel/carousel-list/carousel-list.component.ts"],"names":[],"mappings":"
|
|
1
|
+
{"version":3,"file":"carousel-list.component.d.ts","sourceRoot":"","sources":["../../../../../src/elements/carousel/carousel-list/carousel-list.component.ts"],"names":[],"mappings":"AAEA,OAAO,EAAE,KAAK,cAAc,EAAQ,UAAU,EAAE,KAAK,cAAc,EAAE,MAAM,KAAK,CAAC;;AAcjF;;;;GAIG;AACH,qBAEM,sBAAuB,SAAQ,2BAAoC;IACvE,OAAuB,MAAM,EAAE,cAAc,CAAS;IAEtD,OAAO,CAAC,aAAa,CAAK;IAC1B,OAAO,CAAC,SAAS,CAAmC;IACpD,OAAO,CAAC,sBAAsB,CAAgC;IAE9D,OAAO,CAAC,mBAAmB,CAgBxB;IAEH,OAAO,CAAC,aAAa,CAwBlB;IAEH,OAAO,CAAC,yBAAyB,CAG9B;;IAQH,8BAA8B;IAC9B,OAAO,CAAC,cAAc;IAItB,OAAO,CAAC,iBAAiB;IAoBzB;;;OAGG;IACH,OAAO,CAAC,eAAe;IAyBvB,OAAO,CAAC,UAAU;IAwBF,iBAAiB,IAAI,IAAI;cAOtB,MAAM,IAAI,cAAc;CAG5C;AAED,OAAO,CAAC,MAAM,CAAC;IACb,UAAU,qBAAqB;QAE7B,mBAAmB,EAAE,sBAAsB,CAAC;KAC7C;CACF"}
|
|
@@ -1,6 +1,7 @@
|
|
|
1
1
|
import { __esDecorate, __runInitializers } from "tslib";
|
|
2
2
|
import { IntersectionController } from "@lit-labs/observers/intersection-controller.js";
|
|
3
|
-
import {
|
|
3
|
+
import { ResizeController } from "@lit-labs/observers/resize-controller.js";
|
|
4
|
+
import { css, LitElement, html } from "lit";
|
|
4
5
|
import { customElement } from "lit/decorators.js";
|
|
5
6
|
import { isArrowKeyPressed } from "../../core/a11y.js";
|
|
6
7
|
import { SbbLanguageController } from "../../core/controllers.js";
|
|
@@ -88,18 +89,9 @@ let SbbCarouselListElement = (() => {
|
|
|
88
89
|
},
|
|
89
90
|
config: { threshold: 0.99, root: this, rootMargin: "100% 0% 100% 0%" }
|
|
90
91
|
});
|
|
91
|
-
this.
|
|
92
|
-
|
|
93
|
-
|
|
94
|
-
if (entry.intersectionRatio > 0) {
|
|
95
|
-
this._readDimensions();
|
|
96
|
-
this._firstVisibleIntersectionController.unobserve(this);
|
|
97
|
-
}
|
|
98
|
-
});
|
|
99
|
-
},
|
|
100
|
-
config: {
|
|
101
|
-
root: !isServer ? document?.documentElement : null
|
|
102
|
-
}
|
|
92
|
+
this._resizeObserverController = new ResizeController(this, {
|
|
93
|
+
skipInitial: true,
|
|
94
|
+
callback: () => this._readDimensions()
|
|
103
95
|
});
|
|
104
96
|
this.addEventListener?.("keydown", (e) => this._onKeyDown(e));
|
|
105
97
|
}
|
|
@@ -139,6 +131,7 @@ let SbbCarouselListElement = (() => {
|
|
|
139
131
|
this._showObserver.observe(item);
|
|
140
132
|
});
|
|
141
133
|
this._observedCarouselItems = carouselItems;
|
|
134
|
+
this._resizeObserverController.unobserve(this);
|
|
142
135
|
}
|
|
143
136
|
}
|
|
144
137
|
_onKeyDown(evt) {
|
|
@@ -180,4 +173,4 @@ let SbbCarouselListElement = (() => {
|
|
|
180
173
|
export {
|
|
181
174
|
SbbCarouselListElement
|
|
182
175
|
};
|
|
183
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"carousel-list.component.js","sources":["../../../../../src/elements/carousel/carousel-list/carousel-list.component.ts"],"sourcesContent":["import { IntersectionController } from '@lit-labs/observers/intersection-controller.js';\nimport { type CSSResultGroup, html, isServer, LitElement, type TemplateResult } from 'lit';\nimport { customElement } from 'lit/decorators.js';\n\nimport { isArrowKeyPressed } from '../../core/a11y.js';\nimport { SbbLanguageController } from '../../core/controllers.js';\nimport { i18nCarouselItemAriaLabel } from '../../core/i18n.js';\nimport { SbbElementInternalsMixin } from '../../core/mixins.js';\nimport type {\n  SbbCarouselItemElement,\n  SbbCarouselItemEventDetail,\n} from '../carousel-item/carousel-item.component.js';\n\nimport style from './carousel-list.scss?lit&inline';\n\n/**\n * It displays a list of `sbb-carousel-item` components.\n *\n * @slot - Use the unnamed slot to add `sbb-carousel-item` elements.\n */\nexport\n@customElement('sbb-carousel-list')\nclass SbbCarouselListElement extends SbbElementInternalsMixin(LitElement) {\n  public static override styles: CSSResultGroup = style;\n\n  private _currentIndex = 0;\n  private _language = new SbbLanguageController(this);\n  private _observedCarouselItems: SbbCarouselItemElement[] = [];\n\n  private _beforeShowObserver = new IntersectionController(this, {\n    target: null,\n    callback: (entry) => {\n      const item = entry.filter((e) => e.isIntersecting && e.target !== this);\n      item.forEach((e) => {\n        const target = e.target as SbbCarouselItemElement;\n        target.dispatchEvent(\n          new CustomEvent<SbbCarouselItemEventDetail>('beforeshow', {\n            detail: { index: this._carouselItems().findIndex((e) => e === target) },\n            bubbles: true,\n            composed: true,\n          }),\n        );\n      });\n    },\n    config: { threshold: 0.01, root: this, rootMargin: '100% 0% 100% 0%' },\n  });\n\n  private _showObserver = new IntersectionController(this, {\n    target: null,\n    callback: (entryArr) => {\n      for (const entry of entryArr) {\n        if (entry.target === this) {\n          continue;\n        }\n        const target = entry.target as SbbCarouselItemElement;\n        if (entry.isIntersecting) {\n          target.ariaHidden = null;\n          this._currentIndex = this._carouselItems().findIndex((el) => el === target);\n          target.dispatchEvent(\n            new CustomEvent<SbbCarouselItemEventDetail>('show', {\n              detail: { index: this._currentIndex },\n              bubbles: true,\n              composed: true,\n            }),\n          );\n        } else {\n          target.ariaHidden = 'true';\n        }\n      }\n    },\n    config: { threshold: 0.99, root: this, rootMargin: '100% 0% 100% 0%' },\n  });\n\n  // Observer to read the dimensions of the first item when it becomes visible.\n  private _firstVisibleIntersectionController = new IntersectionController(this, {\n    callback: (entries) => {\n      entries.forEach((entry) => {\n        if (entry.intersectionRatio > 0) {\n          this._readDimensions();\n          this._firstVisibleIntersectionController.unobserve(this);\n        }\n      });\n    },\n    config: {\n      root: !isServer ? document?.documentElement : null,\n    },\n  });\n\n  public constructor() {\n    super();\n\n    this.addEventListener?.('keydown', (e) => this._onKeyDown(e));\n  }\n\n  /** Gets the slotted items. */\n  private _carouselItems(): SbbCarouselItemElement[] {\n    return Array.from(this.querySelectorAll?.('sbb-carousel-item') ?? []);\n  }\n\n  private _handleSlotchange(): void {\n    // In case of removed carousel items, we need to unobserve the current observers.\n    this._observedCarouselItems.forEach((item) => {\n      this._beforeShowObserver.unobserve(item);\n      this._showObserver.unobserve(item);\n    });\n\n    const carouselItems = this._carouselItems();\n\n    // Set the aria-label if not provided\n    carouselItems.forEach((item, index) => {\n      item.ariaLabel ||= i18nCarouselItemAriaLabel(index + 1, carouselItems.length)[\n        this._language.current\n      ];\n      item.ariaHidden = index === this._currentIndex ? null : 'true';\n    });\n\n    this._readDimensions();\n  }\n\n  /**\n   * Reads the dimensions of the first carousel item and sets the CSS properties accordingly.\n   * Should set the dimensions only once, when the first item becomes visible and if the value is non-zero.\n   */\n  private _readDimensions(): void {\n    const carouselItems = this._carouselItems();\n    if (carouselItems.length === 0) {\n      return;\n    }\n\n    const firstItem = carouselItems[0];\n\n    if (firstItem.clientHeight > 0) {\n      this.style.setProperty('--sbb-carousel-list-height', `${firstItem.clientHeight}px`);\n    }\n\n    if (firstItem.clientWidth > 0) {\n      this.style.setProperty('--sbb-carousel-list-width', `${firstItem.clientWidth}px`);\n\n      // We should only observe the items if they have a non-zero width. Otherwise, an unwanted scrolling can happen.\n      carouselItems.forEach((item) => {\n        this._beforeShowObserver.observe(item);\n        this._showObserver.observe(item);\n      });\n      this._observedCarouselItems = carouselItems;\n    }\n  }\n\n  private _onKeyDown(evt: KeyboardEvent): void {\n    if (!isArrowKeyPressed(evt)) {\n      return;\n    }\n    evt.preventDefault();\n\n    let newIndex = this._currentIndex;\n    const isPrev = evt.key === 'ArrowUp' || evt.key === 'ArrowLeft';\n    const isNext = evt.key === 'ArrowDown' || evt.key === 'ArrowRight';\n\n    if (isPrev) {\n      newIndex = Math.max(0, this._currentIndex - 1);\n    } else if (isNext) {\n      newIndex = Math.min(this._carouselItems().length - 1, this._currentIndex + 1);\n    }\n\n    if (newIndex !== this._currentIndex) {\n      this._currentIndex = newIndex;\n      this.scrollTo({\n        left: this._carouselItems()[this._currentIndex].offsetLeft - this.offsetLeft,\n      });\n    }\n  }\n\n  public override connectedCallback(): void {\n    super.connectedCallback();\n\n    this.internals.ariaLive = 'polite';\n    this.internals.ariaAtomic = 'true';\n  }\n\n  protected override render(): TemplateResult {\n    return html`<slot @slotchange=${this._handleSlotchange}></slot>`;\n  }\n}\n\ndeclare global {\n  interface HTMLElementTagNameMap {\n    // eslint-disable-next-line @typescript-eslint/naming-convention\n    'sbb-carousel-list': SbbCarouselListElement;\n  }\n}\n"],"names":["e"],"mappings":";;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;IAsBM,0BAAsB,MAAA;;0BAD3B,cAAc,mBAAmB,CAAC;;;;oBACE,yBAAyB,UAAU;AAA3C,EAAA,mBAAQ,YAAoC;AAAA,IAkEvE,cAAA;AACE,YAAA;AAhEM,WAAA,gBAAgB;AAChB,WAAA,YAAY,IAAI,sBAAsB,IAAI;AAC1C,WAAA,yBAAmD,CAAA;AAEnD,WAAA,sBAAsB,IAAI,uBAAuB,MAAM;AAAA,QAC7D,QAAQ;AAAA,QACR,UAAU,CAAC,UAAS;AAClB,gBAAM,OAAO,MAAM,OAAO,CAAC,MAAM,EAAE,kBAAkB,EAAE,WAAW,IAAI;AACtE,eAAK,QAAQ,CAAC,MAAK;AACjB,kBAAM,SAAS,EAAE;AACjB,mBAAO,cACL,IAAI,YAAwC,cAAc;AAAA,cACxD,QAAQ,EAAE,OAAO,KAAK,eAAA,EAAiB,UAAU,CAACA,OAAMA,OAAM,MAAM,EAAA;AAAA,cACpE,SAAS;AAAA,cACT,UAAU;AAAA,YAAA,CACX,CAAC;AAAA,UAEN,CAAC;AAAA,QACH;AAAA,QACA,QAAQ,EAAE,WAAW,MAAM,MAAM,MAAM,YAAY,kBAAA;AAAA,MAAiB,CACrE;AAEO,WAAA,gBAAgB,IAAI,uBAAuB,MAAM;AAAA,QACvD,QAAQ;AAAA,QACR,UAAU,CAAC,aAAY;AACrB,qBAAW,SAAS,UAAU;AAC5B,gBAAI,MAAM,WAAW,MAAM;AACzB;AAAA,YACF;AACA,kBAAM,SAAS,MAAM;AACrB,gBAAI,MAAM,gBAAgB;AACxB,qBAAO,aAAa;AACpB,mBAAK,gBAAgB,KAAK,eAAA,EAAiB,UAAU,CAAC,OAAO,OAAO,MAAM;AAC1E,qBAAO,cACL,IAAI,YAAwC,QAAQ;AAAA,gBAClD,QAAQ,EAAE,OAAO,KAAK,cAAA;AAAA,gBACtB,SAAS;AAAA,gBACT,UAAU;AAAA,cAAA,CACX,CAAC;AAAA,YAEN,OAAO;AACL,qBAAO,aAAa;AAAA,YACtB;AAAA,UACF;AAAA,QACF;AAAA,QACA,QAAQ,EAAE,WAAW,MAAM,MAAM,MAAM,YAAY,kBAAA;AAAA,MAAiB,CACrE;AAGO,WAAA,sCAAsC,IAAI,uBAAuB,MAAM;AAAA,QAC7E,UAAU,CAAC,YAAW;AACpB,kBAAQ,QAAQ,CAAC,UAAS;AACxB,gBAAI,MAAM,oBAAoB,GAAG;AAC/B,mBAAK,gBAAA;AACL,mBAAK,oCAAoC,UAAU,IAAI;AAAA,YACzD;AAAA,UACF,CAAC;AAAA,QACH;AAAA,QACA,QAAQ;AAAA,UACN,MAAM,CAAC,WAAW,UAAU,kBAAkB;AAAA,QAAA;AAAA,MAC/C,CACF;AAKC,WAAK,mBAAmB,WAAW,CAAC,MAAM,KAAK,WAAW,CAAC,CAAC;AAAA,IAC9D;AAAA;AAAA,IAGQ,iBAAc;AACpB,aAAO,MAAM,KAAK,KAAK,mBAAmB,mBAAmB,KAAK,EAAE;AAAA,IACtE;AAAA,IAEQ,oBAAiB;AAEvB,WAAK,uBAAuB,QAAQ,CAAC,SAAQ;AAC3C,aAAK,oBAAoB,UAAU,IAAI;AACvC,aAAK,cAAc,UAAU,IAAI;AAAA,MACnC,CAAC;AAED,YAAM,gBAAgB,KAAK,eAAA;AAG3B,oBAAc,QAAQ,CAAC,MAAM,UAAS;AACpC,aAAK,cAAc,0BAA0B,QAAQ,GAAG,cAAc,MAAM,EAC1E,KAAK,UAAU,OAAO;AAExB,aAAK,aAAa,UAAU,KAAK,gBAAgB,OAAO;AAAA,MAC1D,CAAC;AAED,WAAK,gBAAA;AAAA,IACP;AAAA;AAAA;AAAA;AAAA;AAAA,IAMQ,kBAAe;AACrB,YAAM,gBAAgB,KAAK,eAAA;AAC3B,UAAI,cAAc,WAAW,GAAG;AAC9B;AAAA,MACF;AAEA,YAAM,YAAY,cAAc,CAAC;AAEjC,UAAI,UAAU,eAAe,GAAG;AAC9B,aAAK,MAAM,YAAY,8BAA8B,GAAG,UAAU,YAAY,IAAI;AAAA,MACpF;AAEA,UAAI,UAAU,cAAc,GAAG;AAC7B,aAAK,MAAM,YAAY,6BAA6B,GAAG,UAAU,WAAW,IAAI;AAGhF,sBAAc,QAAQ,CAAC,SAAQ;AAC7B,eAAK,oBAAoB,QAAQ,IAAI;AACrC,eAAK,cAAc,QAAQ,IAAI;AAAA,QACjC,CAAC;AACD,aAAK,yBAAyB;AAAA,MAChC;AAAA,IACF;AAAA,IAEQ,WAAW,KAAkB;AACnC,UAAI,CAAC,kBAAkB,GAAG,GAAG;AAC3B;AAAA,MACF;AACA,UAAI,eAAA;AAEJ,UAAI,WAAW,KAAK;AACpB,YAAM,SAAS,IAAI,QAAQ,aAAa,IAAI,QAAQ;AACpD,YAAM,SAAS,IAAI,QAAQ,eAAe,IAAI,QAAQ;AAEtD,UAAI,QAAQ;AACV,mBAAW,KAAK,IAAI,GAAG,KAAK,gBAAgB,CAAC;AAAA,MAC/C,WAAW,QAAQ;AACjB,mBAAW,KAAK,IAAI,KAAK,eAAA,EAAiB,SAAS,GAAG,KAAK,gBAAgB,CAAC;AAAA,MAC9E;AAEA,UAAI,aAAa,KAAK,eAAe;AACnC,aAAK,gBAAgB;AACrB,aAAK,SAAS;AAAA,UACZ,MAAM,KAAK,eAAA,EAAiB,KAAK,aAAa,EAAE,aAAa,KAAK;AAAA,QAAA,CACnE;AAAA,MACH;AAAA,IACF;AAAA,IAEgB,oBAAiB;AAC/B,YAAM,kBAAA;AAEN,WAAK,UAAU,WAAW;AAC1B,WAAK,UAAU,aAAa;AAAA,IAC9B;AAAA,IAEmB,SAAM;AACvB,aAAO,yBAAyB,KAAK,iBAAiB;AAAA,IACxD;AAAA;;AA9JF,iBAAA,MAAA,mBAAA,EAAA,OAAA,WAAA,GAAA,kBAAA,EAAA,MAAA,SAAA,MAAA,WAAA,MAAA,UAAA,UAAA,GAAA,MAAA,uBAAA;;;QACyB,GAAA,SAAyB,OAD5C,kBAAA,YAAA,uBAAA,GAAuB;;;"}
|
|
176
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"carousel-list.component.js","sources":["../../../../../src/elements/carousel/carousel-list/carousel-list.component.ts"],"sourcesContent":["import { IntersectionController } from '@lit-labs/observers/intersection-controller.js';\nimport { ResizeController } from '@lit-labs/observers/resize-controller.js';\nimport { type CSSResultGroup, html, LitElement, type TemplateResult } from 'lit';\nimport { customElement } from 'lit/decorators.js';\n\nimport { isArrowKeyPressed } from '../../core/a11y.js';\nimport { SbbLanguageController } from '../../core/controllers.js';\nimport { i18nCarouselItemAriaLabel } from '../../core/i18n.js';\nimport { SbbElementInternalsMixin } from '../../core/mixins.js';\nimport type {\n  SbbCarouselItemElement,\n  SbbCarouselItemEventDetail,\n} from '../carousel-item/carousel-item.component.js';\n\nimport style from './carousel-list.scss?lit&inline';\n\n/**\n * It displays a list of `sbb-carousel-item` components.\n *\n * @slot - Use the unnamed slot to add `sbb-carousel-item` elements.\n */\nexport\n@customElement('sbb-carousel-list')\nclass SbbCarouselListElement extends SbbElementInternalsMixin(LitElement) {\n  public static override styles: CSSResultGroup = style;\n\n  private _currentIndex = 0;\n  private _language = new SbbLanguageController(this);\n  private _observedCarouselItems: SbbCarouselItemElement[] = [];\n\n  private _beforeShowObserver = new IntersectionController(this, {\n    target: null,\n    callback: (entry) => {\n      const item = entry.filter((e) => e.isIntersecting && e.target !== this);\n      item.forEach((e) => {\n        const target = e.target as SbbCarouselItemElement;\n        target.dispatchEvent(\n          new CustomEvent<SbbCarouselItemEventDetail>('beforeshow', {\n            detail: { index: this._carouselItems().findIndex((e) => e === target) },\n            bubbles: true,\n            composed: true,\n          }),\n        );\n      });\n    },\n    config: { threshold: 0.01, root: this, rootMargin: '100% 0% 100% 0%' },\n  });\n\n  private _showObserver = new IntersectionController(this, {\n    target: null,\n    callback: (entryArr) => {\n      for (const entry of entryArr) {\n        if (entry.target === this) {\n          continue;\n        }\n        const target = entry.target as SbbCarouselItemElement;\n        if (entry.isIntersecting) {\n          target.ariaHidden = null;\n          this._currentIndex = this._carouselItems().findIndex((el) => el === target);\n          target.dispatchEvent(\n            new CustomEvent<SbbCarouselItemEventDetail>('show', {\n              detail: { index: this._currentIndex },\n              bubbles: true,\n              composed: true,\n            }),\n          );\n        } else {\n          target.ariaHidden = 'true';\n        }\n      }\n    },\n    config: { threshold: 0.99, root: this, rootMargin: '100% 0% 100% 0%' },\n  });\n\n  private _resizeObserverController = new ResizeController(this, {\n    skipInitial: true,\n    callback: () => this._readDimensions(),\n  });\n\n  public constructor() {\n    super();\n\n    this.addEventListener?.('keydown', (e) => this._onKeyDown(e));\n  }\n\n  /** Gets the slotted items. */\n  private _carouselItems(): SbbCarouselItemElement[] {\n    return Array.from(this.querySelectorAll?.('sbb-carousel-item') ?? []);\n  }\n\n  private _handleSlotchange(): void {\n    // In case of removed carousel items, we need to unobserve the current observers.\n    this._observedCarouselItems.forEach((item) => {\n      this._beforeShowObserver.unobserve(item);\n      this._showObserver.unobserve(item);\n    });\n\n    const carouselItems = this._carouselItems();\n\n    // Set the aria-label if not provided\n    carouselItems.forEach((item, index) => {\n      item.ariaLabel ||= i18nCarouselItemAriaLabel(index + 1, carouselItems.length)[\n        this._language.current\n      ];\n      item.ariaHidden = index === this._currentIndex ? null : 'true';\n    });\n\n    this._readDimensions();\n  }\n\n  /**\n   * Reads the dimensions of the first carousel item and sets the CSS properties accordingly.\n   * Should set the dimensions only once, when the first item becomes visible and if the value is non-zero.\n   */\n  private _readDimensions(): void {\n    const carouselItems = this._carouselItems();\n    if (carouselItems.length === 0) {\n      return;\n    }\n\n    const firstItem = carouselItems[0];\n\n    if (firstItem.clientHeight > 0) {\n      this.style.setProperty('--sbb-carousel-list-height', `${firstItem.clientHeight}px`);\n    }\n\n    if (firstItem.clientWidth > 0) {\n      this.style.setProperty('--sbb-carousel-list-width', `${firstItem.clientWidth}px`);\n\n      // We should only observe the items if they have a non-zero width. Otherwise, an unwanted scrolling can happen.\n      carouselItems.forEach((item) => {\n        this._beforeShowObserver.observe(item);\n        this._showObserver.observe(item);\n      });\n      this._observedCarouselItems = carouselItems;\n      this._resizeObserverController.unobserve(this);\n    }\n  }\n\n  private _onKeyDown(evt: KeyboardEvent): void {\n    if (!isArrowKeyPressed(evt)) {\n      return;\n    }\n    evt.preventDefault();\n\n    let newIndex = this._currentIndex;\n    const isPrev = evt.key === 'ArrowUp' || evt.key === 'ArrowLeft';\n    const isNext = evt.key === 'ArrowDown' || evt.key === 'ArrowRight';\n\n    if (isPrev) {\n      newIndex = Math.max(0, this._currentIndex - 1);\n    } else if (isNext) {\n      newIndex = Math.min(this._carouselItems().length - 1, this._currentIndex + 1);\n    }\n\n    if (newIndex !== this._currentIndex) {\n      this._currentIndex = newIndex;\n      this.scrollTo({\n        left: this._carouselItems()[this._currentIndex].offsetLeft - this.offsetLeft,\n      });\n    }\n  }\n\n  public override connectedCallback(): void {\n    super.connectedCallback();\n\n    this.internals.ariaLive = 'polite';\n    this.internals.ariaAtomic = 'true';\n  }\n\n  protected override render(): TemplateResult {\n    return html`<slot @slotchange=${this._handleSlotchange}></slot>`;\n  }\n}\n\ndeclare global {\n  interface HTMLElementTagNameMap {\n    // eslint-disable-next-line @typescript-eslint/naming-convention\n    'sbb-carousel-list': SbbCarouselListElement;\n  }\n}\n"],"names":["e"],"mappings":"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"}
|