@ultraviolet/ui 1.53.0 → 1.53.2
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/components/Expandable/index.cjs +4 -9
- package/dist/components/Expandable/index.d.ts +0 -2
- package/dist/components/Expandable/index.js +4 -9
- package/dist/components/SelectInputV2/Dropdown.cjs +11 -11
- package/dist/components/SelectInputV2/Dropdown.js +11 -11
- package/dist/components/SelectInputV2/SelectBar.cjs +12 -26
- package/dist/components/SelectInputV2/SelectBar.js +12 -26
- package/dist/components/SelectInputV2/index.cjs +7 -7
- package/dist/components/SelectInputV2/index.js +7 -7
- package/dist/components/SelectInputV2/types.d.ts +1 -1
- package/dist/components/UnitInput/index.cjs +180 -0
- package/dist/components/UnitInput/index.d.ts +31 -0
- package/dist/components/UnitInput/index.js +178 -0
- package/dist/components/index.d.ts +1 -0
- package/dist/index.cjs +2 -0
- package/dist/index.js +2 -0
- package/package.json +2 -2
|
@@ -0,0 +1,180 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, Symbol.toStringTag, { value: "Module" });
|
|
3
|
+
const jsxRuntime = require("@emotion/react/jsx-runtime");
|
|
4
|
+
const _styled = require("@emotion/styled/base");
|
|
5
|
+
const icons = require("@ultraviolet/icons");
|
|
6
|
+
const React = require("react");
|
|
7
|
+
const index$1 = require("../SelectInputV2/index.cjs");
|
|
8
|
+
const index = require("../Stack/index.cjs");
|
|
9
|
+
const index$2 = require("../Text/index.cjs");
|
|
10
|
+
const _interopDefaultCompat = (e) => e && typeof e === "object" && "default" in e ? e : { default: e };
|
|
11
|
+
const _styled__default = /* @__PURE__ */ _interopDefaultCompat(_styled);
|
|
12
|
+
const INPUT_SIZE_HEIGHT = {
|
|
13
|
+
large: 48,
|
|
14
|
+
medium: 40,
|
|
15
|
+
small: 32
|
|
16
|
+
};
|
|
17
|
+
const StyledNumberInputWrapper = /* @__PURE__ */ _styled__default.default("div", process.env.NODE_ENV === "production" ? {
|
|
18
|
+
target: "ex81jph3"
|
|
19
|
+
} : {
|
|
20
|
+
target: "ex81jph3",
|
|
21
|
+
label: "StyledNumberInputWrapper"
|
|
22
|
+
})("display:flex;align-items:center;flex-direction:row;padding-right:", ({
|
|
23
|
+
theme
|
|
24
|
+
}) => theme.space["2"], ";border-right:1px solid ", ({
|
|
25
|
+
theme
|
|
26
|
+
}) => 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"]} */"));
|
|
27
|
+
const StyledInput = /* @__PURE__ */ _styled__default.default("input", process.env.NODE_ENV === "production" ? {
|
|
28
|
+
target: "ex81jph2"
|
|
29
|
+
} : {
|
|
30
|
+
target: "ex81jph2",
|
|
31
|
+
label: "StyledInput"
|
|
32
|
+
})("flex:1;border:none;outline:none;height:100%;padding-left:", ({
|
|
33
|
+
theme
|
|
34
|
+
}) => theme.space["2"], ";background:transparent;color:", ({
|
|
35
|
+
theme
|
|
36
|
+
}) => theme.colors.neutral.text, ";&::placeholder{color:", ({
|
|
37
|
+
theme
|
|
38
|
+
}) => theme.colors.neutral.textWeak, ";}&:disabled{cursor:not-allowed;&::placeholder{color:", ({
|
|
39
|
+
theme
|
|
40
|
+
}) => 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"]} */"));
|
|
41
|
+
const UnitInputWrapper = /* @__PURE__ */ _styled__default.default(index.Stack, process.env.NODE_ENV === "production" ? {
|
|
42
|
+
target: "ex81jph1"
|
|
43
|
+
} : {
|
|
44
|
+
target: "ex81jph1",
|
|
45
|
+
label: "UnitInputWrapper"
|
|
46
|
+
})("border:1px solid ", ({
|
|
47
|
+
theme
|
|
48
|
+
}) => theme.colors.neutral.border, ";border-radius:", ({
|
|
49
|
+
theme
|
|
50
|
+
}) => theme.radii.default, ";&:not([data-disabled='true']):not([data-readonly='true']):not(\n [data-success='true']\n ):not([data-error='true']):focus,:not([data-disabled='true']):not([data-readonly='true']):not(\n [data-success='true']\n ):not([data-error='true'])active{box-shadow:", ({
|
|
51
|
+
theme
|
|
52
|
+
}) => theme.shadows.focusPrimary, ";border-color:", ({
|
|
53
|
+
theme
|
|
54
|
+
}) => theme.colors.primary.borderHover, ";&>", StyledNumberInputWrapper, "{border-right-color:", ({
|
|
55
|
+
theme
|
|
56
|
+
}) => theme.colors.primary.border, ";}}&:not([data-disabled='true']):not([data-error='true']):not(\n [data-success='true']\n ):not([data-readonly='true']):hover,:not([data-disabled='true']):not([data-error='true']):not(\n [data-success='true']\n ):focus{text-decoration:none;border-color:", ({
|
|
57
|
+
theme
|
|
58
|
+
}) => theme.colors.primary.border, ";&>", StyledNumberInputWrapper, "{border-right-color:", ({
|
|
59
|
+
theme
|
|
60
|
+
}) => theme.colors.primary.border, ";}}&[data-readonly='true']{background:", ({
|
|
61
|
+
theme
|
|
62
|
+
}) => theme.colors.neutral.backgroundWeak, ";border-color:", ({
|
|
63
|
+
theme
|
|
64
|
+
}) => theme.colors.neutral.border, ";cursor:default;}&[data-readonly='true']:active{border-color:", ({
|
|
65
|
+
theme
|
|
66
|
+
}) => theme.colors.neutral.border, ";}&[data-size='small']{height:", INPUT_SIZE_HEIGHT["small"], "px;}&[data-size='medium']{height:", INPUT_SIZE_HEIGHT["medium"], "px;}&[data-size='large']{height:", INPUT_SIZE_HEIGHT["large"], "px;}&[data-success='true']{border:1px solid ", ({
|
|
67
|
+
theme
|
|
68
|
+
}) => theme.colors.success.border, ";&>", StyledNumberInputWrapper, "{border-right-color:", ({
|
|
69
|
+
theme
|
|
70
|
+
}) => theme.colors.success.border, ";}}&[data-success='true']:active{border:1px solid ", ({
|
|
71
|
+
theme
|
|
72
|
+
}) => theme.colors.success.border, ";box-shadow:", ({
|
|
73
|
+
theme
|
|
74
|
+
}) => theme.shadows.focusSuccess, ";&>", StyledNumberInputWrapper, "{border-right-color:", ({
|
|
75
|
+
theme
|
|
76
|
+
}) => theme.colors.success.border, ";}}&[data-error='true']{border:1px solid ", ({
|
|
77
|
+
theme
|
|
78
|
+
}) => theme.colors.danger.border, ";&>", StyledNumberInputWrapper, "{border-right-color:", ({
|
|
79
|
+
theme
|
|
80
|
+
}) => theme.colors.danger.border, ";}&>", StyledNumberInputWrapper, ":hover{border-right-color:", ({
|
|
81
|
+
theme
|
|
82
|
+
}) => theme.colors.danger.border, ";}}&[data-error='true']:active{border:1px solid ", ({
|
|
83
|
+
theme
|
|
84
|
+
}) => theme.colors.danger.border, ";box-shadow:", ({
|
|
85
|
+
theme
|
|
86
|
+
}) => theme.shadows.focusDanger, ";&>", StyledNumberInputWrapper, "{border-right-color:", ({
|
|
87
|
+
theme
|
|
88
|
+
}) => theme.colors.danger.border, ";}&>", StyledNumberInputWrapper, ":hover{border-right-color:", ({
|
|
89
|
+
theme
|
|
90
|
+
}) => theme.colors.danger.border, ";}}&[data-disabled='true']{background:", ({
|
|
91
|
+
theme
|
|
92
|
+
}) => theme.colors.neutral.backgroundDisabled, ";border-color:", ({
|
|
93
|
+
theme
|
|
94
|
+
}) => theme.colors.neutral.borderDisabled, ";cursor:not-allowed;&>", StyledNumberInputWrapper, "{border-right-color:", ({
|
|
95
|
+
theme
|
|
96
|
+
}) => 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"]} */"));
|
|
97
|
+
const CustomSelectInput = /* @__PURE__ */ _styled__default.default(index$1.SelectInputV2, process.env.NODE_ENV === "production" ? {
|
|
98
|
+
target: "ex81jph0"
|
|
99
|
+
} : {
|
|
100
|
+
target: "ex81jph0",
|
|
101
|
+
label: "CustomSelectInput"
|
|
102
|
+
})("#unit{border:none;background:transparent;}", ({
|
|
103
|
+
width
|
|
104
|
+
}) => 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"]} */"));
|
|
105
|
+
const UnitInput = ({
|
|
106
|
+
id,
|
|
107
|
+
name = "",
|
|
108
|
+
max = Number.MAX_SAFE_INTEGER,
|
|
109
|
+
min = 0,
|
|
110
|
+
autoFocus = false,
|
|
111
|
+
size = "large",
|
|
112
|
+
placeholder = "0",
|
|
113
|
+
placeholderUnit = "Select item",
|
|
114
|
+
onChange,
|
|
115
|
+
onChangeUnitValue,
|
|
116
|
+
value,
|
|
117
|
+
unitValue,
|
|
118
|
+
selectInputWidth = 200,
|
|
119
|
+
disabled = false,
|
|
120
|
+
options,
|
|
121
|
+
className,
|
|
122
|
+
label,
|
|
123
|
+
step = 1,
|
|
124
|
+
error,
|
|
125
|
+
required,
|
|
126
|
+
helper,
|
|
127
|
+
unitError,
|
|
128
|
+
success,
|
|
129
|
+
"data-testid": dataTestId,
|
|
130
|
+
width,
|
|
131
|
+
labelInformation,
|
|
132
|
+
readOnly,
|
|
133
|
+
onFocus,
|
|
134
|
+
onBlur,
|
|
135
|
+
onKeyDown
|
|
136
|
+
}) => {
|
|
137
|
+
const [val, setVal] = React.useState(value);
|
|
138
|
+
const localId = React.useId();
|
|
139
|
+
const sentiment = React.useMemo(() => {
|
|
140
|
+
if (error) {
|
|
141
|
+
return "danger";
|
|
142
|
+
}
|
|
143
|
+
if (success) {
|
|
144
|
+
return "success";
|
|
145
|
+
}
|
|
146
|
+
return "neutral";
|
|
147
|
+
}, [error, success]);
|
|
148
|
+
return /* @__PURE__ */ jsxRuntime.jsxs(index.Stack, { gap: 0.5, children: [
|
|
149
|
+
label ? /* @__PURE__ */ jsxRuntime.jsxs(index.Stack, { direction: "row", gap: 0.5, alignItems: "center", children: [
|
|
150
|
+
/* @__PURE__ */ jsxRuntime.jsx(index$2.Text, { as: "label", variant: size === "large" ? "bodyStrong" : "bodySmallStrong", disabled, htmlFor: id ?? localId, children: label }),
|
|
151
|
+
required ? /* @__PURE__ */ jsxRuntime.jsx(icons.Icon, { name: "asterisk", sentiment: "danger", size: 8 }) : null,
|
|
152
|
+
labelInformation ?? null
|
|
153
|
+
] }) : null,
|
|
154
|
+
/* @__PURE__ */ jsxRuntime.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: [
|
|
155
|
+
/* @__PURE__ */ jsxRuntime.jsxs(StyledNumberInputWrapper, { id: "input-field", children: [
|
|
156
|
+
/* @__PURE__ */ jsxRuntime.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) => {
|
|
157
|
+
const numericValue = Number.parseInt(event.target.value, 10);
|
|
158
|
+
if (numericValue > max) {
|
|
159
|
+
setVal(max);
|
|
160
|
+
onChange?.(max);
|
|
161
|
+
} else if (numericValue < min) {
|
|
162
|
+
setVal(min);
|
|
163
|
+
onChange?.(min);
|
|
164
|
+
} else {
|
|
165
|
+
setVal(numericValue);
|
|
166
|
+
onChange?.(numericValue);
|
|
167
|
+
}
|
|
168
|
+
}, placeholder, max, readOnly, min, step, "data-success": success, "data-error": error, "data-testid": "unit-input", required, className }),
|
|
169
|
+
error ? /* @__PURE__ */ jsxRuntime.jsx(icons.Icon, { name: "alert", sentiment: "danger" }) : null,
|
|
170
|
+
success && !error ? /* @__PURE__ */ jsxRuntime.jsx(icons.Icon, { name: "checkbox-circle-outline", sentiment: "success" }) : null
|
|
171
|
+
] }),
|
|
172
|
+
/* @__PURE__ */ jsxRuntime.jsx(CustomSelectInput, { width: selectInputWidth, "data-disabled": disabled, id: "unit", name: `${name}-unit`, onChange: (newValue) => {
|
|
173
|
+
onChangeUnitValue?.(newValue);
|
|
174
|
+
}, error: unitError, value: unitValue, options, searchable: false, clearable: false, placeholder: placeholderUnit, disabled: disabled || options.length === 1, size, multiselect: false, readOnly })
|
|
175
|
+
] }),
|
|
176
|
+
error || typeof success === "string" || typeof helper === "string" ? /* @__PURE__ */ jsxRuntime.jsx(index$2.Text, { as: "p", variant: "caption", sentiment, disabled, children: error || success || helper }) : null,
|
|
177
|
+
!error && !success && typeof helper !== "string" && helper ? helper : null
|
|
178
|
+
] });
|
|
179
|
+
};
|
|
180
|
+
exports.UnitInput = UnitInput;
|
|
@@ -0,0 +1,31 @@
|
|
|
1
|
+
import type { ComponentProps, InputHTMLAttributes, ReactNode } from 'react';
|
|
2
|
+
import type { OptionType } from '../SelectInputV2/types';
|
|
3
|
+
import { Stack } from '../Stack';
|
|
4
|
+
type UnitInputValue = {
|
|
5
|
+
inputValue: number;
|
|
6
|
+
unit: string;
|
|
7
|
+
};
|
|
8
|
+
type UnitInputProps = {
|
|
9
|
+
className?: string;
|
|
10
|
+
max?: number;
|
|
11
|
+
min?: number;
|
|
12
|
+
value?: UnitInputValue['inputValue'];
|
|
13
|
+
unitValue?: UnitInputValue['unit'];
|
|
14
|
+
onChange?: (value: UnitInputValue['inputValue']) => void;
|
|
15
|
+
onChangeUnitValue?: (values: string) => void;
|
|
16
|
+
options: OptionType[];
|
|
17
|
+
selectInputWidth?: number;
|
|
18
|
+
size?: 'small' | 'medium' | 'large';
|
|
19
|
+
'data-testid'?: string;
|
|
20
|
+
helper?: string;
|
|
21
|
+
unitError?: string;
|
|
22
|
+
width?: ComponentProps<typeof Stack>['width'];
|
|
23
|
+
placeholderUnit?: string;
|
|
24
|
+
error?: boolean | string;
|
|
25
|
+
success?: boolean | string;
|
|
26
|
+
label?: string;
|
|
27
|
+
labelInformation?: ReactNode;
|
|
28
|
+
step?: number | string;
|
|
29
|
+
} & Pick<InputHTMLAttributes<HTMLInputElement>, 'onFocus' | 'onBlur' | 'name' | 'id' | 'placeholder' | 'disabled' | 'readOnly' | 'required' | 'autoFocus' | 'onKeyDown'>;
|
|
30
|
+
export declare const UnitInput: ({ id, name, max, min, autoFocus, size, placeholder, placeholderUnit, onChange, onChangeUnitValue, value, unitValue, selectInputWidth, disabled, options, className, label, step, error, required, helper, unitError, success, "data-testid": dataTestId, width, labelInformation, readOnly, onFocus, onBlur, onKeyDown, }: UnitInputProps) => import("@emotion/react/jsx-runtime").JSX.Element;
|
|
31
|
+
export {};
|