@ultraviolet/ui 1.81.2 → 1.82.0

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 (57) hide show
  1. package/dist/components/CheckboxGroup/index.cjs +3 -3
  2. package/dist/components/CheckboxGroup/index.d.ts +1 -1
  3. package/dist/components/CheckboxGroup/index.js +3 -3
  4. package/dist/components/Chip/ChipContext.d.ts +1 -1
  5. package/dist/components/DateInput/components/Popup.cjs +2 -2
  6. package/dist/components/DateInput/components/Popup.d.ts +1 -1
  7. package/dist/components/DateInput/components/Popup.js +2 -2
  8. package/dist/components/Drawer/index.cjs +114 -0
  9. package/dist/components/Drawer/index.js +112 -0
  10. package/dist/components/Expandable/index.cjs +8 -4
  11. package/dist/components/Expandable/index.js +8 -4
  12. package/dist/components/List/HeaderRow.cjs +7 -7
  13. package/dist/components/List/HeaderRow.js +7 -7
  14. package/dist/components/List/Row.cjs +15 -8
  15. package/dist/components/List/Row.js +15 -8
  16. package/dist/components/List/constants.cjs +4 -0
  17. package/dist/components/List/constants.d.ts +2 -2
  18. package/dist/components/List/constants.js +4 -0
  19. package/dist/components/List/index.cjs +5 -5
  20. package/dist/components/List/index.js +5 -5
  21. package/dist/components/MenuV2/index.d.ts +1 -1
  22. package/dist/components/NumberInput/index.cjs +6 -6
  23. package/dist/components/NumberInput/index.js +6 -6
  24. package/dist/components/Popover/index.d.ts +1 -1
  25. package/dist/components/Popup/helpers.d.ts +2 -2
  26. package/dist/components/Popup/index.cjs +5 -5
  27. package/dist/components/Popup/index.d.ts +1 -1
  28. package/dist/components/Popup/index.js +5 -5
  29. package/dist/components/RadioGroup/index.cjs +2 -2
  30. package/dist/components/RadioGroup/index.d.ts +1 -1
  31. package/dist/components/RadioGroup/index.js +2 -2
  32. package/dist/components/SelectInputV2/Dropdown.cjs +9 -9
  33. package/dist/components/SelectInputV2/Dropdown.d.ts +1 -1
  34. package/dist/components/SelectInputV2/Dropdown.js +9 -9
  35. package/dist/components/SelectInputV2/SelectBar.cjs +5 -5
  36. package/dist/components/SelectInputV2/SelectBar.d.ts +1 -1
  37. package/dist/components/SelectInputV2/SelectBar.js +5 -5
  38. package/dist/components/SelectInputV2/SelectInputProvider.d.ts +1 -1
  39. package/dist/components/SelectInputV2/index.cjs +2 -2
  40. package/dist/components/SelectInputV2/index.js +2 -2
  41. package/dist/components/SelectableCardGroup/index.cjs +2 -2
  42. package/dist/components/SelectableCardGroup/index.d.ts +1 -1
  43. package/dist/components/SelectableCardGroup/index.js +2 -2
  44. package/dist/components/Slider/styles.d.ts +1 -1
  45. package/dist/components/Table/index.cjs +5 -5
  46. package/dist/components/Table/index.js +5 -5
  47. package/dist/components/Tabs/index.cjs +3 -3
  48. package/dist/components/Tabs/index.d.ts +3 -3
  49. package/dist/components/Tabs/index.js +3 -3
  50. package/dist/components/ToggleGroup/index.cjs +2 -2
  51. package/dist/components/ToggleGroup/index.d.ts +1 -1
  52. package/dist/components/ToggleGroup/index.js +2 -2
  53. package/dist/components/index.d.ts +1 -0
  54. package/dist/index.cjs +2 -0
  55. package/dist/index.js +2 -0
  56. package/dist/style.css +4 -0
  57. package/package.json +8 -8
@@ -25,7 +25,7 @@ const StyledSelectButton = /* @__PURE__ */ _styled(Button, process.env.NODE_ENV
25
25
  label: "StyledSelectButton"
26
26
  })("margin:0 ", ({
27
27
  theme
28
- }) => theme.space["1"], ";width:32px;height:32px;" + (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/NumberInput/index.tsx"],"names":[],"mappings":"AAqCyC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/NumberInput/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { MinusIcon, PlusIcon } from '@ultraviolet/icons'\nimport type {\n  ChangeEventHandler,\n  FocusEventHandler,\n  InputHTMLAttributes,\n  KeyboardEventHandler,\n  MutableRefObject,\n} from 'react'\nimport { useId, useMemo, useRef, useState } from 'react'\nimport { Button } from '../Button'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\nimport { Tooltip } from '../Tooltip'\nimport {\n  bounded,\n  getMinusRoundedValue,\n  getPlusRoundedValue,\n  roundStep,\n} from './helpers'\n\nconst containerSizes = {\n  large: 48,\n  medium: 40,\n  small: 32,\n}\n\ntype ContainerSizesType = keyof typeof containerSizes\n\nconst iconSizes = {\n  large: 26,\n  medium: 24,\n  small: 22,\n}\n\nconst BASE_INPUT_WIDTH = 34\n\nconst StyledSelectButton = styled(Button)`\n  margin: 0 ${({ theme }) => theme.space['1']};\n  width: 32px;\n  height: 32px;\n`\n\nconst StyledCenterBox = styled('div', {\n  shouldForwardProp: prop => !['size'].includes(prop),\n})<{ size: ContainerSizesType }>`\n  display: flex;\n  flex: 1;\n  flex-direction: row;\n  height: ${({ size }) => (size === 'small' ? '24px' : '32px')};\n  align-items: center;\n  outline: none;\n  justify-content: center;\n  border-radius: ${({ theme }) => theme.radii.default};\n  border: 1px solid transparent;\n  max-width: 100%;\n`\n\nconst StyledInput = styled.input`\n  color: ${({ theme }) => theme.colors.neutral.text};\n  background-color: transparent;\n  font-size: ${({ theme }) => theme.typography.bodyStrong.fontSize};\n  border: none;\n  outline: none;\n  position: relative;\n  margin-right: ${({ theme }) => theme.space['0.5']};\n  max-width: 100%;\n  font-weight: ${({ theme }) => theme.typography.bodyStrong.weight};\n  text-align: center;\n\n  &::-webkit-outer-spin-button,\n  &::-webkit-inner-spin-button {\n    -webkit-appearance: none;\n    margin: 0;\n  }\n\n  ::placeholder {\n    color: ${({ theme }) => theme.colors.neutral.textWeak};\n  }\n\n  -moz-appearance: textfield;\n\n  &[disabled] {\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n    cursor: not-allowed;\n  }\n`\n\nconst StyledText = styled('span', {\n  shouldForwardProp: prop => !['disabled'].includes(prop),\n})<{ disabled: boolean }>`\n  color: ${({ theme, disabled }) =>\n    disabled ? theme.colors.neutral.textDisabled : theme.colors.neutral.text};\n  user-select: none;\n  margin-right: ${({ theme }) => theme.space['1']};\n`\n\nconst StyledContainer = styled('div', {\n  shouldForwardProp: prop => !['size'].includes(prop),\n})<{ size: ContainerSizesType }>`\n  background-color: ${({ theme }) => theme.colors.neutral.background};\n  display: flex;\n  flex-direction: row;\n  align-items: center;\n  align-self: stretch;\n  font-weight: 500;\n  height: ${({ size }) => containerSizes[size]}px;\n  border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  border-radius: ${({ theme }) => theme.radii.default};\n\n  &[data-error='true'] {\n    border: 1px solid ${({ theme }) => theme.colors.danger.border};\n  }\n\n  &[aria-disabled='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    cursor: not-allowed;\n  }\n\n  &:not([aria-disabled='true']) {\n    ${StyledCenterBox}:hover,\n    ${StyledCenterBox}:focus {\n      border: 1px solid ${({ theme }) => theme.colors.primary.borderHover};\n    }\n\n    ${StyledCenterBox}:focus-within {\n      box-shadow: ${({ theme }) => theme.shadows.focusPrimary};\n      border: 1px solid ${({ theme }) => theme.colors.primary.borderHover};\n    }\n  }\n`\n\ntype NumberInputProps = {\n  disabled?: boolean\n  maxValue?: number\n  minValue?: number\n  name?: string\n  onChange?: (input: number | undefined) => void\n  onMaxCrossed?: () => void\n  onMinCrossed?: () => void\n  size?: ContainerSizesType\n  /**\n   * Define how much will stepper increase / decrease each time you click on + / - button.\n   */\n  step?: number\n  /**\n   * Text displayed into component at the right of number value.\n   */\n  text?: string\n  defaultValue?: number\n  value?: number | null\n  disabledTooltip?: string\n  className?: string\n  'data-testid'?: string\n  label?: string\n  'aria-label'?: string\n  'aria-describedby'?: string\n  id?: string\n  placeholder?: string\n  error?: string | boolean\n} & Omit<\n  InputHTMLAttributes<HTMLInputElement>,\n  'size' | 'onChange' | 'value' | 'defaultValue'\n>\n\n/**\n * @deprecated This component is deprecated. Please use `NumberInputV2` instead.\n */\nexport const NumberInput = ({\n  disabled = false,\n  maxValue,\n  minValue = 0,\n  name = 'numberinput',\n  onChange,\n  onFocus,\n  onBlur,\n  onMaxCrossed,\n  onMinCrossed,\n  size = 'large',\n  step = 1,\n  text,\n  defaultValue,\n  value,\n  disabledTooltip,\n  className,\n  label,\n  id,\n  placeholder,\n  error,\n  'aria-label': ariaLabel,\n  'aria-describedby': ariaDescribedBy,\n  'data-testid': dataTestId,\n}: NumberInputProps) => {\n  const inputRef =\n    useRef<HTMLInputElement>() as MutableRefObject<HTMLInputElement>\n\n  const uniqueId = useId()\n\n  // local state used if component is not controlled (no value prop provided)\n  const [inputValue, setInputValue] = useState<number | undefined>(() => {\n    if (defaultValue && minValue && defaultValue < minValue) {\n      return minValue\n    }\n    if (defaultValue && maxValue && defaultValue > maxValue) {\n      return maxValue\n    }\n\n    return defaultValue\n  })\n\n  const currentValue =\n    value !== undefined && value !== null ? value : inputValue\n\n  const setValue = (\n    newValue: number | undefined,\n    /**\n     * If true, will check if newValue is between minValue and maxValue and set it to minValue or maxValue if it's not.\n     */\n    hasMinMaxVerification = true,\n  ) => {\n    let nextValue = newValue\n    if (value === undefined && hasMinMaxVerification) {\n      if (newValue !== undefined && newValue < minValue) {\n        nextValue = minValue\n      }\n\n      if (\n        newValue !== undefined &&\n        maxValue !== undefined &&\n        newValue > maxValue\n      ) {\n        nextValue = maxValue\n      }\n    }\n    setInputValue(nextValue)\n    onChange?.(nextValue)\n  }\n\n  const offsetFn = (direction: number) => () => {\n    const localValue = currentValue ?? 0\n    const newValue =\n      localValue % step === 0 ? localValue + step * direction : localValue\n    const roundedValue = roundStep(newValue, step, direction)\n\n    setValue(roundedValue)\n  }\n\n  const handleChange: ChangeEventHandler<HTMLInputElement> = event => {\n    setValue(\n      event.currentTarget.value ? Number(event.currentTarget.value) : undefined,\n      false,\n    )\n  }\n\n  const handleOnBlur: FocusEventHandler<HTMLInputElement> = event => {\n    if (currentValue) {\n      const boundedValue = bounded(\n        currentValue,\n        minValue ?? currentValue,\n        maxValue ?? currentValue,\n      )\n\n      if (maxValue && currentValue > maxValue) onMaxCrossed?.()\n      if (minValue && currentValue < minValue) onMinCrossed?.()\n\n      setValue(boundedValue)\n\n      onBlur?.(event)\n    }\n  }\n\n  const onKeyDown: KeyboardEventHandler = event => {\n    if (event.key === 'ArrowUp') {\n      event.stopPropagation()\n      event.preventDefault()\n\n      const direction = 1\n      const localValue = currentValue ?? 0\n\n      const newValue =\n        localValue % step === 0 ? localValue + step * direction : localValue\n      const roundedValue = roundStep(newValue, step, direction)\n\n      if (maxValue === undefined) {\n        setValue(roundedValue)\n\n        return\n      }\n\n      setValue(Math.min(roundedValue, maxValue))\n    }\n\n    if (event.key === 'ArrowDown') {\n      event.stopPropagation()\n      event.preventDefault()\n\n      const direction = -1\n      const localValue = currentValue ?? 0\n\n      const newValue =\n        localValue % step === 0 ? localValue + step * direction : localValue\n      const roundedValue = roundStep(newValue, step, direction)\n\n      setValue(Math.max(roundedValue, minValue))\n    }\n  }\n\n  const isMinusDisabled = useMemo(() => {\n    if (disabled) return true\n    if (currentValue === undefined) return false\n    if (getMinusRoundedValue(currentValue, step) < minValue) {\n      return true\n    }\n\n    return disabled\n  }, [currentValue, disabled, minValue, step])\n\n  const isPlusDisabled = useMemo(() => {\n    if (disabled) return true\n    if (currentValue === undefined) return false\n    if (maxValue && getPlusRoundedValue(currentValue, step) > maxValue) {\n      return true\n    }\n\n    return disabled\n  }, [currentValue, disabled, maxValue, step])\n\n  const inputWidth = useMemo(() => {\n    if (placeholder && currentValue === undefined) {\n      return placeholder.length * 12\n    }\n\n    if (currentValue !== undefined) {\n      return currentValue.toString().length * 16\n    }\n\n    return BASE_INPUT_WIDTH\n  }, [currentValue, placeholder])\n\n  return (\n    <Stack gap={1}>\n      {label ? (\n        <Text variant=\"bodyStrong\" as=\"label\" htmlFor={id || uniqueId}>\n          {label}\n        </Text>\n      ) : null}\n      <Stack gap={0.5}>\n        <StyledContainer\n          aria-disabled={disabled}\n          data-error={!!error}\n          size={size}\n          className={className}\n          data-testid={dataTestId}\n        >\n          <Tooltip text={isMinusDisabled && disabledTooltip}>\n            <StyledSelectButton\n              onClick={offsetFn(-1)}\n              disabled={isMinusDisabled}\n              aria-label=\"Minus\"\n              type=\"button\"\n              variant=\"ghost\"\n              sentiment=\"primary\"\n              size=\"small\"\n            >\n              <MinusIcon\n                size={iconSizes[size]}\n                sentiment=\"primary\"\n                disabled={isMinusDisabled}\n              />\n            </StyledSelectButton>\n          </Tooltip>\n\n          <StyledCenterBox\n            size={size}\n            onClick={() => {\n              if (inputRef?.current) {\n                inputRef.current.focus()\n              }\n            }}\n            aria-live=\"assertive\"\n            role=\"status\"\n          >\n            <StyledInput\n              disabled={disabled}\n              name={name}\n              onBlur={handleOnBlur}\n              onChange={handleChange}\n              onFocus={onFocus}\n              onKeyDown={onKeyDown}\n              ref={inputRef}\n              style={{\n                width: inputWidth,\n              }}\n              value={currentValue !== undefined ? currentValue.toString() : ''} // A dom element can only have string attributes.\n              type=\"number\"\n              id={id || uniqueId}\n              aria-label={!label && !ariaLabel ? 'Number Input' : ariaLabel}\n              aria-describedby={ariaDescribedBy}\n              placeholder={placeholder}\n            />\n            {currentValue !== undefined ? (\n              <StyledText disabled={disabled}>{text}</StyledText>\n            ) : null}\n          </StyledCenterBox>\n\n          <Tooltip text={isPlusDisabled && disabledTooltip}>\n            <StyledSelectButton\n              onClick={offsetFn(1)}\n              disabled={isPlusDisabled}\n              aria-label=\"Plus\"\n              type=\"button\"\n              variant=\"ghost\"\n              sentiment=\"primary\"\n              size=\"small\"\n            >\n              <PlusIcon\n                size={iconSizes[size]}\n                sentiment=\"primary\"\n                disabled={isPlusDisabled}\n              />\n            </StyledSelectButton>\n          </Tooltip>\n        </StyledContainer>\n        {typeof error === 'string' ? (\n          <Text\n            as=\"span\"\n            variant=\"bodySmall\"\n            sentiment=\"danger\"\n            prominence=\"weak\"\n          >\n            {error}\n          </Text>\n        ) : null}\n      </Stack>\n    </Stack>\n  )\n}\n"]} */"));
28
+ }) => theme.space["1"], ";width:32px;height:32px;" + (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/NumberInput/index.tsx"],"names":[],"mappings":"AAoCyC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/NumberInput/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { MinusIcon, PlusIcon } from '@ultraviolet/icons'\nimport type {\n  ChangeEventHandler,\n  FocusEventHandler,\n  InputHTMLAttributes,\n  KeyboardEventHandler,\n} from 'react'\nimport { useId, useMemo, useRef, useState } from 'react'\nimport { Button } from '../Button'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\nimport { Tooltip } from '../Tooltip'\nimport {\n  bounded,\n  getMinusRoundedValue,\n  getPlusRoundedValue,\n  roundStep,\n} from './helpers'\n\nconst containerSizes = {\n  large: 48,\n  medium: 40,\n  small: 32,\n}\n\ntype ContainerSizesType = keyof typeof containerSizes\n\nconst iconSizes = {\n  large: 26,\n  medium: 24,\n  small: 22,\n}\n\nconst BASE_INPUT_WIDTH = 34\n\nconst StyledSelectButton = styled(Button)`\n  margin: 0 ${({ theme }) => theme.space['1']};\n  width: 32px;\n  height: 32px;\n`\n\nconst StyledCenterBox = styled('div', {\n  shouldForwardProp: prop => !['size'].includes(prop),\n})<{ size: ContainerSizesType }>`\n  display: flex;\n  flex: 1;\n  flex-direction: row;\n  height: ${({ size }) => (size === 'small' ? '24px' : '32px')};\n  align-items: center;\n  outline: none;\n  justify-content: center;\n  border-radius: ${({ theme }) => theme.radii.default};\n  border: 1px solid transparent;\n  max-width: 100%;\n`\n\nconst StyledInput = styled.input`\n  color: ${({ theme }) => theme.colors.neutral.text};\n  background-color: transparent;\n  font-size: ${({ theme }) => theme.typography.bodyStrong.fontSize};\n  border: none;\n  outline: none;\n  position: relative;\n  margin-right: ${({ theme }) => theme.space['0.5']};\n  max-width: 100%;\n  font-weight: ${({ theme }) => theme.typography.bodyStrong.weight};\n  text-align: center;\n\n  &::-webkit-outer-spin-button,\n  &::-webkit-inner-spin-button {\n    -webkit-appearance: none;\n    margin: 0;\n  }\n\n  ::placeholder {\n    color: ${({ theme }) => theme.colors.neutral.textWeak};\n  }\n\n  -moz-appearance: textfield;\n\n  &[disabled] {\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n    cursor: not-allowed;\n  }\n`\n\nconst StyledText = styled('span', {\n  shouldForwardProp: prop => !['disabled'].includes(prop),\n})<{ disabled: boolean }>`\n  color: ${({ theme, disabled }) =>\n    disabled ? theme.colors.neutral.textDisabled : theme.colors.neutral.text};\n  user-select: none;\n  margin-right: ${({ theme }) => theme.space['1']};\n`\n\nconst StyledContainer = styled('div', {\n  shouldForwardProp: prop => !['size'].includes(prop),\n})<{ size: ContainerSizesType }>`\n  background-color: ${({ theme }) => theme.colors.neutral.background};\n  display: flex;\n  flex-direction: row;\n  align-items: center;\n  align-self: stretch;\n  font-weight: 500;\n  height: ${({ size }) => containerSizes[size]}px;\n  border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  border-radius: ${({ theme }) => theme.radii.default};\n\n  &[data-error='true'] {\n    border: 1px solid ${({ theme }) => theme.colors.danger.border};\n  }\n\n  &[aria-disabled='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    cursor: not-allowed;\n  }\n\n  &:not([aria-disabled='true']) {\n    ${StyledCenterBox}:hover,\n    ${StyledCenterBox}:focus {\n      border: 1px solid ${({ theme }) => theme.colors.primary.borderHover};\n    }\n\n    ${StyledCenterBox}:focus-within {\n      box-shadow: ${({ theme }) => theme.shadows.focusPrimary};\n      border: 1px solid ${({ theme }) => theme.colors.primary.borderHover};\n    }\n  }\n`\n\ntype NumberInputProps = {\n  disabled?: boolean\n  maxValue?: number\n  minValue?: number\n  name?: string\n  onChange?: (input: number | undefined) => void\n  onMaxCrossed?: () => void\n  onMinCrossed?: () => void\n  size?: ContainerSizesType\n  /**\n   * Define how much will stepper increase / decrease each time you click on + / - button.\n   */\n  step?: number\n  /**\n   * Text displayed into component at the right of number value.\n   */\n  text?: string\n  defaultValue?: number\n  value?: number | null\n  disabledTooltip?: string\n  className?: string\n  'data-testid'?: string\n  label?: string\n  'aria-label'?: string\n  'aria-describedby'?: string\n  id?: string\n  placeholder?: string\n  error?: string | boolean\n} & Omit<\n  InputHTMLAttributes<HTMLInputElement>,\n  'size' | 'onChange' | 'value' | 'defaultValue'\n>\n\n/**\n * @deprecated This component is deprecated. Please use `NumberInputV2` instead.\n */\nexport const NumberInput = ({\n  disabled = false,\n  maxValue,\n  minValue = 0,\n  name = 'numberinput',\n  onChange,\n  onFocus,\n  onBlur,\n  onMaxCrossed,\n  onMinCrossed,\n  size = 'large',\n  step = 1,\n  text,\n  defaultValue,\n  value,\n  disabledTooltip,\n  className,\n  label,\n  id,\n  placeholder,\n  error,\n  'aria-label': ariaLabel,\n  'aria-describedby': ariaDescribedBy,\n  'data-testid': dataTestId,\n}: NumberInputProps) => {\n  const inputRef = useRef<HTMLInputElement>(null)\n\n  const uniqueId = useId()\n\n  // local state used if component is not controlled (no value prop provided)\n  const [inputValue, setInputValue] = useState<number | undefined>(() => {\n    if (defaultValue && minValue && defaultValue < minValue) {\n      return minValue\n    }\n    if (defaultValue && maxValue && defaultValue > maxValue) {\n      return maxValue\n    }\n\n    return defaultValue\n  })\n\n  const currentValue =\n    value !== undefined && value !== null ? value : inputValue\n\n  const setValue = (\n    newValue: number | undefined,\n    /**\n     * If true, will check if newValue is between minValue and maxValue and set it to minValue or maxValue if it's not.\n     */\n    hasMinMaxVerification = true,\n  ) => {\n    let nextValue = newValue\n    if (value === undefined && hasMinMaxVerification) {\n      if (newValue !== undefined && newValue < minValue) {\n        nextValue = minValue\n      }\n\n      if (\n        newValue !== undefined &&\n        maxValue !== undefined &&\n        newValue > maxValue\n      ) {\n        nextValue = maxValue\n      }\n    }\n    setInputValue(nextValue)\n    onChange?.(nextValue)\n  }\n\n  const offsetFn = (direction: number) => () => {\n    const localValue = currentValue ?? 0\n    const newValue =\n      localValue % step === 0 ? localValue + step * direction : localValue\n    const roundedValue = roundStep(newValue, step, direction)\n\n    setValue(roundedValue)\n  }\n\n  const handleChange: ChangeEventHandler<HTMLInputElement> = event => {\n    setValue(\n      event.currentTarget.value ? Number(event.currentTarget.value) : undefined,\n      false,\n    )\n  }\n\n  const handleOnBlur: FocusEventHandler<HTMLInputElement> = event => {\n    if (currentValue) {\n      const boundedValue = bounded(\n        currentValue,\n        minValue ?? currentValue,\n        maxValue ?? currentValue,\n      )\n\n      if (maxValue && currentValue > maxValue) onMaxCrossed?.()\n      if (minValue && currentValue < minValue) onMinCrossed?.()\n\n      setValue(boundedValue)\n\n      onBlur?.(event)\n    }\n  }\n\n  const onKeyDown: KeyboardEventHandler = event => {\n    if (event.key === 'ArrowUp') {\n      event.stopPropagation()\n      event.preventDefault()\n\n      const direction = 1\n      const localValue = currentValue ?? 0\n\n      const newValue =\n        localValue % step === 0 ? localValue + step * direction : localValue\n      const roundedValue = roundStep(newValue, step, direction)\n\n      if (maxValue === undefined) {\n        setValue(roundedValue)\n\n        return\n      }\n\n      setValue(Math.min(roundedValue, maxValue))\n    }\n\n    if (event.key === 'ArrowDown') {\n      event.stopPropagation()\n      event.preventDefault()\n\n      const direction = -1\n      const localValue = currentValue ?? 0\n\n      const newValue =\n        localValue % step === 0 ? localValue + step * direction : localValue\n      const roundedValue = roundStep(newValue, step, direction)\n\n      setValue(Math.max(roundedValue, minValue))\n    }\n  }\n\n  const isMinusDisabled = useMemo(() => {\n    if (disabled) return true\n    if (currentValue === undefined) return false\n    if (getMinusRoundedValue(currentValue, step) < minValue) {\n      return true\n    }\n\n    return disabled\n  }, [currentValue, disabled, minValue, step])\n\n  const isPlusDisabled = useMemo(() => {\n    if (disabled) return true\n    if (currentValue === undefined) return false\n    if (maxValue && getPlusRoundedValue(currentValue, step) > maxValue) {\n      return true\n    }\n\n    return disabled\n  }, [currentValue, disabled, maxValue, step])\n\n  const inputWidth = useMemo(() => {\n    if (placeholder && currentValue === undefined) {\n      return placeholder.length * 12\n    }\n\n    if (currentValue !== undefined) {\n      return currentValue.toString().length * 16\n    }\n\n    return BASE_INPUT_WIDTH\n  }, [currentValue, placeholder])\n\n  return (\n    <Stack gap={1}>\n      {label ? (\n        <Text variant=\"bodyStrong\" as=\"label\" htmlFor={id || uniqueId}>\n          {label}\n        </Text>\n      ) : null}\n      <Stack gap={0.5}>\n        <StyledContainer\n          aria-disabled={disabled}\n          data-error={!!error}\n          size={size}\n          className={className}\n          data-testid={dataTestId}\n        >\n          <Tooltip text={isMinusDisabled && disabledTooltip}>\n            <StyledSelectButton\n              onClick={offsetFn(-1)}\n              disabled={isMinusDisabled}\n              aria-label=\"Minus\"\n              type=\"button\"\n              variant=\"ghost\"\n              sentiment=\"primary\"\n              size=\"small\"\n            >\n              <MinusIcon\n                size={iconSizes[size]}\n                sentiment=\"primary\"\n                disabled={isMinusDisabled}\n              />\n            </StyledSelectButton>\n          </Tooltip>\n\n          <StyledCenterBox\n            size={size}\n            onClick={() => {\n              if (inputRef?.current) {\n                inputRef.current.focus()\n              }\n            }}\n            aria-live=\"assertive\"\n            role=\"status\"\n          >\n            <StyledInput\n              disabled={disabled}\n              name={name}\n              onBlur={handleOnBlur}\n              onChange={handleChange}\n              onFocus={onFocus}\n              onKeyDown={onKeyDown}\n              ref={inputRef}\n              style={{\n                width: inputWidth,\n              }}\n              value={currentValue !== undefined ? currentValue.toString() : ''} // A dom element can only have string attributes.\n              type=\"number\"\n              id={id || uniqueId}\n              aria-label={!label && !ariaLabel ? 'Number Input' : ariaLabel}\n              aria-describedby={ariaDescribedBy}\n              placeholder={placeholder}\n            />\n            {currentValue !== undefined ? (\n              <StyledText disabled={disabled}>{text}</StyledText>\n            ) : null}\n          </StyledCenterBox>\n\n          <Tooltip text={isPlusDisabled && disabledTooltip}>\n            <StyledSelectButton\n              onClick={offsetFn(1)}\n              disabled={isPlusDisabled}\n              aria-label=\"Plus\"\n              type=\"button\"\n              variant=\"ghost\"\n              sentiment=\"primary\"\n              size=\"small\"\n            >\n              <PlusIcon\n                size={iconSizes[size]}\n                sentiment=\"primary\"\n                disabled={isPlusDisabled}\n              />\n            </StyledSelectButton>\n          </Tooltip>\n        </StyledContainer>\n        {typeof error === 'string' ? (\n          <Text\n            as=\"span\"\n            variant=\"bodySmall\"\n            sentiment=\"danger\"\n            prominence=\"weak\"\n          >\n            {error}\n          </Text>\n        ) : null}\n      </Stack>\n    </Stack>\n  )\n}\n"]} */"));
29
29
  const StyledCenterBox = /* @__PURE__ */ _styled("div", process.env.NODE_ENV === "production" ? {
30
30
  shouldForwardProp: (prop) => !["size"].includes(prop),
31
31
  target: "exvap483"
@@ -37,7 +37,7 @@ const StyledCenterBox = /* @__PURE__ */ _styled("div", process.env.NODE_ENV ===
37
37
  size
38
38
  }) => size === "small" ? "24px" : "32px", ";align-items:center;outline:none;justify-content:center;border-radius:", ({
39
39
  theme
40
- }) => theme.radii.default, ";border:1px solid transparent;max-width:100%;" + (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/NumberInput/index.tsx"],"names":[],"mappings":"AA6CgC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/NumberInput/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { MinusIcon, PlusIcon } from '@ultraviolet/icons'\nimport type {\n  ChangeEventHandler,\n  FocusEventHandler,\n  InputHTMLAttributes,\n  KeyboardEventHandler,\n  MutableRefObject,\n} from 'react'\nimport { useId, useMemo, useRef, useState } from 'react'\nimport { Button } from '../Button'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\nimport { Tooltip } from '../Tooltip'\nimport {\n  bounded,\n  getMinusRoundedValue,\n  getPlusRoundedValue,\n  roundStep,\n} from './helpers'\n\nconst containerSizes = {\n  large: 48,\n  medium: 40,\n  small: 32,\n}\n\ntype ContainerSizesType = keyof typeof containerSizes\n\nconst iconSizes = {\n  large: 26,\n  medium: 24,\n  small: 22,\n}\n\nconst BASE_INPUT_WIDTH = 34\n\nconst StyledSelectButton = styled(Button)`\n  margin: 0 ${({ theme }) => theme.space['1']};\n  width: 32px;\n  height: 32px;\n`\n\nconst StyledCenterBox = styled('div', {\n  shouldForwardProp: prop => !['size'].includes(prop),\n})<{ size: ContainerSizesType }>`\n  display: flex;\n  flex: 1;\n  flex-direction: row;\n  height: ${({ size }) => (size === 'small' ? '24px' : '32px')};\n  align-items: center;\n  outline: none;\n  justify-content: center;\n  border-radius: ${({ theme }) => theme.radii.default};\n  border: 1px solid transparent;\n  max-width: 100%;\n`\n\nconst StyledInput = styled.input`\n  color: ${({ theme }) => theme.colors.neutral.text};\n  background-color: transparent;\n  font-size: ${({ theme }) => theme.typography.bodyStrong.fontSize};\n  border: none;\n  outline: none;\n  position: relative;\n  margin-right: ${({ theme }) => theme.space['0.5']};\n  max-width: 100%;\n  font-weight: ${({ theme }) => theme.typography.bodyStrong.weight};\n  text-align: center;\n\n  &::-webkit-outer-spin-button,\n  &::-webkit-inner-spin-button {\n    -webkit-appearance: none;\n    margin: 0;\n  }\n\n  ::placeholder {\n    color: ${({ theme }) => theme.colors.neutral.textWeak};\n  }\n\n  -moz-appearance: textfield;\n\n  &[disabled] {\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n    cursor: not-allowed;\n  }\n`\n\nconst StyledText = styled('span', {\n  shouldForwardProp: prop => !['disabled'].includes(prop),\n})<{ disabled: boolean }>`\n  color: ${({ theme, disabled }) =>\n    disabled ? theme.colors.neutral.textDisabled : theme.colors.neutral.text};\n  user-select: none;\n  margin-right: ${({ theme }) => theme.space['1']};\n`\n\nconst StyledContainer = styled('div', {\n  shouldForwardProp: prop => !['size'].includes(prop),\n})<{ size: ContainerSizesType }>`\n  background-color: ${({ theme }) => theme.colors.neutral.background};\n  display: flex;\n  flex-direction: row;\n  align-items: center;\n  align-self: stretch;\n  font-weight: 500;\n  height: ${({ size }) => containerSizes[size]}px;\n  border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  border-radius: ${({ theme }) => theme.radii.default};\n\n  &[data-error='true'] {\n    border: 1px solid ${({ theme }) => theme.colors.danger.border};\n  }\n\n  &[aria-disabled='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    cursor: not-allowed;\n  }\n\n  &:not([aria-disabled='true']) {\n    ${StyledCenterBox}:hover,\n    ${StyledCenterBox}:focus {\n      border: 1px solid ${({ theme }) => theme.colors.primary.borderHover};\n    }\n\n    ${StyledCenterBox}:focus-within {\n      box-shadow: ${({ theme }) => theme.shadows.focusPrimary};\n      border: 1px solid ${({ theme }) => theme.colors.primary.borderHover};\n    }\n  }\n`\n\ntype NumberInputProps = {\n  disabled?: boolean\n  maxValue?: number\n  minValue?: number\n  name?: string\n  onChange?: (input: number | undefined) => void\n  onMaxCrossed?: () => void\n  onMinCrossed?: () => void\n  size?: ContainerSizesType\n  /**\n   * Define how much will stepper increase / decrease each time you click on + / - button.\n   */\n  step?: number\n  /**\n   * Text displayed into component at the right of number value.\n   */\n  text?: string\n  defaultValue?: number\n  value?: number | null\n  disabledTooltip?: string\n  className?: string\n  'data-testid'?: string\n  label?: string\n  'aria-label'?: string\n  'aria-describedby'?: string\n  id?: string\n  placeholder?: string\n  error?: string | boolean\n} & Omit<\n  InputHTMLAttributes<HTMLInputElement>,\n  'size' | 'onChange' | 'value' | 'defaultValue'\n>\n\n/**\n * @deprecated This component is deprecated. Please use `NumberInputV2` instead.\n */\nexport const NumberInput = ({\n  disabled = false,\n  maxValue,\n  minValue = 0,\n  name = 'numberinput',\n  onChange,\n  onFocus,\n  onBlur,\n  onMaxCrossed,\n  onMinCrossed,\n  size = 'large',\n  step = 1,\n  text,\n  defaultValue,\n  value,\n  disabledTooltip,\n  className,\n  label,\n  id,\n  placeholder,\n  error,\n  'aria-label': ariaLabel,\n  'aria-describedby': ariaDescribedBy,\n  'data-testid': dataTestId,\n}: NumberInputProps) => {\n  const inputRef =\n    useRef<HTMLInputElement>() as MutableRefObject<HTMLInputElement>\n\n  const uniqueId = useId()\n\n  // local state used if component is not controlled (no value prop provided)\n  const [inputValue, setInputValue] = useState<number | undefined>(() => {\n    if (defaultValue && minValue && defaultValue < minValue) {\n      return minValue\n    }\n    if (defaultValue && maxValue && defaultValue > maxValue) {\n      return maxValue\n    }\n\n    return defaultValue\n  })\n\n  const currentValue =\n    value !== undefined && value !== null ? value : inputValue\n\n  const setValue = (\n    newValue: number | undefined,\n    /**\n     * If true, will check if newValue is between minValue and maxValue and set it to minValue or maxValue if it's not.\n     */\n    hasMinMaxVerification = true,\n  ) => {\n    let nextValue = newValue\n    if (value === undefined && hasMinMaxVerification) {\n      if (newValue !== undefined && newValue < minValue) {\n        nextValue = minValue\n      }\n\n      if (\n        newValue !== undefined &&\n        maxValue !== undefined &&\n        newValue > maxValue\n      ) {\n        nextValue = maxValue\n      }\n    }\n    setInputValue(nextValue)\n    onChange?.(nextValue)\n  }\n\n  const offsetFn = (direction: number) => () => {\n    const localValue = currentValue ?? 0\n    const newValue =\n      localValue % step === 0 ? localValue + step * direction : localValue\n    const roundedValue = roundStep(newValue, step, direction)\n\n    setValue(roundedValue)\n  }\n\n  const handleChange: ChangeEventHandler<HTMLInputElement> = event => {\n    setValue(\n      event.currentTarget.value ? Number(event.currentTarget.value) : undefined,\n      false,\n    )\n  }\n\n  const handleOnBlur: FocusEventHandler<HTMLInputElement> = event => {\n    if (currentValue) {\n      const boundedValue = bounded(\n        currentValue,\n        minValue ?? currentValue,\n        maxValue ?? currentValue,\n      )\n\n      if (maxValue && currentValue > maxValue) onMaxCrossed?.()\n      if (minValue && currentValue < minValue) onMinCrossed?.()\n\n      setValue(boundedValue)\n\n      onBlur?.(event)\n    }\n  }\n\n  const onKeyDown: KeyboardEventHandler = event => {\n    if (event.key === 'ArrowUp') {\n      event.stopPropagation()\n      event.preventDefault()\n\n      const direction = 1\n      const localValue = currentValue ?? 0\n\n      const newValue =\n        localValue % step === 0 ? localValue + step * direction : localValue\n      const roundedValue = roundStep(newValue, step, direction)\n\n      if (maxValue === undefined) {\n        setValue(roundedValue)\n\n        return\n      }\n\n      setValue(Math.min(roundedValue, maxValue))\n    }\n\n    if (event.key === 'ArrowDown') {\n      event.stopPropagation()\n      event.preventDefault()\n\n      const direction = -1\n      const localValue = currentValue ?? 0\n\n      const newValue =\n        localValue % step === 0 ? localValue + step * direction : localValue\n      const roundedValue = roundStep(newValue, step, direction)\n\n      setValue(Math.max(roundedValue, minValue))\n    }\n  }\n\n  const isMinusDisabled = useMemo(() => {\n    if (disabled) return true\n    if (currentValue === undefined) return false\n    if (getMinusRoundedValue(currentValue, step) < minValue) {\n      return true\n    }\n\n    return disabled\n  }, [currentValue, disabled, minValue, step])\n\n  const isPlusDisabled = useMemo(() => {\n    if (disabled) return true\n    if (currentValue === undefined) return false\n    if (maxValue && getPlusRoundedValue(currentValue, step) > maxValue) {\n      return true\n    }\n\n    return disabled\n  }, [currentValue, disabled, maxValue, step])\n\n  const inputWidth = useMemo(() => {\n    if (placeholder && currentValue === undefined) {\n      return placeholder.length * 12\n    }\n\n    if (currentValue !== undefined) {\n      return currentValue.toString().length * 16\n    }\n\n    return BASE_INPUT_WIDTH\n  }, [currentValue, placeholder])\n\n  return (\n    <Stack gap={1}>\n      {label ? (\n        <Text variant=\"bodyStrong\" as=\"label\" htmlFor={id || uniqueId}>\n          {label}\n        </Text>\n      ) : null}\n      <Stack gap={0.5}>\n        <StyledContainer\n          aria-disabled={disabled}\n          data-error={!!error}\n          size={size}\n          className={className}\n          data-testid={dataTestId}\n        >\n          <Tooltip text={isMinusDisabled && disabledTooltip}>\n            <StyledSelectButton\n              onClick={offsetFn(-1)}\n              disabled={isMinusDisabled}\n              aria-label=\"Minus\"\n              type=\"button\"\n              variant=\"ghost\"\n              sentiment=\"primary\"\n              size=\"small\"\n            >\n              <MinusIcon\n                size={iconSizes[size]}\n                sentiment=\"primary\"\n                disabled={isMinusDisabled}\n              />\n            </StyledSelectButton>\n          </Tooltip>\n\n          <StyledCenterBox\n            size={size}\n            onClick={() => {\n              if (inputRef?.current) {\n                inputRef.current.focus()\n              }\n            }}\n            aria-live=\"assertive\"\n            role=\"status\"\n          >\n            <StyledInput\n              disabled={disabled}\n              name={name}\n              onBlur={handleOnBlur}\n              onChange={handleChange}\n              onFocus={onFocus}\n              onKeyDown={onKeyDown}\n              ref={inputRef}\n              style={{\n                width: inputWidth,\n              }}\n              value={currentValue !== undefined ? currentValue.toString() : ''} // A dom element can only have string attributes.\n              type=\"number\"\n              id={id || uniqueId}\n              aria-label={!label && !ariaLabel ? 'Number Input' : ariaLabel}\n              aria-describedby={ariaDescribedBy}\n              placeholder={placeholder}\n            />\n            {currentValue !== undefined ? (\n              <StyledText disabled={disabled}>{text}</StyledText>\n            ) : null}\n          </StyledCenterBox>\n\n          <Tooltip text={isPlusDisabled && disabledTooltip}>\n            <StyledSelectButton\n              onClick={offsetFn(1)}\n              disabled={isPlusDisabled}\n              aria-label=\"Plus\"\n              type=\"button\"\n              variant=\"ghost\"\n              sentiment=\"primary\"\n              size=\"small\"\n            >\n              <PlusIcon\n                size={iconSizes[size]}\n                sentiment=\"primary\"\n                disabled={isPlusDisabled}\n              />\n            </StyledSelectButton>\n          </Tooltip>\n        </StyledContainer>\n        {typeof error === 'string' ? (\n          <Text\n            as=\"span\"\n            variant=\"bodySmall\"\n            sentiment=\"danger\"\n            prominence=\"weak\"\n          >\n            {error}\n          </Text>\n        ) : null}\n      </Stack>\n    </Stack>\n  )\n}\n"]} */"));
40
+ }) => theme.radii.default, ";border:1px solid transparent;max-width:100%;" + (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/NumberInput/index.tsx"],"names":[],"mappings":"AA4CgC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/NumberInput/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { MinusIcon, PlusIcon } from '@ultraviolet/icons'\nimport type {\n  ChangeEventHandler,\n  FocusEventHandler,\n  InputHTMLAttributes,\n  KeyboardEventHandler,\n} from 'react'\nimport { useId, useMemo, useRef, useState } from 'react'\nimport { Button } from '../Button'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\nimport { Tooltip } from '../Tooltip'\nimport {\n  bounded,\n  getMinusRoundedValue,\n  getPlusRoundedValue,\n  roundStep,\n} from './helpers'\n\nconst containerSizes = {\n  large: 48,\n  medium: 40,\n  small: 32,\n}\n\ntype ContainerSizesType = keyof typeof containerSizes\n\nconst iconSizes = {\n  large: 26,\n  medium: 24,\n  small: 22,\n}\n\nconst BASE_INPUT_WIDTH = 34\n\nconst StyledSelectButton = styled(Button)`\n  margin: 0 ${({ theme }) => theme.space['1']};\n  width: 32px;\n  height: 32px;\n`\n\nconst StyledCenterBox = styled('div', {\n  shouldForwardProp: prop => !['size'].includes(prop),\n})<{ size: ContainerSizesType }>`\n  display: flex;\n  flex: 1;\n  flex-direction: row;\n  height: ${({ size }) => (size === 'small' ? '24px' : '32px')};\n  align-items: center;\n  outline: none;\n  justify-content: center;\n  border-radius: ${({ theme }) => theme.radii.default};\n  border: 1px solid transparent;\n  max-width: 100%;\n`\n\nconst StyledInput = styled.input`\n  color: ${({ theme }) => theme.colors.neutral.text};\n  background-color: transparent;\n  font-size: ${({ theme }) => theme.typography.bodyStrong.fontSize};\n  border: none;\n  outline: none;\n  position: relative;\n  margin-right: ${({ theme }) => theme.space['0.5']};\n  max-width: 100%;\n  font-weight: ${({ theme }) => theme.typography.bodyStrong.weight};\n  text-align: center;\n\n  &::-webkit-outer-spin-button,\n  &::-webkit-inner-spin-button {\n    -webkit-appearance: none;\n    margin: 0;\n  }\n\n  ::placeholder {\n    color: ${({ theme }) => theme.colors.neutral.textWeak};\n  }\n\n  -moz-appearance: textfield;\n\n  &[disabled] {\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n    cursor: not-allowed;\n  }\n`\n\nconst StyledText = styled('span', {\n  shouldForwardProp: prop => !['disabled'].includes(prop),\n})<{ disabled: boolean }>`\n  color: ${({ theme, disabled }) =>\n    disabled ? theme.colors.neutral.textDisabled : theme.colors.neutral.text};\n  user-select: none;\n  margin-right: ${({ theme }) => theme.space['1']};\n`\n\nconst StyledContainer = styled('div', {\n  shouldForwardProp: prop => !['size'].includes(prop),\n})<{ size: ContainerSizesType }>`\n  background-color: ${({ theme }) => theme.colors.neutral.background};\n  display: flex;\n  flex-direction: row;\n  align-items: center;\n  align-self: stretch;\n  font-weight: 500;\n  height: ${({ size }) => containerSizes[size]}px;\n  border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  border-radius: ${({ theme }) => theme.radii.default};\n\n  &[data-error='true'] {\n    border: 1px solid ${({ theme }) => theme.colors.danger.border};\n  }\n\n  &[aria-disabled='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    cursor: not-allowed;\n  }\n\n  &:not([aria-disabled='true']) {\n    ${StyledCenterBox}:hover,\n    ${StyledCenterBox}:focus {\n      border: 1px solid ${({ theme }) => theme.colors.primary.borderHover};\n    }\n\n    ${StyledCenterBox}:focus-within {\n      box-shadow: ${({ theme }) => theme.shadows.focusPrimary};\n      border: 1px solid ${({ theme }) => theme.colors.primary.borderHover};\n    }\n  }\n`\n\ntype NumberInputProps = {\n  disabled?: boolean\n  maxValue?: number\n  minValue?: number\n  name?: string\n  onChange?: (input: number | undefined) => void\n  onMaxCrossed?: () => void\n  onMinCrossed?: () => void\n  size?: ContainerSizesType\n  /**\n   * Define how much will stepper increase / decrease each time you click on + / - button.\n   */\n  step?: number\n  /**\n   * Text displayed into component at the right of number value.\n   */\n  text?: string\n  defaultValue?: number\n  value?: number | null\n  disabledTooltip?: string\n  className?: string\n  'data-testid'?: string\n  label?: string\n  'aria-label'?: string\n  'aria-describedby'?: string\n  id?: string\n  placeholder?: string\n  error?: string | boolean\n} & Omit<\n  InputHTMLAttributes<HTMLInputElement>,\n  'size' | 'onChange' | 'value' | 'defaultValue'\n>\n\n/**\n * @deprecated This component is deprecated. Please use `NumberInputV2` instead.\n */\nexport const NumberInput = ({\n  disabled = false,\n  maxValue,\n  minValue = 0,\n  name = 'numberinput',\n  onChange,\n  onFocus,\n  onBlur,\n  onMaxCrossed,\n  onMinCrossed,\n  size = 'large',\n  step = 1,\n  text,\n  defaultValue,\n  value,\n  disabledTooltip,\n  className,\n  label,\n  id,\n  placeholder,\n  error,\n  'aria-label': ariaLabel,\n  'aria-describedby': ariaDescribedBy,\n  'data-testid': dataTestId,\n}: NumberInputProps) => {\n  const inputRef = useRef<HTMLInputElement>(null)\n\n  const uniqueId = useId()\n\n  // local state used if component is not controlled (no value prop provided)\n  const [inputValue, setInputValue] = useState<number | undefined>(() => {\n    if (defaultValue && minValue && defaultValue < minValue) {\n      return minValue\n    }\n    if (defaultValue && maxValue && defaultValue > maxValue) {\n      return maxValue\n    }\n\n    return defaultValue\n  })\n\n  const currentValue =\n    value !== undefined && value !== null ? value : inputValue\n\n  const setValue = (\n    newValue: number | undefined,\n    /**\n     * If true, will check if newValue is between minValue and maxValue and set it to minValue or maxValue if it's not.\n     */\n    hasMinMaxVerification = true,\n  ) => {\n    let nextValue = newValue\n    if (value === undefined && hasMinMaxVerification) {\n      if (newValue !== undefined && newValue < minValue) {\n        nextValue = minValue\n      }\n\n      if (\n        newValue !== undefined &&\n        maxValue !== undefined &&\n        newValue > maxValue\n      ) {\n        nextValue = maxValue\n      }\n    }\n    setInputValue(nextValue)\n    onChange?.(nextValue)\n  }\n\n  const offsetFn = (direction: number) => () => {\n    const localValue = currentValue ?? 0\n    const newValue =\n      localValue % step === 0 ? localValue + step * direction : localValue\n    const roundedValue = roundStep(newValue, step, direction)\n\n    setValue(roundedValue)\n  }\n\n  const handleChange: ChangeEventHandler<HTMLInputElement> = event => {\n    setValue(\n      event.currentTarget.value ? Number(event.currentTarget.value) : undefined,\n      false,\n    )\n  }\n\n  const handleOnBlur: FocusEventHandler<HTMLInputElement> = event => {\n    if (currentValue) {\n      const boundedValue = bounded(\n        currentValue,\n        minValue ?? currentValue,\n        maxValue ?? currentValue,\n      )\n\n      if (maxValue && currentValue > maxValue) onMaxCrossed?.()\n      if (minValue && currentValue < minValue) onMinCrossed?.()\n\n      setValue(boundedValue)\n\n      onBlur?.(event)\n    }\n  }\n\n  const onKeyDown: KeyboardEventHandler = event => {\n    if (event.key === 'ArrowUp') {\n      event.stopPropagation()\n      event.preventDefault()\n\n      const direction = 1\n      const localValue = currentValue ?? 0\n\n      const newValue =\n        localValue % step === 0 ? localValue + step * direction : localValue\n      const roundedValue = roundStep(newValue, step, direction)\n\n      if (maxValue === undefined) {\n        setValue(roundedValue)\n\n        return\n      }\n\n      setValue(Math.min(roundedValue, maxValue))\n    }\n\n    if (event.key === 'ArrowDown') {\n      event.stopPropagation()\n      event.preventDefault()\n\n      const direction = -1\n      const localValue = currentValue ?? 0\n\n      const newValue =\n        localValue % step === 0 ? localValue + step * direction : localValue\n      const roundedValue = roundStep(newValue, step, direction)\n\n      setValue(Math.max(roundedValue, minValue))\n    }\n  }\n\n  const isMinusDisabled = useMemo(() => {\n    if (disabled) return true\n    if (currentValue === undefined) return false\n    if (getMinusRoundedValue(currentValue, step) < minValue) {\n      return true\n    }\n\n    return disabled\n  }, [currentValue, disabled, minValue, step])\n\n  const isPlusDisabled = useMemo(() => {\n    if (disabled) return true\n    if (currentValue === undefined) return false\n    if (maxValue && getPlusRoundedValue(currentValue, step) > maxValue) {\n      return true\n    }\n\n    return disabled\n  }, [currentValue, disabled, maxValue, step])\n\n  const inputWidth = useMemo(() => {\n    if (placeholder && currentValue === undefined) {\n      return placeholder.length * 12\n    }\n\n    if (currentValue !== undefined) {\n      return currentValue.toString().length * 16\n    }\n\n    return BASE_INPUT_WIDTH\n  }, [currentValue, placeholder])\n\n  return (\n    <Stack gap={1}>\n      {label ? (\n        <Text variant=\"bodyStrong\" as=\"label\" htmlFor={id || uniqueId}>\n          {label}\n        </Text>\n      ) : null}\n      <Stack gap={0.5}>\n        <StyledContainer\n          aria-disabled={disabled}\n          data-error={!!error}\n          size={size}\n          className={className}\n          data-testid={dataTestId}\n        >\n          <Tooltip text={isMinusDisabled && disabledTooltip}>\n            <StyledSelectButton\n              onClick={offsetFn(-1)}\n              disabled={isMinusDisabled}\n              aria-label=\"Minus\"\n              type=\"button\"\n              variant=\"ghost\"\n              sentiment=\"primary\"\n              size=\"small\"\n            >\n              <MinusIcon\n                size={iconSizes[size]}\n                sentiment=\"primary\"\n                disabled={isMinusDisabled}\n              />\n            </StyledSelectButton>\n          </Tooltip>\n\n          <StyledCenterBox\n            size={size}\n            onClick={() => {\n              if (inputRef?.current) {\n                inputRef.current.focus()\n              }\n            }}\n            aria-live=\"assertive\"\n            role=\"status\"\n          >\n            <StyledInput\n              disabled={disabled}\n              name={name}\n              onBlur={handleOnBlur}\n              onChange={handleChange}\n              onFocus={onFocus}\n              onKeyDown={onKeyDown}\n              ref={inputRef}\n              style={{\n                width: inputWidth,\n              }}\n              value={currentValue !== undefined ? currentValue.toString() : ''} // A dom element can only have string attributes.\n              type=\"number\"\n              id={id || uniqueId}\n              aria-label={!label && !ariaLabel ? 'Number Input' : ariaLabel}\n              aria-describedby={ariaDescribedBy}\n              placeholder={placeholder}\n            />\n            {currentValue !== undefined ? (\n              <StyledText disabled={disabled}>{text}</StyledText>\n            ) : null}\n          </StyledCenterBox>\n\n          <Tooltip text={isPlusDisabled && disabledTooltip}>\n            <StyledSelectButton\n              onClick={offsetFn(1)}\n              disabled={isPlusDisabled}\n              aria-label=\"Plus\"\n              type=\"button\"\n              variant=\"ghost\"\n              sentiment=\"primary\"\n              size=\"small\"\n            >\n              <PlusIcon\n                size={iconSizes[size]}\n                sentiment=\"primary\"\n                disabled={isPlusDisabled}\n              />\n            </StyledSelectButton>\n          </Tooltip>\n        </StyledContainer>\n        {typeof error === 'string' ? (\n          <Text\n            as=\"span\"\n            variant=\"bodySmall\"\n            sentiment=\"danger\"\n            prominence=\"weak\"\n          >\n            {error}\n          </Text>\n        ) : null}\n      </Stack>\n    </Stack>\n  )\n}\n"]} */"));
41
41
  const StyledInput = /* @__PURE__ */ _styled("input", process.env.NODE_ENV === "production" ? {
42
42
  target: "exvap482"
43
43
  } : {
@@ -55,7 +55,7 @@ const StyledInput = /* @__PURE__ */ _styled("input", process.env.NODE_ENV === "p
55
55
  theme
56
56
  }) => theme.colors.neutral.textWeak, ";}-moz-appearance:textfield;&[disabled]{color:", ({
57
57
  theme
58
- }) => theme.colors.neutral.textDisabled, ";cursor:not-allowed;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/NumberInput/index.tsx"],"names":[],"mappings":"AA0DgC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/NumberInput/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { MinusIcon, PlusIcon } from '@ultraviolet/icons'\nimport type {\n  ChangeEventHandler,\n  FocusEventHandler,\n  InputHTMLAttributes,\n  KeyboardEventHandler,\n  MutableRefObject,\n} from 'react'\nimport { useId, useMemo, useRef, useState } from 'react'\nimport { Button } from '../Button'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\nimport { Tooltip } from '../Tooltip'\nimport {\n  bounded,\n  getMinusRoundedValue,\n  getPlusRoundedValue,\n  roundStep,\n} from './helpers'\n\nconst containerSizes = {\n  large: 48,\n  medium: 40,\n  small: 32,\n}\n\ntype ContainerSizesType = keyof typeof containerSizes\n\nconst iconSizes = {\n  large: 26,\n  medium: 24,\n  small: 22,\n}\n\nconst BASE_INPUT_WIDTH = 34\n\nconst StyledSelectButton = styled(Button)`\n  margin: 0 ${({ theme }) => theme.space['1']};\n  width: 32px;\n  height: 32px;\n`\n\nconst StyledCenterBox = styled('div', {\n  shouldForwardProp: prop => !['size'].includes(prop),\n})<{ size: ContainerSizesType }>`\n  display: flex;\n  flex: 1;\n  flex-direction: row;\n  height: ${({ size }) => (size === 'small' ? '24px' : '32px')};\n  align-items: center;\n  outline: none;\n  justify-content: center;\n  border-radius: ${({ theme }) => theme.radii.default};\n  border: 1px solid transparent;\n  max-width: 100%;\n`\n\nconst StyledInput = styled.input`\n  color: ${({ theme }) => theme.colors.neutral.text};\n  background-color: transparent;\n  font-size: ${({ theme }) => theme.typography.bodyStrong.fontSize};\n  border: none;\n  outline: none;\n  position: relative;\n  margin-right: ${({ theme }) => theme.space['0.5']};\n  max-width: 100%;\n  font-weight: ${({ theme }) => theme.typography.bodyStrong.weight};\n  text-align: center;\n\n  &::-webkit-outer-spin-button,\n  &::-webkit-inner-spin-button {\n    -webkit-appearance: none;\n    margin: 0;\n  }\n\n  ::placeholder {\n    color: ${({ theme }) => theme.colors.neutral.textWeak};\n  }\n\n  -moz-appearance: textfield;\n\n  &[disabled] {\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n    cursor: not-allowed;\n  }\n`\n\nconst StyledText = styled('span', {\n  shouldForwardProp: prop => !['disabled'].includes(prop),\n})<{ disabled: boolean }>`\n  color: ${({ theme, disabled }) =>\n    disabled ? theme.colors.neutral.textDisabled : theme.colors.neutral.text};\n  user-select: none;\n  margin-right: ${({ theme }) => theme.space['1']};\n`\n\nconst StyledContainer = styled('div', {\n  shouldForwardProp: prop => !['size'].includes(prop),\n})<{ size: ContainerSizesType }>`\n  background-color: ${({ theme }) => theme.colors.neutral.background};\n  display: flex;\n  flex-direction: row;\n  align-items: center;\n  align-self: stretch;\n  font-weight: 500;\n  height: ${({ size }) => containerSizes[size]}px;\n  border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  border-radius: ${({ theme }) => theme.radii.default};\n\n  &[data-error='true'] {\n    border: 1px solid ${({ theme }) => theme.colors.danger.border};\n  }\n\n  &[aria-disabled='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    cursor: not-allowed;\n  }\n\n  &:not([aria-disabled='true']) {\n    ${StyledCenterBox}:hover,\n    ${StyledCenterBox}:focus {\n      border: 1px solid ${({ theme }) => theme.colors.primary.borderHover};\n    }\n\n    ${StyledCenterBox}:focus-within {\n      box-shadow: ${({ theme }) => theme.shadows.focusPrimary};\n      border: 1px solid ${({ theme }) => theme.colors.primary.borderHover};\n    }\n  }\n`\n\ntype NumberInputProps = {\n  disabled?: boolean\n  maxValue?: number\n  minValue?: number\n  name?: string\n  onChange?: (input: number | undefined) => void\n  onMaxCrossed?: () => void\n  onMinCrossed?: () => void\n  size?: ContainerSizesType\n  /**\n   * Define how much will stepper increase / decrease each time you click on + / - button.\n   */\n  step?: number\n  /**\n   * Text displayed into component at the right of number value.\n   */\n  text?: string\n  defaultValue?: number\n  value?: number | null\n  disabledTooltip?: string\n  className?: string\n  'data-testid'?: string\n  label?: string\n  'aria-label'?: string\n  'aria-describedby'?: string\n  id?: string\n  placeholder?: string\n  error?: string | boolean\n} & Omit<\n  InputHTMLAttributes<HTMLInputElement>,\n  'size' | 'onChange' | 'value' | 'defaultValue'\n>\n\n/**\n * @deprecated This component is deprecated. Please use `NumberInputV2` instead.\n */\nexport const NumberInput = ({\n  disabled = false,\n  maxValue,\n  minValue = 0,\n  name = 'numberinput',\n  onChange,\n  onFocus,\n  onBlur,\n  onMaxCrossed,\n  onMinCrossed,\n  size = 'large',\n  step = 1,\n  text,\n  defaultValue,\n  value,\n  disabledTooltip,\n  className,\n  label,\n  id,\n  placeholder,\n  error,\n  'aria-label': ariaLabel,\n  'aria-describedby': ariaDescribedBy,\n  'data-testid': dataTestId,\n}: NumberInputProps) => {\n  const inputRef =\n    useRef<HTMLInputElement>() as MutableRefObject<HTMLInputElement>\n\n  const uniqueId = useId()\n\n  // local state used if component is not controlled (no value prop provided)\n  const [inputValue, setInputValue] = useState<number | undefined>(() => {\n    if (defaultValue && minValue && defaultValue < minValue) {\n      return minValue\n    }\n    if (defaultValue && maxValue && defaultValue > maxValue) {\n      return maxValue\n    }\n\n    return defaultValue\n  })\n\n  const currentValue =\n    value !== undefined && value !== null ? value : inputValue\n\n  const setValue = (\n    newValue: number | undefined,\n    /**\n     * If true, will check if newValue is between minValue and maxValue and set it to minValue or maxValue if it's not.\n     */\n    hasMinMaxVerification = true,\n  ) => {\n    let nextValue = newValue\n    if (value === undefined && hasMinMaxVerification) {\n      if (newValue !== undefined && newValue < minValue) {\n        nextValue = minValue\n      }\n\n      if (\n        newValue !== undefined &&\n        maxValue !== undefined &&\n        newValue > maxValue\n      ) {\n        nextValue = maxValue\n      }\n    }\n    setInputValue(nextValue)\n    onChange?.(nextValue)\n  }\n\n  const offsetFn = (direction: number) => () => {\n    const localValue = currentValue ?? 0\n    const newValue =\n      localValue % step === 0 ? localValue + step * direction : localValue\n    const roundedValue = roundStep(newValue, step, direction)\n\n    setValue(roundedValue)\n  }\n\n  const handleChange: ChangeEventHandler<HTMLInputElement> = event => {\n    setValue(\n      event.currentTarget.value ? Number(event.currentTarget.value) : undefined,\n      false,\n    )\n  }\n\n  const handleOnBlur: FocusEventHandler<HTMLInputElement> = event => {\n    if (currentValue) {\n      const boundedValue = bounded(\n        currentValue,\n        minValue ?? currentValue,\n        maxValue ?? currentValue,\n      )\n\n      if (maxValue && currentValue > maxValue) onMaxCrossed?.()\n      if (minValue && currentValue < minValue) onMinCrossed?.()\n\n      setValue(boundedValue)\n\n      onBlur?.(event)\n    }\n  }\n\n  const onKeyDown: KeyboardEventHandler = event => {\n    if (event.key === 'ArrowUp') {\n      event.stopPropagation()\n      event.preventDefault()\n\n      const direction = 1\n      const localValue = currentValue ?? 0\n\n      const newValue =\n        localValue % step === 0 ? localValue + step * direction : localValue\n      const roundedValue = roundStep(newValue, step, direction)\n\n      if (maxValue === undefined) {\n        setValue(roundedValue)\n\n        return\n      }\n\n      setValue(Math.min(roundedValue, maxValue))\n    }\n\n    if (event.key === 'ArrowDown') {\n      event.stopPropagation()\n      event.preventDefault()\n\n      const direction = -1\n      const localValue = currentValue ?? 0\n\n      const newValue =\n        localValue % step === 0 ? localValue + step * direction : localValue\n      const roundedValue = roundStep(newValue, step, direction)\n\n      setValue(Math.max(roundedValue, minValue))\n    }\n  }\n\n  const isMinusDisabled = useMemo(() => {\n    if (disabled) return true\n    if (currentValue === undefined) return false\n    if (getMinusRoundedValue(currentValue, step) < minValue) {\n      return true\n    }\n\n    return disabled\n  }, [currentValue, disabled, minValue, step])\n\n  const isPlusDisabled = useMemo(() => {\n    if (disabled) return true\n    if (currentValue === undefined) return false\n    if (maxValue && getPlusRoundedValue(currentValue, step) > maxValue) {\n      return true\n    }\n\n    return disabled\n  }, [currentValue, disabled, maxValue, step])\n\n  const inputWidth = useMemo(() => {\n    if (placeholder && currentValue === undefined) {\n      return placeholder.length * 12\n    }\n\n    if (currentValue !== undefined) {\n      return currentValue.toString().length * 16\n    }\n\n    return BASE_INPUT_WIDTH\n  }, [currentValue, placeholder])\n\n  return (\n    <Stack gap={1}>\n      {label ? (\n        <Text variant=\"bodyStrong\" as=\"label\" htmlFor={id || uniqueId}>\n          {label}\n        </Text>\n      ) : null}\n      <Stack gap={0.5}>\n        <StyledContainer\n          aria-disabled={disabled}\n          data-error={!!error}\n          size={size}\n          className={className}\n          data-testid={dataTestId}\n        >\n          <Tooltip text={isMinusDisabled && disabledTooltip}>\n            <StyledSelectButton\n              onClick={offsetFn(-1)}\n              disabled={isMinusDisabled}\n              aria-label=\"Minus\"\n              type=\"button\"\n              variant=\"ghost\"\n              sentiment=\"primary\"\n              size=\"small\"\n            >\n              <MinusIcon\n                size={iconSizes[size]}\n                sentiment=\"primary\"\n                disabled={isMinusDisabled}\n              />\n            </StyledSelectButton>\n          </Tooltip>\n\n          <StyledCenterBox\n            size={size}\n            onClick={() => {\n              if (inputRef?.current) {\n                inputRef.current.focus()\n              }\n            }}\n            aria-live=\"assertive\"\n            role=\"status\"\n          >\n            <StyledInput\n              disabled={disabled}\n              name={name}\n              onBlur={handleOnBlur}\n              onChange={handleChange}\n              onFocus={onFocus}\n              onKeyDown={onKeyDown}\n              ref={inputRef}\n              style={{\n                width: inputWidth,\n              }}\n              value={currentValue !== undefined ? currentValue.toString() : ''} // A dom element can only have string attributes.\n              type=\"number\"\n              id={id || uniqueId}\n              aria-label={!label && !ariaLabel ? 'Number Input' : ariaLabel}\n              aria-describedby={ariaDescribedBy}\n              placeholder={placeholder}\n            />\n            {currentValue !== undefined ? (\n              <StyledText disabled={disabled}>{text}</StyledText>\n            ) : null}\n          </StyledCenterBox>\n\n          <Tooltip text={isPlusDisabled && disabledTooltip}>\n            <StyledSelectButton\n              onClick={offsetFn(1)}\n              disabled={isPlusDisabled}\n              aria-label=\"Plus\"\n              type=\"button\"\n              variant=\"ghost\"\n              sentiment=\"primary\"\n              size=\"small\"\n            >\n              <PlusIcon\n                size={iconSizes[size]}\n                sentiment=\"primary\"\n                disabled={isPlusDisabled}\n              />\n            </StyledSelectButton>\n          </Tooltip>\n        </StyledContainer>\n        {typeof error === 'string' ? (\n          <Text\n            as=\"span\"\n            variant=\"bodySmall\"\n            sentiment=\"danger\"\n            prominence=\"weak\"\n          >\n            {error}\n          </Text>\n        ) : null}\n      </Stack>\n    </Stack>\n  )\n}\n"]} */"));
58
+ }) => theme.colors.neutral.textDisabled, ";cursor:not-allowed;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/NumberInput/index.tsx"],"names":[],"mappings":"AAyDgC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/NumberInput/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { MinusIcon, PlusIcon } from '@ultraviolet/icons'\nimport type {\n  ChangeEventHandler,\n  FocusEventHandler,\n  InputHTMLAttributes,\n  KeyboardEventHandler,\n} from 'react'\nimport { useId, useMemo, useRef, useState } from 'react'\nimport { Button } from '../Button'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\nimport { Tooltip } from '../Tooltip'\nimport {\n  bounded,\n  getMinusRoundedValue,\n  getPlusRoundedValue,\n  roundStep,\n} from './helpers'\n\nconst containerSizes = {\n  large: 48,\n  medium: 40,\n  small: 32,\n}\n\ntype ContainerSizesType = keyof typeof containerSizes\n\nconst iconSizes = {\n  large: 26,\n  medium: 24,\n  small: 22,\n}\n\nconst BASE_INPUT_WIDTH = 34\n\nconst StyledSelectButton = styled(Button)`\n  margin: 0 ${({ theme }) => theme.space['1']};\n  width: 32px;\n  height: 32px;\n`\n\nconst StyledCenterBox = styled('div', {\n  shouldForwardProp: prop => !['size'].includes(prop),\n})<{ size: ContainerSizesType }>`\n  display: flex;\n  flex: 1;\n  flex-direction: row;\n  height: ${({ size }) => (size === 'small' ? '24px' : '32px')};\n  align-items: center;\n  outline: none;\n  justify-content: center;\n  border-radius: ${({ theme }) => theme.radii.default};\n  border: 1px solid transparent;\n  max-width: 100%;\n`\n\nconst StyledInput = styled.input`\n  color: ${({ theme }) => theme.colors.neutral.text};\n  background-color: transparent;\n  font-size: ${({ theme }) => theme.typography.bodyStrong.fontSize};\n  border: none;\n  outline: none;\n  position: relative;\n  margin-right: ${({ theme }) => theme.space['0.5']};\n  max-width: 100%;\n  font-weight: ${({ theme }) => theme.typography.bodyStrong.weight};\n  text-align: center;\n\n  &::-webkit-outer-spin-button,\n  &::-webkit-inner-spin-button {\n    -webkit-appearance: none;\n    margin: 0;\n  }\n\n  ::placeholder {\n    color: ${({ theme }) => theme.colors.neutral.textWeak};\n  }\n\n  -moz-appearance: textfield;\n\n  &[disabled] {\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n    cursor: not-allowed;\n  }\n`\n\nconst StyledText = styled('span', {\n  shouldForwardProp: prop => !['disabled'].includes(prop),\n})<{ disabled: boolean }>`\n  color: ${({ theme, disabled }) =>\n    disabled ? theme.colors.neutral.textDisabled : theme.colors.neutral.text};\n  user-select: none;\n  margin-right: ${({ theme }) => theme.space['1']};\n`\n\nconst StyledContainer = styled('div', {\n  shouldForwardProp: prop => !['size'].includes(prop),\n})<{ size: ContainerSizesType }>`\n  background-color: ${({ theme }) => theme.colors.neutral.background};\n  display: flex;\n  flex-direction: row;\n  align-items: center;\n  align-self: stretch;\n  font-weight: 500;\n  height: ${({ size }) => containerSizes[size]}px;\n  border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  border-radius: ${({ theme }) => theme.radii.default};\n\n  &[data-error='true'] {\n    border: 1px solid ${({ theme }) => theme.colors.danger.border};\n  }\n\n  &[aria-disabled='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    cursor: not-allowed;\n  }\n\n  &:not([aria-disabled='true']) {\n    ${StyledCenterBox}:hover,\n    ${StyledCenterBox}:focus {\n      border: 1px solid ${({ theme }) => theme.colors.primary.borderHover};\n    }\n\n    ${StyledCenterBox}:focus-within {\n      box-shadow: ${({ theme }) => theme.shadows.focusPrimary};\n      border: 1px solid ${({ theme }) => theme.colors.primary.borderHover};\n    }\n  }\n`\n\ntype NumberInputProps = {\n  disabled?: boolean\n  maxValue?: number\n  minValue?: number\n  name?: string\n  onChange?: (input: number | undefined) => void\n  onMaxCrossed?: () => void\n  onMinCrossed?: () => void\n  size?: ContainerSizesType\n  /**\n   * Define how much will stepper increase / decrease each time you click on + / - button.\n   */\n  step?: number\n  /**\n   * Text displayed into component at the right of number value.\n   */\n  text?: string\n  defaultValue?: number\n  value?: number | null\n  disabledTooltip?: string\n  className?: string\n  'data-testid'?: string\n  label?: string\n  'aria-label'?: string\n  'aria-describedby'?: string\n  id?: string\n  placeholder?: string\n  error?: string | boolean\n} & Omit<\n  InputHTMLAttributes<HTMLInputElement>,\n  'size' | 'onChange' | 'value' | 'defaultValue'\n>\n\n/**\n * @deprecated This component is deprecated. Please use `NumberInputV2` instead.\n */\nexport const NumberInput = ({\n  disabled = false,\n  maxValue,\n  minValue = 0,\n  name = 'numberinput',\n  onChange,\n  onFocus,\n  onBlur,\n  onMaxCrossed,\n  onMinCrossed,\n  size = 'large',\n  step = 1,\n  text,\n  defaultValue,\n  value,\n  disabledTooltip,\n  className,\n  label,\n  id,\n  placeholder,\n  error,\n  'aria-label': ariaLabel,\n  'aria-describedby': ariaDescribedBy,\n  'data-testid': dataTestId,\n}: NumberInputProps) => {\n  const inputRef = useRef<HTMLInputElement>(null)\n\n  const uniqueId = useId()\n\n  // local state used if component is not controlled (no value prop provided)\n  const [inputValue, setInputValue] = useState<number | undefined>(() => {\n    if (defaultValue && minValue && defaultValue < minValue) {\n      return minValue\n    }\n    if (defaultValue && maxValue && defaultValue > maxValue) {\n      return maxValue\n    }\n\n    return defaultValue\n  })\n\n  const currentValue =\n    value !== undefined && value !== null ? value : inputValue\n\n  const setValue = (\n    newValue: number | undefined,\n    /**\n     * If true, will check if newValue is between minValue and maxValue and set it to minValue or maxValue if it's not.\n     */\n    hasMinMaxVerification = true,\n  ) => {\n    let nextValue = newValue\n    if (value === undefined && hasMinMaxVerification) {\n      if (newValue !== undefined && newValue < minValue) {\n        nextValue = minValue\n      }\n\n      if (\n        newValue !== undefined &&\n        maxValue !== undefined &&\n        newValue > maxValue\n      ) {\n        nextValue = maxValue\n      }\n    }\n    setInputValue(nextValue)\n    onChange?.(nextValue)\n  }\n\n  const offsetFn = (direction: number) => () => {\n    const localValue = currentValue ?? 0\n    const newValue =\n      localValue % step === 0 ? localValue + step * direction : localValue\n    const roundedValue = roundStep(newValue, step, direction)\n\n    setValue(roundedValue)\n  }\n\n  const handleChange: ChangeEventHandler<HTMLInputElement> = event => {\n    setValue(\n      event.currentTarget.value ? Number(event.currentTarget.value) : undefined,\n      false,\n    )\n  }\n\n  const handleOnBlur: FocusEventHandler<HTMLInputElement> = event => {\n    if (currentValue) {\n      const boundedValue = bounded(\n        currentValue,\n        minValue ?? currentValue,\n        maxValue ?? currentValue,\n      )\n\n      if (maxValue && currentValue > maxValue) onMaxCrossed?.()\n      if (minValue && currentValue < minValue) onMinCrossed?.()\n\n      setValue(boundedValue)\n\n      onBlur?.(event)\n    }\n  }\n\n  const onKeyDown: KeyboardEventHandler = event => {\n    if (event.key === 'ArrowUp') {\n      event.stopPropagation()\n      event.preventDefault()\n\n      const direction = 1\n      const localValue = currentValue ?? 0\n\n      const newValue =\n        localValue % step === 0 ? localValue + step * direction : localValue\n      const roundedValue = roundStep(newValue, step, direction)\n\n      if (maxValue === undefined) {\n        setValue(roundedValue)\n\n        return\n      }\n\n      setValue(Math.min(roundedValue, maxValue))\n    }\n\n    if (event.key === 'ArrowDown') {\n      event.stopPropagation()\n      event.preventDefault()\n\n      const direction = -1\n      const localValue = currentValue ?? 0\n\n      const newValue =\n        localValue % step === 0 ? localValue + step * direction : localValue\n      const roundedValue = roundStep(newValue, step, direction)\n\n      setValue(Math.max(roundedValue, minValue))\n    }\n  }\n\n  const isMinusDisabled = useMemo(() => {\n    if (disabled) return true\n    if (currentValue === undefined) return false\n    if (getMinusRoundedValue(currentValue, step) < minValue) {\n      return true\n    }\n\n    return disabled\n  }, [currentValue, disabled, minValue, step])\n\n  const isPlusDisabled = useMemo(() => {\n    if (disabled) return true\n    if (currentValue === undefined) return false\n    if (maxValue && getPlusRoundedValue(currentValue, step) > maxValue) {\n      return true\n    }\n\n    return disabled\n  }, [currentValue, disabled, maxValue, step])\n\n  const inputWidth = useMemo(() => {\n    if (placeholder && currentValue === undefined) {\n      return placeholder.length * 12\n    }\n\n    if (currentValue !== undefined) {\n      return currentValue.toString().length * 16\n    }\n\n    return BASE_INPUT_WIDTH\n  }, [currentValue, placeholder])\n\n  return (\n    <Stack gap={1}>\n      {label ? (\n        <Text variant=\"bodyStrong\" as=\"label\" htmlFor={id || uniqueId}>\n          {label}\n        </Text>\n      ) : null}\n      <Stack gap={0.5}>\n        <StyledContainer\n          aria-disabled={disabled}\n          data-error={!!error}\n          size={size}\n          className={className}\n          data-testid={dataTestId}\n        >\n          <Tooltip text={isMinusDisabled && disabledTooltip}>\n            <StyledSelectButton\n              onClick={offsetFn(-1)}\n              disabled={isMinusDisabled}\n              aria-label=\"Minus\"\n              type=\"button\"\n              variant=\"ghost\"\n              sentiment=\"primary\"\n              size=\"small\"\n            >\n              <MinusIcon\n                size={iconSizes[size]}\n                sentiment=\"primary\"\n                disabled={isMinusDisabled}\n              />\n            </StyledSelectButton>\n          </Tooltip>\n\n          <StyledCenterBox\n            size={size}\n            onClick={() => {\n              if (inputRef?.current) {\n                inputRef.current.focus()\n              }\n            }}\n            aria-live=\"assertive\"\n            role=\"status\"\n          >\n            <StyledInput\n              disabled={disabled}\n              name={name}\n              onBlur={handleOnBlur}\n              onChange={handleChange}\n              onFocus={onFocus}\n              onKeyDown={onKeyDown}\n              ref={inputRef}\n              style={{\n                width: inputWidth,\n              }}\n              value={currentValue !== undefined ? currentValue.toString() : ''} // A dom element can only have string attributes.\n              type=\"number\"\n              id={id || uniqueId}\n              aria-label={!label && !ariaLabel ? 'Number Input' : ariaLabel}\n              aria-describedby={ariaDescribedBy}\n              placeholder={placeholder}\n            />\n            {currentValue !== undefined ? (\n              <StyledText disabled={disabled}>{text}</StyledText>\n            ) : null}\n          </StyledCenterBox>\n\n          <Tooltip text={isPlusDisabled && disabledTooltip}>\n            <StyledSelectButton\n              onClick={offsetFn(1)}\n              disabled={isPlusDisabled}\n              aria-label=\"Plus\"\n              type=\"button\"\n              variant=\"ghost\"\n              sentiment=\"primary\"\n              size=\"small\"\n            >\n              <PlusIcon\n                size={iconSizes[size]}\n                sentiment=\"primary\"\n                disabled={isPlusDisabled}\n              />\n            </StyledSelectButton>\n          </Tooltip>\n        </StyledContainer>\n        {typeof error === 'string' ? (\n          <Text\n            as=\"span\"\n            variant=\"bodySmall\"\n            sentiment=\"danger\"\n            prominence=\"weak\"\n          >\n            {error}\n          </Text>\n        ) : null}\n      </Stack>\n    </Stack>\n  )\n}\n"]} */"));
59
59
  const StyledText = /* @__PURE__ */ _styled("span", process.env.NODE_ENV === "production" ? {
60
60
  shouldForwardProp: (prop) => !["disabled"].includes(prop),
61
61
  target: "exvap481"
@@ -68,7 +68,7 @@ const StyledText = /* @__PURE__ */ _styled("span", process.env.NODE_ENV === "pro
68
68
  disabled
69
69
  }) => disabled ? theme.colors.neutral.textDisabled : theme.colors.neutral.text, ";user-select:none;margin-right:", ({
70
70
  theme
71
- }) => theme.space["1"], ";" + (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/NumberInput/index.tsx"],"names":[],"mappings":"AA0FyB","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/NumberInput/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { MinusIcon, PlusIcon } from '@ultraviolet/icons'\nimport type {\n  ChangeEventHandler,\n  FocusEventHandler,\n  InputHTMLAttributes,\n  KeyboardEventHandler,\n  MutableRefObject,\n} from 'react'\nimport { useId, useMemo, useRef, useState } from 'react'\nimport { Button } from '../Button'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\nimport { Tooltip } from '../Tooltip'\nimport {\n  bounded,\n  getMinusRoundedValue,\n  getPlusRoundedValue,\n  roundStep,\n} from './helpers'\n\nconst containerSizes = {\n  large: 48,\n  medium: 40,\n  small: 32,\n}\n\ntype ContainerSizesType = keyof typeof containerSizes\n\nconst iconSizes = {\n  large: 26,\n  medium: 24,\n  small: 22,\n}\n\nconst BASE_INPUT_WIDTH = 34\n\nconst StyledSelectButton = styled(Button)`\n  margin: 0 ${({ theme }) => theme.space['1']};\n  width: 32px;\n  height: 32px;\n`\n\nconst StyledCenterBox = styled('div', {\n  shouldForwardProp: prop => !['size'].includes(prop),\n})<{ size: ContainerSizesType }>`\n  display: flex;\n  flex: 1;\n  flex-direction: row;\n  height: ${({ size }) => (size === 'small' ? '24px' : '32px')};\n  align-items: center;\n  outline: none;\n  justify-content: center;\n  border-radius: ${({ theme }) => theme.radii.default};\n  border: 1px solid transparent;\n  max-width: 100%;\n`\n\nconst StyledInput = styled.input`\n  color: ${({ theme }) => theme.colors.neutral.text};\n  background-color: transparent;\n  font-size: ${({ theme }) => theme.typography.bodyStrong.fontSize};\n  border: none;\n  outline: none;\n  position: relative;\n  margin-right: ${({ theme }) => theme.space['0.5']};\n  max-width: 100%;\n  font-weight: ${({ theme }) => theme.typography.bodyStrong.weight};\n  text-align: center;\n\n  &::-webkit-outer-spin-button,\n  &::-webkit-inner-spin-button {\n    -webkit-appearance: none;\n    margin: 0;\n  }\n\n  ::placeholder {\n    color: ${({ theme }) => theme.colors.neutral.textWeak};\n  }\n\n  -moz-appearance: textfield;\n\n  &[disabled] {\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n    cursor: not-allowed;\n  }\n`\n\nconst StyledText = styled('span', {\n  shouldForwardProp: prop => !['disabled'].includes(prop),\n})<{ disabled: boolean }>`\n  color: ${({ theme, disabled }) =>\n    disabled ? theme.colors.neutral.textDisabled : theme.colors.neutral.text};\n  user-select: none;\n  margin-right: ${({ theme }) => theme.space['1']};\n`\n\nconst StyledContainer = styled('div', {\n  shouldForwardProp: prop => !['size'].includes(prop),\n})<{ size: ContainerSizesType }>`\n  background-color: ${({ theme }) => theme.colors.neutral.background};\n  display: flex;\n  flex-direction: row;\n  align-items: center;\n  align-self: stretch;\n  font-weight: 500;\n  height: ${({ size }) => containerSizes[size]}px;\n  border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  border-radius: ${({ theme }) => theme.radii.default};\n\n  &[data-error='true'] {\n    border: 1px solid ${({ theme }) => theme.colors.danger.border};\n  }\n\n  &[aria-disabled='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    cursor: not-allowed;\n  }\n\n  &:not([aria-disabled='true']) {\n    ${StyledCenterBox}:hover,\n    ${StyledCenterBox}:focus {\n      border: 1px solid ${({ theme }) => theme.colors.primary.borderHover};\n    }\n\n    ${StyledCenterBox}:focus-within {\n      box-shadow: ${({ theme }) => theme.shadows.focusPrimary};\n      border: 1px solid ${({ theme }) => theme.colors.primary.borderHover};\n    }\n  }\n`\n\ntype NumberInputProps = {\n  disabled?: boolean\n  maxValue?: number\n  minValue?: number\n  name?: string\n  onChange?: (input: number | undefined) => void\n  onMaxCrossed?: () => void\n  onMinCrossed?: () => void\n  size?: ContainerSizesType\n  /**\n   * Define how much will stepper increase / decrease each time you click on + / - button.\n   */\n  step?: number\n  /**\n   * Text displayed into component at the right of number value.\n   */\n  text?: string\n  defaultValue?: number\n  value?: number | null\n  disabledTooltip?: string\n  className?: string\n  'data-testid'?: string\n  label?: string\n  'aria-label'?: string\n  'aria-describedby'?: string\n  id?: string\n  placeholder?: string\n  error?: string | boolean\n} & Omit<\n  InputHTMLAttributes<HTMLInputElement>,\n  'size' | 'onChange' | 'value' | 'defaultValue'\n>\n\n/**\n * @deprecated This component is deprecated. Please use `NumberInputV2` instead.\n */\nexport const NumberInput = ({\n  disabled = false,\n  maxValue,\n  minValue = 0,\n  name = 'numberinput',\n  onChange,\n  onFocus,\n  onBlur,\n  onMaxCrossed,\n  onMinCrossed,\n  size = 'large',\n  step = 1,\n  text,\n  defaultValue,\n  value,\n  disabledTooltip,\n  className,\n  label,\n  id,\n  placeholder,\n  error,\n  'aria-label': ariaLabel,\n  'aria-describedby': ariaDescribedBy,\n  'data-testid': dataTestId,\n}: NumberInputProps) => {\n  const inputRef =\n    useRef<HTMLInputElement>() as MutableRefObject<HTMLInputElement>\n\n  const uniqueId = useId()\n\n  // local state used if component is not controlled (no value prop provided)\n  const [inputValue, setInputValue] = useState<number | undefined>(() => {\n    if (defaultValue && minValue && defaultValue < minValue) {\n      return minValue\n    }\n    if (defaultValue && maxValue && defaultValue > maxValue) {\n      return maxValue\n    }\n\n    return defaultValue\n  })\n\n  const currentValue =\n    value !== undefined && value !== null ? value : inputValue\n\n  const setValue = (\n    newValue: number | undefined,\n    /**\n     * If true, will check if newValue is between minValue and maxValue and set it to minValue or maxValue if it's not.\n     */\n    hasMinMaxVerification = true,\n  ) => {\n    let nextValue = newValue\n    if (value === undefined && hasMinMaxVerification) {\n      if (newValue !== undefined && newValue < minValue) {\n        nextValue = minValue\n      }\n\n      if (\n        newValue !== undefined &&\n        maxValue !== undefined &&\n        newValue > maxValue\n      ) {\n        nextValue = maxValue\n      }\n    }\n    setInputValue(nextValue)\n    onChange?.(nextValue)\n  }\n\n  const offsetFn = (direction: number) => () => {\n    const localValue = currentValue ?? 0\n    const newValue =\n      localValue % step === 0 ? localValue + step * direction : localValue\n    const roundedValue = roundStep(newValue, step, direction)\n\n    setValue(roundedValue)\n  }\n\n  const handleChange: ChangeEventHandler<HTMLInputElement> = event => {\n    setValue(\n      event.currentTarget.value ? Number(event.currentTarget.value) : undefined,\n      false,\n    )\n  }\n\n  const handleOnBlur: FocusEventHandler<HTMLInputElement> = event => {\n    if (currentValue) {\n      const boundedValue = bounded(\n        currentValue,\n        minValue ?? currentValue,\n        maxValue ?? currentValue,\n      )\n\n      if (maxValue && currentValue > maxValue) onMaxCrossed?.()\n      if (minValue && currentValue < minValue) onMinCrossed?.()\n\n      setValue(boundedValue)\n\n      onBlur?.(event)\n    }\n  }\n\n  const onKeyDown: KeyboardEventHandler = event => {\n    if (event.key === 'ArrowUp') {\n      event.stopPropagation()\n      event.preventDefault()\n\n      const direction = 1\n      const localValue = currentValue ?? 0\n\n      const newValue =\n        localValue % step === 0 ? localValue + step * direction : localValue\n      const roundedValue = roundStep(newValue, step, direction)\n\n      if (maxValue === undefined) {\n        setValue(roundedValue)\n\n        return\n      }\n\n      setValue(Math.min(roundedValue, maxValue))\n    }\n\n    if (event.key === 'ArrowDown') {\n      event.stopPropagation()\n      event.preventDefault()\n\n      const direction = -1\n      const localValue = currentValue ?? 0\n\n      const newValue =\n        localValue % step === 0 ? localValue + step * direction : localValue\n      const roundedValue = roundStep(newValue, step, direction)\n\n      setValue(Math.max(roundedValue, minValue))\n    }\n  }\n\n  const isMinusDisabled = useMemo(() => {\n    if (disabled) return true\n    if (currentValue === undefined) return false\n    if (getMinusRoundedValue(currentValue, step) < minValue) {\n      return true\n    }\n\n    return disabled\n  }, [currentValue, disabled, minValue, step])\n\n  const isPlusDisabled = useMemo(() => {\n    if (disabled) return true\n    if (currentValue === undefined) return false\n    if (maxValue && getPlusRoundedValue(currentValue, step) > maxValue) {\n      return true\n    }\n\n    return disabled\n  }, [currentValue, disabled, maxValue, step])\n\n  const inputWidth = useMemo(() => {\n    if (placeholder && currentValue === undefined) {\n      return placeholder.length * 12\n    }\n\n    if (currentValue !== undefined) {\n      return currentValue.toString().length * 16\n    }\n\n    return BASE_INPUT_WIDTH\n  }, [currentValue, placeholder])\n\n  return (\n    <Stack gap={1}>\n      {label ? (\n        <Text variant=\"bodyStrong\" as=\"label\" htmlFor={id || uniqueId}>\n          {label}\n        </Text>\n      ) : null}\n      <Stack gap={0.5}>\n        <StyledContainer\n          aria-disabled={disabled}\n          data-error={!!error}\n          size={size}\n          className={className}\n          data-testid={dataTestId}\n        >\n          <Tooltip text={isMinusDisabled && disabledTooltip}>\n            <StyledSelectButton\n              onClick={offsetFn(-1)}\n              disabled={isMinusDisabled}\n              aria-label=\"Minus\"\n              type=\"button\"\n              variant=\"ghost\"\n              sentiment=\"primary\"\n              size=\"small\"\n            >\n              <MinusIcon\n                size={iconSizes[size]}\n                sentiment=\"primary\"\n                disabled={isMinusDisabled}\n              />\n            </StyledSelectButton>\n          </Tooltip>\n\n          <StyledCenterBox\n            size={size}\n            onClick={() => {\n              if (inputRef?.current) {\n                inputRef.current.focus()\n              }\n            }}\n            aria-live=\"assertive\"\n            role=\"status\"\n          >\n            <StyledInput\n              disabled={disabled}\n              name={name}\n              onBlur={handleOnBlur}\n              onChange={handleChange}\n              onFocus={onFocus}\n              onKeyDown={onKeyDown}\n              ref={inputRef}\n              style={{\n                width: inputWidth,\n              }}\n              value={currentValue !== undefined ? currentValue.toString() : ''} // A dom element can only have string attributes.\n              type=\"number\"\n              id={id || uniqueId}\n              aria-label={!label && !ariaLabel ? 'Number Input' : ariaLabel}\n              aria-describedby={ariaDescribedBy}\n              placeholder={placeholder}\n            />\n            {currentValue !== undefined ? (\n              <StyledText disabled={disabled}>{text}</StyledText>\n            ) : null}\n          </StyledCenterBox>\n\n          <Tooltip text={isPlusDisabled && disabledTooltip}>\n            <StyledSelectButton\n              onClick={offsetFn(1)}\n              disabled={isPlusDisabled}\n              aria-label=\"Plus\"\n              type=\"button\"\n              variant=\"ghost\"\n              sentiment=\"primary\"\n              size=\"small\"\n            >\n              <PlusIcon\n                size={iconSizes[size]}\n                sentiment=\"primary\"\n                disabled={isPlusDisabled}\n              />\n            </StyledSelectButton>\n          </Tooltip>\n        </StyledContainer>\n        {typeof error === 'string' ? (\n          <Text\n            as=\"span\"\n            variant=\"bodySmall\"\n            sentiment=\"danger\"\n            prominence=\"weak\"\n          >\n            {error}\n          </Text>\n        ) : null}\n      </Stack>\n    </Stack>\n  )\n}\n"]} */"));
71
+ }) => theme.space["1"], ";" + (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/NumberInput/index.tsx"],"names":[],"mappings":"AAyFyB","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/NumberInput/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { MinusIcon, PlusIcon } from '@ultraviolet/icons'\nimport type {\n  ChangeEventHandler,\n  FocusEventHandler,\n  InputHTMLAttributes,\n  KeyboardEventHandler,\n} from 'react'\nimport { useId, useMemo, useRef, useState } from 'react'\nimport { Button } from '../Button'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\nimport { Tooltip } from '../Tooltip'\nimport {\n  bounded,\n  getMinusRoundedValue,\n  getPlusRoundedValue,\n  roundStep,\n} from './helpers'\n\nconst containerSizes = {\n  large: 48,\n  medium: 40,\n  small: 32,\n}\n\ntype ContainerSizesType = keyof typeof containerSizes\n\nconst iconSizes = {\n  large: 26,\n  medium: 24,\n  small: 22,\n}\n\nconst BASE_INPUT_WIDTH = 34\n\nconst StyledSelectButton = styled(Button)`\n  margin: 0 ${({ theme }) => theme.space['1']};\n  width: 32px;\n  height: 32px;\n`\n\nconst StyledCenterBox = styled('div', {\n  shouldForwardProp: prop => !['size'].includes(prop),\n})<{ size: ContainerSizesType }>`\n  display: flex;\n  flex: 1;\n  flex-direction: row;\n  height: ${({ size }) => (size === 'small' ? '24px' : '32px')};\n  align-items: center;\n  outline: none;\n  justify-content: center;\n  border-radius: ${({ theme }) => theme.radii.default};\n  border: 1px solid transparent;\n  max-width: 100%;\n`\n\nconst StyledInput = styled.input`\n  color: ${({ theme }) => theme.colors.neutral.text};\n  background-color: transparent;\n  font-size: ${({ theme }) => theme.typography.bodyStrong.fontSize};\n  border: none;\n  outline: none;\n  position: relative;\n  margin-right: ${({ theme }) => theme.space['0.5']};\n  max-width: 100%;\n  font-weight: ${({ theme }) => theme.typography.bodyStrong.weight};\n  text-align: center;\n\n  &::-webkit-outer-spin-button,\n  &::-webkit-inner-spin-button {\n    -webkit-appearance: none;\n    margin: 0;\n  }\n\n  ::placeholder {\n    color: ${({ theme }) => theme.colors.neutral.textWeak};\n  }\n\n  -moz-appearance: textfield;\n\n  &[disabled] {\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n    cursor: not-allowed;\n  }\n`\n\nconst StyledText = styled('span', {\n  shouldForwardProp: prop => !['disabled'].includes(prop),\n})<{ disabled: boolean }>`\n  color: ${({ theme, disabled }) =>\n    disabled ? theme.colors.neutral.textDisabled : theme.colors.neutral.text};\n  user-select: none;\n  margin-right: ${({ theme }) => theme.space['1']};\n`\n\nconst StyledContainer = styled('div', {\n  shouldForwardProp: prop => !['size'].includes(prop),\n})<{ size: ContainerSizesType }>`\n  background-color: ${({ theme }) => theme.colors.neutral.background};\n  display: flex;\n  flex-direction: row;\n  align-items: center;\n  align-self: stretch;\n  font-weight: 500;\n  height: ${({ size }) => containerSizes[size]}px;\n  border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  border-radius: ${({ theme }) => theme.radii.default};\n\n  &[data-error='true'] {\n    border: 1px solid ${({ theme }) => theme.colors.danger.border};\n  }\n\n  &[aria-disabled='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    cursor: not-allowed;\n  }\n\n  &:not([aria-disabled='true']) {\n    ${StyledCenterBox}:hover,\n    ${StyledCenterBox}:focus {\n      border: 1px solid ${({ theme }) => theme.colors.primary.borderHover};\n    }\n\n    ${StyledCenterBox}:focus-within {\n      box-shadow: ${({ theme }) => theme.shadows.focusPrimary};\n      border: 1px solid ${({ theme }) => theme.colors.primary.borderHover};\n    }\n  }\n`\n\ntype NumberInputProps = {\n  disabled?: boolean\n  maxValue?: number\n  minValue?: number\n  name?: string\n  onChange?: (input: number | undefined) => void\n  onMaxCrossed?: () => void\n  onMinCrossed?: () => void\n  size?: ContainerSizesType\n  /**\n   * Define how much will stepper increase / decrease each time you click on + / - button.\n   */\n  step?: number\n  /**\n   * Text displayed into component at the right of number value.\n   */\n  text?: string\n  defaultValue?: number\n  value?: number | null\n  disabledTooltip?: string\n  className?: string\n  'data-testid'?: string\n  label?: string\n  'aria-label'?: string\n  'aria-describedby'?: string\n  id?: string\n  placeholder?: string\n  error?: string | boolean\n} & Omit<\n  InputHTMLAttributes<HTMLInputElement>,\n  'size' | 'onChange' | 'value' | 'defaultValue'\n>\n\n/**\n * @deprecated This component is deprecated. Please use `NumberInputV2` instead.\n */\nexport const NumberInput = ({\n  disabled = false,\n  maxValue,\n  minValue = 0,\n  name = 'numberinput',\n  onChange,\n  onFocus,\n  onBlur,\n  onMaxCrossed,\n  onMinCrossed,\n  size = 'large',\n  step = 1,\n  text,\n  defaultValue,\n  value,\n  disabledTooltip,\n  className,\n  label,\n  id,\n  placeholder,\n  error,\n  'aria-label': ariaLabel,\n  'aria-describedby': ariaDescribedBy,\n  'data-testid': dataTestId,\n}: NumberInputProps) => {\n  const inputRef = useRef<HTMLInputElement>(null)\n\n  const uniqueId = useId()\n\n  // local state used if component is not controlled (no value prop provided)\n  const [inputValue, setInputValue] = useState<number | undefined>(() => {\n    if (defaultValue && minValue && defaultValue < minValue) {\n      return minValue\n    }\n    if (defaultValue && maxValue && defaultValue > maxValue) {\n      return maxValue\n    }\n\n    return defaultValue\n  })\n\n  const currentValue =\n    value !== undefined && value !== null ? value : inputValue\n\n  const setValue = (\n    newValue: number | undefined,\n    /**\n     * If true, will check if newValue is between minValue and maxValue and set it to minValue or maxValue if it's not.\n     */\n    hasMinMaxVerification = true,\n  ) => {\n    let nextValue = newValue\n    if (value === undefined && hasMinMaxVerification) {\n      if (newValue !== undefined && newValue < minValue) {\n        nextValue = minValue\n      }\n\n      if (\n        newValue !== undefined &&\n        maxValue !== undefined &&\n        newValue > maxValue\n      ) {\n        nextValue = maxValue\n      }\n    }\n    setInputValue(nextValue)\n    onChange?.(nextValue)\n  }\n\n  const offsetFn = (direction: number) => () => {\n    const localValue = currentValue ?? 0\n    const newValue =\n      localValue % step === 0 ? localValue + step * direction : localValue\n    const roundedValue = roundStep(newValue, step, direction)\n\n    setValue(roundedValue)\n  }\n\n  const handleChange: ChangeEventHandler<HTMLInputElement> = event => {\n    setValue(\n      event.currentTarget.value ? Number(event.currentTarget.value) : undefined,\n      false,\n    )\n  }\n\n  const handleOnBlur: FocusEventHandler<HTMLInputElement> = event => {\n    if (currentValue) {\n      const boundedValue = bounded(\n        currentValue,\n        minValue ?? currentValue,\n        maxValue ?? currentValue,\n      )\n\n      if (maxValue && currentValue > maxValue) onMaxCrossed?.()\n      if (minValue && currentValue < minValue) onMinCrossed?.()\n\n      setValue(boundedValue)\n\n      onBlur?.(event)\n    }\n  }\n\n  const onKeyDown: KeyboardEventHandler = event => {\n    if (event.key === 'ArrowUp') {\n      event.stopPropagation()\n      event.preventDefault()\n\n      const direction = 1\n      const localValue = currentValue ?? 0\n\n      const newValue =\n        localValue % step === 0 ? localValue + step * direction : localValue\n      const roundedValue = roundStep(newValue, step, direction)\n\n      if (maxValue === undefined) {\n        setValue(roundedValue)\n\n        return\n      }\n\n      setValue(Math.min(roundedValue, maxValue))\n    }\n\n    if (event.key === 'ArrowDown') {\n      event.stopPropagation()\n      event.preventDefault()\n\n      const direction = -1\n      const localValue = currentValue ?? 0\n\n      const newValue =\n        localValue % step === 0 ? localValue + step * direction : localValue\n      const roundedValue = roundStep(newValue, step, direction)\n\n      setValue(Math.max(roundedValue, minValue))\n    }\n  }\n\n  const isMinusDisabled = useMemo(() => {\n    if (disabled) return true\n    if (currentValue === undefined) return false\n    if (getMinusRoundedValue(currentValue, step) < minValue) {\n      return true\n    }\n\n    return disabled\n  }, [currentValue, disabled, minValue, step])\n\n  const isPlusDisabled = useMemo(() => {\n    if (disabled) return true\n    if (currentValue === undefined) return false\n    if (maxValue && getPlusRoundedValue(currentValue, step) > maxValue) {\n      return true\n    }\n\n    return disabled\n  }, [currentValue, disabled, maxValue, step])\n\n  const inputWidth = useMemo(() => {\n    if (placeholder && currentValue === undefined) {\n      return placeholder.length * 12\n    }\n\n    if (currentValue !== undefined) {\n      return currentValue.toString().length * 16\n    }\n\n    return BASE_INPUT_WIDTH\n  }, [currentValue, placeholder])\n\n  return (\n    <Stack gap={1}>\n      {label ? (\n        <Text variant=\"bodyStrong\" as=\"label\" htmlFor={id || uniqueId}>\n          {label}\n        </Text>\n      ) : null}\n      <Stack gap={0.5}>\n        <StyledContainer\n          aria-disabled={disabled}\n          data-error={!!error}\n          size={size}\n          className={className}\n          data-testid={dataTestId}\n        >\n          <Tooltip text={isMinusDisabled && disabledTooltip}>\n            <StyledSelectButton\n              onClick={offsetFn(-1)}\n              disabled={isMinusDisabled}\n              aria-label=\"Minus\"\n              type=\"button\"\n              variant=\"ghost\"\n              sentiment=\"primary\"\n              size=\"small\"\n            >\n              <MinusIcon\n                size={iconSizes[size]}\n                sentiment=\"primary\"\n                disabled={isMinusDisabled}\n              />\n            </StyledSelectButton>\n          </Tooltip>\n\n          <StyledCenterBox\n            size={size}\n            onClick={() => {\n              if (inputRef?.current) {\n                inputRef.current.focus()\n              }\n            }}\n            aria-live=\"assertive\"\n            role=\"status\"\n          >\n            <StyledInput\n              disabled={disabled}\n              name={name}\n              onBlur={handleOnBlur}\n              onChange={handleChange}\n              onFocus={onFocus}\n              onKeyDown={onKeyDown}\n              ref={inputRef}\n              style={{\n                width: inputWidth,\n              }}\n              value={currentValue !== undefined ? currentValue.toString() : ''} // A dom element can only have string attributes.\n              type=\"number\"\n              id={id || uniqueId}\n              aria-label={!label && !ariaLabel ? 'Number Input' : ariaLabel}\n              aria-describedby={ariaDescribedBy}\n              placeholder={placeholder}\n            />\n            {currentValue !== undefined ? (\n              <StyledText disabled={disabled}>{text}</StyledText>\n            ) : null}\n          </StyledCenterBox>\n\n          <Tooltip text={isPlusDisabled && disabledTooltip}>\n            <StyledSelectButton\n              onClick={offsetFn(1)}\n              disabled={isPlusDisabled}\n              aria-label=\"Plus\"\n              type=\"button\"\n              variant=\"ghost\"\n              sentiment=\"primary\"\n              size=\"small\"\n            >\n              <PlusIcon\n                size={iconSizes[size]}\n                sentiment=\"primary\"\n                disabled={isPlusDisabled}\n              />\n            </StyledSelectButton>\n          </Tooltip>\n        </StyledContainer>\n        {typeof error === 'string' ? (\n          <Text\n            as=\"span\"\n            variant=\"bodySmall\"\n            sentiment=\"danger\"\n            prominence=\"weak\"\n          >\n            {error}\n          </Text>\n        ) : null}\n      </Stack>\n    </Stack>\n  )\n}\n"]} */"));
72
72
  const StyledContainer = /* @__PURE__ */ _styled("div", process.env.NODE_ENV === "production" ? {
73
73
  shouldForwardProp: (prop) => !["size"].includes(prop),
74
74
  target: "exvap480"
@@ -94,7 +94,7 @@ const StyledContainer = /* @__PURE__ */ _styled("div", process.env.NODE_ENV ===
94
94
  theme
95
95
  }) => theme.shadows.focusPrimary, ";border:1px solid ", ({
96
96
  theme
97
- }) => theme.colors.primary.borderHover, ";}}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/NumberInput/index.tsx"],"names":[],"mappings":"AAmGgC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/NumberInput/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { MinusIcon, PlusIcon } from '@ultraviolet/icons'\nimport type {\n  ChangeEventHandler,\n  FocusEventHandler,\n  InputHTMLAttributes,\n  KeyboardEventHandler,\n  MutableRefObject,\n} from 'react'\nimport { useId, useMemo, useRef, useState } from 'react'\nimport { Button } from '../Button'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\nimport { Tooltip } from '../Tooltip'\nimport {\n  bounded,\n  getMinusRoundedValue,\n  getPlusRoundedValue,\n  roundStep,\n} from './helpers'\n\nconst containerSizes = {\n  large: 48,\n  medium: 40,\n  small: 32,\n}\n\ntype ContainerSizesType = keyof typeof containerSizes\n\nconst iconSizes = {\n  large: 26,\n  medium: 24,\n  small: 22,\n}\n\nconst BASE_INPUT_WIDTH = 34\n\nconst StyledSelectButton = styled(Button)`\n  margin: 0 ${({ theme }) => theme.space['1']};\n  width: 32px;\n  height: 32px;\n`\n\nconst StyledCenterBox = styled('div', {\n  shouldForwardProp: prop => !['size'].includes(prop),\n})<{ size: ContainerSizesType }>`\n  display: flex;\n  flex: 1;\n  flex-direction: row;\n  height: ${({ size }) => (size === 'small' ? '24px' : '32px')};\n  align-items: center;\n  outline: none;\n  justify-content: center;\n  border-radius: ${({ theme }) => theme.radii.default};\n  border: 1px solid transparent;\n  max-width: 100%;\n`\n\nconst StyledInput = styled.input`\n  color: ${({ theme }) => theme.colors.neutral.text};\n  background-color: transparent;\n  font-size: ${({ theme }) => theme.typography.bodyStrong.fontSize};\n  border: none;\n  outline: none;\n  position: relative;\n  margin-right: ${({ theme }) => theme.space['0.5']};\n  max-width: 100%;\n  font-weight: ${({ theme }) => theme.typography.bodyStrong.weight};\n  text-align: center;\n\n  &::-webkit-outer-spin-button,\n  &::-webkit-inner-spin-button {\n    -webkit-appearance: none;\n    margin: 0;\n  }\n\n  ::placeholder {\n    color: ${({ theme }) => theme.colors.neutral.textWeak};\n  }\n\n  -moz-appearance: textfield;\n\n  &[disabled] {\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n    cursor: not-allowed;\n  }\n`\n\nconst StyledText = styled('span', {\n  shouldForwardProp: prop => !['disabled'].includes(prop),\n})<{ disabled: boolean }>`\n  color: ${({ theme, disabled }) =>\n    disabled ? theme.colors.neutral.textDisabled : theme.colors.neutral.text};\n  user-select: none;\n  margin-right: ${({ theme }) => theme.space['1']};\n`\n\nconst StyledContainer = styled('div', {\n  shouldForwardProp: prop => !['size'].includes(prop),\n})<{ size: ContainerSizesType }>`\n  background-color: ${({ theme }) => theme.colors.neutral.background};\n  display: flex;\n  flex-direction: row;\n  align-items: center;\n  align-self: stretch;\n  font-weight: 500;\n  height: ${({ size }) => containerSizes[size]}px;\n  border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  border-radius: ${({ theme }) => theme.radii.default};\n\n  &[data-error='true'] {\n    border: 1px solid ${({ theme }) => theme.colors.danger.border};\n  }\n\n  &[aria-disabled='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    cursor: not-allowed;\n  }\n\n  &:not([aria-disabled='true']) {\n    ${StyledCenterBox}:hover,\n    ${StyledCenterBox}:focus {\n      border: 1px solid ${({ theme }) => theme.colors.primary.borderHover};\n    }\n\n    ${StyledCenterBox}:focus-within {\n      box-shadow: ${({ theme }) => theme.shadows.focusPrimary};\n      border: 1px solid ${({ theme }) => theme.colors.primary.borderHover};\n    }\n  }\n`\n\ntype NumberInputProps = {\n  disabled?: boolean\n  maxValue?: number\n  minValue?: number\n  name?: string\n  onChange?: (input: number | undefined) => void\n  onMaxCrossed?: () => void\n  onMinCrossed?: () => void\n  size?: ContainerSizesType\n  /**\n   * Define how much will stepper increase / decrease each time you click on + / - button.\n   */\n  step?: number\n  /**\n   * Text displayed into component at the right of number value.\n   */\n  text?: string\n  defaultValue?: number\n  value?: number | null\n  disabledTooltip?: string\n  className?: string\n  'data-testid'?: string\n  label?: string\n  'aria-label'?: string\n  'aria-describedby'?: string\n  id?: string\n  placeholder?: string\n  error?: string | boolean\n} & Omit<\n  InputHTMLAttributes<HTMLInputElement>,\n  'size' | 'onChange' | 'value' | 'defaultValue'\n>\n\n/**\n * @deprecated This component is deprecated. Please use `NumberInputV2` instead.\n */\nexport const NumberInput = ({\n  disabled = false,\n  maxValue,\n  minValue = 0,\n  name = 'numberinput',\n  onChange,\n  onFocus,\n  onBlur,\n  onMaxCrossed,\n  onMinCrossed,\n  size = 'large',\n  step = 1,\n  text,\n  defaultValue,\n  value,\n  disabledTooltip,\n  className,\n  label,\n  id,\n  placeholder,\n  error,\n  'aria-label': ariaLabel,\n  'aria-describedby': ariaDescribedBy,\n  'data-testid': dataTestId,\n}: NumberInputProps) => {\n  const inputRef =\n    useRef<HTMLInputElement>() as MutableRefObject<HTMLInputElement>\n\n  const uniqueId = useId()\n\n  // local state used if component is not controlled (no value prop provided)\n  const [inputValue, setInputValue] = useState<number | undefined>(() => {\n    if (defaultValue && minValue && defaultValue < minValue) {\n      return minValue\n    }\n    if (defaultValue && maxValue && defaultValue > maxValue) {\n      return maxValue\n    }\n\n    return defaultValue\n  })\n\n  const currentValue =\n    value !== undefined && value !== null ? value : inputValue\n\n  const setValue = (\n    newValue: number | undefined,\n    /**\n     * If true, will check if newValue is between minValue and maxValue and set it to minValue or maxValue if it's not.\n     */\n    hasMinMaxVerification = true,\n  ) => {\n    let nextValue = newValue\n    if (value === undefined && hasMinMaxVerification) {\n      if (newValue !== undefined && newValue < minValue) {\n        nextValue = minValue\n      }\n\n      if (\n        newValue !== undefined &&\n        maxValue !== undefined &&\n        newValue > maxValue\n      ) {\n        nextValue = maxValue\n      }\n    }\n    setInputValue(nextValue)\n    onChange?.(nextValue)\n  }\n\n  const offsetFn = (direction: number) => () => {\n    const localValue = currentValue ?? 0\n    const newValue =\n      localValue % step === 0 ? localValue + step * direction : localValue\n    const roundedValue = roundStep(newValue, step, direction)\n\n    setValue(roundedValue)\n  }\n\n  const handleChange: ChangeEventHandler<HTMLInputElement> = event => {\n    setValue(\n      event.currentTarget.value ? Number(event.currentTarget.value) : undefined,\n      false,\n    )\n  }\n\n  const handleOnBlur: FocusEventHandler<HTMLInputElement> = event => {\n    if (currentValue) {\n      const boundedValue = bounded(\n        currentValue,\n        minValue ?? currentValue,\n        maxValue ?? currentValue,\n      )\n\n      if (maxValue && currentValue > maxValue) onMaxCrossed?.()\n      if (minValue && currentValue < minValue) onMinCrossed?.()\n\n      setValue(boundedValue)\n\n      onBlur?.(event)\n    }\n  }\n\n  const onKeyDown: KeyboardEventHandler = event => {\n    if (event.key === 'ArrowUp') {\n      event.stopPropagation()\n      event.preventDefault()\n\n      const direction = 1\n      const localValue = currentValue ?? 0\n\n      const newValue =\n        localValue % step === 0 ? localValue + step * direction : localValue\n      const roundedValue = roundStep(newValue, step, direction)\n\n      if (maxValue === undefined) {\n        setValue(roundedValue)\n\n        return\n      }\n\n      setValue(Math.min(roundedValue, maxValue))\n    }\n\n    if (event.key === 'ArrowDown') {\n      event.stopPropagation()\n      event.preventDefault()\n\n      const direction = -1\n      const localValue = currentValue ?? 0\n\n      const newValue =\n        localValue % step === 0 ? localValue + step * direction : localValue\n      const roundedValue = roundStep(newValue, step, direction)\n\n      setValue(Math.max(roundedValue, minValue))\n    }\n  }\n\n  const isMinusDisabled = useMemo(() => {\n    if (disabled) return true\n    if (currentValue === undefined) return false\n    if (getMinusRoundedValue(currentValue, step) < minValue) {\n      return true\n    }\n\n    return disabled\n  }, [currentValue, disabled, minValue, step])\n\n  const isPlusDisabled = useMemo(() => {\n    if (disabled) return true\n    if (currentValue === undefined) return false\n    if (maxValue && getPlusRoundedValue(currentValue, step) > maxValue) {\n      return true\n    }\n\n    return disabled\n  }, [currentValue, disabled, maxValue, step])\n\n  const inputWidth = useMemo(() => {\n    if (placeholder && currentValue === undefined) {\n      return placeholder.length * 12\n    }\n\n    if (currentValue !== undefined) {\n      return currentValue.toString().length * 16\n    }\n\n    return BASE_INPUT_WIDTH\n  }, [currentValue, placeholder])\n\n  return (\n    <Stack gap={1}>\n      {label ? (\n        <Text variant=\"bodyStrong\" as=\"label\" htmlFor={id || uniqueId}>\n          {label}\n        </Text>\n      ) : null}\n      <Stack gap={0.5}>\n        <StyledContainer\n          aria-disabled={disabled}\n          data-error={!!error}\n          size={size}\n          className={className}\n          data-testid={dataTestId}\n        >\n          <Tooltip text={isMinusDisabled && disabledTooltip}>\n            <StyledSelectButton\n              onClick={offsetFn(-1)}\n              disabled={isMinusDisabled}\n              aria-label=\"Minus\"\n              type=\"button\"\n              variant=\"ghost\"\n              sentiment=\"primary\"\n              size=\"small\"\n            >\n              <MinusIcon\n                size={iconSizes[size]}\n                sentiment=\"primary\"\n                disabled={isMinusDisabled}\n              />\n            </StyledSelectButton>\n          </Tooltip>\n\n          <StyledCenterBox\n            size={size}\n            onClick={() => {\n              if (inputRef?.current) {\n                inputRef.current.focus()\n              }\n            }}\n            aria-live=\"assertive\"\n            role=\"status\"\n          >\n            <StyledInput\n              disabled={disabled}\n              name={name}\n              onBlur={handleOnBlur}\n              onChange={handleChange}\n              onFocus={onFocus}\n              onKeyDown={onKeyDown}\n              ref={inputRef}\n              style={{\n                width: inputWidth,\n              }}\n              value={currentValue !== undefined ? currentValue.toString() : ''} // A dom element can only have string attributes.\n              type=\"number\"\n              id={id || uniqueId}\n              aria-label={!label && !ariaLabel ? 'Number Input' : ariaLabel}\n              aria-describedby={ariaDescribedBy}\n              placeholder={placeholder}\n            />\n            {currentValue !== undefined ? (\n              <StyledText disabled={disabled}>{text}</StyledText>\n            ) : null}\n          </StyledCenterBox>\n\n          <Tooltip text={isPlusDisabled && disabledTooltip}>\n            <StyledSelectButton\n              onClick={offsetFn(1)}\n              disabled={isPlusDisabled}\n              aria-label=\"Plus\"\n              type=\"button\"\n              variant=\"ghost\"\n              sentiment=\"primary\"\n              size=\"small\"\n            >\n              <PlusIcon\n                size={iconSizes[size]}\n                sentiment=\"primary\"\n                disabled={isPlusDisabled}\n              />\n            </StyledSelectButton>\n          </Tooltip>\n        </StyledContainer>\n        {typeof error === 'string' ? (\n          <Text\n            as=\"span\"\n            variant=\"bodySmall\"\n            sentiment=\"danger\"\n            prominence=\"weak\"\n          >\n            {error}\n          </Text>\n        ) : null}\n      </Stack>\n    </Stack>\n  )\n}\n"]} */"));
97
+ }) => theme.colors.primary.borderHover, ";}}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/NumberInput/index.tsx"],"names":[],"mappings":"AAkGgC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/NumberInput/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { MinusIcon, PlusIcon } from '@ultraviolet/icons'\nimport type {\n  ChangeEventHandler,\n  FocusEventHandler,\n  InputHTMLAttributes,\n  KeyboardEventHandler,\n} from 'react'\nimport { useId, useMemo, useRef, useState } from 'react'\nimport { Button } from '../Button'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\nimport { Tooltip } from '../Tooltip'\nimport {\n  bounded,\n  getMinusRoundedValue,\n  getPlusRoundedValue,\n  roundStep,\n} from './helpers'\n\nconst containerSizes = {\n  large: 48,\n  medium: 40,\n  small: 32,\n}\n\ntype ContainerSizesType = keyof typeof containerSizes\n\nconst iconSizes = {\n  large: 26,\n  medium: 24,\n  small: 22,\n}\n\nconst BASE_INPUT_WIDTH = 34\n\nconst StyledSelectButton = styled(Button)`\n  margin: 0 ${({ theme }) => theme.space['1']};\n  width: 32px;\n  height: 32px;\n`\n\nconst StyledCenterBox = styled('div', {\n  shouldForwardProp: prop => !['size'].includes(prop),\n})<{ size: ContainerSizesType }>`\n  display: flex;\n  flex: 1;\n  flex-direction: row;\n  height: ${({ size }) => (size === 'small' ? '24px' : '32px')};\n  align-items: center;\n  outline: none;\n  justify-content: center;\n  border-radius: ${({ theme }) => theme.radii.default};\n  border: 1px solid transparent;\n  max-width: 100%;\n`\n\nconst StyledInput = styled.input`\n  color: ${({ theme }) => theme.colors.neutral.text};\n  background-color: transparent;\n  font-size: ${({ theme }) => theme.typography.bodyStrong.fontSize};\n  border: none;\n  outline: none;\n  position: relative;\n  margin-right: ${({ theme }) => theme.space['0.5']};\n  max-width: 100%;\n  font-weight: ${({ theme }) => theme.typography.bodyStrong.weight};\n  text-align: center;\n\n  &::-webkit-outer-spin-button,\n  &::-webkit-inner-spin-button {\n    -webkit-appearance: none;\n    margin: 0;\n  }\n\n  ::placeholder {\n    color: ${({ theme }) => theme.colors.neutral.textWeak};\n  }\n\n  -moz-appearance: textfield;\n\n  &[disabled] {\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n    cursor: not-allowed;\n  }\n`\n\nconst StyledText = styled('span', {\n  shouldForwardProp: prop => !['disabled'].includes(prop),\n})<{ disabled: boolean }>`\n  color: ${({ theme, disabled }) =>\n    disabled ? theme.colors.neutral.textDisabled : theme.colors.neutral.text};\n  user-select: none;\n  margin-right: ${({ theme }) => theme.space['1']};\n`\n\nconst StyledContainer = styled('div', {\n  shouldForwardProp: prop => !['size'].includes(prop),\n})<{ size: ContainerSizesType }>`\n  background-color: ${({ theme }) => theme.colors.neutral.background};\n  display: flex;\n  flex-direction: row;\n  align-items: center;\n  align-self: stretch;\n  font-weight: 500;\n  height: ${({ size }) => containerSizes[size]}px;\n  border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  border-radius: ${({ theme }) => theme.radii.default};\n\n  &[data-error='true'] {\n    border: 1px solid ${({ theme }) => theme.colors.danger.border};\n  }\n\n  &[aria-disabled='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    cursor: not-allowed;\n  }\n\n  &:not([aria-disabled='true']) {\n    ${StyledCenterBox}:hover,\n    ${StyledCenterBox}:focus {\n      border: 1px solid ${({ theme }) => theme.colors.primary.borderHover};\n    }\n\n    ${StyledCenterBox}:focus-within {\n      box-shadow: ${({ theme }) => theme.shadows.focusPrimary};\n      border: 1px solid ${({ theme }) => theme.colors.primary.borderHover};\n    }\n  }\n`\n\ntype NumberInputProps = {\n  disabled?: boolean\n  maxValue?: number\n  minValue?: number\n  name?: string\n  onChange?: (input: number | undefined) => void\n  onMaxCrossed?: () => void\n  onMinCrossed?: () => void\n  size?: ContainerSizesType\n  /**\n   * Define how much will stepper increase / decrease each time you click on + / - button.\n   */\n  step?: number\n  /**\n   * Text displayed into component at the right of number value.\n   */\n  text?: string\n  defaultValue?: number\n  value?: number | null\n  disabledTooltip?: string\n  className?: string\n  'data-testid'?: string\n  label?: string\n  'aria-label'?: string\n  'aria-describedby'?: string\n  id?: string\n  placeholder?: string\n  error?: string | boolean\n} & Omit<\n  InputHTMLAttributes<HTMLInputElement>,\n  'size' | 'onChange' | 'value' | 'defaultValue'\n>\n\n/**\n * @deprecated This component is deprecated. Please use `NumberInputV2` instead.\n */\nexport const NumberInput = ({\n  disabled = false,\n  maxValue,\n  minValue = 0,\n  name = 'numberinput',\n  onChange,\n  onFocus,\n  onBlur,\n  onMaxCrossed,\n  onMinCrossed,\n  size = 'large',\n  step = 1,\n  text,\n  defaultValue,\n  value,\n  disabledTooltip,\n  className,\n  label,\n  id,\n  placeholder,\n  error,\n  'aria-label': ariaLabel,\n  'aria-describedby': ariaDescribedBy,\n  'data-testid': dataTestId,\n}: NumberInputProps) => {\n  const inputRef = useRef<HTMLInputElement>(null)\n\n  const uniqueId = useId()\n\n  // local state used if component is not controlled (no value prop provided)\n  const [inputValue, setInputValue] = useState<number | undefined>(() => {\n    if (defaultValue && minValue && defaultValue < minValue) {\n      return minValue\n    }\n    if (defaultValue && maxValue && defaultValue > maxValue) {\n      return maxValue\n    }\n\n    return defaultValue\n  })\n\n  const currentValue =\n    value !== undefined && value !== null ? value : inputValue\n\n  const setValue = (\n    newValue: number | undefined,\n    /**\n     * If true, will check if newValue is between minValue and maxValue and set it to minValue or maxValue if it's not.\n     */\n    hasMinMaxVerification = true,\n  ) => {\n    let nextValue = newValue\n    if (value === undefined && hasMinMaxVerification) {\n      if (newValue !== undefined && newValue < minValue) {\n        nextValue = minValue\n      }\n\n      if (\n        newValue !== undefined &&\n        maxValue !== undefined &&\n        newValue > maxValue\n      ) {\n        nextValue = maxValue\n      }\n    }\n    setInputValue(nextValue)\n    onChange?.(nextValue)\n  }\n\n  const offsetFn = (direction: number) => () => {\n    const localValue = currentValue ?? 0\n    const newValue =\n      localValue % step === 0 ? localValue + step * direction : localValue\n    const roundedValue = roundStep(newValue, step, direction)\n\n    setValue(roundedValue)\n  }\n\n  const handleChange: ChangeEventHandler<HTMLInputElement> = event => {\n    setValue(\n      event.currentTarget.value ? Number(event.currentTarget.value) : undefined,\n      false,\n    )\n  }\n\n  const handleOnBlur: FocusEventHandler<HTMLInputElement> = event => {\n    if (currentValue) {\n      const boundedValue = bounded(\n        currentValue,\n        minValue ?? currentValue,\n        maxValue ?? currentValue,\n      )\n\n      if (maxValue && currentValue > maxValue) onMaxCrossed?.()\n      if (minValue && currentValue < minValue) onMinCrossed?.()\n\n      setValue(boundedValue)\n\n      onBlur?.(event)\n    }\n  }\n\n  const onKeyDown: KeyboardEventHandler = event => {\n    if (event.key === 'ArrowUp') {\n      event.stopPropagation()\n      event.preventDefault()\n\n      const direction = 1\n      const localValue = currentValue ?? 0\n\n      const newValue =\n        localValue % step === 0 ? localValue + step * direction : localValue\n      const roundedValue = roundStep(newValue, step, direction)\n\n      if (maxValue === undefined) {\n        setValue(roundedValue)\n\n        return\n      }\n\n      setValue(Math.min(roundedValue, maxValue))\n    }\n\n    if (event.key === 'ArrowDown') {\n      event.stopPropagation()\n      event.preventDefault()\n\n      const direction = -1\n      const localValue = currentValue ?? 0\n\n      const newValue =\n        localValue % step === 0 ? localValue + step * direction : localValue\n      const roundedValue = roundStep(newValue, step, direction)\n\n      setValue(Math.max(roundedValue, minValue))\n    }\n  }\n\n  const isMinusDisabled = useMemo(() => {\n    if (disabled) return true\n    if (currentValue === undefined) return false\n    if (getMinusRoundedValue(currentValue, step) < minValue) {\n      return true\n    }\n\n    return disabled\n  }, [currentValue, disabled, minValue, step])\n\n  const isPlusDisabled = useMemo(() => {\n    if (disabled) return true\n    if (currentValue === undefined) return false\n    if (maxValue && getPlusRoundedValue(currentValue, step) > maxValue) {\n      return true\n    }\n\n    return disabled\n  }, [currentValue, disabled, maxValue, step])\n\n  const inputWidth = useMemo(() => {\n    if (placeholder && currentValue === undefined) {\n      return placeholder.length * 12\n    }\n\n    if (currentValue !== undefined) {\n      return currentValue.toString().length * 16\n    }\n\n    return BASE_INPUT_WIDTH\n  }, [currentValue, placeholder])\n\n  return (\n    <Stack gap={1}>\n      {label ? (\n        <Text variant=\"bodyStrong\" as=\"label\" htmlFor={id || uniqueId}>\n          {label}\n        </Text>\n      ) : null}\n      <Stack gap={0.5}>\n        <StyledContainer\n          aria-disabled={disabled}\n          data-error={!!error}\n          size={size}\n          className={className}\n          data-testid={dataTestId}\n        >\n          <Tooltip text={isMinusDisabled && disabledTooltip}>\n            <StyledSelectButton\n              onClick={offsetFn(-1)}\n              disabled={isMinusDisabled}\n              aria-label=\"Minus\"\n              type=\"button\"\n              variant=\"ghost\"\n              sentiment=\"primary\"\n              size=\"small\"\n            >\n              <MinusIcon\n                size={iconSizes[size]}\n                sentiment=\"primary\"\n                disabled={isMinusDisabled}\n              />\n            </StyledSelectButton>\n          </Tooltip>\n\n          <StyledCenterBox\n            size={size}\n            onClick={() => {\n              if (inputRef?.current) {\n                inputRef.current.focus()\n              }\n            }}\n            aria-live=\"assertive\"\n            role=\"status\"\n          >\n            <StyledInput\n              disabled={disabled}\n              name={name}\n              onBlur={handleOnBlur}\n              onChange={handleChange}\n              onFocus={onFocus}\n              onKeyDown={onKeyDown}\n              ref={inputRef}\n              style={{\n                width: inputWidth,\n              }}\n              value={currentValue !== undefined ? currentValue.toString() : ''} // A dom element can only have string attributes.\n              type=\"number\"\n              id={id || uniqueId}\n              aria-label={!label && !ariaLabel ? 'Number Input' : ariaLabel}\n              aria-describedby={ariaDescribedBy}\n              placeholder={placeholder}\n            />\n            {currentValue !== undefined ? (\n              <StyledText disabled={disabled}>{text}</StyledText>\n            ) : null}\n          </StyledCenterBox>\n\n          <Tooltip text={isPlusDisabled && disabledTooltip}>\n            <StyledSelectButton\n              onClick={offsetFn(1)}\n              disabled={isPlusDisabled}\n              aria-label=\"Plus\"\n              type=\"button\"\n              variant=\"ghost\"\n              sentiment=\"primary\"\n              size=\"small\"\n            >\n              <PlusIcon\n                size={iconSizes[size]}\n                sentiment=\"primary\"\n                disabled={isPlusDisabled}\n              />\n            </StyledSelectButton>\n          </Tooltip>\n        </StyledContainer>\n        {typeof error === 'string' ? (\n          <Text\n            as=\"span\"\n            variant=\"bodySmall\"\n            sentiment=\"danger\"\n            prominence=\"weak\"\n          >\n            {error}\n          </Text>\n        ) : null}\n      </Stack>\n    </Stack>\n  )\n}\n"]} */"));
98
98
  const NumberInput = ({
99
99
  disabled = false,
100
100
  maxValue,
@@ -120,7 +120,7 @@ const NumberInput = ({
120
120
  "aria-describedby": ariaDescribedBy,
121
121
  "data-testid": dataTestId
122
122
  }) => {
123
- const inputRef = useRef();
123
+ const inputRef = useRef(null);
124
124
  const uniqueId = useId();
125
125
  const [inputValue, setInputValue] = useState(() => {
126
126
  if (defaultValue && minValue && defaultValue < minValue) {
@@ -34,7 +34,7 @@ export declare const Popover: import("react").ForwardRefExoticComponent<{
34
34
  onFocus: () => void;
35
35
  onPointerEnter: () => void;
36
36
  onPointerLeave: () => void;
37
- ref: import("react").RefObject<HTMLDivElement>;
37
+ ref: import("react").RefObject<HTMLDivElement | null>;
38
38
  }) => ReactNode);
39
39
  maxWidth?: number | string;
40
40
  placement?: import("../Popup/helpers").PopupPlacement;
@@ -13,8 +13,8 @@ export declare const DEFAULT_POSITIONS: {
13
13
  };
14
14
  type ComputePositionsTypes = {
15
15
  placement: PopupPlacement;
16
- childrenRef: RefObject<HTMLDivElement>;
17
- popupRef: RefObject<HTMLDivElement>;
16
+ childrenRef: RefObject<HTMLDivElement | null>;
17
+ popupRef: RefObject<HTMLDivElement | null>;
18
18
  popupPortalTarget: HTMLElement;
19
19
  hasArrow: boolean;
20
20
  align: PopupAlign;