@ultraviolet/ui 1.75.3 → 1.75.4

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 (31) hide show
  1. package/dist/components/Banner/assets/default-image-small.svg.cjs +1 -1
  2. package/dist/components/Banner/assets/default-image-small.svg.js +1 -1
  3. package/dist/components/Banner/assets/default-image.svg.cjs +1 -1
  4. package/dist/components/Banner/assets/default-image.svg.js +1 -1
  5. package/dist/components/Card/index.cjs +2 -2
  6. package/dist/components/Card/index.js +2 -2
  7. package/dist/components/MenuV2/index.cjs +5 -4
  8. package/dist/components/MenuV2/index.d.ts +2 -1
  9. package/dist/components/MenuV2/index.js +5 -4
  10. package/dist/components/Meter/index.cjs +3 -5
  11. package/dist/components/Meter/index.js +3 -5
  12. package/dist/components/Popover/index.cjs +5 -4
  13. package/dist/components/Popover/index.d.ts +2 -1
  14. package/dist/components/Popover/index.js +5 -4
  15. package/dist/components/Popup/helpers.cjs +15 -13
  16. package/dist/components/Popup/helpers.d.ts +3 -1
  17. package/dist/components/Popup/helpers.js +15 -13
  18. package/dist/components/Popup/index.cjs +7 -5
  19. package/dist/components/Popup/index.d.ts +5 -1
  20. package/dist/components/Popup/index.js +7 -5
  21. package/dist/components/SelectInputV2/Dropdown.cjs +25 -14
  22. package/dist/components/SelectInputV2/Dropdown.d.ts +1 -1
  23. package/dist/components/SelectInputV2/Dropdown.js +25 -14
  24. package/dist/components/SelectInputV2/SelectBar.cjs +7 -14
  25. package/dist/components/SelectInputV2/SelectBar.js +7 -14
  26. package/dist/components/SelectInputV2/SelectInputProvider.cjs +1 -1
  27. package/dist/components/SelectInputV2/SelectInputProvider.js +1 -1
  28. package/dist/components/SelectInputV2/index.cjs +3 -3
  29. package/dist/components/SelectInputV2/index.d.ts +1 -1
  30. package/dist/components/SelectInputV2/index.js +3 -3
  31. package/package.json +5 -5
@@ -35,7 +35,7 @@ const StyledPopup = /* @__PURE__ */ _styled__default.default(index.Popup, proces
35
35
  theme
36
36
  }) => theme.space[0], ";margin-bottom:", ({
37
37
  theme
38
- }) => theme.space[10], ";" + (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":"AA2DiC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/Dropdown.tsx","sourcesContent":["import { useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport { useCallback, useEffect, useMemo, useRef, useState } from 'react'\nimport type {\n  Dispatch,\n  KeyboardEvent,\n  ReactNode,\n  RefObject,\n  SetStateAction,\n} 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, INPUT_SIZE_HEIGHT, type OptionType } from './types'\n\nconst DROPDOWN_MAX_HEIGHT = 256\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  size: 'small' | 'medium' | 'large'\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  min-width: 320px;\n  background-color: ${({ theme }) => theme.colors.other.elevation.background.raised};\n  color: ${({ theme }) => theme.colors.neutral.text};\n  box-shadow: ${({ theme }) => `${theme.shadows.raised[0]}, ${theme.shadows.raised[1]}`};\n  padding: ${({ theme }) => theme.space[0]};\n  margin-bottom: ${({ theme }) => theme.space[10]};\n`\n\nconst DropdownContainer = styled(Stack)<{ 'data-grouped': boolean }>`\n  max-height: ${DROPDOWN_MAX_HEIGHT}px;\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 }) =>\n      theme.colors.neutral.backgroundWeak};\n  }\n\n  &[data-selectgroup='true']:focus {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundHover};\n  }\n`\nconst DropdownGroupWrapper = styled.div`\n  position: sticky;\n  top: 0px;\n`\nconst DropdownItem = styled.div<{\n  'aria-selected': boolean\n  'aria-disabled': boolean\n}>`\n  text-align:left;\n  border: none;\n  background-color: ${({ theme }) => theme.colors.other.elevation.background.raised};\n\n  padding: ${({ theme }) => theme.space['1.5']} ${({ theme }) =>\n    theme.space['2']} ${({ theme }) => theme.space['1.5']} ${({ theme }) =>\n    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  &[aria-selected='true'] {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n  }\n\n  &[aria-disabled=\"true\"] {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n  }\n\n  &[aria-disabled=\"true\"]:hover, [aria-disabled=\"true\"]:focus {\n    background-color: ${({ theme }) =>\n      theme.colors.neutral.backgroundStrongDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textStrongDisabled};\n    cursor: not-allowed;\n    outline: none;\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 > div[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 > div[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 = (event: KeyboardEvent<HTMLDivElement>) => {\n  if (event.key === 'ArrowDown') {\n    event.preventDefault()\n    moveFocusDown()\n  }\n\n  if (event.key === 'ArrowUp') {\n    event.preventDefault()\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: globalThis.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<HTMLDivElement>(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={handleKeyDownSelect}\n      data-grouped\n    >\n      {isLoading ? (\n        <Skeleton variant=\"block\" />\n      ) : (\n        <>\n          {selectAll && multiselect ? (\n            <Stack id=\"items\">\n              <DropdownItem\n                aria-disabled={false}\n                aria-selected={selectedData.allSelected}\n                aria-label=\"select-all\"\n                data-testid=\"select-all\"\n                id=\"select-all\"\n                tabIndex={0}\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                  {group ? (\n                    <DropdownGroup\n                      key={group}\n                      type=\"button\"\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\n                          variant=\"caption\"\n                          as=\"span\"\n                          placement=\"left\"\n                          sentiment=\"neutral\"\n                        >\n                          {group.toUpperCase()}\n                        </Text>\n                      )}\n                    </DropdownGroup>\n                  ) : null}\n                </DropdownGroupWrapper>\n              ) : null}\n              <Stack id=\"items\" gap=\"0.25\">\n                {displayedOptions[group].map((option, indexOption) => (\n                  <DropdownItem\n                    key={option.value}\n                    aria-disabled={!!option.disabled}\n                    tabIndex={!option.disabled ? 0 : -1}\n                    aria-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              </Stack>\n            </Stack>\n          ))}\n        </>\n      )}\n      {loadMore ? <LoadMore>{loadMore}</LoadMore> : null}\n    </DropdownContainer>\n  ) : (\n    <DropdownContainer\n      role=\"listbox\"\n      tabIndex={-1}\n      id=\"select-dropdown\"\n      onKeyDown={handleKeyDownSelect}\n      gap={0.25}\n      data-grouped={false}\n    >\n      {selectAll && multiselect ? (\n        <Stack id=\"items\" gap={0.25} tabIndex={-1}>\n          <DropdownItem\n            aria-disabled={false}\n            aria-selected={selectedData.allSelected}\n            aria-label=\"select-all\"\n            data-testid=\"select-all\"\n            tabIndex={0}\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              aria-disabled={!!option.disabled}\n              aria-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              tabIndex={!option.disabled ? 0 : -1}\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  size,\n}: DropdownProps) => {\n  const {\n    setIsDropdownVisible,\n    isDropdownVisible,\n    onSearch,\n    searchInput,\n    options,\n    displayedOptions,\n  } = useSelectInput()\n  const theme = useTheme()\n  const [searchBarActive, setSearchBarActive] = useState(false)\n  const [defaultSearchValue, setDefaultSearch] = useState<string | null>(null)\n  const ref = useRef<HTMLDivElement>(null)\n  const [search, setSearch] = useState<string>('')\n  const [maxWidth, setWidth] = useState<string | number>()\n\n  useEffect(() => {\n    if (refSelect.current && isDropdownVisible) {\n      const position =\n        refSelect.current.getBoundingClientRect().bottom +\n        DROPDOWN_MAX_HEIGHT +\n        INPUT_SIZE_HEIGHT[size] +\n        parseInt(theme.space['5'], 10)\n      const overflow = position - window.innerHeight\n      if (overflow > 0) {\n        const modalElement = document.getElementById('backdrop-modal')\n\n        if (modalElement) {\n          modalElement.scrollBy({ top: overflow, behavior: 'smooth' })\n        } else window.scrollBy({ top: overflow, behavior: 'smooth' })\n      }\n    }\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, [isDropdownVisible, refSelect, size, ref.current])\n\n  const resizeDropdown = useCallback(() => {\n    if (\n      refSelect.current &&\n      refSelect.current.getBoundingClientRect().width > 0\n    ) {\n      setWidth(refSelect.current.getBoundingClientRect().width)\n    }\n  }, [refSelect])\n\n  useEffect(() => {\n    resizeDropdown()\n\n    window.addEventListener('resize', resizeDropdown)\n\n    return () => window.removeEventListener('resize', resizeDropdown)\n  }, [resizeDropdown])\n\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      disableAnimation\n      maxWidth={maxWidth ?? refSelect.current?.offsetWidth}\n      hasArrow={false}\n      ref={ref}\n      tabIndex={0}\n      role=\"dialog\"\n      debounceDelay={0}\n      containerFullWidth\n    >\n      {children}\n    </StyledPopup>\n  )\n}\n"]} */"));
38
+ }) => theme.space[10], ";" + (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":"AA2DiC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/Dropdown.tsx","sourcesContent":["import { useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport { useCallback, useEffect, useMemo, useRef, useState } from 'react'\nimport type {\n  Dispatch,\n  KeyboardEvent,\n  ReactNode,\n  RefObject,\n  SetStateAction,\n} 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, INPUT_SIZE_HEIGHT, type OptionType } from './types'\n\nconst DROPDOWN_MAX_HEIGHT = 256\n\nexport type DropdownProps = {\n  children: ReactNode\n  emptyState: ReactNode\n  descriptionDirection: 'row' | 'column'\n  searchable: boolean\n  placeholder: string\n  footer?: ((closeDropdown: () => void) => ReactNode) | ReactNode\n  refSelect: RefObject<HTMLDivElement>\n  loadMore?: ReactNode\n  optionalInfoPlacement: 'left' | 'right'\n  isLoading?: boolean\n  size: 'small' | 'medium' | 'large'\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  min-width: 320px;\n  background-color: ${({ theme }) => theme.colors.other.elevation.background.raised};\n  color: ${({ theme }) => theme.colors.neutral.text};\n  box-shadow: ${({ theme }) => `${theme.shadows.raised[0]}, ${theme.shadows.raised[1]}`};\n  padding: ${({ theme }) => theme.space[0]};\n  margin-bottom: ${({ theme }) => theme.space[10]};\n`\n\nconst DropdownContainer = styled(Stack)<{ 'data-grouped': boolean }>`\n  max-height: ${DROPDOWN_MAX_HEIGHT}px;\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 }) =>\n      theme.colors.neutral.backgroundWeak};\n  }\n\n  &[data-selectgroup='true']:focus {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundHover};\n  }\n`\nconst DropdownGroupWrapper = styled.div`\n  position: sticky;\n  top: 0px;\n`\nconst DropdownItem = styled.div<{\n  'aria-selected': boolean\n  'aria-disabled': boolean\n}>`\n  text-align:left;\n  border: none;\n  background-color: ${({ theme }) => theme.colors.other.elevation.background.raised};\n\n  padding: ${({ theme }) => theme.space['1.5']} ${({ theme }) =>\n    theme.space['2']} ${({ theme }) => theme.space['1.5']} ${({ theme }) =>\n    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  &[aria-selected='true'] {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n  }\n\n  &[aria-disabled=\"true\"] {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n  }\n\n  &[aria-disabled=\"true\"]:hover, [aria-disabled=\"true\"]:focus {\n    background-color: ${({ theme }) =>\n      theme.colors.neutral.backgroundStrongDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textStrongDisabled};\n    cursor: not-allowed;\n    outline: none;\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 > div[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 > div[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 = (event: KeyboardEvent<HTMLDivElement>) => {\n  if (event.key === 'ArrowDown') {\n    event.preventDefault()\n    moveFocusDown()\n  }\n\n  if (event.key === 'ArrowUp') {\n    event.preventDefault()\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)\n    onSearch(options)\n  }\n}\n\nconst handleKeyDown = (\n  event: globalThis.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<HTMLDivElement>(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={handleKeyDownSelect}\n      data-grouped\n    >\n      {isLoading ? (\n        <Skeleton variant=\"block\" />\n      ) : (\n        <>\n          {selectAll && multiselect ? (\n            <Stack id=\"items\">\n              <DropdownItem\n                aria-disabled={false}\n                aria-selected={selectedData.allSelected}\n                aria-label=\"select-all\"\n                data-testid=\"select-all\"\n                id=\"select-all\"\n                tabIndex={0}\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                  {group ? (\n                    <DropdownGroup\n                      key={group}\n                      type=\"button\"\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\n                          variant=\"caption\"\n                          as=\"span\"\n                          placement=\"left\"\n                          sentiment=\"neutral\"\n                        >\n                          {group.toUpperCase()}\n                        </Text>\n                      )}\n                    </DropdownGroup>\n                  ) : null}\n                </DropdownGroupWrapper>\n              ) : null}\n              <Stack id=\"items\" gap=\"0.25\">\n                {displayedOptions[group].map((option, indexOption) => (\n                  <DropdownItem\n                    key={option.value}\n                    aria-disabled={!!option.disabled}\n                    tabIndex={!option.disabled ? 0 : -1}\n                    aria-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              </Stack>\n            </Stack>\n          ))}\n        </>\n      )}\n      {loadMore ? <LoadMore>{loadMore}</LoadMore> : null}\n    </DropdownContainer>\n  ) : (\n    <DropdownContainer\n      role=\"listbox\"\n      tabIndex={-1}\n      id=\"select-dropdown\"\n      onKeyDown={handleKeyDownSelect}\n      gap={0.25}\n      data-grouped={false}\n    >\n      {selectAll && multiselect ? (\n        <Stack id=\"items\" gap={0.25} tabIndex={-1}>\n          <DropdownItem\n            aria-disabled={false}\n            aria-selected={selectedData.allSelected}\n            aria-label=\"select-all\"\n            data-testid=\"select-all\"\n            tabIndex={0}\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              aria-disabled={!!option.disabled}\n              aria-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              tabIndex={!option.disabled ? 0 : -1}\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  size,\n}: DropdownProps) => {\n  const {\n    setIsDropdownVisible,\n    isDropdownVisible,\n    onSearch,\n    searchInput,\n    options,\n    displayedOptions,\n    numberOfOptions,\n  } = useSelectInput()\n  const theme = useTheme()\n  const [searchBarActive, setSearchBarActive] = useState(false)\n  const [defaultSearchValue, setDefaultSearch] = useState<string | null>(null)\n  const ref = useRef<HTMLDivElement>(null)\n  const [search, setSearch] = useState<string>('')\n  const [maxWidth, setWidth] = useState<string | number>()\n\n  useEffect(() => {\n    if (refSelect.current && isDropdownVisible) {\n      const position =\n        refSelect.current.getBoundingClientRect().bottom +\n        DROPDOWN_MAX_HEIGHT +\n        INPUT_SIZE_HEIGHT[size] +\n        parseInt(theme.space['5'], 10)\n      const overflow = position - window.innerHeight\n      if (overflow > 0) {\n        const modalElement = document.getElementById('backdrop-modal')\n\n        if (modalElement) {\n          modalElement.scrollBy({ top: overflow, behavior: 'smooth' })\n        } else window.scrollBy({ top: overflow, behavior: 'smooth' })\n      }\n    }\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, [isDropdownVisible, refSelect, size, ref.current])\n\n  const resizeDropdown = useCallback(() => {\n    if (\n      refSelect.current &&\n      refSelect.current.getBoundingClientRect().width > 0\n    ) {\n      setWidth(refSelect.current.getBoundingClientRect().width)\n    }\n  }, [refSelect])\n\n  useEffect(() => {\n    resizeDropdown()\n\n    window.addEventListener('resize', resizeDropdown)\n\n    return () => window.removeEventListener('resize', resizeDropdown)\n  }, [resizeDropdown])\n\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\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  // No data is displayed (because of the search or because no data is provided)\n  // Set to true when noData by default\n  const isEmpty = useMemo(() => {\n    if (numberOfOptions === 0) return true\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, numberOfOptions])\n\n  const computedFooter = useMemo(() => {\n    if (footer) {\n      if (typeof footer === 'function') {\n        return (\n          <PopupFooter>{footer(() => setIsDropdownVisible(false))}</PopupFooter>\n        )\n      }\n\n      return <PopupFooter>{footer}</PopupFooter>\n    }\n\n    return null\n  }, [footer, setIsDropdownVisible])\n\n  return (\n    <StyledPopup\n      visible={isDropdownVisible}\n      text={\n        <Stack>\n          {searchable && !isLoading && numberOfOptions >= 6 ? (\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          {computedFooter}\n        </Stack>\n      }\n      placement=\"bottom\"\n      disableAnimation\n      maxWidth={maxWidth ?? refSelect.current?.offsetWidth}\n      hasArrow={false}\n      ref={ref}\n      tabIndex={0}\n      role=\"dialog\"\n      debounceDelay={0}\n      containerFullWidth\n      align=\"start\"\n    >\n      {children}\n    </StyledPopup>\n  )\n}\n"]} */"));
39
39
  const DropdownContainer = /* @__PURE__ */ _styled__default.default(index$1.Stack, process.env.NODE_ENV === "production" ? {
40
40
  target: "eeucz6c7"
41
41
  } : {
@@ -49,7 +49,7 @@ const DropdownContainer = /* @__PURE__ */ _styled__default.default(index$1.Stack
49
49
  theme
50
50
  }) => theme.space[0.5], ';&[data-grouped="true"]{padding-top:', ({
51
51
  theme
52
- }) => 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":"AAqEoE","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/Dropdown.tsx","sourcesContent":["import { useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport { useCallback, useEffect, useMemo, useRef, useState } from 'react'\nimport type {\n  Dispatch,\n  KeyboardEvent,\n  ReactNode,\n  RefObject,\n  SetStateAction,\n} 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, INPUT_SIZE_HEIGHT, type OptionType } from './types'\n\nconst DROPDOWN_MAX_HEIGHT = 256\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  size: 'small' | 'medium' | 'large'\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  min-width: 320px;\n  background-color: ${({ theme }) => theme.colors.other.elevation.background.raised};\n  color: ${({ theme }) => theme.colors.neutral.text};\n  box-shadow: ${({ theme }) => `${theme.shadows.raised[0]}, ${theme.shadows.raised[1]}`};\n  padding: ${({ theme }) => theme.space[0]};\n  margin-bottom: ${({ theme }) => theme.space[10]};\n`\n\nconst DropdownContainer = styled(Stack)<{ 'data-grouped': boolean }>`\n  max-height: ${DROPDOWN_MAX_HEIGHT}px;\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 }) =>\n      theme.colors.neutral.backgroundWeak};\n  }\n\n  &[data-selectgroup='true']:focus {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundHover};\n  }\n`\nconst DropdownGroupWrapper = styled.div`\n  position: sticky;\n  top: 0px;\n`\nconst DropdownItem = styled.div<{\n  'aria-selected': boolean\n  'aria-disabled': boolean\n}>`\n  text-align:left;\n  border: none;\n  background-color: ${({ theme }) => theme.colors.other.elevation.background.raised};\n\n  padding: ${({ theme }) => theme.space['1.5']} ${({ theme }) =>\n    theme.space['2']} ${({ theme }) => theme.space['1.5']} ${({ theme }) =>\n    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  &[aria-selected='true'] {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n  }\n\n  &[aria-disabled=\"true\"] {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n  }\n\n  &[aria-disabled=\"true\"]:hover, [aria-disabled=\"true\"]:focus {\n    background-color: ${({ theme }) =>\n      theme.colors.neutral.backgroundStrongDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textStrongDisabled};\n    cursor: not-allowed;\n    outline: none;\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 > div[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 > div[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 = (event: KeyboardEvent<HTMLDivElement>) => {\n  if (event.key === 'ArrowDown') {\n    event.preventDefault()\n    moveFocusDown()\n  }\n\n  if (event.key === 'ArrowUp') {\n    event.preventDefault()\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: globalThis.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<HTMLDivElement>(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={handleKeyDownSelect}\n      data-grouped\n    >\n      {isLoading ? (\n        <Skeleton variant=\"block\" />\n      ) : (\n        <>\n          {selectAll && multiselect ? (\n            <Stack id=\"items\">\n              <DropdownItem\n                aria-disabled={false}\n                aria-selected={selectedData.allSelected}\n                aria-label=\"select-all\"\n                data-testid=\"select-all\"\n                id=\"select-all\"\n                tabIndex={0}\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                  {group ? (\n                    <DropdownGroup\n                      key={group}\n                      type=\"button\"\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\n                          variant=\"caption\"\n                          as=\"span\"\n                          placement=\"left\"\n                          sentiment=\"neutral\"\n                        >\n                          {group.toUpperCase()}\n                        </Text>\n                      )}\n                    </DropdownGroup>\n                  ) : null}\n                </DropdownGroupWrapper>\n              ) : null}\n              <Stack id=\"items\" gap=\"0.25\">\n                {displayedOptions[group].map((option, indexOption) => (\n                  <DropdownItem\n                    key={option.value}\n                    aria-disabled={!!option.disabled}\n                    tabIndex={!option.disabled ? 0 : -1}\n                    aria-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              </Stack>\n            </Stack>\n          ))}\n        </>\n      )}\n      {loadMore ? <LoadMore>{loadMore}</LoadMore> : null}\n    </DropdownContainer>\n  ) : (\n    <DropdownContainer\n      role=\"listbox\"\n      tabIndex={-1}\n      id=\"select-dropdown\"\n      onKeyDown={handleKeyDownSelect}\n      gap={0.25}\n      data-grouped={false}\n    >\n      {selectAll && multiselect ? (\n        <Stack id=\"items\" gap={0.25} tabIndex={-1}>\n          <DropdownItem\n            aria-disabled={false}\n            aria-selected={selectedData.allSelected}\n            aria-label=\"select-all\"\n            data-testid=\"select-all\"\n            tabIndex={0}\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              aria-disabled={!!option.disabled}\n              aria-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              tabIndex={!option.disabled ? 0 : -1}\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  size,\n}: DropdownProps) => {\n  const {\n    setIsDropdownVisible,\n    isDropdownVisible,\n    onSearch,\n    searchInput,\n    options,\n    displayedOptions,\n  } = useSelectInput()\n  const theme = useTheme()\n  const [searchBarActive, setSearchBarActive] = useState(false)\n  const [defaultSearchValue, setDefaultSearch] = useState<string | null>(null)\n  const ref = useRef<HTMLDivElement>(null)\n  const [search, setSearch] = useState<string>('')\n  const [maxWidth, setWidth] = useState<string | number>()\n\n  useEffect(() => {\n    if (refSelect.current && isDropdownVisible) {\n      const position =\n        refSelect.current.getBoundingClientRect().bottom +\n        DROPDOWN_MAX_HEIGHT +\n        INPUT_SIZE_HEIGHT[size] +\n        parseInt(theme.space['5'], 10)\n      const overflow = position - window.innerHeight\n      if (overflow > 0) {\n        const modalElement = document.getElementById('backdrop-modal')\n\n        if (modalElement) {\n          modalElement.scrollBy({ top: overflow, behavior: 'smooth' })\n        } else window.scrollBy({ top: overflow, behavior: 'smooth' })\n      }\n    }\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, [isDropdownVisible, refSelect, size, ref.current])\n\n  const resizeDropdown = useCallback(() => {\n    if (\n      refSelect.current &&\n      refSelect.current.getBoundingClientRect().width > 0\n    ) {\n      setWidth(refSelect.current.getBoundingClientRect().width)\n    }\n  }, [refSelect])\n\n  useEffect(() => {\n    resizeDropdown()\n\n    window.addEventListener('resize', resizeDropdown)\n\n    return () => window.removeEventListener('resize', resizeDropdown)\n  }, [resizeDropdown])\n\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      disableAnimation\n      maxWidth={maxWidth ?? refSelect.current?.offsetWidth}\n      hasArrow={false}\n      ref={ref}\n      tabIndex={0}\n      role=\"dialog\"\n      debounceDelay={0}\n      containerFullWidth\n    >\n      {children}\n    </StyledPopup>\n  )\n}\n"]} */"));
52
+ }) => 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":"AAqEoE","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/Dropdown.tsx","sourcesContent":["import { useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport { useCallback, useEffect, useMemo, useRef, useState } from 'react'\nimport type {\n  Dispatch,\n  KeyboardEvent,\n  ReactNode,\n  RefObject,\n  SetStateAction,\n} 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, INPUT_SIZE_HEIGHT, type OptionType } from './types'\n\nconst DROPDOWN_MAX_HEIGHT = 256\n\nexport type DropdownProps = {\n  children: ReactNode\n  emptyState: ReactNode\n  descriptionDirection: 'row' | 'column'\n  searchable: boolean\n  placeholder: string\n  footer?: ((closeDropdown: () => void) => ReactNode) | ReactNode\n  refSelect: RefObject<HTMLDivElement>\n  loadMore?: ReactNode\n  optionalInfoPlacement: 'left' | 'right'\n  isLoading?: boolean\n  size: 'small' | 'medium' | 'large'\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  min-width: 320px;\n  background-color: ${({ theme }) => theme.colors.other.elevation.background.raised};\n  color: ${({ theme }) => theme.colors.neutral.text};\n  box-shadow: ${({ theme }) => `${theme.shadows.raised[0]}, ${theme.shadows.raised[1]}`};\n  padding: ${({ theme }) => theme.space[0]};\n  margin-bottom: ${({ theme }) => theme.space[10]};\n`\n\nconst DropdownContainer = styled(Stack)<{ 'data-grouped': boolean }>`\n  max-height: ${DROPDOWN_MAX_HEIGHT}px;\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 }) =>\n      theme.colors.neutral.backgroundWeak};\n  }\n\n  &[data-selectgroup='true']:focus {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundHover};\n  }\n`\nconst DropdownGroupWrapper = styled.div`\n  position: sticky;\n  top: 0px;\n`\nconst DropdownItem = styled.div<{\n  'aria-selected': boolean\n  'aria-disabled': boolean\n}>`\n  text-align:left;\n  border: none;\n  background-color: ${({ theme }) => theme.colors.other.elevation.background.raised};\n\n  padding: ${({ theme }) => theme.space['1.5']} ${({ theme }) =>\n    theme.space['2']} ${({ theme }) => theme.space['1.5']} ${({ theme }) =>\n    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  &[aria-selected='true'] {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n  }\n\n  &[aria-disabled=\"true\"] {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n  }\n\n  &[aria-disabled=\"true\"]:hover, [aria-disabled=\"true\"]:focus {\n    background-color: ${({ theme }) =>\n      theme.colors.neutral.backgroundStrongDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textStrongDisabled};\n    cursor: not-allowed;\n    outline: none;\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 > div[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 > div[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 = (event: KeyboardEvent<HTMLDivElement>) => {\n  if (event.key === 'ArrowDown') {\n    event.preventDefault()\n    moveFocusDown()\n  }\n\n  if (event.key === 'ArrowUp') {\n    event.preventDefault()\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)\n    onSearch(options)\n  }\n}\n\nconst handleKeyDown = (\n  event: globalThis.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<HTMLDivElement>(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={handleKeyDownSelect}\n      data-grouped\n    >\n      {isLoading ? (\n        <Skeleton variant=\"block\" />\n      ) : (\n        <>\n          {selectAll && multiselect ? (\n            <Stack id=\"items\">\n              <DropdownItem\n                aria-disabled={false}\n                aria-selected={selectedData.allSelected}\n                aria-label=\"select-all\"\n                data-testid=\"select-all\"\n                id=\"select-all\"\n                tabIndex={0}\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                  {group ? (\n                    <DropdownGroup\n                      key={group}\n                      type=\"button\"\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\n                          variant=\"caption\"\n                          as=\"span\"\n                          placement=\"left\"\n                          sentiment=\"neutral\"\n                        >\n                          {group.toUpperCase()}\n                        </Text>\n                      )}\n                    </DropdownGroup>\n                  ) : null}\n                </DropdownGroupWrapper>\n              ) : null}\n              <Stack id=\"items\" gap=\"0.25\">\n                {displayedOptions[group].map((option, indexOption) => (\n                  <DropdownItem\n                    key={option.value}\n                    aria-disabled={!!option.disabled}\n                    tabIndex={!option.disabled ? 0 : -1}\n                    aria-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              </Stack>\n            </Stack>\n          ))}\n        </>\n      )}\n      {loadMore ? <LoadMore>{loadMore}</LoadMore> : null}\n    </DropdownContainer>\n  ) : (\n    <DropdownContainer\n      role=\"listbox\"\n      tabIndex={-1}\n      id=\"select-dropdown\"\n      onKeyDown={handleKeyDownSelect}\n      gap={0.25}\n      data-grouped={false}\n    >\n      {selectAll && multiselect ? (\n        <Stack id=\"items\" gap={0.25} tabIndex={-1}>\n          <DropdownItem\n            aria-disabled={false}\n            aria-selected={selectedData.allSelected}\n            aria-label=\"select-all\"\n            data-testid=\"select-all\"\n            tabIndex={0}\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              aria-disabled={!!option.disabled}\n              aria-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              tabIndex={!option.disabled ? 0 : -1}\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  size,\n}: DropdownProps) => {\n  const {\n    setIsDropdownVisible,\n    isDropdownVisible,\n    onSearch,\n    searchInput,\n    options,\n    displayedOptions,\n    numberOfOptions,\n  } = useSelectInput()\n  const theme = useTheme()\n  const [searchBarActive, setSearchBarActive] = useState(false)\n  const [defaultSearchValue, setDefaultSearch] = useState<string | null>(null)\n  const ref = useRef<HTMLDivElement>(null)\n  const [search, setSearch] = useState<string>('')\n  const [maxWidth, setWidth] = useState<string | number>()\n\n  useEffect(() => {\n    if (refSelect.current && isDropdownVisible) {\n      const position =\n        refSelect.current.getBoundingClientRect().bottom +\n        DROPDOWN_MAX_HEIGHT +\n        INPUT_SIZE_HEIGHT[size] +\n        parseInt(theme.space['5'], 10)\n      const overflow = position - window.innerHeight\n      if (overflow > 0) {\n        const modalElement = document.getElementById('backdrop-modal')\n\n        if (modalElement) {\n          modalElement.scrollBy({ top: overflow, behavior: 'smooth' })\n        } else window.scrollBy({ top: overflow, behavior: 'smooth' })\n      }\n    }\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, [isDropdownVisible, refSelect, size, ref.current])\n\n  const resizeDropdown = useCallback(() => {\n    if (\n      refSelect.current &&\n      refSelect.current.getBoundingClientRect().width > 0\n    ) {\n      setWidth(refSelect.current.getBoundingClientRect().width)\n    }\n  }, [refSelect])\n\n  useEffect(() => {\n    resizeDropdown()\n\n    window.addEventListener('resize', resizeDropdown)\n\n    return () => window.removeEventListener('resize', resizeDropdown)\n  }, [resizeDropdown])\n\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\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  // No data is displayed (because of the search or because no data is provided)\n  // Set to true when noData by default\n  const isEmpty = useMemo(() => {\n    if (numberOfOptions === 0) return true\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, numberOfOptions])\n\n  const computedFooter = useMemo(() => {\n    if (footer) {\n      if (typeof footer === 'function') {\n        return (\n          <PopupFooter>{footer(() => setIsDropdownVisible(false))}</PopupFooter>\n        )\n      }\n\n      return <PopupFooter>{footer}</PopupFooter>\n    }\n\n    return null\n  }, [footer, setIsDropdownVisible])\n\n  return (\n    <StyledPopup\n      visible={isDropdownVisible}\n      text={\n        <Stack>\n          {searchable && !isLoading && numberOfOptions >= 6 ? (\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          {computedFooter}\n        </Stack>\n      }\n      placement=\"bottom\"\n      disableAnimation\n      maxWidth={maxWidth ?? refSelect.current?.offsetWidth}\n      hasArrow={false}\n      ref={ref}\n      tabIndex={0}\n      role=\"dialog\"\n      debounceDelay={0}\n      containerFullWidth\n      align=\"start\"\n    >\n      {children}\n    </StyledPopup>\n  )\n}\n"]} */"));
53
53
  const DropdownGroup = /* @__PURE__ */ _styled__default.default("button", process.env.NODE_ENV === "production" ? {
54
54
  target: "eeucz6c6"
55
55
  } : {
@@ -75,7 +75,7 @@ const DropdownGroup = /* @__PURE__ */ _styled__default.default("button", process
75
75
  theme
76
76
  }) => theme.colors.neutral.backgroundWeak, ";}&[data-selectgroup='true']:focus{background-color:", ({
77
77
  theme
78
- }) => 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":"AAgFoE","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/Dropdown.tsx","sourcesContent":["import { useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport { useCallback, useEffect, useMemo, useRef, useState } from 'react'\nimport type {\n  Dispatch,\n  KeyboardEvent,\n  ReactNode,\n  RefObject,\n  SetStateAction,\n} 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, INPUT_SIZE_HEIGHT, type OptionType } from './types'\n\nconst DROPDOWN_MAX_HEIGHT = 256\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  size: 'small' | 'medium' | 'large'\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  min-width: 320px;\n  background-color: ${({ theme }) => theme.colors.other.elevation.background.raised};\n  color: ${({ theme }) => theme.colors.neutral.text};\n  box-shadow: ${({ theme }) => `${theme.shadows.raised[0]}, ${theme.shadows.raised[1]}`};\n  padding: ${({ theme }) => theme.space[0]};\n  margin-bottom: ${({ theme }) => theme.space[10]};\n`\n\nconst DropdownContainer = styled(Stack)<{ 'data-grouped': boolean }>`\n  max-height: ${DROPDOWN_MAX_HEIGHT}px;\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 }) =>\n      theme.colors.neutral.backgroundWeak};\n  }\n\n  &[data-selectgroup='true']:focus {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundHover};\n  }\n`\nconst DropdownGroupWrapper = styled.div`\n  position: sticky;\n  top: 0px;\n`\nconst DropdownItem = styled.div<{\n  'aria-selected': boolean\n  'aria-disabled': boolean\n}>`\n  text-align:left;\n  border: none;\n  background-color: ${({ theme }) => theme.colors.other.elevation.background.raised};\n\n  padding: ${({ theme }) => theme.space['1.5']} ${({ theme }) =>\n    theme.space['2']} ${({ theme }) => theme.space['1.5']} ${({ theme }) =>\n    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  &[aria-selected='true'] {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n  }\n\n  &[aria-disabled=\"true\"] {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n  }\n\n  &[aria-disabled=\"true\"]:hover, [aria-disabled=\"true\"]:focus {\n    background-color: ${({ theme }) =>\n      theme.colors.neutral.backgroundStrongDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textStrongDisabled};\n    cursor: not-allowed;\n    outline: none;\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 > div[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 > div[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 = (event: KeyboardEvent<HTMLDivElement>) => {\n  if (event.key === 'ArrowDown') {\n    event.preventDefault()\n    moveFocusDown()\n  }\n\n  if (event.key === 'ArrowUp') {\n    event.preventDefault()\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: globalThis.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<HTMLDivElement>(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={handleKeyDownSelect}\n      data-grouped\n    >\n      {isLoading ? (\n        <Skeleton variant=\"block\" />\n      ) : (\n        <>\n          {selectAll && multiselect ? (\n            <Stack id=\"items\">\n              <DropdownItem\n                aria-disabled={false}\n                aria-selected={selectedData.allSelected}\n                aria-label=\"select-all\"\n                data-testid=\"select-all\"\n                id=\"select-all\"\n                tabIndex={0}\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                  {group ? (\n                    <DropdownGroup\n                      key={group}\n                      type=\"button\"\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\n                          variant=\"caption\"\n                          as=\"span\"\n                          placement=\"left\"\n                          sentiment=\"neutral\"\n                        >\n                          {group.toUpperCase()}\n                        </Text>\n                      )}\n                    </DropdownGroup>\n                  ) : null}\n                </DropdownGroupWrapper>\n              ) : null}\n              <Stack id=\"items\" gap=\"0.25\">\n                {displayedOptions[group].map((option, indexOption) => (\n                  <DropdownItem\n                    key={option.value}\n                    aria-disabled={!!option.disabled}\n                    tabIndex={!option.disabled ? 0 : -1}\n                    aria-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              </Stack>\n            </Stack>\n          ))}\n        </>\n      )}\n      {loadMore ? <LoadMore>{loadMore}</LoadMore> : null}\n    </DropdownContainer>\n  ) : (\n    <DropdownContainer\n      role=\"listbox\"\n      tabIndex={-1}\n      id=\"select-dropdown\"\n      onKeyDown={handleKeyDownSelect}\n      gap={0.25}\n      data-grouped={false}\n    >\n      {selectAll && multiselect ? (\n        <Stack id=\"items\" gap={0.25} tabIndex={-1}>\n          <DropdownItem\n            aria-disabled={false}\n            aria-selected={selectedData.allSelected}\n            aria-label=\"select-all\"\n            data-testid=\"select-all\"\n            tabIndex={0}\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              aria-disabled={!!option.disabled}\n              aria-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              tabIndex={!option.disabled ? 0 : -1}\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  size,\n}: DropdownProps) => {\n  const {\n    setIsDropdownVisible,\n    isDropdownVisible,\n    onSearch,\n    searchInput,\n    options,\n    displayedOptions,\n  } = useSelectInput()\n  const theme = useTheme()\n  const [searchBarActive, setSearchBarActive] = useState(false)\n  const [defaultSearchValue, setDefaultSearch] = useState<string | null>(null)\n  const ref = useRef<HTMLDivElement>(null)\n  const [search, setSearch] = useState<string>('')\n  const [maxWidth, setWidth] = useState<string | number>()\n\n  useEffect(() => {\n    if (refSelect.current && isDropdownVisible) {\n      const position =\n        refSelect.current.getBoundingClientRect().bottom +\n        DROPDOWN_MAX_HEIGHT +\n        INPUT_SIZE_HEIGHT[size] +\n        parseInt(theme.space['5'], 10)\n      const overflow = position - window.innerHeight\n      if (overflow > 0) {\n        const modalElement = document.getElementById('backdrop-modal')\n\n        if (modalElement) {\n          modalElement.scrollBy({ top: overflow, behavior: 'smooth' })\n        } else window.scrollBy({ top: overflow, behavior: 'smooth' })\n      }\n    }\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, [isDropdownVisible, refSelect, size, ref.current])\n\n  const resizeDropdown = useCallback(() => {\n    if (\n      refSelect.current &&\n      refSelect.current.getBoundingClientRect().width > 0\n    ) {\n      setWidth(refSelect.current.getBoundingClientRect().width)\n    }\n  }, [refSelect])\n\n  useEffect(() => {\n    resizeDropdown()\n\n    window.addEventListener('resize', resizeDropdown)\n\n    return () => window.removeEventListener('resize', resizeDropdown)\n  }, [resizeDropdown])\n\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      disableAnimation\n      maxWidth={maxWidth ?? refSelect.current?.offsetWidth}\n      hasArrow={false}\n      ref={ref}\n      tabIndex={0}\n      role=\"dialog\"\n      debounceDelay={0}\n      containerFullWidth\n    >\n      {children}\n    </StyledPopup>\n  )\n}\n"]} */"));
78
+ }) => 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":"AAgFoE","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/Dropdown.tsx","sourcesContent":["import { useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport { useCallback, useEffect, useMemo, useRef, useState } from 'react'\nimport type {\n  Dispatch,\n  KeyboardEvent,\n  ReactNode,\n  RefObject,\n  SetStateAction,\n} 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, INPUT_SIZE_HEIGHT, type OptionType } from './types'\n\nconst DROPDOWN_MAX_HEIGHT = 256\n\nexport type DropdownProps = {\n  children: ReactNode\n  emptyState: ReactNode\n  descriptionDirection: 'row' | 'column'\n  searchable: boolean\n  placeholder: string\n  footer?: ((closeDropdown: () => void) => ReactNode) | ReactNode\n  refSelect: RefObject<HTMLDivElement>\n  loadMore?: ReactNode\n  optionalInfoPlacement: 'left' | 'right'\n  isLoading?: boolean\n  size: 'small' | 'medium' | 'large'\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  min-width: 320px;\n  background-color: ${({ theme }) => theme.colors.other.elevation.background.raised};\n  color: ${({ theme }) => theme.colors.neutral.text};\n  box-shadow: ${({ theme }) => `${theme.shadows.raised[0]}, ${theme.shadows.raised[1]}`};\n  padding: ${({ theme }) => theme.space[0]};\n  margin-bottom: ${({ theme }) => theme.space[10]};\n`\n\nconst DropdownContainer = styled(Stack)<{ 'data-grouped': boolean }>`\n  max-height: ${DROPDOWN_MAX_HEIGHT}px;\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 }) =>\n      theme.colors.neutral.backgroundWeak};\n  }\n\n  &[data-selectgroup='true']:focus {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundHover};\n  }\n`\nconst DropdownGroupWrapper = styled.div`\n  position: sticky;\n  top: 0px;\n`\nconst DropdownItem = styled.div<{\n  'aria-selected': boolean\n  'aria-disabled': boolean\n}>`\n  text-align:left;\n  border: none;\n  background-color: ${({ theme }) => theme.colors.other.elevation.background.raised};\n\n  padding: ${({ theme }) => theme.space['1.5']} ${({ theme }) =>\n    theme.space['2']} ${({ theme }) => theme.space['1.5']} ${({ theme }) =>\n    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  &[aria-selected='true'] {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n  }\n\n  &[aria-disabled=\"true\"] {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n  }\n\n  &[aria-disabled=\"true\"]:hover, [aria-disabled=\"true\"]:focus {\n    background-color: ${({ theme }) =>\n      theme.colors.neutral.backgroundStrongDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textStrongDisabled};\n    cursor: not-allowed;\n    outline: none;\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 > div[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 > div[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 = (event: KeyboardEvent<HTMLDivElement>) => {\n  if (event.key === 'ArrowDown') {\n    event.preventDefault()\n    moveFocusDown()\n  }\n\n  if (event.key === 'ArrowUp') {\n    event.preventDefault()\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)\n    onSearch(options)\n  }\n}\n\nconst handleKeyDown = (\n  event: globalThis.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<HTMLDivElement>(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={handleKeyDownSelect}\n      data-grouped\n    >\n      {isLoading ? (\n        <Skeleton variant=\"block\" />\n      ) : (\n        <>\n          {selectAll && multiselect ? (\n            <Stack id=\"items\">\n              <DropdownItem\n                aria-disabled={false}\n                aria-selected={selectedData.allSelected}\n                aria-label=\"select-all\"\n                data-testid=\"select-all\"\n                id=\"select-all\"\n                tabIndex={0}\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                  {group ? (\n                    <DropdownGroup\n                      key={group}\n                      type=\"button\"\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\n                          variant=\"caption\"\n                          as=\"span\"\n                          placement=\"left\"\n                          sentiment=\"neutral\"\n                        >\n                          {group.toUpperCase()}\n                        </Text>\n                      )}\n                    </DropdownGroup>\n                  ) : null}\n                </DropdownGroupWrapper>\n              ) : null}\n              <Stack id=\"items\" gap=\"0.25\">\n                {displayedOptions[group].map((option, indexOption) => (\n                  <DropdownItem\n                    key={option.value}\n                    aria-disabled={!!option.disabled}\n                    tabIndex={!option.disabled ? 0 : -1}\n                    aria-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              </Stack>\n            </Stack>\n          ))}\n        </>\n      )}\n      {loadMore ? <LoadMore>{loadMore}</LoadMore> : null}\n    </DropdownContainer>\n  ) : (\n    <DropdownContainer\n      role=\"listbox\"\n      tabIndex={-1}\n      id=\"select-dropdown\"\n      onKeyDown={handleKeyDownSelect}\n      gap={0.25}\n      data-grouped={false}\n    >\n      {selectAll && multiselect ? (\n        <Stack id=\"items\" gap={0.25} tabIndex={-1}>\n          <DropdownItem\n            aria-disabled={false}\n            aria-selected={selectedData.allSelected}\n            aria-label=\"select-all\"\n            data-testid=\"select-all\"\n            tabIndex={0}\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              aria-disabled={!!option.disabled}\n              aria-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              tabIndex={!option.disabled ? 0 : -1}\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  size,\n}: DropdownProps) => {\n  const {\n    setIsDropdownVisible,\n    isDropdownVisible,\n    onSearch,\n    searchInput,\n    options,\n    displayedOptions,\n    numberOfOptions,\n  } = useSelectInput()\n  const theme = useTheme()\n  const [searchBarActive, setSearchBarActive] = useState(false)\n  const [defaultSearchValue, setDefaultSearch] = useState<string | null>(null)\n  const ref = useRef<HTMLDivElement>(null)\n  const [search, setSearch] = useState<string>('')\n  const [maxWidth, setWidth] = useState<string | number>()\n\n  useEffect(() => {\n    if (refSelect.current && isDropdownVisible) {\n      const position =\n        refSelect.current.getBoundingClientRect().bottom +\n        DROPDOWN_MAX_HEIGHT +\n        INPUT_SIZE_HEIGHT[size] +\n        parseInt(theme.space['5'], 10)\n      const overflow = position - window.innerHeight\n      if (overflow > 0) {\n        const modalElement = document.getElementById('backdrop-modal')\n\n        if (modalElement) {\n          modalElement.scrollBy({ top: overflow, behavior: 'smooth' })\n        } else window.scrollBy({ top: overflow, behavior: 'smooth' })\n      }\n    }\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, [isDropdownVisible, refSelect, size, ref.current])\n\n  const resizeDropdown = useCallback(() => {\n    if (\n      refSelect.current &&\n      refSelect.current.getBoundingClientRect().width > 0\n    ) {\n      setWidth(refSelect.current.getBoundingClientRect().width)\n    }\n  }, [refSelect])\n\n  useEffect(() => {\n    resizeDropdown()\n\n    window.addEventListener('resize', resizeDropdown)\n\n    return () => window.removeEventListener('resize', resizeDropdown)\n  }, [resizeDropdown])\n\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\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  // No data is displayed (because of the search or because no data is provided)\n  // Set to true when noData by default\n  const isEmpty = useMemo(() => {\n    if (numberOfOptions === 0) return true\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, numberOfOptions])\n\n  const computedFooter = useMemo(() => {\n    if (footer) {\n      if (typeof footer === 'function') {\n        return (\n          <PopupFooter>{footer(() => setIsDropdownVisible(false))}</PopupFooter>\n        )\n      }\n\n      return <PopupFooter>{footer}</PopupFooter>\n    }\n\n    return null\n  }, [footer, setIsDropdownVisible])\n\n  return (\n    <StyledPopup\n      visible={isDropdownVisible}\n      text={\n        <Stack>\n          {searchable && !isLoading && numberOfOptions >= 6 ? (\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          {computedFooter}\n        </Stack>\n      }\n      placement=\"bottom\"\n      disableAnimation\n      maxWidth={maxWidth ?? refSelect.current?.offsetWidth}\n      hasArrow={false}\n      ref={ref}\n      tabIndex={0}\n      role=\"dialog\"\n      debounceDelay={0}\n      containerFullWidth\n      align=\"start\"\n    >\n      {children}\n    </StyledPopup>\n  )\n}\n"]} */"));
79
79
  const DropdownGroupWrapper = /* @__PURE__ */ _styled__default.default("div", process.env.NODE_ENV === "production" ? {
80
80
  target: "eeucz6c5"
81
81
  } : {
@@ -87,7 +87,7 @@ const DropdownGroupWrapper = /* @__PURE__ */ _styled__default.default("div", pro
87
87
  } : {
88
88
  name: "vkf8h3",
89
89
  styles: "position:sticky;top:0px",
90
- 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":"AA8GuC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/Dropdown.tsx","sourcesContent":["import { useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport { useCallback, useEffect, useMemo, useRef, useState } from 'react'\nimport type {\n  Dispatch,\n  KeyboardEvent,\n  ReactNode,\n  RefObject,\n  SetStateAction,\n} 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, INPUT_SIZE_HEIGHT, type OptionType } from './types'\n\nconst DROPDOWN_MAX_HEIGHT = 256\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  size: 'small' | 'medium' | 'large'\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  min-width: 320px;\n  background-color: ${({ theme }) => theme.colors.other.elevation.background.raised};\n  color: ${({ theme }) => theme.colors.neutral.text};\n  box-shadow: ${({ theme }) => `${theme.shadows.raised[0]}, ${theme.shadows.raised[1]}`};\n  padding: ${({ theme }) => theme.space[0]};\n  margin-bottom: ${({ theme }) => theme.space[10]};\n`\n\nconst DropdownContainer = styled(Stack)<{ 'data-grouped': boolean }>`\n  max-height: ${DROPDOWN_MAX_HEIGHT}px;\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 }) =>\n      theme.colors.neutral.backgroundWeak};\n  }\n\n  &[data-selectgroup='true']:focus {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundHover};\n  }\n`\nconst DropdownGroupWrapper = styled.div`\n  position: sticky;\n  top: 0px;\n`\nconst DropdownItem = styled.div<{\n  'aria-selected': boolean\n  'aria-disabled': boolean\n}>`\n  text-align:left;\n  border: none;\n  background-color: ${({ theme }) => theme.colors.other.elevation.background.raised};\n\n  padding: ${({ theme }) => theme.space['1.5']} ${({ theme }) =>\n    theme.space['2']} ${({ theme }) => theme.space['1.5']} ${({ theme }) =>\n    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  &[aria-selected='true'] {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n  }\n\n  &[aria-disabled=\"true\"] {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n  }\n\n  &[aria-disabled=\"true\"]:hover, [aria-disabled=\"true\"]:focus {\n    background-color: ${({ theme }) =>\n      theme.colors.neutral.backgroundStrongDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textStrongDisabled};\n    cursor: not-allowed;\n    outline: none;\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 > div[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 > div[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 = (event: KeyboardEvent<HTMLDivElement>) => {\n  if (event.key === 'ArrowDown') {\n    event.preventDefault()\n    moveFocusDown()\n  }\n\n  if (event.key === 'ArrowUp') {\n    event.preventDefault()\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: globalThis.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<HTMLDivElement>(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={handleKeyDownSelect}\n      data-grouped\n    >\n      {isLoading ? (\n        <Skeleton variant=\"block\" />\n      ) : (\n        <>\n          {selectAll && multiselect ? (\n            <Stack id=\"items\">\n              <DropdownItem\n                aria-disabled={false}\n                aria-selected={selectedData.allSelected}\n                aria-label=\"select-all\"\n                data-testid=\"select-all\"\n                id=\"select-all\"\n                tabIndex={0}\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                  {group ? (\n                    <DropdownGroup\n                      key={group}\n                      type=\"button\"\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\n                          variant=\"caption\"\n                          as=\"span\"\n                          placement=\"left\"\n                          sentiment=\"neutral\"\n                        >\n                          {group.toUpperCase()}\n                        </Text>\n                      )}\n                    </DropdownGroup>\n                  ) : null}\n                </DropdownGroupWrapper>\n              ) : null}\n              <Stack id=\"items\" gap=\"0.25\">\n                {displayedOptions[group].map((option, indexOption) => (\n                  <DropdownItem\n                    key={option.value}\n                    aria-disabled={!!option.disabled}\n                    tabIndex={!option.disabled ? 0 : -1}\n                    aria-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              </Stack>\n            </Stack>\n          ))}\n        </>\n      )}\n      {loadMore ? <LoadMore>{loadMore}</LoadMore> : null}\n    </DropdownContainer>\n  ) : (\n    <DropdownContainer\n      role=\"listbox\"\n      tabIndex={-1}\n      id=\"select-dropdown\"\n      onKeyDown={handleKeyDownSelect}\n      gap={0.25}\n      data-grouped={false}\n    >\n      {selectAll && multiselect ? (\n        <Stack id=\"items\" gap={0.25} tabIndex={-1}>\n          <DropdownItem\n            aria-disabled={false}\n            aria-selected={selectedData.allSelected}\n            aria-label=\"select-all\"\n            data-testid=\"select-all\"\n            tabIndex={0}\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              aria-disabled={!!option.disabled}\n              aria-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              tabIndex={!option.disabled ? 0 : -1}\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  size,\n}: DropdownProps) => {\n  const {\n    setIsDropdownVisible,\n    isDropdownVisible,\n    onSearch,\n    searchInput,\n    options,\n    displayedOptions,\n  } = useSelectInput()\n  const theme = useTheme()\n  const [searchBarActive, setSearchBarActive] = useState(false)\n  const [defaultSearchValue, setDefaultSearch] = useState<string | null>(null)\n  const ref = useRef<HTMLDivElement>(null)\n  const [search, setSearch] = useState<string>('')\n  const [maxWidth, setWidth] = useState<string | number>()\n\n  useEffect(() => {\n    if (refSelect.current && isDropdownVisible) {\n      const position =\n        refSelect.current.getBoundingClientRect().bottom +\n        DROPDOWN_MAX_HEIGHT +\n        INPUT_SIZE_HEIGHT[size] +\n        parseInt(theme.space['5'], 10)\n      const overflow = position - window.innerHeight\n      if (overflow > 0) {\n        const modalElement = document.getElementById('backdrop-modal')\n\n        if (modalElement) {\n          modalElement.scrollBy({ top: overflow, behavior: 'smooth' })\n        } else window.scrollBy({ top: overflow, behavior: 'smooth' })\n      }\n    }\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, [isDropdownVisible, refSelect, size, ref.current])\n\n  const resizeDropdown = useCallback(() => {\n    if (\n      refSelect.current &&\n      refSelect.current.getBoundingClientRect().width > 0\n    ) {\n      setWidth(refSelect.current.getBoundingClientRect().width)\n    }\n  }, [refSelect])\n\n  useEffect(() => {\n    resizeDropdown()\n\n    window.addEventListener('resize', resizeDropdown)\n\n    return () => window.removeEventListener('resize', resizeDropdown)\n  }, [resizeDropdown])\n\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      disableAnimation\n      maxWidth={maxWidth ?? refSelect.current?.offsetWidth}\n      hasArrow={false}\n      ref={ref}\n      tabIndex={0}\n      role=\"dialog\"\n      debounceDelay={0}\n      containerFullWidth\n    >\n      {children}\n    </StyledPopup>\n  )\n}\n"]} */",
90
+ 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":"AA8GuC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/Dropdown.tsx","sourcesContent":["import { useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport { useCallback, useEffect, useMemo, useRef, useState } from 'react'\nimport type {\n  Dispatch,\n  KeyboardEvent,\n  ReactNode,\n  RefObject,\n  SetStateAction,\n} 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, INPUT_SIZE_HEIGHT, type OptionType } from './types'\n\nconst DROPDOWN_MAX_HEIGHT = 256\n\nexport type DropdownProps = {\n  children: ReactNode\n  emptyState: ReactNode\n  descriptionDirection: 'row' | 'column'\n  searchable: boolean\n  placeholder: string\n  footer?: ((closeDropdown: () => void) => ReactNode) | ReactNode\n  refSelect: RefObject<HTMLDivElement>\n  loadMore?: ReactNode\n  optionalInfoPlacement: 'left' | 'right'\n  isLoading?: boolean\n  size: 'small' | 'medium' | 'large'\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  min-width: 320px;\n  background-color: ${({ theme }) => theme.colors.other.elevation.background.raised};\n  color: ${({ theme }) => theme.colors.neutral.text};\n  box-shadow: ${({ theme }) => `${theme.shadows.raised[0]}, ${theme.shadows.raised[1]}`};\n  padding: ${({ theme }) => theme.space[0]};\n  margin-bottom: ${({ theme }) => theme.space[10]};\n`\n\nconst DropdownContainer = styled(Stack)<{ 'data-grouped': boolean }>`\n  max-height: ${DROPDOWN_MAX_HEIGHT}px;\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 }) =>\n      theme.colors.neutral.backgroundWeak};\n  }\n\n  &[data-selectgroup='true']:focus {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundHover};\n  }\n`\nconst DropdownGroupWrapper = styled.div`\n  position: sticky;\n  top: 0px;\n`\nconst DropdownItem = styled.div<{\n  'aria-selected': boolean\n  'aria-disabled': boolean\n}>`\n  text-align:left;\n  border: none;\n  background-color: ${({ theme }) => theme.colors.other.elevation.background.raised};\n\n  padding: ${({ theme }) => theme.space['1.5']} ${({ theme }) =>\n    theme.space['2']} ${({ theme }) => theme.space['1.5']} ${({ theme }) =>\n    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  &[aria-selected='true'] {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n  }\n\n  &[aria-disabled=\"true\"] {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n  }\n\n  &[aria-disabled=\"true\"]:hover, [aria-disabled=\"true\"]:focus {\n    background-color: ${({ theme }) =>\n      theme.colors.neutral.backgroundStrongDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textStrongDisabled};\n    cursor: not-allowed;\n    outline: none;\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 > div[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 > div[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 = (event: KeyboardEvent<HTMLDivElement>) => {\n  if (event.key === 'ArrowDown') {\n    event.preventDefault()\n    moveFocusDown()\n  }\n\n  if (event.key === 'ArrowUp') {\n    event.preventDefault()\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)\n    onSearch(options)\n  }\n}\n\nconst handleKeyDown = (\n  event: globalThis.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<HTMLDivElement>(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={handleKeyDownSelect}\n      data-grouped\n    >\n      {isLoading ? (\n        <Skeleton variant=\"block\" />\n      ) : (\n        <>\n          {selectAll && multiselect ? (\n            <Stack id=\"items\">\n              <DropdownItem\n                aria-disabled={false}\n                aria-selected={selectedData.allSelected}\n                aria-label=\"select-all\"\n                data-testid=\"select-all\"\n                id=\"select-all\"\n                tabIndex={0}\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                  {group ? (\n                    <DropdownGroup\n                      key={group}\n                      type=\"button\"\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\n                          variant=\"caption\"\n                          as=\"span\"\n                          placement=\"left\"\n                          sentiment=\"neutral\"\n                        >\n                          {group.toUpperCase()}\n                        </Text>\n                      )}\n                    </DropdownGroup>\n                  ) : null}\n                </DropdownGroupWrapper>\n              ) : null}\n              <Stack id=\"items\" gap=\"0.25\">\n                {displayedOptions[group].map((option, indexOption) => (\n                  <DropdownItem\n                    key={option.value}\n                    aria-disabled={!!option.disabled}\n                    tabIndex={!option.disabled ? 0 : -1}\n                    aria-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              </Stack>\n            </Stack>\n          ))}\n        </>\n      )}\n      {loadMore ? <LoadMore>{loadMore}</LoadMore> : null}\n    </DropdownContainer>\n  ) : (\n    <DropdownContainer\n      role=\"listbox\"\n      tabIndex={-1}\n      id=\"select-dropdown\"\n      onKeyDown={handleKeyDownSelect}\n      gap={0.25}\n      data-grouped={false}\n    >\n      {selectAll && multiselect ? (\n        <Stack id=\"items\" gap={0.25} tabIndex={-1}>\n          <DropdownItem\n            aria-disabled={false}\n            aria-selected={selectedData.allSelected}\n            aria-label=\"select-all\"\n            data-testid=\"select-all\"\n            tabIndex={0}\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              aria-disabled={!!option.disabled}\n              aria-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              tabIndex={!option.disabled ? 0 : -1}\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  size,\n}: DropdownProps) => {\n  const {\n    setIsDropdownVisible,\n    isDropdownVisible,\n    onSearch,\n    searchInput,\n    options,\n    displayedOptions,\n    numberOfOptions,\n  } = useSelectInput()\n  const theme = useTheme()\n  const [searchBarActive, setSearchBarActive] = useState(false)\n  const [defaultSearchValue, setDefaultSearch] = useState<string | null>(null)\n  const ref = useRef<HTMLDivElement>(null)\n  const [search, setSearch] = useState<string>('')\n  const [maxWidth, setWidth] = useState<string | number>()\n\n  useEffect(() => {\n    if (refSelect.current && isDropdownVisible) {\n      const position =\n        refSelect.current.getBoundingClientRect().bottom +\n        DROPDOWN_MAX_HEIGHT +\n        INPUT_SIZE_HEIGHT[size] +\n        parseInt(theme.space['5'], 10)\n      const overflow = position - window.innerHeight\n      if (overflow > 0) {\n        const modalElement = document.getElementById('backdrop-modal')\n\n        if (modalElement) {\n          modalElement.scrollBy({ top: overflow, behavior: 'smooth' })\n        } else window.scrollBy({ top: overflow, behavior: 'smooth' })\n      }\n    }\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, [isDropdownVisible, refSelect, size, ref.current])\n\n  const resizeDropdown = useCallback(() => {\n    if (\n      refSelect.current &&\n      refSelect.current.getBoundingClientRect().width > 0\n    ) {\n      setWidth(refSelect.current.getBoundingClientRect().width)\n    }\n  }, [refSelect])\n\n  useEffect(() => {\n    resizeDropdown()\n\n    window.addEventListener('resize', resizeDropdown)\n\n    return () => window.removeEventListener('resize', resizeDropdown)\n  }, [resizeDropdown])\n\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\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  // No data is displayed (because of the search or because no data is provided)\n  // Set to true when noData by default\n  const isEmpty = useMemo(() => {\n    if (numberOfOptions === 0) return true\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, numberOfOptions])\n\n  const computedFooter = useMemo(() => {\n    if (footer) {\n      if (typeof footer === 'function') {\n        return (\n          <PopupFooter>{footer(() => setIsDropdownVisible(false))}</PopupFooter>\n        )\n      }\n\n      return <PopupFooter>{footer}</PopupFooter>\n    }\n\n    return null\n  }, [footer, setIsDropdownVisible])\n\n  return (\n    <StyledPopup\n      visible={isDropdownVisible}\n      text={\n        <Stack>\n          {searchable && !isLoading && numberOfOptions >= 6 ? (\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          {computedFooter}\n        </Stack>\n      }\n      placement=\"bottom\"\n      disableAnimation\n      maxWidth={maxWidth ?? refSelect.current?.offsetWidth}\n      hasArrow={false}\n      ref={ref}\n      tabIndex={0}\n      role=\"dialog\"\n      debounceDelay={0}\n      containerFullWidth\n      align=\"start\"\n    >\n      {children}\n    </StyledPopup>\n  )\n}\n"]} */",
91
91
  toString: _EMOTION_STRINGIFIED_CSS_ERROR__
92
92
  });
93
93
  const DropdownItem = /* @__PURE__ */ _styled__default.default("div", process.env.NODE_ENV === "production" ? {
@@ -127,7 +127,7 @@ const DropdownItem = /* @__PURE__ */ _styled__default.default("div", process.env
127
127
  theme
128
128
  }) => theme.colors.neutral.backgroundStrongDisabled, ";color:", ({
129
129
  theme
130
- }) => 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":"AAqHE","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/Dropdown.tsx","sourcesContent":["import { useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport { useCallback, useEffect, useMemo, useRef, useState } from 'react'\nimport type {\n  Dispatch,\n  KeyboardEvent,\n  ReactNode,\n  RefObject,\n  SetStateAction,\n} 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, INPUT_SIZE_HEIGHT, type OptionType } from './types'\n\nconst DROPDOWN_MAX_HEIGHT = 256\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  size: 'small' | 'medium' | 'large'\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  min-width: 320px;\n  background-color: ${({ theme }) => theme.colors.other.elevation.background.raised};\n  color: ${({ theme }) => theme.colors.neutral.text};\n  box-shadow: ${({ theme }) => `${theme.shadows.raised[0]}, ${theme.shadows.raised[1]}`};\n  padding: ${({ theme }) => theme.space[0]};\n  margin-bottom: ${({ theme }) => theme.space[10]};\n`\n\nconst DropdownContainer = styled(Stack)<{ 'data-grouped': boolean }>`\n  max-height: ${DROPDOWN_MAX_HEIGHT}px;\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 }) =>\n      theme.colors.neutral.backgroundWeak};\n  }\n\n  &[data-selectgroup='true']:focus {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundHover};\n  }\n`\nconst DropdownGroupWrapper = styled.div`\n  position: sticky;\n  top: 0px;\n`\nconst DropdownItem = styled.div<{\n  'aria-selected': boolean\n  'aria-disabled': boolean\n}>`\n  text-align:left;\n  border: none;\n  background-color: ${({ theme }) => theme.colors.other.elevation.background.raised};\n\n  padding: ${({ theme }) => theme.space['1.5']} ${({ theme }) =>\n    theme.space['2']} ${({ theme }) => theme.space['1.5']} ${({ theme }) =>\n    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  &[aria-selected='true'] {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n  }\n\n  &[aria-disabled=\"true\"] {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n  }\n\n  &[aria-disabled=\"true\"]:hover, [aria-disabled=\"true\"]:focus {\n    background-color: ${({ theme }) =>\n      theme.colors.neutral.backgroundStrongDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textStrongDisabled};\n    cursor: not-allowed;\n    outline: none;\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 > div[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 > div[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 = (event: KeyboardEvent<HTMLDivElement>) => {\n  if (event.key === 'ArrowDown') {\n    event.preventDefault()\n    moveFocusDown()\n  }\n\n  if (event.key === 'ArrowUp') {\n    event.preventDefault()\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: globalThis.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<HTMLDivElement>(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={handleKeyDownSelect}\n      data-grouped\n    >\n      {isLoading ? (\n        <Skeleton variant=\"block\" />\n      ) : (\n        <>\n          {selectAll && multiselect ? (\n            <Stack id=\"items\">\n              <DropdownItem\n                aria-disabled={false}\n                aria-selected={selectedData.allSelected}\n                aria-label=\"select-all\"\n                data-testid=\"select-all\"\n                id=\"select-all\"\n                tabIndex={0}\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                  {group ? (\n                    <DropdownGroup\n                      key={group}\n                      type=\"button\"\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\n                          variant=\"caption\"\n                          as=\"span\"\n                          placement=\"left\"\n                          sentiment=\"neutral\"\n                        >\n                          {group.toUpperCase()}\n                        </Text>\n                      )}\n                    </DropdownGroup>\n                  ) : null}\n                </DropdownGroupWrapper>\n              ) : null}\n              <Stack id=\"items\" gap=\"0.25\">\n                {displayedOptions[group].map((option, indexOption) => (\n                  <DropdownItem\n                    key={option.value}\n                    aria-disabled={!!option.disabled}\n                    tabIndex={!option.disabled ? 0 : -1}\n                    aria-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              </Stack>\n            </Stack>\n          ))}\n        </>\n      )}\n      {loadMore ? <LoadMore>{loadMore}</LoadMore> : null}\n    </DropdownContainer>\n  ) : (\n    <DropdownContainer\n      role=\"listbox\"\n      tabIndex={-1}\n      id=\"select-dropdown\"\n      onKeyDown={handleKeyDownSelect}\n      gap={0.25}\n      data-grouped={false}\n    >\n      {selectAll && multiselect ? (\n        <Stack id=\"items\" gap={0.25} tabIndex={-1}>\n          <DropdownItem\n            aria-disabled={false}\n            aria-selected={selectedData.allSelected}\n            aria-label=\"select-all\"\n            data-testid=\"select-all\"\n            tabIndex={0}\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              aria-disabled={!!option.disabled}\n              aria-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              tabIndex={!option.disabled ? 0 : -1}\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  size,\n}: DropdownProps) => {\n  const {\n    setIsDropdownVisible,\n    isDropdownVisible,\n    onSearch,\n    searchInput,\n    options,\n    displayedOptions,\n  } = useSelectInput()\n  const theme = useTheme()\n  const [searchBarActive, setSearchBarActive] = useState(false)\n  const [defaultSearchValue, setDefaultSearch] = useState<string | null>(null)\n  const ref = useRef<HTMLDivElement>(null)\n  const [search, setSearch] = useState<string>('')\n  const [maxWidth, setWidth] = useState<string | number>()\n\n  useEffect(() => {\n    if (refSelect.current && isDropdownVisible) {\n      const position =\n        refSelect.current.getBoundingClientRect().bottom +\n        DROPDOWN_MAX_HEIGHT +\n        INPUT_SIZE_HEIGHT[size] +\n        parseInt(theme.space['5'], 10)\n      const overflow = position - window.innerHeight\n      if (overflow > 0) {\n        const modalElement = document.getElementById('backdrop-modal')\n\n        if (modalElement) {\n          modalElement.scrollBy({ top: overflow, behavior: 'smooth' })\n        } else window.scrollBy({ top: overflow, behavior: 'smooth' })\n      }\n    }\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, [isDropdownVisible, refSelect, size, ref.current])\n\n  const resizeDropdown = useCallback(() => {\n    if (\n      refSelect.current &&\n      refSelect.current.getBoundingClientRect().width > 0\n    ) {\n      setWidth(refSelect.current.getBoundingClientRect().width)\n    }\n  }, [refSelect])\n\n  useEffect(() => {\n    resizeDropdown()\n\n    window.addEventListener('resize', resizeDropdown)\n\n    return () => window.removeEventListener('resize', resizeDropdown)\n  }, [resizeDropdown])\n\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      disableAnimation\n      maxWidth={maxWidth ?? refSelect.current?.offsetWidth}\n      hasArrow={false}\n      ref={ref}\n      tabIndex={0}\n      role=\"dialog\"\n      debounceDelay={0}\n      containerFullWidth\n    >\n      {children}\n    </StyledPopup>\n  )\n}\n"]} */"));
130
+ }) => 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":"AAqHE","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/Dropdown.tsx","sourcesContent":["import { useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport { useCallback, useEffect, useMemo, useRef, useState } from 'react'\nimport type {\n  Dispatch,\n  KeyboardEvent,\n  ReactNode,\n  RefObject,\n  SetStateAction,\n} 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, INPUT_SIZE_HEIGHT, type OptionType } from './types'\n\nconst DROPDOWN_MAX_HEIGHT = 256\n\nexport type DropdownProps = {\n  children: ReactNode\n  emptyState: ReactNode\n  descriptionDirection: 'row' | 'column'\n  searchable: boolean\n  placeholder: string\n  footer?: ((closeDropdown: () => void) => ReactNode) | ReactNode\n  refSelect: RefObject<HTMLDivElement>\n  loadMore?: ReactNode\n  optionalInfoPlacement: 'left' | 'right'\n  isLoading?: boolean\n  size: 'small' | 'medium' | 'large'\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  min-width: 320px;\n  background-color: ${({ theme }) => theme.colors.other.elevation.background.raised};\n  color: ${({ theme }) => theme.colors.neutral.text};\n  box-shadow: ${({ theme }) => `${theme.shadows.raised[0]}, ${theme.shadows.raised[1]}`};\n  padding: ${({ theme }) => theme.space[0]};\n  margin-bottom: ${({ theme }) => theme.space[10]};\n`\n\nconst DropdownContainer = styled(Stack)<{ 'data-grouped': boolean }>`\n  max-height: ${DROPDOWN_MAX_HEIGHT}px;\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 }) =>\n      theme.colors.neutral.backgroundWeak};\n  }\n\n  &[data-selectgroup='true']:focus {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundHover};\n  }\n`\nconst DropdownGroupWrapper = styled.div`\n  position: sticky;\n  top: 0px;\n`\nconst DropdownItem = styled.div<{\n  'aria-selected': boolean\n  'aria-disabled': boolean\n}>`\n  text-align:left;\n  border: none;\n  background-color: ${({ theme }) => theme.colors.other.elevation.background.raised};\n\n  padding: ${({ theme }) => theme.space['1.5']} ${({ theme }) =>\n    theme.space['2']} ${({ theme }) => theme.space['1.5']} ${({ theme }) =>\n    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  &[aria-selected='true'] {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n  }\n\n  &[aria-disabled=\"true\"] {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n  }\n\n  &[aria-disabled=\"true\"]:hover, [aria-disabled=\"true\"]:focus {\n    background-color: ${({ theme }) =>\n      theme.colors.neutral.backgroundStrongDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textStrongDisabled};\n    cursor: not-allowed;\n    outline: none;\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 > div[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 > div[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 = (event: KeyboardEvent<HTMLDivElement>) => {\n  if (event.key === 'ArrowDown') {\n    event.preventDefault()\n    moveFocusDown()\n  }\n\n  if (event.key === 'ArrowUp') {\n    event.preventDefault()\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)\n    onSearch(options)\n  }\n}\n\nconst handleKeyDown = (\n  event: globalThis.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<HTMLDivElement>(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={handleKeyDownSelect}\n      data-grouped\n    >\n      {isLoading ? (\n        <Skeleton variant=\"block\" />\n      ) : (\n        <>\n          {selectAll && multiselect ? (\n            <Stack id=\"items\">\n              <DropdownItem\n                aria-disabled={false}\n                aria-selected={selectedData.allSelected}\n                aria-label=\"select-all\"\n                data-testid=\"select-all\"\n                id=\"select-all\"\n                tabIndex={0}\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                  {group ? (\n                    <DropdownGroup\n                      key={group}\n                      type=\"button\"\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\n                          variant=\"caption\"\n                          as=\"span\"\n                          placement=\"left\"\n                          sentiment=\"neutral\"\n                        >\n                          {group.toUpperCase()}\n                        </Text>\n                      )}\n                    </DropdownGroup>\n                  ) : null}\n                </DropdownGroupWrapper>\n              ) : null}\n              <Stack id=\"items\" gap=\"0.25\">\n                {displayedOptions[group].map((option, indexOption) => (\n                  <DropdownItem\n                    key={option.value}\n                    aria-disabled={!!option.disabled}\n                    tabIndex={!option.disabled ? 0 : -1}\n                    aria-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              </Stack>\n            </Stack>\n          ))}\n        </>\n      )}\n      {loadMore ? <LoadMore>{loadMore}</LoadMore> : null}\n    </DropdownContainer>\n  ) : (\n    <DropdownContainer\n      role=\"listbox\"\n      tabIndex={-1}\n      id=\"select-dropdown\"\n      onKeyDown={handleKeyDownSelect}\n      gap={0.25}\n      data-grouped={false}\n    >\n      {selectAll && multiselect ? (\n        <Stack id=\"items\" gap={0.25} tabIndex={-1}>\n          <DropdownItem\n            aria-disabled={false}\n            aria-selected={selectedData.allSelected}\n            aria-label=\"select-all\"\n            data-testid=\"select-all\"\n            tabIndex={0}\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              aria-disabled={!!option.disabled}\n              aria-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              tabIndex={!option.disabled ? 0 : -1}\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  size,\n}: DropdownProps) => {\n  const {\n    setIsDropdownVisible,\n    isDropdownVisible,\n    onSearch,\n    searchInput,\n    options,\n    displayedOptions,\n    numberOfOptions,\n  } = useSelectInput()\n  const theme = useTheme()\n  const [searchBarActive, setSearchBarActive] = useState(false)\n  const [defaultSearchValue, setDefaultSearch] = useState<string | null>(null)\n  const ref = useRef<HTMLDivElement>(null)\n  const [search, setSearch] = useState<string>('')\n  const [maxWidth, setWidth] = useState<string | number>()\n\n  useEffect(() => {\n    if (refSelect.current && isDropdownVisible) {\n      const position =\n        refSelect.current.getBoundingClientRect().bottom +\n        DROPDOWN_MAX_HEIGHT +\n        INPUT_SIZE_HEIGHT[size] +\n        parseInt(theme.space['5'], 10)\n      const overflow = position - window.innerHeight\n      if (overflow > 0) {\n        const modalElement = document.getElementById('backdrop-modal')\n\n        if (modalElement) {\n          modalElement.scrollBy({ top: overflow, behavior: 'smooth' })\n        } else window.scrollBy({ top: overflow, behavior: 'smooth' })\n      }\n    }\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, [isDropdownVisible, refSelect, size, ref.current])\n\n  const resizeDropdown = useCallback(() => {\n    if (\n      refSelect.current &&\n      refSelect.current.getBoundingClientRect().width > 0\n    ) {\n      setWidth(refSelect.current.getBoundingClientRect().width)\n    }\n  }, [refSelect])\n\n  useEffect(() => {\n    resizeDropdown()\n\n    window.addEventListener('resize', resizeDropdown)\n\n    return () => window.removeEventListener('resize', resizeDropdown)\n  }, [resizeDropdown])\n\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\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  // No data is displayed (because of the search or because no data is provided)\n  // Set to true when noData by default\n  const isEmpty = useMemo(() => {\n    if (numberOfOptions === 0) return true\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, numberOfOptions])\n\n  const computedFooter = useMemo(() => {\n    if (footer) {\n      if (typeof footer === 'function') {\n        return (\n          <PopupFooter>{footer(() => setIsDropdownVisible(false))}</PopupFooter>\n        )\n      }\n\n      return <PopupFooter>{footer}</PopupFooter>\n    }\n\n    return null\n  }, [footer, setIsDropdownVisible])\n\n  return (\n    <StyledPopup\n      visible={isDropdownVisible}\n      text={\n        <Stack>\n          {searchable && !isLoading && numberOfOptions >= 6 ? (\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          {computedFooter}\n        </Stack>\n      }\n      placement=\"bottom\"\n      disableAnimation\n      maxWidth={maxWidth ?? refSelect.current?.offsetWidth}\n      hasArrow={false}\n      ref={ref}\n      tabIndex={0}\n      role=\"dialog\"\n      debounceDelay={0}\n      containerFullWidth\n      align=\"start\"\n    >\n      {children}\n    </StyledPopup>\n  )\n}\n"]} */"));
131
131
  const PopupFooter = /* @__PURE__ */ _styled__default.default("div", process.env.NODE_ENV === "production" ? {
132
132
  target: "eeucz6c3"
133
133
  } : {
@@ -143,7 +143,7 @@ const PopupFooter = /* @__PURE__ */ _styled__default.default("div", process.env.
143
143
  theme
144
144
  }) => theme.space[2], ";box-shadow:", ({
145
145
  theme
146
- }) => 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":"AA4J8B","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/Dropdown.tsx","sourcesContent":["import { useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport { useCallback, useEffect, useMemo, useRef, useState } from 'react'\nimport type {\n  Dispatch,\n  KeyboardEvent,\n  ReactNode,\n  RefObject,\n  SetStateAction,\n} 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, INPUT_SIZE_HEIGHT, type OptionType } from './types'\n\nconst DROPDOWN_MAX_HEIGHT = 256\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  size: 'small' | 'medium' | 'large'\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  min-width: 320px;\n  background-color: ${({ theme }) => theme.colors.other.elevation.background.raised};\n  color: ${({ theme }) => theme.colors.neutral.text};\n  box-shadow: ${({ theme }) => `${theme.shadows.raised[0]}, ${theme.shadows.raised[1]}`};\n  padding: ${({ theme }) => theme.space[0]};\n  margin-bottom: ${({ theme }) => theme.space[10]};\n`\n\nconst DropdownContainer = styled(Stack)<{ 'data-grouped': boolean }>`\n  max-height: ${DROPDOWN_MAX_HEIGHT}px;\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 }) =>\n      theme.colors.neutral.backgroundWeak};\n  }\n\n  &[data-selectgroup='true']:focus {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundHover};\n  }\n`\nconst DropdownGroupWrapper = styled.div`\n  position: sticky;\n  top: 0px;\n`\nconst DropdownItem = styled.div<{\n  'aria-selected': boolean\n  'aria-disabled': boolean\n}>`\n  text-align:left;\n  border: none;\n  background-color: ${({ theme }) => theme.colors.other.elevation.background.raised};\n\n  padding: ${({ theme }) => theme.space['1.5']} ${({ theme }) =>\n    theme.space['2']} ${({ theme }) => theme.space['1.5']} ${({ theme }) =>\n    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  &[aria-selected='true'] {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n  }\n\n  &[aria-disabled=\"true\"] {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n  }\n\n  &[aria-disabled=\"true\"]:hover, [aria-disabled=\"true\"]:focus {\n    background-color: ${({ theme }) =>\n      theme.colors.neutral.backgroundStrongDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textStrongDisabled};\n    cursor: not-allowed;\n    outline: none;\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 > div[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 > div[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 = (event: KeyboardEvent<HTMLDivElement>) => {\n  if (event.key === 'ArrowDown') {\n    event.preventDefault()\n    moveFocusDown()\n  }\n\n  if (event.key === 'ArrowUp') {\n    event.preventDefault()\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: globalThis.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<HTMLDivElement>(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={handleKeyDownSelect}\n      data-grouped\n    >\n      {isLoading ? (\n        <Skeleton variant=\"block\" />\n      ) : (\n        <>\n          {selectAll && multiselect ? (\n            <Stack id=\"items\">\n              <DropdownItem\n                aria-disabled={false}\n                aria-selected={selectedData.allSelected}\n                aria-label=\"select-all\"\n                data-testid=\"select-all\"\n                id=\"select-all\"\n                tabIndex={0}\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                  {group ? (\n                    <DropdownGroup\n                      key={group}\n                      type=\"button\"\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\n                          variant=\"caption\"\n                          as=\"span\"\n                          placement=\"left\"\n                          sentiment=\"neutral\"\n                        >\n                          {group.toUpperCase()}\n                        </Text>\n                      )}\n                    </DropdownGroup>\n                  ) : null}\n                </DropdownGroupWrapper>\n              ) : null}\n              <Stack id=\"items\" gap=\"0.25\">\n                {displayedOptions[group].map((option, indexOption) => (\n                  <DropdownItem\n                    key={option.value}\n                    aria-disabled={!!option.disabled}\n                    tabIndex={!option.disabled ? 0 : -1}\n                    aria-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              </Stack>\n            </Stack>\n          ))}\n        </>\n      )}\n      {loadMore ? <LoadMore>{loadMore}</LoadMore> : null}\n    </DropdownContainer>\n  ) : (\n    <DropdownContainer\n      role=\"listbox\"\n      tabIndex={-1}\n      id=\"select-dropdown\"\n      onKeyDown={handleKeyDownSelect}\n      gap={0.25}\n      data-grouped={false}\n    >\n      {selectAll && multiselect ? (\n        <Stack id=\"items\" gap={0.25} tabIndex={-1}>\n          <DropdownItem\n            aria-disabled={false}\n            aria-selected={selectedData.allSelected}\n            aria-label=\"select-all\"\n            data-testid=\"select-all\"\n            tabIndex={0}\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              aria-disabled={!!option.disabled}\n              aria-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              tabIndex={!option.disabled ? 0 : -1}\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  size,\n}: DropdownProps) => {\n  const {\n    setIsDropdownVisible,\n    isDropdownVisible,\n    onSearch,\n    searchInput,\n    options,\n    displayedOptions,\n  } = useSelectInput()\n  const theme = useTheme()\n  const [searchBarActive, setSearchBarActive] = useState(false)\n  const [defaultSearchValue, setDefaultSearch] = useState<string | null>(null)\n  const ref = useRef<HTMLDivElement>(null)\n  const [search, setSearch] = useState<string>('')\n  const [maxWidth, setWidth] = useState<string | number>()\n\n  useEffect(() => {\n    if (refSelect.current && isDropdownVisible) {\n      const position =\n        refSelect.current.getBoundingClientRect().bottom +\n        DROPDOWN_MAX_HEIGHT +\n        INPUT_SIZE_HEIGHT[size] +\n        parseInt(theme.space['5'], 10)\n      const overflow = position - window.innerHeight\n      if (overflow > 0) {\n        const modalElement = document.getElementById('backdrop-modal')\n\n        if (modalElement) {\n          modalElement.scrollBy({ top: overflow, behavior: 'smooth' })\n        } else window.scrollBy({ top: overflow, behavior: 'smooth' })\n      }\n    }\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, [isDropdownVisible, refSelect, size, ref.current])\n\n  const resizeDropdown = useCallback(() => {\n    if (\n      refSelect.current &&\n      refSelect.current.getBoundingClientRect().width > 0\n    ) {\n      setWidth(refSelect.current.getBoundingClientRect().width)\n    }\n  }, [refSelect])\n\n  useEffect(() => {\n    resizeDropdown()\n\n    window.addEventListener('resize', resizeDropdown)\n\n    return () => window.removeEventListener('resize', resizeDropdown)\n  }, [resizeDropdown])\n\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      disableAnimation\n      maxWidth={maxWidth ?? refSelect.current?.offsetWidth}\n      hasArrow={false}\n      ref={ref}\n      tabIndex={0}\n      role=\"dialog\"\n      debounceDelay={0}\n      containerFullWidth\n    >\n      {children}\n    </StyledPopup>\n  )\n}\n"]} */"));
146
+ }) => 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":"AA4J8B","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/Dropdown.tsx","sourcesContent":["import { useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport { useCallback, useEffect, useMemo, useRef, useState } from 'react'\nimport type {\n  Dispatch,\n  KeyboardEvent,\n  ReactNode,\n  RefObject,\n  SetStateAction,\n} 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, INPUT_SIZE_HEIGHT, type OptionType } from './types'\n\nconst DROPDOWN_MAX_HEIGHT = 256\n\nexport type DropdownProps = {\n  children: ReactNode\n  emptyState: ReactNode\n  descriptionDirection: 'row' | 'column'\n  searchable: boolean\n  placeholder: string\n  footer?: ((closeDropdown: () => void) => ReactNode) | ReactNode\n  refSelect: RefObject<HTMLDivElement>\n  loadMore?: ReactNode\n  optionalInfoPlacement: 'left' | 'right'\n  isLoading?: boolean\n  size: 'small' | 'medium' | 'large'\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  min-width: 320px;\n  background-color: ${({ theme }) => theme.colors.other.elevation.background.raised};\n  color: ${({ theme }) => theme.colors.neutral.text};\n  box-shadow: ${({ theme }) => `${theme.shadows.raised[0]}, ${theme.shadows.raised[1]}`};\n  padding: ${({ theme }) => theme.space[0]};\n  margin-bottom: ${({ theme }) => theme.space[10]};\n`\n\nconst DropdownContainer = styled(Stack)<{ 'data-grouped': boolean }>`\n  max-height: ${DROPDOWN_MAX_HEIGHT}px;\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 }) =>\n      theme.colors.neutral.backgroundWeak};\n  }\n\n  &[data-selectgroup='true']:focus {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundHover};\n  }\n`\nconst DropdownGroupWrapper = styled.div`\n  position: sticky;\n  top: 0px;\n`\nconst DropdownItem = styled.div<{\n  'aria-selected': boolean\n  'aria-disabled': boolean\n}>`\n  text-align:left;\n  border: none;\n  background-color: ${({ theme }) => theme.colors.other.elevation.background.raised};\n\n  padding: ${({ theme }) => theme.space['1.5']} ${({ theme }) =>\n    theme.space['2']} ${({ theme }) => theme.space['1.5']} ${({ theme }) =>\n    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  &[aria-selected='true'] {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n  }\n\n  &[aria-disabled=\"true\"] {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n  }\n\n  &[aria-disabled=\"true\"]:hover, [aria-disabled=\"true\"]:focus {\n    background-color: ${({ theme }) =>\n      theme.colors.neutral.backgroundStrongDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textStrongDisabled};\n    cursor: not-allowed;\n    outline: none;\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 > div[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 > div[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 = (event: KeyboardEvent<HTMLDivElement>) => {\n  if (event.key === 'ArrowDown') {\n    event.preventDefault()\n    moveFocusDown()\n  }\n\n  if (event.key === 'ArrowUp') {\n    event.preventDefault()\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)\n    onSearch(options)\n  }\n}\n\nconst handleKeyDown = (\n  event: globalThis.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<HTMLDivElement>(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={handleKeyDownSelect}\n      data-grouped\n    >\n      {isLoading ? (\n        <Skeleton variant=\"block\" />\n      ) : (\n        <>\n          {selectAll && multiselect ? (\n            <Stack id=\"items\">\n              <DropdownItem\n                aria-disabled={false}\n                aria-selected={selectedData.allSelected}\n                aria-label=\"select-all\"\n                data-testid=\"select-all\"\n                id=\"select-all\"\n                tabIndex={0}\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                  {group ? (\n                    <DropdownGroup\n                      key={group}\n                      type=\"button\"\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\n                          variant=\"caption\"\n                          as=\"span\"\n                          placement=\"left\"\n                          sentiment=\"neutral\"\n                        >\n                          {group.toUpperCase()}\n                        </Text>\n                      )}\n                    </DropdownGroup>\n                  ) : null}\n                </DropdownGroupWrapper>\n              ) : null}\n              <Stack id=\"items\" gap=\"0.25\">\n                {displayedOptions[group].map((option, indexOption) => (\n                  <DropdownItem\n                    key={option.value}\n                    aria-disabled={!!option.disabled}\n                    tabIndex={!option.disabled ? 0 : -1}\n                    aria-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              </Stack>\n            </Stack>\n          ))}\n        </>\n      )}\n      {loadMore ? <LoadMore>{loadMore}</LoadMore> : null}\n    </DropdownContainer>\n  ) : (\n    <DropdownContainer\n      role=\"listbox\"\n      tabIndex={-1}\n      id=\"select-dropdown\"\n      onKeyDown={handleKeyDownSelect}\n      gap={0.25}\n      data-grouped={false}\n    >\n      {selectAll && multiselect ? (\n        <Stack id=\"items\" gap={0.25} tabIndex={-1}>\n          <DropdownItem\n            aria-disabled={false}\n            aria-selected={selectedData.allSelected}\n            aria-label=\"select-all\"\n            data-testid=\"select-all\"\n            tabIndex={0}\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              aria-disabled={!!option.disabled}\n              aria-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              tabIndex={!option.disabled ? 0 : -1}\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  size,\n}: DropdownProps) => {\n  const {\n    setIsDropdownVisible,\n    isDropdownVisible,\n    onSearch,\n    searchInput,\n    options,\n    displayedOptions,\n    numberOfOptions,\n  } = useSelectInput()\n  const theme = useTheme()\n  const [searchBarActive, setSearchBarActive] = useState(false)\n  const [defaultSearchValue, setDefaultSearch] = useState<string | null>(null)\n  const ref = useRef<HTMLDivElement>(null)\n  const [search, setSearch] = useState<string>('')\n  const [maxWidth, setWidth] = useState<string | number>()\n\n  useEffect(() => {\n    if (refSelect.current && isDropdownVisible) {\n      const position =\n        refSelect.current.getBoundingClientRect().bottom +\n        DROPDOWN_MAX_HEIGHT +\n        INPUT_SIZE_HEIGHT[size] +\n        parseInt(theme.space['5'], 10)\n      const overflow = position - window.innerHeight\n      if (overflow > 0) {\n        const modalElement = document.getElementById('backdrop-modal')\n\n        if (modalElement) {\n          modalElement.scrollBy({ top: overflow, behavior: 'smooth' })\n        } else window.scrollBy({ top: overflow, behavior: 'smooth' })\n      }\n    }\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, [isDropdownVisible, refSelect, size, ref.current])\n\n  const resizeDropdown = useCallback(() => {\n    if (\n      refSelect.current &&\n      refSelect.current.getBoundingClientRect().width > 0\n    ) {\n      setWidth(refSelect.current.getBoundingClientRect().width)\n    }\n  }, [refSelect])\n\n  useEffect(() => {\n    resizeDropdown()\n\n    window.addEventListener('resize', resizeDropdown)\n\n    return () => window.removeEventListener('resize', resizeDropdown)\n  }, [resizeDropdown])\n\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\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  // No data is displayed (because of the search or because no data is provided)\n  // Set to true when noData by default\n  const isEmpty = useMemo(() => {\n    if (numberOfOptions === 0) return true\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, numberOfOptions])\n\n  const computedFooter = useMemo(() => {\n    if (footer) {\n      if (typeof footer === 'function') {\n        return (\n          <PopupFooter>{footer(() => setIsDropdownVisible(false))}</PopupFooter>\n        )\n      }\n\n      return <PopupFooter>{footer}</PopupFooter>\n    }\n\n    return null\n  }, [footer, setIsDropdownVisible])\n\n  return (\n    <StyledPopup\n      visible={isDropdownVisible}\n      text={\n        <Stack>\n          {searchable && !isLoading && numberOfOptions >= 6 ? (\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          {computedFooter}\n        </Stack>\n      }\n      placement=\"bottom\"\n      disableAnimation\n      maxWidth={maxWidth ?? refSelect.current?.offsetWidth}\n      hasArrow={false}\n      ref={ref}\n      tabIndex={0}\n      role=\"dialog\"\n      debounceDelay={0}\n      containerFullWidth\n      align=\"start\"\n    >\n      {children}\n    </StyledPopup>\n  )\n}\n"]} */"));
147
147
  const StyledCheckbox = /* @__PURE__ */ _styled__default.default(index$2.Checkbox, process.env.NODE_ENV === "production" ? {
148
148
  target: "eeucz6c2"
149
149
  } : {
@@ -155,7 +155,7 @@ const StyledCheckbox = /* @__PURE__ */ _styled__default.default(index$2.Checkbox
155
155
  } : {
156
156
  name: "1l9xw77",
157
157
  styles: "width:100%;position:static;text-align:left;align-items:center;pointer-events:none",
158
- 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":"AAkKuC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/Dropdown.tsx","sourcesContent":["import { useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport { useCallback, useEffect, useMemo, useRef, useState } from 'react'\nimport type {\n  Dispatch,\n  KeyboardEvent,\n  ReactNode,\n  RefObject,\n  SetStateAction,\n} 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, INPUT_SIZE_HEIGHT, type OptionType } from './types'\n\nconst DROPDOWN_MAX_HEIGHT = 256\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  size: 'small' | 'medium' | 'large'\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  min-width: 320px;\n  background-color: ${({ theme }) => theme.colors.other.elevation.background.raised};\n  color: ${({ theme }) => theme.colors.neutral.text};\n  box-shadow: ${({ theme }) => `${theme.shadows.raised[0]}, ${theme.shadows.raised[1]}`};\n  padding: ${({ theme }) => theme.space[0]};\n  margin-bottom: ${({ theme }) => theme.space[10]};\n`\n\nconst DropdownContainer = styled(Stack)<{ 'data-grouped': boolean }>`\n  max-height: ${DROPDOWN_MAX_HEIGHT}px;\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 }) =>\n      theme.colors.neutral.backgroundWeak};\n  }\n\n  &[data-selectgroup='true']:focus {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundHover};\n  }\n`\nconst DropdownGroupWrapper = styled.div`\n  position: sticky;\n  top: 0px;\n`\nconst DropdownItem = styled.div<{\n  'aria-selected': boolean\n  'aria-disabled': boolean\n}>`\n  text-align:left;\n  border: none;\n  background-color: ${({ theme }) => theme.colors.other.elevation.background.raised};\n\n  padding: ${({ theme }) => theme.space['1.5']} ${({ theme }) =>\n    theme.space['2']} ${({ theme }) => theme.space['1.5']} ${({ theme }) =>\n    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  &[aria-selected='true'] {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n  }\n\n  &[aria-disabled=\"true\"] {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n  }\n\n  &[aria-disabled=\"true\"]:hover, [aria-disabled=\"true\"]:focus {\n    background-color: ${({ theme }) =>\n      theme.colors.neutral.backgroundStrongDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textStrongDisabled};\n    cursor: not-allowed;\n    outline: none;\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 > div[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 > div[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 = (event: KeyboardEvent<HTMLDivElement>) => {\n  if (event.key === 'ArrowDown') {\n    event.preventDefault()\n    moveFocusDown()\n  }\n\n  if (event.key === 'ArrowUp') {\n    event.preventDefault()\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: globalThis.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<HTMLDivElement>(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={handleKeyDownSelect}\n      data-grouped\n    >\n      {isLoading ? (\n        <Skeleton variant=\"block\" />\n      ) : (\n        <>\n          {selectAll && multiselect ? (\n            <Stack id=\"items\">\n              <DropdownItem\n                aria-disabled={false}\n                aria-selected={selectedData.allSelected}\n                aria-label=\"select-all\"\n                data-testid=\"select-all\"\n                id=\"select-all\"\n                tabIndex={0}\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                  {group ? (\n                    <DropdownGroup\n                      key={group}\n                      type=\"button\"\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\n                          variant=\"caption\"\n                          as=\"span\"\n                          placement=\"left\"\n                          sentiment=\"neutral\"\n                        >\n                          {group.toUpperCase()}\n                        </Text>\n                      )}\n                    </DropdownGroup>\n                  ) : null}\n                </DropdownGroupWrapper>\n              ) : null}\n              <Stack id=\"items\" gap=\"0.25\">\n                {displayedOptions[group].map((option, indexOption) => (\n                  <DropdownItem\n                    key={option.value}\n                    aria-disabled={!!option.disabled}\n                    tabIndex={!option.disabled ? 0 : -1}\n                    aria-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              </Stack>\n            </Stack>\n          ))}\n        </>\n      )}\n      {loadMore ? <LoadMore>{loadMore}</LoadMore> : null}\n    </DropdownContainer>\n  ) : (\n    <DropdownContainer\n      role=\"listbox\"\n      tabIndex={-1}\n      id=\"select-dropdown\"\n      onKeyDown={handleKeyDownSelect}\n      gap={0.25}\n      data-grouped={false}\n    >\n      {selectAll && multiselect ? (\n        <Stack id=\"items\" gap={0.25} tabIndex={-1}>\n          <DropdownItem\n            aria-disabled={false}\n            aria-selected={selectedData.allSelected}\n            aria-label=\"select-all\"\n            data-testid=\"select-all\"\n            tabIndex={0}\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              aria-disabled={!!option.disabled}\n              aria-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              tabIndex={!option.disabled ? 0 : -1}\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  size,\n}: DropdownProps) => {\n  const {\n    setIsDropdownVisible,\n    isDropdownVisible,\n    onSearch,\n    searchInput,\n    options,\n    displayedOptions,\n  } = useSelectInput()\n  const theme = useTheme()\n  const [searchBarActive, setSearchBarActive] = useState(false)\n  const [defaultSearchValue, setDefaultSearch] = useState<string | null>(null)\n  const ref = useRef<HTMLDivElement>(null)\n  const [search, setSearch] = useState<string>('')\n  const [maxWidth, setWidth] = useState<string | number>()\n\n  useEffect(() => {\n    if (refSelect.current && isDropdownVisible) {\n      const position =\n        refSelect.current.getBoundingClientRect().bottom +\n        DROPDOWN_MAX_HEIGHT +\n        INPUT_SIZE_HEIGHT[size] +\n        parseInt(theme.space['5'], 10)\n      const overflow = position - window.innerHeight\n      if (overflow > 0) {\n        const modalElement = document.getElementById('backdrop-modal')\n\n        if (modalElement) {\n          modalElement.scrollBy({ top: overflow, behavior: 'smooth' })\n        } else window.scrollBy({ top: overflow, behavior: 'smooth' })\n      }\n    }\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, [isDropdownVisible, refSelect, size, ref.current])\n\n  const resizeDropdown = useCallback(() => {\n    if (\n      refSelect.current &&\n      refSelect.current.getBoundingClientRect().width > 0\n    ) {\n      setWidth(refSelect.current.getBoundingClientRect().width)\n    }\n  }, [refSelect])\n\n  useEffect(() => {\n    resizeDropdown()\n\n    window.addEventListener('resize', resizeDropdown)\n\n    return () => window.removeEventListener('resize', resizeDropdown)\n  }, [resizeDropdown])\n\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      disableAnimation\n      maxWidth={maxWidth ?? refSelect.current?.offsetWidth}\n      hasArrow={false}\n      ref={ref}\n      tabIndex={0}\n      role=\"dialog\"\n      debounceDelay={0}\n      containerFullWidth\n    >\n      {children}\n    </StyledPopup>\n  )\n}\n"]} */",
158
+ 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":"AAkKuC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/Dropdown.tsx","sourcesContent":["import { useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport { useCallback, useEffect, useMemo, useRef, useState } from 'react'\nimport type {\n  Dispatch,\n  KeyboardEvent,\n  ReactNode,\n  RefObject,\n  SetStateAction,\n} 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, INPUT_SIZE_HEIGHT, type OptionType } from './types'\n\nconst DROPDOWN_MAX_HEIGHT = 256\n\nexport type DropdownProps = {\n  children: ReactNode\n  emptyState: ReactNode\n  descriptionDirection: 'row' | 'column'\n  searchable: boolean\n  placeholder: string\n  footer?: ((closeDropdown: () => void) => ReactNode) | ReactNode\n  refSelect: RefObject<HTMLDivElement>\n  loadMore?: ReactNode\n  optionalInfoPlacement: 'left' | 'right'\n  isLoading?: boolean\n  size: 'small' | 'medium' | 'large'\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  min-width: 320px;\n  background-color: ${({ theme }) => theme.colors.other.elevation.background.raised};\n  color: ${({ theme }) => theme.colors.neutral.text};\n  box-shadow: ${({ theme }) => `${theme.shadows.raised[0]}, ${theme.shadows.raised[1]}`};\n  padding: ${({ theme }) => theme.space[0]};\n  margin-bottom: ${({ theme }) => theme.space[10]};\n`\n\nconst DropdownContainer = styled(Stack)<{ 'data-grouped': boolean }>`\n  max-height: ${DROPDOWN_MAX_HEIGHT}px;\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 }) =>\n      theme.colors.neutral.backgroundWeak};\n  }\n\n  &[data-selectgroup='true']:focus {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundHover};\n  }\n`\nconst DropdownGroupWrapper = styled.div`\n  position: sticky;\n  top: 0px;\n`\nconst DropdownItem = styled.div<{\n  'aria-selected': boolean\n  'aria-disabled': boolean\n}>`\n  text-align:left;\n  border: none;\n  background-color: ${({ theme }) => theme.colors.other.elevation.background.raised};\n\n  padding: ${({ theme }) => theme.space['1.5']} ${({ theme }) =>\n    theme.space['2']} ${({ theme }) => theme.space['1.5']} ${({ theme }) =>\n    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  &[aria-selected='true'] {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n  }\n\n  &[aria-disabled=\"true\"] {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n  }\n\n  &[aria-disabled=\"true\"]:hover, [aria-disabled=\"true\"]:focus {\n    background-color: ${({ theme }) =>\n      theme.colors.neutral.backgroundStrongDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textStrongDisabled};\n    cursor: not-allowed;\n    outline: none;\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 > div[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 > div[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 = (event: KeyboardEvent<HTMLDivElement>) => {\n  if (event.key === 'ArrowDown') {\n    event.preventDefault()\n    moveFocusDown()\n  }\n\n  if (event.key === 'ArrowUp') {\n    event.preventDefault()\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)\n    onSearch(options)\n  }\n}\n\nconst handleKeyDown = (\n  event: globalThis.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<HTMLDivElement>(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={handleKeyDownSelect}\n      data-grouped\n    >\n      {isLoading ? (\n        <Skeleton variant=\"block\" />\n      ) : (\n        <>\n          {selectAll && multiselect ? (\n            <Stack id=\"items\">\n              <DropdownItem\n                aria-disabled={false}\n                aria-selected={selectedData.allSelected}\n                aria-label=\"select-all\"\n                data-testid=\"select-all\"\n                id=\"select-all\"\n                tabIndex={0}\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                  {group ? (\n                    <DropdownGroup\n                      key={group}\n                      type=\"button\"\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\n                          variant=\"caption\"\n                          as=\"span\"\n                          placement=\"left\"\n                          sentiment=\"neutral\"\n                        >\n                          {group.toUpperCase()}\n                        </Text>\n                      )}\n                    </DropdownGroup>\n                  ) : null}\n                </DropdownGroupWrapper>\n              ) : null}\n              <Stack id=\"items\" gap=\"0.25\">\n                {displayedOptions[group].map((option, indexOption) => (\n                  <DropdownItem\n                    key={option.value}\n                    aria-disabled={!!option.disabled}\n                    tabIndex={!option.disabled ? 0 : -1}\n                    aria-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              </Stack>\n            </Stack>\n          ))}\n        </>\n      )}\n      {loadMore ? <LoadMore>{loadMore}</LoadMore> : null}\n    </DropdownContainer>\n  ) : (\n    <DropdownContainer\n      role=\"listbox\"\n      tabIndex={-1}\n      id=\"select-dropdown\"\n      onKeyDown={handleKeyDownSelect}\n      gap={0.25}\n      data-grouped={false}\n    >\n      {selectAll && multiselect ? (\n        <Stack id=\"items\" gap={0.25} tabIndex={-1}>\n          <DropdownItem\n            aria-disabled={false}\n            aria-selected={selectedData.allSelected}\n            aria-label=\"select-all\"\n            data-testid=\"select-all\"\n            tabIndex={0}\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              aria-disabled={!!option.disabled}\n              aria-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              tabIndex={!option.disabled ? 0 : -1}\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  size,\n}: DropdownProps) => {\n  const {\n    setIsDropdownVisible,\n    isDropdownVisible,\n    onSearch,\n    searchInput,\n    options,\n    displayedOptions,\n    numberOfOptions,\n  } = useSelectInput()\n  const theme = useTheme()\n  const [searchBarActive, setSearchBarActive] = useState(false)\n  const [defaultSearchValue, setDefaultSearch] = useState<string | null>(null)\n  const ref = useRef<HTMLDivElement>(null)\n  const [search, setSearch] = useState<string>('')\n  const [maxWidth, setWidth] = useState<string | number>()\n\n  useEffect(() => {\n    if (refSelect.current && isDropdownVisible) {\n      const position =\n        refSelect.current.getBoundingClientRect().bottom +\n        DROPDOWN_MAX_HEIGHT +\n        INPUT_SIZE_HEIGHT[size] +\n        parseInt(theme.space['5'], 10)\n      const overflow = position - window.innerHeight\n      if (overflow > 0) {\n        const modalElement = document.getElementById('backdrop-modal')\n\n        if (modalElement) {\n          modalElement.scrollBy({ top: overflow, behavior: 'smooth' })\n        } else window.scrollBy({ top: overflow, behavior: 'smooth' })\n      }\n    }\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, [isDropdownVisible, refSelect, size, ref.current])\n\n  const resizeDropdown = useCallback(() => {\n    if (\n      refSelect.current &&\n      refSelect.current.getBoundingClientRect().width > 0\n    ) {\n      setWidth(refSelect.current.getBoundingClientRect().width)\n    }\n  }, [refSelect])\n\n  useEffect(() => {\n    resizeDropdown()\n\n    window.addEventListener('resize', resizeDropdown)\n\n    return () => window.removeEventListener('resize', resizeDropdown)\n  }, [resizeDropdown])\n\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\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  // No data is displayed (because of the search or because no data is provided)\n  // Set to true when noData by default\n  const isEmpty = useMemo(() => {\n    if (numberOfOptions === 0) return true\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, numberOfOptions])\n\n  const computedFooter = useMemo(() => {\n    if (footer) {\n      if (typeof footer === 'function') {\n        return (\n          <PopupFooter>{footer(() => setIsDropdownVisible(false))}</PopupFooter>\n        )\n      }\n\n      return <PopupFooter>{footer}</PopupFooter>\n    }\n\n    return null\n  }, [footer, setIsDropdownVisible])\n\n  return (\n    <StyledPopup\n      visible={isDropdownVisible}\n      text={\n        <Stack>\n          {searchable && !isLoading && numberOfOptions >= 6 ? (\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          {computedFooter}\n        </Stack>\n      }\n      placement=\"bottom\"\n      disableAnimation\n      maxWidth={maxWidth ?? refSelect.current?.offsetWidth}\n      hasArrow={false}\n      ref={ref}\n      tabIndex={0}\n      role=\"dialog\"\n      debounceDelay={0}\n      containerFullWidth\n      align=\"start\"\n    >\n      {children}\n    </StyledPopup>\n  )\n}\n"]} */",
159
159
  toString: _EMOTION_STRINGIFIED_CSS_ERROR__
160
160
  });
161
161
  const EmptyState = /* @__PURE__ */ _styled__default.default(index$1.Stack, process.env.NODE_ENV === "production" ? {
@@ -165,7 +165,7 @@ const EmptyState = /* @__PURE__ */ _styled__default.default(index$1.Stack, proce
165
165
  label: "EmptyState"
166
166
  })("padding:", ({
167
167
  theme
168
- }) => 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":"AA0KgC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/Dropdown.tsx","sourcesContent":["import { useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport { useCallback, useEffect, useMemo, useRef, useState } from 'react'\nimport type {\n  Dispatch,\n  KeyboardEvent,\n  ReactNode,\n  RefObject,\n  SetStateAction,\n} 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, INPUT_SIZE_HEIGHT, type OptionType } from './types'\n\nconst DROPDOWN_MAX_HEIGHT = 256\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  size: 'small' | 'medium' | 'large'\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  min-width: 320px;\n  background-color: ${({ theme }) => theme.colors.other.elevation.background.raised};\n  color: ${({ theme }) => theme.colors.neutral.text};\n  box-shadow: ${({ theme }) => `${theme.shadows.raised[0]}, ${theme.shadows.raised[1]}`};\n  padding: ${({ theme }) => theme.space[0]};\n  margin-bottom: ${({ theme }) => theme.space[10]};\n`\n\nconst DropdownContainer = styled(Stack)<{ 'data-grouped': boolean }>`\n  max-height: ${DROPDOWN_MAX_HEIGHT}px;\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 }) =>\n      theme.colors.neutral.backgroundWeak};\n  }\n\n  &[data-selectgroup='true']:focus {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundHover};\n  }\n`\nconst DropdownGroupWrapper = styled.div`\n  position: sticky;\n  top: 0px;\n`\nconst DropdownItem = styled.div<{\n  'aria-selected': boolean\n  'aria-disabled': boolean\n}>`\n  text-align:left;\n  border: none;\n  background-color: ${({ theme }) => theme.colors.other.elevation.background.raised};\n\n  padding: ${({ theme }) => theme.space['1.5']} ${({ theme }) =>\n    theme.space['2']} ${({ theme }) => theme.space['1.5']} ${({ theme }) =>\n    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  &[aria-selected='true'] {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n  }\n\n  &[aria-disabled=\"true\"] {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n  }\n\n  &[aria-disabled=\"true\"]:hover, [aria-disabled=\"true\"]:focus {\n    background-color: ${({ theme }) =>\n      theme.colors.neutral.backgroundStrongDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textStrongDisabled};\n    cursor: not-allowed;\n    outline: none;\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 > div[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 > div[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 = (event: KeyboardEvent<HTMLDivElement>) => {\n  if (event.key === 'ArrowDown') {\n    event.preventDefault()\n    moveFocusDown()\n  }\n\n  if (event.key === 'ArrowUp') {\n    event.preventDefault()\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: globalThis.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<HTMLDivElement>(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={handleKeyDownSelect}\n      data-grouped\n    >\n      {isLoading ? (\n        <Skeleton variant=\"block\" />\n      ) : (\n        <>\n          {selectAll && multiselect ? (\n            <Stack id=\"items\">\n              <DropdownItem\n                aria-disabled={false}\n                aria-selected={selectedData.allSelected}\n                aria-label=\"select-all\"\n                data-testid=\"select-all\"\n                id=\"select-all\"\n                tabIndex={0}\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                  {group ? (\n                    <DropdownGroup\n                      key={group}\n                      type=\"button\"\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\n                          variant=\"caption\"\n                          as=\"span\"\n                          placement=\"left\"\n                          sentiment=\"neutral\"\n                        >\n                          {group.toUpperCase()}\n                        </Text>\n                      )}\n                    </DropdownGroup>\n                  ) : null}\n                </DropdownGroupWrapper>\n              ) : null}\n              <Stack id=\"items\" gap=\"0.25\">\n                {displayedOptions[group].map((option, indexOption) => (\n                  <DropdownItem\n                    key={option.value}\n                    aria-disabled={!!option.disabled}\n                    tabIndex={!option.disabled ? 0 : -1}\n                    aria-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              </Stack>\n            </Stack>\n          ))}\n        </>\n      )}\n      {loadMore ? <LoadMore>{loadMore}</LoadMore> : null}\n    </DropdownContainer>\n  ) : (\n    <DropdownContainer\n      role=\"listbox\"\n      tabIndex={-1}\n      id=\"select-dropdown\"\n      onKeyDown={handleKeyDownSelect}\n      gap={0.25}\n      data-grouped={false}\n    >\n      {selectAll && multiselect ? (\n        <Stack id=\"items\" gap={0.25} tabIndex={-1}>\n          <DropdownItem\n            aria-disabled={false}\n            aria-selected={selectedData.allSelected}\n            aria-label=\"select-all\"\n            data-testid=\"select-all\"\n            tabIndex={0}\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              aria-disabled={!!option.disabled}\n              aria-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              tabIndex={!option.disabled ? 0 : -1}\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  size,\n}: DropdownProps) => {\n  const {\n    setIsDropdownVisible,\n    isDropdownVisible,\n    onSearch,\n    searchInput,\n    options,\n    displayedOptions,\n  } = useSelectInput()\n  const theme = useTheme()\n  const [searchBarActive, setSearchBarActive] = useState(false)\n  const [defaultSearchValue, setDefaultSearch] = useState<string | null>(null)\n  const ref = useRef<HTMLDivElement>(null)\n  const [search, setSearch] = useState<string>('')\n  const [maxWidth, setWidth] = useState<string | number>()\n\n  useEffect(() => {\n    if (refSelect.current && isDropdownVisible) {\n      const position =\n        refSelect.current.getBoundingClientRect().bottom +\n        DROPDOWN_MAX_HEIGHT +\n        INPUT_SIZE_HEIGHT[size] +\n        parseInt(theme.space['5'], 10)\n      const overflow = position - window.innerHeight\n      if (overflow > 0) {\n        const modalElement = document.getElementById('backdrop-modal')\n\n        if (modalElement) {\n          modalElement.scrollBy({ top: overflow, behavior: 'smooth' })\n        } else window.scrollBy({ top: overflow, behavior: 'smooth' })\n      }\n    }\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, [isDropdownVisible, refSelect, size, ref.current])\n\n  const resizeDropdown = useCallback(() => {\n    if (\n      refSelect.current &&\n      refSelect.current.getBoundingClientRect().width > 0\n    ) {\n      setWidth(refSelect.current.getBoundingClientRect().width)\n    }\n  }, [refSelect])\n\n  useEffect(() => {\n    resizeDropdown()\n\n    window.addEventListener('resize', resizeDropdown)\n\n    return () => window.removeEventListener('resize', resizeDropdown)\n  }, [resizeDropdown])\n\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      disableAnimation\n      maxWidth={maxWidth ?? refSelect.current?.offsetWidth}\n      hasArrow={false}\n      ref={ref}\n      tabIndex={0}\n      role=\"dialog\"\n      debounceDelay={0}\n      containerFullWidth\n    >\n      {children}\n    </StyledPopup>\n  )\n}\n"]} */"));
168
+ }) => 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":"AA0KgC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/Dropdown.tsx","sourcesContent":["import { useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport { useCallback, useEffect, useMemo, useRef, useState } from 'react'\nimport type {\n  Dispatch,\n  KeyboardEvent,\n  ReactNode,\n  RefObject,\n  SetStateAction,\n} 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, INPUT_SIZE_HEIGHT, type OptionType } from './types'\n\nconst DROPDOWN_MAX_HEIGHT = 256\n\nexport type DropdownProps = {\n  children: ReactNode\n  emptyState: ReactNode\n  descriptionDirection: 'row' | 'column'\n  searchable: boolean\n  placeholder: string\n  footer?: ((closeDropdown: () => void) => ReactNode) | ReactNode\n  refSelect: RefObject<HTMLDivElement>\n  loadMore?: ReactNode\n  optionalInfoPlacement: 'left' | 'right'\n  isLoading?: boolean\n  size: 'small' | 'medium' | 'large'\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  min-width: 320px;\n  background-color: ${({ theme }) => theme.colors.other.elevation.background.raised};\n  color: ${({ theme }) => theme.colors.neutral.text};\n  box-shadow: ${({ theme }) => `${theme.shadows.raised[0]}, ${theme.shadows.raised[1]}`};\n  padding: ${({ theme }) => theme.space[0]};\n  margin-bottom: ${({ theme }) => theme.space[10]};\n`\n\nconst DropdownContainer = styled(Stack)<{ 'data-grouped': boolean }>`\n  max-height: ${DROPDOWN_MAX_HEIGHT}px;\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 }) =>\n      theme.colors.neutral.backgroundWeak};\n  }\n\n  &[data-selectgroup='true']:focus {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundHover};\n  }\n`\nconst DropdownGroupWrapper = styled.div`\n  position: sticky;\n  top: 0px;\n`\nconst DropdownItem = styled.div<{\n  'aria-selected': boolean\n  'aria-disabled': boolean\n}>`\n  text-align:left;\n  border: none;\n  background-color: ${({ theme }) => theme.colors.other.elevation.background.raised};\n\n  padding: ${({ theme }) => theme.space['1.5']} ${({ theme }) =>\n    theme.space['2']} ${({ theme }) => theme.space['1.5']} ${({ theme }) =>\n    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  &[aria-selected='true'] {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n  }\n\n  &[aria-disabled=\"true\"] {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n  }\n\n  &[aria-disabled=\"true\"]:hover, [aria-disabled=\"true\"]:focus {\n    background-color: ${({ theme }) =>\n      theme.colors.neutral.backgroundStrongDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textStrongDisabled};\n    cursor: not-allowed;\n    outline: none;\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 > div[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 > div[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 = (event: KeyboardEvent<HTMLDivElement>) => {\n  if (event.key === 'ArrowDown') {\n    event.preventDefault()\n    moveFocusDown()\n  }\n\n  if (event.key === 'ArrowUp') {\n    event.preventDefault()\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)\n    onSearch(options)\n  }\n}\n\nconst handleKeyDown = (\n  event: globalThis.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<HTMLDivElement>(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={handleKeyDownSelect}\n      data-grouped\n    >\n      {isLoading ? (\n        <Skeleton variant=\"block\" />\n      ) : (\n        <>\n          {selectAll && multiselect ? (\n            <Stack id=\"items\">\n              <DropdownItem\n                aria-disabled={false}\n                aria-selected={selectedData.allSelected}\n                aria-label=\"select-all\"\n                data-testid=\"select-all\"\n                id=\"select-all\"\n                tabIndex={0}\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                  {group ? (\n                    <DropdownGroup\n                      key={group}\n                      type=\"button\"\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\n                          variant=\"caption\"\n                          as=\"span\"\n                          placement=\"left\"\n                          sentiment=\"neutral\"\n                        >\n                          {group.toUpperCase()}\n                        </Text>\n                      )}\n                    </DropdownGroup>\n                  ) : null}\n                </DropdownGroupWrapper>\n              ) : null}\n              <Stack id=\"items\" gap=\"0.25\">\n                {displayedOptions[group].map((option, indexOption) => (\n                  <DropdownItem\n                    key={option.value}\n                    aria-disabled={!!option.disabled}\n                    tabIndex={!option.disabled ? 0 : -1}\n                    aria-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              </Stack>\n            </Stack>\n          ))}\n        </>\n      )}\n      {loadMore ? <LoadMore>{loadMore}</LoadMore> : null}\n    </DropdownContainer>\n  ) : (\n    <DropdownContainer\n      role=\"listbox\"\n      tabIndex={-1}\n      id=\"select-dropdown\"\n      onKeyDown={handleKeyDownSelect}\n      gap={0.25}\n      data-grouped={false}\n    >\n      {selectAll && multiselect ? (\n        <Stack id=\"items\" gap={0.25} tabIndex={-1}>\n          <DropdownItem\n            aria-disabled={false}\n            aria-selected={selectedData.allSelected}\n            aria-label=\"select-all\"\n            data-testid=\"select-all\"\n            tabIndex={0}\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              aria-disabled={!!option.disabled}\n              aria-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              tabIndex={!option.disabled ? 0 : -1}\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  size,\n}: DropdownProps) => {\n  const {\n    setIsDropdownVisible,\n    isDropdownVisible,\n    onSearch,\n    searchInput,\n    options,\n    displayedOptions,\n    numberOfOptions,\n  } = useSelectInput()\n  const theme = useTheme()\n  const [searchBarActive, setSearchBarActive] = useState(false)\n  const [defaultSearchValue, setDefaultSearch] = useState<string | null>(null)\n  const ref = useRef<HTMLDivElement>(null)\n  const [search, setSearch] = useState<string>('')\n  const [maxWidth, setWidth] = useState<string | number>()\n\n  useEffect(() => {\n    if (refSelect.current && isDropdownVisible) {\n      const position =\n        refSelect.current.getBoundingClientRect().bottom +\n        DROPDOWN_MAX_HEIGHT +\n        INPUT_SIZE_HEIGHT[size] +\n        parseInt(theme.space['5'], 10)\n      const overflow = position - window.innerHeight\n      if (overflow > 0) {\n        const modalElement = document.getElementById('backdrop-modal')\n\n        if (modalElement) {\n          modalElement.scrollBy({ top: overflow, behavior: 'smooth' })\n        } else window.scrollBy({ top: overflow, behavior: 'smooth' })\n      }\n    }\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, [isDropdownVisible, refSelect, size, ref.current])\n\n  const resizeDropdown = useCallback(() => {\n    if (\n      refSelect.current &&\n      refSelect.current.getBoundingClientRect().width > 0\n    ) {\n      setWidth(refSelect.current.getBoundingClientRect().width)\n    }\n  }, [refSelect])\n\n  useEffect(() => {\n    resizeDropdown()\n\n    window.addEventListener('resize', resizeDropdown)\n\n    return () => window.removeEventListener('resize', resizeDropdown)\n  }, [resizeDropdown])\n\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\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  // No data is displayed (because of the search or because no data is provided)\n  // Set to true when noData by default\n  const isEmpty = useMemo(() => {\n    if (numberOfOptions === 0) return true\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, numberOfOptions])\n\n  const computedFooter = useMemo(() => {\n    if (footer) {\n      if (typeof footer === 'function') {\n        return (\n          <PopupFooter>{footer(() => setIsDropdownVisible(false))}</PopupFooter>\n        )\n      }\n\n      return <PopupFooter>{footer}</PopupFooter>\n    }\n\n    return null\n  }, [footer, setIsDropdownVisible])\n\n  return (\n    <StyledPopup\n      visible={isDropdownVisible}\n      text={\n        <Stack>\n          {searchable && !isLoading && numberOfOptions >= 6 ? (\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          {computedFooter}\n        </Stack>\n      }\n      placement=\"bottom\"\n      disableAnimation\n      maxWidth={maxWidth ?? refSelect.current?.offsetWidth}\n      hasArrow={false}\n      ref={ref}\n      tabIndex={0}\n      role=\"dialog\"\n      debounceDelay={0}\n      containerFullWidth\n      align=\"start\"\n    >\n      {children}\n    </StyledPopup>\n  )\n}\n"]} */"));
169
169
  const LoadMore = /* @__PURE__ */ _styled__default.default(index$1.Stack, process.env.NODE_ENV === "production" ? {
170
170
  target: "eeucz6c0"
171
171
  } : {
@@ -173,7 +173,7 @@ const LoadMore = /* @__PURE__ */ _styled__default.default(index$1.Stack, process
173
173
  label: "LoadMore"
174
174
  })("padding:", ({
175
175
  theme
176
- }) => 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":"AA6K8B","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/Dropdown.tsx","sourcesContent":["import { useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport { useCallback, useEffect, useMemo, useRef, useState } from 'react'\nimport type {\n  Dispatch,\n  KeyboardEvent,\n  ReactNode,\n  RefObject,\n  SetStateAction,\n} 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, INPUT_SIZE_HEIGHT, type OptionType } from './types'\n\nconst DROPDOWN_MAX_HEIGHT = 256\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  size: 'small' | 'medium' | 'large'\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  min-width: 320px;\n  background-color: ${({ theme }) => theme.colors.other.elevation.background.raised};\n  color: ${({ theme }) => theme.colors.neutral.text};\n  box-shadow: ${({ theme }) => `${theme.shadows.raised[0]}, ${theme.shadows.raised[1]}`};\n  padding: ${({ theme }) => theme.space[0]};\n  margin-bottom: ${({ theme }) => theme.space[10]};\n`\n\nconst DropdownContainer = styled(Stack)<{ 'data-grouped': boolean }>`\n  max-height: ${DROPDOWN_MAX_HEIGHT}px;\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 }) =>\n      theme.colors.neutral.backgroundWeak};\n  }\n\n  &[data-selectgroup='true']:focus {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundHover};\n  }\n`\nconst DropdownGroupWrapper = styled.div`\n  position: sticky;\n  top: 0px;\n`\nconst DropdownItem = styled.div<{\n  'aria-selected': boolean\n  'aria-disabled': boolean\n}>`\n  text-align:left;\n  border: none;\n  background-color: ${({ theme }) => theme.colors.other.elevation.background.raised};\n\n  padding: ${({ theme }) => theme.space['1.5']} ${({ theme }) =>\n    theme.space['2']} ${({ theme }) => theme.space['1.5']} ${({ theme }) =>\n    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  &[aria-selected='true'] {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n  }\n\n  &[aria-disabled=\"true\"] {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n  }\n\n  &[aria-disabled=\"true\"]:hover, [aria-disabled=\"true\"]:focus {\n    background-color: ${({ theme }) =>\n      theme.colors.neutral.backgroundStrongDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textStrongDisabled};\n    cursor: not-allowed;\n    outline: none;\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 > div[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 > div[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 = (event: KeyboardEvent<HTMLDivElement>) => {\n  if (event.key === 'ArrowDown') {\n    event.preventDefault()\n    moveFocusDown()\n  }\n\n  if (event.key === 'ArrowUp') {\n    event.preventDefault()\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: globalThis.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<HTMLDivElement>(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={handleKeyDownSelect}\n      data-grouped\n    >\n      {isLoading ? (\n        <Skeleton variant=\"block\" />\n      ) : (\n        <>\n          {selectAll && multiselect ? (\n            <Stack id=\"items\">\n              <DropdownItem\n                aria-disabled={false}\n                aria-selected={selectedData.allSelected}\n                aria-label=\"select-all\"\n                data-testid=\"select-all\"\n                id=\"select-all\"\n                tabIndex={0}\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                  {group ? (\n                    <DropdownGroup\n                      key={group}\n                      type=\"button\"\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\n                          variant=\"caption\"\n                          as=\"span\"\n                          placement=\"left\"\n                          sentiment=\"neutral\"\n                        >\n                          {group.toUpperCase()}\n                        </Text>\n                      )}\n                    </DropdownGroup>\n                  ) : null}\n                </DropdownGroupWrapper>\n              ) : null}\n              <Stack id=\"items\" gap=\"0.25\">\n                {displayedOptions[group].map((option, indexOption) => (\n                  <DropdownItem\n                    key={option.value}\n                    aria-disabled={!!option.disabled}\n                    tabIndex={!option.disabled ? 0 : -1}\n                    aria-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              </Stack>\n            </Stack>\n          ))}\n        </>\n      )}\n      {loadMore ? <LoadMore>{loadMore}</LoadMore> : null}\n    </DropdownContainer>\n  ) : (\n    <DropdownContainer\n      role=\"listbox\"\n      tabIndex={-1}\n      id=\"select-dropdown\"\n      onKeyDown={handleKeyDownSelect}\n      gap={0.25}\n      data-grouped={false}\n    >\n      {selectAll && multiselect ? (\n        <Stack id=\"items\" gap={0.25} tabIndex={-1}>\n          <DropdownItem\n            aria-disabled={false}\n            aria-selected={selectedData.allSelected}\n            aria-label=\"select-all\"\n            data-testid=\"select-all\"\n            tabIndex={0}\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              aria-disabled={!!option.disabled}\n              aria-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              tabIndex={!option.disabled ? 0 : -1}\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  size,\n}: DropdownProps) => {\n  const {\n    setIsDropdownVisible,\n    isDropdownVisible,\n    onSearch,\n    searchInput,\n    options,\n    displayedOptions,\n  } = useSelectInput()\n  const theme = useTheme()\n  const [searchBarActive, setSearchBarActive] = useState(false)\n  const [defaultSearchValue, setDefaultSearch] = useState<string | null>(null)\n  const ref = useRef<HTMLDivElement>(null)\n  const [search, setSearch] = useState<string>('')\n  const [maxWidth, setWidth] = useState<string | number>()\n\n  useEffect(() => {\n    if (refSelect.current && isDropdownVisible) {\n      const position =\n        refSelect.current.getBoundingClientRect().bottom +\n        DROPDOWN_MAX_HEIGHT +\n        INPUT_SIZE_HEIGHT[size] +\n        parseInt(theme.space['5'], 10)\n      const overflow = position - window.innerHeight\n      if (overflow > 0) {\n        const modalElement = document.getElementById('backdrop-modal')\n\n        if (modalElement) {\n          modalElement.scrollBy({ top: overflow, behavior: 'smooth' })\n        } else window.scrollBy({ top: overflow, behavior: 'smooth' })\n      }\n    }\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, [isDropdownVisible, refSelect, size, ref.current])\n\n  const resizeDropdown = useCallback(() => {\n    if (\n      refSelect.current &&\n      refSelect.current.getBoundingClientRect().width > 0\n    ) {\n      setWidth(refSelect.current.getBoundingClientRect().width)\n    }\n  }, [refSelect])\n\n  useEffect(() => {\n    resizeDropdown()\n\n    window.addEventListener('resize', resizeDropdown)\n\n    return () => window.removeEventListener('resize', resizeDropdown)\n  }, [resizeDropdown])\n\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      disableAnimation\n      maxWidth={maxWidth ?? refSelect.current?.offsetWidth}\n      hasArrow={false}\n      ref={ref}\n      tabIndex={0}\n      role=\"dialog\"\n      debounceDelay={0}\n      containerFullWidth\n    >\n      {children}\n    </StyledPopup>\n  )\n}\n"]} */"));
176
+ }) => 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":"AA6K8B","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/Dropdown.tsx","sourcesContent":["import { useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport { useCallback, useEffect, useMemo, useRef, useState } from 'react'\nimport type {\n  Dispatch,\n  KeyboardEvent,\n  ReactNode,\n  RefObject,\n  SetStateAction,\n} 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, INPUT_SIZE_HEIGHT, type OptionType } from './types'\n\nconst DROPDOWN_MAX_HEIGHT = 256\n\nexport type DropdownProps = {\n  children: ReactNode\n  emptyState: ReactNode\n  descriptionDirection: 'row' | 'column'\n  searchable: boolean\n  placeholder: string\n  footer?: ((closeDropdown: () => void) => ReactNode) | ReactNode\n  refSelect: RefObject<HTMLDivElement>\n  loadMore?: ReactNode\n  optionalInfoPlacement: 'left' | 'right'\n  isLoading?: boolean\n  size: 'small' | 'medium' | 'large'\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  min-width: 320px;\n  background-color: ${({ theme }) => theme.colors.other.elevation.background.raised};\n  color: ${({ theme }) => theme.colors.neutral.text};\n  box-shadow: ${({ theme }) => `${theme.shadows.raised[0]}, ${theme.shadows.raised[1]}`};\n  padding: ${({ theme }) => theme.space[0]};\n  margin-bottom: ${({ theme }) => theme.space[10]};\n`\n\nconst DropdownContainer = styled(Stack)<{ 'data-grouped': boolean }>`\n  max-height: ${DROPDOWN_MAX_HEIGHT}px;\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 }) =>\n      theme.colors.neutral.backgroundWeak};\n  }\n\n  &[data-selectgroup='true']:focus {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundHover};\n  }\n`\nconst DropdownGroupWrapper = styled.div`\n  position: sticky;\n  top: 0px;\n`\nconst DropdownItem = styled.div<{\n  'aria-selected': boolean\n  'aria-disabled': boolean\n}>`\n  text-align:left;\n  border: none;\n  background-color: ${({ theme }) => theme.colors.other.elevation.background.raised};\n\n  padding: ${({ theme }) => theme.space['1.5']} ${({ theme }) =>\n    theme.space['2']} ${({ theme }) => theme.space['1.5']} ${({ theme }) =>\n    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  &[aria-selected='true'] {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n  }\n\n  &[aria-disabled=\"true\"] {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n  }\n\n  &[aria-disabled=\"true\"]:hover, [aria-disabled=\"true\"]:focus {\n    background-color: ${({ theme }) =>\n      theme.colors.neutral.backgroundStrongDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textStrongDisabled};\n    cursor: not-allowed;\n    outline: none;\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 > div[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 > div[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 = (event: KeyboardEvent<HTMLDivElement>) => {\n  if (event.key === 'ArrowDown') {\n    event.preventDefault()\n    moveFocusDown()\n  }\n\n  if (event.key === 'ArrowUp') {\n    event.preventDefault()\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)\n    onSearch(options)\n  }\n}\n\nconst handleKeyDown = (\n  event: globalThis.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<HTMLDivElement>(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={handleKeyDownSelect}\n      data-grouped\n    >\n      {isLoading ? (\n        <Skeleton variant=\"block\" />\n      ) : (\n        <>\n          {selectAll && multiselect ? (\n            <Stack id=\"items\">\n              <DropdownItem\n                aria-disabled={false}\n                aria-selected={selectedData.allSelected}\n                aria-label=\"select-all\"\n                data-testid=\"select-all\"\n                id=\"select-all\"\n                tabIndex={0}\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                  {group ? (\n                    <DropdownGroup\n                      key={group}\n                      type=\"button\"\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\n                          variant=\"caption\"\n                          as=\"span\"\n                          placement=\"left\"\n                          sentiment=\"neutral\"\n                        >\n                          {group.toUpperCase()}\n                        </Text>\n                      )}\n                    </DropdownGroup>\n                  ) : null}\n                </DropdownGroupWrapper>\n              ) : null}\n              <Stack id=\"items\" gap=\"0.25\">\n                {displayedOptions[group].map((option, indexOption) => (\n                  <DropdownItem\n                    key={option.value}\n                    aria-disabled={!!option.disabled}\n                    tabIndex={!option.disabled ? 0 : -1}\n                    aria-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              </Stack>\n            </Stack>\n          ))}\n        </>\n      )}\n      {loadMore ? <LoadMore>{loadMore}</LoadMore> : null}\n    </DropdownContainer>\n  ) : (\n    <DropdownContainer\n      role=\"listbox\"\n      tabIndex={-1}\n      id=\"select-dropdown\"\n      onKeyDown={handleKeyDownSelect}\n      gap={0.25}\n      data-grouped={false}\n    >\n      {selectAll && multiselect ? (\n        <Stack id=\"items\" gap={0.25} tabIndex={-1}>\n          <DropdownItem\n            aria-disabled={false}\n            aria-selected={selectedData.allSelected}\n            aria-label=\"select-all\"\n            data-testid=\"select-all\"\n            tabIndex={0}\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              aria-disabled={!!option.disabled}\n              aria-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              tabIndex={!option.disabled ? 0 : -1}\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  size,\n}: DropdownProps) => {\n  const {\n    setIsDropdownVisible,\n    isDropdownVisible,\n    onSearch,\n    searchInput,\n    options,\n    displayedOptions,\n    numberOfOptions,\n  } = useSelectInput()\n  const theme = useTheme()\n  const [searchBarActive, setSearchBarActive] = useState(false)\n  const [defaultSearchValue, setDefaultSearch] = useState<string | null>(null)\n  const ref = useRef<HTMLDivElement>(null)\n  const [search, setSearch] = useState<string>('')\n  const [maxWidth, setWidth] = useState<string | number>()\n\n  useEffect(() => {\n    if (refSelect.current && isDropdownVisible) {\n      const position =\n        refSelect.current.getBoundingClientRect().bottom +\n        DROPDOWN_MAX_HEIGHT +\n        INPUT_SIZE_HEIGHT[size] +\n        parseInt(theme.space['5'], 10)\n      const overflow = position - window.innerHeight\n      if (overflow > 0) {\n        const modalElement = document.getElementById('backdrop-modal')\n\n        if (modalElement) {\n          modalElement.scrollBy({ top: overflow, behavior: 'smooth' })\n        } else window.scrollBy({ top: overflow, behavior: 'smooth' })\n      }\n    }\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, [isDropdownVisible, refSelect, size, ref.current])\n\n  const resizeDropdown = useCallback(() => {\n    if (\n      refSelect.current &&\n      refSelect.current.getBoundingClientRect().width > 0\n    ) {\n      setWidth(refSelect.current.getBoundingClientRect().width)\n    }\n  }, [refSelect])\n\n  useEffect(() => {\n    resizeDropdown()\n\n    window.addEventListener('resize', resizeDropdown)\n\n    return () => window.removeEventListener('resize', resizeDropdown)\n  }, [resizeDropdown])\n\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\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  // No data is displayed (because of the search or because no data is provided)\n  // Set to true when noData by default\n  const isEmpty = useMemo(() => {\n    if (numberOfOptions === 0) return true\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, numberOfOptions])\n\n  const computedFooter = useMemo(() => {\n    if (footer) {\n      if (typeof footer === 'function') {\n        return (\n          <PopupFooter>{footer(() => setIsDropdownVisible(false))}</PopupFooter>\n        )\n      }\n\n      return <PopupFooter>{footer}</PopupFooter>\n    }\n\n    return null\n  }, [footer, setIsDropdownVisible])\n\n  return (\n    <StyledPopup\n      visible={isDropdownVisible}\n      text={\n        <Stack>\n          {searchable && !isLoading && numberOfOptions >= 6 ? (\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          {computedFooter}\n        </Stack>\n      }\n      placement=\"bottom\"\n      disableAnimation\n      maxWidth={maxWidth ?? refSelect.current?.offsetWidth}\n      hasArrow={false}\n      ref={ref}\n      tabIndex={0}\n      role=\"dialog\"\n      debounceDelay={0}\n      containerFullWidth\n      align=\"start\"\n    >\n      {children}\n    </StyledPopup>\n  )\n}\n"]} */"));
177
177
  const moveFocusDown = () => {
178
178
  const options = document.querySelectorAll('#items > div[role="option"]:not([disabled])');
179
179
  const activeItem = document.activeElement;
@@ -384,7 +384,8 @@ const Dropdown = ({
384
384
  onSearch,
385
385
  searchInput,
386
386
  options,
387
- displayedOptions
387
+ displayedOptions,
388
+ numberOfOptions
388
389
  } = SelectInputProvider.useSelectInput();
389
390
  const theme = react.useTheme();
390
391
  const [searchBarActive, setSearchBarActive] = React.useState(false);
@@ -442,6 +443,7 @@ const Dropdown = ({
442
443
  };
443
444
  }, [isDropdownVisible, searchBarActive, options, onSearch, search, refSelect, setDefaultSearch, setIsDropdownVisible, searchable]);
444
445
  const isEmpty = React.useMemo(() => {
446
+ if (numberOfOptions === 0) return true;
445
447
  if (Array.isArray(displayedOptions)) {
446
448
  return !(displayedOptions.length > 0);
447
449
  }
@@ -452,11 +454,20 @@ const Dropdown = ({
452
454
  }
453
455
  }
454
456
  return true;
455
- }, [displayedOptions]);
457
+ }, [displayedOptions, numberOfOptions]);
458
+ const computedFooter = React.useMemo(() => {
459
+ if (footer) {
460
+ if (typeof footer === "function") {
461
+ return /* @__PURE__ */ jsxRuntime.jsx(PopupFooter, { children: footer(() => setIsDropdownVisible(false)) });
462
+ }
463
+ return /* @__PURE__ */ jsxRuntime.jsx(PopupFooter, { children: footer });
464
+ }
465
+ return null;
466
+ }, [footer, setIsDropdownVisible]);
456
467
  return /* @__PURE__ */ jsxRuntime.jsx(StyledPopup, { visible: isDropdownVisible, text: /* @__PURE__ */ jsxRuntime.jsxs(index$1.Stack, { children: [
457
- searchable && !isLoading ? /* @__PURE__ */ jsxRuntime.jsx(SearchBarDropdown.SearchBarDropdown, { placeholder, displayedOptions, setSearchBarActive }) : null,
468
+ searchable && !isLoading && numberOfOptions >= 6 ? /* @__PURE__ */ jsxRuntime.jsx(SearchBarDropdown.SearchBarDropdown, { placeholder, displayedOptions, setSearchBarActive }) : null,
458
469
  /* @__PURE__ */ jsxRuntime.jsx(CreateDropdown, { isEmpty, emptyState, descriptionDirection, loadMore, optionalInfoPlacement, defaultSearchValue, isLoading }),
459
- footer ? /* @__PURE__ */ jsxRuntime.jsx(PopupFooter, { children: footer }) : null
460
- ] }), placement: "bottom", disableAnimation: true, maxWidth: maxWidth ?? refSelect.current?.offsetWidth, hasArrow: false, ref, tabIndex: 0, role: "dialog", debounceDelay: 0, containerFullWidth: true, children });
470
+ computedFooter
471
+ ] }), placement: "bottom", disableAnimation: true, maxWidth: maxWidth ?? refSelect.current?.offsetWidth, hasArrow: false, ref, tabIndex: 0, role: "dialog", debounceDelay: 0, containerFullWidth: true, align: "start", children });
461
472
  };
462
473
  exports.Dropdown = Dropdown;