@sbb-esta/lyne-elements 1.14.0 → 1.15.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/autocomplete/autocomplete.d.ts +1 -1
- package/autocomplete/autocomplete.d.ts.map +1 -1
- package/autocomplete-grid/autocomplete-grid/autocomplete-grid.d.ts +1 -1
- package/autocomplete-grid/autocomplete-grid/autocomplete-grid.d.ts.map +1 -1
- package/autocomplete-grid/autocomplete-grid.js +6 -6
- package/autocomplete.js +20 -19
- package/calendar/calendar.d.ts +1 -1
- package/calendar/calendar.d.ts.map +1 -1
- package/calendar.js +175 -178
- package/checkbox/checkbox-panel/checkbox-panel.d.ts +1 -1
- package/checkbox/checkbox-panel/checkbox-panel.d.ts.map +1 -1
- package/checkbox/checkbox-panel.js +5 -5
- package/checkbox/common.js +1 -1
- package/clock/clock.d.ts +1 -1
- package/clock/clock.d.ts.map +1 -1
- package/clock.js +2 -2
- package/container/sticky-bar/sticky-bar.d.ts +29 -2
- package/container/sticky-bar/sticky-bar.d.ts.map +1 -1
- package/container/sticky-bar.js +69 -31
- package/core/controllers/media-matchers-controller.d.ts +32 -0
- package/core/controllers/media-matchers-controller.d.ts.map +1 -0
- package/core/controllers.d.ts +1 -0
- package/core/controllers.d.ts.map +1 -1
- package/core/controllers.js +75 -33
- package/core/dom/breakpoint.d.ts +2 -1
- package/core/dom/breakpoint.d.ts.map +1 -1
- package/core/dom.js +7 -7
- package/core/mixins/form-associated-checkbox-mixin.d.ts.map +1 -1
- package/core/mixins.js +8 -1
- package/core/styles/core/mediaqueries.scss +1 -2
- package/core/styles/mixins/buttons.scss +5 -5
- package/core/testing/event-spy.d.ts +6 -4
- package/core/testing/event-spy.d.ts.map +1 -1
- package/core/testing.js +82 -55
- package/custom-elements.json +650 -74
- package/datepicker/datepicker/datepicker.d.ts.map +1 -1
- package/development/autocomplete/autocomplete.d.ts +1 -1
- package/development/autocomplete/autocomplete.d.ts.map +1 -1
- package/development/autocomplete-grid/autocomplete-grid/autocomplete-grid.d.ts +1 -1
- package/development/autocomplete-grid/autocomplete-grid/autocomplete-grid.d.ts.map +1 -1
- package/development/autocomplete-grid/autocomplete-grid.js +4 -3
- package/development/autocomplete.js +4 -3
- package/development/calendar/calendar.d.ts +1 -1
- package/development/calendar/calendar.d.ts.map +1 -1
- package/development/calendar.js +40 -44
- package/development/checkbox/checkbox-panel/checkbox-panel.d.ts +1 -1
- package/development/checkbox/checkbox-panel/checkbox-panel.d.ts.map +1 -1
- package/development/checkbox/checkbox-panel.js +2 -2
- package/development/checkbox/common.js +2 -2
- package/development/clock/clock.d.ts +1 -1
- package/development/clock/clock.d.ts.map +1 -1
- package/development/clock.js +3 -3
- package/development/container/sticky-bar/sticky-bar.d.ts +29 -2
- package/development/container/sticky-bar/sticky-bar.d.ts.map +1 -1
- package/development/container/sticky-bar.js +149 -36
- package/development/core/controllers/media-matchers-controller.d.ts +32 -0
- package/development/core/controllers/media-matchers-controller.d.ts.map +1 -0
- package/development/core/controllers.d.ts +1 -0
- package/development/core/controllers.d.ts.map +1 -1
- package/development/core/controllers.js +65 -1
- package/development/core/dom/breakpoint.d.ts +2 -1
- package/development/core/dom/breakpoint.d.ts.map +1 -1
- package/development/core/dom.js +2 -2
- package/development/core/mixins/form-associated-checkbox-mixin.d.ts.map +1 -1
- package/development/core/mixins.js +9 -2
- package/development/core/testing/event-spy.d.ts +6 -4
- package/development/core/testing/event-spy.d.ts.map +1 -1
- package/development/core/testing.js +35 -1
- package/development/datepicker/datepicker/datepicker.d.ts.map +1 -1
- package/development/datepicker/datepicker.js +1 -1
- package/development/dialog/dialog.js +2 -2
- package/development/expansion-panel/expansion-panel-header/expansion-panel-header.d.ts +3 -1
- package/development/expansion-panel/expansion-panel-header/expansion-panel-header.d.ts.map +1 -1
- package/development/expansion-panel/expansion-panel-header.js +8 -4
- package/development/file-selector/file-selector.d.ts +5 -0
- package/development/file-selector/file-selector.d.ts.map +1 -1
- package/development/file-selector.js +8 -1
- package/development/flip-card/flip-card/flip-card.d.ts +4 -0
- package/development/flip-card/flip-card/flip-card.d.ts.map +1 -1
- package/development/flip-card/flip-card-details.js +4 -6
- package/development/flip-card/flip-card-summary/flip-card-summary.d.ts +1 -1
- package/development/flip-card/flip-card-summary/flip-card-summary.d.ts.map +1 -1
- package/development/flip-card/flip-card-summary.js +9 -10
- package/development/flip-card/flip-card.js +26 -2
- package/development/form-field/form-field.js +5 -3
- package/development/icon/icon.d.ts +1 -1
- package/development/icon.js +1 -1
- package/development/image/image.d.ts.map +1 -1
- package/development/image.js +3 -19
- package/development/map-container.js +3 -2
- package/development/menu/menu/menu.d.ts +1 -0
- package/development/menu/menu/menu.d.ts.map +1 -1
- package/development/menu/menu.js +15 -11
- package/development/navigation/navigation-section.js +4 -14
- package/development/navigation/navigation.js +2 -7
- package/development/notification.js +3 -2
- package/development/paginator/paginator/paginator.d.ts +1 -5
- package/development/paginator/paginator/paginator.d.ts.map +1 -1
- package/development/paginator/paginator.js +19 -27
- package/development/popover/popover/popover.d.ts.map +1 -1
- package/development/popover/popover.js +9 -16
- package/development/radio-button/radio-button-panel/radio-button-panel.d.ts +1 -1
- package/development/radio-button/radio-button-panel/radio-button-panel.d.ts.map +1 -1
- package/development/radio-button/radio-button-panel.js +2 -2
- package/development/sbb-tokens-BdGhUJjM.js +33 -0
- package/development/select/select.d.ts +6 -5
- package/development/select/select.d.ts.map +1 -1
- package/development/select.js +23 -14
- package/development/slider/slider.d.ts +5 -0
- package/development/slider/slider.d.ts.map +1 -1
- package/development/slider.js +8 -1
- package/development/table/table-wrapper/table-wrapper.d.ts +1 -1
- package/development/table/table-wrapper/table-wrapper.d.ts.map +1 -1
- package/development/table/table-wrapper.js +1 -1
- package/development/timetable-occupancy-icon/timetable-occupancy-icon.d.ts +2 -1
- package/development/timetable-occupancy-icon/timetable-occupancy-icon.d.ts.map +1 -1
- package/development/timetable-occupancy-icon.js +11 -9
- package/development/toggle-check/toggle-check.d.ts +1 -1
- package/development/toggle-check/toggle-check.d.ts.map +1 -1
- package/development/toggle-check.js +2 -2
- package/dialog/dialog.js +1 -1
- package/expansion-panel/expansion-panel-header/expansion-panel-header.d.ts +3 -1
- package/expansion-panel/expansion-panel-header/expansion-panel-header.d.ts.map +1 -1
- package/expansion-panel/expansion-panel-header.js +26 -24
- package/file-selector/file-selector.d.ts +5 -0
- package/file-selector/file-selector.d.ts.map +1 -1
- package/file-selector.js +7 -0
- package/flip-card/flip-card/flip-card.d.ts +4 -0
- package/flip-card/flip-card/flip-card.d.ts.map +1 -1
- package/flip-card/flip-card-details.js +6 -6
- package/flip-card/flip-card-summary/flip-card-summary.d.ts +1 -1
- package/flip-card/flip-card-summary/flip-card-summary.d.ts.map +1 -1
- package/flip-card/flip-card-summary.js +8 -8
- package/flip-card/flip-card.js +56 -42
- package/icon/icon.d.ts +1 -1
- package/image/image.d.ts.map +1 -1
- package/image.js +46 -45
- package/map-container.js +7 -7
- package/menu/menu/menu.d.ts +1 -0
- package/menu/menu/menu.d.ts.map +1 -1
- package/menu/menu.js +45 -41
- package/navigation/navigation-section.js +7 -7
- package/navigation/navigation.js +8 -8
- package/package.json +2 -2
- package/paginator/paginator/paginator.d.ts +1 -5
- package/paginator/paginator/paginator.d.ts.map +1 -1
- package/paginator/paginator.js +50 -52
- package/popover/popover/popover.d.ts.map +1 -1
- package/popover/popover.js +42 -46
- package/radio-button/radio-button-panel/radio-button-panel.d.ts +1 -1
- package/radio-button/radio-button-panel/radio-button-panel.d.ts.map +1 -1
- package/radio-button/radio-button-panel.js +7 -7
- package/sbb-tokens-Dx20OtVg.js +18 -0
- package/select/select.d.ts +6 -5
- package/select/select.d.ts.map +1 -1
- package/select.js +55 -53
- package/slider/slider.d.ts +5 -0
- package/slider/slider.d.ts.map +1 -1
- package/slider.js +12 -5
- package/table/table-wrapper/table-wrapper.d.ts +1 -1
- package/table/table-wrapper/table-wrapper.d.ts.map +1 -1
- package/timetable-occupancy-icon/timetable-occupancy-icon.d.ts +2 -1
- package/timetable-occupancy-icon/timetable-occupancy-icon.d.ts.map +1 -1
- package/timetable-occupancy-icon.js +45 -44
- package/toggle-check/toggle-check.d.ts +1 -1
- package/toggle-check/toggle-check.d.ts.map +1 -1
- package/toggle-check.js +1 -1
|
@@ -7,9 +7,11 @@ var __privateAdd = (obj, member, value) => member.has(obj) ? __typeError("Cannot
|
|
|
7
7
|
var __privateSet = (obj, member, value, setter) => (__accessCheck(obj, member, "write to private field"), setter ? setter.call(obj, value) : member.set(obj, value), value);
|
|
8
8
|
import { __runInitializers, __esDecorate } from "tslib";
|
|
9
9
|
import { IntersectionController } from "@lit-labs/observers/intersection-controller.js";
|
|
10
|
-
import { css,
|
|
10
|
+
import { css, LitElement, html } from "lit";
|
|
11
11
|
import { customElement, property } from "lit/decorators.js";
|
|
12
12
|
import { hostAttributes } from "../core/decorators.js";
|
|
13
|
+
import { EventEmitter } from "../core/eventing.js";
|
|
14
|
+
import { SbbUpdateSchedulerMixin } from "../core/mixins.js";
|
|
13
15
|
const style = css`*,
|
|
14
16
|
::before,
|
|
15
17
|
::after {
|
|
@@ -17,7 +19,10 @@ const style = css`*,
|
|
|
17
19
|
}
|
|
18
20
|
|
|
19
21
|
:host {
|
|
22
|
+
--sbb-sticky-bar-position: sticky;
|
|
20
23
|
--sbb-sticky-bar-padding-block: var(--sbb-spacing-responsive-xs);
|
|
24
|
+
--sbb-sticky-bar-border-radius: var(--sbb-border-radius-8x);
|
|
25
|
+
--sbb-sticky-bar-animation-easing: var(--sbb-animation-easing);
|
|
21
26
|
--sbb-sticky-bar-fade-in-animation-duration: var(
|
|
22
27
|
--sbb-disable-animation-zero-time,
|
|
23
28
|
var(--sbb-animation-duration-5x)
|
|
@@ -26,31 +31,71 @@ const style = css`*,
|
|
|
26
31
|
--sbb-disable-animation-zero-time,
|
|
27
32
|
var(--sbb-animation-duration-2x)
|
|
28
33
|
);
|
|
29
|
-
--sbb-sticky-bar-animation-
|
|
30
|
-
|
|
34
|
+
--sbb-sticky-bar-slide-vertically-animation-duration: var(
|
|
35
|
+
--sbb-disable-animation-zero-duration,
|
|
36
|
+
var(--sbb-animation-duration-4x)
|
|
37
|
+
);
|
|
38
|
+
--sbb-sticky-bar-slide-vertically-animation-easing: ease-out;
|
|
39
|
+
--sbb-sticky-bar-slide-vertically-animation-delay: 0s;
|
|
40
|
+
--sbb-sticky-bar-slide-vertically-animation-name: unset;
|
|
41
|
+
--_sbb-sticky-bar-background-animation-duration: var(
|
|
42
|
+
--sbb-sticky-bar-fade-out-animation-duration
|
|
43
|
+
);
|
|
44
|
+
--_sbb-sticky-bar-intersector-background-color: transparent;
|
|
45
|
+
--_sbb-sticky-bar-forced-colors-border: none;
|
|
31
46
|
display: contents;
|
|
32
47
|
}
|
|
33
48
|
|
|
34
|
-
:host([data-sticking]) {
|
|
49
|
+
:host([data-sticking]:not([data-state=unsticky])) {
|
|
35
50
|
--sbb-sticky-bar-sticky-background-color: var(
|
|
36
51
|
--sbb-container-background-color,
|
|
37
52
|
var(--sbb-color-white)
|
|
38
53
|
);
|
|
54
|
+
--_sbb-sticky-bar-intersector-background-color: var(--sbb-sticky-bar-sticky-background-color);
|
|
55
|
+
--_sbb-sticky-bar-background-animation-duration: var(--sbb-sticky-bar-fade-in-animation-duration);
|
|
56
|
+
}
|
|
57
|
+
@media (forced-colors: active) {
|
|
58
|
+
:host([data-sticking]:not([data-state=unsticky])) {
|
|
59
|
+
--_sbb-sticky-bar-forced-colors-border: var(--sbb-border-width-1x) solid CanvasText;
|
|
60
|
+
}
|
|
39
61
|
}
|
|
40
62
|
|
|
41
|
-
:host([data-sticking][color=white]) {
|
|
63
|
+
:host([data-sticking]:not([data-state=unsticky])[color=white]) {
|
|
42
64
|
--sbb-sticky-bar-sticky-background-color: var(--sbb-color-white);
|
|
43
65
|
}
|
|
44
66
|
|
|
45
|
-
:host([data-sticking][color=milk]) {
|
|
67
|
+
:host([data-sticking]:not([data-state=unsticky])[color=milk]) {
|
|
46
68
|
--sbb-sticky-bar-sticky-background-color: var(--sbb-color-milk);
|
|
47
69
|
}
|
|
48
70
|
|
|
71
|
+
:host(:is([data-sticking]:is([data-slide-vertically], [data-state=sticking], [data-state=unsticking]),
|
|
72
|
+
[data-state=unsticky])) {
|
|
73
|
+
--_sbb-sticky-bar-background-animation-duration: 0s;
|
|
74
|
+
}
|
|
75
|
+
|
|
76
|
+
:host([data-sticking]:is([data-slide-vertically]:not([data-state=unsticky], [data-state=unsticking]),
|
|
77
|
+
[data-state=sticking])) {
|
|
78
|
+
--sbb-sticky-bar-slide-vertically-animation-name: slide-in;
|
|
79
|
+
}
|
|
80
|
+
|
|
81
|
+
:host([data-sticking][data-state=unsticking]) {
|
|
82
|
+
--sbb-sticky-bar-slide-vertically-animation-name: slide-out;
|
|
83
|
+
}
|
|
84
|
+
|
|
85
|
+
:host(:is(:not([data-initialized]), [data-state=unsticky])) {
|
|
86
|
+
--sbb-sticky-bar-position: relative;
|
|
87
|
+
}
|
|
88
|
+
|
|
49
89
|
.sbb-sticky-bar__wrapper {
|
|
50
|
-
position: sticky;
|
|
90
|
+
position: var(--sbb-sticky-bar-position);
|
|
51
91
|
inset-block-end: 0;
|
|
52
92
|
display: block;
|
|
53
93
|
z-index: var(--sbb-sticky-bar-z-index);
|
|
94
|
+
animation-name: var(--sbb-sticky-bar-slide-vertically-animation-name);
|
|
95
|
+
animation-duration: var(--sbb-sticky-bar-slide-vertically-animation-duration);
|
|
96
|
+
animation-timing-function: var(--sbb-sticky-bar-slide-vertically-animation-easing);
|
|
97
|
+
animation-delay: var(--sbb-sticky-bar-slide-vertically-animation-delay);
|
|
98
|
+
animation-fill-mode: backwards;
|
|
54
99
|
}
|
|
55
100
|
.sbb-sticky-bar__wrapper::after, .sbb-sticky-bar__wrapper::before {
|
|
56
101
|
content: "";
|
|
@@ -75,18 +120,10 @@ const style = css`*,
|
|
|
75
120
|
}
|
|
76
121
|
.sbb-sticky-bar__wrapper::after {
|
|
77
122
|
background-color: var(--sbb-sticky-bar-sticky-background-color, transparent);
|
|
78
|
-
transition: background-color var(--sbb-sticky-bar-fade-out-animation-duration) var(--sbb-sticky-bar-animation-easing);
|
|
79
123
|
border-start-start-radius: var(--sbb-sticky-bar-border-radius);
|
|
80
124
|
border-start-end-radius: var(--sbb-sticky-bar-border-radius);
|
|
81
|
-
|
|
82
|
-
:
|
|
83
|
-
transition-duration: var(--sbb-sticky-bar-fade-in-animation-duration);
|
|
84
|
-
}
|
|
85
|
-
@media (forced-colors: active) {
|
|
86
|
-
:host([data-sticking]) .sbb-sticky-bar__wrapper::after {
|
|
87
|
-
border-block-start: var(--sbb-border-width-1x) solid CanvasText;
|
|
88
|
-
border-radius: 0;
|
|
89
|
-
}
|
|
125
|
+
transition: background-color var(--_sbb-sticky-bar-background-animation-duration) var(--sbb-sticky-bar-animation-easing);
|
|
126
|
+
border: var(--_sbb-sticky-bar-forced-colors-border);
|
|
90
127
|
}
|
|
91
128
|
|
|
92
129
|
.sbb-sticky-bar {
|
|
@@ -101,12 +138,11 @@ const style = css`*,
|
|
|
101
138
|
z-index: -1;
|
|
102
139
|
border-start-start-radius: var(--sbb-sticky-bar-border-radius);
|
|
103
140
|
border-start-end-radius: var(--sbb-sticky-bar-border-radius);
|
|
104
|
-
transition: box-shadow var(--
|
|
141
|
+
transition: box-shadow var(--_sbb-sticky-bar-background-animation-duration) var(--sbb-sticky-bar-animation-easing);
|
|
105
142
|
clip-path: polygon(-50% calc(-1 * var(--sbb-shadow-elevation-level-11-shadow-1-blur)), 150% calc(-1 * var(--sbb-shadow-elevation-level-11-shadow-1-blur)), 150% 50%, -50% 50%);
|
|
106
143
|
}
|
|
107
|
-
:host([data-sticking]) .sbb-sticky-bar::before {
|
|
144
|
+
:host([data-sticking]:not([data-state=unsticky])) .sbb-sticky-bar::before {
|
|
108
145
|
box-shadow: var(--sbb-shadow-elevation-level-11-shadow-2-offset-x) var(--sbb-shadow-elevation-level-11-shadow-2-offset-y) var(--sbb-shadow-elevation-level-11-shadow-2-blur) var(--sbb-shadow-elevation-level-11-shadow-2-spread) var(--sbb-shadow-elevation-level-11-soft-2-color), var(--sbb-shadow-elevation-level-11-shadow-1-offset-x) var(--sbb-shadow-elevation-level-11-shadow-1-offset-y) var(--sbb-shadow-elevation-level-11-shadow-1-blur) var(--sbb-shadow-elevation-level-11-shadow-1-spread) var(--sbb-shadow-elevation-level-11-soft-1-color);
|
|
109
|
-
transition-duration: var(--sbb-sticky-bar-fade-in-animation-duration);
|
|
110
146
|
}
|
|
111
147
|
:host(:not([data-expanded])) .sbb-sticky-bar {
|
|
112
148
|
padding-inline: var(--sbb-layout-base-offset-responsive);
|
|
@@ -131,13 +167,26 @@ const style = css`*,
|
|
|
131
167
|
position: absolute;
|
|
132
168
|
width: 100%;
|
|
133
169
|
height: calc(var(--sbb-sticky-bar-bottom-overlapping-height, 0) + 1px);
|
|
134
|
-
background-color:
|
|
170
|
+
background-color: var(--_sbb-sticky-bar-intersector-background-color);
|
|
135
171
|
pointer-events: none;
|
|
136
|
-
transition: background-color var(--
|
|
172
|
+
transition: background-color var(--_sbb-sticky-bar-background-animation-duration) var(--sbb-sticky-bar-animation-easing);
|
|
173
|
+
}
|
|
174
|
+
|
|
175
|
+
@keyframes slide-in {
|
|
176
|
+
from {
|
|
177
|
+
transform: translateY(100%);
|
|
178
|
+
}
|
|
179
|
+
to {
|
|
180
|
+
transform: translateY(0%);
|
|
181
|
+
}
|
|
137
182
|
}
|
|
138
|
-
|
|
139
|
-
|
|
140
|
-
|
|
183
|
+
@keyframes slide-out {
|
|
184
|
+
from {
|
|
185
|
+
transform: translateY(0%);
|
|
186
|
+
}
|
|
187
|
+
to {
|
|
188
|
+
transform: translateY(100%);
|
|
189
|
+
}
|
|
141
190
|
}`;
|
|
142
191
|
let SbbStickyBarElement = (() => {
|
|
143
192
|
var _color_accessor_storage, _a;
|
|
@@ -147,21 +196,24 @@ let SbbStickyBarElement = (() => {
|
|
|
147
196
|
let _classDescriptor;
|
|
148
197
|
let _classExtraInitializers = [];
|
|
149
198
|
let _classThis;
|
|
150
|
-
let _classSuper = LitElement;
|
|
199
|
+
let _classSuper = SbbUpdateSchedulerMixin(LitElement);
|
|
151
200
|
let _color_decorators;
|
|
152
201
|
let _color_initializers = [];
|
|
153
202
|
let _color_extraInitializers = [];
|
|
154
|
-
_a = class extends _classSuper {
|
|
203
|
+
var SbbStickyBarElement2 = (_a = class extends _classSuper {
|
|
155
204
|
constructor() {
|
|
156
205
|
super(...arguments);
|
|
157
206
|
__privateAdd(this, _color_accessor_storage);
|
|
158
207
|
__privateSet(this, _color_accessor_storage, __runInitializers(this, _color_initializers, null));
|
|
159
|
-
this.
|
|
208
|
+
this._willStick = (__runInitializers(this, _color_extraInitializers), new EventEmitter(this, SbbStickyBarElement2.events.willStick));
|
|
209
|
+
this._didStick = new EventEmitter(this, SbbStickyBarElement2.events.didStick);
|
|
210
|
+
this._willUnstick = new EventEmitter(this, SbbStickyBarElement2.events.willUnstick);
|
|
211
|
+
this._didUnstick = new EventEmitter(this, SbbStickyBarElement2.events.didUnstick);
|
|
160
212
|
this._observer = new IntersectionController(this, {
|
|
161
213
|
// Although `this` is observed, we have to postpone observing
|
|
162
214
|
// into firstUpdated() to achieve a correct initial state.
|
|
163
215
|
target: null,
|
|
164
|
-
callback: (entries) => this.
|
|
216
|
+
callback: (entries) => this._detectStickyState(entries[0])
|
|
165
217
|
});
|
|
166
218
|
}
|
|
167
219
|
/** Color of the container, like transparent, white etc. */
|
|
@@ -171,8 +223,17 @@ let SbbStickyBarElement = (() => {
|
|
|
171
223
|
set color(value) {
|
|
172
224
|
__privateSet(this, _color_accessor_storage, value);
|
|
173
225
|
}
|
|
226
|
+
/** The state of the component. */
|
|
227
|
+
set _state(state) {
|
|
228
|
+
this.setAttribute("data-state", state);
|
|
229
|
+
}
|
|
230
|
+
get _state() {
|
|
231
|
+
return this.getAttribute("data-state");
|
|
232
|
+
}
|
|
174
233
|
connectedCallback() {
|
|
175
234
|
super.connectedCallback();
|
|
235
|
+
this._state = "sticky";
|
|
236
|
+
this.startUpdate();
|
|
176
237
|
const container = this.closest("sbb-container");
|
|
177
238
|
if (container) {
|
|
178
239
|
this.toggleAttribute("data-expanded", container.expanded);
|
|
@@ -181,6 +242,10 @@ let SbbStickyBarElement = (() => {
|
|
|
181
242
|
this._observer.observe(this._intersector);
|
|
182
243
|
}
|
|
183
244
|
}
|
|
245
|
+
disconnectedCallback() {
|
|
246
|
+
super.disconnectedCallback();
|
|
247
|
+
this.toggleAttribute("data-initialized", false);
|
|
248
|
+
}
|
|
184
249
|
firstUpdated(changedProperties) {
|
|
185
250
|
super.firstUpdated(changedProperties);
|
|
186
251
|
if (!this._intersector) {
|
|
@@ -189,12 +254,55 @@ let SbbStickyBarElement = (() => {
|
|
|
189
254
|
}
|
|
190
255
|
this._observer.observe(this);
|
|
191
256
|
}
|
|
192
|
-
|
|
193
|
-
|
|
257
|
+
_detectStickyState(entry) {
|
|
258
|
+
var _a2;
|
|
259
|
+
this.toggleAttribute("data-initialized", true);
|
|
260
|
+
const isSticky = !entry.isIntersecting && entry.boundingClientRect.top > 0;
|
|
261
|
+
const intersectorRect = (_a2 = this._intersector) == null ? void 0 : _a2.getBoundingClientRect();
|
|
262
|
+
const stickyBarRect = this.shadowRoot.querySelector(".sbb-sticky-bar__wrapper").getBoundingClientRect();
|
|
263
|
+
const HEIGHT_TOLERANCE = 30;
|
|
264
|
+
this.toggleAttribute("data-slide-vertically", isSticky && this._intersector && Math.abs(intersectorRect.bottom - stickyBarRect.bottom) > HEIGHT_TOLERANCE);
|
|
265
|
+
this.toggleAttribute("data-sticking", isSticky);
|
|
266
|
+
this.completeUpdate();
|
|
267
|
+
}
|
|
268
|
+
/** Animates from normal content flow position to `position: sticky`. */
|
|
269
|
+
stick() {
|
|
270
|
+
if (this._state !== "unsticky" || !this._willStick.emit()) {
|
|
271
|
+
return;
|
|
272
|
+
}
|
|
273
|
+
this._state = "sticking";
|
|
274
|
+
if (!this.hasAttribute("data-sticking")) {
|
|
275
|
+
this._stickyCallback();
|
|
276
|
+
}
|
|
277
|
+
}
|
|
278
|
+
/** Animates `position: sticky` to normal content flow position. */
|
|
279
|
+
unstick() {
|
|
280
|
+
if (this._state !== "sticky" || !this._willUnstick.emit()) {
|
|
281
|
+
return;
|
|
282
|
+
}
|
|
283
|
+
this._state = "unsticking";
|
|
284
|
+
if (!this.hasAttribute("data-sticking")) {
|
|
285
|
+
this._unstickyCallback();
|
|
286
|
+
}
|
|
287
|
+
}
|
|
288
|
+
_stickyCallback() {
|
|
289
|
+
this._state = "sticky";
|
|
290
|
+
this._didStick.emit();
|
|
291
|
+
}
|
|
292
|
+
_unstickyCallback() {
|
|
293
|
+
this._didUnstick.emit();
|
|
294
|
+
this._state = "unsticky";
|
|
295
|
+
}
|
|
296
|
+
_onAnimationEnd(event) {
|
|
297
|
+
if ((this._state === "sticking" || this._state === "sticky") && event.animationName === "slide-in") {
|
|
298
|
+
this._stickyCallback();
|
|
299
|
+
} else if (this._state === "unsticking" && event.animationName === "slide-out") {
|
|
300
|
+
this._unstickyCallback();
|
|
301
|
+
}
|
|
194
302
|
}
|
|
195
303
|
render() {
|
|
196
304
|
return html`
|
|
197
|
-
<div class="sbb-sticky-bar__wrapper">
|
|
305
|
+
<div class="sbb-sticky-bar__wrapper" @animationend=${this._onAnimationEnd}>
|
|
198
306
|
<div class="sbb-sticky-bar">
|
|
199
307
|
<slot></slot>
|
|
200
308
|
</div>
|
|
@@ -209,12 +317,17 @@ let SbbStickyBarElement = (() => {
|
|
|
209
317
|
obj.color = value;
|
|
210
318
|
} }, metadata: _metadata }, _color_initializers, _color_extraInitializers);
|
|
211
319
|
__esDecorate(null, _classDescriptor = { value: _classThis }, _classDecorators, { kind: "class", name: _classThis.name, metadata: _metadata }, null, _classExtraInitializers);
|
|
212
|
-
_classThis = _classDescriptor.value;
|
|
320
|
+
SbbStickyBarElement2 = _classThis = _classDescriptor.value;
|
|
213
321
|
if (_metadata) Object.defineProperty(_classThis, Symbol.metadata, { enumerable: true, configurable: true, writable: true, value: _metadata });
|
|
214
|
-
})(), _a.styles = style,
|
|
215
|
-
|
|
322
|
+
})(), _a.styles = style, _a.events = {
|
|
323
|
+
willStick: "willStick",
|
|
324
|
+
didStick: "didStick",
|
|
325
|
+
willUnstick: "willUnstick",
|
|
326
|
+
didUnstick: "didUnstick"
|
|
327
|
+
}, __runInitializers(_classThis, _classExtraInitializers), _a);
|
|
328
|
+
return SbbStickyBarElement2 = _classThis;
|
|
216
329
|
})();
|
|
217
330
|
export {
|
|
218
331
|
SbbStickyBarElement
|
|
219
332
|
};
|
|
220
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic3RpY2t5LWJhci5qcyIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vc3JjL2VsZW1lbnRzL2NvbnRhaW5lci9zdGlja3ktYmFyL3N0aWNreS1iYXIudHMiXSwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgSW50ZXJzZWN0aW9uQ29udHJvbGxlciB9IGZyb20gJ0BsaXQtbGFicy9vYnNlcnZlcnMvaW50ZXJzZWN0aW9uLWNvbnRyb2xsZXIuanMnO1xuaW1wb3J0IHtcbiAgdHlwZSBDU1NSZXN1bHRHcm91cCxcbiAgaHRtbCxcbiAgTGl0RWxlbWVudCxcbiAgdHlwZSBQcm9wZXJ0eVZhbHVlcyxcbiAgdHlwZSBUZW1wbGF0ZVJlc3VsdCxcbn0gZnJvbSAnbGl0JztcbmltcG9ydCB7IGN1c3RvbUVsZW1lbnQsIHByb3BlcnR5IH0gZnJvbSAnbGl0L2RlY29yYXRvcnMuanMnO1xuXG5pbXBvcnQgeyBob3N0QXR0cmlidXRlcyB9IGZyb20gJy4uLy4uL2NvcmUvZGVjb3JhdG9ycy5qcyc7XG5cbmltcG9ydCBzdHlsZSBmcm9tICcuL3N0aWNreS1iYXIuc2Nzcz9saXQmaW5saW5lJztcblxuLyoqXG4gKiBBIGNvbnRhaW5lciB0aGF0IHN0aWNrcyB0byB0aGUgYm90dG9tIG9mIHRoZSBwYWdlIGlmIHNsb3R0ZWQgaW50byBgc2JiLWNvbnRhaW5lcmAuXG4gKlxuICogQHNsb3QgLSBVc2UgdGhlIHVubmFtZWQgc2xvdCB0byBhZGQgY29udGVudCB0byB0aGUgc3RpY2t5IGJhci5cbiAqIEBjc3Nwcm9wIFstLXNiYi1zdGlja3ktYmFyLXBhZGRpbmctYmxvY2s9dmFyKC0tc2JiLXNwYWNpbmctcmVzcG9uc2l2ZS14cyldIC0gQmxvY2sgcGFkZGluZyBvZiB0aGUgc3RpY2t5IGJhci5cbiAqIEBjc3Nwcm9wIFstLXNiYi1zdGlja3ktYmFyLWJvdHRvbS1vdmVybGFwcGluZy1oZWlnaHQ9MHB4XSAtIERlZmluZSBhbiBhZGRpdGlvbmFsIGFyZWEgd2hlcmVcbiAqIHRoZSBzdGlja3kgYmFyIG92ZXJsYXBzIHRoZSBmb2xsb3dpbmcgY29udGVudCBvbiB0aGUgYm90dG9tLlxuICogVGhpcyBhcmVhIGJlY29tZXMgdmlzaWJsZSB3aGVuIHRoZSBzdGlja3kgYmFyIHRyYW5zaXRpb25zIGZyb20gc3RpY2t5IHRvIHRoZSBub3JtYWwgZG9jdW1lbnQgZmxvdy5cbiAqIEBjc3Nwcm9wIFstLXNiYi1zdGlja3ktYmFyLXotaW5kZXhdIC0gVG8gc3BlY2lmeSBhIGN1c3RvbSBzdGFjayBvcmRlcixcbiAqIHRoZSBgei1pbmRleGAgY2FuIGJlIG92ZXJyaWRkZW4gYnkgZGVmaW5pbmcgdGhpcyBDU1MgdmFyaWFibGUuXG4gKi9cbmV4cG9ydFxuQGN1c3RvbUVsZW1lbnQoJ3NiYi1zdGlja3ktYmFyJylcbkBob3N0QXR0cmlidXRlcyh7XG4gIHNsb3Q6ICdzdGlja3ktYmFyJyxcbn0pXG5jbGFzcyBTYmJTdGlja3lCYXJFbGVtZW50IGV4dGVuZHMgTGl0RWxlbWVudCB7XG4gIHB1YmxpYyBzdGF0aWMgb3ZlcnJpZGUgc3R5bGVzOiBDU1NSZXN1bHRHcm91cCA9IHN0eWxlO1xuXG4gIC8qKiBDb2xvciBvZiB0aGUgY29udGFpbmVyLCBsaWtlIHRyYW5zcGFyZW50LCB3aGl0ZSBldGMuICovXG4gIEBwcm9wZXJ0eSh7IHJlZmxlY3Q6IHRydWUgfSkgcHVibGljIGFjY2Vzc29yIGNvbG9yOiAnd2hpdGUnIHwgJ21pbGsnIHwgbnVsbCA9IG51bGw7XG5cbiAgcHJpdmF0ZSBfaW50ZXJzZWN0b3I/OiBIVE1MU3BhbkVsZW1lbnQ7XG4gIHByaXZhdGUgX29ic2VydmVyID0gbmV3IEludGVyc2VjdGlvbkNvbnRyb2xsZXIodGhpcywge1xuICAgIC8vIEFsdGhvdWdoIGB0aGlzYCBpcyBvYnNlcnZlZCwgd2UgaGF2ZSB0byBwb3N0cG9uZSBvYnNlcnZpbmdcbiAgICAvLyBpbnRvIGZpcnN0VXBkYXRlZCgpIHRvIGFjaGlldmUgYSBjb3JyZWN0IGluaXRpYWwgc3RhdGUuXG4gICAgdGFyZ2V0OiBudWxsLFxuICAgIGNhbGxiYWNrOiAoZW50cmllcykgPT4gdGhpcy5fdG9nZ2xlU2hhZG93VmlzaWJpbGl0eShlbnRyaWVzWzBdKSxcbiAgfSk7XG5cbiAgcHVibGljIG92ZXJyaWRlIGNvbm5lY3RlZENhbGxiYWNrKCk6IHZvaWQge1xuICAgIHN1cGVyLmNvbm5lY3RlZENhbGxiYWNrKCk7XG5cbiAgICBjb25zdCBjb250YWluZXIgPSB0aGlzLmNsb3Nlc3QoJ3NiYi1jb250YWluZXInKTtcbiAgICBpZiAoY29udGFpbmVyKSB7XG4gICAgICB0aGlzLnRvZ2dsZUF0dHJpYnV0ZSgnZGF0YS1leHBhbmRlZCcsIGNvbnRhaW5lci5leHBhbmRlZCk7XG4gICAgfVxuICAgIGlmICh0aGlzLl9pbnRlcnNlY3Rvcikge1xuICAgICAgdGhpcy5fb2JzZXJ2ZXIub2JzZXJ2ZSh0aGlzLl9pbnRlcnNlY3Rvcik7XG4gICAgfVxuICB9XG5cbiAgcHJvdGVjdGVkIG92ZXJyaWRlIGZpcnN0VXBkYXRlZChjaGFuZ2VkUHJvcGVydGllczogUHJvcGVydHlWYWx1ZXM8dGhpcz4pOiB2b2lkIHtcbiAgICBzdXBlci5maXJzdFVwZGF0ZWQoY2hhbmdlZFByb3BlcnRpZXMpO1xuXG4gICAgaWYgKCF0aGlzLl9pbnRlcnNlY3Rvcikge1xuICAgICAgdGhpcy5faW50ZXJzZWN0b3IgPSB0aGlzLnNoYWRvd1Jvb3QhLnF1ZXJ5U2VsZWN0b3IoJy5zYmItc3RpY2t5LWJhcl9faW50ZXJzZWN0b3InKSE7XG4gICAgICB0aGlzLl9vYnNlcnZlci5vYnNlcnZlKHRoaXMuX2ludGVyc2VjdG9yKTtcbiAgICB9XG4gICAgdGhpcy5fb2JzZXJ2ZXIub2JzZXJ2ZSh0aGlzKTtcbiAgfVxuXG4gIHByaXZhdGUgX3RvZ2dsZVNoYWRvd1Zpc2liaWxpdHkoZW50cnk6IEludGVyc2VjdGlvbk9ic2VydmVyRW50cnkpOiB2b2lkIHtcbiAgICB0aGlzLnRvZ2dsZUF0dHJpYnV0ZShcbiAgICAgICdkYXRhLXN0aWNraW5nJyxcbiAgICAgICFlbnRyeS5pc0ludGVyc2VjdGluZyAmJiBlbnRyeS5ib3VuZGluZ0NsaWVudFJlY3QudG9wID4gMCxcbiAgICApO1xuICB9XG5cbiAgcHJvdGVjdGVkIG92ZXJyaWRlIHJlbmRlcigpOiBUZW1wbGF0ZVJlc3VsdCB7XG4gICAgcmV0dXJuIGh0bWxgXG4gICAgICA8ZGl2IGNsYXNzPVwic2JiLXN0aWNreS1iYXJfX3dyYXBwZXJcIj5cbiAgICAgICAgPGRpdiBjbGFzcz1cInNiYi1zdGlja3ktYmFyXCI+XG4gICAgICAgICAgPHNsb3Q+PC9zbG90PlxuICAgICAgICA8L2Rpdj5cbiAgICAgIDwvZGl2PlxuICAgICAgPGRpdiBjbGFzcz1cInNiYi1zdGlja3ktYmFyX19pbnRlcnNlY3RvclwiPjwvZGl2PlxuICAgIGA7XG4gIH1cbn1cblxuZGVjbGFyZSBnbG9iYWwge1xuICBpbnRlcmZhY2UgSFRNTEVsZW1lbnRUYWdOYW1lTWFwIHtcbiAgICAvLyBlc2xpbnQtZGlzYWJsZS1uZXh0LWxpbmUgQHR5cGVzY3JpcHQtZXNsaW50L25hbWluZy1jb252ZW50aW9uXG4gICAgJ3NiYi1zdGlja3ktYmFyJzogU2JiU3RpY2t5QmFyRWxlbWVudDtcbiAgfVxufVxuIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7OztJQThCTSx1QkFBbUIsTUFBQTs7QUFKeEIsTUFBQSxtQkFBQSxDQUFBLGNBQWMsZ0JBQWdCLEdBQzlCLGVBQWU7QUFBQSxJQUNkLE1BQU07QUFBQSxFQUFBLENBQ1AsQ0FBQzs7OztvQkFDZ0M7Ozs7QUFBUixFQUFBLG1CQUFRLFlBQVU7QUFBQTs7QUFJYjtBQUFnQix5QkFBQSx5QkFBQSxrQkFBQSxNQUFBLHFCQUFpQyxJQUFJO0FBRTFFLFdBQUEsZUFBK0Isa0JBQUEsTUFBQSx3QkFBQTtBQUMvQixXQUFBLFlBQVksSUFBSSx1QkFBdUIsTUFBTTtBQUFBO0FBQUE7QUFBQSxRQUduRCxRQUFRO0FBQUEsUUFDUixVQUFVLENBQUMsWUFBWSxLQUFLLHdCQUF3QixRQUFRLENBQUMsQ0FBQztBQUFBLE1BQUEsQ0FDL0Q7QUFBQSxJQUFBO0FBQUE7QUFBQSxJQVI0QixJQUFnQixRQUFzQztBQUFBLGFBQUEsbUJBQUE7QUFBQSxJQUFBO0FBQUEsSUFBdEQsSUFBZ0IsTUFBc0MsT0FBQTtBQUFBLHlCQUFBLHlCQUFBO0FBQUEsSUFBQTtBQUFBLElBVW5FLG9CQUFpQjtBQUMvQixZQUFNLGtCQUFpQjtBQUVqQixZQUFBLFlBQVksS0FBSyxRQUFRLGVBQWU7QUFDOUMsVUFBSSxXQUFXO0FBQ1IsYUFBQSxnQkFBZ0IsaUJBQWlCLFVBQVUsUUFBUTtBQUFBLE1BQUE7QUFFMUQsVUFBSSxLQUFLLGNBQWM7QUFDaEIsYUFBQSxVQUFVLFFBQVEsS0FBSyxZQUFZO0FBQUEsTUFBQTtBQUFBLElBQzFDO0FBQUEsSUFHaUIsYUFBYSxtQkFBdUM7QUFDckUsWUFBTSxhQUFhLGlCQUFpQjtBQUVoQyxVQUFBLENBQUMsS0FBSyxjQUFjO0FBQ3RCLGFBQUssZUFBZSxLQUFLLFdBQVksY0FBYyw4QkFBOEI7QUFDNUUsYUFBQSxVQUFVLFFBQVEsS0FBSyxZQUFZO0FBQUEsTUFBQTtBQUVyQyxXQUFBLFVBQVUsUUFBUSxJQUFJO0FBQUEsSUFBQTtBQUFBLElBR3JCLHdCQUF3QixPQUFnQztBQUN6RCxXQUFBLGdCQUNILGlCQUNBLENBQUMsTUFBTSxrQkFBa0IsTUFBTSxtQkFBbUIsTUFBTSxDQUFDO0FBQUEsSUFBQTtBQUFBLElBSTFDLFNBQU07QUFDaEIsYUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUEsSUFBQTtBQUFBLEtBeENvQjs7QUFBNUIsd0JBQUEsQ0FBQSxTQUFTLEVBQUUsU0FBUyxLQUFNLENBQUEsQ0FBQztBQUFpQixpQkFBQSxJQUFBLE1BQUEsbUJBQUEsRUFBQSxNQUFBLFlBQUEsTUFBQSxTQUFBLFFBQUEsT0FBQSxTQUFBLE9BQUEsUUFBQSxFQUFBLEtBQUEsQ0FBQSxRQUFBLFdBQUEsS0FBQSxLQUFBLENBQUEsUUFBQSxJQUFBLE9BQUEsS0FBQSxDQUFBLEtBQUEsVUFBQTtBQUFBLFVBQUEsUUFBc0M7QUFBQSxTQUFBLFVBQUEsYUFBQSxxQkFBQSx3QkFBQTtBQUpyRixpQkFxREMsTUFBQSxtQkFBQSxFQUFBLE9BQUEsV0FBQSxHQUFBLGtCQUFBLEVBQUEsTUFBQSxTQUFBLE1BQUEsV0FBQSxNQUFBLFVBQUEsVUFBQSxHQUFBLE1BQUEsdUJBQUE7OztRQXBEd0IsR0FBTSxTQUFtQixPQUQ1QyxrQkFBbUIsWUFBQSx1QkFBQSxHQUFDOzs7In0=
|
|
333
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"sticky-bar.js","sources":["../../../../src/elements/container/sticky-bar/sticky-bar.ts"],"sourcesContent":["import { IntersectionController } from '@lit-labs/observers/intersection-controller.js';\nimport {\n  type CSSResultGroup,\n  html,\n  LitElement,\n  type PropertyValues,\n  type TemplateResult,\n} from 'lit';\nimport { customElement, property } from 'lit/decorators.js';\n\nimport { hostAttributes } from '../../core/decorators.js';\nimport { EventEmitter } from '../../core/eventing.js';\nimport { SbbUpdateSchedulerMixin } from '../../core/mixins.js';\n\nimport style from './sticky-bar.scss?lit&inline';\n\ntype StickyState = 'sticking' | 'sticky' | 'unsticking' | 'unsticky';\n\n/**\n * A container that sticks to the bottom of the page if slotted into `sbb-container`.\n *\n * @slot - Use the unnamed slot to add content to the sticky bar.\n * @event {CustomEvent<void>} willStick - Emits when the animation from normal content flow to `position: sticky` starts. Can be canceled.\n * @event {CustomEvent<void>} didStick - Emits when the animation from normal content flow to `position: sticky` ends.\n * @event {CustomEvent<void>} willUnstick - Emits when the animation from `position: sticky` to normal content flow starts. Can be canceled.\n * @event {CustomEvent<void>} didUnstick - Emits when the animation from `position: sticky` to normal content flow ends.\n * @cssprop [--sbb-sticky-bar-padding-block=var(--sbb-spacing-responsive-xs)] - Block padding of the sticky bar.\n * @cssprop [--sbb-sticky-bar-bottom-overlapping-height=0px] - Define an additional area where\n * the sticky bar overlaps the following content on the bottom.\n * This area becomes visible when the sticky bar transitions from sticky to the normal document flow.\n * @cssprop [--sbb-sticky-bar-z-index] - To specify a custom stack order,\n * the `z-index` can be overridden by defining this CSS variable.\n */\nexport\n@customElement('sbb-sticky-bar')\n@hostAttributes({\n  slot: 'sticky-bar',\n})\nclass SbbStickyBarElement extends SbbUpdateSchedulerMixin(LitElement) {\n  public static override styles: CSSResultGroup = style;\n\n  public static readonly events = {\n    willStick: 'willStick',\n    didStick: 'didStick',\n    willUnstick: 'willUnstick',\n    didUnstick: 'didUnstick',\n  } as const;\n\n  /** Color of the container, like transparent, white etc. */\n  @property({ reflect: true }) public accessor color: 'white' | 'milk' | null = null;\n\n  /** The state of the component. */\n  private set _state(state: StickyState) {\n    this.setAttribute('data-state', state);\n  }\n  private get _state(): StickyState {\n    return this.getAttribute('data-state') as StickyState;\n  }\n\n  private _willStick: EventEmitter = new EventEmitter(this, SbbStickyBarElement.events.willStick);\n  private _didStick: EventEmitter = new EventEmitter(this, SbbStickyBarElement.events.didStick);\n  private _willUnstick: EventEmitter = new EventEmitter(\n    this,\n    SbbStickyBarElement.events.willUnstick,\n  );\n  private _didUnstick: EventEmitter = new EventEmitter(this, SbbStickyBarElement.events.didUnstick);\n\n  private _intersector?: HTMLSpanElement;\n  private _observer = new IntersectionController(this, {\n    // Although `this` is observed, we have to postpone observing\n    // into firstUpdated() to achieve a correct initial state.\n    target: null,\n    callback: (entries) => this._detectStickyState(entries[0]),\n  });\n\n  public override connectedCallback(): void {\n    super.connectedCallback();\n    this._state = 'sticky';\n\n    // Sticky bar needs to be hidden until first observer callback\n    this.startUpdate();\n\n    const container = this.closest('sbb-container');\n    if (container) {\n      this.toggleAttribute('data-expanded', container.expanded);\n    }\n    if (this._intersector) {\n      this._observer.observe(this._intersector);\n    }\n  }\n\n  public override disconnectedCallback(): void {\n    super.disconnectedCallback();\n\n    this.toggleAttribute('data-initialized', false);\n  }\n\n  protected override firstUpdated(changedProperties: PropertyValues<this>): void {\n    super.firstUpdated(changedProperties);\n\n    if (!this._intersector) {\n      this._intersector = this.shadowRoot!.querySelector('.sbb-sticky-bar__intersector')!;\n      this._observer.observe(this._intersector);\n    }\n    this._observer.observe(this);\n  }\n\n  private _detectStickyState(entry: IntersectionObserverEntry): void {\n    this.toggleAttribute('data-initialized', true);\n\n    const isSticky = !entry.isIntersecting && entry.boundingClientRect.top > 0;\n\n    // To optimize the visual perception of the sticky bar, we have certain cases (e.g. on page load)\n    // where we want the sticky bar to slide in from the bottom.\n    // To decide whether to slide in from the bottom up,\n    // we check how far away the sticky bar is from the intersector element. When scrolling fast, the\n    // difference can vary slightly. To account for this we add a height tolerance.\n    // This value was found by trial and error.\n    const intersectorRect = this._intersector?.getBoundingClientRect();\n    const stickyBarRect = this.shadowRoot!.querySelector(\n      '.sbb-sticky-bar__wrapper',\n    )!.getBoundingClientRect();\n    const HEIGHT_TOLERANCE = 30;\n\n    this.toggleAttribute(\n      'data-slide-vertically',\n      isSticky &&\n        this._intersector &&\n        Math.abs(intersectorRect!.bottom - stickyBarRect.bottom) > HEIGHT_TOLERANCE,\n    );\n\n    // Toggling data-sticking has to be after data-slide-vertically (prevents background color transition)\n    this.toggleAttribute('data-sticking', isSticky);\n\n    // Sticky bar needs to be hidden until first observer callback\n    this.completeUpdate();\n  }\n\n  /** Animates from normal content flow position to `position: sticky`. */\n  public stick(): void {\n    if (this._state !== 'unsticky' || !this._willStick.emit()) {\n      return;\n    }\n\n    this._state = 'sticking';\n    if (!this.hasAttribute('data-sticking')) {\n      this._stickyCallback();\n    }\n  }\n\n  /** Animates `position: sticky` to normal content flow position. */\n  public unstick(): void {\n    if (this._state !== 'sticky' || !this._willUnstick.emit()) {\n      return;\n    }\n\n    this._state = 'unsticking';\n\n    if (!this.hasAttribute('data-sticking')) {\n      this._unstickyCallback();\n    }\n  }\n\n  private _stickyCallback(): void {\n    this._state = 'sticky';\n    this._didStick.emit();\n  }\n\n  private _unstickyCallback(): void {\n    this._didUnstick.emit();\n    this._state = 'unsticky';\n  }\n\n  private _onAnimationEnd(event: AnimationEvent): void {\n    if (\n      (this._state === 'sticking' || this._state === 'sticky') &&\n      event.animationName === 'slide-in'\n    ) {\n      this._stickyCallback();\n    } else if (this._state === 'unsticking' && event.animationName === 'slide-out') {\n      this._unstickyCallback();\n    }\n  }\n\n  protected override render(): TemplateResult {\n    return html`\n      <div class=\"sbb-sticky-bar__wrapper\" @animationend=${this._onAnimationEnd}>\n        <div class=\"sbb-sticky-bar\">\n          <slot></slot>\n        </div>\n      </div>\n      <div class=\"sbb-sticky-bar__intersector\"></div>\n    `;\n  }\n}\n\ndeclare global {\n  interface HTMLElementTagNameMap {\n    // eslint-disable-next-line @typescript-eslint/naming-convention\n    'sbb-sticky-bar': SbbStickyBarElement;\n  }\n}\n"],"names":["SbbStickyBarElement","_a"],"mappings":"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"}
|
|
@@ -0,0 +1,32 @@
|
|
|
1
|
+
import { ReactiveController, ReactiveControllerHost } from 'lit';
|
|
2
|
+
export declare const SbbMediaQueryForcedColors = "(forced-colors: active)";
|
|
3
|
+
export declare const SbbMediaQueryHover = "(any-hover: hover)";
|
|
4
|
+
export declare const SbbMediaQueryPointerCoarse = "(pointer: coarse)";
|
|
5
|
+
export declare const SbbMediaQueryBreakpointMediumAndAbove: string;
|
|
6
|
+
export declare const SbbMediaQueryBreakpointSmallAndBelow: string;
|
|
7
|
+
/**
|
|
8
|
+
* A callback, which is invoked when the associated media query match
|
|
9
|
+
* status changes.
|
|
10
|
+
*/
|
|
11
|
+
export type SbbMediaMatcherHandler = (matches: boolean) => void;
|
|
12
|
+
/**
|
|
13
|
+
* This controller allows listening to media query changes.
|
|
14
|
+
*
|
|
15
|
+
* @example
|
|
16
|
+
* new SbbMediaMatcherController(this, {
|
|
17
|
+
* [SbbForcedColorsQuery]: (matches) => doSomething(matches),
|
|
18
|
+
* })
|
|
19
|
+
*/
|
|
20
|
+
export declare class SbbMediaMatcherController implements ReactiveController {
|
|
21
|
+
private _queries;
|
|
22
|
+
constructor(host: ReactiveControllerHost, _queries: Record<string, SbbMediaMatcherHandler>);
|
|
23
|
+
/**
|
|
24
|
+
* Returns whether the given query matches. Returns null with SSR.
|
|
25
|
+
* @param query The query to check against.
|
|
26
|
+
* @returns Whether the query matches or null with SSR.
|
|
27
|
+
*/
|
|
28
|
+
matches(query: string): boolean | null;
|
|
29
|
+
hostConnected(): void;
|
|
30
|
+
hostDisconnected(): void;
|
|
31
|
+
}
|
|
32
|
+
//# sourceMappingURL=media-matchers-controller.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"media-matchers-controller.d.ts","sourceRoot":"","sources":["../../../../../src/elements/core/controllers/media-matchers-controller.ts"],"names":[],"mappings":"AAKA,OAAO,EAAY,KAAK,kBAAkB,EAAE,KAAK,sBAAsB,EAAE,MAAM,KAAK,CAAC;AAKrF,eAAO,MAAM,yBAAyB,4BAA4B,CAAC;AACnE,eAAO,MAAM,kBAAkB,uBAAuB,CAAC;AACvD,eAAO,MAAM,0BAA0B,sBAAsB,CAAC;AAC9D,eAAO,MAAM,qCAAqC,QAAuD,CAAC;AAC1G,eAAO,MAAM,oCAAoC,QAAsD,CAAC;AAGxG;;;GAGG;AACH,MAAM,MAAM,sBAAsB,GAAG,CAAC,OAAO,EAAE,OAAO,KAAK,IAAI,CAAC;AAehE;;;;;;;GAOG;AACH,qBAAa,yBAA0B,YAAW,kBAAkB;IAGhE,OAAO,CAAC,QAAQ;gBADhB,IAAI,EAAE,sBAAsB,EACpB,QAAQ,EAAE,MAAM,CAAC,MAAM,EAAE,sBAAsB,CAAC;IAK1D;;;;OAIG;IACI,OAAO,CAAC,KAAK,EAAE,MAAM,GAAG,OAAO,GAAG,IAAI;IAYtC,aAAa,IAAI,IAAI;IAoBrB,gBAAgB,IAAI,IAAI;CAWhC"}
|
|
@@ -1,5 +1,6 @@
|
|
|
1
1
|
export * from './controllers/connected-abort-controller.js';
|
|
2
2
|
export * from './controllers/inert-controller.js';
|
|
3
3
|
export * from './controllers/language-controller.js';
|
|
4
|
+
export * from './controllers/media-matchers-controller.js';
|
|
4
5
|
export * from './controllers/slot-state-controller.js';
|
|
5
6
|
//# sourceMappingURL=controllers.d.ts.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"controllers.d.ts","sourceRoot":"","sources":["../../../../src/elements/core/controllers.ts"],"names":[],"mappings":"AAAA,cAAc,6CAA6C,CAAC;AAC5D,cAAc,mCAAmC,CAAC;AAClD,cAAc,sCAAsC,CAAC;AACrD,cAAc,wCAAwC,CAAC"}
|
|
1
|
+
{"version":3,"file":"controllers.d.ts","sourceRoot":"","sources":["../../../../src/elements/core/controllers.ts"],"names":[],"mappings":"AAAA,cAAc,6CAA6C,CAAC;AAC5D,cAAc,mCAAmC,CAAC;AAClD,cAAc,sCAAsC,CAAC;AACrD,cAAc,4CAA4C,CAAC;AAC3D,cAAc,wCAAwC,CAAC"}
|
|
@@ -1,5 +1,6 @@
|
|
|
1
1
|
import { isServer } from "lit";
|
|
2
2
|
import { readConfig } from "./config.js";
|
|
3
|
+
import { S as SbbTypoScaleDefault, g as SbbBreakpointMediumMin, f as SbbBreakpointSmallMax } from "../sbb-tokens-BdGhUJjM.js";
|
|
3
4
|
class SbbConnectedAbortController {
|
|
4
5
|
get signal() {
|
|
5
6
|
var _a;
|
|
@@ -167,6 +168,63 @@ _SbbLanguageController._observerConfig = {
|
|
|
167
168
|
attributeOldValue: true
|
|
168
169
|
};
|
|
169
170
|
let SbbLanguageController = _SbbLanguageController;
|
|
171
|
+
const pxToRem = (px) => px / SbbTypoScaleDefault;
|
|
172
|
+
const SbbMediaQueryForcedColors = "(forced-colors: active)";
|
|
173
|
+
const SbbMediaQueryHover = "(any-hover: hover)";
|
|
174
|
+
const SbbMediaQueryPointerCoarse = "(pointer: coarse)";
|
|
175
|
+
const SbbMediaQueryBreakpointMediumAndAbove = `(min-width: ${pxToRem(SbbBreakpointMediumMin)}rem)`;
|
|
176
|
+
const SbbMediaQueryBreakpointSmallAndBelow = `(max-width: ${pxToRem(SbbBreakpointSmallMax)}rem)`;
|
|
177
|
+
const mediaQueryRegistry = /* @__PURE__ */ new Map();
|
|
178
|
+
class SbbMediaMatcherController {
|
|
179
|
+
constructor(host, _queries) {
|
|
180
|
+
this._queries = _queries;
|
|
181
|
+
host.addController(this);
|
|
182
|
+
}
|
|
183
|
+
/**
|
|
184
|
+
* Returns whether the given query matches. Returns null with SSR.
|
|
185
|
+
* @param query The query to check against.
|
|
186
|
+
* @returns Whether the query matches or null with SSR.
|
|
187
|
+
*/
|
|
188
|
+
matches(query) {
|
|
189
|
+
if (isServer) {
|
|
190
|
+
return null;
|
|
191
|
+
}
|
|
192
|
+
const mediaQuery = mediaQueryRegistry.get(query);
|
|
193
|
+
if (mediaQuery) {
|
|
194
|
+
return mediaQuery.mediaQueryList.matches;
|
|
195
|
+
} else {
|
|
196
|
+
return matchMedia(query).matches;
|
|
197
|
+
}
|
|
198
|
+
}
|
|
199
|
+
hostConnected() {
|
|
200
|
+
if (isServer) {
|
|
201
|
+
return;
|
|
202
|
+
}
|
|
203
|
+
for (const [query, handler] of Object.entries(this._queries)) {
|
|
204
|
+
const mediaQuery = mediaQueryRegistry.get(query);
|
|
205
|
+
if (mediaQuery) {
|
|
206
|
+
mediaQuery.handlers.add(handler);
|
|
207
|
+
} else {
|
|
208
|
+
const mediaQueryList = matchMedia(query);
|
|
209
|
+
const handlers = /* @__PURE__ */ new Set([handler]);
|
|
210
|
+
const eventHandler = (e) => handlers.forEach((h) => h(e.matches));
|
|
211
|
+
mediaQueryList.addEventListener("change", eventHandler);
|
|
212
|
+
mediaQueryRegistry.set(query, { mediaQueryList, handlers, eventHandler });
|
|
213
|
+
}
|
|
214
|
+
}
|
|
215
|
+
}
|
|
216
|
+
hostDisconnected() {
|
|
217
|
+
for (const [query, handler] of Object.entries(this._queries)) {
|
|
218
|
+
const mediaQuery = mediaQueryRegistry.get(query);
|
|
219
|
+
if (mediaQuery) {
|
|
220
|
+
mediaQuery.handlers.delete(handler);
|
|
221
|
+
if (!mediaQuery.handlers.size) {
|
|
222
|
+
mediaQueryRegistry.delete(query);
|
|
223
|
+
}
|
|
224
|
+
}
|
|
225
|
+
}
|
|
226
|
+
}
|
|
227
|
+
}
|
|
170
228
|
class SbbSlotStateController {
|
|
171
229
|
constructor(_host, _onChangeCallback = null) {
|
|
172
230
|
this._host = _host;
|
|
@@ -215,6 +273,12 @@ export {
|
|
|
215
273
|
SbbConnectedAbortController,
|
|
216
274
|
SbbInertController,
|
|
217
275
|
SbbLanguageController,
|
|
276
|
+
SbbMediaMatcherController,
|
|
277
|
+
SbbMediaQueryBreakpointMediumAndAbove,
|
|
278
|
+
SbbMediaQueryBreakpointSmallAndBelow,
|
|
279
|
+
SbbMediaQueryForcedColors,
|
|
280
|
+
SbbMediaQueryHover,
|
|
281
|
+
SbbMediaQueryPointerCoarse,
|
|
218
282
|
SbbSlotStateController
|
|
219
283
|
};
|
|
220
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"controllers.js","sources":["../../../../src/elements/core/controllers/connected-abort-controller.ts","../../../../src/elements/core/controllers/inert-controller.ts","../../../../src/elements/core/controllers/language-controller.ts","../../../../src/elements/core/controllers/slot-state-controller.ts"],"sourcesContent":["import type { ReactiveController, ReactiveControllerHost } from 'lit';\n\nexport class SbbConnectedAbortController implements ReactiveController {\n  private _abortController?: AbortController = new AbortController();\n\n  public get signal(): AbortSignal | undefined {\n    return this._abortController?.signal;\n  }\n\n  public constructor(private _host: ReactiveControllerHost) {\n    this._host.addController(this);\n  }\n\n  public hostConnected(): void {\n    if (!this._abortController) {\n      this._abortController = new AbortController();\n    }\n  }\n\n  public hostDisconnected(): void {\n    this._abortController?.abort();\n    this._abortController = undefined;\n  }\n}\n","import type { ReactiveController, ReactiveControllerHost } from 'lit';\n\nimport type { SbbOpenCloseBaseElement } from '../base-elements/open-close-base-element.js';\n\nconst IGNORED_ELEMENTS = ['script', 'head', 'template', 'style'];\nconst inertElements = new Set<HTMLElement>();\nconst inertOverlays = new Set<HTMLElement>();\n\nexport class SbbInertController implements ReactiveController {\n  public constructor(\n    private _host: ReactiveControllerHost & SbbOpenCloseBaseElement,\n    private _inertElements = inertElements,\n    private _inertOverlays = inertOverlays,\n  ) {\n    this._host.addController?.(this);\n  }\n\n  public hostConnected(): void {\n    if (this._host.isOpen) {\n      this.activate();\n    }\n  }\n\n  public hostDisconnected(): void {\n    if (this._inertOverlays.has(this._host)) {\n      this.deactivate();\n    }\n  }\n\n  /** Applies inert state to every other element on the page except the overlay. */\n  public activate(): void {\n    // Remove inert state from previous opened overlay\n    if (this._inertOverlays.size) {\n      this._removeInertAttributes();\n    }\n\n    this._inertOverlays.add(this._host);\n    this._addInertAttributes();\n  }\n\n  /** Removes inert state. */\n  public deactivate(): void {\n    if (this._currentOverlay() !== this._host) {\n      // If e.g. a component gets disconnected, it could be that it is not the top most.\n      // In this case, we can directly remove it, as there is currently no inert state applied.\n      if (this._inertOverlays.has(this._host)) {\n        this._inertOverlays.delete(this._host);\n      } else if (import.meta.env.DEV) {\n        console.warn(\n          'Trying to remove inert state of an overlay which never had an applied inert state.',\n          this._host,\n        );\n      }\n\n      return;\n    }\n\n    this._removeInertAttributes();\n    this._inertOverlays.delete(this._host);\n\n    // If there is as previous opened overlay, set its inert state again.\n    if (this._inertOverlays.size) {\n      this._addInertAttributes();\n    }\n  }\n\n  private _currentOverlay(): HTMLElement | null {\n    return [...this._inertOverlays].pop() ?? null;\n  }\n\n  private _removeInertAttributes(): void {\n    this._inertElements.forEach((element: HTMLElement): void => {\n      if (!element) {\n        return;\n      }\n\n      if (element.hasAttribute('data-sbb-inert')) {\n        element.inert = false;\n        element.removeAttribute('data-sbb-inert');\n      }\n\n      if (element.hasAttribute('data-sbb-aria-hidden')) {\n        element.removeAttribute('aria-hidden');\n        element.removeAttribute('data-sbb-aria-hidden');\n      }\n    });\n    this._inertElements.clear();\n  }\n\n  private _addInertAttributes(): void {\n    let element: Element | null = this._currentOverlay();\n\n    while (element !== document.documentElement && element !== null) {\n      Array.from((element?.parentElement ?? element?.getRootNode())?.childNodes ?? [])\n        .filter(\n          (child): child is HTMLElement =>\n            child !== element &&\n            child instanceof window.HTMLElement &&\n            !IGNORED_ELEMENTS.includes(child.localName),\n        )\n        .forEach((element) => {\n          this._inertElements.add(element);\n\n          if (!element.inert) {\n            element.inert = true;\n            element.toggleAttribute('data-sbb-inert', true);\n          }\n\n          if (!element.hasAttribute('aria-hidden')) {\n            element.setAttribute('aria-hidden', 'true');\n            element.toggleAttribute('data-sbb-aria-hidden', true);\n          }\n        });\n\n      // We need to pierce through Shadow DOM boundary\n      element = element?.parentElement ?? (element?.getRootNode() as ShadowRoot)?.host ?? null;\n    }\n  }\n}\n","import { isServer, type ReactiveController, type ReactiveControllerHost } from 'lit';\n\nimport { readConfig } from '../config.js';\n\n/**\n * The LanguageController is a reactive controller that observes the \"lang\" attribute\n * of the <html> tag.\n * On change of the \"lang\" attribute, it will request an update of connected\n * components.\n *\n * @see https://developer.mozilla.org/en-US/docs/Web/HTML/Global_attributes/lang\n */\nexport class SbbLanguageController implements ReactiveController {\n  private static readonly _defaultLanguage = 'en';\n  private static readonly _supportedLocales = ['en', 'de', 'fr', 'it'];\n\n  /** A set of connected components that should be notified on language change. */\n  private static readonly _listeners = new Set<SbbLanguageController>();\n\n  /** MutationObserver that observes the \"lang\" attribute of the <html> element. */\n  private static readonly _observer = !isServer\n    ? new MutationObserver((mutations) => {\n        if (mutations[0].oldValue !== document.documentElement.getAttribute('lang')) {\n          SbbLanguageController._listeners.forEach((l) => l._callHandlers());\n        }\n      })\n    : null;\n  private static readonly _observerConfig = {\n    attributeFilter: ['lang'],\n    attributeOldValue: true,\n  };\n\n  /** Get the current language. */\n  public static get current(): string {\n    const language =\n      (readConfig().language ??\n        (isServer\n          ? SbbLanguageController._defaultLanguage\n          : document.documentElement.getAttribute('lang'))) ||\n      SbbLanguageController._defaultLanguage;\n\n    // Support e.g. cases like `de-ch`.\n    const langAttributeNormalized = language.split('-')[0];\n    return SbbLanguageController._supportedLocales.includes(langAttributeNormalized)\n      ? langAttributeNormalized\n      : SbbLanguageController._defaultLanguage;\n  }\n\n  /** Get the current language. */\n  public get current(): string {\n    return SbbLanguageController.current;\n  }\n\n  private _previousLanguage?: string;\n  private _handlers: (() => void)[] = [];\n\n  public constructor(private _host: ReactiveControllerHost) {\n    this._host.addController(this);\n  }\n\n  /** Add a language change handler. */\n  public withHandler(handler: () => void): this {\n    // We use unshift here, to prepend additional handlers.\n    // This ensures that requestUpdate is called after the other handlers.\n    this._handlers.unshift(handler);\n    return this;\n  }\n\n  public hostConnected(): void {\n    if (isServer) {\n      return;\n    }\n    if (!SbbLanguageController._listeners.size) {\n      SbbLanguageController._observer!.observe(\n        document.documentElement,\n        SbbLanguageController._observerConfig,\n      );\n    }\n\n    SbbLanguageController._listeners.add(this);\n    if (this._previousLanguage !== this.current) {\n      this._callHandlers(this._previousLanguage !== undefined);\n    }\n  }\n\n  public hostDisconnected(): void {\n    if (isServer) {\n      return;\n    }\n    this._previousLanguage = this.current;\n    SbbLanguageController._listeners.delete(this);\n    if (!SbbLanguageController._listeners.size) {\n      SbbLanguageController._observer!.disconnect();\n    }\n  }\n\n  private _callHandlers(requestUpdate = true): void {\n    this._handlers.forEach((h) => h());\n    if (requestUpdate) {\n      this._host.requestUpdate();\n    }\n  }\n}\n","import type { ReactiveController, ReactiveControllerHost } from 'lit';\n\n/**\n * This controller checks for slotted children. From these it generates\n * a list of used slot names (`unnamed` for children without a slot attribute)\n * and adds this to the `data-slot-names` attribute, as a space separated list.\n *\n * This allows CSS attribute selector to display/hide/configure a section\n * of the component as required (see [attr~=value] selector specifically).\n *\n * @example\n * .example {\n *   display: none;\n *\n *   :host([data-slot-names~=\"icon\"]) & {\n *     display: inline;\n *   }\n * }\n *\n * https://developer.mozilla.org/en-US/docs/Web/CSS/Attribute_selectors\n */\nexport class SbbSlotStateController implements ReactiveController {\n  public readonly slots = new Set<string>();\n\n  public constructor(\n    private _host: ReactiveControllerHost & HTMLElement,\n    private _onChangeCallback: (() => void) | null = null,\n  ) {\n    this._host.addController(this);\n  }\n\n  public hostConnected(): void {\n    // TODO: Check if this is really needed with SSR.\n    this._syncSlots(...this._host.querySelectorAll!('slot'));\n    this._host.shadowRoot?.addEventListener('slotchange', this._slotchangeHandler);\n  }\n\n  public hostDisconnected(): void {\n    this._host.shadowRoot?.removeEventListener('slotchange', this._slotchangeHandler);\n  }\n\n  // We avoid using AbortController here, as it would mean creating\n  // a new instance for every NamedSlotStateController instance.\n  private _slotchangeHandler = (event: Event): void => {\n    this._syncSlots(event.target as HTMLSlotElement);\n  };\n\n  private _syncSlots(...slots: HTMLSlotElement[]): void {\n    for (const slot of slots) {\n      const slotName = slot.name || 'unnamed';\n      // We want to check, whether an element is slotted or a text node with actual content.\n      if (slot.assignedNodes().some((n) => 'tagName' in n || n.textContent?.trim())) {\n        this.slots.add(slotName);\n      } else {\n        this.slots.delete(slotName);\n      }\n    }\n\n    const oldValue = this._host.getAttribute('data-slot-names');\n    const joinedSlotNames = [...this.slots].sort().join(' ');\n    if (!joinedSlotNames) {\n      this._host.removeAttribute('data-slot-names');\n    } else if (this._host.getAttribute('data-slot-names') !== joinedSlotNames) {\n      this._host.setAttribute('data-slot-names', joinedSlotNames);\n    }\n\n    if (joinedSlotNames !== oldValue) {\n      this._onChangeCallback?.();\n    }\n  }\n}\n"],"names":["element","_a"],"mappings":"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"}
|
|
284
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"controllers.js","sources":["../../../../src/elements/core/controllers/connected-abort-controller.ts","../../../../src/elements/core/controllers/inert-controller.ts","../../../../src/elements/core/controllers/language-controller.ts","../../../../src/elements/core/controllers/media-matchers-controller.ts","../../../../src/elements/core/controllers/slot-state-controller.ts"],"sourcesContent":["import type { ReactiveController, ReactiveControllerHost } from 'lit';\n\nexport class SbbConnectedAbortController implements ReactiveController {\n  private _abortController?: AbortController = new AbortController();\n\n  public get signal(): AbortSignal | undefined {\n    return this._abortController?.signal;\n  }\n\n  public constructor(private _host: ReactiveControllerHost) {\n    this._host.addController(this);\n  }\n\n  public hostConnected(): void {\n    if (!this._abortController) {\n      this._abortController = new AbortController();\n    }\n  }\n\n  public hostDisconnected(): void {\n    this._abortController?.abort();\n    this._abortController = undefined;\n  }\n}\n","import type { ReactiveController, ReactiveControllerHost } from 'lit';\n\nimport type { SbbOpenCloseBaseElement } from '../base-elements/open-close-base-element.js';\n\nconst IGNORED_ELEMENTS = ['script', 'head', 'template', 'style'];\nconst inertElements = new Set<HTMLElement>();\nconst inertOverlays = new Set<HTMLElement>();\n\nexport class SbbInertController implements ReactiveController {\n  public constructor(\n    private _host: ReactiveControllerHost & SbbOpenCloseBaseElement,\n    private _inertElements = inertElements,\n    private _inertOverlays = inertOverlays,\n  ) {\n    this._host.addController?.(this);\n  }\n\n  public hostConnected(): void {\n    if (this._host.isOpen) {\n      this.activate();\n    }\n  }\n\n  public hostDisconnected(): void {\n    if (this._inertOverlays.has(this._host)) {\n      this.deactivate();\n    }\n  }\n\n  /** Applies inert state to every other element on the page except the overlay. */\n  public activate(): void {\n    // Remove inert state from previous opened overlay\n    if (this._inertOverlays.size) {\n      this._removeInertAttributes();\n    }\n\n    this._inertOverlays.add(this._host);\n    this._addInertAttributes();\n  }\n\n  /** Removes inert state. */\n  public deactivate(): void {\n    if (this._currentOverlay() !== this._host) {\n      // If e.g. a component gets disconnected, it could be that it is not the top most.\n      // In this case, we can directly remove it, as there is currently no inert state applied.\n      if (this._inertOverlays.has(this._host)) {\n        this._inertOverlays.delete(this._host);\n      } else if (import.meta.env.DEV) {\n        console.warn(\n          'Trying to remove inert state of an overlay which never had an applied inert state.',\n          this._host,\n        );\n      }\n\n      return;\n    }\n\n    this._removeInertAttributes();\n    this._inertOverlays.delete(this._host);\n\n    // If there is as previous opened overlay, set its inert state again.\n    if (this._inertOverlays.size) {\n      this._addInertAttributes();\n    }\n  }\n\n  private _currentOverlay(): HTMLElement | null {\n    return [...this._inertOverlays].pop() ?? null;\n  }\n\n  private _removeInertAttributes(): void {\n    this._inertElements.forEach((element: HTMLElement): void => {\n      if (!element) {\n        return;\n      }\n\n      if (element.hasAttribute('data-sbb-inert')) {\n        element.inert = false;\n        element.removeAttribute('data-sbb-inert');\n      }\n\n      if (element.hasAttribute('data-sbb-aria-hidden')) {\n        element.removeAttribute('aria-hidden');\n        element.removeAttribute('data-sbb-aria-hidden');\n      }\n    });\n    this._inertElements.clear();\n  }\n\n  private _addInertAttributes(): void {\n    let element: Element | null = this._currentOverlay();\n\n    while (element !== document.documentElement && element !== null) {\n      Array.from((element?.parentElement ?? element?.getRootNode())?.childNodes ?? [])\n        .filter(\n          (child): child is HTMLElement =>\n            child !== element &&\n            child instanceof window.HTMLElement &&\n            !IGNORED_ELEMENTS.includes(child.localName),\n        )\n        .forEach((element) => {\n          this._inertElements.add(element);\n\n          if (!element.inert) {\n            element.inert = true;\n            element.toggleAttribute('data-sbb-inert', true);\n          }\n\n          if (!element.hasAttribute('aria-hidden')) {\n            element.setAttribute('aria-hidden', 'true');\n            element.toggleAttribute('data-sbb-aria-hidden', true);\n          }\n        });\n\n      // We need to pierce through Shadow DOM boundary\n      element = element?.parentElement ?? (element?.getRootNode() as ShadowRoot)?.host ?? null;\n    }\n  }\n}\n","import { isServer, type ReactiveController, type ReactiveControllerHost } from 'lit';\n\nimport { readConfig } from '../config.js';\n\n/**\n * The LanguageController is a reactive controller that observes the \"lang\" attribute\n * of the <html> tag.\n * On change of the \"lang\" attribute, it will request an update of connected\n * components.\n *\n * @see https://developer.mozilla.org/en-US/docs/Web/HTML/Global_attributes/lang\n */\nexport class SbbLanguageController implements ReactiveController {\n  private static readonly _defaultLanguage = 'en';\n  private static readonly _supportedLocales = ['en', 'de', 'fr', 'it'];\n\n  /** A set of connected components that should be notified on language change. */\n  private static readonly _listeners = new Set<SbbLanguageController>();\n\n  /** MutationObserver that observes the \"lang\" attribute of the <html> element. */\n  private static readonly _observer = !isServer\n    ? new MutationObserver((mutations) => {\n        if (mutations[0].oldValue !== document.documentElement.getAttribute('lang')) {\n          SbbLanguageController._listeners.forEach((l) => l._callHandlers());\n        }\n      })\n    : null;\n  private static readonly _observerConfig = {\n    attributeFilter: ['lang'],\n    attributeOldValue: true,\n  };\n\n  /** Get the current language. */\n  public static get current(): string {\n    const language =\n      (readConfig().language ??\n        (isServer\n          ? SbbLanguageController._defaultLanguage\n          : document.documentElement.getAttribute('lang'))) ||\n      SbbLanguageController._defaultLanguage;\n\n    // Support e.g. cases like `de-ch`.\n    const langAttributeNormalized = language.split('-')[0];\n    return SbbLanguageController._supportedLocales.includes(langAttributeNormalized)\n      ? langAttributeNormalized\n      : SbbLanguageController._defaultLanguage;\n  }\n\n  /** Get the current language. */\n  public get current(): string {\n    return SbbLanguageController.current;\n  }\n\n  private _previousLanguage?: string;\n  private _handlers: (() => void)[] = [];\n\n  public constructor(private _host: ReactiveControllerHost) {\n    this._host.addController(this);\n  }\n\n  /** Add a language change handler. */\n  public withHandler(handler: () => void): this {\n    // We use unshift here, to prepend additional handlers.\n    // This ensures that requestUpdate is called after the other handlers.\n    this._handlers.unshift(handler);\n    return this;\n  }\n\n  public hostConnected(): void {\n    if (isServer) {\n      return;\n    }\n    if (!SbbLanguageController._listeners.size) {\n      SbbLanguageController._observer!.observe(\n        document.documentElement,\n        SbbLanguageController._observerConfig,\n      );\n    }\n\n    SbbLanguageController._listeners.add(this);\n    if (this._previousLanguage !== this.current) {\n      this._callHandlers(this._previousLanguage !== undefined);\n    }\n  }\n\n  public hostDisconnected(): void {\n    if (isServer) {\n      return;\n    }\n    this._previousLanguage = this.current;\n    SbbLanguageController._listeners.delete(this);\n    if (!SbbLanguageController._listeners.size) {\n      SbbLanguageController._observer!.disconnect();\n    }\n  }\n\n  private _callHandlers(requestUpdate = true): void {\n    this._handlers.forEach((h) => h());\n    if (requestUpdate) {\n      this._host.requestUpdate();\n    }\n  }\n}\n","import {\n  SbbBreakpointMediumMin,\n  SbbBreakpointSmallMax,\n  SbbTypoScaleDefault,\n} from '@sbb-esta/lyne-design-tokens';\nimport { isServer, type ReactiveController, type ReactiveControllerHost } from 'lit';\n\nconst pxToRem = (px: number): number => px / SbbTypoScaleDefault;\n\n/* eslint-disable @typescript-eslint/naming-convention */\nexport const SbbMediaQueryForcedColors = '(forced-colors: active)';\nexport const SbbMediaQueryHover = '(any-hover: hover)';\nexport const SbbMediaQueryPointerCoarse = '(pointer: coarse)';\nexport const SbbMediaQueryBreakpointMediumAndAbove = `(min-width: ${pxToRem(SbbBreakpointMediumMin)}rem)`;\nexport const SbbMediaQueryBreakpointSmallAndBelow = `(max-width: ${pxToRem(SbbBreakpointSmallMax)}rem)`;\n/* eslint-enable @typescript-eslint/naming-convention */\n\n/**\n * A callback, which is invoked when the associated media query match\n * status changes.\n */\nexport type SbbMediaMatcherHandler = (matches: boolean) => void;\n\ninterface MediaQueryEntry {\n  mediaQueryList: MediaQueryList;\n  eventHandler: (event: MediaQueryListEvent) => void;\n  handlers: Set<SbbMediaMatcherHandler>;\n}\n\n/**\n * We want to cache MediaQueryList instances and corresponding\n * event handlers, as a multitude of event handlers on global objects\n * can degrade performance with time.\n */\nconst mediaQueryRegistry = new Map<string, MediaQueryEntry>();\n\n/**\n * This controller allows listening to media query changes.\n *\n * @example\n * new SbbMediaMatcherController(this, {\n *   [SbbForcedColorsQuery]: (matches) => doSomething(matches),\n * })\n */\nexport class SbbMediaMatcherController implements ReactiveController {\n  public constructor(\n    host: ReactiveControllerHost,\n    private _queries: Record<string, SbbMediaMatcherHandler>,\n  ) {\n    host.addController(this);\n  }\n\n  /**\n   * Returns whether the given query matches. Returns null with SSR.\n   * @param query The query to check against.\n   * @returns Whether the query matches or null with SSR.\n   */\n  public matches(query: string): boolean | null {\n    if (isServer) {\n      return null;\n    }\n    const mediaQuery = mediaQueryRegistry.get(query);\n    if (mediaQuery) {\n      return mediaQuery.mediaQueryList.matches;\n    } else {\n      return matchMedia(query).matches;\n    }\n  }\n\n  public hostConnected(): void {\n    if (isServer) {\n      return;\n    }\n\n    for (const [query, handler] of Object.entries(this._queries)) {\n      const mediaQuery = mediaQueryRegistry.get(query);\n      if (mediaQuery) {\n        mediaQuery.handlers.add(handler);\n      } else {\n        const mediaQueryList = matchMedia(query);\n        const handlers = new Set([handler]);\n        const eventHandler = (e: MediaQueryListEvent): void =>\n          handlers.forEach((h) => h(e.matches));\n        mediaQueryList.addEventListener('change', eventHandler);\n        mediaQueryRegistry.set(query, { mediaQueryList, handlers, eventHandler });\n      }\n    }\n  }\n\n  public hostDisconnected(): void {\n    for (const [query, handler] of Object.entries(this._queries)) {\n      const mediaQuery = mediaQueryRegistry.get(query);\n      if (mediaQuery) {\n        mediaQuery.handlers.delete(handler);\n        if (!mediaQuery.handlers.size) {\n          mediaQueryRegistry.delete(query);\n        }\n      }\n    }\n  }\n}\n","import type { ReactiveController, ReactiveControllerHost } from 'lit';\n\n/**\n * This controller checks for slotted children. From these it generates\n * a list of used slot names (`unnamed` for children without a slot attribute)\n * and adds this to the `data-slot-names` attribute, as a space separated list.\n *\n * This allows CSS attribute selector to display/hide/configure a section\n * of the component as required (see [attr~=value] selector specifically).\n *\n * @example\n * .example {\n *   display: none;\n *\n *   :host([data-slot-names~=\"icon\"]) & {\n *     display: inline;\n *   }\n * }\n *\n * https://developer.mozilla.org/en-US/docs/Web/CSS/Attribute_selectors\n */\nexport class SbbSlotStateController implements ReactiveController {\n  public readonly slots = new Set<string>();\n\n  public constructor(\n    private _host: ReactiveControllerHost & HTMLElement,\n    private _onChangeCallback: (() => void) | null = null,\n  ) {\n    this._host.addController(this);\n  }\n\n  public hostConnected(): void {\n    // TODO: Check if this is really needed with SSR.\n    this._syncSlots(...this._host.querySelectorAll!('slot'));\n    this._host.shadowRoot?.addEventListener('slotchange', this._slotchangeHandler);\n  }\n\n  public hostDisconnected(): void {\n    this._host.shadowRoot?.removeEventListener('slotchange', this._slotchangeHandler);\n  }\n\n  // We avoid using AbortController here, as it would mean creating\n  // a new instance for every SbbSlotStateController instance.\n  private _slotchangeHandler = (event: Event): void => {\n    this._syncSlots(event.target as HTMLSlotElement);\n  };\n\n  private _syncSlots(...slots: HTMLSlotElement[]): void {\n    for (const slot of slots) {\n      const slotName = slot.name || 'unnamed';\n      // We want to check, whether an element is slotted or a text node with actual content.\n      if (slot.assignedNodes().some((n) => 'tagName' in n || n.textContent?.trim())) {\n        this.slots.add(slotName);\n      } else {\n        this.slots.delete(slotName);\n      }\n    }\n\n    const oldValue = this._host.getAttribute('data-slot-names');\n    const joinedSlotNames = [...this.slots].sort().join(' ');\n    if (!joinedSlotNames) {\n      this._host.removeAttribute('data-slot-names');\n    } else if (this._host.getAttribute('data-slot-names') !== joinedSlotNames) {\n      this._host.setAttribute('data-slot-names', joinedSlotNames);\n    }\n\n    if (joinedSlotNames !== oldValue) {\n      this._onChangeCallback?.();\n    }\n  }\n}\n"],"names":["element","_a"],"mappings":"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"}
|
|
@@ -6,9 +6,10 @@ export type Breakpoint = (typeof breakpoints)[number];
|
|
|
6
6
|
*
|
|
7
7
|
* @param from The breakpoint corresponding to the `min-width` value of the media query (optional).
|
|
8
8
|
* @param to The breakpoint corresponding to the `max-width` value of the media query (optional).
|
|
9
|
+
* @param properties Whether the max breakpoint should be included
|
|
9
10
|
* @returns A boolean indicating whether the window matches the breakpoint.
|
|
10
11
|
*/
|
|
11
12
|
export declare function isBreakpoint(from?: Breakpoint, to?: Breakpoint, properties?: {
|
|
12
13
|
includeMaxBreakpoint: boolean;
|
|
13
|
-
}): boolean;
|
|
14
|
+
}): boolean | null;
|
|
14
15
|
//# sourceMappingURL=breakpoint.d.ts.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"breakpoint.d.ts","sourceRoot":"","sources":["../../../../../src/elements/core/dom/breakpoint.ts"],"names":[],"mappings":"AAEA,eAAO,MAAM,WAAW,yEAA0E,CAAC;AACnG,MAAM,MAAM,UAAU,GAAG,CAAC,OAAO,WAAW,CAAC,CAAC,MAAM,CAAC,CAAC;AAEtD
|
|
1
|
+
{"version":3,"file":"breakpoint.d.ts","sourceRoot":"","sources":["../../../../../src/elements/core/dom/breakpoint.ts"],"names":[],"mappings":"AAEA,eAAO,MAAM,WAAW,yEAA0E,CAAC;AACnG,MAAM,MAAM,UAAU,GAAG,CAAC,OAAO,WAAW,CAAC,CAAC,MAAM,CAAC,CAAC;AAEtD;;;;;;;;GAQG;AACH,wBAAgB,YAAY,CAC1B,IAAI,CAAC,EAAE,UAAU,EACjB,EAAE,CAAC,EAAE,UAAU,EACf,UAAU,CAAC,EAAE;IAAE,oBAAoB,EAAE,OAAO,CAAA;CAAE,GAC7C,OAAO,GAAG,IAAI,CAsBhB"}
|