import type { UserMessageEntry, AssistantMessageEntry, UserMessageImage } from "../../src/lib/types"; import { cn, esc } from "../../src/lib/utils"; import { MessageResponse } from "../ai-elements/message"; import { Reasoning, ReasoningTrigger, ReasoningContent } from "../ai-elements/reasoning"; // ============================================================================= // 用户消息 — 右对齐,深色反转背景,无气泡边框 // Anthropic: right-aligned, inverted dark bg, rounded-xl with bottom-right notch // ============================================================================= interface UserBubbleProps { entry: UserMessageEntry; } export function UserBubble({ entry }: UserBubbleProps) { return (
{/* 图片附件 */} {entry.images && entry.images.length > 0 && (
{entry.images.map((img, i) => ( ))}
)} {/* 文本内容 */} {entry.content && (
{esc(entry.content)}
)}
); } // ============================================================================= // 助手消息 — 左对齐,无背景卡片,编辑式排版 // Anthropic: avatar + plain text, no bubble/card wrapper, serif body font // ============================================================================= interface AssistantBubbleProps { entry: AssistantMessageEntry; isStreaming?: boolean; } export function AssistantBubble({ entry, isStreaming }: AssistantBubbleProps) { return (
{/* Orange triangle avatar */}
{/* 内容 — 无卡片背景,直接排版 */}
{/* Sender label */} Claude {entry.chunks.map((chunk, i) => { if (chunk.type === "thought") { const isLastChunk = i === entry.chunks.length - 1; const isThoughtStreaming = isStreaming && isLastChunk; return (
{chunk.text}
); } // 普通消息块 — 直接输出,无包裹卡片 return (
{chunk.text}
); })}
); } // ============================================================================= // 图片缩略图 — 点击放大 // ============================================================================= function ImageThumbnail({ image }: { image: UserMessageImage }) { const dataUrl = `data:${image.mimeType};base64,${image.data}`; return ( ); }