- ChatPanelContext with localStorage persistence (nexus:chat-panel-open) - ChatInput with Enter/Shift+Enter/Escape keyboard shortcuts and auto-resize - ChatMessage renders user bubbles (bg-secondary) and assistant markdown via ChatMarkdownMessage - ChatInput.test.tsx with 6 passing tests (keyboard shortcuts, max-height, submit state)
92 lines
2.7 KiB
TypeScript
92 lines
2.7 KiB
TypeScript
import { useEffect, useRef, useState } from "react";
|
|
import { Send, Loader2 } from "lucide-react";
|
|
import { Button } from "@/components/ui/button";
|
|
import { cn } from "../lib/utils";
|
|
|
|
interface ChatInputProps {
|
|
onSend: (content: string) => void;
|
|
isSubmitting?: boolean;
|
|
disabled?: boolean;
|
|
}
|
|
|
|
export function ChatInput({ onSend, isSubmitting = false, disabled = false }: ChatInputProps) {
|
|
const [value, setValue] = useState("");
|
|
const textareaRef = useRef<HTMLTextAreaElement>(null);
|
|
|
|
// Auto-resize fallback for browsers without field-sizing: content support
|
|
useEffect(() => {
|
|
const el = textareaRef.current;
|
|
if (!el) return;
|
|
el.style.height = "auto";
|
|
el.style.height = `${el.scrollHeight}px`;
|
|
}, [value]);
|
|
|
|
function submit() {
|
|
const trimmed = value.trim();
|
|
if (!trimmed || isSubmitting || disabled) return;
|
|
onSend(trimmed);
|
|
setValue("");
|
|
if (textareaRef.current) {
|
|
textareaRef.current.style.height = "auto";
|
|
}
|
|
}
|
|
|
|
function handleKeyDown(e: React.KeyboardEvent<HTMLTextAreaElement>) {
|
|
if (e.key === "Enter" && !e.shiftKey && !e.nativeEvent.isComposing) {
|
|
e.preventDefault();
|
|
submit();
|
|
} else if (e.key === "Escape") {
|
|
e.preventDefault();
|
|
setValue("");
|
|
if (textareaRef.current) {
|
|
textareaRef.current.style.height = "auto";
|
|
}
|
|
}
|
|
// Shift+Enter falls through to default behavior (inserts newline)
|
|
}
|
|
|
|
const isEmpty = value.trim().length === 0;
|
|
|
|
return (
|
|
<form
|
|
onSubmit={(e) => {
|
|
e.preventDefault();
|
|
submit();
|
|
}}
|
|
className="flex items-end gap-2"
|
|
>
|
|
<textarea
|
|
ref={textareaRef}
|
|
value={value}
|
|
onChange={(e) => setValue(e.target.value)}
|
|
onKeyDown={handleKeyDown}
|
|
placeholder="Message your agent..."
|
|
rows={1}
|
|
disabled={disabled}
|
|
className={cn(
|
|
"flex w-full rounded-md border border-input bg-transparent px-3 py-2 text-sm",
|
|
"placeholder:text-muted-foreground",
|
|
"focus-visible:outline-none focus-visible:ring-1 focus-visible:ring-ring",
|
|
"disabled:cursor-not-allowed disabled:opacity-50",
|
|
"resize-none min-h-[40px] max-h-[160px] overflow-y-auto",
|
|
// field-sizing: content for modern browsers (Chrome 123+, Firefox 129+)
|
|
"[field-sizing:content]",
|
|
)}
|
|
/>
|
|
<Button
|
|
type="submit"
|
|
variant="ghost"
|
|
size="icon"
|
|
disabled={isEmpty || isSubmitting || disabled}
|
|
aria-label="Send message"
|
|
className="shrink-0"
|
|
>
|
|
{isSubmitting ? (
|
|
<Loader2 className="h-4 w-4 animate-spin" />
|
|
) : (
|
|
<Send className="h-4 w-4" />
|
|
)}
|
|
</Button>
|
|
</form>
|
|
);
|
|
}
|