2024-08-23 09:18:49 +02:00
|
|
|
import { clsx } from "clsx";
|
|
|
|
import React, { forwardRef, useEffect, useRef, useState } from "react";
|
|
|
|
|
|
|
|
import "./input.less";
|
|
|
|
|
|
|
|
interface InputDecorationProps {
|
|
|
|
startDecoration?: React.ReactNode;
|
|
|
|
endDecoration?: React.ReactNode;
|
|
|
|
}
|
|
|
|
|
|
|
|
interface InputProps {
|
|
|
|
label?: string;
|
|
|
|
value?: string;
|
|
|
|
className?: string;
|
|
|
|
onChange?: (value: string) => void;
|
|
|
|
onKeyDown?: (event: React.KeyboardEvent<HTMLInputElement>) => void;
|
|
|
|
onFocus?: () => void;
|
|
|
|
onBlur?: () => void;
|
|
|
|
placeholder?: string;
|
|
|
|
defaultValue?: string;
|
|
|
|
decoration?: InputDecorationProps;
|
|
|
|
required?: boolean;
|
|
|
|
maxLength?: number;
|
|
|
|
autoFocus?: boolean;
|
|
|
|
disabled?: boolean;
|
|
|
|
isNumber?: boolean;
|
2024-09-03 01:48:10 +02:00
|
|
|
inputRef?: React.MutableRefObject<HTMLInputElement>;
|
2024-08-23 09:18:49 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
const Input = forwardRef<HTMLDivElement, InputProps>(
|
|
|
|
(
|
|
|
|
{
|
|
|
|
label,
|
|
|
|
value,
|
|
|
|
className,
|
|
|
|
onChange,
|
|
|
|
onKeyDown,
|
|
|
|
onFocus,
|
|
|
|
onBlur,
|
|
|
|
placeholder,
|
|
|
|
defaultValue = "",
|
|
|
|
decoration,
|
|
|
|
required,
|
|
|
|
maxLength,
|
|
|
|
autoFocus,
|
|
|
|
disabled,
|
|
|
|
isNumber,
|
2024-09-03 01:48:10 +02:00
|
|
|
inputRef,
|
2024-08-23 09:18:49 +02:00
|
|
|
}: InputProps,
|
|
|
|
ref
|
|
|
|
) => {
|
|
|
|
const [focused, setFocused] = useState(false);
|
|
|
|
const [internalValue, setInternalValue] = useState(defaultValue);
|
|
|
|
const [error, setError] = useState(false);
|
|
|
|
const [hasContent, setHasContent] = useState(Boolean(value || defaultValue));
|
2024-09-03 01:48:10 +02:00
|
|
|
const internalInputRef = useRef<HTMLInputElement>(null);
|
2024-08-23 09:18:49 +02:00
|
|
|
|
|
|
|
useEffect(() => {
|
|
|
|
if (value !== undefined) {
|
|
|
|
setFocused(Boolean(value));
|
|
|
|
}
|
|
|
|
}, [value]);
|
|
|
|
|
|
|
|
const handleComponentFocus = () => {
|
2024-09-03 01:48:10 +02:00
|
|
|
if (internalInputRef.current && !internalInputRef.current.contains(document.activeElement)) {
|
|
|
|
internalInputRef.current.focus();
|
2024-08-23 09:18:49 +02:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
const handleComponentBlur = () => {
|
2024-09-03 01:48:10 +02:00
|
|
|
if (internalInputRef.current?.contains(document.activeElement)) {
|
|
|
|
internalInputRef.current.blur();
|
2024-08-23 09:18:49 +02:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2024-09-03 01:48:10 +02:00
|
|
|
const handleSetInputRef = (elem: HTMLInputElement) => {
|
|
|
|
if (inputRef) {
|
|
|
|
inputRef.current = elem;
|
|
|
|
}
|
|
|
|
internalInputRef.current = elem;
|
|
|
|
};
|
|
|
|
|
2024-08-23 09:18:49 +02:00
|
|
|
const handleFocus = () => {
|
|
|
|
setFocused(true);
|
|
|
|
onFocus && onFocus();
|
|
|
|
};
|
|
|
|
|
|
|
|
const handleBlur = () => {
|
2024-09-03 01:48:10 +02:00
|
|
|
if (internalInputRef.current) {
|
|
|
|
const inputValue = internalInputRef.current.value;
|
2024-08-23 09:18:49 +02:00
|
|
|
if (required && !inputValue) {
|
|
|
|
setError(true);
|
|
|
|
setFocused(false);
|
|
|
|
} else {
|
|
|
|
setError(false);
|
|
|
|
setFocused(false);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
onBlur && onBlur();
|
|
|
|
};
|
|
|
|
|
|
|
|
const handleInputChange = (e: React.ChangeEvent<HTMLInputElement>) => {
|
|
|
|
const inputValue = e.target.value;
|
|
|
|
|
|
|
|
if (isNumber && inputValue !== "" && !/^\d*$/.test(inputValue)) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (required && !inputValue) {
|
|
|
|
setError(true);
|
|
|
|
setHasContent(false);
|
|
|
|
} else {
|
|
|
|
setError(false);
|
|
|
|
setHasContent(Boolean(inputValue));
|
|
|
|
}
|
|
|
|
|
|
|
|
if (value === undefined) {
|
|
|
|
setInternalValue(inputValue);
|
|
|
|
}
|
|
|
|
|
|
|
|
onChange && onChange(inputValue);
|
|
|
|
};
|
|
|
|
|
|
|
|
const inputValue = value ?? internalValue;
|
|
|
|
|
|
|
|
return (
|
|
|
|
<div
|
|
|
|
ref={ref}
|
|
|
|
className={clsx("input", className, {
|
|
|
|
focused: focused,
|
|
|
|
error: error,
|
|
|
|
disabled: disabled,
|
|
|
|
"no-label": !label,
|
|
|
|
})}
|
|
|
|
onFocus={handleComponentFocus}
|
|
|
|
onBlur={handleComponentBlur}
|
|
|
|
tabIndex={-1}
|
|
|
|
>
|
|
|
|
{decoration?.startDecoration && <>{decoration.startDecoration}</>}
|
|
|
|
<div className="input-inner">
|
|
|
|
{label && (
|
|
|
|
<label
|
|
|
|
className={clsx("input-inner-label", {
|
|
|
|
float: hasContent || focused || placeholder,
|
|
|
|
"offset-left": decoration?.startDecoration,
|
|
|
|
})}
|
|
|
|
htmlFor={label}
|
|
|
|
>
|
|
|
|
{label}
|
|
|
|
</label>
|
|
|
|
)}
|
|
|
|
<input
|
|
|
|
className={clsx("input-inner-input", {
|
|
|
|
"offset-left": decoration?.startDecoration,
|
|
|
|
})}
|
2024-09-03 01:48:10 +02:00
|
|
|
ref={handleSetInputRef}
|
2024-08-23 09:18:49 +02:00
|
|
|
id={label}
|
|
|
|
value={inputValue}
|
|
|
|
onChange={handleInputChange}
|
|
|
|
onFocus={handleFocus}
|
|
|
|
onBlur={handleBlur}
|
|
|
|
onKeyDown={onKeyDown}
|
|
|
|
placeholder={placeholder}
|
|
|
|
maxLength={maxLength}
|
|
|
|
autoFocus={autoFocus}
|
|
|
|
disabled={disabled}
|
|
|
|
/>
|
|
|
|
</div>
|
|
|
|
{decoration?.endDecoration && <>{decoration.endDecoration}</>}
|
|
|
|
</div>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
);
|
|
|
|
|
|
|
|
export { Input };
|
|
|
|
export type { InputDecorationProps, InputProps };
|