import React from "react"; import TextareaAutosize from "react-textarea-autosize"; import { useTranslation } from "react-i18next"; import { I18nKey } from "#/i18n/declaration"; import { cn } from "#/utils/utils"; import { SubmitButton } from "#/components/shared/buttons/submit-button"; import { StopButton } from "#/components/shared/buttons/stop-button"; interface ChatInputProps { name?: string; button?: "submit" | "stop"; disabled?: boolean; showButton?: boolean; value?: string; maxRows?: number; onSubmit: (message: string) => void; onStop?: () => void; onChange?: (message: string) => void; onFocus?: () => void; onBlur?: () => void; onImagePaste?: (files: File[]) => void; className?: React.HTMLAttributes["className"]; buttonClassName?: React.HTMLAttributes["className"]; } export function ChatInput({ name, button = "submit", disabled, showButton = true, value, maxRows = 4, onSubmit, onStop, onChange, onFocus, onBlur, onImagePaste, className, buttonClassName, }: ChatInputProps) { const { t } = useTranslation(); const textareaRef = React.useRef(null); const [isDraggingOver, setIsDraggingOver] = React.useState(false); const handlePaste = (event: React.ClipboardEvent) => { // Only handle paste if we have an image paste handler and there are files if (onImagePaste && event.clipboardData.files.length > 0) { const files = Array.from(event.clipboardData.files).filter((file) => file.type.startsWith("image/"), ); // Only prevent default if we found image files to handle if (files.length > 0) { event.preventDefault(); onImagePaste(files); } } // For text paste, let the default behavior handle it }; const handleDragOver = (event: React.DragEvent) => { event.preventDefault(); if (event.dataTransfer.types.includes("Files")) { setIsDraggingOver(true); } }; const handleDragLeave = (event: React.DragEvent) => { event.preventDefault(); setIsDraggingOver(false); }; const handleDrop = (event: React.DragEvent) => { event.preventDefault(); setIsDraggingOver(false); if (onImagePaste && event.dataTransfer.files.length > 0) { const files = Array.from(event.dataTransfer.files).filter((file) => file.type.startsWith("image/"), ); if (files.length > 0) { onImagePaste(files); } } }; const handleSubmitMessage = () => { const message = value || textareaRef.current?.value || ""; if (message.trim()) { onSubmit(message); onChange?.(""); if (textareaRef.current) { textareaRef.current.value = ""; } } }; const handleKeyPress = (event: React.KeyboardEvent) => { if ( event.key === "Enter" && !event.shiftKey && !disabled && !event.nativeEvent.isComposing ) { event.preventDefault(); handleSubmitMessage(); } }; const handleChange = (event: React.ChangeEvent) => { onChange?.(event.target.value); }; return (
{showButton && (
{button === "submit" && ( )} {button === "stop" && ( )}
)}
); }