@sbb-esta/lyne-elements 1.5.0 → 1.6.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (51) hide show
  1. package/alert/alert/alert.d.ts +22 -16
  2. package/alert/alert/alert.d.ts.map +1 -1
  3. package/alert/alert-group/alert-group.d.ts +1 -1
  4. package/alert/alert-group/alert-group.d.ts.map +1 -1
  5. package/alert/alert-group.js +31 -26
  6. package/alert/alert.js +45 -37
  7. package/clock/clock.d.ts.map +1 -1
  8. package/clock.js +14 -14
  9. package/core/testing/wait-for-event.d.ts +2 -0
  10. package/core/testing/wait-for-event.d.ts.map +1 -0
  11. package/core/testing.d.ts +1 -0
  12. package/core/testing.d.ts.map +1 -1
  13. package/core/testing.js +49 -36
  14. package/custom-elements.json +211 -51
  15. package/development/alert/alert/alert.d.ts +22 -16
  16. package/development/alert/alert/alert.d.ts.map +1 -1
  17. package/development/alert/alert-group/alert-group.d.ts +1 -1
  18. package/development/alert/alert-group/alert-group.d.ts.map +1 -1
  19. package/development/alert/alert-group.js +16 -11
  20. package/development/alert/alert.js +65 -24
  21. package/development/clock/clock.d.ts.map +1 -1
  22. package/development/clock.js +3 -2
  23. package/development/core/testing/wait-for-event.d.ts +2 -0
  24. package/development/core/testing/wait-for-event.d.ts.map +1 -0
  25. package/development/core/testing.d.ts +1 -0
  26. package/development/core/testing.d.ts.map +1 -1
  27. package/development/core/testing.js +17 -1
  28. package/development/icon/icon.d.ts +9 -1
  29. package/development/icon/icon.d.ts.map +1 -1
  30. package/development/icon.js +16 -2
  31. package/development/menu/menu/menu.d.ts.map +1 -1
  32. package/development/menu/menu.js +2 -1
  33. package/development/navigation/navigation/navigation.d.ts.map +1 -1
  34. package/development/navigation/navigation.js +2 -1
  35. package/development/notification/notification.d.ts +4 -4
  36. package/development/notification.js +1 -1
  37. package/development/overlay/overlay-base-element.d.ts.map +1 -1
  38. package/development/overlay.js +35 -17
  39. package/development/toggle/toggle.js +1 -1
  40. package/icon/icon.d.ts +9 -1
  41. package/icon/icon.d.ts.map +1 -1
  42. package/icon.js +71 -62
  43. package/menu/menu/menu.d.ts.map +1 -1
  44. package/menu/menu.js +3 -3
  45. package/navigation/navigation/navigation.d.ts.map +1 -1
  46. package/navigation/navigation.js +4 -4
  47. package/notification/notification.d.ts +4 -4
  48. package/overlay/overlay-base-element.d.ts.map +1 -1
  49. package/overlay.js +33 -33
  50. package/package.json +1 -1
  51. package/toggle/toggle.js +1 -1
@@ -1,5 +1,5 @@
1
- import { state, property, customElement } from "lit/decorators.js";
2
1
  import { isServer, css, LitElement, html, nothing } from "lit";
2
+ import { state, property, customElement } from "lit/decorators.js";
3
3
  import { unsafeHTML } from "lit/directives/unsafe-html.js";
4
4
  import { until } from "lit/directives/until.js";
5
5
  import { hostAttributes } from "./core/decorators.js";
@@ -240,6 +240,7 @@ let SbbIconElement = class extends SbbIconBase {
240
240
  constructor() {
241
241
  super(...arguments);
242
242
  this._defaultAriaLabel = "";
243
+ this._sbbAngularCompatibility = false;
243
244
  }
244
245
  async fetchSvgIcon(namespace, name) {
245
246
  if (this.getAttribute("aria-label") === this._defaultAriaLabel) {
@@ -257,16 +258,29 @@ let SbbIconElement = class extends SbbIconBase {
257
258
  this.loadSvgIcon(this.name);
258
259
  }
259
260
  }
261
+ attributeChangedCallback(name, _old, value) {
262
+ if (name === "svgicon") {
263
+ this._sbbAngularCompatibility = !!value;
264
+ } else {
265
+ super.attributeChangedCallback(name, _old, value);
266
+ }
267
+ }
260
268
  firstUpdated(changedProperties) {
261
269
  super.firstUpdated(changedProperties);
262
270
  if (!this.hasAttribute("aria-hidden")) {
263
271
  this.setAttribute("aria-hidden", "true");
264
272
  }
265
273
  }
274
+ render() {
275
+ return this._sbbAngularCompatibility ? html`<slot></slot>` : super.render();
276
+ }
266
277
  };
267
278
  __decorateClass$1([
268
279
  property({ reflect: true })
269
280
  ], SbbIconElement.prototype, "name", 2);
281
+ __decorateClass$1([
282
+ state()
283
+ ], SbbIconElement.prototype, "_sbbAngularCompatibility", 2);
270
284
  SbbIconElement = __decorateClass$1([
271
285
  customElement("sbb-icon")
272
286
  ], SbbIconElement);
@@ -302,4 +316,4 @@ export {
302
316
  isValid,
303
317
  validateContent
304
318
  };
305
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"icon.js","sources":["../../../src/elements/icon/icon-validate.ts","../../../src/elements/icon/icon-request.ts","../../../src/elements/icon/icon-base.ts","../../../src/elements/icon/icon.ts","../../../src/elements/icon/icon-name-mixin.ts"],"sourcesContent":["const isStr = (val: any): val is string => typeof val === 'string';\n\n/**\n * Sanitizes the SVG element and all its child nodes.\n * Does not allow `<script>` elements, or any attributes that start with `on`, such as `onclick`.\n */\nexport const isValid = (elm: HTMLElement): boolean => {\n  if (elm.nodeType === 1) {\n    // is an Element node like <div>, <p> or <svg>\n    if (elm.nodeName.toLowerCase() === 'script') {\n      return false;\n    }\n\n    // do not allow attributes starting with `on`\n    for (let i = 0; i < elm.attributes.length; i++) {\n      const val = elm.attributes[i].name;\n      if (isStr(val) && val.toLowerCase().indexOf('on') === 0) {\n        return false;\n      }\n    }\n\n    for (let i = 0; i < elm.childNodes.length; i++) {\n      if (!isValid(elm.childNodes[i] as any)) {\n        return false;\n      }\n    }\n  }\n  return true;\n};\n\n/**\n * Validates the SVG content by checking that it has only one root element `<svg>`,\n * adding the `color-immutable` class if the `colorImmutable` option is set to `true`,\n * and sanitizing the provided content as long as the `sanitize` property is not explicitly set to `false`.\n */\nexport const validateContent = (\n  svgContent: string,\n  sanitize = true,\n  colorImmutable = false,\n): string => {\n  const div = document.createElement('div');\n  div.innerHTML = svgContent;\n\n  for (let i = div.childNodes.length - 1; i >= 0; i--) {\n    if (div.childNodes[i].nodeName.toLowerCase() !== 'svg') {\n      div.removeChild(div.childNodes[i]);\n    }\n  }\n\n  // must only have 1 root element\n  const svgElm = div.firstElementChild;\n\n  if (svgElm && svgElm.nodeName.toLowerCase() === 'svg') {\n    if (colorImmutable && !svgElm.classList.contains('color-immutable')) {\n      svgElm.classList.add('color-immutable');\n    }\n\n    // do not sanitize the svg element\n    if (sanitize === false) {\n      return div.innerHTML;\n    }\n\n    // sanitize the svg element\n    if (isValid(svgElm as any)) {\n      return div.innerHTML;\n    }\n  }\n  return '';\n};\n","import { isServer } from 'lit';\n\nimport type { SbbIconConfig } from '../core/config.js';\nimport { readConfig } from '../core/config.js';\n\nimport { validateContent } from './icon-validate.js';\n\nconst iconCdn = 'https://icons.app.sbb.ch/';\n\nconst iconNamespaces = new Map<string, string>()\n  .set('default', `${iconCdn}icons/`)\n  .set('picto', `${iconCdn}picto/`);\nconst requests = new Map<string, Promise<any>>();\n\n/** Fetches icon svg content from providers and asserts only one request per icon is made. */\nexport const getSvgContent = (\n  namespace: string,\n  name: string,\n  sanitize: boolean,\n): Promise<string> => {\n  const config: SbbIconConfig = readConfig().icon ?? {};\n\n  const resolvedNamespace = config.namespaces?.get(namespace) ?? iconNamespaces.get(namespace);\n  if (resolvedNamespace == null) {\n    throw Error(\n      `Unable to find the namespace \"${namespace}\". Please register your custom namespace.`,\n    );\n  }\n  const url = `${resolvedNamespace}${name}.svg`;\n\n  // Check if we already have a request for this url\n  let req = requests.get(url);\n\n  if (!req) {\n    // We cannot support server side rendered icons (yet), as the validation\n    // is done via DOM, which is not available during SSR.\n    if (typeof fetch !== 'undefined' && !isServer) {\n      const interceptor = config.interceptor ?? ((i) => i.request());\n\n      req = interceptor({\n        namespace,\n        name,\n        url,\n        request: () =>\n          fetch(url)\n            .then(async (response) => {\n              if (!response.ok) {\n                throw new Error('Failed to load icon ' + namespace + ':' + name);\n              }\n              return validateContent(await response.text(), sanitize);\n            })\n            .catch((error) => {\n              throw Error(error);\n            }),\n      });\n      // Cache for the same requests\n      requests.set(url, req);\n    } else {\n      // Resolve promise for ssr scenarios\n      return Promise.resolve('');\n    }\n  }\n\n  return req;\n};\n","import type { CSSResultGroup, PropertyValues, TemplateResult } from 'lit';\nimport { html, LitElement } from 'lit';\nimport { property, state } from 'lit/decorators.js';\nimport type { DirectiveResult } from 'lit/directive.js';\nimport type { UnsafeHTMLDirective } from 'lit/directives/unsafe-html.js';\nimport { unsafeHTML } from 'lit/directives/unsafe-html.js';\nimport { until } from 'lit/directives/until.js';\n\nimport { hostAttributes } from '../core/decorators.js';\n\nimport { getSvgContent } from './icon-request.js';\nimport style from './icon.scss?lit&inline';\n\n/**\n * @cssprop [--sbb-icon-svg-width=auto] - Can be used to set a custom width.\n * @cssprop [--sbb-icon-svg-height=auto] - Can be used to set a custom height.\n */\n@hostAttributes({\n  'data-namespace': SbbIconBase._defaultNamespace,\n  'data-empty': '',\n})\nexport abstract class SbbIconBase extends LitElement {\n  public static override styles: CSSResultGroup = style;\n  private static readonly _defaultNamespace = 'default';\n\n  @state() private _svgNamespace = SbbIconBase._defaultNamespace;\n\n  /**\n   * The icon svg content rendered on the page: <svg>...</svg>.\n   */\n  @state() private _svgIcon?: Promise<DirectiveResult<typeof UnsafeHTMLDirective>>;\n\n  /**\n   * When set to `true`, SVG content that is HTTP fetched will not be checked\n   * if the response SVG content has any `<script>` elements, or any attributes\n   * that start with `on`, such as `onclick`.\n   * @default false\n   */\n  @property({ attribute: 'no-sanitize', type: Boolean }) public noSanitize = false;\n\n  protected async loadSvgIcon(iconName: string): Promise<void> {\n    if (!iconName) {\n      return;\n    }\n\n    const [namespace, name] = this._splitIconName(iconName);\n    this._svgNamespace = namespace;\n    this.setAttribute('data-namespace', this._svgNamespace);\n\n    const svgIcon = this.fetchSvgIcon(this._svgNamespace, name);\n    this._svgIcon = svgIcon.then((v) => unsafeHTML(v));\n    try {\n      this.toggleAttribute('data-empty', !(await svgIcon));\n    } catch {\n      this.toggleAttribute('data-empty', true);\n    }\n  }\n\n  protected async fetchSvgIcon(namespace: string, name: string): Promise<string> {\n    return await getSvgContent(namespace, name, !this.noSanitize);\n  }\n\n  private _splitIconName(iconName: string): [string, string] {\n    if (!iconName) {\n      return ['', ''];\n    }\n    const parts = iconName.split(':');\n    switch (parts.length) {\n      case 1:\n        // Use default namespace if empty.\n        return [SbbIconBase._defaultNamespace, parts[0]];\n      case 2:\n        return parts as [string, string];\n      default:\n        throw Error(`Invalid icon name: \"${iconName}\"`);\n    }\n  }\n\n  protected override firstUpdated(changedProperties: PropertyValues<this>): void {\n    super.firstUpdated(changedProperties);\n\n    this.setAttribute('role', this.getAttribute('role') ?? 'img');\n  }\n\n  protected override render(): TemplateResult {\n    return html`<span class=\"sbb-icon-inner\"\n      >${until(\n        this._svgIcon,\n        // To reserve space, we need an empty svg to apply dimension to.\n        html`<svg width=\"0\" height=\"0\"></svg>`,\n      )}</span\n    >`;\n  }\n}\n","import type { PropertyValues } from 'lit';\nimport { customElement, property } from 'lit/decorators.js';\n\nimport { SbbIconBase } from './icon-base.js';\n\n/**\n * It displays an icon loaded from a registered namespace.\n */\n@customElement('sbb-icon')\nexport class SbbIconElement extends SbbIconBase {\n  /**\n   * The provided name consisting of the namespace and the name of the icon.\n   * If the namespace is missing, the default namespace \"sbb\" will be used.\n   * E.g. `name` (will use \"sbb\" as namespace) or `namespace:name`.\n   */\n  @property({ reflect: true }) public name!: string;\n\n  private _defaultAriaLabel = '';\n\n  protected override async fetchSvgIcon(namespace: string, name: string): Promise<string> {\n    // If the icon is changing, and we were using the defaultAriaLabel, reset it\n    if (this.getAttribute('aria-label') === this._defaultAriaLabel) {\n      this.removeAttribute('aria-label');\n    }\n\n    this._defaultAriaLabel = `Icon ${name.replace(/-/g, ' ')}`;\n\n    // generate a default label in case user does not provide their own\n    // and aria-hidden is set to \"false\"\n    if (this.getAttribute('aria-hidden') === 'false' && !this.hasAttribute('aria-label') && name) {\n      this.setAttribute('aria-label', this._defaultAriaLabel);\n    }\n\n    return super.fetchSvgIcon(namespace, name);\n  }\n\n  protected override willUpdate(changedProperties: PropertyValues<this>): void {\n    super.willUpdate(changedProperties);\n\n    if (changedProperties.has('name')) {\n      this.loadSvgIcon(this.name);\n    }\n  }\n\n  protected override firstUpdated(changedProperties: PropertyValues<this>): void {\n    super.firstUpdated(changedProperties);\n\n    if (!this.hasAttribute('aria-hidden')) {\n      this.setAttribute('aria-hidden', 'true');\n    }\n  }\n}\n\ndeclare global {\n  interface HTMLElementTagNameMap {\n    // eslint-disable-next-line @typescript-eslint/naming-convention\n    'sbb-icon': SbbIconElement;\n  }\n}\n","import { html, type LitElement, nothing, type TemplateResult } from 'lit';\nimport { property } from 'lit/decorators.js';\n\nimport type { AbstractConstructor } from '../core/mixins.js';\n\nimport './icon.js';\n\nexport declare class SbbIconNameMixinType {\n  public iconName?: string;\n  protected renderIconSlot(classname?: string): TemplateResult;\n}\n\n/**\n * Enhance your component with an iconName property and provides a template for the icon slot.\n */\n// eslint-disable-next-line @typescript-eslint/naming-convention\nexport const SbbIconNameMixin = <T extends AbstractConstructor<LitElement>>(\n  superClass: T,\n): AbstractConstructor<SbbIconNameMixinType> & T => {\n  abstract class SbbIconNameElement extends superClass implements Partial<SbbIconNameMixinType> {\n    /**\n     * The icon name we want to use, choose from the small icon variants\n     * from the ui-icons category from here\n     * https://icons.app.sbb.ch.\n     */\n    @property({ attribute: 'icon-name', reflect: true }) public iconName?: string;\n\n    protected renderIconSlot(classname?: string): TemplateResult {\n      return html`\n        <slot name=\"icon\">\n          ${this.iconName\n            ? html`<sbb-icon name=${this.iconName} class=${classname || nothing}></sbb-icon>`\n            : nothing}\n        </slot>\n      `;\n    }\n  }\n\n  return SbbIconNameElement as unknown as AbstractConstructor<SbbIconNameMixinType> & T;\n};\n"],"names":["__decorateClass"],"mappings":"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"}
319
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"icon.js","sources":["../../../src/elements/icon/icon-validate.ts","../../../src/elements/icon/icon-request.ts","../../../src/elements/icon/icon-base.ts","../../../src/elements/icon/icon.ts","../../../src/elements/icon/icon-name-mixin.ts"],"sourcesContent":["const isStr = (val: any): val is string => typeof val === 'string';\n\n/**\n * Sanitizes the SVG element and all its child nodes.\n * Does not allow `<script>` elements, or any attributes that start with `on`, such as `onclick`.\n */\nexport const isValid = (elm: HTMLElement): boolean => {\n  if (elm.nodeType === 1) {\n    // is an Element node like <div>, <p> or <svg>\n    if (elm.nodeName.toLowerCase() === 'script') {\n      return false;\n    }\n\n    // do not allow attributes starting with `on`\n    for (let i = 0; i < elm.attributes.length; i++) {\n      const val = elm.attributes[i].name;\n      if (isStr(val) && val.toLowerCase().indexOf('on') === 0) {\n        return false;\n      }\n    }\n\n    for (let i = 0; i < elm.childNodes.length; i++) {\n      if (!isValid(elm.childNodes[i] as any)) {\n        return false;\n      }\n    }\n  }\n  return true;\n};\n\n/**\n * Validates the SVG content by checking that it has only one root element `<svg>`,\n * adding the `color-immutable` class if the `colorImmutable` option is set to `true`,\n * and sanitizing the provided content as long as the `sanitize` property is not explicitly set to `false`.\n */\nexport const validateContent = (\n  svgContent: string,\n  sanitize = true,\n  colorImmutable = false,\n): string => {\n  const div = document.createElement('div');\n  div.innerHTML = svgContent;\n\n  for (let i = div.childNodes.length - 1; i >= 0; i--) {\n    if (div.childNodes[i].nodeName.toLowerCase() !== 'svg') {\n      div.removeChild(div.childNodes[i]);\n    }\n  }\n\n  // must only have 1 root element\n  const svgElm = div.firstElementChild;\n\n  if (svgElm && svgElm.nodeName.toLowerCase() === 'svg') {\n    if (colorImmutable && !svgElm.classList.contains('color-immutable')) {\n      svgElm.classList.add('color-immutable');\n    }\n\n    // do not sanitize the svg element\n    if (sanitize === false) {\n      return div.innerHTML;\n    }\n\n    // sanitize the svg element\n    if (isValid(svgElm as any)) {\n      return div.innerHTML;\n    }\n  }\n  return '';\n};\n","import { isServer } from 'lit';\n\nimport type { SbbIconConfig } from '../core/config.js';\nimport { readConfig } from '../core/config.js';\n\nimport { validateContent } from './icon-validate.js';\n\nconst iconCdn = 'https://icons.app.sbb.ch/';\n\nconst iconNamespaces = new Map<string, string>()\n  .set('default', `${iconCdn}icons/`)\n  .set('picto', `${iconCdn}picto/`);\nconst requests = new Map<string, Promise<any>>();\n\n/** Fetches icon svg content from providers and asserts only one request per icon is made. */\nexport const getSvgContent = (\n  namespace: string,\n  name: string,\n  sanitize: boolean,\n): Promise<string> => {\n  const config: SbbIconConfig = readConfig().icon ?? {};\n\n  const resolvedNamespace = config.namespaces?.get(namespace) ?? iconNamespaces.get(namespace);\n  if (resolvedNamespace == null) {\n    throw Error(\n      `Unable to find the namespace \"${namespace}\". Please register your custom namespace.`,\n    );\n  }\n  const url = `${resolvedNamespace}${name}.svg`;\n\n  // Check if we already have a request for this url\n  let req = requests.get(url);\n\n  if (!req) {\n    // We cannot support server side rendered icons (yet), as the validation\n    // is done via DOM, which is not available during SSR.\n    if (typeof fetch !== 'undefined' && !isServer) {\n      const interceptor = config.interceptor ?? ((i) => i.request());\n\n      req = interceptor({\n        namespace,\n        name,\n        url,\n        request: () =>\n          fetch(url)\n            .then(async (response) => {\n              if (!response.ok) {\n                throw new Error('Failed to load icon ' + namespace + ':' + name);\n              }\n              return validateContent(await response.text(), sanitize);\n            })\n            .catch((error) => {\n              throw Error(error);\n            }),\n      });\n      // Cache for the same requests\n      requests.set(url, req);\n    } else {\n      // Resolve promise for ssr scenarios\n      return Promise.resolve('');\n    }\n  }\n\n  return req;\n};\n","import type { CSSResultGroup, PropertyValues, TemplateResult } from 'lit';\nimport { html, LitElement } from 'lit';\nimport { property, state } from 'lit/decorators.js';\nimport type { DirectiveResult } from 'lit/directive.js';\nimport type { UnsafeHTMLDirective } from 'lit/directives/unsafe-html.js';\nimport { unsafeHTML } from 'lit/directives/unsafe-html.js';\nimport { until } from 'lit/directives/until.js';\n\nimport { hostAttributes } from '../core/decorators.js';\n\nimport { getSvgContent } from './icon-request.js';\nimport style from './icon.scss?lit&inline';\n\n/**\n * @cssprop [--sbb-icon-svg-width=auto] - Can be used to set a custom width.\n * @cssprop [--sbb-icon-svg-height=auto] - Can be used to set a custom height.\n */\n@hostAttributes({\n  'data-namespace': SbbIconBase._defaultNamespace,\n  'data-empty': '',\n})\nexport abstract class SbbIconBase extends LitElement {\n  public static override styles: CSSResultGroup = style;\n  private static readonly _defaultNamespace = 'default';\n\n  @state() private _svgNamespace = SbbIconBase._defaultNamespace;\n\n  /**\n   * The icon svg content rendered on the page: <svg>...</svg>.\n   */\n  @state() private _svgIcon?: Promise<DirectiveResult<typeof UnsafeHTMLDirective>>;\n\n  /**\n   * When set to `true`, SVG content that is HTTP fetched will not be checked\n   * if the response SVG content has any `<script>` elements, or any attributes\n   * that start with `on`, such as `onclick`.\n   * @default false\n   */\n  @property({ attribute: 'no-sanitize', type: Boolean }) public noSanitize = false;\n\n  protected async loadSvgIcon(iconName: string): Promise<void> {\n    if (!iconName) {\n      return;\n    }\n\n    const [namespace, name] = this._splitIconName(iconName);\n    this._svgNamespace = namespace;\n    this.setAttribute('data-namespace', this._svgNamespace);\n\n    const svgIcon = this.fetchSvgIcon(this._svgNamespace, name);\n    this._svgIcon = svgIcon.then((v) => unsafeHTML(v));\n    try {\n      this.toggleAttribute('data-empty', !(await svgIcon));\n    } catch {\n      this.toggleAttribute('data-empty', true);\n    }\n  }\n\n  protected async fetchSvgIcon(namespace: string, name: string): Promise<string> {\n    return await getSvgContent(namespace, name, !this.noSanitize);\n  }\n\n  private _splitIconName(iconName: string): [string, string] {\n    if (!iconName) {\n      return ['', ''];\n    }\n    const parts = iconName.split(':');\n    switch (parts.length) {\n      case 1:\n        // Use default namespace if empty.\n        return [SbbIconBase._defaultNamespace, parts[0]];\n      case 2:\n        return parts as [string, string];\n      default:\n        throw Error(`Invalid icon name: \"${iconName}\"`);\n    }\n  }\n\n  protected override firstUpdated(changedProperties: PropertyValues<this>): void {\n    super.firstUpdated(changedProperties);\n\n    this.setAttribute('role', this.getAttribute('role') ?? 'img');\n  }\n\n  protected override render(): TemplateResult {\n    return html`<span class=\"sbb-icon-inner\"\n      >${until(\n        this._svgIcon,\n        // To reserve space, we need an empty svg to apply dimension to.\n        html`<svg width=\"0\" height=\"0\"></svg>`,\n      )}</span\n    >`;\n  }\n}\n","import { html, type PropertyValues, type TemplateResult } from 'lit';\nimport { customElement, property, state } from 'lit/decorators.js';\n\nimport { SbbIconBase } from './icon-base.js';\n\n/**\n * It displays an icon loaded from a registered namespace.\n */\n@customElement('sbb-icon')\nexport class SbbIconElement extends SbbIconBase {\n  /**\n   * The provided name consisting of the namespace and the name of the icon.\n   * If the namespace is missing, the default namespace \"sbb\" will be used.\n   * E.g. `name` (will use \"sbb\" as namespace) or `namespace:name`.\n   */\n  @property({ reflect: true }) public name!: string;\n\n  private _defaultAriaLabel = '';\n\n  /**\n   * The sbb-angular library has a sbb-icon component as well. In order to provide\n   * compatibility with it (as some icons are used internally inside the other sbb-angular\n   * components) we need to check whether the attribute svgicon is used.\n   */\n  @state() private _sbbAngularCompatibility = false;\n\n  protected override async fetchSvgIcon(namespace: string, name: string): Promise<string> {\n    // If the icon is changing, and we were using the defaultAriaLabel, reset it\n    if (this.getAttribute('aria-label') === this._defaultAriaLabel) {\n      this.removeAttribute('aria-label');\n    }\n\n    this._defaultAriaLabel = `Icon ${name.replace(/-/g, ' ')}`;\n\n    // generate a default label in case user does not provide their own\n    // and aria-hidden is set to \"false\"\n    if (this.getAttribute('aria-hidden') === 'false' && !this.hasAttribute('aria-label') && name) {\n      this.setAttribute('aria-label', this._defaultAriaLabel);\n    }\n\n    return super.fetchSvgIcon(namespace, name);\n  }\n\n  protected override willUpdate(changedProperties: PropertyValues<this>): void {\n    super.willUpdate(changedProperties);\n\n    if (changedProperties.has('name')) {\n      this.loadSvgIcon(this.name);\n    }\n  }\n\n  public override attributeChangedCallback(\n    name: string,\n    _old: string | null,\n    value: string | null,\n  ): void {\n    if (name === 'svgicon') {\n      this._sbbAngularCompatibility = !!value;\n    } else {\n      super.attributeChangedCallback(name, _old, value);\n    }\n  }\n\n  protected override firstUpdated(changedProperties: PropertyValues<this>): void {\n    super.firstUpdated(changedProperties);\n\n    if (!this.hasAttribute('aria-hidden')) {\n      this.setAttribute('aria-hidden', 'true');\n    }\n  }\n\n  protected override render(): TemplateResult {\n    return this._sbbAngularCompatibility ? html`<slot></slot>` : super.render();\n  }\n}\n\ndeclare global {\n  interface HTMLElementTagNameMap {\n    // eslint-disable-next-line @typescript-eslint/naming-convention\n    'sbb-icon': SbbIconElement;\n  }\n}\n","import { html, type LitElement, nothing, type TemplateResult } from 'lit';\nimport { property } from 'lit/decorators.js';\n\nimport type { AbstractConstructor } from '../core/mixins.js';\n\nimport './icon.js';\n\nexport declare class SbbIconNameMixinType {\n  public iconName?: string;\n  protected renderIconSlot(classname?: string): TemplateResult;\n}\n\n/**\n * Enhance your component with an iconName property and provides a template for the icon slot.\n */\n// eslint-disable-next-line @typescript-eslint/naming-convention\nexport const SbbIconNameMixin = <T extends AbstractConstructor<LitElement>>(\n  superClass: T,\n): AbstractConstructor<SbbIconNameMixinType> & T => {\n  abstract class SbbIconNameElement extends superClass implements Partial<SbbIconNameMixinType> {\n    /**\n     * The icon name we want to use, choose from the small icon variants\n     * from the ui-icons category from here\n     * https://icons.app.sbb.ch.\n     */\n    @property({ attribute: 'icon-name', reflect: true }) public iconName?: string;\n\n    protected renderIconSlot(classname?: string): TemplateResult {\n      return html`\n        <slot name=\"icon\">\n          ${this.iconName\n            ? html`<sbb-icon name=${this.iconName} class=${classname || nothing}></sbb-icon>`\n            : nothing}\n        </slot>\n      `;\n    }\n  }\n\n  return SbbIconNameElement as unknown as AbstractConstructor<SbbIconNameMixinType> & T;\n};\n"],"names":["__decorateClass"],"mappings":"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"}
@@ -1 +1 @@
1
- {"version":3,"file":"menu.d.ts","sourceRoot":"","sources":["../../../../../src/elements/menu/menu/menu.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,KAAK,cAAc,EAAQ,KAAK,cAAc,EAAE,MAAM,KAAK,CAAC;AAYrE,OAAO,EAAE,uBAAuB,EAAE,MAAM,6BAA6B,CAAC;AAYtE,OAAO,KAAK,EAAE,oBAAoB,EAAE,MAAM,mBAAmB,CAAC;AAC9D,OAAO,KAAK,EAAE,kBAAkB,EAAE,MAAM,iBAAiB,CAAC;;AAkB1D;;;;;;;;;;;GAWG;AACH,qBACa,cAAe,SAAQ,mBAGV;IACxB,OAAuB,MAAM,EAAE,cAAc,CAAS;IACtD,mBAA4B,mBAAmB,WAAwC;IAEvF;;;OAGG;IACH,IACW,OAAO,CAAC,KAAK,EAAE,MAAM,GAAG,WAAW,GAAG,IAAI,EAIpD;IACD,IAAW,OAAO,IAAI,MAAM,GAAG,WAAW,GAAG,IAAI,CAEhD;IACD,OAAO,CAAC,QAAQ,CAAqC;IAErD;;;OAGG;IACyD,sBAAsB,CAAC,EAAE,MAAM,CAAC;IAE5F,OAAO,CAAC,KAAK,CAAkB;IAC/B,OAAO,CAAC,eAAe,CAA4B;IACnD,OAAO,CAAC,yBAAyB,CAAkB;IACnD,OAAO,CAAC,eAAe,CAAmB;IAC1C,OAAO,CAAC,uBAAuB,CAAmB;IAClD,OAAO,CAAC,MAAM,CAAyC;IACvD,OAAO,CAAC,aAAa,CAAyB;IAC9C,OAAO,CAAC,cAAc,CAA0B;IAEhD;;OAEG;IACI,IAAI,IAAI,IAAI;IAmBnB;;OAEG;IACI,KAAK,IAAI,IAAI;IAapB;;OAEG;IACH,OAAO,CAAC,QAAQ;IAOhB,OAAO,CAAC,cAAc;YAsBR,eAAe;IAY7B,OAAO,CAAC,2BAA2B;IAWnB,iBAAiB,IAAI,IAAI;IAoBzB,oBAAoB,IAAI,IAAI;IAQ5C,OAAO,CAAC,cAAc;IAmBtB,OAAO,CAAC,UAAU;IAsBlB,OAAO,CAAC,mBAAmB;IAwB3B,OAAO,CAAC,+BAA+B;IAQvC,OAAO,CAAC,oBAAoB,CAE1B;IAGF,OAAO,CAAC,qBAAqB,CAI3B;IAMF,OAAO,CAAC,mBAAmB;IA8B3B,OAAO,CAAC,aAAa;IAOrB,OAAO,CAAC,gBAAgB;cAyBL,MAAM,IAAI,cAAc;CAqB5C;AAED,OAAO,CAAC,MAAM,CAAC;IACb,UAAU,qBAAqB;QAE7B,UAAU,EAAE,cAAc,CAAC;KAC5B;CACF"}
1
+ {"version":3,"file":"menu.d.ts","sourceRoot":"","sources":["../../../../../src/elements/menu/menu/menu.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,KAAK,cAAc,EAAQ,KAAK,cAAc,EAAE,MAAM,KAAK,CAAC;AAYrE,OAAO,EAAE,uBAAuB,EAAE,MAAM,6BAA6B,CAAC;AAYtE,OAAO,KAAK,EAAE,oBAAoB,EAAE,MAAM,mBAAmB,CAAC;AAC9D,OAAO,KAAK,EAAE,kBAAkB,EAAE,MAAM,iBAAiB,CAAC;;AAkB1D;;;;;;;;;;;GAWG;AACH,qBACa,cAAe,SAAQ,mBAGV;IACxB,OAAuB,MAAM,EAAE,cAAc,CAAS;IACtD,mBAA4B,mBAAmB,WAAwC;IAEvF;;;OAGG;IACH,IACW,OAAO,CAAC,KAAK,EAAE,MAAM,GAAG,WAAW,GAAG,IAAI,EAIpD;IACD,IAAW,OAAO,IAAI,MAAM,GAAG,WAAW,GAAG,IAAI,CAEhD;IACD,OAAO,CAAC,QAAQ,CAAqC;IAErD;;;OAGG;IACyD,sBAAsB,CAAC,EAAE,MAAM,CAAC;IAE5F,OAAO,CAAC,KAAK,CAAkB;IAC/B,OAAO,CAAC,eAAe,CAA4B;IACnD,OAAO,CAAC,yBAAyB,CAAkB;IACnD,OAAO,CAAC,eAAe,CAAmB;IAC1C,OAAO,CAAC,uBAAuB,CAAmB;IAClD,OAAO,CAAC,MAAM,CAAyC;IACvD,OAAO,CAAC,aAAa,CAAyB;IAC9C,OAAO,CAAC,cAAc,CAA0B;IAEhD;;OAEG;IACI,IAAI,IAAI,IAAI;IAmBnB;;OAEG;IACI,KAAK,IAAI,IAAI;IAapB;;OAEG;IACH,OAAO,CAAC,QAAQ;IAOhB,OAAO,CAAC,cAAc;YAsBR,eAAe;IAY7B,OAAO,CAAC,2BAA2B;IAWnB,iBAAiB,IAAI,IAAI;IAoBzB,oBAAoB,IAAI,IAAI;IAS5C,OAAO,CAAC,cAAc;IAmBtB,OAAO,CAAC,UAAU;IAsBlB,OAAO,CAAC,mBAAmB;IAwB3B,OAAO,CAAC,+BAA+B;IAQvC,OAAO,CAAC,oBAAoB,CAE1B;IAGF,OAAO,CAAC,qBAAqB,CAI3B;IAMF,OAAO,CAAC,mBAAmB;IA8B3B,OAAO,CAAC,aAAa;IAOrB,OAAO,CAAC,gBAAgB;cAyBL,MAAM,IAAI,cAAc;CAqB5C;AAED,OAAO,CAAC,MAAM,CAAC;IACb,UAAU,qBAAqB;QAE7B,UAAU,EAAE,cAAc,CAAC;KAC5B;CACF"}
@@ -397,6 +397,7 @@ let SbbMenuElement = class extends SbbNamedSlotListMixin(SbbOpenCloseBaseElement
397
397
  (_b = this._windowEventsController) == null ? void 0 : _b.abort();
398
398
  this._focusHandler.disconnect();
399
399
  removeInertMechanism();
400
+ this._scrollHandler.enableScroll();
400
401
  }
401
402
  _checkListCase(event) {
402
403
  var _a;
@@ -540,4 +541,4 @@ SbbMenuElement = __decorateClass([
540
541
  export {
541
542
  SbbMenuElement
542
543
  };
543
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"menu.js","sources":["../../../../src/elements/menu/menu/menu.ts"],"sourcesContent":["import { type CSSResultGroup, html, type TemplateResult } from 'lit';\nimport { customElement, property } from 'lit/decorators.js';\nimport { ref } from 'lit/directives/ref.js';\n\nimport {\n  getNextElementIndex,\n  interactivityChecker,\n  IS_FOCUSABLE_QUERY,\n  isArrowKeyPressed,\n  SbbFocusHandler,\n  setModalityOnNextFocus,\n} from '../../core/a11y.js';\nimport { SbbOpenCloseBaseElement } from '../../core/base-elements.js';\nimport { SbbConnectedAbortController } from '../../core/controllers.js';\nimport { findReferencedElement, isBreakpoint, SbbScrollHandler } from '../../core/dom.js';\nimport { SbbNamedSlotListMixin } from '../../core/mixins.js';\nimport {\n  applyInertMechanism,\n  getElementPosition,\n  isEventOnElement,\n  removeAriaOverlayTriggerAttributes,\n  removeInertMechanism,\n  setAriaOverlayTriggerAttributes,\n} from '../../core/overlay.js';\nimport type { SbbMenuButtonElement } from '../menu-button.js';\nimport type { SbbMenuLinkElement } from '../menu-link.js';\n\nimport style from './menu.scss?lit&inline';\n\nconst MENU_OFFSET = 8;\nconst INTERACTIVE_ELEMENTS = [\n  'A',\n  'BUTTON',\n  'SBB-BUTTON',\n  'SBB-BUTTON-LINK',\n  'SBB-LINK',\n  'SBB-BLOCK-LINK',\n  'SBB-LINK-BUTTON',\n  'SBB-BLOCK-LINK-BUTTON',\n];\n\nlet nextId = 0;\n\n/**\n * It displays a contextual menu with one or more action element.\n *\n * @slot - Use the unnamed slot to add `sbb-menu-button`/`sbb-menu-link` or other elements to the menu.\n * @event {CustomEvent<void>} willOpen - Emits whenever the `sbb-menu` starts the opening transition. Can be canceled.\n * @event {CustomEvent<void>} didOpen - Emits whenever the `sbb-menu` is opened.\n * @event {CustomEvent<void>} willClose - Emits whenever the `sbb-menu` begins the closing transition. Can be canceled.\n * @event {CustomEvent<void>} didClose - Emits whenever the `sbb-menu` is closed.\n * @cssprop [--sbb-menu-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 */\n@customElement('sbb-menu')\nexport class SbbMenuElement extends SbbNamedSlotListMixin<\n  SbbMenuButtonElement | SbbMenuLinkElement,\n  typeof SbbOpenCloseBaseElement\n>(SbbOpenCloseBaseElement) {\n  public static override styles: CSSResultGroup = style;\n  protected override readonly listChildLocalNames = ['sbb-menu-button', 'sbb-menu-link'];\n\n  /**\n   * The element that will trigger the menu overlay.\n   * Accepts both a string (id of an element) or an HTML element.\n   */\n  @property()\n  public set trigger(value: string | HTMLElement | null) {\n    const oldValue = this._trigger;\n    this._trigger = value;\n    this._removeTriggerClickListener(this._trigger, oldValue);\n  }\n  public get trigger(): string | HTMLElement | null {\n    return this._trigger;\n  }\n  private _trigger: string | HTMLElement | null = null;\n\n  /**\n   * This will be forwarded as aria-label to the inner list.\n   * Used only if the menu automatically renders the actions inside as a list.\n   */\n  @property({ attribute: 'list-accessibility-label' }) public listAccessibilityLabel?: string;\n\n  private _menu!: HTMLDivElement;\n  private _triggerElement: HTMLElement | null = null;\n  private _isPointerDownEventOnMenu: boolean = false;\n  private _menuController!: AbortController;\n  private _windowEventsController!: AbortController;\n  private _abort = new SbbConnectedAbortController(this);\n  private _focusHandler = new SbbFocusHandler();\n  private _scrollHandler = new SbbScrollHandler();\n\n  /**\n   * Opens the menu on trigger click.\n   */\n  public open(): void {\n    if (this.state === 'closing' || !this._menu) {\n      return;\n    }\n\n    if (!this.willOpen.emit()) {\n      return;\n    }\n\n    this.state = 'opening';\n    this._setMenuPosition();\n    this._triggerElement?.setAttribute('aria-expanded', 'true');\n\n    // Starting from breakpoint medium, disable scroll\n    if (!isBreakpoint('medium')) {\n      this._scrollHandler.disableScroll();\n    }\n  }\n\n  /**\n   * Closes the menu.\n   */\n  public close(): void {\n    if (this.state === 'opening') {\n      return;\n    }\n\n    if (!this.willClose.emit()) {\n      return;\n    }\n\n    this.state = 'closing';\n    this._triggerElement?.setAttribute('aria-expanded', 'false');\n  }\n\n  /**\n   * Handles click and checks if its target is a sbb-menu-button/sbb-menu-link.\n   */\n  private _onClick(event: Event): void {\n    const target = event.target as HTMLElement | undefined;\n    if (target?.localName === 'sbb-menu-button' || target?.localName === 'sbb-menu-link') {\n      this.close();\n    }\n  }\n\n  private _handleKeyDown(evt: KeyboardEvent): void {\n    if (!isArrowKeyPressed(evt)) {\n      return;\n    }\n    evt.preventDefault();\n\n    const enabledActions: Element[] = Array.from(\n      this.querySelectorAll<SbbMenuButtonElement | SbbMenuLinkElement>(\n        'sbb-menu-button, sbb-menu-link',\n      ),\n    ).filter(\n      (el: SbbMenuButtonElement | SbbMenuLinkElement) =>\n        !el.disabled && interactivityChecker.isVisible(el),\n    );\n\n    const current = enabledActions.findIndex((e: Element) => e === evt.target);\n    const nextIndex = getNextElementIndex(evt, current, enabledActions.length);\n\n    (enabledActions[nextIndex] as HTMLElement).focus();\n  }\n\n  // Closes the menu on \"Esc\" key pressed and traps focus within the menu.\n  private async _onKeydownEvent(event: KeyboardEvent): Promise<void> {\n    if (this.state !== 'opened') {\n      return;\n    }\n\n    if (event.key === 'Escape') {\n      this.close();\n      return;\n    }\n  }\n\n  // Removes trigger click listener on trigger change.\n  private _removeTriggerClickListener(\n    newValue: string | HTMLElement | null,\n    oldValue: string | HTMLElement | null,\n  ): void {\n    if (newValue !== oldValue) {\n      this._menuController?.abort();\n      this._windowEventsController?.abort();\n      this._configure(this.trigger);\n    }\n  }\n\n  public override connectedCallback(): void {\n    super.connectedCallback();\n    const signal = this._abort.signal;\n    this.addEventListener('click', (e) => this._onClick(e), { signal });\n    this.addEventListener('keydown', (e) => this._handleKeyDown(e), { signal });\n    // Due to the fact that menu can both be a list and just a container, we need to check its\n    // state before the SbbNamedSlotListMixin handles the slotchange event, in order to avoid\n    // it interpreting the non list case as a list.\n    this.shadowRoot?.addEventListener('slotchange', (e) => this._checkListCase(e), {\n      signal,\n      capture: true,\n    });\n    // Validate trigger element and attach event listeners\n    this._configure(this.trigger);\n\n    if (this.state === 'opened') {\n      applyInertMechanism(this);\n    }\n  }\n\n  public override disconnectedCallback(): void {\n    super.disconnectedCallback();\n    this._menuController?.abort();\n    this._windowEventsController?.abort();\n    this._focusHandler.disconnect();\n    removeInertMechanism();\n  }\n\n  private _checkListCase(event: Event): void {\n    // If all children are sbb-menu-button/menu-link instances, we render them as a list.\n    if (\n      this.children?.length &&\n      Array.from(this.children ?? []).every(\n        (c) => c.localName === 'sbb-menu-button' || c.localName === 'sbb-menu-link',\n      )\n    ) {\n      return;\n    }\n\n    event.stopImmediatePropagation();\n    if (this.listChildren.length) {\n      this.listChildren.forEach((c) => c.removeAttribute('slot'));\n      this.listChildren = [];\n    }\n  }\n\n  // Check if the trigger is valid and attach click event listeners.\n  private _configure(trigger: string | HTMLElement | null): void {\n    removeAriaOverlayTriggerAttributes(this._triggerElement);\n\n    if (!trigger) {\n      return;\n    }\n\n    this._triggerElement = findReferencedElement(trigger);\n\n    if (!this._triggerElement) {\n      return;\n    }\n\n    this.id = this.id || `sbb-menu-${nextId++}`;\n    setAriaOverlayTriggerAttributes(this._triggerElement, 'menu', this.id, this.state);\n    this._menuController?.abort();\n    this._menuController = new AbortController();\n    this._triggerElement.addEventListener('click', () => this.open(), {\n      signal: this._menuController.signal,\n    });\n  }\n\n  private _attachWindowEvents(): void {\n    this._windowEventsController = new AbortController();\n    document.addEventListener('scroll', () => this._setMenuPosition(), {\n      passive: true,\n      signal: this._windowEventsController.signal,\n    });\n    window.addEventListener('resize', () => this._setMenuPosition(), {\n      passive: true,\n      signal: this._windowEventsController.signal,\n    });\n    window.addEventListener('keydown', (event: KeyboardEvent) => this._onKeydownEvent(event), {\n      signal: this._windowEventsController.signal,\n    });\n\n    // Close menu on backdrop click\n    window.addEventListener('pointerdown', this._pointerDownListener, {\n      signal: this._windowEventsController.signal,\n    });\n    window.addEventListener('pointerup', this._closeOnBackdropClick, {\n      signal: this._windowEventsController.signal,\n    });\n  }\n\n  // Close menu at any click on an interactive element inside the <sbb-menu> that bubbles to the container.\n  private _closeOnInteractiveElementClick(event: Event): void {\n    const target = event.target as HTMLElement;\n    if (INTERACTIVE_ELEMENTS.includes(target.nodeName) && !target.hasAttribute('disabled')) {\n      this.close();\n    }\n  }\n\n  // Check if the pointerdown event target is triggered on the menu.\n  private _pointerDownListener = (event: PointerEvent): void => {\n    this._isPointerDownEventOnMenu = isEventOnElement(this._menu, event);\n  };\n\n  // Close menu on backdrop click.\n  private _closeOnBackdropClick = (event: PointerEvent): void => {\n    if (!this._isPointerDownEventOnMenu && !isEventOnElement(this._menu, event)) {\n      this.close();\n    }\n  };\n\n  // Set menu position (x, y) to '0' once the menu is closed and the transition ended to prevent the\n  // viewport from overflowing. And set the focus to the first focusable element once the menu is open.\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 _onMenuAnimationEnd(event: AnimationEvent): void {\n    if (event.animationName === 'open' && this.state === 'opening') {\n      this.state = 'opened';\n      this.didOpen.emit();\n      applyInertMechanism(this);\n      this._setMenuFocus();\n      this._focusHandler.trap(this);\n      this._attachWindowEvents();\n    } else if (event.animationName === 'close' && this.state === 'closing') {\n      this.state = 'closed';\n      this._menu?.firstElementChild?.scrollTo(0, 0);\n      removeInertMechanism();\n      setModalityOnNextFocus(this._triggerElement);\n      // Manually focus last focused element\n      this._triggerElement?.focus({\n        // When inside the sbb-header, we prevent the scroll to avoid the snapping to the top of the page\n        preventScroll:\n          this._triggerElement.localName === 'sbb-header-button' ||\n          this._triggerElement.localName === 'sbb-header-link',\n      });\n      this.didClose.emit();\n      this._windowEventsController?.abort();\n      this._focusHandler.disconnect();\n\n      // Starting from breakpoint medium, enable scroll\n      this._scrollHandler.enableScroll();\n    }\n  }\n\n  // Set focus on the first focusable element.\n  private _setMenuFocus(): void {\n    const firstFocusable = this.querySelector(IS_FOCUSABLE_QUERY) as HTMLElement;\n    setModalityOnNextFocus(firstFocusable);\n    firstFocusable.focus();\n  }\n\n  // Set menu position and max height if the breakpoint is medium-ultra.\n  private _setMenuPosition(): void {\n    // Starting from breakpoint medium\n    if (\n      !isBreakpoint('medium') ||\n      !this._menu ||\n      !this._triggerElement ||\n      this.state === 'closing'\n    ) {\n      return;\n    }\n\n    const menuPosition = getElementPosition(\n      this.shadowRoot!.querySelector('.sbb-menu__content')!,\n      this._triggerElement,\n      this.shadowRoot!.querySelector('.sbb-menu__container')!,\n      {\n        verticalOffset: MENU_OFFSET,\n      },\n    );\n\n    this.style.setProperty('--sbb-menu-position-x', `${menuPosition.left}px`);\n    this.style.setProperty('--sbb-menu-position-y', `${menuPosition.top}px`);\n    this.style.setProperty('--sbb-menu-max-height', menuPosition.maxHeight);\n  }\n\n  protected override render(): TemplateResult {\n    // TODO: Handle case with other elements than sbb-menu-button/sbb-menu-link.\n    return html`\n      <div class=\"sbb-menu__container\">\n        <div\n          @animationend=${(event: AnimationEvent) => this._onMenuAnimationEnd(event)}\n          class=\"sbb-menu\"\n          ${ref((el?: Element) => (this._menu = el as HTMLDivElement))}\n        >\n          <div\n            @click=${(event: Event) => this._closeOnInteractiveElementClick(event)}\n            class=\"sbb-menu__content\"\n          >\n            ${this.listChildren.length\n              ? this.renderList({ class: 'sbb-menu-list', ariaLabel: this.listAccessibilityLabel })\n              : html`<slot></slot>`}\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-menu': SbbMenuElement;\n  }\n}\n"],"names":[],"mappings":"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"}
544
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"menu.js","sources":["../../../../src/elements/menu/menu/menu.ts"],"sourcesContent":["import { type CSSResultGroup, html, type TemplateResult } from 'lit';\nimport { customElement, property } from 'lit/decorators.js';\nimport { ref } from 'lit/directives/ref.js';\n\nimport {\n  getNextElementIndex,\n  interactivityChecker,\n  IS_FOCUSABLE_QUERY,\n  isArrowKeyPressed,\n  SbbFocusHandler,\n  setModalityOnNextFocus,\n} from '../../core/a11y.js';\nimport { SbbOpenCloseBaseElement } from '../../core/base-elements.js';\nimport { SbbConnectedAbortController } from '../../core/controllers.js';\nimport { findReferencedElement, isBreakpoint, SbbScrollHandler } from '../../core/dom.js';\nimport { SbbNamedSlotListMixin } from '../../core/mixins.js';\nimport {\n  applyInertMechanism,\n  getElementPosition,\n  isEventOnElement,\n  removeAriaOverlayTriggerAttributes,\n  removeInertMechanism,\n  setAriaOverlayTriggerAttributes,\n} from '../../core/overlay.js';\nimport type { SbbMenuButtonElement } from '../menu-button.js';\nimport type { SbbMenuLinkElement } from '../menu-link.js';\n\nimport style from './menu.scss?lit&inline';\n\nconst MENU_OFFSET = 8;\nconst INTERACTIVE_ELEMENTS = [\n  'A',\n  'BUTTON',\n  'SBB-BUTTON',\n  'SBB-BUTTON-LINK',\n  'SBB-LINK',\n  'SBB-BLOCK-LINK',\n  'SBB-LINK-BUTTON',\n  'SBB-BLOCK-LINK-BUTTON',\n];\n\nlet nextId = 0;\n\n/**\n * It displays a contextual menu with one or more action element.\n *\n * @slot - Use the unnamed slot to add `sbb-menu-button`/`sbb-menu-link` or other elements to the menu.\n * @event {CustomEvent<void>} willOpen - Emits whenever the `sbb-menu` starts the opening transition. Can be canceled.\n * @event {CustomEvent<void>} didOpen - Emits whenever the `sbb-menu` is opened.\n * @event {CustomEvent<void>} willClose - Emits whenever the `sbb-menu` begins the closing transition. Can be canceled.\n * @event {CustomEvent<void>} didClose - Emits whenever the `sbb-menu` is closed.\n * @cssprop [--sbb-menu-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 */\n@customElement('sbb-menu')\nexport class SbbMenuElement extends SbbNamedSlotListMixin<\n  SbbMenuButtonElement | SbbMenuLinkElement,\n  typeof SbbOpenCloseBaseElement\n>(SbbOpenCloseBaseElement) {\n  public static override styles: CSSResultGroup = style;\n  protected override readonly listChildLocalNames = ['sbb-menu-button', 'sbb-menu-link'];\n\n  /**\n   * The element that will trigger the menu overlay.\n   * Accepts both a string (id of an element) or an HTML element.\n   */\n  @property()\n  public set trigger(value: string | HTMLElement | null) {\n    const oldValue = this._trigger;\n    this._trigger = value;\n    this._removeTriggerClickListener(this._trigger, oldValue);\n  }\n  public get trigger(): string | HTMLElement | null {\n    return this._trigger;\n  }\n  private _trigger: string | HTMLElement | null = null;\n\n  /**\n   * This will be forwarded as aria-label to the inner list.\n   * Used only if the menu automatically renders the actions inside as a list.\n   */\n  @property({ attribute: 'list-accessibility-label' }) public listAccessibilityLabel?: string;\n\n  private _menu!: HTMLDivElement;\n  private _triggerElement: HTMLElement | null = null;\n  private _isPointerDownEventOnMenu: boolean = false;\n  private _menuController!: AbortController;\n  private _windowEventsController!: AbortController;\n  private _abort = new SbbConnectedAbortController(this);\n  private _focusHandler = new SbbFocusHandler();\n  private _scrollHandler = new SbbScrollHandler();\n\n  /**\n   * Opens the menu on trigger click.\n   */\n  public open(): void {\n    if (this.state === 'closing' || !this._menu) {\n      return;\n    }\n\n    if (!this.willOpen.emit()) {\n      return;\n    }\n\n    this.state = 'opening';\n    this._setMenuPosition();\n    this._triggerElement?.setAttribute('aria-expanded', 'true');\n\n    // Starting from breakpoint medium, disable scroll\n    if (!isBreakpoint('medium')) {\n      this._scrollHandler.disableScroll();\n    }\n  }\n\n  /**\n   * Closes the menu.\n   */\n  public close(): void {\n    if (this.state === 'opening') {\n      return;\n    }\n\n    if (!this.willClose.emit()) {\n      return;\n    }\n\n    this.state = 'closing';\n    this._triggerElement?.setAttribute('aria-expanded', 'false');\n  }\n\n  /**\n   * Handles click and checks if its target is a sbb-menu-button/sbb-menu-link.\n   */\n  private _onClick(event: Event): void {\n    const target = event.target as HTMLElement | undefined;\n    if (target?.localName === 'sbb-menu-button' || target?.localName === 'sbb-menu-link') {\n      this.close();\n    }\n  }\n\n  private _handleKeyDown(evt: KeyboardEvent): void {\n    if (!isArrowKeyPressed(evt)) {\n      return;\n    }\n    evt.preventDefault();\n\n    const enabledActions: Element[] = Array.from(\n      this.querySelectorAll<SbbMenuButtonElement | SbbMenuLinkElement>(\n        'sbb-menu-button, sbb-menu-link',\n      ),\n    ).filter(\n      (el: SbbMenuButtonElement | SbbMenuLinkElement) =>\n        !el.disabled && interactivityChecker.isVisible(el),\n    );\n\n    const current = enabledActions.findIndex((e: Element) => e === evt.target);\n    const nextIndex = getNextElementIndex(evt, current, enabledActions.length);\n\n    (enabledActions[nextIndex] as HTMLElement).focus();\n  }\n\n  // Closes the menu on \"Esc\" key pressed and traps focus within the menu.\n  private async _onKeydownEvent(event: KeyboardEvent): Promise<void> {\n    if (this.state !== 'opened') {\n      return;\n    }\n\n    if (event.key === 'Escape') {\n      this.close();\n      return;\n    }\n  }\n\n  // Removes trigger click listener on trigger change.\n  private _removeTriggerClickListener(\n    newValue: string | HTMLElement | null,\n    oldValue: string | HTMLElement | null,\n  ): void {\n    if (newValue !== oldValue) {\n      this._menuController?.abort();\n      this._windowEventsController?.abort();\n      this._configure(this.trigger);\n    }\n  }\n\n  public override connectedCallback(): void {\n    super.connectedCallback();\n    const signal = this._abort.signal;\n    this.addEventListener('click', (e) => this._onClick(e), { signal });\n    this.addEventListener('keydown', (e) => this._handleKeyDown(e), { signal });\n    // Due to the fact that menu can both be a list and just a container, we need to check its\n    // state before the SbbNamedSlotListMixin handles the slotchange event, in order to avoid\n    // it interpreting the non list case as a list.\n    this.shadowRoot?.addEventListener('slotchange', (e) => this._checkListCase(e), {\n      signal,\n      capture: true,\n    });\n    // Validate trigger element and attach event listeners\n    this._configure(this.trigger);\n\n    if (this.state === 'opened') {\n      applyInertMechanism(this);\n    }\n  }\n\n  public override disconnectedCallback(): void {\n    super.disconnectedCallback();\n    this._menuController?.abort();\n    this._windowEventsController?.abort();\n    this._focusHandler.disconnect();\n    removeInertMechanism();\n    this._scrollHandler.enableScroll();\n  }\n\n  private _checkListCase(event: Event): void {\n    // If all children are sbb-menu-button/menu-link instances, we render them as a list.\n    if (\n      this.children?.length &&\n      Array.from(this.children ?? []).every(\n        (c) => c.localName === 'sbb-menu-button' || c.localName === 'sbb-menu-link',\n      )\n    ) {\n      return;\n    }\n\n    event.stopImmediatePropagation();\n    if (this.listChildren.length) {\n      this.listChildren.forEach((c) => c.removeAttribute('slot'));\n      this.listChildren = [];\n    }\n  }\n\n  // Check if the trigger is valid and attach click event listeners.\n  private _configure(trigger: string | HTMLElement | null): void {\n    removeAriaOverlayTriggerAttributes(this._triggerElement);\n\n    if (!trigger) {\n      return;\n    }\n\n    this._triggerElement = findReferencedElement(trigger);\n\n    if (!this._triggerElement) {\n      return;\n    }\n\n    this.id = this.id || `sbb-menu-${nextId++}`;\n    setAriaOverlayTriggerAttributes(this._triggerElement, 'menu', this.id, this.state);\n    this._menuController?.abort();\n    this._menuController = new AbortController();\n    this._triggerElement.addEventListener('click', () => this.open(), {\n      signal: this._menuController.signal,\n    });\n  }\n\n  private _attachWindowEvents(): void {\n    this._windowEventsController = new AbortController();\n    document.addEventListener('scroll', () => this._setMenuPosition(), {\n      passive: true,\n      signal: this._windowEventsController.signal,\n    });\n    window.addEventListener('resize', () => this._setMenuPosition(), {\n      passive: true,\n      signal: this._windowEventsController.signal,\n    });\n    window.addEventListener('keydown', (event: KeyboardEvent) => this._onKeydownEvent(event), {\n      signal: this._windowEventsController.signal,\n    });\n\n    // Close menu on backdrop click\n    window.addEventListener('pointerdown', this._pointerDownListener, {\n      signal: this._windowEventsController.signal,\n    });\n    window.addEventListener('pointerup', this._closeOnBackdropClick, {\n      signal: this._windowEventsController.signal,\n    });\n  }\n\n  // Close menu at any click on an interactive element inside the <sbb-menu> that bubbles to the container.\n  private _closeOnInteractiveElementClick(event: Event): void {\n    const target = event.target as HTMLElement;\n    if (INTERACTIVE_ELEMENTS.includes(target.nodeName) && !target.hasAttribute('disabled')) {\n      this.close();\n    }\n  }\n\n  // Check if the pointerdown event target is triggered on the menu.\n  private _pointerDownListener = (event: PointerEvent): void => {\n    this._isPointerDownEventOnMenu = isEventOnElement(this._menu, event);\n  };\n\n  // Close menu on backdrop click.\n  private _closeOnBackdropClick = (event: PointerEvent): void => {\n    if (!this._isPointerDownEventOnMenu && !isEventOnElement(this._menu, event)) {\n      this.close();\n    }\n  };\n\n  // Set menu position (x, y) to '0' once the menu is closed and the transition ended to prevent the\n  // viewport from overflowing. And set the focus to the first focusable element once the menu is open.\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 _onMenuAnimationEnd(event: AnimationEvent): void {\n    if (event.animationName === 'open' && this.state === 'opening') {\n      this.state = 'opened';\n      this.didOpen.emit();\n      applyInertMechanism(this);\n      this._setMenuFocus();\n      this._focusHandler.trap(this);\n      this._attachWindowEvents();\n    } else if (event.animationName === 'close' && this.state === 'closing') {\n      this.state = 'closed';\n      this._menu?.firstElementChild?.scrollTo(0, 0);\n      removeInertMechanism();\n      setModalityOnNextFocus(this._triggerElement);\n      // Manually focus last focused element\n      this._triggerElement?.focus({\n        // When inside the sbb-header, we prevent the scroll to avoid the snapping to the top of the page\n        preventScroll:\n          this._triggerElement.localName === 'sbb-header-button' ||\n          this._triggerElement.localName === 'sbb-header-link',\n      });\n      this.didClose.emit();\n      this._windowEventsController?.abort();\n      this._focusHandler.disconnect();\n\n      // Starting from breakpoint medium, enable scroll\n      this._scrollHandler.enableScroll();\n    }\n  }\n\n  // Set focus on the first focusable element.\n  private _setMenuFocus(): void {\n    const firstFocusable = this.querySelector(IS_FOCUSABLE_QUERY) as HTMLElement;\n    setModalityOnNextFocus(firstFocusable);\n    firstFocusable.focus();\n  }\n\n  // Set menu position and max height if the breakpoint is medium-ultra.\n  private _setMenuPosition(): void {\n    // Starting from breakpoint medium\n    if (\n      !isBreakpoint('medium') ||\n      !this._menu ||\n      !this._triggerElement ||\n      this.state === 'closing'\n    ) {\n      return;\n    }\n\n    const menuPosition = getElementPosition(\n      this.shadowRoot!.querySelector('.sbb-menu__content')!,\n      this._triggerElement,\n      this.shadowRoot!.querySelector('.sbb-menu__container')!,\n      {\n        verticalOffset: MENU_OFFSET,\n      },\n    );\n\n    this.style.setProperty('--sbb-menu-position-x', `${menuPosition.left}px`);\n    this.style.setProperty('--sbb-menu-position-y', `${menuPosition.top}px`);\n    this.style.setProperty('--sbb-menu-max-height', menuPosition.maxHeight);\n  }\n\n  protected override render(): TemplateResult {\n    // TODO: Handle case with other elements than sbb-menu-button/sbb-menu-link.\n    return html`\n      <div class=\"sbb-menu__container\">\n        <div\n          @animationend=${(event: AnimationEvent) => this._onMenuAnimationEnd(event)}\n          class=\"sbb-menu\"\n          ${ref((el?: Element) => (this._menu = el as HTMLDivElement))}\n        >\n          <div\n            @click=${(event: Event) => this._closeOnInteractiveElementClick(event)}\n            class=\"sbb-menu__content\"\n          >\n            ${this.listChildren.length\n              ? this.renderList({ class: 'sbb-menu-list', ariaLabel: this.listAccessibilityLabel })\n              : html`<slot></slot>`}\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-menu': SbbMenuElement;\n  }\n}\n"],"names":[],"mappings":"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"}
@@ -1 +1 @@
1
- {"version":3,"file":"navigation.d.ts","sourceRoot":"","sources":["../../../../../src/elements/navigation/navigation/navigation.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,cAAc,EAAE,cAAc,EAAE,MAAM,KAAK,CAAC;AAM1D,OAAO,EAAE,uBAAuB,EAAE,MAAM,6BAA6B,CAAC;AAmBtE,OAAO,oCAAoC,CAAC;;AAW5C;;;;;;;;;;;GAWG;AACH,qBAIa,oBAAqB,SAAQ,yBAAgD;IACxF,OAAuB,MAAM,EAAE,cAAc,CAAS;IAEtD;;;OAGG;IACH,IACW,OAAO,CAAC,KAAK,EAAE,MAAM,GAAG,WAAW,GAAG,IAAI,EAIpD;IACD,IAAW,OAAO,IAAI,MAAM,GAAG,WAAW,GAAG,IAAI,CAEhD;IACD,OAAO,CAAC,QAAQ,CAAqC;IAErD;;OAEG;IAC0D,uBAAuB,EAChF,MAAM,GACN,SAAS,CAAC;IAEd;;OAEG;IACM,OAAO,CAAC,wBAAwB,CAA4B;IAErE,IAAW,uBAAuB,IAAI,WAAW,GAAG,IAAI,CAEvD;IAED,OAAO,CAAC,WAAW,CAAkB;IACrC,OAAO,CAAC,yBAAyB,CAAe;IAChD,OAAO,CAAC,eAAe,CAA4B;IACnD,OAAO,CAAC,qBAAqB,CAAmB;IAChD,OAAO,CAAC,uBAAuB,CAAmB;IAClD,OAAO,CAAC,MAAM,CAAyC;IACvD,OAAO,CAAC,SAAS,CAAmC;IACpD,OAAO,CAAC,aAAa,CAAyB;IAC9C,OAAO,CAAC,cAAc,CAA0B;IAChD,OAAO,CAAC,+BAA+B,CAAkB;IACzD,OAAO,CAAC,sBAAsB,CAA8C;IAC5E,OAAO,CAAC,mBAAmB,CAEzB;IACF,OAAO,CAAC,yBAAyB,CAAgE;IAEjG;;OAEG;IACI,IAAI,IAAI,IAAI;IAkBnB,OAAO,CAAC,mBAAmB;IAO3B,OAAO,CAAC,mBAAmB;IAO3B;;OAEG;IACI,KAAK,IAAI,IAAI;IAcpB,OAAO,CAAC,2BAA2B;IAYnC,OAAO,CAAC,UAAU;IAqBlB,OAAO,CAAC,gBAAgB,CAEtB;IAIF,OAAO,CAAC,eAAe;IA4BvB,OAAO,CAAC,aAAa;IASrB,OAAO,CAAC,mBAAmB;IAO3B,OAAO,CAAC,sBAAsB;IAS9B,OAAO,CAAC,eAAe;IAQvB,OAAO,CAAC,eAAe;IAOvB,OAAO,CAAC,mBAAmB;IAS3B,OAAO,CAAC,oBAAoB,CAS1B;IAGF,OAAO,CAAC,qBAAqB,CAI3B;IAGF,OAAO,CAAC,0BAA0B;IAWlC,OAAO,CAAC,mBAAmB;IAkBX,iBAAiB,IAAI,IAAI;IAgBzB,oBAAoB,IAAI,IAAI;cAUzB,MAAM,IAAI,cAAc;CAqC5C;AAED,OAAO,CAAC,MAAM,CAAC;IACb,UAAU,qBAAqB;QAE7B,gBAAgB,EAAE,oBAAoB,CAAC;KACxC;CACF"}
1
+ {"version":3,"file":"navigation.d.ts","sourceRoot":"","sources":["../../../../../src/elements/navigation/navigation/navigation.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,cAAc,EAAE,cAAc,EAAE,MAAM,KAAK,CAAC;AAM1D,OAAO,EAAE,uBAAuB,EAAE,MAAM,6BAA6B,CAAC;AAmBtE,OAAO,oCAAoC,CAAC;;AAW5C;;;;;;;;;;;GAWG;AACH,qBAIa,oBAAqB,SAAQ,yBAAgD;IACxF,OAAuB,MAAM,EAAE,cAAc,CAAS;IAEtD;;;OAGG;IACH,IACW,OAAO,CAAC,KAAK,EAAE,MAAM,GAAG,WAAW,GAAG,IAAI,EAIpD;IACD,IAAW,OAAO,IAAI,MAAM,GAAG,WAAW,GAAG,IAAI,CAEhD;IACD,OAAO,CAAC,QAAQ,CAAqC;IAErD;;OAEG;IAC0D,uBAAuB,EAChF,MAAM,GACN,SAAS,CAAC;IAEd;;OAEG;IACM,OAAO,CAAC,wBAAwB,CAA4B;IAErE,IAAW,uBAAuB,IAAI,WAAW,GAAG,IAAI,CAEvD;IAED,OAAO,CAAC,WAAW,CAAkB;IACrC,OAAO,CAAC,yBAAyB,CAAe;IAChD,OAAO,CAAC,eAAe,CAA4B;IACnD,OAAO,CAAC,qBAAqB,CAAmB;IAChD,OAAO,CAAC,uBAAuB,CAAmB;IAClD,OAAO,CAAC,MAAM,CAAyC;IACvD,OAAO,CAAC,SAAS,CAAmC;IACpD,OAAO,CAAC,aAAa,CAAyB;IAC9C,OAAO,CAAC,cAAc,CAA0B;IAChD,OAAO,CAAC,+BAA+B,CAAkB;IACzD,OAAO,CAAC,sBAAsB,CAA8C;IAC5E,OAAO,CAAC,mBAAmB,CAEzB;IACF,OAAO,CAAC,yBAAyB,CAAgE;IAEjG;;OAEG;IACI,IAAI,IAAI,IAAI;IAkBnB,OAAO,CAAC,mBAAmB;IAO3B,OAAO,CAAC,mBAAmB;IAO3B;;OAEG;IACI,KAAK,IAAI,IAAI;IAcpB,OAAO,CAAC,2BAA2B;IAYnC,OAAO,CAAC,UAAU;IAqBlB,OAAO,CAAC,gBAAgB,CAEtB;IAIF,OAAO,CAAC,eAAe;IA4BvB,OAAO,CAAC,aAAa;IASrB,OAAO,CAAC,mBAAmB;IAO3B,OAAO,CAAC,sBAAsB;IAS9B,OAAO,CAAC,eAAe;IAQvB,OAAO,CAAC,eAAe;IAOvB,OAAO,CAAC,mBAAmB;IAS3B,OAAO,CAAC,oBAAoB,CAS1B;IAGF,OAAO,CAAC,qBAAqB,CAI3B;IAGF,OAAO,CAAC,0BAA0B;IAWlC,OAAO,CAAC,mBAAmB;IAkBX,iBAAiB,IAAI,IAAI;IAgBzB,oBAAoB,IAAI,IAAI;cAWzB,MAAM,IAAI,cAAc;CAqC5C;AAED,OAAO,CAAC,MAAM,CAAC;IACb,UAAU,qBAAqB;QAE7B,gBAAgB,EAAE,oBAAoB,CAAC;KACxC;CACF"}
@@ -543,6 +543,7 @@ let SbbNavigationElement = class extends SbbUpdateSchedulerMixin(SbbOpenCloseBas
543
543
  this._navigationObserver.disconnect();
544
544
  this._navigationResizeObserver.disconnect();
545
545
  removeInertMechanism();
546
+ this._scrollHandler.enableScroll();
546
547
  }
547
548
  render() {
548
549
  const closeButton = html`
@@ -600,4 +601,4 @@ SbbNavigationElement = __decorateClass([
600
601
  export {
601
602
  SbbNavigationElement
602
603
  };
603
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"navigation.js","sources":["../../../../src/elements/navigation/navigation/navigation.ts"],"sourcesContent":["import type { CSSResultGroup, TemplateResult } from 'lit';\nimport { html } from 'lit';\nimport { customElement, property, state } from 'lit/decorators.js';\nimport { ref } from 'lit/directives/ref.js';\n\nimport { SbbFocusHandler, setModalityOnNextFocus } from '../../core/a11y.js';\nimport { SbbOpenCloseBaseElement } from '../../core/base-elements.js';\nimport { SbbConnectedAbortController, SbbLanguageController } from '../../core/controllers.js';\nimport { hostAttributes } from '../../core/decorators.js';\nimport { findReferencedElement, SbbScrollHandler } from '../../core/dom.js';\nimport { i18nCloseNavigation } from '../../core/i18n.js';\nimport { SbbUpdateSchedulerMixin } from '../../core/mixins.js';\nimport { AgnosticMutationObserver, AgnosticResizeObserver } from '../../core/observers.js';\nimport {\n  applyInertMechanism,\n  isEventOnElement,\n  removeAriaOverlayTriggerAttributes,\n  removeInertMechanism,\n  setAriaOverlayTriggerAttributes,\n} from '../../core/overlay.js';\nimport type { SbbNavigationButtonElement } from '../navigation-button.js';\nimport type { SbbNavigationLinkElement } from '../navigation-link.js';\n\nimport style from './navigation.scss?lit&inline';\n\nimport '../../button/transparent-button.js';\n\n/** Configuration for the attribute to look at if a navigation section is displayed */\nconst navigationObserverConfig: MutationObserverInit = {\n  subtree: true,\n  attributeFilter: ['data-state'],\n};\n\nlet nextId = 0;\nconst DEBOUNCE_TIME = 150;\n\n/**\n * It displays a navigation menu, wrapping one or more `sbb-navigation-*` components.\n *\n * @slot - Use the unnamed slot to add `sbb-navigation-button`/`sbb-navigation-link` elements into the sbb-navigation menu.\n * @event {CustomEvent<void>} willOpen - Emits whenever the `sbb-navigation` begins the opening transition. Can be canceled.\n * @event {CustomEvent<void>} didOpen - Emits whenever the `sbb-navigation` is opened.\n * @event {CustomEvent<void>} willClose - Emits whenever the `sbb-navigation` begins the closing transition. Can be canceled.\n * @event {CustomEvent<void>} didClose - Emits whenever the `sbb-navigation` is closed.\n * @cssprop [--sbb-navigation-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 */\n@customElement('sbb-navigation')\n@hostAttributes({\n  role: 'navigation',\n})\nexport class SbbNavigationElement extends SbbUpdateSchedulerMixin(SbbOpenCloseBaseElement) {\n  public static override styles: CSSResultGroup = style;\n\n  /**\n   * The element that will trigger the navigation.\n   * Accepts both a string (id of an element) or an HTML element.\n   */\n  @property()\n  public set trigger(value: string | HTMLElement | null) {\n    const oldValue = this._trigger;\n    this._trigger = value;\n    this._removeTriggerClickListener(this._trigger, oldValue);\n  }\n  public get trigger(): string | HTMLElement | null {\n    return this._trigger;\n  }\n  private _trigger: string | HTMLElement | null = null;\n\n  /**\n   * This will be forwarded as aria-label to the close button element.\n   */\n  @property({ attribute: 'accessibility-close-label' }) public accessibilityCloseLabel:\n    | string\n    | undefined;\n\n  /**\n   * Whether a navigation section is displayed.\n   */\n  @state() private _activeNavigationSection: HTMLElement | null = null;\n\n  public get activeNavigationSection(): HTMLElement | null {\n    return this._activeNavigationSection;\n  }\n\n  private _navigation!: HTMLDivElement;\n  private _navigationContentElement!: HTMLElement;\n  private _triggerElement: HTMLElement | null = null;\n  private _navigationController!: AbortController;\n  private _windowEventsController!: AbortController;\n  private _abort = new SbbConnectedAbortController(this);\n  private _language = new SbbLanguageController(this);\n  private _focusHandler = new SbbFocusHandler();\n  private _scrollHandler = new SbbScrollHandler();\n  private _isPointerDownEventOnNavigation: boolean = false;\n  private _resizeObserverTimeout: ReturnType<typeof setTimeout> | null = null;\n  private _navigationObserver = new AgnosticMutationObserver((mutationsList: MutationRecord[]) =>\n    this._onNavigationSectionChange(mutationsList),\n  );\n  private _navigationResizeObserver = new AgnosticResizeObserver(() => this._onNavigationResize());\n\n  /**\n   * Opens the navigation.\n   */\n  public open(): void {\n    if (this.state !== 'closed' || !this._navigation) {\n      return;\n    }\n\n    if (!this.willOpen.emit()) {\n      return;\n    }\n    this.state = 'opening';\n    this._checkActiveActions();\n    this._checkActiveSection();\n    this.startUpdate();\n\n    // Disable scrolling for content below the navigation\n    this._scrollHandler.disableScroll();\n    this._triggerElement?.setAttribute('aria-expanded', 'true');\n  }\n\n  private _checkActiveSection(): void {\n    const activeAction = this.querySelector(\n      'sbb-navigation-button.sbb-active',\n    ) as SbbNavigationButtonElement;\n    activeAction?.connectedSection?.open();\n  }\n\n  private _checkActiveActions(): void {\n    const activeActions = Array.from(\n      this.querySelectorAll(':is(sbb-navigation-button, sbb-navigation-link).sbb-active'),\n    ) as (SbbNavigationButtonElement | SbbNavigationLinkElement)[];\n    activeActions?.forEach((action) => action.marker?.select(action));\n  }\n\n  /**\n   * Closes the navigation.\n   */\n  public close(): void {\n    if (this.state !== 'opened') {\n      return;\n    }\n\n    if (!this.willClose.emit()) {\n      return;\n    }\n    this.state = 'closing';\n    this.startUpdate();\n    this._triggerElement?.setAttribute('aria-expanded', 'false');\n  }\n\n  // Removes trigger click listener on trigger change.\n  private _removeTriggerClickListener(\n    newValue: string | HTMLElement | null,\n    oldValue: string | HTMLElement | null,\n  ): void {\n    if (newValue !== oldValue) {\n      this._navigationController?.abort();\n      this._windowEventsController?.abort();\n      this._configure(this.trigger);\n    }\n  }\n\n  // Check if the trigger is valid and attach click event listeners.\n  private _configure(trigger: string | HTMLElement | null): void {\n    removeAriaOverlayTriggerAttributes(this._triggerElement);\n\n    if (!trigger) {\n      return;\n    }\n\n    this._triggerElement = findReferencedElement(trigger);\n\n    if (!this._triggerElement) {\n      return;\n    }\n\n    setAriaOverlayTriggerAttributes(this._triggerElement, 'menu', this.id, this.state);\n    this._navigationController?.abort();\n    this._navigationController = new AbortController();\n    this._triggerElement.addEventListener('click', () => this.open(), {\n      signal: this._navigationController.signal,\n    });\n  }\n\n  private _trapFocusFilter = (el: HTMLElement): boolean => {\n    return el.nodeName !== 'SBB-NAVIGATION-SECTION' || el.getAttribute('data-state') === 'opened';\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 _onAnimationEnd(event: AnimationEvent): void {\n    if (event.animationName === 'open' && this.state === 'opening') {\n      this.state = 'opened';\n      this.didOpen.emit();\n      this._navigationResizeObserver.observe(this);\n      applyInertMechanism(this);\n      this._focusHandler.trap(this, { filter: this._trapFocusFilter });\n      this._attachWindowEvents();\n      this._setNavigationFocus();\n    } else if (event.animationName === 'close' && this.state === 'closing') {\n      this.state = 'closed';\n      this._navigationContentElement.scrollTo(0, 0);\n      setModalityOnNextFocus(this._triggerElement);\n      removeInertMechanism();\n      // To enable focusing other element than the trigger, we need to call focus() a second time.\n      this._triggerElement?.focus();\n      this.didClose.emit();\n      this._navigationResizeObserver.unobserve(this);\n      this._resetMarkers();\n      this._windowEventsController?.abort();\n      this._focusHandler.disconnect();\n\n      // Enable scrolling for content below the navigation\n      this._scrollHandler.enableScroll();\n    }\n    this.completeUpdate();\n  }\n\n  private _resetMarkers(): void {\n    const activeActions = Array.from(\n      this.querySelectorAll(\n        ':is(sbb-navigation-button, sbb-navigation-link)[data-action-active]:not(.sbb-active)',\n      ),\n    ) as (SbbNavigationButtonElement | SbbNavigationLinkElement)[];\n    activeActions?.forEach((action) => action.marker?.reset());\n  }\n\n  private _attachWindowEvents(): void {\n    this._windowEventsController = new AbortController();\n    window.addEventListener('keydown', (event: KeyboardEvent) => this._onKeydownEvent(event), {\n      signal: this._windowEventsController.signal,\n    });\n  }\n\n  private _handleNavigationClose(event: Event): void {\n    const composedPathElements = event\n      .composedPath()\n      .filter((el) => el instanceof window.HTMLElement);\n    if (composedPathElements.some((el) => this._isCloseElement(el as HTMLElement))) {\n      this.close();\n    }\n  }\n\n  private _isCloseElement(element: HTMLElement): boolean {\n    return (\n      element.nodeName === 'A' ||\n      (element.hasAttribute('sbb-navigation-close') && !element.hasAttribute('disabled'))\n    );\n  }\n\n  // Closes the navigation on \"Esc\" key pressed.\n  private _onKeydownEvent(event: KeyboardEvent): void {\n    if (this.state === 'opened' && event.key === 'Escape') {\n      this.close();\n    }\n  }\n\n  // Set focus on the first focusable element.\n  private _setNavigationFocus(): void {\n    const closeButton = this.shadowRoot!.querySelector(\n      '#sbb-navigation-close-button',\n    ) as HTMLElement;\n    setModalityOnNextFocus(closeButton);\n    closeButton.focus();\n  }\n\n  // Check if the pointerdown event target is triggered on the navigation.\n  private _pointerDownListener = (event: PointerEvent): void => {\n    this._isPointerDownEventOnNavigation =\n      isEventOnElement(this._navigation, event) ||\n      isEventOnElement(\n        this.querySelector(\n          'sbb-navigation-section[data-state=\"opened\"]',\n        )?.shadowRoot?.querySelector('nav.sbb-navigation-section') as HTMLElement,\n        event,\n      );\n  };\n\n  // Close navigation on backdrop click.\n  private _closeOnBackdropClick = (event: PointerEvent): void => {\n    if (!this._isPointerDownEventOnNavigation && !isEventOnElement(this._navigation, event)) {\n      this.close();\n    }\n  };\n\n  // Observe changes on navigation section data-state.\n  private _onNavigationSectionChange(mutationsList: MutationRecord[]): void {\n    for (const mutation of mutationsList) {\n      if ((mutation.target as HTMLElement).nodeName === 'SBB-NAVIGATION-SECTION') {\n        this._activeNavigationSection = this.querySelector(\n          'sbb-navigation-section[data-state=\"opening\"], sbb-navigation-section[data-state=\"opened\"]',\n        );\n        this.toggleAttribute('data-has-navigation-section', !!this._activeNavigationSection);\n      }\n    }\n  }\n\n  private _onNavigationResize(): void {\n    if (this.state !== 'opened') {\n      return;\n    }\n\n    if (this._resizeObserverTimeout) {\n      clearTimeout(this._resizeObserverTimeout);\n    }\n\n    this.toggleAttribute('data-resize-disable-animation', true);\n\n    // Disable the animation when resizing the navigation to avoid strange height transition effects.\n    this._resizeObserverTimeout = setTimeout(\n      () => this.removeAttribute('data-resize-disable-animation'),\n      DEBOUNCE_TIME,\n    );\n  }\n\n  public override connectedCallback(): void {\n    super.connectedCallback();\n    this.id ||= `sbb-navigation-${nextId++}`;\n    const signal = this._abort.signal;\n    this.addEventListener('click', (e) => this._handleNavigationClose(e), { signal });\n    // Validate trigger element and attach event listeners\n    this._configure(this.trigger);\n    this._navigationObserver.observe(this, navigationObserverConfig);\n    this.addEventListener('pointerup', (event) => this._closeOnBackdropClick(event), { signal });\n    this.addEventListener('pointerdown', (event) => this._pointerDownListener(event), { signal });\n\n    if (this.state === 'opened') {\n      applyInertMechanism(this);\n    }\n  }\n\n  public override disconnectedCallback(): void {\n    super.disconnectedCallback();\n    this._navigationController?.abort();\n    this._windowEventsController?.abort();\n    this._focusHandler.disconnect();\n    this._navigationObserver.disconnect();\n    this._navigationResizeObserver.disconnect();\n    removeInertMechanism();\n  }\n\n  protected override render(): TemplateResult {\n    const closeButton = html`\n      <sbb-transparent-button\n        id=\"sbb-navigation-close-button\"\n        class=\"sbb-navigation__close\"\n        aria-label=${this.accessibilityCloseLabel || i18nCloseNavigation[this._language.current]}\n        aria-controls=\"sbb-navigation-overlay\"\n        negative\n        size=\"m\"\n        type=\"button\"\n        icon-name=\"cross-small\"\n        sbb-navigation-close\n      ></sbb-transparent-button>\n    `;\n\n    return html`\n      <div class=\"sbb-navigation__container\">\n        <div\n          id=\"sbb-navigation-overlay\"\n          @animationend=${(event: AnimationEvent) => this._onAnimationEnd(event)}\n          class=\"sbb-navigation\"\n          ${ref((navigationRef?: Element) => (this._navigation = navigationRef as HTMLDivElement))}\n        >\n          <div class=\"sbb-navigation__header\">${closeButton}</div>\n          <div class=\"sbb-navigation__wrapper\">\n            <div\n              class=\"sbb-navigation__content\"\n              ${ref((el?: Element) => (this._navigationContentElement = el as HTMLElement))}\n            >\n              <slot></slot>\n            </div>\n          </div>\n        </div>\n        <slot name=\"navigation-section\"></slot>\n      </div>\n    `;\n  }\n}\n\ndeclare global {\n  interface HTMLElementTagNameMap {\n    // eslint-disable-next-line @typescript-eslint/naming-convention\n    'sbb-navigation': SbbNavigationElement;\n  }\n}\n"],"names":[],"mappings":"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"}
604
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"navigation.js","sources":["../../../../src/elements/navigation/navigation/navigation.ts"],"sourcesContent":["import type { CSSResultGroup, TemplateResult } from 'lit';\nimport { html } from 'lit';\nimport { customElement, property, state } from 'lit/decorators.js';\nimport { ref } from 'lit/directives/ref.js';\n\nimport { SbbFocusHandler, setModalityOnNextFocus } from '../../core/a11y.js';\nimport { SbbOpenCloseBaseElement } from '../../core/base-elements.js';\nimport { SbbConnectedAbortController, SbbLanguageController } from '../../core/controllers.js';\nimport { hostAttributes } from '../../core/decorators.js';\nimport { findReferencedElement, SbbScrollHandler } from '../../core/dom.js';\nimport { i18nCloseNavigation } from '../../core/i18n.js';\nimport { SbbUpdateSchedulerMixin } from '../../core/mixins.js';\nimport { AgnosticMutationObserver, AgnosticResizeObserver } from '../../core/observers.js';\nimport {\n  applyInertMechanism,\n  isEventOnElement,\n  removeAriaOverlayTriggerAttributes,\n  removeInertMechanism,\n  setAriaOverlayTriggerAttributes,\n} from '../../core/overlay.js';\nimport type { SbbNavigationButtonElement } from '../navigation-button.js';\nimport type { SbbNavigationLinkElement } from '../navigation-link.js';\n\nimport style from './navigation.scss?lit&inline';\n\nimport '../../button/transparent-button.js';\n\n/** Configuration for the attribute to look at if a navigation section is displayed */\nconst navigationObserverConfig: MutationObserverInit = {\n  subtree: true,\n  attributeFilter: ['data-state'],\n};\n\nlet nextId = 0;\nconst DEBOUNCE_TIME = 150;\n\n/**\n * It displays a navigation menu, wrapping one or more `sbb-navigation-*` components.\n *\n * @slot - Use the unnamed slot to add `sbb-navigation-button`/`sbb-navigation-link` elements into the sbb-navigation menu.\n * @event {CustomEvent<void>} willOpen - Emits whenever the `sbb-navigation` begins the opening transition. Can be canceled.\n * @event {CustomEvent<void>} didOpen - Emits whenever the `sbb-navigation` is opened.\n * @event {CustomEvent<void>} willClose - Emits whenever the `sbb-navigation` begins the closing transition. Can be canceled.\n * @event {CustomEvent<void>} didClose - Emits whenever the `sbb-navigation` is closed.\n * @cssprop [--sbb-navigation-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 */\n@customElement('sbb-navigation')\n@hostAttributes({\n  role: 'navigation',\n})\nexport class SbbNavigationElement extends SbbUpdateSchedulerMixin(SbbOpenCloseBaseElement) {\n  public static override styles: CSSResultGroup = style;\n\n  /**\n   * The element that will trigger the navigation.\n   * Accepts both a string (id of an element) or an HTML element.\n   */\n  @property()\n  public set trigger(value: string | HTMLElement | null) {\n    const oldValue = this._trigger;\n    this._trigger = value;\n    this._removeTriggerClickListener(this._trigger, oldValue);\n  }\n  public get trigger(): string | HTMLElement | null {\n    return this._trigger;\n  }\n  private _trigger: string | HTMLElement | null = null;\n\n  /**\n   * This will be forwarded as aria-label to the close button element.\n   */\n  @property({ attribute: 'accessibility-close-label' }) public accessibilityCloseLabel:\n    | string\n    | undefined;\n\n  /**\n   * Whether a navigation section is displayed.\n   */\n  @state() private _activeNavigationSection: HTMLElement | null = null;\n\n  public get activeNavigationSection(): HTMLElement | null {\n    return this._activeNavigationSection;\n  }\n\n  private _navigation!: HTMLDivElement;\n  private _navigationContentElement!: HTMLElement;\n  private _triggerElement: HTMLElement | null = null;\n  private _navigationController!: AbortController;\n  private _windowEventsController!: AbortController;\n  private _abort = new SbbConnectedAbortController(this);\n  private _language = new SbbLanguageController(this);\n  private _focusHandler = new SbbFocusHandler();\n  private _scrollHandler = new SbbScrollHandler();\n  private _isPointerDownEventOnNavigation: boolean = false;\n  private _resizeObserverTimeout: ReturnType<typeof setTimeout> | null = null;\n  private _navigationObserver = new AgnosticMutationObserver((mutationsList: MutationRecord[]) =>\n    this._onNavigationSectionChange(mutationsList),\n  );\n  private _navigationResizeObserver = new AgnosticResizeObserver(() => this._onNavigationResize());\n\n  /**\n   * Opens the navigation.\n   */\n  public open(): void {\n    if (this.state !== 'closed' || !this._navigation) {\n      return;\n    }\n\n    if (!this.willOpen.emit()) {\n      return;\n    }\n    this.state = 'opening';\n    this._checkActiveActions();\n    this._checkActiveSection();\n    this.startUpdate();\n\n    // Disable scrolling for content below the navigation\n    this._scrollHandler.disableScroll();\n    this._triggerElement?.setAttribute('aria-expanded', 'true');\n  }\n\n  private _checkActiveSection(): void {\n    const activeAction = this.querySelector(\n      'sbb-navigation-button.sbb-active',\n    ) as SbbNavigationButtonElement;\n    activeAction?.connectedSection?.open();\n  }\n\n  private _checkActiveActions(): void {\n    const activeActions = Array.from(\n      this.querySelectorAll(':is(sbb-navigation-button, sbb-navigation-link).sbb-active'),\n    ) as (SbbNavigationButtonElement | SbbNavigationLinkElement)[];\n    activeActions?.forEach((action) => action.marker?.select(action));\n  }\n\n  /**\n   * Closes the navigation.\n   */\n  public close(): void {\n    if (this.state !== 'opened') {\n      return;\n    }\n\n    if (!this.willClose.emit()) {\n      return;\n    }\n    this.state = 'closing';\n    this.startUpdate();\n    this._triggerElement?.setAttribute('aria-expanded', 'false');\n  }\n\n  // Removes trigger click listener on trigger change.\n  private _removeTriggerClickListener(\n    newValue: string | HTMLElement | null,\n    oldValue: string | HTMLElement | null,\n  ): void {\n    if (newValue !== oldValue) {\n      this._navigationController?.abort();\n      this._windowEventsController?.abort();\n      this._configure(this.trigger);\n    }\n  }\n\n  // Check if the trigger is valid and attach click event listeners.\n  private _configure(trigger: string | HTMLElement | null): void {\n    removeAriaOverlayTriggerAttributes(this._triggerElement);\n\n    if (!trigger) {\n      return;\n    }\n\n    this._triggerElement = findReferencedElement(trigger);\n\n    if (!this._triggerElement) {\n      return;\n    }\n\n    setAriaOverlayTriggerAttributes(this._triggerElement, 'menu', this.id, this.state);\n    this._navigationController?.abort();\n    this._navigationController = new AbortController();\n    this._triggerElement.addEventListener('click', () => this.open(), {\n      signal: this._navigationController.signal,\n    });\n  }\n\n  private _trapFocusFilter = (el: HTMLElement): boolean => {\n    return el.nodeName !== 'SBB-NAVIGATION-SECTION' || el.getAttribute('data-state') === 'opened';\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 _onAnimationEnd(event: AnimationEvent): void {\n    if (event.animationName === 'open' && this.state === 'opening') {\n      this.state = 'opened';\n      this.didOpen.emit();\n      this._navigationResizeObserver.observe(this);\n      applyInertMechanism(this);\n      this._focusHandler.trap(this, { filter: this._trapFocusFilter });\n      this._attachWindowEvents();\n      this._setNavigationFocus();\n    } else if (event.animationName === 'close' && this.state === 'closing') {\n      this.state = 'closed';\n      this._navigationContentElement.scrollTo(0, 0);\n      setModalityOnNextFocus(this._triggerElement);\n      removeInertMechanism();\n      // To enable focusing other element than the trigger, we need to call focus() a second time.\n      this._triggerElement?.focus();\n      this.didClose.emit();\n      this._navigationResizeObserver.unobserve(this);\n      this._resetMarkers();\n      this._windowEventsController?.abort();\n      this._focusHandler.disconnect();\n\n      // Enable scrolling for content below the navigation\n      this._scrollHandler.enableScroll();\n    }\n    this.completeUpdate();\n  }\n\n  private _resetMarkers(): void {\n    const activeActions = Array.from(\n      this.querySelectorAll(\n        ':is(sbb-navigation-button, sbb-navigation-link)[data-action-active]:not(.sbb-active)',\n      ),\n    ) as (SbbNavigationButtonElement | SbbNavigationLinkElement)[];\n    activeActions?.forEach((action) => action.marker?.reset());\n  }\n\n  private _attachWindowEvents(): void {\n    this._windowEventsController = new AbortController();\n    window.addEventListener('keydown', (event: KeyboardEvent) => this._onKeydownEvent(event), {\n      signal: this._windowEventsController.signal,\n    });\n  }\n\n  private _handleNavigationClose(event: Event): void {\n    const composedPathElements = event\n      .composedPath()\n      .filter((el) => el instanceof window.HTMLElement);\n    if (composedPathElements.some((el) => this._isCloseElement(el as HTMLElement))) {\n      this.close();\n    }\n  }\n\n  private _isCloseElement(element: HTMLElement): boolean {\n    return (\n      element.nodeName === 'A' ||\n      (element.hasAttribute('sbb-navigation-close') && !element.hasAttribute('disabled'))\n    );\n  }\n\n  // Closes the navigation on \"Esc\" key pressed.\n  private _onKeydownEvent(event: KeyboardEvent): void {\n    if (this.state === 'opened' && event.key === 'Escape') {\n      this.close();\n    }\n  }\n\n  // Set focus on the first focusable element.\n  private _setNavigationFocus(): void {\n    const closeButton = this.shadowRoot!.querySelector(\n      '#sbb-navigation-close-button',\n    ) as HTMLElement;\n    setModalityOnNextFocus(closeButton);\n    closeButton.focus();\n  }\n\n  // Check if the pointerdown event target is triggered on the navigation.\n  private _pointerDownListener = (event: PointerEvent): void => {\n    this._isPointerDownEventOnNavigation =\n      isEventOnElement(this._navigation, event) ||\n      isEventOnElement(\n        this.querySelector(\n          'sbb-navigation-section[data-state=\"opened\"]',\n        )?.shadowRoot?.querySelector('nav.sbb-navigation-section') as HTMLElement,\n        event,\n      );\n  };\n\n  // Close navigation on backdrop click.\n  private _closeOnBackdropClick = (event: PointerEvent): void => {\n    if (!this._isPointerDownEventOnNavigation && !isEventOnElement(this._navigation, event)) {\n      this.close();\n    }\n  };\n\n  // Observe changes on navigation section data-state.\n  private _onNavigationSectionChange(mutationsList: MutationRecord[]): void {\n    for (const mutation of mutationsList) {\n      if ((mutation.target as HTMLElement).nodeName === 'SBB-NAVIGATION-SECTION') {\n        this._activeNavigationSection = this.querySelector(\n          'sbb-navigation-section[data-state=\"opening\"], sbb-navigation-section[data-state=\"opened\"]',\n        );\n        this.toggleAttribute('data-has-navigation-section', !!this._activeNavigationSection);\n      }\n    }\n  }\n\n  private _onNavigationResize(): void {\n    if (this.state !== 'opened') {\n      return;\n    }\n\n    if (this._resizeObserverTimeout) {\n      clearTimeout(this._resizeObserverTimeout);\n    }\n\n    this.toggleAttribute('data-resize-disable-animation', true);\n\n    // Disable the animation when resizing the navigation to avoid strange height transition effects.\n    this._resizeObserverTimeout = setTimeout(\n      () => this.removeAttribute('data-resize-disable-animation'),\n      DEBOUNCE_TIME,\n    );\n  }\n\n  public override connectedCallback(): void {\n    super.connectedCallback();\n    this.id ||= `sbb-navigation-${nextId++}`;\n    const signal = this._abort.signal;\n    this.addEventListener('click', (e) => this._handleNavigationClose(e), { signal });\n    // Validate trigger element and attach event listeners\n    this._configure(this.trigger);\n    this._navigationObserver.observe(this, navigationObserverConfig);\n    this.addEventListener('pointerup', (event) => this._closeOnBackdropClick(event), { signal });\n    this.addEventListener('pointerdown', (event) => this._pointerDownListener(event), { signal });\n\n    if (this.state === 'opened') {\n      applyInertMechanism(this);\n    }\n  }\n\n  public override disconnectedCallback(): void {\n    super.disconnectedCallback();\n    this._navigationController?.abort();\n    this._windowEventsController?.abort();\n    this._focusHandler.disconnect();\n    this._navigationObserver.disconnect();\n    this._navigationResizeObserver.disconnect();\n    removeInertMechanism();\n    this._scrollHandler.enableScroll();\n  }\n\n  protected override render(): TemplateResult {\n    const closeButton = html`\n      <sbb-transparent-button\n        id=\"sbb-navigation-close-button\"\n        class=\"sbb-navigation__close\"\n        aria-label=${this.accessibilityCloseLabel || i18nCloseNavigation[this._language.current]}\n        aria-controls=\"sbb-navigation-overlay\"\n        negative\n        size=\"m\"\n        type=\"button\"\n        icon-name=\"cross-small\"\n        sbb-navigation-close\n      ></sbb-transparent-button>\n    `;\n\n    return html`\n      <div class=\"sbb-navigation__container\">\n        <div\n          id=\"sbb-navigation-overlay\"\n          @animationend=${(event: AnimationEvent) => this._onAnimationEnd(event)}\n          class=\"sbb-navigation\"\n          ${ref((navigationRef?: Element) => (this._navigation = navigationRef as HTMLDivElement))}\n        >\n          <div class=\"sbb-navigation__header\">${closeButton}</div>\n          <div class=\"sbb-navigation__wrapper\">\n            <div\n              class=\"sbb-navigation__content\"\n              ${ref((el?: Element) => (this._navigationContentElement = el as HTMLElement))}\n            >\n              <slot></slot>\n            </div>\n          </div>\n        </div>\n        <slot name=\"navigation-section\"></slot>\n      </div>\n    `;\n  }\n}\n\ndeclare global {\n  interface HTMLElementTagNameMap {\n    // eslint-disable-next-line @typescript-eslint/naming-convention\n    'sbb-navigation': SbbNavigationElement;\n  }\n}\n"],"names":[],"mappings":"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"}
@@ -6,10 +6,10 @@ import { SbbTitleLevel } from '../title.js';
6
6
  *
7
7
  * @slot - Use the unnamed slot to add content to the notification message.
8
8
  * @slot title - Use this to provide a notification title (optional).
9
- * @event {CustomEvent<void>} willOpen - Emits whenever the `sbb-notification` starts the opening transition.
10
- * @event {CustomEvent<void>} didOpen - Emits whenever the `sbb-notification` is opened.
11
- * @event {CustomEvent<void>} willClose - Emits whenever the `sbb-notification` begins the closing transition.
12
- * @event {CustomEvent<void>} didClose - Emits whenever the `sbb-notification` is closed.
9
+ * @event {CustomEvent<void>} willOpen - Emits when the opening animation starts.
10
+ * @event {CustomEvent<void>} didOpen - Emits when the opening animation ends.
11
+ * @event {CustomEvent<void>} willClose - Emits when the closing animation starts.
12
+ * @event {CustomEvent<void>} didClose - Emits when the closing animation ends.
13
13
  * @cssprop [--sbb-notification-margin=0] - Can be used to modify the margin in order to get a smoother animation.
14
14
  * See style section for more information.
15
15
  */
@@ -474,4 +474,4 @@ SbbNotificationElement = __decorateClass([
474
474
  export {
475
475
  SbbNotificationElement
476
476
  };
477
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"notification.js","sources":["../../../src/elements/notification/notification.ts"],"sourcesContent":["import type { CSSResultGroup, PropertyValues, TemplateResult } from 'lit';\nimport { html, LitElement, nothing } from 'lit';\nimport { customElement, property } from 'lit/decorators.js';\n\nimport { SbbLanguageController } from '../core/controllers.js';\nimport { slotState } from '../core/decorators.js';\nimport { EventEmitter } from '../core/eventing.js';\nimport { i18nCloseNotification } from '../core/i18n.js';\nimport type { SbbOpenedClosedState } from '../core/interfaces.js';\nimport { AgnosticResizeObserver } from '../core/observers.js';\nimport type { SbbTitleLevel } from '../title.js';\n\nimport style from './notification.scss?lit&inline';\n\nimport '../button/secondary-button.js';\nimport '../divider.js';\nimport '../icon.js';\nimport '../title.js';\n\nconst notificationTypes = new Map([\n  ['info', 'circle-information-small'],\n  ['success', 'circle-tick-small'],\n  ['warn', 'circle-exclamation-point-small'],\n  ['error', 'circle-cross-small'],\n]);\n\nconst DEBOUNCE_TIME = 150;\n\n/**\n * It displays messages which require a user's attention without interrupting its tasks.\n *\n * @slot - Use the unnamed slot to add content to the notification message.\n * @slot title - Use this to provide a notification title (optional).\n * @event {CustomEvent<void>} willOpen - Emits whenever the `sbb-notification` starts the opening transition.\n * @event {CustomEvent<void>} didOpen - Emits whenever the `sbb-notification` is opened.\n * @event {CustomEvent<void>} willClose - Emits whenever the `sbb-notification` begins the closing transition.\n * @event {CustomEvent<void>} didClose - Emits whenever the `sbb-notification` is closed.\n * @cssprop [--sbb-notification-margin=0] - Can be used to modify the margin in order to get a smoother animation.\n * See style section for more information.\n */\n@customElement('sbb-notification')\n@slotState()\nexport class SbbNotificationElement extends LitElement {\n  // FIXME inheriting from SbbOpenCloseBaseElement requires: https://github.com/open-wc/custom-elements-manifest/issues/253\n  public static override styles: CSSResultGroup = style;\n  public static readonly events = {\n    willOpen: 'willOpen',\n    didOpen: 'didOpen',\n    willClose: 'willClose',\n    didClose: 'didClose',\n  } as const;\n\n  /** The type of the notification. */\n  @property({ reflect: true }) public type: 'info' | 'success' | 'warn' | 'error' = 'info';\n\n  /** Content of title. */\n  @property({ attribute: 'title-content', reflect: true }) public titleContent?: string;\n\n  /** Level of title, it will be rendered as heading tag (e.g. h3). Defaults to level 3. */\n  @property({ attribute: 'title-level' }) public titleLevel: SbbTitleLevel = '3';\n\n  /**\n   * Whether the notification is readonly.\n   * In readonly mode, there is no dismiss button offered to the user.\n   */\n  @property({ reflect: true, type: Boolean }) public readonly = false;\n\n  /** Size variant, either s or m. */\n  @property({ reflect: true }) public size: 'm' | 's' = 'm';\n\n  /** The enabled animations. */\n  @property({ reflect: true }) public animation: 'open' | 'close' | 'all' | 'none' = 'all';\n\n  /** The state of the notification. */\n  private set _state(state: SbbOpenedClosedState) {\n    this.setAttribute('data-state', state);\n  }\n  private get _state(): SbbOpenedClosedState {\n    return this.getAttribute('data-state') as SbbOpenedClosedState;\n  }\n\n  private _notificationElement!: HTMLElement;\n  private _resizeObserverTimeout: ReturnType<typeof setTimeout> | null = null;\n  private _language = new SbbLanguageController(this);\n  private _notificationResizeObserver = new AgnosticResizeObserver(() =>\n    this._onNotificationResize(),\n  );\n\n  /** Emits whenever the `sbb-notification` starts the opening transition. */\n  private _willOpen: EventEmitter<void> = new EventEmitter(\n    this,\n    SbbNotificationElement.events.willOpen,\n  );\n\n  /** Emits whenever the `sbb-notification` is opened. */\n  private _didOpen: EventEmitter<void> = new EventEmitter(\n    this,\n    SbbNotificationElement.events.didOpen,\n  );\n\n  /** Emits whenever the `sbb-notification` begins the closing transition. */\n  private _willClose: EventEmitter<void> = new EventEmitter(\n    this,\n    SbbNotificationElement.events.willClose,\n  );\n\n  /** Emits whenever the `sbb-notification` is closed. */\n  private _didClose: EventEmitter<void> = new EventEmitter(\n    this,\n    SbbNotificationElement.events.didClose,\n  );\n\n  private _open(): void {\n    if (this._state === 'closed') {\n      this._state = 'opening';\n      this._willOpen.emit();\n    }\n  }\n\n  public close(): void {\n    if (this._state === 'opened') {\n      this._state = 'closing';\n      this._willClose.emit();\n    }\n  }\n\n  public override connectedCallback(): void {\n    this._state ||= 'closed';\n\n    super.connectedCallback();\n  }\n\n  protected override async firstUpdated(changedProperties: PropertyValues<this>): Promise<void> {\n    super.firstUpdated(changedProperties);\n\n    this._notificationElement = this.shadowRoot?.querySelector(\n      '.sbb-notification__wrapper',\n    ) as HTMLElement;\n    // We need to wait for the component's `updateComplete` in order to set the correct\n    // height to the notification element.\n    await this.updateComplete;\n    this._setNotificationHeight();\n    this._open();\n  }\n\n  public override disconnectedCallback(): void {\n    super.disconnectedCallback();\n    this._notificationResizeObserver.disconnect();\n  }\n\n  private _setNotificationHeight(): void {\n    if (!this._notificationElement?.scrollHeight) {\n      return;\n    }\n    const notificationHeight = `${this._notificationElement.scrollHeight}px`;\n    this.style.setProperty('--sbb-notification-height', notificationHeight);\n  }\n\n  private _onNotificationResize(): void {\n    if (this._state !== 'opened') {\n      return;\n    }\n\n    if (this._resizeObserverTimeout) {\n      clearTimeout(this._resizeObserverTimeout);\n    }\n\n    // Disable the animation when resizing the notification to avoid strange height transition effects.\n    this.toggleAttribute('data-resize-disable-animation', true);\n    this._resizeObserverTimeout = setTimeout(\n      () => this.removeAttribute('data-resize-disable-animation'),\n      DEBOUNCE_TIME,\n    );\n\n    // To avoid ResizeObserver loops, we set the height a tick later.\n    setTimeout(() => this._setNotificationHeight());\n  }\n\n  private _onNotificationAnimationEnd(event: AnimationEvent): void {\n    if (this._state === 'opening' && event.animationName === 'open') {\n      this._handleOpening();\n    }\n\n    if (this._state === 'closing' && event.animationName === 'close-height') {\n      this._handleClosing();\n    }\n  }\n\n  private _handleOpening(): void {\n    this._state = 'opened';\n    this._didOpen.emit();\n    this._notificationResizeObserver.observe(this._notificationElement);\n  }\n\n  private _handleClosing(): void {\n    this._state = 'closed';\n    this._didClose.emit();\n    this._notificationResizeObserver.unobserve(this._notificationElement);\n    setTimeout(() => this.remove());\n  }\n\n  protected override render(): TemplateResult {\n    return html`\n      <div\n        class=\"sbb-notification__wrapper\"\n        @animationend=${(event: AnimationEvent) => this._onNotificationAnimationEnd(event)}\n      >\n        <div class=\"sbb-notification\">\n          <sbb-icon\n            class=\"sbb-notification__icon\"\n            name=${notificationTypes.get(this.type)!}\n          ></sbb-icon>\n\n          <span class=\"sbb-notification__content\">\n            <sbb-title\n              class=\"sbb-notification__title\"\n              level=${this.titleLevel}\n              visual-level=${this.size === 'm' ? '5' : '6'}\n            >\n              <slot name=\"title\">${this.titleContent}</slot>\n            </sbb-title>\n            <slot></slot>\n          </span>\n\n          ${!this.readonly\n            ? html`<span class=\"sbb-notification__close-wrapper\">\n                <sbb-divider class=\"sbb-notification__divider\" orientation=\"vertical\"></sbb-divider>\n                <sbb-secondary-button\n                  size=${this.size}\n                  icon-name=\"cross-small\"\n                  @click=${() => this.close()}\n                  aria-label=${i18nCloseNotification[this._language.current]}\n                  class=\"sbb-notification__close\"\n                ></sbb-secondary-button>\n              </span>`\n            : nothing}\n        </div>\n      </div>\n    `;\n  }\n}\n\ndeclare global {\n  interface HTMLElementTagNameMap {\n    // eslint-disable-next-line @typescript-eslint/naming-convention\n    'sbb-notification': SbbNotificationElement;\n  }\n}\n"],"names":[],"mappings":"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"}
477
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"notification.js","sources":["../../../src/elements/notification/notification.ts"],"sourcesContent":["import type { CSSResultGroup, PropertyValues, TemplateResult } from 'lit';\nimport { html, LitElement, nothing } from 'lit';\nimport { customElement, property } from 'lit/decorators.js';\n\nimport { SbbLanguageController } from '../core/controllers.js';\nimport { slotState } from '../core/decorators.js';\nimport { EventEmitter } from '../core/eventing.js';\nimport { i18nCloseNotification } from '../core/i18n.js';\nimport type { SbbOpenedClosedState } from '../core/interfaces.js';\nimport { AgnosticResizeObserver } from '../core/observers.js';\nimport type { SbbTitleLevel } from '../title.js';\n\nimport style from './notification.scss?lit&inline';\n\nimport '../button/secondary-button.js';\nimport '../divider.js';\nimport '../icon.js';\nimport '../title.js';\n\nconst notificationTypes = new Map([\n  ['info', 'circle-information-small'],\n  ['success', 'circle-tick-small'],\n  ['warn', 'circle-exclamation-point-small'],\n  ['error', 'circle-cross-small'],\n]);\n\nconst DEBOUNCE_TIME = 150;\n\n/**\n * It displays messages which require a user's attention without interrupting its tasks.\n *\n * @slot - Use the unnamed slot to add content to the notification message.\n * @slot title - Use this to provide a notification title (optional).\n * @event {CustomEvent<void>} willOpen - Emits when the opening animation starts.\n * @event {CustomEvent<void>} didOpen - Emits when the opening animation ends.\n * @event {CustomEvent<void>} willClose - Emits when the closing animation starts.\n * @event {CustomEvent<void>} didClose - Emits when the closing animation ends.\n * @cssprop [--sbb-notification-margin=0] - Can be used to modify the margin in order to get a smoother animation.\n * See style section for more information.\n */\n@customElement('sbb-notification')\n@slotState()\nexport class SbbNotificationElement extends LitElement {\n  // FIXME inheriting from SbbOpenCloseBaseElement requires: https://github.com/open-wc/custom-elements-manifest/issues/253\n  public static override styles: CSSResultGroup = style;\n  public static readonly events = {\n    willOpen: 'willOpen',\n    didOpen: 'didOpen',\n    willClose: 'willClose',\n    didClose: 'didClose',\n  } as const;\n\n  /** The type of the notification. */\n  @property({ reflect: true }) public type: 'info' | 'success' | 'warn' | 'error' = 'info';\n\n  /** Content of title. */\n  @property({ attribute: 'title-content', reflect: true }) public titleContent?: string;\n\n  /** Level of title, it will be rendered as heading tag (e.g. h3). Defaults to level 3. */\n  @property({ attribute: 'title-level' }) public titleLevel: SbbTitleLevel = '3';\n\n  /**\n   * Whether the notification is readonly.\n   * In readonly mode, there is no dismiss button offered to the user.\n   */\n  @property({ reflect: true, type: Boolean }) public readonly = false;\n\n  /** Size variant, either s or m. */\n  @property({ reflect: true }) public size: 'm' | 's' = 'm';\n\n  /** The enabled animations. */\n  @property({ reflect: true }) public animation: 'open' | 'close' | 'all' | 'none' = 'all';\n\n  /** The state of the notification. */\n  private set _state(state: SbbOpenedClosedState) {\n    this.setAttribute('data-state', state);\n  }\n  private get _state(): SbbOpenedClosedState {\n    return this.getAttribute('data-state') as SbbOpenedClosedState;\n  }\n\n  private _notificationElement!: HTMLElement;\n  private _resizeObserverTimeout: ReturnType<typeof setTimeout> | null = null;\n  private _language = new SbbLanguageController(this);\n  private _notificationResizeObserver = new AgnosticResizeObserver(() =>\n    this._onNotificationResize(),\n  );\n\n  /** Emits whenever the `sbb-notification` starts the opening transition. */\n  private _willOpen: EventEmitter<void> = new EventEmitter(\n    this,\n    SbbNotificationElement.events.willOpen,\n  );\n\n  /** Emits whenever the `sbb-notification` is opened. */\n  private _didOpen: EventEmitter<void> = new EventEmitter(\n    this,\n    SbbNotificationElement.events.didOpen,\n  );\n\n  /** Emits whenever the `sbb-notification` begins the closing transition. */\n  private _willClose: EventEmitter<void> = new EventEmitter(\n    this,\n    SbbNotificationElement.events.willClose,\n  );\n\n  /** Emits whenever the `sbb-notification` is closed. */\n  private _didClose: EventEmitter<void> = new EventEmitter(\n    this,\n    SbbNotificationElement.events.didClose,\n  );\n\n  private _open(): void {\n    if (this._state === 'closed') {\n      this._state = 'opening';\n      this._willOpen.emit();\n    }\n  }\n\n  public close(): void {\n    if (this._state === 'opened') {\n      this._state = 'closing';\n      this._willClose.emit();\n    }\n  }\n\n  public override connectedCallback(): void {\n    this._state ||= 'closed';\n\n    super.connectedCallback();\n  }\n\n  protected override async firstUpdated(changedProperties: PropertyValues<this>): Promise<void> {\n    super.firstUpdated(changedProperties);\n\n    this._notificationElement = this.shadowRoot?.querySelector(\n      '.sbb-notification__wrapper',\n    ) as HTMLElement;\n    // We need to wait for the component's `updateComplete` in order to set the correct\n    // height to the notification element.\n    await this.updateComplete;\n    this._setNotificationHeight();\n    this._open();\n  }\n\n  public override disconnectedCallback(): void {\n    super.disconnectedCallback();\n    this._notificationResizeObserver.disconnect();\n  }\n\n  private _setNotificationHeight(): void {\n    if (!this._notificationElement?.scrollHeight) {\n      return;\n    }\n    const notificationHeight = `${this._notificationElement.scrollHeight}px`;\n    this.style.setProperty('--sbb-notification-height', notificationHeight);\n  }\n\n  private _onNotificationResize(): void {\n    if (this._state !== 'opened') {\n      return;\n    }\n\n    if (this._resizeObserverTimeout) {\n      clearTimeout(this._resizeObserverTimeout);\n    }\n\n    // Disable the animation when resizing the notification to avoid strange height transition effects.\n    this.toggleAttribute('data-resize-disable-animation', true);\n    this._resizeObserverTimeout = setTimeout(\n      () => this.removeAttribute('data-resize-disable-animation'),\n      DEBOUNCE_TIME,\n    );\n\n    // To avoid ResizeObserver loops, we set the height a tick later.\n    setTimeout(() => this._setNotificationHeight());\n  }\n\n  private _onNotificationAnimationEnd(event: AnimationEvent): void {\n    if (this._state === 'opening' && event.animationName === 'open') {\n      this._handleOpening();\n    }\n\n    if (this._state === 'closing' && event.animationName === 'close-height') {\n      this._handleClosing();\n    }\n  }\n\n  private _handleOpening(): void {\n    this._state = 'opened';\n    this._didOpen.emit();\n    this._notificationResizeObserver.observe(this._notificationElement);\n  }\n\n  private _handleClosing(): void {\n    this._state = 'closed';\n    this._didClose.emit();\n    this._notificationResizeObserver.unobserve(this._notificationElement);\n    setTimeout(() => this.remove());\n  }\n\n  protected override render(): TemplateResult {\n    return html`\n      <div\n        class=\"sbb-notification__wrapper\"\n        @animationend=${(event: AnimationEvent) => this._onNotificationAnimationEnd(event)}\n      >\n        <div class=\"sbb-notification\">\n          <sbb-icon\n            class=\"sbb-notification__icon\"\n            name=${notificationTypes.get(this.type)!}\n          ></sbb-icon>\n\n          <span class=\"sbb-notification__content\">\n            <sbb-title\n              class=\"sbb-notification__title\"\n              level=${this.titleLevel}\n              visual-level=${this.size === 'm' ? '5' : '6'}\n            >\n              <slot name=\"title\">${this.titleContent}</slot>\n            </sbb-title>\n            <slot></slot>\n          </span>\n\n          ${!this.readonly\n            ? html`<span class=\"sbb-notification__close-wrapper\">\n                <sbb-divider class=\"sbb-notification__divider\" orientation=\"vertical\"></sbb-divider>\n                <sbb-secondary-button\n                  size=${this.size}\n                  icon-name=\"cross-small\"\n                  @click=${() => this.close()}\n                  aria-label=${i18nCloseNotification[this._language.current]}\n                  class=\"sbb-notification__close\"\n                ></sbb-secondary-button>\n              </span>`\n            : nothing}\n        </div>\n      </div>\n    `;\n  }\n}\n\ndeclare global {\n  interface HTMLElementTagNameMap {\n    // eslint-disable-next-line @typescript-eslint/naming-convention\n    'sbb-notification': SbbNotificationElement;\n  }\n}\n"],"names":[],"mappings":"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"}
@@ -1 +1 @@
1
- {"version":3,"file":"overlay-base-element.d.ts","sourceRoot":"","sources":["../../../../src/elements/overlay/overlay-base-element.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,KAAK,cAAc,EAAE,MAAM,KAAK,CAAC;AAG1C,OAAO,EAAE,eAAe,EAAE,MAAM,iBAAiB,CAAC;AAClD,OAAO,EAAE,uBAAuB,EAAE,MAAM,0BAA0B,CAAC;AACnE,OAAO,EAAE,qBAAqB,EAAE,MAAM,wBAAwB,CAAC;AAC/D,OAAO,EAAe,gBAAgB,EAAE,MAAM,gBAAgB,CAAC;AAC/D,OAAO,EAAE,YAAY,EAAE,MAAM,qBAAqB,CAAC;AAEnD,OAAO,KAAK,EAAE,2BAA2B,EAAE,MAAM,uBAAuB,CAAC;AAGzE,OAAO,KAAK,EAAE,0BAA0B,EAAE,MAAM,0BAA0B,CAAC;AAG3E,eAAO,MAAM,WAAW,EAAE,qBAAqB,EAAO,CAAC;;AAEvD,8BAAsB,qBAAsB,SAAQ,0BAAyC;IAC3F,kHAAkH;IAC3D,kBAAkB,EAAE,MAAM,GAAG,SAAS,CAAC;IAE9F,8CAA8C;IAC9C,UAAmB,QAAQ,EAAE,YAAY,CAAC,2BAA2B,CAAC,CAGpE;IAGF,SAAS,CAAC,kBAAkB,CAAC,EAAE,WAAW,CAAC;IAC3C,SAAS,CAAC,mBAAmB,CAAC,EAAE,WAAW,CAAC;IAC5C,SAAS,CAAC,iBAAiB,EAAG,eAAe,CAAC;IAC9C,SAAS,CAAC,qBAAqB,EAAG,eAAe,CAAC;IAClD,SAAS,CAAC,YAAY,kBAAyB;IAC/C,SAAS,CAAC,aAAa,mBAA0B;IACjD,SAAS,CAAC,WAAW,EAAE,GAAG,CAAC;IAC3B,SAAS,CAAC,iBAAiB,UAAS;IACpC,SAAS,CAAC,WAAW,EAAG,0BAA0B,CAAC;IACnD,SAAS,CAAC,QAAQ,wBAAmC;IAErD,SAAS,CAAC,QAAQ,CAAC,qBAAqB,CAAC,KAAK,EAAE,cAAc,GAAG,IAAI;IACrE,SAAS,CAAC,QAAQ,CAAC,eAAe,IAAI,IAAI;IAC1C,SAAS,CAAC,QAAQ,CAAC,cAAc,EAAE,MAAM,CAAC;IAE1C,4BAA4B;IACrB,KAAK,CAAC,MAAM,CAAC,EAAE,GAAG,EAAE,MAAM,CAAC,EAAE,WAAW,GAAG,GAAG;IAmBrC,iBAAiB,IAAI,IAAI;cAUtB,YAAY,CAAC,iBAAiB,EAAE,cAAc,CAAC,IAAI,CAAC,GAAG,IAAI;IAO9D,oBAAoB,IAAI,IAAI;IAS5C,SAAS,CAAC,uBAAuB,IAAI,IAAI;IAkBzC,SAAS,CAAC,cAAc,CAAC,KAAK,EAAE,aAAa,GAAG,IAAI;IAWpD,SAAS,CAAC,kCAAkC,IAAI,IAAI;IAKpD,SAAS,CAAC,2BAA2B,CAAC,KAAK,EAAE,KAAK,GAAG,IAAI;IAoBzD,SAAS,CAAC,wBAAwB,IAAI,IAAI;IAI1C,SAAS,CAAC,qBAAqB,CAAC,KAAK,CAAC,EAAE,MAAM,GAAG,IAAI;CAStD"}
1
+ {"version":3,"file":"overlay-base-element.d.ts","sourceRoot":"","sources":["../../../../src/elements/overlay/overlay-base-element.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,KAAK,cAAc,EAAE,MAAM,KAAK,CAAC;AAG1C,OAAO,EAAE,eAAe,EAAE,MAAM,iBAAiB,CAAC;AAClD,OAAO,EAAE,uBAAuB,EAAE,MAAM,0BAA0B,CAAC;AACnE,OAAO,EAAE,qBAAqB,EAAE,MAAM,wBAAwB,CAAC;AAC/D,OAAO,EAAe,gBAAgB,EAAE,MAAM,gBAAgB,CAAC;AAC/D,OAAO,EAAE,YAAY,EAAE,MAAM,qBAAqB,CAAC;AAEnD,OAAO,KAAK,EAAE,2BAA2B,EAAE,MAAM,uBAAuB,CAAC;AAGzE,OAAO,KAAK,EAAE,0BAA0B,EAAE,MAAM,0BAA0B,CAAC;AAG3E,eAAO,MAAM,WAAW,EAAE,qBAAqB,EAAO,CAAC;;AAEvD,8BAAsB,qBAAsB,SAAQ,0BAAyC;IAC3F,kHAAkH;IAC3D,kBAAkB,EAAE,MAAM,GAAG,SAAS,CAAC;IAE9F,8CAA8C;IAC9C,UAAmB,QAAQ,EAAE,YAAY,CAAC,2BAA2B,CAAC,CAGpE;IAGF,SAAS,CAAC,kBAAkB,CAAC,EAAE,WAAW,CAAC;IAC3C,SAAS,CAAC,mBAAmB,CAAC,EAAE,WAAW,CAAC;IAC5C,SAAS,CAAC,iBAAiB,EAAG,eAAe,CAAC;IAC9C,SAAS,CAAC,qBAAqB,EAAG,eAAe,CAAC;IAClD,SAAS,CAAC,YAAY,kBAAyB;IAC/C,SAAS,CAAC,aAAa,mBAA0B;IACjD,SAAS,CAAC,WAAW,EAAE,GAAG,CAAC;IAC3B,SAAS,CAAC,iBAAiB,UAAS;IACpC,SAAS,CAAC,WAAW,EAAG,0BAA0B,CAAC;IACnD,SAAS,CAAC,QAAQ,wBAAmC;IAErD,SAAS,CAAC,QAAQ,CAAC,qBAAqB,CAAC,KAAK,EAAE,cAAc,GAAG,IAAI;IACrE,SAAS,CAAC,QAAQ,CAAC,eAAe,IAAI,IAAI;IAC1C,SAAS,CAAC,QAAQ,CAAC,cAAc,EAAE,MAAM,CAAC;IAE1C,4BAA4B;IACrB,KAAK,CAAC,MAAM,CAAC,EAAE,GAAG,EAAE,MAAM,CAAC,EAAE,WAAW,GAAG,GAAG;IAmBrC,iBAAiB,IAAI,IAAI;cAUtB,YAAY,CAAC,iBAAiB,EAAE,cAAc,CAAC,IAAI,CAAC,GAAG,IAAI;IAO9D,oBAAoB,IAAI,IAAI;IAU5C,SAAS,CAAC,uBAAuB,IAAI,IAAI;IAkBzC,SAAS,CAAC,cAAc,CAAC,KAAK,EAAE,aAAa,GAAG,IAAI;IAWpD,SAAS,CAAC,kCAAkC,IAAI,IAAI;IAKpD,SAAS,CAAC,2BAA2B,CAAC,KAAK,EAAE,KAAK,GAAG,IAAI;IAoBzD,SAAS,CAAC,wBAAwB,IAAI,IAAI;IAI1C,SAAS,CAAC,qBAAqB,CAAC,KAAK,CAAC,EAAE,MAAM,GAAG,IAAI;CAStD"}