@ultraviolet/ui 1.68.0 → 1.68.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/components/DateInput/index.cjs +4 -2
- package/dist/components/DateInput/index.js +4 -2
- package/dist/components/Notification/index.cjs +1 -1
- package/dist/components/Notification/index.js +1 -1
- package/dist/components/NumberInput/index.cjs +5 -5
- package/dist/components/NumberInput/index.d.ts +2 -2
- package/dist/components/NumberInput/index.js +5 -5
- package/dist/components/PieChart/Legends.cjs +10 -10
- package/dist/components/PieChart/Legends.d.ts +1 -1
- package/dist/components/PieChart/Legends.js +10 -10
- package/dist/components/Toaster/index.cjs +2 -2
- package/dist/components/Toaster/index.js +2 -2
- package/package.json +7 -9
|
@@ -29,7 +29,7 @@ const StyledSelectButton = /* @__PURE__ */ _styled__default.default(index.Button
|
|
|
29
29
|
label: "StyledSelectButton"
|
|
30
30
|
})("margin:0 ", ({
|
|
31
31
|
theme
|
|
32
|
-
}) => 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 { Icon } 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              <Icon\n                name=\"minus\"\n                size={iconSizes[size]}\n                color=\"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              <Icon\n                name=\"plus\"\n                size={iconSizes[size]}\n                color=\"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"]} */"));
|
|
32
|
+
}) => 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 { Icon } 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              <Icon\n                name=\"minus\"\n                size={iconSizes[size]}\n                color=\"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              <Icon\n                name=\"plus\"\n                size={iconSizes[size]}\n                color=\"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"]} */"));
|
|
33
33
|
const StyledCenterBox = /* @__PURE__ */ _styled__default.default("div", process.env.NODE_ENV === "production" ? {
|
|
34
34
|
shouldForwardProp: (prop) => !["size"].includes(prop),
|
|
35
35
|
target: "exvap483"
|
|
@@ -41,7 +41,7 @@ const StyledCenterBox = /* @__PURE__ */ _styled__default.default("div", process.
|
|
|
41
41
|
size
|
|
42
42
|
}) => size === "small" ? "24px" : "32px", ";align-items:center;outline:none;justify-content:center;border-radius:", ({
|
|
43
43
|
theme
|
|
44
|
-
}) => 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 { Icon } 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              <Icon\n                name=\"minus\"\n                size={iconSizes[size]}\n                color=\"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              <Icon\n                name=\"plus\"\n                size={iconSizes[size]}\n                color=\"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"]} */"));
|
|
44
|
+
}) => 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 { Icon } 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              <Icon\n                name=\"minus\"\n                size={iconSizes[size]}\n                color=\"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              <Icon\n                name=\"plus\"\n                size={iconSizes[size]}\n                color=\"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"]} */"));
|
|
45
45
|
const StyledInput = /* @__PURE__ */ _styled__default.default("input", process.env.NODE_ENV === "production" ? {
|
|
46
46
|
target: "exvap482"
|
|
47
47
|
} : {
|
|
@@ -59,7 +59,7 @@ const StyledInput = /* @__PURE__ */ _styled__default.default("input", process.en
|
|
|
59
59
|
theme
|
|
60
60
|
}) => theme.colors.neutral.textWeak, ";}-moz-appearance:textfield;&[disabled]{color:", ({
|
|
61
61
|
theme
|
|
62
|
-
}) => 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 { Icon } 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              <Icon\n                name=\"minus\"\n                size={iconSizes[size]}\n                color=\"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              <Icon\n                name=\"plus\"\n                size={iconSizes[size]}\n                color=\"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"]} */"));
|
|
62
|
+
}) => 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 { Icon } 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              <Icon\n                name=\"minus\"\n                size={iconSizes[size]}\n                color=\"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              <Icon\n                name=\"plus\"\n                size={iconSizes[size]}\n                color=\"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"]} */"));
|
|
63
63
|
const StyledText = /* @__PURE__ */ _styled__default.default("span", process.env.NODE_ENV === "production" ? {
|
|
64
64
|
shouldForwardProp: (prop) => !["disabled"].includes(prop),
|
|
65
65
|
target: "exvap481"
|
|
@@ -72,7 +72,7 @@ const StyledText = /* @__PURE__ */ _styled__default.default("span", process.env.
|
|
|
72
72
|
disabled
|
|
73
73
|
}) => disabled ? theme.colors.neutral.textDisabled : theme.colors.neutral.text, ";user-select:none;margin-right:", ({
|
|
74
74
|
theme
|
|
75
|
-
}) => 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 { Icon } 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              <Icon\n                name=\"minus\"\n                size={iconSizes[size]}\n                color=\"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              <Icon\n                name=\"plus\"\n                size={iconSizes[size]}\n                color=\"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"]} */"));
|
|
75
|
+
}) => 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 { Icon } 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              <Icon\n                name=\"minus\"\n                size={iconSizes[size]}\n                color=\"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              <Icon\n                name=\"plus\"\n                size={iconSizes[size]}\n                color=\"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"]} */"));
|
|
76
76
|
const StyledContainer = /* @__PURE__ */ _styled__default.default("div", process.env.NODE_ENV === "production" ? {
|
|
77
77
|
shouldForwardProp: (prop) => !["size"].includes(prop),
|
|
78
78
|
target: "exvap480"
|
|
@@ -98,7 +98,7 @@ const StyledContainer = /* @__PURE__ */ _styled__default.default("div", process.
|
|
|
98
98
|
theme
|
|
99
99
|
}) => theme.shadows.focusPrimary, ";border:1px solid ", ({
|
|
100
100
|
theme
|
|
101
|
-
}) => 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 { Icon } 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              <Icon\n                name=\"minus\"\n                size={iconSizes[size]}\n                color=\"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              <Icon\n                name=\"plus\"\n                size={iconSizes[size]}\n                color=\"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"]} */"));
|
|
101
|
+
}) => 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 { Icon } 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              <Icon\n                name=\"minus\"\n                size={iconSizes[size]}\n                color=\"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              <Icon\n                name=\"plus\"\n                size={iconSizes[size]}\n                color=\"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"]} */"));
|
|
102
102
|
const NumberInput = ({
|
|
103
103
|
disabled = false,
|
|
104
104
|
maxValue,
|
|
@@ -11,8 +11,8 @@ type NumberInputProps = {
|
|
|
11
11
|
minValue?: number;
|
|
12
12
|
name?: string;
|
|
13
13
|
onChange?: (input: number | undefined) => void;
|
|
14
|
-
onMaxCrossed
|
|
15
|
-
onMinCrossed
|
|
14
|
+
onMaxCrossed?: () => void;
|
|
15
|
+
onMinCrossed?: () => void;
|
|
16
16
|
size?: ContainerSizesType;
|
|
17
17
|
/**
|
|
18
18
|
* Define how much will stepper increase / decrease each time you click on + / - button.
|