@ultraviolet/ui 1.70.1 → 1.70.2
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/AvatarV2/index.cjs +2 -2
- package/dist/components/Button/index.d.ts +6 -6
- package/dist/components/Expandable/index.d.ts +1 -1
- package/dist/components/List/Row.d.ts +1 -1
- package/dist/components/Menu/index.d.ts +6 -6
- package/dist/components/Modal/components/Dialog.d.ts +1 -1
- package/dist/components/Popup/animations.d.ts +2 -2
- package/dist/components/Radio/index.d.ts +1 -1
- package/dist/components/Row/index.d.ts +1 -1
- package/dist/components/SelectInputV2/Dropdown.cjs +41 -31
- package/dist/components/SelectInputV2/Dropdown.js +42 -32
- package/dist/components/SelectInputV2/DropdownOption.cjs +48 -20
- package/dist/components/SelectInputV2/DropdownOption.js +39 -11
- package/dist/components/SelectInputV2/SelectBar.cjs +72 -30
- package/dist/components/SelectInputV2/SelectBar.js +66 -24
- package/dist/components/SelectableCard/index.cjs +111 -15
- package/dist/components/SelectableCard/index.d.ts +9 -1
- package/dist/components/SelectableCard/index.js +94 -16
- package/dist/components/Slider/components/Options.d.ts +1 -1
- package/dist/components/Slider/styles.d.ts +1 -1
- package/dist/components/Stack/index.d.ts +1 -1
- package/dist/components/Tabs/Tab.d.ts +1 -1
- package/dist/components/TextInputV2/index.d.ts +2 -2
- package/dist/types.d.ts +7 -0
- package/dist/utils/animations.d.ts +27 -27
- package/package.json +2 -2
|
@@ -6,9 +6,9 @@ const legacy = require("@ultraviolet/icons/legacy");
|
|
|
6
6
|
const React = require("react");
|
|
7
7
|
const index$4 = require("../Button/index.cjs");
|
|
8
8
|
const index = require("../Stack/index.cjs");
|
|
9
|
-
const index$
|
|
10
|
-
const index$
|
|
11
|
-
const index$
|
|
9
|
+
const index$2 = require("../Tag/index.cjs");
|
|
10
|
+
const index$1 = require("../Text/index.cjs");
|
|
11
|
+
const index$3 = require("../Tooltip/index.cjs");
|
|
12
12
|
const SelectInputProvider = require("./SelectInputProvider.cjs");
|
|
13
13
|
const findOptionInOptions = require("./findOptionInOptions.cjs");
|
|
14
14
|
const types = require("./types.cjs");
|
|
@@ -18,17 +18,31 @@ function _EMOTION_STRINGIFIED_CSS_ERROR__() {
|
|
|
18
18
|
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).";
|
|
19
19
|
}
|
|
20
20
|
const StateStack = /* @__PURE__ */ _styled__default.default(index.Stack, process.env.NODE_ENV === "production" ? {
|
|
21
|
-
target: "
|
|
21
|
+
target: "ejy0aca4"
|
|
22
22
|
} : {
|
|
23
|
-
target: "
|
|
23
|
+
target: "ejy0aca4",
|
|
24
24
|
label: "StateStack"
|
|
25
25
|
})("padding-right:", ({
|
|
26
26
|
theme
|
|
27
|
-
}) => 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":"AAwCgC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/SelectBar.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { Icon } from '@ultraviolet/icons/legacy'\nimport type { RefObject } from 'react'\nimport { useEffect, useMemo, useRef, useState } from 'react'\nimport { Button } from '../Button'\nimport { Stack } from '../Stack'\nimport { Tag } from '../Tag'\nimport { Text } from '../Text'\nimport { Tooltip } from '../Tooltip'\nimport { useSelectInput } from './SelectInputProvider'\nimport { findOptionInOptions } from './findOptionInOptions'\nimport type { DataType, OptionType } from './types'\nimport { INPUT_SIZE_HEIGHT, SIZES_TAG } from './types'\n\ntype SelectBarProps = {\n  size: 'small' | 'medium' | 'large'\n  clearable: boolean\n  disabled: boolean\n  readOnly: boolean\n  placeholder: string\n  success?: string\n  error?: string\n  autoFocus?: boolean\n  innerRef: RefObject<HTMLDivElement>\n  id?: string\n  'data-testid': string\n  label?: string\n  tooltip?: string\n}\n\ntype DisplayValuesProps = {\n  refTag: RefObject<HTMLDivElement>\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`\n\nconst StyledInputWrapper = styled(Stack)<{\n  'data-readonly': boolean\n  'data-disabled': boolean\n  'data-size': 'small' | 'medium' | 'large'\n  'data-state': 'neutral' | 'success' | 'danger'\n  'data-dropdownvisible': boolean\n  'aria-label'?: string\n}>`\n  display: flex;\n  padding: ${({ theme }) => theme.space[1]};\n  padding-right: 0;\n  padding-left: ${({ theme }) => theme.space[2]};\n  cursor: pointer;\n  box-shadow: none;\n  background: ${({ theme }) => theme.colors.neutral.background};\n  border-radius: ${({ theme }) => theme.radii.default};\n  width: 100%;\n\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  &[data-size='small'] {\n    height: ${INPUT_SIZE_HEIGHT.small}px;\n    padding-left: ${({ theme }) => theme.space[1]};\n  }\n  &[data-size='medium'] {\n    height: ${INPUT_SIZE_HEIGHT.medium}px;\n  }\n  &[data-size='large'] {\n    height: ${INPUT_SIZE_HEIGHT.large}px;\n  }\n  &[data-state='neutral'] {\n    border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  }\n  &[data-state='success'] {\n    border: 1px solid ${({ theme }) => theme.colors.success.border};\n  }\n  &[data-state='danger'] {\n    border: 1px solid ${({ theme }) => theme.colors.danger.border};\n  }\n\n  &:not([data-disabled='true']):not([data-readonly]):hover {\n    border-color: ${({ theme }) => theme.colors.primary.border};\n  }\n\n  &:not([data-disabled='true']):not([data-readonly]):active {\n    box-shadow: ${({ theme }) => theme.shadows.focusPrimary};\n  }\n\n  &:not([data-disabled='true']):hover,\n  :not([data-disabled='true']):focus {\n    border-color: ${({ theme }) => theme.colors.primary.borderHover};\n    outline: none;\n  }\n\n  &[data-dropdownvisible='true'] {\n    border-color: ${({ theme }) => theme.colors.primary.borderHover};\n  }\n`\nconst CustomTag = styled(Tag)`\n  height: fit-content;\n  width: fit-content;\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          <Icon name=\"plus\" />\n          {overflowAmount}\n        </Tag>\n      ) : null}\n    </Stack>\n  ) : (\n    <Text as=\"p\" variant={size === 'large' ? 'body' : 'bodySmall'}>\n      {selectedData.selectedValues[0]\n        ? findOptionInOptions(options, selectedData.selectedValues[0])?.label\n        : null}\n    </Text>\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 [overflowed, setOverflowed] = useState(false)\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          setOverflowAmount(computedOverflowAmount)\n        } else {\n          computedNonOverflowedValues = [\n            ...computedNonOverflowedValues,\n            selectedOption,\n          ]\n          setNonOverFlowedValues(computedNonOverflowedValues)\n          tagsWidth += totalTagWidth\n        }\n      }\n    }\n    if (computedOverflowAmount === 0) {\n      setOverflowed(false)\n    } else {\n      setOverflowed(true)\n    }\n    setOverflowAmount(computedOverflowAmount)\n  }, [options, selectedData.selectedValues, width])\n\n  useEffect(() => {\n    setSelectedData({ type: 'update' })\n  }, [setSelectedData, options])\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\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        {selectedData.selectedValues.length > 0 ? (\n          <DisplayValues\n            refTag={refTag}\n            nonOverflowedValues={nonOverflowedValues}\n            disabled={disabled}\n            readOnly={readOnly}\n            overflowed={overflowed}\n            overflowAmount={overflowAmount}\n            size={size}\n          />\n        ) : (\n          <Text\n            as=\"p\"\n            variant={size === 'large' ? 'body' : 'bodySmall'}\n            sentiment=\"neutral\"\n            disabled={disabled}\n            prominence=\"weak\"\n          >\n            {placeholder}\n          </Text>\n        )}\n        <StateStack direction=\"row\" gap={1} alignItems=\"center\">\n          {error ? <Icon name=\"alert\" sentiment=\"danger\" /> : null}\n          {success && !error ? (\n            <Icon name=\"checkbox-circle-outline\" sentiment=\"success\" />\n          ) : null}\n          {clearable && selectedData.selectedValues.length > 0 ? (\n            <Button\n              aria-label=\"clear value\"\n              disabled={\n                disabled || ![selectedData.selectedValues[0]] || readOnly\n              }\n              variant=\"ghost\"\n              size=\"small\"\n              icon=\"close\"\n              onClick={event => {\n                event.stopPropagation()\n                setSelectedData({ type: 'clearAll' })\n                if (multiselect) {\n                  onChange?.([])\n                } else {\n                  onChange?.('')\n                }\n              }}\n              sentiment=\"neutral\"\n              data-testid=\"clear-all\"\n            />\n          ) : null}\n          <Icon\n            aria-label=\"show dropdown\"\n            size=\"small\"\n            name=\"arrow-down\"\n            sentiment=\"neutral\"\n            disabled={disabled || readOnly}\n          />\n        </StateStack>\n      </StyledInputWrapper>\n    </Tooltip>\n  )\n}\n"]} */"));
|
|
27
|
+
}) => 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":"AAwCgC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/SelectBar.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { Icon } from '@ultraviolet/icons/legacy'\nimport type { RefObject } from 'react'\nimport { useEffect, useMemo, useRef, useState } from 'react'\nimport { Button } from '../Button'\nimport { Stack } from '../Stack'\nimport { Tag } from '../Tag'\nimport { Text } from '../Text'\nimport { Tooltip } from '../Tooltip'\nimport { useSelectInput } from './SelectInputProvider'\nimport { findOptionInOptions } from './findOptionInOptions'\nimport type { DataType, OptionType } from './types'\nimport { INPUT_SIZE_HEIGHT, SIZES_TAG } from './types'\n\ntype SelectBarProps = {\n  size: 'small' | 'medium' | 'large'\n  clearable: boolean\n  disabled: boolean\n  readOnly: boolean\n  placeholder: string\n  success?: string\n  error?: string\n  autoFocus?: boolean\n  innerRef: RefObject<HTMLDivElement>\n  id?: string\n  'data-testid': string\n  label?: string\n  tooltip?: string\n}\n\ntype DisplayValuesProps = {\n  refTag: RefObject<HTMLDivElement>\n  nonOverflowedValues: OptionType[]\n  disabled: boolean\n  readOnly: boolean\n  overflowed: boolean\n  overflowAmount: number\n  size: 'small' | 'medium' | 'large'\n}\n\nconst StateStack = styled(Stack)`\n  padding-right: ${({ theme }) => theme.space['2']};\n  display: flex;\n`\nconst Placeholder = styled(Text)`\nuser-select: none;\n`\n\nconst StyledInputWrapper = styled(Stack)<{\n  'data-readonly': boolean\n  'data-disabled': boolean\n  'data-size': 'small' | 'medium' | 'large'\n  'data-state': 'neutral' | 'success' | 'danger'\n  'data-dropdownvisible': boolean\n  'aria-label'?: string\n}>`\n  display: flex;\n  padding: ${({ theme }) => theme.space[1]};\n  padding-right: 0;\n  padding-left: ${({ theme }) => theme.space[2]};\n  cursor: pointer;\n  box-shadow: none;\n  background: ${({ theme }) => theme.colors.neutral.background};\n  border-radius: ${({ theme }) => theme.radii.default};\n  width: 100%;\n  overflow: hidden;\n\n  &[data-size='small'] {\n    height: ${INPUT_SIZE_HEIGHT.small}px;\n    padding-left: ${({ theme }) => theme.space[1]};\n  }\n  &[data-size='medium'] {\n    height: ${INPUT_SIZE_HEIGHT.medium}px;\n  }\n  &[data-size='large'] {\n    height: ${INPUT_SIZE_HEIGHT.large}px;\n  }\n  &[data-state='neutral'] {\n    border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n\n    &:not([data-disabled=\"true\"]):not([data-readonly=\"true\"]):active {\n      border-color: ${({ theme }) => theme.colors.primary.borderHover};\n      box-shadow: ${({ theme }) => theme.shadows.focusPrimary};\n    }\n    &:not([data-disabled='true']):hover,\n    :not([data-disabled='true']):focus {\n      border-color: ${({ theme }) => theme.colors.primary.borderHover};\n      outline: none;\n    }\n\n    &[data-dropdownvisible='true'] {\n    border-color: ${({ theme }) => theme.colors.primary.borderHover};\n  }\n  }\n\n  &[data-state='success'] {\n    border: 1px solid ${({ theme }) => theme.colors.success.border};\n    &:not([data-disabled=\"true\"]):not([data-readonly=\"true\"]):active {\n      border-color: ${({ theme }) => theme.colors.success.borderHover};\n      box-shadow: ${({ theme }) => theme.shadows.focusSuccess};\n    }\n  \n    &[data-dropdownvisible='true'] {\n      border-color: ${({ theme }) => theme.colors.success.borderHover};\n    }\n  }\n\n  &[data-state='danger'] {\n    border: 1px solid ${({ theme }) => theme.colors.danger.border};\n\n    &:not([data-disabled=\"true\"]):not([data-readonly=\"true\"]):active {\n      border-color: ${({ theme }) => theme.colors.danger.borderHover};\n      box-shadow: ${({ theme }) => theme.shadows.focusDanger};\n    }\n  \n    &[data-dropdownvisible='true'] {\n      border-color: ${({ theme }) => theme.colors.danger.borderHover};\n    }\n  }\n\n  &:not([data-disabled='true']):not([data-readonly]):hover {\n    border-color: ${({ theme }) => theme.colors.primary.border};\n  }\n\n  &[data-readonly='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n    border-color: ${({ theme }) => theme.colors.neutral.border};\n    cursor: default;\n  }\n\n  &[data-disabled='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    border-color: ${({ theme }) => theme.colors.neutral.borderDisabled};\n    cursor: not-allowed;\n  }\n`\nconst CustomTag = styled(Tag)`\n  height: fit-content;\n  width: fit-content;\n`\nconst SelectedValues = styled(Text)`\ntext-overflow: ellipsis;\noverflow: hidden; \n`\n\nconst isValidSelectedValue = (selectedValue: string, options: DataType) =>\n  !Array.isArray(options)\n    ? Object.keys(options).some(group =>\n        options[group].some(\n          option => option.value === selectedValue && !option.disabled,\n        ),\n      )\n    : options.some(option => option.value === selectedValue && !option.disabled)\n\nconst DisplayValues = ({\n  refTag,\n  nonOverflowedValues,\n  disabled,\n  readOnly,\n  overflowed,\n  overflowAmount,\n  size,\n}: DisplayValuesProps) => {\n  const { multiselect, selectedData, setSelectedData, options, onChange } =\n    useSelectInput()\n\n  return multiselect ? (\n    <Stack\n      direction=\"row\"\n      gap=\"1\"\n      wrap=\"nowrap\"\n      ref={refTag}\n      alignItems=\"center\"\n    >\n      {nonOverflowedValues.map(option => (\n        <CustomTag\n          data-testid=\"selected-options-tags\"\n          key={option?.value}\n          sentiment=\"neutral\"\n          disabled={disabled}\n          onClose={\n            !readOnly\n              ? event => {\n                  event.stopPropagation()\n                  setSelectedData({\n                    type: 'selectOption',\n                    clickedOption: option,\n                  })\n                  const newSelectedValues = selectedData.selectedValues?.filter(\n                    val => val !== option.value,\n                  )\n                  onChange?.(newSelectedValues)\n                }\n              : undefined\n          }\n        >\n          {option?.label}\n        </CustomTag>\n      ))}\n      {overflowed ? (\n        <Tag\n          sentiment=\"neutral\"\n          disabled={disabled}\n          key=\"+\"\n          data-testid=\"plus-tag\"\n          aria-label=\"Plus tag\"\n        >\n          <Icon name=\"plus\" />\n          {overflowAmount}\n        </Tag>\n      ) : null}\n    </Stack>\n  ) : (\n    <SelectedValues as=\"div\" variant={size === 'large' ? 'body' : 'bodySmall'}>\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 [overflowed, setOverflowed] = useState(false)\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          setOverflowAmount(computedOverflowAmount)\n        } else {\n          computedNonOverflowedValues = [\n            ...computedNonOverflowedValues,\n            selectedOption,\n          ]\n          setNonOverFlowedValues(computedNonOverflowedValues)\n          tagsWidth += totalTagWidth\n        }\n      }\n    }\n    if (computedOverflowAmount === 0) {\n      setOverflowed(false)\n    } else {\n      setOverflowed(true)\n    }\n    setOverflowAmount(computedOverflowAmount)\n  }, [options, selectedData.selectedValues, width])\n\n  useEffect(() => {\n    setSelectedData({ type: 'update' })\n  }, [setSelectedData, options])\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\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        {selectedData.selectedValues.length > 0 ? (\n          <DisplayValues\n            refTag={refTag}\n            nonOverflowedValues={nonOverflowedValues}\n            disabled={disabled}\n            readOnly={readOnly}\n            overflowed={overflowed}\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 ? <Icon name=\"alert\" sentiment=\"danger\" /> : null}\n          {success && !error ? (\n            <Icon name=\"checkbox-circle-outline\" sentiment=\"success\" />\n          ) : null}\n          {clearable && selectedData.selectedValues.length > 0 ? (\n            <Button\n              aria-label=\"clear value\"\n              disabled={disabled || !selectedData.selectedValues[0] || readOnly}\n              variant=\"ghost\"\n              size=\"small\"\n              icon=\"close\"\n              onClick={event => {\n                event.stopPropagation()\n                setSelectedData({ type: 'clearAll' })\n                if (multiselect) {\n                  onChange?.([])\n                } else {\n                  onChange?.('')\n                }\n              }}\n              sentiment=\"neutral\"\n              data-testid=\"clear-all\"\n            />\n          ) : null}\n          <Icon\n            aria-label=\"show dropdown\"\n            size=\"small\"\n            name=\"arrow-down\"\n            sentiment=\"neutral\"\n            disabled={disabled || readOnly}\n          />\n        </StateStack>\n      </StyledInputWrapper>\n    </Tooltip>\n  )\n}\n"]} */"));
|
|
28
|
+
const Placeholder = /* @__PURE__ */ _styled__default.default(index$1.Text, process.env.NODE_ENV === "production" ? {
|
|
29
|
+
target: "ejy0aca3"
|
|
30
|
+
} : {
|
|
31
|
+
target: "ejy0aca3",
|
|
32
|
+
label: "Placeholder"
|
|
33
|
+
})(process.env.NODE_ENV === "production" ? {
|
|
34
|
+
name: "oldbq4",
|
|
35
|
+
styles: "user-select:none"
|
|
36
|
+
} : {
|
|
37
|
+
name: "oldbq4",
|
|
38
|
+
styles: "user-select:none",
|
|
39
|
+
map: "/*# 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":"AA4CgC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/SelectBar.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { Icon } from '@ultraviolet/icons/legacy'\nimport type { RefObject } from 'react'\nimport { useEffect, useMemo, useRef, useState } from 'react'\nimport { Button } from '../Button'\nimport { Stack } from '../Stack'\nimport { Tag } from '../Tag'\nimport { Text } from '../Text'\nimport { Tooltip } from '../Tooltip'\nimport { useSelectInput } from './SelectInputProvider'\nimport { findOptionInOptions } from './findOptionInOptions'\nimport type { DataType, OptionType } from './types'\nimport { INPUT_SIZE_HEIGHT, SIZES_TAG } from './types'\n\ntype SelectBarProps = {\n  size: 'small' | 'medium' | 'large'\n  clearable: boolean\n  disabled: boolean\n  readOnly: boolean\n  placeholder: string\n  success?: string\n  error?: string\n  autoFocus?: boolean\n  innerRef: RefObject<HTMLDivElement>\n  id?: string\n  'data-testid': string\n  label?: string\n  tooltip?: string\n}\n\ntype DisplayValuesProps = {\n  refTag: RefObject<HTMLDivElement>\n  nonOverflowedValues: OptionType[]\n  disabled: boolean\n  readOnly: boolean\n  overflowed: boolean\n  overflowAmount: number\n  size: 'small' | 'medium' | 'large'\n}\n\nconst StateStack = styled(Stack)`\n  padding-right: ${({ theme }) => theme.space['2']};\n  display: flex;\n`\nconst Placeholder = styled(Text)`\nuser-select: none;\n`\n\nconst StyledInputWrapper = styled(Stack)<{\n  'data-readonly': boolean\n  'data-disabled': boolean\n  'data-size': 'small' | 'medium' | 'large'\n  'data-state': 'neutral' | 'success' | 'danger'\n  'data-dropdownvisible': boolean\n  'aria-label'?: string\n}>`\n  display: flex;\n  padding: ${({ theme }) => theme.space[1]};\n  padding-right: 0;\n  padding-left: ${({ theme }) => theme.space[2]};\n  cursor: pointer;\n  box-shadow: none;\n  background: ${({ theme }) => theme.colors.neutral.background};\n  border-radius: ${({ theme }) => theme.radii.default};\n  width: 100%;\n  overflow: hidden;\n\n  &[data-size='small'] {\n    height: ${INPUT_SIZE_HEIGHT.small}px;\n    padding-left: ${({ theme }) => theme.space[1]};\n  }\n  &[data-size='medium'] {\n    height: ${INPUT_SIZE_HEIGHT.medium}px;\n  }\n  &[data-size='large'] {\n    height: ${INPUT_SIZE_HEIGHT.large}px;\n  }\n  &[data-state='neutral'] {\n    border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n\n    &:not([data-disabled=\"true\"]):not([data-readonly=\"true\"]):active {\n      border-color: ${({ theme }) => theme.colors.primary.borderHover};\n      box-shadow: ${({ theme }) => theme.shadows.focusPrimary};\n    }\n    &:not([data-disabled='true']):hover,\n    :not([data-disabled='true']):focus {\n      border-color: ${({ theme }) => theme.colors.primary.borderHover};\n      outline: none;\n    }\n\n    &[data-dropdownvisible='true'] {\n    border-color: ${({ theme }) => theme.colors.primary.borderHover};\n  }\n  }\n\n  &[data-state='success'] {\n    border: 1px solid ${({ theme }) => theme.colors.success.border};\n    &:not([data-disabled=\"true\"]):not([data-readonly=\"true\"]):active {\n      border-color: ${({ theme }) => theme.colors.success.borderHover};\n      box-shadow: ${({ theme }) => theme.shadows.focusSuccess};\n    }\n  \n    &[data-dropdownvisible='true'] {\n      border-color: ${({ theme }) => theme.colors.success.borderHover};\n    }\n  }\n\n  &[data-state='danger'] {\n    border: 1px solid ${({ theme }) => theme.colors.danger.border};\n\n    &:not([data-disabled=\"true\"]):not([data-readonly=\"true\"]):active {\n      border-color: ${({ theme }) => theme.colors.danger.borderHover};\n      box-shadow: ${({ theme }) => theme.shadows.focusDanger};\n    }\n  \n    &[data-dropdownvisible='true'] {\n      border-color: ${({ theme }) => theme.colors.danger.borderHover};\n    }\n  }\n\n  &:not([data-disabled='true']):not([data-readonly]):hover {\n    border-color: ${({ theme }) => theme.colors.primary.border};\n  }\n\n  &[data-readonly='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n    border-color: ${({ theme }) => theme.colors.neutral.border};\n    cursor: default;\n  }\n\n  &[data-disabled='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    border-color: ${({ theme }) => theme.colors.neutral.borderDisabled};\n    cursor: not-allowed;\n  }\n`\nconst CustomTag = styled(Tag)`\n  height: fit-content;\n  width: fit-content;\n`\nconst SelectedValues = styled(Text)`\ntext-overflow: ellipsis;\noverflow: hidden; \n`\n\nconst isValidSelectedValue = (selectedValue: string, options: DataType) =>\n  !Array.isArray(options)\n    ? Object.keys(options).some(group =>\n        options[group].some(\n          option => option.value === selectedValue && !option.disabled,\n        ),\n      )\n    : options.some(option => option.value === selectedValue && !option.disabled)\n\nconst DisplayValues = ({\n  refTag,\n  nonOverflowedValues,\n  disabled,\n  readOnly,\n  overflowed,\n  overflowAmount,\n  size,\n}: DisplayValuesProps) => {\n  const { multiselect, selectedData, setSelectedData, options, onChange } =\n    useSelectInput()\n\n  return multiselect ? (\n    <Stack\n      direction=\"row\"\n      gap=\"1\"\n      wrap=\"nowrap\"\n      ref={refTag}\n      alignItems=\"center\"\n    >\n      {nonOverflowedValues.map(option => (\n        <CustomTag\n          data-testid=\"selected-options-tags\"\n          key={option?.value}\n          sentiment=\"neutral\"\n          disabled={disabled}\n          onClose={\n            !readOnly\n              ? event => {\n                  event.stopPropagation()\n                  setSelectedData({\n                    type: 'selectOption',\n                    clickedOption: option,\n                  })\n                  const newSelectedValues = selectedData.selectedValues?.filter(\n                    val => val !== option.value,\n                  )\n                  onChange?.(newSelectedValues)\n                }\n              : undefined\n          }\n        >\n          {option?.label}\n        </CustomTag>\n      ))}\n      {overflowed ? (\n        <Tag\n          sentiment=\"neutral\"\n          disabled={disabled}\n          key=\"+\"\n          data-testid=\"plus-tag\"\n          aria-label=\"Plus tag\"\n        >\n          <Icon name=\"plus\" />\n          {overflowAmount}\n        </Tag>\n      ) : null}\n    </Stack>\n  ) : (\n    <SelectedValues as=\"div\" variant={size === 'large' ? 'body' : 'bodySmall'}>\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 [overflowed, setOverflowed] = useState(false)\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          setOverflowAmount(computedOverflowAmount)\n        } else {\n          computedNonOverflowedValues = [\n            ...computedNonOverflowedValues,\n            selectedOption,\n          ]\n          setNonOverFlowedValues(computedNonOverflowedValues)\n          tagsWidth += totalTagWidth\n        }\n      }\n    }\n    if (computedOverflowAmount === 0) {\n      setOverflowed(false)\n    } else {\n      setOverflowed(true)\n    }\n    setOverflowAmount(computedOverflowAmount)\n  }, [options, selectedData.selectedValues, width])\n\n  useEffect(() => {\n    setSelectedData({ type: 'update' })\n  }, [setSelectedData, options])\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\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        {selectedData.selectedValues.length > 0 ? (\n          <DisplayValues\n            refTag={refTag}\n            nonOverflowedValues={nonOverflowedValues}\n            disabled={disabled}\n            readOnly={readOnly}\n            overflowed={overflowed}\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 ? <Icon name=\"alert\" sentiment=\"danger\" /> : null}\n          {success && !error ? (\n            <Icon name=\"checkbox-circle-outline\" sentiment=\"success\" />\n          ) : null}\n          {clearable && selectedData.selectedValues.length > 0 ? (\n            <Button\n              aria-label=\"clear value\"\n              disabled={disabled || !selectedData.selectedValues[0] || readOnly}\n              variant=\"ghost\"\n              size=\"small\"\n              icon=\"close\"\n              onClick={event => {\n                event.stopPropagation()\n                setSelectedData({ type: 'clearAll' })\n                if (multiselect) {\n                  onChange?.([])\n                } else {\n                  onChange?.('')\n                }\n              }}\n              sentiment=\"neutral\"\n              data-testid=\"clear-all\"\n            />\n          ) : null}\n          <Icon\n            aria-label=\"show dropdown\"\n            size=\"small\"\n            name=\"arrow-down\"\n            sentiment=\"neutral\"\n            disabled={disabled || readOnly}\n          />\n        </StateStack>\n      </StyledInputWrapper>\n    </Tooltip>\n  )\n}\n"]} */",
|
|
40
|
+
toString: _EMOTION_STRINGIFIED_CSS_ERROR__
|
|
41
|
+
});
|
|
28
42
|
const StyledInputWrapper = /* @__PURE__ */ _styled__default.default(index.Stack, process.env.NODE_ENV === "production" ? {
|
|
29
|
-
target: "
|
|
43
|
+
target: "ejy0aca2"
|
|
30
44
|
} : {
|
|
31
|
-
target: "
|
|
45
|
+
target: "ejy0aca2",
|
|
32
46
|
label: "StyledInputWrapper"
|
|
33
47
|
})("display:flex;padding:", ({
|
|
34
48
|
theme
|
|
@@ -38,35 +52,49 @@ const StyledInputWrapper = /* @__PURE__ */ _styled__default.default(index.Stack,
|
|
|
38
52
|
theme
|
|
39
53
|
}) => theme.colors.neutral.background, ";border-radius:", ({
|
|
40
54
|
theme
|
|
41
|
-
}) => theme.radii.default, ";width:100
|
|
55
|
+
}) => theme.radii.default, ";width:100%;overflow:hidden;&[data-size='small']{height:", types.INPUT_SIZE_HEIGHT.small, "px;padding-left:", ({
|
|
42
56
|
theme
|
|
43
|
-
}) => theme.
|
|
57
|
+
}) => theme.space[1], ";}&[data-size='medium']{height:", types.INPUT_SIZE_HEIGHT.medium, "px;}&[data-size='large']{height:", types.INPUT_SIZE_HEIGHT.large, "px;}&[data-state='neutral']{border:1px solid ", ({
|
|
44
58
|
theme
|
|
45
|
-
}) => theme.colors.neutral.border, "
|
|
59
|
+
}) => theme.colors.neutral.border, ';&:not([data-disabled="true"]):not([data-readonly="true"]):active{border-color:', ({
|
|
46
60
|
theme
|
|
47
|
-
}) => theme.colors.
|
|
61
|
+
}) => theme.colors.primary.borderHover, ";box-shadow:", ({
|
|
48
62
|
theme
|
|
49
|
-
}) => theme.
|
|
63
|
+
}) => theme.shadows.focusPrimary, ";}&:not([data-disabled='true']):hover,:not([data-disabled='true']):focus{border-color:", ({
|
|
50
64
|
theme
|
|
51
|
-
}) => theme.
|
|
65
|
+
}) => theme.colors.primary.borderHover, ";outline:none;}&[data-dropdownvisible='true']{border-color:", ({
|
|
52
66
|
theme
|
|
53
|
-
}) => theme.colors.
|
|
67
|
+
}) => theme.colors.primary.borderHover, ";}}&[data-state='success']{border:1px solid ", ({
|
|
54
68
|
theme
|
|
55
|
-
}) => theme.colors.success.border, "
|
|
69
|
+
}) => theme.colors.success.border, ';&:not([data-disabled="true"]):not([data-readonly="true"]):active{border-color:', ({
|
|
56
70
|
theme
|
|
57
|
-
}) => theme.colors.
|
|
71
|
+
}) => theme.colors.success.borderHover, ";box-shadow:", ({
|
|
58
72
|
theme
|
|
59
|
-
}) => theme.
|
|
73
|
+
}) => theme.shadows.focusSuccess, ";}&[data-dropdownvisible='true']{border-color:", ({
|
|
60
74
|
theme
|
|
61
|
-
}) => theme.
|
|
75
|
+
}) => theme.colors.success.borderHover, ";}}&[data-state='danger']{border:1px solid ", ({
|
|
62
76
|
theme
|
|
63
|
-
}) => theme.colors.
|
|
77
|
+
}) => theme.colors.danger.border, ';&:not([data-disabled="true"]):not([data-readonly="true"]):active{border-color:', ({
|
|
64
78
|
theme
|
|
65
|
-
}) => theme.colors.primary.borderHover, ";}" + (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":"AAoDE","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/SelectBar.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { Icon } from '@ultraviolet/icons/legacy'\nimport type { RefObject } from 'react'\nimport { useEffect, useMemo, useRef, useState } from 'react'\nimport { Button } from '../Button'\nimport { Stack } from '../Stack'\nimport { Tag } from '../Tag'\nimport { Text } from '../Text'\nimport { Tooltip } from '../Tooltip'\nimport { useSelectInput } from './SelectInputProvider'\nimport { findOptionInOptions } from './findOptionInOptions'\nimport type { DataType, OptionType } from './types'\nimport { INPUT_SIZE_HEIGHT, SIZES_TAG } from './types'\n\ntype SelectBarProps = {\n  size: 'small' | 'medium' | 'large'\n  clearable: boolean\n  disabled: boolean\n  readOnly: boolean\n  placeholder: string\n  success?: string\n  error?: string\n  autoFocus?: boolean\n  innerRef: RefObject<HTMLDivElement>\n  id?: string\n  'data-testid': string\n  label?: string\n  tooltip?: string\n}\n\ntype DisplayValuesProps = {\n  refTag: RefObject<HTMLDivElement>\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`\n\nconst StyledInputWrapper = styled(Stack)<{\n  'data-readonly': boolean\n  'data-disabled': boolean\n  'data-size': 'small' | 'medium' | 'large'\n  'data-state': 'neutral' | 'success' | 'danger'\n  'data-dropdownvisible': boolean\n  'aria-label'?: string\n}>`\n  display: flex;\n  padding: ${({ theme }) => theme.space[1]};\n  padding-right: 0;\n  padding-left: ${({ theme }) => theme.space[2]};\n  cursor: pointer;\n  box-shadow: none;\n  background: ${({ theme }) => theme.colors.neutral.background};\n  border-radius: ${({ theme }) => theme.radii.default};\n  width: 100%;\n\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  &[data-size='small'] {\n    height: ${INPUT_SIZE_HEIGHT.small}px;\n    padding-left: ${({ theme }) => theme.space[1]};\n  }\n  &[data-size='medium'] {\n    height: ${INPUT_SIZE_HEIGHT.medium}px;\n  }\n  &[data-size='large'] {\n    height: ${INPUT_SIZE_HEIGHT.large}px;\n  }\n  &[data-state='neutral'] {\n    border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  }\n  &[data-state='success'] {\n    border: 1px solid ${({ theme }) => theme.colors.success.border};\n  }\n  &[data-state='danger'] {\n    border: 1px solid ${({ theme }) => theme.colors.danger.border};\n  }\n\n  &:not([data-disabled='true']):not([data-readonly]):hover {\n    border-color: ${({ theme }) => theme.colors.primary.border};\n  }\n\n  &:not([data-disabled='true']):not([data-readonly]):active {\n    box-shadow: ${({ theme }) => theme.shadows.focusPrimary};\n  }\n\n  &:not([data-disabled='true']):hover,\n  :not([data-disabled='true']):focus {\n    border-color: ${({ theme }) => theme.colors.primary.borderHover};\n    outline: none;\n  }\n\n  &[data-dropdownvisible='true'] {\n    border-color: ${({ theme }) => theme.colors.primary.borderHover};\n  }\n`\nconst CustomTag = styled(Tag)`\n  height: fit-content;\n  width: fit-content;\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          <Icon name=\"plus\" />\n          {overflowAmount}\n        </Tag>\n      ) : null}\n    </Stack>\n  ) : (\n    <Text as=\"p\" variant={size === 'large' ? 'body' : 'bodySmall'}>\n      {selectedData.selectedValues[0]\n        ? findOptionInOptions(options, selectedData.selectedValues[0])?.label\n        : null}\n    </Text>\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 [overflowed, setOverflowed] = useState(false)\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          setOverflowAmount(computedOverflowAmount)\n        } else {\n          computedNonOverflowedValues = [\n            ...computedNonOverflowedValues,\n            selectedOption,\n          ]\n          setNonOverFlowedValues(computedNonOverflowedValues)\n          tagsWidth += totalTagWidth\n        }\n      }\n    }\n    if (computedOverflowAmount === 0) {\n      setOverflowed(false)\n    } else {\n      setOverflowed(true)\n    }\n    setOverflowAmount(computedOverflowAmount)\n  }, [options, selectedData.selectedValues, width])\n\n  useEffect(() => {\n    setSelectedData({ type: 'update' })\n  }, [setSelectedData, options])\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\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        {selectedData.selectedValues.length > 0 ? (\n          <DisplayValues\n            refTag={refTag}\n            nonOverflowedValues={nonOverflowedValues}\n            disabled={disabled}\n            readOnly={readOnly}\n            overflowed={overflowed}\n            overflowAmount={overflowAmount}\n            size={size}\n          />\n        ) : (\n          <Text\n            as=\"p\"\n            variant={size === 'large' ? 'body' : 'bodySmall'}\n            sentiment=\"neutral\"\n            disabled={disabled}\n            prominence=\"weak\"\n          >\n            {placeholder}\n          </Text>\n        )}\n        <StateStack direction=\"row\" gap={1} alignItems=\"center\">\n          {error ? <Icon name=\"alert\" sentiment=\"danger\" /> : null}\n          {success && !error ? (\n            <Icon name=\"checkbox-circle-outline\" sentiment=\"success\" />\n          ) : null}\n          {clearable && selectedData.selectedValues.length > 0 ? (\n            <Button\n              aria-label=\"clear value\"\n              disabled={\n                disabled || ![selectedData.selectedValues[0]] || readOnly\n              }\n              variant=\"ghost\"\n              size=\"small\"\n              icon=\"close\"\n              onClick={event => {\n                event.stopPropagation()\n                setSelectedData({ type: 'clearAll' })\n                if (multiselect) {\n                  onChange?.([])\n                } else {\n                  onChange?.('')\n                }\n              }}\n              sentiment=\"neutral\"\n              data-testid=\"clear-all\"\n            />\n          ) : null}\n          <Icon\n            aria-label=\"show dropdown\"\n            size=\"small\"\n            name=\"arrow-down\"\n            sentiment=\"neutral\"\n            disabled={disabled || readOnly}\n          />\n        </StateStack>\n      </StyledInputWrapper>\n    </Tooltip>\n  )\n}\n"]} */"));
|
|
66
|
-
|
|
67
|
-
|
|
79
|
+
}) => theme.colors.danger.borderHover, ";box-shadow:", ({
|
|
80
|
+
theme
|
|
81
|
+
}) => theme.shadows.focusDanger, ";}&[data-dropdownvisible='true']{border-color:", ({
|
|
82
|
+
theme
|
|
83
|
+
}) => theme.colors.danger.borderHover, ";}}&:not([data-disabled='true']):not([data-readonly]):hover{border-color:", ({
|
|
84
|
+
theme
|
|
85
|
+
}) => theme.colors.primary.border, ";}&[data-readonly='true']{background:", ({
|
|
86
|
+
theme
|
|
87
|
+
}) => theme.colors.neutral.backgroundWeak, ";border-color:", ({
|
|
88
|
+
theme
|
|
89
|
+
}) => theme.colors.neutral.border, ";cursor:default;}&[data-disabled='true']{background:", ({
|
|
90
|
+
theme
|
|
91
|
+
}) => theme.colors.neutral.backgroundDisabled, ";border-color:", ({
|
|
92
|
+
theme
|
|
93
|
+
}) => 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":"AAuDE","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/SelectBar.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { Icon } from '@ultraviolet/icons/legacy'\nimport type { RefObject } from 'react'\nimport { useEffect, useMemo, useRef, useState } from 'react'\nimport { Button } from '../Button'\nimport { Stack } from '../Stack'\nimport { Tag } from '../Tag'\nimport { Text } from '../Text'\nimport { Tooltip } from '../Tooltip'\nimport { useSelectInput } from './SelectInputProvider'\nimport { findOptionInOptions } from './findOptionInOptions'\nimport type { DataType, OptionType } from './types'\nimport { INPUT_SIZE_HEIGHT, SIZES_TAG } from './types'\n\ntype SelectBarProps = {\n  size: 'small' | 'medium' | 'large'\n  clearable: boolean\n  disabled: boolean\n  readOnly: boolean\n  placeholder: string\n  success?: string\n  error?: string\n  autoFocus?: boolean\n  innerRef: RefObject<HTMLDivElement>\n  id?: string\n  'data-testid': string\n  label?: string\n  tooltip?: string\n}\n\ntype DisplayValuesProps = {\n  refTag: RefObject<HTMLDivElement>\n  nonOverflowedValues: OptionType[]\n  disabled: boolean\n  readOnly: boolean\n  overflowed: boolean\n  overflowAmount: number\n  size: 'small' | 'medium' | 'large'\n}\n\nconst StateStack = styled(Stack)`\n  padding-right: ${({ theme }) => theme.space['2']};\n  display: flex;\n`\nconst Placeholder = styled(Text)`\nuser-select: none;\n`\n\nconst StyledInputWrapper = styled(Stack)<{\n  'data-readonly': boolean\n  'data-disabled': boolean\n  'data-size': 'small' | 'medium' | 'large'\n  'data-state': 'neutral' | 'success' | 'danger'\n  'data-dropdownvisible': boolean\n  'aria-label'?: string\n}>`\n  display: flex;\n  padding: ${({ theme }) => theme.space[1]};\n  padding-right: 0;\n  padding-left: ${({ theme }) => theme.space[2]};\n  cursor: pointer;\n  box-shadow: none;\n  background: ${({ theme }) => theme.colors.neutral.background};\n  border-radius: ${({ theme }) => theme.radii.default};\n  width: 100%;\n  overflow: hidden;\n\n  &[data-size='small'] {\n    height: ${INPUT_SIZE_HEIGHT.small}px;\n    padding-left: ${({ theme }) => theme.space[1]};\n  }\n  &[data-size='medium'] {\n    height: ${INPUT_SIZE_HEIGHT.medium}px;\n  }\n  &[data-size='large'] {\n    height: ${INPUT_SIZE_HEIGHT.large}px;\n  }\n  &[data-state='neutral'] {\n    border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n\n    &:not([data-disabled=\"true\"]):not([data-readonly=\"true\"]):active {\n      border-color: ${({ theme }) => theme.colors.primary.borderHover};\n      box-shadow: ${({ theme }) => theme.shadows.focusPrimary};\n    }\n    &:not([data-disabled='true']):hover,\n    :not([data-disabled='true']):focus {\n      border-color: ${({ theme }) => theme.colors.primary.borderHover};\n      outline: none;\n    }\n\n    &[data-dropdownvisible='true'] {\n    border-color: ${({ theme }) => theme.colors.primary.borderHover};\n  }\n  }\n\n  &[data-state='success'] {\n    border: 1px solid ${({ theme }) => theme.colors.success.border};\n    &:not([data-disabled=\"true\"]):not([data-readonly=\"true\"]):active {\n      border-color: ${({ theme }) => theme.colors.success.borderHover};\n      box-shadow: ${({ theme }) => theme.shadows.focusSuccess};\n    }\n  \n    &[data-dropdownvisible='true'] {\n      border-color: ${({ theme }) => theme.colors.success.borderHover};\n    }\n  }\n\n  &[data-state='danger'] {\n    border: 1px solid ${({ theme }) => theme.colors.danger.border};\n\n    &:not([data-disabled=\"true\"]):not([data-readonly=\"true\"]):active {\n      border-color: ${({ theme }) => theme.colors.danger.borderHover};\n      box-shadow: ${({ theme }) => theme.shadows.focusDanger};\n    }\n  \n    &[data-dropdownvisible='true'] {\n      border-color: ${({ theme }) => theme.colors.danger.borderHover};\n    }\n  }\n\n  &:not([data-disabled='true']):not([data-readonly]):hover {\n    border-color: ${({ theme }) => theme.colors.primary.border};\n  }\n\n  &[data-readonly='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n    border-color: ${({ theme }) => theme.colors.neutral.border};\n    cursor: default;\n  }\n\n  &[data-disabled='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    border-color: ${({ theme }) => theme.colors.neutral.borderDisabled};\n    cursor: not-allowed;\n  }\n`\nconst CustomTag = styled(Tag)`\n  height: fit-content;\n  width: fit-content;\n`\nconst SelectedValues = styled(Text)`\ntext-overflow: ellipsis;\noverflow: hidden; \n`\n\nconst isValidSelectedValue = (selectedValue: string, options: DataType) =>\n  !Array.isArray(options)\n    ? Object.keys(options).some(group =>\n        options[group].some(\n          option => option.value === selectedValue && !option.disabled,\n        ),\n      )\n    : options.some(option => option.value === selectedValue && !option.disabled)\n\nconst DisplayValues = ({\n  refTag,\n  nonOverflowedValues,\n  disabled,\n  readOnly,\n  overflowed,\n  overflowAmount,\n  size,\n}: DisplayValuesProps) => {\n  const { multiselect, selectedData, setSelectedData, options, onChange } =\n    useSelectInput()\n\n  return multiselect ? (\n    <Stack\n      direction=\"row\"\n      gap=\"1\"\n      wrap=\"nowrap\"\n      ref={refTag}\n      alignItems=\"center\"\n    >\n      {nonOverflowedValues.map(option => (\n        <CustomTag\n          data-testid=\"selected-options-tags\"\n          key={option?.value}\n          sentiment=\"neutral\"\n          disabled={disabled}\n          onClose={\n            !readOnly\n              ? event => {\n                  event.stopPropagation()\n                  setSelectedData({\n                    type: 'selectOption',\n                    clickedOption: option,\n                  })\n                  const newSelectedValues = selectedData.selectedValues?.filter(\n                    val => val !== option.value,\n                  )\n                  onChange?.(newSelectedValues)\n                }\n              : undefined\n          }\n        >\n          {option?.label}\n        </CustomTag>\n      ))}\n      {overflowed ? (\n        <Tag\n          sentiment=\"neutral\"\n          disabled={disabled}\n          key=\"+\"\n          data-testid=\"plus-tag\"\n          aria-label=\"Plus tag\"\n        >\n          <Icon name=\"plus\" />\n          {overflowAmount}\n        </Tag>\n      ) : null}\n    </Stack>\n  ) : (\n    <SelectedValues as=\"div\" variant={size === 'large' ? 'body' : 'bodySmall'}>\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 [overflowed, setOverflowed] = useState(false)\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          setOverflowAmount(computedOverflowAmount)\n        } else {\n          computedNonOverflowedValues = [\n            ...computedNonOverflowedValues,\n            selectedOption,\n          ]\n          setNonOverFlowedValues(computedNonOverflowedValues)\n          tagsWidth += totalTagWidth\n        }\n      }\n    }\n    if (computedOverflowAmount === 0) {\n      setOverflowed(false)\n    } else {\n      setOverflowed(true)\n    }\n    setOverflowAmount(computedOverflowAmount)\n  }, [options, selectedData.selectedValues, width])\n\n  useEffect(() => {\n    setSelectedData({ type: 'update' })\n  }, [setSelectedData, options])\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\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        {selectedData.selectedValues.length > 0 ? (\n          <DisplayValues\n            refTag={refTag}\n            nonOverflowedValues={nonOverflowedValues}\n            disabled={disabled}\n            readOnly={readOnly}\n            overflowed={overflowed}\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 ? <Icon name=\"alert\" sentiment=\"danger\" /> : null}\n          {success && !error ? (\n            <Icon name=\"checkbox-circle-outline\" sentiment=\"success\" />\n          ) : null}\n          {clearable && selectedData.selectedValues.length > 0 ? (\n            <Button\n              aria-label=\"clear value\"\n              disabled={disabled || !selectedData.selectedValues[0] || readOnly}\n              variant=\"ghost\"\n              size=\"small\"\n              icon=\"close\"\n              onClick={event => {\n                event.stopPropagation()\n                setSelectedData({ type: 'clearAll' })\n                if (multiselect) {\n                  onChange?.([])\n                } else {\n                  onChange?.('')\n                }\n              }}\n              sentiment=\"neutral\"\n              data-testid=\"clear-all\"\n            />\n          ) : null}\n          <Icon\n            aria-label=\"show dropdown\"\n            size=\"small\"\n            name=\"arrow-down\"\n            sentiment=\"neutral\"\n            disabled={disabled || readOnly}\n          />\n        </StateStack>\n      </StyledInputWrapper>\n    </Tooltip>\n  )\n}\n"]} */"));
|
|
94
|
+
const CustomTag = /* @__PURE__ */ _styled__default.default(index$2.Tag, process.env.NODE_ENV === "production" ? {
|
|
95
|
+
target: "ejy0aca1"
|
|
68
96
|
} : {
|
|
69
|
-
target: "
|
|
97
|
+
target: "ejy0aca1",
|
|
70
98
|
label: "CustomTag"
|
|
71
99
|
})(process.env.NODE_ENV === "production" ? {
|
|
72
100
|
name: "1snt5jp",
|
|
@@ -74,7 +102,21 @@ const CustomTag = /* @__PURE__ */ _styled__default.default(index$1.Tag, process.
|
|
|
74
102
|
} : {
|
|
75
103
|
name: "1snt5jp",
|
|
76
104
|
styles: "height:fit-content;width:fit-content",
|
|
77
|
-
map: "/*# 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":"AAgH6B","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/SelectBar.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { Icon } from '@ultraviolet/icons/legacy'\nimport type { RefObject } from 'react'\nimport { useEffect, useMemo, useRef, useState } from 'react'\nimport { Button } from '../Button'\nimport { Stack } from '../Stack'\nimport { Tag } from '../Tag'\nimport { Text } from '../Text'\nimport { Tooltip } from '../Tooltip'\nimport { useSelectInput } from './SelectInputProvider'\nimport { findOptionInOptions } from './findOptionInOptions'\nimport type { DataType, OptionType } from './types'\nimport { INPUT_SIZE_HEIGHT, SIZES_TAG } from './types'\n\ntype SelectBarProps = {\n  size: 'small' | 'medium' | 'large'\n  clearable: boolean\n  disabled: boolean\n  readOnly: boolean\n  placeholder: string\n  success?: string\n  error?: string\n  autoFocus?: boolean\n  innerRef: RefObject<HTMLDivElement>\n  id?: string\n  'data-testid': string\n  label?: string\n  tooltip?: string\n}\n\ntype DisplayValuesProps = {\n  refTag: RefObject<HTMLDivElement>\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`\n\nconst StyledInputWrapper = styled(Stack)<{\n  'data-readonly': boolean\n  'data-disabled': boolean\n  'data-size': 'small' | 'medium' | 'large'\n  'data-state': 'neutral' | 'success' | 'danger'\n  'data-dropdownvisible': boolean\n  'aria-label'?: string\n}>`\n  display: flex;\n  padding: ${({ theme }) => theme.space[1]};\n  padding-right: 0;\n  padding-left: ${({ theme }) => theme.space[2]};\n  cursor: pointer;\n  box-shadow: none;\n  background: ${({ theme }) => theme.colors.neutral.background};\n  border-radius: ${({ theme }) => theme.radii.default};\n  width: 100%;\n\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  &[data-size='small'] {\n    height: ${INPUT_SIZE_HEIGHT.small}px;\n    padding-left: ${({ theme }) => theme.space[1]};\n  }\n  &[data-size='medium'] {\n    height: ${INPUT_SIZE_HEIGHT.medium}px;\n  }\n  &[data-size='large'] {\n    height: ${INPUT_SIZE_HEIGHT.large}px;\n  }\n  &[data-state='neutral'] {\n    border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  }\n  &[data-state='success'] {\n    border: 1px solid ${({ theme }) => theme.colors.success.border};\n  }\n  &[data-state='danger'] {\n    border: 1px solid ${({ theme }) => theme.colors.danger.border};\n  }\n\n  &:not([data-disabled='true']):not([data-readonly]):hover {\n    border-color: ${({ theme }) => theme.colors.primary.border};\n  }\n\n  &:not([data-disabled='true']):not([data-readonly]):active {\n    box-shadow: ${({ theme }) => theme.shadows.focusPrimary};\n  }\n\n  &:not([data-disabled='true']):hover,\n  :not([data-disabled='true']):focus {\n    border-color: ${({ theme }) => theme.colors.primary.borderHover};\n    outline: none;\n  }\n\n  &[data-dropdownvisible='true'] {\n    border-color: ${({ theme }) => theme.colors.primary.borderHover};\n  }\n`\nconst CustomTag = styled(Tag)`\n  height: fit-content;\n  width: fit-content;\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          <Icon name=\"plus\" />\n          {overflowAmount}\n        </Tag>\n      ) : null}\n    </Stack>\n  ) : (\n    <Text as=\"p\" variant={size === 'large' ? 'body' : 'bodySmall'}>\n      {selectedData.selectedValues[0]\n        ? findOptionInOptions(options, selectedData.selectedValues[0])?.label\n        : null}\n    </Text>\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 [overflowed, setOverflowed] = useState(false)\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          setOverflowAmount(computedOverflowAmount)\n        } else {\n          computedNonOverflowedValues = [\n            ...computedNonOverflowedValues,\n            selectedOption,\n          ]\n          setNonOverFlowedValues(computedNonOverflowedValues)\n          tagsWidth += totalTagWidth\n        }\n      }\n    }\n    if (computedOverflowAmount === 0) {\n      setOverflowed(false)\n    } else {\n      setOverflowed(true)\n    }\n    setOverflowAmount(computedOverflowAmount)\n  }, [options, selectedData.selectedValues, width])\n\n  useEffect(() => {\n    setSelectedData({ type: 'update' })\n  }, [setSelectedData, options])\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\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        {selectedData.selectedValues.length > 0 ? (\n          <DisplayValues\n            refTag={refTag}\n            nonOverflowedValues={nonOverflowedValues}\n            disabled={disabled}\n            readOnly={readOnly}\n            overflowed={overflowed}\n            overflowAmount={overflowAmount}\n            size={size}\n          />\n        ) : (\n          <Text\n            as=\"p\"\n            variant={size === 'large' ? 'body' : 'bodySmall'}\n            sentiment=\"neutral\"\n            disabled={disabled}\n            prominence=\"weak\"\n          >\n            {placeholder}\n          </Text>\n        )}\n        <StateStack direction=\"row\" gap={1} alignItems=\"center\">\n          {error ? <Icon name=\"alert\" sentiment=\"danger\" /> : null}\n          {success && !error ? (\n            <Icon name=\"checkbox-circle-outline\" sentiment=\"success\" />\n          ) : null}\n          {clearable && selectedData.selectedValues.length > 0 ? (\n            <Button\n              aria-label=\"clear value\"\n              disabled={\n                disabled || ![selectedData.selectedValues[0]] || readOnly\n              }\n              variant=\"ghost\"\n              size=\"small\"\n              icon=\"close\"\n              onClick={event => {\n                event.stopPropagation()\n                setSelectedData({ type: 'clearAll' })\n                if (multiselect) {\n                  onChange?.([])\n                } else {\n                  onChange?.('')\n                }\n              }}\n              sentiment=\"neutral\"\n              data-testid=\"clear-all\"\n            />\n          ) : null}\n          <Icon\n            aria-label=\"show dropdown\"\n            size=\"small\"\n            name=\"arrow-down\"\n            sentiment=\"neutral\"\n            disabled={disabled || readOnly}\n          />\n        </StateStack>\n      </StyledInputWrapper>\n    </Tooltip>\n  )\n}\n"]} */",
|
|
105
|
+
map: "/*# 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":"AAwI6B","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/SelectBar.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { Icon } from '@ultraviolet/icons/legacy'\nimport type { RefObject } from 'react'\nimport { useEffect, useMemo, useRef, useState } from 'react'\nimport { Button } from '../Button'\nimport { Stack } from '../Stack'\nimport { Tag } from '../Tag'\nimport { Text } from '../Text'\nimport { Tooltip } from '../Tooltip'\nimport { useSelectInput } from './SelectInputProvider'\nimport { findOptionInOptions } from './findOptionInOptions'\nimport type { DataType, OptionType } from './types'\nimport { INPUT_SIZE_HEIGHT, SIZES_TAG } from './types'\n\ntype SelectBarProps = {\n  size: 'small' | 'medium' | 'large'\n  clearable: boolean\n  disabled: boolean\n  readOnly: boolean\n  placeholder: string\n  success?: string\n  error?: string\n  autoFocus?: boolean\n  innerRef: RefObject<HTMLDivElement>\n  id?: string\n  'data-testid': string\n  label?: string\n  tooltip?: string\n}\n\ntype DisplayValuesProps = {\n  refTag: RefObject<HTMLDivElement>\n  nonOverflowedValues: OptionType[]\n  disabled: boolean\n  readOnly: boolean\n  overflowed: boolean\n  overflowAmount: number\n  size: 'small' | 'medium' | 'large'\n}\n\nconst StateStack = styled(Stack)`\n  padding-right: ${({ theme }) => theme.space['2']};\n  display: flex;\n`\nconst Placeholder = styled(Text)`\nuser-select: none;\n`\n\nconst StyledInputWrapper = styled(Stack)<{\n  'data-readonly': boolean\n  'data-disabled': boolean\n  'data-size': 'small' | 'medium' | 'large'\n  'data-state': 'neutral' | 'success' | 'danger'\n  'data-dropdownvisible': boolean\n  'aria-label'?: string\n}>`\n  display: flex;\n  padding: ${({ theme }) => theme.space[1]};\n  padding-right: 0;\n  padding-left: ${({ theme }) => theme.space[2]};\n  cursor: pointer;\n  box-shadow: none;\n  background: ${({ theme }) => theme.colors.neutral.background};\n  border-radius: ${({ theme }) => theme.radii.default};\n  width: 100%;\n  overflow: hidden;\n\n  &[data-size='small'] {\n    height: ${INPUT_SIZE_HEIGHT.small}px;\n    padding-left: ${({ theme }) => theme.space[1]};\n  }\n  &[data-size='medium'] {\n    height: ${INPUT_SIZE_HEIGHT.medium}px;\n  }\n  &[data-size='large'] {\n    height: ${INPUT_SIZE_HEIGHT.large}px;\n  }\n  &[data-state='neutral'] {\n    border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n\n    &:not([data-disabled=\"true\"]):not([data-readonly=\"true\"]):active {\n      border-color: ${({ theme }) => theme.colors.primary.borderHover};\n      box-shadow: ${({ theme }) => theme.shadows.focusPrimary};\n    }\n    &:not([data-disabled='true']):hover,\n    :not([data-disabled='true']):focus {\n      border-color: ${({ theme }) => theme.colors.primary.borderHover};\n      outline: none;\n    }\n\n    &[data-dropdownvisible='true'] {\n    border-color: ${({ theme }) => theme.colors.primary.borderHover};\n  }\n  }\n\n  &[data-state='success'] {\n    border: 1px solid ${({ theme }) => theme.colors.success.border};\n    &:not([data-disabled=\"true\"]):not([data-readonly=\"true\"]):active {\n      border-color: ${({ theme }) => theme.colors.success.borderHover};\n      box-shadow: ${({ theme }) => theme.shadows.focusSuccess};\n    }\n  \n    &[data-dropdownvisible='true'] {\n      border-color: ${({ theme }) => theme.colors.success.borderHover};\n    }\n  }\n\n  &[data-state='danger'] {\n    border: 1px solid ${({ theme }) => theme.colors.danger.border};\n\n    &:not([data-disabled=\"true\"]):not([data-readonly=\"true\"]):active {\n      border-color: ${({ theme }) => theme.colors.danger.borderHover};\n      box-shadow: ${({ theme }) => theme.shadows.focusDanger};\n    }\n  \n    &[data-dropdownvisible='true'] {\n      border-color: ${({ theme }) => theme.colors.danger.borderHover};\n    }\n  }\n\n  &:not([data-disabled='true']):not([data-readonly]):hover {\n    border-color: ${({ theme }) => theme.colors.primary.border};\n  }\n\n  &[data-readonly='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n    border-color: ${({ theme }) => theme.colors.neutral.border};\n    cursor: default;\n  }\n\n  &[data-disabled='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    border-color: ${({ theme }) => theme.colors.neutral.borderDisabled};\n    cursor: not-allowed;\n  }\n`\nconst CustomTag = styled(Tag)`\n  height: fit-content;\n  width: fit-content;\n`\nconst SelectedValues = styled(Text)`\ntext-overflow: ellipsis;\noverflow: hidden; \n`\n\nconst isValidSelectedValue = (selectedValue: string, options: DataType) =>\n  !Array.isArray(options)\n    ? Object.keys(options).some(group =>\n        options[group].some(\n          option => option.value === selectedValue && !option.disabled,\n        ),\n      )\n    : options.some(option => option.value === selectedValue && !option.disabled)\n\nconst DisplayValues = ({\n  refTag,\n  nonOverflowedValues,\n  disabled,\n  readOnly,\n  overflowed,\n  overflowAmount,\n  size,\n}: DisplayValuesProps) => {\n  const { multiselect, selectedData, setSelectedData, options, onChange } =\n    useSelectInput()\n\n  return multiselect ? (\n    <Stack\n      direction=\"row\"\n      gap=\"1\"\n      wrap=\"nowrap\"\n      ref={refTag}\n      alignItems=\"center\"\n    >\n      {nonOverflowedValues.map(option => (\n        <CustomTag\n          data-testid=\"selected-options-tags\"\n          key={option?.value}\n          sentiment=\"neutral\"\n          disabled={disabled}\n          onClose={\n            !readOnly\n              ? event => {\n                  event.stopPropagation()\n                  setSelectedData({\n                    type: 'selectOption',\n                    clickedOption: option,\n                  })\n                  const newSelectedValues = selectedData.selectedValues?.filter(\n                    val => val !== option.value,\n                  )\n                  onChange?.(newSelectedValues)\n                }\n              : undefined\n          }\n        >\n          {option?.label}\n        </CustomTag>\n      ))}\n      {overflowed ? (\n        <Tag\n          sentiment=\"neutral\"\n          disabled={disabled}\n          key=\"+\"\n          data-testid=\"plus-tag\"\n          aria-label=\"Plus tag\"\n        >\n          <Icon name=\"plus\" />\n          {overflowAmount}\n        </Tag>\n      ) : null}\n    </Stack>\n  ) : (\n    <SelectedValues as=\"div\" variant={size === 'large' ? 'body' : 'bodySmall'}>\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 [overflowed, setOverflowed] = useState(false)\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          setOverflowAmount(computedOverflowAmount)\n        } else {\n          computedNonOverflowedValues = [\n            ...computedNonOverflowedValues,\n            selectedOption,\n          ]\n          setNonOverFlowedValues(computedNonOverflowedValues)\n          tagsWidth += totalTagWidth\n        }\n      }\n    }\n    if (computedOverflowAmount === 0) {\n      setOverflowed(false)\n    } else {\n      setOverflowed(true)\n    }\n    setOverflowAmount(computedOverflowAmount)\n  }, [options, selectedData.selectedValues, width])\n\n  useEffect(() => {\n    setSelectedData({ type: 'update' })\n  }, [setSelectedData, options])\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\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        {selectedData.selectedValues.length > 0 ? (\n          <DisplayValues\n            refTag={refTag}\n            nonOverflowedValues={nonOverflowedValues}\n            disabled={disabled}\n            readOnly={readOnly}\n            overflowed={overflowed}\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 ? <Icon name=\"alert\" sentiment=\"danger\" /> : null}\n          {success && !error ? (\n            <Icon name=\"checkbox-circle-outline\" sentiment=\"success\" />\n          ) : null}\n          {clearable && selectedData.selectedValues.length > 0 ? (\n            <Button\n              aria-label=\"clear value\"\n              disabled={disabled || !selectedData.selectedValues[0] || readOnly}\n              variant=\"ghost\"\n              size=\"small\"\n              icon=\"close\"\n              onClick={event => {\n                event.stopPropagation()\n                setSelectedData({ type: 'clearAll' })\n                if (multiselect) {\n                  onChange?.([])\n                } else {\n                  onChange?.('')\n                }\n              }}\n              sentiment=\"neutral\"\n              data-testid=\"clear-all\"\n            />\n          ) : null}\n          <Icon\n            aria-label=\"show dropdown\"\n            size=\"small\"\n            name=\"arrow-down\"\n            sentiment=\"neutral\"\n            disabled={disabled || readOnly}\n          />\n        </StateStack>\n      </StyledInputWrapper>\n    </Tooltip>\n  )\n}\n"]} */",
|
|
106
|
+
toString: _EMOTION_STRINGIFIED_CSS_ERROR__
|
|
107
|
+
});
|
|
108
|
+
const SelectedValues = /* @__PURE__ */ _styled__default.default(index$1.Text, process.env.NODE_ENV === "production" ? {
|
|
109
|
+
target: "ejy0aca0"
|
|
110
|
+
} : {
|
|
111
|
+
target: "ejy0aca0",
|
|
112
|
+
label: "SelectedValues"
|
|
113
|
+
})(process.env.NODE_ENV === "production" ? {
|
|
114
|
+
name: "fzu03x",
|
|
115
|
+
styles: "text-overflow:ellipsis;overflow:hidden"
|
|
116
|
+
} : {
|
|
117
|
+
name: "fzu03x",
|
|
118
|
+
styles: "text-overflow:ellipsis;overflow:hidden",
|
|
119
|
+
map: "/*# 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":"AA4ImC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SelectInputV2/SelectBar.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { Icon } from '@ultraviolet/icons/legacy'\nimport type { RefObject } from 'react'\nimport { useEffect, useMemo, useRef, useState } from 'react'\nimport { Button } from '../Button'\nimport { Stack } from '../Stack'\nimport { Tag } from '../Tag'\nimport { Text } from '../Text'\nimport { Tooltip } from '../Tooltip'\nimport { useSelectInput } from './SelectInputProvider'\nimport { findOptionInOptions } from './findOptionInOptions'\nimport type { DataType, OptionType } from './types'\nimport { INPUT_SIZE_HEIGHT, SIZES_TAG } from './types'\n\ntype SelectBarProps = {\n  size: 'small' | 'medium' | 'large'\n  clearable: boolean\n  disabled: boolean\n  readOnly: boolean\n  placeholder: string\n  success?: string\n  error?: string\n  autoFocus?: boolean\n  innerRef: RefObject<HTMLDivElement>\n  id?: string\n  'data-testid': string\n  label?: string\n  tooltip?: string\n}\n\ntype DisplayValuesProps = {\n  refTag: RefObject<HTMLDivElement>\n  nonOverflowedValues: OptionType[]\n  disabled: boolean\n  readOnly: boolean\n  overflowed: boolean\n  overflowAmount: number\n  size: 'small' | 'medium' | 'large'\n}\n\nconst StateStack = styled(Stack)`\n  padding-right: ${({ theme }) => theme.space['2']};\n  display: flex;\n`\nconst Placeholder = styled(Text)`\nuser-select: none;\n`\n\nconst StyledInputWrapper = styled(Stack)<{\n  'data-readonly': boolean\n  'data-disabled': boolean\n  'data-size': 'small' | 'medium' | 'large'\n  'data-state': 'neutral' | 'success' | 'danger'\n  'data-dropdownvisible': boolean\n  'aria-label'?: string\n}>`\n  display: flex;\n  padding: ${({ theme }) => theme.space[1]};\n  padding-right: 0;\n  padding-left: ${({ theme }) => theme.space[2]};\n  cursor: pointer;\n  box-shadow: none;\n  background: ${({ theme }) => theme.colors.neutral.background};\n  border-radius: ${({ theme }) => theme.radii.default};\n  width: 100%;\n  overflow: hidden;\n\n  &[data-size='small'] {\n    height: ${INPUT_SIZE_HEIGHT.small}px;\n    padding-left: ${({ theme }) => theme.space[1]};\n  }\n  &[data-size='medium'] {\n    height: ${INPUT_SIZE_HEIGHT.medium}px;\n  }\n  &[data-size='large'] {\n    height: ${INPUT_SIZE_HEIGHT.large}px;\n  }\n  &[data-state='neutral'] {\n    border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n\n    &:not([data-disabled=\"true\"]):not([data-readonly=\"true\"]):active {\n      border-color: ${({ theme }) => theme.colors.primary.borderHover};\n      box-shadow: ${({ theme }) => theme.shadows.focusPrimary};\n    }\n    &:not([data-disabled='true']):hover,\n    :not([data-disabled='true']):focus {\n      border-color: ${({ theme }) => theme.colors.primary.borderHover};\n      outline: none;\n    }\n\n    &[data-dropdownvisible='true'] {\n    border-color: ${({ theme }) => theme.colors.primary.borderHover};\n  }\n  }\n\n  &[data-state='success'] {\n    border: 1px solid ${({ theme }) => theme.colors.success.border};\n    &:not([data-disabled=\"true\"]):not([data-readonly=\"true\"]):active {\n      border-color: ${({ theme }) => theme.colors.success.borderHover};\n      box-shadow: ${({ theme }) => theme.shadows.focusSuccess};\n    }\n  \n    &[data-dropdownvisible='true'] {\n      border-color: ${({ theme }) => theme.colors.success.borderHover};\n    }\n  }\n\n  &[data-state='danger'] {\n    border: 1px solid ${({ theme }) => theme.colors.danger.border};\n\n    &:not([data-disabled=\"true\"]):not([data-readonly=\"true\"]):active {\n      border-color: ${({ theme }) => theme.colors.danger.borderHover};\n      box-shadow: ${({ theme }) => theme.shadows.focusDanger};\n    }\n  \n    &[data-dropdownvisible='true'] {\n      border-color: ${({ theme }) => theme.colors.danger.borderHover};\n    }\n  }\n\n  &:not([data-disabled='true']):not([data-readonly]):hover {\n    border-color: ${({ theme }) => theme.colors.primary.border};\n  }\n\n  &[data-readonly='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n    border-color: ${({ theme }) => theme.colors.neutral.border};\n    cursor: default;\n  }\n\n  &[data-disabled='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    border-color: ${({ theme }) => theme.colors.neutral.borderDisabled};\n    cursor: not-allowed;\n  }\n`\nconst CustomTag = styled(Tag)`\n  height: fit-content;\n  width: fit-content;\n`\nconst SelectedValues = styled(Text)`\ntext-overflow: ellipsis;\noverflow: hidden; \n`\n\nconst isValidSelectedValue = (selectedValue: string, options: DataType) =>\n  !Array.isArray(options)\n    ? Object.keys(options).some(group =>\n        options[group].some(\n          option => option.value === selectedValue && !option.disabled,\n        ),\n      )\n    : options.some(option => option.value === selectedValue && !option.disabled)\n\nconst DisplayValues = ({\n  refTag,\n  nonOverflowedValues,\n  disabled,\n  readOnly,\n  overflowed,\n  overflowAmount,\n  size,\n}: DisplayValuesProps) => {\n  const { multiselect, selectedData, setSelectedData, options, onChange } =\n    useSelectInput()\n\n  return multiselect ? (\n    <Stack\n      direction=\"row\"\n      gap=\"1\"\n      wrap=\"nowrap\"\n      ref={refTag}\n      alignItems=\"center\"\n    >\n      {nonOverflowedValues.map(option => (\n        <CustomTag\n          data-testid=\"selected-options-tags\"\n          key={option?.value}\n          sentiment=\"neutral\"\n          disabled={disabled}\n          onClose={\n            !readOnly\n              ? event => {\n                  event.stopPropagation()\n                  setSelectedData({\n                    type: 'selectOption',\n                    clickedOption: option,\n                  })\n                  const newSelectedValues = selectedData.selectedValues?.filter(\n                    val => val !== option.value,\n                  )\n                  onChange?.(newSelectedValues)\n                }\n              : undefined\n          }\n        >\n          {option?.label}\n        </CustomTag>\n      ))}\n      {overflowed ? (\n        <Tag\n          sentiment=\"neutral\"\n          disabled={disabled}\n          key=\"+\"\n          data-testid=\"plus-tag\"\n          aria-label=\"Plus tag\"\n        >\n          <Icon name=\"plus\" />\n          {overflowAmount}\n        </Tag>\n      ) : null}\n    </Stack>\n  ) : (\n    <SelectedValues as=\"div\" variant={size === 'large' ? 'body' : 'bodySmall'}>\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 [overflowed, setOverflowed] = useState(false)\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          setOverflowAmount(computedOverflowAmount)\n        } else {\n          computedNonOverflowedValues = [\n            ...computedNonOverflowedValues,\n            selectedOption,\n          ]\n          setNonOverFlowedValues(computedNonOverflowedValues)\n          tagsWidth += totalTagWidth\n        }\n      }\n    }\n    if (computedOverflowAmount === 0) {\n      setOverflowed(false)\n    } else {\n      setOverflowed(true)\n    }\n    setOverflowAmount(computedOverflowAmount)\n  }, [options, selectedData.selectedValues, width])\n\n  useEffect(() => {\n    setSelectedData({ type: 'update' })\n  }, [setSelectedData, options])\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\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        {selectedData.selectedValues.length > 0 ? (\n          <DisplayValues\n            refTag={refTag}\n            nonOverflowedValues={nonOverflowedValues}\n            disabled={disabled}\n            readOnly={readOnly}\n            overflowed={overflowed}\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 ? <Icon name=\"alert\" sentiment=\"danger\" /> : null}\n          {success && !error ? (\n            <Icon name=\"checkbox-circle-outline\" sentiment=\"success\" />\n          ) : null}\n          {clearable && selectedData.selectedValues.length > 0 ? (\n            <Button\n              aria-label=\"clear value\"\n              disabled={disabled || !selectedData.selectedValues[0] || readOnly}\n              variant=\"ghost\"\n              size=\"small\"\n              icon=\"close\"\n              onClick={event => {\n                event.stopPropagation()\n                setSelectedData({ type: 'clearAll' })\n                if (multiselect) {\n                  onChange?.([])\n                } else {\n                  onChange?.('')\n                }\n              }}\n              sentiment=\"neutral\"\n              data-testid=\"clear-all\"\n            />\n          ) : null}\n          <Icon\n            aria-label=\"show dropdown\"\n            size=\"small\"\n            name=\"arrow-down\"\n            sentiment=\"neutral\"\n            disabled={disabled || readOnly}\n          />\n        </StateStack>\n      </StyledInputWrapper>\n    </Tooltip>\n  )\n}\n"]} */",
|
|
78
120
|
toString: _EMOTION_STRINGIFIED_CSS_ERROR__
|
|
79
121
|
});
|
|
80
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);
|
|
@@ -104,11 +146,11 @@ const DisplayValues = ({
|
|
|
104
146
|
const newSelectedValues = selectedData.selectedValues?.filter((val) => val !== option.value);
|
|
105
147
|
onChange?.(newSelectedValues);
|
|
106
148
|
} : void 0, children: option?.label }, option?.value)),
|
|
107
|
-
overflowed ? /* @__PURE__ */ jsxRuntime.jsxs(index$
|
|
149
|
+
overflowed ? /* @__PURE__ */ jsxRuntime.jsxs(index$2.Tag, { sentiment: "neutral", disabled, "data-testid": "plus-tag", "aria-label": "Plus tag", children: [
|
|
108
150
|
/* @__PURE__ */ jsxRuntime.jsx(legacy.Icon, { name: "plus" }),
|
|
109
151
|
overflowAmount
|
|
110
152
|
] }, "+") : null
|
|
111
|
-
] }) : /* @__PURE__ */ jsxRuntime.jsx(
|
|
153
|
+
] }) : /* @__PURE__ */ jsxRuntime.jsx(SelectedValues, { as: "div", variant: size === "large" ? "body" : "bodySmall", children: selectedData.selectedValues[0] ? findOptionInOptions.findOptionInOptions(options, selectedData.selectedValues[0])?.label : null });
|
|
112
154
|
};
|
|
113
155
|
const SelectBar = ({
|
|
114
156
|
size,
|
|
@@ -187,7 +229,7 @@ const SelectBar = ({
|
|
|
187
229
|
type: "update"
|
|
188
230
|
});
|
|
189
231
|
}, [setSelectedData, options]);
|
|
190
|
-
return /* @__PURE__ */ jsxRuntime.jsx(index$
|
|
232
|
+
return /* @__PURE__ */ jsxRuntime.jsx(index$3.Tooltip, { text: tooltip, children: /* @__PURE__ */ jsxRuntime.jsxs(StyledInputWrapper, { role: "combobox", id, "data-disabled": disabled, "data-readonly": readOnly, "data-size": size, "data-dropdownvisible": isDropdownVisible, "data-state": state, direction: "row", wrap: "nowrap", gap: "1", justifyContent: "space-between", alignItems: "center", onClick: openable ? () => setIsDropdownVisible(!isDropdownVisible) : void 0, "data-testid": dataTestId, autoFocus, onKeyDown: (event) => {
|
|
191
233
|
if (event.key === "ArrowDown") {
|
|
192
234
|
if (!isDropdownVisible) {
|
|
193
235
|
setIsDropdownVisible(true);
|
|
@@ -197,11 +239,11 @@ const SelectBar = ({
|
|
|
197
239
|
}
|
|
198
240
|
return ["Enter", " "].includes(event.key) && openable ? setIsDropdownVisible(!isDropdownVisible) : null;
|
|
199
241
|
}, ref: innerRef, "aria-haspopup": "listbox", "aria-expanded": isDropdownVisible, tabIndex: 0, "aria-label": label, children: [
|
|
200
|
-
selectedData.selectedValues.length > 0 ? /* @__PURE__ */ jsxRuntime.jsx(DisplayValues, { refTag, nonOverflowedValues, disabled, readOnly, overflowed, overflowAmount, size }) : /* @__PURE__ */ jsxRuntime.jsx(
|
|
242
|
+
selectedData.selectedValues.length > 0 ? /* @__PURE__ */ jsxRuntime.jsx(DisplayValues, { refTag, nonOverflowedValues, disabled, readOnly, overflowed, overflowAmount, size }) : /* @__PURE__ */ jsxRuntime.jsx(Placeholder, { as: "p", variant: size === "large" ? "body" : "bodySmall", sentiment: "neutral", disabled, prominence: "weak", children: placeholder }),
|
|
201
243
|
/* @__PURE__ */ jsxRuntime.jsxs(StateStack, { direction: "row", gap: 1, alignItems: "center", children: [
|
|
202
244
|
error ? /* @__PURE__ */ jsxRuntime.jsx(legacy.Icon, { name: "alert", sentiment: "danger" }) : null,
|
|
203
245
|
success && !error ? /* @__PURE__ */ jsxRuntime.jsx(legacy.Icon, { name: "checkbox-circle-outline", sentiment: "success" }) : null,
|
|
204
|
-
clearable && selectedData.selectedValues.length > 0 ? /* @__PURE__ */ jsxRuntime.jsx(index$4.Button, { "aria-label": "clear value", disabled: disabled || !
|
|
246
|
+
clearable && selectedData.selectedValues.length > 0 ? /* @__PURE__ */ jsxRuntime.jsx(index$4.Button, { "aria-label": "clear value", disabled: disabled || !selectedData.selectedValues[0] || readOnly, variant: "ghost", size: "small", icon: "close", onClick: (event) => {
|
|
205
247
|
event.stopPropagation();
|
|
206
248
|
setSelectedData({
|
|
207
249
|
type: "clearAll"
|