@ultraviolet/ui 2.0.0-beta.1 → 2.0.0-beta.3

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (46) hide show
  1. package/dist/components/ActionBar/index.cjs +13 -4
  2. package/dist/components/ActionBar/index.js +13 -4
  3. package/dist/components/BarChart/index.cjs +2 -35
  4. package/dist/components/BarChart/index.js +2 -35
  5. package/dist/components/CheckboxGroup/index.cjs +11 -13
  6. package/dist/components/CheckboxGroup/index.d.ts +3 -2
  7. package/dist/components/CheckboxGroup/index.js +5 -7
  8. package/dist/components/Chip/index.d.ts +1 -1
  9. package/dist/components/LineChart/index.cjs +2 -35
  10. package/dist/components/LineChart/index.js +2 -35
  11. package/dist/components/MenuV2/MenuContent.cjs +48 -14
  12. package/dist/components/MenuV2/MenuContent.d.ts +1 -0
  13. package/dist/components/MenuV2/MenuContent.js +44 -10
  14. package/dist/components/MenuV2/index.d.ts +1 -0
  15. package/dist/components/MenuV2/types.d.ts +1 -0
  16. package/dist/components/Modal/components/Dialog.cjs +5 -5
  17. package/dist/components/Modal/components/Dialog.js +5 -5
  18. package/dist/components/Notification/index.cjs +8 -4
  19. package/dist/components/Notification/index.js +10 -6
  20. package/dist/components/PieChart/index.cjs +5 -4
  21. package/dist/components/PieChart/index.js +5 -4
  22. package/dist/components/PieChart/types.d.ts +4 -0
  23. package/dist/components/RadioGroup/index.cjs +10 -12
  24. package/dist/components/RadioGroup/index.d.ts +3 -2
  25. package/dist/components/RadioGroup/index.js +4 -6
  26. package/dist/components/SelectableCardGroup/index.cjs +9 -13
  27. package/dist/components/SelectableCardGroup/index.d.ts +3 -2
  28. package/dist/components/SelectableCardGroup/index.js +5 -9
  29. package/dist/components/Separator/index.cjs +7 -9
  30. package/dist/components/Separator/index.js +7 -9
  31. package/dist/components/Slider/components/DoubleSlider.cjs +24 -23
  32. package/dist/components/Slider/components/DoubleSlider.js +24 -23
  33. package/dist/components/Slider/components/SingleSlider.cjs +23 -12
  34. package/dist/components/Slider/components/SingleSlider.js +23 -12
  35. package/dist/components/ToggleGroup/index.cjs +10 -12
  36. package/dist/components/ToggleGroup/index.d.ts +3 -2
  37. package/dist/components/ToggleGroup/index.js +4 -6
  38. package/dist/helpers/nivoTheme.cjs +22 -0
  39. package/dist/helpers/nivoTheme.d.ts +20 -0
  40. package/dist/helpers/nivoTheme.js +22 -0
  41. package/dist/theme/index.cjs +2 -2
  42. package/dist/theme/index.d.ts +9 -155
  43. package/dist/theme/index.js +2 -2
  44. package/dist/utils/responsive/utilities.cjs +1 -1
  45. package/dist/utils/responsive/utilities.js +5 -5
  46. package/package.json +8 -8
@@ -28,7 +28,7 @@ const StyledTextValue = /* @__PURE__ */ _styled__default.default(index$3.Text, p
28
28
  }) => double && isColumn ? null : theme.space["5"], ";align-self:", ({
29
29
  double,
30
30
  isColumn
31
- }) => double || !isColumn ? "center" : "end", ";" + (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/Slider/components/SingleSlider.tsx"],"names":[],"mappings":"AAgB0C","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Slider/components/SingleSlider.tsx","sourcesContent":["'use client'\n\nimport { useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport { useCallback, useEffect, useId, useMemo, useRef, useState } from 'react'\nimport { Label } from '../../Label'\nimport { NumberInputV2 } from '../../NumberInputV2'\nimport { Stack } from '../../Stack'\nimport { Text } from '../../Text'\nimport { THUMB_SIZE } from '../constant'\nimport { StyledTooltip, thumbStyle, trackStyle } from '../styles'\nimport type { SingleSliderProps } from '../types'\nimport { Options } from './Options'\n\nconst StyledTextValue = styled(Text, {\n  shouldForwardProp: prop => !['double', 'isColumn'].includes(prop),\n})<{ double: boolean; isColumn: boolean }>`\n  min-width: ${({ theme, double, isColumn }) => (double && isColumn ? null : theme.space['5'])};\n  align-self: ${({ double, isColumn }) => (double || !isColumn ? 'center' : 'end')};\n`\n\nconst SliderElement = styled('input', {\n  shouldForwardProp: prop => !['themeSlider', 'left'].includes(prop),\n})<{ themeSlider: string; disabled: boolean; left: number }>`\n    appearance: none;\n    height: ${({ theme }) => theme.space['1']};\n    width: 100%;\n    position: relative;\n    background-color: ${({ theme }) => theme.colors.neutral.borderWeak};\n\n    border-radius: ${({ theme }) => theme.radii.default};\n    background-image:linear-gradient(${({ theme }) => theme.colors.primary.border}, ${({ theme }) => theme.colors.primary.border});\n    background-repeat: no-repeat;\n    align-self: center;\n    outline: none;\n\n    &:focus {\n        &::-moz-range-thumb {\n          border: ${({ theme, disabled }) => (disabled ? null : `1.5px solid ${theme.colors.primary.border}`)};\n          box-shadow: ${({ theme, disabled }) => (disabled ? null : theme.shadows.focusPrimary)};\n        }\n\n        &::-webkit-slider-thumb {\n          border: ${({ theme, disabled }) => (disabled ? null : `1.5px solid ${theme.colors.primary.border}`)};\n          box-shadow: ${({ theme, disabled }) => (disabled ? null : theme.shadows.focusPrimary)};\n        }\n  }\n    &[data-error='true']{\n        background-image:linear-gradient(${({ theme }) => theme.colors.danger.backgroundStrong}, ${({ theme }) => theme.colors.danger.backgroundStrong});\n    }\n\n    &[data-direction='column'] {\n      align-self: baseline;\n    }\n\n    &[aria-disabled='true']{\n      background-image:linear-gradient(${({ theme }) => theme.colors.primary.borderDisabled}, ${({ theme }) => theme.colors.primary.borderDisabled});\n    }\n\n    /* Mozilla */\n    ::-moz-range-track {\n        ${trackStyle}\n    }\n    ::-moz-range-thumb {\n        ${({ theme, themeSlider, disabled, left }) => thumbStyle(theme, themeSlider, disabled, left, false)}\n    }\n\n    /* Other browsers */\n    ::-webkit-slider-runnable-track {\n        ${trackStyle}\n    }\n    ::-webkit-slider-thumb {\n        ${({ theme, themeSlider, disabled, left }) => thumbStyle(theme, themeSlider, disabled, left, false)}\n    }\n`\n\nconst StyledNumberInputV2 = styled(NumberInputV2)`\n  min-width: ${({ theme }) => theme.space['5']};\n`\n\nexport const SingleSlider = ({\n  name,\n  tooltip,\n  direction,\n  disabled,\n  error,\n  'data-testid': dataTestId,\n  value,\n  onChange,\n  min = 0,\n  max = 100,\n  step = 1,\n  id,\n  onBlur,\n  unit,\n  options,\n  onFocus,\n  className,\n  label,\n  input,\n  prefix,\n  suffix,\n  required,\n  'aria-label': ariaLabel,\n  tooltipPosition,\n}: SingleSliderProps) => {\n  const localId = useId()\n  const { theme } = useTheme()\n  const finalId = id ?? localId\n  const safeValue = value ?? min\n  const [selectedIndex, setSelectedIndex] = useState(safeValue)\n  const refSlider = useRef<HTMLInputElement>(null)\n  const [sliderWidth, setWidth] = useState(0)\n\n  useEffect(() => {\n    setWidth(Number(refSlider.current?.offsetWidth))\n  }, [refSlider])\n\n  const ticks = useMemo(() => {\n    if (options) {\n      return options.map((element, index) => ({\n        value: index,\n        label: element.label,\n      }))\n    }\n\n    return []\n  }, [options])\n\n  const internalOnChange = useCallback(\n    (newValue: number) => {\n      setSelectedIndex(newValue ?? min)\n      onChange?.(newValue ?? min)\n    },\n    [min, onChange],\n  )\n\n  // Make sure that min <= value <= max\n  useEffect(() => {\n    if (value < min) {\n      internalOnChange(min)\n    } else if (value > max) {\n      internalOnChange(max)\n    } else {\n      setSelectedIndex(() => value ?? min)\n    }\n  }, [value, max, min, internalOnChange])\n\n  // Get slider size\n  useEffect(() => {\n    const setWidthResize = () => {\n      setWidth(Number(refSlider.current?.offsetWidth))\n    }\n    window.addEventListener('resize', setWidthResize)\n\n    return () => {\n      window.removeEventListener('resize', setWidthResize)\n    }\n  }, [])\n\n  const leftPosition = useMemo(\n    () => ((selectedIndex - min) * 100) / (max - min),\n    [selectedIndex, max, min],\n  )\n\n  const getBackgroundSize = useMemo(\n    () => ({\n      backgroundSize: `${leftPosition}% 100%`,\n    }),\n    [leftPosition],\n  )\n\n  const styledValue = (valueNumber: string | number | null) =>\n    input && !options ? (\n      <StyledNumberInputV2\n        value={\n          typeof valueNumber === 'string'\n            ? Number.parseFloat(valueNumber)\n            : valueNumber\n        }\n        size=\"small\"\n        min={min}\n        aria-label=\"input\"\n        max={max}\n        step={step}\n        controls={false}\n        disabled={disabled}\n        data-testid={dataTestId ? `${dataTestId}-input` : 'slider-input'}\n        unit={typeof suffix === 'string' ? suffix : unit}\n        onChange={newVal => {\n          if (newVal) {\n            internalOnChange(newVal)\n          } else internalOnChange(0)\n        }}\n        onBlur={event => {\n          if (!event.target.value) internalOnChange(min)\n        }}\n      />\n    ) : (\n      <StyledTextValue\n        as=\"span\"\n        variant=\"bodySmall\"\n        sentiment=\"neutral\"\n        placement={direction === 'column' ? 'right' : 'center'}\n        double={false}\n        isColumn={direction === 'column'}\n        data-testid={dataTestId ? `${dataTestId}-value` : 'slider-value'}\n      >\n        {prefix}\n        {valueNumber}\n        {suffix ?? unit}\n      </StyledTextValue>\n    )\n\n  const tooltipText = useMemo(() => {\n    if (tooltip === true) {\n      return selectedIndex\n    }\n    if (tooltip) {\n      return tooltip\n    }\n\n    return undefined\n  }, [tooltip, selectedIndex])\n\n  const placementTooltip =\n    ((selectedIndex - min) / (max - min)) * (sliderWidth - THUMB_SIZE) +\n    THUMB_SIZE / 2 -\n    sliderWidth / 2\n\n  const valueToShow = options ? options[selectedIndex]?.value : selectedIndex\n\n  return (\n    <Stack gap={1} direction={direction} justifyContent=\"left\">\n      {label ? (\n        <Stack\n          justifyContent=\"space-between\"\n          direction=\"row\"\n          alignItems=\"center\"\n        >\n          <Label htmlFor={finalId} required={required}>\n            {label}\n          </Label>\n          {direction === 'column' ? styledValue(valueToShow) : null}\n        </Stack>\n      ) : null}\n\n      <Stack direction=\"row\" justifyContent=\"end\">\n        {direction === 'column' && !label ? styledValue(valueToShow) : null}\n      </Stack>\n      <Stack direction=\"column\" width=\"100%\" gap={1} justifyContent=\"center\">\n        <StyledTooltip\n          text={tooltipText}\n          placement={tooltipPosition}\n          left={placementTooltip}\n        >\n          <SliderElement\n            type=\"range\"\n            value={selectedIndex}\n            onChange={event => {\n              internalOnChange(Number.parseFloat(event.target.value))\n            }}\n            min={min}\n            max={max}\n            tabIndex={0}\n            step={step}\n            name={name}\n            disabled={!!disabled}\n            aria-disabled={disabled}\n            data-testid={dataTestId}\n            id={finalId}\n            onBlur={onBlur}\n            onFocus={onFocus}\n            aria-label={ariaLabel ?? name}\n            className={className}\n            style={getBackgroundSize}\n            data-error={!!error}\n            data-direction={direction}\n            themeSlider={theme}\n            ref={refSlider}\n            left={leftPosition}\n          />\n        </StyledTooltip>\n        {options ? (\n          <Options\n            ticks={ticks}\n            min={min}\n            max={max}\n            sliderWidth={sliderWidth}\n            value={selectedIndex}\n            step={step}\n          />\n        ) : null}\n      </Stack>\n      {direction === 'row' ? styledValue(valueToShow) : null}\n    </Stack>\n  )\n}\n"]} */"));
31
+ }) => double || !isColumn ? "center" : "end", ";" + (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/Slider/components/SingleSlider.tsx"],"names":[],"mappings":"AAgB0C","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Slider/components/SingleSlider.tsx","sourcesContent":["'use client'\n\nimport { useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport { useCallback, useEffect, useId, useMemo, useRef, useState } from 'react'\nimport { Label } from '../../Label'\nimport { NumberInputV2 } from '../../NumberInputV2'\nimport { Stack } from '../../Stack'\nimport { Text } from '../../Text'\nimport { THUMB_SIZE } from '../constant'\nimport { StyledTooltip, thumbStyle, trackStyle } from '../styles'\nimport type { SingleSliderProps } from '../types'\nimport { Options } from './Options'\n\nconst StyledTextValue = styled(Text, {\n  shouldForwardProp: prop => !['double', 'isColumn'].includes(prop),\n})<{ double: boolean; isColumn: boolean }>`\n  min-width: ${({ theme, double, isColumn }) => (double && isColumn ? null : theme.space['5'])};\n  align-self: ${({ double, isColumn }) => (double || !isColumn ? 'center' : 'end')};\n`\n\nconst SliderElement = styled('input', {\n  shouldForwardProp: prop => !['themeSlider', 'left'].includes(prop),\n})<{ themeSlider: string; disabled: boolean; left: number }>`\n    appearance: none;\n    height: ${({ theme }) => theme.space['1']};\n    width: 100%;\n    position: relative;\n    background-color: ${({ theme }) => theme.colors.neutral.borderWeak};\n\n    border-radius: ${({ theme }) => theme.radii.default};\n    background-image:linear-gradient(${({ theme }) => theme.colors.primary.border}, ${({ theme }) => theme.colors.primary.border});\n    background-repeat: no-repeat;\n    align-self: center;\n    outline: none;\n\n    &:focus {\n        &::-moz-range-thumb {\n          border: ${({ theme, disabled }) => (disabled ? null : `1.5px solid ${theme.colors.primary.border}`)};\n          box-shadow: ${({ theme, disabled }) => (disabled ? null : theme.shadows.focusPrimary)};\n        }\n\n        &::-webkit-slider-thumb {\n          border: ${({ theme, disabled }) => (disabled ? null : `1.5px solid ${theme.colors.primary.border}`)};\n          box-shadow: ${({ theme, disabled }) => (disabled ? null : theme.shadows.focusPrimary)};\n        }\n  }\n    &[data-error='true']{\n        background-image:linear-gradient(${({ theme }) => theme.colors.danger.backgroundStrong}, ${({ theme }) => theme.colors.danger.backgroundStrong});\n    }\n\n    &[data-direction='column'] {\n      align-self: baseline;\n    }\n\n    &[aria-disabled='true']{\n      background-image:linear-gradient(${({ theme }) => theme.colors.primary.borderDisabled}, ${({ theme }) => theme.colors.primary.borderDisabled});\n    }\n\n    /* Mozilla */\n    ::-moz-range-track {\n        ${trackStyle}\n    }\n    ::-moz-range-thumb {\n        ${({ theme, themeSlider, disabled, left }) => thumbStyle(theme, themeSlider, disabled, left, false)}\n    }\n\n    /* Other browsers */\n    ::-webkit-slider-runnable-track {\n        ${trackStyle}\n    }\n    ::-webkit-slider-thumb {\n        ${({ theme, themeSlider, disabled, left }) => thumbStyle(theme, themeSlider, disabled, left, false)}\n    }\n`\n\nconst StyledNumberInputV2 = styled(NumberInputV2)`\n  min-width: ${({ theme }) => theme.space['5']};\n`\n\nexport const SingleSlider = ({\n  name,\n  tooltip,\n  direction,\n  disabled,\n  error,\n  'data-testid': dataTestId,\n  value,\n  onChange,\n  min = 0,\n  max = 100,\n  step = 1,\n  id,\n  onBlur,\n  unit,\n  options,\n  onFocus,\n  className,\n  label,\n  input,\n  prefix,\n  suffix,\n  required,\n  'aria-label': ariaLabel,\n  tooltipPosition,\n}: SingleSliderProps) => {\n  const localId = useId()\n  const { theme } = useTheme()\n  const finalId = id ?? localId\n  const safeValue = value ?? min\n  const [selectedIndex, setSelectedIndex] = useState(safeValue)\n  const refSlider = useRef<HTMLInputElement>(null)\n  const [sliderWidth, setWidth] = useState(0)\n  const [inputValue, setInputValue] = useState<number | null>(safeValue)\n\n  useEffect(() => {\n    setInputValue(selectedIndex)\n  }, [selectedIndex])\n\n  useEffect(() => {\n    setWidth(Number(refSlider.current?.offsetWidth))\n  }, [refSlider])\n\n  const ticks = useMemo(() => {\n    if (options) {\n      return options.map((element, index) => ({\n        value: index,\n        label: element.label,\n      }))\n    }\n\n    return []\n  }, [options])\n\n  const internalOnChange = useCallback(\n    (newValue: number) => {\n      if (!newValue) {\n        setSelectedIndex(min)\n        onChange?.(min)\n      } else if (newValue < min) {\n        setSelectedIndex(min)\n        onChange?.(min)\n      } else if (newValue > max) {\n        setSelectedIndex(max)\n        onChange?.(max)\n      } else {\n        setSelectedIndex(newValue)\n        onChange?.(newValue)\n      }\n    },\n    [max, min, onChange],\n  )\n\n  // Make sure that min <= value <= max\n  useEffect(() => {\n    if (value < min) {\n      internalOnChange(min)\n    } else if (value > max) {\n      internalOnChange(max)\n    } else {\n      setSelectedIndex(() => value ?? min)\n    }\n  }, [value, max, min, internalOnChange])\n\n  // Get slider size\n  useEffect(() => {\n    const setWidthResize = () => {\n      setWidth(Number(refSlider.current?.offsetWidth))\n    }\n    window.addEventListener('resize', setWidthResize)\n\n    return () => {\n      window.removeEventListener('resize', setWidthResize)\n    }\n  }, [])\n\n  const leftPosition = useMemo(\n    () => ((selectedIndex - min) * 100) / (max - min),\n    [selectedIndex, max, min],\n  )\n\n  const getBackgroundSize = useMemo(\n    () => ({\n      backgroundSize: `${leftPosition}% 100%`,\n    }),\n    [leftPosition],\n  )\n\n  const styledValue = (valueNumber: string | number | null) =>\n    input && !options ? (\n      <StyledNumberInputV2\n        value={inputValue}\n        size=\"small\"\n        min={min}\n        aria-label=\"input\"\n        max={max}\n        step={step}\n        controls={false}\n        disabled={disabled}\n        data-testid={dataTestId ? `${dataTestId}-input` : 'slider-input'}\n        unit={typeof suffix === 'string' ? suffix : unit}\n        onChange={setInputValue}\n        onBlur={event => {\n          internalOnChange(Number.parseFloat(event.target.value))\n        }}\n      />\n    ) : (\n      <StyledTextValue\n        as=\"span\"\n        variant=\"bodySmall\"\n        sentiment=\"neutral\"\n        placement={direction === 'column' ? 'right' : 'center'}\n        double={false}\n        isColumn={direction === 'column'}\n        data-testid={dataTestId ? `${dataTestId}-value` : 'slider-value'}\n      >\n        {prefix}\n        {valueNumber}\n        {suffix ?? unit}\n      </StyledTextValue>\n    )\n\n  const tooltipText = useMemo(() => {\n    if (tooltip === true) {\n      return selectedIndex\n    }\n    if (tooltip) {\n      return tooltip\n    }\n\n    return undefined\n  }, [tooltip, selectedIndex])\n\n  const placementTooltip =\n    ((selectedIndex - min) / (max - min)) * (sliderWidth - THUMB_SIZE) +\n    THUMB_SIZE / 2 -\n    sliderWidth / 2\n\n  const valueToShow = options ? options[selectedIndex]?.value : selectedIndex\n\n  return (\n    <Stack gap={1} direction={direction} justifyContent=\"left\">\n      {label ? (\n        <Stack\n          justifyContent=\"space-between\"\n          direction=\"row\"\n          alignItems=\"center\"\n        >\n          <Label htmlFor={finalId} required={required}>\n            {label}\n          </Label>\n          {direction === 'column' ? styledValue(valueToShow) : null}\n        </Stack>\n      ) : null}\n\n      <Stack direction=\"row\" justifyContent=\"end\">\n        {direction === 'column' && !label ? styledValue(valueToShow) : null}\n      </Stack>\n      <Stack direction=\"column\" width=\"100%\" gap={1} justifyContent=\"center\">\n        <StyledTooltip\n          text={tooltipText}\n          placement={tooltipPosition}\n          left={placementTooltip}\n        >\n          <SliderElement\n            type=\"range\"\n            value={selectedIndex}\n            onChange={event => {\n              internalOnChange(Number.parseFloat(event.target.value))\n            }}\n            min={min}\n            max={max}\n            tabIndex={0}\n            step={step}\n            name={name}\n            disabled={!!disabled}\n            aria-disabled={disabled}\n            data-testid={dataTestId}\n            id={finalId}\n            onBlur={onBlur}\n            onFocus={onFocus}\n            aria-label={ariaLabel ?? name}\n            className={className}\n            style={getBackgroundSize}\n            data-error={!!error}\n            data-direction={direction}\n            themeSlider={theme}\n            ref={refSlider}\n            left={leftPosition}\n          />\n        </StyledTooltip>\n        {options ? (\n          <Options\n            ticks={ticks}\n            min={min}\n            max={max}\n            sliderWidth={sliderWidth}\n            value={selectedIndex}\n            step={step}\n          />\n        ) : null}\n      </Stack>\n      {direction === 'row' ? styledValue(valueToShow) : null}\n    </Stack>\n  )\n}\n"]} */"));
32
32
  const SliderElement = /* @__PURE__ */ _styled__default.default("input", process.env.NODE_ENV === "production" ? {
33
33
  shouldForwardProp: (prop) => !["themeSlider", "left"].includes(prop),
34
34
  target: "elmzsg31"
@@ -76,7 +76,7 @@ const SliderElement = /* @__PURE__ */ _styled__default.default("input", process.
76
76
  themeSlider,
77
77
  disabled,
78
78
  left
79
- }) => styles.thumbStyle(theme, themeSlider, disabled, left, false), ";}" + (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/Slider/components/SingleSlider.tsx"],"names":[],"mappings":"AAuB4D","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Slider/components/SingleSlider.tsx","sourcesContent":["'use client'\n\nimport { useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport { useCallback, useEffect, useId, useMemo, useRef, useState } from 'react'\nimport { Label } from '../../Label'\nimport { NumberInputV2 } from '../../NumberInputV2'\nimport { Stack } from '../../Stack'\nimport { Text } from '../../Text'\nimport { THUMB_SIZE } from '../constant'\nimport { StyledTooltip, thumbStyle, trackStyle } from '../styles'\nimport type { SingleSliderProps } from '../types'\nimport { Options } from './Options'\n\nconst StyledTextValue = styled(Text, {\n  shouldForwardProp: prop => !['double', 'isColumn'].includes(prop),\n})<{ double: boolean; isColumn: boolean }>`\n  min-width: ${({ theme, double, isColumn }) => (double && isColumn ? null : theme.space['5'])};\n  align-self: ${({ double, isColumn }) => (double || !isColumn ? 'center' : 'end')};\n`\n\nconst SliderElement = styled('input', {\n  shouldForwardProp: prop => !['themeSlider', 'left'].includes(prop),\n})<{ themeSlider: string; disabled: boolean; left: number }>`\n    appearance: none;\n    height: ${({ theme }) => theme.space['1']};\n    width: 100%;\n    position: relative;\n    background-color: ${({ theme }) => theme.colors.neutral.borderWeak};\n\n    border-radius: ${({ theme }) => theme.radii.default};\n    background-image:linear-gradient(${({ theme }) => theme.colors.primary.border}, ${({ theme }) => theme.colors.primary.border});\n    background-repeat: no-repeat;\n    align-self: center;\n    outline: none;\n\n    &:focus {\n        &::-moz-range-thumb {\n          border: ${({ theme, disabled }) => (disabled ? null : `1.5px solid ${theme.colors.primary.border}`)};\n          box-shadow: ${({ theme, disabled }) => (disabled ? null : theme.shadows.focusPrimary)};\n        }\n\n        &::-webkit-slider-thumb {\n          border: ${({ theme, disabled }) => (disabled ? null : `1.5px solid ${theme.colors.primary.border}`)};\n          box-shadow: ${({ theme, disabled }) => (disabled ? null : theme.shadows.focusPrimary)};\n        }\n  }\n    &[data-error='true']{\n        background-image:linear-gradient(${({ theme }) => theme.colors.danger.backgroundStrong}, ${({ theme }) => theme.colors.danger.backgroundStrong});\n    }\n\n    &[data-direction='column'] {\n      align-self: baseline;\n    }\n\n    &[aria-disabled='true']{\n      background-image:linear-gradient(${({ theme }) => theme.colors.primary.borderDisabled}, ${({ theme }) => theme.colors.primary.borderDisabled});\n    }\n\n    /* Mozilla */\n    ::-moz-range-track {\n        ${trackStyle}\n    }\n    ::-moz-range-thumb {\n        ${({ theme, themeSlider, disabled, left }) => thumbStyle(theme, themeSlider, disabled, left, false)}\n    }\n\n    /* Other browsers */\n    ::-webkit-slider-runnable-track {\n        ${trackStyle}\n    }\n    ::-webkit-slider-thumb {\n        ${({ theme, themeSlider, disabled, left }) => thumbStyle(theme, themeSlider, disabled, left, false)}\n    }\n`\n\nconst StyledNumberInputV2 = styled(NumberInputV2)`\n  min-width: ${({ theme }) => theme.space['5']};\n`\n\nexport const SingleSlider = ({\n  name,\n  tooltip,\n  direction,\n  disabled,\n  error,\n  'data-testid': dataTestId,\n  value,\n  onChange,\n  min = 0,\n  max = 100,\n  step = 1,\n  id,\n  onBlur,\n  unit,\n  options,\n  onFocus,\n  className,\n  label,\n  input,\n  prefix,\n  suffix,\n  required,\n  'aria-label': ariaLabel,\n  tooltipPosition,\n}: SingleSliderProps) => {\n  const localId = useId()\n  const { theme } = useTheme()\n  const finalId = id ?? localId\n  const safeValue = value ?? min\n  const [selectedIndex, setSelectedIndex] = useState(safeValue)\n  const refSlider = useRef<HTMLInputElement>(null)\n  const [sliderWidth, setWidth] = useState(0)\n\n  useEffect(() => {\n    setWidth(Number(refSlider.current?.offsetWidth))\n  }, [refSlider])\n\n  const ticks = useMemo(() => {\n    if (options) {\n      return options.map((element, index) => ({\n        value: index,\n        label: element.label,\n      }))\n    }\n\n    return []\n  }, [options])\n\n  const internalOnChange = useCallback(\n    (newValue: number) => {\n      setSelectedIndex(newValue ?? min)\n      onChange?.(newValue ?? min)\n    },\n    [min, onChange],\n  )\n\n  // Make sure that min <= value <= max\n  useEffect(() => {\n    if (value < min) {\n      internalOnChange(min)\n    } else if (value > max) {\n      internalOnChange(max)\n    } else {\n      setSelectedIndex(() => value ?? min)\n    }\n  }, [value, max, min, internalOnChange])\n\n  // Get slider size\n  useEffect(() => {\n    const setWidthResize = () => {\n      setWidth(Number(refSlider.current?.offsetWidth))\n    }\n    window.addEventListener('resize', setWidthResize)\n\n    return () => {\n      window.removeEventListener('resize', setWidthResize)\n    }\n  }, [])\n\n  const leftPosition = useMemo(\n    () => ((selectedIndex - min) * 100) / (max - min),\n    [selectedIndex, max, min],\n  )\n\n  const getBackgroundSize = useMemo(\n    () => ({\n      backgroundSize: `${leftPosition}% 100%`,\n    }),\n    [leftPosition],\n  )\n\n  const styledValue = (valueNumber: string | number | null) =>\n    input && !options ? (\n      <StyledNumberInputV2\n        value={\n          typeof valueNumber === 'string'\n            ? Number.parseFloat(valueNumber)\n            : valueNumber\n        }\n        size=\"small\"\n        min={min}\n        aria-label=\"input\"\n        max={max}\n        step={step}\n        controls={false}\n        disabled={disabled}\n        data-testid={dataTestId ? `${dataTestId}-input` : 'slider-input'}\n        unit={typeof suffix === 'string' ? suffix : unit}\n        onChange={newVal => {\n          if (newVal) {\n            internalOnChange(newVal)\n          } else internalOnChange(0)\n        }}\n        onBlur={event => {\n          if (!event.target.value) internalOnChange(min)\n        }}\n      />\n    ) : (\n      <StyledTextValue\n        as=\"span\"\n        variant=\"bodySmall\"\n        sentiment=\"neutral\"\n        placement={direction === 'column' ? 'right' : 'center'}\n        double={false}\n        isColumn={direction === 'column'}\n        data-testid={dataTestId ? `${dataTestId}-value` : 'slider-value'}\n      >\n        {prefix}\n        {valueNumber}\n        {suffix ?? unit}\n      </StyledTextValue>\n    )\n\n  const tooltipText = useMemo(() => {\n    if (tooltip === true) {\n      return selectedIndex\n    }\n    if (tooltip) {\n      return tooltip\n    }\n\n    return undefined\n  }, [tooltip, selectedIndex])\n\n  const placementTooltip =\n    ((selectedIndex - min) / (max - min)) * (sliderWidth - THUMB_SIZE) +\n    THUMB_SIZE / 2 -\n    sliderWidth / 2\n\n  const valueToShow = options ? options[selectedIndex]?.value : selectedIndex\n\n  return (\n    <Stack gap={1} direction={direction} justifyContent=\"left\">\n      {label ? (\n        <Stack\n          justifyContent=\"space-between\"\n          direction=\"row\"\n          alignItems=\"center\"\n        >\n          <Label htmlFor={finalId} required={required}>\n            {label}\n          </Label>\n          {direction === 'column' ? styledValue(valueToShow) : null}\n        </Stack>\n      ) : null}\n\n      <Stack direction=\"row\" justifyContent=\"end\">\n        {direction === 'column' && !label ? styledValue(valueToShow) : null}\n      </Stack>\n      <Stack direction=\"column\" width=\"100%\" gap={1} justifyContent=\"center\">\n        <StyledTooltip\n          text={tooltipText}\n          placement={tooltipPosition}\n          left={placementTooltip}\n        >\n          <SliderElement\n            type=\"range\"\n            value={selectedIndex}\n            onChange={event => {\n              internalOnChange(Number.parseFloat(event.target.value))\n            }}\n            min={min}\n            max={max}\n            tabIndex={0}\n            step={step}\n            name={name}\n            disabled={!!disabled}\n            aria-disabled={disabled}\n            data-testid={dataTestId}\n            id={finalId}\n            onBlur={onBlur}\n            onFocus={onFocus}\n            aria-label={ariaLabel ?? name}\n            className={className}\n            style={getBackgroundSize}\n            data-error={!!error}\n            data-direction={direction}\n            themeSlider={theme}\n            ref={refSlider}\n            left={leftPosition}\n          />\n        </StyledTooltip>\n        {options ? (\n          <Options\n            ticks={ticks}\n            min={min}\n            max={max}\n            sliderWidth={sliderWidth}\n            value={selectedIndex}\n            step={step}\n          />\n        ) : null}\n      </Stack>\n      {direction === 'row' ? styledValue(valueToShow) : null}\n    </Stack>\n  )\n}\n"]} */"));
79
+ }) => styles.thumbStyle(theme, themeSlider, disabled, left, false), ";}" + (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/Slider/components/SingleSlider.tsx"],"names":[],"mappings":"AAuB4D","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Slider/components/SingleSlider.tsx","sourcesContent":["'use client'\n\nimport { useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport { useCallback, useEffect, useId, useMemo, useRef, useState } from 'react'\nimport { Label } from '../../Label'\nimport { NumberInputV2 } from '../../NumberInputV2'\nimport { Stack } from '../../Stack'\nimport { Text } from '../../Text'\nimport { THUMB_SIZE } from '../constant'\nimport { StyledTooltip, thumbStyle, trackStyle } from '../styles'\nimport type { SingleSliderProps } from '../types'\nimport { Options } from './Options'\n\nconst StyledTextValue = styled(Text, {\n  shouldForwardProp: prop => !['double', 'isColumn'].includes(prop),\n})<{ double: boolean; isColumn: boolean }>`\n  min-width: ${({ theme, double, isColumn }) => (double && isColumn ? null : theme.space['5'])};\n  align-self: ${({ double, isColumn }) => (double || !isColumn ? 'center' : 'end')};\n`\n\nconst SliderElement = styled('input', {\n  shouldForwardProp: prop => !['themeSlider', 'left'].includes(prop),\n})<{ themeSlider: string; disabled: boolean; left: number }>`\n    appearance: none;\n    height: ${({ theme }) => theme.space['1']};\n    width: 100%;\n    position: relative;\n    background-color: ${({ theme }) => theme.colors.neutral.borderWeak};\n\n    border-radius: ${({ theme }) => theme.radii.default};\n    background-image:linear-gradient(${({ theme }) => theme.colors.primary.border}, ${({ theme }) => theme.colors.primary.border});\n    background-repeat: no-repeat;\n    align-self: center;\n    outline: none;\n\n    &:focus {\n        &::-moz-range-thumb {\n          border: ${({ theme, disabled }) => (disabled ? null : `1.5px solid ${theme.colors.primary.border}`)};\n          box-shadow: ${({ theme, disabled }) => (disabled ? null : theme.shadows.focusPrimary)};\n        }\n\n        &::-webkit-slider-thumb {\n          border: ${({ theme, disabled }) => (disabled ? null : `1.5px solid ${theme.colors.primary.border}`)};\n          box-shadow: ${({ theme, disabled }) => (disabled ? null : theme.shadows.focusPrimary)};\n        }\n  }\n    &[data-error='true']{\n        background-image:linear-gradient(${({ theme }) => theme.colors.danger.backgroundStrong}, ${({ theme }) => theme.colors.danger.backgroundStrong});\n    }\n\n    &[data-direction='column'] {\n      align-self: baseline;\n    }\n\n    &[aria-disabled='true']{\n      background-image:linear-gradient(${({ theme }) => theme.colors.primary.borderDisabled}, ${({ theme }) => theme.colors.primary.borderDisabled});\n    }\n\n    /* Mozilla */\n    ::-moz-range-track {\n        ${trackStyle}\n    }\n    ::-moz-range-thumb {\n        ${({ theme, themeSlider, disabled, left }) => thumbStyle(theme, themeSlider, disabled, left, false)}\n    }\n\n    /* Other browsers */\n    ::-webkit-slider-runnable-track {\n        ${trackStyle}\n    }\n    ::-webkit-slider-thumb {\n        ${({ theme, themeSlider, disabled, left }) => thumbStyle(theme, themeSlider, disabled, left, false)}\n    }\n`\n\nconst StyledNumberInputV2 = styled(NumberInputV2)`\n  min-width: ${({ theme }) => theme.space['5']};\n`\n\nexport const SingleSlider = ({\n  name,\n  tooltip,\n  direction,\n  disabled,\n  error,\n  'data-testid': dataTestId,\n  value,\n  onChange,\n  min = 0,\n  max = 100,\n  step = 1,\n  id,\n  onBlur,\n  unit,\n  options,\n  onFocus,\n  className,\n  label,\n  input,\n  prefix,\n  suffix,\n  required,\n  'aria-label': ariaLabel,\n  tooltipPosition,\n}: SingleSliderProps) => {\n  const localId = useId()\n  const { theme } = useTheme()\n  const finalId = id ?? localId\n  const safeValue = value ?? min\n  const [selectedIndex, setSelectedIndex] = useState(safeValue)\n  const refSlider = useRef<HTMLInputElement>(null)\n  const [sliderWidth, setWidth] = useState(0)\n  const [inputValue, setInputValue] = useState<number | null>(safeValue)\n\n  useEffect(() => {\n    setInputValue(selectedIndex)\n  }, [selectedIndex])\n\n  useEffect(() => {\n    setWidth(Number(refSlider.current?.offsetWidth))\n  }, [refSlider])\n\n  const ticks = useMemo(() => {\n    if (options) {\n      return options.map((element, index) => ({\n        value: index,\n        label: element.label,\n      }))\n    }\n\n    return []\n  }, [options])\n\n  const internalOnChange = useCallback(\n    (newValue: number) => {\n      if (!newValue) {\n        setSelectedIndex(min)\n        onChange?.(min)\n      } else if (newValue < min) {\n        setSelectedIndex(min)\n        onChange?.(min)\n      } else if (newValue > max) {\n        setSelectedIndex(max)\n        onChange?.(max)\n      } else {\n        setSelectedIndex(newValue)\n        onChange?.(newValue)\n      }\n    },\n    [max, min, onChange],\n  )\n\n  // Make sure that min <= value <= max\n  useEffect(() => {\n    if (value < min) {\n      internalOnChange(min)\n    } else if (value > max) {\n      internalOnChange(max)\n    } else {\n      setSelectedIndex(() => value ?? min)\n    }\n  }, [value, max, min, internalOnChange])\n\n  // Get slider size\n  useEffect(() => {\n    const setWidthResize = () => {\n      setWidth(Number(refSlider.current?.offsetWidth))\n    }\n    window.addEventListener('resize', setWidthResize)\n\n    return () => {\n      window.removeEventListener('resize', setWidthResize)\n    }\n  }, [])\n\n  const leftPosition = useMemo(\n    () => ((selectedIndex - min) * 100) / (max - min),\n    [selectedIndex, max, min],\n  )\n\n  const getBackgroundSize = useMemo(\n    () => ({\n      backgroundSize: `${leftPosition}% 100%`,\n    }),\n    [leftPosition],\n  )\n\n  const styledValue = (valueNumber: string | number | null) =>\n    input && !options ? (\n      <StyledNumberInputV2\n        value={inputValue}\n        size=\"small\"\n        min={min}\n        aria-label=\"input\"\n        max={max}\n        step={step}\n        controls={false}\n        disabled={disabled}\n        data-testid={dataTestId ? `${dataTestId}-input` : 'slider-input'}\n        unit={typeof suffix === 'string' ? suffix : unit}\n        onChange={setInputValue}\n        onBlur={event => {\n          internalOnChange(Number.parseFloat(event.target.value))\n        }}\n      />\n    ) : (\n      <StyledTextValue\n        as=\"span\"\n        variant=\"bodySmall\"\n        sentiment=\"neutral\"\n        placement={direction === 'column' ? 'right' : 'center'}\n        double={false}\n        isColumn={direction === 'column'}\n        data-testid={dataTestId ? `${dataTestId}-value` : 'slider-value'}\n      >\n        {prefix}\n        {valueNumber}\n        {suffix ?? unit}\n      </StyledTextValue>\n    )\n\n  const tooltipText = useMemo(() => {\n    if (tooltip === true) {\n      return selectedIndex\n    }\n    if (tooltip) {\n      return tooltip\n    }\n\n    return undefined\n  }, [tooltip, selectedIndex])\n\n  const placementTooltip =\n    ((selectedIndex - min) / (max - min)) * (sliderWidth - THUMB_SIZE) +\n    THUMB_SIZE / 2 -\n    sliderWidth / 2\n\n  const valueToShow = options ? options[selectedIndex]?.value : selectedIndex\n\n  return (\n    <Stack gap={1} direction={direction} justifyContent=\"left\">\n      {label ? (\n        <Stack\n          justifyContent=\"space-between\"\n          direction=\"row\"\n          alignItems=\"center\"\n        >\n          <Label htmlFor={finalId} required={required}>\n            {label}\n          </Label>\n          {direction === 'column' ? styledValue(valueToShow) : null}\n        </Stack>\n      ) : null}\n\n      <Stack direction=\"row\" justifyContent=\"end\">\n        {direction === 'column' && !label ? styledValue(valueToShow) : null}\n      </Stack>\n      <Stack direction=\"column\" width=\"100%\" gap={1} justifyContent=\"center\">\n        <StyledTooltip\n          text={tooltipText}\n          placement={tooltipPosition}\n          left={placementTooltip}\n        >\n          <SliderElement\n            type=\"range\"\n            value={selectedIndex}\n            onChange={event => {\n              internalOnChange(Number.parseFloat(event.target.value))\n            }}\n            min={min}\n            max={max}\n            tabIndex={0}\n            step={step}\n            name={name}\n            disabled={!!disabled}\n            aria-disabled={disabled}\n            data-testid={dataTestId}\n            id={finalId}\n            onBlur={onBlur}\n            onFocus={onFocus}\n            aria-label={ariaLabel ?? name}\n            className={className}\n            style={getBackgroundSize}\n            data-error={!!error}\n            data-direction={direction}\n            themeSlider={theme}\n            ref={refSlider}\n            left={leftPosition}\n          />\n        </StyledTooltip>\n        {options ? (\n          <Options\n            ticks={ticks}\n            min={min}\n            max={max}\n            sliderWidth={sliderWidth}\n            value={selectedIndex}\n            step={step}\n          />\n        ) : null}\n      </Stack>\n      {direction === 'row' ? styledValue(valueToShow) : null}\n    </Stack>\n  )\n}\n"]} */"));
80
80
  const StyledNumberInputV2 = /* @__PURE__ */ _styled__default.default(index$2.NumberInputV2, process.env.NODE_ENV === "production" ? {
81
81
  target: "elmzsg30"
82
82
  } : {
@@ -84,7 +84,7 @@ const StyledNumberInputV2 = /* @__PURE__ */ _styled__default.default(index$2.Num
84
84
  label: "StyledNumberInputV2"
85
85
  })("min-width:", ({
86
86
  theme
87
- }) => theme.space["5"], ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Slider/components/SingleSlider.tsx"],"names":[],"mappings":"AA4EiD","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Slider/components/SingleSlider.tsx","sourcesContent":["'use client'\n\nimport { useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport { useCallback, useEffect, useId, useMemo, useRef, useState } from 'react'\nimport { Label } from '../../Label'\nimport { NumberInputV2 } from '../../NumberInputV2'\nimport { Stack } from '../../Stack'\nimport { Text } from '../../Text'\nimport { THUMB_SIZE } from '../constant'\nimport { StyledTooltip, thumbStyle, trackStyle } from '../styles'\nimport type { SingleSliderProps } from '../types'\nimport { Options } from './Options'\n\nconst StyledTextValue = styled(Text, {\n  shouldForwardProp: prop => !['double', 'isColumn'].includes(prop),\n})<{ double: boolean; isColumn: boolean }>`\n  min-width: ${({ theme, double, isColumn }) => (double && isColumn ? null : theme.space['5'])};\n  align-self: ${({ double, isColumn }) => (double || !isColumn ? 'center' : 'end')};\n`\n\nconst SliderElement = styled('input', {\n  shouldForwardProp: prop => !['themeSlider', 'left'].includes(prop),\n})<{ themeSlider: string; disabled: boolean; left: number }>`\n    appearance: none;\n    height: ${({ theme }) => theme.space['1']};\n    width: 100%;\n    position: relative;\n    background-color: ${({ theme }) => theme.colors.neutral.borderWeak};\n\n    border-radius: ${({ theme }) => theme.radii.default};\n    background-image:linear-gradient(${({ theme }) => theme.colors.primary.border}, ${({ theme }) => theme.colors.primary.border});\n    background-repeat: no-repeat;\n    align-self: center;\n    outline: none;\n\n    &:focus {\n        &::-moz-range-thumb {\n          border: ${({ theme, disabled }) => (disabled ? null : `1.5px solid ${theme.colors.primary.border}`)};\n          box-shadow: ${({ theme, disabled }) => (disabled ? null : theme.shadows.focusPrimary)};\n        }\n\n        &::-webkit-slider-thumb {\n          border: ${({ theme, disabled }) => (disabled ? null : `1.5px solid ${theme.colors.primary.border}`)};\n          box-shadow: ${({ theme, disabled }) => (disabled ? null : theme.shadows.focusPrimary)};\n        }\n  }\n    &[data-error='true']{\n        background-image:linear-gradient(${({ theme }) => theme.colors.danger.backgroundStrong}, ${({ theme }) => theme.colors.danger.backgroundStrong});\n    }\n\n    &[data-direction='column'] {\n      align-self: baseline;\n    }\n\n    &[aria-disabled='true']{\n      background-image:linear-gradient(${({ theme }) => theme.colors.primary.borderDisabled}, ${({ theme }) => theme.colors.primary.borderDisabled});\n    }\n\n    /* Mozilla */\n    ::-moz-range-track {\n        ${trackStyle}\n    }\n    ::-moz-range-thumb {\n        ${({ theme, themeSlider, disabled, left }) => thumbStyle(theme, themeSlider, disabled, left, false)}\n    }\n\n    /* Other browsers */\n    ::-webkit-slider-runnable-track {\n        ${trackStyle}\n    }\n    ::-webkit-slider-thumb {\n        ${({ theme, themeSlider, disabled, left }) => thumbStyle(theme, themeSlider, disabled, left, false)}\n    }\n`\n\nconst StyledNumberInputV2 = styled(NumberInputV2)`\n  min-width: ${({ theme }) => theme.space['5']};\n`\n\nexport const SingleSlider = ({\n  name,\n  tooltip,\n  direction,\n  disabled,\n  error,\n  'data-testid': dataTestId,\n  value,\n  onChange,\n  min = 0,\n  max = 100,\n  step = 1,\n  id,\n  onBlur,\n  unit,\n  options,\n  onFocus,\n  className,\n  label,\n  input,\n  prefix,\n  suffix,\n  required,\n  'aria-label': ariaLabel,\n  tooltipPosition,\n}: SingleSliderProps) => {\n  const localId = useId()\n  const { theme } = useTheme()\n  const finalId = id ?? localId\n  const safeValue = value ?? min\n  const [selectedIndex, setSelectedIndex] = useState(safeValue)\n  const refSlider = useRef<HTMLInputElement>(null)\n  const [sliderWidth, setWidth] = useState(0)\n\n  useEffect(() => {\n    setWidth(Number(refSlider.current?.offsetWidth))\n  }, [refSlider])\n\n  const ticks = useMemo(() => {\n    if (options) {\n      return options.map((element, index) => ({\n        value: index,\n        label: element.label,\n      }))\n    }\n\n    return []\n  }, [options])\n\n  const internalOnChange = useCallback(\n    (newValue: number) => {\n      setSelectedIndex(newValue ?? min)\n      onChange?.(newValue ?? min)\n    },\n    [min, onChange],\n  )\n\n  // Make sure that min <= value <= max\n  useEffect(() => {\n    if (value < min) {\n      internalOnChange(min)\n    } else if (value > max) {\n      internalOnChange(max)\n    } else {\n      setSelectedIndex(() => value ?? min)\n    }\n  }, [value, max, min, internalOnChange])\n\n  // Get slider size\n  useEffect(() => {\n    const setWidthResize = () => {\n      setWidth(Number(refSlider.current?.offsetWidth))\n    }\n    window.addEventListener('resize', setWidthResize)\n\n    return () => {\n      window.removeEventListener('resize', setWidthResize)\n    }\n  }, [])\n\n  const leftPosition = useMemo(\n    () => ((selectedIndex - min) * 100) / (max - min),\n    [selectedIndex, max, min],\n  )\n\n  const getBackgroundSize = useMemo(\n    () => ({\n      backgroundSize: `${leftPosition}% 100%`,\n    }),\n    [leftPosition],\n  )\n\n  const styledValue = (valueNumber: string | number | null) =>\n    input && !options ? (\n      <StyledNumberInputV2\n        value={\n          typeof valueNumber === 'string'\n            ? Number.parseFloat(valueNumber)\n            : valueNumber\n        }\n        size=\"small\"\n        min={min}\n        aria-label=\"input\"\n        max={max}\n        step={step}\n        controls={false}\n        disabled={disabled}\n        data-testid={dataTestId ? `${dataTestId}-input` : 'slider-input'}\n        unit={typeof suffix === 'string' ? suffix : unit}\n        onChange={newVal => {\n          if (newVal) {\n            internalOnChange(newVal)\n          } else internalOnChange(0)\n        }}\n        onBlur={event => {\n          if (!event.target.value) internalOnChange(min)\n        }}\n      />\n    ) : (\n      <StyledTextValue\n        as=\"span\"\n        variant=\"bodySmall\"\n        sentiment=\"neutral\"\n        placement={direction === 'column' ? 'right' : 'center'}\n        double={false}\n        isColumn={direction === 'column'}\n        data-testid={dataTestId ? `${dataTestId}-value` : 'slider-value'}\n      >\n        {prefix}\n        {valueNumber}\n        {suffix ?? unit}\n      </StyledTextValue>\n    )\n\n  const tooltipText = useMemo(() => {\n    if (tooltip === true) {\n      return selectedIndex\n    }\n    if (tooltip) {\n      return tooltip\n    }\n\n    return undefined\n  }, [tooltip, selectedIndex])\n\n  const placementTooltip =\n    ((selectedIndex - min) / (max - min)) * (sliderWidth - THUMB_SIZE) +\n    THUMB_SIZE / 2 -\n    sliderWidth / 2\n\n  const valueToShow = options ? options[selectedIndex]?.value : selectedIndex\n\n  return (\n    <Stack gap={1} direction={direction} justifyContent=\"left\">\n      {label ? (\n        <Stack\n          justifyContent=\"space-between\"\n          direction=\"row\"\n          alignItems=\"center\"\n        >\n          <Label htmlFor={finalId} required={required}>\n            {label}\n          </Label>\n          {direction === 'column' ? styledValue(valueToShow) : null}\n        </Stack>\n      ) : null}\n\n      <Stack direction=\"row\" justifyContent=\"end\">\n        {direction === 'column' && !label ? styledValue(valueToShow) : null}\n      </Stack>\n      <Stack direction=\"column\" width=\"100%\" gap={1} justifyContent=\"center\">\n        <StyledTooltip\n          text={tooltipText}\n          placement={tooltipPosition}\n          left={placementTooltip}\n        >\n          <SliderElement\n            type=\"range\"\n            value={selectedIndex}\n            onChange={event => {\n              internalOnChange(Number.parseFloat(event.target.value))\n            }}\n            min={min}\n            max={max}\n            tabIndex={0}\n            step={step}\n            name={name}\n            disabled={!!disabled}\n            aria-disabled={disabled}\n            data-testid={dataTestId}\n            id={finalId}\n            onBlur={onBlur}\n            onFocus={onFocus}\n            aria-label={ariaLabel ?? name}\n            className={className}\n            style={getBackgroundSize}\n            data-error={!!error}\n            data-direction={direction}\n            themeSlider={theme}\n            ref={refSlider}\n            left={leftPosition}\n          />\n        </StyledTooltip>\n        {options ? (\n          <Options\n            ticks={ticks}\n            min={min}\n            max={max}\n            sliderWidth={sliderWidth}\n            value={selectedIndex}\n            step={step}\n          />\n        ) : null}\n      </Stack>\n      {direction === 'row' ? styledValue(valueToShow) : null}\n    </Stack>\n  )\n}\n"]} */"));
87
+ }) => theme.space["5"], ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Slider/components/SingleSlider.tsx"],"names":[],"mappings":"AA4EiD","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Slider/components/SingleSlider.tsx","sourcesContent":["'use client'\n\nimport { useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport { useCallback, useEffect, useId, useMemo, useRef, useState } from 'react'\nimport { Label } from '../../Label'\nimport { NumberInputV2 } from '../../NumberInputV2'\nimport { Stack } from '../../Stack'\nimport { Text } from '../../Text'\nimport { THUMB_SIZE } from '../constant'\nimport { StyledTooltip, thumbStyle, trackStyle } from '../styles'\nimport type { SingleSliderProps } from '../types'\nimport { Options } from './Options'\n\nconst StyledTextValue = styled(Text, {\n  shouldForwardProp: prop => !['double', 'isColumn'].includes(prop),\n})<{ double: boolean; isColumn: boolean }>`\n  min-width: ${({ theme, double, isColumn }) => (double && isColumn ? null : theme.space['5'])};\n  align-self: ${({ double, isColumn }) => (double || !isColumn ? 'center' : 'end')};\n`\n\nconst SliderElement = styled('input', {\n  shouldForwardProp: prop => !['themeSlider', 'left'].includes(prop),\n})<{ themeSlider: string; disabled: boolean; left: number }>`\n    appearance: none;\n    height: ${({ theme }) => theme.space['1']};\n    width: 100%;\n    position: relative;\n    background-color: ${({ theme }) => theme.colors.neutral.borderWeak};\n\n    border-radius: ${({ theme }) => theme.radii.default};\n    background-image:linear-gradient(${({ theme }) => theme.colors.primary.border}, ${({ theme }) => theme.colors.primary.border});\n    background-repeat: no-repeat;\n    align-self: center;\n    outline: none;\n\n    &:focus {\n        &::-moz-range-thumb {\n          border: ${({ theme, disabled }) => (disabled ? null : `1.5px solid ${theme.colors.primary.border}`)};\n          box-shadow: ${({ theme, disabled }) => (disabled ? null : theme.shadows.focusPrimary)};\n        }\n\n        &::-webkit-slider-thumb {\n          border: ${({ theme, disabled }) => (disabled ? null : `1.5px solid ${theme.colors.primary.border}`)};\n          box-shadow: ${({ theme, disabled }) => (disabled ? null : theme.shadows.focusPrimary)};\n        }\n  }\n    &[data-error='true']{\n        background-image:linear-gradient(${({ theme }) => theme.colors.danger.backgroundStrong}, ${({ theme }) => theme.colors.danger.backgroundStrong});\n    }\n\n    &[data-direction='column'] {\n      align-self: baseline;\n    }\n\n    &[aria-disabled='true']{\n      background-image:linear-gradient(${({ theme }) => theme.colors.primary.borderDisabled}, ${({ theme }) => theme.colors.primary.borderDisabled});\n    }\n\n    /* Mozilla */\n    ::-moz-range-track {\n        ${trackStyle}\n    }\n    ::-moz-range-thumb {\n        ${({ theme, themeSlider, disabled, left }) => thumbStyle(theme, themeSlider, disabled, left, false)}\n    }\n\n    /* Other browsers */\n    ::-webkit-slider-runnable-track {\n        ${trackStyle}\n    }\n    ::-webkit-slider-thumb {\n        ${({ theme, themeSlider, disabled, left }) => thumbStyle(theme, themeSlider, disabled, left, false)}\n    }\n`\n\nconst StyledNumberInputV2 = styled(NumberInputV2)`\n  min-width: ${({ theme }) => theme.space['5']};\n`\n\nexport const SingleSlider = ({\n  name,\n  tooltip,\n  direction,\n  disabled,\n  error,\n  'data-testid': dataTestId,\n  value,\n  onChange,\n  min = 0,\n  max = 100,\n  step = 1,\n  id,\n  onBlur,\n  unit,\n  options,\n  onFocus,\n  className,\n  label,\n  input,\n  prefix,\n  suffix,\n  required,\n  'aria-label': ariaLabel,\n  tooltipPosition,\n}: SingleSliderProps) => {\n  const localId = useId()\n  const { theme } = useTheme()\n  const finalId = id ?? localId\n  const safeValue = value ?? min\n  const [selectedIndex, setSelectedIndex] = useState(safeValue)\n  const refSlider = useRef<HTMLInputElement>(null)\n  const [sliderWidth, setWidth] = useState(0)\n  const [inputValue, setInputValue] = useState<number | null>(safeValue)\n\n  useEffect(() => {\n    setInputValue(selectedIndex)\n  }, [selectedIndex])\n\n  useEffect(() => {\n    setWidth(Number(refSlider.current?.offsetWidth))\n  }, [refSlider])\n\n  const ticks = useMemo(() => {\n    if (options) {\n      return options.map((element, index) => ({\n        value: index,\n        label: element.label,\n      }))\n    }\n\n    return []\n  }, [options])\n\n  const internalOnChange = useCallback(\n    (newValue: number) => {\n      if (!newValue) {\n        setSelectedIndex(min)\n        onChange?.(min)\n      } else if (newValue < min) {\n        setSelectedIndex(min)\n        onChange?.(min)\n      } else if (newValue > max) {\n        setSelectedIndex(max)\n        onChange?.(max)\n      } else {\n        setSelectedIndex(newValue)\n        onChange?.(newValue)\n      }\n    },\n    [max, min, onChange],\n  )\n\n  // Make sure that min <= value <= max\n  useEffect(() => {\n    if (value < min) {\n      internalOnChange(min)\n    } else if (value > max) {\n      internalOnChange(max)\n    } else {\n      setSelectedIndex(() => value ?? min)\n    }\n  }, [value, max, min, internalOnChange])\n\n  // Get slider size\n  useEffect(() => {\n    const setWidthResize = () => {\n      setWidth(Number(refSlider.current?.offsetWidth))\n    }\n    window.addEventListener('resize', setWidthResize)\n\n    return () => {\n      window.removeEventListener('resize', setWidthResize)\n    }\n  }, [])\n\n  const leftPosition = useMemo(\n    () => ((selectedIndex - min) * 100) / (max - min),\n    [selectedIndex, max, min],\n  )\n\n  const getBackgroundSize = useMemo(\n    () => ({\n      backgroundSize: `${leftPosition}% 100%`,\n    }),\n    [leftPosition],\n  )\n\n  const styledValue = (valueNumber: string | number | null) =>\n    input && !options ? (\n      <StyledNumberInputV2\n        value={inputValue}\n        size=\"small\"\n        min={min}\n        aria-label=\"input\"\n        max={max}\n        step={step}\n        controls={false}\n        disabled={disabled}\n        data-testid={dataTestId ? `${dataTestId}-input` : 'slider-input'}\n        unit={typeof suffix === 'string' ? suffix : unit}\n        onChange={setInputValue}\n        onBlur={event => {\n          internalOnChange(Number.parseFloat(event.target.value))\n        }}\n      />\n    ) : (\n      <StyledTextValue\n        as=\"span\"\n        variant=\"bodySmall\"\n        sentiment=\"neutral\"\n        placement={direction === 'column' ? 'right' : 'center'}\n        double={false}\n        isColumn={direction === 'column'}\n        data-testid={dataTestId ? `${dataTestId}-value` : 'slider-value'}\n      >\n        {prefix}\n        {valueNumber}\n        {suffix ?? unit}\n      </StyledTextValue>\n    )\n\n  const tooltipText = useMemo(() => {\n    if (tooltip === true) {\n      return selectedIndex\n    }\n    if (tooltip) {\n      return tooltip\n    }\n\n    return undefined\n  }, [tooltip, selectedIndex])\n\n  const placementTooltip =\n    ((selectedIndex - min) / (max - min)) * (sliderWidth - THUMB_SIZE) +\n    THUMB_SIZE / 2 -\n    sliderWidth / 2\n\n  const valueToShow = options ? options[selectedIndex]?.value : selectedIndex\n\n  return (\n    <Stack gap={1} direction={direction} justifyContent=\"left\">\n      {label ? (\n        <Stack\n          justifyContent=\"space-between\"\n          direction=\"row\"\n          alignItems=\"center\"\n        >\n          <Label htmlFor={finalId} required={required}>\n            {label}\n          </Label>\n          {direction === 'column' ? styledValue(valueToShow) : null}\n        </Stack>\n      ) : null}\n\n      <Stack direction=\"row\" justifyContent=\"end\">\n        {direction === 'column' && !label ? styledValue(valueToShow) : null}\n      </Stack>\n      <Stack direction=\"column\" width=\"100%\" gap={1} justifyContent=\"center\">\n        <StyledTooltip\n          text={tooltipText}\n          placement={tooltipPosition}\n          left={placementTooltip}\n        >\n          <SliderElement\n            type=\"range\"\n            value={selectedIndex}\n            onChange={event => {\n              internalOnChange(Number.parseFloat(event.target.value))\n            }}\n            min={min}\n            max={max}\n            tabIndex={0}\n            step={step}\n            name={name}\n            disabled={!!disabled}\n            aria-disabled={disabled}\n            data-testid={dataTestId}\n            id={finalId}\n            onBlur={onBlur}\n            onFocus={onFocus}\n            aria-label={ariaLabel ?? name}\n            className={className}\n            style={getBackgroundSize}\n            data-error={!!error}\n            data-direction={direction}\n            themeSlider={theme}\n            ref={refSlider}\n            left={leftPosition}\n          />\n        </StyledTooltip>\n        {options ? (\n          <Options\n            ticks={ticks}\n            min={min}\n            max={max}\n            sliderWidth={sliderWidth}\n            value={selectedIndex}\n            step={step}\n          />\n        ) : null}\n      </Stack>\n      {direction === 'row' ? styledValue(valueToShow) : null}\n    </Stack>\n  )\n}\n"]} */"));
88
88
  const SingleSlider = ({
89
89
  name,
90
90
  tooltip,
@@ -120,6 +120,10 @@ const SingleSlider = ({
120
120
  const [selectedIndex, setSelectedIndex] = React.useState(safeValue);
121
121
  const refSlider = React.useRef(null);
122
122
  const [sliderWidth, setWidth] = React.useState(0);
123
+ const [inputValue, setInputValue] = React.useState(safeValue);
124
+ React.useEffect(() => {
125
+ setInputValue(selectedIndex);
126
+ }, [selectedIndex]);
123
127
  React.useEffect(() => {
124
128
  setWidth(Number(refSlider.current?.offsetWidth));
125
129
  }, [refSlider]);
@@ -133,9 +137,20 @@ const SingleSlider = ({
133
137
  return [];
134
138
  }, [options]);
135
139
  const internalOnChange = React.useCallback((newValue) => {
136
- setSelectedIndex(newValue ?? min);
137
- onChange?.(newValue ?? min);
138
- }, [min, onChange]);
140
+ if (!newValue) {
141
+ setSelectedIndex(min);
142
+ onChange?.(min);
143
+ } else if (newValue < min) {
144
+ setSelectedIndex(min);
145
+ onChange?.(min);
146
+ } else if (newValue > max) {
147
+ setSelectedIndex(max);
148
+ onChange?.(max);
149
+ } else {
150
+ setSelectedIndex(newValue);
151
+ onChange?.(newValue);
152
+ }
153
+ }, [max, min, onChange]);
139
154
  React.useEffect(() => {
140
155
  if (value < min) {
141
156
  internalOnChange(min);
@@ -158,12 +173,8 @@ const SingleSlider = ({
158
173
  const getBackgroundSize = React.useMemo(() => ({
159
174
  backgroundSize: `${leftPosition}% 100%`
160
175
  }), [leftPosition]);
161
- const styledValue = (valueNumber) => input && !options ? /* @__PURE__ */ jsxRuntime.jsx(StyledNumberInputV2, { value: typeof valueNumber === "string" ? Number.parseFloat(valueNumber) : valueNumber, size: "small", min, "aria-label": "input", max, step, controls: false, disabled, "data-testid": dataTestId ? `${dataTestId}-input` : "slider-input", unit: typeof suffix === "string" ? suffix : unit, onChange: (newVal) => {
162
- if (newVal) {
163
- internalOnChange(newVal);
164
- } else internalOnChange(0);
165
- }, onBlur: (event) => {
166
- if (!event.target.value) internalOnChange(min);
176
+ const styledValue = (valueNumber) => input && !options ? /* @__PURE__ */ jsxRuntime.jsx(StyledNumberInputV2, { value: inputValue, size: "small", min, "aria-label": "input", max, step, controls: false, disabled, "data-testid": dataTestId ? `${dataTestId}-input` : "slider-input", unit: typeof suffix === "string" ? suffix : unit, onChange: setInputValue, onBlur: (event) => {
177
+ internalOnChange(Number.parseFloat(event.target.value));
167
178
  } }) : /* @__PURE__ */ jsxRuntime.jsxs(StyledTextValue, { as: "span", variant: "bodySmall", sentiment: "neutral", placement: direction === "column" ? "right" : "center", double: false, isColumn: direction === "column", "data-testid": dataTestId ? `${dataTestId}-value` : "slider-value", children: [
168
179
  prefix,
169
180
  valueNumber,
@@ -24,7 +24,7 @@ const StyledTextValue = /* @__PURE__ */ _styled(Text, process.env.NODE_ENV === "
24
24
  }) => double && isColumn ? null : theme.space["5"], ";align-self:", ({
25
25
  double,
26
26
  isColumn
27
- }) => double || !isColumn ? "center" : "end", ";" + (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/Slider/components/SingleSlider.tsx"],"names":[],"mappings":"AAgB0C","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Slider/components/SingleSlider.tsx","sourcesContent":["'use client'\n\nimport { useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport { useCallback, useEffect, useId, useMemo, useRef, useState } from 'react'\nimport { Label } from '../../Label'\nimport { NumberInputV2 } from '../../NumberInputV2'\nimport { Stack } from '../../Stack'\nimport { Text } from '../../Text'\nimport { THUMB_SIZE } from '../constant'\nimport { StyledTooltip, thumbStyle, trackStyle } from '../styles'\nimport type { SingleSliderProps } from '../types'\nimport { Options } from './Options'\n\nconst StyledTextValue = styled(Text, {\n  shouldForwardProp: prop => !['double', 'isColumn'].includes(prop),\n})<{ double: boolean; isColumn: boolean }>`\n  min-width: ${({ theme, double, isColumn }) => (double && isColumn ? null : theme.space['5'])};\n  align-self: ${({ double, isColumn }) => (double || !isColumn ? 'center' : 'end')};\n`\n\nconst SliderElement = styled('input', {\n  shouldForwardProp: prop => !['themeSlider', 'left'].includes(prop),\n})<{ themeSlider: string; disabled: boolean; left: number }>`\n    appearance: none;\n    height: ${({ theme }) => theme.space['1']};\n    width: 100%;\n    position: relative;\n    background-color: ${({ theme }) => theme.colors.neutral.borderWeak};\n\n    border-radius: ${({ theme }) => theme.radii.default};\n    background-image:linear-gradient(${({ theme }) => theme.colors.primary.border}, ${({ theme }) => theme.colors.primary.border});\n    background-repeat: no-repeat;\n    align-self: center;\n    outline: none;\n\n    &:focus {\n        &::-moz-range-thumb {\n          border: ${({ theme, disabled }) => (disabled ? null : `1.5px solid ${theme.colors.primary.border}`)};\n          box-shadow: ${({ theme, disabled }) => (disabled ? null : theme.shadows.focusPrimary)};\n        }\n\n        &::-webkit-slider-thumb {\n          border: ${({ theme, disabled }) => (disabled ? null : `1.5px solid ${theme.colors.primary.border}`)};\n          box-shadow: ${({ theme, disabled }) => (disabled ? null : theme.shadows.focusPrimary)};\n        }\n  }\n    &[data-error='true']{\n        background-image:linear-gradient(${({ theme }) => theme.colors.danger.backgroundStrong}, ${({ theme }) => theme.colors.danger.backgroundStrong});\n    }\n\n    &[data-direction='column'] {\n      align-self: baseline;\n    }\n\n    &[aria-disabled='true']{\n      background-image:linear-gradient(${({ theme }) => theme.colors.primary.borderDisabled}, ${({ theme }) => theme.colors.primary.borderDisabled});\n    }\n\n    /* Mozilla */\n    ::-moz-range-track {\n        ${trackStyle}\n    }\n    ::-moz-range-thumb {\n        ${({ theme, themeSlider, disabled, left }) => thumbStyle(theme, themeSlider, disabled, left, false)}\n    }\n\n    /* Other browsers */\n    ::-webkit-slider-runnable-track {\n        ${trackStyle}\n    }\n    ::-webkit-slider-thumb {\n        ${({ theme, themeSlider, disabled, left }) => thumbStyle(theme, themeSlider, disabled, left, false)}\n    }\n`\n\nconst StyledNumberInputV2 = styled(NumberInputV2)`\n  min-width: ${({ theme }) => theme.space['5']};\n`\n\nexport const SingleSlider = ({\n  name,\n  tooltip,\n  direction,\n  disabled,\n  error,\n  'data-testid': dataTestId,\n  value,\n  onChange,\n  min = 0,\n  max = 100,\n  step = 1,\n  id,\n  onBlur,\n  unit,\n  options,\n  onFocus,\n  className,\n  label,\n  input,\n  prefix,\n  suffix,\n  required,\n  'aria-label': ariaLabel,\n  tooltipPosition,\n}: SingleSliderProps) => {\n  const localId = useId()\n  const { theme } = useTheme()\n  const finalId = id ?? localId\n  const safeValue = value ?? min\n  const [selectedIndex, setSelectedIndex] = useState(safeValue)\n  const refSlider = useRef<HTMLInputElement>(null)\n  const [sliderWidth, setWidth] = useState(0)\n\n  useEffect(() => {\n    setWidth(Number(refSlider.current?.offsetWidth))\n  }, [refSlider])\n\n  const ticks = useMemo(() => {\n    if (options) {\n      return options.map((element, index) => ({\n        value: index,\n        label: element.label,\n      }))\n    }\n\n    return []\n  }, [options])\n\n  const internalOnChange = useCallback(\n    (newValue: number) => {\n      setSelectedIndex(newValue ?? min)\n      onChange?.(newValue ?? min)\n    },\n    [min, onChange],\n  )\n\n  // Make sure that min <= value <= max\n  useEffect(() => {\n    if (value < min) {\n      internalOnChange(min)\n    } else if (value > max) {\n      internalOnChange(max)\n    } else {\n      setSelectedIndex(() => value ?? min)\n    }\n  }, [value, max, min, internalOnChange])\n\n  // Get slider size\n  useEffect(() => {\n    const setWidthResize = () => {\n      setWidth(Number(refSlider.current?.offsetWidth))\n    }\n    window.addEventListener('resize', setWidthResize)\n\n    return () => {\n      window.removeEventListener('resize', setWidthResize)\n    }\n  }, [])\n\n  const leftPosition = useMemo(\n    () => ((selectedIndex - min) * 100) / (max - min),\n    [selectedIndex, max, min],\n  )\n\n  const getBackgroundSize = useMemo(\n    () => ({\n      backgroundSize: `${leftPosition}% 100%`,\n    }),\n    [leftPosition],\n  )\n\n  const styledValue = (valueNumber: string | number | null) =>\n    input && !options ? (\n      <StyledNumberInputV2\n        value={\n          typeof valueNumber === 'string'\n            ? Number.parseFloat(valueNumber)\n            : valueNumber\n        }\n        size=\"small\"\n        min={min}\n        aria-label=\"input\"\n        max={max}\n        step={step}\n        controls={false}\n        disabled={disabled}\n        data-testid={dataTestId ? `${dataTestId}-input` : 'slider-input'}\n        unit={typeof suffix === 'string' ? suffix : unit}\n        onChange={newVal => {\n          if (newVal) {\n            internalOnChange(newVal)\n          } else internalOnChange(0)\n        }}\n        onBlur={event => {\n          if (!event.target.value) internalOnChange(min)\n        }}\n      />\n    ) : (\n      <StyledTextValue\n        as=\"span\"\n        variant=\"bodySmall\"\n        sentiment=\"neutral\"\n        placement={direction === 'column' ? 'right' : 'center'}\n        double={false}\n        isColumn={direction === 'column'}\n        data-testid={dataTestId ? `${dataTestId}-value` : 'slider-value'}\n      >\n        {prefix}\n        {valueNumber}\n        {suffix ?? unit}\n      </StyledTextValue>\n    )\n\n  const tooltipText = useMemo(() => {\n    if (tooltip === true) {\n      return selectedIndex\n    }\n    if (tooltip) {\n      return tooltip\n    }\n\n    return undefined\n  }, [tooltip, selectedIndex])\n\n  const placementTooltip =\n    ((selectedIndex - min) / (max - min)) * (sliderWidth - THUMB_SIZE) +\n    THUMB_SIZE / 2 -\n    sliderWidth / 2\n\n  const valueToShow = options ? options[selectedIndex]?.value : selectedIndex\n\n  return (\n    <Stack gap={1} direction={direction} justifyContent=\"left\">\n      {label ? (\n        <Stack\n          justifyContent=\"space-between\"\n          direction=\"row\"\n          alignItems=\"center\"\n        >\n          <Label htmlFor={finalId} required={required}>\n            {label}\n          </Label>\n          {direction === 'column' ? styledValue(valueToShow) : null}\n        </Stack>\n      ) : null}\n\n      <Stack direction=\"row\" justifyContent=\"end\">\n        {direction === 'column' && !label ? styledValue(valueToShow) : null}\n      </Stack>\n      <Stack direction=\"column\" width=\"100%\" gap={1} justifyContent=\"center\">\n        <StyledTooltip\n          text={tooltipText}\n          placement={tooltipPosition}\n          left={placementTooltip}\n        >\n          <SliderElement\n            type=\"range\"\n            value={selectedIndex}\n            onChange={event => {\n              internalOnChange(Number.parseFloat(event.target.value))\n            }}\n            min={min}\n            max={max}\n            tabIndex={0}\n            step={step}\n            name={name}\n            disabled={!!disabled}\n            aria-disabled={disabled}\n            data-testid={dataTestId}\n            id={finalId}\n            onBlur={onBlur}\n            onFocus={onFocus}\n            aria-label={ariaLabel ?? name}\n            className={className}\n            style={getBackgroundSize}\n            data-error={!!error}\n            data-direction={direction}\n            themeSlider={theme}\n            ref={refSlider}\n            left={leftPosition}\n          />\n        </StyledTooltip>\n        {options ? (\n          <Options\n            ticks={ticks}\n            min={min}\n            max={max}\n            sliderWidth={sliderWidth}\n            value={selectedIndex}\n            step={step}\n          />\n        ) : null}\n      </Stack>\n      {direction === 'row' ? styledValue(valueToShow) : null}\n    </Stack>\n  )\n}\n"]} */"));
27
+ }) => double || !isColumn ? "center" : "end", ";" + (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/Slider/components/SingleSlider.tsx"],"names":[],"mappings":"AAgB0C","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Slider/components/SingleSlider.tsx","sourcesContent":["'use client'\n\nimport { useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport { useCallback, useEffect, useId, useMemo, useRef, useState } from 'react'\nimport { Label } from '../../Label'\nimport { NumberInputV2 } from '../../NumberInputV2'\nimport { Stack } from '../../Stack'\nimport { Text } from '../../Text'\nimport { THUMB_SIZE } from '../constant'\nimport { StyledTooltip, thumbStyle, trackStyle } from '../styles'\nimport type { SingleSliderProps } from '../types'\nimport { Options } from './Options'\n\nconst StyledTextValue = styled(Text, {\n  shouldForwardProp: prop => !['double', 'isColumn'].includes(prop),\n})<{ double: boolean; isColumn: boolean }>`\n  min-width: ${({ theme, double, isColumn }) => (double && isColumn ? null : theme.space['5'])};\n  align-self: ${({ double, isColumn }) => (double || !isColumn ? 'center' : 'end')};\n`\n\nconst SliderElement = styled('input', {\n  shouldForwardProp: prop => !['themeSlider', 'left'].includes(prop),\n})<{ themeSlider: string; disabled: boolean; left: number }>`\n    appearance: none;\n    height: ${({ theme }) => theme.space['1']};\n    width: 100%;\n    position: relative;\n    background-color: ${({ theme }) => theme.colors.neutral.borderWeak};\n\n    border-radius: ${({ theme }) => theme.radii.default};\n    background-image:linear-gradient(${({ theme }) => theme.colors.primary.border}, ${({ theme }) => theme.colors.primary.border});\n    background-repeat: no-repeat;\n    align-self: center;\n    outline: none;\n\n    &:focus {\n        &::-moz-range-thumb {\n          border: ${({ theme, disabled }) => (disabled ? null : `1.5px solid ${theme.colors.primary.border}`)};\n          box-shadow: ${({ theme, disabled }) => (disabled ? null : theme.shadows.focusPrimary)};\n        }\n\n        &::-webkit-slider-thumb {\n          border: ${({ theme, disabled }) => (disabled ? null : `1.5px solid ${theme.colors.primary.border}`)};\n          box-shadow: ${({ theme, disabled }) => (disabled ? null : theme.shadows.focusPrimary)};\n        }\n  }\n    &[data-error='true']{\n        background-image:linear-gradient(${({ theme }) => theme.colors.danger.backgroundStrong}, ${({ theme }) => theme.colors.danger.backgroundStrong});\n    }\n\n    &[data-direction='column'] {\n      align-self: baseline;\n    }\n\n    &[aria-disabled='true']{\n      background-image:linear-gradient(${({ theme }) => theme.colors.primary.borderDisabled}, ${({ theme }) => theme.colors.primary.borderDisabled});\n    }\n\n    /* Mozilla */\n    ::-moz-range-track {\n        ${trackStyle}\n    }\n    ::-moz-range-thumb {\n        ${({ theme, themeSlider, disabled, left }) => thumbStyle(theme, themeSlider, disabled, left, false)}\n    }\n\n    /* Other browsers */\n    ::-webkit-slider-runnable-track {\n        ${trackStyle}\n    }\n    ::-webkit-slider-thumb {\n        ${({ theme, themeSlider, disabled, left }) => thumbStyle(theme, themeSlider, disabled, left, false)}\n    }\n`\n\nconst StyledNumberInputV2 = styled(NumberInputV2)`\n  min-width: ${({ theme }) => theme.space['5']};\n`\n\nexport const SingleSlider = ({\n  name,\n  tooltip,\n  direction,\n  disabled,\n  error,\n  'data-testid': dataTestId,\n  value,\n  onChange,\n  min = 0,\n  max = 100,\n  step = 1,\n  id,\n  onBlur,\n  unit,\n  options,\n  onFocus,\n  className,\n  label,\n  input,\n  prefix,\n  suffix,\n  required,\n  'aria-label': ariaLabel,\n  tooltipPosition,\n}: SingleSliderProps) => {\n  const localId = useId()\n  const { theme } = useTheme()\n  const finalId = id ?? localId\n  const safeValue = value ?? min\n  const [selectedIndex, setSelectedIndex] = useState(safeValue)\n  const refSlider = useRef<HTMLInputElement>(null)\n  const [sliderWidth, setWidth] = useState(0)\n  const [inputValue, setInputValue] = useState<number | null>(safeValue)\n\n  useEffect(() => {\n    setInputValue(selectedIndex)\n  }, [selectedIndex])\n\n  useEffect(() => {\n    setWidth(Number(refSlider.current?.offsetWidth))\n  }, [refSlider])\n\n  const ticks = useMemo(() => {\n    if (options) {\n      return options.map((element, index) => ({\n        value: index,\n        label: element.label,\n      }))\n    }\n\n    return []\n  }, [options])\n\n  const internalOnChange = useCallback(\n    (newValue: number) => {\n      if (!newValue) {\n        setSelectedIndex(min)\n        onChange?.(min)\n      } else if (newValue < min) {\n        setSelectedIndex(min)\n        onChange?.(min)\n      } else if (newValue > max) {\n        setSelectedIndex(max)\n        onChange?.(max)\n      } else {\n        setSelectedIndex(newValue)\n        onChange?.(newValue)\n      }\n    },\n    [max, min, onChange],\n  )\n\n  // Make sure that min <= value <= max\n  useEffect(() => {\n    if (value < min) {\n      internalOnChange(min)\n    } else if (value > max) {\n      internalOnChange(max)\n    } else {\n      setSelectedIndex(() => value ?? min)\n    }\n  }, [value, max, min, internalOnChange])\n\n  // Get slider size\n  useEffect(() => {\n    const setWidthResize = () => {\n      setWidth(Number(refSlider.current?.offsetWidth))\n    }\n    window.addEventListener('resize', setWidthResize)\n\n    return () => {\n      window.removeEventListener('resize', setWidthResize)\n    }\n  }, [])\n\n  const leftPosition = useMemo(\n    () => ((selectedIndex - min) * 100) / (max - min),\n    [selectedIndex, max, min],\n  )\n\n  const getBackgroundSize = useMemo(\n    () => ({\n      backgroundSize: `${leftPosition}% 100%`,\n    }),\n    [leftPosition],\n  )\n\n  const styledValue = (valueNumber: string | number | null) =>\n    input && !options ? (\n      <StyledNumberInputV2\n        value={inputValue}\n        size=\"small\"\n        min={min}\n        aria-label=\"input\"\n        max={max}\n        step={step}\n        controls={false}\n        disabled={disabled}\n        data-testid={dataTestId ? `${dataTestId}-input` : 'slider-input'}\n        unit={typeof suffix === 'string' ? suffix : unit}\n        onChange={setInputValue}\n        onBlur={event => {\n          internalOnChange(Number.parseFloat(event.target.value))\n        }}\n      />\n    ) : (\n      <StyledTextValue\n        as=\"span\"\n        variant=\"bodySmall\"\n        sentiment=\"neutral\"\n        placement={direction === 'column' ? 'right' : 'center'}\n        double={false}\n        isColumn={direction === 'column'}\n        data-testid={dataTestId ? `${dataTestId}-value` : 'slider-value'}\n      >\n        {prefix}\n        {valueNumber}\n        {suffix ?? unit}\n      </StyledTextValue>\n    )\n\n  const tooltipText = useMemo(() => {\n    if (tooltip === true) {\n      return selectedIndex\n    }\n    if (tooltip) {\n      return tooltip\n    }\n\n    return undefined\n  }, [tooltip, selectedIndex])\n\n  const placementTooltip =\n    ((selectedIndex - min) / (max - min)) * (sliderWidth - THUMB_SIZE) +\n    THUMB_SIZE / 2 -\n    sliderWidth / 2\n\n  const valueToShow = options ? options[selectedIndex]?.value : selectedIndex\n\n  return (\n    <Stack gap={1} direction={direction} justifyContent=\"left\">\n      {label ? (\n        <Stack\n          justifyContent=\"space-between\"\n          direction=\"row\"\n          alignItems=\"center\"\n        >\n          <Label htmlFor={finalId} required={required}>\n            {label}\n          </Label>\n          {direction === 'column' ? styledValue(valueToShow) : null}\n        </Stack>\n      ) : null}\n\n      <Stack direction=\"row\" justifyContent=\"end\">\n        {direction === 'column' && !label ? styledValue(valueToShow) : null}\n      </Stack>\n      <Stack direction=\"column\" width=\"100%\" gap={1} justifyContent=\"center\">\n        <StyledTooltip\n          text={tooltipText}\n          placement={tooltipPosition}\n          left={placementTooltip}\n        >\n          <SliderElement\n            type=\"range\"\n            value={selectedIndex}\n            onChange={event => {\n              internalOnChange(Number.parseFloat(event.target.value))\n            }}\n            min={min}\n            max={max}\n            tabIndex={0}\n            step={step}\n            name={name}\n            disabled={!!disabled}\n            aria-disabled={disabled}\n            data-testid={dataTestId}\n            id={finalId}\n            onBlur={onBlur}\n            onFocus={onFocus}\n            aria-label={ariaLabel ?? name}\n            className={className}\n            style={getBackgroundSize}\n            data-error={!!error}\n            data-direction={direction}\n            themeSlider={theme}\n            ref={refSlider}\n            left={leftPosition}\n          />\n        </StyledTooltip>\n        {options ? (\n          <Options\n            ticks={ticks}\n            min={min}\n            max={max}\n            sliderWidth={sliderWidth}\n            value={selectedIndex}\n            step={step}\n          />\n        ) : null}\n      </Stack>\n      {direction === 'row' ? styledValue(valueToShow) : null}\n    </Stack>\n  )\n}\n"]} */"));
28
28
  const SliderElement = /* @__PURE__ */ _styled("input", process.env.NODE_ENV === "production" ? {
29
29
  shouldForwardProp: (prop) => !["themeSlider", "left"].includes(prop),
30
30
  target: "elmzsg31"
@@ -72,7 +72,7 @@ const SliderElement = /* @__PURE__ */ _styled("input", process.env.NODE_ENV ===
72
72
  themeSlider,
73
73
  disabled,
74
74
  left
75
- }) => thumbStyle(theme, themeSlider, disabled, left, false), ";}" + (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/Slider/components/SingleSlider.tsx"],"names":[],"mappings":"AAuB4D","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Slider/components/SingleSlider.tsx","sourcesContent":["'use client'\n\nimport { useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport { useCallback, useEffect, useId, useMemo, useRef, useState } from 'react'\nimport { Label } from '../../Label'\nimport { NumberInputV2 } from '../../NumberInputV2'\nimport { Stack } from '../../Stack'\nimport { Text } from '../../Text'\nimport { THUMB_SIZE } from '../constant'\nimport { StyledTooltip, thumbStyle, trackStyle } from '../styles'\nimport type { SingleSliderProps } from '../types'\nimport { Options } from './Options'\n\nconst StyledTextValue = styled(Text, {\n  shouldForwardProp: prop => !['double', 'isColumn'].includes(prop),\n})<{ double: boolean; isColumn: boolean }>`\n  min-width: ${({ theme, double, isColumn }) => (double && isColumn ? null : theme.space['5'])};\n  align-self: ${({ double, isColumn }) => (double || !isColumn ? 'center' : 'end')};\n`\n\nconst SliderElement = styled('input', {\n  shouldForwardProp: prop => !['themeSlider', 'left'].includes(prop),\n})<{ themeSlider: string; disabled: boolean; left: number }>`\n    appearance: none;\n    height: ${({ theme }) => theme.space['1']};\n    width: 100%;\n    position: relative;\n    background-color: ${({ theme }) => theme.colors.neutral.borderWeak};\n\n    border-radius: ${({ theme }) => theme.radii.default};\n    background-image:linear-gradient(${({ theme }) => theme.colors.primary.border}, ${({ theme }) => theme.colors.primary.border});\n    background-repeat: no-repeat;\n    align-self: center;\n    outline: none;\n\n    &:focus {\n        &::-moz-range-thumb {\n          border: ${({ theme, disabled }) => (disabled ? null : `1.5px solid ${theme.colors.primary.border}`)};\n          box-shadow: ${({ theme, disabled }) => (disabled ? null : theme.shadows.focusPrimary)};\n        }\n\n        &::-webkit-slider-thumb {\n          border: ${({ theme, disabled }) => (disabled ? null : `1.5px solid ${theme.colors.primary.border}`)};\n          box-shadow: ${({ theme, disabled }) => (disabled ? null : theme.shadows.focusPrimary)};\n        }\n  }\n    &[data-error='true']{\n        background-image:linear-gradient(${({ theme }) => theme.colors.danger.backgroundStrong}, ${({ theme }) => theme.colors.danger.backgroundStrong});\n    }\n\n    &[data-direction='column'] {\n      align-self: baseline;\n    }\n\n    &[aria-disabled='true']{\n      background-image:linear-gradient(${({ theme }) => theme.colors.primary.borderDisabled}, ${({ theme }) => theme.colors.primary.borderDisabled});\n    }\n\n    /* Mozilla */\n    ::-moz-range-track {\n        ${trackStyle}\n    }\n    ::-moz-range-thumb {\n        ${({ theme, themeSlider, disabled, left }) => thumbStyle(theme, themeSlider, disabled, left, false)}\n    }\n\n    /* Other browsers */\n    ::-webkit-slider-runnable-track {\n        ${trackStyle}\n    }\n    ::-webkit-slider-thumb {\n        ${({ theme, themeSlider, disabled, left }) => thumbStyle(theme, themeSlider, disabled, left, false)}\n    }\n`\n\nconst StyledNumberInputV2 = styled(NumberInputV2)`\n  min-width: ${({ theme }) => theme.space['5']};\n`\n\nexport const SingleSlider = ({\n  name,\n  tooltip,\n  direction,\n  disabled,\n  error,\n  'data-testid': dataTestId,\n  value,\n  onChange,\n  min = 0,\n  max = 100,\n  step = 1,\n  id,\n  onBlur,\n  unit,\n  options,\n  onFocus,\n  className,\n  label,\n  input,\n  prefix,\n  suffix,\n  required,\n  'aria-label': ariaLabel,\n  tooltipPosition,\n}: SingleSliderProps) => {\n  const localId = useId()\n  const { theme } = useTheme()\n  const finalId = id ?? localId\n  const safeValue = value ?? min\n  const [selectedIndex, setSelectedIndex] = useState(safeValue)\n  const refSlider = useRef<HTMLInputElement>(null)\n  const [sliderWidth, setWidth] = useState(0)\n\n  useEffect(() => {\n    setWidth(Number(refSlider.current?.offsetWidth))\n  }, [refSlider])\n\n  const ticks = useMemo(() => {\n    if (options) {\n      return options.map((element, index) => ({\n        value: index,\n        label: element.label,\n      }))\n    }\n\n    return []\n  }, [options])\n\n  const internalOnChange = useCallback(\n    (newValue: number) => {\n      setSelectedIndex(newValue ?? min)\n      onChange?.(newValue ?? min)\n    },\n    [min, onChange],\n  )\n\n  // Make sure that min <= value <= max\n  useEffect(() => {\n    if (value < min) {\n      internalOnChange(min)\n    } else if (value > max) {\n      internalOnChange(max)\n    } else {\n      setSelectedIndex(() => value ?? min)\n    }\n  }, [value, max, min, internalOnChange])\n\n  // Get slider size\n  useEffect(() => {\n    const setWidthResize = () => {\n      setWidth(Number(refSlider.current?.offsetWidth))\n    }\n    window.addEventListener('resize', setWidthResize)\n\n    return () => {\n      window.removeEventListener('resize', setWidthResize)\n    }\n  }, [])\n\n  const leftPosition = useMemo(\n    () => ((selectedIndex - min) * 100) / (max - min),\n    [selectedIndex, max, min],\n  )\n\n  const getBackgroundSize = useMemo(\n    () => ({\n      backgroundSize: `${leftPosition}% 100%`,\n    }),\n    [leftPosition],\n  )\n\n  const styledValue = (valueNumber: string | number | null) =>\n    input && !options ? (\n      <StyledNumberInputV2\n        value={\n          typeof valueNumber === 'string'\n            ? Number.parseFloat(valueNumber)\n            : valueNumber\n        }\n        size=\"small\"\n        min={min}\n        aria-label=\"input\"\n        max={max}\n        step={step}\n        controls={false}\n        disabled={disabled}\n        data-testid={dataTestId ? `${dataTestId}-input` : 'slider-input'}\n        unit={typeof suffix === 'string' ? suffix : unit}\n        onChange={newVal => {\n          if (newVal) {\n            internalOnChange(newVal)\n          } else internalOnChange(0)\n        }}\n        onBlur={event => {\n          if (!event.target.value) internalOnChange(min)\n        }}\n      />\n    ) : (\n      <StyledTextValue\n        as=\"span\"\n        variant=\"bodySmall\"\n        sentiment=\"neutral\"\n        placement={direction === 'column' ? 'right' : 'center'}\n        double={false}\n        isColumn={direction === 'column'}\n        data-testid={dataTestId ? `${dataTestId}-value` : 'slider-value'}\n      >\n        {prefix}\n        {valueNumber}\n        {suffix ?? unit}\n      </StyledTextValue>\n    )\n\n  const tooltipText = useMemo(() => {\n    if (tooltip === true) {\n      return selectedIndex\n    }\n    if (tooltip) {\n      return tooltip\n    }\n\n    return undefined\n  }, [tooltip, selectedIndex])\n\n  const placementTooltip =\n    ((selectedIndex - min) / (max - min)) * (sliderWidth - THUMB_SIZE) +\n    THUMB_SIZE / 2 -\n    sliderWidth / 2\n\n  const valueToShow = options ? options[selectedIndex]?.value : selectedIndex\n\n  return (\n    <Stack gap={1} direction={direction} justifyContent=\"left\">\n      {label ? (\n        <Stack\n          justifyContent=\"space-between\"\n          direction=\"row\"\n          alignItems=\"center\"\n        >\n          <Label htmlFor={finalId} required={required}>\n            {label}\n          </Label>\n          {direction === 'column' ? styledValue(valueToShow) : null}\n        </Stack>\n      ) : null}\n\n      <Stack direction=\"row\" justifyContent=\"end\">\n        {direction === 'column' && !label ? styledValue(valueToShow) : null}\n      </Stack>\n      <Stack direction=\"column\" width=\"100%\" gap={1} justifyContent=\"center\">\n        <StyledTooltip\n          text={tooltipText}\n          placement={tooltipPosition}\n          left={placementTooltip}\n        >\n          <SliderElement\n            type=\"range\"\n            value={selectedIndex}\n            onChange={event => {\n              internalOnChange(Number.parseFloat(event.target.value))\n            }}\n            min={min}\n            max={max}\n            tabIndex={0}\n            step={step}\n            name={name}\n            disabled={!!disabled}\n            aria-disabled={disabled}\n            data-testid={dataTestId}\n            id={finalId}\n            onBlur={onBlur}\n            onFocus={onFocus}\n            aria-label={ariaLabel ?? name}\n            className={className}\n            style={getBackgroundSize}\n            data-error={!!error}\n            data-direction={direction}\n            themeSlider={theme}\n            ref={refSlider}\n            left={leftPosition}\n          />\n        </StyledTooltip>\n        {options ? (\n          <Options\n            ticks={ticks}\n            min={min}\n            max={max}\n            sliderWidth={sliderWidth}\n            value={selectedIndex}\n            step={step}\n          />\n        ) : null}\n      </Stack>\n      {direction === 'row' ? styledValue(valueToShow) : null}\n    </Stack>\n  )\n}\n"]} */"));
75
+ }) => thumbStyle(theme, themeSlider, disabled, left, false), ";}" + (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/Slider/components/SingleSlider.tsx"],"names":[],"mappings":"AAuB4D","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Slider/components/SingleSlider.tsx","sourcesContent":["'use client'\n\nimport { useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport { useCallback, useEffect, useId, useMemo, useRef, useState } from 'react'\nimport { Label } from '../../Label'\nimport { NumberInputV2 } from '../../NumberInputV2'\nimport { Stack } from '../../Stack'\nimport { Text } from '../../Text'\nimport { THUMB_SIZE } from '../constant'\nimport { StyledTooltip, thumbStyle, trackStyle } from '../styles'\nimport type { SingleSliderProps } from '../types'\nimport { Options } from './Options'\n\nconst StyledTextValue = styled(Text, {\n  shouldForwardProp: prop => !['double', 'isColumn'].includes(prop),\n})<{ double: boolean; isColumn: boolean }>`\n  min-width: ${({ theme, double, isColumn }) => (double && isColumn ? null : theme.space['5'])};\n  align-self: ${({ double, isColumn }) => (double || !isColumn ? 'center' : 'end')};\n`\n\nconst SliderElement = styled('input', {\n  shouldForwardProp: prop => !['themeSlider', 'left'].includes(prop),\n})<{ themeSlider: string; disabled: boolean; left: number }>`\n    appearance: none;\n    height: ${({ theme }) => theme.space['1']};\n    width: 100%;\n    position: relative;\n    background-color: ${({ theme }) => theme.colors.neutral.borderWeak};\n\n    border-radius: ${({ theme }) => theme.radii.default};\n    background-image:linear-gradient(${({ theme }) => theme.colors.primary.border}, ${({ theme }) => theme.colors.primary.border});\n    background-repeat: no-repeat;\n    align-self: center;\n    outline: none;\n\n    &:focus {\n        &::-moz-range-thumb {\n          border: ${({ theme, disabled }) => (disabled ? null : `1.5px solid ${theme.colors.primary.border}`)};\n          box-shadow: ${({ theme, disabled }) => (disabled ? null : theme.shadows.focusPrimary)};\n        }\n\n        &::-webkit-slider-thumb {\n          border: ${({ theme, disabled }) => (disabled ? null : `1.5px solid ${theme.colors.primary.border}`)};\n          box-shadow: ${({ theme, disabled }) => (disabled ? null : theme.shadows.focusPrimary)};\n        }\n  }\n    &[data-error='true']{\n        background-image:linear-gradient(${({ theme }) => theme.colors.danger.backgroundStrong}, ${({ theme }) => theme.colors.danger.backgroundStrong});\n    }\n\n    &[data-direction='column'] {\n      align-self: baseline;\n    }\n\n    &[aria-disabled='true']{\n      background-image:linear-gradient(${({ theme }) => theme.colors.primary.borderDisabled}, ${({ theme }) => theme.colors.primary.borderDisabled});\n    }\n\n    /* Mozilla */\n    ::-moz-range-track {\n        ${trackStyle}\n    }\n    ::-moz-range-thumb {\n        ${({ theme, themeSlider, disabled, left }) => thumbStyle(theme, themeSlider, disabled, left, false)}\n    }\n\n    /* Other browsers */\n    ::-webkit-slider-runnable-track {\n        ${trackStyle}\n    }\n    ::-webkit-slider-thumb {\n        ${({ theme, themeSlider, disabled, left }) => thumbStyle(theme, themeSlider, disabled, left, false)}\n    }\n`\n\nconst StyledNumberInputV2 = styled(NumberInputV2)`\n  min-width: ${({ theme }) => theme.space['5']};\n`\n\nexport const SingleSlider = ({\n  name,\n  tooltip,\n  direction,\n  disabled,\n  error,\n  'data-testid': dataTestId,\n  value,\n  onChange,\n  min = 0,\n  max = 100,\n  step = 1,\n  id,\n  onBlur,\n  unit,\n  options,\n  onFocus,\n  className,\n  label,\n  input,\n  prefix,\n  suffix,\n  required,\n  'aria-label': ariaLabel,\n  tooltipPosition,\n}: SingleSliderProps) => {\n  const localId = useId()\n  const { theme } = useTheme()\n  const finalId = id ?? localId\n  const safeValue = value ?? min\n  const [selectedIndex, setSelectedIndex] = useState(safeValue)\n  const refSlider = useRef<HTMLInputElement>(null)\n  const [sliderWidth, setWidth] = useState(0)\n  const [inputValue, setInputValue] = useState<number | null>(safeValue)\n\n  useEffect(() => {\n    setInputValue(selectedIndex)\n  }, [selectedIndex])\n\n  useEffect(() => {\n    setWidth(Number(refSlider.current?.offsetWidth))\n  }, [refSlider])\n\n  const ticks = useMemo(() => {\n    if (options) {\n      return options.map((element, index) => ({\n        value: index,\n        label: element.label,\n      }))\n    }\n\n    return []\n  }, [options])\n\n  const internalOnChange = useCallback(\n    (newValue: number) => {\n      if (!newValue) {\n        setSelectedIndex(min)\n        onChange?.(min)\n      } else if (newValue < min) {\n        setSelectedIndex(min)\n        onChange?.(min)\n      } else if (newValue > max) {\n        setSelectedIndex(max)\n        onChange?.(max)\n      } else {\n        setSelectedIndex(newValue)\n        onChange?.(newValue)\n      }\n    },\n    [max, min, onChange],\n  )\n\n  // Make sure that min <= value <= max\n  useEffect(() => {\n    if (value < min) {\n      internalOnChange(min)\n    } else if (value > max) {\n      internalOnChange(max)\n    } else {\n      setSelectedIndex(() => value ?? min)\n    }\n  }, [value, max, min, internalOnChange])\n\n  // Get slider size\n  useEffect(() => {\n    const setWidthResize = () => {\n      setWidth(Number(refSlider.current?.offsetWidth))\n    }\n    window.addEventListener('resize', setWidthResize)\n\n    return () => {\n      window.removeEventListener('resize', setWidthResize)\n    }\n  }, [])\n\n  const leftPosition = useMemo(\n    () => ((selectedIndex - min) * 100) / (max - min),\n    [selectedIndex, max, min],\n  )\n\n  const getBackgroundSize = useMemo(\n    () => ({\n      backgroundSize: `${leftPosition}% 100%`,\n    }),\n    [leftPosition],\n  )\n\n  const styledValue = (valueNumber: string | number | null) =>\n    input && !options ? (\n      <StyledNumberInputV2\n        value={inputValue}\n        size=\"small\"\n        min={min}\n        aria-label=\"input\"\n        max={max}\n        step={step}\n        controls={false}\n        disabled={disabled}\n        data-testid={dataTestId ? `${dataTestId}-input` : 'slider-input'}\n        unit={typeof suffix === 'string' ? suffix : unit}\n        onChange={setInputValue}\n        onBlur={event => {\n          internalOnChange(Number.parseFloat(event.target.value))\n        }}\n      />\n    ) : (\n      <StyledTextValue\n        as=\"span\"\n        variant=\"bodySmall\"\n        sentiment=\"neutral\"\n        placement={direction === 'column' ? 'right' : 'center'}\n        double={false}\n        isColumn={direction === 'column'}\n        data-testid={dataTestId ? `${dataTestId}-value` : 'slider-value'}\n      >\n        {prefix}\n        {valueNumber}\n        {suffix ?? unit}\n      </StyledTextValue>\n    )\n\n  const tooltipText = useMemo(() => {\n    if (tooltip === true) {\n      return selectedIndex\n    }\n    if (tooltip) {\n      return tooltip\n    }\n\n    return undefined\n  }, [tooltip, selectedIndex])\n\n  const placementTooltip =\n    ((selectedIndex - min) / (max - min)) * (sliderWidth - THUMB_SIZE) +\n    THUMB_SIZE / 2 -\n    sliderWidth / 2\n\n  const valueToShow = options ? options[selectedIndex]?.value : selectedIndex\n\n  return (\n    <Stack gap={1} direction={direction} justifyContent=\"left\">\n      {label ? (\n        <Stack\n          justifyContent=\"space-between\"\n          direction=\"row\"\n          alignItems=\"center\"\n        >\n          <Label htmlFor={finalId} required={required}>\n            {label}\n          </Label>\n          {direction === 'column' ? styledValue(valueToShow) : null}\n        </Stack>\n      ) : null}\n\n      <Stack direction=\"row\" justifyContent=\"end\">\n        {direction === 'column' && !label ? styledValue(valueToShow) : null}\n      </Stack>\n      <Stack direction=\"column\" width=\"100%\" gap={1} justifyContent=\"center\">\n        <StyledTooltip\n          text={tooltipText}\n          placement={tooltipPosition}\n          left={placementTooltip}\n        >\n          <SliderElement\n            type=\"range\"\n            value={selectedIndex}\n            onChange={event => {\n              internalOnChange(Number.parseFloat(event.target.value))\n            }}\n            min={min}\n            max={max}\n            tabIndex={0}\n            step={step}\n            name={name}\n            disabled={!!disabled}\n            aria-disabled={disabled}\n            data-testid={dataTestId}\n            id={finalId}\n            onBlur={onBlur}\n            onFocus={onFocus}\n            aria-label={ariaLabel ?? name}\n            className={className}\n            style={getBackgroundSize}\n            data-error={!!error}\n            data-direction={direction}\n            themeSlider={theme}\n            ref={refSlider}\n            left={leftPosition}\n          />\n        </StyledTooltip>\n        {options ? (\n          <Options\n            ticks={ticks}\n            min={min}\n            max={max}\n            sliderWidth={sliderWidth}\n            value={selectedIndex}\n            step={step}\n          />\n        ) : null}\n      </Stack>\n      {direction === 'row' ? styledValue(valueToShow) : null}\n    </Stack>\n  )\n}\n"]} */"));
76
76
  const StyledNumberInputV2 = /* @__PURE__ */ _styled(NumberInputV2, process.env.NODE_ENV === "production" ? {
77
77
  target: "elmzsg30"
78
78
  } : {
@@ -80,7 +80,7 @@ const StyledNumberInputV2 = /* @__PURE__ */ _styled(NumberInputV2, process.env.N
80
80
  label: "StyledNumberInputV2"
81
81
  })("min-width:", ({
82
82
  theme
83
- }) => theme.space["5"], ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Slider/components/SingleSlider.tsx"],"names":[],"mappings":"AA4EiD","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Slider/components/SingleSlider.tsx","sourcesContent":["'use client'\n\nimport { useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport { useCallback, useEffect, useId, useMemo, useRef, useState } from 'react'\nimport { Label } from '../../Label'\nimport { NumberInputV2 } from '../../NumberInputV2'\nimport { Stack } from '../../Stack'\nimport { Text } from '../../Text'\nimport { THUMB_SIZE } from '../constant'\nimport { StyledTooltip, thumbStyle, trackStyle } from '../styles'\nimport type { SingleSliderProps } from '../types'\nimport { Options } from './Options'\n\nconst StyledTextValue = styled(Text, {\n  shouldForwardProp: prop => !['double', 'isColumn'].includes(prop),\n})<{ double: boolean; isColumn: boolean }>`\n  min-width: ${({ theme, double, isColumn }) => (double && isColumn ? null : theme.space['5'])};\n  align-self: ${({ double, isColumn }) => (double || !isColumn ? 'center' : 'end')};\n`\n\nconst SliderElement = styled('input', {\n  shouldForwardProp: prop => !['themeSlider', 'left'].includes(prop),\n})<{ themeSlider: string; disabled: boolean; left: number }>`\n    appearance: none;\n    height: ${({ theme }) => theme.space['1']};\n    width: 100%;\n    position: relative;\n    background-color: ${({ theme }) => theme.colors.neutral.borderWeak};\n\n    border-radius: ${({ theme }) => theme.radii.default};\n    background-image:linear-gradient(${({ theme }) => theme.colors.primary.border}, ${({ theme }) => theme.colors.primary.border});\n    background-repeat: no-repeat;\n    align-self: center;\n    outline: none;\n\n    &:focus {\n        &::-moz-range-thumb {\n          border: ${({ theme, disabled }) => (disabled ? null : `1.5px solid ${theme.colors.primary.border}`)};\n          box-shadow: ${({ theme, disabled }) => (disabled ? null : theme.shadows.focusPrimary)};\n        }\n\n        &::-webkit-slider-thumb {\n          border: ${({ theme, disabled }) => (disabled ? null : `1.5px solid ${theme.colors.primary.border}`)};\n          box-shadow: ${({ theme, disabled }) => (disabled ? null : theme.shadows.focusPrimary)};\n        }\n  }\n    &[data-error='true']{\n        background-image:linear-gradient(${({ theme }) => theme.colors.danger.backgroundStrong}, ${({ theme }) => theme.colors.danger.backgroundStrong});\n    }\n\n    &[data-direction='column'] {\n      align-self: baseline;\n    }\n\n    &[aria-disabled='true']{\n      background-image:linear-gradient(${({ theme }) => theme.colors.primary.borderDisabled}, ${({ theme }) => theme.colors.primary.borderDisabled});\n    }\n\n    /* Mozilla */\n    ::-moz-range-track {\n        ${trackStyle}\n    }\n    ::-moz-range-thumb {\n        ${({ theme, themeSlider, disabled, left }) => thumbStyle(theme, themeSlider, disabled, left, false)}\n    }\n\n    /* Other browsers */\n    ::-webkit-slider-runnable-track {\n        ${trackStyle}\n    }\n    ::-webkit-slider-thumb {\n        ${({ theme, themeSlider, disabled, left }) => thumbStyle(theme, themeSlider, disabled, left, false)}\n    }\n`\n\nconst StyledNumberInputV2 = styled(NumberInputV2)`\n  min-width: ${({ theme }) => theme.space['5']};\n`\n\nexport const SingleSlider = ({\n  name,\n  tooltip,\n  direction,\n  disabled,\n  error,\n  'data-testid': dataTestId,\n  value,\n  onChange,\n  min = 0,\n  max = 100,\n  step = 1,\n  id,\n  onBlur,\n  unit,\n  options,\n  onFocus,\n  className,\n  label,\n  input,\n  prefix,\n  suffix,\n  required,\n  'aria-label': ariaLabel,\n  tooltipPosition,\n}: SingleSliderProps) => {\n  const localId = useId()\n  const { theme } = useTheme()\n  const finalId = id ?? localId\n  const safeValue = value ?? min\n  const [selectedIndex, setSelectedIndex] = useState(safeValue)\n  const refSlider = useRef<HTMLInputElement>(null)\n  const [sliderWidth, setWidth] = useState(0)\n\n  useEffect(() => {\n    setWidth(Number(refSlider.current?.offsetWidth))\n  }, [refSlider])\n\n  const ticks = useMemo(() => {\n    if (options) {\n      return options.map((element, index) => ({\n        value: index,\n        label: element.label,\n      }))\n    }\n\n    return []\n  }, [options])\n\n  const internalOnChange = useCallback(\n    (newValue: number) => {\n      setSelectedIndex(newValue ?? min)\n      onChange?.(newValue ?? min)\n    },\n    [min, onChange],\n  )\n\n  // Make sure that min <= value <= max\n  useEffect(() => {\n    if (value < min) {\n      internalOnChange(min)\n    } else if (value > max) {\n      internalOnChange(max)\n    } else {\n      setSelectedIndex(() => value ?? min)\n    }\n  }, [value, max, min, internalOnChange])\n\n  // Get slider size\n  useEffect(() => {\n    const setWidthResize = () => {\n      setWidth(Number(refSlider.current?.offsetWidth))\n    }\n    window.addEventListener('resize', setWidthResize)\n\n    return () => {\n      window.removeEventListener('resize', setWidthResize)\n    }\n  }, [])\n\n  const leftPosition = useMemo(\n    () => ((selectedIndex - min) * 100) / (max - min),\n    [selectedIndex, max, min],\n  )\n\n  const getBackgroundSize = useMemo(\n    () => ({\n      backgroundSize: `${leftPosition}% 100%`,\n    }),\n    [leftPosition],\n  )\n\n  const styledValue = (valueNumber: string | number | null) =>\n    input && !options ? (\n      <StyledNumberInputV2\n        value={\n          typeof valueNumber === 'string'\n            ? Number.parseFloat(valueNumber)\n            : valueNumber\n        }\n        size=\"small\"\n        min={min}\n        aria-label=\"input\"\n        max={max}\n        step={step}\n        controls={false}\n        disabled={disabled}\n        data-testid={dataTestId ? `${dataTestId}-input` : 'slider-input'}\n        unit={typeof suffix === 'string' ? suffix : unit}\n        onChange={newVal => {\n          if (newVal) {\n            internalOnChange(newVal)\n          } else internalOnChange(0)\n        }}\n        onBlur={event => {\n          if (!event.target.value) internalOnChange(min)\n        }}\n      />\n    ) : (\n      <StyledTextValue\n        as=\"span\"\n        variant=\"bodySmall\"\n        sentiment=\"neutral\"\n        placement={direction === 'column' ? 'right' : 'center'}\n        double={false}\n        isColumn={direction === 'column'}\n        data-testid={dataTestId ? `${dataTestId}-value` : 'slider-value'}\n      >\n        {prefix}\n        {valueNumber}\n        {suffix ?? unit}\n      </StyledTextValue>\n    )\n\n  const tooltipText = useMemo(() => {\n    if (tooltip === true) {\n      return selectedIndex\n    }\n    if (tooltip) {\n      return tooltip\n    }\n\n    return undefined\n  }, [tooltip, selectedIndex])\n\n  const placementTooltip =\n    ((selectedIndex - min) / (max - min)) * (sliderWidth - THUMB_SIZE) +\n    THUMB_SIZE / 2 -\n    sliderWidth / 2\n\n  const valueToShow = options ? options[selectedIndex]?.value : selectedIndex\n\n  return (\n    <Stack gap={1} direction={direction} justifyContent=\"left\">\n      {label ? (\n        <Stack\n          justifyContent=\"space-between\"\n          direction=\"row\"\n          alignItems=\"center\"\n        >\n          <Label htmlFor={finalId} required={required}>\n            {label}\n          </Label>\n          {direction === 'column' ? styledValue(valueToShow) : null}\n        </Stack>\n      ) : null}\n\n      <Stack direction=\"row\" justifyContent=\"end\">\n        {direction === 'column' && !label ? styledValue(valueToShow) : null}\n      </Stack>\n      <Stack direction=\"column\" width=\"100%\" gap={1} justifyContent=\"center\">\n        <StyledTooltip\n          text={tooltipText}\n          placement={tooltipPosition}\n          left={placementTooltip}\n        >\n          <SliderElement\n            type=\"range\"\n            value={selectedIndex}\n            onChange={event => {\n              internalOnChange(Number.parseFloat(event.target.value))\n            }}\n            min={min}\n            max={max}\n            tabIndex={0}\n            step={step}\n            name={name}\n            disabled={!!disabled}\n            aria-disabled={disabled}\n            data-testid={dataTestId}\n            id={finalId}\n            onBlur={onBlur}\n            onFocus={onFocus}\n            aria-label={ariaLabel ?? name}\n            className={className}\n            style={getBackgroundSize}\n            data-error={!!error}\n            data-direction={direction}\n            themeSlider={theme}\n            ref={refSlider}\n            left={leftPosition}\n          />\n        </StyledTooltip>\n        {options ? (\n          <Options\n            ticks={ticks}\n            min={min}\n            max={max}\n            sliderWidth={sliderWidth}\n            value={selectedIndex}\n            step={step}\n          />\n        ) : null}\n      </Stack>\n      {direction === 'row' ? styledValue(valueToShow) : null}\n    </Stack>\n  )\n}\n"]} */"));
83
+ }) => theme.space["5"], ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Slider/components/SingleSlider.tsx"],"names":[],"mappings":"AA4EiD","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Slider/components/SingleSlider.tsx","sourcesContent":["'use client'\n\nimport { useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport { useCallback, useEffect, useId, useMemo, useRef, useState } from 'react'\nimport { Label } from '../../Label'\nimport { NumberInputV2 } from '../../NumberInputV2'\nimport { Stack } from '../../Stack'\nimport { Text } from '../../Text'\nimport { THUMB_SIZE } from '../constant'\nimport { StyledTooltip, thumbStyle, trackStyle } from '../styles'\nimport type { SingleSliderProps } from '../types'\nimport { Options } from './Options'\n\nconst StyledTextValue = styled(Text, {\n  shouldForwardProp: prop => !['double', 'isColumn'].includes(prop),\n})<{ double: boolean; isColumn: boolean }>`\n  min-width: ${({ theme, double, isColumn }) => (double && isColumn ? null : theme.space['5'])};\n  align-self: ${({ double, isColumn }) => (double || !isColumn ? 'center' : 'end')};\n`\n\nconst SliderElement = styled('input', {\n  shouldForwardProp: prop => !['themeSlider', 'left'].includes(prop),\n})<{ themeSlider: string; disabled: boolean; left: number }>`\n    appearance: none;\n    height: ${({ theme }) => theme.space['1']};\n    width: 100%;\n    position: relative;\n    background-color: ${({ theme }) => theme.colors.neutral.borderWeak};\n\n    border-radius: ${({ theme }) => theme.radii.default};\n    background-image:linear-gradient(${({ theme }) => theme.colors.primary.border}, ${({ theme }) => theme.colors.primary.border});\n    background-repeat: no-repeat;\n    align-self: center;\n    outline: none;\n\n    &:focus {\n        &::-moz-range-thumb {\n          border: ${({ theme, disabled }) => (disabled ? null : `1.5px solid ${theme.colors.primary.border}`)};\n          box-shadow: ${({ theme, disabled }) => (disabled ? null : theme.shadows.focusPrimary)};\n        }\n\n        &::-webkit-slider-thumb {\n          border: ${({ theme, disabled }) => (disabled ? null : `1.5px solid ${theme.colors.primary.border}`)};\n          box-shadow: ${({ theme, disabled }) => (disabled ? null : theme.shadows.focusPrimary)};\n        }\n  }\n    &[data-error='true']{\n        background-image:linear-gradient(${({ theme }) => theme.colors.danger.backgroundStrong}, ${({ theme }) => theme.colors.danger.backgroundStrong});\n    }\n\n    &[data-direction='column'] {\n      align-self: baseline;\n    }\n\n    &[aria-disabled='true']{\n      background-image:linear-gradient(${({ theme }) => theme.colors.primary.borderDisabled}, ${({ theme }) => theme.colors.primary.borderDisabled});\n    }\n\n    /* Mozilla */\n    ::-moz-range-track {\n        ${trackStyle}\n    }\n    ::-moz-range-thumb {\n        ${({ theme, themeSlider, disabled, left }) => thumbStyle(theme, themeSlider, disabled, left, false)}\n    }\n\n    /* Other browsers */\n    ::-webkit-slider-runnable-track {\n        ${trackStyle}\n    }\n    ::-webkit-slider-thumb {\n        ${({ theme, themeSlider, disabled, left }) => thumbStyle(theme, themeSlider, disabled, left, false)}\n    }\n`\n\nconst StyledNumberInputV2 = styled(NumberInputV2)`\n  min-width: ${({ theme }) => theme.space['5']};\n`\n\nexport const SingleSlider = ({\n  name,\n  tooltip,\n  direction,\n  disabled,\n  error,\n  'data-testid': dataTestId,\n  value,\n  onChange,\n  min = 0,\n  max = 100,\n  step = 1,\n  id,\n  onBlur,\n  unit,\n  options,\n  onFocus,\n  className,\n  label,\n  input,\n  prefix,\n  suffix,\n  required,\n  'aria-label': ariaLabel,\n  tooltipPosition,\n}: SingleSliderProps) => {\n  const localId = useId()\n  const { theme } = useTheme()\n  const finalId = id ?? localId\n  const safeValue = value ?? min\n  const [selectedIndex, setSelectedIndex] = useState(safeValue)\n  const refSlider = useRef<HTMLInputElement>(null)\n  const [sliderWidth, setWidth] = useState(0)\n  const [inputValue, setInputValue] = useState<number | null>(safeValue)\n\n  useEffect(() => {\n    setInputValue(selectedIndex)\n  }, [selectedIndex])\n\n  useEffect(() => {\n    setWidth(Number(refSlider.current?.offsetWidth))\n  }, [refSlider])\n\n  const ticks = useMemo(() => {\n    if (options) {\n      return options.map((element, index) => ({\n        value: index,\n        label: element.label,\n      }))\n    }\n\n    return []\n  }, [options])\n\n  const internalOnChange = useCallback(\n    (newValue: number) => {\n      if (!newValue) {\n        setSelectedIndex(min)\n        onChange?.(min)\n      } else if (newValue < min) {\n        setSelectedIndex(min)\n        onChange?.(min)\n      } else if (newValue > max) {\n        setSelectedIndex(max)\n        onChange?.(max)\n      } else {\n        setSelectedIndex(newValue)\n        onChange?.(newValue)\n      }\n    },\n    [max, min, onChange],\n  )\n\n  // Make sure that min <= value <= max\n  useEffect(() => {\n    if (value < min) {\n      internalOnChange(min)\n    } else if (value > max) {\n      internalOnChange(max)\n    } else {\n      setSelectedIndex(() => value ?? min)\n    }\n  }, [value, max, min, internalOnChange])\n\n  // Get slider size\n  useEffect(() => {\n    const setWidthResize = () => {\n      setWidth(Number(refSlider.current?.offsetWidth))\n    }\n    window.addEventListener('resize', setWidthResize)\n\n    return () => {\n      window.removeEventListener('resize', setWidthResize)\n    }\n  }, [])\n\n  const leftPosition = useMemo(\n    () => ((selectedIndex - min) * 100) / (max - min),\n    [selectedIndex, max, min],\n  )\n\n  const getBackgroundSize = useMemo(\n    () => ({\n      backgroundSize: `${leftPosition}% 100%`,\n    }),\n    [leftPosition],\n  )\n\n  const styledValue = (valueNumber: string | number | null) =>\n    input && !options ? (\n      <StyledNumberInputV2\n        value={inputValue}\n        size=\"small\"\n        min={min}\n        aria-label=\"input\"\n        max={max}\n        step={step}\n        controls={false}\n        disabled={disabled}\n        data-testid={dataTestId ? `${dataTestId}-input` : 'slider-input'}\n        unit={typeof suffix === 'string' ? suffix : unit}\n        onChange={setInputValue}\n        onBlur={event => {\n          internalOnChange(Number.parseFloat(event.target.value))\n        }}\n      />\n    ) : (\n      <StyledTextValue\n        as=\"span\"\n        variant=\"bodySmall\"\n        sentiment=\"neutral\"\n        placement={direction === 'column' ? 'right' : 'center'}\n        double={false}\n        isColumn={direction === 'column'}\n        data-testid={dataTestId ? `${dataTestId}-value` : 'slider-value'}\n      >\n        {prefix}\n        {valueNumber}\n        {suffix ?? unit}\n      </StyledTextValue>\n    )\n\n  const tooltipText = useMemo(() => {\n    if (tooltip === true) {\n      return selectedIndex\n    }\n    if (tooltip) {\n      return tooltip\n    }\n\n    return undefined\n  }, [tooltip, selectedIndex])\n\n  const placementTooltip =\n    ((selectedIndex - min) / (max - min)) * (sliderWidth - THUMB_SIZE) +\n    THUMB_SIZE / 2 -\n    sliderWidth / 2\n\n  const valueToShow = options ? options[selectedIndex]?.value : selectedIndex\n\n  return (\n    <Stack gap={1} direction={direction} justifyContent=\"left\">\n      {label ? (\n        <Stack\n          justifyContent=\"space-between\"\n          direction=\"row\"\n          alignItems=\"center\"\n        >\n          <Label htmlFor={finalId} required={required}>\n            {label}\n          </Label>\n          {direction === 'column' ? styledValue(valueToShow) : null}\n        </Stack>\n      ) : null}\n\n      <Stack direction=\"row\" justifyContent=\"end\">\n        {direction === 'column' && !label ? styledValue(valueToShow) : null}\n      </Stack>\n      <Stack direction=\"column\" width=\"100%\" gap={1} justifyContent=\"center\">\n        <StyledTooltip\n          text={tooltipText}\n          placement={tooltipPosition}\n          left={placementTooltip}\n        >\n          <SliderElement\n            type=\"range\"\n            value={selectedIndex}\n            onChange={event => {\n              internalOnChange(Number.parseFloat(event.target.value))\n            }}\n            min={min}\n            max={max}\n            tabIndex={0}\n            step={step}\n            name={name}\n            disabled={!!disabled}\n            aria-disabled={disabled}\n            data-testid={dataTestId}\n            id={finalId}\n            onBlur={onBlur}\n            onFocus={onFocus}\n            aria-label={ariaLabel ?? name}\n            className={className}\n            style={getBackgroundSize}\n            data-error={!!error}\n            data-direction={direction}\n            themeSlider={theme}\n            ref={refSlider}\n            left={leftPosition}\n          />\n        </StyledTooltip>\n        {options ? (\n          <Options\n            ticks={ticks}\n            min={min}\n            max={max}\n            sliderWidth={sliderWidth}\n            value={selectedIndex}\n            step={step}\n          />\n        ) : null}\n      </Stack>\n      {direction === 'row' ? styledValue(valueToShow) : null}\n    </Stack>\n  )\n}\n"]} */"));
84
84
  const SingleSlider = ({
85
85
  name,
86
86
  tooltip,
@@ -116,6 +116,10 @@ const SingleSlider = ({
116
116
  const [selectedIndex, setSelectedIndex] = useState(safeValue);
117
117
  const refSlider = useRef(null);
118
118
  const [sliderWidth, setWidth] = useState(0);
119
+ const [inputValue, setInputValue] = useState(safeValue);
120
+ useEffect(() => {
121
+ setInputValue(selectedIndex);
122
+ }, [selectedIndex]);
119
123
  useEffect(() => {
120
124
  setWidth(Number(refSlider.current?.offsetWidth));
121
125
  }, [refSlider]);
@@ -129,9 +133,20 @@ const SingleSlider = ({
129
133
  return [];
130
134
  }, [options]);
131
135
  const internalOnChange = useCallback((newValue) => {
132
- setSelectedIndex(newValue ?? min);
133
- onChange?.(newValue ?? min);
134
- }, [min, onChange]);
136
+ if (!newValue) {
137
+ setSelectedIndex(min);
138
+ onChange?.(min);
139
+ } else if (newValue < min) {
140
+ setSelectedIndex(min);
141
+ onChange?.(min);
142
+ } else if (newValue > max) {
143
+ setSelectedIndex(max);
144
+ onChange?.(max);
145
+ } else {
146
+ setSelectedIndex(newValue);
147
+ onChange?.(newValue);
148
+ }
149
+ }, [max, min, onChange]);
135
150
  useEffect(() => {
136
151
  if (value < min) {
137
152
  internalOnChange(min);
@@ -154,12 +169,8 @@ const SingleSlider = ({
154
169
  const getBackgroundSize = useMemo(() => ({
155
170
  backgroundSize: `${leftPosition}% 100%`
156
171
  }), [leftPosition]);
157
- const styledValue = (valueNumber) => input && !options ? /* @__PURE__ */ jsx(StyledNumberInputV2, { value: typeof valueNumber === "string" ? Number.parseFloat(valueNumber) : valueNumber, size: "small", min, "aria-label": "input", max, step, controls: false, disabled, "data-testid": dataTestId ? `${dataTestId}-input` : "slider-input", unit: typeof suffix === "string" ? suffix : unit, onChange: (newVal) => {
158
- if (newVal) {
159
- internalOnChange(newVal);
160
- } else internalOnChange(0);
161
- }, onBlur: (event) => {
162
- if (!event.target.value) internalOnChange(min);
172
+ const styledValue = (valueNumber) => input && !options ? /* @__PURE__ */ jsx(StyledNumberInputV2, { value: inputValue, size: "small", min, "aria-label": "input", max, step, controls: false, disabled, "data-testid": dataTestId ? `${dataTestId}-input` : "slider-input", unit: typeof suffix === "string" ? suffix : unit, onChange: setInputValue, onBlur: (event) => {
173
+ internalOnChange(Number.parseFloat(event.target.value));
163
174
  } }) : /* @__PURE__ */ jsxs(StyledTextValue, { as: "span", variant: "bodySmall", sentiment: "neutral", placement: direction === "column" ? "right" : "center", double: false, isColumn: direction === "column", "data-testid": dataTestId ? `${dataTestId}-value` : "slider-value", children: [
164
175
  prefix,
165
176
  valueNumber,
@@ -4,9 +4,10 @@ Object.defineProperty(exports, Symbol.toStringTag, { value: "Module" });
4
4
  const jsxRuntime = require("@emotion/react/jsx-runtime");
5
5
  const _styled = require("@emotion/styled/base");
6
6
  const React = require("react");
7
+ const index$1 = require("../Label/index.cjs");
7
8
  const index = require("../Stack/index.cjs");
8
- const index$1 = require("../Text/index.cjs");
9
- const index$2 = require("../Toggle/index.cjs");
9
+ const index$2 = require("../Text/index.cjs");
10
+ const index$3 = require("../Toggle/index.cjs");
10
11
  const _interopDefaultCompat = (e) => e && typeof e === "object" && "default" in e ? e : { default: e };
11
12
  const _styled__default = /* @__PURE__ */ _interopDefaultCompat(_styled);
12
13
  function _EMOTION_STRINGIFIED_CSS_ERROR__() {
@@ -35,7 +36,7 @@ const ToggleGroupToggle = ({
35
36
  } = context;
36
37
  const ToggleName = `${groupName}.${name}`;
37
38
  const ToggleValue = `${value}`;
38
- return /* @__PURE__ */ jsxRuntime.jsx(index$2.Toggle, { onChange, checked: groupValues?.includes(ToggleValue), disabled, name: ToggleName, value: ToggleValue, helper, className, "data-testid": dataTestId, label, error: error || contextError });
39
+ return /* @__PURE__ */ jsxRuntime.jsx(index$3.Toggle, { onChange, checked: groupValues?.includes(ToggleValue), disabled, name: ToggleName, value: ToggleValue, helper, className, "data-testid": dataTestId, label, error: error || contextError });
39
40
  };
40
41
  const FieldSet = /* @__PURE__ */ _styled__default.default("fieldset", process.env.NODE_ENV === "production" ? {
41
42
  target: "e1qvneex0"
@@ -47,11 +48,12 @@ const FieldSet = /* @__PURE__ */ _styled__default.default("fieldset", process.en
47
48
  styles: "border:none;padding:0;margin:0"
48
49
  } : {
49
50
  name: "7o2an9",
50
- styles: "border:none;padding:0;margin:0/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi9ob21lL3J1bm5lci93b3JrL3VsdHJhdmlvbGV0L3VsdHJhdmlvbGV0L3BhY2thZ2VzL3VpL3NyYy9jb21wb25lbnRzL1RvZ2dsZUdyb3VwL2luZGV4LnRzeCJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUF5RWdDIiwiZmlsZSI6Ii9ob21lL3J1bm5lci93b3JrL3VsdHJhdmlvbGV0L3VsdHJhdmlvbGV0L3BhY2thZ2VzL3VpL3NyYy9jb21wb25lbnRzL1RvZ2dsZUdyb3VwL2luZGV4LnRzeCIsInNvdXJjZXNDb250ZW50IjpbIid1c2UgY2xpZW50J1xuXG5pbXBvcnQgc3R5bGVkIGZyb20gJ0BlbW90aW9uL3N0eWxlZCdcbmltcG9ydCB7XG4gIHR5cGUgQ29tcG9uZW50UHJvcHMsXG4gIHR5cGUgSW5wdXRIVE1MQXR0cmlidXRlcyxcbiAgdHlwZSBSZWFjdE5vZGUsXG4gIGNyZWF0ZUNvbnRleHQsXG4gIHVzZUNvbnRleHQsXG4gIHVzZU1lbW8sXG59IGZyb20gJ3JlYWN0J1xuaW1wb3J0IHsgU3RhY2sgfSBmcm9tICcuLi9TdGFjaydcbmltcG9ydCB7IFRleHQgfSBmcm9tICcuLi9UZXh0J1xuaW1wb3J0IHsgVG9nZ2xlIH0gZnJvbSAnLi4vVG9nZ2xlJ1xuXG50eXBlIFRvZ2dsZUdyb3VwQ29udGV4dFR5cGUgPSB7XG4gIGdyb3VwTmFtZTogc3RyaW5nXG4gIGdyb3VwVmFsdWVzOiBzdHJpbmdbXVxuICBlcnJvcjogYm9vbGVhblxufSAmIFJlcXVpcmVkPFBpY2s8SW5wdXRIVE1MQXR0cmlidXRlczxIVE1MSW5wdXRFbGVtZW50PiwgJ29uQ2hhbmdlJz4+ICZcbiAgUGljazxJbnB1dEhUTUxBdHRyaWJ1dGVzPEhUTUxJbnB1dEVsZW1lbnQ+LCAncmVxdWlyZWQnPlxuXG5jb25zdCBUb2dnbGVHcm91cENvbnRleHQgPSBjcmVhdGVDb250ZXh0PFRvZ2dsZUdyb3VwQ29udGV4dFR5cGUgfCB1bmRlZmluZWQ+KFxuICB1bmRlZmluZWQsXG4pXG5cbnR5cGUgVG9nZ2xlR3JvdXBUb2dnbGVQcm9wcyA9IE9taXQ8XG4gIENvbXBvbmVudFByb3BzPHR5cGVvZiBUb2dnbGU+LFxuICAnb25DaGFuZ2UnIHwgJ2NoZWNrZWQnIHwgJ3JlcXVpcmVkJ1xuPiAmIHtcbiAgdmFsdWU6IHN0cmluZ1xufVxuXG4vKipcbiAqIFRvZ2dsZUdyb3VwIGlzIGEgY29tcG9uZW50IHRoYXQgYWxsb3dzIHlvdSB0byBncm91cCBhIHNldCBvZiBUb2dnbGUgY29tcG9uZW50cyB0b2dldGhlciB1bmRlciB0aGUgc2FtZSBsZWdlbmQuXG4gKi9cbmV4cG9ydCBjb25zdCBUb2dnbGVHcm91cFRvZ2dsZSA9ICh7XG4gIGRpc2FibGVkLFxuICBuYW1lLFxuICB2YWx1ZSxcbiAgbGFiZWwsXG4gIGhlbHBlcixcbiAgZXJyb3IsXG4gIGNsYXNzTmFtZSxcbiAgJ2RhdGEtdGVzdGlkJzogZGF0YVRlc3RJZCxcbn06IFRvZ2dsZUdyb3VwVG9nZ2xlUHJvcHMpID0+IHtcbiAgY29uc3QgY29udGV4dCA9IHVzZUNvbnRleHQoVG9nZ2xlR3JvdXBDb250ZXh0KVxuXG4gIGlmICghY29udGV4dCkge1xuICAgIHRocm93IG5ldyBFcnJvcignVG9nZ2xlR3JvdXAuVG9nZ2xlIGNhbiBvbmx5IGJlIHVzZWQgaW5zaWRlIGEgVG9nZ2xlR3JvdXAnKVxuICB9XG5cbiAgY29uc3QgeyBncm91cE5hbWUsIG9uQ2hhbmdlLCBncm91cFZhbHVlcywgZXJyb3I6IGNvbnRleHRFcnJvciB9ID0gY29udGV4dFxuXG4gIGNvbnN0IFRvZ2dsZU5hbWUgPSBgJHtncm91cE5hbWV9LiR7bmFtZX1gXG4gIGNvbnN0IFRvZ2dsZVZhbHVlID0gYCR7dmFsdWV9YFxuXG4gIHJldHVybiAoXG4gICAgPFRvZ2dsZVxuICAgICAgb25DaGFuZ2U9e29uQ2hhbmdlfVxuICAgICAgY2hlY2tlZD17Z3JvdXBWYWx1ZXM/LmluY2x1ZGVzKFRvZ2dsZVZhbHVlKX1cbiAgICAgIGRpc2FibGVkPXtkaXNhYmxlZH1cbiAgICAgIG5hbWU9e1RvZ2dsZU5hbWV9XG4gICAgICB2YWx1ZT17VG9nZ2xlVmFsdWV9XG4gICAgICBoZWxwZXI9e2hlbHBlcn1cbiAgICAgIGNsYXNzTmFtZT17Y2xhc3NOYW1lfVxuICAgICAgZGF0YS10ZXN0aWQ9e2RhdGFUZXN0SWR9XG4gICAgICBsYWJlbD17bGFiZWx9XG4gICAgICBlcnJvcj17ZXJyb3IgfHwgY29udGV4dEVycm9yfVxuICAgIC8+XG4gIClcbn1cblxuY29uc3QgRmllbGRTZXQgPSBzdHlsZWQuZmllbGRzZXRgXG4gIGJvcmRlcjogbm9uZTtcbiAgcGFkZGluZzogMDtcbiAgbWFyZ2luOiAwO1xuYFxuXG50eXBlIFRvZ2dsZUdyb3VwUHJvcHMgPSB7XG4gIGxlZ2VuZD86IFJlYWN0Tm9kZVxuICB2YWx1ZT86IHN0cmluZ1tdXG4gIGNsYXNzTmFtZT86IHN0cmluZ1xuICBoZWxwZXI/OiBSZWFjdE5vZGVcbiAgZXJyb3I/OiBSZWFjdE5vZGVcbiAgZGlyZWN0aW9uPzogJ3JvdycgfCAnY29sdW1uJ1xuICBjaGlsZHJlbjogUmVhY3ROb2RlXG4gIHJlcXVpcmVkPzogYm9vbGVhblxuICBkZXNjcmlwdGlvbj86IFJlYWN0Tm9kZVxufSAmIFJlcXVpcmVkPFBpY2s8SW5wdXRIVE1MQXR0cmlidXRlczxIVE1MSW5wdXRFbGVtZW50PiwgJ29uQ2hhbmdlJyB8ICduYW1lJz4+ICZcbiAgUGljazxJbnB1dEhUTUxBdHRyaWJ1dGVzPEhUTUxJbnB1dEVsZW1lbnQ+LCAncmVxdWlyZWQnPlxuXG5leHBvcnQgY29uc3QgVG9nZ2xlR3JvdXAgPSAoe1xuICBsZWdlbmQsXG4gIHZhbHVlLFxuICBjbGFzc05hbWUsXG4gIGhlbHBlcixcbiAgZXJyb3IsXG4gIGRpcmVjdGlvbiA9ICdjb2x1bW4nLFxuICBjaGlsZHJlbixcbiAgb25DaGFuZ2UsXG4gIG5hbWUsXG4gIGRlc2NyaXB0aW9uLFxuICByZXF1aXJlZCA9IGZhbHNlLFxufTogVG9nZ2xlR3JvdXBQcm9wcykgPT4ge1xuICBjb25zdCBjb250ZXh0VmFsdWUgPSB1c2VNZW1vKFxuICAgICgpID0+ICh7XG4gICAgICBncm91cE5hbWU6IG5hbWUsXG4gICAgICBncm91cFZhbHVlczogdmFsdWUgPz8gW10sXG4gICAgICBvbkNoYW5nZSxcbiAgICAgIGVycm9yOiAhIWVycm9yLFxuICAgIH0pLFxuICAgIFtuYW1lLCB2YWx1ZSwgb25DaGFuZ2UsIGVycm9yXSxcbiAgKVxuXG4gIHJldHVybiAoXG4gICAgPFRvZ2dsZUdyb3VwQ29udGV4dC5Qcm92aWRlciB2YWx1ZT17Y29udGV4dFZhbHVlfT5cbiAgICAgIDxTdGFjayBnYXA9ezF9PlxuICAgICAgICA8RmllbGRTZXQgY2xhc3NOYW1lPXtjbGFzc05hbWV9PlxuICAgICAgICAgIDxTdGFjayBnYXA9ezEuNX0+XG4gICAgICAgICAgICB7bGVnZW5kIHx8IGRlc2NyaXB0aW9uID8gKFxuICAgICAgICAgICAgICA8U3RhY2sgZ2FwPXswLjV9PlxuICAgICAgICAgICAgICAgIHtsZWdlbmQgPyAoXG4gICAgICAgICAgICAgICAgICA8VGV4dFxuICAgICAgICAgICAgICAgICAgICBhcz1cImxlZ2VuZFwiXG4gICAgICAgICAgICAgICAgICAgIHZhcmlhbnQ9XCJib2R5U3Ryb25nXCJcbiAgICAgICAgICAgICAgICAgICAgc2VudGltZW50PVwibmV1dHJhbFwiXG4gICAgICAgICAgICAgICAgICAgIHByb21pbmVuY2U9XCJzdHJvbmdcIlxuICAgICAgICAgICAgICAgICAgPlxuICAgICAgICAgICAgICAgICAgICB7bGVnZW5kfSZuYnNwO1xuICAgICAgICAgICAgICAgICAgICB7cmVxdWlyZWQgPyAoXG4gICAgICAgICAgICAgICAgICAgICAgPFRleHQgYXM9XCJzdXBcIiB2YXJpYW50PVwiYm9keVwiIHNlbnRpbWVudD1cImRhbmdlclwiPlxuICAgICAgICAgICAgICAgICAgICAgICAgKlxuICAgICAgICAgICAgICAgICAgICAgIDwvVGV4dD5cbiAgICAgICAgICAgICAgICAgICAgKSA6IG51bGx9XG4gICAgICAgICAgICAgICAgICA8L1RleHQ+XG4gICAgICAgICAgICAgICAgKSA6IG51bGx9XG4gICAgICAgICAgICAgICAge2Rlc2NyaXB0aW9uID8gKFxuICAgICAgICAgICAgICAgICAgPFRleHRcbiAgICAgICAgICAgICAgICAgICAgdmFyaWFudD1cImJvZHlTbWFsbFwiXG4gICAgICAgICAgICAgICAgICAgIGFzPXt0eXBlb2YgZGVzY3JpcHRpb24gPT09ICdzdHJpbmcnID8gJ3AnIDogJ2Rpdid9XG4gICAgICAgICAgICAgICAgICAgIHByb21pbmVuY2U9XCJ3ZWFrXCJcbiAgICAgICAgICAgICAgICAgICAgc2VudGltZW50PVwibmV1dHJhbFwiXG4gICAgICAgICAgICAgICAgICA+XG4gICAgICAgICAgICAgICAgICAgIHtkZXNjcmlwdGlvbn1cbiAgICAgICAgICAgICAgICAgIDwvVGV4dD5cbiAgICAgICAgICAgICAgICApIDogbnVsbH1cbiAgICAgICAgICAgICAgPC9TdGFjaz5cbiAgICAgICAgICAgICkgOiBudWxsfVxuICAgICAgICAgICAgPFN0YWNrIGdhcD17Mn0gZGlyZWN0aW9uPXtkaXJlY3Rpb259PlxuICAgICAgICAgICAgICB7Y2hpbGRyZW59XG4gICAgICAgICAgICA8L1N0YWNrPlxuICAgICAgICAgIDwvU3RhY2s+XG4gICAgICAgIDwvRmllbGRTZXQ+XG4gICAgICAgIHtoZWxwZXIgPyAoXG4gICAgICAgICAgPFRleHQgYXM9XCJwXCIgc2VudGltZW50PVwibmV1dHJhbFwiIHZhcmlhbnQ9XCJjYXB0aW9uXCIgcHJvbWluZW5jZT1cIndlYWtcIj5cbiAgICAgICAgICAgIHtoZWxwZXJ9XG4gICAgICAgICAgPC9UZXh0PlxuICAgICAgICApIDogbnVsbH1cbiAgICAgICAge2Vycm9yID8gKFxuICAgICAgICAgIDxUZXh0IGFzPVwicFwiIHZhcmlhbnQ9XCJib2R5U21hbGxcIiBzZW50aW1lbnQ9XCJkYW5nZXJcIiBwcm9taW5lbmNlPVwid2Vha1wiPlxuICAgICAgICAgICAge2Vycm9yfVxuICAgICAgICAgIDwvVGV4dD5cbiAgICAgICAgKSA6IG51bGx9XG4gICAgICA8L1N0YWNrPlxuICAgIDwvVG9nZ2xlR3JvdXBDb250ZXh0LlByb3ZpZGVyPlxuICApXG59XG5cblRvZ2dsZUdyb3VwLlRvZ2dsZSA9IFRvZ2dsZUdyb3VwVG9nZ2xlXG4iXX0= */",
51
+ styles: "border:none;padding:0;margin:0/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi9ob21lL3J1bm5lci93b3JrL3VsdHJhdmlvbGV0L3VsdHJhdmlvbGV0L3BhY2thZ2VzL3VpL3NyYy9jb21wb25lbnRzL1RvZ2dsZUdyb3VwL2luZGV4LnRzeCJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUEwRWdDIiwiZmlsZSI6Ii9ob21lL3J1bm5lci93b3JrL3VsdHJhdmlvbGV0L3VsdHJhdmlvbGV0L3BhY2thZ2VzL3VpL3NyYy9jb21wb25lbnRzL1RvZ2dsZUdyb3VwL2luZGV4LnRzeCIsInNvdXJjZXNDb250ZW50IjpbIid1c2UgY2xpZW50J1xuXG5pbXBvcnQgc3R5bGVkIGZyb20gJ0BlbW90aW9uL3N0eWxlZCdcbmltcG9ydCB7XG4gIHR5cGUgQ29tcG9uZW50UHJvcHMsXG4gIHR5cGUgSW5wdXRIVE1MQXR0cmlidXRlcyxcbiAgdHlwZSBSZWFjdE5vZGUsXG4gIGNyZWF0ZUNvbnRleHQsXG4gIHVzZUNvbnRleHQsXG4gIHVzZU1lbW8sXG59IGZyb20gJ3JlYWN0J1xuaW1wb3J0IHsgTGFiZWwgfSBmcm9tICcuLi9MYWJlbCdcbmltcG9ydCB7IFN0YWNrIH0gZnJvbSAnLi4vU3RhY2snXG5pbXBvcnQgeyBUZXh0IH0gZnJvbSAnLi4vVGV4dCdcbmltcG9ydCB7IFRvZ2dsZSB9IGZyb20gJy4uL1RvZ2dsZSdcblxudHlwZSBUb2dnbGVHcm91cENvbnRleHRUeXBlID0ge1xuICBncm91cE5hbWU6IHN0cmluZ1xuICBncm91cFZhbHVlczogc3RyaW5nW11cbiAgZXJyb3I6IGJvb2xlYW5cbn0gJiBSZXF1aXJlZDxQaWNrPElucHV0SFRNTEF0dHJpYnV0ZXM8SFRNTElucHV0RWxlbWVudD4sICdvbkNoYW5nZSc+PiAmXG4gIFBpY2s8SW5wdXRIVE1MQXR0cmlidXRlczxIVE1MSW5wdXRFbGVtZW50PiwgJ3JlcXVpcmVkJz5cblxuY29uc3QgVG9nZ2xlR3JvdXBDb250ZXh0ID0gY3JlYXRlQ29udGV4dDxUb2dnbGVHcm91cENvbnRleHRUeXBlIHwgdW5kZWZpbmVkPihcbiAgdW5kZWZpbmVkLFxuKVxuXG50eXBlIFRvZ2dsZUdyb3VwVG9nZ2xlUHJvcHMgPSBPbWl0PFxuICBDb21wb25lbnRQcm9wczx0eXBlb2YgVG9nZ2xlPixcbiAgJ29uQ2hhbmdlJyB8ICdjaGVja2VkJyB8ICdyZXF1aXJlZCdcbj4gJiB7XG4gIHZhbHVlOiBzdHJpbmdcbn1cblxuLyoqXG4gKiBUb2dnbGVHcm91cCBpcyBhIGNvbXBvbmVudCB0aGF0IGFsbG93cyB5b3UgdG8gZ3JvdXAgYSBzZXQgb2YgVG9nZ2xlIGNvbXBvbmVudHMgdG9nZXRoZXIgdW5kZXIgdGhlIHNhbWUgbGVnZW5kLlxuICovXG5leHBvcnQgY29uc3QgVG9nZ2xlR3JvdXBUb2dnbGUgPSAoe1xuICBkaXNhYmxlZCxcbiAgbmFtZSxcbiAgdmFsdWUsXG4gIGxhYmVsLFxuICBoZWxwZXIsXG4gIGVycm9yLFxuICBjbGFzc05hbWUsXG4gICdkYXRhLXRlc3RpZCc6IGRhdGFUZXN0SWQsXG59OiBUb2dnbGVHcm91cFRvZ2dsZVByb3BzKSA9PiB7XG4gIGNvbnN0IGNvbnRleHQgPSB1c2VDb250ZXh0KFRvZ2dsZUdyb3VwQ29udGV4dClcblxuICBpZiAoIWNvbnRleHQpIHtcbiAgICB0aHJvdyBuZXcgRXJyb3IoJ1RvZ2dsZUdyb3VwLlRvZ2dsZSBjYW4gb25seSBiZSB1c2VkIGluc2lkZSBhIFRvZ2dsZUdyb3VwJylcbiAgfVxuXG4gIGNvbnN0IHsgZ3JvdXBOYW1lLCBvbkNoYW5nZSwgZ3JvdXBWYWx1ZXMsIGVycm9yOiBjb250ZXh0RXJyb3IgfSA9IGNvbnRleHRcblxuICBjb25zdCBUb2dnbGVOYW1lID0gYCR7Z3JvdXBOYW1lfS4ke25hbWV9YFxuICBjb25zdCBUb2dnbGVWYWx1ZSA9IGAke3ZhbHVlfWBcblxuICByZXR1cm4gKFxuICAgIDxUb2dnbGVcbiAgICAgIG9uQ2hhbmdlPXtvbkNoYW5nZX1cbiAgICAgIGNoZWNrZWQ9e2dyb3VwVmFsdWVzPy5pbmNsdWRlcyhUb2dnbGVWYWx1ZSl9XG4gICAgICBkaXNhYmxlZD17ZGlzYWJsZWR9XG4gICAgICBuYW1lPXtUb2dnbGVOYW1lfVxuICAgICAgdmFsdWU9e1RvZ2dsZVZhbHVlfVxuICAgICAgaGVscGVyPXtoZWxwZXJ9XG4gICAgICBjbGFzc05hbWU9e2NsYXNzTmFtZX1cbiAgICAgIGRhdGEtdGVzdGlkPXtkYXRhVGVzdElkfVxuICAgICAgbGFiZWw9e2xhYmVsfVxuICAgICAgZXJyb3I9e2Vycm9yIHx8IGNvbnRleHRFcnJvcn1cbiAgICAvPlxuICApXG59XG5cbmNvbnN0IEZpZWxkU2V0ID0gc3R5bGVkLmZpZWxkc2V0YFxuICBib3JkZXI6IG5vbmU7XG4gIHBhZGRpbmc6IDA7XG4gIG1hcmdpbjogMDtcbmBcblxudHlwZSBUb2dnbGVHcm91cFByb3BzID0ge1xuICBsZWdlbmQ/OiBzdHJpbmdcbiAgbGVnZW5kRGVzY3JpcHRpb24/OiBSZWFjdE5vZGVcbiAgdmFsdWU/OiBzdHJpbmdbXVxuICBjbGFzc05hbWU/OiBzdHJpbmdcbiAgaGVscGVyPzogUmVhY3ROb2RlXG4gIGVycm9yPzogUmVhY3ROb2RlXG4gIGRpcmVjdGlvbj86ICdyb3cnIHwgJ2NvbHVtbidcbiAgY2hpbGRyZW46IFJlYWN0Tm9kZVxuICByZXF1aXJlZD86IGJvb2xlYW5cbiAgZGVzY3JpcHRpb24/OiBSZWFjdE5vZGVcbn0gJiBSZXF1aXJlZDxQaWNrPElucHV0SFRNTEF0dHJpYnV0ZXM8SFRNTElucHV0RWxlbWVudD4sICdvbkNoYW5nZScgfCAnbmFtZSc+PiAmXG4gIFBpY2s8SW5wdXRIVE1MQXR0cmlidXRlczxIVE1MSW5wdXRFbGVtZW50PiwgJ3JlcXVpcmVkJz5cblxuZXhwb3J0IGNvbnN0IFRvZ2dsZUdyb3VwID0gKHtcbiAgbGVnZW5kLFxuICBsZWdlbmREZXNjcmlwdGlvbixcbiAgdmFsdWUsXG4gIGNsYXNzTmFtZSxcbiAgaGVscGVyLFxuICBlcnJvcixcbiAgZGlyZWN0aW9uID0gJ2NvbHVtbicsXG4gIGNoaWxkcmVuLFxuICBvbkNoYW5nZSxcbiAgbmFtZSxcbiAgZGVzY3JpcHRpb24sXG4gIHJlcXVpcmVkID0gZmFsc2UsXG59OiBUb2dnbGVHcm91cFByb3BzKSA9PiB7XG4gIGNvbnN0IGNvbnRleHRWYWx1ZSA9IHVzZU1lbW8oXG4gICAgKCkgPT4gKHtcbiAgICAgIGdyb3VwTmFtZTogbmFtZSxcbiAgICAgIGdyb3VwVmFsdWVzOiB2YWx1ZSA/PyBbXSxcbiAgICAgIG9uQ2hhbmdlLFxuICAgICAgZXJyb3I6ICEhZXJyb3IsXG4gICAgfSksXG4gICAgW25hbWUsIHZhbHVlLCBvbkNoYW5nZSwgZXJyb3JdLFxuICApXG5cbiAgcmV0dXJuIChcbiAgICA8VG9nZ2xlR3JvdXBDb250ZXh0LlByb3ZpZGVyIHZhbHVlPXtjb250ZXh0VmFsdWV9PlxuICAgICAgPFN0YWNrIGdhcD17MX0+XG4gICAgICAgIDxGaWVsZFNldCBjbGFzc05hbWU9e2NsYXNzTmFtZX0+XG4gICAgICAgICAgPFN0YWNrIGdhcD17MS41fT5cbiAgICAgICAgICAgIHtsZWdlbmQgfHwgZGVzY3JpcHRpb24gPyAoXG4gICAgICAgICAgICAgIDxTdGFjayBnYXA9ezAuNX0+XG4gICAgICAgICAgICAgICAge2xlZ2VuZCA/IChcbiAgICAgICAgICAgICAgICAgIDxMYWJlbFxuICAgICAgICAgICAgICAgICAgICBhcz1cImxlZ2VuZFwiXG4gICAgICAgICAgICAgICAgICAgIHJlcXVpcmVkPXtyZXF1aXJlZH1cbiAgICAgICAgICAgICAgICAgICAgbGFiZWxEZXNjcmlwdGlvbj17bGVnZW5kRGVzY3JpcHRpb259XG4gICAgICAgICAgICAgICAgICA+XG4gICAgICAgICAgICAgICAgICAgIHtsZWdlbmR9XG4gICAgICAgICAgICAgICAgICA8L0xhYmVsPlxuICAgICAgICAgICAgICAgICkgOiBudWxsfVxuICAgICAgICAgICAgICAgIHtkZXNjcmlwdGlvbiA/IChcbiAgICAgICAgICAgICAgICAgIDxUZXh0XG4gICAgICAgICAgICAgICAgICAgIHZhcmlhbnQ9XCJib2R5U21hbGxcIlxuICAgICAgICAgICAgICAgICAgICBhcz17dHlwZW9mIGRlc2NyaXB0aW9uID09PSAnc3RyaW5nJyA/ICdwJyA6ICdkaXYnfVxuICAgICAgICAgICAgICAgICAgICBwcm9taW5lbmNlPVwid2Vha1wiXG4gICAgICAgICAgICAgICAgICAgIHNlbnRpbWVudD1cIm5ldXRyYWxcIlxuICAgICAgICAgICAgICAgICAgPlxuICAgICAgICAgICAgICAgICAgICB7ZGVzY3JpcHRpb259XG4gICAgICAgICAgICAgICAgICA8L1RleHQ+XG4gICAgICAgICAgICAgICAgKSA6IG51bGx9XG4gICAgICAgICAgICAgIDwvU3RhY2s+XG4gICAgICAgICAgICApIDogbnVsbH1cbiAgICAgICAgICAgIDxTdGFjayBnYXA9ezJ9IGRpcmVjdGlvbj17ZGlyZWN0aW9ufT5cbiAgICAgICAgICAgICAge2NoaWxkcmVufVxuICAgICAgICAgICAgPC9TdGFjaz5cbiAgICAgICAgICA8L1N0YWNrPlxuICAgICAgICA8L0ZpZWxkU2V0PlxuICAgICAgICB7aGVscGVyID8gKFxuICAgICAgICAgIDxUZXh0IGFzPVwicFwiIHNlbnRpbWVudD1cIm5ldXRyYWxcIiB2YXJpYW50PVwiY2FwdGlvblwiIHByb21pbmVuY2U9XCJ3ZWFrXCI+XG4gICAgICAgICAgICB7aGVscGVyfVxuICAgICAgICAgIDwvVGV4dD5cbiAgICAgICAgKSA6IG51bGx9XG4gICAgICAgIHtlcnJvciA/IChcbiAgICAgICAgICA8VGV4dCBhcz1cInBcIiB2YXJpYW50PVwiYm9keVNtYWxsXCIgc2VudGltZW50PVwiZGFuZ2VyXCIgcHJvbWluZW5jZT1cIndlYWtcIj5cbiAgICAgICAgICAgIHtlcnJvcn1cbiAgICAgICAgICA8L1RleHQ+XG4gICAgICAgICkgOiBudWxsfVxuICAgICAgPC9TdGFjaz5cbiAgICA8L1RvZ2dsZUdyb3VwQ29udGV4dC5Qcm92aWRlcj5cbiAgKVxufVxuXG5Ub2dnbGVHcm91cC5Ub2dnbGUgPSBUb2dnbGVHcm91cFRvZ2dsZVxuIl19 */",
51
52
  toString: _EMOTION_STRINGIFIED_CSS_ERROR__
52
53
  });
53
54
  const ToggleGroup = ({
54
55
  legend,
56
+ legendDescription,
55
57
  value,
56
58
  className,
57
59
  helper,
@@ -72,17 +74,13 @@ const ToggleGroup = ({
72
74
  return /* @__PURE__ */ jsxRuntime.jsx(ToggleGroupContext.Provider, { value: contextValue, children: /* @__PURE__ */ jsxRuntime.jsxs(index.Stack, { gap: 1, children: [
73
75
  /* @__PURE__ */ jsxRuntime.jsx(FieldSet, { className, children: /* @__PURE__ */ jsxRuntime.jsxs(index.Stack, { gap: 1.5, children: [
74
76
  legend || description ? /* @__PURE__ */ jsxRuntime.jsxs(index.Stack, { gap: 0.5, children: [
75
- legend ? /* @__PURE__ */ jsxRuntime.jsxs(index$1.Text, { as: "legend", variant: "bodyStrong", sentiment: "neutral", prominence: "strong", children: [
76
- legend,
77
- " ",
78
- required ? /* @__PURE__ */ jsxRuntime.jsx(index$1.Text, { as: "sup", variant: "body", sentiment: "danger", children: "*" }) : null
79
- ] }) : null,
80
- description ? /* @__PURE__ */ jsxRuntime.jsx(index$1.Text, { variant: "bodySmall", as: typeof description === "string" ? "p" : "div", prominence: "weak", sentiment: "neutral", children: description }) : null
77
+ legend ? /* @__PURE__ */ jsxRuntime.jsx(index$1.Label, { as: "legend", required, labelDescription: legendDescription, children: legend }) : null,
78
+ description ? /* @__PURE__ */ jsxRuntime.jsx(index$2.Text, { variant: "bodySmall", as: typeof description === "string" ? "p" : "div", prominence: "weak", sentiment: "neutral", children: description }) : null
81
79
  ] }) : null,
82
80
  /* @__PURE__ */ jsxRuntime.jsx(index.Stack, { gap: 2, direction, children })
83
81
  ] }) }),
84
- helper ? /* @__PURE__ */ jsxRuntime.jsx(index$1.Text, { as: "p", sentiment: "neutral", variant: "caption", prominence: "weak", children: helper }) : null,
85
- error ? /* @__PURE__ */ jsxRuntime.jsx(index$1.Text, { as: "p", variant: "bodySmall", sentiment: "danger", prominence: "weak", children: error }) : null
82
+ helper ? /* @__PURE__ */ jsxRuntime.jsx(index$2.Text, { as: "p", sentiment: "neutral", variant: "caption", prominence: "weak", children: helper }) : null,
83
+ error ? /* @__PURE__ */ jsxRuntime.jsx(index$2.Text, { as: "p", variant: "bodySmall", sentiment: "danger", prominence: "weak", children: error }) : null
86
84
  ] }) });
87
85
  };
88
86
  ToggleGroup.Toggle = ToggleGroupToggle;
@@ -8,7 +8,8 @@ type ToggleGroupToggleProps = Omit<ComponentProps<typeof Toggle>, 'onChange' | '
8
8
  */
9
9
  export declare const ToggleGroupToggle: ({ disabled, name, value, label, helper, error, className, "data-testid": dataTestId, }: ToggleGroupToggleProps) => import("@emotion/react/jsx-runtime").JSX.Element;
10
10
  type ToggleGroupProps = {
11
- legend?: ReactNode;
11
+ legend?: string;
12
+ legendDescription?: ReactNode;
12
13
  value?: string[];
13
14
  className?: string;
14
15
  helper?: ReactNode;
@@ -19,7 +20,7 @@ type ToggleGroupProps = {
19
20
  description?: ReactNode;
20
21
  } & Required<Pick<InputHTMLAttributes<HTMLInputElement>, 'onChange' | 'name'>> & Pick<InputHTMLAttributes<HTMLInputElement>, 'required'>;
21
22
  export declare const ToggleGroup: {
22
- ({ legend, value, className, helper, error, direction, children, onChange, name, description, required, }: ToggleGroupProps): import("@emotion/react/jsx-runtime").JSX.Element;
23
+ ({ legend, legendDescription, value, className, helper, error, direction, children, onChange, name, description, required, }: ToggleGroupProps): import("@emotion/react/jsx-runtime").JSX.Element;
23
24
  Toggle: ({ disabled, name, value, label, helper, error, className, "data-testid": dataTestId, }: ToggleGroupToggleProps) => import("@emotion/react/jsx-runtime").JSX.Element;
24
25
  };
25
26
  export {};