'use client'; import { DefaultChatTransport } from 'ai'; import { useChat } from '@ai-sdk/react'; import { useEffect, useState } from 'react'; import useSWR, { useSWRConfig } from 'swr'; import { ChatHeader } from '@/components/chat-header'; import type { Vote } from '@/lib/db/schema'; import { fetcher, fetchWithErrorHandlers, generateUUID } from '@/lib/utils'; import { Artifact } from './artifact'; import { MultimodalInput } from './multimodal-input'; import { Messages } from './messages'; import type { VisibilityType } from './visibility-selector'; import { useArtifactSelector } from '@/hooks/use-artifact'; import { unstable_serialize } from 'swr/infinite'; import { getChatHistoryPaginationKey } from './sidebar-history'; import { toast } from './toast'; import type { Session } from 'next-auth'; import { useSearchParams } from 'next/navigation'; import { useChatVisibility } from '@/hooks/use-chat-visibility'; import { useAutoResume } from '@/hooks/use-auto-resume'; import { ChatSDKError } from '@/lib/errors'; import type { Attachment, ChatMessage } from '@/lib/types'; import { useDataStream } from './data-stream-provider'; export function Chat({ id, initialMessages, initialChatModel, initialVisibilityType, isReadonly, session, autoResume, }: { id: string; initialMessages: ChatMessage[]; initialChatModel: string; initialVisibilityType: VisibilityType; isReadonly: boolean; session: Session; autoResume: boolean; }) { const { visibilityType } = useChatVisibility({ chatId: id, initialVisibilityType, }); const { mutate } = useSWRConfig(); const { setDataStream } = useDataStream(); const [input, setInput] = useState(''); const { messages, setMessages, sendMessage, status, stop, regenerate, resumeStream, } = useChat({ id, messages: initialMessages, experimental_throttle: 100, generateId: generateUUID, transport: new DefaultChatTransport({ api: '/api/chat', fetch: fetchWithErrorHandlers, prepareSendMessagesRequest({ messages, id, body }) { return { body: { id, message: messages.at(-1), selectedChatModel: initialChatModel, selectedVisibilityType: visibilityType, ...body, }, }; }, }), onData: (dataPart) => { setDataStream((ds) => (ds ? [...ds, dataPart] : [])); }, onFinish: () => { mutate(unstable_serialize(getChatHistoryPaginationKey)); }, onError: (error) => { if (error instanceof ChatSDKError) { toast({ type: 'error', description: error.message, }); } }, }); const searchParams = useSearchParams(); const query = searchParams.get('query'); const [hasAppendedQuery, setHasAppendedQuery] = useState(false); useEffect(() => { if (query && !hasAppendedQuery) { sendMessage({ role: 'user' as const, parts: [{ type: 'text', text: query }], }); setHasAppendedQuery(true); window.history.replaceState({}, '', `/chat/${id}`); } }, [query, sendMessage, hasAppendedQuery, id]); const { data: votes } = useSWR>( messages.length >= 2 ? `/api/vote?chatId=${id}` : null, fetcher, ); const [attachments, setAttachments] = useState>([]); const isArtifactVisible = useArtifactSelector((state) => state.isVisible); useAutoResume({ autoResume, initialMessages, resumeStream, setMessages, }); return ( <>
{!isReadonly && (
)}
); }