@sbb-esta/lyne-elements-dev 4.11.0-dev.1777283963 → 4.12.0-dev.1777289332

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.
@@ -280,6 +280,9 @@ $theme: 'standard' !default;
280
280
  @use '../../tooltip/tooltip.global' as tooltip with (
281
281
  $theme: $theme
282
282
  );
283
+ @use '../../toast/toast.global' as toast with (
284
+ $theme: $theme
285
+ );
283
286
  @use '../../visual-checkbox/visual-checkbox.global' as visual-checkbox with (
284
287
  $theme: $theme
285
288
  );
@@ -372,6 +375,7 @@ $theme: 'standard' !default;
372
375
  @include toggle.base;
373
376
  @include toggle-option.base;
374
377
  @include tooltip.base;
378
+ @include toast.base;
375
379
  @include visual-checkbox.base;
376
380
 
377
381
  @include a11y.if-forced-colors {
package/core.css CHANGED
@@ -1855,6 +1855,18 @@ slot[name=error]::slotted(*) {
1855
1855
  --sbb-overlay-position-try-fallbacks:
1856
1856
  block-start span-inline-end, block-start span-inline-start, block-end,
1857
1857
  block-end span-inline-end, block-end span-inline-start;
1858
+ --sbb-toast-max-width: 25rem;
1859
+ --sbb-toast-animation-timing-function: ease;
1860
+ --sbb-toast-text-color: var(--sbb-color-3-inverted);
1861
+ --sbb-toast-background-color: var(--sbb-background-color-1-inverted);
1862
+ --sbb-toast-margin: var(--sbb-spacing-responsive-s);
1863
+ --sbb-toast-padding-block: var(--sbb-spacing-responsive-xxxs);
1864
+ --sbb-toast-padding-inline: var(--sbb-spacing-responsive-xs);
1865
+ --sbb-toast-border-radius: var(--sbb-border-radius-4x);
1866
+ --sbb-toast-container-position: fixed;
1867
+ --sbb-toast-horizontal-position: initial;
1868
+ --sbb-toast-vertical-position: initial;
1869
+ --sbb-toast-gap: var(--sbb-spacing-fixed-2x);
1858
1870
  --sbb-visual-checkbox-dimension: var(--sbb-checkbox-dimension-m);
1859
1871
  --sbb-visual-checkbox-selection-color: var(--sbb-color-primary);
1860
1872
  --sbb-visual-checkbox-selection-color-negative: var(--sbb-color-primary85);
@@ -1,2 +1,2 @@
1
- import { t as SbbToastElement } from "../toast.component-Cgr5ZGOr.js";
1
+ import { t as SbbToastElement } from "../toast.component-IhHx0Qt3.js";
2
2
  export { SbbToastElement };
@@ -0,0 +1,275 @@
1
+ import { __esDecorate, __runInitializers } from "tslib";
2
+ import { html, nothing, unsafeCSS } from "lit";
3
+ import { property } from "lit/decorators.js";
4
+ import { SbbOpenCloseBaseElement } from "./core/base-elements.js";
5
+ import { forceType } from "./core/decorators.js";
6
+ import { isLean, isZeroAnimationDuration } from "./core/dom.js";
7
+ import { composedPathHasAttribute } from "./core/eventing.js";
8
+ import { SbbReadonlyMixin } from "./core/mixins.js";
9
+ import { boxSizingStyles } from "./core/styles.js";
10
+ import { SbbTransparentButtonElement } from "./button.pure.js";
11
+ import { SbbDarkModeController, SbbLanguageController } from "./core/controllers.js";
12
+ import { i18nCloseAlert } from "./core/i18n.js";
13
+ import { SbbIconNameMixin } from "./icon.js";
14
+ import "./divider.js";
15
+ //#region src/elements/toast/toast.scss?inline
16
+ var toast_default = ":host {\n --sbb-toast-animation-duration: var(\n --sbb-disable-animation-duration,\n var(--sbb-animation-duration-6x)\n );\n --sbb-focus-outline-color: var(--sbb-focus-outline-color-dark);\n display: none;\n}\n\n:host(:is(:is(:state(state-opening),[state--state-opening]), :is(:state(state-opened),[state--state-opened]), :is(:state(state-closing),[state--state-closing]))) {\n --sbb-toast-container-display: flex;\n display: block;\n}\n\n:host([position=bottom-left]),\n:host([position=bottom-start]) {\n --sbb-toast-horizontal-position: flex-start;\n --sbb-toast-vertical-position: flex-end;\n}\n\n:host([position=bottom-center]) {\n --sbb-toast-horizontal-position: center;\n --sbb-toast-vertical-position: flex-end;\n}\n\n:host([position=bottom-right]),\n:host([position=bottom-end]) {\n --sbb-toast-horizontal-position: flex-end;\n --sbb-toast-vertical-position: flex-end;\n}\n\n:host([position=top-left]),\n:host([position=top-start]) {\n --sbb-toast-horizontal-position: flex-start;\n --sbb-toast-vertical-position: flex-start;\n}\n\n:host([position=top-center]) {\n --sbb-toast-horizontal-position: center;\n --sbb-toast-vertical-position: flex-start;\n}\n\n:host([position=top-right]),\n:host([position=top-end]) {\n --sbb-toast-horizontal-position: flex-end;\n --sbb-toast-vertical-position: flex-start;\n}\n\n:host(:is(:state(dark),[state--dark])) {\n --sbb-focus-outline-color: var(--sbb-focus-outline-color-default);\n}\n\n.sbb-toast__overlay-container {\n position: var(--sbb-toast-container-position);\n pointer-events: none;\n inset: 0;\n display: var(--sbb-toast-container-display, none);\n justify-content: var(--sbb-toast-horizontal-position);\n align-items: var(--sbb-toast-vertical-position);\n}\n\n.sbb-toast {\n z-index: var(--sbb-toast-z-index, var(--sbb-overlay-default-z-index));\n pointer-events: auto;\n margin: var(--sbb-toast-margin);\n border-radius: var(--sbb-toast-border-radius);\n max-width: var(--sbb-toast-max-width);\n display: flex;\n color: var(--sbb-toast-text-color);\n background-color: var(--sbb-toast-background-color);\n}\n:host(:is(:state(state-opening),[state--state-opening])) .sbb-toast {\n animation-name: open;\n animation-duration: var(--sbb-toast-animation-duration);\n animation-timing-function: var(--sbb-toast-animation-timing-function);\n}\n:host(:is(:state(state-closing),[state--state-closing])) .sbb-toast {\n animation-name: close;\n animation-duration: var(--sbb-toast-animation-duration);\n animation-timing-function: var(--sbb-toast-animation-timing-function);\n}\n\n.sbb-toast-wrapper {\n display: flex;\n align-items: center;\n gap: var(--sbb-toast-gap);\n padding-block: var(--sbb-toast-padding-block);\n padding-inline: var(--sbb-toast-padding-inline);\n}\n\nslot:not([name=icon]),\n.sbb-toast__close {\n color-scheme: only light;\n}\n\n.sbb-toast__content {\n --sbb-title-font-size: var(--sbb-title-font-size-level-6-lean, var(--sbb-heading-font-size-6));\n --sbb-title-line-height: var(--sbb-typo-line-height-text);\n margin-block: var(--sbb-title-margin-block, 0);\n margin-inline: 0;\n font-family: var(--sbb-typo-font-family);\n font-weight: bold;\n font-size: var(--sbb-title-font-size);\n letter-spacing: var(--sbb-typo-letter-spacing-heading);\n line-height: var(--sbb-title-line-height, var(--sbb-typo-line-height-heading));\n flex: 1 1 0;\n}\n\n.sbb-toast__close {\n display: flex;\n align-items: center;\n}\n\n.sbb-toast__close-divider {\n padding-block: var(--sbb-toast-padding-block);\n}\n\n.sbb-toast__close-button {\n margin-inline: var(--sbb-spacing-responsive-xxxs);\n}\n\n@keyframes open {\n from {\n visibility: hidden;\n opacity: 0;\n translate: 0 var(--sbb-spacing-fixed-2x);\n }\n to {\n visibility: visible;\n opacity: 1;\n translate: 0;\n }\n}\n@keyframes close {\n from {\n visibility: visible;\n opacity: 1;\n translate: 0;\n }\n to {\n visibility: hidden;\n opacity: 0;\n translate: 0 var(--sbb-spacing-fixed-2x);\n }\n}";
17
+ //#endregion
18
+ //#region src/elements/toast/toast.component.ts
19
+ SbbTransparentButtonElement.define();
20
+ var toastRefs = /* @__PURE__ */ new Set();
21
+ /**
22
+ * It displays a toast notification.
23
+ *
24
+ * @slot - Use the unnamed slot to add content to the `sbb-toast`.
25
+ * @slot icon - Assign a custom icon via slot.
26
+ * @slot action - Provide a custom action for this toast.
27
+ * @cssprop [--sbb-toast-z-index=var(--sbb-overlay-default-z-index)] - To specify a custom stack order,
28
+ * the `z-index` can be overridden by defining this CSS variable. The default `z-index` of the
29
+ * component is set to `var(--sbb-overlay-default-z-index)` with a value of `1000`.
30
+ */
31
+ var SbbToastElement = (() => {
32
+ let _classSuper = SbbIconNameMixin(SbbReadonlyMixin(SbbOpenCloseBaseElement));
33
+ let _timeout_decorators;
34
+ let _timeout_initializers = [];
35
+ let _timeout_extraInitializers = [];
36
+ let _position_decorators;
37
+ let _position_initializers = [];
38
+ let _position_extraInitializers = [];
39
+ let _politeness_decorators;
40
+ let _politeness_initializers = [];
41
+ let _politeness_extraInitializers = [];
42
+ return class SbbToastElement extends _classSuper {
43
+ static {
44
+ const _metadata = typeof Symbol === "function" && Symbol.metadata ? Object.create(_classSuper[Symbol.metadata] ?? null) : void 0;
45
+ _timeout_decorators = [forceType(), property({ type: Number })];
46
+ _position_decorators = [property({ reflect: true })];
47
+ _politeness_decorators = [property()];
48
+ __esDecorate(this, null, _timeout_decorators, {
49
+ kind: "accessor",
50
+ name: "timeout",
51
+ static: false,
52
+ private: false,
53
+ access: {
54
+ has: (obj) => "timeout" in obj,
55
+ get: (obj) => obj.timeout,
56
+ set: (obj, value) => {
57
+ obj.timeout = value;
58
+ }
59
+ },
60
+ metadata: _metadata
61
+ }, _timeout_initializers, _timeout_extraInitializers);
62
+ __esDecorate(this, null, _position_decorators, {
63
+ kind: "accessor",
64
+ name: "position",
65
+ static: false,
66
+ private: false,
67
+ access: {
68
+ has: (obj) => "position" in obj,
69
+ get: (obj) => obj.position,
70
+ set: (obj, value) => {
71
+ obj.position = value;
72
+ }
73
+ },
74
+ metadata: _metadata
75
+ }, _position_initializers, _position_extraInitializers);
76
+ __esDecorate(this, null, _politeness_decorators, {
77
+ kind: "accessor",
78
+ name: "politeness",
79
+ static: false,
80
+ private: false,
81
+ access: {
82
+ has: (obj) => "politeness" in obj,
83
+ get: (obj) => obj.politeness,
84
+ set: (obj, value) => {
85
+ obj.politeness = value;
86
+ }
87
+ },
88
+ metadata: _metadata
89
+ }, _politeness_initializers, _politeness_extraInitializers);
90
+ if (_metadata) Object.defineProperty(this, Symbol.metadata, {
91
+ enumerable: true,
92
+ configurable: true,
93
+ writable: true,
94
+ value: _metadata
95
+ });
96
+ }
97
+ static {
98
+ this.elementName = "sbb-toast";
99
+ }
100
+ static {
101
+ this.styles = [boxSizingStyles, unsafeCSS(toast_default)];
102
+ }
103
+ #timeout_accessor_storage;
104
+ /**
105
+ * The length of time in milliseconds to wait before automatically dismissing the toast.
106
+ * If 0 (default), it stays open indefinitely.
107
+ * From accessibility perspective, it is recommended to set a timeout of at least 20 seconds.
108
+ */
109
+ get timeout() {
110
+ return this.#timeout_accessor_storage;
111
+ }
112
+ set timeout(value) {
113
+ this.#timeout_accessor_storage = value;
114
+ }
115
+ #position_accessor_storage;
116
+ /** The position where to place the toast. */
117
+ get position() {
118
+ return this.#position_accessor_storage;
119
+ }
120
+ set position(value) {
121
+ this.#position_accessor_storage = value;
122
+ }
123
+ #politeness_accessor_storage;
124
+ /**
125
+ * The ARIA politeness level.
126
+ * Check https://developer.mozilla.org/en-US/docs/Web/Accessibility/ARIA/ARIA_Live_Regions#live_regions for further info
127
+ */
128
+ get politeness() {
129
+ return this.#politeness_accessor_storage;
130
+ }
131
+ set politeness(value) {
132
+ this.#politeness_accessor_storage = value;
133
+ }
134
+ constructor() {
135
+ super();
136
+ this.#timeout_accessor_storage = __runInitializers(this, _timeout_initializers, 0);
137
+ this.#position_accessor_storage = (__runInitializers(this, _timeout_extraInitializers), __runInitializers(this, _position_initializers, "bottom-center"));
138
+ this.#politeness_accessor_storage = (__runInitializers(this, _position_extraInitializers), __runInitializers(this, _politeness_initializers, "polite"));
139
+ this._closeTimeout = __runInitializers(this, _politeness_extraInitializers);
140
+ this._language = new SbbLanguageController(this);
141
+ this._darkModeController = new SbbDarkModeController(this, () => {
142
+ this._syncSlottedElements();
143
+ this.requestUpdate();
144
+ });
145
+ this.addEventListener?.("click", (e) => this._onClick(e));
146
+ }
147
+ connectedCallback() {
148
+ this.popover = "manual";
149
+ super.connectedCallback();
150
+ toastRefs.add(this);
151
+ }
152
+ disconnectedCallback() {
153
+ super.disconnectedCallback();
154
+ clearTimeout(this._closeTimeout);
155
+ toastRefs.delete(this);
156
+ }
157
+ willUpdate(changedProperties) {
158
+ super.willUpdate(changedProperties);
159
+ if (changedProperties.has("politeness")) this.internals.ariaLive = this.politeness;
160
+ }
161
+ /**
162
+ * Open the toast.
163
+ * If there are other opened toasts in the page, close them first.
164
+ */
165
+ open() {
166
+ if (this.state === "opened" || this.state === "opening" || !this.dispatchBeforeOpenEvent()) return;
167
+ this._closeOtherToasts();
168
+ this.showPopover?.();
169
+ this.state = "opening";
170
+ if (this._isZeroAnimationDuration()) this._handleOpening();
171
+ }
172
+ /**
173
+ * Close the toast.
174
+ */
175
+ close() {
176
+ if (this.state === "closed" || this.state === "closing" || !this.dispatchBeforeCloseEvent()) return;
177
+ clearTimeout(this._closeTimeout);
178
+ this.state = "closing";
179
+ if (this._isZeroAnimationDuration()) this._handleClosing();
180
+ }
181
+ _handleClosing() {
182
+ this.state = "closed";
183
+ this.hidePopover?.();
184
+ this.dispatchCloseEvent();
185
+ }
186
+ _handleOpening() {
187
+ this.state = "opened";
188
+ this.dispatchOpenEvent();
189
+ if (this.timeout) {
190
+ const global = globalThis;
191
+ const setTimeout = global[global.Zone?.__symbol__?.("setTimeout")] ?? global.setTimeout;
192
+ this._closeTimeout = setTimeout(() => this.close(), this.timeout);
193
+ }
194
+ }
195
+ _onClick(event) {
196
+ const closeElement = composedPathHasAttribute(event, "sbb-toast-close", this);
197
+ if (closeElement && !closeElement.hasAttribute("disabled")) this.close();
198
+ }
199
+ _isZeroAnimationDuration() {
200
+ return isZeroAnimationDuration(this, "--sbb-toast-animation-duration");
201
+ }
202
+ /**
203
+ * Slotted text nodes are not read correctly by screen readers on Chrome.
204
+ * To address the problem, if there is at least a root text node,
205
+ * we wrap the whole content in a <span> tag
206
+ */
207
+ _onContentSlotChange(event) {
208
+ const slotNodes = event.target.assignedNodes();
209
+ if (slotNodes.some((el) => el.nodeType === Node.TEXT_NODE)) {
210
+ const span = document.createElement("span");
211
+ this.prepend(span);
212
+ span.append(...slotNodes);
213
+ }
214
+ this._syncSlottedElements();
215
+ }
216
+ _syncSlottedElements() {
217
+ this.querySelectorAll("sbb-transparent-button, sbb-transparent-button-link").forEach((btn) => {
218
+ btn.negative = this._isLightMode();
219
+ btn.size = isLean() ? "s" : "m";
220
+ });
221
+ this.querySelectorAll("sbb-link, sbb-link-button, sbb-link-static").forEach((link) => {
222
+ link.negative = this._isLightMode();
223
+ });
224
+ }
225
+ _onToastAnimationEnd(event) {
226
+ if (event.animationName === "open" && this.state === "opening") this._handleOpening();
227
+ else if (event.animationName === "close" && this.state === "closing") this._handleClosing();
228
+ }
229
+ /**
230
+ * Since we do not stack toasts, we force the closing on other existing opened ones
231
+ */
232
+ _closeOtherToasts() {
233
+ toastRefs.forEach((t) => {
234
+ if (t.state === "opened" || t.state === "opening") t.close();
235
+ });
236
+ }
237
+ _isLightMode() {
238
+ return !this._darkModeController.matches();
239
+ }
240
+ render() {
241
+ return html`
242
+ <div class="sbb-toast__overlay-container">
243
+ <div class="sbb-toast" @animationend=${this._onToastAnimationEnd}>
244
+ <div class="sbb-toast-wrapper">
245
+ ${this.renderIconSlot()}
246
+ <div class="sbb-toast__content">
247
+ <slot @slotchange=${this._onContentSlotChange}></slot>
248
+ </div>
249
+ <slot name="action" @slotchange=${this._syncSlottedElements}></slot>
250
+ </div>
251
+ <div class="sbb-toast__close">
252
+ <sbb-divider
253
+ class="sbb-toast__close-divider"
254
+ orientation="vertical"
255
+ ?negative=${this._isLightMode()}
256
+ ></sbb-divider>
257
+ ${!this.readOnly ? html`<sbb-transparent-button
258
+ class="sbb-toast__close-button"
259
+ icon-name="cross-small"
260
+ ?negative=${this._isLightMode()}
261
+ size="m"
262
+ aria-label=${i18nCloseAlert[this._language.current]}
263
+ sbb-toast-close
264
+ ></sbb-transparent-button>` : nothing}
265
+ </div>
266
+ </div>
267
+ </div>
268
+ `;
269
+ }
270
+ };
271
+ })();
272
+ //#endregion
273
+ export { SbbToastElement as t };
274
+
275
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"toast.component-IhHx0Qt3.js","names":[],"sources":["../../../src/elements/toast/toast.scss?inline","../../../src/elements/toast/toast.component.ts"],"sourcesContent":["@use '../core/styles' as sbb;\n\n:host {\n  --sbb-toast-animation-duration: var(\n    --sbb-disable-animation-duration,\n    var(--sbb-animation-duration-6x)\n  );\n  --sbb-focus-outline-color: var(--sbb-focus-outline-color-dark);\n\n  display: none;\n}\n\n:host(:is(:state(state-opening), :state(state-opened), :state(state-closing))) {\n  --sbb-toast-container-display: flex;\n\n  display: block;\n}\n\n:host([position='bottom-left']),\n:host([position='bottom-start']) {\n  --sbb-toast-horizontal-position: flex-start;\n  --sbb-toast-vertical-position: flex-end;\n}\n\n:host([position='bottom-center']) {\n  --sbb-toast-horizontal-position: center;\n  --sbb-toast-vertical-position: flex-end;\n}\n\n:host([position='bottom-right']),\n:host([position='bottom-end']) {\n  --sbb-toast-horizontal-position: flex-end;\n  --sbb-toast-vertical-position: flex-end;\n}\n\n:host([position='top-left']),\n:host([position='top-start']) {\n  --sbb-toast-horizontal-position: flex-start;\n  --sbb-toast-vertical-position: flex-start;\n}\n\n:host([position='top-center']) {\n  --sbb-toast-horizontal-position: center;\n  --sbb-toast-vertical-position: flex-start;\n}\n\n:host([position='top-right']),\n:host([position='top-end']) {\n  --sbb-toast-horizontal-position: flex-end;\n  --sbb-toast-vertical-position: flex-start;\n}\n\n:host(:state(dark)) {\n  --sbb-focus-outline-color: var(--sbb-focus-outline-color-default);\n}\n\n.sbb-toast__overlay-container {\n  position: var(--sbb-toast-container-position);\n  pointer-events: none;\n  inset: 0;\n  display: var(--sbb-toast-container-display, none);\n  justify-content: var(--sbb-toast-horizontal-position);\n  align-items: var(--sbb-toast-vertical-position);\n}\n\n.sbb-toast {\n  z-index: var(--sbb-toast-z-index, var(--sbb-overlay-default-z-index));\n  pointer-events: auto;\n  margin: var(--sbb-toast-margin);\n  border-radius: var(--sbb-toast-border-radius);\n  max-width: var(--sbb-toast-max-width);\n  display: flex;\n  color: var(--sbb-toast-text-color);\n  background-color: var(--sbb-toast-background-color);\n\n  :host(:state(state-opening)) & {\n    animation: {\n      name: open;\n      duration: var(--sbb-toast-animation-duration);\n      timing-function: var(--sbb-toast-animation-timing-function);\n    }\n  }\n\n  :host(:state(state-closing)) & {\n    animation: {\n      name: close;\n      duration: var(--sbb-toast-animation-duration);\n      timing-function: var(--sbb-toast-animation-timing-function);\n    }\n  }\n}\n\n.sbb-toast-wrapper {\n  display: flex;\n  align-items: center;\n  gap: var(--sbb-toast-gap);\n  padding-block: var(--sbb-toast-padding-block);\n  padding-inline: var(--sbb-toast-padding-inline);\n}\n\nslot:not([name='icon']),\n.sbb-toast__close {\n  // As the slotted content is displayed on a white background, we need to force light mode for them.\n  color-scheme: only light;\n}\n\n.sbb-toast__content {\n  @include sbb.title-6($exclude-spacing: true);\n\n  flex: 1 1 0;\n}\n\n.sbb-toast__close {\n  display: flex;\n  align-items: center;\n}\n\n.sbb-toast__close-divider {\n  padding-block: var(--sbb-toast-padding-block);\n}\n\n.sbb-toast__close-button {\n  margin-inline: var(--sbb-spacing-responsive-xxxs);\n}\n\n@keyframes open {\n  from {\n    visibility: hidden;\n    opacity: 0;\n    translate: 0 var(--sbb-spacing-fixed-2x);\n  }\n\n  to {\n    visibility: visible;\n    opacity: 1;\n    translate: 0;\n  }\n}\n\n@keyframes close {\n  from {\n    visibility: visible;\n    opacity: 1;\n    translate: 0;\n  }\n\n  to {\n    visibility: hidden;\n    opacity: 0;\n    translate: 0 var(--sbb-spacing-fixed-2x);\n  }\n}\n","import {\n  html,\n  nothing,\n  unsafeCSS,\n  type CSSResultGroup,\n  type PropertyValues,\n  type TemplateResult,\n} from 'lit';\nimport { property } from 'lit/decorators.js';\n\nimport {\n  SbbTransparentButtonElement,\n  type SbbTransparentButtonLinkElement,\n} from '../button.pure.ts';\nimport { SbbOpenCloseBaseElement } from '../core/base-elements.ts';\nimport { SbbDarkModeController, SbbLanguageController } from '../core/controllers.ts';\nimport { forceType } from '../core/decorators.ts';\nimport { isLean, isZeroAnimationDuration } from '../core/dom.ts';\nimport { composedPathHasAttribute } from '../core/eventing.ts';\nimport { i18nCloseAlert } from '../core/i18n.ts';\nimport { SbbReadonlyMixin } from '../core/mixins.ts';\nimport { boxSizingStyles } from '../core/styles.ts';\nimport { SbbIconNameMixin } from '../icon.ts';\nimport type { SbbLinkButtonElement, SbbLinkElement, SbbLinkStaticElement } from '../link.ts';\n\nimport style from './toast.scss?inline';\n\nimport '../divider.ts';\n\n// TODO(breaking-change): Remove call to define.\nSbbTransparentButtonElement.define();\n\ntype SbbToastPositionVertical = 'top' | 'bottom';\ntype SbbToastPositionHorizontal = 'left' | 'start' | 'center' | 'right' | 'end';\nexport type SbbToastPosition = `${SbbToastPositionVertical}-${SbbToastPositionHorizontal}`;\n\n// A global collection of existing toasts\nconst toastRefs = new Set<SbbToastElement>();\n\n/**\n * It displays a toast notification.\n *\n * @slot - Use the unnamed slot to add content to the `sbb-toast`.\n * @slot icon - Assign a custom icon via slot.\n * @slot action - Provide a custom action for this toast.\n * @cssprop [--sbb-toast-z-index=var(--sbb-overlay-default-z-index)] - To specify a custom stack order,\n * the `z-index` can be overridden by defining this CSS variable. The default `z-index` of the\n * component is set to `var(--sbb-overlay-default-z-index)` with a value of `1000`.\n */\nexport class SbbToastElement extends SbbIconNameMixin(SbbReadonlyMixin(SbbOpenCloseBaseElement)) {\n  public static override readonly elementName: string = 'sbb-toast';\n  public static override styles: CSSResultGroup = [boxSizingStyles, unsafeCSS(style)];\n\n  /**\n   * The length of time in milliseconds to wait before automatically dismissing the toast.\n   * If 0 (default), it stays open indefinitely.\n   * From accessibility perspective, it is recommended to set a timeout of at least 20 seconds.\n   */\n  @forceType()\n  @property({ type: Number })\n  public accessor timeout: number = 0;\n\n  /** The position where to place the toast. */\n  @property({ reflect: true }) public accessor position: SbbToastPosition = 'bottom-center';\n\n  /**\n   * The ARIA politeness level.\n   * Check https://developer.mozilla.org/en-US/docs/Web/Accessibility/ARIA/ARIA_Live_Regions#live_regions for further info\n   */\n  @property() public accessor politeness: 'polite' | 'assertive' | 'off' = 'polite';\n\n  private _closeTimeout?: ReturnType<typeof setTimeout>;\n  private _language = new SbbLanguageController(this);\n  private _darkModeController = new SbbDarkModeController(this, () => {\n    this._syncSlottedElements();\n    this.requestUpdate();\n  });\n\n  public constructor() {\n    super();\n    this.addEventListener?.('click', (e) => this._onClick(e));\n  }\n\n  public override connectedCallback(): void {\n    this.popover = 'manual';\n    super.connectedCallback();\n\n    // Add this toast to the global collection\n    toastRefs.add(this);\n  }\n\n  public override disconnectedCallback(): void {\n    super.disconnectedCallback();\n    clearTimeout(this._closeTimeout);\n\n    // Remove this instance\n    toastRefs.delete(this);\n  }\n\n  protected override willUpdate(changedProperties: PropertyValues<this>): void {\n    super.willUpdate(changedProperties);\n\n    if (changedProperties.has('politeness')) {\n      this.internals.ariaLive = this.politeness;\n    }\n  }\n\n  /**\n   * Open the toast.\n   * If there are other opened toasts in the page, close them first.\n   */\n  public open(): void {\n    if (this.state === 'opened' || this.state === 'opening' || !this.dispatchBeforeOpenEvent()) {\n      return;\n    }\n\n    this._closeOtherToasts();\n    this.showPopover?.();\n    this.state = 'opening';\n\n    // If the animation duration is zero, the animationend event is not always fired reliably.\n    // In this case we directly set the `opened` state.\n    if (this._isZeroAnimationDuration()) {\n      this._handleOpening();\n    }\n  }\n\n  /**\n   * Close the toast.\n   */\n  public close(): void {\n    if (this.state === 'closed' || this.state === 'closing' || !this.dispatchBeforeCloseEvent()) {\n      return;\n    }\n\n    clearTimeout(this._closeTimeout);\n    this.state = 'closing';\n\n    // If the animation duration is zero, the animationend event is not always fired reliably.\n    // In this case we directly set the `closed` state.\n    if (this._isZeroAnimationDuration()) {\n      this._handleClosing();\n    }\n  }\n\n  private _handleClosing(): void {\n    this.state = 'closed';\n    this.hidePopover?.();\n    this.dispatchCloseEvent();\n  }\n\n  private _handleOpening(): void {\n    this.state = 'opened';\n    this.dispatchOpenEvent();\n\n    // Start the countdown to close it\n    if (this.timeout) {\n      // Workaround for https://github.com/sbb-design-systems/lyne-angular/issues/190\n      // If zone.js is loaded, setTimeout is wrapped and tracked which will mark\n      // anything in Angular as unstable as long as setTimeout is not finished.\n      // This only needs to be fixed in places where we actually want to wait a\n      // specific amount of time without an interaction (e.g. for this case).\n      const global = globalThis as any;\n      const setTimeout: typeof globalThis.setTimeout =\n        global[global.Zone?.__symbol__?.('setTimeout') as string] ?? global.setTimeout;\n\n      this._closeTimeout = setTimeout(() => this.close(), this.timeout);\n    }\n  }\n\n  // Close the toast on click of any element that has the 'sbb-toast-close' attribute.\n  private _onClick(event: Event): void {\n    const closeElement = composedPathHasAttribute(event, 'sbb-toast-close', this);\n\n    if (closeElement && !closeElement.hasAttribute('disabled')) {\n      this.close();\n    }\n  }\n\n  private _isZeroAnimationDuration(): boolean {\n    return isZeroAnimationDuration(this, '--sbb-toast-animation-duration');\n  }\n\n  /**\n   * Slotted text nodes are not read correctly by screen readers on Chrome.\n   * To address the problem, if there is at least a root text node,\n   * we wrap the whole content in a <span> tag\n   */\n  private _onContentSlotChange(event: Event): void {\n    const slotNodes = (event.target as HTMLSlotElement).assignedNodes();\n\n    if (slotNodes.some((el) => el.nodeType === Node.TEXT_NODE)) {\n      const span = document.createElement('span');\n      this.prepend(span);\n      span.append(...slotNodes);\n    }\n    this._syncSlottedElements();\n  }\n\n  private _syncSlottedElements(): void {\n    // Force the visual state on slotted buttons\n    this.querySelectorAll<SbbTransparentButtonElement | SbbTransparentButtonLinkElement>(\n      'sbb-transparent-button, sbb-transparent-button-link',\n    ).forEach((btn) => {\n      btn.negative = this._isLightMode();\n      btn.size = isLean() ? 's' : 'm';\n    });\n\n    // Force negative on slotted links\n    this.querySelectorAll<SbbLinkElement | SbbLinkButtonElement | SbbLinkStaticElement>(\n      'sbb-link, sbb-link-button, sbb-link-static',\n    ).forEach((link) => {\n      link.negative = this._isLightMode();\n    });\n  }\n\n  // In rare cases it can be that the animationEnd event is triggered twice.\n  // To avoid entering a corrupt state, exit when state is not expected.\n  private _onToastAnimationEnd(event: AnimationEvent): void {\n    // On toast opened\n    if (event.animationName === 'open' && this.state === 'opening') {\n      this._handleOpening();\n    } else if (event.animationName === 'close' && this.state === 'closing') {\n      this._handleClosing();\n    }\n  }\n\n  /**\n   * Since we do not stack toasts, we force the closing on other existing opened ones\n   */\n  private _closeOtherToasts(): void {\n    toastRefs.forEach((t) => {\n      if (t.state === 'opened' || t.state === 'opening') {\n        t.close();\n      }\n    });\n  }\n\n  private _isLightMode(): boolean {\n    return !this._darkModeController.matches();\n  }\n\n  protected override render(): TemplateResult {\n    return html`\n      <div class=\"sbb-toast__overlay-container\">\n        <div class=\"sbb-toast\" @animationend=${this._onToastAnimationEnd}>\n          <div class=\"sbb-toast-wrapper\">\n            ${this.renderIconSlot()}\n            <div class=\"sbb-toast__content\">\n              <slot @slotchange=${this._onContentSlotChange}></slot>\n            </div>\n            <slot name=\"action\" @slotchange=${this._syncSlottedElements}></slot>\n          </div>\n          <div class=\"sbb-toast__close\">\n            <sbb-divider\n              class=\"sbb-toast__close-divider\"\n              orientation=\"vertical\"\n              ?negative=${this._isLightMode()}\n            ></sbb-divider>\n            ${!this.readOnly\n              ? html`<sbb-transparent-button\n                  class=\"sbb-toast__close-button\"\n                  icon-name=\"cross-small\"\n                  ?negative=${this._isLightMode()}\n                  size=\"m\"\n                  aria-label=${i18nCloseAlert[this._language.current]}\n                  sbb-toast-close\n                ></sbb-transparent-button>`\n              : nothing}\n          </div>\n        </div>\n      </div>\n    `;\n  }\n}\n\ndeclare global {\n  interface HTMLElementTagNameMap {\n    // eslint-disable-next-line @typescript-eslint/naming-convention\n    'sbb-toast': SbbToastElement;\n  }\n}\n"],"mappings":"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"}
@@ -1,4 +1,4 @@
1
- import { t as SbbToastElement } from "./toast.component-Cgr5ZGOr.js";
1
+ import { t as SbbToastElement } from "./toast.component-IhHx0Qt3.js";
2
2
  import "./toast.pure.js";
3
3
  //#region src/elements/toast.ts
4
4
  /** @entrypoint */
@@ -1,2 +1,2 @@
1
- import { t as SbbToastElement } from "./toast.component-Cgr5ZGOr.js";
1
+ import { t as SbbToastElement } from "./toast.component-IhHx0Qt3.js";
2
2
  export { SbbToastElement };
@@ -1959,6 +1959,18 @@ slot[name=error]::slotted(*) {
1959
1959
  --sbb-overlay-position-try-fallbacks:
1960
1960
  block-start span-inline-end, block-start span-inline-start, block-end,
1961
1961
  block-end span-inline-end, block-end span-inline-start;
1962
+ --sbb-toast-max-width: 25rem;
1963
+ --sbb-toast-animation-timing-function: ease;
1964
+ --sbb-toast-text-color: var(--sbb-color-3-inverted);
1965
+ --sbb-toast-background-color: var(--sbb-background-color-1-inverted);
1966
+ --sbb-toast-margin: var(--sbb-spacing-responsive-s);
1967
+ --sbb-toast-padding-block: var(--sbb-spacing-responsive-xxxs);
1968
+ --sbb-toast-padding-inline: var(--sbb-spacing-responsive-xs);
1969
+ --sbb-toast-border-radius: var(--sbb-border-radius-4x);
1970
+ --sbb-toast-container-position: fixed;
1971
+ --sbb-toast-horizontal-position: initial;
1972
+ --sbb-toast-vertical-position: initial;
1973
+ --sbb-toast-gap: var(--sbb-spacing-fixed-2x);
1962
1974
  --sbb-visual-checkbox-dimension: var(--sbb-checkbox-dimension-m);
1963
1975
  --sbb-visual-checkbox-selection-color: var(--sbb-color-primary);
1964
1976
  --sbb-visual-checkbox-selection-color-negative: var(--sbb-color-primary85);
package/package.json CHANGED
@@ -1,12 +1,12 @@
1
1
  {
2
2
  "name": "@sbb-esta/lyne-elements-dev",
3
- "version": "4.11.0-dev.1777283963",
3
+ "version": "4.12.0-dev.1777289332",
4
4
  "description": "Lyne Design System",
5
5
  "keywords": [
6
6
  "design system",
7
7
  "web components",
8
8
  "lit",
9
- "https://github.com/sbb-design-systems/lyne-components/commit/64eae912da9a5e3799fbd38bdcf4cdc939140d02"
9
+ "https://github.com/sbb-design-systems/lyne-components/commit/ebb76c66e4dfd356b75a8307023b739fff832cfb"
10
10
  ],
11
11
  "type": "module",
12
12
  "exports": {
package/safety-theme.css CHANGED
@@ -1959,6 +1959,18 @@ slot[name=error]::slotted(*) {
1959
1959
  --sbb-overlay-position-try-fallbacks:
1960
1960
  block-start span-inline-end, block-start span-inline-start, block-end,
1961
1961
  block-end span-inline-end, block-end span-inline-start;
1962
+ --sbb-toast-max-width: 25rem;
1963
+ --sbb-toast-animation-timing-function: ease;
1964
+ --sbb-toast-text-color: var(--sbb-color-3-inverted);
1965
+ --sbb-toast-background-color: var(--sbb-background-color-1-inverted);
1966
+ --sbb-toast-margin: var(--sbb-spacing-responsive-s);
1967
+ --sbb-toast-padding-block: var(--sbb-spacing-responsive-xxxs);
1968
+ --sbb-toast-padding-inline: var(--sbb-spacing-responsive-xs);
1969
+ --sbb-toast-border-radius: var(--sbb-border-radius-4x);
1970
+ --sbb-toast-container-position: fixed;
1971
+ --sbb-toast-horizontal-position: initial;
1972
+ --sbb-toast-vertical-position: initial;
1973
+ --sbb-toast-gap: var(--sbb-spacing-fixed-2x);
1962
1974
  --sbb-visual-checkbox-dimension: var(--sbb-checkbox-dimension-m);
1963
1975
  --sbb-visual-checkbox-selection-color: var(--sbb-color-primary);
1964
1976
  --sbb-visual-checkbox-selection-color-negative: var(--sbb-color-primary85);
@@ -1959,6 +1959,18 @@ slot[name=error]::slotted(*) {
1959
1959
  --sbb-overlay-position-try-fallbacks:
1960
1960
  block-start span-inline-end, block-start span-inline-start, block-end,
1961
1961
  block-end span-inline-end, block-end span-inline-start;
1962
+ --sbb-toast-max-width: 25rem;
1963
+ --sbb-toast-animation-timing-function: ease;
1964
+ --sbb-toast-text-color: var(--sbb-color-3-inverted);
1965
+ --sbb-toast-background-color: var(--sbb-background-color-1-inverted);
1966
+ --sbb-toast-margin: var(--sbb-spacing-responsive-s);
1967
+ --sbb-toast-padding-block: var(--sbb-spacing-responsive-xxxs);
1968
+ --sbb-toast-padding-inline: var(--sbb-spacing-responsive-xs);
1969
+ --sbb-toast-border-radius: var(--sbb-border-radius-4x);
1970
+ --sbb-toast-container-position: fixed;
1971
+ --sbb-toast-horizontal-position: initial;
1972
+ --sbb-toast-vertical-position: initial;
1973
+ --sbb-toast-gap: var(--sbb-spacing-fixed-2x);
1962
1974
  --sbb-visual-checkbox-dimension: var(--sbb-checkbox-dimension-m);
1963
1975
  --sbb-visual-checkbox-selection-color: var(--sbb-color-primary);
1964
1976
  --sbb-visual-checkbox-selection-color-negative: var(--sbb-color-primary85);
@@ -1,2 +1,2 @@
1
- import { t as e } from "../toast.component-CGVqkd0_.js";
1
+ import { t as e } from "../toast.component-CORnf2Tl.js";
2
2
  export { e as SbbToastElement };
@@ -13,7 +13,7 @@ import { i18nCloseAlert as g } from "./core/i18n.js";
13
13
  import { SbbIconNameMixin as _ } from "./icon.js";
14
14
  import "./divider.js";
15
15
  //#region src/elements/toast/toast.scss?inline
16
- var v = ":host{--sbb-toast-max-width: 25rem;--sbb-toast-animation-duration: var( --sbb-disable-animation-duration, var(--sbb-animation-duration-6x) );--sbb-toast-animation-timing-function: ease;--sbb-toast-text-color: var(--sbb-color-3-inverted);--sbb-toast-background-color: var(--sbb-background-color-1-inverted);--sbb-toast-margin: var(--sbb-spacing-responsive-s);--sbb-toast-padding-block: var(--sbb-spacing-responsive-xxxs);--sbb-toast-padding-inline: var(--sbb-spacing-responsive-xs);--sbb-toast-border-radius: var(--sbb-border-radius-4x);--sbb-toast-container-display: none;--sbb-toast-container-position: fixed;--sbb-toast-horizontal-position: initial;--sbb-toast-vertical-position: initial;--sbb-toast-gap: var(--sbb-spacing-fixed-2x);--sbb-focus-outline-color: var(--sbb-focus-outline-color-dark);display:none}:host(:is(:is(:state(state-opening),[state--state-opening]),:is(:state(state-opened),[state--state-opened]),:is(:state(state-closing),[state--state-closing]))){--sbb-toast-container-display: flex;display:block}:host([position=bottom-left]),:host([position=bottom-start]){--sbb-toast-horizontal-position: flex-start;--sbb-toast-vertical-position: flex-end}:host([position=bottom-center]){--sbb-toast-horizontal-position: center;--sbb-toast-vertical-position: flex-end}:host([position=bottom-right]),:host([position=bottom-end]){--sbb-toast-horizontal-position: flex-end;--sbb-toast-vertical-position: flex-end}:host([position=top-left]),:host([position=top-start]){--sbb-toast-horizontal-position: flex-start;--sbb-toast-vertical-position: flex-start}:host([position=top-center]){--sbb-toast-horizontal-position: center;--sbb-toast-vertical-position: flex-start}:host([position=top-right]),:host([position=top-end]){--sbb-toast-horizontal-position: flex-end;--sbb-toast-vertical-position: flex-start}:host(:is(:state(dark),[state--dark])){--sbb-focus-outline-color: var(--sbb-focus-outline-color-default)}.sbb-toast__overlay-container{position:var(--sbb-toast-container-position);pointer-events:none;inset:0;display:var(--sbb-toast-container-display);justify-content:var(--sbb-toast-horizontal-position);align-items:var(--sbb-toast-vertical-position)}.sbb-toast{z-index:var(--sbb-toast-z-index, var(--sbb-overlay-default-z-index));pointer-events:auto;margin:var(--sbb-toast-margin);border-radius:var(--sbb-toast-border-radius);max-width:var(--sbb-toast-max-width);display:flex;color:var(--sbb-toast-text-color);background-color:var(--sbb-toast-background-color)}:host(:is(:state(state-opening),[state--state-opening])) .sbb-toast{animation-name:open;animation-duration:var(--sbb-toast-animation-duration);animation-timing-function:var(--sbb-toast-animation-timing-function)}:host(:is(:state(state-closing),[state--state-closing])) .sbb-toast{animation-name:close;animation-duration:var(--sbb-toast-animation-duration);animation-timing-function:var(--sbb-toast-animation-timing-function)}.sbb-toast-wrapper{display:flex;align-items:center;gap:var(--sbb-toast-gap);padding-block:var(--sbb-toast-padding-block);padding-inline:var(--sbb-toast-padding-inline)}slot:not([name=icon]),.sbb-toast__close{color-scheme:only light}.sbb-toast__content{--sbb-title-font-size: var(--sbb-title-font-size-level-6-lean, var(--sbb-heading-font-size-6));--sbb-title-line-height: var(--sbb-typo-line-height-text);margin-block:var(--sbb-title-margin-block, 0);margin-inline:0;font-family:var(--sbb-typo-font-family);font-weight:700;font-size:var(--sbb-title-font-size);letter-spacing:var(--sbb-typo-letter-spacing-heading);line-height:var(--sbb-title-line-height, var(--sbb-typo-line-height-heading));flex:1 1 0}.sbb-toast__close{display:flex;align-items:center}.sbb-toast__close-divider{padding-block:var(--sbb-toast-padding-block)}.sbb-toast__close-button{margin-inline:var(--sbb-spacing-responsive-xxxs)}@keyframes open{0%{visibility:hidden;opacity:0;translate:0 var(--sbb-spacing-fixed-2x)}to{visibility:visible;opacity:1;translate:0}}@keyframes close{0%{visibility:visible;opacity:1;translate:0}to{visibility:hidden;opacity:0;translate:0 var(--sbb-spacing-fixed-2x)}}";
16
+ var v = ":host{--sbb-toast-animation-duration: var( --sbb-disable-animation-duration, var(--sbb-animation-duration-6x) );--sbb-focus-outline-color: var(--sbb-focus-outline-color-dark);display:none}:host(:is(:is(:state(state-opening),[state--state-opening]),:is(:state(state-opened),[state--state-opened]),:is(:state(state-closing),[state--state-closing]))){--sbb-toast-container-display: flex;display:block}:host([position=bottom-left]),:host([position=bottom-start]){--sbb-toast-horizontal-position: flex-start;--sbb-toast-vertical-position: flex-end}:host([position=bottom-center]){--sbb-toast-horizontal-position: center;--sbb-toast-vertical-position: flex-end}:host([position=bottom-right]),:host([position=bottom-end]){--sbb-toast-horizontal-position: flex-end;--sbb-toast-vertical-position: flex-end}:host([position=top-left]),:host([position=top-start]){--sbb-toast-horizontal-position: flex-start;--sbb-toast-vertical-position: flex-start}:host([position=top-center]){--sbb-toast-horizontal-position: center;--sbb-toast-vertical-position: flex-start}:host([position=top-right]),:host([position=top-end]){--sbb-toast-horizontal-position: flex-end;--sbb-toast-vertical-position: flex-start}:host(:is(:state(dark),[state--dark])){--sbb-focus-outline-color: var(--sbb-focus-outline-color-default)}.sbb-toast__overlay-container{position:var(--sbb-toast-container-position);pointer-events:none;inset:0;display:var(--sbb-toast-container-display, none);justify-content:var(--sbb-toast-horizontal-position);align-items:var(--sbb-toast-vertical-position)}.sbb-toast{z-index:var(--sbb-toast-z-index, var(--sbb-overlay-default-z-index));pointer-events:auto;margin:var(--sbb-toast-margin);border-radius:var(--sbb-toast-border-radius);max-width:var(--sbb-toast-max-width);display:flex;color:var(--sbb-toast-text-color);background-color:var(--sbb-toast-background-color)}:host(:is(:state(state-opening),[state--state-opening])) .sbb-toast{animation-name:open;animation-duration:var(--sbb-toast-animation-duration);animation-timing-function:var(--sbb-toast-animation-timing-function)}:host(:is(:state(state-closing),[state--state-closing])) .sbb-toast{animation-name:close;animation-duration:var(--sbb-toast-animation-duration);animation-timing-function:var(--sbb-toast-animation-timing-function)}.sbb-toast-wrapper{display:flex;align-items:center;gap:var(--sbb-toast-gap);padding-block:var(--sbb-toast-padding-block);padding-inline:var(--sbb-toast-padding-inline)}slot:not([name=icon]),.sbb-toast__close{color-scheme:only light}.sbb-toast__content{--sbb-title-font-size: var(--sbb-title-font-size-level-6-lean, var(--sbb-heading-font-size-6));--sbb-title-line-height: var(--sbb-typo-line-height-text);margin-block:var(--sbb-title-margin-block, 0);margin-inline:0;font-family:var(--sbb-typo-font-family);font-weight:700;font-size:var(--sbb-title-font-size);letter-spacing:var(--sbb-typo-letter-spacing-heading);line-height:var(--sbb-title-line-height, var(--sbb-typo-line-height-heading));flex:1 1 0}.sbb-toast__close{display:flex;align-items:center}.sbb-toast__close-divider{padding-block:var(--sbb-toast-padding-block)}.sbb-toast__close-button{margin-inline:var(--sbb-spacing-responsive-xxxs)}@keyframes open{0%{visibility:hidden;opacity:0;translate:0 var(--sbb-spacing-fixed-2x)}to{visibility:visible;opacity:1;translate:0}}@keyframes close{0%{visibility:visible;opacity:1;translate:0}to{visibility:hidden;opacity:0;translate:0 var(--sbb-spacing-fixed-2x)}}";
17
17
  //#endregion
18
18
  //#region src/elements/toast/toast.component.ts
19
19
  p.define();
package/toast.js CHANGED
@@ -1,4 +1,4 @@
1
- import { t as e } from "./toast.component-CGVqkd0_.js";
1
+ import { t as e } from "./toast.component-CORnf2Tl.js";
2
2
  import "./toast.pure.js";
3
3
  //#region src/elements/toast.ts
4
4
  e.define();
package/toast.pure.js CHANGED
@@ -1,2 +1,2 @@
1
- import { t as e } from "./toast.component-CGVqkd0_.js";
1
+ import { t as e } from "./toast.component-CORnf2Tl.js";
2
2
  export { e as SbbToastElement };
@@ -1,275 +0,0 @@
1
- import { __esDecorate, __runInitializers } from "tslib";
2
- import { html, nothing, unsafeCSS } from "lit";
3
- import { property } from "lit/decorators.js";
4
- import { SbbOpenCloseBaseElement } from "./core/base-elements.js";
5
- import { forceType } from "./core/decorators.js";
6
- import { isLean, isZeroAnimationDuration } from "./core/dom.js";
7
- import { composedPathHasAttribute } from "./core/eventing.js";
8
- import { SbbReadonlyMixin } from "./core/mixins.js";
9
- import { boxSizingStyles } from "./core/styles.js";
10
- import { SbbTransparentButtonElement } from "./button.pure.js";
11
- import { SbbDarkModeController, SbbLanguageController } from "./core/controllers.js";
12
- import { i18nCloseAlert } from "./core/i18n.js";
13
- import { SbbIconNameMixin } from "./icon.js";
14
- import "./divider.js";
15
- //#region src/elements/toast/toast.scss?inline
16
- var toast_default = ":host {\n --sbb-toast-max-width: 25rem;\n --sbb-toast-animation-duration: var(\n --sbb-disable-animation-duration,\n var(--sbb-animation-duration-6x)\n );\n --sbb-toast-animation-timing-function: ease;\n --sbb-toast-text-color: var(--sbb-color-3-inverted);\n --sbb-toast-background-color: var(--sbb-background-color-1-inverted);\n --sbb-toast-margin: var(--sbb-spacing-responsive-s);\n --sbb-toast-padding-block: var(--sbb-spacing-responsive-xxxs);\n --sbb-toast-padding-inline: var(--sbb-spacing-responsive-xs);\n --sbb-toast-border-radius: var(--sbb-border-radius-4x);\n --sbb-toast-container-display: none;\n --sbb-toast-container-position: fixed;\n --sbb-toast-horizontal-position: initial;\n --sbb-toast-vertical-position: initial;\n --sbb-toast-gap: var(--sbb-spacing-fixed-2x);\n --sbb-focus-outline-color: var(--sbb-focus-outline-color-dark);\n display: none;\n}\n\n:host(:is(:is(:state(state-opening),[state--state-opening]), :is(:state(state-opened),[state--state-opened]), :is(:state(state-closing),[state--state-closing]))) {\n --sbb-toast-container-display: flex;\n display: block;\n}\n\n:host([position=bottom-left]),\n:host([position=bottom-start]) {\n --sbb-toast-horizontal-position: flex-start;\n --sbb-toast-vertical-position: flex-end;\n}\n\n:host([position=bottom-center]) {\n --sbb-toast-horizontal-position: center;\n --sbb-toast-vertical-position: flex-end;\n}\n\n:host([position=bottom-right]),\n:host([position=bottom-end]) {\n --sbb-toast-horizontal-position: flex-end;\n --sbb-toast-vertical-position: flex-end;\n}\n\n:host([position=top-left]),\n:host([position=top-start]) {\n --sbb-toast-horizontal-position: flex-start;\n --sbb-toast-vertical-position: flex-start;\n}\n\n:host([position=top-center]) {\n --sbb-toast-horizontal-position: center;\n --sbb-toast-vertical-position: flex-start;\n}\n\n:host([position=top-right]),\n:host([position=top-end]) {\n --sbb-toast-horizontal-position: flex-end;\n --sbb-toast-vertical-position: flex-start;\n}\n\n:host(:is(:state(dark),[state--dark])) {\n --sbb-focus-outline-color: var(--sbb-focus-outline-color-default);\n}\n\n.sbb-toast__overlay-container {\n position: var(--sbb-toast-container-position);\n pointer-events: none;\n inset: 0;\n display: var(--sbb-toast-container-display);\n justify-content: var(--sbb-toast-horizontal-position);\n align-items: var(--sbb-toast-vertical-position);\n}\n\n.sbb-toast {\n z-index: var(--sbb-toast-z-index, var(--sbb-overlay-default-z-index));\n pointer-events: auto;\n margin: var(--sbb-toast-margin);\n border-radius: var(--sbb-toast-border-radius);\n max-width: var(--sbb-toast-max-width);\n display: flex;\n color: var(--sbb-toast-text-color);\n background-color: var(--sbb-toast-background-color);\n}\n:host(:is(:state(state-opening),[state--state-opening])) .sbb-toast {\n animation-name: open;\n animation-duration: var(--sbb-toast-animation-duration);\n animation-timing-function: var(--sbb-toast-animation-timing-function);\n}\n:host(:is(:state(state-closing),[state--state-closing])) .sbb-toast {\n animation-name: close;\n animation-duration: var(--sbb-toast-animation-duration);\n animation-timing-function: var(--sbb-toast-animation-timing-function);\n}\n\n.sbb-toast-wrapper {\n display: flex;\n align-items: center;\n gap: var(--sbb-toast-gap);\n padding-block: var(--sbb-toast-padding-block);\n padding-inline: var(--sbb-toast-padding-inline);\n}\n\nslot:not([name=icon]),\n.sbb-toast__close {\n color-scheme: only light;\n}\n\n.sbb-toast__content {\n --sbb-title-font-size: var(--sbb-title-font-size-level-6-lean, var(--sbb-heading-font-size-6));\n --sbb-title-line-height: var(--sbb-typo-line-height-text);\n margin-block: var(--sbb-title-margin-block, 0);\n margin-inline: 0;\n font-family: var(--sbb-typo-font-family);\n font-weight: bold;\n font-size: var(--sbb-title-font-size);\n letter-spacing: var(--sbb-typo-letter-spacing-heading);\n line-height: var(--sbb-title-line-height, var(--sbb-typo-line-height-heading));\n flex: 1 1 0;\n}\n\n.sbb-toast__close {\n display: flex;\n align-items: center;\n}\n\n.sbb-toast__close-divider {\n padding-block: var(--sbb-toast-padding-block);\n}\n\n.sbb-toast__close-button {\n margin-inline: var(--sbb-spacing-responsive-xxxs);\n}\n\n@keyframes open {\n from {\n visibility: hidden;\n opacity: 0;\n translate: 0 var(--sbb-spacing-fixed-2x);\n }\n to {\n visibility: visible;\n opacity: 1;\n translate: 0;\n }\n}\n@keyframes close {\n from {\n visibility: visible;\n opacity: 1;\n translate: 0;\n }\n to {\n visibility: hidden;\n opacity: 0;\n translate: 0 var(--sbb-spacing-fixed-2x);\n }\n}";
17
- //#endregion
18
- //#region src/elements/toast/toast.component.ts
19
- SbbTransparentButtonElement.define();
20
- var toastRefs = /* @__PURE__ */ new Set();
21
- /**
22
- * It displays a toast notification.
23
- *
24
- * @slot - Use the unnamed slot to add content to the `sbb-toast`.
25
- * @slot icon - Assign a custom icon via slot.
26
- * @slot action - Provide a custom action for this toast.
27
- * @cssprop [--sbb-toast-z-index=var(--sbb-overlay-default-z-index)] - To specify a custom stack order,
28
- * the `z-index` can be overridden by defining this CSS variable. The default `z-index` of the
29
- * component is set to `var(--sbb-overlay-default-z-index)` with a value of `1000`.
30
- */
31
- var SbbToastElement = (() => {
32
- let _classSuper = SbbIconNameMixin(SbbReadonlyMixin(SbbOpenCloseBaseElement));
33
- let _timeout_decorators;
34
- let _timeout_initializers = [];
35
- let _timeout_extraInitializers = [];
36
- let _position_decorators;
37
- let _position_initializers = [];
38
- let _position_extraInitializers = [];
39
- let _politeness_decorators;
40
- let _politeness_initializers = [];
41
- let _politeness_extraInitializers = [];
42
- return class SbbToastElement extends _classSuper {
43
- static {
44
- const _metadata = typeof Symbol === "function" && Symbol.metadata ? Object.create(_classSuper[Symbol.metadata] ?? null) : void 0;
45
- _timeout_decorators = [forceType(), property({ type: Number })];
46
- _position_decorators = [property({ reflect: true })];
47
- _politeness_decorators = [property()];
48
- __esDecorate(this, null, _timeout_decorators, {
49
- kind: "accessor",
50
- name: "timeout",
51
- static: false,
52
- private: false,
53
- access: {
54
- has: (obj) => "timeout" in obj,
55
- get: (obj) => obj.timeout,
56
- set: (obj, value) => {
57
- obj.timeout = value;
58
- }
59
- },
60
- metadata: _metadata
61
- }, _timeout_initializers, _timeout_extraInitializers);
62
- __esDecorate(this, null, _position_decorators, {
63
- kind: "accessor",
64
- name: "position",
65
- static: false,
66
- private: false,
67
- access: {
68
- has: (obj) => "position" in obj,
69
- get: (obj) => obj.position,
70
- set: (obj, value) => {
71
- obj.position = value;
72
- }
73
- },
74
- metadata: _metadata
75
- }, _position_initializers, _position_extraInitializers);
76
- __esDecorate(this, null, _politeness_decorators, {
77
- kind: "accessor",
78
- name: "politeness",
79
- static: false,
80
- private: false,
81
- access: {
82
- has: (obj) => "politeness" in obj,
83
- get: (obj) => obj.politeness,
84
- set: (obj, value) => {
85
- obj.politeness = value;
86
- }
87
- },
88
- metadata: _metadata
89
- }, _politeness_initializers, _politeness_extraInitializers);
90
- if (_metadata) Object.defineProperty(this, Symbol.metadata, {
91
- enumerable: true,
92
- configurable: true,
93
- writable: true,
94
- value: _metadata
95
- });
96
- }
97
- static {
98
- this.elementName = "sbb-toast";
99
- }
100
- static {
101
- this.styles = [boxSizingStyles, unsafeCSS(toast_default)];
102
- }
103
- #timeout_accessor_storage;
104
- /**
105
- * The length of time in milliseconds to wait before automatically dismissing the toast.
106
- * If 0 (default), it stays open indefinitely.
107
- * From accessibility perspective, it is recommended to set a timeout of at least 20 seconds.
108
- */
109
- get timeout() {
110
- return this.#timeout_accessor_storage;
111
- }
112
- set timeout(value) {
113
- this.#timeout_accessor_storage = value;
114
- }
115
- #position_accessor_storage;
116
- /** The position where to place the toast. */
117
- get position() {
118
- return this.#position_accessor_storage;
119
- }
120
- set position(value) {
121
- this.#position_accessor_storage = value;
122
- }
123
- #politeness_accessor_storage;
124
- /**
125
- * The ARIA politeness level.
126
- * Check https://developer.mozilla.org/en-US/docs/Web/Accessibility/ARIA/ARIA_Live_Regions#live_regions for further info
127
- */
128
- get politeness() {
129
- return this.#politeness_accessor_storage;
130
- }
131
- set politeness(value) {
132
- this.#politeness_accessor_storage = value;
133
- }
134
- constructor() {
135
- super();
136
- this.#timeout_accessor_storage = __runInitializers(this, _timeout_initializers, 0);
137
- this.#position_accessor_storage = (__runInitializers(this, _timeout_extraInitializers), __runInitializers(this, _position_initializers, "bottom-center"));
138
- this.#politeness_accessor_storage = (__runInitializers(this, _position_extraInitializers), __runInitializers(this, _politeness_initializers, "polite"));
139
- this._closeTimeout = __runInitializers(this, _politeness_extraInitializers);
140
- this._language = new SbbLanguageController(this);
141
- this._darkModeController = new SbbDarkModeController(this, () => {
142
- this._syncSlottedElements();
143
- this.requestUpdate();
144
- });
145
- this.addEventListener?.("click", (e) => this._onClick(e));
146
- }
147
- connectedCallback() {
148
- this.popover = "manual";
149
- super.connectedCallback();
150
- toastRefs.add(this);
151
- }
152
- disconnectedCallback() {
153
- super.disconnectedCallback();
154
- clearTimeout(this._closeTimeout);
155
- toastRefs.delete(this);
156
- }
157
- willUpdate(changedProperties) {
158
- super.willUpdate(changedProperties);
159
- if (changedProperties.has("politeness")) this.internals.ariaLive = this.politeness;
160
- }
161
- /**
162
- * Open the toast.
163
- * If there are other opened toasts in the page, close them first.
164
- */
165
- open() {
166
- if (this.state === "opened" || this.state === "opening" || !this.dispatchBeforeOpenEvent()) return;
167
- this._closeOtherToasts();
168
- this.showPopover?.();
169
- this.state = "opening";
170
- if (this._isZeroAnimationDuration()) this._handleOpening();
171
- }
172
- /**
173
- * Close the toast.
174
- */
175
- close() {
176
- if (this.state === "closed" || this.state === "closing" || !this.dispatchBeforeCloseEvent()) return;
177
- clearTimeout(this._closeTimeout);
178
- this.state = "closing";
179
- if (this._isZeroAnimationDuration()) this._handleClosing();
180
- }
181
- _handleClosing() {
182
- this.state = "closed";
183
- this.hidePopover?.();
184
- this.dispatchCloseEvent();
185
- }
186
- _handleOpening() {
187
- this.state = "opened";
188
- this.dispatchOpenEvent();
189
- if (this.timeout) {
190
- const global = globalThis;
191
- const setTimeout = global[global.Zone?.__symbol__?.("setTimeout")] ?? global.setTimeout;
192
- this._closeTimeout = setTimeout(() => this.close(), this.timeout);
193
- }
194
- }
195
- _onClick(event) {
196
- const closeElement = composedPathHasAttribute(event, "sbb-toast-close", this);
197
- if (closeElement && !closeElement.hasAttribute("disabled")) this.close();
198
- }
199
- _isZeroAnimationDuration() {
200
- return isZeroAnimationDuration(this, "--sbb-toast-animation-duration");
201
- }
202
- /**
203
- * Slotted text nodes are not read correctly by screen readers on Chrome.
204
- * To address the problem, if there is at least a root text node,
205
- * we wrap the whole content in a <span> tag
206
- */
207
- _onContentSlotChange(event) {
208
- const slotNodes = event.target.assignedNodes();
209
- if (slotNodes.some((el) => el.nodeType === Node.TEXT_NODE)) {
210
- const span = document.createElement("span");
211
- this.prepend(span);
212
- span.append(...slotNodes);
213
- }
214
- this._syncSlottedElements();
215
- }
216
- _syncSlottedElements() {
217
- this.querySelectorAll("sbb-transparent-button, sbb-transparent-button-link").forEach((btn) => {
218
- btn.negative = this._isLightMode();
219
- btn.size = isLean() ? "s" : "m";
220
- });
221
- this.querySelectorAll("sbb-link, sbb-link-button, sbb-link-static").forEach((link) => {
222
- link.negative = this._isLightMode();
223
- });
224
- }
225
- _onToastAnimationEnd(event) {
226
- if (event.animationName === "open" && this.state === "opening") this._handleOpening();
227
- else if (event.animationName === "close" && this.state === "closing") this._handleClosing();
228
- }
229
- /**
230
- * Since we do not stack toasts, we force the closing on other existing opened ones
231
- */
232
- _closeOtherToasts() {
233
- toastRefs.forEach((t) => {
234
- if (t.state === "opened" || t.state === "opening") t.close();
235
- });
236
- }
237
- _isLightMode() {
238
- return !this._darkModeController.matches();
239
- }
240
- render() {
241
- return html`
242
- <div class="sbb-toast__overlay-container">
243
- <div class="sbb-toast" @animationend=${this._onToastAnimationEnd}>
244
- <div class="sbb-toast-wrapper">
245
- ${this.renderIconSlot()}
246
- <div class="sbb-toast__content">
247
- <slot @slotchange=${this._onContentSlotChange}></slot>
248
- </div>
249
- <slot name="action" @slotchange=${this._syncSlottedElements}></slot>
250
- </div>
251
- <div class="sbb-toast__close">
252
- <sbb-divider
253
- class="sbb-toast__close-divider"
254
- orientation="vertical"
255
- ?negative=${this._isLightMode()}
256
- ></sbb-divider>
257
- ${!this.readOnly ? html`<sbb-transparent-button
258
- class="sbb-toast__close-button"
259
- icon-name="cross-small"
260
- ?negative=${this._isLightMode()}
261
- size="m"
262
- aria-label=${i18nCloseAlert[this._language.current]}
263
- sbb-toast-close
264
- ></sbb-transparent-button>` : nothing}
265
- </div>
266
- </div>
267
- </div>
268
- `;
269
- }
270
- };
271
- })();
272
- //#endregion
273
- export { SbbToastElement as t };
274
-
275
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"toast.component-Cgr5ZGOr.js","names":[],"sources":["../../../src/elements/toast/toast.scss?inline","../../../src/elements/toast/toast.component.ts"],"sourcesContent":["@use '../core/styles' as sbb;\n\n:host {\n  --sbb-toast-max-width: #{sbb.px-to-rem-build(400)};\n  --sbb-toast-animation-duration: var(\n    --sbb-disable-animation-duration,\n    var(--sbb-animation-duration-6x)\n  );\n  --sbb-toast-animation-timing-function: ease;\n  --sbb-toast-text-color: var(--sbb-color-3-inverted);\n  --sbb-toast-background-color: var(--sbb-background-color-1-inverted);\n  --sbb-toast-margin: var(--sbb-spacing-responsive-s);\n  --sbb-toast-padding-block: var(--sbb-spacing-responsive-xxxs);\n  --sbb-toast-padding-inline: var(--sbb-spacing-responsive-xs);\n  --sbb-toast-border-radius: var(--sbb-border-radius-4x);\n  --sbb-toast-container-display: none;\n  --sbb-toast-container-position: fixed;\n  --sbb-toast-horizontal-position: initial;\n  --sbb-toast-vertical-position: initial;\n  --sbb-toast-gap: var(--sbb-spacing-fixed-2x);\n  --sbb-focus-outline-color: var(--sbb-focus-outline-color-dark);\n\n  display: none;\n}\n\n:host(:is(:state(state-opening), :state(state-opened), :state(state-closing))) {\n  --sbb-toast-container-display: flex;\n\n  display: block;\n}\n\n:host([position='bottom-left']),\n:host([position='bottom-start']) {\n  --sbb-toast-horizontal-position: flex-start;\n  --sbb-toast-vertical-position: flex-end;\n}\n\n:host([position='bottom-center']) {\n  --sbb-toast-horizontal-position: center;\n  --sbb-toast-vertical-position: flex-end;\n}\n\n:host([position='bottom-right']),\n:host([position='bottom-end']) {\n  --sbb-toast-horizontal-position: flex-end;\n  --sbb-toast-vertical-position: flex-end;\n}\n\n:host([position='top-left']),\n:host([position='top-start']) {\n  --sbb-toast-horizontal-position: flex-start;\n  --sbb-toast-vertical-position: flex-start;\n}\n\n:host([position='top-center']) {\n  --sbb-toast-horizontal-position: center;\n  --sbb-toast-vertical-position: flex-start;\n}\n\n:host([position='top-right']),\n:host([position='top-end']) {\n  --sbb-toast-horizontal-position: flex-end;\n  --sbb-toast-vertical-position: flex-start;\n}\n\n:host(:state(dark)) {\n  --sbb-focus-outline-color: var(--sbb-focus-outline-color-default);\n}\n\n.sbb-toast__overlay-container {\n  position: var(--sbb-toast-container-position);\n  pointer-events: none;\n  inset: 0;\n  display: var(--sbb-toast-container-display);\n  justify-content: var(--sbb-toast-horizontal-position);\n  align-items: var(--sbb-toast-vertical-position);\n}\n\n.sbb-toast {\n  z-index: var(--sbb-toast-z-index, var(--sbb-overlay-default-z-index));\n  pointer-events: auto;\n  margin: var(--sbb-toast-margin);\n  border-radius: var(--sbb-toast-border-radius);\n  max-width: var(--sbb-toast-max-width);\n  display: flex;\n  color: var(--sbb-toast-text-color);\n  background-color: var(--sbb-toast-background-color);\n\n  :host(:state(state-opening)) & {\n    animation: {\n      name: open;\n      duration: var(--sbb-toast-animation-duration);\n      timing-function: var(--sbb-toast-animation-timing-function);\n    }\n  }\n\n  :host(:state(state-closing)) & {\n    animation: {\n      name: close;\n      duration: var(--sbb-toast-animation-duration);\n      timing-function: var(--sbb-toast-animation-timing-function);\n    }\n  }\n}\n\n.sbb-toast-wrapper {\n  display: flex;\n  align-items: center;\n  gap: var(--sbb-toast-gap);\n  padding-block: var(--sbb-toast-padding-block);\n  padding-inline: var(--sbb-toast-padding-inline);\n}\n\nslot:not([name='icon']),\n.sbb-toast__close {\n  // As the slotted content is displayed on a white background, we need to force light mode for them.\n  color-scheme: only light;\n}\n\n.sbb-toast__content {\n  @include sbb.title-6($exclude-spacing: true);\n\n  flex: 1 1 0;\n}\n\n.sbb-toast__close {\n  display: flex;\n  align-items: center;\n}\n\n.sbb-toast__close-divider {\n  padding-block: var(--sbb-toast-padding-block);\n}\n\n.sbb-toast__close-button {\n  margin-inline: var(--sbb-spacing-responsive-xxxs);\n}\n\n@keyframes open {\n  from {\n    visibility: hidden;\n    opacity: 0;\n    translate: 0 var(--sbb-spacing-fixed-2x);\n  }\n\n  to {\n    visibility: visible;\n    opacity: 1;\n    translate: 0;\n  }\n}\n\n@keyframes close {\n  from {\n    visibility: visible;\n    opacity: 1;\n    translate: 0;\n  }\n\n  to {\n    visibility: hidden;\n    opacity: 0;\n    translate: 0 var(--sbb-spacing-fixed-2x);\n  }\n}\n","import {\n  html,\n  nothing,\n  unsafeCSS,\n  type CSSResultGroup,\n  type PropertyValues,\n  type TemplateResult,\n} from 'lit';\nimport { property } from 'lit/decorators.js';\n\nimport {\n  SbbTransparentButtonElement,\n  type SbbTransparentButtonLinkElement,\n} from '../button.pure.ts';\nimport { SbbOpenCloseBaseElement } from '../core/base-elements.ts';\nimport { SbbDarkModeController, SbbLanguageController } from '../core/controllers.ts';\nimport { forceType } from '../core/decorators.ts';\nimport { isLean, isZeroAnimationDuration } from '../core/dom.ts';\nimport { composedPathHasAttribute } from '../core/eventing.ts';\nimport { i18nCloseAlert } from '../core/i18n.ts';\nimport { SbbReadonlyMixin } from '../core/mixins.ts';\nimport { boxSizingStyles } from '../core/styles.ts';\nimport { SbbIconNameMixin } from '../icon.ts';\nimport type { SbbLinkButtonElement, SbbLinkElement, SbbLinkStaticElement } from '../link.ts';\n\nimport style from './toast.scss?inline';\n\nimport '../divider.ts';\n\n// TODO(breaking-change): Remove call to define.\nSbbTransparentButtonElement.define();\n\ntype SbbToastPositionVertical = 'top' | 'bottom';\ntype SbbToastPositionHorizontal = 'left' | 'start' | 'center' | 'right' | 'end';\nexport type SbbToastPosition = `${SbbToastPositionVertical}-${SbbToastPositionHorizontal}`;\n\n// A global collection of existing toasts\nconst toastRefs = new Set<SbbToastElement>();\n\n/**\n * It displays a toast notification.\n *\n * @slot - Use the unnamed slot to add content to the `sbb-toast`.\n * @slot icon - Assign a custom icon via slot.\n * @slot action - Provide a custom action for this toast.\n * @cssprop [--sbb-toast-z-index=var(--sbb-overlay-default-z-index)] - To specify a custom stack order,\n * the `z-index` can be overridden by defining this CSS variable. The default `z-index` of the\n * component is set to `var(--sbb-overlay-default-z-index)` with a value of `1000`.\n */\nexport class SbbToastElement extends SbbIconNameMixin(SbbReadonlyMixin(SbbOpenCloseBaseElement)) {\n  public static override readonly elementName: string = 'sbb-toast';\n  public static override styles: CSSResultGroup = [boxSizingStyles, unsafeCSS(style)];\n\n  /**\n   * The length of time in milliseconds to wait before automatically dismissing the toast.\n   * If 0 (default), it stays open indefinitely.\n   * From accessibility perspective, it is recommended to set a timeout of at least 20 seconds.\n   */\n  @forceType()\n  @property({ type: Number })\n  public accessor timeout: number = 0;\n\n  /** The position where to place the toast. */\n  @property({ reflect: true }) public accessor position: SbbToastPosition = 'bottom-center';\n\n  /**\n   * The ARIA politeness level.\n   * Check https://developer.mozilla.org/en-US/docs/Web/Accessibility/ARIA/ARIA_Live_Regions#live_regions for further info\n   */\n  @property() public accessor politeness: 'polite' | 'assertive' | 'off' = 'polite';\n\n  private _closeTimeout?: ReturnType<typeof setTimeout>;\n  private _language = new SbbLanguageController(this);\n  private _darkModeController = new SbbDarkModeController(this, () => {\n    this._syncSlottedElements();\n    this.requestUpdate();\n  });\n\n  public constructor() {\n    super();\n    this.addEventListener?.('click', (e) => this._onClick(e));\n  }\n\n  public override connectedCallback(): void {\n    this.popover = 'manual';\n    super.connectedCallback();\n\n    // Add this toast to the global collection\n    toastRefs.add(this);\n  }\n\n  public override disconnectedCallback(): void {\n    super.disconnectedCallback();\n    clearTimeout(this._closeTimeout);\n\n    // Remove this instance\n    toastRefs.delete(this);\n  }\n\n  protected override willUpdate(changedProperties: PropertyValues<this>): void {\n    super.willUpdate(changedProperties);\n\n    if (changedProperties.has('politeness')) {\n      this.internals.ariaLive = this.politeness;\n    }\n  }\n\n  /**\n   * Open the toast.\n   * If there are other opened toasts in the page, close them first.\n   */\n  public open(): void {\n    if (this.state === 'opened' || this.state === 'opening' || !this.dispatchBeforeOpenEvent()) {\n      return;\n    }\n\n    this._closeOtherToasts();\n    this.showPopover?.();\n    this.state = 'opening';\n\n    // If the animation duration is zero, the animationend event is not always fired reliably.\n    // In this case we directly set the `opened` state.\n    if (this._isZeroAnimationDuration()) {\n      this._handleOpening();\n    }\n  }\n\n  /**\n   * Close the toast.\n   */\n  public close(): void {\n    if (this.state === 'closed' || this.state === 'closing' || !this.dispatchBeforeCloseEvent()) {\n      return;\n    }\n\n    clearTimeout(this._closeTimeout);\n    this.state = 'closing';\n\n    // If the animation duration is zero, the animationend event is not always fired reliably.\n    // In this case we directly set the `closed` state.\n    if (this._isZeroAnimationDuration()) {\n      this._handleClosing();\n    }\n  }\n\n  private _handleClosing(): void {\n    this.state = 'closed';\n    this.hidePopover?.();\n    this.dispatchCloseEvent();\n  }\n\n  private _handleOpening(): void {\n    this.state = 'opened';\n    this.dispatchOpenEvent();\n\n    // Start the countdown to close it\n    if (this.timeout) {\n      // Workaround for https://github.com/sbb-design-systems/lyne-angular/issues/190\n      // If zone.js is loaded, setTimeout is wrapped and tracked which will mark\n      // anything in Angular as unstable as long as setTimeout is not finished.\n      // This only needs to be fixed in places where we actually want to wait a\n      // specific amount of time without an interaction (e.g. for this case).\n      const global = globalThis as any;\n      const setTimeout: typeof globalThis.setTimeout =\n        global[global.Zone?.__symbol__?.('setTimeout') as string] ?? global.setTimeout;\n\n      this._closeTimeout = setTimeout(() => this.close(), this.timeout);\n    }\n  }\n\n  // Close the toast on click of any element that has the 'sbb-toast-close' attribute.\n  private _onClick(event: Event): void {\n    const closeElement = composedPathHasAttribute(event, 'sbb-toast-close', this);\n\n    if (closeElement && !closeElement.hasAttribute('disabled')) {\n      this.close();\n    }\n  }\n\n  private _isZeroAnimationDuration(): boolean {\n    return isZeroAnimationDuration(this, '--sbb-toast-animation-duration');\n  }\n\n  /**\n   * Slotted text nodes are not read correctly by screen readers on Chrome.\n   * To address the problem, if there is at least a root text node,\n   * we wrap the whole content in a <span> tag\n   */\n  private _onContentSlotChange(event: Event): void {\n    const slotNodes = (event.target as HTMLSlotElement).assignedNodes();\n\n    if (slotNodes.some((el) => el.nodeType === Node.TEXT_NODE)) {\n      const span = document.createElement('span');\n      this.prepend(span);\n      span.append(...slotNodes);\n    }\n    this._syncSlottedElements();\n  }\n\n  private _syncSlottedElements(): void {\n    // Force the visual state on slotted buttons\n    this.querySelectorAll<SbbTransparentButtonElement | SbbTransparentButtonLinkElement>(\n      'sbb-transparent-button, sbb-transparent-button-link',\n    ).forEach((btn) => {\n      btn.negative = this._isLightMode();\n      btn.size = isLean() ? 's' : 'm';\n    });\n\n    // Force negative on slotted links\n    this.querySelectorAll<SbbLinkElement | SbbLinkButtonElement | SbbLinkStaticElement>(\n      'sbb-link, sbb-link-button, sbb-link-static',\n    ).forEach((link) => {\n      link.negative = this._isLightMode();\n    });\n  }\n\n  // In rare cases it can be that the animationEnd event is triggered twice.\n  // To avoid entering a corrupt state, exit when state is not expected.\n  private _onToastAnimationEnd(event: AnimationEvent): void {\n    // On toast opened\n    if (event.animationName === 'open' && this.state === 'opening') {\n      this._handleOpening();\n    } else if (event.animationName === 'close' && this.state === 'closing') {\n      this._handleClosing();\n    }\n  }\n\n  /**\n   * Since we do not stack toasts, we force the closing on other existing opened ones\n   */\n  private _closeOtherToasts(): void {\n    toastRefs.forEach((t) => {\n      if (t.state === 'opened' || t.state === 'opening') {\n        t.close();\n      }\n    });\n  }\n\n  private _isLightMode(): boolean {\n    return !this._darkModeController.matches();\n  }\n\n  protected override render(): TemplateResult {\n    return html`\n      <div class=\"sbb-toast__overlay-container\">\n        <div class=\"sbb-toast\" @animationend=${this._onToastAnimationEnd}>\n          <div class=\"sbb-toast-wrapper\">\n            ${this.renderIconSlot()}\n            <div class=\"sbb-toast__content\">\n              <slot @slotchange=${this._onContentSlotChange}></slot>\n            </div>\n            <slot name=\"action\" @slotchange=${this._syncSlottedElements}></slot>\n          </div>\n          <div class=\"sbb-toast__close\">\n            <sbb-divider\n              class=\"sbb-toast__close-divider\"\n              orientation=\"vertical\"\n              ?negative=${this._isLightMode()}\n            ></sbb-divider>\n            ${!this.readOnly\n              ? html`<sbb-transparent-button\n                  class=\"sbb-toast__close-button\"\n                  icon-name=\"cross-small\"\n                  ?negative=${this._isLightMode()}\n                  size=\"m\"\n                  aria-label=${i18nCloseAlert[this._language.current]}\n                  sbb-toast-close\n                ></sbb-transparent-button>`\n              : nothing}\n          </div>\n        </div>\n      </div>\n    `;\n  }\n}\n\ndeclare global {\n  interface HTMLElementTagNameMap {\n    // eslint-disable-next-line @typescript-eslint/naming-convention\n    'sbb-toast': SbbToastElement;\n  }\n}\n"],"mappings":"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"}