98 lines
2.5 KiB
TypeScript
98 lines
2.5 KiB
TypeScript
import { InputHTMLAttributes, forwardRef } from 'react';
|
|
import classNames from 'classnames';
|
|
import { Icon, IconName } from '../icon';
|
|
import {
|
|
includesLeftPadding,
|
|
includesRightPadding,
|
|
} from '../../utils/class-names';
|
|
|
|
interface InputProps extends InputHTMLAttributes<HTMLInputElement> {
|
|
hasError?: boolean;
|
|
disabled?: boolean;
|
|
className?: string;
|
|
prependIconName?: IconName;
|
|
appendIconName?: IconName;
|
|
}
|
|
export const inputClassNames = ({
|
|
hasError,
|
|
className,
|
|
}: {
|
|
hasError?: boolean;
|
|
className?: string;
|
|
}) => {
|
|
return classNames(
|
|
[
|
|
'flex items-center w-full',
|
|
'box-border',
|
|
'border',
|
|
'bg-clip-padding',
|
|
'border-black-60 dark:border-white-60',
|
|
'bg-black-25 dark:bg-white-25',
|
|
'text-black-60 dark:text-white-60',
|
|
'text-ui',
|
|
'focus-visible:shadow-focus dark:focus-visible:shadow-focus-dark',
|
|
'focus-visible:outline-0',
|
|
'disabled:bg-black-10 disabled:dark:bg-white-10',
|
|
],
|
|
{
|
|
'pl-8': !includesLeftPadding(className),
|
|
'pr-8': !includesRightPadding(className),
|
|
'border-vega-pink dark:border-vega-pink': hasError,
|
|
},
|
|
className
|
|
);
|
|
};
|
|
|
|
export const inputStyle = ({
|
|
style,
|
|
disabled,
|
|
}: {
|
|
style?: React.CSSProperties;
|
|
disabled?: boolean;
|
|
}) =>
|
|
disabled
|
|
? {
|
|
...style,
|
|
backgroundImage:
|
|
'url()',
|
|
}
|
|
: style;
|
|
|
|
export const Input = forwardRef<HTMLInputElement, InputProps>(
|
|
({ prependIconName, appendIconName, className, ...props }, ref) => {
|
|
className = `${className} h-28`;
|
|
if (prependIconName) {
|
|
className += ' pl-28';
|
|
}
|
|
if (appendIconName) {
|
|
className += ' pr-28';
|
|
}
|
|
const input = (
|
|
<input
|
|
{...props}
|
|
ref={ref}
|
|
className={classNames(inputClassNames({ className, ...props }))}
|
|
/>
|
|
);
|
|
const iconName = prependIconName || appendIconName;
|
|
if (iconName !== undefined) {
|
|
const iconClassName = classNames(
|
|
['fill-black-60 dark:fill-white-60', 'absolute', 'z-10'],
|
|
{
|
|
'left-8': prependIconName,
|
|
'right-8': appendIconName,
|
|
}
|
|
);
|
|
const icon = <Icon name={iconName} className={iconClassName} size={16} />;
|
|
return (
|
|
<div className="inline-flex items-center relative">
|
|
{prependIconName && icon}
|
|
{input}
|
|
{appendIconName && icon}
|
|
</div>
|
|
);
|
|
}
|
|
return input;
|
|
}
|
|
);
|