@sbb-esta/lyne-elements 3.12.0 → 3.12.1

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.
@@ -335,12 +335,13 @@ const style = css`*,
335
335
  const ariaRoleOnHost = isSafari;
336
336
  let nextId = 0;
337
337
  let SbbSelectElement = (() => {
338
- var _placeholder_accessor_storage, _multiple_accessor_storage, _disabled_accessor_storage, _value_accessor_storage, _size_accessor_storage, __displayValue_accessor_storage, _a;
338
+ var _placeholder_accessor_storage, _multiple_accessor_storage, _disabled_accessor_storage, _size_accessor_storage, __displayValue_accessor_storage, _a;
339
339
  let _classDecorators = [customElement("sbb-select")];
340
340
  let _classDescriptor;
341
341
  let _classExtraInitializers = [];
342
342
  let _classThis;
343
343
  let _classSuper = SbbUpdateSchedulerMixin(SbbDisabledMixin(SbbNegativeMixin(SbbHydrationMixin(SbbRequiredMixin(SbbReadonlyMixin(SbbFormAssociatedMixin(SbbOpenCloseBaseElement)))))));
344
+ let _instanceExtraInitializers = [];
344
345
  let _placeholder_decorators;
345
346
  let _placeholder_initializers = [];
346
347
  let _placeholder_extraInitializers = [];
@@ -350,9 +351,7 @@ let SbbSelectElement = (() => {
350
351
  let _disabled_decorators;
351
352
  let _disabled_initializers = [];
352
353
  let _disabled_extraInitializers = [];
353
- let _value_decorators;
354
- let _value_initializers = [];
355
- let _value_extraInitializers = [];
354
+ let _set_value_decorators;
356
355
  let _size_decorators;
357
356
  let _size_initializers = [];
358
357
  let _size_extraInitializers = [];
@@ -365,14 +364,13 @@ let SbbSelectElement = (() => {
365
364
  __privateAdd(this, _placeholder_accessor_storage);
366
365
  __privateAdd(this, _multiple_accessor_storage);
367
366
  __privateAdd(this, _disabled_accessor_storage);
368
- __privateAdd(this, _value_accessor_storage);
369
367
  __privateAdd(this, _size_accessor_storage);
370
368
  __privateAdd(this, __displayValue_accessor_storage);
371
- __privateSet(this, _placeholder_accessor_storage, __runInitializers(this, _placeholder_initializers, ""));
369
+ __privateSet(this, _placeholder_accessor_storage, (__runInitializers(this, _instanceExtraInitializers), __runInitializers(this, _placeholder_initializers, "")));
372
370
  __privateSet(this, _multiple_accessor_storage, (__runInitializers(this, _placeholder_extraInitializers), __runInitializers(this, _multiple_initializers, false)));
373
371
  __privateSet(this, _disabled_accessor_storage, (__runInitializers(this, _multiple_extraInitializers), __runInitializers(this, _disabled_initializers, false)));
374
- __privateSet(this, _value_accessor_storage, (__runInitializers(this, _disabled_extraInitializers), __runInitializers(this, _value_initializers, null)));
375
- __privateSet(this, _size_accessor_storage, (__runInitializers(this, _value_extraInitializers), __runInitializers(this, _size_initializers, isLean() ? "s" : "m")));
372
+ this._value = (__runInitializers(this, _disabled_extraInitializers), null);
373
+ __privateSet(this, _size_accessor_storage, __runInitializers(this, _size_initializers, isLean() ? "s" : "m"));
376
374
  __privateSet(this, __displayValue_accessor_storage, (__runInitializers(this, _size_extraInitializers), __runInitializers(this, __displayValue_initializers, null)));
377
375
  this._originResizeObserver = (__runInitializers(this, __displayValue_extraInitializers), new ResizeController(this, {
378
376
  target: null,
@@ -390,6 +388,7 @@ let SbbSelectElement = (() => {
390
388
  this._didLoad = false;
391
389
  this._isPointerDownEventOnMenu = false;
392
390
  this._languageController = new SbbLanguageController(this);
391
+ this._isValueManuallyAssigned = false;
393
392
  this._pointerDownListener = (event) => {
394
393
  this._isPointerDownEventOnMenu = isEventOnElement(this._overlay, event);
395
394
  };
@@ -400,6 +399,7 @@ let SbbSelectElement = (() => {
400
399
  };
401
400
  this.addEventListener?.("optionselectionchange", (e) => this._onOptionChanged(e));
402
401
  this.addEventListener?.("optionLabelChanged", (e) => this._onOptionLabelChanged(e));
402
+ this.addEventListener?.("ɵoptgroupslotchange", () => this._onSlotChange(), { capture: true });
403
403
  this.addEventListener?.("click", (e) => {
404
404
  const target = e.target;
405
405
  if (target.localName === "sbb-option") {
@@ -437,11 +437,13 @@ let SbbSelectElement = (() => {
437
437
  __privateSet(this, _disabled_accessor_storage, value);
438
438
  }
439
439
  /** Value of the form element. */
440
- get value() {
441
- return __privateGet(this, _value_accessor_storage);
442
- }
443
440
  set value(value) {
444
- __privateSet(this, _value_accessor_storage, value);
441
+ this._value = value;
442
+ this._updateOptionsFromValue();
443
+ this._isValueManuallyAssigned = true;
444
+ }
445
+ get value() {
446
+ return this._value;
445
447
  }
446
448
  /**
447
449
  * Size variant, either m or s.
@@ -474,12 +476,14 @@ let SbbSelectElement = (() => {
474
476
  get inputElement() {
475
477
  return this._triggerElement;
476
478
  }
477
- /** Gets all the SbbOptionElement projected in the select. */
478
- get _options() {
479
- return Array.from(this.querySelectorAll?.("sbb-option") ?? []);
480
- }
481
- get _filteredOptions() {
482
- return this._options.filter((opt) => !opt.disabled && !opt.hasAttribute("data-group-disabled"));
479
+ /** Returns all SbbOptionElements from this sbb-select instance. */
480
+ get options() {
481
+ const options = [];
482
+ this.querySelectorAll?.("sbb-option").forEach((option) => {
483
+ customElements.upgrade(option);
484
+ options.push(option);
485
+ });
486
+ return options;
483
487
  }
484
488
  _syncAriaLabels() {
485
489
  if (!this._triggerElement || isServer) {
@@ -494,7 +498,7 @@ let SbbSelectElement = (() => {
494
498
  }
495
499
  /** Opens the selection panel. */
496
500
  open() {
497
- if (this.state !== "closed" || !this._overlay || this._options.length === 0 || this.disabled || this.formDisabled || !this.dispatchBeforeOpenEvent()) {
501
+ if (this.state !== "closed" || !this._overlay || this.options.length === 0 || this.disabled || this.formDisabled || !this.dispatchBeforeOpenEvent()) {
498
502
  return;
499
503
  }
500
504
  this.shadowRoot?.querySelector(".sbb-select__container")?.showPopover?.();
@@ -527,6 +531,9 @@ let SbbSelectElement = (() => {
527
531
  getDisplayValue() {
528
532
  return this._displayValue ?? "";
529
533
  }
534
+ _selectableOptions() {
535
+ return this.options.filter((opt) => !opt.disabled && !opt.hasAttribute("data-group-disabled"));
536
+ }
530
537
  /** Listens to option changes. */
531
538
  _onOptionChanged(event) {
532
539
  const target = event.target;
@@ -543,11 +550,12 @@ let SbbSelectElement = (() => {
543
550
  if (!Array.isArray(selected) && target !== selected || Array.isArray(selected) && !selected.includes(target)) {
544
551
  return;
545
552
  }
546
- this._updateDisplayValue(selected);
553
+ this._updateDisplayValue();
547
554
  }
548
- _updateDisplayValue(selected) {
555
+ _updateDisplayValue() {
556
+ const selected = this._getSelected();
549
557
  if (Array.isArray(selected)) {
550
- this._displayValue = selected.map((o) => o.textContent).join(", ") || null;
558
+ this._displayValue = selected.map((o) => o.textContent?.trim()).join(", ") || null;
551
559
  } else if (selected) {
552
560
  this._displayValue = selected?.textContent?.trim() || null;
553
561
  } else {
@@ -562,10 +570,11 @@ let SbbSelectElement = (() => {
562
570
  */
563
571
  _onMultipleChanged(isChangingToMultiple) {
564
572
  if (isChangingToMultiple) {
565
- this.value = this.value !== null && this.value !== void 0 ? [this.value] : [];
573
+ this._value = this._isValueManuallyAssigned || this.value != null ? [this.value] : [];
566
574
  } else if (Array.isArray(this.value)) {
567
- this.value = this.value.length ? this.value[0] : null;
575
+ this._value = this.value.length ? this.value[0] : null;
568
576
  }
577
+ this._updateDisplayValue();
569
578
  }
570
579
  /**
571
580
  * If the `disabled` or the `readonly` properties are set, and the panel is open, close it.
@@ -575,23 +584,6 @@ let SbbSelectElement = (() => {
575
584
  this.close();
576
585
  }
577
586
  }
578
- /** Sets the _displayValue by checking the internal sbb-options and setting the correct `selected` value on them. */
579
- _onValueChanged(newValue) {
580
- const options = this._filteredOptions;
581
- if (!Array.isArray(newValue)) {
582
- const optionElement = options.find((o) => o.value === newValue) ?? null;
583
- if (optionElement) {
584
- optionElement.selected = true;
585
- }
586
- options.filter((o) => (o.value ?? o.getAttribute("value")) !== newValue).forEach((o) => o.selected = false);
587
- this._updateDisplayValue(optionElement);
588
- } else {
589
- options.filter((o) => !newValue.includes(o.value)).forEach((e) => e.selected = false);
590
- const selectedElements = options.filter((o) => newValue.includes(o.value));
591
- selectedElements.forEach((o) => o.selected = true);
592
- this._updateDisplayValue(selectedElements);
593
- }
594
- }
595
587
  firstUpdated(changedProperties) {
596
588
  super.firstUpdated(changedProperties);
597
589
  if (!isNextjs()) {
@@ -643,9 +635,6 @@ let SbbSelectElement = (() => {
643
635
  }
644
636
  willUpdate(changedProperties) {
645
637
  super.willUpdate(changedProperties);
646
- if (changedProperties.has("value") && (this._didLoad || this.value)) {
647
- this._onValueChanged(this.value);
648
- }
649
638
  if (changedProperties.has("negative") || changedProperties.has("multiple")) {
650
639
  this._syncProperties();
651
640
  }
@@ -669,16 +658,22 @@ let SbbSelectElement = (() => {
669
658
  * @internal
670
659
  */
671
660
  formStateRestoreCallback(state2, _reason) {
672
- if (typeof state2 === "string" || state2 == null) {
673
- this.value = state2 ?? null;
674
- } else if (state2 instanceof FormData) {
675
- this._readFormData(state2).then((array) => {
676
- this.value = this.multiple ? array : array[0];
677
- });
661
+ try {
662
+ const { value, manuallyAssigned } = JSON.parse(state2);
663
+ const values = Array.isArray(value) ? value : [value];
664
+ if (values.some((v) => v !== null && typeof v === "object")) {
665
+ console.warn(`Restoring complex objects is not supported for sbb-select with state ${state2}`);
666
+ return;
667
+ }
668
+ this._isValueManuallyAssigned = manuallyAssigned;
669
+ this._value = value;
670
+ this._updateOptionsFromValue();
671
+ } catch {
672
+ console.warn(`Failed to restore sbb-select with state ${state2}`);
678
673
  }
679
674
  }
680
- async _readFormData(formData) {
681
- return Promise.all(formData.getAll(this.name).map(async (entry) => entry instanceof Blob ? JSON.parse(await entry.text()) : entry));
675
+ formState() {
676
+ return JSON.stringify({ value: this.value, manuallyAssigned: this._isValueManuallyAssigned });
682
677
  }
683
678
  _syncProperties() {
684
679
  this.querySelectorAll?.("sbb-divider, sbb-option-hint").forEach((el) => el.negative = this.negative);
@@ -693,7 +688,7 @@ let SbbSelectElement = (() => {
693
688
  }
694
689
  validate() {
695
690
  super.validate();
696
- if (this.required && this._options.every((o) => o.value !== this.value)) {
691
+ if (this.required && (this.options.every((o) => o.value !== this.value) || !this._isValueManuallyAssigned && this.value == null)) {
697
692
  this.setValidityFlag("valueMissing", i18nSelectionRequired[this._languageController.current]);
698
693
  } else {
699
694
  this.removeValidityFlag("valueMissing");
@@ -758,23 +753,22 @@ let SbbSelectElement = (() => {
758
753
  this.dispatchCloseEvent();
759
754
  }
760
755
  /** When an option is selected, updates the displayValue; it also closes the select if not `multiple`. */
761
- _onOptionSelected(optionSelectionChange) {
756
+ _onOptionSelected(option) {
762
757
  if (!this.multiple) {
763
- this._filteredOptions.filter((option) => option.id !== optionSelectionChange.id).forEach((option) => option.selected = false);
764
- this.value = optionSelectionChange.value;
765
- } else {
766
- if (!this.value) {
767
- this.value = [optionSelectionChange.value];
768
- } else if (Array.isArray(this.value) && !this.value.includes(optionSelectionChange.value)) {
769
- this.value = [...this.value, optionSelectionChange.value];
770
- }
758
+ this._value = option.value;
759
+ } else if (!this.value) {
760
+ this._value = [option.value];
761
+ } else if (Array.isArray(this.value) && !this.value.includes(option.value)) {
762
+ this._value = [...this.value, option.value];
771
763
  }
764
+ this._updateOptionsFromValue();
772
765
  this._dispatchInputEvents();
773
766
  }
774
767
  /** When an option is unselected in `multiple`, removes it from value and updates displayValue. */
775
768
  _onOptionDeselected(optionSelectionChange) {
776
769
  if (this.multiple && Array.isArray(this.value)) {
777
- this.value = this.value.filter((el) => el !== optionSelectionChange.value);
770
+ this._value = this.value.filter((el) => el !== optionSelectionChange.value);
771
+ this._updateOptionsFromValue();
778
772
  this._dispatchInputEvents();
779
773
  }
780
774
  }
@@ -858,7 +852,7 @@ let SbbSelectElement = (() => {
858
852
  case "End":
859
853
  case "PageDown":
860
854
  event.preventDefault();
861
- this._setNextActiveOption(event, this._filteredOptions.length - 1);
855
+ this._setNextActiveOption(event, this._selectableOptions().length - 1);
862
856
  break;
863
857
  }
864
858
  }
@@ -872,17 +866,18 @@ let SbbSelectElement = (() => {
872
866
  this._searchTimeout = setTimeout(() => this._searchString = "", 1e3);
873
867
  this._searchString += event.key;
874
868
  const indexForSlice = this._activeItemIndex + 1;
869
+ const selectableOptions = this._selectableOptions();
875
870
  const filteredOptionsSorted = [
876
- ...this._filteredOptions.slice(indexForSlice),
877
- ...this._filteredOptions.slice(0, indexForSlice)
871
+ ...selectableOptions.slice(indexForSlice),
872
+ ...selectableOptions.slice(0, indexForSlice)
878
873
  ];
879
874
  const match = filteredOptionsSorted.find((option) => option.textContent?.toLowerCase().indexOf(this._searchString.toLowerCase()) === 0);
880
875
  if (match) {
881
- this._setNextActiveOption(event, this._filteredOptions.indexOf(match));
876
+ this._setNextActiveOption(event, selectableOptions.indexOf(match));
882
877
  } else if (this._searchString.length > 1 && new RegExp(`^${this._searchString.charAt(0)}*$`).test(this._searchString)) {
883
878
  const firstMatch = filteredOptionsSorted.find((option) => option.textContent?.toLowerCase().indexOf(this._searchString[0].toLowerCase()) === 0);
884
879
  if (firstMatch) {
885
- this._setNextActiveOption(event, this._filteredOptions.indexOf(firstMatch));
880
+ this._setNextActiveOption(event, selectableOptions.indexOf(firstMatch));
886
881
  }
887
882
  } else {
888
883
  clearTimeout(this._searchTimeout);
@@ -890,7 +885,7 @@ let SbbSelectElement = (() => {
890
885
  }
891
886
  }
892
887
  _selectByKeyboard() {
893
- const activeOption = this._filteredOptions[this._activeItemIndex];
888
+ const activeOption = this._selectableOptions()[this._activeItemIndex];
894
889
  if (this.multiple) {
895
890
  activeOption?.["selectViaUserInteraction"](!activeOption.selected);
896
891
  } else {
@@ -898,7 +893,7 @@ let SbbSelectElement = (() => {
898
893
  }
899
894
  }
900
895
  _setNextActiveOption(event, index) {
901
- const filteredOptions = this._filteredOptions;
896
+ const filteredOptions = this._selectableOptions();
902
897
  if (filteredOptions.length === 0) {
903
898
  return;
904
899
  }
@@ -930,14 +925,25 @@ let SbbSelectElement = (() => {
930
925
  }
931
926
  }
932
927
  _resetActiveElement() {
933
- const activeElement = this._filteredOptions[this._activeItemIndex];
928
+ const activeElement = this._selectableOptions()[this._activeItemIndex];
934
929
  if (activeElement) {
935
930
  activeElement.setActive(false);
936
931
  }
937
932
  this._activeItemIndex = -1;
938
933
  this._triggerElement.removeAttribute("aria-activedescendant");
939
934
  }
940
- _setValueFromSelected() {
935
+ _updateOptionsFromValue() {
936
+ const value = Array.isArray(this.value) ? this.value : [this.value];
937
+ for (const option of this.options) {
938
+ option.selected = value.includes(option.value);
939
+ if (option.selected) ;
940
+ }
941
+ this._updateDisplayValue();
942
+ if (!Array.isArray(this.value)) {
943
+ this._activeItemIndex = this._selectableOptions().findIndex((option) => option.value === this.value);
944
+ }
945
+ }
946
+ _updateValueFromOptions() {
941
947
  const selected = this._getSelected();
942
948
  if (Array.isArray(selected)) {
943
949
  if (selected && selected.length > 0) {
@@ -945,20 +951,26 @@ let SbbSelectElement = (() => {
945
951
  for (const option of selected) {
946
952
  value.push(option.value);
947
953
  }
948
- this.value = value;
954
+ this._value = value;
949
955
  }
950
956
  } else if (selected) {
951
- this._activeItemIndex = this._filteredOptions.findIndex((option) => option === selected);
952
- this.value = selected.value;
953
- } else if (this.value) {
954
- this._onValueChanged(this.value);
957
+ this._activeItemIndex = this._selectableOptions().findIndex((option) => option === selected);
958
+ this._value = selected.value;
959
+ }
960
+ this._updateDisplayValue();
961
+ }
962
+ _onSlotChange() {
963
+ if (this._isValueManuallyAssigned) {
964
+ this._updateOptionsFromValue();
965
+ } else {
966
+ this._updateValueFromOptions();
955
967
  }
956
968
  }
957
969
  _getSelected() {
958
970
  if (this.multiple) {
959
- return this._filteredOptions.filter((option) => option.selected);
971
+ return this.options.filter((option) => option.selected);
960
972
  } else {
961
- return this._filteredOptions.find((option) => option.selected) ?? null;
973
+ return this.options.find((option) => option.selected) ?? null;
962
974
  }
963
975
  }
964
976
  _toggleOpening() {
@@ -1027,19 +1039,19 @@ let SbbSelectElement = (() => {
1027
1039
  ?aria-multiselectable=${this.multiple}
1028
1040
  ${ref((containerRef) => this._optionContainer = containerRef)}
1029
1041
  >
1030
- <slot @slotchange=${this._setValueFromSelected}></slot>
1042
+ <slot @slotchange=${this._onSlotChange}></slot>
1031
1043
  </div>
1032
1044
  </div>
1033
1045
  </div>
1034
1046
  </div>
1035
1047
  `;
1036
1048
  }
1037
- }, _placeholder_accessor_storage = new WeakMap(), _multiple_accessor_storage = new WeakMap(), _disabled_accessor_storage = new WeakMap(), _value_accessor_storage = new WeakMap(), _size_accessor_storage = new WeakMap(), __displayValue_accessor_storage = new WeakMap(), _classThis = _a, (() => {
1049
+ }, _placeholder_accessor_storage = new WeakMap(), _multiple_accessor_storage = new WeakMap(), _disabled_accessor_storage = new WeakMap(), _size_accessor_storage = new WeakMap(), __displayValue_accessor_storage = new WeakMap(), _classThis = _a, (() => {
1038
1050
  const _metadata = typeof Symbol === "function" && Symbol.metadata ? Object.create(_classSuper[Symbol.metadata] ?? null) : void 0;
1039
1051
  _placeholder_decorators = [forceType(), property()];
1040
1052
  _multiple_decorators = [forceType(), handleDistinctChange((e, newValue) => e._onMultipleChanged(newValue)), property({ reflect: true, type: Boolean })];
1041
1053
  _disabled_decorators = [forceType(), handleDistinctChange((e, newValue) => e._closeOnDisabledReadonlyChanged(newValue)), property({ reflect: true, type: Boolean }), getOverride((e, v) => v || e.isDisabledExternally())];
1042
- _value_decorators = [property()];
1054
+ _set_value_decorators = [property()];
1043
1055
  _size_decorators = [property({ reflect: true })];
1044
1056
  __displayValue_decorators = [state()];
1045
1057
  __esDecorate(_a, null, _placeholder_decorators, { kind: "accessor", name: "placeholder", static: false, private: false, access: { has: (obj) => "placeholder" in obj, get: (obj) => obj.placeholder, set: (obj, value) => {
@@ -1051,9 +1063,9 @@ let SbbSelectElement = (() => {
1051
1063
  __esDecorate(_a, null, _disabled_decorators, { kind: "accessor", name: "disabled", static: false, private: false, access: { has: (obj) => "disabled" in obj, get: (obj) => obj.disabled, set: (obj, value) => {
1052
1064
  obj.disabled = value;
1053
1065
  } }, metadata: _metadata }, _disabled_initializers, _disabled_extraInitializers);
1054
- __esDecorate(_a, null, _value_decorators, { kind: "accessor", name: "value", static: false, private: false, access: { has: (obj) => "value" in obj, get: (obj) => obj.value, set: (obj, value) => {
1066
+ __esDecorate(_a, null, _set_value_decorators, { kind: "setter", name: "value", static: false, private: false, access: { has: (obj) => "value" in obj, set: (obj, value) => {
1055
1067
  obj.value = value;
1056
- } }, metadata: _metadata }, _value_initializers, _value_extraInitializers);
1068
+ } }, metadata: _metadata }, null, _instanceExtraInitializers);
1057
1069
  __esDecorate(_a, null, _size_decorators, { kind: "accessor", name: "size", static: false, private: false, access: { has: (obj) => "size" in obj, get: (obj) => obj.size, set: (obj, value) => {
1058
1070
  obj.size = value;
1059
1071
  } }, metadata: _metadata }, _size_initializers, _size_extraInitializers);
@@ -1077,4 +1089,4 @@ let SbbSelectElement = (() => {
1077
1089
  export {
1078
1090
  SbbSelectElement
1079
1091
  };
1080
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"select.component.js","sources":["../../../../src/elements/select/select.component.ts"],"sourcesContent":["import { MutationController } from '@lit-labs/observers/mutation-controller.js';\nimport { ResizeController } from '@lit-labs/observers/resize-controller.js';\nimport type { CSSResultGroup, PropertyDeclaration, PropertyValues, TemplateResult } from 'lit';\nimport { html, isServer, nothing } from 'lit';\nimport { customElement, property, state } from 'lit/decorators.js';\nimport { ref } from 'lit/directives/ref.js';\nimport { until } from 'lit/directives/until.js';\n\nimport { getNextElementIndex } from '../core/a11y.js';\nimport { SbbOpenCloseBaseElement } from '../core/base-elements.js';\nimport { SbbEscapableOverlayController, SbbLanguageController } from '../core/controllers.js';\nimport { forceType, getOverride, handleDistinctChange } from '../core/decorators.js';\nimport {\n  isLean,\n  isNextjs,\n  isSafari,\n  isZeroAnimationDuration,\n  setOrRemoveAttribute,\n} from '../core/dom.js';\nimport { i18nSelectionRequired } from '../core/i18n.js';\nimport {\n  type FormRestoreReason,\n  type FormRestoreState,\n  SbbDisabledMixin,\n  SbbFormAssociatedMixin,\n  SbbHydrationMixin,\n  SbbNegativeMixin,\n  SbbReadonlyMixin,\n  SbbRequiredMixin,\n  SbbUpdateSchedulerMixin,\n} from '../core/mixins.js';\nimport { isEventOnElement, overlayGapFixCorners, setOverlayPosition } from '../core/overlay.js';\nimport type { SbbDividerElement } from '../divider.js';\nimport type { SbbOptGroupElement, SbbOptionElement, SbbOptionHintElement } from '../option.js';\n\nimport style from './select.scss?lit&inline';\n\n/**\n * On Safari, the aria role 'listbox' must be on the host element, or else VoiceOver won't work at all.\n * On the other hand, JAWS and NVDA need the role to be an \"immediate parent\" to the options, or else optgroups won't work.\n */\nconst ariaRoleOnHost = isSafari;\n\nlet nextId = 0;\n\n/**\n * It displays a panel with selectable options.\n *\n * @slot - Use the unnamed slot to add options.\n * @cssprop [--sbb-select-z-index=var(--sbb-overlay-default-z-index)] - To specify a custom stack order,\n * the `z-index` can be overridden by defining this CSS variable. The default `z-index` of the\n * component is set to `var(--sbb-overlay-default-z-index)` with a value of `1000`.\n * @overrideType value - (T = string | string[]) | null\n */\nexport\n@customElement('sbb-select')\nclass SbbSelectElement<T = string> extends SbbUpdateSchedulerMixin(\n  SbbDisabledMixin(\n    SbbNegativeMixin(\n      SbbHydrationMixin(\n        SbbRequiredMixin(\n          SbbReadonlyMixin(\n            SbbFormAssociatedMixin<typeof SbbOpenCloseBaseElement>(SbbOpenCloseBaseElement),\n          ),\n        ),\n      ),\n    ),\n  ),\n) {\n  public static override readonly role = ariaRoleOnHost ? 'listbox' : null;\n  public static override styles: CSSResultGroup = style;\n\n  // TODO: fix using ...super.events requires: https://github.com/sbb-design-systems/lyne-components/issues/2600\n  public static override readonly events = {\n    change: 'change',\n    input: 'input',\n    displayvaluechange: 'displayvaluechange',\n    beforeopen: 'beforeopen',\n    open: 'open',\n    beforeclose: 'beforeclose',\n    close: 'close',\n  } as const;\n\n  /** The placeholder used if no value has been selected. */\n  @forceType()\n  @property()\n  public accessor placeholder: string = '';\n\n  /** Whether the select allows for multiple selection. */\n  @forceType()\n  @handleDistinctChange((e: SbbSelectElement<T>, newValue: boolean) =>\n    e._onMultipleChanged(newValue),\n  )\n  @property({ reflect: true, type: Boolean })\n  public accessor multiple: boolean = false;\n\n  @forceType()\n  @handleDistinctChange((e: SbbSelectElement<T>, newValue: boolean) =>\n    e._closeOnDisabledReadonlyChanged(newValue),\n  )\n  @property({ reflect: true, type: Boolean })\n  @getOverride((e: SbbSelectElement<T>, v: boolean): boolean => v || e.isDisabledExternally())\n  public override accessor disabled: boolean = false;\n\n  /** Value of the form element. */\n  @property()\n  public accessor value: T | T[] | null = null;\n\n  /**\n   * Size variant, either m or s.\n   * @default 'm' / 's' (lean)\n   */\n  @property({ reflect: true }) public accessor size: 'm' | 's' = isLean() ? 's' : 'm';\n\n  /**\n   * Form type of element.\n   * @default 'select-one / select-multiple'\n   */\n  public override get type(): string {\n    return this.multiple ? 'select-multiple' : 'select-one';\n  }\n\n  /** The value displayed by the component. */\n  @state() private accessor _displayValue: string | null = null;\n\n  private _originResizeObserver = new ResizeController(this, {\n    target: null,\n    skipInitial: true,\n    callback: () => {\n      if (this.isOpen) {\n        this._setOverlayPosition();\n      }\n    },\n  });\n\n  private _overlay!: HTMLElement;\n  private _optionContainer!: HTMLElement;\n  private _originElement!: HTMLElement;\n  private _triggerElement!: HTMLElement;\n  private _openPanelEventsController!: AbortController;\n  private _escapableOverlayController = new SbbEscapableOverlayController(this);\n  private _overlayId = `sbb-select-${++nextId}`;\n  private _activeItemIndex = -1;\n  private _searchTimeout?: ReturnType<typeof setTimeout>;\n  private _searchString = '';\n  private _didLoad = false;\n  private _isPointerDownEventOnMenu: boolean = false;\n  private _languageController = new SbbLanguageController(this);\n\n  /**\n   * The 'combobox' input element\n   * @internal\n   */\n  public get inputElement(): HTMLElement {\n    return this._triggerElement;\n  }\n\n  /** Gets all the SbbOptionElement projected in the select. */\n  private get _options(): SbbOptionElement<T>[] {\n    return Array.from(this.querySelectorAll?.<SbbOptionElement<T>>('sbb-option') ?? []);\n  }\n\n  private get _filteredOptions(): SbbOptionElement<T>[] {\n    return this._options.filter((opt) => !opt.disabled && !opt.hasAttribute('data-group-disabled'));\n  }\n\n  public constructor() {\n    super();\n    this.addEventListener?.('optionselectionchange', (e: Event) => this._onOptionChanged(e));\n    this.addEventListener?.('optionLabelChanged', (e: Event) => this._onOptionLabelChanged(e));\n    this.addEventListener?.('click', (e: MouseEvent) => {\n      const target = e.target as SbbSelectElement<T> | SbbOptionElement<T>;\n      if (target.localName === 'sbb-option') {\n        // Option click\n        if (!this.multiple && !target.disabled) {\n          this.close();\n          this.focus();\n        }\n      } else {\n        this._toggleOpening();\n      }\n    });\n\n    this.addController(\n      new MutationController(this, {\n        config: { attributeFilter: ['aria-labelledby', 'aria-label', 'aria-describedby'] },\n        callback: () => this._syncAriaLabels(),\n      }),\n    );\n  }\n\n  private _syncAriaLabels(): void {\n    if (!this._triggerElement || isServer) {\n      return;\n    }\n\n    setOrRemoveAttribute(\n      this._triggerElement,\n      'aria-labelledby',\n      this.getAttribute('aria-labelledby'),\n    );\n    setOrRemoveAttribute(this._triggerElement, 'aria-label', this.getAttribute('aria-label'));\n    setOrRemoveAttribute(\n      this._triggerElement,\n      'aria-describedby',\n      this.getAttribute('aria-describedby'),\n    );\n\n    // Using the associated labels is only a fallback.\n    // The drawback is, that it doesn't get updated automatically when the list of label changes.\n    if (\n      !this.getAttribute('aria-label') &&\n      !this.getAttribute('aria-labelledby') &&\n      this.internals.labels.length\n    ) {\n      this._triggerElement?.setAttribute(\n        'aria-label',\n        Array.from(this.internals.labels)\n          .map((label) => label.textContent)\n          .join(', '),\n      );\n    }\n  }\n\n  /** Opens the selection panel. */\n  public open(): void {\n    if (\n      this.state !== 'closed' ||\n      !this._overlay ||\n      this._options.length === 0 ||\n      this.disabled ||\n      this.formDisabled ||\n      !this.dispatchBeforeOpenEvent()\n    ) {\n      return;\n    }\n\n    this.shadowRoot?.querySelector<HTMLDivElement>('.sbb-select__container')?.showPopover?.();\n    this.state = 'opening';\n    this.toggleAttribute('data-expanded', true);\n    this._setOverlayPosition();\n\n    // If the animation duration is zero, the animationend event is not always fired reliably.\n    // In this case we directly set the `opened` state.\n    if (this._isZeroAnimationDuration()) {\n      this._handleOpening();\n    }\n  }\n\n  /** Closes the selection panel. */\n  public close(): void {\n    if (this.state !== 'opened' || !this.dispatchBeforeCloseEvent()) {\n      return;\n    }\n\n    this.state = 'closing';\n    this.toggleAttribute('data-expanded', false);\n    this._openPanelEventsController.abort();\n    if (this._originElement) {\n      this._originResizeObserver.unobserve(this._originElement);\n    }\n\n    // If the animation duration is zero, the animationend event is not always fired reliably.\n    // In this case we directly set the `closed` state.\n    if (this._isZeroAnimationDuration()) {\n      this._handleClosing();\n    }\n  }\n\n  private _isZeroAnimationDuration(): boolean {\n    return isZeroAnimationDuration(this, '--sbb-options-panel-animation-duration');\n  }\n\n  /** Gets the current displayed value. */\n  public getDisplayValue(): string {\n    return this._displayValue ?? '';\n  }\n\n  /** Listens to option changes. */\n  private _onOptionChanged(event: Event): void {\n    const target = event.target as SbbOptionElement<T>;\n    if (target.selected) {\n      this._onOptionSelected(target);\n    } else {\n      this._onOptionDeselected(target);\n    }\n  }\n\n  /** Listens to option changes. */\n  private _onOptionLabelChanged(event: Event): void {\n    const target = event.target as SbbOptionElement<T>;\n    const selected = this._getSelected();\n\n    if (\n      (!Array.isArray(selected) && target !== selected) ||\n      (Array.isArray(selected) && !selected.includes(target))\n    ) {\n      return;\n    }\n\n    this._updateDisplayValue(selected);\n  }\n\n  private _updateDisplayValue(selected: SbbOptionElement<T> | SbbOptionElement<T>[] | null): void {\n    if (Array.isArray(selected)) {\n      this._displayValue = selected.map((o) => o.textContent).join(', ') || null;\n    } else if (selected) {\n      this._displayValue = selected?.textContent?.trim() || null;\n    } else {\n      this._displayValue = null;\n    }\n\n    /** @internal */\n    this.dispatchEvent(new Event('displayvaluechange', { bubbles: true, composed: true }));\n  }\n\n  /**\n   * The `value` property should be adapted when the `multiple` property changes:\n   *   - if it changes to true, the 'value' is set to an array;\n   *   - if it changes to false, the first available option is set as 'value' otherwise it's set to null.\n   */\n  private _onMultipleChanged(isChangingToMultiple: boolean): void {\n    if (isChangingToMultiple) {\n      this.value = this.value !== null && this.value !== undefined ? [this.value as T] : [];\n    } else if (Array.isArray(this.value)) {\n      this.value = this.value.length ? this.value[0] : null;\n    }\n  }\n\n  /**\n   * If the `disabled` or the `readonly` properties are set, and the panel is open, close it.\n   */\n  private _closeOnDisabledReadonlyChanged(newValue: boolean): void {\n    if (this.isOpen && newValue) {\n      this.close();\n    }\n  }\n\n  /** Sets the _displayValue by checking the internal sbb-options and setting the correct `selected` value on them. */\n  private _onValueChanged(newValue: T | T[]): void {\n    const options = this._filteredOptions;\n    if (!Array.isArray(newValue)) {\n      const optionElement = options.find((o) => o.value === newValue) ?? null;\n      if (optionElement) {\n        optionElement.selected = true;\n      }\n      options\n        .filter((o) => (o.value ?? o.getAttribute('value')) !== newValue)\n        .forEach((o) => (o.selected = false));\n      this._updateDisplayValue(optionElement);\n    } else {\n      options.filter((o) => !newValue.includes(o.value)).forEach((e) => (e.selected = false));\n      const selectedElements = options.filter((o) => newValue.includes(o.value));\n      selectedElements.forEach((o) => (o.selected = true));\n      this._updateDisplayValue(selectedElements);\n    }\n  }\n\n  protected override firstUpdated(changedProperties: PropertyValues<this>): void {\n    super.firstUpdated(changedProperties);\n\n    // Wait for ssr hydration\n    if (!isNextjs()) {\n      this.startUpdate();\n      this._setupSelect();\n    }\n  }\n\n  /** @internal */\n  public override focus(): void {\n    // Forward focus to the trigger element.\n    this._triggerElement?.focus();\n  }\n\n  /** @internal */\n  public override blur(): void {\n    // Forward blur to the trigger element.\n    this._triggerElement?.blur();\n  }\n\n  /**\n   * Removes element's first attribute whose qualified name is qualifiedName.\n   *\n   * [MDN Reference](https://developer.mozilla.org/docs/Web/API/Element/removeAttribute)\n   * @internal We need to override this due to a hydration issue with Next.js.\n   */\n  public override removeAttribute(qualifiedName: string): void {\n    // In Next.js the hydration needs to finish before we can manipulate the light DOM.\n    // @lit/react calls removeAttribute('defer-hydration') in a useLayoutEffect,\n    // which is done after hydration is finished. Due to this we intercept this call\n    // in overriding removeAttribute to finish initialization of the sbb-select.\n    // https://github.com/lit/lit/blob/main/packages/react/src/create-component.ts#L293-L296\n    // We also need to wait for update complete in order to be sure that the shadow DOM has been rendered.\n    if (isNextjs() && qualifiedName === 'defer-hydration' && !this._didLoad) {\n      this.updateComplete.then(() => this._setupSelect());\n    }\n    super.removeAttribute(qualifiedName);\n  }\n\n  public override connectedCallback(): void {\n    super.connectedCallback();\n\n    if (ariaRoleOnHost) {\n      this.id ||= this._overlayId;\n    }\n\n    const formField = this.closest?.('sbb-form-field') ?? this.closest?.('[data-form-field]');\n\n    if (formField) {\n      this.negative = formField.hasAttribute('negative');\n    }\n    this._syncProperties();\n    this._syncAriaLabels();\n\n    if (this._didLoad) {\n      this._setupOrigin();\n      this._setupTrigger();\n    }\n  }\n\n  public override requestUpdate(\n    name?: PropertyKey,\n    oldValue?: unknown,\n    options?: PropertyDeclaration,\n  ): void {\n    super.requestUpdate(name, oldValue, options);\n    if (!name && this.hasUpdated) {\n      setTimeout(() => this._syncAriaLabels());\n    }\n  }\n\n  protected override willUpdate(changedProperties: PropertyValues<this>): void {\n    super.willUpdate(changedProperties);\n\n    if (changedProperties.has('value') && (this._didLoad || this.value)) {\n      this._onValueChanged(this.value!);\n    }\n    if (changedProperties.has('negative') || changedProperties.has('multiple')) {\n      this._syncProperties();\n    }\n    if (changedProperties.has('readOnly')) {\n      this._closeOnDisabledReadonlyChanged(this.readOnly);\n    }\n  }\n\n  public override disconnectedCallback(): void {\n    super.disconnectedCallback();\n    this.prepend(this._triggerElement); // Take back the trigger element previously moved to the light DOM\n    this._openPanelEventsController?.abort();\n  }\n\n  /**\n   * The reset value is the attribute value (the setup value), null otherwise.\n   * @internal\n   */\n  public formResetCallback(): void {\n    this.value = (this.hasAttribute('value') ? this.getAttribute('value') : null) as T;\n  }\n\n  /**\n   * @internal\n   */\n  public formStateRestoreCallback(\n    state: FormRestoreState | null,\n    _reason: FormRestoreReason,\n  ): void {\n    if (typeof state === 'string' || state == null) {\n      this.value = (state as T) ?? null;\n    } else if (state instanceof FormData) {\n      this._readFormData(state).then((array) => {\n        this.value = this.multiple ? array : array[0];\n      });\n    }\n  }\n\n  private async _readFormData(formData: FormData): Promise<T[]> {\n    return Promise.all(\n      formData\n        .getAll(this.name)\n        .map(async (entry) =>\n          entry instanceof Blob ? JSON.parse(await entry.text()) : (entry as T),\n        ),\n    );\n  }\n\n  private _syncProperties(): void {\n    this.querySelectorAll?.<SbbDividerElement | SbbOptionHintElement>(\n      'sbb-divider, sbb-option-hint',\n    ).forEach((el) => (el.negative = this.negative));\n\n    this.querySelectorAll?.<SbbOptionElement<T> | SbbOptGroupElement>(\n      'sbb-option, sbb-optgroup',\n    ).forEach((element) => {\n      element.toggleAttribute('data-negative', this.negative);\n      element.toggleAttribute('data-multiple', this.multiple);\n    });\n\n    this.querySelectorAll?.<SbbOptionElement<T> | SbbOptGroupElement>(\n      'sbb-option, sbb-optgroup',\n    ).forEach((e) => e.requestUpdate?.());\n  }\n\n  protected override shouldValidate(name: PropertyKey | undefined): boolean {\n    return super.shouldValidate(name) || name === 'value' || name === 'required';\n  }\n\n  protected override validate(): void {\n    super.validate();\n    if (this.required && this._options.every((o) => o.value !== this.value)) {\n      this.setValidityFlag('valueMissing', i18nSelectionRequired[this._languageController.current]);\n    } else {\n      this.removeValidityFlag('valueMissing');\n    }\n  }\n\n  private _setupSelect(): void {\n    this._setupOrigin();\n    this._setupTrigger();\n    this._didLoad = true;\n    this.completeUpdate();\n  }\n\n  /** Sets the originElement; if the component is used in a sbb-form-field uses it, otherwise uses the parentElement. */\n  private _setupOrigin(): void {\n    const formField = this.closest?.('sbb-form-field');\n    if (this._originElement) {\n      this._originResizeObserver.unobserve(this._originElement);\n    }\n    this._originElement =\n      formField?.shadowRoot?.querySelector?.('#overlay-anchor') ?? this.parentElement!;\n    if (this._originElement) {\n      this.toggleAttribute(\n        'data-option-panel-origin-borderless',\n        !!formField?.hasAttribute?.('borderless'),\n      );\n\n      if (this.isOpen) {\n        this._originResizeObserver.observe(this._originElement);\n      }\n    }\n  }\n\n  /**\n   * To assess screen-readers problems caused by the interaction between aria patterns and shadow DOM,\n   * we are forced to move the 'combobox' trigger element to the light DOM\n   */\n  private _setupTrigger(): void {\n    // Move the trigger before the sbb-select\n    this.parentElement!.insertBefore?.(this._triggerElement, this);\n  }\n\n  private _setOverlayPosition(): void {\n    setOverlayPosition(\n      this._overlay,\n      this._originElement,\n      this._optionContainer,\n      this.shadowRoot!.querySelector('.sbb-select__container')!,\n      this,\n    );\n  }\n\n  // In rare cases it can be that the animationEnd event is triggered twice.\n  // To avoid entering a corrupt state, exit when state is not expected.\n  private _onAnimationEnd(event: AnimationEvent): void {\n    if (event.animationName === 'open' && this.state === 'opening') {\n      this._handleOpening();\n    } else if (event.animationName === 'close' && this.state === 'closing') {\n      this._handleClosing();\n    }\n  }\n\n  private _handleOpening(): void {\n    this.state = 'opened';\n    this._attachOpenPanelEvents();\n    this._triggerElement.setAttribute('aria-expanded', 'true');\n    this._escapableOverlayController.connect();\n    if (this._originElement) {\n      this._originResizeObserver.observe(this._originElement);\n    }\n    this.dispatchOpenEvent();\n  }\n\n  private _handleClosing(): void {\n    this.state = 'closed';\n    this.shadowRoot?.querySelector<HTMLDivElement>('.sbb-select__container')?.hidePopover?.();\n    this._triggerElement.setAttribute('aria-expanded', 'false');\n    this._resetActiveElement();\n    this._optionContainer.scrollTop = 0;\n    this._escapableOverlayController.disconnect();\n    this.dispatchCloseEvent();\n  }\n\n  /** When an option is selected, updates the displayValue; it also closes the select if not `multiple`. */\n  private _onOptionSelected(optionSelectionChange: SbbOptionElement<T>): void {\n    if (!this.multiple) {\n      this._filteredOptions\n        .filter((option) => option.id !== optionSelectionChange.id)\n        .forEach((option) => (option.selected = false));\n      this.value = optionSelectionChange.value;\n    } else {\n      if (!this.value) {\n        this.value = [optionSelectionChange.value!];\n      } else if (Array.isArray(this.value) && !this.value.includes(optionSelectionChange.value!)) {\n        this.value = [...this.value, optionSelectionChange.value!];\n      }\n    }\n\n    this._dispatchInputEvents();\n  }\n\n  /** When an option is unselected in `multiple`, removes it from value and updates displayValue. */\n  private _onOptionDeselected(optionSelectionChange: SbbOptionElement<T>): void {\n    if (this.multiple && Array.isArray(this.value)) {\n      this.value = this.value.filter((el) => el !== optionSelectionChange.value);\n      this._dispatchInputEvents();\n    }\n  }\n\n  private _dispatchInputEvents(): void {\n    /** The input event fires when the value has been changed as a direct result of a user action. */\n    this.dispatchEvent(\n      new InputEvent('input', {\n        bubbles: true,\n        composed: true,\n      }),\n    );\n\n    /**\n     * The change event is fired when the user modifies the element's value.\n     * Unlike the input event, the change event is not necessarily fired\n     * for each alteration to an element's value.\n     */\n    this.dispatchEvent(new Event('change', { bubbles: true }));\n  }\n\n  private _attachOpenPanelEvents(): void {\n    this._openPanelEventsController = new AbortController();\n\n    // Recalculate the overlay position on scroll and window resize\n    document.addEventListener('scroll', () => this._setOverlayPosition(), {\n      passive: true,\n      signal: this._openPanelEventsController.signal,\n      // Without capture, other scroll contexts would not bubble to this event listener.\n      // Capture allows us to react to all scroll contexts in this DOM.\n      capture: true,\n    });\n    window.addEventListener('resize', () => this._setOverlayPosition(), {\n      passive: true,\n      signal: this._openPanelEventsController.signal,\n    });\n\n    // Close menu on backdrop click\n    window.addEventListener('pointerdown', (ev) => this._pointerDownListener(ev), {\n      signal: this._openPanelEventsController.signal,\n    });\n    window.addEventListener('pointerup', (ev) => this._closeOnBackdropClick(ev), {\n      signal: this._openPanelEventsController.signal,\n    });\n  }\n\n  private _onKeyDown(event: KeyboardEvent): void {\n    if (this.readOnly) {\n      return;\n    }\n\n    if (this.state === 'opened') {\n      this._openedPanelKeyboardInteraction(event);\n    } else if (this.state === 'closed') {\n      this._closedPanelKeyboardInteraction(event);\n    }\n  }\n\n  private _closedPanelKeyboardInteraction(event: KeyboardEvent): void {\n    if (this._checkForLetterSelection(event)) {\n      return this._setNextActiveOptionByText(event);\n    }\n\n    switch (event.key) {\n      case 'Enter':\n      case ' ':\n      case 'ArrowDown':\n      case 'ArrowUp':\n        event.preventDefault();\n        this.open();\n        break;\n    }\n  }\n\n  private _openedPanelKeyboardInteraction(event: KeyboardEvent): void {\n    if (this.readOnly || this.state !== 'opened') {\n      return;\n    }\n\n    if (this._checkForLetterSelection(event)) {\n      return this._setNextActiveOptionByText(event);\n    }\n\n    switch (event.key) {\n      case 'Tab':\n        this.close();\n        break;\n\n      case 'Enter':\n      case ' ':\n        event.preventDefault();\n        this._selectByKeyboard();\n        break;\n\n      case 'ArrowDown':\n      case 'ArrowUp':\n        event.preventDefault();\n        this._setNextActiveOption(event);\n        break;\n\n      case 'Home':\n      case 'PageUp':\n        event.preventDefault();\n        this._setNextActiveOption(event, 0);\n        break;\n\n      case 'End':\n      case 'PageDown':\n        event.preventDefault();\n        this._setNextActiveOption(event, this._filteredOptions.length - 1);\n        break;\n    }\n  }\n\n  private _checkForLetterSelection(event: KeyboardEvent): boolean {\n    return (\n      event.key === 'Backspace' ||\n      event.key === 'Clear' ||\n      (event.key.length === 1 &&\n        event.key !== ' ' &&\n        !event.altKey &&\n        !event.ctrlKey &&\n        !event.metaKey)\n    );\n  }\n\n  private _setNextActiveOptionByText(event: KeyboardEvent): void {\n    // Set timeout and the string to search.\n    if (typeof this._searchTimeout === typeof setTimeout) {\n      clearTimeout(this._searchTimeout);\n    }\n    this._searchTimeout = setTimeout(() => (this._searchString = ''), 1000);\n    this._searchString += event.key;\n\n    // Reorder the _filteredOption array to have the last selected element at the bottom.\n    const indexForSlice: number = this._activeItemIndex + 1;\n    const filteredOptionsSorted = [\n      ...this._filteredOptions.slice(indexForSlice),\n      ...this._filteredOptions.slice(0, indexForSlice),\n    ];\n\n    const match = filteredOptionsSorted.find(\n      (option) => option.textContent?.toLowerCase().indexOf(this._searchString.toLowerCase()) === 0,\n    );\n    if (match) {\n      // If an exact match has been found, go to that option.\n      this._setNextActiveOption(event, this._filteredOptions.indexOf(match));\n    } else if (\n      this._searchString.length > 1 &&\n      new RegExp(`^${this._searchString.charAt(0)}*$`).test(this._searchString)\n    ) {\n      // If no exact match has been found but the string to search is made by the same repeated letter,\n      // go to the first element, if exists, that matches the letter.\n      const firstMatch = filteredOptionsSorted.find(\n        (option) =>\n          option.textContent?.toLowerCase().indexOf(this._searchString[0].toLowerCase()) === 0,\n      );\n      if (firstMatch) {\n        this._setNextActiveOption(event, this._filteredOptions.indexOf(firstMatch));\n      }\n    } else {\n      // No match found, clear the timeout and the search term.\n      clearTimeout(this._searchTimeout);\n      this._searchString = '';\n    }\n  }\n\n  private _selectByKeyboard(): void {\n    const activeOption = this._filteredOptions[this._activeItemIndex];\n\n    if (this.multiple) {\n      activeOption?.['selectViaUserInteraction'](!activeOption.selected);\n    } else {\n      this.close();\n    }\n  }\n\n  private _setNextActiveOption(event: KeyboardEvent, index?: number): void {\n    const filteredOptions = this._filteredOptions;\n\n    // Prevent keyboard navigation if all options are disabled\n    if (filteredOptions.length === 0) {\n      return;\n    }\n\n    const nextIndex =\n      index ?? getNextElementIndex(event, this._activeItemIndex, filteredOptions.length);\n    const nextOption = filteredOptions[nextIndex];\n    const activeOption = filteredOptions[this._activeItemIndex];\n\n    this._setActiveElement(nextOption, activeOption);\n\n    if (!this.multiple) {\n      this._setSelectedElement(nextOption, activeOption);\n    } else if (event?.shiftKey) {\n      nextOption['selectViaUserInteraction'](!nextOption.selected);\n    }\n    this._activeItemIndex = nextIndex;\n  }\n\n  private _setActiveElement(\n    nextActiveOption: SbbOptionElement<T>,\n    lastActiveOption: SbbOptionElement<T> | null = null,\n    setActiveDescendant = true,\n  ): void {\n    nextActiveOption.setActive(true);\n    nextActiveOption.scrollIntoView({ block: 'nearest' });\n\n    if (setActiveDescendant) {\n      this._triggerElement.setAttribute('aria-activedescendant', nextActiveOption.id);\n    }\n\n    // Reset the previous\n    if (lastActiveOption && lastActiveOption !== nextActiveOption) {\n      lastActiveOption.setActive(false);\n    }\n  }\n\n  private _setSelectedElement(\n    nextActiveOption: SbbOptionElement<T>,\n    lastActiveOption: SbbOptionElement<T>,\n  ): void {\n    nextActiveOption['selectViaUserInteraction'](true);\n\n    if (lastActiveOption && lastActiveOption !== nextActiveOption) {\n      lastActiveOption['selectViaUserInteraction'](false);\n    }\n  }\n\n  private _resetActiveElement(): void {\n    const activeElement = this._filteredOptions[this._activeItemIndex];\n\n    if (activeElement) {\n      activeElement.setActive(false);\n    }\n    this._activeItemIndex = -1;\n    this._triggerElement.removeAttribute('aria-activedescendant');\n  }\n\n  // Check if the pointerdown event target is triggered on the menu.\n  private _pointerDownListener = (event: PointerEvent): void => {\n    this._isPointerDownEventOnMenu = isEventOnElement(this._overlay, event);\n  };\n\n  // Close menu on backdrop click.\n  private _closeOnBackdropClick = (event: PointerEvent): void => {\n    if (!this._isPointerDownEventOnMenu && !isEventOnElement(this._overlay, event)) {\n      this.close();\n    }\n  };\n\n  private _setValueFromSelected(): void {\n    const selected = this._getSelected();\n\n    if (Array.isArray(selected)) {\n      if (selected && selected.length > 0) {\n        const value: T[] = [];\n        for (const option of selected) {\n          value.push(option.value!);\n        }\n        this.value = value;\n      }\n    } else if (selected) {\n      this._activeItemIndex = this._filteredOptions.findIndex((option) => option === selected);\n      this.value = selected.value;\n    } else if (this.value) {\n      // If we arrive here without any options being selected,\n      // we should try to check the current value against the available options\n      // (and select it if any match is found).\n      this._onValueChanged(this.value);\n    }\n  }\n\n  private _getSelected(): SbbOptionElement<T> | SbbOptionElement<T>[] | null {\n    if (this.multiple) {\n      return this._filteredOptions.filter((option) => option.selected);\n    } else {\n      return this._filteredOptions.find((option) => option.selected) ?? null;\n    }\n  }\n\n  private _toggleOpening(): void {\n    if (this.disabled || this.formDisabled || this.readOnly) {\n      return;\n    }\n    this._triggerElement?.focus();\n\n    switch (this.state) {\n      case 'opened': {\n        this.close();\n        break;\n      }\n      case 'closed': {\n        this.open();\n        break;\n      }\n      default:\n        break;\n    }\n  }\n\n  private _spreadDeferredDisplayValue(\n    placeholder: TemplateResult,\n  ): (TemplateResult | Promise<TemplateResult>)[] {\n    return [this._deferredDisplayValue(placeholder), placeholder];\n  }\n\n  private async _deferredDisplayValue(placeholder: TemplateResult): Promise<TemplateResult> {\n    if (this.hydrationRequired) {\n      await this.hydrationComplete;\n    }\n    return this._displayValue ? html`${this._displayValue}` : placeholder;\n  }\n\n  protected override render(): TemplateResult {\n    return html`\n      <!-- This element is visually hidden and will be appended to the light DOM to allow screen\n      readers to work properly -->\n      <div\n        class=\"sbb-screen-reader-only sbb-select-trigger\"\n        tabindex=${this.disabled || this.formDisabled ? nothing : '0'}\n        role=\"combobox\"\n        aria-haspopup=\"listbox\"\n        aria-expanded=\"false\"\n        aria-required=${this.required.toString()}\n        aria-controls=${this._overlayId}\n        aria-owns=${this._overlayId}\n        @keydown=${this._onKeyDown}\n        @click=${this._toggleOpening}\n        ${ref((ref) => (this._triggerElement = ref as HTMLElement))}\n      >\n        ${until(...this._spreadDeferredDisplayValue(html`<span>${this.placeholder}</span>`))}\n      </div>\n\n      <!-- Visually display the value -->\n      <div class=\"sbb-select__trigger\" aria-hidden=\"true\">\n        ${until(\n          ...this._spreadDeferredDisplayValue(\n            html`<span class=\"sbb-select__trigger--placeholder\">${this.placeholder}</span>`,\n          ),\n        )}\n      </div>\n\n      <div class=\"sbb-select__gap-fix\"></div>\n      <div class=\"sbb-select__container\" popover=\"manual\">\n        <div class=\"sbb-select__gap-fix\">${overlayGapFixCorners()}</div>\n        <div\n          @animationend=${this._onAnimationEnd}\n          class=\"sbb-select__panel\"\n          ${ref((dialogRef) => (this._overlay = dialogRef as HTMLElement))}\n        >\n          <div class=\"sbb-select__wrapper\">\n            <div\n              id=${!ariaRoleOnHost ? this._overlayId : nothing}\n              class=\"sbb-select__options\"\n              role=${!ariaRoleOnHost ? 'listbox' : nothing}\n              ?aria-multiselectable=${this.multiple}\n              ${ref((containerRef) => (this._optionContainer = containerRef as HTMLElement))}\n            >\n              <slot @slotchange=${this._setValueFromSelected}></slot>\n            </div>\n          </div>\n        </div>\n      </div>\n    `;\n  }\n}\n\ndeclare global {\n  interface HTMLElementTagNameMap {\n    // eslint-disable-next-line @typescript-eslint/naming-convention\n    'sbb-select': SbbSelectElement;\n  }\n}\n"],"names":["state","ref"],"mappings":"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"}
1092
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"select.component.js","sources":["../../../../src/elements/select/select.component.ts"],"sourcesContent":["import { MutationController } from '@lit-labs/observers/mutation-controller.js';\nimport { ResizeController } from '@lit-labs/observers/resize-controller.js';\nimport type { CSSResultGroup, PropertyDeclaration, PropertyValues, TemplateResult } from 'lit';\nimport { html, isServer, nothing } from 'lit';\nimport { customElement, property, state } from 'lit/decorators.js';\nimport { ref } from 'lit/directives/ref.js';\nimport { until } from 'lit/directives/until.js';\n\nimport { getNextElementIndex } from '../core/a11y.js';\nimport { SbbOpenCloseBaseElement } from '../core/base-elements.js';\nimport { SbbEscapableOverlayController, SbbLanguageController } from '../core/controllers.js';\nimport { forceType, getOverride, handleDistinctChange } from '../core/decorators.js';\nimport {\n  isLean,\n  isNextjs,\n  isSafari,\n  isZeroAnimationDuration,\n  setOrRemoveAttribute,\n} from '../core/dom.js';\nimport { i18nSelectionRequired } from '../core/i18n.js';\nimport {\n  type FormRestoreReason,\n  type FormRestoreState,\n  SbbDisabledMixin,\n  SbbFormAssociatedMixin,\n  SbbHydrationMixin,\n  SbbNegativeMixin,\n  SbbReadonlyMixin,\n  SbbRequiredMixin,\n  SbbUpdateSchedulerMixin,\n} from '../core/mixins.js';\nimport { isEventOnElement, overlayGapFixCorners, setOverlayPosition } from '../core/overlay.js';\nimport type { SbbDividerElement } from '../divider.js';\nimport type { SbbOptGroupElement, SbbOptionElement, SbbOptionHintElement } from '../option.js';\n\nimport style from './select.scss?lit&inline';\n\n/**\n * On Safari, the aria role 'listbox' must be on the host element, or else VoiceOver won't work at all.\n * On the other hand, JAWS and NVDA need the role to be an \"immediate parent\" to the options, or else optgroups won't work.\n */\nconst ariaRoleOnHost = isSafari;\n\nlet nextId = 0;\n\n/**\n * It displays a panel with selectable options.\n *\n * @slot - Use the unnamed slot to add options.\n * @cssprop [--sbb-select-z-index=var(--sbb-overlay-default-z-index)] - To specify a custom stack order,\n * the `z-index` can be overridden by defining this CSS variable. The default `z-index` of the\n * component is set to `var(--sbb-overlay-default-z-index)` with a value of `1000`.\n * @overrideType value - (T = string | string[]) | null\n */\nexport\n@customElement('sbb-select')\nclass SbbSelectElement<T = string> extends SbbUpdateSchedulerMixin(\n  SbbDisabledMixin(\n    SbbNegativeMixin(\n      SbbHydrationMixin(\n        SbbRequiredMixin(\n          SbbReadonlyMixin(\n            SbbFormAssociatedMixin<typeof SbbOpenCloseBaseElement>(SbbOpenCloseBaseElement),\n          ),\n        ),\n      ),\n    ),\n  ),\n) {\n  public static override readonly role = ariaRoleOnHost ? 'listbox' : null;\n  public static override styles: CSSResultGroup = style;\n\n  // TODO: fix using ...super.events requires: https://github.com/sbb-design-systems/lyne-components/issues/2600\n  public static override readonly events = {\n    change: 'change',\n    input: 'input',\n    displayvaluechange: 'displayvaluechange',\n    beforeopen: 'beforeopen',\n    open: 'open',\n    beforeclose: 'beforeclose',\n    close: 'close',\n  } as const;\n\n  /** The placeholder used if no value has been selected. */\n  @forceType()\n  @property()\n  public accessor placeholder: string = '';\n\n  /** Whether the select allows for multiple selection. */\n  @forceType()\n  @handleDistinctChange((e: SbbSelectElement<T>, newValue: boolean) =>\n    e._onMultipleChanged(newValue),\n  )\n  @property({ reflect: true, type: Boolean })\n  public accessor multiple: boolean = false;\n\n  @forceType()\n  @handleDistinctChange((e: SbbSelectElement<T>, newValue: boolean) =>\n    e._closeOnDisabledReadonlyChanged(newValue),\n  )\n  @property({ reflect: true, type: Boolean })\n  @getOverride((e: SbbSelectElement<T>, v: boolean): boolean => v || e.isDisabledExternally())\n  public override accessor disabled: boolean = false;\n\n  /** Value of the form element. */\n  @property()\n  public set value(value: T[] | T) {\n    this._value = value;\n    this._updateOptionsFromValue();\n    this._isValueManuallyAssigned = true;\n  }\n  public get value(): T[] | T | null {\n    return this._value;\n  }\n  private _value: T | T[] | null = null;\n\n  /**\n   * Size variant, either m or s.\n   * @default 'm' / 's' (lean)\n   */\n  @property({ reflect: true }) public accessor size: 'm' | 's' = isLean() ? 's' : 'm';\n\n  /**\n   * Form type of element.\n   * @default 'select-one / select-multiple'\n   */\n  public override get type(): string {\n    return this.multiple ? 'select-multiple' : 'select-one';\n  }\n\n  /** The value displayed by the component. */\n  @state() private accessor _displayValue: string | null = null;\n\n  private _originResizeObserver = new ResizeController(this, {\n    target: null,\n    skipInitial: true,\n    callback: () => {\n      if (this.isOpen) {\n        this._setOverlayPosition();\n      }\n    },\n  });\n\n  private _overlay!: HTMLElement;\n  private _optionContainer!: HTMLElement;\n  private _originElement!: HTMLElement;\n  private _triggerElement!: HTMLElement;\n  private _openPanelEventsController!: AbortController;\n  private _escapableOverlayController = new SbbEscapableOverlayController(this);\n  private _overlayId = `sbb-select-${++nextId}`;\n  private _activeItemIndex = -1;\n  private _searchTimeout?: ReturnType<typeof setTimeout>;\n  private _searchString = '';\n  private _didLoad = false;\n  private _isPointerDownEventOnMenu: boolean = false;\n  private _languageController = new SbbLanguageController(this);\n  private _isValueManuallyAssigned = false;\n\n  /**\n   * The 'combobox' input element\n   * @internal\n   */\n  public get inputElement(): HTMLElement {\n    return this._triggerElement;\n  }\n\n  /** Returns all SbbOptionElements from this sbb-select instance. */\n  public get options(): SbbOptionElement<T>[] {\n    const options: SbbOptionElement<T>[] = [];\n    this.querySelectorAll?.<SbbOptionElement<T>>('sbb-option').forEach((option) => {\n      customElements.upgrade(option);\n      options.push(option);\n    });\n    return options;\n  }\n\n  public constructor() {\n    super();\n    this.addEventListener?.('optionselectionchange', (e: Event) => this._onOptionChanged(e));\n    this.addEventListener?.('optionLabelChanged', (e: Event) => this._onOptionLabelChanged(e));\n    this.addEventListener?.('ɵoptgroupslotchange', () => this._onSlotChange(), { capture: true });\n    this.addEventListener?.('click', (e: MouseEvent) => {\n      const target = e.target as SbbSelectElement<T> | SbbOptionElement<T>;\n      if (target.localName === 'sbb-option') {\n        // Option click\n        if (!this.multiple && !target.disabled) {\n          this.close();\n          this.focus();\n        }\n      } else {\n        this._toggleOpening();\n      }\n    });\n\n    this.addController(\n      new MutationController(this, {\n        config: { attributeFilter: ['aria-labelledby', 'aria-label', 'aria-describedby'] },\n        callback: () => this._syncAriaLabels(),\n      }),\n    );\n  }\n\n  private _syncAriaLabels(): void {\n    if (!this._triggerElement || isServer) {\n      return;\n    }\n\n    setOrRemoveAttribute(\n      this._triggerElement,\n      'aria-labelledby',\n      this.getAttribute('aria-labelledby'),\n    );\n    setOrRemoveAttribute(this._triggerElement, 'aria-label', this.getAttribute('aria-label'));\n    setOrRemoveAttribute(\n      this._triggerElement,\n      'aria-describedby',\n      this.getAttribute('aria-describedby'),\n    );\n\n    // Using the associated labels is only a fallback.\n    // The drawback is, that it doesn't get updated automatically when the list of label changes.\n    if (\n      !this.getAttribute('aria-label') &&\n      !this.getAttribute('aria-labelledby') &&\n      this.internals.labels.length\n    ) {\n      this._triggerElement?.setAttribute(\n        'aria-label',\n        Array.from(this.internals.labels)\n          .map((label) => label.textContent)\n          .join(', '),\n      );\n    }\n  }\n\n  /** Opens the selection panel. */\n  public open(): void {\n    if (\n      this.state !== 'closed' ||\n      !this._overlay ||\n      this.options.length === 0 ||\n      this.disabled ||\n      this.formDisabled ||\n      !this.dispatchBeforeOpenEvent()\n    ) {\n      return;\n    }\n\n    this.shadowRoot?.querySelector<HTMLDivElement>('.sbb-select__container')?.showPopover?.();\n    this.state = 'opening';\n    this.toggleAttribute('data-expanded', true);\n    this._setOverlayPosition();\n\n    // If the animation duration is zero, the animationend event is not always fired reliably.\n    // In this case we directly set the `opened` state.\n    if (this._isZeroAnimationDuration()) {\n      this._handleOpening();\n    }\n  }\n\n  /** Closes the selection panel. */\n  public close(): void {\n    if (this.state !== 'opened' || !this.dispatchBeforeCloseEvent()) {\n      return;\n    }\n\n    this.state = 'closing';\n    this.toggleAttribute('data-expanded', false);\n    this._openPanelEventsController.abort();\n    if (this._originElement) {\n      this._originResizeObserver.unobserve(this._originElement);\n    }\n\n    // If the animation duration is zero, the animationend event is not always fired reliably.\n    // In this case we directly set the `closed` state.\n    if (this._isZeroAnimationDuration()) {\n      this._handleClosing();\n    }\n  }\n\n  private _isZeroAnimationDuration(): boolean {\n    return isZeroAnimationDuration(this, '--sbb-options-panel-animation-duration');\n  }\n\n  /** Gets the current displayed value. */\n  public getDisplayValue(): string {\n    return this._displayValue ?? '';\n  }\n\n  private _selectableOptions(): SbbOptionElement<T>[] {\n    return this.options.filter((opt) => !opt.disabled && !opt.hasAttribute('data-group-disabled'));\n  }\n\n  /** Listens to option changes. */\n  private _onOptionChanged(event: Event): void {\n    const target = event.target as SbbOptionElement<T>;\n    if (target.selected) {\n      this._onOptionSelected(target);\n    } else {\n      this._onOptionDeselected(target);\n    }\n  }\n\n  /** Listens to option changes. */\n  private _onOptionLabelChanged(event: Event): void {\n    const target = event.target as SbbOptionElement<T>;\n    const selected = this._getSelected();\n\n    if (\n      (!Array.isArray(selected) && target !== selected) ||\n      (Array.isArray(selected) && !selected.includes(target))\n    ) {\n      return;\n    }\n\n    this._updateDisplayValue();\n  }\n\n  private _updateDisplayValue(): void {\n    const selected = this._getSelected();\n    if (Array.isArray(selected)) {\n      this._displayValue = selected.map((o) => o.textContent?.trim()).join(', ') || null;\n    } else if (selected) {\n      this._displayValue = selected?.textContent?.trim() || null;\n    } else {\n      this._displayValue = null;\n    }\n\n    /** @internal */\n    this.dispatchEvent(new Event('displayvaluechange', { bubbles: true, composed: true }));\n  }\n\n  /**\n   * The `value` property should be adapted when the `multiple` property changes:\n   *   - if it changes to true, the 'value' is set to an array;\n   *   - if it changes to false, the first available option is set as 'value' otherwise it's set to null.\n   */\n  private _onMultipleChanged(isChangingToMultiple: boolean): void {\n    if (isChangingToMultiple) {\n      this._value = this._isValueManuallyAssigned || this.value != null ? [this.value as T] : [];\n    } else if (Array.isArray(this.value)) {\n      this._value = this.value.length ? this.value[0] : null;\n    }\n    this._updateDisplayValue();\n  }\n\n  /**\n   * If the `disabled` or the `readonly` properties are set, and the panel is open, close it.\n   */\n  private _closeOnDisabledReadonlyChanged(newValue: boolean): void {\n    if (this.isOpen && newValue) {\n      this.close();\n    }\n  }\n\n  protected override firstUpdated(changedProperties: PropertyValues<this>): void {\n    super.firstUpdated(changedProperties);\n\n    // Wait for ssr hydration\n    if (!isNextjs()) {\n      this.startUpdate();\n      this._setupSelect();\n    }\n  }\n\n  /** @internal */\n  public override focus(): void {\n    // Forward focus to the trigger element.\n    this._triggerElement?.focus();\n  }\n\n  /** @internal */\n  public override blur(): void {\n    // Forward blur to the trigger element.\n    this._triggerElement?.blur();\n  }\n\n  /**\n   * Removes element's first attribute whose qualified name is qualifiedName.\n   *\n   * [MDN Reference](https://developer.mozilla.org/docs/Web/API/Element/removeAttribute)\n   * @internal We need to override this due to a hydration issue with Next.js.\n   */\n  public override removeAttribute(qualifiedName: string): void {\n    // In Next.js the hydration needs to finish before we can manipulate the light DOM.\n    // @lit/react calls removeAttribute('defer-hydration') in a useLayoutEffect,\n    // which is done after hydration is finished. Due to this we intercept this call\n    // in overriding removeAttribute to finish initialization of the sbb-select.\n    // https://github.com/lit/lit/blob/main/packages/react/src/create-component.ts#L293-L296\n    // We also need to wait for update complete in order to be sure that the shadow DOM has been rendered.\n    if (isNextjs() && qualifiedName === 'defer-hydration' && !this._didLoad) {\n      this.updateComplete.then(() => this._setupSelect());\n    }\n    super.removeAttribute(qualifiedName);\n  }\n\n  public override connectedCallback(): void {\n    super.connectedCallback();\n\n    if (ariaRoleOnHost) {\n      this.id ||= this._overlayId;\n    }\n\n    const formField = this.closest?.('sbb-form-field') ?? this.closest?.('[data-form-field]');\n\n    if (formField) {\n      this.negative = formField.hasAttribute('negative');\n    }\n    this._syncProperties();\n    this._syncAriaLabels();\n\n    if (this._didLoad) {\n      this._setupOrigin();\n      this._setupTrigger();\n    }\n  }\n\n  public override requestUpdate(\n    name?: PropertyKey,\n    oldValue?: unknown,\n    options?: PropertyDeclaration,\n  ): void {\n    super.requestUpdate(name, oldValue, options);\n    if (!name && this.hasUpdated) {\n      setTimeout(() => this._syncAriaLabels());\n    }\n  }\n\n  protected override willUpdate(changedProperties: PropertyValues<this>): void {\n    super.willUpdate(changedProperties);\n\n    if (changedProperties.has('negative') || changedProperties.has('multiple')) {\n      this._syncProperties();\n    }\n    if (changedProperties.has('readOnly')) {\n      this._closeOnDisabledReadonlyChanged(this.readOnly);\n    }\n  }\n\n  public override disconnectedCallback(): void {\n    super.disconnectedCallback();\n    this.prepend(this._triggerElement); // Take back the trigger element previously moved to the light DOM\n    this._openPanelEventsController?.abort();\n  }\n\n  /**\n   * The reset value is the attribute value (the setup value), null otherwise.\n   * @internal\n   */\n  public formResetCallback(): void {\n    this.value = (this.hasAttribute('value') ? this.getAttribute('value') : null) as T;\n  }\n\n  /**\n   * @internal\n   */\n  public formStateRestoreCallback(\n    state: FormRestoreState | null,\n    _reason: FormRestoreReason,\n  ): void {\n    try {\n      const { value, manuallyAssigned } = JSON.parse(state as string) as {\n        value: T | T[];\n        manuallyAssigned: boolean;\n      };\n\n      const values = Array.isArray(value) ? value : [value];\n\n      if (values.some((v) => v !== null && typeof v === 'object')) {\n        console.warn(\n          `Restoring complex objects is not supported for sbb-select with state ${state}`,\n        );\n        return;\n      }\n\n      this._isValueManuallyAssigned = manuallyAssigned;\n      this._value = value;\n      this._updateOptionsFromValue();\n    } catch {\n      console.warn(`Failed to restore sbb-select with state ${state}`);\n    }\n  }\n\n  protected override formState(): FormRestoreState {\n    return JSON.stringify({ value: this.value, manuallyAssigned: this._isValueManuallyAssigned });\n  }\n\n  private _syncProperties(): void {\n    this.querySelectorAll?.<SbbDividerElement | SbbOptionHintElement>(\n      'sbb-divider, sbb-option-hint',\n    ).forEach((el) => (el.negative = this.negative));\n\n    this.querySelectorAll?.<SbbOptionElement<T> | SbbOptGroupElement>(\n      'sbb-option, sbb-optgroup',\n    ).forEach((element) => {\n      element.toggleAttribute('data-negative', this.negative);\n      element.toggleAttribute('data-multiple', this.multiple);\n    });\n\n    this.querySelectorAll?.<SbbOptionElement<T> | SbbOptGroupElement>(\n      'sbb-option, sbb-optgroup',\n    ).forEach((e) => e.requestUpdate?.());\n  }\n\n  protected override shouldValidate(name: PropertyKey | undefined): boolean {\n    return super.shouldValidate(name) || name === 'value' || name === 'required';\n  }\n\n  protected override validate(): void {\n    super.validate();\n    if (\n      this.required &&\n      (this.options.every((o) => o.value !== this.value) ||\n        (!this._isValueManuallyAssigned && this.value == null))\n    ) {\n      this.setValidityFlag('valueMissing', i18nSelectionRequired[this._languageController.current]);\n    } else {\n      this.removeValidityFlag('valueMissing');\n    }\n  }\n\n  private _setupSelect(): void {\n    this._setupOrigin();\n    this._setupTrigger();\n    this._didLoad = true;\n    this.completeUpdate();\n  }\n\n  /** Sets the originElement; if the component is used in a sbb-form-field uses it, otherwise uses the parentElement. */\n  private _setupOrigin(): void {\n    const formField = this.closest?.('sbb-form-field');\n    if (this._originElement) {\n      this._originResizeObserver.unobserve(this._originElement);\n    }\n    this._originElement =\n      formField?.shadowRoot?.querySelector?.('#overlay-anchor') ?? this.parentElement!;\n    if (this._originElement) {\n      this.toggleAttribute(\n        'data-option-panel-origin-borderless',\n        !!formField?.hasAttribute?.('borderless'),\n      );\n\n      if (this.isOpen) {\n        this._originResizeObserver.observe(this._originElement);\n      }\n    }\n  }\n\n  /**\n   * To assess screen-readers problems caused by the interaction between aria patterns and shadow DOM,\n   * we are forced to move the 'combobox' trigger element to the light DOM\n   */\n  private _setupTrigger(): void {\n    // Move the trigger before the sbb-select\n    this.parentElement!.insertBefore?.(this._triggerElement, this);\n  }\n\n  private _setOverlayPosition(): void {\n    setOverlayPosition(\n      this._overlay,\n      this._originElement,\n      this._optionContainer,\n      this.shadowRoot!.querySelector('.sbb-select__container')!,\n      this,\n    );\n  }\n\n  // In rare cases it can be that the animationEnd event is triggered twice.\n  // To avoid entering a corrupt state, exit when state is not expected.\n  private _onAnimationEnd(event: AnimationEvent): void {\n    if (event.animationName === 'open' && this.state === 'opening') {\n      this._handleOpening();\n    } else if (event.animationName === 'close' && this.state === 'closing') {\n      this._handleClosing();\n    }\n  }\n\n  private _handleOpening(): void {\n    this.state = 'opened';\n    this._attachOpenPanelEvents();\n    this._triggerElement.setAttribute('aria-expanded', 'true');\n    this._escapableOverlayController.connect();\n    if (this._originElement) {\n      this._originResizeObserver.observe(this._originElement);\n    }\n    this.dispatchOpenEvent();\n  }\n\n  private _handleClosing(): void {\n    this.state = 'closed';\n    this.shadowRoot?.querySelector<HTMLDivElement>('.sbb-select__container')?.hidePopover?.();\n    this._triggerElement.setAttribute('aria-expanded', 'false');\n    this._resetActiveElement();\n    this._optionContainer.scrollTop = 0;\n    this._escapableOverlayController.disconnect();\n    this.dispatchCloseEvent();\n  }\n\n  /** When an option is selected, updates the displayValue; it also closes the select if not `multiple`. */\n  private _onOptionSelected(option: SbbOptionElement<T>): void {\n    if (!this.multiple) {\n      this._value = option.value;\n    } else if (!this.value) {\n      this._value = [option.value!];\n    } else if (Array.isArray(this.value) && !this.value.includes(option.value!)) {\n      this._value = [...this.value, option.value!];\n    }\n\n    this._updateOptionsFromValue();\n    this._dispatchInputEvents();\n  }\n\n  /** When an option is unselected in `multiple`, removes it from value and updates displayValue. */\n  private _onOptionDeselected(optionSelectionChange: SbbOptionElement<T>): void {\n    if (this.multiple && Array.isArray(this.value)) {\n      this._value = this.value.filter((el) => el !== optionSelectionChange.value);\n      this._updateOptionsFromValue();\n      this._dispatchInputEvents();\n    }\n  }\n\n  private _dispatchInputEvents(): void {\n    /** The input event fires when the value has been changed as a direct result of a user action. */\n    this.dispatchEvent(\n      new InputEvent('input', {\n        bubbles: true,\n        composed: true,\n      }),\n    );\n\n    /**\n     * The change event is fired when the user modifies the element's value.\n     * Unlike the input event, the change event is not necessarily fired\n     * for each alteration to an element's value.\n     */\n    this.dispatchEvent(new Event('change', { bubbles: true }));\n  }\n\n  private _attachOpenPanelEvents(): void {\n    this._openPanelEventsController = new AbortController();\n\n    // Recalculate the overlay position on scroll and window resize\n    document.addEventListener('scroll', () => this._setOverlayPosition(), {\n      passive: true,\n      signal: this._openPanelEventsController.signal,\n      // Without capture, other scroll contexts would not bubble to this event listener.\n      // Capture allows us to react to all scroll contexts in this DOM.\n      capture: true,\n    });\n    window.addEventListener('resize', () => this._setOverlayPosition(), {\n      passive: true,\n      signal: this._openPanelEventsController.signal,\n    });\n\n    // Close menu on backdrop click\n    window.addEventListener('pointerdown', (ev) => this._pointerDownListener(ev), {\n      signal: this._openPanelEventsController.signal,\n    });\n    window.addEventListener('pointerup', (ev) => this._closeOnBackdropClick(ev), {\n      signal: this._openPanelEventsController.signal,\n    });\n  }\n\n  private _onKeyDown(event: KeyboardEvent): void {\n    if (this.readOnly) {\n      return;\n    }\n\n    if (this.state === 'opened') {\n      this._openedPanelKeyboardInteraction(event);\n    } else if (this.state === 'closed') {\n      this._closedPanelKeyboardInteraction(event);\n    }\n  }\n\n  private _closedPanelKeyboardInteraction(event: KeyboardEvent): void {\n    if (this._checkForLetterSelection(event)) {\n      return this._setNextActiveOptionByText(event);\n    }\n\n    switch (event.key) {\n      case 'Enter':\n      case ' ':\n      case 'ArrowDown':\n      case 'ArrowUp':\n        event.preventDefault();\n        this.open();\n        break;\n    }\n  }\n\n  private _openedPanelKeyboardInteraction(event: KeyboardEvent): void {\n    if (this.readOnly || this.state !== 'opened') {\n      return;\n    }\n\n    if (this._checkForLetterSelection(event)) {\n      return this._setNextActiveOptionByText(event);\n    }\n\n    switch (event.key) {\n      case 'Tab':\n        this.close();\n        break;\n\n      case 'Enter':\n      case ' ':\n        event.preventDefault();\n        this._selectByKeyboard();\n        break;\n\n      case 'ArrowDown':\n      case 'ArrowUp':\n        event.preventDefault();\n        this._setNextActiveOption(event);\n        break;\n\n      case 'Home':\n      case 'PageUp':\n        event.preventDefault();\n        this._setNextActiveOption(event, 0);\n        break;\n\n      case 'End':\n      case 'PageDown':\n        event.preventDefault();\n        this._setNextActiveOption(event, this._selectableOptions().length - 1);\n        break;\n    }\n  }\n\n  private _checkForLetterSelection(event: KeyboardEvent): boolean {\n    return (\n      event.key === 'Backspace' ||\n      event.key === 'Clear' ||\n      (event.key.length === 1 &&\n        event.key !== ' ' &&\n        !event.altKey &&\n        !event.ctrlKey &&\n        !event.metaKey)\n    );\n  }\n\n  private _setNextActiveOptionByText(event: KeyboardEvent): void {\n    // Set timeout and the string to search.\n    if (typeof this._searchTimeout === typeof setTimeout) {\n      clearTimeout(this._searchTimeout);\n    }\n    this._searchTimeout = setTimeout(() => (this._searchString = ''), 1000);\n    this._searchString += event.key;\n\n    // Reorder the _filteredOption array to have the last selected element at the bottom.\n    const indexForSlice: number = this._activeItemIndex + 1;\n    const selectableOptions = this._selectableOptions();\n    const filteredOptionsSorted = [\n      ...selectableOptions.slice(indexForSlice),\n      ...selectableOptions.slice(0, indexForSlice),\n    ];\n\n    const match = filteredOptionsSorted.find(\n      (option) => option.textContent?.toLowerCase().indexOf(this._searchString.toLowerCase()) === 0,\n    );\n    if (match) {\n      // If an exact match has been found, go to that option.\n      this._setNextActiveOption(event, selectableOptions.indexOf(match));\n    } else if (\n      this._searchString.length > 1 &&\n      new RegExp(`^${this._searchString.charAt(0)}*$`).test(this._searchString)\n    ) {\n      // If no exact match has been found but the string to search is made by the same repeated letter,\n      // go to the first element, if exists, that matches the letter.\n      const firstMatch = filteredOptionsSorted.find(\n        (option) =>\n          option.textContent?.toLowerCase().indexOf(this._searchString[0].toLowerCase()) === 0,\n      );\n      if (firstMatch) {\n        this._setNextActiveOption(event, selectableOptions.indexOf(firstMatch));\n      }\n    } else {\n      // No match found, clear the timeout and the search term.\n      clearTimeout(this._searchTimeout);\n      this._searchString = '';\n    }\n  }\n\n  private _selectByKeyboard(): void {\n    const activeOption = this._selectableOptions()[this._activeItemIndex];\n\n    if (this.multiple) {\n      activeOption?.['selectViaUserInteraction'](!activeOption.selected);\n    } else {\n      this.close();\n    }\n  }\n\n  private _setNextActiveOption(event: KeyboardEvent, index?: number): void {\n    const filteredOptions = this._selectableOptions();\n\n    // Prevent keyboard navigation if all options are disabled\n    if (filteredOptions.length === 0) {\n      return;\n    }\n\n    const nextIndex =\n      index ?? getNextElementIndex(event, this._activeItemIndex, filteredOptions.length);\n    const nextOption = filteredOptions[nextIndex];\n    const activeOption = filteredOptions[this._activeItemIndex];\n\n    this._setActiveElement(nextOption, activeOption);\n\n    if (!this.multiple) {\n      this._setSelectedElement(nextOption, activeOption);\n    } else if (event?.shiftKey) {\n      nextOption['selectViaUserInteraction'](!nextOption.selected);\n    }\n    this._activeItemIndex = nextIndex;\n  }\n\n  private _setActiveElement(\n    nextActiveOption: SbbOptionElement<T>,\n    lastActiveOption: SbbOptionElement<T> | null = null,\n    setActiveDescendant = true,\n  ): void {\n    nextActiveOption.setActive(true);\n    nextActiveOption.scrollIntoView({ block: 'nearest' });\n\n    if (setActiveDescendant) {\n      this._triggerElement.setAttribute('aria-activedescendant', nextActiveOption.id);\n    }\n\n    // Reset the previous\n    if (lastActiveOption && lastActiveOption !== nextActiveOption) {\n      lastActiveOption.setActive(false);\n    }\n  }\n\n  private _setSelectedElement(\n    nextActiveOption: SbbOptionElement<T>,\n    lastActiveOption: SbbOptionElement<T>,\n  ): void {\n    nextActiveOption['selectViaUserInteraction'](true);\n\n    if (lastActiveOption && lastActiveOption !== nextActiveOption) {\n      lastActiveOption['selectViaUserInteraction'](false);\n    }\n  }\n\n  private _resetActiveElement(): void {\n    const activeElement = this._selectableOptions()[this._activeItemIndex];\n\n    if (activeElement) {\n      activeElement.setActive(false);\n    }\n    this._activeItemIndex = -1;\n    this._triggerElement.removeAttribute('aria-activedescendant');\n  }\n\n  // Check if the pointerdown event target is triggered on the menu.\n  private _pointerDownListener = (event: PointerEvent): void => {\n    this._isPointerDownEventOnMenu = isEventOnElement(this._overlay, event);\n  };\n\n  // Close menu on backdrop click.\n  private _closeOnBackdropClick = (event: PointerEvent): void => {\n    if (!this._isPointerDownEventOnMenu && !isEventOnElement(this._overlay, event)) {\n      this.close();\n    }\n  };\n\n  private _updateOptionsFromValue(): void {\n    const value = Array.isArray(this.value) ? this.value : [this.value];\n\n    const displayValues = [];\n    for (const option of this.options) {\n      option.selected = value.includes(option.value);\n      if (option.selected) {\n        displayValues.push(option);\n      }\n    }\n\n    this._updateDisplayValue();\n\n    if (!Array.isArray(this.value)) {\n      this._activeItemIndex = this._selectableOptions().findIndex(\n        (option) => option.value === this.value,\n      );\n    }\n  }\n\n  private _updateValueFromOptions(): void {\n    const selected = this._getSelected();\n\n    if (Array.isArray(selected)) {\n      if (selected && selected.length > 0) {\n        const value: T[] = [];\n        for (const option of selected) {\n          value.push(option.value!);\n        }\n        this._value = value;\n      }\n    } else if (selected) {\n      this._activeItemIndex = this._selectableOptions().findIndex((option) => option === selected);\n      this._value = selected.value;\n    }\n\n    this._updateDisplayValue();\n  }\n\n  private _onSlotChange(): void {\n    if (this._isValueManuallyAssigned) {\n      this._updateOptionsFromValue();\n    } else {\n      this._updateValueFromOptions();\n    }\n  }\n\n  private _getSelected(): SbbOptionElement<T> | SbbOptionElement<T>[] | null {\n    if (this.multiple) {\n      return this.options.filter((option) => option.selected);\n    } else {\n      return this.options.find((option) => option.selected) ?? null;\n    }\n  }\n\n  private _toggleOpening(): void {\n    if (this.disabled || this.formDisabled || this.readOnly) {\n      return;\n    }\n    this._triggerElement?.focus();\n\n    switch (this.state) {\n      case 'opened': {\n        this.close();\n        break;\n      }\n      case 'closed': {\n        this.open();\n        break;\n      }\n      default:\n        break;\n    }\n  }\n\n  private _spreadDeferredDisplayValue(\n    placeholder: TemplateResult,\n  ): (TemplateResult | Promise<TemplateResult>)[] {\n    return [this._deferredDisplayValue(placeholder), placeholder];\n  }\n\n  private async _deferredDisplayValue(placeholder: TemplateResult): Promise<TemplateResult> {\n    if (this.hydrationRequired) {\n      await this.hydrationComplete;\n    }\n    return this._displayValue ? html`${this._displayValue}` : placeholder;\n  }\n\n  protected override render(): TemplateResult {\n    return html`\n      <!-- This element is visually hidden and will be appended to the light DOM to allow screen\n      readers to work properly -->\n      <div\n        class=\"sbb-screen-reader-only sbb-select-trigger\"\n        tabindex=${this.disabled || this.formDisabled ? nothing : '0'}\n        role=\"combobox\"\n        aria-haspopup=\"listbox\"\n        aria-expanded=\"false\"\n        aria-required=${this.required.toString()}\n        aria-controls=${this._overlayId}\n        aria-owns=${this._overlayId}\n        @keydown=${this._onKeyDown}\n        @click=${this._toggleOpening}\n        ${ref((ref) => (this._triggerElement = ref as HTMLElement))}\n      >\n        ${until(...this._spreadDeferredDisplayValue(html`<span>${this.placeholder}</span>`))}\n      </div>\n\n      <!-- Visually display the value -->\n      <div class=\"sbb-select__trigger\" aria-hidden=\"true\">\n        ${until(\n          ...this._spreadDeferredDisplayValue(\n            html`<span class=\"sbb-select__trigger--placeholder\">${this.placeholder}</span>`,\n          ),\n        )}\n      </div>\n\n      <div class=\"sbb-select__gap-fix\"></div>\n      <div class=\"sbb-select__container\" popover=\"manual\">\n        <div class=\"sbb-select__gap-fix\">${overlayGapFixCorners()}</div>\n        <div\n          @animationend=${this._onAnimationEnd}\n          class=\"sbb-select__panel\"\n          ${ref((dialogRef) => (this._overlay = dialogRef as HTMLElement))}\n        >\n          <div class=\"sbb-select__wrapper\">\n            <div\n              id=${!ariaRoleOnHost ? this._overlayId : nothing}\n              class=\"sbb-select__options\"\n              role=${!ariaRoleOnHost ? 'listbox' : nothing}\n              ?aria-multiselectable=${this.multiple}\n              ${ref((containerRef) => (this._optionContainer = containerRef as HTMLElement))}\n            >\n              <slot @slotchange=${this._onSlotChange}></slot>\n            </div>\n          </div>\n        </div>\n      </div>\n    `;\n  }\n}\n\ndeclare global {\n  interface HTMLElementTagNameMap {\n    // eslint-disable-next-line @typescript-eslint/naming-convention\n    'sbb-select': SbbSelectElement;\n  }\n}\n"],"names":["state","ref"],"mappings":"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"}