Trigger /<name>, dropdown lists all skills filtered by name prefix, arg passthrough sends the rest as the user message. Synthetic skill_use tool_use renders identically to model-invoked skills.
204 lines
7.3 KiB
TypeScript
204 lines
7.3 KiB
TypeScript
import { useCallback, useEffect, useRef, useState } from 'react';
|
|
import { ChevronDown, Square, X } from 'lucide-react';
|
|
import { toast } from 'sonner';
|
|
import { api } from '@/api/client';
|
|
import { useSessionStream } from '@/hooks/useSessionStream';
|
|
import { useChatContextStats } from '@/hooks/useChatContextStats';
|
|
import { MessageList } from '@/components/MessageList';
|
|
import { ChatInput } from '@/components/ChatInput';
|
|
import { ChatContextPopover } from '@/components/ChatContextPopover';
|
|
import {
|
|
DropdownMenu,
|
|
DropdownMenuContent,
|
|
DropdownMenuItem,
|
|
DropdownMenuTrigger,
|
|
} from '@/components/ui/dropdown-menu';
|
|
|
|
interface Props {
|
|
sessionId: string;
|
|
chatId: string;
|
|
projectId: string;
|
|
// Batch 9: optional, threaded down to ChatInput's agent picker.
|
|
agentId?: string | null;
|
|
onAgentChange?: (agentId: string | null) => void | Promise<void>;
|
|
sessionChats?: import('@/api/types').Chat[];
|
|
// v1.9: threaded down to ChatInput's + menu (Web search quick toggle).
|
|
// null means "inherit project default" — ChatInput PATCHes with the
|
|
// opposite of the effective value.
|
|
webSearchEnabled?: boolean | null;
|
|
}
|
|
|
|
export function ChatPane({ sessionId, chatId, projectId, agentId, onAgentChange, sessionChats, webSearchEnabled }: Props) {
|
|
const stream = useSessionStream(sessionId);
|
|
const lastErrorRef = useRef<string | null>(null);
|
|
const [queue, setQueue] = useState<string[]>([]);
|
|
const processingRef = useRef(false);
|
|
|
|
useEffect(() => {
|
|
if (stream.error && stream.error !== lastErrorRef.current) {
|
|
lastErrorRef.current = stream.error;
|
|
toast.error(stream.error);
|
|
}
|
|
if (!stream.error) {
|
|
lastErrorRef.current = null;
|
|
}
|
|
}, [stream.error]);
|
|
|
|
const chatMessages = stream.messages.filter((m) => m.chat_id === chatId);
|
|
const streaming = chatMessages.some((m) => m.status === 'streaming');
|
|
const contextStats = useChatContextStats(chatId, chatMessages);
|
|
|
|
// Auto-send next queued message when streaming completes
|
|
useEffect(() => {
|
|
if (streaming || queue.length === 0 || processingRef.current) return;
|
|
processingRef.current = true;
|
|
const next = queue[0]!;
|
|
setQueue((prev) => prev.slice(1));
|
|
api.messages.send(chatId, next)
|
|
.catch((err) => toast.error(err instanceof Error ? err.message : 'queue send failed'))
|
|
.finally(() => { processingRef.current = false; });
|
|
}, [streaming, queue, chatId]);
|
|
|
|
const handleSend = useCallback(async (content: string) => {
|
|
const trimmed = content.trim();
|
|
if (!trimmed) return;
|
|
if (trimmed === '/compact') {
|
|
try {
|
|
await api.chats.compact(chatId);
|
|
} catch (err) {
|
|
toast.error(err instanceof Error ? err.message : 'compact failed');
|
|
}
|
|
return;
|
|
}
|
|
if (streaming) {
|
|
setQueue((prev) => [...prev, trimmed]);
|
|
return;
|
|
}
|
|
await api.messages.send(chatId, trimmed);
|
|
}, [chatId, streaming]);
|
|
|
|
async function handleStop() {
|
|
try {
|
|
await api.chats.stop(chatId);
|
|
} catch (err) {
|
|
toast.error(err instanceof Error ? err.message : 'stop failed');
|
|
}
|
|
}
|
|
|
|
const handleForceSend = useCallback(async (content: string) => {
|
|
const trimmed = content.trim();
|
|
if (!trimmed) return;
|
|
try {
|
|
await api.chats.forceSend(chatId, trimmed);
|
|
setQueue([]);
|
|
} catch (err) {
|
|
toast.error(err instanceof Error ? err.message : 'force send failed');
|
|
}
|
|
}, [chatId]);
|
|
|
|
// Batch 9.6: slash-command dispatch. Sent regardless of streaming state —
|
|
// matches the existing /compact precedent (which also fires immediately).
|
|
// Empty args go to the server as null; the server fills in a default user
|
|
// message ("Apply this skill.") so the model has something to act on.
|
|
const handleSlashCommand = useCallback(async (skillName: string, userMessage: string) => {
|
|
try {
|
|
await api.chats.skillInvoke(chatId, skillName, userMessage.length > 0 ? userMessage : null);
|
|
} catch (err) {
|
|
toast.error(err instanceof Error ? err.message : `/${skillName} failed`);
|
|
}
|
|
}, [chatId]);
|
|
|
|
function removeQueued(idx: number) {
|
|
setQueue((prev) => prev.filter((_, i) => i !== idx));
|
|
}
|
|
|
|
async function forceSendQueued(idx: number) {
|
|
const msg = queue[idx];
|
|
if (!msg) return;
|
|
setQueue((prev) => prev.filter((_, i) => i !== idx));
|
|
try {
|
|
await api.chats.forceSend(chatId, msg);
|
|
} catch (err) {
|
|
toast.error(err instanceof Error ? err.message : 'force send failed');
|
|
}
|
|
}
|
|
|
|
return (
|
|
<div className="flex flex-col h-full min-h-0">
|
|
<MessageList messages={chatMessages} sessionChats={sessionChats} />
|
|
|
|
{/* Queued messages */}
|
|
{queue.length > 0 && (
|
|
<div className="border-t">
|
|
<div className="max-w-[1000px] mx-auto w-full px-4 py-1 space-y-1">
|
|
{queue.map((msg, i) => (
|
|
<div key={i} className="flex items-center gap-2 text-xs text-muted-foreground bg-muted/30 rounded px-2 py-1">
|
|
<span className="font-medium shrink-0">Queued:</span>
|
|
<span className="truncate flex-1">{msg}</span>
|
|
<DropdownMenu>
|
|
<DropdownMenuTrigger asChild>
|
|
<button
|
|
type="button"
|
|
className="inline-flex items-center justify-center p-0.5 hover:bg-muted rounded shrink-0 max-md:min-h-[44px] max-md:min-w-[44px]"
|
|
aria-label="Queued message options"
|
|
>
|
|
<ChevronDown size={12} />
|
|
</button>
|
|
</DropdownMenuTrigger>
|
|
<DropdownMenuContent align="end">
|
|
<DropdownMenuItem onSelect={() => { /* default: queued, nothing to do */ }}>
|
|
Send when done
|
|
</DropdownMenuItem>
|
|
<DropdownMenuItem onSelect={() => void forceSendQueued(i)}>
|
|
Force send now
|
|
</DropdownMenuItem>
|
|
</DropdownMenuContent>
|
|
</DropdownMenu>
|
|
<button
|
|
type="button"
|
|
onClick={() => removeQueued(i)}
|
|
className="inline-flex items-center justify-center p-0.5 hover:bg-muted rounded shrink-0 max-md:min-h-[44px] max-md:min-w-[44px]"
|
|
aria-label="Cancel queued message"
|
|
>
|
|
<X size={12} />
|
|
</button>
|
|
</div>
|
|
))}
|
|
</div>
|
|
</div>
|
|
)}
|
|
|
|
{/* Stop button when streaming */}
|
|
{streaming && (
|
|
<div className="border-t py-1">
|
|
<div className="max-w-[1000px] mx-auto w-full flex justify-center">
|
|
<button
|
|
type="button"
|
|
onClick={() => void handleStop()}
|
|
className="flex items-center gap-1.5 text-xs px-3 py-1 rounded-full border hover:bg-muted text-muted-foreground hover:text-foreground max-md:min-h-[44px] max-md:px-5"
|
|
>
|
|
<Square size={10} className="fill-current" />
|
|
Stop generating
|
|
</button>
|
|
</div>
|
|
</div>
|
|
)}
|
|
|
|
<div className="relative">
|
|
<ChatContextPopover stats={contextStats} />
|
|
<ChatInput
|
|
disabled={false}
|
|
projectId={projectId}
|
|
sessionId={sessionId}
|
|
agentId={agentId}
|
|
onAgentChange={onAgentChange}
|
|
webSearchEnabled={webSearchEnabled}
|
|
onSend={handleSend}
|
|
onForceSend={streaming ? handleForceSend : undefined}
|
|
onSlashCommand={handleSlashCommand}
|
|
/>
|
|
</div>
|
|
</div>
|
|
);
|
|
}
|