@sbb-esta/lyne-elements-dev 4.7.0-dev.1772631734 → 4.7.0-dev.1772638329

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.
@@ -260,7 +260,7 @@ let SbbToastElement = (() => {
260
260
  * If there are other opened toasts in the page, close them first.
261
261
  */
262
262
  open() {
263
- if (this.state !== "closed" || !this.dispatchBeforeOpenEvent()) {
263
+ if (this.state === "opened" || this.state === "opening" || !this.dispatchBeforeOpenEvent()) {
264
264
  return;
265
265
  }
266
266
  this._closeOtherToasts();
@@ -274,7 +274,7 @@ let SbbToastElement = (() => {
274
274
  * Close the toast.
275
275
  */
276
276
  close() {
277
- if (this.state !== "opened" || !this.dispatchBeforeCloseEvent()) {
277
+ if (this.state === "closed" || this.state === "closing" || !this.dispatchBeforeCloseEvent()) {
278
278
  return;
279
279
  }
280
280
  clearTimeout(this._closeTimeout);
@@ -344,7 +344,7 @@ let SbbToastElement = (() => {
344
344
  */
345
345
  _closeOtherToasts() {
346
346
  toastRefs.forEach((t) => {
347
- if (t.isOpen) {
347
+ if (t.state === "opened" || t.state === "opening") {
348
348
  t.close();
349
349
  }
350
350
  });
@@ -405,4 +405,4 @@ let SbbToastElement = (() => {
405
405
  export {
406
406
  SbbToastElement
407
407
  };
408
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"toast.component.js","sources":["../../../../src/elements/toast/toast.component.ts"],"sourcesContent":["import type { CSSResultGroup, PropertyValues, TemplateResult } from 'lit';\nimport { html, nothing } from 'lit';\nimport { customElement, property } from 'lit/decorators.js';\n\nimport type { SbbTransparentButtonElement, SbbTransparentButtonLinkElement } from '../button.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 { SbbHydrationMixin, 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?lit&inline';\n\nimport '../button/transparent-button.ts';\nimport '../divider.ts';\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\n@customElement('sbb-toast')\nclass SbbToastElement extends SbbIconNameMixin(\n  SbbHydrationMixin(SbbReadonlyMixin(SbbOpenCloseBaseElement)),\n) {\n  public static override styles: CSSResultGroup = [boxSizingStyles, 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 !== 'closed' || !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 !== 'opened' || !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.isOpen) {\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"],"names":[],"mappings":"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"}
408
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"toast.component.js","sources":["../../../../src/elements/toast/toast.component.ts"],"sourcesContent":["import type { CSSResultGroup, PropertyValues, TemplateResult } from 'lit';\nimport { html, nothing } from 'lit';\nimport { customElement, property } from 'lit/decorators.js';\n\nimport type { SbbTransparentButtonElement, SbbTransparentButtonLinkElement } from '../button.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 { SbbHydrationMixin, 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?lit&inline';\n\nimport '../button/transparent-button.ts';\nimport '../divider.ts';\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\n@customElement('sbb-toast')\nclass SbbToastElement extends SbbIconNameMixin(\n  SbbHydrationMixin(SbbReadonlyMixin(SbbOpenCloseBaseElement)),\n) {\n  public static override styles: CSSResultGroup = [boxSizingStyles, 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"],"names":[],"mappings":"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"}
package/package.json CHANGED
@@ -1,13 +1,13 @@
1
1
  {
2
2
  "name": "@sbb-esta/lyne-elements-dev",
3
- "version": "4.7.0-dev.1772631734",
3
+ "version": "4.7.0-dev.1772638329",
4
4
  "description": "Lyne Design System",
5
5
  "keywords": [
6
6
  "design system",
7
7
  "web components",
8
8
  "lit",
9
9
  "storybook",
10
- "https://github.com/sbb-design-systems/lyne-components/commit/15ca867493f336f96f99a558d0d13195eebe8a8e"
10
+ "https://github.com/sbb-design-systems/lyne-components/commit/00e09d3ca8e505598817488aacad3311d6a325e0"
11
11
  ],
12
12
  "type": "module",
13
13
  "exports": {
@@ -73,13 +73,13 @@ let bt = (() => {
73
73
  * If there are other opened toasts in the page, close them first.
74
74
  */
75
75
  open() {
76
- this.state !== "closed" || !this.dispatchBeforeOpenEvent() || (this._closeOtherToasts(), this.showPopover?.(), this.state = "opening", this._isZeroAnimationDuration() && this._handleOpening());
76
+ this.state === "opened" || this.state === "opening" || !this.dispatchBeforeOpenEvent() || (this._closeOtherToasts(), this.showPopover?.(), this.state = "opening", this._isZeroAnimationDuration() && this._handleOpening());
77
77
  }
78
78
  /**
79
79
  * Close the toast.
80
80
  */
81
81
  close() {
82
- this.state !== "opened" || !this.dispatchBeforeCloseEvent() || (clearTimeout(this._closeTimeout), this.state = "closing", this._isZeroAnimationDuration() && this._handleClosing());
82
+ this.state === "closed" || this.state === "closing" || !this.dispatchBeforeCloseEvent() || (clearTimeout(this._closeTimeout), this.state = "closing", this._isZeroAnimationDuration() && this._handleClosing());
83
83
  }
84
84
  _handleClosing() {
85
85
  this.state = "closed", this.hidePopover?.(), this.dispatchCloseEvent();
@@ -128,7 +128,7 @@ let bt = (() => {
128
128
  */
129
129
  _closeOtherToasts() {
130
130
  v.forEach((t) => {
131
- t.isOpen && t.close();
131
+ (t.state === "opened" || t.state === "opening") && t.close();
132
132
  });
133
133
  }
134
134
  _isLightMode() {