@ultraviolet/ui 1.58.0 → 1.59.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -19,7 +19,7 @@ const BasicPrefixStack = /* @__PURE__ */ _styled(Stack, process.env.NODE_ENV ===
19
19
  label: "BasicPrefixStack"
20
20
  })("padding:", ({
21
21
  theme
22
- }) => theme.space["2"], ";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":"AAkB6C","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextInputV2/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { Icon } from '@ultraviolet/icons'\nimport type { InputHTMLAttributes, ReactNode } from 'react'\nimport { forwardRef, useId, useMemo, useState } from 'react'\nimport { Button } from '../Button'\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  large: 48,\n  medium: 40,\n  small: 32,\n} as const\ntype TextInputSize = keyof typeof TEXTINPUT_SIZE_HEIGHT\n\nexport const BasicPrefixStack = styled(Stack)`\n  padding: ${({ theme }) => theme.space['2']};\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\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 }) => TEXTINPUT_SIZE_HEIGHT[size]}px;\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  onChange?: (newValue: string) => 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} & 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>\n\n/**\n * This component offers an extended input HTML\n */\nexport const TextInputV2 = forwardRef<HTMLInputElement, TextInputProps>(\n  (\n    {\n      id,\n      className,\n      tabIndex,\n      value,\n      onChange,\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    },\n    ref,\n  ) => {\n    const localId = useId()\n    const [hasFocus, setHasFocus] = useState(false)\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 computedClearable = clearable && !!value\n\n    return (\n      <Stack gap={0.5} className={className}>\n        {label || labelDescription ? (\n          <Stack direction=\"row\" gap=\"1\" alignItems=\"center\">\n            {label ? (\n              <Stack direction=\"row\" gap=\"0.5\" alignItems=\"start\">\n                <Text\n                  as=\"label\"\n                  variant={size === 'large' ? 'bodyStrong' : 'bodySmallStrong'}\n                  sentiment=\"neutral\"\n                  htmlFor={id ?? localId}\n                >\n                  {label}\n                </Text>\n                {required ? (\n                  <Icon name=\"asterisk\" color=\"danger\" size={8} />\n                ) : null}\n              </Stack>\n            ) : null}\n            {labelDescription ?? null}\n          </Stack>\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 direction=\"row\" alignItems=\"center\">\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={ref}\n                value={value === null || value === undefined ? '' : value}\n                onChange={event => {\n                  onChange?.(event.currentTarget.value)\n                }}\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              />\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                        onChange?.('')\n                      }}\n                      sentiment=\"neutral\"\n                    />\n                  ) : null}\n                  {success ? (\n                    <Icon\n                      name=\"checkbox-circle-outline\"\n                      sentiment=\"success\"\n                      size=\"small\"\n                      disabled={disabled}\n                    />\n                  ) : null}\n                  {error ? (\n                    <Icon\n                      name=\"alert\"\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"]} */"));
22
+ }) => theme.space["2"], ";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":"AAkB6C","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextInputV2/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { Icon } from '@ultraviolet/icons'\nimport type { InputHTMLAttributes, ReactNode } from 'react'\nimport { forwardRef, useId, useMemo, useState } from 'react'\nimport { Button } from '../Button'\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  large: 48,\n  medium: 40,\n  small: 32,\n} as const\ntype TextInputSize = keyof typeof TEXTINPUT_SIZE_HEIGHT\n\nexport const BasicPrefixStack = styled(Stack)`\n  padding: ${({ theme }) => theme.space['2']};\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\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 }) => TEXTINPUT_SIZE_HEIGHT[size]}px;\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  onChange?: (newValue: string) => 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} & 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  | 'role'\n  | 'aria-live'\n  | 'aria-atomic'\n>\n\n/**\n * This component offers an extended input HTML\n */\nexport const TextInputV2 = forwardRef<HTMLInputElement, TextInputProps>(\n  (\n    {\n      id,\n      className,\n      tabIndex,\n      value,\n      onChange,\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      role,\n      'aria-live': ariaLive,\n      'aria-atomic': ariaAtomic,\n    },\n    ref,\n  ) => {\n    const localId = useId()\n    const [hasFocus, setHasFocus] = useState(false)\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 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          <Stack direction=\"row\" gap=\"1\" alignItems=\"center\">\n            {label ? (\n              <Stack direction=\"row\" gap=\"0.5\" alignItems=\"start\">\n                <Text\n                  as=\"label\"\n                  id={ariaLabelledBy}\n                  variant={size === 'large' ? 'bodyStrong' : 'bodySmallStrong'}\n                  sentiment=\"neutral\"\n                  htmlFor={id ?? localId}\n                >\n                  {label}\n                </Text>\n                {required ? (\n                  <Icon name=\"asterisk\" color=\"danger\" size={8} />\n                ) : null}\n              </Stack>\n            ) : null}\n            {labelDescription ?? null}\n          </Stack>\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 direction=\"row\" alignItems=\"center\">\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={ref}\n                value={value === null || value === undefined ? '' : value}\n                onChange={event => {\n                  onChange?.(event.currentTarget.value)\n                }}\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              />\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                        onChange?.('')\n                      }}\n                      sentiment=\"neutral\"\n                    />\n                  ) : null}\n                  {success ? (\n                    <Icon\n                      name=\"checkbox-circle-outline\"\n                      sentiment=\"success\"\n                      size=\"small\"\n                      disabled={disabled}\n                    />\n                  ) : null}\n                  {error ? (\n                    <Icon\n                      name=\"alert\"\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"]} */"));
23
23
  const StateStack = /* @__PURE__ */ _styled(Stack, process.env.NODE_ENV === "production" ? {
24
24
  target: "e7tir8v4"
25
25
  } : {
@@ -27,7 +27,7 @@ const StateStack = /* @__PURE__ */ _styled(Stack, process.env.NODE_ENV === "prod
27
27
  label: "StateStack"
28
28
  })("padding:", ({
29
29
  theme
30
- }) => `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":"AAwBgC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextInputV2/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { Icon } from '@ultraviolet/icons'\nimport type { InputHTMLAttributes, ReactNode } from 'react'\nimport { forwardRef, useId, useMemo, useState } from 'react'\nimport { Button } from '../Button'\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  large: 48,\n  medium: 40,\n  small: 32,\n} as const\ntype TextInputSize = keyof typeof TEXTINPUT_SIZE_HEIGHT\n\nexport const BasicPrefixStack = styled(Stack)`\n  padding: ${({ theme }) => theme.space['2']};\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\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 }) => TEXTINPUT_SIZE_HEIGHT[size]}px;\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  onChange?: (newValue: string) => 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} & 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>\n\n/**\n * This component offers an extended input HTML\n */\nexport const TextInputV2 = forwardRef<HTMLInputElement, TextInputProps>(\n  (\n    {\n      id,\n      className,\n      tabIndex,\n      value,\n      onChange,\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    },\n    ref,\n  ) => {\n    const localId = useId()\n    const [hasFocus, setHasFocus] = useState(false)\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 computedClearable = clearable && !!value\n\n    return (\n      <Stack gap={0.5} className={className}>\n        {label || labelDescription ? (\n          <Stack direction=\"row\" gap=\"1\" alignItems=\"center\">\n            {label ? (\n              <Stack direction=\"row\" gap=\"0.5\" alignItems=\"start\">\n                <Text\n                  as=\"label\"\n                  variant={size === 'large' ? 'bodyStrong' : 'bodySmallStrong'}\n                  sentiment=\"neutral\"\n                  htmlFor={id ?? localId}\n                >\n                  {label}\n                </Text>\n                {required ? (\n                  <Icon name=\"asterisk\" color=\"danger\" size={8} />\n                ) : null}\n              </Stack>\n            ) : null}\n            {labelDescription ?? null}\n          </Stack>\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 direction=\"row\" alignItems=\"center\">\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={ref}\n                value={value === null || value === undefined ? '' : value}\n                onChange={event => {\n                  onChange?.(event.currentTarget.value)\n                }}\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              />\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                        onChange?.('')\n                      }}\n                      sentiment=\"neutral\"\n                    />\n                  ) : null}\n                  {success ? (\n                    <Icon\n                      name=\"checkbox-circle-outline\"\n                      sentiment=\"success\"\n                      size=\"small\"\n                      disabled={disabled}\n                    />\n                  ) : null}\n                  {error ? (\n                    <Icon\n                      name=\"alert\"\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"]} */"));
30
+ }) => `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":"AAwBgC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextInputV2/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { Icon } from '@ultraviolet/icons'\nimport type { InputHTMLAttributes, ReactNode } from 'react'\nimport { forwardRef, useId, useMemo, useState } from 'react'\nimport { Button } from '../Button'\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  large: 48,\n  medium: 40,\n  small: 32,\n} as const\ntype TextInputSize = keyof typeof TEXTINPUT_SIZE_HEIGHT\n\nexport const BasicPrefixStack = styled(Stack)`\n  padding: ${({ theme }) => theme.space['2']};\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\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 }) => TEXTINPUT_SIZE_HEIGHT[size]}px;\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  onChange?: (newValue: string) => 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} & 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  | 'role'\n  | 'aria-live'\n  | 'aria-atomic'\n>\n\n/**\n * This component offers an extended input HTML\n */\nexport const TextInputV2 = forwardRef<HTMLInputElement, TextInputProps>(\n  (\n    {\n      id,\n      className,\n      tabIndex,\n      value,\n      onChange,\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      role,\n      'aria-live': ariaLive,\n      'aria-atomic': ariaAtomic,\n    },\n    ref,\n  ) => {\n    const localId = useId()\n    const [hasFocus, setHasFocus] = useState(false)\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 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          <Stack direction=\"row\" gap=\"1\" alignItems=\"center\">\n            {label ? (\n              <Stack direction=\"row\" gap=\"0.5\" alignItems=\"start\">\n                <Text\n                  as=\"label\"\n                  id={ariaLabelledBy}\n                  variant={size === 'large' ? 'bodyStrong' : 'bodySmallStrong'}\n                  sentiment=\"neutral\"\n                  htmlFor={id ?? localId}\n                >\n                  {label}\n                </Text>\n                {required ? (\n                  <Icon name=\"asterisk\" color=\"danger\" size={8} />\n                ) : null}\n              </Stack>\n            ) : null}\n            {labelDescription ?? null}\n          </Stack>\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 direction=\"row\" alignItems=\"center\">\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={ref}\n                value={value === null || value === undefined ? '' : value}\n                onChange={event => {\n                  onChange?.(event.currentTarget.value)\n                }}\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              />\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                        onChange?.('')\n                      }}\n                      sentiment=\"neutral\"\n                    />\n                  ) : null}\n                  {success ? (\n                    <Icon\n                      name=\"checkbox-circle-outline\"\n                      sentiment=\"success\"\n                      size=\"small\"\n                      disabled={disabled}\n                    />\n                  ) : null}\n                  {error ? (\n                    <Icon\n                      name=\"alert\"\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
31
  const BasicSuffixStack = /* @__PURE__ */ _styled(Stack, process.env.NODE_ENV === "production" ? {
32
32
  target: "e7tir8v3"
33
33
  } : {
@@ -35,7 +35,7 @@ const BasicSuffixStack = /* @__PURE__ */ _styled(Stack, process.env.NODE_ENV ===
35
35
  label: "BasicSuffixStack"
36
36
  })("padding:", ({
37
37
  theme
38
- }) => `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":"AA4B6C","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextInputV2/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { Icon } from '@ultraviolet/icons'\nimport type { InputHTMLAttributes, ReactNode } from 'react'\nimport { forwardRef, useId, useMemo, useState } from 'react'\nimport { Button } from '../Button'\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  large: 48,\n  medium: 40,\n  small: 32,\n} as const\ntype TextInputSize = keyof typeof TEXTINPUT_SIZE_HEIGHT\n\nexport const BasicPrefixStack = styled(Stack)`\n  padding: ${({ theme }) => theme.space['2']};\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\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 }) => TEXTINPUT_SIZE_HEIGHT[size]}px;\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  onChange?: (newValue: string) => 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} & 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>\n\n/**\n * This component offers an extended input HTML\n */\nexport const TextInputV2 = forwardRef<HTMLInputElement, TextInputProps>(\n  (\n    {\n      id,\n      className,\n      tabIndex,\n      value,\n      onChange,\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    },\n    ref,\n  ) => {\n    const localId = useId()\n    const [hasFocus, setHasFocus] = useState(false)\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 computedClearable = clearable && !!value\n\n    return (\n      <Stack gap={0.5} className={className}>\n        {label || labelDescription ? (\n          <Stack direction=\"row\" gap=\"1\" alignItems=\"center\">\n            {label ? (\n              <Stack direction=\"row\" gap=\"0.5\" alignItems=\"start\">\n                <Text\n                  as=\"label\"\n                  variant={size === 'large' ? 'bodyStrong' : 'bodySmallStrong'}\n                  sentiment=\"neutral\"\n                  htmlFor={id ?? localId}\n                >\n                  {label}\n                </Text>\n                {required ? (\n                  <Icon name=\"asterisk\" color=\"danger\" size={8} />\n                ) : null}\n              </Stack>\n            ) : null}\n            {labelDescription ?? null}\n          </Stack>\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 direction=\"row\" alignItems=\"center\">\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={ref}\n                value={value === null || value === undefined ? '' : value}\n                onChange={event => {\n                  onChange?.(event.currentTarget.value)\n                }}\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              />\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                        onChange?.('')\n                      }}\n                      sentiment=\"neutral\"\n                    />\n                  ) : null}\n                  {success ? (\n                    <Icon\n                      name=\"checkbox-circle-outline\"\n                      sentiment=\"success\"\n                      size=\"small\"\n                      disabled={disabled}\n                    />\n                  ) : null}\n                  {error ? (\n                    <Icon\n                      name=\"alert\"\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"]} */"));
38
+ }) => `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":"AA4B6C","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextInputV2/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { Icon } from '@ultraviolet/icons'\nimport type { InputHTMLAttributes, ReactNode } from 'react'\nimport { forwardRef, useId, useMemo, useState } from 'react'\nimport { Button } from '../Button'\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  large: 48,\n  medium: 40,\n  small: 32,\n} as const\ntype TextInputSize = keyof typeof TEXTINPUT_SIZE_HEIGHT\n\nexport const BasicPrefixStack = styled(Stack)`\n  padding: ${({ theme }) => theme.space['2']};\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\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 }) => TEXTINPUT_SIZE_HEIGHT[size]}px;\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  onChange?: (newValue: string) => 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} & 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  | 'role'\n  | 'aria-live'\n  | 'aria-atomic'\n>\n\n/**\n * This component offers an extended input HTML\n */\nexport const TextInputV2 = forwardRef<HTMLInputElement, TextInputProps>(\n  (\n    {\n      id,\n      className,\n      tabIndex,\n      value,\n      onChange,\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      role,\n      'aria-live': ariaLive,\n      'aria-atomic': ariaAtomic,\n    },\n    ref,\n  ) => {\n    const localId = useId()\n    const [hasFocus, setHasFocus] = useState(false)\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 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          <Stack direction=\"row\" gap=\"1\" alignItems=\"center\">\n            {label ? (\n              <Stack direction=\"row\" gap=\"0.5\" alignItems=\"start\">\n                <Text\n                  as=\"label\"\n                  id={ariaLabelledBy}\n                  variant={size === 'large' ? 'bodyStrong' : 'bodySmallStrong'}\n                  sentiment=\"neutral\"\n                  htmlFor={id ?? localId}\n                >\n                  {label}\n                </Text>\n                {required ? (\n                  <Icon name=\"asterisk\" color=\"danger\" size={8} />\n                ) : null}\n              </Stack>\n            ) : null}\n            {labelDescription ?? null}\n          </Stack>\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 direction=\"row\" alignItems=\"center\">\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={ref}\n                value={value === null || value === undefined ? '' : value}\n                onChange={event => {\n                  onChange?.(event.currentTarget.value)\n                }}\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              />\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                        onChange?.('')\n                      }}\n                      sentiment=\"neutral\"\n                    />\n                  ) : null}\n                  {success ? (\n                    <Icon\n                      name=\"checkbox-circle-outline\"\n                      sentiment=\"success\"\n                      size=\"small\"\n                      disabled={disabled}\n                    />\n                  ) : null}\n                  {error ? (\n                    <Icon\n                      name=\"alert\"\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
39
  const CTASuffixStack = /* @__PURE__ */ _styled(Stack, process.env.NODE_ENV === "production" ? {
40
40
  target: "e7tir8v2"
41
41
  } : {
@@ -43,7 +43,7 @@ const CTASuffixStack = /* @__PURE__ */ _styled(Stack, process.env.NODE_ENV === "
43
43
  label: "CTASuffixStack"
44
44
  })("padding:", ({
45
45
  theme
46
- }) => `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":"AAkCoC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextInputV2/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { Icon } from '@ultraviolet/icons'\nimport type { InputHTMLAttributes, ReactNode } from 'react'\nimport { forwardRef, useId, useMemo, useState } from 'react'\nimport { Button } from '../Button'\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  large: 48,\n  medium: 40,\n  small: 32,\n} as const\ntype TextInputSize = keyof typeof TEXTINPUT_SIZE_HEIGHT\n\nexport const BasicPrefixStack = styled(Stack)`\n  padding: ${({ theme }) => theme.space['2']};\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\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 }) => TEXTINPUT_SIZE_HEIGHT[size]}px;\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  onChange?: (newValue: string) => 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} & 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>\n\n/**\n * This component offers an extended input HTML\n */\nexport const TextInputV2 = forwardRef<HTMLInputElement, TextInputProps>(\n  (\n    {\n      id,\n      className,\n      tabIndex,\n      value,\n      onChange,\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    },\n    ref,\n  ) => {\n    const localId = useId()\n    const [hasFocus, setHasFocus] = useState(false)\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 computedClearable = clearable && !!value\n\n    return (\n      <Stack gap={0.5} className={className}>\n        {label || labelDescription ? (\n          <Stack direction=\"row\" gap=\"1\" alignItems=\"center\">\n            {label ? (\n              <Stack direction=\"row\" gap=\"0.5\" alignItems=\"start\">\n                <Text\n                  as=\"label\"\n                  variant={size === 'large' ? 'bodyStrong' : 'bodySmallStrong'}\n                  sentiment=\"neutral\"\n                  htmlFor={id ?? localId}\n                >\n                  {label}\n                </Text>\n                {required ? (\n                  <Icon name=\"asterisk\" color=\"danger\" size={8} />\n                ) : null}\n              </Stack>\n            ) : null}\n            {labelDescription ?? null}\n          </Stack>\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 direction=\"row\" alignItems=\"center\">\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={ref}\n                value={value === null || value === undefined ? '' : value}\n                onChange={event => {\n                  onChange?.(event.currentTarget.value)\n                }}\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              />\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                        onChange?.('')\n                      }}\n                      sentiment=\"neutral\"\n                    />\n                  ) : null}\n                  {success ? (\n                    <Icon\n                      name=\"checkbox-circle-outline\"\n                      sentiment=\"success\"\n                      size=\"small\"\n                      disabled={disabled}\n                    />\n                  ) : null}\n                  {error ? (\n                    <Icon\n                      name=\"alert\"\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"]} */"));
46
+ }) => `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":"AAkCoC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextInputV2/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { Icon } from '@ultraviolet/icons'\nimport type { InputHTMLAttributes, ReactNode } from 'react'\nimport { forwardRef, useId, useMemo, useState } from 'react'\nimport { Button } from '../Button'\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  large: 48,\n  medium: 40,\n  small: 32,\n} as const\ntype TextInputSize = keyof typeof TEXTINPUT_SIZE_HEIGHT\n\nexport const BasicPrefixStack = styled(Stack)`\n  padding: ${({ theme }) => theme.space['2']};\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\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 }) => TEXTINPUT_SIZE_HEIGHT[size]}px;\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  onChange?: (newValue: string) => 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} & 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  | 'role'\n  | 'aria-live'\n  | 'aria-atomic'\n>\n\n/**\n * This component offers an extended input HTML\n */\nexport const TextInputV2 = forwardRef<HTMLInputElement, TextInputProps>(\n  (\n    {\n      id,\n      className,\n      tabIndex,\n      value,\n      onChange,\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      role,\n      'aria-live': ariaLive,\n      'aria-atomic': ariaAtomic,\n    },\n    ref,\n  ) => {\n    const localId = useId()\n    const [hasFocus, setHasFocus] = useState(false)\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 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          <Stack direction=\"row\" gap=\"1\" alignItems=\"center\">\n            {label ? (\n              <Stack direction=\"row\" gap=\"0.5\" alignItems=\"start\">\n                <Text\n                  as=\"label\"\n                  id={ariaLabelledBy}\n                  variant={size === 'large' ? 'bodyStrong' : 'bodySmallStrong'}\n                  sentiment=\"neutral\"\n                  htmlFor={id ?? localId}\n                >\n                  {label}\n                </Text>\n                {required ? (\n                  <Icon name=\"asterisk\" color=\"danger\" size={8} />\n                ) : null}\n              </Stack>\n            ) : null}\n            {labelDescription ?? null}\n          </Stack>\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 direction=\"row\" alignItems=\"center\">\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={ref}\n                value={value === null || value === undefined ? '' : value}\n                onChange={event => {\n                  onChange?.(event.currentTarget.value)\n                }}\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              />\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                        onChange?.('')\n                      }}\n                      sentiment=\"neutral\"\n                    />\n                  ) : null}\n                  {success ? (\n                    <Icon\n                      name=\"checkbox-circle-outline\"\n                      sentiment=\"success\"\n                      size=\"small\"\n                      disabled={disabled}\n                    />\n                  ) : null}\n                  {error ? (\n                    <Icon\n                      name=\"alert\"\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
47
  const StyledInput = /* @__PURE__ */ _styled("input", process.env.NODE_ENV === "production" ? {
48
48
  target: "e7tir8v1"
49
49
  } : {
@@ -55,7 +55,7 @@ const StyledInput = /* @__PURE__ */ _styled("input", process.env.NODE_ENV === "p
55
55
  theme
56
56
  }) => theme.typography.bodySmall.fontSize, ";&[data-size='large']{font-size:", ({
57
57
  theme
58
- }) => theme.typography.body.fontSize, ";}" + (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":"AA0CE","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextInputV2/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { Icon } from '@ultraviolet/icons'\nimport type { InputHTMLAttributes, ReactNode } from 'react'\nimport { forwardRef, useId, useMemo, useState } from 'react'\nimport { Button } from '../Button'\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  large: 48,\n  medium: 40,\n  small: 32,\n} as const\ntype TextInputSize = keyof typeof TEXTINPUT_SIZE_HEIGHT\n\nexport const BasicPrefixStack = styled(Stack)`\n  padding: ${({ theme }) => theme.space['2']};\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\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 }) => TEXTINPUT_SIZE_HEIGHT[size]}px;\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  onChange?: (newValue: string) => 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} & 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>\n\n/**\n * This component offers an extended input HTML\n */\nexport const TextInputV2 = forwardRef<HTMLInputElement, TextInputProps>(\n  (\n    {\n      id,\n      className,\n      tabIndex,\n      value,\n      onChange,\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    },\n    ref,\n  ) => {\n    const localId = useId()\n    const [hasFocus, setHasFocus] = useState(false)\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 computedClearable = clearable && !!value\n\n    return (\n      <Stack gap={0.5} className={className}>\n        {label || labelDescription ? (\n          <Stack direction=\"row\" gap=\"1\" alignItems=\"center\">\n            {label ? (\n              <Stack direction=\"row\" gap=\"0.5\" alignItems=\"start\">\n                <Text\n                  as=\"label\"\n                  variant={size === 'large' ? 'bodyStrong' : 'bodySmallStrong'}\n                  sentiment=\"neutral\"\n                  htmlFor={id ?? localId}\n                >\n                  {label}\n                </Text>\n                {required ? (\n                  <Icon name=\"asterisk\" color=\"danger\" size={8} />\n                ) : null}\n              </Stack>\n            ) : null}\n            {labelDescription ?? null}\n          </Stack>\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 direction=\"row\" alignItems=\"center\">\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={ref}\n                value={value === null || value === undefined ? '' : value}\n                onChange={event => {\n                  onChange?.(event.currentTarget.value)\n                }}\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              />\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                        onChange?.('')\n                      }}\n                      sentiment=\"neutral\"\n                    />\n                  ) : null}\n                  {success ? (\n                    <Icon\n                      name=\"checkbox-circle-outline\"\n                      sentiment=\"success\"\n                      size=\"small\"\n                      disabled={disabled}\n                    />\n                  ) : null}\n                  {error ? (\n                    <Icon\n                      name=\"alert\"\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"]} */"));
58
+ }) => theme.typography.body.fontSize, ";}" + (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":"AA0CE","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextInputV2/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { Icon } from '@ultraviolet/icons'\nimport type { InputHTMLAttributes, ReactNode } from 'react'\nimport { forwardRef, useId, useMemo, useState } from 'react'\nimport { Button } from '../Button'\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  large: 48,\n  medium: 40,\n  small: 32,\n} as const\ntype TextInputSize = keyof typeof TEXTINPUT_SIZE_HEIGHT\n\nexport const BasicPrefixStack = styled(Stack)`\n  padding: ${({ theme }) => theme.space['2']};\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\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 }) => TEXTINPUT_SIZE_HEIGHT[size]}px;\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  onChange?: (newValue: string) => 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} & 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  | 'role'\n  | 'aria-live'\n  | 'aria-atomic'\n>\n\n/**\n * This component offers an extended input HTML\n */\nexport const TextInputV2 = forwardRef<HTMLInputElement, TextInputProps>(\n  (\n    {\n      id,\n      className,\n      tabIndex,\n      value,\n      onChange,\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      role,\n      'aria-live': ariaLive,\n      'aria-atomic': ariaAtomic,\n    },\n    ref,\n  ) => {\n    const localId = useId()\n    const [hasFocus, setHasFocus] = useState(false)\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 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          <Stack direction=\"row\" gap=\"1\" alignItems=\"center\">\n            {label ? (\n              <Stack direction=\"row\" gap=\"0.5\" alignItems=\"start\">\n                <Text\n                  as=\"label\"\n                  id={ariaLabelledBy}\n                  variant={size === 'large' ? 'bodyStrong' : 'bodySmallStrong'}\n                  sentiment=\"neutral\"\n                  htmlFor={id ?? localId}\n                >\n                  {label}\n                </Text>\n                {required ? (\n                  <Icon name=\"asterisk\" color=\"danger\" size={8} />\n                ) : null}\n              </Stack>\n            ) : null}\n            {labelDescription ?? null}\n          </Stack>\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 direction=\"row\" alignItems=\"center\">\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={ref}\n                value={value === null || value === undefined ? '' : value}\n                onChange={event => {\n                  onChange?.(event.currentTarget.value)\n                }}\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              />\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                        onChange?.('')\n                      }}\n                      sentiment=\"neutral\"\n                    />\n                  ) : null}\n                  {success ? (\n                    <Icon\n                      name=\"checkbox-circle-outline\"\n                      sentiment=\"success\"\n                      size=\"small\"\n                      disabled={disabled}\n                    />\n                  ) : null}\n                  {error ? (\n                    <Icon\n                      name=\"alert\"\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"]} */"));
59
59
  const StyledInputWrapper = /* @__PURE__ */ _styled("div", process.env.NODE_ENV === "production" ? {
60
60
  shouldForwardProp: (prop) => !["hasFocus", "size"].includes(prop),
61
61
  target: "e7tir8v0"
@@ -99,7 +99,7 @@ const StyledInputWrapper = /* @__PURE__ */ _styled("div", process.env.NODE_ENV =
99
99
  }) => hasFocus ? `
100
100
  box-shadow: ${theme.shadows.focusPrimary};
101
101
  border: 1px solid ${theme.colors.primary.border};
102
- ` : 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":"AA+D2B","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextInputV2/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { Icon } from '@ultraviolet/icons'\nimport type { InputHTMLAttributes, ReactNode } from 'react'\nimport { forwardRef, useId, useMemo, useState } from 'react'\nimport { Button } from '../Button'\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  large: 48,\n  medium: 40,\n  small: 32,\n} as const\ntype TextInputSize = keyof typeof TEXTINPUT_SIZE_HEIGHT\n\nexport const BasicPrefixStack = styled(Stack)`\n  padding: ${({ theme }) => theme.space['2']};\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\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 }) => TEXTINPUT_SIZE_HEIGHT[size]}px;\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  onChange?: (newValue: string) => 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} & 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>\n\n/**\n * This component offers an extended input HTML\n */\nexport const TextInputV2 = forwardRef<HTMLInputElement, TextInputProps>(\n  (\n    {\n      id,\n      className,\n      tabIndex,\n      value,\n      onChange,\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    },\n    ref,\n  ) => {\n    const localId = useId()\n    const [hasFocus, setHasFocus] = useState(false)\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 computedClearable = clearable && !!value\n\n    return (\n      <Stack gap={0.5} className={className}>\n        {label || labelDescription ? (\n          <Stack direction=\"row\" gap=\"1\" alignItems=\"center\">\n            {label ? (\n              <Stack direction=\"row\" gap=\"0.5\" alignItems=\"start\">\n                <Text\n                  as=\"label\"\n                  variant={size === 'large' ? 'bodyStrong' : 'bodySmallStrong'}\n                  sentiment=\"neutral\"\n                  htmlFor={id ?? localId}\n                >\n                  {label}\n                </Text>\n                {required ? (\n                  <Icon name=\"asterisk\" color=\"danger\" size={8} />\n                ) : null}\n              </Stack>\n            ) : null}\n            {labelDescription ?? null}\n          </Stack>\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 direction=\"row\" alignItems=\"center\">\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={ref}\n                value={value === null || value === undefined ? '' : value}\n                onChange={event => {\n                  onChange?.(event.currentTarget.value)\n                }}\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              />\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                        onChange?.('')\n                      }}\n                      sentiment=\"neutral\"\n                    />\n                  ) : null}\n                  {success ? (\n                    <Icon\n                      name=\"checkbox-circle-outline\"\n                      sentiment=\"success\"\n                      size=\"small\"\n                      disabled={disabled}\n                    />\n                  ) : null}\n                  {error ? (\n                    <Icon\n                      name=\"alert\"\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"]} */"));
102
+ ` : 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":"AA+D2B","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextInputV2/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { Icon } from '@ultraviolet/icons'\nimport type { InputHTMLAttributes, ReactNode } from 'react'\nimport { forwardRef, useId, useMemo, useState } from 'react'\nimport { Button } from '../Button'\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  large: 48,\n  medium: 40,\n  small: 32,\n} as const\ntype TextInputSize = keyof typeof TEXTINPUT_SIZE_HEIGHT\n\nexport const BasicPrefixStack = styled(Stack)`\n  padding: ${({ theme }) => theme.space['2']};\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\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 }) => TEXTINPUT_SIZE_HEIGHT[size]}px;\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  onChange?: (newValue: string) => 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} & 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  | 'role'\n  | 'aria-live'\n  | 'aria-atomic'\n>\n\n/**\n * This component offers an extended input HTML\n */\nexport const TextInputV2 = forwardRef<HTMLInputElement, TextInputProps>(\n  (\n    {\n      id,\n      className,\n      tabIndex,\n      value,\n      onChange,\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      role,\n      'aria-live': ariaLive,\n      'aria-atomic': ariaAtomic,\n    },\n    ref,\n  ) => {\n    const localId = useId()\n    const [hasFocus, setHasFocus] = useState(false)\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 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          <Stack direction=\"row\" gap=\"1\" alignItems=\"center\">\n            {label ? (\n              <Stack direction=\"row\" gap=\"0.5\" alignItems=\"start\">\n                <Text\n                  as=\"label\"\n                  id={ariaLabelledBy}\n                  variant={size === 'large' ? 'bodyStrong' : 'bodySmallStrong'}\n                  sentiment=\"neutral\"\n                  htmlFor={id ?? localId}\n                >\n                  {label}\n                </Text>\n                {required ? (\n                  <Icon name=\"asterisk\" color=\"danger\" size={8} />\n                ) : null}\n              </Stack>\n            ) : null}\n            {labelDescription ?? null}\n          </Stack>\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 direction=\"row\" alignItems=\"center\">\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={ref}\n                value={value === null || value === undefined ? '' : value}\n                onChange={event => {\n                  onChange?.(event.currentTarget.value)\n                }}\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              />\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                        onChange?.('')\n                      }}\n                      sentiment=\"neutral\"\n                    />\n                  ) : null}\n                  {success ? (\n                    <Icon\n                      name=\"checkbox-circle-outline\"\n                      sentiment=\"success\"\n                      size=\"small\"\n                      disabled={disabled}\n                    />\n                  ) : null}\n                  {error ? (\n                    <Icon\n                      name=\"alert\"\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"]} */"));
103
103
  const TextInputV2 = forwardRef(({
104
104
  id,
105
105
  className,
@@ -133,7 +133,10 @@ const TextInputV2 = forwardRef(({
133
133
  "aria-labelledby": ariaLabelledBy,
134
134
  "aria-label": ariaLabel,
135
135
  autoComplete,
136
- onKeyDown
136
+ onKeyDown,
137
+ role,
138
+ "aria-live": ariaLive,
139
+ "aria-atomic": ariaAtomic
137
140
  }, ref) => {
138
141
  const localId = useId();
139
142
  const [hasFocus, setHasFocus] = useState(false);
@@ -149,10 +152,10 @@ const TextInputV2 = forwardRef(({
149
152
  return "neutral";
150
153
  }, [error, success]);
151
154
  const computedClearable = clearable && !!value;
152
- return /* @__PURE__ */ jsxs(Stack, { gap: 0.5, className, children: [
155
+ return /* @__PURE__ */ jsxs(Stack, { gap: 0.5, className, role, "aria-live": ariaLive, "aria-atomic": ariaAtomic, children: [
153
156
  label || labelDescription ? /* @__PURE__ */ jsxs(Stack, { direction: "row", gap: "1", alignItems: "center", children: [
154
157
  label ? /* @__PURE__ */ jsxs(Stack, { direction: "row", gap: "0.5", alignItems: "start", children: [
155
- /* @__PURE__ */ jsx(Text, { as: "label", variant: size === "large" ? "bodyStrong" : "bodySmallStrong", sentiment: "neutral", htmlFor: id ?? localId, children: label }),
158
+ /* @__PURE__ */ jsx(Text, { as: "label", id: ariaLabelledBy, variant: size === "large" ? "bodyStrong" : "bodySmallStrong", sentiment: "neutral", htmlFor: id ?? localId, children: label }),
156
159
  required ? /* @__PURE__ */ jsx(Icon, { name: "asterisk", color: "danger", size: 8 }) : null
157
160
  ] }) : null,
158
161
  labelDescription ?? null
@@ -0,0 +1,4 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, Symbol.toStringTag, { value: "Module" });
3
+ const isClientSide = !!(typeof window !== "undefined" && window?.document?.createElement);
4
+ exports.isClientSide = isClientSide;
@@ -0,0 +1,4 @@
1
+ /**
2
+ * Check if the code is running on the client side. This is useful for checking if the code is running in the browser and not in SSR.
3
+ */
4
+ export declare const isClientSide: boolean;
@@ -0,0 +1,4 @@
1
+ const isClientSide = !!(typeof window !== "undefined" && window?.document?.createElement);
2
+ export {
3
+ isClientSide
4
+ };
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@ultraviolet/ui",
3
- "version": "1.58.0",
3
+ "version": "1.59.0",
4
4
  "description": "Ultraviolet UI",
5
5
  "homepage": "https://github.com/scaleway/ultraviolet#readme",
6
6
  "repository": {