@ultraviolet/ui 1.51.2 → 1.51.3

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 (127) hide show
  1. package/dist/components/ActionBar/index.js +16 -26
  2. package/dist/components/Alert/index.js +77 -88
  3. package/dist/components/Avatar/index.js +29 -26
  4. package/dist/components/Badge/index.js +36 -52
  5. package/dist/components/Banner/index.js +64 -139
  6. package/dist/components/BarChart/Tooltip.js +23 -18
  7. package/dist/components/BarChart/index.js +29 -42
  8. package/dist/components/BarStack/index.js +89 -163
  9. package/dist/components/Breadcrumbs/index.js +30 -33
  10. package/dist/components/Bullet/index.js +24 -28
  11. package/dist/components/Button/index.js +117 -159
  12. package/dist/components/Card/index.js +45 -52
  13. package/dist/components/Carousel/index.js +75 -99
  14. package/dist/components/Checkbox/index.js +246 -343
  15. package/dist/components/CheckboxGroup/index.js +60 -56
  16. package/dist/components/CopyButton/index.js +1 -15
  17. package/dist/components/DateInput/index.js +124 -243
  18. package/dist/components/Dialog/Context.js +1 -3
  19. package/dist/components/Dialog/index.js +22 -43
  20. package/dist/components/Dialog/subComponents/Button.js +1 -10
  21. package/dist/components/Dialog/subComponents/Stack.js +3 -1
  22. package/dist/components/Dialog/subComponents/Text.js +3 -1
  23. package/dist/components/EmptyState/index.js +58 -50
  24. package/dist/components/Expandable/index.js +19 -23
  25. package/dist/components/GlobalAlert/GlobalAlertLink.js +4 -20
  26. package/dist/components/GlobalAlert/index.js +34 -87
  27. package/dist/components/LineChart/CustomLegend.js +76 -72
  28. package/dist/components/LineChart/Tooltip.js +26 -37
  29. package/dist/components/LineChart/helpers.js +11 -9
  30. package/dist/components/LineChart/index.js +38 -55
  31. package/dist/components/Link/index.js +86 -125
  32. package/dist/components/List/Body.js +12 -7
  33. package/dist/components/List/Cell.js +36 -33
  34. package/dist/components/List/HeaderCell.js +54 -68
  35. package/dist/components/List/HeaderRow.js +16 -18
  36. package/dist/components/List/ListContext.js +68 -87
  37. package/dist/components/List/Row.js +136 -169
  38. package/dist/components/List/SelectBar.js +31 -14
  39. package/dist/components/List/SkeletonRows.js +42 -23
  40. package/dist/components/List/index.js +39 -59
  41. package/dist/components/Loader/index.js +44 -65
  42. package/dist/components/Menu/Item.js +52 -62
  43. package/dist/components/Menu/index.js +105 -171
  44. package/dist/components/MenuV2/Group.js +14 -16
  45. package/dist/components/MenuV2/Item.js +67 -74
  46. package/dist/components/MenuV2/index.js +76 -110
  47. package/dist/components/Meter/index.js +44 -57
  48. package/dist/components/Modal/Dialog.js +44 -100
  49. package/dist/components/Modal/index.js +23 -58
  50. package/dist/components/Notice/index.js +13 -29
  51. package/dist/components/Notification/index.js +9 -48
  52. package/dist/components/NumberInput/index.js +113 -200
  53. package/dist/components/NumberInputV2/index.js +200 -349
  54. package/dist/components/Pagination/getPageNumbers.js +3 -1
  55. package/dist/components/Pagination/index.js +45 -81
  56. package/dist/components/PasswordCheck/index.js +10 -15
  57. package/dist/components/PasswordStrengthMeter/index.js +66 -68
  58. package/dist/components/PieChart/Legends.js +121 -99
  59. package/dist/components/PieChart/Tooltip.js +35 -15
  60. package/dist/components/PieChart/index.js +68 -87
  61. package/dist/components/Popover/index.js +57 -106
  62. package/dist/components/Popup/helpers.js +24 -12
  63. package/dist/components/Popup/index.js +263 -351
  64. package/dist/components/ProgressBar/index.js +33 -51
  65. package/dist/components/Radio/index.js +119 -188
  66. package/dist/components/RadioGroup/index.js +46 -51
  67. package/dist/components/Row/index.js +11 -20
  68. package/dist/components/SelectInput/index.js +165 -212
  69. package/dist/components/SelectInputV2/Dropdown.js +233 -473
  70. package/dist/components/SelectInputV2/DropdownOption.js +44 -115
  71. package/dist/components/SelectInputV2/SearchBarDropdown.js +28 -44
  72. package/dist/components/SelectInputV2/SelectBar.js +137 -240
  73. package/dist/components/SelectInputV2/SelectInputProvider.js +45 -77
  74. package/dist/components/SelectInputV2/index.js +39 -98
  75. package/dist/components/SelectableCard/index.js +127 -183
  76. package/dist/components/SelectableCardGroup/index.js +50 -55
  77. package/dist/components/Separator/index.js +44 -64
  78. package/dist/components/Skeleton/Block.js +28 -20
  79. package/dist/components/Skeleton/Blocks.js +26 -16
  80. package/dist/components/Skeleton/BoxWithIcon.js +24 -17
  81. package/dist/components/Skeleton/Donut.js +46 -21
  82. package/dist/components/Skeleton/IconSkeleton.js +19 -9
  83. package/dist/components/Skeleton/Line.js +11 -8
  84. package/dist/components/Skeleton/List.js +48 -21
  85. package/dist/components/Skeleton/Slider.js +36 -22
  86. package/dist/components/Skeleton/Square.js +9 -7
  87. package/dist/components/Skeleton/index.js +38 -45
  88. package/dist/components/Snippet/index.js +139 -128
  89. package/dist/components/Stack/index.js +10 -17
  90. package/dist/components/Status/index.js +42 -27
  91. package/dist/components/StepList/index.js +34 -45
  92. package/dist/components/Stepper/index.js +106 -114
  93. package/dist/components/SwitchButton/FocusOverlay.js +18 -27
  94. package/dist/components/SwitchButton/index.js +39 -103
  95. package/dist/components/Table/Body.js +3 -1
  96. package/dist/components/Table/Cell.js +15 -6
  97. package/dist/components/Table/Header.js +12 -5
  98. package/dist/components/Table/HeaderCell.js +48 -91
  99. package/dist/components/Table/HeaderRow.js +11 -13
  100. package/dist/components/Table/Row.js +39 -44
  101. package/dist/components/Table/SelectBar.js +31 -14
  102. package/dist/components/Table/SkeletonRows.js +43 -11
  103. package/dist/components/Table/TableContext.js +29 -44
  104. package/dist/components/Table/index.js +37 -51
  105. package/dist/components/Tabs/Tab.js +108 -158
  106. package/dist/components/Tabs/TabMenu.js +52 -53
  107. package/dist/components/Tabs/TabMenuItem.js +19 -24
  108. package/dist/components/Tabs/TabsContext.js +1 -3
  109. package/dist/components/Tabs/index.js +44 -79
  110. package/dist/components/Tag/index.js +66 -75
  111. package/dist/components/TagInput/index.js +125 -188
  112. package/dist/components/TagList/index.js +51 -56
  113. package/dist/components/Text/index.js +10 -35
  114. package/dist/components/TextArea/index.js +140 -208
  115. package/dist/components/TextInput/index.js +326 -411
  116. package/dist/components/TextInputV2/index.js +170 -294
  117. package/dist/components/TimeInput/index.js +3 -174
  118. package/dist/components/Toaster/index.js +26 -74
  119. package/dist/components/Toggle/index.js +112 -172
  120. package/dist/components/ToggleGroup/index.js +44 -35
  121. package/dist/components/Tooltip/index.js +34 -41
  122. package/dist/components/VerificationCode/index.js +58 -63
  123. package/dist/helpers/legend.js +4 -4
  124. package/dist/theme/index.js +10 -13
  125. package/dist/utils/ids.js +3 -1
  126. package/dist/utils/responsive/Breakpoint.js +8 -6
  127. package/package.json +4 -4
@@ -1,5 +1,5 @@
1
1
  import { jsx, jsxs, Fragment } from "@emotion/react/jsx-runtime";
2
- import styled from "@emotion/styled";
2
+ import _styled from "@emotion/styled/base";
3
3
  import { useState, useRef, useEffect, useMemo } from "react";
4
4
  import { Checkbox } from "../Checkbox/index.js";
5
5
  import { Popup } from "../Popup/index.js";
@@ -9,131 +9,164 @@ import { Text } from "../Text/index.js";
9
9
  import { DisplayOption } from "./DropdownOption.js";
10
10
  import { SearchBarDropdown } from "./SearchBarDropdown.js";
11
11
  import { useSelectInput } from "./SelectInputProvider.js";
12
- const NON_SEARCHABLE_KEYS = [
13
- "Tab",
14
- " ",
15
- "Enter",
16
- "CapsLock",
17
- "Shift",
18
- "ArrowDown",
19
- "ArrowUp",
20
- "ArrowLeft",
21
- "ArrowRight",
22
- "Escape"
23
- ];
24
- const StyledPopup = styled(Popup)`
25
- width: 100%;
26
- background-color: ${({ theme }) => theme.colors.neutral.background};
27
- color: ${({ theme }) => theme.colors.neutral.text};
28
- box-shadow: ${({ theme }) => theme.shadows.dropdown};
29
- padding: ${({ theme }) => theme.space[0]};
30
- `;
31
- const DropdownContainer = styled(Stack)`
32
- max-height: 256px;
33
- overflow-y: scroll;
34
- padding: ${({ theme }) => theme.space[0]};
35
- padding-bottom: ${({ theme }) => theme.space[0.5]};
36
- padding-top: ${({ theme }) => theme.space[0.5]};
37
-
38
- &[data-grouped='true'] {
39
- padding-top: ${({ theme }) => theme.space[0]};
40
- }
41
- `;
42
- const DropdownGroup = styled.button`
43
- display: flex;
44
- width: 100%;
45
- justify-content: left;
46
- align-items: center;
47
- border: none;
48
- background-color: ${({ theme }) => theme.colors.neutral.backgroundWeak};
49
- position: sticky;
50
- top: 0px;
51
- padding-right: ${({ theme }) => theme.space[2]};
52
- padding-left: ${({ theme }) => theme.space[2]};
53
- height: ${({ theme }) => theme.space[4]};
54
- text-align: left;
55
- margin-bottom: ${({ theme }) => theme.space["0.25"]};
56
-
57
- &:focus {
58
- background-color: ${({ theme }) => theme.colors.neutral.backgroundWeak};
59
- outline: none;
60
- }
61
-
62
- &[data-selectgroup='true'] {
63
- padding-left: ${({ theme }) => theme.space[2]};
64
- border-left: ${({ theme }) => theme.space[0.5]} solid ${({ theme }) => theme.colors.neutral.backgroundWeak};
65
- }
66
-
67
- &[data-selectgroup='true']:focus {
68
- background-color: ${({ theme }) => theme.colors.neutral.backgroundHover};
69
- }
12
+ function _EMOTION_STRINGIFIED_CSS_ERROR__() {
13
+ return "You have tried to stringify object returned from `css` function. It isn't supposed to be used directly (e.g. as value of the `className` prop), but rather handed to emotion so it can handle it (e.g. as value of `css` prop).";
70
14
  }
71
- `;
72
- const DropdownGroupWrapper = styled.div`
73
- position: sticky;
74
- top: 0px;
75
- `;
76
- const DropdownItem = styled.button`
77
- text-align:left;
78
- border: none;
79
- background-color: ${({ theme }) => theme.colors.neutral.background};
80
-
81
- padding: ${({ theme }) => theme.space["1.5"]} ${({ theme }) => theme.space["2"]} ${({ theme }) => theme.space["1.5"]} ${({ theme }) => theme.space["2"]};
82
- margin-left: ${({ theme }) => theme.space["0.5"]};
83
- margin-right: ${({ theme }) => theme.space["0.5"]};
84
-
85
- color: ${({ theme }) => theme.colors.neutral.text};
86
- border-radius: ${({ theme }) => theme.radii.default};
87
-
88
- &:hover, :focus {
89
- background-color: ${({ theme }) => theme.colors.primary.background};
90
- color: ${({ theme }) => theme.colors.primary.text};
91
- cursor: pointer;
92
- outline: none;
93
- }
94
-
95
- &[data-selected='true'] {
96
- background-color: ${({ theme }) => theme.colors.primary.background};
97
- }
98
-
99
- &[disabled] {
100
- background-color: ${({ theme }) => theme.colors.neutral.backgroundDisabled};
101
- color: ${({ theme }) => theme.colors.neutral.textDisabled};
102
-
103
- }
104
-
105
- &[disabled]:hover, [disabled]:focus {
106
- background-color: ${({ theme }) => theme.colors.neutral.backgroundStrongDisabled};
107
- color: ${({ theme }) => theme.colors.neutral.textStrongDisabled};
108
- cursor: not-allowed;
109
- outline: none;
110
- }
111
-
112
- }
113
- `;
114
- const PopupFooter = styled.div`
115
- width: 100%;
116
- padding: ${({ theme }) => theme.space[1.5]} ${({ theme }) => theme.space[2]}
117
- ${({ theme }) => theme.space[1.5]} ${({ theme }) => theme.space[2]};
118
- box-shadow: ${({ theme }) => theme.shadows.dropdown};
119
- `;
120
- const StyledCheckbox = styled(Checkbox)`
121
- width: 100%;
122
- position: static;
123
- text-align: left;
124
- align-items: center;
125
- pointer-events: none;
126
- `;
127
- const EmptyState = styled(Stack)`
128
- padding: ${({ theme }) => theme.space[2]};
129
- `;
130
- const LoadMore = styled(Stack)`
131
- padding: ${({ theme }) => theme.space[0.5]};
132
- `;
15
+ const NON_SEARCHABLE_KEYS = ["Tab", " ", "Enter", "CapsLock", "Shift", "ArrowDown", "ArrowUp", "ArrowLeft", "ArrowRight", "Escape"];
16
+ const StyledPopup = /* @__PURE__ */ _styled(Popup, process.env.NODE_ENV === "production" ? {
17
+ target: "eeucz6c8"
18
+ } : {
19
+ target: "eeucz6c8",
20
+ label: "StyledPopup"
21
+ })("width:100%;background-color:", ({
22
+ theme
23
+ }) => theme.colors.neutral.background, ";color:", ({
24
+ theme
25
+ }) => theme.colors.neutral.text, ";box-shadow:", ({
26
+ theme
27
+ }) => theme.shadows.dropdown, ";padding:", ({
28
+ theme
29
+ }) => theme.space[0], ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/Dropdown.tsx"],"names":[],"mappings":"AAiDiC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/Dropdown.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { useEffect, useMemo, useRef, useState } from 'react'\nimport type { Dispatch, ReactNode, RefObject, SetStateAction } from 'react'\nimport { Checkbox } from '../Checkbox'\nimport { Popup } from '../Popup'\nimport { Skeleton } from '../Skeleton'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\nimport { DisplayOption } from './DropdownOption'\nimport { SearchBarDropdown } from './SearchBarDropdown'\nimport { useSelectInput } from './SelectInputProvider'\nimport type { DataType, OptionType } from './types'\n\nexport type DropdownProps = {\n  children: ReactNode\n  emptyState: ReactNode\n  descriptionDirection: 'row' | 'column'\n  searchable: boolean\n  placeholder: string\n  footer?: ReactNode\n  refSelect: RefObject<HTMLDivElement>\n  loadMore?: ReactNode\n  optionalInfoPlacement: 'left' | 'right'\n  isLoading?: boolean\n}\n\nexport type CreateDropdownProps = {\n  isEmpty: boolean\n  emptyState: ReactNode\n  descriptionDirection: 'row' | 'column'\n  loadMore?: ReactNode\n  optionalInfoPlacement: 'left' | 'right'\n  defaultSearchValue: string | null\n  isLoading?: boolean\n}\n\nconst NON_SEARCHABLE_KEYS = [\n  'Tab',\n  ' ',\n  'Enter',\n  'CapsLock',\n  'Shift',\n  'ArrowDown',\n  'ArrowUp',\n  'ArrowLeft',\n  'ArrowRight',\n  'Escape',\n]\n\nconst StyledPopup = styled(Popup)`\n  width: 100%;\n  background-color: ${({ theme }) => theme.colors.neutral.background};\n  color: ${({ theme }) => theme.colors.neutral.text};\n  box-shadow: ${({ theme }) => theme.shadows.dropdown};\n  padding: ${({ theme }) => theme.space[0]};\n`\n\nconst DropdownContainer = styled(Stack)<{ 'data-grouped': boolean }>`\n  max-height: 256px;\n  overflow-y: scroll;\n  padding: ${({ theme }) => theme.space[0]};\n  padding-bottom: ${({ theme }) => theme.space[0.5]};\n  padding-top: ${({ theme }) => theme.space[0.5]};\n\n  &[data-grouped='true'] {\n    padding-top: ${({ theme }) => theme.space[0]};\n  }\n`\nconst DropdownGroup = styled.button<{ 'data-selectgroup': boolean }>`\n  display: flex;\n  width: 100%;\n  justify-content: left;\n  align-items: center;\n  border: none;\n  background-color: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n  position: sticky;\n  top: 0px;\n  padding-right: ${({ theme }) => theme.space[2]};\n  padding-left: ${({ theme }) => theme.space[2]};\n  height: ${({ theme }) => theme.space[4]};\n  text-align: left;\n  margin-bottom: ${({ theme }) => theme.space['0.25']};\n\n  &:focus {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n    outline: none;\n  }\n\n  &[data-selectgroup='true'] {\n    padding-left: ${({ theme }) => theme.space[2]};\n    border-left: ${({ theme }) => theme.space[0.5]} solid ${({ theme }) => theme.colors.neutral.backgroundWeak};\n  }\n\n  &[data-selectgroup='true']:focus {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundHover};\n  }\n}\n`\nconst DropdownGroupWrapper = styled.div`\n  position: sticky;\n  top: 0px;\n`\nconst DropdownItem = styled.button<{\n  'data-selected': boolean\n  disabled: boolean\n}>`\n  text-align:left;\n  border: none;\n  background-color: ${({ theme }) => theme.colors.neutral.background};\n\n  padding: ${({ theme }) => theme.space['1.5']} ${({ theme }) => theme.space['2']} ${({ theme }) => theme.space['1.5']} ${({ theme }) => theme.space['2']};\n  margin-left: ${({ theme }) => theme.space['0.5']};\n  margin-right: ${({ theme }) => theme.space['0.5']};\n\n  color:  ${({ theme }) => theme.colors.neutral.text};\n  border-radius: ${({ theme }) => theme.radii.default};\n\n  &:hover, :focus {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n    color: ${({ theme }) => theme.colors.primary.text};\n    cursor: pointer;\n    outline: none;\n  }\n\n  &[data-selected='true'] {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n  }\n\n  &[disabled] {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n\n  }\n\n  &[disabled]:hover, [disabled]:focus {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundStrongDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textStrongDisabled};\n    cursor: not-allowed;\n    outline: none;\n  }\n\n  }\n`\nconst PopupFooter = styled.div`\n  width: 100%;\n  padding: ${({ theme }) => theme.space[1.5]} ${({ theme }) => theme.space[2]}\n    ${({ theme }) => theme.space[1.5]} ${({ theme }) => theme.space[2]};\n  box-shadow: ${({ theme }) => theme.shadows.dropdown};\n`\nconst StyledCheckbox = styled(Checkbox)`\n  width: 100%;\n  position: static;\n  text-align: left;\n  align-items: center;\n  pointer-events: none;\n` // pointer-events: none prevents any error when using the checkbox in a form since it is an unnamed input\n\nconst EmptyState = styled(Stack)`\n  padding: ${({ theme }) => theme.space[2]};\n`\nconst LoadMore = styled(Stack)`\n  padding: ${({ theme }) => theme.space[0.5]};\n`\n\nconst moveFocusDown = () => {\n  const options = document.querySelectorAll(\n    '#items > button[role=\"option\"]:not([disabled])',\n  )\n  const activeItem = document.activeElement\n  if (options) {\n    for (let i = 0; i < options?.length; i += 1) {\n      const listLength = options.length\n      if (activeItem === options[i] && activeItem !== options[listLength - 1]) {\n        ;(options[i + 1] as HTMLElement).focus()\n      }\n    }\n  }\n}\nconst moveFocusUp = () => {\n  const options = document.querySelectorAll(\n    '#items > button[role=\"option\"]:not([disabled])',\n  )\n  const activeItem = document.activeElement\n\n  if (options) {\n    for (let i = 0; i < options.length; i += 1) {\n      if (activeItem === options[i] && activeItem !== options[0]) {\n        ;(options[i - 1] as HTMLElement).focus()\n      }\n    }\n  }\n}\nconst handleKeyDownSelect = (key: string) => {\n  if (key === 'ArrowDown') {\n    moveFocusDown()\n  }\n  if (key === 'ArrowUp') {\n    moveFocusUp()\n  }\n}\nconst handleClickOutside = (\n  event: MouseEvent,\n  ref: RefObject<HTMLDivElement>,\n  setIsDropdownVisibile: Dispatch<SetStateAction<boolean>>,\n  refSelect: RefObject<HTMLDivElement>,\n  onSearch: Dispatch<SetStateAction<DataType>>,\n  options: DataType,\n) => {\n  if (\n    ref.current &&\n    !ref.current.contains(event.target as Node) &&\n    !refSelect.current?.contains(event.target as Node)\n  ) {\n    setIsDropdownVisibile(false) // hide dropdown when clicking outside of the dropdown\n    onSearch(options) // reset displayed options to default when dropdown is hidden\n  }\n}\n\nconst handleKeyDown = (\n  event: KeyboardEvent,\n  ref: RefObject<HTMLDivElement>,\n  options: DataType,\n  searchBarActive: boolean,\n  setSearch: Dispatch<SetStateAction<string>>,\n  setDefaultSearch: Dispatch<SetStateAction<string | null>>,\n  search: string,\n) => {\n  // Deals with default search\n  if (\n    ref.current &&\n    !searchBarActive &&\n    !NON_SEARCHABLE_KEYS.includes(event.key) &&\n    document.activeElement?.ariaLabel !== 'search-bar'\n  ) {\n    const currentSearch = search + event.key\n    setSearch(currentSearch)\n    ref.current.focus()\n    if (!Array.isArray(options)) {\n      const closestOptions = { ...options }\n      Object.keys(closestOptions).map((group: string) => {\n        closestOptions[group] = closestOptions[group].filter(option =>\n          option.searchText\n            ? option.searchText.toLocaleLowerCase().startsWith(currentSearch)\n            : option.value.toLocaleLowerCase().startsWith(currentSearch),\n        )\n\n        return null\n      })\n      const closestOption = closestOptions[Object.keys(closestOptions)[0]][0]\n      if (closestOption) {\n        setDefaultSearch(closestOption.searchText ?? closestOption.value)\n      } else {\n        setDefaultSearch(null)\n      }\n    } else {\n      const closestOption = [...options].filter(option =>\n        option.searchText\n          ? option.searchText.toLocaleLowerCase().startsWith(currentSearch)\n          : option.value.toLocaleLowerCase().startsWith(currentSearch),\n      )[0]\n      if (closestOption) {\n        setDefaultSearch(closestOption.searchText ?? closestOption.value)\n      } else {\n        setDefaultSearch(null)\n      }\n    }\n  }\n}\nconst CreateDropdown = ({\n  isEmpty,\n  emptyState,\n  descriptionDirection,\n  loadMore,\n  optionalInfoPlacement,\n  defaultSearchValue,\n  isLoading,\n}: CreateDropdownProps) => {\n  const {\n    setIsDropdownVisible,\n    onChange,\n    options,\n    multiselect,\n    selectAll,\n    selectAllGroup,\n    displayedOptions,\n    setSelectedData,\n    selectedData,\n  } = useSelectInput()\n  const focusedItemRef = useRef<HTMLButtonElement>(null)\n\n  useEffect(() => {\n    if (defaultSearchValue && focusedItemRef?.current) {\n      focusedItemRef.current.focus()\n    }\n  }, [defaultSearchValue])\n\n  if (isEmpty) {\n    return (\n      <EmptyState gap={2} alignItems=\"center\">\n        {emptyState ?? (\n          <Text variant=\"bodyStrong\" as=\"p\">\n            No options\n          </Text>\n        )}\n      </EmptyState>\n    )\n  }\n\n  const handleClick = (clickedOption: OptionType, group?: string) => {\n    setSelectedData({ type: 'selectOption', clickedOption, group })\n    if (multiselect) {\n      if (selectedData.selectedValues.includes(clickedOption.value)) {\n        onChange?.(\n          selectedData.selectedValues.filter(\n            val => val !== clickedOption.value,\n          ),\n        )\n      } else {\n        onChange?.([...selectedData.selectedValues, clickedOption.value])\n      }\n    } else {\n      onChange?.(clickedOption.value)\n    }\n    setIsDropdownVisible(multiselect) // hide the dropdown on click when single select only\n  }\n\n  const selectAllOptions = () => {\n    if (multiselect) {\n      setSelectedData({ type: 'selectAll' })\n      if (selectedData.allSelected && onChange) {\n        onChange([])\n      } else {\n        const allValues: OptionType[] = []\n        if (!Array.isArray(options)) {\n          Object.keys(options).map((group: string) =>\n            options[group].map(option => {\n              if (!option.disabled) {\n                allValues.push(option)\n              }\n\n              return null\n            }),\n          )\n        } else {\n          options.map(option => allValues.push(option))\n        }\n        onChange?.(allValues.map(value => value.value))\n      }\n    }\n  }\n\n  const handleSelectGroup = (group: string) => {\n    if (multiselect) {\n      setSelectedData({ type: 'selectGroup', selectedGroup: group })\n      if (!Array.isArray(options)) {\n        if (selectedData.selectedGroups.includes(group)) {\n          const newSelectedValues = [...selectedData.selectedValues].filter(\n            selectedValue =>\n              !options[group].find(option => option.value === selectedValue),\n          )\n          onChange?.(newSelectedValues)\n        } else {\n          const newSelectedValues = [...selectedData.selectedValues]\n\n          options[group].map(option =>\n            newSelectedValues.includes(option.value) || option.disabled\n              ? null\n              : newSelectedValues.push(option.value),\n          )\n          onChange?.(newSelectedValues)\n        }\n      }\n    }\n  }\n\n  return !Array.isArray(displayedOptions) ? (\n    <DropdownContainer\n      role=\"listbox\"\n      id=\"select-dropdown\"\n      onKeyDown={event => {\n        event.preventDefault()\n        handleKeyDownSelect(event.key)\n      }}\n      data-grouped\n    >\n      {isLoading ? (\n        <Skeleton variant=\"block\" />\n      ) : (\n        <>\n          {selectAll && multiselect ? (\n            <Stack id=\"items\">\n              <DropdownItem\n                disabled={false}\n                data-selected={selectedData.allSelected}\n                aria-label=\"select-all\"\n                data-testid=\"select-all\"\n                id=\"select-all\"\n                role=\"option\"\n                onKeyDown={event =>\n                  [' ', 'Enter'].includes(event.key) ? selectAllOptions() : null\n                }\n                onClick={selectAllOptions}\n              >\n                <StyledCheckbox\n                  checked={selectedData.allSelected}\n                  disabled={false}\n                  value=\"select-all\"\n                  data-testid=\"select-all-checkbox\"\n                  tabIndex={-1}\n                >\n                  <Stack direction=\"column\">\n                    <Text as=\"span\" variant=\"body\" placement=\"left\">\n                      {selectAll.label}\n                    </Text>\n                    <Text\n                      as=\"span\"\n                      variant=\"bodySmall\"\n                      sentiment=\"neutral\"\n                      placement=\"left\"\n                      prominence=\"weak\"\n                    >\n                      {selectAll.description}\n                    </Text>\n                  </Stack>\n                </StyledCheckbox>\n              </DropdownItem>\n            </Stack>\n          ) : null}\n          {Object.keys(displayedOptions).map((group, index) => (\n            <Stack key={group} gap={0.25}>\n              {displayedOptions[group].length > 0 ? (\n                <DropdownGroupWrapper id={selectAllGroup ? 'items' : undefined}>\n                  <DropdownGroup\n                    key={group}\n                    tabIndex={selectAllGroup ? 0 : -1}\n                    onKeyDown={event => {\n                      if ([' ', 'Enter'].includes(event.key)) {\n                        event.preventDefault()\n                        handleSelectGroup(group)\n                      }\n                    }}\n                    data-selectgroup={selectAllGroup}\n                    role=\"group\"\n                    data-testid={`group-${index}`}\n                    onClick={() =>\n                      selectAllGroup ? handleSelectGroup(group) : null\n                    }\n                  >\n                    {selectAllGroup ? (\n                      <StyledCheckbox\n                        checked={selectedData.selectedGroups.includes(group)}\n                        disabled={false}\n                        value={group}\n                        data-testid=\"select-group\"\n                        tabIndex={-1}\n                      >\n                        <Text variant=\"caption\" as=\"span\" placement=\"left\">\n                          {group.toUpperCase()}\n                        </Text>\n                      </StyledCheckbox>\n                    ) : (\n                      <Text variant=\"caption\" as=\"span\" placement=\"left\">\n                        {group.toUpperCase()}\n                      </Text>\n                    )}\n                  </DropdownGroup>\n                </DropdownGroupWrapper>\n              ) : null}\n              <Stack id=\"items\" gap=\"0.25\">\n                {displayedOptions[group].map((option, indexOption) => (\n                  <DropdownItem\n                    key={option.value}\n                    disabled={option.disabled}\n                    data-selected={\n                      selectedData.selectedValues.includes(option.value) &&\n                      !option.disabled\n                    }\n                    aria-label={option.value}\n                    data-testid={`option-${option.value}`}\n                    id={`option-${indexOption}`}\n                    role=\"option\"\n                    onClick={() => {\n                      if (!option.disabled) {\n                        handleClick(option, group)\n                      }\n                    }}\n                    onKeyDown={event =>\n                      [' ', 'Enter'].includes(event.key)\n                        ? handleClick(option, group)\n                        : null\n                    }\n                    ref={\n                      option.value === defaultSearchValue ||\n                      option.searchText === defaultSearchValue\n                        ? focusedItemRef\n                        : null\n                    }\n                  >\n                    {multiselect ? (\n                      <StyledCheckbox\n                        checked={\n                          selectedData.selectedValues.includes(option.value) &&\n                          !option.disabled\n                        }\n                        disabled={option.disabled}\n                        value={option.value}\n                        tabIndex={-1}\n                      >\n                        <DisplayOption\n                          option={option}\n                          descriptionDirection={descriptionDirection}\n                          optionalInfoPlacement={optionalInfoPlacement}\n                        />\n                      </StyledCheckbox>\n                    ) : (\n                      <DisplayOption\n                        option={option}\n                        descriptionDirection={descriptionDirection}\n                        optionalInfoPlacement={optionalInfoPlacement}\n                      />\n                    )}\n                  </DropdownItem>\n                ))}\n                {loadMore ? <LoadMore>{loadMore}</LoadMore> : null}\n              </Stack>\n            </Stack>\n          ))}\n        </>\n      )}\n    </DropdownContainer>\n  ) : (\n    <DropdownContainer\n      role=\"listbox\"\n      id=\"select-dropdown\"\n      onKeyDown={event => {\n        event.preventDefault()\n        handleKeyDownSelect(event.key)\n      }}\n      gap={0.25}\n      data-grouped={false}\n    >\n      {selectAll && multiselect ? (\n        <Stack id=\"items\" gap={0.25}>\n          <DropdownItem\n            disabled={false}\n            data-selected={selectedData.allSelected}\n            aria-label=\"select-all\"\n            data-testid=\"select-all\"\n            role=\"option\"\n            onKeyDown={event =>\n              [' ', 'Enter'].includes(event.key) ? selectAllOptions() : null\n            }\n            onClick={selectAllOptions}\n          >\n            <StyledCheckbox\n              checked={selectedData.allSelected}\n              disabled={false}\n              value=\"select-all\"\n              data-testid=\"select-all-checkbox\"\n              tabIndex={-1}\n            >\n              <Stack direction=\"column\">\n                <Text as=\"span\" variant=\"body\" placement=\"left\">\n                  {selectAll.label}\n                </Text>\n                <Text\n                  as=\"span\"\n                  variant=\"bodySmall\"\n                  sentiment=\"neutral\"\n                  placement=\"left\"\n                  prominence=\"weak\"\n                >\n                  {selectAll.description}\n                </Text>\n              </Stack>\n            </StyledCheckbox>\n          </DropdownItem>\n        </Stack>\n      ) : null}\n      <Stack id=\"items\" gap={0.25}>\n        {isLoading ? (\n          <Skeleton variant=\"block\" />\n        ) : (\n          displayedOptions.map((option, index) => (\n            <DropdownItem\n              key={option.value}\n              disabled={option.disabled}\n              data-selected={\n                selectedData.selectedValues.includes(option.value) &&\n                !option.disabled\n              }\n              onClick={() => {\n                if (!option.disabled) {\n                  handleClick(option)\n                }\n              }}\n              aria-label={option.value}\n              data-testid={`option-${option.value}`}\n              id={`option-${index}`}\n              role=\"option\"\n              ref={\n                option.value === defaultSearchValue ||\n                option.searchText === defaultSearchValue\n                  ? focusedItemRef\n                  : null\n              }\n              onKeyDown={event =>\n                [' ', 'Enter'].includes(event.key) ? handleClick(option) : null\n              }\n            >\n              {multiselect ? (\n                <StyledCheckbox\n                  checked={\n                    selectedData.selectedValues.includes(option.value) &&\n                    !option.disabled\n                  }\n                  disabled={option.disabled}\n                  value={option.value}\n                  tabIndex={-1}\n                >\n                  <DisplayOption\n                    option={option}\n                    descriptionDirection={descriptionDirection}\n                    optionalInfoPlacement={optionalInfoPlacement}\n                  />\n                </StyledCheckbox>\n              ) : (\n                <DisplayOption\n                  option={option}\n                  descriptionDirection={descriptionDirection}\n                  optionalInfoPlacement={optionalInfoPlacement}\n                />\n              )}\n            </DropdownItem>\n          ))\n        )}\n        {loadMore ? <LoadMore>{loadMore}</LoadMore> : null}\n      </Stack>\n    </DropdownContainer>\n  )\n}\nexport const Dropdown = ({\n  children,\n  emptyState,\n  descriptionDirection,\n  searchable,\n  placeholder,\n  footer,\n  refSelect,\n  loadMore,\n  optionalInfoPlacement,\n  isLoading,\n}: DropdownProps) => {\n  const {\n    setIsDropdownVisible,\n    isDropdownVisible,\n    onSearch,\n    searchInput,\n    options,\n    displayedOptions,\n  } = useSelectInput()\n  const [searchBarActive, setSearchBarActive] = useState(false)\n  const [defaultSearchValue, setDefaultSearch] = useState<string | null>(null)\n  const maxWidth = refSelect.current?.offsetWidth\n  const ref = useRef<HTMLDivElement>(null)\n  const [search, setSearch] = useState<string>('')\n  useEffect(() => {\n    if (!searchInput) {\n      onSearch(options)\n    }\n  }, [onSearch, options, searchInput])\n\n  useEffect(() => {\n    if (!isDropdownVisible) {\n      setDefaultSearch(null)\n      setSearch('')\n    }\n\n    document.addEventListener('mousedown', event =>\n      handleClickOutside(\n        event,\n        ref,\n        setIsDropdownVisible,\n        refSelect,\n        onSearch,\n        options,\n      ),\n    )\n\n    if (!searchable) {\n      document.addEventListener('keydown', event =>\n        handleKeyDown(\n          event,\n          ref,\n          options,\n          searchBarActive,\n          setSearch,\n          setDefaultSearch,\n          search,\n        ),\n      )\n    }\n\n    return () => {\n      document.removeEventListener('mousedown', event =>\n        handleClickOutside(\n          event,\n          ref,\n          setIsDropdownVisible,\n          refSelect,\n          onSearch,\n          options,\n        ),\n      )\n      if (!searchable) {\n        document.removeEventListener('keydown', event =>\n          handleKeyDown(\n            event,\n            ref,\n            options,\n            searchBarActive,\n            setSearch,\n            setDefaultSearch,\n            search,\n          ),\n        )\n      }\n    }\n  }, [\n    isDropdownVisible,\n    searchBarActive,\n    options,\n    onSearch,\n    search,\n    refSelect,\n    setDefaultSearch,\n    setIsDropdownVisible,\n    searchable,\n  ])\n\n  const isEmpty = useMemo(() => {\n    if (Array.isArray(displayedOptions)) {\n      return !(displayedOptions.length > 0)\n    }\n    const groups = Object.keys(displayedOptions)\n    for (const group of groups) {\n      if (displayedOptions[group].length !== 0) {\n        return false\n      }\n    }\n\n    return true\n  }, [displayedOptions])\n\n  return (\n    <StyledPopup\n      visible={isDropdownVisible}\n      text={\n        <Stack>\n          {searchable && !isLoading ? (\n            <SearchBarDropdown\n              placeholder={placeholder}\n              displayedOptions={displayedOptions}\n              setSearchBarActive={setSearchBarActive}\n            />\n          ) : null}\n          <CreateDropdown\n            isEmpty={isEmpty}\n            emptyState={emptyState}\n            descriptionDirection={descriptionDirection}\n            loadMore={loadMore}\n            optionalInfoPlacement={optionalInfoPlacement}\n            defaultSearchValue={defaultSearchValue}\n            isLoading={isLoading}\n          />\n          {footer ? <PopupFooter>{footer}</PopupFooter> : null}\n        </Stack>\n      }\n      placement=\"bottom\"\n      containerFullWidth\n      disableAnimation\n      maxWidth={maxWidth}\n      hasArrow={false}\n      ref={ref}\n      tabIndex={0}\n      role=\"dialog\"\n      debounceDelay={0}\n    >\n      {children}\n    </StyledPopup>\n  )\n}\n"]} */"));
30
+ const DropdownContainer = /* @__PURE__ */ _styled(Stack, process.env.NODE_ENV === "production" ? {
31
+ target: "eeucz6c7"
32
+ } : {
33
+ target: "eeucz6c7",
34
+ label: "DropdownContainer"
35
+ })("max-height:256px;overflow-y:scroll;padding:", ({
36
+ theme
37
+ }) => theme.space[0], ";padding-bottom:", ({
38
+ theme
39
+ }) => theme.space[0.5], ";padding-top:", ({
40
+ theme
41
+ }) => theme.space[0.5], ";&[data-grouped='true']{padding-top:", ({
42
+ theme
43
+ }) => theme.space[0], ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/Dropdown.tsx"],"names":[],"mappings":"AAyDoE","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/Dropdown.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { useEffect, useMemo, useRef, useState } from 'react'\nimport type { Dispatch, ReactNode, RefObject, SetStateAction } from 'react'\nimport { Checkbox } from '../Checkbox'\nimport { Popup } from '../Popup'\nimport { Skeleton } from '../Skeleton'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\nimport { DisplayOption } from './DropdownOption'\nimport { SearchBarDropdown } from './SearchBarDropdown'\nimport { useSelectInput } from './SelectInputProvider'\nimport type { DataType, OptionType } from './types'\n\nexport type DropdownProps = {\n  children: ReactNode\n  emptyState: ReactNode\n  descriptionDirection: 'row' | 'column'\n  searchable: boolean\n  placeholder: string\n  footer?: ReactNode\n  refSelect: RefObject<HTMLDivElement>\n  loadMore?: ReactNode\n  optionalInfoPlacement: 'left' | 'right'\n  isLoading?: boolean\n}\n\nexport type CreateDropdownProps = {\n  isEmpty: boolean\n  emptyState: ReactNode\n  descriptionDirection: 'row' | 'column'\n  loadMore?: ReactNode\n  optionalInfoPlacement: 'left' | 'right'\n  defaultSearchValue: string | null\n  isLoading?: boolean\n}\n\nconst NON_SEARCHABLE_KEYS = [\n  'Tab',\n  ' ',\n  'Enter',\n  'CapsLock',\n  'Shift',\n  'ArrowDown',\n  'ArrowUp',\n  'ArrowLeft',\n  'ArrowRight',\n  'Escape',\n]\n\nconst StyledPopup = styled(Popup)`\n  width: 100%;\n  background-color: ${({ theme }) => theme.colors.neutral.background};\n  color: ${({ theme }) => theme.colors.neutral.text};\n  box-shadow: ${({ theme }) => theme.shadows.dropdown};\n  padding: ${({ theme }) => theme.space[0]};\n`\n\nconst DropdownContainer = styled(Stack)<{ 'data-grouped': boolean }>`\n  max-height: 256px;\n  overflow-y: scroll;\n  padding: ${({ theme }) => theme.space[0]};\n  padding-bottom: ${({ theme }) => theme.space[0.5]};\n  padding-top: ${({ theme }) => theme.space[0.5]};\n\n  &[data-grouped='true'] {\n    padding-top: ${({ theme }) => theme.space[0]};\n  }\n`\nconst DropdownGroup = styled.button<{ 'data-selectgroup': boolean }>`\n  display: flex;\n  width: 100%;\n  justify-content: left;\n  align-items: center;\n  border: none;\n  background-color: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n  position: sticky;\n  top: 0px;\n  padding-right: ${({ theme }) => theme.space[2]};\n  padding-left: ${({ theme }) => theme.space[2]};\n  height: ${({ theme }) => theme.space[4]};\n  text-align: left;\n  margin-bottom: ${({ theme }) => theme.space['0.25']};\n\n  &:focus {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n    outline: none;\n  }\n\n  &[data-selectgroup='true'] {\n    padding-left: ${({ theme }) => theme.space[2]};\n    border-left: ${({ theme }) => theme.space[0.5]} solid ${({ theme }) => theme.colors.neutral.backgroundWeak};\n  }\n\n  &[data-selectgroup='true']:focus {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundHover};\n  }\n}\n`\nconst DropdownGroupWrapper = styled.div`\n  position: sticky;\n  top: 0px;\n`\nconst DropdownItem = styled.button<{\n  'data-selected': boolean\n  disabled: boolean\n}>`\n  text-align:left;\n  border: none;\n  background-color: ${({ theme }) => theme.colors.neutral.background};\n\n  padding: ${({ theme }) => theme.space['1.5']} ${({ theme }) => theme.space['2']} ${({ theme }) => theme.space['1.5']} ${({ theme }) => theme.space['2']};\n  margin-left: ${({ theme }) => theme.space['0.5']};\n  margin-right: ${({ theme }) => theme.space['0.5']};\n\n  color:  ${({ theme }) => theme.colors.neutral.text};\n  border-radius: ${({ theme }) => theme.radii.default};\n\n  &:hover, :focus {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n    color: ${({ theme }) => theme.colors.primary.text};\n    cursor: pointer;\n    outline: none;\n  }\n\n  &[data-selected='true'] {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n  }\n\n  &[disabled] {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n\n  }\n\n  &[disabled]:hover, [disabled]:focus {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundStrongDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textStrongDisabled};\n    cursor: not-allowed;\n    outline: none;\n  }\n\n  }\n`\nconst PopupFooter = styled.div`\n  width: 100%;\n  padding: ${({ theme }) => theme.space[1.5]} ${({ theme }) => theme.space[2]}\n    ${({ theme }) => theme.space[1.5]} ${({ theme }) => theme.space[2]};\n  box-shadow: ${({ theme }) => theme.shadows.dropdown};\n`\nconst StyledCheckbox = styled(Checkbox)`\n  width: 100%;\n  position: static;\n  text-align: left;\n  align-items: center;\n  pointer-events: none;\n` // pointer-events: none prevents any error when using the checkbox in a form since it is an unnamed input\n\nconst EmptyState = styled(Stack)`\n  padding: ${({ theme }) => theme.space[2]};\n`\nconst LoadMore = styled(Stack)`\n  padding: ${({ theme }) => theme.space[0.5]};\n`\n\nconst moveFocusDown = () => {\n  const options = document.querySelectorAll(\n    '#items > button[role=\"option\"]:not([disabled])',\n  )\n  const activeItem = document.activeElement\n  if (options) {\n    for (let i = 0; i < options?.length; i += 1) {\n      const listLength = options.length\n      if (activeItem === options[i] && activeItem !== options[listLength - 1]) {\n        ;(options[i + 1] as HTMLElement).focus()\n      }\n    }\n  }\n}\nconst moveFocusUp = () => {\n  const options = document.querySelectorAll(\n    '#items > button[role=\"option\"]:not([disabled])',\n  )\n  const activeItem = document.activeElement\n\n  if (options) {\n    for (let i = 0; i < options.length; i += 1) {\n      if (activeItem === options[i] && activeItem !== options[0]) {\n        ;(options[i - 1] as HTMLElement).focus()\n      }\n    }\n  }\n}\nconst handleKeyDownSelect = (key: string) => {\n  if (key === 'ArrowDown') {\n    moveFocusDown()\n  }\n  if (key === 'ArrowUp') {\n    moveFocusUp()\n  }\n}\nconst handleClickOutside = (\n  event: MouseEvent,\n  ref: RefObject<HTMLDivElement>,\n  setIsDropdownVisibile: Dispatch<SetStateAction<boolean>>,\n  refSelect: RefObject<HTMLDivElement>,\n  onSearch: Dispatch<SetStateAction<DataType>>,\n  options: DataType,\n) => {\n  if (\n    ref.current &&\n    !ref.current.contains(event.target as Node) &&\n    !refSelect.current?.contains(event.target as Node)\n  ) {\n    setIsDropdownVisibile(false) // hide dropdown when clicking outside of the dropdown\n    onSearch(options) // reset displayed options to default when dropdown is hidden\n  }\n}\n\nconst handleKeyDown = (\n  event: KeyboardEvent,\n  ref: RefObject<HTMLDivElement>,\n  options: DataType,\n  searchBarActive: boolean,\n  setSearch: Dispatch<SetStateAction<string>>,\n  setDefaultSearch: Dispatch<SetStateAction<string | null>>,\n  search: string,\n) => {\n  // Deals with default search\n  if (\n    ref.current &&\n    !searchBarActive &&\n    !NON_SEARCHABLE_KEYS.includes(event.key) &&\n    document.activeElement?.ariaLabel !== 'search-bar'\n  ) {\n    const currentSearch = search + event.key\n    setSearch(currentSearch)\n    ref.current.focus()\n    if (!Array.isArray(options)) {\n      const closestOptions = { ...options }\n      Object.keys(closestOptions).map((group: string) => {\n        closestOptions[group] = closestOptions[group].filter(option =>\n          option.searchText\n            ? option.searchText.toLocaleLowerCase().startsWith(currentSearch)\n            : option.value.toLocaleLowerCase().startsWith(currentSearch),\n        )\n\n        return null\n      })\n      const closestOption = closestOptions[Object.keys(closestOptions)[0]][0]\n      if (closestOption) {\n        setDefaultSearch(closestOption.searchText ?? closestOption.value)\n      } else {\n        setDefaultSearch(null)\n      }\n    } else {\n      const closestOption = [...options].filter(option =>\n        option.searchText\n          ? option.searchText.toLocaleLowerCase().startsWith(currentSearch)\n          : option.value.toLocaleLowerCase().startsWith(currentSearch),\n      )[0]\n      if (closestOption) {\n        setDefaultSearch(closestOption.searchText ?? closestOption.value)\n      } else {\n        setDefaultSearch(null)\n      }\n    }\n  }\n}\nconst CreateDropdown = ({\n  isEmpty,\n  emptyState,\n  descriptionDirection,\n  loadMore,\n  optionalInfoPlacement,\n  defaultSearchValue,\n  isLoading,\n}: CreateDropdownProps) => {\n  const {\n    setIsDropdownVisible,\n    onChange,\n    options,\n    multiselect,\n    selectAll,\n    selectAllGroup,\n    displayedOptions,\n    setSelectedData,\n    selectedData,\n  } = useSelectInput()\n  const focusedItemRef = useRef<HTMLButtonElement>(null)\n\n  useEffect(() => {\n    if (defaultSearchValue && focusedItemRef?.current) {\n      focusedItemRef.current.focus()\n    }\n  }, [defaultSearchValue])\n\n  if (isEmpty) {\n    return (\n      <EmptyState gap={2} alignItems=\"center\">\n        {emptyState ?? (\n          <Text variant=\"bodyStrong\" as=\"p\">\n            No options\n          </Text>\n        )}\n      </EmptyState>\n    )\n  }\n\n  const handleClick = (clickedOption: OptionType, group?: string) => {\n    setSelectedData({ type: 'selectOption', clickedOption, group })\n    if (multiselect) {\n      if (selectedData.selectedValues.includes(clickedOption.value)) {\n        onChange?.(\n          selectedData.selectedValues.filter(\n            val => val !== clickedOption.value,\n          ),\n        )\n      } else {\n        onChange?.([...selectedData.selectedValues, clickedOption.value])\n      }\n    } else {\n      onChange?.(clickedOption.value)\n    }\n    setIsDropdownVisible(multiselect) // hide the dropdown on click when single select only\n  }\n\n  const selectAllOptions = () => {\n    if (multiselect) {\n      setSelectedData({ type: 'selectAll' })\n      if (selectedData.allSelected && onChange) {\n        onChange([])\n      } else {\n        const allValues: OptionType[] = []\n        if (!Array.isArray(options)) {\n          Object.keys(options).map((group: string) =>\n            options[group].map(option => {\n              if (!option.disabled) {\n                allValues.push(option)\n              }\n\n              return null\n            }),\n          )\n        } else {\n          options.map(option => allValues.push(option))\n        }\n        onChange?.(allValues.map(value => value.value))\n      }\n    }\n  }\n\n  const handleSelectGroup = (group: string) => {\n    if (multiselect) {\n      setSelectedData({ type: 'selectGroup', selectedGroup: group })\n      if (!Array.isArray(options)) {\n        if (selectedData.selectedGroups.includes(group)) {\n          const newSelectedValues = [...selectedData.selectedValues].filter(\n            selectedValue =>\n              !options[group].find(option => option.value === selectedValue),\n          )\n          onChange?.(newSelectedValues)\n        } else {\n          const newSelectedValues = [...selectedData.selectedValues]\n\n          options[group].map(option =>\n            newSelectedValues.includes(option.value) || option.disabled\n              ? null\n              : newSelectedValues.push(option.value),\n          )\n          onChange?.(newSelectedValues)\n        }\n      }\n    }\n  }\n\n  return !Array.isArray(displayedOptions) ? (\n    <DropdownContainer\n      role=\"listbox\"\n      id=\"select-dropdown\"\n      onKeyDown={event => {\n        event.preventDefault()\n        handleKeyDownSelect(event.key)\n      }}\n      data-grouped\n    >\n      {isLoading ? (\n        <Skeleton variant=\"block\" />\n      ) : (\n        <>\n          {selectAll && multiselect ? (\n            <Stack id=\"items\">\n              <DropdownItem\n                disabled={false}\n                data-selected={selectedData.allSelected}\n                aria-label=\"select-all\"\n                data-testid=\"select-all\"\n                id=\"select-all\"\n                role=\"option\"\n                onKeyDown={event =>\n                  [' ', 'Enter'].includes(event.key) ? selectAllOptions() : null\n                }\n                onClick={selectAllOptions}\n              >\n                <StyledCheckbox\n                  checked={selectedData.allSelected}\n                  disabled={false}\n                  value=\"select-all\"\n                  data-testid=\"select-all-checkbox\"\n                  tabIndex={-1}\n                >\n                  <Stack direction=\"column\">\n                    <Text as=\"span\" variant=\"body\" placement=\"left\">\n                      {selectAll.label}\n                    </Text>\n                    <Text\n                      as=\"span\"\n                      variant=\"bodySmall\"\n                      sentiment=\"neutral\"\n                      placement=\"left\"\n                      prominence=\"weak\"\n                    >\n                      {selectAll.description}\n                    </Text>\n                  </Stack>\n                </StyledCheckbox>\n              </DropdownItem>\n            </Stack>\n          ) : null}\n          {Object.keys(displayedOptions).map((group, index) => (\n            <Stack key={group} gap={0.25}>\n              {displayedOptions[group].length > 0 ? (\n                <DropdownGroupWrapper id={selectAllGroup ? 'items' : undefined}>\n                  <DropdownGroup\n                    key={group}\n                    tabIndex={selectAllGroup ? 0 : -1}\n                    onKeyDown={event => {\n                      if ([' ', 'Enter'].includes(event.key)) {\n                        event.preventDefault()\n                        handleSelectGroup(group)\n                      }\n                    }}\n                    data-selectgroup={selectAllGroup}\n                    role=\"group\"\n                    data-testid={`group-${index}`}\n                    onClick={() =>\n                      selectAllGroup ? handleSelectGroup(group) : null\n                    }\n                  >\n                    {selectAllGroup ? (\n                      <StyledCheckbox\n                        checked={selectedData.selectedGroups.includes(group)}\n                        disabled={false}\n                        value={group}\n                        data-testid=\"select-group\"\n                        tabIndex={-1}\n                      >\n                        <Text variant=\"caption\" as=\"span\" placement=\"left\">\n                          {group.toUpperCase()}\n                        </Text>\n                      </StyledCheckbox>\n                    ) : (\n                      <Text variant=\"caption\" as=\"span\" placement=\"left\">\n                        {group.toUpperCase()}\n                      </Text>\n                    )}\n                  </DropdownGroup>\n                </DropdownGroupWrapper>\n              ) : null}\n              <Stack id=\"items\" gap=\"0.25\">\n                {displayedOptions[group].map((option, indexOption) => (\n                  <DropdownItem\n                    key={option.value}\n                    disabled={option.disabled}\n                    data-selected={\n                      selectedData.selectedValues.includes(option.value) &&\n                      !option.disabled\n                    }\n                    aria-label={option.value}\n                    data-testid={`option-${option.value}`}\n                    id={`option-${indexOption}`}\n                    role=\"option\"\n                    onClick={() => {\n                      if (!option.disabled) {\n                        handleClick(option, group)\n                      }\n                    }}\n                    onKeyDown={event =>\n                      [' ', 'Enter'].includes(event.key)\n                        ? handleClick(option, group)\n                        : null\n                    }\n                    ref={\n                      option.value === defaultSearchValue ||\n                      option.searchText === defaultSearchValue\n                        ? focusedItemRef\n                        : null\n                    }\n                  >\n                    {multiselect ? (\n                      <StyledCheckbox\n                        checked={\n                          selectedData.selectedValues.includes(option.value) &&\n                          !option.disabled\n                        }\n                        disabled={option.disabled}\n                        value={option.value}\n                        tabIndex={-1}\n                      >\n                        <DisplayOption\n                          option={option}\n                          descriptionDirection={descriptionDirection}\n                          optionalInfoPlacement={optionalInfoPlacement}\n                        />\n                      </StyledCheckbox>\n                    ) : (\n                      <DisplayOption\n                        option={option}\n                        descriptionDirection={descriptionDirection}\n                        optionalInfoPlacement={optionalInfoPlacement}\n                      />\n                    )}\n                  </DropdownItem>\n                ))}\n                {loadMore ? <LoadMore>{loadMore}</LoadMore> : null}\n              </Stack>\n            </Stack>\n          ))}\n        </>\n      )}\n    </DropdownContainer>\n  ) : (\n    <DropdownContainer\n      role=\"listbox\"\n      id=\"select-dropdown\"\n      onKeyDown={event => {\n        event.preventDefault()\n        handleKeyDownSelect(event.key)\n      }}\n      gap={0.25}\n      data-grouped={false}\n    >\n      {selectAll && multiselect ? (\n        <Stack id=\"items\" gap={0.25}>\n          <DropdownItem\n            disabled={false}\n            data-selected={selectedData.allSelected}\n            aria-label=\"select-all\"\n            data-testid=\"select-all\"\n            role=\"option\"\n            onKeyDown={event =>\n              [' ', 'Enter'].includes(event.key) ? selectAllOptions() : null\n            }\n            onClick={selectAllOptions}\n          >\n            <StyledCheckbox\n              checked={selectedData.allSelected}\n              disabled={false}\n              value=\"select-all\"\n              data-testid=\"select-all-checkbox\"\n              tabIndex={-1}\n            >\n              <Stack direction=\"column\">\n                <Text as=\"span\" variant=\"body\" placement=\"left\">\n                  {selectAll.label}\n                </Text>\n                <Text\n                  as=\"span\"\n                  variant=\"bodySmall\"\n                  sentiment=\"neutral\"\n                  placement=\"left\"\n                  prominence=\"weak\"\n                >\n                  {selectAll.description}\n                </Text>\n              </Stack>\n            </StyledCheckbox>\n          </DropdownItem>\n        </Stack>\n      ) : null}\n      <Stack id=\"items\" gap={0.25}>\n        {isLoading ? (\n          <Skeleton variant=\"block\" />\n        ) : (\n          displayedOptions.map((option, index) => (\n            <DropdownItem\n              key={option.value}\n              disabled={option.disabled}\n              data-selected={\n                selectedData.selectedValues.includes(option.value) &&\n                !option.disabled\n              }\n              onClick={() => {\n                if (!option.disabled) {\n                  handleClick(option)\n                }\n              }}\n              aria-label={option.value}\n              data-testid={`option-${option.value}`}\n              id={`option-${index}`}\n              role=\"option\"\n              ref={\n                option.value === defaultSearchValue ||\n                option.searchText === defaultSearchValue\n                  ? focusedItemRef\n                  : null\n              }\n              onKeyDown={event =>\n                [' ', 'Enter'].includes(event.key) ? handleClick(option) : null\n              }\n            >\n              {multiselect ? (\n                <StyledCheckbox\n                  checked={\n                    selectedData.selectedValues.includes(option.value) &&\n                    !option.disabled\n                  }\n                  disabled={option.disabled}\n                  value={option.value}\n                  tabIndex={-1}\n                >\n                  <DisplayOption\n                    option={option}\n                    descriptionDirection={descriptionDirection}\n                    optionalInfoPlacement={optionalInfoPlacement}\n                  />\n                </StyledCheckbox>\n              ) : (\n                <DisplayOption\n                  option={option}\n                  descriptionDirection={descriptionDirection}\n                  optionalInfoPlacement={optionalInfoPlacement}\n                />\n              )}\n            </DropdownItem>\n          ))\n        )}\n        {loadMore ? <LoadMore>{loadMore}</LoadMore> : null}\n      </Stack>\n    </DropdownContainer>\n  )\n}\nexport const Dropdown = ({\n  children,\n  emptyState,\n  descriptionDirection,\n  searchable,\n  placeholder,\n  footer,\n  refSelect,\n  loadMore,\n  optionalInfoPlacement,\n  isLoading,\n}: DropdownProps) => {\n  const {\n    setIsDropdownVisible,\n    isDropdownVisible,\n    onSearch,\n    searchInput,\n    options,\n    displayedOptions,\n  } = useSelectInput()\n  const [searchBarActive, setSearchBarActive] = useState(false)\n  const [defaultSearchValue, setDefaultSearch] = useState<string | null>(null)\n  const maxWidth = refSelect.current?.offsetWidth\n  const ref = useRef<HTMLDivElement>(null)\n  const [search, setSearch] = useState<string>('')\n  useEffect(() => {\n    if (!searchInput) {\n      onSearch(options)\n    }\n  }, [onSearch, options, searchInput])\n\n  useEffect(() => {\n    if (!isDropdownVisible) {\n      setDefaultSearch(null)\n      setSearch('')\n    }\n\n    document.addEventListener('mousedown', event =>\n      handleClickOutside(\n        event,\n        ref,\n        setIsDropdownVisible,\n        refSelect,\n        onSearch,\n        options,\n      ),\n    )\n\n    if (!searchable) {\n      document.addEventListener('keydown', event =>\n        handleKeyDown(\n          event,\n          ref,\n          options,\n          searchBarActive,\n          setSearch,\n          setDefaultSearch,\n          search,\n        ),\n      )\n    }\n\n    return () => {\n      document.removeEventListener('mousedown', event =>\n        handleClickOutside(\n          event,\n          ref,\n          setIsDropdownVisible,\n          refSelect,\n          onSearch,\n          options,\n        ),\n      )\n      if (!searchable) {\n        document.removeEventListener('keydown', event =>\n          handleKeyDown(\n            event,\n            ref,\n            options,\n            searchBarActive,\n            setSearch,\n            setDefaultSearch,\n            search,\n          ),\n        )\n      }\n    }\n  }, [\n    isDropdownVisible,\n    searchBarActive,\n    options,\n    onSearch,\n    search,\n    refSelect,\n    setDefaultSearch,\n    setIsDropdownVisible,\n    searchable,\n  ])\n\n  const isEmpty = useMemo(() => {\n    if (Array.isArray(displayedOptions)) {\n      return !(displayedOptions.length > 0)\n    }\n    const groups = Object.keys(displayedOptions)\n    for (const group of groups) {\n      if (displayedOptions[group].length !== 0) {\n        return false\n      }\n    }\n\n    return true\n  }, [displayedOptions])\n\n  return (\n    <StyledPopup\n      visible={isDropdownVisible}\n      text={\n        <Stack>\n          {searchable && !isLoading ? (\n            <SearchBarDropdown\n              placeholder={placeholder}\n              displayedOptions={displayedOptions}\n              setSearchBarActive={setSearchBarActive}\n            />\n          ) : null}\n          <CreateDropdown\n            isEmpty={isEmpty}\n            emptyState={emptyState}\n            descriptionDirection={descriptionDirection}\n            loadMore={loadMore}\n            optionalInfoPlacement={optionalInfoPlacement}\n            defaultSearchValue={defaultSearchValue}\n            isLoading={isLoading}\n          />\n          {footer ? <PopupFooter>{footer}</PopupFooter> : null}\n        </Stack>\n      }\n      placement=\"bottom\"\n      containerFullWidth\n      disableAnimation\n      maxWidth={maxWidth}\n      hasArrow={false}\n      ref={ref}\n      tabIndex={0}\n      role=\"dialog\"\n      debounceDelay={0}\n    >\n      {children}\n    </StyledPopup>\n  )\n}\n"]} */"));
44
+ const DropdownGroup = /* @__PURE__ */ _styled("button", process.env.NODE_ENV === "production" ? {
45
+ target: "eeucz6c6"
46
+ } : {
47
+ target: "eeucz6c6",
48
+ label: "DropdownGroup"
49
+ })("display:flex;width:100%;justify-content:left;align-items:center;border:none;background-color:", ({
50
+ theme
51
+ }) => theme.colors.neutral.backgroundWeak, ";position:sticky;top:0px;padding-right:", ({
52
+ theme
53
+ }) => theme.space[2], ";padding-left:", ({
54
+ theme
55
+ }) => theme.space[2], ";height:", ({
56
+ theme
57
+ }) => theme.space[4], ";text-align:left;margin-bottom:", ({
58
+ theme
59
+ }) => theme.space["0.25"], ";&:focus{background-color:", ({
60
+ theme
61
+ }) => theme.colors.neutral.backgroundWeak, ";outline:none;}&[data-selectgroup='true']{padding-left:", ({
62
+ theme
63
+ }) => theme.space[2], ";border-left:", ({
64
+ theme
65
+ }) => theme.space[0.5], " solid ", ({
66
+ theme
67
+ }) => theme.colors.neutral.backgroundWeak, ";}&[data-selectgroup='true']:focus{background-color:", ({
68
+ theme
69
+ }) => theme.colors.neutral.backgroundHover, ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/Dropdown.tsx"],"names":[],"mappings":"AAoEoE","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/Dropdown.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { useEffect, useMemo, useRef, useState } from 'react'\nimport type { Dispatch, ReactNode, RefObject, SetStateAction } from 'react'\nimport { Checkbox } from '../Checkbox'\nimport { Popup } from '../Popup'\nimport { Skeleton } from '../Skeleton'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\nimport { DisplayOption } from './DropdownOption'\nimport { SearchBarDropdown } from './SearchBarDropdown'\nimport { useSelectInput } from './SelectInputProvider'\nimport type { DataType, OptionType } from './types'\n\nexport type DropdownProps = {\n  children: ReactNode\n  emptyState: ReactNode\n  descriptionDirection: 'row' | 'column'\n  searchable: boolean\n  placeholder: string\n  footer?: ReactNode\n  refSelect: RefObject<HTMLDivElement>\n  loadMore?: ReactNode\n  optionalInfoPlacement: 'left' | 'right'\n  isLoading?: boolean\n}\n\nexport type CreateDropdownProps = {\n  isEmpty: boolean\n  emptyState: ReactNode\n  descriptionDirection: 'row' | 'column'\n  loadMore?: ReactNode\n  optionalInfoPlacement: 'left' | 'right'\n  defaultSearchValue: string | null\n  isLoading?: boolean\n}\n\nconst NON_SEARCHABLE_KEYS = [\n  'Tab',\n  ' ',\n  'Enter',\n  'CapsLock',\n  'Shift',\n  'ArrowDown',\n  'ArrowUp',\n  'ArrowLeft',\n  'ArrowRight',\n  'Escape',\n]\n\nconst StyledPopup = styled(Popup)`\n  width: 100%;\n  background-color: ${({ theme }) => theme.colors.neutral.background};\n  color: ${({ theme }) => theme.colors.neutral.text};\n  box-shadow: ${({ theme }) => theme.shadows.dropdown};\n  padding: ${({ theme }) => theme.space[0]};\n`\n\nconst DropdownContainer = styled(Stack)<{ 'data-grouped': boolean }>`\n  max-height: 256px;\n  overflow-y: scroll;\n  padding: ${({ theme }) => theme.space[0]};\n  padding-bottom: ${({ theme }) => theme.space[0.5]};\n  padding-top: ${({ theme }) => theme.space[0.5]};\n\n  &[data-grouped='true'] {\n    padding-top: ${({ theme }) => theme.space[0]};\n  }\n`\nconst DropdownGroup = styled.button<{ 'data-selectgroup': boolean }>`\n  display: flex;\n  width: 100%;\n  justify-content: left;\n  align-items: center;\n  border: none;\n  background-color: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n  position: sticky;\n  top: 0px;\n  padding-right: ${({ theme }) => theme.space[2]};\n  padding-left: ${({ theme }) => theme.space[2]};\n  height: ${({ theme }) => theme.space[4]};\n  text-align: left;\n  margin-bottom: ${({ theme }) => theme.space['0.25']};\n\n  &:focus {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n    outline: none;\n  }\n\n  &[data-selectgroup='true'] {\n    padding-left: ${({ theme }) => theme.space[2]};\n    border-left: ${({ theme }) => theme.space[0.5]} solid ${({ theme }) => theme.colors.neutral.backgroundWeak};\n  }\n\n  &[data-selectgroup='true']:focus {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundHover};\n  }\n}\n`\nconst DropdownGroupWrapper = styled.div`\n  position: sticky;\n  top: 0px;\n`\nconst DropdownItem = styled.button<{\n  'data-selected': boolean\n  disabled: boolean\n}>`\n  text-align:left;\n  border: none;\n  background-color: ${({ theme }) => theme.colors.neutral.background};\n\n  padding: ${({ theme }) => theme.space['1.5']} ${({ theme }) => theme.space['2']} ${({ theme }) => theme.space['1.5']} ${({ theme }) => theme.space['2']};\n  margin-left: ${({ theme }) => theme.space['0.5']};\n  margin-right: ${({ theme }) => theme.space['0.5']};\n\n  color:  ${({ theme }) => theme.colors.neutral.text};\n  border-radius: ${({ theme }) => theme.radii.default};\n\n  &:hover, :focus {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n    color: ${({ theme }) => theme.colors.primary.text};\n    cursor: pointer;\n    outline: none;\n  }\n\n  &[data-selected='true'] {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n  }\n\n  &[disabled] {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n\n  }\n\n  &[disabled]:hover, [disabled]:focus {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundStrongDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textStrongDisabled};\n    cursor: not-allowed;\n    outline: none;\n  }\n\n  }\n`\nconst PopupFooter = styled.div`\n  width: 100%;\n  padding: ${({ theme }) => theme.space[1.5]} ${({ theme }) => theme.space[2]}\n    ${({ theme }) => theme.space[1.5]} ${({ theme }) => theme.space[2]};\n  box-shadow: ${({ theme }) => theme.shadows.dropdown};\n`\nconst StyledCheckbox = styled(Checkbox)`\n  width: 100%;\n  position: static;\n  text-align: left;\n  align-items: center;\n  pointer-events: none;\n` // pointer-events: none prevents any error when using the checkbox in a form since it is an unnamed input\n\nconst EmptyState = styled(Stack)`\n  padding: ${({ theme }) => theme.space[2]};\n`\nconst LoadMore = styled(Stack)`\n  padding: ${({ theme }) => theme.space[0.5]};\n`\n\nconst moveFocusDown = () => {\n  const options = document.querySelectorAll(\n    '#items > button[role=\"option\"]:not([disabled])',\n  )\n  const activeItem = document.activeElement\n  if (options) {\n    for (let i = 0; i < options?.length; i += 1) {\n      const listLength = options.length\n      if (activeItem === options[i] && activeItem !== options[listLength - 1]) {\n        ;(options[i + 1] as HTMLElement).focus()\n      }\n    }\n  }\n}\nconst moveFocusUp = () => {\n  const options = document.querySelectorAll(\n    '#items > button[role=\"option\"]:not([disabled])',\n  )\n  const activeItem = document.activeElement\n\n  if (options) {\n    for (let i = 0; i < options.length; i += 1) {\n      if (activeItem === options[i] && activeItem !== options[0]) {\n        ;(options[i - 1] as HTMLElement).focus()\n      }\n    }\n  }\n}\nconst handleKeyDownSelect = (key: string) => {\n  if (key === 'ArrowDown') {\n    moveFocusDown()\n  }\n  if (key === 'ArrowUp') {\n    moveFocusUp()\n  }\n}\nconst handleClickOutside = (\n  event: MouseEvent,\n  ref: RefObject<HTMLDivElement>,\n  setIsDropdownVisibile: Dispatch<SetStateAction<boolean>>,\n  refSelect: RefObject<HTMLDivElement>,\n  onSearch: Dispatch<SetStateAction<DataType>>,\n  options: DataType,\n) => {\n  if (\n    ref.current &&\n    !ref.current.contains(event.target as Node) &&\n    !refSelect.current?.contains(event.target as Node)\n  ) {\n    setIsDropdownVisibile(false) // hide dropdown when clicking outside of the dropdown\n    onSearch(options) // reset displayed options to default when dropdown is hidden\n  }\n}\n\nconst handleKeyDown = (\n  event: KeyboardEvent,\n  ref: RefObject<HTMLDivElement>,\n  options: DataType,\n  searchBarActive: boolean,\n  setSearch: Dispatch<SetStateAction<string>>,\n  setDefaultSearch: Dispatch<SetStateAction<string | null>>,\n  search: string,\n) => {\n  // Deals with default search\n  if (\n    ref.current &&\n    !searchBarActive &&\n    !NON_SEARCHABLE_KEYS.includes(event.key) &&\n    document.activeElement?.ariaLabel !== 'search-bar'\n  ) {\n    const currentSearch = search + event.key\n    setSearch(currentSearch)\n    ref.current.focus()\n    if (!Array.isArray(options)) {\n      const closestOptions = { ...options }\n      Object.keys(closestOptions).map((group: string) => {\n        closestOptions[group] = closestOptions[group].filter(option =>\n          option.searchText\n            ? option.searchText.toLocaleLowerCase().startsWith(currentSearch)\n            : option.value.toLocaleLowerCase().startsWith(currentSearch),\n        )\n\n        return null\n      })\n      const closestOption = closestOptions[Object.keys(closestOptions)[0]][0]\n      if (closestOption) {\n        setDefaultSearch(closestOption.searchText ?? closestOption.value)\n      } else {\n        setDefaultSearch(null)\n      }\n    } else {\n      const closestOption = [...options].filter(option =>\n        option.searchText\n          ? option.searchText.toLocaleLowerCase().startsWith(currentSearch)\n          : option.value.toLocaleLowerCase().startsWith(currentSearch),\n      )[0]\n      if (closestOption) {\n        setDefaultSearch(closestOption.searchText ?? closestOption.value)\n      } else {\n        setDefaultSearch(null)\n      }\n    }\n  }\n}\nconst CreateDropdown = ({\n  isEmpty,\n  emptyState,\n  descriptionDirection,\n  loadMore,\n  optionalInfoPlacement,\n  defaultSearchValue,\n  isLoading,\n}: CreateDropdownProps) => {\n  const {\n    setIsDropdownVisible,\n    onChange,\n    options,\n    multiselect,\n    selectAll,\n    selectAllGroup,\n    displayedOptions,\n    setSelectedData,\n    selectedData,\n  } = useSelectInput()\n  const focusedItemRef = useRef<HTMLButtonElement>(null)\n\n  useEffect(() => {\n    if (defaultSearchValue && focusedItemRef?.current) {\n      focusedItemRef.current.focus()\n    }\n  }, [defaultSearchValue])\n\n  if (isEmpty) {\n    return (\n      <EmptyState gap={2} alignItems=\"center\">\n        {emptyState ?? (\n          <Text variant=\"bodyStrong\" as=\"p\">\n            No options\n          </Text>\n        )}\n      </EmptyState>\n    )\n  }\n\n  const handleClick = (clickedOption: OptionType, group?: string) => {\n    setSelectedData({ type: 'selectOption', clickedOption, group })\n    if (multiselect) {\n      if (selectedData.selectedValues.includes(clickedOption.value)) {\n        onChange?.(\n          selectedData.selectedValues.filter(\n            val => val !== clickedOption.value,\n          ),\n        )\n      } else {\n        onChange?.([...selectedData.selectedValues, clickedOption.value])\n      }\n    } else {\n      onChange?.(clickedOption.value)\n    }\n    setIsDropdownVisible(multiselect) // hide the dropdown on click when single select only\n  }\n\n  const selectAllOptions = () => {\n    if (multiselect) {\n      setSelectedData({ type: 'selectAll' })\n      if (selectedData.allSelected && onChange) {\n        onChange([])\n      } else {\n        const allValues: OptionType[] = []\n        if (!Array.isArray(options)) {\n          Object.keys(options).map((group: string) =>\n            options[group].map(option => {\n              if (!option.disabled) {\n                allValues.push(option)\n              }\n\n              return null\n            }),\n          )\n        } else {\n          options.map(option => allValues.push(option))\n        }\n        onChange?.(allValues.map(value => value.value))\n      }\n    }\n  }\n\n  const handleSelectGroup = (group: string) => {\n    if (multiselect) {\n      setSelectedData({ type: 'selectGroup', selectedGroup: group })\n      if (!Array.isArray(options)) {\n        if (selectedData.selectedGroups.includes(group)) {\n          const newSelectedValues = [...selectedData.selectedValues].filter(\n            selectedValue =>\n              !options[group].find(option => option.value === selectedValue),\n          )\n          onChange?.(newSelectedValues)\n        } else {\n          const newSelectedValues = [...selectedData.selectedValues]\n\n          options[group].map(option =>\n            newSelectedValues.includes(option.value) || option.disabled\n              ? null\n              : newSelectedValues.push(option.value),\n          )\n          onChange?.(newSelectedValues)\n        }\n      }\n    }\n  }\n\n  return !Array.isArray(displayedOptions) ? (\n    <DropdownContainer\n      role=\"listbox\"\n      id=\"select-dropdown\"\n      onKeyDown={event => {\n        event.preventDefault()\n        handleKeyDownSelect(event.key)\n      }}\n      data-grouped\n    >\n      {isLoading ? (\n        <Skeleton variant=\"block\" />\n      ) : (\n        <>\n          {selectAll && multiselect ? (\n            <Stack id=\"items\">\n              <DropdownItem\n                disabled={false}\n                data-selected={selectedData.allSelected}\n                aria-label=\"select-all\"\n                data-testid=\"select-all\"\n                id=\"select-all\"\n                role=\"option\"\n                onKeyDown={event =>\n                  [' ', 'Enter'].includes(event.key) ? selectAllOptions() : null\n                }\n                onClick={selectAllOptions}\n              >\n                <StyledCheckbox\n                  checked={selectedData.allSelected}\n                  disabled={false}\n                  value=\"select-all\"\n                  data-testid=\"select-all-checkbox\"\n                  tabIndex={-1}\n                >\n                  <Stack direction=\"column\">\n                    <Text as=\"span\" variant=\"body\" placement=\"left\">\n                      {selectAll.label}\n                    </Text>\n                    <Text\n                      as=\"span\"\n                      variant=\"bodySmall\"\n                      sentiment=\"neutral\"\n                      placement=\"left\"\n                      prominence=\"weak\"\n                    >\n                      {selectAll.description}\n                    </Text>\n                  </Stack>\n                </StyledCheckbox>\n              </DropdownItem>\n            </Stack>\n          ) : null}\n          {Object.keys(displayedOptions).map((group, index) => (\n            <Stack key={group} gap={0.25}>\n              {displayedOptions[group].length > 0 ? (\n                <DropdownGroupWrapper id={selectAllGroup ? 'items' : undefined}>\n                  <DropdownGroup\n                    key={group}\n                    tabIndex={selectAllGroup ? 0 : -1}\n                    onKeyDown={event => {\n                      if ([' ', 'Enter'].includes(event.key)) {\n                        event.preventDefault()\n                        handleSelectGroup(group)\n                      }\n                    }}\n                    data-selectgroup={selectAllGroup}\n                    role=\"group\"\n                    data-testid={`group-${index}`}\n                    onClick={() =>\n                      selectAllGroup ? handleSelectGroup(group) : null\n                    }\n                  >\n                    {selectAllGroup ? (\n                      <StyledCheckbox\n                        checked={selectedData.selectedGroups.includes(group)}\n                        disabled={false}\n                        value={group}\n                        data-testid=\"select-group\"\n                        tabIndex={-1}\n                      >\n                        <Text variant=\"caption\" as=\"span\" placement=\"left\">\n                          {group.toUpperCase()}\n                        </Text>\n                      </StyledCheckbox>\n                    ) : (\n                      <Text variant=\"caption\" as=\"span\" placement=\"left\">\n                        {group.toUpperCase()}\n                      </Text>\n                    )}\n                  </DropdownGroup>\n                </DropdownGroupWrapper>\n              ) : null}\n              <Stack id=\"items\" gap=\"0.25\">\n                {displayedOptions[group].map((option, indexOption) => (\n                  <DropdownItem\n                    key={option.value}\n                    disabled={option.disabled}\n                    data-selected={\n                      selectedData.selectedValues.includes(option.value) &&\n                      !option.disabled\n                    }\n                    aria-label={option.value}\n                    data-testid={`option-${option.value}`}\n                    id={`option-${indexOption}`}\n                    role=\"option\"\n                    onClick={() => {\n                      if (!option.disabled) {\n                        handleClick(option, group)\n                      }\n                    }}\n                    onKeyDown={event =>\n                      [' ', 'Enter'].includes(event.key)\n                        ? handleClick(option, group)\n                        : null\n                    }\n                    ref={\n                      option.value === defaultSearchValue ||\n                      option.searchText === defaultSearchValue\n                        ? focusedItemRef\n                        : null\n                    }\n                  >\n                    {multiselect ? (\n                      <StyledCheckbox\n                        checked={\n                          selectedData.selectedValues.includes(option.value) &&\n                          !option.disabled\n                        }\n                        disabled={option.disabled}\n                        value={option.value}\n                        tabIndex={-1}\n                      >\n                        <DisplayOption\n                          option={option}\n                          descriptionDirection={descriptionDirection}\n                          optionalInfoPlacement={optionalInfoPlacement}\n                        />\n                      </StyledCheckbox>\n                    ) : (\n                      <DisplayOption\n                        option={option}\n                        descriptionDirection={descriptionDirection}\n                        optionalInfoPlacement={optionalInfoPlacement}\n                      />\n                    )}\n                  </DropdownItem>\n                ))}\n                {loadMore ? <LoadMore>{loadMore}</LoadMore> : null}\n              </Stack>\n            </Stack>\n          ))}\n        </>\n      )}\n    </DropdownContainer>\n  ) : (\n    <DropdownContainer\n      role=\"listbox\"\n      id=\"select-dropdown\"\n      onKeyDown={event => {\n        event.preventDefault()\n        handleKeyDownSelect(event.key)\n      }}\n      gap={0.25}\n      data-grouped={false}\n    >\n      {selectAll && multiselect ? (\n        <Stack id=\"items\" gap={0.25}>\n          <DropdownItem\n            disabled={false}\n            data-selected={selectedData.allSelected}\n            aria-label=\"select-all\"\n            data-testid=\"select-all\"\n            role=\"option\"\n            onKeyDown={event =>\n              [' ', 'Enter'].includes(event.key) ? selectAllOptions() : null\n            }\n            onClick={selectAllOptions}\n          >\n            <StyledCheckbox\n              checked={selectedData.allSelected}\n              disabled={false}\n              value=\"select-all\"\n              data-testid=\"select-all-checkbox\"\n              tabIndex={-1}\n            >\n              <Stack direction=\"column\">\n                <Text as=\"span\" variant=\"body\" placement=\"left\">\n                  {selectAll.label}\n                </Text>\n                <Text\n                  as=\"span\"\n                  variant=\"bodySmall\"\n                  sentiment=\"neutral\"\n                  placement=\"left\"\n                  prominence=\"weak\"\n                >\n                  {selectAll.description}\n                </Text>\n              </Stack>\n            </StyledCheckbox>\n          </DropdownItem>\n        </Stack>\n      ) : null}\n      <Stack id=\"items\" gap={0.25}>\n        {isLoading ? (\n          <Skeleton variant=\"block\" />\n        ) : (\n          displayedOptions.map((option, index) => (\n            <DropdownItem\n              key={option.value}\n              disabled={option.disabled}\n              data-selected={\n                selectedData.selectedValues.includes(option.value) &&\n                !option.disabled\n              }\n              onClick={() => {\n                if (!option.disabled) {\n                  handleClick(option)\n                }\n              }}\n              aria-label={option.value}\n              data-testid={`option-${option.value}`}\n              id={`option-${index}`}\n              role=\"option\"\n              ref={\n                option.value === defaultSearchValue ||\n                option.searchText === defaultSearchValue\n                  ? focusedItemRef\n                  : null\n              }\n              onKeyDown={event =>\n                [' ', 'Enter'].includes(event.key) ? handleClick(option) : null\n              }\n            >\n              {multiselect ? (\n                <StyledCheckbox\n                  checked={\n                    selectedData.selectedValues.includes(option.value) &&\n                    !option.disabled\n                  }\n                  disabled={option.disabled}\n                  value={option.value}\n                  tabIndex={-1}\n                >\n                  <DisplayOption\n                    option={option}\n                    descriptionDirection={descriptionDirection}\n                    optionalInfoPlacement={optionalInfoPlacement}\n                  />\n                </StyledCheckbox>\n              ) : (\n                <DisplayOption\n                  option={option}\n                  descriptionDirection={descriptionDirection}\n                  optionalInfoPlacement={optionalInfoPlacement}\n                />\n              )}\n            </DropdownItem>\n          ))\n        )}\n        {loadMore ? <LoadMore>{loadMore}</LoadMore> : null}\n      </Stack>\n    </DropdownContainer>\n  )\n}\nexport const Dropdown = ({\n  children,\n  emptyState,\n  descriptionDirection,\n  searchable,\n  placeholder,\n  footer,\n  refSelect,\n  loadMore,\n  optionalInfoPlacement,\n  isLoading,\n}: DropdownProps) => {\n  const {\n    setIsDropdownVisible,\n    isDropdownVisible,\n    onSearch,\n    searchInput,\n    options,\n    displayedOptions,\n  } = useSelectInput()\n  const [searchBarActive, setSearchBarActive] = useState(false)\n  const [defaultSearchValue, setDefaultSearch] = useState<string | null>(null)\n  const maxWidth = refSelect.current?.offsetWidth\n  const ref = useRef<HTMLDivElement>(null)\n  const [search, setSearch] = useState<string>('')\n  useEffect(() => {\n    if (!searchInput) {\n      onSearch(options)\n    }\n  }, [onSearch, options, searchInput])\n\n  useEffect(() => {\n    if (!isDropdownVisible) {\n      setDefaultSearch(null)\n      setSearch('')\n    }\n\n    document.addEventListener('mousedown', event =>\n      handleClickOutside(\n        event,\n        ref,\n        setIsDropdownVisible,\n        refSelect,\n        onSearch,\n        options,\n      ),\n    )\n\n    if (!searchable) {\n      document.addEventListener('keydown', event =>\n        handleKeyDown(\n          event,\n          ref,\n          options,\n          searchBarActive,\n          setSearch,\n          setDefaultSearch,\n          search,\n        ),\n      )\n    }\n\n    return () => {\n      document.removeEventListener('mousedown', event =>\n        handleClickOutside(\n          event,\n          ref,\n          setIsDropdownVisible,\n          refSelect,\n          onSearch,\n          options,\n        ),\n      )\n      if (!searchable) {\n        document.removeEventListener('keydown', event =>\n          handleKeyDown(\n            event,\n            ref,\n            options,\n            searchBarActive,\n            setSearch,\n            setDefaultSearch,\n            search,\n          ),\n        )\n      }\n    }\n  }, [\n    isDropdownVisible,\n    searchBarActive,\n    options,\n    onSearch,\n    search,\n    refSelect,\n    setDefaultSearch,\n    setIsDropdownVisible,\n    searchable,\n  ])\n\n  const isEmpty = useMemo(() => {\n    if (Array.isArray(displayedOptions)) {\n      return !(displayedOptions.length > 0)\n    }\n    const groups = Object.keys(displayedOptions)\n    for (const group of groups) {\n      if (displayedOptions[group].length !== 0) {\n        return false\n      }\n    }\n\n    return true\n  }, [displayedOptions])\n\n  return (\n    <StyledPopup\n      visible={isDropdownVisible}\n      text={\n        <Stack>\n          {searchable && !isLoading ? (\n            <SearchBarDropdown\n              placeholder={placeholder}\n              displayedOptions={displayedOptions}\n              setSearchBarActive={setSearchBarActive}\n            />\n          ) : null}\n          <CreateDropdown\n            isEmpty={isEmpty}\n            emptyState={emptyState}\n            descriptionDirection={descriptionDirection}\n            loadMore={loadMore}\n            optionalInfoPlacement={optionalInfoPlacement}\n            defaultSearchValue={defaultSearchValue}\n            isLoading={isLoading}\n          />\n          {footer ? <PopupFooter>{footer}</PopupFooter> : null}\n        </Stack>\n      }\n      placement=\"bottom\"\n      containerFullWidth\n      disableAnimation\n      maxWidth={maxWidth}\n      hasArrow={false}\n      ref={ref}\n      tabIndex={0}\n      role=\"dialog\"\n      debounceDelay={0}\n    >\n      {children}\n    </StyledPopup>\n  )\n}\n"]} */"));
70
+ const DropdownGroupWrapper = /* @__PURE__ */ _styled("div", process.env.NODE_ENV === "production" ? {
71
+ target: "eeucz6c5"
72
+ } : {
73
+ target: "eeucz6c5",
74
+ label: "DropdownGroupWrapper"
75
+ })(process.env.NODE_ENV === "production" ? {
76
+ name: "vkf8h3",
77
+ styles: "position:sticky;top:0px"
78
+ } : {
79
+ name: "vkf8h3",
80
+ styles: "position:sticky;top:0px",
81
+ map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/Dropdown.tsx"],"names":[],"mappings":"AAkGuC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/Dropdown.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { useEffect, useMemo, useRef, useState } from 'react'\nimport type { Dispatch, ReactNode, RefObject, SetStateAction } from 'react'\nimport { Checkbox } from '../Checkbox'\nimport { Popup } from '../Popup'\nimport { Skeleton } from '../Skeleton'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\nimport { DisplayOption } from './DropdownOption'\nimport { SearchBarDropdown } from './SearchBarDropdown'\nimport { useSelectInput } from './SelectInputProvider'\nimport type { DataType, OptionType } from './types'\n\nexport type DropdownProps = {\n  children: ReactNode\n  emptyState: ReactNode\n  descriptionDirection: 'row' | 'column'\n  searchable: boolean\n  placeholder: string\n  footer?: ReactNode\n  refSelect: RefObject<HTMLDivElement>\n  loadMore?: ReactNode\n  optionalInfoPlacement: 'left' | 'right'\n  isLoading?: boolean\n}\n\nexport type CreateDropdownProps = {\n  isEmpty: boolean\n  emptyState: ReactNode\n  descriptionDirection: 'row' | 'column'\n  loadMore?: ReactNode\n  optionalInfoPlacement: 'left' | 'right'\n  defaultSearchValue: string | null\n  isLoading?: boolean\n}\n\nconst NON_SEARCHABLE_KEYS = [\n  'Tab',\n  ' ',\n  'Enter',\n  'CapsLock',\n  'Shift',\n  'ArrowDown',\n  'ArrowUp',\n  'ArrowLeft',\n  'ArrowRight',\n  'Escape',\n]\n\nconst StyledPopup = styled(Popup)`\n  width: 100%;\n  background-color: ${({ theme }) => theme.colors.neutral.background};\n  color: ${({ theme }) => theme.colors.neutral.text};\n  box-shadow: ${({ theme }) => theme.shadows.dropdown};\n  padding: ${({ theme }) => theme.space[0]};\n`\n\nconst DropdownContainer = styled(Stack)<{ 'data-grouped': boolean }>`\n  max-height: 256px;\n  overflow-y: scroll;\n  padding: ${({ theme }) => theme.space[0]};\n  padding-bottom: ${({ theme }) => theme.space[0.5]};\n  padding-top: ${({ theme }) => theme.space[0.5]};\n\n  &[data-grouped='true'] {\n    padding-top: ${({ theme }) => theme.space[0]};\n  }\n`\nconst DropdownGroup = styled.button<{ 'data-selectgroup': boolean }>`\n  display: flex;\n  width: 100%;\n  justify-content: left;\n  align-items: center;\n  border: none;\n  background-color: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n  position: sticky;\n  top: 0px;\n  padding-right: ${({ theme }) => theme.space[2]};\n  padding-left: ${({ theme }) => theme.space[2]};\n  height: ${({ theme }) => theme.space[4]};\n  text-align: left;\n  margin-bottom: ${({ theme }) => theme.space['0.25']};\n\n  &:focus {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n    outline: none;\n  }\n\n  &[data-selectgroup='true'] {\n    padding-left: ${({ theme }) => theme.space[2]};\n    border-left: ${({ theme }) => theme.space[0.5]} solid ${({ theme }) => theme.colors.neutral.backgroundWeak};\n  }\n\n  &[data-selectgroup='true']:focus {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundHover};\n  }\n}\n`\nconst DropdownGroupWrapper = styled.div`\n  position: sticky;\n  top: 0px;\n`\nconst DropdownItem = styled.button<{\n  'data-selected': boolean\n  disabled: boolean\n}>`\n  text-align:left;\n  border: none;\n  background-color: ${({ theme }) => theme.colors.neutral.background};\n\n  padding: ${({ theme }) => theme.space['1.5']} ${({ theme }) => theme.space['2']} ${({ theme }) => theme.space['1.5']} ${({ theme }) => theme.space['2']};\n  margin-left: ${({ theme }) => theme.space['0.5']};\n  margin-right: ${({ theme }) => theme.space['0.5']};\n\n  color:  ${({ theme }) => theme.colors.neutral.text};\n  border-radius: ${({ theme }) => theme.radii.default};\n\n  &:hover, :focus {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n    color: ${({ theme }) => theme.colors.primary.text};\n    cursor: pointer;\n    outline: none;\n  }\n\n  &[data-selected='true'] {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n  }\n\n  &[disabled] {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n\n  }\n\n  &[disabled]:hover, [disabled]:focus {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundStrongDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textStrongDisabled};\n    cursor: not-allowed;\n    outline: none;\n  }\n\n  }\n`\nconst PopupFooter = styled.div`\n  width: 100%;\n  padding: ${({ theme }) => theme.space[1.5]} ${({ theme }) => theme.space[2]}\n    ${({ theme }) => theme.space[1.5]} ${({ theme }) => theme.space[2]};\n  box-shadow: ${({ theme }) => theme.shadows.dropdown};\n`\nconst StyledCheckbox = styled(Checkbox)`\n  width: 100%;\n  position: static;\n  text-align: left;\n  align-items: center;\n  pointer-events: none;\n` // pointer-events: none prevents any error when using the checkbox in a form since it is an unnamed input\n\nconst EmptyState = styled(Stack)`\n  padding: ${({ theme }) => theme.space[2]};\n`\nconst LoadMore = styled(Stack)`\n  padding: ${({ theme }) => theme.space[0.5]};\n`\n\nconst moveFocusDown = () => {\n  const options = document.querySelectorAll(\n    '#items > button[role=\"option\"]:not([disabled])',\n  )\n  const activeItem = document.activeElement\n  if (options) {\n    for (let i = 0; i < options?.length; i += 1) {\n      const listLength = options.length\n      if (activeItem === options[i] && activeItem !== options[listLength - 1]) {\n        ;(options[i + 1] as HTMLElement).focus()\n      }\n    }\n  }\n}\nconst moveFocusUp = () => {\n  const options = document.querySelectorAll(\n    '#items > button[role=\"option\"]:not([disabled])',\n  )\n  const activeItem = document.activeElement\n\n  if (options) {\n    for (let i = 0; i < options.length; i += 1) {\n      if (activeItem === options[i] && activeItem !== options[0]) {\n        ;(options[i - 1] as HTMLElement).focus()\n      }\n    }\n  }\n}\nconst handleKeyDownSelect = (key: string) => {\n  if (key === 'ArrowDown') {\n    moveFocusDown()\n  }\n  if (key === 'ArrowUp') {\n    moveFocusUp()\n  }\n}\nconst handleClickOutside = (\n  event: MouseEvent,\n  ref: RefObject<HTMLDivElement>,\n  setIsDropdownVisibile: Dispatch<SetStateAction<boolean>>,\n  refSelect: RefObject<HTMLDivElement>,\n  onSearch: Dispatch<SetStateAction<DataType>>,\n  options: DataType,\n) => {\n  if (\n    ref.current &&\n    !ref.current.contains(event.target as Node) &&\n    !refSelect.current?.contains(event.target as Node)\n  ) {\n    setIsDropdownVisibile(false) // hide dropdown when clicking outside of the dropdown\n    onSearch(options) // reset displayed options to default when dropdown is hidden\n  }\n}\n\nconst handleKeyDown = (\n  event: KeyboardEvent,\n  ref: RefObject<HTMLDivElement>,\n  options: DataType,\n  searchBarActive: boolean,\n  setSearch: Dispatch<SetStateAction<string>>,\n  setDefaultSearch: Dispatch<SetStateAction<string | null>>,\n  search: string,\n) => {\n  // Deals with default search\n  if (\n    ref.current &&\n    !searchBarActive &&\n    !NON_SEARCHABLE_KEYS.includes(event.key) &&\n    document.activeElement?.ariaLabel !== 'search-bar'\n  ) {\n    const currentSearch = search + event.key\n    setSearch(currentSearch)\n    ref.current.focus()\n    if (!Array.isArray(options)) {\n      const closestOptions = { ...options }\n      Object.keys(closestOptions).map((group: string) => {\n        closestOptions[group] = closestOptions[group].filter(option =>\n          option.searchText\n            ? option.searchText.toLocaleLowerCase().startsWith(currentSearch)\n            : option.value.toLocaleLowerCase().startsWith(currentSearch),\n        )\n\n        return null\n      })\n      const closestOption = closestOptions[Object.keys(closestOptions)[0]][0]\n      if (closestOption) {\n        setDefaultSearch(closestOption.searchText ?? closestOption.value)\n      } else {\n        setDefaultSearch(null)\n      }\n    } else {\n      const closestOption = [...options].filter(option =>\n        option.searchText\n          ? option.searchText.toLocaleLowerCase().startsWith(currentSearch)\n          : option.value.toLocaleLowerCase().startsWith(currentSearch),\n      )[0]\n      if (closestOption) {\n        setDefaultSearch(closestOption.searchText ?? closestOption.value)\n      } else {\n        setDefaultSearch(null)\n      }\n    }\n  }\n}\nconst CreateDropdown = ({\n  isEmpty,\n  emptyState,\n  descriptionDirection,\n  loadMore,\n  optionalInfoPlacement,\n  defaultSearchValue,\n  isLoading,\n}: CreateDropdownProps) => {\n  const {\n    setIsDropdownVisible,\n    onChange,\n    options,\n    multiselect,\n    selectAll,\n    selectAllGroup,\n    displayedOptions,\n    setSelectedData,\n    selectedData,\n  } = useSelectInput()\n  const focusedItemRef = useRef<HTMLButtonElement>(null)\n\n  useEffect(() => {\n    if (defaultSearchValue && focusedItemRef?.current) {\n      focusedItemRef.current.focus()\n    }\n  }, [defaultSearchValue])\n\n  if (isEmpty) {\n    return (\n      <EmptyState gap={2} alignItems=\"center\">\n        {emptyState ?? (\n          <Text variant=\"bodyStrong\" as=\"p\">\n            No options\n          </Text>\n        )}\n      </EmptyState>\n    )\n  }\n\n  const handleClick = (clickedOption: OptionType, group?: string) => {\n    setSelectedData({ type: 'selectOption', clickedOption, group })\n    if (multiselect) {\n      if (selectedData.selectedValues.includes(clickedOption.value)) {\n        onChange?.(\n          selectedData.selectedValues.filter(\n            val => val !== clickedOption.value,\n          ),\n        )\n      } else {\n        onChange?.([...selectedData.selectedValues, clickedOption.value])\n      }\n    } else {\n      onChange?.(clickedOption.value)\n    }\n    setIsDropdownVisible(multiselect) // hide the dropdown on click when single select only\n  }\n\n  const selectAllOptions = () => {\n    if (multiselect) {\n      setSelectedData({ type: 'selectAll' })\n      if (selectedData.allSelected && onChange) {\n        onChange([])\n      } else {\n        const allValues: OptionType[] = []\n        if (!Array.isArray(options)) {\n          Object.keys(options).map((group: string) =>\n            options[group].map(option => {\n              if (!option.disabled) {\n                allValues.push(option)\n              }\n\n              return null\n            }),\n          )\n        } else {\n          options.map(option => allValues.push(option))\n        }\n        onChange?.(allValues.map(value => value.value))\n      }\n    }\n  }\n\n  const handleSelectGroup = (group: string) => {\n    if (multiselect) {\n      setSelectedData({ type: 'selectGroup', selectedGroup: group })\n      if (!Array.isArray(options)) {\n        if (selectedData.selectedGroups.includes(group)) {\n          const newSelectedValues = [...selectedData.selectedValues].filter(\n            selectedValue =>\n              !options[group].find(option => option.value === selectedValue),\n          )\n          onChange?.(newSelectedValues)\n        } else {\n          const newSelectedValues = [...selectedData.selectedValues]\n\n          options[group].map(option =>\n            newSelectedValues.includes(option.value) || option.disabled\n              ? null\n              : newSelectedValues.push(option.value),\n          )\n          onChange?.(newSelectedValues)\n        }\n      }\n    }\n  }\n\n  return !Array.isArray(displayedOptions) ? (\n    <DropdownContainer\n      role=\"listbox\"\n      id=\"select-dropdown\"\n      onKeyDown={event => {\n        event.preventDefault()\n        handleKeyDownSelect(event.key)\n      }}\n      data-grouped\n    >\n      {isLoading ? (\n        <Skeleton variant=\"block\" />\n      ) : (\n        <>\n          {selectAll && multiselect ? (\n            <Stack id=\"items\">\n              <DropdownItem\n                disabled={false}\n                data-selected={selectedData.allSelected}\n                aria-label=\"select-all\"\n                data-testid=\"select-all\"\n                id=\"select-all\"\n                role=\"option\"\n                onKeyDown={event =>\n                  [' ', 'Enter'].includes(event.key) ? selectAllOptions() : null\n                }\n                onClick={selectAllOptions}\n              >\n                <StyledCheckbox\n                  checked={selectedData.allSelected}\n                  disabled={false}\n                  value=\"select-all\"\n                  data-testid=\"select-all-checkbox\"\n                  tabIndex={-1}\n                >\n                  <Stack direction=\"column\">\n                    <Text as=\"span\" variant=\"body\" placement=\"left\">\n                      {selectAll.label}\n                    </Text>\n                    <Text\n                      as=\"span\"\n                      variant=\"bodySmall\"\n                      sentiment=\"neutral\"\n                      placement=\"left\"\n                      prominence=\"weak\"\n                    >\n                      {selectAll.description}\n                    </Text>\n                  </Stack>\n                </StyledCheckbox>\n              </DropdownItem>\n            </Stack>\n          ) : null}\n          {Object.keys(displayedOptions).map((group, index) => (\n            <Stack key={group} gap={0.25}>\n              {displayedOptions[group].length > 0 ? (\n                <DropdownGroupWrapper id={selectAllGroup ? 'items' : undefined}>\n                  <DropdownGroup\n                    key={group}\n                    tabIndex={selectAllGroup ? 0 : -1}\n                    onKeyDown={event => {\n                      if ([' ', 'Enter'].includes(event.key)) {\n                        event.preventDefault()\n                        handleSelectGroup(group)\n                      }\n                    }}\n                    data-selectgroup={selectAllGroup}\n                    role=\"group\"\n                    data-testid={`group-${index}`}\n                    onClick={() =>\n                      selectAllGroup ? handleSelectGroup(group) : null\n                    }\n                  >\n                    {selectAllGroup ? (\n                      <StyledCheckbox\n                        checked={selectedData.selectedGroups.includes(group)}\n                        disabled={false}\n                        value={group}\n                        data-testid=\"select-group\"\n                        tabIndex={-1}\n                      >\n                        <Text variant=\"caption\" as=\"span\" placement=\"left\">\n                          {group.toUpperCase()}\n                        </Text>\n                      </StyledCheckbox>\n                    ) : (\n                      <Text variant=\"caption\" as=\"span\" placement=\"left\">\n                        {group.toUpperCase()}\n                      </Text>\n                    )}\n                  </DropdownGroup>\n                </DropdownGroupWrapper>\n              ) : null}\n              <Stack id=\"items\" gap=\"0.25\">\n                {displayedOptions[group].map((option, indexOption) => (\n                  <DropdownItem\n                    key={option.value}\n                    disabled={option.disabled}\n                    data-selected={\n                      selectedData.selectedValues.includes(option.value) &&\n                      !option.disabled\n                    }\n                    aria-label={option.value}\n                    data-testid={`option-${option.value}`}\n                    id={`option-${indexOption}`}\n                    role=\"option\"\n                    onClick={() => {\n                      if (!option.disabled) {\n                        handleClick(option, group)\n                      }\n                    }}\n                    onKeyDown={event =>\n                      [' ', 'Enter'].includes(event.key)\n                        ? handleClick(option, group)\n                        : null\n                    }\n                    ref={\n                      option.value === defaultSearchValue ||\n                      option.searchText === defaultSearchValue\n                        ? focusedItemRef\n                        : null\n                    }\n                  >\n                    {multiselect ? (\n                      <StyledCheckbox\n                        checked={\n                          selectedData.selectedValues.includes(option.value) &&\n                          !option.disabled\n                        }\n                        disabled={option.disabled}\n                        value={option.value}\n                        tabIndex={-1}\n                      >\n                        <DisplayOption\n                          option={option}\n                          descriptionDirection={descriptionDirection}\n                          optionalInfoPlacement={optionalInfoPlacement}\n                        />\n                      </StyledCheckbox>\n                    ) : (\n                      <DisplayOption\n                        option={option}\n                        descriptionDirection={descriptionDirection}\n                        optionalInfoPlacement={optionalInfoPlacement}\n                      />\n                    )}\n                  </DropdownItem>\n                ))}\n                {loadMore ? <LoadMore>{loadMore}</LoadMore> : null}\n              </Stack>\n            </Stack>\n          ))}\n        </>\n      )}\n    </DropdownContainer>\n  ) : (\n    <DropdownContainer\n      role=\"listbox\"\n      id=\"select-dropdown\"\n      onKeyDown={event => {\n        event.preventDefault()\n        handleKeyDownSelect(event.key)\n      }}\n      gap={0.25}\n      data-grouped={false}\n    >\n      {selectAll && multiselect ? (\n        <Stack id=\"items\" gap={0.25}>\n          <DropdownItem\n            disabled={false}\n            data-selected={selectedData.allSelected}\n            aria-label=\"select-all\"\n            data-testid=\"select-all\"\n            role=\"option\"\n            onKeyDown={event =>\n              [' ', 'Enter'].includes(event.key) ? selectAllOptions() : null\n            }\n            onClick={selectAllOptions}\n          >\n            <StyledCheckbox\n              checked={selectedData.allSelected}\n              disabled={false}\n              value=\"select-all\"\n              data-testid=\"select-all-checkbox\"\n              tabIndex={-1}\n            >\n              <Stack direction=\"column\">\n                <Text as=\"span\" variant=\"body\" placement=\"left\">\n                  {selectAll.label}\n                </Text>\n                <Text\n                  as=\"span\"\n                  variant=\"bodySmall\"\n                  sentiment=\"neutral\"\n                  placement=\"left\"\n                  prominence=\"weak\"\n                >\n                  {selectAll.description}\n                </Text>\n              </Stack>\n            </StyledCheckbox>\n          </DropdownItem>\n        </Stack>\n      ) : null}\n      <Stack id=\"items\" gap={0.25}>\n        {isLoading ? (\n          <Skeleton variant=\"block\" />\n        ) : (\n          displayedOptions.map((option, index) => (\n            <DropdownItem\n              key={option.value}\n              disabled={option.disabled}\n              data-selected={\n                selectedData.selectedValues.includes(option.value) &&\n                !option.disabled\n              }\n              onClick={() => {\n                if (!option.disabled) {\n                  handleClick(option)\n                }\n              }}\n              aria-label={option.value}\n              data-testid={`option-${option.value}`}\n              id={`option-${index}`}\n              role=\"option\"\n              ref={\n                option.value === defaultSearchValue ||\n                option.searchText === defaultSearchValue\n                  ? focusedItemRef\n                  : null\n              }\n              onKeyDown={event =>\n                [' ', 'Enter'].includes(event.key) ? handleClick(option) : null\n              }\n            >\n              {multiselect ? (\n                <StyledCheckbox\n                  checked={\n                    selectedData.selectedValues.includes(option.value) &&\n                    !option.disabled\n                  }\n                  disabled={option.disabled}\n                  value={option.value}\n                  tabIndex={-1}\n                >\n                  <DisplayOption\n                    option={option}\n                    descriptionDirection={descriptionDirection}\n                    optionalInfoPlacement={optionalInfoPlacement}\n                  />\n                </StyledCheckbox>\n              ) : (\n                <DisplayOption\n                  option={option}\n                  descriptionDirection={descriptionDirection}\n                  optionalInfoPlacement={optionalInfoPlacement}\n                />\n              )}\n            </DropdownItem>\n          ))\n        )}\n        {loadMore ? <LoadMore>{loadMore}</LoadMore> : null}\n      </Stack>\n    </DropdownContainer>\n  )\n}\nexport const Dropdown = ({\n  children,\n  emptyState,\n  descriptionDirection,\n  searchable,\n  placeholder,\n  footer,\n  refSelect,\n  loadMore,\n  optionalInfoPlacement,\n  isLoading,\n}: DropdownProps) => {\n  const {\n    setIsDropdownVisible,\n    isDropdownVisible,\n    onSearch,\n    searchInput,\n    options,\n    displayedOptions,\n  } = useSelectInput()\n  const [searchBarActive, setSearchBarActive] = useState(false)\n  const [defaultSearchValue, setDefaultSearch] = useState<string | null>(null)\n  const maxWidth = refSelect.current?.offsetWidth\n  const ref = useRef<HTMLDivElement>(null)\n  const [search, setSearch] = useState<string>('')\n  useEffect(() => {\n    if (!searchInput) {\n      onSearch(options)\n    }\n  }, [onSearch, options, searchInput])\n\n  useEffect(() => {\n    if (!isDropdownVisible) {\n      setDefaultSearch(null)\n      setSearch('')\n    }\n\n    document.addEventListener('mousedown', event =>\n      handleClickOutside(\n        event,\n        ref,\n        setIsDropdownVisible,\n        refSelect,\n        onSearch,\n        options,\n      ),\n    )\n\n    if (!searchable) {\n      document.addEventListener('keydown', event =>\n        handleKeyDown(\n          event,\n          ref,\n          options,\n          searchBarActive,\n          setSearch,\n          setDefaultSearch,\n          search,\n        ),\n      )\n    }\n\n    return () => {\n      document.removeEventListener('mousedown', event =>\n        handleClickOutside(\n          event,\n          ref,\n          setIsDropdownVisible,\n          refSelect,\n          onSearch,\n          options,\n        ),\n      )\n      if (!searchable) {\n        document.removeEventListener('keydown', event =>\n          handleKeyDown(\n            event,\n            ref,\n            options,\n            searchBarActive,\n            setSearch,\n            setDefaultSearch,\n            search,\n          ),\n        )\n      }\n    }\n  }, [\n    isDropdownVisible,\n    searchBarActive,\n    options,\n    onSearch,\n    search,\n    refSelect,\n    setDefaultSearch,\n    setIsDropdownVisible,\n    searchable,\n  ])\n\n  const isEmpty = useMemo(() => {\n    if (Array.isArray(displayedOptions)) {\n      return !(displayedOptions.length > 0)\n    }\n    const groups = Object.keys(displayedOptions)\n    for (const group of groups) {\n      if (displayedOptions[group].length !== 0) {\n        return false\n      }\n    }\n\n    return true\n  }, [displayedOptions])\n\n  return (\n    <StyledPopup\n      visible={isDropdownVisible}\n      text={\n        <Stack>\n          {searchable && !isLoading ? (\n            <SearchBarDropdown\n              placeholder={placeholder}\n              displayedOptions={displayedOptions}\n              setSearchBarActive={setSearchBarActive}\n            />\n          ) : null}\n          <CreateDropdown\n            isEmpty={isEmpty}\n            emptyState={emptyState}\n            descriptionDirection={descriptionDirection}\n            loadMore={loadMore}\n            optionalInfoPlacement={optionalInfoPlacement}\n            defaultSearchValue={defaultSearchValue}\n            isLoading={isLoading}\n          />\n          {footer ? <PopupFooter>{footer}</PopupFooter> : null}\n        </Stack>\n      }\n      placement=\"bottom\"\n      containerFullWidth\n      disableAnimation\n      maxWidth={maxWidth}\n      hasArrow={false}\n      ref={ref}\n      tabIndex={0}\n      role=\"dialog\"\n      debounceDelay={0}\n    >\n      {children}\n    </StyledPopup>\n  )\n}\n"]} */",
82
+ toString: _EMOTION_STRINGIFIED_CSS_ERROR__
83
+ });
84
+ const DropdownItem = /* @__PURE__ */ _styled("button", process.env.NODE_ENV === "production" ? {
85
+ target: "eeucz6c4"
86
+ } : {
87
+ target: "eeucz6c4",
88
+ label: "DropdownItem"
89
+ })("text-align:left;border:none;background-color:", ({
90
+ theme
91
+ }) => theme.colors.neutral.background, ";padding:", ({
92
+ theme
93
+ }) => theme.space["1.5"], " ", ({
94
+ theme
95
+ }) => theme.space["2"], " ", ({
96
+ theme
97
+ }) => theme.space["1.5"], " ", ({
98
+ theme
99
+ }) => theme.space["2"], ";margin-left:", ({
100
+ theme
101
+ }) => theme.space["0.5"], ";margin-right:", ({
102
+ theme
103
+ }) => theme.space["0.5"], ";color:", ({
104
+ theme
105
+ }) => theme.colors.neutral.text, ";border-radius:", ({
106
+ theme
107
+ }) => theme.radii.default, ";&:hover,:focus{background-color:", ({
108
+ theme
109
+ }) => theme.colors.primary.background, ";color:", ({
110
+ theme
111
+ }) => theme.colors.primary.text, ";cursor:pointer;outline:none;}&[data-selected='true']{background-color:", ({
112
+ theme
113
+ }) => theme.colors.primary.background, ";}&[disabled]{background-color:", ({
114
+ theme
115
+ }) => theme.colors.neutral.backgroundDisabled, ";color:", ({
116
+ theme
117
+ }) => theme.colors.neutral.textDisabled, ";}&[disabled]:hover,[disabled]:focus{background-color:", ({
118
+ theme
119
+ }) => theme.colors.neutral.backgroundStrongDisabled, ";color:", ({
120
+ theme
121
+ }) => theme.colors.neutral.textStrongDisabled, ";cursor:not-allowed;outline:none;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/Dropdown.tsx"],"names":[],"mappings":"AAyGE","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/Dropdown.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { useEffect, useMemo, useRef, useState } from 'react'\nimport type { Dispatch, ReactNode, RefObject, SetStateAction } from 'react'\nimport { Checkbox } from '../Checkbox'\nimport { Popup } from '../Popup'\nimport { Skeleton } from '../Skeleton'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\nimport { DisplayOption } from './DropdownOption'\nimport { SearchBarDropdown } from './SearchBarDropdown'\nimport { useSelectInput } from './SelectInputProvider'\nimport type { DataType, OptionType } from './types'\n\nexport type DropdownProps = {\n  children: ReactNode\n  emptyState: ReactNode\n  descriptionDirection: 'row' | 'column'\n  searchable: boolean\n  placeholder: string\n  footer?: ReactNode\n  refSelect: RefObject<HTMLDivElement>\n  loadMore?: ReactNode\n  optionalInfoPlacement: 'left' | 'right'\n  isLoading?: boolean\n}\n\nexport type CreateDropdownProps = {\n  isEmpty: boolean\n  emptyState: ReactNode\n  descriptionDirection: 'row' | 'column'\n  loadMore?: ReactNode\n  optionalInfoPlacement: 'left' | 'right'\n  defaultSearchValue: string | null\n  isLoading?: boolean\n}\n\nconst NON_SEARCHABLE_KEYS = [\n  'Tab',\n  ' ',\n  'Enter',\n  'CapsLock',\n  'Shift',\n  'ArrowDown',\n  'ArrowUp',\n  'ArrowLeft',\n  'ArrowRight',\n  'Escape',\n]\n\nconst StyledPopup = styled(Popup)`\n  width: 100%;\n  background-color: ${({ theme }) => theme.colors.neutral.background};\n  color: ${({ theme }) => theme.colors.neutral.text};\n  box-shadow: ${({ theme }) => theme.shadows.dropdown};\n  padding: ${({ theme }) => theme.space[0]};\n`\n\nconst DropdownContainer = styled(Stack)<{ 'data-grouped': boolean }>`\n  max-height: 256px;\n  overflow-y: scroll;\n  padding: ${({ theme }) => theme.space[0]};\n  padding-bottom: ${({ theme }) => theme.space[0.5]};\n  padding-top: ${({ theme }) => theme.space[0.5]};\n\n  &[data-grouped='true'] {\n    padding-top: ${({ theme }) => theme.space[0]};\n  }\n`\nconst DropdownGroup = styled.button<{ 'data-selectgroup': boolean }>`\n  display: flex;\n  width: 100%;\n  justify-content: left;\n  align-items: center;\n  border: none;\n  background-color: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n  position: sticky;\n  top: 0px;\n  padding-right: ${({ theme }) => theme.space[2]};\n  padding-left: ${({ theme }) => theme.space[2]};\n  height: ${({ theme }) => theme.space[4]};\n  text-align: left;\n  margin-bottom: ${({ theme }) => theme.space['0.25']};\n\n  &:focus {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n    outline: none;\n  }\n\n  &[data-selectgroup='true'] {\n    padding-left: ${({ theme }) => theme.space[2]};\n    border-left: ${({ theme }) => theme.space[0.5]} solid ${({ theme }) => theme.colors.neutral.backgroundWeak};\n  }\n\n  &[data-selectgroup='true']:focus {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundHover};\n  }\n}\n`\nconst DropdownGroupWrapper = styled.div`\n  position: sticky;\n  top: 0px;\n`\nconst DropdownItem = styled.button<{\n  'data-selected': boolean\n  disabled: boolean\n}>`\n  text-align:left;\n  border: none;\n  background-color: ${({ theme }) => theme.colors.neutral.background};\n\n  padding: ${({ theme }) => theme.space['1.5']} ${({ theme }) => theme.space['2']} ${({ theme }) => theme.space['1.5']} ${({ theme }) => theme.space['2']};\n  margin-left: ${({ theme }) => theme.space['0.5']};\n  margin-right: ${({ theme }) => theme.space['0.5']};\n\n  color:  ${({ theme }) => theme.colors.neutral.text};\n  border-radius: ${({ theme }) => theme.radii.default};\n\n  &:hover, :focus {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n    color: ${({ theme }) => theme.colors.primary.text};\n    cursor: pointer;\n    outline: none;\n  }\n\n  &[data-selected='true'] {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n  }\n\n  &[disabled] {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n\n  }\n\n  &[disabled]:hover, [disabled]:focus {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundStrongDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textStrongDisabled};\n    cursor: not-allowed;\n    outline: none;\n  }\n\n  }\n`\nconst PopupFooter = styled.div`\n  width: 100%;\n  padding: ${({ theme }) => theme.space[1.5]} ${({ theme }) => theme.space[2]}\n    ${({ theme }) => theme.space[1.5]} ${({ theme }) => theme.space[2]};\n  box-shadow: ${({ theme }) => theme.shadows.dropdown};\n`\nconst StyledCheckbox = styled(Checkbox)`\n  width: 100%;\n  position: static;\n  text-align: left;\n  align-items: center;\n  pointer-events: none;\n` // pointer-events: none prevents any error when using the checkbox in a form since it is an unnamed input\n\nconst EmptyState = styled(Stack)`\n  padding: ${({ theme }) => theme.space[2]};\n`\nconst LoadMore = styled(Stack)`\n  padding: ${({ theme }) => theme.space[0.5]};\n`\n\nconst moveFocusDown = () => {\n  const options = document.querySelectorAll(\n    '#items > button[role=\"option\"]:not([disabled])',\n  )\n  const activeItem = document.activeElement\n  if (options) {\n    for (let i = 0; i < options?.length; i += 1) {\n      const listLength = options.length\n      if (activeItem === options[i] && activeItem !== options[listLength - 1]) {\n        ;(options[i + 1] as HTMLElement).focus()\n      }\n    }\n  }\n}\nconst moveFocusUp = () => {\n  const options = document.querySelectorAll(\n    '#items > button[role=\"option\"]:not([disabled])',\n  )\n  const activeItem = document.activeElement\n\n  if (options) {\n    for (let i = 0; i < options.length; i += 1) {\n      if (activeItem === options[i] && activeItem !== options[0]) {\n        ;(options[i - 1] as HTMLElement).focus()\n      }\n    }\n  }\n}\nconst handleKeyDownSelect = (key: string) => {\n  if (key === 'ArrowDown') {\n    moveFocusDown()\n  }\n  if (key === 'ArrowUp') {\n    moveFocusUp()\n  }\n}\nconst handleClickOutside = (\n  event: MouseEvent,\n  ref: RefObject<HTMLDivElement>,\n  setIsDropdownVisibile: Dispatch<SetStateAction<boolean>>,\n  refSelect: RefObject<HTMLDivElement>,\n  onSearch: Dispatch<SetStateAction<DataType>>,\n  options: DataType,\n) => {\n  if (\n    ref.current &&\n    !ref.current.contains(event.target as Node) &&\n    !refSelect.current?.contains(event.target as Node)\n  ) {\n    setIsDropdownVisibile(false) // hide dropdown when clicking outside of the dropdown\n    onSearch(options) // reset displayed options to default when dropdown is hidden\n  }\n}\n\nconst handleKeyDown = (\n  event: KeyboardEvent,\n  ref: RefObject<HTMLDivElement>,\n  options: DataType,\n  searchBarActive: boolean,\n  setSearch: Dispatch<SetStateAction<string>>,\n  setDefaultSearch: Dispatch<SetStateAction<string | null>>,\n  search: string,\n) => {\n  // Deals with default search\n  if (\n    ref.current &&\n    !searchBarActive &&\n    !NON_SEARCHABLE_KEYS.includes(event.key) &&\n    document.activeElement?.ariaLabel !== 'search-bar'\n  ) {\n    const currentSearch = search + event.key\n    setSearch(currentSearch)\n    ref.current.focus()\n    if (!Array.isArray(options)) {\n      const closestOptions = { ...options }\n      Object.keys(closestOptions).map((group: string) => {\n        closestOptions[group] = closestOptions[group].filter(option =>\n          option.searchText\n            ? option.searchText.toLocaleLowerCase().startsWith(currentSearch)\n            : option.value.toLocaleLowerCase().startsWith(currentSearch),\n        )\n\n        return null\n      })\n      const closestOption = closestOptions[Object.keys(closestOptions)[0]][0]\n      if (closestOption) {\n        setDefaultSearch(closestOption.searchText ?? closestOption.value)\n      } else {\n        setDefaultSearch(null)\n      }\n    } else {\n      const closestOption = [...options].filter(option =>\n        option.searchText\n          ? option.searchText.toLocaleLowerCase().startsWith(currentSearch)\n          : option.value.toLocaleLowerCase().startsWith(currentSearch),\n      )[0]\n      if (closestOption) {\n        setDefaultSearch(closestOption.searchText ?? closestOption.value)\n      } else {\n        setDefaultSearch(null)\n      }\n    }\n  }\n}\nconst CreateDropdown = ({\n  isEmpty,\n  emptyState,\n  descriptionDirection,\n  loadMore,\n  optionalInfoPlacement,\n  defaultSearchValue,\n  isLoading,\n}: CreateDropdownProps) => {\n  const {\n    setIsDropdownVisible,\n    onChange,\n    options,\n    multiselect,\n    selectAll,\n    selectAllGroup,\n    displayedOptions,\n    setSelectedData,\n    selectedData,\n  } = useSelectInput()\n  const focusedItemRef = useRef<HTMLButtonElement>(null)\n\n  useEffect(() => {\n    if (defaultSearchValue && focusedItemRef?.current) {\n      focusedItemRef.current.focus()\n    }\n  }, [defaultSearchValue])\n\n  if (isEmpty) {\n    return (\n      <EmptyState gap={2} alignItems=\"center\">\n        {emptyState ?? (\n          <Text variant=\"bodyStrong\" as=\"p\">\n            No options\n          </Text>\n        )}\n      </EmptyState>\n    )\n  }\n\n  const handleClick = (clickedOption: OptionType, group?: string) => {\n    setSelectedData({ type: 'selectOption', clickedOption, group })\n    if (multiselect) {\n      if (selectedData.selectedValues.includes(clickedOption.value)) {\n        onChange?.(\n          selectedData.selectedValues.filter(\n            val => val !== clickedOption.value,\n          ),\n        )\n      } else {\n        onChange?.([...selectedData.selectedValues, clickedOption.value])\n      }\n    } else {\n      onChange?.(clickedOption.value)\n    }\n    setIsDropdownVisible(multiselect) // hide the dropdown on click when single select only\n  }\n\n  const selectAllOptions = () => {\n    if (multiselect) {\n      setSelectedData({ type: 'selectAll' })\n      if (selectedData.allSelected && onChange) {\n        onChange([])\n      } else {\n        const allValues: OptionType[] = []\n        if (!Array.isArray(options)) {\n          Object.keys(options).map((group: string) =>\n            options[group].map(option => {\n              if (!option.disabled) {\n                allValues.push(option)\n              }\n\n              return null\n            }),\n          )\n        } else {\n          options.map(option => allValues.push(option))\n        }\n        onChange?.(allValues.map(value => value.value))\n      }\n    }\n  }\n\n  const handleSelectGroup = (group: string) => {\n    if (multiselect) {\n      setSelectedData({ type: 'selectGroup', selectedGroup: group })\n      if (!Array.isArray(options)) {\n        if (selectedData.selectedGroups.includes(group)) {\n          const newSelectedValues = [...selectedData.selectedValues].filter(\n            selectedValue =>\n              !options[group].find(option => option.value === selectedValue),\n          )\n          onChange?.(newSelectedValues)\n        } else {\n          const newSelectedValues = [...selectedData.selectedValues]\n\n          options[group].map(option =>\n            newSelectedValues.includes(option.value) || option.disabled\n              ? null\n              : newSelectedValues.push(option.value),\n          )\n          onChange?.(newSelectedValues)\n        }\n      }\n    }\n  }\n\n  return !Array.isArray(displayedOptions) ? (\n    <DropdownContainer\n      role=\"listbox\"\n      id=\"select-dropdown\"\n      onKeyDown={event => {\n        event.preventDefault()\n        handleKeyDownSelect(event.key)\n      }}\n      data-grouped\n    >\n      {isLoading ? (\n        <Skeleton variant=\"block\" />\n      ) : (\n        <>\n          {selectAll && multiselect ? (\n            <Stack id=\"items\">\n              <DropdownItem\n                disabled={false}\n                data-selected={selectedData.allSelected}\n                aria-label=\"select-all\"\n                data-testid=\"select-all\"\n                id=\"select-all\"\n                role=\"option\"\n                onKeyDown={event =>\n                  [' ', 'Enter'].includes(event.key) ? selectAllOptions() : null\n                }\n                onClick={selectAllOptions}\n              >\n                <StyledCheckbox\n                  checked={selectedData.allSelected}\n                  disabled={false}\n                  value=\"select-all\"\n                  data-testid=\"select-all-checkbox\"\n                  tabIndex={-1}\n                >\n                  <Stack direction=\"column\">\n                    <Text as=\"span\" variant=\"body\" placement=\"left\">\n                      {selectAll.label}\n                    </Text>\n                    <Text\n                      as=\"span\"\n                      variant=\"bodySmall\"\n                      sentiment=\"neutral\"\n                      placement=\"left\"\n                      prominence=\"weak\"\n                    >\n                      {selectAll.description}\n                    </Text>\n                  </Stack>\n                </StyledCheckbox>\n              </DropdownItem>\n            </Stack>\n          ) : null}\n          {Object.keys(displayedOptions).map((group, index) => (\n            <Stack key={group} gap={0.25}>\n              {displayedOptions[group].length > 0 ? (\n                <DropdownGroupWrapper id={selectAllGroup ? 'items' : undefined}>\n                  <DropdownGroup\n                    key={group}\n                    tabIndex={selectAllGroup ? 0 : -1}\n                    onKeyDown={event => {\n                      if ([' ', 'Enter'].includes(event.key)) {\n                        event.preventDefault()\n                        handleSelectGroup(group)\n                      }\n                    }}\n                    data-selectgroup={selectAllGroup}\n                    role=\"group\"\n                    data-testid={`group-${index}`}\n                    onClick={() =>\n                      selectAllGroup ? handleSelectGroup(group) : null\n                    }\n                  >\n                    {selectAllGroup ? (\n                      <StyledCheckbox\n                        checked={selectedData.selectedGroups.includes(group)}\n                        disabled={false}\n                        value={group}\n                        data-testid=\"select-group\"\n                        tabIndex={-1}\n                      >\n                        <Text variant=\"caption\" as=\"span\" placement=\"left\">\n                          {group.toUpperCase()}\n                        </Text>\n                      </StyledCheckbox>\n                    ) : (\n                      <Text variant=\"caption\" as=\"span\" placement=\"left\">\n                        {group.toUpperCase()}\n                      </Text>\n                    )}\n                  </DropdownGroup>\n                </DropdownGroupWrapper>\n              ) : null}\n              <Stack id=\"items\" gap=\"0.25\">\n                {displayedOptions[group].map((option, indexOption) => (\n                  <DropdownItem\n                    key={option.value}\n                    disabled={option.disabled}\n                    data-selected={\n                      selectedData.selectedValues.includes(option.value) &&\n                      !option.disabled\n                    }\n                    aria-label={option.value}\n                    data-testid={`option-${option.value}`}\n                    id={`option-${indexOption}`}\n                    role=\"option\"\n                    onClick={() => {\n                      if (!option.disabled) {\n                        handleClick(option, group)\n                      }\n                    }}\n                    onKeyDown={event =>\n                      [' ', 'Enter'].includes(event.key)\n                        ? handleClick(option, group)\n                        : null\n                    }\n                    ref={\n                      option.value === defaultSearchValue ||\n                      option.searchText === defaultSearchValue\n                        ? focusedItemRef\n                        : null\n                    }\n                  >\n                    {multiselect ? (\n                      <StyledCheckbox\n                        checked={\n                          selectedData.selectedValues.includes(option.value) &&\n                          !option.disabled\n                        }\n                        disabled={option.disabled}\n                        value={option.value}\n                        tabIndex={-1}\n                      >\n                        <DisplayOption\n                          option={option}\n                          descriptionDirection={descriptionDirection}\n                          optionalInfoPlacement={optionalInfoPlacement}\n                        />\n                      </StyledCheckbox>\n                    ) : (\n                      <DisplayOption\n                        option={option}\n                        descriptionDirection={descriptionDirection}\n                        optionalInfoPlacement={optionalInfoPlacement}\n                      />\n                    )}\n                  </DropdownItem>\n                ))}\n                {loadMore ? <LoadMore>{loadMore}</LoadMore> : null}\n              </Stack>\n            </Stack>\n          ))}\n        </>\n      )}\n    </DropdownContainer>\n  ) : (\n    <DropdownContainer\n      role=\"listbox\"\n      id=\"select-dropdown\"\n      onKeyDown={event => {\n        event.preventDefault()\n        handleKeyDownSelect(event.key)\n      }}\n      gap={0.25}\n      data-grouped={false}\n    >\n      {selectAll && multiselect ? (\n        <Stack id=\"items\" gap={0.25}>\n          <DropdownItem\n            disabled={false}\n            data-selected={selectedData.allSelected}\n            aria-label=\"select-all\"\n            data-testid=\"select-all\"\n            role=\"option\"\n            onKeyDown={event =>\n              [' ', 'Enter'].includes(event.key) ? selectAllOptions() : null\n            }\n            onClick={selectAllOptions}\n          >\n            <StyledCheckbox\n              checked={selectedData.allSelected}\n              disabled={false}\n              value=\"select-all\"\n              data-testid=\"select-all-checkbox\"\n              tabIndex={-1}\n            >\n              <Stack direction=\"column\">\n                <Text as=\"span\" variant=\"body\" placement=\"left\">\n                  {selectAll.label}\n                </Text>\n                <Text\n                  as=\"span\"\n                  variant=\"bodySmall\"\n                  sentiment=\"neutral\"\n                  placement=\"left\"\n                  prominence=\"weak\"\n                >\n                  {selectAll.description}\n                </Text>\n              </Stack>\n            </StyledCheckbox>\n          </DropdownItem>\n        </Stack>\n      ) : null}\n      <Stack id=\"items\" gap={0.25}>\n        {isLoading ? (\n          <Skeleton variant=\"block\" />\n        ) : (\n          displayedOptions.map((option, index) => (\n            <DropdownItem\n              key={option.value}\n              disabled={option.disabled}\n              data-selected={\n                selectedData.selectedValues.includes(option.value) &&\n                !option.disabled\n              }\n              onClick={() => {\n                if (!option.disabled) {\n                  handleClick(option)\n                }\n              }}\n              aria-label={option.value}\n              data-testid={`option-${option.value}`}\n              id={`option-${index}`}\n              role=\"option\"\n              ref={\n                option.value === defaultSearchValue ||\n                option.searchText === defaultSearchValue\n                  ? focusedItemRef\n                  : null\n              }\n              onKeyDown={event =>\n                [' ', 'Enter'].includes(event.key) ? handleClick(option) : null\n              }\n            >\n              {multiselect ? (\n                <StyledCheckbox\n                  checked={\n                    selectedData.selectedValues.includes(option.value) &&\n                    !option.disabled\n                  }\n                  disabled={option.disabled}\n                  value={option.value}\n                  tabIndex={-1}\n                >\n                  <DisplayOption\n                    option={option}\n                    descriptionDirection={descriptionDirection}\n                    optionalInfoPlacement={optionalInfoPlacement}\n                  />\n                </StyledCheckbox>\n              ) : (\n                <DisplayOption\n                  option={option}\n                  descriptionDirection={descriptionDirection}\n                  optionalInfoPlacement={optionalInfoPlacement}\n                />\n              )}\n            </DropdownItem>\n          ))\n        )}\n        {loadMore ? <LoadMore>{loadMore}</LoadMore> : null}\n      </Stack>\n    </DropdownContainer>\n  )\n}\nexport const Dropdown = ({\n  children,\n  emptyState,\n  descriptionDirection,\n  searchable,\n  placeholder,\n  footer,\n  refSelect,\n  loadMore,\n  optionalInfoPlacement,\n  isLoading,\n}: DropdownProps) => {\n  const {\n    setIsDropdownVisible,\n    isDropdownVisible,\n    onSearch,\n    searchInput,\n    options,\n    displayedOptions,\n  } = useSelectInput()\n  const [searchBarActive, setSearchBarActive] = useState(false)\n  const [defaultSearchValue, setDefaultSearch] = useState<string | null>(null)\n  const maxWidth = refSelect.current?.offsetWidth\n  const ref = useRef<HTMLDivElement>(null)\n  const [search, setSearch] = useState<string>('')\n  useEffect(() => {\n    if (!searchInput) {\n      onSearch(options)\n    }\n  }, [onSearch, options, searchInput])\n\n  useEffect(() => {\n    if (!isDropdownVisible) {\n      setDefaultSearch(null)\n      setSearch('')\n    }\n\n    document.addEventListener('mousedown', event =>\n      handleClickOutside(\n        event,\n        ref,\n        setIsDropdownVisible,\n        refSelect,\n        onSearch,\n        options,\n      ),\n    )\n\n    if (!searchable) {\n      document.addEventListener('keydown', event =>\n        handleKeyDown(\n          event,\n          ref,\n          options,\n          searchBarActive,\n          setSearch,\n          setDefaultSearch,\n          search,\n        ),\n      )\n    }\n\n    return () => {\n      document.removeEventListener('mousedown', event =>\n        handleClickOutside(\n          event,\n          ref,\n          setIsDropdownVisible,\n          refSelect,\n          onSearch,\n          options,\n        ),\n      )\n      if (!searchable) {\n        document.removeEventListener('keydown', event =>\n          handleKeyDown(\n            event,\n            ref,\n            options,\n            searchBarActive,\n            setSearch,\n            setDefaultSearch,\n            search,\n          ),\n        )\n      }\n    }\n  }, [\n    isDropdownVisible,\n    searchBarActive,\n    options,\n    onSearch,\n    search,\n    refSelect,\n    setDefaultSearch,\n    setIsDropdownVisible,\n    searchable,\n  ])\n\n  const isEmpty = useMemo(() => {\n    if (Array.isArray(displayedOptions)) {\n      return !(displayedOptions.length > 0)\n    }\n    const groups = Object.keys(displayedOptions)\n    for (const group of groups) {\n      if (displayedOptions[group].length !== 0) {\n        return false\n      }\n    }\n\n    return true\n  }, [displayedOptions])\n\n  return (\n    <StyledPopup\n      visible={isDropdownVisible}\n      text={\n        <Stack>\n          {searchable && !isLoading ? (\n            <SearchBarDropdown\n              placeholder={placeholder}\n              displayedOptions={displayedOptions}\n              setSearchBarActive={setSearchBarActive}\n            />\n          ) : null}\n          <CreateDropdown\n            isEmpty={isEmpty}\n            emptyState={emptyState}\n            descriptionDirection={descriptionDirection}\n            loadMore={loadMore}\n            optionalInfoPlacement={optionalInfoPlacement}\n            defaultSearchValue={defaultSearchValue}\n            isLoading={isLoading}\n          />\n          {footer ? <PopupFooter>{footer}</PopupFooter> : null}\n        </Stack>\n      }\n      placement=\"bottom\"\n      containerFullWidth\n      disableAnimation\n      maxWidth={maxWidth}\n      hasArrow={false}\n      ref={ref}\n      tabIndex={0}\n      role=\"dialog\"\n      debounceDelay={0}\n    >\n      {children}\n    </StyledPopup>\n  )\n}\n"]} */"));
122
+ const PopupFooter = /* @__PURE__ */ _styled("div", process.env.NODE_ENV === "production" ? {
123
+ target: "eeucz6c3"
124
+ } : {
125
+ target: "eeucz6c3",
126
+ label: "PopupFooter"
127
+ })("width:100%;padding:", ({
128
+ theme
129
+ }) => theme.space[1.5], " ", ({
130
+ theme
131
+ }) => theme.space[2], " ", ({
132
+ theme
133
+ }) => theme.space[1.5], " ", ({
134
+ theme
135
+ }) => theme.space[2], ";box-shadow:", ({
136
+ theme
137
+ }) => theme.shadows.dropdown, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/Dropdown.tsx"],"names":[],"mappings":"AA+I8B","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/Dropdown.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { useEffect, useMemo, useRef, useState } from 'react'\nimport type { Dispatch, ReactNode, RefObject, SetStateAction } from 'react'\nimport { Checkbox } from '../Checkbox'\nimport { Popup } from '../Popup'\nimport { Skeleton } from '../Skeleton'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\nimport { DisplayOption } from './DropdownOption'\nimport { SearchBarDropdown } from './SearchBarDropdown'\nimport { useSelectInput } from './SelectInputProvider'\nimport type { DataType, OptionType } from './types'\n\nexport type DropdownProps = {\n  children: ReactNode\n  emptyState: ReactNode\n  descriptionDirection: 'row' | 'column'\n  searchable: boolean\n  placeholder: string\n  footer?: ReactNode\n  refSelect: RefObject<HTMLDivElement>\n  loadMore?: ReactNode\n  optionalInfoPlacement: 'left' | 'right'\n  isLoading?: boolean\n}\n\nexport type CreateDropdownProps = {\n  isEmpty: boolean\n  emptyState: ReactNode\n  descriptionDirection: 'row' | 'column'\n  loadMore?: ReactNode\n  optionalInfoPlacement: 'left' | 'right'\n  defaultSearchValue: string | null\n  isLoading?: boolean\n}\n\nconst NON_SEARCHABLE_KEYS = [\n  'Tab',\n  ' ',\n  'Enter',\n  'CapsLock',\n  'Shift',\n  'ArrowDown',\n  'ArrowUp',\n  'ArrowLeft',\n  'ArrowRight',\n  'Escape',\n]\n\nconst StyledPopup = styled(Popup)`\n  width: 100%;\n  background-color: ${({ theme }) => theme.colors.neutral.background};\n  color: ${({ theme }) => theme.colors.neutral.text};\n  box-shadow: ${({ theme }) => theme.shadows.dropdown};\n  padding: ${({ theme }) => theme.space[0]};\n`\n\nconst DropdownContainer = styled(Stack)<{ 'data-grouped': boolean }>`\n  max-height: 256px;\n  overflow-y: scroll;\n  padding: ${({ theme }) => theme.space[0]};\n  padding-bottom: ${({ theme }) => theme.space[0.5]};\n  padding-top: ${({ theme }) => theme.space[0.5]};\n\n  &[data-grouped='true'] {\n    padding-top: ${({ theme }) => theme.space[0]};\n  }\n`\nconst DropdownGroup = styled.button<{ 'data-selectgroup': boolean }>`\n  display: flex;\n  width: 100%;\n  justify-content: left;\n  align-items: center;\n  border: none;\n  background-color: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n  position: sticky;\n  top: 0px;\n  padding-right: ${({ theme }) => theme.space[2]};\n  padding-left: ${({ theme }) => theme.space[2]};\n  height: ${({ theme }) => theme.space[4]};\n  text-align: left;\n  margin-bottom: ${({ theme }) => theme.space['0.25']};\n\n  &:focus {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n    outline: none;\n  }\n\n  &[data-selectgroup='true'] {\n    padding-left: ${({ theme }) => theme.space[2]};\n    border-left: ${({ theme }) => theme.space[0.5]} solid ${({ theme }) => theme.colors.neutral.backgroundWeak};\n  }\n\n  &[data-selectgroup='true']:focus {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundHover};\n  }\n}\n`\nconst DropdownGroupWrapper = styled.div`\n  position: sticky;\n  top: 0px;\n`\nconst DropdownItem = styled.button<{\n  'data-selected': boolean\n  disabled: boolean\n}>`\n  text-align:left;\n  border: none;\n  background-color: ${({ theme }) => theme.colors.neutral.background};\n\n  padding: ${({ theme }) => theme.space['1.5']} ${({ theme }) => theme.space['2']} ${({ theme }) => theme.space['1.5']} ${({ theme }) => theme.space['2']};\n  margin-left: ${({ theme }) => theme.space['0.5']};\n  margin-right: ${({ theme }) => theme.space['0.5']};\n\n  color:  ${({ theme }) => theme.colors.neutral.text};\n  border-radius: ${({ theme }) => theme.radii.default};\n\n  &:hover, :focus {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n    color: ${({ theme }) => theme.colors.primary.text};\n    cursor: pointer;\n    outline: none;\n  }\n\n  &[data-selected='true'] {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n  }\n\n  &[disabled] {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n\n  }\n\n  &[disabled]:hover, [disabled]:focus {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundStrongDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textStrongDisabled};\n    cursor: not-allowed;\n    outline: none;\n  }\n\n  }\n`\nconst PopupFooter = styled.div`\n  width: 100%;\n  padding: ${({ theme }) => theme.space[1.5]} ${({ theme }) => theme.space[2]}\n    ${({ theme }) => theme.space[1.5]} ${({ theme }) => theme.space[2]};\n  box-shadow: ${({ theme }) => theme.shadows.dropdown};\n`\nconst StyledCheckbox = styled(Checkbox)`\n  width: 100%;\n  position: static;\n  text-align: left;\n  align-items: center;\n  pointer-events: none;\n` // pointer-events: none prevents any error when using the checkbox in a form since it is an unnamed input\n\nconst EmptyState = styled(Stack)`\n  padding: ${({ theme }) => theme.space[2]};\n`\nconst LoadMore = styled(Stack)`\n  padding: ${({ theme }) => theme.space[0.5]};\n`\n\nconst moveFocusDown = () => {\n  const options = document.querySelectorAll(\n    '#items > button[role=\"option\"]:not([disabled])',\n  )\n  const activeItem = document.activeElement\n  if (options) {\n    for (let i = 0; i < options?.length; i += 1) {\n      const listLength = options.length\n      if (activeItem === options[i] && activeItem !== options[listLength - 1]) {\n        ;(options[i + 1] as HTMLElement).focus()\n      }\n    }\n  }\n}\nconst moveFocusUp = () => {\n  const options = document.querySelectorAll(\n    '#items > button[role=\"option\"]:not([disabled])',\n  )\n  const activeItem = document.activeElement\n\n  if (options) {\n    for (let i = 0; i < options.length; i += 1) {\n      if (activeItem === options[i] && activeItem !== options[0]) {\n        ;(options[i - 1] as HTMLElement).focus()\n      }\n    }\n  }\n}\nconst handleKeyDownSelect = (key: string) => {\n  if (key === 'ArrowDown') {\n    moveFocusDown()\n  }\n  if (key === 'ArrowUp') {\n    moveFocusUp()\n  }\n}\nconst handleClickOutside = (\n  event: MouseEvent,\n  ref: RefObject<HTMLDivElement>,\n  setIsDropdownVisibile: Dispatch<SetStateAction<boolean>>,\n  refSelect: RefObject<HTMLDivElement>,\n  onSearch: Dispatch<SetStateAction<DataType>>,\n  options: DataType,\n) => {\n  if (\n    ref.current &&\n    !ref.current.contains(event.target as Node) &&\n    !refSelect.current?.contains(event.target as Node)\n  ) {\n    setIsDropdownVisibile(false) // hide dropdown when clicking outside of the dropdown\n    onSearch(options) // reset displayed options to default when dropdown is hidden\n  }\n}\n\nconst handleKeyDown = (\n  event: KeyboardEvent,\n  ref: RefObject<HTMLDivElement>,\n  options: DataType,\n  searchBarActive: boolean,\n  setSearch: Dispatch<SetStateAction<string>>,\n  setDefaultSearch: Dispatch<SetStateAction<string | null>>,\n  search: string,\n) => {\n  // Deals with default search\n  if (\n    ref.current &&\n    !searchBarActive &&\n    !NON_SEARCHABLE_KEYS.includes(event.key) &&\n    document.activeElement?.ariaLabel !== 'search-bar'\n  ) {\n    const currentSearch = search + event.key\n    setSearch(currentSearch)\n    ref.current.focus()\n    if (!Array.isArray(options)) {\n      const closestOptions = { ...options }\n      Object.keys(closestOptions).map((group: string) => {\n        closestOptions[group] = closestOptions[group].filter(option =>\n          option.searchText\n            ? option.searchText.toLocaleLowerCase().startsWith(currentSearch)\n            : option.value.toLocaleLowerCase().startsWith(currentSearch),\n        )\n\n        return null\n      })\n      const closestOption = closestOptions[Object.keys(closestOptions)[0]][0]\n      if (closestOption) {\n        setDefaultSearch(closestOption.searchText ?? closestOption.value)\n      } else {\n        setDefaultSearch(null)\n      }\n    } else {\n      const closestOption = [...options].filter(option =>\n        option.searchText\n          ? option.searchText.toLocaleLowerCase().startsWith(currentSearch)\n          : option.value.toLocaleLowerCase().startsWith(currentSearch),\n      )[0]\n      if (closestOption) {\n        setDefaultSearch(closestOption.searchText ?? closestOption.value)\n      } else {\n        setDefaultSearch(null)\n      }\n    }\n  }\n}\nconst CreateDropdown = ({\n  isEmpty,\n  emptyState,\n  descriptionDirection,\n  loadMore,\n  optionalInfoPlacement,\n  defaultSearchValue,\n  isLoading,\n}: CreateDropdownProps) => {\n  const {\n    setIsDropdownVisible,\n    onChange,\n    options,\n    multiselect,\n    selectAll,\n    selectAllGroup,\n    displayedOptions,\n    setSelectedData,\n    selectedData,\n  } = useSelectInput()\n  const focusedItemRef = useRef<HTMLButtonElement>(null)\n\n  useEffect(() => {\n    if (defaultSearchValue && focusedItemRef?.current) {\n      focusedItemRef.current.focus()\n    }\n  }, [defaultSearchValue])\n\n  if (isEmpty) {\n    return (\n      <EmptyState gap={2} alignItems=\"center\">\n        {emptyState ?? (\n          <Text variant=\"bodyStrong\" as=\"p\">\n            No options\n          </Text>\n        )}\n      </EmptyState>\n    )\n  }\n\n  const handleClick = (clickedOption: OptionType, group?: string) => {\n    setSelectedData({ type: 'selectOption', clickedOption, group })\n    if (multiselect) {\n      if (selectedData.selectedValues.includes(clickedOption.value)) {\n        onChange?.(\n          selectedData.selectedValues.filter(\n            val => val !== clickedOption.value,\n          ),\n        )\n      } else {\n        onChange?.([...selectedData.selectedValues, clickedOption.value])\n      }\n    } else {\n      onChange?.(clickedOption.value)\n    }\n    setIsDropdownVisible(multiselect) // hide the dropdown on click when single select only\n  }\n\n  const selectAllOptions = () => {\n    if (multiselect) {\n      setSelectedData({ type: 'selectAll' })\n      if (selectedData.allSelected && onChange) {\n        onChange([])\n      } else {\n        const allValues: OptionType[] = []\n        if (!Array.isArray(options)) {\n          Object.keys(options).map((group: string) =>\n            options[group].map(option => {\n              if (!option.disabled) {\n                allValues.push(option)\n              }\n\n              return null\n            }),\n          )\n        } else {\n          options.map(option => allValues.push(option))\n        }\n        onChange?.(allValues.map(value => value.value))\n      }\n    }\n  }\n\n  const handleSelectGroup = (group: string) => {\n    if (multiselect) {\n      setSelectedData({ type: 'selectGroup', selectedGroup: group })\n      if (!Array.isArray(options)) {\n        if (selectedData.selectedGroups.includes(group)) {\n          const newSelectedValues = [...selectedData.selectedValues].filter(\n            selectedValue =>\n              !options[group].find(option => option.value === selectedValue),\n          )\n          onChange?.(newSelectedValues)\n        } else {\n          const newSelectedValues = [...selectedData.selectedValues]\n\n          options[group].map(option =>\n            newSelectedValues.includes(option.value) || option.disabled\n              ? null\n              : newSelectedValues.push(option.value),\n          )\n          onChange?.(newSelectedValues)\n        }\n      }\n    }\n  }\n\n  return !Array.isArray(displayedOptions) ? (\n    <DropdownContainer\n      role=\"listbox\"\n      id=\"select-dropdown\"\n      onKeyDown={event => {\n        event.preventDefault()\n        handleKeyDownSelect(event.key)\n      }}\n      data-grouped\n    >\n      {isLoading ? (\n        <Skeleton variant=\"block\" />\n      ) : (\n        <>\n          {selectAll && multiselect ? (\n            <Stack id=\"items\">\n              <DropdownItem\n                disabled={false}\n                data-selected={selectedData.allSelected}\n                aria-label=\"select-all\"\n                data-testid=\"select-all\"\n                id=\"select-all\"\n                role=\"option\"\n                onKeyDown={event =>\n                  [' ', 'Enter'].includes(event.key) ? selectAllOptions() : null\n                }\n                onClick={selectAllOptions}\n              >\n                <StyledCheckbox\n                  checked={selectedData.allSelected}\n                  disabled={false}\n                  value=\"select-all\"\n                  data-testid=\"select-all-checkbox\"\n                  tabIndex={-1}\n                >\n                  <Stack direction=\"column\">\n                    <Text as=\"span\" variant=\"body\" placement=\"left\">\n                      {selectAll.label}\n                    </Text>\n                    <Text\n                      as=\"span\"\n                      variant=\"bodySmall\"\n                      sentiment=\"neutral\"\n                      placement=\"left\"\n                      prominence=\"weak\"\n                    >\n                      {selectAll.description}\n                    </Text>\n                  </Stack>\n                </StyledCheckbox>\n              </DropdownItem>\n            </Stack>\n          ) : null}\n          {Object.keys(displayedOptions).map((group, index) => (\n            <Stack key={group} gap={0.25}>\n              {displayedOptions[group].length > 0 ? (\n                <DropdownGroupWrapper id={selectAllGroup ? 'items' : undefined}>\n                  <DropdownGroup\n                    key={group}\n                    tabIndex={selectAllGroup ? 0 : -1}\n                    onKeyDown={event => {\n                      if ([' ', 'Enter'].includes(event.key)) {\n                        event.preventDefault()\n                        handleSelectGroup(group)\n                      }\n                    }}\n                    data-selectgroup={selectAllGroup}\n                    role=\"group\"\n                    data-testid={`group-${index}`}\n                    onClick={() =>\n                      selectAllGroup ? handleSelectGroup(group) : null\n                    }\n                  >\n                    {selectAllGroup ? (\n                      <StyledCheckbox\n                        checked={selectedData.selectedGroups.includes(group)}\n                        disabled={false}\n                        value={group}\n                        data-testid=\"select-group\"\n                        tabIndex={-1}\n                      >\n                        <Text variant=\"caption\" as=\"span\" placement=\"left\">\n                          {group.toUpperCase()}\n                        </Text>\n                      </StyledCheckbox>\n                    ) : (\n                      <Text variant=\"caption\" as=\"span\" placement=\"left\">\n                        {group.toUpperCase()}\n                      </Text>\n                    )}\n                  </DropdownGroup>\n                </DropdownGroupWrapper>\n              ) : null}\n              <Stack id=\"items\" gap=\"0.25\">\n                {displayedOptions[group].map((option, indexOption) => (\n                  <DropdownItem\n                    key={option.value}\n                    disabled={option.disabled}\n                    data-selected={\n                      selectedData.selectedValues.includes(option.value) &&\n                      !option.disabled\n                    }\n                    aria-label={option.value}\n                    data-testid={`option-${option.value}`}\n                    id={`option-${indexOption}`}\n                    role=\"option\"\n                    onClick={() => {\n                      if (!option.disabled) {\n                        handleClick(option, group)\n                      }\n                    }}\n                    onKeyDown={event =>\n                      [' ', 'Enter'].includes(event.key)\n                        ? handleClick(option, group)\n                        : null\n                    }\n                    ref={\n                      option.value === defaultSearchValue ||\n                      option.searchText === defaultSearchValue\n                        ? focusedItemRef\n                        : null\n                    }\n                  >\n                    {multiselect ? (\n                      <StyledCheckbox\n                        checked={\n                          selectedData.selectedValues.includes(option.value) &&\n                          !option.disabled\n                        }\n                        disabled={option.disabled}\n                        value={option.value}\n                        tabIndex={-1}\n                      >\n                        <DisplayOption\n                          option={option}\n                          descriptionDirection={descriptionDirection}\n                          optionalInfoPlacement={optionalInfoPlacement}\n                        />\n                      </StyledCheckbox>\n                    ) : (\n                      <DisplayOption\n                        option={option}\n                        descriptionDirection={descriptionDirection}\n                        optionalInfoPlacement={optionalInfoPlacement}\n                      />\n                    )}\n                  </DropdownItem>\n                ))}\n                {loadMore ? <LoadMore>{loadMore}</LoadMore> : null}\n              </Stack>\n            </Stack>\n          ))}\n        </>\n      )}\n    </DropdownContainer>\n  ) : (\n    <DropdownContainer\n      role=\"listbox\"\n      id=\"select-dropdown\"\n      onKeyDown={event => {\n        event.preventDefault()\n        handleKeyDownSelect(event.key)\n      }}\n      gap={0.25}\n      data-grouped={false}\n    >\n      {selectAll && multiselect ? (\n        <Stack id=\"items\" gap={0.25}>\n          <DropdownItem\n            disabled={false}\n            data-selected={selectedData.allSelected}\n            aria-label=\"select-all\"\n            data-testid=\"select-all\"\n            role=\"option\"\n            onKeyDown={event =>\n              [' ', 'Enter'].includes(event.key) ? selectAllOptions() : null\n            }\n            onClick={selectAllOptions}\n          >\n            <StyledCheckbox\n              checked={selectedData.allSelected}\n              disabled={false}\n              value=\"select-all\"\n              data-testid=\"select-all-checkbox\"\n              tabIndex={-1}\n            >\n              <Stack direction=\"column\">\n                <Text as=\"span\" variant=\"body\" placement=\"left\">\n                  {selectAll.label}\n                </Text>\n                <Text\n                  as=\"span\"\n                  variant=\"bodySmall\"\n                  sentiment=\"neutral\"\n                  placement=\"left\"\n                  prominence=\"weak\"\n                >\n                  {selectAll.description}\n                </Text>\n              </Stack>\n            </StyledCheckbox>\n          </DropdownItem>\n        </Stack>\n      ) : null}\n      <Stack id=\"items\" gap={0.25}>\n        {isLoading ? (\n          <Skeleton variant=\"block\" />\n        ) : (\n          displayedOptions.map((option, index) => (\n            <DropdownItem\n              key={option.value}\n              disabled={option.disabled}\n              data-selected={\n                selectedData.selectedValues.includes(option.value) &&\n                !option.disabled\n              }\n              onClick={() => {\n                if (!option.disabled) {\n                  handleClick(option)\n                }\n              }}\n              aria-label={option.value}\n              data-testid={`option-${option.value}`}\n              id={`option-${index}`}\n              role=\"option\"\n              ref={\n                option.value === defaultSearchValue ||\n                option.searchText === defaultSearchValue\n                  ? focusedItemRef\n                  : null\n              }\n              onKeyDown={event =>\n                [' ', 'Enter'].includes(event.key) ? handleClick(option) : null\n              }\n            >\n              {multiselect ? (\n                <StyledCheckbox\n                  checked={\n                    selectedData.selectedValues.includes(option.value) &&\n                    !option.disabled\n                  }\n                  disabled={option.disabled}\n                  value={option.value}\n                  tabIndex={-1}\n                >\n                  <DisplayOption\n                    option={option}\n                    descriptionDirection={descriptionDirection}\n                    optionalInfoPlacement={optionalInfoPlacement}\n                  />\n                </StyledCheckbox>\n              ) : (\n                <DisplayOption\n                  option={option}\n                  descriptionDirection={descriptionDirection}\n                  optionalInfoPlacement={optionalInfoPlacement}\n                />\n              )}\n            </DropdownItem>\n          ))\n        )}\n        {loadMore ? <LoadMore>{loadMore}</LoadMore> : null}\n      </Stack>\n    </DropdownContainer>\n  )\n}\nexport const Dropdown = ({\n  children,\n  emptyState,\n  descriptionDirection,\n  searchable,\n  placeholder,\n  footer,\n  refSelect,\n  loadMore,\n  optionalInfoPlacement,\n  isLoading,\n}: DropdownProps) => {\n  const {\n    setIsDropdownVisible,\n    isDropdownVisible,\n    onSearch,\n    searchInput,\n    options,\n    displayedOptions,\n  } = useSelectInput()\n  const [searchBarActive, setSearchBarActive] = useState(false)\n  const [defaultSearchValue, setDefaultSearch] = useState<string | null>(null)\n  const maxWidth = refSelect.current?.offsetWidth\n  const ref = useRef<HTMLDivElement>(null)\n  const [search, setSearch] = useState<string>('')\n  useEffect(() => {\n    if (!searchInput) {\n      onSearch(options)\n    }\n  }, [onSearch, options, searchInput])\n\n  useEffect(() => {\n    if (!isDropdownVisible) {\n      setDefaultSearch(null)\n      setSearch('')\n    }\n\n    document.addEventListener('mousedown', event =>\n      handleClickOutside(\n        event,\n        ref,\n        setIsDropdownVisible,\n        refSelect,\n        onSearch,\n        options,\n      ),\n    )\n\n    if (!searchable) {\n      document.addEventListener('keydown', event =>\n        handleKeyDown(\n          event,\n          ref,\n          options,\n          searchBarActive,\n          setSearch,\n          setDefaultSearch,\n          search,\n        ),\n      )\n    }\n\n    return () => {\n      document.removeEventListener('mousedown', event =>\n        handleClickOutside(\n          event,\n          ref,\n          setIsDropdownVisible,\n          refSelect,\n          onSearch,\n          options,\n        ),\n      )\n      if (!searchable) {\n        document.removeEventListener('keydown', event =>\n          handleKeyDown(\n            event,\n            ref,\n            options,\n            searchBarActive,\n            setSearch,\n            setDefaultSearch,\n            search,\n          ),\n        )\n      }\n    }\n  }, [\n    isDropdownVisible,\n    searchBarActive,\n    options,\n    onSearch,\n    search,\n    refSelect,\n    setDefaultSearch,\n    setIsDropdownVisible,\n    searchable,\n  ])\n\n  const isEmpty = useMemo(() => {\n    if (Array.isArray(displayedOptions)) {\n      return !(displayedOptions.length > 0)\n    }\n    const groups = Object.keys(displayedOptions)\n    for (const group of groups) {\n      if (displayedOptions[group].length !== 0) {\n        return false\n      }\n    }\n\n    return true\n  }, [displayedOptions])\n\n  return (\n    <StyledPopup\n      visible={isDropdownVisible}\n      text={\n        <Stack>\n          {searchable && !isLoading ? (\n            <SearchBarDropdown\n              placeholder={placeholder}\n              displayedOptions={displayedOptions}\n              setSearchBarActive={setSearchBarActive}\n            />\n          ) : null}\n          <CreateDropdown\n            isEmpty={isEmpty}\n            emptyState={emptyState}\n            descriptionDirection={descriptionDirection}\n            loadMore={loadMore}\n            optionalInfoPlacement={optionalInfoPlacement}\n            defaultSearchValue={defaultSearchValue}\n            isLoading={isLoading}\n          />\n          {footer ? <PopupFooter>{footer}</PopupFooter> : null}\n        </Stack>\n      }\n      placement=\"bottom\"\n      containerFullWidth\n      disableAnimation\n      maxWidth={maxWidth}\n      hasArrow={false}\n      ref={ref}\n      tabIndex={0}\n      role=\"dialog\"\n      debounceDelay={0}\n    >\n      {children}\n    </StyledPopup>\n  )\n}\n"]} */"));
138
+ const StyledCheckbox = /* @__PURE__ */ _styled(Checkbox, process.env.NODE_ENV === "production" ? {
139
+ target: "eeucz6c2"
140
+ } : {
141
+ target: "eeucz6c2",
142
+ label: "StyledCheckbox"
143
+ })(process.env.NODE_ENV === "production" ? {
144
+ name: "1l9xw77",
145
+ styles: "width:100%;position:static;text-align:left;align-items:center;pointer-events:none"
146
+ } : {
147
+ name: "1l9xw77",
148
+ styles: "width:100%;position:static;text-align:left;align-items:center;pointer-events:none",
149
+ map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/Dropdown.tsx"],"names":[],"mappings":"AAqJuC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/Dropdown.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { useEffect, useMemo, useRef, useState } from 'react'\nimport type { Dispatch, ReactNode, RefObject, SetStateAction } from 'react'\nimport { Checkbox } from '../Checkbox'\nimport { Popup } from '../Popup'\nimport { Skeleton } from '../Skeleton'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\nimport { DisplayOption } from './DropdownOption'\nimport { SearchBarDropdown } from './SearchBarDropdown'\nimport { useSelectInput } from './SelectInputProvider'\nimport type { DataType, OptionType } from './types'\n\nexport type DropdownProps = {\n  children: ReactNode\n  emptyState: ReactNode\n  descriptionDirection: 'row' | 'column'\n  searchable: boolean\n  placeholder: string\n  footer?: ReactNode\n  refSelect: RefObject<HTMLDivElement>\n  loadMore?: ReactNode\n  optionalInfoPlacement: 'left' | 'right'\n  isLoading?: boolean\n}\n\nexport type CreateDropdownProps = {\n  isEmpty: boolean\n  emptyState: ReactNode\n  descriptionDirection: 'row' | 'column'\n  loadMore?: ReactNode\n  optionalInfoPlacement: 'left' | 'right'\n  defaultSearchValue: string | null\n  isLoading?: boolean\n}\n\nconst NON_SEARCHABLE_KEYS = [\n  'Tab',\n  ' ',\n  'Enter',\n  'CapsLock',\n  'Shift',\n  'ArrowDown',\n  'ArrowUp',\n  'ArrowLeft',\n  'ArrowRight',\n  'Escape',\n]\n\nconst StyledPopup = styled(Popup)`\n  width: 100%;\n  background-color: ${({ theme }) => theme.colors.neutral.background};\n  color: ${({ theme }) => theme.colors.neutral.text};\n  box-shadow: ${({ theme }) => theme.shadows.dropdown};\n  padding: ${({ theme }) => theme.space[0]};\n`\n\nconst DropdownContainer = styled(Stack)<{ 'data-grouped': boolean }>`\n  max-height: 256px;\n  overflow-y: scroll;\n  padding: ${({ theme }) => theme.space[0]};\n  padding-bottom: ${({ theme }) => theme.space[0.5]};\n  padding-top: ${({ theme }) => theme.space[0.5]};\n\n  &[data-grouped='true'] {\n    padding-top: ${({ theme }) => theme.space[0]};\n  }\n`\nconst DropdownGroup = styled.button<{ 'data-selectgroup': boolean }>`\n  display: flex;\n  width: 100%;\n  justify-content: left;\n  align-items: center;\n  border: none;\n  background-color: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n  position: sticky;\n  top: 0px;\n  padding-right: ${({ theme }) => theme.space[2]};\n  padding-left: ${({ theme }) => theme.space[2]};\n  height: ${({ theme }) => theme.space[4]};\n  text-align: left;\n  margin-bottom: ${({ theme }) => theme.space['0.25']};\n\n  &:focus {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n    outline: none;\n  }\n\n  &[data-selectgroup='true'] {\n    padding-left: ${({ theme }) => theme.space[2]};\n    border-left: ${({ theme }) => theme.space[0.5]} solid ${({ theme }) => theme.colors.neutral.backgroundWeak};\n  }\n\n  &[data-selectgroup='true']:focus {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundHover};\n  }\n}\n`\nconst DropdownGroupWrapper = styled.div`\n  position: sticky;\n  top: 0px;\n`\nconst DropdownItem = styled.button<{\n  'data-selected': boolean\n  disabled: boolean\n}>`\n  text-align:left;\n  border: none;\n  background-color: ${({ theme }) => theme.colors.neutral.background};\n\n  padding: ${({ theme }) => theme.space['1.5']} ${({ theme }) => theme.space['2']} ${({ theme }) => theme.space['1.5']} ${({ theme }) => theme.space['2']};\n  margin-left: ${({ theme }) => theme.space['0.5']};\n  margin-right: ${({ theme }) => theme.space['0.5']};\n\n  color:  ${({ theme }) => theme.colors.neutral.text};\n  border-radius: ${({ theme }) => theme.radii.default};\n\n  &:hover, :focus {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n    color: ${({ theme }) => theme.colors.primary.text};\n    cursor: pointer;\n    outline: none;\n  }\n\n  &[data-selected='true'] {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n  }\n\n  &[disabled] {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n\n  }\n\n  &[disabled]:hover, [disabled]:focus {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundStrongDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textStrongDisabled};\n    cursor: not-allowed;\n    outline: none;\n  }\n\n  }\n`\nconst PopupFooter = styled.div`\n  width: 100%;\n  padding: ${({ theme }) => theme.space[1.5]} ${({ theme }) => theme.space[2]}\n    ${({ theme }) => theme.space[1.5]} ${({ theme }) => theme.space[2]};\n  box-shadow: ${({ theme }) => theme.shadows.dropdown};\n`\nconst StyledCheckbox = styled(Checkbox)`\n  width: 100%;\n  position: static;\n  text-align: left;\n  align-items: center;\n  pointer-events: none;\n` // pointer-events: none prevents any error when using the checkbox in a form since it is an unnamed input\n\nconst EmptyState = styled(Stack)`\n  padding: ${({ theme }) => theme.space[2]};\n`\nconst LoadMore = styled(Stack)`\n  padding: ${({ theme }) => theme.space[0.5]};\n`\n\nconst moveFocusDown = () => {\n  const options = document.querySelectorAll(\n    '#items > button[role=\"option\"]:not([disabled])',\n  )\n  const activeItem = document.activeElement\n  if (options) {\n    for (let i = 0; i < options?.length; i += 1) {\n      const listLength = options.length\n      if (activeItem === options[i] && activeItem !== options[listLength - 1]) {\n        ;(options[i + 1] as HTMLElement).focus()\n      }\n    }\n  }\n}\nconst moveFocusUp = () => {\n  const options = document.querySelectorAll(\n    '#items > button[role=\"option\"]:not([disabled])',\n  )\n  const activeItem = document.activeElement\n\n  if (options) {\n    for (let i = 0; i < options.length; i += 1) {\n      if (activeItem === options[i] && activeItem !== options[0]) {\n        ;(options[i - 1] as HTMLElement).focus()\n      }\n    }\n  }\n}\nconst handleKeyDownSelect = (key: string) => {\n  if (key === 'ArrowDown') {\n    moveFocusDown()\n  }\n  if (key === 'ArrowUp') {\n    moveFocusUp()\n  }\n}\nconst handleClickOutside = (\n  event: MouseEvent,\n  ref: RefObject<HTMLDivElement>,\n  setIsDropdownVisibile: Dispatch<SetStateAction<boolean>>,\n  refSelect: RefObject<HTMLDivElement>,\n  onSearch: Dispatch<SetStateAction<DataType>>,\n  options: DataType,\n) => {\n  if (\n    ref.current &&\n    !ref.current.contains(event.target as Node) &&\n    !refSelect.current?.contains(event.target as Node)\n  ) {\n    setIsDropdownVisibile(false) // hide dropdown when clicking outside of the dropdown\n    onSearch(options) // reset displayed options to default when dropdown is hidden\n  }\n}\n\nconst handleKeyDown = (\n  event: KeyboardEvent,\n  ref: RefObject<HTMLDivElement>,\n  options: DataType,\n  searchBarActive: boolean,\n  setSearch: Dispatch<SetStateAction<string>>,\n  setDefaultSearch: Dispatch<SetStateAction<string | null>>,\n  search: string,\n) => {\n  // Deals with default search\n  if (\n    ref.current &&\n    !searchBarActive &&\n    !NON_SEARCHABLE_KEYS.includes(event.key) &&\n    document.activeElement?.ariaLabel !== 'search-bar'\n  ) {\n    const currentSearch = search + event.key\n    setSearch(currentSearch)\n    ref.current.focus()\n    if (!Array.isArray(options)) {\n      const closestOptions = { ...options }\n      Object.keys(closestOptions).map((group: string) => {\n        closestOptions[group] = closestOptions[group].filter(option =>\n          option.searchText\n            ? option.searchText.toLocaleLowerCase().startsWith(currentSearch)\n            : option.value.toLocaleLowerCase().startsWith(currentSearch),\n        )\n\n        return null\n      })\n      const closestOption = closestOptions[Object.keys(closestOptions)[0]][0]\n      if (closestOption) {\n        setDefaultSearch(closestOption.searchText ?? closestOption.value)\n      } else {\n        setDefaultSearch(null)\n      }\n    } else {\n      const closestOption = [...options].filter(option =>\n        option.searchText\n          ? option.searchText.toLocaleLowerCase().startsWith(currentSearch)\n          : option.value.toLocaleLowerCase().startsWith(currentSearch),\n      )[0]\n      if (closestOption) {\n        setDefaultSearch(closestOption.searchText ?? closestOption.value)\n      } else {\n        setDefaultSearch(null)\n      }\n    }\n  }\n}\nconst CreateDropdown = ({\n  isEmpty,\n  emptyState,\n  descriptionDirection,\n  loadMore,\n  optionalInfoPlacement,\n  defaultSearchValue,\n  isLoading,\n}: CreateDropdownProps) => {\n  const {\n    setIsDropdownVisible,\n    onChange,\n    options,\n    multiselect,\n    selectAll,\n    selectAllGroup,\n    displayedOptions,\n    setSelectedData,\n    selectedData,\n  } = useSelectInput()\n  const focusedItemRef = useRef<HTMLButtonElement>(null)\n\n  useEffect(() => {\n    if (defaultSearchValue && focusedItemRef?.current) {\n      focusedItemRef.current.focus()\n    }\n  }, [defaultSearchValue])\n\n  if (isEmpty) {\n    return (\n      <EmptyState gap={2} alignItems=\"center\">\n        {emptyState ?? (\n          <Text variant=\"bodyStrong\" as=\"p\">\n            No options\n          </Text>\n        )}\n      </EmptyState>\n    )\n  }\n\n  const handleClick = (clickedOption: OptionType, group?: string) => {\n    setSelectedData({ type: 'selectOption', clickedOption, group })\n    if (multiselect) {\n      if (selectedData.selectedValues.includes(clickedOption.value)) {\n        onChange?.(\n          selectedData.selectedValues.filter(\n            val => val !== clickedOption.value,\n          ),\n        )\n      } else {\n        onChange?.([...selectedData.selectedValues, clickedOption.value])\n      }\n    } else {\n      onChange?.(clickedOption.value)\n    }\n    setIsDropdownVisible(multiselect) // hide the dropdown on click when single select only\n  }\n\n  const selectAllOptions = () => {\n    if (multiselect) {\n      setSelectedData({ type: 'selectAll' })\n      if (selectedData.allSelected && onChange) {\n        onChange([])\n      } else {\n        const allValues: OptionType[] = []\n        if (!Array.isArray(options)) {\n          Object.keys(options).map((group: string) =>\n            options[group].map(option => {\n              if (!option.disabled) {\n                allValues.push(option)\n              }\n\n              return null\n            }),\n          )\n        } else {\n          options.map(option => allValues.push(option))\n        }\n        onChange?.(allValues.map(value => value.value))\n      }\n    }\n  }\n\n  const handleSelectGroup = (group: string) => {\n    if (multiselect) {\n      setSelectedData({ type: 'selectGroup', selectedGroup: group })\n      if (!Array.isArray(options)) {\n        if (selectedData.selectedGroups.includes(group)) {\n          const newSelectedValues = [...selectedData.selectedValues].filter(\n            selectedValue =>\n              !options[group].find(option => option.value === selectedValue),\n          )\n          onChange?.(newSelectedValues)\n        } else {\n          const newSelectedValues = [...selectedData.selectedValues]\n\n          options[group].map(option =>\n            newSelectedValues.includes(option.value) || option.disabled\n              ? null\n              : newSelectedValues.push(option.value),\n          )\n          onChange?.(newSelectedValues)\n        }\n      }\n    }\n  }\n\n  return !Array.isArray(displayedOptions) ? (\n    <DropdownContainer\n      role=\"listbox\"\n      id=\"select-dropdown\"\n      onKeyDown={event => {\n        event.preventDefault()\n        handleKeyDownSelect(event.key)\n      }}\n      data-grouped\n    >\n      {isLoading ? (\n        <Skeleton variant=\"block\" />\n      ) : (\n        <>\n          {selectAll && multiselect ? (\n            <Stack id=\"items\">\n              <DropdownItem\n                disabled={false}\n                data-selected={selectedData.allSelected}\n                aria-label=\"select-all\"\n                data-testid=\"select-all\"\n                id=\"select-all\"\n                role=\"option\"\n                onKeyDown={event =>\n                  [' ', 'Enter'].includes(event.key) ? selectAllOptions() : null\n                }\n                onClick={selectAllOptions}\n              >\n                <StyledCheckbox\n                  checked={selectedData.allSelected}\n                  disabled={false}\n                  value=\"select-all\"\n                  data-testid=\"select-all-checkbox\"\n                  tabIndex={-1}\n                >\n                  <Stack direction=\"column\">\n                    <Text as=\"span\" variant=\"body\" placement=\"left\">\n                      {selectAll.label}\n                    </Text>\n                    <Text\n                      as=\"span\"\n                      variant=\"bodySmall\"\n                      sentiment=\"neutral\"\n                      placement=\"left\"\n                      prominence=\"weak\"\n                    >\n                      {selectAll.description}\n                    </Text>\n                  </Stack>\n                </StyledCheckbox>\n              </DropdownItem>\n            </Stack>\n          ) : null}\n          {Object.keys(displayedOptions).map((group, index) => (\n            <Stack key={group} gap={0.25}>\n              {displayedOptions[group].length > 0 ? (\n                <DropdownGroupWrapper id={selectAllGroup ? 'items' : undefined}>\n                  <DropdownGroup\n                    key={group}\n                    tabIndex={selectAllGroup ? 0 : -1}\n                    onKeyDown={event => {\n                      if ([' ', 'Enter'].includes(event.key)) {\n                        event.preventDefault()\n                        handleSelectGroup(group)\n                      }\n                    }}\n                    data-selectgroup={selectAllGroup}\n                    role=\"group\"\n                    data-testid={`group-${index}`}\n                    onClick={() =>\n                      selectAllGroup ? handleSelectGroup(group) : null\n                    }\n                  >\n                    {selectAllGroup ? (\n                      <StyledCheckbox\n                        checked={selectedData.selectedGroups.includes(group)}\n                        disabled={false}\n                        value={group}\n                        data-testid=\"select-group\"\n                        tabIndex={-1}\n                      >\n                        <Text variant=\"caption\" as=\"span\" placement=\"left\">\n                          {group.toUpperCase()}\n                        </Text>\n                      </StyledCheckbox>\n                    ) : (\n                      <Text variant=\"caption\" as=\"span\" placement=\"left\">\n                        {group.toUpperCase()}\n                      </Text>\n                    )}\n                  </DropdownGroup>\n                </DropdownGroupWrapper>\n              ) : null}\n              <Stack id=\"items\" gap=\"0.25\">\n                {displayedOptions[group].map((option, indexOption) => (\n                  <DropdownItem\n                    key={option.value}\n                    disabled={option.disabled}\n                    data-selected={\n                      selectedData.selectedValues.includes(option.value) &&\n                      !option.disabled\n                    }\n                    aria-label={option.value}\n                    data-testid={`option-${option.value}`}\n                    id={`option-${indexOption}`}\n                    role=\"option\"\n                    onClick={() => {\n                      if (!option.disabled) {\n                        handleClick(option, group)\n                      }\n                    }}\n                    onKeyDown={event =>\n                      [' ', 'Enter'].includes(event.key)\n                        ? handleClick(option, group)\n                        : null\n                    }\n                    ref={\n                      option.value === defaultSearchValue ||\n                      option.searchText === defaultSearchValue\n                        ? focusedItemRef\n                        : null\n                    }\n                  >\n                    {multiselect ? (\n                      <StyledCheckbox\n                        checked={\n                          selectedData.selectedValues.includes(option.value) &&\n                          !option.disabled\n                        }\n                        disabled={option.disabled}\n                        value={option.value}\n                        tabIndex={-1}\n                      >\n                        <DisplayOption\n                          option={option}\n                          descriptionDirection={descriptionDirection}\n                          optionalInfoPlacement={optionalInfoPlacement}\n                        />\n                      </StyledCheckbox>\n                    ) : (\n                      <DisplayOption\n                        option={option}\n                        descriptionDirection={descriptionDirection}\n                        optionalInfoPlacement={optionalInfoPlacement}\n                      />\n                    )}\n                  </DropdownItem>\n                ))}\n                {loadMore ? <LoadMore>{loadMore}</LoadMore> : null}\n              </Stack>\n            </Stack>\n          ))}\n        </>\n      )}\n    </DropdownContainer>\n  ) : (\n    <DropdownContainer\n      role=\"listbox\"\n      id=\"select-dropdown\"\n      onKeyDown={event => {\n        event.preventDefault()\n        handleKeyDownSelect(event.key)\n      }}\n      gap={0.25}\n      data-grouped={false}\n    >\n      {selectAll && multiselect ? (\n        <Stack id=\"items\" gap={0.25}>\n          <DropdownItem\n            disabled={false}\n            data-selected={selectedData.allSelected}\n            aria-label=\"select-all\"\n            data-testid=\"select-all\"\n            role=\"option\"\n            onKeyDown={event =>\n              [' ', 'Enter'].includes(event.key) ? selectAllOptions() : null\n            }\n            onClick={selectAllOptions}\n          >\n            <StyledCheckbox\n              checked={selectedData.allSelected}\n              disabled={false}\n              value=\"select-all\"\n              data-testid=\"select-all-checkbox\"\n              tabIndex={-1}\n            >\n              <Stack direction=\"column\">\n                <Text as=\"span\" variant=\"body\" placement=\"left\">\n                  {selectAll.label}\n                </Text>\n                <Text\n                  as=\"span\"\n                  variant=\"bodySmall\"\n                  sentiment=\"neutral\"\n                  placement=\"left\"\n                  prominence=\"weak\"\n                >\n                  {selectAll.description}\n                </Text>\n              </Stack>\n            </StyledCheckbox>\n          </DropdownItem>\n        </Stack>\n      ) : null}\n      <Stack id=\"items\" gap={0.25}>\n        {isLoading ? (\n          <Skeleton variant=\"block\" />\n        ) : (\n          displayedOptions.map((option, index) => (\n            <DropdownItem\n              key={option.value}\n              disabled={option.disabled}\n              data-selected={\n                selectedData.selectedValues.includes(option.value) &&\n                !option.disabled\n              }\n              onClick={() => {\n                if (!option.disabled) {\n                  handleClick(option)\n                }\n              }}\n              aria-label={option.value}\n              data-testid={`option-${option.value}`}\n              id={`option-${index}`}\n              role=\"option\"\n              ref={\n                option.value === defaultSearchValue ||\n                option.searchText === defaultSearchValue\n                  ? focusedItemRef\n                  : null\n              }\n              onKeyDown={event =>\n                [' ', 'Enter'].includes(event.key) ? handleClick(option) : null\n              }\n            >\n              {multiselect ? (\n                <StyledCheckbox\n                  checked={\n                    selectedData.selectedValues.includes(option.value) &&\n                    !option.disabled\n                  }\n                  disabled={option.disabled}\n                  value={option.value}\n                  tabIndex={-1}\n                >\n                  <DisplayOption\n                    option={option}\n                    descriptionDirection={descriptionDirection}\n                    optionalInfoPlacement={optionalInfoPlacement}\n                  />\n                </StyledCheckbox>\n              ) : (\n                <DisplayOption\n                  option={option}\n                  descriptionDirection={descriptionDirection}\n                  optionalInfoPlacement={optionalInfoPlacement}\n                />\n              )}\n            </DropdownItem>\n          ))\n        )}\n        {loadMore ? <LoadMore>{loadMore}</LoadMore> : null}\n      </Stack>\n    </DropdownContainer>\n  )\n}\nexport const Dropdown = ({\n  children,\n  emptyState,\n  descriptionDirection,\n  searchable,\n  placeholder,\n  footer,\n  refSelect,\n  loadMore,\n  optionalInfoPlacement,\n  isLoading,\n}: DropdownProps) => {\n  const {\n    setIsDropdownVisible,\n    isDropdownVisible,\n    onSearch,\n    searchInput,\n    options,\n    displayedOptions,\n  } = useSelectInput()\n  const [searchBarActive, setSearchBarActive] = useState(false)\n  const [defaultSearchValue, setDefaultSearch] = useState<string | null>(null)\n  const maxWidth = refSelect.current?.offsetWidth\n  const ref = useRef<HTMLDivElement>(null)\n  const [search, setSearch] = useState<string>('')\n  useEffect(() => {\n    if (!searchInput) {\n      onSearch(options)\n    }\n  }, [onSearch, options, searchInput])\n\n  useEffect(() => {\n    if (!isDropdownVisible) {\n      setDefaultSearch(null)\n      setSearch('')\n    }\n\n    document.addEventListener('mousedown', event =>\n      handleClickOutside(\n        event,\n        ref,\n        setIsDropdownVisible,\n        refSelect,\n        onSearch,\n        options,\n      ),\n    )\n\n    if (!searchable) {\n      document.addEventListener('keydown', event =>\n        handleKeyDown(\n          event,\n          ref,\n          options,\n          searchBarActive,\n          setSearch,\n          setDefaultSearch,\n          search,\n        ),\n      )\n    }\n\n    return () => {\n      document.removeEventListener('mousedown', event =>\n        handleClickOutside(\n          event,\n          ref,\n          setIsDropdownVisible,\n          refSelect,\n          onSearch,\n          options,\n        ),\n      )\n      if (!searchable) {\n        document.removeEventListener('keydown', event =>\n          handleKeyDown(\n            event,\n            ref,\n            options,\n            searchBarActive,\n            setSearch,\n            setDefaultSearch,\n            search,\n          ),\n        )\n      }\n    }\n  }, [\n    isDropdownVisible,\n    searchBarActive,\n    options,\n    onSearch,\n    search,\n    refSelect,\n    setDefaultSearch,\n    setIsDropdownVisible,\n    searchable,\n  ])\n\n  const isEmpty = useMemo(() => {\n    if (Array.isArray(displayedOptions)) {\n      return !(displayedOptions.length > 0)\n    }\n    const groups = Object.keys(displayedOptions)\n    for (const group of groups) {\n      if (displayedOptions[group].length !== 0) {\n        return false\n      }\n    }\n\n    return true\n  }, [displayedOptions])\n\n  return (\n    <StyledPopup\n      visible={isDropdownVisible}\n      text={\n        <Stack>\n          {searchable && !isLoading ? (\n            <SearchBarDropdown\n              placeholder={placeholder}\n              displayedOptions={displayedOptions}\n              setSearchBarActive={setSearchBarActive}\n            />\n          ) : null}\n          <CreateDropdown\n            isEmpty={isEmpty}\n            emptyState={emptyState}\n            descriptionDirection={descriptionDirection}\n            loadMore={loadMore}\n            optionalInfoPlacement={optionalInfoPlacement}\n            defaultSearchValue={defaultSearchValue}\n            isLoading={isLoading}\n          />\n          {footer ? <PopupFooter>{footer}</PopupFooter> : null}\n        </Stack>\n      }\n      placement=\"bottom\"\n      containerFullWidth\n      disableAnimation\n      maxWidth={maxWidth}\n      hasArrow={false}\n      ref={ref}\n      tabIndex={0}\n      role=\"dialog\"\n      debounceDelay={0}\n    >\n      {children}\n    </StyledPopup>\n  )\n}\n"]} */",
150
+ toString: _EMOTION_STRINGIFIED_CSS_ERROR__
151
+ });
152
+ const EmptyState = /* @__PURE__ */ _styled(Stack, process.env.NODE_ENV === "production" ? {
153
+ target: "eeucz6c1"
154
+ } : {
155
+ target: "eeucz6c1",
156
+ label: "EmptyState"
157
+ })("padding:", ({
158
+ theme
159
+ }) => theme.space[2], ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/Dropdown.tsx"],"names":[],"mappings":"AA6JgC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/Dropdown.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { useEffect, useMemo, useRef, useState } from 'react'\nimport type { Dispatch, ReactNode, RefObject, SetStateAction } from 'react'\nimport { Checkbox } from '../Checkbox'\nimport { Popup } from '../Popup'\nimport { Skeleton } from '../Skeleton'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\nimport { DisplayOption } from './DropdownOption'\nimport { SearchBarDropdown } from './SearchBarDropdown'\nimport { useSelectInput } from './SelectInputProvider'\nimport type { DataType, OptionType } from './types'\n\nexport type DropdownProps = {\n  children: ReactNode\n  emptyState: ReactNode\n  descriptionDirection: 'row' | 'column'\n  searchable: boolean\n  placeholder: string\n  footer?: ReactNode\n  refSelect: RefObject<HTMLDivElement>\n  loadMore?: ReactNode\n  optionalInfoPlacement: 'left' | 'right'\n  isLoading?: boolean\n}\n\nexport type CreateDropdownProps = {\n  isEmpty: boolean\n  emptyState: ReactNode\n  descriptionDirection: 'row' | 'column'\n  loadMore?: ReactNode\n  optionalInfoPlacement: 'left' | 'right'\n  defaultSearchValue: string | null\n  isLoading?: boolean\n}\n\nconst NON_SEARCHABLE_KEYS = [\n  'Tab',\n  ' ',\n  'Enter',\n  'CapsLock',\n  'Shift',\n  'ArrowDown',\n  'ArrowUp',\n  'ArrowLeft',\n  'ArrowRight',\n  'Escape',\n]\n\nconst StyledPopup = styled(Popup)`\n  width: 100%;\n  background-color: ${({ theme }) => theme.colors.neutral.background};\n  color: ${({ theme }) => theme.colors.neutral.text};\n  box-shadow: ${({ theme }) => theme.shadows.dropdown};\n  padding: ${({ theme }) => theme.space[0]};\n`\n\nconst DropdownContainer = styled(Stack)<{ 'data-grouped': boolean }>`\n  max-height: 256px;\n  overflow-y: scroll;\n  padding: ${({ theme }) => theme.space[0]};\n  padding-bottom: ${({ theme }) => theme.space[0.5]};\n  padding-top: ${({ theme }) => theme.space[0.5]};\n\n  &[data-grouped='true'] {\n    padding-top: ${({ theme }) => theme.space[0]};\n  }\n`\nconst DropdownGroup = styled.button<{ 'data-selectgroup': boolean }>`\n  display: flex;\n  width: 100%;\n  justify-content: left;\n  align-items: center;\n  border: none;\n  background-color: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n  position: sticky;\n  top: 0px;\n  padding-right: ${({ theme }) => theme.space[2]};\n  padding-left: ${({ theme }) => theme.space[2]};\n  height: ${({ theme }) => theme.space[4]};\n  text-align: left;\n  margin-bottom: ${({ theme }) => theme.space['0.25']};\n\n  &:focus {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n    outline: none;\n  }\n\n  &[data-selectgroup='true'] {\n    padding-left: ${({ theme }) => theme.space[2]};\n    border-left: ${({ theme }) => theme.space[0.5]} solid ${({ theme }) => theme.colors.neutral.backgroundWeak};\n  }\n\n  &[data-selectgroup='true']:focus {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundHover};\n  }\n}\n`\nconst DropdownGroupWrapper = styled.div`\n  position: sticky;\n  top: 0px;\n`\nconst DropdownItem = styled.button<{\n  'data-selected': boolean\n  disabled: boolean\n}>`\n  text-align:left;\n  border: none;\n  background-color: ${({ theme }) => theme.colors.neutral.background};\n\n  padding: ${({ theme }) => theme.space['1.5']} ${({ theme }) => theme.space['2']} ${({ theme }) => theme.space['1.5']} ${({ theme }) => theme.space['2']};\n  margin-left: ${({ theme }) => theme.space['0.5']};\n  margin-right: ${({ theme }) => theme.space['0.5']};\n\n  color:  ${({ theme }) => theme.colors.neutral.text};\n  border-radius: ${({ theme }) => theme.radii.default};\n\n  &:hover, :focus {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n    color: ${({ theme }) => theme.colors.primary.text};\n    cursor: pointer;\n    outline: none;\n  }\n\n  &[data-selected='true'] {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n  }\n\n  &[disabled] {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n\n  }\n\n  &[disabled]:hover, [disabled]:focus {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundStrongDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textStrongDisabled};\n    cursor: not-allowed;\n    outline: none;\n  }\n\n  }\n`\nconst PopupFooter = styled.div`\n  width: 100%;\n  padding: ${({ theme }) => theme.space[1.5]} ${({ theme }) => theme.space[2]}\n    ${({ theme }) => theme.space[1.5]} ${({ theme }) => theme.space[2]};\n  box-shadow: ${({ theme }) => theme.shadows.dropdown};\n`\nconst StyledCheckbox = styled(Checkbox)`\n  width: 100%;\n  position: static;\n  text-align: left;\n  align-items: center;\n  pointer-events: none;\n` // pointer-events: none prevents any error when using the checkbox in a form since it is an unnamed input\n\nconst EmptyState = styled(Stack)`\n  padding: ${({ theme }) => theme.space[2]};\n`\nconst LoadMore = styled(Stack)`\n  padding: ${({ theme }) => theme.space[0.5]};\n`\n\nconst moveFocusDown = () => {\n  const options = document.querySelectorAll(\n    '#items > button[role=\"option\"]:not([disabled])',\n  )\n  const activeItem = document.activeElement\n  if (options) {\n    for (let i = 0; i < options?.length; i += 1) {\n      const listLength = options.length\n      if (activeItem === options[i] && activeItem !== options[listLength - 1]) {\n        ;(options[i + 1] as HTMLElement).focus()\n      }\n    }\n  }\n}\nconst moveFocusUp = () => {\n  const options = document.querySelectorAll(\n    '#items > button[role=\"option\"]:not([disabled])',\n  )\n  const activeItem = document.activeElement\n\n  if (options) {\n    for (let i = 0; i < options.length; i += 1) {\n      if (activeItem === options[i] && activeItem !== options[0]) {\n        ;(options[i - 1] as HTMLElement).focus()\n      }\n    }\n  }\n}\nconst handleKeyDownSelect = (key: string) => {\n  if (key === 'ArrowDown') {\n    moveFocusDown()\n  }\n  if (key === 'ArrowUp') {\n    moveFocusUp()\n  }\n}\nconst handleClickOutside = (\n  event: MouseEvent,\n  ref: RefObject<HTMLDivElement>,\n  setIsDropdownVisibile: Dispatch<SetStateAction<boolean>>,\n  refSelect: RefObject<HTMLDivElement>,\n  onSearch: Dispatch<SetStateAction<DataType>>,\n  options: DataType,\n) => {\n  if (\n    ref.current &&\n    !ref.current.contains(event.target as Node) &&\n    !refSelect.current?.contains(event.target as Node)\n  ) {\n    setIsDropdownVisibile(false) // hide dropdown when clicking outside of the dropdown\n    onSearch(options) // reset displayed options to default when dropdown is hidden\n  }\n}\n\nconst handleKeyDown = (\n  event: KeyboardEvent,\n  ref: RefObject<HTMLDivElement>,\n  options: DataType,\n  searchBarActive: boolean,\n  setSearch: Dispatch<SetStateAction<string>>,\n  setDefaultSearch: Dispatch<SetStateAction<string | null>>,\n  search: string,\n) => {\n  // Deals with default search\n  if (\n    ref.current &&\n    !searchBarActive &&\n    !NON_SEARCHABLE_KEYS.includes(event.key) &&\n    document.activeElement?.ariaLabel !== 'search-bar'\n  ) {\n    const currentSearch = search + event.key\n    setSearch(currentSearch)\n    ref.current.focus()\n    if (!Array.isArray(options)) {\n      const closestOptions = { ...options }\n      Object.keys(closestOptions).map((group: string) => {\n        closestOptions[group] = closestOptions[group].filter(option =>\n          option.searchText\n            ? option.searchText.toLocaleLowerCase().startsWith(currentSearch)\n            : option.value.toLocaleLowerCase().startsWith(currentSearch),\n        )\n\n        return null\n      })\n      const closestOption = closestOptions[Object.keys(closestOptions)[0]][0]\n      if (closestOption) {\n        setDefaultSearch(closestOption.searchText ?? closestOption.value)\n      } else {\n        setDefaultSearch(null)\n      }\n    } else {\n      const closestOption = [...options].filter(option =>\n        option.searchText\n          ? option.searchText.toLocaleLowerCase().startsWith(currentSearch)\n          : option.value.toLocaleLowerCase().startsWith(currentSearch),\n      )[0]\n      if (closestOption) {\n        setDefaultSearch(closestOption.searchText ?? closestOption.value)\n      } else {\n        setDefaultSearch(null)\n      }\n    }\n  }\n}\nconst CreateDropdown = ({\n  isEmpty,\n  emptyState,\n  descriptionDirection,\n  loadMore,\n  optionalInfoPlacement,\n  defaultSearchValue,\n  isLoading,\n}: CreateDropdownProps) => {\n  const {\n    setIsDropdownVisible,\n    onChange,\n    options,\n    multiselect,\n    selectAll,\n    selectAllGroup,\n    displayedOptions,\n    setSelectedData,\n    selectedData,\n  } = useSelectInput()\n  const focusedItemRef = useRef<HTMLButtonElement>(null)\n\n  useEffect(() => {\n    if (defaultSearchValue && focusedItemRef?.current) {\n      focusedItemRef.current.focus()\n    }\n  }, [defaultSearchValue])\n\n  if (isEmpty) {\n    return (\n      <EmptyState gap={2} alignItems=\"center\">\n        {emptyState ?? (\n          <Text variant=\"bodyStrong\" as=\"p\">\n            No options\n          </Text>\n        )}\n      </EmptyState>\n    )\n  }\n\n  const handleClick = (clickedOption: OptionType, group?: string) => {\n    setSelectedData({ type: 'selectOption', clickedOption, group })\n    if (multiselect) {\n      if (selectedData.selectedValues.includes(clickedOption.value)) {\n        onChange?.(\n          selectedData.selectedValues.filter(\n            val => val !== clickedOption.value,\n          ),\n        )\n      } else {\n        onChange?.([...selectedData.selectedValues, clickedOption.value])\n      }\n    } else {\n      onChange?.(clickedOption.value)\n    }\n    setIsDropdownVisible(multiselect) // hide the dropdown on click when single select only\n  }\n\n  const selectAllOptions = () => {\n    if (multiselect) {\n      setSelectedData({ type: 'selectAll' })\n      if (selectedData.allSelected && onChange) {\n        onChange([])\n      } else {\n        const allValues: OptionType[] = []\n        if (!Array.isArray(options)) {\n          Object.keys(options).map((group: string) =>\n            options[group].map(option => {\n              if (!option.disabled) {\n                allValues.push(option)\n              }\n\n              return null\n            }),\n          )\n        } else {\n          options.map(option => allValues.push(option))\n        }\n        onChange?.(allValues.map(value => value.value))\n      }\n    }\n  }\n\n  const handleSelectGroup = (group: string) => {\n    if (multiselect) {\n      setSelectedData({ type: 'selectGroup', selectedGroup: group })\n      if (!Array.isArray(options)) {\n        if (selectedData.selectedGroups.includes(group)) {\n          const newSelectedValues = [...selectedData.selectedValues].filter(\n            selectedValue =>\n              !options[group].find(option => option.value === selectedValue),\n          )\n          onChange?.(newSelectedValues)\n        } else {\n          const newSelectedValues = [...selectedData.selectedValues]\n\n          options[group].map(option =>\n            newSelectedValues.includes(option.value) || option.disabled\n              ? null\n              : newSelectedValues.push(option.value),\n          )\n          onChange?.(newSelectedValues)\n        }\n      }\n    }\n  }\n\n  return !Array.isArray(displayedOptions) ? (\n    <DropdownContainer\n      role=\"listbox\"\n      id=\"select-dropdown\"\n      onKeyDown={event => {\n        event.preventDefault()\n        handleKeyDownSelect(event.key)\n      }}\n      data-grouped\n    >\n      {isLoading ? (\n        <Skeleton variant=\"block\" />\n      ) : (\n        <>\n          {selectAll && multiselect ? (\n            <Stack id=\"items\">\n              <DropdownItem\n                disabled={false}\n                data-selected={selectedData.allSelected}\n                aria-label=\"select-all\"\n                data-testid=\"select-all\"\n                id=\"select-all\"\n                role=\"option\"\n                onKeyDown={event =>\n                  [' ', 'Enter'].includes(event.key) ? selectAllOptions() : null\n                }\n                onClick={selectAllOptions}\n              >\n                <StyledCheckbox\n                  checked={selectedData.allSelected}\n                  disabled={false}\n                  value=\"select-all\"\n                  data-testid=\"select-all-checkbox\"\n                  tabIndex={-1}\n                >\n                  <Stack direction=\"column\">\n                    <Text as=\"span\" variant=\"body\" placement=\"left\">\n                      {selectAll.label}\n                    </Text>\n                    <Text\n                      as=\"span\"\n                      variant=\"bodySmall\"\n                      sentiment=\"neutral\"\n                      placement=\"left\"\n                      prominence=\"weak\"\n                    >\n                      {selectAll.description}\n                    </Text>\n                  </Stack>\n                </StyledCheckbox>\n              </DropdownItem>\n            </Stack>\n          ) : null}\n          {Object.keys(displayedOptions).map((group, index) => (\n            <Stack key={group} gap={0.25}>\n              {displayedOptions[group].length > 0 ? (\n                <DropdownGroupWrapper id={selectAllGroup ? 'items' : undefined}>\n                  <DropdownGroup\n                    key={group}\n                    tabIndex={selectAllGroup ? 0 : -1}\n                    onKeyDown={event => {\n                      if ([' ', 'Enter'].includes(event.key)) {\n                        event.preventDefault()\n                        handleSelectGroup(group)\n                      }\n                    }}\n                    data-selectgroup={selectAllGroup}\n                    role=\"group\"\n                    data-testid={`group-${index}`}\n                    onClick={() =>\n                      selectAllGroup ? handleSelectGroup(group) : null\n                    }\n                  >\n                    {selectAllGroup ? (\n                      <StyledCheckbox\n                        checked={selectedData.selectedGroups.includes(group)}\n                        disabled={false}\n                        value={group}\n                        data-testid=\"select-group\"\n                        tabIndex={-1}\n                      >\n                        <Text variant=\"caption\" as=\"span\" placement=\"left\">\n                          {group.toUpperCase()}\n                        </Text>\n                      </StyledCheckbox>\n                    ) : (\n                      <Text variant=\"caption\" as=\"span\" placement=\"left\">\n                        {group.toUpperCase()}\n                      </Text>\n                    )}\n                  </DropdownGroup>\n                </DropdownGroupWrapper>\n              ) : null}\n              <Stack id=\"items\" gap=\"0.25\">\n                {displayedOptions[group].map((option, indexOption) => (\n                  <DropdownItem\n                    key={option.value}\n                    disabled={option.disabled}\n                    data-selected={\n                      selectedData.selectedValues.includes(option.value) &&\n                      !option.disabled\n                    }\n                    aria-label={option.value}\n                    data-testid={`option-${option.value}`}\n                    id={`option-${indexOption}`}\n                    role=\"option\"\n                    onClick={() => {\n                      if (!option.disabled) {\n                        handleClick(option, group)\n                      }\n                    }}\n                    onKeyDown={event =>\n                      [' ', 'Enter'].includes(event.key)\n                        ? handleClick(option, group)\n                        : null\n                    }\n                    ref={\n                      option.value === defaultSearchValue ||\n                      option.searchText === defaultSearchValue\n                        ? focusedItemRef\n                        : null\n                    }\n                  >\n                    {multiselect ? (\n                      <StyledCheckbox\n                        checked={\n                          selectedData.selectedValues.includes(option.value) &&\n                          !option.disabled\n                        }\n                        disabled={option.disabled}\n                        value={option.value}\n                        tabIndex={-1}\n                      >\n                        <DisplayOption\n                          option={option}\n                          descriptionDirection={descriptionDirection}\n                          optionalInfoPlacement={optionalInfoPlacement}\n                        />\n                      </StyledCheckbox>\n                    ) : (\n                      <DisplayOption\n                        option={option}\n                        descriptionDirection={descriptionDirection}\n                        optionalInfoPlacement={optionalInfoPlacement}\n                      />\n                    )}\n                  </DropdownItem>\n                ))}\n                {loadMore ? <LoadMore>{loadMore}</LoadMore> : null}\n              </Stack>\n            </Stack>\n          ))}\n        </>\n      )}\n    </DropdownContainer>\n  ) : (\n    <DropdownContainer\n      role=\"listbox\"\n      id=\"select-dropdown\"\n      onKeyDown={event => {\n        event.preventDefault()\n        handleKeyDownSelect(event.key)\n      }}\n      gap={0.25}\n      data-grouped={false}\n    >\n      {selectAll && multiselect ? (\n        <Stack id=\"items\" gap={0.25}>\n          <DropdownItem\n            disabled={false}\n            data-selected={selectedData.allSelected}\n            aria-label=\"select-all\"\n            data-testid=\"select-all\"\n            role=\"option\"\n            onKeyDown={event =>\n              [' ', 'Enter'].includes(event.key) ? selectAllOptions() : null\n            }\n            onClick={selectAllOptions}\n          >\n            <StyledCheckbox\n              checked={selectedData.allSelected}\n              disabled={false}\n              value=\"select-all\"\n              data-testid=\"select-all-checkbox\"\n              tabIndex={-1}\n            >\n              <Stack direction=\"column\">\n                <Text as=\"span\" variant=\"body\" placement=\"left\">\n                  {selectAll.label}\n                </Text>\n                <Text\n                  as=\"span\"\n                  variant=\"bodySmall\"\n                  sentiment=\"neutral\"\n                  placement=\"left\"\n                  prominence=\"weak\"\n                >\n                  {selectAll.description}\n                </Text>\n              </Stack>\n            </StyledCheckbox>\n          </DropdownItem>\n        </Stack>\n      ) : null}\n      <Stack id=\"items\" gap={0.25}>\n        {isLoading ? (\n          <Skeleton variant=\"block\" />\n        ) : (\n          displayedOptions.map((option, index) => (\n            <DropdownItem\n              key={option.value}\n              disabled={option.disabled}\n              data-selected={\n                selectedData.selectedValues.includes(option.value) &&\n                !option.disabled\n              }\n              onClick={() => {\n                if (!option.disabled) {\n                  handleClick(option)\n                }\n              }}\n              aria-label={option.value}\n              data-testid={`option-${option.value}`}\n              id={`option-${index}`}\n              role=\"option\"\n              ref={\n                option.value === defaultSearchValue ||\n                option.searchText === defaultSearchValue\n                  ? focusedItemRef\n                  : null\n              }\n              onKeyDown={event =>\n                [' ', 'Enter'].includes(event.key) ? handleClick(option) : null\n              }\n            >\n              {multiselect ? (\n                <StyledCheckbox\n                  checked={\n                    selectedData.selectedValues.includes(option.value) &&\n                    !option.disabled\n                  }\n                  disabled={option.disabled}\n                  value={option.value}\n                  tabIndex={-1}\n                >\n                  <DisplayOption\n                    option={option}\n                    descriptionDirection={descriptionDirection}\n                    optionalInfoPlacement={optionalInfoPlacement}\n                  />\n                </StyledCheckbox>\n              ) : (\n                <DisplayOption\n                  option={option}\n                  descriptionDirection={descriptionDirection}\n                  optionalInfoPlacement={optionalInfoPlacement}\n                />\n              )}\n            </DropdownItem>\n          ))\n        )}\n        {loadMore ? <LoadMore>{loadMore}</LoadMore> : null}\n      </Stack>\n    </DropdownContainer>\n  )\n}\nexport const Dropdown = ({\n  children,\n  emptyState,\n  descriptionDirection,\n  searchable,\n  placeholder,\n  footer,\n  refSelect,\n  loadMore,\n  optionalInfoPlacement,\n  isLoading,\n}: DropdownProps) => {\n  const {\n    setIsDropdownVisible,\n    isDropdownVisible,\n    onSearch,\n    searchInput,\n    options,\n    displayedOptions,\n  } = useSelectInput()\n  const [searchBarActive, setSearchBarActive] = useState(false)\n  const [defaultSearchValue, setDefaultSearch] = useState<string | null>(null)\n  const maxWidth = refSelect.current?.offsetWidth\n  const ref = useRef<HTMLDivElement>(null)\n  const [search, setSearch] = useState<string>('')\n  useEffect(() => {\n    if (!searchInput) {\n      onSearch(options)\n    }\n  }, [onSearch, options, searchInput])\n\n  useEffect(() => {\n    if (!isDropdownVisible) {\n      setDefaultSearch(null)\n      setSearch('')\n    }\n\n    document.addEventListener('mousedown', event =>\n      handleClickOutside(\n        event,\n        ref,\n        setIsDropdownVisible,\n        refSelect,\n        onSearch,\n        options,\n      ),\n    )\n\n    if (!searchable) {\n      document.addEventListener('keydown', event =>\n        handleKeyDown(\n          event,\n          ref,\n          options,\n          searchBarActive,\n          setSearch,\n          setDefaultSearch,\n          search,\n        ),\n      )\n    }\n\n    return () => {\n      document.removeEventListener('mousedown', event =>\n        handleClickOutside(\n          event,\n          ref,\n          setIsDropdownVisible,\n          refSelect,\n          onSearch,\n          options,\n        ),\n      )\n      if (!searchable) {\n        document.removeEventListener('keydown', event =>\n          handleKeyDown(\n            event,\n            ref,\n            options,\n            searchBarActive,\n            setSearch,\n            setDefaultSearch,\n            search,\n          ),\n        )\n      }\n    }\n  }, [\n    isDropdownVisible,\n    searchBarActive,\n    options,\n    onSearch,\n    search,\n    refSelect,\n    setDefaultSearch,\n    setIsDropdownVisible,\n    searchable,\n  ])\n\n  const isEmpty = useMemo(() => {\n    if (Array.isArray(displayedOptions)) {\n      return !(displayedOptions.length > 0)\n    }\n    const groups = Object.keys(displayedOptions)\n    for (const group of groups) {\n      if (displayedOptions[group].length !== 0) {\n        return false\n      }\n    }\n\n    return true\n  }, [displayedOptions])\n\n  return (\n    <StyledPopup\n      visible={isDropdownVisible}\n      text={\n        <Stack>\n          {searchable && !isLoading ? (\n            <SearchBarDropdown\n              placeholder={placeholder}\n              displayedOptions={displayedOptions}\n              setSearchBarActive={setSearchBarActive}\n            />\n          ) : null}\n          <CreateDropdown\n            isEmpty={isEmpty}\n            emptyState={emptyState}\n            descriptionDirection={descriptionDirection}\n            loadMore={loadMore}\n            optionalInfoPlacement={optionalInfoPlacement}\n            defaultSearchValue={defaultSearchValue}\n            isLoading={isLoading}\n          />\n          {footer ? <PopupFooter>{footer}</PopupFooter> : null}\n        </Stack>\n      }\n      placement=\"bottom\"\n      containerFullWidth\n      disableAnimation\n      maxWidth={maxWidth}\n      hasArrow={false}\n      ref={ref}\n      tabIndex={0}\n      role=\"dialog\"\n      debounceDelay={0}\n    >\n      {children}\n    </StyledPopup>\n  )\n}\n"]} */"));
160
+ const LoadMore = /* @__PURE__ */ _styled(Stack, process.env.NODE_ENV === "production" ? {
161
+ target: "eeucz6c0"
162
+ } : {
163
+ target: "eeucz6c0",
164
+ label: "LoadMore"
165
+ })("padding:", ({
166
+ theme
167
+ }) => theme.space[0.5], ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/Dropdown.tsx"],"names":[],"mappings":"AAgK8B","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/Dropdown.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { useEffect, useMemo, useRef, useState } from 'react'\nimport type { Dispatch, ReactNode, RefObject, SetStateAction } from 'react'\nimport { Checkbox } from '../Checkbox'\nimport { Popup } from '../Popup'\nimport { Skeleton } from '../Skeleton'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\nimport { DisplayOption } from './DropdownOption'\nimport { SearchBarDropdown } from './SearchBarDropdown'\nimport { useSelectInput } from './SelectInputProvider'\nimport type { DataType, OptionType } from './types'\n\nexport type DropdownProps = {\n  children: ReactNode\n  emptyState: ReactNode\n  descriptionDirection: 'row' | 'column'\n  searchable: boolean\n  placeholder: string\n  footer?: ReactNode\n  refSelect: RefObject<HTMLDivElement>\n  loadMore?: ReactNode\n  optionalInfoPlacement: 'left' | 'right'\n  isLoading?: boolean\n}\n\nexport type CreateDropdownProps = {\n  isEmpty: boolean\n  emptyState: ReactNode\n  descriptionDirection: 'row' | 'column'\n  loadMore?: ReactNode\n  optionalInfoPlacement: 'left' | 'right'\n  defaultSearchValue: string | null\n  isLoading?: boolean\n}\n\nconst NON_SEARCHABLE_KEYS = [\n  'Tab',\n  ' ',\n  'Enter',\n  'CapsLock',\n  'Shift',\n  'ArrowDown',\n  'ArrowUp',\n  'ArrowLeft',\n  'ArrowRight',\n  'Escape',\n]\n\nconst StyledPopup = styled(Popup)`\n  width: 100%;\n  background-color: ${({ theme }) => theme.colors.neutral.background};\n  color: ${({ theme }) => theme.colors.neutral.text};\n  box-shadow: ${({ theme }) => theme.shadows.dropdown};\n  padding: ${({ theme }) => theme.space[0]};\n`\n\nconst DropdownContainer = styled(Stack)<{ 'data-grouped': boolean }>`\n  max-height: 256px;\n  overflow-y: scroll;\n  padding: ${({ theme }) => theme.space[0]};\n  padding-bottom: ${({ theme }) => theme.space[0.5]};\n  padding-top: ${({ theme }) => theme.space[0.5]};\n\n  &[data-grouped='true'] {\n    padding-top: ${({ theme }) => theme.space[0]};\n  }\n`\nconst DropdownGroup = styled.button<{ 'data-selectgroup': boolean }>`\n  display: flex;\n  width: 100%;\n  justify-content: left;\n  align-items: center;\n  border: none;\n  background-color: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n  position: sticky;\n  top: 0px;\n  padding-right: ${({ theme }) => theme.space[2]};\n  padding-left: ${({ theme }) => theme.space[2]};\n  height: ${({ theme }) => theme.space[4]};\n  text-align: left;\n  margin-bottom: ${({ theme }) => theme.space['0.25']};\n\n  &:focus {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n    outline: none;\n  }\n\n  &[data-selectgroup='true'] {\n    padding-left: ${({ theme }) => theme.space[2]};\n    border-left: ${({ theme }) => theme.space[0.5]} solid ${({ theme }) => theme.colors.neutral.backgroundWeak};\n  }\n\n  &[data-selectgroup='true']:focus {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundHover};\n  }\n}\n`\nconst DropdownGroupWrapper = styled.div`\n  position: sticky;\n  top: 0px;\n`\nconst DropdownItem = styled.button<{\n  'data-selected': boolean\n  disabled: boolean\n}>`\n  text-align:left;\n  border: none;\n  background-color: ${({ theme }) => theme.colors.neutral.background};\n\n  padding: ${({ theme }) => theme.space['1.5']} ${({ theme }) => theme.space['2']} ${({ theme }) => theme.space['1.5']} ${({ theme }) => theme.space['2']};\n  margin-left: ${({ theme }) => theme.space['0.5']};\n  margin-right: ${({ theme }) => theme.space['0.5']};\n\n  color:  ${({ theme }) => theme.colors.neutral.text};\n  border-radius: ${({ theme }) => theme.radii.default};\n\n  &:hover, :focus {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n    color: ${({ theme }) => theme.colors.primary.text};\n    cursor: pointer;\n    outline: none;\n  }\n\n  &[data-selected='true'] {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n  }\n\n  &[disabled] {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n\n  }\n\n  &[disabled]:hover, [disabled]:focus {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundStrongDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textStrongDisabled};\n    cursor: not-allowed;\n    outline: none;\n  }\n\n  }\n`\nconst PopupFooter = styled.div`\n  width: 100%;\n  padding: ${({ theme }) => theme.space[1.5]} ${({ theme }) => theme.space[2]}\n    ${({ theme }) => theme.space[1.5]} ${({ theme }) => theme.space[2]};\n  box-shadow: ${({ theme }) => theme.shadows.dropdown};\n`\nconst StyledCheckbox = styled(Checkbox)`\n  width: 100%;\n  position: static;\n  text-align: left;\n  align-items: center;\n  pointer-events: none;\n` // pointer-events: none prevents any error when using the checkbox in a form since it is an unnamed input\n\nconst EmptyState = styled(Stack)`\n  padding: ${({ theme }) => theme.space[2]};\n`\nconst LoadMore = styled(Stack)`\n  padding: ${({ theme }) => theme.space[0.5]};\n`\n\nconst moveFocusDown = () => {\n  const options = document.querySelectorAll(\n    '#items > button[role=\"option\"]:not([disabled])',\n  )\n  const activeItem = document.activeElement\n  if (options) {\n    for (let i = 0; i < options?.length; i += 1) {\n      const listLength = options.length\n      if (activeItem === options[i] && activeItem !== options[listLength - 1]) {\n        ;(options[i + 1] as HTMLElement).focus()\n      }\n    }\n  }\n}\nconst moveFocusUp = () => {\n  const options = document.querySelectorAll(\n    '#items > button[role=\"option\"]:not([disabled])',\n  )\n  const activeItem = document.activeElement\n\n  if (options) {\n    for (let i = 0; i < options.length; i += 1) {\n      if (activeItem === options[i] && activeItem !== options[0]) {\n        ;(options[i - 1] as HTMLElement).focus()\n      }\n    }\n  }\n}\nconst handleKeyDownSelect = (key: string) => {\n  if (key === 'ArrowDown') {\n    moveFocusDown()\n  }\n  if (key === 'ArrowUp') {\n    moveFocusUp()\n  }\n}\nconst handleClickOutside = (\n  event: MouseEvent,\n  ref: RefObject<HTMLDivElement>,\n  setIsDropdownVisibile: Dispatch<SetStateAction<boolean>>,\n  refSelect: RefObject<HTMLDivElement>,\n  onSearch: Dispatch<SetStateAction<DataType>>,\n  options: DataType,\n) => {\n  if (\n    ref.current &&\n    !ref.current.contains(event.target as Node) &&\n    !refSelect.current?.contains(event.target as Node)\n  ) {\n    setIsDropdownVisibile(false) // hide dropdown when clicking outside of the dropdown\n    onSearch(options) // reset displayed options to default when dropdown is hidden\n  }\n}\n\nconst handleKeyDown = (\n  event: KeyboardEvent,\n  ref: RefObject<HTMLDivElement>,\n  options: DataType,\n  searchBarActive: boolean,\n  setSearch: Dispatch<SetStateAction<string>>,\n  setDefaultSearch: Dispatch<SetStateAction<string | null>>,\n  search: string,\n) => {\n  // Deals with default search\n  if (\n    ref.current &&\n    !searchBarActive &&\n    !NON_SEARCHABLE_KEYS.includes(event.key) &&\n    document.activeElement?.ariaLabel !== 'search-bar'\n  ) {\n    const currentSearch = search + event.key\n    setSearch(currentSearch)\n    ref.current.focus()\n    if (!Array.isArray(options)) {\n      const closestOptions = { ...options }\n      Object.keys(closestOptions).map((group: string) => {\n        closestOptions[group] = closestOptions[group].filter(option =>\n          option.searchText\n            ? option.searchText.toLocaleLowerCase().startsWith(currentSearch)\n            : option.value.toLocaleLowerCase().startsWith(currentSearch),\n        )\n\n        return null\n      })\n      const closestOption = closestOptions[Object.keys(closestOptions)[0]][0]\n      if (closestOption) {\n        setDefaultSearch(closestOption.searchText ?? closestOption.value)\n      } else {\n        setDefaultSearch(null)\n      }\n    } else {\n      const closestOption = [...options].filter(option =>\n        option.searchText\n          ? option.searchText.toLocaleLowerCase().startsWith(currentSearch)\n          : option.value.toLocaleLowerCase().startsWith(currentSearch),\n      )[0]\n      if (closestOption) {\n        setDefaultSearch(closestOption.searchText ?? closestOption.value)\n      } else {\n        setDefaultSearch(null)\n      }\n    }\n  }\n}\nconst CreateDropdown = ({\n  isEmpty,\n  emptyState,\n  descriptionDirection,\n  loadMore,\n  optionalInfoPlacement,\n  defaultSearchValue,\n  isLoading,\n}: CreateDropdownProps) => {\n  const {\n    setIsDropdownVisible,\n    onChange,\n    options,\n    multiselect,\n    selectAll,\n    selectAllGroup,\n    displayedOptions,\n    setSelectedData,\n    selectedData,\n  } = useSelectInput()\n  const focusedItemRef = useRef<HTMLButtonElement>(null)\n\n  useEffect(() => {\n    if (defaultSearchValue && focusedItemRef?.current) {\n      focusedItemRef.current.focus()\n    }\n  }, [defaultSearchValue])\n\n  if (isEmpty) {\n    return (\n      <EmptyState gap={2} alignItems=\"center\">\n        {emptyState ?? (\n          <Text variant=\"bodyStrong\" as=\"p\">\n            No options\n          </Text>\n        )}\n      </EmptyState>\n    )\n  }\n\n  const handleClick = (clickedOption: OptionType, group?: string) => {\n    setSelectedData({ type: 'selectOption', clickedOption, group })\n    if (multiselect) {\n      if (selectedData.selectedValues.includes(clickedOption.value)) {\n        onChange?.(\n          selectedData.selectedValues.filter(\n            val => val !== clickedOption.value,\n          ),\n        )\n      } else {\n        onChange?.([...selectedData.selectedValues, clickedOption.value])\n      }\n    } else {\n      onChange?.(clickedOption.value)\n    }\n    setIsDropdownVisible(multiselect) // hide the dropdown on click when single select only\n  }\n\n  const selectAllOptions = () => {\n    if (multiselect) {\n      setSelectedData({ type: 'selectAll' })\n      if (selectedData.allSelected && onChange) {\n        onChange([])\n      } else {\n        const allValues: OptionType[] = []\n        if (!Array.isArray(options)) {\n          Object.keys(options).map((group: string) =>\n            options[group].map(option => {\n              if (!option.disabled) {\n                allValues.push(option)\n              }\n\n              return null\n            }),\n          )\n        } else {\n          options.map(option => allValues.push(option))\n        }\n        onChange?.(allValues.map(value => value.value))\n      }\n    }\n  }\n\n  const handleSelectGroup = (group: string) => {\n    if (multiselect) {\n      setSelectedData({ type: 'selectGroup', selectedGroup: group })\n      if (!Array.isArray(options)) {\n        if (selectedData.selectedGroups.includes(group)) {\n          const newSelectedValues = [...selectedData.selectedValues].filter(\n            selectedValue =>\n              !options[group].find(option => option.value === selectedValue),\n          )\n          onChange?.(newSelectedValues)\n        } else {\n          const newSelectedValues = [...selectedData.selectedValues]\n\n          options[group].map(option =>\n            newSelectedValues.includes(option.value) || option.disabled\n              ? null\n              : newSelectedValues.push(option.value),\n          )\n          onChange?.(newSelectedValues)\n        }\n      }\n    }\n  }\n\n  return !Array.isArray(displayedOptions) ? (\n    <DropdownContainer\n      role=\"listbox\"\n      id=\"select-dropdown\"\n      onKeyDown={event => {\n        event.preventDefault()\n        handleKeyDownSelect(event.key)\n      }}\n      data-grouped\n    >\n      {isLoading ? (\n        <Skeleton variant=\"block\" />\n      ) : (\n        <>\n          {selectAll && multiselect ? (\n            <Stack id=\"items\">\n              <DropdownItem\n                disabled={false}\n                data-selected={selectedData.allSelected}\n                aria-label=\"select-all\"\n                data-testid=\"select-all\"\n                id=\"select-all\"\n                role=\"option\"\n                onKeyDown={event =>\n                  [' ', 'Enter'].includes(event.key) ? selectAllOptions() : null\n                }\n                onClick={selectAllOptions}\n              >\n                <StyledCheckbox\n                  checked={selectedData.allSelected}\n                  disabled={false}\n                  value=\"select-all\"\n                  data-testid=\"select-all-checkbox\"\n                  tabIndex={-1}\n                >\n                  <Stack direction=\"column\">\n                    <Text as=\"span\" variant=\"body\" placement=\"left\">\n                      {selectAll.label}\n                    </Text>\n                    <Text\n                      as=\"span\"\n                      variant=\"bodySmall\"\n                      sentiment=\"neutral\"\n                      placement=\"left\"\n                      prominence=\"weak\"\n                    >\n                      {selectAll.description}\n                    </Text>\n                  </Stack>\n                </StyledCheckbox>\n              </DropdownItem>\n            </Stack>\n          ) : null}\n          {Object.keys(displayedOptions).map((group, index) => (\n            <Stack key={group} gap={0.25}>\n              {displayedOptions[group].length > 0 ? (\n                <DropdownGroupWrapper id={selectAllGroup ? 'items' : undefined}>\n                  <DropdownGroup\n                    key={group}\n                    tabIndex={selectAllGroup ? 0 : -1}\n                    onKeyDown={event => {\n                      if ([' ', 'Enter'].includes(event.key)) {\n                        event.preventDefault()\n                        handleSelectGroup(group)\n                      }\n                    }}\n                    data-selectgroup={selectAllGroup}\n                    role=\"group\"\n                    data-testid={`group-${index}`}\n                    onClick={() =>\n                      selectAllGroup ? handleSelectGroup(group) : null\n                    }\n                  >\n                    {selectAllGroup ? (\n                      <StyledCheckbox\n                        checked={selectedData.selectedGroups.includes(group)}\n                        disabled={false}\n                        value={group}\n                        data-testid=\"select-group\"\n                        tabIndex={-1}\n                      >\n                        <Text variant=\"caption\" as=\"span\" placement=\"left\">\n                          {group.toUpperCase()}\n                        </Text>\n                      </StyledCheckbox>\n                    ) : (\n                      <Text variant=\"caption\" as=\"span\" placement=\"left\">\n                        {group.toUpperCase()}\n                      </Text>\n                    )}\n                  </DropdownGroup>\n                </DropdownGroupWrapper>\n              ) : null}\n              <Stack id=\"items\" gap=\"0.25\">\n                {displayedOptions[group].map((option, indexOption) => (\n                  <DropdownItem\n                    key={option.value}\n                    disabled={option.disabled}\n                    data-selected={\n                      selectedData.selectedValues.includes(option.value) &&\n                      !option.disabled\n                    }\n                    aria-label={option.value}\n                    data-testid={`option-${option.value}`}\n                    id={`option-${indexOption}`}\n                    role=\"option\"\n                    onClick={() => {\n                      if (!option.disabled) {\n                        handleClick(option, group)\n                      }\n                    }}\n                    onKeyDown={event =>\n                      [' ', 'Enter'].includes(event.key)\n                        ? handleClick(option, group)\n                        : null\n                    }\n                    ref={\n                      option.value === defaultSearchValue ||\n                      option.searchText === defaultSearchValue\n                        ? focusedItemRef\n                        : null\n                    }\n                  >\n                    {multiselect ? (\n                      <StyledCheckbox\n                        checked={\n                          selectedData.selectedValues.includes(option.value) &&\n                          !option.disabled\n                        }\n                        disabled={option.disabled}\n                        value={option.value}\n                        tabIndex={-1}\n                      >\n                        <DisplayOption\n                          option={option}\n                          descriptionDirection={descriptionDirection}\n                          optionalInfoPlacement={optionalInfoPlacement}\n                        />\n                      </StyledCheckbox>\n                    ) : (\n                      <DisplayOption\n                        option={option}\n                        descriptionDirection={descriptionDirection}\n                        optionalInfoPlacement={optionalInfoPlacement}\n                      />\n                    )}\n                  </DropdownItem>\n                ))}\n                {loadMore ? <LoadMore>{loadMore}</LoadMore> : null}\n              </Stack>\n            </Stack>\n          ))}\n        </>\n      )}\n    </DropdownContainer>\n  ) : (\n    <DropdownContainer\n      role=\"listbox\"\n      id=\"select-dropdown\"\n      onKeyDown={event => {\n        event.preventDefault()\n        handleKeyDownSelect(event.key)\n      }}\n      gap={0.25}\n      data-grouped={false}\n    >\n      {selectAll && multiselect ? (\n        <Stack id=\"items\" gap={0.25}>\n          <DropdownItem\n            disabled={false}\n            data-selected={selectedData.allSelected}\n            aria-label=\"select-all\"\n            data-testid=\"select-all\"\n            role=\"option\"\n            onKeyDown={event =>\n              [' ', 'Enter'].includes(event.key) ? selectAllOptions() : null\n            }\n            onClick={selectAllOptions}\n          >\n            <StyledCheckbox\n              checked={selectedData.allSelected}\n              disabled={false}\n              value=\"select-all\"\n              data-testid=\"select-all-checkbox\"\n              tabIndex={-1}\n            >\n              <Stack direction=\"column\">\n                <Text as=\"span\" variant=\"body\" placement=\"left\">\n                  {selectAll.label}\n                </Text>\n                <Text\n                  as=\"span\"\n                  variant=\"bodySmall\"\n                  sentiment=\"neutral\"\n                  placement=\"left\"\n                  prominence=\"weak\"\n                >\n                  {selectAll.description}\n                </Text>\n              </Stack>\n            </StyledCheckbox>\n          </DropdownItem>\n        </Stack>\n      ) : null}\n      <Stack id=\"items\" gap={0.25}>\n        {isLoading ? (\n          <Skeleton variant=\"block\" />\n        ) : (\n          displayedOptions.map((option, index) => (\n            <DropdownItem\n              key={option.value}\n              disabled={option.disabled}\n              data-selected={\n                selectedData.selectedValues.includes(option.value) &&\n                !option.disabled\n              }\n              onClick={() => {\n                if (!option.disabled) {\n                  handleClick(option)\n                }\n              }}\n              aria-label={option.value}\n              data-testid={`option-${option.value}`}\n              id={`option-${index}`}\n              role=\"option\"\n              ref={\n                option.value === defaultSearchValue ||\n                option.searchText === defaultSearchValue\n                  ? focusedItemRef\n                  : null\n              }\n              onKeyDown={event =>\n                [' ', 'Enter'].includes(event.key) ? handleClick(option) : null\n              }\n            >\n              {multiselect ? (\n                <StyledCheckbox\n                  checked={\n                    selectedData.selectedValues.includes(option.value) &&\n                    !option.disabled\n                  }\n                  disabled={option.disabled}\n                  value={option.value}\n                  tabIndex={-1}\n                >\n                  <DisplayOption\n                    option={option}\n                    descriptionDirection={descriptionDirection}\n                    optionalInfoPlacement={optionalInfoPlacement}\n                  />\n                </StyledCheckbox>\n              ) : (\n                <DisplayOption\n                  option={option}\n                  descriptionDirection={descriptionDirection}\n                  optionalInfoPlacement={optionalInfoPlacement}\n                />\n              )}\n            </DropdownItem>\n          ))\n        )}\n        {loadMore ? <LoadMore>{loadMore}</LoadMore> : null}\n      </Stack>\n    </DropdownContainer>\n  )\n}\nexport const Dropdown = ({\n  children,\n  emptyState,\n  descriptionDirection,\n  searchable,\n  placeholder,\n  footer,\n  refSelect,\n  loadMore,\n  optionalInfoPlacement,\n  isLoading,\n}: DropdownProps) => {\n  const {\n    setIsDropdownVisible,\n    isDropdownVisible,\n    onSearch,\n    searchInput,\n    options,\n    displayedOptions,\n  } = useSelectInput()\n  const [searchBarActive, setSearchBarActive] = useState(false)\n  const [defaultSearchValue, setDefaultSearch] = useState<string | null>(null)\n  const maxWidth = refSelect.current?.offsetWidth\n  const ref = useRef<HTMLDivElement>(null)\n  const [search, setSearch] = useState<string>('')\n  useEffect(() => {\n    if (!searchInput) {\n      onSearch(options)\n    }\n  }, [onSearch, options, searchInput])\n\n  useEffect(() => {\n    if (!isDropdownVisible) {\n      setDefaultSearch(null)\n      setSearch('')\n    }\n\n    document.addEventListener('mousedown', event =>\n      handleClickOutside(\n        event,\n        ref,\n        setIsDropdownVisible,\n        refSelect,\n        onSearch,\n        options,\n      ),\n    )\n\n    if (!searchable) {\n      document.addEventListener('keydown', event =>\n        handleKeyDown(\n          event,\n          ref,\n          options,\n          searchBarActive,\n          setSearch,\n          setDefaultSearch,\n          search,\n        ),\n      )\n    }\n\n    return () => {\n      document.removeEventListener('mousedown', event =>\n        handleClickOutside(\n          event,\n          ref,\n          setIsDropdownVisible,\n          refSelect,\n          onSearch,\n          options,\n        ),\n      )\n      if (!searchable) {\n        document.removeEventListener('keydown', event =>\n          handleKeyDown(\n            event,\n            ref,\n            options,\n            searchBarActive,\n            setSearch,\n            setDefaultSearch,\n            search,\n          ),\n        )\n      }\n    }\n  }, [\n    isDropdownVisible,\n    searchBarActive,\n    options,\n    onSearch,\n    search,\n    refSelect,\n    setDefaultSearch,\n    setIsDropdownVisible,\n    searchable,\n  ])\n\n  const isEmpty = useMemo(() => {\n    if (Array.isArray(displayedOptions)) {\n      return !(displayedOptions.length > 0)\n    }\n    const groups = Object.keys(displayedOptions)\n    for (const group of groups) {\n      if (displayedOptions[group].length !== 0) {\n        return false\n      }\n    }\n\n    return true\n  }, [displayedOptions])\n\n  return (\n    <StyledPopup\n      visible={isDropdownVisible}\n      text={\n        <Stack>\n          {searchable && !isLoading ? (\n            <SearchBarDropdown\n              placeholder={placeholder}\n              displayedOptions={displayedOptions}\n              setSearchBarActive={setSearchBarActive}\n            />\n          ) : null}\n          <CreateDropdown\n            isEmpty={isEmpty}\n            emptyState={emptyState}\n            descriptionDirection={descriptionDirection}\n            loadMore={loadMore}\n            optionalInfoPlacement={optionalInfoPlacement}\n            defaultSearchValue={defaultSearchValue}\n            isLoading={isLoading}\n          />\n          {footer ? <PopupFooter>{footer}</PopupFooter> : null}\n        </Stack>\n      }\n      placement=\"bottom\"\n      containerFullWidth\n      disableAnimation\n      maxWidth={maxWidth}\n      hasArrow={false}\n      ref={ref}\n      tabIndex={0}\n      role=\"dialog\"\n      debounceDelay={0}\n    >\n      {children}\n    </StyledPopup>\n  )\n}\n"]} */"));
133
168
  const moveFocusDown = () => {
134
- const options = document.querySelectorAll(
135
- '#items > button[role="option"]:not([disabled])'
136
- );
169
+ const options = document.querySelectorAll('#items > button[role="option"]:not([disabled])');
137
170
  const activeItem = document.activeElement;
138
171
  if (options) {
139
172
  for (let i = 0; i < options?.length; i += 1) {
@@ -145,9 +178,7 @@ const moveFocusDown = () => {
145
178
  }
146
179
  };
147
180
  const moveFocusUp = () => {
148
- const options = document.querySelectorAll(
149
- '#items > button[role="option"]:not([disabled])'
150
- );
181
+ const options = document.querySelectorAll('#items > button[role="option"]:not([disabled])');
151
182
  const activeItem = document.activeElement;
152
183
  if (options) {
153
184
  for (let i = 0; i < options.length; i += 1) {
@@ -177,11 +208,11 @@ const handleKeyDown = (event, ref, options, searchBarActive, setSearch, setDefau
177
208
  setSearch(currentSearch);
178
209
  ref.current.focus();
179
210
  if (!Array.isArray(options)) {
180
- const closestOptions = { ...options };
211
+ const closestOptions = {
212
+ ...options
213
+ };
181
214
  Object.keys(closestOptions).map((group) => {
182
- closestOptions[group] = closestOptions[group].filter(
183
- (option) => option.searchText ? option.searchText.toLocaleLowerCase().startsWith(currentSearch) : option.value.toLocaleLowerCase().startsWith(currentSearch)
184
- );
215
+ closestOptions[group] = closestOptions[group].filter((option) => option.searchText ? option.searchText.toLocaleLowerCase().startsWith(currentSearch) : option.value.toLocaleLowerCase().startsWith(currentSearch));
185
216
  return null;
186
217
  });
187
218
  const closestOption = closestOptions[Object.keys(closestOptions)[0]][0];
@@ -191,9 +222,7 @@ const handleKeyDown = (event, ref, options, searchBarActive, setSearch, setDefau
191
222
  setDefaultSearch(null);
192
223
  }
193
224
  } else {
194
- const closestOption = [...options].filter(
195
- (option) => option.searchText ? option.searchText.toLocaleLowerCase().startsWith(currentSearch) : option.value.toLocaleLowerCase().startsWith(currentSearch)
196
- )[0];
225
+ const closestOption = [...options].filter((option) => option.searchText ? option.searchText.toLocaleLowerCase().startsWith(currentSearch) : option.value.toLocaleLowerCase().startsWith(currentSearch))[0];
197
226
  if (closestOption) {
198
227
  setDefaultSearch(closestOption.searchText ?? closestOption.value);
199
228
  } else {
@@ -232,14 +261,14 @@ const CreateDropdown = ({
232
261
  return /* @__PURE__ */ jsx(EmptyState, { gap: 2, alignItems: "center", children: emptyState ?? /* @__PURE__ */ jsx(Text, { variant: "bodyStrong", as: "p", children: "No options" }) });
233
262
  }
234
263
  const handleClick = (clickedOption, group) => {
235
- setSelectedData({ type: "selectOption", clickedOption, group });
264
+ setSelectedData({
265
+ type: "selectOption",
266
+ clickedOption,
267
+ group
268
+ });
236
269
  if (multiselect) {
237
270
  if (selectedData.selectedValues.includes(clickedOption.value)) {
238
- onChange?.(
239
- selectedData.selectedValues.filter(
240
- (val) => val !== clickedOption.value
241
- )
242
- );
271
+ onChange?.(selectedData.selectedValues.filter((val) => val !== clickedOption.value));
243
272
  } else {
244
273
  onChange?.([...selectedData.selectedValues, clickedOption.value]);
245
274
  }
@@ -250,20 +279,20 @@ const CreateDropdown = ({
250
279
  };
251
280
  const selectAllOptions = () => {
252
281
  if (multiselect) {
253
- setSelectedData({ type: "selectAll" });
282
+ setSelectedData({
283
+ type: "selectAll"
284
+ });
254
285
  if (selectedData.allSelected && onChange) {
255
286
  onChange([]);
256
287
  } else {
257
288
  const allValues = [];
258
289
  if (!Array.isArray(options)) {
259
- Object.keys(options).map(
260
- (group) => options[group].map((option) => {
261
- if (!option.disabled) {
262
- allValues.push(option);
263
- }
264
- return null;
265
- })
266
- );
290
+ Object.keys(options).map((group) => options[group].map((option) => {
291
+ if (!option.disabled) {
292
+ allValues.push(option);
293
+ }
294
+ return null;
295
+ }));
267
296
  } else {
268
297
  options.map((option) => allValues.push(option));
269
298
  }
@@ -273,246 +302,63 @@ const CreateDropdown = ({
273
302
  };
274
303
  const handleSelectGroup = (group) => {
275
304
  if (multiselect) {
276
- setSelectedData({ type: "selectGroup", selectedGroup: group });
305
+ setSelectedData({
306
+ type: "selectGroup",
307
+ selectedGroup: group
308
+ });
277
309
  if (!Array.isArray(options)) {
278
310
  if (selectedData.selectedGroups.includes(group)) {
279
- const newSelectedValues = [...selectedData.selectedValues].filter(
280
- (selectedValue) => !options[group].find((option) => option.value === selectedValue)
281
- );
311
+ const newSelectedValues = [...selectedData.selectedValues].filter((selectedValue) => !options[group].find((option) => option.value === selectedValue));
282
312
  onChange?.(newSelectedValues);
283
313
  } else {
284
314
  const newSelectedValues = [...selectedData.selectedValues];
285
- options[group].map(
286
- (option) => newSelectedValues.includes(option.value) || option.disabled ? null : newSelectedValues.push(option.value)
287
- );
315
+ options[group].map((option) => newSelectedValues.includes(option.value) || option.disabled ? null : newSelectedValues.push(option.value));
288
316
  onChange?.(newSelectedValues);
289
317
  }
290
318
  }
291
319
  }
292
320
  };
293
- return !Array.isArray(displayedOptions) ? /* @__PURE__ */ jsx(
294
- DropdownContainer,
295
- {
296
- role: "listbox",
297
- id: "select-dropdown",
298
- onKeyDown: (event) => {
299
- event.preventDefault();
300
- handleKeyDownSelect(event.key);
301
- },
302
- "data-grouped": true,
303
- children: isLoading ? /* @__PURE__ */ jsx(Skeleton, { variant: "block" }) : /* @__PURE__ */ jsxs(Fragment, { children: [
304
- selectAll && multiselect ? /* @__PURE__ */ jsx(Stack, { id: "items", children: /* @__PURE__ */ jsx(
305
- DropdownItem,
306
- {
307
- disabled: false,
308
- "data-selected": selectedData.allSelected,
309
- "aria-label": "select-all",
310
- "data-testid": "select-all",
311
- id: "select-all",
312
- role: "option",
313
- onKeyDown: (event) => [" ", "Enter"].includes(event.key) ? selectAllOptions() : null,
314
- onClick: selectAllOptions,
315
- children: /* @__PURE__ */ jsx(
316
- StyledCheckbox,
317
- {
318
- checked: selectedData.allSelected,
319
- disabled: false,
320
- value: "select-all",
321
- "data-testid": "select-all-checkbox",
322
- tabIndex: -1,
323
- children: /* @__PURE__ */ jsxs(Stack, { direction: "column", children: [
324
- /* @__PURE__ */ jsx(Text, { as: "span", variant: "body", placement: "left", children: selectAll.label }),
325
- /* @__PURE__ */ jsx(
326
- Text,
327
- {
328
- as: "span",
329
- variant: "bodySmall",
330
- sentiment: "neutral",
331
- placement: "left",
332
- prominence: "weak",
333
- children: selectAll.description
334
- }
335
- )
336
- ] })
337
- }
338
- )
321
+ return !Array.isArray(displayedOptions) ? /* @__PURE__ */ jsx(DropdownContainer, { role: "listbox", id: "select-dropdown", onKeyDown: (event) => {
322
+ event.preventDefault();
323
+ handleKeyDownSelect(event.key);
324
+ }, "data-grouped": true, children: isLoading ? /* @__PURE__ */ jsx(Skeleton, { variant: "block" }) : /* @__PURE__ */ jsxs(Fragment, { children: [
325
+ selectAll && multiselect ? /* @__PURE__ */ jsx(Stack, { id: "items", children: /* @__PURE__ */ jsx(DropdownItem, { disabled: false, "data-selected": selectedData.allSelected, "aria-label": "select-all", "data-testid": "select-all", id: "select-all", role: "option", onKeyDown: (event) => [" ", "Enter"].includes(event.key) ? selectAllOptions() : null, onClick: selectAllOptions, children: /* @__PURE__ */ jsx(StyledCheckbox, { checked: selectedData.allSelected, disabled: false, value: "select-all", "data-testid": "select-all-checkbox", tabIndex: -1, children: /* @__PURE__ */ jsxs(Stack, { direction: "column", children: [
326
+ /* @__PURE__ */ jsx(Text, { as: "span", variant: "body", placement: "left", children: selectAll.label }),
327
+ /* @__PURE__ */ jsx(Text, { as: "span", variant: "bodySmall", sentiment: "neutral", placement: "left", prominence: "weak", children: selectAll.description })
328
+ ] }) }) }) }) : null,
329
+ Object.keys(displayedOptions).map((group, index) => /* @__PURE__ */ jsxs(Stack, { gap: 0.25, children: [
330
+ displayedOptions[group].length > 0 ? /* @__PURE__ */ jsx(DropdownGroupWrapper, { id: selectAllGroup ? "items" : void 0, children: /* @__PURE__ */ jsx(DropdownGroup, { tabIndex: selectAllGroup ? 0 : -1, onKeyDown: (event) => {
331
+ if ([" ", "Enter"].includes(event.key)) {
332
+ event.preventDefault();
333
+ handleSelectGroup(group);
334
+ }
335
+ }, "data-selectgroup": selectAllGroup, role: "group", "data-testid": `group-${index}`, onClick: () => selectAllGroup ? handleSelectGroup(group) : null, children: selectAllGroup ? /* @__PURE__ */ jsx(StyledCheckbox, { checked: selectedData.selectedGroups.includes(group), disabled: false, value: group, "data-testid": "select-group", tabIndex: -1, children: /* @__PURE__ */ jsx(Text, { variant: "caption", as: "span", placement: "left", children: group.toUpperCase() }) }) : /* @__PURE__ */ jsx(Text, { variant: "caption", as: "span", placement: "left", children: group.toUpperCase() }) }, group) }) : null,
336
+ /* @__PURE__ */ jsxs(Stack, { id: "items", gap: "0.25", children: [
337
+ displayedOptions[group].map((option, indexOption) => /* @__PURE__ */ jsx(DropdownItem, { disabled: option.disabled, "data-selected": selectedData.selectedValues.includes(option.value) && !option.disabled, "aria-label": option.value, "data-testid": `option-${option.value}`, id: `option-${indexOption}`, role: "option", onClick: () => {
338
+ if (!option.disabled) {
339
+ handleClick(option, group);
339
340
  }
340
- ) }) : null,
341
- Object.keys(displayedOptions).map((group, index) => /* @__PURE__ */ jsxs(Stack, { gap: 0.25, children: [
342
- displayedOptions[group].length > 0 ? /* @__PURE__ */ jsx(DropdownGroupWrapper, { id: selectAllGroup ? "items" : void 0, children: /* @__PURE__ */ jsx(
343
- DropdownGroup,
344
- {
345
- tabIndex: selectAllGroup ? 0 : -1,
346
- onKeyDown: (event) => {
347
- if ([" ", "Enter"].includes(event.key)) {
348
- event.preventDefault();
349
- handleSelectGroup(group);
350
- }
351
- },
352
- "data-selectgroup": selectAllGroup,
353
- role: "group",
354
- "data-testid": `group-${index}`,
355
- onClick: () => selectAllGroup ? handleSelectGroup(group) : null,
356
- children: selectAllGroup ? /* @__PURE__ */ jsx(
357
- StyledCheckbox,
358
- {
359
- checked: selectedData.selectedGroups.includes(group),
360
- disabled: false,
361
- value: group,
362
- "data-testid": "select-group",
363
- tabIndex: -1,
364
- children: /* @__PURE__ */ jsx(Text, { variant: "caption", as: "span", placement: "left", children: group.toUpperCase() })
365
- }
366
- ) : /* @__PURE__ */ jsx(Text, { variant: "caption", as: "span", placement: "left", children: group.toUpperCase() })
367
- },
368
- group
369
- ) }) : null,
370
- /* @__PURE__ */ jsxs(Stack, { id: "items", gap: "0.25", children: [
371
- displayedOptions[group].map((option, indexOption) => /* @__PURE__ */ jsx(
372
- DropdownItem,
373
- {
374
- disabled: option.disabled,
375
- "data-selected": selectedData.selectedValues.includes(option.value) && !option.disabled,
376
- "aria-label": option.value,
377
- "data-testid": `option-${option.value}`,
378
- id: `option-${indexOption}`,
379
- role: "option",
380
- onClick: () => {
381
- if (!option.disabled) {
382
- handleClick(option, group);
383
- }
384
- },
385
- onKeyDown: (event) => [" ", "Enter"].includes(event.key) ? handleClick(option, group) : null,
386
- ref: option.value === defaultSearchValue || option.searchText === defaultSearchValue ? focusedItemRef : null,
387
- children: multiselect ? /* @__PURE__ */ jsx(
388
- StyledCheckbox,
389
- {
390
- checked: selectedData.selectedValues.includes(option.value) && !option.disabled,
391
- disabled: option.disabled,
392
- value: option.value,
393
- tabIndex: -1,
394
- children: /* @__PURE__ */ jsx(
395
- DisplayOption,
396
- {
397
- option,
398
- descriptionDirection,
399
- optionalInfoPlacement
400
- }
401
- )
402
- }
403
- ) : /* @__PURE__ */ jsx(
404
- DisplayOption,
405
- {
406
- option,
407
- descriptionDirection,
408
- optionalInfoPlacement
409
- }
410
- )
411
- },
412
- option.value
413
- )),
414
- loadMore ? /* @__PURE__ */ jsx(LoadMore, { children: loadMore }) : null
415
- ] })
416
- ] }, group))
341
+ }, onKeyDown: (event) => [" ", "Enter"].includes(event.key) ? handleClick(option, group) : null, ref: option.value === defaultSearchValue || option.searchText === defaultSearchValue ? focusedItemRef : null, children: multiselect ? /* @__PURE__ */ jsx(StyledCheckbox, { checked: selectedData.selectedValues.includes(option.value) && !option.disabled, disabled: option.disabled, value: option.value, tabIndex: -1, children: /* @__PURE__ */ jsx(DisplayOption, { option, descriptionDirection, optionalInfoPlacement }) }) : /* @__PURE__ */ jsx(DisplayOption, { option, descriptionDirection, optionalInfoPlacement }) }, option.value)),
342
+ loadMore ? /* @__PURE__ */ jsx(LoadMore, { children: loadMore }) : null
417
343
  ] })
418
- }
419
- ) : /* @__PURE__ */ jsxs(
420
- DropdownContainer,
421
- {
422
- role: "listbox",
423
- id: "select-dropdown",
424
- onKeyDown: (event) => {
425
- event.preventDefault();
426
- handleKeyDownSelect(event.key);
427
- },
428
- gap: 0.25,
429
- "data-grouped": false,
430
- children: [
431
- selectAll && multiselect ? /* @__PURE__ */ jsx(Stack, { id: "items", gap: 0.25, children: /* @__PURE__ */ jsx(
432
- DropdownItem,
433
- {
434
- disabled: false,
435
- "data-selected": selectedData.allSelected,
436
- "aria-label": "select-all",
437
- "data-testid": "select-all",
438
- role: "option",
439
- onKeyDown: (event) => [" ", "Enter"].includes(event.key) ? selectAllOptions() : null,
440
- onClick: selectAllOptions,
441
- children: /* @__PURE__ */ jsx(
442
- StyledCheckbox,
443
- {
444
- checked: selectedData.allSelected,
445
- disabled: false,
446
- value: "select-all",
447
- "data-testid": "select-all-checkbox",
448
- tabIndex: -1,
449
- children: /* @__PURE__ */ jsxs(Stack, { direction: "column", children: [
450
- /* @__PURE__ */ jsx(Text, { as: "span", variant: "body", placement: "left", children: selectAll.label }),
451
- /* @__PURE__ */ jsx(
452
- Text,
453
- {
454
- as: "span",
455
- variant: "bodySmall",
456
- sentiment: "neutral",
457
- placement: "left",
458
- prominence: "weak",
459
- children: selectAll.description
460
- }
461
- )
462
- ] })
463
- }
464
- )
465
- }
466
- ) }) : null,
467
- /* @__PURE__ */ jsxs(Stack, { id: "items", gap: 0.25, children: [
468
- isLoading ? /* @__PURE__ */ jsx(Skeleton, { variant: "block" }) : displayedOptions.map((option, index) => /* @__PURE__ */ jsx(
469
- DropdownItem,
470
- {
471
- disabled: option.disabled,
472
- "data-selected": selectedData.selectedValues.includes(option.value) && !option.disabled,
473
- onClick: () => {
474
- if (!option.disabled) {
475
- handleClick(option);
476
- }
477
- },
478
- "aria-label": option.value,
479
- "data-testid": `option-${option.value}`,
480
- id: `option-${index}`,
481
- role: "option",
482
- ref: option.value === defaultSearchValue || option.searchText === defaultSearchValue ? focusedItemRef : null,
483
- onKeyDown: (event) => [" ", "Enter"].includes(event.key) ? handleClick(option) : null,
484
- children: multiselect ? /* @__PURE__ */ jsx(
485
- StyledCheckbox,
486
- {
487
- checked: selectedData.selectedValues.includes(option.value) && !option.disabled,
488
- disabled: option.disabled,
489
- value: option.value,
490
- tabIndex: -1,
491
- children: /* @__PURE__ */ jsx(
492
- DisplayOption,
493
- {
494
- option,
495
- descriptionDirection,
496
- optionalInfoPlacement
497
- }
498
- )
499
- }
500
- ) : /* @__PURE__ */ jsx(
501
- DisplayOption,
502
- {
503
- option,
504
- descriptionDirection,
505
- optionalInfoPlacement
506
- }
507
- )
508
- },
509
- option.value
510
- )),
511
- loadMore ? /* @__PURE__ */ jsx(LoadMore, { children: loadMore }) : null
512
- ] })
513
- ]
514
- }
515
- );
344
+ ] }, group))
345
+ ] }) }) : /* @__PURE__ */ jsxs(DropdownContainer, { role: "listbox", id: "select-dropdown", onKeyDown: (event) => {
346
+ event.preventDefault();
347
+ handleKeyDownSelect(event.key);
348
+ }, gap: 0.25, "data-grouped": false, children: [
349
+ selectAll && multiselect ? /* @__PURE__ */ jsx(Stack, { id: "items", gap: 0.25, children: /* @__PURE__ */ jsx(DropdownItem, { disabled: false, "data-selected": selectedData.allSelected, "aria-label": "select-all", "data-testid": "select-all", role: "option", onKeyDown: (event) => [" ", "Enter"].includes(event.key) ? selectAllOptions() : null, onClick: selectAllOptions, children: /* @__PURE__ */ jsx(StyledCheckbox, { checked: selectedData.allSelected, disabled: false, value: "select-all", "data-testid": "select-all-checkbox", tabIndex: -1, children: /* @__PURE__ */ jsxs(Stack, { direction: "column", children: [
350
+ /* @__PURE__ */ jsx(Text, { as: "span", variant: "body", placement: "left", children: selectAll.label }),
351
+ /* @__PURE__ */ jsx(Text, { as: "span", variant: "bodySmall", sentiment: "neutral", placement: "left", prominence: "weak", children: selectAll.description })
352
+ ] }) }) }) }) : null,
353
+ /* @__PURE__ */ jsxs(Stack, { id: "items", gap: 0.25, children: [
354
+ isLoading ? /* @__PURE__ */ jsx(Skeleton, { variant: "block" }) : displayedOptions.map((option, index) => /* @__PURE__ */ jsx(DropdownItem, { disabled: option.disabled, "data-selected": selectedData.selectedValues.includes(option.value) && !option.disabled, onClick: () => {
355
+ if (!option.disabled) {
356
+ handleClick(option);
357
+ }
358
+ }, "aria-label": option.value, "data-testid": `option-${option.value}`, id: `option-${index}`, role: "option", ref: option.value === defaultSearchValue || option.searchText === defaultSearchValue ? focusedItemRef : null, onKeyDown: (event) => [" ", "Enter"].includes(event.key) ? handleClick(option) : null, children: multiselect ? /* @__PURE__ */ jsx(StyledCheckbox, { checked: selectedData.selectedValues.includes(option.value) && !option.disabled, disabled: option.disabled, value: option.value, tabIndex: -1, children: /* @__PURE__ */ jsx(DisplayOption, { option, descriptionDirection, optionalInfoPlacement }) }) : /* @__PURE__ */ jsx(DisplayOption, { option, descriptionDirection, optionalInfoPlacement }) }, option.value)),
359
+ loadMore ? /* @__PURE__ */ jsx(LoadMore, { children: loadMore }) : null
360
+ ] })
361
+ ] });
516
362
  };
517
363
  const Dropdown = ({
518
364
  children,
@@ -549,69 +395,17 @@ const Dropdown = ({
549
395
  setDefaultSearch(null);
550
396
  setSearch("");
551
397
  }
552
- document.addEventListener(
553
- "mousedown",
554
- (event) => handleClickOutside(
555
- event,
556
- ref,
557
- setIsDropdownVisible,
558
- refSelect,
559
- onSearch,
560
- options
561
- )
562
- );
398
+ document.addEventListener("mousedown", (event) => handleClickOutside(event, ref, setIsDropdownVisible, refSelect, onSearch, options));
563
399
  if (!searchable) {
564
- document.addEventListener(
565
- "keydown",
566
- (event) => handleKeyDown(
567
- event,
568
- ref,
569
- options,
570
- searchBarActive,
571
- setSearch,
572
- setDefaultSearch,
573
- search
574
- )
575
- );
400
+ document.addEventListener("keydown", (event) => handleKeyDown(event, ref, options, searchBarActive, setSearch, setDefaultSearch, search));
576
401
  }
577
402
  return () => {
578
- document.removeEventListener(
579
- "mousedown",
580
- (event) => handleClickOutside(
581
- event,
582
- ref,
583
- setIsDropdownVisible,
584
- refSelect,
585
- onSearch,
586
- options
587
- )
588
- );
403
+ document.removeEventListener("mousedown", (event) => handleClickOutside(event, ref, setIsDropdownVisible, refSelect, onSearch, options));
589
404
  if (!searchable) {
590
- document.removeEventListener(
591
- "keydown",
592
- (event) => handleKeyDown(
593
- event,
594
- ref,
595
- options,
596
- searchBarActive,
597
- setSearch,
598
- setDefaultSearch,
599
- search
600
- )
601
- );
405
+ document.removeEventListener("keydown", (event) => handleKeyDown(event, ref, options, searchBarActive, setSearch, setDefaultSearch, search));
602
406
  }
603
407
  };
604
- }, [
605
- isDropdownVisible,
606
- searchBarActive,
607
- options,
608
- onSearch,
609
- search,
610
- refSelect,
611
- setDefaultSearch,
612
- setIsDropdownVisible,
613
- searchable
614
- ]);
408
+ }, [isDropdownVisible, searchBarActive, options, onSearch, search, refSelect, setDefaultSearch, setIsDropdownVisible, searchable]);
615
409
  const isEmpty = useMemo(() => {
616
410
  if (Array.isArray(displayedOptions)) {
617
411
  return !(displayedOptions.length > 0);
@@ -624,45 +418,11 @@ const Dropdown = ({
624
418
  }
625
419
  return true;
626
420
  }, [displayedOptions]);
627
- return /* @__PURE__ */ jsx(
628
- StyledPopup,
629
- {
630
- visible: isDropdownVisible,
631
- text: /* @__PURE__ */ jsxs(Stack, { children: [
632
- searchable && !isLoading ? /* @__PURE__ */ jsx(
633
- SearchBarDropdown,
634
- {
635
- placeholder,
636
- displayedOptions,
637
- setSearchBarActive
638
- }
639
- ) : null,
640
- /* @__PURE__ */ jsx(
641
- CreateDropdown,
642
- {
643
- isEmpty,
644
- emptyState,
645
- descriptionDirection,
646
- loadMore,
647
- optionalInfoPlacement,
648
- defaultSearchValue,
649
- isLoading
650
- }
651
- ),
652
- footer ? /* @__PURE__ */ jsx(PopupFooter, { children: footer }) : null
653
- ] }),
654
- placement: "bottom",
655
- containerFullWidth: true,
656
- disableAnimation: true,
657
- maxWidth,
658
- hasArrow: false,
659
- ref,
660
- tabIndex: 0,
661
- role: "dialog",
662
- debounceDelay: 0,
663
- children
664
- }
665
- );
421
+ return /* @__PURE__ */ jsx(StyledPopup, { visible: isDropdownVisible, text: /* @__PURE__ */ jsxs(Stack, { children: [
422
+ searchable && !isLoading ? /* @__PURE__ */ jsx(SearchBarDropdown, { placeholder, displayedOptions, setSearchBarActive }) : null,
423
+ /* @__PURE__ */ jsx(CreateDropdown, { isEmpty, emptyState, descriptionDirection, loadMore, optionalInfoPlacement, defaultSearchValue, isLoading }),
424
+ footer ? /* @__PURE__ */ jsx(PopupFooter, { children: footer }) : null
425
+ ] }), placement: "bottom", containerFullWidth: true, disableAnimation: true, maxWidth, hasArrow: false, ref, tabIndex: 0, role: "dialog", debounceDelay: 0, children });
666
426
  };
667
427
  export {
668
428
  Dropdown