@ultraviolet/ui 1.72.1 → 1.72.2

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.
@@ -1,6 +1,7 @@
1
1
  import { jsx, jsxs, Fragment } from "@emotion/react/jsx-runtime";
2
2
  import _styled from "@emotion/styled/base";
3
- import { useState, useRef, useCallback, useEffect, useMemo } from "react";
3
+ import { useTheme } from "@emotion/react";
4
+ import { useState, useRef, useEffect, useCallback, useMemo } from "react";
4
5
  import { Checkbox } from "../Checkbox/index.js";
5
6
  import { Popup } from "../Popup/index.js";
6
7
  import { Skeleton } from "../Skeleton/index.js";
@@ -9,9 +10,11 @@ import { Text } from "../Text/index.js";
9
10
  import { DisplayOption } from "./DropdownOption.js";
10
11
  import { SearchBarDropdown } from "./SearchBarDropdown.js";
11
12
  import { useSelectInput } from "./SelectInputProvider.js";
13
+ import { INPUT_SIZE_HEIGHT } from "./types.js";
12
14
  function _EMOTION_STRINGIFIED_CSS_ERROR__() {
13
15
  return "You have tried to stringify object returned from `css` function. It isn't supposed to be used directly (e.g. as value of the `className` prop), but rather handed to emotion so it can handle it (e.g. as value of `css` prop).";
14
16
  }
17
+ const DROPDOWN_MAX_HEIGHT = 256;
15
18
  const NON_SEARCHABLE_KEYS = ["Tab", " ", "Enter", "CapsLock", "Shift", "ArrowDown", "ArrowUp", "ArrowLeft", "ArrowRight", "Escape"];
16
19
  const StyledPopup = /* @__PURE__ */ _styled(Popup, process.env.NODE_ENV === "production" ? {
17
20
  target: "eeucz6c8"
@@ -26,13 +29,15 @@ const StyledPopup = /* @__PURE__ */ _styled(Popup, process.env.NODE_ENV === "pro
26
29
  theme
27
30
  }) => `${theme.shadows.raised[0]}, ${theme.shadows.raised[1]}`, ";padding:", ({
28
31
  theme
29
- }) => theme.space[0], ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/Dropdown.tsx"],"names":[],"mappings":"AAiDiC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/Dropdown.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { useCallback, useEffect, useMemo, useRef, useState } from 'react'\nimport type { Dispatch, ReactNode, RefObject, SetStateAction } from 'react'\nimport { Checkbox } from '../Checkbox'\nimport { Popup } from '../Popup'\nimport { Skeleton } from '../Skeleton'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\nimport { DisplayOption } from './DropdownOption'\nimport { SearchBarDropdown } from './SearchBarDropdown'\nimport { useSelectInput } from './SelectInputProvider'\nimport type { DataType, OptionType } from './types'\n\nexport type DropdownProps = {\n  children: ReactNode\n  emptyState: ReactNode\n  descriptionDirection: 'row' | 'column'\n  searchable: boolean\n  placeholder: string\n  footer?: ReactNode\n  refSelect: RefObject<HTMLDivElement>\n  loadMore?: ReactNode\n  optionalInfoPlacement: 'left' | 'right'\n  isLoading?: boolean\n}\n\nexport type CreateDropdownProps = {\n  isEmpty: boolean\n  emptyState: ReactNode\n  descriptionDirection: 'row' | 'column'\n  loadMore?: ReactNode\n  optionalInfoPlacement: 'left' | 'right'\n  defaultSearchValue: string | null\n  isLoading?: boolean\n}\n\nconst NON_SEARCHABLE_KEYS = [\n  'Tab',\n  ' ',\n  'Enter',\n  'CapsLock',\n  'Shift',\n  'ArrowDown',\n  'ArrowUp',\n  'ArrowLeft',\n  'ArrowRight',\n  'Escape',\n]\n\nconst StyledPopup = styled(Popup)`\n  width: 100%;\n  background-color: ${({ theme }) => theme.colors.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`\n\nconst DropdownContainer = styled(Stack)<{ 'data-grouped': boolean }>`\n  max-height: 256px;\n  overflow-y: scroll;\n  padding: ${({ theme }) => theme.space[0]};\n  padding-bottom: ${({ theme }) => theme.space[0.5]};\n  padding-top: ${({ theme }) => theme.space[0.5]};\n\n  &[data-grouped=\"true\"] {\n    padding-top: ${({ theme }) => theme.space[0]};\n  }\n`\nconst DropdownGroup = styled.button<{ 'data-selectgroup': boolean }>`\n  display: flex;\n  width: 100%;\n  justify-content: left;\n  align-items: center;\n  border: none;\n  background-color: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n  position: sticky;\n  top: 0px;\n  padding-right: ${({ theme }) => theme.space[2]};\n  padding-left: ${({ theme }) => theme.space[2]};\n  height: ${({ theme }) => theme.space[4]};\n  text-align: left;\n  margin-bottom: ${({ theme }) => theme.space['0.25']};\n\n  &:focus {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n    outline: none;\n  }\n\n  &[data-selectgroup='true'] {\n    padding-left: ${({ theme }) => theme.space[2]};\n    border-left: ${({ theme }) => theme.space[0.5]} solid ${({ theme }) =>\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.button<{\n  'data-selected': boolean\n  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  &[data-selected='true'] {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n  }\n\n  &[disabled] {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n\n  }\n\n  &[disabled]:hover, [disabled]:focus {\n    background-color: ${({ theme }) =>\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 > button[role=\"option\"]:not([disabled])',\n  )\n  const activeItem = document.activeElement\n  if (options) {\n    for (let i = 0; i < options?.length; i += 1) {\n      const listLength = options.length\n      if (activeItem === options[i] && activeItem !== options[listLength - 1]) {\n        ;(options[i + 1] as HTMLElement).focus()\n      }\n    }\n  }\n}\nconst moveFocusUp = () => {\n  const options = document.querySelectorAll(\n    '#items > button[role=\"option\"]:not([disabled])',\n  )\n  const activeItem = document.activeElement\n\n  if (options) {\n    for (let i = 0; i < options.length; i += 1) {\n      if (activeItem === options[i] && activeItem !== options[0]) {\n        ;(options[i - 1] as HTMLElement).focus()\n      }\n    }\n  }\n}\nconst handleKeyDownSelect = (key: string) => {\n  if (key === 'ArrowDown') {\n    moveFocusDown()\n  }\n  if (key === 'ArrowUp') {\n    moveFocusUp()\n  }\n}\nconst handleClickOutside = (\n  event: MouseEvent,\n  ref: RefObject<HTMLDivElement>,\n  setIsDropdownVisibile: Dispatch<SetStateAction<boolean>>,\n  refSelect: RefObject<HTMLDivElement>,\n  onSearch: Dispatch<SetStateAction<DataType>>,\n  options: DataType,\n) => {\n  if (\n    ref.current &&\n    !ref.current.contains(event.target as Node) &&\n    !refSelect.current?.contains(event.target as Node)\n  ) {\n    setIsDropdownVisibile(false) // hide dropdown when clicking outside of the dropdown\n    onSearch(options) // reset displayed options to default when dropdown is hidden\n  }\n}\n\nconst handleKeyDown = (\n  event: KeyboardEvent,\n  ref: RefObject<HTMLDivElement>,\n  options: DataType,\n  searchBarActive: boolean,\n  setSearch: Dispatch<SetStateAction<string>>,\n  setDefaultSearch: Dispatch<SetStateAction<string | null>>,\n  search: string,\n) => {\n  // Deals with default search\n  if (\n    ref.current &&\n    !searchBarActive &&\n    !NON_SEARCHABLE_KEYS.includes(event.key) &&\n    document.activeElement?.ariaLabel !== 'search-bar'\n  ) {\n    const currentSearch = search + event.key\n    setSearch(currentSearch)\n    ref.current.focus()\n    if (!Array.isArray(options)) {\n      const closestOptions = { ...options }\n      Object.keys(closestOptions).map((group: string) => {\n        closestOptions[group] = closestOptions[group].filter(option =>\n          option.searchText\n            ? option.searchText.toLocaleLowerCase().startsWith(currentSearch)\n            : option.value.toLocaleLowerCase().startsWith(currentSearch),\n        )\n\n        return null\n      })\n      const closestOption = closestOptions[Object.keys(closestOptions)[0]][0]\n      if (closestOption) {\n        setDefaultSearch(closestOption.searchText ?? closestOption.value)\n      } else {\n        setDefaultSearch(null)\n      }\n    } else {\n      const closestOption = [...options].filter(option =>\n        option.searchText\n          ? option.searchText.toLocaleLowerCase().startsWith(currentSearch)\n          : option.value.toLocaleLowerCase().startsWith(currentSearch),\n      )[0]\n      if (closestOption) {\n        setDefaultSearch(closestOption.searchText ?? closestOption.value)\n      } else {\n        setDefaultSearch(null)\n      }\n    }\n  }\n}\nconst CreateDropdown = ({\n  isEmpty,\n  emptyState,\n  descriptionDirection,\n  loadMore,\n  optionalInfoPlacement,\n  defaultSearchValue,\n  isLoading,\n}: CreateDropdownProps) => {\n  const {\n    setIsDropdownVisible,\n    onChange,\n    options,\n    multiselect,\n    selectAll,\n    selectAllGroup,\n    displayedOptions,\n    setSelectedData,\n    selectedData,\n  } = useSelectInput()\n  const focusedItemRef = useRef<HTMLButtonElement>(null)\n\n  useEffect(() => {\n    if (defaultSearchValue && focusedItemRef?.current) {\n      focusedItemRef.current.focus()\n    }\n  }, [defaultSearchValue])\n\n  if (isEmpty) {\n    return (\n      <EmptyState gap={2} alignItems=\"center\">\n        {emptyState ?? (\n          <Text variant=\"bodyStrong\" as=\"p\">\n            No options\n          </Text>\n        )}\n      </EmptyState>\n    )\n  }\n\n  const handleClick = (clickedOption: OptionType, group?: string) => {\n    setSelectedData({ type: 'selectOption', clickedOption, group })\n    if (multiselect) {\n      if (selectedData.selectedValues.includes(clickedOption.value)) {\n        onChange?.(\n          selectedData.selectedValues.filter(\n            val => val !== clickedOption.value,\n          ),\n        )\n      } else {\n        onChange?.([...selectedData.selectedValues, clickedOption.value])\n      }\n    } else {\n      onChange?.(clickedOption.value)\n    }\n    setIsDropdownVisible(multiselect) // hide the dropdown on click when single select only\n  }\n\n  const selectAllOptions = () => {\n    if (multiselect) {\n      setSelectedData({ type: 'selectAll' })\n      if (selectedData.allSelected && onChange) {\n        onChange([])\n      } else {\n        const allValues: OptionType[] = []\n        if (!Array.isArray(options)) {\n          Object.keys(options).map((group: string) =>\n            options[group].map(option => {\n              if (!option.disabled) {\n                allValues.push(option)\n              }\n\n              return null\n            }),\n          )\n        } else {\n          options.map(option => allValues.push(option))\n        }\n        onChange?.(allValues.map(value => value.value))\n      }\n    }\n  }\n\n  const handleSelectGroup = (group: string) => {\n    if (multiselect) {\n      setSelectedData({ type: 'selectGroup', selectedGroup: group })\n      if (!Array.isArray(options)) {\n        if (selectedData.selectedGroups.includes(group)) {\n          const newSelectedValues = [...selectedData.selectedValues].filter(\n            selectedValue =>\n              !options[group].find(option => option.value === selectedValue),\n          )\n          onChange?.(newSelectedValues)\n        } else {\n          const newSelectedValues = [...selectedData.selectedValues]\n\n          options[group].map(option =>\n            newSelectedValues.includes(option.value) || option.disabled\n              ? null\n              : newSelectedValues.push(option.value),\n          )\n          onChange?.(newSelectedValues)\n        }\n      }\n    }\n  }\n\n  return !Array.isArray(displayedOptions) ? (\n    <DropdownContainer\n      role=\"listbox\"\n      id=\"select-dropdown\"\n      onKeyDown={event => {\n        event.preventDefault()\n        handleKeyDownSelect(event.key)\n      }}\n      data-grouped\n    >\n      {isLoading ? (\n        <Skeleton variant=\"block\" />\n      ) : (\n        <>\n          {selectAll && multiselect ? (\n            <Stack id=\"items\">\n              <DropdownItem\n                disabled={false}\n                data-selected={selectedData.allSelected}\n                aria-label=\"select-all\"\n                data-testid=\"select-all\"\n                id=\"select-all\"\n                type=\"button\"\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              (!Array.isArray(options) && options[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                    disabled={!!option.disabled}\n                    type=\"button\"\n                    data-selected={\n                      selectedData.selectedValues.includes(option.value) &&\n                      !option.disabled\n                    }\n                    aria-label={option.value}\n                    data-testid={`option-${option.value}`}\n                    id={`option-${indexOption}`}\n                    role=\"option\"\n                    onClick={() => {\n                      if (!option.disabled) {\n                        handleClick(option, group)\n                      }\n                    }}\n                    onKeyDown={event =>\n                      [' ', 'Enter'].includes(event.key)\n                        ? handleClick(option, group)\n                        : null\n                    }\n                    ref={\n                      option.value === defaultSearchValue ||\n                      option.searchText === defaultSearchValue\n                        ? focusedItemRef\n                        : null\n                    }\n                  >\n                    {multiselect ? (\n                      <StyledCheckbox\n                        checked={\n                          selectedData.selectedValues.includes(option.value) &&\n                          !option.disabled\n                        }\n                        disabled={option.disabled}\n                        value={option.value}\n                        tabIndex={-1}\n                      >\n                        <DisplayOption\n                          option={option}\n                          descriptionDirection={descriptionDirection}\n                          optionalInfoPlacement={optionalInfoPlacement}\n                        />\n                      </StyledCheckbox>\n                    ) : (\n                      <DisplayOption\n                        option={option}\n                        descriptionDirection={descriptionDirection}\n                        optionalInfoPlacement={optionalInfoPlacement}\n                      />\n                    )}\n                  </DropdownItem>\n                ))}\n              </Stack>\n            </Stack>\n          ))}\n        </>\n      )}\n      {loadMore ? <LoadMore>{loadMore}</LoadMore> : null}\n    </DropdownContainer>\n  ) : (\n    <DropdownContainer\n      role=\"listbox\"\n      id=\"select-dropdown\"\n      onKeyDown={event => {\n        event.preventDefault()\n        handleKeyDownSelect(event.key)\n      }}\n      gap={0.25}\n      data-grouped={false}\n    >\n      {selectAll && multiselect ? (\n        <Stack id=\"items\" gap={0.25}>\n          <DropdownItem\n            disabled={false}\n            data-selected={selectedData.allSelected}\n            aria-label=\"select-all\"\n            data-testid=\"select-all\"\n            type=\"button\"\n            role=\"option\"\n            onKeyDown={event =>\n              [' ', 'Enter'].includes(event.key) ? selectAllOptions() : null\n            }\n            onClick={selectAllOptions}\n          >\n            <StyledCheckbox\n              checked={selectedData.allSelected}\n              disabled={false}\n              value=\"select-all\"\n              data-testid=\"select-all-checkbox\"\n              tabIndex={-1}\n            >\n              <Stack direction=\"column\">\n                <Text as=\"span\" variant=\"body\" placement=\"left\">\n                  {selectAll.label}\n                </Text>\n                <Text\n                  as=\"span\"\n                  variant=\"bodySmall\"\n                  sentiment=\"neutral\"\n                  placement=\"left\"\n                  prominence=\"weak\"\n                >\n                  {selectAll.description}\n                </Text>\n              </Stack>\n            </StyledCheckbox>\n          </DropdownItem>\n        </Stack>\n      ) : null}\n      <Stack id=\"items\" gap={0.25}>\n        {isLoading ? (\n          <Skeleton variant=\"block\" />\n        ) : (\n          displayedOptions.map((option, index) => (\n            <DropdownItem\n              key={option.value}\n              disabled={!!option.disabled}\n              data-selected={\n                selectedData.selectedValues.includes(option.value) &&\n                !option.disabled\n              }\n              onClick={() => {\n                if (!option.disabled) {\n                  handleClick(option)\n                }\n              }}\n              aria-label={option.value}\n              data-testid={`option-${option.value}`}\n              id={`option-${index}`}\n              role=\"option\"\n              type=\"button\"\n              ref={\n                option.value === defaultSearchValue ||\n                option.searchText === defaultSearchValue\n                  ? focusedItemRef\n                  : null\n              }\n              onKeyDown={event =>\n                [' ', 'Enter'].includes(event.key) ? handleClick(option) : null\n              }\n            >\n              {multiselect ? (\n                <StyledCheckbox\n                  checked={\n                    selectedData.selectedValues.includes(option.value) &&\n                    !option.disabled\n                  }\n                  disabled={option.disabled}\n                  value={option.value}\n                  tabIndex={-1}\n                >\n                  <DisplayOption\n                    option={option}\n                    descriptionDirection={descriptionDirection}\n                    optionalInfoPlacement={optionalInfoPlacement}\n                  />\n                </StyledCheckbox>\n              ) : (\n                <DisplayOption\n                  option={option}\n                  descriptionDirection={descriptionDirection}\n                  optionalInfoPlacement={optionalInfoPlacement}\n                />\n              )}\n            </DropdownItem>\n          ))\n        )}\n        {loadMore ? <LoadMore>{loadMore}</LoadMore> : null}\n      </Stack>\n    </DropdownContainer>\n  )\n}\nexport const Dropdown = ({\n  children,\n  emptyState,\n  descriptionDirection,\n  searchable,\n  placeholder,\n  footer,\n  refSelect,\n  loadMore,\n  optionalInfoPlacement,\n  isLoading,\n}: DropdownProps) => {\n  const {\n    setIsDropdownVisible,\n    isDropdownVisible,\n    onSearch,\n    searchInput,\n    options,\n    displayedOptions,\n  } = useSelectInput()\n  const [searchBarActive, setSearchBarActive] = useState(false)\n  const [defaultSearchValue, setDefaultSearch] = useState<string | null>(null)\n  const ref = useRef<HTMLDivElement>(null)\n  const [search, setSearch] = useState<string>('')\n  const [maxWidth, setWidth] = useState<string | number>()\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"]} */"));
32
+ }) => theme.space[0], ";margin-bottom:", ({
33
+ theme
34
+ }) => 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  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  'data-selected': boolean\n  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  &[data-selected='true'] {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n  }\n\n  &[disabled] {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n  }\n\n  &[disabled]:hover, [disabled]: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                disabled={false}\n                data-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              (!Array.isArray(options) && options[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                    disabled={!!option.disabled}\n                    tabIndex={!option.disabled ? 0 : -1}\n                    data-selected={\n                      selectedData.selectedValues.includes(option.value) &&\n                      !option.disabled\n                    }\n                    aria-label={option.value}\n                    data-testid={`option-${option.value}`}\n                    id={`option-${indexOption}`}\n                    role=\"option\"\n                    onClick={() => {\n                      if (!option.disabled) {\n                        handleClick(option, group)\n                      }\n                    }}\n                    onKeyDown={event =>\n                      [' ', 'Enter'].includes(event.key)\n                        ? handleClick(option, group)\n                        : null\n                    }\n                    ref={\n                      option.value === defaultSearchValue ||\n                      option.searchText === defaultSearchValue\n                        ? focusedItemRef\n                        : null\n                    }\n                  >\n                    {multiselect ? (\n                      <StyledCheckbox\n                        checked={\n                          selectedData.selectedValues.includes(option.value) &&\n                          !option.disabled\n                        }\n                        disabled={option.disabled}\n                        value={option.value}\n                        tabIndex={-1}\n                      >\n                        <DisplayOption\n                          option={option}\n                          descriptionDirection={descriptionDirection}\n                          optionalInfoPlacement={optionalInfoPlacement}\n                        />\n                      </StyledCheckbox>\n                    ) : (\n                      <DisplayOption\n                        option={option}\n                        descriptionDirection={descriptionDirection}\n                        optionalInfoPlacement={optionalInfoPlacement}\n                      />\n                    )}\n                  </DropdownItem>\n                ))}\n              </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            disabled={false}\n            data-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              disabled={!!option.disabled}\n              data-selected={\n                selectedData.selectedValues.includes(option.value) &&\n                !option.disabled\n              }\n              onClick={() => {\n                if (!option.disabled) {\n                  handleClick(option)\n                }\n              }}\n              aria-label={option.value}\n              data-testid={`option-${option.value}`}\n              id={`option-${index}`}\n              role=\"option\"\n              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"]} */"));
30
35
  const DropdownContainer = /* @__PURE__ */ _styled(Stack, process.env.NODE_ENV === "production" ? {
31
36
  target: "eeucz6c7"
32
37
  } : {
33
38
  target: "eeucz6c7",
34
39
  label: "DropdownContainer"
35
- })("max-height:256px;overflow-y:scroll;padding:", ({
40
+ })("max-height:", DROPDOWN_MAX_HEIGHT, "px;overflow-y:scroll;padding:", ({
36
41
  theme
37
42
  }) => theme.space[0], ";padding-bottom:", ({
38
43
  theme
@@ -40,7 +45,7 @@ const DropdownContainer = /* @__PURE__ */ _styled(Stack, process.env.NODE_ENV ==
40
45
  theme
41
46
  }) => theme.space[0.5], ';&[data-grouped="true"]{padding-top:', ({
42
47
  theme
43
- }) => theme.space[0], ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/Dropdown.tsx"],"names":[],"mappings":"AAyDoE","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/Dropdown.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { useCallback, useEffect, useMemo, useRef, useState } from 'react'\nimport type { Dispatch, ReactNode, RefObject, SetStateAction } from 'react'\nimport { Checkbox } from '../Checkbox'\nimport { Popup } from '../Popup'\nimport { Skeleton } from '../Skeleton'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\nimport { DisplayOption } from './DropdownOption'\nimport { SearchBarDropdown } from './SearchBarDropdown'\nimport { useSelectInput } from './SelectInputProvider'\nimport type { DataType, OptionType } from './types'\n\nexport type DropdownProps = {\n  children: ReactNode\n  emptyState: ReactNode\n  descriptionDirection: 'row' | 'column'\n  searchable: boolean\n  placeholder: string\n  footer?: ReactNode\n  refSelect: RefObject<HTMLDivElement>\n  loadMore?: ReactNode\n  optionalInfoPlacement: 'left' | 'right'\n  isLoading?: boolean\n}\n\nexport type CreateDropdownProps = {\n  isEmpty: boolean\n  emptyState: ReactNode\n  descriptionDirection: 'row' | 'column'\n  loadMore?: ReactNode\n  optionalInfoPlacement: 'left' | 'right'\n  defaultSearchValue: string | null\n  isLoading?: boolean\n}\n\nconst NON_SEARCHABLE_KEYS = [\n  'Tab',\n  ' ',\n  'Enter',\n  'CapsLock',\n  'Shift',\n  'ArrowDown',\n  'ArrowUp',\n  'ArrowLeft',\n  'ArrowRight',\n  'Escape',\n]\n\nconst StyledPopup = styled(Popup)`\n  width: 100%;\n  background-color: ${({ theme }) => theme.colors.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`\n\nconst DropdownContainer = styled(Stack)<{ 'data-grouped': boolean }>`\n  max-height: 256px;\n  overflow-y: scroll;\n  padding: ${({ theme }) => theme.space[0]};\n  padding-bottom: ${({ theme }) => theme.space[0.5]};\n  padding-top: ${({ theme }) => theme.space[0.5]};\n\n  &[data-grouped=\"true\"] {\n    padding-top: ${({ theme }) => theme.space[0]};\n  }\n`\nconst DropdownGroup = styled.button<{ 'data-selectgroup': boolean }>`\n  display: flex;\n  width: 100%;\n  justify-content: left;\n  align-items: center;\n  border: none;\n  background-color: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n  position: sticky;\n  top: 0px;\n  padding-right: ${({ theme }) => theme.space[2]};\n  padding-left: ${({ theme }) => theme.space[2]};\n  height: ${({ theme }) => theme.space[4]};\n  text-align: left;\n  margin-bottom: ${({ theme }) => theme.space['0.25']};\n\n  &:focus {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n    outline: none;\n  }\n\n  &[data-selectgroup='true'] {\n    padding-left: ${({ theme }) => theme.space[2]};\n    border-left: ${({ theme }) => theme.space[0.5]} solid ${({ theme }) =>\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.button<{\n  'data-selected': boolean\n  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  &[data-selected='true'] {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n  }\n\n  &[disabled] {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n\n  }\n\n  &[disabled]:hover, [disabled]:focus {\n    background-color: ${({ theme }) =>\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 > button[role=\"option\"]:not([disabled])',\n  )\n  const activeItem = document.activeElement\n  if (options) {\n    for (let i = 0; i < options?.length; i += 1) {\n      const listLength = options.length\n      if (activeItem === options[i] && activeItem !== options[listLength - 1]) {\n        ;(options[i + 1] as HTMLElement).focus()\n      }\n    }\n  }\n}\nconst moveFocusUp = () => {\n  const options = document.querySelectorAll(\n    '#items > button[role=\"option\"]:not([disabled])',\n  )\n  const activeItem = document.activeElement\n\n  if (options) {\n    for (let i = 0; i < options.length; i += 1) {\n      if (activeItem === options[i] && activeItem !== options[0]) {\n        ;(options[i - 1] as HTMLElement).focus()\n      }\n    }\n  }\n}\nconst handleKeyDownSelect = (key: string) => {\n  if (key === 'ArrowDown') {\n    moveFocusDown()\n  }\n  if (key === 'ArrowUp') {\n    moveFocusUp()\n  }\n}\nconst handleClickOutside = (\n  event: MouseEvent,\n  ref: RefObject<HTMLDivElement>,\n  setIsDropdownVisibile: Dispatch<SetStateAction<boolean>>,\n  refSelect: RefObject<HTMLDivElement>,\n  onSearch: Dispatch<SetStateAction<DataType>>,\n  options: DataType,\n) => {\n  if (\n    ref.current &&\n    !ref.current.contains(event.target as Node) &&\n    !refSelect.current?.contains(event.target as Node)\n  ) {\n    setIsDropdownVisibile(false) // hide dropdown when clicking outside of the dropdown\n    onSearch(options) // reset displayed options to default when dropdown is hidden\n  }\n}\n\nconst handleKeyDown = (\n  event: KeyboardEvent,\n  ref: RefObject<HTMLDivElement>,\n  options: DataType,\n  searchBarActive: boolean,\n  setSearch: Dispatch<SetStateAction<string>>,\n  setDefaultSearch: Dispatch<SetStateAction<string | null>>,\n  search: string,\n) => {\n  // Deals with default search\n  if (\n    ref.current &&\n    !searchBarActive &&\n    !NON_SEARCHABLE_KEYS.includes(event.key) &&\n    document.activeElement?.ariaLabel !== 'search-bar'\n  ) {\n    const currentSearch = search + event.key\n    setSearch(currentSearch)\n    ref.current.focus()\n    if (!Array.isArray(options)) {\n      const closestOptions = { ...options }\n      Object.keys(closestOptions).map((group: string) => {\n        closestOptions[group] = closestOptions[group].filter(option =>\n          option.searchText\n            ? option.searchText.toLocaleLowerCase().startsWith(currentSearch)\n            : option.value.toLocaleLowerCase().startsWith(currentSearch),\n        )\n\n        return null\n      })\n      const closestOption = closestOptions[Object.keys(closestOptions)[0]][0]\n      if (closestOption) {\n        setDefaultSearch(closestOption.searchText ?? closestOption.value)\n      } else {\n        setDefaultSearch(null)\n      }\n    } else {\n      const closestOption = [...options].filter(option =>\n        option.searchText\n          ? option.searchText.toLocaleLowerCase().startsWith(currentSearch)\n          : option.value.toLocaleLowerCase().startsWith(currentSearch),\n      )[0]\n      if (closestOption) {\n        setDefaultSearch(closestOption.searchText ?? closestOption.value)\n      } else {\n        setDefaultSearch(null)\n      }\n    }\n  }\n}\nconst CreateDropdown = ({\n  isEmpty,\n  emptyState,\n  descriptionDirection,\n  loadMore,\n  optionalInfoPlacement,\n  defaultSearchValue,\n  isLoading,\n}: CreateDropdownProps) => {\n  const {\n    setIsDropdownVisible,\n    onChange,\n    options,\n    multiselect,\n    selectAll,\n    selectAllGroup,\n    displayedOptions,\n    setSelectedData,\n    selectedData,\n  } = useSelectInput()\n  const focusedItemRef = useRef<HTMLButtonElement>(null)\n\n  useEffect(() => {\n    if (defaultSearchValue && focusedItemRef?.current) {\n      focusedItemRef.current.focus()\n    }\n  }, [defaultSearchValue])\n\n  if (isEmpty) {\n    return (\n      <EmptyState gap={2} alignItems=\"center\">\n        {emptyState ?? (\n          <Text variant=\"bodyStrong\" as=\"p\">\n            No options\n          </Text>\n        )}\n      </EmptyState>\n    )\n  }\n\n  const handleClick = (clickedOption: OptionType, group?: string) => {\n    setSelectedData({ type: 'selectOption', clickedOption, group })\n    if (multiselect) {\n      if (selectedData.selectedValues.includes(clickedOption.value)) {\n        onChange?.(\n          selectedData.selectedValues.filter(\n            val => val !== clickedOption.value,\n          ),\n        )\n      } else {\n        onChange?.([...selectedData.selectedValues, clickedOption.value])\n      }\n    } else {\n      onChange?.(clickedOption.value)\n    }\n    setIsDropdownVisible(multiselect) // hide the dropdown on click when single select only\n  }\n\n  const selectAllOptions = () => {\n    if (multiselect) {\n      setSelectedData({ type: 'selectAll' })\n      if (selectedData.allSelected && onChange) {\n        onChange([])\n      } else {\n        const allValues: OptionType[] = []\n        if (!Array.isArray(options)) {\n          Object.keys(options).map((group: string) =>\n            options[group].map(option => {\n              if (!option.disabled) {\n                allValues.push(option)\n              }\n\n              return null\n            }),\n          )\n        } else {\n          options.map(option => allValues.push(option))\n        }\n        onChange?.(allValues.map(value => value.value))\n      }\n    }\n  }\n\n  const handleSelectGroup = (group: string) => {\n    if (multiselect) {\n      setSelectedData({ type: 'selectGroup', selectedGroup: group })\n      if (!Array.isArray(options)) {\n        if (selectedData.selectedGroups.includes(group)) {\n          const newSelectedValues = [...selectedData.selectedValues].filter(\n            selectedValue =>\n              !options[group].find(option => option.value === selectedValue),\n          )\n          onChange?.(newSelectedValues)\n        } else {\n          const newSelectedValues = [...selectedData.selectedValues]\n\n          options[group].map(option =>\n            newSelectedValues.includes(option.value) || option.disabled\n              ? null\n              : newSelectedValues.push(option.value),\n          )\n          onChange?.(newSelectedValues)\n        }\n      }\n    }\n  }\n\n  return !Array.isArray(displayedOptions) ? (\n    <DropdownContainer\n      role=\"listbox\"\n      id=\"select-dropdown\"\n      onKeyDown={event => {\n        event.preventDefault()\n        handleKeyDownSelect(event.key)\n      }}\n      data-grouped\n    >\n      {isLoading ? (\n        <Skeleton variant=\"block\" />\n      ) : (\n        <>\n          {selectAll && multiselect ? (\n            <Stack id=\"items\">\n              <DropdownItem\n                disabled={false}\n                data-selected={selectedData.allSelected}\n                aria-label=\"select-all\"\n                data-testid=\"select-all\"\n                id=\"select-all\"\n                type=\"button\"\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              (!Array.isArray(options) && options[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                    disabled={!!option.disabled}\n                    type=\"button\"\n                    data-selected={\n                      selectedData.selectedValues.includes(option.value) &&\n                      !option.disabled\n                    }\n                    aria-label={option.value}\n                    data-testid={`option-${option.value}`}\n                    id={`option-${indexOption}`}\n                    role=\"option\"\n                    onClick={() => {\n                      if (!option.disabled) {\n                        handleClick(option, group)\n                      }\n                    }}\n                    onKeyDown={event =>\n                      [' ', 'Enter'].includes(event.key)\n                        ? handleClick(option, group)\n                        : null\n                    }\n                    ref={\n                      option.value === defaultSearchValue ||\n                      option.searchText === defaultSearchValue\n                        ? focusedItemRef\n                        : null\n                    }\n                  >\n                    {multiselect ? (\n                      <StyledCheckbox\n                        checked={\n                          selectedData.selectedValues.includes(option.value) &&\n                          !option.disabled\n                        }\n                        disabled={option.disabled}\n                        value={option.value}\n                        tabIndex={-1}\n                      >\n                        <DisplayOption\n                          option={option}\n                          descriptionDirection={descriptionDirection}\n                          optionalInfoPlacement={optionalInfoPlacement}\n                        />\n                      </StyledCheckbox>\n                    ) : (\n                      <DisplayOption\n                        option={option}\n                        descriptionDirection={descriptionDirection}\n                        optionalInfoPlacement={optionalInfoPlacement}\n                      />\n                    )}\n                  </DropdownItem>\n                ))}\n              </Stack>\n            </Stack>\n          ))}\n        </>\n      )}\n      {loadMore ? <LoadMore>{loadMore}</LoadMore> : null}\n    </DropdownContainer>\n  ) : (\n    <DropdownContainer\n      role=\"listbox\"\n      id=\"select-dropdown\"\n      onKeyDown={event => {\n        event.preventDefault()\n        handleKeyDownSelect(event.key)\n      }}\n      gap={0.25}\n      data-grouped={false}\n    >\n      {selectAll && multiselect ? (\n        <Stack id=\"items\" gap={0.25}>\n          <DropdownItem\n            disabled={false}\n            data-selected={selectedData.allSelected}\n            aria-label=\"select-all\"\n            data-testid=\"select-all\"\n            type=\"button\"\n            role=\"option\"\n            onKeyDown={event =>\n              [' ', 'Enter'].includes(event.key) ? selectAllOptions() : null\n            }\n            onClick={selectAllOptions}\n          >\n            <StyledCheckbox\n              checked={selectedData.allSelected}\n              disabled={false}\n              value=\"select-all\"\n              data-testid=\"select-all-checkbox\"\n              tabIndex={-1}\n            >\n              <Stack direction=\"column\">\n                <Text as=\"span\" variant=\"body\" placement=\"left\">\n                  {selectAll.label}\n                </Text>\n                <Text\n                  as=\"span\"\n                  variant=\"bodySmall\"\n                  sentiment=\"neutral\"\n                  placement=\"left\"\n                  prominence=\"weak\"\n                >\n                  {selectAll.description}\n                </Text>\n              </Stack>\n            </StyledCheckbox>\n          </DropdownItem>\n        </Stack>\n      ) : null}\n      <Stack id=\"items\" gap={0.25}>\n        {isLoading ? (\n          <Skeleton variant=\"block\" />\n        ) : (\n          displayedOptions.map((option, index) => (\n            <DropdownItem\n              key={option.value}\n              disabled={!!option.disabled}\n              data-selected={\n                selectedData.selectedValues.includes(option.value) &&\n                !option.disabled\n              }\n              onClick={() => {\n                if (!option.disabled) {\n                  handleClick(option)\n                }\n              }}\n              aria-label={option.value}\n              data-testid={`option-${option.value}`}\n              id={`option-${index}`}\n              role=\"option\"\n              type=\"button\"\n              ref={\n                option.value === defaultSearchValue ||\n                option.searchText === defaultSearchValue\n                  ? focusedItemRef\n                  : null\n              }\n              onKeyDown={event =>\n                [' ', 'Enter'].includes(event.key) ? handleClick(option) : null\n              }\n            >\n              {multiselect ? (\n                <StyledCheckbox\n                  checked={\n                    selectedData.selectedValues.includes(option.value) &&\n                    !option.disabled\n                  }\n                  disabled={option.disabled}\n                  value={option.value}\n                  tabIndex={-1}\n                >\n                  <DisplayOption\n                    option={option}\n                    descriptionDirection={descriptionDirection}\n                    optionalInfoPlacement={optionalInfoPlacement}\n                  />\n                </StyledCheckbox>\n              ) : (\n                <DisplayOption\n                  option={option}\n                  descriptionDirection={descriptionDirection}\n                  optionalInfoPlacement={optionalInfoPlacement}\n                />\n              )}\n            </DropdownItem>\n          ))\n        )}\n        {loadMore ? <LoadMore>{loadMore}</LoadMore> : null}\n      </Stack>\n    </DropdownContainer>\n  )\n}\nexport const Dropdown = ({\n  children,\n  emptyState,\n  descriptionDirection,\n  searchable,\n  placeholder,\n  footer,\n  refSelect,\n  loadMore,\n  optionalInfoPlacement,\n  isLoading,\n}: DropdownProps) => {\n  const {\n    setIsDropdownVisible,\n    isDropdownVisible,\n    onSearch,\n    searchInput,\n    options,\n    displayedOptions,\n  } = useSelectInput()\n  const [searchBarActive, setSearchBarActive] = useState(false)\n  const [defaultSearchValue, setDefaultSearch] = useState<string | null>(null)\n  const ref = useRef<HTMLDivElement>(null)\n  const [search, setSearch] = useState<string>('')\n  const [maxWidth, setWidth] = useState<string | number>()\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"]} */"));
48
+ }) => 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":"AAoEoE","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  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  'data-selected': boolean\n  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  &[data-selected='true'] {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n  }\n\n  &[disabled] {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n  }\n\n  &[disabled]:hover, [disabled]: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                disabled={false}\n                data-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              (!Array.isArray(options) && options[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                    disabled={!!option.disabled}\n                    tabIndex={!option.disabled ? 0 : -1}\n                    data-selected={\n                      selectedData.selectedValues.includes(option.value) &&\n                      !option.disabled\n                    }\n                    aria-label={option.value}\n                    data-testid={`option-${option.value}`}\n                    id={`option-${indexOption}`}\n                    role=\"option\"\n                    onClick={() => {\n                      if (!option.disabled) {\n                        handleClick(option, group)\n                      }\n                    }}\n                    onKeyDown={event =>\n                      [' ', 'Enter'].includes(event.key)\n                        ? handleClick(option, group)\n                        : null\n                    }\n                    ref={\n                      option.value === defaultSearchValue ||\n                      option.searchText === defaultSearchValue\n                        ? focusedItemRef\n                        : null\n                    }\n                  >\n                    {multiselect ? (\n                      <StyledCheckbox\n                        checked={\n                          selectedData.selectedValues.includes(option.value) &&\n                          !option.disabled\n                        }\n                        disabled={option.disabled}\n                        value={option.value}\n                        tabIndex={-1}\n                      >\n                        <DisplayOption\n                          option={option}\n                          descriptionDirection={descriptionDirection}\n                          optionalInfoPlacement={optionalInfoPlacement}\n                        />\n                      </StyledCheckbox>\n                    ) : (\n                      <DisplayOption\n                        option={option}\n                        descriptionDirection={descriptionDirection}\n                        optionalInfoPlacement={optionalInfoPlacement}\n                      />\n                    )}\n                  </DropdownItem>\n                ))}\n              </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            disabled={false}\n            data-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              disabled={!!option.disabled}\n              data-selected={\n                selectedData.selectedValues.includes(option.value) &&\n                !option.disabled\n              }\n              onClick={() => {\n                if (!option.disabled) {\n                  handleClick(option)\n                }\n              }}\n              aria-label={option.value}\n              data-testid={`option-${option.value}`}\n              id={`option-${index}`}\n              role=\"option\"\n              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"]} */"));
44
49
  const DropdownGroup = /* @__PURE__ */ _styled("button", process.env.NODE_ENV === "production" ? {
45
50
  target: "eeucz6c6"
46
51
  } : {
@@ -66,7 +71,7 @@ const DropdownGroup = /* @__PURE__ */ _styled("button", process.env.NODE_ENV ===
66
71
  theme
67
72
  }) => theme.colors.neutral.backgroundWeak, ";}&[data-selectgroup='true']:focus{background-color:", ({
68
73
  theme
69
- }) => theme.colors.neutral.backgroundHover, ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/Dropdown.tsx"],"names":[],"mappings":"AAoEoE","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/Dropdown.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { useCallback, useEffect, useMemo, useRef, useState } from 'react'\nimport type { Dispatch, ReactNode, RefObject, SetStateAction } from 'react'\nimport { Checkbox } from '../Checkbox'\nimport { Popup } from '../Popup'\nimport { Skeleton } from '../Skeleton'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\nimport { DisplayOption } from './DropdownOption'\nimport { SearchBarDropdown } from './SearchBarDropdown'\nimport { useSelectInput } from './SelectInputProvider'\nimport type { DataType, OptionType } from './types'\n\nexport type DropdownProps = {\n  children: ReactNode\n  emptyState: ReactNode\n  descriptionDirection: 'row' | 'column'\n  searchable: boolean\n  placeholder: string\n  footer?: ReactNode\n  refSelect: RefObject<HTMLDivElement>\n  loadMore?: ReactNode\n  optionalInfoPlacement: 'left' | 'right'\n  isLoading?: boolean\n}\n\nexport type CreateDropdownProps = {\n  isEmpty: boolean\n  emptyState: ReactNode\n  descriptionDirection: 'row' | 'column'\n  loadMore?: ReactNode\n  optionalInfoPlacement: 'left' | 'right'\n  defaultSearchValue: string | null\n  isLoading?: boolean\n}\n\nconst NON_SEARCHABLE_KEYS = [\n  'Tab',\n  ' ',\n  'Enter',\n  'CapsLock',\n  'Shift',\n  'ArrowDown',\n  'ArrowUp',\n  'ArrowLeft',\n  'ArrowRight',\n  'Escape',\n]\n\nconst StyledPopup = styled(Popup)`\n  width: 100%;\n  background-color: ${({ theme }) => theme.colors.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`\n\nconst DropdownContainer = styled(Stack)<{ 'data-grouped': boolean }>`\n  max-height: 256px;\n  overflow-y: scroll;\n  padding: ${({ theme }) => theme.space[0]};\n  padding-bottom: ${({ theme }) => theme.space[0.5]};\n  padding-top: ${({ theme }) => theme.space[0.5]};\n\n  &[data-grouped=\"true\"] {\n    padding-top: ${({ theme }) => theme.space[0]};\n  }\n`\nconst DropdownGroup = styled.button<{ 'data-selectgroup': boolean }>`\n  display: flex;\n  width: 100%;\n  justify-content: left;\n  align-items: center;\n  border: none;\n  background-color: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n  position: sticky;\n  top: 0px;\n  padding-right: ${({ theme }) => theme.space[2]};\n  padding-left: ${({ theme }) => theme.space[2]};\n  height: ${({ theme }) => theme.space[4]};\n  text-align: left;\n  margin-bottom: ${({ theme }) => theme.space['0.25']};\n\n  &:focus {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n    outline: none;\n  }\n\n  &[data-selectgroup='true'] {\n    padding-left: ${({ theme }) => theme.space[2]};\n    border-left: ${({ theme }) => theme.space[0.5]} solid ${({ theme }) =>\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.button<{\n  'data-selected': boolean\n  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  &[data-selected='true'] {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n  }\n\n  &[disabled] {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n\n  }\n\n  &[disabled]:hover, [disabled]:focus {\n    background-color: ${({ theme }) =>\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 > button[role=\"option\"]:not([disabled])',\n  )\n  const activeItem = document.activeElement\n  if (options) {\n    for (let i = 0; i < options?.length; i += 1) {\n      const listLength = options.length\n      if (activeItem === options[i] && activeItem !== options[listLength - 1]) {\n        ;(options[i + 1] as HTMLElement).focus()\n      }\n    }\n  }\n}\nconst moveFocusUp = () => {\n  const options = document.querySelectorAll(\n    '#items > button[role=\"option\"]:not([disabled])',\n  )\n  const activeItem = document.activeElement\n\n  if (options) {\n    for (let i = 0; i < options.length; i += 1) {\n      if (activeItem === options[i] && activeItem !== options[0]) {\n        ;(options[i - 1] as HTMLElement).focus()\n      }\n    }\n  }\n}\nconst handleKeyDownSelect = (key: string) => {\n  if (key === 'ArrowDown') {\n    moveFocusDown()\n  }\n  if (key === 'ArrowUp') {\n    moveFocusUp()\n  }\n}\nconst handleClickOutside = (\n  event: MouseEvent,\n  ref: RefObject<HTMLDivElement>,\n  setIsDropdownVisibile: Dispatch<SetStateAction<boolean>>,\n  refSelect: RefObject<HTMLDivElement>,\n  onSearch: Dispatch<SetStateAction<DataType>>,\n  options: DataType,\n) => {\n  if (\n    ref.current &&\n    !ref.current.contains(event.target as Node) &&\n    !refSelect.current?.contains(event.target as Node)\n  ) {\n    setIsDropdownVisibile(false) // hide dropdown when clicking outside of the dropdown\n    onSearch(options) // reset displayed options to default when dropdown is hidden\n  }\n}\n\nconst handleKeyDown = (\n  event: KeyboardEvent,\n  ref: RefObject<HTMLDivElement>,\n  options: DataType,\n  searchBarActive: boolean,\n  setSearch: Dispatch<SetStateAction<string>>,\n  setDefaultSearch: Dispatch<SetStateAction<string | null>>,\n  search: string,\n) => {\n  // Deals with default search\n  if (\n    ref.current &&\n    !searchBarActive &&\n    !NON_SEARCHABLE_KEYS.includes(event.key) &&\n    document.activeElement?.ariaLabel !== 'search-bar'\n  ) {\n    const currentSearch = search + event.key\n    setSearch(currentSearch)\n    ref.current.focus()\n    if (!Array.isArray(options)) {\n      const closestOptions = { ...options }\n      Object.keys(closestOptions).map((group: string) => {\n        closestOptions[group] = closestOptions[group].filter(option =>\n          option.searchText\n            ? option.searchText.toLocaleLowerCase().startsWith(currentSearch)\n            : option.value.toLocaleLowerCase().startsWith(currentSearch),\n        )\n\n        return null\n      })\n      const closestOption = closestOptions[Object.keys(closestOptions)[0]][0]\n      if (closestOption) {\n        setDefaultSearch(closestOption.searchText ?? closestOption.value)\n      } else {\n        setDefaultSearch(null)\n      }\n    } else {\n      const closestOption = [...options].filter(option =>\n        option.searchText\n          ? option.searchText.toLocaleLowerCase().startsWith(currentSearch)\n          : option.value.toLocaleLowerCase().startsWith(currentSearch),\n      )[0]\n      if (closestOption) {\n        setDefaultSearch(closestOption.searchText ?? closestOption.value)\n      } else {\n        setDefaultSearch(null)\n      }\n    }\n  }\n}\nconst CreateDropdown = ({\n  isEmpty,\n  emptyState,\n  descriptionDirection,\n  loadMore,\n  optionalInfoPlacement,\n  defaultSearchValue,\n  isLoading,\n}: CreateDropdownProps) => {\n  const {\n    setIsDropdownVisible,\n    onChange,\n    options,\n    multiselect,\n    selectAll,\n    selectAllGroup,\n    displayedOptions,\n    setSelectedData,\n    selectedData,\n  } = useSelectInput()\n  const focusedItemRef = useRef<HTMLButtonElement>(null)\n\n  useEffect(() => {\n    if (defaultSearchValue && focusedItemRef?.current) {\n      focusedItemRef.current.focus()\n    }\n  }, [defaultSearchValue])\n\n  if (isEmpty) {\n    return (\n      <EmptyState gap={2} alignItems=\"center\">\n        {emptyState ?? (\n          <Text variant=\"bodyStrong\" as=\"p\">\n            No options\n          </Text>\n        )}\n      </EmptyState>\n    )\n  }\n\n  const handleClick = (clickedOption: OptionType, group?: string) => {\n    setSelectedData({ type: 'selectOption', clickedOption, group })\n    if (multiselect) {\n      if (selectedData.selectedValues.includes(clickedOption.value)) {\n        onChange?.(\n          selectedData.selectedValues.filter(\n            val => val !== clickedOption.value,\n          ),\n        )\n      } else {\n        onChange?.([...selectedData.selectedValues, clickedOption.value])\n      }\n    } else {\n      onChange?.(clickedOption.value)\n    }\n    setIsDropdownVisible(multiselect) // hide the dropdown on click when single select only\n  }\n\n  const selectAllOptions = () => {\n    if (multiselect) {\n      setSelectedData({ type: 'selectAll' })\n      if (selectedData.allSelected && onChange) {\n        onChange([])\n      } else {\n        const allValues: OptionType[] = []\n        if (!Array.isArray(options)) {\n          Object.keys(options).map((group: string) =>\n            options[group].map(option => {\n              if (!option.disabled) {\n                allValues.push(option)\n              }\n\n              return null\n            }),\n          )\n        } else {\n          options.map(option => allValues.push(option))\n        }\n        onChange?.(allValues.map(value => value.value))\n      }\n    }\n  }\n\n  const handleSelectGroup = (group: string) => {\n    if (multiselect) {\n      setSelectedData({ type: 'selectGroup', selectedGroup: group })\n      if (!Array.isArray(options)) {\n        if (selectedData.selectedGroups.includes(group)) {\n          const newSelectedValues = [...selectedData.selectedValues].filter(\n            selectedValue =>\n              !options[group].find(option => option.value === selectedValue),\n          )\n          onChange?.(newSelectedValues)\n        } else {\n          const newSelectedValues = [...selectedData.selectedValues]\n\n          options[group].map(option =>\n            newSelectedValues.includes(option.value) || option.disabled\n              ? null\n              : newSelectedValues.push(option.value),\n          )\n          onChange?.(newSelectedValues)\n        }\n      }\n    }\n  }\n\n  return !Array.isArray(displayedOptions) ? (\n    <DropdownContainer\n      role=\"listbox\"\n      id=\"select-dropdown\"\n      onKeyDown={event => {\n        event.preventDefault()\n        handleKeyDownSelect(event.key)\n      }}\n      data-grouped\n    >\n      {isLoading ? (\n        <Skeleton variant=\"block\" />\n      ) : (\n        <>\n          {selectAll && multiselect ? (\n            <Stack id=\"items\">\n              <DropdownItem\n                disabled={false}\n                data-selected={selectedData.allSelected}\n                aria-label=\"select-all\"\n                data-testid=\"select-all\"\n                id=\"select-all\"\n                type=\"button\"\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              (!Array.isArray(options) && options[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                    disabled={!!option.disabled}\n                    type=\"button\"\n                    data-selected={\n                      selectedData.selectedValues.includes(option.value) &&\n                      !option.disabled\n                    }\n                    aria-label={option.value}\n                    data-testid={`option-${option.value}`}\n                    id={`option-${indexOption}`}\n                    role=\"option\"\n                    onClick={() => {\n                      if (!option.disabled) {\n                        handleClick(option, group)\n                      }\n                    }}\n                    onKeyDown={event =>\n                      [' ', 'Enter'].includes(event.key)\n                        ? handleClick(option, group)\n                        : null\n                    }\n                    ref={\n                      option.value === defaultSearchValue ||\n                      option.searchText === defaultSearchValue\n                        ? focusedItemRef\n                        : null\n                    }\n                  >\n                    {multiselect ? (\n                      <StyledCheckbox\n                        checked={\n                          selectedData.selectedValues.includes(option.value) &&\n                          !option.disabled\n                        }\n                        disabled={option.disabled}\n                        value={option.value}\n                        tabIndex={-1}\n                      >\n                        <DisplayOption\n                          option={option}\n                          descriptionDirection={descriptionDirection}\n                          optionalInfoPlacement={optionalInfoPlacement}\n                        />\n                      </StyledCheckbox>\n                    ) : (\n                      <DisplayOption\n                        option={option}\n                        descriptionDirection={descriptionDirection}\n                        optionalInfoPlacement={optionalInfoPlacement}\n                      />\n                    )}\n                  </DropdownItem>\n                ))}\n              </Stack>\n            </Stack>\n          ))}\n        </>\n      )}\n      {loadMore ? <LoadMore>{loadMore}</LoadMore> : null}\n    </DropdownContainer>\n  ) : (\n    <DropdownContainer\n      role=\"listbox\"\n      id=\"select-dropdown\"\n      onKeyDown={event => {\n        event.preventDefault()\n        handleKeyDownSelect(event.key)\n      }}\n      gap={0.25}\n      data-grouped={false}\n    >\n      {selectAll && multiselect ? (\n        <Stack id=\"items\" gap={0.25}>\n          <DropdownItem\n            disabled={false}\n            data-selected={selectedData.allSelected}\n            aria-label=\"select-all\"\n            data-testid=\"select-all\"\n            type=\"button\"\n            role=\"option\"\n            onKeyDown={event =>\n              [' ', 'Enter'].includes(event.key) ? selectAllOptions() : null\n            }\n            onClick={selectAllOptions}\n          >\n            <StyledCheckbox\n              checked={selectedData.allSelected}\n              disabled={false}\n              value=\"select-all\"\n              data-testid=\"select-all-checkbox\"\n              tabIndex={-1}\n            >\n              <Stack direction=\"column\">\n                <Text as=\"span\" variant=\"body\" placement=\"left\">\n                  {selectAll.label}\n                </Text>\n                <Text\n                  as=\"span\"\n                  variant=\"bodySmall\"\n                  sentiment=\"neutral\"\n                  placement=\"left\"\n                  prominence=\"weak\"\n                >\n                  {selectAll.description}\n                </Text>\n              </Stack>\n            </StyledCheckbox>\n          </DropdownItem>\n        </Stack>\n      ) : null}\n      <Stack id=\"items\" gap={0.25}>\n        {isLoading ? (\n          <Skeleton variant=\"block\" />\n        ) : (\n          displayedOptions.map((option, index) => (\n            <DropdownItem\n              key={option.value}\n              disabled={!!option.disabled}\n              data-selected={\n                selectedData.selectedValues.includes(option.value) &&\n                !option.disabled\n              }\n              onClick={() => {\n                if (!option.disabled) {\n                  handleClick(option)\n                }\n              }}\n              aria-label={option.value}\n              data-testid={`option-${option.value}`}\n              id={`option-${index}`}\n              role=\"option\"\n              type=\"button\"\n              ref={\n                option.value === defaultSearchValue ||\n                option.searchText === defaultSearchValue\n                  ? focusedItemRef\n                  : null\n              }\n              onKeyDown={event =>\n                [' ', 'Enter'].includes(event.key) ? handleClick(option) : null\n              }\n            >\n              {multiselect ? (\n                <StyledCheckbox\n                  checked={\n                    selectedData.selectedValues.includes(option.value) &&\n                    !option.disabled\n                  }\n                  disabled={option.disabled}\n                  value={option.value}\n                  tabIndex={-1}\n                >\n                  <DisplayOption\n                    option={option}\n                    descriptionDirection={descriptionDirection}\n                    optionalInfoPlacement={optionalInfoPlacement}\n                  />\n                </StyledCheckbox>\n              ) : (\n                <DisplayOption\n                  option={option}\n                  descriptionDirection={descriptionDirection}\n                  optionalInfoPlacement={optionalInfoPlacement}\n                />\n              )}\n            </DropdownItem>\n          ))\n        )}\n        {loadMore ? <LoadMore>{loadMore}</LoadMore> : null}\n      </Stack>\n    </DropdownContainer>\n  )\n}\nexport const Dropdown = ({\n  children,\n  emptyState,\n  descriptionDirection,\n  searchable,\n  placeholder,\n  footer,\n  refSelect,\n  loadMore,\n  optionalInfoPlacement,\n  isLoading,\n}: DropdownProps) => {\n  const {\n    setIsDropdownVisible,\n    isDropdownVisible,\n    onSearch,\n    searchInput,\n    options,\n    displayedOptions,\n  } = useSelectInput()\n  const [searchBarActive, setSearchBarActive] = useState(false)\n  const [defaultSearchValue, setDefaultSearch] = useState<string | null>(null)\n  const ref = useRef<HTMLDivElement>(null)\n  const [search, setSearch] = useState<string>('')\n  const [maxWidth, setWidth] = useState<string | number>()\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"]} */"));
74
+ }) => 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":"AA+EoE","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  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  'data-selected': boolean\n  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  &[data-selected='true'] {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n  }\n\n  &[disabled] {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n  }\n\n  &[disabled]:hover, [disabled]: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                disabled={false}\n                data-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              (!Array.isArray(options) && options[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                    disabled={!!option.disabled}\n                    tabIndex={!option.disabled ? 0 : -1}\n                    data-selected={\n                      selectedData.selectedValues.includes(option.value) &&\n                      !option.disabled\n                    }\n                    aria-label={option.value}\n                    data-testid={`option-${option.value}`}\n                    id={`option-${indexOption}`}\n                    role=\"option\"\n                    onClick={() => {\n                      if (!option.disabled) {\n                        handleClick(option, group)\n                      }\n                    }}\n                    onKeyDown={event =>\n                      [' ', 'Enter'].includes(event.key)\n                        ? handleClick(option, group)\n                        : null\n                    }\n                    ref={\n                      option.value === defaultSearchValue ||\n                      option.searchText === defaultSearchValue\n                        ? focusedItemRef\n                        : null\n                    }\n                  >\n                    {multiselect ? (\n                      <StyledCheckbox\n                        checked={\n                          selectedData.selectedValues.includes(option.value) &&\n                          !option.disabled\n                        }\n                        disabled={option.disabled}\n                        value={option.value}\n                        tabIndex={-1}\n                      >\n                        <DisplayOption\n                          option={option}\n                          descriptionDirection={descriptionDirection}\n                          optionalInfoPlacement={optionalInfoPlacement}\n                        />\n                      </StyledCheckbox>\n                    ) : (\n                      <DisplayOption\n                        option={option}\n                        descriptionDirection={descriptionDirection}\n                        optionalInfoPlacement={optionalInfoPlacement}\n                      />\n                    )}\n                  </DropdownItem>\n                ))}\n              </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            disabled={false}\n            data-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              disabled={!!option.disabled}\n              data-selected={\n                selectedData.selectedValues.includes(option.value) &&\n                !option.disabled\n              }\n              onClick={() => {\n                if (!option.disabled) {\n                  handleClick(option)\n                }\n              }}\n              aria-label={option.value}\n              data-testid={`option-${option.value}`}\n              id={`option-${index}`}\n              role=\"option\"\n              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"]} */"));
70
75
  const DropdownGroupWrapper = /* @__PURE__ */ _styled("div", process.env.NODE_ENV === "production" ? {
71
76
  target: "eeucz6c5"
72
77
  } : {
@@ -78,10 +83,10 @@ const DropdownGroupWrapper = /* @__PURE__ */ _styled("div", process.env.NODE_ENV
78
83
  } : {
79
84
  name: "vkf8h3",
80
85
  styles: "position:sticky;top:0px",
81
- map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/Dropdown.tsx"],"names":[],"mappings":"AAkGuC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/Dropdown.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { useCallback, useEffect, useMemo, useRef, useState } from 'react'\nimport type { Dispatch, ReactNode, RefObject, SetStateAction } from 'react'\nimport { Checkbox } from '../Checkbox'\nimport { Popup } from '../Popup'\nimport { Skeleton } from '../Skeleton'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\nimport { DisplayOption } from './DropdownOption'\nimport { SearchBarDropdown } from './SearchBarDropdown'\nimport { useSelectInput } from './SelectInputProvider'\nimport type { DataType, OptionType } from './types'\n\nexport type DropdownProps = {\n  children: ReactNode\n  emptyState: ReactNode\n  descriptionDirection: 'row' | 'column'\n  searchable: boolean\n  placeholder: string\n  footer?: ReactNode\n  refSelect: RefObject<HTMLDivElement>\n  loadMore?: ReactNode\n  optionalInfoPlacement: 'left' | 'right'\n  isLoading?: boolean\n}\n\nexport type CreateDropdownProps = {\n  isEmpty: boolean\n  emptyState: ReactNode\n  descriptionDirection: 'row' | 'column'\n  loadMore?: ReactNode\n  optionalInfoPlacement: 'left' | 'right'\n  defaultSearchValue: string | null\n  isLoading?: boolean\n}\n\nconst NON_SEARCHABLE_KEYS = [\n  'Tab',\n  ' ',\n  'Enter',\n  'CapsLock',\n  'Shift',\n  'ArrowDown',\n  'ArrowUp',\n  'ArrowLeft',\n  'ArrowRight',\n  'Escape',\n]\n\nconst StyledPopup = styled(Popup)`\n  width: 100%;\n  background-color: ${({ theme }) => theme.colors.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`\n\nconst DropdownContainer = styled(Stack)<{ 'data-grouped': boolean }>`\n  max-height: 256px;\n  overflow-y: scroll;\n  padding: ${({ theme }) => theme.space[0]};\n  padding-bottom: ${({ theme }) => theme.space[0.5]};\n  padding-top: ${({ theme }) => theme.space[0.5]};\n\n  &[data-grouped=\"true\"] {\n    padding-top: ${({ theme }) => theme.space[0]};\n  }\n`\nconst DropdownGroup = styled.button<{ 'data-selectgroup': boolean }>`\n  display: flex;\n  width: 100%;\n  justify-content: left;\n  align-items: center;\n  border: none;\n  background-color: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n  position: sticky;\n  top: 0px;\n  padding-right: ${({ theme }) => theme.space[2]};\n  padding-left: ${({ theme }) => theme.space[2]};\n  height: ${({ theme }) => theme.space[4]};\n  text-align: left;\n  margin-bottom: ${({ theme }) => theme.space['0.25']};\n\n  &:focus {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n    outline: none;\n  }\n\n  &[data-selectgroup='true'] {\n    padding-left: ${({ theme }) => theme.space[2]};\n    border-left: ${({ theme }) => theme.space[0.5]} solid ${({ theme }) =>\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.button<{\n  'data-selected': boolean\n  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  &[data-selected='true'] {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n  }\n\n  &[disabled] {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n\n  }\n\n  &[disabled]:hover, [disabled]:focus {\n    background-color: ${({ theme }) =>\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 > button[role=\"option\"]:not([disabled])',\n  )\n  const activeItem = document.activeElement\n  if (options) {\n    for (let i = 0; i < options?.length; i += 1) {\n      const listLength = options.length\n      if (activeItem === options[i] && activeItem !== options[listLength - 1]) {\n        ;(options[i + 1] as HTMLElement).focus()\n      }\n    }\n  }\n}\nconst moveFocusUp = () => {\n  const options = document.querySelectorAll(\n    '#items > button[role=\"option\"]:not([disabled])',\n  )\n  const activeItem = document.activeElement\n\n  if (options) {\n    for (let i = 0; i < options.length; i += 1) {\n      if (activeItem === options[i] && activeItem !== options[0]) {\n        ;(options[i - 1] as HTMLElement).focus()\n      }\n    }\n  }\n}\nconst handleKeyDownSelect = (key: string) => {\n  if (key === 'ArrowDown') {\n    moveFocusDown()\n  }\n  if (key === 'ArrowUp') {\n    moveFocusUp()\n  }\n}\nconst handleClickOutside = (\n  event: MouseEvent,\n  ref: RefObject<HTMLDivElement>,\n  setIsDropdownVisibile: Dispatch<SetStateAction<boolean>>,\n  refSelect: RefObject<HTMLDivElement>,\n  onSearch: Dispatch<SetStateAction<DataType>>,\n  options: DataType,\n) => {\n  if (\n    ref.current &&\n    !ref.current.contains(event.target as Node) &&\n    !refSelect.current?.contains(event.target as Node)\n  ) {\n    setIsDropdownVisibile(false) // hide dropdown when clicking outside of the dropdown\n    onSearch(options) // reset displayed options to default when dropdown is hidden\n  }\n}\n\nconst handleKeyDown = (\n  event: KeyboardEvent,\n  ref: RefObject<HTMLDivElement>,\n  options: DataType,\n  searchBarActive: boolean,\n  setSearch: Dispatch<SetStateAction<string>>,\n  setDefaultSearch: Dispatch<SetStateAction<string | null>>,\n  search: string,\n) => {\n  // Deals with default search\n  if (\n    ref.current &&\n    !searchBarActive &&\n    !NON_SEARCHABLE_KEYS.includes(event.key) &&\n    document.activeElement?.ariaLabel !== 'search-bar'\n  ) {\n    const currentSearch = search + event.key\n    setSearch(currentSearch)\n    ref.current.focus()\n    if (!Array.isArray(options)) {\n      const closestOptions = { ...options }\n      Object.keys(closestOptions).map((group: string) => {\n        closestOptions[group] = closestOptions[group].filter(option =>\n          option.searchText\n            ? option.searchText.toLocaleLowerCase().startsWith(currentSearch)\n            : option.value.toLocaleLowerCase().startsWith(currentSearch),\n        )\n\n        return null\n      })\n      const closestOption = closestOptions[Object.keys(closestOptions)[0]][0]\n      if (closestOption) {\n        setDefaultSearch(closestOption.searchText ?? closestOption.value)\n      } else {\n        setDefaultSearch(null)\n      }\n    } else {\n      const closestOption = [...options].filter(option =>\n        option.searchText\n          ? option.searchText.toLocaleLowerCase().startsWith(currentSearch)\n          : option.value.toLocaleLowerCase().startsWith(currentSearch),\n      )[0]\n      if (closestOption) {\n        setDefaultSearch(closestOption.searchText ?? closestOption.value)\n      } else {\n        setDefaultSearch(null)\n      }\n    }\n  }\n}\nconst CreateDropdown = ({\n  isEmpty,\n  emptyState,\n  descriptionDirection,\n  loadMore,\n  optionalInfoPlacement,\n  defaultSearchValue,\n  isLoading,\n}: CreateDropdownProps) => {\n  const {\n    setIsDropdownVisible,\n    onChange,\n    options,\n    multiselect,\n    selectAll,\n    selectAllGroup,\n    displayedOptions,\n    setSelectedData,\n    selectedData,\n  } = useSelectInput()\n  const focusedItemRef = useRef<HTMLButtonElement>(null)\n\n  useEffect(() => {\n    if (defaultSearchValue && focusedItemRef?.current) {\n      focusedItemRef.current.focus()\n    }\n  }, [defaultSearchValue])\n\n  if (isEmpty) {\n    return (\n      <EmptyState gap={2} alignItems=\"center\">\n        {emptyState ?? (\n          <Text variant=\"bodyStrong\" as=\"p\">\n            No options\n          </Text>\n        )}\n      </EmptyState>\n    )\n  }\n\n  const handleClick = (clickedOption: OptionType, group?: string) => {\n    setSelectedData({ type: 'selectOption', clickedOption, group })\n    if (multiselect) {\n      if (selectedData.selectedValues.includes(clickedOption.value)) {\n        onChange?.(\n          selectedData.selectedValues.filter(\n            val => val !== clickedOption.value,\n          ),\n        )\n      } else {\n        onChange?.([...selectedData.selectedValues, clickedOption.value])\n      }\n    } else {\n      onChange?.(clickedOption.value)\n    }\n    setIsDropdownVisible(multiselect) // hide the dropdown on click when single select only\n  }\n\n  const selectAllOptions = () => {\n    if (multiselect) {\n      setSelectedData({ type: 'selectAll' })\n      if (selectedData.allSelected && onChange) {\n        onChange([])\n      } else {\n        const allValues: OptionType[] = []\n        if (!Array.isArray(options)) {\n          Object.keys(options).map((group: string) =>\n            options[group].map(option => {\n              if (!option.disabled) {\n                allValues.push(option)\n              }\n\n              return null\n            }),\n          )\n        } else {\n          options.map(option => allValues.push(option))\n        }\n        onChange?.(allValues.map(value => value.value))\n      }\n    }\n  }\n\n  const handleSelectGroup = (group: string) => {\n    if (multiselect) {\n      setSelectedData({ type: 'selectGroup', selectedGroup: group })\n      if (!Array.isArray(options)) {\n        if (selectedData.selectedGroups.includes(group)) {\n          const newSelectedValues = [...selectedData.selectedValues].filter(\n            selectedValue =>\n              !options[group].find(option => option.value === selectedValue),\n          )\n          onChange?.(newSelectedValues)\n        } else {\n          const newSelectedValues = [...selectedData.selectedValues]\n\n          options[group].map(option =>\n            newSelectedValues.includes(option.value) || option.disabled\n              ? null\n              : newSelectedValues.push(option.value),\n          )\n          onChange?.(newSelectedValues)\n        }\n      }\n    }\n  }\n\n  return !Array.isArray(displayedOptions) ? (\n    <DropdownContainer\n      role=\"listbox\"\n      id=\"select-dropdown\"\n      onKeyDown={event => {\n        event.preventDefault()\n        handleKeyDownSelect(event.key)\n      }}\n      data-grouped\n    >\n      {isLoading ? (\n        <Skeleton variant=\"block\" />\n      ) : (\n        <>\n          {selectAll && multiselect ? (\n            <Stack id=\"items\">\n              <DropdownItem\n                disabled={false}\n                data-selected={selectedData.allSelected}\n                aria-label=\"select-all\"\n                data-testid=\"select-all\"\n                id=\"select-all\"\n                type=\"button\"\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              (!Array.isArray(options) && options[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                    disabled={!!option.disabled}\n                    type=\"button\"\n                    data-selected={\n                      selectedData.selectedValues.includes(option.value) &&\n                      !option.disabled\n                    }\n                    aria-label={option.value}\n                    data-testid={`option-${option.value}`}\n                    id={`option-${indexOption}`}\n                    role=\"option\"\n                    onClick={() => {\n                      if (!option.disabled) {\n                        handleClick(option, group)\n                      }\n                    }}\n                    onKeyDown={event =>\n                      [' ', 'Enter'].includes(event.key)\n                        ? handleClick(option, group)\n                        : null\n                    }\n                    ref={\n                      option.value === defaultSearchValue ||\n                      option.searchText === defaultSearchValue\n                        ? focusedItemRef\n                        : null\n                    }\n                  >\n                    {multiselect ? (\n                      <StyledCheckbox\n                        checked={\n                          selectedData.selectedValues.includes(option.value) &&\n                          !option.disabled\n                        }\n                        disabled={option.disabled}\n                        value={option.value}\n                        tabIndex={-1}\n                      >\n                        <DisplayOption\n                          option={option}\n                          descriptionDirection={descriptionDirection}\n                          optionalInfoPlacement={optionalInfoPlacement}\n                        />\n                      </StyledCheckbox>\n                    ) : (\n                      <DisplayOption\n                        option={option}\n                        descriptionDirection={descriptionDirection}\n                        optionalInfoPlacement={optionalInfoPlacement}\n                      />\n                    )}\n                  </DropdownItem>\n                ))}\n              </Stack>\n            </Stack>\n          ))}\n        </>\n      )}\n      {loadMore ? <LoadMore>{loadMore}</LoadMore> : null}\n    </DropdownContainer>\n  ) : (\n    <DropdownContainer\n      role=\"listbox\"\n      id=\"select-dropdown\"\n      onKeyDown={event => {\n        event.preventDefault()\n        handleKeyDownSelect(event.key)\n      }}\n      gap={0.25}\n      data-grouped={false}\n    >\n      {selectAll && multiselect ? (\n        <Stack id=\"items\" gap={0.25}>\n          <DropdownItem\n            disabled={false}\n            data-selected={selectedData.allSelected}\n            aria-label=\"select-all\"\n            data-testid=\"select-all\"\n            type=\"button\"\n            role=\"option\"\n            onKeyDown={event =>\n              [' ', 'Enter'].includes(event.key) ? selectAllOptions() : null\n            }\n            onClick={selectAllOptions}\n          >\n            <StyledCheckbox\n              checked={selectedData.allSelected}\n              disabled={false}\n              value=\"select-all\"\n              data-testid=\"select-all-checkbox\"\n              tabIndex={-1}\n            >\n              <Stack direction=\"column\">\n                <Text as=\"span\" variant=\"body\" placement=\"left\">\n                  {selectAll.label}\n                </Text>\n                <Text\n                  as=\"span\"\n                  variant=\"bodySmall\"\n                  sentiment=\"neutral\"\n                  placement=\"left\"\n                  prominence=\"weak\"\n                >\n                  {selectAll.description}\n                </Text>\n              </Stack>\n            </StyledCheckbox>\n          </DropdownItem>\n        </Stack>\n      ) : null}\n      <Stack id=\"items\" gap={0.25}>\n        {isLoading ? (\n          <Skeleton variant=\"block\" />\n        ) : (\n          displayedOptions.map((option, index) => (\n            <DropdownItem\n              key={option.value}\n              disabled={!!option.disabled}\n              data-selected={\n                selectedData.selectedValues.includes(option.value) &&\n                !option.disabled\n              }\n              onClick={() => {\n                if (!option.disabled) {\n                  handleClick(option)\n                }\n              }}\n              aria-label={option.value}\n              data-testid={`option-${option.value}`}\n              id={`option-${index}`}\n              role=\"option\"\n              type=\"button\"\n              ref={\n                option.value === defaultSearchValue ||\n                option.searchText === defaultSearchValue\n                  ? focusedItemRef\n                  : null\n              }\n              onKeyDown={event =>\n                [' ', 'Enter'].includes(event.key) ? handleClick(option) : null\n              }\n            >\n              {multiselect ? (\n                <StyledCheckbox\n                  checked={\n                    selectedData.selectedValues.includes(option.value) &&\n                    !option.disabled\n                  }\n                  disabled={option.disabled}\n                  value={option.value}\n                  tabIndex={-1}\n                >\n                  <DisplayOption\n                    option={option}\n                    descriptionDirection={descriptionDirection}\n                    optionalInfoPlacement={optionalInfoPlacement}\n                  />\n                </StyledCheckbox>\n              ) : (\n                <DisplayOption\n                  option={option}\n                  descriptionDirection={descriptionDirection}\n                  optionalInfoPlacement={optionalInfoPlacement}\n                />\n              )}\n            </DropdownItem>\n          ))\n        )}\n        {loadMore ? <LoadMore>{loadMore}</LoadMore> : null}\n      </Stack>\n    </DropdownContainer>\n  )\n}\nexport const Dropdown = ({\n  children,\n  emptyState,\n  descriptionDirection,\n  searchable,\n  placeholder,\n  footer,\n  refSelect,\n  loadMore,\n  optionalInfoPlacement,\n  isLoading,\n}: DropdownProps) => {\n  const {\n    setIsDropdownVisible,\n    isDropdownVisible,\n    onSearch,\n    searchInput,\n    options,\n    displayedOptions,\n  } = useSelectInput()\n  const [searchBarActive, setSearchBarActive] = useState(false)\n  const [defaultSearchValue, setDefaultSearch] = useState<string | null>(null)\n  const ref = useRef<HTMLDivElement>(null)\n  const [search, setSearch] = useState<string>('')\n  const [maxWidth, setWidth] = useState<string | number>()\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"]} */",
86
+ 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":"AA6GuC","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  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  'data-selected': boolean\n  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  &[data-selected='true'] {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n  }\n\n  &[disabled] {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n  }\n\n  &[disabled]:hover, [disabled]: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                disabled={false}\n                data-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              (!Array.isArray(options) && options[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                    disabled={!!option.disabled}\n                    tabIndex={!option.disabled ? 0 : -1}\n                    data-selected={\n                      selectedData.selectedValues.includes(option.value) &&\n                      !option.disabled\n                    }\n                    aria-label={option.value}\n                    data-testid={`option-${option.value}`}\n                    id={`option-${indexOption}`}\n                    role=\"option\"\n                    onClick={() => {\n                      if (!option.disabled) {\n                        handleClick(option, group)\n                      }\n                    }}\n                    onKeyDown={event =>\n                      [' ', 'Enter'].includes(event.key)\n                        ? handleClick(option, group)\n                        : null\n                    }\n                    ref={\n                      option.value === defaultSearchValue ||\n                      option.searchText === defaultSearchValue\n                        ? focusedItemRef\n                        : null\n                    }\n                  >\n                    {multiselect ? (\n                      <StyledCheckbox\n                        checked={\n                          selectedData.selectedValues.includes(option.value) &&\n                          !option.disabled\n                        }\n                        disabled={option.disabled}\n                        value={option.value}\n                        tabIndex={-1}\n                      >\n                        <DisplayOption\n                          option={option}\n                          descriptionDirection={descriptionDirection}\n                          optionalInfoPlacement={optionalInfoPlacement}\n                        />\n                      </StyledCheckbox>\n                    ) : (\n                      <DisplayOption\n                        option={option}\n                        descriptionDirection={descriptionDirection}\n                        optionalInfoPlacement={optionalInfoPlacement}\n                      />\n                    )}\n                  </DropdownItem>\n                ))}\n              </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            disabled={false}\n            data-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              disabled={!!option.disabled}\n              data-selected={\n                selectedData.selectedValues.includes(option.value) &&\n                !option.disabled\n              }\n              onClick={() => {\n                if (!option.disabled) {\n                  handleClick(option)\n                }\n              }}\n              aria-label={option.value}\n              data-testid={`option-${option.value}`}\n              id={`option-${index}`}\n              role=\"option\"\n              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"]} */",
82
87
  toString: _EMOTION_STRINGIFIED_CSS_ERROR__
83
88
  });
84
- const DropdownItem = /* @__PURE__ */ _styled("button", process.env.NODE_ENV === "production" ? {
89
+ const DropdownItem = /* @__PURE__ */ _styled("div", process.env.NODE_ENV === "production" ? {
85
90
  target: "eeucz6c4"
86
91
  } : {
87
92
  target: "eeucz6c4",
@@ -118,7 +123,7 @@ const DropdownItem = /* @__PURE__ */ _styled("button", process.env.NODE_ENV ===
118
123
  theme
119
124
  }) => theme.colors.neutral.backgroundStrongDisabled, ";color:", ({
120
125
  theme
121
- }) => theme.colors.neutral.textStrongDisabled, ";cursor:not-allowed;outline:none;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/Dropdown.tsx"],"names":[],"mappings":"AAyGE","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/Dropdown.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { useCallback, useEffect, useMemo, useRef, useState } from 'react'\nimport type { Dispatch, ReactNode, RefObject, SetStateAction } from 'react'\nimport { Checkbox } from '../Checkbox'\nimport { Popup } from '../Popup'\nimport { Skeleton } from '../Skeleton'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\nimport { DisplayOption } from './DropdownOption'\nimport { SearchBarDropdown } from './SearchBarDropdown'\nimport { useSelectInput } from './SelectInputProvider'\nimport type { DataType, OptionType } from './types'\n\nexport type DropdownProps = {\n  children: ReactNode\n  emptyState: ReactNode\n  descriptionDirection: 'row' | 'column'\n  searchable: boolean\n  placeholder: string\n  footer?: ReactNode\n  refSelect: RefObject<HTMLDivElement>\n  loadMore?: ReactNode\n  optionalInfoPlacement: 'left' | 'right'\n  isLoading?: boolean\n}\n\nexport type CreateDropdownProps = {\n  isEmpty: boolean\n  emptyState: ReactNode\n  descriptionDirection: 'row' | 'column'\n  loadMore?: ReactNode\n  optionalInfoPlacement: 'left' | 'right'\n  defaultSearchValue: string | null\n  isLoading?: boolean\n}\n\nconst NON_SEARCHABLE_KEYS = [\n  'Tab',\n  ' ',\n  'Enter',\n  'CapsLock',\n  'Shift',\n  'ArrowDown',\n  'ArrowUp',\n  'ArrowLeft',\n  'ArrowRight',\n  'Escape',\n]\n\nconst StyledPopup = styled(Popup)`\n  width: 100%;\n  background-color: ${({ theme }) => theme.colors.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`\n\nconst DropdownContainer = styled(Stack)<{ 'data-grouped': boolean }>`\n  max-height: 256px;\n  overflow-y: scroll;\n  padding: ${({ theme }) => theme.space[0]};\n  padding-bottom: ${({ theme }) => theme.space[0.5]};\n  padding-top: ${({ theme }) => theme.space[0.5]};\n\n  &[data-grouped=\"true\"] {\n    padding-top: ${({ theme }) => theme.space[0]};\n  }\n`\nconst DropdownGroup = styled.button<{ 'data-selectgroup': boolean }>`\n  display: flex;\n  width: 100%;\n  justify-content: left;\n  align-items: center;\n  border: none;\n  background-color: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n  position: sticky;\n  top: 0px;\n  padding-right: ${({ theme }) => theme.space[2]};\n  padding-left: ${({ theme }) => theme.space[2]};\n  height: ${({ theme }) => theme.space[4]};\n  text-align: left;\n  margin-bottom: ${({ theme }) => theme.space['0.25']};\n\n  &:focus {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n    outline: none;\n  }\n\n  &[data-selectgroup='true'] {\n    padding-left: ${({ theme }) => theme.space[2]};\n    border-left: ${({ theme }) => theme.space[0.5]} solid ${({ theme }) =>\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.button<{\n  'data-selected': boolean\n  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  &[data-selected='true'] {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n  }\n\n  &[disabled] {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n\n  }\n\n  &[disabled]:hover, [disabled]:focus {\n    background-color: ${({ theme }) =>\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 > button[role=\"option\"]:not([disabled])',\n  )\n  const activeItem = document.activeElement\n  if (options) {\n    for (let i = 0; i < options?.length; i += 1) {\n      const listLength = options.length\n      if (activeItem === options[i] && activeItem !== options[listLength - 1]) {\n        ;(options[i + 1] as HTMLElement).focus()\n      }\n    }\n  }\n}\nconst moveFocusUp = () => {\n  const options = document.querySelectorAll(\n    '#items > button[role=\"option\"]:not([disabled])',\n  )\n  const activeItem = document.activeElement\n\n  if (options) {\n    for (let i = 0; i < options.length; i += 1) {\n      if (activeItem === options[i] && activeItem !== options[0]) {\n        ;(options[i - 1] as HTMLElement).focus()\n      }\n    }\n  }\n}\nconst handleKeyDownSelect = (key: string) => {\n  if (key === 'ArrowDown') {\n    moveFocusDown()\n  }\n  if (key === 'ArrowUp') {\n    moveFocusUp()\n  }\n}\nconst handleClickOutside = (\n  event: MouseEvent,\n  ref: RefObject<HTMLDivElement>,\n  setIsDropdownVisibile: Dispatch<SetStateAction<boolean>>,\n  refSelect: RefObject<HTMLDivElement>,\n  onSearch: Dispatch<SetStateAction<DataType>>,\n  options: DataType,\n) => {\n  if (\n    ref.current &&\n    !ref.current.contains(event.target as Node) &&\n    !refSelect.current?.contains(event.target as Node)\n  ) {\n    setIsDropdownVisibile(false) // hide dropdown when clicking outside of the dropdown\n    onSearch(options) // reset displayed options to default when dropdown is hidden\n  }\n}\n\nconst handleKeyDown = (\n  event: KeyboardEvent,\n  ref: RefObject<HTMLDivElement>,\n  options: DataType,\n  searchBarActive: boolean,\n  setSearch: Dispatch<SetStateAction<string>>,\n  setDefaultSearch: Dispatch<SetStateAction<string | null>>,\n  search: string,\n) => {\n  // Deals with default search\n  if (\n    ref.current &&\n    !searchBarActive &&\n    !NON_SEARCHABLE_KEYS.includes(event.key) &&\n    document.activeElement?.ariaLabel !== 'search-bar'\n  ) {\n    const currentSearch = search + event.key\n    setSearch(currentSearch)\n    ref.current.focus()\n    if (!Array.isArray(options)) {\n      const closestOptions = { ...options }\n      Object.keys(closestOptions).map((group: string) => {\n        closestOptions[group] = closestOptions[group].filter(option =>\n          option.searchText\n            ? option.searchText.toLocaleLowerCase().startsWith(currentSearch)\n            : option.value.toLocaleLowerCase().startsWith(currentSearch),\n        )\n\n        return null\n      })\n      const closestOption = closestOptions[Object.keys(closestOptions)[0]][0]\n      if (closestOption) {\n        setDefaultSearch(closestOption.searchText ?? closestOption.value)\n      } else {\n        setDefaultSearch(null)\n      }\n    } else {\n      const closestOption = [...options].filter(option =>\n        option.searchText\n          ? option.searchText.toLocaleLowerCase().startsWith(currentSearch)\n          : option.value.toLocaleLowerCase().startsWith(currentSearch),\n      )[0]\n      if (closestOption) {\n        setDefaultSearch(closestOption.searchText ?? closestOption.value)\n      } else {\n        setDefaultSearch(null)\n      }\n    }\n  }\n}\nconst CreateDropdown = ({\n  isEmpty,\n  emptyState,\n  descriptionDirection,\n  loadMore,\n  optionalInfoPlacement,\n  defaultSearchValue,\n  isLoading,\n}: CreateDropdownProps) => {\n  const {\n    setIsDropdownVisible,\n    onChange,\n    options,\n    multiselect,\n    selectAll,\n    selectAllGroup,\n    displayedOptions,\n    setSelectedData,\n    selectedData,\n  } = useSelectInput()\n  const focusedItemRef = useRef<HTMLButtonElement>(null)\n\n  useEffect(() => {\n    if (defaultSearchValue && focusedItemRef?.current) {\n      focusedItemRef.current.focus()\n    }\n  }, [defaultSearchValue])\n\n  if (isEmpty) {\n    return (\n      <EmptyState gap={2} alignItems=\"center\">\n        {emptyState ?? (\n          <Text variant=\"bodyStrong\" as=\"p\">\n            No options\n          </Text>\n        )}\n      </EmptyState>\n    )\n  }\n\n  const handleClick = (clickedOption: OptionType, group?: string) => {\n    setSelectedData({ type: 'selectOption', clickedOption, group })\n    if (multiselect) {\n      if (selectedData.selectedValues.includes(clickedOption.value)) {\n        onChange?.(\n          selectedData.selectedValues.filter(\n            val => val !== clickedOption.value,\n          ),\n        )\n      } else {\n        onChange?.([...selectedData.selectedValues, clickedOption.value])\n      }\n    } else {\n      onChange?.(clickedOption.value)\n    }\n    setIsDropdownVisible(multiselect) // hide the dropdown on click when single select only\n  }\n\n  const selectAllOptions = () => {\n    if (multiselect) {\n      setSelectedData({ type: 'selectAll' })\n      if (selectedData.allSelected && onChange) {\n        onChange([])\n      } else {\n        const allValues: OptionType[] = []\n        if (!Array.isArray(options)) {\n          Object.keys(options).map((group: string) =>\n            options[group].map(option => {\n              if (!option.disabled) {\n                allValues.push(option)\n              }\n\n              return null\n            }),\n          )\n        } else {\n          options.map(option => allValues.push(option))\n        }\n        onChange?.(allValues.map(value => value.value))\n      }\n    }\n  }\n\n  const handleSelectGroup = (group: string) => {\n    if (multiselect) {\n      setSelectedData({ type: 'selectGroup', selectedGroup: group })\n      if (!Array.isArray(options)) {\n        if (selectedData.selectedGroups.includes(group)) {\n          const newSelectedValues = [...selectedData.selectedValues].filter(\n            selectedValue =>\n              !options[group].find(option => option.value === selectedValue),\n          )\n          onChange?.(newSelectedValues)\n        } else {\n          const newSelectedValues = [...selectedData.selectedValues]\n\n          options[group].map(option =>\n            newSelectedValues.includes(option.value) || option.disabled\n              ? null\n              : newSelectedValues.push(option.value),\n          )\n          onChange?.(newSelectedValues)\n        }\n      }\n    }\n  }\n\n  return !Array.isArray(displayedOptions) ? (\n    <DropdownContainer\n      role=\"listbox\"\n      id=\"select-dropdown\"\n      onKeyDown={event => {\n        event.preventDefault()\n        handleKeyDownSelect(event.key)\n      }}\n      data-grouped\n    >\n      {isLoading ? (\n        <Skeleton variant=\"block\" />\n      ) : (\n        <>\n          {selectAll && multiselect ? (\n            <Stack id=\"items\">\n              <DropdownItem\n                disabled={false}\n                data-selected={selectedData.allSelected}\n                aria-label=\"select-all\"\n                data-testid=\"select-all\"\n                id=\"select-all\"\n                type=\"button\"\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              (!Array.isArray(options) && options[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                    disabled={!!option.disabled}\n                    type=\"button\"\n                    data-selected={\n                      selectedData.selectedValues.includes(option.value) &&\n                      !option.disabled\n                    }\n                    aria-label={option.value}\n                    data-testid={`option-${option.value}`}\n                    id={`option-${indexOption}`}\n                    role=\"option\"\n                    onClick={() => {\n                      if (!option.disabled) {\n                        handleClick(option, group)\n                      }\n                    }}\n                    onKeyDown={event =>\n                      [' ', 'Enter'].includes(event.key)\n                        ? handleClick(option, group)\n                        : null\n                    }\n                    ref={\n                      option.value === defaultSearchValue ||\n                      option.searchText === defaultSearchValue\n                        ? focusedItemRef\n                        : null\n                    }\n                  >\n                    {multiselect ? (\n                      <StyledCheckbox\n                        checked={\n                          selectedData.selectedValues.includes(option.value) &&\n                          !option.disabled\n                        }\n                        disabled={option.disabled}\n                        value={option.value}\n                        tabIndex={-1}\n                      >\n                        <DisplayOption\n                          option={option}\n                          descriptionDirection={descriptionDirection}\n                          optionalInfoPlacement={optionalInfoPlacement}\n                        />\n                      </StyledCheckbox>\n                    ) : (\n                      <DisplayOption\n                        option={option}\n                        descriptionDirection={descriptionDirection}\n                        optionalInfoPlacement={optionalInfoPlacement}\n                      />\n                    )}\n                  </DropdownItem>\n                ))}\n              </Stack>\n            </Stack>\n          ))}\n        </>\n      )}\n      {loadMore ? <LoadMore>{loadMore}</LoadMore> : null}\n    </DropdownContainer>\n  ) : (\n    <DropdownContainer\n      role=\"listbox\"\n      id=\"select-dropdown\"\n      onKeyDown={event => {\n        event.preventDefault()\n        handleKeyDownSelect(event.key)\n      }}\n      gap={0.25}\n      data-grouped={false}\n    >\n      {selectAll && multiselect ? (\n        <Stack id=\"items\" gap={0.25}>\n          <DropdownItem\n            disabled={false}\n            data-selected={selectedData.allSelected}\n            aria-label=\"select-all\"\n            data-testid=\"select-all\"\n            type=\"button\"\n            role=\"option\"\n            onKeyDown={event =>\n              [' ', 'Enter'].includes(event.key) ? selectAllOptions() : null\n            }\n            onClick={selectAllOptions}\n          >\n            <StyledCheckbox\n              checked={selectedData.allSelected}\n              disabled={false}\n              value=\"select-all\"\n              data-testid=\"select-all-checkbox\"\n              tabIndex={-1}\n            >\n              <Stack direction=\"column\">\n                <Text as=\"span\" variant=\"body\" placement=\"left\">\n                  {selectAll.label}\n                </Text>\n                <Text\n                  as=\"span\"\n                  variant=\"bodySmall\"\n                  sentiment=\"neutral\"\n                  placement=\"left\"\n                  prominence=\"weak\"\n                >\n                  {selectAll.description}\n                </Text>\n              </Stack>\n            </StyledCheckbox>\n          </DropdownItem>\n        </Stack>\n      ) : null}\n      <Stack id=\"items\" gap={0.25}>\n        {isLoading ? (\n          <Skeleton variant=\"block\" />\n        ) : (\n          displayedOptions.map((option, index) => (\n            <DropdownItem\n              key={option.value}\n              disabled={!!option.disabled}\n              data-selected={\n                selectedData.selectedValues.includes(option.value) &&\n                !option.disabled\n              }\n              onClick={() => {\n                if (!option.disabled) {\n                  handleClick(option)\n                }\n              }}\n              aria-label={option.value}\n              data-testid={`option-${option.value}`}\n              id={`option-${index}`}\n              role=\"option\"\n              type=\"button\"\n              ref={\n                option.value === defaultSearchValue ||\n                option.searchText === defaultSearchValue\n                  ? focusedItemRef\n                  : null\n              }\n              onKeyDown={event =>\n                [' ', 'Enter'].includes(event.key) ? handleClick(option) : null\n              }\n            >\n              {multiselect ? (\n                <StyledCheckbox\n                  checked={\n                    selectedData.selectedValues.includes(option.value) &&\n                    !option.disabled\n                  }\n                  disabled={option.disabled}\n                  value={option.value}\n                  tabIndex={-1}\n                >\n                  <DisplayOption\n                    option={option}\n                    descriptionDirection={descriptionDirection}\n                    optionalInfoPlacement={optionalInfoPlacement}\n                  />\n                </StyledCheckbox>\n              ) : (\n                <DisplayOption\n                  option={option}\n                  descriptionDirection={descriptionDirection}\n                  optionalInfoPlacement={optionalInfoPlacement}\n                />\n              )}\n            </DropdownItem>\n          ))\n        )}\n        {loadMore ? <LoadMore>{loadMore}</LoadMore> : null}\n      </Stack>\n    </DropdownContainer>\n  )\n}\nexport const Dropdown = ({\n  children,\n  emptyState,\n  descriptionDirection,\n  searchable,\n  placeholder,\n  footer,\n  refSelect,\n  loadMore,\n  optionalInfoPlacement,\n  isLoading,\n}: DropdownProps) => {\n  const {\n    setIsDropdownVisible,\n    isDropdownVisible,\n    onSearch,\n    searchInput,\n    options,\n    displayedOptions,\n  } = useSelectInput()\n  const [searchBarActive, setSearchBarActive] = useState(false)\n  const [defaultSearchValue, setDefaultSearch] = useState<string | null>(null)\n  const ref = useRef<HTMLDivElement>(null)\n  const [search, setSearch] = useState<string>('')\n  const [maxWidth, setWidth] = useState<string | number>()\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"]} */"));
126
+ }) => 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":"AAoHE","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  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  'data-selected': boolean\n  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  &[data-selected='true'] {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n  }\n\n  &[disabled] {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n  }\n\n  &[disabled]:hover, [disabled]: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                disabled={false}\n                data-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              (!Array.isArray(options) && options[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                    disabled={!!option.disabled}\n                    tabIndex={!option.disabled ? 0 : -1}\n                    data-selected={\n                      selectedData.selectedValues.includes(option.value) &&\n                      !option.disabled\n                    }\n                    aria-label={option.value}\n                    data-testid={`option-${option.value}`}\n                    id={`option-${indexOption}`}\n                    role=\"option\"\n                    onClick={() => {\n                      if (!option.disabled) {\n                        handleClick(option, group)\n                      }\n                    }}\n                    onKeyDown={event =>\n                      [' ', 'Enter'].includes(event.key)\n                        ? handleClick(option, group)\n                        : null\n                    }\n                    ref={\n                      option.value === defaultSearchValue ||\n                      option.searchText === defaultSearchValue\n                        ? focusedItemRef\n                        : null\n                    }\n                  >\n                    {multiselect ? (\n                      <StyledCheckbox\n                        checked={\n                          selectedData.selectedValues.includes(option.value) &&\n                          !option.disabled\n                        }\n                        disabled={option.disabled}\n                        value={option.value}\n                        tabIndex={-1}\n                      >\n                        <DisplayOption\n                          option={option}\n                          descriptionDirection={descriptionDirection}\n                          optionalInfoPlacement={optionalInfoPlacement}\n                        />\n                      </StyledCheckbox>\n                    ) : (\n                      <DisplayOption\n                        option={option}\n                        descriptionDirection={descriptionDirection}\n                        optionalInfoPlacement={optionalInfoPlacement}\n                      />\n                    )}\n                  </DropdownItem>\n                ))}\n              </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            disabled={false}\n            data-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              disabled={!!option.disabled}\n              data-selected={\n                selectedData.selectedValues.includes(option.value) &&\n                !option.disabled\n              }\n              onClick={() => {\n                if (!option.disabled) {\n                  handleClick(option)\n                }\n              }}\n              aria-label={option.value}\n              data-testid={`option-${option.value}`}\n              id={`option-${index}`}\n              role=\"option\"\n              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"]} */"));
122
127
  const PopupFooter = /* @__PURE__ */ _styled("div", process.env.NODE_ENV === "production" ? {
123
128
  target: "eeucz6c3"
124
129
  } : {
@@ -134,7 +139,7 @@ const PopupFooter = /* @__PURE__ */ _styled("div", process.env.NODE_ENV === "pro
134
139
  theme
135
140
  }) => theme.space[2], ";box-shadow:", ({
136
141
  theme
137
- }) => theme.shadows.dropdown, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/Dropdown.tsx"],"names":[],"mappings":"AAiJ8B","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/Dropdown.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { useCallback, useEffect, useMemo, useRef, useState } from 'react'\nimport type { Dispatch, ReactNode, RefObject, SetStateAction } from 'react'\nimport { Checkbox } from '../Checkbox'\nimport { Popup } from '../Popup'\nimport { Skeleton } from '../Skeleton'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\nimport { DisplayOption } from './DropdownOption'\nimport { SearchBarDropdown } from './SearchBarDropdown'\nimport { useSelectInput } from './SelectInputProvider'\nimport type { DataType, OptionType } from './types'\n\nexport type DropdownProps = {\n  children: ReactNode\n  emptyState: ReactNode\n  descriptionDirection: 'row' | 'column'\n  searchable: boolean\n  placeholder: string\n  footer?: ReactNode\n  refSelect: RefObject<HTMLDivElement>\n  loadMore?: ReactNode\n  optionalInfoPlacement: 'left' | 'right'\n  isLoading?: boolean\n}\n\nexport type CreateDropdownProps = {\n  isEmpty: boolean\n  emptyState: ReactNode\n  descriptionDirection: 'row' | 'column'\n  loadMore?: ReactNode\n  optionalInfoPlacement: 'left' | 'right'\n  defaultSearchValue: string | null\n  isLoading?: boolean\n}\n\nconst NON_SEARCHABLE_KEYS = [\n  'Tab',\n  ' ',\n  'Enter',\n  'CapsLock',\n  'Shift',\n  'ArrowDown',\n  'ArrowUp',\n  'ArrowLeft',\n  'ArrowRight',\n  'Escape',\n]\n\nconst StyledPopup = styled(Popup)`\n  width: 100%;\n  background-color: ${({ theme }) => theme.colors.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`\n\nconst DropdownContainer = styled(Stack)<{ 'data-grouped': boolean }>`\n  max-height: 256px;\n  overflow-y: scroll;\n  padding: ${({ theme }) => theme.space[0]};\n  padding-bottom: ${({ theme }) => theme.space[0.5]};\n  padding-top: ${({ theme }) => theme.space[0.5]};\n\n  &[data-grouped=\"true\"] {\n    padding-top: ${({ theme }) => theme.space[0]};\n  }\n`\nconst DropdownGroup = styled.button<{ 'data-selectgroup': boolean }>`\n  display: flex;\n  width: 100%;\n  justify-content: left;\n  align-items: center;\n  border: none;\n  background-color: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n  position: sticky;\n  top: 0px;\n  padding-right: ${({ theme }) => theme.space[2]};\n  padding-left: ${({ theme }) => theme.space[2]};\n  height: ${({ theme }) => theme.space[4]};\n  text-align: left;\n  margin-bottom: ${({ theme }) => theme.space['0.25']};\n\n  &:focus {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n    outline: none;\n  }\n\n  &[data-selectgroup='true'] {\n    padding-left: ${({ theme }) => theme.space[2]};\n    border-left: ${({ theme }) => theme.space[0.5]} solid ${({ theme }) =>\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.button<{\n  'data-selected': boolean\n  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  &[data-selected='true'] {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n  }\n\n  &[disabled] {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n\n  }\n\n  &[disabled]:hover, [disabled]:focus {\n    background-color: ${({ theme }) =>\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 > button[role=\"option\"]:not([disabled])',\n  )\n  const activeItem = document.activeElement\n  if (options) {\n    for (let i = 0; i < options?.length; i += 1) {\n      const listLength = options.length\n      if (activeItem === options[i] && activeItem !== options[listLength - 1]) {\n        ;(options[i + 1] as HTMLElement).focus()\n      }\n    }\n  }\n}\nconst moveFocusUp = () => {\n  const options = document.querySelectorAll(\n    '#items > button[role=\"option\"]:not([disabled])',\n  )\n  const activeItem = document.activeElement\n\n  if (options) {\n    for (let i = 0; i < options.length; i += 1) {\n      if (activeItem === options[i] && activeItem !== options[0]) {\n        ;(options[i - 1] as HTMLElement).focus()\n      }\n    }\n  }\n}\nconst handleKeyDownSelect = (key: string) => {\n  if (key === 'ArrowDown') {\n    moveFocusDown()\n  }\n  if (key === 'ArrowUp') {\n    moveFocusUp()\n  }\n}\nconst handleClickOutside = (\n  event: MouseEvent,\n  ref: RefObject<HTMLDivElement>,\n  setIsDropdownVisibile: Dispatch<SetStateAction<boolean>>,\n  refSelect: RefObject<HTMLDivElement>,\n  onSearch: Dispatch<SetStateAction<DataType>>,\n  options: DataType,\n) => {\n  if (\n    ref.current &&\n    !ref.current.contains(event.target as Node) &&\n    !refSelect.current?.contains(event.target as Node)\n  ) {\n    setIsDropdownVisibile(false) // hide dropdown when clicking outside of the dropdown\n    onSearch(options) // reset displayed options to default when dropdown is hidden\n  }\n}\n\nconst handleKeyDown = (\n  event: KeyboardEvent,\n  ref: RefObject<HTMLDivElement>,\n  options: DataType,\n  searchBarActive: boolean,\n  setSearch: Dispatch<SetStateAction<string>>,\n  setDefaultSearch: Dispatch<SetStateAction<string | null>>,\n  search: string,\n) => {\n  // Deals with default search\n  if (\n    ref.current &&\n    !searchBarActive &&\n    !NON_SEARCHABLE_KEYS.includes(event.key) &&\n    document.activeElement?.ariaLabel !== 'search-bar'\n  ) {\n    const currentSearch = search + event.key\n    setSearch(currentSearch)\n    ref.current.focus()\n    if (!Array.isArray(options)) {\n      const closestOptions = { ...options }\n      Object.keys(closestOptions).map((group: string) => {\n        closestOptions[group] = closestOptions[group].filter(option =>\n          option.searchText\n            ? option.searchText.toLocaleLowerCase().startsWith(currentSearch)\n            : option.value.toLocaleLowerCase().startsWith(currentSearch),\n        )\n\n        return null\n      })\n      const closestOption = closestOptions[Object.keys(closestOptions)[0]][0]\n      if (closestOption) {\n        setDefaultSearch(closestOption.searchText ?? closestOption.value)\n      } else {\n        setDefaultSearch(null)\n      }\n    } else {\n      const closestOption = [...options].filter(option =>\n        option.searchText\n          ? option.searchText.toLocaleLowerCase().startsWith(currentSearch)\n          : option.value.toLocaleLowerCase().startsWith(currentSearch),\n      )[0]\n      if (closestOption) {\n        setDefaultSearch(closestOption.searchText ?? closestOption.value)\n      } else {\n        setDefaultSearch(null)\n      }\n    }\n  }\n}\nconst CreateDropdown = ({\n  isEmpty,\n  emptyState,\n  descriptionDirection,\n  loadMore,\n  optionalInfoPlacement,\n  defaultSearchValue,\n  isLoading,\n}: CreateDropdownProps) => {\n  const {\n    setIsDropdownVisible,\n    onChange,\n    options,\n    multiselect,\n    selectAll,\n    selectAllGroup,\n    displayedOptions,\n    setSelectedData,\n    selectedData,\n  } = useSelectInput()\n  const focusedItemRef = useRef<HTMLButtonElement>(null)\n\n  useEffect(() => {\n    if (defaultSearchValue && focusedItemRef?.current) {\n      focusedItemRef.current.focus()\n    }\n  }, [defaultSearchValue])\n\n  if (isEmpty) {\n    return (\n      <EmptyState gap={2} alignItems=\"center\">\n        {emptyState ?? (\n          <Text variant=\"bodyStrong\" as=\"p\">\n            No options\n          </Text>\n        )}\n      </EmptyState>\n    )\n  }\n\n  const handleClick = (clickedOption: OptionType, group?: string) => {\n    setSelectedData({ type: 'selectOption', clickedOption, group })\n    if (multiselect) {\n      if (selectedData.selectedValues.includes(clickedOption.value)) {\n        onChange?.(\n          selectedData.selectedValues.filter(\n            val => val !== clickedOption.value,\n          ),\n        )\n      } else {\n        onChange?.([...selectedData.selectedValues, clickedOption.value])\n      }\n    } else {\n      onChange?.(clickedOption.value)\n    }\n    setIsDropdownVisible(multiselect) // hide the dropdown on click when single select only\n  }\n\n  const selectAllOptions = () => {\n    if (multiselect) {\n      setSelectedData({ type: 'selectAll' })\n      if (selectedData.allSelected && onChange) {\n        onChange([])\n      } else {\n        const allValues: OptionType[] = []\n        if (!Array.isArray(options)) {\n          Object.keys(options).map((group: string) =>\n            options[group].map(option => {\n              if (!option.disabled) {\n                allValues.push(option)\n              }\n\n              return null\n            }),\n          )\n        } else {\n          options.map(option => allValues.push(option))\n        }\n        onChange?.(allValues.map(value => value.value))\n      }\n    }\n  }\n\n  const handleSelectGroup = (group: string) => {\n    if (multiselect) {\n      setSelectedData({ type: 'selectGroup', selectedGroup: group })\n      if (!Array.isArray(options)) {\n        if (selectedData.selectedGroups.includes(group)) {\n          const newSelectedValues = [...selectedData.selectedValues].filter(\n            selectedValue =>\n              !options[group].find(option => option.value === selectedValue),\n          )\n          onChange?.(newSelectedValues)\n        } else {\n          const newSelectedValues = [...selectedData.selectedValues]\n\n          options[group].map(option =>\n            newSelectedValues.includes(option.value) || option.disabled\n              ? null\n              : newSelectedValues.push(option.value),\n          )\n          onChange?.(newSelectedValues)\n        }\n      }\n    }\n  }\n\n  return !Array.isArray(displayedOptions) ? (\n    <DropdownContainer\n      role=\"listbox\"\n      id=\"select-dropdown\"\n      onKeyDown={event => {\n        event.preventDefault()\n        handleKeyDownSelect(event.key)\n      }}\n      data-grouped\n    >\n      {isLoading ? (\n        <Skeleton variant=\"block\" />\n      ) : (\n        <>\n          {selectAll && multiselect ? (\n            <Stack id=\"items\">\n              <DropdownItem\n                disabled={false}\n                data-selected={selectedData.allSelected}\n                aria-label=\"select-all\"\n                data-testid=\"select-all\"\n                id=\"select-all\"\n                type=\"button\"\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              (!Array.isArray(options) && options[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                    disabled={!!option.disabled}\n                    type=\"button\"\n                    data-selected={\n                      selectedData.selectedValues.includes(option.value) &&\n                      !option.disabled\n                    }\n                    aria-label={option.value}\n                    data-testid={`option-${option.value}`}\n                    id={`option-${indexOption}`}\n                    role=\"option\"\n                    onClick={() => {\n                      if (!option.disabled) {\n                        handleClick(option, group)\n                      }\n                    }}\n                    onKeyDown={event =>\n                      [' ', 'Enter'].includes(event.key)\n                        ? handleClick(option, group)\n                        : null\n                    }\n                    ref={\n                      option.value === defaultSearchValue ||\n                      option.searchText === defaultSearchValue\n                        ? focusedItemRef\n                        : null\n                    }\n                  >\n                    {multiselect ? (\n                      <StyledCheckbox\n                        checked={\n                          selectedData.selectedValues.includes(option.value) &&\n                          !option.disabled\n                        }\n                        disabled={option.disabled}\n                        value={option.value}\n                        tabIndex={-1}\n                      >\n                        <DisplayOption\n                          option={option}\n                          descriptionDirection={descriptionDirection}\n                          optionalInfoPlacement={optionalInfoPlacement}\n                        />\n                      </StyledCheckbox>\n                    ) : (\n                      <DisplayOption\n                        option={option}\n                        descriptionDirection={descriptionDirection}\n                        optionalInfoPlacement={optionalInfoPlacement}\n                      />\n                    )}\n                  </DropdownItem>\n                ))}\n              </Stack>\n            </Stack>\n          ))}\n        </>\n      )}\n      {loadMore ? <LoadMore>{loadMore}</LoadMore> : null}\n    </DropdownContainer>\n  ) : (\n    <DropdownContainer\n      role=\"listbox\"\n      id=\"select-dropdown\"\n      onKeyDown={event => {\n        event.preventDefault()\n        handleKeyDownSelect(event.key)\n      }}\n      gap={0.25}\n      data-grouped={false}\n    >\n      {selectAll && multiselect ? (\n        <Stack id=\"items\" gap={0.25}>\n          <DropdownItem\n            disabled={false}\n            data-selected={selectedData.allSelected}\n            aria-label=\"select-all\"\n            data-testid=\"select-all\"\n            type=\"button\"\n            role=\"option\"\n            onKeyDown={event =>\n              [' ', 'Enter'].includes(event.key) ? selectAllOptions() : null\n            }\n            onClick={selectAllOptions}\n          >\n            <StyledCheckbox\n              checked={selectedData.allSelected}\n              disabled={false}\n              value=\"select-all\"\n              data-testid=\"select-all-checkbox\"\n              tabIndex={-1}\n            >\n              <Stack direction=\"column\">\n                <Text as=\"span\" variant=\"body\" placement=\"left\">\n                  {selectAll.label}\n                </Text>\n                <Text\n                  as=\"span\"\n                  variant=\"bodySmall\"\n                  sentiment=\"neutral\"\n                  placement=\"left\"\n                  prominence=\"weak\"\n                >\n                  {selectAll.description}\n                </Text>\n              </Stack>\n            </StyledCheckbox>\n          </DropdownItem>\n        </Stack>\n      ) : null}\n      <Stack id=\"items\" gap={0.25}>\n        {isLoading ? (\n          <Skeleton variant=\"block\" />\n        ) : (\n          displayedOptions.map((option, index) => (\n            <DropdownItem\n              key={option.value}\n              disabled={!!option.disabled}\n              data-selected={\n                selectedData.selectedValues.includes(option.value) &&\n                !option.disabled\n              }\n              onClick={() => {\n                if (!option.disabled) {\n                  handleClick(option)\n                }\n              }}\n              aria-label={option.value}\n              data-testid={`option-${option.value}`}\n              id={`option-${index}`}\n              role=\"option\"\n              type=\"button\"\n              ref={\n                option.value === defaultSearchValue ||\n                option.searchText === defaultSearchValue\n                  ? focusedItemRef\n                  : null\n              }\n              onKeyDown={event =>\n                [' ', 'Enter'].includes(event.key) ? handleClick(option) : null\n              }\n            >\n              {multiselect ? (\n                <StyledCheckbox\n                  checked={\n                    selectedData.selectedValues.includes(option.value) &&\n                    !option.disabled\n                  }\n                  disabled={option.disabled}\n                  value={option.value}\n                  tabIndex={-1}\n                >\n                  <DisplayOption\n                    option={option}\n                    descriptionDirection={descriptionDirection}\n                    optionalInfoPlacement={optionalInfoPlacement}\n                  />\n                </StyledCheckbox>\n              ) : (\n                <DisplayOption\n                  option={option}\n                  descriptionDirection={descriptionDirection}\n                  optionalInfoPlacement={optionalInfoPlacement}\n                />\n              )}\n            </DropdownItem>\n          ))\n        )}\n        {loadMore ? <LoadMore>{loadMore}</LoadMore> : null}\n      </Stack>\n    </DropdownContainer>\n  )\n}\nexport const Dropdown = ({\n  children,\n  emptyState,\n  descriptionDirection,\n  searchable,\n  placeholder,\n  footer,\n  refSelect,\n  loadMore,\n  optionalInfoPlacement,\n  isLoading,\n}: DropdownProps) => {\n  const {\n    setIsDropdownVisible,\n    isDropdownVisible,\n    onSearch,\n    searchInput,\n    options,\n    displayedOptions,\n  } = useSelectInput()\n  const [searchBarActive, setSearchBarActive] = useState(false)\n  const [defaultSearchValue, setDefaultSearch] = useState<string | null>(null)\n  const ref = useRef<HTMLDivElement>(null)\n  const [search, setSearch] = useState<string>('')\n  const [maxWidth, setWidth] = useState<string | number>()\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"]} */"));
142
+ }) => 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":"AA2J8B","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  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  'data-selected': boolean\n  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  &[data-selected='true'] {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n  }\n\n  &[disabled] {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n  }\n\n  &[disabled]:hover, [disabled]: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                disabled={false}\n                data-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              (!Array.isArray(options) && options[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                    disabled={!!option.disabled}\n                    tabIndex={!option.disabled ? 0 : -1}\n                    data-selected={\n                      selectedData.selectedValues.includes(option.value) &&\n                      !option.disabled\n                    }\n                    aria-label={option.value}\n                    data-testid={`option-${option.value}`}\n                    id={`option-${indexOption}`}\n                    role=\"option\"\n                    onClick={() => {\n                      if (!option.disabled) {\n                        handleClick(option, group)\n                      }\n                    }}\n                    onKeyDown={event =>\n                      [' ', 'Enter'].includes(event.key)\n                        ? handleClick(option, group)\n                        : null\n                    }\n                    ref={\n                      option.value === defaultSearchValue ||\n                      option.searchText === defaultSearchValue\n                        ? focusedItemRef\n                        : null\n                    }\n                  >\n                    {multiselect ? (\n                      <StyledCheckbox\n                        checked={\n                          selectedData.selectedValues.includes(option.value) &&\n                          !option.disabled\n                        }\n                        disabled={option.disabled}\n                        value={option.value}\n                        tabIndex={-1}\n                      >\n                        <DisplayOption\n                          option={option}\n                          descriptionDirection={descriptionDirection}\n                          optionalInfoPlacement={optionalInfoPlacement}\n                        />\n                      </StyledCheckbox>\n                    ) : (\n                      <DisplayOption\n                        option={option}\n                        descriptionDirection={descriptionDirection}\n                        optionalInfoPlacement={optionalInfoPlacement}\n                      />\n                    )}\n                  </DropdownItem>\n                ))}\n              </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            disabled={false}\n            data-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              disabled={!!option.disabled}\n              data-selected={\n                selectedData.selectedValues.includes(option.value) &&\n                !option.disabled\n              }\n              onClick={() => {\n                if (!option.disabled) {\n                  handleClick(option)\n                }\n              }}\n              aria-label={option.value}\n              data-testid={`option-${option.value}`}\n              id={`option-${index}`}\n              role=\"option\"\n              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"]} */"));
138
143
  const StyledCheckbox = /* @__PURE__ */ _styled(Checkbox, process.env.NODE_ENV === "production" ? {
139
144
  target: "eeucz6c2"
140
145
  } : {
@@ -146,7 +151,7 @@ const StyledCheckbox = /* @__PURE__ */ _styled(Checkbox, process.env.NODE_ENV ==
146
151
  } : {
147
152
  name: "1l9xw77",
148
153
  styles: "width:100%;position:static;text-align:left;align-items:center;pointer-events:none",
149
- map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/Dropdown.tsx"],"names":[],"mappings":"AAuJuC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/Dropdown.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { useCallback, useEffect, useMemo, useRef, useState } from 'react'\nimport type { Dispatch, ReactNode, RefObject, SetStateAction } from 'react'\nimport { Checkbox } from '../Checkbox'\nimport { Popup } from '../Popup'\nimport { Skeleton } from '../Skeleton'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\nimport { DisplayOption } from './DropdownOption'\nimport { SearchBarDropdown } from './SearchBarDropdown'\nimport { useSelectInput } from './SelectInputProvider'\nimport type { DataType, OptionType } from './types'\n\nexport type DropdownProps = {\n  children: ReactNode\n  emptyState: ReactNode\n  descriptionDirection: 'row' | 'column'\n  searchable: boolean\n  placeholder: string\n  footer?: ReactNode\n  refSelect: RefObject<HTMLDivElement>\n  loadMore?: ReactNode\n  optionalInfoPlacement: 'left' | 'right'\n  isLoading?: boolean\n}\n\nexport type CreateDropdownProps = {\n  isEmpty: boolean\n  emptyState: ReactNode\n  descriptionDirection: 'row' | 'column'\n  loadMore?: ReactNode\n  optionalInfoPlacement: 'left' | 'right'\n  defaultSearchValue: string | null\n  isLoading?: boolean\n}\n\nconst NON_SEARCHABLE_KEYS = [\n  'Tab',\n  ' ',\n  'Enter',\n  'CapsLock',\n  'Shift',\n  'ArrowDown',\n  'ArrowUp',\n  'ArrowLeft',\n  'ArrowRight',\n  'Escape',\n]\n\nconst StyledPopup = styled(Popup)`\n  width: 100%;\n  background-color: ${({ theme }) => theme.colors.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`\n\nconst DropdownContainer = styled(Stack)<{ 'data-grouped': boolean }>`\n  max-height: 256px;\n  overflow-y: scroll;\n  padding: ${({ theme }) => theme.space[0]};\n  padding-bottom: ${({ theme }) => theme.space[0.5]};\n  padding-top: ${({ theme }) => theme.space[0.5]};\n\n  &[data-grouped=\"true\"] {\n    padding-top: ${({ theme }) => theme.space[0]};\n  }\n`\nconst DropdownGroup = styled.button<{ 'data-selectgroup': boolean }>`\n  display: flex;\n  width: 100%;\n  justify-content: left;\n  align-items: center;\n  border: none;\n  background-color: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n  position: sticky;\n  top: 0px;\n  padding-right: ${({ theme }) => theme.space[2]};\n  padding-left: ${({ theme }) => theme.space[2]};\n  height: ${({ theme }) => theme.space[4]};\n  text-align: left;\n  margin-bottom: ${({ theme }) => theme.space['0.25']};\n\n  &:focus {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n    outline: none;\n  }\n\n  &[data-selectgroup='true'] {\n    padding-left: ${({ theme }) => theme.space[2]};\n    border-left: ${({ theme }) => theme.space[0.5]} solid ${({ theme }) =>\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.button<{\n  'data-selected': boolean\n  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  &[data-selected='true'] {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n  }\n\n  &[disabled] {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n\n  }\n\n  &[disabled]:hover, [disabled]:focus {\n    background-color: ${({ theme }) =>\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 > button[role=\"option\"]:not([disabled])',\n  )\n  const activeItem = document.activeElement\n  if (options) {\n    for (let i = 0; i < options?.length; i += 1) {\n      const listLength = options.length\n      if (activeItem === options[i] && activeItem !== options[listLength - 1]) {\n        ;(options[i + 1] as HTMLElement).focus()\n      }\n    }\n  }\n}\nconst moveFocusUp = () => {\n  const options = document.querySelectorAll(\n    '#items > button[role=\"option\"]:not([disabled])',\n  )\n  const activeItem = document.activeElement\n\n  if (options) {\n    for (let i = 0; i < options.length; i += 1) {\n      if (activeItem === options[i] && activeItem !== options[0]) {\n        ;(options[i - 1] as HTMLElement).focus()\n      }\n    }\n  }\n}\nconst handleKeyDownSelect = (key: string) => {\n  if (key === 'ArrowDown') {\n    moveFocusDown()\n  }\n  if (key === 'ArrowUp') {\n    moveFocusUp()\n  }\n}\nconst handleClickOutside = (\n  event: MouseEvent,\n  ref: RefObject<HTMLDivElement>,\n  setIsDropdownVisibile: Dispatch<SetStateAction<boolean>>,\n  refSelect: RefObject<HTMLDivElement>,\n  onSearch: Dispatch<SetStateAction<DataType>>,\n  options: DataType,\n) => {\n  if (\n    ref.current &&\n    !ref.current.contains(event.target as Node) &&\n    !refSelect.current?.contains(event.target as Node)\n  ) {\n    setIsDropdownVisibile(false) // hide dropdown when clicking outside of the dropdown\n    onSearch(options) // reset displayed options to default when dropdown is hidden\n  }\n}\n\nconst handleKeyDown = (\n  event: KeyboardEvent,\n  ref: RefObject<HTMLDivElement>,\n  options: DataType,\n  searchBarActive: boolean,\n  setSearch: Dispatch<SetStateAction<string>>,\n  setDefaultSearch: Dispatch<SetStateAction<string | null>>,\n  search: string,\n) => {\n  // Deals with default search\n  if (\n    ref.current &&\n    !searchBarActive &&\n    !NON_SEARCHABLE_KEYS.includes(event.key) &&\n    document.activeElement?.ariaLabel !== 'search-bar'\n  ) {\n    const currentSearch = search + event.key\n    setSearch(currentSearch)\n    ref.current.focus()\n    if (!Array.isArray(options)) {\n      const closestOptions = { ...options }\n      Object.keys(closestOptions).map((group: string) => {\n        closestOptions[group] = closestOptions[group].filter(option =>\n          option.searchText\n            ? option.searchText.toLocaleLowerCase().startsWith(currentSearch)\n            : option.value.toLocaleLowerCase().startsWith(currentSearch),\n        )\n\n        return null\n      })\n      const closestOption = closestOptions[Object.keys(closestOptions)[0]][0]\n      if (closestOption) {\n        setDefaultSearch(closestOption.searchText ?? closestOption.value)\n      } else {\n        setDefaultSearch(null)\n      }\n    } else {\n      const closestOption = [...options].filter(option =>\n        option.searchText\n          ? option.searchText.toLocaleLowerCase().startsWith(currentSearch)\n          : option.value.toLocaleLowerCase().startsWith(currentSearch),\n      )[0]\n      if (closestOption) {\n        setDefaultSearch(closestOption.searchText ?? closestOption.value)\n      } else {\n        setDefaultSearch(null)\n      }\n    }\n  }\n}\nconst CreateDropdown = ({\n  isEmpty,\n  emptyState,\n  descriptionDirection,\n  loadMore,\n  optionalInfoPlacement,\n  defaultSearchValue,\n  isLoading,\n}: CreateDropdownProps) => {\n  const {\n    setIsDropdownVisible,\n    onChange,\n    options,\n    multiselect,\n    selectAll,\n    selectAllGroup,\n    displayedOptions,\n    setSelectedData,\n    selectedData,\n  } = useSelectInput()\n  const focusedItemRef = useRef<HTMLButtonElement>(null)\n\n  useEffect(() => {\n    if (defaultSearchValue && focusedItemRef?.current) {\n      focusedItemRef.current.focus()\n    }\n  }, [defaultSearchValue])\n\n  if (isEmpty) {\n    return (\n      <EmptyState gap={2} alignItems=\"center\">\n        {emptyState ?? (\n          <Text variant=\"bodyStrong\" as=\"p\">\n            No options\n          </Text>\n        )}\n      </EmptyState>\n    )\n  }\n\n  const handleClick = (clickedOption: OptionType, group?: string) => {\n    setSelectedData({ type: 'selectOption', clickedOption, group })\n    if (multiselect) {\n      if (selectedData.selectedValues.includes(clickedOption.value)) {\n        onChange?.(\n          selectedData.selectedValues.filter(\n            val => val !== clickedOption.value,\n          ),\n        )\n      } else {\n        onChange?.([...selectedData.selectedValues, clickedOption.value])\n      }\n    } else {\n      onChange?.(clickedOption.value)\n    }\n    setIsDropdownVisible(multiselect) // hide the dropdown on click when single select only\n  }\n\n  const selectAllOptions = () => {\n    if (multiselect) {\n      setSelectedData({ type: 'selectAll' })\n      if (selectedData.allSelected && onChange) {\n        onChange([])\n      } else {\n        const allValues: OptionType[] = []\n        if (!Array.isArray(options)) {\n          Object.keys(options).map((group: string) =>\n            options[group].map(option => {\n              if (!option.disabled) {\n                allValues.push(option)\n              }\n\n              return null\n            }),\n          )\n        } else {\n          options.map(option => allValues.push(option))\n        }\n        onChange?.(allValues.map(value => value.value))\n      }\n    }\n  }\n\n  const handleSelectGroup = (group: string) => {\n    if (multiselect) {\n      setSelectedData({ type: 'selectGroup', selectedGroup: group })\n      if (!Array.isArray(options)) {\n        if (selectedData.selectedGroups.includes(group)) {\n          const newSelectedValues = [...selectedData.selectedValues].filter(\n            selectedValue =>\n              !options[group].find(option => option.value === selectedValue),\n          )\n          onChange?.(newSelectedValues)\n        } else {\n          const newSelectedValues = [...selectedData.selectedValues]\n\n          options[group].map(option =>\n            newSelectedValues.includes(option.value) || option.disabled\n              ? null\n              : newSelectedValues.push(option.value),\n          )\n          onChange?.(newSelectedValues)\n        }\n      }\n    }\n  }\n\n  return !Array.isArray(displayedOptions) ? (\n    <DropdownContainer\n      role=\"listbox\"\n      id=\"select-dropdown\"\n      onKeyDown={event => {\n        event.preventDefault()\n        handleKeyDownSelect(event.key)\n      }}\n      data-grouped\n    >\n      {isLoading ? (\n        <Skeleton variant=\"block\" />\n      ) : (\n        <>\n          {selectAll && multiselect ? (\n            <Stack id=\"items\">\n              <DropdownItem\n                disabled={false}\n                data-selected={selectedData.allSelected}\n                aria-label=\"select-all\"\n                data-testid=\"select-all\"\n                id=\"select-all\"\n                type=\"button\"\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              (!Array.isArray(options) && options[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                    disabled={!!option.disabled}\n                    type=\"button\"\n                    data-selected={\n                      selectedData.selectedValues.includes(option.value) &&\n                      !option.disabled\n                    }\n                    aria-label={option.value}\n                    data-testid={`option-${option.value}`}\n                    id={`option-${indexOption}`}\n                    role=\"option\"\n                    onClick={() => {\n                      if (!option.disabled) {\n                        handleClick(option, group)\n                      }\n                    }}\n                    onKeyDown={event =>\n                      [' ', 'Enter'].includes(event.key)\n                        ? handleClick(option, group)\n                        : null\n                    }\n                    ref={\n                      option.value === defaultSearchValue ||\n                      option.searchText === defaultSearchValue\n                        ? focusedItemRef\n                        : null\n                    }\n                  >\n                    {multiselect ? (\n                      <StyledCheckbox\n                        checked={\n                          selectedData.selectedValues.includes(option.value) &&\n                          !option.disabled\n                        }\n                        disabled={option.disabled}\n                        value={option.value}\n                        tabIndex={-1}\n                      >\n                        <DisplayOption\n                          option={option}\n                          descriptionDirection={descriptionDirection}\n                          optionalInfoPlacement={optionalInfoPlacement}\n                        />\n                      </StyledCheckbox>\n                    ) : (\n                      <DisplayOption\n                        option={option}\n                        descriptionDirection={descriptionDirection}\n                        optionalInfoPlacement={optionalInfoPlacement}\n                      />\n                    )}\n                  </DropdownItem>\n                ))}\n              </Stack>\n            </Stack>\n          ))}\n        </>\n      )}\n      {loadMore ? <LoadMore>{loadMore}</LoadMore> : null}\n    </DropdownContainer>\n  ) : (\n    <DropdownContainer\n      role=\"listbox\"\n      id=\"select-dropdown\"\n      onKeyDown={event => {\n        event.preventDefault()\n        handleKeyDownSelect(event.key)\n      }}\n      gap={0.25}\n      data-grouped={false}\n    >\n      {selectAll && multiselect ? (\n        <Stack id=\"items\" gap={0.25}>\n          <DropdownItem\n            disabled={false}\n            data-selected={selectedData.allSelected}\n            aria-label=\"select-all\"\n            data-testid=\"select-all\"\n            type=\"button\"\n            role=\"option\"\n            onKeyDown={event =>\n              [' ', 'Enter'].includes(event.key) ? selectAllOptions() : null\n            }\n            onClick={selectAllOptions}\n          >\n            <StyledCheckbox\n              checked={selectedData.allSelected}\n              disabled={false}\n              value=\"select-all\"\n              data-testid=\"select-all-checkbox\"\n              tabIndex={-1}\n            >\n              <Stack direction=\"column\">\n                <Text as=\"span\" variant=\"body\" placement=\"left\">\n                  {selectAll.label}\n                </Text>\n                <Text\n                  as=\"span\"\n                  variant=\"bodySmall\"\n                  sentiment=\"neutral\"\n                  placement=\"left\"\n                  prominence=\"weak\"\n                >\n                  {selectAll.description}\n                </Text>\n              </Stack>\n            </StyledCheckbox>\n          </DropdownItem>\n        </Stack>\n      ) : null}\n      <Stack id=\"items\" gap={0.25}>\n        {isLoading ? (\n          <Skeleton variant=\"block\" />\n        ) : (\n          displayedOptions.map((option, index) => (\n            <DropdownItem\n              key={option.value}\n              disabled={!!option.disabled}\n              data-selected={\n                selectedData.selectedValues.includes(option.value) &&\n                !option.disabled\n              }\n              onClick={() => {\n                if (!option.disabled) {\n                  handleClick(option)\n                }\n              }}\n              aria-label={option.value}\n              data-testid={`option-${option.value}`}\n              id={`option-${index}`}\n              role=\"option\"\n              type=\"button\"\n              ref={\n                option.value === defaultSearchValue ||\n                option.searchText === defaultSearchValue\n                  ? focusedItemRef\n                  : null\n              }\n              onKeyDown={event =>\n                [' ', 'Enter'].includes(event.key) ? handleClick(option) : null\n              }\n            >\n              {multiselect ? (\n                <StyledCheckbox\n                  checked={\n                    selectedData.selectedValues.includes(option.value) &&\n                    !option.disabled\n                  }\n                  disabled={option.disabled}\n                  value={option.value}\n                  tabIndex={-1}\n                >\n                  <DisplayOption\n                    option={option}\n                    descriptionDirection={descriptionDirection}\n                    optionalInfoPlacement={optionalInfoPlacement}\n                  />\n                </StyledCheckbox>\n              ) : (\n                <DisplayOption\n                  option={option}\n                  descriptionDirection={descriptionDirection}\n                  optionalInfoPlacement={optionalInfoPlacement}\n                />\n              )}\n            </DropdownItem>\n          ))\n        )}\n        {loadMore ? <LoadMore>{loadMore}</LoadMore> : null}\n      </Stack>\n    </DropdownContainer>\n  )\n}\nexport const Dropdown = ({\n  children,\n  emptyState,\n  descriptionDirection,\n  searchable,\n  placeholder,\n  footer,\n  refSelect,\n  loadMore,\n  optionalInfoPlacement,\n  isLoading,\n}: DropdownProps) => {\n  const {\n    setIsDropdownVisible,\n    isDropdownVisible,\n    onSearch,\n    searchInput,\n    options,\n    displayedOptions,\n  } = useSelectInput()\n  const [searchBarActive, setSearchBarActive] = useState(false)\n  const [defaultSearchValue, setDefaultSearch] = useState<string | null>(null)\n  const ref = useRef<HTMLDivElement>(null)\n  const [search, setSearch] = useState<string>('')\n  const [maxWidth, setWidth] = useState<string | number>()\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"]} */",
154
+ 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":"AAiKuC","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  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  'data-selected': boolean\n  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  &[data-selected='true'] {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n  }\n\n  &[disabled] {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n  }\n\n  &[disabled]:hover, [disabled]: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                disabled={false}\n                data-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              (!Array.isArray(options) && options[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                    disabled={!!option.disabled}\n                    tabIndex={!option.disabled ? 0 : -1}\n                    data-selected={\n                      selectedData.selectedValues.includes(option.value) &&\n                      !option.disabled\n                    }\n                    aria-label={option.value}\n                    data-testid={`option-${option.value}`}\n                    id={`option-${indexOption}`}\n                    role=\"option\"\n                    onClick={() => {\n                      if (!option.disabled) {\n                        handleClick(option, group)\n                      }\n                    }}\n                    onKeyDown={event =>\n                      [' ', 'Enter'].includes(event.key)\n                        ? handleClick(option, group)\n                        : null\n                    }\n                    ref={\n                      option.value === defaultSearchValue ||\n                      option.searchText === defaultSearchValue\n                        ? focusedItemRef\n                        : null\n                    }\n                  >\n                    {multiselect ? (\n                      <StyledCheckbox\n                        checked={\n                          selectedData.selectedValues.includes(option.value) &&\n                          !option.disabled\n                        }\n                        disabled={option.disabled}\n                        value={option.value}\n                        tabIndex={-1}\n                      >\n                        <DisplayOption\n                          option={option}\n                          descriptionDirection={descriptionDirection}\n                          optionalInfoPlacement={optionalInfoPlacement}\n                        />\n                      </StyledCheckbox>\n                    ) : (\n                      <DisplayOption\n                        option={option}\n                        descriptionDirection={descriptionDirection}\n                        optionalInfoPlacement={optionalInfoPlacement}\n                      />\n                    )}\n                  </DropdownItem>\n                ))}\n              </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            disabled={false}\n            data-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              disabled={!!option.disabled}\n              data-selected={\n                selectedData.selectedValues.includes(option.value) &&\n                !option.disabled\n              }\n              onClick={() => {\n                if (!option.disabled) {\n                  handleClick(option)\n                }\n              }}\n              aria-label={option.value}\n              data-testid={`option-${option.value}`}\n              id={`option-${index}`}\n              role=\"option\"\n              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"]} */",
150
155
  toString: _EMOTION_STRINGIFIED_CSS_ERROR__
151
156
  });
152
157
  const EmptyState = /* @__PURE__ */ _styled(Stack, process.env.NODE_ENV === "production" ? {
@@ -156,7 +161,7 @@ const EmptyState = /* @__PURE__ */ _styled(Stack, process.env.NODE_ENV === "prod
156
161
  label: "EmptyState"
157
162
  })("padding:", ({
158
163
  theme
159
- }) => theme.space[2], ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/Dropdown.tsx"],"names":[],"mappings":"AA+JgC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/Dropdown.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { useCallback, useEffect, useMemo, useRef, useState } from 'react'\nimport type { Dispatch, ReactNode, RefObject, SetStateAction } from 'react'\nimport { Checkbox } from '../Checkbox'\nimport { Popup } from '../Popup'\nimport { Skeleton } from '../Skeleton'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\nimport { DisplayOption } from './DropdownOption'\nimport { SearchBarDropdown } from './SearchBarDropdown'\nimport { useSelectInput } from './SelectInputProvider'\nimport type { DataType, OptionType } from './types'\n\nexport type DropdownProps = {\n  children: ReactNode\n  emptyState: ReactNode\n  descriptionDirection: 'row' | 'column'\n  searchable: boolean\n  placeholder: string\n  footer?: ReactNode\n  refSelect: RefObject<HTMLDivElement>\n  loadMore?: ReactNode\n  optionalInfoPlacement: 'left' | 'right'\n  isLoading?: boolean\n}\n\nexport type CreateDropdownProps = {\n  isEmpty: boolean\n  emptyState: ReactNode\n  descriptionDirection: 'row' | 'column'\n  loadMore?: ReactNode\n  optionalInfoPlacement: 'left' | 'right'\n  defaultSearchValue: string | null\n  isLoading?: boolean\n}\n\nconst NON_SEARCHABLE_KEYS = [\n  'Tab',\n  ' ',\n  'Enter',\n  'CapsLock',\n  'Shift',\n  'ArrowDown',\n  'ArrowUp',\n  'ArrowLeft',\n  'ArrowRight',\n  'Escape',\n]\n\nconst StyledPopup = styled(Popup)`\n  width: 100%;\n  background-color: ${({ theme }) => theme.colors.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`\n\nconst DropdownContainer = styled(Stack)<{ 'data-grouped': boolean }>`\n  max-height: 256px;\n  overflow-y: scroll;\n  padding: ${({ theme }) => theme.space[0]};\n  padding-bottom: ${({ theme }) => theme.space[0.5]};\n  padding-top: ${({ theme }) => theme.space[0.5]};\n\n  &[data-grouped=\"true\"] {\n    padding-top: ${({ theme }) => theme.space[0]};\n  }\n`\nconst DropdownGroup = styled.button<{ 'data-selectgroup': boolean }>`\n  display: flex;\n  width: 100%;\n  justify-content: left;\n  align-items: center;\n  border: none;\n  background-color: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n  position: sticky;\n  top: 0px;\n  padding-right: ${({ theme }) => theme.space[2]};\n  padding-left: ${({ theme }) => theme.space[2]};\n  height: ${({ theme }) => theme.space[4]};\n  text-align: left;\n  margin-bottom: ${({ theme }) => theme.space['0.25']};\n\n  &:focus {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n    outline: none;\n  }\n\n  &[data-selectgroup='true'] {\n    padding-left: ${({ theme }) => theme.space[2]};\n    border-left: ${({ theme }) => theme.space[0.5]} solid ${({ theme }) =>\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.button<{\n  'data-selected': boolean\n  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  &[data-selected='true'] {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n  }\n\n  &[disabled] {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n\n  }\n\n  &[disabled]:hover, [disabled]:focus {\n    background-color: ${({ theme }) =>\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 > button[role=\"option\"]:not([disabled])',\n  )\n  const activeItem = document.activeElement\n  if (options) {\n    for (let i = 0; i < options?.length; i += 1) {\n      const listLength = options.length\n      if (activeItem === options[i] && activeItem !== options[listLength - 1]) {\n        ;(options[i + 1] as HTMLElement).focus()\n      }\n    }\n  }\n}\nconst moveFocusUp = () => {\n  const options = document.querySelectorAll(\n    '#items > button[role=\"option\"]:not([disabled])',\n  )\n  const activeItem = document.activeElement\n\n  if (options) {\n    for (let i = 0; i < options.length; i += 1) {\n      if (activeItem === options[i] && activeItem !== options[0]) {\n        ;(options[i - 1] as HTMLElement).focus()\n      }\n    }\n  }\n}\nconst handleKeyDownSelect = (key: string) => {\n  if (key === 'ArrowDown') {\n    moveFocusDown()\n  }\n  if (key === 'ArrowUp') {\n    moveFocusUp()\n  }\n}\nconst handleClickOutside = (\n  event: MouseEvent,\n  ref: RefObject<HTMLDivElement>,\n  setIsDropdownVisibile: Dispatch<SetStateAction<boolean>>,\n  refSelect: RefObject<HTMLDivElement>,\n  onSearch: Dispatch<SetStateAction<DataType>>,\n  options: DataType,\n) => {\n  if (\n    ref.current &&\n    !ref.current.contains(event.target as Node) &&\n    !refSelect.current?.contains(event.target as Node)\n  ) {\n    setIsDropdownVisibile(false) // hide dropdown when clicking outside of the dropdown\n    onSearch(options) // reset displayed options to default when dropdown is hidden\n  }\n}\n\nconst handleKeyDown = (\n  event: KeyboardEvent,\n  ref: RefObject<HTMLDivElement>,\n  options: DataType,\n  searchBarActive: boolean,\n  setSearch: Dispatch<SetStateAction<string>>,\n  setDefaultSearch: Dispatch<SetStateAction<string | null>>,\n  search: string,\n) => {\n  // Deals with default search\n  if (\n    ref.current &&\n    !searchBarActive &&\n    !NON_SEARCHABLE_KEYS.includes(event.key) &&\n    document.activeElement?.ariaLabel !== 'search-bar'\n  ) {\n    const currentSearch = search + event.key\n    setSearch(currentSearch)\n    ref.current.focus()\n    if (!Array.isArray(options)) {\n      const closestOptions = { ...options }\n      Object.keys(closestOptions).map((group: string) => {\n        closestOptions[group] = closestOptions[group].filter(option =>\n          option.searchText\n            ? option.searchText.toLocaleLowerCase().startsWith(currentSearch)\n            : option.value.toLocaleLowerCase().startsWith(currentSearch),\n        )\n\n        return null\n      })\n      const closestOption = closestOptions[Object.keys(closestOptions)[0]][0]\n      if (closestOption) {\n        setDefaultSearch(closestOption.searchText ?? closestOption.value)\n      } else {\n        setDefaultSearch(null)\n      }\n    } else {\n      const closestOption = [...options].filter(option =>\n        option.searchText\n          ? option.searchText.toLocaleLowerCase().startsWith(currentSearch)\n          : option.value.toLocaleLowerCase().startsWith(currentSearch),\n      )[0]\n      if (closestOption) {\n        setDefaultSearch(closestOption.searchText ?? closestOption.value)\n      } else {\n        setDefaultSearch(null)\n      }\n    }\n  }\n}\nconst CreateDropdown = ({\n  isEmpty,\n  emptyState,\n  descriptionDirection,\n  loadMore,\n  optionalInfoPlacement,\n  defaultSearchValue,\n  isLoading,\n}: CreateDropdownProps) => {\n  const {\n    setIsDropdownVisible,\n    onChange,\n    options,\n    multiselect,\n    selectAll,\n    selectAllGroup,\n    displayedOptions,\n    setSelectedData,\n    selectedData,\n  } = useSelectInput()\n  const focusedItemRef = useRef<HTMLButtonElement>(null)\n\n  useEffect(() => {\n    if (defaultSearchValue && focusedItemRef?.current) {\n      focusedItemRef.current.focus()\n    }\n  }, [defaultSearchValue])\n\n  if (isEmpty) {\n    return (\n      <EmptyState gap={2} alignItems=\"center\">\n        {emptyState ?? (\n          <Text variant=\"bodyStrong\" as=\"p\">\n            No options\n          </Text>\n        )}\n      </EmptyState>\n    )\n  }\n\n  const handleClick = (clickedOption: OptionType, group?: string) => {\n    setSelectedData({ type: 'selectOption', clickedOption, group })\n    if (multiselect) {\n      if (selectedData.selectedValues.includes(clickedOption.value)) {\n        onChange?.(\n          selectedData.selectedValues.filter(\n            val => val !== clickedOption.value,\n          ),\n        )\n      } else {\n        onChange?.([...selectedData.selectedValues, clickedOption.value])\n      }\n    } else {\n      onChange?.(clickedOption.value)\n    }\n    setIsDropdownVisible(multiselect) // hide the dropdown on click when single select only\n  }\n\n  const selectAllOptions = () => {\n    if (multiselect) {\n      setSelectedData({ type: 'selectAll' })\n      if (selectedData.allSelected && onChange) {\n        onChange([])\n      } else {\n        const allValues: OptionType[] = []\n        if (!Array.isArray(options)) {\n          Object.keys(options).map((group: string) =>\n            options[group].map(option => {\n              if (!option.disabled) {\n                allValues.push(option)\n              }\n\n              return null\n            }),\n          )\n        } else {\n          options.map(option => allValues.push(option))\n        }\n        onChange?.(allValues.map(value => value.value))\n      }\n    }\n  }\n\n  const handleSelectGroup = (group: string) => {\n    if (multiselect) {\n      setSelectedData({ type: 'selectGroup', selectedGroup: group })\n      if (!Array.isArray(options)) {\n        if (selectedData.selectedGroups.includes(group)) {\n          const newSelectedValues = [...selectedData.selectedValues].filter(\n            selectedValue =>\n              !options[group].find(option => option.value === selectedValue),\n          )\n          onChange?.(newSelectedValues)\n        } else {\n          const newSelectedValues = [...selectedData.selectedValues]\n\n          options[group].map(option =>\n            newSelectedValues.includes(option.value) || option.disabled\n              ? null\n              : newSelectedValues.push(option.value),\n          )\n          onChange?.(newSelectedValues)\n        }\n      }\n    }\n  }\n\n  return !Array.isArray(displayedOptions) ? (\n    <DropdownContainer\n      role=\"listbox\"\n      id=\"select-dropdown\"\n      onKeyDown={event => {\n        event.preventDefault()\n        handleKeyDownSelect(event.key)\n      }}\n      data-grouped\n    >\n      {isLoading ? (\n        <Skeleton variant=\"block\" />\n      ) : (\n        <>\n          {selectAll && multiselect ? (\n            <Stack id=\"items\">\n              <DropdownItem\n                disabled={false}\n                data-selected={selectedData.allSelected}\n                aria-label=\"select-all\"\n                data-testid=\"select-all\"\n                id=\"select-all\"\n                type=\"button\"\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              (!Array.isArray(options) && options[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                    disabled={!!option.disabled}\n                    type=\"button\"\n                    data-selected={\n                      selectedData.selectedValues.includes(option.value) &&\n                      !option.disabled\n                    }\n                    aria-label={option.value}\n                    data-testid={`option-${option.value}`}\n                    id={`option-${indexOption}`}\n                    role=\"option\"\n                    onClick={() => {\n                      if (!option.disabled) {\n                        handleClick(option, group)\n                      }\n                    }}\n                    onKeyDown={event =>\n                      [' ', 'Enter'].includes(event.key)\n                        ? handleClick(option, group)\n                        : null\n                    }\n                    ref={\n                      option.value === defaultSearchValue ||\n                      option.searchText === defaultSearchValue\n                        ? focusedItemRef\n                        : null\n                    }\n                  >\n                    {multiselect ? (\n                      <StyledCheckbox\n                        checked={\n                          selectedData.selectedValues.includes(option.value) &&\n                          !option.disabled\n                        }\n                        disabled={option.disabled}\n                        value={option.value}\n                        tabIndex={-1}\n                      >\n                        <DisplayOption\n                          option={option}\n                          descriptionDirection={descriptionDirection}\n                          optionalInfoPlacement={optionalInfoPlacement}\n                        />\n                      </StyledCheckbox>\n                    ) : (\n                      <DisplayOption\n                        option={option}\n                        descriptionDirection={descriptionDirection}\n                        optionalInfoPlacement={optionalInfoPlacement}\n                      />\n                    )}\n                  </DropdownItem>\n                ))}\n              </Stack>\n            </Stack>\n          ))}\n        </>\n      )}\n      {loadMore ? <LoadMore>{loadMore}</LoadMore> : null}\n    </DropdownContainer>\n  ) : (\n    <DropdownContainer\n      role=\"listbox\"\n      id=\"select-dropdown\"\n      onKeyDown={event => {\n        event.preventDefault()\n        handleKeyDownSelect(event.key)\n      }}\n      gap={0.25}\n      data-grouped={false}\n    >\n      {selectAll && multiselect ? (\n        <Stack id=\"items\" gap={0.25}>\n          <DropdownItem\n            disabled={false}\n            data-selected={selectedData.allSelected}\n            aria-label=\"select-all\"\n            data-testid=\"select-all\"\n            type=\"button\"\n            role=\"option\"\n            onKeyDown={event =>\n              [' ', 'Enter'].includes(event.key) ? selectAllOptions() : null\n            }\n            onClick={selectAllOptions}\n          >\n            <StyledCheckbox\n              checked={selectedData.allSelected}\n              disabled={false}\n              value=\"select-all\"\n              data-testid=\"select-all-checkbox\"\n              tabIndex={-1}\n            >\n              <Stack direction=\"column\">\n                <Text as=\"span\" variant=\"body\" placement=\"left\">\n                  {selectAll.label}\n                </Text>\n                <Text\n                  as=\"span\"\n                  variant=\"bodySmall\"\n                  sentiment=\"neutral\"\n                  placement=\"left\"\n                  prominence=\"weak\"\n                >\n                  {selectAll.description}\n                </Text>\n              </Stack>\n            </StyledCheckbox>\n          </DropdownItem>\n        </Stack>\n      ) : null}\n      <Stack id=\"items\" gap={0.25}>\n        {isLoading ? (\n          <Skeleton variant=\"block\" />\n        ) : (\n          displayedOptions.map((option, index) => (\n            <DropdownItem\n              key={option.value}\n              disabled={!!option.disabled}\n              data-selected={\n                selectedData.selectedValues.includes(option.value) &&\n                !option.disabled\n              }\n              onClick={() => {\n                if (!option.disabled) {\n                  handleClick(option)\n                }\n              }}\n              aria-label={option.value}\n              data-testid={`option-${option.value}`}\n              id={`option-${index}`}\n              role=\"option\"\n              type=\"button\"\n              ref={\n                option.value === defaultSearchValue ||\n                option.searchText === defaultSearchValue\n                  ? focusedItemRef\n                  : null\n              }\n              onKeyDown={event =>\n                [' ', 'Enter'].includes(event.key) ? handleClick(option) : null\n              }\n            >\n              {multiselect ? (\n                <StyledCheckbox\n                  checked={\n                    selectedData.selectedValues.includes(option.value) &&\n                    !option.disabled\n                  }\n                  disabled={option.disabled}\n                  value={option.value}\n                  tabIndex={-1}\n                >\n                  <DisplayOption\n                    option={option}\n                    descriptionDirection={descriptionDirection}\n                    optionalInfoPlacement={optionalInfoPlacement}\n                  />\n                </StyledCheckbox>\n              ) : (\n                <DisplayOption\n                  option={option}\n                  descriptionDirection={descriptionDirection}\n                  optionalInfoPlacement={optionalInfoPlacement}\n                />\n              )}\n            </DropdownItem>\n          ))\n        )}\n        {loadMore ? <LoadMore>{loadMore}</LoadMore> : null}\n      </Stack>\n    </DropdownContainer>\n  )\n}\nexport const Dropdown = ({\n  children,\n  emptyState,\n  descriptionDirection,\n  searchable,\n  placeholder,\n  footer,\n  refSelect,\n  loadMore,\n  optionalInfoPlacement,\n  isLoading,\n}: DropdownProps) => {\n  const {\n    setIsDropdownVisible,\n    isDropdownVisible,\n    onSearch,\n    searchInput,\n    options,\n    displayedOptions,\n  } = useSelectInput()\n  const [searchBarActive, setSearchBarActive] = useState(false)\n  const [defaultSearchValue, setDefaultSearch] = useState<string | null>(null)\n  const ref = useRef<HTMLDivElement>(null)\n  const [search, setSearch] = useState<string>('')\n  const [maxWidth, setWidth] = useState<string | number>()\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"]} */"));
164
+ }) => 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":"AAyKgC","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  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  'data-selected': boolean\n  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  &[data-selected='true'] {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n  }\n\n  &[disabled] {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n  }\n\n  &[disabled]:hover, [disabled]: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                disabled={false}\n                data-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              (!Array.isArray(options) && options[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                    disabled={!!option.disabled}\n                    tabIndex={!option.disabled ? 0 : -1}\n                    data-selected={\n                      selectedData.selectedValues.includes(option.value) &&\n                      !option.disabled\n                    }\n                    aria-label={option.value}\n                    data-testid={`option-${option.value}`}\n                    id={`option-${indexOption}`}\n                    role=\"option\"\n                    onClick={() => {\n                      if (!option.disabled) {\n                        handleClick(option, group)\n                      }\n                    }}\n                    onKeyDown={event =>\n                      [' ', 'Enter'].includes(event.key)\n                        ? handleClick(option, group)\n                        : null\n                    }\n                    ref={\n                      option.value === defaultSearchValue ||\n                      option.searchText === defaultSearchValue\n                        ? focusedItemRef\n                        : null\n                    }\n                  >\n                    {multiselect ? (\n                      <StyledCheckbox\n                        checked={\n                          selectedData.selectedValues.includes(option.value) &&\n                          !option.disabled\n                        }\n                        disabled={option.disabled}\n                        value={option.value}\n                        tabIndex={-1}\n                      >\n                        <DisplayOption\n                          option={option}\n                          descriptionDirection={descriptionDirection}\n                          optionalInfoPlacement={optionalInfoPlacement}\n                        />\n                      </StyledCheckbox>\n                    ) : (\n                      <DisplayOption\n                        option={option}\n                        descriptionDirection={descriptionDirection}\n                        optionalInfoPlacement={optionalInfoPlacement}\n                      />\n                    )}\n                  </DropdownItem>\n                ))}\n              </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            disabled={false}\n            data-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              disabled={!!option.disabled}\n              data-selected={\n                selectedData.selectedValues.includes(option.value) &&\n                !option.disabled\n              }\n              onClick={() => {\n                if (!option.disabled) {\n                  handleClick(option)\n                }\n              }}\n              aria-label={option.value}\n              data-testid={`option-${option.value}`}\n              id={`option-${index}`}\n              role=\"option\"\n              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"]} */"));
160
165
  const LoadMore = /* @__PURE__ */ _styled(Stack, process.env.NODE_ENV === "production" ? {
161
166
  target: "eeucz6c0"
162
167
  } : {
@@ -164,9 +169,9 @@ const LoadMore = /* @__PURE__ */ _styled(Stack, process.env.NODE_ENV === "produc
164
169
  label: "LoadMore"
165
170
  })("padding:", ({
166
171
  theme
167
- }) => theme.space[0.5], ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/Dropdown.tsx"],"names":[],"mappings":"AAkK8B","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/Dropdown.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { useCallback, useEffect, useMemo, useRef, useState } from 'react'\nimport type { Dispatch, ReactNode, RefObject, SetStateAction } from 'react'\nimport { Checkbox } from '../Checkbox'\nimport { Popup } from '../Popup'\nimport { Skeleton } from '../Skeleton'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\nimport { DisplayOption } from './DropdownOption'\nimport { SearchBarDropdown } from './SearchBarDropdown'\nimport { useSelectInput } from './SelectInputProvider'\nimport type { DataType, OptionType } from './types'\n\nexport type DropdownProps = {\n  children: ReactNode\n  emptyState: ReactNode\n  descriptionDirection: 'row' | 'column'\n  searchable: boolean\n  placeholder: string\n  footer?: ReactNode\n  refSelect: RefObject<HTMLDivElement>\n  loadMore?: ReactNode\n  optionalInfoPlacement: 'left' | 'right'\n  isLoading?: boolean\n}\n\nexport type CreateDropdownProps = {\n  isEmpty: boolean\n  emptyState: ReactNode\n  descriptionDirection: 'row' | 'column'\n  loadMore?: ReactNode\n  optionalInfoPlacement: 'left' | 'right'\n  defaultSearchValue: string | null\n  isLoading?: boolean\n}\n\nconst NON_SEARCHABLE_KEYS = [\n  'Tab',\n  ' ',\n  'Enter',\n  'CapsLock',\n  'Shift',\n  'ArrowDown',\n  'ArrowUp',\n  'ArrowLeft',\n  'ArrowRight',\n  'Escape',\n]\n\nconst StyledPopup = styled(Popup)`\n  width: 100%;\n  background-color: ${({ theme }) => theme.colors.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`\n\nconst DropdownContainer = styled(Stack)<{ 'data-grouped': boolean }>`\n  max-height: 256px;\n  overflow-y: scroll;\n  padding: ${({ theme }) => theme.space[0]};\n  padding-bottom: ${({ theme }) => theme.space[0.5]};\n  padding-top: ${({ theme }) => theme.space[0.5]};\n\n  &[data-grouped=\"true\"] {\n    padding-top: ${({ theme }) => theme.space[0]};\n  }\n`\nconst DropdownGroup = styled.button<{ 'data-selectgroup': boolean }>`\n  display: flex;\n  width: 100%;\n  justify-content: left;\n  align-items: center;\n  border: none;\n  background-color: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n  position: sticky;\n  top: 0px;\n  padding-right: ${({ theme }) => theme.space[2]};\n  padding-left: ${({ theme }) => theme.space[2]};\n  height: ${({ theme }) => theme.space[4]};\n  text-align: left;\n  margin-bottom: ${({ theme }) => theme.space['0.25']};\n\n  &:focus {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n    outline: none;\n  }\n\n  &[data-selectgroup='true'] {\n    padding-left: ${({ theme }) => theme.space[2]};\n    border-left: ${({ theme }) => theme.space[0.5]} solid ${({ theme }) =>\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.button<{\n  'data-selected': boolean\n  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  &[data-selected='true'] {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n  }\n\n  &[disabled] {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n\n  }\n\n  &[disabled]:hover, [disabled]:focus {\n    background-color: ${({ theme }) =>\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 > button[role=\"option\"]:not([disabled])',\n  )\n  const activeItem = document.activeElement\n  if (options) {\n    for (let i = 0; i < options?.length; i += 1) {\n      const listLength = options.length\n      if (activeItem === options[i] && activeItem !== options[listLength - 1]) {\n        ;(options[i + 1] as HTMLElement).focus()\n      }\n    }\n  }\n}\nconst moveFocusUp = () => {\n  const options = document.querySelectorAll(\n    '#items > button[role=\"option\"]:not([disabled])',\n  )\n  const activeItem = document.activeElement\n\n  if (options) {\n    for (let i = 0; i < options.length; i += 1) {\n      if (activeItem === options[i] && activeItem !== options[0]) {\n        ;(options[i - 1] as HTMLElement).focus()\n      }\n    }\n  }\n}\nconst handleKeyDownSelect = (key: string) => {\n  if (key === 'ArrowDown') {\n    moveFocusDown()\n  }\n  if (key === 'ArrowUp') {\n    moveFocusUp()\n  }\n}\nconst handleClickOutside = (\n  event: MouseEvent,\n  ref: RefObject<HTMLDivElement>,\n  setIsDropdownVisibile: Dispatch<SetStateAction<boolean>>,\n  refSelect: RefObject<HTMLDivElement>,\n  onSearch: Dispatch<SetStateAction<DataType>>,\n  options: DataType,\n) => {\n  if (\n    ref.current &&\n    !ref.current.contains(event.target as Node) &&\n    !refSelect.current?.contains(event.target as Node)\n  ) {\n    setIsDropdownVisibile(false) // hide dropdown when clicking outside of the dropdown\n    onSearch(options) // reset displayed options to default when dropdown is hidden\n  }\n}\n\nconst handleKeyDown = (\n  event: KeyboardEvent,\n  ref: RefObject<HTMLDivElement>,\n  options: DataType,\n  searchBarActive: boolean,\n  setSearch: Dispatch<SetStateAction<string>>,\n  setDefaultSearch: Dispatch<SetStateAction<string | null>>,\n  search: string,\n) => {\n  // Deals with default search\n  if (\n    ref.current &&\n    !searchBarActive &&\n    !NON_SEARCHABLE_KEYS.includes(event.key) &&\n    document.activeElement?.ariaLabel !== 'search-bar'\n  ) {\n    const currentSearch = search + event.key\n    setSearch(currentSearch)\n    ref.current.focus()\n    if (!Array.isArray(options)) {\n      const closestOptions = { ...options }\n      Object.keys(closestOptions).map((group: string) => {\n        closestOptions[group] = closestOptions[group].filter(option =>\n          option.searchText\n            ? option.searchText.toLocaleLowerCase().startsWith(currentSearch)\n            : option.value.toLocaleLowerCase().startsWith(currentSearch),\n        )\n\n        return null\n      })\n      const closestOption = closestOptions[Object.keys(closestOptions)[0]][0]\n      if (closestOption) {\n        setDefaultSearch(closestOption.searchText ?? closestOption.value)\n      } else {\n        setDefaultSearch(null)\n      }\n    } else {\n      const closestOption = [...options].filter(option =>\n        option.searchText\n          ? option.searchText.toLocaleLowerCase().startsWith(currentSearch)\n          : option.value.toLocaleLowerCase().startsWith(currentSearch),\n      )[0]\n      if (closestOption) {\n        setDefaultSearch(closestOption.searchText ?? closestOption.value)\n      } else {\n        setDefaultSearch(null)\n      }\n    }\n  }\n}\nconst CreateDropdown = ({\n  isEmpty,\n  emptyState,\n  descriptionDirection,\n  loadMore,\n  optionalInfoPlacement,\n  defaultSearchValue,\n  isLoading,\n}: CreateDropdownProps) => {\n  const {\n    setIsDropdownVisible,\n    onChange,\n    options,\n    multiselect,\n    selectAll,\n    selectAllGroup,\n    displayedOptions,\n    setSelectedData,\n    selectedData,\n  } = useSelectInput()\n  const focusedItemRef = useRef<HTMLButtonElement>(null)\n\n  useEffect(() => {\n    if (defaultSearchValue && focusedItemRef?.current) {\n      focusedItemRef.current.focus()\n    }\n  }, [defaultSearchValue])\n\n  if (isEmpty) {\n    return (\n      <EmptyState gap={2} alignItems=\"center\">\n        {emptyState ?? (\n          <Text variant=\"bodyStrong\" as=\"p\">\n            No options\n          </Text>\n        )}\n      </EmptyState>\n    )\n  }\n\n  const handleClick = (clickedOption: OptionType, group?: string) => {\n    setSelectedData({ type: 'selectOption', clickedOption, group })\n    if (multiselect) {\n      if (selectedData.selectedValues.includes(clickedOption.value)) {\n        onChange?.(\n          selectedData.selectedValues.filter(\n            val => val !== clickedOption.value,\n          ),\n        )\n      } else {\n        onChange?.([...selectedData.selectedValues, clickedOption.value])\n      }\n    } else {\n      onChange?.(clickedOption.value)\n    }\n    setIsDropdownVisible(multiselect) // hide the dropdown on click when single select only\n  }\n\n  const selectAllOptions = () => {\n    if (multiselect) {\n      setSelectedData({ type: 'selectAll' })\n      if (selectedData.allSelected && onChange) {\n        onChange([])\n      } else {\n        const allValues: OptionType[] = []\n        if (!Array.isArray(options)) {\n          Object.keys(options).map((group: string) =>\n            options[group].map(option => {\n              if (!option.disabled) {\n                allValues.push(option)\n              }\n\n              return null\n            }),\n          )\n        } else {\n          options.map(option => allValues.push(option))\n        }\n        onChange?.(allValues.map(value => value.value))\n      }\n    }\n  }\n\n  const handleSelectGroup = (group: string) => {\n    if (multiselect) {\n      setSelectedData({ type: 'selectGroup', selectedGroup: group })\n      if (!Array.isArray(options)) {\n        if (selectedData.selectedGroups.includes(group)) {\n          const newSelectedValues = [...selectedData.selectedValues].filter(\n            selectedValue =>\n              !options[group].find(option => option.value === selectedValue),\n          )\n          onChange?.(newSelectedValues)\n        } else {\n          const newSelectedValues = [...selectedData.selectedValues]\n\n          options[group].map(option =>\n            newSelectedValues.includes(option.value) || option.disabled\n              ? null\n              : newSelectedValues.push(option.value),\n          )\n          onChange?.(newSelectedValues)\n        }\n      }\n    }\n  }\n\n  return !Array.isArray(displayedOptions) ? (\n    <DropdownContainer\n      role=\"listbox\"\n      id=\"select-dropdown\"\n      onKeyDown={event => {\n        event.preventDefault()\n        handleKeyDownSelect(event.key)\n      }}\n      data-grouped\n    >\n      {isLoading ? (\n        <Skeleton variant=\"block\" />\n      ) : (\n        <>\n          {selectAll && multiselect ? (\n            <Stack id=\"items\">\n              <DropdownItem\n                disabled={false}\n                data-selected={selectedData.allSelected}\n                aria-label=\"select-all\"\n                data-testid=\"select-all\"\n                id=\"select-all\"\n                type=\"button\"\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              (!Array.isArray(options) && options[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                    disabled={!!option.disabled}\n                    type=\"button\"\n                    data-selected={\n                      selectedData.selectedValues.includes(option.value) &&\n                      !option.disabled\n                    }\n                    aria-label={option.value}\n                    data-testid={`option-${option.value}`}\n                    id={`option-${indexOption}`}\n                    role=\"option\"\n                    onClick={() => {\n                      if (!option.disabled) {\n                        handleClick(option, group)\n                      }\n                    }}\n                    onKeyDown={event =>\n                      [' ', 'Enter'].includes(event.key)\n                        ? handleClick(option, group)\n                        : null\n                    }\n                    ref={\n                      option.value === defaultSearchValue ||\n                      option.searchText === defaultSearchValue\n                        ? focusedItemRef\n                        : null\n                    }\n                  >\n                    {multiselect ? (\n                      <StyledCheckbox\n                        checked={\n                          selectedData.selectedValues.includes(option.value) &&\n                          !option.disabled\n                        }\n                        disabled={option.disabled}\n                        value={option.value}\n                        tabIndex={-1}\n                      >\n                        <DisplayOption\n                          option={option}\n                          descriptionDirection={descriptionDirection}\n                          optionalInfoPlacement={optionalInfoPlacement}\n                        />\n                      </StyledCheckbox>\n                    ) : (\n                      <DisplayOption\n                        option={option}\n                        descriptionDirection={descriptionDirection}\n                        optionalInfoPlacement={optionalInfoPlacement}\n                      />\n                    )}\n                  </DropdownItem>\n                ))}\n              </Stack>\n            </Stack>\n          ))}\n        </>\n      )}\n      {loadMore ? <LoadMore>{loadMore}</LoadMore> : null}\n    </DropdownContainer>\n  ) : (\n    <DropdownContainer\n      role=\"listbox\"\n      id=\"select-dropdown\"\n      onKeyDown={event => {\n        event.preventDefault()\n        handleKeyDownSelect(event.key)\n      }}\n      gap={0.25}\n      data-grouped={false}\n    >\n      {selectAll && multiselect ? (\n        <Stack id=\"items\" gap={0.25}>\n          <DropdownItem\n            disabled={false}\n            data-selected={selectedData.allSelected}\n            aria-label=\"select-all\"\n            data-testid=\"select-all\"\n            type=\"button\"\n            role=\"option\"\n            onKeyDown={event =>\n              [' ', 'Enter'].includes(event.key) ? selectAllOptions() : null\n            }\n            onClick={selectAllOptions}\n          >\n            <StyledCheckbox\n              checked={selectedData.allSelected}\n              disabled={false}\n              value=\"select-all\"\n              data-testid=\"select-all-checkbox\"\n              tabIndex={-1}\n            >\n              <Stack direction=\"column\">\n                <Text as=\"span\" variant=\"body\" placement=\"left\">\n                  {selectAll.label}\n                </Text>\n                <Text\n                  as=\"span\"\n                  variant=\"bodySmall\"\n                  sentiment=\"neutral\"\n                  placement=\"left\"\n                  prominence=\"weak\"\n                >\n                  {selectAll.description}\n                </Text>\n              </Stack>\n            </StyledCheckbox>\n          </DropdownItem>\n        </Stack>\n      ) : null}\n      <Stack id=\"items\" gap={0.25}>\n        {isLoading ? (\n          <Skeleton variant=\"block\" />\n        ) : (\n          displayedOptions.map((option, index) => (\n            <DropdownItem\n              key={option.value}\n              disabled={!!option.disabled}\n              data-selected={\n                selectedData.selectedValues.includes(option.value) &&\n                !option.disabled\n              }\n              onClick={() => {\n                if (!option.disabled) {\n                  handleClick(option)\n                }\n              }}\n              aria-label={option.value}\n              data-testid={`option-${option.value}`}\n              id={`option-${index}`}\n              role=\"option\"\n              type=\"button\"\n              ref={\n                option.value === defaultSearchValue ||\n                option.searchText === defaultSearchValue\n                  ? focusedItemRef\n                  : null\n              }\n              onKeyDown={event =>\n                [' ', 'Enter'].includes(event.key) ? handleClick(option) : null\n              }\n            >\n              {multiselect ? (\n                <StyledCheckbox\n                  checked={\n                    selectedData.selectedValues.includes(option.value) &&\n                    !option.disabled\n                  }\n                  disabled={option.disabled}\n                  value={option.value}\n                  tabIndex={-1}\n                >\n                  <DisplayOption\n                    option={option}\n                    descriptionDirection={descriptionDirection}\n                    optionalInfoPlacement={optionalInfoPlacement}\n                  />\n                </StyledCheckbox>\n              ) : (\n                <DisplayOption\n                  option={option}\n                  descriptionDirection={descriptionDirection}\n                  optionalInfoPlacement={optionalInfoPlacement}\n                />\n              )}\n            </DropdownItem>\n          ))\n        )}\n        {loadMore ? <LoadMore>{loadMore}</LoadMore> : null}\n      </Stack>\n    </DropdownContainer>\n  )\n}\nexport const Dropdown = ({\n  children,\n  emptyState,\n  descriptionDirection,\n  searchable,\n  placeholder,\n  footer,\n  refSelect,\n  loadMore,\n  optionalInfoPlacement,\n  isLoading,\n}: DropdownProps) => {\n  const {\n    setIsDropdownVisible,\n    isDropdownVisible,\n    onSearch,\n    searchInput,\n    options,\n    displayedOptions,\n  } = useSelectInput()\n  const [searchBarActive, setSearchBarActive] = useState(false)\n  const [defaultSearchValue, setDefaultSearch] = useState<string | null>(null)\n  const ref = useRef<HTMLDivElement>(null)\n  const [search, setSearch] = useState<string>('')\n  const [maxWidth, setWidth] = useState<string | number>()\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"]} */"));
172
+ }) => 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":"AA4K8B","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  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  'data-selected': boolean\n  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  &[data-selected='true'] {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n  }\n\n  &[disabled] {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n  }\n\n  &[disabled]:hover, [disabled]: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                disabled={false}\n                data-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              (!Array.isArray(options) && options[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                    disabled={!!option.disabled}\n                    tabIndex={!option.disabled ? 0 : -1}\n                    data-selected={\n                      selectedData.selectedValues.includes(option.value) &&\n                      !option.disabled\n                    }\n                    aria-label={option.value}\n                    data-testid={`option-${option.value}`}\n                    id={`option-${indexOption}`}\n                    role=\"option\"\n                    onClick={() => {\n                      if (!option.disabled) {\n                        handleClick(option, group)\n                      }\n                    }}\n                    onKeyDown={event =>\n                      [' ', 'Enter'].includes(event.key)\n                        ? handleClick(option, group)\n                        : null\n                    }\n                    ref={\n                      option.value === defaultSearchValue ||\n                      option.searchText === defaultSearchValue\n                        ? focusedItemRef\n                        : null\n                    }\n                  >\n                    {multiselect ? (\n                      <StyledCheckbox\n                        checked={\n                          selectedData.selectedValues.includes(option.value) &&\n                          !option.disabled\n                        }\n                        disabled={option.disabled}\n                        value={option.value}\n                        tabIndex={-1}\n                      >\n                        <DisplayOption\n                          option={option}\n                          descriptionDirection={descriptionDirection}\n                          optionalInfoPlacement={optionalInfoPlacement}\n                        />\n                      </StyledCheckbox>\n                    ) : (\n                      <DisplayOption\n                        option={option}\n                        descriptionDirection={descriptionDirection}\n                        optionalInfoPlacement={optionalInfoPlacement}\n                      />\n                    )}\n                  </DropdownItem>\n                ))}\n              </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            disabled={false}\n            data-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              disabled={!!option.disabled}\n              data-selected={\n                selectedData.selectedValues.includes(option.value) &&\n                !option.disabled\n              }\n              onClick={() => {\n                if (!option.disabled) {\n                  handleClick(option)\n                }\n              }}\n              aria-label={option.value}\n              data-testid={`option-${option.value}`}\n              id={`option-${index}`}\n              role=\"option\"\n              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
173
  const moveFocusDown = () => {
169
- const options = document.querySelectorAll('#items > button[role="option"]:not([disabled])');
174
+ const options = document.querySelectorAll('#items > div[role="option"]:not([disabled])');
170
175
  const activeItem = document.activeElement;
171
176
  if (options) {
172
177
  for (let i = 0; i < options?.length; i += 1) {
@@ -178,7 +183,7 @@ const moveFocusDown = () => {
178
183
  }
179
184
  };
180
185
  const moveFocusUp = () => {
181
- const options = document.querySelectorAll('#items > button[role="option"]:not([disabled])');
186
+ const options = document.querySelectorAll('#items > div[role="option"]:not([disabled])');
182
187
  const activeItem = document.activeElement;
183
188
  if (options) {
184
189
  for (let i = 0; i < options.length; i += 1) {
@@ -188,11 +193,13 @@ const moveFocusUp = () => {
188
193
  }
189
194
  }
190
195
  };
191
- const handleKeyDownSelect = (key) => {
192
- if (key === "ArrowDown") {
196
+ const handleKeyDownSelect = (event) => {
197
+ if (event.key === "ArrowDown") {
198
+ event.preventDefault();
193
199
  moveFocusDown();
194
200
  }
195
- if (key === "ArrowUp") {
201
+ if (event.key === "ArrowUp") {
202
+ event.preventDefault();
196
203
  moveFocusUp();
197
204
  }
198
205
  };
@@ -318,12 +325,9 @@ const CreateDropdown = ({
318
325
  }
319
326
  }
320
327
  };
321
- return !Array.isArray(displayedOptions) ? /* @__PURE__ */ jsxs(DropdownContainer, { role: "listbox", id: "select-dropdown", onKeyDown: (event) => {
322
- event.preventDefault();
323
- handleKeyDownSelect(event.key);
324
- }, "data-grouped": true, children: [
328
+ return !Array.isArray(displayedOptions) ? /* @__PURE__ */ jsxs(DropdownContainer, { role: "listbox", id: "select-dropdown", onKeyDown: handleKeyDownSelect, "data-grouped": true, children: [
325
329
  isLoading ? /* @__PURE__ */ jsx(Skeleton, { variant: "block" }) : /* @__PURE__ */ jsxs(Fragment, { children: [
326
- selectAll && multiselect ? /* @__PURE__ */ jsx(Stack, { id: "items", children: /* @__PURE__ */ jsx(DropdownItem, { disabled: false, "data-selected": selectedData.allSelected, "aria-label": "select-all", "data-testid": "select-all", id: "select-all", type: "button", role: "option", onKeyDown: (event) => [" ", "Enter"].includes(event.key) ? selectAllOptions() : null, onClick: selectAllOptions, children: /* @__PURE__ */ jsx(StyledCheckbox, { checked: selectedData.allSelected, disabled: false, value: "select-all", "data-testid": "select-all-checkbox", tabIndex: -1, children: /* @__PURE__ */ jsxs(Stack, { direction: "column", children: [
330
+ selectAll && multiselect ? /* @__PURE__ */ jsx(Stack, { id: "items", children: /* @__PURE__ */ jsx(DropdownItem, { disabled: false, "data-selected": selectedData.allSelected, "aria-label": "select-all", "data-testid": "select-all", id: "select-all", tabIndex: 0, role: "option", onKeyDown: (event) => [" ", "Enter"].includes(event.key) ? selectAllOptions() : null, onClick: selectAllOptions, children: /* @__PURE__ */ jsx(StyledCheckbox, { checked: selectedData.allSelected, disabled: false, value: "select-all", "data-testid": "select-all-checkbox", tabIndex: -1, children: /* @__PURE__ */ jsxs(Stack, { direction: "column", children: [
327
331
  /* @__PURE__ */ jsx(Text, { as: "span", variant: "body", placement: "left", children: selectAll.label }),
328
332
  /* @__PURE__ */ jsx(Text, { as: "span", variant: "bodySmall", sentiment: "neutral", placement: "left", prominence: "weak", children: selectAll.description })
329
333
  ] }) }) }) }) : null,
@@ -334,7 +338,7 @@ const CreateDropdown = ({
334
338
  handleSelectGroup(group);
335
339
  }
336
340
  }, "data-selectgroup": selectAllGroup, role: "group", "data-testid": `group-${index}`, onClick: () => selectAllGroup ? handleSelectGroup(group) : null, children: selectAllGroup ? /* @__PURE__ */ jsx(StyledCheckbox, { checked: selectedData.selectedGroups.includes(group), disabled: false, value: group, "data-testid": "select-group", tabIndex: -1, children: /* @__PURE__ */ jsx(Text, { variant: "caption", as: "span", placement: "left", children: group.toUpperCase() }) }) : /* @__PURE__ */ jsx(Text, { variant: "caption", as: "span", placement: "left", sentiment: "neutral", children: group.toUpperCase() }) }, group) : null }) : null,
337
- /* @__PURE__ */ jsx(Stack, { id: "items", gap: "0.25", children: displayedOptions[group].map((option, indexOption) => /* @__PURE__ */ jsx(DropdownItem, { disabled: !!option.disabled, type: "button", "data-selected": selectedData.selectedValues.includes(option.value) && !option.disabled, "aria-label": option.value, "data-testid": `option-${option.value}`, id: `option-${indexOption}`, role: "option", onClick: () => {
341
+ /* @__PURE__ */ jsx(Stack, { id: "items", gap: "0.25", children: displayedOptions[group].map((option, indexOption) => /* @__PURE__ */ jsx(DropdownItem, { disabled: !!option.disabled, tabIndex: !option.disabled ? 0 : -1, "data-selected": selectedData.selectedValues.includes(option.value) && !option.disabled, "aria-label": option.value, "data-testid": `option-${option.value}`, id: `option-${indexOption}`, role: "option", onClick: () => {
338
342
  if (!option.disabled) {
339
343
  handleClick(option, group);
340
344
  }
@@ -342,11 +346,8 @@ const CreateDropdown = ({
342
346
  ] }, group))
343
347
  ] }),
344
348
  loadMore ? /* @__PURE__ */ jsx(LoadMore, { children: loadMore }) : null
345
- ] }) : /* @__PURE__ */ jsxs(DropdownContainer, { role: "listbox", id: "select-dropdown", onKeyDown: (event) => {
346
- event.preventDefault();
347
- handleKeyDownSelect(event.key);
348
- }, gap: 0.25, "data-grouped": false, children: [
349
- selectAll && multiselect ? /* @__PURE__ */ jsx(Stack, { id: "items", gap: 0.25, children: /* @__PURE__ */ jsx(DropdownItem, { disabled: false, "data-selected": selectedData.allSelected, "aria-label": "select-all", "data-testid": "select-all", type: "button", role: "option", onKeyDown: (event) => [" ", "Enter"].includes(event.key) ? selectAllOptions() : null, onClick: selectAllOptions, children: /* @__PURE__ */ jsx(StyledCheckbox, { checked: selectedData.allSelected, disabled: false, value: "select-all", "data-testid": "select-all-checkbox", tabIndex: -1, children: /* @__PURE__ */ jsxs(Stack, { direction: "column", children: [
349
+ ] }) : /* @__PURE__ */ jsxs(DropdownContainer, { role: "listbox", tabIndex: -1, id: "select-dropdown", onKeyDown: handleKeyDownSelect, gap: 0.25, "data-grouped": false, children: [
350
+ selectAll && multiselect ? /* @__PURE__ */ jsx(Stack, { id: "items", gap: 0.25, tabIndex: -1, children: /* @__PURE__ */ jsx(DropdownItem, { disabled: false, "data-selected": selectedData.allSelected, "aria-label": "select-all", "data-testid": "select-all", tabIndex: 0, role: "option", onKeyDown: (event) => [" ", "Enter"].includes(event.key) ? selectAllOptions() : null, onClick: selectAllOptions, children: /* @__PURE__ */ jsx(StyledCheckbox, { checked: selectedData.allSelected, disabled: false, value: "select-all", "data-testid": "select-all-checkbox", tabIndex: -1, children: /* @__PURE__ */ jsxs(Stack, { direction: "column", children: [
350
351
  /* @__PURE__ */ jsx(Text, { as: "span", variant: "body", placement: "left", children: selectAll.label }),
351
352
  /* @__PURE__ */ jsx(Text, { as: "span", variant: "bodySmall", sentiment: "neutral", placement: "left", prominence: "weak", children: selectAll.description })
352
353
  ] }) }) }) }) : null,
@@ -355,7 +356,7 @@ const CreateDropdown = ({
355
356
  if (!option.disabled) {
356
357
  handleClick(option);
357
358
  }
358
- }, "aria-label": option.value, "data-testid": `option-${option.value}`, id: `option-${index}`, role: "option", type: "button", ref: option.value === defaultSearchValue || option.searchText === defaultSearchValue ? focusedItemRef : null, onKeyDown: (event) => [" ", "Enter"].includes(event.key) ? handleClick(option) : null, children: multiselect ? /* @__PURE__ */ jsx(StyledCheckbox, { checked: selectedData.selectedValues.includes(option.value) && !option.disabled, disabled: option.disabled, value: option.value, tabIndex: -1, children: /* @__PURE__ */ jsx(DisplayOption, { option, descriptionDirection, optionalInfoPlacement }) }) : /* @__PURE__ */ jsx(DisplayOption, { option, descriptionDirection, optionalInfoPlacement }) }, option.value)),
359
+ }, "aria-label": option.value, "data-testid": `option-${option.value}`, id: `option-${index}`, role: "option", tabIndex: !option.disabled ? 0 : -1, ref: option.value === defaultSearchValue || option.searchText === defaultSearchValue ? focusedItemRef : null, onKeyDown: (event) => [" ", "Enter"].includes(event.key) ? handleClick(option) : null, children: multiselect ? /* @__PURE__ */ jsx(StyledCheckbox, { checked: selectedData.selectedValues.includes(option.value) && !option.disabled, disabled: option.disabled, value: option.value, tabIndex: -1, children: /* @__PURE__ */ jsx(DisplayOption, { option, descriptionDirection, optionalInfoPlacement }) }) : /* @__PURE__ */ jsx(DisplayOption, { option, descriptionDirection, optionalInfoPlacement }) }, option.value)),
359
360
  loadMore ? /* @__PURE__ */ jsx(LoadMore, { children: loadMore }) : null
360
361
  ] })
361
362
  ] });
@@ -370,7 +371,8 @@ const Dropdown = ({
370
371
  refSelect,
371
372
  loadMore,
372
373
  optionalInfoPlacement,
373
- isLoading
374
+ isLoading,
375
+ size
374
376
  }) => {
375
377
  const {
376
378
  setIsDropdownVisible,
@@ -380,11 +382,30 @@ const Dropdown = ({
380
382
  options,
381
383
  displayedOptions
382
384
  } = useSelectInput();
385
+ const theme = useTheme();
383
386
  const [searchBarActive, setSearchBarActive] = useState(false);
384
387
  const [defaultSearchValue, setDefaultSearch] = useState(null);
385
388
  const ref = useRef(null);
386
389
  const [search, setSearch] = useState("");
387
390
  const [maxWidth, setWidth] = useState();
391
+ useEffect(() => {
392
+ if (refSelect.current && isDropdownVisible) {
393
+ const position = refSelect.current.getBoundingClientRect().bottom + DROPDOWN_MAX_HEIGHT + INPUT_SIZE_HEIGHT[size] + parseInt(theme.space["5"], 10);
394
+ const overflow = position - window.innerHeight;
395
+ if (overflow > 0) {
396
+ const modalElement = document.getElementById("backdrop-modal");
397
+ if (modalElement) {
398
+ modalElement.scrollBy({
399
+ top: overflow,
400
+ behavior: "smooth"
401
+ });
402
+ } else window.scrollBy({
403
+ top: overflow,
404
+ behavior: "smooth"
405
+ });
406
+ }
407
+ }
408
+ }, [isDropdownVisible, refSelect, size, ref.current]);
388
409
  const resizeDropdown = useCallback(() => {
389
410
  if (refSelect.current && refSelect.current.getBoundingClientRect().width > 0) {
390
411
  setWidth(refSelect.current.getBoundingClientRect().width);