@ultraviolet/ui 1.92.4 → 1.92.6

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.
@@ -1,7 +1,7 @@
1
1
  import { jsxs, jsx } from "@emotion/react/jsx-runtime";
2
2
  import _styled from "@emotion/styled/base";
3
3
  import { AlertCircleIcon, CheckCircleIcon } from "@ultraviolet/icons";
4
- import { useState, useId, useMemo } from "react";
4
+ import { useState, useId, useMemo, useEffect } from "react";
5
5
  import { Label } from "../Label/index.js";
6
6
  import { SelectInputV2 } from "../SelectInputV2/index.js";
7
7
  import { Stack } from "../Stack/index.js";
@@ -24,7 +24,7 @@ const StyledNumberInputWrapper = /* @__PURE__ */ _styled("div", process.env.NODE
24
24
  theme
25
25
  }) => theme.space["2"], ";border-right:1px solid ", ({
26
26
  theme
27
- }) => 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":"AAgB2C","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/UnitInput/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { AlertCircleIcon, CheckCircleIcon } from '@ultraviolet/icons'\nimport type { ComponentProps, InputHTMLAttributes, ReactNode } from 'react'\nimport { useId, useMemo, useState } from 'react'\nimport { Label } from '../Label'\nimport { SelectInputV2 } from '../SelectInputV2'\nimport type { OptionType } from '../SelectInputV2/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  font-size: ${({ theme }) => theme.typography.bodySmall.fontSize};\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  background-color: ${({ theme }) => theme.colors.neutral.background};\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`\nconst SelectInputWrapper = styled.div<{\n  width: number | string\n}>`\n${({ width }) => width && `width: ${typeof width === 'number' ? `${width}px` : width};`}\ndisplay: flex;\n`\n\nconst CustomSelectInput = styled(SelectInputV2)<{\n  'data-disabled': boolean\n}>`\n  #unit {\n    border: none;\n    background: transparent;\n  }\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} & 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}: 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  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        <SelectInputWrapper width={selectInputWidth}>\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            searchable={false}\n            clearable={false}\n            placeholder={placeholderUnit}\n            disabled={disabled || options.length === 1}\n            size={size}\n            multiselect={false}\n            readOnly={readOnly}\n          />\n        </SelectInputWrapper>\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"]} */"));
27
+ }) => 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":"AAgB2C","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/UnitInput/index.tsx","sourcesContent":["import 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 { SelectInputV2 } from '../SelectInputV2'\nimport type { OptionType } from '../SelectInputV2/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  font-size: ${({ theme }) => theme.typography.bodySmall.fontSize};\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  background-color: ${({ theme }) => theme.colors.neutral.background};\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`\nconst SelectInputWrapper = styled.div<{\n  width: number | string\n}>`\n${({ width }) => width && `width: ${typeof width === 'number' ? `${width}px` : width};`}\ndisplay: flex;\n`\n\nconst CustomSelectInput = styled(SelectInputV2)<{\n  'data-disabled': boolean\n}>`\n  #unit {\n    border: none;\n    background: transparent;\n  }\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 SelectInputV2>['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        <SelectInputWrapper width={selectInputWidth}>\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            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        </SelectInputWrapper>\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"]} */"));
28
28
  const StyledInput = /* @__PURE__ */ _styled("input", process.env.NODE_ENV === "production" ? {
29
29
  target: "ex81jph3"
30
30
  } : {
@@ -44,7 +44,7 @@ const StyledInput = /* @__PURE__ */ _styled("input", process.env.NODE_ENV === "p
44
44
  theme
45
45
  }) => theme.colors.neutral.textWeak, ";}&:disabled{cursor:not-allowed;&::placeholder{color:", ({
46
46
  theme
47
- }) => 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":"AA0BgC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/UnitInput/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { AlertCircleIcon, CheckCircleIcon } from '@ultraviolet/icons'\nimport type { ComponentProps, InputHTMLAttributes, ReactNode } from 'react'\nimport { useId, useMemo, useState } from 'react'\nimport { Label } from '../Label'\nimport { SelectInputV2 } from '../SelectInputV2'\nimport type { OptionType } from '../SelectInputV2/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  font-size: ${({ theme }) => theme.typography.bodySmall.fontSize};\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  background-color: ${({ theme }) => theme.colors.neutral.background};\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`\nconst SelectInputWrapper = styled.div<{\n  width: number | string\n}>`\n${({ width }) => width && `width: ${typeof width === 'number' ? `${width}px` : width};`}\ndisplay: flex;\n`\n\nconst CustomSelectInput = styled(SelectInputV2)<{\n  'data-disabled': boolean\n}>`\n  #unit {\n    border: none;\n    background: transparent;\n  }\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} & 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}: 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  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        <SelectInputWrapper width={selectInputWidth}>\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            searchable={false}\n            clearable={false}\n            placeholder={placeholderUnit}\n            disabled={disabled || options.length === 1}\n            size={size}\n            multiselect={false}\n            readOnly={readOnly}\n          />\n        </SelectInputWrapper>\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"]} */"));
47
+ }) => 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":"AA0BgC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/UnitInput/index.tsx","sourcesContent":["import 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 { SelectInputV2 } from '../SelectInputV2'\nimport type { OptionType } from '../SelectInputV2/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  font-size: ${({ theme }) => theme.typography.bodySmall.fontSize};\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  background-color: ${({ theme }) => theme.colors.neutral.background};\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`\nconst SelectInputWrapper = styled.div<{\n  width: number | string\n}>`\n${({ width }) => width && `width: ${typeof width === 'number' ? `${width}px` : width};`}\ndisplay: flex;\n`\n\nconst CustomSelectInput = styled(SelectInputV2)<{\n  'data-disabled': boolean\n}>`\n  #unit {\n    border: none;\n    background: transparent;\n  }\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 SelectInputV2>['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        <SelectInputWrapper width={selectInputWidth}>\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            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        </SelectInputWrapper>\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"]} */"));
48
48
  const UnitInputWrapper = /* @__PURE__ */ _styled(Stack, process.env.NODE_ENV === "production" ? {
49
49
  target: "ex81jph2"
50
50
  } : {
@@ -112,7 +112,7 @@ const UnitInputWrapper = /* @__PURE__ */ _styled(Stack, process.env.NODE_ENV ===
112
112
  theme
113
113
  }) => theme.colors.neutral.borderDisabled, ";cursor:not-allowed;&>", StyledNumberInputWrapper, "{border-right-color:", ({
114
114
  theme
115
- }) => 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":"AA+DE","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/UnitInput/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { AlertCircleIcon, CheckCircleIcon } from '@ultraviolet/icons'\nimport type { ComponentProps, InputHTMLAttributes, ReactNode } from 'react'\nimport { useId, useMemo, useState } from 'react'\nimport { Label } from '../Label'\nimport { SelectInputV2 } from '../SelectInputV2'\nimport type { OptionType } from '../SelectInputV2/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  font-size: ${({ theme }) => theme.typography.bodySmall.fontSize};\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  background-color: ${({ theme }) => theme.colors.neutral.background};\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`\nconst SelectInputWrapper = styled.div<{\n  width: number | string\n}>`\n${({ width }) => width && `width: ${typeof width === 'number' ? `${width}px` : width};`}\ndisplay: flex;\n`\n\nconst CustomSelectInput = styled(SelectInputV2)<{\n  'data-disabled': boolean\n}>`\n  #unit {\n    border: none;\n    background: transparent;\n  }\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} & 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}: 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  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        <SelectInputWrapper width={selectInputWidth}>\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            searchable={false}\n            clearable={false}\n            placeholder={placeholderUnit}\n            disabled={disabled || options.length === 1}\n            size={size}\n            multiselect={false}\n            readOnly={readOnly}\n          />\n        </SelectInputWrapper>\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"]} */"));
115
+ }) => 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":"AA+DE","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/UnitInput/index.tsx","sourcesContent":["import 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 { SelectInputV2 } from '../SelectInputV2'\nimport type { OptionType } from '../SelectInputV2/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  font-size: ${({ theme }) => theme.typography.bodySmall.fontSize};\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  background-color: ${({ theme }) => theme.colors.neutral.background};\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`\nconst SelectInputWrapper = styled.div<{\n  width: number | string\n}>`\n${({ width }) => width && `width: ${typeof width === 'number' ? `${width}px` : width};`}\ndisplay: flex;\n`\n\nconst CustomSelectInput = styled(SelectInputV2)<{\n  'data-disabled': boolean\n}>`\n  #unit {\n    border: none;\n    background: transparent;\n  }\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 SelectInputV2>['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        <SelectInputWrapper width={selectInputWidth}>\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            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        </SelectInputWrapper>\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"]} */"));
116
116
  const SelectInputWrapper = /* @__PURE__ */ _styled("div", process.env.NODE_ENV === "production" ? {
117
117
  target: "ex81jph1"
118
118
  } : {
@@ -120,7 +120,7 @@ const SelectInputWrapper = /* @__PURE__ */ _styled("div", process.env.NODE_ENV =
120
120
  label: "SelectInputWrapper"
121
121
  })(({
122
122
  width
123
- }) => width && `width: ${typeof width === "number" ? `${width}px` : width};`, " display:flex;" + (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":"AA+KE","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/UnitInput/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { AlertCircleIcon, CheckCircleIcon } from '@ultraviolet/icons'\nimport type { ComponentProps, InputHTMLAttributes, ReactNode } from 'react'\nimport { useId, useMemo, useState } from 'react'\nimport { Label } from '../Label'\nimport { SelectInputV2 } from '../SelectInputV2'\nimport type { OptionType } from '../SelectInputV2/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  font-size: ${({ theme }) => theme.typography.bodySmall.fontSize};\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  background-color: ${({ theme }) => theme.colors.neutral.background};\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`\nconst SelectInputWrapper = styled.div<{\n  width: number | string\n}>`\n${({ width }) => width && `width: ${typeof width === 'number' ? `${width}px` : width};`}\ndisplay: flex;\n`\n\nconst CustomSelectInput = styled(SelectInputV2)<{\n  'data-disabled': boolean\n}>`\n  #unit {\n    border: none;\n    background: transparent;\n  }\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} & 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}: 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  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        <SelectInputWrapper width={selectInputWidth}>\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            searchable={false}\n            clearable={false}\n            placeholder={placeholderUnit}\n            disabled={disabled || options.length === 1}\n            size={size}\n            multiselect={false}\n            readOnly={readOnly}\n          />\n        </SelectInputWrapper>\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"]} */"));
123
+ }) => width && `width: ${typeof width === "number" ? `${width}px` : width};`, " display:flex;" + (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":"AA+KE","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/UnitInput/index.tsx","sourcesContent":["import 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 { SelectInputV2 } from '../SelectInputV2'\nimport type { OptionType } from '../SelectInputV2/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  font-size: ${({ theme }) => theme.typography.bodySmall.fontSize};\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  background-color: ${({ theme }) => theme.colors.neutral.background};\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`\nconst SelectInputWrapper = styled.div<{\n  width: number | string\n}>`\n${({ width }) => width && `width: ${typeof width === 'number' ? `${width}px` : width};`}\ndisplay: flex;\n`\n\nconst CustomSelectInput = styled(SelectInputV2)<{\n  'data-disabled': boolean\n}>`\n  #unit {\n    border: none;\n    background: transparent;\n  }\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 SelectInputV2>['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        <SelectInputWrapper width={selectInputWidth}>\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            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        </SelectInputWrapper>\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"]} */"));
124
124
  const CustomSelectInput = /* @__PURE__ */ _styled(SelectInputV2, process.env.NODE_ENV === "production" ? {
125
125
  target: "ex81jph0"
126
126
  } : {
@@ -131,7 +131,7 @@ const CustomSelectInput = /* @__PURE__ */ _styled(SelectInputV2, process.env.NOD
131
131
  styles: "#unit{border:none;background:transparent;}#unit:focus,#unit:active{box-shadow:none;}"
132
132
  } : {
133
133
  name: "5y7rrb",
134
- styles: "#unit{border:none;background:transparent;}#unit:focus,#unit:active{box-shadow:none;}/*# 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":"AAsLE","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/UnitInput/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { AlertCircleIcon, CheckCircleIcon } from '@ultraviolet/icons'\nimport type { ComponentProps, InputHTMLAttributes, ReactNode } from 'react'\nimport { useId, useMemo, useState } from 'react'\nimport { Label } from '../Label'\nimport { SelectInputV2 } from '../SelectInputV2'\nimport type { OptionType } from '../SelectInputV2/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  font-size: ${({ theme }) => theme.typography.bodySmall.fontSize};\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  background-color: ${({ theme }) => theme.colors.neutral.background};\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`\nconst SelectInputWrapper = styled.div<{\n  width: number | string\n}>`\n${({ width }) => width && `width: ${typeof width === 'number' ? `${width}px` : width};`}\ndisplay: flex;\n`\n\nconst CustomSelectInput = styled(SelectInputV2)<{\n  'data-disabled': boolean\n}>`\n  #unit {\n    border: none;\n    background: transparent;\n  }\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} & 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}: 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  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        <SelectInputWrapper width={selectInputWidth}>\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            searchable={false}\n            clearable={false}\n            placeholder={placeholderUnit}\n            disabled={disabled || options.length === 1}\n            size={size}\n            multiselect={false}\n            readOnly={readOnly}\n          />\n        </SelectInputWrapper>\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"]} */",
134
+ styles: "#unit{border:none;background:transparent;}#unit:focus,#unit:active{box-shadow:none;}/*# 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":"AAsLE","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/UnitInput/index.tsx","sourcesContent":["import 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 { SelectInputV2 } from '../SelectInputV2'\nimport type { OptionType } from '../SelectInputV2/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  font-size: ${({ theme }) => theme.typography.bodySmall.fontSize};\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  background-color: ${({ theme }) => theme.colors.neutral.background};\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`\nconst SelectInputWrapper = styled.div<{\n  width: number | string\n}>`\n${({ width }) => width && `width: ${typeof width === 'number' ? `${width}px` : width};`}\ndisplay: flex;\n`\n\nconst CustomSelectInput = styled(SelectInputV2)<{\n  'data-disabled': boolean\n}>`\n  #unit {\n    border: none;\n    background: transparent;\n  }\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 SelectInputV2>['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        <SelectInputWrapper width={selectInputWidth}>\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            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        </SelectInputWrapper>\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"]} */",
135
135
  toString: _EMOTION_STRINGIFIED_CSS_ERROR__
136
136
  });
137
137
  const UnitInput = ({
@@ -164,7 +164,8 @@ const UnitInput = ({
164
164
  readOnly,
165
165
  onFocus,
166
166
  onBlur,
167
- onKeyDown
167
+ onKeyDown,
168
+ dropdownAlign
168
169
  }) => {
169
170
  const [val, setVal] = useState(value);
170
171
  const localId = useId();
@@ -177,6 +178,11 @@ const UnitInput = ({
177
178
  }
178
179
  return "neutral";
179
180
  }, [error, success]);
181
+ useEffect(() => {
182
+ if (value !== void 0) {
183
+ setVal(value);
184
+ }
185
+ }, [value]);
180
186
  return /* @__PURE__ */ jsxs(Stack, { gap: 0.5, children: [
181
187
  label || labelInformation ? /* @__PURE__ */ jsx(Label, { labelDescription: labelInformation, required, size, htmlFor: id ?? localId, children: label }) : null,
182
188
  /* @__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: [
@@ -200,7 +206,7 @@ const UnitInput = ({
200
206
  " ",
201
207
  /* @__PURE__ */ jsx(SelectInputWrapper, { width: selectInputWidth, children: /* @__PURE__ */ jsx(CustomSelectInput, { "data-disabled": disabled, id: "unit", name: `${name}-unit`, onChange: (newValue) => {
202
208
  onChangeUnitValue?.(newValue);
203
- }, error: unitError, value: unitValue, options, searchable: false, clearable: false, placeholder: placeholderUnit, disabled: disabled || options.length === 1, size, multiselect: false, readOnly }) })
209
+ }, error: unitError, value: unitValue, options, searchable: false, clearable: false, placeholder: placeholderUnit, disabled, size, multiselect: false, readOnly, dropdownAlign }) })
204
210
  ] }),
205
211
  error || typeof success === "string" || typeof helper === "string" ? /* @__PURE__ */ jsx(Text, { as: "p", variant: "caption", sentiment, disabled, prominence: sentiment === "neutral" ? "weak" : "default", children: error || success || helper }) : null,
206
212
  !error && !success && typeof helper !== "string" && helper ? helper : null
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@ultraviolet/ui",
3
- "version": "1.92.4",
3
+ "version": "1.92.6",
4
4
  "description": "Ultraviolet UI",
5
5
  "homepage": "https://github.com/scaleway/ultraviolet#readme",
6
6
  "repository": {
@@ -83,8 +83,8 @@
83
83
  "react-toastify": "11.0.5",
84
84
  "react-use-clipboard": "1.0.9",
85
85
  "reakit": "1.3.11",
86
- "@ultraviolet/icons": "3.12.5",
87
- "@ultraviolet/themes": "1.17.0"
86
+ "@ultraviolet/themes": "1.17.0",
87
+ "@ultraviolet/icons": "3.12.5"
88
88
  },
89
89
  "scripts": {
90
90
  "type:generate": "tsc --declaration -p tsconfig.build.json",