@ultraviolet/ui 2.0.0-beta.11 → 2.0.0-beta.13

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (32) hide show
  1. package/dist/components/Button/index.cjs +11 -5
  2. package/dist/components/Button/index.d.ts +6 -0
  3. package/dist/components/Button/index.js +11 -5
  4. package/dist/components/Dialog/index.cjs +2 -2
  5. package/dist/components/Dialog/index.d.ts +6 -1
  6. package/dist/components/Dialog/index.js +2 -2
  7. package/dist/components/Drawer/index.cjs +44 -22
  8. package/dist/components/Drawer/index.d.ts +12 -1
  9. package/dist/components/Drawer/index.js +44 -22
  10. package/dist/components/Expandable/index.cjs +12 -3
  11. package/dist/components/Expandable/index.js +12 -3
  12. package/dist/components/List/HeaderCell.cjs +3 -3
  13. package/dist/components/List/HeaderCell.js +4 -4
  14. package/dist/components/List/Row.cjs +12 -10
  15. package/dist/components/List/Row.d.ts +4 -1
  16. package/dist/components/List/Row.js +12 -10
  17. package/dist/components/List/index.d.ts +2 -0
  18. package/dist/components/Modal/components/Disclosure.cjs +1 -1
  19. package/dist/components/Modal/components/Disclosure.js +1 -1
  20. package/dist/components/SelectInput/Dropdown.cjs +11 -11
  21. package/dist/components/SelectInput/Dropdown.js +11 -11
  22. package/dist/components/Table/HeaderCell.cjs +4 -4
  23. package/dist/components/Table/HeaderCell.js +5 -5
  24. package/dist/components/UnitInput/index.cjs +5 -5
  25. package/dist/components/UnitInput/index.js +5 -5
  26. package/dist/components/index.d.ts +0 -1
  27. package/dist/index.cjs +77 -79
  28. package/dist/index.js +0 -2
  29. package/package.json +2 -2
  30. package/dist/components/PasswordStrengthMeter/index.cjs +0 -98
  31. package/dist/components/PasswordStrengthMeter/index.d.ts +0 -39
  32. package/dist/components/PasswordStrengthMeter/index.js +0 -96
@@ -22,7 +22,7 @@ const StyledNumberInputWrapper = /* @__PURE__ */ _styled("div", process.env.NODE
22
22
  theme
23
23
  }) => theme.space["2"], ";border-right:1px solid ", ({
24
24
  theme
25
- }) => theme.colors.neutral.border, ";width:100%;height: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/UnitInput/index.tsx"],"names":[],"mappings":"AAkB2C","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/UnitInput/index.tsx","sourcesContent":["'use client'\n\nimport styled from '@emotion/styled'\nimport { AlertCircleIcon, CheckCircleIcon } from '@ultraviolet/icons'\nimport type { ComponentProps, InputHTMLAttributes, ReactNode } from 'react'\nimport { useEffect, useId, useMemo, useState } from 'react'\nimport { Label } from '../Label'\nimport { SelectInput } from '../SelectInput'\nimport type { OptionType } from '../SelectInput/types'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\n\nconst INPUT_SIZE_HEIGHT = {\n  small: '400', // sizing theme tokens key\n  medium: '500',\n  large: '600',\n} as const\n\nconst StyledNumberInputWrapper = styled.div`\n  display: flex;\n  align-items: center;\n  flex-direction: row;\n  padding-right: ${({ theme }) => theme.space['2']};\n  border-right: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  width: 100%;\n  height: 100%;\n`\n\nconst StyledInput = styled.input`\n  flex: 1;\n  border: none;\n  outline: none;\n  height: 100%;\n  width: 100%;\n  padding-left: ${({ theme }) => theme.space['2']};\n  background: transparent;\n  color: ${({ theme }) => theme.colors.neutral.text};\n  &[data-size=\"small\"] {\n    padding-left: ${({ theme }) => theme.space['1']};\n  }\n\n  &[data-size=\"large\"] {\n    font-size: ${({ theme }) => theme.typography.body.fontSize};\n  }\n\n  &::placeholder {\n    color: ${({ theme }) => theme.colors.neutral.textWeak};\n  }\n\n  &:disabled {\n    cursor: not-allowed;\n\n    &::placeholder {\n      color: ${({ theme }) => theme.colors.neutral.textWeakDisabled};\n    }\n  }\n`\n\nconst UnitInputWrapper = styled(Stack)<{\n  'data-size': 'small' | 'medium' | 'large'\n  'data-success': boolean\n  'data-error': boolean\n  'data-disabled': boolean\n  'data-readonly': boolean\n}>`\n  border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  border-radius: ${({ theme }) => theme.radii.default};\n\n  &:not([data-disabled='true']):not([data-readonly='true']):not(\n      [data-success='true']\n    ):not([data-error='true']):focus,\n  :not([data-disabled='true']):not([data-readonly='true']):not(\n      [data-success='true']\n    ):not([data-error='true']):active {\n    box-shadow: ${({ theme }) => theme.shadows.focusPrimary};\n    border-color: ${({ theme }) => theme.colors.primary.borderHover};\n\n    & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.primary.border};\n    }\n  }\n\n  &:not([data-disabled='true']):not([data-readonly='true']):not(\n      [data-success='true']\n    ):not([data-error='true']):focus-within  {\n      border-color: ${({ theme }) => theme.colors.primary.borderHover};\n      & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.primary.border};\n    }\n    }\n\n  &:not([data-disabled='true']):not([data-error='true']):not(\n      [data-success='true']\n    ):not([data-readonly='true']):hover,\n  :not([data-disabled='true']):not([data-error='true']):not(\n      [data-success='true']\n    ):focus {\n    text-decoration: none;\n    border-color: ${({ theme }) => theme.colors.primary.border};\n    & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.primary.border};\n    }\n  }\n\n  &[data-readonly='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n    border-color: ${({ theme }) => theme.colors.neutral.border};\n    cursor: default;\n  }\n\n  &[data-readonly='true']:active {\n    border-color: ${({ theme }) => theme.colors.neutral.border};\n  }\n\n  &[data-size='small'] {\n    height: ${({ theme }) => theme.sizing[INPUT_SIZE_HEIGHT.small]};\n  }\n  &[data-size='medium'] {\n    height: ${({ theme }) => theme.sizing[INPUT_SIZE_HEIGHT.medium]};\n  }\n  &[data-size='large'] {\n    height: ${({ theme }) => theme.sizing[INPUT_SIZE_HEIGHT.large]};\n  }\n\n  &[data-success='true'] {\n    border: 1px solid ${({ theme }) => theme.colors.success.border};\n    & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.success.border};\n    }\n  }\n  &[data-success='true']:active {\n    border: 1px solid ${({ theme }) => theme.colors.success.border};\n    box-shadow: ${({ theme }) => theme.shadows.focusSuccess};\n\n    & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.success.border};\n    }\n  }\n\n  &[data-error='true'] {\n    border: 1px solid ${({ theme }) => theme.colors.danger.border};\n\n    & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.danger.border};\n    }\n\n    & > ${StyledNumberInputWrapper}:hover {\n      border-right-color: ${({ theme }) => theme.colors.danger.border};\n    }\n  }\n\n  &[data-error='true']:active {\n    border: 1px solid ${({ theme }) => theme.colors.danger.border};\n    box-shadow: ${({ theme }) => theme.shadows.focusDanger};\n\n    & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.danger.border};\n    }\n    & > ${StyledNumberInputWrapper}:hover {\n      border-right-color: ${({ theme }) => theme.colors.danger.border};\n    }\n  }\n\n  &[data-disabled='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    border-color: ${({ theme }) => theme.colors.neutral.borderDisabled};\n    cursor: not-allowed;\n\n    & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.neutral.borderDisabled};\n    }\n  }\n`\n\nconst CustomSelectInput = styled(SelectInput)<{\n  width?: number | string\n  maxWidth?: number | string\n  'data-disabled': boolean\n}>`\n  #unit {\n    border: none;\n    background: transparent;\n  }\n\n  ${({ width }) =>\n    width && `width: ${typeof width === 'string' ? width : `${width}px`};`}\n  ${({ maxWidth }) =>\n    maxWidth &&\n    `max-width: ${typeof maxWidth === 'string' ? maxWidth : `${maxWidth}px`};`}\n\n  #unit:focus,\n  #unit:active {\n    box-shadow: none;\n  }\n`\ntype UnitInputValue = { inputValue: number; unit: string }\n\ntype UnitInputProps = {\n  className?: string\n  max?: number\n  min?: number\n  value?: UnitInputValue['inputValue']\n  unitValue?: UnitInputValue['unit']\n  onChange?: (value: UnitInputValue['inputValue']) => void\n  onChangeUnitValue?: (values: string) => void\n  options: OptionType[]\n  selectInputWidth?: number | string\n  size?: 'small' | 'medium' | 'large'\n  'data-testid'?: string\n  helper?: string\n  unitError?: string\n  width?: ComponentProps<typeof Stack>['width']\n  placeholderUnit?: string\n  error?: boolean | string\n  success?: boolean | string\n  label?: string\n  labelInformation?: ReactNode\n  step?: number | string\n  dropdownAlign?: ComponentProps<typeof SelectInput>['dropdownAlign']\n} & Pick<\n  InputHTMLAttributes<HTMLInputElement>,\n  | 'onFocus'\n  | 'onBlur'\n  | 'name'\n  | 'id'\n  | 'placeholder'\n  | 'disabled'\n  | 'readOnly'\n  | 'required'\n  | 'autoFocus'\n  | 'onKeyDown'\n>\n\nexport const UnitInput = ({\n  id,\n  name = '',\n  max = Number.MAX_SAFE_INTEGER,\n  min = 0,\n  autoFocus = false,\n  size = 'large',\n  placeholder = '0',\n  placeholderUnit = 'Select item',\n  onChange,\n  onChangeUnitValue,\n  value,\n  unitValue,\n  selectInputWidth = '12.6rem',\n  disabled = false,\n  options,\n  className,\n  label,\n  step = 1,\n  error,\n  required,\n  helper,\n  unitError,\n  success,\n  'data-testid': dataTestId,\n  width,\n  labelInformation,\n  readOnly,\n  onFocus,\n  onBlur,\n  onKeyDown,\n  dropdownAlign,\n}: UnitInputProps) => {\n  const [val, setVal] = useState(value)\n  const localId = useId()\n  const sentiment = useMemo(() => {\n    if (error) {\n      return 'danger'\n    }\n    if (success) {\n      return 'success'\n    }\n\n    return 'neutral'\n  }, [error, success])\n\n  useEffect(() => {\n    if (value !== undefined) {\n      setVal(value)\n    }\n  }, [value])\n\n  return (\n    <Stack gap={0.5}>\n      {label || labelInformation ? (\n        <Label\n          labelDescription={labelInformation}\n          required={required}\n          size={size}\n          htmlFor={id ?? localId}\n        >\n          {label}\n        </Label>\n      ) : null}\n      <UnitInputWrapper\n        direction=\"row\"\n        data-testid={dataTestId}\n        data-size={size}\n        width={width}\n        data-success={!!success}\n        data-error={!!error}\n        data-disabled={!!disabled}\n        data-readonly={!!readOnly}\n        onFocus={onFocus}\n        onBlur={onBlur}\n        onKeyDown={onKeyDown}\n      >\n        <StyledNumberInputWrapper id=\"input-field\">\n          <StyledInput\n            type=\"number\"\n            aria-invalid={!!error}\n            autoFocus={autoFocus}\n            disabled={disabled}\n            name={`${name}-value`}\n            width={width}\n            id={id ? `${id}-value` : undefined}\n            value={val}\n            onChange={event => {\n              const numericValue = Number.parseInt(event.target.value, 10)\n              if (numericValue > max) {\n                setVal(max)\n                onChange?.(max)\n              } else if (numericValue < min) {\n                setVal(min)\n                onChange?.(min)\n              } else {\n                setVal(numericValue)\n                onChange?.(numericValue)\n              }\n            }}\n            placeholder={placeholder}\n            max={max}\n            readOnly={readOnly}\n            min={min}\n            step={step}\n            data-success={success}\n            data-error={error}\n            data-testid=\"unit-input\"\n            required={required}\n            className={className}\n            data-size={size}\n          />\n          {error ? <AlertCircleIcon sentiment=\"danger\" /> : null}\n          {success && !error ? <CheckCircleIcon sentiment=\"success\" /> : null}\n        </StyledNumberInputWrapper>\n        <CustomSelectInput\n          data-disabled={disabled}\n          id=\"unit\"\n          name={`${name}-unit`}\n          onChange={(newValue: string) => {\n            onChangeUnitValue?.(newValue)\n          }}\n          error={unitError}\n          value={unitValue}\n          options={options}\n          width={selectInputWidth}\n          searchable={false}\n          clearable={false}\n          placeholder={placeholderUnit}\n          disabled={disabled}\n          size={size}\n          multiselect={false}\n          readOnly={readOnly}\n          dropdownAlign={dropdownAlign}\n        />\n      </UnitInputWrapper>\n      {error || typeof success === 'string' || typeof helper === 'string' ? (\n        <Text\n          as=\"p\"\n          variant=\"caption\"\n          sentiment={sentiment}\n          disabled={disabled}\n          prominence={sentiment === 'neutral' ? 'weak' : 'default'}\n        >\n          {error || success || helper}\n        </Text>\n      ) : null}\n      {!error && !success && typeof helper !== 'string' && helper\n        ? helper\n        : null}\n    </Stack>\n  )\n}\n"]} */"));
25
+ }) => theme.colors.neutral.border, ";width:100%;height: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/UnitInput/index.tsx"],"names":[],"mappings":"AAkB2C","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/UnitInput/index.tsx","sourcesContent":["'use client'\n\nimport styled from '@emotion/styled'\nimport { AlertCircleIcon, CheckCircleIcon } from '@ultraviolet/icons'\nimport type { ComponentProps, InputHTMLAttributes, ReactNode } from 'react'\nimport { useEffect, useId, useMemo, useState } from 'react'\nimport { Label } from '../Label'\nimport { SelectInput } from '../SelectInput'\nimport type { OptionType } from '../SelectInput/types'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\n\nconst INPUT_SIZE_HEIGHT = {\n  small: '400', // sizing theme tokens key\n  medium: '500',\n  large: '600',\n} as const\n\nconst StyledNumberInputWrapper = styled.div`\n  display: flex;\n  align-items: center;\n  flex-direction: row;\n  padding-right: ${({ theme }) => theme.space['2']};\n  border-right: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  width: 100%;\n  height: 100%;\n`\n\nconst StyledInput = styled.input`\n  flex: 1;\n  border: none;\n  outline: none;\n  height: 100%;\n  width: 100%;\n  padding-left: ${({ theme }) => theme.space['2']};\n  background: transparent;\n  color: ${({ theme }) => theme.colors.neutral.text};\n  &[data-size=\"small\"] {\n    padding-left: ${({ theme }) => theme.space['1']};\n  }\n\n  &[data-size=\"large\"] {\n    font-size: ${({ theme }) => theme.typography.body.fontSize};\n  }\n\n  &::placeholder {\n    color: ${({ theme }) => theme.colors.neutral.textWeak};\n  }\n\n  &:disabled {\n    cursor: not-allowed;\n\n    &::placeholder {\n      color: ${({ theme }) => theme.colors.neutral.textWeakDisabled};\n    }\n  }\n`\n\nconst UnitInputWrapper = styled(Stack)<{\n  'data-size': 'small' | 'medium' | 'large'\n  'data-success': boolean\n  'data-error': boolean\n  'data-disabled': boolean\n  'data-readonly': boolean\n}>`\n  border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  border-radius: ${({ theme }) => theme.radii.default};\n\n  &:not([data-disabled='true']):not([data-readonly='true']):not(\n      [data-success='true']\n    ):not([data-error='true']):focus,\n  :not([data-disabled='true']):not([data-readonly='true']):not(\n      [data-success='true']\n    ):not([data-error='true']):active {\n    box-shadow: ${({ theme }) => theme.shadows.focusPrimary};\n    border-color: ${({ theme }) => theme.colors.primary.borderHover};\n\n    & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.primary.border};\n    }\n  }\n\n  &:not([data-disabled='true']):not([data-readonly='true']):not(\n      [data-success='true']\n    ):not([data-error='true']):focus-within  {\n      border-color: ${({ theme }) => theme.colors.primary.borderHover};\n      & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.primary.border};\n    }\n    }\n\n  &:not([data-disabled='true']):not([data-error='true']):not(\n      [data-success='true']\n    ):not([data-readonly='true']):hover,\n  :not([data-disabled='true']):not([data-error='true']):not(\n      [data-success='true']\n    ):focus {\n    text-decoration: none;\n    border-color: ${({ theme }) => theme.colors.primary.border};\n    & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.primary.border};\n    }\n  }\n\n  &[data-readonly='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n    border-color: ${({ theme }) => theme.colors.neutral.border};\n    cursor: default;\n  }\n\n  &[data-readonly='true']:active {\n    border-color: ${({ theme }) => theme.colors.neutral.border};\n  }\n\n  &[data-size='small'] {\n    height: ${({ theme }) => theme.sizing[INPUT_SIZE_HEIGHT.small]};\n  }\n  &[data-size='medium'] {\n    height: ${({ theme }) => theme.sizing[INPUT_SIZE_HEIGHT.medium]};\n  }\n  &[data-size='large'] {\n    height: ${({ theme }) => theme.sizing[INPUT_SIZE_HEIGHT.large]};\n  }\n\n  &[data-success='true'] {\n    border: 1px solid ${({ theme }) => theme.colors.success.border};\n    & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.success.border};\n    }\n  }\n  &[data-success='true']:active {\n    border: 1px solid ${({ theme }) => theme.colors.success.border};\n    box-shadow: ${({ theme }) => theme.shadows.focusSuccess};\n\n    & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.success.border};\n    }\n  }\n\n  &[data-error='true'] {\n    border: 1px solid ${({ theme }) => theme.colors.danger.border};\n\n    & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.danger.border};\n    }\n\n    & > ${StyledNumberInputWrapper}:hover {\n      border-right-color: ${({ theme }) => theme.colors.danger.border};\n    }\n  }\n\n  &[data-error='true']:active {\n    border: 1px solid ${({ theme }) => theme.colors.danger.border};\n    box-shadow: ${({ theme }) => theme.shadows.focusDanger};\n\n    & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.danger.border};\n    }\n    & > ${StyledNumberInputWrapper}:hover {\n      border-right-color: ${({ theme }) => theme.colors.danger.border};\n    }\n  }\n\n  &[data-disabled='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    border-color: ${({ theme }) => theme.colors.neutral.borderDisabled};\n    cursor: not-allowed;\n\n    & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.neutral.borderDisabled};\n    }\n  }\n`\n\nconst CustomSelectInput = styled(SelectInput)<{\n  width?: number | string\n  maxWidth?: number | string\n  'data-disabled': boolean\n}>`\n  #unit {\n    border: none;\n    background: transparent;\n  }\n\n  ${({ width }) =>\n    width && `width: ${typeof width === 'string' ? width : `${width}px`};`}\n  ${({ maxWidth }) =>\n    maxWidth &&\n    `max-width: ${typeof maxWidth === 'string' ? maxWidth : `${maxWidth}px`};`}\n\n  #unit:focus,\n  #unit:active {\n    box-shadow: none;\n  }\n`\ntype UnitInputValue = { inputValue: number; unit: string }\n\ntype UnitInputProps = {\n  className?: string\n  max?: number\n  min?: number\n  value?: UnitInputValue['inputValue']\n  unitValue?: UnitInputValue['unit']\n  onChange?: (value: UnitInputValue['inputValue']) => void\n  onChangeUnitValue?: (values: string) => void\n  options: OptionType[]\n  selectInputWidth?: number | string\n  size?: 'small' | 'medium' | 'large'\n  'data-testid'?: string\n  helper?: string\n  unitError?: string\n  width?: ComponentProps<typeof Stack>['width']\n  placeholderUnit?: string\n  error?: boolean | string\n  success?: boolean | string\n  label?: string\n  labelInformation?: ReactNode\n  step?: number | string\n  dropdownAlign?: ComponentProps<typeof SelectInput>['dropdownAlign']\n} & Pick<\n  InputHTMLAttributes<HTMLInputElement>,\n  | 'onFocus'\n  | 'onBlur'\n  | 'name'\n  | 'id'\n  | 'placeholder'\n  | 'disabled'\n  | 'readOnly'\n  | 'required'\n  | 'autoFocus'\n  | 'onKeyDown'\n>\n\nexport const UnitInput = ({\n  id,\n  name = '',\n  max = Number.MAX_SAFE_INTEGER,\n  min = 0,\n  autoFocus = false,\n  size = 'large',\n  placeholder = '0',\n  placeholderUnit = 'Select item',\n  onChange,\n  onChangeUnitValue,\n  value,\n  unitValue,\n  selectInputWidth = '12.6rem',\n  disabled = false,\n  options,\n  className,\n  label,\n  step = 1,\n  error,\n  required,\n  helper,\n  unitError,\n  success,\n  'data-testid': dataTestId,\n  width,\n  labelInformation,\n  readOnly,\n  onFocus,\n  onBlur,\n  onKeyDown,\n  dropdownAlign,\n}: UnitInputProps) => {\n  const [val, setVal] = useState(value)\n  const localId = useId()\n  const sentiment = useMemo(() => {\n    if (error) {\n      return 'danger'\n    }\n    if (success) {\n      return 'success'\n    }\n\n    return 'neutral'\n  }, [error, success])\n\n  useEffect(() => {\n    if (value !== undefined) {\n      setVal(value)\n    }\n  }, [value])\n\n  return (\n    <Stack gap={0.5}>\n      {label || labelInformation ? (\n        <Label\n          labelDescription={labelInformation}\n          required={required}\n          size={size}\n          htmlFor={id ?? localId}\n        >\n          {label}\n        </Label>\n      ) : null}\n      <UnitInputWrapper\n        direction=\"row\"\n        data-testid={dataTestId}\n        data-size={size}\n        width={width}\n        data-success={!!success}\n        data-error={!!error}\n        data-disabled={!!disabled}\n        data-readonly={!!readOnly}\n        onFocus={onFocus}\n        onBlur={onBlur}\n        onKeyDown={onKeyDown}\n      >\n        <StyledNumberInputWrapper id=\"input-field\">\n          <StyledInput\n            type=\"number\"\n            aria-invalid={!!error}\n            autoFocus={autoFocus}\n            disabled={disabled}\n            name={`${name}-value`}\n            width={width}\n            id={id ?? localId}\n            value={val}\n            onChange={event => {\n              const numericValue = Number.parseInt(event.target.value, 10)\n              if (numericValue > max) {\n                setVal(max)\n                onChange?.(max)\n              } else if (numericValue < min) {\n                setVal(min)\n                onChange?.(min)\n              } else {\n                setVal(numericValue)\n                onChange?.(numericValue)\n              }\n            }}\n            placeholder={placeholder}\n            max={max}\n            readOnly={readOnly}\n            min={min}\n            step={step}\n            data-success={success}\n            data-error={error}\n            data-testid=\"unit-input\"\n            required={required}\n            className={className}\n            data-size={size}\n          />\n          {error ? <AlertCircleIcon sentiment=\"danger\" /> : null}\n          {success && !error ? <CheckCircleIcon sentiment=\"success\" /> : null}\n        </StyledNumberInputWrapper>\n        <CustomSelectInput\n          data-disabled={disabled}\n          id=\"unit\"\n          name={`${name}-unit`}\n          onChange={(newValue: string) => {\n            onChangeUnitValue?.(newValue)\n          }}\n          error={unitError}\n          value={unitValue}\n          options={options}\n          width={selectInputWidth}\n          searchable={false}\n          clearable={false}\n          placeholder={placeholderUnit}\n          disabled={disabled}\n          size={size}\n          multiselect={false}\n          readOnly={readOnly}\n          dropdownAlign={dropdownAlign}\n        />\n      </UnitInputWrapper>\n      {error || typeof success === 'string' || typeof helper === 'string' ? (\n        <Text\n          as=\"p\"\n          variant=\"caption\"\n          sentiment={sentiment}\n          disabled={disabled}\n          prominence={sentiment === 'neutral' ? 'weak' : 'default'}\n        >\n          {error || success || helper}\n        </Text>\n      ) : null}\n      {!error && !success && typeof helper !== 'string' && helper\n        ? helper\n        : null}\n    </Stack>\n  )\n}\n"]} */"));
26
26
  const StyledInput = /* @__PURE__ */ _styled("input", process.env.NODE_ENV === "production" ? {
27
27
  target: "ex81jph2"
28
28
  } : {
@@ -40,7 +40,7 @@ const StyledInput = /* @__PURE__ */ _styled("input", process.env.NODE_ENV === "p
40
40
  theme
41
41
  }) => theme.colors.neutral.textWeak, ";}&:disabled{cursor:not-allowed;&::placeholder{color:", ({
42
42
  theme
43
- }) => theme.colors.neutral.textWeakDisabled, ";}}" + (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/UnitInput/index.tsx"],"names":[],"mappings":"AA4BgC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/UnitInput/index.tsx","sourcesContent":["'use client'\n\nimport styled from '@emotion/styled'\nimport { AlertCircleIcon, CheckCircleIcon } from '@ultraviolet/icons'\nimport type { ComponentProps, InputHTMLAttributes, ReactNode } from 'react'\nimport { useEffect, useId, useMemo, useState } from 'react'\nimport { Label } from '../Label'\nimport { SelectInput } from '../SelectInput'\nimport type { OptionType } from '../SelectInput/types'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\n\nconst INPUT_SIZE_HEIGHT = {\n  small: '400', // sizing theme tokens key\n  medium: '500',\n  large: '600',\n} as const\n\nconst StyledNumberInputWrapper = styled.div`\n  display: flex;\n  align-items: center;\n  flex-direction: row;\n  padding-right: ${({ theme }) => theme.space['2']};\n  border-right: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  width: 100%;\n  height: 100%;\n`\n\nconst StyledInput = styled.input`\n  flex: 1;\n  border: none;\n  outline: none;\n  height: 100%;\n  width: 100%;\n  padding-left: ${({ theme }) => theme.space['2']};\n  background: transparent;\n  color: ${({ theme }) => theme.colors.neutral.text};\n  &[data-size=\"small\"] {\n    padding-left: ${({ theme }) => theme.space['1']};\n  }\n\n  &[data-size=\"large\"] {\n    font-size: ${({ theme }) => theme.typography.body.fontSize};\n  }\n\n  &::placeholder {\n    color: ${({ theme }) => theme.colors.neutral.textWeak};\n  }\n\n  &:disabled {\n    cursor: not-allowed;\n\n    &::placeholder {\n      color: ${({ theme }) => theme.colors.neutral.textWeakDisabled};\n    }\n  }\n`\n\nconst UnitInputWrapper = styled(Stack)<{\n  'data-size': 'small' | 'medium' | 'large'\n  'data-success': boolean\n  'data-error': boolean\n  'data-disabled': boolean\n  'data-readonly': boolean\n}>`\n  border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  border-radius: ${({ theme }) => theme.radii.default};\n\n  &:not([data-disabled='true']):not([data-readonly='true']):not(\n      [data-success='true']\n    ):not([data-error='true']):focus,\n  :not([data-disabled='true']):not([data-readonly='true']):not(\n      [data-success='true']\n    ):not([data-error='true']):active {\n    box-shadow: ${({ theme }) => theme.shadows.focusPrimary};\n    border-color: ${({ theme }) => theme.colors.primary.borderHover};\n\n    & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.primary.border};\n    }\n  }\n\n  &:not([data-disabled='true']):not([data-readonly='true']):not(\n      [data-success='true']\n    ):not([data-error='true']):focus-within  {\n      border-color: ${({ theme }) => theme.colors.primary.borderHover};\n      & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.primary.border};\n    }\n    }\n\n  &:not([data-disabled='true']):not([data-error='true']):not(\n      [data-success='true']\n    ):not([data-readonly='true']):hover,\n  :not([data-disabled='true']):not([data-error='true']):not(\n      [data-success='true']\n    ):focus {\n    text-decoration: none;\n    border-color: ${({ theme }) => theme.colors.primary.border};\n    & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.primary.border};\n    }\n  }\n\n  &[data-readonly='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n    border-color: ${({ theme }) => theme.colors.neutral.border};\n    cursor: default;\n  }\n\n  &[data-readonly='true']:active {\n    border-color: ${({ theme }) => theme.colors.neutral.border};\n  }\n\n  &[data-size='small'] {\n    height: ${({ theme }) => theme.sizing[INPUT_SIZE_HEIGHT.small]};\n  }\n  &[data-size='medium'] {\n    height: ${({ theme }) => theme.sizing[INPUT_SIZE_HEIGHT.medium]};\n  }\n  &[data-size='large'] {\n    height: ${({ theme }) => theme.sizing[INPUT_SIZE_HEIGHT.large]};\n  }\n\n  &[data-success='true'] {\n    border: 1px solid ${({ theme }) => theme.colors.success.border};\n    & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.success.border};\n    }\n  }\n  &[data-success='true']:active {\n    border: 1px solid ${({ theme }) => theme.colors.success.border};\n    box-shadow: ${({ theme }) => theme.shadows.focusSuccess};\n\n    & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.success.border};\n    }\n  }\n\n  &[data-error='true'] {\n    border: 1px solid ${({ theme }) => theme.colors.danger.border};\n\n    & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.danger.border};\n    }\n\n    & > ${StyledNumberInputWrapper}:hover {\n      border-right-color: ${({ theme }) => theme.colors.danger.border};\n    }\n  }\n\n  &[data-error='true']:active {\n    border: 1px solid ${({ theme }) => theme.colors.danger.border};\n    box-shadow: ${({ theme }) => theme.shadows.focusDanger};\n\n    & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.danger.border};\n    }\n    & > ${StyledNumberInputWrapper}:hover {\n      border-right-color: ${({ theme }) => theme.colors.danger.border};\n    }\n  }\n\n  &[data-disabled='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    border-color: ${({ theme }) => theme.colors.neutral.borderDisabled};\n    cursor: not-allowed;\n\n    & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.neutral.borderDisabled};\n    }\n  }\n`\n\nconst CustomSelectInput = styled(SelectInput)<{\n  width?: number | string\n  maxWidth?: number | string\n  'data-disabled': boolean\n}>`\n  #unit {\n    border: none;\n    background: transparent;\n  }\n\n  ${({ width }) =>\n    width && `width: ${typeof width === 'string' ? width : `${width}px`};`}\n  ${({ maxWidth }) =>\n    maxWidth &&\n    `max-width: ${typeof maxWidth === 'string' ? maxWidth : `${maxWidth}px`};`}\n\n  #unit:focus,\n  #unit:active {\n    box-shadow: none;\n  }\n`\ntype UnitInputValue = { inputValue: number; unit: string }\n\ntype UnitInputProps = {\n  className?: string\n  max?: number\n  min?: number\n  value?: UnitInputValue['inputValue']\n  unitValue?: UnitInputValue['unit']\n  onChange?: (value: UnitInputValue['inputValue']) => void\n  onChangeUnitValue?: (values: string) => void\n  options: OptionType[]\n  selectInputWidth?: number | string\n  size?: 'small' | 'medium' | 'large'\n  'data-testid'?: string\n  helper?: string\n  unitError?: string\n  width?: ComponentProps<typeof Stack>['width']\n  placeholderUnit?: string\n  error?: boolean | string\n  success?: boolean | string\n  label?: string\n  labelInformation?: ReactNode\n  step?: number | string\n  dropdownAlign?: ComponentProps<typeof SelectInput>['dropdownAlign']\n} & Pick<\n  InputHTMLAttributes<HTMLInputElement>,\n  | 'onFocus'\n  | 'onBlur'\n  | 'name'\n  | 'id'\n  | 'placeholder'\n  | 'disabled'\n  | 'readOnly'\n  | 'required'\n  | 'autoFocus'\n  | 'onKeyDown'\n>\n\nexport const UnitInput = ({\n  id,\n  name = '',\n  max = Number.MAX_SAFE_INTEGER,\n  min = 0,\n  autoFocus = false,\n  size = 'large',\n  placeholder = '0',\n  placeholderUnit = 'Select item',\n  onChange,\n  onChangeUnitValue,\n  value,\n  unitValue,\n  selectInputWidth = '12.6rem',\n  disabled = false,\n  options,\n  className,\n  label,\n  step = 1,\n  error,\n  required,\n  helper,\n  unitError,\n  success,\n  'data-testid': dataTestId,\n  width,\n  labelInformation,\n  readOnly,\n  onFocus,\n  onBlur,\n  onKeyDown,\n  dropdownAlign,\n}: UnitInputProps) => {\n  const [val, setVal] = useState(value)\n  const localId = useId()\n  const sentiment = useMemo(() => {\n    if (error) {\n      return 'danger'\n    }\n    if (success) {\n      return 'success'\n    }\n\n    return 'neutral'\n  }, [error, success])\n\n  useEffect(() => {\n    if (value !== undefined) {\n      setVal(value)\n    }\n  }, [value])\n\n  return (\n    <Stack gap={0.5}>\n      {label || labelInformation ? (\n        <Label\n          labelDescription={labelInformation}\n          required={required}\n          size={size}\n          htmlFor={id ?? localId}\n        >\n          {label}\n        </Label>\n      ) : null}\n      <UnitInputWrapper\n        direction=\"row\"\n        data-testid={dataTestId}\n        data-size={size}\n        width={width}\n        data-success={!!success}\n        data-error={!!error}\n        data-disabled={!!disabled}\n        data-readonly={!!readOnly}\n        onFocus={onFocus}\n        onBlur={onBlur}\n        onKeyDown={onKeyDown}\n      >\n        <StyledNumberInputWrapper id=\"input-field\">\n          <StyledInput\n            type=\"number\"\n            aria-invalid={!!error}\n            autoFocus={autoFocus}\n            disabled={disabled}\n            name={`${name}-value`}\n            width={width}\n            id={id ? `${id}-value` : undefined}\n            value={val}\n            onChange={event => {\n              const numericValue = Number.parseInt(event.target.value, 10)\n              if (numericValue > max) {\n                setVal(max)\n                onChange?.(max)\n              } else if (numericValue < min) {\n                setVal(min)\n                onChange?.(min)\n              } else {\n                setVal(numericValue)\n                onChange?.(numericValue)\n              }\n            }}\n            placeholder={placeholder}\n            max={max}\n            readOnly={readOnly}\n            min={min}\n            step={step}\n            data-success={success}\n            data-error={error}\n            data-testid=\"unit-input\"\n            required={required}\n            className={className}\n            data-size={size}\n          />\n          {error ? <AlertCircleIcon sentiment=\"danger\" /> : null}\n          {success && !error ? <CheckCircleIcon sentiment=\"success\" /> : null}\n        </StyledNumberInputWrapper>\n        <CustomSelectInput\n          data-disabled={disabled}\n          id=\"unit\"\n          name={`${name}-unit`}\n          onChange={(newValue: string) => {\n            onChangeUnitValue?.(newValue)\n          }}\n          error={unitError}\n          value={unitValue}\n          options={options}\n          width={selectInputWidth}\n          searchable={false}\n          clearable={false}\n          placeholder={placeholderUnit}\n          disabled={disabled}\n          size={size}\n          multiselect={false}\n          readOnly={readOnly}\n          dropdownAlign={dropdownAlign}\n        />\n      </UnitInputWrapper>\n      {error || typeof success === 'string' || typeof helper === 'string' ? (\n        <Text\n          as=\"p\"\n          variant=\"caption\"\n          sentiment={sentiment}\n          disabled={disabled}\n          prominence={sentiment === 'neutral' ? 'weak' : 'default'}\n        >\n          {error || success || helper}\n        </Text>\n      ) : null}\n      {!error && !success && typeof helper !== 'string' && helper\n        ? helper\n        : null}\n    </Stack>\n  )\n}\n"]} */"));
43
+ }) => theme.colors.neutral.textWeakDisabled, ";}}" + (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/UnitInput/index.tsx"],"names":[],"mappings":"AA4BgC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/UnitInput/index.tsx","sourcesContent":["'use client'\n\nimport styled from '@emotion/styled'\nimport { AlertCircleIcon, CheckCircleIcon } from '@ultraviolet/icons'\nimport type { ComponentProps, InputHTMLAttributes, ReactNode } from 'react'\nimport { useEffect, useId, useMemo, useState } from 'react'\nimport { Label } from '../Label'\nimport { SelectInput } from '../SelectInput'\nimport type { OptionType } from '../SelectInput/types'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\n\nconst INPUT_SIZE_HEIGHT = {\n  small: '400', // sizing theme tokens key\n  medium: '500',\n  large: '600',\n} as const\n\nconst StyledNumberInputWrapper = styled.div`\n  display: flex;\n  align-items: center;\n  flex-direction: row;\n  padding-right: ${({ theme }) => theme.space['2']};\n  border-right: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  width: 100%;\n  height: 100%;\n`\n\nconst StyledInput = styled.input`\n  flex: 1;\n  border: none;\n  outline: none;\n  height: 100%;\n  width: 100%;\n  padding-left: ${({ theme }) => theme.space['2']};\n  background: transparent;\n  color: ${({ theme }) => theme.colors.neutral.text};\n  &[data-size=\"small\"] {\n    padding-left: ${({ theme }) => theme.space['1']};\n  }\n\n  &[data-size=\"large\"] {\n    font-size: ${({ theme }) => theme.typography.body.fontSize};\n  }\n\n  &::placeholder {\n    color: ${({ theme }) => theme.colors.neutral.textWeak};\n  }\n\n  &:disabled {\n    cursor: not-allowed;\n\n    &::placeholder {\n      color: ${({ theme }) => theme.colors.neutral.textWeakDisabled};\n    }\n  }\n`\n\nconst UnitInputWrapper = styled(Stack)<{\n  'data-size': 'small' | 'medium' | 'large'\n  'data-success': boolean\n  'data-error': boolean\n  'data-disabled': boolean\n  'data-readonly': boolean\n}>`\n  border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  border-radius: ${({ theme }) => theme.radii.default};\n\n  &:not([data-disabled='true']):not([data-readonly='true']):not(\n      [data-success='true']\n    ):not([data-error='true']):focus,\n  :not([data-disabled='true']):not([data-readonly='true']):not(\n      [data-success='true']\n    ):not([data-error='true']):active {\n    box-shadow: ${({ theme }) => theme.shadows.focusPrimary};\n    border-color: ${({ theme }) => theme.colors.primary.borderHover};\n\n    & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.primary.border};\n    }\n  }\n\n  &:not([data-disabled='true']):not([data-readonly='true']):not(\n      [data-success='true']\n    ):not([data-error='true']):focus-within  {\n      border-color: ${({ theme }) => theme.colors.primary.borderHover};\n      & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.primary.border};\n    }\n    }\n\n  &:not([data-disabled='true']):not([data-error='true']):not(\n      [data-success='true']\n    ):not([data-readonly='true']):hover,\n  :not([data-disabled='true']):not([data-error='true']):not(\n      [data-success='true']\n    ):focus {\n    text-decoration: none;\n    border-color: ${({ theme }) => theme.colors.primary.border};\n    & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.primary.border};\n    }\n  }\n\n  &[data-readonly='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n    border-color: ${({ theme }) => theme.colors.neutral.border};\n    cursor: default;\n  }\n\n  &[data-readonly='true']:active {\n    border-color: ${({ theme }) => theme.colors.neutral.border};\n  }\n\n  &[data-size='small'] {\n    height: ${({ theme }) => theme.sizing[INPUT_SIZE_HEIGHT.small]};\n  }\n  &[data-size='medium'] {\n    height: ${({ theme }) => theme.sizing[INPUT_SIZE_HEIGHT.medium]};\n  }\n  &[data-size='large'] {\n    height: ${({ theme }) => theme.sizing[INPUT_SIZE_HEIGHT.large]};\n  }\n\n  &[data-success='true'] {\n    border: 1px solid ${({ theme }) => theme.colors.success.border};\n    & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.success.border};\n    }\n  }\n  &[data-success='true']:active {\n    border: 1px solid ${({ theme }) => theme.colors.success.border};\n    box-shadow: ${({ theme }) => theme.shadows.focusSuccess};\n\n    & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.success.border};\n    }\n  }\n\n  &[data-error='true'] {\n    border: 1px solid ${({ theme }) => theme.colors.danger.border};\n\n    & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.danger.border};\n    }\n\n    & > ${StyledNumberInputWrapper}:hover {\n      border-right-color: ${({ theme }) => theme.colors.danger.border};\n    }\n  }\n\n  &[data-error='true']:active {\n    border: 1px solid ${({ theme }) => theme.colors.danger.border};\n    box-shadow: ${({ theme }) => theme.shadows.focusDanger};\n\n    & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.danger.border};\n    }\n    & > ${StyledNumberInputWrapper}:hover {\n      border-right-color: ${({ theme }) => theme.colors.danger.border};\n    }\n  }\n\n  &[data-disabled='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    border-color: ${({ theme }) => theme.colors.neutral.borderDisabled};\n    cursor: not-allowed;\n\n    & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.neutral.borderDisabled};\n    }\n  }\n`\n\nconst CustomSelectInput = styled(SelectInput)<{\n  width?: number | string\n  maxWidth?: number | string\n  'data-disabled': boolean\n}>`\n  #unit {\n    border: none;\n    background: transparent;\n  }\n\n  ${({ width }) =>\n    width && `width: ${typeof width === 'string' ? width : `${width}px`};`}\n  ${({ maxWidth }) =>\n    maxWidth &&\n    `max-width: ${typeof maxWidth === 'string' ? maxWidth : `${maxWidth}px`};`}\n\n  #unit:focus,\n  #unit:active {\n    box-shadow: none;\n  }\n`\ntype UnitInputValue = { inputValue: number; unit: string }\n\ntype UnitInputProps = {\n  className?: string\n  max?: number\n  min?: number\n  value?: UnitInputValue['inputValue']\n  unitValue?: UnitInputValue['unit']\n  onChange?: (value: UnitInputValue['inputValue']) => void\n  onChangeUnitValue?: (values: string) => void\n  options: OptionType[]\n  selectInputWidth?: number | string\n  size?: 'small' | 'medium' | 'large'\n  'data-testid'?: string\n  helper?: string\n  unitError?: string\n  width?: ComponentProps<typeof Stack>['width']\n  placeholderUnit?: string\n  error?: boolean | string\n  success?: boolean | string\n  label?: string\n  labelInformation?: ReactNode\n  step?: number | string\n  dropdownAlign?: ComponentProps<typeof SelectInput>['dropdownAlign']\n} & Pick<\n  InputHTMLAttributes<HTMLInputElement>,\n  | 'onFocus'\n  | 'onBlur'\n  | 'name'\n  | 'id'\n  | 'placeholder'\n  | 'disabled'\n  | 'readOnly'\n  | 'required'\n  | 'autoFocus'\n  | 'onKeyDown'\n>\n\nexport const UnitInput = ({\n  id,\n  name = '',\n  max = Number.MAX_SAFE_INTEGER,\n  min = 0,\n  autoFocus = false,\n  size = 'large',\n  placeholder = '0',\n  placeholderUnit = 'Select item',\n  onChange,\n  onChangeUnitValue,\n  value,\n  unitValue,\n  selectInputWidth = '12.6rem',\n  disabled = false,\n  options,\n  className,\n  label,\n  step = 1,\n  error,\n  required,\n  helper,\n  unitError,\n  success,\n  'data-testid': dataTestId,\n  width,\n  labelInformation,\n  readOnly,\n  onFocus,\n  onBlur,\n  onKeyDown,\n  dropdownAlign,\n}: UnitInputProps) => {\n  const [val, setVal] = useState(value)\n  const localId = useId()\n  const sentiment = useMemo(() => {\n    if (error) {\n      return 'danger'\n    }\n    if (success) {\n      return 'success'\n    }\n\n    return 'neutral'\n  }, [error, success])\n\n  useEffect(() => {\n    if (value !== undefined) {\n      setVal(value)\n    }\n  }, [value])\n\n  return (\n    <Stack gap={0.5}>\n      {label || labelInformation ? (\n        <Label\n          labelDescription={labelInformation}\n          required={required}\n          size={size}\n          htmlFor={id ?? localId}\n        >\n          {label}\n        </Label>\n      ) : null}\n      <UnitInputWrapper\n        direction=\"row\"\n        data-testid={dataTestId}\n        data-size={size}\n        width={width}\n        data-success={!!success}\n        data-error={!!error}\n        data-disabled={!!disabled}\n        data-readonly={!!readOnly}\n        onFocus={onFocus}\n        onBlur={onBlur}\n        onKeyDown={onKeyDown}\n      >\n        <StyledNumberInputWrapper id=\"input-field\">\n          <StyledInput\n            type=\"number\"\n            aria-invalid={!!error}\n            autoFocus={autoFocus}\n            disabled={disabled}\n            name={`${name}-value`}\n            width={width}\n            id={id ?? localId}\n            value={val}\n            onChange={event => {\n              const numericValue = Number.parseInt(event.target.value, 10)\n              if (numericValue > max) {\n                setVal(max)\n                onChange?.(max)\n              } else if (numericValue < min) {\n                setVal(min)\n                onChange?.(min)\n              } else {\n                setVal(numericValue)\n                onChange?.(numericValue)\n              }\n            }}\n            placeholder={placeholder}\n            max={max}\n            readOnly={readOnly}\n            min={min}\n            step={step}\n            data-success={success}\n            data-error={error}\n            data-testid=\"unit-input\"\n            required={required}\n            className={className}\n            data-size={size}\n          />\n          {error ? <AlertCircleIcon sentiment=\"danger\" /> : null}\n          {success && !error ? <CheckCircleIcon sentiment=\"success\" /> : null}\n        </StyledNumberInputWrapper>\n        <CustomSelectInput\n          data-disabled={disabled}\n          id=\"unit\"\n          name={`${name}-unit`}\n          onChange={(newValue: string) => {\n            onChangeUnitValue?.(newValue)\n          }}\n          error={unitError}\n          value={unitValue}\n          options={options}\n          width={selectInputWidth}\n          searchable={false}\n          clearable={false}\n          placeholder={placeholderUnit}\n          disabled={disabled}\n          size={size}\n          multiselect={false}\n          readOnly={readOnly}\n          dropdownAlign={dropdownAlign}\n        />\n      </UnitInputWrapper>\n      {error || typeof success === 'string' || typeof helper === 'string' ? (\n        <Text\n          as=\"p\"\n          variant=\"caption\"\n          sentiment={sentiment}\n          disabled={disabled}\n          prominence={sentiment === 'neutral' ? 'weak' : 'default'}\n        >\n          {error || success || helper}\n        </Text>\n      ) : null}\n      {!error && !success && typeof helper !== 'string' && helper\n        ? helper\n        : null}\n    </Stack>\n  )\n}\n"]} */"));
44
44
  const UnitInputWrapper = /* @__PURE__ */ _styled(Stack, process.env.NODE_ENV === "production" ? {
45
45
  target: "ex81jph1"
46
46
  } : {
@@ -106,7 +106,7 @@ const UnitInputWrapper = /* @__PURE__ */ _styled(Stack, process.env.NODE_ENV ===
106
106
  theme
107
107
  }) => theme.colors.neutral.borderDisabled, ";cursor:not-allowed;&>", StyledNumberInputWrapper, "{border-right-color:", ({
108
108
  theme
109
- }) => theme.colors.neutral.borderDisabled, ";}}" + (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/UnitInput/index.tsx"],"names":[],"mappings":"AAgEE","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/UnitInput/index.tsx","sourcesContent":["'use client'\n\nimport styled from '@emotion/styled'\nimport { AlertCircleIcon, CheckCircleIcon } from '@ultraviolet/icons'\nimport type { ComponentProps, InputHTMLAttributes, ReactNode } from 'react'\nimport { useEffect, useId, useMemo, useState } from 'react'\nimport { Label } from '../Label'\nimport { SelectInput } from '../SelectInput'\nimport type { OptionType } from '../SelectInput/types'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\n\nconst INPUT_SIZE_HEIGHT = {\n  small: '400', // sizing theme tokens key\n  medium: '500',\n  large: '600',\n} as const\n\nconst StyledNumberInputWrapper = styled.div`\n  display: flex;\n  align-items: center;\n  flex-direction: row;\n  padding-right: ${({ theme }) => theme.space['2']};\n  border-right: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  width: 100%;\n  height: 100%;\n`\n\nconst StyledInput = styled.input`\n  flex: 1;\n  border: none;\n  outline: none;\n  height: 100%;\n  width: 100%;\n  padding-left: ${({ theme }) => theme.space['2']};\n  background: transparent;\n  color: ${({ theme }) => theme.colors.neutral.text};\n  &[data-size=\"small\"] {\n    padding-left: ${({ theme }) => theme.space['1']};\n  }\n\n  &[data-size=\"large\"] {\n    font-size: ${({ theme }) => theme.typography.body.fontSize};\n  }\n\n  &::placeholder {\n    color: ${({ theme }) => theme.colors.neutral.textWeak};\n  }\n\n  &:disabled {\n    cursor: not-allowed;\n\n    &::placeholder {\n      color: ${({ theme }) => theme.colors.neutral.textWeakDisabled};\n    }\n  }\n`\n\nconst UnitInputWrapper = styled(Stack)<{\n  'data-size': 'small' | 'medium' | 'large'\n  'data-success': boolean\n  'data-error': boolean\n  'data-disabled': boolean\n  'data-readonly': boolean\n}>`\n  border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  border-radius: ${({ theme }) => theme.radii.default};\n\n  &:not([data-disabled='true']):not([data-readonly='true']):not(\n      [data-success='true']\n    ):not([data-error='true']):focus,\n  :not([data-disabled='true']):not([data-readonly='true']):not(\n      [data-success='true']\n    ):not([data-error='true']):active {\n    box-shadow: ${({ theme }) => theme.shadows.focusPrimary};\n    border-color: ${({ theme }) => theme.colors.primary.borderHover};\n\n    & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.primary.border};\n    }\n  }\n\n  &:not([data-disabled='true']):not([data-readonly='true']):not(\n      [data-success='true']\n    ):not([data-error='true']):focus-within  {\n      border-color: ${({ theme }) => theme.colors.primary.borderHover};\n      & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.primary.border};\n    }\n    }\n\n  &:not([data-disabled='true']):not([data-error='true']):not(\n      [data-success='true']\n    ):not([data-readonly='true']):hover,\n  :not([data-disabled='true']):not([data-error='true']):not(\n      [data-success='true']\n    ):focus {\n    text-decoration: none;\n    border-color: ${({ theme }) => theme.colors.primary.border};\n    & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.primary.border};\n    }\n  }\n\n  &[data-readonly='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n    border-color: ${({ theme }) => theme.colors.neutral.border};\n    cursor: default;\n  }\n\n  &[data-readonly='true']:active {\n    border-color: ${({ theme }) => theme.colors.neutral.border};\n  }\n\n  &[data-size='small'] {\n    height: ${({ theme }) => theme.sizing[INPUT_SIZE_HEIGHT.small]};\n  }\n  &[data-size='medium'] {\n    height: ${({ theme }) => theme.sizing[INPUT_SIZE_HEIGHT.medium]};\n  }\n  &[data-size='large'] {\n    height: ${({ theme }) => theme.sizing[INPUT_SIZE_HEIGHT.large]};\n  }\n\n  &[data-success='true'] {\n    border: 1px solid ${({ theme }) => theme.colors.success.border};\n    & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.success.border};\n    }\n  }\n  &[data-success='true']:active {\n    border: 1px solid ${({ theme }) => theme.colors.success.border};\n    box-shadow: ${({ theme }) => theme.shadows.focusSuccess};\n\n    & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.success.border};\n    }\n  }\n\n  &[data-error='true'] {\n    border: 1px solid ${({ theme }) => theme.colors.danger.border};\n\n    & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.danger.border};\n    }\n\n    & > ${StyledNumberInputWrapper}:hover {\n      border-right-color: ${({ theme }) => theme.colors.danger.border};\n    }\n  }\n\n  &[data-error='true']:active {\n    border: 1px solid ${({ theme }) => theme.colors.danger.border};\n    box-shadow: ${({ theme }) => theme.shadows.focusDanger};\n\n    & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.danger.border};\n    }\n    & > ${StyledNumberInputWrapper}:hover {\n      border-right-color: ${({ theme }) => theme.colors.danger.border};\n    }\n  }\n\n  &[data-disabled='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    border-color: ${({ theme }) => theme.colors.neutral.borderDisabled};\n    cursor: not-allowed;\n\n    & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.neutral.borderDisabled};\n    }\n  }\n`\n\nconst CustomSelectInput = styled(SelectInput)<{\n  width?: number | string\n  maxWidth?: number | string\n  'data-disabled': boolean\n}>`\n  #unit {\n    border: none;\n    background: transparent;\n  }\n\n  ${({ width }) =>\n    width && `width: ${typeof width === 'string' ? width : `${width}px`};`}\n  ${({ maxWidth }) =>\n    maxWidth &&\n    `max-width: ${typeof maxWidth === 'string' ? maxWidth : `${maxWidth}px`};`}\n\n  #unit:focus,\n  #unit:active {\n    box-shadow: none;\n  }\n`\ntype UnitInputValue = { inputValue: number; unit: string }\n\ntype UnitInputProps = {\n  className?: string\n  max?: number\n  min?: number\n  value?: UnitInputValue['inputValue']\n  unitValue?: UnitInputValue['unit']\n  onChange?: (value: UnitInputValue['inputValue']) => void\n  onChangeUnitValue?: (values: string) => void\n  options: OptionType[]\n  selectInputWidth?: number | string\n  size?: 'small' | 'medium' | 'large'\n  'data-testid'?: string\n  helper?: string\n  unitError?: string\n  width?: ComponentProps<typeof Stack>['width']\n  placeholderUnit?: string\n  error?: boolean | string\n  success?: boolean | string\n  label?: string\n  labelInformation?: ReactNode\n  step?: number | string\n  dropdownAlign?: ComponentProps<typeof SelectInput>['dropdownAlign']\n} & Pick<\n  InputHTMLAttributes<HTMLInputElement>,\n  | 'onFocus'\n  | 'onBlur'\n  | 'name'\n  | 'id'\n  | 'placeholder'\n  | 'disabled'\n  | 'readOnly'\n  | 'required'\n  | 'autoFocus'\n  | 'onKeyDown'\n>\n\nexport const UnitInput = ({\n  id,\n  name = '',\n  max = Number.MAX_SAFE_INTEGER,\n  min = 0,\n  autoFocus = false,\n  size = 'large',\n  placeholder = '0',\n  placeholderUnit = 'Select item',\n  onChange,\n  onChangeUnitValue,\n  value,\n  unitValue,\n  selectInputWidth = '12.6rem',\n  disabled = false,\n  options,\n  className,\n  label,\n  step = 1,\n  error,\n  required,\n  helper,\n  unitError,\n  success,\n  'data-testid': dataTestId,\n  width,\n  labelInformation,\n  readOnly,\n  onFocus,\n  onBlur,\n  onKeyDown,\n  dropdownAlign,\n}: UnitInputProps) => {\n  const [val, setVal] = useState(value)\n  const localId = useId()\n  const sentiment = useMemo(() => {\n    if (error) {\n      return 'danger'\n    }\n    if (success) {\n      return 'success'\n    }\n\n    return 'neutral'\n  }, [error, success])\n\n  useEffect(() => {\n    if (value !== undefined) {\n      setVal(value)\n    }\n  }, [value])\n\n  return (\n    <Stack gap={0.5}>\n      {label || labelInformation ? (\n        <Label\n          labelDescription={labelInformation}\n          required={required}\n          size={size}\n          htmlFor={id ?? localId}\n        >\n          {label}\n        </Label>\n      ) : null}\n      <UnitInputWrapper\n        direction=\"row\"\n        data-testid={dataTestId}\n        data-size={size}\n        width={width}\n        data-success={!!success}\n        data-error={!!error}\n        data-disabled={!!disabled}\n        data-readonly={!!readOnly}\n        onFocus={onFocus}\n        onBlur={onBlur}\n        onKeyDown={onKeyDown}\n      >\n        <StyledNumberInputWrapper id=\"input-field\">\n          <StyledInput\n            type=\"number\"\n            aria-invalid={!!error}\n            autoFocus={autoFocus}\n            disabled={disabled}\n            name={`${name}-value`}\n            width={width}\n            id={id ? `${id}-value` : undefined}\n            value={val}\n            onChange={event => {\n              const numericValue = Number.parseInt(event.target.value, 10)\n              if (numericValue > max) {\n                setVal(max)\n                onChange?.(max)\n              } else if (numericValue < min) {\n                setVal(min)\n                onChange?.(min)\n              } else {\n                setVal(numericValue)\n                onChange?.(numericValue)\n              }\n            }}\n            placeholder={placeholder}\n            max={max}\n            readOnly={readOnly}\n            min={min}\n            step={step}\n            data-success={success}\n            data-error={error}\n            data-testid=\"unit-input\"\n            required={required}\n            className={className}\n            data-size={size}\n          />\n          {error ? <AlertCircleIcon sentiment=\"danger\" /> : null}\n          {success && !error ? <CheckCircleIcon sentiment=\"success\" /> : null}\n        </StyledNumberInputWrapper>\n        <CustomSelectInput\n          data-disabled={disabled}\n          id=\"unit\"\n          name={`${name}-unit`}\n          onChange={(newValue: string) => {\n            onChangeUnitValue?.(newValue)\n          }}\n          error={unitError}\n          value={unitValue}\n          options={options}\n          width={selectInputWidth}\n          searchable={false}\n          clearable={false}\n          placeholder={placeholderUnit}\n          disabled={disabled}\n          size={size}\n          multiselect={false}\n          readOnly={readOnly}\n          dropdownAlign={dropdownAlign}\n        />\n      </UnitInputWrapper>\n      {error || typeof success === 'string' || typeof helper === 'string' ? (\n        <Text\n          as=\"p\"\n          variant=\"caption\"\n          sentiment={sentiment}\n          disabled={disabled}\n          prominence={sentiment === 'neutral' ? 'weak' : 'default'}\n        >\n          {error || success || helper}\n        </Text>\n      ) : null}\n      {!error && !success && typeof helper !== 'string' && helper\n        ? helper\n        : null}\n    </Stack>\n  )\n}\n"]} */"));
109
+ }) => theme.colors.neutral.borderDisabled, ";}}" + (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/UnitInput/index.tsx"],"names":[],"mappings":"AAgEE","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/UnitInput/index.tsx","sourcesContent":["'use client'\n\nimport styled from '@emotion/styled'\nimport { AlertCircleIcon, CheckCircleIcon } from '@ultraviolet/icons'\nimport type { ComponentProps, InputHTMLAttributes, ReactNode } from 'react'\nimport { useEffect, useId, useMemo, useState } from 'react'\nimport { Label } from '../Label'\nimport { SelectInput } from '../SelectInput'\nimport type { OptionType } from '../SelectInput/types'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\n\nconst INPUT_SIZE_HEIGHT = {\n  small: '400', // sizing theme tokens key\n  medium: '500',\n  large: '600',\n} as const\n\nconst StyledNumberInputWrapper = styled.div`\n  display: flex;\n  align-items: center;\n  flex-direction: row;\n  padding-right: ${({ theme }) => theme.space['2']};\n  border-right: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  width: 100%;\n  height: 100%;\n`\n\nconst StyledInput = styled.input`\n  flex: 1;\n  border: none;\n  outline: none;\n  height: 100%;\n  width: 100%;\n  padding-left: ${({ theme }) => theme.space['2']};\n  background: transparent;\n  color: ${({ theme }) => theme.colors.neutral.text};\n  &[data-size=\"small\"] {\n    padding-left: ${({ theme }) => theme.space['1']};\n  }\n\n  &[data-size=\"large\"] {\n    font-size: ${({ theme }) => theme.typography.body.fontSize};\n  }\n\n  &::placeholder {\n    color: ${({ theme }) => theme.colors.neutral.textWeak};\n  }\n\n  &:disabled {\n    cursor: not-allowed;\n\n    &::placeholder {\n      color: ${({ theme }) => theme.colors.neutral.textWeakDisabled};\n    }\n  }\n`\n\nconst UnitInputWrapper = styled(Stack)<{\n  'data-size': 'small' | 'medium' | 'large'\n  'data-success': boolean\n  'data-error': boolean\n  'data-disabled': boolean\n  'data-readonly': boolean\n}>`\n  border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  border-radius: ${({ theme }) => theme.radii.default};\n\n  &:not([data-disabled='true']):not([data-readonly='true']):not(\n      [data-success='true']\n    ):not([data-error='true']):focus,\n  :not([data-disabled='true']):not([data-readonly='true']):not(\n      [data-success='true']\n    ):not([data-error='true']):active {\n    box-shadow: ${({ theme }) => theme.shadows.focusPrimary};\n    border-color: ${({ theme }) => theme.colors.primary.borderHover};\n\n    & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.primary.border};\n    }\n  }\n\n  &:not([data-disabled='true']):not([data-readonly='true']):not(\n      [data-success='true']\n    ):not([data-error='true']):focus-within  {\n      border-color: ${({ theme }) => theme.colors.primary.borderHover};\n      & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.primary.border};\n    }\n    }\n\n  &:not([data-disabled='true']):not([data-error='true']):not(\n      [data-success='true']\n    ):not([data-readonly='true']):hover,\n  :not([data-disabled='true']):not([data-error='true']):not(\n      [data-success='true']\n    ):focus {\n    text-decoration: none;\n    border-color: ${({ theme }) => theme.colors.primary.border};\n    & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.primary.border};\n    }\n  }\n\n  &[data-readonly='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n    border-color: ${({ theme }) => theme.colors.neutral.border};\n    cursor: default;\n  }\n\n  &[data-readonly='true']:active {\n    border-color: ${({ theme }) => theme.colors.neutral.border};\n  }\n\n  &[data-size='small'] {\n    height: ${({ theme }) => theme.sizing[INPUT_SIZE_HEIGHT.small]};\n  }\n  &[data-size='medium'] {\n    height: ${({ theme }) => theme.sizing[INPUT_SIZE_HEIGHT.medium]};\n  }\n  &[data-size='large'] {\n    height: ${({ theme }) => theme.sizing[INPUT_SIZE_HEIGHT.large]};\n  }\n\n  &[data-success='true'] {\n    border: 1px solid ${({ theme }) => theme.colors.success.border};\n    & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.success.border};\n    }\n  }\n  &[data-success='true']:active {\n    border: 1px solid ${({ theme }) => theme.colors.success.border};\n    box-shadow: ${({ theme }) => theme.shadows.focusSuccess};\n\n    & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.success.border};\n    }\n  }\n\n  &[data-error='true'] {\n    border: 1px solid ${({ theme }) => theme.colors.danger.border};\n\n    & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.danger.border};\n    }\n\n    & > ${StyledNumberInputWrapper}:hover {\n      border-right-color: ${({ theme }) => theme.colors.danger.border};\n    }\n  }\n\n  &[data-error='true']:active {\n    border: 1px solid ${({ theme }) => theme.colors.danger.border};\n    box-shadow: ${({ theme }) => theme.shadows.focusDanger};\n\n    & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.danger.border};\n    }\n    & > ${StyledNumberInputWrapper}:hover {\n      border-right-color: ${({ theme }) => theme.colors.danger.border};\n    }\n  }\n\n  &[data-disabled='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    border-color: ${({ theme }) => theme.colors.neutral.borderDisabled};\n    cursor: not-allowed;\n\n    & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.neutral.borderDisabled};\n    }\n  }\n`\n\nconst CustomSelectInput = styled(SelectInput)<{\n  width?: number | string\n  maxWidth?: number | string\n  'data-disabled': boolean\n}>`\n  #unit {\n    border: none;\n    background: transparent;\n  }\n\n  ${({ width }) =>\n    width && `width: ${typeof width === 'string' ? width : `${width}px`};`}\n  ${({ maxWidth }) =>\n    maxWidth &&\n    `max-width: ${typeof maxWidth === 'string' ? maxWidth : `${maxWidth}px`};`}\n\n  #unit:focus,\n  #unit:active {\n    box-shadow: none;\n  }\n`\ntype UnitInputValue = { inputValue: number; unit: string }\n\ntype UnitInputProps = {\n  className?: string\n  max?: number\n  min?: number\n  value?: UnitInputValue['inputValue']\n  unitValue?: UnitInputValue['unit']\n  onChange?: (value: UnitInputValue['inputValue']) => void\n  onChangeUnitValue?: (values: string) => void\n  options: OptionType[]\n  selectInputWidth?: number | string\n  size?: 'small' | 'medium' | 'large'\n  'data-testid'?: string\n  helper?: string\n  unitError?: string\n  width?: ComponentProps<typeof Stack>['width']\n  placeholderUnit?: string\n  error?: boolean | string\n  success?: boolean | string\n  label?: string\n  labelInformation?: ReactNode\n  step?: number | string\n  dropdownAlign?: ComponentProps<typeof SelectInput>['dropdownAlign']\n} & Pick<\n  InputHTMLAttributes<HTMLInputElement>,\n  | 'onFocus'\n  | 'onBlur'\n  | 'name'\n  | 'id'\n  | 'placeholder'\n  | 'disabled'\n  | 'readOnly'\n  | 'required'\n  | 'autoFocus'\n  | 'onKeyDown'\n>\n\nexport const UnitInput = ({\n  id,\n  name = '',\n  max = Number.MAX_SAFE_INTEGER,\n  min = 0,\n  autoFocus = false,\n  size = 'large',\n  placeholder = '0',\n  placeholderUnit = 'Select item',\n  onChange,\n  onChangeUnitValue,\n  value,\n  unitValue,\n  selectInputWidth = '12.6rem',\n  disabled = false,\n  options,\n  className,\n  label,\n  step = 1,\n  error,\n  required,\n  helper,\n  unitError,\n  success,\n  'data-testid': dataTestId,\n  width,\n  labelInformation,\n  readOnly,\n  onFocus,\n  onBlur,\n  onKeyDown,\n  dropdownAlign,\n}: UnitInputProps) => {\n  const [val, setVal] = useState(value)\n  const localId = useId()\n  const sentiment = useMemo(() => {\n    if (error) {\n      return 'danger'\n    }\n    if (success) {\n      return 'success'\n    }\n\n    return 'neutral'\n  }, [error, success])\n\n  useEffect(() => {\n    if (value !== undefined) {\n      setVal(value)\n    }\n  }, [value])\n\n  return (\n    <Stack gap={0.5}>\n      {label || labelInformation ? (\n        <Label\n          labelDescription={labelInformation}\n          required={required}\n          size={size}\n          htmlFor={id ?? localId}\n        >\n          {label}\n        </Label>\n      ) : null}\n      <UnitInputWrapper\n        direction=\"row\"\n        data-testid={dataTestId}\n        data-size={size}\n        width={width}\n        data-success={!!success}\n        data-error={!!error}\n        data-disabled={!!disabled}\n        data-readonly={!!readOnly}\n        onFocus={onFocus}\n        onBlur={onBlur}\n        onKeyDown={onKeyDown}\n      >\n        <StyledNumberInputWrapper id=\"input-field\">\n          <StyledInput\n            type=\"number\"\n            aria-invalid={!!error}\n            autoFocus={autoFocus}\n            disabled={disabled}\n            name={`${name}-value`}\n            width={width}\n            id={id ?? localId}\n            value={val}\n            onChange={event => {\n              const numericValue = Number.parseInt(event.target.value, 10)\n              if (numericValue > max) {\n                setVal(max)\n                onChange?.(max)\n              } else if (numericValue < min) {\n                setVal(min)\n                onChange?.(min)\n              } else {\n                setVal(numericValue)\n                onChange?.(numericValue)\n              }\n            }}\n            placeholder={placeholder}\n            max={max}\n            readOnly={readOnly}\n            min={min}\n            step={step}\n            data-success={success}\n            data-error={error}\n            data-testid=\"unit-input\"\n            required={required}\n            className={className}\n            data-size={size}\n          />\n          {error ? <AlertCircleIcon sentiment=\"danger\" /> : null}\n          {success && !error ? <CheckCircleIcon sentiment=\"success\" /> : null}\n        </StyledNumberInputWrapper>\n        <CustomSelectInput\n          data-disabled={disabled}\n          id=\"unit\"\n          name={`${name}-unit`}\n          onChange={(newValue: string) => {\n            onChangeUnitValue?.(newValue)\n          }}\n          error={unitError}\n          value={unitValue}\n          options={options}\n          width={selectInputWidth}\n          searchable={false}\n          clearable={false}\n          placeholder={placeholderUnit}\n          disabled={disabled}\n          size={size}\n          multiselect={false}\n          readOnly={readOnly}\n          dropdownAlign={dropdownAlign}\n        />\n      </UnitInputWrapper>\n      {error || typeof success === 'string' || typeof helper === 'string' ? (\n        <Text\n          as=\"p\"\n          variant=\"caption\"\n          sentiment={sentiment}\n          disabled={disabled}\n          prominence={sentiment === 'neutral' ? 'weak' : 'default'}\n        >\n          {error || success || helper}\n        </Text>\n      ) : null}\n      {!error && !success && typeof helper !== 'string' && helper\n        ? helper\n        : null}\n    </Stack>\n  )\n}\n"]} */"));
110
110
  const CustomSelectInput = /* @__PURE__ */ _styled(SelectInput, process.env.NODE_ENV === "production" ? {
111
111
  target: "ex81jph0"
112
112
  } : {
@@ -116,7 +116,7 @@ const CustomSelectInput = /* @__PURE__ */ _styled(SelectInput, process.env.NODE_
116
116
  width
117
117
  }) => width && `width: ${typeof width === "string" ? width : `${width}px`};`, " ", ({
118
118
  maxWidth
119
- }) => maxWidth && `max-width: ${typeof maxWidth === "string" ? maxWidth : `${maxWidth}px`};`, " #unit:focus,#unit:active{box-shadow:none;}" + (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/UnitInput/index.tsx"],"names":[],"mappings":"AAkLE","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/UnitInput/index.tsx","sourcesContent":["'use client'\n\nimport styled from '@emotion/styled'\nimport { AlertCircleIcon, CheckCircleIcon } from '@ultraviolet/icons'\nimport type { ComponentProps, InputHTMLAttributes, ReactNode } from 'react'\nimport { useEffect, useId, useMemo, useState } from 'react'\nimport { Label } from '../Label'\nimport { SelectInput } from '../SelectInput'\nimport type { OptionType } from '../SelectInput/types'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\n\nconst INPUT_SIZE_HEIGHT = {\n  small: '400', // sizing theme tokens key\n  medium: '500',\n  large: '600',\n} as const\n\nconst StyledNumberInputWrapper = styled.div`\n  display: flex;\n  align-items: center;\n  flex-direction: row;\n  padding-right: ${({ theme }) => theme.space['2']};\n  border-right: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  width: 100%;\n  height: 100%;\n`\n\nconst StyledInput = styled.input`\n  flex: 1;\n  border: none;\n  outline: none;\n  height: 100%;\n  width: 100%;\n  padding-left: ${({ theme }) => theme.space['2']};\n  background: transparent;\n  color: ${({ theme }) => theme.colors.neutral.text};\n  &[data-size=\"small\"] {\n    padding-left: ${({ theme }) => theme.space['1']};\n  }\n\n  &[data-size=\"large\"] {\n    font-size: ${({ theme }) => theme.typography.body.fontSize};\n  }\n\n  &::placeholder {\n    color: ${({ theme }) => theme.colors.neutral.textWeak};\n  }\n\n  &:disabled {\n    cursor: not-allowed;\n\n    &::placeholder {\n      color: ${({ theme }) => theme.colors.neutral.textWeakDisabled};\n    }\n  }\n`\n\nconst UnitInputWrapper = styled(Stack)<{\n  'data-size': 'small' | 'medium' | 'large'\n  'data-success': boolean\n  'data-error': boolean\n  'data-disabled': boolean\n  'data-readonly': boolean\n}>`\n  border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  border-radius: ${({ theme }) => theme.radii.default};\n\n  &:not([data-disabled='true']):not([data-readonly='true']):not(\n      [data-success='true']\n    ):not([data-error='true']):focus,\n  :not([data-disabled='true']):not([data-readonly='true']):not(\n      [data-success='true']\n    ):not([data-error='true']):active {\n    box-shadow: ${({ theme }) => theme.shadows.focusPrimary};\n    border-color: ${({ theme }) => theme.colors.primary.borderHover};\n\n    & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.primary.border};\n    }\n  }\n\n  &:not([data-disabled='true']):not([data-readonly='true']):not(\n      [data-success='true']\n    ):not([data-error='true']):focus-within  {\n      border-color: ${({ theme }) => theme.colors.primary.borderHover};\n      & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.primary.border};\n    }\n    }\n\n  &:not([data-disabled='true']):not([data-error='true']):not(\n      [data-success='true']\n    ):not([data-readonly='true']):hover,\n  :not([data-disabled='true']):not([data-error='true']):not(\n      [data-success='true']\n    ):focus {\n    text-decoration: none;\n    border-color: ${({ theme }) => theme.colors.primary.border};\n    & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.primary.border};\n    }\n  }\n\n  &[data-readonly='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n    border-color: ${({ theme }) => theme.colors.neutral.border};\n    cursor: default;\n  }\n\n  &[data-readonly='true']:active {\n    border-color: ${({ theme }) => theme.colors.neutral.border};\n  }\n\n  &[data-size='small'] {\n    height: ${({ theme }) => theme.sizing[INPUT_SIZE_HEIGHT.small]};\n  }\n  &[data-size='medium'] {\n    height: ${({ theme }) => theme.sizing[INPUT_SIZE_HEIGHT.medium]};\n  }\n  &[data-size='large'] {\n    height: ${({ theme }) => theme.sizing[INPUT_SIZE_HEIGHT.large]};\n  }\n\n  &[data-success='true'] {\n    border: 1px solid ${({ theme }) => theme.colors.success.border};\n    & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.success.border};\n    }\n  }\n  &[data-success='true']:active {\n    border: 1px solid ${({ theme }) => theme.colors.success.border};\n    box-shadow: ${({ theme }) => theme.shadows.focusSuccess};\n\n    & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.success.border};\n    }\n  }\n\n  &[data-error='true'] {\n    border: 1px solid ${({ theme }) => theme.colors.danger.border};\n\n    & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.danger.border};\n    }\n\n    & > ${StyledNumberInputWrapper}:hover {\n      border-right-color: ${({ theme }) => theme.colors.danger.border};\n    }\n  }\n\n  &[data-error='true']:active {\n    border: 1px solid ${({ theme }) => theme.colors.danger.border};\n    box-shadow: ${({ theme }) => theme.shadows.focusDanger};\n\n    & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.danger.border};\n    }\n    & > ${StyledNumberInputWrapper}:hover {\n      border-right-color: ${({ theme }) => theme.colors.danger.border};\n    }\n  }\n\n  &[data-disabled='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    border-color: ${({ theme }) => theme.colors.neutral.borderDisabled};\n    cursor: not-allowed;\n\n    & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.neutral.borderDisabled};\n    }\n  }\n`\n\nconst CustomSelectInput = styled(SelectInput)<{\n  width?: number | string\n  maxWidth?: number | string\n  'data-disabled': boolean\n}>`\n  #unit {\n    border: none;\n    background: transparent;\n  }\n\n  ${({ width }) =>\n    width && `width: ${typeof width === 'string' ? width : `${width}px`};`}\n  ${({ maxWidth }) =>\n    maxWidth &&\n    `max-width: ${typeof maxWidth === 'string' ? maxWidth : `${maxWidth}px`};`}\n\n  #unit:focus,\n  #unit:active {\n    box-shadow: none;\n  }\n`\ntype UnitInputValue = { inputValue: number; unit: string }\n\ntype UnitInputProps = {\n  className?: string\n  max?: number\n  min?: number\n  value?: UnitInputValue['inputValue']\n  unitValue?: UnitInputValue['unit']\n  onChange?: (value: UnitInputValue['inputValue']) => void\n  onChangeUnitValue?: (values: string) => void\n  options: OptionType[]\n  selectInputWidth?: number | string\n  size?: 'small' | 'medium' | 'large'\n  'data-testid'?: string\n  helper?: string\n  unitError?: string\n  width?: ComponentProps<typeof Stack>['width']\n  placeholderUnit?: string\n  error?: boolean | string\n  success?: boolean | string\n  label?: string\n  labelInformation?: ReactNode\n  step?: number | string\n  dropdownAlign?: ComponentProps<typeof SelectInput>['dropdownAlign']\n} & Pick<\n  InputHTMLAttributes<HTMLInputElement>,\n  | 'onFocus'\n  | 'onBlur'\n  | 'name'\n  | 'id'\n  | 'placeholder'\n  | 'disabled'\n  | 'readOnly'\n  | 'required'\n  | 'autoFocus'\n  | 'onKeyDown'\n>\n\nexport const UnitInput = ({\n  id,\n  name = '',\n  max = Number.MAX_SAFE_INTEGER,\n  min = 0,\n  autoFocus = false,\n  size = 'large',\n  placeholder = '0',\n  placeholderUnit = 'Select item',\n  onChange,\n  onChangeUnitValue,\n  value,\n  unitValue,\n  selectInputWidth = '12.6rem',\n  disabled = false,\n  options,\n  className,\n  label,\n  step = 1,\n  error,\n  required,\n  helper,\n  unitError,\n  success,\n  'data-testid': dataTestId,\n  width,\n  labelInformation,\n  readOnly,\n  onFocus,\n  onBlur,\n  onKeyDown,\n  dropdownAlign,\n}: UnitInputProps) => {\n  const [val, setVal] = useState(value)\n  const localId = useId()\n  const sentiment = useMemo(() => {\n    if (error) {\n      return 'danger'\n    }\n    if (success) {\n      return 'success'\n    }\n\n    return 'neutral'\n  }, [error, success])\n\n  useEffect(() => {\n    if (value !== undefined) {\n      setVal(value)\n    }\n  }, [value])\n\n  return (\n    <Stack gap={0.5}>\n      {label || labelInformation ? (\n        <Label\n          labelDescription={labelInformation}\n          required={required}\n          size={size}\n          htmlFor={id ?? localId}\n        >\n          {label}\n        </Label>\n      ) : null}\n      <UnitInputWrapper\n        direction=\"row\"\n        data-testid={dataTestId}\n        data-size={size}\n        width={width}\n        data-success={!!success}\n        data-error={!!error}\n        data-disabled={!!disabled}\n        data-readonly={!!readOnly}\n        onFocus={onFocus}\n        onBlur={onBlur}\n        onKeyDown={onKeyDown}\n      >\n        <StyledNumberInputWrapper id=\"input-field\">\n          <StyledInput\n            type=\"number\"\n            aria-invalid={!!error}\n            autoFocus={autoFocus}\n            disabled={disabled}\n            name={`${name}-value`}\n            width={width}\n            id={id ? `${id}-value` : undefined}\n            value={val}\n            onChange={event => {\n              const numericValue = Number.parseInt(event.target.value, 10)\n              if (numericValue > max) {\n                setVal(max)\n                onChange?.(max)\n              } else if (numericValue < min) {\n                setVal(min)\n                onChange?.(min)\n              } else {\n                setVal(numericValue)\n                onChange?.(numericValue)\n              }\n            }}\n            placeholder={placeholder}\n            max={max}\n            readOnly={readOnly}\n            min={min}\n            step={step}\n            data-success={success}\n            data-error={error}\n            data-testid=\"unit-input\"\n            required={required}\n            className={className}\n            data-size={size}\n          />\n          {error ? <AlertCircleIcon sentiment=\"danger\" /> : null}\n          {success && !error ? <CheckCircleIcon sentiment=\"success\" /> : null}\n        </StyledNumberInputWrapper>\n        <CustomSelectInput\n          data-disabled={disabled}\n          id=\"unit\"\n          name={`${name}-unit`}\n          onChange={(newValue: string) => {\n            onChangeUnitValue?.(newValue)\n          }}\n          error={unitError}\n          value={unitValue}\n          options={options}\n          width={selectInputWidth}\n          searchable={false}\n          clearable={false}\n          placeholder={placeholderUnit}\n          disabled={disabled}\n          size={size}\n          multiselect={false}\n          readOnly={readOnly}\n          dropdownAlign={dropdownAlign}\n        />\n      </UnitInputWrapper>\n      {error || typeof success === 'string' || typeof helper === 'string' ? (\n        <Text\n          as=\"p\"\n          variant=\"caption\"\n          sentiment={sentiment}\n          disabled={disabled}\n          prominence={sentiment === 'neutral' ? 'weak' : 'default'}\n        >\n          {error || success || helper}\n        </Text>\n      ) : null}\n      {!error && !success && typeof helper !== 'string' && helper\n        ? helper\n        : null}\n    </Stack>\n  )\n}\n"]} */"));
119
+ }) => maxWidth && `max-width: ${typeof maxWidth === "string" ? maxWidth : `${maxWidth}px`};`, " #unit:focus,#unit:active{box-shadow:none;}" + (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/UnitInput/index.tsx"],"names":[],"mappings":"AAkLE","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/UnitInput/index.tsx","sourcesContent":["'use client'\n\nimport styled from '@emotion/styled'\nimport { AlertCircleIcon, CheckCircleIcon } from '@ultraviolet/icons'\nimport type { ComponentProps, InputHTMLAttributes, ReactNode } from 'react'\nimport { useEffect, useId, useMemo, useState } from 'react'\nimport { Label } from '../Label'\nimport { SelectInput } from '../SelectInput'\nimport type { OptionType } from '../SelectInput/types'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\n\nconst INPUT_SIZE_HEIGHT = {\n  small: '400', // sizing theme tokens key\n  medium: '500',\n  large: '600',\n} as const\n\nconst StyledNumberInputWrapper = styled.div`\n  display: flex;\n  align-items: center;\n  flex-direction: row;\n  padding-right: ${({ theme }) => theme.space['2']};\n  border-right: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  width: 100%;\n  height: 100%;\n`\n\nconst StyledInput = styled.input`\n  flex: 1;\n  border: none;\n  outline: none;\n  height: 100%;\n  width: 100%;\n  padding-left: ${({ theme }) => theme.space['2']};\n  background: transparent;\n  color: ${({ theme }) => theme.colors.neutral.text};\n  &[data-size=\"small\"] {\n    padding-left: ${({ theme }) => theme.space['1']};\n  }\n\n  &[data-size=\"large\"] {\n    font-size: ${({ theme }) => theme.typography.body.fontSize};\n  }\n\n  &::placeholder {\n    color: ${({ theme }) => theme.colors.neutral.textWeak};\n  }\n\n  &:disabled {\n    cursor: not-allowed;\n\n    &::placeholder {\n      color: ${({ theme }) => theme.colors.neutral.textWeakDisabled};\n    }\n  }\n`\n\nconst UnitInputWrapper = styled(Stack)<{\n  'data-size': 'small' | 'medium' | 'large'\n  'data-success': boolean\n  'data-error': boolean\n  'data-disabled': boolean\n  'data-readonly': boolean\n}>`\n  border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  border-radius: ${({ theme }) => theme.radii.default};\n\n  &:not([data-disabled='true']):not([data-readonly='true']):not(\n      [data-success='true']\n    ):not([data-error='true']):focus,\n  :not([data-disabled='true']):not([data-readonly='true']):not(\n      [data-success='true']\n    ):not([data-error='true']):active {\n    box-shadow: ${({ theme }) => theme.shadows.focusPrimary};\n    border-color: ${({ theme }) => theme.colors.primary.borderHover};\n\n    & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.primary.border};\n    }\n  }\n\n  &:not([data-disabled='true']):not([data-readonly='true']):not(\n      [data-success='true']\n    ):not([data-error='true']):focus-within  {\n      border-color: ${({ theme }) => theme.colors.primary.borderHover};\n      & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.primary.border};\n    }\n    }\n\n  &:not([data-disabled='true']):not([data-error='true']):not(\n      [data-success='true']\n    ):not([data-readonly='true']):hover,\n  :not([data-disabled='true']):not([data-error='true']):not(\n      [data-success='true']\n    ):focus {\n    text-decoration: none;\n    border-color: ${({ theme }) => theme.colors.primary.border};\n    & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.primary.border};\n    }\n  }\n\n  &[data-readonly='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n    border-color: ${({ theme }) => theme.colors.neutral.border};\n    cursor: default;\n  }\n\n  &[data-readonly='true']:active {\n    border-color: ${({ theme }) => theme.colors.neutral.border};\n  }\n\n  &[data-size='small'] {\n    height: ${({ theme }) => theme.sizing[INPUT_SIZE_HEIGHT.small]};\n  }\n  &[data-size='medium'] {\n    height: ${({ theme }) => theme.sizing[INPUT_SIZE_HEIGHT.medium]};\n  }\n  &[data-size='large'] {\n    height: ${({ theme }) => theme.sizing[INPUT_SIZE_HEIGHT.large]};\n  }\n\n  &[data-success='true'] {\n    border: 1px solid ${({ theme }) => theme.colors.success.border};\n    & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.success.border};\n    }\n  }\n  &[data-success='true']:active {\n    border: 1px solid ${({ theme }) => theme.colors.success.border};\n    box-shadow: ${({ theme }) => theme.shadows.focusSuccess};\n\n    & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.success.border};\n    }\n  }\n\n  &[data-error='true'] {\n    border: 1px solid ${({ theme }) => theme.colors.danger.border};\n\n    & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.danger.border};\n    }\n\n    & > ${StyledNumberInputWrapper}:hover {\n      border-right-color: ${({ theme }) => theme.colors.danger.border};\n    }\n  }\n\n  &[data-error='true']:active {\n    border: 1px solid ${({ theme }) => theme.colors.danger.border};\n    box-shadow: ${({ theme }) => theme.shadows.focusDanger};\n\n    & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.danger.border};\n    }\n    & > ${StyledNumberInputWrapper}:hover {\n      border-right-color: ${({ theme }) => theme.colors.danger.border};\n    }\n  }\n\n  &[data-disabled='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    border-color: ${({ theme }) => theme.colors.neutral.borderDisabled};\n    cursor: not-allowed;\n\n    & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.neutral.borderDisabled};\n    }\n  }\n`\n\nconst CustomSelectInput = styled(SelectInput)<{\n  width?: number | string\n  maxWidth?: number | string\n  'data-disabled': boolean\n}>`\n  #unit {\n    border: none;\n    background: transparent;\n  }\n\n  ${({ width }) =>\n    width && `width: ${typeof width === 'string' ? width : `${width}px`};`}\n  ${({ maxWidth }) =>\n    maxWidth &&\n    `max-width: ${typeof maxWidth === 'string' ? maxWidth : `${maxWidth}px`};`}\n\n  #unit:focus,\n  #unit:active {\n    box-shadow: none;\n  }\n`\ntype UnitInputValue = { inputValue: number; unit: string }\n\ntype UnitInputProps = {\n  className?: string\n  max?: number\n  min?: number\n  value?: UnitInputValue['inputValue']\n  unitValue?: UnitInputValue['unit']\n  onChange?: (value: UnitInputValue['inputValue']) => void\n  onChangeUnitValue?: (values: string) => void\n  options: OptionType[]\n  selectInputWidth?: number | string\n  size?: 'small' | 'medium' | 'large'\n  'data-testid'?: string\n  helper?: string\n  unitError?: string\n  width?: ComponentProps<typeof Stack>['width']\n  placeholderUnit?: string\n  error?: boolean | string\n  success?: boolean | string\n  label?: string\n  labelInformation?: ReactNode\n  step?: number | string\n  dropdownAlign?: ComponentProps<typeof SelectInput>['dropdownAlign']\n} & Pick<\n  InputHTMLAttributes<HTMLInputElement>,\n  | 'onFocus'\n  | 'onBlur'\n  | 'name'\n  | 'id'\n  | 'placeholder'\n  | 'disabled'\n  | 'readOnly'\n  | 'required'\n  | 'autoFocus'\n  | 'onKeyDown'\n>\n\nexport const UnitInput = ({\n  id,\n  name = '',\n  max = Number.MAX_SAFE_INTEGER,\n  min = 0,\n  autoFocus = false,\n  size = 'large',\n  placeholder = '0',\n  placeholderUnit = 'Select item',\n  onChange,\n  onChangeUnitValue,\n  value,\n  unitValue,\n  selectInputWidth = '12.6rem',\n  disabled = false,\n  options,\n  className,\n  label,\n  step = 1,\n  error,\n  required,\n  helper,\n  unitError,\n  success,\n  'data-testid': dataTestId,\n  width,\n  labelInformation,\n  readOnly,\n  onFocus,\n  onBlur,\n  onKeyDown,\n  dropdownAlign,\n}: UnitInputProps) => {\n  const [val, setVal] = useState(value)\n  const localId = useId()\n  const sentiment = useMemo(() => {\n    if (error) {\n      return 'danger'\n    }\n    if (success) {\n      return 'success'\n    }\n\n    return 'neutral'\n  }, [error, success])\n\n  useEffect(() => {\n    if (value !== undefined) {\n      setVal(value)\n    }\n  }, [value])\n\n  return (\n    <Stack gap={0.5}>\n      {label || labelInformation ? (\n        <Label\n          labelDescription={labelInformation}\n          required={required}\n          size={size}\n          htmlFor={id ?? localId}\n        >\n          {label}\n        </Label>\n      ) : null}\n      <UnitInputWrapper\n        direction=\"row\"\n        data-testid={dataTestId}\n        data-size={size}\n        width={width}\n        data-success={!!success}\n        data-error={!!error}\n        data-disabled={!!disabled}\n        data-readonly={!!readOnly}\n        onFocus={onFocus}\n        onBlur={onBlur}\n        onKeyDown={onKeyDown}\n      >\n        <StyledNumberInputWrapper id=\"input-field\">\n          <StyledInput\n            type=\"number\"\n            aria-invalid={!!error}\n            autoFocus={autoFocus}\n            disabled={disabled}\n            name={`${name}-value`}\n            width={width}\n            id={id ?? localId}\n            value={val}\n            onChange={event => {\n              const numericValue = Number.parseInt(event.target.value, 10)\n              if (numericValue > max) {\n                setVal(max)\n                onChange?.(max)\n              } else if (numericValue < min) {\n                setVal(min)\n                onChange?.(min)\n              } else {\n                setVal(numericValue)\n                onChange?.(numericValue)\n              }\n            }}\n            placeholder={placeholder}\n            max={max}\n            readOnly={readOnly}\n            min={min}\n            step={step}\n            data-success={success}\n            data-error={error}\n            data-testid=\"unit-input\"\n            required={required}\n            className={className}\n            data-size={size}\n          />\n          {error ? <AlertCircleIcon sentiment=\"danger\" /> : null}\n          {success && !error ? <CheckCircleIcon sentiment=\"success\" /> : null}\n        </StyledNumberInputWrapper>\n        <CustomSelectInput\n          data-disabled={disabled}\n          id=\"unit\"\n          name={`${name}-unit`}\n          onChange={(newValue: string) => {\n            onChangeUnitValue?.(newValue)\n          }}\n          error={unitError}\n          value={unitValue}\n          options={options}\n          width={selectInputWidth}\n          searchable={false}\n          clearable={false}\n          placeholder={placeholderUnit}\n          disabled={disabled}\n          size={size}\n          multiselect={false}\n          readOnly={readOnly}\n          dropdownAlign={dropdownAlign}\n        />\n      </UnitInputWrapper>\n      {error || typeof success === 'string' || typeof helper === 'string' ? (\n        <Text\n          as=\"p\"\n          variant=\"caption\"\n          sentiment={sentiment}\n          disabled={disabled}\n          prominence={sentiment === 'neutral' ? 'weak' : 'default'}\n        >\n          {error || success || helper}\n        </Text>\n      ) : null}\n      {!error && !success && typeof helper !== 'string' && helper\n        ? helper\n        : null}\n    </Stack>\n  )\n}\n"]} */"));
120
120
  const UnitInput = ({
121
121
  id,
122
122
  name = "",
@@ -170,7 +170,7 @@ const UnitInput = ({
170
170
  label || labelInformation ? /* @__PURE__ */ jsx(Label, { labelDescription: labelInformation, required, size, htmlFor: id ?? localId, children: label }) : null,
171
171
  /* @__PURE__ */ jsxs(UnitInputWrapper, { direction: "row", "data-testid": dataTestId, "data-size": size, width, "data-success": !!success, "data-error": !!error, "data-disabled": !!disabled, "data-readonly": !!readOnly, onFocus, onBlur, onKeyDown, children: [
172
172
  /* @__PURE__ */ jsxs(StyledNumberInputWrapper, { id: "input-field", children: [
173
- /* @__PURE__ */ jsx(StyledInput, { type: "number", "aria-invalid": !!error, autoFocus, disabled, name: `${name}-value`, width, id: id ? `${id}-value` : void 0, value: val, onChange: (event) => {
173
+ /* @__PURE__ */ jsx(StyledInput, { type: "number", "aria-invalid": !!error, autoFocus, disabled, name: `${name}-value`, width, id: id ?? localId, value: val, onChange: (event) => {
174
174
  const numericValue = Number.parseInt(event.target.value, 10);
175
175
  if (numericValue > max) {
176
176
  setVal(max);
@@ -33,7 +33,6 @@ export { NotificationContainer, notification } from './Notification';
33
33
  export { NumberInput } from './NumberInput';
34
34
  export { Pagination } from './Pagination';
35
35
  export { PasswordCheck } from './PasswordCheck';
36
- export { PasswordStrengthMeter } from './PasswordStrengthMeter';
37
36
  export { PieChart } from './PieChart';
38
37
  export { Popover } from './Popover';
39
38
  export { Popup } from './Popup';
package/dist/index.cjs CHANGED
@@ -43,45 +43,44 @@ const index$w = require("./components/Notification/index.cjs");
43
43
  const index$x = require("./components/NumberInput/index.cjs");
44
44
  const index$y = require("./components/Pagination/index.cjs");
45
45
  const index$z = require("./components/PasswordCheck/index.cjs");
46
- const index$A = require("./components/PasswordStrengthMeter/index.cjs");
47
- const index$B = require("./components/PieChart/index.cjs");
48
- const index$C = require("./components/Popover/index.cjs");
49
- const index$D = require("./components/Popup/index.cjs");
50
- const index$E = require("./components/ProgressBar/index.cjs");
51
- const index$F = require("./components/Radio/index.cjs");
52
- const index$G = require("./components/RadioGroup/index.cjs");
53
- const index$H = require("./components/Row/index.cjs");
54
- const index$I = require("./components/SearchInput/index.cjs");
55
- const index$J = require("./components/SelectableCard/index.cjs");
56
- const index$K = require("./components/SelectableCardGroup/index.cjs");
57
- const index$L = require("./components/SelectableCardOptionGroup/index.cjs");
58
- const index$M = require("./components/SelectInput/index.cjs");
59
- const index$N = require("./components/Separator/index.cjs");
60
- const index$O = require("./components/Skeleton/index.cjs");
61
- const index$P = require("./components/Slider/index.cjs");
62
- const index$Q = require("./components/Snippet/index.cjs");
63
- const index$R = require("./components/Stack/index.cjs");
64
- const index$S = require("./components/Status/index.cjs");
65
- const index$T = require("./components/StepList/index.cjs");
66
- const index$U = require("./components/Stepper/index.cjs");
67
- const index$V = require("./components/SwitchButton/index.cjs");
68
- const index$W = require("./components/Table/index.cjs");
69
- const index$X = require("./components/Tabs/index.cjs");
70
- const index$Y = require("./components/Tag/index.cjs");
71
- const index$Z = require("./components/TagInput/index.cjs");
72
- const index$_ = require("./components/TagList/index.cjs");
73
- const index$$ = require("./components/Text/index.cjs");
74
- const index$10 = require("./components/TextArea/index.cjs");
75
- const index$11 = require("./components/TextInput/index.cjs");
76
- const index$12 = require("./components/TimeInput/index.cjs");
77
- const index$13 = require("./components/Toaster/index.cjs");
78
- const index$14 = require("./components/Toggle/index.cjs");
79
- const index$15 = require("./components/ToggleGroup/index.cjs");
80
- const index$16 = require("./components/Tooltip/index.cjs");
81
- const index$17 = require("./components/UnitInput/index.cjs");
82
- const index$18 = require("./components/VerificationCode/index.cjs");
83
- const index$19 = require("./components/Drawer/index.cjs");
84
- const index$1a = require("./components/InfiniteScroll/index.cjs");
46
+ const index$A = require("./components/PieChart/index.cjs");
47
+ const index$B = require("./components/Popover/index.cjs");
48
+ const index$C = require("./components/Popup/index.cjs");
49
+ const index$D = require("./components/ProgressBar/index.cjs");
50
+ const index$E = require("./components/Radio/index.cjs");
51
+ const index$F = require("./components/RadioGroup/index.cjs");
52
+ const index$G = require("./components/Row/index.cjs");
53
+ const index$H = require("./components/SearchInput/index.cjs");
54
+ const index$I = require("./components/SelectableCard/index.cjs");
55
+ const index$J = require("./components/SelectableCardGroup/index.cjs");
56
+ const index$K = require("./components/SelectableCardOptionGroup/index.cjs");
57
+ const index$L = require("./components/SelectInput/index.cjs");
58
+ const index$M = require("./components/Separator/index.cjs");
59
+ const index$N = require("./components/Skeleton/index.cjs");
60
+ const index$O = require("./components/Slider/index.cjs");
61
+ const index$P = require("./components/Snippet/index.cjs");
62
+ const index$Q = require("./components/Stack/index.cjs");
63
+ const index$R = require("./components/Status/index.cjs");
64
+ const index$S = require("./components/StepList/index.cjs");
65
+ const index$T = require("./components/Stepper/index.cjs");
66
+ const index$U = require("./components/SwitchButton/index.cjs");
67
+ const index$V = require("./components/Table/index.cjs");
68
+ const index$W = require("./components/Tabs/index.cjs");
69
+ const index$X = require("./components/Tag/index.cjs");
70
+ const index$Y = require("./components/TagInput/index.cjs");
71
+ const index$Z = require("./components/TagList/index.cjs");
72
+ const index$_ = require("./components/Text/index.cjs");
73
+ const index$$ = require("./components/TextArea/index.cjs");
74
+ const index$10 = require("./components/TextInput/index.cjs");
75
+ const index$11 = require("./components/TimeInput/index.cjs");
76
+ const index$12 = require("./components/Toaster/index.cjs");
77
+ const index$13 = require("./components/Toggle/index.cjs");
78
+ const index$14 = require("./components/ToggleGroup/index.cjs");
79
+ const index$15 = require("./components/Tooltip/index.cjs");
80
+ const index$16 = require("./components/UnitInput/index.cjs");
81
+ const index$17 = require("./components/VerificationCode/index.cjs");
82
+ const index$18 = require("./components/Drawer/index.cjs");
83
+ const index$19 = require("./components/InfiniteScroll/index.cjs");
85
84
  exports.extendTheme = index.extendTheme;
86
85
  exports.ThemeRegistry = ThemeRegistry;
87
86
  exports.Breakpoint = Breakpoint;
@@ -161,43 +160,42 @@ exports.notification = index$w.notification;
161
160
  exports.NumberInput = index$x.NumberInput;
162
161
  exports.Pagination = index$y.Pagination;
163
162
  exports.PasswordCheck = index$z.PasswordCheck;
164
- exports.PasswordStrengthMeter = index$A.PasswordStrengthMeter;
165
- exports.PieChart = index$B.PieChart;
166
- exports.Popover = index$C.Popover;
167
- exports.Popup = index$D.Popup;
168
- exports.ProgressBar = index$E.ProgressBar;
169
- exports.Radio = index$F.Radio;
170
- exports.RadioGroup = index$G.RadioGroup;
171
- exports.Row = index$H.Row;
172
- exports.SearchInput = index$I.SearchInput;
173
- exports.SelectableCard = index$J.SelectableCard;
174
- exports.SelectableCardGroup = index$K.SelectableCardGroup;
175
- exports.SelectableCardOptionGroup = index$L.SelectableCardOptionGroup;
176
- exports.SelectInput = index$M.SelectInput;
177
- exports.Separator = index$N.Separator;
178
- exports.Skeleton = index$O.Skeleton;
179
- exports.Slider = index$P.Slider;
180
- exports.Snippet = index$Q.Snippet;
181
- exports.Stack = index$R.Stack;
182
- exports.Status = index$S.Status;
183
- exports.StepList = index$T.StepList;
184
- exports.Stepper = index$U.Stepper;
185
- exports.SwitchButton = index$V.SwitchButton;
186
- exports.Table = index$W.Table;
187
- exports.Tabs = index$X.Tabs;
188
- exports.Tag = index$Y.Tag;
189
- exports.TagInput = index$Z.TagInput;
190
- exports.TagList = index$_.TagList;
191
- exports.Text = index$$.Text;
192
- exports.TextArea = index$10.TextArea;
193
- exports.TextInput = index$11.TextInput;
194
- exports.TimeInput = index$12.TimeInput;
195
- exports.ToastContainer = index$13.ToastContainer;
196
- exports.toast = index$13.toast;
197
- exports.Toggle = index$14.Toggle;
198
- exports.ToggleGroup = index$15.ToggleGroup;
199
- exports.Tooltip = index$16.Tooltip;
200
- exports.UnitInput = index$17.UnitInput;
201
- exports.VerificationCode = index$18.VerificationCode;
202
- exports.Drawer = index$19.Drawer;
203
- exports.InfiniteScroll = index$1a.InfiniteScroll;
163
+ exports.PieChart = index$A.PieChart;
164
+ exports.Popover = index$B.Popover;
165
+ exports.Popup = index$C.Popup;
166
+ exports.ProgressBar = index$D.ProgressBar;
167
+ exports.Radio = index$E.Radio;
168
+ exports.RadioGroup = index$F.RadioGroup;
169
+ exports.Row = index$G.Row;
170
+ exports.SearchInput = index$H.SearchInput;
171
+ exports.SelectableCard = index$I.SelectableCard;
172
+ exports.SelectableCardGroup = index$J.SelectableCardGroup;
173
+ exports.SelectableCardOptionGroup = index$K.SelectableCardOptionGroup;
174
+ exports.SelectInput = index$L.SelectInput;
175
+ exports.Separator = index$M.Separator;
176
+ exports.Skeleton = index$N.Skeleton;
177
+ exports.Slider = index$O.Slider;
178
+ exports.Snippet = index$P.Snippet;
179
+ exports.Stack = index$Q.Stack;
180
+ exports.Status = index$R.Status;
181
+ exports.StepList = index$S.StepList;
182
+ exports.Stepper = index$T.Stepper;
183
+ exports.SwitchButton = index$U.SwitchButton;
184
+ exports.Table = index$V.Table;
185
+ exports.Tabs = index$W.Tabs;
186
+ exports.Tag = index$X.Tag;
187
+ exports.TagInput = index$Y.TagInput;
188
+ exports.TagList = index$Z.TagList;
189
+ exports.Text = index$_.Text;
190
+ exports.TextArea = index$$.TextArea;
191
+ exports.TextInput = index$10.TextInput;
192
+ exports.TimeInput = index$11.TimeInput;
193
+ exports.ToastContainer = index$12.ToastContainer;
194
+ exports.toast = index$12.toast;
195
+ exports.Toggle = index$13.Toggle;
196
+ exports.ToggleGroup = index$14.ToggleGroup;
197
+ exports.Tooltip = index$15.Tooltip;
198
+ exports.UnitInput = index$16.UnitInput;
199
+ exports.VerificationCode = index$17.VerificationCode;
200
+ exports.Drawer = index$18.Drawer;
201
+ exports.InfiniteScroll = index$19.InfiniteScroll;
package/dist/index.js CHANGED
@@ -41,7 +41,6 @@ import { NotificationContainer, notification } from "./components/Notification/i
41
41
  import { NumberInput } from "./components/NumberInput/index.js";
42
42
  import { Pagination } from "./components/Pagination/index.js";
43
43
  import { PasswordCheck } from "./components/PasswordCheck/index.js";
44
- import { PasswordStrengthMeter } from "./components/PasswordStrengthMeter/index.js";
45
44
  import { PieChart } from "./components/PieChart/index.js";
46
45
  import { Popover } from "./components/Popover/index.js";
47
46
  import { Popup } from "./components/Popup/index.js";
@@ -120,7 +119,6 @@ export {
120
119
  NumberInput,
121
120
  Pagination,
122
121
  PasswordCheck,
123
- PasswordStrengthMeter,
124
122
  PieChart,
125
123
  Popover,
126
124
  Popup,
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@ultraviolet/ui",
3
- "version": "2.0.0-beta.11",
3
+ "version": "2.0.0-beta.13",
4
4
  "description": "Ultraviolet UI",
5
5
  "homepage": "https://github.com/scaleway/ultraviolet#readme",
6
6
  "repository": {
@@ -84,7 +84,7 @@
84
84
  "react-toastify": "11.0.5",
85
85
  "react-use-clipboard": "1.0.9",
86
86
  "@ultraviolet/themes": "2.0.0-beta.2",
87
- "@ultraviolet/icons": "4.0.0-beta.5"
87
+ "@ultraviolet/icons": "4.0.0-beta.6"
88
88
  },
89
89
  "scripts": {
90
90
  "type:generate": "tsc --declaration -p tsconfig.build.json",