@sbb-esta/lyne-elements-dev 4.9.0-dev.1775050914 → 4.9.0-dev.1775054429

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (51) hide show
  1. package/autocomplete/autocomplete-base-element.js +1 -1
  2. package/autocomplete/autocomplete.component.js +1 -1
  3. package/{autocomplete-base-element-DFRA5eoY.js → autocomplete-base-element-DarHnnYC.js} +1 -1
  4. package/autocomplete.js +1 -1
  5. package/autocomplete.pure.js +1 -1
  6. package/custom-elements.json +24 -24
  7. package/date-input/date-input.component.js +1 -1
  8. package/{date-input.component-C6UN7HzO.js → date-input.component-B3f_GThS.js} +1 -1
  9. package/date-input.js +1 -1
  10. package/date-input.pure.js +1 -1
  11. package/development/autocomplete/autocomplete-base-element.js +1 -1
  12. package/development/autocomplete/autocomplete.component.js +1 -1
  13. package/development/autocomplete-base-element-YwfObJzm.js +678 -0
  14. package/development/autocomplete.js +1 -1
  15. package/development/autocomplete.pure.js +1 -1
  16. package/development/date-input/date-input.component.js +1 -1
  17. package/development/date-input.component-UP1afzgy.js +307 -0
  18. package/development/date-input.js +1 -1
  19. package/development/date-input.pure.js +1 -1
  20. package/development/form-field/form-field/form-field.component.js +1 -1
  21. package/development/form-field/form-field.js +1 -1
  22. package/development/form-field.component-DnUA6Uds.js +650 -0
  23. package/development/form-field.js +1 -1
  24. package/development/form-field.pure.js +1 -1
  25. package/development/select/select.component.js +1 -1
  26. package/development/select.component-Il_cpCFV.js +784 -0
  27. package/development/select.js +1 -1
  28. package/development/select.pure.js +1 -1
  29. package/development/time-input/time-input.component.js +1 -1
  30. package/development/time-input.component-XS0aJ6M7.js +191 -0
  31. package/development/time-input.js +1 -1
  32. package/development/time-input.pure.js +1 -1
  33. package/form-field/form-field/form-field.component.js +1 -1
  34. package/form-field/form-field.js +1 -1
  35. package/{form-field.component-BKjqLhmy.js → form-field.component-D2R9BpyQ.js} +1 -1
  36. package/form-field.js +1 -1
  37. package/form-field.pure.js +1 -1
  38. package/package.json +2 -2
  39. package/select/select.component.js +1 -1
  40. package/{select.component-A-TwBa65.js → select.component-DMzvfiNM.js} +1 -1
  41. package/select.js +1 -1
  42. package/select.pure.js +1 -1
  43. package/time-input/time-input.component.js +1 -1
  44. package/{time-input.component-KSyUufTP.js → time-input.component-Jv2xorV5.js} +1 -1
  45. package/time-input.js +1 -1
  46. package/time-input.pure.js +1 -1
  47. package/development/autocomplete-base-element-CNoHrX63.js +0 -678
  48. package/development/date-input.component-B7Eq5hCB.js +0 -307
  49. package/development/form-field.component-KEFq2rmY.js +0 -650
  50. package/development/select.component-DNMAVhaz.js +0 -784
  51. package/development/time-input.component-B4HBAe0K.js +0 -191
@@ -0,0 +1,650 @@
1
+ import { __esDecorate, __runInitializers } from "tslib";
2
+ import { html, isServer, nothing, unsafeCSS } from "lit";
3
+ import { property, state } from "lit/decorators.js";
4
+ import { SbbElement } from "./core/base-elements.js";
5
+ import { forceType } from "./core/decorators.js";
6
+ import { isLean } from "./core/dom.js";
7
+ import { SbbNegativeMixin, appendAriaElements, removeAriaElements } from "./core/mixins.js";
8
+ import { boxSizingStyles } from "./core/styles.js";
9
+ import { SbbLanguageController } from "./core/controllers.js";
10
+ import { i18nOptional } from "./core/i18n.js";
11
+ import "./icon.js";
12
+ import { sbbInputModalityDetector } from "./core/a11y.js";
13
+ //#region src/elements/form-field/form-field/form-field.scss?inline
14
+ var form_field_default = "@charset \"UTF-8\";\n:host {\n --sbb-form-field-error-divider-width: var(--sbb-spacing-fixed-1x);\n --sbb-form-field-error-padding-block-start: var(--sbb-form-field-error-divider-width);\n --sbb-form-field-select-inline-padding-end: calc(\n var(--sbb-icon-svg-width) + var(--sbb-form-field-gap)\n );\n --sbb-form-field-label-size: calc(\n var(--sbb-form-field-label-text-size) * var(--sbb-typo-line-height-text)\n );\n --sbb-form-field-text-line-height: calc(\n var(--sbb-form-field-input-text-size) * var(--sbb-typo-line-height-text)\n );\n --sbb-form-field-margin-block-start: calc(\n (\n var(--sbb-form-field-min-height) - var(--sbb-form-field-label-size) - var(\n --sbb-form-field-text-line-height\n ) +\n var(--_sbb-form-field-label-to-input-overlapping)\n ) /\n 2\n );\n --sbb-icon-svg-width: var(--sbb-form-field-icon-size);\n --sbb-icon-svg-height: var(--sbb-form-field-icon-size);\n --sbb-focus-outline-color: var(--sbb-focus-outline-color-default);\n --sbb-focus-outline-color: light-dark(\n var(--sbb-focus-outline-color-default),\n var(--sbb-focus-outline-color-dark)\n );\n display: inline-block;\n color: var(--sbb-form-field-color);\n font-weight: normal;\n}\n\n:host(:where(:not([width=collapse]))) {\n min-width: 9.375rem;\n width: min(18.75rem, 100%);\n}\n\n:host([negative]) {\n color: var(--sbb-color-3-negative);\n --sbb-form-field-background-color: var(--sbb-background-color-1-negative);\n --sbb-form-field-border-color: var(--sbb-border-color-5);\n --sbb-form-field-label-color: var(--sbb-color-5);\n --sbb-form-field-prefix-color: var(--sbb-color-5);\n --sbb-form-field-text-color: var(--sbb-color-3-negative);\n --sbb-focus-outline-color: var(--sbb-focus-outline-color-dark);\n --sbb-form-field-arrow-color: var(--sbb-color-3-negative);\n}\n@media (forced-colors: active) {\n :host([negative]) {\n --sbb-form-field-border-color: ButtonBorder;\n }\n}\n\n:host([size=s]) {\n --sbb-form-field-min-height: var(--sbb-size-element-xs);\n --sbb-form-field-padding-inline: var(--sbb-spacing-fixed-2x);\n --sbb-form-field-input-text-size: var(--sbb-text-font-size-s);\n --sbb-form-field-label-text-size: var(--sbb-text-font-size-xxs);\n --_sbb-form-field-label-to-input-overlapping: 0.625rem;\n --_sbb-form-field-floating-label-transform: 0.34375rem;\n --_sbb-form-field-spacer-margin-block-end: -0.53125rem;\n}\n@media (min-width: calc(64rem)) {\n :host([size=s]) {\n --_sbb-form-field-label-to-input-overlapping: 0.6875rem;\n --_sbb-form-field-floating-label-transform: 0.3125rem;\n --_sbb-form-field-spacer-margin-block-end: -0.5rem;\n }\n}\n\n:host([size=m]) {\n --sbb-form-field-min-height: var(--sbb-size-element-m);\n --sbb-form-field-padding-inline: var(--sbb-spacing-fixed-3x);\n --sbb-form-field-input-text-size: var(--sbb-text-font-size-m);\n --sbb-form-field-label-text-size: var(--sbb-text-font-size-xs);\n --_sbb-form-field-label-to-input-overlapping: var(--sbb-spacing-fixed-1x);\n --_sbb-form-field-floating-label-transform: 0.53125rem;\n --_sbb-form-field-spacer-margin-block-end: calc(\n -1 * var(--_sbb-form-field-label-to-input-overlapping)\n );\n}\n@media (min-width: calc(64rem)) {\n :host([size=m]) {\n --_sbb-form-field-floating-label-transform: 0.65625rem;\n }\n}\n\n:host([size=l]) {\n --sbb-form-field-min-height: var(--sbb-size-element-l);\n --sbb-form-field-padding-inline: var(--sbb-spacing-responsive-xxxs);\n}\n\n:host([error-space=reserve]) {\n --sbb-form-field-error-min-height: calc(\n var(--sbb-typo-line-height-text) * var(--sbb-text-font-size-xs)\n );\n --sbb-form-field-error-padding-block-start-override: var(--sbb-form-field-error-divider-width);\n}\n\n:host(:where(:is(:state(readonly),[state--readonly]), :is(:state(disabled),[state--disabled]))) {\n --sbb-form-field-background-color: var(--sbb-background-color-3);\n --sbb-form-field-border-color: var(--sbb-color-graphite);\n --sbb-form-field-border-color: light-dark(var(--sbb-color-graphite), var(--sbb-color-smoke));\n --sbb-form-field-arrow-color: var(--sbb-color-granite);\n --sbb-form-field-arrow-color: light-dark(var(--sbb-color-granite), var(--sbb-color-storm));\n}\n@media (forced-colors: active) {\n :host(:where(:is(:state(readonly),[state--readonly]), :is(:state(disabled),[state--disabled]))) {\n --sbb-form-field-border-color: ButtonBorder;\n }\n}\n\n:host(:where(:is(:state(readonly),[state--readonly]), :is(:state(disabled),[state--disabled]))[negative]) {\n --sbb-form-field-background-color: var(--sbb-background-color-3-negative);\n --sbb-form-field-border-color: var(--sbb-border-color-5);\n --sbb-form-field-arrow-color: var(--sbb-border-color-5);\n}\n\n:host(:where(:is(:state(disabled),[state--disabled]))) {\n --sbb-form-field-label-color: var(--sbb-color-granite);\n --sbb-form-field-label-color: light-dark(var(--sbb-color-granite), var(--sbb-color-smoke));\n --sbb-form-field-prefix-color: var(--sbb-color-granite);\n --sbb-form-field-prefix-color: light-dark(var(--sbb-color-granite), var(--sbb-color-storm));\n --sbb-form-field-border-style: dashed;\n}\n@media (forced-colors: active) {\n :host(:where(:is(:state(disabled),[state--disabled]))) {\n --sbb-form-field-label-color: GrayText !important;\n --sbb-form-field-prefix-color: GrayText !important;\n --sbb-form-field-text-color: GrayText !important;\n --sbb-form-field-border-color: GrayText !important;\n }\n}\n\n:host(:where(:is(:state(disabled),[state--disabled]))[negative]) {\n --sbb-form-field-text-color: var(--sbb-color-5);\n --sbb-form-field-label-color: var(--sbb-color-5);\n --sbb-form-field-prefix-color: var(--sbb-color-5);\n}\n\n:host(:where(:is(:state(readonly),[state--readonly])):not([negative])) {\n --sbb-form-field-label-color: var(--sbb-color-granite);\n --sbb-form-field-label-color: light-dark(var(--sbb-color-granite), var(--sbb-color-smoke));\n --sbb-form-field-prefix-color: var(--sbb-color-granite);\n --sbb-form-field-prefix-color: light-dark(var(--sbb-color-granite), var(--sbb-color-storm));\n}\n\n:host(:where(:is(:state(focus),[state--focus]))) {\n --sbb-form-field-border-color: var(--sbb-border-color-3);\n --sbb-form-field-prefix-color: var(--sbb-color-3);\n --sbb-form-field-border-width: var(--sbb-border-width-2x);\n}\n\n:host(:where(:is(:state(focus),[state--focus]))[negative]) {\n --sbb-form-field-border-color: var(--sbb-border-color-3-negative);\n --sbb-form-field-prefix-color: var(--sbb-color-3-negative);\n}\n\n@media (forced-colors: active) {\n :host(:is(:state(focus),[state--focus])) {\n --sbb-form-field-border-color: Highlight;\n --sbb-form-field-prefix-color: Highlight;\n }\n}\n\n:host(:not(:where(:is(:state(has-error),[state--has-error])))) {\n --sbb-form-field-error-padding-block-start: 0;\n}\n\n:host([floating-label]) {\n --sbb-select-placeholder-color: transparent;\n}\n@media (forced-colors: active) {\n :host([floating-label]) {\n --sbb-select-placeholder-color: Canvas;\n }\n}\n\n:host(:is(:not(:is(:state(slotted-label),[state--slotted-label])), [hidden-label])) {\n --sbb-form-field-label-size: 0rem;\n --_sbb-form-field-label-to-input-overlapping: 0rem;\n}\n\n:host(:where(:is(:state(input-type-sbb-slider),[state--input-type-sbb-slider]))) {\n --sbb-form-field-overflow: visible;\n}\n\n.sbb-form-field__space-wrapper {\n display: flex;\n flex-direction: column;\n height: 100%;\n}\n.sbb-form-field__space-wrapper::before {\n content: \"​\";\n user-select: none;\n width: 0;\n height: 0;\n}\n\n.sbb-form-field__wrapper {\n display: flex;\n flex: 1;\n gap: var(--sbb-form-field-gap);\n padding-inline: var(--sbb-form-field-padding-inline);\n border-radius: var(--sbb-form-field-border-radius);\n min-height: var(--sbb-form-field-min-height);\n max-height: var(--sbb-form-field-max-height);\n background-color: var(--sbb-form-field-background-color);\n position: relative;\n}\n.sbb-form-field__wrapper::before {\n content: \"\";\n display: block;\n position: absolute;\n inset: 0;\n border: var(--sbb-form-field-border-width) var(--sbb-form-field-border-style) var(--sbb-form-field-border-color);\n border-radius: var(--sbb-form-field-border-radius);\n pointer-events: none;\n}\n:host(:where(:is(:state(focus),[state--focus]):is(:state(focus-origin-keyboard),[state--focus-origin-keyboard]))) .sbb-form-field__wrapper {\n outline-offset: var(--sbb-focus-outline-offset);\n outline: var(--sbb-focus-outline-color) var(--sbb-focus-outline-style, solid) var(--sbb-focus-outline-width);\n outline-offset: var(--sbb-form-field-outline-offset, var(--sbb-focus-outline-offset));\n}\n@media (forced-colors: none) {\n :host(:is([borderless], :is(:state(input-type-sbb-slider),[state--input-type-sbb-slider]))) .sbb-form-field__wrapper::before {\n border-color: transparent;\n }\n :host(:where(:is(:state(focus),[state--focus]), :is(:state(disabled),[state--disabled]))[borderless]) .sbb-form-field__wrapper::after {\n content: \"\";\n position: absolute;\n border-block-end: var(--sbb-border-width-1x) var(--sbb-form-field-border-style) var(--sbb-form-field-border-color);\n inset-inline: var(--sbb-form-field-padding-inline);\n inset-block-end: 0;\n z-index: var(--sbb-form-field-focus-underline-z-index, initial);\n }\n}\n\n::slotted(*[slot=prefix]) {\n color: var(--sbb-form-field-prefix-color);\n}\n\n::slotted(*[slot=prefix]),\n::slotted(*[slot=suffix]) {\n display: flex;\n min-width: var(--sbb-icon-svg-width);\n margin-block-start: calc((var(--sbb-form-field-min-height) - var(--sbb-size-icon-ui-small)) / 2);\n}\n\n:host(:where(:is(:state(empty),[state--empty]), :is(:state(disabled),[state--disabled]), :is(:state(readonly),[state--readonly]))) ::slotted(sbb-form-field-clear) {\n display: none;\n}\n\n::slotted(sbb-datepicker-toggle) {\n padding-block-end: calc((var(--sbb-form-field-min-height) - var(--sbb-size-icon-ui-small)) / 2);\n}\n\n@media (forced-colors: active) {\n ::slotted(*[slot=suffix]) {\n color: var(--sbb-form-field-prefix-color);\n }\n}\n.sbb-form-field__select-input-icon {\n position: absolute;\n top: 50%;\n transform: translateY(-50%);\n position: absolute;\n inset-inline-end: 0;\n margin-block-start: calc(-1 * var(--sbb-form-field-margin-block-start) / 2);\n pointer-events: none;\n color: var(--sbb-form-field-arrow-color);\n}\n\n.sbb-form-field__input-container {\n display: flex;\n flex-direction: column;\n flex: 1 1 auto;\n position: relative;\n margin-block-start: var(--sbb-form-field-margin-block-start);\n min-width: 0;\n}\n\n.sbb-form-field__label-spacer {\n display: flex;\n height: calc(var(--sbb-form-field-label-text-size) * var(--sbb-typo-line-height-text));\n margin-block-end: var(--_sbb-form-field-spacer-margin-block-end);\n}\n\n:host(:not(:is(:state(slotted-label),[state--slotted-label]))) :is(.sbb-form-field__label, .sbb-form-field__label-spacer) {\n display: none;\n}\n:host([hidden-label]) :is(.sbb-form-field__label, .sbb-form-field__label-spacer) {\n border: 0;\n clip-path: rect(0 0 0 0);\n height: 1px;\n margin: -1px;\n overflow: hidden;\n padding: 0;\n position: absolute;\n white-space: nowrap;\n width: 1px;\n}\n\n.sbb-form-field__label {\n display: flex;\n max-width: 100%;\n cursor: var(--sbb-cursor-default);\n position: absolute;\n inset-block-start: 0;\n color: var(--sbb-form-field-label-color);\n font-size: var(--sbb-form-field-label-text-size);\n letter-spacing: var(--sbb-typo-letter-spacing-text);\n}\n:host(:where(:is(:state(input-type-select),[state--input-type-select]), :is(:state(input-type-sbb-select),[state--input-type-sbb-select]))) .sbb-form-field__label {\n padding-inline-end: var(--sbb-form-field-select-inline-padding-end);\n}\n:host([floating-label]) .sbb-form-field__label {\n transform-origin: 0 0;\n pointer-events: none;\n backface-visibility: hidden;\n will-change: transform, font-size;\n transition-duration: var(--sbb-disable-animation-duration, var(--sbb-animation-duration-2x));\n transition-timing-function: var(--sbb-animation-easing);\n transition-property: transform, font-size;\n}\n:host([floating-label]:is(:not(:where(:is(:state(focus),[state--focus])):not(:where(:is(:state(input-type-sbb-select),[state--input-type-sbb-select]))),\n:where(:is(:state(has-popup-open),[state--has-popup-open]))),\n:where(:is(:state(readonly),[state--readonly]))):where(:is(:state(empty),[state--empty]))) .sbb-form-field__label {\n font-size: var(--sbb-form-field-input-text-size);\n transform: translateY(var(--_sbb-form-field-floating-label-transform));\n}\n@media (forced-colors: active) {\n :host(:where(:is(:state(input-type-textarea),[state--input-type-textarea]))) .sbb-form-field__label {\n z-index: 1;\n }\n}\n\n.sbb-form-field__label-ellipsis {\n overflow: hidden;\n white-space: nowrap;\n text-overflow: ellipsis;\n}\n\n.sbb-form-field__input {\n display: flex;\n flex: 1;\n align-items: start;\n}\n:host([size=s]:is(:is(:state(input-type-input),[state--input-type-input]),\n:is(:state(input-type-select),[state--input-type-select]),\n:is(:state(input-type-sbb-select),[state--input-type-sbb-select]),\n:is(:state(input-type-sbb-date-input),[state--input-type-sbb-date-input]),\n:is(:state(input-type-sbb-time-input),[state--input-type-sbb-time-input]))) .sbb-form-field__input {\n margin-block-end: -0.125rem;\n}\n\n.sbb-form-field__error {\n display: flex;\n min-height: var(--sbb-form-field-error-min-height);\n flex-direction: column;\n margin-block-start: var(--sbb-form-field-error-padding-block-start-override, var(--sbb-form-field-error-padding-block-start));\n}";
15
+ //#endregion
16
+ //#region src/elements/form-field/form-field/form-field.component.ts
17
+ var nextId = 0;
18
+ var patchedInputs = /* @__PURE__ */ new WeakMap();
19
+ var nativeInputElements = [
20
+ "input",
21
+ "textarea",
22
+ "select"
23
+ ];
24
+ var SbbFormFieldControlEvent = class extends Event {
25
+ get control() {
26
+ return this._control;
27
+ }
28
+ constructor(control) {
29
+ super("formfieldcontrol");
30
+ this._control = control;
31
+ }
32
+ };
33
+ /**
34
+ * It wraps an input element adding label, errors, icon, etc.
35
+ *
36
+ * @slot - Use this slot to render an input/select or a supported non-native element.
37
+ * @slot label - Use this slot to render a label.
38
+ * @slot prefix - Use this slot to render an icon on the left side of the input.
39
+ * @slot suffix - Use this slot to render an icon on the right side of the input.
40
+ * @slot error - Use this slot to render an error.
41
+ *
42
+ * @cssprop [--sbb-form-field-outline-offset] - To override the focus outline offset,
43
+ * @cssprop [--sbb-form-field-focus-underline-z-index] - To override the z-index of the focus underline effect,
44
+ */
45
+ var SbbFormFieldElement = (() => {
46
+ let _classSuper = SbbNegativeMixin(SbbElement);
47
+ let _errorSpace_decorators;
48
+ let _errorSpace_initializers = [];
49
+ let _errorSpace_extraInitializers = [];
50
+ let _optional_decorators;
51
+ let _optional_initializers = [];
52
+ let _optional_extraInitializers = [];
53
+ let _size_decorators;
54
+ let _size_initializers = [];
55
+ let _size_extraInitializers = [];
56
+ let _borderless_decorators;
57
+ let _borderless_initializers = [];
58
+ let _borderless_extraInitializers = [];
59
+ let _width_decorators;
60
+ let _width_initializers = [];
61
+ let _width_extraInitializers = [];
62
+ let _hiddenLabel_decorators;
63
+ let _hiddenLabel_initializers = [];
64
+ let _hiddenLabel_extraInitializers = [];
65
+ let _floatingLabel_decorators;
66
+ let _floatingLabel_initializers = [];
67
+ let _floatingLabel_extraInitializers = [];
68
+ let __errorElements_decorators;
69
+ let __errorElements_initializers = [];
70
+ let __errorElements_extraInitializers = [];
71
+ let __input_decorators;
72
+ let __input_initializers = [];
73
+ let __input_extraInitializers = [];
74
+ let __label_decorators;
75
+ let __label_initializers = [];
76
+ let __label_extraInitializers = [];
77
+ return class SbbFormFieldElement extends _classSuper {
78
+ static {
79
+ const _metadata = typeof Symbol === "function" && Symbol.metadata ? Object.create(_classSuper[Symbol.metadata] ?? null) : void 0;
80
+ _errorSpace_decorators = [property({
81
+ attribute: "error-space",
82
+ reflect: true
83
+ })];
84
+ _optional_decorators = [forceType(), property({ type: Boolean })];
85
+ _size_decorators = [property({ reflect: true })];
86
+ _borderless_decorators = [forceType(), property({
87
+ reflect: true,
88
+ type: Boolean
89
+ })];
90
+ _width_decorators = [property({ reflect: true })];
91
+ _hiddenLabel_decorators = [forceType(), property({
92
+ attribute: "hidden-label",
93
+ reflect: true,
94
+ type: Boolean
95
+ })];
96
+ _floatingLabel_decorators = [forceType(), property({
97
+ attribute: "floating-label",
98
+ reflect: true,
99
+ type: Boolean
100
+ })];
101
+ __errorElements_decorators = [state()];
102
+ __input_decorators = [state()];
103
+ __label_decorators = [state()];
104
+ __esDecorate(this, null, _errorSpace_decorators, {
105
+ kind: "accessor",
106
+ name: "errorSpace",
107
+ static: false,
108
+ private: false,
109
+ access: {
110
+ has: (obj) => "errorSpace" in obj,
111
+ get: (obj) => obj.errorSpace,
112
+ set: (obj, value) => {
113
+ obj.errorSpace = value;
114
+ }
115
+ },
116
+ metadata: _metadata
117
+ }, _errorSpace_initializers, _errorSpace_extraInitializers);
118
+ __esDecorate(this, null, _optional_decorators, {
119
+ kind: "accessor",
120
+ name: "optional",
121
+ static: false,
122
+ private: false,
123
+ access: {
124
+ has: (obj) => "optional" in obj,
125
+ get: (obj) => obj.optional,
126
+ set: (obj, value) => {
127
+ obj.optional = value;
128
+ }
129
+ },
130
+ metadata: _metadata
131
+ }, _optional_initializers, _optional_extraInitializers);
132
+ __esDecorate(this, null, _size_decorators, {
133
+ kind: "accessor",
134
+ name: "size",
135
+ static: false,
136
+ private: false,
137
+ access: {
138
+ has: (obj) => "size" in obj,
139
+ get: (obj) => obj.size,
140
+ set: (obj, value) => {
141
+ obj.size = value;
142
+ }
143
+ },
144
+ metadata: _metadata
145
+ }, _size_initializers, _size_extraInitializers);
146
+ __esDecorate(this, null, _borderless_decorators, {
147
+ kind: "accessor",
148
+ name: "borderless",
149
+ static: false,
150
+ private: false,
151
+ access: {
152
+ has: (obj) => "borderless" in obj,
153
+ get: (obj) => obj.borderless,
154
+ set: (obj, value) => {
155
+ obj.borderless = value;
156
+ }
157
+ },
158
+ metadata: _metadata
159
+ }, _borderless_initializers, _borderless_extraInitializers);
160
+ __esDecorate(this, null, _width_decorators, {
161
+ kind: "accessor",
162
+ name: "width",
163
+ static: false,
164
+ private: false,
165
+ access: {
166
+ has: (obj) => "width" in obj,
167
+ get: (obj) => obj.width,
168
+ set: (obj, value) => {
169
+ obj.width = value;
170
+ }
171
+ },
172
+ metadata: _metadata
173
+ }, _width_initializers, _width_extraInitializers);
174
+ __esDecorate(this, null, _hiddenLabel_decorators, {
175
+ kind: "accessor",
176
+ name: "hiddenLabel",
177
+ static: false,
178
+ private: false,
179
+ access: {
180
+ has: (obj) => "hiddenLabel" in obj,
181
+ get: (obj) => obj.hiddenLabel,
182
+ set: (obj, value) => {
183
+ obj.hiddenLabel = value;
184
+ }
185
+ },
186
+ metadata: _metadata
187
+ }, _hiddenLabel_initializers, _hiddenLabel_extraInitializers);
188
+ __esDecorate(this, null, _floatingLabel_decorators, {
189
+ kind: "accessor",
190
+ name: "floatingLabel",
191
+ static: false,
192
+ private: false,
193
+ access: {
194
+ has: (obj) => "floatingLabel" in obj,
195
+ get: (obj) => obj.floatingLabel,
196
+ set: (obj, value) => {
197
+ obj.floatingLabel = value;
198
+ }
199
+ },
200
+ metadata: _metadata
201
+ }, _floatingLabel_initializers, _floatingLabel_extraInitializers);
202
+ __esDecorate(this, null, __errorElements_decorators, {
203
+ kind: "accessor",
204
+ name: "_errorElements",
205
+ static: false,
206
+ private: false,
207
+ access: {
208
+ has: (obj) => "_errorElements" in obj,
209
+ get: (obj) => obj._errorElements,
210
+ set: (obj, value) => {
211
+ obj._errorElements = value;
212
+ }
213
+ },
214
+ metadata: _metadata
215
+ }, __errorElements_initializers, __errorElements_extraInitializers);
216
+ __esDecorate(this, null, __input_decorators, {
217
+ kind: "accessor",
218
+ name: "_input",
219
+ static: false,
220
+ private: false,
221
+ access: {
222
+ has: (obj) => "_input" in obj,
223
+ get: (obj) => obj._input,
224
+ set: (obj, value) => {
225
+ obj._input = value;
226
+ }
227
+ },
228
+ metadata: _metadata
229
+ }, __input_initializers, __input_extraInitializers);
230
+ __esDecorate(this, null, __label_decorators, {
231
+ kind: "accessor",
232
+ name: "_label",
233
+ static: false,
234
+ private: false,
235
+ access: {
236
+ has: (obj) => "_label" in obj,
237
+ get: (obj) => obj._label,
238
+ set: (obj, value) => {
239
+ obj._label = value;
240
+ }
241
+ },
242
+ metadata: _metadata
243
+ }, __label_initializers, __label_extraInitializers);
244
+ if (_metadata) Object.defineProperty(this, Symbol.metadata, {
245
+ enumerable: true,
246
+ configurable: true,
247
+ writable: true,
248
+ value: _metadata
249
+ });
250
+ }
251
+ static {
252
+ this.elementName = "sbb-form-field";
253
+ }
254
+ static {
255
+ this.styles = [boxSizingStyles, unsafeCSS(form_field_default)];
256
+ }
257
+ #errorSpace_accessor_storage;
258
+ /**
259
+ * Whether to reserve space for an error message.
260
+ * `none` does not reserve any space.
261
+ * `reserve` does reserve one row for an error message.
262
+ */
263
+ get errorSpace() {
264
+ return this.#errorSpace_accessor_storage;
265
+ }
266
+ set errorSpace(value) {
267
+ this.#errorSpace_accessor_storage = value;
268
+ }
269
+ #optional_accessor_storage;
270
+ /**
271
+ * Indicates whether the input is optional.
272
+ * @deprecated Set the (optional) label text manually. Will be removed with next major version.
273
+ */
274
+ get optional() {
275
+ return this.#optional_accessor_storage;
276
+ }
277
+ set optional(value) {
278
+ this.#optional_accessor_storage = value;
279
+ }
280
+ #size_accessor_storage;
281
+ /**
282
+ * Size variant, either l, m or s.
283
+ * @default 'm' / 's' (lean)
284
+ */
285
+ get size() {
286
+ return this.#size_accessor_storage;
287
+ }
288
+ set size(value) {
289
+ this.#size_accessor_storage = value;
290
+ }
291
+ #borderless_accessor_storage;
292
+ /** Whether to display the form field without a border. */
293
+ get borderless() {
294
+ return this.#borderless_accessor_storage;
295
+ }
296
+ set borderless(value) {
297
+ this.#borderless_accessor_storage = value;
298
+ }
299
+ #width_accessor_storage;
300
+ /** Defines the width of the component:
301
+ * - `default`: the component has defined width and min-width;
302
+ * - `collapse`: the component adapts itself to its inner input content. */
303
+ get width() {
304
+ return this.#width_accessor_storage;
305
+ }
306
+ set width(value) {
307
+ this.#width_accessor_storage = value;
308
+ }
309
+ #hiddenLabel_accessor_storage;
310
+ /** Whether to visually hide the label. If hidden, screen readers will still read it. */
311
+ get hiddenLabel() {
312
+ return this.#hiddenLabel_accessor_storage;
313
+ }
314
+ set hiddenLabel(value) {
315
+ this.#hiddenLabel_accessor_storage = value;
316
+ }
317
+ #floatingLabel_accessor_storage;
318
+ /** Whether the label should float. If activated, the placeholder of the input is hidden. */
319
+ get floatingLabel() {
320
+ return this.#floatingLabel_accessor_storage;
321
+ }
322
+ set floatingLabel(value) {
323
+ this.#floatingLabel_accessor_storage = value;
324
+ }
325
+ #_errorElements_accessor_storage;
326
+ /** It is used internally to get the `error` slot. */
327
+ get _errorElements() {
328
+ return this.#_errorElements_accessor_storage;
329
+ }
330
+ set _errorElements(value) {
331
+ this.#_errorElements_accessor_storage = value;
332
+ }
333
+ #_input_accessor_storage;
334
+ /** Reference to the slotted input element. */
335
+ get _input() {
336
+ return this.#_input_accessor_storage;
337
+ }
338
+ set _input(value) {
339
+ this.#_input_accessor_storage = value;
340
+ }
341
+ #_label_accessor_storage;
342
+ /** Reference to the slotted label elements. */
343
+ get _label() {
344
+ return this.#_label_accessor_storage;
345
+ }
346
+ set _label(value) {
347
+ this.#_label_accessor_storage = value;
348
+ }
349
+ /** Returns the input element. */
350
+ get inputElement() {
351
+ return this._input;
352
+ }
353
+ /** Reference to the slotted label. */
354
+ get label() {
355
+ return this._label;
356
+ }
357
+ constructor() {
358
+ super();
359
+ this._excludedFocusElements = [
360
+ "button",
361
+ "sbb-popover",
362
+ "sbb-option",
363
+ "sbb-chip"
364
+ ];
365
+ this._floatingLabelSupportedInputElements = [
366
+ "input",
367
+ "select",
368
+ "textarea",
369
+ "sbb-select"
370
+ ];
371
+ this._floatingLabelSupportedInputTypes = [
372
+ "email",
373
+ "number",
374
+ "password",
375
+ "search",
376
+ "tel",
377
+ "text",
378
+ "url"
379
+ ];
380
+ this.#errorSpace_accessor_storage = __runInitializers(this, _errorSpace_initializers, "none");
381
+ this.#optional_accessor_storage = (__runInitializers(this, _errorSpace_extraInitializers), __runInitializers(this, _optional_initializers, false));
382
+ this.#size_accessor_storage = (__runInitializers(this, _optional_extraInitializers), __runInitializers(this, _size_initializers, isLean() ? "s" : "m"));
383
+ this.#borderless_accessor_storage = (__runInitializers(this, _size_extraInitializers), __runInitializers(this, _borderless_initializers, false));
384
+ this.#width_accessor_storage = (__runInitializers(this, _borderless_extraInitializers), __runInitializers(this, _width_initializers, "default"));
385
+ this.#hiddenLabel_accessor_storage = (__runInitializers(this, _width_extraInitializers), __runInitializers(this, _hiddenLabel_initializers, false));
386
+ this.#floatingLabel_accessor_storage = (__runInitializers(this, _hiddenLabel_extraInitializers), __runInitializers(this, _floatingLabel_initializers, false));
387
+ this.#_errorElements_accessor_storage = (__runInitializers(this, _floatingLabel_extraInitializers), __runInitializers(this, __errorElements_initializers, []));
388
+ this.#_input_accessor_storage = (__runInitializers(this, __errorElements_extraInitializers), __runInitializers(this, __input_initializers, null));
389
+ this.#_label_accessor_storage = (__runInitializers(this, __input_extraInitializers), __runInitializers(this, __label_initializers, void 0));
390
+ this._language = (__runInitializers(this, __label_extraInitializers), new SbbLanguageController(this));
391
+ /**
392
+ * Listens to the changes on `readonly` and `disabled` attributes of `<input>`.
393
+ */
394
+ this._formFieldAttributeObserver = !isServer ? new MutationObserver((mutations) => {
395
+ if (mutations.some((m) => m.type === "attributes") && this._input) {
396
+ this._readInputState();
397
+ this._registerInputFormListener();
398
+ this._checkAndUpdateInputEmpty();
399
+ }
400
+ }) : null;
401
+ this._inputFormAbortController = new AbortController();
402
+ this._control = null;
403
+ this.addEventListener?.("focusin", (event) => {
404
+ if (event.target === this.inputElement || event.target === this.inputElement?.inputElement) {
405
+ this.internals.states.add("focus");
406
+ this.internals.states.add(`focus-origin-${sbbInputModalityDetector.mostRecentModality}`);
407
+ }
408
+ }, { passive: true });
409
+ this.addEventListener?.("focusout", (event) => {
410
+ if (event.target === this.inputElement || event.target === this.inputElement?.inputElement) {
411
+ this._checkAndUpdateInputEmpty();
412
+ this.internals.states.delete("focus");
413
+ for (const state of this.internals.states) if (state.startsWith("focus-origin-")) this.internals.states.delete(state);
414
+ }
415
+ }, { passive: true });
416
+ this.addEventListener("input", () => this._checkAndUpdateInputEmpty());
417
+ this.addEventListener("displayvaluechange", () => this._checkAndUpdateInputEmpty());
418
+ this.addEventListener("invalid", (e) => e.preventDefault(), { capture: true });
419
+ this.addEventListener("formfieldcontrol", (e) => {
420
+ this._control = e.control;
421
+ if (this._connectInputElement() === "unchanged") {
422
+ this._assignErrorMessageElements();
423
+ this._readInputState();
424
+ this._checkAndUpdateInputEmpty();
425
+ }
426
+ });
427
+ }
428
+ connectedCallback() {
429
+ super.connectedCallback();
430
+ this._assignSlots();
431
+ this._connectInputElement();
432
+ this._syncNegative();
433
+ this._syncSize();
434
+ }
435
+ willUpdate(changedProperties) {
436
+ super.willUpdate(changedProperties);
437
+ if (changedProperties.has("negative")) this._syncNegative();
438
+ if (changedProperties.has("size")) this._syncSize();
439
+ }
440
+ disconnectedCallback() {
441
+ super.disconnectedCallback();
442
+ this._formFieldAttributeObserver?.disconnect();
443
+ this._inputFormAbortController.abort();
444
+ if (this._input?.localName === "input") this._unpatchInputValue();
445
+ }
446
+ _handleWrapperClick(event) {
447
+ if (this._isElementFocusExcluded(event)) return;
448
+ if (this._control?.onContainerClick) this._control?.onContainerClick(event);
449
+ else if (event.target.localName !== "label") {
450
+ if (this._input?.localName === "sbb-select" && event.target.localName !== "sbb-select") this._input.click();
451
+ this._input?.focus();
452
+ }
453
+ }
454
+ _isElementFocusExcluded(event) {
455
+ return event.composedPath().some((el) => el instanceof window.HTMLElement && (el.getAttribute("tabindex") === "0" || el.hasAttribute("contenteditable")) || this._excludedFocusElements.includes(el.localName));
456
+ }
457
+ _onSlotLabelChange() {
458
+ this._label = Array.from(this.querySelectorAll("label"))[0];
459
+ this._syncLabelInputReferences();
460
+ }
461
+ /**
462
+ * It is used internally to assign the attributes of `<input>` to `_id` and `_input` and to observe the native readonly and disabled attributes.
463
+ */
464
+ _onSlotInputChange() {
465
+ this._assignSlots();
466
+ this._connectInputElement();
467
+ }
468
+ _assignSlots() {
469
+ this.querySelectorAll("label:not([slot])").forEach((e) => e.setAttribute("slot", "label"));
470
+ this.querySelectorAll("sbb-error:not([slot])").forEach((e) => e.setAttribute("slot", "error"));
471
+ }
472
+ _connectInputElement() {
473
+ let newInput;
474
+ if (this._control?.id) newInput = this.getRootNode().getElementById(this._control.id);
475
+ else {
476
+ const inputCandidates = Array.from(this.querySelectorAll("*:not([slot],sbb-chip-group)"));
477
+ newInput = (inputCandidates.find((e) => this._isInputElement(e)) || inputCandidates[0]) ?? null;
478
+ }
479
+ if (newInput === this._input) return "unchanged";
480
+ else if (this._input) {
481
+ this.internals.states.delete(`input-type-${this._input.localName}`);
482
+ if (this._input.localName === "input") this._unpatchInputValue();
483
+ }
484
+ if (!newInput) {
485
+ this._input = null;
486
+ return "no-input";
487
+ }
488
+ this._input = newInput;
489
+ this._registerInputFormListener();
490
+ this._assignErrorMessageElements();
491
+ this._readInputState();
492
+ this._checkAndUpdateInputEmpty();
493
+ if (this._input.localName === "textarea") this._input.setAttribute("rows", this._input.getAttribute("rows") || "3");
494
+ else if (this._input.localName === "input") this._patchInputValue();
495
+ else if ((this._input.localName === "select" || this._input.localName === "sbb-select") && !this.hasUpdated) this.hydrationComplete.then(() => this.requestUpdate());
496
+ this._formFieldAttributeObserver?.disconnect();
497
+ this._formFieldAttributeObserver?.observe(this._input, {
498
+ attributes: true,
499
+ attributeFilter: [
500
+ "readonly",
501
+ "disabled",
502
+ "form",
503
+ "class",
504
+ "data-expanded"
505
+ ]
506
+ });
507
+ this.internals.states.add(`input-type-${this._input.localName}`);
508
+ this._syncLabelInputReferences();
509
+ return "changed";
510
+ }
511
+ _syncLabelInputReferences() {
512
+ if (!this._input || !this._label) return;
513
+ if (nativeInputElements.includes(this._input.localName) || customElements.get(this._input.localName)?.formAssociated) {
514
+ this._input.id ||= `sbb-form-field-input-${nextId++}`;
515
+ this._label.htmlFor = this._input.id;
516
+ } else {
517
+ this._label.id ||= `sbb-form-field-label-${nextId++}`;
518
+ const labelledby = this._input.getAttribute("aria-labelledby")?.split(" ").filter((l) => !!l && l !== this._label.id) ?? [];
519
+ this._input.setAttribute("aria-labelledby", [...labelledby, this._label.id].join(" "));
520
+ }
521
+ }
522
+ _isInputElement(input) {
523
+ return nativeInputElements.includes(input.localName) || !!customElements.get(input.localName)?.formAssociated;
524
+ }
525
+ _readInputState() {
526
+ this.toggleState("readonly", this._control?.readOnly ?? this._input.hasAttribute("readonly"));
527
+ this.toggleState("disabled", this._control?.disabled ?? this._input.hasAttribute("disabled"));
528
+ this.toggleState("has-popup-open", this._input.hasAttribute("data-expanded"));
529
+ }
530
+ _registerInputFormListener() {
531
+ this._inputFormAbortController.abort();
532
+ const { signal } = this._inputFormAbortController = new AbortController();
533
+ const form = this._input?.form ?? this._input?.closest("form");
534
+ const resetHandler = () => setTimeout(() => this.reset());
535
+ form?.addEventListener("reset", resetHandler, { signal });
536
+ }
537
+ _patchInputValue() {
538
+ const inputElement = this._input;
539
+ if (!inputElement || patchedInputs.has(inputElement)) return;
540
+ const originalDescriptor = Object.getOwnPropertyDescriptor(Object.getPrototypeOf(inputElement), "value");
541
+ if (!originalDescriptor || !originalDescriptor.set || !originalDescriptor.get) return;
542
+ patchedInputs.set(inputElement, originalDescriptor);
543
+ const { get: getter, set: setter } = originalDescriptor;
544
+ const checkAndUpdateInputEmpty = () => this._checkAndUpdateInputEmpty();
545
+ Object.defineProperty(inputElement, "value", {
546
+ ...originalDescriptor,
547
+ get() {
548
+ return getter.call(this);
549
+ },
550
+ set(newValue) {
551
+ setter.call(this, newValue);
552
+ checkAndUpdateInputEmpty();
553
+ }
554
+ });
555
+ }
556
+ _unpatchInputValue() {
557
+ const inputElement = this._input;
558
+ const originalDescriptor = patchedInputs.get(inputElement);
559
+ if (originalDescriptor) {
560
+ Object.defineProperty(inputElement, "value", originalDescriptor);
561
+ patchedInputs.delete(inputElement);
562
+ }
563
+ }
564
+ _checkAndUpdateInputEmpty() {
565
+ this.toggleState("empty", this._control?.empty ?? (((this._floatingLabelSupportedInputElements.includes(this._input?.localName) || this._input?.constructor?.formFieldAssociated) ?? false) && this._isInputEmpty()));
566
+ }
567
+ _isInputEmpty() {
568
+ const chipGroupElem = this.querySelector("sbb-chip-group");
569
+ if (chipGroupElem) return this._isInputValueEmpty() && (Array.isArray(chipGroupElem.value) ? chipGroupElem.value.length === 0 : !chipGroupElem.querySelector("sbb-chip"));
570
+ else if (this._input instanceof HTMLInputElement) return this._floatingLabelSupportedInputTypes.includes(this._input.type) && this._isInputValueEmpty();
571
+ else if (this._input instanceof HTMLSelectElement) return this._input.selectedOptions?.item(0)?.label?.trim() === "";
572
+ else if (this._input?.localName === "sbb-select") return this._input.getDisplayValue?.()?.trim() === "";
573
+ else return this._isInputValueEmpty();
574
+ }
575
+ _isInputValueEmpty() {
576
+ const value = this._input.value;
577
+ return [
578
+ "",
579
+ void 0,
580
+ null
581
+ ].includes(value) || Array.isArray(value) && value.length === 0;
582
+ }
583
+ /**
584
+ * It is used internally to set the aria-describedby attribute for the slotted input referencing available <sbb-error> instances.
585
+ */
586
+ _onSlotErrorChange(event) {
587
+ const errorElements = event.target.assignedElements();
588
+ if (this._input && this._input.ariaDescribedByElements?.length) this._input.ariaDescribedByElements = removeAriaElements(this._input.ariaDescribedByElements, ...this._errorElements ?? []);
589
+ this._errorElements = errorElements;
590
+ for (const el of this._errorElements) el.role ||= "status";
591
+ this._assignErrorMessageElements();
592
+ this.toggleState("has-error", !!this._errorElements.length);
593
+ this._syncNegative();
594
+ }
595
+ _assignErrorMessageElements() {
596
+ if (this._input) this._input.ariaDescribedByElements = appendAriaElements(this._input.ariaDescribedByElements, ...this._errorElements);
597
+ }
598
+ /** Manually reset the form field. Currently, this only resets the floating label. */
599
+ reset() {
600
+ this._checkAndUpdateInputEmpty();
601
+ }
602
+ /** Manually clears the input value. It only works for inputs, selects are not supported. */
603
+ clear() {
604
+ if (this._input?.localName !== "input") return;
605
+ this._input.value = "";
606
+ this._checkAndUpdateInputEmpty();
607
+ }
608
+ _syncNegative() {
609
+ this.querySelectorAll?.("sbb-error,sbb-mini-button,sbb-mini-button-link,sbb-form-field-clear,sbb-datepicker-next-day,sbb-datepicker-previous-day,sbb-datepicker-toggle,sbb-select,sbb-autocomplete,sbb-autocomplete-grid,sbb-chip-group").forEach((element) => element.toggleAttribute("negative", this.negative));
610
+ }
611
+ _syncSize() {
612
+ this.querySelectorAll?.("sbb-autocomplete,sbb-autocomplete-grid,sbb-select").forEach((element) => element.size = this.size === "s" ? "s" : "m");
613
+ }
614
+ render() {
615
+ return html`
616
+ <div class="sbb-form-field__space-wrapper">
617
+ ${""}
618
+ <div @click=${this._handleWrapperClick} class="sbb-form-field__wrapper" id="overlay-anchor">
619
+ <slot name="prefix" @slotchange=${this._syncNegative}></slot>
620
+ <div class="sbb-form-field__input-container">
621
+ <span class="sbb-form-field__label-spacer" aria-hidden="true"></span>
622
+ <span class="sbb-form-field__label">
623
+ <span class="sbb-form-field__label-ellipsis">
624
+ <slot name="label" @slotchange=${this._onSlotLabelChange}></slot>
625
+ ${this.optional ? html` <span aria-hidden="true"> ${i18nOptional[this._language.current]} </span>` : nothing}
626
+ </span>
627
+ </span>
628
+ <div class="sbb-form-field__input">
629
+ <slot @slotchange=${this._onSlotInputChange}></slot>
630
+ </div>
631
+ ${this.hasUpdated && ["select", "sbb-select"].includes(this._input?.localName) ? html`<sbb-icon
632
+ name="chevron-small-down-small"
633
+ class="sbb-form-field__select-input-icon"
634
+ ></sbb-icon>` : nothing}
635
+ </div>
636
+ <slot name="suffix" @slotchange=${this._syncNegative}></slot>
637
+ </div>
638
+
639
+ <div class="sbb-form-field__error">
640
+ <slot name="error" @slotchange=${this._onSlotErrorChange}></slot>
641
+ </div>
642
+ </div>
643
+ `;
644
+ }
645
+ };
646
+ })();
647
+ //#endregion
648
+ export { SbbFormFieldElement as n, SbbFormFieldControlEvent as t };
649
+
650
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"form-field.component-DnUA6Uds.js","names":[],"sources":["../../../src/elements/form-field/form-field/form-field.scss?inline","../../../src/elements/form-field/form-field/form-field.component.ts"],"sourcesContent":["@use '../../core/styles' as sbb;\n\n:host {\n  --sbb-form-field-error-divider-width: var(--sbb-spacing-fixed-1x);\n  --sbb-form-field-error-padding-block-start: var(--sbb-form-field-error-divider-width);\n  --sbb-form-field-select-inline-padding-end: calc(\n    var(--sbb-icon-svg-width) + var(--sbb-form-field-gap)\n  );\n  --sbb-form-field-label-size: calc(\n    var(--sbb-form-field-label-text-size) * var(--sbb-typo-line-height-text)\n  );\n  --sbb-form-field-text-line-height: calc(\n    var(--sbb-form-field-input-text-size) * var(--sbb-typo-line-height-text)\n  );\n  --sbb-form-field-margin-block-start: calc(\n    (\n        var(--sbb-form-field-min-height) - var(--sbb-form-field-label-size) - var(\n            --sbb-form-field-text-line-height\n          ) +\n          var(--_sbb-form-field-label-to-input-overlapping)\n      ) /\n      2\n  );\n\n  // Lock sbb-icon size\n  --sbb-icon-svg-width: var(--sbb-form-field-icon-size);\n  --sbb-icon-svg-height: var(--sbb-form-field-icon-size);\n\n  // As the form field has always a white background, we have to fix the focus outline color\n  // to default color for cases where the form field is used in a negative context.\n  --sbb-focus-outline-color: light-dark(\n    var(--sbb-focus-outline-color-default),\n    var(--sbb-focus-outline-color-dark)\n  );\n\n  display: inline-block;\n  color: var(--sbb-form-field-color);\n  font-weight: normal;\n}\n\n:host(:where(:not([width='collapse']))) {\n  min-width: #{sbb.px-to-rem-build(150)};\n  width: min(#{sbb.px-to-rem-build(300)}, 100%);\n}\n\n:host([negative]) {\n  color: var(--sbb-color-3-negative);\n\n  --sbb-form-field-background-color: var(--sbb-background-color-1-negative);\n  --sbb-form-field-border-color: var(--sbb-border-color-5);\n  --sbb-form-field-label-color: var(--sbb-color-5);\n  --sbb-form-field-prefix-color: var(--sbb-color-5);\n  --sbb-form-field-text-color: var(--sbb-color-3-negative);\n  --sbb-focus-outline-color: var(--sbb-focus-outline-color-dark);\n  --sbb-form-field-arrow-color: var(--sbb-color-3-negative);\n\n  @include sbb.if-forced-colors {\n    --sbb-form-field-border-color: ButtonBorder;\n  }\n}\n\n:host([size='s']) {\n  --sbb-form-field-min-height: var(--sbb-size-element-xs);\n  --sbb-form-field-padding-inline: var(--sbb-spacing-fixed-2x);\n  --sbb-form-field-input-text-size: var(--sbb-text-font-size-s);\n  --sbb-form-field-label-text-size: var(--sbb-text-font-size-xxs);\n\n  // Values found by try and error\n  --_sbb-form-field-label-to-input-overlapping: #{sbb.px-to-rem-build(10)};\n  --_sbb-form-field-floating-label-transform: #{sbb.px-to-rem-build(5.5)};\n  --_sbb-form-field-spacer-margin-block-end: #{sbb.px-to-rem-build(-8.5)};\n\n  @include sbb.mq($from: large) {\n    // Values found by try and error\n    --_sbb-form-field-label-to-input-overlapping: #{sbb.px-to-rem-build(11)};\n    --_sbb-form-field-floating-label-transform: #{sbb.px-to-rem-build(5)};\n    --_sbb-form-field-spacer-margin-block-end: #{sbb.px-to-rem-build(-8)};\n  }\n}\n\n:host([size='m']) {\n  --sbb-form-field-min-height: var(--sbb-size-element-m);\n  --sbb-form-field-padding-inline: var(--sbb-spacing-fixed-3x);\n  --sbb-form-field-input-text-size: var(--sbb-text-font-size-m);\n  --sbb-form-field-label-text-size: var(--sbb-text-font-size-xs);\n  --_sbb-form-field-label-to-input-overlapping: var(--sbb-spacing-fixed-1x);\n  --_sbb-form-field-floating-label-transform: #{sbb.px-to-rem-build(8.5)};\n  --_sbb-form-field-spacer-margin-block-end: calc(\n    -1 * var(--_sbb-form-field-label-to-input-overlapping)\n  );\n\n  @include sbb.mq($from: large) {\n    --_sbb-form-field-floating-label-transform: #{sbb.px-to-rem-build(10.5)};\n  }\n}\n\n:host([size='l']) {\n  --sbb-form-field-min-height: var(--sbb-size-element-l);\n  --sbb-form-field-padding-inline: var(--sbb-spacing-responsive-xxxs);\n}\n\n:host([error-space='reserve']) {\n  --sbb-form-field-error-min-height: calc(\n    var(--sbb-typo-line-height-text) * var(--sbb-text-font-size-xs)\n  );\n  --sbb-form-field-error-padding-block-start-override: var(--sbb-form-field-error-divider-width);\n}\n\n:host(:where(:state(readonly), :state(disabled))) {\n  --sbb-form-field-background-color: var(--sbb-background-color-3);\n  --sbb-form-field-border-color: light-dark(var(--sbb-color-graphite), var(--sbb-color-smoke));\n  --sbb-form-field-arrow-color: light-dark(var(--sbb-color-granite), var(--sbb-color-storm));\n\n  @include sbb.if-forced-colors {\n    --sbb-form-field-border-color: ButtonBorder;\n  }\n}\n\n:host(:where(:state(readonly), :state(disabled))[negative]) {\n  --sbb-form-field-background-color: var(--sbb-background-color-3-negative);\n  --sbb-form-field-border-color: var(--sbb-border-color-5);\n  --sbb-form-field-arrow-color: var(--sbb-border-color-5);\n}\n\n:host(:where(:state(disabled))) {\n  --sbb-form-field-label-color: light-dark(var(--sbb-color-granite), var(--sbb-color-smoke));\n  --sbb-form-field-prefix-color: light-dark(var(--sbb-color-granite), var(--sbb-color-storm));\n  --sbb-form-field-border-style: dashed;\n\n  @include sbb.if-forced-colors {\n    --sbb-form-field-label-color: GrayText !important;\n    --sbb-form-field-prefix-color: GrayText !important;\n    --sbb-form-field-text-color: GrayText !important;\n    --sbb-form-field-border-color: GrayText !important;\n  }\n}\n\n// :where is to align specificity\n:host(:where(:state(disabled))[negative]) {\n  --sbb-form-field-text-color: var(--sbb-color-5);\n  --sbb-form-field-label-color: var(--sbb-color-5);\n  --sbb-form-field-prefix-color: var(--sbb-color-5);\n}\n\n// :where is to align specificity\n:host(:where(:state(readonly)):not([negative])) {\n  --sbb-form-field-label-color: light-dark(var(--sbb-color-granite), var(--sbb-color-smoke));\n  --sbb-form-field-prefix-color: light-dark(var(--sbb-color-granite), var(--sbb-color-storm));\n}\n\n// :where is to align specificity\n:host(:where(:state(focus))) {\n  --sbb-form-field-border-color: var(--sbb-border-color-3);\n  --sbb-form-field-prefix-color: var(--sbb-color-3);\n  --sbb-form-field-border-width: var(--sbb-border-width-2x);\n}\n\n// :where is to align specificity\n:host(:where(:state(focus))[negative]) {\n  --sbb-form-field-border-color: var(--sbb-border-color-3-negative);\n  --sbb-form-field-prefix-color: var(--sbb-color-3-negative);\n}\n\n// We must not use :where here, as it needs the added specificity to properly work\n:host(:state(focus)) {\n  @include sbb.if-forced-colors {\n    --sbb-form-field-border-color: Highlight;\n    --sbb-form-field-prefix-color: Highlight;\n  }\n}\n\n// :where is to align specificity\n:host(:not(:where(:state(has-error)))) {\n  --sbb-form-field-error-padding-block-start: 0;\n}\n\n:host([floating-label]) {\n  --sbb-select-placeholder-color: transparent;\n\n  @include sbb.if-forced-colors {\n    --sbb-select-placeholder-color: Canvas;\n  }\n}\n\n:host(:is(:not(:state(slotted-label)), [hidden-label])) {\n  --sbb-form-field-label-size: 0rem;\n  --_sbb-form-field-label-to-input-overlapping: 0rem;\n}\n\n// Should be after other definitions to override overflow\n// :where is to align specificity\n:host(:where(:state(input-type-sbb-slider))) {\n  --sbb-form-field-overflow: visible;\n}\n\n.sbb-form-field__space-wrapper {\n  display: flex;\n  flex-direction: column;\n  height: 100%;\n\n  @include sbb.zero-width-space;\n}\n\n.sbb-form-field__wrapper {\n  display: flex;\n  flex: 1;\n  gap: var(--sbb-form-field-gap);\n  padding-inline: var(--sbb-form-field-padding-inline);\n  border-radius: var(--sbb-form-field-border-radius);\n  min-height: var(--sbb-form-field-min-height);\n  max-height: var(--sbb-form-field-max-height);\n  background-color: var(--sbb-form-field-background-color);\n  position: relative;\n\n  // Use ::before to avoid content shifting when border width changes.\n  &::before {\n    content: '';\n    display: block;\n    position: absolute;\n    inset: 0;\n    border: var(--sbb-form-field-border-width) var(--sbb-form-field-border-style)\n      var(--sbb-form-field-border-color);\n    border-radius: var(--sbb-form-field-border-radius);\n    pointer-events: none;\n  }\n\n  // :where is to align specificity\n  :host(:where(:state(focus):state(focus-origin-keyboard))) & {\n    @include sbb.focus-outline;\n\n    outline-offset: var(--sbb-form-field-outline-offset, var(--sbb-focus-outline-offset));\n  }\n\n  // In high contrast, there is no borderless variant\n  @media (forced-colors: none) {\n    :host(:is([borderless], :state(input-type-sbb-slider))) & {\n      &::before {\n        border-color: transparent;\n      }\n    }\n\n    :host(:where(:state(focus), :state(disabled))[borderless]) & {\n      &::after {\n        content: '';\n        position: absolute;\n        border-block-end: var(--sbb-border-width-1x) var(--sbb-form-field-border-style)\n          var(--sbb-form-field-border-color);\n        inset-inline: var(--sbb-form-field-padding-inline);\n        inset-block-end: 0;\n        z-index: var(--sbb-form-field-focus-underline-z-index, initial);\n      }\n    }\n  }\n}\n\n::slotted(*[slot='prefix']) {\n  color: var(--sbb-form-field-prefix-color);\n}\n\n// Ensure slotted inline elements are vertically centered\n::slotted(*[slot='prefix']),\n::slotted(*[slot='suffix']) {\n  display: flex;\n  min-width: var(--sbb-icon-svg-width);\n  margin-block-start: calc((var(--sbb-form-field-min-height) - var(--sbb-size-icon-ui-small)) / 2);\n}\n\n::slotted(sbb-form-field-clear) {\n  :host(:where(:state(empty), :state(disabled), :state(readonly))) & {\n    display: none;\n  }\n}\n\n// As the calendar should be shown below the form field border, we have to stretch the toggle's height.\n// TODO: Maybe move this to the datepicker-toggle component.\n::slotted(sbb-datepicker-toggle) {\n  padding-block-end: calc((var(--sbb-form-field-min-height) - var(--sbb-size-icon-ui-small)) / 2);\n}\n\n@include sbb.if-forced-colors {\n  // Align with prefix color\n  ::slotted(*[slot='suffix']) {\n    color: var(--sbb-form-field-prefix-color);\n  }\n}\n\n.sbb-form-field__select-input-icon {\n  @include sbb.absolute-center-y;\n\n  position: absolute;\n  inset-inline-end: 0;\n  margin-block-start: calc(-1 * var(--sbb-form-field-margin-block-start) / 2);\n  pointer-events: none;\n  color: var(--sbb-form-field-arrow-color);\n}\n\n.sbb-form-field__input-container {\n  display: flex;\n  flex-direction: column;\n  flex: 1 1 auto;\n  position: relative;\n  margin-block-start: var(--sbb-form-field-margin-block-start);\n\n  // Prevents overflowing parent\n  min-width: 0;\n}\n\n.sbb-form-field__label-spacer {\n  display: flex;\n  height: calc(var(--sbb-form-field-label-text-size) * var(--sbb-typo-line-height-text));\n\n  // Moves label down and input up to meet positioning requirements\n  margin-block-end: var(--_sbb-form-field-spacer-margin-block-end);\n}\n\n// To avoid doubled payload, we group the rules.\n:is(.sbb-form-field__label, .sbb-form-field__label-spacer) {\n  :host(:not(:state(slotted-label))) & {\n    display: none;\n  }\n\n  :host([hidden-label]) & {\n    @include sbb.screen-reader-only;\n  }\n}\n\n.sbb-form-field__label {\n  display: flex;\n  max-width: 100%;\n  cursor: var(--sbb-cursor-default);\n  position: absolute;\n  inset-block-start: 0;\n  color: var(--sbb-form-field-label-color);\n  font-size: var(--sbb-form-field-label-text-size);\n  letter-spacing: var(--sbb-typo-letter-spacing-text);\n\n  :host(:where(:state(input-type-select), :state(input-type-sbb-select))) & {\n    padding-inline-end: var(--sbb-form-field-select-inline-padding-end);\n  }\n\n  :host([floating-label]) & {\n    transform-origin: 0 0;\n    pointer-events: none; // We shouldn't catch mouse events (let them through).\n    backface-visibility: hidden;\n    will-change: transform, font-size;\n\n    transition: {\n      duration: var(--sbb-disable-animation-duration, var(--sbb-animation-duration-2x));\n      timing-function: var(--sbb-animation-easing);\n      property: transform, font-size;\n    }\n  }\n\n  // If floating-label is activated and there is no focus in it (except for select) and no popup is open\n  // and input is empty then apply the label transition.\n  // If it is empty and readonly, always apply transition\n  // :where is to align specificity\n  :host(\n      [floating-label]:is(\n          :not(\n            :where(:state(focus)):not(:where(:state(input-type-sbb-select))),\n            :where(:state(has-popup-open))\n          ),\n          :where(:state(readonly))\n        ):where(:state(empty))\n    )\n    & {\n    font-size: var(--sbb-form-field-input-text-size);\n    transform: translateY(var(--_sbb-form-field-floating-label-transform));\n  }\n\n  // :where is to align specificity\n  :host(:where(:state(input-type-textarea))) & {\n    @include sbb.if-forced-colors {\n      // The background of the textarea is set to Canvas and therefore hides the label behind.\n      // By setting the z-index to 1, we move the label in front of the textarea.\n      z-index: 1;\n    }\n  }\n}\n\n.sbb-form-field__label-ellipsis {\n  @include sbb.ellipsis;\n}\n\n.sbb-form-field__input {\n  display: flex;\n  flex: 1;\n  align-items: start;\n\n  :host(\n      [size='s']:is(\n        :state(input-type-input),\n        :state(input-type-select),\n        :state(input-type-sbb-select),\n        :state(input-type-sbb-date-input),\n        :state(input-type-sbb-time-input)\n      )\n    )\n    & {\n    // In size s, the natural height of the text input is too small.\n    // To not reserve too much space, we decrease the height.\n    margin-block-end: #{sbb.px-to-rem-build(-2)};\n  }\n}\n\n.sbb-form-field__error {\n  display: flex;\n  min-height: var(--sbb-form-field-error-min-height);\n  flex-direction: column;\n  margin-block-start: var(\n    --sbb-form-field-error-padding-block-start-override,\n    var(--sbb-form-field-error-padding-block-start)\n  );\n}\n","import {\n  type CSSResultGroup,\n  html,\n  isServer,\n  nothing,\n  type PropertyValues,\n  type TemplateResult,\n  unsafeCSS,\n} from 'lit';\nimport { property, state } from 'lit/decorators.js';\n\nimport type { SbbAutocompleteBaseElement } from '../../autocomplete.ts';\nimport type { SbbChipGroupElement } from '../../chip.ts';\nimport { sbbInputModalityDetector } from '../../core/a11y.ts';\nimport { SbbElement } from '../../core/base-elements.ts';\nimport { SbbLanguageController } from '../../core/controllers.ts';\nimport { forceType } from '../../core/decorators.ts';\nimport { isLean } from '../../core/dom.ts';\nimport { i18nOptional } from '../../core/i18n.ts';\nimport {\n  appendAriaElements,\n  removeAriaElements,\n  type SbbFormAssociatedInputMixinType,\n  SbbNegativeMixin,\n} from '../../core/mixins.ts';\nimport { boxSizingStyles } from '../../core/styles.ts';\nimport type { SbbSelectElement } from '../../select.ts';\n\nimport style from './form-field.scss?inline';\n\nimport '../../icon.ts';\n\nlet nextId = 0;\n\nconst patchedInputs = new WeakMap<HTMLInputElement, PropertyDescriptor>();\nconst nativeInputElements = ['input', 'textarea', 'select'];\n\n/** An interface which allows a control to work inside a `SbbFormField`. */\nexport interface SbbFormFieldElementControl {\n  /** The id of the form field control. */\n  readonly id: string;\n  /** Whether the control is empty. */\n  readonly empty: boolean;\n  /** Whether the control is readonly. */\n  readonly readOnly?: boolean;\n  /** Whether the control is disabled. */\n  readonly disabled: boolean;\n\n  /**\n   * Handles a click on the control's container.\n   * If not implemented, focus() of the element is called.\n   */\n  onContainerClick?(event: MouseEvent): void;\n}\n\nexport class SbbFormFieldControlEvent extends Event {\n  private _control: SbbFormFieldElementControl | null;\n\n  public get control(): SbbFormFieldElementControl | null {\n    return this._control;\n  }\n\n  public constructor(control: SbbFormFieldElementControl | null) {\n    super('formfieldcontrol');\n    this._control = control;\n  }\n}\n\n/**\n * It wraps an input element adding label, errors, icon, etc.\n *\n * @slot - Use this slot to render an input/select or a supported non-native element.\n * @slot label - Use this slot to render a label.\n * @slot prefix - Use this slot to render an icon on the left side of the input.\n * @slot suffix - Use this slot to render an icon on the right side of the input.\n * @slot error - Use this slot to render an error.\n *\n * @cssprop [--sbb-form-field-outline-offset] - To override the focus outline offset,\n * @cssprop [--sbb-form-field-focus-underline-z-index] - To override the z-index of the focus underline effect,\n */\nexport class SbbFormFieldElement extends SbbNegativeMixin(SbbElement) {\n  public static override readonly elementName: string = 'sbb-form-field';\n  public static override styles: CSSResultGroup = [boxSizingStyles, unsafeCSS(style)];\n\n  // List of elements that should not focus input on click\n  private readonly _excludedFocusElements = ['button', 'sbb-popover', 'sbb-option', 'sbb-chip'];\n\n  private readonly _floatingLabelSupportedInputElements = [\n    'input',\n    'select',\n    'textarea',\n    'sbb-select',\n  ];\n\n  private readonly _floatingLabelSupportedInputTypes = [\n    'email',\n    'number',\n    'password',\n    'search',\n    'tel',\n    'text',\n    'url',\n  ];\n\n  /**\n   * Whether to reserve space for an error message.\n   * `none` does not reserve any space.\n   * `reserve` does reserve one row for an error message.\n   */\n  @property({ attribute: 'error-space', reflect: true })\n  public accessor errorSpace: 'none' | 'reserve' = 'none';\n\n  /**\n   * Indicates whether the input is optional.\n   * @deprecated Set the (optional) label text manually. Will be removed with next major version.\n   */\n  @forceType()\n  @property({ type: Boolean })\n  public accessor optional: boolean = false;\n\n  /**\n   * Size variant, either l, m or s.\n   * @default 'm' / 's' (lean)\n   */\n  @property({ reflect: true }) public accessor size: 'l' | 'm' | 's' = isLean() ? 's' : 'm';\n\n  /** Whether to display the form field without a border. */\n  @forceType()\n  @property({ reflect: true, type: Boolean })\n  public accessor borderless: boolean = false;\n\n  /** Defines the width of the component:\n   * - `default`: the component has defined width and min-width;\n   * - `collapse`: the component adapts itself to its inner input content. */\n  @property({ reflect: true }) public accessor width: 'default' | 'collapse' = 'default';\n\n  /** Whether to visually hide the label. If hidden, screen readers will still read it. */\n  @forceType()\n  @property({ attribute: 'hidden-label', reflect: true, type: Boolean })\n  public accessor hiddenLabel: boolean = false;\n\n  /** Whether the label should float. If activated, the placeholder of the input is hidden. */\n  @forceType()\n  @property({ attribute: 'floating-label', reflect: true, type: Boolean })\n  public accessor floatingLabel: boolean = false;\n\n  /** It is used internally to get the `error` slot. */\n  @state() private accessor _errorElements: Element[] = [];\n\n  /** Reference to the slotted input element. */\n  @state() private accessor _input: HTMLInputElement | HTMLSelectElement | HTMLElement | null =\n    null;\n\n  /** Reference to the slotted label elements. */\n  @state() private accessor _label!: HTMLLabelElement;\n\n  /** Returns the input element. */\n  public get inputElement(): HTMLInputElement | HTMLSelectElement | HTMLElement | null {\n    return this._input;\n  }\n\n  /** Reference to the slotted label. */\n  public get label(): HTMLLabelElement | null {\n    return this._label;\n  }\n\n  private _language = new SbbLanguageController(this);\n\n  /**\n   * Listens to the changes on `readonly` and `disabled` attributes of `<input>`.\n   */\n  private _formFieldAttributeObserver = !isServer\n    ? new MutationObserver((mutations: MutationRecord[]) => {\n        if (mutations.some((m) => m.type === 'attributes') && this._input) {\n          this._readInputState();\n          this._registerInputFormListener();\n          this._checkAndUpdateInputEmpty();\n        }\n      })\n    : null;\n\n  private _inputFormAbortController = new AbortController();\n  private _control: SbbFormFieldElementControl | null = null;\n\n  public constructor() {\n    super();\n    this.addEventListener?.(\n      'focusin',\n      (event) => {\n        if (\n          event.target === this.inputElement ||\n          event.target === (this.inputElement as SbbSelectElement | undefined)?.inputElement\n        ) {\n          this.internals.states.add('focus');\n          this.internals.states.add(`focus-origin-${sbbInputModalityDetector.mostRecentModality}`);\n        }\n      },\n      { passive: true },\n    );\n    this.addEventListener?.(\n      'focusout',\n      (event) => {\n        if (\n          event.target === this.inputElement ||\n          event.target === (this.inputElement as SbbSelectElement | undefined)?.inputElement\n        ) {\n          this._checkAndUpdateInputEmpty();\n          this.internals.states.delete('focus');\n          for (const state of this.internals.states) {\n            if (state.startsWith('focus-origin-')) {\n              this.internals.states.delete(state);\n            }\n          }\n        }\n      },\n      { passive: true },\n    );\n    this.addEventListener('input', () => this._checkAndUpdateInputEmpty());\n    this.addEventListener('displayvaluechange', () => this._checkAndUpdateInputEmpty());\n    // We want to prevent the native browser validation message popover\n    // to be shown. This also prevents a bug in WebKit, which would not\n    // allow host as the validity anchor: https://bugs.webkit.org/show_bug.cgi?id=269832\n    // This is duplicated from the form associated mixin for the native\n    // input controls (e.g. <input> or <select>).\n    this.addEventListener('invalid', (e) => e.preventDefault(), { capture: true });\n    this.addEventListener('formfieldcontrol', (e: SbbFormFieldControlEvent) => {\n      this._control = e.control;\n      if (this._connectInputElement() === 'unchanged') {\n        this._assignErrorMessageElements();\n        this._readInputState();\n        this._checkAndUpdateInputEmpty();\n      }\n    });\n  }\n\n  public override connectedCallback(): void {\n    super.connectedCallback();\n    this._assignSlots();\n    this._connectInputElement();\n    this._syncNegative();\n    this._syncSize();\n  }\n\n  protected override willUpdate(changedProperties: PropertyValues<this>): void {\n    super.willUpdate(changedProperties);\n\n    if (changedProperties.has('negative')) {\n      this._syncNegative();\n    }\n    if (changedProperties.has('size')) {\n      this._syncSize();\n    }\n  }\n\n  public override disconnectedCallback(): void {\n    super.disconnectedCallback();\n    this._formFieldAttributeObserver?.disconnect();\n    this._inputFormAbortController.abort();\n    if (this._input?.localName === 'input') {\n      this._unpatchInputValue();\n    }\n  }\n\n  private _handleWrapperClick(event: MouseEvent): void {\n    if (this._isElementFocusExcluded(event)) {\n      return;\n    }\n\n    if (this._control?.onContainerClick) {\n      this._control?.onContainerClick(event);\n    } else if ((event.target as Element).localName !== 'label') {\n      if (\n        this._input?.localName === 'sbb-select' &&\n        (event.target as HTMLElement).localName !== 'sbb-select'\n      ) {\n        this._input.click();\n      }\n      this._input?.focus();\n    }\n  }\n\n  private _isElementFocusExcluded(event: Event): boolean {\n    return event\n      .composedPath()\n      .some(\n        (el) =>\n          (el instanceof window.HTMLElement &&\n            (el.getAttribute('tabindex') === '0' || el.hasAttribute('contenteditable'))) ||\n          this._excludedFocusElements.includes((el as HTMLElement).localName),\n      );\n  }\n\n  private _onSlotLabelChange(): void {\n    const labels = Array.from(this.querySelectorAll('label'));\n    if (import.meta.env.DEV && labels.length > 1) {\n      console.warn(\n        `Detected more than one label in sbb-form-field#${this.id}. Only one label is supported.`,\n      );\n    }\n    this._label = labels[0];\n    this._syncLabelInputReferences();\n  }\n\n  /**\n   * It is used internally to assign the attributes of `<input>` to `_id` and `_input` and to observe the native readonly and disabled attributes.\n   */\n  private _onSlotInputChange(): void {\n    this._assignSlots();\n    this._connectInputElement();\n  }\n\n  private _assignSlots(): void {\n    this.querySelectorAll('label:not([slot])').forEach((e) => e.setAttribute('slot', 'label'));\n    this.querySelectorAll('sbb-error:not([slot])').forEach((e) => e.setAttribute('slot', 'error'));\n  }\n\n  private _connectInputElement(): 'changed' | 'no-input' | 'unchanged' {\n    let newInput: HTMLElement | null;\n    if (this._control?.id) {\n      newInput = (this.getRootNode() as Document | ShadowRoot).getElementById(this._control.id);\n    } else {\n      // Find the slotted input element, even if it's nested (e.g. chip group)\n      const inputCandidates = Array.from(\n        this.querySelectorAll<HTMLElement>('*:not([slot],sbb-chip-group)'),\n      );\n      newInput =\n        (inputCandidates.find((e) => this._isInputElement(e)) || inputCandidates[0]) ?? null;\n    }\n\n    if (newInput === this._input) {\n      return 'unchanged';\n    } else if (this._input) {\n      this.internals.states.delete(`input-type-${this._input.localName}`);\n      if (this._input.localName === 'input') {\n        this._unpatchInputValue();\n      }\n    }\n\n    if (!newInput) {\n      this._input = null;\n      return 'no-input';\n    }\n\n    this._input = newInput;\n    this._registerInputFormListener();\n    this._assignErrorMessageElements();\n    this._readInputState();\n    this._checkAndUpdateInputEmpty();\n\n    if (this._input.localName === 'textarea') {\n      this._input.setAttribute('rows', this._input.getAttribute('rows') || '3');\n    } else if (this._input.localName === 'input') {\n      this._patchInputValue();\n    } else if (\n      (this._input.localName === 'select' || this._input.localName === 'sbb-select') &&\n      !this.hasUpdated\n    ) {\n      // Due to SSR the dropdown icon for selects cannot be rendered in the first render pass.\n      // Due to this we schedule a re-render to render the icon.\n      this.hydrationComplete.then(() => this.requestUpdate());\n    }\n\n    this._formFieldAttributeObserver?.disconnect();\n    this._formFieldAttributeObserver?.observe(this._input, {\n      attributes: true,\n      attributeFilter: ['readonly', 'disabled', 'form', 'class', 'data-expanded'],\n    });\n    this.internals.states.add(`input-type-${this._input.localName}`);\n    this._syncLabelInputReferences();\n    return 'changed';\n  }\n\n  private _syncLabelInputReferences(): void {\n    if (!this._input || !this._label) {\n      return;\n    }\n\n    if (\n      nativeInputElements.includes(this._input.localName) ||\n      (customElements.get(this._input.localName) as { formAssociated?: boolean } | undefined)\n        ?.formAssociated\n    ) {\n      // For native input elements we use the `for` attribute on the label to reference the input\n      // via id reference.\n      this._input.id ||= `sbb-form-field-input-${nextId++}`;\n      this._label.htmlFor = this._input.id;\n    } else {\n      // For non-native input elements, that do not support references via the label for attribute,\n      // we use aria-labelledby on the input element to reference the label element via id.\n      // TODO: Migrate to ariaLabelledByElements when available:\n      // https://developer.mozilla.org/en-US/docs/Web/API/Element/ariaLabelledByElements\n      this._label.id ||= `sbb-form-field-label-${nextId++}`;\n      const labelledby =\n        this._input\n          .getAttribute('aria-labelledby')\n          ?.split(' ')\n          .filter((l) => !!l && l !== this._label!.id) ?? [];\n      this._input.setAttribute('aria-labelledby', [...labelledby, this._label.id].join(' '));\n    }\n  }\n\n  private _isInputElement(input: Element): boolean {\n    return (\n      nativeInputElements.includes(input.localName) ||\n      !!(customElements.get(input.localName) as { formAssociated?: boolean } | undefined)\n        ?.formAssociated\n    );\n  }\n\n  private _readInputState(): void {\n    this.toggleState('readonly', this._control?.readOnly ?? this._input!.hasAttribute('readonly'));\n    this.toggleState('disabled', this._control?.disabled ?? this._input!.hasAttribute('disabled'));\n    this.toggleState('has-popup-open', this._input!.hasAttribute('data-expanded'));\n  }\n\n  private _registerInputFormListener(): void {\n    this._inputFormAbortController.abort();\n    const { signal } = (this._inputFormAbortController = new AbortController());\n\n    const form =\n      (this._input as HTMLInputElement | HTMLSelectElement | undefined)?.form ??\n      this._input?.closest('form');\n    // Timeout needed to have value updated\n    const resetHandler = (): unknown => setTimeout(() => this.reset());\n    form?.addEventListener('reset', resetHandler, { signal });\n  }\n\n  // We need to patch the value property of the HTMLInputElement in order\n  // to be able to reset the floating label in the empty state.\n  private _patchInputValue(): void {\n    const inputElement = this._input as HTMLInputElement;\n    if (!inputElement || patchedInputs.has(inputElement)) {\n      return;\n    }\n\n    const originalDescriptor = Object.getOwnPropertyDescriptor(\n      Object.getPrototypeOf(inputElement),\n      'value',\n    );\n\n    if (!originalDescriptor || !originalDescriptor.set || !originalDescriptor.get) {\n      return;\n    }\n\n    patchedInputs.set(inputElement, originalDescriptor);\n\n    const { get: getter, set: setter } = originalDescriptor;\n    const checkAndUpdateInputEmpty = (): void => this._checkAndUpdateInputEmpty();\n\n    Object.defineProperty(inputElement, 'value', {\n      ...originalDescriptor,\n      get() {\n        return getter.call(this);\n      },\n      set(newValue) {\n        setter.call(this, newValue);\n        checkAndUpdateInputEmpty();\n      },\n    });\n  }\n\n  private _unpatchInputValue(): void {\n    const inputElement = this._input as HTMLInputElement;\n    const originalDescriptor = patchedInputs.get(inputElement);\n    if (originalDescriptor) {\n      Object.defineProperty(inputElement, 'value', originalDescriptor);\n      patchedInputs.delete(inputElement);\n    }\n  }\n\n  private _checkAndUpdateInputEmpty(): void {\n    this.toggleState(\n      'empty',\n      this._control?.empty ??\n        (((this._floatingLabelSupportedInputElements.includes(this._input?.localName as string) ||\n          (this._input?.constructor as undefined | typeof SbbFormAssociatedInputMixinType)\n            ?.formFieldAssociated) ??\n          false) &&\n          this._isInputEmpty()),\n    );\n  }\n\n  private _isInputEmpty(): boolean {\n    const chipGroupElem: SbbChipGroupElement | null = this.querySelector('sbb-chip-group');\n    if (chipGroupElem) {\n      return (\n        this._isInputValueEmpty() &&\n        (Array.isArray(chipGroupElem.value)\n          ? chipGroupElem.value.length === 0\n          : !chipGroupElem.querySelector('sbb-chip'))\n      );\n    } else if (this._input instanceof HTMLInputElement) {\n      return (\n        this._floatingLabelSupportedInputTypes.includes(this._input.type) &&\n        this._isInputValueEmpty()\n      );\n    } else if (this._input instanceof HTMLSelectElement) {\n      return this._input.selectedOptions?.item(0)?.label?.trim() === '';\n    } else if (this._input?.localName === 'sbb-select') {\n      return (this._input as SbbSelectElement).getDisplayValue?.()?.trim() === '';\n    } else {\n      return this._isInputValueEmpty();\n    }\n  }\n\n  private _isInputValueEmpty(): boolean {\n    const value = (this._input as { value: string }).value;\n    return ['', undefined, null].includes(value) || (Array.isArray(value) && value.length === 0);\n  }\n\n  /**\n   * It is used internally to set the aria-describedby attribute for the slotted input referencing available <sbb-error> instances.\n   */\n  private _onSlotErrorChange(event: Event): void {\n    const errorElements = (event.target as HTMLSlotElement).assignedElements();\n    if (this._input && this._input.ariaDescribedByElements?.length) {\n      this._input.ariaDescribedByElements = removeAriaElements(\n        this._input.ariaDescribedByElements,\n        ...(this._errorElements ?? []),\n      );\n    }\n\n    this._errorElements = errorElements;\n    for (const el of this._errorElements) {\n      // Instead of defining a container with an aria-live region as expected, we had to change\n      // setting it for every slotted element to properly work in all browsers and screen reader combinations.\n      el.role ||= 'status';\n    }\n\n    this._assignErrorMessageElements();\n    this.toggleState('has-error', !!this._errorElements.length);\n    this._syncNegative();\n  }\n\n  private _assignErrorMessageElements(): void {\n    if (this._input) {\n      this._input.ariaDescribedByElements = appendAriaElements(\n        this._input.ariaDescribedByElements,\n        ...this._errorElements,\n      );\n    }\n  }\n\n  /** Manually reset the form field. Currently, this only resets the floating label. */\n  public reset(): void {\n    this._checkAndUpdateInputEmpty();\n  }\n\n  /** Manually clears the input value. It only works for inputs, selects are not supported. */\n  public clear(): void {\n    if ((this._input?.localName as string) !== 'input') {\n      return;\n    }\n    (this._input as { value: string }).value = '';\n    this._checkAndUpdateInputEmpty();\n  }\n\n  private _syncNegative(): void {\n    this.querySelectorAll?.(\n      'sbb-error,sbb-mini-button,sbb-mini-button-link,sbb-form-field-clear,sbb-datepicker-next-day,sbb-datepicker-previous-day,sbb-datepicker-toggle,sbb-select,sbb-autocomplete,sbb-autocomplete-grid,sbb-chip-group',\n    ).forEach((element) => element.toggleAttribute('negative', this.negative));\n  }\n\n  private _syncSize(): void {\n    this.querySelectorAll?.<SbbAutocompleteBaseElement | SbbSelectElement>(\n      'sbb-autocomplete,sbb-autocomplete-grid,sbb-select',\n    ).forEach((element) => (element.size = this.size === 's' ? 's' : 'm'));\n  }\n\n  protected override render(): TemplateResult {\n    return html`\n      <div class=\"sbb-form-field__space-wrapper\">\n        ${/* Queried by id from the autocomplete/select to be used as the anchor element */ ''}\n        <div @click=${this._handleWrapperClick} class=\"sbb-form-field__wrapper\" id=\"overlay-anchor\">\n          <slot name=\"prefix\" @slotchange=${this._syncNegative}></slot>\n          <div class=\"sbb-form-field__input-container\">\n            <span class=\"sbb-form-field__label-spacer\" aria-hidden=\"true\"></span>\n            <span class=\"sbb-form-field__label\">\n              <span class=\"sbb-form-field__label-ellipsis\">\n                <slot name=\"label\" @slotchange=${this._onSlotLabelChange}></slot>\n                ${this.optional\n                  ? html` <span aria-hidden=\"true\"> ${i18nOptional[this._language.current]} </span>`\n                  : nothing}\n              </span>\n            </span>\n            <div class=\"sbb-form-field__input\">\n              <slot @slotchange=${this._onSlotInputChange}></slot>\n            </div>\n            ${this.hasUpdated && ['select', 'sbb-select'].includes(this._input?.localName as string)\n              ? html`<sbb-icon\n                  name=\"chevron-small-down-small\"\n                  class=\"sbb-form-field__select-input-icon\"\n                ></sbb-icon>`\n              : nothing}\n          </div>\n          <slot name=\"suffix\" @slotchange=${this._syncNegative}></slot>\n        </div>\n\n        <div class=\"sbb-form-field__error\">\n          <slot name=\"error\" @slotchange=${this._onSlotErrorChange}></slot>\n        </div>\n      </div>\n    `;\n  }\n}\n\ndeclare global {\n  interface HTMLElementTagNameMap {\n    // eslint-disable-next-line @typescript-eslint/naming-convention\n    'sbb-form-field': SbbFormFieldElement;\n  }\n\n  interface HTMLElementEventMap {\n    formfieldcontrol: SbbFormFieldControlEvent;\n  }\n}\n"],"mappings":"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"}