@ultraviolet/ui 1.61.1 → 1.63.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (50) hide show
  1. package/dist/components/Badge/index.cjs +9 -5
  2. package/dist/components/Badge/index.d.ts +5 -0
  3. package/dist/components/Badge/index.js +10 -6
  4. package/dist/components/ExpandableCard/components/Title.cjs +11 -0
  5. package/dist/components/ExpandableCard/components/Title.d.ts +7 -0
  6. package/dist/components/ExpandableCard/components/Title.js +11 -0
  7. package/dist/components/ExpandableCard/index.cjs +87 -0
  8. package/dist/components/ExpandableCard/index.d.ts +22 -0
  9. package/dist/components/ExpandableCard/index.js +85 -0
  10. package/dist/components/Modal/ModalContent.cjs +53 -0
  11. package/dist/components/Modal/ModalContent.d.ts +17 -0
  12. package/dist/components/Modal/ModalContent.js +51 -0
  13. package/dist/components/Modal/ModalProvider.cjs +40 -0
  14. package/dist/components/Modal/ModalProvider.d.ts +14 -0
  15. package/dist/components/Modal/ModalProvider.js +40 -0
  16. package/dist/components/Modal/components/Dialog.cjs +162 -0
  17. package/dist/components/Modal/components/Dialog.d.ts +13 -0
  18. package/dist/components/Modal/components/Dialog.js +160 -0
  19. package/dist/components/Modal/{Disclosure.d.ts → components/Disclosure.d.ts} +1 -1
  20. package/dist/components/Modal/index.cjs +5 -29
  21. package/dist/components/Modal/index.js +6 -28
  22. package/dist/components/Slider/components/DoubleSlider.cjs +53 -63
  23. package/dist/components/Slider/components/DoubleSlider.d.ts +1 -1
  24. package/dist/components/Slider/components/DoubleSlider.js +55 -65
  25. package/dist/components/Slider/components/Options.cjs +29 -7
  26. package/dist/components/Slider/components/Options.d.ts +23 -2
  27. package/dist/components/Slider/components/Options.js +26 -6
  28. package/dist/components/Slider/components/SingleSlider.cjs +32 -42
  29. package/dist/components/Slider/components/SingleSlider.d.ts +1 -1
  30. package/dist/components/Slider/components/SingleSlider.js +33 -43
  31. package/dist/components/Slider/index.cjs +4 -5
  32. package/dist/components/Slider/index.d.ts +1 -1
  33. package/dist/components/Slider/index.js +4 -5
  34. package/dist/components/Slider/styles.cjs +15 -52
  35. package/dist/components/Slider/styles.d.ts +1 -31
  36. package/dist/components/Slider/styles.js +13 -50
  37. package/dist/components/Slider/types.d.ts +2 -18
  38. package/dist/components/TextArea/index.cjs +5 -5
  39. package/dist/components/TextArea/index.js +5 -5
  40. package/dist/components/UnitInput/index.cjs +5 -5
  41. package/dist/components/UnitInput/index.js +5 -5
  42. package/dist/components/index.d.ts +1 -0
  43. package/dist/index.cjs +110 -108
  44. package/dist/index.js +2 -0
  45. package/package.json +4 -4
  46. package/dist/components/Modal/Dialog.cjs +0 -144
  47. package/dist/components/Modal/Dialog.d.ts +0 -11
  48. package/dist/components/Modal/Dialog.js +0 -142
  49. /package/dist/components/Modal/{Disclosure.cjs → components/Disclosure.cjs} +0 -0
  50. /package/dist/components/Modal/{Disclosure.js → components/Disclosure.js} +0 -0
@@ -19,7 +19,7 @@ const StyledNumberInputWrapper = /* @__PURE__ */ _styled("div", process.env.NODE
19
19
  theme
20
20
  }) => theme.space["2"], ";border-right:1px solid ", ({
21
21
  theme
22
- }) => 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":"AAe2C","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/UnitInput/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { Icon } from '@ultraviolet/icons'\nimport type { ComponentProps, InputHTMLAttributes, ReactNode } from 'react'\nimport { useId, useMemo, useState } from 'react'\nimport { SelectInputV2 } from '../SelectInputV2'\nimport type { OptionType } from '../SelectInputV2/types'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\n\nconst INPUT_SIZE_HEIGHT: Record<string, number> = {\n  large: 48,\n  medium: 40,\n  small: 32,\n}\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  padding-left: ${({ theme }) => theme.space['2']};\n  background: transparent;\n  color: ${({ theme }) => theme.colors.neutral.text};\n  &::placeholder {\n    color: ${({ theme }) => theme.colors.neutral.textWeak};\n  }\n\n  &:disabled {\n    cursor: not-allowed;\n\n    &::placeholder {\n      color: ${({ theme }) => theme.colors.neutral.textWeakDisabled};\n    }\n  }\n`\n\nconst UnitInputWrapper = styled(Stack)<{\n  'data-size': 'small' | 'medium' | 'large'\n  'data-success': boolean\n  'data-error': boolean\n  'data-disabled': boolean\n  'data-readonly': boolean\n}>`\n  border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  border-radius: ${({ theme }) => theme.radii.default};\n\n  &:not([data-disabled='true']):not([data-readonly='true']):not(\n      [data-success='true']\n    ):not([data-error='true']):focus,\n  :not([data-disabled='true']):not([data-readonly='true']):not(\n      [data-success='true']\n    ):not([data-error='true'])active {\n    box-shadow: ${({ theme }) => theme.shadows.focusPrimary};\n    border-color: ${({ theme }) => theme.colors.primary.borderHover};\n\n    & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.primary.border};\n    }\n  }\n\n  &:not([data-disabled='true']):not([data-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: ${INPUT_SIZE_HEIGHT['small']}px;\n  }\n  &[data-size='medium'] {\n    height: ${INPUT_SIZE_HEIGHT['medium']}px;\n  }\n  &[data-size='large'] {\n    height: ${INPUT_SIZE_HEIGHT['large']}px;\n  }\n\n  &[data-success='true'] {\n    border: 1px solid ${({ theme }) => theme.colors.success.border};\n    & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.success.border};\n    }\n  }\n  &[data-success='true']:active {\n    border: 1px solid ${({ theme }) => theme.colors.success.border};\n    box-shadow: ${({ theme }) => theme.shadows.focusSuccess};\n\n    & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.success.border};\n    }\n  }\n\n  &[data-error='true'] {\n    border: 1px solid ${({ theme }) => theme.colors.danger.border};\n\n    & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.danger.border};\n    }\n\n    & > ${StyledNumberInputWrapper}:hover {\n      border-right-color: ${({ theme }) => theme.colors.danger.border};\n    }\n  }\n\n  &[data-error='true']:active {\n    border: 1px solid ${({ theme }) => theme.colors.danger.border};\n    box-shadow: ${({ theme }) => theme.shadows.focusDanger};\n\n    & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.danger.border};\n    }\n    & > ${StyledNumberInputWrapper}:hover {\n      border-right-color: ${({ theme }) => theme.colors.danger.border};\n    }\n  }\n\n  &[data-disabled='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    border-color: ${({ theme }) => theme.colors.neutral.borderDisabled};\n    cursor: not-allowed;\n\n    & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.neutral.borderDisabled};\n    }\n  }\n`\n\nconst CustomSelectInput = styled(SelectInputV2)<{\n  width?: number\n  'data-disabled': boolean\n}>`\n  #unit {\n    border: none;\n    background: transparent;\n  }\n\n  ${({ width }) => width && `width: ${width}px;`}\n\n  #unit:focus,\n  #unit:active {\n    box-shadow: none;\n  }\n`\ntype UnitInputValue = { inputValue: number; unit: string }\n\ntype UnitInputProps = {\n  className?: string\n  max?: number\n  min?: number\n  value?: UnitInputValue['inputValue']\n  unitValue?: UnitInputValue['unit']\n  onChange?: (value: UnitInputValue['inputValue']) => void\n  onChangeUnitValue?: (values: string) => void\n  options: OptionType[]\n  selectInputWidth?: number\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 = 200,\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 ? (\n        <Stack direction=\"row\" gap={0.5} alignItems=\"center\">\n          <Text\n            as=\"label\"\n            variant={size === 'large' ? 'bodyStrong' : 'bodySmallStrong'}\n            disabled={disabled}\n            htmlFor={id ?? localId}\n          >\n            {label}\n          </Text>\n          {required ? (\n            <Icon name=\"asterisk\" sentiment=\"danger\" size={8} />\n          ) : null}\n          {labelInformation ?? null}\n        </Stack>\n      ) : null}\n      <UnitInputWrapper\n        direction=\"row\"\n        data-testid={dataTestId}\n        data-size={size}\n        width={width}\n        id={id}\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            data-readOnly={readOnly}\n            name={`${name}-value`}\n            id={id ? `${id}-value` : undefined}\n            value={val}\n            onChange={event => {\n              const numericValue = Number.parseInt(event.target.value, 10)\n              if (numericValue > max) {\n                setVal(max)\n                onChange?.(max)\n              } else if (numericValue < min) {\n                setVal(min)\n                onChange?.(min)\n              } else {\n                setVal(numericValue)\n                onChange?.(numericValue)\n              }\n            }}\n            placeholder={placeholder}\n            max={max}\n            readOnly={readOnly}\n            min={min}\n            step={step}\n            data-success={success}\n            data-error={error}\n            data-testid=\"unit-input\"\n            required={required}\n            className={className}\n          />\n          {error ? <Icon name=\"alert\" sentiment=\"danger\" /> : null}\n          {success && !error ? (\n            <Icon name=\"checkbox-circle-outline\" sentiment=\"success\" />\n          ) : null}\n        </StyledNumberInputWrapper>\n        <CustomSelectInput\n          width={selectInputWidth}\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      </UnitInputWrapper>\n      {error || typeof success === 'string' || typeof helper === 'string' ? (\n        <Text\n          as=\"p\"\n          variant=\"caption\"\n          sentiment={sentiment}\n          disabled={disabled}\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"]} */"));
22
+ }) => 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":"AAe2C","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/UnitInput/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { Icon } from '@ultraviolet/icons'\nimport type { ComponentProps, InputHTMLAttributes, ReactNode } from 'react'\nimport { useId, useMemo, useState } from 'react'\nimport { SelectInputV2 } from '../SelectInputV2'\nimport type { OptionType } from '../SelectInputV2/types'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\n\nconst INPUT_SIZE_HEIGHT: Record<string, number> = {\n  large: 48,\n  medium: 40,\n  small: 32,\n}\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  padding-left: ${({ theme }) => theme.space['2']};\n  background: transparent;\n  color: ${({ theme }) => theme.colors.neutral.text};\n  &::placeholder {\n    color: ${({ theme }) => theme.colors.neutral.textWeak};\n  }\n\n  &:disabled {\n    cursor: not-allowed;\n\n    &::placeholder {\n      color: ${({ theme }) => theme.colors.neutral.textWeakDisabled};\n    }\n  }\n`\n\nconst UnitInputWrapper = styled(Stack)<{\n  'data-size': 'small' | 'medium' | 'large'\n  'data-success': boolean\n  'data-error': boolean\n  'data-disabled': boolean\n  'data-readonly': boolean\n}>`\n  border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  border-radius: ${({ theme }) => theme.radii.default};\n\n  &:not([data-disabled='true']):not([data-readonly='true']):not(\n      [data-success='true']\n    ):not([data-error='true']):focus,\n  :not([data-disabled='true']):not([data-readonly='true']):not(\n      [data-success='true']\n    ):not([data-error='true'])active {\n    box-shadow: ${({ theme }) => theme.shadows.focusPrimary};\n    border-color: ${({ theme }) => theme.colors.primary.borderHover};\n\n    & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.primary.border};\n    }\n  }\n\n  &:not([data-disabled='true']):not([data-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: ${INPUT_SIZE_HEIGHT['small']}px;\n  }\n  &[data-size='medium'] {\n    height: ${INPUT_SIZE_HEIGHT['medium']}px;\n  }\n  &[data-size='large'] {\n    height: ${INPUT_SIZE_HEIGHT['large']}px;\n  }\n\n  &[data-success='true'] {\n    border: 1px solid ${({ theme }) => theme.colors.success.border};\n    & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.success.border};\n    }\n  }\n  &[data-success='true']:active {\n    border: 1px solid ${({ theme }) => theme.colors.success.border};\n    box-shadow: ${({ theme }) => theme.shadows.focusSuccess};\n\n    & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.success.border};\n    }\n  }\n\n  &[data-error='true'] {\n    border: 1px solid ${({ theme }) => theme.colors.danger.border};\n\n    & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.danger.border};\n    }\n\n    & > ${StyledNumberInputWrapper}:hover {\n      border-right-color: ${({ theme }) => theme.colors.danger.border};\n    }\n  }\n\n  &[data-error='true']:active {\n    border: 1px solid ${({ theme }) => theme.colors.danger.border};\n    box-shadow: ${({ theme }) => theme.shadows.focusDanger};\n\n    & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.danger.border};\n    }\n    & > ${StyledNumberInputWrapper}:hover {\n      border-right-color: ${({ theme }) => theme.colors.danger.border};\n    }\n  }\n\n  &[data-disabled='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    border-color: ${({ theme }) => theme.colors.neutral.borderDisabled};\n    cursor: not-allowed;\n\n    & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.neutral.borderDisabled};\n    }\n  }\n`\n\nconst CustomSelectInput = styled(SelectInputV2)<{\n  width?: number\n  'data-disabled': boolean\n}>`\n  #unit {\n    border: none;\n    background: transparent;\n  }\n\n  ${({ width }) => width && `width: ${width}px;`}\n\n  #unit:focus,\n  #unit:active {\n    box-shadow: none;\n  }\n`\ntype UnitInputValue = { inputValue: number; unit: string }\n\ntype UnitInputProps = {\n  className?: string\n  max?: number\n  min?: number\n  value?: UnitInputValue['inputValue']\n  unitValue?: UnitInputValue['unit']\n  onChange?: (value: UnitInputValue['inputValue']) => void\n  onChangeUnitValue?: (values: string) => void\n  options: OptionType[]\n  selectInputWidth?: number\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 = 200,\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 ? (\n        <Stack direction=\"row\" gap={0.5} alignItems=\"center\">\n          <Text\n            as=\"label\"\n            variant={size === 'large' ? 'bodyStrong' : 'bodySmallStrong'}\n            disabled={disabled}\n            htmlFor={id ?? localId}\n          >\n            {label}\n          </Text>\n          {required ? (\n            <Icon name=\"asterisk\" sentiment=\"danger\" size={8} />\n          ) : null}\n          {labelInformation ?? null}\n        </Stack>\n      ) : null}\n      <UnitInputWrapper\n        direction=\"row\"\n        data-testid={dataTestId}\n        data-size={size}\n        width={width}\n        id={id}\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            id={id ? `${id}-value` : undefined}\n            value={val}\n            onChange={event => {\n              const numericValue = Number.parseInt(event.target.value, 10)\n              if (numericValue > max) {\n                setVal(max)\n                onChange?.(max)\n              } else if (numericValue < min) {\n                setVal(min)\n                onChange?.(min)\n              } else {\n                setVal(numericValue)\n                onChange?.(numericValue)\n              }\n            }}\n            placeholder={placeholder}\n            max={max}\n            readOnly={readOnly}\n            min={min}\n            step={step}\n            data-success={success}\n            data-error={error}\n            data-testid=\"unit-input\"\n            required={required}\n            className={className}\n          />\n          {error ? <Icon name=\"alert\" sentiment=\"danger\" /> : null}\n          {success && !error ? (\n            <Icon name=\"checkbox-circle-outline\" sentiment=\"success\" />\n          ) : null}\n        </StyledNumberInputWrapper>\n        <CustomSelectInput\n          width={selectInputWidth}\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      </UnitInputWrapper>\n      {error || typeof success === 'string' || typeof helper === 'string' ? (\n        <Text\n          as=\"p\"\n          variant=\"caption\"\n          sentiment={sentiment}\n          disabled={disabled}\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"]} */"));
23
23
  const StyledInput = /* @__PURE__ */ _styled("input", process.env.NODE_ENV === "production" ? {
24
24
  target: "ex81jph2"
25
25
  } : {
@@ -33,7 +33,7 @@ const StyledInput = /* @__PURE__ */ _styled("input", process.env.NODE_ENV === "p
33
33
  theme
34
34
  }) => theme.colors.neutral.textWeak, ";}&:disabled{cursor:not-allowed;&::placeholder{color:", ({
35
35
  theme
36
- }) => 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":"AAyBgC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/UnitInput/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { Icon } from '@ultraviolet/icons'\nimport type { ComponentProps, InputHTMLAttributes, ReactNode } from 'react'\nimport { useId, useMemo, useState } from 'react'\nimport { SelectInputV2 } from '../SelectInputV2'\nimport type { OptionType } from '../SelectInputV2/types'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\n\nconst INPUT_SIZE_HEIGHT: Record<string, number> = {\n  large: 48,\n  medium: 40,\n  small: 32,\n}\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  padding-left: ${({ theme }) => theme.space['2']};\n  background: transparent;\n  color: ${({ theme }) => theme.colors.neutral.text};\n  &::placeholder {\n    color: ${({ theme }) => theme.colors.neutral.textWeak};\n  }\n\n  &:disabled {\n    cursor: not-allowed;\n\n    &::placeholder {\n      color: ${({ theme }) => theme.colors.neutral.textWeakDisabled};\n    }\n  }\n`\n\nconst UnitInputWrapper = styled(Stack)<{\n  'data-size': 'small' | 'medium' | 'large'\n  'data-success': boolean\n  'data-error': boolean\n  'data-disabled': boolean\n  'data-readonly': boolean\n}>`\n  border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  border-radius: ${({ theme }) => theme.radii.default};\n\n  &:not([data-disabled='true']):not([data-readonly='true']):not(\n      [data-success='true']\n    ):not([data-error='true']):focus,\n  :not([data-disabled='true']):not([data-readonly='true']):not(\n      [data-success='true']\n    ):not([data-error='true'])active {\n    box-shadow: ${({ theme }) => theme.shadows.focusPrimary};\n    border-color: ${({ theme }) => theme.colors.primary.borderHover};\n\n    & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.primary.border};\n    }\n  }\n\n  &:not([data-disabled='true']):not([data-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: ${INPUT_SIZE_HEIGHT['small']}px;\n  }\n  &[data-size='medium'] {\n    height: ${INPUT_SIZE_HEIGHT['medium']}px;\n  }\n  &[data-size='large'] {\n    height: ${INPUT_SIZE_HEIGHT['large']}px;\n  }\n\n  &[data-success='true'] {\n    border: 1px solid ${({ theme }) => theme.colors.success.border};\n    & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.success.border};\n    }\n  }\n  &[data-success='true']:active {\n    border: 1px solid ${({ theme }) => theme.colors.success.border};\n    box-shadow: ${({ theme }) => theme.shadows.focusSuccess};\n\n    & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.success.border};\n    }\n  }\n\n  &[data-error='true'] {\n    border: 1px solid ${({ theme }) => theme.colors.danger.border};\n\n    & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.danger.border};\n    }\n\n    & > ${StyledNumberInputWrapper}:hover {\n      border-right-color: ${({ theme }) => theme.colors.danger.border};\n    }\n  }\n\n  &[data-error='true']:active {\n    border: 1px solid ${({ theme }) => theme.colors.danger.border};\n    box-shadow: ${({ theme }) => theme.shadows.focusDanger};\n\n    & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.danger.border};\n    }\n    & > ${StyledNumberInputWrapper}:hover {\n      border-right-color: ${({ theme }) => theme.colors.danger.border};\n    }\n  }\n\n  &[data-disabled='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    border-color: ${({ theme }) => theme.colors.neutral.borderDisabled};\n    cursor: not-allowed;\n\n    & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.neutral.borderDisabled};\n    }\n  }\n`\n\nconst CustomSelectInput = styled(SelectInputV2)<{\n  width?: number\n  'data-disabled': boolean\n}>`\n  #unit {\n    border: none;\n    background: transparent;\n  }\n\n  ${({ width }) => width && `width: ${width}px;`}\n\n  #unit:focus,\n  #unit:active {\n    box-shadow: none;\n  }\n`\ntype UnitInputValue = { inputValue: number; unit: string }\n\ntype UnitInputProps = {\n  className?: string\n  max?: number\n  min?: number\n  value?: UnitInputValue['inputValue']\n  unitValue?: UnitInputValue['unit']\n  onChange?: (value: UnitInputValue['inputValue']) => void\n  onChangeUnitValue?: (values: string) => void\n  options: OptionType[]\n  selectInputWidth?: number\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 = 200,\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 ? (\n        <Stack direction=\"row\" gap={0.5} alignItems=\"center\">\n          <Text\n            as=\"label\"\n            variant={size === 'large' ? 'bodyStrong' : 'bodySmallStrong'}\n            disabled={disabled}\n            htmlFor={id ?? localId}\n          >\n            {label}\n          </Text>\n          {required ? (\n            <Icon name=\"asterisk\" sentiment=\"danger\" size={8} />\n          ) : null}\n          {labelInformation ?? null}\n        </Stack>\n      ) : null}\n      <UnitInputWrapper\n        direction=\"row\"\n        data-testid={dataTestId}\n        data-size={size}\n        width={width}\n        id={id}\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            data-readOnly={readOnly}\n            name={`${name}-value`}\n            id={id ? `${id}-value` : undefined}\n            value={val}\n            onChange={event => {\n              const numericValue = Number.parseInt(event.target.value, 10)\n              if (numericValue > max) {\n                setVal(max)\n                onChange?.(max)\n              } else if (numericValue < min) {\n                setVal(min)\n                onChange?.(min)\n              } else {\n                setVal(numericValue)\n                onChange?.(numericValue)\n              }\n            }}\n            placeholder={placeholder}\n            max={max}\n            readOnly={readOnly}\n            min={min}\n            step={step}\n            data-success={success}\n            data-error={error}\n            data-testid=\"unit-input\"\n            required={required}\n            className={className}\n          />\n          {error ? <Icon name=\"alert\" sentiment=\"danger\" /> : null}\n          {success && !error ? (\n            <Icon name=\"checkbox-circle-outline\" sentiment=\"success\" />\n          ) : null}\n        </StyledNumberInputWrapper>\n        <CustomSelectInput\n          width={selectInputWidth}\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      </UnitInputWrapper>\n      {error || typeof success === 'string' || typeof helper === 'string' ? (\n        <Text\n          as=\"p\"\n          variant=\"caption\"\n          sentiment={sentiment}\n          disabled={disabled}\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"]} */"));
36
+ }) => 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":"AAyBgC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/UnitInput/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { Icon } from '@ultraviolet/icons'\nimport type { ComponentProps, InputHTMLAttributes, ReactNode } from 'react'\nimport { useId, useMemo, useState } from 'react'\nimport { SelectInputV2 } from '../SelectInputV2'\nimport type { OptionType } from '../SelectInputV2/types'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\n\nconst INPUT_SIZE_HEIGHT: Record<string, number> = {\n  large: 48,\n  medium: 40,\n  small: 32,\n}\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  padding-left: ${({ theme }) => theme.space['2']};\n  background: transparent;\n  color: ${({ theme }) => theme.colors.neutral.text};\n  &::placeholder {\n    color: ${({ theme }) => theme.colors.neutral.textWeak};\n  }\n\n  &:disabled {\n    cursor: not-allowed;\n\n    &::placeholder {\n      color: ${({ theme }) => theme.colors.neutral.textWeakDisabled};\n    }\n  }\n`\n\nconst UnitInputWrapper = styled(Stack)<{\n  'data-size': 'small' | 'medium' | 'large'\n  'data-success': boolean\n  'data-error': boolean\n  'data-disabled': boolean\n  'data-readonly': boolean\n}>`\n  border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  border-radius: ${({ theme }) => theme.radii.default};\n\n  &:not([data-disabled='true']):not([data-readonly='true']):not(\n      [data-success='true']\n    ):not([data-error='true']):focus,\n  :not([data-disabled='true']):not([data-readonly='true']):not(\n      [data-success='true']\n    ):not([data-error='true'])active {\n    box-shadow: ${({ theme }) => theme.shadows.focusPrimary};\n    border-color: ${({ theme }) => theme.colors.primary.borderHover};\n\n    & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.primary.border};\n    }\n  }\n\n  &:not([data-disabled='true']):not([data-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: ${INPUT_SIZE_HEIGHT['small']}px;\n  }\n  &[data-size='medium'] {\n    height: ${INPUT_SIZE_HEIGHT['medium']}px;\n  }\n  &[data-size='large'] {\n    height: ${INPUT_SIZE_HEIGHT['large']}px;\n  }\n\n  &[data-success='true'] {\n    border: 1px solid ${({ theme }) => theme.colors.success.border};\n    & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.success.border};\n    }\n  }\n  &[data-success='true']:active {\n    border: 1px solid ${({ theme }) => theme.colors.success.border};\n    box-shadow: ${({ theme }) => theme.shadows.focusSuccess};\n\n    & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.success.border};\n    }\n  }\n\n  &[data-error='true'] {\n    border: 1px solid ${({ theme }) => theme.colors.danger.border};\n\n    & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.danger.border};\n    }\n\n    & > ${StyledNumberInputWrapper}:hover {\n      border-right-color: ${({ theme }) => theme.colors.danger.border};\n    }\n  }\n\n  &[data-error='true']:active {\n    border: 1px solid ${({ theme }) => theme.colors.danger.border};\n    box-shadow: ${({ theme }) => theme.shadows.focusDanger};\n\n    & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.danger.border};\n    }\n    & > ${StyledNumberInputWrapper}:hover {\n      border-right-color: ${({ theme }) => theme.colors.danger.border};\n    }\n  }\n\n  &[data-disabled='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    border-color: ${({ theme }) => theme.colors.neutral.borderDisabled};\n    cursor: not-allowed;\n\n    & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.neutral.borderDisabled};\n    }\n  }\n`\n\nconst CustomSelectInput = styled(SelectInputV2)<{\n  width?: number\n  'data-disabled': boolean\n}>`\n  #unit {\n    border: none;\n    background: transparent;\n  }\n\n  ${({ width }) => width && `width: ${width}px;`}\n\n  #unit:focus,\n  #unit:active {\n    box-shadow: none;\n  }\n`\ntype UnitInputValue = { inputValue: number; unit: string }\n\ntype UnitInputProps = {\n  className?: string\n  max?: number\n  min?: number\n  value?: UnitInputValue['inputValue']\n  unitValue?: UnitInputValue['unit']\n  onChange?: (value: UnitInputValue['inputValue']) => void\n  onChangeUnitValue?: (values: string) => void\n  options: OptionType[]\n  selectInputWidth?: number\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 = 200,\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 ? (\n        <Stack direction=\"row\" gap={0.5} alignItems=\"center\">\n          <Text\n            as=\"label\"\n            variant={size === 'large' ? 'bodyStrong' : 'bodySmallStrong'}\n            disabled={disabled}\n            htmlFor={id ?? localId}\n          >\n            {label}\n          </Text>\n          {required ? (\n            <Icon name=\"asterisk\" sentiment=\"danger\" size={8} />\n          ) : null}\n          {labelInformation ?? null}\n        </Stack>\n      ) : null}\n      <UnitInputWrapper\n        direction=\"row\"\n        data-testid={dataTestId}\n        data-size={size}\n        width={width}\n        id={id}\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            id={id ? `${id}-value` : undefined}\n            value={val}\n            onChange={event => {\n              const numericValue = Number.parseInt(event.target.value, 10)\n              if (numericValue > max) {\n                setVal(max)\n                onChange?.(max)\n              } else if (numericValue < min) {\n                setVal(min)\n                onChange?.(min)\n              } else {\n                setVal(numericValue)\n                onChange?.(numericValue)\n              }\n            }}\n            placeholder={placeholder}\n            max={max}\n            readOnly={readOnly}\n            min={min}\n            step={step}\n            data-success={success}\n            data-error={error}\n            data-testid=\"unit-input\"\n            required={required}\n            className={className}\n          />\n          {error ? <Icon name=\"alert\" sentiment=\"danger\" /> : null}\n          {success && !error ? (\n            <Icon name=\"checkbox-circle-outline\" sentiment=\"success\" />\n          ) : null}\n        </StyledNumberInputWrapper>\n        <CustomSelectInput\n          width={selectInputWidth}\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      </UnitInputWrapper>\n      {error || typeof success === 'string' || typeof helper === 'string' ? (\n        <Text\n          as=\"p\"\n          variant=\"caption\"\n          sentiment={sentiment}\n          disabled={disabled}\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"]} */"));
37
37
  const UnitInputWrapper = /* @__PURE__ */ _styled(Stack, process.env.NODE_ENV === "production" ? {
38
38
  target: "ex81jph1"
39
39
  } : {
@@ -89,7 +89,7 @@ const UnitInputWrapper = /* @__PURE__ */ _styled(Stack, process.env.NODE_ENV ===
89
89
  theme
90
90
  }) => theme.colors.neutral.borderDisabled, ";cursor:not-allowed;&>", StyledNumberInputWrapper, "{border-right-color:", ({
91
91
  theme
92
- }) => 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":"AAoDE","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/UnitInput/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { Icon } from '@ultraviolet/icons'\nimport type { ComponentProps, InputHTMLAttributes, ReactNode } from 'react'\nimport { useId, useMemo, useState } from 'react'\nimport { SelectInputV2 } from '../SelectInputV2'\nimport type { OptionType } from '../SelectInputV2/types'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\n\nconst INPUT_SIZE_HEIGHT: Record<string, number> = {\n  large: 48,\n  medium: 40,\n  small: 32,\n}\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  padding-left: ${({ theme }) => theme.space['2']};\n  background: transparent;\n  color: ${({ theme }) => theme.colors.neutral.text};\n  &::placeholder {\n    color: ${({ theme }) => theme.colors.neutral.textWeak};\n  }\n\n  &:disabled {\n    cursor: not-allowed;\n\n    &::placeholder {\n      color: ${({ theme }) => theme.colors.neutral.textWeakDisabled};\n    }\n  }\n`\n\nconst UnitInputWrapper = styled(Stack)<{\n  'data-size': 'small' | 'medium' | 'large'\n  'data-success': boolean\n  'data-error': boolean\n  'data-disabled': boolean\n  'data-readonly': boolean\n}>`\n  border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  border-radius: ${({ theme }) => theme.radii.default};\n\n  &:not([data-disabled='true']):not([data-readonly='true']):not(\n      [data-success='true']\n    ):not([data-error='true']):focus,\n  :not([data-disabled='true']):not([data-readonly='true']):not(\n      [data-success='true']\n    ):not([data-error='true'])active {\n    box-shadow: ${({ theme }) => theme.shadows.focusPrimary};\n    border-color: ${({ theme }) => theme.colors.primary.borderHover};\n\n    & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.primary.border};\n    }\n  }\n\n  &:not([data-disabled='true']):not([data-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: ${INPUT_SIZE_HEIGHT['small']}px;\n  }\n  &[data-size='medium'] {\n    height: ${INPUT_SIZE_HEIGHT['medium']}px;\n  }\n  &[data-size='large'] {\n    height: ${INPUT_SIZE_HEIGHT['large']}px;\n  }\n\n  &[data-success='true'] {\n    border: 1px solid ${({ theme }) => theme.colors.success.border};\n    & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.success.border};\n    }\n  }\n  &[data-success='true']:active {\n    border: 1px solid ${({ theme }) => theme.colors.success.border};\n    box-shadow: ${({ theme }) => theme.shadows.focusSuccess};\n\n    & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.success.border};\n    }\n  }\n\n  &[data-error='true'] {\n    border: 1px solid ${({ theme }) => theme.colors.danger.border};\n\n    & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.danger.border};\n    }\n\n    & > ${StyledNumberInputWrapper}:hover {\n      border-right-color: ${({ theme }) => theme.colors.danger.border};\n    }\n  }\n\n  &[data-error='true']:active {\n    border: 1px solid ${({ theme }) => theme.colors.danger.border};\n    box-shadow: ${({ theme }) => theme.shadows.focusDanger};\n\n    & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.danger.border};\n    }\n    & > ${StyledNumberInputWrapper}:hover {\n      border-right-color: ${({ theme }) => theme.colors.danger.border};\n    }\n  }\n\n  &[data-disabled='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    border-color: ${({ theme }) => theme.colors.neutral.borderDisabled};\n    cursor: not-allowed;\n\n    & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.neutral.borderDisabled};\n    }\n  }\n`\n\nconst CustomSelectInput = styled(SelectInputV2)<{\n  width?: number\n  'data-disabled': boolean\n}>`\n  #unit {\n    border: none;\n    background: transparent;\n  }\n\n  ${({ width }) => width && `width: ${width}px;`}\n\n  #unit:focus,\n  #unit:active {\n    box-shadow: none;\n  }\n`\ntype UnitInputValue = { inputValue: number; unit: string }\n\ntype UnitInputProps = {\n  className?: string\n  max?: number\n  min?: number\n  value?: UnitInputValue['inputValue']\n  unitValue?: UnitInputValue['unit']\n  onChange?: (value: UnitInputValue['inputValue']) => void\n  onChangeUnitValue?: (values: string) => void\n  options: OptionType[]\n  selectInputWidth?: number\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 = 200,\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 ? (\n        <Stack direction=\"row\" gap={0.5} alignItems=\"center\">\n          <Text\n            as=\"label\"\n            variant={size === 'large' ? 'bodyStrong' : 'bodySmallStrong'}\n            disabled={disabled}\n            htmlFor={id ?? localId}\n          >\n            {label}\n          </Text>\n          {required ? (\n            <Icon name=\"asterisk\" sentiment=\"danger\" size={8} />\n          ) : null}\n          {labelInformation ?? null}\n        </Stack>\n      ) : null}\n      <UnitInputWrapper\n        direction=\"row\"\n        data-testid={dataTestId}\n        data-size={size}\n        width={width}\n        id={id}\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            data-readOnly={readOnly}\n            name={`${name}-value`}\n            id={id ? `${id}-value` : undefined}\n            value={val}\n            onChange={event => {\n              const numericValue = Number.parseInt(event.target.value, 10)\n              if (numericValue > max) {\n                setVal(max)\n                onChange?.(max)\n              } else if (numericValue < min) {\n                setVal(min)\n                onChange?.(min)\n              } else {\n                setVal(numericValue)\n                onChange?.(numericValue)\n              }\n            }}\n            placeholder={placeholder}\n            max={max}\n            readOnly={readOnly}\n            min={min}\n            step={step}\n            data-success={success}\n            data-error={error}\n            data-testid=\"unit-input\"\n            required={required}\n            className={className}\n          />\n          {error ? <Icon name=\"alert\" sentiment=\"danger\" /> : null}\n          {success && !error ? (\n            <Icon name=\"checkbox-circle-outline\" sentiment=\"success\" />\n          ) : null}\n        </StyledNumberInputWrapper>\n        <CustomSelectInput\n          width={selectInputWidth}\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      </UnitInputWrapper>\n      {error || typeof success === 'string' || typeof helper === 'string' ? (\n        <Text\n          as=\"p\"\n          variant=\"caption\"\n          sentiment={sentiment}\n          disabled={disabled}\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"]} */"));
92
+ }) => 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":"AAoDE","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/UnitInput/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { Icon } from '@ultraviolet/icons'\nimport type { ComponentProps, InputHTMLAttributes, ReactNode } from 'react'\nimport { useId, useMemo, useState } from 'react'\nimport { SelectInputV2 } from '../SelectInputV2'\nimport type { OptionType } from '../SelectInputV2/types'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\n\nconst INPUT_SIZE_HEIGHT: Record<string, number> = {\n  large: 48,\n  medium: 40,\n  small: 32,\n}\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  padding-left: ${({ theme }) => theme.space['2']};\n  background: transparent;\n  color: ${({ theme }) => theme.colors.neutral.text};\n  &::placeholder {\n    color: ${({ theme }) => theme.colors.neutral.textWeak};\n  }\n\n  &:disabled {\n    cursor: not-allowed;\n\n    &::placeholder {\n      color: ${({ theme }) => theme.colors.neutral.textWeakDisabled};\n    }\n  }\n`\n\nconst UnitInputWrapper = styled(Stack)<{\n  'data-size': 'small' | 'medium' | 'large'\n  'data-success': boolean\n  'data-error': boolean\n  'data-disabled': boolean\n  'data-readonly': boolean\n}>`\n  border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  border-radius: ${({ theme }) => theme.radii.default};\n\n  &:not([data-disabled='true']):not([data-readonly='true']):not(\n      [data-success='true']\n    ):not([data-error='true']):focus,\n  :not([data-disabled='true']):not([data-readonly='true']):not(\n      [data-success='true']\n    ):not([data-error='true'])active {\n    box-shadow: ${({ theme }) => theme.shadows.focusPrimary};\n    border-color: ${({ theme }) => theme.colors.primary.borderHover};\n\n    & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.primary.border};\n    }\n  }\n\n  &:not([data-disabled='true']):not([data-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: ${INPUT_SIZE_HEIGHT['small']}px;\n  }\n  &[data-size='medium'] {\n    height: ${INPUT_SIZE_HEIGHT['medium']}px;\n  }\n  &[data-size='large'] {\n    height: ${INPUT_SIZE_HEIGHT['large']}px;\n  }\n\n  &[data-success='true'] {\n    border: 1px solid ${({ theme }) => theme.colors.success.border};\n    & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.success.border};\n    }\n  }\n  &[data-success='true']:active {\n    border: 1px solid ${({ theme }) => theme.colors.success.border};\n    box-shadow: ${({ theme }) => theme.shadows.focusSuccess};\n\n    & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.success.border};\n    }\n  }\n\n  &[data-error='true'] {\n    border: 1px solid ${({ theme }) => theme.colors.danger.border};\n\n    & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.danger.border};\n    }\n\n    & > ${StyledNumberInputWrapper}:hover {\n      border-right-color: ${({ theme }) => theme.colors.danger.border};\n    }\n  }\n\n  &[data-error='true']:active {\n    border: 1px solid ${({ theme }) => theme.colors.danger.border};\n    box-shadow: ${({ theme }) => theme.shadows.focusDanger};\n\n    & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.danger.border};\n    }\n    & > ${StyledNumberInputWrapper}:hover {\n      border-right-color: ${({ theme }) => theme.colors.danger.border};\n    }\n  }\n\n  &[data-disabled='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    border-color: ${({ theme }) => theme.colors.neutral.borderDisabled};\n    cursor: not-allowed;\n\n    & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.neutral.borderDisabled};\n    }\n  }\n`\n\nconst CustomSelectInput = styled(SelectInputV2)<{\n  width?: number\n  'data-disabled': boolean\n}>`\n  #unit {\n    border: none;\n    background: transparent;\n  }\n\n  ${({ width }) => width && `width: ${width}px;`}\n\n  #unit:focus,\n  #unit:active {\n    box-shadow: none;\n  }\n`\ntype UnitInputValue = { inputValue: number; unit: string }\n\ntype UnitInputProps = {\n  className?: string\n  max?: number\n  min?: number\n  value?: UnitInputValue['inputValue']\n  unitValue?: UnitInputValue['unit']\n  onChange?: (value: UnitInputValue['inputValue']) => void\n  onChangeUnitValue?: (values: string) => void\n  options: OptionType[]\n  selectInputWidth?: number\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 = 200,\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 ? (\n        <Stack direction=\"row\" gap={0.5} alignItems=\"center\">\n          <Text\n            as=\"label\"\n            variant={size === 'large' ? 'bodyStrong' : 'bodySmallStrong'}\n            disabled={disabled}\n            htmlFor={id ?? localId}\n          >\n            {label}\n          </Text>\n          {required ? (\n            <Icon name=\"asterisk\" sentiment=\"danger\" size={8} />\n          ) : null}\n          {labelInformation ?? null}\n        </Stack>\n      ) : null}\n      <UnitInputWrapper\n        direction=\"row\"\n        data-testid={dataTestId}\n        data-size={size}\n        width={width}\n        id={id}\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            id={id ? `${id}-value` : undefined}\n            value={val}\n            onChange={event => {\n              const numericValue = Number.parseInt(event.target.value, 10)\n              if (numericValue > max) {\n                setVal(max)\n                onChange?.(max)\n              } else if (numericValue < min) {\n                setVal(min)\n                onChange?.(min)\n              } else {\n                setVal(numericValue)\n                onChange?.(numericValue)\n              }\n            }}\n            placeholder={placeholder}\n            max={max}\n            readOnly={readOnly}\n            min={min}\n            step={step}\n            data-success={success}\n            data-error={error}\n            data-testid=\"unit-input\"\n            required={required}\n            className={className}\n          />\n          {error ? <Icon name=\"alert\" sentiment=\"danger\" /> : null}\n          {success && !error ? (\n            <Icon name=\"checkbox-circle-outline\" sentiment=\"success\" />\n          ) : null}\n        </StyledNumberInputWrapper>\n        <CustomSelectInput\n          width={selectInputWidth}\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      </UnitInputWrapper>\n      {error || typeof success === 'string' || typeof helper === 'string' ? (\n        <Text\n          as=\"p\"\n          variant=\"caption\"\n          sentiment={sentiment}\n          disabled={disabled}\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"]} */"));
93
93
  const CustomSelectInput = /* @__PURE__ */ _styled(SelectInputV2, process.env.NODE_ENV === "production" ? {
94
94
  target: "ex81jph0"
95
95
  } : {
@@ -97,7 +97,7 @@ const CustomSelectInput = /* @__PURE__ */ _styled(SelectInputV2, process.env.NOD
97
97
  label: "CustomSelectInput"
98
98
  })("#unit{border:none;background:transparent;}", ({
99
99
  width
100
- }) => width && `width: ${width}px;`, " #unit:focus,#unit:active{box-shadow:none;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/UnitInput/index.tsx"],"names":[],"mappings":"AA4JE","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/UnitInput/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { Icon } from '@ultraviolet/icons'\nimport type { ComponentProps, InputHTMLAttributes, ReactNode } from 'react'\nimport { useId, useMemo, useState } from 'react'\nimport { SelectInputV2 } from '../SelectInputV2'\nimport type { OptionType } from '../SelectInputV2/types'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\n\nconst INPUT_SIZE_HEIGHT: Record<string, number> = {\n  large: 48,\n  medium: 40,\n  small: 32,\n}\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  padding-left: ${({ theme }) => theme.space['2']};\n  background: transparent;\n  color: ${({ theme }) => theme.colors.neutral.text};\n  &::placeholder {\n    color: ${({ theme }) => theme.colors.neutral.textWeak};\n  }\n\n  &:disabled {\n    cursor: not-allowed;\n\n    &::placeholder {\n      color: ${({ theme }) => theme.colors.neutral.textWeakDisabled};\n    }\n  }\n`\n\nconst UnitInputWrapper = styled(Stack)<{\n  'data-size': 'small' | 'medium' | 'large'\n  'data-success': boolean\n  'data-error': boolean\n  'data-disabled': boolean\n  'data-readonly': boolean\n}>`\n  border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  border-radius: ${({ theme }) => theme.radii.default};\n\n  &:not([data-disabled='true']):not([data-readonly='true']):not(\n      [data-success='true']\n    ):not([data-error='true']):focus,\n  :not([data-disabled='true']):not([data-readonly='true']):not(\n      [data-success='true']\n    ):not([data-error='true'])active {\n    box-shadow: ${({ theme }) => theme.shadows.focusPrimary};\n    border-color: ${({ theme }) => theme.colors.primary.borderHover};\n\n    & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.primary.border};\n    }\n  }\n\n  &:not([data-disabled='true']):not([data-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: ${INPUT_SIZE_HEIGHT['small']}px;\n  }\n  &[data-size='medium'] {\n    height: ${INPUT_SIZE_HEIGHT['medium']}px;\n  }\n  &[data-size='large'] {\n    height: ${INPUT_SIZE_HEIGHT['large']}px;\n  }\n\n  &[data-success='true'] {\n    border: 1px solid ${({ theme }) => theme.colors.success.border};\n    & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.success.border};\n    }\n  }\n  &[data-success='true']:active {\n    border: 1px solid ${({ theme }) => theme.colors.success.border};\n    box-shadow: ${({ theme }) => theme.shadows.focusSuccess};\n\n    & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.success.border};\n    }\n  }\n\n  &[data-error='true'] {\n    border: 1px solid ${({ theme }) => theme.colors.danger.border};\n\n    & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.danger.border};\n    }\n\n    & > ${StyledNumberInputWrapper}:hover {\n      border-right-color: ${({ theme }) => theme.colors.danger.border};\n    }\n  }\n\n  &[data-error='true']:active {\n    border: 1px solid ${({ theme }) => theme.colors.danger.border};\n    box-shadow: ${({ theme }) => theme.shadows.focusDanger};\n\n    & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.danger.border};\n    }\n    & > ${StyledNumberInputWrapper}:hover {\n      border-right-color: ${({ theme }) => theme.colors.danger.border};\n    }\n  }\n\n  &[data-disabled='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    border-color: ${({ theme }) => theme.colors.neutral.borderDisabled};\n    cursor: not-allowed;\n\n    & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.neutral.borderDisabled};\n    }\n  }\n`\n\nconst CustomSelectInput = styled(SelectInputV2)<{\n  width?: number\n  'data-disabled': boolean\n}>`\n  #unit {\n    border: none;\n    background: transparent;\n  }\n\n  ${({ width }) => width && `width: ${width}px;`}\n\n  #unit:focus,\n  #unit:active {\n    box-shadow: none;\n  }\n`\ntype UnitInputValue = { inputValue: number; unit: string }\n\ntype UnitInputProps = {\n  className?: string\n  max?: number\n  min?: number\n  value?: UnitInputValue['inputValue']\n  unitValue?: UnitInputValue['unit']\n  onChange?: (value: UnitInputValue['inputValue']) => void\n  onChangeUnitValue?: (values: string) => void\n  options: OptionType[]\n  selectInputWidth?: number\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 = 200,\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 ? (\n        <Stack direction=\"row\" gap={0.5} alignItems=\"center\">\n          <Text\n            as=\"label\"\n            variant={size === 'large' ? 'bodyStrong' : 'bodySmallStrong'}\n            disabled={disabled}\n            htmlFor={id ?? localId}\n          >\n            {label}\n          </Text>\n          {required ? (\n            <Icon name=\"asterisk\" sentiment=\"danger\" size={8} />\n          ) : null}\n          {labelInformation ?? null}\n        </Stack>\n      ) : null}\n      <UnitInputWrapper\n        direction=\"row\"\n        data-testid={dataTestId}\n        data-size={size}\n        width={width}\n        id={id}\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            data-readOnly={readOnly}\n            name={`${name}-value`}\n            id={id ? `${id}-value` : undefined}\n            value={val}\n            onChange={event => {\n              const numericValue = Number.parseInt(event.target.value, 10)\n              if (numericValue > max) {\n                setVal(max)\n                onChange?.(max)\n              } else if (numericValue < min) {\n                setVal(min)\n                onChange?.(min)\n              } else {\n                setVal(numericValue)\n                onChange?.(numericValue)\n              }\n            }}\n            placeholder={placeholder}\n            max={max}\n            readOnly={readOnly}\n            min={min}\n            step={step}\n            data-success={success}\n            data-error={error}\n            data-testid=\"unit-input\"\n            required={required}\n            className={className}\n          />\n          {error ? <Icon name=\"alert\" sentiment=\"danger\" /> : null}\n          {success && !error ? (\n            <Icon name=\"checkbox-circle-outline\" sentiment=\"success\" />\n          ) : null}\n        </StyledNumberInputWrapper>\n        <CustomSelectInput\n          width={selectInputWidth}\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      </UnitInputWrapper>\n      {error || typeof success === 'string' || typeof helper === 'string' ? (\n        <Text\n          as=\"p\"\n          variant=\"caption\"\n          sentiment={sentiment}\n          disabled={disabled}\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"]} */"));
100
+ }) => width && `width: ${width}px;`, " #unit:focus,#unit:active{box-shadow:none;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/UnitInput/index.tsx"],"names":[],"mappings":"AA4JE","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/UnitInput/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { Icon } from '@ultraviolet/icons'\nimport type { ComponentProps, InputHTMLAttributes, ReactNode } from 'react'\nimport { useId, useMemo, useState } from 'react'\nimport { SelectInputV2 } from '../SelectInputV2'\nimport type { OptionType } from '../SelectInputV2/types'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\n\nconst INPUT_SIZE_HEIGHT: Record<string, number> = {\n  large: 48,\n  medium: 40,\n  small: 32,\n}\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  padding-left: ${({ theme }) => theme.space['2']};\n  background: transparent;\n  color: ${({ theme }) => theme.colors.neutral.text};\n  &::placeholder {\n    color: ${({ theme }) => theme.colors.neutral.textWeak};\n  }\n\n  &:disabled {\n    cursor: not-allowed;\n\n    &::placeholder {\n      color: ${({ theme }) => theme.colors.neutral.textWeakDisabled};\n    }\n  }\n`\n\nconst UnitInputWrapper = styled(Stack)<{\n  'data-size': 'small' | 'medium' | 'large'\n  'data-success': boolean\n  'data-error': boolean\n  'data-disabled': boolean\n  'data-readonly': boolean\n}>`\n  border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  border-radius: ${({ theme }) => theme.radii.default};\n\n  &:not([data-disabled='true']):not([data-readonly='true']):not(\n      [data-success='true']\n    ):not([data-error='true']):focus,\n  :not([data-disabled='true']):not([data-readonly='true']):not(\n      [data-success='true']\n    ):not([data-error='true'])active {\n    box-shadow: ${({ theme }) => theme.shadows.focusPrimary};\n    border-color: ${({ theme }) => theme.colors.primary.borderHover};\n\n    & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.primary.border};\n    }\n  }\n\n  &:not([data-disabled='true']):not([data-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: ${INPUT_SIZE_HEIGHT['small']}px;\n  }\n  &[data-size='medium'] {\n    height: ${INPUT_SIZE_HEIGHT['medium']}px;\n  }\n  &[data-size='large'] {\n    height: ${INPUT_SIZE_HEIGHT['large']}px;\n  }\n\n  &[data-success='true'] {\n    border: 1px solid ${({ theme }) => theme.colors.success.border};\n    & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.success.border};\n    }\n  }\n  &[data-success='true']:active {\n    border: 1px solid ${({ theme }) => theme.colors.success.border};\n    box-shadow: ${({ theme }) => theme.shadows.focusSuccess};\n\n    & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.success.border};\n    }\n  }\n\n  &[data-error='true'] {\n    border: 1px solid ${({ theme }) => theme.colors.danger.border};\n\n    & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.danger.border};\n    }\n\n    & > ${StyledNumberInputWrapper}:hover {\n      border-right-color: ${({ theme }) => theme.colors.danger.border};\n    }\n  }\n\n  &[data-error='true']:active {\n    border: 1px solid ${({ theme }) => theme.colors.danger.border};\n    box-shadow: ${({ theme }) => theme.shadows.focusDanger};\n\n    & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.danger.border};\n    }\n    & > ${StyledNumberInputWrapper}:hover {\n      border-right-color: ${({ theme }) => theme.colors.danger.border};\n    }\n  }\n\n  &[data-disabled='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    border-color: ${({ theme }) => theme.colors.neutral.borderDisabled};\n    cursor: not-allowed;\n\n    & > ${StyledNumberInputWrapper} {\n      border-right-color: ${({ theme }) => theme.colors.neutral.borderDisabled};\n    }\n  }\n`\n\nconst CustomSelectInput = styled(SelectInputV2)<{\n  width?: number\n  'data-disabled': boolean\n}>`\n  #unit {\n    border: none;\n    background: transparent;\n  }\n\n  ${({ width }) => width && `width: ${width}px;`}\n\n  #unit:focus,\n  #unit:active {\n    box-shadow: none;\n  }\n`\ntype UnitInputValue = { inputValue: number; unit: string }\n\ntype UnitInputProps = {\n  className?: string\n  max?: number\n  min?: number\n  value?: UnitInputValue['inputValue']\n  unitValue?: UnitInputValue['unit']\n  onChange?: (value: UnitInputValue['inputValue']) => void\n  onChangeUnitValue?: (values: string) => void\n  options: OptionType[]\n  selectInputWidth?: number\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 = 200,\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 ? (\n        <Stack direction=\"row\" gap={0.5} alignItems=\"center\">\n          <Text\n            as=\"label\"\n            variant={size === 'large' ? 'bodyStrong' : 'bodySmallStrong'}\n            disabled={disabled}\n            htmlFor={id ?? localId}\n          >\n            {label}\n          </Text>\n          {required ? (\n            <Icon name=\"asterisk\" sentiment=\"danger\" size={8} />\n          ) : null}\n          {labelInformation ?? null}\n        </Stack>\n      ) : null}\n      <UnitInputWrapper\n        direction=\"row\"\n        data-testid={dataTestId}\n        data-size={size}\n        width={width}\n        id={id}\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            id={id ? `${id}-value` : undefined}\n            value={val}\n            onChange={event => {\n              const numericValue = Number.parseInt(event.target.value, 10)\n              if (numericValue > max) {\n                setVal(max)\n                onChange?.(max)\n              } else if (numericValue < min) {\n                setVal(min)\n                onChange?.(min)\n              } else {\n                setVal(numericValue)\n                onChange?.(numericValue)\n              }\n            }}\n            placeholder={placeholder}\n            max={max}\n            readOnly={readOnly}\n            min={min}\n            step={step}\n            data-success={success}\n            data-error={error}\n            data-testid=\"unit-input\"\n            required={required}\n            className={className}\n          />\n          {error ? <Icon name=\"alert\" sentiment=\"danger\" /> : null}\n          {success && !error ? (\n            <Icon name=\"checkbox-circle-outline\" sentiment=\"success\" />\n          ) : null}\n        </StyledNumberInputWrapper>\n        <CustomSelectInput\n          width={selectInputWidth}\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      </UnitInputWrapper>\n      {error || typeof success === 'string' || typeof helper === 'string' ? (\n        <Text\n          as=\"p\"\n          variant=\"caption\"\n          sentiment={sentiment}\n          disabled={disabled}\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"]} */"));
101
101
  const UnitInput = ({
102
102
  id,
103
103
  name = "",
@@ -149,7 +149,7 @@ const UnitInput = ({
149
149
  ] }) : null,
150
150
  /* @__PURE__ */ jsxs(UnitInputWrapper, { direction: "row", "data-testid": dataTestId, "data-size": size, width, id, "data-success": !!success, "data-error": !!error, "data-disabled": !!disabled, "data-readonly": !!readOnly, onFocus, onBlur, onKeyDown, children: [
151
151
  /* @__PURE__ */ jsxs(StyledNumberInputWrapper, { id: "input-field", children: [
152
- /* @__PURE__ */ jsx(StyledInput, { type: "number", "aria-invalid": !!error, autoFocus, disabled, "data-readOnly": readOnly, name: `${name}-value`, id: id ? `${id}-value` : void 0, value: val, onChange: (event) => {
152
+ /* @__PURE__ */ jsx(StyledInput, { type: "number", "aria-invalid": !!error, autoFocus, disabled, name: `${name}-value`, id: id ? `${id}-value` : void 0, value: val, onChange: (event) => {
153
153
  const numericValue = Number.parseInt(event.target.value, 10);
154
154
  if (numericValue > max) {
155
155
  setVal(max);
@@ -17,6 +17,7 @@ export { DateInput } from './DateInput';
17
17
  export { Dialog } from './Dialog';
18
18
  export { EmptyState } from './EmptyState';
19
19
  export { Expandable } from './Expandable';
20
+ export { ExpandableCard } from './ExpandableCard';
20
21
  export { GlobalAlert } from './GlobalAlert';
21
22
  export { LineChart } from './LineChart';
22
23
  export { Link } from './Link';
package/dist/index.cjs CHANGED
@@ -26,59 +26,60 @@ const index$g = require("./components/DateInput/index.cjs");
26
26
  const index$h = require("./components/Dialog/index.cjs");
27
27
  const index$i = require("./components/EmptyState/index.cjs");
28
28
  const index$j = require("./components/Expandable/index.cjs");
29
- const index$k = require("./components/GlobalAlert/index.cjs");
30
- const index$l = require("./components/LineChart/index.cjs");
31
- const index$m = require("./components/Link/index.cjs");
32
- const index$n = require("./components/List/index.cjs");
33
- const index$o = require("./components/Loader/index.cjs");
34
- const index$p = require("./components/Menu/index.cjs");
35
- const index$q = require("./components/MenuV2/index.cjs");
36
- const index$r = require("./components/Meter/index.cjs");
37
- const index$s = require("./components/Modal/index.cjs");
38
- const index$t = require("./components/Notice/index.cjs");
39
- const index$u = require("./components/Notification/index.cjs");
40
- const index$v = require("./components/NumberInput/index.cjs");
41
- const index$w = require("./components/NumberInputV2/index.cjs");
42
- const index$x = require("./components/Pagination/index.cjs");
43
- const index$y = require("./components/PasswordCheck/index.cjs");
44
- const index$z = require("./components/PasswordStrengthMeter/index.cjs");
45
- const index$A = require("./components/PieChart/index.cjs");
46
- const index$B = require("./components/Popover/index.cjs");
47
- const index$C = require("./components/Popup/index.cjs");
48
- const index$D = require("./components/ProgressBar/index.cjs");
49
- const index$E = require("./components/Radio/index.cjs");
50
- const index$F = require("./components/RadioGroup/index.cjs");
51
- const index$G = require("./components/Row/index.cjs");
52
- const index$H = require("./components/SearchInput/index.cjs");
53
- const index$I = require("./components/SelectableCard/index.cjs");
54
- const index$J = require("./components/SelectableCardGroup/index.cjs");
55
- const index$K = require("./components/SelectInput/index.cjs");
56
- const index$L = require("./components/SelectInputV2/index.cjs");
57
- const index$M = require("./components/Separator/index.cjs");
58
- const index$N = require("./components/Skeleton/index.cjs");
59
- const index$O = require("./components/Slider/index.cjs");
60
- const index$P = require("./components/Snippet/index.cjs");
61
- const index$Q = require("./components/Stack/index.cjs");
62
- const index$R = require("./components/Status/index.cjs");
63
- const index$S = require("./components/StepList/index.cjs");
64
- const index$T = require("./components/Stepper/index.cjs");
65
- const index$U = require("./components/SwitchButton/index.cjs");
66
- const index$V = require("./components/Table/index.cjs");
67
- const index$W = require("./components/Tabs/index.cjs");
68
- const index$X = require("./components/Tag/index.cjs");
69
- const index$Y = require("./components/TagInput/index.cjs");
70
- const index$Z = require("./components/TagList/index.cjs");
71
- const index$_ = require("./components/Text/index.cjs");
72
- const index$$ = require("./components/TextArea/index.cjs");
73
- const index$10 = require("./components/TextInput/index.cjs");
74
- const index$11 = require("./components/TextInputV2/index.cjs");
75
- const index$12 = require("./components/TimeInput/index.cjs");
76
- const index$13 = require("./components/Toaster/index.cjs");
77
- const index$14 = require("./components/Toggle/index.cjs");
78
- const index$15 = require("./components/ToggleGroup/index.cjs");
79
- const index$16 = require("./components/Tooltip/index.cjs");
80
- const index$17 = require("./components/UnitInput/index.cjs");
81
- const index$18 = require("./components/VerificationCode/index.cjs");
29
+ const index$k = require("./components/ExpandableCard/index.cjs");
30
+ const index$l = require("./components/GlobalAlert/index.cjs");
31
+ const index$m = require("./components/LineChart/index.cjs");
32
+ const index$n = require("./components/Link/index.cjs");
33
+ const index$o = require("./components/List/index.cjs");
34
+ const index$p = require("./components/Loader/index.cjs");
35
+ const index$q = require("./components/Menu/index.cjs");
36
+ const index$r = require("./components/MenuV2/index.cjs");
37
+ const index$s = require("./components/Meter/index.cjs");
38
+ const index$t = require("./components/Modal/index.cjs");
39
+ const index$u = require("./components/Notice/index.cjs");
40
+ const index$v = require("./components/Notification/index.cjs");
41
+ const index$w = require("./components/NumberInput/index.cjs");
42
+ const index$x = require("./components/NumberInputV2/index.cjs");
43
+ const index$y = require("./components/Pagination/index.cjs");
44
+ const index$z = require("./components/PasswordCheck/index.cjs");
45
+ const index$A = require("./components/PasswordStrengthMeter/index.cjs");
46
+ const index$B = require("./components/PieChart/index.cjs");
47
+ const index$C = require("./components/Popover/index.cjs");
48
+ const index$D = require("./components/Popup/index.cjs");
49
+ const index$E = require("./components/ProgressBar/index.cjs");
50
+ const index$F = require("./components/Radio/index.cjs");
51
+ const index$G = require("./components/RadioGroup/index.cjs");
52
+ const index$H = require("./components/Row/index.cjs");
53
+ const index$I = require("./components/SearchInput/index.cjs");
54
+ const index$J = require("./components/SelectableCard/index.cjs");
55
+ const index$K = require("./components/SelectableCardGroup/index.cjs");
56
+ const index$L = require("./components/SelectInput/index.cjs");
57
+ const index$M = require("./components/SelectInputV2/index.cjs");
58
+ const index$N = require("./components/Separator/index.cjs");
59
+ const index$O = require("./components/Skeleton/index.cjs");
60
+ const index$P = require("./components/Slider/index.cjs");
61
+ const index$Q = require("./components/Snippet/index.cjs");
62
+ const index$R = require("./components/Stack/index.cjs");
63
+ const index$S = require("./components/Status/index.cjs");
64
+ const index$T = require("./components/StepList/index.cjs");
65
+ const index$U = require("./components/Stepper/index.cjs");
66
+ const index$V = require("./components/SwitchButton/index.cjs");
67
+ const index$W = require("./components/Table/index.cjs");
68
+ const index$X = require("./components/Tabs/index.cjs");
69
+ const index$Y = require("./components/Tag/index.cjs");
70
+ const index$Z = require("./components/TagInput/index.cjs");
71
+ const index$_ = require("./components/TagList/index.cjs");
72
+ const index$$ = require("./components/Text/index.cjs");
73
+ const index$10 = require("./components/TextArea/index.cjs");
74
+ const index$11 = require("./components/TextInput/index.cjs");
75
+ const index$12 = require("./components/TextInputV2/index.cjs");
76
+ const index$13 = require("./components/TimeInput/index.cjs");
77
+ const index$14 = require("./components/Toaster/index.cjs");
78
+ const index$15 = require("./components/Toggle/index.cjs");
79
+ const index$16 = require("./components/ToggleGroup/index.cjs");
80
+ const index$17 = require("./components/Tooltip/index.cjs");
81
+ const index$18 = require("./components/UnitInput/index.cjs");
82
+ const index$19 = require("./components/VerificationCode/index.cjs");
82
83
  const icons = require("@ultraviolet/icons");
83
84
  exports.extendTheme = index.extendTheme;
84
85
  Object.defineProperty(exports, "darkTheme", {
@@ -141,61 +142,62 @@ exports.DateInput = index$g.DateInput;
141
142
  exports.Dialog = index$h.Dialog;
142
143
  exports.EmptyState = index$i.EmptyState;
143
144
  exports.Expandable = index$j.Expandable;
144
- exports.GlobalAlert = index$k.GlobalAlert;
145
- exports.LineChart = index$l.LineChart;
146
- exports.Link = index$m.Link;
147
- exports.List = index$n.List;
148
- exports.Loader = index$o.Loader;
149
- exports.Menu = index$p.Menu;
150
- exports.MenuV2 = index$q.MenuV2;
151
- exports.Meter = index$r.Meter;
152
- exports.Modal = index$s.Modal;
153
- exports.Notice = index$t.Notice;
154
- exports.NotificationContainer = index$u.NotificationContainer;
155
- exports.notification = index$u.notification;
156
- exports.NumberInput = index$v.NumberInput;
157
- exports.NumberInputV2 = index$w.NumberInputV2;
158
- exports.Pagination = index$x.Pagination;
159
- exports.PasswordCheck = index$y.PasswordCheck;
160
- exports.PasswordStrengthMeter = index$z.PasswordStrengthMeter;
161
- exports.PieChart = index$A.PieChart;
162
- exports.Popover = index$B.Popover;
163
- exports.Popup = index$C.Popup;
164
- exports.ProgressBar = index$D.ProgressBar;
165
- exports.Radio = index$E.Radio;
166
- exports.RadioGroup = index$F.RadioGroup;
167
- exports.Row = index$G.Row;
168
- exports.SearchInput = index$H.SearchInput;
169
- exports.SelectableCard = index$I.SelectableCard;
170
- exports.SelectableCardGroup = index$J.SelectableCardGroup;
171
- exports.SelectInput = index$K.SelectInput;
172
- exports.SelectInputV2 = index$L.SelectInputV2;
173
- exports.Separator = index$M.Separator;
174
- exports.Skeleton = index$N.Skeleton;
175
- exports.Slider = index$O.Slider;
176
- exports.Snippet = index$P.Snippet;
177
- exports.Stack = index$Q.Stack;
178
- exports.Status = index$R.Status;
179
- exports.StepList = index$S.StepList;
180
- exports.Stepper = index$T.Stepper;
181
- exports.SwitchButton = index$U.SwitchButton;
182
- exports.Table = index$V.Table;
183
- exports.Tabs = index$W.Tabs;
184
- exports.Tag = index$X.Tag;
185
- exports.TagInput = index$Y.TagInput;
186
- exports.TagList = index$Z.TagList;
187
- exports.Text = index$_.Text;
188
- exports.TextArea = index$$.TextArea;
189
- exports.TextInput = index$10.TextInput;
190
- exports.TextInputV2 = index$11.TextInputV2;
191
- exports.TimeInput = index$12.TimeInput;
192
- exports.ToastContainer = index$13.ToastContainer;
193
- exports.toast = index$13.toast;
194
- exports.Toggle = index$14.Toggle;
195
- exports.ToggleGroup = index$15.ToggleGroup;
196
- exports.Tooltip = index$16.Tooltip;
197
- exports.UnitInput = index$17.UnitInput;
198
- exports.VerificationCode = index$18.VerificationCode;
145
+ exports.ExpandableCard = index$k.ExpandableCard;
146
+ exports.GlobalAlert = index$l.GlobalAlert;
147
+ exports.LineChart = index$m.LineChart;
148
+ exports.Link = index$n.Link;
149
+ exports.List = index$o.List;
150
+ exports.Loader = index$p.Loader;
151
+ exports.Menu = index$q.Menu;
152
+ exports.MenuV2 = index$r.MenuV2;
153
+ exports.Meter = index$s.Meter;
154
+ exports.Modal = index$t.Modal;
155
+ exports.Notice = index$u.Notice;
156
+ exports.NotificationContainer = index$v.NotificationContainer;
157
+ exports.notification = index$v.notification;
158
+ exports.NumberInput = index$w.NumberInput;
159
+ exports.NumberInputV2 = index$x.NumberInputV2;
160
+ exports.Pagination = index$y.Pagination;
161
+ exports.PasswordCheck = index$z.PasswordCheck;
162
+ exports.PasswordStrengthMeter = index$A.PasswordStrengthMeter;
163
+ exports.PieChart = index$B.PieChart;
164
+ exports.Popover = index$C.Popover;
165
+ exports.Popup = index$D.Popup;
166
+ exports.ProgressBar = index$E.ProgressBar;
167
+ exports.Radio = index$F.Radio;
168
+ exports.RadioGroup = index$G.RadioGroup;
169
+ exports.Row = index$H.Row;
170
+ exports.SearchInput = index$I.SearchInput;
171
+ exports.SelectableCard = index$J.SelectableCard;
172
+ exports.SelectableCardGroup = index$K.SelectableCardGroup;
173
+ exports.SelectInput = index$L.SelectInput;
174
+ exports.SelectInputV2 = index$M.SelectInputV2;
175
+ exports.Separator = index$N.Separator;
176
+ exports.Skeleton = index$O.Skeleton;
177
+ exports.Slider = index$P.Slider;
178
+ exports.Snippet = index$Q.Snippet;
179
+ exports.Stack = index$R.Stack;
180
+ exports.Status = index$S.Status;
181
+ exports.StepList = index$T.StepList;
182
+ exports.Stepper = index$U.Stepper;
183
+ exports.SwitchButton = index$V.SwitchButton;
184
+ exports.Table = index$W.Table;
185
+ exports.Tabs = index$X.Tabs;
186
+ exports.Tag = index$Y.Tag;
187
+ exports.TagInput = index$Z.TagInput;
188
+ exports.TagList = index$_.TagList;
189
+ exports.Text = index$$.Text;
190
+ exports.TextArea = index$10.TextArea;
191
+ exports.TextInput = index$11.TextInput;
192
+ exports.TextInputV2 = index$12.TextInputV2;
193
+ exports.TimeInput = index$13.TimeInput;
194
+ exports.ToastContainer = index$14.ToastContainer;
195
+ exports.toast = index$14.toast;
196
+ exports.Toggle = index$15.Toggle;
197
+ exports.ToggleGroup = index$16.ToggleGroup;
198
+ exports.Tooltip = index$17.Tooltip;
199
+ exports.UnitInput = index$18.UnitInput;
200
+ exports.VerificationCode = index$19.VerificationCode;
199
201
  Object.defineProperty(exports, "Icon", {
200
202
  enumerable: true,
201
203
  get: () => icons.Icon
package/dist/index.js CHANGED
@@ -24,6 +24,7 @@ import { DateInput } from "./components/DateInput/index.js";
24
24
  import { Dialog } from "./components/Dialog/index.js";
25
25
  import { EmptyState } from "./components/EmptyState/index.js";
26
26
  import { Expandable } from "./components/Expandable/index.js";
27
+ import { ExpandableCard } from "./components/ExpandableCard/index.js";
27
28
  import { GlobalAlert } from "./components/GlobalAlert/index.js";
28
29
  import { LineChart } from "./components/LineChart/index.js";
29
30
  import { Link } from "./components/Link/index.js";
@@ -100,6 +101,7 @@ export {
100
101
  Dialog,
101
102
  EmptyState,
102
103
  Expandable,
104
+ ExpandableCard,
103
105
  GlobalAlert,
104
106
  Icon,
105
107
  LineChart,
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@ultraviolet/ui",
3
- "version": "1.61.1",
3
+ "version": "1.63.0",
4
4
  "description": "Ultraviolet UI",
5
5
  "homepage": "https://github.com/scaleway/ultraviolet#readme",
6
6
  "repository": {
@@ -60,7 +60,7 @@
60
60
  "react-dom": "18.3.1"
61
61
  },
62
62
  "devDependencies": {
63
- "@babel/core": "7.24.7",
63
+ "@babel/core": "7.24.9",
64
64
  "@emotion/react": "11.11.4",
65
65
  "@emotion/styled": "11.11.5",
66
66
  "@types/react": "18.3.3",
@@ -68,7 +68,7 @@
68
68
  "@types/react-dom": "18.3.0",
69
69
  "react": "18.3.1",
70
70
  "react-dom": "18.3.1",
71
- "@ultraviolet/icons": "2.13.1",
71
+ "@ultraviolet/icons": "2.14.0",
72
72
  "@ultraviolet/themes": "1.12.2",
73
73
  "@utils/test": "0.0.1"
74
74
  },
@@ -88,7 +88,7 @@
88
88
  "react-use-clipboard": "1.0.9",
89
89
  "reakit": "1.3.11",
90
90
  "@ultraviolet/themes": "1.12.2",
91
- "@ultraviolet/icons": "2.13.1"
91
+ "@ultraviolet/icons": "2.14.0"
92
92
  },
93
93
  "scripts": {
94
94
  "prebuild": "shx rm -rf dist",