@ultraviolet/ui 2.0.4 → 2.0.6

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.
@@ -2,7 +2,7 @@
2
2
  import { jsx, jsxs } from "@emotion/react/jsx-runtime";
3
3
  import _styled from "@emotion/styled/base";
4
4
  import { AlertCircleIcon, CheckCircleIcon, CloseIcon, ArrowDownIcon, PlusIcon } from "@ultraviolet/icons";
5
- import { useRef, useState, useMemo, useEffect } from "react";
5
+ import { useState, useRef, useMemo, useEffect, useLayoutEffect } from "react";
6
6
  import { Button } from "../Button/index.js";
7
7
  import { StyledChildrenContainer } from "../Popup/index.js";
8
8
  import { Stack } from "../Stack/index.js";
@@ -26,7 +26,7 @@ const StateStack = /* @__PURE__ */ _styled(Stack, process.env.NODE_ENV === "prod
26
26
  label: "StateStack"
27
27
  })("padding-right:", ({
28
28
  theme
29
- }) => 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/SelectInput/SelectBar.tsx"],"names":[],"mappings":"AA6DgC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInput/SelectBar.tsx","sourcesContent":["'use client'\n\nimport styled from '@emotion/styled'\nimport {\n  AlertCircleIcon,\n  ArrowDownIcon,\n  CheckCircleIcon,\n  CloseIcon,\n  PlusIcon,\n} from '@ultraviolet/icons'\nimport type { ReactNode, RefObject } from 'react'\nimport { useEffect, useMemo, useRef, useState } from 'react'\nimport { Button } from '../Button'\nimport { StyledChildrenContainer } from '../Popup'\nimport { Stack } from '../Stack'\nimport { Tag } from '../Tag'\nimport { Text } from '../Text'\nimport { Tooltip } from '../Tooltip'\nimport { findOptionInOptions } from './findOptionInOptions'\nimport { useSelectInput } from './SelectInputProvider'\nimport type { OptionType } from './types'\nimport { INPUT_SIZE_HEIGHT } from './types'\n\nconst SIZES_TAG = {\n  gap: 8,\n  paddings: 16,\n  plusTag: 48,\n}\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  dropdownId?: string\n}\n\ntype DisplayValuesProps = {\n  refTag: RefObject<HTMLDivElement | null>\n  nonOverflowedValues: OptionType[]\n  potentiallyNonOverflowedValues: OptionType[]\n  disabled: boolean\n  readOnly: boolean\n  overflowed: boolean\n  overflowAmount: number\n  measureRef: RefObject<HTMLDivElement | null>\n  size: 'small' | 'medium' | 'large'\n  lastElementMaxWidth: number\n  overflow?: boolean\n  refPlusTag: RefObject<HTMLDivElement | null>\n}\n\nconst StateStack = styled(Stack)`\n  padding-right: ${({ theme }) => theme.space['2']};\n  display: flex;\n`\nconst Placeholder = styled(Text)`\nuser-select: none;\nalign-self: center;\n`\n\nconst StyledInputWrapper = styled.div<{\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: grid;\n  width: 100%;\n  gap: ${({ theme }) => theme.space[1]};\n  grid-template-columns: 1fr auto ;\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\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  shouldForwardProp: prop => !['lastElementMaxWidth', 'hidden'].includes(prop),\n})<{\n  lastElementMaxWidth?: number\n  hidden?: boolean\n}>`\n  height: max-content;\n  width: fit-content;\n  min-width: ${({ lastElementMaxWidth }) =>\n    lastElementMaxWidth ? 'auto' : 'fit-content'};\n\n  max-width: ${({ lastElementMaxWidth, hidden }) =>\n    lastElementMaxWidth && !hidden ? `${lastElementMaxWidth}px` : '100%'};\n\n  ${({ hidden }) =>\n    hidden\n      ? 'visibility: hidden;'\n      : `\n  text-overflow: ellipsis;\n  overflow: hidden;`}\n\n  & > ${StyledChildrenContainer} {\n    overflow: hidden;\n  }\n`\n\nconst SelectedValues = styled(Text)`\n  text-overflow: ellipsis;\n  overflow: hidden;\n  white-space: nowrap;\n  align-self: center;\n`\nconst PlusTag = styled(Tag)`\nwidth: ${({ theme }) => theme.sizing[500]};\n;\n`\n\nconst MultiselectStack = styled(Stack)`\noverflow: hidden;\nmax-width: 100%;\nheight: 100%;\n`\n\nconst DisplayValues = ({\n  refTag,\n  nonOverflowedValues,\n  potentiallyNonOverflowedValues,\n  disabled,\n  readOnly,\n  overflowed,\n  overflowAmount,\n  size,\n  measureRef,\n  lastElementMaxWidth,\n  overflow,\n  refPlusTag,\n}: DisplayValuesProps) => {\n  const { multiselect, selectedData, setSelectedData, options, onChange } =\n    useSelectInput()\n\n  return multiselect ? (\n    <MultiselectStack\n      alignItems=\"center\"\n      direction=\"row\"\n      gap=\"1\"\n      ref={refTag}\n      wrap=\"nowrap\"\n    >\n      {/* Hidden div to measure the width of the tags */}\n      <div\n        ref={measureRef}\n        style={{\n          position: 'absolute',\n        }}\n      >\n        {potentiallyNonOverflowedValues.map(option => (\n          <CustomTag\n            className={option.value}\n            hidden\n            key={option.value}\n            onClose={() => {}}\n          >\n            {option?.label}\n          </CustomTag>\n        ))}\n      </div>\n      {nonOverflowedValues.map((option, index) => (\n        <CustomTag\n          data-testid=\"selected-options-tags\"\n          disabled={disabled}\n          key={option?.value}\n          lastElementMaxWidth={\n            index === nonOverflowedValues.length - 1 && overflow\n              ? lastElementMaxWidth\n              : 0\n          }\n          onClose={\n            !readOnly\n              ? event => {\n                  event.stopPropagation()\n                  setSelectedData({\n                    clickedOption: option,\n                    type: 'selectOption',\n                  })\n                  const newSelectedValues = selectedData.selectedValues?.filter(\n                    val => val !== option.value,\n                  )\n                  onChange?.(newSelectedValues)\n                }\n              : undefined\n          }\n          sentiment=\"neutral\"\n        >\n          {option?.label}\n        </CustomTag>\n      ))}\n\n      {overflowed ? (\n        <Stack justifyContent=\"center\" ref={refPlusTag}>\n          <PlusTag\n            aria-label=\"Plus tag\"\n            data-testid=\"plus-tag\"\n            disabled={disabled}\n            key=\"+\"\n            sentiment=\"neutral\"\n          >\n            <PlusIcon size=\"xsmall\" />\n            {overflowAmount}\n          </PlusTag>\n        </Stack>\n      ) : null}\n    </MultiselectStack>\n  ) : (\n    <SelectedValues\n      as=\"div\"\n      disabled={disabled}\n      prominence=\"default\"\n      sentiment=\"neutral\"\n      variant={size === 'large' ? 'body' : 'bodySmall'}\n    >\n      {selectedData.selectedValues[0]\n        ? findOptionInOptions(options, selectedData.selectedValues[0])?.label\n        : null}\n    </SelectedValues>\n  )\n}\n\nconst 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  dropdownId,\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 measureRef = useRef<HTMLDivElement>(null)\n  const arrowRef = useRef<HTMLDivElement>(null)\n  const refPlusTag = useRef<HTMLDivElement>(null)\n  // width - width of the arrow (in px) - padding between tags (in px)\n  const [innerWidth, setInnerWidth] = useState(\n    innerRef.current?.offsetWidth ??\n      0 - (arrowRef.current?.offsetWidth ?? 0) - SIZES_TAG.paddings,\n  )\n  const [overflowAmount, setOverflowAmount] = useState(0)\n  const [overflow, setOverflow] = useState(false)\n  const [lastElementMaxWidth, setLastElementMaxWidth] = 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 potentiallyNonOverflowedValues = useMemo(\n    () =>\n      selectedData.selectedValues\n        .map(selectedValue => findOptionInOptions(options, selectedValue))\n        .filter((option): option is OptionType => !!option),\n    [options, selectedData.selectedValues],\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    if (selectedData.selectedValues.length === 0) {\n      setOverflowAmount(0)\n      setNonOverFlowedValues([])\n    }\n    if (measureRef.current && selectedData.selectedValues.length > 0) {\n      const toMeasureElements: HTMLCollection = measureRef.current.children\n      const toMeasureElementsArray = [...toMeasureElements]\n\n      const {\n        measuredVisibleTags,\n        measuredHiddenTags,\n        accumulatedWidth,\n        lastVisibleElementWidth,\n        lastVisibleLabel,\n      } = toMeasureElementsArray.reduce(\n        (\n          accumulator: {\n            measuredVisibleTags: OptionType[]\n            measuredHiddenTags: number\n            accumulatedWidth: number\n            lastVisibleElementWidth: number\n            lastVisibleLabel: ReactNode\n          },\n          currentValue,\n          index,\n        ) => {\n          const elementWidth = (currentValue as HTMLDivElement).offsetWidth\n\n          const newAccumulatedWidth =\n            accumulator.accumulatedWidth + elementWidth + SIZES_TAG.gap\n\n          const canBeVisible = newAccumulatedWidth <= innerWidth\n\n          return {\n            accumulatedWidth: !canBeVisible\n              ? accumulator.accumulatedWidth\n              : newAccumulatedWidth,\n            lastVisibleElementWidth: canBeVisible\n              ? elementWidth\n              : accumulator.lastVisibleElementWidth,\n            lastVisibleLabel: canBeVisible\n              ? potentiallyNonOverflowedValues[index].label\n              : accumulator.lastVisibleLabel,\n            measuredHiddenTags:\n              accumulator.measuredHiddenTags + (!canBeVisible ? 1 : 0),\n            measuredVisibleTags: [\n              ...accumulator.measuredVisibleTags,\n              canBeVisible && potentiallyNonOverflowedValues[index],\n            ].filter(Boolean) as OptionType[],\n          }\n        },\n        {\n          accumulatedWidth: 0,\n          lastVisibleElementWidth: 0,\n          lastVisibleLabel: '',\n          measuredHiddenTags: 0,\n          measuredVisibleTags: [],\n        },\n      )\n\n      const additionnalElementsWidth =\n        SIZES_TAG.paddings + (refPlusTag.current?.offsetWidth ?? 0)\n      const finalWidth =\n        accumulatedWidth + (measuredHiddenTags ? additionnalElementsWidth : 0)\n\n      const overflowPx = finalWidth - innerWidth\n      const hasOverflow = overflowPx > 0\n      const hasHiddenTags = measuredHiddenTags > 0\n      const lastVisibleElementMaxSize = lastVisibleElementWidth - overflowPx\n\n      // If only one element is selected and it is hidden, we need to show it\n      if (measuredHiddenTags === 1 && measuredVisibleTags.length === 0) {\n        setOverflowAmount(0)\n        setNonOverFlowedValues([potentiallyNonOverflowedValues[0]])\n\n        const newOverflowPx =\n          lastVisibleElementWidth +\n          (measuredHiddenTags > 1 ? additionnalElementsWidth : 0) -\n          innerWidth\n        setLastElementMaxWidth(lastVisibleElementWidth - newOverflowPx)\n        setOverflow(true)\n      }\n\n      // If it overflows with the last tag, we need to add an ellipsis to the last element if there is enough space (>60px)\n      // and if it is a string (do not cut ReactNode label)\n      // else we hide it completely and add it to the overflow amount\n      else if (\n        hasOverflow &&\n        hasHiddenTags &&\n        (lastVisibleElementMaxSize > 65 ||\n          (measuredVisibleTags.length === 1 &&\n            lastVisibleElementMaxSize > 65)) &&\n        typeof lastVisibleLabel === 'string'\n      ) {\n        setLastElementMaxWidth(lastVisibleElementMaxSize)\n        setOverflow(true)\n        setOverflowAmount(measuredHiddenTags)\n        setNonOverFlowedValues(measuredVisibleTags)\n      } else if (hasOverflow && hasHiddenTags) {\n        setLastElementMaxWidth(0)\n        setOverflow(false)\n        setOverflowAmount(measuredHiddenTags + 1)\n        setNonOverFlowedValues(measuredVisibleTags.slice(0, -1))\n      }\n      // Otherwise, we have enough space to show all tags\n      else {\n        setOverflow(false)\n        setOverflowAmount(measuredHiddenTags)\n        setNonOverFlowedValues(measuredVisibleTags)\n      }\n    }\n  }, [\n    selectedData.selectedValues.length,\n    innerWidth,\n    potentiallyNonOverflowedValues,\n  ])\n\n  useEffect(() => {\n    setSelectedData({ type: 'update' })\n  }, [setSelectedData, options])\n\n  useEffect(() => {\n    const getWidth = () => {\n      if (refTag.current) {\n        setInnerWidth(refTag.current.offsetWidth)\n      } else {\n        setInnerWidth(\n          innerRef.current?.offsetWidth ??\n            0 - (arrowRef.current?.offsetWidth ?? 0) - SIZES_TAG.paddings,\n        )\n      }\n    }\n    getWidth()\n    window.addEventListener('resize', getWidth)\n\n    return () => window.removeEventListener('resize', getWidth)\n  }, [innerRef, refTag, selectedData.selectedValues])\n\n  const shouldDisplayValues = useMemo(() => {\n    if (multiselect) {\n      return (\n        potentiallyNonOverflowedValues.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    options,\n    potentiallyNonOverflowedValues.length,\n    selectedData.selectedValues,\n  ])\n\n  return (\n    <Tooltip text={tooltip}>\n      <StyledInputWrapper\n        aria-controls={dropdownId}\n        aria-expanded={isDropdownVisible}\n        aria-label={label}\n        autoFocus={autoFocus}\n        data-disabled={disabled}\n        data-dropdownvisible={isDropdownVisible}\n        data-readonly={readOnly}\n        data-size={size}\n        data-state={state}\n        data-testid={dataTestId}\n        id={id}\n        onClick={\n          openable ? () => setIsDropdownVisible(!isDropdownVisible) : undefined\n        }\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        role=\"combobox\"\n        tabIndex={0}\n      >\n        {shouldDisplayValues ? (\n          <DisplayValues\n            disabled={disabled}\n            lastElementMaxWidth={lastElementMaxWidth}\n            measureRef={measureRef}\n            nonOverflowedValues={nonOverflowedValues}\n            overflow={overflow}\n            overflowAmount={overflowAmount}\n            overflowed={!!overflowAmount}\n            potentiallyNonOverflowedValues={potentiallyNonOverflowedValues}\n            readOnly={readOnly}\n            refPlusTag={refPlusTag}\n            refTag={refTag}\n            size={size}\n          />\n        ) : (\n          <Placeholder\n            as=\"p\"\n            disabled={disabled}\n            prominence=\"weak\"\n            sentiment=\"neutral\"\n            variant={size === 'large' ? 'body' : 'bodySmall'}\n          >\n            {placeholder}\n          </Placeholder>\n        )}\n        <StateStack alignItems=\"center\" direction=\"row\" gap={1} ref={arrowRef}>\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              data-testid=\"clear-all\"\n              disabled={disabled || !selectedData.selectedValues[0] || readOnly}\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              size=\"small\"\n              variant=\"ghost\"\n            >\n              <CloseIcon />\n            </Button>\n          ) : null}\n          <ArrowDownIcon\n            aria-label=\"show dropdown\"\n            disabled={disabled || readOnly}\n            sentiment=\"neutral\"\n            size=\"small\"\n          />\n        </StateStack>\n      </StyledInputWrapper>\n    </Tooltip>\n  )\n}\n\nexport { SelectBar, StyledInputWrapper }\n"]} */"));
29
+ }) => 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/SelectInput/SelectBar.tsx"],"names":[],"mappings":"AA8DgC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInput/SelectBar.tsx","sourcesContent":["'use client'\n\nimport styled from '@emotion/styled'\nimport {\n  AlertCircleIcon,\n  ArrowDownIcon,\n  CheckCircleIcon,\n  CloseIcon,\n  PlusIcon,\n} from '@ultraviolet/icons'\nimport type { ReactNode, RefObject } from 'react'\nimport { useEffect, useLayoutEffect, useMemo, useRef, useState } from 'react'\nimport { Button } from '../Button'\nimport { StyledChildrenContainer } from '../Popup'\nimport { Stack } from '../Stack'\nimport { Tag } from '../Tag'\nimport { Text } from '../Text'\nimport { Tooltip } from '../Tooltip'\nimport { findOptionInOptions } from './findOptionInOptions'\nimport { useSelectInput } from './SelectInputProvider'\nimport type { OptionType } from './types'\nimport { INPUT_SIZE_HEIGHT } from './types'\n\nconst SIZES_TAG = {\n  gap: 8,\n  paddings: 16,\n  plusTag: 48,\n}\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  dropdownId?: string\n}\n\ntype DisplayValuesProps = {\n  refTag: RefObject<HTMLDivElement | null>\n  nonOverflowedValues: OptionType[]\n  potentiallyNonOverflowedValues: OptionType[]\n  disabled: boolean\n  readOnly: boolean\n  overflowed: boolean\n  overflowAmount: number\n  measureRef: RefObject<HTMLDivElement | null>\n  size: 'small' | 'medium' | 'large'\n  lastElementMaxWidth: number\n  overflow?: boolean\n  refPlusTag: RefObject<HTMLDivElement | null>\n  displayShadowCopy?: boolean\n}\n\nconst StateStack = styled(Stack)`\n  padding-right: ${({ theme }) => theme.space['2']};\n  display: flex;\n`\nconst Placeholder = styled(Text)`\nuser-select: none;\nalign-self: center;\n`\n\nconst StyledInputWrapper = styled.div<{\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: grid;\n  width: 100%;\n  gap: ${({ theme }) => theme.space[1]};\n  grid-template-columns: 1fr auto ;\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\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  shouldForwardProp: prop => !['lastElementMaxWidth', 'hidden'].includes(prop),\n})<{\n  lastElementMaxWidth?: number\n  hidden?: boolean\n}>`\n  height: max-content;\n  width: fit-content;\n  min-width: ${({ lastElementMaxWidth }) =>\n    lastElementMaxWidth ? 'auto' : 'fit-content'};\n\n  max-width: ${({ lastElementMaxWidth, hidden }) =>\n    lastElementMaxWidth && !hidden ? `${lastElementMaxWidth}px` : '100%'};\n\n  ${({ hidden }) =>\n    hidden\n      ? 'visibility: hidden;'\n      : `\n  text-overflow: ellipsis;\n  overflow: hidden;`}\n\n  & > ${StyledChildrenContainer} {\n    overflow: hidden;\n  }\n`\n\nconst SelectedValues = styled(Text)`\n  text-overflow: ellipsis;\n  overflow: hidden;\n  white-space: nowrap;\n  align-self: center;\n`\nconst PlusTag = styled(Tag)`\nwidth: ${({ theme }) => theme.sizing[500]};\n;\n`\n\nconst MultiselectStack = styled(Stack)`\noverflow: hidden;\nmax-width: 100%;\nheight: 100%;\n`\n\nconst DisplayValues = ({\n  refTag,\n  nonOverflowedValues,\n  potentiallyNonOverflowedValues,\n  disabled,\n  readOnly,\n  overflowed,\n  overflowAmount,\n  size,\n  measureRef,\n  lastElementMaxWidth,\n  overflow,\n  refPlusTag,\n  displayShadowCopy,\n}: DisplayValuesProps) => {\n  const { multiselect, selectedData, setSelectedData, options, onChange } =\n    useSelectInput()\n\n  return multiselect ? (\n    <MultiselectStack\n      alignItems=\"center\"\n      direction=\"row\"\n      gap=\"1\"\n      ref={refTag}\n      wrap=\"nowrap\"\n    >\n      {/* Hidden div to measure the width of the tags */}\n      {displayShadowCopy ? (\n        <div\n          ref={measureRef}\n          style={{\n            position: 'absolute',\n          }}\n        >\n          {potentiallyNonOverflowedValues.map(option => (\n            <CustomTag\n              className={option.value}\n              hidden\n              key={option.value}\n              onClose={() => {}}\n            >\n              {option?.label}\n            </CustomTag>\n          ))}\n        </div>\n      ) : null}\n      {nonOverflowedValues.map((option, index) => (\n        <CustomTag\n          data-testid=\"selected-options-tags\"\n          disabled={disabled}\n          key={option?.value}\n          lastElementMaxWidth={\n            index === nonOverflowedValues.length - 1 && overflow\n              ? lastElementMaxWidth\n              : 0\n          }\n          onClose={\n            !readOnly\n              ? event => {\n                  event.stopPropagation()\n                  setSelectedData({\n                    clickedOption: option,\n                    type: 'selectOption',\n                  })\n                  const newSelectedValues = selectedData.selectedValues?.filter(\n                    val => val !== option.value,\n                  )\n                  onChange?.(newSelectedValues)\n                }\n              : undefined\n          }\n          sentiment=\"neutral\"\n        >\n          {option?.label}\n        </CustomTag>\n      ))}\n\n      {overflowed ? (\n        <Stack justifyContent=\"center\" ref={refPlusTag}>\n          <PlusTag\n            aria-label=\"Plus tag\"\n            data-testid=\"plus-tag\"\n            disabled={disabled}\n            key=\"+\"\n            sentiment=\"neutral\"\n          >\n            <PlusIcon size=\"xsmall\" />\n            {overflowAmount}\n          </PlusTag>\n        </Stack>\n      ) : null}\n    </MultiselectStack>\n  ) : (\n    <SelectedValues\n      as=\"div\"\n      disabled={disabled}\n      prominence=\"default\"\n      sentiment=\"neutral\"\n      variant={size === 'large' ? 'body' : 'bodySmall'}\n    >\n      {selectedData.selectedValues[0]\n        ? findOptionInOptions(options, selectedData.selectedValues[0])?.label\n        : null}\n    </SelectedValues>\n  )\n}\n\nconst 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  dropdownId,\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 [displayShadowCopy, setDisplayShadowCopy] = useState(false)\n  const refTag = useRef<HTMLDivElement>(null)\n  const measureRef = useRef<HTMLDivElement>(null)\n  const arrowRef = useRef<HTMLDivElement>(null)\n  const refPlusTag = useRef<HTMLDivElement>(null)\n  // width - width of the arrow (in px) - padding between tags (in px)\n  const [innerWidth, setInnerWidth] = useState(\n    innerRef.current?.offsetWidth ??\n      0 - (arrowRef.current?.offsetWidth ?? 0) - SIZES_TAG.paddings,\n  )\n  const [overflowAmount, setOverflowAmount] = useState(0)\n  const [overflow, setOverflow] = useState(false)\n  const [lastElementMaxWidth, setLastElementMaxWidth] = 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 potentiallyNonOverflowedValues = useMemo(\n    () =>\n      selectedData.selectedValues\n        .map(selectedValue => findOptionInOptions(options, selectedValue))\n        .filter((option): option is OptionType => !!option),\n    [options, selectedData.selectedValues],\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  // First we display shadow copy of tags to render it\n  useEffect(() => {\n    setDisplayShadowCopy(true)\n  }, [selectedData.selectedValues.length])\n\n  // We then want to measure the tags length before displaying them\n  // so we can determine if there is an overflow or not\n  // We use useLayoutEffect to ensure the measurement is done before the browser paints\n  useLayoutEffect(() => {\n    if (selectedData.selectedValues.length === 0) {\n      setOverflowAmount(0)\n      setNonOverFlowedValues([])\n    }\n\n    if (measureRef.current && selectedData.selectedValues.length > 0) {\n      const toMeasureElements: HTMLCollection = measureRef.current.children\n      const toMeasureElementsArray = [...toMeasureElements]\n\n      const {\n        measuredVisibleTags,\n        measuredHiddenTags,\n        accumulatedWidth,\n        lastVisibleElementWidth,\n        lastVisibleLabel,\n      } = toMeasureElementsArray.reduce(\n        (\n          accumulator: {\n            measuredVisibleTags: OptionType[]\n            measuredHiddenTags: number\n            accumulatedWidth: number\n            lastVisibleElementWidth: number\n            lastVisibleLabel: ReactNode\n          },\n          currentValue,\n          index,\n        ) => {\n          const elementWidth = (currentValue as HTMLDivElement).offsetWidth\n\n          const newAccumulatedWidth =\n            accumulator.accumulatedWidth + elementWidth + SIZES_TAG.gap\n\n          const canBeVisible = newAccumulatedWidth <= innerWidth\n\n          return {\n            accumulatedWidth: !canBeVisible\n              ? accumulator.accumulatedWidth\n              : newAccumulatedWidth,\n            lastVisibleElementWidth: canBeVisible\n              ? elementWidth\n              : accumulator.lastVisibleElementWidth,\n            lastVisibleLabel: canBeVisible\n              ? potentiallyNonOverflowedValues[index].label\n              : accumulator.lastVisibleLabel,\n            measuredHiddenTags:\n              accumulator.measuredHiddenTags + (!canBeVisible ? 1 : 0),\n            measuredVisibleTags: [\n              ...accumulator.measuredVisibleTags,\n              canBeVisible && potentiallyNonOverflowedValues[index],\n            ].filter(Boolean) as OptionType[],\n          }\n        },\n        {\n          accumulatedWidth: 0,\n          lastVisibleElementWidth: 0,\n          lastVisibleLabel: '',\n          measuredHiddenTags: 0,\n          measuredVisibleTags: [],\n        },\n      )\n\n      const additionnalElementsWidth =\n        SIZES_TAG.paddings + (refPlusTag.current?.offsetWidth ?? 0)\n      const finalWidth =\n        accumulatedWidth + (measuredHiddenTags ? additionnalElementsWidth : 0)\n\n      const overflowPx = finalWidth - innerWidth\n      const hasOverflow = overflowPx > 0\n      const hasHiddenTags = measuredHiddenTags > 0\n      const lastVisibleElementMaxSize = lastVisibleElementWidth - overflowPx\n\n      // If only one element is selected and it is hidden, we need to show it\n      if (measuredHiddenTags === 1 && measuredVisibleTags.length === 0) {\n        setOverflowAmount(0)\n        setNonOverFlowedValues([potentiallyNonOverflowedValues[0]])\n\n        const newOverflowPx =\n          lastVisibleElementWidth +\n          (measuredHiddenTags > 1 ? additionnalElementsWidth : 0) -\n          innerWidth\n        setLastElementMaxWidth(lastVisibleElementWidth - newOverflowPx)\n        setOverflow(true)\n      }\n\n      // If it overflows with the last tag, we need to add an ellipsis to the last element if there is enough space (>60px)\n      // and if it is a string (do not cut ReactNode label)\n      // else we hide it completely and add it to the overflow amount\n      else if (\n        hasOverflow &&\n        hasHiddenTags &&\n        (lastVisibleElementMaxSize > 65 ||\n          (measuredVisibleTags.length === 1 &&\n            lastVisibleElementMaxSize > 65)) &&\n        typeof lastVisibleLabel === 'string'\n      ) {\n        setLastElementMaxWidth(lastVisibleElementMaxSize)\n        setOverflow(true)\n        setOverflowAmount(measuredHiddenTags)\n        setNonOverFlowedValues(measuredVisibleTags)\n      } else if (hasOverflow && hasHiddenTags) {\n        setLastElementMaxWidth(0)\n        setOverflow(false)\n        setOverflowAmount(measuredHiddenTags + 1)\n        setNonOverFlowedValues(measuredVisibleTags.slice(0, -1))\n      }\n      // Otherwise, we have enough space to show all tags\n      else {\n        setOverflow(false)\n        setOverflowAmount(measuredHiddenTags)\n        setNonOverFlowedValues(measuredVisibleTags)\n      }\n    }\n    setDisplayShadowCopy(false)\n  }, [\n    displayShadowCopy,\n    innerWidth,\n    potentiallyNonOverflowedValues,\n    selectedData.selectedValues.length,\n  ])\n\n  useEffect(() => {\n    setSelectedData({ type: 'update' })\n  }, [setSelectedData, options])\n\n  useEffect(() => {\n    const getWidth = () => {\n      if (refTag.current) {\n        setInnerWidth(refTag.current.offsetWidth)\n      } else {\n        setInnerWidth(\n          innerRef.current?.offsetWidth ??\n            0 - (arrowRef.current?.offsetWidth ?? 0) - SIZES_TAG.paddings,\n        )\n      }\n    }\n    getWidth()\n    window.addEventListener('resize', getWidth)\n\n    return () => window.removeEventListener('resize', getWidth)\n  }, [innerRef, refTag, selectedData.selectedValues])\n\n  const shouldDisplayValues = useMemo(() => {\n    if (multiselect) {\n      return (\n        potentiallyNonOverflowedValues.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    options,\n    potentiallyNonOverflowedValues.length,\n    selectedData.selectedValues,\n  ])\n\n  return (\n    <Tooltip text={tooltip}>\n      <StyledInputWrapper\n        aria-controls={dropdownId}\n        aria-expanded={isDropdownVisible}\n        aria-label={label}\n        autoFocus={autoFocus}\n        data-disabled={disabled}\n        data-dropdownvisible={isDropdownVisible}\n        data-readonly={readOnly}\n        data-size={size}\n        data-state={state}\n        data-testid={dataTestId}\n        id={id}\n        onClick={\n          openable ? () => setIsDropdownVisible(!isDropdownVisible) : undefined\n        }\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        role=\"combobox\"\n        tabIndex={0}\n      >\n        {shouldDisplayValues ? (\n          <DisplayValues\n            disabled={disabled}\n            displayShadowCopy={displayShadowCopy}\n            lastElementMaxWidth={lastElementMaxWidth}\n            measureRef={measureRef}\n            nonOverflowedValues={nonOverflowedValues}\n            overflow={overflow}\n            overflowAmount={overflowAmount}\n            overflowed={!!overflowAmount}\n            potentiallyNonOverflowedValues={potentiallyNonOverflowedValues}\n            readOnly={readOnly}\n            refPlusTag={refPlusTag}\n            refTag={refTag}\n            size={size}\n          />\n        ) : (\n          <Placeholder\n            as=\"p\"\n            disabled={disabled}\n            prominence=\"weak\"\n            sentiment=\"neutral\"\n            variant={size === 'large' ? 'body' : 'bodySmall'}\n          >\n            {placeholder}\n          </Placeholder>\n        )}\n        <StateStack alignItems=\"center\" direction=\"row\" gap={1} ref={arrowRef}>\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              data-testid=\"clear-all\"\n              disabled={disabled || !selectedData.selectedValues[0] || readOnly}\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              size=\"small\"\n              variant=\"ghost\"\n            >\n              <CloseIcon />\n            </Button>\n          ) : null}\n          <ArrowDownIcon\n            aria-label=\"show dropdown\"\n            disabled={disabled || readOnly}\n            sentiment=\"neutral\"\n            size=\"small\"\n          />\n        </StateStack>\n      </StyledInputWrapper>\n    </Tooltip>\n  )\n}\n\nexport { SelectBar, StyledInputWrapper }\n"]} */"));
30
30
  const Placeholder = /* @__PURE__ */ _styled(Text, process.env.NODE_ENV === "production" ? {
31
31
  target: "e1pocxam5"
32
32
  } : {
@@ -37,7 +37,7 @@ const Placeholder = /* @__PURE__ */ _styled(Text, process.env.NODE_ENV === "prod
37
37
  styles: "user-select:none;align-self:center"
38
38
  } : {
39
39
  name: "14eg98m",
40
- styles: "user-select:none;align-self:center/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInput/SelectBar.tsx"],"names":[],"mappings":"AAiEgC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInput/SelectBar.tsx","sourcesContent":["'use client'\n\nimport styled from '@emotion/styled'\nimport {\n  AlertCircleIcon,\n  ArrowDownIcon,\n  CheckCircleIcon,\n  CloseIcon,\n  PlusIcon,\n} from '@ultraviolet/icons'\nimport type { ReactNode, RefObject } from 'react'\nimport { useEffect, useMemo, useRef, useState } from 'react'\nimport { Button } from '../Button'\nimport { StyledChildrenContainer } from '../Popup'\nimport { Stack } from '../Stack'\nimport { Tag } from '../Tag'\nimport { Text } from '../Text'\nimport { Tooltip } from '../Tooltip'\nimport { findOptionInOptions } from './findOptionInOptions'\nimport { useSelectInput } from './SelectInputProvider'\nimport type { OptionType } from './types'\nimport { INPUT_SIZE_HEIGHT } from './types'\n\nconst SIZES_TAG = {\n  gap: 8,\n  paddings: 16,\n  plusTag: 48,\n}\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  dropdownId?: string\n}\n\ntype DisplayValuesProps = {\n  refTag: RefObject<HTMLDivElement | null>\n  nonOverflowedValues: OptionType[]\n  potentiallyNonOverflowedValues: OptionType[]\n  disabled: boolean\n  readOnly: boolean\n  overflowed: boolean\n  overflowAmount: number\n  measureRef: RefObject<HTMLDivElement | null>\n  size: 'small' | 'medium' | 'large'\n  lastElementMaxWidth: number\n  overflow?: boolean\n  refPlusTag: RefObject<HTMLDivElement | null>\n}\n\nconst StateStack = styled(Stack)`\n  padding-right: ${({ theme }) => theme.space['2']};\n  display: flex;\n`\nconst Placeholder = styled(Text)`\nuser-select: none;\nalign-self: center;\n`\n\nconst StyledInputWrapper = styled.div<{\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: grid;\n  width: 100%;\n  gap: ${({ theme }) => theme.space[1]};\n  grid-template-columns: 1fr auto ;\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\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  shouldForwardProp: prop => !['lastElementMaxWidth', 'hidden'].includes(prop),\n})<{\n  lastElementMaxWidth?: number\n  hidden?: boolean\n}>`\n  height: max-content;\n  width: fit-content;\n  min-width: ${({ lastElementMaxWidth }) =>\n    lastElementMaxWidth ? 'auto' : 'fit-content'};\n\n  max-width: ${({ lastElementMaxWidth, hidden }) =>\n    lastElementMaxWidth && !hidden ? `${lastElementMaxWidth}px` : '100%'};\n\n  ${({ hidden }) =>\n    hidden\n      ? 'visibility: hidden;'\n      : `\n  text-overflow: ellipsis;\n  overflow: hidden;`}\n\n  & > ${StyledChildrenContainer} {\n    overflow: hidden;\n  }\n`\n\nconst SelectedValues = styled(Text)`\n  text-overflow: ellipsis;\n  overflow: hidden;\n  white-space: nowrap;\n  align-self: center;\n`\nconst PlusTag = styled(Tag)`\nwidth: ${({ theme }) => theme.sizing[500]};\n;\n`\n\nconst MultiselectStack = styled(Stack)`\noverflow: hidden;\nmax-width: 100%;\nheight: 100%;\n`\n\nconst DisplayValues = ({\n  refTag,\n  nonOverflowedValues,\n  potentiallyNonOverflowedValues,\n  disabled,\n  readOnly,\n  overflowed,\n  overflowAmount,\n  size,\n  measureRef,\n  lastElementMaxWidth,\n  overflow,\n  refPlusTag,\n}: DisplayValuesProps) => {\n  const { multiselect, selectedData, setSelectedData, options, onChange } =\n    useSelectInput()\n\n  return multiselect ? (\n    <MultiselectStack\n      alignItems=\"center\"\n      direction=\"row\"\n      gap=\"1\"\n      ref={refTag}\n      wrap=\"nowrap\"\n    >\n      {/* Hidden div to measure the width of the tags */}\n      <div\n        ref={measureRef}\n        style={{\n          position: 'absolute',\n        }}\n      >\n        {potentiallyNonOverflowedValues.map(option => (\n          <CustomTag\n            className={option.value}\n            hidden\n            key={option.value}\n            onClose={() => {}}\n          >\n            {option?.label}\n          </CustomTag>\n        ))}\n      </div>\n      {nonOverflowedValues.map((option, index) => (\n        <CustomTag\n          data-testid=\"selected-options-tags\"\n          disabled={disabled}\n          key={option?.value}\n          lastElementMaxWidth={\n            index === nonOverflowedValues.length - 1 && overflow\n              ? lastElementMaxWidth\n              : 0\n          }\n          onClose={\n            !readOnly\n              ? event => {\n                  event.stopPropagation()\n                  setSelectedData({\n                    clickedOption: option,\n                    type: 'selectOption',\n                  })\n                  const newSelectedValues = selectedData.selectedValues?.filter(\n                    val => val !== option.value,\n                  )\n                  onChange?.(newSelectedValues)\n                }\n              : undefined\n          }\n          sentiment=\"neutral\"\n        >\n          {option?.label}\n        </CustomTag>\n      ))}\n\n      {overflowed ? (\n        <Stack justifyContent=\"center\" ref={refPlusTag}>\n          <PlusTag\n            aria-label=\"Plus tag\"\n            data-testid=\"plus-tag\"\n            disabled={disabled}\n            key=\"+\"\n            sentiment=\"neutral\"\n          >\n            <PlusIcon size=\"xsmall\" />\n            {overflowAmount}\n          </PlusTag>\n        </Stack>\n      ) : null}\n    </MultiselectStack>\n  ) : (\n    <SelectedValues\n      as=\"div\"\n      disabled={disabled}\n      prominence=\"default\"\n      sentiment=\"neutral\"\n      variant={size === 'large' ? 'body' : 'bodySmall'}\n    >\n      {selectedData.selectedValues[0]\n        ? findOptionInOptions(options, selectedData.selectedValues[0])?.label\n        : null}\n    </SelectedValues>\n  )\n}\n\nconst 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  dropdownId,\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 measureRef = useRef<HTMLDivElement>(null)\n  const arrowRef = useRef<HTMLDivElement>(null)\n  const refPlusTag = useRef<HTMLDivElement>(null)\n  // width - width of the arrow (in px) - padding between tags (in px)\n  const [innerWidth, setInnerWidth] = useState(\n    innerRef.current?.offsetWidth ??\n      0 - (arrowRef.current?.offsetWidth ?? 0) - SIZES_TAG.paddings,\n  )\n  const [overflowAmount, setOverflowAmount] = useState(0)\n  const [overflow, setOverflow] = useState(false)\n  const [lastElementMaxWidth, setLastElementMaxWidth] = 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 potentiallyNonOverflowedValues = useMemo(\n    () =>\n      selectedData.selectedValues\n        .map(selectedValue => findOptionInOptions(options, selectedValue))\n        .filter((option): option is OptionType => !!option),\n    [options, selectedData.selectedValues],\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    if (selectedData.selectedValues.length === 0) {\n      setOverflowAmount(0)\n      setNonOverFlowedValues([])\n    }\n    if (measureRef.current && selectedData.selectedValues.length > 0) {\n      const toMeasureElements: HTMLCollection = measureRef.current.children\n      const toMeasureElementsArray = [...toMeasureElements]\n\n      const {\n        measuredVisibleTags,\n        measuredHiddenTags,\n        accumulatedWidth,\n        lastVisibleElementWidth,\n        lastVisibleLabel,\n      } = toMeasureElementsArray.reduce(\n        (\n          accumulator: {\n            measuredVisibleTags: OptionType[]\n            measuredHiddenTags: number\n            accumulatedWidth: number\n            lastVisibleElementWidth: number\n            lastVisibleLabel: ReactNode\n          },\n          currentValue,\n          index,\n        ) => {\n          const elementWidth = (currentValue as HTMLDivElement).offsetWidth\n\n          const newAccumulatedWidth =\n            accumulator.accumulatedWidth + elementWidth + SIZES_TAG.gap\n\n          const canBeVisible = newAccumulatedWidth <= innerWidth\n\n          return {\n            accumulatedWidth: !canBeVisible\n              ? accumulator.accumulatedWidth\n              : newAccumulatedWidth,\n            lastVisibleElementWidth: canBeVisible\n              ? elementWidth\n              : accumulator.lastVisibleElementWidth,\n            lastVisibleLabel: canBeVisible\n              ? potentiallyNonOverflowedValues[index].label\n              : accumulator.lastVisibleLabel,\n            measuredHiddenTags:\n              accumulator.measuredHiddenTags + (!canBeVisible ? 1 : 0),\n            measuredVisibleTags: [\n              ...accumulator.measuredVisibleTags,\n              canBeVisible && potentiallyNonOverflowedValues[index],\n            ].filter(Boolean) as OptionType[],\n          }\n        },\n        {\n          accumulatedWidth: 0,\n          lastVisibleElementWidth: 0,\n          lastVisibleLabel: '',\n          measuredHiddenTags: 0,\n          measuredVisibleTags: [],\n        },\n      )\n\n      const additionnalElementsWidth =\n        SIZES_TAG.paddings + (refPlusTag.current?.offsetWidth ?? 0)\n      const finalWidth =\n        accumulatedWidth + (measuredHiddenTags ? additionnalElementsWidth : 0)\n\n      const overflowPx = finalWidth - innerWidth\n      const hasOverflow = overflowPx > 0\n      const hasHiddenTags = measuredHiddenTags > 0\n      const lastVisibleElementMaxSize = lastVisibleElementWidth - overflowPx\n\n      // If only one element is selected and it is hidden, we need to show it\n      if (measuredHiddenTags === 1 && measuredVisibleTags.length === 0) {\n        setOverflowAmount(0)\n        setNonOverFlowedValues([potentiallyNonOverflowedValues[0]])\n\n        const newOverflowPx =\n          lastVisibleElementWidth +\n          (measuredHiddenTags > 1 ? additionnalElementsWidth : 0) -\n          innerWidth\n        setLastElementMaxWidth(lastVisibleElementWidth - newOverflowPx)\n        setOverflow(true)\n      }\n\n      // If it overflows with the last tag, we need to add an ellipsis to the last element if there is enough space (>60px)\n      // and if it is a string (do not cut ReactNode label)\n      // else we hide it completely and add it to the overflow amount\n      else if (\n        hasOverflow &&\n        hasHiddenTags &&\n        (lastVisibleElementMaxSize > 65 ||\n          (measuredVisibleTags.length === 1 &&\n            lastVisibleElementMaxSize > 65)) &&\n        typeof lastVisibleLabel === 'string'\n      ) {\n        setLastElementMaxWidth(lastVisibleElementMaxSize)\n        setOverflow(true)\n        setOverflowAmount(measuredHiddenTags)\n        setNonOverFlowedValues(measuredVisibleTags)\n      } else if (hasOverflow && hasHiddenTags) {\n        setLastElementMaxWidth(0)\n        setOverflow(false)\n        setOverflowAmount(measuredHiddenTags + 1)\n        setNonOverFlowedValues(measuredVisibleTags.slice(0, -1))\n      }\n      // Otherwise, we have enough space to show all tags\n      else {\n        setOverflow(false)\n        setOverflowAmount(measuredHiddenTags)\n        setNonOverFlowedValues(measuredVisibleTags)\n      }\n    }\n  }, [\n    selectedData.selectedValues.length,\n    innerWidth,\n    potentiallyNonOverflowedValues,\n  ])\n\n  useEffect(() => {\n    setSelectedData({ type: 'update' })\n  }, [setSelectedData, options])\n\n  useEffect(() => {\n    const getWidth = () => {\n      if (refTag.current) {\n        setInnerWidth(refTag.current.offsetWidth)\n      } else {\n        setInnerWidth(\n          innerRef.current?.offsetWidth ??\n            0 - (arrowRef.current?.offsetWidth ?? 0) - SIZES_TAG.paddings,\n        )\n      }\n    }\n    getWidth()\n    window.addEventListener('resize', getWidth)\n\n    return () => window.removeEventListener('resize', getWidth)\n  }, [innerRef, refTag, selectedData.selectedValues])\n\n  const shouldDisplayValues = useMemo(() => {\n    if (multiselect) {\n      return (\n        potentiallyNonOverflowedValues.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    options,\n    potentiallyNonOverflowedValues.length,\n    selectedData.selectedValues,\n  ])\n\n  return (\n    <Tooltip text={tooltip}>\n      <StyledInputWrapper\n        aria-controls={dropdownId}\n        aria-expanded={isDropdownVisible}\n        aria-label={label}\n        autoFocus={autoFocus}\n        data-disabled={disabled}\n        data-dropdownvisible={isDropdownVisible}\n        data-readonly={readOnly}\n        data-size={size}\n        data-state={state}\n        data-testid={dataTestId}\n        id={id}\n        onClick={\n          openable ? () => setIsDropdownVisible(!isDropdownVisible) : undefined\n        }\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        role=\"combobox\"\n        tabIndex={0}\n      >\n        {shouldDisplayValues ? (\n          <DisplayValues\n            disabled={disabled}\n            lastElementMaxWidth={lastElementMaxWidth}\n            measureRef={measureRef}\n            nonOverflowedValues={nonOverflowedValues}\n            overflow={overflow}\n            overflowAmount={overflowAmount}\n            overflowed={!!overflowAmount}\n            potentiallyNonOverflowedValues={potentiallyNonOverflowedValues}\n            readOnly={readOnly}\n            refPlusTag={refPlusTag}\n            refTag={refTag}\n            size={size}\n          />\n        ) : (\n          <Placeholder\n            as=\"p\"\n            disabled={disabled}\n            prominence=\"weak\"\n            sentiment=\"neutral\"\n            variant={size === 'large' ? 'body' : 'bodySmall'}\n          >\n            {placeholder}\n          </Placeholder>\n        )}\n        <StateStack alignItems=\"center\" direction=\"row\" gap={1} ref={arrowRef}>\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              data-testid=\"clear-all\"\n              disabled={disabled || !selectedData.selectedValues[0] || readOnly}\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              size=\"small\"\n              variant=\"ghost\"\n            >\n              <CloseIcon />\n            </Button>\n          ) : null}\n          <ArrowDownIcon\n            aria-label=\"show dropdown\"\n            disabled={disabled || readOnly}\n            sentiment=\"neutral\"\n            size=\"small\"\n          />\n        </StateStack>\n      </StyledInputWrapper>\n    </Tooltip>\n  )\n}\n\nexport { SelectBar, StyledInputWrapper }\n"]} */",
40
+ styles: "user-select:none;align-self:center/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInput/SelectBar.tsx"],"names":[],"mappings":"AAkEgC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInput/SelectBar.tsx","sourcesContent":["'use client'\n\nimport styled from '@emotion/styled'\nimport {\n  AlertCircleIcon,\n  ArrowDownIcon,\n  CheckCircleIcon,\n  CloseIcon,\n  PlusIcon,\n} from '@ultraviolet/icons'\nimport type { ReactNode, RefObject } from 'react'\nimport { useEffect, useLayoutEffect, useMemo, useRef, useState } from 'react'\nimport { Button } from '../Button'\nimport { StyledChildrenContainer } from '../Popup'\nimport { Stack } from '../Stack'\nimport { Tag } from '../Tag'\nimport { Text } from '../Text'\nimport { Tooltip } from '../Tooltip'\nimport { findOptionInOptions } from './findOptionInOptions'\nimport { useSelectInput } from './SelectInputProvider'\nimport type { OptionType } from './types'\nimport { INPUT_SIZE_HEIGHT } from './types'\n\nconst SIZES_TAG = {\n  gap: 8,\n  paddings: 16,\n  plusTag: 48,\n}\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  dropdownId?: string\n}\n\ntype DisplayValuesProps = {\n  refTag: RefObject<HTMLDivElement | null>\n  nonOverflowedValues: OptionType[]\n  potentiallyNonOverflowedValues: OptionType[]\n  disabled: boolean\n  readOnly: boolean\n  overflowed: boolean\n  overflowAmount: number\n  measureRef: RefObject<HTMLDivElement | null>\n  size: 'small' | 'medium' | 'large'\n  lastElementMaxWidth: number\n  overflow?: boolean\n  refPlusTag: RefObject<HTMLDivElement | null>\n  displayShadowCopy?: boolean\n}\n\nconst StateStack = styled(Stack)`\n  padding-right: ${({ theme }) => theme.space['2']};\n  display: flex;\n`\nconst Placeholder = styled(Text)`\nuser-select: none;\nalign-self: center;\n`\n\nconst StyledInputWrapper = styled.div<{\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: grid;\n  width: 100%;\n  gap: ${({ theme }) => theme.space[1]};\n  grid-template-columns: 1fr auto ;\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\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  shouldForwardProp: prop => !['lastElementMaxWidth', 'hidden'].includes(prop),\n})<{\n  lastElementMaxWidth?: number\n  hidden?: boolean\n}>`\n  height: max-content;\n  width: fit-content;\n  min-width: ${({ lastElementMaxWidth }) =>\n    lastElementMaxWidth ? 'auto' : 'fit-content'};\n\n  max-width: ${({ lastElementMaxWidth, hidden }) =>\n    lastElementMaxWidth && !hidden ? `${lastElementMaxWidth}px` : '100%'};\n\n  ${({ hidden }) =>\n    hidden\n      ? 'visibility: hidden;'\n      : `\n  text-overflow: ellipsis;\n  overflow: hidden;`}\n\n  & > ${StyledChildrenContainer} {\n    overflow: hidden;\n  }\n`\n\nconst SelectedValues = styled(Text)`\n  text-overflow: ellipsis;\n  overflow: hidden;\n  white-space: nowrap;\n  align-self: center;\n`\nconst PlusTag = styled(Tag)`\nwidth: ${({ theme }) => theme.sizing[500]};\n;\n`\n\nconst MultiselectStack = styled(Stack)`\noverflow: hidden;\nmax-width: 100%;\nheight: 100%;\n`\n\nconst DisplayValues = ({\n  refTag,\n  nonOverflowedValues,\n  potentiallyNonOverflowedValues,\n  disabled,\n  readOnly,\n  overflowed,\n  overflowAmount,\n  size,\n  measureRef,\n  lastElementMaxWidth,\n  overflow,\n  refPlusTag,\n  displayShadowCopy,\n}: DisplayValuesProps) => {\n  const { multiselect, selectedData, setSelectedData, options, onChange } =\n    useSelectInput()\n\n  return multiselect ? (\n    <MultiselectStack\n      alignItems=\"center\"\n      direction=\"row\"\n      gap=\"1\"\n      ref={refTag}\n      wrap=\"nowrap\"\n    >\n      {/* Hidden div to measure the width of the tags */}\n      {displayShadowCopy ? (\n        <div\n          ref={measureRef}\n          style={{\n            position: 'absolute',\n          }}\n        >\n          {potentiallyNonOverflowedValues.map(option => (\n            <CustomTag\n              className={option.value}\n              hidden\n              key={option.value}\n              onClose={() => {}}\n            >\n              {option?.label}\n            </CustomTag>\n          ))}\n        </div>\n      ) : null}\n      {nonOverflowedValues.map((option, index) => (\n        <CustomTag\n          data-testid=\"selected-options-tags\"\n          disabled={disabled}\n          key={option?.value}\n          lastElementMaxWidth={\n            index === nonOverflowedValues.length - 1 && overflow\n              ? lastElementMaxWidth\n              : 0\n          }\n          onClose={\n            !readOnly\n              ? event => {\n                  event.stopPropagation()\n                  setSelectedData({\n                    clickedOption: option,\n                    type: 'selectOption',\n                  })\n                  const newSelectedValues = selectedData.selectedValues?.filter(\n                    val => val !== option.value,\n                  )\n                  onChange?.(newSelectedValues)\n                }\n              : undefined\n          }\n          sentiment=\"neutral\"\n        >\n          {option?.label}\n        </CustomTag>\n      ))}\n\n      {overflowed ? (\n        <Stack justifyContent=\"center\" ref={refPlusTag}>\n          <PlusTag\n            aria-label=\"Plus tag\"\n            data-testid=\"plus-tag\"\n            disabled={disabled}\n            key=\"+\"\n            sentiment=\"neutral\"\n          >\n            <PlusIcon size=\"xsmall\" />\n            {overflowAmount}\n          </PlusTag>\n        </Stack>\n      ) : null}\n    </MultiselectStack>\n  ) : (\n    <SelectedValues\n      as=\"div\"\n      disabled={disabled}\n      prominence=\"default\"\n      sentiment=\"neutral\"\n      variant={size === 'large' ? 'body' : 'bodySmall'}\n    >\n      {selectedData.selectedValues[0]\n        ? findOptionInOptions(options, selectedData.selectedValues[0])?.label\n        : null}\n    </SelectedValues>\n  )\n}\n\nconst 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  dropdownId,\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 [displayShadowCopy, setDisplayShadowCopy] = useState(false)\n  const refTag = useRef<HTMLDivElement>(null)\n  const measureRef = useRef<HTMLDivElement>(null)\n  const arrowRef = useRef<HTMLDivElement>(null)\n  const refPlusTag = useRef<HTMLDivElement>(null)\n  // width - width of the arrow (in px) - padding between tags (in px)\n  const [innerWidth, setInnerWidth] = useState(\n    innerRef.current?.offsetWidth ??\n      0 - (arrowRef.current?.offsetWidth ?? 0) - SIZES_TAG.paddings,\n  )\n  const [overflowAmount, setOverflowAmount] = useState(0)\n  const [overflow, setOverflow] = useState(false)\n  const [lastElementMaxWidth, setLastElementMaxWidth] = 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 potentiallyNonOverflowedValues = useMemo(\n    () =>\n      selectedData.selectedValues\n        .map(selectedValue => findOptionInOptions(options, selectedValue))\n        .filter((option): option is OptionType => !!option),\n    [options, selectedData.selectedValues],\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  // First we display shadow copy of tags to render it\n  useEffect(() => {\n    setDisplayShadowCopy(true)\n  }, [selectedData.selectedValues.length])\n\n  // We then want to measure the tags length before displaying them\n  // so we can determine if there is an overflow or not\n  // We use useLayoutEffect to ensure the measurement is done before the browser paints\n  useLayoutEffect(() => {\n    if (selectedData.selectedValues.length === 0) {\n      setOverflowAmount(0)\n      setNonOverFlowedValues([])\n    }\n\n    if (measureRef.current && selectedData.selectedValues.length > 0) {\n      const toMeasureElements: HTMLCollection = measureRef.current.children\n      const toMeasureElementsArray = [...toMeasureElements]\n\n      const {\n        measuredVisibleTags,\n        measuredHiddenTags,\n        accumulatedWidth,\n        lastVisibleElementWidth,\n        lastVisibleLabel,\n      } = toMeasureElementsArray.reduce(\n        (\n          accumulator: {\n            measuredVisibleTags: OptionType[]\n            measuredHiddenTags: number\n            accumulatedWidth: number\n            lastVisibleElementWidth: number\n            lastVisibleLabel: ReactNode\n          },\n          currentValue,\n          index,\n        ) => {\n          const elementWidth = (currentValue as HTMLDivElement).offsetWidth\n\n          const newAccumulatedWidth =\n            accumulator.accumulatedWidth + elementWidth + SIZES_TAG.gap\n\n          const canBeVisible = newAccumulatedWidth <= innerWidth\n\n          return {\n            accumulatedWidth: !canBeVisible\n              ? accumulator.accumulatedWidth\n              : newAccumulatedWidth,\n            lastVisibleElementWidth: canBeVisible\n              ? elementWidth\n              : accumulator.lastVisibleElementWidth,\n            lastVisibleLabel: canBeVisible\n              ? potentiallyNonOverflowedValues[index].label\n              : accumulator.lastVisibleLabel,\n            measuredHiddenTags:\n              accumulator.measuredHiddenTags + (!canBeVisible ? 1 : 0),\n            measuredVisibleTags: [\n              ...accumulator.measuredVisibleTags,\n              canBeVisible && potentiallyNonOverflowedValues[index],\n            ].filter(Boolean) as OptionType[],\n          }\n        },\n        {\n          accumulatedWidth: 0,\n          lastVisibleElementWidth: 0,\n          lastVisibleLabel: '',\n          measuredHiddenTags: 0,\n          measuredVisibleTags: [],\n        },\n      )\n\n      const additionnalElementsWidth =\n        SIZES_TAG.paddings + (refPlusTag.current?.offsetWidth ?? 0)\n      const finalWidth =\n        accumulatedWidth + (measuredHiddenTags ? additionnalElementsWidth : 0)\n\n      const overflowPx = finalWidth - innerWidth\n      const hasOverflow = overflowPx > 0\n      const hasHiddenTags = measuredHiddenTags > 0\n      const lastVisibleElementMaxSize = lastVisibleElementWidth - overflowPx\n\n      // If only one element is selected and it is hidden, we need to show it\n      if (measuredHiddenTags === 1 && measuredVisibleTags.length === 0) {\n        setOverflowAmount(0)\n        setNonOverFlowedValues([potentiallyNonOverflowedValues[0]])\n\n        const newOverflowPx =\n          lastVisibleElementWidth +\n          (measuredHiddenTags > 1 ? additionnalElementsWidth : 0) -\n          innerWidth\n        setLastElementMaxWidth(lastVisibleElementWidth - newOverflowPx)\n        setOverflow(true)\n      }\n\n      // If it overflows with the last tag, we need to add an ellipsis to the last element if there is enough space (>60px)\n      // and if it is a string (do not cut ReactNode label)\n      // else we hide it completely and add it to the overflow amount\n      else if (\n        hasOverflow &&\n        hasHiddenTags &&\n        (lastVisibleElementMaxSize > 65 ||\n          (measuredVisibleTags.length === 1 &&\n            lastVisibleElementMaxSize > 65)) &&\n        typeof lastVisibleLabel === 'string'\n      ) {\n        setLastElementMaxWidth(lastVisibleElementMaxSize)\n        setOverflow(true)\n        setOverflowAmount(measuredHiddenTags)\n        setNonOverFlowedValues(measuredVisibleTags)\n      } else if (hasOverflow && hasHiddenTags) {\n        setLastElementMaxWidth(0)\n        setOverflow(false)\n        setOverflowAmount(measuredHiddenTags + 1)\n        setNonOverFlowedValues(measuredVisibleTags.slice(0, -1))\n      }\n      // Otherwise, we have enough space to show all tags\n      else {\n        setOverflow(false)\n        setOverflowAmount(measuredHiddenTags)\n        setNonOverFlowedValues(measuredVisibleTags)\n      }\n    }\n    setDisplayShadowCopy(false)\n  }, [\n    displayShadowCopy,\n    innerWidth,\n    potentiallyNonOverflowedValues,\n    selectedData.selectedValues.length,\n  ])\n\n  useEffect(() => {\n    setSelectedData({ type: 'update' })\n  }, [setSelectedData, options])\n\n  useEffect(() => {\n    const getWidth = () => {\n      if (refTag.current) {\n        setInnerWidth(refTag.current.offsetWidth)\n      } else {\n        setInnerWidth(\n          innerRef.current?.offsetWidth ??\n            0 - (arrowRef.current?.offsetWidth ?? 0) - SIZES_TAG.paddings,\n        )\n      }\n    }\n    getWidth()\n    window.addEventListener('resize', getWidth)\n\n    return () => window.removeEventListener('resize', getWidth)\n  }, [innerRef, refTag, selectedData.selectedValues])\n\n  const shouldDisplayValues = useMemo(() => {\n    if (multiselect) {\n      return (\n        potentiallyNonOverflowedValues.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    options,\n    potentiallyNonOverflowedValues.length,\n    selectedData.selectedValues,\n  ])\n\n  return (\n    <Tooltip text={tooltip}>\n      <StyledInputWrapper\n        aria-controls={dropdownId}\n        aria-expanded={isDropdownVisible}\n        aria-label={label}\n        autoFocus={autoFocus}\n        data-disabled={disabled}\n        data-dropdownvisible={isDropdownVisible}\n        data-readonly={readOnly}\n        data-size={size}\n        data-state={state}\n        data-testid={dataTestId}\n        id={id}\n        onClick={\n          openable ? () => setIsDropdownVisible(!isDropdownVisible) : undefined\n        }\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        role=\"combobox\"\n        tabIndex={0}\n      >\n        {shouldDisplayValues ? (\n          <DisplayValues\n            disabled={disabled}\n            displayShadowCopy={displayShadowCopy}\n            lastElementMaxWidth={lastElementMaxWidth}\n            measureRef={measureRef}\n            nonOverflowedValues={nonOverflowedValues}\n            overflow={overflow}\n            overflowAmount={overflowAmount}\n            overflowed={!!overflowAmount}\n            potentiallyNonOverflowedValues={potentiallyNonOverflowedValues}\n            readOnly={readOnly}\n            refPlusTag={refPlusTag}\n            refTag={refTag}\n            size={size}\n          />\n        ) : (\n          <Placeholder\n            as=\"p\"\n            disabled={disabled}\n            prominence=\"weak\"\n            sentiment=\"neutral\"\n            variant={size === 'large' ? 'body' : 'bodySmall'}\n          >\n            {placeholder}\n          </Placeholder>\n        )}\n        <StateStack alignItems=\"center\" direction=\"row\" gap={1} ref={arrowRef}>\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              data-testid=\"clear-all\"\n              disabled={disabled || !selectedData.selectedValues[0] || readOnly}\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              size=\"small\"\n              variant=\"ghost\"\n            >\n              <CloseIcon />\n            </Button>\n          ) : null}\n          <ArrowDownIcon\n            aria-label=\"show dropdown\"\n            disabled={disabled || readOnly}\n            sentiment=\"neutral\"\n            size=\"small\"\n          />\n        </StateStack>\n      </StyledInputWrapper>\n    </Tooltip>\n  )\n}\n\nexport { SelectBar, StyledInputWrapper }\n"]} */",
41
41
  toString: _EMOTION_STRINGIFIED_CSS_ERROR__
42
42
  });
43
43
  const StyledInputWrapper = /* @__PURE__ */ _styled("div", process.env.NODE_ENV === "production" ? {
@@ -99,7 +99,7 @@ const StyledInputWrapper = /* @__PURE__ */ _styled("div", process.env.NODE_ENV =
99
99
  theme
100
100
  }) => theme.colors.neutral.backgroundDisabled, ";border-color:", ({
101
101
  theme
102
- }) => 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/SelectInput/SelectBar.tsx"],"names":[],"mappings":"AA6EE","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInput/SelectBar.tsx","sourcesContent":["'use client'\n\nimport styled from '@emotion/styled'\nimport {\n  AlertCircleIcon,\n  ArrowDownIcon,\n  CheckCircleIcon,\n  CloseIcon,\n  PlusIcon,\n} from '@ultraviolet/icons'\nimport type { ReactNode, RefObject } from 'react'\nimport { useEffect, useMemo, useRef, useState } from 'react'\nimport { Button } from '../Button'\nimport { StyledChildrenContainer } from '../Popup'\nimport { Stack } from '../Stack'\nimport { Tag } from '../Tag'\nimport { Text } from '../Text'\nimport { Tooltip } from '../Tooltip'\nimport { findOptionInOptions } from './findOptionInOptions'\nimport { useSelectInput } from './SelectInputProvider'\nimport type { OptionType } from './types'\nimport { INPUT_SIZE_HEIGHT } from './types'\n\nconst SIZES_TAG = {\n  gap: 8,\n  paddings: 16,\n  plusTag: 48,\n}\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  dropdownId?: string\n}\n\ntype DisplayValuesProps = {\n  refTag: RefObject<HTMLDivElement | null>\n  nonOverflowedValues: OptionType[]\n  potentiallyNonOverflowedValues: OptionType[]\n  disabled: boolean\n  readOnly: boolean\n  overflowed: boolean\n  overflowAmount: number\n  measureRef: RefObject<HTMLDivElement | null>\n  size: 'small' | 'medium' | 'large'\n  lastElementMaxWidth: number\n  overflow?: boolean\n  refPlusTag: RefObject<HTMLDivElement | null>\n}\n\nconst StateStack = styled(Stack)`\n  padding-right: ${({ theme }) => theme.space['2']};\n  display: flex;\n`\nconst Placeholder = styled(Text)`\nuser-select: none;\nalign-self: center;\n`\n\nconst StyledInputWrapper = styled.div<{\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: grid;\n  width: 100%;\n  gap: ${({ theme }) => theme.space[1]};\n  grid-template-columns: 1fr auto ;\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\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  shouldForwardProp: prop => !['lastElementMaxWidth', 'hidden'].includes(prop),\n})<{\n  lastElementMaxWidth?: number\n  hidden?: boolean\n}>`\n  height: max-content;\n  width: fit-content;\n  min-width: ${({ lastElementMaxWidth }) =>\n    lastElementMaxWidth ? 'auto' : 'fit-content'};\n\n  max-width: ${({ lastElementMaxWidth, hidden }) =>\n    lastElementMaxWidth && !hidden ? `${lastElementMaxWidth}px` : '100%'};\n\n  ${({ hidden }) =>\n    hidden\n      ? 'visibility: hidden;'\n      : `\n  text-overflow: ellipsis;\n  overflow: hidden;`}\n\n  & > ${StyledChildrenContainer} {\n    overflow: hidden;\n  }\n`\n\nconst SelectedValues = styled(Text)`\n  text-overflow: ellipsis;\n  overflow: hidden;\n  white-space: nowrap;\n  align-self: center;\n`\nconst PlusTag = styled(Tag)`\nwidth: ${({ theme }) => theme.sizing[500]};\n;\n`\n\nconst MultiselectStack = styled(Stack)`\noverflow: hidden;\nmax-width: 100%;\nheight: 100%;\n`\n\nconst DisplayValues = ({\n  refTag,\n  nonOverflowedValues,\n  potentiallyNonOverflowedValues,\n  disabled,\n  readOnly,\n  overflowed,\n  overflowAmount,\n  size,\n  measureRef,\n  lastElementMaxWidth,\n  overflow,\n  refPlusTag,\n}: DisplayValuesProps) => {\n  const { multiselect, selectedData, setSelectedData, options, onChange } =\n    useSelectInput()\n\n  return multiselect ? (\n    <MultiselectStack\n      alignItems=\"center\"\n      direction=\"row\"\n      gap=\"1\"\n      ref={refTag}\n      wrap=\"nowrap\"\n    >\n      {/* Hidden div to measure the width of the tags */}\n      <div\n        ref={measureRef}\n        style={{\n          position: 'absolute',\n        }}\n      >\n        {potentiallyNonOverflowedValues.map(option => (\n          <CustomTag\n            className={option.value}\n            hidden\n            key={option.value}\n            onClose={() => {}}\n          >\n            {option?.label}\n          </CustomTag>\n        ))}\n      </div>\n      {nonOverflowedValues.map((option, index) => (\n        <CustomTag\n          data-testid=\"selected-options-tags\"\n          disabled={disabled}\n          key={option?.value}\n          lastElementMaxWidth={\n            index === nonOverflowedValues.length - 1 && overflow\n              ? lastElementMaxWidth\n              : 0\n          }\n          onClose={\n            !readOnly\n              ? event => {\n                  event.stopPropagation()\n                  setSelectedData({\n                    clickedOption: option,\n                    type: 'selectOption',\n                  })\n                  const newSelectedValues = selectedData.selectedValues?.filter(\n                    val => val !== option.value,\n                  )\n                  onChange?.(newSelectedValues)\n                }\n              : undefined\n          }\n          sentiment=\"neutral\"\n        >\n          {option?.label}\n        </CustomTag>\n      ))}\n\n      {overflowed ? (\n        <Stack justifyContent=\"center\" ref={refPlusTag}>\n          <PlusTag\n            aria-label=\"Plus tag\"\n            data-testid=\"plus-tag\"\n            disabled={disabled}\n            key=\"+\"\n            sentiment=\"neutral\"\n          >\n            <PlusIcon size=\"xsmall\" />\n            {overflowAmount}\n          </PlusTag>\n        </Stack>\n      ) : null}\n    </MultiselectStack>\n  ) : (\n    <SelectedValues\n      as=\"div\"\n      disabled={disabled}\n      prominence=\"default\"\n      sentiment=\"neutral\"\n      variant={size === 'large' ? 'body' : 'bodySmall'}\n    >\n      {selectedData.selectedValues[0]\n        ? findOptionInOptions(options, selectedData.selectedValues[0])?.label\n        : null}\n    </SelectedValues>\n  )\n}\n\nconst 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  dropdownId,\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 measureRef = useRef<HTMLDivElement>(null)\n  const arrowRef = useRef<HTMLDivElement>(null)\n  const refPlusTag = useRef<HTMLDivElement>(null)\n  // width - width of the arrow (in px) - padding between tags (in px)\n  const [innerWidth, setInnerWidth] = useState(\n    innerRef.current?.offsetWidth ??\n      0 - (arrowRef.current?.offsetWidth ?? 0) - SIZES_TAG.paddings,\n  )\n  const [overflowAmount, setOverflowAmount] = useState(0)\n  const [overflow, setOverflow] = useState(false)\n  const [lastElementMaxWidth, setLastElementMaxWidth] = 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 potentiallyNonOverflowedValues = useMemo(\n    () =>\n      selectedData.selectedValues\n        .map(selectedValue => findOptionInOptions(options, selectedValue))\n        .filter((option): option is OptionType => !!option),\n    [options, selectedData.selectedValues],\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    if (selectedData.selectedValues.length === 0) {\n      setOverflowAmount(0)\n      setNonOverFlowedValues([])\n    }\n    if (measureRef.current && selectedData.selectedValues.length > 0) {\n      const toMeasureElements: HTMLCollection = measureRef.current.children\n      const toMeasureElementsArray = [...toMeasureElements]\n\n      const {\n        measuredVisibleTags,\n        measuredHiddenTags,\n        accumulatedWidth,\n        lastVisibleElementWidth,\n        lastVisibleLabel,\n      } = toMeasureElementsArray.reduce(\n        (\n          accumulator: {\n            measuredVisibleTags: OptionType[]\n            measuredHiddenTags: number\n            accumulatedWidth: number\n            lastVisibleElementWidth: number\n            lastVisibleLabel: ReactNode\n          },\n          currentValue,\n          index,\n        ) => {\n          const elementWidth = (currentValue as HTMLDivElement).offsetWidth\n\n          const newAccumulatedWidth =\n            accumulator.accumulatedWidth + elementWidth + SIZES_TAG.gap\n\n          const canBeVisible = newAccumulatedWidth <= innerWidth\n\n          return {\n            accumulatedWidth: !canBeVisible\n              ? accumulator.accumulatedWidth\n              : newAccumulatedWidth,\n            lastVisibleElementWidth: canBeVisible\n              ? elementWidth\n              : accumulator.lastVisibleElementWidth,\n            lastVisibleLabel: canBeVisible\n              ? potentiallyNonOverflowedValues[index].label\n              : accumulator.lastVisibleLabel,\n            measuredHiddenTags:\n              accumulator.measuredHiddenTags + (!canBeVisible ? 1 : 0),\n            measuredVisibleTags: [\n              ...accumulator.measuredVisibleTags,\n              canBeVisible && potentiallyNonOverflowedValues[index],\n            ].filter(Boolean) as OptionType[],\n          }\n        },\n        {\n          accumulatedWidth: 0,\n          lastVisibleElementWidth: 0,\n          lastVisibleLabel: '',\n          measuredHiddenTags: 0,\n          measuredVisibleTags: [],\n        },\n      )\n\n      const additionnalElementsWidth =\n        SIZES_TAG.paddings + (refPlusTag.current?.offsetWidth ?? 0)\n      const finalWidth =\n        accumulatedWidth + (measuredHiddenTags ? additionnalElementsWidth : 0)\n\n      const overflowPx = finalWidth - innerWidth\n      const hasOverflow = overflowPx > 0\n      const hasHiddenTags = measuredHiddenTags > 0\n      const lastVisibleElementMaxSize = lastVisibleElementWidth - overflowPx\n\n      // If only one element is selected and it is hidden, we need to show it\n      if (measuredHiddenTags === 1 && measuredVisibleTags.length === 0) {\n        setOverflowAmount(0)\n        setNonOverFlowedValues([potentiallyNonOverflowedValues[0]])\n\n        const newOverflowPx =\n          lastVisibleElementWidth +\n          (measuredHiddenTags > 1 ? additionnalElementsWidth : 0) -\n          innerWidth\n        setLastElementMaxWidth(lastVisibleElementWidth - newOverflowPx)\n        setOverflow(true)\n      }\n\n      // If it overflows with the last tag, we need to add an ellipsis to the last element if there is enough space (>60px)\n      // and if it is a string (do not cut ReactNode label)\n      // else we hide it completely and add it to the overflow amount\n      else if (\n        hasOverflow &&\n        hasHiddenTags &&\n        (lastVisibleElementMaxSize > 65 ||\n          (measuredVisibleTags.length === 1 &&\n            lastVisibleElementMaxSize > 65)) &&\n        typeof lastVisibleLabel === 'string'\n      ) {\n        setLastElementMaxWidth(lastVisibleElementMaxSize)\n        setOverflow(true)\n        setOverflowAmount(measuredHiddenTags)\n        setNonOverFlowedValues(measuredVisibleTags)\n      } else if (hasOverflow && hasHiddenTags) {\n        setLastElementMaxWidth(0)\n        setOverflow(false)\n        setOverflowAmount(measuredHiddenTags + 1)\n        setNonOverFlowedValues(measuredVisibleTags.slice(0, -1))\n      }\n      // Otherwise, we have enough space to show all tags\n      else {\n        setOverflow(false)\n        setOverflowAmount(measuredHiddenTags)\n        setNonOverFlowedValues(measuredVisibleTags)\n      }\n    }\n  }, [\n    selectedData.selectedValues.length,\n    innerWidth,\n    potentiallyNonOverflowedValues,\n  ])\n\n  useEffect(() => {\n    setSelectedData({ type: 'update' })\n  }, [setSelectedData, options])\n\n  useEffect(() => {\n    const getWidth = () => {\n      if (refTag.current) {\n        setInnerWidth(refTag.current.offsetWidth)\n      } else {\n        setInnerWidth(\n          innerRef.current?.offsetWidth ??\n            0 - (arrowRef.current?.offsetWidth ?? 0) - SIZES_TAG.paddings,\n        )\n      }\n    }\n    getWidth()\n    window.addEventListener('resize', getWidth)\n\n    return () => window.removeEventListener('resize', getWidth)\n  }, [innerRef, refTag, selectedData.selectedValues])\n\n  const shouldDisplayValues = useMemo(() => {\n    if (multiselect) {\n      return (\n        potentiallyNonOverflowedValues.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    options,\n    potentiallyNonOverflowedValues.length,\n    selectedData.selectedValues,\n  ])\n\n  return (\n    <Tooltip text={tooltip}>\n      <StyledInputWrapper\n        aria-controls={dropdownId}\n        aria-expanded={isDropdownVisible}\n        aria-label={label}\n        autoFocus={autoFocus}\n        data-disabled={disabled}\n        data-dropdownvisible={isDropdownVisible}\n        data-readonly={readOnly}\n        data-size={size}\n        data-state={state}\n        data-testid={dataTestId}\n        id={id}\n        onClick={\n          openable ? () => setIsDropdownVisible(!isDropdownVisible) : undefined\n        }\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        role=\"combobox\"\n        tabIndex={0}\n      >\n        {shouldDisplayValues ? (\n          <DisplayValues\n            disabled={disabled}\n            lastElementMaxWidth={lastElementMaxWidth}\n            measureRef={measureRef}\n            nonOverflowedValues={nonOverflowedValues}\n            overflow={overflow}\n            overflowAmount={overflowAmount}\n            overflowed={!!overflowAmount}\n            potentiallyNonOverflowedValues={potentiallyNonOverflowedValues}\n            readOnly={readOnly}\n            refPlusTag={refPlusTag}\n            refTag={refTag}\n            size={size}\n          />\n        ) : (\n          <Placeholder\n            as=\"p\"\n            disabled={disabled}\n            prominence=\"weak\"\n            sentiment=\"neutral\"\n            variant={size === 'large' ? 'body' : 'bodySmall'}\n          >\n            {placeholder}\n          </Placeholder>\n        )}\n        <StateStack alignItems=\"center\" direction=\"row\" gap={1} ref={arrowRef}>\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              data-testid=\"clear-all\"\n              disabled={disabled || !selectedData.selectedValues[0] || readOnly}\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              size=\"small\"\n              variant=\"ghost\"\n            >\n              <CloseIcon />\n            </Button>\n          ) : null}\n          <ArrowDownIcon\n            aria-label=\"show dropdown\"\n            disabled={disabled || readOnly}\n            sentiment=\"neutral\"\n            size=\"small\"\n          />\n        </StateStack>\n      </StyledInputWrapper>\n    </Tooltip>\n  )\n}\n\nexport { SelectBar, StyledInputWrapper }\n"]} */"));
102
+ }) => 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/SelectInput/SelectBar.tsx"],"names":[],"mappings":"AA8EE","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInput/SelectBar.tsx","sourcesContent":["'use client'\n\nimport styled from '@emotion/styled'\nimport {\n  AlertCircleIcon,\n  ArrowDownIcon,\n  CheckCircleIcon,\n  CloseIcon,\n  PlusIcon,\n} from '@ultraviolet/icons'\nimport type { ReactNode, RefObject } from 'react'\nimport { useEffect, useLayoutEffect, useMemo, useRef, useState } from 'react'\nimport { Button } from '../Button'\nimport { StyledChildrenContainer } from '../Popup'\nimport { Stack } from '../Stack'\nimport { Tag } from '../Tag'\nimport { Text } from '../Text'\nimport { Tooltip } from '../Tooltip'\nimport { findOptionInOptions } from './findOptionInOptions'\nimport { useSelectInput } from './SelectInputProvider'\nimport type { OptionType } from './types'\nimport { INPUT_SIZE_HEIGHT } from './types'\n\nconst SIZES_TAG = {\n  gap: 8,\n  paddings: 16,\n  plusTag: 48,\n}\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  dropdownId?: string\n}\n\ntype DisplayValuesProps = {\n  refTag: RefObject<HTMLDivElement | null>\n  nonOverflowedValues: OptionType[]\n  potentiallyNonOverflowedValues: OptionType[]\n  disabled: boolean\n  readOnly: boolean\n  overflowed: boolean\n  overflowAmount: number\n  measureRef: RefObject<HTMLDivElement | null>\n  size: 'small' | 'medium' | 'large'\n  lastElementMaxWidth: number\n  overflow?: boolean\n  refPlusTag: RefObject<HTMLDivElement | null>\n  displayShadowCopy?: boolean\n}\n\nconst StateStack = styled(Stack)`\n  padding-right: ${({ theme }) => theme.space['2']};\n  display: flex;\n`\nconst Placeholder = styled(Text)`\nuser-select: none;\nalign-self: center;\n`\n\nconst StyledInputWrapper = styled.div<{\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: grid;\n  width: 100%;\n  gap: ${({ theme }) => theme.space[1]};\n  grid-template-columns: 1fr auto ;\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\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  shouldForwardProp: prop => !['lastElementMaxWidth', 'hidden'].includes(prop),\n})<{\n  lastElementMaxWidth?: number\n  hidden?: boolean\n}>`\n  height: max-content;\n  width: fit-content;\n  min-width: ${({ lastElementMaxWidth }) =>\n    lastElementMaxWidth ? 'auto' : 'fit-content'};\n\n  max-width: ${({ lastElementMaxWidth, hidden }) =>\n    lastElementMaxWidth && !hidden ? `${lastElementMaxWidth}px` : '100%'};\n\n  ${({ hidden }) =>\n    hidden\n      ? 'visibility: hidden;'\n      : `\n  text-overflow: ellipsis;\n  overflow: hidden;`}\n\n  & > ${StyledChildrenContainer} {\n    overflow: hidden;\n  }\n`\n\nconst SelectedValues = styled(Text)`\n  text-overflow: ellipsis;\n  overflow: hidden;\n  white-space: nowrap;\n  align-self: center;\n`\nconst PlusTag = styled(Tag)`\nwidth: ${({ theme }) => theme.sizing[500]};\n;\n`\n\nconst MultiselectStack = styled(Stack)`\noverflow: hidden;\nmax-width: 100%;\nheight: 100%;\n`\n\nconst DisplayValues = ({\n  refTag,\n  nonOverflowedValues,\n  potentiallyNonOverflowedValues,\n  disabled,\n  readOnly,\n  overflowed,\n  overflowAmount,\n  size,\n  measureRef,\n  lastElementMaxWidth,\n  overflow,\n  refPlusTag,\n  displayShadowCopy,\n}: DisplayValuesProps) => {\n  const { multiselect, selectedData, setSelectedData, options, onChange } =\n    useSelectInput()\n\n  return multiselect ? (\n    <MultiselectStack\n      alignItems=\"center\"\n      direction=\"row\"\n      gap=\"1\"\n      ref={refTag}\n      wrap=\"nowrap\"\n    >\n      {/* Hidden div to measure the width of the tags */}\n      {displayShadowCopy ? (\n        <div\n          ref={measureRef}\n          style={{\n            position: 'absolute',\n          }}\n        >\n          {potentiallyNonOverflowedValues.map(option => (\n            <CustomTag\n              className={option.value}\n              hidden\n              key={option.value}\n              onClose={() => {}}\n            >\n              {option?.label}\n            </CustomTag>\n          ))}\n        </div>\n      ) : null}\n      {nonOverflowedValues.map((option, index) => (\n        <CustomTag\n          data-testid=\"selected-options-tags\"\n          disabled={disabled}\n          key={option?.value}\n          lastElementMaxWidth={\n            index === nonOverflowedValues.length - 1 && overflow\n              ? lastElementMaxWidth\n              : 0\n          }\n          onClose={\n            !readOnly\n              ? event => {\n                  event.stopPropagation()\n                  setSelectedData({\n                    clickedOption: option,\n                    type: 'selectOption',\n                  })\n                  const newSelectedValues = selectedData.selectedValues?.filter(\n                    val => val !== option.value,\n                  )\n                  onChange?.(newSelectedValues)\n                }\n              : undefined\n          }\n          sentiment=\"neutral\"\n        >\n          {option?.label}\n        </CustomTag>\n      ))}\n\n      {overflowed ? (\n        <Stack justifyContent=\"center\" ref={refPlusTag}>\n          <PlusTag\n            aria-label=\"Plus tag\"\n            data-testid=\"plus-tag\"\n            disabled={disabled}\n            key=\"+\"\n            sentiment=\"neutral\"\n          >\n            <PlusIcon size=\"xsmall\" />\n            {overflowAmount}\n          </PlusTag>\n        </Stack>\n      ) : null}\n    </MultiselectStack>\n  ) : (\n    <SelectedValues\n      as=\"div\"\n      disabled={disabled}\n      prominence=\"default\"\n      sentiment=\"neutral\"\n      variant={size === 'large' ? 'body' : 'bodySmall'}\n    >\n      {selectedData.selectedValues[0]\n        ? findOptionInOptions(options, selectedData.selectedValues[0])?.label\n        : null}\n    </SelectedValues>\n  )\n}\n\nconst 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  dropdownId,\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 [displayShadowCopy, setDisplayShadowCopy] = useState(false)\n  const refTag = useRef<HTMLDivElement>(null)\n  const measureRef = useRef<HTMLDivElement>(null)\n  const arrowRef = useRef<HTMLDivElement>(null)\n  const refPlusTag = useRef<HTMLDivElement>(null)\n  // width - width of the arrow (in px) - padding between tags (in px)\n  const [innerWidth, setInnerWidth] = useState(\n    innerRef.current?.offsetWidth ??\n      0 - (arrowRef.current?.offsetWidth ?? 0) - SIZES_TAG.paddings,\n  )\n  const [overflowAmount, setOverflowAmount] = useState(0)\n  const [overflow, setOverflow] = useState(false)\n  const [lastElementMaxWidth, setLastElementMaxWidth] = 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 potentiallyNonOverflowedValues = useMemo(\n    () =>\n      selectedData.selectedValues\n        .map(selectedValue => findOptionInOptions(options, selectedValue))\n        .filter((option): option is OptionType => !!option),\n    [options, selectedData.selectedValues],\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  // First we display shadow copy of tags to render it\n  useEffect(() => {\n    setDisplayShadowCopy(true)\n  }, [selectedData.selectedValues.length])\n\n  // We then want to measure the tags length before displaying them\n  // so we can determine if there is an overflow or not\n  // We use useLayoutEffect to ensure the measurement is done before the browser paints\n  useLayoutEffect(() => {\n    if (selectedData.selectedValues.length === 0) {\n      setOverflowAmount(0)\n      setNonOverFlowedValues([])\n    }\n\n    if (measureRef.current && selectedData.selectedValues.length > 0) {\n      const toMeasureElements: HTMLCollection = measureRef.current.children\n      const toMeasureElementsArray = [...toMeasureElements]\n\n      const {\n        measuredVisibleTags,\n        measuredHiddenTags,\n        accumulatedWidth,\n        lastVisibleElementWidth,\n        lastVisibleLabel,\n      } = toMeasureElementsArray.reduce(\n        (\n          accumulator: {\n            measuredVisibleTags: OptionType[]\n            measuredHiddenTags: number\n            accumulatedWidth: number\n            lastVisibleElementWidth: number\n            lastVisibleLabel: ReactNode\n          },\n          currentValue,\n          index,\n        ) => {\n          const elementWidth = (currentValue as HTMLDivElement).offsetWidth\n\n          const newAccumulatedWidth =\n            accumulator.accumulatedWidth + elementWidth + SIZES_TAG.gap\n\n          const canBeVisible = newAccumulatedWidth <= innerWidth\n\n          return {\n            accumulatedWidth: !canBeVisible\n              ? accumulator.accumulatedWidth\n              : newAccumulatedWidth,\n            lastVisibleElementWidth: canBeVisible\n              ? elementWidth\n              : accumulator.lastVisibleElementWidth,\n            lastVisibleLabel: canBeVisible\n              ? potentiallyNonOverflowedValues[index].label\n              : accumulator.lastVisibleLabel,\n            measuredHiddenTags:\n              accumulator.measuredHiddenTags + (!canBeVisible ? 1 : 0),\n            measuredVisibleTags: [\n              ...accumulator.measuredVisibleTags,\n              canBeVisible && potentiallyNonOverflowedValues[index],\n            ].filter(Boolean) as OptionType[],\n          }\n        },\n        {\n          accumulatedWidth: 0,\n          lastVisibleElementWidth: 0,\n          lastVisibleLabel: '',\n          measuredHiddenTags: 0,\n          measuredVisibleTags: [],\n        },\n      )\n\n      const additionnalElementsWidth =\n        SIZES_TAG.paddings + (refPlusTag.current?.offsetWidth ?? 0)\n      const finalWidth =\n        accumulatedWidth + (measuredHiddenTags ? additionnalElementsWidth : 0)\n\n      const overflowPx = finalWidth - innerWidth\n      const hasOverflow = overflowPx > 0\n      const hasHiddenTags = measuredHiddenTags > 0\n      const lastVisibleElementMaxSize = lastVisibleElementWidth - overflowPx\n\n      // If only one element is selected and it is hidden, we need to show it\n      if (measuredHiddenTags === 1 && measuredVisibleTags.length === 0) {\n        setOverflowAmount(0)\n        setNonOverFlowedValues([potentiallyNonOverflowedValues[0]])\n\n        const newOverflowPx =\n          lastVisibleElementWidth +\n          (measuredHiddenTags > 1 ? additionnalElementsWidth : 0) -\n          innerWidth\n        setLastElementMaxWidth(lastVisibleElementWidth - newOverflowPx)\n        setOverflow(true)\n      }\n\n      // If it overflows with the last tag, we need to add an ellipsis to the last element if there is enough space (>60px)\n      // and if it is a string (do not cut ReactNode label)\n      // else we hide it completely and add it to the overflow amount\n      else if (\n        hasOverflow &&\n        hasHiddenTags &&\n        (lastVisibleElementMaxSize > 65 ||\n          (measuredVisibleTags.length === 1 &&\n            lastVisibleElementMaxSize > 65)) &&\n        typeof lastVisibleLabel === 'string'\n      ) {\n        setLastElementMaxWidth(lastVisibleElementMaxSize)\n        setOverflow(true)\n        setOverflowAmount(measuredHiddenTags)\n        setNonOverFlowedValues(measuredVisibleTags)\n      } else if (hasOverflow && hasHiddenTags) {\n        setLastElementMaxWidth(0)\n        setOverflow(false)\n        setOverflowAmount(measuredHiddenTags + 1)\n        setNonOverFlowedValues(measuredVisibleTags.slice(0, -1))\n      }\n      // Otherwise, we have enough space to show all tags\n      else {\n        setOverflow(false)\n        setOverflowAmount(measuredHiddenTags)\n        setNonOverFlowedValues(measuredVisibleTags)\n      }\n    }\n    setDisplayShadowCopy(false)\n  }, [\n    displayShadowCopy,\n    innerWidth,\n    potentiallyNonOverflowedValues,\n    selectedData.selectedValues.length,\n  ])\n\n  useEffect(() => {\n    setSelectedData({ type: 'update' })\n  }, [setSelectedData, options])\n\n  useEffect(() => {\n    const getWidth = () => {\n      if (refTag.current) {\n        setInnerWidth(refTag.current.offsetWidth)\n      } else {\n        setInnerWidth(\n          innerRef.current?.offsetWidth ??\n            0 - (arrowRef.current?.offsetWidth ?? 0) - SIZES_TAG.paddings,\n        )\n      }\n    }\n    getWidth()\n    window.addEventListener('resize', getWidth)\n\n    return () => window.removeEventListener('resize', getWidth)\n  }, [innerRef, refTag, selectedData.selectedValues])\n\n  const shouldDisplayValues = useMemo(() => {\n    if (multiselect) {\n      return (\n        potentiallyNonOverflowedValues.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    options,\n    potentiallyNonOverflowedValues.length,\n    selectedData.selectedValues,\n  ])\n\n  return (\n    <Tooltip text={tooltip}>\n      <StyledInputWrapper\n        aria-controls={dropdownId}\n        aria-expanded={isDropdownVisible}\n        aria-label={label}\n        autoFocus={autoFocus}\n        data-disabled={disabled}\n        data-dropdownvisible={isDropdownVisible}\n        data-readonly={readOnly}\n        data-size={size}\n        data-state={state}\n        data-testid={dataTestId}\n        id={id}\n        onClick={\n          openable ? () => setIsDropdownVisible(!isDropdownVisible) : undefined\n        }\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        role=\"combobox\"\n        tabIndex={0}\n      >\n        {shouldDisplayValues ? (\n          <DisplayValues\n            disabled={disabled}\n            displayShadowCopy={displayShadowCopy}\n            lastElementMaxWidth={lastElementMaxWidth}\n            measureRef={measureRef}\n            nonOverflowedValues={nonOverflowedValues}\n            overflow={overflow}\n            overflowAmount={overflowAmount}\n            overflowed={!!overflowAmount}\n            potentiallyNonOverflowedValues={potentiallyNonOverflowedValues}\n            readOnly={readOnly}\n            refPlusTag={refPlusTag}\n            refTag={refTag}\n            size={size}\n          />\n        ) : (\n          <Placeholder\n            as=\"p\"\n            disabled={disabled}\n            prominence=\"weak\"\n            sentiment=\"neutral\"\n            variant={size === 'large' ? 'body' : 'bodySmall'}\n          >\n            {placeholder}\n          </Placeholder>\n        )}\n        <StateStack alignItems=\"center\" direction=\"row\" gap={1} ref={arrowRef}>\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              data-testid=\"clear-all\"\n              disabled={disabled || !selectedData.selectedValues[0] || readOnly}\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              size=\"small\"\n              variant=\"ghost\"\n            >\n              <CloseIcon />\n            </Button>\n          ) : null}\n          <ArrowDownIcon\n            aria-label=\"show dropdown\"\n            disabled={disabled || readOnly}\n            sentiment=\"neutral\"\n            size=\"small\"\n          />\n        </StateStack>\n      </StyledInputWrapper>\n    </Tooltip>\n  )\n}\n\nexport { SelectBar, StyledInputWrapper }\n"]} */"));
103
103
  const CustomTag = /* @__PURE__ */ _styled(Tag, process.env.NODE_ENV === "production" ? {
104
104
  shouldForwardProp: (prop) => !["lastElementMaxWidth", "hidden"].includes(prop),
105
105
  target: "e1pocxam3"
@@ -116,7 +116,7 @@ const CustomTag = /* @__PURE__ */ _styled(Tag, process.env.NODE_ENV === "product
116
116
  hidden
117
117
  }) => hidden ? "visibility: hidden;" : `
118
118
  text-overflow: ellipsis;
119
- overflow: hidden;`, " &>", StyledChildrenContainer, "{overflow:hidden;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInput/SelectBar.tsx"],"names":[],"mappings":"AA0KE","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInput/SelectBar.tsx","sourcesContent":["'use client'\n\nimport styled from '@emotion/styled'\nimport {\n  AlertCircleIcon,\n  ArrowDownIcon,\n  CheckCircleIcon,\n  CloseIcon,\n  PlusIcon,\n} from '@ultraviolet/icons'\nimport type { ReactNode, RefObject } from 'react'\nimport { useEffect, useMemo, useRef, useState } from 'react'\nimport { Button } from '../Button'\nimport { StyledChildrenContainer } from '../Popup'\nimport { Stack } from '../Stack'\nimport { Tag } from '../Tag'\nimport { Text } from '../Text'\nimport { Tooltip } from '../Tooltip'\nimport { findOptionInOptions } from './findOptionInOptions'\nimport { useSelectInput } from './SelectInputProvider'\nimport type { OptionType } from './types'\nimport { INPUT_SIZE_HEIGHT } from './types'\n\nconst SIZES_TAG = {\n  gap: 8,\n  paddings: 16,\n  plusTag: 48,\n}\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  dropdownId?: string\n}\n\ntype DisplayValuesProps = {\n  refTag: RefObject<HTMLDivElement | null>\n  nonOverflowedValues: OptionType[]\n  potentiallyNonOverflowedValues: OptionType[]\n  disabled: boolean\n  readOnly: boolean\n  overflowed: boolean\n  overflowAmount: number\n  measureRef: RefObject<HTMLDivElement | null>\n  size: 'small' | 'medium' | 'large'\n  lastElementMaxWidth: number\n  overflow?: boolean\n  refPlusTag: RefObject<HTMLDivElement | null>\n}\n\nconst StateStack = styled(Stack)`\n  padding-right: ${({ theme }) => theme.space['2']};\n  display: flex;\n`\nconst Placeholder = styled(Text)`\nuser-select: none;\nalign-self: center;\n`\n\nconst StyledInputWrapper = styled.div<{\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: grid;\n  width: 100%;\n  gap: ${({ theme }) => theme.space[1]};\n  grid-template-columns: 1fr auto ;\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\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  shouldForwardProp: prop => !['lastElementMaxWidth', 'hidden'].includes(prop),\n})<{\n  lastElementMaxWidth?: number\n  hidden?: boolean\n}>`\n  height: max-content;\n  width: fit-content;\n  min-width: ${({ lastElementMaxWidth }) =>\n    lastElementMaxWidth ? 'auto' : 'fit-content'};\n\n  max-width: ${({ lastElementMaxWidth, hidden }) =>\n    lastElementMaxWidth && !hidden ? `${lastElementMaxWidth}px` : '100%'};\n\n  ${({ hidden }) =>\n    hidden\n      ? 'visibility: hidden;'\n      : `\n  text-overflow: ellipsis;\n  overflow: hidden;`}\n\n  & > ${StyledChildrenContainer} {\n    overflow: hidden;\n  }\n`\n\nconst SelectedValues = styled(Text)`\n  text-overflow: ellipsis;\n  overflow: hidden;\n  white-space: nowrap;\n  align-self: center;\n`\nconst PlusTag = styled(Tag)`\nwidth: ${({ theme }) => theme.sizing[500]};\n;\n`\n\nconst MultiselectStack = styled(Stack)`\noverflow: hidden;\nmax-width: 100%;\nheight: 100%;\n`\n\nconst DisplayValues = ({\n  refTag,\n  nonOverflowedValues,\n  potentiallyNonOverflowedValues,\n  disabled,\n  readOnly,\n  overflowed,\n  overflowAmount,\n  size,\n  measureRef,\n  lastElementMaxWidth,\n  overflow,\n  refPlusTag,\n}: DisplayValuesProps) => {\n  const { multiselect, selectedData, setSelectedData, options, onChange } =\n    useSelectInput()\n\n  return multiselect ? (\n    <MultiselectStack\n      alignItems=\"center\"\n      direction=\"row\"\n      gap=\"1\"\n      ref={refTag}\n      wrap=\"nowrap\"\n    >\n      {/* Hidden div to measure the width of the tags */}\n      <div\n        ref={measureRef}\n        style={{\n          position: 'absolute',\n        }}\n      >\n        {potentiallyNonOverflowedValues.map(option => (\n          <CustomTag\n            className={option.value}\n            hidden\n            key={option.value}\n            onClose={() => {}}\n          >\n            {option?.label}\n          </CustomTag>\n        ))}\n      </div>\n      {nonOverflowedValues.map((option, index) => (\n        <CustomTag\n          data-testid=\"selected-options-tags\"\n          disabled={disabled}\n          key={option?.value}\n          lastElementMaxWidth={\n            index === nonOverflowedValues.length - 1 && overflow\n              ? lastElementMaxWidth\n              : 0\n          }\n          onClose={\n            !readOnly\n              ? event => {\n                  event.stopPropagation()\n                  setSelectedData({\n                    clickedOption: option,\n                    type: 'selectOption',\n                  })\n                  const newSelectedValues = selectedData.selectedValues?.filter(\n                    val => val !== option.value,\n                  )\n                  onChange?.(newSelectedValues)\n                }\n              : undefined\n          }\n          sentiment=\"neutral\"\n        >\n          {option?.label}\n        </CustomTag>\n      ))}\n\n      {overflowed ? (\n        <Stack justifyContent=\"center\" ref={refPlusTag}>\n          <PlusTag\n            aria-label=\"Plus tag\"\n            data-testid=\"plus-tag\"\n            disabled={disabled}\n            key=\"+\"\n            sentiment=\"neutral\"\n          >\n            <PlusIcon size=\"xsmall\" />\n            {overflowAmount}\n          </PlusTag>\n        </Stack>\n      ) : null}\n    </MultiselectStack>\n  ) : (\n    <SelectedValues\n      as=\"div\"\n      disabled={disabled}\n      prominence=\"default\"\n      sentiment=\"neutral\"\n      variant={size === 'large' ? 'body' : 'bodySmall'}\n    >\n      {selectedData.selectedValues[0]\n        ? findOptionInOptions(options, selectedData.selectedValues[0])?.label\n        : null}\n    </SelectedValues>\n  )\n}\n\nconst 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  dropdownId,\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 measureRef = useRef<HTMLDivElement>(null)\n  const arrowRef = useRef<HTMLDivElement>(null)\n  const refPlusTag = useRef<HTMLDivElement>(null)\n  // width - width of the arrow (in px) - padding between tags (in px)\n  const [innerWidth, setInnerWidth] = useState(\n    innerRef.current?.offsetWidth ??\n      0 - (arrowRef.current?.offsetWidth ?? 0) - SIZES_TAG.paddings,\n  )\n  const [overflowAmount, setOverflowAmount] = useState(0)\n  const [overflow, setOverflow] = useState(false)\n  const [lastElementMaxWidth, setLastElementMaxWidth] = 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 potentiallyNonOverflowedValues = useMemo(\n    () =>\n      selectedData.selectedValues\n        .map(selectedValue => findOptionInOptions(options, selectedValue))\n        .filter((option): option is OptionType => !!option),\n    [options, selectedData.selectedValues],\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    if (selectedData.selectedValues.length === 0) {\n      setOverflowAmount(0)\n      setNonOverFlowedValues([])\n    }\n    if (measureRef.current && selectedData.selectedValues.length > 0) {\n      const toMeasureElements: HTMLCollection = measureRef.current.children\n      const toMeasureElementsArray = [...toMeasureElements]\n\n      const {\n        measuredVisibleTags,\n        measuredHiddenTags,\n        accumulatedWidth,\n        lastVisibleElementWidth,\n        lastVisibleLabel,\n      } = toMeasureElementsArray.reduce(\n        (\n          accumulator: {\n            measuredVisibleTags: OptionType[]\n            measuredHiddenTags: number\n            accumulatedWidth: number\n            lastVisibleElementWidth: number\n            lastVisibleLabel: ReactNode\n          },\n          currentValue,\n          index,\n        ) => {\n          const elementWidth = (currentValue as HTMLDivElement).offsetWidth\n\n          const newAccumulatedWidth =\n            accumulator.accumulatedWidth + elementWidth + SIZES_TAG.gap\n\n          const canBeVisible = newAccumulatedWidth <= innerWidth\n\n          return {\n            accumulatedWidth: !canBeVisible\n              ? accumulator.accumulatedWidth\n              : newAccumulatedWidth,\n            lastVisibleElementWidth: canBeVisible\n              ? elementWidth\n              : accumulator.lastVisibleElementWidth,\n            lastVisibleLabel: canBeVisible\n              ? potentiallyNonOverflowedValues[index].label\n              : accumulator.lastVisibleLabel,\n            measuredHiddenTags:\n              accumulator.measuredHiddenTags + (!canBeVisible ? 1 : 0),\n            measuredVisibleTags: [\n              ...accumulator.measuredVisibleTags,\n              canBeVisible && potentiallyNonOverflowedValues[index],\n            ].filter(Boolean) as OptionType[],\n          }\n        },\n        {\n          accumulatedWidth: 0,\n          lastVisibleElementWidth: 0,\n          lastVisibleLabel: '',\n          measuredHiddenTags: 0,\n          measuredVisibleTags: [],\n        },\n      )\n\n      const additionnalElementsWidth =\n        SIZES_TAG.paddings + (refPlusTag.current?.offsetWidth ?? 0)\n      const finalWidth =\n        accumulatedWidth + (measuredHiddenTags ? additionnalElementsWidth : 0)\n\n      const overflowPx = finalWidth - innerWidth\n      const hasOverflow = overflowPx > 0\n      const hasHiddenTags = measuredHiddenTags > 0\n      const lastVisibleElementMaxSize = lastVisibleElementWidth - overflowPx\n\n      // If only one element is selected and it is hidden, we need to show it\n      if (measuredHiddenTags === 1 && measuredVisibleTags.length === 0) {\n        setOverflowAmount(0)\n        setNonOverFlowedValues([potentiallyNonOverflowedValues[0]])\n\n        const newOverflowPx =\n          lastVisibleElementWidth +\n          (measuredHiddenTags > 1 ? additionnalElementsWidth : 0) -\n          innerWidth\n        setLastElementMaxWidth(lastVisibleElementWidth - newOverflowPx)\n        setOverflow(true)\n      }\n\n      // If it overflows with the last tag, we need to add an ellipsis to the last element if there is enough space (>60px)\n      // and if it is a string (do not cut ReactNode label)\n      // else we hide it completely and add it to the overflow amount\n      else if (\n        hasOverflow &&\n        hasHiddenTags &&\n        (lastVisibleElementMaxSize > 65 ||\n          (measuredVisibleTags.length === 1 &&\n            lastVisibleElementMaxSize > 65)) &&\n        typeof lastVisibleLabel === 'string'\n      ) {\n        setLastElementMaxWidth(lastVisibleElementMaxSize)\n        setOverflow(true)\n        setOverflowAmount(measuredHiddenTags)\n        setNonOverFlowedValues(measuredVisibleTags)\n      } else if (hasOverflow && hasHiddenTags) {\n        setLastElementMaxWidth(0)\n        setOverflow(false)\n        setOverflowAmount(measuredHiddenTags + 1)\n        setNonOverFlowedValues(measuredVisibleTags.slice(0, -1))\n      }\n      // Otherwise, we have enough space to show all tags\n      else {\n        setOverflow(false)\n        setOverflowAmount(measuredHiddenTags)\n        setNonOverFlowedValues(measuredVisibleTags)\n      }\n    }\n  }, [\n    selectedData.selectedValues.length,\n    innerWidth,\n    potentiallyNonOverflowedValues,\n  ])\n\n  useEffect(() => {\n    setSelectedData({ type: 'update' })\n  }, [setSelectedData, options])\n\n  useEffect(() => {\n    const getWidth = () => {\n      if (refTag.current) {\n        setInnerWidth(refTag.current.offsetWidth)\n      } else {\n        setInnerWidth(\n          innerRef.current?.offsetWidth ??\n            0 - (arrowRef.current?.offsetWidth ?? 0) - SIZES_TAG.paddings,\n        )\n      }\n    }\n    getWidth()\n    window.addEventListener('resize', getWidth)\n\n    return () => window.removeEventListener('resize', getWidth)\n  }, [innerRef, refTag, selectedData.selectedValues])\n\n  const shouldDisplayValues = useMemo(() => {\n    if (multiselect) {\n      return (\n        potentiallyNonOverflowedValues.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    options,\n    potentiallyNonOverflowedValues.length,\n    selectedData.selectedValues,\n  ])\n\n  return (\n    <Tooltip text={tooltip}>\n      <StyledInputWrapper\n        aria-controls={dropdownId}\n        aria-expanded={isDropdownVisible}\n        aria-label={label}\n        autoFocus={autoFocus}\n        data-disabled={disabled}\n        data-dropdownvisible={isDropdownVisible}\n        data-readonly={readOnly}\n        data-size={size}\n        data-state={state}\n        data-testid={dataTestId}\n        id={id}\n        onClick={\n          openable ? () => setIsDropdownVisible(!isDropdownVisible) : undefined\n        }\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        role=\"combobox\"\n        tabIndex={0}\n      >\n        {shouldDisplayValues ? (\n          <DisplayValues\n            disabled={disabled}\n            lastElementMaxWidth={lastElementMaxWidth}\n            measureRef={measureRef}\n            nonOverflowedValues={nonOverflowedValues}\n            overflow={overflow}\n            overflowAmount={overflowAmount}\n            overflowed={!!overflowAmount}\n            potentiallyNonOverflowedValues={potentiallyNonOverflowedValues}\n            readOnly={readOnly}\n            refPlusTag={refPlusTag}\n            refTag={refTag}\n            size={size}\n          />\n        ) : (\n          <Placeholder\n            as=\"p\"\n            disabled={disabled}\n            prominence=\"weak\"\n            sentiment=\"neutral\"\n            variant={size === 'large' ? 'body' : 'bodySmall'}\n          >\n            {placeholder}\n          </Placeholder>\n        )}\n        <StateStack alignItems=\"center\" direction=\"row\" gap={1} ref={arrowRef}>\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              data-testid=\"clear-all\"\n              disabled={disabled || !selectedData.selectedValues[0] || readOnly}\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              size=\"small\"\n              variant=\"ghost\"\n            >\n              <CloseIcon />\n            </Button>\n          ) : null}\n          <ArrowDownIcon\n            aria-label=\"show dropdown\"\n            disabled={disabled || readOnly}\n            sentiment=\"neutral\"\n            size=\"small\"\n          />\n        </StateStack>\n      </StyledInputWrapper>\n    </Tooltip>\n  )\n}\n\nexport { SelectBar, StyledInputWrapper }\n"]} */"));
119
+ overflow: hidden;`, " &>", StyledChildrenContainer, "{overflow:hidden;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInput/SelectBar.tsx"],"names":[],"mappings":"AA2KE","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInput/SelectBar.tsx","sourcesContent":["'use client'\n\nimport styled from '@emotion/styled'\nimport {\n  AlertCircleIcon,\n  ArrowDownIcon,\n  CheckCircleIcon,\n  CloseIcon,\n  PlusIcon,\n} from '@ultraviolet/icons'\nimport type { ReactNode, RefObject } from 'react'\nimport { useEffect, useLayoutEffect, useMemo, useRef, useState } from 'react'\nimport { Button } from '../Button'\nimport { StyledChildrenContainer } from '../Popup'\nimport { Stack } from '../Stack'\nimport { Tag } from '../Tag'\nimport { Text } from '../Text'\nimport { Tooltip } from '../Tooltip'\nimport { findOptionInOptions } from './findOptionInOptions'\nimport { useSelectInput } from './SelectInputProvider'\nimport type { OptionType } from './types'\nimport { INPUT_SIZE_HEIGHT } from './types'\n\nconst SIZES_TAG = {\n  gap: 8,\n  paddings: 16,\n  plusTag: 48,\n}\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  dropdownId?: string\n}\n\ntype DisplayValuesProps = {\n  refTag: RefObject<HTMLDivElement | null>\n  nonOverflowedValues: OptionType[]\n  potentiallyNonOverflowedValues: OptionType[]\n  disabled: boolean\n  readOnly: boolean\n  overflowed: boolean\n  overflowAmount: number\n  measureRef: RefObject<HTMLDivElement | null>\n  size: 'small' | 'medium' | 'large'\n  lastElementMaxWidth: number\n  overflow?: boolean\n  refPlusTag: RefObject<HTMLDivElement | null>\n  displayShadowCopy?: boolean\n}\n\nconst StateStack = styled(Stack)`\n  padding-right: ${({ theme }) => theme.space['2']};\n  display: flex;\n`\nconst Placeholder = styled(Text)`\nuser-select: none;\nalign-self: center;\n`\n\nconst StyledInputWrapper = styled.div<{\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: grid;\n  width: 100%;\n  gap: ${({ theme }) => theme.space[1]};\n  grid-template-columns: 1fr auto ;\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\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  shouldForwardProp: prop => !['lastElementMaxWidth', 'hidden'].includes(prop),\n})<{\n  lastElementMaxWidth?: number\n  hidden?: boolean\n}>`\n  height: max-content;\n  width: fit-content;\n  min-width: ${({ lastElementMaxWidth }) =>\n    lastElementMaxWidth ? 'auto' : 'fit-content'};\n\n  max-width: ${({ lastElementMaxWidth, hidden }) =>\n    lastElementMaxWidth && !hidden ? `${lastElementMaxWidth}px` : '100%'};\n\n  ${({ hidden }) =>\n    hidden\n      ? 'visibility: hidden;'\n      : `\n  text-overflow: ellipsis;\n  overflow: hidden;`}\n\n  & > ${StyledChildrenContainer} {\n    overflow: hidden;\n  }\n`\n\nconst SelectedValues = styled(Text)`\n  text-overflow: ellipsis;\n  overflow: hidden;\n  white-space: nowrap;\n  align-self: center;\n`\nconst PlusTag = styled(Tag)`\nwidth: ${({ theme }) => theme.sizing[500]};\n;\n`\n\nconst MultiselectStack = styled(Stack)`\noverflow: hidden;\nmax-width: 100%;\nheight: 100%;\n`\n\nconst DisplayValues = ({\n  refTag,\n  nonOverflowedValues,\n  potentiallyNonOverflowedValues,\n  disabled,\n  readOnly,\n  overflowed,\n  overflowAmount,\n  size,\n  measureRef,\n  lastElementMaxWidth,\n  overflow,\n  refPlusTag,\n  displayShadowCopy,\n}: DisplayValuesProps) => {\n  const { multiselect, selectedData, setSelectedData, options, onChange } =\n    useSelectInput()\n\n  return multiselect ? (\n    <MultiselectStack\n      alignItems=\"center\"\n      direction=\"row\"\n      gap=\"1\"\n      ref={refTag}\n      wrap=\"nowrap\"\n    >\n      {/* Hidden div to measure the width of the tags */}\n      {displayShadowCopy ? (\n        <div\n          ref={measureRef}\n          style={{\n            position: 'absolute',\n          }}\n        >\n          {potentiallyNonOverflowedValues.map(option => (\n            <CustomTag\n              className={option.value}\n              hidden\n              key={option.value}\n              onClose={() => {}}\n            >\n              {option?.label}\n            </CustomTag>\n          ))}\n        </div>\n      ) : null}\n      {nonOverflowedValues.map((option, index) => (\n        <CustomTag\n          data-testid=\"selected-options-tags\"\n          disabled={disabled}\n          key={option?.value}\n          lastElementMaxWidth={\n            index === nonOverflowedValues.length - 1 && overflow\n              ? lastElementMaxWidth\n              : 0\n          }\n          onClose={\n            !readOnly\n              ? event => {\n                  event.stopPropagation()\n                  setSelectedData({\n                    clickedOption: option,\n                    type: 'selectOption',\n                  })\n                  const newSelectedValues = selectedData.selectedValues?.filter(\n                    val => val !== option.value,\n                  )\n                  onChange?.(newSelectedValues)\n                }\n              : undefined\n          }\n          sentiment=\"neutral\"\n        >\n          {option?.label}\n        </CustomTag>\n      ))}\n\n      {overflowed ? (\n        <Stack justifyContent=\"center\" ref={refPlusTag}>\n          <PlusTag\n            aria-label=\"Plus tag\"\n            data-testid=\"plus-tag\"\n            disabled={disabled}\n            key=\"+\"\n            sentiment=\"neutral\"\n          >\n            <PlusIcon size=\"xsmall\" />\n            {overflowAmount}\n          </PlusTag>\n        </Stack>\n      ) : null}\n    </MultiselectStack>\n  ) : (\n    <SelectedValues\n      as=\"div\"\n      disabled={disabled}\n      prominence=\"default\"\n      sentiment=\"neutral\"\n      variant={size === 'large' ? 'body' : 'bodySmall'}\n    >\n      {selectedData.selectedValues[0]\n        ? findOptionInOptions(options, selectedData.selectedValues[0])?.label\n        : null}\n    </SelectedValues>\n  )\n}\n\nconst 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  dropdownId,\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 [displayShadowCopy, setDisplayShadowCopy] = useState(false)\n  const refTag = useRef<HTMLDivElement>(null)\n  const measureRef = useRef<HTMLDivElement>(null)\n  const arrowRef = useRef<HTMLDivElement>(null)\n  const refPlusTag = useRef<HTMLDivElement>(null)\n  // width - width of the arrow (in px) - padding between tags (in px)\n  const [innerWidth, setInnerWidth] = useState(\n    innerRef.current?.offsetWidth ??\n      0 - (arrowRef.current?.offsetWidth ?? 0) - SIZES_TAG.paddings,\n  )\n  const [overflowAmount, setOverflowAmount] = useState(0)\n  const [overflow, setOverflow] = useState(false)\n  const [lastElementMaxWidth, setLastElementMaxWidth] = 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 potentiallyNonOverflowedValues = useMemo(\n    () =>\n      selectedData.selectedValues\n        .map(selectedValue => findOptionInOptions(options, selectedValue))\n        .filter((option): option is OptionType => !!option),\n    [options, selectedData.selectedValues],\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  // First we display shadow copy of tags to render it\n  useEffect(() => {\n    setDisplayShadowCopy(true)\n  }, [selectedData.selectedValues.length])\n\n  // We then want to measure the tags length before displaying them\n  // so we can determine if there is an overflow or not\n  // We use useLayoutEffect to ensure the measurement is done before the browser paints\n  useLayoutEffect(() => {\n    if (selectedData.selectedValues.length === 0) {\n      setOverflowAmount(0)\n      setNonOverFlowedValues([])\n    }\n\n    if (measureRef.current && selectedData.selectedValues.length > 0) {\n      const toMeasureElements: HTMLCollection = measureRef.current.children\n      const toMeasureElementsArray = [...toMeasureElements]\n\n      const {\n        measuredVisibleTags,\n        measuredHiddenTags,\n        accumulatedWidth,\n        lastVisibleElementWidth,\n        lastVisibleLabel,\n      } = toMeasureElementsArray.reduce(\n        (\n          accumulator: {\n            measuredVisibleTags: OptionType[]\n            measuredHiddenTags: number\n            accumulatedWidth: number\n            lastVisibleElementWidth: number\n            lastVisibleLabel: ReactNode\n          },\n          currentValue,\n          index,\n        ) => {\n          const elementWidth = (currentValue as HTMLDivElement).offsetWidth\n\n          const newAccumulatedWidth =\n            accumulator.accumulatedWidth + elementWidth + SIZES_TAG.gap\n\n          const canBeVisible = newAccumulatedWidth <= innerWidth\n\n          return {\n            accumulatedWidth: !canBeVisible\n              ? accumulator.accumulatedWidth\n              : newAccumulatedWidth,\n            lastVisibleElementWidth: canBeVisible\n              ? elementWidth\n              : accumulator.lastVisibleElementWidth,\n            lastVisibleLabel: canBeVisible\n              ? potentiallyNonOverflowedValues[index].label\n              : accumulator.lastVisibleLabel,\n            measuredHiddenTags:\n              accumulator.measuredHiddenTags + (!canBeVisible ? 1 : 0),\n            measuredVisibleTags: [\n              ...accumulator.measuredVisibleTags,\n              canBeVisible && potentiallyNonOverflowedValues[index],\n            ].filter(Boolean) as OptionType[],\n          }\n        },\n        {\n          accumulatedWidth: 0,\n          lastVisibleElementWidth: 0,\n          lastVisibleLabel: '',\n          measuredHiddenTags: 0,\n          measuredVisibleTags: [],\n        },\n      )\n\n      const additionnalElementsWidth =\n        SIZES_TAG.paddings + (refPlusTag.current?.offsetWidth ?? 0)\n      const finalWidth =\n        accumulatedWidth + (measuredHiddenTags ? additionnalElementsWidth : 0)\n\n      const overflowPx = finalWidth - innerWidth\n      const hasOverflow = overflowPx > 0\n      const hasHiddenTags = measuredHiddenTags > 0\n      const lastVisibleElementMaxSize = lastVisibleElementWidth - overflowPx\n\n      // If only one element is selected and it is hidden, we need to show it\n      if (measuredHiddenTags === 1 && measuredVisibleTags.length === 0) {\n        setOverflowAmount(0)\n        setNonOverFlowedValues([potentiallyNonOverflowedValues[0]])\n\n        const newOverflowPx =\n          lastVisibleElementWidth +\n          (measuredHiddenTags > 1 ? additionnalElementsWidth : 0) -\n          innerWidth\n        setLastElementMaxWidth(lastVisibleElementWidth - newOverflowPx)\n        setOverflow(true)\n      }\n\n      // If it overflows with the last tag, we need to add an ellipsis to the last element if there is enough space (>60px)\n      // and if it is a string (do not cut ReactNode label)\n      // else we hide it completely and add it to the overflow amount\n      else if (\n        hasOverflow &&\n        hasHiddenTags &&\n        (lastVisibleElementMaxSize > 65 ||\n          (measuredVisibleTags.length === 1 &&\n            lastVisibleElementMaxSize > 65)) &&\n        typeof lastVisibleLabel === 'string'\n      ) {\n        setLastElementMaxWidth(lastVisibleElementMaxSize)\n        setOverflow(true)\n        setOverflowAmount(measuredHiddenTags)\n        setNonOverFlowedValues(measuredVisibleTags)\n      } else if (hasOverflow && hasHiddenTags) {\n        setLastElementMaxWidth(0)\n        setOverflow(false)\n        setOverflowAmount(measuredHiddenTags + 1)\n        setNonOverFlowedValues(measuredVisibleTags.slice(0, -1))\n      }\n      // Otherwise, we have enough space to show all tags\n      else {\n        setOverflow(false)\n        setOverflowAmount(measuredHiddenTags)\n        setNonOverFlowedValues(measuredVisibleTags)\n      }\n    }\n    setDisplayShadowCopy(false)\n  }, [\n    displayShadowCopy,\n    innerWidth,\n    potentiallyNonOverflowedValues,\n    selectedData.selectedValues.length,\n  ])\n\n  useEffect(() => {\n    setSelectedData({ type: 'update' })\n  }, [setSelectedData, options])\n\n  useEffect(() => {\n    const getWidth = () => {\n      if (refTag.current) {\n        setInnerWidth(refTag.current.offsetWidth)\n      } else {\n        setInnerWidth(\n          innerRef.current?.offsetWidth ??\n            0 - (arrowRef.current?.offsetWidth ?? 0) - SIZES_TAG.paddings,\n        )\n      }\n    }\n    getWidth()\n    window.addEventListener('resize', getWidth)\n\n    return () => window.removeEventListener('resize', getWidth)\n  }, [innerRef, refTag, selectedData.selectedValues])\n\n  const shouldDisplayValues = useMemo(() => {\n    if (multiselect) {\n      return (\n        potentiallyNonOverflowedValues.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    options,\n    potentiallyNonOverflowedValues.length,\n    selectedData.selectedValues,\n  ])\n\n  return (\n    <Tooltip text={tooltip}>\n      <StyledInputWrapper\n        aria-controls={dropdownId}\n        aria-expanded={isDropdownVisible}\n        aria-label={label}\n        autoFocus={autoFocus}\n        data-disabled={disabled}\n        data-dropdownvisible={isDropdownVisible}\n        data-readonly={readOnly}\n        data-size={size}\n        data-state={state}\n        data-testid={dataTestId}\n        id={id}\n        onClick={\n          openable ? () => setIsDropdownVisible(!isDropdownVisible) : undefined\n        }\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        role=\"combobox\"\n        tabIndex={0}\n      >\n        {shouldDisplayValues ? (\n          <DisplayValues\n            disabled={disabled}\n            displayShadowCopy={displayShadowCopy}\n            lastElementMaxWidth={lastElementMaxWidth}\n            measureRef={measureRef}\n            nonOverflowedValues={nonOverflowedValues}\n            overflow={overflow}\n            overflowAmount={overflowAmount}\n            overflowed={!!overflowAmount}\n            potentiallyNonOverflowedValues={potentiallyNonOverflowedValues}\n            readOnly={readOnly}\n            refPlusTag={refPlusTag}\n            refTag={refTag}\n            size={size}\n          />\n        ) : (\n          <Placeholder\n            as=\"p\"\n            disabled={disabled}\n            prominence=\"weak\"\n            sentiment=\"neutral\"\n            variant={size === 'large' ? 'body' : 'bodySmall'}\n          >\n            {placeholder}\n          </Placeholder>\n        )}\n        <StateStack alignItems=\"center\" direction=\"row\" gap={1} ref={arrowRef}>\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              data-testid=\"clear-all\"\n              disabled={disabled || !selectedData.selectedValues[0] || readOnly}\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              size=\"small\"\n              variant=\"ghost\"\n            >\n              <CloseIcon />\n            </Button>\n          ) : null}\n          <ArrowDownIcon\n            aria-label=\"show dropdown\"\n            disabled={disabled || readOnly}\n            sentiment=\"neutral\"\n            size=\"small\"\n          />\n        </StateStack>\n      </StyledInputWrapper>\n    </Tooltip>\n  )\n}\n\nexport { SelectBar, StyledInputWrapper }\n"]} */"));
120
120
  const SelectedValues = /* @__PURE__ */ _styled(Text, process.env.NODE_ENV === "production" ? {
121
121
  target: "e1pocxam2"
122
122
  } : {
@@ -127,7 +127,7 @@ const SelectedValues = /* @__PURE__ */ _styled(Text, process.env.NODE_ENV === "p
127
127
  styles: "text-overflow:ellipsis;overflow:hidden;white-space:nowrap;align-self:center"
128
128
  } : {
129
129
  name: "et8r3r",
130
- styles: "text-overflow:ellipsis;overflow:hidden;white-space:nowrap;align-self:center/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInput/SelectBar.tsx"],"names":[],"mappings":"AA+LmC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInput/SelectBar.tsx","sourcesContent":["'use client'\n\nimport styled from '@emotion/styled'\nimport {\n  AlertCircleIcon,\n  ArrowDownIcon,\n  CheckCircleIcon,\n  CloseIcon,\n  PlusIcon,\n} from '@ultraviolet/icons'\nimport type { ReactNode, RefObject } from 'react'\nimport { useEffect, useMemo, useRef, useState } from 'react'\nimport { Button } from '../Button'\nimport { StyledChildrenContainer } from '../Popup'\nimport { Stack } from '../Stack'\nimport { Tag } from '../Tag'\nimport { Text } from '../Text'\nimport { Tooltip } from '../Tooltip'\nimport { findOptionInOptions } from './findOptionInOptions'\nimport { useSelectInput } from './SelectInputProvider'\nimport type { OptionType } from './types'\nimport { INPUT_SIZE_HEIGHT } from './types'\n\nconst SIZES_TAG = {\n  gap: 8,\n  paddings: 16,\n  plusTag: 48,\n}\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  dropdownId?: string\n}\n\ntype DisplayValuesProps = {\n  refTag: RefObject<HTMLDivElement | null>\n  nonOverflowedValues: OptionType[]\n  potentiallyNonOverflowedValues: OptionType[]\n  disabled: boolean\n  readOnly: boolean\n  overflowed: boolean\n  overflowAmount: number\n  measureRef: RefObject<HTMLDivElement | null>\n  size: 'small' | 'medium' | 'large'\n  lastElementMaxWidth: number\n  overflow?: boolean\n  refPlusTag: RefObject<HTMLDivElement | null>\n}\n\nconst StateStack = styled(Stack)`\n  padding-right: ${({ theme }) => theme.space['2']};\n  display: flex;\n`\nconst Placeholder = styled(Text)`\nuser-select: none;\nalign-self: center;\n`\n\nconst StyledInputWrapper = styled.div<{\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: grid;\n  width: 100%;\n  gap: ${({ theme }) => theme.space[1]};\n  grid-template-columns: 1fr auto ;\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\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  shouldForwardProp: prop => !['lastElementMaxWidth', 'hidden'].includes(prop),\n})<{\n  lastElementMaxWidth?: number\n  hidden?: boolean\n}>`\n  height: max-content;\n  width: fit-content;\n  min-width: ${({ lastElementMaxWidth }) =>\n    lastElementMaxWidth ? 'auto' : 'fit-content'};\n\n  max-width: ${({ lastElementMaxWidth, hidden }) =>\n    lastElementMaxWidth && !hidden ? `${lastElementMaxWidth}px` : '100%'};\n\n  ${({ hidden }) =>\n    hidden\n      ? 'visibility: hidden;'\n      : `\n  text-overflow: ellipsis;\n  overflow: hidden;`}\n\n  & > ${StyledChildrenContainer} {\n    overflow: hidden;\n  }\n`\n\nconst SelectedValues = styled(Text)`\n  text-overflow: ellipsis;\n  overflow: hidden;\n  white-space: nowrap;\n  align-self: center;\n`\nconst PlusTag = styled(Tag)`\nwidth: ${({ theme }) => theme.sizing[500]};\n;\n`\n\nconst MultiselectStack = styled(Stack)`\noverflow: hidden;\nmax-width: 100%;\nheight: 100%;\n`\n\nconst DisplayValues = ({\n  refTag,\n  nonOverflowedValues,\n  potentiallyNonOverflowedValues,\n  disabled,\n  readOnly,\n  overflowed,\n  overflowAmount,\n  size,\n  measureRef,\n  lastElementMaxWidth,\n  overflow,\n  refPlusTag,\n}: DisplayValuesProps) => {\n  const { multiselect, selectedData, setSelectedData, options, onChange } =\n    useSelectInput()\n\n  return multiselect ? (\n    <MultiselectStack\n      alignItems=\"center\"\n      direction=\"row\"\n      gap=\"1\"\n      ref={refTag}\n      wrap=\"nowrap\"\n    >\n      {/* Hidden div to measure the width of the tags */}\n      <div\n        ref={measureRef}\n        style={{\n          position: 'absolute',\n        }}\n      >\n        {potentiallyNonOverflowedValues.map(option => (\n          <CustomTag\n            className={option.value}\n            hidden\n            key={option.value}\n            onClose={() => {}}\n          >\n            {option?.label}\n          </CustomTag>\n        ))}\n      </div>\n      {nonOverflowedValues.map((option, index) => (\n        <CustomTag\n          data-testid=\"selected-options-tags\"\n          disabled={disabled}\n          key={option?.value}\n          lastElementMaxWidth={\n            index === nonOverflowedValues.length - 1 && overflow\n              ? lastElementMaxWidth\n              : 0\n          }\n          onClose={\n            !readOnly\n              ? event => {\n                  event.stopPropagation()\n                  setSelectedData({\n                    clickedOption: option,\n                    type: 'selectOption',\n                  })\n                  const newSelectedValues = selectedData.selectedValues?.filter(\n                    val => val !== option.value,\n                  )\n                  onChange?.(newSelectedValues)\n                }\n              : undefined\n          }\n          sentiment=\"neutral\"\n        >\n          {option?.label}\n        </CustomTag>\n      ))}\n\n      {overflowed ? (\n        <Stack justifyContent=\"center\" ref={refPlusTag}>\n          <PlusTag\n            aria-label=\"Plus tag\"\n            data-testid=\"plus-tag\"\n            disabled={disabled}\n            key=\"+\"\n            sentiment=\"neutral\"\n          >\n            <PlusIcon size=\"xsmall\" />\n            {overflowAmount}\n          </PlusTag>\n        </Stack>\n      ) : null}\n    </MultiselectStack>\n  ) : (\n    <SelectedValues\n      as=\"div\"\n      disabled={disabled}\n      prominence=\"default\"\n      sentiment=\"neutral\"\n      variant={size === 'large' ? 'body' : 'bodySmall'}\n    >\n      {selectedData.selectedValues[0]\n        ? findOptionInOptions(options, selectedData.selectedValues[0])?.label\n        : null}\n    </SelectedValues>\n  )\n}\n\nconst 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  dropdownId,\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 measureRef = useRef<HTMLDivElement>(null)\n  const arrowRef = useRef<HTMLDivElement>(null)\n  const refPlusTag = useRef<HTMLDivElement>(null)\n  // width - width of the arrow (in px) - padding between tags (in px)\n  const [innerWidth, setInnerWidth] = useState(\n    innerRef.current?.offsetWidth ??\n      0 - (arrowRef.current?.offsetWidth ?? 0) - SIZES_TAG.paddings,\n  )\n  const [overflowAmount, setOverflowAmount] = useState(0)\n  const [overflow, setOverflow] = useState(false)\n  const [lastElementMaxWidth, setLastElementMaxWidth] = 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 potentiallyNonOverflowedValues = useMemo(\n    () =>\n      selectedData.selectedValues\n        .map(selectedValue => findOptionInOptions(options, selectedValue))\n        .filter((option): option is OptionType => !!option),\n    [options, selectedData.selectedValues],\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    if (selectedData.selectedValues.length === 0) {\n      setOverflowAmount(0)\n      setNonOverFlowedValues([])\n    }\n    if (measureRef.current && selectedData.selectedValues.length > 0) {\n      const toMeasureElements: HTMLCollection = measureRef.current.children\n      const toMeasureElementsArray = [...toMeasureElements]\n\n      const {\n        measuredVisibleTags,\n        measuredHiddenTags,\n        accumulatedWidth,\n        lastVisibleElementWidth,\n        lastVisibleLabel,\n      } = toMeasureElementsArray.reduce(\n        (\n          accumulator: {\n            measuredVisibleTags: OptionType[]\n            measuredHiddenTags: number\n            accumulatedWidth: number\n            lastVisibleElementWidth: number\n            lastVisibleLabel: ReactNode\n          },\n          currentValue,\n          index,\n        ) => {\n          const elementWidth = (currentValue as HTMLDivElement).offsetWidth\n\n          const newAccumulatedWidth =\n            accumulator.accumulatedWidth + elementWidth + SIZES_TAG.gap\n\n          const canBeVisible = newAccumulatedWidth <= innerWidth\n\n          return {\n            accumulatedWidth: !canBeVisible\n              ? accumulator.accumulatedWidth\n              : newAccumulatedWidth,\n            lastVisibleElementWidth: canBeVisible\n              ? elementWidth\n              : accumulator.lastVisibleElementWidth,\n            lastVisibleLabel: canBeVisible\n              ? potentiallyNonOverflowedValues[index].label\n              : accumulator.lastVisibleLabel,\n            measuredHiddenTags:\n              accumulator.measuredHiddenTags + (!canBeVisible ? 1 : 0),\n            measuredVisibleTags: [\n              ...accumulator.measuredVisibleTags,\n              canBeVisible && potentiallyNonOverflowedValues[index],\n            ].filter(Boolean) as OptionType[],\n          }\n        },\n        {\n          accumulatedWidth: 0,\n          lastVisibleElementWidth: 0,\n          lastVisibleLabel: '',\n          measuredHiddenTags: 0,\n          measuredVisibleTags: [],\n        },\n      )\n\n      const additionnalElementsWidth =\n        SIZES_TAG.paddings + (refPlusTag.current?.offsetWidth ?? 0)\n      const finalWidth =\n        accumulatedWidth + (measuredHiddenTags ? additionnalElementsWidth : 0)\n\n      const overflowPx = finalWidth - innerWidth\n      const hasOverflow = overflowPx > 0\n      const hasHiddenTags = measuredHiddenTags > 0\n      const lastVisibleElementMaxSize = lastVisibleElementWidth - overflowPx\n\n      // If only one element is selected and it is hidden, we need to show it\n      if (measuredHiddenTags === 1 && measuredVisibleTags.length === 0) {\n        setOverflowAmount(0)\n        setNonOverFlowedValues([potentiallyNonOverflowedValues[0]])\n\n        const newOverflowPx =\n          lastVisibleElementWidth +\n          (measuredHiddenTags > 1 ? additionnalElementsWidth : 0) -\n          innerWidth\n        setLastElementMaxWidth(lastVisibleElementWidth - newOverflowPx)\n        setOverflow(true)\n      }\n\n      // If it overflows with the last tag, we need to add an ellipsis to the last element if there is enough space (>60px)\n      // and if it is a string (do not cut ReactNode label)\n      // else we hide it completely and add it to the overflow amount\n      else if (\n        hasOverflow &&\n        hasHiddenTags &&\n        (lastVisibleElementMaxSize > 65 ||\n          (measuredVisibleTags.length === 1 &&\n            lastVisibleElementMaxSize > 65)) &&\n        typeof lastVisibleLabel === 'string'\n      ) {\n        setLastElementMaxWidth(lastVisibleElementMaxSize)\n        setOverflow(true)\n        setOverflowAmount(measuredHiddenTags)\n        setNonOverFlowedValues(measuredVisibleTags)\n      } else if (hasOverflow && hasHiddenTags) {\n        setLastElementMaxWidth(0)\n        setOverflow(false)\n        setOverflowAmount(measuredHiddenTags + 1)\n        setNonOverFlowedValues(measuredVisibleTags.slice(0, -1))\n      }\n      // Otherwise, we have enough space to show all tags\n      else {\n        setOverflow(false)\n        setOverflowAmount(measuredHiddenTags)\n        setNonOverFlowedValues(measuredVisibleTags)\n      }\n    }\n  }, [\n    selectedData.selectedValues.length,\n    innerWidth,\n    potentiallyNonOverflowedValues,\n  ])\n\n  useEffect(() => {\n    setSelectedData({ type: 'update' })\n  }, [setSelectedData, options])\n\n  useEffect(() => {\n    const getWidth = () => {\n      if (refTag.current) {\n        setInnerWidth(refTag.current.offsetWidth)\n      } else {\n        setInnerWidth(\n          innerRef.current?.offsetWidth ??\n            0 - (arrowRef.current?.offsetWidth ?? 0) - SIZES_TAG.paddings,\n        )\n      }\n    }\n    getWidth()\n    window.addEventListener('resize', getWidth)\n\n    return () => window.removeEventListener('resize', getWidth)\n  }, [innerRef, refTag, selectedData.selectedValues])\n\n  const shouldDisplayValues = useMemo(() => {\n    if (multiselect) {\n      return (\n        potentiallyNonOverflowedValues.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    options,\n    potentiallyNonOverflowedValues.length,\n    selectedData.selectedValues,\n  ])\n\n  return (\n    <Tooltip text={tooltip}>\n      <StyledInputWrapper\n        aria-controls={dropdownId}\n        aria-expanded={isDropdownVisible}\n        aria-label={label}\n        autoFocus={autoFocus}\n        data-disabled={disabled}\n        data-dropdownvisible={isDropdownVisible}\n        data-readonly={readOnly}\n        data-size={size}\n        data-state={state}\n        data-testid={dataTestId}\n        id={id}\n        onClick={\n          openable ? () => setIsDropdownVisible(!isDropdownVisible) : undefined\n        }\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        role=\"combobox\"\n        tabIndex={0}\n      >\n        {shouldDisplayValues ? (\n          <DisplayValues\n            disabled={disabled}\n            lastElementMaxWidth={lastElementMaxWidth}\n            measureRef={measureRef}\n            nonOverflowedValues={nonOverflowedValues}\n            overflow={overflow}\n            overflowAmount={overflowAmount}\n            overflowed={!!overflowAmount}\n            potentiallyNonOverflowedValues={potentiallyNonOverflowedValues}\n            readOnly={readOnly}\n            refPlusTag={refPlusTag}\n            refTag={refTag}\n            size={size}\n          />\n        ) : (\n          <Placeholder\n            as=\"p\"\n            disabled={disabled}\n            prominence=\"weak\"\n            sentiment=\"neutral\"\n            variant={size === 'large' ? 'body' : 'bodySmall'}\n          >\n            {placeholder}\n          </Placeholder>\n        )}\n        <StateStack alignItems=\"center\" direction=\"row\" gap={1} ref={arrowRef}>\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              data-testid=\"clear-all\"\n              disabled={disabled || !selectedData.selectedValues[0] || readOnly}\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              size=\"small\"\n              variant=\"ghost\"\n            >\n              <CloseIcon />\n            </Button>\n          ) : null}\n          <ArrowDownIcon\n            aria-label=\"show dropdown\"\n            disabled={disabled || readOnly}\n            sentiment=\"neutral\"\n            size=\"small\"\n          />\n        </StateStack>\n      </StyledInputWrapper>\n    </Tooltip>\n  )\n}\n\nexport { SelectBar, StyledInputWrapper }\n"]} */",
130
+ styles: "text-overflow:ellipsis;overflow:hidden;white-space:nowrap;align-self:center/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInput/SelectBar.tsx"],"names":[],"mappings":"AAgMmC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInput/SelectBar.tsx","sourcesContent":["'use client'\n\nimport styled from '@emotion/styled'\nimport {\n  AlertCircleIcon,\n  ArrowDownIcon,\n  CheckCircleIcon,\n  CloseIcon,\n  PlusIcon,\n} from '@ultraviolet/icons'\nimport type { ReactNode, RefObject } from 'react'\nimport { useEffect, useLayoutEffect, useMemo, useRef, useState } from 'react'\nimport { Button } from '../Button'\nimport { StyledChildrenContainer } from '../Popup'\nimport { Stack } from '../Stack'\nimport { Tag } from '../Tag'\nimport { Text } from '../Text'\nimport { Tooltip } from '../Tooltip'\nimport { findOptionInOptions } from './findOptionInOptions'\nimport { useSelectInput } from './SelectInputProvider'\nimport type { OptionType } from './types'\nimport { INPUT_SIZE_HEIGHT } from './types'\n\nconst SIZES_TAG = {\n  gap: 8,\n  paddings: 16,\n  plusTag: 48,\n}\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  dropdownId?: string\n}\n\ntype DisplayValuesProps = {\n  refTag: RefObject<HTMLDivElement | null>\n  nonOverflowedValues: OptionType[]\n  potentiallyNonOverflowedValues: OptionType[]\n  disabled: boolean\n  readOnly: boolean\n  overflowed: boolean\n  overflowAmount: number\n  measureRef: RefObject<HTMLDivElement | null>\n  size: 'small' | 'medium' | 'large'\n  lastElementMaxWidth: number\n  overflow?: boolean\n  refPlusTag: RefObject<HTMLDivElement | null>\n  displayShadowCopy?: boolean\n}\n\nconst StateStack = styled(Stack)`\n  padding-right: ${({ theme }) => theme.space['2']};\n  display: flex;\n`\nconst Placeholder = styled(Text)`\nuser-select: none;\nalign-self: center;\n`\n\nconst StyledInputWrapper = styled.div<{\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: grid;\n  width: 100%;\n  gap: ${({ theme }) => theme.space[1]};\n  grid-template-columns: 1fr auto ;\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\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  shouldForwardProp: prop => !['lastElementMaxWidth', 'hidden'].includes(prop),\n})<{\n  lastElementMaxWidth?: number\n  hidden?: boolean\n}>`\n  height: max-content;\n  width: fit-content;\n  min-width: ${({ lastElementMaxWidth }) =>\n    lastElementMaxWidth ? 'auto' : 'fit-content'};\n\n  max-width: ${({ lastElementMaxWidth, hidden }) =>\n    lastElementMaxWidth && !hidden ? `${lastElementMaxWidth}px` : '100%'};\n\n  ${({ hidden }) =>\n    hidden\n      ? 'visibility: hidden;'\n      : `\n  text-overflow: ellipsis;\n  overflow: hidden;`}\n\n  & > ${StyledChildrenContainer} {\n    overflow: hidden;\n  }\n`\n\nconst SelectedValues = styled(Text)`\n  text-overflow: ellipsis;\n  overflow: hidden;\n  white-space: nowrap;\n  align-self: center;\n`\nconst PlusTag = styled(Tag)`\nwidth: ${({ theme }) => theme.sizing[500]};\n;\n`\n\nconst MultiselectStack = styled(Stack)`\noverflow: hidden;\nmax-width: 100%;\nheight: 100%;\n`\n\nconst DisplayValues = ({\n  refTag,\n  nonOverflowedValues,\n  potentiallyNonOverflowedValues,\n  disabled,\n  readOnly,\n  overflowed,\n  overflowAmount,\n  size,\n  measureRef,\n  lastElementMaxWidth,\n  overflow,\n  refPlusTag,\n  displayShadowCopy,\n}: DisplayValuesProps) => {\n  const { multiselect, selectedData, setSelectedData, options, onChange } =\n    useSelectInput()\n\n  return multiselect ? (\n    <MultiselectStack\n      alignItems=\"center\"\n      direction=\"row\"\n      gap=\"1\"\n      ref={refTag}\n      wrap=\"nowrap\"\n    >\n      {/* Hidden div to measure the width of the tags */}\n      {displayShadowCopy ? (\n        <div\n          ref={measureRef}\n          style={{\n            position: 'absolute',\n          }}\n        >\n          {potentiallyNonOverflowedValues.map(option => (\n            <CustomTag\n              className={option.value}\n              hidden\n              key={option.value}\n              onClose={() => {}}\n            >\n              {option?.label}\n            </CustomTag>\n          ))}\n        </div>\n      ) : null}\n      {nonOverflowedValues.map((option, index) => (\n        <CustomTag\n          data-testid=\"selected-options-tags\"\n          disabled={disabled}\n          key={option?.value}\n          lastElementMaxWidth={\n            index === nonOverflowedValues.length - 1 && overflow\n              ? lastElementMaxWidth\n              : 0\n          }\n          onClose={\n            !readOnly\n              ? event => {\n                  event.stopPropagation()\n                  setSelectedData({\n                    clickedOption: option,\n                    type: 'selectOption',\n                  })\n                  const newSelectedValues = selectedData.selectedValues?.filter(\n                    val => val !== option.value,\n                  )\n                  onChange?.(newSelectedValues)\n                }\n              : undefined\n          }\n          sentiment=\"neutral\"\n        >\n          {option?.label}\n        </CustomTag>\n      ))}\n\n      {overflowed ? (\n        <Stack justifyContent=\"center\" ref={refPlusTag}>\n          <PlusTag\n            aria-label=\"Plus tag\"\n            data-testid=\"plus-tag\"\n            disabled={disabled}\n            key=\"+\"\n            sentiment=\"neutral\"\n          >\n            <PlusIcon size=\"xsmall\" />\n            {overflowAmount}\n          </PlusTag>\n        </Stack>\n      ) : null}\n    </MultiselectStack>\n  ) : (\n    <SelectedValues\n      as=\"div\"\n      disabled={disabled}\n      prominence=\"default\"\n      sentiment=\"neutral\"\n      variant={size === 'large' ? 'body' : 'bodySmall'}\n    >\n      {selectedData.selectedValues[0]\n        ? findOptionInOptions(options, selectedData.selectedValues[0])?.label\n        : null}\n    </SelectedValues>\n  )\n}\n\nconst 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  dropdownId,\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 [displayShadowCopy, setDisplayShadowCopy] = useState(false)\n  const refTag = useRef<HTMLDivElement>(null)\n  const measureRef = useRef<HTMLDivElement>(null)\n  const arrowRef = useRef<HTMLDivElement>(null)\n  const refPlusTag = useRef<HTMLDivElement>(null)\n  // width - width of the arrow (in px) - padding between tags (in px)\n  const [innerWidth, setInnerWidth] = useState(\n    innerRef.current?.offsetWidth ??\n      0 - (arrowRef.current?.offsetWidth ?? 0) - SIZES_TAG.paddings,\n  )\n  const [overflowAmount, setOverflowAmount] = useState(0)\n  const [overflow, setOverflow] = useState(false)\n  const [lastElementMaxWidth, setLastElementMaxWidth] = 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 potentiallyNonOverflowedValues = useMemo(\n    () =>\n      selectedData.selectedValues\n        .map(selectedValue => findOptionInOptions(options, selectedValue))\n        .filter((option): option is OptionType => !!option),\n    [options, selectedData.selectedValues],\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  // First we display shadow copy of tags to render it\n  useEffect(() => {\n    setDisplayShadowCopy(true)\n  }, [selectedData.selectedValues.length])\n\n  // We then want to measure the tags length before displaying them\n  // so we can determine if there is an overflow or not\n  // We use useLayoutEffect to ensure the measurement is done before the browser paints\n  useLayoutEffect(() => {\n    if (selectedData.selectedValues.length === 0) {\n      setOverflowAmount(0)\n      setNonOverFlowedValues([])\n    }\n\n    if (measureRef.current && selectedData.selectedValues.length > 0) {\n      const toMeasureElements: HTMLCollection = measureRef.current.children\n      const toMeasureElementsArray = [...toMeasureElements]\n\n      const {\n        measuredVisibleTags,\n        measuredHiddenTags,\n        accumulatedWidth,\n        lastVisibleElementWidth,\n        lastVisibleLabel,\n      } = toMeasureElementsArray.reduce(\n        (\n          accumulator: {\n            measuredVisibleTags: OptionType[]\n            measuredHiddenTags: number\n            accumulatedWidth: number\n            lastVisibleElementWidth: number\n            lastVisibleLabel: ReactNode\n          },\n          currentValue,\n          index,\n        ) => {\n          const elementWidth = (currentValue as HTMLDivElement).offsetWidth\n\n          const newAccumulatedWidth =\n            accumulator.accumulatedWidth + elementWidth + SIZES_TAG.gap\n\n          const canBeVisible = newAccumulatedWidth <= innerWidth\n\n          return {\n            accumulatedWidth: !canBeVisible\n              ? accumulator.accumulatedWidth\n              : newAccumulatedWidth,\n            lastVisibleElementWidth: canBeVisible\n              ? elementWidth\n              : accumulator.lastVisibleElementWidth,\n            lastVisibleLabel: canBeVisible\n              ? potentiallyNonOverflowedValues[index].label\n              : accumulator.lastVisibleLabel,\n            measuredHiddenTags:\n              accumulator.measuredHiddenTags + (!canBeVisible ? 1 : 0),\n            measuredVisibleTags: [\n              ...accumulator.measuredVisibleTags,\n              canBeVisible && potentiallyNonOverflowedValues[index],\n            ].filter(Boolean) as OptionType[],\n          }\n        },\n        {\n          accumulatedWidth: 0,\n          lastVisibleElementWidth: 0,\n          lastVisibleLabel: '',\n          measuredHiddenTags: 0,\n          measuredVisibleTags: [],\n        },\n      )\n\n      const additionnalElementsWidth =\n        SIZES_TAG.paddings + (refPlusTag.current?.offsetWidth ?? 0)\n      const finalWidth =\n        accumulatedWidth + (measuredHiddenTags ? additionnalElementsWidth : 0)\n\n      const overflowPx = finalWidth - innerWidth\n      const hasOverflow = overflowPx > 0\n      const hasHiddenTags = measuredHiddenTags > 0\n      const lastVisibleElementMaxSize = lastVisibleElementWidth - overflowPx\n\n      // If only one element is selected and it is hidden, we need to show it\n      if (measuredHiddenTags === 1 && measuredVisibleTags.length === 0) {\n        setOverflowAmount(0)\n        setNonOverFlowedValues([potentiallyNonOverflowedValues[0]])\n\n        const newOverflowPx =\n          lastVisibleElementWidth +\n          (measuredHiddenTags > 1 ? additionnalElementsWidth : 0) -\n          innerWidth\n        setLastElementMaxWidth(lastVisibleElementWidth - newOverflowPx)\n        setOverflow(true)\n      }\n\n      // If it overflows with the last tag, we need to add an ellipsis to the last element if there is enough space (>60px)\n      // and if it is a string (do not cut ReactNode label)\n      // else we hide it completely and add it to the overflow amount\n      else if (\n        hasOverflow &&\n        hasHiddenTags &&\n        (lastVisibleElementMaxSize > 65 ||\n          (measuredVisibleTags.length === 1 &&\n            lastVisibleElementMaxSize > 65)) &&\n        typeof lastVisibleLabel === 'string'\n      ) {\n        setLastElementMaxWidth(lastVisibleElementMaxSize)\n        setOverflow(true)\n        setOverflowAmount(measuredHiddenTags)\n        setNonOverFlowedValues(measuredVisibleTags)\n      } else if (hasOverflow && hasHiddenTags) {\n        setLastElementMaxWidth(0)\n        setOverflow(false)\n        setOverflowAmount(measuredHiddenTags + 1)\n        setNonOverFlowedValues(measuredVisibleTags.slice(0, -1))\n      }\n      // Otherwise, we have enough space to show all tags\n      else {\n        setOverflow(false)\n        setOverflowAmount(measuredHiddenTags)\n        setNonOverFlowedValues(measuredVisibleTags)\n      }\n    }\n    setDisplayShadowCopy(false)\n  }, [\n    displayShadowCopy,\n    innerWidth,\n    potentiallyNonOverflowedValues,\n    selectedData.selectedValues.length,\n  ])\n\n  useEffect(() => {\n    setSelectedData({ type: 'update' })\n  }, [setSelectedData, options])\n\n  useEffect(() => {\n    const getWidth = () => {\n      if (refTag.current) {\n        setInnerWidth(refTag.current.offsetWidth)\n      } else {\n        setInnerWidth(\n          innerRef.current?.offsetWidth ??\n            0 - (arrowRef.current?.offsetWidth ?? 0) - SIZES_TAG.paddings,\n        )\n      }\n    }\n    getWidth()\n    window.addEventListener('resize', getWidth)\n\n    return () => window.removeEventListener('resize', getWidth)\n  }, [innerRef, refTag, selectedData.selectedValues])\n\n  const shouldDisplayValues = useMemo(() => {\n    if (multiselect) {\n      return (\n        potentiallyNonOverflowedValues.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    options,\n    potentiallyNonOverflowedValues.length,\n    selectedData.selectedValues,\n  ])\n\n  return (\n    <Tooltip text={tooltip}>\n      <StyledInputWrapper\n        aria-controls={dropdownId}\n        aria-expanded={isDropdownVisible}\n        aria-label={label}\n        autoFocus={autoFocus}\n        data-disabled={disabled}\n        data-dropdownvisible={isDropdownVisible}\n        data-readonly={readOnly}\n        data-size={size}\n        data-state={state}\n        data-testid={dataTestId}\n        id={id}\n        onClick={\n          openable ? () => setIsDropdownVisible(!isDropdownVisible) : undefined\n        }\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        role=\"combobox\"\n        tabIndex={0}\n      >\n        {shouldDisplayValues ? (\n          <DisplayValues\n            disabled={disabled}\n            displayShadowCopy={displayShadowCopy}\n            lastElementMaxWidth={lastElementMaxWidth}\n            measureRef={measureRef}\n            nonOverflowedValues={nonOverflowedValues}\n            overflow={overflow}\n            overflowAmount={overflowAmount}\n            overflowed={!!overflowAmount}\n            potentiallyNonOverflowedValues={potentiallyNonOverflowedValues}\n            readOnly={readOnly}\n            refPlusTag={refPlusTag}\n            refTag={refTag}\n            size={size}\n          />\n        ) : (\n          <Placeholder\n            as=\"p\"\n            disabled={disabled}\n            prominence=\"weak\"\n            sentiment=\"neutral\"\n            variant={size === 'large' ? 'body' : 'bodySmall'}\n          >\n            {placeholder}\n          </Placeholder>\n        )}\n        <StateStack alignItems=\"center\" direction=\"row\" gap={1} ref={arrowRef}>\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              data-testid=\"clear-all\"\n              disabled={disabled || !selectedData.selectedValues[0] || readOnly}\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              size=\"small\"\n              variant=\"ghost\"\n            >\n              <CloseIcon />\n            </Button>\n          ) : null}\n          <ArrowDownIcon\n            aria-label=\"show dropdown\"\n            disabled={disabled || readOnly}\n            sentiment=\"neutral\"\n            size=\"small\"\n          />\n        </StateStack>\n      </StyledInputWrapper>\n    </Tooltip>\n  )\n}\n\nexport { SelectBar, StyledInputWrapper }\n"]} */",
131
131
  toString: _EMOTION_STRINGIFIED_CSS_ERROR__
132
132
  });
133
133
  const PlusTag = /* @__PURE__ */ _styled(Tag, process.env.NODE_ENV === "production" ? {
@@ -137,7 +137,7 @@ const PlusTag = /* @__PURE__ */ _styled(Tag, process.env.NODE_ENV === "productio
137
137
  label: "PlusTag"
138
138
  })("width:", ({
139
139
  theme
140
- }) => theme.sizing[500], ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInput/SelectBar.tsx"],"names":[],"mappings":"AAqM2B","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInput/SelectBar.tsx","sourcesContent":["'use client'\n\nimport styled from '@emotion/styled'\nimport {\n  AlertCircleIcon,\n  ArrowDownIcon,\n  CheckCircleIcon,\n  CloseIcon,\n  PlusIcon,\n} from '@ultraviolet/icons'\nimport type { ReactNode, RefObject } from 'react'\nimport { useEffect, useMemo, useRef, useState } from 'react'\nimport { Button } from '../Button'\nimport { StyledChildrenContainer } from '../Popup'\nimport { Stack } from '../Stack'\nimport { Tag } from '../Tag'\nimport { Text } from '../Text'\nimport { Tooltip } from '../Tooltip'\nimport { findOptionInOptions } from './findOptionInOptions'\nimport { useSelectInput } from './SelectInputProvider'\nimport type { OptionType } from './types'\nimport { INPUT_SIZE_HEIGHT } from './types'\n\nconst SIZES_TAG = {\n  gap: 8,\n  paddings: 16,\n  plusTag: 48,\n}\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  dropdownId?: string\n}\n\ntype DisplayValuesProps = {\n  refTag: RefObject<HTMLDivElement | null>\n  nonOverflowedValues: OptionType[]\n  potentiallyNonOverflowedValues: OptionType[]\n  disabled: boolean\n  readOnly: boolean\n  overflowed: boolean\n  overflowAmount: number\n  measureRef: RefObject<HTMLDivElement | null>\n  size: 'small' | 'medium' | 'large'\n  lastElementMaxWidth: number\n  overflow?: boolean\n  refPlusTag: RefObject<HTMLDivElement | null>\n}\n\nconst StateStack = styled(Stack)`\n  padding-right: ${({ theme }) => theme.space['2']};\n  display: flex;\n`\nconst Placeholder = styled(Text)`\nuser-select: none;\nalign-self: center;\n`\n\nconst StyledInputWrapper = styled.div<{\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: grid;\n  width: 100%;\n  gap: ${({ theme }) => theme.space[1]};\n  grid-template-columns: 1fr auto ;\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\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  shouldForwardProp: prop => !['lastElementMaxWidth', 'hidden'].includes(prop),\n})<{\n  lastElementMaxWidth?: number\n  hidden?: boolean\n}>`\n  height: max-content;\n  width: fit-content;\n  min-width: ${({ lastElementMaxWidth }) =>\n    lastElementMaxWidth ? 'auto' : 'fit-content'};\n\n  max-width: ${({ lastElementMaxWidth, hidden }) =>\n    lastElementMaxWidth && !hidden ? `${lastElementMaxWidth}px` : '100%'};\n\n  ${({ hidden }) =>\n    hidden\n      ? 'visibility: hidden;'\n      : `\n  text-overflow: ellipsis;\n  overflow: hidden;`}\n\n  & > ${StyledChildrenContainer} {\n    overflow: hidden;\n  }\n`\n\nconst SelectedValues = styled(Text)`\n  text-overflow: ellipsis;\n  overflow: hidden;\n  white-space: nowrap;\n  align-self: center;\n`\nconst PlusTag = styled(Tag)`\nwidth: ${({ theme }) => theme.sizing[500]};\n;\n`\n\nconst MultiselectStack = styled(Stack)`\noverflow: hidden;\nmax-width: 100%;\nheight: 100%;\n`\n\nconst DisplayValues = ({\n  refTag,\n  nonOverflowedValues,\n  potentiallyNonOverflowedValues,\n  disabled,\n  readOnly,\n  overflowed,\n  overflowAmount,\n  size,\n  measureRef,\n  lastElementMaxWidth,\n  overflow,\n  refPlusTag,\n}: DisplayValuesProps) => {\n  const { multiselect, selectedData, setSelectedData, options, onChange } =\n    useSelectInput()\n\n  return multiselect ? (\n    <MultiselectStack\n      alignItems=\"center\"\n      direction=\"row\"\n      gap=\"1\"\n      ref={refTag}\n      wrap=\"nowrap\"\n    >\n      {/* Hidden div to measure the width of the tags */}\n      <div\n        ref={measureRef}\n        style={{\n          position: 'absolute',\n        }}\n      >\n        {potentiallyNonOverflowedValues.map(option => (\n          <CustomTag\n            className={option.value}\n            hidden\n            key={option.value}\n            onClose={() => {}}\n          >\n            {option?.label}\n          </CustomTag>\n        ))}\n      </div>\n      {nonOverflowedValues.map((option, index) => (\n        <CustomTag\n          data-testid=\"selected-options-tags\"\n          disabled={disabled}\n          key={option?.value}\n          lastElementMaxWidth={\n            index === nonOverflowedValues.length - 1 && overflow\n              ? lastElementMaxWidth\n              : 0\n          }\n          onClose={\n            !readOnly\n              ? event => {\n                  event.stopPropagation()\n                  setSelectedData({\n                    clickedOption: option,\n                    type: 'selectOption',\n                  })\n                  const newSelectedValues = selectedData.selectedValues?.filter(\n                    val => val !== option.value,\n                  )\n                  onChange?.(newSelectedValues)\n                }\n              : undefined\n          }\n          sentiment=\"neutral\"\n        >\n          {option?.label}\n        </CustomTag>\n      ))}\n\n      {overflowed ? (\n        <Stack justifyContent=\"center\" ref={refPlusTag}>\n          <PlusTag\n            aria-label=\"Plus tag\"\n            data-testid=\"plus-tag\"\n            disabled={disabled}\n            key=\"+\"\n            sentiment=\"neutral\"\n          >\n            <PlusIcon size=\"xsmall\" />\n            {overflowAmount}\n          </PlusTag>\n        </Stack>\n      ) : null}\n    </MultiselectStack>\n  ) : (\n    <SelectedValues\n      as=\"div\"\n      disabled={disabled}\n      prominence=\"default\"\n      sentiment=\"neutral\"\n      variant={size === 'large' ? 'body' : 'bodySmall'}\n    >\n      {selectedData.selectedValues[0]\n        ? findOptionInOptions(options, selectedData.selectedValues[0])?.label\n        : null}\n    </SelectedValues>\n  )\n}\n\nconst 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  dropdownId,\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 measureRef = useRef<HTMLDivElement>(null)\n  const arrowRef = useRef<HTMLDivElement>(null)\n  const refPlusTag = useRef<HTMLDivElement>(null)\n  // width - width of the arrow (in px) - padding between tags (in px)\n  const [innerWidth, setInnerWidth] = useState(\n    innerRef.current?.offsetWidth ??\n      0 - (arrowRef.current?.offsetWidth ?? 0) - SIZES_TAG.paddings,\n  )\n  const [overflowAmount, setOverflowAmount] = useState(0)\n  const [overflow, setOverflow] = useState(false)\n  const [lastElementMaxWidth, setLastElementMaxWidth] = 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 potentiallyNonOverflowedValues = useMemo(\n    () =>\n      selectedData.selectedValues\n        .map(selectedValue => findOptionInOptions(options, selectedValue))\n        .filter((option): option is OptionType => !!option),\n    [options, selectedData.selectedValues],\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    if (selectedData.selectedValues.length === 0) {\n      setOverflowAmount(0)\n      setNonOverFlowedValues([])\n    }\n    if (measureRef.current && selectedData.selectedValues.length > 0) {\n      const toMeasureElements: HTMLCollection = measureRef.current.children\n      const toMeasureElementsArray = [...toMeasureElements]\n\n      const {\n        measuredVisibleTags,\n        measuredHiddenTags,\n        accumulatedWidth,\n        lastVisibleElementWidth,\n        lastVisibleLabel,\n      } = toMeasureElementsArray.reduce(\n        (\n          accumulator: {\n            measuredVisibleTags: OptionType[]\n            measuredHiddenTags: number\n            accumulatedWidth: number\n            lastVisibleElementWidth: number\n            lastVisibleLabel: ReactNode\n          },\n          currentValue,\n          index,\n        ) => {\n          const elementWidth = (currentValue as HTMLDivElement).offsetWidth\n\n          const newAccumulatedWidth =\n            accumulator.accumulatedWidth + elementWidth + SIZES_TAG.gap\n\n          const canBeVisible = newAccumulatedWidth <= innerWidth\n\n          return {\n            accumulatedWidth: !canBeVisible\n              ? accumulator.accumulatedWidth\n              : newAccumulatedWidth,\n            lastVisibleElementWidth: canBeVisible\n              ? elementWidth\n              : accumulator.lastVisibleElementWidth,\n            lastVisibleLabel: canBeVisible\n              ? potentiallyNonOverflowedValues[index].label\n              : accumulator.lastVisibleLabel,\n            measuredHiddenTags:\n              accumulator.measuredHiddenTags + (!canBeVisible ? 1 : 0),\n            measuredVisibleTags: [\n              ...accumulator.measuredVisibleTags,\n              canBeVisible && potentiallyNonOverflowedValues[index],\n            ].filter(Boolean) as OptionType[],\n          }\n        },\n        {\n          accumulatedWidth: 0,\n          lastVisibleElementWidth: 0,\n          lastVisibleLabel: '',\n          measuredHiddenTags: 0,\n          measuredVisibleTags: [],\n        },\n      )\n\n      const additionnalElementsWidth =\n        SIZES_TAG.paddings + (refPlusTag.current?.offsetWidth ?? 0)\n      const finalWidth =\n        accumulatedWidth + (measuredHiddenTags ? additionnalElementsWidth : 0)\n\n      const overflowPx = finalWidth - innerWidth\n      const hasOverflow = overflowPx > 0\n      const hasHiddenTags = measuredHiddenTags > 0\n      const lastVisibleElementMaxSize = lastVisibleElementWidth - overflowPx\n\n      // If only one element is selected and it is hidden, we need to show it\n      if (measuredHiddenTags === 1 && measuredVisibleTags.length === 0) {\n        setOverflowAmount(0)\n        setNonOverFlowedValues([potentiallyNonOverflowedValues[0]])\n\n        const newOverflowPx =\n          lastVisibleElementWidth +\n          (measuredHiddenTags > 1 ? additionnalElementsWidth : 0) -\n          innerWidth\n        setLastElementMaxWidth(lastVisibleElementWidth - newOverflowPx)\n        setOverflow(true)\n      }\n\n      // If it overflows with the last tag, we need to add an ellipsis to the last element if there is enough space (>60px)\n      // and if it is a string (do not cut ReactNode label)\n      // else we hide it completely and add it to the overflow amount\n      else if (\n        hasOverflow &&\n        hasHiddenTags &&\n        (lastVisibleElementMaxSize > 65 ||\n          (measuredVisibleTags.length === 1 &&\n            lastVisibleElementMaxSize > 65)) &&\n        typeof lastVisibleLabel === 'string'\n      ) {\n        setLastElementMaxWidth(lastVisibleElementMaxSize)\n        setOverflow(true)\n        setOverflowAmount(measuredHiddenTags)\n        setNonOverFlowedValues(measuredVisibleTags)\n      } else if (hasOverflow && hasHiddenTags) {\n        setLastElementMaxWidth(0)\n        setOverflow(false)\n        setOverflowAmount(measuredHiddenTags + 1)\n        setNonOverFlowedValues(measuredVisibleTags.slice(0, -1))\n      }\n      // Otherwise, we have enough space to show all tags\n      else {\n        setOverflow(false)\n        setOverflowAmount(measuredHiddenTags)\n        setNonOverFlowedValues(measuredVisibleTags)\n      }\n    }\n  }, [\n    selectedData.selectedValues.length,\n    innerWidth,\n    potentiallyNonOverflowedValues,\n  ])\n\n  useEffect(() => {\n    setSelectedData({ type: 'update' })\n  }, [setSelectedData, options])\n\n  useEffect(() => {\n    const getWidth = () => {\n      if (refTag.current) {\n        setInnerWidth(refTag.current.offsetWidth)\n      } else {\n        setInnerWidth(\n          innerRef.current?.offsetWidth ??\n            0 - (arrowRef.current?.offsetWidth ?? 0) - SIZES_TAG.paddings,\n        )\n      }\n    }\n    getWidth()\n    window.addEventListener('resize', getWidth)\n\n    return () => window.removeEventListener('resize', getWidth)\n  }, [innerRef, refTag, selectedData.selectedValues])\n\n  const shouldDisplayValues = useMemo(() => {\n    if (multiselect) {\n      return (\n        potentiallyNonOverflowedValues.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    options,\n    potentiallyNonOverflowedValues.length,\n    selectedData.selectedValues,\n  ])\n\n  return (\n    <Tooltip text={tooltip}>\n      <StyledInputWrapper\n        aria-controls={dropdownId}\n        aria-expanded={isDropdownVisible}\n        aria-label={label}\n        autoFocus={autoFocus}\n        data-disabled={disabled}\n        data-dropdownvisible={isDropdownVisible}\n        data-readonly={readOnly}\n        data-size={size}\n        data-state={state}\n        data-testid={dataTestId}\n        id={id}\n        onClick={\n          openable ? () => setIsDropdownVisible(!isDropdownVisible) : undefined\n        }\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        role=\"combobox\"\n        tabIndex={0}\n      >\n        {shouldDisplayValues ? (\n          <DisplayValues\n            disabled={disabled}\n            lastElementMaxWidth={lastElementMaxWidth}\n            measureRef={measureRef}\n            nonOverflowedValues={nonOverflowedValues}\n            overflow={overflow}\n            overflowAmount={overflowAmount}\n            overflowed={!!overflowAmount}\n            potentiallyNonOverflowedValues={potentiallyNonOverflowedValues}\n            readOnly={readOnly}\n            refPlusTag={refPlusTag}\n            refTag={refTag}\n            size={size}\n          />\n        ) : (\n          <Placeholder\n            as=\"p\"\n            disabled={disabled}\n            prominence=\"weak\"\n            sentiment=\"neutral\"\n            variant={size === 'large' ? 'body' : 'bodySmall'}\n          >\n            {placeholder}\n          </Placeholder>\n        )}\n        <StateStack alignItems=\"center\" direction=\"row\" gap={1} ref={arrowRef}>\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              data-testid=\"clear-all\"\n              disabled={disabled || !selectedData.selectedValues[0] || readOnly}\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              size=\"small\"\n              variant=\"ghost\"\n            >\n              <CloseIcon />\n            </Button>\n          ) : null}\n          <ArrowDownIcon\n            aria-label=\"show dropdown\"\n            disabled={disabled || readOnly}\n            sentiment=\"neutral\"\n            size=\"small\"\n          />\n        </StateStack>\n      </StyledInputWrapper>\n    </Tooltip>\n  )\n}\n\nexport { SelectBar, StyledInputWrapper }\n"]} */"));
140
+ }) => theme.sizing[500], ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInput/SelectBar.tsx"],"names":[],"mappings":"AAsM2B","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInput/SelectBar.tsx","sourcesContent":["'use client'\n\nimport styled from '@emotion/styled'\nimport {\n  AlertCircleIcon,\n  ArrowDownIcon,\n  CheckCircleIcon,\n  CloseIcon,\n  PlusIcon,\n} from '@ultraviolet/icons'\nimport type { ReactNode, RefObject } from 'react'\nimport { useEffect, useLayoutEffect, useMemo, useRef, useState } from 'react'\nimport { Button } from '../Button'\nimport { StyledChildrenContainer } from '../Popup'\nimport { Stack } from '../Stack'\nimport { Tag } from '../Tag'\nimport { Text } from '../Text'\nimport { Tooltip } from '../Tooltip'\nimport { findOptionInOptions } from './findOptionInOptions'\nimport { useSelectInput } from './SelectInputProvider'\nimport type { OptionType } from './types'\nimport { INPUT_SIZE_HEIGHT } from './types'\n\nconst SIZES_TAG = {\n  gap: 8,\n  paddings: 16,\n  plusTag: 48,\n}\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  dropdownId?: string\n}\n\ntype DisplayValuesProps = {\n  refTag: RefObject<HTMLDivElement | null>\n  nonOverflowedValues: OptionType[]\n  potentiallyNonOverflowedValues: OptionType[]\n  disabled: boolean\n  readOnly: boolean\n  overflowed: boolean\n  overflowAmount: number\n  measureRef: RefObject<HTMLDivElement | null>\n  size: 'small' | 'medium' | 'large'\n  lastElementMaxWidth: number\n  overflow?: boolean\n  refPlusTag: RefObject<HTMLDivElement | null>\n  displayShadowCopy?: boolean\n}\n\nconst StateStack = styled(Stack)`\n  padding-right: ${({ theme }) => theme.space['2']};\n  display: flex;\n`\nconst Placeholder = styled(Text)`\nuser-select: none;\nalign-self: center;\n`\n\nconst StyledInputWrapper = styled.div<{\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: grid;\n  width: 100%;\n  gap: ${({ theme }) => theme.space[1]};\n  grid-template-columns: 1fr auto ;\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\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  shouldForwardProp: prop => !['lastElementMaxWidth', 'hidden'].includes(prop),\n})<{\n  lastElementMaxWidth?: number\n  hidden?: boolean\n}>`\n  height: max-content;\n  width: fit-content;\n  min-width: ${({ lastElementMaxWidth }) =>\n    lastElementMaxWidth ? 'auto' : 'fit-content'};\n\n  max-width: ${({ lastElementMaxWidth, hidden }) =>\n    lastElementMaxWidth && !hidden ? `${lastElementMaxWidth}px` : '100%'};\n\n  ${({ hidden }) =>\n    hidden\n      ? 'visibility: hidden;'\n      : `\n  text-overflow: ellipsis;\n  overflow: hidden;`}\n\n  & > ${StyledChildrenContainer} {\n    overflow: hidden;\n  }\n`\n\nconst SelectedValues = styled(Text)`\n  text-overflow: ellipsis;\n  overflow: hidden;\n  white-space: nowrap;\n  align-self: center;\n`\nconst PlusTag = styled(Tag)`\nwidth: ${({ theme }) => theme.sizing[500]};\n;\n`\n\nconst MultiselectStack = styled(Stack)`\noverflow: hidden;\nmax-width: 100%;\nheight: 100%;\n`\n\nconst DisplayValues = ({\n  refTag,\n  nonOverflowedValues,\n  potentiallyNonOverflowedValues,\n  disabled,\n  readOnly,\n  overflowed,\n  overflowAmount,\n  size,\n  measureRef,\n  lastElementMaxWidth,\n  overflow,\n  refPlusTag,\n  displayShadowCopy,\n}: DisplayValuesProps) => {\n  const { multiselect, selectedData, setSelectedData, options, onChange } =\n    useSelectInput()\n\n  return multiselect ? (\n    <MultiselectStack\n      alignItems=\"center\"\n      direction=\"row\"\n      gap=\"1\"\n      ref={refTag}\n      wrap=\"nowrap\"\n    >\n      {/* Hidden div to measure the width of the tags */}\n      {displayShadowCopy ? (\n        <div\n          ref={measureRef}\n          style={{\n            position: 'absolute',\n          }}\n        >\n          {potentiallyNonOverflowedValues.map(option => (\n            <CustomTag\n              className={option.value}\n              hidden\n              key={option.value}\n              onClose={() => {}}\n            >\n              {option?.label}\n            </CustomTag>\n          ))}\n        </div>\n      ) : null}\n      {nonOverflowedValues.map((option, index) => (\n        <CustomTag\n          data-testid=\"selected-options-tags\"\n          disabled={disabled}\n          key={option?.value}\n          lastElementMaxWidth={\n            index === nonOverflowedValues.length - 1 && overflow\n              ? lastElementMaxWidth\n              : 0\n          }\n          onClose={\n            !readOnly\n              ? event => {\n                  event.stopPropagation()\n                  setSelectedData({\n                    clickedOption: option,\n                    type: 'selectOption',\n                  })\n                  const newSelectedValues = selectedData.selectedValues?.filter(\n                    val => val !== option.value,\n                  )\n                  onChange?.(newSelectedValues)\n                }\n              : undefined\n          }\n          sentiment=\"neutral\"\n        >\n          {option?.label}\n        </CustomTag>\n      ))}\n\n      {overflowed ? (\n        <Stack justifyContent=\"center\" ref={refPlusTag}>\n          <PlusTag\n            aria-label=\"Plus tag\"\n            data-testid=\"plus-tag\"\n            disabled={disabled}\n            key=\"+\"\n            sentiment=\"neutral\"\n          >\n            <PlusIcon size=\"xsmall\" />\n            {overflowAmount}\n          </PlusTag>\n        </Stack>\n      ) : null}\n    </MultiselectStack>\n  ) : (\n    <SelectedValues\n      as=\"div\"\n      disabled={disabled}\n      prominence=\"default\"\n      sentiment=\"neutral\"\n      variant={size === 'large' ? 'body' : 'bodySmall'}\n    >\n      {selectedData.selectedValues[0]\n        ? findOptionInOptions(options, selectedData.selectedValues[0])?.label\n        : null}\n    </SelectedValues>\n  )\n}\n\nconst 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  dropdownId,\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 [displayShadowCopy, setDisplayShadowCopy] = useState(false)\n  const refTag = useRef<HTMLDivElement>(null)\n  const measureRef = useRef<HTMLDivElement>(null)\n  const arrowRef = useRef<HTMLDivElement>(null)\n  const refPlusTag = useRef<HTMLDivElement>(null)\n  // width - width of the arrow (in px) - padding between tags (in px)\n  const [innerWidth, setInnerWidth] = useState(\n    innerRef.current?.offsetWidth ??\n      0 - (arrowRef.current?.offsetWidth ?? 0) - SIZES_TAG.paddings,\n  )\n  const [overflowAmount, setOverflowAmount] = useState(0)\n  const [overflow, setOverflow] = useState(false)\n  const [lastElementMaxWidth, setLastElementMaxWidth] = 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 potentiallyNonOverflowedValues = useMemo(\n    () =>\n      selectedData.selectedValues\n        .map(selectedValue => findOptionInOptions(options, selectedValue))\n        .filter((option): option is OptionType => !!option),\n    [options, selectedData.selectedValues],\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  // First we display shadow copy of tags to render it\n  useEffect(() => {\n    setDisplayShadowCopy(true)\n  }, [selectedData.selectedValues.length])\n\n  // We then want to measure the tags length before displaying them\n  // so we can determine if there is an overflow or not\n  // We use useLayoutEffect to ensure the measurement is done before the browser paints\n  useLayoutEffect(() => {\n    if (selectedData.selectedValues.length === 0) {\n      setOverflowAmount(0)\n      setNonOverFlowedValues([])\n    }\n\n    if (measureRef.current && selectedData.selectedValues.length > 0) {\n      const toMeasureElements: HTMLCollection = measureRef.current.children\n      const toMeasureElementsArray = [...toMeasureElements]\n\n      const {\n        measuredVisibleTags,\n        measuredHiddenTags,\n        accumulatedWidth,\n        lastVisibleElementWidth,\n        lastVisibleLabel,\n      } = toMeasureElementsArray.reduce(\n        (\n          accumulator: {\n            measuredVisibleTags: OptionType[]\n            measuredHiddenTags: number\n            accumulatedWidth: number\n            lastVisibleElementWidth: number\n            lastVisibleLabel: ReactNode\n          },\n          currentValue,\n          index,\n        ) => {\n          const elementWidth = (currentValue as HTMLDivElement).offsetWidth\n\n          const newAccumulatedWidth =\n            accumulator.accumulatedWidth + elementWidth + SIZES_TAG.gap\n\n          const canBeVisible = newAccumulatedWidth <= innerWidth\n\n          return {\n            accumulatedWidth: !canBeVisible\n              ? accumulator.accumulatedWidth\n              : newAccumulatedWidth,\n            lastVisibleElementWidth: canBeVisible\n              ? elementWidth\n              : accumulator.lastVisibleElementWidth,\n            lastVisibleLabel: canBeVisible\n              ? potentiallyNonOverflowedValues[index].label\n              : accumulator.lastVisibleLabel,\n            measuredHiddenTags:\n              accumulator.measuredHiddenTags + (!canBeVisible ? 1 : 0),\n            measuredVisibleTags: [\n              ...accumulator.measuredVisibleTags,\n              canBeVisible && potentiallyNonOverflowedValues[index],\n            ].filter(Boolean) as OptionType[],\n          }\n        },\n        {\n          accumulatedWidth: 0,\n          lastVisibleElementWidth: 0,\n          lastVisibleLabel: '',\n          measuredHiddenTags: 0,\n          measuredVisibleTags: [],\n        },\n      )\n\n      const additionnalElementsWidth =\n        SIZES_TAG.paddings + (refPlusTag.current?.offsetWidth ?? 0)\n      const finalWidth =\n        accumulatedWidth + (measuredHiddenTags ? additionnalElementsWidth : 0)\n\n      const overflowPx = finalWidth - innerWidth\n      const hasOverflow = overflowPx > 0\n      const hasHiddenTags = measuredHiddenTags > 0\n      const lastVisibleElementMaxSize = lastVisibleElementWidth - overflowPx\n\n      // If only one element is selected and it is hidden, we need to show it\n      if (measuredHiddenTags === 1 && measuredVisibleTags.length === 0) {\n        setOverflowAmount(0)\n        setNonOverFlowedValues([potentiallyNonOverflowedValues[0]])\n\n        const newOverflowPx =\n          lastVisibleElementWidth +\n          (measuredHiddenTags > 1 ? additionnalElementsWidth : 0) -\n          innerWidth\n        setLastElementMaxWidth(lastVisibleElementWidth - newOverflowPx)\n        setOverflow(true)\n      }\n\n      // If it overflows with the last tag, we need to add an ellipsis to the last element if there is enough space (>60px)\n      // and if it is a string (do not cut ReactNode label)\n      // else we hide it completely and add it to the overflow amount\n      else if (\n        hasOverflow &&\n        hasHiddenTags &&\n        (lastVisibleElementMaxSize > 65 ||\n          (measuredVisibleTags.length === 1 &&\n            lastVisibleElementMaxSize > 65)) &&\n        typeof lastVisibleLabel === 'string'\n      ) {\n        setLastElementMaxWidth(lastVisibleElementMaxSize)\n        setOverflow(true)\n        setOverflowAmount(measuredHiddenTags)\n        setNonOverFlowedValues(measuredVisibleTags)\n      } else if (hasOverflow && hasHiddenTags) {\n        setLastElementMaxWidth(0)\n        setOverflow(false)\n        setOverflowAmount(measuredHiddenTags + 1)\n        setNonOverFlowedValues(measuredVisibleTags.slice(0, -1))\n      }\n      // Otherwise, we have enough space to show all tags\n      else {\n        setOverflow(false)\n        setOverflowAmount(measuredHiddenTags)\n        setNonOverFlowedValues(measuredVisibleTags)\n      }\n    }\n    setDisplayShadowCopy(false)\n  }, [\n    displayShadowCopy,\n    innerWidth,\n    potentiallyNonOverflowedValues,\n    selectedData.selectedValues.length,\n  ])\n\n  useEffect(() => {\n    setSelectedData({ type: 'update' })\n  }, [setSelectedData, options])\n\n  useEffect(() => {\n    const getWidth = () => {\n      if (refTag.current) {\n        setInnerWidth(refTag.current.offsetWidth)\n      } else {\n        setInnerWidth(\n          innerRef.current?.offsetWidth ??\n            0 - (arrowRef.current?.offsetWidth ?? 0) - SIZES_TAG.paddings,\n        )\n      }\n    }\n    getWidth()\n    window.addEventListener('resize', getWidth)\n\n    return () => window.removeEventListener('resize', getWidth)\n  }, [innerRef, refTag, selectedData.selectedValues])\n\n  const shouldDisplayValues = useMemo(() => {\n    if (multiselect) {\n      return (\n        potentiallyNonOverflowedValues.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    options,\n    potentiallyNonOverflowedValues.length,\n    selectedData.selectedValues,\n  ])\n\n  return (\n    <Tooltip text={tooltip}>\n      <StyledInputWrapper\n        aria-controls={dropdownId}\n        aria-expanded={isDropdownVisible}\n        aria-label={label}\n        autoFocus={autoFocus}\n        data-disabled={disabled}\n        data-dropdownvisible={isDropdownVisible}\n        data-readonly={readOnly}\n        data-size={size}\n        data-state={state}\n        data-testid={dataTestId}\n        id={id}\n        onClick={\n          openable ? () => setIsDropdownVisible(!isDropdownVisible) : undefined\n        }\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        role=\"combobox\"\n        tabIndex={0}\n      >\n        {shouldDisplayValues ? (\n          <DisplayValues\n            disabled={disabled}\n            displayShadowCopy={displayShadowCopy}\n            lastElementMaxWidth={lastElementMaxWidth}\n            measureRef={measureRef}\n            nonOverflowedValues={nonOverflowedValues}\n            overflow={overflow}\n            overflowAmount={overflowAmount}\n            overflowed={!!overflowAmount}\n            potentiallyNonOverflowedValues={potentiallyNonOverflowedValues}\n            readOnly={readOnly}\n            refPlusTag={refPlusTag}\n            refTag={refTag}\n            size={size}\n          />\n        ) : (\n          <Placeholder\n            as=\"p\"\n            disabled={disabled}\n            prominence=\"weak\"\n            sentiment=\"neutral\"\n            variant={size === 'large' ? 'body' : 'bodySmall'}\n          >\n            {placeholder}\n          </Placeholder>\n        )}\n        <StateStack alignItems=\"center\" direction=\"row\" gap={1} ref={arrowRef}>\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              data-testid=\"clear-all\"\n              disabled={disabled || !selectedData.selectedValues[0] || readOnly}\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              size=\"small\"\n              variant=\"ghost\"\n            >\n              <CloseIcon />\n            </Button>\n          ) : null}\n          <ArrowDownIcon\n            aria-label=\"show dropdown\"\n            disabled={disabled || readOnly}\n            sentiment=\"neutral\"\n            size=\"small\"\n          />\n        </StateStack>\n      </StyledInputWrapper>\n    </Tooltip>\n  )\n}\n\nexport { SelectBar, StyledInputWrapper }\n"]} */"));
141
141
  const MultiselectStack = /* @__PURE__ */ _styled(Stack, process.env.NODE_ENV === "production" ? {
142
142
  target: "e1pocxam0"
143
143
  } : {
@@ -148,7 +148,7 @@ const MultiselectStack = /* @__PURE__ */ _styled(Stack, process.env.NODE_ENV ===
148
148
  styles: "overflow:hidden;max-width:100%;height:100%"
149
149
  } : {
150
150
  name: "1llx5fl",
151
- styles: "overflow:hidden;max-width:100%;height:100%/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInput/SelectBar.tsx"],"names":[],"mappings":"AA0MsC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInput/SelectBar.tsx","sourcesContent":["'use client'\n\nimport styled from '@emotion/styled'\nimport {\n  AlertCircleIcon,\n  ArrowDownIcon,\n  CheckCircleIcon,\n  CloseIcon,\n  PlusIcon,\n} from '@ultraviolet/icons'\nimport type { ReactNode, RefObject } from 'react'\nimport { useEffect, useMemo, useRef, useState } from 'react'\nimport { Button } from '../Button'\nimport { StyledChildrenContainer } from '../Popup'\nimport { Stack } from '../Stack'\nimport { Tag } from '../Tag'\nimport { Text } from '../Text'\nimport { Tooltip } from '../Tooltip'\nimport { findOptionInOptions } from './findOptionInOptions'\nimport { useSelectInput } from './SelectInputProvider'\nimport type { OptionType } from './types'\nimport { INPUT_SIZE_HEIGHT } from './types'\n\nconst SIZES_TAG = {\n  gap: 8,\n  paddings: 16,\n  plusTag: 48,\n}\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  dropdownId?: string\n}\n\ntype DisplayValuesProps = {\n  refTag: RefObject<HTMLDivElement | null>\n  nonOverflowedValues: OptionType[]\n  potentiallyNonOverflowedValues: OptionType[]\n  disabled: boolean\n  readOnly: boolean\n  overflowed: boolean\n  overflowAmount: number\n  measureRef: RefObject<HTMLDivElement | null>\n  size: 'small' | 'medium' | 'large'\n  lastElementMaxWidth: number\n  overflow?: boolean\n  refPlusTag: RefObject<HTMLDivElement | null>\n}\n\nconst StateStack = styled(Stack)`\n  padding-right: ${({ theme }) => theme.space['2']};\n  display: flex;\n`\nconst Placeholder = styled(Text)`\nuser-select: none;\nalign-self: center;\n`\n\nconst StyledInputWrapper = styled.div<{\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: grid;\n  width: 100%;\n  gap: ${({ theme }) => theme.space[1]};\n  grid-template-columns: 1fr auto ;\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\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  shouldForwardProp: prop => !['lastElementMaxWidth', 'hidden'].includes(prop),\n})<{\n  lastElementMaxWidth?: number\n  hidden?: boolean\n}>`\n  height: max-content;\n  width: fit-content;\n  min-width: ${({ lastElementMaxWidth }) =>\n    lastElementMaxWidth ? 'auto' : 'fit-content'};\n\n  max-width: ${({ lastElementMaxWidth, hidden }) =>\n    lastElementMaxWidth && !hidden ? `${lastElementMaxWidth}px` : '100%'};\n\n  ${({ hidden }) =>\n    hidden\n      ? 'visibility: hidden;'\n      : `\n  text-overflow: ellipsis;\n  overflow: hidden;`}\n\n  & > ${StyledChildrenContainer} {\n    overflow: hidden;\n  }\n`\n\nconst SelectedValues = styled(Text)`\n  text-overflow: ellipsis;\n  overflow: hidden;\n  white-space: nowrap;\n  align-self: center;\n`\nconst PlusTag = styled(Tag)`\nwidth: ${({ theme }) => theme.sizing[500]};\n;\n`\n\nconst MultiselectStack = styled(Stack)`\noverflow: hidden;\nmax-width: 100%;\nheight: 100%;\n`\n\nconst DisplayValues = ({\n  refTag,\n  nonOverflowedValues,\n  potentiallyNonOverflowedValues,\n  disabled,\n  readOnly,\n  overflowed,\n  overflowAmount,\n  size,\n  measureRef,\n  lastElementMaxWidth,\n  overflow,\n  refPlusTag,\n}: DisplayValuesProps) => {\n  const { multiselect, selectedData, setSelectedData, options, onChange } =\n    useSelectInput()\n\n  return multiselect ? (\n    <MultiselectStack\n      alignItems=\"center\"\n      direction=\"row\"\n      gap=\"1\"\n      ref={refTag}\n      wrap=\"nowrap\"\n    >\n      {/* Hidden div to measure the width of the tags */}\n      <div\n        ref={measureRef}\n        style={{\n          position: 'absolute',\n        }}\n      >\n        {potentiallyNonOverflowedValues.map(option => (\n          <CustomTag\n            className={option.value}\n            hidden\n            key={option.value}\n            onClose={() => {}}\n          >\n            {option?.label}\n          </CustomTag>\n        ))}\n      </div>\n      {nonOverflowedValues.map((option, index) => (\n        <CustomTag\n          data-testid=\"selected-options-tags\"\n          disabled={disabled}\n          key={option?.value}\n          lastElementMaxWidth={\n            index === nonOverflowedValues.length - 1 && overflow\n              ? lastElementMaxWidth\n              : 0\n          }\n          onClose={\n            !readOnly\n              ? event => {\n                  event.stopPropagation()\n                  setSelectedData({\n                    clickedOption: option,\n                    type: 'selectOption',\n                  })\n                  const newSelectedValues = selectedData.selectedValues?.filter(\n                    val => val !== option.value,\n                  )\n                  onChange?.(newSelectedValues)\n                }\n              : undefined\n          }\n          sentiment=\"neutral\"\n        >\n          {option?.label}\n        </CustomTag>\n      ))}\n\n      {overflowed ? (\n        <Stack justifyContent=\"center\" ref={refPlusTag}>\n          <PlusTag\n            aria-label=\"Plus tag\"\n            data-testid=\"plus-tag\"\n            disabled={disabled}\n            key=\"+\"\n            sentiment=\"neutral\"\n          >\n            <PlusIcon size=\"xsmall\" />\n            {overflowAmount}\n          </PlusTag>\n        </Stack>\n      ) : null}\n    </MultiselectStack>\n  ) : (\n    <SelectedValues\n      as=\"div\"\n      disabled={disabled}\n      prominence=\"default\"\n      sentiment=\"neutral\"\n      variant={size === 'large' ? 'body' : 'bodySmall'}\n    >\n      {selectedData.selectedValues[0]\n        ? findOptionInOptions(options, selectedData.selectedValues[0])?.label\n        : null}\n    </SelectedValues>\n  )\n}\n\nconst 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  dropdownId,\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 measureRef = useRef<HTMLDivElement>(null)\n  const arrowRef = useRef<HTMLDivElement>(null)\n  const refPlusTag = useRef<HTMLDivElement>(null)\n  // width - width of the arrow (in px) - padding between tags (in px)\n  const [innerWidth, setInnerWidth] = useState(\n    innerRef.current?.offsetWidth ??\n      0 - (arrowRef.current?.offsetWidth ?? 0) - SIZES_TAG.paddings,\n  )\n  const [overflowAmount, setOverflowAmount] = useState(0)\n  const [overflow, setOverflow] = useState(false)\n  const [lastElementMaxWidth, setLastElementMaxWidth] = 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 potentiallyNonOverflowedValues = useMemo(\n    () =>\n      selectedData.selectedValues\n        .map(selectedValue => findOptionInOptions(options, selectedValue))\n        .filter((option): option is OptionType => !!option),\n    [options, selectedData.selectedValues],\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    if (selectedData.selectedValues.length === 0) {\n      setOverflowAmount(0)\n      setNonOverFlowedValues([])\n    }\n    if (measureRef.current && selectedData.selectedValues.length > 0) {\n      const toMeasureElements: HTMLCollection = measureRef.current.children\n      const toMeasureElementsArray = [...toMeasureElements]\n\n      const {\n        measuredVisibleTags,\n        measuredHiddenTags,\n        accumulatedWidth,\n        lastVisibleElementWidth,\n        lastVisibleLabel,\n      } = toMeasureElementsArray.reduce(\n        (\n          accumulator: {\n            measuredVisibleTags: OptionType[]\n            measuredHiddenTags: number\n            accumulatedWidth: number\n            lastVisibleElementWidth: number\n            lastVisibleLabel: ReactNode\n          },\n          currentValue,\n          index,\n        ) => {\n          const elementWidth = (currentValue as HTMLDivElement).offsetWidth\n\n          const newAccumulatedWidth =\n            accumulator.accumulatedWidth + elementWidth + SIZES_TAG.gap\n\n          const canBeVisible = newAccumulatedWidth <= innerWidth\n\n          return {\n            accumulatedWidth: !canBeVisible\n              ? accumulator.accumulatedWidth\n              : newAccumulatedWidth,\n            lastVisibleElementWidth: canBeVisible\n              ? elementWidth\n              : accumulator.lastVisibleElementWidth,\n            lastVisibleLabel: canBeVisible\n              ? potentiallyNonOverflowedValues[index].label\n              : accumulator.lastVisibleLabel,\n            measuredHiddenTags:\n              accumulator.measuredHiddenTags + (!canBeVisible ? 1 : 0),\n            measuredVisibleTags: [\n              ...accumulator.measuredVisibleTags,\n              canBeVisible && potentiallyNonOverflowedValues[index],\n            ].filter(Boolean) as OptionType[],\n          }\n        },\n        {\n          accumulatedWidth: 0,\n          lastVisibleElementWidth: 0,\n          lastVisibleLabel: '',\n          measuredHiddenTags: 0,\n          measuredVisibleTags: [],\n        },\n      )\n\n      const additionnalElementsWidth =\n        SIZES_TAG.paddings + (refPlusTag.current?.offsetWidth ?? 0)\n      const finalWidth =\n        accumulatedWidth + (measuredHiddenTags ? additionnalElementsWidth : 0)\n\n      const overflowPx = finalWidth - innerWidth\n      const hasOverflow = overflowPx > 0\n      const hasHiddenTags = measuredHiddenTags > 0\n      const lastVisibleElementMaxSize = lastVisibleElementWidth - overflowPx\n\n      // If only one element is selected and it is hidden, we need to show it\n      if (measuredHiddenTags === 1 && measuredVisibleTags.length === 0) {\n        setOverflowAmount(0)\n        setNonOverFlowedValues([potentiallyNonOverflowedValues[0]])\n\n        const newOverflowPx =\n          lastVisibleElementWidth +\n          (measuredHiddenTags > 1 ? additionnalElementsWidth : 0) -\n          innerWidth\n        setLastElementMaxWidth(lastVisibleElementWidth - newOverflowPx)\n        setOverflow(true)\n      }\n\n      // If it overflows with the last tag, we need to add an ellipsis to the last element if there is enough space (>60px)\n      // and if it is a string (do not cut ReactNode label)\n      // else we hide it completely and add it to the overflow amount\n      else if (\n        hasOverflow &&\n        hasHiddenTags &&\n        (lastVisibleElementMaxSize > 65 ||\n          (measuredVisibleTags.length === 1 &&\n            lastVisibleElementMaxSize > 65)) &&\n        typeof lastVisibleLabel === 'string'\n      ) {\n        setLastElementMaxWidth(lastVisibleElementMaxSize)\n        setOverflow(true)\n        setOverflowAmount(measuredHiddenTags)\n        setNonOverFlowedValues(measuredVisibleTags)\n      } else if (hasOverflow && hasHiddenTags) {\n        setLastElementMaxWidth(0)\n        setOverflow(false)\n        setOverflowAmount(measuredHiddenTags + 1)\n        setNonOverFlowedValues(measuredVisibleTags.slice(0, -1))\n      }\n      // Otherwise, we have enough space to show all tags\n      else {\n        setOverflow(false)\n        setOverflowAmount(measuredHiddenTags)\n        setNonOverFlowedValues(measuredVisibleTags)\n      }\n    }\n  }, [\n    selectedData.selectedValues.length,\n    innerWidth,\n    potentiallyNonOverflowedValues,\n  ])\n\n  useEffect(() => {\n    setSelectedData({ type: 'update' })\n  }, [setSelectedData, options])\n\n  useEffect(() => {\n    const getWidth = () => {\n      if (refTag.current) {\n        setInnerWidth(refTag.current.offsetWidth)\n      } else {\n        setInnerWidth(\n          innerRef.current?.offsetWidth ??\n            0 - (arrowRef.current?.offsetWidth ?? 0) - SIZES_TAG.paddings,\n        )\n      }\n    }\n    getWidth()\n    window.addEventListener('resize', getWidth)\n\n    return () => window.removeEventListener('resize', getWidth)\n  }, [innerRef, refTag, selectedData.selectedValues])\n\n  const shouldDisplayValues = useMemo(() => {\n    if (multiselect) {\n      return (\n        potentiallyNonOverflowedValues.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    options,\n    potentiallyNonOverflowedValues.length,\n    selectedData.selectedValues,\n  ])\n\n  return (\n    <Tooltip text={tooltip}>\n      <StyledInputWrapper\n        aria-controls={dropdownId}\n        aria-expanded={isDropdownVisible}\n        aria-label={label}\n        autoFocus={autoFocus}\n        data-disabled={disabled}\n        data-dropdownvisible={isDropdownVisible}\n        data-readonly={readOnly}\n        data-size={size}\n        data-state={state}\n        data-testid={dataTestId}\n        id={id}\n        onClick={\n          openable ? () => setIsDropdownVisible(!isDropdownVisible) : undefined\n        }\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        role=\"combobox\"\n        tabIndex={0}\n      >\n        {shouldDisplayValues ? (\n          <DisplayValues\n            disabled={disabled}\n            lastElementMaxWidth={lastElementMaxWidth}\n            measureRef={measureRef}\n            nonOverflowedValues={nonOverflowedValues}\n            overflow={overflow}\n            overflowAmount={overflowAmount}\n            overflowed={!!overflowAmount}\n            potentiallyNonOverflowedValues={potentiallyNonOverflowedValues}\n            readOnly={readOnly}\n            refPlusTag={refPlusTag}\n            refTag={refTag}\n            size={size}\n          />\n        ) : (\n          <Placeholder\n            as=\"p\"\n            disabled={disabled}\n            prominence=\"weak\"\n            sentiment=\"neutral\"\n            variant={size === 'large' ? 'body' : 'bodySmall'}\n          >\n            {placeholder}\n          </Placeholder>\n        )}\n        <StateStack alignItems=\"center\" direction=\"row\" gap={1} ref={arrowRef}>\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              data-testid=\"clear-all\"\n              disabled={disabled || !selectedData.selectedValues[0] || readOnly}\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              size=\"small\"\n              variant=\"ghost\"\n            >\n              <CloseIcon />\n            </Button>\n          ) : null}\n          <ArrowDownIcon\n            aria-label=\"show dropdown\"\n            disabled={disabled || readOnly}\n            sentiment=\"neutral\"\n            size=\"small\"\n          />\n        </StateStack>\n      </StyledInputWrapper>\n    </Tooltip>\n  )\n}\n\nexport { SelectBar, StyledInputWrapper }\n"]} */",
151
+ styles: "overflow:hidden;max-width:100%;height:100%/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInput/SelectBar.tsx"],"names":[],"mappings":"AA2MsC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInput/SelectBar.tsx","sourcesContent":["'use client'\n\nimport styled from '@emotion/styled'\nimport {\n  AlertCircleIcon,\n  ArrowDownIcon,\n  CheckCircleIcon,\n  CloseIcon,\n  PlusIcon,\n} from '@ultraviolet/icons'\nimport type { ReactNode, RefObject } from 'react'\nimport { useEffect, useLayoutEffect, useMemo, useRef, useState } from 'react'\nimport { Button } from '../Button'\nimport { StyledChildrenContainer } from '../Popup'\nimport { Stack } from '../Stack'\nimport { Tag } from '../Tag'\nimport { Text } from '../Text'\nimport { Tooltip } from '../Tooltip'\nimport { findOptionInOptions } from './findOptionInOptions'\nimport { useSelectInput } from './SelectInputProvider'\nimport type { OptionType } from './types'\nimport { INPUT_SIZE_HEIGHT } from './types'\n\nconst SIZES_TAG = {\n  gap: 8,\n  paddings: 16,\n  plusTag: 48,\n}\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  dropdownId?: string\n}\n\ntype DisplayValuesProps = {\n  refTag: RefObject<HTMLDivElement | null>\n  nonOverflowedValues: OptionType[]\n  potentiallyNonOverflowedValues: OptionType[]\n  disabled: boolean\n  readOnly: boolean\n  overflowed: boolean\n  overflowAmount: number\n  measureRef: RefObject<HTMLDivElement | null>\n  size: 'small' | 'medium' | 'large'\n  lastElementMaxWidth: number\n  overflow?: boolean\n  refPlusTag: RefObject<HTMLDivElement | null>\n  displayShadowCopy?: boolean\n}\n\nconst StateStack = styled(Stack)`\n  padding-right: ${({ theme }) => theme.space['2']};\n  display: flex;\n`\nconst Placeholder = styled(Text)`\nuser-select: none;\nalign-self: center;\n`\n\nconst StyledInputWrapper = styled.div<{\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: grid;\n  width: 100%;\n  gap: ${({ theme }) => theme.space[1]};\n  grid-template-columns: 1fr auto ;\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\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  shouldForwardProp: prop => !['lastElementMaxWidth', 'hidden'].includes(prop),\n})<{\n  lastElementMaxWidth?: number\n  hidden?: boolean\n}>`\n  height: max-content;\n  width: fit-content;\n  min-width: ${({ lastElementMaxWidth }) =>\n    lastElementMaxWidth ? 'auto' : 'fit-content'};\n\n  max-width: ${({ lastElementMaxWidth, hidden }) =>\n    lastElementMaxWidth && !hidden ? `${lastElementMaxWidth}px` : '100%'};\n\n  ${({ hidden }) =>\n    hidden\n      ? 'visibility: hidden;'\n      : `\n  text-overflow: ellipsis;\n  overflow: hidden;`}\n\n  & > ${StyledChildrenContainer} {\n    overflow: hidden;\n  }\n`\n\nconst SelectedValues = styled(Text)`\n  text-overflow: ellipsis;\n  overflow: hidden;\n  white-space: nowrap;\n  align-self: center;\n`\nconst PlusTag = styled(Tag)`\nwidth: ${({ theme }) => theme.sizing[500]};\n;\n`\n\nconst MultiselectStack = styled(Stack)`\noverflow: hidden;\nmax-width: 100%;\nheight: 100%;\n`\n\nconst DisplayValues = ({\n  refTag,\n  nonOverflowedValues,\n  potentiallyNonOverflowedValues,\n  disabled,\n  readOnly,\n  overflowed,\n  overflowAmount,\n  size,\n  measureRef,\n  lastElementMaxWidth,\n  overflow,\n  refPlusTag,\n  displayShadowCopy,\n}: DisplayValuesProps) => {\n  const { multiselect, selectedData, setSelectedData, options, onChange } =\n    useSelectInput()\n\n  return multiselect ? (\n    <MultiselectStack\n      alignItems=\"center\"\n      direction=\"row\"\n      gap=\"1\"\n      ref={refTag}\n      wrap=\"nowrap\"\n    >\n      {/* Hidden div to measure the width of the tags */}\n      {displayShadowCopy ? (\n        <div\n          ref={measureRef}\n          style={{\n            position: 'absolute',\n          }}\n        >\n          {potentiallyNonOverflowedValues.map(option => (\n            <CustomTag\n              className={option.value}\n              hidden\n              key={option.value}\n              onClose={() => {}}\n            >\n              {option?.label}\n            </CustomTag>\n          ))}\n        </div>\n      ) : null}\n      {nonOverflowedValues.map((option, index) => (\n        <CustomTag\n          data-testid=\"selected-options-tags\"\n          disabled={disabled}\n          key={option?.value}\n          lastElementMaxWidth={\n            index === nonOverflowedValues.length - 1 && overflow\n              ? lastElementMaxWidth\n              : 0\n          }\n          onClose={\n            !readOnly\n              ? event => {\n                  event.stopPropagation()\n                  setSelectedData({\n                    clickedOption: option,\n                    type: 'selectOption',\n                  })\n                  const newSelectedValues = selectedData.selectedValues?.filter(\n                    val => val !== option.value,\n                  )\n                  onChange?.(newSelectedValues)\n                }\n              : undefined\n          }\n          sentiment=\"neutral\"\n        >\n          {option?.label}\n        </CustomTag>\n      ))}\n\n      {overflowed ? (\n        <Stack justifyContent=\"center\" ref={refPlusTag}>\n          <PlusTag\n            aria-label=\"Plus tag\"\n            data-testid=\"plus-tag\"\n            disabled={disabled}\n            key=\"+\"\n            sentiment=\"neutral\"\n          >\n            <PlusIcon size=\"xsmall\" />\n            {overflowAmount}\n          </PlusTag>\n        </Stack>\n      ) : null}\n    </MultiselectStack>\n  ) : (\n    <SelectedValues\n      as=\"div\"\n      disabled={disabled}\n      prominence=\"default\"\n      sentiment=\"neutral\"\n      variant={size === 'large' ? 'body' : 'bodySmall'}\n    >\n      {selectedData.selectedValues[0]\n        ? findOptionInOptions(options, selectedData.selectedValues[0])?.label\n        : null}\n    </SelectedValues>\n  )\n}\n\nconst 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  dropdownId,\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 [displayShadowCopy, setDisplayShadowCopy] = useState(false)\n  const refTag = useRef<HTMLDivElement>(null)\n  const measureRef = useRef<HTMLDivElement>(null)\n  const arrowRef = useRef<HTMLDivElement>(null)\n  const refPlusTag = useRef<HTMLDivElement>(null)\n  // width - width of the arrow (in px) - padding between tags (in px)\n  const [innerWidth, setInnerWidth] = useState(\n    innerRef.current?.offsetWidth ??\n      0 - (arrowRef.current?.offsetWidth ?? 0) - SIZES_TAG.paddings,\n  )\n  const [overflowAmount, setOverflowAmount] = useState(0)\n  const [overflow, setOverflow] = useState(false)\n  const [lastElementMaxWidth, setLastElementMaxWidth] = 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 potentiallyNonOverflowedValues = useMemo(\n    () =>\n      selectedData.selectedValues\n        .map(selectedValue => findOptionInOptions(options, selectedValue))\n        .filter((option): option is OptionType => !!option),\n    [options, selectedData.selectedValues],\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  // First we display shadow copy of tags to render it\n  useEffect(() => {\n    setDisplayShadowCopy(true)\n  }, [selectedData.selectedValues.length])\n\n  // We then want to measure the tags length before displaying them\n  // so we can determine if there is an overflow or not\n  // We use useLayoutEffect to ensure the measurement is done before the browser paints\n  useLayoutEffect(() => {\n    if (selectedData.selectedValues.length === 0) {\n      setOverflowAmount(0)\n      setNonOverFlowedValues([])\n    }\n\n    if (measureRef.current && selectedData.selectedValues.length > 0) {\n      const toMeasureElements: HTMLCollection = measureRef.current.children\n      const toMeasureElementsArray = [...toMeasureElements]\n\n      const {\n        measuredVisibleTags,\n        measuredHiddenTags,\n        accumulatedWidth,\n        lastVisibleElementWidth,\n        lastVisibleLabel,\n      } = toMeasureElementsArray.reduce(\n        (\n          accumulator: {\n            measuredVisibleTags: OptionType[]\n            measuredHiddenTags: number\n            accumulatedWidth: number\n            lastVisibleElementWidth: number\n            lastVisibleLabel: ReactNode\n          },\n          currentValue,\n          index,\n        ) => {\n          const elementWidth = (currentValue as HTMLDivElement).offsetWidth\n\n          const newAccumulatedWidth =\n            accumulator.accumulatedWidth + elementWidth + SIZES_TAG.gap\n\n          const canBeVisible = newAccumulatedWidth <= innerWidth\n\n          return {\n            accumulatedWidth: !canBeVisible\n              ? accumulator.accumulatedWidth\n              : newAccumulatedWidth,\n            lastVisibleElementWidth: canBeVisible\n              ? elementWidth\n              : accumulator.lastVisibleElementWidth,\n            lastVisibleLabel: canBeVisible\n              ? potentiallyNonOverflowedValues[index].label\n              : accumulator.lastVisibleLabel,\n            measuredHiddenTags:\n              accumulator.measuredHiddenTags + (!canBeVisible ? 1 : 0),\n            measuredVisibleTags: [\n              ...accumulator.measuredVisibleTags,\n              canBeVisible && potentiallyNonOverflowedValues[index],\n            ].filter(Boolean) as OptionType[],\n          }\n        },\n        {\n          accumulatedWidth: 0,\n          lastVisibleElementWidth: 0,\n          lastVisibleLabel: '',\n          measuredHiddenTags: 0,\n          measuredVisibleTags: [],\n        },\n      )\n\n      const additionnalElementsWidth =\n        SIZES_TAG.paddings + (refPlusTag.current?.offsetWidth ?? 0)\n      const finalWidth =\n        accumulatedWidth + (measuredHiddenTags ? additionnalElementsWidth : 0)\n\n      const overflowPx = finalWidth - innerWidth\n      const hasOverflow = overflowPx > 0\n      const hasHiddenTags = measuredHiddenTags > 0\n      const lastVisibleElementMaxSize = lastVisibleElementWidth - overflowPx\n\n      // If only one element is selected and it is hidden, we need to show it\n      if (measuredHiddenTags === 1 && measuredVisibleTags.length === 0) {\n        setOverflowAmount(0)\n        setNonOverFlowedValues([potentiallyNonOverflowedValues[0]])\n\n        const newOverflowPx =\n          lastVisibleElementWidth +\n          (measuredHiddenTags > 1 ? additionnalElementsWidth : 0) -\n          innerWidth\n        setLastElementMaxWidth(lastVisibleElementWidth - newOverflowPx)\n        setOverflow(true)\n      }\n\n      // If it overflows with the last tag, we need to add an ellipsis to the last element if there is enough space (>60px)\n      // and if it is a string (do not cut ReactNode label)\n      // else we hide it completely and add it to the overflow amount\n      else if (\n        hasOverflow &&\n        hasHiddenTags &&\n        (lastVisibleElementMaxSize > 65 ||\n          (measuredVisibleTags.length === 1 &&\n            lastVisibleElementMaxSize > 65)) &&\n        typeof lastVisibleLabel === 'string'\n      ) {\n        setLastElementMaxWidth(lastVisibleElementMaxSize)\n        setOverflow(true)\n        setOverflowAmount(measuredHiddenTags)\n        setNonOverFlowedValues(measuredVisibleTags)\n      } else if (hasOverflow && hasHiddenTags) {\n        setLastElementMaxWidth(0)\n        setOverflow(false)\n        setOverflowAmount(measuredHiddenTags + 1)\n        setNonOverFlowedValues(measuredVisibleTags.slice(0, -1))\n      }\n      // Otherwise, we have enough space to show all tags\n      else {\n        setOverflow(false)\n        setOverflowAmount(measuredHiddenTags)\n        setNonOverFlowedValues(measuredVisibleTags)\n      }\n    }\n    setDisplayShadowCopy(false)\n  }, [\n    displayShadowCopy,\n    innerWidth,\n    potentiallyNonOverflowedValues,\n    selectedData.selectedValues.length,\n  ])\n\n  useEffect(() => {\n    setSelectedData({ type: 'update' })\n  }, [setSelectedData, options])\n\n  useEffect(() => {\n    const getWidth = () => {\n      if (refTag.current) {\n        setInnerWidth(refTag.current.offsetWidth)\n      } else {\n        setInnerWidth(\n          innerRef.current?.offsetWidth ??\n            0 - (arrowRef.current?.offsetWidth ?? 0) - SIZES_TAG.paddings,\n        )\n      }\n    }\n    getWidth()\n    window.addEventListener('resize', getWidth)\n\n    return () => window.removeEventListener('resize', getWidth)\n  }, [innerRef, refTag, selectedData.selectedValues])\n\n  const shouldDisplayValues = useMemo(() => {\n    if (multiselect) {\n      return (\n        potentiallyNonOverflowedValues.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    options,\n    potentiallyNonOverflowedValues.length,\n    selectedData.selectedValues,\n  ])\n\n  return (\n    <Tooltip text={tooltip}>\n      <StyledInputWrapper\n        aria-controls={dropdownId}\n        aria-expanded={isDropdownVisible}\n        aria-label={label}\n        autoFocus={autoFocus}\n        data-disabled={disabled}\n        data-dropdownvisible={isDropdownVisible}\n        data-readonly={readOnly}\n        data-size={size}\n        data-state={state}\n        data-testid={dataTestId}\n        id={id}\n        onClick={\n          openable ? () => setIsDropdownVisible(!isDropdownVisible) : undefined\n        }\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        role=\"combobox\"\n        tabIndex={0}\n      >\n        {shouldDisplayValues ? (\n          <DisplayValues\n            disabled={disabled}\n            displayShadowCopy={displayShadowCopy}\n            lastElementMaxWidth={lastElementMaxWidth}\n            measureRef={measureRef}\n            nonOverflowedValues={nonOverflowedValues}\n            overflow={overflow}\n            overflowAmount={overflowAmount}\n            overflowed={!!overflowAmount}\n            potentiallyNonOverflowedValues={potentiallyNonOverflowedValues}\n            readOnly={readOnly}\n            refPlusTag={refPlusTag}\n            refTag={refTag}\n            size={size}\n          />\n        ) : (\n          <Placeholder\n            as=\"p\"\n            disabled={disabled}\n            prominence=\"weak\"\n            sentiment=\"neutral\"\n            variant={size === 'large' ? 'body' : 'bodySmall'}\n          >\n            {placeholder}\n          </Placeholder>\n        )}\n        <StateStack alignItems=\"center\" direction=\"row\" gap={1} ref={arrowRef}>\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              data-testid=\"clear-all\"\n              disabled={disabled || !selectedData.selectedValues[0] || readOnly}\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              size=\"small\"\n              variant=\"ghost\"\n            >\n              <CloseIcon />\n            </Button>\n          ) : null}\n          <ArrowDownIcon\n            aria-label=\"show dropdown\"\n            disabled={disabled || readOnly}\n            sentiment=\"neutral\"\n            size=\"small\"\n          />\n        </StateStack>\n      </StyledInputWrapper>\n    </Tooltip>\n  )\n}\n\nexport { SelectBar, StyledInputWrapper }\n"]} */",
152
152
  toString: _EMOTION_STRINGIFIED_CSS_ERROR__
153
153
  });
154
154
  const DisplayValues = ({
@@ -163,7 +163,8 @@ const DisplayValues = ({
163
163
  measureRef,
164
164
  lastElementMaxWidth,
165
165
  overflow,
166
- refPlusTag
166
+ refPlusTag,
167
+ displayShadowCopy
167
168
  }) => {
168
169
  const {
169
170
  multiselect,
@@ -173,10 +174,10 @@ const DisplayValues = ({
173
174
  onChange
174
175
  } = useSelectInput();
175
176
  return multiselect ? /* @__PURE__ */ jsxs(MultiselectStack, { alignItems: "center", direction: "row", gap: "1", ref: refTag, wrap: "nowrap", children: [
176
- /* @__PURE__ */ jsx("div", { ref: measureRef, style: {
177
+ displayShadowCopy ? /* @__PURE__ */ jsx("div", { ref: measureRef, style: {
177
178
  position: "absolute"
178
179
  }, children: potentiallyNonOverflowedValues.map((option) => /* @__PURE__ */ jsx(CustomTag, { className: option.value, hidden: true, onClose: () => {
179
- }, children: option?.label }, option.value)) }),
180
+ }, children: option?.label }, option.value)) }) : null,
180
181
  nonOverflowedValues.map((option, index) => /* @__PURE__ */ jsx(CustomTag, { "data-testid": "selected-options-tags", disabled, lastElementMaxWidth: index === nonOverflowedValues.length - 1 && overflow ? lastElementMaxWidth : 0, onClose: !readOnly ? (event) => {
181
182
  event.stopPropagation();
182
183
  setSelectedData({
@@ -218,6 +219,7 @@ const SelectBar = ({
218
219
  multiselect
219
220
  } = useSelectInput();
220
221
  const openable = !(readOnly || disabled);
222
+ const [displayShadowCopy, setDisplayShadowCopy] = useState(false);
221
223
  const refTag = useRef(null);
222
224
  const measureRef = useRef(null);
223
225
  const arrowRef = useRef(null);
@@ -244,6 +246,9 @@ const SelectBar = ({
244
246
  return "neutral";
245
247
  }, [error, success]);
246
248
  useEffect(() => {
249
+ setDisplayShadowCopy(true);
250
+ }, [selectedData.selectedValues.length]);
251
+ useLayoutEffect(() => {
247
252
  if (selectedData.selectedValues.length === 0) {
248
253
  setOverflowAmount(0);
249
254
  setNonOverFlowedValues([]);
@@ -303,7 +308,8 @@ const SelectBar = ({
303
308
  setNonOverFlowedValues(measuredVisibleTags);
304
309
  }
305
310
  }
306
- }, [selectedData.selectedValues.length, innerWidth, potentiallyNonOverflowedValues]);
311
+ setDisplayShadowCopy(false);
312
+ }, [displayShadowCopy, innerWidth, potentiallyNonOverflowedValues, selectedData.selectedValues.length]);
307
313
  useEffect(() => {
308
314
  setSelectedData({
309
315
  type: "update"
@@ -340,7 +346,7 @@ const SelectBar = ({
340
346
  }
341
347
  return ["Enter", " "].includes(event.key) && openable ? setIsDropdownVisible(!isDropdownVisible) : null;
342
348
  }, ref: innerRef, role: "combobox", tabIndex: 0, children: [
343
- shouldDisplayValues ? /* @__PURE__ */ jsx(DisplayValues, { disabled, lastElementMaxWidth, measureRef, nonOverflowedValues, overflow, overflowAmount, overflowed: !!overflowAmount, potentiallyNonOverflowedValues, readOnly, refPlusTag, refTag, size }) : /* @__PURE__ */ jsx(Placeholder, { as: "p", disabled, prominence: "weak", sentiment: "neutral", variant: size === "large" ? "body" : "bodySmall", children: placeholder }),
349
+ shouldDisplayValues ? /* @__PURE__ */ jsx(DisplayValues, { disabled, displayShadowCopy, lastElementMaxWidth, measureRef, nonOverflowedValues, overflow, overflowAmount, overflowed: !!overflowAmount, potentiallyNonOverflowedValues, readOnly, refPlusTag, refTag, size }) : /* @__PURE__ */ jsx(Placeholder, { as: "p", disabled, prominence: "weak", sentiment: "neutral", variant: size === "large" ? "body" : "bodySmall", children: placeholder }),
344
350
  /* @__PURE__ */ jsxs(StateStack, { alignItems: "center", direction: "row", gap: 1, ref: arrowRef, children: [
345
351
  error ? /* @__PURE__ */ jsx(AlertCircleIcon, { sentiment: "danger" }) : null,
346
352
  success && !error ? /* @__PURE__ */ jsx(CheckCircleIcon, { sentiment: "success" }) : null,