@ultraviolet/ui 1.88.1 → 1.89.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (82) hide show
  1. package/dist/components/Banner/index.cjs +7 -7
  2. package/dist/components/Banner/index.js +7 -7
  3. package/dist/components/Checkbox/index.cjs +12 -9
  4. package/dist/components/Checkbox/index.d.ts +16 -0
  5. package/dist/components/Checkbox/index.js +13 -10
  6. package/dist/components/Chip/index.d.ts +1 -1
  7. package/dist/components/Dialog/{subComponents → components}/CancelButton.cjs +3 -2
  8. package/dist/components/Dialog/{subComponents → components}/CancelButton.d.ts +2 -1
  9. package/dist/components/Dialog/{subComponents → components}/CancelButton.js +3 -2
  10. package/dist/components/Dialog/index.cjs +7 -7
  11. package/dist/components/Dialog/index.d.ts +2 -1
  12. package/dist/components/Dialog/index.js +7 -7
  13. package/dist/components/Label/index.cjs +33 -7
  14. package/dist/components/Label/index.d.ts +9 -2
  15. package/dist/components/Label/index.js +31 -7
  16. package/dist/components/List/ListContext.cjs +71 -67
  17. package/dist/components/List/ListContext.d.ts +5 -4
  18. package/dist/components/List/ListContext.js +72 -68
  19. package/dist/components/List/Row.cjs +19 -31
  20. package/dist/components/List/Row.js +19 -31
  21. package/dist/components/List/index.cjs +19 -4
  22. package/dist/components/List/index.js +20 -5
  23. package/dist/components/Popup/index.cjs +21 -9
  24. package/dist/components/Popup/index.js +21 -9
  25. package/dist/components/Radio/index.cjs +18 -14
  26. package/dist/components/Radio/index.d.ts +13 -1
  27. package/dist/components/Radio/index.js +19 -15
  28. package/dist/components/SelectInputV2/Dropdown.cjs +10 -10
  29. package/dist/components/SelectInputV2/Dropdown.js +10 -10
  30. package/dist/components/SelectInputV2/SelectBar.cjs +11 -10
  31. package/dist/components/SelectInputV2/SelectBar.d.ts +25 -1
  32. package/dist/components/SelectInputV2/SelectBar.js +12 -11
  33. package/dist/components/SelectInputV2/index.cjs +3 -3
  34. package/dist/components/SelectInputV2/index.d.ts +1 -1
  35. package/dist/components/SelectInputV2/index.js +3 -3
  36. package/dist/components/SelectableCard/index.cjs +86 -41
  37. package/dist/components/SelectableCard/index.js +88 -43
  38. package/dist/components/SelectableCardOptionGroup/Provider.cjs +24 -0
  39. package/dist/components/SelectableCardOptionGroup/Provider.d.ts +15 -0
  40. package/dist/components/SelectableCardOptionGroup/Provider.js +24 -0
  41. package/dist/components/SelectableCardOptionGroup/components/Image.cjs +39 -0
  42. package/dist/components/SelectableCardOptionGroup/components/Image.d.ts +9 -0
  43. package/dist/components/SelectableCardOptionGroup/components/Image.js +37 -0
  44. package/dist/components/SelectableCardOptionGroup/components/Option.cjs +118 -0
  45. package/dist/components/SelectableCardOptionGroup/components/Option.d.ts +17 -0
  46. package/dist/components/SelectableCardOptionGroup/components/Option.js +116 -0
  47. package/dist/components/SelectableCardOptionGroup/index.cjs +67 -0
  48. package/dist/components/SelectableCardOptionGroup/index.d.ts +46 -0
  49. package/dist/components/SelectableCardOptionGroup/index.js +65 -0
  50. package/dist/components/SelectableCardOptionGroup/types.d.ts +1 -0
  51. package/dist/components/Slider/components/DoubleSlider.cjs +12 -12
  52. package/dist/components/Slider/components/DoubleSlider.js +8 -8
  53. package/dist/components/Slider/components/SingleSlider.cjs +22 -14
  54. package/dist/components/Slider/components/SingleSlider.js +19 -11
  55. package/dist/components/Table/Row.cjs +20 -32
  56. package/dist/components/Table/Row.js +20 -32
  57. package/dist/components/Table/index.cjs +20 -4
  58. package/dist/components/Table/index.d.ts +6 -5
  59. package/dist/components/Table/index.js +21 -5
  60. package/dist/components/Text/index.cjs +4 -3
  61. package/dist/components/Text/index.d.ts +2 -1
  62. package/dist/components/Text/index.js +4 -3
  63. package/dist/components/index.d.ts +1 -0
  64. package/dist/index.cjs +61 -59
  65. package/dist/index.js +2 -0
  66. package/dist/types.d.ts +6 -0
  67. package/package.json +2 -4
  68. package/dist/components/Slider/components/Label.cjs +0 -17
  69. package/dist/components/Slider/components/Label.d.ts +0 -2
  70. package/dist/components/Slider/components/Label.js +0 -17
  71. /package/dist/components/Dialog/{subComponents → components}/Button.cjs +0 -0
  72. /package/dist/components/Dialog/{subComponents → components}/Button.d.ts +0 -0
  73. /package/dist/components/Dialog/{subComponents → components}/Button.js +0 -0
  74. /package/dist/components/Dialog/{subComponents → components}/Buttons.cjs +0 -0
  75. /package/dist/components/Dialog/{subComponents → components}/Buttons.d.ts +0 -0
  76. /package/dist/components/Dialog/{subComponents → components}/Buttons.js +0 -0
  77. /package/dist/components/Dialog/{subComponents → components}/Stack.cjs +0 -0
  78. /package/dist/components/Dialog/{subComponents → components}/Stack.d.ts +0 -0
  79. /package/dist/components/Dialog/{subComponents → components}/Stack.js +0 -0
  80. /package/dist/components/Dialog/{subComponents → components}/Text.cjs +0 -0
  81. /package/dist/components/Dialog/{subComponents → components}/Text.d.ts +0 -0
  82. /package/dist/components/Dialog/{subComponents → components}/Text.js +0 -0
@@ -20,7 +20,7 @@ const StateStack = /* @__PURE__ */ _styled(Stack, process.env.NODE_ENV === "prod
20
20
  label: "StateStack"
21
21
  })("padding-right:", ({
22
22
  theme
23
- }) => theme.space["2"], ";display:flex;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/SelectBar.tsx"],"names":[],"mappings":"AA6CgC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/SelectBar.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport {\n  AlertCircleIcon,\n  ArrowDownIcon,\n  CheckCircleIcon,\n  PlusIcon,\n} from '@ultraviolet/icons'\nimport type { RefObject } from 'react'\nimport { useEffect, useMemo, useRef, useState } from 'react'\nimport { Button } from '../Button'\nimport { Stack } from '../Stack'\nimport { Tag } from '../Tag'\nimport { Text } from '../Text'\nimport { Tooltip } from '../Tooltip'\nimport { useSelectInput } from './SelectInputProvider'\nimport { findOptionInOptions } from './findOptionInOptions'\nimport type { DataType, OptionType } from './types'\nimport { INPUT_SIZE_HEIGHT, SIZES_TAG } from './types'\n\ntype SelectBarProps = {\n  size: 'small' | 'medium' | 'large'\n  clearable: boolean\n  disabled: boolean\n  readOnly: boolean\n  placeholder: string\n  success?: string\n  error?: string\n  autoFocus?: boolean\n  innerRef: RefObject<HTMLDivElement | null>\n  id?: string\n  'data-testid': string\n  label?: string\n  tooltip?: string\n}\n\ntype DisplayValuesProps = {\n  refTag: RefObject<HTMLDivElement | null>\n  nonOverflowedValues: OptionType[]\n  disabled: boolean\n  readOnly: boolean\n  overflowed: boolean\n  overflowAmount: number\n  size: 'small' | 'medium' | 'large'\n}\n\nconst StateStack = styled(Stack)`\n  padding-right: ${({ theme }) => theme.space['2']};\n  display: flex;\n`\nconst Placeholder = styled(Text)`\nuser-select: none;\n`\n\nconst StyledInputWrapper = styled(Stack)<{\n  'data-readonly': boolean\n  'data-disabled': boolean\n  'data-size': 'small' | 'medium' | 'large'\n  'data-state': 'neutral' | 'success' | 'danger'\n  'data-dropdownvisible': boolean\n  'aria-label'?: string\n}>`\n  display: flex;\n  padding: ${({ theme }) => theme.space[1]};\n  padding-right: 0;\n  padding-left: ${({ theme }) => theme.space[2]};\n  cursor: pointer;\n  box-shadow: none;\n  background: ${({ theme }) => theme.colors.neutral.background};\n  border-radius: ${({ theme }) => theme.radii.default};\n  width: 100%;\n  overflow: hidden;\n\n  &[data-size='small'] {\n    height: ${({ theme }) => theme.sizing[INPUT_SIZE_HEIGHT.small]};\n    padding-left: ${({ theme }) => theme.space[1]};\n  }\n  &[data-size='medium'] {\n    height: ${({ theme }) => theme.sizing[INPUT_SIZE_HEIGHT.medium]};\n  }\n  &[data-size='large'] {\n    height: ${({ theme }) => theme.sizing[INPUT_SIZE_HEIGHT.large]};\n  }\n  &[data-state='neutral'] {\n    border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n\n    &:not([data-disabled=\"true\"]):not([data-readonly=\"true\"]):active {\n      border-color: ${({ theme }) => theme.colors.primary.borderHover};\n      box-shadow: ${({ theme }) => theme.shadows.focusPrimary};\n    }\n    &:not([data-disabled='true']):hover,\n    :not([data-disabled='true']):focus {\n      border-color: ${({ theme }) => theme.colors.primary.borderHover};\n      outline: none;\n    }\n\n    &[data-dropdownvisible='true'] {\n    border-color: ${({ theme }) => theme.colors.primary.borderHover};\n  }\n  }\n\n  &[data-state='success'] {\n    border: 1px solid ${({ theme }) => theme.colors.success.border};\n    &:not([data-disabled=\"true\"]):not([data-readonly=\"true\"]):active {\n      border-color: ${({ theme }) => theme.colors.success.borderHover};\n      box-shadow: ${({ theme }) => theme.shadows.focusSuccess};\n    }\n\n    &[data-dropdownvisible='true'] {\n      border-color: ${({ theme }) => theme.colors.success.borderHover};\n    }\n  }\n\n  &[data-state='danger'] {\n    border: 1px solid ${({ theme }) => theme.colors.danger.border};\n\n    &:not([data-disabled=\"true\"]):not([data-readonly=\"true\"]):active {\n      border-color: ${({ theme }) => theme.colors.danger.borderHover};\n      box-shadow: ${({ theme }) => theme.shadows.focusDanger};\n    }\n\n    &[data-dropdownvisible='true'] {\n      border-color: ${({ theme }) => theme.colors.danger.borderHover};\n    }\n  }\n\n  &:not([data-disabled='true']):not([data-readonly]):hover {\n    border-color: ${({ theme }) => theme.colors.primary.border};\n  }\n\n  &[data-readonly='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n    border-color: ${({ theme }) => theme.colors.neutral.border};\n    cursor: default;\n  }\n\n  &[data-disabled='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    border-color: ${({ theme }) => theme.colors.neutral.borderDisabled};\n    cursor: not-allowed;\n  }\n`\nconst CustomTag = styled(Tag)`\n  height: fit-content;\n  width: fit-content;\n`\nconst SelectedValues = styled(Text)`\ntext-overflow: ellipsis;\noverflow: hidden;\n`\n\nconst isValidSelectedValue = (selectedValue: string, options: DataType) =>\n  !Array.isArray(options)\n    ? Object.keys(options).some(group =>\n        options[group].some(\n          option => option.value === selectedValue && !option.disabled,\n        ),\n      )\n    : options.some(option => option.value === selectedValue && !option.disabled)\n\nconst DisplayValues = ({\n  refTag,\n  nonOverflowedValues,\n  disabled,\n  readOnly,\n  overflowed,\n  overflowAmount,\n  size,\n}: DisplayValuesProps) => {\n  const { multiselect, selectedData, setSelectedData, options, onChange } =\n    useSelectInput()\n\n  return multiselect ? (\n    <Stack\n      direction=\"row\"\n      gap=\"1\"\n      wrap=\"nowrap\"\n      ref={refTag}\n      alignItems=\"center\"\n    >\n      {nonOverflowedValues.map(option => (\n        <CustomTag\n          data-testid=\"selected-options-tags\"\n          key={option?.value}\n          sentiment=\"neutral\"\n          disabled={disabled}\n          onClose={\n            !readOnly\n              ? event => {\n                  event.stopPropagation()\n                  setSelectedData({\n                    type: 'selectOption',\n                    clickedOption: option,\n                  })\n                  const newSelectedValues = selectedData.selectedValues?.filter(\n                    val => val !== option.value,\n                  )\n                  onChange?.(newSelectedValues)\n                }\n              : undefined\n          }\n        >\n          {option?.label}\n        </CustomTag>\n      ))}\n      {overflowed ? (\n        <Tag\n          sentiment=\"neutral\"\n          disabled={disabled}\n          key=\"+\"\n          data-testid=\"plus-tag\"\n          aria-label=\"Plus tag\"\n        >\n          <PlusIcon />\n          {overflowAmount}\n        </Tag>\n      ) : null}\n    </Stack>\n  ) : (\n    <SelectedValues\n      as=\"div\"\n      variant={size === 'large' ? 'body' : 'bodySmall'}\n      disabled={disabled}\n      prominence=\"default\"\n      sentiment=\"neutral\"\n    >\n      {selectedData.selectedValues[0]\n        ? findOptionInOptions(options, selectedData.selectedValues[0])?.label\n        : null}\n    </SelectedValues>\n  )\n}\n\nexport const SelectBar = ({\n  size,\n  clearable,\n  disabled,\n  readOnly,\n  placeholder,\n  success,\n  error,\n  autoFocus,\n  tooltip,\n  innerRef,\n  id,\n  'data-testid': dataTestId,\n  label,\n}: SelectBarProps) => {\n  const {\n    isDropdownVisible,\n    onChange,\n    setIsDropdownVisible,\n    options,\n    selectedData,\n    setSelectedData,\n    multiselect,\n  } = useSelectInput()\n  const openable = !(readOnly || disabled)\n  const refTag = useRef<HTMLDivElement>(null)\n  const width = innerRef.current?.offsetWidth\n  const [overflowAmount, setOverflowAmount] = useState(0)\n  const [nonOverflowedValues, setNonOverFlowedValues] = useState<OptionType[]>(\n    () => {\n      if (selectedData.selectedValues[0]) {\n        const firstSelectOption = findOptionInOptions(\n          options,\n          selectedData.selectedValues[0],\n        )\n\n        return firstSelectOption ? [firstSelectOption] : []\n      }\n\n      return []\n    },\n  )\n\n  const state = useMemo(() => {\n    if (error) {\n      return 'danger'\n    }\n    if (success) {\n      return 'success'\n    }\n\n    return 'neutral'\n  }, [error, success])\n\n  useEffect(() => {\n    // When too many items are selected, too avoid overflow, compute the number of tags to display and add a + tag\n    let tagsWidth = 0\n    let computedOverflowAmount = 0\n    let computedNonOverflowedValues: OptionType[] = []\n    const newSelectedValues = selectedData.selectedValues.filter(\n      selectedValue => isValidSelectedValue(selectedValue, options),\n    )\n    for (const selectedValue of newSelectedValues) {\n      const selectedOption = findOptionInOptions(options, selectedValue)\n      if (\n        selectedOption?.label &&\n        width &&\n        isValidSelectedValue(selectedValue, options)\n      ) {\n        const lengthValue = selectedOption.value.length // Find a better way to find the number of displayed characters?\n        const totalTagWidth =\n          SIZES_TAG.tagWidth + SIZES_TAG.letterWidth * lengthValue\n        if (totalTagWidth + tagsWidth > width - 100) {\n          computedOverflowAmount += 1\n        } else {\n          computedNonOverflowedValues = [\n            ...computedNonOverflowedValues,\n            selectedOption,\n          ]\n          tagsWidth += totalTagWidth\n        }\n      }\n    }\n    setNonOverFlowedValues(computedNonOverflowedValues)\n    setOverflowAmount(computedOverflowAmount)\n  }, [options, selectedData.selectedValues, width])\n\n  useEffect(() => {\n    setSelectedData({ type: 'update' })\n  }, [setSelectedData, options])\n\n  const shouldDisplayValues = useMemo(() => {\n    if (multiselect) {\n      return (\n        nonOverflowedValues.length > 0 ||\n        selectedData.selectedValues.some(\n          selectedValue =>\n            findOptionInOptions(options, selectedValue) !== undefined,\n        )\n      )\n    }\n\n    return (\n      selectedData.selectedValues[0] !== undefined &&\n      findOptionInOptions(options, selectedData.selectedValues[0]) !== undefined\n    )\n  }, [\n    multiselect,\n    nonOverflowedValues.length,\n    options,\n    selectedData.selectedValues,\n  ])\n\n  return (\n    <Tooltip text={tooltip}>\n      <StyledInputWrapper\n        role=\"combobox\"\n        id={id}\n        data-disabled={disabled}\n        data-readonly={readOnly}\n        data-size={size}\n        data-dropdownvisible={isDropdownVisible}\n        data-state={state}\n        direction=\"row\"\n        wrap=\"nowrap\"\n        gap=\"1\"\n        justifyContent=\"space-between\"\n        alignItems=\"center\"\n        onClick={\n          openable ? () => setIsDropdownVisible(!isDropdownVisible) : undefined\n        }\n        data-testid={dataTestId}\n        autoFocus={autoFocus}\n        onKeyDown={event => {\n          if (event.key === 'ArrowDown') {\n            if (!isDropdownVisible) {\n              setIsDropdownVisible(true)\n            } else {\n              document.getElementById(`option-0`)?.focus()\n            }\n          }\n          if (event.key === ' ') {\n            event.preventDefault()\n          }\n\n          return ['Enter', ' '].includes(event.key) && openable\n            ? setIsDropdownVisible(!isDropdownVisible)\n            : null\n        }}\n        ref={innerRef}\n        aria-haspopup=\"listbox\"\n        aria-expanded={isDropdownVisible}\n        tabIndex={0}\n        aria-label={label}\n      >\n        {shouldDisplayValues ? (\n          <DisplayValues\n            refTag={refTag}\n            nonOverflowedValues={nonOverflowedValues}\n            disabled={disabled}\n            readOnly={readOnly}\n            overflowed={!!overflowAmount}\n            overflowAmount={overflowAmount}\n            size={size}\n          />\n        ) : (\n          <Placeholder\n            as=\"p\"\n            variant={size === 'large' ? 'body' : 'bodySmall'}\n            sentiment=\"neutral\"\n            disabled={disabled}\n            prominence=\"weak\"\n          >\n            {placeholder}\n          </Placeholder>\n        )}\n        <StateStack direction=\"row\" gap={1} alignItems=\"center\">\n          {error ? <AlertCircleIcon sentiment=\"danger\" /> : null}\n          {success && !error ? <CheckCircleIcon sentiment=\"success\" /> : null}\n          {clearable && selectedData.selectedValues.length > 0 ? (\n            <Button\n              aria-label=\"clear value\"\n              disabled={disabled || !selectedData.selectedValues[0] || readOnly}\n              variant=\"ghost\"\n              size=\"small\"\n              icon=\"close\"\n              onClick={event => {\n                event.stopPropagation()\n                setSelectedData({ type: 'clearAll' })\n                if (multiselect) {\n                  onChange?.([])\n                } else {\n                  onChange?.('')\n                }\n              }}\n              sentiment=\"neutral\"\n              data-testid=\"clear-all\"\n            />\n          ) : null}\n          <ArrowDownIcon\n            aria-label=\"show dropdown\"\n            size=\"small\"\n            sentiment=\"neutral\"\n            disabled={disabled || readOnly}\n          />\n        </StateStack>\n      </StyledInputWrapper>\n    </Tooltip>\n  )\n}\n"]} */"));
23
+ }) => theme.space["2"], ";display:flex;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/SelectBar.tsx"],"names":[],"mappings":"AA6CgC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/SelectBar.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport {\n  AlertCircleIcon,\n  ArrowDownIcon,\n  CheckCircleIcon,\n  PlusIcon,\n} from '@ultraviolet/icons'\nimport type { RefObject } from 'react'\nimport { useEffect, useMemo, useRef, useState } from 'react'\nimport { Button } from '../Button'\nimport { Stack } from '../Stack'\nimport { Tag } from '../Tag'\nimport { Text } from '../Text'\nimport { Tooltip } from '../Tooltip'\nimport { useSelectInput } from './SelectInputProvider'\nimport { findOptionInOptions } from './findOptionInOptions'\nimport type { DataType, OptionType } from './types'\nimport { INPUT_SIZE_HEIGHT, SIZES_TAG } from './types'\n\ntype SelectBarProps = {\n  size: 'small' | 'medium' | 'large'\n  clearable: boolean\n  disabled: boolean\n  readOnly: boolean\n  placeholder: string\n  success?: string\n  error?: string | boolean\n  autoFocus?: boolean\n  innerRef: RefObject<HTMLDivElement | null>\n  id?: string\n  'data-testid': string\n  label?: string\n  tooltip?: string\n}\n\ntype DisplayValuesProps = {\n  refTag: RefObject<HTMLDivElement | null>\n  nonOverflowedValues: OptionType[]\n  disabled: boolean\n  readOnly: boolean\n  overflowed: boolean\n  overflowAmount: number\n  size: 'small' | 'medium' | 'large'\n}\n\nconst StateStack = styled(Stack)`\n  padding-right: ${({ theme }) => theme.space['2']};\n  display: flex;\n`\nconst Placeholder = styled(Text)`\nuser-select: none;\n`\n\nexport const StyledInputWrapper = styled(Stack)<{\n  'data-readonly': boolean\n  'data-disabled': boolean\n  'data-size': 'small' | 'medium' | 'large'\n  'data-state': 'neutral' | 'success' | 'danger'\n  'data-dropdownvisible': boolean\n  'aria-label'?: string\n}>`\n  display: flex;\n  padding: ${({ theme }) => theme.space[1]};\n  padding-right: 0;\n  padding-left: ${({ theme }) => theme.space[2]};\n  cursor: pointer;\n  box-shadow: none;\n  background: ${({ theme }) => theme.colors.neutral.background};\n  border-radius: ${({ theme }) => theme.radii.default};\n  width: 100%;\n  overflow: hidden;\n\n  &[data-size='small'] {\n    height: ${({ theme }) => theme.sizing[INPUT_SIZE_HEIGHT.small]};\n    padding-left: ${({ theme }) => theme.space[1]};\n  }\n  &[data-size='medium'] {\n    height: ${({ theme }) => theme.sizing[INPUT_SIZE_HEIGHT.medium]};\n  }\n  &[data-size='large'] {\n    height: ${({ theme }) => theme.sizing[INPUT_SIZE_HEIGHT.large]};\n  }\n  &[data-state='neutral'] {\n    border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n\n    &:not([data-disabled=\"true\"]):not([data-readonly=\"true\"]):active {\n      border-color: ${({ theme }) => theme.colors.primary.borderHover};\n      box-shadow: ${({ theme }) => theme.shadows.focusPrimary};\n    }\n    &:not([data-disabled='true']):hover {\n      border-color: ${({ theme }) => theme.colors.primary.borderHover};\n      outline: none;\n    }\n\n    &:not([data-disabled='true']):focus-visible {\n      outline: 5px auto Highlight;\n      outline: 5px auto -webkit-focus-ring-color;\n    }\n\n    &[data-dropdownvisible='true'] {\n    border-color: ${({ theme }) => theme.colors.primary.borderHover};\n  }\n  }\n\n  &[data-state='success'] {\n    border: 1px solid ${({ theme }) => theme.colors.success.border};\n    &:not([data-disabled=\"true\"]):not([data-readonly=\"true\"]):active {\n      border-color: ${({ theme }) => theme.colors.success.borderHover};\n      box-shadow: ${({ theme }) => theme.shadows.focusSuccess};\n    }\n\n    &[data-dropdownvisible='true'] {\n      border-color: ${({ theme }) => theme.colors.success.borderHover};\n    }\n  }\n\n  &[data-state='danger'] {\n    border: 1px solid ${({ theme }) => theme.colors.danger.border};\n\n    &:not([data-disabled=\"true\"]):not([data-readonly=\"true\"]):active {\n      border-color: ${({ theme }) => theme.colors.danger.borderHover};\n      box-shadow: ${({ theme }) => theme.shadows.focusDanger};\n    }\n\n    &[data-dropdownvisible='true'] {\n      border-color: ${({ theme }) => theme.colors.danger.borderHover};\n    }\n  }\n\n  &:not([data-disabled='true']):not([data-readonly]):hover {\n    border-color: ${({ theme }) => theme.colors.primary.border};\n  }\n\n  &[data-readonly='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n    border-color: ${({ theme }) => theme.colors.neutral.border};\n    cursor: default;\n  }\n\n  &[data-disabled='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    border-color: ${({ theme }) => theme.colors.neutral.borderDisabled};\n    cursor: not-allowed;\n  }\n`\n\nconst CustomTag = styled(Tag)`\n  height: fit-content;\n  width: fit-content;\n`\n\nconst SelectedValues = styled(Text)`\n  text-overflow: ellipsis;\n  overflow: hidden;\n  white-space: nowrap;\n`\n\nconst isValidSelectedValue = (selectedValue: string, options: DataType) =>\n  !Array.isArray(options)\n    ? Object.keys(options).some(group =>\n        options[group].some(\n          option => option.value === selectedValue && !option.disabled,\n        ),\n      )\n    : options.some(option => option.value === selectedValue && !option.disabled)\n\nconst DisplayValues = ({\n  refTag,\n  nonOverflowedValues,\n  disabled,\n  readOnly,\n  overflowed,\n  overflowAmount,\n  size,\n}: DisplayValuesProps) => {\n  const { multiselect, selectedData, setSelectedData, options, onChange } =\n    useSelectInput()\n\n  return multiselect ? (\n    <Stack\n      direction=\"row\"\n      gap=\"1\"\n      wrap=\"nowrap\"\n      ref={refTag}\n      alignItems=\"center\"\n    >\n      {nonOverflowedValues.map(option => (\n        <CustomTag\n          data-testid=\"selected-options-tags\"\n          key={option?.value}\n          sentiment=\"neutral\"\n          disabled={disabled}\n          onClose={\n            !readOnly\n              ? event => {\n                  event.stopPropagation()\n                  setSelectedData({\n                    type: 'selectOption',\n                    clickedOption: option,\n                  })\n                  const newSelectedValues = selectedData.selectedValues?.filter(\n                    val => val !== option.value,\n                  )\n                  onChange?.(newSelectedValues)\n                }\n              : undefined\n          }\n        >\n          {option?.label}\n        </CustomTag>\n      ))}\n      {overflowed ? (\n        <Tag\n          sentiment=\"neutral\"\n          disabled={disabled}\n          key=\"+\"\n          data-testid=\"plus-tag\"\n          aria-label=\"Plus tag\"\n        >\n          <PlusIcon />\n          {overflowAmount}\n        </Tag>\n      ) : null}\n    </Stack>\n  ) : (\n    <SelectedValues\n      as=\"div\"\n      variant={size === 'large' ? 'body' : 'bodySmall'}\n      disabled={disabled}\n      prominence=\"default\"\n      sentiment=\"neutral\"\n    >\n      {selectedData.selectedValues[0]\n        ? findOptionInOptions(options, selectedData.selectedValues[0])?.label\n        : null}\n    </SelectedValues>\n  )\n}\n\nexport const SelectBar = ({\n  size,\n  clearable,\n  disabled,\n  readOnly,\n  placeholder,\n  success,\n  error,\n  autoFocus,\n  tooltip,\n  innerRef,\n  id,\n  'data-testid': dataTestId,\n  label,\n}: SelectBarProps) => {\n  const {\n    isDropdownVisible,\n    onChange,\n    setIsDropdownVisible,\n    options,\n    selectedData,\n    setSelectedData,\n    multiselect,\n  } = useSelectInput()\n  const openable = !(readOnly || disabled)\n  const refTag = useRef<HTMLDivElement>(null)\n  const width = innerRef.current?.offsetWidth\n  const [overflowAmount, setOverflowAmount] = useState(0)\n  const [nonOverflowedValues, setNonOverFlowedValues] = useState<OptionType[]>(\n    () => {\n      if (selectedData.selectedValues[0]) {\n        const firstSelectOption = findOptionInOptions(\n          options,\n          selectedData.selectedValues[0],\n        )\n\n        return firstSelectOption ? [firstSelectOption] : []\n      }\n\n      return []\n    },\n  )\n\n  const state = useMemo(() => {\n    if (error) {\n      return 'danger'\n    }\n    if (success) {\n      return 'success'\n    }\n\n    return 'neutral'\n  }, [error, success])\n\n  useEffect(() => {\n    // When too many items are selected, too avoid overflow, compute the number of tags to display and add a + tag\n    let tagsWidth = 0\n    let computedOverflowAmount = 0\n    let computedNonOverflowedValues: OptionType[] = []\n    const newSelectedValues = selectedData.selectedValues.filter(\n      selectedValue => isValidSelectedValue(selectedValue, options),\n    )\n    for (const selectedValue of newSelectedValues) {\n      const selectedOption = findOptionInOptions(options, selectedValue)\n      if (\n        selectedOption?.label &&\n        width &&\n        isValidSelectedValue(selectedValue, options)\n      ) {\n        const lengthValue = selectedOption.value.length // Find a better way to find the number of displayed characters?\n        const totalTagWidth =\n          SIZES_TAG.tagWidth + SIZES_TAG.letterWidth * lengthValue\n        if (totalTagWidth + tagsWidth > width - 100) {\n          computedOverflowAmount += 1\n        } else {\n          computedNonOverflowedValues = [\n            ...computedNonOverflowedValues,\n            selectedOption,\n          ]\n          tagsWidth += totalTagWidth\n        }\n      }\n    }\n    setNonOverFlowedValues(computedNonOverflowedValues)\n    setOverflowAmount(computedOverflowAmount)\n  }, [options, selectedData.selectedValues, width])\n\n  useEffect(() => {\n    setSelectedData({ type: 'update' })\n  }, [setSelectedData, options])\n\n  const shouldDisplayValues = useMemo(() => {\n    if (multiselect) {\n      return (\n        nonOverflowedValues.length > 0 ||\n        selectedData.selectedValues.some(\n          selectedValue =>\n            findOptionInOptions(options, selectedValue) !== undefined,\n        )\n      )\n    }\n\n    return (\n      selectedData.selectedValues[0] !== undefined &&\n      findOptionInOptions(options, selectedData.selectedValues[0]) !== undefined\n    )\n  }, [\n    multiselect,\n    nonOverflowedValues.length,\n    options,\n    selectedData.selectedValues,\n  ])\n\n  return (\n    <Tooltip text={tooltip}>\n      <StyledInputWrapper\n        role=\"combobox\"\n        id={id}\n        data-disabled={disabled}\n        data-readonly={readOnly}\n        data-size={size}\n        data-dropdownvisible={isDropdownVisible}\n        data-state={state}\n        direction=\"row\"\n        wrap=\"nowrap\"\n        gap=\"1\"\n        justifyContent=\"space-between\"\n        alignItems=\"center\"\n        onClick={\n          openable ? () => setIsDropdownVisible(!isDropdownVisible) : undefined\n        }\n        data-testid={dataTestId}\n        autoFocus={autoFocus}\n        onKeyDown={event => {\n          if (event.key === 'ArrowDown') {\n            if (!isDropdownVisible) {\n              setIsDropdownVisible(true)\n            } else {\n              document.getElementById(`option-0`)?.focus()\n            }\n          }\n          if (event.key === ' ') {\n            event.preventDefault()\n          }\n\n          return ['Enter', ' '].includes(event.key) && openable\n            ? setIsDropdownVisible(!isDropdownVisible)\n            : null\n        }}\n        ref={innerRef}\n        aria-haspopup=\"listbox\"\n        aria-expanded={isDropdownVisible}\n        tabIndex={0}\n        aria-label={label}\n      >\n        {shouldDisplayValues ? (\n          <DisplayValues\n            refTag={refTag}\n            nonOverflowedValues={nonOverflowedValues}\n            disabled={disabled}\n            readOnly={readOnly}\n            overflowed={!!overflowAmount}\n            overflowAmount={overflowAmount}\n            size={size}\n          />\n        ) : (\n          <Placeholder\n            as=\"p\"\n            variant={size === 'large' ? 'body' : 'bodySmall'}\n            sentiment=\"neutral\"\n            disabled={disabled}\n            prominence=\"weak\"\n          >\n            {placeholder}\n          </Placeholder>\n        )}\n        <StateStack direction=\"row\" gap={1} alignItems=\"center\">\n          {error ? <AlertCircleIcon sentiment=\"danger\" /> : null}\n          {success && !error ? <CheckCircleIcon sentiment=\"success\" /> : null}\n          {clearable && selectedData.selectedValues.length > 0 ? (\n            <Button\n              aria-label=\"clear value\"\n              disabled={disabled || !selectedData.selectedValues[0] || readOnly}\n              variant=\"ghost\"\n              size=\"small\"\n              icon=\"close\"\n              onClick={event => {\n                event.stopPropagation()\n                setSelectedData({ type: 'clearAll' })\n                if (multiselect) {\n                  onChange?.([])\n                } else {\n                  onChange?.('')\n                }\n              }}\n              sentiment=\"neutral\"\n              data-testid=\"clear-all\"\n            />\n          ) : null}\n          <ArrowDownIcon\n            aria-label=\"show dropdown\"\n            size=\"small\"\n            sentiment=\"neutral\"\n            disabled={disabled || readOnly}\n          />\n        </StateStack>\n      </StyledInputWrapper>\n    </Tooltip>\n  )\n}\n"]} */"));
24
24
  const Placeholder = /* @__PURE__ */ _styled(Text, process.env.NODE_ENV === "production" ? {
25
25
  target: "ejy0aca3"
26
26
  } : {
@@ -31,7 +31,7 @@ const Placeholder = /* @__PURE__ */ _styled(Text, process.env.NODE_ENV === "prod
31
31
  styles: "user-select:none"
32
32
  } : {
33
33
  name: "oldbq4",
34
- styles: "user-select:none/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/SelectBar.tsx"],"names":[],"mappings":"AAiDgC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/SelectBar.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport {\n  AlertCircleIcon,\n  ArrowDownIcon,\n  CheckCircleIcon,\n  PlusIcon,\n} from '@ultraviolet/icons'\nimport type { RefObject } from 'react'\nimport { useEffect, useMemo, useRef, useState } from 'react'\nimport { Button } from '../Button'\nimport { Stack } from '../Stack'\nimport { Tag } from '../Tag'\nimport { Text } from '../Text'\nimport { Tooltip } from '../Tooltip'\nimport { useSelectInput } from './SelectInputProvider'\nimport { findOptionInOptions } from './findOptionInOptions'\nimport type { DataType, OptionType } from './types'\nimport { INPUT_SIZE_HEIGHT, SIZES_TAG } from './types'\n\ntype SelectBarProps = {\n  size: 'small' | 'medium' | 'large'\n  clearable: boolean\n  disabled: boolean\n  readOnly: boolean\n  placeholder: string\n  success?: string\n  error?: string\n  autoFocus?: boolean\n  innerRef: RefObject<HTMLDivElement | null>\n  id?: string\n  'data-testid': string\n  label?: string\n  tooltip?: string\n}\n\ntype DisplayValuesProps = {\n  refTag: RefObject<HTMLDivElement | null>\n  nonOverflowedValues: OptionType[]\n  disabled: boolean\n  readOnly: boolean\n  overflowed: boolean\n  overflowAmount: number\n  size: 'small' | 'medium' | 'large'\n}\n\nconst StateStack = styled(Stack)`\n  padding-right: ${({ theme }) => theme.space['2']};\n  display: flex;\n`\nconst Placeholder = styled(Text)`\nuser-select: none;\n`\n\nconst StyledInputWrapper = styled(Stack)<{\n  'data-readonly': boolean\n  'data-disabled': boolean\n  'data-size': 'small' | 'medium' | 'large'\n  'data-state': 'neutral' | 'success' | 'danger'\n  'data-dropdownvisible': boolean\n  'aria-label'?: string\n}>`\n  display: flex;\n  padding: ${({ theme }) => theme.space[1]};\n  padding-right: 0;\n  padding-left: ${({ theme }) => theme.space[2]};\n  cursor: pointer;\n  box-shadow: none;\n  background: ${({ theme }) => theme.colors.neutral.background};\n  border-radius: ${({ theme }) => theme.radii.default};\n  width: 100%;\n  overflow: hidden;\n\n  &[data-size='small'] {\n    height: ${({ theme }) => theme.sizing[INPUT_SIZE_HEIGHT.small]};\n    padding-left: ${({ theme }) => theme.space[1]};\n  }\n  &[data-size='medium'] {\n    height: ${({ theme }) => theme.sizing[INPUT_SIZE_HEIGHT.medium]};\n  }\n  &[data-size='large'] {\n    height: ${({ theme }) => theme.sizing[INPUT_SIZE_HEIGHT.large]};\n  }\n  &[data-state='neutral'] {\n    border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n\n    &:not([data-disabled=\"true\"]):not([data-readonly=\"true\"]):active {\n      border-color: ${({ theme }) => theme.colors.primary.borderHover};\n      box-shadow: ${({ theme }) => theme.shadows.focusPrimary};\n    }\n    &:not([data-disabled='true']):hover,\n    :not([data-disabled='true']):focus {\n      border-color: ${({ theme }) => theme.colors.primary.borderHover};\n      outline: none;\n    }\n\n    &[data-dropdownvisible='true'] {\n    border-color: ${({ theme }) => theme.colors.primary.borderHover};\n  }\n  }\n\n  &[data-state='success'] {\n    border: 1px solid ${({ theme }) => theme.colors.success.border};\n    &:not([data-disabled=\"true\"]):not([data-readonly=\"true\"]):active {\n      border-color: ${({ theme }) => theme.colors.success.borderHover};\n      box-shadow: ${({ theme }) => theme.shadows.focusSuccess};\n    }\n\n    &[data-dropdownvisible='true'] {\n      border-color: ${({ theme }) => theme.colors.success.borderHover};\n    }\n  }\n\n  &[data-state='danger'] {\n    border: 1px solid ${({ theme }) => theme.colors.danger.border};\n\n    &:not([data-disabled=\"true\"]):not([data-readonly=\"true\"]):active {\n      border-color: ${({ theme }) => theme.colors.danger.borderHover};\n      box-shadow: ${({ theme }) => theme.shadows.focusDanger};\n    }\n\n    &[data-dropdownvisible='true'] {\n      border-color: ${({ theme }) => theme.colors.danger.borderHover};\n    }\n  }\n\n  &:not([data-disabled='true']):not([data-readonly]):hover {\n    border-color: ${({ theme }) => theme.colors.primary.border};\n  }\n\n  &[data-readonly='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n    border-color: ${({ theme }) => theme.colors.neutral.border};\n    cursor: default;\n  }\n\n  &[data-disabled='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    border-color: ${({ theme }) => theme.colors.neutral.borderDisabled};\n    cursor: not-allowed;\n  }\n`\nconst CustomTag = styled(Tag)`\n  height: fit-content;\n  width: fit-content;\n`\nconst SelectedValues = styled(Text)`\ntext-overflow: ellipsis;\noverflow: hidden;\n`\n\nconst isValidSelectedValue = (selectedValue: string, options: DataType) =>\n  !Array.isArray(options)\n    ? Object.keys(options).some(group =>\n        options[group].some(\n          option => option.value === selectedValue && !option.disabled,\n        ),\n      )\n    : options.some(option => option.value === selectedValue && !option.disabled)\n\nconst DisplayValues = ({\n  refTag,\n  nonOverflowedValues,\n  disabled,\n  readOnly,\n  overflowed,\n  overflowAmount,\n  size,\n}: DisplayValuesProps) => {\n  const { multiselect, selectedData, setSelectedData, options, onChange } =\n    useSelectInput()\n\n  return multiselect ? (\n    <Stack\n      direction=\"row\"\n      gap=\"1\"\n      wrap=\"nowrap\"\n      ref={refTag}\n      alignItems=\"center\"\n    >\n      {nonOverflowedValues.map(option => (\n        <CustomTag\n          data-testid=\"selected-options-tags\"\n          key={option?.value}\n          sentiment=\"neutral\"\n          disabled={disabled}\n          onClose={\n            !readOnly\n              ? event => {\n                  event.stopPropagation()\n                  setSelectedData({\n                    type: 'selectOption',\n                    clickedOption: option,\n                  })\n                  const newSelectedValues = selectedData.selectedValues?.filter(\n                    val => val !== option.value,\n                  )\n                  onChange?.(newSelectedValues)\n                }\n              : undefined\n          }\n        >\n          {option?.label}\n        </CustomTag>\n      ))}\n      {overflowed ? (\n        <Tag\n          sentiment=\"neutral\"\n          disabled={disabled}\n          key=\"+\"\n          data-testid=\"plus-tag\"\n          aria-label=\"Plus tag\"\n        >\n          <PlusIcon />\n          {overflowAmount}\n        </Tag>\n      ) : null}\n    </Stack>\n  ) : (\n    <SelectedValues\n      as=\"div\"\n      variant={size === 'large' ? 'body' : 'bodySmall'}\n      disabled={disabled}\n      prominence=\"default\"\n      sentiment=\"neutral\"\n    >\n      {selectedData.selectedValues[0]\n        ? findOptionInOptions(options, selectedData.selectedValues[0])?.label\n        : null}\n    </SelectedValues>\n  )\n}\n\nexport const SelectBar = ({\n  size,\n  clearable,\n  disabled,\n  readOnly,\n  placeholder,\n  success,\n  error,\n  autoFocus,\n  tooltip,\n  innerRef,\n  id,\n  'data-testid': dataTestId,\n  label,\n}: SelectBarProps) => {\n  const {\n    isDropdownVisible,\n    onChange,\n    setIsDropdownVisible,\n    options,\n    selectedData,\n    setSelectedData,\n    multiselect,\n  } = useSelectInput()\n  const openable = !(readOnly || disabled)\n  const refTag = useRef<HTMLDivElement>(null)\n  const width = innerRef.current?.offsetWidth\n  const [overflowAmount, setOverflowAmount] = useState(0)\n  const [nonOverflowedValues, setNonOverFlowedValues] = useState<OptionType[]>(\n    () => {\n      if (selectedData.selectedValues[0]) {\n        const firstSelectOption = findOptionInOptions(\n          options,\n          selectedData.selectedValues[0],\n        )\n\n        return firstSelectOption ? [firstSelectOption] : []\n      }\n\n      return []\n    },\n  )\n\n  const state = useMemo(() => {\n    if (error) {\n      return 'danger'\n    }\n    if (success) {\n      return 'success'\n    }\n\n    return 'neutral'\n  }, [error, success])\n\n  useEffect(() => {\n    // When too many items are selected, too avoid overflow, compute the number of tags to display and add a + tag\n    let tagsWidth = 0\n    let computedOverflowAmount = 0\n    let computedNonOverflowedValues: OptionType[] = []\n    const newSelectedValues = selectedData.selectedValues.filter(\n      selectedValue => isValidSelectedValue(selectedValue, options),\n    )\n    for (const selectedValue of newSelectedValues) {\n      const selectedOption = findOptionInOptions(options, selectedValue)\n      if (\n        selectedOption?.label &&\n        width &&\n        isValidSelectedValue(selectedValue, options)\n      ) {\n        const lengthValue = selectedOption.value.length // Find a better way to find the number of displayed characters?\n        const totalTagWidth =\n          SIZES_TAG.tagWidth + SIZES_TAG.letterWidth * lengthValue\n        if (totalTagWidth + tagsWidth > width - 100) {\n          computedOverflowAmount += 1\n        } else {\n          computedNonOverflowedValues = [\n            ...computedNonOverflowedValues,\n            selectedOption,\n          ]\n          tagsWidth += totalTagWidth\n        }\n      }\n    }\n    setNonOverFlowedValues(computedNonOverflowedValues)\n    setOverflowAmount(computedOverflowAmount)\n  }, [options, selectedData.selectedValues, width])\n\n  useEffect(() => {\n    setSelectedData({ type: 'update' })\n  }, [setSelectedData, options])\n\n  const shouldDisplayValues = useMemo(() => {\n    if (multiselect) {\n      return (\n        nonOverflowedValues.length > 0 ||\n        selectedData.selectedValues.some(\n          selectedValue =>\n            findOptionInOptions(options, selectedValue) !== undefined,\n        )\n      )\n    }\n\n    return (\n      selectedData.selectedValues[0] !== undefined &&\n      findOptionInOptions(options, selectedData.selectedValues[0]) !== undefined\n    )\n  }, [\n    multiselect,\n    nonOverflowedValues.length,\n    options,\n    selectedData.selectedValues,\n  ])\n\n  return (\n    <Tooltip text={tooltip}>\n      <StyledInputWrapper\n        role=\"combobox\"\n        id={id}\n        data-disabled={disabled}\n        data-readonly={readOnly}\n        data-size={size}\n        data-dropdownvisible={isDropdownVisible}\n        data-state={state}\n        direction=\"row\"\n        wrap=\"nowrap\"\n        gap=\"1\"\n        justifyContent=\"space-between\"\n        alignItems=\"center\"\n        onClick={\n          openable ? () => setIsDropdownVisible(!isDropdownVisible) : undefined\n        }\n        data-testid={dataTestId}\n        autoFocus={autoFocus}\n        onKeyDown={event => {\n          if (event.key === 'ArrowDown') {\n            if (!isDropdownVisible) {\n              setIsDropdownVisible(true)\n            } else {\n              document.getElementById(`option-0`)?.focus()\n            }\n          }\n          if (event.key === ' ') {\n            event.preventDefault()\n          }\n\n          return ['Enter', ' '].includes(event.key) && openable\n            ? setIsDropdownVisible(!isDropdownVisible)\n            : null\n        }}\n        ref={innerRef}\n        aria-haspopup=\"listbox\"\n        aria-expanded={isDropdownVisible}\n        tabIndex={0}\n        aria-label={label}\n      >\n        {shouldDisplayValues ? (\n          <DisplayValues\n            refTag={refTag}\n            nonOverflowedValues={nonOverflowedValues}\n            disabled={disabled}\n            readOnly={readOnly}\n            overflowed={!!overflowAmount}\n            overflowAmount={overflowAmount}\n            size={size}\n          />\n        ) : (\n          <Placeholder\n            as=\"p\"\n            variant={size === 'large' ? 'body' : 'bodySmall'}\n            sentiment=\"neutral\"\n            disabled={disabled}\n            prominence=\"weak\"\n          >\n            {placeholder}\n          </Placeholder>\n        )}\n        <StateStack direction=\"row\" gap={1} alignItems=\"center\">\n          {error ? <AlertCircleIcon sentiment=\"danger\" /> : null}\n          {success && !error ? <CheckCircleIcon sentiment=\"success\" /> : null}\n          {clearable && selectedData.selectedValues.length > 0 ? (\n            <Button\n              aria-label=\"clear value\"\n              disabled={disabled || !selectedData.selectedValues[0] || readOnly}\n              variant=\"ghost\"\n              size=\"small\"\n              icon=\"close\"\n              onClick={event => {\n                event.stopPropagation()\n                setSelectedData({ type: 'clearAll' })\n                if (multiselect) {\n                  onChange?.([])\n                } else {\n                  onChange?.('')\n                }\n              }}\n              sentiment=\"neutral\"\n              data-testid=\"clear-all\"\n            />\n          ) : null}\n          <ArrowDownIcon\n            aria-label=\"show dropdown\"\n            size=\"small\"\n            sentiment=\"neutral\"\n            disabled={disabled || readOnly}\n          />\n        </StateStack>\n      </StyledInputWrapper>\n    </Tooltip>\n  )\n}\n"]} */",
34
+ styles: "user-select:none/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/SelectBar.tsx"],"names":[],"mappings":"AAiDgC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/SelectBar.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport {\n  AlertCircleIcon,\n  ArrowDownIcon,\n  CheckCircleIcon,\n  PlusIcon,\n} from '@ultraviolet/icons'\nimport type { RefObject } from 'react'\nimport { useEffect, useMemo, useRef, useState } from 'react'\nimport { Button } from '../Button'\nimport { Stack } from '../Stack'\nimport { Tag } from '../Tag'\nimport { Text } from '../Text'\nimport { Tooltip } from '../Tooltip'\nimport { useSelectInput } from './SelectInputProvider'\nimport { findOptionInOptions } from './findOptionInOptions'\nimport type { DataType, OptionType } from './types'\nimport { INPUT_SIZE_HEIGHT, SIZES_TAG } from './types'\n\ntype SelectBarProps = {\n  size: 'small' | 'medium' | 'large'\n  clearable: boolean\n  disabled: boolean\n  readOnly: boolean\n  placeholder: string\n  success?: string\n  error?: string | boolean\n  autoFocus?: boolean\n  innerRef: RefObject<HTMLDivElement | null>\n  id?: string\n  'data-testid': string\n  label?: string\n  tooltip?: string\n}\n\ntype DisplayValuesProps = {\n  refTag: RefObject<HTMLDivElement | null>\n  nonOverflowedValues: OptionType[]\n  disabled: boolean\n  readOnly: boolean\n  overflowed: boolean\n  overflowAmount: number\n  size: 'small' | 'medium' | 'large'\n}\n\nconst StateStack = styled(Stack)`\n  padding-right: ${({ theme }) => theme.space['2']};\n  display: flex;\n`\nconst Placeholder = styled(Text)`\nuser-select: none;\n`\n\nexport const StyledInputWrapper = styled(Stack)<{\n  'data-readonly': boolean\n  'data-disabled': boolean\n  'data-size': 'small' | 'medium' | 'large'\n  'data-state': 'neutral' | 'success' | 'danger'\n  'data-dropdownvisible': boolean\n  'aria-label'?: string\n}>`\n  display: flex;\n  padding: ${({ theme }) => theme.space[1]};\n  padding-right: 0;\n  padding-left: ${({ theme }) => theme.space[2]};\n  cursor: pointer;\n  box-shadow: none;\n  background: ${({ theme }) => theme.colors.neutral.background};\n  border-radius: ${({ theme }) => theme.radii.default};\n  width: 100%;\n  overflow: hidden;\n\n  &[data-size='small'] {\n    height: ${({ theme }) => theme.sizing[INPUT_SIZE_HEIGHT.small]};\n    padding-left: ${({ theme }) => theme.space[1]};\n  }\n  &[data-size='medium'] {\n    height: ${({ theme }) => theme.sizing[INPUT_SIZE_HEIGHT.medium]};\n  }\n  &[data-size='large'] {\n    height: ${({ theme }) => theme.sizing[INPUT_SIZE_HEIGHT.large]};\n  }\n  &[data-state='neutral'] {\n    border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n\n    &:not([data-disabled=\"true\"]):not([data-readonly=\"true\"]):active {\n      border-color: ${({ theme }) => theme.colors.primary.borderHover};\n      box-shadow: ${({ theme }) => theme.shadows.focusPrimary};\n    }\n    &:not([data-disabled='true']):hover {\n      border-color: ${({ theme }) => theme.colors.primary.borderHover};\n      outline: none;\n    }\n\n    &:not([data-disabled='true']):focus-visible {\n      outline: 5px auto Highlight;\n      outline: 5px auto -webkit-focus-ring-color;\n    }\n\n    &[data-dropdownvisible='true'] {\n    border-color: ${({ theme }) => theme.colors.primary.borderHover};\n  }\n  }\n\n  &[data-state='success'] {\n    border: 1px solid ${({ theme }) => theme.colors.success.border};\n    &:not([data-disabled=\"true\"]):not([data-readonly=\"true\"]):active {\n      border-color: ${({ theme }) => theme.colors.success.borderHover};\n      box-shadow: ${({ theme }) => theme.shadows.focusSuccess};\n    }\n\n    &[data-dropdownvisible='true'] {\n      border-color: ${({ theme }) => theme.colors.success.borderHover};\n    }\n  }\n\n  &[data-state='danger'] {\n    border: 1px solid ${({ theme }) => theme.colors.danger.border};\n\n    &:not([data-disabled=\"true\"]):not([data-readonly=\"true\"]):active {\n      border-color: ${({ theme }) => theme.colors.danger.borderHover};\n      box-shadow: ${({ theme }) => theme.shadows.focusDanger};\n    }\n\n    &[data-dropdownvisible='true'] {\n      border-color: ${({ theme }) => theme.colors.danger.borderHover};\n    }\n  }\n\n  &:not([data-disabled='true']):not([data-readonly]):hover {\n    border-color: ${({ theme }) => theme.colors.primary.border};\n  }\n\n  &[data-readonly='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n    border-color: ${({ theme }) => theme.colors.neutral.border};\n    cursor: default;\n  }\n\n  &[data-disabled='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    border-color: ${({ theme }) => theme.colors.neutral.borderDisabled};\n    cursor: not-allowed;\n  }\n`\n\nconst CustomTag = styled(Tag)`\n  height: fit-content;\n  width: fit-content;\n`\n\nconst SelectedValues = styled(Text)`\n  text-overflow: ellipsis;\n  overflow: hidden;\n  white-space: nowrap;\n`\n\nconst isValidSelectedValue = (selectedValue: string, options: DataType) =>\n  !Array.isArray(options)\n    ? Object.keys(options).some(group =>\n        options[group].some(\n          option => option.value === selectedValue && !option.disabled,\n        ),\n      )\n    : options.some(option => option.value === selectedValue && !option.disabled)\n\nconst DisplayValues = ({\n  refTag,\n  nonOverflowedValues,\n  disabled,\n  readOnly,\n  overflowed,\n  overflowAmount,\n  size,\n}: DisplayValuesProps) => {\n  const { multiselect, selectedData, setSelectedData, options, onChange } =\n    useSelectInput()\n\n  return multiselect ? (\n    <Stack\n      direction=\"row\"\n      gap=\"1\"\n      wrap=\"nowrap\"\n      ref={refTag}\n      alignItems=\"center\"\n    >\n      {nonOverflowedValues.map(option => (\n        <CustomTag\n          data-testid=\"selected-options-tags\"\n          key={option?.value}\n          sentiment=\"neutral\"\n          disabled={disabled}\n          onClose={\n            !readOnly\n              ? event => {\n                  event.stopPropagation()\n                  setSelectedData({\n                    type: 'selectOption',\n                    clickedOption: option,\n                  })\n                  const newSelectedValues = selectedData.selectedValues?.filter(\n                    val => val !== option.value,\n                  )\n                  onChange?.(newSelectedValues)\n                }\n              : undefined\n          }\n        >\n          {option?.label}\n        </CustomTag>\n      ))}\n      {overflowed ? (\n        <Tag\n          sentiment=\"neutral\"\n          disabled={disabled}\n          key=\"+\"\n          data-testid=\"plus-tag\"\n          aria-label=\"Plus tag\"\n        >\n          <PlusIcon />\n          {overflowAmount}\n        </Tag>\n      ) : null}\n    </Stack>\n  ) : (\n    <SelectedValues\n      as=\"div\"\n      variant={size === 'large' ? 'body' : 'bodySmall'}\n      disabled={disabled}\n      prominence=\"default\"\n      sentiment=\"neutral\"\n    >\n      {selectedData.selectedValues[0]\n        ? findOptionInOptions(options, selectedData.selectedValues[0])?.label\n        : null}\n    </SelectedValues>\n  )\n}\n\nexport const SelectBar = ({\n  size,\n  clearable,\n  disabled,\n  readOnly,\n  placeholder,\n  success,\n  error,\n  autoFocus,\n  tooltip,\n  innerRef,\n  id,\n  'data-testid': dataTestId,\n  label,\n}: SelectBarProps) => {\n  const {\n    isDropdownVisible,\n    onChange,\n    setIsDropdownVisible,\n    options,\n    selectedData,\n    setSelectedData,\n    multiselect,\n  } = useSelectInput()\n  const openable = !(readOnly || disabled)\n  const refTag = useRef<HTMLDivElement>(null)\n  const width = innerRef.current?.offsetWidth\n  const [overflowAmount, setOverflowAmount] = useState(0)\n  const [nonOverflowedValues, setNonOverFlowedValues] = useState<OptionType[]>(\n    () => {\n      if (selectedData.selectedValues[0]) {\n        const firstSelectOption = findOptionInOptions(\n          options,\n          selectedData.selectedValues[0],\n        )\n\n        return firstSelectOption ? [firstSelectOption] : []\n      }\n\n      return []\n    },\n  )\n\n  const state = useMemo(() => {\n    if (error) {\n      return 'danger'\n    }\n    if (success) {\n      return 'success'\n    }\n\n    return 'neutral'\n  }, [error, success])\n\n  useEffect(() => {\n    // When too many items are selected, too avoid overflow, compute the number of tags to display and add a + tag\n    let tagsWidth = 0\n    let computedOverflowAmount = 0\n    let computedNonOverflowedValues: OptionType[] = []\n    const newSelectedValues = selectedData.selectedValues.filter(\n      selectedValue => isValidSelectedValue(selectedValue, options),\n    )\n    for (const selectedValue of newSelectedValues) {\n      const selectedOption = findOptionInOptions(options, selectedValue)\n      if (\n        selectedOption?.label &&\n        width &&\n        isValidSelectedValue(selectedValue, options)\n      ) {\n        const lengthValue = selectedOption.value.length // Find a better way to find the number of displayed characters?\n        const totalTagWidth =\n          SIZES_TAG.tagWidth + SIZES_TAG.letterWidth * lengthValue\n        if (totalTagWidth + tagsWidth > width - 100) {\n          computedOverflowAmount += 1\n        } else {\n          computedNonOverflowedValues = [\n            ...computedNonOverflowedValues,\n            selectedOption,\n          ]\n          tagsWidth += totalTagWidth\n        }\n      }\n    }\n    setNonOverFlowedValues(computedNonOverflowedValues)\n    setOverflowAmount(computedOverflowAmount)\n  }, [options, selectedData.selectedValues, width])\n\n  useEffect(() => {\n    setSelectedData({ type: 'update' })\n  }, [setSelectedData, options])\n\n  const shouldDisplayValues = useMemo(() => {\n    if (multiselect) {\n      return (\n        nonOverflowedValues.length > 0 ||\n        selectedData.selectedValues.some(\n          selectedValue =>\n            findOptionInOptions(options, selectedValue) !== undefined,\n        )\n      )\n    }\n\n    return (\n      selectedData.selectedValues[0] !== undefined &&\n      findOptionInOptions(options, selectedData.selectedValues[0]) !== undefined\n    )\n  }, [\n    multiselect,\n    nonOverflowedValues.length,\n    options,\n    selectedData.selectedValues,\n  ])\n\n  return (\n    <Tooltip text={tooltip}>\n      <StyledInputWrapper\n        role=\"combobox\"\n        id={id}\n        data-disabled={disabled}\n        data-readonly={readOnly}\n        data-size={size}\n        data-dropdownvisible={isDropdownVisible}\n        data-state={state}\n        direction=\"row\"\n        wrap=\"nowrap\"\n        gap=\"1\"\n        justifyContent=\"space-between\"\n        alignItems=\"center\"\n        onClick={\n          openable ? () => setIsDropdownVisible(!isDropdownVisible) : undefined\n        }\n        data-testid={dataTestId}\n        autoFocus={autoFocus}\n        onKeyDown={event => {\n          if (event.key === 'ArrowDown') {\n            if (!isDropdownVisible) {\n              setIsDropdownVisible(true)\n            } else {\n              document.getElementById(`option-0`)?.focus()\n            }\n          }\n          if (event.key === ' ') {\n            event.preventDefault()\n          }\n\n          return ['Enter', ' '].includes(event.key) && openable\n            ? setIsDropdownVisible(!isDropdownVisible)\n            : null\n        }}\n        ref={innerRef}\n        aria-haspopup=\"listbox\"\n        aria-expanded={isDropdownVisible}\n        tabIndex={0}\n        aria-label={label}\n      >\n        {shouldDisplayValues ? (\n          <DisplayValues\n            refTag={refTag}\n            nonOverflowedValues={nonOverflowedValues}\n            disabled={disabled}\n            readOnly={readOnly}\n            overflowed={!!overflowAmount}\n            overflowAmount={overflowAmount}\n            size={size}\n          />\n        ) : (\n          <Placeholder\n            as=\"p\"\n            variant={size === 'large' ? 'body' : 'bodySmall'}\n            sentiment=\"neutral\"\n            disabled={disabled}\n            prominence=\"weak\"\n          >\n            {placeholder}\n          </Placeholder>\n        )}\n        <StateStack direction=\"row\" gap={1} alignItems=\"center\">\n          {error ? <AlertCircleIcon sentiment=\"danger\" /> : null}\n          {success && !error ? <CheckCircleIcon sentiment=\"success\" /> : null}\n          {clearable && selectedData.selectedValues.length > 0 ? (\n            <Button\n              aria-label=\"clear value\"\n              disabled={disabled || !selectedData.selectedValues[0] || readOnly}\n              variant=\"ghost\"\n              size=\"small\"\n              icon=\"close\"\n              onClick={event => {\n                event.stopPropagation()\n                setSelectedData({ type: 'clearAll' })\n                if (multiselect) {\n                  onChange?.([])\n                } else {\n                  onChange?.('')\n                }\n              }}\n              sentiment=\"neutral\"\n              data-testid=\"clear-all\"\n            />\n          ) : null}\n          <ArrowDownIcon\n            aria-label=\"show dropdown\"\n            size=\"small\"\n            sentiment=\"neutral\"\n            disabled={disabled || readOnly}\n          />\n        </StateStack>\n      </StyledInputWrapper>\n    </Tooltip>\n  )\n}\n"]} */",
35
35
  toString: _EMOTION_STRINGIFIED_CSS_ERROR__
36
36
  });
37
37
  const StyledInputWrapper = /* @__PURE__ */ _styled(Stack, process.env.NODE_ENV === "production" ? {
@@ -61,9 +61,9 @@ const StyledInputWrapper = /* @__PURE__ */ _styled(Stack, process.env.NODE_ENV =
61
61
  theme
62
62
  }) => theme.colors.primary.borderHover, ";box-shadow:", ({
63
63
  theme
64
- }) => theme.shadows.focusPrimary, ";}&:not([data-disabled='true']):hover,:not([data-disabled='true']):focus{border-color:", ({
64
+ }) => theme.shadows.focusPrimary, ";}&:not([data-disabled='true']):hover{border-color:", ({
65
65
  theme
66
- }) => theme.colors.primary.borderHover, ";outline:none;}&[data-dropdownvisible='true']{border-color:", ({
66
+ }) => theme.colors.primary.borderHover, ";outline:none;}&:not([data-disabled='true']):focus-visible{outline:5px auto Highlight;outline:5px auto -webkit-focus-ring-color;}&[data-dropdownvisible='true']{border-color:", ({
67
67
  theme
68
68
  }) => theme.colors.primary.borderHover, ";}}&[data-state='success']{border:1px solid ", ({
69
69
  theme
@@ -91,7 +91,7 @@ const StyledInputWrapper = /* @__PURE__ */ _styled(Stack, process.env.NODE_ENV =
91
91
  theme
92
92
  }) => theme.colors.neutral.backgroundDisabled, ";border-color:", ({
93
93
  theme
94
- }) => theme.colors.neutral.borderDisabled, ";cursor:not-allowed;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/SelectBar.tsx"],"names":[],"mappings":"AA4DE","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/SelectBar.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport {\n  AlertCircleIcon,\n  ArrowDownIcon,\n  CheckCircleIcon,\n  PlusIcon,\n} from '@ultraviolet/icons'\nimport type { RefObject } from 'react'\nimport { useEffect, useMemo, useRef, useState } from 'react'\nimport { Button } from '../Button'\nimport { Stack } from '../Stack'\nimport { Tag } from '../Tag'\nimport { Text } from '../Text'\nimport { Tooltip } from '../Tooltip'\nimport { useSelectInput } from './SelectInputProvider'\nimport { findOptionInOptions } from './findOptionInOptions'\nimport type { DataType, OptionType } from './types'\nimport { INPUT_SIZE_HEIGHT, SIZES_TAG } from './types'\n\ntype SelectBarProps = {\n  size: 'small' | 'medium' | 'large'\n  clearable: boolean\n  disabled: boolean\n  readOnly: boolean\n  placeholder: string\n  success?: string\n  error?: string\n  autoFocus?: boolean\n  innerRef: RefObject<HTMLDivElement | null>\n  id?: string\n  'data-testid': string\n  label?: string\n  tooltip?: string\n}\n\ntype DisplayValuesProps = {\n  refTag: RefObject<HTMLDivElement | null>\n  nonOverflowedValues: OptionType[]\n  disabled: boolean\n  readOnly: boolean\n  overflowed: boolean\n  overflowAmount: number\n  size: 'small' | 'medium' | 'large'\n}\n\nconst StateStack = styled(Stack)`\n  padding-right: ${({ theme }) => theme.space['2']};\n  display: flex;\n`\nconst Placeholder = styled(Text)`\nuser-select: none;\n`\n\nconst StyledInputWrapper = styled(Stack)<{\n  'data-readonly': boolean\n  'data-disabled': boolean\n  'data-size': 'small' | 'medium' | 'large'\n  'data-state': 'neutral' | 'success' | 'danger'\n  'data-dropdownvisible': boolean\n  'aria-label'?: string\n}>`\n  display: flex;\n  padding: ${({ theme }) => theme.space[1]};\n  padding-right: 0;\n  padding-left: ${({ theme }) => theme.space[2]};\n  cursor: pointer;\n  box-shadow: none;\n  background: ${({ theme }) => theme.colors.neutral.background};\n  border-radius: ${({ theme }) => theme.radii.default};\n  width: 100%;\n  overflow: hidden;\n\n  &[data-size='small'] {\n    height: ${({ theme }) => theme.sizing[INPUT_SIZE_HEIGHT.small]};\n    padding-left: ${({ theme }) => theme.space[1]};\n  }\n  &[data-size='medium'] {\n    height: ${({ theme }) => theme.sizing[INPUT_SIZE_HEIGHT.medium]};\n  }\n  &[data-size='large'] {\n    height: ${({ theme }) => theme.sizing[INPUT_SIZE_HEIGHT.large]};\n  }\n  &[data-state='neutral'] {\n    border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n\n    &:not([data-disabled=\"true\"]):not([data-readonly=\"true\"]):active {\n      border-color: ${({ theme }) => theme.colors.primary.borderHover};\n      box-shadow: ${({ theme }) => theme.shadows.focusPrimary};\n    }\n    &:not([data-disabled='true']):hover,\n    :not([data-disabled='true']):focus {\n      border-color: ${({ theme }) => theme.colors.primary.borderHover};\n      outline: none;\n    }\n\n    &[data-dropdownvisible='true'] {\n    border-color: ${({ theme }) => theme.colors.primary.borderHover};\n  }\n  }\n\n  &[data-state='success'] {\n    border: 1px solid ${({ theme }) => theme.colors.success.border};\n    &:not([data-disabled=\"true\"]):not([data-readonly=\"true\"]):active {\n      border-color: ${({ theme }) => theme.colors.success.borderHover};\n      box-shadow: ${({ theme }) => theme.shadows.focusSuccess};\n    }\n\n    &[data-dropdownvisible='true'] {\n      border-color: ${({ theme }) => theme.colors.success.borderHover};\n    }\n  }\n\n  &[data-state='danger'] {\n    border: 1px solid ${({ theme }) => theme.colors.danger.border};\n\n    &:not([data-disabled=\"true\"]):not([data-readonly=\"true\"]):active {\n      border-color: ${({ theme }) => theme.colors.danger.borderHover};\n      box-shadow: ${({ theme }) => theme.shadows.focusDanger};\n    }\n\n    &[data-dropdownvisible='true'] {\n      border-color: ${({ theme }) => theme.colors.danger.borderHover};\n    }\n  }\n\n  &:not([data-disabled='true']):not([data-readonly]):hover {\n    border-color: ${({ theme }) => theme.colors.primary.border};\n  }\n\n  &[data-readonly='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n    border-color: ${({ theme }) => theme.colors.neutral.border};\n    cursor: default;\n  }\n\n  &[data-disabled='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    border-color: ${({ theme }) => theme.colors.neutral.borderDisabled};\n    cursor: not-allowed;\n  }\n`\nconst CustomTag = styled(Tag)`\n  height: fit-content;\n  width: fit-content;\n`\nconst SelectedValues = styled(Text)`\ntext-overflow: ellipsis;\noverflow: hidden;\n`\n\nconst isValidSelectedValue = (selectedValue: string, options: DataType) =>\n  !Array.isArray(options)\n    ? Object.keys(options).some(group =>\n        options[group].some(\n          option => option.value === selectedValue && !option.disabled,\n        ),\n      )\n    : options.some(option => option.value === selectedValue && !option.disabled)\n\nconst DisplayValues = ({\n  refTag,\n  nonOverflowedValues,\n  disabled,\n  readOnly,\n  overflowed,\n  overflowAmount,\n  size,\n}: DisplayValuesProps) => {\n  const { multiselect, selectedData, setSelectedData, options, onChange } =\n    useSelectInput()\n\n  return multiselect ? (\n    <Stack\n      direction=\"row\"\n      gap=\"1\"\n      wrap=\"nowrap\"\n      ref={refTag}\n      alignItems=\"center\"\n    >\n      {nonOverflowedValues.map(option => (\n        <CustomTag\n          data-testid=\"selected-options-tags\"\n          key={option?.value}\n          sentiment=\"neutral\"\n          disabled={disabled}\n          onClose={\n            !readOnly\n              ? event => {\n                  event.stopPropagation()\n                  setSelectedData({\n                    type: 'selectOption',\n                    clickedOption: option,\n                  })\n                  const newSelectedValues = selectedData.selectedValues?.filter(\n                    val => val !== option.value,\n                  )\n                  onChange?.(newSelectedValues)\n                }\n              : undefined\n          }\n        >\n          {option?.label}\n        </CustomTag>\n      ))}\n      {overflowed ? (\n        <Tag\n          sentiment=\"neutral\"\n          disabled={disabled}\n          key=\"+\"\n          data-testid=\"plus-tag\"\n          aria-label=\"Plus tag\"\n        >\n          <PlusIcon />\n          {overflowAmount}\n        </Tag>\n      ) : null}\n    </Stack>\n  ) : (\n    <SelectedValues\n      as=\"div\"\n      variant={size === 'large' ? 'body' : 'bodySmall'}\n      disabled={disabled}\n      prominence=\"default\"\n      sentiment=\"neutral\"\n    >\n      {selectedData.selectedValues[0]\n        ? findOptionInOptions(options, selectedData.selectedValues[0])?.label\n        : null}\n    </SelectedValues>\n  )\n}\n\nexport const SelectBar = ({\n  size,\n  clearable,\n  disabled,\n  readOnly,\n  placeholder,\n  success,\n  error,\n  autoFocus,\n  tooltip,\n  innerRef,\n  id,\n  'data-testid': dataTestId,\n  label,\n}: SelectBarProps) => {\n  const {\n    isDropdownVisible,\n    onChange,\n    setIsDropdownVisible,\n    options,\n    selectedData,\n    setSelectedData,\n    multiselect,\n  } = useSelectInput()\n  const openable = !(readOnly || disabled)\n  const refTag = useRef<HTMLDivElement>(null)\n  const width = innerRef.current?.offsetWidth\n  const [overflowAmount, setOverflowAmount] = useState(0)\n  const [nonOverflowedValues, setNonOverFlowedValues] = useState<OptionType[]>(\n    () => {\n      if (selectedData.selectedValues[0]) {\n        const firstSelectOption = findOptionInOptions(\n          options,\n          selectedData.selectedValues[0],\n        )\n\n        return firstSelectOption ? [firstSelectOption] : []\n      }\n\n      return []\n    },\n  )\n\n  const state = useMemo(() => {\n    if (error) {\n      return 'danger'\n    }\n    if (success) {\n      return 'success'\n    }\n\n    return 'neutral'\n  }, [error, success])\n\n  useEffect(() => {\n    // When too many items are selected, too avoid overflow, compute the number of tags to display and add a + tag\n    let tagsWidth = 0\n    let computedOverflowAmount = 0\n    let computedNonOverflowedValues: OptionType[] = []\n    const newSelectedValues = selectedData.selectedValues.filter(\n      selectedValue => isValidSelectedValue(selectedValue, options),\n    )\n    for (const selectedValue of newSelectedValues) {\n      const selectedOption = findOptionInOptions(options, selectedValue)\n      if (\n        selectedOption?.label &&\n        width &&\n        isValidSelectedValue(selectedValue, options)\n      ) {\n        const lengthValue = selectedOption.value.length // Find a better way to find the number of displayed characters?\n        const totalTagWidth =\n          SIZES_TAG.tagWidth + SIZES_TAG.letterWidth * lengthValue\n        if (totalTagWidth + tagsWidth > width - 100) {\n          computedOverflowAmount += 1\n        } else {\n          computedNonOverflowedValues = [\n            ...computedNonOverflowedValues,\n            selectedOption,\n          ]\n          tagsWidth += totalTagWidth\n        }\n      }\n    }\n    setNonOverFlowedValues(computedNonOverflowedValues)\n    setOverflowAmount(computedOverflowAmount)\n  }, [options, selectedData.selectedValues, width])\n\n  useEffect(() => {\n    setSelectedData({ type: 'update' })\n  }, [setSelectedData, options])\n\n  const shouldDisplayValues = useMemo(() => {\n    if (multiselect) {\n      return (\n        nonOverflowedValues.length > 0 ||\n        selectedData.selectedValues.some(\n          selectedValue =>\n            findOptionInOptions(options, selectedValue) !== undefined,\n        )\n      )\n    }\n\n    return (\n      selectedData.selectedValues[0] !== undefined &&\n      findOptionInOptions(options, selectedData.selectedValues[0]) !== undefined\n    )\n  }, [\n    multiselect,\n    nonOverflowedValues.length,\n    options,\n    selectedData.selectedValues,\n  ])\n\n  return (\n    <Tooltip text={tooltip}>\n      <StyledInputWrapper\n        role=\"combobox\"\n        id={id}\n        data-disabled={disabled}\n        data-readonly={readOnly}\n        data-size={size}\n        data-dropdownvisible={isDropdownVisible}\n        data-state={state}\n        direction=\"row\"\n        wrap=\"nowrap\"\n        gap=\"1\"\n        justifyContent=\"space-between\"\n        alignItems=\"center\"\n        onClick={\n          openable ? () => setIsDropdownVisible(!isDropdownVisible) : undefined\n        }\n        data-testid={dataTestId}\n        autoFocus={autoFocus}\n        onKeyDown={event => {\n          if (event.key === 'ArrowDown') {\n            if (!isDropdownVisible) {\n              setIsDropdownVisible(true)\n            } else {\n              document.getElementById(`option-0`)?.focus()\n            }\n          }\n          if (event.key === ' ') {\n            event.preventDefault()\n          }\n\n          return ['Enter', ' '].includes(event.key) && openable\n            ? setIsDropdownVisible(!isDropdownVisible)\n            : null\n        }}\n        ref={innerRef}\n        aria-haspopup=\"listbox\"\n        aria-expanded={isDropdownVisible}\n        tabIndex={0}\n        aria-label={label}\n      >\n        {shouldDisplayValues ? (\n          <DisplayValues\n            refTag={refTag}\n            nonOverflowedValues={nonOverflowedValues}\n            disabled={disabled}\n            readOnly={readOnly}\n            overflowed={!!overflowAmount}\n            overflowAmount={overflowAmount}\n            size={size}\n          />\n        ) : (\n          <Placeholder\n            as=\"p\"\n            variant={size === 'large' ? 'body' : 'bodySmall'}\n            sentiment=\"neutral\"\n            disabled={disabled}\n            prominence=\"weak\"\n          >\n            {placeholder}\n          </Placeholder>\n        )}\n        <StateStack direction=\"row\" gap={1} alignItems=\"center\">\n          {error ? <AlertCircleIcon sentiment=\"danger\" /> : null}\n          {success && !error ? <CheckCircleIcon sentiment=\"success\" /> : null}\n          {clearable && selectedData.selectedValues.length > 0 ? (\n            <Button\n              aria-label=\"clear value\"\n              disabled={disabled || !selectedData.selectedValues[0] || readOnly}\n              variant=\"ghost\"\n              size=\"small\"\n              icon=\"close\"\n              onClick={event => {\n                event.stopPropagation()\n                setSelectedData({ type: 'clearAll' })\n                if (multiselect) {\n                  onChange?.([])\n                } else {\n                  onChange?.('')\n                }\n              }}\n              sentiment=\"neutral\"\n              data-testid=\"clear-all\"\n            />\n          ) : null}\n          <ArrowDownIcon\n            aria-label=\"show dropdown\"\n            size=\"small\"\n            sentiment=\"neutral\"\n            disabled={disabled || readOnly}\n          />\n        </StateStack>\n      </StyledInputWrapper>\n    </Tooltip>\n  )\n}\n"]} */"));
94
+ }) => theme.colors.neutral.borderDisabled, ";cursor:not-allowed;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/SelectBar.tsx"],"names":[],"mappings":"AA4DE","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/SelectBar.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport {\n  AlertCircleIcon,\n  ArrowDownIcon,\n  CheckCircleIcon,\n  PlusIcon,\n} from '@ultraviolet/icons'\nimport type { RefObject } from 'react'\nimport { useEffect, useMemo, useRef, useState } from 'react'\nimport { Button } from '../Button'\nimport { Stack } from '../Stack'\nimport { Tag } from '../Tag'\nimport { Text } from '../Text'\nimport { Tooltip } from '../Tooltip'\nimport { useSelectInput } from './SelectInputProvider'\nimport { findOptionInOptions } from './findOptionInOptions'\nimport type { DataType, OptionType } from './types'\nimport { INPUT_SIZE_HEIGHT, SIZES_TAG } from './types'\n\ntype SelectBarProps = {\n  size: 'small' | 'medium' | 'large'\n  clearable: boolean\n  disabled: boolean\n  readOnly: boolean\n  placeholder: string\n  success?: string\n  error?: string | boolean\n  autoFocus?: boolean\n  innerRef: RefObject<HTMLDivElement | null>\n  id?: string\n  'data-testid': string\n  label?: string\n  tooltip?: string\n}\n\ntype DisplayValuesProps = {\n  refTag: RefObject<HTMLDivElement | null>\n  nonOverflowedValues: OptionType[]\n  disabled: boolean\n  readOnly: boolean\n  overflowed: boolean\n  overflowAmount: number\n  size: 'small' | 'medium' | 'large'\n}\n\nconst StateStack = styled(Stack)`\n  padding-right: ${({ theme }) => theme.space['2']};\n  display: flex;\n`\nconst Placeholder = styled(Text)`\nuser-select: none;\n`\n\nexport const StyledInputWrapper = styled(Stack)<{\n  'data-readonly': boolean\n  'data-disabled': boolean\n  'data-size': 'small' | 'medium' | 'large'\n  'data-state': 'neutral' | 'success' | 'danger'\n  'data-dropdownvisible': boolean\n  'aria-label'?: string\n}>`\n  display: flex;\n  padding: ${({ theme }) => theme.space[1]};\n  padding-right: 0;\n  padding-left: ${({ theme }) => theme.space[2]};\n  cursor: pointer;\n  box-shadow: none;\n  background: ${({ theme }) => theme.colors.neutral.background};\n  border-radius: ${({ theme }) => theme.radii.default};\n  width: 100%;\n  overflow: hidden;\n\n  &[data-size='small'] {\n    height: ${({ theme }) => theme.sizing[INPUT_SIZE_HEIGHT.small]};\n    padding-left: ${({ theme }) => theme.space[1]};\n  }\n  &[data-size='medium'] {\n    height: ${({ theme }) => theme.sizing[INPUT_SIZE_HEIGHT.medium]};\n  }\n  &[data-size='large'] {\n    height: ${({ theme }) => theme.sizing[INPUT_SIZE_HEIGHT.large]};\n  }\n  &[data-state='neutral'] {\n    border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n\n    &:not([data-disabled=\"true\"]):not([data-readonly=\"true\"]):active {\n      border-color: ${({ theme }) => theme.colors.primary.borderHover};\n      box-shadow: ${({ theme }) => theme.shadows.focusPrimary};\n    }\n    &:not([data-disabled='true']):hover {\n      border-color: ${({ theme }) => theme.colors.primary.borderHover};\n      outline: none;\n    }\n\n    &:not([data-disabled='true']):focus-visible {\n      outline: 5px auto Highlight;\n      outline: 5px auto -webkit-focus-ring-color;\n    }\n\n    &[data-dropdownvisible='true'] {\n    border-color: ${({ theme }) => theme.colors.primary.borderHover};\n  }\n  }\n\n  &[data-state='success'] {\n    border: 1px solid ${({ theme }) => theme.colors.success.border};\n    &:not([data-disabled=\"true\"]):not([data-readonly=\"true\"]):active {\n      border-color: ${({ theme }) => theme.colors.success.borderHover};\n      box-shadow: ${({ theme }) => theme.shadows.focusSuccess};\n    }\n\n    &[data-dropdownvisible='true'] {\n      border-color: ${({ theme }) => theme.colors.success.borderHover};\n    }\n  }\n\n  &[data-state='danger'] {\n    border: 1px solid ${({ theme }) => theme.colors.danger.border};\n\n    &:not([data-disabled=\"true\"]):not([data-readonly=\"true\"]):active {\n      border-color: ${({ theme }) => theme.colors.danger.borderHover};\n      box-shadow: ${({ theme }) => theme.shadows.focusDanger};\n    }\n\n    &[data-dropdownvisible='true'] {\n      border-color: ${({ theme }) => theme.colors.danger.borderHover};\n    }\n  }\n\n  &:not([data-disabled='true']):not([data-readonly]):hover {\n    border-color: ${({ theme }) => theme.colors.primary.border};\n  }\n\n  &[data-readonly='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n    border-color: ${({ theme }) => theme.colors.neutral.border};\n    cursor: default;\n  }\n\n  &[data-disabled='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    border-color: ${({ theme }) => theme.colors.neutral.borderDisabled};\n    cursor: not-allowed;\n  }\n`\n\nconst CustomTag = styled(Tag)`\n  height: fit-content;\n  width: fit-content;\n`\n\nconst SelectedValues = styled(Text)`\n  text-overflow: ellipsis;\n  overflow: hidden;\n  white-space: nowrap;\n`\n\nconst isValidSelectedValue = (selectedValue: string, options: DataType) =>\n  !Array.isArray(options)\n    ? Object.keys(options).some(group =>\n        options[group].some(\n          option => option.value === selectedValue && !option.disabled,\n        ),\n      )\n    : options.some(option => option.value === selectedValue && !option.disabled)\n\nconst DisplayValues = ({\n  refTag,\n  nonOverflowedValues,\n  disabled,\n  readOnly,\n  overflowed,\n  overflowAmount,\n  size,\n}: DisplayValuesProps) => {\n  const { multiselect, selectedData, setSelectedData, options, onChange } =\n    useSelectInput()\n\n  return multiselect ? (\n    <Stack\n      direction=\"row\"\n      gap=\"1\"\n      wrap=\"nowrap\"\n      ref={refTag}\n      alignItems=\"center\"\n    >\n      {nonOverflowedValues.map(option => (\n        <CustomTag\n          data-testid=\"selected-options-tags\"\n          key={option?.value}\n          sentiment=\"neutral\"\n          disabled={disabled}\n          onClose={\n            !readOnly\n              ? event => {\n                  event.stopPropagation()\n                  setSelectedData({\n                    type: 'selectOption',\n                    clickedOption: option,\n                  })\n                  const newSelectedValues = selectedData.selectedValues?.filter(\n                    val => val !== option.value,\n                  )\n                  onChange?.(newSelectedValues)\n                }\n              : undefined\n          }\n        >\n          {option?.label}\n        </CustomTag>\n      ))}\n      {overflowed ? (\n        <Tag\n          sentiment=\"neutral\"\n          disabled={disabled}\n          key=\"+\"\n          data-testid=\"plus-tag\"\n          aria-label=\"Plus tag\"\n        >\n          <PlusIcon />\n          {overflowAmount}\n        </Tag>\n      ) : null}\n    </Stack>\n  ) : (\n    <SelectedValues\n      as=\"div\"\n      variant={size === 'large' ? 'body' : 'bodySmall'}\n      disabled={disabled}\n      prominence=\"default\"\n      sentiment=\"neutral\"\n    >\n      {selectedData.selectedValues[0]\n        ? findOptionInOptions(options, selectedData.selectedValues[0])?.label\n        : null}\n    </SelectedValues>\n  )\n}\n\nexport const SelectBar = ({\n  size,\n  clearable,\n  disabled,\n  readOnly,\n  placeholder,\n  success,\n  error,\n  autoFocus,\n  tooltip,\n  innerRef,\n  id,\n  'data-testid': dataTestId,\n  label,\n}: SelectBarProps) => {\n  const {\n    isDropdownVisible,\n    onChange,\n    setIsDropdownVisible,\n    options,\n    selectedData,\n    setSelectedData,\n    multiselect,\n  } = useSelectInput()\n  const openable = !(readOnly || disabled)\n  const refTag = useRef<HTMLDivElement>(null)\n  const width = innerRef.current?.offsetWidth\n  const [overflowAmount, setOverflowAmount] = useState(0)\n  const [nonOverflowedValues, setNonOverFlowedValues] = useState<OptionType[]>(\n    () => {\n      if (selectedData.selectedValues[0]) {\n        const firstSelectOption = findOptionInOptions(\n          options,\n          selectedData.selectedValues[0],\n        )\n\n        return firstSelectOption ? [firstSelectOption] : []\n      }\n\n      return []\n    },\n  )\n\n  const state = useMemo(() => {\n    if (error) {\n      return 'danger'\n    }\n    if (success) {\n      return 'success'\n    }\n\n    return 'neutral'\n  }, [error, success])\n\n  useEffect(() => {\n    // When too many items are selected, too avoid overflow, compute the number of tags to display and add a + tag\n    let tagsWidth = 0\n    let computedOverflowAmount = 0\n    let computedNonOverflowedValues: OptionType[] = []\n    const newSelectedValues = selectedData.selectedValues.filter(\n      selectedValue => isValidSelectedValue(selectedValue, options),\n    )\n    for (const selectedValue of newSelectedValues) {\n      const selectedOption = findOptionInOptions(options, selectedValue)\n      if (\n        selectedOption?.label &&\n        width &&\n        isValidSelectedValue(selectedValue, options)\n      ) {\n        const lengthValue = selectedOption.value.length // Find a better way to find the number of displayed characters?\n        const totalTagWidth =\n          SIZES_TAG.tagWidth + SIZES_TAG.letterWidth * lengthValue\n        if (totalTagWidth + tagsWidth > width - 100) {\n          computedOverflowAmount += 1\n        } else {\n          computedNonOverflowedValues = [\n            ...computedNonOverflowedValues,\n            selectedOption,\n          ]\n          tagsWidth += totalTagWidth\n        }\n      }\n    }\n    setNonOverFlowedValues(computedNonOverflowedValues)\n    setOverflowAmount(computedOverflowAmount)\n  }, [options, selectedData.selectedValues, width])\n\n  useEffect(() => {\n    setSelectedData({ type: 'update' })\n  }, [setSelectedData, options])\n\n  const shouldDisplayValues = useMemo(() => {\n    if (multiselect) {\n      return (\n        nonOverflowedValues.length > 0 ||\n        selectedData.selectedValues.some(\n          selectedValue =>\n            findOptionInOptions(options, selectedValue) !== undefined,\n        )\n      )\n    }\n\n    return (\n      selectedData.selectedValues[0] !== undefined &&\n      findOptionInOptions(options, selectedData.selectedValues[0]) !== undefined\n    )\n  }, [\n    multiselect,\n    nonOverflowedValues.length,\n    options,\n    selectedData.selectedValues,\n  ])\n\n  return (\n    <Tooltip text={tooltip}>\n      <StyledInputWrapper\n        role=\"combobox\"\n        id={id}\n        data-disabled={disabled}\n        data-readonly={readOnly}\n        data-size={size}\n        data-dropdownvisible={isDropdownVisible}\n        data-state={state}\n        direction=\"row\"\n        wrap=\"nowrap\"\n        gap=\"1\"\n        justifyContent=\"space-between\"\n        alignItems=\"center\"\n        onClick={\n          openable ? () => setIsDropdownVisible(!isDropdownVisible) : undefined\n        }\n        data-testid={dataTestId}\n        autoFocus={autoFocus}\n        onKeyDown={event => {\n          if (event.key === 'ArrowDown') {\n            if (!isDropdownVisible) {\n              setIsDropdownVisible(true)\n            } else {\n              document.getElementById(`option-0`)?.focus()\n            }\n          }\n          if (event.key === ' ') {\n            event.preventDefault()\n          }\n\n          return ['Enter', ' '].includes(event.key) && openable\n            ? setIsDropdownVisible(!isDropdownVisible)\n            : null\n        }}\n        ref={innerRef}\n        aria-haspopup=\"listbox\"\n        aria-expanded={isDropdownVisible}\n        tabIndex={0}\n        aria-label={label}\n      >\n        {shouldDisplayValues ? (\n          <DisplayValues\n            refTag={refTag}\n            nonOverflowedValues={nonOverflowedValues}\n            disabled={disabled}\n            readOnly={readOnly}\n            overflowed={!!overflowAmount}\n            overflowAmount={overflowAmount}\n            size={size}\n          />\n        ) : (\n          <Placeholder\n            as=\"p\"\n            variant={size === 'large' ? 'body' : 'bodySmall'}\n            sentiment=\"neutral\"\n            disabled={disabled}\n            prominence=\"weak\"\n          >\n            {placeholder}\n          </Placeholder>\n        )}\n        <StateStack direction=\"row\" gap={1} alignItems=\"center\">\n          {error ? <AlertCircleIcon sentiment=\"danger\" /> : null}\n          {success && !error ? <CheckCircleIcon sentiment=\"success\" /> : null}\n          {clearable && selectedData.selectedValues.length > 0 ? (\n            <Button\n              aria-label=\"clear value\"\n              disabled={disabled || !selectedData.selectedValues[0] || readOnly}\n              variant=\"ghost\"\n              size=\"small\"\n              icon=\"close\"\n              onClick={event => {\n                event.stopPropagation()\n                setSelectedData({ type: 'clearAll' })\n                if (multiselect) {\n                  onChange?.([])\n                } else {\n                  onChange?.('')\n                }\n              }}\n              sentiment=\"neutral\"\n              data-testid=\"clear-all\"\n            />\n          ) : null}\n          <ArrowDownIcon\n            aria-label=\"show dropdown\"\n            size=\"small\"\n            sentiment=\"neutral\"\n            disabled={disabled || readOnly}\n          />\n        </StateStack>\n      </StyledInputWrapper>\n    </Tooltip>\n  )\n}\n"]} */"));
95
95
  const CustomTag = /* @__PURE__ */ _styled(Tag, process.env.NODE_ENV === "production" ? {
96
96
  target: "ejy0aca1"
97
97
  } : {
@@ -102,7 +102,7 @@ const CustomTag = /* @__PURE__ */ _styled(Tag, process.env.NODE_ENV === "product
102
102
  styles: "height:fit-content;width:fit-content"
103
103
  } : {
104
104
  name: "1snt5jp",
105
- styles: "height:fit-content;width:fit-content/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/SelectBar.tsx"],"names":[],"mappings":"AA6I6B","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/SelectBar.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport {\n  AlertCircleIcon,\n  ArrowDownIcon,\n  CheckCircleIcon,\n  PlusIcon,\n} from '@ultraviolet/icons'\nimport type { RefObject } from 'react'\nimport { useEffect, useMemo, useRef, useState } from 'react'\nimport { Button } from '../Button'\nimport { Stack } from '../Stack'\nimport { Tag } from '../Tag'\nimport { Text } from '../Text'\nimport { Tooltip } from '../Tooltip'\nimport { useSelectInput } from './SelectInputProvider'\nimport { findOptionInOptions } from './findOptionInOptions'\nimport type { DataType, OptionType } from './types'\nimport { INPUT_SIZE_HEIGHT, SIZES_TAG } from './types'\n\ntype SelectBarProps = {\n  size: 'small' | 'medium' | 'large'\n  clearable: boolean\n  disabled: boolean\n  readOnly: boolean\n  placeholder: string\n  success?: string\n  error?: string\n  autoFocus?: boolean\n  innerRef: RefObject<HTMLDivElement | null>\n  id?: string\n  'data-testid': string\n  label?: string\n  tooltip?: string\n}\n\ntype DisplayValuesProps = {\n  refTag: RefObject<HTMLDivElement | null>\n  nonOverflowedValues: OptionType[]\n  disabled: boolean\n  readOnly: boolean\n  overflowed: boolean\n  overflowAmount: number\n  size: 'small' | 'medium' | 'large'\n}\n\nconst StateStack = styled(Stack)`\n  padding-right: ${({ theme }) => theme.space['2']};\n  display: flex;\n`\nconst Placeholder = styled(Text)`\nuser-select: none;\n`\n\nconst StyledInputWrapper = styled(Stack)<{\n  'data-readonly': boolean\n  'data-disabled': boolean\n  'data-size': 'small' | 'medium' | 'large'\n  'data-state': 'neutral' | 'success' | 'danger'\n  'data-dropdownvisible': boolean\n  'aria-label'?: string\n}>`\n  display: flex;\n  padding: ${({ theme }) => theme.space[1]};\n  padding-right: 0;\n  padding-left: ${({ theme }) => theme.space[2]};\n  cursor: pointer;\n  box-shadow: none;\n  background: ${({ theme }) => theme.colors.neutral.background};\n  border-radius: ${({ theme }) => theme.radii.default};\n  width: 100%;\n  overflow: hidden;\n\n  &[data-size='small'] {\n    height: ${({ theme }) => theme.sizing[INPUT_SIZE_HEIGHT.small]};\n    padding-left: ${({ theme }) => theme.space[1]};\n  }\n  &[data-size='medium'] {\n    height: ${({ theme }) => theme.sizing[INPUT_SIZE_HEIGHT.medium]};\n  }\n  &[data-size='large'] {\n    height: ${({ theme }) => theme.sizing[INPUT_SIZE_HEIGHT.large]};\n  }\n  &[data-state='neutral'] {\n    border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n\n    &:not([data-disabled=\"true\"]):not([data-readonly=\"true\"]):active {\n      border-color: ${({ theme }) => theme.colors.primary.borderHover};\n      box-shadow: ${({ theme }) => theme.shadows.focusPrimary};\n    }\n    &:not([data-disabled='true']):hover,\n    :not([data-disabled='true']):focus {\n      border-color: ${({ theme }) => theme.colors.primary.borderHover};\n      outline: none;\n    }\n\n    &[data-dropdownvisible='true'] {\n    border-color: ${({ theme }) => theme.colors.primary.borderHover};\n  }\n  }\n\n  &[data-state='success'] {\n    border: 1px solid ${({ theme }) => theme.colors.success.border};\n    &:not([data-disabled=\"true\"]):not([data-readonly=\"true\"]):active {\n      border-color: ${({ theme }) => theme.colors.success.borderHover};\n      box-shadow: ${({ theme }) => theme.shadows.focusSuccess};\n    }\n\n    &[data-dropdownvisible='true'] {\n      border-color: ${({ theme }) => theme.colors.success.borderHover};\n    }\n  }\n\n  &[data-state='danger'] {\n    border: 1px solid ${({ theme }) => theme.colors.danger.border};\n\n    &:not([data-disabled=\"true\"]):not([data-readonly=\"true\"]):active {\n      border-color: ${({ theme }) => theme.colors.danger.borderHover};\n      box-shadow: ${({ theme }) => theme.shadows.focusDanger};\n    }\n\n    &[data-dropdownvisible='true'] {\n      border-color: ${({ theme }) => theme.colors.danger.borderHover};\n    }\n  }\n\n  &:not([data-disabled='true']):not([data-readonly]):hover {\n    border-color: ${({ theme }) => theme.colors.primary.border};\n  }\n\n  &[data-readonly='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n    border-color: ${({ theme }) => theme.colors.neutral.border};\n    cursor: default;\n  }\n\n  &[data-disabled='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    border-color: ${({ theme }) => theme.colors.neutral.borderDisabled};\n    cursor: not-allowed;\n  }\n`\nconst CustomTag = styled(Tag)`\n  height: fit-content;\n  width: fit-content;\n`\nconst SelectedValues = styled(Text)`\ntext-overflow: ellipsis;\noverflow: hidden;\n`\n\nconst isValidSelectedValue = (selectedValue: string, options: DataType) =>\n  !Array.isArray(options)\n    ? Object.keys(options).some(group =>\n        options[group].some(\n          option => option.value === selectedValue && !option.disabled,\n        ),\n      )\n    : options.some(option => option.value === selectedValue && !option.disabled)\n\nconst DisplayValues = ({\n  refTag,\n  nonOverflowedValues,\n  disabled,\n  readOnly,\n  overflowed,\n  overflowAmount,\n  size,\n}: DisplayValuesProps) => {\n  const { multiselect, selectedData, setSelectedData, options, onChange } =\n    useSelectInput()\n\n  return multiselect ? (\n    <Stack\n      direction=\"row\"\n      gap=\"1\"\n      wrap=\"nowrap\"\n      ref={refTag}\n      alignItems=\"center\"\n    >\n      {nonOverflowedValues.map(option => (\n        <CustomTag\n          data-testid=\"selected-options-tags\"\n          key={option?.value}\n          sentiment=\"neutral\"\n          disabled={disabled}\n          onClose={\n            !readOnly\n              ? event => {\n                  event.stopPropagation()\n                  setSelectedData({\n                    type: 'selectOption',\n                    clickedOption: option,\n                  })\n                  const newSelectedValues = selectedData.selectedValues?.filter(\n                    val => val !== option.value,\n                  )\n                  onChange?.(newSelectedValues)\n                }\n              : undefined\n          }\n        >\n          {option?.label}\n        </CustomTag>\n      ))}\n      {overflowed ? (\n        <Tag\n          sentiment=\"neutral\"\n          disabled={disabled}\n          key=\"+\"\n          data-testid=\"plus-tag\"\n          aria-label=\"Plus tag\"\n        >\n          <PlusIcon />\n          {overflowAmount}\n        </Tag>\n      ) : null}\n    </Stack>\n  ) : (\n    <SelectedValues\n      as=\"div\"\n      variant={size === 'large' ? 'body' : 'bodySmall'}\n      disabled={disabled}\n      prominence=\"default\"\n      sentiment=\"neutral\"\n    >\n      {selectedData.selectedValues[0]\n        ? findOptionInOptions(options, selectedData.selectedValues[0])?.label\n        : null}\n    </SelectedValues>\n  )\n}\n\nexport const SelectBar = ({\n  size,\n  clearable,\n  disabled,\n  readOnly,\n  placeholder,\n  success,\n  error,\n  autoFocus,\n  tooltip,\n  innerRef,\n  id,\n  'data-testid': dataTestId,\n  label,\n}: SelectBarProps) => {\n  const {\n    isDropdownVisible,\n    onChange,\n    setIsDropdownVisible,\n    options,\n    selectedData,\n    setSelectedData,\n    multiselect,\n  } = useSelectInput()\n  const openable = !(readOnly || disabled)\n  const refTag = useRef<HTMLDivElement>(null)\n  const width = innerRef.current?.offsetWidth\n  const [overflowAmount, setOverflowAmount] = useState(0)\n  const [nonOverflowedValues, setNonOverFlowedValues] = useState<OptionType[]>(\n    () => {\n      if (selectedData.selectedValues[0]) {\n        const firstSelectOption = findOptionInOptions(\n          options,\n          selectedData.selectedValues[0],\n        )\n\n        return firstSelectOption ? [firstSelectOption] : []\n      }\n\n      return []\n    },\n  )\n\n  const state = useMemo(() => {\n    if (error) {\n      return 'danger'\n    }\n    if (success) {\n      return 'success'\n    }\n\n    return 'neutral'\n  }, [error, success])\n\n  useEffect(() => {\n    // When too many items are selected, too avoid overflow, compute the number of tags to display and add a + tag\n    let tagsWidth = 0\n    let computedOverflowAmount = 0\n    let computedNonOverflowedValues: OptionType[] = []\n    const newSelectedValues = selectedData.selectedValues.filter(\n      selectedValue => isValidSelectedValue(selectedValue, options),\n    )\n    for (const selectedValue of newSelectedValues) {\n      const selectedOption = findOptionInOptions(options, selectedValue)\n      if (\n        selectedOption?.label &&\n        width &&\n        isValidSelectedValue(selectedValue, options)\n      ) {\n        const lengthValue = selectedOption.value.length // Find a better way to find the number of displayed characters?\n        const totalTagWidth =\n          SIZES_TAG.tagWidth + SIZES_TAG.letterWidth * lengthValue\n        if (totalTagWidth + tagsWidth > width - 100) {\n          computedOverflowAmount += 1\n        } else {\n          computedNonOverflowedValues = [\n            ...computedNonOverflowedValues,\n            selectedOption,\n          ]\n          tagsWidth += totalTagWidth\n        }\n      }\n    }\n    setNonOverFlowedValues(computedNonOverflowedValues)\n    setOverflowAmount(computedOverflowAmount)\n  }, [options, selectedData.selectedValues, width])\n\n  useEffect(() => {\n    setSelectedData({ type: 'update' })\n  }, [setSelectedData, options])\n\n  const shouldDisplayValues = useMemo(() => {\n    if (multiselect) {\n      return (\n        nonOverflowedValues.length > 0 ||\n        selectedData.selectedValues.some(\n          selectedValue =>\n            findOptionInOptions(options, selectedValue) !== undefined,\n        )\n      )\n    }\n\n    return (\n      selectedData.selectedValues[0] !== undefined &&\n      findOptionInOptions(options, selectedData.selectedValues[0]) !== undefined\n    )\n  }, [\n    multiselect,\n    nonOverflowedValues.length,\n    options,\n    selectedData.selectedValues,\n  ])\n\n  return (\n    <Tooltip text={tooltip}>\n      <StyledInputWrapper\n        role=\"combobox\"\n        id={id}\n        data-disabled={disabled}\n        data-readonly={readOnly}\n        data-size={size}\n        data-dropdownvisible={isDropdownVisible}\n        data-state={state}\n        direction=\"row\"\n        wrap=\"nowrap\"\n        gap=\"1\"\n        justifyContent=\"space-between\"\n        alignItems=\"center\"\n        onClick={\n          openable ? () => setIsDropdownVisible(!isDropdownVisible) : undefined\n        }\n        data-testid={dataTestId}\n        autoFocus={autoFocus}\n        onKeyDown={event => {\n          if (event.key === 'ArrowDown') {\n            if (!isDropdownVisible) {\n              setIsDropdownVisible(true)\n            } else {\n              document.getElementById(`option-0`)?.focus()\n            }\n          }\n          if (event.key === ' ') {\n            event.preventDefault()\n          }\n\n          return ['Enter', ' '].includes(event.key) && openable\n            ? setIsDropdownVisible(!isDropdownVisible)\n            : null\n        }}\n        ref={innerRef}\n        aria-haspopup=\"listbox\"\n        aria-expanded={isDropdownVisible}\n        tabIndex={0}\n        aria-label={label}\n      >\n        {shouldDisplayValues ? (\n          <DisplayValues\n            refTag={refTag}\n            nonOverflowedValues={nonOverflowedValues}\n            disabled={disabled}\n            readOnly={readOnly}\n            overflowed={!!overflowAmount}\n            overflowAmount={overflowAmount}\n            size={size}\n          />\n        ) : (\n          <Placeholder\n            as=\"p\"\n            variant={size === 'large' ? 'body' : 'bodySmall'}\n            sentiment=\"neutral\"\n            disabled={disabled}\n            prominence=\"weak\"\n          >\n            {placeholder}\n          </Placeholder>\n        )}\n        <StateStack direction=\"row\" gap={1} alignItems=\"center\">\n          {error ? <AlertCircleIcon sentiment=\"danger\" /> : null}\n          {success && !error ? <CheckCircleIcon sentiment=\"success\" /> : null}\n          {clearable && selectedData.selectedValues.length > 0 ? (\n            <Button\n              aria-label=\"clear value\"\n              disabled={disabled || !selectedData.selectedValues[0] || readOnly}\n              variant=\"ghost\"\n              size=\"small\"\n              icon=\"close\"\n              onClick={event => {\n                event.stopPropagation()\n                setSelectedData({ type: 'clearAll' })\n                if (multiselect) {\n                  onChange?.([])\n                } else {\n                  onChange?.('')\n                }\n              }}\n              sentiment=\"neutral\"\n              data-testid=\"clear-all\"\n            />\n          ) : null}\n          <ArrowDownIcon\n            aria-label=\"show dropdown\"\n            size=\"small\"\n            sentiment=\"neutral\"\n            disabled={disabled || readOnly}\n          />\n        </StateStack>\n      </StyledInputWrapper>\n    </Tooltip>\n  )\n}\n"]} */",
105
+ styles: "height:fit-content;width:fit-content/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/SelectBar.tsx"],"names":[],"mappings":"AAkJ6B","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/SelectBar.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport {\n  AlertCircleIcon,\n  ArrowDownIcon,\n  CheckCircleIcon,\n  PlusIcon,\n} from '@ultraviolet/icons'\nimport type { RefObject } from 'react'\nimport { useEffect, useMemo, useRef, useState } from 'react'\nimport { Button } from '../Button'\nimport { Stack } from '../Stack'\nimport { Tag } from '../Tag'\nimport { Text } from '../Text'\nimport { Tooltip } from '../Tooltip'\nimport { useSelectInput } from './SelectInputProvider'\nimport { findOptionInOptions } from './findOptionInOptions'\nimport type { DataType, OptionType } from './types'\nimport { INPUT_SIZE_HEIGHT, SIZES_TAG } from './types'\n\ntype SelectBarProps = {\n  size: 'small' | 'medium' | 'large'\n  clearable: boolean\n  disabled: boolean\n  readOnly: boolean\n  placeholder: string\n  success?: string\n  error?: string | boolean\n  autoFocus?: boolean\n  innerRef: RefObject<HTMLDivElement | null>\n  id?: string\n  'data-testid': string\n  label?: string\n  tooltip?: string\n}\n\ntype DisplayValuesProps = {\n  refTag: RefObject<HTMLDivElement | null>\n  nonOverflowedValues: OptionType[]\n  disabled: boolean\n  readOnly: boolean\n  overflowed: boolean\n  overflowAmount: number\n  size: 'small' | 'medium' | 'large'\n}\n\nconst StateStack = styled(Stack)`\n  padding-right: ${({ theme }) => theme.space['2']};\n  display: flex;\n`\nconst Placeholder = styled(Text)`\nuser-select: none;\n`\n\nexport const StyledInputWrapper = styled(Stack)<{\n  'data-readonly': boolean\n  'data-disabled': boolean\n  'data-size': 'small' | 'medium' | 'large'\n  'data-state': 'neutral' | 'success' | 'danger'\n  'data-dropdownvisible': boolean\n  'aria-label'?: string\n}>`\n  display: flex;\n  padding: ${({ theme }) => theme.space[1]};\n  padding-right: 0;\n  padding-left: ${({ theme }) => theme.space[2]};\n  cursor: pointer;\n  box-shadow: none;\n  background: ${({ theme }) => theme.colors.neutral.background};\n  border-radius: ${({ theme }) => theme.radii.default};\n  width: 100%;\n  overflow: hidden;\n\n  &[data-size='small'] {\n    height: ${({ theme }) => theme.sizing[INPUT_SIZE_HEIGHT.small]};\n    padding-left: ${({ theme }) => theme.space[1]};\n  }\n  &[data-size='medium'] {\n    height: ${({ theme }) => theme.sizing[INPUT_SIZE_HEIGHT.medium]};\n  }\n  &[data-size='large'] {\n    height: ${({ theme }) => theme.sizing[INPUT_SIZE_HEIGHT.large]};\n  }\n  &[data-state='neutral'] {\n    border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n\n    &:not([data-disabled=\"true\"]):not([data-readonly=\"true\"]):active {\n      border-color: ${({ theme }) => theme.colors.primary.borderHover};\n      box-shadow: ${({ theme }) => theme.shadows.focusPrimary};\n    }\n    &:not([data-disabled='true']):hover {\n      border-color: ${({ theme }) => theme.colors.primary.borderHover};\n      outline: none;\n    }\n\n    &:not([data-disabled='true']):focus-visible {\n      outline: 5px auto Highlight;\n      outline: 5px auto -webkit-focus-ring-color;\n    }\n\n    &[data-dropdownvisible='true'] {\n    border-color: ${({ theme }) => theme.colors.primary.borderHover};\n  }\n  }\n\n  &[data-state='success'] {\n    border: 1px solid ${({ theme }) => theme.colors.success.border};\n    &:not([data-disabled=\"true\"]):not([data-readonly=\"true\"]):active {\n      border-color: ${({ theme }) => theme.colors.success.borderHover};\n      box-shadow: ${({ theme }) => theme.shadows.focusSuccess};\n    }\n\n    &[data-dropdownvisible='true'] {\n      border-color: ${({ theme }) => theme.colors.success.borderHover};\n    }\n  }\n\n  &[data-state='danger'] {\n    border: 1px solid ${({ theme }) => theme.colors.danger.border};\n\n    &:not([data-disabled=\"true\"]):not([data-readonly=\"true\"]):active {\n      border-color: ${({ theme }) => theme.colors.danger.borderHover};\n      box-shadow: ${({ theme }) => theme.shadows.focusDanger};\n    }\n\n    &[data-dropdownvisible='true'] {\n      border-color: ${({ theme }) => theme.colors.danger.borderHover};\n    }\n  }\n\n  &:not([data-disabled='true']):not([data-readonly]):hover {\n    border-color: ${({ theme }) => theme.colors.primary.border};\n  }\n\n  &[data-readonly='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n    border-color: ${({ theme }) => theme.colors.neutral.border};\n    cursor: default;\n  }\n\n  &[data-disabled='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    border-color: ${({ theme }) => theme.colors.neutral.borderDisabled};\n    cursor: not-allowed;\n  }\n`\n\nconst CustomTag = styled(Tag)`\n  height: fit-content;\n  width: fit-content;\n`\n\nconst SelectedValues = styled(Text)`\n  text-overflow: ellipsis;\n  overflow: hidden;\n  white-space: nowrap;\n`\n\nconst isValidSelectedValue = (selectedValue: string, options: DataType) =>\n  !Array.isArray(options)\n    ? Object.keys(options).some(group =>\n        options[group].some(\n          option => option.value === selectedValue && !option.disabled,\n        ),\n      )\n    : options.some(option => option.value === selectedValue && !option.disabled)\n\nconst DisplayValues = ({\n  refTag,\n  nonOverflowedValues,\n  disabled,\n  readOnly,\n  overflowed,\n  overflowAmount,\n  size,\n}: DisplayValuesProps) => {\n  const { multiselect, selectedData, setSelectedData, options, onChange } =\n    useSelectInput()\n\n  return multiselect ? (\n    <Stack\n      direction=\"row\"\n      gap=\"1\"\n      wrap=\"nowrap\"\n      ref={refTag}\n      alignItems=\"center\"\n    >\n      {nonOverflowedValues.map(option => (\n        <CustomTag\n          data-testid=\"selected-options-tags\"\n          key={option?.value}\n          sentiment=\"neutral\"\n          disabled={disabled}\n          onClose={\n            !readOnly\n              ? event => {\n                  event.stopPropagation()\n                  setSelectedData({\n                    type: 'selectOption',\n                    clickedOption: option,\n                  })\n                  const newSelectedValues = selectedData.selectedValues?.filter(\n                    val => val !== option.value,\n                  )\n                  onChange?.(newSelectedValues)\n                }\n              : undefined\n          }\n        >\n          {option?.label}\n        </CustomTag>\n      ))}\n      {overflowed ? (\n        <Tag\n          sentiment=\"neutral\"\n          disabled={disabled}\n          key=\"+\"\n          data-testid=\"plus-tag\"\n          aria-label=\"Plus tag\"\n        >\n          <PlusIcon />\n          {overflowAmount}\n        </Tag>\n      ) : null}\n    </Stack>\n  ) : (\n    <SelectedValues\n      as=\"div\"\n      variant={size === 'large' ? 'body' : 'bodySmall'}\n      disabled={disabled}\n      prominence=\"default\"\n      sentiment=\"neutral\"\n    >\n      {selectedData.selectedValues[0]\n        ? findOptionInOptions(options, selectedData.selectedValues[0])?.label\n        : null}\n    </SelectedValues>\n  )\n}\n\nexport const SelectBar = ({\n  size,\n  clearable,\n  disabled,\n  readOnly,\n  placeholder,\n  success,\n  error,\n  autoFocus,\n  tooltip,\n  innerRef,\n  id,\n  'data-testid': dataTestId,\n  label,\n}: SelectBarProps) => {\n  const {\n    isDropdownVisible,\n    onChange,\n    setIsDropdownVisible,\n    options,\n    selectedData,\n    setSelectedData,\n    multiselect,\n  } = useSelectInput()\n  const openable = !(readOnly || disabled)\n  const refTag = useRef<HTMLDivElement>(null)\n  const width = innerRef.current?.offsetWidth\n  const [overflowAmount, setOverflowAmount] = useState(0)\n  const [nonOverflowedValues, setNonOverFlowedValues] = useState<OptionType[]>(\n    () => {\n      if (selectedData.selectedValues[0]) {\n        const firstSelectOption = findOptionInOptions(\n          options,\n          selectedData.selectedValues[0],\n        )\n\n        return firstSelectOption ? [firstSelectOption] : []\n      }\n\n      return []\n    },\n  )\n\n  const state = useMemo(() => {\n    if (error) {\n      return 'danger'\n    }\n    if (success) {\n      return 'success'\n    }\n\n    return 'neutral'\n  }, [error, success])\n\n  useEffect(() => {\n    // When too many items are selected, too avoid overflow, compute the number of tags to display and add a + tag\n    let tagsWidth = 0\n    let computedOverflowAmount = 0\n    let computedNonOverflowedValues: OptionType[] = []\n    const newSelectedValues = selectedData.selectedValues.filter(\n      selectedValue => isValidSelectedValue(selectedValue, options),\n    )\n    for (const selectedValue of newSelectedValues) {\n      const selectedOption = findOptionInOptions(options, selectedValue)\n      if (\n        selectedOption?.label &&\n        width &&\n        isValidSelectedValue(selectedValue, options)\n      ) {\n        const lengthValue = selectedOption.value.length // Find a better way to find the number of displayed characters?\n        const totalTagWidth =\n          SIZES_TAG.tagWidth + SIZES_TAG.letterWidth * lengthValue\n        if (totalTagWidth + tagsWidth > width - 100) {\n          computedOverflowAmount += 1\n        } else {\n          computedNonOverflowedValues = [\n            ...computedNonOverflowedValues,\n            selectedOption,\n          ]\n          tagsWidth += totalTagWidth\n        }\n      }\n    }\n    setNonOverFlowedValues(computedNonOverflowedValues)\n    setOverflowAmount(computedOverflowAmount)\n  }, [options, selectedData.selectedValues, width])\n\n  useEffect(() => {\n    setSelectedData({ type: 'update' })\n  }, [setSelectedData, options])\n\n  const shouldDisplayValues = useMemo(() => {\n    if (multiselect) {\n      return (\n        nonOverflowedValues.length > 0 ||\n        selectedData.selectedValues.some(\n          selectedValue =>\n            findOptionInOptions(options, selectedValue) !== undefined,\n        )\n      )\n    }\n\n    return (\n      selectedData.selectedValues[0] !== undefined &&\n      findOptionInOptions(options, selectedData.selectedValues[0]) !== undefined\n    )\n  }, [\n    multiselect,\n    nonOverflowedValues.length,\n    options,\n    selectedData.selectedValues,\n  ])\n\n  return (\n    <Tooltip text={tooltip}>\n      <StyledInputWrapper\n        role=\"combobox\"\n        id={id}\n        data-disabled={disabled}\n        data-readonly={readOnly}\n        data-size={size}\n        data-dropdownvisible={isDropdownVisible}\n        data-state={state}\n        direction=\"row\"\n        wrap=\"nowrap\"\n        gap=\"1\"\n        justifyContent=\"space-between\"\n        alignItems=\"center\"\n        onClick={\n          openable ? () => setIsDropdownVisible(!isDropdownVisible) : undefined\n        }\n        data-testid={dataTestId}\n        autoFocus={autoFocus}\n        onKeyDown={event => {\n          if (event.key === 'ArrowDown') {\n            if (!isDropdownVisible) {\n              setIsDropdownVisible(true)\n            } else {\n              document.getElementById(`option-0`)?.focus()\n            }\n          }\n          if (event.key === ' ') {\n            event.preventDefault()\n          }\n\n          return ['Enter', ' '].includes(event.key) && openable\n            ? setIsDropdownVisible(!isDropdownVisible)\n            : null\n        }}\n        ref={innerRef}\n        aria-haspopup=\"listbox\"\n        aria-expanded={isDropdownVisible}\n        tabIndex={0}\n        aria-label={label}\n      >\n        {shouldDisplayValues ? (\n          <DisplayValues\n            refTag={refTag}\n            nonOverflowedValues={nonOverflowedValues}\n            disabled={disabled}\n            readOnly={readOnly}\n            overflowed={!!overflowAmount}\n            overflowAmount={overflowAmount}\n            size={size}\n          />\n        ) : (\n          <Placeholder\n            as=\"p\"\n            variant={size === 'large' ? 'body' : 'bodySmall'}\n            sentiment=\"neutral\"\n            disabled={disabled}\n            prominence=\"weak\"\n          >\n            {placeholder}\n          </Placeholder>\n        )}\n        <StateStack direction=\"row\" gap={1} alignItems=\"center\">\n          {error ? <AlertCircleIcon sentiment=\"danger\" /> : null}\n          {success && !error ? <CheckCircleIcon sentiment=\"success\" /> : null}\n          {clearable && selectedData.selectedValues.length > 0 ? (\n            <Button\n              aria-label=\"clear value\"\n              disabled={disabled || !selectedData.selectedValues[0] || readOnly}\n              variant=\"ghost\"\n              size=\"small\"\n              icon=\"close\"\n              onClick={event => {\n                event.stopPropagation()\n                setSelectedData({ type: 'clearAll' })\n                if (multiselect) {\n                  onChange?.([])\n                } else {\n                  onChange?.('')\n                }\n              }}\n              sentiment=\"neutral\"\n              data-testid=\"clear-all\"\n            />\n          ) : null}\n          <ArrowDownIcon\n            aria-label=\"show dropdown\"\n            size=\"small\"\n            sentiment=\"neutral\"\n            disabled={disabled || readOnly}\n          />\n        </StateStack>\n      </StyledInputWrapper>\n    </Tooltip>\n  )\n}\n"]} */",
106
106
  toString: _EMOTION_STRINGIFIED_CSS_ERROR__
107
107
  });
108
108
  const SelectedValues = /* @__PURE__ */ _styled(Text, process.env.NODE_ENV === "production" ? {
@@ -111,11 +111,11 @@ const SelectedValues = /* @__PURE__ */ _styled(Text, process.env.NODE_ENV === "p
111
111
  target: "ejy0aca0",
112
112
  label: "SelectedValues"
113
113
  })(process.env.NODE_ENV === "production" ? {
114
- name: "fzu03x",
115
- styles: "text-overflow:ellipsis;overflow:hidden"
114
+ name: "12wal98",
115
+ styles: "text-overflow:ellipsis;overflow:hidden;white-space:nowrap"
116
116
  } : {
117
- name: "fzu03x",
118
- styles: "text-overflow:ellipsis;overflow:hidden/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/SelectBar.tsx"],"names":[],"mappings":"AAiJmC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/SelectBar.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport {\n  AlertCircleIcon,\n  ArrowDownIcon,\n  CheckCircleIcon,\n  PlusIcon,\n} from '@ultraviolet/icons'\nimport type { RefObject } from 'react'\nimport { useEffect, useMemo, useRef, useState } from 'react'\nimport { Button } from '../Button'\nimport { Stack } from '../Stack'\nimport { Tag } from '../Tag'\nimport { Text } from '../Text'\nimport { Tooltip } from '../Tooltip'\nimport { useSelectInput } from './SelectInputProvider'\nimport { findOptionInOptions } from './findOptionInOptions'\nimport type { DataType, OptionType } from './types'\nimport { INPUT_SIZE_HEIGHT, SIZES_TAG } from './types'\n\ntype SelectBarProps = {\n  size: 'small' | 'medium' | 'large'\n  clearable: boolean\n  disabled: boolean\n  readOnly: boolean\n  placeholder: string\n  success?: string\n  error?: string\n  autoFocus?: boolean\n  innerRef: RefObject<HTMLDivElement | null>\n  id?: string\n  'data-testid': string\n  label?: string\n  tooltip?: string\n}\n\ntype DisplayValuesProps = {\n  refTag: RefObject<HTMLDivElement | null>\n  nonOverflowedValues: OptionType[]\n  disabled: boolean\n  readOnly: boolean\n  overflowed: boolean\n  overflowAmount: number\n  size: 'small' | 'medium' | 'large'\n}\n\nconst StateStack = styled(Stack)`\n  padding-right: ${({ theme }) => theme.space['2']};\n  display: flex;\n`\nconst Placeholder = styled(Text)`\nuser-select: none;\n`\n\nconst StyledInputWrapper = styled(Stack)<{\n  'data-readonly': boolean\n  'data-disabled': boolean\n  'data-size': 'small' | 'medium' | 'large'\n  'data-state': 'neutral' | 'success' | 'danger'\n  'data-dropdownvisible': boolean\n  'aria-label'?: string\n}>`\n  display: flex;\n  padding: ${({ theme }) => theme.space[1]};\n  padding-right: 0;\n  padding-left: ${({ theme }) => theme.space[2]};\n  cursor: pointer;\n  box-shadow: none;\n  background: ${({ theme }) => theme.colors.neutral.background};\n  border-radius: ${({ theme }) => theme.radii.default};\n  width: 100%;\n  overflow: hidden;\n\n  &[data-size='small'] {\n    height: ${({ theme }) => theme.sizing[INPUT_SIZE_HEIGHT.small]};\n    padding-left: ${({ theme }) => theme.space[1]};\n  }\n  &[data-size='medium'] {\n    height: ${({ theme }) => theme.sizing[INPUT_SIZE_HEIGHT.medium]};\n  }\n  &[data-size='large'] {\n    height: ${({ theme }) => theme.sizing[INPUT_SIZE_HEIGHT.large]};\n  }\n  &[data-state='neutral'] {\n    border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n\n    &:not([data-disabled=\"true\"]):not([data-readonly=\"true\"]):active {\n      border-color: ${({ theme }) => theme.colors.primary.borderHover};\n      box-shadow: ${({ theme }) => theme.shadows.focusPrimary};\n    }\n    &:not([data-disabled='true']):hover,\n    :not([data-disabled='true']):focus {\n      border-color: ${({ theme }) => theme.colors.primary.borderHover};\n      outline: none;\n    }\n\n    &[data-dropdownvisible='true'] {\n    border-color: ${({ theme }) => theme.colors.primary.borderHover};\n  }\n  }\n\n  &[data-state='success'] {\n    border: 1px solid ${({ theme }) => theme.colors.success.border};\n    &:not([data-disabled=\"true\"]):not([data-readonly=\"true\"]):active {\n      border-color: ${({ theme }) => theme.colors.success.borderHover};\n      box-shadow: ${({ theme }) => theme.shadows.focusSuccess};\n    }\n\n    &[data-dropdownvisible='true'] {\n      border-color: ${({ theme }) => theme.colors.success.borderHover};\n    }\n  }\n\n  &[data-state='danger'] {\n    border: 1px solid ${({ theme }) => theme.colors.danger.border};\n\n    &:not([data-disabled=\"true\"]):not([data-readonly=\"true\"]):active {\n      border-color: ${({ theme }) => theme.colors.danger.borderHover};\n      box-shadow: ${({ theme }) => theme.shadows.focusDanger};\n    }\n\n    &[data-dropdownvisible='true'] {\n      border-color: ${({ theme }) => theme.colors.danger.borderHover};\n    }\n  }\n\n  &:not([data-disabled='true']):not([data-readonly]):hover {\n    border-color: ${({ theme }) => theme.colors.primary.border};\n  }\n\n  &[data-readonly='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n    border-color: ${({ theme }) => theme.colors.neutral.border};\n    cursor: default;\n  }\n\n  &[data-disabled='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    border-color: ${({ theme }) => theme.colors.neutral.borderDisabled};\n    cursor: not-allowed;\n  }\n`\nconst CustomTag = styled(Tag)`\n  height: fit-content;\n  width: fit-content;\n`\nconst SelectedValues = styled(Text)`\ntext-overflow: ellipsis;\noverflow: hidden;\n`\n\nconst isValidSelectedValue = (selectedValue: string, options: DataType) =>\n  !Array.isArray(options)\n    ? Object.keys(options).some(group =>\n        options[group].some(\n          option => option.value === selectedValue && !option.disabled,\n        ),\n      )\n    : options.some(option => option.value === selectedValue && !option.disabled)\n\nconst DisplayValues = ({\n  refTag,\n  nonOverflowedValues,\n  disabled,\n  readOnly,\n  overflowed,\n  overflowAmount,\n  size,\n}: DisplayValuesProps) => {\n  const { multiselect, selectedData, setSelectedData, options, onChange } =\n    useSelectInput()\n\n  return multiselect ? (\n    <Stack\n      direction=\"row\"\n      gap=\"1\"\n      wrap=\"nowrap\"\n      ref={refTag}\n      alignItems=\"center\"\n    >\n      {nonOverflowedValues.map(option => (\n        <CustomTag\n          data-testid=\"selected-options-tags\"\n          key={option?.value}\n          sentiment=\"neutral\"\n          disabled={disabled}\n          onClose={\n            !readOnly\n              ? event => {\n                  event.stopPropagation()\n                  setSelectedData({\n                    type: 'selectOption',\n                    clickedOption: option,\n                  })\n                  const newSelectedValues = selectedData.selectedValues?.filter(\n                    val => val !== option.value,\n                  )\n                  onChange?.(newSelectedValues)\n                }\n              : undefined\n          }\n        >\n          {option?.label}\n        </CustomTag>\n      ))}\n      {overflowed ? (\n        <Tag\n          sentiment=\"neutral\"\n          disabled={disabled}\n          key=\"+\"\n          data-testid=\"plus-tag\"\n          aria-label=\"Plus tag\"\n        >\n          <PlusIcon />\n          {overflowAmount}\n        </Tag>\n      ) : null}\n    </Stack>\n  ) : (\n    <SelectedValues\n      as=\"div\"\n      variant={size === 'large' ? 'body' : 'bodySmall'}\n      disabled={disabled}\n      prominence=\"default\"\n      sentiment=\"neutral\"\n    >\n      {selectedData.selectedValues[0]\n        ? findOptionInOptions(options, selectedData.selectedValues[0])?.label\n        : null}\n    </SelectedValues>\n  )\n}\n\nexport const SelectBar = ({\n  size,\n  clearable,\n  disabled,\n  readOnly,\n  placeholder,\n  success,\n  error,\n  autoFocus,\n  tooltip,\n  innerRef,\n  id,\n  'data-testid': dataTestId,\n  label,\n}: SelectBarProps) => {\n  const {\n    isDropdownVisible,\n    onChange,\n    setIsDropdownVisible,\n    options,\n    selectedData,\n    setSelectedData,\n    multiselect,\n  } = useSelectInput()\n  const openable = !(readOnly || disabled)\n  const refTag = useRef<HTMLDivElement>(null)\n  const width = innerRef.current?.offsetWidth\n  const [overflowAmount, setOverflowAmount] = useState(0)\n  const [nonOverflowedValues, setNonOverFlowedValues] = useState<OptionType[]>(\n    () => {\n      if (selectedData.selectedValues[0]) {\n        const firstSelectOption = findOptionInOptions(\n          options,\n          selectedData.selectedValues[0],\n        )\n\n        return firstSelectOption ? [firstSelectOption] : []\n      }\n\n      return []\n    },\n  )\n\n  const state = useMemo(() => {\n    if (error) {\n      return 'danger'\n    }\n    if (success) {\n      return 'success'\n    }\n\n    return 'neutral'\n  }, [error, success])\n\n  useEffect(() => {\n    // When too many items are selected, too avoid overflow, compute the number of tags to display and add a + tag\n    let tagsWidth = 0\n    let computedOverflowAmount = 0\n    let computedNonOverflowedValues: OptionType[] = []\n    const newSelectedValues = selectedData.selectedValues.filter(\n      selectedValue => isValidSelectedValue(selectedValue, options),\n    )\n    for (const selectedValue of newSelectedValues) {\n      const selectedOption = findOptionInOptions(options, selectedValue)\n      if (\n        selectedOption?.label &&\n        width &&\n        isValidSelectedValue(selectedValue, options)\n      ) {\n        const lengthValue = selectedOption.value.length // Find a better way to find the number of displayed characters?\n        const totalTagWidth =\n          SIZES_TAG.tagWidth + SIZES_TAG.letterWidth * lengthValue\n        if (totalTagWidth + tagsWidth > width - 100) {\n          computedOverflowAmount += 1\n        } else {\n          computedNonOverflowedValues = [\n            ...computedNonOverflowedValues,\n            selectedOption,\n          ]\n          tagsWidth += totalTagWidth\n        }\n      }\n    }\n    setNonOverFlowedValues(computedNonOverflowedValues)\n    setOverflowAmount(computedOverflowAmount)\n  }, [options, selectedData.selectedValues, width])\n\n  useEffect(() => {\n    setSelectedData({ type: 'update' })\n  }, [setSelectedData, options])\n\n  const shouldDisplayValues = useMemo(() => {\n    if (multiselect) {\n      return (\n        nonOverflowedValues.length > 0 ||\n        selectedData.selectedValues.some(\n          selectedValue =>\n            findOptionInOptions(options, selectedValue) !== undefined,\n        )\n      )\n    }\n\n    return (\n      selectedData.selectedValues[0] !== undefined &&\n      findOptionInOptions(options, selectedData.selectedValues[0]) !== undefined\n    )\n  }, [\n    multiselect,\n    nonOverflowedValues.length,\n    options,\n    selectedData.selectedValues,\n  ])\n\n  return (\n    <Tooltip text={tooltip}>\n      <StyledInputWrapper\n        role=\"combobox\"\n        id={id}\n        data-disabled={disabled}\n        data-readonly={readOnly}\n        data-size={size}\n        data-dropdownvisible={isDropdownVisible}\n        data-state={state}\n        direction=\"row\"\n        wrap=\"nowrap\"\n        gap=\"1\"\n        justifyContent=\"space-between\"\n        alignItems=\"center\"\n        onClick={\n          openable ? () => setIsDropdownVisible(!isDropdownVisible) : undefined\n        }\n        data-testid={dataTestId}\n        autoFocus={autoFocus}\n        onKeyDown={event => {\n          if (event.key === 'ArrowDown') {\n            if (!isDropdownVisible) {\n              setIsDropdownVisible(true)\n            } else {\n              document.getElementById(`option-0`)?.focus()\n            }\n          }\n          if (event.key === ' ') {\n            event.preventDefault()\n          }\n\n          return ['Enter', ' '].includes(event.key) && openable\n            ? setIsDropdownVisible(!isDropdownVisible)\n            : null\n        }}\n        ref={innerRef}\n        aria-haspopup=\"listbox\"\n        aria-expanded={isDropdownVisible}\n        tabIndex={0}\n        aria-label={label}\n      >\n        {shouldDisplayValues ? (\n          <DisplayValues\n            refTag={refTag}\n            nonOverflowedValues={nonOverflowedValues}\n            disabled={disabled}\n            readOnly={readOnly}\n            overflowed={!!overflowAmount}\n            overflowAmount={overflowAmount}\n            size={size}\n          />\n        ) : (\n          <Placeholder\n            as=\"p\"\n            variant={size === 'large' ? 'body' : 'bodySmall'}\n            sentiment=\"neutral\"\n            disabled={disabled}\n            prominence=\"weak\"\n          >\n            {placeholder}\n          </Placeholder>\n        )}\n        <StateStack direction=\"row\" gap={1} alignItems=\"center\">\n          {error ? <AlertCircleIcon sentiment=\"danger\" /> : null}\n          {success && !error ? <CheckCircleIcon sentiment=\"success\" /> : null}\n          {clearable && selectedData.selectedValues.length > 0 ? (\n            <Button\n              aria-label=\"clear value\"\n              disabled={disabled || !selectedData.selectedValues[0] || readOnly}\n              variant=\"ghost\"\n              size=\"small\"\n              icon=\"close\"\n              onClick={event => {\n                event.stopPropagation()\n                setSelectedData({ type: 'clearAll' })\n                if (multiselect) {\n                  onChange?.([])\n                } else {\n                  onChange?.('')\n                }\n              }}\n              sentiment=\"neutral\"\n              data-testid=\"clear-all\"\n            />\n          ) : null}\n          <ArrowDownIcon\n            aria-label=\"show dropdown\"\n            size=\"small\"\n            sentiment=\"neutral\"\n            disabled={disabled || readOnly}\n          />\n        </StateStack>\n      </StyledInputWrapper>\n    </Tooltip>\n  )\n}\n"]} */",
117
+ name: "12wal98",
118
+ styles: "text-overflow:ellipsis;overflow:hidden;white-space:nowrap/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/SelectBar.tsx"],"names":[],"mappings":"AAuJmC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/SelectBar.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport {\n  AlertCircleIcon,\n  ArrowDownIcon,\n  CheckCircleIcon,\n  PlusIcon,\n} from '@ultraviolet/icons'\nimport type { RefObject } from 'react'\nimport { useEffect, useMemo, useRef, useState } from 'react'\nimport { Button } from '../Button'\nimport { Stack } from '../Stack'\nimport { Tag } from '../Tag'\nimport { Text } from '../Text'\nimport { Tooltip } from '../Tooltip'\nimport { useSelectInput } from './SelectInputProvider'\nimport { findOptionInOptions } from './findOptionInOptions'\nimport type { DataType, OptionType } from './types'\nimport { INPUT_SIZE_HEIGHT, SIZES_TAG } from './types'\n\ntype SelectBarProps = {\n  size: 'small' | 'medium' | 'large'\n  clearable: boolean\n  disabled: boolean\n  readOnly: boolean\n  placeholder: string\n  success?: string\n  error?: string | boolean\n  autoFocus?: boolean\n  innerRef: RefObject<HTMLDivElement | null>\n  id?: string\n  'data-testid': string\n  label?: string\n  tooltip?: string\n}\n\ntype DisplayValuesProps = {\n  refTag: RefObject<HTMLDivElement | null>\n  nonOverflowedValues: OptionType[]\n  disabled: boolean\n  readOnly: boolean\n  overflowed: boolean\n  overflowAmount: number\n  size: 'small' | 'medium' | 'large'\n}\n\nconst StateStack = styled(Stack)`\n  padding-right: ${({ theme }) => theme.space['2']};\n  display: flex;\n`\nconst Placeholder = styled(Text)`\nuser-select: none;\n`\n\nexport const StyledInputWrapper = styled(Stack)<{\n  'data-readonly': boolean\n  'data-disabled': boolean\n  'data-size': 'small' | 'medium' | 'large'\n  'data-state': 'neutral' | 'success' | 'danger'\n  'data-dropdownvisible': boolean\n  'aria-label'?: string\n}>`\n  display: flex;\n  padding: ${({ theme }) => theme.space[1]};\n  padding-right: 0;\n  padding-left: ${({ theme }) => theme.space[2]};\n  cursor: pointer;\n  box-shadow: none;\n  background: ${({ theme }) => theme.colors.neutral.background};\n  border-radius: ${({ theme }) => theme.radii.default};\n  width: 100%;\n  overflow: hidden;\n\n  &[data-size='small'] {\n    height: ${({ theme }) => theme.sizing[INPUT_SIZE_HEIGHT.small]};\n    padding-left: ${({ theme }) => theme.space[1]};\n  }\n  &[data-size='medium'] {\n    height: ${({ theme }) => theme.sizing[INPUT_SIZE_HEIGHT.medium]};\n  }\n  &[data-size='large'] {\n    height: ${({ theme }) => theme.sizing[INPUT_SIZE_HEIGHT.large]};\n  }\n  &[data-state='neutral'] {\n    border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n\n    &:not([data-disabled=\"true\"]):not([data-readonly=\"true\"]):active {\n      border-color: ${({ theme }) => theme.colors.primary.borderHover};\n      box-shadow: ${({ theme }) => theme.shadows.focusPrimary};\n    }\n    &:not([data-disabled='true']):hover {\n      border-color: ${({ theme }) => theme.colors.primary.borderHover};\n      outline: none;\n    }\n\n    &:not([data-disabled='true']):focus-visible {\n      outline: 5px auto Highlight;\n      outline: 5px auto -webkit-focus-ring-color;\n    }\n\n    &[data-dropdownvisible='true'] {\n    border-color: ${({ theme }) => theme.colors.primary.borderHover};\n  }\n  }\n\n  &[data-state='success'] {\n    border: 1px solid ${({ theme }) => theme.colors.success.border};\n    &:not([data-disabled=\"true\"]):not([data-readonly=\"true\"]):active {\n      border-color: ${({ theme }) => theme.colors.success.borderHover};\n      box-shadow: ${({ theme }) => theme.shadows.focusSuccess};\n    }\n\n    &[data-dropdownvisible='true'] {\n      border-color: ${({ theme }) => theme.colors.success.borderHover};\n    }\n  }\n\n  &[data-state='danger'] {\n    border: 1px solid ${({ theme }) => theme.colors.danger.border};\n\n    &:not([data-disabled=\"true\"]):not([data-readonly=\"true\"]):active {\n      border-color: ${({ theme }) => theme.colors.danger.borderHover};\n      box-shadow: ${({ theme }) => theme.shadows.focusDanger};\n    }\n\n    &[data-dropdownvisible='true'] {\n      border-color: ${({ theme }) => theme.colors.danger.borderHover};\n    }\n  }\n\n  &:not([data-disabled='true']):not([data-readonly]):hover {\n    border-color: ${({ theme }) => theme.colors.primary.border};\n  }\n\n  &[data-readonly='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n    border-color: ${({ theme }) => theme.colors.neutral.border};\n    cursor: default;\n  }\n\n  &[data-disabled='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    border-color: ${({ theme }) => theme.colors.neutral.borderDisabled};\n    cursor: not-allowed;\n  }\n`\n\nconst CustomTag = styled(Tag)`\n  height: fit-content;\n  width: fit-content;\n`\n\nconst SelectedValues = styled(Text)`\n  text-overflow: ellipsis;\n  overflow: hidden;\n  white-space: nowrap;\n`\n\nconst isValidSelectedValue = (selectedValue: string, options: DataType) =>\n  !Array.isArray(options)\n    ? Object.keys(options).some(group =>\n        options[group].some(\n          option => option.value === selectedValue && !option.disabled,\n        ),\n      )\n    : options.some(option => option.value === selectedValue && !option.disabled)\n\nconst DisplayValues = ({\n  refTag,\n  nonOverflowedValues,\n  disabled,\n  readOnly,\n  overflowed,\n  overflowAmount,\n  size,\n}: DisplayValuesProps) => {\n  const { multiselect, selectedData, setSelectedData, options, onChange } =\n    useSelectInput()\n\n  return multiselect ? (\n    <Stack\n      direction=\"row\"\n      gap=\"1\"\n      wrap=\"nowrap\"\n      ref={refTag}\n      alignItems=\"center\"\n    >\n      {nonOverflowedValues.map(option => (\n        <CustomTag\n          data-testid=\"selected-options-tags\"\n          key={option?.value}\n          sentiment=\"neutral\"\n          disabled={disabled}\n          onClose={\n            !readOnly\n              ? event => {\n                  event.stopPropagation()\n                  setSelectedData({\n                    type: 'selectOption',\n                    clickedOption: option,\n                  })\n                  const newSelectedValues = selectedData.selectedValues?.filter(\n                    val => val !== option.value,\n                  )\n                  onChange?.(newSelectedValues)\n                }\n              : undefined\n          }\n        >\n          {option?.label}\n        </CustomTag>\n      ))}\n      {overflowed ? (\n        <Tag\n          sentiment=\"neutral\"\n          disabled={disabled}\n          key=\"+\"\n          data-testid=\"plus-tag\"\n          aria-label=\"Plus tag\"\n        >\n          <PlusIcon />\n          {overflowAmount}\n        </Tag>\n      ) : null}\n    </Stack>\n  ) : (\n    <SelectedValues\n      as=\"div\"\n      variant={size === 'large' ? 'body' : 'bodySmall'}\n      disabled={disabled}\n      prominence=\"default\"\n      sentiment=\"neutral\"\n    >\n      {selectedData.selectedValues[0]\n        ? findOptionInOptions(options, selectedData.selectedValues[0])?.label\n        : null}\n    </SelectedValues>\n  )\n}\n\nexport const SelectBar = ({\n  size,\n  clearable,\n  disabled,\n  readOnly,\n  placeholder,\n  success,\n  error,\n  autoFocus,\n  tooltip,\n  innerRef,\n  id,\n  'data-testid': dataTestId,\n  label,\n}: SelectBarProps) => {\n  const {\n    isDropdownVisible,\n    onChange,\n    setIsDropdownVisible,\n    options,\n    selectedData,\n    setSelectedData,\n    multiselect,\n  } = useSelectInput()\n  const openable = !(readOnly || disabled)\n  const refTag = useRef<HTMLDivElement>(null)\n  const width = innerRef.current?.offsetWidth\n  const [overflowAmount, setOverflowAmount] = useState(0)\n  const [nonOverflowedValues, setNonOverFlowedValues] = useState<OptionType[]>(\n    () => {\n      if (selectedData.selectedValues[0]) {\n        const firstSelectOption = findOptionInOptions(\n          options,\n          selectedData.selectedValues[0],\n        )\n\n        return firstSelectOption ? [firstSelectOption] : []\n      }\n\n      return []\n    },\n  )\n\n  const state = useMemo(() => {\n    if (error) {\n      return 'danger'\n    }\n    if (success) {\n      return 'success'\n    }\n\n    return 'neutral'\n  }, [error, success])\n\n  useEffect(() => {\n    // When too many items are selected, too avoid overflow, compute the number of tags to display and add a + tag\n    let tagsWidth = 0\n    let computedOverflowAmount = 0\n    let computedNonOverflowedValues: OptionType[] = []\n    const newSelectedValues = selectedData.selectedValues.filter(\n      selectedValue => isValidSelectedValue(selectedValue, options),\n    )\n    for (const selectedValue of newSelectedValues) {\n      const selectedOption = findOptionInOptions(options, selectedValue)\n      if (\n        selectedOption?.label &&\n        width &&\n        isValidSelectedValue(selectedValue, options)\n      ) {\n        const lengthValue = selectedOption.value.length // Find a better way to find the number of displayed characters?\n        const totalTagWidth =\n          SIZES_TAG.tagWidth + SIZES_TAG.letterWidth * lengthValue\n        if (totalTagWidth + tagsWidth > width - 100) {\n          computedOverflowAmount += 1\n        } else {\n          computedNonOverflowedValues = [\n            ...computedNonOverflowedValues,\n            selectedOption,\n          ]\n          tagsWidth += totalTagWidth\n        }\n      }\n    }\n    setNonOverFlowedValues(computedNonOverflowedValues)\n    setOverflowAmount(computedOverflowAmount)\n  }, [options, selectedData.selectedValues, width])\n\n  useEffect(() => {\n    setSelectedData({ type: 'update' })\n  }, [setSelectedData, options])\n\n  const shouldDisplayValues = useMemo(() => {\n    if (multiselect) {\n      return (\n        nonOverflowedValues.length > 0 ||\n        selectedData.selectedValues.some(\n          selectedValue =>\n            findOptionInOptions(options, selectedValue) !== undefined,\n        )\n      )\n    }\n\n    return (\n      selectedData.selectedValues[0] !== undefined &&\n      findOptionInOptions(options, selectedData.selectedValues[0]) !== undefined\n    )\n  }, [\n    multiselect,\n    nonOverflowedValues.length,\n    options,\n    selectedData.selectedValues,\n  ])\n\n  return (\n    <Tooltip text={tooltip}>\n      <StyledInputWrapper\n        role=\"combobox\"\n        id={id}\n        data-disabled={disabled}\n        data-readonly={readOnly}\n        data-size={size}\n        data-dropdownvisible={isDropdownVisible}\n        data-state={state}\n        direction=\"row\"\n        wrap=\"nowrap\"\n        gap=\"1\"\n        justifyContent=\"space-between\"\n        alignItems=\"center\"\n        onClick={\n          openable ? () => setIsDropdownVisible(!isDropdownVisible) : undefined\n        }\n        data-testid={dataTestId}\n        autoFocus={autoFocus}\n        onKeyDown={event => {\n          if (event.key === 'ArrowDown') {\n            if (!isDropdownVisible) {\n              setIsDropdownVisible(true)\n            } else {\n              document.getElementById(`option-0`)?.focus()\n            }\n          }\n          if (event.key === ' ') {\n            event.preventDefault()\n          }\n\n          return ['Enter', ' '].includes(event.key) && openable\n            ? setIsDropdownVisible(!isDropdownVisible)\n            : null\n        }}\n        ref={innerRef}\n        aria-haspopup=\"listbox\"\n        aria-expanded={isDropdownVisible}\n        tabIndex={0}\n        aria-label={label}\n      >\n        {shouldDisplayValues ? (\n          <DisplayValues\n            refTag={refTag}\n            nonOverflowedValues={nonOverflowedValues}\n            disabled={disabled}\n            readOnly={readOnly}\n            overflowed={!!overflowAmount}\n            overflowAmount={overflowAmount}\n            size={size}\n          />\n        ) : (\n          <Placeholder\n            as=\"p\"\n            variant={size === 'large' ? 'body' : 'bodySmall'}\n            sentiment=\"neutral\"\n            disabled={disabled}\n            prominence=\"weak\"\n          >\n            {placeholder}\n          </Placeholder>\n        )}\n        <StateStack direction=\"row\" gap={1} alignItems=\"center\">\n          {error ? <AlertCircleIcon sentiment=\"danger\" /> : null}\n          {success && !error ? <CheckCircleIcon sentiment=\"success\" /> : null}\n          {clearable && selectedData.selectedValues.length > 0 ? (\n            <Button\n              aria-label=\"clear value\"\n              disabled={disabled || !selectedData.selectedValues[0] || readOnly}\n              variant=\"ghost\"\n              size=\"small\"\n              icon=\"close\"\n              onClick={event => {\n                event.stopPropagation()\n                setSelectedData({ type: 'clearAll' })\n                if (multiselect) {\n                  onChange?.([])\n                } else {\n                  onChange?.('')\n                }\n              }}\n              sentiment=\"neutral\"\n              data-testid=\"clear-all\"\n            />\n          ) : null}\n          <ArrowDownIcon\n            aria-label=\"show dropdown\"\n            size=\"small\"\n            sentiment=\"neutral\"\n            disabled={disabled || readOnly}\n          />\n        </StateStack>\n      </StyledInputWrapper>\n    </Tooltip>\n  )\n}\n"]} */",
119
119
  toString: _EMOTION_STRINGIFIED_CSS_ERROR__
120
120
  });
121
121
  const isValidSelectedValue = (selectedValue, options) => !Array.isArray(options) ? Object.keys(options).some((group) => options[group].some((option) => option.value === selectedValue && !option.disabled)) : options.some((option) => option.value === selectedValue && !option.disabled);
@@ -260,5 +260,6 @@ const SelectBar = ({
260
260
  ] }) });
261
261
  };
262
262
  export {
263
- SelectBar
263
+ SelectBar,
264
+ StyledInputWrapper
264
265
  };
@@ -24,7 +24,7 @@ const SelectInputContainer = /* @__PURE__ */ _styled__default.default("div", pro
24
24
  styles: "width:100%"
25
25
  } : {
26
26
  name: "1d3w5wq",
27
- styles: "width:100%/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/index.tsx"],"names":[],"mappings":"AA8HuC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { useId, useRef } from 'react'\nimport type { HTMLAttributes, ReactNode } from 'react'\nimport { Label } from '../Label'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\nimport { Dropdown } from './Dropdown'\nimport { SelectBar } from './SelectBar'\nimport { SelectInputProvider } from './SelectInputProvider'\nimport type { DataType } from './types'\n\ntype SelectInputV2Props<IsMulti extends undefined | boolean = false> = {\n  /**\n   * Input name\n   */\n  name: string\n  /**\n   * Place holder when no value defined\n   */\n  placeholder?: string\n  /**\n   * When searchable, placeholder when no value is searched\n   */\n  placeholderSearch?: string\n  /**\n   * Label of the component\n   */\n  label?: string\n  /**\n   * Helper text to give more information to the user\n   */\n  helper?: string\n  /**\n   * Selectable options\n   */\n  options: DataType\n  /**\n   * Message to show when no option available\n   */\n  emptyState?: ReactNode\n  /**\n   * Whether it is possible to search through the input\n   */\n  searchable?: boolean\n  /**\n   * Whether the component in disabled\n   */\n  disabled?: boolean\n  /**\n   * Whether the component in readOnly\n   */\n  readOnly?: boolean\n  /**\n   * Whether it is possible to clear the search input\n   */\n  clearable?: boolean\n  /**\n   * Size of the input\n   */\n  size?: 'small' | 'medium' | 'large'\n  /**\n   * Whether field is required\n   */\n  required?: boolean\n  /**\n   * More information regarding/description ofs the selectInput\n   */\n  labelDescription?: ReactNode\n  /**\n   * Whether option description is on the right of the option or under it\n   */\n  descriptionDirection?: 'row' | 'column'\n  /**\n   * Where to place the additional info prop\n   */\n  optionalInfoPlacement?: 'left' | 'right'\n  /**\n   * To add custom fixed elements at the bottom of the dropdown\n   */\n  footer?: ((closeDropdown: () => void) => ReactNode) | ReactNode\n  /**\n   * Display an error message under the select bar\n   */\n  error?: string\n  /**\n   * Display a success message under the select bar\n   */\n  success?: string\n  /**\n   * Load more button to implement lazy loading\n   */\n  loadMore?: ReactNode\n  /**\n   * When the options are loading, display a skeleton\n   */\n  isLoading?: boolean\n  /**\n   * Add a tooltip around the select bar\n   */\n  tooltip?: string\n  /**\n   * Adds an option to select every selectable options\n   */\n  selectAll?: { label: ReactNode; description?: string }\n  /**\n   * When options are group, define a option to select every selectable options of a group\n   */\n  selectAllGroup?: boolean\n  autofocus?: boolean\n  /**\n   * Whether it is possible to select multiple options\n   */\n  multiselect?: IsMulti\n  /**\n   * Default value, must be one of the options\n   */\n  value?: IsMulti extends true ? string[] : string\n  onChange?: IsMulti extends true\n    ? (value: string[]) => void\n    : (value: string) => void\n  'data-testid'?: string\n} & Pick<\n  HTMLAttributes<HTMLDivElement>,\n  'id' | 'onBlur' | 'onFocus' | 'aria-label' | 'className'\n>\n\nconst SelectInputContainer = styled.div`\n  width: 100%;\n`\nconst HelperText = styled(Text)`\n  padding-top: ${({ theme }) => theme.space['0.5']};\n`\n\n/**\n * SelectInputV2 component is used to select one or many elements from a selection.\n */\nexport const SelectInputV2 = <IsMulti extends undefined | boolean>({\n  name,\n  id,\n  onBlur,\n  onFocus,\n  onChange,\n  'aria-label': ariaLabel,\n  value,\n  label,\n  helper,\n  options,\n  size = 'large',\n  emptyState,\n  descriptionDirection = 'column',\n  success,\n  error,\n  'data-testid': dataTestId,\n  className,\n  tooltip,\n  footer,\n  placeholderSearch = 'Search in list',\n  placeholder = 'Select item',\n  searchable = true,\n  disabled = false,\n  readOnly = false,\n  clearable = false,\n  multiselect = false,\n  required = false,\n  labelDescription,\n  autofocus,\n  loadMore,\n  optionalInfoPlacement = 'right',\n  isLoading,\n  selectAll,\n  selectAllGroup = false,\n}: SelectInputV2Props<IsMulti>) => {\n  const localId = useId()\n  const finalId = id ?? localId\n  const ref = useRef<HTMLDivElement | null>(null)\n  const numberOfOptions = Array.isArray(options)\n    ? options.length\n    : Object.values(options).reduce(\n        (acc, current) =>\n          acc + current.filter(option => !option.disabled).length,\n        0,\n      )\n  const finalDataTestId = dataTestId ?? `select-input-${name}`\n\n  return (\n    <SelectInputProvider\n      options={options}\n      multiselect={multiselect}\n      selectAll={selectAll}\n      value={value}\n      selectAllGroup={selectAllGroup}\n      numberOfOptions={numberOfOptions}\n      onChange={onChange}\n      refSelect={ref}\n    >\n      <SelectInputContainer\n        onBlur={onBlur}\n        onFocus={onFocus}\n        className={className}\n        aria-label={name}\n      >\n        <Dropdown\n          emptyState={emptyState}\n          descriptionDirection={descriptionDirection}\n          searchable={searchable}\n          placeholder={placeholderSearch}\n          footer={footer}\n          refSelect={ref}\n          loadMore={loadMore}\n          optionalInfoPlacement={optionalInfoPlacement}\n          isLoading={isLoading}\n          size={size}\n        >\n          <Stack gap={0.5} aria-label={ariaLabel}>\n            {label || labelDescription ? (\n              <Label\n                htmlFor={finalId}\n                labelDescription={labelDescription}\n                required={required}\n                size={size}\n              >\n                {label}\n              </Label>\n            ) : null}\n            <SelectBar\n              size={size}\n              data-testid={finalDataTestId}\n              clearable={clearable}\n              readOnly={readOnly}\n              disabled={disabled}\n              placeholder={placeholder}\n              success={success}\n              error={error}\n              autoFocus={autofocus}\n              innerRef={ref}\n              id={finalId}\n              label={label}\n              tooltip={tooltip}\n            />\n          </Stack>\n        </Dropdown>\n        {!error && !success && helper ? (\n          <HelperText\n            variant=\"caption\"\n            as=\"p\"\n            sentiment=\"neutral\"\n            prominence=\"weak\"\n          >\n            {helper}\n          </HelperText>\n        ) : null}\n        {error || success ? (\n          <HelperText\n            variant=\"caption\"\n            as=\"p\"\n            sentiment={error ? 'danger' : 'success'}\n            prominence=\"default\"\n          >\n            {error || success}\n          </HelperText>\n        ) : null}\n      </SelectInputContainer>\n    </SelectInputProvider>\n  )\n}\n"]} */",
27
+ styles: "width:100%/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/index.tsx"],"names":[],"mappings":"AA8HuC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { useId, useRef } from 'react'\nimport type { HTMLAttributes, ReactNode } from 'react'\nimport { Label } from '../Label'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\nimport { Dropdown } from './Dropdown'\nimport { SelectBar } from './SelectBar'\nimport { SelectInputProvider } from './SelectInputProvider'\nimport type { DataType } from './types'\n\ntype SelectInputV2Props<IsMulti extends undefined | boolean = false> = {\n  /**\n   * Input name\n   */\n  name: string\n  /**\n   * Place holder when no value defined\n   */\n  placeholder?: string\n  /**\n   * When searchable, placeholder when no value is searched\n   */\n  placeholderSearch?: string\n  /**\n   * Label of the component\n   */\n  label?: string\n  /**\n   * Helper text to give more information to the user\n   */\n  helper?: string\n  /**\n   * Selectable options\n   */\n  options: DataType\n  /**\n   * Message to show when no option available\n   */\n  emptyState?: ReactNode\n  /**\n   * Whether it is possible to search through the input\n   */\n  searchable?: boolean\n  /**\n   * Whether the component in disabled\n   */\n  disabled?: boolean\n  /**\n   * Whether the component in readOnly\n   */\n  readOnly?: boolean\n  /**\n   * Whether it is possible to clear the search input\n   */\n  clearable?: boolean\n  /**\n   * Size of the input\n   */\n  size?: 'small' | 'medium' | 'large'\n  /**\n   * Whether field is required\n   */\n  required?: boolean\n  /**\n   * More information regarding/description ofs the selectInput\n   */\n  labelDescription?: ReactNode\n  /**\n   * Whether option description is on the right of the option or under it\n   */\n  descriptionDirection?: 'row' | 'column'\n  /**\n   * Where to place the additional info prop\n   */\n  optionalInfoPlacement?: 'left' | 'right'\n  /**\n   * To add custom fixed elements at the bottom of the dropdown\n   */\n  footer?: ((closeDropdown: () => void) => ReactNode) | ReactNode\n  /**\n   * Display an error message under the select bar\n   */\n  error?: string | boolean\n  /**\n   * Display a success message under the select bar\n   */\n  success?: string\n  /**\n   * Load more button to implement lazy loading\n   */\n  loadMore?: ReactNode\n  /**\n   * When the options are loading, display a skeleton\n   */\n  isLoading?: boolean\n  /**\n   * Add a tooltip around the select bar\n   */\n  tooltip?: string\n  /**\n   * Adds an option to select every selectable options\n   */\n  selectAll?: { label: ReactNode; description?: string }\n  /**\n   * When options are group, define a option to select every selectable options of a group\n   */\n  selectAllGroup?: boolean\n  autofocus?: boolean\n  /**\n   * Whether it is possible to select multiple options\n   */\n  multiselect?: IsMulti\n  /**\n   * Default value, must be one of the options\n   */\n  value?: IsMulti extends true ? string[] : string\n  onChange?: IsMulti extends true\n    ? (value: string[]) => void\n    : (value: string) => void\n  'data-testid'?: string\n} & Pick<\n  HTMLAttributes<HTMLDivElement>,\n  'id' | 'onBlur' | 'onFocus' | 'aria-label' | 'className'\n>\n\nconst SelectInputContainer = styled.div`\n  width: 100%;\n`\nconst HelperText = styled(Text)`\n  padding-top: ${({ theme }) => theme.space['0.5']};\n`\n\n/**\n * SelectInputV2 component is used to select one or many elements from a selection.\n */\nexport const SelectInputV2 = <IsMulti extends undefined | boolean>({\n  name,\n  id,\n  onBlur,\n  onFocus,\n  onChange,\n  'aria-label': ariaLabel,\n  value,\n  label,\n  helper,\n  options,\n  size = 'large',\n  emptyState,\n  descriptionDirection = 'column',\n  success,\n  error,\n  'data-testid': dataTestId,\n  className,\n  tooltip,\n  footer,\n  placeholderSearch = 'Search in list',\n  placeholder = 'Select item',\n  searchable = true,\n  disabled = false,\n  readOnly = false,\n  clearable = false,\n  multiselect = false,\n  required = false,\n  labelDescription,\n  autofocus,\n  loadMore,\n  optionalInfoPlacement = 'right',\n  isLoading,\n  selectAll,\n  selectAllGroup = false,\n}: SelectInputV2Props<IsMulti>) => {\n  const localId = useId()\n  const finalId = id ?? localId\n  const ref = useRef<HTMLDivElement | null>(null)\n  const numberOfOptions = Array.isArray(options)\n    ? options.length\n    : Object.values(options).reduce(\n        (acc, current) =>\n          acc + current.filter(option => !option.disabled).length,\n        0,\n      )\n  const finalDataTestId = dataTestId ?? `select-input-${name}`\n\n  return (\n    <SelectInputProvider\n      options={options}\n      multiselect={multiselect}\n      selectAll={selectAll}\n      value={value}\n      selectAllGroup={selectAllGroup}\n      numberOfOptions={numberOfOptions}\n      onChange={onChange}\n      refSelect={ref}\n    >\n      <SelectInputContainer\n        onBlur={onBlur}\n        onFocus={onFocus}\n        className={className}\n        aria-label={name}\n      >\n        <Dropdown\n          emptyState={emptyState}\n          descriptionDirection={descriptionDirection}\n          searchable={searchable}\n          placeholder={placeholderSearch}\n          footer={footer}\n          refSelect={ref}\n          loadMore={loadMore}\n          optionalInfoPlacement={optionalInfoPlacement}\n          isLoading={isLoading}\n          size={size}\n        >\n          <Stack gap={0.5} aria-label={ariaLabel}>\n            {label || labelDescription ? (\n              <Label\n                htmlFor={finalId}\n                labelDescription={labelDescription}\n                required={required}\n                size={size}\n              >\n                {label}\n              </Label>\n            ) : null}\n            <SelectBar\n              size={size}\n              data-testid={finalDataTestId}\n              clearable={clearable}\n              readOnly={readOnly}\n              disabled={disabled}\n              placeholder={placeholder}\n              success={success}\n              error={error}\n              autoFocus={autofocus}\n              innerRef={ref}\n              id={finalId}\n              label={label}\n              tooltip={tooltip}\n            />\n          </Stack>\n        </Dropdown>\n        {!error && !success && helper ? (\n          <HelperText\n            variant=\"caption\"\n            as=\"p\"\n            sentiment=\"neutral\"\n            prominence=\"weak\"\n          >\n            {helper}\n          </HelperText>\n        ) : null}\n        {(error && typeof error === 'string') || success ? (\n          <HelperText\n            variant=\"caption\"\n            as=\"p\"\n            sentiment={error ? 'danger' : 'success'}\n            prominence=\"default\"\n          >\n            {error || success}\n          </HelperText>\n        ) : null}\n      </SelectInputContainer>\n    </SelectInputProvider>\n  )\n}\n"]} */",
28
28
  toString: _EMOTION_STRINGIFIED_CSS_ERROR__
29
29
  });
30
30
  const HelperText = /* @__PURE__ */ _styled__default.default(index$2.Text, process.env.NODE_ENV === "production" ? {
@@ -34,7 +34,7 @@ const HelperText = /* @__PURE__ */ _styled__default.default(index$2.Text, proces
34
34
  label: "HelperText"
35
35
  })("padding-top:", ({
36
36
  theme
37
- }) => theme.space["0.5"], ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/index.tsx"],"names":[],"mappings":"AAiI+B","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { useId, useRef } from 'react'\nimport type { HTMLAttributes, ReactNode } from 'react'\nimport { Label } from '../Label'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\nimport { Dropdown } from './Dropdown'\nimport { SelectBar } from './SelectBar'\nimport { SelectInputProvider } from './SelectInputProvider'\nimport type { DataType } from './types'\n\ntype SelectInputV2Props<IsMulti extends undefined | boolean = false> = {\n  /**\n   * Input name\n   */\n  name: string\n  /**\n   * Place holder when no value defined\n   */\n  placeholder?: string\n  /**\n   * When searchable, placeholder when no value is searched\n   */\n  placeholderSearch?: string\n  /**\n   * Label of the component\n   */\n  label?: string\n  /**\n   * Helper text to give more information to the user\n   */\n  helper?: string\n  /**\n   * Selectable options\n   */\n  options: DataType\n  /**\n   * Message to show when no option available\n   */\n  emptyState?: ReactNode\n  /**\n   * Whether it is possible to search through the input\n   */\n  searchable?: boolean\n  /**\n   * Whether the component in disabled\n   */\n  disabled?: boolean\n  /**\n   * Whether the component in readOnly\n   */\n  readOnly?: boolean\n  /**\n   * Whether it is possible to clear the search input\n   */\n  clearable?: boolean\n  /**\n   * Size of the input\n   */\n  size?: 'small' | 'medium' | 'large'\n  /**\n   * Whether field is required\n   */\n  required?: boolean\n  /**\n   * More information regarding/description ofs the selectInput\n   */\n  labelDescription?: ReactNode\n  /**\n   * Whether option description is on the right of the option or under it\n   */\n  descriptionDirection?: 'row' | 'column'\n  /**\n   * Where to place the additional info prop\n   */\n  optionalInfoPlacement?: 'left' | 'right'\n  /**\n   * To add custom fixed elements at the bottom of the dropdown\n   */\n  footer?: ((closeDropdown: () => void) => ReactNode) | ReactNode\n  /**\n   * Display an error message under the select bar\n   */\n  error?: string\n  /**\n   * Display a success message under the select bar\n   */\n  success?: string\n  /**\n   * Load more button to implement lazy loading\n   */\n  loadMore?: ReactNode\n  /**\n   * When the options are loading, display a skeleton\n   */\n  isLoading?: boolean\n  /**\n   * Add a tooltip around the select bar\n   */\n  tooltip?: string\n  /**\n   * Adds an option to select every selectable options\n   */\n  selectAll?: { label: ReactNode; description?: string }\n  /**\n   * When options are group, define a option to select every selectable options of a group\n   */\n  selectAllGroup?: boolean\n  autofocus?: boolean\n  /**\n   * Whether it is possible to select multiple options\n   */\n  multiselect?: IsMulti\n  /**\n   * Default value, must be one of the options\n   */\n  value?: IsMulti extends true ? string[] : string\n  onChange?: IsMulti extends true\n    ? (value: string[]) => void\n    : (value: string) => void\n  'data-testid'?: string\n} & Pick<\n  HTMLAttributes<HTMLDivElement>,\n  'id' | 'onBlur' | 'onFocus' | 'aria-label' | 'className'\n>\n\nconst SelectInputContainer = styled.div`\n  width: 100%;\n`\nconst HelperText = styled(Text)`\n  padding-top: ${({ theme }) => theme.space['0.5']};\n`\n\n/**\n * SelectInputV2 component is used to select one or many elements from a selection.\n */\nexport const SelectInputV2 = <IsMulti extends undefined | boolean>({\n  name,\n  id,\n  onBlur,\n  onFocus,\n  onChange,\n  'aria-label': ariaLabel,\n  value,\n  label,\n  helper,\n  options,\n  size = 'large',\n  emptyState,\n  descriptionDirection = 'column',\n  success,\n  error,\n  'data-testid': dataTestId,\n  className,\n  tooltip,\n  footer,\n  placeholderSearch = 'Search in list',\n  placeholder = 'Select item',\n  searchable = true,\n  disabled = false,\n  readOnly = false,\n  clearable = false,\n  multiselect = false,\n  required = false,\n  labelDescription,\n  autofocus,\n  loadMore,\n  optionalInfoPlacement = 'right',\n  isLoading,\n  selectAll,\n  selectAllGroup = false,\n}: SelectInputV2Props<IsMulti>) => {\n  const localId = useId()\n  const finalId = id ?? localId\n  const ref = useRef<HTMLDivElement | null>(null)\n  const numberOfOptions = Array.isArray(options)\n    ? options.length\n    : Object.values(options).reduce(\n        (acc, current) =>\n          acc + current.filter(option => !option.disabled).length,\n        0,\n      )\n  const finalDataTestId = dataTestId ?? `select-input-${name}`\n\n  return (\n    <SelectInputProvider\n      options={options}\n      multiselect={multiselect}\n      selectAll={selectAll}\n      value={value}\n      selectAllGroup={selectAllGroup}\n      numberOfOptions={numberOfOptions}\n      onChange={onChange}\n      refSelect={ref}\n    >\n      <SelectInputContainer\n        onBlur={onBlur}\n        onFocus={onFocus}\n        className={className}\n        aria-label={name}\n      >\n        <Dropdown\n          emptyState={emptyState}\n          descriptionDirection={descriptionDirection}\n          searchable={searchable}\n          placeholder={placeholderSearch}\n          footer={footer}\n          refSelect={ref}\n          loadMore={loadMore}\n          optionalInfoPlacement={optionalInfoPlacement}\n          isLoading={isLoading}\n          size={size}\n        >\n          <Stack gap={0.5} aria-label={ariaLabel}>\n            {label || labelDescription ? (\n              <Label\n                htmlFor={finalId}\n                labelDescription={labelDescription}\n                required={required}\n                size={size}\n              >\n                {label}\n              </Label>\n            ) : null}\n            <SelectBar\n              size={size}\n              data-testid={finalDataTestId}\n              clearable={clearable}\n              readOnly={readOnly}\n              disabled={disabled}\n              placeholder={placeholder}\n              success={success}\n              error={error}\n              autoFocus={autofocus}\n              innerRef={ref}\n              id={finalId}\n              label={label}\n              tooltip={tooltip}\n            />\n          </Stack>\n        </Dropdown>\n        {!error && !success && helper ? (\n          <HelperText\n            variant=\"caption\"\n            as=\"p\"\n            sentiment=\"neutral\"\n            prominence=\"weak\"\n          >\n            {helper}\n          </HelperText>\n        ) : null}\n        {error || success ? (\n          <HelperText\n            variant=\"caption\"\n            as=\"p\"\n            sentiment={error ? 'danger' : 'success'}\n            prominence=\"default\"\n          >\n            {error || success}\n          </HelperText>\n        ) : null}\n      </SelectInputContainer>\n    </SelectInputProvider>\n  )\n}\n"]} */"));
37
+ }) => theme.space["0.5"], ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/index.tsx"],"names":[],"mappings":"AAiI+B","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { useId, useRef } from 'react'\nimport type { HTMLAttributes, ReactNode } from 'react'\nimport { Label } from '../Label'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\nimport { Dropdown } from './Dropdown'\nimport { SelectBar } from './SelectBar'\nimport { SelectInputProvider } from './SelectInputProvider'\nimport type { DataType } from './types'\n\ntype SelectInputV2Props<IsMulti extends undefined | boolean = false> = {\n  /**\n   * Input name\n   */\n  name: string\n  /**\n   * Place holder when no value defined\n   */\n  placeholder?: string\n  /**\n   * When searchable, placeholder when no value is searched\n   */\n  placeholderSearch?: string\n  /**\n   * Label of the component\n   */\n  label?: string\n  /**\n   * Helper text to give more information to the user\n   */\n  helper?: string\n  /**\n   * Selectable options\n   */\n  options: DataType\n  /**\n   * Message to show when no option available\n   */\n  emptyState?: ReactNode\n  /**\n   * Whether it is possible to search through the input\n   */\n  searchable?: boolean\n  /**\n   * Whether the component in disabled\n   */\n  disabled?: boolean\n  /**\n   * Whether the component in readOnly\n   */\n  readOnly?: boolean\n  /**\n   * Whether it is possible to clear the search input\n   */\n  clearable?: boolean\n  /**\n   * Size of the input\n   */\n  size?: 'small' | 'medium' | 'large'\n  /**\n   * Whether field is required\n   */\n  required?: boolean\n  /**\n   * More information regarding/description ofs the selectInput\n   */\n  labelDescription?: ReactNode\n  /**\n   * Whether option description is on the right of the option or under it\n   */\n  descriptionDirection?: 'row' | 'column'\n  /**\n   * Where to place the additional info prop\n   */\n  optionalInfoPlacement?: 'left' | 'right'\n  /**\n   * To add custom fixed elements at the bottom of the dropdown\n   */\n  footer?: ((closeDropdown: () => void) => ReactNode) | ReactNode\n  /**\n   * Display an error message under the select bar\n   */\n  error?: string | boolean\n  /**\n   * Display a success message under the select bar\n   */\n  success?: string\n  /**\n   * Load more button to implement lazy loading\n   */\n  loadMore?: ReactNode\n  /**\n   * When the options are loading, display a skeleton\n   */\n  isLoading?: boolean\n  /**\n   * Add a tooltip around the select bar\n   */\n  tooltip?: string\n  /**\n   * Adds an option to select every selectable options\n   */\n  selectAll?: { label: ReactNode; description?: string }\n  /**\n   * When options are group, define a option to select every selectable options of a group\n   */\n  selectAllGroup?: boolean\n  autofocus?: boolean\n  /**\n   * Whether it is possible to select multiple options\n   */\n  multiselect?: IsMulti\n  /**\n   * Default value, must be one of the options\n   */\n  value?: IsMulti extends true ? string[] : string\n  onChange?: IsMulti extends true\n    ? (value: string[]) => void\n    : (value: string) => void\n  'data-testid'?: string\n} & Pick<\n  HTMLAttributes<HTMLDivElement>,\n  'id' | 'onBlur' | 'onFocus' | 'aria-label' | 'className'\n>\n\nconst SelectInputContainer = styled.div`\n  width: 100%;\n`\nconst HelperText = styled(Text)`\n  padding-top: ${({ theme }) => theme.space['0.5']};\n`\n\n/**\n * SelectInputV2 component is used to select one or many elements from a selection.\n */\nexport const SelectInputV2 = <IsMulti extends undefined | boolean>({\n  name,\n  id,\n  onBlur,\n  onFocus,\n  onChange,\n  'aria-label': ariaLabel,\n  value,\n  label,\n  helper,\n  options,\n  size = 'large',\n  emptyState,\n  descriptionDirection = 'column',\n  success,\n  error,\n  'data-testid': dataTestId,\n  className,\n  tooltip,\n  footer,\n  placeholderSearch = 'Search in list',\n  placeholder = 'Select item',\n  searchable = true,\n  disabled = false,\n  readOnly = false,\n  clearable = false,\n  multiselect = false,\n  required = false,\n  labelDescription,\n  autofocus,\n  loadMore,\n  optionalInfoPlacement = 'right',\n  isLoading,\n  selectAll,\n  selectAllGroup = false,\n}: SelectInputV2Props<IsMulti>) => {\n  const localId = useId()\n  const finalId = id ?? localId\n  const ref = useRef<HTMLDivElement | null>(null)\n  const numberOfOptions = Array.isArray(options)\n    ? options.length\n    : Object.values(options).reduce(\n        (acc, current) =>\n          acc + current.filter(option => !option.disabled).length,\n        0,\n      )\n  const finalDataTestId = dataTestId ?? `select-input-${name}`\n\n  return (\n    <SelectInputProvider\n      options={options}\n      multiselect={multiselect}\n      selectAll={selectAll}\n      value={value}\n      selectAllGroup={selectAllGroup}\n      numberOfOptions={numberOfOptions}\n      onChange={onChange}\n      refSelect={ref}\n    >\n      <SelectInputContainer\n        onBlur={onBlur}\n        onFocus={onFocus}\n        className={className}\n        aria-label={name}\n      >\n        <Dropdown\n          emptyState={emptyState}\n          descriptionDirection={descriptionDirection}\n          searchable={searchable}\n          placeholder={placeholderSearch}\n          footer={footer}\n          refSelect={ref}\n          loadMore={loadMore}\n          optionalInfoPlacement={optionalInfoPlacement}\n          isLoading={isLoading}\n          size={size}\n        >\n          <Stack gap={0.5} aria-label={ariaLabel}>\n            {label || labelDescription ? (\n              <Label\n                htmlFor={finalId}\n                labelDescription={labelDescription}\n                required={required}\n                size={size}\n              >\n                {label}\n              </Label>\n            ) : null}\n            <SelectBar\n              size={size}\n              data-testid={finalDataTestId}\n              clearable={clearable}\n              readOnly={readOnly}\n              disabled={disabled}\n              placeholder={placeholder}\n              success={success}\n              error={error}\n              autoFocus={autofocus}\n              innerRef={ref}\n              id={finalId}\n              label={label}\n              tooltip={tooltip}\n            />\n          </Stack>\n        </Dropdown>\n        {!error && !success && helper ? (\n          <HelperText\n            variant=\"caption\"\n            as=\"p\"\n            sentiment=\"neutral\"\n            prominence=\"weak\"\n          >\n            {helper}\n          </HelperText>\n        ) : null}\n        {(error && typeof error === 'string') || success ? (\n          <HelperText\n            variant=\"caption\"\n            as=\"p\"\n            sentiment={error ? 'danger' : 'success'}\n            prominence=\"default\"\n          >\n            {error || success}\n          </HelperText>\n        ) : null}\n      </SelectInputContainer>\n    </SelectInputProvider>\n  )\n}\n"]} */"));
38
38
  const SelectInputV2 = ({
39
39
  name,
40
40
  id,
@@ -82,7 +82,7 @@ const SelectInputV2 = ({
82
82
  /* @__PURE__ */ jsxRuntime.jsx(SelectBar.SelectBar, { size, "data-testid": finalDataTestId, clearable, readOnly, disabled, placeholder, success, error, autoFocus: autofocus, innerRef: ref, id: finalId, label, tooltip })
83
83
  ] }) }),
84
84
  !error && !success && helper ? /* @__PURE__ */ jsxRuntime.jsx(HelperText, { variant: "caption", as: "p", sentiment: "neutral", prominence: "weak", children: helper }) : null,
85
- error || success ? /* @__PURE__ */ jsxRuntime.jsx(HelperText, { variant: "caption", as: "p", sentiment: error ? "danger" : "success", prominence: "default", children: error || success }) : null
85
+ error && typeof error === "string" || success ? /* @__PURE__ */ jsxRuntime.jsx(HelperText, { variant: "caption", as: "p", sentiment: error ? "danger" : "success", prominence: "default", children: error || success }) : null
86
86
  ] }) });
87
87
  };
88
88
  exports.SelectInputV2 = SelectInputV2;