2022-08-31 02:35:06 -07:00
|
|
|
import classNames from 'clsx';
|
2022-03-21 11:09:01 -07:00
|
|
|
import React from 'react';
|
|
|
|
import { defineMessages, useIntl } from 'react-intl';
|
|
|
|
|
2022-04-04 08:53:47 -07:00
|
|
|
import Icon from '../icon/icon';
|
2022-04-07 11:47:06 -07:00
|
|
|
import SvgIcon from '../icon/svg-icon';
|
2022-03-21 11:09:01 -07:00
|
|
|
import Tooltip from '../tooltip/tooltip';
|
|
|
|
|
|
|
|
const messages = defineMessages({
|
|
|
|
showPassword: { id: 'input.password.show_password', defaultMessage: 'Show password' },
|
|
|
|
hidePassword: { id: 'input.password.hide_password', defaultMessage: 'Hide password' },
|
|
|
|
});
|
|
|
|
|
2022-10-04 14:00:14 -07:00
|
|
|
/** Possible theme names for an Input. */
|
2022-11-07 13:30:53 -08:00
|
|
|
type InputThemes = 'normal' | 'search'
|
2022-10-04 14:00:14 -07:00
|
|
|
|
2022-10-04 13:39:51 -07:00
|
|
|
interface IInput extends Pick<React.InputHTMLAttributes<HTMLInputElement>, 'maxLength' | 'onChange' | 'onBlur' | 'type' | 'autoComplete' | 'autoCorrect' | 'autoCapitalize' | 'required' | 'disabled' | 'onClick' | 'readOnly' | 'min' | 'pattern' | 'onKeyDown' | 'onKeyUp' | 'onFocus' | 'style' | 'id'> {
|
2022-04-30 21:39:58 -07:00
|
|
|
/** Put the cursor into the input on mount. */
|
2022-03-21 11:09:01 -07:00
|
|
|
autoFocus?: boolean,
|
2022-04-30 21:39:58 -07:00
|
|
|
/** The initial text in the input. */
|
2022-03-21 11:09:01 -07:00
|
|
|
defaultValue?: string,
|
2022-04-30 21:39:58 -07:00
|
|
|
/** Extra class names for the <input> element. */
|
2022-03-21 11:09:01 -07:00
|
|
|
className?: string,
|
2022-05-02 19:38:15 -07:00
|
|
|
/** Extra class names for the outer <div> element. */
|
|
|
|
outerClassName?: string,
|
2022-09-07 05:21:14 -07:00
|
|
|
/** URL to the svg icon. Cannot be used with prepend. */
|
2022-03-21 11:09:01 -07:00
|
|
|
icon?: string,
|
2022-04-30 21:39:58 -07:00
|
|
|
/** Internal input name. */
|
2022-03-21 11:09:01 -07:00
|
|
|
name?: string,
|
2022-04-30 21:39:58 -07:00
|
|
|
/** Text to display before a value is entered. */
|
2022-03-21 11:09:01 -07:00
|
|
|
placeholder?: string,
|
2022-04-30 21:39:58 -07:00
|
|
|
/** Text in the input. */
|
2022-05-05 14:35:30 -07:00
|
|
|
value?: string | number,
|
2022-04-30 21:39:58 -07:00
|
|
|
/** Change event handler for the input. */
|
2022-04-07 10:39:22 -07:00
|
|
|
onChange?: (event: React.ChangeEvent<HTMLInputElement>) => void,
|
2022-07-14 08:55:00 -07:00
|
|
|
/** An element to display as prefix to input. Cannot be used with icon. */
|
2022-09-07 05:21:14 -07:00
|
|
|
prepend?: React.ReactElement,
|
|
|
|
/** An element to display as suffix to input. Cannot be used with password type. */
|
|
|
|
append?: React.ReactElement,
|
2022-10-04 14:00:14 -07:00
|
|
|
/** Theme to style the input with. */
|
|
|
|
theme?: InputThemes,
|
2022-03-21 11:09:01 -07:00
|
|
|
}
|
|
|
|
|
2022-04-30 21:39:58 -07:00
|
|
|
/** Form input element. */
|
2022-03-21 11:09:01 -07:00
|
|
|
const Input = React.forwardRef<HTMLInputElement, IInput>(
|
|
|
|
(props, ref) => {
|
|
|
|
const intl = useIntl();
|
|
|
|
|
2023-01-10 10:41:32 -08:00
|
|
|
const { type = 'text', icon, className, outerClassName, append, prepend, theme = 'normal', ...filteredProps } = props;
|
2022-03-21 11:09:01 -07:00
|
|
|
|
|
|
|
const [revealed, setRevealed] = React.useState(false);
|
|
|
|
|
|
|
|
const isPassword = type === 'password';
|
|
|
|
|
|
|
|
const togglePassword = React.useCallback(() => {
|
|
|
|
setRevealed((prev) => !prev);
|
|
|
|
}, []);
|
|
|
|
|
|
|
|
return (
|
2022-09-07 05:21:14 -07:00
|
|
|
<div
|
|
|
|
className={
|
2022-10-05 13:22:13 -07:00
|
|
|
classNames('relative', {
|
2022-10-04 14:00:14 -07:00
|
|
|
'rounded-md': theme !== 'search',
|
|
|
|
'rounded-full': theme === 'search',
|
2022-10-05 13:13:29 -07:00
|
|
|
'mt-1': !String(outerClassName).includes('mt-'),
|
|
|
|
[String(outerClassName)]: typeof outerClassName !== 'undefined',
|
2022-09-07 05:21:14 -07:00
|
|
|
})
|
|
|
|
}
|
|
|
|
>
|
2022-03-21 11:09:01 -07:00
|
|
|
{icon ? (
|
2023-02-01 14:13:42 -08:00
|
|
|
<div className='pointer-events-none absolute inset-y-0 left-0 flex items-center pl-3'>
|
2022-07-22 10:30:16 -07:00
|
|
|
<Icon src={icon} className='h-4 w-4 text-gray-700 dark:text-gray-600' aria-hidden='true' />
|
2022-03-21 11:09:01 -07:00
|
|
|
</div>
|
|
|
|
) : null}
|
|
|
|
|
2022-09-07 05:21:14 -07:00
|
|
|
{prepend ? (
|
2022-07-14 08:55:00 -07:00
|
|
|
<div className='absolute inset-y-0 left-0 flex items-center'>
|
2022-09-07 05:21:14 -07:00
|
|
|
{prepend}
|
2022-07-14 08:55:00 -07:00
|
|
|
</div>
|
|
|
|
) : null}
|
|
|
|
|
2022-03-21 11:09:01 -07:00
|
|
|
<input
|
|
|
|
{...filteredProps}
|
|
|
|
type={revealed ? 'text' : type}
|
|
|
|
ref={ref}
|
2022-10-05 13:13:29 -07:00
|
|
|
className={classNames('text-base placeholder:text-gray-600 dark:placeholder:text-gray-600', {
|
|
|
|
'text-gray-900 dark:text-gray-100 block w-full sm:text-sm dark:ring-1 dark:ring-gray-800 focus:ring-primary-500 focus:border-primary-500 dark:focus:ring-primary-500 dark:focus:border-primary-500':
|
2022-10-04 14:14:08 -07:00
|
|
|
['normal', 'search'].includes(theme),
|
|
|
|
'rounded-md bg-white dark:bg-gray-900 border-gray-400 dark:border-gray-800': theme === 'normal',
|
2022-10-04 14:00:14 -07:00
|
|
|
'rounded-full bg-gray-200 border-gray-200 dark:bg-gray-800 dark:border-gray-800 focus:bg-white': theme === 'search',
|
2022-11-25 09:04:11 -08:00
|
|
|
'pr-7 rtl:pl-7 rtl:pr-3': isPassword || append,
|
2022-03-21 11:09:01 -07:00
|
|
|
'pl-8': typeof icon !== 'undefined',
|
2022-09-07 05:21:14 -07:00
|
|
|
'pl-16': typeof prepend !== 'undefined',
|
2022-03-24 12:27:27 -07:00
|
|
|
}, className)}
|
2022-03-21 11:09:01 -07:00
|
|
|
/>
|
|
|
|
|
2022-09-07 05:21:14 -07:00
|
|
|
{append ? (
|
2023-02-01 14:13:42 -08:00
|
|
|
<div className='absolute inset-y-0 right-0 flex items-center pr-3 rtl:left-0 rtl:right-auto'>
|
2022-09-07 05:21:14 -07:00
|
|
|
{append}
|
|
|
|
</div>
|
|
|
|
) : null}
|
|
|
|
|
2022-03-21 11:09:01 -07:00
|
|
|
{isPassword ? (
|
|
|
|
<Tooltip
|
|
|
|
text={
|
|
|
|
revealed ?
|
|
|
|
intl.formatMessage(messages.hidePassword) :
|
|
|
|
intl.formatMessage(messages.showPassword)
|
|
|
|
}
|
|
|
|
>
|
2023-02-01 14:13:42 -08:00
|
|
|
<div className='absolute inset-y-0 right-0 flex items-center rtl:left-0 rtl:right-auto'>
|
2022-03-21 11:09:01 -07:00
|
|
|
<button
|
|
|
|
type='button'
|
|
|
|
onClick={togglePassword}
|
|
|
|
tabIndex={-1}
|
2023-02-01 14:13:42 -08:00
|
|
|
className='h-full px-2 text-gray-700 hover:text-gray-500 focus:ring-2 focus:ring-primary-500 dark:text-gray-600 dark:hover:text-gray-400'
|
2022-03-21 11:09:01 -07:00
|
|
|
>
|
2022-04-07 11:47:06 -07:00
|
|
|
<SvgIcon
|
2022-07-09 09:20:02 -07:00
|
|
|
src={revealed ? require('@tabler/icons/eye-off.svg') : require('@tabler/icons/eye.svg')}
|
2022-03-21 11:09:01 -07:00
|
|
|
className='h-4 w-4'
|
|
|
|
/>
|
|
|
|
</button>
|
|
|
|
</div>
|
|
|
|
</Tooltip>
|
|
|
|
) : null}
|
|
|
|
</div>
|
|
|
|
);
|
|
|
|
},
|
|
|
|
);
|
|
|
|
|
2022-10-04 14:00:14 -07:00
|
|
|
export {
|
|
|
|
Input as default,
|
|
|
|
InputThemes,
|
|
|
|
};
|