2025-05-14 11:19:06 -06:00
|
|
|
import { ArrowRight, ArrowUp, Square } from 'lucide-react';
|
2024-06-23 10:46:22 +05:30
|
|
|
import { useEffect, useRef, useState } from 'react';
|
2024-04-09 16:21:05 +05:30
|
|
|
import TextareaAutosize from 'react-textarea-autosize';
|
2025-05-05 00:04:00 -06:00
|
|
|
import { File } from './ChatWindow';
|
2024-05-31 11:02:37 +05:30
|
|
|
import Attach from './MessageInputActions/Attach';
|
2025-05-05 00:04:00 -06:00
|
|
|
import Focus from './MessageInputActions/Focus';
|
|
|
|
|
import ModelSelector from './MessageInputActions/ModelSelector';
|
2025-02-16 15:02:05 -07:00
|
|
|
import Optimization from './MessageInputActions/Optimization';
|
2025-05-27 12:53:30 -06:00
|
|
|
import SystemPromptSelector from './MessageInputActions/SystemPromptSelector'; // Import new component
|
2024-04-09 16:21:05 +05:30
|
|
|
|
|
|
|
|
const MessageInput = ({
|
|
|
|
|
sendMessage,
|
2024-04-24 10:06:56 +05:30
|
|
|
loading,
|
2024-11-23 15:04:19 +05:30
|
|
|
fileIds,
|
|
|
|
|
setFileIds,
|
|
|
|
|
files,
|
|
|
|
|
setFiles,
|
2025-02-16 15:02:05 -07:00
|
|
|
optimizationMode,
|
|
|
|
|
setOptimizationMode,
|
2025-05-05 00:04:00 -06:00
|
|
|
focusMode,
|
|
|
|
|
setFocusMode,
|
|
|
|
|
firstMessage,
|
2025-05-14 11:19:06 -06:00
|
|
|
onCancel,
|
2025-05-27 12:53:30 -06:00
|
|
|
systemPromptIds,
|
|
|
|
|
setSystemPromptIds,
|
2024-04-09 16:21:05 +05:30
|
|
|
}: {
|
2025-05-27 12:53:30 -06:00
|
|
|
sendMessage: (
|
|
|
|
|
message: string,
|
|
|
|
|
options?: {
|
|
|
|
|
messageId?: string; // For rewrites/edits
|
|
|
|
|
selectedSystemPromptIds?: string[];
|
|
|
|
|
},
|
|
|
|
|
) => void;
|
2024-04-24 10:06:56 +05:30
|
|
|
loading: boolean;
|
2024-11-23 15:04:19 +05:30
|
|
|
fileIds: string[];
|
|
|
|
|
setFileIds: (fileIds: string[]) => void;
|
|
|
|
|
files: File[];
|
|
|
|
|
setFiles: (files: File[]) => void;
|
2025-02-16 15:02:05 -07:00
|
|
|
optimizationMode: string;
|
|
|
|
|
setOptimizationMode: (mode: string) => void;
|
2025-05-05 00:04:00 -06:00
|
|
|
focusMode: string;
|
|
|
|
|
setFocusMode: (mode: string) => void;
|
|
|
|
|
firstMessage: boolean;
|
2025-05-14 11:19:06 -06:00
|
|
|
onCancel?: () => void;
|
2025-05-27 12:53:30 -06:00
|
|
|
systemPromptIds: string[];
|
|
|
|
|
setSystemPromptIds: (ids: string[]) => void;
|
2024-04-09 16:21:05 +05:30
|
|
|
}) => {
|
|
|
|
|
const [message, setMessage] = useState('');
|
2025-05-05 00:04:00 -06:00
|
|
|
const [selectedModel, setSelectedModel] = useState<{
|
|
|
|
|
provider: string;
|
|
|
|
|
model: string;
|
|
|
|
|
} | null>(null);
|
2024-04-09 16:21:05 +05:30
|
|
|
|
|
|
|
|
useEffect(() => {
|
2025-05-05 00:04:00 -06:00
|
|
|
// Load saved model preferences from localStorage
|
|
|
|
|
const chatModelProvider = localStorage.getItem('chatModelProvider');
|
|
|
|
|
const chatModel = localStorage.getItem('chatModel');
|
|
|
|
|
|
|
|
|
|
if (chatModelProvider && chatModel) {
|
|
|
|
|
setSelectedModel({
|
|
|
|
|
provider: chatModelProvider,
|
|
|
|
|
model: chatModel,
|
|
|
|
|
});
|
2024-04-09 16:21:05 +05:30
|
|
|
}
|
2025-05-27 12:53:30 -06:00
|
|
|
|
|
|
|
|
const storedPromptIds = localStorage.getItem('selectedSystemPromptIds');
|
|
|
|
|
if (storedPromptIds) {
|
|
|
|
|
try {
|
|
|
|
|
const parsedIds = JSON.parse(storedPromptIds);
|
|
|
|
|
if (Array.isArray(parsedIds)) {
|
|
|
|
|
setSystemPromptIds(parsedIds);
|
|
|
|
|
}
|
|
|
|
|
} catch (e) {
|
|
|
|
|
console.error(
|
|
|
|
|
'Failed to parse selectedSystemPromptIds from localStorage',
|
|
|
|
|
e,
|
|
|
|
|
);
|
|
|
|
|
localStorage.removeItem('selectedSystemPromptIds'); // Clear corrupted data
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}, [setSystemPromptIds]);
|
|
|
|
|
|
|
|
|
|
useEffect(() => {
|
|
|
|
|
if (systemPromptIds.length > 0) {
|
|
|
|
|
localStorage.setItem(
|
|
|
|
|
'selectedSystemPromptIds',
|
|
|
|
|
JSON.stringify(systemPromptIds),
|
|
|
|
|
);
|
|
|
|
|
} else {
|
|
|
|
|
// Remove from localStorage if no prompts are selected to keep it clean
|
|
|
|
|
localStorage.removeItem('selectedSystemPromptIds');
|
|
|
|
|
}
|
|
|
|
|
}, [systemPromptIds]);
|
2024-04-09 16:21:05 +05:30
|
|
|
|
2024-06-23 10:46:22 +05:30
|
|
|
const inputRef = useRef<HTMLTextAreaElement | null>(null);
|
|
|
|
|
|
|
|
|
|
useEffect(() => {
|
2024-09-02 11:44:40 +05:30
|
|
|
const handleKeyDown = (e: KeyboardEvent) => {
|
|
|
|
|
const activeElement = document.activeElement;
|
|
|
|
|
const isInputFocused =
|
|
|
|
|
activeElement?.tagName === 'INPUT' ||
|
|
|
|
|
activeElement?.tagName === 'TEXTAREA' ||
|
|
|
|
|
activeElement?.hasAttribute('contenteditable');
|
|
|
|
|
if (e.key === '/' && !isInputFocused) {
|
|
|
|
|
e.preventDefault();
|
|
|
|
|
inputRef.current?.focus();
|
|
|
|
|
}
|
|
|
|
|
};
|
2024-06-23 10:46:22 +05:30
|
|
|
document.addEventListener('keydown', handleKeyDown);
|
|
|
|
|
return () => {
|
|
|
|
|
document.removeEventListener('keydown', handleKeyDown);
|
|
|
|
|
};
|
|
|
|
|
}, []);
|
|
|
|
|
|
2025-05-05 00:04:00 -06:00
|
|
|
// Function to handle message submission
|
|
|
|
|
const handleSubmitMessage = () => {
|
|
|
|
|
// Only submit if we have a non-empty message and not currently loading
|
|
|
|
|
if (loading || message.trim().length === 0) return;
|
|
|
|
|
|
|
|
|
|
// Make sure the selected model is used when sending a message
|
|
|
|
|
if (selectedModel) {
|
|
|
|
|
localStorage.setItem('chatModelProvider', selectedModel.provider);
|
|
|
|
|
localStorage.setItem('chatModel', selectedModel.model);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
sendMessage(message);
|
|
|
|
|
setMessage('');
|
|
|
|
|
};
|
|
|
|
|
|
2025-05-07 22:12:06 -06:00
|
|
|
return (
|
2024-04-09 16:21:05 +05:30
|
|
|
<form
|
|
|
|
|
onSubmit={(e) => {
|
|
|
|
|
e.preventDefault();
|
2025-05-05 00:04:00 -06:00
|
|
|
handleSubmitMessage();
|
2024-04-09 16:21:05 +05:30
|
|
|
}}
|
|
|
|
|
onKeyDown={(e) => {
|
2025-05-05 00:04:00 -06:00
|
|
|
if (e.key === 'Enter' && !e.shiftKey) {
|
2024-04-09 16:21:05 +05:30
|
|
|
e.preventDefault();
|
2025-05-05 00:04:00 -06:00
|
|
|
handleSubmitMessage();
|
2024-04-09 16:21:05 +05:30
|
|
|
}
|
|
|
|
|
}}
|
2025-05-05 00:04:00 -06:00
|
|
|
className="w-full"
|
2024-04-09 16:21:05 +05:30
|
|
|
>
|
2025-05-05 00:04:00 -06:00
|
|
|
<div className="flex flex-col bg-light-secondary dark:bg-dark-secondary px-5 pt-5 pb-2 rounded-lg w-full border border-light-200 dark:border-dark-200">
|
|
|
|
|
<TextareaAutosize
|
|
|
|
|
ref={inputRef}
|
|
|
|
|
value={message}
|
|
|
|
|
onChange={(e) => setMessage(e.target.value)}
|
|
|
|
|
minRows={2}
|
|
|
|
|
className="bg-transparent placeholder:text-black/50 dark:placeholder:text-white/50 text-sm text-black dark:text-white resize-none focus:outline-none w-full max-h-24 lg:max-h-36 xl:max-h-48"
|
2025-05-07 22:12:06 -06:00
|
|
|
placeholder={firstMessage ? 'Ask anything...' : 'Ask a follow-up'}
|
2025-05-05 00:04:00 -06:00
|
|
|
/>
|
|
|
|
|
<div className="flex flex-row items-center justify-between mt-4">
|
2025-05-27 12:53:30 -06:00
|
|
|
<div className="flex flex-row items-center space-x-2">
|
2025-05-05 00:04:00 -06:00
|
|
|
<Focus focusMode={focusMode} setFocusMode={setFocusMode} />
|
|
|
|
|
<Attach
|
2025-02-16 15:02:05 -07:00
|
|
|
fileIds={fileIds}
|
|
|
|
|
setFileIds={setFileIds}
|
|
|
|
|
files={files}
|
|
|
|
|
setFiles={setFiles}
|
2025-06-28 17:59:12 -06:00
|
|
|
optimizationMode={optimizationMode}
|
2025-05-05 00:04:00 -06:00
|
|
|
/>
|
2025-05-27 12:53:30 -06:00
|
|
|
</div>
|
|
|
|
|
<div className="flex flex-row items-center space-x-2">
|
2025-05-05 00:04:00 -06:00
|
|
|
<ModelSelector
|
|
|
|
|
selectedModel={selectedModel}
|
|
|
|
|
setSelectedModel={setSelectedModel}
|
2025-02-16 15:02:05 -07:00
|
|
|
/>
|
2025-05-27 12:53:30 -06:00
|
|
|
<SystemPromptSelector
|
|
|
|
|
selectedPromptIds={systemPromptIds}
|
|
|
|
|
onSelectedPromptIdsChange={setSystemPromptIds}
|
|
|
|
|
/>
|
2025-02-16 15:02:05 -07:00
|
|
|
<Optimization
|
|
|
|
|
optimizationMode={optimizationMode}
|
|
|
|
|
setOptimizationMode={setOptimizationMode}
|
|
|
|
|
/>
|
2025-05-14 11:19:06 -06:00
|
|
|
{loading ? (
|
|
|
|
|
<button
|
|
|
|
|
type="button"
|
|
|
|
|
className="bg-red-700 text-white hover:bg-red-800 transition duration-100 rounded-full p-2 relative group"
|
|
|
|
|
onClick={onCancel}
|
|
|
|
|
aria-label="Cancel"
|
|
|
|
|
>
|
|
|
|
|
{loading && (
|
|
|
|
|
<div className="absolute inset-0 rounded-full border-2 border-white/30 border-t-white animate-spin" />
|
|
|
|
|
)}
|
|
|
|
|
<span className="relative flex items-center justify-center w-[17px] h-[17px]">
|
|
|
|
|
<Square size={17} className="text-white" />
|
|
|
|
|
</span>
|
|
|
|
|
</button>
|
|
|
|
|
) : (
|
|
|
|
|
<button
|
|
|
|
|
disabled={message.trim().length === 0}
|
|
|
|
|
className="bg-[#24A0ED] text-white disabled:text-black/50 dark:disabled:text-white/50 disabled:bg-[#e0e0dc] dark:disabled:bg-[#ececec21] hover:bg-opacity-85 transition duration-100 rounded-full p-2"
|
|
|
|
|
type="submit"
|
|
|
|
|
>
|
|
|
|
|
{firstMessage ? (
|
|
|
|
|
<ArrowRight className="bg-background" size={17} />
|
|
|
|
|
) : (
|
|
|
|
|
<ArrowUp className="bg-background" size={17} />
|
|
|
|
|
)}
|
|
|
|
|
</button>
|
|
|
|
|
)}
|
2024-04-09 16:21:05 +05:30
|
|
|
</div>
|
|
|
|
|
</div>
|
2025-05-05 00:04:00 -06:00
|
|
|
</div>
|
2024-04-09 16:21:05 +05:30
|
|
|
</form>
|
|
|
|
|
);
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
export default MessageInput;
|