@ultraviolet/ui 1.95.9 → 1.95.10
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.
|
@@ -28,7 +28,7 @@ const BasicPrefixStack = /* @__PURE__ */ _styled__default.default(index.Stack, p
|
|
|
28
28
|
theme
|
|
29
29
|
}) => theme.space["2"], ';&[data-size="small"]{padding:', ({
|
|
30
30
|
theme
|
|
31
|
-
}) => theme.space["1"], ";}border-right:1px solid;border-color:inherit;" + (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/TextInputV2/index.tsx"],"names":[],"mappings":"AAkC6C","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextInputV2/index.tsx","sourcesContent":["'use client'\n\nimport styled from '@emotion/styled'\nimport { AlertCircleIcon, CheckCircleIcon } from '@ultraviolet/icons'\nimport type {\n  ChangeEvent,\n  ChangeEventHandler,\n  InputHTMLAttributes,\n  ReactNode,\n} from 'react'\nimport {\n  forwardRef,\n  useCallback,\n  useId,\n  useImperativeHandle,\n  useMemo,\n  useRef,\n  useState,\n} from 'react'\nimport { Button } from '../Button'\nimport { Label } from '../Label'\nimport { Loader } from '../Loader'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\nimport { Tooltip } from '../Tooltip'\n\n// SIZE\nexport const TEXTINPUT_SIZE_HEIGHT = {\n  small: '400', // sizing theme tokens key\n  medium: '500',\n  large: '600',\n} as const\ntype TextInputSize = keyof typeof TEXTINPUT_SIZE_HEIGHT\n\nexport const BasicPrefixStack = styled(Stack)`\n  padding: ${({ theme }) => theme.space['2']};\n\n  &[data-size=\"small\"] {\n    padding: ${({ theme }) => theme.space['1']};\n  }\n  border-right: 1px solid;\n  border-color: inherit;\n`\n\nconst StateStack = styled(Stack)`\n  padding: ${({ theme }) => `0 ${theme.space['2']}`};\n`\n\nexport const BasicSuffixStack = styled(Stack)`\n  padding: ${({ theme }) => `0 ${theme.space['2']}`};\n  border-left: 1px solid;\n  border-color: inherit;\n`\n\nconst CTASuffixStack = styled(Stack)`\n  padding: ${({ theme }) => `0 ${theme.space['1']}`};\n  border-left: 1px solid;\n  border-color: inherit;\n`\n\nexport const StyledInput = styled.input<{\n  'data-size': TextInputSize\n}>`\n  flex: 1;\n  border: none;\n  outline: none;\n  height: 100%;\n  width: 100%;\n  padding-left: ${({ theme }) => theme.space['2']};\n  background: transparent;\n  font-size: ${({ theme }) => theme.typography.bodySmall.fontSize};\n\n  &[data-size='large'] {\n    font-size: ${({ theme }) => theme.typography.body.fontSize};\n  }\n\n  &[data-size='small'] {\n    padding-left: ${({ theme }) => theme.space['1']};\n  }\n`\n\ntype StyledInputWrapperProps = {\n  hasFocus: boolean\n  size: TextInputSize\n}\nconst StyledInputWrapper = styled('div', {\n  shouldForwardProp: prop => !['hasFocus', 'size'].includes(prop),\n})<StyledInputWrapperProps>`\n  display: flex;\n  flex-direction: row;\n  height: ${({ size, theme }) => theme.sizing[TEXTINPUT_SIZE_HEIGHT[size]]};\n\n  background: ${({ theme }) => theme.colors.neutral.background};\n  border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  border-radius: ${({ theme }) => theme.radii.default};\n\n  & > ${StyledInput} {\n    color: ${({ theme }) => theme.colors.neutral.text};\n\n    &::placeholder {\n      color: ${({ theme }) => theme.colors.neutral.textWeak};\n    }\n  }\n\n  &[data-success='true'] {\n    border-color: ${({ theme }) => theme.colors.success.border};\n  }\n\n  &[data-error='true'] {\n    border-color: ${({ theme }) => theme.colors.danger.border};\n  }\n\n  &[data-readonly='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n    border-color: ${({ theme }) => theme.colors.neutral.border};\n  }\n\n  &[data-disabled='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    border-color: ${({ theme }) => theme.colors.neutral.borderDisabled};\n\n    & > ${StyledInput} {\n      color: ${({ theme }) => theme.colors.neutral.textDisabled};\n\n      &::placeholder {\n        color: ${({ theme }) => theme.colors.neutral.textWeakDisabled};\n      }\n    }\n  }\n\n  &:not([data-disabled='true']):not([data-readonly]):hover {\n    border-color: ${({ theme }) => theme.colors.primary.border};\n  }\n\n  ${({ theme, hasFocus }) =>\n    hasFocus\n      ? `\n  box-shadow: ${theme.shadows.focusPrimary};\n  border: 1px solid ${theme.colors.primary.border};\n`\n      : null};\n`\n\ntype TextInputProps = {\n  className?: string\n  clearable?: boolean\n  'data-testid'?: string\n  error?: string\n  helper?: ReactNode\n  label?: string\n  labelDescription?: ReactNode\n  loading?: boolean\n  minLength?: number\n  maxLength?: number\n  onRandomize?: () => void\n  prefix?: ReactNode\n  size?: TextInputSize\n  success?: string | boolean\n  suffix?: ReactNode\n  tooltip?: string\n  type?: 'text' | 'password' | 'url' | 'email'\n  value?: string\n  defaultValue?: string\n  onChangeValue?: (value: string) => void\n} & Pick<\n  InputHTMLAttributes<HTMLInputElement>,\n  | 'onFocus'\n  | 'onBlur'\n  | 'name'\n  | 'id'\n  | 'placeholder'\n  | 'aria-label'\n  | 'aria-labelledby'\n  | 'disabled'\n  | 'readOnly'\n  | 'required'\n  | 'autoFocus'\n  | 'tabIndex'\n  | 'autoComplete'\n  | 'onKeyDown'\n  | 'onKeyUp'\n  | 'role'\n  | 'aria-live'\n  | 'aria-atomic'\n  | 'onChange'\n>\n\n/**\n * This component offers an extended input HTML. The component can be controlled or uncontrolled.\n * To control the component, you need to pass the value and the `onChange` function.\n * If you don't pass the `onChange` function, the component will be uncontrolled and you can set the default value using `defaultValue`\n */\nexport const TextInputV2 = forwardRef<HTMLInputElement, TextInputProps>(\n  (\n    {\n      id,\n      className,\n      tabIndex,\n      value,\n      onChange,\n      onChangeValue,\n      placeholder,\n      disabled = false,\n      readOnly = false,\n      success,\n      error,\n      helper,\n      tooltip,\n      label,\n      autoFocus,\n      required = false,\n      'data-testid': dataTestId,\n      name,\n      onFocus,\n      onBlur,\n      clearable = false,\n      labelDescription,\n      type = 'text',\n      prefix,\n      suffix,\n      size = 'large',\n      loading,\n      onRandomize,\n      minLength,\n      maxLength,\n      'aria-labelledby': ariaLabelledBy,\n      'aria-label': ariaLabel,\n      autoComplete,\n      onKeyDown,\n      onKeyUp,\n      role,\n      'aria-live': ariaLive,\n      'aria-atomic': ariaAtomic,\n      defaultValue,\n    },\n    ref,\n  ) => {\n    const localId = useId()\n    const [hasFocus, setHasFocus] = useState(false)\n    const inputRef = useRef<HTMLInputElement>(null)\n    useImperativeHandle(ref, () => inputRef.current as HTMLInputElement)\n\n    const [isPasswordVisible, setIsPasswordVisible] = useState(false)\n    const computedType =\n      type === 'password' && isPasswordVisible ? 'text' : type\n\n    const sentiment = useMemo(() => {\n      if (error) {\n        return 'danger'\n      }\n\n      if (success) {\n        return 'success'\n      }\n\n      return 'neutral'\n    }, [error, success])\n\n    const onChangeCallback: ChangeEventHandler<HTMLInputElement> = useCallback(\n      event => {\n        onChange?.(event)\n        onChangeValue?.(event.target.value)\n      },\n      [onChange, onChangeValue],\n    )\n\n    const computedClearable = clearable && !!value\n\n    return (\n      <Stack\n        gap={0.5}\n        className={className}\n        role={role}\n        aria-live={ariaLive}\n        aria-atomic={ariaAtomic}\n      >\n        {label || labelDescription ? (\n          <Label\n            labelDescription={labelDescription}\n            required={required}\n            size={size}\n            htmlFor={id ?? localId}\n            id={ariaLabelledBy}\n          >\n            {label}\n          </Label>\n        ) : null}\n        <div>\n          <Tooltip text={tooltip}>\n            <StyledInputWrapper\n              hasFocus={hasFocus}\n              data-disabled={disabled}\n              data-readonly={readOnly}\n              data-success={!!success}\n              data-error={!!error}\n              size={size}\n            >\n              {prefix ? (\n                <BasicPrefixStack\n                  direction=\"row\"\n                  alignItems=\"center\"\n                  data-size={size}\n                >\n                  {typeof prefix === 'string' ? (\n                    <Text\n                      as=\"span\"\n                      sentiment=\"neutral\"\n                      variant=\"bodySmall\"\n                      disabled={disabled}\n                    >\n                      {prefix}\n                    </Text>\n                  ) : (\n                    prefix\n                  )}\n                </BasicPrefixStack>\n              ) : null}\n              <StyledInput\n                type={computedType}\n                aria-invalid={!!error}\n                id={id ?? localId}\n                tabIndex={tabIndex}\n                autoFocus={autoFocus}\n                disabled={disabled}\n                ref={inputRef}\n                value={value}\n                defaultValue={defaultValue}\n                onChange={onChangeCallback}\n                data-size={size}\n                placeholder={placeholder}\n                data-testid={dataTestId}\n                name={name}\n                onFocus={event => {\n                  setHasFocus(true)\n                  onFocus?.(event)\n                }}\n                onBlur={event => {\n                  setHasFocus(false)\n                  onBlur?.(event)\n                }}\n                readOnly={readOnly}\n                minLength={minLength}\n                maxLength={maxLength}\n                aria-labelledby={ariaLabelledBy}\n                aria-label={ariaLabel}\n                autoComplete={autoComplete}\n                required={required}\n                onKeyDown={onKeyDown}\n                onKeyUp={onKeyUp}\n              />\n              {success || error || loading || computedClearable ? (\n                <StateStack direction=\"row\" gap={1} alignItems=\"center\">\n                  {computedClearable ? (\n                    <Button\n                      aria-label=\"clear value\"\n                      disabled={disabled || !value}\n                      variant=\"ghost\"\n                      size={size === 'small' ? 'xsmall' : 'small'}\n                      icon=\"close\"\n                      onClick={() => {\n                        if (inputRef?.current) {\n                          inputRef.current.value = ''\n                          onChangeCallback({\n                            target: { value: '' },\n                            currentTarget: { value: '' },\n                          } as ChangeEvent<HTMLInputElement>)\n                        }\n                      }}\n                      sentiment=\"neutral\"\n                    />\n                  ) : null}\n                  {success ? (\n                    <CheckCircleIcon\n                      sentiment=\"success\"\n                      size=\"small\"\n                      disabled={disabled}\n                    />\n                  ) : null}\n                  {error ? (\n                    <AlertCircleIcon\n                      sentiment=\"danger\"\n                      size=\"small\"\n                      disabled={disabled}\n                    />\n                  ) : null}\n                  {loading && !disabled ? <Loader active size={16} /> : null}\n                </StateStack>\n              ) : null}\n              {suffix ? (\n                <BasicSuffixStack direction=\"row\" alignItems=\"center\">\n                  {typeof suffix === 'string' ? (\n                    <Text\n                      as=\"span\"\n                      sentiment=\"neutral\"\n                      variant=\"bodySmall\"\n                      disabled={disabled}\n                    >\n                      {suffix}\n                    </Text>\n                  ) : (\n                    suffix\n                  )}\n                </BasicSuffixStack>\n              ) : null}\n              {type === 'password' ? (\n                <CTASuffixStack direction=\"row\" alignItems=\"center\">\n                  <Button\n                    disabled={disabled}\n                    data-testid={\n                      dataTestId ? `${dataTestId}-visibility-button` : undefined\n                    }\n                    aria-label={isPasswordVisible ? 'hide' : 'show'}\n                    onClick={() => {\n                      setIsPasswordVisible(!isPasswordVisible)\n                    }}\n                    variant=\"ghost\"\n                    sentiment=\"neutral\"\n                    icon={isPasswordVisible ? 'eye-off' : 'eye'}\n                    size={size === 'small' ? 'xsmall' : 'small'}\n                  />\n                </CTASuffixStack>\n              ) : null}\n              {onRandomize ? (\n                <CTASuffixStack direction=\"row\" alignItems=\"center\">\n                  <Button\n                    disabled={disabled}\n                    icon=\"auto-fix\"\n                    size={size === 'small' ? 'xsmall' : 'small'}\n                    variant=\"ghost\"\n                    sentiment=\"neutral\"\n                    onClick={onRandomize}\n                  />\n                </CTASuffixStack>\n              ) : null}\n            </StyledInputWrapper>\n          </Tooltip>\n        </div>\n        {error || typeof success === 'string' || typeof helper === 'string' ? (\n          <Text\n            as=\"p\"\n            variant=\"caption\"\n            sentiment={sentiment}\n            prominence={!error && !success ? 'weak' : 'default'}\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)\n"]} */"));
|
|
31
|
+
}) => theme.space["1"], ";}border-right:1px solid;border-color:inherit;" + (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/TextInputV2/index.tsx"],"names":[],"mappings":"AAkC6C","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextInputV2/index.tsx","sourcesContent":["'use client'\n\nimport styled from '@emotion/styled'\nimport { AlertCircleIcon, CheckCircleIcon } from '@ultraviolet/icons'\nimport type {\n  ChangeEvent,\n  ChangeEventHandler,\n  InputHTMLAttributes,\n  ReactNode,\n} from 'react'\nimport {\n  forwardRef,\n  useCallback,\n  useId,\n  useImperativeHandle,\n  useMemo,\n  useRef,\n  useState,\n} from 'react'\nimport { Button } from '../Button'\nimport { Label } from '../Label'\nimport { Loader } from '../Loader'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\nimport { Tooltip } from '../Tooltip'\n\n// SIZE\nexport const TEXTINPUT_SIZE_HEIGHT = {\n  small: '400', // sizing theme tokens key\n  medium: '500',\n  large: '600',\n} as const\ntype TextInputSize = keyof typeof TEXTINPUT_SIZE_HEIGHT\n\nexport const BasicPrefixStack = styled(Stack)`\n  padding: ${({ theme }) => theme.space['2']};\n\n  &[data-size=\"small\"] {\n    padding: ${({ theme }) => theme.space['1']};\n  }\n  border-right: 1px solid;\n  border-color: inherit;\n`\n\nconst StateStack = styled(Stack)`\n  padding: ${({ theme }) => `0 ${theme.space['2']}`};\n`\n\nexport const BasicSuffixStack = styled(Stack)`\n  padding: ${({ theme }) => `0 ${theme.space['2']}`};\n  border-left: 1px solid;\n  border-color: inherit;\n`\n\nconst CTASuffixStack = styled(Stack)`\n  padding: ${({ theme }) => `0 ${theme.space['1']}`};\n  border-left: 1px solid;\n  border-color: inherit;\n`\n\nexport const StyledInput = styled.input<{\n  'data-size': TextInputSize\n}>`\n  flex: 1;\n  border: none;\n  outline: none;\n  height: 100%;\n  width: 100%;\n  padding-left: ${({ theme }) => theme.space['2']};\n  background: transparent;\n  font-size: ${({ theme }) => theme.typography.bodySmall.fontSize};\n\n  &[data-size='large'] {\n    font-size: ${({ theme }) => theme.typography.body.fontSize};\n  }\n\n  &[data-size='small'] {\n    padding-left: ${({ theme }) => theme.space['1']};\n  }\n`\n\ntype StyledInputWrapperProps = {\n  hasFocus: boolean\n  size: TextInputSize\n}\nconst StyledInputWrapper = styled('div', {\n  shouldForwardProp: prop => !['hasFocus', 'size'].includes(prop),\n})<StyledInputWrapperProps>`\n  display: flex;\n  flex-direction: row;\n  height: ${({ size, theme }) => theme.sizing[TEXTINPUT_SIZE_HEIGHT[size]]};\n\n  background: ${({ theme }) => theme.colors.neutral.background};\n  border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  border-radius: ${({ theme }) => theme.radii.default};\n\n  & > ${StyledInput} {\n    color: ${({ theme }) => theme.colors.neutral.text};\n\n    &::placeholder {\n      color: ${({ theme }) => theme.colors.neutral.textWeak};\n    }\n  }\n\n  &[data-success='true'] {\n    border-color: ${({ theme }) => theme.colors.success.border};\n  }\n\n  &[data-error='true'] {\n    border-color: ${({ theme }) => theme.colors.danger.border};\n  }\n\n  &[data-readonly='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n    border-color: ${({ theme }) => theme.colors.neutral.border};\n  }\n\n  &[data-disabled='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    border-color: ${({ theme }) => theme.colors.neutral.borderDisabled};\n\n    & > ${StyledInput} {\n      color: ${({ theme }) => theme.colors.neutral.textDisabled};\n\n      &::placeholder {\n        color: ${({ theme }) => theme.colors.neutral.textWeakDisabled};\n      }\n    }\n  }\n\n  &:not([data-disabled='true']):not([data-readonly]):hover {\n    border-color: ${({ theme }) => theme.colors.primary.border};\n  }\n\n  ${({ theme, hasFocus }) =>\n    hasFocus\n      ? `\n  box-shadow: ${theme.shadows.focusPrimary};\n  border: 1px solid ${theme.colors.primary.border};\n`\n      : null};\n`\n\ntype TextInputProps = {\n  className?: string\n  clearable?: boolean\n  'data-testid'?: string\n  error?: string\n  helper?: ReactNode\n  label?: string\n  labelDescription?: ReactNode\n  loading?: boolean\n  minLength?: number\n  maxLength?: number\n  onRandomize?: () => void\n  prefix?: ReactNode\n  size?: TextInputSize\n  success?: string | boolean\n  suffix?: ReactNode\n  tooltip?: string\n  type?: 'text' | 'password' | 'url' | 'email'\n  value?: string\n  defaultValue?: string\n  onChangeValue?: (value: string) => void\n} & Pick<\n  InputHTMLAttributes<HTMLInputElement>,\n  | 'onFocus'\n  | 'onBlur'\n  | 'name'\n  | 'id'\n  | 'placeholder'\n  | 'aria-label'\n  | 'aria-labelledby'\n  | 'disabled'\n  | 'readOnly'\n  | 'required'\n  | 'autoFocus'\n  | 'tabIndex'\n  | 'autoComplete'\n  | 'onKeyDown'\n  | 'onKeyUp'\n  | 'role'\n  | 'aria-live'\n  | 'aria-atomic'\n  | 'onChange'\n>\n\n/**\n * This component offers an extended input HTML. The component can be controlled or uncontrolled.\n * To control the component, you need to pass the value and the `onChange` function.\n * If you don't pass the `onChange` function, the component will be uncontrolled and you can set the default value using `defaultValue`\n */\nexport const TextInputV2 = forwardRef<HTMLInputElement, TextInputProps>(\n  (\n    {\n      id,\n      className,\n      tabIndex,\n      value,\n      onChange,\n      onChangeValue,\n      placeholder,\n      disabled = false,\n      readOnly = false,\n      success,\n      error,\n      helper,\n      tooltip,\n      label,\n      autoFocus,\n      required = false,\n      'data-testid': dataTestId,\n      name,\n      onFocus,\n      onBlur,\n      clearable = false,\n      labelDescription,\n      type = 'text',\n      prefix,\n      suffix,\n      size = 'large',\n      loading,\n      onRandomize,\n      minLength,\n      maxLength,\n      'aria-labelledby': ariaLabelledBy,\n      'aria-label': ariaLabel,\n      autoComplete,\n      onKeyDown,\n      onKeyUp,\n      role,\n      'aria-live': ariaLive,\n      'aria-atomic': ariaAtomic,\n      defaultValue,\n    },\n    ref,\n  ) => {\n    const localId = useId()\n    const [hasFocus, setHasFocus] = useState(false)\n    const [localValue, setLocalValue] = useState(defaultValue)\n    const inputRef = useRef<HTMLInputElement>(null)\n    useImperativeHandle(ref, () => inputRef.current as HTMLInputElement)\n\n    const [isPasswordVisible, setIsPasswordVisible] = useState(false)\n    const computedType =\n      type === 'password' && isPasswordVisible ? 'text' : type\n\n    const sentiment = useMemo(() => {\n      if (error) {\n        return 'danger'\n      }\n\n      if (success) {\n        return 'success'\n      }\n\n      return 'neutral'\n    }, [error, success])\n\n    const onChangeCallback: ChangeEventHandler<HTMLInputElement> = useCallback(\n      event => {\n        onChange?.(event)\n        onChangeValue?.(event.target.value)\n        setLocalValue(event.target.value)\n      },\n      [onChange, onChangeValue],\n    )\n\n    const computedValue = value !== undefined ? value : localValue\n\n    const computedClearable = clearable && !!computedValue\n\n    return (\n      <Stack\n        gap={0.5}\n        className={className}\n        role={role}\n        aria-live={ariaLive}\n        aria-atomic={ariaAtomic}\n      >\n        {label || labelDescription ? (\n          <Label\n            labelDescription={labelDescription}\n            required={required}\n            size={size}\n            htmlFor={id ?? localId}\n            id={ariaLabelledBy}\n          >\n            {label}\n          </Label>\n        ) : null}\n        <div>\n          <Tooltip text={tooltip}>\n            <StyledInputWrapper\n              hasFocus={hasFocus}\n              data-disabled={disabled}\n              data-readonly={readOnly}\n              data-success={!!success}\n              data-error={!!error}\n              size={size}\n            >\n              {prefix ? (\n                <BasicPrefixStack\n                  direction=\"row\"\n                  alignItems=\"center\"\n                  data-size={size}\n                >\n                  {typeof prefix === 'string' ? (\n                    <Text\n                      as=\"span\"\n                      sentiment=\"neutral\"\n                      variant=\"bodySmall\"\n                      disabled={disabled}\n                    >\n                      {prefix}\n                    </Text>\n                  ) : (\n                    prefix\n                  )}\n                </BasicPrefixStack>\n              ) : null}\n              <StyledInput\n                type={computedType}\n                aria-invalid={!!error}\n                id={id ?? localId}\n                tabIndex={tabIndex}\n                autoFocus={autoFocus}\n                disabled={disabled}\n                ref={inputRef}\n                value={value}\n                defaultValue={defaultValue}\n                onChange={onChangeCallback}\n                data-size={size}\n                placeholder={placeholder}\n                data-testid={dataTestId}\n                name={name}\n                onFocus={event => {\n                  setHasFocus(true)\n                  onFocus?.(event)\n                }}\n                onBlur={event => {\n                  setHasFocus(false)\n                  onBlur?.(event)\n                }}\n                readOnly={readOnly}\n                minLength={minLength}\n                maxLength={maxLength}\n                aria-labelledby={ariaLabelledBy}\n                aria-label={ariaLabel}\n                autoComplete={autoComplete}\n                required={required}\n                onKeyDown={onKeyDown}\n                onKeyUp={onKeyUp}\n              />\n              {success || error || loading || computedClearable ? (\n                <StateStack direction=\"row\" gap={1} alignItems=\"center\">\n                  {computedClearable ? (\n                    <Button\n                      aria-label=\"clear value\"\n                      disabled={disabled || !computedValue}\n                      variant=\"ghost\"\n                      size={size === 'small' ? 'xsmall' : 'small'}\n                      icon=\"close\"\n                      onClick={() => {\n                        if (inputRef?.current) {\n                          inputRef.current.value = ''\n                          setLocalValue('')\n                          onChangeCallback({\n                            target: { value: '' },\n                            currentTarget: { value: '' },\n                          } as ChangeEvent<HTMLInputElement>)\n                        }\n                      }}\n                      sentiment=\"neutral\"\n                    />\n                  ) : null}\n                  {success ? (\n                    <CheckCircleIcon\n                      sentiment=\"success\"\n                      size=\"small\"\n                      disabled={disabled}\n                    />\n                  ) : null}\n                  {error ? (\n                    <AlertCircleIcon\n                      sentiment=\"danger\"\n                      size=\"small\"\n                      disabled={disabled}\n                    />\n                  ) : null}\n                  {loading && !disabled ? <Loader active size={16} /> : null}\n                </StateStack>\n              ) : null}\n              {suffix ? (\n                <BasicSuffixStack direction=\"row\" alignItems=\"center\">\n                  {typeof suffix === 'string' ? (\n                    <Text\n                      as=\"span\"\n                      sentiment=\"neutral\"\n                      variant=\"bodySmall\"\n                      disabled={disabled}\n                    >\n                      {suffix}\n                    </Text>\n                  ) : (\n                    suffix\n                  )}\n                </BasicSuffixStack>\n              ) : null}\n              {type === 'password' ? (\n                <CTASuffixStack direction=\"row\" alignItems=\"center\">\n                  <Button\n                    disabled={disabled}\n                    data-testid={\n                      dataTestId ? `${dataTestId}-visibility-button` : undefined\n                    }\n                    aria-label={isPasswordVisible ? 'hide' : 'show'}\n                    onClick={() => {\n                      setIsPasswordVisible(!isPasswordVisible)\n                    }}\n                    variant=\"ghost\"\n                    sentiment=\"neutral\"\n                    icon={isPasswordVisible ? 'eye-off' : 'eye'}\n                    size={size === 'small' ? 'xsmall' : 'small'}\n                  />\n                </CTASuffixStack>\n              ) : null}\n              {onRandomize ? (\n                <CTASuffixStack direction=\"row\" alignItems=\"center\">\n                  <Button\n                    disabled={disabled}\n                    icon=\"auto-fix\"\n                    size={size === 'small' ? 'xsmall' : 'small'}\n                    variant=\"ghost\"\n                    sentiment=\"neutral\"\n                    onClick={onRandomize}\n                  />\n                </CTASuffixStack>\n              ) : null}\n            </StyledInputWrapper>\n          </Tooltip>\n        </div>\n        {error || typeof success === 'string' || typeof helper === 'string' ? (\n          <Text\n            as=\"p\"\n            variant=\"caption\"\n            sentiment={sentiment}\n            prominence={!error && !success ? 'weak' : 'default'}\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)\n"]} */"));
|
|
32
32
|
const StateStack = /* @__PURE__ */ _styled__default.default(index.Stack, process.env.NODE_ENV === "production" ? {
|
|
33
33
|
target: "e7tir8v4"
|
|
34
34
|
} : {
|
|
@@ -36,7 +36,7 @@ const StateStack = /* @__PURE__ */ _styled__default.default(index.Stack, process
|
|
|
36
36
|
label: "StateStack"
|
|
37
37
|
})("padding:", ({
|
|
38
38
|
theme
|
|
39
|
-
}) => `0 ${theme.space["2"]}`, ";" + (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/TextInputV2/index.tsx"],"names":[],"mappings":"AA4CgC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextInputV2/index.tsx","sourcesContent":["'use client'\n\nimport styled from '@emotion/styled'\nimport { AlertCircleIcon, CheckCircleIcon } from '@ultraviolet/icons'\nimport type {\n  ChangeEvent,\n  ChangeEventHandler,\n  InputHTMLAttributes,\n  ReactNode,\n} from 'react'\nimport {\n  forwardRef,\n  useCallback,\n  useId,\n  useImperativeHandle,\n  useMemo,\n  useRef,\n  useState,\n} from 'react'\nimport { Button } from '../Button'\nimport { Label } from '../Label'\nimport { Loader } from '../Loader'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\nimport { Tooltip } from '../Tooltip'\n\n// SIZE\nexport const TEXTINPUT_SIZE_HEIGHT = {\n  small: '400', // sizing theme tokens key\n  medium: '500',\n  large: '600',\n} as const\ntype TextInputSize = keyof typeof TEXTINPUT_SIZE_HEIGHT\n\nexport const BasicPrefixStack = styled(Stack)`\n  padding: ${({ theme }) => theme.space['2']};\n\n  &[data-size=\"small\"] {\n    padding: ${({ theme }) => theme.space['1']};\n  }\n  border-right: 1px solid;\n  border-color: inherit;\n`\n\nconst StateStack = styled(Stack)`\n  padding: ${({ theme }) => `0 ${theme.space['2']}`};\n`\n\nexport const BasicSuffixStack = styled(Stack)`\n  padding: ${({ theme }) => `0 ${theme.space['2']}`};\n  border-left: 1px solid;\n  border-color: inherit;\n`\n\nconst CTASuffixStack = styled(Stack)`\n  padding: ${({ theme }) => `0 ${theme.space['1']}`};\n  border-left: 1px solid;\n  border-color: inherit;\n`\n\nexport const StyledInput = styled.input<{\n  'data-size': TextInputSize\n}>`\n  flex: 1;\n  border: none;\n  outline: none;\n  height: 100%;\n  width: 100%;\n  padding-left: ${({ theme }) => theme.space['2']};\n  background: transparent;\n  font-size: ${({ theme }) => theme.typography.bodySmall.fontSize};\n\n  &[data-size='large'] {\n    font-size: ${({ theme }) => theme.typography.body.fontSize};\n  }\n\n  &[data-size='small'] {\n    padding-left: ${({ theme }) => theme.space['1']};\n  }\n`\n\ntype StyledInputWrapperProps = {\n  hasFocus: boolean\n  size: TextInputSize\n}\nconst StyledInputWrapper = styled('div', {\n  shouldForwardProp: prop => !['hasFocus', 'size'].includes(prop),\n})<StyledInputWrapperProps>`\n  display: flex;\n  flex-direction: row;\n  height: ${({ size, theme }) => theme.sizing[TEXTINPUT_SIZE_HEIGHT[size]]};\n\n  background: ${({ theme }) => theme.colors.neutral.background};\n  border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  border-radius: ${({ theme }) => theme.radii.default};\n\n  & > ${StyledInput} {\n    color: ${({ theme }) => theme.colors.neutral.text};\n\n    &::placeholder {\n      color: ${({ theme }) => theme.colors.neutral.textWeak};\n    }\n  }\n\n  &[data-success='true'] {\n    border-color: ${({ theme }) => theme.colors.success.border};\n  }\n\n  &[data-error='true'] {\n    border-color: ${({ theme }) => theme.colors.danger.border};\n  }\n\n  &[data-readonly='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n    border-color: ${({ theme }) => theme.colors.neutral.border};\n  }\n\n  &[data-disabled='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    border-color: ${({ theme }) => theme.colors.neutral.borderDisabled};\n\n    & > ${StyledInput} {\n      color: ${({ theme }) => theme.colors.neutral.textDisabled};\n\n      &::placeholder {\n        color: ${({ theme }) => theme.colors.neutral.textWeakDisabled};\n      }\n    }\n  }\n\n  &:not([data-disabled='true']):not([data-readonly]):hover {\n    border-color: ${({ theme }) => theme.colors.primary.border};\n  }\n\n  ${({ theme, hasFocus }) =>\n    hasFocus\n      ? `\n  box-shadow: ${theme.shadows.focusPrimary};\n  border: 1px solid ${theme.colors.primary.border};\n`\n      : null};\n`\n\ntype TextInputProps = {\n  className?: string\n  clearable?: boolean\n  'data-testid'?: string\n  error?: string\n  helper?: ReactNode\n  label?: string\n  labelDescription?: ReactNode\n  loading?: boolean\n  minLength?: number\n  maxLength?: number\n  onRandomize?: () => void\n  prefix?: ReactNode\n  size?: TextInputSize\n  success?: string | boolean\n  suffix?: ReactNode\n  tooltip?: string\n  type?: 'text' | 'password' | 'url' | 'email'\n  value?: string\n  defaultValue?: string\n  onChangeValue?: (value: string) => void\n} & Pick<\n  InputHTMLAttributes<HTMLInputElement>,\n  | 'onFocus'\n  | 'onBlur'\n  | 'name'\n  | 'id'\n  | 'placeholder'\n  | 'aria-label'\n  | 'aria-labelledby'\n  | 'disabled'\n  | 'readOnly'\n  | 'required'\n  | 'autoFocus'\n  | 'tabIndex'\n  | 'autoComplete'\n  | 'onKeyDown'\n  | 'onKeyUp'\n  | 'role'\n  | 'aria-live'\n  | 'aria-atomic'\n  | 'onChange'\n>\n\n/**\n * This component offers an extended input HTML. The component can be controlled or uncontrolled.\n * To control the component, you need to pass the value and the `onChange` function.\n * If you don't pass the `onChange` function, the component will be uncontrolled and you can set the default value using `defaultValue`\n */\nexport const TextInputV2 = forwardRef<HTMLInputElement, TextInputProps>(\n  (\n    {\n      id,\n      className,\n      tabIndex,\n      value,\n      onChange,\n      onChangeValue,\n      placeholder,\n      disabled = false,\n      readOnly = false,\n      success,\n      error,\n      helper,\n      tooltip,\n      label,\n      autoFocus,\n      required = false,\n      'data-testid': dataTestId,\n      name,\n      onFocus,\n      onBlur,\n      clearable = false,\n      labelDescription,\n      type = 'text',\n      prefix,\n      suffix,\n      size = 'large',\n      loading,\n      onRandomize,\n      minLength,\n      maxLength,\n      'aria-labelledby': ariaLabelledBy,\n      'aria-label': ariaLabel,\n      autoComplete,\n      onKeyDown,\n      onKeyUp,\n      role,\n      'aria-live': ariaLive,\n      'aria-atomic': ariaAtomic,\n      defaultValue,\n    },\n    ref,\n  ) => {\n    const localId = useId()\n    const [hasFocus, setHasFocus] = useState(false)\n    const inputRef = useRef<HTMLInputElement>(null)\n    useImperativeHandle(ref, () => inputRef.current as HTMLInputElement)\n\n    const [isPasswordVisible, setIsPasswordVisible] = useState(false)\n    const computedType =\n      type === 'password' && isPasswordVisible ? 'text' : type\n\n    const sentiment = useMemo(() => {\n      if (error) {\n        return 'danger'\n      }\n\n      if (success) {\n        return 'success'\n      }\n\n      return 'neutral'\n    }, [error, success])\n\n    const onChangeCallback: ChangeEventHandler<HTMLInputElement> = useCallback(\n      event => {\n        onChange?.(event)\n        onChangeValue?.(event.target.value)\n      },\n      [onChange, onChangeValue],\n    )\n\n    const computedClearable = clearable && !!value\n\n    return (\n      <Stack\n        gap={0.5}\n        className={className}\n        role={role}\n        aria-live={ariaLive}\n        aria-atomic={ariaAtomic}\n      >\n        {label || labelDescription ? (\n          <Label\n            labelDescription={labelDescription}\n            required={required}\n            size={size}\n            htmlFor={id ?? localId}\n            id={ariaLabelledBy}\n          >\n            {label}\n          </Label>\n        ) : null}\n        <div>\n          <Tooltip text={tooltip}>\n            <StyledInputWrapper\n              hasFocus={hasFocus}\n              data-disabled={disabled}\n              data-readonly={readOnly}\n              data-success={!!success}\n              data-error={!!error}\n              size={size}\n            >\n              {prefix ? (\n                <BasicPrefixStack\n                  direction=\"row\"\n                  alignItems=\"center\"\n                  data-size={size}\n                >\n                  {typeof prefix === 'string' ? (\n                    <Text\n                      as=\"span\"\n                      sentiment=\"neutral\"\n                      variant=\"bodySmall\"\n                      disabled={disabled}\n                    >\n                      {prefix}\n                    </Text>\n                  ) : (\n                    prefix\n                  )}\n                </BasicPrefixStack>\n              ) : null}\n              <StyledInput\n                type={computedType}\n                aria-invalid={!!error}\n                id={id ?? localId}\n                tabIndex={tabIndex}\n                autoFocus={autoFocus}\n                disabled={disabled}\n                ref={inputRef}\n                value={value}\n                defaultValue={defaultValue}\n                onChange={onChangeCallback}\n                data-size={size}\n                placeholder={placeholder}\n                data-testid={dataTestId}\n                name={name}\n                onFocus={event => {\n                  setHasFocus(true)\n                  onFocus?.(event)\n                }}\n                onBlur={event => {\n                  setHasFocus(false)\n                  onBlur?.(event)\n                }}\n                readOnly={readOnly}\n                minLength={minLength}\n                maxLength={maxLength}\n                aria-labelledby={ariaLabelledBy}\n                aria-label={ariaLabel}\n                autoComplete={autoComplete}\n                required={required}\n                onKeyDown={onKeyDown}\n                onKeyUp={onKeyUp}\n              />\n              {success || error || loading || computedClearable ? (\n                <StateStack direction=\"row\" gap={1} alignItems=\"center\">\n                  {computedClearable ? (\n                    <Button\n                      aria-label=\"clear value\"\n                      disabled={disabled || !value}\n                      variant=\"ghost\"\n                      size={size === 'small' ? 'xsmall' : 'small'}\n                      icon=\"close\"\n                      onClick={() => {\n                        if (inputRef?.current) {\n                          inputRef.current.value = ''\n                          onChangeCallback({\n                            target: { value: '' },\n                            currentTarget: { value: '' },\n                          } as ChangeEvent<HTMLInputElement>)\n                        }\n                      }}\n                      sentiment=\"neutral\"\n                    />\n                  ) : null}\n                  {success ? (\n                    <CheckCircleIcon\n                      sentiment=\"success\"\n                      size=\"small\"\n                      disabled={disabled}\n                    />\n                  ) : null}\n                  {error ? (\n                    <AlertCircleIcon\n                      sentiment=\"danger\"\n                      size=\"small\"\n                      disabled={disabled}\n                    />\n                  ) : null}\n                  {loading && !disabled ? <Loader active size={16} /> : null}\n                </StateStack>\n              ) : null}\n              {suffix ? (\n                <BasicSuffixStack direction=\"row\" alignItems=\"center\">\n                  {typeof suffix === 'string' ? (\n                    <Text\n                      as=\"span\"\n                      sentiment=\"neutral\"\n                      variant=\"bodySmall\"\n                      disabled={disabled}\n                    >\n                      {suffix}\n                    </Text>\n                  ) : (\n                    suffix\n                  )}\n                </BasicSuffixStack>\n              ) : null}\n              {type === 'password' ? (\n                <CTASuffixStack direction=\"row\" alignItems=\"center\">\n                  <Button\n                    disabled={disabled}\n                    data-testid={\n                      dataTestId ? `${dataTestId}-visibility-button` : undefined\n                    }\n                    aria-label={isPasswordVisible ? 'hide' : 'show'}\n                    onClick={() => {\n                      setIsPasswordVisible(!isPasswordVisible)\n                    }}\n                    variant=\"ghost\"\n                    sentiment=\"neutral\"\n                    icon={isPasswordVisible ? 'eye-off' : 'eye'}\n                    size={size === 'small' ? 'xsmall' : 'small'}\n                  />\n                </CTASuffixStack>\n              ) : null}\n              {onRandomize ? (\n                <CTASuffixStack direction=\"row\" alignItems=\"center\">\n                  <Button\n                    disabled={disabled}\n                    icon=\"auto-fix\"\n                    size={size === 'small' ? 'xsmall' : 'small'}\n                    variant=\"ghost\"\n                    sentiment=\"neutral\"\n                    onClick={onRandomize}\n                  />\n                </CTASuffixStack>\n              ) : null}\n            </StyledInputWrapper>\n          </Tooltip>\n        </div>\n        {error || typeof success === 'string' || typeof helper === 'string' ? (\n          <Text\n            as=\"p\"\n            variant=\"caption\"\n            sentiment={sentiment}\n            prominence={!error && !success ? 'weak' : 'default'}\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)\n"]} */"));
|
|
39
|
+
}) => `0 ${theme.space["2"]}`, ";" + (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/TextInputV2/index.tsx"],"names":[],"mappings":"AA4CgC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextInputV2/index.tsx","sourcesContent":["'use client'\n\nimport styled from '@emotion/styled'\nimport { AlertCircleIcon, CheckCircleIcon } from '@ultraviolet/icons'\nimport type {\n  ChangeEvent,\n  ChangeEventHandler,\n  InputHTMLAttributes,\n  ReactNode,\n} from 'react'\nimport {\n  forwardRef,\n  useCallback,\n  useId,\n  useImperativeHandle,\n  useMemo,\n  useRef,\n  useState,\n} from 'react'\nimport { Button } from '../Button'\nimport { Label } from '../Label'\nimport { Loader } from '../Loader'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\nimport { Tooltip } from '../Tooltip'\n\n// SIZE\nexport const TEXTINPUT_SIZE_HEIGHT = {\n  small: '400', // sizing theme tokens key\n  medium: '500',\n  large: '600',\n} as const\ntype TextInputSize = keyof typeof TEXTINPUT_SIZE_HEIGHT\n\nexport const BasicPrefixStack = styled(Stack)`\n  padding: ${({ theme }) => theme.space['2']};\n\n  &[data-size=\"small\"] {\n    padding: ${({ theme }) => theme.space['1']};\n  }\n  border-right: 1px solid;\n  border-color: inherit;\n`\n\nconst StateStack = styled(Stack)`\n  padding: ${({ theme }) => `0 ${theme.space['2']}`};\n`\n\nexport const BasicSuffixStack = styled(Stack)`\n  padding: ${({ theme }) => `0 ${theme.space['2']}`};\n  border-left: 1px solid;\n  border-color: inherit;\n`\n\nconst CTASuffixStack = styled(Stack)`\n  padding: ${({ theme }) => `0 ${theme.space['1']}`};\n  border-left: 1px solid;\n  border-color: inherit;\n`\n\nexport const StyledInput = styled.input<{\n  'data-size': TextInputSize\n}>`\n  flex: 1;\n  border: none;\n  outline: none;\n  height: 100%;\n  width: 100%;\n  padding-left: ${({ theme }) => theme.space['2']};\n  background: transparent;\n  font-size: ${({ theme }) => theme.typography.bodySmall.fontSize};\n\n  &[data-size='large'] {\n    font-size: ${({ theme }) => theme.typography.body.fontSize};\n  }\n\n  &[data-size='small'] {\n    padding-left: ${({ theme }) => theme.space['1']};\n  }\n`\n\ntype StyledInputWrapperProps = {\n  hasFocus: boolean\n  size: TextInputSize\n}\nconst StyledInputWrapper = styled('div', {\n  shouldForwardProp: prop => !['hasFocus', 'size'].includes(prop),\n})<StyledInputWrapperProps>`\n  display: flex;\n  flex-direction: row;\n  height: ${({ size, theme }) => theme.sizing[TEXTINPUT_SIZE_HEIGHT[size]]};\n\n  background: ${({ theme }) => theme.colors.neutral.background};\n  border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  border-radius: ${({ theme }) => theme.radii.default};\n\n  & > ${StyledInput} {\n    color: ${({ theme }) => theme.colors.neutral.text};\n\n    &::placeholder {\n      color: ${({ theme }) => theme.colors.neutral.textWeak};\n    }\n  }\n\n  &[data-success='true'] {\n    border-color: ${({ theme }) => theme.colors.success.border};\n  }\n\n  &[data-error='true'] {\n    border-color: ${({ theme }) => theme.colors.danger.border};\n  }\n\n  &[data-readonly='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n    border-color: ${({ theme }) => theme.colors.neutral.border};\n  }\n\n  &[data-disabled='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    border-color: ${({ theme }) => theme.colors.neutral.borderDisabled};\n\n    & > ${StyledInput} {\n      color: ${({ theme }) => theme.colors.neutral.textDisabled};\n\n      &::placeholder {\n        color: ${({ theme }) => theme.colors.neutral.textWeakDisabled};\n      }\n    }\n  }\n\n  &:not([data-disabled='true']):not([data-readonly]):hover {\n    border-color: ${({ theme }) => theme.colors.primary.border};\n  }\n\n  ${({ theme, hasFocus }) =>\n    hasFocus\n      ? `\n  box-shadow: ${theme.shadows.focusPrimary};\n  border: 1px solid ${theme.colors.primary.border};\n`\n      : null};\n`\n\ntype TextInputProps = {\n  className?: string\n  clearable?: boolean\n  'data-testid'?: string\n  error?: string\n  helper?: ReactNode\n  label?: string\n  labelDescription?: ReactNode\n  loading?: boolean\n  minLength?: number\n  maxLength?: number\n  onRandomize?: () => void\n  prefix?: ReactNode\n  size?: TextInputSize\n  success?: string | boolean\n  suffix?: ReactNode\n  tooltip?: string\n  type?: 'text' | 'password' | 'url' | 'email'\n  value?: string\n  defaultValue?: string\n  onChangeValue?: (value: string) => void\n} & Pick<\n  InputHTMLAttributes<HTMLInputElement>,\n  | 'onFocus'\n  | 'onBlur'\n  | 'name'\n  | 'id'\n  | 'placeholder'\n  | 'aria-label'\n  | 'aria-labelledby'\n  | 'disabled'\n  | 'readOnly'\n  | 'required'\n  | 'autoFocus'\n  | 'tabIndex'\n  | 'autoComplete'\n  | 'onKeyDown'\n  | 'onKeyUp'\n  | 'role'\n  | 'aria-live'\n  | 'aria-atomic'\n  | 'onChange'\n>\n\n/**\n * This component offers an extended input HTML. The component can be controlled or uncontrolled.\n * To control the component, you need to pass the value and the `onChange` function.\n * If you don't pass the `onChange` function, the component will be uncontrolled and you can set the default value using `defaultValue`\n */\nexport const TextInputV2 = forwardRef<HTMLInputElement, TextInputProps>(\n  (\n    {\n      id,\n      className,\n      tabIndex,\n      value,\n      onChange,\n      onChangeValue,\n      placeholder,\n      disabled = false,\n      readOnly = false,\n      success,\n      error,\n      helper,\n      tooltip,\n      label,\n      autoFocus,\n      required = false,\n      'data-testid': dataTestId,\n      name,\n      onFocus,\n      onBlur,\n      clearable = false,\n      labelDescription,\n      type = 'text',\n      prefix,\n      suffix,\n      size = 'large',\n      loading,\n      onRandomize,\n      minLength,\n      maxLength,\n      'aria-labelledby': ariaLabelledBy,\n      'aria-label': ariaLabel,\n      autoComplete,\n      onKeyDown,\n      onKeyUp,\n      role,\n      'aria-live': ariaLive,\n      'aria-atomic': ariaAtomic,\n      defaultValue,\n    },\n    ref,\n  ) => {\n    const localId = useId()\n    const [hasFocus, setHasFocus] = useState(false)\n    const [localValue, setLocalValue] = useState(defaultValue)\n    const inputRef = useRef<HTMLInputElement>(null)\n    useImperativeHandle(ref, () => inputRef.current as HTMLInputElement)\n\n    const [isPasswordVisible, setIsPasswordVisible] = useState(false)\n    const computedType =\n      type === 'password' && isPasswordVisible ? 'text' : type\n\n    const sentiment = useMemo(() => {\n      if (error) {\n        return 'danger'\n      }\n\n      if (success) {\n        return 'success'\n      }\n\n      return 'neutral'\n    }, [error, success])\n\n    const onChangeCallback: ChangeEventHandler<HTMLInputElement> = useCallback(\n      event => {\n        onChange?.(event)\n        onChangeValue?.(event.target.value)\n        setLocalValue(event.target.value)\n      },\n      [onChange, onChangeValue],\n    )\n\n    const computedValue = value !== undefined ? value : localValue\n\n    const computedClearable = clearable && !!computedValue\n\n    return (\n      <Stack\n        gap={0.5}\n        className={className}\n        role={role}\n        aria-live={ariaLive}\n        aria-atomic={ariaAtomic}\n      >\n        {label || labelDescription ? (\n          <Label\n            labelDescription={labelDescription}\n            required={required}\n            size={size}\n            htmlFor={id ?? localId}\n            id={ariaLabelledBy}\n          >\n            {label}\n          </Label>\n        ) : null}\n        <div>\n          <Tooltip text={tooltip}>\n            <StyledInputWrapper\n              hasFocus={hasFocus}\n              data-disabled={disabled}\n              data-readonly={readOnly}\n              data-success={!!success}\n              data-error={!!error}\n              size={size}\n            >\n              {prefix ? (\n                <BasicPrefixStack\n                  direction=\"row\"\n                  alignItems=\"center\"\n                  data-size={size}\n                >\n                  {typeof prefix === 'string' ? (\n                    <Text\n                      as=\"span\"\n                      sentiment=\"neutral\"\n                      variant=\"bodySmall\"\n                      disabled={disabled}\n                    >\n                      {prefix}\n                    </Text>\n                  ) : (\n                    prefix\n                  )}\n                </BasicPrefixStack>\n              ) : null}\n              <StyledInput\n                type={computedType}\n                aria-invalid={!!error}\n                id={id ?? localId}\n                tabIndex={tabIndex}\n                autoFocus={autoFocus}\n                disabled={disabled}\n                ref={inputRef}\n                value={value}\n                defaultValue={defaultValue}\n                onChange={onChangeCallback}\n                data-size={size}\n                placeholder={placeholder}\n                data-testid={dataTestId}\n                name={name}\n                onFocus={event => {\n                  setHasFocus(true)\n                  onFocus?.(event)\n                }}\n                onBlur={event => {\n                  setHasFocus(false)\n                  onBlur?.(event)\n                }}\n                readOnly={readOnly}\n                minLength={minLength}\n                maxLength={maxLength}\n                aria-labelledby={ariaLabelledBy}\n                aria-label={ariaLabel}\n                autoComplete={autoComplete}\n                required={required}\n                onKeyDown={onKeyDown}\n                onKeyUp={onKeyUp}\n              />\n              {success || error || loading || computedClearable ? (\n                <StateStack direction=\"row\" gap={1} alignItems=\"center\">\n                  {computedClearable ? (\n                    <Button\n                      aria-label=\"clear value\"\n                      disabled={disabled || !computedValue}\n                      variant=\"ghost\"\n                      size={size === 'small' ? 'xsmall' : 'small'}\n                      icon=\"close\"\n                      onClick={() => {\n                        if (inputRef?.current) {\n                          inputRef.current.value = ''\n                          setLocalValue('')\n                          onChangeCallback({\n                            target: { value: '' },\n                            currentTarget: { value: '' },\n                          } as ChangeEvent<HTMLInputElement>)\n                        }\n                      }}\n                      sentiment=\"neutral\"\n                    />\n                  ) : null}\n                  {success ? (\n                    <CheckCircleIcon\n                      sentiment=\"success\"\n                      size=\"small\"\n                      disabled={disabled}\n                    />\n                  ) : null}\n                  {error ? (\n                    <AlertCircleIcon\n                      sentiment=\"danger\"\n                      size=\"small\"\n                      disabled={disabled}\n                    />\n                  ) : null}\n                  {loading && !disabled ? <Loader active size={16} /> : null}\n                </StateStack>\n              ) : null}\n              {suffix ? (\n                <BasicSuffixStack direction=\"row\" alignItems=\"center\">\n                  {typeof suffix === 'string' ? (\n                    <Text\n                      as=\"span\"\n                      sentiment=\"neutral\"\n                      variant=\"bodySmall\"\n                      disabled={disabled}\n                    >\n                      {suffix}\n                    </Text>\n                  ) : (\n                    suffix\n                  )}\n                </BasicSuffixStack>\n              ) : null}\n              {type === 'password' ? (\n                <CTASuffixStack direction=\"row\" alignItems=\"center\">\n                  <Button\n                    disabled={disabled}\n                    data-testid={\n                      dataTestId ? `${dataTestId}-visibility-button` : undefined\n                    }\n                    aria-label={isPasswordVisible ? 'hide' : 'show'}\n                    onClick={() => {\n                      setIsPasswordVisible(!isPasswordVisible)\n                    }}\n                    variant=\"ghost\"\n                    sentiment=\"neutral\"\n                    icon={isPasswordVisible ? 'eye-off' : 'eye'}\n                    size={size === 'small' ? 'xsmall' : 'small'}\n                  />\n                </CTASuffixStack>\n              ) : null}\n              {onRandomize ? (\n                <CTASuffixStack direction=\"row\" alignItems=\"center\">\n                  <Button\n                    disabled={disabled}\n                    icon=\"auto-fix\"\n                    size={size === 'small' ? 'xsmall' : 'small'}\n                    variant=\"ghost\"\n                    sentiment=\"neutral\"\n                    onClick={onRandomize}\n                  />\n                </CTASuffixStack>\n              ) : null}\n            </StyledInputWrapper>\n          </Tooltip>\n        </div>\n        {error || typeof success === 'string' || typeof helper === 'string' ? (\n          <Text\n            as=\"p\"\n            variant=\"caption\"\n            sentiment={sentiment}\n            prominence={!error && !success ? 'weak' : 'default'}\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)\n"]} */"));
|
|
40
40
|
const BasicSuffixStack = /* @__PURE__ */ _styled__default.default(index.Stack, process.env.NODE_ENV === "production" ? {
|
|
41
41
|
target: "e7tir8v3"
|
|
42
42
|
} : {
|
|
@@ -44,7 +44,7 @@ const BasicSuffixStack = /* @__PURE__ */ _styled__default.default(index.Stack, p
|
|
|
44
44
|
label: "BasicSuffixStack"
|
|
45
45
|
})("padding:", ({
|
|
46
46
|
theme
|
|
47
|
-
}) => `0 ${theme.space["2"]}`, ";border-left:1px solid;border-color:inherit;" + (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/TextInputV2/index.tsx"],"names":[],"mappings":"AAgD6C","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextInputV2/index.tsx","sourcesContent":["'use client'\n\nimport styled from '@emotion/styled'\nimport { AlertCircleIcon, CheckCircleIcon } from '@ultraviolet/icons'\nimport type {\n  ChangeEvent,\n  ChangeEventHandler,\n  InputHTMLAttributes,\n  ReactNode,\n} from 'react'\nimport {\n  forwardRef,\n  useCallback,\n  useId,\n  useImperativeHandle,\n  useMemo,\n  useRef,\n  useState,\n} from 'react'\nimport { Button } from '../Button'\nimport { Label } from '../Label'\nimport { Loader } from '../Loader'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\nimport { Tooltip } from '../Tooltip'\n\n// SIZE\nexport const TEXTINPUT_SIZE_HEIGHT = {\n  small: '400', // sizing theme tokens key\n  medium: '500',\n  large: '600',\n} as const\ntype TextInputSize = keyof typeof TEXTINPUT_SIZE_HEIGHT\n\nexport const BasicPrefixStack = styled(Stack)`\n  padding: ${({ theme }) => theme.space['2']};\n\n  &[data-size=\"small\"] {\n    padding: ${({ theme }) => theme.space['1']};\n  }\n  border-right: 1px solid;\n  border-color: inherit;\n`\n\nconst StateStack = styled(Stack)`\n  padding: ${({ theme }) => `0 ${theme.space['2']}`};\n`\n\nexport const BasicSuffixStack = styled(Stack)`\n  padding: ${({ theme }) => `0 ${theme.space['2']}`};\n  border-left: 1px solid;\n  border-color: inherit;\n`\n\nconst CTASuffixStack = styled(Stack)`\n  padding: ${({ theme }) => `0 ${theme.space['1']}`};\n  border-left: 1px solid;\n  border-color: inherit;\n`\n\nexport const StyledInput = styled.input<{\n  'data-size': TextInputSize\n}>`\n  flex: 1;\n  border: none;\n  outline: none;\n  height: 100%;\n  width: 100%;\n  padding-left: ${({ theme }) => theme.space['2']};\n  background: transparent;\n  font-size: ${({ theme }) => theme.typography.bodySmall.fontSize};\n\n  &[data-size='large'] {\n    font-size: ${({ theme }) => theme.typography.body.fontSize};\n  }\n\n  &[data-size='small'] {\n    padding-left: ${({ theme }) => theme.space['1']};\n  }\n`\n\ntype StyledInputWrapperProps = {\n  hasFocus: boolean\n  size: TextInputSize\n}\nconst StyledInputWrapper = styled('div', {\n  shouldForwardProp: prop => !['hasFocus', 'size'].includes(prop),\n})<StyledInputWrapperProps>`\n  display: flex;\n  flex-direction: row;\n  height: ${({ size, theme }) => theme.sizing[TEXTINPUT_SIZE_HEIGHT[size]]};\n\n  background: ${({ theme }) => theme.colors.neutral.background};\n  border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  border-radius: ${({ theme }) => theme.radii.default};\n\n  & > ${StyledInput} {\n    color: ${({ theme }) => theme.colors.neutral.text};\n\n    &::placeholder {\n      color: ${({ theme }) => theme.colors.neutral.textWeak};\n    }\n  }\n\n  &[data-success='true'] {\n    border-color: ${({ theme }) => theme.colors.success.border};\n  }\n\n  &[data-error='true'] {\n    border-color: ${({ theme }) => theme.colors.danger.border};\n  }\n\n  &[data-readonly='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n    border-color: ${({ theme }) => theme.colors.neutral.border};\n  }\n\n  &[data-disabled='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    border-color: ${({ theme }) => theme.colors.neutral.borderDisabled};\n\n    & > ${StyledInput} {\n      color: ${({ theme }) => theme.colors.neutral.textDisabled};\n\n      &::placeholder {\n        color: ${({ theme }) => theme.colors.neutral.textWeakDisabled};\n      }\n    }\n  }\n\n  &:not([data-disabled='true']):not([data-readonly]):hover {\n    border-color: ${({ theme }) => theme.colors.primary.border};\n  }\n\n  ${({ theme, hasFocus }) =>\n    hasFocus\n      ? `\n  box-shadow: ${theme.shadows.focusPrimary};\n  border: 1px solid ${theme.colors.primary.border};\n`\n      : null};\n`\n\ntype TextInputProps = {\n  className?: string\n  clearable?: boolean\n  'data-testid'?: string\n  error?: string\n  helper?: ReactNode\n  label?: string\n  labelDescription?: ReactNode\n  loading?: boolean\n  minLength?: number\n  maxLength?: number\n  onRandomize?: () => void\n  prefix?: ReactNode\n  size?: TextInputSize\n  success?: string | boolean\n  suffix?: ReactNode\n  tooltip?: string\n  type?: 'text' | 'password' | 'url' | 'email'\n  value?: string\n  defaultValue?: string\n  onChangeValue?: (value: string) => void\n} & Pick<\n  InputHTMLAttributes<HTMLInputElement>,\n  | 'onFocus'\n  | 'onBlur'\n  | 'name'\n  | 'id'\n  | 'placeholder'\n  | 'aria-label'\n  | 'aria-labelledby'\n  | 'disabled'\n  | 'readOnly'\n  | 'required'\n  | 'autoFocus'\n  | 'tabIndex'\n  | 'autoComplete'\n  | 'onKeyDown'\n  | 'onKeyUp'\n  | 'role'\n  | 'aria-live'\n  | 'aria-atomic'\n  | 'onChange'\n>\n\n/**\n * This component offers an extended input HTML. The component can be controlled or uncontrolled.\n * To control the component, you need to pass the value and the `onChange` function.\n * If you don't pass the `onChange` function, the component will be uncontrolled and you can set the default value using `defaultValue`\n */\nexport const TextInputV2 = forwardRef<HTMLInputElement, TextInputProps>(\n  (\n    {\n      id,\n      className,\n      tabIndex,\n      value,\n      onChange,\n      onChangeValue,\n      placeholder,\n      disabled = false,\n      readOnly = false,\n      success,\n      error,\n      helper,\n      tooltip,\n      label,\n      autoFocus,\n      required = false,\n      'data-testid': dataTestId,\n      name,\n      onFocus,\n      onBlur,\n      clearable = false,\n      labelDescription,\n      type = 'text',\n      prefix,\n      suffix,\n      size = 'large',\n      loading,\n      onRandomize,\n      minLength,\n      maxLength,\n      'aria-labelledby': ariaLabelledBy,\n      'aria-label': ariaLabel,\n      autoComplete,\n      onKeyDown,\n      onKeyUp,\n      role,\n      'aria-live': ariaLive,\n      'aria-atomic': ariaAtomic,\n      defaultValue,\n    },\n    ref,\n  ) => {\n    const localId = useId()\n    const [hasFocus, setHasFocus] = useState(false)\n    const inputRef = useRef<HTMLInputElement>(null)\n    useImperativeHandle(ref, () => inputRef.current as HTMLInputElement)\n\n    const [isPasswordVisible, setIsPasswordVisible] = useState(false)\n    const computedType =\n      type === 'password' && isPasswordVisible ? 'text' : type\n\n    const sentiment = useMemo(() => {\n      if (error) {\n        return 'danger'\n      }\n\n      if (success) {\n        return 'success'\n      }\n\n      return 'neutral'\n    }, [error, success])\n\n    const onChangeCallback: ChangeEventHandler<HTMLInputElement> = useCallback(\n      event => {\n        onChange?.(event)\n        onChangeValue?.(event.target.value)\n      },\n      [onChange, onChangeValue],\n    )\n\n    const computedClearable = clearable && !!value\n\n    return (\n      <Stack\n        gap={0.5}\n        className={className}\n        role={role}\n        aria-live={ariaLive}\n        aria-atomic={ariaAtomic}\n      >\n        {label || labelDescription ? (\n          <Label\n            labelDescription={labelDescription}\n            required={required}\n            size={size}\n            htmlFor={id ?? localId}\n            id={ariaLabelledBy}\n          >\n            {label}\n          </Label>\n        ) : null}\n        <div>\n          <Tooltip text={tooltip}>\n            <StyledInputWrapper\n              hasFocus={hasFocus}\n              data-disabled={disabled}\n              data-readonly={readOnly}\n              data-success={!!success}\n              data-error={!!error}\n              size={size}\n            >\n              {prefix ? (\n                <BasicPrefixStack\n                  direction=\"row\"\n                  alignItems=\"center\"\n                  data-size={size}\n                >\n                  {typeof prefix === 'string' ? (\n                    <Text\n                      as=\"span\"\n                      sentiment=\"neutral\"\n                      variant=\"bodySmall\"\n                      disabled={disabled}\n                    >\n                      {prefix}\n                    </Text>\n                  ) : (\n                    prefix\n                  )}\n                </BasicPrefixStack>\n              ) : null}\n              <StyledInput\n                type={computedType}\n                aria-invalid={!!error}\n                id={id ?? localId}\n                tabIndex={tabIndex}\n                autoFocus={autoFocus}\n                disabled={disabled}\n                ref={inputRef}\n                value={value}\n                defaultValue={defaultValue}\n                onChange={onChangeCallback}\n                data-size={size}\n                placeholder={placeholder}\n                data-testid={dataTestId}\n                name={name}\n                onFocus={event => {\n                  setHasFocus(true)\n                  onFocus?.(event)\n                }}\n                onBlur={event => {\n                  setHasFocus(false)\n                  onBlur?.(event)\n                }}\n                readOnly={readOnly}\n                minLength={minLength}\n                maxLength={maxLength}\n                aria-labelledby={ariaLabelledBy}\n                aria-label={ariaLabel}\n                autoComplete={autoComplete}\n                required={required}\n                onKeyDown={onKeyDown}\n                onKeyUp={onKeyUp}\n              />\n              {success || error || loading || computedClearable ? (\n                <StateStack direction=\"row\" gap={1} alignItems=\"center\">\n                  {computedClearable ? (\n                    <Button\n                      aria-label=\"clear value\"\n                      disabled={disabled || !value}\n                      variant=\"ghost\"\n                      size={size === 'small' ? 'xsmall' : 'small'}\n                      icon=\"close\"\n                      onClick={() => {\n                        if (inputRef?.current) {\n                          inputRef.current.value = ''\n                          onChangeCallback({\n                            target: { value: '' },\n                            currentTarget: { value: '' },\n                          } as ChangeEvent<HTMLInputElement>)\n                        }\n                      }}\n                      sentiment=\"neutral\"\n                    />\n                  ) : null}\n                  {success ? (\n                    <CheckCircleIcon\n                      sentiment=\"success\"\n                      size=\"small\"\n                      disabled={disabled}\n                    />\n                  ) : null}\n                  {error ? (\n                    <AlertCircleIcon\n                      sentiment=\"danger\"\n                      size=\"small\"\n                      disabled={disabled}\n                    />\n                  ) : null}\n                  {loading && !disabled ? <Loader active size={16} /> : null}\n                </StateStack>\n              ) : null}\n              {suffix ? (\n                <BasicSuffixStack direction=\"row\" alignItems=\"center\">\n                  {typeof suffix === 'string' ? (\n                    <Text\n                      as=\"span\"\n                      sentiment=\"neutral\"\n                      variant=\"bodySmall\"\n                      disabled={disabled}\n                    >\n                      {suffix}\n                    </Text>\n                  ) : (\n                    suffix\n                  )}\n                </BasicSuffixStack>\n              ) : null}\n              {type === 'password' ? (\n                <CTASuffixStack direction=\"row\" alignItems=\"center\">\n                  <Button\n                    disabled={disabled}\n                    data-testid={\n                      dataTestId ? `${dataTestId}-visibility-button` : undefined\n                    }\n                    aria-label={isPasswordVisible ? 'hide' : 'show'}\n                    onClick={() => {\n                      setIsPasswordVisible(!isPasswordVisible)\n                    }}\n                    variant=\"ghost\"\n                    sentiment=\"neutral\"\n                    icon={isPasswordVisible ? 'eye-off' : 'eye'}\n                    size={size === 'small' ? 'xsmall' : 'small'}\n                  />\n                </CTASuffixStack>\n              ) : null}\n              {onRandomize ? (\n                <CTASuffixStack direction=\"row\" alignItems=\"center\">\n                  <Button\n                    disabled={disabled}\n                    icon=\"auto-fix\"\n                    size={size === 'small' ? 'xsmall' : 'small'}\n                    variant=\"ghost\"\n                    sentiment=\"neutral\"\n                    onClick={onRandomize}\n                  />\n                </CTASuffixStack>\n              ) : null}\n            </StyledInputWrapper>\n          </Tooltip>\n        </div>\n        {error || typeof success === 'string' || typeof helper === 'string' ? (\n          <Text\n            as=\"p\"\n            variant=\"caption\"\n            sentiment={sentiment}\n            prominence={!error && !success ? 'weak' : 'default'}\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)\n"]} */"));
|
|
47
|
+
}) => `0 ${theme.space["2"]}`, ";border-left:1px solid;border-color:inherit;" + (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/TextInputV2/index.tsx"],"names":[],"mappings":"AAgD6C","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextInputV2/index.tsx","sourcesContent":["'use client'\n\nimport styled from '@emotion/styled'\nimport { AlertCircleIcon, CheckCircleIcon } from '@ultraviolet/icons'\nimport type {\n  ChangeEvent,\n  ChangeEventHandler,\n  InputHTMLAttributes,\n  ReactNode,\n} from 'react'\nimport {\n  forwardRef,\n  useCallback,\n  useId,\n  useImperativeHandle,\n  useMemo,\n  useRef,\n  useState,\n} from 'react'\nimport { Button } from '../Button'\nimport { Label } from '../Label'\nimport { Loader } from '../Loader'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\nimport { Tooltip } from '../Tooltip'\n\n// SIZE\nexport const TEXTINPUT_SIZE_HEIGHT = {\n  small: '400', // sizing theme tokens key\n  medium: '500',\n  large: '600',\n} as const\ntype TextInputSize = keyof typeof TEXTINPUT_SIZE_HEIGHT\n\nexport const BasicPrefixStack = styled(Stack)`\n  padding: ${({ theme }) => theme.space['2']};\n\n  &[data-size=\"small\"] {\n    padding: ${({ theme }) => theme.space['1']};\n  }\n  border-right: 1px solid;\n  border-color: inherit;\n`\n\nconst StateStack = styled(Stack)`\n  padding: ${({ theme }) => `0 ${theme.space['2']}`};\n`\n\nexport const BasicSuffixStack = styled(Stack)`\n  padding: ${({ theme }) => `0 ${theme.space['2']}`};\n  border-left: 1px solid;\n  border-color: inherit;\n`\n\nconst CTASuffixStack = styled(Stack)`\n  padding: ${({ theme }) => `0 ${theme.space['1']}`};\n  border-left: 1px solid;\n  border-color: inherit;\n`\n\nexport const StyledInput = styled.input<{\n  'data-size': TextInputSize\n}>`\n  flex: 1;\n  border: none;\n  outline: none;\n  height: 100%;\n  width: 100%;\n  padding-left: ${({ theme }) => theme.space['2']};\n  background: transparent;\n  font-size: ${({ theme }) => theme.typography.bodySmall.fontSize};\n\n  &[data-size='large'] {\n    font-size: ${({ theme }) => theme.typography.body.fontSize};\n  }\n\n  &[data-size='small'] {\n    padding-left: ${({ theme }) => theme.space['1']};\n  }\n`\n\ntype StyledInputWrapperProps = {\n  hasFocus: boolean\n  size: TextInputSize\n}\nconst StyledInputWrapper = styled('div', {\n  shouldForwardProp: prop => !['hasFocus', 'size'].includes(prop),\n})<StyledInputWrapperProps>`\n  display: flex;\n  flex-direction: row;\n  height: ${({ size, theme }) => theme.sizing[TEXTINPUT_SIZE_HEIGHT[size]]};\n\n  background: ${({ theme }) => theme.colors.neutral.background};\n  border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  border-radius: ${({ theme }) => theme.radii.default};\n\n  & > ${StyledInput} {\n    color: ${({ theme }) => theme.colors.neutral.text};\n\n    &::placeholder {\n      color: ${({ theme }) => theme.colors.neutral.textWeak};\n    }\n  }\n\n  &[data-success='true'] {\n    border-color: ${({ theme }) => theme.colors.success.border};\n  }\n\n  &[data-error='true'] {\n    border-color: ${({ theme }) => theme.colors.danger.border};\n  }\n\n  &[data-readonly='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n    border-color: ${({ theme }) => theme.colors.neutral.border};\n  }\n\n  &[data-disabled='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    border-color: ${({ theme }) => theme.colors.neutral.borderDisabled};\n\n    & > ${StyledInput} {\n      color: ${({ theme }) => theme.colors.neutral.textDisabled};\n\n      &::placeholder {\n        color: ${({ theme }) => theme.colors.neutral.textWeakDisabled};\n      }\n    }\n  }\n\n  &:not([data-disabled='true']):not([data-readonly]):hover {\n    border-color: ${({ theme }) => theme.colors.primary.border};\n  }\n\n  ${({ theme, hasFocus }) =>\n    hasFocus\n      ? `\n  box-shadow: ${theme.shadows.focusPrimary};\n  border: 1px solid ${theme.colors.primary.border};\n`\n      : null};\n`\n\ntype TextInputProps = {\n  className?: string\n  clearable?: boolean\n  'data-testid'?: string\n  error?: string\n  helper?: ReactNode\n  label?: string\n  labelDescription?: ReactNode\n  loading?: boolean\n  minLength?: number\n  maxLength?: number\n  onRandomize?: () => void\n  prefix?: ReactNode\n  size?: TextInputSize\n  success?: string | boolean\n  suffix?: ReactNode\n  tooltip?: string\n  type?: 'text' | 'password' | 'url' | 'email'\n  value?: string\n  defaultValue?: string\n  onChangeValue?: (value: string) => void\n} & Pick<\n  InputHTMLAttributes<HTMLInputElement>,\n  | 'onFocus'\n  | 'onBlur'\n  | 'name'\n  | 'id'\n  | 'placeholder'\n  | 'aria-label'\n  | 'aria-labelledby'\n  | 'disabled'\n  | 'readOnly'\n  | 'required'\n  | 'autoFocus'\n  | 'tabIndex'\n  | 'autoComplete'\n  | 'onKeyDown'\n  | 'onKeyUp'\n  | 'role'\n  | 'aria-live'\n  | 'aria-atomic'\n  | 'onChange'\n>\n\n/**\n * This component offers an extended input HTML. The component can be controlled or uncontrolled.\n * To control the component, you need to pass the value and the `onChange` function.\n * If you don't pass the `onChange` function, the component will be uncontrolled and you can set the default value using `defaultValue`\n */\nexport const TextInputV2 = forwardRef<HTMLInputElement, TextInputProps>(\n  (\n    {\n      id,\n      className,\n      tabIndex,\n      value,\n      onChange,\n      onChangeValue,\n      placeholder,\n      disabled = false,\n      readOnly = false,\n      success,\n      error,\n      helper,\n      tooltip,\n      label,\n      autoFocus,\n      required = false,\n      'data-testid': dataTestId,\n      name,\n      onFocus,\n      onBlur,\n      clearable = false,\n      labelDescription,\n      type = 'text',\n      prefix,\n      suffix,\n      size = 'large',\n      loading,\n      onRandomize,\n      minLength,\n      maxLength,\n      'aria-labelledby': ariaLabelledBy,\n      'aria-label': ariaLabel,\n      autoComplete,\n      onKeyDown,\n      onKeyUp,\n      role,\n      'aria-live': ariaLive,\n      'aria-atomic': ariaAtomic,\n      defaultValue,\n    },\n    ref,\n  ) => {\n    const localId = useId()\n    const [hasFocus, setHasFocus] = useState(false)\n    const [localValue, setLocalValue] = useState(defaultValue)\n    const inputRef = useRef<HTMLInputElement>(null)\n    useImperativeHandle(ref, () => inputRef.current as HTMLInputElement)\n\n    const [isPasswordVisible, setIsPasswordVisible] = useState(false)\n    const computedType =\n      type === 'password' && isPasswordVisible ? 'text' : type\n\n    const sentiment = useMemo(() => {\n      if (error) {\n        return 'danger'\n      }\n\n      if (success) {\n        return 'success'\n      }\n\n      return 'neutral'\n    }, [error, success])\n\n    const onChangeCallback: ChangeEventHandler<HTMLInputElement> = useCallback(\n      event => {\n        onChange?.(event)\n        onChangeValue?.(event.target.value)\n        setLocalValue(event.target.value)\n      },\n      [onChange, onChangeValue],\n    )\n\n    const computedValue = value !== undefined ? value : localValue\n\n    const computedClearable = clearable && !!computedValue\n\n    return (\n      <Stack\n        gap={0.5}\n        className={className}\n        role={role}\n        aria-live={ariaLive}\n        aria-atomic={ariaAtomic}\n      >\n        {label || labelDescription ? (\n          <Label\n            labelDescription={labelDescription}\n            required={required}\n            size={size}\n            htmlFor={id ?? localId}\n            id={ariaLabelledBy}\n          >\n            {label}\n          </Label>\n        ) : null}\n        <div>\n          <Tooltip text={tooltip}>\n            <StyledInputWrapper\n              hasFocus={hasFocus}\n              data-disabled={disabled}\n              data-readonly={readOnly}\n              data-success={!!success}\n              data-error={!!error}\n              size={size}\n            >\n              {prefix ? (\n                <BasicPrefixStack\n                  direction=\"row\"\n                  alignItems=\"center\"\n                  data-size={size}\n                >\n                  {typeof prefix === 'string' ? (\n                    <Text\n                      as=\"span\"\n                      sentiment=\"neutral\"\n                      variant=\"bodySmall\"\n                      disabled={disabled}\n                    >\n                      {prefix}\n                    </Text>\n                  ) : (\n                    prefix\n                  )}\n                </BasicPrefixStack>\n              ) : null}\n              <StyledInput\n                type={computedType}\n                aria-invalid={!!error}\n                id={id ?? localId}\n                tabIndex={tabIndex}\n                autoFocus={autoFocus}\n                disabled={disabled}\n                ref={inputRef}\n                value={value}\n                defaultValue={defaultValue}\n                onChange={onChangeCallback}\n                data-size={size}\n                placeholder={placeholder}\n                data-testid={dataTestId}\n                name={name}\n                onFocus={event => {\n                  setHasFocus(true)\n                  onFocus?.(event)\n                }}\n                onBlur={event => {\n                  setHasFocus(false)\n                  onBlur?.(event)\n                }}\n                readOnly={readOnly}\n                minLength={minLength}\n                maxLength={maxLength}\n                aria-labelledby={ariaLabelledBy}\n                aria-label={ariaLabel}\n                autoComplete={autoComplete}\n                required={required}\n                onKeyDown={onKeyDown}\n                onKeyUp={onKeyUp}\n              />\n              {success || error || loading || computedClearable ? (\n                <StateStack direction=\"row\" gap={1} alignItems=\"center\">\n                  {computedClearable ? (\n                    <Button\n                      aria-label=\"clear value\"\n                      disabled={disabled || !computedValue}\n                      variant=\"ghost\"\n                      size={size === 'small' ? 'xsmall' : 'small'}\n                      icon=\"close\"\n                      onClick={() => {\n                        if (inputRef?.current) {\n                          inputRef.current.value = ''\n                          setLocalValue('')\n                          onChangeCallback({\n                            target: { value: '' },\n                            currentTarget: { value: '' },\n                          } as ChangeEvent<HTMLInputElement>)\n                        }\n                      }}\n                      sentiment=\"neutral\"\n                    />\n                  ) : null}\n                  {success ? (\n                    <CheckCircleIcon\n                      sentiment=\"success\"\n                      size=\"small\"\n                      disabled={disabled}\n                    />\n                  ) : null}\n                  {error ? (\n                    <AlertCircleIcon\n                      sentiment=\"danger\"\n                      size=\"small\"\n                      disabled={disabled}\n                    />\n                  ) : null}\n                  {loading && !disabled ? <Loader active size={16} /> : null}\n                </StateStack>\n              ) : null}\n              {suffix ? (\n                <BasicSuffixStack direction=\"row\" alignItems=\"center\">\n                  {typeof suffix === 'string' ? (\n                    <Text\n                      as=\"span\"\n                      sentiment=\"neutral\"\n                      variant=\"bodySmall\"\n                      disabled={disabled}\n                    >\n                      {suffix}\n                    </Text>\n                  ) : (\n                    suffix\n                  )}\n                </BasicSuffixStack>\n              ) : null}\n              {type === 'password' ? (\n                <CTASuffixStack direction=\"row\" alignItems=\"center\">\n                  <Button\n                    disabled={disabled}\n                    data-testid={\n                      dataTestId ? `${dataTestId}-visibility-button` : undefined\n                    }\n                    aria-label={isPasswordVisible ? 'hide' : 'show'}\n                    onClick={() => {\n                      setIsPasswordVisible(!isPasswordVisible)\n                    }}\n                    variant=\"ghost\"\n                    sentiment=\"neutral\"\n                    icon={isPasswordVisible ? 'eye-off' : 'eye'}\n                    size={size === 'small' ? 'xsmall' : 'small'}\n                  />\n                </CTASuffixStack>\n              ) : null}\n              {onRandomize ? (\n                <CTASuffixStack direction=\"row\" alignItems=\"center\">\n                  <Button\n                    disabled={disabled}\n                    icon=\"auto-fix\"\n                    size={size === 'small' ? 'xsmall' : 'small'}\n                    variant=\"ghost\"\n                    sentiment=\"neutral\"\n                    onClick={onRandomize}\n                  />\n                </CTASuffixStack>\n              ) : null}\n            </StyledInputWrapper>\n          </Tooltip>\n        </div>\n        {error || typeof success === 'string' || typeof helper === 'string' ? (\n          <Text\n            as=\"p\"\n            variant=\"caption\"\n            sentiment={sentiment}\n            prominence={!error && !success ? 'weak' : 'default'}\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)\n"]} */"));
|
|
48
48
|
const CTASuffixStack = /* @__PURE__ */ _styled__default.default(index.Stack, process.env.NODE_ENV === "production" ? {
|
|
49
49
|
target: "e7tir8v2"
|
|
50
50
|
} : {
|
|
@@ -52,7 +52,7 @@ const CTASuffixStack = /* @__PURE__ */ _styled__default.default(index.Stack, pro
|
|
|
52
52
|
label: "CTASuffixStack"
|
|
53
53
|
})("padding:", ({
|
|
54
54
|
theme
|
|
55
|
-
}) => `0 ${theme.space["1"]}`, ";border-left:1px solid;border-color:inherit;" + (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/TextInputV2/index.tsx"],"names":[],"mappings":"AAsDoC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextInputV2/index.tsx","sourcesContent":["'use client'\n\nimport styled from '@emotion/styled'\nimport { AlertCircleIcon, CheckCircleIcon } from '@ultraviolet/icons'\nimport type {\n  ChangeEvent,\n  ChangeEventHandler,\n  InputHTMLAttributes,\n  ReactNode,\n} from 'react'\nimport {\n  forwardRef,\n  useCallback,\n  useId,\n  useImperativeHandle,\n  useMemo,\n  useRef,\n  useState,\n} from 'react'\nimport { Button } from '../Button'\nimport { Label } from '../Label'\nimport { Loader } from '../Loader'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\nimport { Tooltip } from '../Tooltip'\n\n// SIZE\nexport const TEXTINPUT_SIZE_HEIGHT = {\n  small: '400', // sizing theme tokens key\n  medium: '500',\n  large: '600',\n} as const\ntype TextInputSize = keyof typeof TEXTINPUT_SIZE_HEIGHT\n\nexport const BasicPrefixStack = styled(Stack)`\n  padding: ${({ theme }) => theme.space['2']};\n\n  &[data-size=\"small\"] {\n    padding: ${({ theme }) => theme.space['1']};\n  }\n  border-right: 1px solid;\n  border-color: inherit;\n`\n\nconst StateStack = styled(Stack)`\n  padding: ${({ theme }) => `0 ${theme.space['2']}`};\n`\n\nexport const BasicSuffixStack = styled(Stack)`\n  padding: ${({ theme }) => `0 ${theme.space['2']}`};\n  border-left: 1px solid;\n  border-color: inherit;\n`\n\nconst CTASuffixStack = styled(Stack)`\n  padding: ${({ theme }) => `0 ${theme.space['1']}`};\n  border-left: 1px solid;\n  border-color: inherit;\n`\n\nexport const StyledInput = styled.input<{\n  'data-size': TextInputSize\n}>`\n  flex: 1;\n  border: none;\n  outline: none;\n  height: 100%;\n  width: 100%;\n  padding-left: ${({ theme }) => theme.space['2']};\n  background: transparent;\n  font-size: ${({ theme }) => theme.typography.bodySmall.fontSize};\n\n  &[data-size='large'] {\n    font-size: ${({ theme }) => theme.typography.body.fontSize};\n  }\n\n  &[data-size='small'] {\n    padding-left: ${({ theme }) => theme.space['1']};\n  }\n`\n\ntype StyledInputWrapperProps = {\n  hasFocus: boolean\n  size: TextInputSize\n}\nconst StyledInputWrapper = styled('div', {\n  shouldForwardProp: prop => !['hasFocus', 'size'].includes(prop),\n})<StyledInputWrapperProps>`\n  display: flex;\n  flex-direction: row;\n  height: ${({ size, theme }) => theme.sizing[TEXTINPUT_SIZE_HEIGHT[size]]};\n\n  background: ${({ theme }) => theme.colors.neutral.background};\n  border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  border-radius: ${({ theme }) => theme.radii.default};\n\n  & > ${StyledInput} {\n    color: ${({ theme }) => theme.colors.neutral.text};\n\n    &::placeholder {\n      color: ${({ theme }) => theme.colors.neutral.textWeak};\n    }\n  }\n\n  &[data-success='true'] {\n    border-color: ${({ theme }) => theme.colors.success.border};\n  }\n\n  &[data-error='true'] {\n    border-color: ${({ theme }) => theme.colors.danger.border};\n  }\n\n  &[data-readonly='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n    border-color: ${({ theme }) => theme.colors.neutral.border};\n  }\n\n  &[data-disabled='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    border-color: ${({ theme }) => theme.colors.neutral.borderDisabled};\n\n    & > ${StyledInput} {\n      color: ${({ theme }) => theme.colors.neutral.textDisabled};\n\n      &::placeholder {\n        color: ${({ theme }) => theme.colors.neutral.textWeakDisabled};\n      }\n    }\n  }\n\n  &:not([data-disabled='true']):not([data-readonly]):hover {\n    border-color: ${({ theme }) => theme.colors.primary.border};\n  }\n\n  ${({ theme, hasFocus }) =>\n    hasFocus\n      ? `\n  box-shadow: ${theme.shadows.focusPrimary};\n  border: 1px solid ${theme.colors.primary.border};\n`\n      : null};\n`\n\ntype TextInputProps = {\n  className?: string\n  clearable?: boolean\n  'data-testid'?: string\n  error?: string\n  helper?: ReactNode\n  label?: string\n  labelDescription?: ReactNode\n  loading?: boolean\n  minLength?: number\n  maxLength?: number\n  onRandomize?: () => void\n  prefix?: ReactNode\n  size?: TextInputSize\n  success?: string | boolean\n  suffix?: ReactNode\n  tooltip?: string\n  type?: 'text' | 'password' | 'url' | 'email'\n  value?: string\n  defaultValue?: string\n  onChangeValue?: (value: string) => void\n} & Pick<\n  InputHTMLAttributes<HTMLInputElement>,\n  | 'onFocus'\n  | 'onBlur'\n  | 'name'\n  | 'id'\n  | 'placeholder'\n  | 'aria-label'\n  | 'aria-labelledby'\n  | 'disabled'\n  | 'readOnly'\n  | 'required'\n  | 'autoFocus'\n  | 'tabIndex'\n  | 'autoComplete'\n  | 'onKeyDown'\n  | 'onKeyUp'\n  | 'role'\n  | 'aria-live'\n  | 'aria-atomic'\n  | 'onChange'\n>\n\n/**\n * This component offers an extended input HTML. The component can be controlled or uncontrolled.\n * To control the component, you need to pass the value and the `onChange` function.\n * If you don't pass the `onChange` function, the component will be uncontrolled and you can set the default value using `defaultValue`\n */\nexport const TextInputV2 = forwardRef<HTMLInputElement, TextInputProps>(\n  (\n    {\n      id,\n      className,\n      tabIndex,\n      value,\n      onChange,\n      onChangeValue,\n      placeholder,\n      disabled = false,\n      readOnly = false,\n      success,\n      error,\n      helper,\n      tooltip,\n      label,\n      autoFocus,\n      required = false,\n      'data-testid': dataTestId,\n      name,\n      onFocus,\n      onBlur,\n      clearable = false,\n      labelDescription,\n      type = 'text',\n      prefix,\n      suffix,\n      size = 'large',\n      loading,\n      onRandomize,\n      minLength,\n      maxLength,\n      'aria-labelledby': ariaLabelledBy,\n      'aria-label': ariaLabel,\n      autoComplete,\n      onKeyDown,\n      onKeyUp,\n      role,\n      'aria-live': ariaLive,\n      'aria-atomic': ariaAtomic,\n      defaultValue,\n    },\n    ref,\n  ) => {\n    const localId = useId()\n    const [hasFocus, setHasFocus] = useState(false)\n    const inputRef = useRef<HTMLInputElement>(null)\n    useImperativeHandle(ref, () => inputRef.current as HTMLInputElement)\n\n    const [isPasswordVisible, setIsPasswordVisible] = useState(false)\n    const computedType =\n      type === 'password' && isPasswordVisible ? 'text' : type\n\n    const sentiment = useMemo(() => {\n      if (error) {\n        return 'danger'\n      }\n\n      if (success) {\n        return 'success'\n      }\n\n      return 'neutral'\n    }, [error, success])\n\n    const onChangeCallback: ChangeEventHandler<HTMLInputElement> = useCallback(\n      event => {\n        onChange?.(event)\n        onChangeValue?.(event.target.value)\n      },\n      [onChange, onChangeValue],\n    )\n\n    const computedClearable = clearable && !!value\n\n    return (\n      <Stack\n        gap={0.5}\n        className={className}\n        role={role}\n        aria-live={ariaLive}\n        aria-atomic={ariaAtomic}\n      >\n        {label || labelDescription ? (\n          <Label\n            labelDescription={labelDescription}\n            required={required}\n            size={size}\n            htmlFor={id ?? localId}\n            id={ariaLabelledBy}\n          >\n            {label}\n          </Label>\n        ) : null}\n        <div>\n          <Tooltip text={tooltip}>\n            <StyledInputWrapper\n              hasFocus={hasFocus}\n              data-disabled={disabled}\n              data-readonly={readOnly}\n              data-success={!!success}\n              data-error={!!error}\n              size={size}\n            >\n              {prefix ? (\n                <BasicPrefixStack\n                  direction=\"row\"\n                  alignItems=\"center\"\n                  data-size={size}\n                >\n                  {typeof prefix === 'string' ? (\n                    <Text\n                      as=\"span\"\n                      sentiment=\"neutral\"\n                      variant=\"bodySmall\"\n                      disabled={disabled}\n                    >\n                      {prefix}\n                    </Text>\n                  ) : (\n                    prefix\n                  )}\n                </BasicPrefixStack>\n              ) : null}\n              <StyledInput\n                type={computedType}\n                aria-invalid={!!error}\n                id={id ?? localId}\n                tabIndex={tabIndex}\n                autoFocus={autoFocus}\n                disabled={disabled}\n                ref={inputRef}\n                value={value}\n                defaultValue={defaultValue}\n                onChange={onChangeCallback}\n                data-size={size}\n                placeholder={placeholder}\n                data-testid={dataTestId}\n                name={name}\n                onFocus={event => {\n                  setHasFocus(true)\n                  onFocus?.(event)\n                }}\n                onBlur={event => {\n                  setHasFocus(false)\n                  onBlur?.(event)\n                }}\n                readOnly={readOnly}\n                minLength={minLength}\n                maxLength={maxLength}\n                aria-labelledby={ariaLabelledBy}\n                aria-label={ariaLabel}\n                autoComplete={autoComplete}\n                required={required}\n                onKeyDown={onKeyDown}\n                onKeyUp={onKeyUp}\n              />\n              {success || error || loading || computedClearable ? (\n                <StateStack direction=\"row\" gap={1} alignItems=\"center\">\n                  {computedClearable ? (\n                    <Button\n                      aria-label=\"clear value\"\n                      disabled={disabled || !value}\n                      variant=\"ghost\"\n                      size={size === 'small' ? 'xsmall' : 'small'}\n                      icon=\"close\"\n                      onClick={() => {\n                        if (inputRef?.current) {\n                          inputRef.current.value = ''\n                          onChangeCallback({\n                            target: { value: '' },\n                            currentTarget: { value: '' },\n                          } as ChangeEvent<HTMLInputElement>)\n                        }\n                      }}\n                      sentiment=\"neutral\"\n                    />\n                  ) : null}\n                  {success ? (\n                    <CheckCircleIcon\n                      sentiment=\"success\"\n                      size=\"small\"\n                      disabled={disabled}\n                    />\n                  ) : null}\n                  {error ? (\n                    <AlertCircleIcon\n                      sentiment=\"danger\"\n                      size=\"small\"\n                      disabled={disabled}\n                    />\n                  ) : null}\n                  {loading && !disabled ? <Loader active size={16} /> : null}\n                </StateStack>\n              ) : null}\n              {suffix ? (\n                <BasicSuffixStack direction=\"row\" alignItems=\"center\">\n                  {typeof suffix === 'string' ? (\n                    <Text\n                      as=\"span\"\n                      sentiment=\"neutral\"\n                      variant=\"bodySmall\"\n                      disabled={disabled}\n                    >\n                      {suffix}\n                    </Text>\n                  ) : (\n                    suffix\n                  )}\n                </BasicSuffixStack>\n              ) : null}\n              {type === 'password' ? (\n                <CTASuffixStack direction=\"row\" alignItems=\"center\">\n                  <Button\n                    disabled={disabled}\n                    data-testid={\n                      dataTestId ? `${dataTestId}-visibility-button` : undefined\n                    }\n                    aria-label={isPasswordVisible ? 'hide' : 'show'}\n                    onClick={() => {\n                      setIsPasswordVisible(!isPasswordVisible)\n                    }}\n                    variant=\"ghost\"\n                    sentiment=\"neutral\"\n                    icon={isPasswordVisible ? 'eye-off' : 'eye'}\n                    size={size === 'small' ? 'xsmall' : 'small'}\n                  />\n                </CTASuffixStack>\n              ) : null}\n              {onRandomize ? (\n                <CTASuffixStack direction=\"row\" alignItems=\"center\">\n                  <Button\n                    disabled={disabled}\n                    icon=\"auto-fix\"\n                    size={size === 'small' ? 'xsmall' : 'small'}\n                    variant=\"ghost\"\n                    sentiment=\"neutral\"\n                    onClick={onRandomize}\n                  />\n                </CTASuffixStack>\n              ) : null}\n            </StyledInputWrapper>\n          </Tooltip>\n        </div>\n        {error || typeof success === 'string' || typeof helper === 'string' ? (\n          <Text\n            as=\"p\"\n            variant=\"caption\"\n            sentiment={sentiment}\n            prominence={!error && !success ? 'weak' : 'default'}\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)\n"]} */"));
|
|
55
|
+
}) => `0 ${theme.space["1"]}`, ";border-left:1px solid;border-color:inherit;" + (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/TextInputV2/index.tsx"],"names":[],"mappings":"AAsDoC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextInputV2/index.tsx","sourcesContent":["'use client'\n\nimport styled from '@emotion/styled'\nimport { AlertCircleIcon, CheckCircleIcon } from '@ultraviolet/icons'\nimport type {\n  ChangeEvent,\n  ChangeEventHandler,\n  InputHTMLAttributes,\n  ReactNode,\n} from 'react'\nimport {\n  forwardRef,\n  useCallback,\n  useId,\n  useImperativeHandle,\n  useMemo,\n  useRef,\n  useState,\n} from 'react'\nimport { Button } from '../Button'\nimport { Label } from '../Label'\nimport { Loader } from '../Loader'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\nimport { Tooltip } from '../Tooltip'\n\n// SIZE\nexport const TEXTINPUT_SIZE_HEIGHT = {\n  small: '400', // sizing theme tokens key\n  medium: '500',\n  large: '600',\n} as const\ntype TextInputSize = keyof typeof TEXTINPUT_SIZE_HEIGHT\n\nexport const BasicPrefixStack = styled(Stack)`\n  padding: ${({ theme }) => theme.space['2']};\n\n  &[data-size=\"small\"] {\n    padding: ${({ theme }) => theme.space['1']};\n  }\n  border-right: 1px solid;\n  border-color: inherit;\n`\n\nconst StateStack = styled(Stack)`\n  padding: ${({ theme }) => `0 ${theme.space['2']}`};\n`\n\nexport const BasicSuffixStack = styled(Stack)`\n  padding: ${({ theme }) => `0 ${theme.space['2']}`};\n  border-left: 1px solid;\n  border-color: inherit;\n`\n\nconst CTASuffixStack = styled(Stack)`\n  padding: ${({ theme }) => `0 ${theme.space['1']}`};\n  border-left: 1px solid;\n  border-color: inherit;\n`\n\nexport const StyledInput = styled.input<{\n  'data-size': TextInputSize\n}>`\n  flex: 1;\n  border: none;\n  outline: none;\n  height: 100%;\n  width: 100%;\n  padding-left: ${({ theme }) => theme.space['2']};\n  background: transparent;\n  font-size: ${({ theme }) => theme.typography.bodySmall.fontSize};\n\n  &[data-size='large'] {\n    font-size: ${({ theme }) => theme.typography.body.fontSize};\n  }\n\n  &[data-size='small'] {\n    padding-left: ${({ theme }) => theme.space['1']};\n  }\n`\n\ntype StyledInputWrapperProps = {\n  hasFocus: boolean\n  size: TextInputSize\n}\nconst StyledInputWrapper = styled('div', {\n  shouldForwardProp: prop => !['hasFocus', 'size'].includes(prop),\n})<StyledInputWrapperProps>`\n  display: flex;\n  flex-direction: row;\n  height: ${({ size, theme }) => theme.sizing[TEXTINPUT_SIZE_HEIGHT[size]]};\n\n  background: ${({ theme }) => theme.colors.neutral.background};\n  border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  border-radius: ${({ theme }) => theme.radii.default};\n\n  & > ${StyledInput} {\n    color: ${({ theme }) => theme.colors.neutral.text};\n\n    &::placeholder {\n      color: ${({ theme }) => theme.colors.neutral.textWeak};\n    }\n  }\n\n  &[data-success='true'] {\n    border-color: ${({ theme }) => theme.colors.success.border};\n  }\n\n  &[data-error='true'] {\n    border-color: ${({ theme }) => theme.colors.danger.border};\n  }\n\n  &[data-readonly='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n    border-color: ${({ theme }) => theme.colors.neutral.border};\n  }\n\n  &[data-disabled='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    border-color: ${({ theme }) => theme.colors.neutral.borderDisabled};\n\n    & > ${StyledInput} {\n      color: ${({ theme }) => theme.colors.neutral.textDisabled};\n\n      &::placeholder {\n        color: ${({ theme }) => theme.colors.neutral.textWeakDisabled};\n      }\n    }\n  }\n\n  &:not([data-disabled='true']):not([data-readonly]):hover {\n    border-color: ${({ theme }) => theme.colors.primary.border};\n  }\n\n  ${({ theme, hasFocus }) =>\n    hasFocus\n      ? `\n  box-shadow: ${theme.shadows.focusPrimary};\n  border: 1px solid ${theme.colors.primary.border};\n`\n      : null};\n`\n\ntype TextInputProps = {\n  className?: string\n  clearable?: boolean\n  'data-testid'?: string\n  error?: string\n  helper?: ReactNode\n  label?: string\n  labelDescription?: ReactNode\n  loading?: boolean\n  minLength?: number\n  maxLength?: number\n  onRandomize?: () => void\n  prefix?: ReactNode\n  size?: TextInputSize\n  success?: string | boolean\n  suffix?: ReactNode\n  tooltip?: string\n  type?: 'text' | 'password' | 'url' | 'email'\n  value?: string\n  defaultValue?: string\n  onChangeValue?: (value: string) => void\n} & Pick<\n  InputHTMLAttributes<HTMLInputElement>,\n  | 'onFocus'\n  | 'onBlur'\n  | 'name'\n  | 'id'\n  | 'placeholder'\n  | 'aria-label'\n  | 'aria-labelledby'\n  | 'disabled'\n  | 'readOnly'\n  | 'required'\n  | 'autoFocus'\n  | 'tabIndex'\n  | 'autoComplete'\n  | 'onKeyDown'\n  | 'onKeyUp'\n  | 'role'\n  | 'aria-live'\n  | 'aria-atomic'\n  | 'onChange'\n>\n\n/**\n * This component offers an extended input HTML. The component can be controlled or uncontrolled.\n * To control the component, you need to pass the value and the `onChange` function.\n * If you don't pass the `onChange` function, the component will be uncontrolled and you can set the default value using `defaultValue`\n */\nexport const TextInputV2 = forwardRef<HTMLInputElement, TextInputProps>(\n  (\n    {\n      id,\n      className,\n      tabIndex,\n      value,\n      onChange,\n      onChangeValue,\n      placeholder,\n      disabled = false,\n      readOnly = false,\n      success,\n      error,\n      helper,\n      tooltip,\n      label,\n      autoFocus,\n      required = false,\n      'data-testid': dataTestId,\n      name,\n      onFocus,\n      onBlur,\n      clearable = false,\n      labelDescription,\n      type = 'text',\n      prefix,\n      suffix,\n      size = 'large',\n      loading,\n      onRandomize,\n      minLength,\n      maxLength,\n      'aria-labelledby': ariaLabelledBy,\n      'aria-label': ariaLabel,\n      autoComplete,\n      onKeyDown,\n      onKeyUp,\n      role,\n      'aria-live': ariaLive,\n      'aria-atomic': ariaAtomic,\n      defaultValue,\n    },\n    ref,\n  ) => {\n    const localId = useId()\n    const [hasFocus, setHasFocus] = useState(false)\n    const [localValue, setLocalValue] = useState(defaultValue)\n    const inputRef = useRef<HTMLInputElement>(null)\n    useImperativeHandle(ref, () => inputRef.current as HTMLInputElement)\n\n    const [isPasswordVisible, setIsPasswordVisible] = useState(false)\n    const computedType =\n      type === 'password' && isPasswordVisible ? 'text' : type\n\n    const sentiment = useMemo(() => {\n      if (error) {\n        return 'danger'\n      }\n\n      if (success) {\n        return 'success'\n      }\n\n      return 'neutral'\n    }, [error, success])\n\n    const onChangeCallback: ChangeEventHandler<HTMLInputElement> = useCallback(\n      event => {\n        onChange?.(event)\n        onChangeValue?.(event.target.value)\n        setLocalValue(event.target.value)\n      },\n      [onChange, onChangeValue],\n    )\n\n    const computedValue = value !== undefined ? value : localValue\n\n    const computedClearable = clearable && !!computedValue\n\n    return (\n      <Stack\n        gap={0.5}\n        className={className}\n        role={role}\n        aria-live={ariaLive}\n        aria-atomic={ariaAtomic}\n      >\n        {label || labelDescription ? (\n          <Label\n            labelDescription={labelDescription}\n            required={required}\n            size={size}\n            htmlFor={id ?? localId}\n            id={ariaLabelledBy}\n          >\n            {label}\n          </Label>\n        ) : null}\n        <div>\n          <Tooltip text={tooltip}>\n            <StyledInputWrapper\n              hasFocus={hasFocus}\n              data-disabled={disabled}\n              data-readonly={readOnly}\n              data-success={!!success}\n              data-error={!!error}\n              size={size}\n            >\n              {prefix ? (\n                <BasicPrefixStack\n                  direction=\"row\"\n                  alignItems=\"center\"\n                  data-size={size}\n                >\n                  {typeof prefix === 'string' ? (\n                    <Text\n                      as=\"span\"\n                      sentiment=\"neutral\"\n                      variant=\"bodySmall\"\n                      disabled={disabled}\n                    >\n                      {prefix}\n                    </Text>\n                  ) : (\n                    prefix\n                  )}\n                </BasicPrefixStack>\n              ) : null}\n              <StyledInput\n                type={computedType}\n                aria-invalid={!!error}\n                id={id ?? localId}\n                tabIndex={tabIndex}\n                autoFocus={autoFocus}\n                disabled={disabled}\n                ref={inputRef}\n                value={value}\n                defaultValue={defaultValue}\n                onChange={onChangeCallback}\n                data-size={size}\n                placeholder={placeholder}\n                data-testid={dataTestId}\n                name={name}\n                onFocus={event => {\n                  setHasFocus(true)\n                  onFocus?.(event)\n                }}\n                onBlur={event => {\n                  setHasFocus(false)\n                  onBlur?.(event)\n                }}\n                readOnly={readOnly}\n                minLength={minLength}\n                maxLength={maxLength}\n                aria-labelledby={ariaLabelledBy}\n                aria-label={ariaLabel}\n                autoComplete={autoComplete}\n                required={required}\n                onKeyDown={onKeyDown}\n                onKeyUp={onKeyUp}\n              />\n              {success || error || loading || computedClearable ? (\n                <StateStack direction=\"row\" gap={1} alignItems=\"center\">\n                  {computedClearable ? (\n                    <Button\n                      aria-label=\"clear value\"\n                      disabled={disabled || !computedValue}\n                      variant=\"ghost\"\n                      size={size === 'small' ? 'xsmall' : 'small'}\n                      icon=\"close\"\n                      onClick={() => {\n                        if (inputRef?.current) {\n                          inputRef.current.value = ''\n                          setLocalValue('')\n                          onChangeCallback({\n                            target: { value: '' },\n                            currentTarget: { value: '' },\n                          } as ChangeEvent<HTMLInputElement>)\n                        }\n                      }}\n                      sentiment=\"neutral\"\n                    />\n                  ) : null}\n                  {success ? (\n                    <CheckCircleIcon\n                      sentiment=\"success\"\n                      size=\"small\"\n                      disabled={disabled}\n                    />\n                  ) : null}\n                  {error ? (\n                    <AlertCircleIcon\n                      sentiment=\"danger\"\n                      size=\"small\"\n                      disabled={disabled}\n                    />\n                  ) : null}\n                  {loading && !disabled ? <Loader active size={16} /> : null}\n                </StateStack>\n              ) : null}\n              {suffix ? (\n                <BasicSuffixStack direction=\"row\" alignItems=\"center\">\n                  {typeof suffix === 'string' ? (\n                    <Text\n                      as=\"span\"\n                      sentiment=\"neutral\"\n                      variant=\"bodySmall\"\n                      disabled={disabled}\n                    >\n                      {suffix}\n                    </Text>\n                  ) : (\n                    suffix\n                  )}\n                </BasicSuffixStack>\n              ) : null}\n              {type === 'password' ? (\n                <CTASuffixStack direction=\"row\" alignItems=\"center\">\n                  <Button\n                    disabled={disabled}\n                    data-testid={\n                      dataTestId ? `${dataTestId}-visibility-button` : undefined\n                    }\n                    aria-label={isPasswordVisible ? 'hide' : 'show'}\n                    onClick={() => {\n                      setIsPasswordVisible(!isPasswordVisible)\n                    }}\n                    variant=\"ghost\"\n                    sentiment=\"neutral\"\n                    icon={isPasswordVisible ? 'eye-off' : 'eye'}\n                    size={size === 'small' ? 'xsmall' : 'small'}\n                  />\n                </CTASuffixStack>\n              ) : null}\n              {onRandomize ? (\n                <CTASuffixStack direction=\"row\" alignItems=\"center\">\n                  <Button\n                    disabled={disabled}\n                    icon=\"auto-fix\"\n                    size={size === 'small' ? 'xsmall' : 'small'}\n                    variant=\"ghost\"\n                    sentiment=\"neutral\"\n                    onClick={onRandomize}\n                  />\n                </CTASuffixStack>\n              ) : null}\n            </StyledInputWrapper>\n          </Tooltip>\n        </div>\n        {error || typeof success === 'string' || typeof helper === 'string' ? (\n          <Text\n            as=\"p\"\n            variant=\"caption\"\n            sentiment={sentiment}\n            prominence={!error && !success ? 'weak' : 'default'}\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)\n"]} */"));
|
|
56
56
|
const StyledInput = /* @__PURE__ */ _styled__default.default("input", process.env.NODE_ENV === "production" ? {
|
|
57
57
|
target: "e7tir8v1"
|
|
58
58
|
} : {
|
|
@@ -66,7 +66,7 @@ const StyledInput = /* @__PURE__ */ _styled__default.default("input", process.en
|
|
|
66
66
|
theme
|
|
67
67
|
}) => theme.typography.body.fontSize, ";}&[data-size='small']{padding-left:", ({
|
|
68
68
|
theme
|
|
69
|
-
}) => theme.space["1"], ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextInputV2/index.tsx"],"names":[],"mappings":"AA8DE","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextInputV2/index.tsx","sourcesContent":["'use client'\n\nimport styled from '@emotion/styled'\nimport { AlertCircleIcon, CheckCircleIcon } from '@ultraviolet/icons'\nimport type {\n  ChangeEvent,\n  ChangeEventHandler,\n  InputHTMLAttributes,\n  ReactNode,\n} from 'react'\nimport {\n  forwardRef,\n  useCallback,\n  useId,\n  useImperativeHandle,\n  useMemo,\n  useRef,\n  useState,\n} from 'react'\nimport { Button } from '../Button'\nimport { Label } from '../Label'\nimport { Loader } from '../Loader'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\nimport { Tooltip } from '../Tooltip'\n\n// SIZE\nexport const TEXTINPUT_SIZE_HEIGHT = {\n  small: '400', // sizing theme tokens key\n  medium: '500',\n  large: '600',\n} as const\ntype TextInputSize = keyof typeof TEXTINPUT_SIZE_HEIGHT\n\nexport const BasicPrefixStack = styled(Stack)`\n  padding: ${({ theme }) => theme.space['2']};\n\n  &[data-size=\"small\"] {\n    padding: ${({ theme }) => theme.space['1']};\n  }\n  border-right: 1px solid;\n  border-color: inherit;\n`\n\nconst StateStack = styled(Stack)`\n  padding: ${({ theme }) => `0 ${theme.space['2']}`};\n`\n\nexport const BasicSuffixStack = styled(Stack)`\n  padding: ${({ theme }) => `0 ${theme.space['2']}`};\n  border-left: 1px solid;\n  border-color: inherit;\n`\n\nconst CTASuffixStack = styled(Stack)`\n  padding: ${({ theme }) => `0 ${theme.space['1']}`};\n  border-left: 1px solid;\n  border-color: inherit;\n`\n\nexport const StyledInput = styled.input<{\n  'data-size': TextInputSize\n}>`\n  flex: 1;\n  border: none;\n  outline: none;\n  height: 100%;\n  width: 100%;\n  padding-left: ${({ theme }) => theme.space['2']};\n  background: transparent;\n  font-size: ${({ theme }) => theme.typography.bodySmall.fontSize};\n\n  &[data-size='large'] {\n    font-size: ${({ theme }) => theme.typography.body.fontSize};\n  }\n\n  &[data-size='small'] {\n    padding-left: ${({ theme }) => theme.space['1']};\n  }\n`\n\ntype StyledInputWrapperProps = {\n  hasFocus: boolean\n  size: TextInputSize\n}\nconst StyledInputWrapper = styled('div', {\n  shouldForwardProp: prop => !['hasFocus', 'size'].includes(prop),\n})<StyledInputWrapperProps>`\n  display: flex;\n  flex-direction: row;\n  height: ${({ size, theme }) => theme.sizing[TEXTINPUT_SIZE_HEIGHT[size]]};\n\n  background: ${({ theme }) => theme.colors.neutral.background};\n  border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  border-radius: ${({ theme }) => theme.radii.default};\n\n  & > ${StyledInput} {\n    color: ${({ theme }) => theme.colors.neutral.text};\n\n    &::placeholder {\n      color: ${({ theme }) => theme.colors.neutral.textWeak};\n    }\n  }\n\n  &[data-success='true'] {\n    border-color: ${({ theme }) => theme.colors.success.border};\n  }\n\n  &[data-error='true'] {\n    border-color: ${({ theme }) => theme.colors.danger.border};\n  }\n\n  &[data-readonly='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n    border-color: ${({ theme }) => theme.colors.neutral.border};\n  }\n\n  &[data-disabled='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    border-color: ${({ theme }) => theme.colors.neutral.borderDisabled};\n\n    & > ${StyledInput} {\n      color: ${({ theme }) => theme.colors.neutral.textDisabled};\n\n      &::placeholder {\n        color: ${({ theme }) => theme.colors.neutral.textWeakDisabled};\n      }\n    }\n  }\n\n  &:not([data-disabled='true']):not([data-readonly]):hover {\n    border-color: ${({ theme }) => theme.colors.primary.border};\n  }\n\n  ${({ theme, hasFocus }) =>\n    hasFocus\n      ? `\n  box-shadow: ${theme.shadows.focusPrimary};\n  border: 1px solid ${theme.colors.primary.border};\n`\n      : null};\n`\n\ntype TextInputProps = {\n  className?: string\n  clearable?: boolean\n  'data-testid'?: string\n  error?: string\n  helper?: ReactNode\n  label?: string\n  labelDescription?: ReactNode\n  loading?: boolean\n  minLength?: number\n  maxLength?: number\n  onRandomize?: () => void\n  prefix?: ReactNode\n  size?: TextInputSize\n  success?: string | boolean\n  suffix?: ReactNode\n  tooltip?: string\n  type?: 'text' | 'password' | 'url' | 'email'\n  value?: string\n  defaultValue?: string\n  onChangeValue?: (value: string) => void\n} & Pick<\n  InputHTMLAttributes<HTMLInputElement>,\n  | 'onFocus'\n  | 'onBlur'\n  | 'name'\n  | 'id'\n  | 'placeholder'\n  | 'aria-label'\n  | 'aria-labelledby'\n  | 'disabled'\n  | 'readOnly'\n  | 'required'\n  | 'autoFocus'\n  | 'tabIndex'\n  | 'autoComplete'\n  | 'onKeyDown'\n  | 'onKeyUp'\n  | 'role'\n  | 'aria-live'\n  | 'aria-atomic'\n  | 'onChange'\n>\n\n/**\n * This component offers an extended input HTML. The component can be controlled or uncontrolled.\n * To control the component, you need to pass the value and the `onChange` function.\n * If you don't pass the `onChange` function, the component will be uncontrolled and you can set the default value using `defaultValue`\n */\nexport const TextInputV2 = forwardRef<HTMLInputElement, TextInputProps>(\n  (\n    {\n      id,\n      className,\n      tabIndex,\n      value,\n      onChange,\n      onChangeValue,\n      placeholder,\n      disabled = false,\n      readOnly = false,\n      success,\n      error,\n      helper,\n      tooltip,\n      label,\n      autoFocus,\n      required = false,\n      'data-testid': dataTestId,\n      name,\n      onFocus,\n      onBlur,\n      clearable = false,\n      labelDescription,\n      type = 'text',\n      prefix,\n      suffix,\n      size = 'large',\n      loading,\n      onRandomize,\n      minLength,\n      maxLength,\n      'aria-labelledby': ariaLabelledBy,\n      'aria-label': ariaLabel,\n      autoComplete,\n      onKeyDown,\n      onKeyUp,\n      role,\n      'aria-live': ariaLive,\n      'aria-atomic': ariaAtomic,\n      defaultValue,\n    },\n    ref,\n  ) => {\n    const localId = useId()\n    const [hasFocus, setHasFocus] = useState(false)\n    const inputRef = useRef<HTMLInputElement>(null)\n    useImperativeHandle(ref, () => inputRef.current as HTMLInputElement)\n\n    const [isPasswordVisible, setIsPasswordVisible] = useState(false)\n    const computedType =\n      type === 'password' && isPasswordVisible ? 'text' : type\n\n    const sentiment = useMemo(() => {\n      if (error) {\n        return 'danger'\n      }\n\n      if (success) {\n        return 'success'\n      }\n\n      return 'neutral'\n    }, [error, success])\n\n    const onChangeCallback: ChangeEventHandler<HTMLInputElement> = useCallback(\n      event => {\n        onChange?.(event)\n        onChangeValue?.(event.target.value)\n      },\n      [onChange, onChangeValue],\n    )\n\n    const computedClearable = clearable && !!value\n\n    return (\n      <Stack\n        gap={0.5}\n        className={className}\n        role={role}\n        aria-live={ariaLive}\n        aria-atomic={ariaAtomic}\n      >\n        {label || labelDescription ? (\n          <Label\n            labelDescription={labelDescription}\n            required={required}\n            size={size}\n            htmlFor={id ?? localId}\n            id={ariaLabelledBy}\n          >\n            {label}\n          </Label>\n        ) : null}\n        <div>\n          <Tooltip text={tooltip}>\n            <StyledInputWrapper\n              hasFocus={hasFocus}\n              data-disabled={disabled}\n              data-readonly={readOnly}\n              data-success={!!success}\n              data-error={!!error}\n              size={size}\n            >\n              {prefix ? (\n                <BasicPrefixStack\n                  direction=\"row\"\n                  alignItems=\"center\"\n                  data-size={size}\n                >\n                  {typeof prefix === 'string' ? (\n                    <Text\n                      as=\"span\"\n                      sentiment=\"neutral\"\n                      variant=\"bodySmall\"\n                      disabled={disabled}\n                    >\n                      {prefix}\n                    </Text>\n                  ) : (\n                    prefix\n                  )}\n                </BasicPrefixStack>\n              ) : null}\n              <StyledInput\n                type={computedType}\n                aria-invalid={!!error}\n                id={id ?? localId}\n                tabIndex={tabIndex}\n                autoFocus={autoFocus}\n                disabled={disabled}\n                ref={inputRef}\n                value={value}\n                defaultValue={defaultValue}\n                onChange={onChangeCallback}\n                data-size={size}\n                placeholder={placeholder}\n                data-testid={dataTestId}\n                name={name}\n                onFocus={event => {\n                  setHasFocus(true)\n                  onFocus?.(event)\n                }}\n                onBlur={event => {\n                  setHasFocus(false)\n                  onBlur?.(event)\n                }}\n                readOnly={readOnly}\n                minLength={minLength}\n                maxLength={maxLength}\n                aria-labelledby={ariaLabelledBy}\n                aria-label={ariaLabel}\n                autoComplete={autoComplete}\n                required={required}\n                onKeyDown={onKeyDown}\n                onKeyUp={onKeyUp}\n              />\n              {success || error || loading || computedClearable ? (\n                <StateStack direction=\"row\" gap={1} alignItems=\"center\">\n                  {computedClearable ? (\n                    <Button\n                      aria-label=\"clear value\"\n                      disabled={disabled || !value}\n                      variant=\"ghost\"\n                      size={size === 'small' ? 'xsmall' : 'small'}\n                      icon=\"close\"\n                      onClick={() => {\n                        if (inputRef?.current) {\n                          inputRef.current.value = ''\n                          onChangeCallback({\n                            target: { value: '' },\n                            currentTarget: { value: '' },\n                          } as ChangeEvent<HTMLInputElement>)\n                        }\n                      }}\n                      sentiment=\"neutral\"\n                    />\n                  ) : null}\n                  {success ? (\n                    <CheckCircleIcon\n                      sentiment=\"success\"\n                      size=\"small\"\n                      disabled={disabled}\n                    />\n                  ) : null}\n                  {error ? (\n                    <AlertCircleIcon\n                      sentiment=\"danger\"\n                      size=\"small\"\n                      disabled={disabled}\n                    />\n                  ) : null}\n                  {loading && !disabled ? <Loader active size={16} /> : null}\n                </StateStack>\n              ) : null}\n              {suffix ? (\n                <BasicSuffixStack direction=\"row\" alignItems=\"center\">\n                  {typeof suffix === 'string' ? (\n                    <Text\n                      as=\"span\"\n                      sentiment=\"neutral\"\n                      variant=\"bodySmall\"\n                      disabled={disabled}\n                    >\n                      {suffix}\n                    </Text>\n                  ) : (\n                    suffix\n                  )}\n                </BasicSuffixStack>\n              ) : null}\n              {type === 'password' ? (\n                <CTASuffixStack direction=\"row\" alignItems=\"center\">\n                  <Button\n                    disabled={disabled}\n                    data-testid={\n                      dataTestId ? `${dataTestId}-visibility-button` : undefined\n                    }\n                    aria-label={isPasswordVisible ? 'hide' : 'show'}\n                    onClick={() => {\n                      setIsPasswordVisible(!isPasswordVisible)\n                    }}\n                    variant=\"ghost\"\n                    sentiment=\"neutral\"\n                    icon={isPasswordVisible ? 'eye-off' : 'eye'}\n                    size={size === 'small' ? 'xsmall' : 'small'}\n                  />\n                </CTASuffixStack>\n              ) : null}\n              {onRandomize ? (\n                <CTASuffixStack direction=\"row\" alignItems=\"center\">\n                  <Button\n                    disabled={disabled}\n                    icon=\"auto-fix\"\n                    size={size === 'small' ? 'xsmall' : 'small'}\n                    variant=\"ghost\"\n                    sentiment=\"neutral\"\n                    onClick={onRandomize}\n                  />\n                </CTASuffixStack>\n              ) : null}\n            </StyledInputWrapper>\n          </Tooltip>\n        </div>\n        {error || typeof success === 'string' || typeof helper === 'string' ? (\n          <Text\n            as=\"p\"\n            variant=\"caption\"\n            sentiment={sentiment}\n            prominence={!error && !success ? 'weak' : 'default'}\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)\n"]} */"));
|
|
69
|
+
}) => theme.space["1"], ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextInputV2/index.tsx"],"names":[],"mappings":"AA8DE","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextInputV2/index.tsx","sourcesContent":["'use client'\n\nimport styled from '@emotion/styled'\nimport { AlertCircleIcon, CheckCircleIcon } from '@ultraviolet/icons'\nimport type {\n  ChangeEvent,\n  ChangeEventHandler,\n  InputHTMLAttributes,\n  ReactNode,\n} from 'react'\nimport {\n  forwardRef,\n  useCallback,\n  useId,\n  useImperativeHandle,\n  useMemo,\n  useRef,\n  useState,\n} from 'react'\nimport { Button } from '../Button'\nimport { Label } from '../Label'\nimport { Loader } from '../Loader'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\nimport { Tooltip } from '../Tooltip'\n\n// SIZE\nexport const TEXTINPUT_SIZE_HEIGHT = {\n  small: '400', // sizing theme tokens key\n  medium: '500',\n  large: '600',\n} as const\ntype TextInputSize = keyof typeof TEXTINPUT_SIZE_HEIGHT\n\nexport const BasicPrefixStack = styled(Stack)`\n  padding: ${({ theme }) => theme.space['2']};\n\n  &[data-size=\"small\"] {\n    padding: ${({ theme }) => theme.space['1']};\n  }\n  border-right: 1px solid;\n  border-color: inherit;\n`\n\nconst StateStack = styled(Stack)`\n  padding: ${({ theme }) => `0 ${theme.space['2']}`};\n`\n\nexport const BasicSuffixStack = styled(Stack)`\n  padding: ${({ theme }) => `0 ${theme.space['2']}`};\n  border-left: 1px solid;\n  border-color: inherit;\n`\n\nconst CTASuffixStack = styled(Stack)`\n  padding: ${({ theme }) => `0 ${theme.space['1']}`};\n  border-left: 1px solid;\n  border-color: inherit;\n`\n\nexport const StyledInput = styled.input<{\n  'data-size': TextInputSize\n}>`\n  flex: 1;\n  border: none;\n  outline: none;\n  height: 100%;\n  width: 100%;\n  padding-left: ${({ theme }) => theme.space['2']};\n  background: transparent;\n  font-size: ${({ theme }) => theme.typography.bodySmall.fontSize};\n\n  &[data-size='large'] {\n    font-size: ${({ theme }) => theme.typography.body.fontSize};\n  }\n\n  &[data-size='small'] {\n    padding-left: ${({ theme }) => theme.space['1']};\n  }\n`\n\ntype StyledInputWrapperProps = {\n  hasFocus: boolean\n  size: TextInputSize\n}\nconst StyledInputWrapper = styled('div', {\n  shouldForwardProp: prop => !['hasFocus', 'size'].includes(prop),\n})<StyledInputWrapperProps>`\n  display: flex;\n  flex-direction: row;\n  height: ${({ size, theme }) => theme.sizing[TEXTINPUT_SIZE_HEIGHT[size]]};\n\n  background: ${({ theme }) => theme.colors.neutral.background};\n  border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  border-radius: ${({ theme }) => theme.radii.default};\n\n  & > ${StyledInput} {\n    color: ${({ theme }) => theme.colors.neutral.text};\n\n    &::placeholder {\n      color: ${({ theme }) => theme.colors.neutral.textWeak};\n    }\n  }\n\n  &[data-success='true'] {\n    border-color: ${({ theme }) => theme.colors.success.border};\n  }\n\n  &[data-error='true'] {\n    border-color: ${({ theme }) => theme.colors.danger.border};\n  }\n\n  &[data-readonly='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n    border-color: ${({ theme }) => theme.colors.neutral.border};\n  }\n\n  &[data-disabled='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    border-color: ${({ theme }) => theme.colors.neutral.borderDisabled};\n\n    & > ${StyledInput} {\n      color: ${({ theme }) => theme.colors.neutral.textDisabled};\n\n      &::placeholder {\n        color: ${({ theme }) => theme.colors.neutral.textWeakDisabled};\n      }\n    }\n  }\n\n  &:not([data-disabled='true']):not([data-readonly]):hover {\n    border-color: ${({ theme }) => theme.colors.primary.border};\n  }\n\n  ${({ theme, hasFocus }) =>\n    hasFocus\n      ? `\n  box-shadow: ${theme.shadows.focusPrimary};\n  border: 1px solid ${theme.colors.primary.border};\n`\n      : null};\n`\n\ntype TextInputProps = {\n  className?: string\n  clearable?: boolean\n  'data-testid'?: string\n  error?: string\n  helper?: ReactNode\n  label?: string\n  labelDescription?: ReactNode\n  loading?: boolean\n  minLength?: number\n  maxLength?: number\n  onRandomize?: () => void\n  prefix?: ReactNode\n  size?: TextInputSize\n  success?: string | boolean\n  suffix?: ReactNode\n  tooltip?: string\n  type?: 'text' | 'password' | 'url' | 'email'\n  value?: string\n  defaultValue?: string\n  onChangeValue?: (value: string) => void\n} & Pick<\n  InputHTMLAttributes<HTMLInputElement>,\n  | 'onFocus'\n  | 'onBlur'\n  | 'name'\n  | 'id'\n  | 'placeholder'\n  | 'aria-label'\n  | 'aria-labelledby'\n  | 'disabled'\n  | 'readOnly'\n  | 'required'\n  | 'autoFocus'\n  | 'tabIndex'\n  | 'autoComplete'\n  | 'onKeyDown'\n  | 'onKeyUp'\n  | 'role'\n  | 'aria-live'\n  | 'aria-atomic'\n  | 'onChange'\n>\n\n/**\n * This component offers an extended input HTML. The component can be controlled or uncontrolled.\n * To control the component, you need to pass the value and the `onChange` function.\n * If you don't pass the `onChange` function, the component will be uncontrolled and you can set the default value using `defaultValue`\n */\nexport const TextInputV2 = forwardRef<HTMLInputElement, TextInputProps>(\n  (\n    {\n      id,\n      className,\n      tabIndex,\n      value,\n      onChange,\n      onChangeValue,\n      placeholder,\n      disabled = false,\n      readOnly = false,\n      success,\n      error,\n      helper,\n      tooltip,\n      label,\n      autoFocus,\n      required = false,\n      'data-testid': dataTestId,\n      name,\n      onFocus,\n      onBlur,\n      clearable = false,\n      labelDescription,\n      type = 'text',\n      prefix,\n      suffix,\n      size = 'large',\n      loading,\n      onRandomize,\n      minLength,\n      maxLength,\n      'aria-labelledby': ariaLabelledBy,\n      'aria-label': ariaLabel,\n      autoComplete,\n      onKeyDown,\n      onKeyUp,\n      role,\n      'aria-live': ariaLive,\n      'aria-atomic': ariaAtomic,\n      defaultValue,\n    },\n    ref,\n  ) => {\n    const localId = useId()\n    const [hasFocus, setHasFocus] = useState(false)\n    const [localValue, setLocalValue] = useState(defaultValue)\n    const inputRef = useRef<HTMLInputElement>(null)\n    useImperativeHandle(ref, () => inputRef.current as HTMLInputElement)\n\n    const [isPasswordVisible, setIsPasswordVisible] = useState(false)\n    const computedType =\n      type === 'password' && isPasswordVisible ? 'text' : type\n\n    const sentiment = useMemo(() => {\n      if (error) {\n        return 'danger'\n      }\n\n      if (success) {\n        return 'success'\n      }\n\n      return 'neutral'\n    }, [error, success])\n\n    const onChangeCallback: ChangeEventHandler<HTMLInputElement> = useCallback(\n      event => {\n        onChange?.(event)\n        onChangeValue?.(event.target.value)\n        setLocalValue(event.target.value)\n      },\n      [onChange, onChangeValue],\n    )\n\n    const computedValue = value !== undefined ? value : localValue\n\n    const computedClearable = clearable && !!computedValue\n\n    return (\n      <Stack\n        gap={0.5}\n        className={className}\n        role={role}\n        aria-live={ariaLive}\n        aria-atomic={ariaAtomic}\n      >\n        {label || labelDescription ? (\n          <Label\n            labelDescription={labelDescription}\n            required={required}\n            size={size}\n            htmlFor={id ?? localId}\n            id={ariaLabelledBy}\n          >\n            {label}\n          </Label>\n        ) : null}\n        <div>\n          <Tooltip text={tooltip}>\n            <StyledInputWrapper\n              hasFocus={hasFocus}\n              data-disabled={disabled}\n              data-readonly={readOnly}\n              data-success={!!success}\n              data-error={!!error}\n              size={size}\n            >\n              {prefix ? (\n                <BasicPrefixStack\n                  direction=\"row\"\n                  alignItems=\"center\"\n                  data-size={size}\n                >\n                  {typeof prefix === 'string' ? (\n                    <Text\n                      as=\"span\"\n                      sentiment=\"neutral\"\n                      variant=\"bodySmall\"\n                      disabled={disabled}\n                    >\n                      {prefix}\n                    </Text>\n                  ) : (\n                    prefix\n                  )}\n                </BasicPrefixStack>\n              ) : null}\n              <StyledInput\n                type={computedType}\n                aria-invalid={!!error}\n                id={id ?? localId}\n                tabIndex={tabIndex}\n                autoFocus={autoFocus}\n                disabled={disabled}\n                ref={inputRef}\n                value={value}\n                defaultValue={defaultValue}\n                onChange={onChangeCallback}\n                data-size={size}\n                placeholder={placeholder}\n                data-testid={dataTestId}\n                name={name}\n                onFocus={event => {\n                  setHasFocus(true)\n                  onFocus?.(event)\n                }}\n                onBlur={event => {\n                  setHasFocus(false)\n                  onBlur?.(event)\n                }}\n                readOnly={readOnly}\n                minLength={minLength}\n                maxLength={maxLength}\n                aria-labelledby={ariaLabelledBy}\n                aria-label={ariaLabel}\n                autoComplete={autoComplete}\n                required={required}\n                onKeyDown={onKeyDown}\n                onKeyUp={onKeyUp}\n              />\n              {success || error || loading || computedClearable ? (\n                <StateStack direction=\"row\" gap={1} alignItems=\"center\">\n                  {computedClearable ? (\n                    <Button\n                      aria-label=\"clear value\"\n                      disabled={disabled || !computedValue}\n                      variant=\"ghost\"\n                      size={size === 'small' ? 'xsmall' : 'small'}\n                      icon=\"close\"\n                      onClick={() => {\n                        if (inputRef?.current) {\n                          inputRef.current.value = ''\n                          setLocalValue('')\n                          onChangeCallback({\n                            target: { value: '' },\n                            currentTarget: { value: '' },\n                          } as ChangeEvent<HTMLInputElement>)\n                        }\n                      }}\n                      sentiment=\"neutral\"\n                    />\n                  ) : null}\n                  {success ? (\n                    <CheckCircleIcon\n                      sentiment=\"success\"\n                      size=\"small\"\n                      disabled={disabled}\n                    />\n                  ) : null}\n                  {error ? (\n                    <AlertCircleIcon\n                      sentiment=\"danger\"\n                      size=\"small\"\n                      disabled={disabled}\n                    />\n                  ) : null}\n                  {loading && !disabled ? <Loader active size={16} /> : null}\n                </StateStack>\n              ) : null}\n              {suffix ? (\n                <BasicSuffixStack direction=\"row\" alignItems=\"center\">\n                  {typeof suffix === 'string' ? (\n                    <Text\n                      as=\"span\"\n                      sentiment=\"neutral\"\n                      variant=\"bodySmall\"\n                      disabled={disabled}\n                    >\n                      {suffix}\n                    </Text>\n                  ) : (\n                    suffix\n                  )}\n                </BasicSuffixStack>\n              ) : null}\n              {type === 'password' ? (\n                <CTASuffixStack direction=\"row\" alignItems=\"center\">\n                  <Button\n                    disabled={disabled}\n                    data-testid={\n                      dataTestId ? `${dataTestId}-visibility-button` : undefined\n                    }\n                    aria-label={isPasswordVisible ? 'hide' : 'show'}\n                    onClick={() => {\n                      setIsPasswordVisible(!isPasswordVisible)\n                    }}\n                    variant=\"ghost\"\n                    sentiment=\"neutral\"\n                    icon={isPasswordVisible ? 'eye-off' : 'eye'}\n                    size={size === 'small' ? 'xsmall' : 'small'}\n                  />\n                </CTASuffixStack>\n              ) : null}\n              {onRandomize ? (\n                <CTASuffixStack direction=\"row\" alignItems=\"center\">\n                  <Button\n                    disabled={disabled}\n                    icon=\"auto-fix\"\n                    size={size === 'small' ? 'xsmall' : 'small'}\n                    variant=\"ghost\"\n                    sentiment=\"neutral\"\n                    onClick={onRandomize}\n                  />\n                </CTASuffixStack>\n              ) : null}\n            </StyledInputWrapper>\n          </Tooltip>\n        </div>\n        {error || typeof success === 'string' || typeof helper === 'string' ? (\n          <Text\n            as=\"p\"\n            variant=\"caption\"\n            sentiment={sentiment}\n            prominence={!error && !success ? 'weak' : 'default'}\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)\n"]} */"));
|
|
70
70
|
const StyledInputWrapper = /* @__PURE__ */ _styled__default.default("div", process.env.NODE_ENV === "production" ? {
|
|
71
71
|
shouldForwardProp: (prop) => !["hasFocus", "size"].includes(prop),
|
|
72
72
|
target: "e7tir8v0"
|
|
@@ -111,7 +111,7 @@ const StyledInputWrapper = /* @__PURE__ */ _styled__default.default("div", proce
|
|
|
111
111
|
}) => hasFocus ? `
|
|
112
112
|
box-shadow: ${theme.shadows.focusPrimary};
|
|
113
113
|
border: 1px solid ${theme.colors.primary.border};
|
|
114
|
-
` : null, ";" + (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/TextInputV2/index.tsx"],"names":[],"mappings":"AAuF2B","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextInputV2/index.tsx","sourcesContent":["'use client'\n\nimport styled from '@emotion/styled'\nimport { AlertCircleIcon, CheckCircleIcon } from '@ultraviolet/icons'\nimport type {\n  ChangeEvent,\n  ChangeEventHandler,\n  InputHTMLAttributes,\n  ReactNode,\n} from 'react'\nimport {\n  forwardRef,\n  useCallback,\n  useId,\n  useImperativeHandle,\n  useMemo,\n  useRef,\n  useState,\n} from 'react'\nimport { Button } from '../Button'\nimport { Label } from '../Label'\nimport { Loader } from '../Loader'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\nimport { Tooltip } from '../Tooltip'\n\n// SIZE\nexport const TEXTINPUT_SIZE_HEIGHT = {\n  small: '400', // sizing theme tokens key\n  medium: '500',\n  large: '600',\n} as const\ntype TextInputSize = keyof typeof TEXTINPUT_SIZE_HEIGHT\n\nexport const BasicPrefixStack = styled(Stack)`\n  padding: ${({ theme }) => theme.space['2']};\n\n  &[data-size=\"small\"] {\n    padding: ${({ theme }) => theme.space['1']};\n  }\n  border-right: 1px solid;\n  border-color: inherit;\n`\n\nconst StateStack = styled(Stack)`\n  padding: ${({ theme }) => `0 ${theme.space['2']}`};\n`\n\nexport const BasicSuffixStack = styled(Stack)`\n  padding: ${({ theme }) => `0 ${theme.space['2']}`};\n  border-left: 1px solid;\n  border-color: inherit;\n`\n\nconst CTASuffixStack = styled(Stack)`\n  padding: ${({ theme }) => `0 ${theme.space['1']}`};\n  border-left: 1px solid;\n  border-color: inherit;\n`\n\nexport const StyledInput = styled.input<{\n  'data-size': TextInputSize\n}>`\n  flex: 1;\n  border: none;\n  outline: none;\n  height: 100%;\n  width: 100%;\n  padding-left: ${({ theme }) => theme.space['2']};\n  background: transparent;\n  font-size: ${({ theme }) => theme.typography.bodySmall.fontSize};\n\n  &[data-size='large'] {\n    font-size: ${({ theme }) => theme.typography.body.fontSize};\n  }\n\n  &[data-size='small'] {\n    padding-left: ${({ theme }) => theme.space['1']};\n  }\n`\n\ntype StyledInputWrapperProps = {\n  hasFocus: boolean\n  size: TextInputSize\n}\nconst StyledInputWrapper = styled('div', {\n  shouldForwardProp: prop => !['hasFocus', 'size'].includes(prop),\n})<StyledInputWrapperProps>`\n  display: flex;\n  flex-direction: row;\n  height: ${({ size, theme }) => theme.sizing[TEXTINPUT_SIZE_HEIGHT[size]]};\n\n  background: ${({ theme }) => theme.colors.neutral.background};\n  border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  border-radius: ${({ theme }) => theme.radii.default};\n\n  & > ${StyledInput} {\n    color: ${({ theme }) => theme.colors.neutral.text};\n\n    &::placeholder {\n      color: ${({ theme }) => theme.colors.neutral.textWeak};\n    }\n  }\n\n  &[data-success='true'] {\n    border-color: ${({ theme }) => theme.colors.success.border};\n  }\n\n  &[data-error='true'] {\n    border-color: ${({ theme }) => theme.colors.danger.border};\n  }\n\n  &[data-readonly='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n    border-color: ${({ theme }) => theme.colors.neutral.border};\n  }\n\n  &[data-disabled='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    border-color: ${({ theme }) => theme.colors.neutral.borderDisabled};\n\n    & > ${StyledInput} {\n      color: ${({ theme }) => theme.colors.neutral.textDisabled};\n\n      &::placeholder {\n        color: ${({ theme }) => theme.colors.neutral.textWeakDisabled};\n      }\n    }\n  }\n\n  &:not([data-disabled='true']):not([data-readonly]):hover {\n    border-color: ${({ theme }) => theme.colors.primary.border};\n  }\n\n  ${({ theme, hasFocus }) =>\n    hasFocus\n      ? `\n  box-shadow: ${theme.shadows.focusPrimary};\n  border: 1px solid ${theme.colors.primary.border};\n`\n      : null};\n`\n\ntype TextInputProps = {\n  className?: string\n  clearable?: boolean\n  'data-testid'?: string\n  error?: string\n  helper?: ReactNode\n  label?: string\n  labelDescription?: ReactNode\n  loading?: boolean\n  minLength?: number\n  maxLength?: number\n  onRandomize?: () => void\n  prefix?: ReactNode\n  size?: TextInputSize\n  success?: string | boolean\n  suffix?: ReactNode\n  tooltip?: string\n  type?: 'text' | 'password' | 'url' | 'email'\n  value?: string\n  defaultValue?: string\n  onChangeValue?: (value: string) => void\n} & Pick<\n  InputHTMLAttributes<HTMLInputElement>,\n  | 'onFocus'\n  | 'onBlur'\n  | 'name'\n  | 'id'\n  | 'placeholder'\n  | 'aria-label'\n  | 'aria-labelledby'\n  | 'disabled'\n  | 'readOnly'\n  | 'required'\n  | 'autoFocus'\n  | 'tabIndex'\n  | 'autoComplete'\n  | 'onKeyDown'\n  | 'onKeyUp'\n  | 'role'\n  | 'aria-live'\n  | 'aria-atomic'\n  | 'onChange'\n>\n\n/**\n * This component offers an extended input HTML. The component can be controlled or uncontrolled.\n * To control the component, you need to pass the value and the `onChange` function.\n * If you don't pass the `onChange` function, the component will be uncontrolled and you can set the default value using `defaultValue`\n */\nexport const TextInputV2 = forwardRef<HTMLInputElement, TextInputProps>(\n  (\n    {\n      id,\n      className,\n      tabIndex,\n      value,\n      onChange,\n      onChangeValue,\n      placeholder,\n      disabled = false,\n      readOnly = false,\n      success,\n      error,\n      helper,\n      tooltip,\n      label,\n      autoFocus,\n      required = false,\n      'data-testid': dataTestId,\n      name,\n      onFocus,\n      onBlur,\n      clearable = false,\n      labelDescription,\n      type = 'text',\n      prefix,\n      suffix,\n      size = 'large',\n      loading,\n      onRandomize,\n      minLength,\n      maxLength,\n      'aria-labelledby': ariaLabelledBy,\n      'aria-label': ariaLabel,\n      autoComplete,\n      onKeyDown,\n      onKeyUp,\n      role,\n      'aria-live': ariaLive,\n      'aria-atomic': ariaAtomic,\n      defaultValue,\n    },\n    ref,\n  ) => {\n    const localId = useId()\n    const [hasFocus, setHasFocus] = useState(false)\n    const inputRef = useRef<HTMLInputElement>(null)\n    useImperativeHandle(ref, () => inputRef.current as HTMLInputElement)\n\n    const [isPasswordVisible, setIsPasswordVisible] = useState(false)\n    const computedType =\n      type === 'password' && isPasswordVisible ? 'text' : type\n\n    const sentiment = useMemo(() => {\n      if (error) {\n        return 'danger'\n      }\n\n      if (success) {\n        return 'success'\n      }\n\n      return 'neutral'\n    }, [error, success])\n\n    const onChangeCallback: ChangeEventHandler<HTMLInputElement> = useCallback(\n      event => {\n        onChange?.(event)\n        onChangeValue?.(event.target.value)\n      },\n      [onChange, onChangeValue],\n    )\n\n    const computedClearable = clearable && !!value\n\n    return (\n      <Stack\n        gap={0.5}\n        className={className}\n        role={role}\n        aria-live={ariaLive}\n        aria-atomic={ariaAtomic}\n      >\n        {label || labelDescription ? (\n          <Label\n            labelDescription={labelDescription}\n            required={required}\n            size={size}\n            htmlFor={id ?? localId}\n            id={ariaLabelledBy}\n          >\n            {label}\n          </Label>\n        ) : null}\n        <div>\n          <Tooltip text={tooltip}>\n            <StyledInputWrapper\n              hasFocus={hasFocus}\n              data-disabled={disabled}\n              data-readonly={readOnly}\n              data-success={!!success}\n              data-error={!!error}\n              size={size}\n            >\n              {prefix ? (\n                <BasicPrefixStack\n                  direction=\"row\"\n                  alignItems=\"center\"\n                  data-size={size}\n                >\n                  {typeof prefix === 'string' ? (\n                    <Text\n                      as=\"span\"\n                      sentiment=\"neutral\"\n                      variant=\"bodySmall\"\n                      disabled={disabled}\n                    >\n                      {prefix}\n                    </Text>\n                  ) : (\n                    prefix\n                  )}\n                </BasicPrefixStack>\n              ) : null}\n              <StyledInput\n                type={computedType}\n                aria-invalid={!!error}\n                id={id ?? localId}\n                tabIndex={tabIndex}\n                autoFocus={autoFocus}\n                disabled={disabled}\n                ref={inputRef}\n                value={value}\n                defaultValue={defaultValue}\n                onChange={onChangeCallback}\n                data-size={size}\n                placeholder={placeholder}\n                data-testid={dataTestId}\n                name={name}\n                onFocus={event => {\n                  setHasFocus(true)\n                  onFocus?.(event)\n                }}\n                onBlur={event => {\n                  setHasFocus(false)\n                  onBlur?.(event)\n                }}\n                readOnly={readOnly}\n                minLength={minLength}\n                maxLength={maxLength}\n                aria-labelledby={ariaLabelledBy}\n                aria-label={ariaLabel}\n                autoComplete={autoComplete}\n                required={required}\n                onKeyDown={onKeyDown}\n                onKeyUp={onKeyUp}\n              />\n              {success || error || loading || computedClearable ? (\n                <StateStack direction=\"row\" gap={1} alignItems=\"center\">\n                  {computedClearable ? (\n                    <Button\n                      aria-label=\"clear value\"\n                      disabled={disabled || !value}\n                      variant=\"ghost\"\n                      size={size === 'small' ? 'xsmall' : 'small'}\n                      icon=\"close\"\n                      onClick={() => {\n                        if (inputRef?.current) {\n                          inputRef.current.value = ''\n                          onChangeCallback({\n                            target: { value: '' },\n                            currentTarget: { value: '' },\n                          } as ChangeEvent<HTMLInputElement>)\n                        }\n                      }}\n                      sentiment=\"neutral\"\n                    />\n                  ) : null}\n                  {success ? (\n                    <CheckCircleIcon\n                      sentiment=\"success\"\n                      size=\"small\"\n                      disabled={disabled}\n                    />\n                  ) : null}\n                  {error ? (\n                    <AlertCircleIcon\n                      sentiment=\"danger\"\n                      size=\"small\"\n                      disabled={disabled}\n                    />\n                  ) : null}\n                  {loading && !disabled ? <Loader active size={16} /> : null}\n                </StateStack>\n              ) : null}\n              {suffix ? (\n                <BasicSuffixStack direction=\"row\" alignItems=\"center\">\n                  {typeof suffix === 'string' ? (\n                    <Text\n                      as=\"span\"\n                      sentiment=\"neutral\"\n                      variant=\"bodySmall\"\n                      disabled={disabled}\n                    >\n                      {suffix}\n                    </Text>\n                  ) : (\n                    suffix\n                  )}\n                </BasicSuffixStack>\n              ) : null}\n              {type === 'password' ? (\n                <CTASuffixStack direction=\"row\" alignItems=\"center\">\n                  <Button\n                    disabled={disabled}\n                    data-testid={\n                      dataTestId ? `${dataTestId}-visibility-button` : undefined\n                    }\n                    aria-label={isPasswordVisible ? 'hide' : 'show'}\n                    onClick={() => {\n                      setIsPasswordVisible(!isPasswordVisible)\n                    }}\n                    variant=\"ghost\"\n                    sentiment=\"neutral\"\n                    icon={isPasswordVisible ? 'eye-off' : 'eye'}\n                    size={size === 'small' ? 'xsmall' : 'small'}\n                  />\n                </CTASuffixStack>\n              ) : null}\n              {onRandomize ? (\n                <CTASuffixStack direction=\"row\" alignItems=\"center\">\n                  <Button\n                    disabled={disabled}\n                    icon=\"auto-fix\"\n                    size={size === 'small' ? 'xsmall' : 'small'}\n                    variant=\"ghost\"\n                    sentiment=\"neutral\"\n                    onClick={onRandomize}\n                  />\n                </CTASuffixStack>\n              ) : null}\n            </StyledInputWrapper>\n          </Tooltip>\n        </div>\n        {error || typeof success === 'string' || typeof helper === 'string' ? (\n          <Text\n            as=\"p\"\n            variant=\"caption\"\n            sentiment={sentiment}\n            prominence={!error && !success ? 'weak' : 'default'}\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)\n"]} */"));
|
|
114
|
+
` : null, ";" + (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/TextInputV2/index.tsx"],"names":[],"mappings":"AAuF2B","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextInputV2/index.tsx","sourcesContent":["'use client'\n\nimport styled from '@emotion/styled'\nimport { AlertCircleIcon, CheckCircleIcon } from '@ultraviolet/icons'\nimport type {\n  ChangeEvent,\n  ChangeEventHandler,\n  InputHTMLAttributes,\n  ReactNode,\n} from 'react'\nimport {\n  forwardRef,\n  useCallback,\n  useId,\n  useImperativeHandle,\n  useMemo,\n  useRef,\n  useState,\n} from 'react'\nimport { Button } from '../Button'\nimport { Label } from '../Label'\nimport { Loader } from '../Loader'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\nimport { Tooltip } from '../Tooltip'\n\n// SIZE\nexport const TEXTINPUT_SIZE_HEIGHT = {\n  small: '400', // sizing theme tokens key\n  medium: '500',\n  large: '600',\n} as const\ntype TextInputSize = keyof typeof TEXTINPUT_SIZE_HEIGHT\n\nexport const BasicPrefixStack = styled(Stack)`\n  padding: ${({ theme }) => theme.space['2']};\n\n  &[data-size=\"small\"] {\n    padding: ${({ theme }) => theme.space['1']};\n  }\n  border-right: 1px solid;\n  border-color: inherit;\n`\n\nconst StateStack = styled(Stack)`\n  padding: ${({ theme }) => `0 ${theme.space['2']}`};\n`\n\nexport const BasicSuffixStack = styled(Stack)`\n  padding: ${({ theme }) => `0 ${theme.space['2']}`};\n  border-left: 1px solid;\n  border-color: inherit;\n`\n\nconst CTASuffixStack = styled(Stack)`\n  padding: ${({ theme }) => `0 ${theme.space['1']}`};\n  border-left: 1px solid;\n  border-color: inherit;\n`\n\nexport const StyledInput = styled.input<{\n  'data-size': TextInputSize\n}>`\n  flex: 1;\n  border: none;\n  outline: none;\n  height: 100%;\n  width: 100%;\n  padding-left: ${({ theme }) => theme.space['2']};\n  background: transparent;\n  font-size: ${({ theme }) => theme.typography.bodySmall.fontSize};\n\n  &[data-size='large'] {\n    font-size: ${({ theme }) => theme.typography.body.fontSize};\n  }\n\n  &[data-size='small'] {\n    padding-left: ${({ theme }) => theme.space['1']};\n  }\n`\n\ntype StyledInputWrapperProps = {\n  hasFocus: boolean\n  size: TextInputSize\n}\nconst StyledInputWrapper = styled('div', {\n  shouldForwardProp: prop => !['hasFocus', 'size'].includes(prop),\n})<StyledInputWrapperProps>`\n  display: flex;\n  flex-direction: row;\n  height: ${({ size, theme }) => theme.sizing[TEXTINPUT_SIZE_HEIGHT[size]]};\n\n  background: ${({ theme }) => theme.colors.neutral.background};\n  border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  border-radius: ${({ theme }) => theme.radii.default};\n\n  & > ${StyledInput} {\n    color: ${({ theme }) => theme.colors.neutral.text};\n\n    &::placeholder {\n      color: ${({ theme }) => theme.colors.neutral.textWeak};\n    }\n  }\n\n  &[data-success='true'] {\n    border-color: ${({ theme }) => theme.colors.success.border};\n  }\n\n  &[data-error='true'] {\n    border-color: ${({ theme }) => theme.colors.danger.border};\n  }\n\n  &[data-readonly='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n    border-color: ${({ theme }) => theme.colors.neutral.border};\n  }\n\n  &[data-disabled='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    border-color: ${({ theme }) => theme.colors.neutral.borderDisabled};\n\n    & > ${StyledInput} {\n      color: ${({ theme }) => theme.colors.neutral.textDisabled};\n\n      &::placeholder {\n        color: ${({ theme }) => theme.colors.neutral.textWeakDisabled};\n      }\n    }\n  }\n\n  &:not([data-disabled='true']):not([data-readonly]):hover {\n    border-color: ${({ theme }) => theme.colors.primary.border};\n  }\n\n  ${({ theme, hasFocus }) =>\n    hasFocus\n      ? `\n  box-shadow: ${theme.shadows.focusPrimary};\n  border: 1px solid ${theme.colors.primary.border};\n`\n      : null};\n`\n\ntype TextInputProps = {\n  className?: string\n  clearable?: boolean\n  'data-testid'?: string\n  error?: string\n  helper?: ReactNode\n  label?: string\n  labelDescription?: ReactNode\n  loading?: boolean\n  minLength?: number\n  maxLength?: number\n  onRandomize?: () => void\n  prefix?: ReactNode\n  size?: TextInputSize\n  success?: string | boolean\n  suffix?: ReactNode\n  tooltip?: string\n  type?: 'text' | 'password' | 'url' | 'email'\n  value?: string\n  defaultValue?: string\n  onChangeValue?: (value: string) => void\n} & Pick<\n  InputHTMLAttributes<HTMLInputElement>,\n  | 'onFocus'\n  | 'onBlur'\n  | 'name'\n  | 'id'\n  | 'placeholder'\n  | 'aria-label'\n  | 'aria-labelledby'\n  | 'disabled'\n  | 'readOnly'\n  | 'required'\n  | 'autoFocus'\n  | 'tabIndex'\n  | 'autoComplete'\n  | 'onKeyDown'\n  | 'onKeyUp'\n  | 'role'\n  | 'aria-live'\n  | 'aria-atomic'\n  | 'onChange'\n>\n\n/**\n * This component offers an extended input HTML. The component can be controlled or uncontrolled.\n * To control the component, you need to pass the value and the `onChange` function.\n * If you don't pass the `onChange` function, the component will be uncontrolled and you can set the default value using `defaultValue`\n */\nexport const TextInputV2 = forwardRef<HTMLInputElement, TextInputProps>(\n  (\n    {\n      id,\n      className,\n      tabIndex,\n      value,\n      onChange,\n      onChangeValue,\n      placeholder,\n      disabled = false,\n      readOnly = false,\n      success,\n      error,\n      helper,\n      tooltip,\n      label,\n      autoFocus,\n      required = false,\n      'data-testid': dataTestId,\n      name,\n      onFocus,\n      onBlur,\n      clearable = false,\n      labelDescription,\n      type = 'text',\n      prefix,\n      suffix,\n      size = 'large',\n      loading,\n      onRandomize,\n      minLength,\n      maxLength,\n      'aria-labelledby': ariaLabelledBy,\n      'aria-label': ariaLabel,\n      autoComplete,\n      onKeyDown,\n      onKeyUp,\n      role,\n      'aria-live': ariaLive,\n      'aria-atomic': ariaAtomic,\n      defaultValue,\n    },\n    ref,\n  ) => {\n    const localId = useId()\n    const [hasFocus, setHasFocus] = useState(false)\n    const [localValue, setLocalValue] = useState(defaultValue)\n    const inputRef = useRef<HTMLInputElement>(null)\n    useImperativeHandle(ref, () => inputRef.current as HTMLInputElement)\n\n    const [isPasswordVisible, setIsPasswordVisible] = useState(false)\n    const computedType =\n      type === 'password' && isPasswordVisible ? 'text' : type\n\n    const sentiment = useMemo(() => {\n      if (error) {\n        return 'danger'\n      }\n\n      if (success) {\n        return 'success'\n      }\n\n      return 'neutral'\n    }, [error, success])\n\n    const onChangeCallback: ChangeEventHandler<HTMLInputElement> = useCallback(\n      event => {\n        onChange?.(event)\n        onChangeValue?.(event.target.value)\n        setLocalValue(event.target.value)\n      },\n      [onChange, onChangeValue],\n    )\n\n    const computedValue = value !== undefined ? value : localValue\n\n    const computedClearable = clearable && !!computedValue\n\n    return (\n      <Stack\n        gap={0.5}\n        className={className}\n        role={role}\n        aria-live={ariaLive}\n        aria-atomic={ariaAtomic}\n      >\n        {label || labelDescription ? (\n          <Label\n            labelDescription={labelDescription}\n            required={required}\n            size={size}\n            htmlFor={id ?? localId}\n            id={ariaLabelledBy}\n          >\n            {label}\n          </Label>\n        ) : null}\n        <div>\n          <Tooltip text={tooltip}>\n            <StyledInputWrapper\n              hasFocus={hasFocus}\n              data-disabled={disabled}\n              data-readonly={readOnly}\n              data-success={!!success}\n              data-error={!!error}\n              size={size}\n            >\n              {prefix ? (\n                <BasicPrefixStack\n                  direction=\"row\"\n                  alignItems=\"center\"\n                  data-size={size}\n                >\n                  {typeof prefix === 'string' ? (\n                    <Text\n                      as=\"span\"\n                      sentiment=\"neutral\"\n                      variant=\"bodySmall\"\n                      disabled={disabled}\n                    >\n                      {prefix}\n                    </Text>\n                  ) : (\n                    prefix\n                  )}\n                </BasicPrefixStack>\n              ) : null}\n              <StyledInput\n                type={computedType}\n                aria-invalid={!!error}\n                id={id ?? localId}\n                tabIndex={tabIndex}\n                autoFocus={autoFocus}\n                disabled={disabled}\n                ref={inputRef}\n                value={value}\n                defaultValue={defaultValue}\n                onChange={onChangeCallback}\n                data-size={size}\n                placeholder={placeholder}\n                data-testid={dataTestId}\n                name={name}\n                onFocus={event => {\n                  setHasFocus(true)\n                  onFocus?.(event)\n                }}\n                onBlur={event => {\n                  setHasFocus(false)\n                  onBlur?.(event)\n                }}\n                readOnly={readOnly}\n                minLength={minLength}\n                maxLength={maxLength}\n                aria-labelledby={ariaLabelledBy}\n                aria-label={ariaLabel}\n                autoComplete={autoComplete}\n                required={required}\n                onKeyDown={onKeyDown}\n                onKeyUp={onKeyUp}\n              />\n              {success || error || loading || computedClearable ? (\n                <StateStack direction=\"row\" gap={1} alignItems=\"center\">\n                  {computedClearable ? (\n                    <Button\n                      aria-label=\"clear value\"\n                      disabled={disabled || !computedValue}\n                      variant=\"ghost\"\n                      size={size === 'small' ? 'xsmall' : 'small'}\n                      icon=\"close\"\n                      onClick={() => {\n                        if (inputRef?.current) {\n                          inputRef.current.value = ''\n                          setLocalValue('')\n                          onChangeCallback({\n                            target: { value: '' },\n                            currentTarget: { value: '' },\n                          } as ChangeEvent<HTMLInputElement>)\n                        }\n                      }}\n                      sentiment=\"neutral\"\n                    />\n                  ) : null}\n                  {success ? (\n                    <CheckCircleIcon\n                      sentiment=\"success\"\n                      size=\"small\"\n                      disabled={disabled}\n                    />\n                  ) : null}\n                  {error ? (\n                    <AlertCircleIcon\n                      sentiment=\"danger\"\n                      size=\"small\"\n                      disabled={disabled}\n                    />\n                  ) : null}\n                  {loading && !disabled ? <Loader active size={16} /> : null}\n                </StateStack>\n              ) : null}\n              {suffix ? (\n                <BasicSuffixStack direction=\"row\" alignItems=\"center\">\n                  {typeof suffix === 'string' ? (\n                    <Text\n                      as=\"span\"\n                      sentiment=\"neutral\"\n                      variant=\"bodySmall\"\n                      disabled={disabled}\n                    >\n                      {suffix}\n                    </Text>\n                  ) : (\n                    suffix\n                  )}\n                </BasicSuffixStack>\n              ) : null}\n              {type === 'password' ? (\n                <CTASuffixStack direction=\"row\" alignItems=\"center\">\n                  <Button\n                    disabled={disabled}\n                    data-testid={\n                      dataTestId ? `${dataTestId}-visibility-button` : undefined\n                    }\n                    aria-label={isPasswordVisible ? 'hide' : 'show'}\n                    onClick={() => {\n                      setIsPasswordVisible(!isPasswordVisible)\n                    }}\n                    variant=\"ghost\"\n                    sentiment=\"neutral\"\n                    icon={isPasswordVisible ? 'eye-off' : 'eye'}\n                    size={size === 'small' ? 'xsmall' : 'small'}\n                  />\n                </CTASuffixStack>\n              ) : null}\n              {onRandomize ? (\n                <CTASuffixStack direction=\"row\" alignItems=\"center\">\n                  <Button\n                    disabled={disabled}\n                    icon=\"auto-fix\"\n                    size={size === 'small' ? 'xsmall' : 'small'}\n                    variant=\"ghost\"\n                    sentiment=\"neutral\"\n                    onClick={onRandomize}\n                  />\n                </CTASuffixStack>\n              ) : null}\n            </StyledInputWrapper>\n          </Tooltip>\n        </div>\n        {error || typeof success === 'string' || typeof helper === 'string' ? (\n          <Text\n            as=\"p\"\n            variant=\"caption\"\n            sentiment={sentiment}\n            prominence={!error && !success ? 'weak' : 'default'}\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)\n"]} */"));
|
|
115
115
|
const TextInputV2 = React.forwardRef(({
|
|
116
116
|
id,
|
|
117
117
|
className,
|
|
@@ -155,6 +155,7 @@ const TextInputV2 = React.forwardRef(({
|
|
|
155
155
|
}, ref) => {
|
|
156
156
|
const localId = React.useId();
|
|
157
157
|
const [hasFocus, setHasFocus] = React.useState(false);
|
|
158
|
+
const [localValue, setLocalValue] = React.useState(defaultValue);
|
|
158
159
|
const inputRef = React.useRef(null);
|
|
159
160
|
React.useImperativeHandle(ref, () => inputRef.current);
|
|
160
161
|
const [isPasswordVisible, setIsPasswordVisible] = React.useState(false);
|
|
@@ -171,8 +172,10 @@ const TextInputV2 = React.forwardRef(({
|
|
|
171
172
|
const onChangeCallback = React.useCallback((event) => {
|
|
172
173
|
onChange?.(event);
|
|
173
174
|
onChangeValue?.(event.target.value);
|
|
175
|
+
setLocalValue(event.target.value);
|
|
174
176
|
}, [onChange, onChangeValue]);
|
|
175
|
-
const
|
|
177
|
+
const computedValue = value !== void 0 ? value : localValue;
|
|
178
|
+
const computedClearable = clearable && !!computedValue;
|
|
176
179
|
return /* @__PURE__ */ jsxRuntime.jsxs(index.Stack, { gap: 0.5, className, role, "aria-live": ariaLive, "aria-atomic": ariaAtomic, children: [
|
|
177
180
|
label || labelDescription ? /* @__PURE__ */ jsxRuntime.jsx(index$1.Label, { labelDescription, required, size, htmlFor: id ?? localId, id: ariaLabelledBy, children: label }) : null,
|
|
178
181
|
/* @__PURE__ */ jsxRuntime.jsx("div", { children: /* @__PURE__ */ jsxRuntime.jsx(index$2.Tooltip, { text: tooltip, children: /* @__PURE__ */ jsxRuntime.jsxs(StyledInputWrapper, { hasFocus, "data-disabled": disabled, "data-readonly": readOnly, "data-success": !!success, "data-error": !!error, size, children: [
|
|
@@ -185,9 +188,10 @@ const TextInputV2 = React.forwardRef(({
|
|
|
185
188
|
onBlur?.(event);
|
|
186
189
|
}, readOnly, minLength, maxLength, "aria-labelledby": ariaLabelledBy, "aria-label": ariaLabel, autoComplete, required, onKeyDown, onKeyUp }),
|
|
187
190
|
success || error || loading || computedClearable ? /* @__PURE__ */ jsxRuntime.jsxs(StateStack, { direction: "row", gap: 1, alignItems: "center", children: [
|
|
188
|
-
computedClearable ? /* @__PURE__ */ jsxRuntime.jsx(index$4.Button, { "aria-label": "clear value", disabled: disabled || !
|
|
191
|
+
computedClearable ? /* @__PURE__ */ jsxRuntime.jsx(index$4.Button, { "aria-label": "clear value", disabled: disabled || !computedValue, variant: "ghost", size: size === "small" ? "xsmall" : "small", icon: "close", onClick: () => {
|
|
189
192
|
if (inputRef?.current) {
|
|
190
193
|
inputRef.current.value = "";
|
|
194
|
+
setLocalValue("");
|
|
191
195
|
onChangeCallback({
|
|
192
196
|
target: {
|
|
193
197
|
value: ""
|