mirror of
https://github.com/toeverything/AFFiNE.git
synced 2026-02-12 04:18:54 +00:00
124 lines
3.0 KiB
TypeScript
124 lines
3.0 KiB
TypeScript
import { assignInlineVars } from '@vanilla-extract/dynamic';
|
|
import clsx from 'clsx';
|
|
import type {
|
|
ChangeEvent,
|
|
CSSProperties,
|
|
FocusEvent,
|
|
FocusEventHandler,
|
|
ForwardedRef,
|
|
InputHTMLAttributes,
|
|
KeyboardEvent,
|
|
KeyboardEventHandler,
|
|
ReactNode,
|
|
} from 'react';
|
|
import { forwardRef, useCallback, useState } from 'react';
|
|
|
|
import { input, inputWrapper, widthVar } from './style.css';
|
|
|
|
export type InputProps = {
|
|
disabled?: boolean;
|
|
width?: CSSProperties['width'];
|
|
onChange?: (value: string) => void;
|
|
onBlur?: FocusEventHandler<HTMLInputElement>;
|
|
onKeyDown?: KeyboardEventHandler<HTMLInputElement>;
|
|
noBorder?: boolean;
|
|
status?: 'error' | 'success' | 'warning' | 'default';
|
|
size?: 'default' | 'large' | 'extraLarge';
|
|
preFix?: ReactNode;
|
|
endFix?: ReactNode;
|
|
type?: HTMLInputElement['type'];
|
|
inputStyle?: CSSProperties;
|
|
onEnter?: () => void;
|
|
} & Omit<InputHTMLAttributes<HTMLInputElement>, 'onChange' | 'size'>;
|
|
|
|
export const Input = forwardRef<HTMLInputElement, InputProps>(function Input(
|
|
{
|
|
disabled,
|
|
width,
|
|
onChange: propsOnChange,
|
|
noBorder = false,
|
|
className,
|
|
status = 'default',
|
|
style = {},
|
|
inputStyle = {},
|
|
size = 'default',
|
|
onFocus,
|
|
onBlur,
|
|
preFix,
|
|
endFix,
|
|
onEnter,
|
|
onKeyDown,
|
|
...otherProps
|
|
}: InputProps,
|
|
ref: ForwardedRef<HTMLInputElement>
|
|
) {
|
|
const [isFocus, setIsFocus] = useState(false);
|
|
|
|
return (
|
|
<div
|
|
className={clsx(inputWrapper, className, {
|
|
// status
|
|
disabled: disabled,
|
|
'no-border': noBorder,
|
|
focus: isFocus,
|
|
// color
|
|
error: status === 'error',
|
|
success: status === 'success',
|
|
warning: status === 'warning',
|
|
default: status === 'default',
|
|
// size
|
|
large: size === 'large',
|
|
'extra-large': size === 'extraLarge',
|
|
})}
|
|
style={{
|
|
...assignInlineVars({
|
|
[widthVar]: width ? `${width}px` : '100%',
|
|
}),
|
|
...style,
|
|
}}
|
|
>
|
|
{preFix}
|
|
<input
|
|
className={clsx(input, {
|
|
large: size === 'large',
|
|
'extra-large': size === 'extraLarge',
|
|
})}
|
|
ref={ref}
|
|
disabled={disabled}
|
|
style={inputStyle}
|
|
onFocus={useCallback(
|
|
(e: FocusEvent<HTMLInputElement>) => {
|
|
setIsFocus(true);
|
|
onFocus?.(e);
|
|
},
|
|
[onFocus]
|
|
)}
|
|
onBlur={useCallback(
|
|
(e: FocusEvent<HTMLInputElement>) => {
|
|
setIsFocus(false);
|
|
onBlur?.(e);
|
|
},
|
|
[onBlur]
|
|
)}
|
|
onChange={useCallback(
|
|
(e: ChangeEvent<HTMLInputElement>) => {
|
|
propsOnChange?.(e.target.value);
|
|
},
|
|
[propsOnChange]
|
|
)}
|
|
onKeyDown={useCallback(
|
|
(e: KeyboardEvent<HTMLInputElement>) => {
|
|
if (e.key === 'Enter') {
|
|
onEnter?.();
|
|
}
|
|
onKeyDown?.(e);
|
|
},
|
|
[onKeyDown, onEnter]
|
|
)}
|
|
{...otherProps}
|
|
/>
|
|
{endFix}
|
|
</div>
|
|
);
|
|
});
|