@sbb-esta/lyne-elements 1.11.0 → 1.11.2

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.
@@ -189,7 +189,7 @@ const IS_FOCUSABLE_QUERY = [
189
189
  "details",
190
190
  "summary:not(:disabled)",
191
191
  "[tabindex]"
192
- ].map((selector) => `${selector}:not([disabled],[tabindex="-1"])`).join(",");
192
+ ].map((selector) => `${selector}:not([disabled]:not([disabled-interactive]),[tabindex="-1"])`).join(",");
193
193
  function getFocusableElements(elements, properties) {
194
194
  const focusableEls = /* @__PURE__ */ new Set();
195
195
  function getFocusables(elements2, filter) {
@@ -291,4 +291,4 @@ export {
291
291
  sbbInputModalityDetector,
292
292
  setModalityOnNextFocus
293
293
  };
294
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"a11y.js","sources":["../../../../src/elements/core/a11y/arrow-navigation.ts","../../../../src/elements/core/a11y/fake-event-detection.ts","../../../../src/elements/core/a11y/input-modality-detector.ts","../../../../src/elements/core/a11y/focus-visible-within-controller.ts","../../../../src/elements/core/a11y/interactivity-checker.ts","../../../../src/elements/core/a11y/focus.ts"],"sourcesContent":["interface PrevAndNextKeys {\n  prevKey: string;\n  nextKey: string;\n}\n\nfunction getPrevAndNextKeys(): PrevAndNextKeys {\n  if (document.documentElement.matches(':dir(rtl)')) {\n    return { prevKey: 'ArrowRight', nextKey: 'ArrowLeft' };\n  } else {\n    return { prevKey: 'ArrowLeft', nextKey: 'ArrowRight' };\n  }\n}\n\n/**\n * Check if the key pressed is among those allowed for navigation.\n * @param event The keyboard event to check.\n */\nexport function isArrowKeyPressed(event: KeyboardEvent): boolean {\n  return ['ArrowRight', 'ArrowLeft', 'ArrowUp', 'ArrowDown'].includes(event.key);\n}\n\n/**\n * Check if the key pressed should move the navigation to the next element.\n * @param event The keyboard event to check.\n */\nexport function isPreviousArrowKeyPressed(event: KeyboardEvent): boolean {\n  return event.key === 'ArrowUp' || event.key === getPrevAndNextKeys().prevKey;\n}\n\n/**\n * Check if the key pressed should move the navigation to the next element.\n * @param event The keyboard event to check.\n */\nexport function isNextArrowKeyPressed(event: KeyboardEvent): boolean {\n  return event.key === 'ArrowDown' || event.key === getPrevAndNextKeys().nextKey;\n}\n\n/**\n * Check if the key pressed is among those allowed for navigation.\n * @param event The keyboard event to check.\n */\nexport function isArrowKeyOrPageKeysPressed(event: KeyboardEvent): boolean {\n  return isArrowKeyPressed(event) || ['PageUp', 'PageDown', 'Home', 'End'].includes(event.key);\n}\n\n/**\n * Calculate the index of the next element based on the given offset.\n * @param currentIndex The index of the current element.\n * @param maxIndex The maximum permitted value (e.g. array size).\n * @param offset The amount to move by.\n */\nfunction calcNextIndexInRange(currentIndex: number, maxIndex: number, offset: number): number {\n  return (currentIndex + offset + maxIndex) % maxIndex;\n}\n\nfunction getLastIndex(size: number): number {\n  return size - 1;\n}\n\nconst firstIndex = 0;\n\n/**\n * Gets the index of the element to move to, based on the keyboard input, the current element in the list and the list size.\n * @param event The keyboard event to check.\n * @param current The index of the current element in the list.\n * @param size The size of the list.\n * @returns if it is a 'previous' event, returns the index of the previous element,\n * or the index of the last one if the current element is the first in the list;\n * if it is a 'next' event, returns the index of the next element,\n * or the index of the first one, if the current is the last in the list.\n */\nexport function getNextElementIndex(event: KeyboardEvent, current: number, size: number): number {\n  const { prevKey, nextKey } = getPrevAndNextKeys();\n\n  if (event.key === prevKey || event.key === 'ArrowUp') {\n    return current < firstIndex ? getLastIndex(size) : calcNextIndexInRange(current, size, -1);\n  } else if (event.key === nextKey || event.key === 'ArrowDown') {\n    return current >= size ? firstIndex : calcNextIndexInRange(current, size, 1);\n  }\n  return current;\n}\n","// This implementation is taken from https://github.com/angular/components/blob/main/src/cdk/a11y/fake-event-detection.ts\n\n/** Gets whether an event could be a faked `mousedown` event dispatched by a screen reader. */\nexport function isFakeMousedownFromScreenReader(event: MouseEvent): boolean {\n  // Some screen readers will dispatch a fake `mousedown` event when pressing enter or space on\n  // a clickable element. We can distinguish these events when `event.buttons` is zero, or\n  // `event.detail` is zero depending on the browser:\n  // - `event.buttons` works on Firefox, but fails on Chrome.\n  // - `detail` works on Chrome, but fails on Firefox.\n  return event.buttons === 0 || event.detail === 0;\n}\n\n/** Gets whether an event could be a faked `touchstart` event dispatched by a screen reader. */\nexport function isFakeTouchstartFromScreenReader(event: TouchEvent): boolean {\n  const touch: Touch | undefined =\n    (event.touches && event.touches[0]) || (event.changedTouches && event.changedTouches[0]);\n\n  // A fake `touchstart` can be distinguished from a real one by looking at the `identifier`\n  // which is typically >= 0 on a real device versus -1 from a screen reader. Just to be safe,\n  // we can also look at `radiusX` and `radiusY`. This behavior was observed against a Windows 10\n  // device with a touch screen running NVDA v2020.4 and Firefox 85 or Chrome 88.\n  return (\n    !!touch &&\n    touch.identifier === -1 &&\n    (touch.radiusX == null || touch.radiusX === 1) &&\n    (touch.radiusY == null || touch.radiusY === 1)\n  );\n}\n","// This implementation is inspired by https://github.com/angular/components/blob/main/src/cdk/a11y/input-modality/input-modality-detector.ts\n\nimport { isServer } from 'lit';\n\nimport { getEventTarget } from '../eventing.js';\n\nimport {\n  isFakeMousedownFromScreenReader,\n  isFakeTouchstartFromScreenReader,\n} from './fake-event-detection.js';\n\nexport type SbbInputModality = 'touch' | 'mouse' | 'keyboard' | 'program' | null;\n\n/** Options to configure the behavior of the SbbInputModalityDetector. */\ninterface SbbInputModalityDetectorOptions {\n  /** Keys to ignore when detecting keyboard input modality. */\n  ignoreKeys?: number[];\n}\n\nconst SHIFT = 16;\nconst CONTROL = 17;\nconst ALT = 18;\nconst META = 91; // WIN_KEY_LEFT\nconst MAC_META = 224;\n\n/**\n * Default options for the SbbInputModalityDetector.\n *\n * Modifier keys are ignored by default (i.e. when pressed won't cause the service to detect\n * keyboard input modality) for two reasons:\n *\n * 1. Modifier keys are commonly used with mouse to perform actions such as 'right click' or 'open\n *    in new tab', and are thus less representative of actual keyboard interaction.\n * 2. VoiceOver triggers some keyboard events when linearly navigating with Control + Option (but\n *    confusingly not with Caps Lock). Thus, to have parity with other screen readers, we ignore\n *    these keys so, as to not update the input modality.\n *\n * Note that we do not by default ignore the right Meta key on Safari because it has the same key\n * code as the ContextMenu key on other browsers. When we switch to using event.key, we can\n * distinguish between the two.\n */\nconst inputModalityDetectorOptions: SbbInputModalityDetectorOptions = {\n  ignoreKeys: [ALT, CONTROL, MAC_META, META, SHIFT],\n};\n\n/**\n * The amount of time needed to pass after a touchstart event in order for a subsequent mousedown\n * event to be attributed as mouse and not touch.\n *\n * Through trial and error it was found that a value of around 650ms seems appropriate.\n */\nconst touchBufferMs = 650;\n\n/**\n * Event listener options that enable capturing and also mark the listener as passive if the browser\n * supports it.\n */\nconst modalityEventListenerOptions = {\n  passive: true,\n  capture: true,\n};\n\n/**\n * Service that detects the user's input modality.\n *\n * This service does not update the input modality when a user navigates with a screen reader\n * (e.g. linear navigation with VoiceOver, object navigation / browse mode with NVDA, virtual PC\n * cursor mode with JAWS). This is in part due to technical limitations (i.e. keyboard events do not\n * fire as expected in these modes) but is also arguably the correct behavior. Navigating with a\n * screen reader is akin to visually scanning a page, and should not be interpreted as actual user\n * input interaction.\n *\n * When a user is not navigating but *interacting* with a screen reader, this service attempts to\n * update the input modality to keyboard, but in general this service's behavior is largely\n * undefined.\n */\nclass SbbInputModalityDetector {\n  /** The most recently detected input modality. */\n  public get mostRecentModality(): SbbInputModality | null {\n    return this._mostRecentModality;\n  }\n\n  /**\n   * The most recent modality must be initialised with the value 'mouse' to cover the case where an action is\n   * performed but no mouse or keyboard event has yet occurred on the page (e.g. `sbb-popover` with hover trigger).\n   */\n  private _mostRecentModality: SbbInputModality | null = 'mouse';\n\n  /**\n   * The most recently detected input modality event target. Is null if no input modality has been\n   * detected or if the associated event target is null for some unknown reason.\n   */\n  public get mostRecentTarget(): HTMLElement | null {\n    return this._mostRecentTarget;\n  }\n  private _mostRecentTarget: HTMLElement | null = null;\n\n  /** Options for this SbbInputModalityDetector. */\n  private readonly _options: SbbInputModalityDetectorOptions = {\n    ...inputModalityDetectorOptions,\n  };\n\n  /**\n   * The timestamp of the last touch input modality. Used to determine whether mousedown events\n   * should be attributed to mouse or touch.\n   */\n  private _lastTouchMs = 0;\n\n  public constructor() {\n    if (!isServer) {\n      document.addEventListener('keydown', this._onKeydown, modalityEventListenerOptions);\n      document.addEventListener('mousedown', this._onMousedown, modalityEventListenerOptions);\n      document.addEventListener('touchstart', this._onTouchstart, modalityEventListenerOptions);\n    }\n  }\n\n  public reset(): void {\n    this._mostRecentModality = 'mouse';\n    this._mostRecentTarget = null;\n    this._lastTouchMs = 0;\n  }\n\n  /**\n   * Handles keydown events. Must be an arrow function in order to preserve the context when it gets\n   * bound.\n   */\n  private _onKeydown = (event: KeyboardEvent): void => {\n    // If this is one of the keys we should ignore, then ignore it and don't update the input\n    // modality to keyboard.\n    if (this._options?.ignoreKeys?.some((keyCode) => keyCode === event.keyCode)) {\n      return;\n    }\n\n    this._mostRecentModality = 'keyboard';\n    this._mostRecentTarget = getEventTarget(event);\n  };\n\n  /**\n   * Handles mousedown events. Must be an arrow function in order to preserve the context when it\n   * gets bound.\n   */\n  private _onMousedown = (event: MouseEvent): void => {\n    // Touches trigger both touch and mouse events, so we need to distinguish between mouse events\n    // that were triggered via mouse vs touch. To do so, check if the mouse event occurs closely\n    // after the previous touch event.\n    if (Date.now() - this._lastTouchMs < touchBufferMs) {\n      return;\n    }\n\n    // Fake mousedown events are fired by some screen readers when controls are activated by the\n    // screen reader. Attribute them to keyboard input modality.\n    this._mostRecentModality = isFakeMousedownFromScreenReader(event) ? 'keyboard' : 'mouse';\n    this._mostRecentTarget = getEventTarget(event);\n  };\n\n  /**\n   * Handles touchstart events. Must be an arrow function in order to preserve the context when it\n   * gets bound.\n   */\n  private _onTouchstart = (event: TouchEvent): void => {\n    // Same scenario as mentioned in _onMousedown, but on touch screen devices, fake touchstart\n    // events are fired. Again, attribute to keyboard input modality.\n    if (isFakeTouchstartFromScreenReader(event)) {\n      this._mostRecentModality = 'keyboard';\n      this._mostRecentTarget = getEventTarget(event);\n      return;\n    }\n\n    // Store the timestamp of this touch event, as it's used to distinguish between mouse events\n    // triggered via mouse vs touch.\n    this._lastTouchMs = Date.now();\n\n    this._mostRecentModality = 'touch';\n    this._mostRecentTarget = getEventTarget(event);\n  };\n}\n\nexport const sbbInputModalityDetector = new SbbInputModalityDetector();\n\n// Set the input modality in order to avoid showing the outline in Safari.\nexport function setModalityOnNextFocus(elementToFocus: HTMLElement | null | undefined): void {\n  if (!elementToFocus) {\n    return;\n  }\n\n  const mostRecentModality = sbbInputModalityDetector.mostRecentModality;\n\n  // Set focus origin to element which should receive focus\n  if (!(elementToFocus && mostRecentModality !== null)) {\n    return;\n  }\n  elementToFocus.addEventListener(\n    'focus',\n    () => {\n      elementToFocus.setAttribute('data-focus-origin', mostRecentModality);\n      elementToFocus.addEventListener(\n        'blur',\n        () => elementToFocus.removeAttribute('data-focus-origin'),\n        {\n          once: true,\n        },\n      );\n    },\n    { once: true },\n  );\n}\n","import type { ReactiveController, ReactiveControllerHost } from 'lit';\n\nimport { sbbInputModalityDetector } from './input-modality-detector.js';\n\n// Determine whether the element has a visible focus within.\nexport class SbbFocusVisibleWithinController implements ReactiveController {\n  public constructor(private _host: ReactiveControllerHost & HTMLElement) {\n    this._host.addController(this);\n  }\n\n  private _focusinHandler = (): void => {\n    this._host.toggleAttribute(\n      'data-has-visible-focus-within',\n      sbbInputModalityDetector.mostRecentModality === 'keyboard',\n    );\n  };\n\n  private _focusoutHandler = (): void => {\n    this._host.removeAttribute('data-has-visible-focus-within');\n  };\n\n  public hostConnected(): void {\n    this._host.addEventListener('focusin', this._focusinHandler);\n    this._host.addEventListener('focusout', this._focusoutHandler);\n  }\n\n  public hostDisconnected(): void {\n    this._host.removeEventListener('focusin', this._focusinHandler);\n    this._host.removeEventListener('focusout', this._focusoutHandler);\n  }\n}\n","export function hasGeometry(element: HTMLElement): boolean {\n  // Use logic from jQuery to check for an invisible element.\n  // See https://github.com/jquery/jquery/blob/master/src/css/hiddenVisibleSelectors.js#L12\n  return !!(\n    element.offsetWidth ||\n    element.offsetHeight ||\n    (typeof element.getClientRects === 'function' && element.getClientRects().length)\n  );\n}\n\nexport interface InterfaceInteractivityChecker {\n  isVisible(element: HTMLElement): boolean;\n}\n\nexport class InteractivityChecker implements InterfaceInteractivityChecker {\n  /**\n   * Gets whether an element is visible for the purposes of interactivity.\n   *\n   * This will capture states like `display: none` and `visibility: hidden`, but not things like\n   * being clipped by an `overflow: hidden` parent or being outside the viewport.\n   *\n   * @returns Whether the element is visible.\n   */\n  public isVisible(element: HTMLElement): boolean {\n    return hasGeometry(element) && getComputedStyle(element).visibility === 'visible';\n  }\n}\n\nclass InteractivityCheckerMock implements InterfaceInteractivityChecker {\n  public isVisible(): boolean {\n    return true;\n  }\n}\n\n// For unit tests we need to pretend that an element is always visible.\n// This can be done by checking if visibility property is set to empty.\nexport const interactivityChecker: InterfaceInteractivityChecker =\n  typeof getComputedStyle === 'undefined' ||\n  getComputedStyle(document.documentElement).visibility === ''\n    ? new InteractivityCheckerMock()\n    : new InteractivityChecker();\n","import { interactivityChecker } from './interactivity-checker.js';\n\nexport const IS_FOCUSABLE_QUERY = [\n  'button',\n  '[href]',\n  'input',\n  'select',\n  'textarea',\n  'details',\n  'summary:not(:disabled)',\n  '[tabindex]',\n]\n  .map((selector) => `${selector}:not([disabled],[tabindex=\"-1\"])`)\n  .join(',');\n\n// Note: the use of this function for more complex scenarios (with many nested elements) may be expensive.\nexport function getFocusableElements(\n  elements: HTMLElement[],\n  properties?: {\n    filter?: (el: HTMLElement) => boolean;\n    findFirstFocusable?: boolean;\n    includeInvisibleElements?: boolean;\n  },\n): HTMLElement[] {\n  const focusableEls = new Set<HTMLElement>();\n\n  function getFocusables(elements: HTMLElement[], filter?: (el: HTMLElement) => boolean): void {\n    for (const el of elements) {\n      if (filter && !filter(el)) {\n        continue;\n      }\n\n      if (el.nodeName === 'SLOT') {\n        getFocusables(\n          Array.from((el as HTMLSlotElement).assignedElements()) as HTMLElement[],\n          filter,\n        );\n        continue;\n      }\n\n      if (\n        el.matches(IS_FOCUSABLE_QUERY) &&\n        (properties?.includeInvisibleElements ?? interactivityChecker.isVisible(el))\n      ) {\n        focusableEls.add(el);\n      }\n\n      if (properties?.findFirstFocusable && focusableEls.size > 0) {\n        break;\n      }\n\n      if (el.children.length || el.shadowRoot?.children.length) {\n        const children = Array.from(el.children).length\n          ? (Array.from(el.children) as HTMLElement[])\n          : (Array.from(el.shadowRoot!.children) as HTMLElement[]);\n        getFocusables(children, filter);\n      }\n    }\n  }\n  getFocusables(elements, properties?.filter);\n\n  return [...focusableEls];\n}\n\nexport function getFirstFocusableElement(\n  elements: HTMLElement[],\n  filter?: (el: HTMLElement) => boolean,\n): HTMLElement | null {\n  const focusableElements = getFocusableElements(elements, {\n    filter,\n    findFirstFocusable: true,\n  });\n  return focusableElements.length ? focusableElements[0] : null;\n}\n\nexport class SbbFocusHandler {\n  private _controller = new AbortController();\n\n  /**\n   * @param element in which the focus should be trapped.\n   * @param options options object.\n   * @param options.filter filter function which is applied during searching for focusable element. If an element is filtered, also child elements are filtered.\n   * @param options.postFilter filter function which is applied after collecting focusable elements.\n   */\n  public trap(\n    element: HTMLElement,\n    options?: {\n      filter?: (el: HTMLElement) => boolean;\n      postFilter?: (el: HTMLElement) => boolean;\n    },\n  ): void {\n    element.addEventListener(\n      'keydown',\n      (event) => {\n        if (event.key !== 'Tab') {\n          return;\n        }\n\n        // Dynamically get first and last focusable element, as this might have changed since opening overlay\n        const elementChildren: HTMLElement[] = Array.from(\n          element.shadowRoot!.children || [],\n        ) as HTMLElement[];\n        const focusableElements = getFocusableElements(elementChildren, {\n          filter: options?.filter,\n        });\n        const filteredFocusableElements = focusableElements.filter(\n          options?.postFilter ?? (() => true),\n        );\n\n        if (!filteredFocusableElements.length) {\n          return;\n        }\n\n        const firstFocusable = filteredFocusableElements[0] as HTMLElement;\n        const lastFocusable = filteredFocusableElements[\n          filteredFocusableElements.length - 1\n        ] as HTMLElement;\n\n        const [pivot, next] = event.shiftKey\n          ? [firstFocusable, lastFocusable]\n          : [lastFocusable, firstFocusable];\n\n        if (\n          (firstFocusable.getRootNode() as Document | ShadowRoot).activeElement === pivot ||\n          (lastFocusable.getRootNode() as Document | ShadowRoot).activeElement === pivot\n        ) {\n          next.focus();\n          event.preventDefault();\n        }\n      },\n      { signal: this._controller.signal },\n    );\n  }\n\n  public disconnect(): void {\n    this._controller.abort();\n    this._controller = new AbortController();\n  }\n}\n"],"names":["elements"],"mappings":"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"}
294
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"a11y.js","sources":["../../../../src/elements/core/a11y/arrow-navigation.ts","../../../../src/elements/core/a11y/fake-event-detection.ts","../../../../src/elements/core/a11y/input-modality-detector.ts","../../../../src/elements/core/a11y/focus-visible-within-controller.ts","../../../../src/elements/core/a11y/interactivity-checker.ts","../../../../src/elements/core/a11y/focus.ts"],"sourcesContent":["interface PrevAndNextKeys {\n  prevKey: string;\n  nextKey: string;\n}\n\nfunction getPrevAndNextKeys(): PrevAndNextKeys {\n  if (document.documentElement.matches(':dir(rtl)')) {\n    return { prevKey: 'ArrowRight', nextKey: 'ArrowLeft' };\n  } else {\n    return { prevKey: 'ArrowLeft', nextKey: 'ArrowRight' };\n  }\n}\n\n/**\n * Check if the key pressed is among those allowed for navigation.\n * @param event The keyboard event to check.\n */\nexport function isArrowKeyPressed(event: KeyboardEvent): boolean {\n  return ['ArrowRight', 'ArrowLeft', 'ArrowUp', 'ArrowDown'].includes(event.key);\n}\n\n/**\n * Check if the key pressed should move the navigation to the next element.\n * @param event The keyboard event to check.\n */\nexport function isPreviousArrowKeyPressed(event: KeyboardEvent): boolean {\n  return event.key === 'ArrowUp' || event.key === getPrevAndNextKeys().prevKey;\n}\n\n/**\n * Check if the key pressed should move the navigation to the next element.\n * @param event The keyboard event to check.\n */\nexport function isNextArrowKeyPressed(event: KeyboardEvent): boolean {\n  return event.key === 'ArrowDown' || event.key === getPrevAndNextKeys().nextKey;\n}\n\n/**\n * Check if the key pressed is among those allowed for navigation.\n * @param event The keyboard event to check.\n */\nexport function isArrowKeyOrPageKeysPressed(event: KeyboardEvent): boolean {\n  return isArrowKeyPressed(event) || ['PageUp', 'PageDown', 'Home', 'End'].includes(event.key);\n}\n\n/**\n * Calculate the index of the next element based on the given offset.\n * @param currentIndex The index of the current element.\n * @param maxIndex The maximum permitted value (e.g. array size).\n * @param offset The amount to move by.\n */\nfunction calcNextIndexInRange(currentIndex: number, maxIndex: number, offset: number): number {\n  return (currentIndex + offset + maxIndex) % maxIndex;\n}\n\nfunction getLastIndex(size: number): number {\n  return size - 1;\n}\n\nconst firstIndex = 0;\n\n/**\n * Gets the index of the element to move to, based on the keyboard input, the current element in the list and the list size.\n * @param event The keyboard event to check.\n * @param current The index of the current element in the list.\n * @param size The size of the list.\n * @returns if it is a 'previous' event, returns the index of the previous element,\n * or the index of the last one if the current element is the first in the list;\n * if it is a 'next' event, returns the index of the next element,\n * or the index of the first one, if the current is the last in the list.\n */\nexport function getNextElementIndex(event: KeyboardEvent, current: number, size: number): number {\n  const { prevKey, nextKey } = getPrevAndNextKeys();\n\n  if (event.key === prevKey || event.key === 'ArrowUp') {\n    return current < firstIndex ? getLastIndex(size) : calcNextIndexInRange(current, size, -1);\n  } else if (event.key === nextKey || event.key === 'ArrowDown') {\n    return current >= size ? firstIndex : calcNextIndexInRange(current, size, 1);\n  }\n  return current;\n}\n","// This implementation is taken from https://github.com/angular/components/blob/main/src/cdk/a11y/fake-event-detection.ts\n\n/** Gets whether an event could be a faked `mousedown` event dispatched by a screen reader. */\nexport function isFakeMousedownFromScreenReader(event: MouseEvent): boolean {\n  // Some screen readers will dispatch a fake `mousedown` event when pressing enter or space on\n  // a clickable element. We can distinguish these events when `event.buttons` is zero, or\n  // `event.detail` is zero depending on the browser:\n  // - `event.buttons` works on Firefox, but fails on Chrome.\n  // - `detail` works on Chrome, but fails on Firefox.\n  return event.buttons === 0 || event.detail === 0;\n}\n\n/** Gets whether an event could be a faked `touchstart` event dispatched by a screen reader. */\nexport function isFakeTouchstartFromScreenReader(event: TouchEvent): boolean {\n  const touch: Touch | undefined =\n    (event.touches && event.touches[0]) || (event.changedTouches && event.changedTouches[0]);\n\n  // A fake `touchstart` can be distinguished from a real one by looking at the `identifier`\n  // which is typically >= 0 on a real device versus -1 from a screen reader. Just to be safe,\n  // we can also look at `radiusX` and `radiusY`. This behavior was observed against a Windows 10\n  // device with a touch screen running NVDA v2020.4 and Firefox 85 or Chrome 88.\n  return (\n    !!touch &&\n    touch.identifier === -1 &&\n    (touch.radiusX == null || touch.radiusX === 1) &&\n    (touch.radiusY == null || touch.radiusY === 1)\n  );\n}\n","// This implementation is inspired by https://github.com/angular/components/blob/main/src/cdk/a11y/input-modality/input-modality-detector.ts\n\nimport { isServer } from 'lit';\n\nimport { getEventTarget } from '../eventing.js';\n\nimport {\n  isFakeMousedownFromScreenReader,\n  isFakeTouchstartFromScreenReader,\n} from './fake-event-detection.js';\n\nexport type SbbInputModality = 'touch' | 'mouse' | 'keyboard' | 'program' | null;\n\n/** Options to configure the behavior of the SbbInputModalityDetector. */\ninterface SbbInputModalityDetectorOptions {\n  /** Keys to ignore when detecting keyboard input modality. */\n  ignoreKeys?: number[];\n}\n\nconst SHIFT = 16;\nconst CONTROL = 17;\nconst ALT = 18;\nconst META = 91; // WIN_KEY_LEFT\nconst MAC_META = 224;\n\n/**\n * Default options for the SbbInputModalityDetector.\n *\n * Modifier keys are ignored by default (i.e. when pressed won't cause the service to detect\n * keyboard input modality) for two reasons:\n *\n * 1. Modifier keys are commonly used with mouse to perform actions such as 'right click' or 'open\n *    in new tab', and are thus less representative of actual keyboard interaction.\n * 2. VoiceOver triggers some keyboard events when linearly navigating with Control + Option (but\n *    confusingly not with Caps Lock). Thus, to have parity with other screen readers, we ignore\n *    these keys so, as to not update the input modality.\n *\n * Note that we do not by default ignore the right Meta key on Safari because it has the same key\n * code as the ContextMenu key on other browsers. When we switch to using event.key, we can\n * distinguish between the two.\n */\nconst inputModalityDetectorOptions: SbbInputModalityDetectorOptions = {\n  ignoreKeys: [ALT, CONTROL, MAC_META, META, SHIFT],\n};\n\n/**\n * The amount of time needed to pass after a touchstart event in order for a subsequent mousedown\n * event to be attributed as mouse and not touch.\n *\n * Through trial and error it was found that a value of around 650ms seems appropriate.\n */\nconst touchBufferMs = 650;\n\n/**\n * Event listener options that enable capturing and also mark the listener as passive if the browser\n * supports it.\n */\nconst modalityEventListenerOptions = {\n  passive: true,\n  capture: true,\n};\n\n/**\n * Service that detects the user's input modality.\n *\n * This service does not update the input modality when a user navigates with a screen reader\n * (e.g. linear navigation with VoiceOver, object navigation / browse mode with NVDA, virtual PC\n * cursor mode with JAWS). This is in part due to technical limitations (i.e. keyboard events do not\n * fire as expected in these modes) but is also arguably the correct behavior. Navigating with a\n * screen reader is akin to visually scanning a page, and should not be interpreted as actual user\n * input interaction.\n *\n * When a user is not navigating but *interacting* with a screen reader, this service attempts to\n * update the input modality to keyboard, but in general this service's behavior is largely\n * undefined.\n */\nclass SbbInputModalityDetector {\n  /** The most recently detected input modality. */\n  public get mostRecentModality(): SbbInputModality | null {\n    return this._mostRecentModality;\n  }\n\n  /**\n   * The most recent modality must be initialised with the value 'mouse' to cover the case where an action is\n   * performed but no mouse or keyboard event has yet occurred on the page (e.g. `sbb-popover` with hover trigger).\n   */\n  private _mostRecentModality: SbbInputModality | null = 'mouse';\n\n  /**\n   * The most recently detected input modality event target. Is null if no input modality has been\n   * detected or if the associated event target is null for some unknown reason.\n   */\n  public get mostRecentTarget(): HTMLElement | null {\n    return this._mostRecentTarget;\n  }\n  private _mostRecentTarget: HTMLElement | null = null;\n\n  /** Options for this SbbInputModalityDetector. */\n  private readonly _options: SbbInputModalityDetectorOptions = {\n    ...inputModalityDetectorOptions,\n  };\n\n  /**\n   * The timestamp of the last touch input modality. Used to determine whether mousedown events\n   * should be attributed to mouse or touch.\n   */\n  private _lastTouchMs = 0;\n\n  public constructor() {\n    if (!isServer) {\n      document.addEventListener('keydown', this._onKeydown, modalityEventListenerOptions);\n      document.addEventListener('mousedown', this._onMousedown, modalityEventListenerOptions);\n      document.addEventListener('touchstart', this._onTouchstart, modalityEventListenerOptions);\n    }\n  }\n\n  public reset(): void {\n    this._mostRecentModality = 'mouse';\n    this._mostRecentTarget = null;\n    this._lastTouchMs = 0;\n  }\n\n  /**\n   * Handles keydown events. Must be an arrow function in order to preserve the context when it gets\n   * bound.\n   */\n  private _onKeydown = (event: KeyboardEvent): void => {\n    // If this is one of the keys we should ignore, then ignore it and don't update the input\n    // modality to keyboard.\n    if (this._options?.ignoreKeys?.some((keyCode) => keyCode === event.keyCode)) {\n      return;\n    }\n\n    this._mostRecentModality = 'keyboard';\n    this._mostRecentTarget = getEventTarget(event);\n  };\n\n  /**\n   * Handles mousedown events. Must be an arrow function in order to preserve the context when it\n   * gets bound.\n   */\n  private _onMousedown = (event: MouseEvent): void => {\n    // Touches trigger both touch and mouse events, so we need to distinguish between mouse events\n    // that were triggered via mouse vs touch. To do so, check if the mouse event occurs closely\n    // after the previous touch event.\n    if (Date.now() - this._lastTouchMs < touchBufferMs) {\n      return;\n    }\n\n    // Fake mousedown events are fired by some screen readers when controls are activated by the\n    // screen reader. Attribute them to keyboard input modality.\n    this._mostRecentModality = isFakeMousedownFromScreenReader(event) ? 'keyboard' : 'mouse';\n    this._mostRecentTarget = getEventTarget(event);\n  };\n\n  /**\n   * Handles touchstart events. Must be an arrow function in order to preserve the context when it\n   * gets bound.\n   */\n  private _onTouchstart = (event: TouchEvent): void => {\n    // Same scenario as mentioned in _onMousedown, but on touch screen devices, fake touchstart\n    // events are fired. Again, attribute to keyboard input modality.\n    if (isFakeTouchstartFromScreenReader(event)) {\n      this._mostRecentModality = 'keyboard';\n      this._mostRecentTarget = getEventTarget(event);\n      return;\n    }\n\n    // Store the timestamp of this touch event, as it's used to distinguish between mouse events\n    // triggered via mouse vs touch.\n    this._lastTouchMs = Date.now();\n\n    this._mostRecentModality = 'touch';\n    this._mostRecentTarget = getEventTarget(event);\n  };\n}\n\nexport const sbbInputModalityDetector = new SbbInputModalityDetector();\n\n// Set the input modality in order to avoid showing the outline in Safari.\nexport function setModalityOnNextFocus(elementToFocus: HTMLElement | null | undefined): void {\n  if (!elementToFocus) {\n    return;\n  }\n\n  const mostRecentModality = sbbInputModalityDetector.mostRecentModality;\n\n  // Set focus origin to element which should receive focus\n  if (!(elementToFocus && mostRecentModality !== null)) {\n    return;\n  }\n  elementToFocus.addEventListener(\n    'focus',\n    () => {\n      elementToFocus.setAttribute('data-focus-origin', mostRecentModality);\n      elementToFocus.addEventListener(\n        'blur',\n        () => elementToFocus.removeAttribute('data-focus-origin'),\n        {\n          once: true,\n        },\n      );\n    },\n    { once: true },\n  );\n}\n","import type { ReactiveController, ReactiveControllerHost } from 'lit';\n\nimport { sbbInputModalityDetector } from './input-modality-detector.js';\n\n// Determine whether the element has a visible focus within.\nexport class SbbFocusVisibleWithinController implements ReactiveController {\n  public constructor(private _host: ReactiveControllerHost & HTMLElement) {\n    this._host.addController(this);\n  }\n\n  private _focusinHandler = (): void => {\n    this._host.toggleAttribute(\n      'data-has-visible-focus-within',\n      sbbInputModalityDetector.mostRecentModality === 'keyboard',\n    );\n  };\n\n  private _focusoutHandler = (): void => {\n    this._host.removeAttribute('data-has-visible-focus-within');\n  };\n\n  public hostConnected(): void {\n    this._host.addEventListener('focusin', this._focusinHandler);\n    this._host.addEventListener('focusout', this._focusoutHandler);\n  }\n\n  public hostDisconnected(): void {\n    this._host.removeEventListener('focusin', this._focusinHandler);\n    this._host.removeEventListener('focusout', this._focusoutHandler);\n  }\n}\n","export function hasGeometry(element: HTMLElement): boolean {\n  // Use logic from jQuery to check for an invisible element.\n  // See https://github.com/jquery/jquery/blob/master/src/css/hiddenVisibleSelectors.js#L12\n  return !!(\n    element.offsetWidth ||\n    element.offsetHeight ||\n    (typeof element.getClientRects === 'function' && element.getClientRects().length)\n  );\n}\n\nexport interface InterfaceInteractivityChecker {\n  isVisible(element: HTMLElement): boolean;\n}\n\nexport class InteractivityChecker implements InterfaceInteractivityChecker {\n  /**\n   * Gets whether an element is visible for the purposes of interactivity.\n   *\n   * This will capture states like `display: none` and `visibility: hidden`, but not things like\n   * being clipped by an `overflow: hidden` parent or being outside the viewport.\n   *\n   * @returns Whether the element is visible.\n   */\n  public isVisible(element: HTMLElement): boolean {\n    return hasGeometry(element) && getComputedStyle(element).visibility === 'visible';\n  }\n}\n\nclass InteractivityCheckerMock implements InterfaceInteractivityChecker {\n  public isVisible(): boolean {\n    return true;\n  }\n}\n\n// For unit tests we need to pretend that an element is always visible.\n// This can be done by checking if visibility property is set to empty.\nexport const interactivityChecker: InterfaceInteractivityChecker =\n  typeof getComputedStyle === 'undefined' ||\n  getComputedStyle(document.documentElement).visibility === ''\n    ? new InteractivityCheckerMock()\n    : new InteractivityChecker();\n","import { interactivityChecker } from './interactivity-checker.js';\n\nexport const IS_FOCUSABLE_QUERY = [\n  'button',\n  '[href]',\n  'input',\n  'select',\n  'textarea',\n  'details',\n  'summary:not(:disabled)',\n  '[tabindex]',\n]\n  .map((selector) => `${selector}:not([disabled]:not([disabled-interactive]),[tabindex=\"-1\"])`)\n  .join(',');\n\n// Note: the use of this function for more complex scenarios (with many nested elements) may be expensive.\nexport function getFocusableElements(\n  elements: HTMLElement[],\n  properties?: {\n    filter?: (el: HTMLElement) => boolean;\n    findFirstFocusable?: boolean;\n    includeInvisibleElements?: boolean;\n  },\n): HTMLElement[] {\n  const focusableEls = new Set<HTMLElement>();\n\n  function getFocusables(elements: HTMLElement[], filter?: (el: HTMLElement) => boolean): void {\n    for (const el of elements) {\n      if (filter && !filter(el)) {\n        continue;\n      }\n\n      if (el.nodeName === 'SLOT') {\n        getFocusables(\n          Array.from((el as HTMLSlotElement).assignedElements()) as HTMLElement[],\n          filter,\n        );\n        continue;\n      }\n\n      if (\n        el.matches(IS_FOCUSABLE_QUERY) &&\n        (properties?.includeInvisibleElements ?? interactivityChecker.isVisible(el))\n      ) {\n        focusableEls.add(el);\n      }\n\n      if (properties?.findFirstFocusable && focusableEls.size > 0) {\n        break;\n      }\n\n      if (el.children.length || el.shadowRoot?.children.length) {\n        const children = Array.from(el.children).length\n          ? (Array.from(el.children) as HTMLElement[])\n          : (Array.from(el.shadowRoot!.children) as HTMLElement[]);\n        getFocusables(children, filter);\n      }\n    }\n  }\n  getFocusables(elements, properties?.filter);\n\n  return [...focusableEls];\n}\n\nexport function getFirstFocusableElement(\n  elements: HTMLElement[],\n  filter?: (el: HTMLElement) => boolean,\n): HTMLElement | null {\n  const focusableElements = getFocusableElements(elements, {\n    filter,\n    findFirstFocusable: true,\n  });\n  return focusableElements.length ? focusableElements[0] : null;\n}\n\nexport class SbbFocusHandler {\n  private _controller = new AbortController();\n\n  /**\n   * @param element in which the focus should be trapped.\n   * @param options options object.\n   * @param options.filter filter function which is applied during searching for focusable element. If an element is filtered, also child elements are filtered.\n   * @param options.postFilter filter function which is applied after collecting focusable elements.\n   */\n  public trap(\n    element: HTMLElement,\n    options?: {\n      filter?: (el: HTMLElement) => boolean;\n      postFilter?: (el: HTMLElement) => boolean;\n    },\n  ): void {\n    element.addEventListener(\n      'keydown',\n      (event) => {\n        if (event.key !== 'Tab') {\n          return;\n        }\n\n        // Dynamically get first and last focusable element, as this might have changed since opening overlay\n        const elementChildren: HTMLElement[] = Array.from(\n          element.shadowRoot!.children || [],\n        ) as HTMLElement[];\n        const focusableElements = getFocusableElements(elementChildren, {\n          filter: options?.filter,\n        });\n        const filteredFocusableElements = focusableElements.filter(\n          options?.postFilter ?? (() => true),\n        );\n\n        if (!filteredFocusableElements.length) {\n          return;\n        }\n\n        const firstFocusable = filteredFocusableElements[0] as HTMLElement;\n        const lastFocusable = filteredFocusableElements[\n          filteredFocusableElements.length - 1\n        ] as HTMLElement;\n\n        const [pivot, next] = event.shiftKey\n          ? [firstFocusable, lastFocusable]\n          : [lastFocusable, firstFocusable];\n\n        if (\n          (firstFocusable.getRootNode() as Document | ShadowRoot).activeElement === pivot ||\n          (lastFocusable.getRootNode() as Document | ShadowRoot).activeElement === pivot\n        ) {\n          next.focus();\n          event.preventDefault();\n        }\n      },\n      { signal: this._controller.signal },\n    );\n  }\n\n  public disconnect(): void {\n    this._controller.abort();\n    this._controller = new AbortController();\n  }\n}\n"],"names":["elements"],"mappings":"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"}
@@ -1 +1 @@
1
- {"version":3,"file":"action-base-element.d.ts","sourceRoot":"","sources":["../../../../../src/elements/core/base-elements/action-base-element.ts"],"names":[],"mappings":"AAAA,OAAO,EAAQ,UAAU,EAAE,KAAK,cAAc,EAAE,MAAM,KAAK,CAAC;AAgB5D,8BAGsB,oBAAqB,SAAQ,UAAU;IAC3D,SAAS,KAAK,aAAa,IAAI,OAAO,GAAG,SAAS,CAGjD;IAED,SAAS,KAAK,wBAAwB,IAAI,OAAO,GAAG,SAAS,CAE5D;IAEe,iBAAiB,IAAI,IAAI;IAezC,SAAS,CAAC,sBAAsB,IAAI,IAAI;IAexC,6DAA6D;IAC7D,SAAS,CAAC,cAAc,IAAI,cAAc;IAI1C,wDAAwD;cACrC,MAAM,IAAI,cAAc;CAO5C"}
1
+ {"version":3,"file":"action-base-element.d.ts","sourceRoot":"","sources":["../../../../../src/elements/core/base-elements/action-base-element.ts"],"names":[],"mappings":"AAAA,OAAO,EAAQ,UAAU,EAAE,KAAK,cAAc,EAAE,MAAM,KAAK,CAAC;AAe5D,8BAGsB,oBAAqB,SAAQ,UAAU;IAC3D,SAAS,KAAK,aAAa,IAAI,OAAO,GAAG,SAAS,CAGjD;IAED,SAAS,KAAK,wBAAwB,IAAI,OAAO,GAAG,SAAS,CAE5D;IAEe,iBAAiB,IAAI,IAAI;IAezC,SAAS,CAAC,sBAAsB,IAAI,IAAI;IAexC,6DAA6D;IAC7D,SAAS,CAAC,cAAc,IAAI,cAAc;IAI1C,wDAAwD;cACrC,MAAM,IAAI,cAAc;CAG5C"}
@@ -1 +1 @@
1
- {"version":3,"file":"link-base-element.d.ts","sourceRoot":"","sources":["../../../../../src/elements/core/base-elements/link-base-element.ts"],"names":[],"mappings":"AAAA,OAAO,EAA2B,KAAK,cAAc,EAAE,MAAM,KAAK,CAAC;AAGnE,OAAO,EAAE,qBAAqB,EAAE,MAAM,mBAAmB,CAAC;AAK1D,OAAO,EAAE,oBAAoB,EAAE,MAAM,0BAA0B,CAAC;AAEhE,OAAO,6BAA6B,CAAC;AAErC,0DAA0D;AAC1D,MAAM,MAAM,cAAc,GAAG,QAAQ,GAAG,OAAO,GAAG,SAAS,GAAG,MAAM,CAAC;AAErE,uBAAuB;AACvB,8BAGsB,kBAAmB,SAAQ,oBAAoB;IACnE,0CAA0C;IACvB,IAAI,CAAC,EAAE,MAAM,CAAC;IAEjC,uCAAuC;IACpB,MAAM,CAAC,EAAE,cAAc,GAAG,MAAM,CAAC;IAEpD,wEAAwE;IACrD,GAAG,CAAC,EAAE,MAAM,CAAC;IAEhC,kEAAkE;IAC9B,QAAQ,CAAC,EAAE,OAAO,CAAC;IAEvD,wEAAwE;IACjB,kBAAkB,EAAE,MAAM,GAAG,SAAS,CAAC;IAE9F,SAAS,CAAC,QAAQ,wBAAmC;;IASrD,gBAAgB;IACA,KAAK,CAAC,OAAO,CAAC,EAAE,YAAY,GAAG,SAAS,GAAG,IAAI;IAI/D,gBAAgB;IACA,IAAI,IAAI,IAAI;IAI5B,gBAAgB;IACA,KAAK,IAAI,IAAI;IAI7B,OAAO,CAAC,qBAAqB,CAE3B;IAEF,mHAAmH;cAChG,MAAM,IAAI,cAAc;CAqB5C"}
1
+ {"version":3,"file":"link-base-element.d.ts","sourceRoot":"","sources":["../../../../../src/elements/core/base-elements/link-base-element.ts"],"names":[],"mappings":"AAAA,OAAO,EAA2B,KAAK,cAAc,EAAE,MAAM,KAAK,CAAC;AAGnE,OAAO,EAAE,qBAAqB,EAAE,MAAM,mBAAmB,CAAC;AAI1D,OAAO,EAAE,oBAAoB,EAAE,MAAM,0BAA0B,CAAC;AAEhE,OAAO,6BAA6B,CAAC;AAErC,0DAA0D;AAC1D,MAAM,MAAM,cAAc,GAAG,QAAQ,GAAG,OAAO,GAAG,SAAS,GAAG,MAAM,CAAC;AAErE,uBAAuB;AACvB,8BAGsB,kBAAmB,SAAQ,oBAAoB;IACnE,0CAA0C;IACvB,IAAI,CAAC,EAAE,MAAM,CAAC;IAEjC,uCAAuC;IACpB,MAAM,CAAC,EAAE,cAAc,GAAG,MAAM,CAAC;IAEpD,wEAAwE;IACrD,GAAG,CAAC,EAAE,MAAM,CAAC;IAEhC,kEAAkE;IAC9B,QAAQ,CAAC,EAAE,OAAO,CAAC;IAEvD,wEAAwE;IACjB,kBAAkB,EAAE,MAAM,GAAG,SAAS,CAAC;IAE9F,SAAS,CAAC,QAAQ,wBAAmC;;IASrD,gBAAgB;IACA,KAAK,CAAC,OAAO,CAAC,EAAE,YAAY,GAAG,SAAS,GAAG,IAAI;IAI/D,gBAAgB;IACA,IAAI,IAAI,IAAI;IAI5B,gBAAgB;IACA,KAAK,IAAI,IAAI;IAI7B,OAAO,CAAC,qBAAqB,CAE3B;IAEF,mHAAmH;cAChG,MAAM,IAAI,cAAc;CAqB5C"}
@@ -1,6 +1,5 @@
1
1
  import { LitElement, html, isServer, nothing } from "lit";
2
2
  import { hostAttributes } from "./decorators.js";
3
- import { getLocalName } from "./dom.js";
4
3
  import { property } from "lit/decorators.js";
5
4
  import { isEventPrevented, EventEmitter } from "./eventing.js";
6
5
  import { SbbLanguageController } from "./controllers.js";
@@ -54,11 +53,7 @@ let SbbActionBaseElement = class extends LitElement {
54
53
  }
55
54
  /** Default render method for button-like components. */
56
55
  render() {
57
- return html`
58
- <span class="sbb-action-base ${this.localName ?? getLocalName(this)}">
59
- ${this.renderTemplate()}
60
- </span>
61
- `;
56
+ return html` <span class="sbb-action-base ${this.localName}">${this.renderTemplate()}</span> `;
62
57
  }
63
58
  };
64
59
  SbbActionBaseElement = __decorateClass$2([
@@ -219,7 +214,7 @@ let SbbLinkBaseElement = class extends SbbActionBaseElement {
219
214
  render() {
220
215
  return html`
221
216
  <a
222
- class="sbb-action-base ${this.localName ?? getLocalName(this)}"
217
+ class="sbb-action-base ${this.localName}"
223
218
  href=${this.href || nothing}
224
219
  ?download=${this.download}
225
220
  target=${this.target || nothing}
@@ -302,4 +297,4 @@ export {
302
297
  SbbLinkBaseElement,
303
298
  SbbOpenCloseBaseElement
304
299
  };
305
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"base-elements.js","sources":["../../../../src/elements/core/base-elements/action-base-element.ts","../../../../src/elements/core/base-elements/button-base-element.ts","../../../../src/elements/core/base-elements/link-base-element.ts","../../../../src/elements/core/base-elements/open-close-base-element.ts"],"sourcesContent":["import { html, LitElement, type TemplateResult } from 'lit';\n\nimport { hostAttributes } from '../decorators.js';\nimport { getLocalName } from '../dom.js';\n\n/**\n * Whenever an element can be disabled it has disabled property\n * or formDisabled if it's a form element.\n * Because we can't use types here directly we created this helper type.\n */\ntype MaybeDisabled = {\n  disabled?: boolean;\n  formDisabled?: boolean;\n  disabledInteractive?: boolean;\n};\n\n@hostAttributes({\n  'data-action': '',\n})\nexport abstract class SbbActionBaseElement extends LitElement {\n  protected get maybeDisabled(): boolean | undefined {\n    const maybeDisabled = this as MaybeDisabled;\n    return maybeDisabled.disabled || maybeDisabled.formDisabled;\n  }\n\n  protected get maybeDisabledInteractive(): boolean | undefined {\n    return (this as MaybeDisabled).disabledInteractive;\n  }\n\n  public override connectedCallback(): void {\n    super.connectedCallback();\n\n    if (\n      import.meta.env.DEV &&\n      (this.hasAttribute('data-link') || this.hasAttribute('data-button')) &&\n      this.parentElement?.closest('[data-link], [data-button], a, button')\n    ) {\n      console.warn(\n        `Nested action element detected (${this.localName} inside ${this.parentElement!.closest('[data-link], [data-button], a, button')!.localName}). Maybe use a static variant for the inner action element?`,\n        this,\n      );\n    }\n  }\n\n  protected setupBaseEventHandlers(): void {\n    this.addEventListener(\n      'click',\n      (event) => {\n        if (this.maybeDisabled && !this.maybeDisabledInteractive) {\n          event.preventDefault();\n          event.stopImmediatePropagation();\n        }\n      },\n      // capture is necessary here, as this event handler needs to be executed before any other\n      // in order to stop immediate propagation in the disabled case.\n      { capture: true },\n    );\n  }\n\n  /** Override this method to render the component template. */\n  protected renderTemplate(): TemplateResult {\n    throw new Error('Implementation needed!');\n  }\n\n  /** Default render method for button-like components. */\n  protected override render(): TemplateResult {\n    return html`\n      <span class=\"sbb-action-base ${this.localName ?? getLocalName(this)}\">\n        ${this.renderTemplate()}\n      </span>\n    `;\n  }\n}\n","import { isServer } from 'lit';\nimport { property } from 'lit/decorators.js';\n\nimport { hostAttributes } from '../decorators.js';\nimport { isEventPrevented } from '../eventing.js';\n\nimport { SbbActionBaseElement } from './action-base-element.js';\n\n/** Enumeration for type attribute in <button> HTML tag. */\nexport type SbbButtonType = 'button' | 'reset' | 'submit';\n\n/** Button base class. */\n@hostAttributes({\n  role: 'button',\n  tabindex: '0',\n  'data-button': '',\n})\nexport abstract class SbbButtonBaseElement extends SbbActionBaseElement {\n  /** The type attribute to use for the button. */\n  @property() public type: SbbButtonType = 'button';\n\n  /**\n   * The name of the button element.\n   *\n   * @description Developer note: In this case updating the attribute must be synchronous.\n   * Due to this it is implemented as a getter/setter and the attributeChangedCallback() handles the diff check.\n   */\n  @property()\n  public set name(name: string) {\n    this.setAttribute('name', `${name}`);\n  }\n  public get name(): string {\n    return this.getAttribute('name') ?? '';\n  }\n\n  /**\n   * The value of the button element.\n   *\n   * @description Developer note: In this case updating the attribute must be synchronous.\n   * Due to this it is implemented as a getter/setter and the attributeChangedCallback() handles the diff check.\n   */\n  @property()\n  public set value(value: string) {\n    this.setAttribute('value', `${value}`);\n  }\n  public get value(): string {\n    return this.getAttribute('value') ?? '';\n  }\n\n  /** The <form> element to associate the button with. */\n  @property() public form?: string;\n\n  public constructor() {\n    super();\n    if (!isServer) {\n      this.setupBaseEventHandlers();\n\n      const passiveOptions = { passive: true };\n      this.addEventListener('click', this._handleButtonClick);\n      this.addEventListener('keydown', this._preventScrollOnSpaceKeydown);\n      this.addEventListener('keyup', this._dispatchClickEventOnSpaceKeyup, passiveOptions);\n      this.addEventListener('blur', this._removeActiveMarker, passiveOptions);\n      this.addEventListener(\n        'keypress',\n        (event: KeyboardEvent): void => {\n          if (event.key === 'Enter' || event.key === '\\n') {\n            this._dispatchClickEvent(event);\n          }\n        },\n        passiveOptions,\n      );\n    }\n  }\n\n  private _handleButtonClick = async (event: MouseEvent): Promise<void> => {\n    if (this.type === 'button' || (await isEventPrevented(event))) {\n      return;\n    }\n\n    // Use querySelector with form and id selector, as the form property must\n    // reference a valid <form> element\n    const form = this.form\n      ? (this.ownerDocument.querySelector(`form#${this.form}`) as HTMLFormElement)\n      : this.closest('form');\n    if (!form) {\n      return;\n    } else if (this.type === 'submit') {\n      // `form.requestSubmit(element);` seems not to work for CustomElements, so the `element` parameter has been removed;\n      // TODO: Check if solved in any way, see https://github.com/WICG/webcomponents/issues/814#issuecomment-1218452137\n      form.requestSubmit();\n    } else if (this.type === 'reset') {\n      form.reset();\n    }\n  };\n\n  /**\n   * Prevents scrolling from pressing Space, when the event target is an action element.\n   * Also sets data-active attribute.\n   * @param event The origin event.\n   */\n  private _preventScrollOnSpaceKeydown = (event: KeyboardEvent): void => {\n    if (event.key === ' ') {\n      event.preventDefault();\n      (event.target as HTMLElement).toggleAttribute('data-active', true);\n    }\n  };\n\n  private _removeActiveMarker = (event: Event): void => {\n    (event.target as HTMLElement).removeAttribute('data-active');\n  };\n\n  /**\n   * Dispatches a 'click' PointerEvent if the original keyboard event is a 'Space' press.\n   * As verified with the native button, when 'Space' is pressed, a 'click' event is dispatched\n   * after the 'keyup' event.\n   * @param event The origin event.\n   */\n  private _dispatchClickEventOnSpaceKeyup = (event: KeyboardEvent): void => {\n    if (event.key === ' ') {\n      this._removeActiveMarker(event);\n      this._dispatchClickEvent(event);\n    }\n  };\n\n  private _dispatchClickEvent = (event: KeyboardEvent): void => {\n    const { altKey, ctrlKey, metaKey, shiftKey } = event;\n    (event.target as Element).dispatchEvent(\n      new PointerEvent('click', {\n        bubbles: true,\n        cancelable: true,\n        composed: true,\n        pointerId: -1,\n        pointerType: '',\n        altKey,\n        ctrlKey,\n        metaKey,\n        shiftKey,\n      }),\n    );\n  };\n\n  public override attributeChangedCallback(\n    name: string,\n    old: string | null,\n    value: string | null,\n  ): void {\n    if (!['name', 'value'].includes(name) || old !== value) {\n      super.attributeChangedCallback(name, old, value);\n    }\n  }\n}\n","import { html, isServer, nothing, type TemplateResult } from 'lit';\nimport { property } from 'lit/decorators.js';\n\nimport { SbbLanguageController } from '../controllers.js';\nimport { hostAttributes } from '../decorators.js';\nimport { getLocalName } from '../dom.js';\nimport { i18nTargetOpensInNewWindow } from '../i18n.js';\n\nimport { SbbActionBaseElement } from './action-base-element.js';\n\nimport '../../screen-reader-only.js';\n\n/** Enumeration for 'target' attribute in <a> HTML tag. */\nexport type LinkTargetType = '_blank' | '_self' | '_parent' | '_top';\n\n/** Link base class. */\n@hostAttributes({\n  'data-link': '',\n})\nexport abstract class SbbLinkBaseElement extends SbbActionBaseElement {\n  /** The href value you want to link to. */\n  @property() public href?: string;\n\n  /** Where to display the linked URL. */\n  @property() public target?: LinkTargetType | string;\n\n  /** The relationship of the linked URL as space-separated link types. */\n  @property() public rel?: string;\n\n  /** Whether the browser will show the download dialog on click. */\n  @property({ type: Boolean }) public download?: boolean;\n\n  /** This will be forwarded as aria-label to the inner anchor element. */\n  @property({ attribute: 'accessibility-label' }) public accessibilityLabel: string | undefined;\n\n  protected language = new SbbLanguageController(this);\n\n  public constructor() {\n    super();\n    if (!isServer) {\n      this.setupBaseEventHandlers();\n    }\n  }\n\n  /** @internal */\n  public override focus(options?: FocusOptions | undefined): void {\n    this.shadowRoot!.querySelector<HTMLAnchorElement>('a')?.focus(options);\n  }\n\n  /** @internal */\n  public override blur(): void {\n    this.shadowRoot!.querySelector<HTMLAnchorElement>('a')?.blur();\n  }\n\n  /** @internal */\n  public override click(): void {\n    this.shadowRoot!.querySelector<HTMLAnchorElement>('a')?.click();\n  }\n\n  private _evaluateRelAttribute = (): string | typeof nothing => {\n    return this.rel ? this.rel : this.target === '_blank' ? 'external noopener nofollow' : nothing;\n  };\n\n  /** Default render method for link-like components. Can be overridden if the LinkRenderVariables are not needed. */\n  protected override render(): TemplateResult {\n    return html`\n      <a\n        class=\"sbb-action-base ${this.localName ?? getLocalName(this)}\"\n        href=${this.href || nothing}\n        ?download=${this.download}\n        target=${this.target || nothing}\n        rel=${this._evaluateRelAttribute()}\n        aria-label=${this.accessibilityLabel || nothing}\n        tabindex=${this.maybeDisabled && !this.maybeDisabledInteractive ? '-1' : nothing}\n        aria-disabled=${this.maybeDisabled ? 'true' : nothing}\n      >\n        ${this.renderTemplate()}\n        ${!!this.href && this.target === '_blank'\n          ? html`<sbb-screen-reader-only\n              >. ${i18nTargetOpensInNewWindow[this.language.current]}</sbb-screen-reader-only\n            >`\n          : nothing}\n      </a>\n    `;\n  }\n}\n","import { LitElement } from 'lit';\n\nimport { EventEmitter } from '../eventing.js';\nimport type { SbbOpenedClosedState } from '../interfaces.js';\n\n/**\n * Base class for overlay components.\n *\n * @event willOpen - Emits whenever the component starts the opening transition. Can be canceled.\n * @event didOpen - Emits whenever the component is opened.\n * @event willClose - Emits whenever the component begins the closing transition. Can be canceled.\n * @event didClose - Emits whenever the component is closed.\n */\nexport abstract class SbbOpenCloseBaseElement extends LitElement {\n  public static readonly events = {\n    willOpen: 'willOpen',\n    didOpen: 'didOpen',\n    willClose: 'willClose',\n    didClose: 'didClose',\n  } as const;\n\n  /** The state of the component. */\n  protected set state(state: SbbOpenedClosedState) {\n    this.setAttribute('data-state', state);\n  }\n  protected get state(): SbbOpenedClosedState {\n    return this.getAttribute('data-state') as SbbOpenedClosedState;\n  }\n\n  /** Whether the element is open. */\n  public get isOpen(): boolean {\n    return this.state === 'opened';\n  }\n\n  /** Emits whenever the component starts the opening transition. */\n  protected willOpen: EventEmitter = new EventEmitter(\n    this,\n    SbbOpenCloseBaseElement.events.willOpen,\n  );\n\n  /** Emits whenever the component is opened. */\n  protected didOpen: EventEmitter = new EventEmitter(this, SbbOpenCloseBaseElement.events.didOpen);\n\n  /** Emits whenever the component begins the closing transition. */\n  protected willClose: EventEmitter = new EventEmitter(\n    this,\n    SbbOpenCloseBaseElement.events.willClose,\n  );\n\n  /** Emits whenever the component is closed. */\n  protected didClose: EventEmitter = new EventEmitter(\n    this,\n    SbbOpenCloseBaseElement.events.didClose,\n  );\n\n  /** Opens the component. */\n  public abstract open(): void;\n  /** Closes the component. */\n  public abstract close(): void;\n\n  public override connectedCallback(): void {\n    super.connectedCallback();\n    this.state ||= 'closed';\n  }\n}\n\ndeclare global {\n  interface GlobalEventHandlersEventMap {\n    willOpen: CustomEvent<void>;\n    willClose: CustomEvent<void>;\n    didOpen: CustomEvent<void>;\n    didClose: CustomEvent<void>;\n  }\n}\n"],"names":["__decorateClass"],"mappings":"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"}
300
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"base-elements.js","sources":["../../../../src/elements/core/base-elements/action-base-element.ts","../../../../src/elements/core/base-elements/button-base-element.ts","../../../../src/elements/core/base-elements/link-base-element.ts","../../../../src/elements/core/base-elements/open-close-base-element.ts"],"sourcesContent":["import { html, LitElement, type TemplateResult } from 'lit';\n\nimport { hostAttributes } from '../decorators.js';\n\n/**\n * Whenever an element can be disabled it has disabled property\n * or formDisabled if it's a form element.\n * Because we can't use types here directly we created this helper type.\n */\ntype MaybeDisabled = {\n  disabled?: boolean;\n  formDisabled?: boolean;\n  disabledInteractive?: boolean;\n};\n\n@hostAttributes({\n  'data-action': '',\n})\nexport abstract class SbbActionBaseElement extends LitElement {\n  protected get maybeDisabled(): boolean | undefined {\n    const maybeDisabled = this as MaybeDisabled;\n    return maybeDisabled.disabled || maybeDisabled.formDisabled;\n  }\n\n  protected get maybeDisabledInteractive(): boolean | undefined {\n    return (this as MaybeDisabled).disabledInteractive;\n  }\n\n  public override connectedCallback(): void {\n    super.connectedCallback();\n\n    if (\n      import.meta.env.DEV &&\n      (this.hasAttribute('data-link') || this.hasAttribute('data-button')) &&\n      this.parentElement?.closest('[data-link], [data-button], a, button')\n    ) {\n      console.warn(\n        `Nested action element detected (${this.localName} inside ${this.parentElement!.closest('[data-link], [data-button], a, button')!.localName}). Maybe use a static variant for the inner action element?`,\n        this,\n      );\n    }\n  }\n\n  protected setupBaseEventHandlers(): void {\n    this.addEventListener(\n      'click',\n      (event) => {\n        if (this.maybeDisabled && !this.maybeDisabledInteractive) {\n          event.preventDefault();\n          event.stopImmediatePropagation();\n        }\n      },\n      // capture is necessary here, as this event handler needs to be executed before any other\n      // in order to stop immediate propagation in the disabled case.\n      { capture: true },\n    );\n  }\n\n  /** Override this method to render the component template. */\n  protected renderTemplate(): TemplateResult {\n    throw new Error('Implementation needed!');\n  }\n\n  /** Default render method for button-like components. */\n  protected override render(): TemplateResult {\n    return html` <span class=\"sbb-action-base ${this.localName}\">${this.renderTemplate()}</span> `;\n  }\n}\n","import { isServer } from 'lit';\nimport { property } from 'lit/decorators.js';\n\nimport { hostAttributes } from '../decorators.js';\nimport { isEventPrevented } from '../eventing.js';\n\nimport { SbbActionBaseElement } from './action-base-element.js';\n\n/** Enumeration for type attribute in <button> HTML tag. */\nexport type SbbButtonType = 'button' | 'reset' | 'submit';\n\n/** Button base class. */\n@hostAttributes({\n  role: 'button',\n  tabindex: '0',\n  'data-button': '',\n})\nexport abstract class SbbButtonBaseElement extends SbbActionBaseElement {\n  /** The type attribute to use for the button. */\n  @property() public type: SbbButtonType = 'button';\n\n  /**\n   * The name of the button element.\n   *\n   * @description Developer note: In this case updating the attribute must be synchronous.\n   * Due to this it is implemented as a getter/setter and the attributeChangedCallback() handles the diff check.\n   */\n  @property()\n  public set name(name: string) {\n    this.setAttribute('name', `${name}`);\n  }\n  public get name(): string {\n    return this.getAttribute('name') ?? '';\n  }\n\n  /**\n   * The value of the button element.\n   *\n   * @description Developer note: In this case updating the attribute must be synchronous.\n   * Due to this it is implemented as a getter/setter and the attributeChangedCallback() handles the diff check.\n   */\n  @property()\n  public set value(value: string) {\n    this.setAttribute('value', `${value}`);\n  }\n  public get value(): string {\n    return this.getAttribute('value') ?? '';\n  }\n\n  /** The <form> element to associate the button with. */\n  @property() public form?: string;\n\n  public constructor() {\n    super();\n    if (!isServer) {\n      this.setupBaseEventHandlers();\n\n      const passiveOptions = { passive: true };\n      this.addEventListener('click', this._handleButtonClick);\n      this.addEventListener('keydown', this._preventScrollOnSpaceKeydown);\n      this.addEventListener('keyup', this._dispatchClickEventOnSpaceKeyup, passiveOptions);\n      this.addEventListener('blur', this._removeActiveMarker, passiveOptions);\n      this.addEventListener(\n        'keypress',\n        (event: KeyboardEvent): void => {\n          if (event.key === 'Enter' || event.key === '\\n') {\n            this._dispatchClickEvent(event);\n          }\n        },\n        passiveOptions,\n      );\n    }\n  }\n\n  private _handleButtonClick = async (event: MouseEvent): Promise<void> => {\n    if (this.type === 'button' || (await isEventPrevented(event))) {\n      return;\n    }\n\n    // Use querySelector with form and id selector, as the form property must\n    // reference a valid <form> element\n    const form = this.form\n      ? (this.ownerDocument.querySelector(`form#${this.form}`) as HTMLFormElement)\n      : this.closest('form');\n    if (!form) {\n      return;\n    } else if (this.type === 'submit') {\n      // `form.requestSubmit(element);` seems not to work for CustomElements, so the `element` parameter has been removed;\n      // TODO: Check if solved in any way, see https://github.com/WICG/webcomponents/issues/814#issuecomment-1218452137\n      form.requestSubmit();\n    } else if (this.type === 'reset') {\n      form.reset();\n    }\n  };\n\n  /**\n   * Prevents scrolling from pressing Space, when the event target is an action element.\n   * Also sets data-active attribute.\n   * @param event The origin event.\n   */\n  private _preventScrollOnSpaceKeydown = (event: KeyboardEvent): void => {\n    if (event.key === ' ') {\n      event.preventDefault();\n      (event.target as HTMLElement).toggleAttribute('data-active', true);\n    }\n  };\n\n  private _removeActiveMarker = (event: Event): void => {\n    (event.target as HTMLElement).removeAttribute('data-active');\n  };\n\n  /**\n   * Dispatches a 'click' PointerEvent if the original keyboard event is a 'Space' press.\n   * As verified with the native button, when 'Space' is pressed, a 'click' event is dispatched\n   * after the 'keyup' event.\n   * @param event The origin event.\n   */\n  private _dispatchClickEventOnSpaceKeyup = (event: KeyboardEvent): void => {\n    if (event.key === ' ') {\n      this._removeActiveMarker(event);\n      this._dispatchClickEvent(event);\n    }\n  };\n\n  private _dispatchClickEvent = (event: KeyboardEvent): void => {\n    const { altKey, ctrlKey, metaKey, shiftKey } = event;\n    (event.target as Element).dispatchEvent(\n      new PointerEvent('click', {\n        bubbles: true,\n        cancelable: true,\n        composed: true,\n        pointerId: -1,\n        pointerType: '',\n        altKey,\n        ctrlKey,\n        metaKey,\n        shiftKey,\n      }),\n    );\n  };\n\n  public override attributeChangedCallback(\n    name: string,\n    old: string | null,\n    value: string | null,\n  ): void {\n    if (!['name', 'value'].includes(name) || old !== value) {\n      super.attributeChangedCallback(name, old, value);\n    }\n  }\n}\n","import { html, isServer, nothing, type TemplateResult } from 'lit';\nimport { property } from 'lit/decorators.js';\n\nimport { SbbLanguageController } from '../controllers.js';\nimport { hostAttributes } from '../decorators.js';\nimport { i18nTargetOpensInNewWindow } from '../i18n.js';\n\nimport { SbbActionBaseElement } from './action-base-element.js';\n\nimport '../../screen-reader-only.js';\n\n/** Enumeration for 'target' attribute in <a> HTML tag. */\nexport type LinkTargetType = '_blank' | '_self' | '_parent' | '_top';\n\n/** Link base class. */\n@hostAttributes({\n  'data-link': '',\n})\nexport abstract class SbbLinkBaseElement extends SbbActionBaseElement {\n  /** The href value you want to link to. */\n  @property() public href?: string;\n\n  /** Where to display the linked URL. */\n  @property() public target?: LinkTargetType | string;\n\n  /** The relationship of the linked URL as space-separated link types. */\n  @property() public rel?: string;\n\n  /** Whether the browser will show the download dialog on click. */\n  @property({ type: Boolean }) public download?: boolean;\n\n  /** This will be forwarded as aria-label to the inner anchor element. */\n  @property({ attribute: 'accessibility-label' }) public accessibilityLabel: string | undefined;\n\n  protected language = new SbbLanguageController(this);\n\n  public constructor() {\n    super();\n    if (!isServer) {\n      this.setupBaseEventHandlers();\n    }\n  }\n\n  /** @internal */\n  public override focus(options?: FocusOptions | undefined): void {\n    this.shadowRoot!.querySelector<HTMLAnchorElement>('a')?.focus(options);\n  }\n\n  /** @internal */\n  public override blur(): void {\n    this.shadowRoot!.querySelector<HTMLAnchorElement>('a')?.blur();\n  }\n\n  /** @internal */\n  public override click(): void {\n    this.shadowRoot!.querySelector<HTMLAnchorElement>('a')?.click();\n  }\n\n  private _evaluateRelAttribute = (): string | typeof nothing => {\n    return this.rel ? this.rel : this.target === '_blank' ? 'external noopener nofollow' : nothing;\n  };\n\n  /** Default render method for link-like components. Can be overridden if the LinkRenderVariables are not needed. */\n  protected override render(): TemplateResult {\n    return html`\n      <a\n        class=\"sbb-action-base ${this.localName}\"\n        href=${this.href || nothing}\n        ?download=${this.download}\n        target=${this.target || nothing}\n        rel=${this._evaluateRelAttribute()}\n        aria-label=${this.accessibilityLabel || nothing}\n        tabindex=${this.maybeDisabled && !this.maybeDisabledInteractive ? '-1' : nothing}\n        aria-disabled=${this.maybeDisabled ? 'true' : nothing}\n      >\n        ${this.renderTemplate()}\n        ${!!this.href && this.target === '_blank'\n          ? html`<sbb-screen-reader-only\n              >. ${i18nTargetOpensInNewWindow[this.language.current]}</sbb-screen-reader-only\n            >`\n          : nothing}\n      </a>\n    `;\n  }\n}\n","import { LitElement } from 'lit';\n\nimport { EventEmitter } from '../eventing.js';\nimport type { SbbOpenedClosedState } from '../interfaces.js';\n\n/**\n * Base class for overlay components.\n *\n * @event willOpen - Emits whenever the component starts the opening transition. Can be canceled.\n * @event didOpen - Emits whenever the component is opened.\n * @event willClose - Emits whenever the component begins the closing transition. Can be canceled.\n * @event didClose - Emits whenever the component is closed.\n */\nexport abstract class SbbOpenCloseBaseElement extends LitElement {\n  public static readonly events = {\n    willOpen: 'willOpen',\n    didOpen: 'didOpen',\n    willClose: 'willClose',\n    didClose: 'didClose',\n  } as const;\n\n  /** The state of the component. */\n  protected set state(state: SbbOpenedClosedState) {\n    this.setAttribute('data-state', state);\n  }\n  protected get state(): SbbOpenedClosedState {\n    return this.getAttribute('data-state') as SbbOpenedClosedState;\n  }\n\n  /** Whether the element is open. */\n  public get isOpen(): boolean {\n    return this.state === 'opened';\n  }\n\n  /** Emits whenever the component starts the opening transition. */\n  protected willOpen: EventEmitter = new EventEmitter(\n    this,\n    SbbOpenCloseBaseElement.events.willOpen,\n  );\n\n  /** Emits whenever the component is opened. */\n  protected didOpen: EventEmitter = new EventEmitter(this, SbbOpenCloseBaseElement.events.didOpen);\n\n  /** Emits whenever the component begins the closing transition. */\n  protected willClose: EventEmitter = new EventEmitter(\n    this,\n    SbbOpenCloseBaseElement.events.willClose,\n  );\n\n  /** Emits whenever the component is closed. */\n  protected didClose: EventEmitter = new EventEmitter(\n    this,\n    SbbOpenCloseBaseElement.events.didClose,\n  );\n\n  /** Opens the component. */\n  public abstract open(): void;\n  /** Closes the component. */\n  public abstract close(): void;\n\n  public override connectedCallback(): void {\n    super.connectedCallback();\n    this.state ||= 'closed';\n  }\n}\n\ndeclare global {\n  interface GlobalEventHandlersEventMap {\n    willOpen: CustomEvent<void>;\n    willClose: CustomEvent<void>;\n    didOpen: CustomEvent<void>;\n    didClose: CustomEvent<void>;\n  }\n}\n"],"names":["__decorateClass"],"mappings":"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"}
@@ -6,6 +6,8 @@ import { LitElement } from 'lit';
6
6
  * https://github.com/lit/lit/blob/main/packages/labs/ssr-dom-shim/src.ts (See CustomElementRegistryShim)
7
7
  *
8
8
  * Can be removed once https://github.com/lit/lit/pull/4553 is merged/released.
9
+ *
10
+ * @deprecated Will be removed with next major version
9
11
  */
10
12
  export declare function getLocalName(element: LitElement): string;
11
13
  //# sourceMappingURL=ssr.d.ts.map
@@ -1 +1 @@
1
- {"version":3,"file":"ssr.d.ts","sourceRoot":"","sources":["../../../../../src/elements/core/dom/ssr.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,UAAU,EAAE,MAAM,KAAK,CAAC;AAItC;;;;;;;GAOG;AACH,wBAAgB,YAAY,CAAC,OAAO,EAAE,UAAU,GAAG,MAAM,CAcxD"}
1
+ {"version":3,"file":"ssr.d.ts","sourceRoot":"","sources":["../../../../../src/elements/core/dom/ssr.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,UAAU,EAAE,MAAM,KAAK,CAAC;AAItC;;;;;;;;;GASG;AACH,wBAAgB,YAAY,CAAC,OAAO,EAAE,UAAU,GAAG,MAAM,CAcxD"}
@@ -142,4 +142,4 @@ export {
142
142
  pageScrollDisabled,
143
143
  setOrRemoveAttribute
144
144
  };
145
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"dom.js","sources":["../../../../src/elements/core/dom/breakpoint.ts","../../../../src/elements/core/dom/find-referenced-element.ts","../../../../src/elements/core/dom/get-document-writing-mode.ts","../../../../src/elements/core/dom/host-context.ts","../../../../src/elements/core/dom/input-element.ts","../../../../src/elements/core/dom/set-or-remove-attribute.ts","../../../../src/elements/core/dom/platform.ts","../../../../src/elements/core/dom/scroll.ts","../../../../src/elements/core/dom/ssr.ts"],"sourcesContent":["import { isServer } from 'lit';\n\nexport const breakpoints = ['zero', 'micro', 'small', 'medium', 'wide', 'large', 'ultra'] as const;\nexport type Breakpoint = (typeof breakpoints)[number];\n\n/**\n * Checks whether the document matches a particular media query.\n * It will rely on the global CSS variables to determine the value of the breakpoints.\n *\n * @param from The breakpoint corresponding to the `min-width` value of the media query (optional).\n * @param to The breakpoint corresponding to the `max-width` value of the media query (optional).\n * @returns A boolean indicating whether the window matches the breakpoint.\n */\nexport function isBreakpoint(\n  from?: Breakpoint,\n  to?: Breakpoint,\n  properties?: { includeMaxBreakpoint: boolean },\n): boolean {\n  if (isServer) {\n    // TODO: Remove and decide case by case what should be done on consuming end\n    return false;\n  }\n\n  const computedStyle = getComputedStyle(document.documentElement);\n  const breakpointMin = from ? computedStyle.getPropertyValue(`--sbb-breakpoint-${from}-min`) : '';\n  const breakpointMax = to\n    ? `${\n        parseFloat(\n          computedStyle.getPropertyValue(\n            `--sbb-breakpoint-${to}-${properties?.includeMaxBreakpoint ? 'max' : 'min'}`,\n          ),\n        ) - (properties?.includeMaxBreakpoint ? 0 : 0.0625)\n      }rem`\n    : ''; // subtract 1px (0.0625rem) from the max-width breakpoint\n\n  const minWidth = breakpointMin && `(min-width: ${breakpointMin})`;\n  const maxWidth = breakpointMax && `(max-width: ${breakpointMax})`;\n  const and = breakpointMin && breakpointMax && ' and ';\n\n  return window.matchMedia(`${minWidth}${and}${maxWidth}`).matches;\n}\n","import { isServer } from 'lit';\n\n/**\n *  Check whether it's a string or an HTMLElement, if it's a string queries the element with the\n *  corresponding id.\n *  @param reference either the wanted id or the HTMLElement\n */\nexport function findReferencedElement<T extends HTMLElement = HTMLElement>(\n  reference: string | HTMLElement,\n): T | null {\n  if (isServer) {\n    return null;\n  } else if (typeof reference === 'string') {\n    return document.getElementById(reference) as T;\n  } else if (reference instanceof window.Element) {\n    return reference as T;\n  }\n  return null;\n}\n","import { isServer } from 'lit';\n\n/**\n * @deprecated use :dir() selector instead\n */\nexport const getDocumentWritingMode = (): string =>\n  (!isServer && document.documentElement.getAttribute('dir')) || 'ltr';\n","import { isServer } from 'lit';\n\n/**\n * Looks for the closest element matching the given selector starting from the given element.\n * Returns null, if none of the ancestor match.\n *\n * @param selector The selector to match ancestor against.\n * @param element The base element from which to start the search.\n * @returns The closest element matching the selector or null if none is found.\n */\nexport function hostContext(selector: string, element: Element): Element | null {\n  if (isServer) {\n    return null;\n  }\n  // Start with parent element in order to avoid finding element itself\n  element = element.parentElement ?? (element.getRootNode() as ShadowRoot).host;\n  while (element && (element as any) !== document && (element as any) !== window) {\n    const match = element.closest(selector);\n    if (match) {\n      return match;\n    }\n\n    element = (element.getRootNode() as ShadowRoot).host;\n  }\n\n  return null;\n}\n\n// A list of elements that should not allow another anchor or button element inside them.\n// Needs to be extended if additional elements fall into this category.\nexport const ACTION_ELEMENTS = 'a,button,sbb-teaser-hero,sbb-teaser';\n","import { findReferencedElement } from './find-referenced-element.js';\n\n/**\n * Given an element, returns the related input reference, if it exists respecting following priority.\n * 1. Input field in `sbb-form-field` (if trigger is undefiend)\n * 2. Input referenced by id (trigger is string)\n * 3. Input referenced directly (trigger is HTMLElement)\n * @param element The starting SbbDatepickerElement element.\n * @param trigger The id or the reference of the input.\n */\nexport function findInput(\n  element: HTMLElement,\n  trigger?: string | HTMLElement | null,\n): HTMLInputElement | null {\n  if (!trigger) {\n    const parent = element.closest?.('sbb-form-field');\n    return parent?.querySelector('input') as HTMLInputElement | null;\n  }\n\n  return findReferencedElement<HTMLInputElement>(trigger);\n}\n","/**\n * Set the attribute only if value is not 'false', otherwise remove attribute.\n * @param element The element that will have the attribute\n * @param attribute The attribute name\n * @param value The attribute value\n */\nexport function setOrRemoveAttribute(element: HTMLElement, attribute: string, value?: any): void {\n  if (!value) {\n    element.removeAttribute(attribute);\n  } else {\n    element.setAttribute(attribute, value);\n  }\n}\n","// This file is freely inspired by Angular Material solution.\n// Reference: https://github.com/angular/components/blob/main/src/cdk/platform/platform.ts\nimport { isServer } from 'lit';\n\n// Whether the current platform supports the V8 Break Iterator. The V8 check\n// is necessary to detect all Blink based browsers.\nlet hasV8BreakIterator: boolean;\n\n// We need a try/catch around the reference to `Intl`, because accessing it in some cases can\n// cause IE to throw. These cases are tied to particular versions of Windows and can happen if\n// the consumer is providing a polyfilled `Map`. See:\n// https://github.com/Microsoft/ChakraCore/issues/3189\n// https://github.com/angular/components/issues/15687\ntry {\n  hasV8BreakIterator = typeof Intl !== 'undefined' && (Intl as any).v8BreakIterator;\n} catch {\n  hasV8BreakIterator = false;\n}\n\n/** Whether the current browser is Microsoft Edge. */\nexport const isEdge = !isServer && /(edge)/i.test(navigator.userAgent);\n\n/** Whether the current rendering engine is Microsoft Trident. */\nexport const isTrident = !isServer && /(msie|trident)/i.test(navigator.userAgent);\n\n// EdgeHTML and Trident mock Blink specific things and need to be excluded from this check.\n/** Whether the current rendering engine is Blink. */\nexport const isBlink =\n  !isServer &&\n  !!((window as any).chrome || hasV8BreakIterator) &&\n  typeof CSS !== 'undefined' &&\n  !isEdge &&\n  !isTrident;\n\n// Webkit is part of the userAgent in EdgeHTML, Blink and Trident. Therefore, we need to\n// ensure that Webkit runs standalone and is not used as another engine's base.\n/** Whether the current rendering engine is WebKit. */\nexport const isWebkit =\n  !isServer && /AppleWebKit/i.test(navigator.userAgent) && !isBlink && !isEdge && !isTrident;\n\n/** Whether the current platform is Apple iOS. */\nexport const isIOS =\n  !isServer && /iPad|iPhone|iPod/.test(navigator.userAgent) && !('MSStream' in window);\n\n// It's difficult to detect the plain Gecko engine, because most of the browsers identify\n// them self as Gecko-like browsers and modify the userAgent's according to that.\n// Since we only cover one explicit Firefox case, we can simply check for Firefox\n// instead of having an unstable check for Gecko.\n/** Whether the current browser is Firefox. */\nexport const isFirefox = !isServer && /(firefox|minefield)/i.test(navigator.userAgent);\n\n/** Whether the current platform is Android. */\n// Trident on mobile adds the android platform to the userAgent to trick detections.\nexport const isAndroid = !isServer && /android/i.test(navigator.userAgent) && !isTrident;\n\n// Safari browsers will include the Safari keyword in their userAgent. Some browsers may fake\n// this and just place the Safari keyword in the userAgent. To be more safe about Safari every\n// Safari browser should also use Webkit as its layout engine.\n/** Whether the current browser is Safari. */\nexport const isSafari = !isServer && /safari/i.test(navigator.userAgent) && isWebkit;\n\n/**\n * We just check for the blink engine.\n */\nexport const isChromium = isBlink;\n\n/** Whether the application is being rendered in a Next.js environment. */\nexport const isNextjs = (): boolean => !!(globalThis as { next?: object }).next;\n","export function pageScrollDisabled(): boolean {\n  return document.body.hasAttribute('data-sbb-scroll-disabled');\n}\n\n/**\n * Handle the page scroll, allowing to disable/enable the window scroll avoiding a potential\n * content shift caused by the disappearance/appearance of the scrollbar.\n */\nexport class SbbScrollHandler {\n  private _position!: string;\n  private _overflow!: string;\n  private _marginInlineEnd!: string;\n\n  public disableScroll(): void {\n    if (pageScrollDisabled()) {\n      return;\n    }\n\n    // Save any pre-existing styles to reapply them to the body when enabling the scroll again.\n    this._position = document.body.style.position;\n    this._overflow = document.body.style.overflow;\n    this._marginInlineEnd = document.body.style.marginInlineEnd;\n\n    const scrollbarWidth = window.innerWidth - document.documentElement.clientWidth;\n\n    document.body.style.overflow = 'hidden';\n    document.body.style.position = 'relative';\n    document.body.style.marginInlineEnd = `${scrollbarWidth}px`;\n    document.body.style.setProperty('--sbb-scrollbar-width', `${scrollbarWidth}px`);\n\n    document.body.toggleAttribute('data-sbb-scroll-disabled', true);\n  }\n\n  public enableScroll(): void {\n    if (!pageScrollDisabled()) {\n      return;\n    }\n\n    // Revert body inline styles.\n    document.body.style.position = this._position || '';\n    document.body.style.overflow = this._overflow || '';\n    document.body.style.marginInlineEnd = this._marginInlineEnd || '';\n    document.body.style.setProperty('--sbb-scrollbar-width', '0');\n\n    document.body.removeAttribute('data-sbb-scroll-disabled');\n  }\n}\n","import type { LitElement } from 'lit';\n\nconst localNameCache = new Map<unknown, string>();\n\n/**\n * In SSR the local/tag name is not available on the class instance, but it is available\n * in the shim customElements registry.\n *\n * https://github.com/lit/lit/blob/main/packages/labs/ssr-dom-shim/src.ts (See CustomElementRegistryShim)\n *\n * Can be removed once https://github.com/lit/lit/pull/4553 is merged/released.\n */\nexport function getLocalName(element: LitElement): string {\n  if (localNameCache.has(element.constructor)) {\n    return localNameCache.get(element.constructor)!;\n  }\n\n  const definitions = // eslint-disable-next-line @typescript-eslint/naming-convention\n    (customElements as unknown as { __definitions: Map<string, { ctor: unknown }> }).__definitions;\n  for (const [key, value] of definitions) {\n    if (value.ctor === element.constructor) {\n      localNameCache.set(element.constructor, key);\n      return key;\n    }\n  }\n  throw new Error(`Given element ${element.constructor.name} has not been registered yet.`);\n}\n"],"names":[],"mappings":"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"}
145
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"dom.js","sources":["../../../../src/elements/core/dom/breakpoint.ts","../../../../src/elements/core/dom/find-referenced-element.ts","../../../../src/elements/core/dom/get-document-writing-mode.ts","../../../../src/elements/core/dom/host-context.ts","../../../../src/elements/core/dom/input-element.ts","../../../../src/elements/core/dom/set-or-remove-attribute.ts","../../../../src/elements/core/dom/platform.ts","../../../../src/elements/core/dom/scroll.ts","../../../../src/elements/core/dom/ssr.ts"],"sourcesContent":["import { isServer } from 'lit';\n\nexport const breakpoints = ['zero', 'micro', 'small', 'medium', 'wide', 'large', 'ultra'] as const;\nexport type Breakpoint = (typeof breakpoints)[number];\n\n/**\n * Checks whether the document matches a particular media query.\n * It will rely on the global CSS variables to determine the value of the breakpoints.\n *\n * @param from The breakpoint corresponding to the `min-width` value of the media query (optional).\n * @param to The breakpoint corresponding to the `max-width` value of the media query (optional).\n * @returns A boolean indicating whether the window matches the breakpoint.\n */\nexport function isBreakpoint(\n  from?: Breakpoint,\n  to?: Breakpoint,\n  properties?: { includeMaxBreakpoint: boolean },\n): boolean {\n  if (isServer) {\n    // TODO: Remove and decide case by case what should be done on consuming end\n    return false;\n  }\n\n  const computedStyle = getComputedStyle(document.documentElement);\n  const breakpointMin = from ? computedStyle.getPropertyValue(`--sbb-breakpoint-${from}-min`) : '';\n  const breakpointMax = to\n    ? `${\n        parseFloat(\n          computedStyle.getPropertyValue(\n            `--sbb-breakpoint-${to}-${properties?.includeMaxBreakpoint ? 'max' : 'min'}`,\n          ),\n        ) - (properties?.includeMaxBreakpoint ? 0 : 0.0625)\n      }rem`\n    : ''; // subtract 1px (0.0625rem) from the max-width breakpoint\n\n  const minWidth = breakpointMin && `(min-width: ${breakpointMin})`;\n  const maxWidth = breakpointMax && `(max-width: ${breakpointMax})`;\n  const and = breakpointMin && breakpointMax && ' and ';\n\n  return window.matchMedia(`${minWidth}${and}${maxWidth}`).matches;\n}\n","import { isServer } from 'lit';\n\n/**\n *  Check whether it's a string or an HTMLElement, if it's a string queries the element with the\n *  corresponding id.\n *  @param reference either the wanted id or the HTMLElement\n */\nexport function findReferencedElement<T extends HTMLElement = HTMLElement>(\n  reference: string | HTMLElement,\n): T | null {\n  if (isServer) {\n    return null;\n  } else if (typeof reference === 'string') {\n    return document.getElementById(reference) as T;\n  } else if (reference instanceof window.Element) {\n    return reference as T;\n  }\n  return null;\n}\n","import { isServer } from 'lit';\n\n/**\n * @deprecated use :dir() selector instead\n */\nexport const getDocumentWritingMode = (): string =>\n  (!isServer && document.documentElement.getAttribute('dir')) || 'ltr';\n","import { isServer } from 'lit';\n\n/**\n * Looks for the closest element matching the given selector starting from the given element.\n * Returns null, if none of the ancestor match.\n *\n * @param selector The selector to match ancestor against.\n * @param element The base element from which to start the search.\n * @returns The closest element matching the selector or null if none is found.\n */\nexport function hostContext(selector: string, element: Element): Element | null {\n  if (isServer) {\n    return null;\n  }\n  // Start with parent element in order to avoid finding element itself\n  element = element.parentElement ?? (element.getRootNode() as ShadowRoot).host;\n  while (element && (element as any) !== document && (element as any) !== window) {\n    const match = element.closest(selector);\n    if (match) {\n      return match;\n    }\n\n    element = (element.getRootNode() as ShadowRoot).host;\n  }\n\n  return null;\n}\n\n// A list of elements that should not allow another anchor or button element inside them.\n// Needs to be extended if additional elements fall into this category.\nexport const ACTION_ELEMENTS = 'a,button,sbb-teaser-hero,sbb-teaser';\n","import { findReferencedElement } from './find-referenced-element.js';\n\n/**\n * Given an element, returns the related input reference, if it exists respecting following priority.\n * 1. Input field in `sbb-form-field` (if trigger is undefiend)\n * 2. Input referenced by id (trigger is string)\n * 3. Input referenced directly (trigger is HTMLElement)\n * @param element The starting SbbDatepickerElement element.\n * @param trigger The id or the reference of the input.\n */\nexport function findInput(\n  element: HTMLElement,\n  trigger?: string | HTMLElement | null,\n): HTMLInputElement | null {\n  if (!trigger) {\n    const parent = element.closest?.('sbb-form-field');\n    return parent?.querySelector('input') as HTMLInputElement | null;\n  }\n\n  return findReferencedElement<HTMLInputElement>(trigger);\n}\n","/**\n * Set the attribute only if value is not 'false', otherwise remove attribute.\n * @param element The element that will have the attribute\n * @param attribute The attribute name\n * @param value The attribute value\n */\nexport function setOrRemoveAttribute(element: HTMLElement, attribute: string, value?: any): void {\n  if (!value) {\n    element.removeAttribute(attribute);\n  } else {\n    element.setAttribute(attribute, value);\n  }\n}\n","// This file is freely inspired by Angular Material solution.\n// Reference: https://github.com/angular/components/blob/main/src/cdk/platform/platform.ts\nimport { isServer } from 'lit';\n\n// Whether the current platform supports the V8 Break Iterator. The V8 check\n// is necessary to detect all Blink based browsers.\nlet hasV8BreakIterator: boolean;\n\n// We need a try/catch around the reference to `Intl`, because accessing it in some cases can\n// cause IE to throw. These cases are tied to particular versions of Windows and can happen if\n// the consumer is providing a polyfilled `Map`. See:\n// https://github.com/Microsoft/ChakraCore/issues/3189\n// https://github.com/angular/components/issues/15687\ntry {\n  hasV8BreakIterator = typeof Intl !== 'undefined' && (Intl as any).v8BreakIterator;\n} catch {\n  hasV8BreakIterator = false;\n}\n\n/** Whether the current browser is Microsoft Edge. */\nexport const isEdge = !isServer && /(edge)/i.test(navigator.userAgent);\n\n/** Whether the current rendering engine is Microsoft Trident. */\nexport const isTrident = !isServer && /(msie|trident)/i.test(navigator.userAgent);\n\n// EdgeHTML and Trident mock Blink specific things and need to be excluded from this check.\n/** Whether the current rendering engine is Blink. */\nexport const isBlink =\n  !isServer &&\n  !!((window as any).chrome || hasV8BreakIterator) &&\n  typeof CSS !== 'undefined' &&\n  !isEdge &&\n  !isTrident;\n\n// Webkit is part of the userAgent in EdgeHTML, Blink and Trident. Therefore, we need to\n// ensure that Webkit runs standalone and is not used as another engine's base.\n/** Whether the current rendering engine is WebKit. */\nexport const isWebkit =\n  !isServer && /AppleWebKit/i.test(navigator.userAgent) && !isBlink && !isEdge && !isTrident;\n\n/** Whether the current platform is Apple iOS. */\nexport const isIOS =\n  !isServer && /iPad|iPhone|iPod/.test(navigator.userAgent) && !('MSStream' in window);\n\n// It's difficult to detect the plain Gecko engine, because most of the browsers identify\n// them self as Gecko-like browsers and modify the userAgent's according to that.\n// Since we only cover one explicit Firefox case, we can simply check for Firefox\n// instead of having an unstable check for Gecko.\n/** Whether the current browser is Firefox. */\nexport const isFirefox = !isServer && /(firefox|minefield)/i.test(navigator.userAgent);\n\n/** Whether the current platform is Android. */\n// Trident on mobile adds the android platform to the userAgent to trick detections.\nexport const isAndroid = !isServer && /android/i.test(navigator.userAgent) && !isTrident;\n\n// Safari browsers will include the Safari keyword in their userAgent. Some browsers may fake\n// this and just place the Safari keyword in the userAgent. To be more safe about Safari every\n// Safari browser should also use Webkit as its layout engine.\n/** Whether the current browser is Safari. */\nexport const isSafari = !isServer && /safari/i.test(navigator.userAgent) && isWebkit;\n\n/**\n * We just check for the blink engine.\n */\nexport const isChromium = isBlink;\n\n/** Whether the application is being rendered in a Next.js environment. */\nexport const isNextjs = (): boolean => !!(globalThis as { next?: object }).next;\n","export function pageScrollDisabled(): boolean {\n  return document.body.hasAttribute('data-sbb-scroll-disabled');\n}\n\n/**\n * Handle the page scroll, allowing to disable/enable the window scroll avoiding a potential\n * content shift caused by the disappearance/appearance of the scrollbar.\n */\nexport class SbbScrollHandler {\n  private _position!: string;\n  private _overflow!: string;\n  private _marginInlineEnd!: string;\n\n  public disableScroll(): void {\n    if (pageScrollDisabled()) {\n      return;\n    }\n\n    // Save any pre-existing styles to reapply them to the body when enabling the scroll again.\n    this._position = document.body.style.position;\n    this._overflow = document.body.style.overflow;\n    this._marginInlineEnd = document.body.style.marginInlineEnd;\n\n    const scrollbarWidth = window.innerWidth - document.documentElement.clientWidth;\n\n    document.body.style.overflow = 'hidden';\n    document.body.style.position = 'relative';\n    document.body.style.marginInlineEnd = `${scrollbarWidth}px`;\n    document.body.style.setProperty('--sbb-scrollbar-width', `${scrollbarWidth}px`);\n\n    document.body.toggleAttribute('data-sbb-scroll-disabled', true);\n  }\n\n  public enableScroll(): void {\n    if (!pageScrollDisabled()) {\n      return;\n    }\n\n    // Revert body inline styles.\n    document.body.style.position = this._position || '';\n    document.body.style.overflow = this._overflow || '';\n    document.body.style.marginInlineEnd = this._marginInlineEnd || '';\n    document.body.style.setProperty('--sbb-scrollbar-width', '0');\n\n    document.body.removeAttribute('data-sbb-scroll-disabled');\n  }\n}\n","import type { LitElement } from 'lit';\n\nconst localNameCache = new Map<unknown, string>();\n\n/**\n * In SSR the local/tag name is not available on the class instance, but it is available\n * in the shim customElements registry.\n *\n * https://github.com/lit/lit/blob/main/packages/labs/ssr-dom-shim/src.ts (See CustomElementRegistryShim)\n *\n * Can be removed once https://github.com/lit/lit/pull/4553 is merged/released.\n *\n * @deprecated Will be removed with next major version\n */\nexport function getLocalName(element: LitElement): string {\n  if (localNameCache.has(element.constructor)) {\n    return localNameCache.get(element.constructor)!;\n  }\n\n  const definitions = // eslint-disable-next-line @typescript-eslint/naming-convention\n    (customElements as unknown as { __definitions: Map<string, { ctor: unknown }> }).__definitions;\n  for (const [key, value] of definitions) {\n    if (value.ctor === element.constructor) {\n      localNameCache.set(element.constructor, key);\n      return key;\n    }\n  }\n  throw new Error(`Given element ${element.constructor.name} has not been registered yet.`);\n}\n"],"names":[],"mappings":";AAEa,MAAA,cAAc,CAAC,QAAQ,SAAS,SAAS,UAAU,QAAQ,SAAS,OAAO;AAWxE,SAAA,aACd,MACA,IACA,YACS;AACT,MAAI,UAAU;AAEL,WAAA;AAAA,EACT;AAEM,QAAA,gBAAgB,iBAAiB,SAAS,eAAe;AAC/D,QAAM,gBAAgB,OAAO,cAAc,iBAAiB,oBAAoB,IAAI,MAAM,IAAI;AACxF,QAAA,gBAAgB,KAClB,GACE;AAAA,IACE,cAAc;AAAA,MACZ,oBAAoB,EAAE,KAAI,yCAAY,wBAAuB,QAAQ,KAAK;AAAA,IAC5E;AAAA,EAAA,MACG,yCAAY,wBAAuB,IAAI,OAC9C,QACA;AAEE,QAAA,WAAW,iBAAiB,eAAe,aAAa;AACxD,QAAA,WAAW,iBAAiB,eAAe,aAAa;AACxD,QAAA,MAAM,iBAAiB,iBAAiB;AAEvC,SAAA,OAAO,WAAW,GAAG,QAAQ,GAAG,GAAG,GAAG,QAAQ,EAAE,EAAE;AAC3D;ACjCO,SAAS,sBACd,WACU;AACV,MAAI,UAAU;AACL,WAAA;AAAA,EAAA,WACE,OAAO,cAAc,UAAU;AACjC,WAAA,SAAS,eAAe,SAAS;AAAA,EAAA,WAC/B,qBAAqB,OAAO,SAAS;AACvC,WAAA;AAAA,EACT;AACO,SAAA;AACT;ACba,MAAA,yBAAyB,MACnC,CAAC,YAAY,SAAS,gBAAgB,aAAa,KAAK,KAAM;ACIjD,SAAA,YAAY,UAAkB,SAAkC;AAC9E,MAAI,UAAU;AACL,WAAA;AAAA,EACT;AAEA,YAAU,QAAQ,iBAAkB,QAAQ,YAAA,EAA6B;AACzE,SAAO,WAAY,YAAoB,YAAa,YAAoB,QAAQ;AACxE,UAAA,QAAQ,QAAQ,QAAQ,QAAQ;AACtC,QAAI,OAAO;AACF,aAAA;AAAA,IACT;AAEW,cAAA,QAAQ,YAA6B,EAAA;AAAA,EAClD;AAEO,SAAA;AACT;AAIO,MAAM,kBAAkB;ACpBf,SAAA,UACd,SACA,SACyB;;AACzB,MAAI,CAAC,SAAS;AACN,UAAA,UAAS,aAAQ,YAAR,iCAAkB;AAC1B,WAAA,iCAAQ,cAAc;AAAA,EAC/B;AAEA,SAAO,sBAAwC,OAAO;AACxD;ACdgB,SAAA,qBAAqB,SAAsB,WAAmB,OAAmB;AAC/F,MAAI,CAAC,OAAO;AACV,YAAQ,gBAAgB,SAAS;AAAA,EAAA,OAC5B;AACG,YAAA,aAAa,WAAW,KAAK;AAAA,EACvC;AACF;ACNA,IAAI;AAOJ,IAAI;AACmB,uBAAA,OAAO,SAAS,eAAgB,KAAa;AACpE,QAAQ;AACe,uBAAA;AACvB;AAGO,MAAM,SAAS,CAAC,YAAY,UAAU,KAAK,UAAU,SAAS;AAG9D,MAAM,YAAY,CAAC,YAAY,kBAAkB,KAAK,UAAU,SAAS;AAIzE,MAAM,UACX,CAAC,YACD,CAAC,EAAG,OAAe,UAAU,uBAC7B,OAAO,QAAQ,eACf,CAAC,UACD,CAAC;AAKI,MAAM,WACX,CAAC,YAAY,eAAe,KAAK,UAAU,SAAS,KAAK,CAAC,WAAW,CAAC,UAAU,CAAC;AAGtE,MAAA,QACX,CAAC,YAAY,mBAAmB,KAAK,UAAU,SAAS,KAAK,EAAE,cAAc;AAOxE,MAAM,YAAY,CAAC,YAAY,uBAAuB,KAAK,UAAU,SAAS;AAIxE,MAAA,YAAY,CAAC,YAAY,WAAW,KAAK,UAAU,SAAS,KAAK,CAAC;AAMxE,MAAM,WAAW,CAAC,YAAY,UAAU,KAAK,UAAU,SAAS,KAAK;AAKrE,MAAM,aAAa;AAGnB,MAAM,WAAW,MAAe,CAAC,CAAE,WAAiC;ACnEpE,SAAS,qBAA8B;AACrC,SAAA,SAAS,KAAK,aAAa,0BAA0B;AAC9D;AAMO,MAAM,iBAAiB;AAAA,EAKrB,gBAAsB;AAC3B,QAAI,sBAAsB;AACxB;AAAA,IACF;AAGK,SAAA,YAAY,SAAS,KAAK,MAAM;AAChC,SAAA,YAAY,SAAS,KAAK,MAAM;AAChC,SAAA,mBAAmB,SAAS,KAAK,MAAM;AAE5C,UAAM,iBAAiB,OAAO,aAAa,SAAS,gBAAgB;AAE3D,aAAA,KAAK,MAAM,WAAW;AACtB,aAAA,KAAK,MAAM,WAAW;AAC/B,aAAS,KAAK,MAAM,kBAAkB,GAAG,cAAc;AACvD,aAAS,KAAK,MAAM,YAAY,yBAAyB,GAAG,cAAc,IAAI;AAErE,aAAA,KAAK,gBAAgB,4BAA4B,IAAI;AAAA,EAChE;AAAA,EAEO,eAAqB;AACtB,QAAA,CAAC,sBAAsB;AACzB;AAAA,IACF;AAGA,aAAS,KAAK,MAAM,WAAW,KAAK,aAAa;AACjD,aAAS,KAAK,MAAM,WAAW,KAAK,aAAa;AACjD,aAAS,KAAK,MAAM,kBAAkB,KAAK,oBAAoB;AAC/D,aAAS,KAAK,MAAM,YAAY,yBAAyB,GAAG;AAEnD,aAAA,KAAK,gBAAgB,0BAA0B;AAAA,EAC1D;AACF;AC5CA,MAAM,qCAAqB;AAYpB,SAAS,aAAa,SAA6B;AACxD,MAAI,eAAe,IAAI,QAAQ,WAAW,GAAG;AACpC,WAAA,eAAe,IAAI,QAAQ,WAAW;AAAA,EAC/C;AAEM,QAAA;AAAA;AAAA,IACH,eAAgF;AAAA;AACnF,aAAW,CAAC,KAAK,KAAK,KAAK,aAAa;AAClC,QAAA,MAAM,SAAS,QAAQ,aAAa;AACvB,qBAAA,IAAI,QAAQ,aAAa,GAAG;AACpC,aAAA;AAAA,IACT;AAAA,EACF;AACA,QAAM,IAAI,MAAM,iBAAiB,QAAQ,YAAY,IAAI,+BAA+B;AAC1F;"}
@@ -1 +1 @@
1
- {"version":3,"file":"named-slot-list-mixin.d.ts","sourceRoot":"","sources":["../../../../../src/elements/core/mixins/named-slot-list-mixin.ts"],"names":[],"mappings":"AAAA,OAAO,EAAQ,KAAK,UAAU,EAAW,KAAK,cAAc,EAAE,MAAM,KAAK,CAAC;AAK1E,OAAO,KAAK,EAAE,mBAAmB,EAAE,MAAM,kBAAkB,CAAC;AAC5D,OAAO,EAAqB,KAAK,qBAAqB,EAAE,MAAM,sBAAsB,CAAC;AAErF,OAAO,6BAA6B,CAAC;AAKrC,MAAM,MAAM,sBAAsB,GAAG;IACnC,IAAI,EAAE,MAAM,CAAC;IACb,UAAU,EAAE,OAAO,CAAC;CACrB,CAAC;AAEF;;;;;;;;;;GAUG;AACH,MAAM,MAAM,gBAAgB,CAC1B,CAAC,SAAS,yBAAyB,CAAC,CAAC,CAAC,EACtC,CAAC,SAAS,WAAW,GAAG,WAAW,IACjC,CAAC,GAAG;IAAE,YAAY,EAAE,CAAC,EAAE,CAAA;CAAE,CAAC;AAE9B,MAAM,CAAC,OAAO,CAAC,QAAQ,OAAO,yBAAyB,CACrD,CAAC,SAAS,WAAW,CACrB,SAAQ,qBAAqB;IAC7B,SAAS,CAAC,QAAQ,CAAC,QAAQ,CAAC,mBAAmB,EAAE,MAAM,EAAE,CAAC;IACjD,SAAS,CAAC,YAAY,EAAE,CAAC,EAAE,CAAC;IACrC,SAAS,CAAC,UAAU,CAClB,UAAU,CAAC,EAAE;QACX,KAAK,CAAC,EAAE,MAAM,CAAC;QACf,SAAS,CAAC,EAAE,MAAM,CAAC;QACnB,cAAc,CAAC,EAAE,MAAM,CAAC;KACzB,EACD,kBAAkB,CAAC,EAAE;QACnB,mBAAmB,CAAC,EAAE,MAAM,EAAE,CAAC;KAChC,GACA,cAAc;IACjB,SAAS,CAAC,eAAe,IAAI,sBAAsB,EAAE;IACrD,SAAS,CAAC,gBAAgB,IAAI,cAAc;CAC7C;AAGD,eAAO,MAAM,qBAAqB,GAChC,CAAC,SAAS,WAAW,EACrB,CAAC,SAAS,mBAAmB,CAAC,UAAU,CAAC,cAE7B,CAAC,KACZ,mBAAmB,CAAC,yBAAyB,CAAC,CAAC,CAAC,CAAC,GAAG,CAsItD,CAAC"}
1
+ {"version":3,"file":"named-slot-list-mixin.d.ts","sourceRoot":"","sources":["../../../../../src/elements/core/mixins/named-slot-list-mixin.ts"],"names":[],"mappings":"AAAA,OAAO,EAAQ,KAAK,UAAU,EAAW,KAAK,cAAc,EAAE,MAAM,KAAK,CAAC;AAG1E,OAAO,KAAK,EAAE,mBAAmB,EAAE,MAAM,kBAAkB,CAAC;AAC5D,OAAO,EAAqB,KAAK,qBAAqB,EAAE,MAAM,sBAAsB,CAAC;AAErF,OAAO,6BAA6B,CAAC;AAKrC,MAAM,MAAM,sBAAsB,GAAG;IACnC,IAAI,EAAE,MAAM,CAAC;IACb,UAAU,EAAE,OAAO,CAAC;CACrB,CAAC;AAEF;;;;;;;;;;GAUG;AACH,MAAM,MAAM,gBAAgB,CAC1B,CAAC,SAAS,yBAAyB,CAAC,CAAC,CAAC,EACtC,CAAC,SAAS,WAAW,GAAG,WAAW,IACjC,CAAC,GAAG;IAAE,YAAY,EAAE,CAAC,EAAE,CAAA;CAAE,CAAC;AAE9B,MAAM,CAAC,OAAO,CAAC,QAAQ,OAAO,yBAAyB,CACrD,CAAC,SAAS,WAAW,CACrB,SAAQ,qBAAqB;IAC7B,SAAS,CAAC,QAAQ,CAAC,QAAQ,CAAC,mBAAmB,EAAE,MAAM,EAAE,CAAC;IACjD,SAAS,CAAC,YAAY,EAAE,CAAC,EAAE,CAAC;IACrC,SAAS,CAAC,UAAU,CAClB,UAAU,CAAC,EAAE;QACX,KAAK,CAAC,EAAE,MAAM,CAAC;QACf,SAAS,CAAC,EAAE,MAAM,CAAC;QACnB,cAAc,CAAC,EAAE,MAAM,CAAC;KACzB,EACD,kBAAkB,CAAC,EAAE;QACnB,mBAAmB,CAAC,EAAE,MAAM,EAAE,CAAC;KAChC,GACA,cAAc;IACjB,SAAS,CAAC,eAAe,IAAI,sBAAsB,EAAE;IACrD,SAAS,CAAC,gBAAgB,IAAI,cAAc;CAC7C;AAGD,eAAO,MAAM,qBAAqB,GAChC,CAAC,SAAS,WAAW,EACrB,CAAC,SAAS,mBAAmB,CAAC,UAAU,CAAC,cAE7B,CAAC,KACZ,mBAAmB,CAAC,yBAAyB,CAAC,CAAC,CAAC,CAAC,GAAG,CAsItD,CAAC"}