@ultraviolet/ui 1.80.0 → 1.81.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/Banner/assets/default-image-small.svg.cjs +1 -1
- package/dist/components/Banner/assets/default-image-small.svg.js +1 -1
- package/dist/components/Banner/assets/default-image.svg.cjs +1 -1
- package/dist/components/Banner/assets/default-image.svg.js +1 -1
- package/dist/components/ExpandableCard/index.cjs +8 -8
- package/dist/components/ExpandableCard/index.js +8 -8
- package/dist/components/List/ListContext.cjs +5 -3
- package/dist/components/List/ListContext.d.ts +4 -1
- package/dist/components/List/ListContext.js +5 -3
- package/dist/components/List/Row.cjs +20 -10
- package/dist/components/List/Row.d.ts +3 -0
- package/dist/components/List/Row.js +20 -10
- package/dist/components/List/SkeletonRows.cjs +7 -3
- package/dist/components/List/SkeletonRows.js +7 -3
- package/dist/components/List/index.cjs +19 -3
- package/dist/components/List/index.d.ts +4 -10
- package/dist/components/List/index.js +19 -3
- package/dist/components/List/types.d.ts +9 -0
- package/dist/components/Loader/index.cjs +3 -3
- package/dist/components/Loader/index.d.ts +2 -2
- package/dist/components/Loader/index.js +3 -3
- package/dist/components/Modal/ModalContent.cjs +1 -1
- package/dist/components/Modal/ModalContent.d.ts +1 -1
- package/dist/components/Modal/ModalContent.js +1 -1
- package/dist/components/Notice/index.cjs +3 -3
- package/dist/components/Notice/index.js +3 -3
- package/dist/components/NumberInput/index.cjs +8 -8
- package/dist/components/NumberInput/index.js +8 -8
- package/dist/components/PasswordCheck/index.cjs +3 -3
- package/dist/components/PasswordCheck/index.js +3 -3
- package/dist/components/Popup/animations.d.ts +12 -2
- package/dist/components/ProgressBar/index.cjs +54 -14
- package/dist/components/ProgressBar/index.d.ts +4 -1
- package/dist/components/ProgressBar/index.js +54 -14
- package/dist/components/SearchInput/index.cjs +4 -4
- package/dist/components/SearchInput/index.js +4 -4
- package/dist/components/Slider/components/DoubleSlider.cjs +19 -19
- package/dist/components/Slider/components/DoubleSlider.js +20 -20
- package/dist/components/Slider/components/SingleSlider.cjs +11 -11
- package/dist/components/Slider/components/SingleSlider.js +12 -12
- package/dist/components/Table/Row.cjs +19 -9
- package/dist/components/Table/Row.js +19 -9
- package/dist/components/Table/TableContext.cjs +5 -3
- package/dist/components/Table/TableContext.d.ts +4 -1
- package/dist/components/Table/TableContext.js +5 -3
- package/dist/components/Table/index.cjs +19 -3
- package/dist/components/Table/index.d.ts +4 -11
- package/dist/components/Table/index.js +19 -3
- package/dist/components/Table/types.d.ts +10 -0
- package/dist/components/TagInput/index.cjs +4 -4
- package/dist/components/TagInput/index.js +4 -4
- package/dist/components/VerificationCode/index.cjs +2 -2
- package/dist/components/VerificationCode/index.js +2 -2
- package/dist/utils/animations.d.ts +162 -27
- package/package.json +8 -8
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
import { jsxs, jsx } from "@emotion/react/jsx-runtime";
|
|
2
2
|
import _styled from "@emotion/styled/base";
|
|
3
|
-
import {
|
|
3
|
+
import { MinusIcon, PlusIcon } from "@ultraviolet/icons";
|
|
4
4
|
import { useRef, useId, useState, useMemo } from "react";
|
|
5
5
|
import { Button } from "../Button/index.js";
|
|
6
6
|
import { Stack } from "../Stack/index.js";
|
|
@@ -25,7 +25,7 @@ const StyledSelectButton = /* @__PURE__ */ _styled(Button, process.env.NODE_ENV
|
|
|
25
25
|
label: "StyledSelectButton"
|
|
26
26
|
})("margin:0 ", ({
|
|
27
27
|
theme
|
|
28
|
-
}) => theme.space["1"], ";width:32px;height:32px;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/NumberInput/index.tsx"],"names":[],"mappings":"AAqCyC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/NumberInput/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { Icon } from '@ultraviolet/icons/legacy'\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"]} */"));
|
|
28
|
+
}) => theme.space["1"], ";width:32px;height:32px;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/NumberInput/index.tsx"],"names":[],"mappings":"AAqCyC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/NumberInput/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { MinusIcon, PlusIcon } from '@ultraviolet/icons'\nimport type {\n  ChangeEventHandler,\n  FocusEventHandler,\n  InputHTMLAttributes,\n  KeyboardEventHandler,\n  MutableRefObject,\n} from 'react'\nimport { useId, useMemo, useRef, useState } from 'react'\nimport { Button } from '../Button'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\nimport { Tooltip } from '../Tooltip'\nimport {\n  bounded,\n  getMinusRoundedValue,\n  getPlusRoundedValue,\n  roundStep,\n} from './helpers'\n\nconst containerSizes = {\n  large: 48,\n  medium: 40,\n  small: 32,\n}\n\ntype ContainerSizesType = keyof typeof containerSizes\n\nconst iconSizes = {\n  large: 26,\n  medium: 24,\n  small: 22,\n}\n\nconst BASE_INPUT_WIDTH = 34\n\nconst StyledSelectButton = styled(Button)`\n  margin: 0 ${({ theme }) => theme.space['1']};\n  width: 32px;\n  height: 32px;\n`\n\nconst StyledCenterBox = styled('div', {\n  shouldForwardProp: prop => !['size'].includes(prop),\n})<{ size: ContainerSizesType }>`\n  display: flex;\n  flex: 1;\n  flex-direction: row;\n  height: ${({ size }) => (size === 'small' ? '24px' : '32px')};\n  align-items: center;\n  outline: none;\n  justify-content: center;\n  border-radius: ${({ theme }) => theme.radii.default};\n  border: 1px solid transparent;\n  max-width: 100%;\n`\n\nconst StyledInput = styled.input`\n  color: ${({ theme }) => theme.colors.neutral.text};\n  background-color: transparent;\n  font-size: ${({ theme }) => theme.typography.bodyStrong.fontSize};\n  border: none;\n  outline: none;\n  position: relative;\n  margin-right: ${({ theme }) => theme.space['0.5']};\n  max-width: 100%;\n  font-weight: ${({ theme }) => theme.typography.bodyStrong.weight};\n  text-align: center;\n\n  &::-webkit-outer-spin-button,\n  &::-webkit-inner-spin-button {\n    -webkit-appearance: none;\n    margin: 0;\n  }\n\n  ::placeholder {\n    color: ${({ theme }) => theme.colors.neutral.textWeak};\n  }\n\n  -moz-appearance: textfield;\n\n  &[disabled] {\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n    cursor: not-allowed;\n  }\n`\n\nconst StyledText = styled('span', {\n  shouldForwardProp: prop => !['disabled'].includes(prop),\n})<{ disabled: boolean }>`\n  color: ${({ theme, disabled }) =>\n    disabled ? theme.colors.neutral.textDisabled : theme.colors.neutral.text};\n  user-select: none;\n  margin-right: ${({ theme }) => theme.space['1']};\n`\n\nconst StyledContainer = styled('div', {\n  shouldForwardProp: prop => !['size'].includes(prop),\n})<{ size: ContainerSizesType }>`\n  background-color: ${({ theme }) => theme.colors.neutral.background};\n  display: flex;\n  flex-direction: row;\n  align-items: center;\n  align-self: stretch;\n  font-weight: 500;\n  height: ${({ size }) => containerSizes[size]}px;\n  border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  border-radius: ${({ theme }) => theme.radii.default};\n\n  &[data-error='true'] {\n    border: 1px solid ${({ theme }) => theme.colors.danger.border};\n  }\n\n  &[aria-disabled='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    cursor: not-allowed;\n  }\n\n  &:not([aria-disabled='true']) {\n    ${StyledCenterBox}:hover,\n    ${StyledCenterBox}:focus {\n      border: 1px solid ${({ theme }) => theme.colors.primary.borderHover};\n    }\n\n    ${StyledCenterBox}:focus-within {\n      box-shadow: ${({ theme }) => theme.shadows.focusPrimary};\n      border: 1px solid ${({ theme }) => theme.colors.primary.borderHover};\n    }\n  }\n`\n\ntype NumberInputProps = {\n  disabled?: boolean\n  maxValue?: number\n  minValue?: number\n  name?: string\n  onChange?: (input: number | undefined) => void\n  onMaxCrossed?: () => void\n  onMinCrossed?: () => void\n  size?: ContainerSizesType\n  /**\n   * Define how much will stepper increase / decrease each time you click on + / - button.\n   */\n  step?: number\n  /**\n   * Text displayed into component at the right of number value.\n   */\n  text?: string\n  defaultValue?: number\n  value?: number | null\n  disabledTooltip?: string\n  className?: string\n  'data-testid'?: string\n  label?: string\n  'aria-label'?: string\n  'aria-describedby'?: string\n  id?: string\n  placeholder?: string\n  error?: string | boolean\n} & Omit<\n  InputHTMLAttributes<HTMLInputElement>,\n  'size' | 'onChange' | 'value' | 'defaultValue'\n>\n\n/**\n * @deprecated This component is deprecated. Please use `NumberInputV2` instead.\n */\nexport const NumberInput = ({\n  disabled = false,\n  maxValue,\n  minValue = 0,\n  name = 'numberinput',\n  onChange,\n  onFocus,\n  onBlur,\n  onMaxCrossed,\n  onMinCrossed,\n  size = 'large',\n  step = 1,\n  text,\n  defaultValue,\n  value,\n  disabledTooltip,\n  className,\n  label,\n  id,\n  placeholder,\n  error,\n  'aria-label': ariaLabel,\n  'aria-describedby': ariaDescribedBy,\n  'data-testid': dataTestId,\n}: NumberInputProps) => {\n  const inputRef =\n    useRef<HTMLInputElement>() as MutableRefObject<HTMLInputElement>\n\n  const uniqueId = useId()\n\n  // local state used if component is not controlled (no value prop provided)\n  const [inputValue, setInputValue] = useState<number | undefined>(() => {\n    if (defaultValue && minValue && defaultValue < minValue) {\n      return minValue\n    }\n    if (defaultValue && maxValue && defaultValue > maxValue) {\n      return maxValue\n    }\n\n    return defaultValue\n  })\n\n  const currentValue =\n    value !== undefined && value !== null ? value : inputValue\n\n  const setValue = (\n    newValue: number | undefined,\n    /**\n     * If true, will check if newValue is between minValue and maxValue and set it to minValue or maxValue if it's not.\n     */\n    hasMinMaxVerification = true,\n  ) => {\n    let nextValue = newValue\n    if (value === undefined && hasMinMaxVerification) {\n      if (newValue !== undefined && newValue < minValue) {\n        nextValue = minValue\n      }\n\n      if (\n        newValue !== undefined &&\n        maxValue !== undefined &&\n        newValue > maxValue\n      ) {\n        nextValue = maxValue\n      }\n    }\n    setInputValue(nextValue)\n    onChange?.(nextValue)\n  }\n\n  const offsetFn = (direction: number) => () => {\n    const localValue = currentValue ?? 0\n    const newValue =\n      localValue % step === 0 ? localValue + step * direction : localValue\n    const roundedValue = roundStep(newValue, step, direction)\n\n    setValue(roundedValue)\n  }\n\n  const handleChange: ChangeEventHandler<HTMLInputElement> = event => {\n    setValue(\n      event.currentTarget.value ? Number(event.currentTarget.value) : undefined,\n      false,\n    )\n  }\n\n  const handleOnBlur: FocusEventHandler<HTMLInputElement> = event => {\n    if (currentValue) {\n      const boundedValue = bounded(\n        currentValue,\n        minValue ?? currentValue,\n        maxValue ?? currentValue,\n      )\n\n      if (maxValue && currentValue > maxValue) onMaxCrossed?.()\n      if (minValue && currentValue < minValue) onMinCrossed?.()\n\n      setValue(boundedValue)\n\n      onBlur?.(event)\n    }\n  }\n\n  const onKeyDown: KeyboardEventHandler = event => {\n    if (event.key === 'ArrowUp') {\n      event.stopPropagation()\n      event.preventDefault()\n\n      const direction = 1\n      const localValue = currentValue ?? 0\n\n      const newValue =\n        localValue % step === 0 ? localValue + step * direction : localValue\n      const roundedValue = roundStep(newValue, step, direction)\n\n      if (maxValue === undefined) {\n        setValue(roundedValue)\n\n        return\n      }\n\n      setValue(Math.min(roundedValue, maxValue))\n    }\n\n    if (event.key === 'ArrowDown') {\n      event.stopPropagation()\n      event.preventDefault()\n\n      const direction = -1\n      const localValue = currentValue ?? 0\n\n      const newValue =\n        localValue % step === 0 ? localValue + step * direction : localValue\n      const roundedValue = roundStep(newValue, step, direction)\n\n      setValue(Math.max(roundedValue, minValue))\n    }\n  }\n\n  const isMinusDisabled = useMemo(() => {\n    if (disabled) return true\n    if (currentValue === undefined) return false\n    if (getMinusRoundedValue(currentValue, step) < minValue) {\n      return true\n    }\n\n    return disabled\n  }, [currentValue, disabled, minValue, step])\n\n  const isPlusDisabled = useMemo(() => {\n    if (disabled) return true\n    if (currentValue === undefined) return false\n    if (maxValue && getPlusRoundedValue(currentValue, step) > maxValue) {\n      return true\n    }\n\n    return disabled\n  }, [currentValue, disabled, maxValue, step])\n\n  const inputWidth = useMemo(() => {\n    if (placeholder && currentValue === undefined) {\n      return placeholder.length * 12\n    }\n\n    if (currentValue !== undefined) {\n      return currentValue.toString().length * 16\n    }\n\n    return BASE_INPUT_WIDTH\n  }, [currentValue, placeholder])\n\n  return (\n    <Stack gap={1}>\n      {label ? (\n        <Text variant=\"bodyStrong\" as=\"label\" htmlFor={id || uniqueId}>\n          {label}\n        </Text>\n      ) : null}\n      <Stack gap={0.5}>\n        <StyledContainer\n          aria-disabled={disabled}\n          data-error={!!error}\n          size={size}\n          className={className}\n          data-testid={dataTestId}\n        >\n          <Tooltip text={isMinusDisabled && disabledTooltip}>\n            <StyledSelectButton\n              onClick={offsetFn(-1)}\n              disabled={isMinusDisabled}\n              aria-label=\"Minus\"\n              type=\"button\"\n              variant=\"ghost\"\n              sentiment=\"primary\"\n              size=\"small\"\n            >\n              <MinusIcon\n                size={iconSizes[size]}\n                sentiment=\"primary\"\n                disabled={isMinusDisabled}\n              />\n            </StyledSelectButton>\n          </Tooltip>\n\n          <StyledCenterBox\n            size={size}\n            onClick={() => {\n              if (inputRef?.current) {\n                inputRef.current.focus()\n              }\n            }}\n            aria-live=\"assertive\"\n            role=\"status\"\n          >\n            <StyledInput\n              disabled={disabled}\n              name={name}\n              onBlur={handleOnBlur}\n              onChange={handleChange}\n              onFocus={onFocus}\n              onKeyDown={onKeyDown}\n              ref={inputRef}\n              style={{\n                width: inputWidth,\n              }}\n              value={currentValue !== undefined ? currentValue.toString() : ''} // A dom element can only have string attributes.\n              type=\"number\"\n              id={id || uniqueId}\n              aria-label={!label && !ariaLabel ? 'Number Input' : ariaLabel}\n              aria-describedby={ariaDescribedBy}\n              placeholder={placeholder}\n            />\n            {currentValue !== undefined ? (\n              <StyledText disabled={disabled}>{text}</StyledText>\n            ) : null}\n          </StyledCenterBox>\n\n          <Tooltip text={isPlusDisabled && disabledTooltip}>\n            <StyledSelectButton\n              onClick={offsetFn(1)}\n              disabled={isPlusDisabled}\n              aria-label=\"Plus\"\n              type=\"button\"\n              variant=\"ghost\"\n              sentiment=\"primary\"\n              size=\"small\"\n            >\n              <PlusIcon\n                size={iconSizes[size]}\n                sentiment=\"primary\"\n                disabled={isPlusDisabled}\n              />\n            </StyledSelectButton>\n          </Tooltip>\n        </StyledContainer>\n        {typeof error === 'string' ? (\n          <Text\n            as=\"span\"\n            variant=\"bodySmall\"\n            sentiment=\"danger\"\n            prominence=\"weak\"\n          >\n            {error}\n          </Text>\n        ) : null}\n      </Stack>\n    </Stack>\n  )\n}\n"]} */"));
|
|
29
29
|
const StyledCenterBox = /* @__PURE__ */ _styled("div", process.env.NODE_ENV === "production" ? {
|
|
30
30
|
shouldForwardProp: (prop) => !["size"].includes(prop),
|
|
31
31
|
target: "exvap483"
|
|
@@ -37,7 +37,7 @@ const StyledCenterBox = /* @__PURE__ */ _styled("div", process.env.NODE_ENV ===
|
|
|
37
37
|
size
|
|
38
38
|
}) => size === "small" ? "24px" : "32px", ";align-items:center;outline:none;justify-content:center;border-radius:", ({
|
|
39
39
|
theme
|
|
40
|
-
}) => theme.radii.default, ";border:1px solid transparent;max-width:100%;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/NumberInput/index.tsx"],"names":[],"mappings":"AA6CgC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/NumberInput/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { Icon } from '@ultraviolet/icons/legacy'\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"]} */"));
|
|
40
|
+
}) => theme.radii.default, ";border:1px solid transparent;max-width:100%;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/NumberInput/index.tsx"],"names":[],"mappings":"AA6CgC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/NumberInput/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { MinusIcon, PlusIcon } from '@ultraviolet/icons'\nimport type {\n  ChangeEventHandler,\n  FocusEventHandler,\n  InputHTMLAttributes,\n  KeyboardEventHandler,\n  MutableRefObject,\n} from 'react'\nimport { useId, useMemo, useRef, useState } from 'react'\nimport { Button } from '../Button'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\nimport { Tooltip } from '../Tooltip'\nimport {\n  bounded,\n  getMinusRoundedValue,\n  getPlusRoundedValue,\n  roundStep,\n} from './helpers'\n\nconst containerSizes = {\n  large: 48,\n  medium: 40,\n  small: 32,\n}\n\ntype ContainerSizesType = keyof typeof containerSizes\n\nconst iconSizes = {\n  large: 26,\n  medium: 24,\n  small: 22,\n}\n\nconst BASE_INPUT_WIDTH = 34\n\nconst StyledSelectButton = styled(Button)`\n  margin: 0 ${({ theme }) => theme.space['1']};\n  width: 32px;\n  height: 32px;\n`\n\nconst StyledCenterBox = styled('div', {\n  shouldForwardProp: prop => !['size'].includes(prop),\n})<{ size: ContainerSizesType }>`\n  display: flex;\n  flex: 1;\n  flex-direction: row;\n  height: ${({ size }) => (size === 'small' ? '24px' : '32px')};\n  align-items: center;\n  outline: none;\n  justify-content: center;\n  border-radius: ${({ theme }) => theme.radii.default};\n  border: 1px solid transparent;\n  max-width: 100%;\n`\n\nconst StyledInput = styled.input`\n  color: ${({ theme }) => theme.colors.neutral.text};\n  background-color: transparent;\n  font-size: ${({ theme }) => theme.typography.bodyStrong.fontSize};\n  border: none;\n  outline: none;\n  position: relative;\n  margin-right: ${({ theme }) => theme.space['0.5']};\n  max-width: 100%;\n  font-weight: ${({ theme }) => theme.typography.bodyStrong.weight};\n  text-align: center;\n\n  &::-webkit-outer-spin-button,\n  &::-webkit-inner-spin-button {\n    -webkit-appearance: none;\n    margin: 0;\n  }\n\n  ::placeholder {\n    color: ${({ theme }) => theme.colors.neutral.textWeak};\n  }\n\n  -moz-appearance: textfield;\n\n  &[disabled] {\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n    cursor: not-allowed;\n  }\n`\n\nconst StyledText = styled('span', {\n  shouldForwardProp: prop => !['disabled'].includes(prop),\n})<{ disabled: boolean }>`\n  color: ${({ theme, disabled }) =>\n    disabled ? theme.colors.neutral.textDisabled : theme.colors.neutral.text};\n  user-select: none;\n  margin-right: ${({ theme }) => theme.space['1']};\n`\n\nconst StyledContainer = styled('div', {\n  shouldForwardProp: prop => !['size'].includes(prop),\n})<{ size: ContainerSizesType }>`\n  background-color: ${({ theme }) => theme.colors.neutral.background};\n  display: flex;\n  flex-direction: row;\n  align-items: center;\n  align-self: stretch;\n  font-weight: 500;\n  height: ${({ size }) => containerSizes[size]}px;\n  border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  border-radius: ${({ theme }) => theme.radii.default};\n\n  &[data-error='true'] {\n    border: 1px solid ${({ theme }) => theme.colors.danger.border};\n  }\n\n  &[aria-disabled='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    cursor: not-allowed;\n  }\n\n  &:not([aria-disabled='true']) {\n    ${StyledCenterBox}:hover,\n    ${StyledCenterBox}:focus {\n      border: 1px solid ${({ theme }) => theme.colors.primary.borderHover};\n    }\n\n    ${StyledCenterBox}:focus-within {\n      box-shadow: ${({ theme }) => theme.shadows.focusPrimary};\n      border: 1px solid ${({ theme }) => theme.colors.primary.borderHover};\n    }\n  }\n`\n\ntype NumberInputProps = {\n  disabled?: boolean\n  maxValue?: number\n  minValue?: number\n  name?: string\n  onChange?: (input: number | undefined) => void\n  onMaxCrossed?: () => void\n  onMinCrossed?: () => void\n  size?: ContainerSizesType\n  /**\n   * Define how much will stepper increase / decrease each time you click on + / - button.\n   */\n  step?: number\n  /**\n   * Text displayed into component at the right of number value.\n   */\n  text?: string\n  defaultValue?: number\n  value?: number | null\n  disabledTooltip?: string\n  className?: string\n  'data-testid'?: string\n  label?: string\n  'aria-label'?: string\n  'aria-describedby'?: string\n  id?: string\n  placeholder?: string\n  error?: string | boolean\n} & Omit<\n  InputHTMLAttributes<HTMLInputElement>,\n  'size' | 'onChange' | 'value' | 'defaultValue'\n>\n\n/**\n * @deprecated This component is deprecated. Please use `NumberInputV2` instead.\n */\nexport const NumberInput = ({\n  disabled = false,\n  maxValue,\n  minValue = 0,\n  name = 'numberinput',\n  onChange,\n  onFocus,\n  onBlur,\n  onMaxCrossed,\n  onMinCrossed,\n  size = 'large',\n  step = 1,\n  text,\n  defaultValue,\n  value,\n  disabledTooltip,\n  className,\n  label,\n  id,\n  placeholder,\n  error,\n  'aria-label': ariaLabel,\n  'aria-describedby': ariaDescribedBy,\n  'data-testid': dataTestId,\n}: NumberInputProps) => {\n  const inputRef =\n    useRef<HTMLInputElement>() as MutableRefObject<HTMLInputElement>\n\n  const uniqueId = useId()\n\n  // local state used if component is not controlled (no value prop provided)\n  const [inputValue, setInputValue] = useState<number | undefined>(() => {\n    if (defaultValue && minValue && defaultValue < minValue) {\n      return minValue\n    }\n    if (defaultValue && maxValue && defaultValue > maxValue) {\n      return maxValue\n    }\n\n    return defaultValue\n  })\n\n  const currentValue =\n    value !== undefined && value !== null ? value : inputValue\n\n  const setValue = (\n    newValue: number | undefined,\n    /**\n     * If true, will check if newValue is between minValue and maxValue and set it to minValue or maxValue if it's not.\n     */\n    hasMinMaxVerification = true,\n  ) => {\n    let nextValue = newValue\n    if (value === undefined && hasMinMaxVerification) {\n      if (newValue !== undefined && newValue < minValue) {\n        nextValue = minValue\n      }\n\n      if (\n        newValue !== undefined &&\n        maxValue !== undefined &&\n        newValue > maxValue\n      ) {\n        nextValue = maxValue\n      }\n    }\n    setInputValue(nextValue)\n    onChange?.(nextValue)\n  }\n\n  const offsetFn = (direction: number) => () => {\n    const localValue = currentValue ?? 0\n    const newValue =\n      localValue % step === 0 ? localValue + step * direction : localValue\n    const roundedValue = roundStep(newValue, step, direction)\n\n    setValue(roundedValue)\n  }\n\n  const handleChange: ChangeEventHandler<HTMLInputElement> = event => {\n    setValue(\n      event.currentTarget.value ? Number(event.currentTarget.value) : undefined,\n      false,\n    )\n  }\n\n  const handleOnBlur: FocusEventHandler<HTMLInputElement> = event => {\n    if (currentValue) {\n      const boundedValue = bounded(\n        currentValue,\n        minValue ?? currentValue,\n        maxValue ?? currentValue,\n      )\n\n      if (maxValue && currentValue > maxValue) onMaxCrossed?.()\n      if (minValue && currentValue < minValue) onMinCrossed?.()\n\n      setValue(boundedValue)\n\n      onBlur?.(event)\n    }\n  }\n\n  const onKeyDown: KeyboardEventHandler = event => {\n    if (event.key === 'ArrowUp') {\n      event.stopPropagation()\n      event.preventDefault()\n\n      const direction = 1\n      const localValue = currentValue ?? 0\n\n      const newValue =\n        localValue % step === 0 ? localValue + step * direction : localValue\n      const roundedValue = roundStep(newValue, step, direction)\n\n      if (maxValue === undefined) {\n        setValue(roundedValue)\n\n        return\n      }\n\n      setValue(Math.min(roundedValue, maxValue))\n    }\n\n    if (event.key === 'ArrowDown') {\n      event.stopPropagation()\n      event.preventDefault()\n\n      const direction = -1\n      const localValue = currentValue ?? 0\n\n      const newValue =\n        localValue % step === 0 ? localValue + step * direction : localValue\n      const roundedValue = roundStep(newValue, step, direction)\n\n      setValue(Math.max(roundedValue, minValue))\n    }\n  }\n\n  const isMinusDisabled = useMemo(() => {\n    if (disabled) return true\n    if (currentValue === undefined) return false\n    if (getMinusRoundedValue(currentValue, step) < minValue) {\n      return true\n    }\n\n    return disabled\n  }, [currentValue, disabled, minValue, step])\n\n  const isPlusDisabled = useMemo(() => {\n    if (disabled) return true\n    if (currentValue === undefined) return false\n    if (maxValue && getPlusRoundedValue(currentValue, step) > maxValue) {\n      return true\n    }\n\n    return disabled\n  }, [currentValue, disabled, maxValue, step])\n\n  const inputWidth = useMemo(() => {\n    if (placeholder && currentValue === undefined) {\n      return placeholder.length * 12\n    }\n\n    if (currentValue !== undefined) {\n      return currentValue.toString().length * 16\n    }\n\n    return BASE_INPUT_WIDTH\n  }, [currentValue, placeholder])\n\n  return (\n    <Stack gap={1}>\n      {label ? (\n        <Text variant=\"bodyStrong\" as=\"label\" htmlFor={id || uniqueId}>\n          {label}\n        </Text>\n      ) : null}\n      <Stack gap={0.5}>\n        <StyledContainer\n          aria-disabled={disabled}\n          data-error={!!error}\n          size={size}\n          className={className}\n          data-testid={dataTestId}\n        >\n          <Tooltip text={isMinusDisabled && disabledTooltip}>\n            <StyledSelectButton\n              onClick={offsetFn(-1)}\n              disabled={isMinusDisabled}\n              aria-label=\"Minus\"\n              type=\"button\"\n              variant=\"ghost\"\n              sentiment=\"primary\"\n              size=\"small\"\n            >\n              <MinusIcon\n                size={iconSizes[size]}\n                sentiment=\"primary\"\n                disabled={isMinusDisabled}\n              />\n            </StyledSelectButton>\n          </Tooltip>\n\n          <StyledCenterBox\n            size={size}\n            onClick={() => {\n              if (inputRef?.current) {\n                inputRef.current.focus()\n              }\n            }}\n            aria-live=\"assertive\"\n            role=\"status\"\n          >\n            <StyledInput\n              disabled={disabled}\n              name={name}\n              onBlur={handleOnBlur}\n              onChange={handleChange}\n              onFocus={onFocus}\n              onKeyDown={onKeyDown}\n              ref={inputRef}\n              style={{\n                width: inputWidth,\n              }}\n              value={currentValue !== undefined ? currentValue.toString() : ''} // A dom element can only have string attributes.\n              type=\"number\"\n              id={id || uniqueId}\n              aria-label={!label && !ariaLabel ? 'Number Input' : ariaLabel}\n              aria-describedby={ariaDescribedBy}\n              placeholder={placeholder}\n            />\n            {currentValue !== undefined ? (\n              <StyledText disabled={disabled}>{text}</StyledText>\n            ) : null}\n          </StyledCenterBox>\n\n          <Tooltip text={isPlusDisabled && disabledTooltip}>\n            <StyledSelectButton\n              onClick={offsetFn(1)}\n              disabled={isPlusDisabled}\n              aria-label=\"Plus\"\n              type=\"button\"\n              variant=\"ghost\"\n              sentiment=\"primary\"\n              size=\"small\"\n            >\n              <PlusIcon\n                size={iconSizes[size]}\n                sentiment=\"primary\"\n                disabled={isPlusDisabled}\n              />\n            </StyledSelectButton>\n          </Tooltip>\n        </StyledContainer>\n        {typeof error === 'string' ? (\n          <Text\n            as=\"span\"\n            variant=\"bodySmall\"\n            sentiment=\"danger\"\n            prominence=\"weak\"\n          >\n            {error}\n          </Text>\n        ) : null}\n      </Stack>\n    </Stack>\n  )\n}\n"]} */"));
|
|
41
41
|
const StyledInput = /* @__PURE__ */ _styled("input", process.env.NODE_ENV === "production" ? {
|
|
42
42
|
target: "exvap482"
|
|
43
43
|
} : {
|
|
@@ -55,7 +55,7 @@ const StyledInput = /* @__PURE__ */ _styled("input", process.env.NODE_ENV === "p
|
|
|
55
55
|
theme
|
|
56
56
|
}) => theme.colors.neutral.textWeak, ";}-moz-appearance:textfield;&[disabled]{color:", ({
|
|
57
57
|
theme
|
|
58
|
-
}) => theme.colors.neutral.textDisabled, ";cursor:not-allowed;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/NumberInput/index.tsx"],"names":[],"mappings":"AA0DgC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/NumberInput/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { Icon } from '@ultraviolet/icons/legacy'\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"]} */"));
|
|
58
|
+
}) => theme.colors.neutral.textDisabled, ";cursor:not-allowed;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/NumberInput/index.tsx"],"names":[],"mappings":"AA0DgC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/NumberInput/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { MinusIcon, PlusIcon } from '@ultraviolet/icons'\nimport type {\n  ChangeEventHandler,\n  FocusEventHandler,\n  InputHTMLAttributes,\n  KeyboardEventHandler,\n  MutableRefObject,\n} from 'react'\nimport { useId, useMemo, useRef, useState } from 'react'\nimport { Button } from '../Button'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\nimport { Tooltip } from '../Tooltip'\nimport {\n  bounded,\n  getMinusRoundedValue,\n  getPlusRoundedValue,\n  roundStep,\n} from './helpers'\n\nconst containerSizes = {\n  large: 48,\n  medium: 40,\n  small: 32,\n}\n\ntype ContainerSizesType = keyof typeof containerSizes\n\nconst iconSizes = {\n  large: 26,\n  medium: 24,\n  small: 22,\n}\n\nconst BASE_INPUT_WIDTH = 34\n\nconst StyledSelectButton = styled(Button)`\n  margin: 0 ${({ theme }) => theme.space['1']};\n  width: 32px;\n  height: 32px;\n`\n\nconst StyledCenterBox = styled('div', {\n  shouldForwardProp: prop => !['size'].includes(prop),\n})<{ size: ContainerSizesType }>`\n  display: flex;\n  flex: 1;\n  flex-direction: row;\n  height: ${({ size }) => (size === 'small' ? '24px' : '32px')};\n  align-items: center;\n  outline: none;\n  justify-content: center;\n  border-radius: ${({ theme }) => theme.radii.default};\n  border: 1px solid transparent;\n  max-width: 100%;\n`\n\nconst StyledInput = styled.input`\n  color: ${({ theme }) => theme.colors.neutral.text};\n  background-color: transparent;\n  font-size: ${({ theme }) => theme.typography.bodyStrong.fontSize};\n  border: none;\n  outline: none;\n  position: relative;\n  margin-right: ${({ theme }) => theme.space['0.5']};\n  max-width: 100%;\n  font-weight: ${({ theme }) => theme.typography.bodyStrong.weight};\n  text-align: center;\n\n  &::-webkit-outer-spin-button,\n  &::-webkit-inner-spin-button {\n    -webkit-appearance: none;\n    margin: 0;\n  }\n\n  ::placeholder {\n    color: ${({ theme }) => theme.colors.neutral.textWeak};\n  }\n\n  -moz-appearance: textfield;\n\n  &[disabled] {\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n    cursor: not-allowed;\n  }\n`\n\nconst StyledText = styled('span', {\n  shouldForwardProp: prop => !['disabled'].includes(prop),\n})<{ disabled: boolean }>`\n  color: ${({ theme, disabled }) =>\n    disabled ? theme.colors.neutral.textDisabled : theme.colors.neutral.text};\n  user-select: none;\n  margin-right: ${({ theme }) => theme.space['1']};\n`\n\nconst StyledContainer = styled('div', {\n  shouldForwardProp: prop => !['size'].includes(prop),\n})<{ size: ContainerSizesType }>`\n  background-color: ${({ theme }) => theme.colors.neutral.background};\n  display: flex;\n  flex-direction: row;\n  align-items: center;\n  align-self: stretch;\n  font-weight: 500;\n  height: ${({ size }) => containerSizes[size]}px;\n  border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  border-radius: ${({ theme }) => theme.radii.default};\n\n  &[data-error='true'] {\n    border: 1px solid ${({ theme }) => theme.colors.danger.border};\n  }\n\n  &[aria-disabled='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    cursor: not-allowed;\n  }\n\n  &:not([aria-disabled='true']) {\n    ${StyledCenterBox}:hover,\n    ${StyledCenterBox}:focus {\n      border: 1px solid ${({ theme }) => theme.colors.primary.borderHover};\n    }\n\n    ${StyledCenterBox}:focus-within {\n      box-shadow: ${({ theme }) => theme.shadows.focusPrimary};\n      border: 1px solid ${({ theme }) => theme.colors.primary.borderHover};\n    }\n  }\n`\n\ntype NumberInputProps = {\n  disabled?: boolean\n  maxValue?: number\n  minValue?: number\n  name?: string\n  onChange?: (input: number | undefined) => void\n  onMaxCrossed?: () => void\n  onMinCrossed?: () => void\n  size?: ContainerSizesType\n  /**\n   * Define how much will stepper increase / decrease each time you click on + / - button.\n   */\n  step?: number\n  /**\n   * Text displayed into component at the right of number value.\n   */\n  text?: string\n  defaultValue?: number\n  value?: number | null\n  disabledTooltip?: string\n  className?: string\n  'data-testid'?: string\n  label?: string\n  'aria-label'?: string\n  'aria-describedby'?: string\n  id?: string\n  placeholder?: string\n  error?: string | boolean\n} & Omit<\n  InputHTMLAttributes<HTMLInputElement>,\n  'size' | 'onChange' | 'value' | 'defaultValue'\n>\n\n/**\n * @deprecated This component is deprecated. Please use `NumberInputV2` instead.\n */\nexport const NumberInput = ({\n  disabled = false,\n  maxValue,\n  minValue = 0,\n  name = 'numberinput',\n  onChange,\n  onFocus,\n  onBlur,\n  onMaxCrossed,\n  onMinCrossed,\n  size = 'large',\n  step = 1,\n  text,\n  defaultValue,\n  value,\n  disabledTooltip,\n  className,\n  label,\n  id,\n  placeholder,\n  error,\n  'aria-label': ariaLabel,\n  'aria-describedby': ariaDescribedBy,\n  'data-testid': dataTestId,\n}: NumberInputProps) => {\n  const inputRef =\n    useRef<HTMLInputElement>() as MutableRefObject<HTMLInputElement>\n\n  const uniqueId = useId()\n\n  // local state used if component is not controlled (no value prop provided)\n  const [inputValue, setInputValue] = useState<number | undefined>(() => {\n    if (defaultValue && minValue && defaultValue < minValue) {\n      return minValue\n    }\n    if (defaultValue && maxValue && defaultValue > maxValue) {\n      return maxValue\n    }\n\n    return defaultValue\n  })\n\n  const currentValue =\n    value !== undefined && value !== null ? value : inputValue\n\n  const setValue = (\n    newValue: number | undefined,\n    /**\n     * If true, will check if newValue is between minValue and maxValue and set it to minValue or maxValue if it's not.\n     */\n    hasMinMaxVerification = true,\n  ) => {\n    let nextValue = newValue\n    if (value === undefined && hasMinMaxVerification) {\n      if (newValue !== undefined && newValue < minValue) {\n        nextValue = minValue\n      }\n\n      if (\n        newValue !== undefined &&\n        maxValue !== undefined &&\n        newValue > maxValue\n      ) {\n        nextValue = maxValue\n      }\n    }\n    setInputValue(nextValue)\n    onChange?.(nextValue)\n  }\n\n  const offsetFn = (direction: number) => () => {\n    const localValue = currentValue ?? 0\n    const newValue =\n      localValue % step === 0 ? localValue + step * direction : localValue\n    const roundedValue = roundStep(newValue, step, direction)\n\n    setValue(roundedValue)\n  }\n\n  const handleChange: ChangeEventHandler<HTMLInputElement> = event => {\n    setValue(\n      event.currentTarget.value ? Number(event.currentTarget.value) : undefined,\n      false,\n    )\n  }\n\n  const handleOnBlur: FocusEventHandler<HTMLInputElement> = event => {\n    if (currentValue) {\n      const boundedValue = bounded(\n        currentValue,\n        minValue ?? currentValue,\n        maxValue ?? currentValue,\n      )\n\n      if (maxValue && currentValue > maxValue) onMaxCrossed?.()\n      if (minValue && currentValue < minValue) onMinCrossed?.()\n\n      setValue(boundedValue)\n\n      onBlur?.(event)\n    }\n  }\n\n  const onKeyDown: KeyboardEventHandler = event => {\n    if (event.key === 'ArrowUp') {\n      event.stopPropagation()\n      event.preventDefault()\n\n      const direction = 1\n      const localValue = currentValue ?? 0\n\n      const newValue =\n        localValue % step === 0 ? localValue + step * direction : localValue\n      const roundedValue = roundStep(newValue, step, direction)\n\n      if (maxValue === undefined) {\n        setValue(roundedValue)\n\n        return\n      }\n\n      setValue(Math.min(roundedValue, maxValue))\n    }\n\n    if (event.key === 'ArrowDown') {\n      event.stopPropagation()\n      event.preventDefault()\n\n      const direction = -1\n      const localValue = currentValue ?? 0\n\n      const newValue =\n        localValue % step === 0 ? localValue + step * direction : localValue\n      const roundedValue = roundStep(newValue, step, direction)\n\n      setValue(Math.max(roundedValue, minValue))\n    }\n  }\n\n  const isMinusDisabled = useMemo(() => {\n    if (disabled) return true\n    if (currentValue === undefined) return false\n    if (getMinusRoundedValue(currentValue, step) < minValue) {\n      return true\n    }\n\n    return disabled\n  }, [currentValue, disabled, minValue, step])\n\n  const isPlusDisabled = useMemo(() => {\n    if (disabled) return true\n    if (currentValue === undefined) return false\n    if (maxValue && getPlusRoundedValue(currentValue, step) > maxValue) {\n      return true\n    }\n\n    return disabled\n  }, [currentValue, disabled, maxValue, step])\n\n  const inputWidth = useMemo(() => {\n    if (placeholder && currentValue === undefined) {\n      return placeholder.length * 12\n    }\n\n    if (currentValue !== undefined) {\n      return currentValue.toString().length * 16\n    }\n\n    return BASE_INPUT_WIDTH\n  }, [currentValue, placeholder])\n\n  return (\n    <Stack gap={1}>\n      {label ? (\n        <Text variant=\"bodyStrong\" as=\"label\" htmlFor={id || uniqueId}>\n          {label}\n        </Text>\n      ) : null}\n      <Stack gap={0.5}>\n        <StyledContainer\n          aria-disabled={disabled}\n          data-error={!!error}\n          size={size}\n          className={className}\n          data-testid={dataTestId}\n        >\n          <Tooltip text={isMinusDisabled && disabledTooltip}>\n            <StyledSelectButton\n              onClick={offsetFn(-1)}\n              disabled={isMinusDisabled}\n              aria-label=\"Minus\"\n              type=\"button\"\n              variant=\"ghost\"\n              sentiment=\"primary\"\n              size=\"small\"\n            >\n              <MinusIcon\n                size={iconSizes[size]}\n                sentiment=\"primary\"\n                disabled={isMinusDisabled}\n              />\n            </StyledSelectButton>\n          </Tooltip>\n\n          <StyledCenterBox\n            size={size}\n            onClick={() => {\n              if (inputRef?.current) {\n                inputRef.current.focus()\n              }\n            }}\n            aria-live=\"assertive\"\n            role=\"status\"\n          >\n            <StyledInput\n              disabled={disabled}\n              name={name}\n              onBlur={handleOnBlur}\n              onChange={handleChange}\n              onFocus={onFocus}\n              onKeyDown={onKeyDown}\n              ref={inputRef}\n              style={{\n                width: inputWidth,\n              }}\n              value={currentValue !== undefined ? currentValue.toString() : ''} // A dom element can only have string attributes.\n              type=\"number\"\n              id={id || uniqueId}\n              aria-label={!label && !ariaLabel ? 'Number Input' : ariaLabel}\n              aria-describedby={ariaDescribedBy}\n              placeholder={placeholder}\n            />\n            {currentValue !== undefined ? (\n              <StyledText disabled={disabled}>{text}</StyledText>\n            ) : null}\n          </StyledCenterBox>\n\n          <Tooltip text={isPlusDisabled && disabledTooltip}>\n            <StyledSelectButton\n              onClick={offsetFn(1)}\n              disabled={isPlusDisabled}\n              aria-label=\"Plus\"\n              type=\"button\"\n              variant=\"ghost\"\n              sentiment=\"primary\"\n              size=\"small\"\n            >\n              <PlusIcon\n                size={iconSizes[size]}\n                sentiment=\"primary\"\n                disabled={isPlusDisabled}\n              />\n            </StyledSelectButton>\n          </Tooltip>\n        </StyledContainer>\n        {typeof error === 'string' ? (\n          <Text\n            as=\"span\"\n            variant=\"bodySmall\"\n            sentiment=\"danger\"\n            prominence=\"weak\"\n          >\n            {error}\n          </Text>\n        ) : null}\n      </Stack>\n    </Stack>\n  )\n}\n"]} */"));
|
|
59
59
|
const StyledText = /* @__PURE__ */ _styled("span", process.env.NODE_ENV === "production" ? {
|
|
60
60
|
shouldForwardProp: (prop) => !["disabled"].includes(prop),
|
|
61
61
|
target: "exvap481"
|
|
@@ -68,7 +68,7 @@ const StyledText = /* @__PURE__ */ _styled("span", process.env.NODE_ENV === "pro
|
|
|
68
68
|
disabled
|
|
69
69
|
}) => disabled ? theme.colors.neutral.textDisabled : theme.colors.neutral.text, ";user-select:none;margin-right:", ({
|
|
70
70
|
theme
|
|
71
|
-
}) => theme.space["1"], ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/NumberInput/index.tsx"],"names":[],"mappings":"AA0FyB","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/NumberInput/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { Icon } from '@ultraviolet/icons/legacy'\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"]} */"));
|
|
71
|
+
}) => theme.space["1"], ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/NumberInput/index.tsx"],"names":[],"mappings":"AA0FyB","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/NumberInput/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { MinusIcon, PlusIcon } from '@ultraviolet/icons'\nimport type {\n  ChangeEventHandler,\n  FocusEventHandler,\n  InputHTMLAttributes,\n  KeyboardEventHandler,\n  MutableRefObject,\n} from 'react'\nimport { useId, useMemo, useRef, useState } from 'react'\nimport { Button } from '../Button'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\nimport { Tooltip } from '../Tooltip'\nimport {\n  bounded,\n  getMinusRoundedValue,\n  getPlusRoundedValue,\n  roundStep,\n} from './helpers'\n\nconst containerSizes = {\n  large: 48,\n  medium: 40,\n  small: 32,\n}\n\ntype ContainerSizesType = keyof typeof containerSizes\n\nconst iconSizes = {\n  large: 26,\n  medium: 24,\n  small: 22,\n}\n\nconst BASE_INPUT_WIDTH = 34\n\nconst StyledSelectButton = styled(Button)`\n  margin: 0 ${({ theme }) => theme.space['1']};\n  width: 32px;\n  height: 32px;\n`\n\nconst StyledCenterBox = styled('div', {\n  shouldForwardProp: prop => !['size'].includes(prop),\n})<{ size: ContainerSizesType }>`\n  display: flex;\n  flex: 1;\n  flex-direction: row;\n  height: ${({ size }) => (size === 'small' ? '24px' : '32px')};\n  align-items: center;\n  outline: none;\n  justify-content: center;\n  border-radius: ${({ theme }) => theme.radii.default};\n  border: 1px solid transparent;\n  max-width: 100%;\n`\n\nconst StyledInput = styled.input`\n  color: ${({ theme }) => theme.colors.neutral.text};\n  background-color: transparent;\n  font-size: ${({ theme }) => theme.typography.bodyStrong.fontSize};\n  border: none;\n  outline: none;\n  position: relative;\n  margin-right: ${({ theme }) => theme.space['0.5']};\n  max-width: 100%;\n  font-weight: ${({ theme }) => theme.typography.bodyStrong.weight};\n  text-align: center;\n\n  &::-webkit-outer-spin-button,\n  &::-webkit-inner-spin-button {\n    -webkit-appearance: none;\n    margin: 0;\n  }\n\n  ::placeholder {\n    color: ${({ theme }) => theme.colors.neutral.textWeak};\n  }\n\n  -moz-appearance: textfield;\n\n  &[disabled] {\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n    cursor: not-allowed;\n  }\n`\n\nconst StyledText = styled('span', {\n  shouldForwardProp: prop => !['disabled'].includes(prop),\n})<{ disabled: boolean }>`\n  color: ${({ theme, disabled }) =>\n    disabled ? theme.colors.neutral.textDisabled : theme.colors.neutral.text};\n  user-select: none;\n  margin-right: ${({ theme }) => theme.space['1']};\n`\n\nconst StyledContainer = styled('div', {\n  shouldForwardProp: prop => !['size'].includes(prop),\n})<{ size: ContainerSizesType }>`\n  background-color: ${({ theme }) => theme.colors.neutral.background};\n  display: flex;\n  flex-direction: row;\n  align-items: center;\n  align-self: stretch;\n  font-weight: 500;\n  height: ${({ size }) => containerSizes[size]}px;\n  border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  border-radius: ${({ theme }) => theme.radii.default};\n\n  &[data-error='true'] {\n    border: 1px solid ${({ theme }) => theme.colors.danger.border};\n  }\n\n  &[aria-disabled='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    cursor: not-allowed;\n  }\n\n  &:not([aria-disabled='true']) {\n    ${StyledCenterBox}:hover,\n    ${StyledCenterBox}:focus {\n      border: 1px solid ${({ theme }) => theme.colors.primary.borderHover};\n    }\n\n    ${StyledCenterBox}:focus-within {\n      box-shadow: ${({ theme }) => theme.shadows.focusPrimary};\n      border: 1px solid ${({ theme }) => theme.colors.primary.borderHover};\n    }\n  }\n`\n\ntype NumberInputProps = {\n  disabled?: boolean\n  maxValue?: number\n  minValue?: number\n  name?: string\n  onChange?: (input: number | undefined) => void\n  onMaxCrossed?: () => void\n  onMinCrossed?: () => void\n  size?: ContainerSizesType\n  /**\n   * Define how much will stepper increase / decrease each time you click on + / - button.\n   */\n  step?: number\n  /**\n   * Text displayed into component at the right of number value.\n   */\n  text?: string\n  defaultValue?: number\n  value?: number | null\n  disabledTooltip?: string\n  className?: string\n  'data-testid'?: string\n  label?: string\n  'aria-label'?: string\n  'aria-describedby'?: string\n  id?: string\n  placeholder?: string\n  error?: string | boolean\n} & Omit<\n  InputHTMLAttributes<HTMLInputElement>,\n  'size' | 'onChange' | 'value' | 'defaultValue'\n>\n\n/**\n * @deprecated This component is deprecated. Please use `NumberInputV2` instead.\n */\nexport const NumberInput = ({\n  disabled = false,\n  maxValue,\n  minValue = 0,\n  name = 'numberinput',\n  onChange,\n  onFocus,\n  onBlur,\n  onMaxCrossed,\n  onMinCrossed,\n  size = 'large',\n  step = 1,\n  text,\n  defaultValue,\n  value,\n  disabledTooltip,\n  className,\n  label,\n  id,\n  placeholder,\n  error,\n  'aria-label': ariaLabel,\n  'aria-describedby': ariaDescribedBy,\n  'data-testid': dataTestId,\n}: NumberInputProps) => {\n  const inputRef =\n    useRef<HTMLInputElement>() as MutableRefObject<HTMLInputElement>\n\n  const uniqueId = useId()\n\n  // local state used if component is not controlled (no value prop provided)\n  const [inputValue, setInputValue] = useState<number | undefined>(() => {\n    if (defaultValue && minValue && defaultValue < minValue) {\n      return minValue\n    }\n    if (defaultValue && maxValue && defaultValue > maxValue) {\n      return maxValue\n    }\n\n    return defaultValue\n  })\n\n  const currentValue =\n    value !== undefined && value !== null ? value : inputValue\n\n  const setValue = (\n    newValue: number | undefined,\n    /**\n     * If true, will check if newValue is between minValue and maxValue and set it to minValue or maxValue if it's not.\n     */\n    hasMinMaxVerification = true,\n  ) => {\n    let nextValue = newValue\n    if (value === undefined && hasMinMaxVerification) {\n      if (newValue !== undefined && newValue < minValue) {\n        nextValue = minValue\n      }\n\n      if (\n        newValue !== undefined &&\n        maxValue !== undefined &&\n        newValue > maxValue\n      ) {\n        nextValue = maxValue\n      }\n    }\n    setInputValue(nextValue)\n    onChange?.(nextValue)\n  }\n\n  const offsetFn = (direction: number) => () => {\n    const localValue = currentValue ?? 0\n    const newValue =\n      localValue % step === 0 ? localValue + step * direction : localValue\n    const roundedValue = roundStep(newValue, step, direction)\n\n    setValue(roundedValue)\n  }\n\n  const handleChange: ChangeEventHandler<HTMLInputElement> = event => {\n    setValue(\n      event.currentTarget.value ? Number(event.currentTarget.value) : undefined,\n      false,\n    )\n  }\n\n  const handleOnBlur: FocusEventHandler<HTMLInputElement> = event => {\n    if (currentValue) {\n      const boundedValue = bounded(\n        currentValue,\n        minValue ?? currentValue,\n        maxValue ?? currentValue,\n      )\n\n      if (maxValue && currentValue > maxValue) onMaxCrossed?.()\n      if (minValue && currentValue < minValue) onMinCrossed?.()\n\n      setValue(boundedValue)\n\n      onBlur?.(event)\n    }\n  }\n\n  const onKeyDown: KeyboardEventHandler = event => {\n    if (event.key === 'ArrowUp') {\n      event.stopPropagation()\n      event.preventDefault()\n\n      const direction = 1\n      const localValue = currentValue ?? 0\n\n      const newValue =\n        localValue % step === 0 ? localValue + step * direction : localValue\n      const roundedValue = roundStep(newValue, step, direction)\n\n      if (maxValue === undefined) {\n        setValue(roundedValue)\n\n        return\n      }\n\n      setValue(Math.min(roundedValue, maxValue))\n    }\n\n    if (event.key === 'ArrowDown') {\n      event.stopPropagation()\n      event.preventDefault()\n\n      const direction = -1\n      const localValue = currentValue ?? 0\n\n      const newValue =\n        localValue % step === 0 ? localValue + step * direction : localValue\n      const roundedValue = roundStep(newValue, step, direction)\n\n      setValue(Math.max(roundedValue, minValue))\n    }\n  }\n\n  const isMinusDisabled = useMemo(() => {\n    if (disabled) return true\n    if (currentValue === undefined) return false\n    if (getMinusRoundedValue(currentValue, step) < minValue) {\n      return true\n    }\n\n    return disabled\n  }, [currentValue, disabled, minValue, step])\n\n  const isPlusDisabled = useMemo(() => {\n    if (disabled) return true\n    if (currentValue === undefined) return false\n    if (maxValue && getPlusRoundedValue(currentValue, step) > maxValue) {\n      return true\n    }\n\n    return disabled\n  }, [currentValue, disabled, maxValue, step])\n\n  const inputWidth = useMemo(() => {\n    if (placeholder && currentValue === undefined) {\n      return placeholder.length * 12\n    }\n\n    if (currentValue !== undefined) {\n      return currentValue.toString().length * 16\n    }\n\n    return BASE_INPUT_WIDTH\n  }, [currentValue, placeholder])\n\n  return (\n    <Stack gap={1}>\n      {label ? (\n        <Text variant=\"bodyStrong\" as=\"label\" htmlFor={id || uniqueId}>\n          {label}\n        </Text>\n      ) : null}\n      <Stack gap={0.5}>\n        <StyledContainer\n          aria-disabled={disabled}\n          data-error={!!error}\n          size={size}\n          className={className}\n          data-testid={dataTestId}\n        >\n          <Tooltip text={isMinusDisabled && disabledTooltip}>\n            <StyledSelectButton\n              onClick={offsetFn(-1)}\n              disabled={isMinusDisabled}\n              aria-label=\"Minus\"\n              type=\"button\"\n              variant=\"ghost\"\n              sentiment=\"primary\"\n              size=\"small\"\n            >\n              <MinusIcon\n                size={iconSizes[size]}\n                sentiment=\"primary\"\n                disabled={isMinusDisabled}\n              />\n            </StyledSelectButton>\n          </Tooltip>\n\n          <StyledCenterBox\n            size={size}\n            onClick={() => {\n              if (inputRef?.current) {\n                inputRef.current.focus()\n              }\n            }}\n            aria-live=\"assertive\"\n            role=\"status\"\n          >\n            <StyledInput\n              disabled={disabled}\n              name={name}\n              onBlur={handleOnBlur}\n              onChange={handleChange}\n              onFocus={onFocus}\n              onKeyDown={onKeyDown}\n              ref={inputRef}\n              style={{\n                width: inputWidth,\n              }}\n              value={currentValue !== undefined ? currentValue.toString() : ''} // A dom element can only have string attributes.\n              type=\"number\"\n              id={id || uniqueId}\n              aria-label={!label && !ariaLabel ? 'Number Input' : ariaLabel}\n              aria-describedby={ariaDescribedBy}\n              placeholder={placeholder}\n            />\n            {currentValue !== undefined ? (\n              <StyledText disabled={disabled}>{text}</StyledText>\n            ) : null}\n          </StyledCenterBox>\n\n          <Tooltip text={isPlusDisabled && disabledTooltip}>\n            <StyledSelectButton\n              onClick={offsetFn(1)}\n              disabled={isPlusDisabled}\n              aria-label=\"Plus\"\n              type=\"button\"\n              variant=\"ghost\"\n              sentiment=\"primary\"\n              size=\"small\"\n            >\n              <PlusIcon\n                size={iconSizes[size]}\n                sentiment=\"primary\"\n                disabled={isPlusDisabled}\n              />\n            </StyledSelectButton>\n          </Tooltip>\n        </StyledContainer>\n        {typeof error === 'string' ? (\n          <Text\n            as=\"span\"\n            variant=\"bodySmall\"\n            sentiment=\"danger\"\n            prominence=\"weak\"\n          >\n            {error}\n          </Text>\n        ) : null}\n      </Stack>\n    </Stack>\n  )\n}\n"]} */"));
|
|
72
72
|
const StyledContainer = /* @__PURE__ */ _styled("div", process.env.NODE_ENV === "production" ? {
|
|
73
73
|
shouldForwardProp: (prop) => !["size"].includes(prop),
|
|
74
74
|
target: "exvap480"
|
|
@@ -94,7 +94,7 @@ const StyledContainer = /* @__PURE__ */ _styled("div", process.env.NODE_ENV ===
|
|
|
94
94
|
theme
|
|
95
95
|
}) => theme.shadows.focusPrimary, ";border:1px solid ", ({
|
|
96
96
|
theme
|
|
97
|
-
}) => theme.colors.primary.borderHover, ";}}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/NumberInput/index.tsx"],"names":[],"mappings":"AAmGgC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/NumberInput/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { Icon } from '@ultraviolet/icons/legacy'\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"]} */"));
|
|
97
|
+
}) => theme.colors.primary.borderHover, ";}}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/NumberInput/index.tsx"],"names":[],"mappings":"AAmGgC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/NumberInput/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { MinusIcon, PlusIcon } from '@ultraviolet/icons'\nimport type {\n  ChangeEventHandler,\n  FocusEventHandler,\n  InputHTMLAttributes,\n  KeyboardEventHandler,\n  MutableRefObject,\n} from 'react'\nimport { useId, useMemo, useRef, useState } from 'react'\nimport { Button } from '../Button'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\nimport { Tooltip } from '../Tooltip'\nimport {\n  bounded,\n  getMinusRoundedValue,\n  getPlusRoundedValue,\n  roundStep,\n} from './helpers'\n\nconst containerSizes = {\n  large: 48,\n  medium: 40,\n  small: 32,\n}\n\ntype ContainerSizesType = keyof typeof containerSizes\n\nconst iconSizes = {\n  large: 26,\n  medium: 24,\n  small: 22,\n}\n\nconst BASE_INPUT_WIDTH = 34\n\nconst StyledSelectButton = styled(Button)`\n  margin: 0 ${({ theme }) => theme.space['1']};\n  width: 32px;\n  height: 32px;\n`\n\nconst StyledCenterBox = styled('div', {\n  shouldForwardProp: prop => !['size'].includes(prop),\n})<{ size: ContainerSizesType }>`\n  display: flex;\n  flex: 1;\n  flex-direction: row;\n  height: ${({ size }) => (size === 'small' ? '24px' : '32px')};\n  align-items: center;\n  outline: none;\n  justify-content: center;\n  border-radius: ${({ theme }) => theme.radii.default};\n  border: 1px solid transparent;\n  max-width: 100%;\n`\n\nconst StyledInput = styled.input`\n  color: ${({ theme }) => theme.colors.neutral.text};\n  background-color: transparent;\n  font-size: ${({ theme }) => theme.typography.bodyStrong.fontSize};\n  border: none;\n  outline: none;\n  position: relative;\n  margin-right: ${({ theme }) => theme.space['0.5']};\n  max-width: 100%;\n  font-weight: ${({ theme }) => theme.typography.bodyStrong.weight};\n  text-align: center;\n\n  &::-webkit-outer-spin-button,\n  &::-webkit-inner-spin-button {\n    -webkit-appearance: none;\n    margin: 0;\n  }\n\n  ::placeholder {\n    color: ${({ theme }) => theme.colors.neutral.textWeak};\n  }\n\n  -moz-appearance: textfield;\n\n  &[disabled] {\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n    cursor: not-allowed;\n  }\n`\n\nconst StyledText = styled('span', {\n  shouldForwardProp: prop => !['disabled'].includes(prop),\n})<{ disabled: boolean }>`\n  color: ${({ theme, disabled }) =>\n    disabled ? theme.colors.neutral.textDisabled : theme.colors.neutral.text};\n  user-select: none;\n  margin-right: ${({ theme }) => theme.space['1']};\n`\n\nconst StyledContainer = styled('div', {\n  shouldForwardProp: prop => !['size'].includes(prop),\n})<{ size: ContainerSizesType }>`\n  background-color: ${({ theme }) => theme.colors.neutral.background};\n  display: flex;\n  flex-direction: row;\n  align-items: center;\n  align-self: stretch;\n  font-weight: 500;\n  height: ${({ size }) => containerSizes[size]}px;\n  border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  border-radius: ${({ theme }) => theme.radii.default};\n\n  &[data-error='true'] {\n    border: 1px solid ${({ theme }) => theme.colors.danger.border};\n  }\n\n  &[aria-disabled='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    cursor: not-allowed;\n  }\n\n  &:not([aria-disabled='true']) {\n    ${StyledCenterBox}:hover,\n    ${StyledCenterBox}:focus {\n      border: 1px solid ${({ theme }) => theme.colors.primary.borderHover};\n    }\n\n    ${StyledCenterBox}:focus-within {\n      box-shadow: ${({ theme }) => theme.shadows.focusPrimary};\n      border: 1px solid ${({ theme }) => theme.colors.primary.borderHover};\n    }\n  }\n`\n\ntype NumberInputProps = {\n  disabled?: boolean\n  maxValue?: number\n  minValue?: number\n  name?: string\n  onChange?: (input: number | undefined) => void\n  onMaxCrossed?: () => void\n  onMinCrossed?: () => void\n  size?: ContainerSizesType\n  /**\n   * Define how much will stepper increase / decrease each time you click on + / - button.\n   */\n  step?: number\n  /**\n   * Text displayed into component at the right of number value.\n   */\n  text?: string\n  defaultValue?: number\n  value?: number | null\n  disabledTooltip?: string\n  className?: string\n  'data-testid'?: string\n  label?: string\n  'aria-label'?: string\n  'aria-describedby'?: string\n  id?: string\n  placeholder?: string\n  error?: string | boolean\n} & Omit<\n  InputHTMLAttributes<HTMLInputElement>,\n  'size' | 'onChange' | 'value' | 'defaultValue'\n>\n\n/**\n * @deprecated This component is deprecated. Please use `NumberInputV2` instead.\n */\nexport const NumberInput = ({\n  disabled = false,\n  maxValue,\n  minValue = 0,\n  name = 'numberinput',\n  onChange,\n  onFocus,\n  onBlur,\n  onMaxCrossed,\n  onMinCrossed,\n  size = 'large',\n  step = 1,\n  text,\n  defaultValue,\n  value,\n  disabledTooltip,\n  className,\n  label,\n  id,\n  placeholder,\n  error,\n  'aria-label': ariaLabel,\n  'aria-describedby': ariaDescribedBy,\n  'data-testid': dataTestId,\n}: NumberInputProps) => {\n  const inputRef =\n    useRef<HTMLInputElement>() as MutableRefObject<HTMLInputElement>\n\n  const uniqueId = useId()\n\n  // local state used if component is not controlled (no value prop provided)\n  const [inputValue, setInputValue] = useState<number | undefined>(() => {\n    if (defaultValue && minValue && defaultValue < minValue) {\n      return minValue\n    }\n    if (defaultValue && maxValue && defaultValue > maxValue) {\n      return maxValue\n    }\n\n    return defaultValue\n  })\n\n  const currentValue =\n    value !== undefined && value !== null ? value : inputValue\n\n  const setValue = (\n    newValue: number | undefined,\n    /**\n     * If true, will check if newValue is between minValue and maxValue and set it to minValue or maxValue if it's not.\n     */\n    hasMinMaxVerification = true,\n  ) => {\n    let nextValue = newValue\n    if (value === undefined && hasMinMaxVerification) {\n      if (newValue !== undefined && newValue < minValue) {\n        nextValue = minValue\n      }\n\n      if (\n        newValue !== undefined &&\n        maxValue !== undefined &&\n        newValue > maxValue\n      ) {\n        nextValue = maxValue\n      }\n    }\n    setInputValue(nextValue)\n    onChange?.(nextValue)\n  }\n\n  const offsetFn = (direction: number) => () => {\n    const localValue = currentValue ?? 0\n    const newValue =\n      localValue % step === 0 ? localValue + step * direction : localValue\n    const roundedValue = roundStep(newValue, step, direction)\n\n    setValue(roundedValue)\n  }\n\n  const handleChange: ChangeEventHandler<HTMLInputElement> = event => {\n    setValue(\n      event.currentTarget.value ? Number(event.currentTarget.value) : undefined,\n      false,\n    )\n  }\n\n  const handleOnBlur: FocusEventHandler<HTMLInputElement> = event => {\n    if (currentValue) {\n      const boundedValue = bounded(\n        currentValue,\n        minValue ?? currentValue,\n        maxValue ?? currentValue,\n      )\n\n      if (maxValue && currentValue > maxValue) onMaxCrossed?.()\n      if (minValue && currentValue < minValue) onMinCrossed?.()\n\n      setValue(boundedValue)\n\n      onBlur?.(event)\n    }\n  }\n\n  const onKeyDown: KeyboardEventHandler = event => {\n    if (event.key === 'ArrowUp') {\n      event.stopPropagation()\n      event.preventDefault()\n\n      const direction = 1\n      const localValue = currentValue ?? 0\n\n      const newValue =\n        localValue % step === 0 ? localValue + step * direction : localValue\n      const roundedValue = roundStep(newValue, step, direction)\n\n      if (maxValue === undefined) {\n        setValue(roundedValue)\n\n        return\n      }\n\n      setValue(Math.min(roundedValue, maxValue))\n    }\n\n    if (event.key === 'ArrowDown') {\n      event.stopPropagation()\n      event.preventDefault()\n\n      const direction = -1\n      const localValue = currentValue ?? 0\n\n      const newValue =\n        localValue % step === 0 ? localValue + step * direction : localValue\n      const roundedValue = roundStep(newValue, step, direction)\n\n      setValue(Math.max(roundedValue, minValue))\n    }\n  }\n\n  const isMinusDisabled = useMemo(() => {\n    if (disabled) return true\n    if (currentValue === undefined) return false\n    if (getMinusRoundedValue(currentValue, step) < minValue) {\n      return true\n    }\n\n    return disabled\n  }, [currentValue, disabled, minValue, step])\n\n  const isPlusDisabled = useMemo(() => {\n    if (disabled) return true\n    if (currentValue === undefined) return false\n    if (maxValue && getPlusRoundedValue(currentValue, step) > maxValue) {\n      return true\n    }\n\n    return disabled\n  }, [currentValue, disabled, maxValue, step])\n\n  const inputWidth = useMemo(() => {\n    if (placeholder && currentValue === undefined) {\n      return placeholder.length * 12\n    }\n\n    if (currentValue !== undefined) {\n      return currentValue.toString().length * 16\n    }\n\n    return BASE_INPUT_WIDTH\n  }, [currentValue, placeholder])\n\n  return (\n    <Stack gap={1}>\n      {label ? (\n        <Text variant=\"bodyStrong\" as=\"label\" htmlFor={id || uniqueId}>\n          {label}\n        </Text>\n      ) : null}\n      <Stack gap={0.5}>\n        <StyledContainer\n          aria-disabled={disabled}\n          data-error={!!error}\n          size={size}\n          className={className}\n          data-testid={dataTestId}\n        >\n          <Tooltip text={isMinusDisabled && disabledTooltip}>\n            <StyledSelectButton\n              onClick={offsetFn(-1)}\n              disabled={isMinusDisabled}\n              aria-label=\"Minus\"\n              type=\"button\"\n              variant=\"ghost\"\n              sentiment=\"primary\"\n              size=\"small\"\n            >\n              <MinusIcon\n                size={iconSizes[size]}\n                sentiment=\"primary\"\n                disabled={isMinusDisabled}\n              />\n            </StyledSelectButton>\n          </Tooltip>\n\n          <StyledCenterBox\n            size={size}\n            onClick={() => {\n              if (inputRef?.current) {\n                inputRef.current.focus()\n              }\n            }}\n            aria-live=\"assertive\"\n            role=\"status\"\n          >\n            <StyledInput\n              disabled={disabled}\n              name={name}\n              onBlur={handleOnBlur}\n              onChange={handleChange}\n              onFocus={onFocus}\n              onKeyDown={onKeyDown}\n              ref={inputRef}\n              style={{\n                width: inputWidth,\n              }}\n              value={currentValue !== undefined ? currentValue.toString() : ''} // A dom element can only have string attributes.\n              type=\"number\"\n              id={id || uniqueId}\n              aria-label={!label && !ariaLabel ? 'Number Input' : ariaLabel}\n              aria-describedby={ariaDescribedBy}\n              placeholder={placeholder}\n            />\n            {currentValue !== undefined ? (\n              <StyledText disabled={disabled}>{text}</StyledText>\n            ) : null}\n          </StyledCenterBox>\n\n          <Tooltip text={isPlusDisabled && disabledTooltip}>\n            <StyledSelectButton\n              onClick={offsetFn(1)}\n              disabled={isPlusDisabled}\n              aria-label=\"Plus\"\n              type=\"button\"\n              variant=\"ghost\"\n              sentiment=\"primary\"\n              size=\"small\"\n            >\n              <PlusIcon\n                size={iconSizes[size]}\n                sentiment=\"primary\"\n                disabled={isPlusDisabled}\n              />\n            </StyledSelectButton>\n          </Tooltip>\n        </StyledContainer>\n        {typeof error === 'string' ? (\n          <Text\n            as=\"span\"\n            variant=\"bodySmall\"\n            sentiment=\"danger\"\n            prominence=\"weak\"\n          >\n            {error}\n          </Text>\n        ) : null}\n      </Stack>\n    </Stack>\n  )\n}\n"]} */"));
|
|
98
98
|
const NumberInput = ({
|
|
99
99
|
disabled = false,
|
|
100
100
|
maxValue,
|
|
@@ -216,7 +216,7 @@ const NumberInput = ({
|
|
|
216
216
|
label ? /* @__PURE__ */ jsx(Text, { variant: "bodyStrong", as: "label", htmlFor: id || uniqueId, children: label }) : null,
|
|
217
217
|
/* @__PURE__ */ jsxs(Stack, { gap: 0.5, children: [
|
|
218
218
|
/* @__PURE__ */ jsxs(StyledContainer, { "aria-disabled": disabled, "data-error": !!error, size, className, "data-testid": dataTestId, children: [
|
|
219
|
-
/* @__PURE__ */ jsx(Tooltip, { text: isMinusDisabled && disabledTooltip, children: /* @__PURE__ */ jsx(StyledSelectButton, { onClick: offsetFn(-1), disabled: isMinusDisabled, "aria-label": "Minus", type: "button", variant: "ghost", sentiment: "primary", size: "small", children: /* @__PURE__ */ jsx(
|
|
219
|
+
/* @__PURE__ */ jsx(Tooltip, { text: isMinusDisabled && disabledTooltip, children: /* @__PURE__ */ jsx(StyledSelectButton, { onClick: offsetFn(-1), disabled: isMinusDisabled, "aria-label": "Minus", type: "button", variant: "ghost", sentiment: "primary", size: "small", children: /* @__PURE__ */ jsx(MinusIcon, { size: iconSizes[size], sentiment: "primary", disabled: isMinusDisabled }) }) }),
|
|
220
220
|
/* @__PURE__ */ jsxs(StyledCenterBox, { size, onClick: () => {
|
|
221
221
|
if (inputRef?.current) {
|
|
222
222
|
inputRef.current.focus();
|
|
@@ -245,7 +245,7 @@ const NumberInput = ({
|
|
|
245
245
|
),
|
|
246
246
|
currentValue !== void 0 ? /* @__PURE__ */ jsx(StyledText, { disabled, children: text }) : null
|
|
247
247
|
] }),
|
|
248
|
-
/* @__PURE__ */ jsx(Tooltip, { text: isPlusDisabled && disabledTooltip, children: /* @__PURE__ */ jsx(StyledSelectButton, { onClick: offsetFn(1), disabled: isPlusDisabled, "aria-label": "Plus", type: "button", variant: "ghost", sentiment: "primary", size: "small", children: /* @__PURE__ */ jsx(
|
|
248
|
+
/* @__PURE__ */ jsx(Tooltip, { text: isPlusDisabled && disabledTooltip, children: /* @__PURE__ */ jsx(StyledSelectButton, { onClick: offsetFn(1), disabled: isPlusDisabled, "aria-label": "Plus", type: "button", variant: "ghost", sentiment: "primary", size: "small", children: /* @__PURE__ */ jsx(PlusIcon, { size: iconSizes[size], sentiment: "primary", disabled: isPlusDisabled }) }) })
|
|
249
249
|
] }),
|
|
250
250
|
typeof error === "string" ? /* @__PURE__ */ jsx(Text, { as: "span", variant: "bodySmall", sentiment: "danger", prominence: "weak", children: error }) : null
|
|
251
251
|
] })
|
|
@@ -2,7 +2,7 @@
|
|
|
2
2
|
Object.defineProperty(exports, Symbol.toStringTag, { value: "Module" });
|
|
3
3
|
const jsxRuntime = require("@emotion/react/jsx-runtime");
|
|
4
4
|
const _styled = require("@emotion/styled/base");
|
|
5
|
-
const
|
|
5
|
+
const Icon = require("@ultraviolet/icons");
|
|
6
6
|
const index = require("../Stack/index.cjs");
|
|
7
7
|
const index$1 = require("../Text/index.cjs");
|
|
8
8
|
const _interopDefaultCompat = (e) => e && typeof e === "object" && "default" in e ? e : { default: e };
|
|
@@ -14,13 +14,13 @@ const PasswordCheckContainer = /* @__PURE__ */ _styled__default.default("div", p
|
|
|
14
14
|
label: "PasswordCheckContainer"
|
|
15
15
|
})("display:grid;grid-template-columns:repeat(2, 1fr);gap:", ({
|
|
16
16
|
theme
|
|
17
|
-
}) => theme.space["1"], ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,
|
|
17
|
+
}) => theme.space["1"], ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi9ob21lL3J1bm5lci93b3JrL3VsdHJhdmlvbGV0L3VsdHJhdmlvbGV0L3BhY2thZ2VzL3VpL3NyYy9jb21wb25lbnRzL1Bhc3N3b3JkQ2hlY2svaW5kZXgudHN4Il0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQXdCeUMiLCJmaWxlIjoiL2hvbWUvcnVubmVyL3dvcmsvdWx0cmF2aW9sZXQvdWx0cmF2aW9sZXQvcGFja2FnZXMvdWkvc3JjL2NvbXBvbmVudHMvUGFzc3dvcmRDaGVjay9pbmRleC50c3giLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgc3R5bGVkIGZyb20gJ0BlbW90aW9uL3N0eWxlZCdcbmltcG9ydCB7XG4gIENoZWNrQ2lyY2xlT3V0bGluZUljb24sXG4gIENsb3NlQ2lyY2xlT3V0bGluZUljb24sXG59IGZyb20gJ0B1bHRyYXZpb2xldC9pY29ucydcbmltcG9ydCB7IFN0YWNrIH0gZnJvbSAnLi4vU3RhY2snXG5pbXBvcnQgeyBUZXh0IH0gZnJvbSAnLi4vVGV4dCdcblxudHlwZSBSdWxlID0ge1xuICBuYW1lOiBzdHJpbmdcbiAgdGV4dDogc3RyaW5nXG4gIHZhbGlkOiBib29sZWFuXG59XG5cbnR5cGUgUGFzc3dvcmRDaGVja1Byb3BzID0ge1xuICAvKipcbiAgICogSXMgYW4gYXJyYXkgb2Ygb2JqZWN0IHRoYXQgY29udGFpbnMgcGFzc3dvcmQgcnVsZXMuIGBuYW1lYCBpcyB0aGUgbmFtZSBvZiB0aGUgcnVsZSwgYHRleHRgIHRoZSB0ZXh0IGFzc29jaWF0ZWRcbiAgICogd2l0aCB0aGUgcnVsZSBhbmQgYHZhbGlkYCBpcyBhIGJvb2xlYW4gdGhhdCBkZXRlcm1pbmUgaWYgdGhlIHJ1bGUgaXMgcmVzcGVjdGVkIG9yIG5vdC5cbiAgICovXG4gIHJ1bGVzOiBSdWxlW11cbiAgY2xhc3NOYW1lPzogc3RyaW5nXG4gICdkYXRhLXRlc3RpZCc/OiBzdHJpbmdcbn1cblxuY29uc3QgUGFzc3dvcmRDaGVja0NvbnRhaW5lciA9IHN0eWxlZC5kaXZgXG4gIGRpc3BsYXk6IGdyaWQ7XG4gIGdyaWQtdGVtcGxhdGUtY29sdW1uczogcmVwZWF0KDIsIDFmcik7XG4gIGdhcDogJHsoeyB0aGVtZSB9KSA9PiB0aGVtZS5zcGFjZVsnMSddfTtcbmBcblxuLyoqXG4gKiBQYXNzd29yZENoZWNrIGlzIGEgY29tcG9uZW50IHRoYXQgZGlzcGxheSBhIGxpc3Qgb2YgcGFzc3dvcmQgcnVsZXMgd2l0aCBhIGNoZWNrIG9yIGEgY3Jvc3MgZGVwZW5kaW5nIG9uIHRoZSB2YWxpZGl0eVxuICogb2YgdGhlIHJ1bGUuXG4gKi9cbmV4cG9ydCBjb25zdCBQYXNzd29yZENoZWNrID0gKHtcbiAgcnVsZXMsXG4gIGNsYXNzTmFtZSxcbiAgJ2RhdGEtdGVzdGlkJzogZGF0YVRlc3RJZCxcbn06IFBhc3N3b3JkQ2hlY2tQcm9wcykgPT4gKFxuICA8UGFzc3dvcmRDaGVja0NvbnRhaW5lciBjbGFzc05hbWU9e2NsYXNzTmFtZX0gZGF0YS10ZXN0aWQ9e2RhdGFUZXN0SWR9PlxuICAgIHtydWxlcy5tYXAocnVsZSA9PiAoXG4gICAgICA8U3RhY2sgZGlyZWN0aW9uPVwicm93XCIgZ2FwPXsxfSBhbGlnbkl0ZW1zPVwiY2VudGVyXCIga2V5PXtydWxlLm5hbWV9PlxuICAgICAgICB7cnVsZS52YWxpZCA/IChcbiAgICAgICAgICA8Q2hlY2tDaXJjbGVPdXRsaW5lSWNvblxuICAgICAgICAgICAgc2VudGltZW50PVwic3VjY2Vzc1wiXG4gICAgICAgICAgICBwcm9taW5lbmNlPVwid2Vha1wiXG4gICAgICAgICAgICBzaXplPXsyMH1cbiAgICAgICAgICAvPlxuICAgICAgICApIDogKFxuICAgICAgICAgIDxDbG9zZUNpcmNsZU91dGxpbmVJY29uXG4gICAgICAgICAgICBzZW50aW1lbnQ9XCJuZXV0cmFsXCJcbiAgICAgICAgICAgIHByb21pbmVuY2U9XCJ3ZWFrXCJcbiAgICAgICAgICAgIHNpemU9ezIwfVxuICAgICAgICAgIC8+XG4gICAgICAgICl9XG5cbiAgICAgICAgPFRleHQgYXM9XCJwXCIgdmFyaWFudD1cImJvZHlTbWFsbFwiPlxuICAgICAgICAgIHtydWxlLnRleHR9XG4gICAgICAgIDwvVGV4dD5cbiAgICAgIDwvU3RhY2s+XG4gICAgKSl9XG4gIDwvUGFzc3dvcmRDaGVja0NvbnRhaW5lcj5cbilcbiJdfQ== */"));
|
|
18
18
|
const PasswordCheck = ({
|
|
19
19
|
rules,
|
|
20
20
|
className,
|
|
21
21
|
"data-testid": dataTestId
|
|
22
22
|
}) => /* @__PURE__ */ jsxRuntime.jsx(PasswordCheckContainer, { className, "data-testid": dataTestId, children: rules.map((rule) => /* @__PURE__ */ jsxRuntime.jsxs(index.Stack, { direction: "row", gap: 1, alignItems: "center", children: [
|
|
23
|
-
/* @__PURE__ */ jsxRuntime.jsx(
|
|
23
|
+
rule.valid ? /* @__PURE__ */ jsxRuntime.jsx(Icon.CheckCircleOutlineIcon, { sentiment: "success", prominence: "weak", size: 20 }) : /* @__PURE__ */ jsxRuntime.jsx(Icon.CloseCircleOutlineIcon, { sentiment: "neutral", prominence: "weak", size: 20 }),
|
|
24
24
|
/* @__PURE__ */ jsxRuntime.jsx(index$1.Text, { as: "p", variant: "bodySmall", children: rule.text })
|
|
25
25
|
] }, rule.name)) });
|
|
26
26
|
exports.PasswordCheck = PasswordCheck;
|
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
import { jsx, jsxs } from "@emotion/react/jsx-runtime";
|
|
2
2
|
import _styled from "@emotion/styled/base";
|
|
3
|
-
import {
|
|
3
|
+
import { CheckCircleOutlineIcon, CloseCircleOutlineIcon } from "@ultraviolet/icons";
|
|
4
4
|
import { Stack } from "../Stack/index.js";
|
|
5
5
|
import { Text } from "../Text/index.js";
|
|
6
6
|
const PasswordCheckContainer = /* @__PURE__ */ _styled("div", process.env.NODE_ENV === "production" ? {
|
|
@@ -10,13 +10,13 @@ const PasswordCheckContainer = /* @__PURE__ */ _styled("div", process.env.NODE_E
|
|
|
10
10
|
label: "PasswordCheckContainer"
|
|
11
11
|
})("display:grid;grid-template-columns:repeat(2, 1fr);gap:", ({
|
|
12
12
|
theme
|
|
13
|
-
}) => theme.space["1"], ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,
|
|
13
|
+
}) => theme.space["1"], ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi9ob21lL3J1bm5lci93b3JrL3VsdHJhdmlvbGV0L3VsdHJhdmlvbGV0L3BhY2thZ2VzL3VpL3NyYy9jb21wb25lbnRzL1Bhc3N3b3JkQ2hlY2svaW5kZXgudHN4Il0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQXdCeUMiLCJmaWxlIjoiL2hvbWUvcnVubmVyL3dvcmsvdWx0cmF2aW9sZXQvdWx0cmF2aW9sZXQvcGFja2FnZXMvdWkvc3JjL2NvbXBvbmVudHMvUGFzc3dvcmRDaGVjay9pbmRleC50c3giLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgc3R5bGVkIGZyb20gJ0BlbW90aW9uL3N0eWxlZCdcbmltcG9ydCB7XG4gIENoZWNrQ2lyY2xlT3V0bGluZUljb24sXG4gIENsb3NlQ2lyY2xlT3V0bGluZUljb24sXG59IGZyb20gJ0B1bHRyYXZpb2xldC9pY29ucydcbmltcG9ydCB7IFN0YWNrIH0gZnJvbSAnLi4vU3RhY2snXG5pbXBvcnQgeyBUZXh0IH0gZnJvbSAnLi4vVGV4dCdcblxudHlwZSBSdWxlID0ge1xuICBuYW1lOiBzdHJpbmdcbiAgdGV4dDogc3RyaW5nXG4gIHZhbGlkOiBib29sZWFuXG59XG5cbnR5cGUgUGFzc3dvcmRDaGVja1Byb3BzID0ge1xuICAvKipcbiAgICogSXMgYW4gYXJyYXkgb2Ygb2JqZWN0IHRoYXQgY29udGFpbnMgcGFzc3dvcmQgcnVsZXMuIGBuYW1lYCBpcyB0aGUgbmFtZSBvZiB0aGUgcnVsZSwgYHRleHRgIHRoZSB0ZXh0IGFzc29jaWF0ZWRcbiAgICogd2l0aCB0aGUgcnVsZSBhbmQgYHZhbGlkYCBpcyBhIGJvb2xlYW4gdGhhdCBkZXRlcm1pbmUgaWYgdGhlIHJ1bGUgaXMgcmVzcGVjdGVkIG9yIG5vdC5cbiAgICovXG4gIHJ1bGVzOiBSdWxlW11cbiAgY2xhc3NOYW1lPzogc3RyaW5nXG4gICdkYXRhLXRlc3RpZCc/OiBzdHJpbmdcbn1cblxuY29uc3QgUGFzc3dvcmRDaGVja0NvbnRhaW5lciA9IHN0eWxlZC5kaXZgXG4gIGRpc3BsYXk6IGdyaWQ7XG4gIGdyaWQtdGVtcGxhdGUtY29sdW1uczogcmVwZWF0KDIsIDFmcik7XG4gIGdhcDogJHsoeyB0aGVtZSB9KSA9PiB0aGVtZS5zcGFjZVsnMSddfTtcbmBcblxuLyoqXG4gKiBQYXNzd29yZENoZWNrIGlzIGEgY29tcG9uZW50IHRoYXQgZGlzcGxheSBhIGxpc3Qgb2YgcGFzc3dvcmQgcnVsZXMgd2l0aCBhIGNoZWNrIG9yIGEgY3Jvc3MgZGVwZW5kaW5nIG9uIHRoZSB2YWxpZGl0eVxuICogb2YgdGhlIHJ1bGUuXG4gKi9cbmV4cG9ydCBjb25zdCBQYXNzd29yZENoZWNrID0gKHtcbiAgcnVsZXMsXG4gIGNsYXNzTmFtZSxcbiAgJ2RhdGEtdGVzdGlkJzogZGF0YVRlc3RJZCxcbn06IFBhc3N3b3JkQ2hlY2tQcm9wcykgPT4gKFxuICA8UGFzc3dvcmRDaGVja0NvbnRhaW5lciBjbGFzc05hbWU9e2NsYXNzTmFtZX0gZGF0YS10ZXN0aWQ9e2RhdGFUZXN0SWR9PlxuICAgIHtydWxlcy5tYXAocnVsZSA9PiAoXG4gICAgICA8U3RhY2sgZGlyZWN0aW9uPVwicm93XCIgZ2FwPXsxfSBhbGlnbkl0ZW1zPVwiY2VudGVyXCIga2V5PXtydWxlLm5hbWV9PlxuICAgICAgICB7cnVsZS52YWxpZCA/IChcbiAgICAgICAgICA8Q2hlY2tDaXJjbGVPdXRsaW5lSWNvblxuICAgICAgICAgICAgc2VudGltZW50PVwic3VjY2Vzc1wiXG4gICAgICAgICAgICBwcm9taW5lbmNlPVwid2Vha1wiXG4gICAgICAgICAgICBzaXplPXsyMH1cbiAgICAgICAgICAvPlxuICAgICAgICApIDogKFxuICAgICAgICAgIDxDbG9zZUNpcmNsZU91dGxpbmVJY29uXG4gICAgICAgICAgICBzZW50aW1lbnQ9XCJuZXV0cmFsXCJcbiAgICAgICAgICAgIHByb21pbmVuY2U9XCJ3ZWFrXCJcbiAgICAgICAgICAgIHNpemU9ezIwfVxuICAgICAgICAgIC8+XG4gICAgICAgICl9XG5cbiAgICAgICAgPFRleHQgYXM9XCJwXCIgdmFyaWFudD1cImJvZHlTbWFsbFwiPlxuICAgICAgICAgIHtydWxlLnRleHR9XG4gICAgICAgIDwvVGV4dD5cbiAgICAgIDwvU3RhY2s+XG4gICAgKSl9XG4gIDwvUGFzc3dvcmRDaGVja0NvbnRhaW5lcj5cbilcbiJdfQ== */"));
|
|
14
14
|
const PasswordCheck = ({
|
|
15
15
|
rules,
|
|
16
16
|
className,
|
|
17
17
|
"data-testid": dataTestId
|
|
18
18
|
}) => /* @__PURE__ */ jsx(PasswordCheckContainer, { className, "data-testid": dataTestId, children: rules.map((rule) => /* @__PURE__ */ jsxs(Stack, { direction: "row", gap: 1, alignItems: "center", children: [
|
|
19
|
-
/* @__PURE__ */ jsx(
|
|
19
|
+
rule.valid ? /* @__PURE__ */ jsx(CheckCircleOutlineIcon, { sentiment: "success", prominence: "weak", size: 20 }) : /* @__PURE__ */ jsx(CloseCircleOutlineIcon, { sentiment: "neutral", prominence: "weak", size: 20 }),
|
|
20
20
|
/* @__PURE__ */ jsx(Text, { as: "p", variant: "bodySmall", children: rule.text })
|
|
21
21
|
] }, rule.name)) });
|
|
22
22
|
export {
|
|
@@ -7,5 +7,15 @@ export type PositionsType = {
|
|
|
7
7
|
popupInitialPosition: string;
|
|
8
8
|
popupPosition: string;
|
|
9
9
|
};
|
|
10
|
-
export declare const animation: (positions: PositionsType) =>
|
|
11
|
-
|
|
10
|
+
export declare const animation: (positions: PositionsType) => {
|
|
11
|
+
name: string;
|
|
12
|
+
styles: string;
|
|
13
|
+
anim: 1;
|
|
14
|
+
toString: () => string;
|
|
15
|
+
} & string;
|
|
16
|
+
export declare const exitAnimation: (positions: PositionsType) => {
|
|
17
|
+
name: string;
|
|
18
|
+
styles: string;
|
|
19
|
+
anim: 1;
|
|
20
|
+
toString: () => string;
|
|
21
|
+
} & string;
|