@ultraviolet/ui 1.56.1 → 1.57.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +1 -1
- package/dist/components/Dialog/index.cjs +2 -2
- package/dist/components/Dialog/index.js +2 -2
- package/dist/components/Dialog/subComponents/Text.cjs +1 -1
- package/dist/components/Dialog/subComponents/Text.js +1 -1
- package/dist/components/SearchInput/Key.cjs +30 -0
- package/dist/components/SearchInput/Key.d.ts +6 -0
- package/dist/components/SearchInput/Key.js +28 -0
- package/dist/components/SearchInput/KeyGroup.cjs +10 -0
- package/dist/components/SearchInput/KeyGroup.d.ts +8 -0
- package/dist/components/SearchInput/KeyGroup.js +10 -0
- package/dist/components/SearchInput/index.cjs +38 -7
- package/dist/components/SearchInput/index.d.ts +2 -1
- package/dist/components/SearchInput/index.js +40 -9
- package/dist/components/SearchInput/types.d.ts +5 -1
- package/dist/components/SelectInputV2/Dropdown.cjs +9 -9
- package/dist/components/SelectInputV2/Dropdown.js +9 -9
- package/dist/components/SelectInputV2/SelectBar.cjs +13 -11
- package/dist/components/SelectInputV2/SelectBar.d.ts +2 -1
- package/dist/components/SelectInputV2/SelectBar.js +9 -7
- package/dist/components/SelectInputV2/index.cjs +4 -3
- package/dist/components/SelectInputV2/index.d.ts +5 -1
- package/dist/components/SelectInputV2/index.js +4 -3
- package/dist/components/TextArea/index.cjs +4 -4
- package/dist/components/TextArea/index.js +4 -4
- package/dist/components/TextInputV2/index.cjs +9 -6
- package/dist/components/TextInputV2/index.d.ts +40 -0
- package/dist/components/TextInputV2/index.js +9 -6
- package/package.json +5 -5
|
@@ -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":"AAkBsC","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\nconst 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\nconst 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\nconst 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>\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"]} */"));
|
|
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\nconst 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\nconst 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\nconst 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>\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"]} */"));
|
|
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":"AA4BsC","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\nconst 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\nconst 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\nconst 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>\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"]} */"));
|
|
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\nconst 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\nconst 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\nconst 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>\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"]} */"));
|
|
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\nconst 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\nconst 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\nconst 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>\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"]} */"));
|
|
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\nconst 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\nconst 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\nconst 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>\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"]} */"));
|
|
103
103
|
const TextInputV2 = forwardRef(({
|
|
104
104
|
id,
|
|
105
105
|
className,
|
|
@@ -187,6 +187,9 @@ const TextInputV2 = forwardRef(({
|
|
|
187
187
|
] });
|
|
188
188
|
});
|
|
189
189
|
export {
|
|
190
|
+
BasicPrefixStack,
|
|
191
|
+
BasicSuffixStack,
|
|
192
|
+
StyledInput,
|
|
190
193
|
TEXTINPUT_SIZE_HEIGHT,
|
|
191
194
|
TextInputV2
|
|
192
195
|
};
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@ultraviolet/ui",
|
|
3
|
-
"version": "1.
|
|
3
|
+
"version": "1.57.1",
|
|
4
4
|
"description": "Ultraviolet UI",
|
|
5
5
|
"homepage": "https://github.com/scaleway/ultraviolet#readme",
|
|
6
6
|
"repository": {
|
|
@@ -68,8 +68,8 @@
|
|
|
68
68
|
"@types/react-dom": "18.3.0",
|
|
69
69
|
"react": "18.3.1",
|
|
70
70
|
"react-dom": "18.3.1",
|
|
71
|
-
"@ultraviolet/icons": "2.12.
|
|
72
|
-
"@ultraviolet/themes": "1.12.
|
|
71
|
+
"@ultraviolet/icons": "2.12.17",
|
|
72
|
+
"@ultraviolet/themes": "1.12.1",
|
|
73
73
|
"@utils/test": "0.0.1"
|
|
74
74
|
},
|
|
75
75
|
"dependencies": {
|
|
@@ -87,8 +87,8 @@
|
|
|
87
87
|
"react-toastify": "10.0.5",
|
|
88
88
|
"react-use-clipboard": "1.0.9",
|
|
89
89
|
"reakit": "1.3.11",
|
|
90
|
-
"@ultraviolet/themes": "1.12.
|
|
91
|
-
"@ultraviolet/icons": "2.12.
|
|
90
|
+
"@ultraviolet/themes": "1.12.1",
|
|
91
|
+
"@ultraviolet/icons": "2.12.17"
|
|
92
92
|
},
|
|
93
93
|
"scripts": {
|
|
94
94
|
"prebuild": "shx rm -rf dist",
|