@sbb-esta/lyne-elements 1.8.0 → 1.9.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (43) hide show
  1. package/clock.js +5 -5
  2. package/container/container.js +14 -14
  3. package/core/dom/platform.d.ts +2 -3
  4. package/core/dom/platform.d.ts.map +1 -1
  5. package/core/dom.js +38 -40
  6. package/core/styles/core.scss +6 -0
  7. package/core/styles/mixins/lists.scss +11 -1
  8. package/core/testing/wait-for-image-ready.d.ts.map +1 -1
  9. package/core/testing.js +61 -54
  10. package/core.css +5 -0
  11. package/custom-elements.json +123 -1
  12. package/datepicker/datepicker-toggle.js +3 -3
  13. package/development/clock.js +1 -1
  14. package/development/container/container.js +1 -1
  15. package/development/core/dom/platform.d.ts +2 -3
  16. package/development/core/dom/platform.d.ts.map +1 -1
  17. package/development/core/dom.js +4 -5
  18. package/development/core/testing/wait-for-image-ready.d.ts.map +1 -1
  19. package/development/core/testing.js +16 -2
  20. package/development/datepicker/datepicker-toggle.js +3 -3
  21. package/development/overlay.js +2 -9
  22. package/development/select/select.d.ts +3 -1
  23. package/development/select/select.d.ts.map +1 -1
  24. package/development/select.js +20 -6
  25. package/development/stepper/step-label.js +20 -2
  26. package/development/stepper/stepper/stepper.d.ts +3 -0
  27. package/development/stepper/stepper/stepper.d.ts.map +1 -1
  28. package/development/stepper/stepper.js +16 -1
  29. package/development/toggle/toggle-option/toggle-option.d.ts.map +1 -1
  30. package/development/toggle/toggle-option.js +2 -1
  31. package/lists.css +9 -1
  32. package/overlay.js +1 -1
  33. package/package.json +1 -1
  34. package/select/select.d.ts +3 -1
  35. package/select/select.d.ts.map +1 -1
  36. package/select.js +61 -50
  37. package/standard-theme.css +14 -1
  38. package/stepper/step-label.js +10 -10
  39. package/stepper/stepper/stepper.d.ts +3 -0
  40. package/stepper/stepper/stepper.d.ts.map +1 -1
  41. package/stepper/stepper.js +34 -25
  42. package/toggle/toggle-option/toggle-option.d.ts.map +1 -1
  43. package/toggle/toggle-option.js +1 -0
@@ -9981,6 +9981,42 @@
9981
9981
  }
9982
9982
  }
9983
9983
  },
9984
+ {
9985
+ "kind": "method",
9986
+ "name": "_spreadDeferredDisplayValue",
9987
+ "privacy": "private",
9988
+ "return": {
9989
+ "type": {
9990
+ "text": "(TemplateResult | Promise<TemplateResult>)[]"
9991
+ }
9992
+ },
9993
+ "parameters": [
9994
+ {
9995
+ "name": "placeholder",
9996
+ "type": {
9997
+ "text": "TemplateResult"
9998
+ }
9999
+ }
10000
+ ]
10001
+ },
10002
+ {
10003
+ "kind": "method",
10004
+ "name": "_deferredDisplayValue",
10005
+ "privacy": "private",
10006
+ "return": {
10007
+ "type": {
10008
+ "text": "Promise<TemplateResult>"
10009
+ }
10010
+ },
10011
+ "parameters": [
10012
+ {
10013
+ "name": "placeholder",
10014
+ "type": {
10015
+ "text": "TemplateResult"
10016
+ }
10017
+ }
10018
+ ]
10019
+ },
9984
10020
  {
9985
10021
  "kind": "field",
9986
10022
  "name": "_updatePromise",
@@ -10101,6 +10137,56 @@
10101
10137
  "module": "core/mixins.js"
10102
10138
  }
10103
10139
  },
10140
+ {
10141
+ "kind": "field",
10142
+ "name": "_hydrationRequired",
10143
+ "type": {
10144
+ "text": "boolean"
10145
+ },
10146
+ "privacy": "private",
10147
+ "default": "!!this.shadowRoot",
10148
+ "inheritedFrom": {
10149
+ "name": "SbbHydrationMixin",
10150
+ "module": "core/mixins.js"
10151
+ }
10152
+ },
10153
+ {
10154
+ "kind": "field",
10155
+ "name": "_hydrationComplete",
10156
+ "privacy": "private",
10157
+ "default": "new Promise<boolean>( (resolve) => (this._resolveHydration = resolve), )",
10158
+ "inheritedFrom": {
10159
+ "name": "SbbHydrationMixin",
10160
+ "module": "core/mixins.js"
10161
+ }
10162
+ },
10163
+ {
10164
+ "kind": "field",
10165
+ "name": "_resolveHydration",
10166
+ "type": {
10167
+ "text": "(hydrationRequired: boolean) => void"
10168
+ },
10169
+ "privacy": "private",
10170
+ "inheritedFrom": {
10171
+ "name": "SbbHydrationMixin",
10172
+ "module": "core/mixins.js"
10173
+ }
10174
+ },
10175
+ {
10176
+ "kind": "field",
10177
+ "name": "hydrationRequired",
10178
+ "type": {
10179
+ "text": "boolean"
10180
+ },
10181
+ "privacy": "protected",
10182
+ "description": "Returns whether hydration is required and not completed.",
10183
+ "readonly": true,
10184
+ "inheritedFrom": {
10185
+ "name": "SbbHydrationMixin",
10186
+ "module": "core/mixins.js"
10187
+ },
10188
+ "default": "!!this.shadowRoot"
10189
+ },
10104
10190
  {
10105
10191
  "kind": "field",
10106
10192
  "name": "state",
@@ -10334,6 +10420,10 @@
10334
10420
  {
10335
10421
  "name": "SbbNegativeMixin",
10336
10422
  "module": "core/mixins.js"
10423
+ },
10424
+ {
10425
+ "name": "SbbHydrationMixin",
10426
+ "module": "core/mixins.js"
10337
10427
  }
10338
10428
  ],
10339
10429
  "superclass": {
@@ -30160,7 +30250,8 @@
30160
30250
  {
30161
30251
  "kind": "variable",
30162
30252
  "name": "isChromium",
30163
- "description": "Note: `userAgentData` is still experimental. If possible, avoid to use it.\nhttps://developer.mozilla.org/en-US/docs/Web/API/Navigator/userAgentData"
30253
+ "default": "isBlink",
30254
+ "description": "We just check for the blink engine."
30164
30255
  },
30165
30256
  {
30166
30257
  "kind": "function",
@@ -54905,6 +54996,18 @@
54905
54996
  "attribute": "orientation",
54906
54997
  "reflects": true
54907
54998
  },
54999
+ {
55000
+ "kind": "field",
55001
+ "name": "size",
55002
+ "type": {
55003
+ "text": "'s' | 'm'"
55004
+ },
55005
+ "privacy": "public",
55006
+ "default": "'m'",
55007
+ "description": "Size variant, either s or m.",
55008
+ "attribute": "size",
55009
+ "reflects": true
55010
+ },
54908
55011
  {
54909
55012
  "kind": "field",
54910
55013
  "name": "selected",
@@ -55106,6 +55209,16 @@
55106
55209
  }
55107
55210
  }
55108
55211
  },
55212
+ {
55213
+ "kind": "method",
55214
+ "name": "_proxySize",
55215
+ "privacy": "private",
55216
+ "return": {
55217
+ "type": {
55218
+ "text": "void"
55219
+ }
55220
+ }
55221
+ },
55109
55222
  {
55110
55223
  "kind": "method",
55111
55224
  "name": "_handleKeyDown",
@@ -55202,6 +55315,15 @@
55202
55315
  "description": "Steps orientation, either horizontal or vertical.",
55203
55316
  "fieldName": "orientation"
55204
55317
  },
55318
+ {
55319
+ "name": "size",
55320
+ "type": {
55321
+ "text": "'s' | 'm'"
55322
+ },
55323
+ "default": "'m'",
55324
+ "description": "Size variant, either s or m.",
55325
+ "fieldName": "size"
55326
+ },
55205
55327
  {
55206
55328
  "name": "selected-index",
55207
55329
  "description": "The currently selected step index.",
@@ -10,8 +10,8 @@ import { getDatePicker as y, datepickerControlRegisteredEventFactory as $ } from
10
10
  import "../calendar.js";
11
11
  import "../popover.js";
12
12
  const x = p`*,:before,:after{box-sizing:border-box}:host{--sbb-datepicker-control-radius: var(--sbb-border-radius-4x);--sbb-datepicker-control-color: var(--sbb-color-black);display:inline-flex}sbb-popover-trigger{color:var(--sbb-datepicker-control-color);height:unset}sbb-icon{color:var(--sbb-color-graphite)}`;
13
- var D = Object.defineProperty, A = Object.getOwnPropertyDescriptor, l = (e, t, i, n) => {
14
- for (var a = n > 1 ? void 0 : n ? A(t, i) : t, o = e.length - 1, r; o >= 0; o--)
13
+ var D = Object.defineProperty, O = Object.getOwnPropertyDescriptor, l = (e, t, i, n) => {
14
+ for (var a = n > 1 ? void 0 : n ? O(t, i) : t, o = e.length - 1, r; o >= 0; o--)
15
15
  (r = e[o]) && (a = (n ? r(t, i, a) : r(a)) || a);
16
16
  return n && a && D(t, i, a), a;
17
17
  };
@@ -78,7 +78,7 @@ let s = class extends C(
78
78
  !e || !t || (e.wide = t.wide, e.now = this._nowOrUndefined(), e.dateFilter = t.dateFilter);
79
79
  }
80
80
  _datePickerChanged(e) {
81
- this._datePickerElement = e.target, this._calendarElement && this._datePickerElement.valueAsDate && (this._calendarElement.selected = this._datePickerElement.valueAsDate);
81
+ this._datePickerElement = e.target, this._calendarElement && (this._calendarElement.selected = this._datePickerElement.valueAsDate || void 0);
82
82
  }
83
83
  _assignCalendar(e) {
84
84
  var t;
@@ -63,7 +63,7 @@ const style = css`/**
63
63
  }
64
64
 
65
65
  .sbb-clock__hand-minutes {
66
- transition: transform 0.2s cubic-bezier(0.4, 2.08, 0.55, 0.44);
66
+ transition: transform var(--sbb-disable-animation-zero-time, 0.2s) cubic-bezier(0.4, 2.08, 0.55, 0.44);
67
67
  }
68
68
  @supports not (aspect-ratio: 1/1) {
69
69
  .sbb-clock__hand-minutes {
@@ -84,7 +84,7 @@ const style = css`@charset "UTF-8";
84
84
  position: relative;
85
85
  }
86
86
 
87
- .sbb-container__content {
87
+ :host([data-slot-names~=image]) .sbb-container__content {
88
88
  position: relative;
89
89
  }
90
90
 
@@ -15,10 +15,9 @@ export declare const isAndroid: boolean;
15
15
  /** Whether the current browser is Safari. */
16
16
  export declare const isSafari: boolean;
17
17
  /**
18
- * Note: `userAgentData` is still experimental. If possible, avoid to use it.
19
- * https://developer.mozilla.org/en-US/docs/Web/API/Navigator/userAgentData
18
+ * We just check for the blink engine.
20
19
  */
21
- export declare const isChromium: any;
20
+ export declare const isChromium: boolean;
22
21
  /** Whether the application is being rendered in a Next.js environment. */
23
22
  export declare const isNextjs: () => boolean;
24
23
  //# sourceMappingURL=platform.d.ts.map
@@ -1 +1 @@
1
- {"version":3,"file":"platform.d.ts","sourceRoot":"","sources":["../../../../../src/elements/core/dom/platform.ts"],"names":[],"mappings":"AAmBA,qDAAqD;AACrD,eAAO,MAAM,MAAM,SAAmD,CAAC;AAEvE,iEAAiE;AACjE,eAAO,MAAM,SAAS,SAA2D,CAAC;AAGlF,qDAAqD;AACrD,eAAO,MAAM,OAAO,SAKR,CAAC;AAIb,sDAAsD;AACtD,eAAO,MAAM,QAAQ,SACuE,CAAC;AAE7F,iDAAiD;AACjD,eAAO,MAAM,KAAK,SACoE,CAAC;AAMvF,8CAA8C;AAC9C,eAAO,MAAM,SAAS,SAAgE,CAAC;AAEvF,+CAA+C;AAE/C,eAAO,MAAM,SAAS,SAAkE,CAAC;AAKzF,6CAA6C;AAC7C,eAAO,MAAM,QAAQ,SAA+D,CAAC;AAErF;;;GAGG;AACH,eAAO,MAAM,UAAU,KAEkE,CAAC;AAE1F,0EAA0E;AAC1E,eAAO,MAAM,QAAQ,QAAO,OAAmD,CAAC"}
1
+ {"version":3,"file":"platform.d.ts","sourceRoot":"","sources":["../../../../../src/elements/core/dom/platform.ts"],"names":[],"mappings":"AAmBA,qDAAqD;AACrD,eAAO,MAAM,MAAM,SAAmD,CAAC;AAEvE,iEAAiE;AACjE,eAAO,MAAM,SAAS,SAA2D,CAAC;AAGlF,qDAAqD;AACrD,eAAO,MAAM,OAAO,SAKR,CAAC;AAIb,sDAAsD;AACtD,eAAO,MAAM,QAAQ,SACuE,CAAC;AAE7F,iDAAiD;AACjD,eAAO,MAAM,KAAK,SACoE,CAAC;AAMvF,8CAA8C;AAC9C,eAAO,MAAM,SAAS,SAAgE,CAAC;AAEvF,+CAA+C;AAE/C,eAAO,MAAM,SAAS,SAAkE,CAAC;AAKzF,6CAA6C;AAC7C,eAAO,MAAM,QAAQ,SAA+D,CAAC;AAErF;;GAEG;AACH,eAAO,MAAM,UAAU,SAAU,CAAC;AAElC,0EAA0E;AAC1E,eAAO,MAAM,QAAQ,QAAO,OAAmD,CAAC"}
@@ -1,4 +1,3 @@
1
- var _a, _b;
2
1
  import { isServer } from "lit";
3
2
  const breakpoints = ["zero", "micro", "small", "medium", "wide", "large", "ultra"];
4
3
  function isBreakpoint(from, to, properties) {
@@ -44,9 +43,9 @@ function hostContext(selector, element) {
44
43
  }
45
44
  const ACTION_ELEMENTS = "a,button,sbb-teaser-hero,sbb-teaser";
46
45
  function findInput(element, trigger) {
47
- var _a2;
46
+ var _a;
48
47
  if (!trigger) {
49
- const parent = (_a2 = element.closest) == null ? void 0 : _a2.call(element, "sbb-form-field");
48
+ const parent = (_a = element.closest) == null ? void 0 : _a.call(element, "sbb-form-field");
50
49
  return parent == null ? void 0 : parent.querySelector("input");
51
50
  }
52
51
  return findReferencedElement(trigger);
@@ -72,7 +71,7 @@ const isIOS = !isServer && /iPad|iPhone|iPod/.test(navigator.userAgent) && !("MS
72
71
  const isFirefox = !isServer && /(firefox|minefield)/i.test(navigator.userAgent);
73
72
  const isAndroid = !isServer && /android/i.test(navigator.userAgent) && !isTrident;
74
73
  const isSafari = !isServer && /safari/i.test(navigator.userAgent) && isWebkit;
75
- const isChromium = !isServer && ((_b = (_a = navigator.userAgentData) == null ? void 0 : _a.brands) == null ? void 0 : _b.some((data) => data.brand == "Chromium"));
74
+ const isChromium = isBlink;
76
75
  const isNextjs = () => !!globalThis.next;
77
76
  function pageScrollDisabled() {
78
77
  return document.body.hasAttribute("data-sbb-scroll-disabled");
@@ -143,4 +142,4 @@ export {
143
142
  pageScrollDisabled,
144
143
  setOrRemoveAttribute
145
144
  };
146
- //# 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\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 * Note: `userAgentData` is still experimental. If possible, avoid to use it.\n * https://developer.mozilla.org/en-US/docs/Web/API/Navigator/userAgentData\n */\nexport const isChromium =\n  !isServer &&\n  (navigator as any).userAgentData?.brands?.some((data: any) => data.brand == 'Chromium');\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":["_a"],"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\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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"}
@@ -1 +1 @@
1
- {"version":3,"file":"wait-for-image-ready.d.ts","sourceRoot":"","sources":["../../../../../src/elements/core/testing/wait-for-image-ready.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,eAAe,EAAE,MAAM,gBAAgB,CAAC;AAEtD,wBAAsB,iBAAiB,CACrC,OAAO,EAAE,gBAAgB,GAAG,eAAe,EAC3C,qBAAqB,SAAW,GAC/B,OAAO,CAAC,IAAI,CAAC,CAcf"}
1
+ {"version":3,"file":"wait-for-image-ready.d.ts","sourceRoot":"","sources":["../../../../../src/elements/core/testing/wait-for-image-ready.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,eAAe,EAAE,MAAM,gBAAgB,CAAC;AAGtD,wBAAsB,iBAAiB,CACrC,OAAO,EAAE,gBAAgB,GAAG,eAAe,EAC3C,qBAAqB,SAAW,GAC/B,OAAO,CAAC,IAAI,CAAC,CAkCf"}
@@ -1,3 +1,4 @@
1
+ import { isSafari } from "./dom.js";
1
2
  class EventSpy {
2
3
  constructor(_event, _target = null) {
3
4
  this._event = _event;
@@ -76,18 +77,31 @@ function waitForEvent(element, eventName, timeout = 1e3) {
76
77
  });
77
78
  }
78
79
  async function waitForImageReady(element, timeoutInMilliseconds = 2 * 1e3) {
80
+ var _a;
81
+ const imgElement = element.localName === "sbb-image" ? ((_a = element.shadowRoot) == null ? void 0 : _a.querySelector(".sbb-image__img")) ?? null : element;
82
+ if (!imgElement) {
83
+ throw new Error("img tag not found");
84
+ }
79
85
  if (!element.complete) {
80
86
  await new Promise((resolve, reject) => {
81
87
  const timeout = setTimeout(() => reject("image loading timeout"), timeoutInMilliseconds);
82
88
  element.addEventListener("load", () => {
83
89
  clearTimeout(timeout);
84
- resolve();
90
+ imgElement.decode().then(() => {
91
+ if (isSafari && element.localName === "sbb-image") {
92
+ setTimeout(resolve, 100);
93
+ } else {
94
+ resolve();
95
+ }
96
+ });
85
97
  });
86
98
  element.addEventListener("error", () => {
87
99
  clearTimeout(timeout);
88
100
  reject("image error");
89
101
  });
90
102
  });
103
+ } else {
104
+ await imgElement.decode();
91
105
  }
92
106
  }
93
107
  const isReactiveElement = (element) => {
@@ -117,4 +131,4 @@ export {
117
131
  waitForImageReady,
118
132
  waitForLitRender
119
133
  };
120
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"testing.js","sources":["../../../../src/elements/core/testing/event-spy.ts","../../../../src/elements/core/testing/mocha-extensions.ts","../../../../src/elements/core/testing/scroll.ts","../../../../src/elements/core/testing/wait-for-condition.ts","../../../../src/elements/core/testing/wait-for-event.ts","../../../../src/elements/core/testing/wait-for-image-ready.ts","../../../../src/elements/core/testing/wait-for-render.ts"],"sourcesContent":["/**\n * This class exists to facilitate the test migration from stencil to lit.\n * It mimics the API that stencil provided to test events.\n *\n * TODO: Document me\n */\nexport class EventSpy<T extends Event> {\n  private _count: number = 0;\n  public get count(): number {\n    return this._count;\n  }\n\n  private _events: T[] = [];\n  public get events(): T[] {\n    return this._events;\n  }\n\n  public get firstEvent(): T | null {\n    return this.events.length ? this.events[0] : null;\n  }\n\n  public get lastEvent(): T | null {\n    return this.events.length ? this.events[this.events.length - 1] : null;\n  }\n\n  public constructor(\n    private _event: string,\n    private readonly _target: Node | null = null,\n  ) {\n    if (!this._target) {\n      this._target = document;\n    }\n    this._listenForEvent();\n  }\n\n  private _listenForEvent(): void {\n    this._target?.addEventListener(this._event, (ev) => {\n      this._events.push(ev as T);\n      this._count++;\n    });\n  }\n}\n","import type { Func, Suite, Test } from 'mocha';\n\n/**\n * Skip the `describe` if the condition is not met\n */\nexport const describeIf = (\n  condition: boolean,\n  title: string,\n  fn: (this: Suite) => void,\n): Suite | void => {\n  if (condition) {\n    return Mocha.describe(title, fn);\n  }\n};\n\n/**\n * Skip the `test` if the condition is not met\n */\nexport const testIf = (condition: boolean, title: string, fn?: Func): Test | void => {\n  if (condition) {\n    return Mocha.test(title, fn);\n  }\n};\n","/**\n * Can be used in tests where scroll events are not triggered by default.\n * @param options ScrollToOptions\n */\nexport function mockScrollTo(options: ScrollToOptions): void {\n  window.scrollTo(options);\n\n  // We need to manually dispatching the scroll as the scrollTo\n  // does not fire a scroll event in test env.\n  document.dispatchEvent(new Event('scroll'));\n}\n","export async function waitForCondition(\n  condition: () => boolean | Promise<boolean>,\n  interval = 30,\n  timeoutInMilliseconds = 2 * 1000,\n): Promise<boolean> {\n  const start = new Date();\n  let counter = 0;\n  function checkCondition(): boolean | Promise<boolean> {\n    counter++;\n    return condition();\n  }\n  while (\n    !(await checkCondition()) &&\n    start.getTime() + timeoutInMilliseconds >= new Date().getTime()\n  ) {\n    await new Promise((resolve) => setTimeout(resolve, interval));\n  }\n  if (start.getTime() + timeoutInMilliseconds < new Date().getTime()) {\n    return Promise.reject(\n      `waitForCondition timeout: ${condition.toString()}, attempts: ${counter}, start: ${start.getTime()}, end: ${new Date().getTime()}`,\n    );\n  }\n  return Promise.resolve(true);\n}\n","export function waitForEvent(\n  element: HTMLElement,\n  eventName: string,\n  timeout = 1000,\n): Promise<void> {\n  return new Promise<void>((resolve, reject) => {\n    const signal = AbortSignal.timeout(timeout);\n    const timeoutReached = (): void => reject(`Timeout of ${timeout} reached`);\n    signal.addEventListener('abort', timeoutReached);\n    element.addEventListener(\n      eventName,\n      () => {\n        signal.removeEventListener('abort', timeoutReached);\n        resolve();\n      },\n      { passive: true, signal },\n    );\n  });\n}\n","import type { SbbImageElement } from '../../image.js';\n\nexport async function waitForImageReady(\n  element: HTMLImageElement | SbbImageElement,\n  timeoutInMilliseconds = 2 * 1000,\n): Promise<void> {\n  if (!element.complete) {\n    await new Promise<void>((resolve, reject) => {\n      const timeout = setTimeout(() => reject('image loading timeout'), timeoutInMilliseconds);\n      element.addEventListener('load', () => {\n        clearTimeout(timeout);\n        resolve();\n      });\n      element.addEventListener('error', () => {\n        clearTimeout(timeout);\n        reject('image error');\n      });\n    });\n  }\n}\n","import type { ReactiveElement } from 'lit';\n\nimport type { SbbHydrationMixinType } from '../mixins.js';\n\n/**\n * Tests if an element is a Lit `ReactiveElement`.\n *\n * @param element the element to test.\n * @return true if the element is a `ReactiveElement`.\n */\nexport const isReactiveElement = (\n  element: Element,\n): element is ReactiveElement & Partial<SbbHydrationMixinType> => {\n  return Boolean((element as ReactiveElement).updateComplete);\n};\n\nconst promiseComplete = Promise.resolve();\n\n/**\n * Waits for all Lit `ReactiveElement` children of the given parent node to\n * finish rendering.\n *\n * @param root a parent node to wait for rendering on.\n */\nexport const waitForLitRender = async <\n  T extends HTMLElement | DocumentFragment = HTMLElement | DocumentFragment,\n>(\n  node: T | Promise<T>,\n): Promise<T> => {\n  const root = await node;\n  (root.parentElement ?? root)\n    .querySelectorAll?.('[defer-hydration]')\n    .forEach((e) => e.removeAttribute('defer-hydration'));\n  const completables = [root as Element, ...root.querySelectorAll('*')]\n    .filter(isReactiveElement)\n    .map((e) => [\n      e.updateComplete,\n      e.hydrationComplete ?? promiseComplete,\n      waitForLitRender(e.renderRoot),\n    ])\n    .flat(Infinity);\n  await Promise.all(completables);\n  return root;\n};\n"],"names":[],"mappings":"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"}
134
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"testing.js","sources":["../../../../src/elements/core/testing/event-spy.ts","../../../../src/elements/core/testing/mocha-extensions.ts","../../../../src/elements/core/testing/scroll.ts","../../../../src/elements/core/testing/wait-for-condition.ts","../../../../src/elements/core/testing/wait-for-event.ts","../../../../src/elements/core/testing/wait-for-image-ready.ts","../../../../src/elements/core/testing/wait-for-render.ts"],"sourcesContent":["/**\n * This class exists to facilitate the test migration from stencil to lit.\n * It mimics the API that stencil provided to test events.\n *\n * TODO: Document me\n */\nexport class EventSpy<T extends Event> {\n  private _count: number = 0;\n  public get count(): number {\n    return this._count;\n  }\n\n  private _events: T[] = [];\n  public get events(): T[] {\n    return this._events;\n  }\n\n  public get firstEvent(): T | null {\n    return this.events.length ? this.events[0] : null;\n  }\n\n  public get lastEvent(): T | null {\n    return this.events.length ? this.events[this.events.length - 1] : null;\n  }\n\n  public constructor(\n    private _event: string,\n    private readonly _target: Node | null = null,\n  ) {\n    if (!this._target) {\n      this._target = document;\n    }\n    this._listenForEvent();\n  }\n\n  private _listenForEvent(): void {\n    this._target?.addEventListener(this._event, (ev) => {\n      this._events.push(ev as T);\n      this._count++;\n    });\n  }\n}\n","import type { Func, Suite, Test } from 'mocha';\n\n/**\n * Skip the `describe` if the condition is not met\n */\nexport const describeIf = (\n  condition: boolean,\n  title: string,\n  fn: (this: Suite) => void,\n): Suite | void => {\n  if (condition) {\n    return Mocha.describe(title, fn);\n  }\n};\n\n/**\n * Skip the `test` if the condition is not met\n */\nexport const testIf = (condition: boolean, title: string, fn?: Func): Test | void => {\n  if (condition) {\n    return Mocha.test(title, fn);\n  }\n};\n","/**\n * Can be used in tests where scroll events are not triggered by default.\n * @param options ScrollToOptions\n */\nexport function mockScrollTo(options: ScrollToOptions): void {\n  window.scrollTo(options);\n\n  // We need to manually dispatching the scroll as the scrollTo\n  // does not fire a scroll event in test env.\n  document.dispatchEvent(new Event('scroll'));\n}\n","export async function waitForCondition(\n  condition: () => boolean | Promise<boolean>,\n  interval = 30,\n  timeoutInMilliseconds = 2 * 1000,\n): Promise<boolean> {\n  const start = new Date();\n  let counter = 0;\n  function checkCondition(): boolean | Promise<boolean> {\n    counter++;\n    return condition();\n  }\n  while (\n    !(await checkCondition()) &&\n    start.getTime() + timeoutInMilliseconds >= new Date().getTime()\n  ) {\n    await new Promise((resolve) => setTimeout(resolve, interval));\n  }\n  if (start.getTime() + timeoutInMilliseconds < new Date().getTime()) {\n    return Promise.reject(\n      `waitForCondition timeout: ${condition.toString()}, attempts: ${counter}, start: ${start.getTime()}, end: ${new Date().getTime()}`,\n    );\n  }\n  return Promise.resolve(true);\n}\n","export function waitForEvent(\n  element: HTMLElement,\n  eventName: string,\n  timeout = 1000,\n): Promise<void> {\n  return new Promise<void>((resolve, reject) => {\n    const signal = AbortSignal.timeout(timeout);\n    const timeoutReached = (): void => reject(`Timeout of ${timeout} reached`);\n    signal.addEventListener('abort', timeoutReached);\n    element.addEventListener(\n      eventName,\n      () => {\n        signal.removeEventListener('abort', timeoutReached);\n        resolve();\n      },\n      { passive: true, signal },\n    );\n  });\n}\n","import type { SbbImageElement } from '../../image.js';\nimport { isSafari } from '../dom.js';\n\nexport async function waitForImageReady(\n  element: HTMLImageElement | SbbImageElement,\n  timeoutInMilliseconds = 2 * 1000,\n): Promise<void> {\n  const imgElement =\n    element.localName === 'sbb-image'\n      ? (element.shadowRoot?.querySelector<HTMLImageElement>('.sbb-image__img') ?? null)\n      : (element as HTMLImageElement);\n\n  if (!imgElement) {\n    throw new Error('img tag not found');\n  }\n\n  if (!element.complete) {\n    await new Promise<void>((resolve, reject) => {\n      const timeout = setTimeout(() => reject('image loading timeout'), timeoutInMilliseconds);\n      element.addEventListener('load', () => {\n        clearTimeout(timeout);\n\n        imgElement.decode().then(() => {\n          if (isSafari && element.localName === 'sbb-image') {\n            // On a test page this is only happening once (first time an image is loaded). Therefore, the impact is very small.\n            setTimeout(resolve, 100);\n          } else {\n            resolve();\n          }\n        });\n      });\n\n      element.addEventListener('error', () => {\n        clearTimeout(timeout);\n        reject('image error');\n      });\n    });\n  } else {\n    await imgElement.decode();\n  }\n}\n","import type { ReactiveElement } from 'lit';\n\nimport type { SbbHydrationMixinType } from '../mixins.js';\n\n/**\n * Tests if an element is a Lit `ReactiveElement`.\n *\n * @param element the element to test.\n * @return true if the element is a `ReactiveElement`.\n */\nexport const isReactiveElement = (\n  element: Element,\n): element is ReactiveElement & Partial<SbbHydrationMixinType> => {\n  return Boolean((element as ReactiveElement).updateComplete);\n};\n\nconst promiseComplete = Promise.resolve();\n\n/**\n * Waits for all Lit `ReactiveElement` children of the given parent node to\n * finish rendering.\n *\n * @param root a parent node to wait for rendering on.\n */\nexport const waitForLitRender = async <\n  T extends HTMLElement | DocumentFragment = HTMLElement | DocumentFragment,\n>(\n  node: T | Promise<T>,\n): Promise<T> => {\n  const root = await node;\n  (root.parentElement ?? root)\n    .querySelectorAll?.('[defer-hydration]')\n    .forEach((e) => e.removeAttribute('defer-hydration'));\n  const completables = [root as Element, ...root.querySelectorAll('*')]\n    .filter(isReactiveElement)\n    .map((e) => [\n      e.updateComplete,\n      e.hydrationComplete ?? promiseComplete,\n      waitForLitRender(e.renderRoot),\n    ])\n    .flat(Infinity);\n  await Promise.all(completables);\n  return root;\n};\n"],"names":[],"mappings":";AAMO,MAAM,SAA0B;AAAA,EAmB9B,YACG,QACS,UAAuB,MACxC;AAFQ,SAAA,SAAA;AACS,SAAA,UAAA;AApBnB,SAAQ,SAAiB;AAKzB,SAAQ,UAAe;AAiBjB,QAAA,CAAC,KAAK,SAAS;AACjB,WAAK,UAAU;AAAA,IACjB;AACA,SAAK,gBAAgB;AAAA,EACvB;AAAA,EAzBA,IAAW,QAAgB;AACzB,WAAO,KAAK;AAAA,EACd;AAAA,EAGA,IAAW,SAAc;AACvB,WAAO,KAAK;AAAA,EACd;AAAA,EAEA,IAAW,aAAuB;AAChC,WAAO,KAAK,OAAO,SAAS,KAAK,OAAO,CAAC,IAAI;AAAA,EAC/C;AAAA,EAEA,IAAW,YAAsB;AACxB,WAAA,KAAK,OAAO,SAAS,KAAK,OAAO,KAAK,OAAO,SAAS,CAAC,IAAI;AAAA,EACpE;AAAA,EAYQ,kBAAwB;;AAC9B,eAAK,YAAL,mBAAc,iBAAiB,KAAK,QAAQ,CAAC,OAAO;AAC7C,WAAA,QAAQ,KAAK,EAAO;AACpB,WAAA;AAAA,IAAA;AAAA,EAET;AACF;ACpCO,MAAM,aAAa,CACxB,WACA,OACA,OACiB;AACjB,MAAI,WAAW;AACN,WAAA,MAAM,SAAS,OAAO,EAAE;AAAA,EACjC;AACF;AAKO,MAAM,SAAS,CAAC,WAAoB,OAAe,OAA2B;AACnF,MAAI,WAAW;AACN,WAAA,MAAM,KAAK,OAAO,EAAE;AAAA,EAC7B;AACF;AClBO,SAAS,aAAa,SAAgC;AAC3D,SAAO,SAAS,OAAO;AAIvB,WAAS,cAAc,IAAI,MAAM,QAAQ,CAAC;AAC5C;ACVA,eAAsB,iBACpB,WACA,WAAW,IACX,wBAAwB,IAAI,KACV;AACZ,QAAA,4BAAY;AAClB,MAAI,UAAU;AACd,WAAS,iBAA6C;AACpD;AACA,WAAO,UAAU;AAAA,EACnB;AACA,SACE,CAAE,MAAM,oBACR,MAAM,QAAA,IAAY,0BAA6B,oBAAA,QAAO,WACtD;AACA,UAAM,IAAI,QAAQ,CAAC,YAAY,WAAW,SAAS,QAAQ,CAAC;AAAA,EAC9D;AACI,MAAA,MAAM,YAAY,6CAA4B,KAAK,GAAE,WAAW;AAClE,WAAO,QAAQ;AAAA,MACb,6BAA6B,UAAU,SAAU,CAAA,eAAe,OAAO,YAAY,MAAM,QAAA,CAAS,WAAU,oBAAI,KAAK,GAAE,SAAS;AAAA,IAAA;AAAA,EAEpI;AACO,SAAA,QAAQ,QAAQ,IAAI;AAC7B;ACvBO,SAAS,aACd,SACA,WACA,UAAU,KACK;AACf,SAAO,IAAI,QAAc,CAAC,SAAS,WAAW;AACtC,UAAA,SAAS,YAAY,QAAQ,OAAO;AAC1C,UAAM,iBAAiB,MAAY,OAAO,cAAc,OAAO,UAAU;AAClE,WAAA,iBAAiB,SAAS,cAAc;AACvC,YAAA;AAAA,MACN;AAAA,MACA,MAAM;AACG,eAAA,oBAAoB,SAAS,cAAc;AAC1C;MACV;AAAA,MACA,EAAE,SAAS,MAAM,OAAO;AAAA,IAAA;AAAA,EAC1B,CACD;AACH;ACfA,eAAsB,kBACpB,SACA,wBAAwB,IAAI,KACb;;AACT,QAAA,aACJ,QAAQ,cAAc,gBACjB,aAAQ,eAAR,mBAAoB,cAAgC,uBAAsB,OAC1E;AAEP,MAAI,CAAC,YAAY;AACT,UAAA,IAAI,MAAM,mBAAmB;AAAA,EACrC;AAEI,MAAA,CAAC,QAAQ,UAAU;AACrB,UAAM,IAAI,QAAc,CAAC,SAAS,WAAW;AAC3C,YAAM,UAAU,WAAW,MAAM,OAAO,uBAAuB,GAAG,qBAAqB;AAC/E,cAAA,iBAAiB,QAAQ,MAAM;AACrC,qBAAa,OAAO;AAET,mBAAA,SAAS,KAAK,MAAM;AACzB,cAAA,YAAY,QAAQ,cAAc,aAAa;AAEjD,uBAAW,SAAS,GAAG;AAAA,UAAA,OAClB;AACG;UACV;AAAA,QAAA,CACD;AAAA,MAAA,CACF;AAEO,cAAA,iBAAiB,SAAS,MAAM;AACtC,qBAAa,OAAO;AACpB,eAAO,aAAa;AAAA,MAAA,CACrB;AAAA,IAAA,CACF;AAAA,EAAA,OACI;AACL,UAAM,WAAW;EACnB;AACF;AC9Ba,MAAA,oBAAoB,CAC/B,YACgE;AACzD,SAAA,QAAS,QAA4B,cAAc;AAC5D;AAEA,MAAM,kBAAkB,QAAQ;AAQnB,MAAA,mBAAmB,OAG9B,SACe;;AACf,QAAM,OAAO,MAAM;AACnB,GAAC,gBAAK,iBAAiB,MACpB,qBADF,4BACqB,qBACnB,QAAQ,CAAC,MAAM,EAAE,gBAAgB,iBAAiB;AACrD,QAAM,eAAe,CAAC,MAAiB,GAAG,KAAK,iBAAiB,GAAG,CAAC,EACjE,OAAO,iBAAiB,EACxB,IAAI,CAAC,MAAM;AAAA,IACV,EAAE;AAAA,IACF,EAAE,qBAAqB;AAAA,IACvB,iBAAiB,EAAE,UAAU;AAAA,EAAA,CAC9B,EACA,KAAK,QAAQ;AACV,QAAA,QAAQ,IAAI,YAAY;AACvB,SAAA;AACT;"}
@@ -162,8 +162,8 @@ let SbbDatepickerToggleElement = class extends SbbNegativeMixin(
162
162
  }
163
163
  _datePickerChanged(event) {
164
164
  this._datePickerElement = event.target;
165
- if (this._calendarElement && this._datePickerElement.valueAsDate) {
166
- this._calendarElement.selected = this._datePickerElement.valueAsDate;
165
+ if (this._calendarElement) {
166
+ this._calendarElement.selected = this._datePickerElement.valueAsDate || void 0;
167
167
  }
168
168
  }
169
169
  _assignCalendar(calendar) {
@@ -258,4 +258,4 @@ SbbDatepickerToggleElement = __decorateClass([
258
258
  export {
259
259
  SbbDatepickerToggleElement
260
260
  };
261
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"datepicker-toggle.js","sources":["../../../../src/elements/datepicker/datepicker-toggle/datepicker-toggle.ts"],"sourcesContent":["import type { CSSResultGroup, PropertyValues, TemplateResult } from 'lit';\nimport { html, isServer, LitElement, nothing } from 'lit';\nimport { customElement, property, state } from 'lit/decorators.js';\nimport { ref } from 'lit/directives/ref.js';\n\nimport type { CalendarView, SbbCalendarElement } from '../../calendar.js';\nimport { sbbInputModalityDetector } from '../../core/a11y.js';\nimport { SbbConnectedAbortController, SbbLanguageController } from '../../core/controllers.js';\nimport { hostAttributes } from '../../core/decorators.js';\nimport { i18nShowCalendar } from '../../core/i18n.js';\nimport { SbbHydrationMixin, SbbNegativeMixin } from '../../core/mixins.js';\nimport type { SbbPopoverElement, SbbPopoverTriggerElement } from '../../popover.js';\nimport type { SbbDatepickerElement, SbbInputUpdateEvent } from '../datepicker.js';\nimport { datepickerControlRegisteredEventFactory, getDatePicker } from '../datepicker.js';\n\nimport style from './datepicker-toggle.scss?lit&inline';\n\nimport '../../calendar.js';\nimport '../../popover.js';\n\n/**\n * Combined with a `sbb-datepicker`, it can be used to select a date from a `sbb-calendar`.\n */\n@customElement('sbb-datepicker-toggle')\n@hostAttributes({\n  slot: 'prefix',\n})\nexport class SbbDatepickerToggleElement<T = Date> extends SbbNegativeMixin(\n  SbbHydrationMixin(LitElement),\n) {\n  public static override styles: CSSResultGroup = style;\n\n  /** Datepicker reference. */\n  @property({ attribute: 'date-picker' }) public datePicker?: string | SbbDatepickerElement;\n\n  /** The initial view of calendar which should be displayed on opening. */\n  @property() public view: CalendarView = 'day';\n\n  @state() private _disabled = false;\n\n  @state() private _min: string | number | null | undefined = null;\n\n  @state() private _max: string | number | null | undefined = null;\n\n  @state() private _renderCalendar = false;\n\n  private _datePickerElement: SbbDatepickerElement<T> | null | undefined;\n  private _calendarElement!: SbbCalendarElement<T>;\n  private _triggerElement!: SbbPopoverTriggerElement;\n  private _popoverElement!: SbbPopoverElement;\n  private _datePickerController!: AbortController;\n  private _language = new SbbLanguageController(this);\n  private _abort = new SbbConnectedAbortController(this);\n\n  public constructor() {\n    super();\n    if (!isServer) {\n      this.hydrationComplete.then(() => (this._renderCalendar = true));\n    }\n  }\n\n  /**\n   * Opens the calendar.\n   */\n  public open(): void {\n    if (!this._triggerElement) {\n      this._triggerElement = this.shadowRoot!.querySelector('sbb-popover-trigger')!;\n    }\n    this._triggerElement.click();\n  }\n\n  public override connectedCallback(): void {\n    super.connectedCallback();\n    if (!this.datePicker) {\n      this._init();\n    }\n\n    const formField = this.closest?.('sbb-form-field') ?? this.closest?.('[data-form-field]');\n    if (formField) {\n      this.negative = formField.hasAttribute('negative');\n    }\n\n    this.addEventListener(\n      'click',\n      (event) => {\n        if (event.composedPath()[0] === this) {\n          this.open();\n        }\n      },\n      { signal: this._abort.signal },\n    );\n  }\n\n  public override willUpdate(changedProperties: PropertyValues<this>): void {\n    super.willUpdate(changedProperties);\n\n    if (changedProperties.has('datePicker')) {\n      this._init(this.datePicker);\n    }\n  }\n\n  public override disconnectedCallback(): void {\n    super.disconnectedCallback();\n    this._datePickerController?.abort();\n  }\n\n  private _init(datePicker?: string | SbbDatepickerElement): void {\n    this._datePickerController?.abort();\n    this._datePickerController = new AbortController();\n    this._datePickerElement = getDatePicker<T>(this, datePicker);\n    if (!this._datePickerElement) {\n      // If the component is attached to the DOM before the datepicker, it has to listen for the datepicker init,\n      // assuming that the two components share the same parent element.\n      this.parentElement?.addEventListener(\n        'inputUpdated',\n        (e: Event) => this._init(e.target as SbbDatepickerElement),\n        { once: true, signal: this._datePickerController.signal },\n      );\n      return;\n    }\n\n    this._datePickerElement?.addEventListener(\n      'inputUpdated',\n      (event: CustomEvent<SbbInputUpdateEvent>) => {\n        this._datePickerElement = event.target as SbbDatepickerElement<T>;\n        this._disabled = !!(event.detail.disabled || event.detail.readonly);\n        this._min = event.detail.min;\n        this._max = event.detail.max;\n      },\n      { signal: this._datePickerController.signal },\n    );\n    this._datePickerElement?.addEventListener(\n      'change',\n      (event: Event) => this._datePickerChanged(event),\n      {\n        signal: this._datePickerController.signal,\n      },\n    );\n    this._datePickerElement?.addEventListener(\n      'datePickerUpdated',\n      (event: Event) =>\n        this._configureCalendar(this._calendarElement, event.target as SbbDatepickerElement<T>),\n      { signal: this._datePickerController.signal },\n    );\n    this._datePickerElement.dispatchEvent(datepickerControlRegisteredEventFactory());\n  }\n\n  private _configureCalendar(\n    calendar: SbbCalendarElement<T>,\n    datepicker: SbbDatepickerElement<T>,\n  ): void {\n    if (!calendar || !datepicker) {\n      return;\n    }\n    calendar.wide = datepicker.wide;\n    calendar.now = this._nowOrUndefined();\n    calendar.dateFilter = datepicker.dateFilter;\n  }\n\n  private _datePickerChanged(event: Event): void {\n    this._datePickerElement = event.target as SbbDatepickerElement<T>;\n    if (this._calendarElement && this._datePickerElement.valueAsDate) {\n      this._calendarElement.selected = this._datePickerElement.valueAsDate;\n    }\n  }\n\n  private _assignCalendar(calendar: SbbCalendarElement<T>): void {\n    if (this._calendarElement && this._calendarElement === calendar) {\n      return;\n    }\n    this._calendarElement = calendar;\n    if (\n      !('valueAsDate' in (this._datePickerElement ?? {})) ||\n      !this._calendarElement?.resetPosition\n    ) {\n      return;\n    }\n    this._calendarElement.selected = this._datePickerElement!.valueAsDate ?? undefined;\n    this._configureCalendar(this._calendarElement, this._datePickerElement!);\n    this._calendarElement.resetPosition();\n  }\n\n  protected override updated(changedProperties: PropertyValues<this>): void {\n    super.updated(changedProperties);\n\n    this._popoverElement.trigger = this._triggerElement;\n  }\n\n  private _nowOrUndefined(): T | undefined {\n    return this._datePickerElement?.hasCustomNow() ? this._datePickerElement.now : undefined;\n  }\n\n  protected override render(): TemplateResult {\n    return html`\n      <sbb-popover-trigger\n        icon-name=\"calendar-small\"\n        aria-label=${i18nShowCalendar[this._language.current]}\n        ?disabled=${!isServer && (!this._datePickerElement || this._disabled)}\n        ?negative=${this.negative}\n        data-icon-small\n        ${ref((el?: Element) => (this._triggerElement = el as SbbPopoverTriggerElement))}\n      ></sbb-popover-trigger>\n      <sbb-popover\n        @willOpen=${() => this._calendarElement.resetPosition()}\n        @didOpen=${() => {\n          if (sbbInputModalityDetector.mostRecentModality === 'keyboard') {\n            this._calendarElement.focus();\n          }\n        }}\n        .trigger=${this._triggerElement}\n        hide-close-button\n        ${ref((el?: Element) => (this._popoverElement = el as SbbPopoverElement))}\n      >\n        ${this._renderCalendar\n          ? html`<sbb-calendar\n              .view=${this.view}\n              .min=${this._min}\n              .max=${this._max}\n              .now=${this._nowOrUndefined()}\n              ?wide=${this._datePickerElement?.wide}\n              .dateFilter=${this._datePickerElement?.dateFilter}\n              @dateSelected=${(d: CustomEvent<T>) => {\n                this._calendarElement.selected = d.detail;\n                if (this._datePickerElement) {\n                  this._datePickerElement.valueAsDate = d.detail;\n                }\n              }}\n              ${ref((calendar?: Element) =>\n                this._assignCalendar(calendar as SbbCalendarElement<T>),\n              )}\n            ></sbb-calendar>`\n          : nothing}\n      </sbb-popover>\n    `;\n  }\n}\n\ndeclare global {\n  interface HTMLElementTagNameMap {\n    // eslint-disable-next-line @typescript-eslint/naming-convention\n    'sbb-datepicker-toggle': SbbDatepickerToggleElement;\n  }\n}\n"],"names":[],"mappings":"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"}
261
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"datepicker-toggle.js","sources":["../../../../src/elements/datepicker/datepicker-toggle/datepicker-toggle.ts"],"sourcesContent":["import type { CSSResultGroup, PropertyValues, TemplateResult } from 'lit';\nimport { html, isServer, LitElement, nothing } from 'lit';\nimport { customElement, property, state } from 'lit/decorators.js';\nimport { ref } from 'lit/directives/ref.js';\n\nimport type { CalendarView, SbbCalendarElement } from '../../calendar.js';\nimport { sbbInputModalityDetector } from '../../core/a11y.js';\nimport { SbbConnectedAbortController, SbbLanguageController } from '../../core/controllers.js';\nimport { hostAttributes } from '../../core/decorators.js';\nimport { i18nShowCalendar } from '../../core/i18n.js';\nimport { SbbHydrationMixin, SbbNegativeMixin } from '../../core/mixins.js';\nimport type { SbbPopoverElement, SbbPopoverTriggerElement } from '../../popover.js';\nimport type { SbbDatepickerElement, SbbInputUpdateEvent } from '../datepicker.js';\nimport { datepickerControlRegisteredEventFactory, getDatePicker } from '../datepicker.js';\n\nimport style from './datepicker-toggle.scss?lit&inline';\n\nimport '../../calendar.js';\nimport '../../popover.js';\n\n/**\n * Combined with a `sbb-datepicker`, it can be used to select a date from a `sbb-calendar`.\n */\n@customElement('sbb-datepicker-toggle')\n@hostAttributes({\n  slot: 'prefix',\n})\nexport class SbbDatepickerToggleElement<T = Date> extends SbbNegativeMixin(\n  SbbHydrationMixin(LitElement),\n) {\n  public static override styles: CSSResultGroup = style;\n\n  /** Datepicker reference. */\n  @property({ attribute: 'date-picker' }) public datePicker?: string | SbbDatepickerElement;\n\n  /** The initial view of calendar which should be displayed on opening. */\n  @property() public view: CalendarView = 'day';\n\n  @state() private _disabled = false;\n\n  @state() private _min: string | number | null | undefined = null;\n\n  @state() private _max: string | number | null | undefined = null;\n\n  @state() private _renderCalendar = false;\n\n  private _datePickerElement: SbbDatepickerElement<T> | null | undefined;\n  private _calendarElement!: SbbCalendarElement<T>;\n  private _triggerElement!: SbbPopoverTriggerElement;\n  private _popoverElement!: SbbPopoverElement;\n  private _datePickerController!: AbortController;\n  private _language = new SbbLanguageController(this);\n  private _abort = new SbbConnectedAbortController(this);\n\n  public constructor() {\n    super();\n    if (!isServer) {\n      this.hydrationComplete.then(() => (this._renderCalendar = true));\n    }\n  }\n\n  /**\n   * Opens the calendar.\n   */\n  public open(): void {\n    if (!this._triggerElement) {\n      this._triggerElement = this.shadowRoot!.querySelector('sbb-popover-trigger')!;\n    }\n    this._triggerElement.click();\n  }\n\n  public override connectedCallback(): void {\n    super.connectedCallback();\n    if (!this.datePicker) {\n      this._init();\n    }\n\n    const formField = this.closest?.('sbb-form-field') ?? this.closest?.('[data-form-field]');\n    if (formField) {\n      this.negative = formField.hasAttribute('negative');\n    }\n\n    this.addEventListener(\n      'click',\n      (event) => {\n        if (event.composedPath()[0] === this) {\n          this.open();\n        }\n      },\n      { signal: this._abort.signal },\n    );\n  }\n\n  public override willUpdate(changedProperties: PropertyValues<this>): void {\n    super.willUpdate(changedProperties);\n\n    if (changedProperties.has('datePicker')) {\n      this._init(this.datePicker);\n    }\n  }\n\n  public override disconnectedCallback(): void {\n    super.disconnectedCallback();\n    this._datePickerController?.abort();\n  }\n\n  private _init(datePicker?: string | SbbDatepickerElement): void {\n    this._datePickerController?.abort();\n    this._datePickerController = new AbortController();\n    this._datePickerElement = getDatePicker<T>(this, datePicker);\n    if (!this._datePickerElement) {\n      // If the component is attached to the DOM before the datepicker, it has to listen for the datepicker init,\n      // assuming that the two components share the same parent element.\n      this.parentElement?.addEventListener(\n        'inputUpdated',\n        (e: Event) => this._init(e.target as SbbDatepickerElement),\n        { once: true, signal: this._datePickerController.signal },\n      );\n      return;\n    }\n\n    this._datePickerElement?.addEventListener(\n      'inputUpdated',\n      (event: CustomEvent<SbbInputUpdateEvent>) => {\n        this._datePickerElement = event.target as SbbDatepickerElement<T>;\n        this._disabled = !!(event.detail.disabled || event.detail.readonly);\n        this._min = event.detail.min;\n        this._max = event.detail.max;\n      },\n      { signal: this._datePickerController.signal },\n    );\n    this._datePickerElement?.addEventListener(\n      'change',\n      (event: Event) => this._datePickerChanged(event),\n      {\n        signal: this._datePickerController.signal,\n      },\n    );\n    this._datePickerElement?.addEventListener(\n      'datePickerUpdated',\n      (event: Event) =>\n        this._configureCalendar(this._calendarElement, event.target as SbbDatepickerElement<T>),\n      { signal: this._datePickerController.signal },\n    );\n    this._datePickerElement.dispatchEvent(datepickerControlRegisteredEventFactory());\n  }\n\n  private _configureCalendar(\n    calendar: SbbCalendarElement<T>,\n    datepicker: SbbDatepickerElement<T>,\n  ): void {\n    if (!calendar || !datepicker) {\n      return;\n    }\n    calendar.wide = datepicker.wide;\n    calendar.now = this._nowOrUndefined();\n    calendar.dateFilter = datepicker.dateFilter;\n  }\n\n  private _datePickerChanged(event: Event): void {\n    this._datePickerElement = event.target as SbbDatepickerElement<T>;\n    if (this._calendarElement) {\n      this._calendarElement.selected = this._datePickerElement.valueAsDate || undefined;\n    }\n  }\n\n  private _assignCalendar(calendar: SbbCalendarElement<T>): void {\n    if (this._calendarElement && this._calendarElement === calendar) {\n      return;\n    }\n    this._calendarElement = calendar;\n    if (\n      !('valueAsDate' in (this._datePickerElement ?? {})) ||\n      !this._calendarElement?.resetPosition\n    ) {\n      return;\n    }\n    this._calendarElement.selected = this._datePickerElement!.valueAsDate ?? undefined;\n    this._configureCalendar(this._calendarElement, this._datePickerElement!);\n    this._calendarElement.resetPosition();\n  }\n\n  protected override updated(changedProperties: PropertyValues<this>): void {\n    super.updated(changedProperties);\n\n    this._popoverElement.trigger = this._triggerElement;\n  }\n\n  private _nowOrUndefined(): T | undefined {\n    return this._datePickerElement?.hasCustomNow() ? this._datePickerElement.now : undefined;\n  }\n\n  protected override render(): TemplateResult {\n    return html`\n      <sbb-popover-trigger\n        icon-name=\"calendar-small\"\n        aria-label=${i18nShowCalendar[this._language.current]}\n        ?disabled=${!isServer && (!this._datePickerElement || this._disabled)}\n        ?negative=${this.negative}\n        data-icon-small\n        ${ref((el?: Element) => (this._triggerElement = el as SbbPopoverTriggerElement))}\n      ></sbb-popover-trigger>\n      <sbb-popover\n        @willOpen=${() => this._calendarElement.resetPosition()}\n        @didOpen=${() => {\n          if (sbbInputModalityDetector.mostRecentModality === 'keyboard') {\n            this._calendarElement.focus();\n          }\n        }}\n        .trigger=${this._triggerElement}\n        hide-close-button\n        ${ref((el?: Element) => (this._popoverElement = el as SbbPopoverElement))}\n      >\n        ${this._renderCalendar\n          ? html`<sbb-calendar\n              .view=${this.view}\n              .min=${this._min}\n              .max=${this._max}\n              .now=${this._nowOrUndefined()}\n              ?wide=${this._datePickerElement?.wide}\n              .dateFilter=${this._datePickerElement?.dateFilter}\n              @dateSelected=${(d: CustomEvent<T>) => {\n                this._calendarElement.selected = d.detail;\n                if (this._datePickerElement) {\n                  this._datePickerElement.valueAsDate = d.detail;\n                }\n              }}\n              ${ref((calendar?: Element) =>\n                this._assignCalendar(calendar as SbbCalendarElement<T>),\n              )}\n            ></sbb-calendar>`\n          : nothing}\n      </sbb-popover>\n    `;\n  }\n}\n\ndeclare global {\n  interface HTMLElementTagNameMap {\n    // eslint-disable-next-line @typescript-eslint/naming-convention\n    'sbb-datepicker-toggle': SbbDatepickerToggleElement;\n  }\n}\n"],"names":[],"mappings":"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"}