@ultraviolet/ui 2.0.0-beta.6 → 2.0.0-beta.8
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.
- package/dist/components/Popup/index.cjs +4 -3
- package/dist/components/Popup/index.d.ts +4 -0
- package/dist/components/Popup/index.js +5 -4
- package/dist/components/SelectInputV2/SearchBarDropdown.cjs +16 -8
- package/dist/components/SelectInputV2/SearchBarDropdown.d.ts +3 -1
- package/dist/components/SelectInputV2/SearchBarDropdown.js +17 -9
- package/dist/components/SelectInputV2/SelectBar.cjs +158 -62
- package/dist/components/SelectInputV2/SelectBar.d.ts +4 -20
- package/dist/components/SelectInputV2/SelectBar.js +155 -59
- package/dist/components/SelectInputV2/types.cjs +0 -5
- package/dist/components/SelectInputV2/types.d.ts +0 -4
- package/dist/components/SelectInputV2/types.js +1 -6
- package/dist/utils/searchAlgorithm.cjs +30 -0
- package/dist/utils/searchAlgorithm.d.ts +12 -0
- package/dist/utils/searchAlgorithm.js +30 -0
- package/package.json +2 -2
|
@@ -4,43 +4,50 @@ import _styled from "@emotion/styled/base";
|
|
|
4
4
|
import { AlertCircleIcon, CheckCircleIcon, CloseIcon, ArrowDownIcon, PlusIcon } from "@ultraviolet/icons";
|
|
5
5
|
import { useRef, useState, useMemo, useEffect } from "react";
|
|
6
6
|
import { Button } from "../Button/index.js";
|
|
7
|
+
import { StyledChildrenContainer } from "../Popup/index.js";
|
|
7
8
|
import { Stack } from "../Stack/index.js";
|
|
8
9
|
import { Tag } from "../Tag/index.js";
|
|
9
10
|
import { Text } from "../Text/index.js";
|
|
10
11
|
import { Tooltip } from "../Tooltip/index.js";
|
|
11
12
|
import { useSelectInput } from "./SelectInputProvider.js";
|
|
12
13
|
import { findOptionInOptions } from "./findOptionInOptions.js";
|
|
13
|
-
import {
|
|
14
|
+
import { INPUT_SIZE_HEIGHT } from "./types.js";
|
|
14
15
|
function _EMOTION_STRINGIFIED_CSS_ERROR__() {
|
|
15
16
|
return "You have tried to stringify object returned from `css` function. It isn't supposed to be used directly (e.g. as value of the `className` prop), but rather handed to emotion so it can handle it (e.g. as value of `css` prop).";
|
|
16
17
|
}
|
|
18
|
+
const SIZES_TAG = {
|
|
19
|
+
paddings: 16,
|
|
20
|
+
gap: 8
|
|
21
|
+
};
|
|
17
22
|
const StateStack = /* @__PURE__ */ _styled(Stack, process.env.NODE_ENV === "production" ? {
|
|
18
|
-
target: "
|
|
23
|
+
target: "ejy0aca6"
|
|
19
24
|
} : {
|
|
20
|
-
target: "
|
|
25
|
+
target: "ejy0aca6",
|
|
21
26
|
label: "StateStack"
|
|
22
27
|
})("padding-right:", ({
|
|
23
28
|
theme
|
|
24
|
-
}) => theme.space["2"], ";display:flex;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/SelectBar.tsx"],"names":[],"mappings":"AAgDgC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/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 { RefObject } from 'react'\nimport { useEffect, useMemo, useRef, useState } from 'react'\nimport { Button } from '../Button'\nimport { Stack } from '../Stack'\nimport { Tag } from '../Tag'\nimport { Text } from '../Text'\nimport { Tooltip } from '../Tooltip'\nimport { useSelectInput } from './SelectInputProvider'\nimport { findOptionInOptions } from './findOptionInOptions'\nimport type { DataType, OptionType } from './types'\nimport { INPUT_SIZE_HEIGHT, SIZES_TAG } from './types'\n\ntype SelectBarProps = {\n  size: 'small' | 'medium' | 'large'\n  clearable: boolean\n  disabled: boolean\n  readOnly: boolean\n  placeholder: string\n  success?: string\n  error?: string | boolean\n  autoFocus?: boolean\n  innerRef: RefObject<HTMLDivElement | null>\n  id?: string\n  'data-testid': string\n  label?: string\n  tooltip?: string\n}\n\ntype DisplayValuesProps = {\n  refTag: RefObject<HTMLDivElement | null>\n  nonOverflowedValues: OptionType[]\n  disabled: boolean\n  readOnly: boolean\n  overflowed: boolean\n  overflowAmount: number\n  size: 'small' | 'medium' | 'large'\n}\n\nconst StateStack = styled(Stack)`\n  padding-right: ${({ theme }) => theme.space['2']};\n  display: flex;\n`\nconst Placeholder = styled(Text)`\nuser-select: none;\n`\n\nexport const StyledInputWrapper = styled(Stack)<{\n  'data-readonly': boolean\n  'data-disabled': boolean\n  'data-size': 'small' | 'medium' | 'large'\n  'data-state': 'neutral' | 'success' | 'danger'\n  'data-dropdownvisible': boolean\n  'aria-label'?: string\n}>`\n  display: flex;\n  padding: ${({ theme }) => theme.space[1]};\n  padding-right: 0;\n  padding-left: ${({ theme }) => theme.space[2]};\n  cursor: pointer;\n  box-shadow: none;\n  background: ${({ theme }) => theme.colors.neutral.background};\n  border-radius: ${({ theme }) => theme.radii.default};\n  width: 100%;\n  overflow: hidden;\n\n  &[data-size='small'] {\n    height: ${({ theme }) => theme.sizing[INPUT_SIZE_HEIGHT.small]};\n    padding-left: ${({ theme }) => theme.space[1]};\n  }\n  &[data-size='medium'] {\n    height: ${({ theme }) => theme.sizing[INPUT_SIZE_HEIGHT.medium]};\n  }\n  &[data-size='large'] {\n    height: ${({ theme }) => theme.sizing[INPUT_SIZE_HEIGHT.large]};\n  }\n  &[data-state='neutral'] {\n    border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n\n    &:not([data-disabled=\"true\"]):not([data-readonly=\"true\"]):active {\n      border-color: ${({ theme }) => theme.colors.primary.borderHover};\n      box-shadow: ${({ theme }) => theme.shadows.focusPrimary};\n    }\n    &:not([data-disabled='true']):hover {\n      border-color: ${({ theme }) => theme.colors.primary.borderHover};\n      outline: none;\n    }\n\n    &:not([data-disabled='true']):focus-visible {\n      outline: 5px auto Highlight;\n      outline: 5px auto -webkit-focus-ring-color;\n    }\n\n    &[data-dropdownvisible='true'] {\n    border-color: ${({ theme }) => theme.colors.primary.borderHover};\n  }\n  }\n\n  &[data-state='success'] {\n    border: 1px solid ${({ theme }) => theme.colors.success.border};\n    &:not([data-disabled=\"true\"]):not([data-readonly=\"true\"]):active {\n      border-color: ${({ theme }) => theme.colors.success.borderHover};\n      box-shadow: ${({ theme }) => theme.shadows.focusSuccess};\n    }\n\n    &[data-dropdownvisible='true'] {\n      border-color: ${({ theme }) => theme.colors.success.borderHover};\n    }\n  }\n\n  &[data-state='danger'] {\n    border: 1px solid ${({ theme }) => theme.colors.danger.border};\n\n    &:not([data-disabled=\"true\"]):not([data-readonly=\"true\"]):active {\n      border-color: ${({ theme }) => theme.colors.danger.borderHover};\n      box-shadow: ${({ theme }) => theme.shadows.focusDanger};\n    }\n\n    &[data-dropdownvisible='true'] {\n      border-color: ${({ theme }) => theme.colors.danger.borderHover};\n    }\n  }\n\n  &:not([data-disabled='true']):not([data-readonly]):hover {\n    border-color: ${({ theme }) => theme.colors.primary.border};\n  }\n\n  &[data-readonly='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n    border-color: ${({ theme }) => theme.colors.neutral.border};\n    cursor: default;\n  }\n\n  &[data-disabled='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    border-color: ${({ theme }) => theme.colors.neutral.borderDisabled};\n    cursor: not-allowed;\n  }\n`\n\nconst CustomTag = styled(Tag)`\n  height: fit-content;\n  width: fit-content;\n`\n\nconst SelectedValues = styled(Text)`\n  text-overflow: ellipsis;\n  overflow: hidden;\n  white-space: nowrap;\n`\n\nconst isValidSelectedValue = (selectedValue: string, options: DataType) =>\n  !Array.isArray(options)\n    ? Object.keys(options).some(group =>\n        options[group].some(\n          option => option.value === selectedValue && !option.disabled,\n        ),\n      )\n    : options.some(option => option.value === selectedValue && !option.disabled)\n\nconst DisplayValues = ({\n  refTag,\n  nonOverflowedValues,\n  disabled,\n  readOnly,\n  overflowed,\n  overflowAmount,\n  size,\n}: DisplayValuesProps) => {\n  const { multiselect, selectedData, setSelectedData, options, onChange } =\n    useSelectInput()\n\n  return multiselect ? (\n    <Stack\n      direction=\"row\"\n      gap=\"1\"\n      wrap=\"nowrap\"\n      ref={refTag}\n      alignItems=\"center\"\n    >\n      {nonOverflowedValues.map(option => (\n        <CustomTag\n          data-testid=\"selected-options-tags\"\n          key={option?.value}\n          sentiment=\"neutral\"\n          disabled={disabled}\n          onClose={\n            !readOnly\n              ? event => {\n                  event.stopPropagation()\n                  setSelectedData({\n                    type: 'selectOption',\n                    clickedOption: option,\n                  })\n                  const newSelectedValues = selectedData.selectedValues?.filter(\n                    val => val !== option.value,\n                  )\n                  onChange?.(newSelectedValues)\n                }\n              : undefined\n          }\n        >\n          {option?.label}\n        </CustomTag>\n      ))}\n      {overflowed ? (\n        <Tag\n          sentiment=\"neutral\"\n          disabled={disabled}\n          key=\"+\"\n          data-testid=\"plus-tag\"\n          aria-label=\"Plus tag\"\n        >\n          <PlusIcon />\n          {overflowAmount}\n        </Tag>\n      ) : null}\n    </Stack>\n  ) : (\n    <SelectedValues\n      as=\"div\"\n      variant={size === 'large' ? 'body' : 'bodySmall'}\n      disabled={disabled}\n      prominence=\"default\"\n      sentiment=\"neutral\"\n    >\n      {selectedData.selectedValues[0]\n        ? findOptionInOptions(options, selectedData.selectedValues[0])?.label\n        : null}\n    </SelectedValues>\n  )\n}\n\nexport const SelectBar = ({\n  size,\n  clearable,\n  disabled,\n  readOnly,\n  placeholder,\n  success,\n  error,\n  autoFocus,\n  tooltip,\n  innerRef,\n  id,\n  'data-testid': dataTestId,\n  label,\n}: SelectBarProps) => {\n  const {\n    isDropdownVisible,\n    onChange,\n    setIsDropdownVisible,\n    options,\n    selectedData,\n    setSelectedData,\n    multiselect,\n  } = useSelectInput()\n  const openable = !(readOnly || disabled)\n  const refTag = useRef<HTMLDivElement>(null)\n  const width = innerRef.current?.offsetWidth\n  const [overflowAmount, setOverflowAmount] = useState(0)\n  const [nonOverflowedValues, setNonOverFlowedValues] = useState<OptionType[]>(\n    () => {\n      if (selectedData.selectedValues[0]) {\n        const firstSelectOption = findOptionInOptions(\n          options,\n          selectedData.selectedValues[0],\n        )\n\n        return firstSelectOption ? [firstSelectOption] : []\n      }\n\n      return []\n    },\n  )\n\n  const state = useMemo(() => {\n    if (error) {\n      return 'danger'\n    }\n    if (success) {\n      return 'success'\n    }\n\n    return 'neutral'\n  }, [error, success])\n\n  useEffect(() => {\n    // When too many items are selected, too avoid overflow, compute the number of tags to display and add a + tag\n    let tagsWidth = 0\n    let computedOverflowAmount = 0\n    let computedNonOverflowedValues: OptionType[] = []\n    const newSelectedValues = selectedData.selectedValues.filter(\n      selectedValue => isValidSelectedValue(selectedValue, options),\n    )\n    for (const selectedValue of newSelectedValues) {\n      const selectedOption = findOptionInOptions(options, selectedValue)\n      if (\n        selectedOption?.label &&\n        width &&\n        isValidSelectedValue(selectedValue, options)\n      ) {\n        const lengthValue = selectedOption.value.length // Find a better way to find the number of displayed characters?\n        const totalTagWidth =\n          SIZES_TAG.tagWidth + SIZES_TAG.letterWidth * lengthValue\n        if (totalTagWidth + tagsWidth > width - 100) {\n          computedOverflowAmount += 1\n        } else {\n          computedNonOverflowedValues = [\n            ...computedNonOverflowedValues,\n            selectedOption,\n          ]\n          tagsWidth += totalTagWidth\n        }\n      }\n    }\n    setNonOverFlowedValues(computedNonOverflowedValues)\n    setOverflowAmount(computedOverflowAmount)\n  }, [options, selectedData.selectedValues, width])\n\n  useEffect(() => {\n    setSelectedData({ type: 'update' })\n  }, [setSelectedData, options])\n\n  const shouldDisplayValues = useMemo(() => {\n    if (multiselect) {\n      return (\n        nonOverflowedValues.length > 0 ||\n        selectedData.selectedValues.some(\n          selectedValue =>\n            findOptionInOptions(options, selectedValue) !== undefined,\n        )\n      )\n    }\n\n    return (\n      selectedData.selectedValues[0] !== undefined &&\n      findOptionInOptions(options, selectedData.selectedValues[0]) !== undefined\n    )\n  }, [\n    multiselect,\n    nonOverflowedValues.length,\n    options,\n    selectedData.selectedValues,\n  ])\n\n  return (\n    <Tooltip text={tooltip}>\n      <StyledInputWrapper\n        role=\"combobox\"\n        id={id}\n        data-disabled={disabled}\n        data-readonly={readOnly}\n        data-size={size}\n        data-dropdownvisible={isDropdownVisible}\n        data-state={state}\n        direction=\"row\"\n        wrap=\"nowrap\"\n        gap=\"1\"\n        justifyContent=\"space-between\"\n        alignItems=\"center\"\n        onClick={\n          openable ? () => setIsDropdownVisible(!isDropdownVisible) : undefined\n        }\n        data-testid={dataTestId}\n        autoFocus={autoFocus}\n        onKeyDown={event => {\n          if (event.key === 'ArrowDown') {\n            if (!isDropdownVisible) {\n              setIsDropdownVisible(true)\n            } else {\n              document.getElementById(`option-0`)?.focus()\n            }\n          }\n          if (event.key === ' ') {\n            event.preventDefault()\n          }\n\n          return ['Enter', ' '].includes(event.key) && openable\n            ? setIsDropdownVisible(!isDropdownVisible)\n            : null\n        }}\n        ref={innerRef}\n        aria-haspopup=\"listbox\"\n        aria-expanded={isDropdownVisible}\n        tabIndex={0}\n        aria-label={label}\n      >\n        {shouldDisplayValues ? (\n          <DisplayValues\n            refTag={refTag}\n            nonOverflowedValues={nonOverflowedValues}\n            disabled={disabled}\n            readOnly={readOnly}\n            overflowed={!!overflowAmount}\n            overflowAmount={overflowAmount}\n            size={size}\n          />\n        ) : (\n          <Placeholder\n            as=\"p\"\n            variant={size === 'large' ? 'body' : 'bodySmall'}\n            sentiment=\"neutral\"\n            disabled={disabled}\n            prominence=\"weak\"\n          >\n            {placeholder}\n          </Placeholder>\n        )}\n        <StateStack direction=\"row\" gap={1} alignItems=\"center\">\n          {error ? <AlertCircleIcon sentiment=\"danger\" /> : null}\n          {success && !error ? <CheckCircleIcon sentiment=\"success\" /> : null}\n          {clearable && selectedData.selectedValues.length > 0 ? (\n            <Button\n              aria-label=\"clear value\"\n              disabled={disabled || !selectedData.selectedValues[0] || readOnly}\n              variant=\"ghost\"\n              size=\"small\"\n              onClick={event => {\n                event.stopPropagation()\n                setSelectedData({ type: 'clearAll' })\n                if (multiselect) {\n                  onChange?.([])\n                } else {\n                  onChange?.('')\n                }\n              }}\n              sentiment=\"neutral\"\n              data-testid=\"clear-all\"\n            >\n              <CloseIcon />\n            </Button>\n          ) : null}\n          <ArrowDownIcon\n            aria-label=\"show dropdown\"\n            size=\"small\"\n            sentiment=\"neutral\"\n            disabled={disabled || readOnly}\n          />\n        </StateStack>\n      </StyledInputWrapper>\n    </Tooltip>\n  )\n}\n"]} */"));
|
|
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/SelectInputV2/SelectBar.tsx"],"names":[],"mappings":"AA4DgC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/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 { useSelectInput } from './SelectInputProvider'\nimport { findOptionInOptions } from './findOptionInOptions'\nimport type { OptionType } from './types'\nimport { INPUT_SIZE_HEIGHT } from './types'\n\nconst SIZES_TAG = {\n  paddings: 16,\n  plusTag: 48,\n  gap: 8,\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}\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      direction=\"row\"\n      gap=\"1\"\n      wrap=\"nowrap\"\n      ref={refTag}\n      alignItems=\"center\"\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            onClose={() => {}}\n            className={option.value}\n            key={option.value}\n            hidden\n          >\n            {option?.label}\n          </CustomTag>\n        ))}\n      </div>\n      {nonOverflowedValues.map((option, index) => (\n        <CustomTag\n          data-testid=\"selected-options-tags\"\n          sentiment=\"neutral\"\n          key={option?.value}\n          disabled={disabled}\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                    type: 'selectOption',\n                    clickedOption: option,\n                  })\n                  const newSelectedValues = selectedData.selectedValues?.filter(\n                    val => val !== option.value,\n                  )\n                  onChange?.(newSelectedValues)\n                }\n              : undefined\n          }\n        >\n          {option?.label}\n        </CustomTag>\n      ))}\n\n      {overflowed ? (\n        <Stack ref={refPlusTag} justifyContent=\"center\">\n          <PlusTag\n            sentiment=\"neutral\"\n            disabled={disabled}\n            key=\"+\"\n            data-testid=\"plus-tag\"\n            aria-label=\"Plus tag\"\n          >\n            <PlusIcon size=\"xsmall\" />\n            {overflowAmount}\n          </PlusTag>\n        </Stack>\n      ) : null}\n    </MultiselectStack>\n  ) : (\n    <SelectedValues\n      as=\"div\"\n      variant={size === 'large' ? 'body' : 'bodySmall'}\n      disabled={disabled}\n      prominence=\"default\"\n      sentiment=\"neutral\"\n    >\n      {selectedData.selectedValues[0]\n        ? findOptionInOptions(options, selectedData.selectedValues[0])?.label\n        : null}\n    </SelectedValues>\n  )\n}\n\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}: 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            measuredVisibleTags: [\n              ...accumulator.measuredVisibleTags,\n              canBeVisible && potentiallyNonOverflowedValues[index],\n            ].filter(Boolean) as OptionType[],\n            measuredHiddenTags:\n              accumulator.measuredHiddenTags + (!canBeVisible ? 1 : 0),\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          }\n        },\n        {\n          measuredVisibleTags: [],\n          measuredHiddenTags: 0,\n          accumulatedWidth: 0,\n          lastVisibleElementWidth: 0,\n          lastVisibleLabel: '',\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    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        role=\"combobox\"\n        id={id}\n        data-disabled={disabled}\n        data-readonly={readOnly}\n        data-size={size}\n        data-dropdownvisible={isDropdownVisible}\n        data-state={state}\n        onClick={\n          openable ? () => setIsDropdownVisible(!isDropdownVisible) : undefined\n        }\n        data-testid={dataTestId}\n        autoFocus={autoFocus}\n        onKeyDown={event => {\n          if (event.key === 'ArrowDown') {\n            if (!isDropdownVisible) {\n              setIsDropdownVisible(true)\n            } else {\n              document.getElementById(`option-0`)?.focus()\n            }\n          }\n          if (event.key === ' ') {\n            event.preventDefault()\n          }\n\n          return ['Enter', ' '].includes(event.key) && openable\n            ? setIsDropdownVisible(!isDropdownVisible)\n            : null\n        }}\n        ref={innerRef}\n        aria-haspopup=\"listbox\"\n        aria-expanded={isDropdownVisible}\n        tabIndex={0}\n        aria-label={label}\n      >\n        {shouldDisplayValues ? (\n          <DisplayValues\n            refTag={refTag}\n            nonOverflowedValues={nonOverflowedValues}\n            potentiallyNonOverflowedValues={potentiallyNonOverflowedValues}\n            disabled={disabled}\n            readOnly={readOnly}\n            overflowed={!!overflowAmount}\n            overflowAmount={overflowAmount}\n            size={size}\n            measureRef={measureRef}\n            lastElementMaxWidth={lastElementMaxWidth}\n            overflow={overflow}\n            refPlusTag={refPlusTag}\n          />\n        ) : (\n          <Placeholder\n            as=\"p\"\n            variant={size === 'large' ? 'body' : 'bodySmall'}\n            sentiment=\"neutral\"\n            disabled={disabled}\n            prominence=\"weak\"\n          >\n            {placeholder}\n          </Placeholder>\n        )}\n        <StateStack direction=\"row\" gap={1} alignItems=\"center\" 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              disabled={disabled || !selectedData.selectedValues[0] || readOnly}\n              variant=\"ghost\"\n              size=\"small\"\n              onClick={event => {\n                event.stopPropagation()\n                setSelectedData({ type: 'clearAll' })\n                if (multiselect) {\n                  onChange?.([])\n                } else {\n                  onChange?.('')\n                }\n              }}\n              sentiment=\"neutral\"\n              data-testid=\"clear-all\"\n            >\n              <CloseIcon />\n            </Button>\n          ) : null}\n          <ArrowDownIcon\n            aria-label=\"show dropdown\"\n            size=\"small\"\n            sentiment=\"neutral\"\n            disabled={disabled || readOnly}\n          />\n        </StateStack>\n      </StyledInputWrapper>\n    </Tooltip>\n  )\n}\n\nexport { SelectBar, StyledInputWrapper }\n"]} */"));
|
|
25
30
|
const Placeholder = /* @__PURE__ */ _styled(Text, process.env.NODE_ENV === "production" ? {
|
|
26
|
-
target: "
|
|
31
|
+
target: "ejy0aca5"
|
|
27
32
|
} : {
|
|
28
|
-
target: "
|
|
33
|
+
target: "ejy0aca5",
|
|
29
34
|
label: "Placeholder"
|
|
30
35
|
})(process.env.NODE_ENV === "production" ? {
|
|
31
|
-
name: "
|
|
32
|
-
styles: "user-select:none"
|
|
36
|
+
name: "14eg98m",
|
|
37
|
+
styles: "user-select:none;align-self:center"
|
|
33
38
|
} : {
|
|
34
|
-
name: "
|
|
35
|
-
styles: "user-select:none/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/SelectBar.tsx"],"names":[],"mappings":"AAoDgC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/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 { RefObject } from 'react'\nimport { useEffect, useMemo, useRef, useState } from 'react'\nimport { Button } from '../Button'\nimport { Stack } from '../Stack'\nimport { Tag } from '../Tag'\nimport { Text } from '../Text'\nimport { Tooltip } from '../Tooltip'\nimport { useSelectInput } from './SelectInputProvider'\nimport { findOptionInOptions } from './findOptionInOptions'\nimport type { DataType, OptionType } from './types'\nimport { INPUT_SIZE_HEIGHT, SIZES_TAG } from './types'\n\ntype SelectBarProps = {\n  size: 'small' | 'medium' | 'large'\n  clearable: boolean\n  disabled: boolean\n  readOnly: boolean\n  placeholder: string\n  success?: string\n  error?: string | boolean\n  autoFocus?: boolean\n  innerRef: RefObject<HTMLDivElement | null>\n  id?: string\n  'data-testid': string\n  label?: string\n  tooltip?: string\n}\n\ntype DisplayValuesProps = {\n  refTag: RefObject<HTMLDivElement | null>\n  nonOverflowedValues: OptionType[]\n  disabled: boolean\n  readOnly: boolean\n  overflowed: boolean\n  overflowAmount: number\n  size: 'small' | 'medium' | 'large'\n}\n\nconst StateStack = styled(Stack)`\n  padding-right: ${({ theme }) => theme.space['2']};\n  display: flex;\n`\nconst Placeholder = styled(Text)`\nuser-select: none;\n`\n\nexport const StyledInputWrapper = styled(Stack)<{\n  'data-readonly': boolean\n  'data-disabled': boolean\n  'data-size': 'small' | 'medium' | 'large'\n  'data-state': 'neutral' | 'success' | 'danger'\n  'data-dropdownvisible': boolean\n  'aria-label'?: string\n}>`\n  display: flex;\n  padding: ${({ theme }) => theme.space[1]};\n  padding-right: 0;\n  padding-left: ${({ theme }) => theme.space[2]};\n  cursor: pointer;\n  box-shadow: none;\n  background: ${({ theme }) => theme.colors.neutral.background};\n  border-radius: ${({ theme }) => theme.radii.default};\n  width: 100%;\n  overflow: hidden;\n\n  &[data-size='small'] {\n    height: ${({ theme }) => theme.sizing[INPUT_SIZE_HEIGHT.small]};\n    padding-left: ${({ theme }) => theme.space[1]};\n  }\n  &[data-size='medium'] {\n    height: ${({ theme }) => theme.sizing[INPUT_SIZE_HEIGHT.medium]};\n  }\n  &[data-size='large'] {\n    height: ${({ theme }) => theme.sizing[INPUT_SIZE_HEIGHT.large]};\n  }\n  &[data-state='neutral'] {\n    border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n\n    &:not([data-disabled=\"true\"]):not([data-readonly=\"true\"]):active {\n      border-color: ${({ theme }) => theme.colors.primary.borderHover};\n      box-shadow: ${({ theme }) => theme.shadows.focusPrimary};\n    }\n    &:not([data-disabled='true']):hover {\n      border-color: ${({ theme }) => theme.colors.primary.borderHover};\n      outline: none;\n    }\n\n    &:not([data-disabled='true']):focus-visible {\n      outline: 5px auto Highlight;\n      outline: 5px auto -webkit-focus-ring-color;\n    }\n\n    &[data-dropdownvisible='true'] {\n    border-color: ${({ theme }) => theme.colors.primary.borderHover};\n  }\n  }\n\n  &[data-state='success'] {\n    border: 1px solid ${({ theme }) => theme.colors.success.border};\n    &:not([data-disabled=\"true\"]):not([data-readonly=\"true\"]):active {\n      border-color: ${({ theme }) => theme.colors.success.borderHover};\n      box-shadow: ${({ theme }) => theme.shadows.focusSuccess};\n    }\n\n    &[data-dropdownvisible='true'] {\n      border-color: ${({ theme }) => theme.colors.success.borderHover};\n    }\n  }\n\n  &[data-state='danger'] {\n    border: 1px solid ${({ theme }) => theme.colors.danger.border};\n\n    &:not([data-disabled=\"true\"]):not([data-readonly=\"true\"]):active {\n      border-color: ${({ theme }) => theme.colors.danger.borderHover};\n      box-shadow: ${({ theme }) => theme.shadows.focusDanger};\n    }\n\n    &[data-dropdownvisible='true'] {\n      border-color: ${({ theme }) => theme.colors.danger.borderHover};\n    }\n  }\n\n  &:not([data-disabled='true']):not([data-readonly]):hover {\n    border-color: ${({ theme }) => theme.colors.primary.border};\n  }\n\n  &[data-readonly='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n    border-color: ${({ theme }) => theme.colors.neutral.border};\n    cursor: default;\n  }\n\n  &[data-disabled='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    border-color: ${({ theme }) => theme.colors.neutral.borderDisabled};\n    cursor: not-allowed;\n  }\n`\n\nconst CustomTag = styled(Tag)`\n  height: fit-content;\n  width: fit-content;\n`\n\nconst SelectedValues = styled(Text)`\n  text-overflow: ellipsis;\n  overflow: hidden;\n  white-space: nowrap;\n`\n\nconst isValidSelectedValue = (selectedValue: string, options: DataType) =>\n  !Array.isArray(options)\n    ? Object.keys(options).some(group =>\n        options[group].some(\n          option => option.value === selectedValue && !option.disabled,\n        ),\n      )\n    : options.some(option => option.value === selectedValue && !option.disabled)\n\nconst DisplayValues = ({\n  refTag,\n  nonOverflowedValues,\n  disabled,\n  readOnly,\n  overflowed,\n  overflowAmount,\n  size,\n}: DisplayValuesProps) => {\n  const { multiselect, selectedData, setSelectedData, options, onChange } =\n    useSelectInput()\n\n  return multiselect ? (\n    <Stack\n      direction=\"row\"\n      gap=\"1\"\n      wrap=\"nowrap\"\n      ref={refTag}\n      alignItems=\"center\"\n    >\n      {nonOverflowedValues.map(option => (\n        <CustomTag\n          data-testid=\"selected-options-tags\"\n          key={option?.value}\n          sentiment=\"neutral\"\n          disabled={disabled}\n          onClose={\n            !readOnly\n              ? event => {\n                  event.stopPropagation()\n                  setSelectedData({\n                    type: 'selectOption',\n                    clickedOption: option,\n                  })\n                  const newSelectedValues = selectedData.selectedValues?.filter(\n                    val => val !== option.value,\n                  )\n                  onChange?.(newSelectedValues)\n                }\n              : undefined\n          }\n        >\n          {option?.label}\n        </CustomTag>\n      ))}\n      {overflowed ? (\n        <Tag\n          sentiment=\"neutral\"\n          disabled={disabled}\n          key=\"+\"\n          data-testid=\"plus-tag\"\n          aria-label=\"Plus tag\"\n        >\n          <PlusIcon />\n          {overflowAmount}\n        </Tag>\n      ) : null}\n    </Stack>\n  ) : (\n    <SelectedValues\n      as=\"div\"\n      variant={size === 'large' ? 'body' : 'bodySmall'}\n      disabled={disabled}\n      prominence=\"default\"\n      sentiment=\"neutral\"\n    >\n      {selectedData.selectedValues[0]\n        ? findOptionInOptions(options, selectedData.selectedValues[0])?.label\n        : null}\n    </SelectedValues>\n  )\n}\n\nexport const SelectBar = ({\n  size,\n  clearable,\n  disabled,\n  readOnly,\n  placeholder,\n  success,\n  error,\n  autoFocus,\n  tooltip,\n  innerRef,\n  id,\n  'data-testid': dataTestId,\n  label,\n}: SelectBarProps) => {\n  const {\n    isDropdownVisible,\n    onChange,\n    setIsDropdownVisible,\n    options,\n    selectedData,\n    setSelectedData,\n    multiselect,\n  } = useSelectInput()\n  const openable = !(readOnly || disabled)\n  const refTag = useRef<HTMLDivElement>(null)\n  const width = innerRef.current?.offsetWidth\n  const [overflowAmount, setOverflowAmount] = useState(0)\n  const [nonOverflowedValues, setNonOverFlowedValues] = useState<OptionType[]>(\n    () => {\n      if (selectedData.selectedValues[0]) {\n        const firstSelectOption = findOptionInOptions(\n          options,\n          selectedData.selectedValues[0],\n        )\n\n        return firstSelectOption ? [firstSelectOption] : []\n      }\n\n      return []\n    },\n  )\n\n  const state = useMemo(() => {\n    if (error) {\n      return 'danger'\n    }\n    if (success) {\n      return 'success'\n    }\n\n    return 'neutral'\n  }, [error, success])\n\n  useEffect(() => {\n    // When too many items are selected, too avoid overflow, compute the number of tags to display and add a + tag\n    let tagsWidth = 0\n    let computedOverflowAmount = 0\n    let computedNonOverflowedValues: OptionType[] = []\n    const newSelectedValues = selectedData.selectedValues.filter(\n      selectedValue => isValidSelectedValue(selectedValue, options),\n    )\n    for (const selectedValue of newSelectedValues) {\n      const selectedOption = findOptionInOptions(options, selectedValue)\n      if (\n        selectedOption?.label &&\n        width &&\n        isValidSelectedValue(selectedValue, options)\n      ) {\n        const lengthValue = selectedOption.value.length // Find a better way to find the number of displayed characters?\n        const totalTagWidth =\n          SIZES_TAG.tagWidth + SIZES_TAG.letterWidth * lengthValue\n        if (totalTagWidth + tagsWidth > width - 100) {\n          computedOverflowAmount += 1\n        } else {\n          computedNonOverflowedValues = [\n            ...computedNonOverflowedValues,\n            selectedOption,\n          ]\n          tagsWidth += totalTagWidth\n        }\n      }\n    }\n    setNonOverFlowedValues(computedNonOverflowedValues)\n    setOverflowAmount(computedOverflowAmount)\n  }, [options, selectedData.selectedValues, width])\n\n  useEffect(() => {\n    setSelectedData({ type: 'update' })\n  }, [setSelectedData, options])\n\n  const shouldDisplayValues = useMemo(() => {\n    if (multiselect) {\n      return (\n        nonOverflowedValues.length > 0 ||\n        selectedData.selectedValues.some(\n          selectedValue =>\n            findOptionInOptions(options, selectedValue) !== undefined,\n        )\n      )\n    }\n\n    return (\n      selectedData.selectedValues[0] !== undefined &&\n      findOptionInOptions(options, selectedData.selectedValues[0]) !== undefined\n    )\n  }, [\n    multiselect,\n    nonOverflowedValues.length,\n    options,\n    selectedData.selectedValues,\n  ])\n\n  return (\n    <Tooltip text={tooltip}>\n      <StyledInputWrapper\n        role=\"combobox\"\n        id={id}\n        data-disabled={disabled}\n        data-readonly={readOnly}\n        data-size={size}\n        data-dropdownvisible={isDropdownVisible}\n        data-state={state}\n        direction=\"row\"\n        wrap=\"nowrap\"\n        gap=\"1\"\n        justifyContent=\"space-between\"\n        alignItems=\"center\"\n        onClick={\n          openable ? () => setIsDropdownVisible(!isDropdownVisible) : undefined\n        }\n        data-testid={dataTestId}\n        autoFocus={autoFocus}\n        onKeyDown={event => {\n          if (event.key === 'ArrowDown') {\n            if (!isDropdownVisible) {\n              setIsDropdownVisible(true)\n            } else {\n              document.getElementById(`option-0`)?.focus()\n            }\n          }\n          if (event.key === ' ') {\n            event.preventDefault()\n          }\n\n          return ['Enter', ' '].includes(event.key) && openable\n            ? setIsDropdownVisible(!isDropdownVisible)\n            : null\n        }}\n        ref={innerRef}\n        aria-haspopup=\"listbox\"\n        aria-expanded={isDropdownVisible}\n        tabIndex={0}\n        aria-label={label}\n      >\n        {shouldDisplayValues ? (\n          <DisplayValues\n            refTag={refTag}\n            nonOverflowedValues={nonOverflowedValues}\n            disabled={disabled}\n            readOnly={readOnly}\n            overflowed={!!overflowAmount}\n            overflowAmount={overflowAmount}\n            size={size}\n          />\n        ) : (\n          <Placeholder\n            as=\"p\"\n            variant={size === 'large' ? 'body' : 'bodySmall'}\n            sentiment=\"neutral\"\n            disabled={disabled}\n            prominence=\"weak\"\n          >\n            {placeholder}\n          </Placeholder>\n        )}\n        <StateStack direction=\"row\" gap={1} alignItems=\"center\">\n          {error ? <AlertCircleIcon sentiment=\"danger\" /> : null}\n          {success && !error ? <CheckCircleIcon sentiment=\"success\" /> : null}\n          {clearable && selectedData.selectedValues.length > 0 ? (\n            <Button\n              aria-label=\"clear value\"\n              disabled={disabled || !selectedData.selectedValues[0] || readOnly}\n              variant=\"ghost\"\n              size=\"small\"\n              onClick={event => {\n                event.stopPropagation()\n                setSelectedData({ type: 'clearAll' })\n                if (multiselect) {\n                  onChange?.([])\n                } else {\n                  onChange?.('')\n                }\n              }}\n              sentiment=\"neutral\"\n              data-testid=\"clear-all\"\n            >\n              <CloseIcon />\n            </Button>\n          ) : null}\n          <ArrowDownIcon\n            aria-label=\"show dropdown\"\n            size=\"small\"\n            sentiment=\"neutral\"\n            disabled={disabled || readOnly}\n          />\n        </StateStack>\n      </StyledInputWrapper>\n    </Tooltip>\n  )\n}\n"]} */",
|
|
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/SelectInputV2/SelectBar.tsx"],"names":[],"mappings":"AAgEgC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/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 { useSelectInput } from './SelectInputProvider'\nimport { findOptionInOptions } from './findOptionInOptions'\nimport type { OptionType } from './types'\nimport { INPUT_SIZE_HEIGHT } from './types'\n\nconst SIZES_TAG = {\n  paddings: 16,\n  plusTag: 48,\n  gap: 8,\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}\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      direction=\"row\"\n      gap=\"1\"\n      wrap=\"nowrap\"\n      ref={refTag}\n      alignItems=\"center\"\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            onClose={() => {}}\n            className={option.value}\n            key={option.value}\n            hidden\n          >\n            {option?.label}\n          </CustomTag>\n        ))}\n      </div>\n      {nonOverflowedValues.map((option, index) => (\n        <CustomTag\n          data-testid=\"selected-options-tags\"\n          sentiment=\"neutral\"\n          key={option?.value}\n          disabled={disabled}\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                    type: 'selectOption',\n                    clickedOption: option,\n                  })\n                  const newSelectedValues = selectedData.selectedValues?.filter(\n                    val => val !== option.value,\n                  )\n                  onChange?.(newSelectedValues)\n                }\n              : undefined\n          }\n        >\n          {option?.label}\n        </CustomTag>\n      ))}\n\n      {overflowed ? (\n        <Stack ref={refPlusTag} justifyContent=\"center\">\n          <PlusTag\n            sentiment=\"neutral\"\n            disabled={disabled}\n            key=\"+\"\n            data-testid=\"plus-tag\"\n            aria-label=\"Plus tag\"\n          >\n            <PlusIcon size=\"xsmall\" />\n            {overflowAmount}\n          </PlusTag>\n        </Stack>\n      ) : null}\n    </MultiselectStack>\n  ) : (\n    <SelectedValues\n      as=\"div\"\n      variant={size === 'large' ? 'body' : 'bodySmall'}\n      disabled={disabled}\n      prominence=\"default\"\n      sentiment=\"neutral\"\n    >\n      {selectedData.selectedValues[0]\n        ? findOptionInOptions(options, selectedData.selectedValues[0])?.label\n        : null}\n    </SelectedValues>\n  )\n}\n\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}: 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            measuredVisibleTags: [\n              ...accumulator.measuredVisibleTags,\n              canBeVisible && potentiallyNonOverflowedValues[index],\n            ].filter(Boolean) as OptionType[],\n            measuredHiddenTags:\n              accumulator.measuredHiddenTags + (!canBeVisible ? 1 : 0),\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          }\n        },\n        {\n          measuredVisibleTags: [],\n          measuredHiddenTags: 0,\n          accumulatedWidth: 0,\n          lastVisibleElementWidth: 0,\n          lastVisibleLabel: '',\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    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        role=\"combobox\"\n        id={id}\n        data-disabled={disabled}\n        data-readonly={readOnly}\n        data-size={size}\n        data-dropdownvisible={isDropdownVisible}\n        data-state={state}\n        onClick={\n          openable ? () => setIsDropdownVisible(!isDropdownVisible) : undefined\n        }\n        data-testid={dataTestId}\n        autoFocus={autoFocus}\n        onKeyDown={event => {\n          if (event.key === 'ArrowDown') {\n            if (!isDropdownVisible) {\n              setIsDropdownVisible(true)\n            } else {\n              document.getElementById(`option-0`)?.focus()\n            }\n          }\n          if (event.key === ' ') {\n            event.preventDefault()\n          }\n\n          return ['Enter', ' '].includes(event.key) && openable\n            ? setIsDropdownVisible(!isDropdownVisible)\n            : null\n        }}\n        ref={innerRef}\n        aria-haspopup=\"listbox\"\n        aria-expanded={isDropdownVisible}\n        tabIndex={0}\n        aria-label={label}\n      >\n        {shouldDisplayValues ? (\n          <DisplayValues\n            refTag={refTag}\n            nonOverflowedValues={nonOverflowedValues}\n            potentiallyNonOverflowedValues={potentiallyNonOverflowedValues}\n            disabled={disabled}\n            readOnly={readOnly}\n            overflowed={!!overflowAmount}\n            overflowAmount={overflowAmount}\n            size={size}\n            measureRef={measureRef}\n            lastElementMaxWidth={lastElementMaxWidth}\n            overflow={overflow}\n            refPlusTag={refPlusTag}\n          />\n        ) : (\n          <Placeholder\n            as=\"p\"\n            variant={size === 'large' ? 'body' : 'bodySmall'}\n            sentiment=\"neutral\"\n            disabled={disabled}\n            prominence=\"weak\"\n          >\n            {placeholder}\n          </Placeholder>\n        )}\n        <StateStack direction=\"row\" gap={1} alignItems=\"center\" 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              disabled={disabled || !selectedData.selectedValues[0] || readOnly}\n              variant=\"ghost\"\n              size=\"small\"\n              onClick={event => {\n                event.stopPropagation()\n                setSelectedData({ type: 'clearAll' })\n                if (multiselect) {\n                  onChange?.([])\n                } else {\n                  onChange?.('')\n                }\n              }}\n              sentiment=\"neutral\"\n              data-testid=\"clear-all\"\n            >\n              <CloseIcon />\n            </Button>\n          ) : null}\n          <ArrowDownIcon\n            aria-label=\"show dropdown\"\n            size=\"small\"\n            sentiment=\"neutral\"\n            disabled={disabled || readOnly}\n          />\n        </StateStack>\n      </StyledInputWrapper>\n    </Tooltip>\n  )\n}\n\nexport { SelectBar, StyledInputWrapper }\n"]} */",
|
|
36
41
|
toString: _EMOTION_STRINGIFIED_CSS_ERROR__
|
|
37
42
|
});
|
|
38
|
-
const StyledInputWrapper = /* @__PURE__ */ _styled(
|
|
39
|
-
target: "
|
|
43
|
+
const StyledInputWrapper = /* @__PURE__ */ _styled("div", process.env.NODE_ENV === "production" ? {
|
|
44
|
+
target: "ejy0aca4"
|
|
40
45
|
} : {
|
|
41
|
-
target: "
|
|
46
|
+
target: "ejy0aca4",
|
|
42
47
|
label: "StyledInputWrapper"
|
|
43
|
-
})("display:
|
|
48
|
+
})("display:grid;width:100%;gap:", ({
|
|
49
|
+
theme
|
|
50
|
+
}) => theme.space[1], ";grid-template-columns:1fr auto;padding:", ({
|
|
44
51
|
theme
|
|
45
52
|
}) => theme.space[1], ";padding-right:0;padding-left:", ({
|
|
46
53
|
theme
|
|
@@ -48,7 +55,7 @@ const StyledInputWrapper = /* @__PURE__ */ _styled(Stack, process.env.NODE_ENV =
|
|
|
48
55
|
theme
|
|
49
56
|
}) => theme.colors.neutral.background, ";border-radius:", ({
|
|
50
57
|
theme
|
|
51
|
-
}) => theme.radii.default, ";width:100
|
|
58
|
+
}) => theme.radii.default, ";width:100%;&[data-size='small']{height:", ({
|
|
52
59
|
theme
|
|
53
60
|
}) => theme.sizing[INPUT_SIZE_HEIGHT.small], ";padding-left:", ({
|
|
54
61
|
theme
|
|
@@ -92,42 +99,71 @@ const StyledInputWrapper = /* @__PURE__ */ _styled(Stack, process.env.NODE_ENV =
|
|
|
92
99
|
theme
|
|
93
100
|
}) => theme.colors.neutral.backgroundDisabled, ";border-color:", ({
|
|
94
101
|
theme
|
|
95
|
-
}) => theme.colors.neutral.borderDisabled, ";cursor:not-allowed;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/SelectBar.tsx"],"names":[],"mappings":"AA+DE","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/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 { RefObject } from 'react'\nimport { useEffect, useMemo, useRef, useState } from 'react'\nimport { Button } from '../Button'\nimport { Stack } from '../Stack'\nimport { Tag } from '../Tag'\nimport { Text } from '../Text'\nimport { Tooltip } from '../Tooltip'\nimport { useSelectInput } from './SelectInputProvider'\nimport { findOptionInOptions } from './findOptionInOptions'\nimport type { DataType, OptionType } from './types'\nimport { INPUT_SIZE_HEIGHT, SIZES_TAG } from './types'\n\ntype SelectBarProps = {\n  size: 'small' | 'medium' | 'large'\n  clearable: boolean\n  disabled: boolean\n  readOnly: boolean\n  placeholder: string\n  success?: string\n  error?: string | boolean\n  autoFocus?: boolean\n  innerRef: RefObject<HTMLDivElement | null>\n  id?: string\n  'data-testid': string\n  label?: string\n  tooltip?: string\n}\n\ntype DisplayValuesProps = {\n  refTag: RefObject<HTMLDivElement | null>\n  nonOverflowedValues: OptionType[]\n  disabled: boolean\n  readOnly: boolean\n  overflowed: boolean\n  overflowAmount: number\n  size: 'small' | 'medium' | 'large'\n}\n\nconst StateStack = styled(Stack)`\n  padding-right: ${({ theme }) => theme.space['2']};\n  display: flex;\n`\nconst Placeholder = styled(Text)`\nuser-select: none;\n`\n\nexport const StyledInputWrapper = styled(Stack)<{\n  'data-readonly': boolean\n  'data-disabled': boolean\n  'data-size': 'small' | 'medium' | 'large'\n  'data-state': 'neutral' | 'success' | 'danger'\n  'data-dropdownvisible': boolean\n  'aria-label'?: string\n}>`\n  display: flex;\n  padding: ${({ theme }) => theme.space[1]};\n  padding-right: 0;\n  padding-left: ${({ theme }) => theme.space[2]};\n  cursor: pointer;\n  box-shadow: none;\n  background: ${({ theme }) => theme.colors.neutral.background};\n  border-radius: ${({ theme }) => theme.radii.default};\n  width: 100%;\n  overflow: hidden;\n\n  &[data-size='small'] {\n    height: ${({ theme }) => theme.sizing[INPUT_SIZE_HEIGHT.small]};\n    padding-left: ${({ theme }) => theme.space[1]};\n  }\n  &[data-size='medium'] {\n    height: ${({ theme }) => theme.sizing[INPUT_SIZE_HEIGHT.medium]};\n  }\n  &[data-size='large'] {\n    height: ${({ theme }) => theme.sizing[INPUT_SIZE_HEIGHT.large]};\n  }\n  &[data-state='neutral'] {\n    border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n\n    &:not([data-disabled=\"true\"]):not([data-readonly=\"true\"]):active {\n      border-color: ${({ theme }) => theme.colors.primary.borderHover};\n      box-shadow: ${({ theme }) => theme.shadows.focusPrimary};\n    }\n    &:not([data-disabled='true']):hover {\n      border-color: ${({ theme }) => theme.colors.primary.borderHover};\n      outline: none;\n    }\n\n    &:not([data-disabled='true']):focus-visible {\n      outline: 5px auto Highlight;\n      outline: 5px auto -webkit-focus-ring-color;\n    }\n\n    &[data-dropdownvisible='true'] {\n    border-color: ${({ theme }) => theme.colors.primary.borderHover};\n  }\n  }\n\n  &[data-state='success'] {\n    border: 1px solid ${({ theme }) => theme.colors.success.border};\n    &:not([data-disabled=\"true\"]):not([data-readonly=\"true\"]):active {\n      border-color: ${({ theme }) => theme.colors.success.borderHover};\n      box-shadow: ${({ theme }) => theme.shadows.focusSuccess};\n    }\n\n    &[data-dropdownvisible='true'] {\n      border-color: ${({ theme }) => theme.colors.success.borderHover};\n    }\n  }\n\n  &[data-state='danger'] {\n    border: 1px solid ${({ theme }) => theme.colors.danger.border};\n\n    &:not([data-disabled=\"true\"]):not([data-readonly=\"true\"]):active {\n      border-color: ${({ theme }) => theme.colors.danger.borderHover};\n      box-shadow: ${({ theme }) => theme.shadows.focusDanger};\n    }\n\n    &[data-dropdownvisible='true'] {\n      border-color: ${({ theme }) => theme.colors.danger.borderHover};\n    }\n  }\n\n  &:not([data-disabled='true']):not([data-readonly]):hover {\n    border-color: ${({ theme }) => theme.colors.primary.border};\n  }\n\n  &[data-readonly='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n    border-color: ${({ theme }) => theme.colors.neutral.border};\n    cursor: default;\n  }\n\n  &[data-disabled='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    border-color: ${({ theme }) => theme.colors.neutral.borderDisabled};\n    cursor: not-allowed;\n  }\n`\n\nconst CustomTag = styled(Tag)`\n  height: fit-content;\n  width: fit-content;\n`\n\nconst SelectedValues = styled(Text)`\n  text-overflow: ellipsis;\n  overflow: hidden;\n  white-space: nowrap;\n`\n\nconst isValidSelectedValue = (selectedValue: string, options: DataType) =>\n  !Array.isArray(options)\n    ? Object.keys(options).some(group =>\n        options[group].some(\n          option => option.value === selectedValue && !option.disabled,\n        ),\n      )\n    : options.some(option => option.value === selectedValue && !option.disabled)\n\nconst DisplayValues = ({\n  refTag,\n  nonOverflowedValues,\n  disabled,\n  readOnly,\n  overflowed,\n  overflowAmount,\n  size,\n}: DisplayValuesProps) => {\n  const { multiselect, selectedData, setSelectedData, options, onChange } =\n    useSelectInput()\n\n  return multiselect ? (\n    <Stack\n      direction=\"row\"\n      gap=\"1\"\n      wrap=\"nowrap\"\n      ref={refTag}\n      alignItems=\"center\"\n    >\n      {nonOverflowedValues.map(option => (\n        <CustomTag\n          data-testid=\"selected-options-tags\"\n          key={option?.value}\n          sentiment=\"neutral\"\n          disabled={disabled}\n          onClose={\n            !readOnly\n              ? event => {\n                  event.stopPropagation()\n                  setSelectedData({\n                    type: 'selectOption',\n                    clickedOption: option,\n                  })\n                  const newSelectedValues = selectedData.selectedValues?.filter(\n                    val => val !== option.value,\n                  )\n                  onChange?.(newSelectedValues)\n                }\n              : undefined\n          }\n        >\n          {option?.label}\n        </CustomTag>\n      ))}\n      {overflowed ? (\n        <Tag\n          sentiment=\"neutral\"\n          disabled={disabled}\n          key=\"+\"\n          data-testid=\"plus-tag\"\n          aria-label=\"Plus tag\"\n        >\n          <PlusIcon />\n          {overflowAmount}\n        </Tag>\n      ) : null}\n    </Stack>\n  ) : (\n    <SelectedValues\n      as=\"div\"\n      variant={size === 'large' ? 'body' : 'bodySmall'}\n      disabled={disabled}\n      prominence=\"default\"\n      sentiment=\"neutral\"\n    >\n      {selectedData.selectedValues[0]\n        ? findOptionInOptions(options, selectedData.selectedValues[0])?.label\n        : null}\n    </SelectedValues>\n  )\n}\n\nexport const SelectBar = ({\n  size,\n  clearable,\n  disabled,\n  readOnly,\n  placeholder,\n  success,\n  error,\n  autoFocus,\n  tooltip,\n  innerRef,\n  id,\n  'data-testid': dataTestId,\n  label,\n}: SelectBarProps) => {\n  const {\n    isDropdownVisible,\n    onChange,\n    setIsDropdownVisible,\n    options,\n    selectedData,\n    setSelectedData,\n    multiselect,\n  } = useSelectInput()\n  const openable = !(readOnly || disabled)\n  const refTag = useRef<HTMLDivElement>(null)\n  const width = innerRef.current?.offsetWidth\n  const [overflowAmount, setOverflowAmount] = useState(0)\n  const [nonOverflowedValues, setNonOverFlowedValues] = useState<OptionType[]>(\n    () => {\n      if (selectedData.selectedValues[0]) {\n        const firstSelectOption = findOptionInOptions(\n          options,\n          selectedData.selectedValues[0],\n        )\n\n        return firstSelectOption ? [firstSelectOption] : []\n      }\n\n      return []\n    },\n  )\n\n  const state = useMemo(() => {\n    if (error) {\n      return 'danger'\n    }\n    if (success) {\n      return 'success'\n    }\n\n    return 'neutral'\n  }, [error, success])\n\n  useEffect(() => {\n    // When too many items are selected, too avoid overflow, compute the number of tags to display and add a + tag\n    let tagsWidth = 0\n    let computedOverflowAmount = 0\n    let computedNonOverflowedValues: OptionType[] = []\n    const newSelectedValues = selectedData.selectedValues.filter(\n      selectedValue => isValidSelectedValue(selectedValue, options),\n    )\n    for (const selectedValue of newSelectedValues) {\n      const selectedOption = findOptionInOptions(options, selectedValue)\n      if (\n        selectedOption?.label &&\n        width &&\n        isValidSelectedValue(selectedValue, options)\n      ) {\n        const lengthValue = selectedOption.value.length // Find a better way to find the number of displayed characters?\n        const totalTagWidth =\n          SIZES_TAG.tagWidth + SIZES_TAG.letterWidth * lengthValue\n        if (totalTagWidth + tagsWidth > width - 100) {\n          computedOverflowAmount += 1\n        } else {\n          computedNonOverflowedValues = [\n            ...computedNonOverflowedValues,\n            selectedOption,\n          ]\n          tagsWidth += totalTagWidth\n        }\n      }\n    }\n    setNonOverFlowedValues(computedNonOverflowedValues)\n    setOverflowAmount(computedOverflowAmount)\n  }, [options, selectedData.selectedValues, width])\n\n  useEffect(() => {\n    setSelectedData({ type: 'update' })\n  }, [setSelectedData, options])\n\n  const shouldDisplayValues = useMemo(() => {\n    if (multiselect) {\n      return (\n        nonOverflowedValues.length > 0 ||\n        selectedData.selectedValues.some(\n          selectedValue =>\n            findOptionInOptions(options, selectedValue) !== undefined,\n        )\n      )\n    }\n\n    return (\n      selectedData.selectedValues[0] !== undefined &&\n      findOptionInOptions(options, selectedData.selectedValues[0]) !== undefined\n    )\n  }, [\n    multiselect,\n    nonOverflowedValues.length,\n    options,\n    selectedData.selectedValues,\n  ])\n\n  return (\n    <Tooltip text={tooltip}>\n      <StyledInputWrapper\n        role=\"combobox\"\n        id={id}\n        data-disabled={disabled}\n        data-readonly={readOnly}\n        data-size={size}\n        data-dropdownvisible={isDropdownVisible}\n        data-state={state}\n        direction=\"row\"\n        wrap=\"nowrap\"\n        gap=\"1\"\n        justifyContent=\"space-between\"\n        alignItems=\"center\"\n        onClick={\n          openable ? () => setIsDropdownVisible(!isDropdownVisible) : undefined\n        }\n        data-testid={dataTestId}\n        autoFocus={autoFocus}\n        onKeyDown={event => {\n          if (event.key === 'ArrowDown') {\n            if (!isDropdownVisible) {\n              setIsDropdownVisible(true)\n            } else {\n              document.getElementById(`option-0`)?.focus()\n            }\n          }\n          if (event.key === ' ') {\n            event.preventDefault()\n          }\n\n          return ['Enter', ' '].includes(event.key) && openable\n            ? setIsDropdownVisible(!isDropdownVisible)\n            : null\n        }}\n        ref={innerRef}\n        aria-haspopup=\"listbox\"\n        aria-expanded={isDropdownVisible}\n        tabIndex={0}\n        aria-label={label}\n      >\n        {shouldDisplayValues ? (\n          <DisplayValues\n            refTag={refTag}\n            nonOverflowedValues={nonOverflowedValues}\n            disabled={disabled}\n            readOnly={readOnly}\n            overflowed={!!overflowAmount}\n            overflowAmount={overflowAmount}\n            size={size}\n          />\n        ) : (\n          <Placeholder\n            as=\"p\"\n            variant={size === 'large' ? 'body' : 'bodySmall'}\n            sentiment=\"neutral\"\n            disabled={disabled}\n            prominence=\"weak\"\n          >\n            {placeholder}\n          </Placeholder>\n        )}\n        <StateStack direction=\"row\" gap={1} alignItems=\"center\">\n          {error ? <AlertCircleIcon sentiment=\"danger\" /> : null}\n          {success && !error ? <CheckCircleIcon sentiment=\"success\" /> : null}\n          {clearable && selectedData.selectedValues.length > 0 ? (\n            <Button\n              aria-label=\"clear value\"\n              disabled={disabled || !selectedData.selectedValues[0] || readOnly}\n              variant=\"ghost\"\n              size=\"small\"\n              onClick={event => {\n                event.stopPropagation()\n                setSelectedData({ type: 'clearAll' })\n                if (multiselect) {\n                  onChange?.([])\n                } else {\n                  onChange?.('')\n                }\n              }}\n              sentiment=\"neutral\"\n              data-testid=\"clear-all\"\n            >\n              <CloseIcon />\n            </Button>\n          ) : null}\n          <ArrowDownIcon\n            aria-label=\"show dropdown\"\n            size=\"small\"\n            sentiment=\"neutral\"\n            disabled={disabled || readOnly}\n          />\n        </StateStack>\n      </StyledInputWrapper>\n    </Tooltip>\n  )\n}\n"]} */"));
|
|
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/SelectInputV2/SelectBar.tsx"],"names":[],"mappings":"AA4EE","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/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 { useSelectInput } from './SelectInputProvider'\nimport { findOptionInOptions } from './findOptionInOptions'\nimport type { OptionType } from './types'\nimport { INPUT_SIZE_HEIGHT } from './types'\n\nconst SIZES_TAG = {\n  paddings: 16,\n  plusTag: 48,\n  gap: 8,\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}\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      direction=\"row\"\n      gap=\"1\"\n      wrap=\"nowrap\"\n      ref={refTag}\n      alignItems=\"center\"\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            onClose={() => {}}\n            className={option.value}\n            key={option.value}\n            hidden\n          >\n            {option?.label}\n          </CustomTag>\n        ))}\n      </div>\n      {nonOverflowedValues.map((option, index) => (\n        <CustomTag\n          data-testid=\"selected-options-tags\"\n          sentiment=\"neutral\"\n          key={option?.value}\n          disabled={disabled}\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                    type: 'selectOption',\n                    clickedOption: option,\n                  })\n                  const newSelectedValues = selectedData.selectedValues?.filter(\n                    val => val !== option.value,\n                  )\n                  onChange?.(newSelectedValues)\n                }\n              : undefined\n          }\n        >\n          {option?.label}\n        </CustomTag>\n      ))}\n\n      {overflowed ? (\n        <Stack ref={refPlusTag} justifyContent=\"center\">\n          <PlusTag\n            sentiment=\"neutral\"\n            disabled={disabled}\n            key=\"+\"\n            data-testid=\"plus-tag\"\n            aria-label=\"Plus tag\"\n          >\n            <PlusIcon size=\"xsmall\" />\n            {overflowAmount}\n          </PlusTag>\n        </Stack>\n      ) : null}\n    </MultiselectStack>\n  ) : (\n    <SelectedValues\n      as=\"div\"\n      variant={size === 'large' ? 'body' : 'bodySmall'}\n      disabled={disabled}\n      prominence=\"default\"\n      sentiment=\"neutral\"\n    >\n      {selectedData.selectedValues[0]\n        ? findOptionInOptions(options, selectedData.selectedValues[0])?.label\n        : null}\n    </SelectedValues>\n  )\n}\n\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}: 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            measuredVisibleTags: [\n              ...accumulator.measuredVisibleTags,\n              canBeVisible && potentiallyNonOverflowedValues[index],\n            ].filter(Boolean) as OptionType[],\n            measuredHiddenTags:\n              accumulator.measuredHiddenTags + (!canBeVisible ? 1 : 0),\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          }\n        },\n        {\n          measuredVisibleTags: [],\n          measuredHiddenTags: 0,\n          accumulatedWidth: 0,\n          lastVisibleElementWidth: 0,\n          lastVisibleLabel: '',\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    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        role=\"combobox\"\n        id={id}\n        data-disabled={disabled}\n        data-readonly={readOnly}\n        data-size={size}\n        data-dropdownvisible={isDropdownVisible}\n        data-state={state}\n        onClick={\n          openable ? () => setIsDropdownVisible(!isDropdownVisible) : undefined\n        }\n        data-testid={dataTestId}\n        autoFocus={autoFocus}\n        onKeyDown={event => {\n          if (event.key === 'ArrowDown') {\n            if (!isDropdownVisible) {\n              setIsDropdownVisible(true)\n            } else {\n              document.getElementById(`option-0`)?.focus()\n            }\n          }\n          if (event.key === ' ') {\n            event.preventDefault()\n          }\n\n          return ['Enter', ' '].includes(event.key) && openable\n            ? setIsDropdownVisible(!isDropdownVisible)\n            : null\n        }}\n        ref={innerRef}\n        aria-haspopup=\"listbox\"\n        aria-expanded={isDropdownVisible}\n        tabIndex={0}\n        aria-label={label}\n      >\n        {shouldDisplayValues ? (\n          <DisplayValues\n            refTag={refTag}\n            nonOverflowedValues={nonOverflowedValues}\n            potentiallyNonOverflowedValues={potentiallyNonOverflowedValues}\n            disabled={disabled}\n            readOnly={readOnly}\n            overflowed={!!overflowAmount}\n            overflowAmount={overflowAmount}\n            size={size}\n            measureRef={measureRef}\n            lastElementMaxWidth={lastElementMaxWidth}\n            overflow={overflow}\n            refPlusTag={refPlusTag}\n          />\n        ) : (\n          <Placeholder\n            as=\"p\"\n            variant={size === 'large' ? 'body' : 'bodySmall'}\n            sentiment=\"neutral\"\n            disabled={disabled}\n            prominence=\"weak\"\n          >\n            {placeholder}\n          </Placeholder>\n        )}\n        <StateStack direction=\"row\" gap={1} alignItems=\"center\" 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              disabled={disabled || !selectedData.selectedValues[0] || readOnly}\n              variant=\"ghost\"\n              size=\"small\"\n              onClick={event => {\n                event.stopPropagation()\n                setSelectedData({ type: 'clearAll' })\n                if (multiselect) {\n                  onChange?.([])\n                } else {\n                  onChange?.('')\n                }\n              }}\n              sentiment=\"neutral\"\n              data-testid=\"clear-all\"\n            >\n              <CloseIcon />\n            </Button>\n          ) : null}\n          <ArrowDownIcon\n            aria-label=\"show dropdown\"\n            size=\"small\"\n            sentiment=\"neutral\"\n            disabled={disabled || readOnly}\n          />\n        </StateStack>\n      </StyledInputWrapper>\n    </Tooltip>\n  )\n}\n\nexport { SelectBar, StyledInputWrapper }\n"]} */"));
|
|
96
103
|
const CustomTag = /* @__PURE__ */ _styled(Tag, process.env.NODE_ENV === "production" ? {
|
|
97
|
-
|
|
104
|
+
shouldForwardProp: (prop) => !["lastElementMaxWidth", "hidden"].includes(prop),
|
|
105
|
+
target: "ejy0aca3"
|
|
98
106
|
} : {
|
|
99
|
-
|
|
107
|
+
shouldForwardProp: (prop) => !["lastElementMaxWidth", "hidden"].includes(prop),
|
|
108
|
+
target: "ejy0aca3",
|
|
100
109
|
label: "CustomTag"
|
|
110
|
+
})("height:max-content;width:fit-content;min-width:", ({
|
|
111
|
+
lastElementMaxWidth
|
|
112
|
+
}) => lastElementMaxWidth ? "auto" : "fit-content", ";max-width:", ({
|
|
113
|
+
lastElementMaxWidth,
|
|
114
|
+
hidden
|
|
115
|
+
}) => lastElementMaxWidth && !hidden ? `${lastElementMaxWidth}px` : "100%", ";", ({
|
|
116
|
+
hidden
|
|
117
|
+
}) => hidden ? "visibility: hidden;" : `
|
|
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/SelectInputV2/SelectBar.tsx"],"names":[],"mappings":"AAyKE","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/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 { useSelectInput } from './SelectInputProvider'\nimport { findOptionInOptions } from './findOptionInOptions'\nimport type { OptionType } from './types'\nimport { INPUT_SIZE_HEIGHT } from './types'\n\nconst SIZES_TAG = {\n  paddings: 16,\n  plusTag: 48,\n  gap: 8,\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}\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      direction=\"row\"\n      gap=\"1\"\n      wrap=\"nowrap\"\n      ref={refTag}\n      alignItems=\"center\"\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            onClose={() => {}}\n            className={option.value}\n            key={option.value}\n            hidden\n          >\n            {option?.label}\n          </CustomTag>\n        ))}\n      </div>\n      {nonOverflowedValues.map((option, index) => (\n        <CustomTag\n          data-testid=\"selected-options-tags\"\n          sentiment=\"neutral\"\n          key={option?.value}\n          disabled={disabled}\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                    type: 'selectOption',\n                    clickedOption: option,\n                  })\n                  const newSelectedValues = selectedData.selectedValues?.filter(\n                    val => val !== option.value,\n                  )\n                  onChange?.(newSelectedValues)\n                }\n              : undefined\n          }\n        >\n          {option?.label}\n        </CustomTag>\n      ))}\n\n      {overflowed ? (\n        <Stack ref={refPlusTag} justifyContent=\"center\">\n          <PlusTag\n            sentiment=\"neutral\"\n            disabled={disabled}\n            key=\"+\"\n            data-testid=\"plus-tag\"\n            aria-label=\"Plus tag\"\n          >\n            <PlusIcon size=\"xsmall\" />\n            {overflowAmount}\n          </PlusTag>\n        </Stack>\n      ) : null}\n    </MultiselectStack>\n  ) : (\n    <SelectedValues\n      as=\"div\"\n      variant={size === 'large' ? 'body' : 'bodySmall'}\n      disabled={disabled}\n      prominence=\"default\"\n      sentiment=\"neutral\"\n    >\n      {selectedData.selectedValues[0]\n        ? findOptionInOptions(options, selectedData.selectedValues[0])?.label\n        : null}\n    </SelectedValues>\n  )\n}\n\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}: 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            measuredVisibleTags: [\n              ...accumulator.measuredVisibleTags,\n              canBeVisible && potentiallyNonOverflowedValues[index],\n            ].filter(Boolean) as OptionType[],\n            measuredHiddenTags:\n              accumulator.measuredHiddenTags + (!canBeVisible ? 1 : 0),\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          }\n        },\n        {\n          measuredVisibleTags: [],\n          measuredHiddenTags: 0,\n          accumulatedWidth: 0,\n          lastVisibleElementWidth: 0,\n          lastVisibleLabel: '',\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    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        role=\"combobox\"\n        id={id}\n        data-disabled={disabled}\n        data-readonly={readOnly}\n        data-size={size}\n        data-dropdownvisible={isDropdownVisible}\n        data-state={state}\n        onClick={\n          openable ? () => setIsDropdownVisible(!isDropdownVisible) : undefined\n        }\n        data-testid={dataTestId}\n        autoFocus={autoFocus}\n        onKeyDown={event => {\n          if (event.key === 'ArrowDown') {\n            if (!isDropdownVisible) {\n              setIsDropdownVisible(true)\n            } else {\n              document.getElementById(`option-0`)?.focus()\n            }\n          }\n          if (event.key === ' ') {\n            event.preventDefault()\n          }\n\n          return ['Enter', ' '].includes(event.key) && openable\n            ? setIsDropdownVisible(!isDropdownVisible)\n            : null\n        }}\n        ref={innerRef}\n        aria-haspopup=\"listbox\"\n        aria-expanded={isDropdownVisible}\n        tabIndex={0}\n        aria-label={label}\n      >\n        {shouldDisplayValues ? (\n          <DisplayValues\n            refTag={refTag}\n            nonOverflowedValues={nonOverflowedValues}\n            potentiallyNonOverflowedValues={potentiallyNonOverflowedValues}\n            disabled={disabled}\n            readOnly={readOnly}\n            overflowed={!!overflowAmount}\n            overflowAmount={overflowAmount}\n            size={size}\n            measureRef={measureRef}\n            lastElementMaxWidth={lastElementMaxWidth}\n            overflow={overflow}\n            refPlusTag={refPlusTag}\n          />\n        ) : (\n          <Placeholder\n            as=\"p\"\n            variant={size === 'large' ? 'body' : 'bodySmall'}\n            sentiment=\"neutral\"\n            disabled={disabled}\n            prominence=\"weak\"\n          >\n            {placeholder}\n          </Placeholder>\n        )}\n        <StateStack direction=\"row\" gap={1} alignItems=\"center\" 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              disabled={disabled || !selectedData.selectedValues[0] || readOnly}\n              variant=\"ghost\"\n              size=\"small\"\n              onClick={event => {\n                event.stopPropagation()\n                setSelectedData({ type: 'clearAll' })\n                if (multiselect) {\n                  onChange?.([])\n                } else {\n                  onChange?.('')\n                }\n              }}\n              sentiment=\"neutral\"\n              data-testid=\"clear-all\"\n            >\n              <CloseIcon />\n            </Button>\n          ) : null}\n          <ArrowDownIcon\n            aria-label=\"show dropdown\"\n            size=\"small\"\n            sentiment=\"neutral\"\n            disabled={disabled || readOnly}\n          />\n        </StateStack>\n      </StyledInputWrapper>\n    </Tooltip>\n  )\n}\n\nexport { SelectBar, StyledInputWrapper }\n"]} */"));
|
|
120
|
+
const SelectedValues = /* @__PURE__ */ _styled(Text, process.env.NODE_ENV === "production" ? {
|
|
121
|
+
target: "ejy0aca2"
|
|
122
|
+
} : {
|
|
123
|
+
target: "ejy0aca2",
|
|
124
|
+
label: "SelectedValues"
|
|
101
125
|
})(process.env.NODE_ENV === "production" ? {
|
|
102
|
-
name: "
|
|
103
|
-
styles: "
|
|
126
|
+
name: "et8r3r",
|
|
127
|
+
styles: "text-overflow:ellipsis;overflow:hidden;white-space:nowrap;align-self:center"
|
|
104
128
|
} : {
|
|
105
|
-
name: "
|
|
106
|
-
styles: "height:fit-content;width:fit-content/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/SelectBar.tsx"],"names":[],"mappings":"AAqJ6B","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/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 { RefObject } from 'react'\nimport { useEffect, useMemo, useRef, useState } from 'react'\nimport { Button } from '../Button'\nimport { Stack } from '../Stack'\nimport { Tag } from '../Tag'\nimport { Text } from '../Text'\nimport { Tooltip } from '../Tooltip'\nimport { useSelectInput } from './SelectInputProvider'\nimport { findOptionInOptions } from './findOptionInOptions'\nimport type { DataType, OptionType } from './types'\nimport { INPUT_SIZE_HEIGHT, SIZES_TAG } from './types'\n\ntype SelectBarProps = {\n  size: 'small' | 'medium' | 'large'\n  clearable: boolean\n  disabled: boolean\n  readOnly: boolean\n  placeholder: string\n  success?: string\n  error?: string | boolean\n  autoFocus?: boolean\n  innerRef: RefObject<HTMLDivElement | null>\n  id?: string\n  'data-testid': string\n  label?: string\n  tooltip?: string\n}\n\ntype DisplayValuesProps = {\n  refTag: RefObject<HTMLDivElement | null>\n  nonOverflowedValues: OptionType[]\n  disabled: boolean\n  readOnly: boolean\n  overflowed: boolean\n  overflowAmount: number\n  size: 'small' | 'medium' | 'large'\n}\n\nconst StateStack = styled(Stack)`\n  padding-right: ${({ theme }) => theme.space['2']};\n  display: flex;\n`\nconst Placeholder = styled(Text)`\nuser-select: none;\n`\n\nexport const StyledInputWrapper = styled(Stack)<{\n  'data-readonly': boolean\n  'data-disabled': boolean\n  'data-size': 'small' | 'medium' | 'large'\n  'data-state': 'neutral' | 'success' | 'danger'\n  'data-dropdownvisible': boolean\n  'aria-label'?: string\n}>`\n  display: flex;\n  padding: ${({ theme }) => theme.space[1]};\n  padding-right: 0;\n  padding-left: ${({ theme }) => theme.space[2]};\n  cursor: pointer;\n  box-shadow: none;\n  background: ${({ theme }) => theme.colors.neutral.background};\n  border-radius: ${({ theme }) => theme.radii.default};\n  width: 100%;\n  overflow: hidden;\n\n  &[data-size='small'] {\n    height: ${({ theme }) => theme.sizing[INPUT_SIZE_HEIGHT.small]};\n    padding-left: ${({ theme }) => theme.space[1]};\n  }\n  &[data-size='medium'] {\n    height: ${({ theme }) => theme.sizing[INPUT_SIZE_HEIGHT.medium]};\n  }\n  &[data-size='large'] {\n    height: ${({ theme }) => theme.sizing[INPUT_SIZE_HEIGHT.large]};\n  }\n  &[data-state='neutral'] {\n    border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n\n    &:not([data-disabled=\"true\"]):not([data-readonly=\"true\"]):active {\n      border-color: ${({ theme }) => theme.colors.primary.borderHover};\n      box-shadow: ${({ theme }) => theme.shadows.focusPrimary};\n    }\n    &:not([data-disabled='true']):hover {\n      border-color: ${({ theme }) => theme.colors.primary.borderHover};\n      outline: none;\n    }\n\n    &:not([data-disabled='true']):focus-visible {\n      outline: 5px auto Highlight;\n      outline: 5px auto -webkit-focus-ring-color;\n    }\n\n    &[data-dropdownvisible='true'] {\n    border-color: ${({ theme }) => theme.colors.primary.borderHover};\n  }\n  }\n\n  &[data-state='success'] {\n    border: 1px solid ${({ theme }) => theme.colors.success.border};\n    &:not([data-disabled=\"true\"]):not([data-readonly=\"true\"]):active {\n      border-color: ${({ theme }) => theme.colors.success.borderHover};\n      box-shadow: ${({ theme }) => theme.shadows.focusSuccess};\n    }\n\n    &[data-dropdownvisible='true'] {\n      border-color: ${({ theme }) => theme.colors.success.borderHover};\n    }\n  }\n\n  &[data-state='danger'] {\n    border: 1px solid ${({ theme }) => theme.colors.danger.border};\n\n    &:not([data-disabled=\"true\"]):not([data-readonly=\"true\"]):active {\n      border-color: ${({ theme }) => theme.colors.danger.borderHover};\n      box-shadow: ${({ theme }) => theme.shadows.focusDanger};\n    }\n\n    &[data-dropdownvisible='true'] {\n      border-color: ${({ theme }) => theme.colors.danger.borderHover};\n    }\n  }\n\n  &:not([data-disabled='true']):not([data-readonly]):hover {\n    border-color: ${({ theme }) => theme.colors.primary.border};\n  }\n\n  &[data-readonly='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n    border-color: ${({ theme }) => theme.colors.neutral.border};\n    cursor: default;\n  }\n\n  &[data-disabled='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    border-color: ${({ theme }) => theme.colors.neutral.borderDisabled};\n    cursor: not-allowed;\n  }\n`\n\nconst CustomTag = styled(Tag)`\n  height: fit-content;\n  width: fit-content;\n`\n\nconst SelectedValues = styled(Text)`\n  text-overflow: ellipsis;\n  overflow: hidden;\n  white-space: nowrap;\n`\n\nconst isValidSelectedValue = (selectedValue: string, options: DataType) =>\n  !Array.isArray(options)\n    ? Object.keys(options).some(group =>\n        options[group].some(\n          option => option.value === selectedValue && !option.disabled,\n        ),\n      )\n    : options.some(option => option.value === selectedValue && !option.disabled)\n\nconst DisplayValues = ({\n  refTag,\n  nonOverflowedValues,\n  disabled,\n  readOnly,\n  overflowed,\n  overflowAmount,\n  size,\n}: DisplayValuesProps) => {\n  const { multiselect, selectedData, setSelectedData, options, onChange } =\n    useSelectInput()\n\n  return multiselect ? (\n    <Stack\n      direction=\"row\"\n      gap=\"1\"\n      wrap=\"nowrap\"\n      ref={refTag}\n      alignItems=\"center\"\n    >\n      {nonOverflowedValues.map(option => (\n        <CustomTag\n          data-testid=\"selected-options-tags\"\n          key={option?.value}\n          sentiment=\"neutral\"\n          disabled={disabled}\n          onClose={\n            !readOnly\n              ? event => {\n                  event.stopPropagation()\n                  setSelectedData({\n                    type: 'selectOption',\n                    clickedOption: option,\n                  })\n                  const newSelectedValues = selectedData.selectedValues?.filter(\n                    val => val !== option.value,\n                  )\n                  onChange?.(newSelectedValues)\n                }\n              : undefined\n          }\n        >\n          {option?.label}\n        </CustomTag>\n      ))}\n      {overflowed ? (\n        <Tag\n          sentiment=\"neutral\"\n          disabled={disabled}\n          key=\"+\"\n          data-testid=\"plus-tag\"\n          aria-label=\"Plus tag\"\n        >\n          <PlusIcon />\n          {overflowAmount}\n        </Tag>\n      ) : null}\n    </Stack>\n  ) : (\n    <SelectedValues\n      as=\"div\"\n      variant={size === 'large' ? 'body' : 'bodySmall'}\n      disabled={disabled}\n      prominence=\"default\"\n      sentiment=\"neutral\"\n    >\n      {selectedData.selectedValues[0]\n        ? findOptionInOptions(options, selectedData.selectedValues[0])?.label\n        : null}\n    </SelectedValues>\n  )\n}\n\nexport const SelectBar = ({\n  size,\n  clearable,\n  disabled,\n  readOnly,\n  placeholder,\n  success,\n  error,\n  autoFocus,\n  tooltip,\n  innerRef,\n  id,\n  'data-testid': dataTestId,\n  label,\n}: SelectBarProps) => {\n  const {\n    isDropdownVisible,\n    onChange,\n    setIsDropdownVisible,\n    options,\n    selectedData,\n    setSelectedData,\n    multiselect,\n  } = useSelectInput()\n  const openable = !(readOnly || disabled)\n  const refTag = useRef<HTMLDivElement>(null)\n  const width = innerRef.current?.offsetWidth\n  const [overflowAmount, setOverflowAmount] = useState(0)\n  const [nonOverflowedValues, setNonOverFlowedValues] = useState<OptionType[]>(\n    () => {\n      if (selectedData.selectedValues[0]) {\n        const firstSelectOption = findOptionInOptions(\n          options,\n          selectedData.selectedValues[0],\n        )\n\n        return firstSelectOption ? [firstSelectOption] : []\n      }\n\n      return []\n    },\n  )\n\n  const state = useMemo(() => {\n    if (error) {\n      return 'danger'\n    }\n    if (success) {\n      return 'success'\n    }\n\n    return 'neutral'\n  }, [error, success])\n\n  useEffect(() => {\n    // When too many items are selected, too avoid overflow, compute the number of tags to display and add a + tag\n    let tagsWidth = 0\n    let computedOverflowAmount = 0\n    let computedNonOverflowedValues: OptionType[] = []\n    const newSelectedValues = selectedData.selectedValues.filter(\n      selectedValue => isValidSelectedValue(selectedValue, options),\n    )\n    for (const selectedValue of newSelectedValues) {\n      const selectedOption = findOptionInOptions(options, selectedValue)\n      if (\n        selectedOption?.label &&\n        width &&\n        isValidSelectedValue(selectedValue, options)\n      ) {\n        const lengthValue = selectedOption.value.length // Find a better way to find the number of displayed characters?\n        const totalTagWidth =\n          SIZES_TAG.tagWidth + SIZES_TAG.letterWidth * lengthValue\n        if (totalTagWidth + tagsWidth > width - 100) {\n          computedOverflowAmount += 1\n        } else {\n          computedNonOverflowedValues = [\n            ...computedNonOverflowedValues,\n            selectedOption,\n          ]\n          tagsWidth += totalTagWidth\n        }\n      }\n    }\n    setNonOverFlowedValues(computedNonOverflowedValues)\n    setOverflowAmount(computedOverflowAmount)\n  }, [options, selectedData.selectedValues, width])\n\n  useEffect(() => {\n    setSelectedData({ type: 'update' })\n  }, [setSelectedData, options])\n\n  const shouldDisplayValues = useMemo(() => {\n    if (multiselect) {\n      return (\n        nonOverflowedValues.length > 0 ||\n        selectedData.selectedValues.some(\n          selectedValue =>\n            findOptionInOptions(options, selectedValue) !== undefined,\n        )\n      )\n    }\n\n    return (\n      selectedData.selectedValues[0] !== undefined &&\n      findOptionInOptions(options, selectedData.selectedValues[0]) !== undefined\n    )\n  }, [\n    multiselect,\n    nonOverflowedValues.length,\n    options,\n    selectedData.selectedValues,\n  ])\n\n  return (\n    <Tooltip text={tooltip}>\n      <StyledInputWrapper\n        role=\"combobox\"\n        id={id}\n        data-disabled={disabled}\n        data-readonly={readOnly}\n        data-size={size}\n        data-dropdownvisible={isDropdownVisible}\n        data-state={state}\n        direction=\"row\"\n        wrap=\"nowrap\"\n        gap=\"1\"\n        justifyContent=\"space-between\"\n        alignItems=\"center\"\n        onClick={\n          openable ? () => setIsDropdownVisible(!isDropdownVisible) : undefined\n        }\n        data-testid={dataTestId}\n        autoFocus={autoFocus}\n        onKeyDown={event => {\n          if (event.key === 'ArrowDown') {\n            if (!isDropdownVisible) {\n              setIsDropdownVisible(true)\n            } else {\n              document.getElementById(`option-0`)?.focus()\n            }\n          }\n          if (event.key === ' ') {\n            event.preventDefault()\n          }\n\n          return ['Enter', ' '].includes(event.key) && openable\n            ? setIsDropdownVisible(!isDropdownVisible)\n            : null\n        }}\n        ref={innerRef}\n        aria-haspopup=\"listbox\"\n        aria-expanded={isDropdownVisible}\n        tabIndex={0}\n        aria-label={label}\n      >\n        {shouldDisplayValues ? (\n          <DisplayValues\n            refTag={refTag}\n            nonOverflowedValues={nonOverflowedValues}\n            disabled={disabled}\n            readOnly={readOnly}\n            overflowed={!!overflowAmount}\n            overflowAmount={overflowAmount}\n            size={size}\n          />\n        ) : (\n          <Placeholder\n            as=\"p\"\n            variant={size === 'large' ? 'body' : 'bodySmall'}\n            sentiment=\"neutral\"\n            disabled={disabled}\n            prominence=\"weak\"\n          >\n            {placeholder}\n          </Placeholder>\n        )}\n        <StateStack direction=\"row\" gap={1} alignItems=\"center\">\n          {error ? <AlertCircleIcon sentiment=\"danger\" /> : null}\n          {success && !error ? <CheckCircleIcon sentiment=\"success\" /> : null}\n          {clearable && selectedData.selectedValues.length > 0 ? (\n            <Button\n              aria-label=\"clear value\"\n              disabled={disabled || !selectedData.selectedValues[0] || readOnly}\n              variant=\"ghost\"\n              size=\"small\"\n              onClick={event => {\n                event.stopPropagation()\n                setSelectedData({ type: 'clearAll' })\n                if (multiselect) {\n                  onChange?.([])\n                } else {\n                  onChange?.('')\n                }\n              }}\n              sentiment=\"neutral\"\n              data-testid=\"clear-all\"\n            >\n              <CloseIcon />\n            </Button>\n          ) : null}\n          <ArrowDownIcon\n            aria-label=\"show dropdown\"\n            size=\"small\"\n            sentiment=\"neutral\"\n            disabled={disabled || readOnly}\n          />\n        </StateStack>\n      </StyledInputWrapper>\n    </Tooltip>\n  )\n}\n"]} */",
|
|
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/SelectInputV2/SelectBar.tsx"],"names":[],"mappings":"AA8LmC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/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 { useSelectInput } from './SelectInputProvider'\nimport { findOptionInOptions } from './findOptionInOptions'\nimport type { OptionType } from './types'\nimport { INPUT_SIZE_HEIGHT } from './types'\n\nconst SIZES_TAG = {\n  paddings: 16,\n  plusTag: 48,\n  gap: 8,\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}\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      direction=\"row\"\n      gap=\"1\"\n      wrap=\"nowrap\"\n      ref={refTag}\n      alignItems=\"center\"\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            onClose={() => {}}\n            className={option.value}\n            key={option.value}\n            hidden\n          >\n            {option?.label}\n          </CustomTag>\n        ))}\n      </div>\n      {nonOverflowedValues.map((option, index) => (\n        <CustomTag\n          data-testid=\"selected-options-tags\"\n          sentiment=\"neutral\"\n          key={option?.value}\n          disabled={disabled}\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                    type: 'selectOption',\n                    clickedOption: option,\n                  })\n                  const newSelectedValues = selectedData.selectedValues?.filter(\n                    val => val !== option.value,\n                  )\n                  onChange?.(newSelectedValues)\n                }\n              : undefined\n          }\n        >\n          {option?.label}\n        </CustomTag>\n      ))}\n\n      {overflowed ? (\n        <Stack ref={refPlusTag} justifyContent=\"center\">\n          <PlusTag\n            sentiment=\"neutral\"\n            disabled={disabled}\n            key=\"+\"\n            data-testid=\"plus-tag\"\n            aria-label=\"Plus tag\"\n          >\n            <PlusIcon size=\"xsmall\" />\n            {overflowAmount}\n          </PlusTag>\n        </Stack>\n      ) : null}\n    </MultiselectStack>\n  ) : (\n    <SelectedValues\n      as=\"div\"\n      variant={size === 'large' ? 'body' : 'bodySmall'}\n      disabled={disabled}\n      prominence=\"default\"\n      sentiment=\"neutral\"\n    >\n      {selectedData.selectedValues[0]\n        ? findOptionInOptions(options, selectedData.selectedValues[0])?.label\n        : null}\n    </SelectedValues>\n  )\n}\n\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}: 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            measuredVisibleTags: [\n              ...accumulator.measuredVisibleTags,\n              canBeVisible && potentiallyNonOverflowedValues[index],\n            ].filter(Boolean) as OptionType[],\n            measuredHiddenTags:\n              accumulator.measuredHiddenTags + (!canBeVisible ? 1 : 0),\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          }\n        },\n        {\n          measuredVisibleTags: [],\n          measuredHiddenTags: 0,\n          accumulatedWidth: 0,\n          lastVisibleElementWidth: 0,\n          lastVisibleLabel: '',\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    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        role=\"combobox\"\n        id={id}\n        data-disabled={disabled}\n        data-readonly={readOnly}\n        data-size={size}\n        data-dropdownvisible={isDropdownVisible}\n        data-state={state}\n        onClick={\n          openable ? () => setIsDropdownVisible(!isDropdownVisible) : undefined\n        }\n        data-testid={dataTestId}\n        autoFocus={autoFocus}\n        onKeyDown={event => {\n          if (event.key === 'ArrowDown') {\n            if (!isDropdownVisible) {\n              setIsDropdownVisible(true)\n            } else {\n              document.getElementById(`option-0`)?.focus()\n            }\n          }\n          if (event.key === ' ') {\n            event.preventDefault()\n          }\n\n          return ['Enter', ' '].includes(event.key) && openable\n            ? setIsDropdownVisible(!isDropdownVisible)\n            : null\n        }}\n        ref={innerRef}\n        aria-haspopup=\"listbox\"\n        aria-expanded={isDropdownVisible}\n        tabIndex={0}\n        aria-label={label}\n      >\n        {shouldDisplayValues ? (\n          <DisplayValues\n            refTag={refTag}\n            nonOverflowedValues={nonOverflowedValues}\n            potentiallyNonOverflowedValues={potentiallyNonOverflowedValues}\n            disabled={disabled}\n            readOnly={readOnly}\n            overflowed={!!overflowAmount}\n            overflowAmount={overflowAmount}\n            size={size}\n            measureRef={measureRef}\n            lastElementMaxWidth={lastElementMaxWidth}\n            overflow={overflow}\n            refPlusTag={refPlusTag}\n          />\n        ) : (\n          <Placeholder\n            as=\"p\"\n            variant={size === 'large' ? 'body' : 'bodySmall'}\n            sentiment=\"neutral\"\n            disabled={disabled}\n            prominence=\"weak\"\n          >\n            {placeholder}\n          </Placeholder>\n        )}\n        <StateStack direction=\"row\" gap={1} alignItems=\"center\" 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              disabled={disabled || !selectedData.selectedValues[0] || readOnly}\n              variant=\"ghost\"\n              size=\"small\"\n              onClick={event => {\n                event.stopPropagation()\n                setSelectedData({ type: 'clearAll' })\n                if (multiselect) {\n                  onChange?.([])\n                } else {\n                  onChange?.('')\n                }\n              }}\n              sentiment=\"neutral\"\n              data-testid=\"clear-all\"\n            >\n              <CloseIcon />\n            </Button>\n          ) : null}\n          <ArrowDownIcon\n            aria-label=\"show dropdown\"\n            size=\"small\"\n            sentiment=\"neutral\"\n            disabled={disabled || readOnly}\n          />\n        </StateStack>\n      </StyledInputWrapper>\n    </Tooltip>\n  )\n}\n\nexport { SelectBar, StyledInputWrapper }\n"]} */",
|
|
107
131
|
toString: _EMOTION_STRINGIFIED_CSS_ERROR__
|
|
108
132
|
});
|
|
109
|
-
const
|
|
133
|
+
const PlusTag = /* @__PURE__ */ _styled(Tag, process.env.NODE_ENV === "production" ? {
|
|
134
|
+
target: "ejy0aca1"
|
|
135
|
+
} : {
|
|
136
|
+
target: "ejy0aca1",
|
|
137
|
+
label: "PlusTag"
|
|
138
|
+
})("width:", ({
|
|
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/SelectInputV2/SelectBar.tsx"],"names":[],"mappings":"AAoM2B","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/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 { useSelectInput } from './SelectInputProvider'\nimport { findOptionInOptions } from './findOptionInOptions'\nimport type { OptionType } from './types'\nimport { INPUT_SIZE_HEIGHT } from './types'\n\nconst SIZES_TAG = {\n  paddings: 16,\n  plusTag: 48,\n  gap: 8,\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}\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      direction=\"row\"\n      gap=\"1\"\n      wrap=\"nowrap\"\n      ref={refTag}\n      alignItems=\"center\"\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            onClose={() => {}}\n            className={option.value}\n            key={option.value}\n            hidden\n          >\n            {option?.label}\n          </CustomTag>\n        ))}\n      </div>\n      {nonOverflowedValues.map((option, index) => (\n        <CustomTag\n          data-testid=\"selected-options-tags\"\n          sentiment=\"neutral\"\n          key={option?.value}\n          disabled={disabled}\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                    type: 'selectOption',\n                    clickedOption: option,\n                  })\n                  const newSelectedValues = selectedData.selectedValues?.filter(\n                    val => val !== option.value,\n                  )\n                  onChange?.(newSelectedValues)\n                }\n              : undefined\n          }\n        >\n          {option?.label}\n        </CustomTag>\n      ))}\n\n      {overflowed ? (\n        <Stack ref={refPlusTag} justifyContent=\"center\">\n          <PlusTag\n            sentiment=\"neutral\"\n            disabled={disabled}\n            key=\"+\"\n            data-testid=\"plus-tag\"\n            aria-label=\"Plus tag\"\n          >\n            <PlusIcon size=\"xsmall\" />\n            {overflowAmount}\n          </PlusTag>\n        </Stack>\n      ) : null}\n    </MultiselectStack>\n  ) : (\n    <SelectedValues\n      as=\"div\"\n      variant={size === 'large' ? 'body' : 'bodySmall'}\n      disabled={disabled}\n      prominence=\"default\"\n      sentiment=\"neutral\"\n    >\n      {selectedData.selectedValues[0]\n        ? findOptionInOptions(options, selectedData.selectedValues[0])?.label\n        : null}\n    </SelectedValues>\n  )\n}\n\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}: 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            measuredVisibleTags: [\n              ...accumulator.measuredVisibleTags,\n              canBeVisible && potentiallyNonOverflowedValues[index],\n            ].filter(Boolean) as OptionType[],\n            measuredHiddenTags:\n              accumulator.measuredHiddenTags + (!canBeVisible ? 1 : 0),\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          }\n        },\n        {\n          measuredVisibleTags: [],\n          measuredHiddenTags: 0,\n          accumulatedWidth: 0,\n          lastVisibleElementWidth: 0,\n          lastVisibleLabel: '',\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    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        role=\"combobox\"\n        id={id}\n        data-disabled={disabled}\n        data-readonly={readOnly}\n        data-size={size}\n        data-dropdownvisible={isDropdownVisible}\n        data-state={state}\n        onClick={\n          openable ? () => setIsDropdownVisible(!isDropdownVisible) : undefined\n        }\n        data-testid={dataTestId}\n        autoFocus={autoFocus}\n        onKeyDown={event => {\n          if (event.key === 'ArrowDown') {\n            if (!isDropdownVisible) {\n              setIsDropdownVisible(true)\n            } else {\n              document.getElementById(`option-0`)?.focus()\n            }\n          }\n          if (event.key === ' ') {\n            event.preventDefault()\n          }\n\n          return ['Enter', ' '].includes(event.key) && openable\n            ? setIsDropdownVisible(!isDropdownVisible)\n            : null\n        }}\n        ref={innerRef}\n        aria-haspopup=\"listbox\"\n        aria-expanded={isDropdownVisible}\n        tabIndex={0}\n        aria-label={label}\n      >\n        {shouldDisplayValues ? (\n          <DisplayValues\n            refTag={refTag}\n            nonOverflowedValues={nonOverflowedValues}\n            potentiallyNonOverflowedValues={potentiallyNonOverflowedValues}\n            disabled={disabled}\n            readOnly={readOnly}\n            overflowed={!!overflowAmount}\n            overflowAmount={overflowAmount}\n            size={size}\n            measureRef={measureRef}\n            lastElementMaxWidth={lastElementMaxWidth}\n            overflow={overflow}\n            refPlusTag={refPlusTag}\n          />\n        ) : (\n          <Placeholder\n            as=\"p\"\n            variant={size === 'large' ? 'body' : 'bodySmall'}\n            sentiment=\"neutral\"\n            disabled={disabled}\n            prominence=\"weak\"\n          >\n            {placeholder}\n          </Placeholder>\n        )}\n        <StateStack direction=\"row\" gap={1} alignItems=\"center\" 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              disabled={disabled || !selectedData.selectedValues[0] || readOnly}\n              variant=\"ghost\"\n              size=\"small\"\n              onClick={event => {\n                event.stopPropagation()\n                setSelectedData({ type: 'clearAll' })\n                if (multiselect) {\n                  onChange?.([])\n                } else {\n                  onChange?.('')\n                }\n              }}\n              sentiment=\"neutral\"\n              data-testid=\"clear-all\"\n            >\n              <CloseIcon />\n            </Button>\n          ) : null}\n          <ArrowDownIcon\n            aria-label=\"show dropdown\"\n            size=\"small\"\n            sentiment=\"neutral\"\n            disabled={disabled || readOnly}\n          />\n        </StateStack>\n      </StyledInputWrapper>\n    </Tooltip>\n  )\n}\n\nexport { SelectBar, StyledInputWrapper }\n"]} */"));
|
|
141
|
+
const MultiselectStack = /* @__PURE__ */ _styled(Stack, process.env.NODE_ENV === "production" ? {
|
|
110
142
|
target: "ejy0aca0"
|
|
111
143
|
} : {
|
|
112
144
|
target: "ejy0aca0",
|
|
113
|
-
label: "
|
|
145
|
+
label: "MultiselectStack"
|
|
114
146
|
})(process.env.NODE_ENV === "production" ? {
|
|
115
|
-
name: "
|
|
116
|
-
styles: "
|
|
147
|
+
name: "1llx5fl",
|
|
148
|
+
styles: "overflow:hidden;max-width:100%;height:100%"
|
|
117
149
|
} : {
|
|
118
|
-
name: "
|
|
119
|
-
styles: "text-overflow:ellipsis;overflow:hidden;white-space:nowrap/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/SelectBar.tsx"],"names":[],"mappings":"AA0JmC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/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 { RefObject } from 'react'\nimport { useEffect, useMemo, useRef, useState } from 'react'\nimport { Button } from '../Button'\nimport { Stack } from '../Stack'\nimport { Tag } from '../Tag'\nimport { Text } from '../Text'\nimport { Tooltip } from '../Tooltip'\nimport { useSelectInput } from './SelectInputProvider'\nimport { findOptionInOptions } from './findOptionInOptions'\nimport type { DataType, OptionType } from './types'\nimport { INPUT_SIZE_HEIGHT, SIZES_TAG } from './types'\n\ntype SelectBarProps = {\n  size: 'small' | 'medium' | 'large'\n  clearable: boolean\n  disabled: boolean\n  readOnly: boolean\n  placeholder: string\n  success?: string\n  error?: string | boolean\n  autoFocus?: boolean\n  innerRef: RefObject<HTMLDivElement | null>\n  id?: string\n  'data-testid': string\n  label?: string\n  tooltip?: string\n}\n\ntype DisplayValuesProps = {\n  refTag: RefObject<HTMLDivElement | null>\n  nonOverflowedValues: OptionType[]\n  disabled: boolean\n  readOnly: boolean\n  overflowed: boolean\n  overflowAmount: number\n  size: 'small' | 'medium' | 'large'\n}\n\nconst StateStack = styled(Stack)`\n  padding-right: ${({ theme }) => theme.space['2']};\n  display: flex;\n`\nconst Placeholder = styled(Text)`\nuser-select: none;\n`\n\nexport const StyledInputWrapper = styled(Stack)<{\n  'data-readonly': boolean\n  'data-disabled': boolean\n  'data-size': 'small' | 'medium' | 'large'\n  'data-state': 'neutral' | 'success' | 'danger'\n  'data-dropdownvisible': boolean\n  'aria-label'?: string\n}>`\n  display: flex;\n  padding: ${({ theme }) => theme.space[1]};\n  padding-right: 0;\n  padding-left: ${({ theme }) => theme.space[2]};\n  cursor: pointer;\n  box-shadow: none;\n  background: ${({ theme }) => theme.colors.neutral.background};\n  border-radius: ${({ theme }) => theme.radii.default};\n  width: 100%;\n  overflow: hidden;\n\n  &[data-size='small'] {\n    height: ${({ theme }) => theme.sizing[INPUT_SIZE_HEIGHT.small]};\n    padding-left: ${({ theme }) => theme.space[1]};\n  }\n  &[data-size='medium'] {\n    height: ${({ theme }) => theme.sizing[INPUT_SIZE_HEIGHT.medium]};\n  }\n  &[data-size='large'] {\n    height: ${({ theme }) => theme.sizing[INPUT_SIZE_HEIGHT.large]};\n  }\n  &[data-state='neutral'] {\n    border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n\n    &:not([data-disabled=\"true\"]):not([data-readonly=\"true\"]):active {\n      border-color: ${({ theme }) => theme.colors.primary.borderHover};\n      box-shadow: ${({ theme }) => theme.shadows.focusPrimary};\n    }\n    &:not([data-disabled='true']):hover {\n      border-color: ${({ theme }) => theme.colors.primary.borderHover};\n      outline: none;\n    }\n\n    &:not([data-disabled='true']):focus-visible {\n      outline: 5px auto Highlight;\n      outline: 5px auto -webkit-focus-ring-color;\n    }\n\n    &[data-dropdownvisible='true'] {\n    border-color: ${({ theme }) => theme.colors.primary.borderHover};\n  }\n  }\n\n  &[data-state='success'] {\n    border: 1px solid ${({ theme }) => theme.colors.success.border};\n    &:not([data-disabled=\"true\"]):not([data-readonly=\"true\"]):active {\n      border-color: ${({ theme }) => theme.colors.success.borderHover};\n      box-shadow: ${({ theme }) => theme.shadows.focusSuccess};\n    }\n\n    &[data-dropdownvisible='true'] {\n      border-color: ${({ theme }) => theme.colors.success.borderHover};\n    }\n  }\n\n  &[data-state='danger'] {\n    border: 1px solid ${({ theme }) => theme.colors.danger.border};\n\n    &:not([data-disabled=\"true\"]):not([data-readonly=\"true\"]):active {\n      border-color: ${({ theme }) => theme.colors.danger.borderHover};\n      box-shadow: ${({ theme }) => theme.shadows.focusDanger};\n    }\n\n    &[data-dropdownvisible='true'] {\n      border-color: ${({ theme }) => theme.colors.danger.borderHover};\n    }\n  }\n\n  &:not([data-disabled='true']):not([data-readonly]):hover {\n    border-color: ${({ theme }) => theme.colors.primary.border};\n  }\n\n  &[data-readonly='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n    border-color: ${({ theme }) => theme.colors.neutral.border};\n    cursor: default;\n  }\n\n  &[data-disabled='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    border-color: ${({ theme }) => theme.colors.neutral.borderDisabled};\n    cursor: not-allowed;\n  }\n`\n\nconst CustomTag = styled(Tag)`\n  height: fit-content;\n  width: fit-content;\n`\n\nconst SelectedValues = styled(Text)`\n  text-overflow: ellipsis;\n  overflow: hidden;\n  white-space: nowrap;\n`\n\nconst isValidSelectedValue = (selectedValue: string, options: DataType) =>\n  !Array.isArray(options)\n    ? Object.keys(options).some(group =>\n        options[group].some(\n          option => option.value === selectedValue && !option.disabled,\n        ),\n      )\n    : options.some(option => option.value === selectedValue && !option.disabled)\n\nconst DisplayValues = ({\n  refTag,\n  nonOverflowedValues,\n  disabled,\n  readOnly,\n  overflowed,\n  overflowAmount,\n  size,\n}: DisplayValuesProps) => {\n  const { multiselect, selectedData, setSelectedData, options, onChange } =\n    useSelectInput()\n\n  return multiselect ? (\n    <Stack\n      direction=\"row\"\n      gap=\"1\"\n      wrap=\"nowrap\"\n      ref={refTag}\n      alignItems=\"center\"\n    >\n      {nonOverflowedValues.map(option => (\n        <CustomTag\n          data-testid=\"selected-options-tags\"\n          key={option?.value}\n          sentiment=\"neutral\"\n          disabled={disabled}\n          onClose={\n            !readOnly\n              ? event => {\n                  event.stopPropagation()\n                  setSelectedData({\n                    type: 'selectOption',\n                    clickedOption: option,\n                  })\n                  const newSelectedValues = selectedData.selectedValues?.filter(\n                    val => val !== option.value,\n                  )\n                  onChange?.(newSelectedValues)\n                }\n              : undefined\n          }\n        >\n          {option?.label}\n        </CustomTag>\n      ))}\n      {overflowed ? (\n        <Tag\n          sentiment=\"neutral\"\n          disabled={disabled}\n          key=\"+\"\n          data-testid=\"plus-tag\"\n          aria-label=\"Plus tag\"\n        >\n          <PlusIcon />\n          {overflowAmount}\n        </Tag>\n      ) : null}\n    </Stack>\n  ) : (\n    <SelectedValues\n      as=\"div\"\n      variant={size === 'large' ? 'body' : 'bodySmall'}\n      disabled={disabled}\n      prominence=\"default\"\n      sentiment=\"neutral\"\n    >\n      {selectedData.selectedValues[0]\n        ? findOptionInOptions(options, selectedData.selectedValues[0])?.label\n        : null}\n    </SelectedValues>\n  )\n}\n\nexport const SelectBar = ({\n  size,\n  clearable,\n  disabled,\n  readOnly,\n  placeholder,\n  success,\n  error,\n  autoFocus,\n  tooltip,\n  innerRef,\n  id,\n  'data-testid': dataTestId,\n  label,\n}: SelectBarProps) => {\n  const {\n    isDropdownVisible,\n    onChange,\n    setIsDropdownVisible,\n    options,\n    selectedData,\n    setSelectedData,\n    multiselect,\n  } = useSelectInput()\n  const openable = !(readOnly || disabled)\n  const refTag = useRef<HTMLDivElement>(null)\n  const width = innerRef.current?.offsetWidth\n  const [overflowAmount, setOverflowAmount] = useState(0)\n  const [nonOverflowedValues, setNonOverFlowedValues] = useState<OptionType[]>(\n    () => {\n      if (selectedData.selectedValues[0]) {\n        const firstSelectOption = findOptionInOptions(\n          options,\n          selectedData.selectedValues[0],\n        )\n\n        return firstSelectOption ? [firstSelectOption] : []\n      }\n\n      return []\n    },\n  )\n\n  const state = useMemo(() => {\n    if (error) {\n      return 'danger'\n    }\n    if (success) {\n      return 'success'\n    }\n\n    return 'neutral'\n  }, [error, success])\n\n  useEffect(() => {\n    // When too many items are selected, too avoid overflow, compute the number of tags to display and add a + tag\n    let tagsWidth = 0\n    let computedOverflowAmount = 0\n    let computedNonOverflowedValues: OptionType[] = []\n    const newSelectedValues = selectedData.selectedValues.filter(\n      selectedValue => isValidSelectedValue(selectedValue, options),\n    )\n    for (const selectedValue of newSelectedValues) {\n      const selectedOption = findOptionInOptions(options, selectedValue)\n      if (\n        selectedOption?.label &&\n        width &&\n        isValidSelectedValue(selectedValue, options)\n      ) {\n        const lengthValue = selectedOption.value.length // Find a better way to find the number of displayed characters?\n        const totalTagWidth =\n          SIZES_TAG.tagWidth + SIZES_TAG.letterWidth * lengthValue\n        if (totalTagWidth + tagsWidth > width - 100) {\n          computedOverflowAmount += 1\n        } else {\n          computedNonOverflowedValues = [\n            ...computedNonOverflowedValues,\n            selectedOption,\n          ]\n          tagsWidth += totalTagWidth\n        }\n      }\n    }\n    setNonOverFlowedValues(computedNonOverflowedValues)\n    setOverflowAmount(computedOverflowAmount)\n  }, [options, selectedData.selectedValues, width])\n\n  useEffect(() => {\n    setSelectedData({ type: 'update' })\n  }, [setSelectedData, options])\n\n  const shouldDisplayValues = useMemo(() => {\n    if (multiselect) {\n      return (\n        nonOverflowedValues.length > 0 ||\n        selectedData.selectedValues.some(\n          selectedValue =>\n            findOptionInOptions(options, selectedValue) !== undefined,\n        )\n      )\n    }\n\n    return (\n      selectedData.selectedValues[0] !== undefined &&\n      findOptionInOptions(options, selectedData.selectedValues[0]) !== undefined\n    )\n  }, [\n    multiselect,\n    nonOverflowedValues.length,\n    options,\n    selectedData.selectedValues,\n  ])\n\n  return (\n    <Tooltip text={tooltip}>\n      <StyledInputWrapper\n        role=\"combobox\"\n        id={id}\n        data-disabled={disabled}\n        data-readonly={readOnly}\n        data-size={size}\n        data-dropdownvisible={isDropdownVisible}\n        data-state={state}\n        direction=\"row\"\n        wrap=\"nowrap\"\n        gap=\"1\"\n        justifyContent=\"space-between\"\n        alignItems=\"center\"\n        onClick={\n          openable ? () => setIsDropdownVisible(!isDropdownVisible) : undefined\n        }\n        data-testid={dataTestId}\n        autoFocus={autoFocus}\n        onKeyDown={event => {\n          if (event.key === 'ArrowDown') {\n            if (!isDropdownVisible) {\n              setIsDropdownVisible(true)\n            } else {\n              document.getElementById(`option-0`)?.focus()\n            }\n          }\n          if (event.key === ' ') {\n            event.preventDefault()\n          }\n\n          return ['Enter', ' '].includes(event.key) && openable\n            ? setIsDropdownVisible(!isDropdownVisible)\n            : null\n        }}\n        ref={innerRef}\n        aria-haspopup=\"listbox\"\n        aria-expanded={isDropdownVisible}\n        tabIndex={0}\n        aria-label={label}\n      >\n        {shouldDisplayValues ? (\n          <DisplayValues\n            refTag={refTag}\n            nonOverflowedValues={nonOverflowedValues}\n            disabled={disabled}\n            readOnly={readOnly}\n            overflowed={!!overflowAmount}\n            overflowAmount={overflowAmount}\n            size={size}\n          />\n        ) : (\n          <Placeholder\n            as=\"p\"\n            variant={size === 'large' ? 'body' : 'bodySmall'}\n            sentiment=\"neutral\"\n            disabled={disabled}\n            prominence=\"weak\"\n          >\n            {placeholder}\n          </Placeholder>\n        )}\n        <StateStack direction=\"row\" gap={1} alignItems=\"center\">\n          {error ? <AlertCircleIcon sentiment=\"danger\" /> : null}\n          {success && !error ? <CheckCircleIcon sentiment=\"success\" /> : null}\n          {clearable && selectedData.selectedValues.length > 0 ? (\n            <Button\n              aria-label=\"clear value\"\n              disabled={disabled || !selectedData.selectedValues[0] || readOnly}\n              variant=\"ghost\"\n              size=\"small\"\n              onClick={event => {\n                event.stopPropagation()\n                setSelectedData({ type: 'clearAll' })\n                if (multiselect) {\n                  onChange?.([])\n                } else {\n                  onChange?.('')\n                }\n              }}\n              sentiment=\"neutral\"\n              data-testid=\"clear-all\"\n            >\n              <CloseIcon />\n            </Button>\n          ) : null}\n          <ArrowDownIcon\n            aria-label=\"show dropdown\"\n            size=\"small\"\n            sentiment=\"neutral\"\n            disabled={disabled || readOnly}\n          />\n        </StateStack>\n      </StyledInputWrapper>\n    </Tooltip>\n  )\n}\n"]} */",
|
|
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/SelectInputV2/SelectBar.tsx"],"names":[],"mappings":"AAyMsC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/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 { useSelectInput } from './SelectInputProvider'\nimport { findOptionInOptions } from './findOptionInOptions'\nimport type { OptionType } from './types'\nimport { INPUT_SIZE_HEIGHT } from './types'\n\nconst SIZES_TAG = {\n  paddings: 16,\n  plusTag: 48,\n  gap: 8,\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}\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      direction=\"row\"\n      gap=\"1\"\n      wrap=\"nowrap\"\n      ref={refTag}\n      alignItems=\"center\"\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            onClose={() => {}}\n            className={option.value}\n            key={option.value}\n            hidden\n          >\n            {option?.label}\n          </CustomTag>\n        ))}\n      </div>\n      {nonOverflowedValues.map((option, index) => (\n        <CustomTag\n          data-testid=\"selected-options-tags\"\n          sentiment=\"neutral\"\n          key={option?.value}\n          disabled={disabled}\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                    type: 'selectOption',\n                    clickedOption: option,\n                  })\n                  const newSelectedValues = selectedData.selectedValues?.filter(\n                    val => val !== option.value,\n                  )\n                  onChange?.(newSelectedValues)\n                }\n              : undefined\n          }\n        >\n          {option?.label}\n        </CustomTag>\n      ))}\n\n      {overflowed ? (\n        <Stack ref={refPlusTag} justifyContent=\"center\">\n          <PlusTag\n            sentiment=\"neutral\"\n            disabled={disabled}\n            key=\"+\"\n            data-testid=\"plus-tag\"\n            aria-label=\"Plus tag\"\n          >\n            <PlusIcon size=\"xsmall\" />\n            {overflowAmount}\n          </PlusTag>\n        </Stack>\n      ) : null}\n    </MultiselectStack>\n  ) : (\n    <SelectedValues\n      as=\"div\"\n      variant={size === 'large' ? 'body' : 'bodySmall'}\n      disabled={disabled}\n      prominence=\"default\"\n      sentiment=\"neutral\"\n    >\n      {selectedData.selectedValues[0]\n        ? findOptionInOptions(options, selectedData.selectedValues[0])?.label\n        : null}\n    </SelectedValues>\n  )\n}\n\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}: 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            measuredVisibleTags: [\n              ...accumulator.measuredVisibleTags,\n              canBeVisible && potentiallyNonOverflowedValues[index],\n            ].filter(Boolean) as OptionType[],\n            measuredHiddenTags:\n              accumulator.measuredHiddenTags + (!canBeVisible ? 1 : 0),\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          }\n        },\n        {\n          measuredVisibleTags: [],\n          measuredHiddenTags: 0,\n          accumulatedWidth: 0,\n          lastVisibleElementWidth: 0,\n          lastVisibleLabel: '',\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    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        role=\"combobox\"\n        id={id}\n        data-disabled={disabled}\n        data-readonly={readOnly}\n        data-size={size}\n        data-dropdownvisible={isDropdownVisible}\n        data-state={state}\n        onClick={\n          openable ? () => setIsDropdownVisible(!isDropdownVisible) : undefined\n        }\n        data-testid={dataTestId}\n        autoFocus={autoFocus}\n        onKeyDown={event => {\n          if (event.key === 'ArrowDown') {\n            if (!isDropdownVisible) {\n              setIsDropdownVisible(true)\n            } else {\n              document.getElementById(`option-0`)?.focus()\n            }\n          }\n          if (event.key === ' ') {\n            event.preventDefault()\n          }\n\n          return ['Enter', ' '].includes(event.key) && openable\n            ? setIsDropdownVisible(!isDropdownVisible)\n            : null\n        }}\n        ref={innerRef}\n        aria-haspopup=\"listbox\"\n        aria-expanded={isDropdownVisible}\n        tabIndex={0}\n        aria-label={label}\n      >\n        {shouldDisplayValues ? (\n          <DisplayValues\n            refTag={refTag}\n            nonOverflowedValues={nonOverflowedValues}\n            potentiallyNonOverflowedValues={potentiallyNonOverflowedValues}\n            disabled={disabled}\n            readOnly={readOnly}\n            overflowed={!!overflowAmount}\n            overflowAmount={overflowAmount}\n            size={size}\n            measureRef={measureRef}\n            lastElementMaxWidth={lastElementMaxWidth}\n            overflow={overflow}\n            refPlusTag={refPlusTag}\n          />\n        ) : (\n          <Placeholder\n            as=\"p\"\n            variant={size === 'large' ? 'body' : 'bodySmall'}\n            sentiment=\"neutral\"\n            disabled={disabled}\n            prominence=\"weak\"\n          >\n            {placeholder}\n          </Placeholder>\n        )}\n        <StateStack direction=\"row\" gap={1} alignItems=\"center\" 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              disabled={disabled || !selectedData.selectedValues[0] || readOnly}\n              variant=\"ghost\"\n              size=\"small\"\n              onClick={event => {\n                event.stopPropagation()\n                setSelectedData({ type: 'clearAll' })\n                if (multiselect) {\n                  onChange?.([])\n                } else {\n                  onChange?.('')\n                }\n              }}\n              sentiment=\"neutral\"\n              data-testid=\"clear-all\"\n            >\n              <CloseIcon />\n            </Button>\n          ) : null}\n          <ArrowDownIcon\n            aria-label=\"show dropdown\"\n            size=\"small\"\n            sentiment=\"neutral\"\n            disabled={disabled || readOnly}\n          />\n        </StateStack>\n      </StyledInputWrapper>\n    </Tooltip>\n  )\n}\n\nexport { SelectBar, StyledInputWrapper }\n"]} */",
|
|
120
152
|
toString: _EMOTION_STRINGIFIED_CSS_ERROR__
|
|
121
153
|
});
|
|
122
|
-
const isValidSelectedValue = (selectedValue, options) => !Array.isArray(options) ? Object.keys(options).some((group) => options[group].some((option) => option.value === selectedValue && !option.disabled)) : options.some((option) => option.value === selectedValue && !option.disabled);
|
|
123
154
|
const DisplayValues = ({
|
|
124
155
|
refTag,
|
|
125
156
|
nonOverflowedValues,
|
|
157
|
+
potentiallyNonOverflowedValues,
|
|
126
158
|
disabled,
|
|
127
159
|
readOnly,
|
|
128
160
|
overflowed,
|
|
129
161
|
overflowAmount,
|
|
130
|
-
size
|
|
162
|
+
size,
|
|
163
|
+
measureRef,
|
|
164
|
+
lastElementMaxWidth,
|
|
165
|
+
overflow,
|
|
166
|
+
refPlusTag
|
|
131
167
|
}) => {
|
|
132
168
|
const {
|
|
133
169
|
multiselect,
|
|
@@ -136,8 +172,12 @@ const DisplayValues = ({
|
|
|
136
172
|
options,
|
|
137
173
|
onChange
|
|
138
174
|
} = useSelectInput();
|
|
139
|
-
return multiselect ? /* @__PURE__ */ jsxs(
|
|
140
|
-
|
|
175
|
+
return multiselect ? /* @__PURE__ */ jsxs(MultiselectStack, { direction: "row", gap: "1", wrap: "nowrap", ref: refTag, alignItems: "center", children: [
|
|
176
|
+
/* @__PURE__ */ jsx("div", { ref: measureRef, style: {
|
|
177
|
+
position: "absolute"
|
|
178
|
+
}, children: potentiallyNonOverflowedValues.map((option) => /* @__PURE__ */ jsx(CustomTag, { onClose: () => {
|
|
179
|
+
}, className: option.value, hidden: true, children: option?.label }, option.value)) }),
|
|
180
|
+
nonOverflowedValues.map((option, index) => /* @__PURE__ */ jsx(CustomTag, { "data-testid": "selected-options-tags", sentiment: "neutral", disabled, lastElementMaxWidth: index === nonOverflowedValues.length - 1 && overflow ? lastElementMaxWidth : 0, onClose: !readOnly ? (event) => {
|
|
141
181
|
event.stopPropagation();
|
|
142
182
|
setSelectedData({
|
|
143
183
|
type: "selectOption",
|
|
@@ -146,10 +186,10 @@ const DisplayValues = ({
|
|
|
146
186
|
const newSelectedValues = selectedData.selectedValues?.filter((val) => val !== option.value);
|
|
147
187
|
onChange?.(newSelectedValues);
|
|
148
188
|
} : void 0, children: option?.label }, option?.value)),
|
|
149
|
-
overflowed ? /* @__PURE__ */ jsxs(
|
|
150
|
-
/* @__PURE__ */ jsx(PlusIcon, {}),
|
|
189
|
+
overflowed ? /* @__PURE__ */ jsx(Stack, { ref: refPlusTag, justifyContent: "center", children: /* @__PURE__ */ jsxs(PlusTag, { sentiment: "neutral", disabled, "data-testid": "plus-tag", "aria-label": "Plus tag", children: [
|
|
190
|
+
/* @__PURE__ */ jsx(PlusIcon, { size: "xsmall" }),
|
|
151
191
|
overflowAmount
|
|
152
|
-
] }, "+") : null
|
|
192
|
+
] }, "+") }) : null
|
|
153
193
|
] }) : /* @__PURE__ */ jsx(SelectedValues, { as: "div", variant: size === "large" ? "body" : "bodySmall", disabled, prominence: "default", sentiment: "neutral", children: selectedData.selectedValues[0] ? findOptionInOptions(options, selectedData.selectedValues[0])?.label : null });
|
|
154
194
|
};
|
|
155
195
|
const SelectBar = ({
|
|
@@ -178,8 +218,13 @@ const SelectBar = ({
|
|
|
178
218
|
} = useSelectInput();
|
|
179
219
|
const openable = !(readOnly || disabled);
|
|
180
220
|
const refTag = useRef(null);
|
|
181
|
-
const
|
|
221
|
+
const measureRef = useRef(null);
|
|
222
|
+
const arrowRef = useRef(null);
|
|
223
|
+
const refPlusTag = useRef(null);
|
|
224
|
+
const [innerWidth, setInnerWidth] = useState(innerRef.current?.offsetWidth ?? 0 - (arrowRef.current?.offsetWidth ?? 0) - SIZES_TAG.paddings);
|
|
182
225
|
const [overflowAmount, setOverflowAmount] = useState(0);
|
|
226
|
+
const [overflow, setOverflow] = useState(false);
|
|
227
|
+
const [lastElementMaxWidth, setLastElementMaxWidth] = useState(0);
|
|
183
228
|
const [nonOverflowedValues, setNonOverFlowedValues] = useState(() => {
|
|
184
229
|
if (selectedData.selectedValues[0]) {
|
|
185
230
|
const firstSelectOption = findOptionInOptions(options, selectedData.selectedValues[0]);
|
|
@@ -187,6 +232,7 @@ const SelectBar = ({
|
|
|
187
232
|
}
|
|
188
233
|
return [];
|
|
189
234
|
});
|
|
235
|
+
const potentiallyNonOverflowedValues = useMemo(() => selectedData.selectedValues.map((selectedValue) => findOptionInOptions(options, selectedValue)).filter((option) => !!option), [options, selectedData.selectedValues]);
|
|
190
236
|
const state = useMemo(() => {
|
|
191
237
|
if (error) {
|
|
192
238
|
return "danger";
|
|
@@ -197,38 +243,88 @@ const SelectBar = ({
|
|
|
197
243
|
return "neutral";
|
|
198
244
|
}, [error, success]);
|
|
199
245
|
useEffect(() => {
|
|
200
|
-
|
|
201
|
-
|
|
202
|
-
|
|
203
|
-
|
|
204
|
-
|
|
205
|
-
const
|
|
206
|
-
|
|
207
|
-
|
|
208
|
-
|
|
209
|
-
|
|
210
|
-
|
|
211
|
-
|
|
212
|
-
|
|
213
|
-
|
|
214
|
-
|
|
246
|
+
if (selectedData.selectedValues.length === 0) {
|
|
247
|
+
setOverflowAmount(0);
|
|
248
|
+
setNonOverFlowedValues([]);
|
|
249
|
+
}
|
|
250
|
+
if (measureRef.current && selectedData.selectedValues.length > 0) {
|
|
251
|
+
const toMeasureElements = measureRef.current.children;
|
|
252
|
+
const toMeasureElementsArray = [...toMeasureElements];
|
|
253
|
+
const {
|
|
254
|
+
measuredVisibleTags,
|
|
255
|
+
measuredHiddenTags,
|
|
256
|
+
accumulatedWidth,
|
|
257
|
+
lastVisibleElementWidth,
|
|
258
|
+
lastVisibleLabel
|
|
259
|
+
} = toMeasureElementsArray.reduce((accumulator, currentValue, index) => {
|
|
260
|
+
const elementWidth = currentValue.offsetWidth;
|
|
261
|
+
const newAccumulatedWidth = accumulator.accumulatedWidth + elementWidth + SIZES_TAG.gap;
|
|
262
|
+
const canBeVisible = newAccumulatedWidth <= innerWidth;
|
|
263
|
+
return {
|
|
264
|
+
measuredVisibleTags: [...accumulator.measuredVisibleTags, canBeVisible && potentiallyNonOverflowedValues[index]].filter(Boolean),
|
|
265
|
+
measuredHiddenTags: accumulator.measuredHiddenTags + (!canBeVisible ? 1 : 0),
|
|
266
|
+
accumulatedWidth: !canBeVisible ? accumulator.accumulatedWidth : newAccumulatedWidth,
|
|
267
|
+
lastVisibleElementWidth: canBeVisible ? elementWidth : accumulator.lastVisibleElementWidth,
|
|
268
|
+
lastVisibleLabel: canBeVisible ? potentiallyNonOverflowedValues[index].label : accumulator.lastVisibleLabel
|
|
269
|
+
};
|
|
270
|
+
}, {
|
|
271
|
+
measuredVisibleTags: [],
|
|
272
|
+
measuredHiddenTags: 0,
|
|
273
|
+
accumulatedWidth: 0,
|
|
274
|
+
lastVisibleElementWidth: 0,
|
|
275
|
+
lastVisibleLabel: ""
|
|
276
|
+
});
|
|
277
|
+
const additionnalElementsWidth = SIZES_TAG.paddings + (refPlusTag.current?.offsetWidth ?? 0);
|
|
278
|
+
const finalWidth = accumulatedWidth + (measuredHiddenTags ? additionnalElementsWidth : 0);
|
|
279
|
+
const overflowPx = finalWidth - innerWidth;
|
|
280
|
+
const hasOverflow = overflowPx > 0;
|
|
281
|
+
const hasHiddenTags = measuredHiddenTags > 0;
|
|
282
|
+
const lastVisibleElementMaxSize = lastVisibleElementWidth - overflowPx;
|
|
283
|
+
if (measuredHiddenTags === 1 && measuredVisibleTags.length === 0) {
|
|
284
|
+
setOverflowAmount(0);
|
|
285
|
+
setNonOverFlowedValues([potentiallyNonOverflowedValues[0]]);
|
|
286
|
+
const newOverflowPx = lastVisibleElementWidth + (measuredHiddenTags > 1 ? additionnalElementsWidth : 0) - innerWidth;
|
|
287
|
+
setLastElementMaxWidth(lastVisibleElementWidth - newOverflowPx);
|
|
288
|
+
setOverflow(true);
|
|
289
|
+
} else if (hasOverflow && hasHiddenTags && (lastVisibleElementMaxSize > 65 || measuredVisibleTags.length === 1 && lastVisibleElementMaxSize > 65) && typeof lastVisibleLabel === "string") {
|
|
290
|
+
setLastElementMaxWidth(lastVisibleElementMaxSize);
|
|
291
|
+
setOverflow(true);
|
|
292
|
+
setOverflowAmount(measuredHiddenTags);
|
|
293
|
+
setNonOverFlowedValues(measuredVisibleTags);
|
|
294
|
+
} else if (hasOverflow && hasHiddenTags) {
|
|
295
|
+
setLastElementMaxWidth(0);
|
|
296
|
+
setOverflow(false);
|
|
297
|
+
setOverflowAmount(measuredHiddenTags + 1);
|
|
298
|
+
setNonOverFlowedValues(measuredVisibleTags.slice(0, -1));
|
|
299
|
+
} else {
|
|
300
|
+
setOverflow(false);
|
|
301
|
+
setOverflowAmount(measuredHiddenTags);
|
|
302
|
+
setNonOverFlowedValues(measuredVisibleTags);
|
|
215
303
|
}
|
|
216
304
|
}
|
|
217
|
-
|
|
218
|
-
setOverflowAmount(computedOverflowAmount);
|
|
219
|
-
}, [options, selectedData.selectedValues, width]);
|
|
305
|
+
}, [selectedData.selectedValues.length, innerWidth, potentiallyNonOverflowedValues]);
|
|
220
306
|
useEffect(() => {
|
|
221
307
|
setSelectedData({
|
|
222
308
|
type: "update"
|
|
223
309
|
});
|
|
224
310
|
}, [setSelectedData, options]);
|
|
311
|
+
useEffect(() => {
|
|
312
|
+
const getWidth = () => {
|
|
313
|
+
if (refTag.current) {
|
|
314
|
+
setInnerWidth(refTag.current.offsetWidth);
|
|
315
|
+
} else setInnerWidth(innerRef.current?.offsetWidth ?? 0 - (arrowRef.current?.offsetWidth ?? 0) - SIZES_TAG.paddings);
|
|
316
|
+
};
|
|
317
|
+
getWidth();
|
|
318
|
+
window.addEventListener("resize", getWidth);
|
|
319
|
+
return () => window.removeEventListener("resize", getWidth);
|
|
320
|
+
}, [innerRef, refTag, selectedData.selectedValues]);
|
|
225
321
|
const shouldDisplayValues = useMemo(() => {
|
|
226
322
|
if (multiselect) {
|
|
227
|
-
return
|
|
323
|
+
return potentiallyNonOverflowedValues.length > 0 || selectedData.selectedValues.some((selectedValue) => findOptionInOptions(options, selectedValue) !== void 0);
|
|
228
324
|
}
|
|
229
325
|
return selectedData.selectedValues[0] !== void 0 && findOptionInOptions(options, selectedData.selectedValues[0]) !== void 0;
|
|
230
|
-
}, [multiselect,
|
|
231
|
-
return /* @__PURE__ */ jsx(Tooltip, { text: tooltip, children: /* @__PURE__ */ jsxs(StyledInputWrapper, { role: "combobox", id, "data-disabled": disabled, "data-readonly": readOnly, "data-size": size, "data-dropdownvisible": isDropdownVisible, "data-state": state,
|
|
326
|
+
}, [multiselect, options, potentiallyNonOverflowedValues.length, selectedData.selectedValues]);
|
|
327
|
+
return /* @__PURE__ */ jsx(Tooltip, { text: tooltip, children: /* @__PURE__ */ jsxs(StyledInputWrapper, { role: "combobox", id, "data-disabled": disabled, "data-readonly": readOnly, "data-size": size, "data-dropdownvisible": isDropdownVisible, "data-state": state, onClick: openable ? () => setIsDropdownVisible(!isDropdownVisible) : void 0, "data-testid": dataTestId, autoFocus, onKeyDown: (event) => {
|
|
232
328
|
if (event.key === "ArrowDown") {
|
|
233
329
|
if (!isDropdownVisible) {
|
|
234
330
|
setIsDropdownVisible(true);
|
|
@@ -241,8 +337,8 @@ const SelectBar = ({
|
|
|
241
337
|
}
|
|
242
338
|
return ["Enter", " "].includes(event.key) && openable ? setIsDropdownVisible(!isDropdownVisible) : null;
|
|
243
339
|
}, ref: innerRef, "aria-haspopup": "listbox", "aria-expanded": isDropdownVisible, tabIndex: 0, "aria-label": label, children: [
|
|
244
|
-
shouldDisplayValues ? /* @__PURE__ */ jsx(DisplayValues, { refTag, nonOverflowedValues, disabled, readOnly, overflowed: !!overflowAmount, overflowAmount, size }) : /* @__PURE__ */ jsx(Placeholder, { as: "p", variant: size === "large" ? "body" : "bodySmall", sentiment: "neutral", disabled, prominence: "weak", children: placeholder }),
|
|
245
|
-
/* @__PURE__ */ jsxs(StateStack, { direction: "row", gap: 1, alignItems: "center", children: [
|
|
340
|
+
shouldDisplayValues ? /* @__PURE__ */ jsx(DisplayValues, { refTag, nonOverflowedValues, potentiallyNonOverflowedValues, disabled, readOnly, overflowed: !!overflowAmount, overflowAmount, size, measureRef, lastElementMaxWidth, overflow, refPlusTag }) : /* @__PURE__ */ jsx(Placeholder, { as: "p", variant: size === "large" ? "body" : "bodySmall", sentiment: "neutral", disabled, prominence: "weak", children: placeholder }),
|
|
341
|
+
/* @__PURE__ */ jsxs(StateStack, { direction: "row", gap: 1, alignItems: "center", ref: arrowRef, children: [
|
|
246
342
|
error ? /* @__PURE__ */ jsx(AlertCircleIcon, { sentiment: "danger" }) : null,
|
|
247
343
|
success && !error ? /* @__PURE__ */ jsx(CheckCircleIcon, { sentiment: "success" }) : null,
|
|
248
344
|
clearable && selectedData.selectedValues.length > 0 ? /* @__PURE__ */ jsx(Button, { "aria-label": "clear value", disabled: disabled || !selectedData.selectedValues[0] || readOnly, variant: "ghost", size: "small", onClick: (event) => {
|