@ultraviolet/ui 1.70.2 → 1.71.0

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.
@@ -26,7 +26,7 @@ const StyledPopup = /* @__PURE__ */ _styled(Popup, process.env.NODE_ENV === "pro
26
26
  theme
27
27
  }) => `${theme.shadows.raised[0]}, ${theme.shadows.raised[1]}`, ";padding:", ({
28
28
  theme
29
- }) => theme.space[0], ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/Dropdown.tsx"],"names":[],"mappings":"AAiDiC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/Dropdown.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { 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(refSelect.current?.offsetWidth)\n\n  const resizeDropdown = useCallback(() => {\n    if (refSelect.current) {\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}\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"]} */"));
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"]} */"));
30
30
  const DropdownContainer = /* @__PURE__ */ _styled(Stack, process.env.NODE_ENV === "production" ? {
31
31
  target: "eeucz6c7"
32
32
  } : {
@@ -40,7 +40,7 @@ const DropdownContainer = /* @__PURE__ */ _styled(Stack, process.env.NODE_ENV ==
40
40
  theme
41
41
  }) => theme.space[0.5], ';&[data-grouped="true"]{padding-top:', ({
42
42
  theme
43
- }) => theme.space[0], ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/Dropdown.tsx"],"names":[],"mappings":"AAyDoE","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/Dropdown.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { 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(refSelect.current?.offsetWidth)\n\n  const resizeDropdown = useCallback(() => {\n    if (refSelect.current) {\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}\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"]} */"));
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"]} */"));
44
44
  const DropdownGroup = /* @__PURE__ */ _styled("button", process.env.NODE_ENV === "production" ? {
45
45
  target: "eeucz6c6"
46
46
  } : {
@@ -66,7 +66,7 @@ const DropdownGroup = /* @__PURE__ */ _styled("button", process.env.NODE_ENV ===
66
66
  theme
67
67
  }) => theme.colors.neutral.backgroundWeak, ";}&[data-selectgroup='true']:focus{background-color:", ({
68
68
  theme
69
- }) => theme.colors.neutral.backgroundHover, ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/Dropdown.tsx"],"names":[],"mappings":"AAoEoE","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/Dropdown.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { 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(refSelect.current?.offsetWidth)\n\n  const resizeDropdown = useCallback(() => {\n    if (refSelect.current) {\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}\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"]} */"));
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"]} */"));
70
70
  const DropdownGroupWrapper = /* @__PURE__ */ _styled("div", process.env.NODE_ENV === "production" ? {
71
71
  target: "eeucz6c5"
72
72
  } : {
@@ -78,7 +78,7 @@ const DropdownGroupWrapper = /* @__PURE__ */ _styled("div", process.env.NODE_ENV
78
78
  } : {
79
79
  name: "vkf8h3",
80
80
  styles: "position:sticky;top:0px",
81
- map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/Dropdown.tsx"],"names":[],"mappings":"AAkGuC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/Dropdown.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { 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(refSelect.current?.offsetWidth)\n\n  const resizeDropdown = useCallback(() => {\n    if (refSelect.current) {\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}\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"]} */",
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"]} */",
82
82
  toString: _EMOTION_STRINGIFIED_CSS_ERROR__
83
83
  });
84
84
  const DropdownItem = /* @__PURE__ */ _styled("button", process.env.NODE_ENV === "production" ? {
@@ -118,7 +118,7 @@ const DropdownItem = /* @__PURE__ */ _styled("button", process.env.NODE_ENV ===
118
118
  theme
119
119
  }) => theme.colors.neutral.backgroundStrongDisabled, ";color:", ({
120
120
  theme
121
- }) => theme.colors.neutral.textStrongDisabled, ";cursor:not-allowed;outline:none;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/Dropdown.tsx"],"names":[],"mappings":"AAyGE","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/Dropdown.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { 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(refSelect.current?.offsetWidth)\n\n  const resizeDropdown = useCallback(() => {\n    if (refSelect.current) {\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}\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"]} */"));
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"]} */"));
122
122
  const PopupFooter = /* @__PURE__ */ _styled("div", process.env.NODE_ENV === "production" ? {
123
123
  target: "eeucz6c3"
124
124
  } : {
@@ -134,7 +134,7 @@ const PopupFooter = /* @__PURE__ */ _styled("div", process.env.NODE_ENV === "pro
134
134
  theme
135
135
  }) => theme.space[2], ";box-shadow:", ({
136
136
  theme
137
- }) => theme.shadows.dropdown, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/Dropdown.tsx"],"names":[],"mappings":"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(refSelect.current?.offsetWidth)\n\n  const resizeDropdown = useCallback(() => {\n    if (refSelect.current) {\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}\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"]} */"));
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"]} */"));
138
138
  const StyledCheckbox = /* @__PURE__ */ _styled(Checkbox, process.env.NODE_ENV === "production" ? {
139
139
  target: "eeucz6c2"
140
140
  } : {
@@ -146,7 +146,7 @@ const StyledCheckbox = /* @__PURE__ */ _styled(Checkbox, process.env.NODE_ENV ==
146
146
  } : {
147
147
  name: "1l9xw77",
148
148
  styles: "width:100%;position:static;text-align:left;align-items:center;pointer-events:none",
149
- map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/Dropdown.tsx"],"names":[],"mappings":"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(refSelect.current?.offsetWidth)\n\n  const resizeDropdown = useCallback(() => {\n    if (refSelect.current) {\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}\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"]} */",
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"]} */",
150
150
  toString: _EMOTION_STRINGIFIED_CSS_ERROR__
151
151
  });
152
152
  const EmptyState = /* @__PURE__ */ _styled(Stack, process.env.NODE_ENV === "production" ? {
@@ -156,7 +156,7 @@ const EmptyState = /* @__PURE__ */ _styled(Stack, process.env.NODE_ENV === "prod
156
156
  label: "EmptyState"
157
157
  })("padding:", ({
158
158
  theme
159
- }) => theme.space[2], ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/Dropdown.tsx"],"names":[],"mappings":"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(refSelect.current?.offsetWidth)\n\n  const resizeDropdown = useCallback(() => {\n    if (refSelect.current) {\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}\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"]} */"));
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"]} */"));
160
160
  const LoadMore = /* @__PURE__ */ _styled(Stack, process.env.NODE_ENV === "production" ? {
161
161
  target: "eeucz6c0"
162
162
  } : {
@@ -164,7 +164,7 @@ const LoadMore = /* @__PURE__ */ _styled(Stack, process.env.NODE_ENV === "produc
164
164
  label: "LoadMore"
165
165
  })("padding:", ({
166
166
  theme
167
- }) => theme.space[0.5], ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/Dropdown.tsx"],"names":[],"mappings":"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(refSelect.current?.offsetWidth)\n\n  const resizeDropdown = useCallback(() => {\n    if (refSelect.current) {\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}\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"]} */"));
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"]} */"));
168
168
  const moveFocusDown = () => {
169
169
  const options = document.querySelectorAll('#items > button[role="option"]:not([disabled])');
170
170
  const activeItem = document.activeElement;
@@ -384,9 +384,9 @@ const Dropdown = ({
384
384
  const [defaultSearchValue, setDefaultSearch] = useState(null);
385
385
  const ref = useRef(null);
386
386
  const [search, setSearch] = useState("");
387
- const [maxWidth, setWidth] = useState(refSelect.current?.offsetWidth);
387
+ const [maxWidth, setWidth] = useState();
388
388
  const resizeDropdown = useCallback(() => {
389
- if (refSelect.current) {
389
+ if (refSelect.current && refSelect.current.getBoundingClientRect().width > 0) {
390
390
  setWidth(refSelect.current.getBoundingClientRect().width);
391
391
  }
392
392
  }, [refSelect]);
@@ -432,7 +432,7 @@ const Dropdown = ({
432
432
  searchable && !isLoading ? /* @__PURE__ */ jsx(SearchBarDropdown, { placeholder, displayedOptions, setSearchBarActive }) : null,
433
433
  /* @__PURE__ */ jsx(CreateDropdown, { isEmpty, emptyState, descriptionDirection, loadMore, optionalInfoPlacement, defaultSearchValue, isLoading }),
434
434
  footer ? /* @__PURE__ */ jsx(PopupFooter, { children: footer }) : null
435
- ] }), placement: "bottom", disableAnimation: true, maxWidth, hasArrow: false, ref, tabIndex: 0, role: "dialog", debounceDelay: 0, containerFullWidth: true, children });
435
+ ] }), placement: "bottom", disableAnimation: true, maxWidth: maxWidth ?? refSelect.current?.offsetWidth, hasArrow: false, ref, tabIndex: 0, role: "dialog", debounceDelay: 0, containerFullWidth: true, children });
436
436
  };
437
437
  export {
438
438
  Dropdown