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 (
);
}