next-chat / components /message.tsx
NeoPy's picture
Upload folder using huggingface_hub
867b17d verified
raw
history blame
14 kB
'use client';
import cx from 'classnames';
import { AnimatePresence, motion } from 'framer-motion';
import { memo, useState } from 'react';
import type { Vote } from '@/lib/db/schema';
import { DocumentToolResult } from './document';
import { PencilEditIcon, SparklesIcon } from './icons';
import { Response } from './elements/response';
import { MessageContent } from './elements/message';
import {
Tool,
ToolHeader,
ToolContent,
ToolInput,
ToolOutput,
} from './elements/tool';
import { MessageActions } from './message-actions';
import { PreviewAttachment } from './preview-attachment';
import { Weather } from './weather';
import equal from 'fast-deep-equal';
import { cn, sanitizeText } from '@/lib/utils';
import { Button } from './ui/button';
import { Tooltip, TooltipContent, TooltipTrigger } from './ui/tooltip';
import { MessageEditor } from './message-editor';
import { DocumentPreview } from './document-preview';
import { MessageReasoning } from './message-reasoning';
import type { UseChatHelpers } from '@ai-sdk/react';
import type { ChatMessage } from '@/lib/types';
import { useDataStream } from './data-stream-provider';
// Type narrowing is handled by TypeScript's control flow analysis
// The AI SDK provides proper discriminated unions for tool calls
const PurePreviewMessage = ({
chatId,
message,
vote,
isLoading,
setMessages,
regenerate,
isReadonly,
requiresScrollPadding,
}: {
chatId: string;
message: ChatMessage;
vote: Vote | undefined;
isLoading: boolean;
setMessages: UseChatHelpers<ChatMessage>['setMessages'];
regenerate: UseChatHelpers<ChatMessage>['regenerate'];
isReadonly: boolean;
requiresScrollPadding: boolean;
}) => {
const [mode, setMode] = useState<'view' | 'edit'>('view');
const attachmentsFromMessage = message.parts.filter(
(part) => part.type === 'file',
);
useDataStream();
return (
<AnimatePresence>
<motion.div
data-testid={`message-${message.role}`}
className="px-2 mx-auto w-full max-w-3xl group/message"
initial={{ y: 5, opacity: 0 }}
animate={{ y: 0, opacity: 1 }}
data-role={message.role}
>
<div
className={cn(
'flex gap-3 w-full group-data-[role=user]/message:ml-auto',
{
'max-w-[90%] md:max-w-[85%]': message.role === 'user',
'max-w-[90%] md:max-w-[85%]': message.role === 'assistant',
'w-full': mode === 'edit',
},
)}
>
{message.role === 'assistant' && (
<div className="flex justify-center items-center rounded-full bg-primary text-primary-foreground size-8 shrink-0">
<div className="translate-y-px">
<SparklesIcon size={16} />
</div>
</div>
)}
<div
className={cn('flex flex-col gap-3 w-full', {
'min-h-96': message.role === 'assistant' && requiresScrollPadding,
})}
>
{attachmentsFromMessage.length > 0 && (
<div
data-testid={`message-attachments`}
className="flex flex-row gap-2 justify-end"
>
{attachmentsFromMessage.map((attachment) => (
<PreviewAttachment
key={attachment.url}
attachment={{
name: attachment.filename ?? 'file',
contentType: attachment.mediaType,
url: attachment.url,
}}
/>
))}
</div>
)}
{message.parts?.map((part, index) => {
const { type } = part;
const key = `message-${message.id}-part-${index}`;
if (type === 'reasoning' && part.text?.trim().length > 0) {
return (
<MessageReasoning
key={key}
isLoading={isLoading}
reasoning={part.text}
/>
);
}
if (type === 'text') {
if (mode === 'view') {
return (
<div key={key} className="flex flex-row gap-2 items-start">
{message.role === 'user' && !isReadonly && (
<Tooltip>
<TooltipTrigger asChild>
<Button
data-testid="message-edit-button"
variant="ghost"
size="icon"
className="h-6 w-6 rounded-full opacity-0 text-muted-foreground group-hover/message:opacity-100 transition-opacity"
onClick={() => {
setMode('edit');
}}
>
<PencilEditIcon size={14} />
</Button>
</TooltipTrigger>
<TooltipContent>Edit message</TooltipContent>
</Tooltip>
)}
<MessageContent
data-testid="message-content"
className={cn(
'justify-start items-start text-left rounded-xl px-4 py-3 text-sm',
{
'bg-primary text-primary-foreground self-end':
message.role === 'user',
'bg-muted/50 dark:bg-muted/70 self-start':
message.role === 'assistant',
}
)}
>
<Response>{sanitizeText(part.text)}</Response>
</MessageContent>
</div>
);
}
if (mode === 'edit') {
return (
<div key={key} className="flex flex-row gap-2 items-start w-full">
<div className="size-8" />
<div className="flex-1">
<MessageEditor
key={message.id}
message={message}
setMode={setMode}
setMessages={setMessages}
regenerate={regenerate}
/>
</div>
</div>
);
}
}
if (type === 'tool-getWeather') {
const { toolCallId, state } = part;
return (
<Tool key={toolCallId} defaultOpen={true}>
<ToolHeader type="tool-getWeather" state={state} />
<ToolContent>
{state === 'input-available' && (
<ToolInput input={part.input} />
)}
{state === 'output-available' && (
<ToolOutput
output={<Weather weatherAtLocation={part.output} />}
errorText={undefined}
/>
)}
</ToolContent>
</Tool>
);
}
if (type === 'tool-createDocument') {
const { toolCallId, state } = part;
return (
<Tool key={toolCallId} defaultOpen={true}>
<ToolHeader type="tool-createDocument" state={state} />
<ToolContent>
{state === 'input-available' && (
<ToolInput input={part.input} />
)}
{state === 'output-available' && (
<ToolOutput
output={
'error' in part.output ? (
<div className="p-3 text-red-500 rounded-lg border bg-destructive/10">
Error: {String(part.output.error)}
</div>
) : (
<DocumentPreview
isReadonly={isReadonly}
result={part.output}
/>
)
}
errorText={undefined}
/>
)}
</ToolContent>
</Tool>
);
}
if (type === 'tool-updateDocument') {
const { toolCallId, state } = part;
return (
<Tool key={toolCallId} defaultOpen={true}>
<ToolHeader type="tool-updateDocument" state={state} />
<ToolContent>
{state === 'input-available' && (
<ToolInput input={part.input} />
)}
{state === 'output-available' && (
<ToolOutput
output={
'error' in part.output ? (
<div className="p-3 text-red-500 rounded-lg border bg-destructive/10">
Error: {String(part.output.error)}
</div>
) : (
<DocumentToolResult
type="update"
result={part.output}
isReadonly={isReadonly}
/>
)
}
errorText={undefined}
/>
)}
</ToolContent>
</Tool>
);
}
if (type === 'tool-requestSuggestions') {
const { toolCallId, state } = part;
return (
<Tool key={toolCallId} defaultOpen={true}>
<ToolHeader type="tool-requestSuggestions" state={state} />
<ToolContent>
{state === 'input-available' && (
<ToolInput input={part.input} />
)}
{state === 'output-available' && (
<ToolOutput
output={
'error' in part.output ? (
<div className="p-3 text-red-500 rounded-lg border bg-destructive/10">
Error: {String(part.output.error)}
</div>
) : (
<DocumentToolResult
type="request-suggestions"
result={part.output}
isReadonly={isReadonly}
/>
)
}
errorText={undefined}
/>
)}
</ToolContent>
</Tool>
);
}
})}
{!isReadonly && (
<MessageActions
key={`action-${message.id}`}
chatId={chatId}
message={message}
vote={vote}
isLoading={isLoading}
/>
)}
</div>
{message.role === 'user' && (
<div className="flex justify-center items-center rounded-full bg-muted size-8 shrink-0">
<div className="translate-y-px">
<div className="h-4 w-4 rounded-full bg-foreground"></div>
</div>
</div>
)}
</div>
</motion.div>
</AnimatePresence>
);
};
export const PreviewMessage = memo(
PurePreviewMessage,
(prevProps, nextProps) => {
if (prevProps.isLoading !== nextProps.isLoading) return false;
if (prevProps.message.id !== nextProps.message.id) return false;
if (prevProps.requiresScrollPadding !== nextProps.requiresScrollPadding)
return false;
if (!equal(prevProps.message.parts, nextProps.message.parts)) return false;
if (!equal(prevProps.vote, nextProps.vote)) return false;
return false;
},
);
export const ThinkingMessage = () => {
const role = 'assistant';
return (
<motion.div
data-testid="message-assistant-loading"
className="px-2 mx-auto w-full max-w-3xl group/message"
initial={{ y: 5, opacity: 0 }}
animate={{ y: 0, opacity: 1, transition: { delay: 1 } }}
data-role={role}
>
<div className="flex gap-3 max-w-[85%] md:max-w-[85%]">
<div className="flex justify-center items-center rounded-full bg-primary text-primary-foreground size-8 shrink-0">
<SparklesIcon size={16} />
</div>
<div className="flex flex-col gap-3 w-full">
<div className="bg-muted/50 dark:bg-muted/70 rounded-xl px-4 py-3 text-sm">
<div className="flex items-center space-x-2 text-muted-foreground">
<div className="flex space-x-1">
<div className="h-2 w-2 rounded-full bg-muted-foreground animate-bounce"></div>
<div className="h-2 w-2 rounded-full bg-muted-foreground animate-bounce" style={{ animationDelay: '0.2s' }}></div>
<div className="h-2 w-2 rounded-full bg-muted-foreground animate-bounce" style={{ animationDelay: '0.4s' }}></div>
</div>
<span>Thinking...</span>
</div>
</div>
</div>
</div>
</motion.div>
);
};