@ultraviolet/ui 2.0.4 → 2.0.5
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.
|
@@ -33,7 +33,7 @@ const StyledPopup = /* @__PURE__ */ _styled(Popup, process.env.NODE_ENV === "pro
|
|
|
33
33
|
theme
|
|
34
34
|
}) => theme.space[0], ";margin-bottom:", ({
|
|
35
35
|
theme
|
|
36
|
-
}) => theme.space[10], ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInput/Dropdown.tsx"],"names":[],"mappings":"AA0EiC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInput/Dropdown.tsx","sourcesContent":["'use client'\n\nimport { useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport type {\n  ComponentProps,\n  Dispatch,\n  KeyboardEvent,\n  ReactNode,\n  RefObject,\n  SetStateAction,\n} from 'react'\nimport {\n  useCallback,\n  useContext,\n  useEffect,\n  useMemo,\n  useRef,\n  useState,\n} from 'react'\nimport { Checkbox } from '../Checkbox'\nimport { ModalContext } from '../Modal/ModalProvider'\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'\nimport { INPUT_SIZE_HEIGHT } from './types'\n\nconst DROPDOWN_MAX_HEIGHT = 256\n\nexport type DropdownProps = {\n  id?: string\n  children: ReactNode\n  emptyState: ReactNode\n  descriptionDirection: 'row' | 'column'\n  searchable: boolean\n  placeholder: string\n  footer?: ((closeDropdown: () => void) => ReactNode) | ReactNode\n  refSelect: RefObject<HTMLDivElement | null>\n  loadMore?: ReactNode\n  optionalInfoPlacement: 'left' | 'right'\n  isLoading?: boolean\n  size: 'small' | 'medium' | 'large'\n  dropdownAlign?: ComponentProps<typeof Popup>['align']\n  portalTarget?: ComponentProps<typeof Popup>['portalTarget']\n}\n\nexport type CreateDropdownProps = {\n  isEmpty: boolean\n  emptyState: ReactNode\n  descriptionDirection: 'row' | 'column'\n  loadMore?: ReactNode\n  optionalInfoPlacement: 'left' | 'right'\n  defaultSearchValue: string | null\n  isLoading?: boolean\n}\n\nconst NON_SEARCHABLE_KEYS = [\n  'Tab',\n  ' ',\n  'Enter',\n  'CapsLock',\n  'Shift',\n  'ArrowDown',\n  'ArrowUp',\n  'ArrowLeft',\n  'ArrowRight',\n  'Escape',\n]\n\nconst StyledPopup = styled(Popup)`\n  width: 100%;\n  min-width: 320px;\n  background-color: ${({ theme }) =>\n    theme.colors.other.elevation.background.raised};\n  color: ${({ theme }) => theme.colors.neutral.text};\n  box-shadow: ${({ theme }) =>\n    `${theme.shadows.raised[0]}, ${theme.shadows.raised[1]}`};\n  padding: ${({ theme }) => theme.space[0]};\n  margin-bottom: ${({ theme }) => theme.space[10]};\n`\n\nconst DropdownContainer = styled(Stack)<{ 'data-grouped': boolean }>`\n  max-height: ${DROPDOWN_MAX_HEIGHT}px;\n  overflow-y: auto;\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: 0;\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: 0;\n`\nconst DropdownItem = styled.div<{\n  'aria-selected': boolean\n  'aria-disabled': boolean\n}>`\n  text-align:left;\n  border: none;\n  background-color: ${({ theme }) =>\n    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  border: 1px transparent solid;\n\n  &:hover, :focus {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n    color: ${({ theme }) => theme.colors.primary.text};\n    cursor: pointer;\n  }\n\n  &[aria-selected='true'] {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n  }\n\n  &[aria-disabled=\"true\"] {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n  }\n\n  &[aria-disabled=\"true\"]:hover, [aria-disabled=\"true\"]:focus {\n    background-color: ${({ theme }) =>\n      theme.colors.neutral.backgroundStrongDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textStrongDisabled};\n    cursor: not-allowed;\n    outline: none;\n  }\n`\n\nconst PopupFooter = styled.div`\n  width: 100%;\n  padding: ${({ theme }) => theme.space[1.5]} ${({ theme }) => theme.space[2]}\n    ${({ theme }) => theme.space[1.5]} ${({ theme }) => theme.space[2]};\n  box-shadow: ${({ theme }) => theme.shadows.dropdown};\n`\nconst StyledCheckbox = styled(Checkbox)`\n  width: 100%;\n  position: static;\n  text-align: left;\n  align-items: center;\n  pointer-events: none;\n` // pointer-events: none prevents any error when using the checkbox in a form since it is an unnamed input\n\nconst EmptyState = styled(Stack)`\n  padding: ${({ theme }) => theme.space[2]};\n`\nconst LoadMore = styled(Stack)`\n  padding: ${({ theme }) => theme.space[0.5]};\n`\n\nconst moveFocusDown = () => {\n  const options = document.querySelectorAll(\n    '#items > div[role=\"option\"]:not([disabled])',\n  )\n  const activeItem = document.activeElement\n  if (options) {\n    for (let i = 0; i < options?.length; i += 1) {\n      const listLength = options.length\n      if (activeItem === options[i] && activeItem !== options[listLength - 1]) {\n        ;(options[i + 1] as HTMLElement).focus()\n      }\n    }\n  }\n}\nconst moveFocusUp = () => {\n  const options = document.querySelectorAll(\n    '#items > div[role=\"option\"]:not([disabled])',\n  )\n  const activeItem = document.activeElement\n\n  if (options) {\n    for (let i = 0; i < options.length; i += 1) {\n      if (activeItem === options[i] && activeItem !== options[0]) {\n        ;(options[i - 1] as HTMLElement).focus()\n      }\n    }\n  }\n}\nconst handleKeyDownSelect = (event: KeyboardEvent<HTMLDivElement>) => {\n  if (event.key === 'ArrowDown') {\n    event.preventDefault()\n    moveFocusDown()\n  }\n\n  if (event.key === 'ArrowUp') {\n    event.preventDefault()\n    moveFocusUp()\n  }\n\n  if (event.key === ' ') {\n    // No scroll\n    event.preventDefault()\n  }\n}\n\nconst handleKeyDown = (\n  event: globalThis.KeyboardEvent,\n  ref: RefObject<HTMLDivElement | null>,\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].find(option =>\n        option.searchText\n          ? option.searchText.toLocaleLowerCase().startsWith(currentSearch)\n          : option.value.toLocaleLowerCase().startsWith(currentSearch),\n      )\n      if (closestOption) {\n        setDefaultSearch(closestOption.searchText ?? closestOption.value)\n      } else {\n        setDefaultSearch(null)\n      }\n    }\n  }\n}\nconst CreateDropdown = ({\n  isEmpty,\n  emptyState,\n  descriptionDirection,\n  loadMore,\n  optionalInfoPlacement,\n  defaultSearchValue,\n  isLoading,\n}: CreateDropdownProps) => {\n  const {\n    setIsDropdownVisible,\n    onChange,\n    options,\n    multiselect,\n    selectAll,\n    selectAllGroup,\n    displayedOptions,\n    setSelectedData,\n    selectedData,\n  } = useSelectInput()\n  const focusedItemRef = useRef<HTMLDivElement>(null)\n\n  useEffect(() => {\n    if (defaultSearchValue && focusedItemRef?.current) {\n      focusedItemRef.current.focus()\n    }\n  }, [defaultSearchValue])\n\n  if (isEmpty) {\n    return (\n      <EmptyState alignItems=\"center\" gap={2}>\n        {emptyState ?? (\n          <Text as=\"p\" variant=\"bodyStrong\">\n            No options\n          </Text>\n        )}\n      </EmptyState>\n    )\n  }\n\n  const handleClick = (clickedOption: OptionType, group?: string) => {\n    setSelectedData({ clickedOption, group, type: 'selectOption' })\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({ selectedGroup: group, type: 'selectGroup' })\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      data-grouped\n      id=\"select-dropdown\"\n      onKeyDown={handleKeyDownSelect}\n      role=\"listbox\"\n    >\n      {isLoading ? (\n        <Skeleton variant=\"block\" />\n      ) : (\n        <>\n          {selectAll && multiselect ? (\n            <Stack id=\"items\">\n              <DropdownItem\n                aria-disabled={false}\n                aria-label=\"select-all\"\n                aria-selected={selectedData.allSelected}\n                data-testid=\"select-all\"\n                id=\"select-all\"\n                onClick={selectAllOptions}\n                onKeyDown={event =>\n                  [' ', 'Enter'].includes(event.key) ? selectAllOptions() : null\n                }\n                role=\"option\"\n                tabIndex={0}\n              >\n                <StyledCheckbox\n                  checked={selectedData.allSelected}\n                  data-testid=\"select-all-checkbox\"\n                  disabled={false}\n                  onChange={selectAllOptions}\n                  tabIndex={-1}\n                  value=\"select-all\"\n                >\n                  <Stack direction=\"column\">\n                    <Text as=\"span\" placement=\"left\" variant=\"body\">\n                      {selectAll.label}\n                    </Text>\n                    <Text\n                      as=\"span\"\n                      placement=\"left\"\n                      prominence=\"weak\"\n                      sentiment=\"neutral\"\n                      variant=\"bodySmall\"\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 gap={0.25} key={group}>\n              {displayedOptions[group].length > 0 ? (\n                <DropdownGroupWrapper id={selectAllGroup ? 'items' : undefined}>\n                  {group ? (\n                    <DropdownGroup\n                      data-selectgroup={selectAllGroup}\n                      data-testid={`group-${index}`}\n                      key={group}\n                      onClick={() =>\n                        selectAllGroup ? handleSelectGroup(group) : null\n                      }\n                      onKeyDown={event => {\n                        if ([' ', 'Enter'].includes(event.key)) {\n                          event.preventDefault()\n                          handleSelectGroup(group)\n                        }\n                      }}\n                      role=\"group\"\n                      tabIndex={selectAllGroup ? 0 : -1}\n                      type=\"button\"\n                    >\n                      {selectAllGroup ? (\n                        <StyledCheckbox\n                          checked={selectedData.selectedGroups.includes(group)}\n                          data-testid=\"select-group\"\n                          disabled={false}\n                          onChange={() =>\n                            selectAllGroup ? handleSelectGroup(group) : null\n                          }\n                          tabIndex={-1}\n                          value={group}\n                        >\n                          <Text\n                            as=\"span\"\n                            placement=\"left\"\n                            sentiment=\"neutral\"\n                            variant=\"caption\"\n                          >\n                            {group.toUpperCase()}\n                          </Text>\n                        </StyledCheckbox>\n                      ) : (\n                        <Text\n                          as=\"span\"\n                          placement=\"left\"\n                          sentiment=\"neutral\"\n                          variant=\"caption\"\n                        >\n                          {group.toUpperCase()}\n                        </Text>\n                      )}\n                    </DropdownGroup>\n                  ) : null}\n                </DropdownGroupWrapper>\n              ) : null}\n              <Stack gap=\"0.25\" id=\"items\">\n                {displayedOptions[group].map((option, indexOption) => (\n                  <DropdownItem\n                    aria-disabled={!!option.disabled}\n                    aria-label={option.value}\n                    aria-selected={\n                      selectedData.selectedValues.includes(option.value) &&\n                      !option.disabled\n                    }\n                    data-testid={`option-${option.value}`}\n                    id={`option-${indexOption}`}\n                    key={option.value}\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                    role=\"option\"\n                    tabIndex={!option.disabled ? 0 : -1}\n                  >\n                    {multiselect ? (\n                      <StyledCheckbox\n                        checked={\n                          selectedData.selectedValues.includes(option.value) &&\n                          !option.disabled\n                        }\n                        disabled={option.disabled}\n                        onChange={() => {\n                          if (!option.disabled) {\n                            handleClick(option, group)\n                          }\n                        }}\n                        tabIndex={-1}\n                        value={option.value}\n                      >\n                        <DisplayOption\n                          descriptionDirection={descriptionDirection}\n                          option={option}\n                          optionalInfoPlacement={optionalInfoPlacement}\n                        />\n                      </StyledCheckbox>\n                    ) : (\n                      <DisplayOption\n                        descriptionDirection={descriptionDirection}\n                        option={option}\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      data-grouped={false}\n      gap={0.25}\n      id=\"select-dropdown\"\n      onKeyDown={handleKeyDownSelect}\n      role=\"listbox\"\n    >\n      {selectAll && multiselect ? (\n        <Stack gap={0.25} id=\"items\" tabIndex={-1}>\n          <DropdownItem\n            aria-disabled={false}\n            aria-label=\"select-all\"\n            aria-selected={selectedData.allSelected}\n            data-testid=\"select-all\"\n            onClick={selectAllOptions}\n            onKeyDown={event =>\n              [' ', 'Enter'].includes(event.key) ? selectAllOptions() : null\n            }\n            role=\"option\"\n            tabIndex={0}\n          >\n            <StyledCheckbox\n              checked={selectedData.allSelected}\n              data-testid=\"select-all-checkbox\"\n              disabled={false}\n              onChange={selectAllOptions}\n              tabIndex={-1}\n              value=\"select-all\"\n            >\n              <Stack direction=\"column\">\n                <Text as=\"span\" placement=\"left\" variant=\"body\">\n                  {selectAll.label}\n                </Text>\n                <Text\n                  as=\"span\"\n                  placement=\"left\"\n                  prominence=\"weak\"\n                  sentiment=\"neutral\"\n                  variant=\"bodySmall\"\n                >\n                  {selectAll.description}\n                </Text>\n              </Stack>\n            </StyledCheckbox>\n          </DropdownItem>\n        </Stack>\n      ) : null}\n      <Stack gap={0.25} id=\"items\">\n        {isLoading ? (\n          <Skeleton variant=\"block\" />\n        ) : (\n          displayedOptions.map((option, index) => (\n            <DropdownItem\n              aria-disabled={!!option.disabled}\n              aria-label={option.value}\n              aria-selected={\n                selectedData.selectedValues.includes(option.value) &&\n                !option.disabled\n              }\n              data-testid={`option-${option.value}`}\n              id={`option-${index}`}\n              key={option.value}\n              onClick={() => {\n                if (!option.disabled) {\n                  handleClick(option)\n                }\n              }}\n              onKeyDown={event =>\n                [' ', 'Enter'].includes(event.key) ? handleClick(option) : null\n              }\n              ref={\n                option.value === defaultSearchValue ||\n                option.searchText === defaultSearchValue\n                  ? focusedItemRef\n                  : null\n              }\n              role=\"option\"\n              tabIndex={!option.disabled ? 0 : -1}\n            >\n              {multiselect ? (\n                <StyledCheckbox\n                  checked={\n                    selectedData.selectedValues.includes(option.value) &&\n                    !option.disabled\n                  }\n                  disabled={option.disabled}\n                  onChange={() => {\n                    if (!option.disabled) {\n                      handleClick(option)\n                    }\n                  }}\n                  tabIndex={-1}\n                  value={option.value}\n                >\n                  <DisplayOption\n                    descriptionDirection={descriptionDirection}\n                    option={option}\n                    optionalInfoPlacement={optionalInfoPlacement}\n                  />\n                </StyledCheckbox>\n              ) : (\n                <DisplayOption\n                  descriptionDirection={descriptionDirection}\n                  option={option}\n                  optionalInfoPlacement={optionalInfoPlacement}\n                />\n              )}\n            </DropdownItem>\n          ))\n        )}\n        {loadMore ? <LoadMore>{loadMore}</LoadMore> : null}\n      </Stack>\n    </DropdownContainer>\n  )\n}\nexport const Dropdown = ({\n  children,\n  emptyState,\n  descriptionDirection,\n  searchable,\n  placeholder,\n  footer,\n  refSelect,\n  loadMore,\n  optionalInfoPlacement,\n  isLoading,\n  size,\n  dropdownAlign,\n  portalTarget,\n  id,\n}: DropdownProps) => {\n  const {\n    setIsDropdownVisible,\n    isDropdownVisible,\n    onSearch,\n    searchInput,\n    options,\n    displayedOptions,\n    numberOfOptions,\n  } = useSelectInput()\n  const theme = useTheme()\n  const [searchBarActive, setSearchBarActive] = useState(false)\n  const [defaultSearchValue, setDefaultSearch] = useState<string | null>(null)\n  const ref = useRef<HTMLDivElement>(null)\n  const [search, setSearch] = useState<string>('')\n  const [maxWidth, setWidth] = useState<string | number>()\n  const modalContext = useContext(ModalContext)\n\n  useEffect(() => {\n    if (refSelect.current && isDropdownVisible) {\n      const position =\n        refSelect.current.getBoundingClientRect().bottom +\n        DROPDOWN_MAX_HEIGHT +\n        Number(theme.sizing[INPUT_SIZE_HEIGHT[size]].replace('rem', '')) * 16 +\n        Number.parseInt(theme.space['5'], 10)\n      const overflow = position - window.innerHeight + 32\n      if (overflow > 0 && modalContext) {\n        const currentModal = modalContext.openedModals[0]\n        const modalElement = currentModal?.ref.current\n\n        if (modalElement) {\n          const parentElement = modalElement.parentNode as HTMLElement\n          if (parentElement) {\n            parentElement.scrollBy({\n              behavior: 'smooth',\n              top: overflow,\n            })\n          }\n        } else {\n          window.scrollBy({ behavior: 'smooth', top: overflow })\n        }\n      }\n    }\n    // oxlint-disable-next-line react/exhaustive-deps\n  }, [isDropdownVisible, refSelect, size, ref.current])\n\n  const resizeDropdown = useCallback(() => {\n    if (\n      refSelect.current &&\n      refSelect.current.getBoundingClientRect().width > 0\n    ) {\n      setWidth(refSelect.current.getBoundingClientRect().width)\n    }\n  }, [refSelect])\n\n  useEffect(() => {\n    resizeDropdown()\n\n    window.addEventListener('resize', resizeDropdown)\n\n    return () => window.removeEventListener('resize', resizeDropdown)\n  }, [resizeDropdown])\n\n  useEffect(() => {\n    if (!searchInput) {\n      onSearch(options)\n    }\n  }, [onSearch, options, searchInput])\n\n  useEffect(() => {\n    if (!isDropdownVisible) {\n      setDefaultSearch(null)\n      setSearch('')\n    }\n\n    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      if (!searchable) {\n        document.removeEventListener('keydown', event =>\n          handleKeyDown(\n            event,\n            ref,\n            options,\n            searchBarActive,\n            setSearch,\n            setDefaultSearch,\n            search,\n          ),\n        )\n      }\n    }\n  }, [\n    isDropdownVisible,\n    searchBarActive,\n    options,\n    onSearch,\n    search,\n    refSelect,\n    setDefaultSearch,\n    setIsDropdownVisible,\n    searchable,\n  ])\n\n  // No data is displayed (because of the search or because no data is provided)\n  // Set to true when noData by default\n  const isEmpty = useMemo(() => {\n    if (numberOfOptions === 0) {\n      return true\n    }\n    if (Array.isArray(displayedOptions)) {\n      return displayedOptions.length === 0\n    }\n    const groups = Object.keys(displayedOptions)\n    for (const group of groups) {\n      if (displayedOptions[group].length > 0) {\n        return false\n      }\n    }\n\n    return true\n  }, [displayedOptions, numberOfOptions])\n\n  const computedFooter = useMemo(() => {\n    if (footer && !isEmpty) {\n      if (typeof footer === 'function') {\n        return (\n          <PopupFooter>{footer(() => setIsDropdownVisible(false))}</PopupFooter>\n        )\n      }\n\n      return <PopupFooter>{footer}</PopupFooter>\n    }\n\n    return null\n  }, [isEmpty, footer, setIsDropdownVisible])\n\n  return (\n    <StyledPopup\n      align={dropdownAlign ?? 'start'}\n      containerFullWidth\n      debounceDelay={0}\n      disableAnimation\n      hasArrow={false}\n      hideOnClickOutside\n      id={id}\n      maxWidth={maxWidth ?? refSelect.current?.offsetWidth}\n      onClose={() => setIsDropdownVisible(false)}\n      placement=\"bottom\"\n      portalTarget={portalTarget}\n      ref={ref}\n      role=\"dialog\"\n      tabIndex={-1}\n      text={\n        <Stack>\n          {searchable && !isLoading && numberOfOptions >= 6 ? (\n            <SearchBarDropdown\n              displayedOptions={displayedOptions}\n              placeholder={placeholder}\n              setSearchBarActive={setSearchBarActive}\n            />\n          ) : null}\n          <CreateDropdown\n            defaultSearchValue={defaultSearchValue}\n            descriptionDirection={descriptionDirection}\n            emptyState={emptyState}\n            isEmpty={isEmpty}\n            isLoading={isLoading}\n            loadMore={loadMore}\n            optionalInfoPlacement={optionalInfoPlacement}\n          />\n          {computedFooter}\n        </Stack>\n      }\n      visible={isDropdownVisible}\n    >\n      {children}\n    </StyledPopup>\n  )\n}\n"]} */"));
|
|
36
|
+
}) => theme.space[10], ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInput/Dropdown.tsx"],"names":[],"mappings":"AA0EiC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInput/Dropdown.tsx","sourcesContent":["'use client'\n\nimport { useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport type {\n  ComponentProps,\n  Dispatch,\n  KeyboardEvent,\n  ReactNode,\n  RefObject,\n  SetStateAction,\n} from 'react'\nimport {\n  useCallback,\n  useContext,\n  useEffect,\n  useMemo,\n  useRef,\n  useState,\n} from 'react'\nimport { Checkbox } from '../Checkbox'\nimport { ModalContext } from '../Modal/ModalProvider'\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'\nimport { INPUT_SIZE_HEIGHT } from './types'\n\nconst DROPDOWN_MAX_HEIGHT = 256\n\nexport type DropdownProps = {\n  id?: string\n  children: ReactNode\n  emptyState: ReactNode\n  descriptionDirection: 'row' | 'column'\n  searchable: boolean\n  placeholder: string\n  footer?: ((closeDropdown: () => void) => ReactNode) | ReactNode\n  refSelect: RefObject<HTMLDivElement | null>\n  loadMore?: ReactNode\n  optionalInfoPlacement: 'left' | 'right'\n  isLoading?: boolean\n  size: 'small' | 'medium' | 'large'\n  dropdownAlign?: ComponentProps<typeof Popup>['align']\n  portalTarget?: ComponentProps<typeof Popup>['portalTarget']\n}\n\nexport type CreateDropdownProps = {\n  isEmpty: boolean\n  emptyState: ReactNode\n  descriptionDirection: 'row' | 'column'\n  loadMore?: ReactNode\n  optionalInfoPlacement: 'left' | 'right'\n  defaultSearchValue: string | null\n  isLoading?: boolean\n}\n\nconst NON_SEARCHABLE_KEYS = [\n  'Tab',\n  ' ',\n  'Enter',\n  'CapsLock',\n  'Shift',\n  'ArrowDown',\n  'ArrowUp',\n  'ArrowLeft',\n  'ArrowRight',\n  'Escape',\n]\n\nconst StyledPopup = styled(Popup)`\n  width: 100%;\n  min-width: 320px;\n  background-color: ${({ theme }) =>\n    theme.colors.other.elevation.background.raised};\n  color: ${({ theme }) => theme.colors.neutral.text};\n  box-shadow: ${({ theme }) =>\n    `${theme.shadows.raised[0]}, ${theme.shadows.raised[1]}`};\n  padding: ${({ theme }) => theme.space[0]};\n  margin-bottom: ${({ theme }) => theme.space[10]};\n`\n\nconst DropdownContainer = styled(Stack)<{ 'data-grouped': boolean }>`\n  max-height: ${DROPDOWN_MAX_HEIGHT}px;\n  overflow-y: auto;\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: 0;\n  padding-right: ${({ theme }) => theme.space[2]};\n  padding-left: ${({ theme }) => theme.space[2]};\n  height: ${({ theme }) => theme.space[4]};\n  display: flex;\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: 0;\n`\nconst DropdownItem = styled.div<{\n  'aria-selected': boolean\n  'aria-disabled': boolean\n}>`\n  text-align:left;\n  border: none;\n  background-color: ${({ theme }) =>\n    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  border: 1px transparent solid;\n\n  &:hover, :focus {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n    color: ${({ theme }) => theme.colors.primary.text};\n    cursor: pointer;\n  }\n\n  &[aria-selected='true'] {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n  }\n\n  &[aria-disabled=\"true\"] {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n  }\n\n  &[aria-disabled=\"true\"]:hover, [aria-disabled=\"true\"]:focus {\n    background-color: ${({ theme }) =>\n      theme.colors.neutral.backgroundStrongDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textStrongDisabled};\n    cursor: not-allowed;\n    outline: none;\n  }\n`\n\nconst PopupFooter = styled.div`\n  width: 100%;\n  padding: ${({ theme }) => theme.space[1.5]} ${({ theme }) => theme.space[2]}\n    ${({ theme }) => theme.space[1.5]} ${({ theme }) => theme.space[2]};\n  box-shadow: ${({ theme }) => theme.shadows.dropdown};\n`\nconst StyledCheckbox = styled(Checkbox)`\n  width: 100%;\n  position: static;\n  text-align: left;\n  align-items: center;\n  pointer-events: none;\n` // pointer-events: none prevents any error when using the checkbox in a form since it is an unnamed input\n\nconst EmptyState = styled(Stack)`\n  padding: ${({ theme }) => theme.space[2]};\n`\nconst LoadMore = styled(Stack)`\n  padding: ${({ theme }) => theme.space[0.5]};\n`\n\nconst moveFocusDown = () => {\n  const options = document.querySelectorAll(\n    '#items > div[role=\"option\"]:not([disabled])',\n  )\n  const activeItem = document.activeElement\n  if (options) {\n    for (let i = 0; i < options?.length; i += 1) {\n      const listLength = options.length\n      if (activeItem === options[i] && activeItem !== options[listLength - 1]) {\n        ;(options[i + 1] as HTMLElement).focus()\n      }\n    }\n  }\n}\nconst moveFocusUp = () => {\n  const options = document.querySelectorAll(\n    '#items > div[role=\"option\"]:not([disabled])',\n  )\n  const activeItem = document.activeElement\n\n  if (options) {\n    for (let i = 0; i < options.length; i += 1) {\n      if (activeItem === options[i] && activeItem !== options[0]) {\n        ;(options[i - 1] as HTMLElement).focus()\n      }\n    }\n  }\n}\nconst handleKeyDownSelect = (event: KeyboardEvent<HTMLDivElement>) => {\n  if (event.key === 'ArrowDown') {\n    event.preventDefault()\n    moveFocusDown()\n  }\n\n  if (event.key === 'ArrowUp') {\n    event.preventDefault()\n    moveFocusUp()\n  }\n\n  if (event.key === ' ') {\n    // No scroll\n    event.preventDefault()\n  }\n}\n\nconst handleKeyDown = (\n  event: globalThis.KeyboardEvent,\n  ref: RefObject<HTMLDivElement | null>,\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].find(option =>\n        option.searchText\n          ? option.searchText.toLocaleLowerCase().startsWith(currentSearch)\n          : option.value.toLocaleLowerCase().startsWith(currentSearch),\n      )\n      if (closestOption) {\n        setDefaultSearch(closestOption.searchText ?? closestOption.value)\n      } else {\n        setDefaultSearch(null)\n      }\n    }\n  }\n}\nconst CreateDropdown = ({\n  isEmpty,\n  emptyState,\n  descriptionDirection,\n  loadMore,\n  optionalInfoPlacement,\n  defaultSearchValue,\n  isLoading,\n}: CreateDropdownProps) => {\n  const {\n    setIsDropdownVisible,\n    onChange,\n    options,\n    multiselect,\n    selectAll,\n    selectAllGroup,\n    displayedOptions,\n    setSelectedData,\n    selectedData,\n  } = useSelectInput()\n  const focusedItemRef = useRef<HTMLDivElement>(null)\n\n  useEffect(() => {\n    if (defaultSearchValue && focusedItemRef?.current) {\n      focusedItemRef.current.focus()\n    }\n  }, [defaultSearchValue])\n\n  if (isEmpty) {\n    return (\n      <EmptyState alignItems=\"center\" gap={2}>\n        {emptyState ?? (\n          <Text as=\"p\" variant=\"bodyStrong\">\n            No options\n          </Text>\n        )}\n      </EmptyState>\n    )\n  }\n\n  const handleClick = (clickedOption: OptionType, group?: string) => {\n    setSelectedData({ clickedOption, group, type: 'selectOption' })\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({ selectedGroup: group, type: 'selectGroup' })\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      data-grouped\n      id=\"select-dropdown\"\n      onKeyDown={handleKeyDownSelect}\n      role=\"listbox\"\n    >\n      {isLoading ? (\n        <Skeleton variant=\"block\" />\n      ) : (\n        <>\n          {selectAll && multiselect ? (\n            <Stack id=\"items\">\n              <DropdownItem\n                aria-disabled={false}\n                aria-label=\"select-all\"\n                aria-selected={selectedData.allSelected}\n                data-testid=\"select-all\"\n                id=\"select-all\"\n                onClick={selectAllOptions}\n                onKeyDown={event =>\n                  [' ', 'Enter'].includes(event.key) ? selectAllOptions() : null\n                }\n                role=\"option\"\n                tabIndex={0}\n              >\n                <StyledCheckbox\n                  checked={selectedData.allSelected}\n                  data-testid=\"select-all-checkbox\"\n                  disabled={false}\n                  onChange={selectAllOptions}\n                  tabIndex={-1}\n                  value=\"select-all\"\n                >\n                  <Stack direction=\"column\">\n                    <Text as=\"span\" placement=\"left\" variant=\"body\">\n                      {selectAll.label}\n                    </Text>\n                    <Text\n                      as=\"span\"\n                      placement=\"left\"\n                      prominence=\"weak\"\n                      sentiment=\"neutral\"\n                      variant=\"bodySmall\"\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 gap={0.25} key={group}>\n              {displayedOptions[group].length > 0 ? (\n                <DropdownGroupWrapper id={selectAllGroup ? 'items' : undefined}>\n                  {group ? (\n                    <DropdownGroup\n                      data-selectgroup={selectAllGroup}\n                      data-testid={`group-${index}`}\n                      key={group}\n                      onClick={() =>\n                        selectAllGroup ? handleSelectGroup(group) : null\n                      }\n                      onKeyDown={event => {\n                        if ([' ', 'Enter'].includes(event.key)) {\n                          event.preventDefault()\n                          handleSelectGroup(group)\n                        }\n                      }}\n                      role=\"group\"\n                      tabIndex={selectAllGroup ? 0 : -1}\n                      type=\"button\"\n                    >\n                      {selectAllGroup ? (\n                        <StyledCheckbox\n                          checked={selectedData.selectedGroups.includes(group)}\n                          data-testid=\"select-group\"\n                          disabled={false}\n                          onChange={() =>\n                            selectAllGroup ? handleSelectGroup(group) : null\n                          }\n                          tabIndex={-1}\n                          value={group}\n                        >\n                          <Text\n                            as=\"span\"\n                            placement=\"left\"\n                            sentiment=\"neutral\"\n                            variant=\"caption\"\n                          >\n                            {group.toUpperCase()}\n                          </Text>\n                        </StyledCheckbox>\n                      ) : (\n                        <Text\n                          as=\"span\"\n                          placement=\"left\"\n                          sentiment=\"neutral\"\n                          variant=\"caption\"\n                        >\n                          {group.toUpperCase()}\n                        </Text>\n                      )}\n                    </DropdownGroup>\n                  ) : null}\n                </DropdownGroupWrapper>\n              ) : null}\n              <Stack gap=\"0.25\" id=\"items\">\n                {displayedOptions[group].map((option, indexOption) => (\n                  <DropdownItem\n                    aria-disabled={!!option.disabled}\n                    aria-label={option.value}\n                    aria-selected={\n                      selectedData.selectedValues.includes(option.value) &&\n                      !option.disabled\n                    }\n                    data-testid={`option-${option.value}`}\n                    id={`option-${indexOption}`}\n                    key={option.value}\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                    role=\"option\"\n                    tabIndex={!option.disabled ? 0 : -1}\n                  >\n                    {multiselect ? (\n                      <StyledCheckbox\n                        checked={\n                          selectedData.selectedValues.includes(option.value) &&\n                          !option.disabled\n                        }\n                        disabled={option.disabled}\n                        onChange={() => {\n                          if (!option.disabled) {\n                            handleClick(option, group)\n                          }\n                        }}\n                        tabIndex={-1}\n                        value={option.value}\n                      >\n                        <DisplayOption\n                          descriptionDirection={descriptionDirection}\n                          option={option}\n                          optionalInfoPlacement={optionalInfoPlacement}\n                        />\n                      </StyledCheckbox>\n                    ) : (\n                      <DisplayOption\n                        descriptionDirection={descriptionDirection}\n                        option={option}\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      data-grouped={false}\n      gap={0.25}\n      id=\"select-dropdown\"\n      onKeyDown={handleKeyDownSelect}\n      role=\"listbox\"\n    >\n      {selectAll && multiselect ? (\n        <Stack gap={0.25} id=\"items\" tabIndex={-1}>\n          <DropdownItem\n            aria-disabled={false}\n            aria-label=\"select-all\"\n            aria-selected={selectedData.allSelected}\n            data-testid=\"select-all\"\n            onClick={selectAllOptions}\n            onKeyDown={event =>\n              [' ', 'Enter'].includes(event.key) ? selectAllOptions() : null\n            }\n            role=\"option\"\n            tabIndex={0}\n          >\n            <StyledCheckbox\n              checked={selectedData.allSelected}\n              data-testid=\"select-all-checkbox\"\n              disabled={false}\n              onChange={selectAllOptions}\n              tabIndex={-1}\n              value=\"select-all\"\n            >\n              <Stack direction=\"column\">\n                <Text as=\"span\" placement=\"left\" variant=\"body\">\n                  {selectAll.label}\n                </Text>\n                <Text\n                  as=\"span\"\n                  placement=\"left\"\n                  prominence=\"weak\"\n                  sentiment=\"neutral\"\n                  variant=\"bodySmall\"\n                >\n                  {selectAll.description}\n                </Text>\n              </Stack>\n            </StyledCheckbox>\n          </DropdownItem>\n        </Stack>\n      ) : null}\n      <Stack gap={0.25} id=\"items\">\n        {isLoading ? (\n          <Skeleton variant=\"block\" />\n        ) : (\n          displayedOptions.map((option, index) => (\n            <DropdownItem\n              aria-disabled={!!option.disabled}\n              aria-label={option.value}\n              aria-selected={\n                selectedData.selectedValues.includes(option.value) &&\n                !option.disabled\n              }\n              data-testid={`option-${option.value}`}\n              id={`option-${index}`}\n              key={option.value}\n              onClick={() => {\n                if (!option.disabled) {\n                  handleClick(option)\n                }\n              }}\n              onKeyDown={event =>\n                [' ', 'Enter'].includes(event.key) ? handleClick(option) : null\n              }\n              ref={\n                option.value === defaultSearchValue ||\n                option.searchText === defaultSearchValue\n                  ? focusedItemRef\n                  : null\n              }\n              role=\"option\"\n              tabIndex={!option.disabled ? 0 : -1}\n            >\n              {multiselect ? (\n                <StyledCheckbox\n                  checked={\n                    selectedData.selectedValues.includes(option.value) &&\n                    !option.disabled\n                  }\n                  disabled={option.disabled}\n                  onChange={() => {\n                    if (!option.disabled) {\n                      handleClick(option)\n                    }\n                  }}\n                  tabIndex={-1}\n                  value={option.value}\n                >\n                  <DisplayOption\n                    descriptionDirection={descriptionDirection}\n                    option={option}\n                    optionalInfoPlacement={optionalInfoPlacement}\n                  />\n                </StyledCheckbox>\n              ) : (\n                <DisplayOption\n                  descriptionDirection={descriptionDirection}\n                  option={option}\n                  optionalInfoPlacement={optionalInfoPlacement}\n                />\n              )}\n            </DropdownItem>\n          ))\n        )}\n        {loadMore ? <LoadMore>{loadMore}</LoadMore> : null}\n      </Stack>\n    </DropdownContainer>\n  )\n}\nexport const Dropdown = ({\n  children,\n  emptyState,\n  descriptionDirection,\n  searchable,\n  placeholder,\n  footer,\n  refSelect,\n  loadMore,\n  optionalInfoPlacement,\n  isLoading,\n  size,\n  dropdownAlign,\n  portalTarget,\n  id,\n}: DropdownProps) => {\n  const {\n    setIsDropdownVisible,\n    isDropdownVisible,\n    onSearch,\n    searchInput,\n    options,\n    displayedOptions,\n    numberOfOptions,\n  } = useSelectInput()\n  const theme = useTheme()\n  const [searchBarActive, setSearchBarActive] = useState(false)\n  const [defaultSearchValue, setDefaultSearch] = useState<string | null>(null)\n  const ref = useRef<HTMLDivElement>(null)\n  const [search, setSearch] = useState<string>('')\n  const [maxWidth, setWidth] = useState<string | number>()\n  const modalContext = useContext(ModalContext)\n\n  useEffect(() => {\n    if (refSelect.current && isDropdownVisible) {\n      const position =\n        refSelect.current.getBoundingClientRect().bottom +\n        DROPDOWN_MAX_HEIGHT +\n        Number(theme.sizing[INPUT_SIZE_HEIGHT[size]].replace('rem', '')) * 16 +\n        Number.parseInt(theme.space['5'], 10)\n      const overflow = position - window.innerHeight + 32\n      if (overflow > 0 && modalContext) {\n        const currentModal = modalContext.openedModals[0]\n        const modalElement = currentModal?.ref.current\n\n        if (modalElement) {\n          const parentElement = modalElement.parentNode as HTMLElement\n          if (parentElement) {\n            parentElement.scrollBy({\n              behavior: 'smooth',\n              top: overflow,\n            })\n          }\n        } else {\n          window.scrollBy({ behavior: 'smooth', top: overflow })\n        }\n      }\n    }\n    // oxlint-disable-next-line react/exhaustive-deps\n  }, [isDropdownVisible, refSelect, size, ref.current])\n\n  const resizeDropdown = useCallback(() => {\n    if (\n      refSelect.current &&\n      refSelect.current.getBoundingClientRect().width > 0\n    ) {\n      setWidth(refSelect.current.getBoundingClientRect().width)\n    }\n  }, [refSelect])\n\n  useEffect(() => {\n    resizeDropdown()\n\n    window.addEventListener('resize', resizeDropdown)\n\n    return () => window.removeEventListener('resize', resizeDropdown)\n  }, [resizeDropdown])\n\n  useEffect(() => {\n    if (!searchInput) {\n      onSearch(options)\n    }\n  }, [onSearch, options, searchInput])\n\n  useEffect(() => {\n    if (!isDropdownVisible) {\n      setDefaultSearch(null)\n      setSearch('')\n    }\n\n    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      if (!searchable) {\n        document.removeEventListener('keydown', event =>\n          handleKeyDown(\n            event,\n            ref,\n            options,\n            searchBarActive,\n            setSearch,\n            setDefaultSearch,\n            search,\n          ),\n        )\n      }\n    }\n  }, [\n    isDropdownVisible,\n    searchBarActive,\n    options,\n    onSearch,\n    search,\n    refSelect,\n    setDefaultSearch,\n    setIsDropdownVisible,\n    searchable,\n  ])\n\n  // No data is displayed (because of the search or because no data is provided)\n  // Set to true when noData by default\n  const isEmpty = useMemo(() => {\n    if (numberOfOptions === 0) {\n      return true\n    }\n    if (Array.isArray(displayedOptions)) {\n      return displayedOptions.length === 0\n    }\n    const groups = Object.keys(displayedOptions)\n    for (const group of groups) {\n      if (displayedOptions[group].length > 0) {\n        return false\n      }\n    }\n\n    return true\n  }, [displayedOptions, numberOfOptions])\n\n  const computedFooter = useMemo(() => {\n    if (footer && !isEmpty) {\n      if (typeof footer === 'function') {\n        return (\n          <PopupFooter>{footer(() => setIsDropdownVisible(false))}</PopupFooter>\n        )\n      }\n\n      return <PopupFooter>{footer}</PopupFooter>\n    }\n\n    return null\n  }, [isEmpty, footer, setIsDropdownVisible])\n\n  return (\n    <StyledPopup\n      align={dropdownAlign ?? 'start'}\n      containerFullWidth\n      debounceDelay={0}\n      disableAnimation\n      hasArrow={false}\n      hideOnClickOutside\n      id={id}\n      maxWidth={maxWidth ?? refSelect.current?.offsetWidth}\n      onClose={() => setIsDropdownVisible(false)}\n      placement=\"bottom\"\n      portalTarget={portalTarget}\n      ref={ref}\n      role=\"dialog\"\n      tabIndex={-1}\n      text={\n        <Stack>\n          {searchable && !isLoading && numberOfOptions >= 6 ? (\n            <SearchBarDropdown\n              displayedOptions={displayedOptions}\n              placeholder={placeholder}\n              setSearchBarActive={setSearchBarActive}\n            />\n          ) : null}\n          <CreateDropdown\n            defaultSearchValue={defaultSearchValue}\n            descriptionDirection={descriptionDirection}\n            emptyState={emptyState}\n            isEmpty={isEmpty}\n            isLoading={isLoading}\n            loadMore={loadMore}\n            optionalInfoPlacement={optionalInfoPlacement}\n          />\n          {computedFooter}\n        </Stack>\n      }\n      visible={isDropdownVisible}\n    >\n      {children}\n    </StyledPopup>\n  )\n}\n"]} */"));
|
|
37
37
|
const DropdownContainer = /* @__PURE__ */ _styled(Stack, process.env.NODE_ENV === "production" ? {
|
|
38
38
|
target: "e12qyldb7"
|
|
39
39
|
} : {
|
|
@@ -47,7 +47,7 @@ const DropdownContainer = /* @__PURE__ */ _styled(Stack, process.env.NODE_ENV ==
|
|
|
47
47
|
theme
|
|
48
48
|
}) => theme.space[0.5], ';&[data-grouped="true"]{padding-top:', ({
|
|
49
49
|
theme
|
|
50
|
-
}) => 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/SelectInput/Dropdown.tsx"],"names":[],"mappings":"AAsFoE","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInput/Dropdown.tsx","sourcesContent":["'use client'\n\nimport { useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport type {\n  ComponentProps,\n  Dispatch,\n  KeyboardEvent,\n  ReactNode,\n  RefObject,\n  SetStateAction,\n} from 'react'\nimport {\n  useCallback,\n  useContext,\n  useEffect,\n  useMemo,\n  useRef,\n  useState,\n} from 'react'\nimport { Checkbox } from '../Checkbox'\nimport { ModalContext } from '../Modal/ModalProvider'\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'\nimport { INPUT_SIZE_HEIGHT } from './types'\n\nconst DROPDOWN_MAX_HEIGHT = 256\n\nexport type DropdownProps = {\n  id?: string\n  children: ReactNode\n  emptyState: ReactNode\n  descriptionDirection: 'row' | 'column'\n  searchable: boolean\n  placeholder: string\n  footer?: ((closeDropdown: () => void) => ReactNode) | ReactNode\n  refSelect: RefObject<HTMLDivElement | null>\n  loadMore?: ReactNode\n  optionalInfoPlacement: 'left' | 'right'\n  isLoading?: boolean\n  size: 'small' | 'medium' | 'large'\n  dropdownAlign?: ComponentProps<typeof Popup>['align']\n  portalTarget?: ComponentProps<typeof Popup>['portalTarget']\n}\n\nexport type CreateDropdownProps = {\n  isEmpty: boolean\n  emptyState: ReactNode\n  descriptionDirection: 'row' | 'column'\n  loadMore?: ReactNode\n  optionalInfoPlacement: 'left' | 'right'\n  defaultSearchValue: string | null\n  isLoading?: boolean\n}\n\nconst NON_SEARCHABLE_KEYS = [\n  'Tab',\n  ' ',\n  'Enter',\n  'CapsLock',\n  'Shift',\n  'ArrowDown',\n  'ArrowUp',\n  'ArrowLeft',\n  'ArrowRight',\n  'Escape',\n]\n\nconst StyledPopup = styled(Popup)`\n  width: 100%;\n  min-width: 320px;\n  background-color: ${({ theme }) =>\n    theme.colors.other.elevation.background.raised};\n  color: ${({ theme }) => theme.colors.neutral.text};\n  box-shadow: ${({ theme }) =>\n    `${theme.shadows.raised[0]}, ${theme.shadows.raised[1]}`};\n  padding: ${({ theme }) => theme.space[0]};\n  margin-bottom: ${({ theme }) => theme.space[10]};\n`\n\nconst DropdownContainer = styled(Stack)<{ 'data-grouped': boolean }>`\n  max-height: ${DROPDOWN_MAX_HEIGHT}px;\n  overflow-y: auto;\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: 0;\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: 0;\n`\nconst DropdownItem = styled.div<{\n  'aria-selected': boolean\n  'aria-disabled': boolean\n}>`\n  text-align:left;\n  border: none;\n  background-color: ${({ theme }) =>\n    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  border: 1px transparent solid;\n\n  &:hover, :focus {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n    color: ${({ theme }) => theme.colors.primary.text};\n    cursor: pointer;\n  }\n\n  &[aria-selected='true'] {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n  }\n\n  &[aria-disabled=\"true\"] {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n  }\n\n  &[aria-disabled=\"true\"]:hover, [aria-disabled=\"true\"]:focus {\n    background-color: ${({ theme }) =>\n      theme.colors.neutral.backgroundStrongDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textStrongDisabled};\n    cursor: not-allowed;\n    outline: none;\n  }\n`\n\nconst PopupFooter = styled.div`\n  width: 100%;\n  padding: ${({ theme }) => theme.space[1.5]} ${({ theme }) => theme.space[2]}\n    ${({ theme }) => theme.space[1.5]} ${({ theme }) => theme.space[2]};\n  box-shadow: ${({ theme }) => theme.shadows.dropdown};\n`\nconst StyledCheckbox = styled(Checkbox)`\n  width: 100%;\n  position: static;\n  text-align: left;\n  align-items: center;\n  pointer-events: none;\n` // pointer-events: none prevents any error when using the checkbox in a form since it is an unnamed input\n\nconst EmptyState = styled(Stack)`\n  padding: ${({ theme }) => theme.space[2]};\n`\nconst LoadMore = styled(Stack)`\n  padding: ${({ theme }) => theme.space[0.5]};\n`\n\nconst moveFocusDown = () => {\n  const options = document.querySelectorAll(\n    '#items > div[role=\"option\"]:not([disabled])',\n  )\n  const activeItem = document.activeElement\n  if (options) {\n    for (let i = 0; i < options?.length; i += 1) {\n      const listLength = options.length\n      if (activeItem === options[i] && activeItem !== options[listLength - 1]) {\n        ;(options[i + 1] as HTMLElement).focus()\n      }\n    }\n  }\n}\nconst moveFocusUp = () => {\n  const options = document.querySelectorAll(\n    '#items > div[role=\"option\"]:not([disabled])',\n  )\n  const activeItem = document.activeElement\n\n  if (options) {\n    for (let i = 0; i < options.length; i += 1) {\n      if (activeItem === options[i] && activeItem !== options[0]) {\n        ;(options[i - 1] as HTMLElement).focus()\n      }\n    }\n  }\n}\nconst handleKeyDownSelect = (event: KeyboardEvent<HTMLDivElement>) => {\n  if (event.key === 'ArrowDown') {\n    event.preventDefault()\n    moveFocusDown()\n  }\n\n  if (event.key === 'ArrowUp') {\n    event.preventDefault()\n    moveFocusUp()\n  }\n\n  if (event.key === ' ') {\n    // No scroll\n    event.preventDefault()\n  }\n}\n\nconst handleKeyDown = (\n  event: globalThis.KeyboardEvent,\n  ref: RefObject<HTMLDivElement | null>,\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].find(option =>\n        option.searchText\n          ? option.searchText.toLocaleLowerCase().startsWith(currentSearch)\n          : option.value.toLocaleLowerCase().startsWith(currentSearch),\n      )\n      if (closestOption) {\n        setDefaultSearch(closestOption.searchText ?? closestOption.value)\n      } else {\n        setDefaultSearch(null)\n      }\n    }\n  }\n}\nconst CreateDropdown = ({\n  isEmpty,\n  emptyState,\n  descriptionDirection,\n  loadMore,\n  optionalInfoPlacement,\n  defaultSearchValue,\n  isLoading,\n}: CreateDropdownProps) => {\n  const {\n    setIsDropdownVisible,\n    onChange,\n    options,\n    multiselect,\n    selectAll,\n    selectAllGroup,\n    displayedOptions,\n    setSelectedData,\n    selectedData,\n  } = useSelectInput()\n  const focusedItemRef = useRef<HTMLDivElement>(null)\n\n  useEffect(() => {\n    if (defaultSearchValue && focusedItemRef?.current) {\n      focusedItemRef.current.focus()\n    }\n  }, [defaultSearchValue])\n\n  if (isEmpty) {\n    return (\n      <EmptyState alignItems=\"center\" gap={2}>\n        {emptyState ?? (\n          <Text as=\"p\" variant=\"bodyStrong\">\n            No options\n          </Text>\n        )}\n      </EmptyState>\n    )\n  }\n\n  const handleClick = (clickedOption: OptionType, group?: string) => {\n    setSelectedData({ clickedOption, group, type: 'selectOption' })\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({ selectedGroup: group, type: 'selectGroup' })\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      data-grouped\n      id=\"select-dropdown\"\n      onKeyDown={handleKeyDownSelect}\n      role=\"listbox\"\n    >\n      {isLoading ? (\n        <Skeleton variant=\"block\" />\n      ) : (\n        <>\n          {selectAll && multiselect ? (\n            <Stack id=\"items\">\n              <DropdownItem\n                aria-disabled={false}\n                aria-label=\"select-all\"\n                aria-selected={selectedData.allSelected}\n                data-testid=\"select-all\"\n                id=\"select-all\"\n                onClick={selectAllOptions}\n                onKeyDown={event =>\n                  [' ', 'Enter'].includes(event.key) ? selectAllOptions() : null\n                }\n                role=\"option\"\n                tabIndex={0}\n              >\n                <StyledCheckbox\n                  checked={selectedData.allSelected}\n                  data-testid=\"select-all-checkbox\"\n                  disabled={false}\n                  onChange={selectAllOptions}\n                  tabIndex={-1}\n                  value=\"select-all\"\n                >\n                  <Stack direction=\"column\">\n                    <Text as=\"span\" placement=\"left\" variant=\"body\">\n                      {selectAll.label}\n                    </Text>\n                    <Text\n                      as=\"span\"\n                      placement=\"left\"\n                      prominence=\"weak\"\n                      sentiment=\"neutral\"\n                      variant=\"bodySmall\"\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 gap={0.25} key={group}>\n              {displayedOptions[group].length > 0 ? (\n                <DropdownGroupWrapper id={selectAllGroup ? 'items' : undefined}>\n                  {group ? (\n                    <DropdownGroup\n                      data-selectgroup={selectAllGroup}\n                      data-testid={`group-${index}`}\n                      key={group}\n                      onClick={() =>\n                        selectAllGroup ? handleSelectGroup(group) : null\n                      }\n                      onKeyDown={event => {\n                        if ([' ', 'Enter'].includes(event.key)) {\n                          event.preventDefault()\n                          handleSelectGroup(group)\n                        }\n                      }}\n                      role=\"group\"\n                      tabIndex={selectAllGroup ? 0 : -1}\n                      type=\"button\"\n                    >\n                      {selectAllGroup ? (\n                        <StyledCheckbox\n                          checked={selectedData.selectedGroups.includes(group)}\n                          data-testid=\"select-group\"\n                          disabled={false}\n                          onChange={() =>\n                            selectAllGroup ? handleSelectGroup(group) : null\n                          }\n                          tabIndex={-1}\n                          value={group}\n                        >\n                          <Text\n                            as=\"span\"\n                            placement=\"left\"\n                            sentiment=\"neutral\"\n                            variant=\"caption\"\n                          >\n                            {group.toUpperCase()}\n                          </Text>\n                        </StyledCheckbox>\n                      ) : (\n                        <Text\n                          as=\"span\"\n                          placement=\"left\"\n                          sentiment=\"neutral\"\n                          variant=\"caption\"\n                        >\n                          {group.toUpperCase()}\n                        </Text>\n                      )}\n                    </DropdownGroup>\n                  ) : null}\n                </DropdownGroupWrapper>\n              ) : null}\n              <Stack gap=\"0.25\" id=\"items\">\n                {displayedOptions[group].map((option, indexOption) => (\n                  <DropdownItem\n                    aria-disabled={!!option.disabled}\n                    aria-label={option.value}\n                    aria-selected={\n                      selectedData.selectedValues.includes(option.value) &&\n                      !option.disabled\n                    }\n                    data-testid={`option-${option.value}`}\n                    id={`option-${indexOption}`}\n                    key={option.value}\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                    role=\"option\"\n                    tabIndex={!option.disabled ? 0 : -1}\n                  >\n                    {multiselect ? (\n                      <StyledCheckbox\n                        checked={\n                          selectedData.selectedValues.includes(option.value) &&\n                          !option.disabled\n                        }\n                        disabled={option.disabled}\n                        onChange={() => {\n                          if (!option.disabled) {\n                            handleClick(option, group)\n                          }\n                        }}\n                        tabIndex={-1}\n                        value={option.value}\n                      >\n                        <DisplayOption\n                          descriptionDirection={descriptionDirection}\n                          option={option}\n                          optionalInfoPlacement={optionalInfoPlacement}\n                        />\n                      </StyledCheckbox>\n                    ) : (\n                      <DisplayOption\n                        descriptionDirection={descriptionDirection}\n                        option={option}\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      data-grouped={false}\n      gap={0.25}\n      id=\"select-dropdown\"\n      onKeyDown={handleKeyDownSelect}\n      role=\"listbox\"\n    >\n      {selectAll && multiselect ? (\n        <Stack gap={0.25} id=\"items\" tabIndex={-1}>\n          <DropdownItem\n            aria-disabled={false}\n            aria-label=\"select-all\"\n            aria-selected={selectedData.allSelected}\n            data-testid=\"select-all\"\n            onClick={selectAllOptions}\n            onKeyDown={event =>\n              [' ', 'Enter'].includes(event.key) ? selectAllOptions() : null\n            }\n            role=\"option\"\n            tabIndex={0}\n          >\n            <StyledCheckbox\n              checked={selectedData.allSelected}\n              data-testid=\"select-all-checkbox\"\n              disabled={false}\n              onChange={selectAllOptions}\n              tabIndex={-1}\n              value=\"select-all\"\n            >\n              <Stack direction=\"column\">\n                <Text as=\"span\" placement=\"left\" variant=\"body\">\n                  {selectAll.label}\n                </Text>\n                <Text\n                  as=\"span\"\n                  placement=\"left\"\n                  prominence=\"weak\"\n                  sentiment=\"neutral\"\n                  variant=\"bodySmall\"\n                >\n                  {selectAll.description}\n                </Text>\n              </Stack>\n            </StyledCheckbox>\n          </DropdownItem>\n        </Stack>\n      ) : null}\n      <Stack gap={0.25} id=\"items\">\n        {isLoading ? (\n          <Skeleton variant=\"block\" />\n        ) : (\n          displayedOptions.map((option, index) => (\n            <DropdownItem\n              aria-disabled={!!option.disabled}\n              aria-label={option.value}\n              aria-selected={\n                selectedData.selectedValues.includes(option.value) &&\n                !option.disabled\n              }\n              data-testid={`option-${option.value}`}\n              id={`option-${index}`}\n              key={option.value}\n              onClick={() => {\n                if (!option.disabled) {\n                  handleClick(option)\n                }\n              }}\n              onKeyDown={event =>\n                [' ', 'Enter'].includes(event.key) ? handleClick(option) : null\n              }\n              ref={\n                option.value === defaultSearchValue ||\n                option.searchText === defaultSearchValue\n                  ? focusedItemRef\n                  : null\n              }\n              role=\"option\"\n              tabIndex={!option.disabled ? 0 : -1}\n            >\n              {multiselect ? (\n                <StyledCheckbox\n                  checked={\n                    selectedData.selectedValues.includes(option.value) &&\n                    !option.disabled\n                  }\n                  disabled={option.disabled}\n                  onChange={() => {\n                    if (!option.disabled) {\n                      handleClick(option)\n                    }\n                  }}\n                  tabIndex={-1}\n                  value={option.value}\n                >\n                  <DisplayOption\n                    descriptionDirection={descriptionDirection}\n                    option={option}\n                    optionalInfoPlacement={optionalInfoPlacement}\n                  />\n                </StyledCheckbox>\n              ) : (\n                <DisplayOption\n                  descriptionDirection={descriptionDirection}\n                  option={option}\n                  optionalInfoPlacement={optionalInfoPlacement}\n                />\n              )}\n            </DropdownItem>\n          ))\n        )}\n        {loadMore ? <LoadMore>{loadMore}</LoadMore> : null}\n      </Stack>\n    </DropdownContainer>\n  )\n}\nexport const Dropdown = ({\n  children,\n  emptyState,\n  descriptionDirection,\n  searchable,\n  placeholder,\n  footer,\n  refSelect,\n  loadMore,\n  optionalInfoPlacement,\n  isLoading,\n  size,\n  dropdownAlign,\n  portalTarget,\n  id,\n}: DropdownProps) => {\n  const {\n    setIsDropdownVisible,\n    isDropdownVisible,\n    onSearch,\n    searchInput,\n    options,\n    displayedOptions,\n    numberOfOptions,\n  } = useSelectInput()\n  const theme = useTheme()\n  const [searchBarActive, setSearchBarActive] = useState(false)\n  const [defaultSearchValue, setDefaultSearch] = useState<string | null>(null)\n  const ref = useRef<HTMLDivElement>(null)\n  const [search, setSearch] = useState<string>('')\n  const [maxWidth, setWidth] = useState<string | number>()\n  const modalContext = useContext(ModalContext)\n\n  useEffect(() => {\n    if (refSelect.current && isDropdownVisible) {\n      const position =\n        refSelect.current.getBoundingClientRect().bottom +\n        DROPDOWN_MAX_HEIGHT +\n        Number(theme.sizing[INPUT_SIZE_HEIGHT[size]].replace('rem', '')) * 16 +\n        Number.parseInt(theme.space['5'], 10)\n      const overflow = position - window.innerHeight + 32\n      if (overflow > 0 && modalContext) {\n        const currentModal = modalContext.openedModals[0]\n        const modalElement = currentModal?.ref.current\n\n        if (modalElement) {\n          const parentElement = modalElement.parentNode as HTMLElement\n          if (parentElement) {\n            parentElement.scrollBy({\n              behavior: 'smooth',\n              top: overflow,\n            })\n          }\n        } else {\n          window.scrollBy({ behavior: 'smooth', top: overflow })\n        }\n      }\n    }\n    // oxlint-disable-next-line react/exhaustive-deps\n  }, [isDropdownVisible, refSelect, size, ref.current])\n\n  const resizeDropdown = useCallback(() => {\n    if (\n      refSelect.current &&\n      refSelect.current.getBoundingClientRect().width > 0\n    ) {\n      setWidth(refSelect.current.getBoundingClientRect().width)\n    }\n  }, [refSelect])\n\n  useEffect(() => {\n    resizeDropdown()\n\n    window.addEventListener('resize', resizeDropdown)\n\n    return () => window.removeEventListener('resize', resizeDropdown)\n  }, [resizeDropdown])\n\n  useEffect(() => {\n    if (!searchInput) {\n      onSearch(options)\n    }\n  }, [onSearch, options, searchInput])\n\n  useEffect(() => {\n    if (!isDropdownVisible) {\n      setDefaultSearch(null)\n      setSearch('')\n    }\n\n    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      if (!searchable) {\n        document.removeEventListener('keydown', event =>\n          handleKeyDown(\n            event,\n            ref,\n            options,\n            searchBarActive,\n            setSearch,\n            setDefaultSearch,\n            search,\n          ),\n        )\n      }\n    }\n  }, [\n    isDropdownVisible,\n    searchBarActive,\n    options,\n    onSearch,\n    search,\n    refSelect,\n    setDefaultSearch,\n    setIsDropdownVisible,\n    searchable,\n  ])\n\n  // No data is displayed (because of the search or because no data is provided)\n  // Set to true when noData by default\n  const isEmpty = useMemo(() => {\n    if (numberOfOptions === 0) {\n      return true\n    }\n    if (Array.isArray(displayedOptions)) {\n      return displayedOptions.length === 0\n    }\n    const groups = Object.keys(displayedOptions)\n    for (const group of groups) {\n      if (displayedOptions[group].length > 0) {\n        return false\n      }\n    }\n\n    return true\n  }, [displayedOptions, numberOfOptions])\n\n  const computedFooter = useMemo(() => {\n    if (footer && !isEmpty) {\n      if (typeof footer === 'function') {\n        return (\n          <PopupFooter>{footer(() => setIsDropdownVisible(false))}</PopupFooter>\n        )\n      }\n\n      return <PopupFooter>{footer}</PopupFooter>\n    }\n\n    return null\n  }, [isEmpty, footer, setIsDropdownVisible])\n\n  return (\n    <StyledPopup\n      align={dropdownAlign ?? 'start'}\n      containerFullWidth\n      debounceDelay={0}\n      disableAnimation\n      hasArrow={false}\n      hideOnClickOutside\n      id={id}\n      maxWidth={maxWidth ?? refSelect.current?.offsetWidth}\n      onClose={() => setIsDropdownVisible(false)}\n      placement=\"bottom\"\n      portalTarget={portalTarget}\n      ref={ref}\n      role=\"dialog\"\n      tabIndex={-1}\n      text={\n        <Stack>\n          {searchable && !isLoading && numberOfOptions >= 6 ? (\n            <SearchBarDropdown\n              displayedOptions={displayedOptions}\n              placeholder={placeholder}\n              setSearchBarActive={setSearchBarActive}\n            />\n          ) : null}\n          <CreateDropdown\n            defaultSearchValue={defaultSearchValue}\n            descriptionDirection={descriptionDirection}\n            emptyState={emptyState}\n            isEmpty={isEmpty}\n            isLoading={isLoading}\n            loadMore={loadMore}\n            optionalInfoPlacement={optionalInfoPlacement}\n          />\n          {computedFooter}\n        </Stack>\n      }\n      visible={isDropdownVisible}\n    >\n      {children}\n    </StyledPopup>\n  )\n}\n"]} */"));
|
|
50
|
+
}) => 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/SelectInput/Dropdown.tsx"],"names":[],"mappings":"AAsFoE","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInput/Dropdown.tsx","sourcesContent":["'use client'\n\nimport { useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport type {\n  ComponentProps,\n  Dispatch,\n  KeyboardEvent,\n  ReactNode,\n  RefObject,\n  SetStateAction,\n} from 'react'\nimport {\n  useCallback,\n  useContext,\n  useEffect,\n  useMemo,\n  useRef,\n  useState,\n} from 'react'\nimport { Checkbox } from '../Checkbox'\nimport { ModalContext } from '../Modal/ModalProvider'\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'\nimport { INPUT_SIZE_HEIGHT } from './types'\n\nconst DROPDOWN_MAX_HEIGHT = 256\n\nexport type DropdownProps = {\n  id?: string\n  children: ReactNode\n  emptyState: ReactNode\n  descriptionDirection: 'row' | 'column'\n  searchable: boolean\n  placeholder: string\n  footer?: ((closeDropdown: () => void) => ReactNode) | ReactNode\n  refSelect: RefObject<HTMLDivElement | null>\n  loadMore?: ReactNode\n  optionalInfoPlacement: 'left' | 'right'\n  isLoading?: boolean\n  size: 'small' | 'medium' | 'large'\n  dropdownAlign?: ComponentProps<typeof Popup>['align']\n  portalTarget?: ComponentProps<typeof Popup>['portalTarget']\n}\n\nexport type CreateDropdownProps = {\n  isEmpty: boolean\n  emptyState: ReactNode\n  descriptionDirection: 'row' | 'column'\n  loadMore?: ReactNode\n  optionalInfoPlacement: 'left' | 'right'\n  defaultSearchValue: string | null\n  isLoading?: boolean\n}\n\nconst NON_SEARCHABLE_KEYS = [\n  'Tab',\n  ' ',\n  'Enter',\n  'CapsLock',\n  'Shift',\n  'ArrowDown',\n  'ArrowUp',\n  'ArrowLeft',\n  'ArrowRight',\n  'Escape',\n]\n\nconst StyledPopup = styled(Popup)`\n  width: 100%;\n  min-width: 320px;\n  background-color: ${({ theme }) =>\n    theme.colors.other.elevation.background.raised};\n  color: ${({ theme }) => theme.colors.neutral.text};\n  box-shadow: ${({ theme }) =>\n    `${theme.shadows.raised[0]}, ${theme.shadows.raised[1]}`};\n  padding: ${({ theme }) => theme.space[0]};\n  margin-bottom: ${({ theme }) => theme.space[10]};\n`\n\nconst DropdownContainer = styled(Stack)<{ 'data-grouped': boolean }>`\n  max-height: ${DROPDOWN_MAX_HEIGHT}px;\n  overflow-y: auto;\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: 0;\n  padding-right: ${({ theme }) => theme.space[2]};\n  padding-left: ${({ theme }) => theme.space[2]};\n  height: ${({ theme }) => theme.space[4]};\n  display: flex;\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: 0;\n`\nconst DropdownItem = styled.div<{\n  'aria-selected': boolean\n  'aria-disabled': boolean\n}>`\n  text-align:left;\n  border: none;\n  background-color: ${({ theme }) =>\n    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  border: 1px transparent solid;\n\n  &:hover, :focus {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n    color: ${({ theme }) => theme.colors.primary.text};\n    cursor: pointer;\n  }\n\n  &[aria-selected='true'] {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n  }\n\n  &[aria-disabled=\"true\"] {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n  }\n\n  &[aria-disabled=\"true\"]:hover, [aria-disabled=\"true\"]:focus {\n    background-color: ${({ theme }) =>\n      theme.colors.neutral.backgroundStrongDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textStrongDisabled};\n    cursor: not-allowed;\n    outline: none;\n  }\n`\n\nconst PopupFooter = styled.div`\n  width: 100%;\n  padding: ${({ theme }) => theme.space[1.5]} ${({ theme }) => theme.space[2]}\n    ${({ theme }) => theme.space[1.5]} ${({ theme }) => theme.space[2]};\n  box-shadow: ${({ theme }) => theme.shadows.dropdown};\n`\nconst StyledCheckbox = styled(Checkbox)`\n  width: 100%;\n  position: static;\n  text-align: left;\n  align-items: center;\n  pointer-events: none;\n` // pointer-events: none prevents any error when using the checkbox in a form since it is an unnamed input\n\nconst EmptyState = styled(Stack)`\n  padding: ${({ theme }) => theme.space[2]};\n`\nconst LoadMore = styled(Stack)`\n  padding: ${({ theme }) => theme.space[0.5]};\n`\n\nconst moveFocusDown = () => {\n  const options = document.querySelectorAll(\n    '#items > div[role=\"option\"]:not([disabled])',\n  )\n  const activeItem = document.activeElement\n  if (options) {\n    for (let i = 0; i < options?.length; i += 1) {\n      const listLength = options.length\n      if (activeItem === options[i] && activeItem !== options[listLength - 1]) {\n        ;(options[i + 1] as HTMLElement).focus()\n      }\n    }\n  }\n}\nconst moveFocusUp = () => {\n  const options = document.querySelectorAll(\n    '#items > div[role=\"option\"]:not([disabled])',\n  )\n  const activeItem = document.activeElement\n\n  if (options) {\n    for (let i = 0; i < options.length; i += 1) {\n      if (activeItem === options[i] && activeItem !== options[0]) {\n        ;(options[i - 1] as HTMLElement).focus()\n      }\n    }\n  }\n}\nconst handleKeyDownSelect = (event: KeyboardEvent<HTMLDivElement>) => {\n  if (event.key === 'ArrowDown') {\n    event.preventDefault()\n    moveFocusDown()\n  }\n\n  if (event.key === 'ArrowUp') {\n    event.preventDefault()\n    moveFocusUp()\n  }\n\n  if (event.key === ' ') {\n    // No scroll\n    event.preventDefault()\n  }\n}\n\nconst handleKeyDown = (\n  event: globalThis.KeyboardEvent,\n  ref: RefObject<HTMLDivElement | null>,\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].find(option =>\n        option.searchText\n          ? option.searchText.toLocaleLowerCase().startsWith(currentSearch)\n          : option.value.toLocaleLowerCase().startsWith(currentSearch),\n      )\n      if (closestOption) {\n        setDefaultSearch(closestOption.searchText ?? closestOption.value)\n      } else {\n        setDefaultSearch(null)\n      }\n    }\n  }\n}\nconst CreateDropdown = ({\n  isEmpty,\n  emptyState,\n  descriptionDirection,\n  loadMore,\n  optionalInfoPlacement,\n  defaultSearchValue,\n  isLoading,\n}: CreateDropdownProps) => {\n  const {\n    setIsDropdownVisible,\n    onChange,\n    options,\n    multiselect,\n    selectAll,\n    selectAllGroup,\n    displayedOptions,\n    setSelectedData,\n    selectedData,\n  } = useSelectInput()\n  const focusedItemRef = useRef<HTMLDivElement>(null)\n\n  useEffect(() => {\n    if (defaultSearchValue && focusedItemRef?.current) {\n      focusedItemRef.current.focus()\n    }\n  }, [defaultSearchValue])\n\n  if (isEmpty) {\n    return (\n      <EmptyState alignItems=\"center\" gap={2}>\n        {emptyState ?? (\n          <Text as=\"p\" variant=\"bodyStrong\">\n            No options\n          </Text>\n        )}\n      </EmptyState>\n    )\n  }\n\n  const handleClick = (clickedOption: OptionType, group?: string) => {\n    setSelectedData({ clickedOption, group, type: 'selectOption' })\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({ selectedGroup: group, type: 'selectGroup' })\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      data-grouped\n      id=\"select-dropdown\"\n      onKeyDown={handleKeyDownSelect}\n      role=\"listbox\"\n    >\n      {isLoading ? (\n        <Skeleton variant=\"block\" />\n      ) : (\n        <>\n          {selectAll && multiselect ? (\n            <Stack id=\"items\">\n              <DropdownItem\n                aria-disabled={false}\n                aria-label=\"select-all\"\n                aria-selected={selectedData.allSelected}\n                data-testid=\"select-all\"\n                id=\"select-all\"\n                onClick={selectAllOptions}\n                onKeyDown={event =>\n                  [' ', 'Enter'].includes(event.key) ? selectAllOptions() : null\n                }\n                role=\"option\"\n                tabIndex={0}\n              >\n                <StyledCheckbox\n                  checked={selectedData.allSelected}\n                  data-testid=\"select-all-checkbox\"\n                  disabled={false}\n                  onChange={selectAllOptions}\n                  tabIndex={-1}\n                  value=\"select-all\"\n                >\n                  <Stack direction=\"column\">\n                    <Text as=\"span\" placement=\"left\" variant=\"body\">\n                      {selectAll.label}\n                    </Text>\n                    <Text\n                      as=\"span\"\n                      placement=\"left\"\n                      prominence=\"weak\"\n                      sentiment=\"neutral\"\n                      variant=\"bodySmall\"\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 gap={0.25} key={group}>\n              {displayedOptions[group].length > 0 ? (\n                <DropdownGroupWrapper id={selectAllGroup ? 'items' : undefined}>\n                  {group ? (\n                    <DropdownGroup\n                      data-selectgroup={selectAllGroup}\n                      data-testid={`group-${index}`}\n                      key={group}\n                      onClick={() =>\n                        selectAllGroup ? handleSelectGroup(group) : null\n                      }\n                      onKeyDown={event => {\n                        if ([' ', 'Enter'].includes(event.key)) {\n                          event.preventDefault()\n                          handleSelectGroup(group)\n                        }\n                      }}\n                      role=\"group\"\n                      tabIndex={selectAllGroup ? 0 : -1}\n                      type=\"button\"\n                    >\n                      {selectAllGroup ? (\n                        <StyledCheckbox\n                          checked={selectedData.selectedGroups.includes(group)}\n                          data-testid=\"select-group\"\n                          disabled={false}\n                          onChange={() =>\n                            selectAllGroup ? handleSelectGroup(group) : null\n                          }\n                          tabIndex={-1}\n                          value={group}\n                        >\n                          <Text\n                            as=\"span\"\n                            placement=\"left\"\n                            sentiment=\"neutral\"\n                            variant=\"caption\"\n                          >\n                            {group.toUpperCase()}\n                          </Text>\n                        </StyledCheckbox>\n                      ) : (\n                        <Text\n                          as=\"span\"\n                          placement=\"left\"\n                          sentiment=\"neutral\"\n                          variant=\"caption\"\n                        >\n                          {group.toUpperCase()}\n                        </Text>\n                      )}\n                    </DropdownGroup>\n                  ) : null}\n                </DropdownGroupWrapper>\n              ) : null}\n              <Stack gap=\"0.25\" id=\"items\">\n                {displayedOptions[group].map((option, indexOption) => (\n                  <DropdownItem\n                    aria-disabled={!!option.disabled}\n                    aria-label={option.value}\n                    aria-selected={\n                      selectedData.selectedValues.includes(option.value) &&\n                      !option.disabled\n                    }\n                    data-testid={`option-${option.value}`}\n                    id={`option-${indexOption}`}\n                    key={option.value}\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                    role=\"option\"\n                    tabIndex={!option.disabled ? 0 : -1}\n                  >\n                    {multiselect ? (\n                      <StyledCheckbox\n                        checked={\n                          selectedData.selectedValues.includes(option.value) &&\n                          !option.disabled\n                        }\n                        disabled={option.disabled}\n                        onChange={() => {\n                          if (!option.disabled) {\n                            handleClick(option, group)\n                          }\n                        }}\n                        tabIndex={-1}\n                        value={option.value}\n                      >\n                        <DisplayOption\n                          descriptionDirection={descriptionDirection}\n                          option={option}\n                          optionalInfoPlacement={optionalInfoPlacement}\n                        />\n                      </StyledCheckbox>\n                    ) : (\n                      <DisplayOption\n                        descriptionDirection={descriptionDirection}\n                        option={option}\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      data-grouped={false}\n      gap={0.25}\n      id=\"select-dropdown\"\n      onKeyDown={handleKeyDownSelect}\n      role=\"listbox\"\n    >\n      {selectAll && multiselect ? (\n        <Stack gap={0.25} id=\"items\" tabIndex={-1}>\n          <DropdownItem\n            aria-disabled={false}\n            aria-label=\"select-all\"\n            aria-selected={selectedData.allSelected}\n            data-testid=\"select-all\"\n            onClick={selectAllOptions}\n            onKeyDown={event =>\n              [' ', 'Enter'].includes(event.key) ? selectAllOptions() : null\n            }\n            role=\"option\"\n            tabIndex={0}\n          >\n            <StyledCheckbox\n              checked={selectedData.allSelected}\n              data-testid=\"select-all-checkbox\"\n              disabled={false}\n              onChange={selectAllOptions}\n              tabIndex={-1}\n              value=\"select-all\"\n            >\n              <Stack direction=\"column\">\n                <Text as=\"span\" placement=\"left\" variant=\"body\">\n                  {selectAll.label}\n                </Text>\n                <Text\n                  as=\"span\"\n                  placement=\"left\"\n                  prominence=\"weak\"\n                  sentiment=\"neutral\"\n                  variant=\"bodySmall\"\n                >\n                  {selectAll.description}\n                </Text>\n              </Stack>\n            </StyledCheckbox>\n          </DropdownItem>\n        </Stack>\n      ) : null}\n      <Stack gap={0.25} id=\"items\">\n        {isLoading ? (\n          <Skeleton variant=\"block\" />\n        ) : (\n          displayedOptions.map((option, index) => (\n            <DropdownItem\n              aria-disabled={!!option.disabled}\n              aria-label={option.value}\n              aria-selected={\n                selectedData.selectedValues.includes(option.value) &&\n                !option.disabled\n              }\n              data-testid={`option-${option.value}`}\n              id={`option-${index}`}\n              key={option.value}\n              onClick={() => {\n                if (!option.disabled) {\n                  handleClick(option)\n                }\n              }}\n              onKeyDown={event =>\n                [' ', 'Enter'].includes(event.key) ? handleClick(option) : null\n              }\n              ref={\n                option.value === defaultSearchValue ||\n                option.searchText === defaultSearchValue\n                  ? focusedItemRef\n                  : null\n              }\n              role=\"option\"\n              tabIndex={!option.disabled ? 0 : -1}\n            >\n              {multiselect ? (\n                <StyledCheckbox\n                  checked={\n                    selectedData.selectedValues.includes(option.value) &&\n                    !option.disabled\n                  }\n                  disabled={option.disabled}\n                  onChange={() => {\n                    if (!option.disabled) {\n                      handleClick(option)\n                    }\n                  }}\n                  tabIndex={-1}\n                  value={option.value}\n                >\n                  <DisplayOption\n                    descriptionDirection={descriptionDirection}\n                    option={option}\n                    optionalInfoPlacement={optionalInfoPlacement}\n                  />\n                </StyledCheckbox>\n              ) : (\n                <DisplayOption\n                  descriptionDirection={descriptionDirection}\n                  option={option}\n                  optionalInfoPlacement={optionalInfoPlacement}\n                />\n              )}\n            </DropdownItem>\n          ))\n        )}\n        {loadMore ? <LoadMore>{loadMore}</LoadMore> : null}\n      </Stack>\n    </DropdownContainer>\n  )\n}\nexport const Dropdown = ({\n  children,\n  emptyState,\n  descriptionDirection,\n  searchable,\n  placeholder,\n  footer,\n  refSelect,\n  loadMore,\n  optionalInfoPlacement,\n  isLoading,\n  size,\n  dropdownAlign,\n  portalTarget,\n  id,\n}: DropdownProps) => {\n  const {\n    setIsDropdownVisible,\n    isDropdownVisible,\n    onSearch,\n    searchInput,\n    options,\n    displayedOptions,\n    numberOfOptions,\n  } = useSelectInput()\n  const theme = useTheme()\n  const [searchBarActive, setSearchBarActive] = useState(false)\n  const [defaultSearchValue, setDefaultSearch] = useState<string | null>(null)\n  const ref = useRef<HTMLDivElement>(null)\n  const [search, setSearch] = useState<string>('')\n  const [maxWidth, setWidth] = useState<string | number>()\n  const modalContext = useContext(ModalContext)\n\n  useEffect(() => {\n    if (refSelect.current && isDropdownVisible) {\n      const position =\n        refSelect.current.getBoundingClientRect().bottom +\n        DROPDOWN_MAX_HEIGHT +\n        Number(theme.sizing[INPUT_SIZE_HEIGHT[size]].replace('rem', '')) * 16 +\n        Number.parseInt(theme.space['5'], 10)\n      const overflow = position - window.innerHeight + 32\n      if (overflow > 0 && modalContext) {\n        const currentModal = modalContext.openedModals[0]\n        const modalElement = currentModal?.ref.current\n\n        if (modalElement) {\n          const parentElement = modalElement.parentNode as HTMLElement\n          if (parentElement) {\n            parentElement.scrollBy({\n              behavior: 'smooth',\n              top: overflow,\n            })\n          }\n        } else {\n          window.scrollBy({ behavior: 'smooth', top: overflow })\n        }\n      }\n    }\n    // oxlint-disable-next-line react/exhaustive-deps\n  }, [isDropdownVisible, refSelect, size, ref.current])\n\n  const resizeDropdown = useCallback(() => {\n    if (\n      refSelect.current &&\n      refSelect.current.getBoundingClientRect().width > 0\n    ) {\n      setWidth(refSelect.current.getBoundingClientRect().width)\n    }\n  }, [refSelect])\n\n  useEffect(() => {\n    resizeDropdown()\n\n    window.addEventListener('resize', resizeDropdown)\n\n    return () => window.removeEventListener('resize', resizeDropdown)\n  }, [resizeDropdown])\n\n  useEffect(() => {\n    if (!searchInput) {\n      onSearch(options)\n    }\n  }, [onSearch, options, searchInput])\n\n  useEffect(() => {\n    if (!isDropdownVisible) {\n      setDefaultSearch(null)\n      setSearch('')\n    }\n\n    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      if (!searchable) {\n        document.removeEventListener('keydown', event =>\n          handleKeyDown(\n            event,\n            ref,\n            options,\n            searchBarActive,\n            setSearch,\n            setDefaultSearch,\n            search,\n          ),\n        )\n      }\n    }\n  }, [\n    isDropdownVisible,\n    searchBarActive,\n    options,\n    onSearch,\n    search,\n    refSelect,\n    setDefaultSearch,\n    setIsDropdownVisible,\n    searchable,\n  ])\n\n  // No data is displayed (because of the search or because no data is provided)\n  // Set to true when noData by default\n  const isEmpty = useMemo(() => {\n    if (numberOfOptions === 0) {\n      return true\n    }\n    if (Array.isArray(displayedOptions)) {\n      return displayedOptions.length === 0\n    }\n    const groups = Object.keys(displayedOptions)\n    for (const group of groups) {\n      if (displayedOptions[group].length > 0) {\n        return false\n      }\n    }\n\n    return true\n  }, [displayedOptions, numberOfOptions])\n\n  const computedFooter = useMemo(() => {\n    if (footer && !isEmpty) {\n      if (typeof footer === 'function') {\n        return (\n          <PopupFooter>{footer(() => setIsDropdownVisible(false))}</PopupFooter>\n        )\n      }\n\n      return <PopupFooter>{footer}</PopupFooter>\n    }\n\n    return null\n  }, [isEmpty, footer, setIsDropdownVisible])\n\n  return (\n    <StyledPopup\n      align={dropdownAlign ?? 'start'}\n      containerFullWidth\n      debounceDelay={0}\n      disableAnimation\n      hasArrow={false}\n      hideOnClickOutside\n      id={id}\n      maxWidth={maxWidth ?? refSelect.current?.offsetWidth}\n      onClose={() => setIsDropdownVisible(false)}\n      placement=\"bottom\"\n      portalTarget={portalTarget}\n      ref={ref}\n      role=\"dialog\"\n      tabIndex={-1}\n      text={\n        <Stack>\n          {searchable && !isLoading && numberOfOptions >= 6 ? (\n            <SearchBarDropdown\n              displayedOptions={displayedOptions}\n              placeholder={placeholder}\n              setSearchBarActive={setSearchBarActive}\n            />\n          ) : null}\n          <CreateDropdown\n            defaultSearchValue={defaultSearchValue}\n            descriptionDirection={descriptionDirection}\n            emptyState={emptyState}\n            isEmpty={isEmpty}\n            isLoading={isLoading}\n            loadMore={loadMore}\n            optionalInfoPlacement={optionalInfoPlacement}\n          />\n          {computedFooter}\n        </Stack>\n      }\n      visible={isDropdownVisible}\n    >\n      {children}\n    </StyledPopup>\n  )\n}\n"]} */"));
|
|
51
51
|
const DropdownGroup = /* @__PURE__ */ _styled("button", process.env.NODE_ENV === "production" ? {
|
|
52
52
|
target: "e12qyldb6"
|
|
53
53
|
} : {
|
|
@@ -61,7 +61,7 @@ const DropdownGroup = /* @__PURE__ */ _styled("button", process.env.NODE_ENV ===
|
|
|
61
61
|
theme
|
|
62
62
|
}) => theme.space[2], ";height:", ({
|
|
63
63
|
theme
|
|
64
|
-
}) => theme.space[4], ";text-align:left;margin-bottom:", ({
|
|
64
|
+
}) => theme.space[4], ";display:flex;text-align:left;margin-bottom:", ({
|
|
65
65
|
theme
|
|
66
66
|
}) => theme.space["0.25"], ";&:focus{background-color:", ({
|
|
67
67
|
theme
|
|
@@ -73,7 +73,7 @@ const DropdownGroup = /* @__PURE__ */ _styled("button", process.env.NODE_ENV ===
|
|
|
73
73
|
theme
|
|
74
74
|
}) => theme.colors.neutral.backgroundWeak, ";}&[data-selectgroup='true']:focus{background-color:", ({
|
|
75
75
|
theme
|
|
76
|
-
}) => 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/SelectInput/Dropdown.tsx"],"names":[],"mappings":"AAiGoE","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInput/Dropdown.tsx","sourcesContent":["'use client'\n\nimport { useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport type {\n  ComponentProps,\n  Dispatch,\n  KeyboardEvent,\n  ReactNode,\n  RefObject,\n  SetStateAction,\n} from 'react'\nimport {\n  useCallback,\n  useContext,\n  useEffect,\n  useMemo,\n  useRef,\n  useState,\n} from 'react'\nimport { Checkbox } from '../Checkbox'\nimport { ModalContext } from '../Modal/ModalProvider'\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'\nimport { INPUT_SIZE_HEIGHT } from './types'\n\nconst DROPDOWN_MAX_HEIGHT = 256\n\nexport type DropdownProps = {\n  id?: string\n  children: ReactNode\n  emptyState: ReactNode\n  descriptionDirection: 'row' | 'column'\n  searchable: boolean\n  placeholder: string\n  footer?: ((closeDropdown: () => void) => ReactNode) | ReactNode\n  refSelect: RefObject<HTMLDivElement | null>\n  loadMore?: ReactNode\n  optionalInfoPlacement: 'left' | 'right'\n  isLoading?: boolean\n  size: 'small' | 'medium' | 'large'\n  dropdownAlign?: ComponentProps<typeof Popup>['align']\n  portalTarget?: ComponentProps<typeof Popup>['portalTarget']\n}\n\nexport type CreateDropdownProps = {\n  isEmpty: boolean\n  emptyState: ReactNode\n  descriptionDirection: 'row' | 'column'\n  loadMore?: ReactNode\n  optionalInfoPlacement: 'left' | 'right'\n  defaultSearchValue: string | null\n  isLoading?: boolean\n}\n\nconst NON_SEARCHABLE_KEYS = [\n  'Tab',\n  ' ',\n  'Enter',\n  'CapsLock',\n  'Shift',\n  'ArrowDown',\n  'ArrowUp',\n  'ArrowLeft',\n  'ArrowRight',\n  'Escape',\n]\n\nconst StyledPopup = styled(Popup)`\n  width: 100%;\n  min-width: 320px;\n  background-color: ${({ theme }) =>\n    theme.colors.other.elevation.background.raised};\n  color: ${({ theme }) => theme.colors.neutral.text};\n  box-shadow: ${({ theme }) =>\n    `${theme.shadows.raised[0]}, ${theme.shadows.raised[1]}`};\n  padding: ${({ theme }) => theme.space[0]};\n  margin-bottom: ${({ theme }) => theme.space[10]};\n`\n\nconst DropdownContainer = styled(Stack)<{ 'data-grouped': boolean }>`\n  max-height: ${DROPDOWN_MAX_HEIGHT}px;\n  overflow-y: auto;\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: 0;\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: 0;\n`\nconst DropdownItem = styled.div<{\n  'aria-selected': boolean\n  'aria-disabled': boolean\n}>`\n  text-align:left;\n  border: none;\n  background-color: ${({ theme }) =>\n    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  border: 1px transparent solid;\n\n  &:hover, :focus {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n    color: ${({ theme }) => theme.colors.primary.text};\n    cursor: pointer;\n  }\n\n  &[aria-selected='true'] {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n  }\n\n  &[aria-disabled=\"true\"] {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n  }\n\n  &[aria-disabled=\"true\"]:hover, [aria-disabled=\"true\"]:focus {\n    background-color: ${({ theme }) =>\n      theme.colors.neutral.backgroundStrongDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textStrongDisabled};\n    cursor: not-allowed;\n    outline: none;\n  }\n`\n\nconst PopupFooter = styled.div`\n  width: 100%;\n  padding: ${({ theme }) => theme.space[1.5]} ${({ theme }) => theme.space[2]}\n    ${({ theme }) => theme.space[1.5]} ${({ theme }) => theme.space[2]};\n  box-shadow: ${({ theme }) => theme.shadows.dropdown};\n`\nconst StyledCheckbox = styled(Checkbox)`\n  width: 100%;\n  position: static;\n  text-align: left;\n  align-items: center;\n  pointer-events: none;\n` // pointer-events: none prevents any error when using the checkbox in a form since it is an unnamed input\n\nconst EmptyState = styled(Stack)`\n  padding: ${({ theme }) => theme.space[2]};\n`\nconst LoadMore = styled(Stack)`\n  padding: ${({ theme }) => theme.space[0.5]};\n`\n\nconst moveFocusDown = () => {\n  const options = document.querySelectorAll(\n    '#items > div[role=\"option\"]:not([disabled])',\n  )\n  const activeItem = document.activeElement\n  if (options) {\n    for (let i = 0; i < options?.length; i += 1) {\n      const listLength = options.length\n      if (activeItem === options[i] && activeItem !== options[listLength - 1]) {\n        ;(options[i + 1] as HTMLElement).focus()\n      }\n    }\n  }\n}\nconst moveFocusUp = () => {\n  const options = document.querySelectorAll(\n    '#items > div[role=\"option\"]:not([disabled])',\n  )\n  const activeItem = document.activeElement\n\n  if (options) {\n    for (let i = 0; i < options.length; i += 1) {\n      if (activeItem === options[i] && activeItem !== options[0]) {\n        ;(options[i - 1] as HTMLElement).focus()\n      }\n    }\n  }\n}\nconst handleKeyDownSelect = (event: KeyboardEvent<HTMLDivElement>) => {\n  if (event.key === 'ArrowDown') {\n    event.preventDefault()\n    moveFocusDown()\n  }\n\n  if (event.key === 'ArrowUp') {\n    event.preventDefault()\n    moveFocusUp()\n  }\n\n  if (event.key === ' ') {\n    // No scroll\n    event.preventDefault()\n  }\n}\n\nconst handleKeyDown = (\n  event: globalThis.KeyboardEvent,\n  ref: RefObject<HTMLDivElement | null>,\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].find(option =>\n        option.searchText\n          ? option.searchText.toLocaleLowerCase().startsWith(currentSearch)\n          : option.value.toLocaleLowerCase().startsWith(currentSearch),\n      )\n      if (closestOption) {\n        setDefaultSearch(closestOption.searchText ?? closestOption.value)\n      } else {\n        setDefaultSearch(null)\n      }\n    }\n  }\n}\nconst CreateDropdown = ({\n  isEmpty,\n  emptyState,\n  descriptionDirection,\n  loadMore,\n  optionalInfoPlacement,\n  defaultSearchValue,\n  isLoading,\n}: CreateDropdownProps) => {\n  const {\n    setIsDropdownVisible,\n    onChange,\n    options,\n    multiselect,\n    selectAll,\n    selectAllGroup,\n    displayedOptions,\n    setSelectedData,\n    selectedData,\n  } = useSelectInput()\n  const focusedItemRef = useRef<HTMLDivElement>(null)\n\n  useEffect(() => {\n    if (defaultSearchValue && focusedItemRef?.current) {\n      focusedItemRef.current.focus()\n    }\n  }, [defaultSearchValue])\n\n  if (isEmpty) {\n    return (\n      <EmptyState alignItems=\"center\" gap={2}>\n        {emptyState ?? (\n          <Text as=\"p\" variant=\"bodyStrong\">\n            No options\n          </Text>\n        )}\n      </EmptyState>\n    )\n  }\n\n  const handleClick = (clickedOption: OptionType, group?: string) => {\n    setSelectedData({ clickedOption, group, type: 'selectOption' })\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({ selectedGroup: group, type: 'selectGroup' })\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      data-grouped\n      id=\"select-dropdown\"\n      onKeyDown={handleKeyDownSelect}\n      role=\"listbox\"\n    >\n      {isLoading ? (\n        <Skeleton variant=\"block\" />\n      ) : (\n        <>\n          {selectAll && multiselect ? (\n            <Stack id=\"items\">\n              <DropdownItem\n                aria-disabled={false}\n                aria-label=\"select-all\"\n                aria-selected={selectedData.allSelected}\n                data-testid=\"select-all\"\n                id=\"select-all\"\n                onClick={selectAllOptions}\n                onKeyDown={event =>\n                  [' ', 'Enter'].includes(event.key) ? selectAllOptions() : null\n                }\n                role=\"option\"\n                tabIndex={0}\n              >\n                <StyledCheckbox\n                  checked={selectedData.allSelected}\n                  data-testid=\"select-all-checkbox\"\n                  disabled={false}\n                  onChange={selectAllOptions}\n                  tabIndex={-1}\n                  value=\"select-all\"\n                >\n                  <Stack direction=\"column\">\n                    <Text as=\"span\" placement=\"left\" variant=\"body\">\n                      {selectAll.label}\n                    </Text>\n                    <Text\n                      as=\"span\"\n                      placement=\"left\"\n                      prominence=\"weak\"\n                      sentiment=\"neutral\"\n                      variant=\"bodySmall\"\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 gap={0.25} key={group}>\n              {displayedOptions[group].length > 0 ? (\n                <DropdownGroupWrapper id={selectAllGroup ? 'items' : undefined}>\n                  {group ? (\n                    <DropdownGroup\n                      data-selectgroup={selectAllGroup}\n                      data-testid={`group-${index}`}\n                      key={group}\n                      onClick={() =>\n                        selectAllGroup ? handleSelectGroup(group) : null\n                      }\n                      onKeyDown={event => {\n                        if ([' ', 'Enter'].includes(event.key)) {\n                          event.preventDefault()\n                          handleSelectGroup(group)\n                        }\n                      }}\n                      role=\"group\"\n                      tabIndex={selectAllGroup ? 0 : -1}\n                      type=\"button\"\n                    >\n                      {selectAllGroup ? (\n                        <StyledCheckbox\n                          checked={selectedData.selectedGroups.includes(group)}\n                          data-testid=\"select-group\"\n                          disabled={false}\n                          onChange={() =>\n                            selectAllGroup ? handleSelectGroup(group) : null\n                          }\n                          tabIndex={-1}\n                          value={group}\n                        >\n                          <Text\n                            as=\"span\"\n                            placement=\"left\"\n                            sentiment=\"neutral\"\n                            variant=\"caption\"\n                          >\n                            {group.toUpperCase()}\n                          </Text>\n                        </StyledCheckbox>\n                      ) : (\n                        <Text\n                          as=\"span\"\n                          placement=\"left\"\n                          sentiment=\"neutral\"\n                          variant=\"caption\"\n                        >\n                          {group.toUpperCase()}\n                        </Text>\n                      )}\n                    </DropdownGroup>\n                  ) : null}\n                </DropdownGroupWrapper>\n              ) : null}\n              <Stack gap=\"0.25\" id=\"items\">\n                {displayedOptions[group].map((option, indexOption) => (\n                  <DropdownItem\n                    aria-disabled={!!option.disabled}\n                    aria-label={option.value}\n                    aria-selected={\n                      selectedData.selectedValues.includes(option.value) &&\n                      !option.disabled\n                    }\n                    data-testid={`option-${option.value}`}\n                    id={`option-${indexOption}`}\n                    key={option.value}\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                    role=\"option\"\n                    tabIndex={!option.disabled ? 0 : -1}\n                  >\n                    {multiselect ? (\n                      <StyledCheckbox\n                        checked={\n                          selectedData.selectedValues.includes(option.value) &&\n                          !option.disabled\n                        }\n                        disabled={option.disabled}\n                        onChange={() => {\n                          if (!option.disabled) {\n                            handleClick(option, group)\n                          }\n                        }}\n                        tabIndex={-1}\n                        value={option.value}\n                      >\n                        <DisplayOption\n                          descriptionDirection={descriptionDirection}\n                          option={option}\n                          optionalInfoPlacement={optionalInfoPlacement}\n                        />\n                      </StyledCheckbox>\n                    ) : (\n                      <DisplayOption\n                        descriptionDirection={descriptionDirection}\n                        option={option}\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      data-grouped={false}\n      gap={0.25}\n      id=\"select-dropdown\"\n      onKeyDown={handleKeyDownSelect}\n      role=\"listbox\"\n    >\n      {selectAll && multiselect ? (\n        <Stack gap={0.25} id=\"items\" tabIndex={-1}>\n          <DropdownItem\n            aria-disabled={false}\n            aria-label=\"select-all\"\n            aria-selected={selectedData.allSelected}\n            data-testid=\"select-all\"\n            onClick={selectAllOptions}\n            onKeyDown={event =>\n              [' ', 'Enter'].includes(event.key) ? selectAllOptions() : null\n            }\n            role=\"option\"\n            tabIndex={0}\n          >\n            <StyledCheckbox\n              checked={selectedData.allSelected}\n              data-testid=\"select-all-checkbox\"\n              disabled={false}\n              onChange={selectAllOptions}\n              tabIndex={-1}\n              value=\"select-all\"\n            >\n              <Stack direction=\"column\">\n                <Text as=\"span\" placement=\"left\" variant=\"body\">\n                  {selectAll.label}\n                </Text>\n                <Text\n                  as=\"span\"\n                  placement=\"left\"\n                  prominence=\"weak\"\n                  sentiment=\"neutral\"\n                  variant=\"bodySmall\"\n                >\n                  {selectAll.description}\n                </Text>\n              </Stack>\n            </StyledCheckbox>\n          </DropdownItem>\n        </Stack>\n      ) : null}\n      <Stack gap={0.25} id=\"items\">\n        {isLoading ? (\n          <Skeleton variant=\"block\" />\n        ) : (\n          displayedOptions.map((option, index) => (\n            <DropdownItem\n              aria-disabled={!!option.disabled}\n              aria-label={option.value}\n              aria-selected={\n                selectedData.selectedValues.includes(option.value) &&\n                !option.disabled\n              }\n              data-testid={`option-${option.value}`}\n              id={`option-${index}`}\n              key={option.value}\n              onClick={() => {\n                if (!option.disabled) {\n                  handleClick(option)\n                }\n              }}\n              onKeyDown={event =>\n                [' ', 'Enter'].includes(event.key) ? handleClick(option) : null\n              }\n              ref={\n                option.value === defaultSearchValue ||\n                option.searchText === defaultSearchValue\n                  ? focusedItemRef\n                  : null\n              }\n              role=\"option\"\n              tabIndex={!option.disabled ? 0 : -1}\n            >\n              {multiselect ? (\n                <StyledCheckbox\n                  checked={\n                    selectedData.selectedValues.includes(option.value) &&\n                    !option.disabled\n                  }\n                  disabled={option.disabled}\n                  onChange={() => {\n                    if (!option.disabled) {\n                      handleClick(option)\n                    }\n                  }}\n                  tabIndex={-1}\n                  value={option.value}\n                >\n                  <DisplayOption\n                    descriptionDirection={descriptionDirection}\n                    option={option}\n                    optionalInfoPlacement={optionalInfoPlacement}\n                  />\n                </StyledCheckbox>\n              ) : (\n                <DisplayOption\n                  descriptionDirection={descriptionDirection}\n                  option={option}\n                  optionalInfoPlacement={optionalInfoPlacement}\n                />\n              )}\n            </DropdownItem>\n          ))\n        )}\n        {loadMore ? <LoadMore>{loadMore}</LoadMore> : null}\n      </Stack>\n    </DropdownContainer>\n  )\n}\nexport const Dropdown = ({\n  children,\n  emptyState,\n  descriptionDirection,\n  searchable,\n  placeholder,\n  footer,\n  refSelect,\n  loadMore,\n  optionalInfoPlacement,\n  isLoading,\n  size,\n  dropdownAlign,\n  portalTarget,\n  id,\n}: DropdownProps) => {\n  const {\n    setIsDropdownVisible,\n    isDropdownVisible,\n    onSearch,\n    searchInput,\n    options,\n    displayedOptions,\n    numberOfOptions,\n  } = useSelectInput()\n  const theme = useTheme()\n  const [searchBarActive, setSearchBarActive] = useState(false)\n  const [defaultSearchValue, setDefaultSearch] = useState<string | null>(null)\n  const ref = useRef<HTMLDivElement>(null)\n  const [search, setSearch] = useState<string>('')\n  const [maxWidth, setWidth] = useState<string | number>()\n  const modalContext = useContext(ModalContext)\n\n  useEffect(() => {\n    if (refSelect.current && isDropdownVisible) {\n      const position =\n        refSelect.current.getBoundingClientRect().bottom +\n        DROPDOWN_MAX_HEIGHT +\n        Number(theme.sizing[INPUT_SIZE_HEIGHT[size]].replace('rem', '')) * 16 +\n        Number.parseInt(theme.space['5'], 10)\n      const overflow = position - window.innerHeight + 32\n      if (overflow > 0 && modalContext) {\n        const currentModal = modalContext.openedModals[0]\n        const modalElement = currentModal?.ref.current\n\n        if (modalElement) {\n          const parentElement = modalElement.parentNode as HTMLElement\n          if (parentElement) {\n            parentElement.scrollBy({\n              behavior: 'smooth',\n              top: overflow,\n            })\n          }\n        } else {\n          window.scrollBy({ behavior: 'smooth', top: overflow })\n        }\n      }\n    }\n    // oxlint-disable-next-line react/exhaustive-deps\n  }, [isDropdownVisible, refSelect, size, ref.current])\n\n  const resizeDropdown = useCallback(() => {\n    if (\n      refSelect.current &&\n      refSelect.current.getBoundingClientRect().width > 0\n    ) {\n      setWidth(refSelect.current.getBoundingClientRect().width)\n    }\n  }, [refSelect])\n\n  useEffect(() => {\n    resizeDropdown()\n\n    window.addEventListener('resize', resizeDropdown)\n\n    return () => window.removeEventListener('resize', resizeDropdown)\n  }, [resizeDropdown])\n\n  useEffect(() => {\n    if (!searchInput) {\n      onSearch(options)\n    }\n  }, [onSearch, options, searchInput])\n\n  useEffect(() => {\n    if (!isDropdownVisible) {\n      setDefaultSearch(null)\n      setSearch('')\n    }\n\n    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      if (!searchable) {\n        document.removeEventListener('keydown', event =>\n          handleKeyDown(\n            event,\n            ref,\n            options,\n            searchBarActive,\n            setSearch,\n            setDefaultSearch,\n            search,\n          ),\n        )\n      }\n    }\n  }, [\n    isDropdownVisible,\n    searchBarActive,\n    options,\n    onSearch,\n    search,\n    refSelect,\n    setDefaultSearch,\n    setIsDropdownVisible,\n    searchable,\n  ])\n\n  // No data is displayed (because of the search or because no data is provided)\n  // Set to true when noData by default\n  const isEmpty = useMemo(() => {\n    if (numberOfOptions === 0) {\n      return true\n    }\n    if (Array.isArray(displayedOptions)) {\n      return displayedOptions.length === 0\n    }\n    const groups = Object.keys(displayedOptions)\n    for (const group of groups) {\n      if (displayedOptions[group].length > 0) {\n        return false\n      }\n    }\n\n    return true\n  }, [displayedOptions, numberOfOptions])\n\n  const computedFooter = useMemo(() => {\n    if (footer && !isEmpty) {\n      if (typeof footer === 'function') {\n        return (\n          <PopupFooter>{footer(() => setIsDropdownVisible(false))}</PopupFooter>\n        )\n      }\n\n      return <PopupFooter>{footer}</PopupFooter>\n    }\n\n    return null\n  }, [isEmpty, footer, setIsDropdownVisible])\n\n  return (\n    <StyledPopup\n      align={dropdownAlign ?? 'start'}\n      containerFullWidth\n      debounceDelay={0}\n      disableAnimation\n      hasArrow={false}\n      hideOnClickOutside\n      id={id}\n      maxWidth={maxWidth ?? refSelect.current?.offsetWidth}\n      onClose={() => setIsDropdownVisible(false)}\n      placement=\"bottom\"\n      portalTarget={portalTarget}\n      ref={ref}\n      role=\"dialog\"\n      tabIndex={-1}\n      text={\n        <Stack>\n          {searchable && !isLoading && numberOfOptions >= 6 ? (\n            <SearchBarDropdown\n              displayedOptions={displayedOptions}\n              placeholder={placeholder}\n              setSearchBarActive={setSearchBarActive}\n            />\n          ) : null}\n          <CreateDropdown\n            defaultSearchValue={defaultSearchValue}\n            descriptionDirection={descriptionDirection}\n            emptyState={emptyState}\n            isEmpty={isEmpty}\n            isLoading={isLoading}\n            loadMore={loadMore}\n            optionalInfoPlacement={optionalInfoPlacement}\n          />\n          {computedFooter}\n        </Stack>\n      }\n      visible={isDropdownVisible}\n    >\n      {children}\n    </StyledPopup>\n  )\n}\n"]} */"));
|
|
76
|
+
}) => 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/SelectInput/Dropdown.tsx"],"names":[],"mappings":"AAiGoE","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInput/Dropdown.tsx","sourcesContent":["'use client'\n\nimport { useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport type {\n  ComponentProps,\n  Dispatch,\n  KeyboardEvent,\n  ReactNode,\n  RefObject,\n  SetStateAction,\n} from 'react'\nimport {\n  useCallback,\n  useContext,\n  useEffect,\n  useMemo,\n  useRef,\n  useState,\n} from 'react'\nimport { Checkbox } from '../Checkbox'\nimport { ModalContext } from '../Modal/ModalProvider'\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'\nimport { INPUT_SIZE_HEIGHT } from './types'\n\nconst DROPDOWN_MAX_HEIGHT = 256\n\nexport type DropdownProps = {\n  id?: string\n  children: ReactNode\n  emptyState: ReactNode\n  descriptionDirection: 'row' | 'column'\n  searchable: boolean\n  placeholder: string\n  footer?: ((closeDropdown: () => void) => ReactNode) | ReactNode\n  refSelect: RefObject<HTMLDivElement | null>\n  loadMore?: ReactNode\n  optionalInfoPlacement: 'left' | 'right'\n  isLoading?: boolean\n  size: 'small' | 'medium' | 'large'\n  dropdownAlign?: ComponentProps<typeof Popup>['align']\n  portalTarget?: ComponentProps<typeof Popup>['portalTarget']\n}\n\nexport type CreateDropdownProps = {\n  isEmpty: boolean\n  emptyState: ReactNode\n  descriptionDirection: 'row' | 'column'\n  loadMore?: ReactNode\n  optionalInfoPlacement: 'left' | 'right'\n  defaultSearchValue: string | null\n  isLoading?: boolean\n}\n\nconst NON_SEARCHABLE_KEYS = [\n  'Tab',\n  ' ',\n  'Enter',\n  'CapsLock',\n  'Shift',\n  'ArrowDown',\n  'ArrowUp',\n  'ArrowLeft',\n  'ArrowRight',\n  'Escape',\n]\n\nconst StyledPopup = styled(Popup)`\n  width: 100%;\n  min-width: 320px;\n  background-color: ${({ theme }) =>\n    theme.colors.other.elevation.background.raised};\n  color: ${({ theme }) => theme.colors.neutral.text};\n  box-shadow: ${({ theme }) =>\n    `${theme.shadows.raised[0]}, ${theme.shadows.raised[1]}`};\n  padding: ${({ theme }) => theme.space[0]};\n  margin-bottom: ${({ theme }) => theme.space[10]};\n`\n\nconst DropdownContainer = styled(Stack)<{ 'data-grouped': boolean }>`\n  max-height: ${DROPDOWN_MAX_HEIGHT}px;\n  overflow-y: auto;\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: 0;\n  padding-right: ${({ theme }) => theme.space[2]};\n  padding-left: ${({ theme }) => theme.space[2]};\n  height: ${({ theme }) => theme.space[4]};\n  display: flex;\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: 0;\n`\nconst DropdownItem = styled.div<{\n  'aria-selected': boolean\n  'aria-disabled': boolean\n}>`\n  text-align:left;\n  border: none;\n  background-color: ${({ theme }) =>\n    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  border: 1px transparent solid;\n\n  &:hover, :focus {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n    color: ${({ theme }) => theme.colors.primary.text};\n    cursor: pointer;\n  }\n\n  &[aria-selected='true'] {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n  }\n\n  &[aria-disabled=\"true\"] {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n  }\n\n  &[aria-disabled=\"true\"]:hover, [aria-disabled=\"true\"]:focus {\n    background-color: ${({ theme }) =>\n      theme.colors.neutral.backgroundStrongDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textStrongDisabled};\n    cursor: not-allowed;\n    outline: none;\n  }\n`\n\nconst PopupFooter = styled.div`\n  width: 100%;\n  padding: ${({ theme }) => theme.space[1.5]} ${({ theme }) => theme.space[2]}\n    ${({ theme }) => theme.space[1.5]} ${({ theme }) => theme.space[2]};\n  box-shadow: ${({ theme }) => theme.shadows.dropdown};\n`\nconst StyledCheckbox = styled(Checkbox)`\n  width: 100%;\n  position: static;\n  text-align: left;\n  align-items: center;\n  pointer-events: none;\n` // pointer-events: none prevents any error when using the checkbox in a form since it is an unnamed input\n\nconst EmptyState = styled(Stack)`\n  padding: ${({ theme }) => theme.space[2]};\n`\nconst LoadMore = styled(Stack)`\n  padding: ${({ theme }) => theme.space[0.5]};\n`\n\nconst moveFocusDown = () => {\n  const options = document.querySelectorAll(\n    '#items > div[role=\"option\"]:not([disabled])',\n  )\n  const activeItem = document.activeElement\n  if (options) {\n    for (let i = 0; i < options?.length; i += 1) {\n      const listLength = options.length\n      if (activeItem === options[i] && activeItem !== options[listLength - 1]) {\n        ;(options[i + 1] as HTMLElement).focus()\n      }\n    }\n  }\n}\nconst moveFocusUp = () => {\n  const options = document.querySelectorAll(\n    '#items > div[role=\"option\"]:not([disabled])',\n  )\n  const activeItem = document.activeElement\n\n  if (options) {\n    for (let i = 0; i < options.length; i += 1) {\n      if (activeItem === options[i] && activeItem !== options[0]) {\n        ;(options[i - 1] as HTMLElement).focus()\n      }\n    }\n  }\n}\nconst handleKeyDownSelect = (event: KeyboardEvent<HTMLDivElement>) => {\n  if (event.key === 'ArrowDown') {\n    event.preventDefault()\n    moveFocusDown()\n  }\n\n  if (event.key === 'ArrowUp') {\n    event.preventDefault()\n    moveFocusUp()\n  }\n\n  if (event.key === ' ') {\n    // No scroll\n    event.preventDefault()\n  }\n}\n\nconst handleKeyDown = (\n  event: globalThis.KeyboardEvent,\n  ref: RefObject<HTMLDivElement | null>,\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].find(option =>\n        option.searchText\n          ? option.searchText.toLocaleLowerCase().startsWith(currentSearch)\n          : option.value.toLocaleLowerCase().startsWith(currentSearch),\n      )\n      if (closestOption) {\n        setDefaultSearch(closestOption.searchText ?? closestOption.value)\n      } else {\n        setDefaultSearch(null)\n      }\n    }\n  }\n}\nconst CreateDropdown = ({\n  isEmpty,\n  emptyState,\n  descriptionDirection,\n  loadMore,\n  optionalInfoPlacement,\n  defaultSearchValue,\n  isLoading,\n}: CreateDropdownProps) => {\n  const {\n    setIsDropdownVisible,\n    onChange,\n    options,\n    multiselect,\n    selectAll,\n    selectAllGroup,\n    displayedOptions,\n    setSelectedData,\n    selectedData,\n  } = useSelectInput()\n  const focusedItemRef = useRef<HTMLDivElement>(null)\n\n  useEffect(() => {\n    if (defaultSearchValue && focusedItemRef?.current) {\n      focusedItemRef.current.focus()\n    }\n  }, [defaultSearchValue])\n\n  if (isEmpty) {\n    return (\n      <EmptyState alignItems=\"center\" gap={2}>\n        {emptyState ?? (\n          <Text as=\"p\" variant=\"bodyStrong\">\n            No options\n          </Text>\n        )}\n      </EmptyState>\n    )\n  }\n\n  const handleClick = (clickedOption: OptionType, group?: string) => {\n    setSelectedData({ clickedOption, group, type: 'selectOption' })\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({ selectedGroup: group, type: 'selectGroup' })\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      data-grouped\n      id=\"select-dropdown\"\n      onKeyDown={handleKeyDownSelect}\n      role=\"listbox\"\n    >\n      {isLoading ? (\n        <Skeleton variant=\"block\" />\n      ) : (\n        <>\n          {selectAll && multiselect ? (\n            <Stack id=\"items\">\n              <DropdownItem\n                aria-disabled={false}\n                aria-label=\"select-all\"\n                aria-selected={selectedData.allSelected}\n                data-testid=\"select-all\"\n                id=\"select-all\"\n                onClick={selectAllOptions}\n                onKeyDown={event =>\n                  [' ', 'Enter'].includes(event.key) ? selectAllOptions() : null\n                }\n                role=\"option\"\n                tabIndex={0}\n              >\n                <StyledCheckbox\n                  checked={selectedData.allSelected}\n                  data-testid=\"select-all-checkbox\"\n                  disabled={false}\n                  onChange={selectAllOptions}\n                  tabIndex={-1}\n                  value=\"select-all\"\n                >\n                  <Stack direction=\"column\">\n                    <Text as=\"span\" placement=\"left\" variant=\"body\">\n                      {selectAll.label}\n                    </Text>\n                    <Text\n                      as=\"span\"\n                      placement=\"left\"\n                      prominence=\"weak\"\n                      sentiment=\"neutral\"\n                      variant=\"bodySmall\"\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 gap={0.25} key={group}>\n              {displayedOptions[group].length > 0 ? (\n                <DropdownGroupWrapper id={selectAllGroup ? 'items' : undefined}>\n                  {group ? (\n                    <DropdownGroup\n                      data-selectgroup={selectAllGroup}\n                      data-testid={`group-${index}`}\n                      key={group}\n                      onClick={() =>\n                        selectAllGroup ? handleSelectGroup(group) : null\n                      }\n                      onKeyDown={event => {\n                        if ([' ', 'Enter'].includes(event.key)) {\n                          event.preventDefault()\n                          handleSelectGroup(group)\n                        }\n                      }}\n                      role=\"group\"\n                      tabIndex={selectAllGroup ? 0 : -1}\n                      type=\"button\"\n                    >\n                      {selectAllGroup ? (\n                        <StyledCheckbox\n                          checked={selectedData.selectedGroups.includes(group)}\n                          data-testid=\"select-group\"\n                          disabled={false}\n                          onChange={() =>\n                            selectAllGroup ? handleSelectGroup(group) : null\n                          }\n                          tabIndex={-1}\n                          value={group}\n                        >\n                          <Text\n                            as=\"span\"\n                            placement=\"left\"\n                            sentiment=\"neutral\"\n                            variant=\"caption\"\n                          >\n                            {group.toUpperCase()}\n                          </Text>\n                        </StyledCheckbox>\n                      ) : (\n                        <Text\n                          as=\"span\"\n                          placement=\"left\"\n                          sentiment=\"neutral\"\n                          variant=\"caption\"\n                        >\n                          {group.toUpperCase()}\n                        </Text>\n                      )}\n                    </DropdownGroup>\n                  ) : null}\n                </DropdownGroupWrapper>\n              ) : null}\n              <Stack gap=\"0.25\" id=\"items\">\n                {displayedOptions[group].map((option, indexOption) => (\n                  <DropdownItem\n                    aria-disabled={!!option.disabled}\n                    aria-label={option.value}\n                    aria-selected={\n                      selectedData.selectedValues.includes(option.value) &&\n                      !option.disabled\n                    }\n                    data-testid={`option-${option.value}`}\n                    id={`option-${indexOption}`}\n                    key={option.value}\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                    role=\"option\"\n                    tabIndex={!option.disabled ? 0 : -1}\n                  >\n                    {multiselect ? (\n                      <StyledCheckbox\n                        checked={\n                          selectedData.selectedValues.includes(option.value) &&\n                          !option.disabled\n                        }\n                        disabled={option.disabled}\n                        onChange={() => {\n                          if (!option.disabled) {\n                            handleClick(option, group)\n                          }\n                        }}\n                        tabIndex={-1}\n                        value={option.value}\n                      >\n                        <DisplayOption\n                          descriptionDirection={descriptionDirection}\n                          option={option}\n                          optionalInfoPlacement={optionalInfoPlacement}\n                        />\n                      </StyledCheckbox>\n                    ) : (\n                      <DisplayOption\n                        descriptionDirection={descriptionDirection}\n                        option={option}\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      data-grouped={false}\n      gap={0.25}\n      id=\"select-dropdown\"\n      onKeyDown={handleKeyDownSelect}\n      role=\"listbox\"\n    >\n      {selectAll && multiselect ? (\n        <Stack gap={0.25} id=\"items\" tabIndex={-1}>\n          <DropdownItem\n            aria-disabled={false}\n            aria-label=\"select-all\"\n            aria-selected={selectedData.allSelected}\n            data-testid=\"select-all\"\n            onClick={selectAllOptions}\n            onKeyDown={event =>\n              [' ', 'Enter'].includes(event.key) ? selectAllOptions() : null\n            }\n            role=\"option\"\n            tabIndex={0}\n          >\n            <StyledCheckbox\n              checked={selectedData.allSelected}\n              data-testid=\"select-all-checkbox\"\n              disabled={false}\n              onChange={selectAllOptions}\n              tabIndex={-1}\n              value=\"select-all\"\n            >\n              <Stack direction=\"column\">\n                <Text as=\"span\" placement=\"left\" variant=\"body\">\n                  {selectAll.label}\n                </Text>\n                <Text\n                  as=\"span\"\n                  placement=\"left\"\n                  prominence=\"weak\"\n                  sentiment=\"neutral\"\n                  variant=\"bodySmall\"\n                >\n                  {selectAll.description}\n                </Text>\n              </Stack>\n            </StyledCheckbox>\n          </DropdownItem>\n        </Stack>\n      ) : null}\n      <Stack gap={0.25} id=\"items\">\n        {isLoading ? (\n          <Skeleton variant=\"block\" />\n        ) : (\n          displayedOptions.map((option, index) => (\n            <DropdownItem\n              aria-disabled={!!option.disabled}\n              aria-label={option.value}\n              aria-selected={\n                selectedData.selectedValues.includes(option.value) &&\n                !option.disabled\n              }\n              data-testid={`option-${option.value}`}\n              id={`option-${index}`}\n              key={option.value}\n              onClick={() => {\n                if (!option.disabled) {\n                  handleClick(option)\n                }\n              }}\n              onKeyDown={event =>\n                [' ', 'Enter'].includes(event.key) ? handleClick(option) : null\n              }\n              ref={\n                option.value === defaultSearchValue ||\n                option.searchText === defaultSearchValue\n                  ? focusedItemRef\n                  : null\n              }\n              role=\"option\"\n              tabIndex={!option.disabled ? 0 : -1}\n            >\n              {multiselect ? (\n                <StyledCheckbox\n                  checked={\n                    selectedData.selectedValues.includes(option.value) &&\n                    !option.disabled\n                  }\n                  disabled={option.disabled}\n                  onChange={() => {\n                    if (!option.disabled) {\n                      handleClick(option)\n                    }\n                  }}\n                  tabIndex={-1}\n                  value={option.value}\n                >\n                  <DisplayOption\n                    descriptionDirection={descriptionDirection}\n                    option={option}\n                    optionalInfoPlacement={optionalInfoPlacement}\n                  />\n                </StyledCheckbox>\n              ) : (\n                <DisplayOption\n                  descriptionDirection={descriptionDirection}\n                  option={option}\n                  optionalInfoPlacement={optionalInfoPlacement}\n                />\n              )}\n            </DropdownItem>\n          ))\n        )}\n        {loadMore ? <LoadMore>{loadMore}</LoadMore> : null}\n      </Stack>\n    </DropdownContainer>\n  )\n}\nexport const Dropdown = ({\n  children,\n  emptyState,\n  descriptionDirection,\n  searchable,\n  placeholder,\n  footer,\n  refSelect,\n  loadMore,\n  optionalInfoPlacement,\n  isLoading,\n  size,\n  dropdownAlign,\n  portalTarget,\n  id,\n}: DropdownProps) => {\n  const {\n    setIsDropdownVisible,\n    isDropdownVisible,\n    onSearch,\n    searchInput,\n    options,\n    displayedOptions,\n    numberOfOptions,\n  } = useSelectInput()\n  const theme = useTheme()\n  const [searchBarActive, setSearchBarActive] = useState(false)\n  const [defaultSearchValue, setDefaultSearch] = useState<string | null>(null)\n  const ref = useRef<HTMLDivElement>(null)\n  const [search, setSearch] = useState<string>('')\n  const [maxWidth, setWidth] = useState<string | number>()\n  const modalContext = useContext(ModalContext)\n\n  useEffect(() => {\n    if (refSelect.current && isDropdownVisible) {\n      const position =\n        refSelect.current.getBoundingClientRect().bottom +\n        DROPDOWN_MAX_HEIGHT +\n        Number(theme.sizing[INPUT_SIZE_HEIGHT[size]].replace('rem', '')) * 16 +\n        Number.parseInt(theme.space['5'], 10)\n      const overflow = position - window.innerHeight + 32\n      if (overflow > 0 && modalContext) {\n        const currentModal = modalContext.openedModals[0]\n        const modalElement = currentModal?.ref.current\n\n        if (modalElement) {\n          const parentElement = modalElement.parentNode as HTMLElement\n          if (parentElement) {\n            parentElement.scrollBy({\n              behavior: 'smooth',\n              top: overflow,\n            })\n          }\n        } else {\n          window.scrollBy({ behavior: 'smooth', top: overflow })\n        }\n      }\n    }\n    // oxlint-disable-next-line react/exhaustive-deps\n  }, [isDropdownVisible, refSelect, size, ref.current])\n\n  const resizeDropdown = useCallback(() => {\n    if (\n      refSelect.current &&\n      refSelect.current.getBoundingClientRect().width > 0\n    ) {\n      setWidth(refSelect.current.getBoundingClientRect().width)\n    }\n  }, [refSelect])\n\n  useEffect(() => {\n    resizeDropdown()\n\n    window.addEventListener('resize', resizeDropdown)\n\n    return () => window.removeEventListener('resize', resizeDropdown)\n  }, [resizeDropdown])\n\n  useEffect(() => {\n    if (!searchInput) {\n      onSearch(options)\n    }\n  }, [onSearch, options, searchInput])\n\n  useEffect(() => {\n    if (!isDropdownVisible) {\n      setDefaultSearch(null)\n      setSearch('')\n    }\n\n    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      if (!searchable) {\n        document.removeEventListener('keydown', event =>\n          handleKeyDown(\n            event,\n            ref,\n            options,\n            searchBarActive,\n            setSearch,\n            setDefaultSearch,\n            search,\n          ),\n        )\n      }\n    }\n  }, [\n    isDropdownVisible,\n    searchBarActive,\n    options,\n    onSearch,\n    search,\n    refSelect,\n    setDefaultSearch,\n    setIsDropdownVisible,\n    searchable,\n  ])\n\n  // No data is displayed (because of the search or because no data is provided)\n  // Set to true when noData by default\n  const isEmpty = useMemo(() => {\n    if (numberOfOptions === 0) {\n      return true\n    }\n    if (Array.isArray(displayedOptions)) {\n      return displayedOptions.length === 0\n    }\n    const groups = Object.keys(displayedOptions)\n    for (const group of groups) {\n      if (displayedOptions[group].length > 0) {\n        return false\n      }\n    }\n\n    return true\n  }, [displayedOptions, numberOfOptions])\n\n  const computedFooter = useMemo(() => {\n    if (footer && !isEmpty) {\n      if (typeof footer === 'function') {\n        return (\n          <PopupFooter>{footer(() => setIsDropdownVisible(false))}</PopupFooter>\n        )\n      }\n\n      return <PopupFooter>{footer}</PopupFooter>\n    }\n\n    return null\n  }, [isEmpty, footer, setIsDropdownVisible])\n\n  return (\n    <StyledPopup\n      align={dropdownAlign ?? 'start'}\n      containerFullWidth\n      debounceDelay={0}\n      disableAnimation\n      hasArrow={false}\n      hideOnClickOutside\n      id={id}\n      maxWidth={maxWidth ?? refSelect.current?.offsetWidth}\n      onClose={() => setIsDropdownVisible(false)}\n      placement=\"bottom\"\n      portalTarget={portalTarget}\n      ref={ref}\n      role=\"dialog\"\n      tabIndex={-1}\n      text={\n        <Stack>\n          {searchable && !isLoading && numberOfOptions >= 6 ? (\n            <SearchBarDropdown\n              displayedOptions={displayedOptions}\n              placeholder={placeholder}\n              setSearchBarActive={setSearchBarActive}\n            />\n          ) : null}\n          <CreateDropdown\n            defaultSearchValue={defaultSearchValue}\n            descriptionDirection={descriptionDirection}\n            emptyState={emptyState}\n            isEmpty={isEmpty}\n            isLoading={isLoading}\n            loadMore={loadMore}\n            optionalInfoPlacement={optionalInfoPlacement}\n          />\n          {computedFooter}\n        </Stack>\n      }\n      visible={isDropdownVisible}\n    >\n      {children}\n    </StyledPopup>\n  )\n}\n"]} */"));
|
|
77
77
|
const DropdownGroupWrapper = /* @__PURE__ */ _styled("div", process.env.NODE_ENV === "production" ? {
|
|
78
78
|
target: "e12qyldb5"
|
|
79
79
|
} : {
|
|
@@ -84,7 +84,7 @@ const DropdownGroupWrapper = /* @__PURE__ */ _styled("div", process.env.NODE_ENV
|
|
|
84
84
|
styles: "position:sticky;top:0"
|
|
85
85
|
} : {
|
|
86
86
|
name: "1o5tqc6",
|
|
87
|
-
styles: "position:sticky;top:0/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInput/Dropdown.tsx"],"names":[],"mappings":"AA+HuC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInput/Dropdown.tsx","sourcesContent":["'use client'\n\nimport { useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport type {\n  ComponentProps,\n  Dispatch,\n  KeyboardEvent,\n  ReactNode,\n  RefObject,\n  SetStateAction,\n} from 'react'\nimport {\n  useCallback,\n  useContext,\n  useEffect,\n  useMemo,\n  useRef,\n  useState,\n} from 'react'\nimport { Checkbox } from '../Checkbox'\nimport { ModalContext } from '../Modal/ModalProvider'\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'\nimport { INPUT_SIZE_HEIGHT } from './types'\n\nconst DROPDOWN_MAX_HEIGHT = 256\n\nexport type DropdownProps = {\n  id?: string\n  children: ReactNode\n  emptyState: ReactNode\n  descriptionDirection: 'row' | 'column'\n  searchable: boolean\n  placeholder: string\n  footer?: ((closeDropdown: () => void) => ReactNode) | ReactNode\n  refSelect: RefObject<HTMLDivElement | null>\n  loadMore?: ReactNode\n  optionalInfoPlacement: 'left' | 'right'\n  isLoading?: boolean\n  size: 'small' | 'medium' | 'large'\n  dropdownAlign?: ComponentProps<typeof Popup>['align']\n  portalTarget?: ComponentProps<typeof Popup>['portalTarget']\n}\n\nexport type CreateDropdownProps = {\n  isEmpty: boolean\n  emptyState: ReactNode\n  descriptionDirection: 'row' | 'column'\n  loadMore?: ReactNode\n  optionalInfoPlacement: 'left' | 'right'\n  defaultSearchValue: string | null\n  isLoading?: boolean\n}\n\nconst NON_SEARCHABLE_KEYS = [\n  'Tab',\n  ' ',\n  'Enter',\n  'CapsLock',\n  'Shift',\n  'ArrowDown',\n  'ArrowUp',\n  'ArrowLeft',\n  'ArrowRight',\n  'Escape',\n]\n\nconst StyledPopup = styled(Popup)`\n  width: 100%;\n  min-width: 320px;\n  background-color: ${({ theme }) =>\n    theme.colors.other.elevation.background.raised};\n  color: ${({ theme }) => theme.colors.neutral.text};\n  box-shadow: ${({ theme }) =>\n    `${theme.shadows.raised[0]}, ${theme.shadows.raised[1]}`};\n  padding: ${({ theme }) => theme.space[0]};\n  margin-bottom: ${({ theme }) => theme.space[10]};\n`\n\nconst DropdownContainer = styled(Stack)<{ 'data-grouped': boolean }>`\n  max-height: ${DROPDOWN_MAX_HEIGHT}px;\n  overflow-y: auto;\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: 0;\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: 0;\n`\nconst DropdownItem = styled.div<{\n  'aria-selected': boolean\n  'aria-disabled': boolean\n}>`\n  text-align:left;\n  border: none;\n  background-color: ${({ theme }) =>\n    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  border: 1px transparent solid;\n\n  &:hover, :focus {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n    color: ${({ theme }) => theme.colors.primary.text};\n    cursor: pointer;\n  }\n\n  &[aria-selected='true'] {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n  }\n\n  &[aria-disabled=\"true\"] {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n  }\n\n  &[aria-disabled=\"true\"]:hover, [aria-disabled=\"true\"]:focus {\n    background-color: ${({ theme }) =>\n      theme.colors.neutral.backgroundStrongDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textStrongDisabled};\n    cursor: not-allowed;\n    outline: none;\n  }\n`\n\nconst PopupFooter = styled.div`\n  width: 100%;\n  padding: ${({ theme }) => theme.space[1.5]} ${({ theme }) => theme.space[2]}\n    ${({ theme }) => theme.space[1.5]} ${({ theme }) => theme.space[2]};\n  box-shadow: ${({ theme }) => theme.shadows.dropdown};\n`\nconst StyledCheckbox = styled(Checkbox)`\n  width: 100%;\n  position: static;\n  text-align: left;\n  align-items: center;\n  pointer-events: none;\n` // pointer-events: none prevents any error when using the checkbox in a form since it is an unnamed input\n\nconst EmptyState = styled(Stack)`\n  padding: ${({ theme }) => theme.space[2]};\n`\nconst LoadMore = styled(Stack)`\n  padding: ${({ theme }) => theme.space[0.5]};\n`\n\nconst moveFocusDown = () => {\n  const options = document.querySelectorAll(\n    '#items > div[role=\"option\"]:not([disabled])',\n  )\n  const activeItem = document.activeElement\n  if (options) {\n    for (let i = 0; i < options?.length; i += 1) {\n      const listLength = options.length\n      if (activeItem === options[i] && activeItem !== options[listLength - 1]) {\n        ;(options[i + 1] as HTMLElement).focus()\n      }\n    }\n  }\n}\nconst moveFocusUp = () => {\n  const options = document.querySelectorAll(\n    '#items > div[role=\"option\"]:not([disabled])',\n  )\n  const activeItem = document.activeElement\n\n  if (options) {\n    for (let i = 0; i < options.length; i += 1) {\n      if (activeItem === options[i] && activeItem !== options[0]) {\n        ;(options[i - 1] as HTMLElement).focus()\n      }\n    }\n  }\n}\nconst handleKeyDownSelect = (event: KeyboardEvent<HTMLDivElement>) => {\n  if (event.key === 'ArrowDown') {\n    event.preventDefault()\n    moveFocusDown()\n  }\n\n  if (event.key === 'ArrowUp') {\n    event.preventDefault()\n    moveFocusUp()\n  }\n\n  if (event.key === ' ') {\n    // No scroll\n    event.preventDefault()\n  }\n}\n\nconst handleKeyDown = (\n  event: globalThis.KeyboardEvent,\n  ref: RefObject<HTMLDivElement | null>,\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].find(option =>\n        option.searchText\n          ? option.searchText.toLocaleLowerCase().startsWith(currentSearch)\n          : option.value.toLocaleLowerCase().startsWith(currentSearch),\n      )\n      if (closestOption) {\n        setDefaultSearch(closestOption.searchText ?? closestOption.value)\n      } else {\n        setDefaultSearch(null)\n      }\n    }\n  }\n}\nconst CreateDropdown = ({\n  isEmpty,\n  emptyState,\n  descriptionDirection,\n  loadMore,\n  optionalInfoPlacement,\n  defaultSearchValue,\n  isLoading,\n}: CreateDropdownProps) => {\n  const {\n    setIsDropdownVisible,\n    onChange,\n    options,\n    multiselect,\n    selectAll,\n    selectAllGroup,\n    displayedOptions,\n    setSelectedData,\n    selectedData,\n  } = useSelectInput()\n  const focusedItemRef = useRef<HTMLDivElement>(null)\n\n  useEffect(() => {\n    if (defaultSearchValue && focusedItemRef?.current) {\n      focusedItemRef.current.focus()\n    }\n  }, [defaultSearchValue])\n\n  if (isEmpty) {\n    return (\n      <EmptyState alignItems=\"center\" gap={2}>\n        {emptyState ?? (\n          <Text as=\"p\" variant=\"bodyStrong\">\n            No options\n          </Text>\n        )}\n      </EmptyState>\n    )\n  }\n\n  const handleClick = (clickedOption: OptionType, group?: string) => {\n    setSelectedData({ clickedOption, group, type: 'selectOption' })\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({ selectedGroup: group, type: 'selectGroup' })\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      data-grouped\n      id=\"select-dropdown\"\n      onKeyDown={handleKeyDownSelect}\n      role=\"listbox\"\n    >\n      {isLoading ? (\n        <Skeleton variant=\"block\" />\n      ) : (\n        <>\n          {selectAll && multiselect ? (\n            <Stack id=\"items\">\n              <DropdownItem\n                aria-disabled={false}\n                aria-label=\"select-all\"\n                aria-selected={selectedData.allSelected}\n                data-testid=\"select-all\"\n                id=\"select-all\"\n                onClick={selectAllOptions}\n                onKeyDown={event =>\n                  [' ', 'Enter'].includes(event.key) ? selectAllOptions() : null\n                }\n                role=\"option\"\n                tabIndex={0}\n              >\n                <StyledCheckbox\n                  checked={selectedData.allSelected}\n                  data-testid=\"select-all-checkbox\"\n                  disabled={false}\n                  onChange={selectAllOptions}\n                  tabIndex={-1}\n                  value=\"select-all\"\n                >\n                  <Stack direction=\"column\">\n                    <Text as=\"span\" placement=\"left\" variant=\"body\">\n                      {selectAll.label}\n                    </Text>\n                    <Text\n                      as=\"span\"\n                      placement=\"left\"\n                      prominence=\"weak\"\n                      sentiment=\"neutral\"\n                      variant=\"bodySmall\"\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 gap={0.25} key={group}>\n              {displayedOptions[group].length > 0 ? (\n                <DropdownGroupWrapper id={selectAllGroup ? 'items' : undefined}>\n                  {group ? (\n                    <DropdownGroup\n                      data-selectgroup={selectAllGroup}\n                      data-testid={`group-${index}`}\n                      key={group}\n                      onClick={() =>\n                        selectAllGroup ? handleSelectGroup(group) : null\n                      }\n                      onKeyDown={event => {\n                        if ([' ', 'Enter'].includes(event.key)) {\n                          event.preventDefault()\n                          handleSelectGroup(group)\n                        }\n                      }}\n                      role=\"group\"\n                      tabIndex={selectAllGroup ? 0 : -1}\n                      type=\"button\"\n                    >\n                      {selectAllGroup ? (\n                        <StyledCheckbox\n                          checked={selectedData.selectedGroups.includes(group)}\n                          data-testid=\"select-group\"\n                          disabled={false}\n                          onChange={() =>\n                            selectAllGroup ? handleSelectGroup(group) : null\n                          }\n                          tabIndex={-1}\n                          value={group}\n                        >\n                          <Text\n                            as=\"span\"\n                            placement=\"left\"\n                            sentiment=\"neutral\"\n                            variant=\"caption\"\n                          >\n                            {group.toUpperCase()}\n                          </Text>\n                        </StyledCheckbox>\n                      ) : (\n                        <Text\n                          as=\"span\"\n                          placement=\"left\"\n                          sentiment=\"neutral\"\n                          variant=\"caption\"\n                        >\n                          {group.toUpperCase()}\n                        </Text>\n                      )}\n                    </DropdownGroup>\n                  ) : null}\n                </DropdownGroupWrapper>\n              ) : null}\n              <Stack gap=\"0.25\" id=\"items\">\n                {displayedOptions[group].map((option, indexOption) => (\n                  <DropdownItem\n                    aria-disabled={!!option.disabled}\n                    aria-label={option.value}\n                    aria-selected={\n                      selectedData.selectedValues.includes(option.value) &&\n                      !option.disabled\n                    }\n                    data-testid={`option-${option.value}`}\n                    id={`option-${indexOption}`}\n                    key={option.value}\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                    role=\"option\"\n                    tabIndex={!option.disabled ? 0 : -1}\n                  >\n                    {multiselect ? (\n                      <StyledCheckbox\n                        checked={\n                          selectedData.selectedValues.includes(option.value) &&\n                          !option.disabled\n                        }\n                        disabled={option.disabled}\n                        onChange={() => {\n                          if (!option.disabled) {\n                            handleClick(option, group)\n                          }\n                        }}\n                        tabIndex={-1}\n                        value={option.value}\n                      >\n                        <DisplayOption\n                          descriptionDirection={descriptionDirection}\n                          option={option}\n                          optionalInfoPlacement={optionalInfoPlacement}\n                        />\n                      </StyledCheckbox>\n                    ) : (\n                      <DisplayOption\n                        descriptionDirection={descriptionDirection}\n                        option={option}\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      data-grouped={false}\n      gap={0.25}\n      id=\"select-dropdown\"\n      onKeyDown={handleKeyDownSelect}\n      role=\"listbox\"\n    >\n      {selectAll && multiselect ? (\n        <Stack gap={0.25} id=\"items\" tabIndex={-1}>\n          <DropdownItem\n            aria-disabled={false}\n            aria-label=\"select-all\"\n            aria-selected={selectedData.allSelected}\n            data-testid=\"select-all\"\n            onClick={selectAllOptions}\n            onKeyDown={event =>\n              [' ', 'Enter'].includes(event.key) ? selectAllOptions() : null\n            }\n            role=\"option\"\n            tabIndex={0}\n          >\n            <StyledCheckbox\n              checked={selectedData.allSelected}\n              data-testid=\"select-all-checkbox\"\n              disabled={false}\n              onChange={selectAllOptions}\n              tabIndex={-1}\n              value=\"select-all\"\n            >\n              <Stack direction=\"column\">\n                <Text as=\"span\" placement=\"left\" variant=\"body\">\n                  {selectAll.label}\n                </Text>\n                <Text\n                  as=\"span\"\n                  placement=\"left\"\n                  prominence=\"weak\"\n                  sentiment=\"neutral\"\n                  variant=\"bodySmall\"\n                >\n                  {selectAll.description}\n                </Text>\n              </Stack>\n            </StyledCheckbox>\n          </DropdownItem>\n        </Stack>\n      ) : null}\n      <Stack gap={0.25} id=\"items\">\n        {isLoading ? (\n          <Skeleton variant=\"block\" />\n        ) : (\n          displayedOptions.map((option, index) => (\n            <DropdownItem\n              aria-disabled={!!option.disabled}\n              aria-label={option.value}\n              aria-selected={\n                selectedData.selectedValues.includes(option.value) &&\n                !option.disabled\n              }\n              data-testid={`option-${option.value}`}\n              id={`option-${index}`}\n              key={option.value}\n              onClick={() => {\n                if (!option.disabled) {\n                  handleClick(option)\n                }\n              }}\n              onKeyDown={event =>\n                [' ', 'Enter'].includes(event.key) ? handleClick(option) : null\n              }\n              ref={\n                option.value === defaultSearchValue ||\n                option.searchText === defaultSearchValue\n                  ? focusedItemRef\n                  : null\n              }\n              role=\"option\"\n              tabIndex={!option.disabled ? 0 : -1}\n            >\n              {multiselect ? (\n                <StyledCheckbox\n                  checked={\n                    selectedData.selectedValues.includes(option.value) &&\n                    !option.disabled\n                  }\n                  disabled={option.disabled}\n                  onChange={() => {\n                    if (!option.disabled) {\n                      handleClick(option)\n                    }\n                  }}\n                  tabIndex={-1}\n                  value={option.value}\n                >\n                  <DisplayOption\n                    descriptionDirection={descriptionDirection}\n                    option={option}\n                    optionalInfoPlacement={optionalInfoPlacement}\n                  />\n                </StyledCheckbox>\n              ) : (\n                <DisplayOption\n                  descriptionDirection={descriptionDirection}\n                  option={option}\n                  optionalInfoPlacement={optionalInfoPlacement}\n                />\n              )}\n            </DropdownItem>\n          ))\n        )}\n        {loadMore ? <LoadMore>{loadMore}</LoadMore> : null}\n      </Stack>\n    </DropdownContainer>\n  )\n}\nexport const Dropdown = ({\n  children,\n  emptyState,\n  descriptionDirection,\n  searchable,\n  placeholder,\n  footer,\n  refSelect,\n  loadMore,\n  optionalInfoPlacement,\n  isLoading,\n  size,\n  dropdownAlign,\n  portalTarget,\n  id,\n}: DropdownProps) => {\n  const {\n    setIsDropdownVisible,\n    isDropdownVisible,\n    onSearch,\n    searchInput,\n    options,\n    displayedOptions,\n    numberOfOptions,\n  } = useSelectInput()\n  const theme = useTheme()\n  const [searchBarActive, setSearchBarActive] = useState(false)\n  const [defaultSearchValue, setDefaultSearch] = useState<string | null>(null)\n  const ref = useRef<HTMLDivElement>(null)\n  const [search, setSearch] = useState<string>('')\n  const [maxWidth, setWidth] = useState<string | number>()\n  const modalContext = useContext(ModalContext)\n\n  useEffect(() => {\n    if (refSelect.current && isDropdownVisible) {\n      const position =\n        refSelect.current.getBoundingClientRect().bottom +\n        DROPDOWN_MAX_HEIGHT +\n        Number(theme.sizing[INPUT_SIZE_HEIGHT[size]].replace('rem', '')) * 16 +\n        Number.parseInt(theme.space['5'], 10)\n      const overflow = position - window.innerHeight + 32\n      if (overflow > 0 && modalContext) {\n        const currentModal = modalContext.openedModals[0]\n        const modalElement = currentModal?.ref.current\n\n        if (modalElement) {\n          const parentElement = modalElement.parentNode as HTMLElement\n          if (parentElement) {\n            parentElement.scrollBy({\n              behavior: 'smooth',\n              top: overflow,\n            })\n          }\n        } else {\n          window.scrollBy({ behavior: 'smooth', top: overflow })\n        }\n      }\n    }\n    // oxlint-disable-next-line react/exhaustive-deps\n  }, [isDropdownVisible, refSelect, size, ref.current])\n\n  const resizeDropdown = useCallback(() => {\n    if (\n      refSelect.current &&\n      refSelect.current.getBoundingClientRect().width > 0\n    ) {\n      setWidth(refSelect.current.getBoundingClientRect().width)\n    }\n  }, [refSelect])\n\n  useEffect(() => {\n    resizeDropdown()\n\n    window.addEventListener('resize', resizeDropdown)\n\n    return () => window.removeEventListener('resize', resizeDropdown)\n  }, [resizeDropdown])\n\n  useEffect(() => {\n    if (!searchInput) {\n      onSearch(options)\n    }\n  }, [onSearch, options, searchInput])\n\n  useEffect(() => {\n    if (!isDropdownVisible) {\n      setDefaultSearch(null)\n      setSearch('')\n    }\n\n    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      if (!searchable) {\n        document.removeEventListener('keydown', event =>\n          handleKeyDown(\n            event,\n            ref,\n            options,\n            searchBarActive,\n            setSearch,\n            setDefaultSearch,\n            search,\n          ),\n        )\n      }\n    }\n  }, [\n    isDropdownVisible,\n    searchBarActive,\n    options,\n    onSearch,\n    search,\n    refSelect,\n    setDefaultSearch,\n    setIsDropdownVisible,\n    searchable,\n  ])\n\n  // No data is displayed (because of the search or because no data is provided)\n  // Set to true when noData by default\n  const isEmpty = useMemo(() => {\n    if (numberOfOptions === 0) {\n      return true\n    }\n    if (Array.isArray(displayedOptions)) {\n      return displayedOptions.length === 0\n    }\n    const groups = Object.keys(displayedOptions)\n    for (const group of groups) {\n      if (displayedOptions[group].length > 0) {\n        return false\n      }\n    }\n\n    return true\n  }, [displayedOptions, numberOfOptions])\n\n  const computedFooter = useMemo(() => {\n    if (footer && !isEmpty) {\n      if (typeof footer === 'function') {\n        return (\n          <PopupFooter>{footer(() => setIsDropdownVisible(false))}</PopupFooter>\n        )\n      }\n\n      return <PopupFooter>{footer}</PopupFooter>\n    }\n\n    return null\n  }, [isEmpty, footer, setIsDropdownVisible])\n\n  return (\n    <StyledPopup\n      align={dropdownAlign ?? 'start'}\n      containerFullWidth\n      debounceDelay={0}\n      disableAnimation\n      hasArrow={false}\n      hideOnClickOutside\n      id={id}\n      maxWidth={maxWidth ?? refSelect.current?.offsetWidth}\n      onClose={() => setIsDropdownVisible(false)}\n      placement=\"bottom\"\n      portalTarget={portalTarget}\n      ref={ref}\n      role=\"dialog\"\n      tabIndex={-1}\n      text={\n        <Stack>\n          {searchable && !isLoading && numberOfOptions >= 6 ? (\n            <SearchBarDropdown\n              displayedOptions={displayedOptions}\n              placeholder={placeholder}\n              setSearchBarActive={setSearchBarActive}\n            />\n          ) : null}\n          <CreateDropdown\n            defaultSearchValue={defaultSearchValue}\n            descriptionDirection={descriptionDirection}\n            emptyState={emptyState}\n            isEmpty={isEmpty}\n            isLoading={isLoading}\n            loadMore={loadMore}\n            optionalInfoPlacement={optionalInfoPlacement}\n          />\n          {computedFooter}\n        </Stack>\n      }\n      visible={isDropdownVisible}\n    >\n      {children}\n    </StyledPopup>\n  )\n}\n"]} */",
|
|
87
|
+
styles: "position:sticky;top:0/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInput/Dropdown.tsx"],"names":[],"mappings":"AAgIuC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInput/Dropdown.tsx","sourcesContent":["'use client'\n\nimport { useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport type {\n  ComponentProps,\n  Dispatch,\n  KeyboardEvent,\n  ReactNode,\n  RefObject,\n  SetStateAction,\n} from 'react'\nimport {\n  useCallback,\n  useContext,\n  useEffect,\n  useMemo,\n  useRef,\n  useState,\n} from 'react'\nimport { Checkbox } from '../Checkbox'\nimport { ModalContext } from '../Modal/ModalProvider'\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'\nimport { INPUT_SIZE_HEIGHT } from './types'\n\nconst DROPDOWN_MAX_HEIGHT = 256\n\nexport type DropdownProps = {\n  id?: string\n  children: ReactNode\n  emptyState: ReactNode\n  descriptionDirection: 'row' | 'column'\n  searchable: boolean\n  placeholder: string\n  footer?: ((closeDropdown: () => void) => ReactNode) | ReactNode\n  refSelect: RefObject<HTMLDivElement | null>\n  loadMore?: ReactNode\n  optionalInfoPlacement: 'left' | 'right'\n  isLoading?: boolean\n  size: 'small' | 'medium' | 'large'\n  dropdownAlign?: ComponentProps<typeof Popup>['align']\n  portalTarget?: ComponentProps<typeof Popup>['portalTarget']\n}\n\nexport type CreateDropdownProps = {\n  isEmpty: boolean\n  emptyState: ReactNode\n  descriptionDirection: 'row' | 'column'\n  loadMore?: ReactNode\n  optionalInfoPlacement: 'left' | 'right'\n  defaultSearchValue: string | null\n  isLoading?: boolean\n}\n\nconst NON_SEARCHABLE_KEYS = [\n  'Tab',\n  ' ',\n  'Enter',\n  'CapsLock',\n  'Shift',\n  'ArrowDown',\n  'ArrowUp',\n  'ArrowLeft',\n  'ArrowRight',\n  'Escape',\n]\n\nconst StyledPopup = styled(Popup)`\n  width: 100%;\n  min-width: 320px;\n  background-color: ${({ theme }) =>\n    theme.colors.other.elevation.background.raised};\n  color: ${({ theme }) => theme.colors.neutral.text};\n  box-shadow: ${({ theme }) =>\n    `${theme.shadows.raised[0]}, ${theme.shadows.raised[1]}`};\n  padding: ${({ theme }) => theme.space[0]};\n  margin-bottom: ${({ theme }) => theme.space[10]};\n`\n\nconst DropdownContainer = styled(Stack)<{ 'data-grouped': boolean }>`\n  max-height: ${DROPDOWN_MAX_HEIGHT}px;\n  overflow-y: auto;\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: 0;\n  padding-right: ${({ theme }) => theme.space[2]};\n  padding-left: ${({ theme }) => theme.space[2]};\n  height: ${({ theme }) => theme.space[4]};\n  display: flex;\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: 0;\n`\nconst DropdownItem = styled.div<{\n  'aria-selected': boolean\n  'aria-disabled': boolean\n}>`\n  text-align:left;\n  border: none;\n  background-color: ${({ theme }) =>\n    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  border: 1px transparent solid;\n\n  &:hover, :focus {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n    color: ${({ theme }) => theme.colors.primary.text};\n    cursor: pointer;\n  }\n\n  &[aria-selected='true'] {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n  }\n\n  &[aria-disabled=\"true\"] {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n  }\n\n  &[aria-disabled=\"true\"]:hover, [aria-disabled=\"true\"]:focus {\n    background-color: ${({ theme }) =>\n      theme.colors.neutral.backgroundStrongDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textStrongDisabled};\n    cursor: not-allowed;\n    outline: none;\n  }\n`\n\nconst PopupFooter = styled.div`\n  width: 100%;\n  padding: ${({ theme }) => theme.space[1.5]} ${({ theme }) => theme.space[2]}\n    ${({ theme }) => theme.space[1.5]} ${({ theme }) => theme.space[2]};\n  box-shadow: ${({ theme }) => theme.shadows.dropdown};\n`\nconst StyledCheckbox = styled(Checkbox)`\n  width: 100%;\n  position: static;\n  text-align: left;\n  align-items: center;\n  pointer-events: none;\n` // pointer-events: none prevents any error when using the checkbox in a form since it is an unnamed input\n\nconst EmptyState = styled(Stack)`\n  padding: ${({ theme }) => theme.space[2]};\n`\nconst LoadMore = styled(Stack)`\n  padding: ${({ theme }) => theme.space[0.5]};\n`\n\nconst moveFocusDown = () => {\n  const options = document.querySelectorAll(\n    '#items > div[role=\"option\"]:not([disabled])',\n  )\n  const activeItem = document.activeElement\n  if (options) {\n    for (let i = 0; i < options?.length; i += 1) {\n      const listLength = options.length\n      if (activeItem === options[i] && activeItem !== options[listLength - 1]) {\n        ;(options[i + 1] as HTMLElement).focus()\n      }\n    }\n  }\n}\nconst moveFocusUp = () => {\n  const options = document.querySelectorAll(\n    '#items > div[role=\"option\"]:not([disabled])',\n  )\n  const activeItem = document.activeElement\n\n  if (options) {\n    for (let i = 0; i < options.length; i += 1) {\n      if (activeItem === options[i] && activeItem !== options[0]) {\n        ;(options[i - 1] as HTMLElement).focus()\n      }\n    }\n  }\n}\nconst handleKeyDownSelect = (event: KeyboardEvent<HTMLDivElement>) => {\n  if (event.key === 'ArrowDown') {\n    event.preventDefault()\n    moveFocusDown()\n  }\n\n  if (event.key === 'ArrowUp') {\n    event.preventDefault()\n    moveFocusUp()\n  }\n\n  if (event.key === ' ') {\n    // No scroll\n    event.preventDefault()\n  }\n}\n\nconst handleKeyDown = (\n  event: globalThis.KeyboardEvent,\n  ref: RefObject<HTMLDivElement | null>,\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].find(option =>\n        option.searchText\n          ? option.searchText.toLocaleLowerCase().startsWith(currentSearch)\n          : option.value.toLocaleLowerCase().startsWith(currentSearch),\n      )\n      if (closestOption) {\n        setDefaultSearch(closestOption.searchText ?? closestOption.value)\n      } else {\n        setDefaultSearch(null)\n      }\n    }\n  }\n}\nconst CreateDropdown = ({\n  isEmpty,\n  emptyState,\n  descriptionDirection,\n  loadMore,\n  optionalInfoPlacement,\n  defaultSearchValue,\n  isLoading,\n}: CreateDropdownProps) => {\n  const {\n    setIsDropdownVisible,\n    onChange,\n    options,\n    multiselect,\n    selectAll,\n    selectAllGroup,\n    displayedOptions,\n    setSelectedData,\n    selectedData,\n  } = useSelectInput()\n  const focusedItemRef = useRef<HTMLDivElement>(null)\n\n  useEffect(() => {\n    if (defaultSearchValue && focusedItemRef?.current) {\n      focusedItemRef.current.focus()\n    }\n  }, [defaultSearchValue])\n\n  if (isEmpty) {\n    return (\n      <EmptyState alignItems=\"center\" gap={2}>\n        {emptyState ?? (\n          <Text as=\"p\" variant=\"bodyStrong\">\n            No options\n          </Text>\n        )}\n      </EmptyState>\n    )\n  }\n\n  const handleClick = (clickedOption: OptionType, group?: string) => {\n    setSelectedData({ clickedOption, group, type: 'selectOption' })\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({ selectedGroup: group, type: 'selectGroup' })\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      data-grouped\n      id=\"select-dropdown\"\n      onKeyDown={handleKeyDownSelect}\n      role=\"listbox\"\n    >\n      {isLoading ? (\n        <Skeleton variant=\"block\" />\n      ) : (\n        <>\n          {selectAll && multiselect ? (\n            <Stack id=\"items\">\n              <DropdownItem\n                aria-disabled={false}\n                aria-label=\"select-all\"\n                aria-selected={selectedData.allSelected}\n                data-testid=\"select-all\"\n                id=\"select-all\"\n                onClick={selectAllOptions}\n                onKeyDown={event =>\n                  [' ', 'Enter'].includes(event.key) ? selectAllOptions() : null\n                }\n                role=\"option\"\n                tabIndex={0}\n              >\n                <StyledCheckbox\n                  checked={selectedData.allSelected}\n                  data-testid=\"select-all-checkbox\"\n                  disabled={false}\n                  onChange={selectAllOptions}\n                  tabIndex={-1}\n                  value=\"select-all\"\n                >\n                  <Stack direction=\"column\">\n                    <Text as=\"span\" placement=\"left\" variant=\"body\">\n                      {selectAll.label}\n                    </Text>\n                    <Text\n                      as=\"span\"\n                      placement=\"left\"\n                      prominence=\"weak\"\n                      sentiment=\"neutral\"\n                      variant=\"bodySmall\"\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 gap={0.25} key={group}>\n              {displayedOptions[group].length > 0 ? (\n                <DropdownGroupWrapper id={selectAllGroup ? 'items' : undefined}>\n                  {group ? (\n                    <DropdownGroup\n                      data-selectgroup={selectAllGroup}\n                      data-testid={`group-${index}`}\n                      key={group}\n                      onClick={() =>\n                        selectAllGroup ? handleSelectGroup(group) : null\n                      }\n                      onKeyDown={event => {\n                        if ([' ', 'Enter'].includes(event.key)) {\n                          event.preventDefault()\n                          handleSelectGroup(group)\n                        }\n                      }}\n                      role=\"group\"\n                      tabIndex={selectAllGroup ? 0 : -1}\n                      type=\"button\"\n                    >\n                      {selectAllGroup ? (\n                        <StyledCheckbox\n                          checked={selectedData.selectedGroups.includes(group)}\n                          data-testid=\"select-group\"\n                          disabled={false}\n                          onChange={() =>\n                            selectAllGroup ? handleSelectGroup(group) : null\n                          }\n                          tabIndex={-1}\n                          value={group}\n                        >\n                          <Text\n                            as=\"span\"\n                            placement=\"left\"\n                            sentiment=\"neutral\"\n                            variant=\"caption\"\n                          >\n                            {group.toUpperCase()}\n                          </Text>\n                        </StyledCheckbox>\n                      ) : (\n                        <Text\n                          as=\"span\"\n                          placement=\"left\"\n                          sentiment=\"neutral\"\n                          variant=\"caption\"\n                        >\n                          {group.toUpperCase()}\n                        </Text>\n                      )}\n                    </DropdownGroup>\n                  ) : null}\n                </DropdownGroupWrapper>\n              ) : null}\n              <Stack gap=\"0.25\" id=\"items\">\n                {displayedOptions[group].map((option, indexOption) => (\n                  <DropdownItem\n                    aria-disabled={!!option.disabled}\n                    aria-label={option.value}\n                    aria-selected={\n                      selectedData.selectedValues.includes(option.value) &&\n                      !option.disabled\n                    }\n                    data-testid={`option-${option.value}`}\n                    id={`option-${indexOption}`}\n                    key={option.value}\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                    role=\"option\"\n                    tabIndex={!option.disabled ? 0 : -1}\n                  >\n                    {multiselect ? (\n                      <StyledCheckbox\n                        checked={\n                          selectedData.selectedValues.includes(option.value) &&\n                          !option.disabled\n                        }\n                        disabled={option.disabled}\n                        onChange={() => {\n                          if (!option.disabled) {\n                            handleClick(option, group)\n                          }\n                        }}\n                        tabIndex={-1}\n                        value={option.value}\n                      >\n                        <DisplayOption\n                          descriptionDirection={descriptionDirection}\n                          option={option}\n                          optionalInfoPlacement={optionalInfoPlacement}\n                        />\n                      </StyledCheckbox>\n                    ) : (\n                      <DisplayOption\n                        descriptionDirection={descriptionDirection}\n                        option={option}\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      data-grouped={false}\n      gap={0.25}\n      id=\"select-dropdown\"\n      onKeyDown={handleKeyDownSelect}\n      role=\"listbox\"\n    >\n      {selectAll && multiselect ? (\n        <Stack gap={0.25} id=\"items\" tabIndex={-1}>\n          <DropdownItem\n            aria-disabled={false}\n            aria-label=\"select-all\"\n            aria-selected={selectedData.allSelected}\n            data-testid=\"select-all\"\n            onClick={selectAllOptions}\n            onKeyDown={event =>\n              [' ', 'Enter'].includes(event.key) ? selectAllOptions() : null\n            }\n            role=\"option\"\n            tabIndex={0}\n          >\n            <StyledCheckbox\n              checked={selectedData.allSelected}\n              data-testid=\"select-all-checkbox\"\n              disabled={false}\n              onChange={selectAllOptions}\n              tabIndex={-1}\n              value=\"select-all\"\n            >\n              <Stack direction=\"column\">\n                <Text as=\"span\" placement=\"left\" variant=\"body\">\n                  {selectAll.label}\n                </Text>\n                <Text\n                  as=\"span\"\n                  placement=\"left\"\n                  prominence=\"weak\"\n                  sentiment=\"neutral\"\n                  variant=\"bodySmall\"\n                >\n                  {selectAll.description}\n                </Text>\n              </Stack>\n            </StyledCheckbox>\n          </DropdownItem>\n        </Stack>\n      ) : null}\n      <Stack gap={0.25} id=\"items\">\n        {isLoading ? (\n          <Skeleton variant=\"block\" />\n        ) : (\n          displayedOptions.map((option, index) => (\n            <DropdownItem\n              aria-disabled={!!option.disabled}\n              aria-label={option.value}\n              aria-selected={\n                selectedData.selectedValues.includes(option.value) &&\n                !option.disabled\n              }\n              data-testid={`option-${option.value}`}\n              id={`option-${index}`}\n              key={option.value}\n              onClick={() => {\n                if (!option.disabled) {\n                  handleClick(option)\n                }\n              }}\n              onKeyDown={event =>\n                [' ', 'Enter'].includes(event.key) ? handleClick(option) : null\n              }\n              ref={\n                option.value === defaultSearchValue ||\n                option.searchText === defaultSearchValue\n                  ? focusedItemRef\n                  : null\n              }\n              role=\"option\"\n              tabIndex={!option.disabled ? 0 : -1}\n            >\n              {multiselect ? (\n                <StyledCheckbox\n                  checked={\n                    selectedData.selectedValues.includes(option.value) &&\n                    !option.disabled\n                  }\n                  disabled={option.disabled}\n                  onChange={() => {\n                    if (!option.disabled) {\n                      handleClick(option)\n                    }\n                  }}\n                  tabIndex={-1}\n                  value={option.value}\n                >\n                  <DisplayOption\n                    descriptionDirection={descriptionDirection}\n                    option={option}\n                    optionalInfoPlacement={optionalInfoPlacement}\n                  />\n                </StyledCheckbox>\n              ) : (\n                <DisplayOption\n                  descriptionDirection={descriptionDirection}\n                  option={option}\n                  optionalInfoPlacement={optionalInfoPlacement}\n                />\n              )}\n            </DropdownItem>\n          ))\n        )}\n        {loadMore ? <LoadMore>{loadMore}</LoadMore> : null}\n      </Stack>\n    </DropdownContainer>\n  )\n}\nexport const Dropdown = ({\n  children,\n  emptyState,\n  descriptionDirection,\n  searchable,\n  placeholder,\n  footer,\n  refSelect,\n  loadMore,\n  optionalInfoPlacement,\n  isLoading,\n  size,\n  dropdownAlign,\n  portalTarget,\n  id,\n}: DropdownProps) => {\n  const {\n    setIsDropdownVisible,\n    isDropdownVisible,\n    onSearch,\n    searchInput,\n    options,\n    displayedOptions,\n    numberOfOptions,\n  } = useSelectInput()\n  const theme = useTheme()\n  const [searchBarActive, setSearchBarActive] = useState(false)\n  const [defaultSearchValue, setDefaultSearch] = useState<string | null>(null)\n  const ref = useRef<HTMLDivElement>(null)\n  const [search, setSearch] = useState<string>('')\n  const [maxWidth, setWidth] = useState<string | number>()\n  const modalContext = useContext(ModalContext)\n\n  useEffect(() => {\n    if (refSelect.current && isDropdownVisible) {\n      const position =\n        refSelect.current.getBoundingClientRect().bottom +\n        DROPDOWN_MAX_HEIGHT +\n        Number(theme.sizing[INPUT_SIZE_HEIGHT[size]].replace('rem', '')) * 16 +\n        Number.parseInt(theme.space['5'], 10)\n      const overflow = position - window.innerHeight + 32\n      if (overflow > 0 && modalContext) {\n        const currentModal = modalContext.openedModals[0]\n        const modalElement = currentModal?.ref.current\n\n        if (modalElement) {\n          const parentElement = modalElement.parentNode as HTMLElement\n          if (parentElement) {\n            parentElement.scrollBy({\n              behavior: 'smooth',\n              top: overflow,\n            })\n          }\n        } else {\n          window.scrollBy({ behavior: 'smooth', top: overflow })\n        }\n      }\n    }\n    // oxlint-disable-next-line react/exhaustive-deps\n  }, [isDropdownVisible, refSelect, size, ref.current])\n\n  const resizeDropdown = useCallback(() => {\n    if (\n      refSelect.current &&\n      refSelect.current.getBoundingClientRect().width > 0\n    ) {\n      setWidth(refSelect.current.getBoundingClientRect().width)\n    }\n  }, [refSelect])\n\n  useEffect(() => {\n    resizeDropdown()\n\n    window.addEventListener('resize', resizeDropdown)\n\n    return () => window.removeEventListener('resize', resizeDropdown)\n  }, [resizeDropdown])\n\n  useEffect(() => {\n    if (!searchInput) {\n      onSearch(options)\n    }\n  }, [onSearch, options, searchInput])\n\n  useEffect(() => {\n    if (!isDropdownVisible) {\n      setDefaultSearch(null)\n      setSearch('')\n    }\n\n    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      if (!searchable) {\n        document.removeEventListener('keydown', event =>\n          handleKeyDown(\n            event,\n            ref,\n            options,\n            searchBarActive,\n            setSearch,\n            setDefaultSearch,\n            search,\n          ),\n        )\n      }\n    }\n  }, [\n    isDropdownVisible,\n    searchBarActive,\n    options,\n    onSearch,\n    search,\n    refSelect,\n    setDefaultSearch,\n    setIsDropdownVisible,\n    searchable,\n  ])\n\n  // No data is displayed (because of the search or because no data is provided)\n  // Set to true when noData by default\n  const isEmpty = useMemo(() => {\n    if (numberOfOptions === 0) {\n      return true\n    }\n    if (Array.isArray(displayedOptions)) {\n      return displayedOptions.length === 0\n    }\n    const groups = Object.keys(displayedOptions)\n    for (const group of groups) {\n      if (displayedOptions[group].length > 0) {\n        return false\n      }\n    }\n\n    return true\n  }, [displayedOptions, numberOfOptions])\n\n  const computedFooter = useMemo(() => {\n    if (footer && !isEmpty) {\n      if (typeof footer === 'function') {\n        return (\n          <PopupFooter>{footer(() => setIsDropdownVisible(false))}</PopupFooter>\n        )\n      }\n\n      return <PopupFooter>{footer}</PopupFooter>\n    }\n\n    return null\n  }, [isEmpty, footer, setIsDropdownVisible])\n\n  return (\n    <StyledPopup\n      align={dropdownAlign ?? 'start'}\n      containerFullWidth\n      debounceDelay={0}\n      disableAnimation\n      hasArrow={false}\n      hideOnClickOutside\n      id={id}\n      maxWidth={maxWidth ?? refSelect.current?.offsetWidth}\n      onClose={() => setIsDropdownVisible(false)}\n      placement=\"bottom\"\n      portalTarget={portalTarget}\n      ref={ref}\n      role=\"dialog\"\n      tabIndex={-1}\n      text={\n        <Stack>\n          {searchable && !isLoading && numberOfOptions >= 6 ? (\n            <SearchBarDropdown\n              displayedOptions={displayedOptions}\n              placeholder={placeholder}\n              setSearchBarActive={setSearchBarActive}\n            />\n          ) : null}\n          <CreateDropdown\n            defaultSearchValue={defaultSearchValue}\n            descriptionDirection={descriptionDirection}\n            emptyState={emptyState}\n            isEmpty={isEmpty}\n            isLoading={isLoading}\n            loadMore={loadMore}\n            optionalInfoPlacement={optionalInfoPlacement}\n          />\n          {computedFooter}\n        </Stack>\n      }\n      visible={isDropdownVisible}\n    >\n      {children}\n    </StyledPopup>\n  )\n}\n"]} */",
|
|
88
88
|
toString: _EMOTION_STRINGIFIED_CSS_ERROR__
|
|
89
89
|
});
|
|
90
90
|
const DropdownItem = /* @__PURE__ */ _styled("div", process.env.NODE_ENV === "production" ? {
|
|
@@ -124,7 +124,7 @@ const DropdownItem = /* @__PURE__ */ _styled("div", process.env.NODE_ENV === "pr
|
|
|
124
124
|
theme
|
|
125
125
|
}) => theme.colors.neutral.backgroundStrongDisabled, ";color:", ({
|
|
126
126
|
theme
|
|
127
|
-
}) => 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/SelectInput/Dropdown.tsx"],"names":[],"mappings":"AAsIE","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInput/Dropdown.tsx","sourcesContent":["'use client'\n\nimport { useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport type {\n  ComponentProps,\n  Dispatch,\n  KeyboardEvent,\n  ReactNode,\n  RefObject,\n  SetStateAction,\n} from 'react'\nimport {\n  useCallback,\n  useContext,\n  useEffect,\n  useMemo,\n  useRef,\n  useState,\n} from 'react'\nimport { Checkbox } from '../Checkbox'\nimport { ModalContext } from '../Modal/ModalProvider'\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'\nimport { INPUT_SIZE_HEIGHT } from './types'\n\nconst DROPDOWN_MAX_HEIGHT = 256\n\nexport type DropdownProps = {\n  id?: string\n  children: ReactNode\n  emptyState: ReactNode\n  descriptionDirection: 'row' | 'column'\n  searchable: boolean\n  placeholder: string\n  footer?: ((closeDropdown: () => void) => ReactNode) | ReactNode\n  refSelect: RefObject<HTMLDivElement | null>\n  loadMore?: ReactNode\n  optionalInfoPlacement: 'left' | 'right'\n  isLoading?: boolean\n  size: 'small' | 'medium' | 'large'\n  dropdownAlign?: ComponentProps<typeof Popup>['align']\n  portalTarget?: ComponentProps<typeof Popup>['portalTarget']\n}\n\nexport type CreateDropdownProps = {\n  isEmpty: boolean\n  emptyState: ReactNode\n  descriptionDirection: 'row' | 'column'\n  loadMore?: ReactNode\n  optionalInfoPlacement: 'left' | 'right'\n  defaultSearchValue: string | null\n  isLoading?: boolean\n}\n\nconst NON_SEARCHABLE_KEYS = [\n  'Tab',\n  ' ',\n  'Enter',\n  'CapsLock',\n  'Shift',\n  'ArrowDown',\n  'ArrowUp',\n  'ArrowLeft',\n  'ArrowRight',\n  'Escape',\n]\n\nconst StyledPopup = styled(Popup)`\n  width: 100%;\n  min-width: 320px;\n  background-color: ${({ theme }) =>\n    theme.colors.other.elevation.background.raised};\n  color: ${({ theme }) => theme.colors.neutral.text};\n  box-shadow: ${({ theme }) =>\n    `${theme.shadows.raised[0]}, ${theme.shadows.raised[1]}`};\n  padding: ${({ theme }) => theme.space[0]};\n  margin-bottom: ${({ theme }) => theme.space[10]};\n`\n\nconst DropdownContainer = styled(Stack)<{ 'data-grouped': boolean }>`\n  max-height: ${DROPDOWN_MAX_HEIGHT}px;\n  overflow-y: auto;\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: 0;\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: 0;\n`\nconst DropdownItem = styled.div<{\n  'aria-selected': boolean\n  'aria-disabled': boolean\n}>`\n  text-align:left;\n  border: none;\n  background-color: ${({ theme }) =>\n    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  border: 1px transparent solid;\n\n  &:hover, :focus {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n    color: ${({ theme }) => theme.colors.primary.text};\n    cursor: pointer;\n  }\n\n  &[aria-selected='true'] {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n  }\n\n  &[aria-disabled=\"true\"] {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n  }\n\n  &[aria-disabled=\"true\"]:hover, [aria-disabled=\"true\"]:focus {\n    background-color: ${({ theme }) =>\n      theme.colors.neutral.backgroundStrongDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textStrongDisabled};\n    cursor: not-allowed;\n    outline: none;\n  }\n`\n\nconst PopupFooter = styled.div`\n  width: 100%;\n  padding: ${({ theme }) => theme.space[1.5]} ${({ theme }) => theme.space[2]}\n    ${({ theme }) => theme.space[1.5]} ${({ theme }) => theme.space[2]};\n  box-shadow: ${({ theme }) => theme.shadows.dropdown};\n`\nconst StyledCheckbox = styled(Checkbox)`\n  width: 100%;\n  position: static;\n  text-align: left;\n  align-items: center;\n  pointer-events: none;\n` // pointer-events: none prevents any error when using the checkbox in a form since it is an unnamed input\n\nconst EmptyState = styled(Stack)`\n  padding: ${({ theme }) => theme.space[2]};\n`\nconst LoadMore = styled(Stack)`\n  padding: ${({ theme }) => theme.space[0.5]};\n`\n\nconst moveFocusDown = () => {\n  const options = document.querySelectorAll(\n    '#items > div[role=\"option\"]:not([disabled])',\n  )\n  const activeItem = document.activeElement\n  if (options) {\n    for (let i = 0; i < options?.length; i += 1) {\n      const listLength = options.length\n      if (activeItem === options[i] && activeItem !== options[listLength - 1]) {\n        ;(options[i + 1] as HTMLElement).focus()\n      }\n    }\n  }\n}\nconst moveFocusUp = () => {\n  const options = document.querySelectorAll(\n    '#items > div[role=\"option\"]:not([disabled])',\n  )\n  const activeItem = document.activeElement\n\n  if (options) {\n    for (let i = 0; i < options.length; i += 1) {\n      if (activeItem === options[i] && activeItem !== options[0]) {\n        ;(options[i - 1] as HTMLElement).focus()\n      }\n    }\n  }\n}\nconst handleKeyDownSelect = (event: KeyboardEvent<HTMLDivElement>) => {\n  if (event.key === 'ArrowDown') {\n    event.preventDefault()\n    moveFocusDown()\n  }\n\n  if (event.key === 'ArrowUp') {\n    event.preventDefault()\n    moveFocusUp()\n  }\n\n  if (event.key === ' ') {\n    // No scroll\n    event.preventDefault()\n  }\n}\n\nconst handleKeyDown = (\n  event: globalThis.KeyboardEvent,\n  ref: RefObject<HTMLDivElement | null>,\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].find(option =>\n        option.searchText\n          ? option.searchText.toLocaleLowerCase().startsWith(currentSearch)\n          : option.value.toLocaleLowerCase().startsWith(currentSearch),\n      )\n      if (closestOption) {\n        setDefaultSearch(closestOption.searchText ?? closestOption.value)\n      } else {\n        setDefaultSearch(null)\n      }\n    }\n  }\n}\nconst CreateDropdown = ({\n  isEmpty,\n  emptyState,\n  descriptionDirection,\n  loadMore,\n  optionalInfoPlacement,\n  defaultSearchValue,\n  isLoading,\n}: CreateDropdownProps) => {\n  const {\n    setIsDropdownVisible,\n    onChange,\n    options,\n    multiselect,\n    selectAll,\n    selectAllGroup,\n    displayedOptions,\n    setSelectedData,\n    selectedData,\n  } = useSelectInput()\n  const focusedItemRef = useRef<HTMLDivElement>(null)\n\n  useEffect(() => {\n    if (defaultSearchValue && focusedItemRef?.current) {\n      focusedItemRef.current.focus()\n    }\n  }, [defaultSearchValue])\n\n  if (isEmpty) {\n    return (\n      <EmptyState alignItems=\"center\" gap={2}>\n        {emptyState ?? (\n          <Text as=\"p\" variant=\"bodyStrong\">\n            No options\n          </Text>\n        )}\n      </EmptyState>\n    )\n  }\n\n  const handleClick = (clickedOption: OptionType, group?: string) => {\n    setSelectedData({ clickedOption, group, type: 'selectOption' })\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({ selectedGroup: group, type: 'selectGroup' })\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      data-grouped\n      id=\"select-dropdown\"\n      onKeyDown={handleKeyDownSelect}\n      role=\"listbox\"\n    >\n      {isLoading ? (\n        <Skeleton variant=\"block\" />\n      ) : (\n        <>\n          {selectAll && multiselect ? (\n            <Stack id=\"items\">\n              <DropdownItem\n                aria-disabled={false}\n                aria-label=\"select-all\"\n                aria-selected={selectedData.allSelected}\n                data-testid=\"select-all\"\n                id=\"select-all\"\n                onClick={selectAllOptions}\n                onKeyDown={event =>\n                  [' ', 'Enter'].includes(event.key) ? selectAllOptions() : null\n                }\n                role=\"option\"\n                tabIndex={0}\n              >\n                <StyledCheckbox\n                  checked={selectedData.allSelected}\n                  data-testid=\"select-all-checkbox\"\n                  disabled={false}\n                  onChange={selectAllOptions}\n                  tabIndex={-1}\n                  value=\"select-all\"\n                >\n                  <Stack direction=\"column\">\n                    <Text as=\"span\" placement=\"left\" variant=\"body\">\n                      {selectAll.label}\n                    </Text>\n                    <Text\n                      as=\"span\"\n                      placement=\"left\"\n                      prominence=\"weak\"\n                      sentiment=\"neutral\"\n                      variant=\"bodySmall\"\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 gap={0.25} key={group}>\n              {displayedOptions[group].length > 0 ? (\n                <DropdownGroupWrapper id={selectAllGroup ? 'items' : undefined}>\n                  {group ? (\n                    <DropdownGroup\n                      data-selectgroup={selectAllGroup}\n                      data-testid={`group-${index}`}\n                      key={group}\n                      onClick={() =>\n                        selectAllGroup ? handleSelectGroup(group) : null\n                      }\n                      onKeyDown={event => {\n                        if ([' ', 'Enter'].includes(event.key)) {\n                          event.preventDefault()\n                          handleSelectGroup(group)\n                        }\n                      }}\n                      role=\"group\"\n                      tabIndex={selectAllGroup ? 0 : -1}\n                      type=\"button\"\n                    >\n                      {selectAllGroup ? (\n                        <StyledCheckbox\n                          checked={selectedData.selectedGroups.includes(group)}\n                          data-testid=\"select-group\"\n                          disabled={false}\n                          onChange={() =>\n                            selectAllGroup ? handleSelectGroup(group) : null\n                          }\n                          tabIndex={-1}\n                          value={group}\n                        >\n                          <Text\n                            as=\"span\"\n                            placement=\"left\"\n                            sentiment=\"neutral\"\n                            variant=\"caption\"\n                          >\n                            {group.toUpperCase()}\n                          </Text>\n                        </StyledCheckbox>\n                      ) : (\n                        <Text\n                          as=\"span\"\n                          placement=\"left\"\n                          sentiment=\"neutral\"\n                          variant=\"caption\"\n                        >\n                          {group.toUpperCase()}\n                        </Text>\n                      )}\n                    </DropdownGroup>\n                  ) : null}\n                </DropdownGroupWrapper>\n              ) : null}\n              <Stack gap=\"0.25\" id=\"items\">\n                {displayedOptions[group].map((option, indexOption) => (\n                  <DropdownItem\n                    aria-disabled={!!option.disabled}\n                    aria-label={option.value}\n                    aria-selected={\n                      selectedData.selectedValues.includes(option.value) &&\n                      !option.disabled\n                    }\n                    data-testid={`option-${option.value}`}\n                    id={`option-${indexOption}`}\n                    key={option.value}\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                    role=\"option\"\n                    tabIndex={!option.disabled ? 0 : -1}\n                  >\n                    {multiselect ? (\n                      <StyledCheckbox\n                        checked={\n                          selectedData.selectedValues.includes(option.value) &&\n                          !option.disabled\n                        }\n                        disabled={option.disabled}\n                        onChange={() => {\n                          if (!option.disabled) {\n                            handleClick(option, group)\n                          }\n                        }}\n                        tabIndex={-1}\n                        value={option.value}\n                      >\n                        <DisplayOption\n                          descriptionDirection={descriptionDirection}\n                          option={option}\n                          optionalInfoPlacement={optionalInfoPlacement}\n                        />\n                      </StyledCheckbox>\n                    ) : (\n                      <DisplayOption\n                        descriptionDirection={descriptionDirection}\n                        option={option}\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      data-grouped={false}\n      gap={0.25}\n      id=\"select-dropdown\"\n      onKeyDown={handleKeyDownSelect}\n      role=\"listbox\"\n    >\n      {selectAll && multiselect ? (\n        <Stack gap={0.25} id=\"items\" tabIndex={-1}>\n          <DropdownItem\n            aria-disabled={false}\n            aria-label=\"select-all\"\n            aria-selected={selectedData.allSelected}\n            data-testid=\"select-all\"\n            onClick={selectAllOptions}\n            onKeyDown={event =>\n              [' ', 'Enter'].includes(event.key) ? selectAllOptions() : null\n            }\n            role=\"option\"\n            tabIndex={0}\n          >\n            <StyledCheckbox\n              checked={selectedData.allSelected}\n              data-testid=\"select-all-checkbox\"\n              disabled={false}\n              onChange={selectAllOptions}\n              tabIndex={-1}\n              value=\"select-all\"\n            >\n              <Stack direction=\"column\">\n                <Text as=\"span\" placement=\"left\" variant=\"body\">\n                  {selectAll.label}\n                </Text>\n                <Text\n                  as=\"span\"\n                  placement=\"left\"\n                  prominence=\"weak\"\n                  sentiment=\"neutral\"\n                  variant=\"bodySmall\"\n                >\n                  {selectAll.description}\n                </Text>\n              </Stack>\n            </StyledCheckbox>\n          </DropdownItem>\n        </Stack>\n      ) : null}\n      <Stack gap={0.25} id=\"items\">\n        {isLoading ? (\n          <Skeleton variant=\"block\" />\n        ) : (\n          displayedOptions.map((option, index) => (\n            <DropdownItem\n              aria-disabled={!!option.disabled}\n              aria-label={option.value}\n              aria-selected={\n                selectedData.selectedValues.includes(option.value) &&\n                !option.disabled\n              }\n              data-testid={`option-${option.value}`}\n              id={`option-${index}`}\n              key={option.value}\n              onClick={() => {\n                if (!option.disabled) {\n                  handleClick(option)\n                }\n              }}\n              onKeyDown={event =>\n                [' ', 'Enter'].includes(event.key) ? handleClick(option) : null\n              }\n              ref={\n                option.value === defaultSearchValue ||\n                option.searchText === defaultSearchValue\n                  ? focusedItemRef\n                  : null\n              }\n              role=\"option\"\n              tabIndex={!option.disabled ? 0 : -1}\n            >\n              {multiselect ? (\n                <StyledCheckbox\n                  checked={\n                    selectedData.selectedValues.includes(option.value) &&\n                    !option.disabled\n                  }\n                  disabled={option.disabled}\n                  onChange={() => {\n                    if (!option.disabled) {\n                      handleClick(option)\n                    }\n                  }}\n                  tabIndex={-1}\n                  value={option.value}\n                >\n                  <DisplayOption\n                    descriptionDirection={descriptionDirection}\n                    option={option}\n                    optionalInfoPlacement={optionalInfoPlacement}\n                  />\n                </StyledCheckbox>\n              ) : (\n                <DisplayOption\n                  descriptionDirection={descriptionDirection}\n                  option={option}\n                  optionalInfoPlacement={optionalInfoPlacement}\n                />\n              )}\n            </DropdownItem>\n          ))\n        )}\n        {loadMore ? <LoadMore>{loadMore}</LoadMore> : null}\n      </Stack>\n    </DropdownContainer>\n  )\n}\nexport const Dropdown = ({\n  children,\n  emptyState,\n  descriptionDirection,\n  searchable,\n  placeholder,\n  footer,\n  refSelect,\n  loadMore,\n  optionalInfoPlacement,\n  isLoading,\n  size,\n  dropdownAlign,\n  portalTarget,\n  id,\n}: DropdownProps) => {\n  const {\n    setIsDropdownVisible,\n    isDropdownVisible,\n    onSearch,\n    searchInput,\n    options,\n    displayedOptions,\n    numberOfOptions,\n  } = useSelectInput()\n  const theme = useTheme()\n  const [searchBarActive, setSearchBarActive] = useState(false)\n  const [defaultSearchValue, setDefaultSearch] = useState<string | null>(null)\n  const ref = useRef<HTMLDivElement>(null)\n  const [search, setSearch] = useState<string>('')\n  const [maxWidth, setWidth] = useState<string | number>()\n  const modalContext = useContext(ModalContext)\n\n  useEffect(() => {\n    if (refSelect.current && isDropdownVisible) {\n      const position =\n        refSelect.current.getBoundingClientRect().bottom +\n        DROPDOWN_MAX_HEIGHT +\n        Number(theme.sizing[INPUT_SIZE_HEIGHT[size]].replace('rem', '')) * 16 +\n        Number.parseInt(theme.space['5'], 10)\n      const overflow = position - window.innerHeight + 32\n      if (overflow > 0 && modalContext) {\n        const currentModal = modalContext.openedModals[0]\n        const modalElement = currentModal?.ref.current\n\n        if (modalElement) {\n          const parentElement = modalElement.parentNode as HTMLElement\n          if (parentElement) {\n            parentElement.scrollBy({\n              behavior: 'smooth',\n              top: overflow,\n            })\n          }\n        } else {\n          window.scrollBy({ behavior: 'smooth', top: overflow })\n        }\n      }\n    }\n    // oxlint-disable-next-line react/exhaustive-deps\n  }, [isDropdownVisible, refSelect, size, ref.current])\n\n  const resizeDropdown = useCallback(() => {\n    if (\n      refSelect.current &&\n      refSelect.current.getBoundingClientRect().width > 0\n    ) {\n      setWidth(refSelect.current.getBoundingClientRect().width)\n    }\n  }, [refSelect])\n\n  useEffect(() => {\n    resizeDropdown()\n\n    window.addEventListener('resize', resizeDropdown)\n\n    return () => window.removeEventListener('resize', resizeDropdown)\n  }, [resizeDropdown])\n\n  useEffect(() => {\n    if (!searchInput) {\n      onSearch(options)\n    }\n  }, [onSearch, options, searchInput])\n\n  useEffect(() => {\n    if (!isDropdownVisible) {\n      setDefaultSearch(null)\n      setSearch('')\n    }\n\n    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      if (!searchable) {\n        document.removeEventListener('keydown', event =>\n          handleKeyDown(\n            event,\n            ref,\n            options,\n            searchBarActive,\n            setSearch,\n            setDefaultSearch,\n            search,\n          ),\n        )\n      }\n    }\n  }, [\n    isDropdownVisible,\n    searchBarActive,\n    options,\n    onSearch,\n    search,\n    refSelect,\n    setDefaultSearch,\n    setIsDropdownVisible,\n    searchable,\n  ])\n\n  // No data is displayed (because of the search or because no data is provided)\n  // Set to true when noData by default\n  const isEmpty = useMemo(() => {\n    if (numberOfOptions === 0) {\n      return true\n    }\n    if (Array.isArray(displayedOptions)) {\n      return displayedOptions.length === 0\n    }\n    const groups = Object.keys(displayedOptions)\n    for (const group of groups) {\n      if (displayedOptions[group].length > 0) {\n        return false\n      }\n    }\n\n    return true\n  }, [displayedOptions, numberOfOptions])\n\n  const computedFooter = useMemo(() => {\n    if (footer && !isEmpty) {\n      if (typeof footer === 'function') {\n        return (\n          <PopupFooter>{footer(() => setIsDropdownVisible(false))}</PopupFooter>\n        )\n      }\n\n      return <PopupFooter>{footer}</PopupFooter>\n    }\n\n    return null\n  }, [isEmpty, footer, setIsDropdownVisible])\n\n  return (\n    <StyledPopup\n      align={dropdownAlign ?? 'start'}\n      containerFullWidth\n      debounceDelay={0}\n      disableAnimation\n      hasArrow={false}\n      hideOnClickOutside\n      id={id}\n      maxWidth={maxWidth ?? refSelect.current?.offsetWidth}\n      onClose={() => setIsDropdownVisible(false)}\n      placement=\"bottom\"\n      portalTarget={portalTarget}\n      ref={ref}\n      role=\"dialog\"\n      tabIndex={-1}\n      text={\n        <Stack>\n          {searchable && !isLoading && numberOfOptions >= 6 ? (\n            <SearchBarDropdown\n              displayedOptions={displayedOptions}\n              placeholder={placeholder}\n              setSearchBarActive={setSearchBarActive}\n            />\n          ) : null}\n          <CreateDropdown\n            defaultSearchValue={defaultSearchValue}\n            descriptionDirection={descriptionDirection}\n            emptyState={emptyState}\n            isEmpty={isEmpty}\n            isLoading={isLoading}\n            loadMore={loadMore}\n            optionalInfoPlacement={optionalInfoPlacement}\n          />\n          {computedFooter}\n        </Stack>\n      }\n      visible={isDropdownVisible}\n    >\n      {children}\n    </StyledPopup>\n  )\n}\n"]} */"));
|
|
127
|
+
}) => 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/SelectInput/Dropdown.tsx"],"names":[],"mappings":"AAuIE","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInput/Dropdown.tsx","sourcesContent":["'use client'\n\nimport { useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport type {\n  ComponentProps,\n  Dispatch,\n  KeyboardEvent,\n  ReactNode,\n  RefObject,\n  SetStateAction,\n} from 'react'\nimport {\n  useCallback,\n  useContext,\n  useEffect,\n  useMemo,\n  useRef,\n  useState,\n} from 'react'\nimport { Checkbox } from '../Checkbox'\nimport { ModalContext } from '../Modal/ModalProvider'\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'\nimport { INPUT_SIZE_HEIGHT } from './types'\n\nconst DROPDOWN_MAX_HEIGHT = 256\n\nexport type DropdownProps = {\n  id?: string\n  children: ReactNode\n  emptyState: ReactNode\n  descriptionDirection: 'row' | 'column'\n  searchable: boolean\n  placeholder: string\n  footer?: ((closeDropdown: () => void) => ReactNode) | ReactNode\n  refSelect: RefObject<HTMLDivElement | null>\n  loadMore?: ReactNode\n  optionalInfoPlacement: 'left' | 'right'\n  isLoading?: boolean\n  size: 'small' | 'medium' | 'large'\n  dropdownAlign?: ComponentProps<typeof Popup>['align']\n  portalTarget?: ComponentProps<typeof Popup>['portalTarget']\n}\n\nexport type CreateDropdownProps = {\n  isEmpty: boolean\n  emptyState: ReactNode\n  descriptionDirection: 'row' | 'column'\n  loadMore?: ReactNode\n  optionalInfoPlacement: 'left' | 'right'\n  defaultSearchValue: string | null\n  isLoading?: boolean\n}\n\nconst NON_SEARCHABLE_KEYS = [\n  'Tab',\n  ' ',\n  'Enter',\n  'CapsLock',\n  'Shift',\n  'ArrowDown',\n  'ArrowUp',\n  'ArrowLeft',\n  'ArrowRight',\n  'Escape',\n]\n\nconst StyledPopup = styled(Popup)`\n  width: 100%;\n  min-width: 320px;\n  background-color: ${({ theme }) =>\n    theme.colors.other.elevation.background.raised};\n  color: ${({ theme }) => theme.colors.neutral.text};\n  box-shadow: ${({ theme }) =>\n    `${theme.shadows.raised[0]}, ${theme.shadows.raised[1]}`};\n  padding: ${({ theme }) => theme.space[0]};\n  margin-bottom: ${({ theme }) => theme.space[10]};\n`\n\nconst DropdownContainer = styled(Stack)<{ 'data-grouped': boolean }>`\n  max-height: ${DROPDOWN_MAX_HEIGHT}px;\n  overflow-y: auto;\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: 0;\n  padding-right: ${({ theme }) => theme.space[2]};\n  padding-left: ${({ theme }) => theme.space[2]};\n  height: ${({ theme }) => theme.space[4]};\n  display: flex;\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: 0;\n`\nconst DropdownItem = styled.div<{\n  'aria-selected': boolean\n  'aria-disabled': boolean\n}>`\n  text-align:left;\n  border: none;\n  background-color: ${({ theme }) =>\n    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  border: 1px transparent solid;\n\n  &:hover, :focus {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n    color: ${({ theme }) => theme.colors.primary.text};\n    cursor: pointer;\n  }\n\n  &[aria-selected='true'] {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n  }\n\n  &[aria-disabled=\"true\"] {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n  }\n\n  &[aria-disabled=\"true\"]:hover, [aria-disabled=\"true\"]:focus {\n    background-color: ${({ theme }) =>\n      theme.colors.neutral.backgroundStrongDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textStrongDisabled};\n    cursor: not-allowed;\n    outline: none;\n  }\n`\n\nconst PopupFooter = styled.div`\n  width: 100%;\n  padding: ${({ theme }) => theme.space[1.5]} ${({ theme }) => theme.space[2]}\n    ${({ theme }) => theme.space[1.5]} ${({ theme }) => theme.space[2]};\n  box-shadow: ${({ theme }) => theme.shadows.dropdown};\n`\nconst StyledCheckbox = styled(Checkbox)`\n  width: 100%;\n  position: static;\n  text-align: left;\n  align-items: center;\n  pointer-events: none;\n` // pointer-events: none prevents any error when using the checkbox in a form since it is an unnamed input\n\nconst EmptyState = styled(Stack)`\n  padding: ${({ theme }) => theme.space[2]};\n`\nconst LoadMore = styled(Stack)`\n  padding: ${({ theme }) => theme.space[0.5]};\n`\n\nconst moveFocusDown = () => {\n  const options = document.querySelectorAll(\n    '#items > div[role=\"option\"]:not([disabled])',\n  )\n  const activeItem = document.activeElement\n  if (options) {\n    for (let i = 0; i < options?.length; i += 1) {\n      const listLength = options.length\n      if (activeItem === options[i] && activeItem !== options[listLength - 1]) {\n        ;(options[i + 1] as HTMLElement).focus()\n      }\n    }\n  }\n}\nconst moveFocusUp = () => {\n  const options = document.querySelectorAll(\n    '#items > div[role=\"option\"]:not([disabled])',\n  )\n  const activeItem = document.activeElement\n\n  if (options) {\n    for (let i = 0; i < options.length; i += 1) {\n      if (activeItem === options[i] && activeItem !== options[0]) {\n        ;(options[i - 1] as HTMLElement).focus()\n      }\n    }\n  }\n}\nconst handleKeyDownSelect = (event: KeyboardEvent<HTMLDivElement>) => {\n  if (event.key === 'ArrowDown') {\n    event.preventDefault()\n    moveFocusDown()\n  }\n\n  if (event.key === 'ArrowUp') {\n    event.preventDefault()\n    moveFocusUp()\n  }\n\n  if (event.key === ' ') {\n    // No scroll\n    event.preventDefault()\n  }\n}\n\nconst handleKeyDown = (\n  event: globalThis.KeyboardEvent,\n  ref: RefObject<HTMLDivElement | null>,\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].find(option =>\n        option.searchText\n          ? option.searchText.toLocaleLowerCase().startsWith(currentSearch)\n          : option.value.toLocaleLowerCase().startsWith(currentSearch),\n      )\n      if (closestOption) {\n        setDefaultSearch(closestOption.searchText ?? closestOption.value)\n      } else {\n        setDefaultSearch(null)\n      }\n    }\n  }\n}\nconst CreateDropdown = ({\n  isEmpty,\n  emptyState,\n  descriptionDirection,\n  loadMore,\n  optionalInfoPlacement,\n  defaultSearchValue,\n  isLoading,\n}: CreateDropdownProps) => {\n  const {\n    setIsDropdownVisible,\n    onChange,\n    options,\n    multiselect,\n    selectAll,\n    selectAllGroup,\n    displayedOptions,\n    setSelectedData,\n    selectedData,\n  } = useSelectInput()\n  const focusedItemRef = useRef<HTMLDivElement>(null)\n\n  useEffect(() => {\n    if (defaultSearchValue && focusedItemRef?.current) {\n      focusedItemRef.current.focus()\n    }\n  }, [defaultSearchValue])\n\n  if (isEmpty) {\n    return (\n      <EmptyState alignItems=\"center\" gap={2}>\n        {emptyState ?? (\n          <Text as=\"p\" variant=\"bodyStrong\">\n            No options\n          </Text>\n        )}\n      </EmptyState>\n    )\n  }\n\n  const handleClick = (clickedOption: OptionType, group?: string) => {\n    setSelectedData({ clickedOption, group, type: 'selectOption' })\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({ selectedGroup: group, type: 'selectGroup' })\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      data-grouped\n      id=\"select-dropdown\"\n      onKeyDown={handleKeyDownSelect}\n      role=\"listbox\"\n    >\n      {isLoading ? (\n        <Skeleton variant=\"block\" />\n      ) : (\n        <>\n          {selectAll && multiselect ? (\n            <Stack id=\"items\">\n              <DropdownItem\n                aria-disabled={false}\n                aria-label=\"select-all\"\n                aria-selected={selectedData.allSelected}\n                data-testid=\"select-all\"\n                id=\"select-all\"\n                onClick={selectAllOptions}\n                onKeyDown={event =>\n                  [' ', 'Enter'].includes(event.key) ? selectAllOptions() : null\n                }\n                role=\"option\"\n                tabIndex={0}\n              >\n                <StyledCheckbox\n                  checked={selectedData.allSelected}\n                  data-testid=\"select-all-checkbox\"\n                  disabled={false}\n                  onChange={selectAllOptions}\n                  tabIndex={-1}\n                  value=\"select-all\"\n                >\n                  <Stack direction=\"column\">\n                    <Text as=\"span\" placement=\"left\" variant=\"body\">\n                      {selectAll.label}\n                    </Text>\n                    <Text\n                      as=\"span\"\n                      placement=\"left\"\n                      prominence=\"weak\"\n                      sentiment=\"neutral\"\n                      variant=\"bodySmall\"\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 gap={0.25} key={group}>\n              {displayedOptions[group].length > 0 ? (\n                <DropdownGroupWrapper id={selectAllGroup ? 'items' : undefined}>\n                  {group ? (\n                    <DropdownGroup\n                      data-selectgroup={selectAllGroup}\n                      data-testid={`group-${index}`}\n                      key={group}\n                      onClick={() =>\n                        selectAllGroup ? handleSelectGroup(group) : null\n                      }\n                      onKeyDown={event => {\n                        if ([' ', 'Enter'].includes(event.key)) {\n                          event.preventDefault()\n                          handleSelectGroup(group)\n                        }\n                      }}\n                      role=\"group\"\n                      tabIndex={selectAllGroup ? 0 : -1}\n                      type=\"button\"\n                    >\n                      {selectAllGroup ? (\n                        <StyledCheckbox\n                          checked={selectedData.selectedGroups.includes(group)}\n                          data-testid=\"select-group\"\n                          disabled={false}\n                          onChange={() =>\n                            selectAllGroup ? handleSelectGroup(group) : null\n                          }\n                          tabIndex={-1}\n                          value={group}\n                        >\n                          <Text\n                            as=\"span\"\n                            placement=\"left\"\n                            sentiment=\"neutral\"\n                            variant=\"caption\"\n                          >\n                            {group.toUpperCase()}\n                          </Text>\n                        </StyledCheckbox>\n                      ) : (\n                        <Text\n                          as=\"span\"\n                          placement=\"left\"\n                          sentiment=\"neutral\"\n                          variant=\"caption\"\n                        >\n                          {group.toUpperCase()}\n                        </Text>\n                      )}\n                    </DropdownGroup>\n                  ) : null}\n                </DropdownGroupWrapper>\n              ) : null}\n              <Stack gap=\"0.25\" id=\"items\">\n                {displayedOptions[group].map((option, indexOption) => (\n                  <DropdownItem\n                    aria-disabled={!!option.disabled}\n                    aria-label={option.value}\n                    aria-selected={\n                      selectedData.selectedValues.includes(option.value) &&\n                      !option.disabled\n                    }\n                    data-testid={`option-${option.value}`}\n                    id={`option-${indexOption}`}\n                    key={option.value}\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                    role=\"option\"\n                    tabIndex={!option.disabled ? 0 : -1}\n                  >\n                    {multiselect ? (\n                      <StyledCheckbox\n                        checked={\n                          selectedData.selectedValues.includes(option.value) &&\n                          !option.disabled\n                        }\n                        disabled={option.disabled}\n                        onChange={() => {\n                          if (!option.disabled) {\n                            handleClick(option, group)\n                          }\n                        }}\n                        tabIndex={-1}\n                        value={option.value}\n                      >\n                        <DisplayOption\n                          descriptionDirection={descriptionDirection}\n                          option={option}\n                          optionalInfoPlacement={optionalInfoPlacement}\n                        />\n                      </StyledCheckbox>\n                    ) : (\n                      <DisplayOption\n                        descriptionDirection={descriptionDirection}\n                        option={option}\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      data-grouped={false}\n      gap={0.25}\n      id=\"select-dropdown\"\n      onKeyDown={handleKeyDownSelect}\n      role=\"listbox\"\n    >\n      {selectAll && multiselect ? (\n        <Stack gap={0.25} id=\"items\" tabIndex={-1}>\n          <DropdownItem\n            aria-disabled={false}\n            aria-label=\"select-all\"\n            aria-selected={selectedData.allSelected}\n            data-testid=\"select-all\"\n            onClick={selectAllOptions}\n            onKeyDown={event =>\n              [' ', 'Enter'].includes(event.key) ? selectAllOptions() : null\n            }\n            role=\"option\"\n            tabIndex={0}\n          >\n            <StyledCheckbox\n              checked={selectedData.allSelected}\n              data-testid=\"select-all-checkbox\"\n              disabled={false}\n              onChange={selectAllOptions}\n              tabIndex={-1}\n              value=\"select-all\"\n            >\n              <Stack direction=\"column\">\n                <Text as=\"span\" placement=\"left\" variant=\"body\">\n                  {selectAll.label}\n                </Text>\n                <Text\n                  as=\"span\"\n                  placement=\"left\"\n                  prominence=\"weak\"\n                  sentiment=\"neutral\"\n                  variant=\"bodySmall\"\n                >\n                  {selectAll.description}\n                </Text>\n              </Stack>\n            </StyledCheckbox>\n          </DropdownItem>\n        </Stack>\n      ) : null}\n      <Stack gap={0.25} id=\"items\">\n        {isLoading ? (\n          <Skeleton variant=\"block\" />\n        ) : (\n          displayedOptions.map((option, index) => (\n            <DropdownItem\n              aria-disabled={!!option.disabled}\n              aria-label={option.value}\n              aria-selected={\n                selectedData.selectedValues.includes(option.value) &&\n                !option.disabled\n              }\n              data-testid={`option-${option.value}`}\n              id={`option-${index}`}\n              key={option.value}\n              onClick={() => {\n                if (!option.disabled) {\n                  handleClick(option)\n                }\n              }}\n              onKeyDown={event =>\n                [' ', 'Enter'].includes(event.key) ? handleClick(option) : null\n              }\n              ref={\n                option.value === defaultSearchValue ||\n                option.searchText === defaultSearchValue\n                  ? focusedItemRef\n                  : null\n              }\n              role=\"option\"\n              tabIndex={!option.disabled ? 0 : -1}\n            >\n              {multiselect ? (\n                <StyledCheckbox\n                  checked={\n                    selectedData.selectedValues.includes(option.value) &&\n                    !option.disabled\n                  }\n                  disabled={option.disabled}\n                  onChange={() => {\n                    if (!option.disabled) {\n                      handleClick(option)\n                    }\n                  }}\n                  tabIndex={-1}\n                  value={option.value}\n                >\n                  <DisplayOption\n                    descriptionDirection={descriptionDirection}\n                    option={option}\n                    optionalInfoPlacement={optionalInfoPlacement}\n                  />\n                </StyledCheckbox>\n              ) : (\n                <DisplayOption\n                  descriptionDirection={descriptionDirection}\n                  option={option}\n                  optionalInfoPlacement={optionalInfoPlacement}\n                />\n              )}\n            </DropdownItem>\n          ))\n        )}\n        {loadMore ? <LoadMore>{loadMore}</LoadMore> : null}\n      </Stack>\n    </DropdownContainer>\n  )\n}\nexport const Dropdown = ({\n  children,\n  emptyState,\n  descriptionDirection,\n  searchable,\n  placeholder,\n  footer,\n  refSelect,\n  loadMore,\n  optionalInfoPlacement,\n  isLoading,\n  size,\n  dropdownAlign,\n  portalTarget,\n  id,\n}: DropdownProps) => {\n  const {\n    setIsDropdownVisible,\n    isDropdownVisible,\n    onSearch,\n    searchInput,\n    options,\n    displayedOptions,\n    numberOfOptions,\n  } = useSelectInput()\n  const theme = useTheme()\n  const [searchBarActive, setSearchBarActive] = useState(false)\n  const [defaultSearchValue, setDefaultSearch] = useState<string | null>(null)\n  const ref = useRef<HTMLDivElement>(null)\n  const [search, setSearch] = useState<string>('')\n  const [maxWidth, setWidth] = useState<string | number>()\n  const modalContext = useContext(ModalContext)\n\n  useEffect(() => {\n    if (refSelect.current && isDropdownVisible) {\n      const position =\n        refSelect.current.getBoundingClientRect().bottom +\n        DROPDOWN_MAX_HEIGHT +\n        Number(theme.sizing[INPUT_SIZE_HEIGHT[size]].replace('rem', '')) * 16 +\n        Number.parseInt(theme.space['5'], 10)\n      const overflow = position - window.innerHeight + 32\n      if (overflow > 0 && modalContext) {\n        const currentModal = modalContext.openedModals[0]\n        const modalElement = currentModal?.ref.current\n\n        if (modalElement) {\n          const parentElement = modalElement.parentNode as HTMLElement\n          if (parentElement) {\n            parentElement.scrollBy({\n              behavior: 'smooth',\n              top: overflow,\n            })\n          }\n        } else {\n          window.scrollBy({ behavior: 'smooth', top: overflow })\n        }\n      }\n    }\n    // oxlint-disable-next-line react/exhaustive-deps\n  }, [isDropdownVisible, refSelect, size, ref.current])\n\n  const resizeDropdown = useCallback(() => {\n    if (\n      refSelect.current &&\n      refSelect.current.getBoundingClientRect().width > 0\n    ) {\n      setWidth(refSelect.current.getBoundingClientRect().width)\n    }\n  }, [refSelect])\n\n  useEffect(() => {\n    resizeDropdown()\n\n    window.addEventListener('resize', resizeDropdown)\n\n    return () => window.removeEventListener('resize', resizeDropdown)\n  }, [resizeDropdown])\n\n  useEffect(() => {\n    if (!searchInput) {\n      onSearch(options)\n    }\n  }, [onSearch, options, searchInput])\n\n  useEffect(() => {\n    if (!isDropdownVisible) {\n      setDefaultSearch(null)\n      setSearch('')\n    }\n\n    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      if (!searchable) {\n        document.removeEventListener('keydown', event =>\n          handleKeyDown(\n            event,\n            ref,\n            options,\n            searchBarActive,\n            setSearch,\n            setDefaultSearch,\n            search,\n          ),\n        )\n      }\n    }\n  }, [\n    isDropdownVisible,\n    searchBarActive,\n    options,\n    onSearch,\n    search,\n    refSelect,\n    setDefaultSearch,\n    setIsDropdownVisible,\n    searchable,\n  ])\n\n  // No data is displayed (because of the search or because no data is provided)\n  // Set to true when noData by default\n  const isEmpty = useMemo(() => {\n    if (numberOfOptions === 0) {\n      return true\n    }\n    if (Array.isArray(displayedOptions)) {\n      return displayedOptions.length === 0\n    }\n    const groups = Object.keys(displayedOptions)\n    for (const group of groups) {\n      if (displayedOptions[group].length > 0) {\n        return false\n      }\n    }\n\n    return true\n  }, [displayedOptions, numberOfOptions])\n\n  const computedFooter = useMemo(() => {\n    if (footer && !isEmpty) {\n      if (typeof footer === 'function') {\n        return (\n          <PopupFooter>{footer(() => setIsDropdownVisible(false))}</PopupFooter>\n        )\n      }\n\n      return <PopupFooter>{footer}</PopupFooter>\n    }\n\n    return null\n  }, [isEmpty, footer, setIsDropdownVisible])\n\n  return (\n    <StyledPopup\n      align={dropdownAlign ?? 'start'}\n      containerFullWidth\n      debounceDelay={0}\n      disableAnimation\n      hasArrow={false}\n      hideOnClickOutside\n      id={id}\n      maxWidth={maxWidth ?? refSelect.current?.offsetWidth}\n      onClose={() => setIsDropdownVisible(false)}\n      placement=\"bottom\"\n      portalTarget={portalTarget}\n      ref={ref}\n      role=\"dialog\"\n      tabIndex={-1}\n      text={\n        <Stack>\n          {searchable && !isLoading && numberOfOptions >= 6 ? (\n            <SearchBarDropdown\n              displayedOptions={displayedOptions}\n              placeholder={placeholder}\n              setSearchBarActive={setSearchBarActive}\n            />\n          ) : null}\n          <CreateDropdown\n            defaultSearchValue={defaultSearchValue}\n            descriptionDirection={descriptionDirection}\n            emptyState={emptyState}\n            isEmpty={isEmpty}\n            isLoading={isLoading}\n            loadMore={loadMore}\n            optionalInfoPlacement={optionalInfoPlacement}\n          />\n          {computedFooter}\n        </Stack>\n      }\n      visible={isDropdownVisible}\n    >\n      {children}\n    </StyledPopup>\n  )\n}\n"]} */"));
|
|
128
128
|
const PopupFooter = /* @__PURE__ */ _styled("div", process.env.NODE_ENV === "production" ? {
|
|
129
129
|
target: "e12qyldb3"
|
|
130
130
|
} : {
|
|
@@ -140,7 +140,7 @@ const PopupFooter = /* @__PURE__ */ _styled("div", process.env.NODE_ENV === "pro
|
|
|
140
140
|
theme
|
|
141
141
|
}) => theme.space[2], ";box-shadow:", ({
|
|
142
142
|
theme
|
|
143
|
-
}) => 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/SelectInput/Dropdown.tsx"],"names":[],"mappings":"AA8K8B","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInput/Dropdown.tsx","sourcesContent":["'use client'\n\nimport { useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport type {\n  ComponentProps,\n  Dispatch,\n  KeyboardEvent,\n  ReactNode,\n  RefObject,\n  SetStateAction,\n} from 'react'\nimport {\n  useCallback,\n  useContext,\n  useEffect,\n  useMemo,\n  useRef,\n  useState,\n} from 'react'\nimport { Checkbox } from '../Checkbox'\nimport { ModalContext } from '../Modal/ModalProvider'\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'\nimport { INPUT_SIZE_HEIGHT } from './types'\n\nconst DROPDOWN_MAX_HEIGHT = 256\n\nexport type DropdownProps = {\n  id?: string\n  children: ReactNode\n  emptyState: ReactNode\n  descriptionDirection: 'row' | 'column'\n  searchable: boolean\n  placeholder: string\n  footer?: ((closeDropdown: () => void) => ReactNode) | ReactNode\n  refSelect: RefObject<HTMLDivElement | null>\n  loadMore?: ReactNode\n  optionalInfoPlacement: 'left' | 'right'\n  isLoading?: boolean\n  size: 'small' | 'medium' | 'large'\n  dropdownAlign?: ComponentProps<typeof Popup>['align']\n  portalTarget?: ComponentProps<typeof Popup>['portalTarget']\n}\n\nexport type CreateDropdownProps = {\n  isEmpty: boolean\n  emptyState: ReactNode\n  descriptionDirection: 'row' | 'column'\n  loadMore?: ReactNode\n  optionalInfoPlacement: 'left' | 'right'\n  defaultSearchValue: string | null\n  isLoading?: boolean\n}\n\nconst NON_SEARCHABLE_KEYS = [\n  'Tab',\n  ' ',\n  'Enter',\n  'CapsLock',\n  'Shift',\n  'ArrowDown',\n  'ArrowUp',\n  'ArrowLeft',\n  'ArrowRight',\n  'Escape',\n]\n\nconst StyledPopup = styled(Popup)`\n  width: 100%;\n  min-width: 320px;\n  background-color: ${({ theme }) =>\n    theme.colors.other.elevation.background.raised};\n  color: ${({ theme }) => theme.colors.neutral.text};\n  box-shadow: ${({ theme }) =>\n    `${theme.shadows.raised[0]}, ${theme.shadows.raised[1]}`};\n  padding: ${({ theme }) => theme.space[0]};\n  margin-bottom: ${({ theme }) => theme.space[10]};\n`\n\nconst DropdownContainer = styled(Stack)<{ 'data-grouped': boolean }>`\n  max-height: ${DROPDOWN_MAX_HEIGHT}px;\n  overflow-y: auto;\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: 0;\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: 0;\n`\nconst DropdownItem = styled.div<{\n  'aria-selected': boolean\n  'aria-disabled': boolean\n}>`\n  text-align:left;\n  border: none;\n  background-color: ${({ theme }) =>\n    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  border: 1px transparent solid;\n\n  &:hover, :focus {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n    color: ${({ theme }) => theme.colors.primary.text};\n    cursor: pointer;\n  }\n\n  &[aria-selected='true'] {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n  }\n\n  &[aria-disabled=\"true\"] {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n  }\n\n  &[aria-disabled=\"true\"]:hover, [aria-disabled=\"true\"]:focus {\n    background-color: ${({ theme }) =>\n      theme.colors.neutral.backgroundStrongDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textStrongDisabled};\n    cursor: not-allowed;\n    outline: none;\n  }\n`\n\nconst PopupFooter = styled.div`\n  width: 100%;\n  padding: ${({ theme }) => theme.space[1.5]} ${({ theme }) => theme.space[2]}\n    ${({ theme }) => theme.space[1.5]} ${({ theme }) => theme.space[2]};\n  box-shadow: ${({ theme }) => theme.shadows.dropdown};\n`\nconst StyledCheckbox = styled(Checkbox)`\n  width: 100%;\n  position: static;\n  text-align: left;\n  align-items: center;\n  pointer-events: none;\n` // pointer-events: none prevents any error when using the checkbox in a form since it is an unnamed input\n\nconst EmptyState = styled(Stack)`\n  padding: ${({ theme }) => theme.space[2]};\n`\nconst LoadMore = styled(Stack)`\n  padding: ${({ theme }) => theme.space[0.5]};\n`\n\nconst moveFocusDown = () => {\n  const options = document.querySelectorAll(\n    '#items > div[role=\"option\"]:not([disabled])',\n  )\n  const activeItem = document.activeElement\n  if (options) {\n    for (let i = 0; i < options?.length; i += 1) {\n      const listLength = options.length\n      if (activeItem === options[i] && activeItem !== options[listLength - 1]) {\n        ;(options[i + 1] as HTMLElement).focus()\n      }\n    }\n  }\n}\nconst moveFocusUp = () => {\n  const options = document.querySelectorAll(\n    '#items > div[role=\"option\"]:not([disabled])',\n  )\n  const activeItem = document.activeElement\n\n  if (options) {\n    for (let i = 0; i < options.length; i += 1) {\n      if (activeItem === options[i] && activeItem !== options[0]) {\n        ;(options[i - 1] as HTMLElement).focus()\n      }\n    }\n  }\n}\nconst handleKeyDownSelect = (event: KeyboardEvent<HTMLDivElement>) => {\n  if (event.key === 'ArrowDown') {\n    event.preventDefault()\n    moveFocusDown()\n  }\n\n  if (event.key === 'ArrowUp') {\n    event.preventDefault()\n    moveFocusUp()\n  }\n\n  if (event.key === ' ') {\n    // No scroll\n    event.preventDefault()\n  }\n}\n\nconst handleKeyDown = (\n  event: globalThis.KeyboardEvent,\n  ref: RefObject<HTMLDivElement | null>,\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].find(option =>\n        option.searchText\n          ? option.searchText.toLocaleLowerCase().startsWith(currentSearch)\n          : option.value.toLocaleLowerCase().startsWith(currentSearch),\n      )\n      if (closestOption) {\n        setDefaultSearch(closestOption.searchText ?? closestOption.value)\n      } else {\n        setDefaultSearch(null)\n      }\n    }\n  }\n}\nconst CreateDropdown = ({\n  isEmpty,\n  emptyState,\n  descriptionDirection,\n  loadMore,\n  optionalInfoPlacement,\n  defaultSearchValue,\n  isLoading,\n}: CreateDropdownProps) => {\n  const {\n    setIsDropdownVisible,\n    onChange,\n    options,\n    multiselect,\n    selectAll,\n    selectAllGroup,\n    displayedOptions,\n    setSelectedData,\n    selectedData,\n  } = useSelectInput()\n  const focusedItemRef = useRef<HTMLDivElement>(null)\n\n  useEffect(() => {\n    if (defaultSearchValue && focusedItemRef?.current) {\n      focusedItemRef.current.focus()\n    }\n  }, [defaultSearchValue])\n\n  if (isEmpty) {\n    return (\n      <EmptyState alignItems=\"center\" gap={2}>\n        {emptyState ?? (\n          <Text as=\"p\" variant=\"bodyStrong\">\n            No options\n          </Text>\n        )}\n      </EmptyState>\n    )\n  }\n\n  const handleClick = (clickedOption: OptionType, group?: string) => {\n    setSelectedData({ clickedOption, group, type: 'selectOption' })\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({ selectedGroup: group, type: 'selectGroup' })\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      data-grouped\n      id=\"select-dropdown\"\n      onKeyDown={handleKeyDownSelect}\n      role=\"listbox\"\n    >\n      {isLoading ? (\n        <Skeleton variant=\"block\" />\n      ) : (\n        <>\n          {selectAll && multiselect ? (\n            <Stack id=\"items\">\n              <DropdownItem\n                aria-disabled={false}\n                aria-label=\"select-all\"\n                aria-selected={selectedData.allSelected}\n                data-testid=\"select-all\"\n                id=\"select-all\"\n                onClick={selectAllOptions}\n                onKeyDown={event =>\n                  [' ', 'Enter'].includes(event.key) ? selectAllOptions() : null\n                }\n                role=\"option\"\n                tabIndex={0}\n              >\n                <StyledCheckbox\n                  checked={selectedData.allSelected}\n                  data-testid=\"select-all-checkbox\"\n                  disabled={false}\n                  onChange={selectAllOptions}\n                  tabIndex={-1}\n                  value=\"select-all\"\n                >\n                  <Stack direction=\"column\">\n                    <Text as=\"span\" placement=\"left\" variant=\"body\">\n                      {selectAll.label}\n                    </Text>\n                    <Text\n                      as=\"span\"\n                      placement=\"left\"\n                      prominence=\"weak\"\n                      sentiment=\"neutral\"\n                      variant=\"bodySmall\"\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 gap={0.25} key={group}>\n              {displayedOptions[group].length > 0 ? (\n                <DropdownGroupWrapper id={selectAllGroup ? 'items' : undefined}>\n                  {group ? (\n                    <DropdownGroup\n                      data-selectgroup={selectAllGroup}\n                      data-testid={`group-${index}`}\n                      key={group}\n                      onClick={() =>\n                        selectAllGroup ? handleSelectGroup(group) : null\n                      }\n                      onKeyDown={event => {\n                        if ([' ', 'Enter'].includes(event.key)) {\n                          event.preventDefault()\n                          handleSelectGroup(group)\n                        }\n                      }}\n                      role=\"group\"\n                      tabIndex={selectAllGroup ? 0 : -1}\n                      type=\"button\"\n                    >\n                      {selectAllGroup ? (\n                        <StyledCheckbox\n                          checked={selectedData.selectedGroups.includes(group)}\n                          data-testid=\"select-group\"\n                          disabled={false}\n                          onChange={() =>\n                            selectAllGroup ? handleSelectGroup(group) : null\n                          }\n                          tabIndex={-1}\n                          value={group}\n                        >\n                          <Text\n                            as=\"span\"\n                            placement=\"left\"\n                            sentiment=\"neutral\"\n                            variant=\"caption\"\n                          >\n                            {group.toUpperCase()}\n                          </Text>\n                        </StyledCheckbox>\n                      ) : (\n                        <Text\n                          as=\"span\"\n                          placement=\"left\"\n                          sentiment=\"neutral\"\n                          variant=\"caption\"\n                        >\n                          {group.toUpperCase()}\n                        </Text>\n                      )}\n                    </DropdownGroup>\n                  ) : null}\n                </DropdownGroupWrapper>\n              ) : null}\n              <Stack gap=\"0.25\" id=\"items\">\n                {displayedOptions[group].map((option, indexOption) => (\n                  <DropdownItem\n                    aria-disabled={!!option.disabled}\n                    aria-label={option.value}\n                    aria-selected={\n                      selectedData.selectedValues.includes(option.value) &&\n                      !option.disabled\n                    }\n                    data-testid={`option-${option.value}`}\n                    id={`option-${indexOption}`}\n                    key={option.value}\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                    role=\"option\"\n                    tabIndex={!option.disabled ? 0 : -1}\n                  >\n                    {multiselect ? (\n                      <StyledCheckbox\n                        checked={\n                          selectedData.selectedValues.includes(option.value) &&\n                          !option.disabled\n                        }\n                        disabled={option.disabled}\n                        onChange={() => {\n                          if (!option.disabled) {\n                            handleClick(option, group)\n                          }\n                        }}\n                        tabIndex={-1}\n                        value={option.value}\n                      >\n                        <DisplayOption\n                          descriptionDirection={descriptionDirection}\n                          option={option}\n                          optionalInfoPlacement={optionalInfoPlacement}\n                        />\n                      </StyledCheckbox>\n                    ) : (\n                      <DisplayOption\n                        descriptionDirection={descriptionDirection}\n                        option={option}\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      data-grouped={false}\n      gap={0.25}\n      id=\"select-dropdown\"\n      onKeyDown={handleKeyDownSelect}\n      role=\"listbox\"\n    >\n      {selectAll && multiselect ? (\n        <Stack gap={0.25} id=\"items\" tabIndex={-1}>\n          <DropdownItem\n            aria-disabled={false}\n            aria-label=\"select-all\"\n            aria-selected={selectedData.allSelected}\n            data-testid=\"select-all\"\n            onClick={selectAllOptions}\n            onKeyDown={event =>\n              [' ', 'Enter'].includes(event.key) ? selectAllOptions() : null\n            }\n            role=\"option\"\n            tabIndex={0}\n          >\n            <StyledCheckbox\n              checked={selectedData.allSelected}\n              data-testid=\"select-all-checkbox\"\n              disabled={false}\n              onChange={selectAllOptions}\n              tabIndex={-1}\n              value=\"select-all\"\n            >\n              <Stack direction=\"column\">\n                <Text as=\"span\" placement=\"left\" variant=\"body\">\n                  {selectAll.label}\n                </Text>\n                <Text\n                  as=\"span\"\n                  placement=\"left\"\n                  prominence=\"weak\"\n                  sentiment=\"neutral\"\n                  variant=\"bodySmall\"\n                >\n                  {selectAll.description}\n                </Text>\n              </Stack>\n            </StyledCheckbox>\n          </DropdownItem>\n        </Stack>\n      ) : null}\n      <Stack gap={0.25} id=\"items\">\n        {isLoading ? (\n          <Skeleton variant=\"block\" />\n        ) : (\n          displayedOptions.map((option, index) => (\n            <DropdownItem\n              aria-disabled={!!option.disabled}\n              aria-label={option.value}\n              aria-selected={\n                selectedData.selectedValues.includes(option.value) &&\n                !option.disabled\n              }\n              data-testid={`option-${option.value}`}\n              id={`option-${index}`}\n              key={option.value}\n              onClick={() => {\n                if (!option.disabled) {\n                  handleClick(option)\n                }\n              }}\n              onKeyDown={event =>\n                [' ', 'Enter'].includes(event.key) ? handleClick(option) : null\n              }\n              ref={\n                option.value === defaultSearchValue ||\n                option.searchText === defaultSearchValue\n                  ? focusedItemRef\n                  : null\n              }\n              role=\"option\"\n              tabIndex={!option.disabled ? 0 : -1}\n            >\n              {multiselect ? (\n                <StyledCheckbox\n                  checked={\n                    selectedData.selectedValues.includes(option.value) &&\n                    !option.disabled\n                  }\n                  disabled={option.disabled}\n                  onChange={() => {\n                    if (!option.disabled) {\n                      handleClick(option)\n                    }\n                  }}\n                  tabIndex={-1}\n                  value={option.value}\n                >\n                  <DisplayOption\n                    descriptionDirection={descriptionDirection}\n                    option={option}\n                    optionalInfoPlacement={optionalInfoPlacement}\n                  />\n                </StyledCheckbox>\n              ) : (\n                <DisplayOption\n                  descriptionDirection={descriptionDirection}\n                  option={option}\n                  optionalInfoPlacement={optionalInfoPlacement}\n                />\n              )}\n            </DropdownItem>\n          ))\n        )}\n        {loadMore ? <LoadMore>{loadMore}</LoadMore> : null}\n      </Stack>\n    </DropdownContainer>\n  )\n}\nexport const Dropdown = ({\n  children,\n  emptyState,\n  descriptionDirection,\n  searchable,\n  placeholder,\n  footer,\n  refSelect,\n  loadMore,\n  optionalInfoPlacement,\n  isLoading,\n  size,\n  dropdownAlign,\n  portalTarget,\n  id,\n}: DropdownProps) => {\n  const {\n    setIsDropdownVisible,\n    isDropdownVisible,\n    onSearch,\n    searchInput,\n    options,\n    displayedOptions,\n    numberOfOptions,\n  } = useSelectInput()\n  const theme = useTheme()\n  const [searchBarActive, setSearchBarActive] = useState(false)\n  const [defaultSearchValue, setDefaultSearch] = useState<string | null>(null)\n  const ref = useRef<HTMLDivElement>(null)\n  const [search, setSearch] = useState<string>('')\n  const [maxWidth, setWidth] = useState<string | number>()\n  const modalContext = useContext(ModalContext)\n\n  useEffect(() => {\n    if (refSelect.current && isDropdownVisible) {\n      const position =\n        refSelect.current.getBoundingClientRect().bottom +\n        DROPDOWN_MAX_HEIGHT +\n        Number(theme.sizing[INPUT_SIZE_HEIGHT[size]].replace('rem', '')) * 16 +\n        Number.parseInt(theme.space['5'], 10)\n      const overflow = position - window.innerHeight + 32\n      if (overflow > 0 && modalContext) {\n        const currentModal = modalContext.openedModals[0]\n        const modalElement = currentModal?.ref.current\n\n        if (modalElement) {\n          const parentElement = modalElement.parentNode as HTMLElement\n          if (parentElement) {\n            parentElement.scrollBy({\n              behavior: 'smooth',\n              top: overflow,\n            })\n          }\n        } else {\n          window.scrollBy({ behavior: 'smooth', top: overflow })\n        }\n      }\n    }\n    // oxlint-disable-next-line react/exhaustive-deps\n  }, [isDropdownVisible, refSelect, size, ref.current])\n\n  const resizeDropdown = useCallback(() => {\n    if (\n      refSelect.current &&\n      refSelect.current.getBoundingClientRect().width > 0\n    ) {\n      setWidth(refSelect.current.getBoundingClientRect().width)\n    }\n  }, [refSelect])\n\n  useEffect(() => {\n    resizeDropdown()\n\n    window.addEventListener('resize', resizeDropdown)\n\n    return () => window.removeEventListener('resize', resizeDropdown)\n  }, [resizeDropdown])\n\n  useEffect(() => {\n    if (!searchInput) {\n      onSearch(options)\n    }\n  }, [onSearch, options, searchInput])\n\n  useEffect(() => {\n    if (!isDropdownVisible) {\n      setDefaultSearch(null)\n      setSearch('')\n    }\n\n    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      if (!searchable) {\n        document.removeEventListener('keydown', event =>\n          handleKeyDown(\n            event,\n            ref,\n            options,\n            searchBarActive,\n            setSearch,\n            setDefaultSearch,\n            search,\n          ),\n        )\n      }\n    }\n  }, [\n    isDropdownVisible,\n    searchBarActive,\n    options,\n    onSearch,\n    search,\n    refSelect,\n    setDefaultSearch,\n    setIsDropdownVisible,\n    searchable,\n  ])\n\n  // No data is displayed (because of the search or because no data is provided)\n  // Set to true when noData by default\n  const isEmpty = useMemo(() => {\n    if (numberOfOptions === 0) {\n      return true\n    }\n    if (Array.isArray(displayedOptions)) {\n      return displayedOptions.length === 0\n    }\n    const groups = Object.keys(displayedOptions)\n    for (const group of groups) {\n      if (displayedOptions[group].length > 0) {\n        return false\n      }\n    }\n\n    return true\n  }, [displayedOptions, numberOfOptions])\n\n  const computedFooter = useMemo(() => {\n    if (footer && !isEmpty) {\n      if (typeof footer === 'function') {\n        return (\n          <PopupFooter>{footer(() => setIsDropdownVisible(false))}</PopupFooter>\n        )\n      }\n\n      return <PopupFooter>{footer}</PopupFooter>\n    }\n\n    return null\n  }, [isEmpty, footer, setIsDropdownVisible])\n\n  return (\n    <StyledPopup\n      align={dropdownAlign ?? 'start'}\n      containerFullWidth\n      debounceDelay={0}\n      disableAnimation\n      hasArrow={false}\n      hideOnClickOutside\n      id={id}\n      maxWidth={maxWidth ?? refSelect.current?.offsetWidth}\n      onClose={() => setIsDropdownVisible(false)}\n      placement=\"bottom\"\n      portalTarget={portalTarget}\n      ref={ref}\n      role=\"dialog\"\n      tabIndex={-1}\n      text={\n        <Stack>\n          {searchable && !isLoading && numberOfOptions >= 6 ? (\n            <SearchBarDropdown\n              displayedOptions={displayedOptions}\n              placeholder={placeholder}\n              setSearchBarActive={setSearchBarActive}\n            />\n          ) : null}\n          <CreateDropdown\n            defaultSearchValue={defaultSearchValue}\n            descriptionDirection={descriptionDirection}\n            emptyState={emptyState}\n            isEmpty={isEmpty}\n            isLoading={isLoading}\n            loadMore={loadMore}\n            optionalInfoPlacement={optionalInfoPlacement}\n          />\n          {computedFooter}\n        </Stack>\n      }\n      visible={isDropdownVisible}\n    >\n      {children}\n    </StyledPopup>\n  )\n}\n"]} */"));
|
|
143
|
+
}) => 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/SelectInput/Dropdown.tsx"],"names":[],"mappings":"AA+K8B","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInput/Dropdown.tsx","sourcesContent":["'use client'\n\nimport { useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport type {\n  ComponentProps,\n  Dispatch,\n  KeyboardEvent,\n  ReactNode,\n  RefObject,\n  SetStateAction,\n} from 'react'\nimport {\n  useCallback,\n  useContext,\n  useEffect,\n  useMemo,\n  useRef,\n  useState,\n} from 'react'\nimport { Checkbox } from '../Checkbox'\nimport { ModalContext } from '../Modal/ModalProvider'\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'\nimport { INPUT_SIZE_HEIGHT } from './types'\n\nconst DROPDOWN_MAX_HEIGHT = 256\n\nexport type DropdownProps = {\n  id?: string\n  children: ReactNode\n  emptyState: ReactNode\n  descriptionDirection: 'row' | 'column'\n  searchable: boolean\n  placeholder: string\n  footer?: ((closeDropdown: () => void) => ReactNode) | ReactNode\n  refSelect: RefObject<HTMLDivElement | null>\n  loadMore?: ReactNode\n  optionalInfoPlacement: 'left' | 'right'\n  isLoading?: boolean\n  size: 'small' | 'medium' | 'large'\n  dropdownAlign?: ComponentProps<typeof Popup>['align']\n  portalTarget?: ComponentProps<typeof Popup>['portalTarget']\n}\n\nexport type CreateDropdownProps = {\n  isEmpty: boolean\n  emptyState: ReactNode\n  descriptionDirection: 'row' | 'column'\n  loadMore?: ReactNode\n  optionalInfoPlacement: 'left' | 'right'\n  defaultSearchValue: string | null\n  isLoading?: boolean\n}\n\nconst NON_SEARCHABLE_KEYS = [\n  'Tab',\n  ' ',\n  'Enter',\n  'CapsLock',\n  'Shift',\n  'ArrowDown',\n  'ArrowUp',\n  'ArrowLeft',\n  'ArrowRight',\n  'Escape',\n]\n\nconst StyledPopup = styled(Popup)`\n  width: 100%;\n  min-width: 320px;\n  background-color: ${({ theme }) =>\n    theme.colors.other.elevation.background.raised};\n  color: ${({ theme }) => theme.colors.neutral.text};\n  box-shadow: ${({ theme }) =>\n    `${theme.shadows.raised[0]}, ${theme.shadows.raised[1]}`};\n  padding: ${({ theme }) => theme.space[0]};\n  margin-bottom: ${({ theme }) => theme.space[10]};\n`\n\nconst DropdownContainer = styled(Stack)<{ 'data-grouped': boolean }>`\n  max-height: ${DROPDOWN_MAX_HEIGHT}px;\n  overflow-y: auto;\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: 0;\n  padding-right: ${({ theme }) => theme.space[2]};\n  padding-left: ${({ theme }) => theme.space[2]};\n  height: ${({ theme }) => theme.space[4]};\n  display: flex;\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: 0;\n`\nconst DropdownItem = styled.div<{\n  'aria-selected': boolean\n  'aria-disabled': boolean\n}>`\n  text-align:left;\n  border: none;\n  background-color: ${({ theme }) =>\n    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  border: 1px transparent solid;\n\n  &:hover, :focus {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n    color: ${({ theme }) => theme.colors.primary.text};\n    cursor: pointer;\n  }\n\n  &[aria-selected='true'] {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n  }\n\n  &[aria-disabled=\"true\"] {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n  }\n\n  &[aria-disabled=\"true\"]:hover, [aria-disabled=\"true\"]:focus {\n    background-color: ${({ theme }) =>\n      theme.colors.neutral.backgroundStrongDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textStrongDisabled};\n    cursor: not-allowed;\n    outline: none;\n  }\n`\n\nconst PopupFooter = styled.div`\n  width: 100%;\n  padding: ${({ theme }) => theme.space[1.5]} ${({ theme }) => theme.space[2]}\n    ${({ theme }) => theme.space[1.5]} ${({ theme }) => theme.space[2]};\n  box-shadow: ${({ theme }) => theme.shadows.dropdown};\n`\nconst StyledCheckbox = styled(Checkbox)`\n  width: 100%;\n  position: static;\n  text-align: left;\n  align-items: center;\n  pointer-events: none;\n` // pointer-events: none prevents any error when using the checkbox in a form since it is an unnamed input\n\nconst EmptyState = styled(Stack)`\n  padding: ${({ theme }) => theme.space[2]};\n`\nconst LoadMore = styled(Stack)`\n  padding: ${({ theme }) => theme.space[0.5]};\n`\n\nconst moveFocusDown = () => {\n  const options = document.querySelectorAll(\n    '#items > div[role=\"option\"]:not([disabled])',\n  )\n  const activeItem = document.activeElement\n  if (options) {\n    for (let i = 0; i < options?.length; i += 1) {\n      const listLength = options.length\n      if (activeItem === options[i] && activeItem !== options[listLength - 1]) {\n        ;(options[i + 1] as HTMLElement).focus()\n      }\n    }\n  }\n}\nconst moveFocusUp = () => {\n  const options = document.querySelectorAll(\n    '#items > div[role=\"option\"]:not([disabled])',\n  )\n  const activeItem = document.activeElement\n\n  if (options) {\n    for (let i = 0; i < options.length; i += 1) {\n      if (activeItem === options[i] && activeItem !== options[0]) {\n        ;(options[i - 1] as HTMLElement).focus()\n      }\n    }\n  }\n}\nconst handleKeyDownSelect = (event: KeyboardEvent<HTMLDivElement>) => {\n  if (event.key === 'ArrowDown') {\n    event.preventDefault()\n    moveFocusDown()\n  }\n\n  if (event.key === 'ArrowUp') {\n    event.preventDefault()\n    moveFocusUp()\n  }\n\n  if (event.key === ' ') {\n    // No scroll\n    event.preventDefault()\n  }\n}\n\nconst handleKeyDown = (\n  event: globalThis.KeyboardEvent,\n  ref: RefObject<HTMLDivElement | null>,\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].find(option =>\n        option.searchText\n          ? option.searchText.toLocaleLowerCase().startsWith(currentSearch)\n          : option.value.toLocaleLowerCase().startsWith(currentSearch),\n      )\n      if (closestOption) {\n        setDefaultSearch(closestOption.searchText ?? closestOption.value)\n      } else {\n        setDefaultSearch(null)\n      }\n    }\n  }\n}\nconst CreateDropdown = ({\n  isEmpty,\n  emptyState,\n  descriptionDirection,\n  loadMore,\n  optionalInfoPlacement,\n  defaultSearchValue,\n  isLoading,\n}: CreateDropdownProps) => {\n  const {\n    setIsDropdownVisible,\n    onChange,\n    options,\n    multiselect,\n    selectAll,\n    selectAllGroup,\n    displayedOptions,\n    setSelectedData,\n    selectedData,\n  } = useSelectInput()\n  const focusedItemRef = useRef<HTMLDivElement>(null)\n\n  useEffect(() => {\n    if (defaultSearchValue && focusedItemRef?.current) {\n      focusedItemRef.current.focus()\n    }\n  }, [defaultSearchValue])\n\n  if (isEmpty) {\n    return (\n      <EmptyState alignItems=\"center\" gap={2}>\n        {emptyState ?? (\n          <Text as=\"p\" variant=\"bodyStrong\">\n            No options\n          </Text>\n        )}\n      </EmptyState>\n    )\n  }\n\n  const handleClick = (clickedOption: OptionType, group?: string) => {\n    setSelectedData({ clickedOption, group, type: 'selectOption' })\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({ selectedGroup: group, type: 'selectGroup' })\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      data-grouped\n      id=\"select-dropdown\"\n      onKeyDown={handleKeyDownSelect}\n      role=\"listbox\"\n    >\n      {isLoading ? (\n        <Skeleton variant=\"block\" />\n      ) : (\n        <>\n          {selectAll && multiselect ? (\n            <Stack id=\"items\">\n              <DropdownItem\n                aria-disabled={false}\n                aria-label=\"select-all\"\n                aria-selected={selectedData.allSelected}\n                data-testid=\"select-all\"\n                id=\"select-all\"\n                onClick={selectAllOptions}\n                onKeyDown={event =>\n                  [' ', 'Enter'].includes(event.key) ? selectAllOptions() : null\n                }\n                role=\"option\"\n                tabIndex={0}\n              >\n                <StyledCheckbox\n                  checked={selectedData.allSelected}\n                  data-testid=\"select-all-checkbox\"\n                  disabled={false}\n                  onChange={selectAllOptions}\n                  tabIndex={-1}\n                  value=\"select-all\"\n                >\n                  <Stack direction=\"column\">\n                    <Text as=\"span\" placement=\"left\" variant=\"body\">\n                      {selectAll.label}\n                    </Text>\n                    <Text\n                      as=\"span\"\n                      placement=\"left\"\n                      prominence=\"weak\"\n                      sentiment=\"neutral\"\n                      variant=\"bodySmall\"\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 gap={0.25} key={group}>\n              {displayedOptions[group].length > 0 ? (\n                <DropdownGroupWrapper id={selectAllGroup ? 'items' : undefined}>\n                  {group ? (\n                    <DropdownGroup\n                      data-selectgroup={selectAllGroup}\n                      data-testid={`group-${index}`}\n                      key={group}\n                      onClick={() =>\n                        selectAllGroup ? handleSelectGroup(group) : null\n                      }\n                      onKeyDown={event => {\n                        if ([' ', 'Enter'].includes(event.key)) {\n                          event.preventDefault()\n                          handleSelectGroup(group)\n                        }\n                      }}\n                      role=\"group\"\n                      tabIndex={selectAllGroup ? 0 : -1}\n                      type=\"button\"\n                    >\n                      {selectAllGroup ? (\n                        <StyledCheckbox\n                          checked={selectedData.selectedGroups.includes(group)}\n                          data-testid=\"select-group\"\n                          disabled={false}\n                          onChange={() =>\n                            selectAllGroup ? handleSelectGroup(group) : null\n                          }\n                          tabIndex={-1}\n                          value={group}\n                        >\n                          <Text\n                            as=\"span\"\n                            placement=\"left\"\n                            sentiment=\"neutral\"\n                            variant=\"caption\"\n                          >\n                            {group.toUpperCase()}\n                          </Text>\n                        </StyledCheckbox>\n                      ) : (\n                        <Text\n                          as=\"span\"\n                          placement=\"left\"\n                          sentiment=\"neutral\"\n                          variant=\"caption\"\n                        >\n                          {group.toUpperCase()}\n                        </Text>\n                      )}\n                    </DropdownGroup>\n                  ) : null}\n                </DropdownGroupWrapper>\n              ) : null}\n              <Stack gap=\"0.25\" id=\"items\">\n                {displayedOptions[group].map((option, indexOption) => (\n                  <DropdownItem\n                    aria-disabled={!!option.disabled}\n                    aria-label={option.value}\n                    aria-selected={\n                      selectedData.selectedValues.includes(option.value) &&\n                      !option.disabled\n                    }\n                    data-testid={`option-${option.value}`}\n                    id={`option-${indexOption}`}\n                    key={option.value}\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                    role=\"option\"\n                    tabIndex={!option.disabled ? 0 : -1}\n                  >\n                    {multiselect ? (\n                      <StyledCheckbox\n                        checked={\n                          selectedData.selectedValues.includes(option.value) &&\n                          !option.disabled\n                        }\n                        disabled={option.disabled}\n                        onChange={() => {\n                          if (!option.disabled) {\n                            handleClick(option, group)\n                          }\n                        }}\n                        tabIndex={-1}\n                        value={option.value}\n                      >\n                        <DisplayOption\n                          descriptionDirection={descriptionDirection}\n                          option={option}\n                          optionalInfoPlacement={optionalInfoPlacement}\n                        />\n                      </StyledCheckbox>\n                    ) : (\n                      <DisplayOption\n                        descriptionDirection={descriptionDirection}\n                        option={option}\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      data-grouped={false}\n      gap={0.25}\n      id=\"select-dropdown\"\n      onKeyDown={handleKeyDownSelect}\n      role=\"listbox\"\n    >\n      {selectAll && multiselect ? (\n        <Stack gap={0.25} id=\"items\" tabIndex={-1}>\n          <DropdownItem\n            aria-disabled={false}\n            aria-label=\"select-all\"\n            aria-selected={selectedData.allSelected}\n            data-testid=\"select-all\"\n            onClick={selectAllOptions}\n            onKeyDown={event =>\n              [' ', 'Enter'].includes(event.key) ? selectAllOptions() : null\n            }\n            role=\"option\"\n            tabIndex={0}\n          >\n            <StyledCheckbox\n              checked={selectedData.allSelected}\n              data-testid=\"select-all-checkbox\"\n              disabled={false}\n              onChange={selectAllOptions}\n              tabIndex={-1}\n              value=\"select-all\"\n            >\n              <Stack direction=\"column\">\n                <Text as=\"span\" placement=\"left\" variant=\"body\">\n                  {selectAll.label}\n                </Text>\n                <Text\n                  as=\"span\"\n                  placement=\"left\"\n                  prominence=\"weak\"\n                  sentiment=\"neutral\"\n                  variant=\"bodySmall\"\n                >\n                  {selectAll.description}\n                </Text>\n              </Stack>\n            </StyledCheckbox>\n          </DropdownItem>\n        </Stack>\n      ) : null}\n      <Stack gap={0.25} id=\"items\">\n        {isLoading ? (\n          <Skeleton variant=\"block\" />\n        ) : (\n          displayedOptions.map((option, index) => (\n            <DropdownItem\n              aria-disabled={!!option.disabled}\n              aria-label={option.value}\n              aria-selected={\n                selectedData.selectedValues.includes(option.value) &&\n                !option.disabled\n              }\n              data-testid={`option-${option.value}`}\n              id={`option-${index}`}\n              key={option.value}\n              onClick={() => {\n                if (!option.disabled) {\n                  handleClick(option)\n                }\n              }}\n              onKeyDown={event =>\n                [' ', 'Enter'].includes(event.key) ? handleClick(option) : null\n              }\n              ref={\n                option.value === defaultSearchValue ||\n                option.searchText === defaultSearchValue\n                  ? focusedItemRef\n                  : null\n              }\n              role=\"option\"\n              tabIndex={!option.disabled ? 0 : -1}\n            >\n              {multiselect ? (\n                <StyledCheckbox\n                  checked={\n                    selectedData.selectedValues.includes(option.value) &&\n                    !option.disabled\n                  }\n                  disabled={option.disabled}\n                  onChange={() => {\n                    if (!option.disabled) {\n                      handleClick(option)\n                    }\n                  }}\n                  tabIndex={-1}\n                  value={option.value}\n                >\n                  <DisplayOption\n                    descriptionDirection={descriptionDirection}\n                    option={option}\n                    optionalInfoPlacement={optionalInfoPlacement}\n                  />\n                </StyledCheckbox>\n              ) : (\n                <DisplayOption\n                  descriptionDirection={descriptionDirection}\n                  option={option}\n                  optionalInfoPlacement={optionalInfoPlacement}\n                />\n              )}\n            </DropdownItem>\n          ))\n        )}\n        {loadMore ? <LoadMore>{loadMore}</LoadMore> : null}\n      </Stack>\n    </DropdownContainer>\n  )\n}\nexport const Dropdown = ({\n  children,\n  emptyState,\n  descriptionDirection,\n  searchable,\n  placeholder,\n  footer,\n  refSelect,\n  loadMore,\n  optionalInfoPlacement,\n  isLoading,\n  size,\n  dropdownAlign,\n  portalTarget,\n  id,\n}: DropdownProps) => {\n  const {\n    setIsDropdownVisible,\n    isDropdownVisible,\n    onSearch,\n    searchInput,\n    options,\n    displayedOptions,\n    numberOfOptions,\n  } = useSelectInput()\n  const theme = useTheme()\n  const [searchBarActive, setSearchBarActive] = useState(false)\n  const [defaultSearchValue, setDefaultSearch] = useState<string | null>(null)\n  const ref = useRef<HTMLDivElement>(null)\n  const [search, setSearch] = useState<string>('')\n  const [maxWidth, setWidth] = useState<string | number>()\n  const modalContext = useContext(ModalContext)\n\n  useEffect(() => {\n    if (refSelect.current && isDropdownVisible) {\n      const position =\n        refSelect.current.getBoundingClientRect().bottom +\n        DROPDOWN_MAX_HEIGHT +\n        Number(theme.sizing[INPUT_SIZE_HEIGHT[size]].replace('rem', '')) * 16 +\n        Number.parseInt(theme.space['5'], 10)\n      const overflow = position - window.innerHeight + 32\n      if (overflow > 0 && modalContext) {\n        const currentModal = modalContext.openedModals[0]\n        const modalElement = currentModal?.ref.current\n\n        if (modalElement) {\n          const parentElement = modalElement.parentNode as HTMLElement\n          if (parentElement) {\n            parentElement.scrollBy({\n              behavior: 'smooth',\n              top: overflow,\n            })\n          }\n        } else {\n          window.scrollBy({ behavior: 'smooth', top: overflow })\n        }\n      }\n    }\n    // oxlint-disable-next-line react/exhaustive-deps\n  }, [isDropdownVisible, refSelect, size, ref.current])\n\n  const resizeDropdown = useCallback(() => {\n    if (\n      refSelect.current &&\n      refSelect.current.getBoundingClientRect().width > 0\n    ) {\n      setWidth(refSelect.current.getBoundingClientRect().width)\n    }\n  }, [refSelect])\n\n  useEffect(() => {\n    resizeDropdown()\n\n    window.addEventListener('resize', resizeDropdown)\n\n    return () => window.removeEventListener('resize', resizeDropdown)\n  }, [resizeDropdown])\n\n  useEffect(() => {\n    if (!searchInput) {\n      onSearch(options)\n    }\n  }, [onSearch, options, searchInput])\n\n  useEffect(() => {\n    if (!isDropdownVisible) {\n      setDefaultSearch(null)\n      setSearch('')\n    }\n\n    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      if (!searchable) {\n        document.removeEventListener('keydown', event =>\n          handleKeyDown(\n            event,\n            ref,\n            options,\n            searchBarActive,\n            setSearch,\n            setDefaultSearch,\n            search,\n          ),\n        )\n      }\n    }\n  }, [\n    isDropdownVisible,\n    searchBarActive,\n    options,\n    onSearch,\n    search,\n    refSelect,\n    setDefaultSearch,\n    setIsDropdownVisible,\n    searchable,\n  ])\n\n  // No data is displayed (because of the search or because no data is provided)\n  // Set to true when noData by default\n  const isEmpty = useMemo(() => {\n    if (numberOfOptions === 0) {\n      return true\n    }\n    if (Array.isArray(displayedOptions)) {\n      return displayedOptions.length === 0\n    }\n    const groups = Object.keys(displayedOptions)\n    for (const group of groups) {\n      if (displayedOptions[group].length > 0) {\n        return false\n      }\n    }\n\n    return true\n  }, [displayedOptions, numberOfOptions])\n\n  const computedFooter = useMemo(() => {\n    if (footer && !isEmpty) {\n      if (typeof footer === 'function') {\n        return (\n          <PopupFooter>{footer(() => setIsDropdownVisible(false))}</PopupFooter>\n        )\n      }\n\n      return <PopupFooter>{footer}</PopupFooter>\n    }\n\n    return null\n  }, [isEmpty, footer, setIsDropdownVisible])\n\n  return (\n    <StyledPopup\n      align={dropdownAlign ?? 'start'}\n      containerFullWidth\n      debounceDelay={0}\n      disableAnimation\n      hasArrow={false}\n      hideOnClickOutside\n      id={id}\n      maxWidth={maxWidth ?? refSelect.current?.offsetWidth}\n      onClose={() => setIsDropdownVisible(false)}\n      placement=\"bottom\"\n      portalTarget={portalTarget}\n      ref={ref}\n      role=\"dialog\"\n      tabIndex={-1}\n      text={\n        <Stack>\n          {searchable && !isLoading && numberOfOptions >= 6 ? (\n            <SearchBarDropdown\n              displayedOptions={displayedOptions}\n              placeholder={placeholder}\n              setSearchBarActive={setSearchBarActive}\n            />\n          ) : null}\n          <CreateDropdown\n            defaultSearchValue={defaultSearchValue}\n            descriptionDirection={descriptionDirection}\n            emptyState={emptyState}\n            isEmpty={isEmpty}\n            isLoading={isLoading}\n            loadMore={loadMore}\n            optionalInfoPlacement={optionalInfoPlacement}\n          />\n          {computedFooter}\n        </Stack>\n      }\n      visible={isDropdownVisible}\n    >\n      {children}\n    </StyledPopup>\n  )\n}\n"]} */"));
|
|
144
144
|
const StyledCheckbox = /* @__PURE__ */ _styled(Checkbox, process.env.NODE_ENV === "production" ? {
|
|
145
145
|
target: "e12qyldb2"
|
|
146
146
|
} : {
|
|
@@ -151,7 +151,7 @@ const StyledCheckbox = /* @__PURE__ */ _styled(Checkbox, process.env.NODE_ENV ==
|
|
|
151
151
|
styles: "width:100%;position:static;text-align:left;align-items:center;pointer-events:none"
|
|
152
152
|
} : {
|
|
153
153
|
name: "1l9xw77",
|
|
154
|
-
styles: "width:100%;position:static;text-align:left;align-items:center;pointer-events:none/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInput/Dropdown.tsx"],"names":[],"mappings":"AAoLuC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInput/Dropdown.tsx","sourcesContent":["'use client'\n\nimport { useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport type {\n  ComponentProps,\n  Dispatch,\n  KeyboardEvent,\n  ReactNode,\n  RefObject,\n  SetStateAction,\n} from 'react'\nimport {\n  useCallback,\n  useContext,\n  useEffect,\n  useMemo,\n  useRef,\n  useState,\n} from 'react'\nimport { Checkbox } from '../Checkbox'\nimport { ModalContext } from '../Modal/ModalProvider'\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'\nimport { INPUT_SIZE_HEIGHT } from './types'\n\nconst DROPDOWN_MAX_HEIGHT = 256\n\nexport type DropdownProps = {\n  id?: string\n  children: ReactNode\n  emptyState: ReactNode\n  descriptionDirection: 'row' | 'column'\n  searchable: boolean\n  placeholder: string\n  footer?: ((closeDropdown: () => void) => ReactNode) | ReactNode\n  refSelect: RefObject<HTMLDivElement | null>\n  loadMore?: ReactNode\n  optionalInfoPlacement: 'left' | 'right'\n  isLoading?: boolean\n  size: 'small' | 'medium' | 'large'\n  dropdownAlign?: ComponentProps<typeof Popup>['align']\n  portalTarget?: ComponentProps<typeof Popup>['portalTarget']\n}\n\nexport type CreateDropdownProps = {\n  isEmpty: boolean\n  emptyState: ReactNode\n  descriptionDirection: 'row' | 'column'\n  loadMore?: ReactNode\n  optionalInfoPlacement: 'left' | 'right'\n  defaultSearchValue: string | null\n  isLoading?: boolean\n}\n\nconst NON_SEARCHABLE_KEYS = [\n  'Tab',\n  ' ',\n  'Enter',\n  'CapsLock',\n  'Shift',\n  'ArrowDown',\n  'ArrowUp',\n  'ArrowLeft',\n  'ArrowRight',\n  'Escape',\n]\n\nconst StyledPopup = styled(Popup)`\n  width: 100%;\n  min-width: 320px;\n  background-color: ${({ theme }) =>\n    theme.colors.other.elevation.background.raised};\n  color: ${({ theme }) => theme.colors.neutral.text};\n  box-shadow: ${({ theme }) =>\n    `${theme.shadows.raised[0]}, ${theme.shadows.raised[1]}`};\n  padding: ${({ theme }) => theme.space[0]};\n  margin-bottom: ${({ theme }) => theme.space[10]};\n`\n\nconst DropdownContainer = styled(Stack)<{ 'data-grouped': boolean }>`\n  max-height: ${DROPDOWN_MAX_HEIGHT}px;\n  overflow-y: auto;\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: 0;\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: 0;\n`\nconst DropdownItem = styled.div<{\n  'aria-selected': boolean\n  'aria-disabled': boolean\n}>`\n  text-align:left;\n  border: none;\n  background-color: ${({ theme }) =>\n    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  border: 1px transparent solid;\n\n  &:hover, :focus {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n    color: ${({ theme }) => theme.colors.primary.text};\n    cursor: pointer;\n  }\n\n  &[aria-selected='true'] {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n  }\n\n  &[aria-disabled=\"true\"] {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n  }\n\n  &[aria-disabled=\"true\"]:hover, [aria-disabled=\"true\"]:focus {\n    background-color: ${({ theme }) =>\n      theme.colors.neutral.backgroundStrongDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textStrongDisabled};\n    cursor: not-allowed;\n    outline: none;\n  }\n`\n\nconst PopupFooter = styled.div`\n  width: 100%;\n  padding: ${({ theme }) => theme.space[1.5]} ${({ theme }) => theme.space[2]}\n    ${({ theme }) => theme.space[1.5]} ${({ theme }) => theme.space[2]};\n  box-shadow: ${({ theme }) => theme.shadows.dropdown};\n`\nconst StyledCheckbox = styled(Checkbox)`\n  width: 100%;\n  position: static;\n  text-align: left;\n  align-items: center;\n  pointer-events: none;\n` // pointer-events: none prevents any error when using the checkbox in a form since it is an unnamed input\n\nconst EmptyState = styled(Stack)`\n  padding: ${({ theme }) => theme.space[2]};\n`\nconst LoadMore = styled(Stack)`\n  padding: ${({ theme }) => theme.space[0.5]};\n`\n\nconst moveFocusDown = () => {\n  const options = document.querySelectorAll(\n    '#items > div[role=\"option\"]:not([disabled])',\n  )\n  const activeItem = document.activeElement\n  if (options) {\n    for (let i = 0; i < options?.length; i += 1) {\n      const listLength = options.length\n      if (activeItem === options[i] && activeItem !== options[listLength - 1]) {\n        ;(options[i + 1] as HTMLElement).focus()\n      }\n    }\n  }\n}\nconst moveFocusUp = () => {\n  const options = document.querySelectorAll(\n    '#items > div[role=\"option\"]:not([disabled])',\n  )\n  const activeItem = document.activeElement\n\n  if (options) {\n    for (let i = 0; i < options.length; i += 1) {\n      if (activeItem === options[i] && activeItem !== options[0]) {\n        ;(options[i - 1] as HTMLElement).focus()\n      }\n    }\n  }\n}\nconst handleKeyDownSelect = (event: KeyboardEvent<HTMLDivElement>) => {\n  if (event.key === 'ArrowDown') {\n    event.preventDefault()\n    moveFocusDown()\n  }\n\n  if (event.key === 'ArrowUp') {\n    event.preventDefault()\n    moveFocusUp()\n  }\n\n  if (event.key === ' ') {\n    // No scroll\n    event.preventDefault()\n  }\n}\n\nconst handleKeyDown = (\n  event: globalThis.KeyboardEvent,\n  ref: RefObject<HTMLDivElement | null>,\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].find(option =>\n        option.searchText\n          ? option.searchText.toLocaleLowerCase().startsWith(currentSearch)\n          : option.value.toLocaleLowerCase().startsWith(currentSearch),\n      )\n      if (closestOption) {\n        setDefaultSearch(closestOption.searchText ?? closestOption.value)\n      } else {\n        setDefaultSearch(null)\n      }\n    }\n  }\n}\nconst CreateDropdown = ({\n  isEmpty,\n  emptyState,\n  descriptionDirection,\n  loadMore,\n  optionalInfoPlacement,\n  defaultSearchValue,\n  isLoading,\n}: CreateDropdownProps) => {\n  const {\n    setIsDropdownVisible,\n    onChange,\n    options,\n    multiselect,\n    selectAll,\n    selectAllGroup,\n    displayedOptions,\n    setSelectedData,\n    selectedData,\n  } = useSelectInput()\n  const focusedItemRef = useRef<HTMLDivElement>(null)\n\n  useEffect(() => {\n    if (defaultSearchValue && focusedItemRef?.current) {\n      focusedItemRef.current.focus()\n    }\n  }, [defaultSearchValue])\n\n  if (isEmpty) {\n    return (\n      <EmptyState alignItems=\"center\" gap={2}>\n        {emptyState ?? (\n          <Text as=\"p\" variant=\"bodyStrong\">\n            No options\n          </Text>\n        )}\n      </EmptyState>\n    )\n  }\n\n  const handleClick = (clickedOption: OptionType, group?: string) => {\n    setSelectedData({ clickedOption, group, type: 'selectOption' })\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({ selectedGroup: group, type: 'selectGroup' })\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      data-grouped\n      id=\"select-dropdown\"\n      onKeyDown={handleKeyDownSelect}\n      role=\"listbox\"\n    >\n      {isLoading ? (\n        <Skeleton variant=\"block\" />\n      ) : (\n        <>\n          {selectAll && multiselect ? (\n            <Stack id=\"items\">\n              <DropdownItem\n                aria-disabled={false}\n                aria-label=\"select-all\"\n                aria-selected={selectedData.allSelected}\n                data-testid=\"select-all\"\n                id=\"select-all\"\n                onClick={selectAllOptions}\n                onKeyDown={event =>\n                  [' ', 'Enter'].includes(event.key) ? selectAllOptions() : null\n                }\n                role=\"option\"\n                tabIndex={0}\n              >\n                <StyledCheckbox\n                  checked={selectedData.allSelected}\n                  data-testid=\"select-all-checkbox\"\n                  disabled={false}\n                  onChange={selectAllOptions}\n                  tabIndex={-1}\n                  value=\"select-all\"\n                >\n                  <Stack direction=\"column\">\n                    <Text as=\"span\" placement=\"left\" variant=\"body\">\n                      {selectAll.label}\n                    </Text>\n                    <Text\n                      as=\"span\"\n                      placement=\"left\"\n                      prominence=\"weak\"\n                      sentiment=\"neutral\"\n                      variant=\"bodySmall\"\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 gap={0.25} key={group}>\n              {displayedOptions[group].length > 0 ? (\n                <DropdownGroupWrapper id={selectAllGroup ? 'items' : undefined}>\n                  {group ? (\n                    <DropdownGroup\n                      data-selectgroup={selectAllGroup}\n                      data-testid={`group-${index}`}\n                      key={group}\n                      onClick={() =>\n                        selectAllGroup ? handleSelectGroup(group) : null\n                      }\n                      onKeyDown={event => {\n                        if ([' ', 'Enter'].includes(event.key)) {\n                          event.preventDefault()\n                          handleSelectGroup(group)\n                        }\n                      }}\n                      role=\"group\"\n                      tabIndex={selectAllGroup ? 0 : -1}\n                      type=\"button\"\n                    >\n                      {selectAllGroup ? (\n                        <StyledCheckbox\n                          checked={selectedData.selectedGroups.includes(group)}\n                          data-testid=\"select-group\"\n                          disabled={false}\n                          onChange={() =>\n                            selectAllGroup ? handleSelectGroup(group) : null\n                          }\n                          tabIndex={-1}\n                          value={group}\n                        >\n                          <Text\n                            as=\"span\"\n                            placement=\"left\"\n                            sentiment=\"neutral\"\n                            variant=\"caption\"\n                          >\n                            {group.toUpperCase()}\n                          </Text>\n                        </StyledCheckbox>\n                      ) : (\n                        <Text\n                          as=\"span\"\n                          placement=\"left\"\n                          sentiment=\"neutral\"\n                          variant=\"caption\"\n                        >\n                          {group.toUpperCase()}\n                        </Text>\n                      )}\n                    </DropdownGroup>\n                  ) : null}\n                </DropdownGroupWrapper>\n              ) : null}\n              <Stack gap=\"0.25\" id=\"items\">\n                {displayedOptions[group].map((option, indexOption) => (\n                  <DropdownItem\n                    aria-disabled={!!option.disabled}\n                    aria-label={option.value}\n                    aria-selected={\n                      selectedData.selectedValues.includes(option.value) &&\n                      !option.disabled\n                    }\n                    data-testid={`option-${option.value}`}\n                    id={`option-${indexOption}`}\n                    key={option.value}\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                    role=\"option\"\n                    tabIndex={!option.disabled ? 0 : -1}\n                  >\n                    {multiselect ? (\n                      <StyledCheckbox\n                        checked={\n                          selectedData.selectedValues.includes(option.value) &&\n                          !option.disabled\n                        }\n                        disabled={option.disabled}\n                        onChange={() => {\n                          if (!option.disabled) {\n                            handleClick(option, group)\n                          }\n                        }}\n                        tabIndex={-1}\n                        value={option.value}\n                      >\n                        <DisplayOption\n                          descriptionDirection={descriptionDirection}\n                          option={option}\n                          optionalInfoPlacement={optionalInfoPlacement}\n                        />\n                      </StyledCheckbox>\n                    ) : (\n                      <DisplayOption\n                        descriptionDirection={descriptionDirection}\n                        option={option}\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      data-grouped={false}\n      gap={0.25}\n      id=\"select-dropdown\"\n      onKeyDown={handleKeyDownSelect}\n      role=\"listbox\"\n    >\n      {selectAll && multiselect ? (\n        <Stack gap={0.25} id=\"items\" tabIndex={-1}>\n          <DropdownItem\n            aria-disabled={false}\n            aria-label=\"select-all\"\n            aria-selected={selectedData.allSelected}\n            data-testid=\"select-all\"\n            onClick={selectAllOptions}\n            onKeyDown={event =>\n              [' ', 'Enter'].includes(event.key) ? selectAllOptions() : null\n            }\n            role=\"option\"\n            tabIndex={0}\n          >\n            <StyledCheckbox\n              checked={selectedData.allSelected}\n              data-testid=\"select-all-checkbox\"\n              disabled={false}\n              onChange={selectAllOptions}\n              tabIndex={-1}\n              value=\"select-all\"\n            >\n              <Stack direction=\"column\">\n                <Text as=\"span\" placement=\"left\" variant=\"body\">\n                  {selectAll.label}\n                </Text>\n                <Text\n                  as=\"span\"\n                  placement=\"left\"\n                  prominence=\"weak\"\n                  sentiment=\"neutral\"\n                  variant=\"bodySmall\"\n                >\n                  {selectAll.description}\n                </Text>\n              </Stack>\n            </StyledCheckbox>\n          </DropdownItem>\n        </Stack>\n      ) : null}\n      <Stack gap={0.25} id=\"items\">\n        {isLoading ? (\n          <Skeleton variant=\"block\" />\n        ) : (\n          displayedOptions.map((option, index) => (\n            <DropdownItem\n              aria-disabled={!!option.disabled}\n              aria-label={option.value}\n              aria-selected={\n                selectedData.selectedValues.includes(option.value) &&\n                !option.disabled\n              }\n              data-testid={`option-${option.value}`}\n              id={`option-${index}`}\n              key={option.value}\n              onClick={() => {\n                if (!option.disabled) {\n                  handleClick(option)\n                }\n              }}\n              onKeyDown={event =>\n                [' ', 'Enter'].includes(event.key) ? handleClick(option) : null\n              }\n              ref={\n                option.value === defaultSearchValue ||\n                option.searchText === defaultSearchValue\n                  ? focusedItemRef\n                  : null\n              }\n              role=\"option\"\n              tabIndex={!option.disabled ? 0 : -1}\n            >\n              {multiselect ? (\n                <StyledCheckbox\n                  checked={\n                    selectedData.selectedValues.includes(option.value) &&\n                    !option.disabled\n                  }\n                  disabled={option.disabled}\n                  onChange={() => {\n                    if (!option.disabled) {\n                      handleClick(option)\n                    }\n                  }}\n                  tabIndex={-1}\n                  value={option.value}\n                >\n                  <DisplayOption\n                    descriptionDirection={descriptionDirection}\n                    option={option}\n                    optionalInfoPlacement={optionalInfoPlacement}\n                  />\n                </StyledCheckbox>\n              ) : (\n                <DisplayOption\n                  descriptionDirection={descriptionDirection}\n                  option={option}\n                  optionalInfoPlacement={optionalInfoPlacement}\n                />\n              )}\n            </DropdownItem>\n          ))\n        )}\n        {loadMore ? <LoadMore>{loadMore}</LoadMore> : null}\n      </Stack>\n    </DropdownContainer>\n  )\n}\nexport const Dropdown = ({\n  children,\n  emptyState,\n  descriptionDirection,\n  searchable,\n  placeholder,\n  footer,\n  refSelect,\n  loadMore,\n  optionalInfoPlacement,\n  isLoading,\n  size,\n  dropdownAlign,\n  portalTarget,\n  id,\n}: DropdownProps) => {\n  const {\n    setIsDropdownVisible,\n    isDropdownVisible,\n    onSearch,\n    searchInput,\n    options,\n    displayedOptions,\n    numberOfOptions,\n  } = useSelectInput()\n  const theme = useTheme()\n  const [searchBarActive, setSearchBarActive] = useState(false)\n  const [defaultSearchValue, setDefaultSearch] = useState<string | null>(null)\n  const ref = useRef<HTMLDivElement>(null)\n  const [search, setSearch] = useState<string>('')\n  const [maxWidth, setWidth] = useState<string | number>()\n  const modalContext = useContext(ModalContext)\n\n  useEffect(() => {\n    if (refSelect.current && isDropdownVisible) {\n      const position =\n        refSelect.current.getBoundingClientRect().bottom +\n        DROPDOWN_MAX_HEIGHT +\n        Number(theme.sizing[INPUT_SIZE_HEIGHT[size]].replace('rem', '')) * 16 +\n        Number.parseInt(theme.space['5'], 10)\n      const overflow = position - window.innerHeight + 32\n      if (overflow > 0 && modalContext) {\n        const currentModal = modalContext.openedModals[0]\n        const modalElement = currentModal?.ref.current\n\n        if (modalElement) {\n          const parentElement = modalElement.parentNode as HTMLElement\n          if (parentElement) {\n            parentElement.scrollBy({\n              behavior: 'smooth',\n              top: overflow,\n            })\n          }\n        } else {\n          window.scrollBy({ behavior: 'smooth', top: overflow })\n        }\n      }\n    }\n    // oxlint-disable-next-line react/exhaustive-deps\n  }, [isDropdownVisible, refSelect, size, ref.current])\n\n  const resizeDropdown = useCallback(() => {\n    if (\n      refSelect.current &&\n      refSelect.current.getBoundingClientRect().width > 0\n    ) {\n      setWidth(refSelect.current.getBoundingClientRect().width)\n    }\n  }, [refSelect])\n\n  useEffect(() => {\n    resizeDropdown()\n\n    window.addEventListener('resize', resizeDropdown)\n\n    return () => window.removeEventListener('resize', resizeDropdown)\n  }, [resizeDropdown])\n\n  useEffect(() => {\n    if (!searchInput) {\n      onSearch(options)\n    }\n  }, [onSearch, options, searchInput])\n\n  useEffect(() => {\n    if (!isDropdownVisible) {\n      setDefaultSearch(null)\n      setSearch('')\n    }\n\n    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      if (!searchable) {\n        document.removeEventListener('keydown', event =>\n          handleKeyDown(\n            event,\n            ref,\n            options,\n            searchBarActive,\n            setSearch,\n            setDefaultSearch,\n            search,\n          ),\n        )\n      }\n    }\n  }, [\n    isDropdownVisible,\n    searchBarActive,\n    options,\n    onSearch,\n    search,\n    refSelect,\n    setDefaultSearch,\n    setIsDropdownVisible,\n    searchable,\n  ])\n\n  // No data is displayed (because of the search or because no data is provided)\n  // Set to true when noData by default\n  const isEmpty = useMemo(() => {\n    if (numberOfOptions === 0) {\n      return true\n    }\n    if (Array.isArray(displayedOptions)) {\n      return displayedOptions.length === 0\n    }\n    const groups = Object.keys(displayedOptions)\n    for (const group of groups) {\n      if (displayedOptions[group].length > 0) {\n        return false\n      }\n    }\n\n    return true\n  }, [displayedOptions, numberOfOptions])\n\n  const computedFooter = useMemo(() => {\n    if (footer && !isEmpty) {\n      if (typeof footer === 'function') {\n        return (\n          <PopupFooter>{footer(() => setIsDropdownVisible(false))}</PopupFooter>\n        )\n      }\n\n      return <PopupFooter>{footer}</PopupFooter>\n    }\n\n    return null\n  }, [isEmpty, footer, setIsDropdownVisible])\n\n  return (\n    <StyledPopup\n      align={dropdownAlign ?? 'start'}\n      containerFullWidth\n      debounceDelay={0}\n      disableAnimation\n      hasArrow={false}\n      hideOnClickOutside\n      id={id}\n      maxWidth={maxWidth ?? refSelect.current?.offsetWidth}\n      onClose={() => setIsDropdownVisible(false)}\n      placement=\"bottom\"\n      portalTarget={portalTarget}\n      ref={ref}\n      role=\"dialog\"\n      tabIndex={-1}\n      text={\n        <Stack>\n          {searchable && !isLoading && numberOfOptions >= 6 ? (\n            <SearchBarDropdown\n              displayedOptions={displayedOptions}\n              placeholder={placeholder}\n              setSearchBarActive={setSearchBarActive}\n            />\n          ) : null}\n          <CreateDropdown\n            defaultSearchValue={defaultSearchValue}\n            descriptionDirection={descriptionDirection}\n            emptyState={emptyState}\n            isEmpty={isEmpty}\n            isLoading={isLoading}\n            loadMore={loadMore}\n            optionalInfoPlacement={optionalInfoPlacement}\n          />\n          {computedFooter}\n        </Stack>\n      }\n      visible={isDropdownVisible}\n    >\n      {children}\n    </StyledPopup>\n  )\n}\n"]} */",
|
|
154
|
+
styles: "width:100%;position:static;text-align:left;align-items:center;pointer-events:none/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInput/Dropdown.tsx"],"names":[],"mappings":"AAqLuC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInput/Dropdown.tsx","sourcesContent":["'use client'\n\nimport { useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport type {\n  ComponentProps,\n  Dispatch,\n  KeyboardEvent,\n  ReactNode,\n  RefObject,\n  SetStateAction,\n} from 'react'\nimport {\n  useCallback,\n  useContext,\n  useEffect,\n  useMemo,\n  useRef,\n  useState,\n} from 'react'\nimport { Checkbox } from '../Checkbox'\nimport { ModalContext } from '../Modal/ModalProvider'\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'\nimport { INPUT_SIZE_HEIGHT } from './types'\n\nconst DROPDOWN_MAX_HEIGHT = 256\n\nexport type DropdownProps = {\n  id?: string\n  children: ReactNode\n  emptyState: ReactNode\n  descriptionDirection: 'row' | 'column'\n  searchable: boolean\n  placeholder: string\n  footer?: ((closeDropdown: () => void) => ReactNode) | ReactNode\n  refSelect: RefObject<HTMLDivElement | null>\n  loadMore?: ReactNode\n  optionalInfoPlacement: 'left' | 'right'\n  isLoading?: boolean\n  size: 'small' | 'medium' | 'large'\n  dropdownAlign?: ComponentProps<typeof Popup>['align']\n  portalTarget?: ComponentProps<typeof Popup>['portalTarget']\n}\n\nexport type CreateDropdownProps = {\n  isEmpty: boolean\n  emptyState: ReactNode\n  descriptionDirection: 'row' | 'column'\n  loadMore?: ReactNode\n  optionalInfoPlacement: 'left' | 'right'\n  defaultSearchValue: string | null\n  isLoading?: boolean\n}\n\nconst NON_SEARCHABLE_KEYS = [\n  'Tab',\n  ' ',\n  'Enter',\n  'CapsLock',\n  'Shift',\n  'ArrowDown',\n  'ArrowUp',\n  'ArrowLeft',\n  'ArrowRight',\n  'Escape',\n]\n\nconst StyledPopup = styled(Popup)`\n  width: 100%;\n  min-width: 320px;\n  background-color: ${({ theme }) =>\n    theme.colors.other.elevation.background.raised};\n  color: ${({ theme }) => theme.colors.neutral.text};\n  box-shadow: ${({ theme }) =>\n    `${theme.shadows.raised[0]}, ${theme.shadows.raised[1]}`};\n  padding: ${({ theme }) => theme.space[0]};\n  margin-bottom: ${({ theme }) => theme.space[10]};\n`\n\nconst DropdownContainer = styled(Stack)<{ 'data-grouped': boolean }>`\n  max-height: ${DROPDOWN_MAX_HEIGHT}px;\n  overflow-y: auto;\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: 0;\n  padding-right: ${({ theme }) => theme.space[2]};\n  padding-left: ${({ theme }) => theme.space[2]};\n  height: ${({ theme }) => theme.space[4]};\n  display: flex;\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: 0;\n`\nconst DropdownItem = styled.div<{\n  'aria-selected': boolean\n  'aria-disabled': boolean\n}>`\n  text-align:left;\n  border: none;\n  background-color: ${({ theme }) =>\n    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  border: 1px transparent solid;\n\n  &:hover, :focus {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n    color: ${({ theme }) => theme.colors.primary.text};\n    cursor: pointer;\n  }\n\n  &[aria-selected='true'] {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n  }\n\n  &[aria-disabled=\"true\"] {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n  }\n\n  &[aria-disabled=\"true\"]:hover, [aria-disabled=\"true\"]:focus {\n    background-color: ${({ theme }) =>\n      theme.colors.neutral.backgroundStrongDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textStrongDisabled};\n    cursor: not-allowed;\n    outline: none;\n  }\n`\n\nconst PopupFooter = styled.div`\n  width: 100%;\n  padding: ${({ theme }) => theme.space[1.5]} ${({ theme }) => theme.space[2]}\n    ${({ theme }) => theme.space[1.5]} ${({ theme }) => theme.space[2]};\n  box-shadow: ${({ theme }) => theme.shadows.dropdown};\n`\nconst StyledCheckbox = styled(Checkbox)`\n  width: 100%;\n  position: static;\n  text-align: left;\n  align-items: center;\n  pointer-events: none;\n` // pointer-events: none prevents any error when using the checkbox in a form since it is an unnamed input\n\nconst EmptyState = styled(Stack)`\n  padding: ${({ theme }) => theme.space[2]};\n`\nconst LoadMore = styled(Stack)`\n  padding: ${({ theme }) => theme.space[0.5]};\n`\n\nconst moveFocusDown = () => {\n  const options = document.querySelectorAll(\n    '#items > div[role=\"option\"]:not([disabled])',\n  )\n  const activeItem = document.activeElement\n  if (options) {\n    for (let i = 0; i < options?.length; i += 1) {\n      const listLength = options.length\n      if (activeItem === options[i] && activeItem !== options[listLength - 1]) {\n        ;(options[i + 1] as HTMLElement).focus()\n      }\n    }\n  }\n}\nconst moveFocusUp = () => {\n  const options = document.querySelectorAll(\n    '#items > div[role=\"option\"]:not([disabled])',\n  )\n  const activeItem = document.activeElement\n\n  if (options) {\n    for (let i = 0; i < options.length; i += 1) {\n      if (activeItem === options[i] && activeItem !== options[0]) {\n        ;(options[i - 1] as HTMLElement).focus()\n      }\n    }\n  }\n}\nconst handleKeyDownSelect = (event: KeyboardEvent<HTMLDivElement>) => {\n  if (event.key === 'ArrowDown') {\n    event.preventDefault()\n    moveFocusDown()\n  }\n\n  if (event.key === 'ArrowUp') {\n    event.preventDefault()\n    moveFocusUp()\n  }\n\n  if (event.key === ' ') {\n    // No scroll\n    event.preventDefault()\n  }\n}\n\nconst handleKeyDown = (\n  event: globalThis.KeyboardEvent,\n  ref: RefObject<HTMLDivElement | null>,\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].find(option =>\n        option.searchText\n          ? option.searchText.toLocaleLowerCase().startsWith(currentSearch)\n          : option.value.toLocaleLowerCase().startsWith(currentSearch),\n      )\n      if (closestOption) {\n        setDefaultSearch(closestOption.searchText ?? closestOption.value)\n      } else {\n        setDefaultSearch(null)\n      }\n    }\n  }\n}\nconst CreateDropdown = ({\n  isEmpty,\n  emptyState,\n  descriptionDirection,\n  loadMore,\n  optionalInfoPlacement,\n  defaultSearchValue,\n  isLoading,\n}: CreateDropdownProps) => {\n  const {\n    setIsDropdownVisible,\n    onChange,\n    options,\n    multiselect,\n    selectAll,\n    selectAllGroup,\n    displayedOptions,\n    setSelectedData,\n    selectedData,\n  } = useSelectInput()\n  const focusedItemRef = useRef<HTMLDivElement>(null)\n\n  useEffect(() => {\n    if (defaultSearchValue && focusedItemRef?.current) {\n      focusedItemRef.current.focus()\n    }\n  }, [defaultSearchValue])\n\n  if (isEmpty) {\n    return (\n      <EmptyState alignItems=\"center\" gap={2}>\n        {emptyState ?? (\n          <Text as=\"p\" variant=\"bodyStrong\">\n            No options\n          </Text>\n        )}\n      </EmptyState>\n    )\n  }\n\n  const handleClick = (clickedOption: OptionType, group?: string) => {\n    setSelectedData({ clickedOption, group, type: 'selectOption' })\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({ selectedGroup: group, type: 'selectGroup' })\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      data-grouped\n      id=\"select-dropdown\"\n      onKeyDown={handleKeyDownSelect}\n      role=\"listbox\"\n    >\n      {isLoading ? (\n        <Skeleton variant=\"block\" />\n      ) : (\n        <>\n          {selectAll && multiselect ? (\n            <Stack id=\"items\">\n              <DropdownItem\n                aria-disabled={false}\n                aria-label=\"select-all\"\n                aria-selected={selectedData.allSelected}\n                data-testid=\"select-all\"\n                id=\"select-all\"\n                onClick={selectAllOptions}\n                onKeyDown={event =>\n                  [' ', 'Enter'].includes(event.key) ? selectAllOptions() : null\n                }\n                role=\"option\"\n                tabIndex={0}\n              >\n                <StyledCheckbox\n                  checked={selectedData.allSelected}\n                  data-testid=\"select-all-checkbox\"\n                  disabled={false}\n                  onChange={selectAllOptions}\n                  tabIndex={-1}\n                  value=\"select-all\"\n                >\n                  <Stack direction=\"column\">\n                    <Text as=\"span\" placement=\"left\" variant=\"body\">\n                      {selectAll.label}\n                    </Text>\n                    <Text\n                      as=\"span\"\n                      placement=\"left\"\n                      prominence=\"weak\"\n                      sentiment=\"neutral\"\n                      variant=\"bodySmall\"\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 gap={0.25} key={group}>\n              {displayedOptions[group].length > 0 ? (\n                <DropdownGroupWrapper id={selectAllGroup ? 'items' : undefined}>\n                  {group ? (\n                    <DropdownGroup\n                      data-selectgroup={selectAllGroup}\n                      data-testid={`group-${index}`}\n                      key={group}\n                      onClick={() =>\n                        selectAllGroup ? handleSelectGroup(group) : null\n                      }\n                      onKeyDown={event => {\n                        if ([' ', 'Enter'].includes(event.key)) {\n                          event.preventDefault()\n                          handleSelectGroup(group)\n                        }\n                      }}\n                      role=\"group\"\n                      tabIndex={selectAllGroup ? 0 : -1}\n                      type=\"button\"\n                    >\n                      {selectAllGroup ? (\n                        <StyledCheckbox\n                          checked={selectedData.selectedGroups.includes(group)}\n                          data-testid=\"select-group\"\n                          disabled={false}\n                          onChange={() =>\n                            selectAllGroup ? handleSelectGroup(group) : null\n                          }\n                          tabIndex={-1}\n                          value={group}\n                        >\n                          <Text\n                            as=\"span\"\n                            placement=\"left\"\n                            sentiment=\"neutral\"\n                            variant=\"caption\"\n                          >\n                            {group.toUpperCase()}\n                          </Text>\n                        </StyledCheckbox>\n                      ) : (\n                        <Text\n                          as=\"span\"\n                          placement=\"left\"\n                          sentiment=\"neutral\"\n                          variant=\"caption\"\n                        >\n                          {group.toUpperCase()}\n                        </Text>\n                      )}\n                    </DropdownGroup>\n                  ) : null}\n                </DropdownGroupWrapper>\n              ) : null}\n              <Stack gap=\"0.25\" id=\"items\">\n                {displayedOptions[group].map((option, indexOption) => (\n                  <DropdownItem\n                    aria-disabled={!!option.disabled}\n                    aria-label={option.value}\n                    aria-selected={\n                      selectedData.selectedValues.includes(option.value) &&\n                      !option.disabled\n                    }\n                    data-testid={`option-${option.value}`}\n                    id={`option-${indexOption}`}\n                    key={option.value}\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                    role=\"option\"\n                    tabIndex={!option.disabled ? 0 : -1}\n                  >\n                    {multiselect ? (\n                      <StyledCheckbox\n                        checked={\n                          selectedData.selectedValues.includes(option.value) &&\n                          !option.disabled\n                        }\n                        disabled={option.disabled}\n                        onChange={() => {\n                          if (!option.disabled) {\n                            handleClick(option, group)\n                          }\n                        }}\n                        tabIndex={-1}\n                        value={option.value}\n                      >\n                        <DisplayOption\n                          descriptionDirection={descriptionDirection}\n                          option={option}\n                          optionalInfoPlacement={optionalInfoPlacement}\n                        />\n                      </StyledCheckbox>\n                    ) : (\n                      <DisplayOption\n                        descriptionDirection={descriptionDirection}\n                        option={option}\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      data-grouped={false}\n      gap={0.25}\n      id=\"select-dropdown\"\n      onKeyDown={handleKeyDownSelect}\n      role=\"listbox\"\n    >\n      {selectAll && multiselect ? (\n        <Stack gap={0.25} id=\"items\" tabIndex={-1}>\n          <DropdownItem\n            aria-disabled={false}\n            aria-label=\"select-all\"\n            aria-selected={selectedData.allSelected}\n            data-testid=\"select-all\"\n            onClick={selectAllOptions}\n            onKeyDown={event =>\n              [' ', 'Enter'].includes(event.key) ? selectAllOptions() : null\n            }\n            role=\"option\"\n            tabIndex={0}\n          >\n            <StyledCheckbox\n              checked={selectedData.allSelected}\n              data-testid=\"select-all-checkbox\"\n              disabled={false}\n              onChange={selectAllOptions}\n              tabIndex={-1}\n              value=\"select-all\"\n            >\n              <Stack direction=\"column\">\n                <Text as=\"span\" placement=\"left\" variant=\"body\">\n                  {selectAll.label}\n                </Text>\n                <Text\n                  as=\"span\"\n                  placement=\"left\"\n                  prominence=\"weak\"\n                  sentiment=\"neutral\"\n                  variant=\"bodySmall\"\n                >\n                  {selectAll.description}\n                </Text>\n              </Stack>\n            </StyledCheckbox>\n          </DropdownItem>\n        </Stack>\n      ) : null}\n      <Stack gap={0.25} id=\"items\">\n        {isLoading ? (\n          <Skeleton variant=\"block\" />\n        ) : (\n          displayedOptions.map((option, index) => (\n            <DropdownItem\n              aria-disabled={!!option.disabled}\n              aria-label={option.value}\n              aria-selected={\n                selectedData.selectedValues.includes(option.value) &&\n                !option.disabled\n              }\n              data-testid={`option-${option.value}`}\n              id={`option-${index}`}\n              key={option.value}\n              onClick={() => {\n                if (!option.disabled) {\n                  handleClick(option)\n                }\n              }}\n              onKeyDown={event =>\n                [' ', 'Enter'].includes(event.key) ? handleClick(option) : null\n              }\n              ref={\n                option.value === defaultSearchValue ||\n                option.searchText === defaultSearchValue\n                  ? focusedItemRef\n                  : null\n              }\n              role=\"option\"\n              tabIndex={!option.disabled ? 0 : -1}\n            >\n              {multiselect ? (\n                <StyledCheckbox\n                  checked={\n                    selectedData.selectedValues.includes(option.value) &&\n                    !option.disabled\n                  }\n                  disabled={option.disabled}\n                  onChange={() => {\n                    if (!option.disabled) {\n                      handleClick(option)\n                    }\n                  }}\n                  tabIndex={-1}\n                  value={option.value}\n                >\n                  <DisplayOption\n                    descriptionDirection={descriptionDirection}\n                    option={option}\n                    optionalInfoPlacement={optionalInfoPlacement}\n                  />\n                </StyledCheckbox>\n              ) : (\n                <DisplayOption\n                  descriptionDirection={descriptionDirection}\n                  option={option}\n                  optionalInfoPlacement={optionalInfoPlacement}\n                />\n              )}\n            </DropdownItem>\n          ))\n        )}\n        {loadMore ? <LoadMore>{loadMore}</LoadMore> : null}\n      </Stack>\n    </DropdownContainer>\n  )\n}\nexport const Dropdown = ({\n  children,\n  emptyState,\n  descriptionDirection,\n  searchable,\n  placeholder,\n  footer,\n  refSelect,\n  loadMore,\n  optionalInfoPlacement,\n  isLoading,\n  size,\n  dropdownAlign,\n  portalTarget,\n  id,\n}: DropdownProps) => {\n  const {\n    setIsDropdownVisible,\n    isDropdownVisible,\n    onSearch,\n    searchInput,\n    options,\n    displayedOptions,\n    numberOfOptions,\n  } = useSelectInput()\n  const theme = useTheme()\n  const [searchBarActive, setSearchBarActive] = useState(false)\n  const [defaultSearchValue, setDefaultSearch] = useState<string | null>(null)\n  const ref = useRef<HTMLDivElement>(null)\n  const [search, setSearch] = useState<string>('')\n  const [maxWidth, setWidth] = useState<string | number>()\n  const modalContext = useContext(ModalContext)\n\n  useEffect(() => {\n    if (refSelect.current && isDropdownVisible) {\n      const position =\n        refSelect.current.getBoundingClientRect().bottom +\n        DROPDOWN_MAX_HEIGHT +\n        Number(theme.sizing[INPUT_SIZE_HEIGHT[size]].replace('rem', '')) * 16 +\n        Number.parseInt(theme.space['5'], 10)\n      const overflow = position - window.innerHeight + 32\n      if (overflow > 0 && modalContext) {\n        const currentModal = modalContext.openedModals[0]\n        const modalElement = currentModal?.ref.current\n\n        if (modalElement) {\n          const parentElement = modalElement.parentNode as HTMLElement\n          if (parentElement) {\n            parentElement.scrollBy({\n              behavior: 'smooth',\n              top: overflow,\n            })\n          }\n        } else {\n          window.scrollBy({ behavior: 'smooth', top: overflow })\n        }\n      }\n    }\n    // oxlint-disable-next-line react/exhaustive-deps\n  }, [isDropdownVisible, refSelect, size, ref.current])\n\n  const resizeDropdown = useCallback(() => {\n    if (\n      refSelect.current &&\n      refSelect.current.getBoundingClientRect().width > 0\n    ) {\n      setWidth(refSelect.current.getBoundingClientRect().width)\n    }\n  }, [refSelect])\n\n  useEffect(() => {\n    resizeDropdown()\n\n    window.addEventListener('resize', resizeDropdown)\n\n    return () => window.removeEventListener('resize', resizeDropdown)\n  }, [resizeDropdown])\n\n  useEffect(() => {\n    if (!searchInput) {\n      onSearch(options)\n    }\n  }, [onSearch, options, searchInput])\n\n  useEffect(() => {\n    if (!isDropdownVisible) {\n      setDefaultSearch(null)\n      setSearch('')\n    }\n\n    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      if (!searchable) {\n        document.removeEventListener('keydown', event =>\n          handleKeyDown(\n            event,\n            ref,\n            options,\n            searchBarActive,\n            setSearch,\n            setDefaultSearch,\n            search,\n          ),\n        )\n      }\n    }\n  }, [\n    isDropdownVisible,\n    searchBarActive,\n    options,\n    onSearch,\n    search,\n    refSelect,\n    setDefaultSearch,\n    setIsDropdownVisible,\n    searchable,\n  ])\n\n  // No data is displayed (because of the search or because no data is provided)\n  // Set to true when noData by default\n  const isEmpty = useMemo(() => {\n    if (numberOfOptions === 0) {\n      return true\n    }\n    if (Array.isArray(displayedOptions)) {\n      return displayedOptions.length === 0\n    }\n    const groups = Object.keys(displayedOptions)\n    for (const group of groups) {\n      if (displayedOptions[group].length > 0) {\n        return false\n      }\n    }\n\n    return true\n  }, [displayedOptions, numberOfOptions])\n\n  const computedFooter = useMemo(() => {\n    if (footer && !isEmpty) {\n      if (typeof footer === 'function') {\n        return (\n          <PopupFooter>{footer(() => setIsDropdownVisible(false))}</PopupFooter>\n        )\n      }\n\n      return <PopupFooter>{footer}</PopupFooter>\n    }\n\n    return null\n  }, [isEmpty, footer, setIsDropdownVisible])\n\n  return (\n    <StyledPopup\n      align={dropdownAlign ?? 'start'}\n      containerFullWidth\n      debounceDelay={0}\n      disableAnimation\n      hasArrow={false}\n      hideOnClickOutside\n      id={id}\n      maxWidth={maxWidth ?? refSelect.current?.offsetWidth}\n      onClose={() => setIsDropdownVisible(false)}\n      placement=\"bottom\"\n      portalTarget={portalTarget}\n      ref={ref}\n      role=\"dialog\"\n      tabIndex={-1}\n      text={\n        <Stack>\n          {searchable && !isLoading && numberOfOptions >= 6 ? (\n            <SearchBarDropdown\n              displayedOptions={displayedOptions}\n              placeholder={placeholder}\n              setSearchBarActive={setSearchBarActive}\n            />\n          ) : null}\n          <CreateDropdown\n            defaultSearchValue={defaultSearchValue}\n            descriptionDirection={descriptionDirection}\n            emptyState={emptyState}\n            isEmpty={isEmpty}\n            isLoading={isLoading}\n            loadMore={loadMore}\n            optionalInfoPlacement={optionalInfoPlacement}\n          />\n          {computedFooter}\n        </Stack>\n      }\n      visible={isDropdownVisible}\n    >\n      {children}\n    </StyledPopup>\n  )\n}\n"]} */",
|
|
155
155
|
toString: _EMOTION_STRINGIFIED_CSS_ERROR__
|
|
156
156
|
});
|
|
157
157
|
const EmptyState = /* @__PURE__ */ _styled(Stack, process.env.NODE_ENV === "production" ? {
|
|
@@ -161,7 +161,7 @@ const EmptyState = /* @__PURE__ */ _styled(Stack, process.env.NODE_ENV === "prod
|
|
|
161
161
|
label: "EmptyState"
|
|
162
162
|
})("padding:", ({
|
|
163
163
|
theme
|
|
164
|
-
}) => theme.space[2], ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInput/Dropdown.tsx"],"names":[],"mappings":"AA4LgC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInput/Dropdown.tsx","sourcesContent":["'use client'\n\nimport { useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport type {\n  ComponentProps,\n  Dispatch,\n  KeyboardEvent,\n  ReactNode,\n  RefObject,\n  SetStateAction,\n} from 'react'\nimport {\n  useCallback,\n  useContext,\n  useEffect,\n  useMemo,\n  useRef,\n  useState,\n} from 'react'\nimport { Checkbox } from '../Checkbox'\nimport { ModalContext } from '../Modal/ModalProvider'\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'\nimport { INPUT_SIZE_HEIGHT } from './types'\n\nconst DROPDOWN_MAX_HEIGHT = 256\n\nexport type DropdownProps = {\n  id?: string\n  children: ReactNode\n  emptyState: ReactNode\n  descriptionDirection: 'row' | 'column'\n  searchable: boolean\n  placeholder: string\n  footer?: ((closeDropdown: () => void) => ReactNode) | ReactNode\n  refSelect: RefObject<HTMLDivElement | null>\n  loadMore?: ReactNode\n  optionalInfoPlacement: 'left' | 'right'\n  isLoading?: boolean\n  size: 'small' | 'medium' | 'large'\n  dropdownAlign?: ComponentProps<typeof Popup>['align']\n  portalTarget?: ComponentProps<typeof Popup>['portalTarget']\n}\n\nexport type CreateDropdownProps = {\n  isEmpty: boolean\n  emptyState: ReactNode\n  descriptionDirection: 'row' | 'column'\n  loadMore?: ReactNode\n  optionalInfoPlacement: 'left' | 'right'\n  defaultSearchValue: string | null\n  isLoading?: boolean\n}\n\nconst NON_SEARCHABLE_KEYS = [\n  'Tab',\n  ' ',\n  'Enter',\n  'CapsLock',\n  'Shift',\n  'ArrowDown',\n  'ArrowUp',\n  'ArrowLeft',\n  'ArrowRight',\n  'Escape',\n]\n\nconst StyledPopup = styled(Popup)`\n  width: 100%;\n  min-width: 320px;\n  background-color: ${({ theme }) =>\n    theme.colors.other.elevation.background.raised};\n  color: ${({ theme }) => theme.colors.neutral.text};\n  box-shadow: ${({ theme }) =>\n    `${theme.shadows.raised[0]}, ${theme.shadows.raised[1]}`};\n  padding: ${({ theme }) => theme.space[0]};\n  margin-bottom: ${({ theme }) => theme.space[10]};\n`\n\nconst DropdownContainer = styled(Stack)<{ 'data-grouped': boolean }>`\n  max-height: ${DROPDOWN_MAX_HEIGHT}px;\n  overflow-y: auto;\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: 0;\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: 0;\n`\nconst DropdownItem = styled.div<{\n  'aria-selected': boolean\n  'aria-disabled': boolean\n}>`\n  text-align:left;\n  border: none;\n  background-color: ${({ theme }) =>\n    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  border: 1px transparent solid;\n\n  &:hover, :focus {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n    color: ${({ theme }) => theme.colors.primary.text};\n    cursor: pointer;\n  }\n\n  &[aria-selected='true'] {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n  }\n\n  &[aria-disabled=\"true\"] {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n  }\n\n  &[aria-disabled=\"true\"]:hover, [aria-disabled=\"true\"]:focus {\n    background-color: ${({ theme }) =>\n      theme.colors.neutral.backgroundStrongDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textStrongDisabled};\n    cursor: not-allowed;\n    outline: none;\n  }\n`\n\nconst PopupFooter = styled.div`\n  width: 100%;\n  padding: ${({ theme }) => theme.space[1.5]} ${({ theme }) => theme.space[2]}\n    ${({ theme }) => theme.space[1.5]} ${({ theme }) => theme.space[2]};\n  box-shadow: ${({ theme }) => theme.shadows.dropdown};\n`\nconst StyledCheckbox = styled(Checkbox)`\n  width: 100%;\n  position: static;\n  text-align: left;\n  align-items: center;\n  pointer-events: none;\n` // pointer-events: none prevents any error when using the checkbox in a form since it is an unnamed input\n\nconst EmptyState = styled(Stack)`\n  padding: ${({ theme }) => theme.space[2]};\n`\nconst LoadMore = styled(Stack)`\n  padding: ${({ theme }) => theme.space[0.5]};\n`\n\nconst moveFocusDown = () => {\n  const options = document.querySelectorAll(\n    '#items > div[role=\"option\"]:not([disabled])',\n  )\n  const activeItem = document.activeElement\n  if (options) {\n    for (let i = 0; i < options?.length; i += 1) {\n      const listLength = options.length\n      if (activeItem === options[i] && activeItem !== options[listLength - 1]) {\n        ;(options[i + 1] as HTMLElement).focus()\n      }\n    }\n  }\n}\nconst moveFocusUp = () => {\n  const options = document.querySelectorAll(\n    '#items > div[role=\"option\"]:not([disabled])',\n  )\n  const activeItem = document.activeElement\n\n  if (options) {\n    for (let i = 0; i < options.length; i += 1) {\n      if (activeItem === options[i] && activeItem !== options[0]) {\n        ;(options[i - 1] as HTMLElement).focus()\n      }\n    }\n  }\n}\nconst handleKeyDownSelect = (event: KeyboardEvent<HTMLDivElement>) => {\n  if (event.key === 'ArrowDown') {\n    event.preventDefault()\n    moveFocusDown()\n  }\n\n  if (event.key === 'ArrowUp') {\n    event.preventDefault()\n    moveFocusUp()\n  }\n\n  if (event.key === ' ') {\n    // No scroll\n    event.preventDefault()\n  }\n}\n\nconst handleKeyDown = (\n  event: globalThis.KeyboardEvent,\n  ref: RefObject<HTMLDivElement | null>,\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].find(option =>\n        option.searchText\n          ? option.searchText.toLocaleLowerCase().startsWith(currentSearch)\n          : option.value.toLocaleLowerCase().startsWith(currentSearch),\n      )\n      if (closestOption) {\n        setDefaultSearch(closestOption.searchText ?? closestOption.value)\n      } else {\n        setDefaultSearch(null)\n      }\n    }\n  }\n}\nconst CreateDropdown = ({\n  isEmpty,\n  emptyState,\n  descriptionDirection,\n  loadMore,\n  optionalInfoPlacement,\n  defaultSearchValue,\n  isLoading,\n}: CreateDropdownProps) => {\n  const {\n    setIsDropdownVisible,\n    onChange,\n    options,\n    multiselect,\n    selectAll,\n    selectAllGroup,\n    displayedOptions,\n    setSelectedData,\n    selectedData,\n  } = useSelectInput()\n  const focusedItemRef = useRef<HTMLDivElement>(null)\n\n  useEffect(() => {\n    if (defaultSearchValue && focusedItemRef?.current) {\n      focusedItemRef.current.focus()\n    }\n  }, [defaultSearchValue])\n\n  if (isEmpty) {\n    return (\n      <EmptyState alignItems=\"center\" gap={2}>\n        {emptyState ?? (\n          <Text as=\"p\" variant=\"bodyStrong\">\n            No options\n          </Text>\n        )}\n      </EmptyState>\n    )\n  }\n\n  const handleClick = (clickedOption: OptionType, group?: string) => {\n    setSelectedData({ clickedOption, group, type: 'selectOption' })\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({ selectedGroup: group, type: 'selectGroup' })\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      data-grouped\n      id=\"select-dropdown\"\n      onKeyDown={handleKeyDownSelect}\n      role=\"listbox\"\n    >\n      {isLoading ? (\n        <Skeleton variant=\"block\" />\n      ) : (\n        <>\n          {selectAll && multiselect ? (\n            <Stack id=\"items\">\n              <DropdownItem\n                aria-disabled={false}\n                aria-label=\"select-all\"\n                aria-selected={selectedData.allSelected}\n                data-testid=\"select-all\"\n                id=\"select-all\"\n                onClick={selectAllOptions}\n                onKeyDown={event =>\n                  [' ', 'Enter'].includes(event.key) ? selectAllOptions() : null\n                }\n                role=\"option\"\n                tabIndex={0}\n              >\n                <StyledCheckbox\n                  checked={selectedData.allSelected}\n                  data-testid=\"select-all-checkbox\"\n                  disabled={false}\n                  onChange={selectAllOptions}\n                  tabIndex={-1}\n                  value=\"select-all\"\n                >\n                  <Stack direction=\"column\">\n                    <Text as=\"span\" placement=\"left\" variant=\"body\">\n                      {selectAll.label}\n                    </Text>\n                    <Text\n                      as=\"span\"\n                      placement=\"left\"\n                      prominence=\"weak\"\n                      sentiment=\"neutral\"\n                      variant=\"bodySmall\"\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 gap={0.25} key={group}>\n              {displayedOptions[group].length > 0 ? (\n                <DropdownGroupWrapper id={selectAllGroup ? 'items' : undefined}>\n                  {group ? (\n                    <DropdownGroup\n                      data-selectgroup={selectAllGroup}\n                      data-testid={`group-${index}`}\n                      key={group}\n                      onClick={() =>\n                        selectAllGroup ? handleSelectGroup(group) : null\n                      }\n                      onKeyDown={event => {\n                        if ([' ', 'Enter'].includes(event.key)) {\n                          event.preventDefault()\n                          handleSelectGroup(group)\n                        }\n                      }}\n                      role=\"group\"\n                      tabIndex={selectAllGroup ? 0 : -1}\n                      type=\"button\"\n                    >\n                      {selectAllGroup ? (\n                        <StyledCheckbox\n                          checked={selectedData.selectedGroups.includes(group)}\n                          data-testid=\"select-group\"\n                          disabled={false}\n                          onChange={() =>\n                            selectAllGroup ? handleSelectGroup(group) : null\n                          }\n                          tabIndex={-1}\n                          value={group}\n                        >\n                          <Text\n                            as=\"span\"\n                            placement=\"left\"\n                            sentiment=\"neutral\"\n                            variant=\"caption\"\n                          >\n                            {group.toUpperCase()}\n                          </Text>\n                        </StyledCheckbox>\n                      ) : (\n                        <Text\n                          as=\"span\"\n                          placement=\"left\"\n                          sentiment=\"neutral\"\n                          variant=\"caption\"\n                        >\n                          {group.toUpperCase()}\n                        </Text>\n                      )}\n                    </DropdownGroup>\n                  ) : null}\n                </DropdownGroupWrapper>\n              ) : null}\n              <Stack gap=\"0.25\" id=\"items\">\n                {displayedOptions[group].map((option, indexOption) => (\n                  <DropdownItem\n                    aria-disabled={!!option.disabled}\n                    aria-label={option.value}\n                    aria-selected={\n                      selectedData.selectedValues.includes(option.value) &&\n                      !option.disabled\n                    }\n                    data-testid={`option-${option.value}`}\n                    id={`option-${indexOption}`}\n                    key={option.value}\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                    role=\"option\"\n                    tabIndex={!option.disabled ? 0 : -1}\n                  >\n                    {multiselect ? (\n                      <StyledCheckbox\n                        checked={\n                          selectedData.selectedValues.includes(option.value) &&\n                          !option.disabled\n                        }\n                        disabled={option.disabled}\n                        onChange={() => {\n                          if (!option.disabled) {\n                            handleClick(option, group)\n                          }\n                        }}\n                        tabIndex={-1}\n                        value={option.value}\n                      >\n                        <DisplayOption\n                          descriptionDirection={descriptionDirection}\n                          option={option}\n                          optionalInfoPlacement={optionalInfoPlacement}\n                        />\n                      </StyledCheckbox>\n                    ) : (\n                      <DisplayOption\n                        descriptionDirection={descriptionDirection}\n                        option={option}\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      data-grouped={false}\n      gap={0.25}\n      id=\"select-dropdown\"\n      onKeyDown={handleKeyDownSelect}\n      role=\"listbox\"\n    >\n      {selectAll && multiselect ? (\n        <Stack gap={0.25} id=\"items\" tabIndex={-1}>\n          <DropdownItem\n            aria-disabled={false}\n            aria-label=\"select-all\"\n            aria-selected={selectedData.allSelected}\n            data-testid=\"select-all\"\n            onClick={selectAllOptions}\n            onKeyDown={event =>\n              [' ', 'Enter'].includes(event.key) ? selectAllOptions() : null\n            }\n            role=\"option\"\n            tabIndex={0}\n          >\n            <StyledCheckbox\n              checked={selectedData.allSelected}\n              data-testid=\"select-all-checkbox\"\n              disabled={false}\n              onChange={selectAllOptions}\n              tabIndex={-1}\n              value=\"select-all\"\n            >\n              <Stack direction=\"column\">\n                <Text as=\"span\" placement=\"left\" variant=\"body\">\n                  {selectAll.label}\n                </Text>\n                <Text\n                  as=\"span\"\n                  placement=\"left\"\n                  prominence=\"weak\"\n                  sentiment=\"neutral\"\n                  variant=\"bodySmall\"\n                >\n                  {selectAll.description}\n                </Text>\n              </Stack>\n            </StyledCheckbox>\n          </DropdownItem>\n        </Stack>\n      ) : null}\n      <Stack gap={0.25} id=\"items\">\n        {isLoading ? (\n          <Skeleton variant=\"block\" />\n        ) : (\n          displayedOptions.map((option, index) => (\n            <DropdownItem\n              aria-disabled={!!option.disabled}\n              aria-label={option.value}\n              aria-selected={\n                selectedData.selectedValues.includes(option.value) &&\n                !option.disabled\n              }\n              data-testid={`option-${option.value}`}\n              id={`option-${index}`}\n              key={option.value}\n              onClick={() => {\n                if (!option.disabled) {\n                  handleClick(option)\n                }\n              }}\n              onKeyDown={event =>\n                [' ', 'Enter'].includes(event.key) ? handleClick(option) : null\n              }\n              ref={\n                option.value === defaultSearchValue ||\n                option.searchText === defaultSearchValue\n                  ? focusedItemRef\n                  : null\n              }\n              role=\"option\"\n              tabIndex={!option.disabled ? 0 : -1}\n            >\n              {multiselect ? (\n                <StyledCheckbox\n                  checked={\n                    selectedData.selectedValues.includes(option.value) &&\n                    !option.disabled\n                  }\n                  disabled={option.disabled}\n                  onChange={() => {\n                    if (!option.disabled) {\n                      handleClick(option)\n                    }\n                  }}\n                  tabIndex={-1}\n                  value={option.value}\n                >\n                  <DisplayOption\n                    descriptionDirection={descriptionDirection}\n                    option={option}\n                    optionalInfoPlacement={optionalInfoPlacement}\n                  />\n                </StyledCheckbox>\n              ) : (\n                <DisplayOption\n                  descriptionDirection={descriptionDirection}\n                  option={option}\n                  optionalInfoPlacement={optionalInfoPlacement}\n                />\n              )}\n            </DropdownItem>\n          ))\n        )}\n        {loadMore ? <LoadMore>{loadMore}</LoadMore> : null}\n      </Stack>\n    </DropdownContainer>\n  )\n}\nexport const Dropdown = ({\n  children,\n  emptyState,\n  descriptionDirection,\n  searchable,\n  placeholder,\n  footer,\n  refSelect,\n  loadMore,\n  optionalInfoPlacement,\n  isLoading,\n  size,\n  dropdownAlign,\n  portalTarget,\n  id,\n}: DropdownProps) => {\n  const {\n    setIsDropdownVisible,\n    isDropdownVisible,\n    onSearch,\n    searchInput,\n    options,\n    displayedOptions,\n    numberOfOptions,\n  } = useSelectInput()\n  const theme = useTheme()\n  const [searchBarActive, setSearchBarActive] = useState(false)\n  const [defaultSearchValue, setDefaultSearch] = useState<string | null>(null)\n  const ref = useRef<HTMLDivElement>(null)\n  const [search, setSearch] = useState<string>('')\n  const [maxWidth, setWidth] = useState<string | number>()\n  const modalContext = useContext(ModalContext)\n\n  useEffect(() => {\n    if (refSelect.current && isDropdownVisible) {\n      const position =\n        refSelect.current.getBoundingClientRect().bottom +\n        DROPDOWN_MAX_HEIGHT +\n        Number(theme.sizing[INPUT_SIZE_HEIGHT[size]].replace('rem', '')) * 16 +\n        Number.parseInt(theme.space['5'], 10)\n      const overflow = position - window.innerHeight + 32\n      if (overflow > 0 && modalContext) {\n        const currentModal = modalContext.openedModals[0]\n        const modalElement = currentModal?.ref.current\n\n        if (modalElement) {\n          const parentElement = modalElement.parentNode as HTMLElement\n          if (parentElement) {\n            parentElement.scrollBy({\n              behavior: 'smooth',\n              top: overflow,\n            })\n          }\n        } else {\n          window.scrollBy({ behavior: 'smooth', top: overflow })\n        }\n      }\n    }\n    // oxlint-disable-next-line react/exhaustive-deps\n  }, [isDropdownVisible, refSelect, size, ref.current])\n\n  const resizeDropdown = useCallback(() => {\n    if (\n      refSelect.current &&\n      refSelect.current.getBoundingClientRect().width > 0\n    ) {\n      setWidth(refSelect.current.getBoundingClientRect().width)\n    }\n  }, [refSelect])\n\n  useEffect(() => {\n    resizeDropdown()\n\n    window.addEventListener('resize', resizeDropdown)\n\n    return () => window.removeEventListener('resize', resizeDropdown)\n  }, [resizeDropdown])\n\n  useEffect(() => {\n    if (!searchInput) {\n      onSearch(options)\n    }\n  }, [onSearch, options, searchInput])\n\n  useEffect(() => {\n    if (!isDropdownVisible) {\n      setDefaultSearch(null)\n      setSearch('')\n    }\n\n    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      if (!searchable) {\n        document.removeEventListener('keydown', event =>\n          handleKeyDown(\n            event,\n            ref,\n            options,\n            searchBarActive,\n            setSearch,\n            setDefaultSearch,\n            search,\n          ),\n        )\n      }\n    }\n  }, [\n    isDropdownVisible,\n    searchBarActive,\n    options,\n    onSearch,\n    search,\n    refSelect,\n    setDefaultSearch,\n    setIsDropdownVisible,\n    searchable,\n  ])\n\n  // No data is displayed (because of the search or because no data is provided)\n  // Set to true when noData by default\n  const isEmpty = useMemo(() => {\n    if (numberOfOptions === 0) {\n      return true\n    }\n    if (Array.isArray(displayedOptions)) {\n      return displayedOptions.length === 0\n    }\n    const groups = Object.keys(displayedOptions)\n    for (const group of groups) {\n      if (displayedOptions[group].length > 0) {\n        return false\n      }\n    }\n\n    return true\n  }, [displayedOptions, numberOfOptions])\n\n  const computedFooter = useMemo(() => {\n    if (footer && !isEmpty) {\n      if (typeof footer === 'function') {\n        return (\n          <PopupFooter>{footer(() => setIsDropdownVisible(false))}</PopupFooter>\n        )\n      }\n\n      return <PopupFooter>{footer}</PopupFooter>\n    }\n\n    return null\n  }, [isEmpty, footer, setIsDropdownVisible])\n\n  return (\n    <StyledPopup\n      align={dropdownAlign ?? 'start'}\n      containerFullWidth\n      debounceDelay={0}\n      disableAnimation\n      hasArrow={false}\n      hideOnClickOutside\n      id={id}\n      maxWidth={maxWidth ?? refSelect.current?.offsetWidth}\n      onClose={() => setIsDropdownVisible(false)}\n      placement=\"bottom\"\n      portalTarget={portalTarget}\n      ref={ref}\n      role=\"dialog\"\n      tabIndex={-1}\n      text={\n        <Stack>\n          {searchable && !isLoading && numberOfOptions >= 6 ? (\n            <SearchBarDropdown\n              displayedOptions={displayedOptions}\n              placeholder={placeholder}\n              setSearchBarActive={setSearchBarActive}\n            />\n          ) : null}\n          <CreateDropdown\n            defaultSearchValue={defaultSearchValue}\n            descriptionDirection={descriptionDirection}\n            emptyState={emptyState}\n            isEmpty={isEmpty}\n            isLoading={isLoading}\n            loadMore={loadMore}\n            optionalInfoPlacement={optionalInfoPlacement}\n          />\n          {computedFooter}\n        </Stack>\n      }\n      visible={isDropdownVisible}\n    >\n      {children}\n    </StyledPopup>\n  )\n}\n"]} */"));
|
|
164
|
+
}) => theme.space[2], ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInput/Dropdown.tsx"],"names":[],"mappings":"AA6LgC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInput/Dropdown.tsx","sourcesContent":["'use client'\n\nimport { useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport type {\n  ComponentProps,\n  Dispatch,\n  KeyboardEvent,\n  ReactNode,\n  RefObject,\n  SetStateAction,\n} from 'react'\nimport {\n  useCallback,\n  useContext,\n  useEffect,\n  useMemo,\n  useRef,\n  useState,\n} from 'react'\nimport { Checkbox } from '../Checkbox'\nimport { ModalContext } from '../Modal/ModalProvider'\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'\nimport { INPUT_SIZE_HEIGHT } from './types'\n\nconst DROPDOWN_MAX_HEIGHT = 256\n\nexport type DropdownProps = {\n  id?: string\n  children: ReactNode\n  emptyState: ReactNode\n  descriptionDirection: 'row' | 'column'\n  searchable: boolean\n  placeholder: string\n  footer?: ((closeDropdown: () => void) => ReactNode) | ReactNode\n  refSelect: RefObject<HTMLDivElement | null>\n  loadMore?: ReactNode\n  optionalInfoPlacement: 'left' | 'right'\n  isLoading?: boolean\n  size: 'small' | 'medium' | 'large'\n  dropdownAlign?: ComponentProps<typeof Popup>['align']\n  portalTarget?: ComponentProps<typeof Popup>['portalTarget']\n}\n\nexport type CreateDropdownProps = {\n  isEmpty: boolean\n  emptyState: ReactNode\n  descriptionDirection: 'row' | 'column'\n  loadMore?: ReactNode\n  optionalInfoPlacement: 'left' | 'right'\n  defaultSearchValue: string | null\n  isLoading?: boolean\n}\n\nconst NON_SEARCHABLE_KEYS = [\n  'Tab',\n  ' ',\n  'Enter',\n  'CapsLock',\n  'Shift',\n  'ArrowDown',\n  'ArrowUp',\n  'ArrowLeft',\n  'ArrowRight',\n  'Escape',\n]\n\nconst StyledPopup = styled(Popup)`\n  width: 100%;\n  min-width: 320px;\n  background-color: ${({ theme }) =>\n    theme.colors.other.elevation.background.raised};\n  color: ${({ theme }) => theme.colors.neutral.text};\n  box-shadow: ${({ theme }) =>\n    `${theme.shadows.raised[0]}, ${theme.shadows.raised[1]}`};\n  padding: ${({ theme }) => theme.space[0]};\n  margin-bottom: ${({ theme }) => theme.space[10]};\n`\n\nconst DropdownContainer = styled(Stack)<{ 'data-grouped': boolean }>`\n  max-height: ${DROPDOWN_MAX_HEIGHT}px;\n  overflow-y: auto;\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: 0;\n  padding-right: ${({ theme }) => theme.space[2]};\n  padding-left: ${({ theme }) => theme.space[2]};\n  height: ${({ theme }) => theme.space[4]};\n  display: flex;\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: 0;\n`\nconst DropdownItem = styled.div<{\n  'aria-selected': boolean\n  'aria-disabled': boolean\n}>`\n  text-align:left;\n  border: none;\n  background-color: ${({ theme }) =>\n    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  border: 1px transparent solid;\n\n  &:hover, :focus {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n    color: ${({ theme }) => theme.colors.primary.text};\n    cursor: pointer;\n  }\n\n  &[aria-selected='true'] {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n  }\n\n  &[aria-disabled=\"true\"] {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n  }\n\n  &[aria-disabled=\"true\"]:hover, [aria-disabled=\"true\"]:focus {\n    background-color: ${({ theme }) =>\n      theme.colors.neutral.backgroundStrongDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textStrongDisabled};\n    cursor: not-allowed;\n    outline: none;\n  }\n`\n\nconst PopupFooter = styled.div`\n  width: 100%;\n  padding: ${({ theme }) => theme.space[1.5]} ${({ theme }) => theme.space[2]}\n    ${({ theme }) => theme.space[1.5]} ${({ theme }) => theme.space[2]};\n  box-shadow: ${({ theme }) => theme.shadows.dropdown};\n`\nconst StyledCheckbox = styled(Checkbox)`\n  width: 100%;\n  position: static;\n  text-align: left;\n  align-items: center;\n  pointer-events: none;\n` // pointer-events: none prevents any error when using the checkbox in a form since it is an unnamed input\n\nconst EmptyState = styled(Stack)`\n  padding: ${({ theme }) => theme.space[2]};\n`\nconst LoadMore = styled(Stack)`\n  padding: ${({ theme }) => theme.space[0.5]};\n`\n\nconst moveFocusDown = () => {\n  const options = document.querySelectorAll(\n    '#items > div[role=\"option\"]:not([disabled])',\n  )\n  const activeItem = document.activeElement\n  if (options) {\n    for (let i = 0; i < options?.length; i += 1) {\n      const listLength = options.length\n      if (activeItem === options[i] && activeItem !== options[listLength - 1]) {\n        ;(options[i + 1] as HTMLElement).focus()\n      }\n    }\n  }\n}\nconst moveFocusUp = () => {\n  const options = document.querySelectorAll(\n    '#items > div[role=\"option\"]:not([disabled])',\n  )\n  const activeItem = document.activeElement\n\n  if (options) {\n    for (let i = 0; i < options.length; i += 1) {\n      if (activeItem === options[i] && activeItem !== options[0]) {\n        ;(options[i - 1] as HTMLElement).focus()\n      }\n    }\n  }\n}\nconst handleKeyDownSelect = (event: KeyboardEvent<HTMLDivElement>) => {\n  if (event.key === 'ArrowDown') {\n    event.preventDefault()\n    moveFocusDown()\n  }\n\n  if (event.key === 'ArrowUp') {\n    event.preventDefault()\n    moveFocusUp()\n  }\n\n  if (event.key === ' ') {\n    // No scroll\n    event.preventDefault()\n  }\n}\n\nconst handleKeyDown = (\n  event: globalThis.KeyboardEvent,\n  ref: RefObject<HTMLDivElement | null>,\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].find(option =>\n        option.searchText\n          ? option.searchText.toLocaleLowerCase().startsWith(currentSearch)\n          : option.value.toLocaleLowerCase().startsWith(currentSearch),\n      )\n      if (closestOption) {\n        setDefaultSearch(closestOption.searchText ?? closestOption.value)\n      } else {\n        setDefaultSearch(null)\n      }\n    }\n  }\n}\nconst CreateDropdown = ({\n  isEmpty,\n  emptyState,\n  descriptionDirection,\n  loadMore,\n  optionalInfoPlacement,\n  defaultSearchValue,\n  isLoading,\n}: CreateDropdownProps) => {\n  const {\n    setIsDropdownVisible,\n    onChange,\n    options,\n    multiselect,\n    selectAll,\n    selectAllGroup,\n    displayedOptions,\n    setSelectedData,\n    selectedData,\n  } = useSelectInput()\n  const focusedItemRef = useRef<HTMLDivElement>(null)\n\n  useEffect(() => {\n    if (defaultSearchValue && focusedItemRef?.current) {\n      focusedItemRef.current.focus()\n    }\n  }, [defaultSearchValue])\n\n  if (isEmpty) {\n    return (\n      <EmptyState alignItems=\"center\" gap={2}>\n        {emptyState ?? (\n          <Text as=\"p\" variant=\"bodyStrong\">\n            No options\n          </Text>\n        )}\n      </EmptyState>\n    )\n  }\n\n  const handleClick = (clickedOption: OptionType, group?: string) => {\n    setSelectedData({ clickedOption, group, type: 'selectOption' })\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({ selectedGroup: group, type: 'selectGroup' })\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      data-grouped\n      id=\"select-dropdown\"\n      onKeyDown={handleKeyDownSelect}\n      role=\"listbox\"\n    >\n      {isLoading ? (\n        <Skeleton variant=\"block\" />\n      ) : (\n        <>\n          {selectAll && multiselect ? (\n            <Stack id=\"items\">\n              <DropdownItem\n                aria-disabled={false}\n                aria-label=\"select-all\"\n                aria-selected={selectedData.allSelected}\n                data-testid=\"select-all\"\n                id=\"select-all\"\n                onClick={selectAllOptions}\n                onKeyDown={event =>\n                  [' ', 'Enter'].includes(event.key) ? selectAllOptions() : null\n                }\n                role=\"option\"\n                tabIndex={0}\n              >\n                <StyledCheckbox\n                  checked={selectedData.allSelected}\n                  data-testid=\"select-all-checkbox\"\n                  disabled={false}\n                  onChange={selectAllOptions}\n                  tabIndex={-1}\n                  value=\"select-all\"\n                >\n                  <Stack direction=\"column\">\n                    <Text as=\"span\" placement=\"left\" variant=\"body\">\n                      {selectAll.label}\n                    </Text>\n                    <Text\n                      as=\"span\"\n                      placement=\"left\"\n                      prominence=\"weak\"\n                      sentiment=\"neutral\"\n                      variant=\"bodySmall\"\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 gap={0.25} key={group}>\n              {displayedOptions[group].length > 0 ? (\n                <DropdownGroupWrapper id={selectAllGroup ? 'items' : undefined}>\n                  {group ? (\n                    <DropdownGroup\n                      data-selectgroup={selectAllGroup}\n                      data-testid={`group-${index}`}\n                      key={group}\n                      onClick={() =>\n                        selectAllGroup ? handleSelectGroup(group) : null\n                      }\n                      onKeyDown={event => {\n                        if ([' ', 'Enter'].includes(event.key)) {\n                          event.preventDefault()\n                          handleSelectGroup(group)\n                        }\n                      }}\n                      role=\"group\"\n                      tabIndex={selectAllGroup ? 0 : -1}\n                      type=\"button\"\n                    >\n                      {selectAllGroup ? (\n                        <StyledCheckbox\n                          checked={selectedData.selectedGroups.includes(group)}\n                          data-testid=\"select-group\"\n                          disabled={false}\n                          onChange={() =>\n                            selectAllGroup ? handleSelectGroup(group) : null\n                          }\n                          tabIndex={-1}\n                          value={group}\n                        >\n                          <Text\n                            as=\"span\"\n                            placement=\"left\"\n                            sentiment=\"neutral\"\n                            variant=\"caption\"\n                          >\n                            {group.toUpperCase()}\n                          </Text>\n                        </StyledCheckbox>\n                      ) : (\n                        <Text\n                          as=\"span\"\n                          placement=\"left\"\n                          sentiment=\"neutral\"\n                          variant=\"caption\"\n                        >\n                          {group.toUpperCase()}\n                        </Text>\n                      )}\n                    </DropdownGroup>\n                  ) : null}\n                </DropdownGroupWrapper>\n              ) : null}\n              <Stack gap=\"0.25\" id=\"items\">\n                {displayedOptions[group].map((option, indexOption) => (\n                  <DropdownItem\n                    aria-disabled={!!option.disabled}\n                    aria-label={option.value}\n                    aria-selected={\n                      selectedData.selectedValues.includes(option.value) &&\n                      !option.disabled\n                    }\n                    data-testid={`option-${option.value}`}\n                    id={`option-${indexOption}`}\n                    key={option.value}\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                    role=\"option\"\n                    tabIndex={!option.disabled ? 0 : -1}\n                  >\n                    {multiselect ? (\n                      <StyledCheckbox\n                        checked={\n                          selectedData.selectedValues.includes(option.value) &&\n                          !option.disabled\n                        }\n                        disabled={option.disabled}\n                        onChange={() => {\n                          if (!option.disabled) {\n                            handleClick(option, group)\n                          }\n                        }}\n                        tabIndex={-1}\n                        value={option.value}\n                      >\n                        <DisplayOption\n                          descriptionDirection={descriptionDirection}\n                          option={option}\n                          optionalInfoPlacement={optionalInfoPlacement}\n                        />\n                      </StyledCheckbox>\n                    ) : (\n                      <DisplayOption\n                        descriptionDirection={descriptionDirection}\n                        option={option}\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      data-grouped={false}\n      gap={0.25}\n      id=\"select-dropdown\"\n      onKeyDown={handleKeyDownSelect}\n      role=\"listbox\"\n    >\n      {selectAll && multiselect ? (\n        <Stack gap={0.25} id=\"items\" tabIndex={-1}>\n          <DropdownItem\n            aria-disabled={false}\n            aria-label=\"select-all\"\n            aria-selected={selectedData.allSelected}\n            data-testid=\"select-all\"\n            onClick={selectAllOptions}\n            onKeyDown={event =>\n              [' ', 'Enter'].includes(event.key) ? selectAllOptions() : null\n            }\n            role=\"option\"\n            tabIndex={0}\n          >\n            <StyledCheckbox\n              checked={selectedData.allSelected}\n              data-testid=\"select-all-checkbox\"\n              disabled={false}\n              onChange={selectAllOptions}\n              tabIndex={-1}\n              value=\"select-all\"\n            >\n              <Stack direction=\"column\">\n                <Text as=\"span\" placement=\"left\" variant=\"body\">\n                  {selectAll.label}\n                </Text>\n                <Text\n                  as=\"span\"\n                  placement=\"left\"\n                  prominence=\"weak\"\n                  sentiment=\"neutral\"\n                  variant=\"bodySmall\"\n                >\n                  {selectAll.description}\n                </Text>\n              </Stack>\n            </StyledCheckbox>\n          </DropdownItem>\n        </Stack>\n      ) : null}\n      <Stack gap={0.25} id=\"items\">\n        {isLoading ? (\n          <Skeleton variant=\"block\" />\n        ) : (\n          displayedOptions.map((option, index) => (\n            <DropdownItem\n              aria-disabled={!!option.disabled}\n              aria-label={option.value}\n              aria-selected={\n                selectedData.selectedValues.includes(option.value) &&\n                !option.disabled\n              }\n              data-testid={`option-${option.value}`}\n              id={`option-${index}`}\n              key={option.value}\n              onClick={() => {\n                if (!option.disabled) {\n                  handleClick(option)\n                }\n              }}\n              onKeyDown={event =>\n                [' ', 'Enter'].includes(event.key) ? handleClick(option) : null\n              }\n              ref={\n                option.value === defaultSearchValue ||\n                option.searchText === defaultSearchValue\n                  ? focusedItemRef\n                  : null\n              }\n              role=\"option\"\n              tabIndex={!option.disabled ? 0 : -1}\n            >\n              {multiselect ? (\n                <StyledCheckbox\n                  checked={\n                    selectedData.selectedValues.includes(option.value) &&\n                    !option.disabled\n                  }\n                  disabled={option.disabled}\n                  onChange={() => {\n                    if (!option.disabled) {\n                      handleClick(option)\n                    }\n                  }}\n                  tabIndex={-1}\n                  value={option.value}\n                >\n                  <DisplayOption\n                    descriptionDirection={descriptionDirection}\n                    option={option}\n                    optionalInfoPlacement={optionalInfoPlacement}\n                  />\n                </StyledCheckbox>\n              ) : (\n                <DisplayOption\n                  descriptionDirection={descriptionDirection}\n                  option={option}\n                  optionalInfoPlacement={optionalInfoPlacement}\n                />\n              )}\n            </DropdownItem>\n          ))\n        )}\n        {loadMore ? <LoadMore>{loadMore}</LoadMore> : null}\n      </Stack>\n    </DropdownContainer>\n  )\n}\nexport const Dropdown = ({\n  children,\n  emptyState,\n  descriptionDirection,\n  searchable,\n  placeholder,\n  footer,\n  refSelect,\n  loadMore,\n  optionalInfoPlacement,\n  isLoading,\n  size,\n  dropdownAlign,\n  portalTarget,\n  id,\n}: DropdownProps) => {\n  const {\n    setIsDropdownVisible,\n    isDropdownVisible,\n    onSearch,\n    searchInput,\n    options,\n    displayedOptions,\n    numberOfOptions,\n  } = useSelectInput()\n  const theme = useTheme()\n  const [searchBarActive, setSearchBarActive] = useState(false)\n  const [defaultSearchValue, setDefaultSearch] = useState<string | null>(null)\n  const ref = useRef<HTMLDivElement>(null)\n  const [search, setSearch] = useState<string>('')\n  const [maxWidth, setWidth] = useState<string | number>()\n  const modalContext = useContext(ModalContext)\n\n  useEffect(() => {\n    if (refSelect.current && isDropdownVisible) {\n      const position =\n        refSelect.current.getBoundingClientRect().bottom +\n        DROPDOWN_MAX_HEIGHT +\n        Number(theme.sizing[INPUT_SIZE_HEIGHT[size]].replace('rem', '')) * 16 +\n        Number.parseInt(theme.space['5'], 10)\n      const overflow = position - window.innerHeight + 32\n      if (overflow > 0 && modalContext) {\n        const currentModal = modalContext.openedModals[0]\n        const modalElement = currentModal?.ref.current\n\n        if (modalElement) {\n          const parentElement = modalElement.parentNode as HTMLElement\n          if (parentElement) {\n            parentElement.scrollBy({\n              behavior: 'smooth',\n              top: overflow,\n            })\n          }\n        } else {\n          window.scrollBy({ behavior: 'smooth', top: overflow })\n        }\n      }\n    }\n    // oxlint-disable-next-line react/exhaustive-deps\n  }, [isDropdownVisible, refSelect, size, ref.current])\n\n  const resizeDropdown = useCallback(() => {\n    if (\n      refSelect.current &&\n      refSelect.current.getBoundingClientRect().width > 0\n    ) {\n      setWidth(refSelect.current.getBoundingClientRect().width)\n    }\n  }, [refSelect])\n\n  useEffect(() => {\n    resizeDropdown()\n\n    window.addEventListener('resize', resizeDropdown)\n\n    return () => window.removeEventListener('resize', resizeDropdown)\n  }, [resizeDropdown])\n\n  useEffect(() => {\n    if (!searchInput) {\n      onSearch(options)\n    }\n  }, [onSearch, options, searchInput])\n\n  useEffect(() => {\n    if (!isDropdownVisible) {\n      setDefaultSearch(null)\n      setSearch('')\n    }\n\n    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      if (!searchable) {\n        document.removeEventListener('keydown', event =>\n          handleKeyDown(\n            event,\n            ref,\n            options,\n            searchBarActive,\n            setSearch,\n            setDefaultSearch,\n            search,\n          ),\n        )\n      }\n    }\n  }, [\n    isDropdownVisible,\n    searchBarActive,\n    options,\n    onSearch,\n    search,\n    refSelect,\n    setDefaultSearch,\n    setIsDropdownVisible,\n    searchable,\n  ])\n\n  // No data is displayed (because of the search or because no data is provided)\n  // Set to true when noData by default\n  const isEmpty = useMemo(() => {\n    if (numberOfOptions === 0) {\n      return true\n    }\n    if (Array.isArray(displayedOptions)) {\n      return displayedOptions.length === 0\n    }\n    const groups = Object.keys(displayedOptions)\n    for (const group of groups) {\n      if (displayedOptions[group].length > 0) {\n        return false\n      }\n    }\n\n    return true\n  }, [displayedOptions, numberOfOptions])\n\n  const computedFooter = useMemo(() => {\n    if (footer && !isEmpty) {\n      if (typeof footer === 'function') {\n        return (\n          <PopupFooter>{footer(() => setIsDropdownVisible(false))}</PopupFooter>\n        )\n      }\n\n      return <PopupFooter>{footer}</PopupFooter>\n    }\n\n    return null\n  }, [isEmpty, footer, setIsDropdownVisible])\n\n  return (\n    <StyledPopup\n      align={dropdownAlign ?? 'start'}\n      containerFullWidth\n      debounceDelay={0}\n      disableAnimation\n      hasArrow={false}\n      hideOnClickOutside\n      id={id}\n      maxWidth={maxWidth ?? refSelect.current?.offsetWidth}\n      onClose={() => setIsDropdownVisible(false)}\n      placement=\"bottom\"\n      portalTarget={portalTarget}\n      ref={ref}\n      role=\"dialog\"\n      tabIndex={-1}\n      text={\n        <Stack>\n          {searchable && !isLoading && numberOfOptions >= 6 ? (\n            <SearchBarDropdown\n              displayedOptions={displayedOptions}\n              placeholder={placeholder}\n              setSearchBarActive={setSearchBarActive}\n            />\n          ) : null}\n          <CreateDropdown\n            defaultSearchValue={defaultSearchValue}\n            descriptionDirection={descriptionDirection}\n            emptyState={emptyState}\n            isEmpty={isEmpty}\n            isLoading={isLoading}\n            loadMore={loadMore}\n            optionalInfoPlacement={optionalInfoPlacement}\n          />\n          {computedFooter}\n        </Stack>\n      }\n      visible={isDropdownVisible}\n    >\n      {children}\n    </StyledPopup>\n  )\n}\n"]} */"));
|
|
165
165
|
const LoadMore = /* @__PURE__ */ _styled(Stack, process.env.NODE_ENV === "production" ? {
|
|
166
166
|
target: "e12qyldb0"
|
|
167
167
|
} : {
|
|
@@ -169,7 +169,7 @@ const LoadMore = /* @__PURE__ */ _styled(Stack, process.env.NODE_ENV === "produc
|
|
|
169
169
|
label: "LoadMore"
|
|
170
170
|
})("padding:", ({
|
|
171
171
|
theme
|
|
172
|
-
}) => theme.space[0.5], ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInput/Dropdown.tsx"],"names":[],"mappings":"AA+L8B","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInput/Dropdown.tsx","sourcesContent":["'use client'\n\nimport { useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport type {\n  ComponentProps,\n  Dispatch,\n  KeyboardEvent,\n  ReactNode,\n  RefObject,\n  SetStateAction,\n} from 'react'\nimport {\n  useCallback,\n  useContext,\n  useEffect,\n  useMemo,\n  useRef,\n  useState,\n} from 'react'\nimport { Checkbox } from '../Checkbox'\nimport { ModalContext } from '../Modal/ModalProvider'\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'\nimport { INPUT_SIZE_HEIGHT } from './types'\n\nconst DROPDOWN_MAX_HEIGHT = 256\n\nexport type DropdownProps = {\n  id?: string\n  children: ReactNode\n  emptyState: ReactNode\n  descriptionDirection: 'row' | 'column'\n  searchable: boolean\n  placeholder: string\n  footer?: ((closeDropdown: () => void) => ReactNode) | ReactNode\n  refSelect: RefObject<HTMLDivElement | null>\n  loadMore?: ReactNode\n  optionalInfoPlacement: 'left' | 'right'\n  isLoading?: boolean\n  size: 'small' | 'medium' | 'large'\n  dropdownAlign?: ComponentProps<typeof Popup>['align']\n  portalTarget?: ComponentProps<typeof Popup>['portalTarget']\n}\n\nexport type CreateDropdownProps = {\n  isEmpty: boolean\n  emptyState: ReactNode\n  descriptionDirection: 'row' | 'column'\n  loadMore?: ReactNode\n  optionalInfoPlacement: 'left' | 'right'\n  defaultSearchValue: string | null\n  isLoading?: boolean\n}\n\nconst NON_SEARCHABLE_KEYS = [\n  'Tab',\n  ' ',\n  'Enter',\n  'CapsLock',\n  'Shift',\n  'ArrowDown',\n  'ArrowUp',\n  'ArrowLeft',\n  'ArrowRight',\n  'Escape',\n]\n\nconst StyledPopup = styled(Popup)`\n  width: 100%;\n  min-width: 320px;\n  background-color: ${({ theme }) =>\n    theme.colors.other.elevation.background.raised};\n  color: ${({ theme }) => theme.colors.neutral.text};\n  box-shadow: ${({ theme }) =>\n    `${theme.shadows.raised[0]}, ${theme.shadows.raised[1]}`};\n  padding: ${({ theme }) => theme.space[0]};\n  margin-bottom: ${({ theme }) => theme.space[10]};\n`\n\nconst DropdownContainer = styled(Stack)<{ 'data-grouped': boolean }>`\n  max-height: ${DROPDOWN_MAX_HEIGHT}px;\n  overflow-y: auto;\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: 0;\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: 0;\n`\nconst DropdownItem = styled.div<{\n  'aria-selected': boolean\n  'aria-disabled': boolean\n}>`\n  text-align:left;\n  border: none;\n  background-color: ${({ theme }) =>\n    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  border: 1px transparent solid;\n\n  &:hover, :focus {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n    color: ${({ theme }) => theme.colors.primary.text};\n    cursor: pointer;\n  }\n\n  &[aria-selected='true'] {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n  }\n\n  &[aria-disabled=\"true\"] {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n  }\n\n  &[aria-disabled=\"true\"]:hover, [aria-disabled=\"true\"]:focus {\n    background-color: ${({ theme }) =>\n      theme.colors.neutral.backgroundStrongDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textStrongDisabled};\n    cursor: not-allowed;\n    outline: none;\n  }\n`\n\nconst PopupFooter = styled.div`\n  width: 100%;\n  padding: ${({ theme }) => theme.space[1.5]} ${({ theme }) => theme.space[2]}\n    ${({ theme }) => theme.space[1.5]} ${({ theme }) => theme.space[2]};\n  box-shadow: ${({ theme }) => theme.shadows.dropdown};\n`\nconst StyledCheckbox = styled(Checkbox)`\n  width: 100%;\n  position: static;\n  text-align: left;\n  align-items: center;\n  pointer-events: none;\n` // pointer-events: none prevents any error when using the checkbox in a form since it is an unnamed input\n\nconst EmptyState = styled(Stack)`\n  padding: ${({ theme }) => theme.space[2]};\n`\nconst LoadMore = styled(Stack)`\n  padding: ${({ theme }) => theme.space[0.5]};\n`\n\nconst moveFocusDown = () => {\n  const options = document.querySelectorAll(\n    '#items > div[role=\"option\"]:not([disabled])',\n  )\n  const activeItem = document.activeElement\n  if (options) {\n    for (let i = 0; i < options?.length; i += 1) {\n      const listLength = options.length\n      if (activeItem === options[i] && activeItem !== options[listLength - 1]) {\n        ;(options[i + 1] as HTMLElement).focus()\n      }\n    }\n  }\n}\nconst moveFocusUp = () => {\n  const options = document.querySelectorAll(\n    '#items > div[role=\"option\"]:not([disabled])',\n  )\n  const activeItem = document.activeElement\n\n  if (options) {\n    for (let i = 0; i < options.length; i += 1) {\n      if (activeItem === options[i] && activeItem !== options[0]) {\n        ;(options[i - 1] as HTMLElement).focus()\n      }\n    }\n  }\n}\nconst handleKeyDownSelect = (event: KeyboardEvent<HTMLDivElement>) => {\n  if (event.key === 'ArrowDown') {\n    event.preventDefault()\n    moveFocusDown()\n  }\n\n  if (event.key === 'ArrowUp') {\n    event.preventDefault()\n    moveFocusUp()\n  }\n\n  if (event.key === ' ') {\n    // No scroll\n    event.preventDefault()\n  }\n}\n\nconst handleKeyDown = (\n  event: globalThis.KeyboardEvent,\n  ref: RefObject<HTMLDivElement | null>,\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].find(option =>\n        option.searchText\n          ? option.searchText.toLocaleLowerCase().startsWith(currentSearch)\n          : option.value.toLocaleLowerCase().startsWith(currentSearch),\n      )\n      if (closestOption) {\n        setDefaultSearch(closestOption.searchText ?? closestOption.value)\n      } else {\n        setDefaultSearch(null)\n      }\n    }\n  }\n}\nconst CreateDropdown = ({\n  isEmpty,\n  emptyState,\n  descriptionDirection,\n  loadMore,\n  optionalInfoPlacement,\n  defaultSearchValue,\n  isLoading,\n}: CreateDropdownProps) => {\n  const {\n    setIsDropdownVisible,\n    onChange,\n    options,\n    multiselect,\n    selectAll,\n    selectAllGroup,\n    displayedOptions,\n    setSelectedData,\n    selectedData,\n  } = useSelectInput()\n  const focusedItemRef = useRef<HTMLDivElement>(null)\n\n  useEffect(() => {\n    if (defaultSearchValue && focusedItemRef?.current) {\n      focusedItemRef.current.focus()\n    }\n  }, [defaultSearchValue])\n\n  if (isEmpty) {\n    return (\n      <EmptyState alignItems=\"center\" gap={2}>\n        {emptyState ?? (\n          <Text as=\"p\" variant=\"bodyStrong\">\n            No options\n          </Text>\n        )}\n      </EmptyState>\n    )\n  }\n\n  const handleClick = (clickedOption: OptionType, group?: string) => {\n    setSelectedData({ clickedOption, group, type: 'selectOption' })\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({ selectedGroup: group, type: 'selectGroup' })\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      data-grouped\n      id=\"select-dropdown\"\n      onKeyDown={handleKeyDownSelect}\n      role=\"listbox\"\n    >\n      {isLoading ? (\n        <Skeleton variant=\"block\" />\n      ) : (\n        <>\n          {selectAll && multiselect ? (\n            <Stack id=\"items\">\n              <DropdownItem\n                aria-disabled={false}\n                aria-label=\"select-all\"\n                aria-selected={selectedData.allSelected}\n                data-testid=\"select-all\"\n                id=\"select-all\"\n                onClick={selectAllOptions}\n                onKeyDown={event =>\n                  [' ', 'Enter'].includes(event.key) ? selectAllOptions() : null\n                }\n                role=\"option\"\n                tabIndex={0}\n              >\n                <StyledCheckbox\n                  checked={selectedData.allSelected}\n                  data-testid=\"select-all-checkbox\"\n                  disabled={false}\n                  onChange={selectAllOptions}\n                  tabIndex={-1}\n                  value=\"select-all\"\n                >\n                  <Stack direction=\"column\">\n                    <Text as=\"span\" placement=\"left\" variant=\"body\">\n                      {selectAll.label}\n                    </Text>\n                    <Text\n                      as=\"span\"\n                      placement=\"left\"\n                      prominence=\"weak\"\n                      sentiment=\"neutral\"\n                      variant=\"bodySmall\"\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 gap={0.25} key={group}>\n              {displayedOptions[group].length > 0 ? (\n                <DropdownGroupWrapper id={selectAllGroup ? 'items' : undefined}>\n                  {group ? (\n                    <DropdownGroup\n                      data-selectgroup={selectAllGroup}\n                      data-testid={`group-${index}`}\n                      key={group}\n                      onClick={() =>\n                        selectAllGroup ? handleSelectGroup(group) : null\n                      }\n                      onKeyDown={event => {\n                        if ([' ', 'Enter'].includes(event.key)) {\n                          event.preventDefault()\n                          handleSelectGroup(group)\n                        }\n                      }}\n                      role=\"group\"\n                      tabIndex={selectAllGroup ? 0 : -1}\n                      type=\"button\"\n                    >\n                      {selectAllGroup ? (\n                        <StyledCheckbox\n                          checked={selectedData.selectedGroups.includes(group)}\n                          data-testid=\"select-group\"\n                          disabled={false}\n                          onChange={() =>\n                            selectAllGroup ? handleSelectGroup(group) : null\n                          }\n                          tabIndex={-1}\n                          value={group}\n                        >\n                          <Text\n                            as=\"span\"\n                            placement=\"left\"\n                            sentiment=\"neutral\"\n                            variant=\"caption\"\n                          >\n                            {group.toUpperCase()}\n                          </Text>\n                        </StyledCheckbox>\n                      ) : (\n                        <Text\n                          as=\"span\"\n                          placement=\"left\"\n                          sentiment=\"neutral\"\n                          variant=\"caption\"\n                        >\n                          {group.toUpperCase()}\n                        </Text>\n                      )}\n                    </DropdownGroup>\n                  ) : null}\n                </DropdownGroupWrapper>\n              ) : null}\n              <Stack gap=\"0.25\" id=\"items\">\n                {displayedOptions[group].map((option, indexOption) => (\n                  <DropdownItem\n                    aria-disabled={!!option.disabled}\n                    aria-label={option.value}\n                    aria-selected={\n                      selectedData.selectedValues.includes(option.value) &&\n                      !option.disabled\n                    }\n                    data-testid={`option-${option.value}`}\n                    id={`option-${indexOption}`}\n                    key={option.value}\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                    role=\"option\"\n                    tabIndex={!option.disabled ? 0 : -1}\n                  >\n                    {multiselect ? (\n                      <StyledCheckbox\n                        checked={\n                          selectedData.selectedValues.includes(option.value) &&\n                          !option.disabled\n                        }\n                        disabled={option.disabled}\n                        onChange={() => {\n                          if (!option.disabled) {\n                            handleClick(option, group)\n                          }\n                        }}\n                        tabIndex={-1}\n                        value={option.value}\n                      >\n                        <DisplayOption\n                          descriptionDirection={descriptionDirection}\n                          option={option}\n                          optionalInfoPlacement={optionalInfoPlacement}\n                        />\n                      </StyledCheckbox>\n                    ) : (\n                      <DisplayOption\n                        descriptionDirection={descriptionDirection}\n                        option={option}\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      data-grouped={false}\n      gap={0.25}\n      id=\"select-dropdown\"\n      onKeyDown={handleKeyDownSelect}\n      role=\"listbox\"\n    >\n      {selectAll && multiselect ? (\n        <Stack gap={0.25} id=\"items\" tabIndex={-1}>\n          <DropdownItem\n            aria-disabled={false}\n            aria-label=\"select-all\"\n            aria-selected={selectedData.allSelected}\n            data-testid=\"select-all\"\n            onClick={selectAllOptions}\n            onKeyDown={event =>\n              [' ', 'Enter'].includes(event.key) ? selectAllOptions() : null\n            }\n            role=\"option\"\n            tabIndex={0}\n          >\n            <StyledCheckbox\n              checked={selectedData.allSelected}\n              data-testid=\"select-all-checkbox\"\n              disabled={false}\n              onChange={selectAllOptions}\n              tabIndex={-1}\n              value=\"select-all\"\n            >\n              <Stack direction=\"column\">\n                <Text as=\"span\" placement=\"left\" variant=\"body\">\n                  {selectAll.label}\n                </Text>\n                <Text\n                  as=\"span\"\n                  placement=\"left\"\n                  prominence=\"weak\"\n                  sentiment=\"neutral\"\n                  variant=\"bodySmall\"\n                >\n                  {selectAll.description}\n                </Text>\n              </Stack>\n            </StyledCheckbox>\n          </DropdownItem>\n        </Stack>\n      ) : null}\n      <Stack gap={0.25} id=\"items\">\n        {isLoading ? (\n          <Skeleton variant=\"block\" />\n        ) : (\n          displayedOptions.map((option, index) => (\n            <DropdownItem\n              aria-disabled={!!option.disabled}\n              aria-label={option.value}\n              aria-selected={\n                selectedData.selectedValues.includes(option.value) &&\n                !option.disabled\n              }\n              data-testid={`option-${option.value}`}\n              id={`option-${index}`}\n              key={option.value}\n              onClick={() => {\n                if (!option.disabled) {\n                  handleClick(option)\n                }\n              }}\n              onKeyDown={event =>\n                [' ', 'Enter'].includes(event.key) ? handleClick(option) : null\n              }\n              ref={\n                option.value === defaultSearchValue ||\n                option.searchText === defaultSearchValue\n                  ? focusedItemRef\n                  : null\n              }\n              role=\"option\"\n              tabIndex={!option.disabled ? 0 : -1}\n            >\n              {multiselect ? (\n                <StyledCheckbox\n                  checked={\n                    selectedData.selectedValues.includes(option.value) &&\n                    !option.disabled\n                  }\n                  disabled={option.disabled}\n                  onChange={() => {\n                    if (!option.disabled) {\n                      handleClick(option)\n                    }\n                  }}\n                  tabIndex={-1}\n                  value={option.value}\n                >\n                  <DisplayOption\n                    descriptionDirection={descriptionDirection}\n                    option={option}\n                    optionalInfoPlacement={optionalInfoPlacement}\n                  />\n                </StyledCheckbox>\n              ) : (\n                <DisplayOption\n                  descriptionDirection={descriptionDirection}\n                  option={option}\n                  optionalInfoPlacement={optionalInfoPlacement}\n                />\n              )}\n            </DropdownItem>\n          ))\n        )}\n        {loadMore ? <LoadMore>{loadMore}</LoadMore> : null}\n      </Stack>\n    </DropdownContainer>\n  )\n}\nexport const Dropdown = ({\n  children,\n  emptyState,\n  descriptionDirection,\n  searchable,\n  placeholder,\n  footer,\n  refSelect,\n  loadMore,\n  optionalInfoPlacement,\n  isLoading,\n  size,\n  dropdownAlign,\n  portalTarget,\n  id,\n}: DropdownProps) => {\n  const {\n    setIsDropdownVisible,\n    isDropdownVisible,\n    onSearch,\n    searchInput,\n    options,\n    displayedOptions,\n    numberOfOptions,\n  } = useSelectInput()\n  const theme = useTheme()\n  const [searchBarActive, setSearchBarActive] = useState(false)\n  const [defaultSearchValue, setDefaultSearch] = useState<string | null>(null)\n  const ref = useRef<HTMLDivElement>(null)\n  const [search, setSearch] = useState<string>('')\n  const [maxWidth, setWidth] = useState<string | number>()\n  const modalContext = useContext(ModalContext)\n\n  useEffect(() => {\n    if (refSelect.current && isDropdownVisible) {\n      const position =\n        refSelect.current.getBoundingClientRect().bottom +\n        DROPDOWN_MAX_HEIGHT +\n        Number(theme.sizing[INPUT_SIZE_HEIGHT[size]].replace('rem', '')) * 16 +\n        Number.parseInt(theme.space['5'], 10)\n      const overflow = position - window.innerHeight + 32\n      if (overflow > 0 && modalContext) {\n        const currentModal = modalContext.openedModals[0]\n        const modalElement = currentModal?.ref.current\n\n        if (modalElement) {\n          const parentElement = modalElement.parentNode as HTMLElement\n          if (parentElement) {\n            parentElement.scrollBy({\n              behavior: 'smooth',\n              top: overflow,\n            })\n          }\n        } else {\n          window.scrollBy({ behavior: 'smooth', top: overflow })\n        }\n      }\n    }\n    // oxlint-disable-next-line react/exhaustive-deps\n  }, [isDropdownVisible, refSelect, size, ref.current])\n\n  const resizeDropdown = useCallback(() => {\n    if (\n      refSelect.current &&\n      refSelect.current.getBoundingClientRect().width > 0\n    ) {\n      setWidth(refSelect.current.getBoundingClientRect().width)\n    }\n  }, [refSelect])\n\n  useEffect(() => {\n    resizeDropdown()\n\n    window.addEventListener('resize', resizeDropdown)\n\n    return () => window.removeEventListener('resize', resizeDropdown)\n  }, [resizeDropdown])\n\n  useEffect(() => {\n    if (!searchInput) {\n      onSearch(options)\n    }\n  }, [onSearch, options, searchInput])\n\n  useEffect(() => {\n    if (!isDropdownVisible) {\n      setDefaultSearch(null)\n      setSearch('')\n    }\n\n    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      if (!searchable) {\n        document.removeEventListener('keydown', event =>\n          handleKeyDown(\n            event,\n            ref,\n            options,\n            searchBarActive,\n            setSearch,\n            setDefaultSearch,\n            search,\n          ),\n        )\n      }\n    }\n  }, [\n    isDropdownVisible,\n    searchBarActive,\n    options,\n    onSearch,\n    search,\n    refSelect,\n    setDefaultSearch,\n    setIsDropdownVisible,\n    searchable,\n  ])\n\n  // No data is displayed (because of the search or because no data is provided)\n  // Set to true when noData by default\n  const isEmpty = useMemo(() => {\n    if (numberOfOptions === 0) {\n      return true\n    }\n    if (Array.isArray(displayedOptions)) {\n      return displayedOptions.length === 0\n    }\n    const groups = Object.keys(displayedOptions)\n    for (const group of groups) {\n      if (displayedOptions[group].length > 0) {\n        return false\n      }\n    }\n\n    return true\n  }, [displayedOptions, numberOfOptions])\n\n  const computedFooter = useMemo(() => {\n    if (footer && !isEmpty) {\n      if (typeof footer === 'function') {\n        return (\n          <PopupFooter>{footer(() => setIsDropdownVisible(false))}</PopupFooter>\n        )\n      }\n\n      return <PopupFooter>{footer}</PopupFooter>\n    }\n\n    return null\n  }, [isEmpty, footer, setIsDropdownVisible])\n\n  return (\n    <StyledPopup\n      align={dropdownAlign ?? 'start'}\n      containerFullWidth\n      debounceDelay={0}\n      disableAnimation\n      hasArrow={false}\n      hideOnClickOutside\n      id={id}\n      maxWidth={maxWidth ?? refSelect.current?.offsetWidth}\n      onClose={() => setIsDropdownVisible(false)}\n      placement=\"bottom\"\n      portalTarget={portalTarget}\n      ref={ref}\n      role=\"dialog\"\n      tabIndex={-1}\n      text={\n        <Stack>\n          {searchable && !isLoading && numberOfOptions >= 6 ? (\n            <SearchBarDropdown\n              displayedOptions={displayedOptions}\n              placeholder={placeholder}\n              setSearchBarActive={setSearchBarActive}\n            />\n          ) : null}\n          <CreateDropdown\n            defaultSearchValue={defaultSearchValue}\n            descriptionDirection={descriptionDirection}\n            emptyState={emptyState}\n            isEmpty={isEmpty}\n            isLoading={isLoading}\n            loadMore={loadMore}\n            optionalInfoPlacement={optionalInfoPlacement}\n          />\n          {computedFooter}\n        </Stack>\n      }\n      visible={isDropdownVisible}\n    >\n      {children}\n    </StyledPopup>\n  )\n}\n"]} */"));
|
|
172
|
+
}) => theme.space[0.5], ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInput/Dropdown.tsx"],"names":[],"mappings":"AAgM8B","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInput/Dropdown.tsx","sourcesContent":["'use client'\n\nimport { useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport type {\n  ComponentProps,\n  Dispatch,\n  KeyboardEvent,\n  ReactNode,\n  RefObject,\n  SetStateAction,\n} from 'react'\nimport {\n  useCallback,\n  useContext,\n  useEffect,\n  useMemo,\n  useRef,\n  useState,\n} from 'react'\nimport { Checkbox } from '../Checkbox'\nimport { ModalContext } from '../Modal/ModalProvider'\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'\nimport { INPUT_SIZE_HEIGHT } from './types'\n\nconst DROPDOWN_MAX_HEIGHT = 256\n\nexport type DropdownProps = {\n  id?: string\n  children: ReactNode\n  emptyState: ReactNode\n  descriptionDirection: 'row' | 'column'\n  searchable: boolean\n  placeholder: string\n  footer?: ((closeDropdown: () => void) => ReactNode) | ReactNode\n  refSelect: RefObject<HTMLDivElement | null>\n  loadMore?: ReactNode\n  optionalInfoPlacement: 'left' | 'right'\n  isLoading?: boolean\n  size: 'small' | 'medium' | 'large'\n  dropdownAlign?: ComponentProps<typeof Popup>['align']\n  portalTarget?: ComponentProps<typeof Popup>['portalTarget']\n}\n\nexport type CreateDropdownProps = {\n  isEmpty: boolean\n  emptyState: ReactNode\n  descriptionDirection: 'row' | 'column'\n  loadMore?: ReactNode\n  optionalInfoPlacement: 'left' | 'right'\n  defaultSearchValue: string | null\n  isLoading?: boolean\n}\n\nconst NON_SEARCHABLE_KEYS = [\n  'Tab',\n  ' ',\n  'Enter',\n  'CapsLock',\n  'Shift',\n  'ArrowDown',\n  'ArrowUp',\n  'ArrowLeft',\n  'ArrowRight',\n  'Escape',\n]\n\nconst StyledPopup = styled(Popup)`\n  width: 100%;\n  min-width: 320px;\n  background-color: ${({ theme }) =>\n    theme.colors.other.elevation.background.raised};\n  color: ${({ theme }) => theme.colors.neutral.text};\n  box-shadow: ${({ theme }) =>\n    `${theme.shadows.raised[0]}, ${theme.shadows.raised[1]}`};\n  padding: ${({ theme }) => theme.space[0]};\n  margin-bottom: ${({ theme }) => theme.space[10]};\n`\n\nconst DropdownContainer = styled(Stack)<{ 'data-grouped': boolean }>`\n  max-height: ${DROPDOWN_MAX_HEIGHT}px;\n  overflow-y: auto;\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: 0;\n  padding-right: ${({ theme }) => theme.space[2]};\n  padding-left: ${({ theme }) => theme.space[2]};\n  height: ${({ theme }) => theme.space[4]};\n  display: flex;\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: 0;\n`\nconst DropdownItem = styled.div<{\n  'aria-selected': boolean\n  'aria-disabled': boolean\n}>`\n  text-align:left;\n  border: none;\n  background-color: ${({ theme }) =>\n    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  border: 1px transparent solid;\n\n  &:hover, :focus {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n    color: ${({ theme }) => theme.colors.primary.text};\n    cursor: pointer;\n  }\n\n  &[aria-selected='true'] {\n    background-color: ${({ theme }) => theme.colors.primary.background};\n  }\n\n  &[aria-disabled=\"true\"] {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n  }\n\n  &[aria-disabled=\"true\"]:hover, [aria-disabled=\"true\"]:focus {\n    background-color: ${({ theme }) =>\n      theme.colors.neutral.backgroundStrongDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textStrongDisabled};\n    cursor: not-allowed;\n    outline: none;\n  }\n`\n\nconst PopupFooter = styled.div`\n  width: 100%;\n  padding: ${({ theme }) => theme.space[1.5]} ${({ theme }) => theme.space[2]}\n    ${({ theme }) => theme.space[1.5]} ${({ theme }) => theme.space[2]};\n  box-shadow: ${({ theme }) => theme.shadows.dropdown};\n`\nconst StyledCheckbox = styled(Checkbox)`\n  width: 100%;\n  position: static;\n  text-align: left;\n  align-items: center;\n  pointer-events: none;\n` // pointer-events: none prevents any error when using the checkbox in a form since it is an unnamed input\n\nconst EmptyState = styled(Stack)`\n  padding: ${({ theme }) => theme.space[2]};\n`\nconst LoadMore = styled(Stack)`\n  padding: ${({ theme }) => theme.space[0.5]};\n`\n\nconst moveFocusDown = () => {\n  const options = document.querySelectorAll(\n    '#items > div[role=\"option\"]:not([disabled])',\n  )\n  const activeItem = document.activeElement\n  if (options) {\n    for (let i = 0; i < options?.length; i += 1) {\n      const listLength = options.length\n      if (activeItem === options[i] && activeItem !== options[listLength - 1]) {\n        ;(options[i + 1] as HTMLElement).focus()\n      }\n    }\n  }\n}\nconst moveFocusUp = () => {\n  const options = document.querySelectorAll(\n    '#items > div[role=\"option\"]:not([disabled])',\n  )\n  const activeItem = document.activeElement\n\n  if (options) {\n    for (let i = 0; i < options.length; i += 1) {\n      if (activeItem === options[i] && activeItem !== options[0]) {\n        ;(options[i - 1] as HTMLElement).focus()\n      }\n    }\n  }\n}\nconst handleKeyDownSelect = (event: KeyboardEvent<HTMLDivElement>) => {\n  if (event.key === 'ArrowDown') {\n    event.preventDefault()\n    moveFocusDown()\n  }\n\n  if (event.key === 'ArrowUp') {\n    event.preventDefault()\n    moveFocusUp()\n  }\n\n  if (event.key === ' ') {\n    // No scroll\n    event.preventDefault()\n  }\n}\n\nconst handleKeyDown = (\n  event: globalThis.KeyboardEvent,\n  ref: RefObject<HTMLDivElement | null>,\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].find(option =>\n        option.searchText\n          ? option.searchText.toLocaleLowerCase().startsWith(currentSearch)\n          : option.value.toLocaleLowerCase().startsWith(currentSearch),\n      )\n      if (closestOption) {\n        setDefaultSearch(closestOption.searchText ?? closestOption.value)\n      } else {\n        setDefaultSearch(null)\n      }\n    }\n  }\n}\nconst CreateDropdown = ({\n  isEmpty,\n  emptyState,\n  descriptionDirection,\n  loadMore,\n  optionalInfoPlacement,\n  defaultSearchValue,\n  isLoading,\n}: CreateDropdownProps) => {\n  const {\n    setIsDropdownVisible,\n    onChange,\n    options,\n    multiselect,\n    selectAll,\n    selectAllGroup,\n    displayedOptions,\n    setSelectedData,\n    selectedData,\n  } = useSelectInput()\n  const focusedItemRef = useRef<HTMLDivElement>(null)\n\n  useEffect(() => {\n    if (defaultSearchValue && focusedItemRef?.current) {\n      focusedItemRef.current.focus()\n    }\n  }, [defaultSearchValue])\n\n  if (isEmpty) {\n    return (\n      <EmptyState alignItems=\"center\" gap={2}>\n        {emptyState ?? (\n          <Text as=\"p\" variant=\"bodyStrong\">\n            No options\n          </Text>\n        )}\n      </EmptyState>\n    )\n  }\n\n  const handleClick = (clickedOption: OptionType, group?: string) => {\n    setSelectedData({ clickedOption, group, type: 'selectOption' })\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({ selectedGroup: group, type: 'selectGroup' })\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      data-grouped\n      id=\"select-dropdown\"\n      onKeyDown={handleKeyDownSelect}\n      role=\"listbox\"\n    >\n      {isLoading ? (\n        <Skeleton variant=\"block\" />\n      ) : (\n        <>\n          {selectAll && multiselect ? (\n            <Stack id=\"items\">\n              <DropdownItem\n                aria-disabled={false}\n                aria-label=\"select-all\"\n                aria-selected={selectedData.allSelected}\n                data-testid=\"select-all\"\n                id=\"select-all\"\n                onClick={selectAllOptions}\n                onKeyDown={event =>\n                  [' ', 'Enter'].includes(event.key) ? selectAllOptions() : null\n                }\n                role=\"option\"\n                tabIndex={0}\n              >\n                <StyledCheckbox\n                  checked={selectedData.allSelected}\n                  data-testid=\"select-all-checkbox\"\n                  disabled={false}\n                  onChange={selectAllOptions}\n                  tabIndex={-1}\n                  value=\"select-all\"\n                >\n                  <Stack direction=\"column\">\n                    <Text as=\"span\" placement=\"left\" variant=\"body\">\n                      {selectAll.label}\n                    </Text>\n                    <Text\n                      as=\"span\"\n                      placement=\"left\"\n                      prominence=\"weak\"\n                      sentiment=\"neutral\"\n                      variant=\"bodySmall\"\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 gap={0.25} key={group}>\n              {displayedOptions[group].length > 0 ? (\n                <DropdownGroupWrapper id={selectAllGroup ? 'items' : undefined}>\n                  {group ? (\n                    <DropdownGroup\n                      data-selectgroup={selectAllGroup}\n                      data-testid={`group-${index}`}\n                      key={group}\n                      onClick={() =>\n                        selectAllGroup ? handleSelectGroup(group) : null\n                      }\n                      onKeyDown={event => {\n                        if ([' ', 'Enter'].includes(event.key)) {\n                          event.preventDefault()\n                          handleSelectGroup(group)\n                        }\n                      }}\n                      role=\"group\"\n                      tabIndex={selectAllGroup ? 0 : -1}\n                      type=\"button\"\n                    >\n                      {selectAllGroup ? (\n                        <StyledCheckbox\n                          checked={selectedData.selectedGroups.includes(group)}\n                          data-testid=\"select-group\"\n                          disabled={false}\n                          onChange={() =>\n                            selectAllGroup ? handleSelectGroup(group) : null\n                          }\n                          tabIndex={-1}\n                          value={group}\n                        >\n                          <Text\n                            as=\"span\"\n                            placement=\"left\"\n                            sentiment=\"neutral\"\n                            variant=\"caption\"\n                          >\n                            {group.toUpperCase()}\n                          </Text>\n                        </StyledCheckbox>\n                      ) : (\n                        <Text\n                          as=\"span\"\n                          placement=\"left\"\n                          sentiment=\"neutral\"\n                          variant=\"caption\"\n                        >\n                          {group.toUpperCase()}\n                        </Text>\n                      )}\n                    </DropdownGroup>\n                  ) : null}\n                </DropdownGroupWrapper>\n              ) : null}\n              <Stack gap=\"0.25\" id=\"items\">\n                {displayedOptions[group].map((option, indexOption) => (\n                  <DropdownItem\n                    aria-disabled={!!option.disabled}\n                    aria-label={option.value}\n                    aria-selected={\n                      selectedData.selectedValues.includes(option.value) &&\n                      !option.disabled\n                    }\n                    data-testid={`option-${option.value}`}\n                    id={`option-${indexOption}`}\n                    key={option.value}\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                    role=\"option\"\n                    tabIndex={!option.disabled ? 0 : -1}\n                  >\n                    {multiselect ? (\n                      <StyledCheckbox\n                        checked={\n                          selectedData.selectedValues.includes(option.value) &&\n                          !option.disabled\n                        }\n                        disabled={option.disabled}\n                        onChange={() => {\n                          if (!option.disabled) {\n                            handleClick(option, group)\n                          }\n                        }}\n                        tabIndex={-1}\n                        value={option.value}\n                      >\n                        <DisplayOption\n                          descriptionDirection={descriptionDirection}\n                          option={option}\n                          optionalInfoPlacement={optionalInfoPlacement}\n                        />\n                      </StyledCheckbox>\n                    ) : (\n                      <DisplayOption\n                        descriptionDirection={descriptionDirection}\n                        option={option}\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      data-grouped={false}\n      gap={0.25}\n      id=\"select-dropdown\"\n      onKeyDown={handleKeyDownSelect}\n      role=\"listbox\"\n    >\n      {selectAll && multiselect ? (\n        <Stack gap={0.25} id=\"items\" tabIndex={-1}>\n          <DropdownItem\n            aria-disabled={false}\n            aria-label=\"select-all\"\n            aria-selected={selectedData.allSelected}\n            data-testid=\"select-all\"\n            onClick={selectAllOptions}\n            onKeyDown={event =>\n              [' ', 'Enter'].includes(event.key) ? selectAllOptions() : null\n            }\n            role=\"option\"\n            tabIndex={0}\n          >\n            <StyledCheckbox\n              checked={selectedData.allSelected}\n              data-testid=\"select-all-checkbox\"\n              disabled={false}\n              onChange={selectAllOptions}\n              tabIndex={-1}\n              value=\"select-all\"\n            >\n              <Stack direction=\"column\">\n                <Text as=\"span\" placement=\"left\" variant=\"body\">\n                  {selectAll.label}\n                </Text>\n                <Text\n                  as=\"span\"\n                  placement=\"left\"\n                  prominence=\"weak\"\n                  sentiment=\"neutral\"\n                  variant=\"bodySmall\"\n                >\n                  {selectAll.description}\n                </Text>\n              </Stack>\n            </StyledCheckbox>\n          </DropdownItem>\n        </Stack>\n      ) : null}\n      <Stack gap={0.25} id=\"items\">\n        {isLoading ? (\n          <Skeleton variant=\"block\" />\n        ) : (\n          displayedOptions.map((option, index) => (\n            <DropdownItem\n              aria-disabled={!!option.disabled}\n              aria-label={option.value}\n              aria-selected={\n                selectedData.selectedValues.includes(option.value) &&\n                !option.disabled\n              }\n              data-testid={`option-${option.value}`}\n              id={`option-${index}`}\n              key={option.value}\n              onClick={() => {\n                if (!option.disabled) {\n                  handleClick(option)\n                }\n              }}\n              onKeyDown={event =>\n                [' ', 'Enter'].includes(event.key) ? handleClick(option) : null\n              }\n              ref={\n                option.value === defaultSearchValue ||\n                option.searchText === defaultSearchValue\n                  ? focusedItemRef\n                  : null\n              }\n              role=\"option\"\n              tabIndex={!option.disabled ? 0 : -1}\n            >\n              {multiselect ? (\n                <StyledCheckbox\n                  checked={\n                    selectedData.selectedValues.includes(option.value) &&\n                    !option.disabled\n                  }\n                  disabled={option.disabled}\n                  onChange={() => {\n                    if (!option.disabled) {\n                      handleClick(option)\n                    }\n                  }}\n                  tabIndex={-1}\n                  value={option.value}\n                >\n                  <DisplayOption\n                    descriptionDirection={descriptionDirection}\n                    option={option}\n                    optionalInfoPlacement={optionalInfoPlacement}\n                  />\n                </StyledCheckbox>\n              ) : (\n                <DisplayOption\n                  descriptionDirection={descriptionDirection}\n                  option={option}\n                  optionalInfoPlacement={optionalInfoPlacement}\n                />\n              )}\n            </DropdownItem>\n          ))\n        )}\n        {loadMore ? <LoadMore>{loadMore}</LoadMore> : null}\n      </Stack>\n    </DropdownContainer>\n  )\n}\nexport const Dropdown = ({\n  children,\n  emptyState,\n  descriptionDirection,\n  searchable,\n  placeholder,\n  footer,\n  refSelect,\n  loadMore,\n  optionalInfoPlacement,\n  isLoading,\n  size,\n  dropdownAlign,\n  portalTarget,\n  id,\n}: DropdownProps) => {\n  const {\n    setIsDropdownVisible,\n    isDropdownVisible,\n    onSearch,\n    searchInput,\n    options,\n    displayedOptions,\n    numberOfOptions,\n  } = useSelectInput()\n  const theme = useTheme()\n  const [searchBarActive, setSearchBarActive] = useState(false)\n  const [defaultSearchValue, setDefaultSearch] = useState<string | null>(null)\n  const ref = useRef<HTMLDivElement>(null)\n  const [search, setSearch] = useState<string>('')\n  const [maxWidth, setWidth] = useState<string | number>()\n  const modalContext = useContext(ModalContext)\n\n  useEffect(() => {\n    if (refSelect.current && isDropdownVisible) {\n      const position =\n        refSelect.current.getBoundingClientRect().bottom +\n        DROPDOWN_MAX_HEIGHT +\n        Number(theme.sizing[INPUT_SIZE_HEIGHT[size]].replace('rem', '')) * 16 +\n        Number.parseInt(theme.space['5'], 10)\n      const overflow = position - window.innerHeight + 32\n      if (overflow > 0 && modalContext) {\n        const currentModal = modalContext.openedModals[0]\n        const modalElement = currentModal?.ref.current\n\n        if (modalElement) {\n          const parentElement = modalElement.parentNode as HTMLElement\n          if (parentElement) {\n            parentElement.scrollBy({\n              behavior: 'smooth',\n              top: overflow,\n            })\n          }\n        } else {\n          window.scrollBy({ behavior: 'smooth', top: overflow })\n        }\n      }\n    }\n    // oxlint-disable-next-line react/exhaustive-deps\n  }, [isDropdownVisible, refSelect, size, ref.current])\n\n  const resizeDropdown = useCallback(() => {\n    if (\n      refSelect.current &&\n      refSelect.current.getBoundingClientRect().width > 0\n    ) {\n      setWidth(refSelect.current.getBoundingClientRect().width)\n    }\n  }, [refSelect])\n\n  useEffect(() => {\n    resizeDropdown()\n\n    window.addEventListener('resize', resizeDropdown)\n\n    return () => window.removeEventListener('resize', resizeDropdown)\n  }, [resizeDropdown])\n\n  useEffect(() => {\n    if (!searchInput) {\n      onSearch(options)\n    }\n  }, [onSearch, options, searchInput])\n\n  useEffect(() => {\n    if (!isDropdownVisible) {\n      setDefaultSearch(null)\n      setSearch('')\n    }\n\n    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      if (!searchable) {\n        document.removeEventListener('keydown', event =>\n          handleKeyDown(\n            event,\n            ref,\n            options,\n            searchBarActive,\n            setSearch,\n            setDefaultSearch,\n            search,\n          ),\n        )\n      }\n    }\n  }, [\n    isDropdownVisible,\n    searchBarActive,\n    options,\n    onSearch,\n    search,\n    refSelect,\n    setDefaultSearch,\n    setIsDropdownVisible,\n    searchable,\n  ])\n\n  // No data is displayed (because of the search or because no data is provided)\n  // Set to true when noData by default\n  const isEmpty = useMemo(() => {\n    if (numberOfOptions === 0) {\n      return true\n    }\n    if (Array.isArray(displayedOptions)) {\n      return displayedOptions.length === 0\n    }\n    const groups = Object.keys(displayedOptions)\n    for (const group of groups) {\n      if (displayedOptions[group].length > 0) {\n        return false\n      }\n    }\n\n    return true\n  }, [displayedOptions, numberOfOptions])\n\n  const computedFooter = useMemo(() => {\n    if (footer && !isEmpty) {\n      if (typeof footer === 'function') {\n        return (\n          <PopupFooter>{footer(() => setIsDropdownVisible(false))}</PopupFooter>\n        )\n      }\n\n      return <PopupFooter>{footer}</PopupFooter>\n    }\n\n    return null\n  }, [isEmpty, footer, setIsDropdownVisible])\n\n  return (\n    <StyledPopup\n      align={dropdownAlign ?? 'start'}\n      containerFullWidth\n      debounceDelay={0}\n      disableAnimation\n      hasArrow={false}\n      hideOnClickOutside\n      id={id}\n      maxWidth={maxWidth ?? refSelect.current?.offsetWidth}\n      onClose={() => setIsDropdownVisible(false)}\n      placement=\"bottom\"\n      portalTarget={portalTarget}\n      ref={ref}\n      role=\"dialog\"\n      tabIndex={-1}\n      text={\n        <Stack>\n          {searchable && !isLoading && numberOfOptions >= 6 ? (\n            <SearchBarDropdown\n              displayedOptions={displayedOptions}\n              placeholder={placeholder}\n              setSearchBarActive={setSearchBarActive}\n            />\n          ) : null}\n          <CreateDropdown\n            defaultSearchValue={defaultSearchValue}\n            descriptionDirection={descriptionDirection}\n            emptyState={emptyState}\n            isEmpty={isEmpty}\n            isLoading={isLoading}\n            loadMore={loadMore}\n            optionalInfoPlacement={optionalInfoPlacement}\n          />\n          {computedFooter}\n        </Stack>\n      }\n      visible={isDropdownVisible}\n    >\n      {children}\n    </StyledPopup>\n  )\n}\n"]} */"));
|
|
173
173
|
const moveFocusDown = () => {
|
|
174
174
|
const options = document.querySelectorAll('#items > div[role="option"]:not([disabled])');
|
|
175
175
|
const activeItem = document.activeElement;
|