style: 完成所有文件的lint

This commit is contained in:
claude-code-best
2026-05-01 21:39:30 +08:00
parent d136872cc9
commit 6182015005
1333 changed files with 68255 additions and 77882 deletions

View File

@@ -1,17 +1,17 @@
import { useState, useRef, useCallback, type KeyboardEvent, type ClipboardEvent } from "react";
import { cn } from "../../src/lib/utils";
import { Send, Square, Paperclip, Slash } from "lucide-react";
import type { ChatInputMessage, UserMessageImage } from "../../src/lib/types";
import type { AvailableCommand } from "../../src/acp/types";
import { CommandMenu } from "./CommandMenu";
import imageCompression from "browser-image-compression";
import { useState, useRef, useCallback, type KeyboardEvent, type ClipboardEvent } from 'react';
import { cn } from '../../src/lib/utils';
import { Send, Square, Paperclip, Slash } from 'lucide-react';
import type { ChatInputMessage, UserMessageImage } from '../../src/lib/types';
import type { AvailableCommand } from '../../src/acp/types';
import { CommandMenu } from './CommandMenu';
import imageCompression from 'browser-image-compression';
// 图片压缩配置
const IMAGE_COMPRESSION_OPTIONS = {
maxSizeMB: 2,
maxWidthOrHeight: 2048,
useWebWorker: true,
fileType: "image/jpeg" as const,
fileType: 'image/jpeg' as const,
};
// =============================================================================
@@ -36,15 +36,15 @@ export function ChatInput({
isLoading = false,
onInterrupt,
disabled = false,
placeholder = "给 Claude 发送消息…",
placeholder = '给 Claude 发送消息…',
supportsImages = false,
commands,
className,
}: ChatInputProps) {
const [text, setText] = useState("");
const [text, setText] = useState('');
const [images, setImages] = useState<UserMessageImage[]>([]);
const [showCommandMenu, setShowCommandMenu] = useState(false);
const [commandFilter, setCommandFilter] = useState("");
const [commandFilter, setCommandFilter] = useState('');
const textareaRef = useRef<HTMLTextAreaElement>(null);
const fileInputRef = useRef<HTMLInputElement>(null);
@@ -53,37 +53,37 @@ export function ChatInput({
if ((!trimmed && images.length === 0) || disabled) return;
onSubmit({ text: trimmed, images: images.length > 0 ? images : undefined });
setText("");
setText('');
setImages([]);
setShowCommandMenu(false);
setCommandFilter("");
setCommandFilter('');
// 重置 textarea 高度
if (textareaRef.current) {
textareaRef.current.style.height = "auto";
textareaRef.current.style.height = 'auto';
}
}, [text, images, disabled, onSubmit]);
const handleKeyDown = useCallback(
(e: KeyboardEvent<HTMLTextAreaElement>) => {
if (showCommandMenu) {
if (e.key === "Escape") {
if (e.key === 'Escape') {
e.preventDefault();
setShowCommandMenu(false);
return;
}
// Arrow keys and Enter are handled by CommandMenu via document-level listener
// Don't submit or move cursor when menu is open
if (e.key === "ArrowUp" || e.key === "ArrowDown" || e.key === "Enter") {
if (e.key === 'ArrowUp' || e.key === 'ArrowDown' || e.key === 'Enter') {
e.preventDefault();
return;
}
if (e.key === "Tab") {
if (e.key === 'Tab') {
e.preventDefault();
setShowCommandMenu(false);
return;
}
}
if (e.key === "Enter" && !e.shiftKey && !e.nativeEvent.isComposing) {
if (e.key === 'Enter' && !e.shiftKey && !e.nativeEvent.isComposing) {
e.preventDefault();
if (isLoading) {
onInterrupt?.();
@@ -95,35 +95,41 @@ export function ChatInput({
[handleSubmit, isLoading, onInterrupt, showCommandMenu],
);
const handleInput = useCallback((e: React.ChangeEvent<HTMLTextAreaElement>) => {
const value = e.target.value;
setText(value);
const handleInput = useCallback(
(e: React.ChangeEvent<HTMLTextAreaElement>) => {
const value = e.target.value;
setText(value);
// 检测 slash 命令模式:仅在输入开头输入 / 时触发
if (value.startsWith("/") && commands && commands.length > 0) {
setShowCommandMenu(true);
setCommandFilter(value.slice(1).split(/\s/)[0] || "");
} else if (showCommandMenu) {
setShowCommandMenu(false);
setCommandFilter("");
}
// 检测 slash 命令模式:仅在输入开头输入 / 时触发
if (value.startsWith('/') && commands && commands.length > 0) {
setShowCommandMenu(true);
setCommandFilter(value.slice(1).split(/\s/)[0] || '');
} else if (showCommandMenu) {
setShowCommandMenu(false);
setCommandFilter('');
}
// 自动调整高度
const el = e.target;
el.style.height = "auto";
el.style.height = Math.min(el.scrollHeight, 200) + "px";
}, [commands, showCommandMenu]);
// 自动调整高度
const el = e.target;
el.style.height = 'auto';
el.style.height = Math.min(el.scrollHeight, 200) + 'px';
},
[commands, showCommandMenu],
);
// 粘贴图片
const handlePaste = useCallback(async (e: ClipboardEvent) => {
if (!supportsImages) return;
const files = Array.from(e.clipboardData.files).filter((f) => f.type.startsWith("image/"));
if (files.length === 0) return;
const handlePaste = useCallback(
async (e: ClipboardEvent) => {
if (!supportsImages) return;
const files = Array.from(e.clipboardData.files).filter(f => f.type.startsWith('image/'));
if (files.length === 0) return;
e.preventDefault();
const newImages = await processImageFiles(files);
setImages((prev) => [...prev, ...newImages]);
}, [supportsImages]);
e.preventDefault();
const newImages = await processImageFiles(files);
setImages(prev => [...prev, ...newImages]);
},
[supportsImages],
);
// 选择文件
const handleFileSelect = useCallback(async () => {
@@ -132,32 +138,32 @@ export function ChatInput({
if (!files || files.length === 0) return;
const newImages = await processImageFiles(Array.from(files));
setImages((prev) => [...prev, ...newImages]);
setImages(prev => [...prev, ...newImages]);
// 清空 input 以便重复选择
fileInputRef.current.value = "";
fileInputRef.current.value = '';
}, []);
const removeImage = useCallback((index: number) => {
setImages((prev) => prev.filter((_, i) => i !== index));
setImages(prev => prev.filter((_, i) => i !== index));
}, []);
const handleCommandSelect = useCallback((command: AvailableCommand) => {
setText(`/${command.name} `);
setShowCommandMenu(false);
setCommandFilter("");
setCommandFilter('');
textareaRef.current?.focus();
}, []);
const toggleCommandMenu = useCallback(() => {
if (showCommandMenu) {
setShowCommandMenu(false);
setCommandFilter("");
setCommandFilter('');
} else {
if (!text.startsWith("/")) {
setText("/" + text);
if (!text.startsWith('/')) {
setText('/' + text);
}
setShowCommandMenu(true);
setCommandFilter(text.startsWith("/") ? text.slice(1).split(/\s/)[0] || "" : "");
setCommandFilter(text.startsWith('/') ? text.slice(1).split(/\s/)[0] || '' : '');
textareaRef.current?.focus();
}
}, [showCommandMenu, text]);
@@ -165,7 +171,7 @@ export function ChatInput({
const canSend = (text.trim() || images.length > 0) && !disabled;
return (
<div className={cn("w-full max-w-3xl mx-auto px-4 sm:px-8 pb-4 pt-2", className)}>
<div className={cn('w-full max-w-3xl mx-auto px-4 sm:px-8 pb-4 pt-2', className)}>
<div className="relative">
{/* Slash command menu — floating above input */}
{showCommandMenu && commands && commands.length > 0 && (
@@ -175,127 +181,124 @@ export function ChatInput({
onSelect={handleCommandSelect}
onClose={() => {
setShowCommandMenu(false);
setCommandFilter("");
setCommandFilter('');
}}
className="absolute bottom-full left-0 right-0 mb-1 z-50"
/>
)}
<div className={cn(
"rounded-xl border border-border bg-surface-2 overflow-hidden",
"focus-within:border-brand/50 focus-within:shadow-[0_0_0_3px_rgba(217,119,87,0.15)] transition-all",
)}>
{/* 图片预览 */}
{images.length > 0 && (
<div className="flex flex-wrap gap-2 px-3 pt-3">
{images.map((img, i) => (
<div key={i} className="relative group">
<img
src={`data:${img.mimeType};base64,${img.data}`}
alt={`Attached image ${i + 1}`}
className="h-14 w-14 object-cover rounded-lg border border-border"
/>
<div
className={cn(
'rounded-xl border border-border bg-surface-2 overflow-hidden',
'focus-within:border-brand/50 focus-within:shadow-[0_0_0_3px_rgba(217,119,87,0.15)] transition-all',
)}
>
{/* 图片预览 */}
{images.length > 0 && (
<div className="flex flex-wrap gap-2 px-3 pt-3">
{images.map((img, i) => (
<div key={i} className="relative group">
<img
src={`data:${img.mimeType};base64,${img.data}`}
alt={`Attached image ${i + 1}`}
className="h-14 w-14 object-cover rounded-lg border border-border"
/>
<button
type="button"
onClick={() => removeImage(i)}
className="absolute -top-1.5 -right-1.5 min-h-[32px] min-w-[32px] h-5 w-5 rounded-full bg-surface-2 border border-border flex items-center justify-center text-text-muted hover:text-text-primary text-xs opacity-0 group-hover:opacity-100 transition-opacity"
aria-label={`Remove image ${i + 1}`}
>
{'\u00D7'}
</button>
</div>
))}
</div>
)}
{/* 输入区域 — Anthropic 单行紧凑布局 */}
<div className="flex items-end gap-2 px-3 py-2.5">
{/* 左侧附件按钮 */}
{supportsImages && (
<>
<button
type="button"
onClick={() => removeImage(i)}
className="absolute -top-1.5 -right-1.5 min-h-[32px] min-w-[32px] h-5 w-5 rounded-full bg-surface-2 border border-border flex items-center justify-center text-text-muted hover:text-text-primary text-xs opacity-0 group-hover:opacity-100 transition-opacity"
aria-label={`Remove image ${i + 1}`}
onClick={() => fileInputRef.current?.click()}
className="flex-shrink-0 h-8 w-8 flex items-center justify-center rounded-lg text-text-muted hover:text-text-secondary hover:bg-surface-1/50 transition-colors"
disabled={disabled}
>
{"\u00D7"}
<Paperclip className="h-4 w-4" />
<span className="sr-only">Attach file</span>
</button>
</div>
))}
</div>
)}
<input
ref={fileInputRef}
type="file"
accept="image/*"
multiple
className="hidden"
onChange={handleFileSelect}
/>
</>
)}
{/* 输入区域 — Anthropic 单行紧凑布局 */}
<div className="flex items-end gap-2 px-3 py-2.5">
{/* 左侧附件按钮 */}
{supportsImages && (
<>
{/* Slash 命令按钮 */}
{commands && commands.length > 0 && (
<button
type="button"
onClick={() => fileInputRef.current?.click()}
className="flex-shrink-0 h-8 w-8 flex items-center justify-center rounded-lg text-text-muted hover:text-text-secondary hover:bg-surface-1/50 transition-colors"
onClick={toggleCommandMenu}
className={cn(
'flex-shrink-0 h-8 w-8 flex items-center justify-center rounded-lg transition-colors',
showCommandMenu
? 'bg-brand/15 text-brand'
: 'text-text-muted hover:text-text-secondary hover:bg-surface-1/50',
)}
disabled={disabled}
title="命令列表"
>
<Paperclip className="h-4 w-4" />
<span className="sr-only">Attach file</span>
<Slash className="h-4 w-4" />
</button>
<input
ref={fileInputRef}
type="file"
accept="image/*"
multiple
className="hidden"
onChange={handleFileSelect}
/>
</>
)}
)}
{/* Slash 命令按钮 */}
{commands && commands.length > 0 && (
{/* Textarea — Poppins font */}
<textarea
ref={textareaRef}
value={text}
onChange={handleInput}
onKeyDown={handleKeyDown}
onPaste={handlePaste}
placeholder={placeholder}
disabled={disabled}
rows={1}
className={cn(
'flex-1 resize-none border-none bg-transparent outline-none',
'text-sm text-text-primary placeholder:text-text-muted font-display',
'max-h-[200px] min-h-[24px] leading-normal',
)}
/>
{/* 右侧发送/取消按钮 */}
<button
type="button"
onClick={toggleCommandMenu}
onClick={isLoading ? onInterrupt : handleSubmit}
disabled={!isLoading && !canSend}
className={cn(
"flex-shrink-0 h-8 w-8 flex items-center justify-center rounded-lg transition-colors",
showCommandMenu
? "bg-brand/15 text-brand"
: "text-text-muted hover:text-text-secondary hover:bg-surface-1/50",
'flex-shrink-0 h-8 w-8 flex items-center justify-center rounded-lg transition-all',
isLoading
? 'bg-text-primary text-surface-2 hover:bg-text-secondary'
: canSend
? 'bg-brand text-white hover:bg-brand-light hover:scale-[1.05] active:scale-[0.97]'
: 'bg-surface-1 text-text-muted',
)}
disabled={disabled}
title="命令列表"
>
<Slash className="h-4 w-4" />
{isLoading ? <Square className="h-3.5 w-3.5" fill="currentColor" /> : <Send className="h-4 w-4" />}
</button>
)}
{/* Textarea — Poppins font */}
<textarea
ref={textareaRef}
value={text}
onChange={handleInput}
onKeyDown={handleKeyDown}
onPaste={handlePaste}
placeholder={placeholder}
disabled={disabled}
rows={1}
className={cn(
"flex-1 resize-none border-none bg-transparent outline-none",
"text-sm text-text-primary placeholder:text-text-muted font-display",
"max-h-[200px] min-h-[24px] leading-normal",
)}
/>
{/* 右侧发送/取消按钮 */}
<button
type="button"
onClick={isLoading ? onInterrupt : handleSubmit}
disabled={!isLoading && !canSend}
className={cn(
"flex-shrink-0 h-8 w-8 flex items-center justify-center rounded-lg transition-all",
isLoading
? "bg-text-primary text-surface-2 hover:bg-text-secondary"
: canSend
? "bg-brand text-white hover:bg-brand-light hover:scale-[1.05] active:scale-[0.97]"
: "bg-surface-1 text-text-muted",
)}
>
{isLoading ? (
<Square className="h-3.5 w-3.5" fill="currentColor" />
) : (
<Send className="h-4 w-4" />
)}
</button>
</div>
</div>
</div>
</div>{/* end relative */}
{/* end relative */}
{/* 提示文本 */}
<div className="text-center mt-1.5">
<span className="text-[11px] text-text-muted font-display">
Enter Shift+Enter
</span>
<span className="text-[11px] text-text-muted font-display">Enter Shift+Enter </span>
</div>
</div>
);
@@ -316,23 +319,23 @@ async function processImageFiles(files: File[]): Promise<UserMessageImage[]> {
if (file.size > 2 * 1024 * 1024) {
const compressed = await imageCompression(file, IMAGE_COMPRESSION_OPTIONS);
blob = compressed;
mimeType = "image/jpeg";
mimeType = 'image/jpeg';
}
const base64 = await new Promise<string>((resolve, reject) => {
const reader = new FileReader();
reader.onloadend = () => {
const result = reader.result as string;
const commaIdx = result.indexOf(",");
const commaIdx = result.indexOf(',');
resolve(commaIdx >= 0 ? result.slice(commaIdx + 1) : result);
};
reader.onerror = () => reject(new Error("FileReader error"));
reader.onerror = () => reject(new Error('FileReader error'));
reader.readAsDataURL(blob);
});
results.push({ mimeType, data: base64 });
} catch (err) {
console.error("Failed to process image:", err);
console.error('Failed to process image:', err);
}
}

View File

@@ -1,9 +1,14 @@
import type { ThreadEntry, ToolCallEntry, PlanDisplayEntry } from "../../src/lib/types";
import { cn } from "../../src/lib/utils";
import { UserBubble, AssistantBubble } from "./MessageBubble";
import { ToolCallGroup } from "./ToolCallGroup";
import { PlanDisplay } from "./PlanView";
import { Conversation, ConversationContent, ConversationEmptyState, ConversationScrollButtons } from "../ai-elements/conversation";
import type { ThreadEntry, ToolCallEntry, PlanDisplayEntry } from '../../src/lib/types';
import { cn } from '../../src/lib/utils';
import { UserBubble, AssistantBubble } from './MessageBubble';
import { ToolCallGroup } from './ToolCallGroup';
import { PlanDisplay } from './PlanView';
import {
Conversation,
ConversationContent,
ConversationEmptyState,
ConversationScrollButtons,
} from '../ai-elements/conversation';
// =============================================================================
// 统一聊天视图 — Anthropic 编辑式排版
@@ -22,28 +27,25 @@ export function ChatView({
entries,
isLoading = false,
onPermissionRespond,
emptyTitle = "开始对话",
emptyDescription = "输入消息开始聊天",
emptyTitle = '开始对话',
emptyDescription = '输入消息开始聊天',
}: ChatViewProps) {
// 将相邻的 ToolCallEntry 合并为一组
const grouped = groupToolCalls(entries);
const hasMessages = entries.length > 0;
// 检查是否正在加载(最后一个条目是用户消息)
const showThinking = isLoading && entries.length > 0 && entries[entries.length - 1]?.type === "user_message";
const showThinking = isLoading && entries.length > 0 && entries[entries.length - 1]?.type === 'user_message';
return (
<Conversation className="flex-1">
<ConversationContent>
{!hasMessages ? (
<ConversationEmptyState
title={emptyTitle}
description={emptyDescription}
/>
<ConversationEmptyState title={emptyTitle} description={emptyDescription} />
) : (
<>
{grouped.map((item, i) => {
if (item.type === "single") {
if (item.type === 'single') {
return (
<div key={`entry-${i}`} className={cn(entrySpacing(entries, i))}>
<EntryRenderer entry={item.entry} isLoading={isLoading} onPermissionRespond={onPermissionRespond} />
@@ -63,19 +65,25 @@ export function ChatView({
<div className="flex gap-4 items-start">
<div className="w-8 h-8 rounded-lg bg-brand/8 flex items-center justify-center flex-shrink-0">
<svg width="20" height="20" viewBox="0 0 24 24" aria-hidden="true">
<path d="M4.709 15.955l4.72-2.647.08-.23-.08-.128H9.2l-.79-.048-2.698-.073-2.339-.097-2.266-.122-.571-.121L0 11.784l.055-.352.48-.321.686.06 1.52.103 2.278.158 1.652.097 2.449.255h.389l.055-.157-.134-.098-.103-.097-2.358-1.596-2.552-1.688-1.336-.972-.724-.491-.364-.462-.158-1.008.656-.722.881.06.225.061.893.686 1.908 1.476 2.491 1.833.365.304.145-.103.019-.073-.164-.274-1.355-2.446-1.446-2.49-.644-1.032-.17-.619a2.97 2.97 0 01-.104-.729L6.283.134 6.696 0l.996.134.42.364.62 1.414 1.002 2.229 1.555 3.03.456.898.243.832.091.255h.158V9.01l.128-1.706.237-2.095.23-2.695.08-.76.376-.91.747-.492.584.28.48.685-.067.444-.286 1.851-.559 2.903-.364 1.942h.212l.243-.242.985-1.306 1.652-2.064.73-.82.85-.904.547-.431h1.033l.76 1.129-.34 1.166-1.064 1.347-.881 1.142-1.264 1.7-.79 1.36.073.11.188-.02 2.856-.606 1.543-.28 1.841-.315.833.388.091.395-.328.807-1.969.486-2.309.462-3.439.813-.042.03.049.061 1.549.146.662.036h1.622l3.02.225.79.522.474.638-.079.485-1.215.62-1.64-.389-3.829-.91-1.312-.329h-.182v.11l1.093 1.068 2.006 1.81 2.509 2.33.127.578-.322.455-.34-.049-2.205-1.657-.851-.747-1.926-1.62h-.128v.17l.444.649 2.345 3.521.122 1.08-.17.353-.608.213-.668-.122-1.374-1.925-1.415-2.167-1.143-1.943-.14.08-.674 7.254-.316.37-.729.28-.607-.461-.322-.747.322-1.476.389-1.924.315-1.53.286-1.9.17-.632-.012-.042-.14.018-1.434 1.967-2.18 2.945-1.726 1.845-.414.164-.717-.37.067-.662.401-.589 2.388-3.036 1.44-1.882.93-1.086-.006-.158h-.055L4.132 18.56l-1.13.146-.487-.456.061-.746.231-.243 1.908-1.312-.006.006z" fill="var(--color-brand)" fillRule="nonzero" />
<path
d="M4.709 15.955l4.72-2.647.08-.23-.08-.128H9.2l-.79-.048-2.698-.073-2.339-.097-2.266-.122-.571-.121L0 11.784l.055-.352.48-.321.686.06 1.52.103 2.278.158 1.652.097 2.449.255h.389l.055-.157-.134-.098-.103-.097-2.358-1.596-2.552-1.688-1.336-.972-.724-.491-.364-.462-.158-1.008.656-.722.881.06.225.061.893.686 1.908 1.476 2.491 1.833.365.304.145-.103.019-.073-.164-.274-1.355-2.446-1.446-2.49-.644-1.032-.17-.619a2.97 2.97 0 01-.104-.729L6.283.134 6.696 0l.996.134.42.364.62 1.414 1.002 2.229 1.555 3.03.456.898.243.832.091.255h.158V9.01l.128-1.706.237-2.095.23-2.695.08-.76.376-.91.747-.492.584.28.48.685-.067.444-.286 1.851-.559 2.903-.364 1.942h.212l.243-.242.985-1.306 1.652-2.064.73-.82.85-.904.547-.431h1.033l.76 1.129-.34 1.166-1.064 1.347-.881 1.142-1.264 1.7-.79 1.36.073.11.188-.02 2.856-.606 1.543-.28 1.841-.315.833.388.091.395-.328.807-1.969.486-2.309.462-3.439.813-.042.03.049.061 1.549.146.662.036h1.622l3.02.225.79.522.474.638-.079.485-1.215.62-1.64-.389-3.829-.91-1.312-.329h-.182v.11l1.093 1.068 2.006 1.81 2.509 2.33.127.578-.322.455-.34-.049-2.205-1.657-.851-.747-1.926-1.62h-.128v.17l.444.649 2.345 3.521.122 1.08-.17.353-.608.213-.668-.122-1.374-1.925-1.415-2.167-1.143-1.943-.14.08-.674 7.254-.316.37-.729.28-.607-.461-.322-.747.322-1.476.389-1.924.315-1.53.286-1.9.17-.632-.012-.042-.14.018-1.434 1.967-2.18 2.945-1.726 1.845-.414.164-.717-.37.067-.662.401-.589 2.388-3.036 1.44-1.882.93-1.086-.006-.158h-.055L4.132 18.56l-1.13.146-.487-.456.061-.746.231-.243 1.908-1.312-.006.006z"
fill="var(--color-brand)"
fillRule="nonzero"
/>
</svg>
</div>
<div className="flex items-center gap-1 pt-2">
<span className="chat-typing-indicator" aria-hidden="true">
<span></span><span></span><span></span>
<span></span>
<span></span>
<span></span>
</span>
</div>
</div>
)}
</>
)}
<ConversationScrollButtons hasUserMessages={entries.some((e) => e.type === "user_message")} />
<ConversationScrollButtons hasUserMessages={entries.some(e => e.type === 'user_message')} />
</ConversationContent>
</Conversation>
);
@@ -88,22 +96,22 @@ export function ChatView({
function entrySpacing(entries: ThreadEntry[], index: number): string {
const entry = entries[index];
// 用户消息前后大留白 — Claude.ai 式宽松间距
if (entry?.type === "user_message") {
return "pt-10 pb-3";
if (entry?.type === 'user_message') {
return 'pt-10 pb-3';
}
// 助手消息 — 工具调用紧贴,否则多留白
if (entry?.type === "assistant_message") {
if (entry?.type === 'assistant_message') {
const next = entries[index + 1];
if (next?.type === "tool_call") {
return "pt-3 pb-1";
if (next?.type === 'tool_call') {
return 'pt-3 pb-1';
}
return "pt-3 pb-8";
return 'pt-3 pb-8';
}
// Plan 条目
if (entry?.type === "plan") {
return "pt-3 pb-3";
if (entry?.type === 'plan') {
return 'pt-3 pb-3';
}
return "py-2";
return 'py-2';
}
// =============================================================================
@@ -120,18 +128,13 @@ function EntryRenderer({
onPermissionRespond?: (requestId: string, optionId: string | null, optionKind: string | null) => void;
}) {
switch (entry.type) {
case "user_message":
case 'user_message':
return <UserBubble entry={entry} />;
case "assistant_message":
case 'assistant_message':
return <AssistantBubble entry={entry} isStreaming={isLoading} />;
case "tool_call":
return (
<ToolCallGroup
entries={[entry as ToolCallEntry]}
onPermissionRespond={onPermissionRespond}
/>
);
case "plan":
case 'tool_call':
return <ToolCallGroup entries={[entry as ToolCallEntry]} onPermissionRespond={onPermissionRespond} />;
case 'plan':
return <PlanDisplay entry={entry as PlanDisplayEntry} />;
default:
return null;
@@ -142,9 +145,7 @@ function EntryRenderer({
// 工具调用分组逻辑
// =============================================================================
type GroupedItem =
| { type: "single"; entry: ThreadEntry }
| { type: "tool_group"; entries: ToolCallEntry[] };
type GroupedItem = { type: 'single'; entry: ThreadEntry } | { type: 'tool_group'; entries: ToolCallEntry[] };
function groupToolCalls(entries: ThreadEntry[]): GroupedItem[] {
const result: GroupedItem[] = [];
@@ -152,19 +153,19 @@ function groupToolCalls(entries: ThreadEntry[]): GroupedItem[] {
const flushToolGroup = () => {
if (currentToolGroup.length === 1) {
result.push({ type: "single", entry: currentToolGroup[0] });
result.push({ type: 'single', entry: currentToolGroup[0] });
} else if (currentToolGroup.length > 1) {
result.push({ type: "tool_group", entries: currentToolGroup });
result.push({ type: 'tool_group', entries: currentToolGroup });
}
currentToolGroup = [];
};
for (const entry of entries) {
if (entry.type === "tool_call") {
if (entry.type === 'tool_call') {
currentToolGroup.push(entry);
} else {
flushToolGroup();
result.push({ type: "single", entry });
result.push({ type: 'single', entry });
}
}
flushToolGroup();

View File

@@ -1,6 +1,6 @@
import { useMemo, useRef, useEffect, useState } from "react";
import { cn } from "../../src/lib/utils";
import type { AvailableCommand } from "../../src/acp/types";
import { useMemo, useRef, useEffect, useState } from 'react';
import { cn } from '../../src/lib/utils';
import type { AvailableCommand } from '../../src/acp/types';
// =============================================================================
// Slash command picker — floating above ChatInput
@@ -23,22 +23,14 @@ function prefixMatch(query: string, text: string): boolean {
return text.toLowerCase().startsWith(query.toLowerCase());
}
export function CommandMenu({
commands,
filter,
onSelect,
onClose,
className,
}: CommandMenuProps) {
export function CommandMenu({ commands, filter, onSelect, onClose, className }: CommandMenuProps) {
const containerRef = useRef<HTMLDivElement>(null);
const [activeIndex, setActiveIndex] = useState(0);
// Filter commands by current input
const filtered = useMemo(() => {
if (!filter) return commands;
return commands.filter(
(cmd) => prefixMatch(filter, cmd.name),
);
return commands.filter(cmd => prefixMatch(filter, cmd.name));
}, [commands, filter]);
// Reset active index when filter changes
@@ -53,8 +45,8 @@ export function CommandMenu({
onClose();
}
};
document.addEventListener("mousedown", handleClick);
return () => document.removeEventListener("mousedown", handleClick);
document.addEventListener('mousedown', handleClick);
return () => document.removeEventListener('mousedown', handleClick);
}, [onClose]);
// Handle keyboard navigation (ArrowUp/ArrowDown/Enter) via document-level listener
@@ -62,21 +54,21 @@ export function CommandMenu({
const handleKeyDown = (e: KeyboardEvent) => {
if (filtered.length === 0) return;
if (e.key === "ArrowDown") {
if (e.key === 'ArrowDown') {
e.preventDefault();
setActiveIndex((prev) => (prev + 1) % filtered.length);
} else if (e.key === "ArrowUp") {
setActiveIndex(prev => (prev + 1) % filtered.length);
} else if (e.key === 'ArrowUp') {
e.preventDefault();
setActiveIndex((prev) => (prev - 1 + filtered.length) % filtered.length);
} else if (e.key === "Enter" && !e.shiftKey) {
setActiveIndex(prev => (prev - 1 + filtered.length) % filtered.length);
} else if (e.key === 'Enter' && !e.shiftKey) {
e.preventDefault();
const cmd = filtered[activeIndex];
if (cmd) onSelect(cmd);
}
};
document.addEventListener("keydown", handleKeyDown, true); // capture phase
return () => document.removeEventListener("keydown", handleKeyDown, true);
document.addEventListener('keydown', handleKeyDown, true); // capture phase
return () => document.removeEventListener('keydown', handleKeyDown, true);
}, [filtered, activeIndex, onSelect]);
// Scroll active item into view
@@ -84,22 +76,14 @@ export function CommandMenu({
const container = containerRef.current;
if (!container) return;
const active = container.querySelector("[data-active='true']");
active?.scrollIntoView({ block: "nearest" });
active?.scrollIntoView({ block: 'nearest' });
}, [activeIndex]);
return (
<div
ref={containerRef}
className={cn(
"rounded-xl border border-border bg-surface-2 shadow-lg",
className,
)}
>
<div ref={containerRef} className={cn('rounded-xl border border-border bg-surface-2 shadow-lg', className)}>
<div className="max-h-[320px] overflow-y-auto py-1">
{filtered.length === 0 ? (
<div className="text-xs text-text-muted font-display py-3 text-center">
</div>
<div className="text-xs text-text-muted font-display py-3 text-center"></div>
) : (
filtered.map((cmd, index) => (
<button
@@ -109,25 +93,15 @@ export function CommandMenu({
onClick={() => onSelect(cmd)}
onMouseEnter={() => setActiveIndex(index)}
className={cn(
"flex w-full items-center gap-2 px-3 py-2 cursor-pointer rounded-lg mx-1 text-left",
"transition-colors",
index === activeIndex
? "bg-brand/10 text-text-primary"
: "text-text-secondary hover:bg-surface-1/50",
'flex w-full items-center gap-2 px-3 py-2 cursor-pointer rounded-lg mx-1 text-left',
'transition-colors',
index === activeIndex ? 'bg-brand/10 text-text-primary' : 'text-text-secondary hover:bg-surface-1/50',
)}
style={{ width: "calc(100% - 8px)" }}
style={{ width: 'calc(100% - 8px)' }}
>
<span className="text-sm font-display font-medium text-brand">
/{cmd.name}
</span>
<span className="text-xs text-text-muted truncate flex-1">
{cmd.description}
</span>
{cmd.input?.hint && (
<span className="text-[10px] text-text-muted italic">
{cmd.input.hint}
</span>
)}
<span className="text-sm font-display font-medium text-brand">/{cmd.name}</span>
<span className="text-xs text-text-muted truncate flex-1">{cmd.description}</span>
{cmd.input?.hint && <span className="text-[10px] text-text-muted italic">{cmd.input.hint}</span>}
</button>
))
)}

View File

@@ -1,9 +1,9 @@
import { useState, useRef, useEffect, useCallback } from "react";
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";
import { ChevronDown } from "lucide-react";
import { useState, useRef, useEffect, useCallback } from 'react';
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';
import { ChevronDown } from 'lucide-react';
// 用户消息折叠最大高度px
const COLLAPSED_MAX_HEIGHT = 200;
@@ -48,7 +48,7 @@ export function UserBubble({ entry }: UserBubbleProps) {
<div
ref={contentRef}
className={cn(
"px-5 py-3 text-sm text-white whitespace-pre-wrap font-display leading-relaxed",
'px-5 py-3 text-sm text-white whitespace-pre-wrap font-display leading-relaxed',
!expanded && overflowing && `max-h-[${COLLAPSED_MAX_HEIGHT}px]`,
)}
style={!expanded && overflowing ? { maxHeight: `${COLLAPSED_MAX_HEIGHT}px` } : undefined}
@@ -90,7 +90,11 @@ export function AssistantBubble({ entry, isStreaming }: AssistantBubbleProps) {
{/* Orange triangle avatar */}
<div className="w-8 h-8 rounded-lg bg-brand/8 flex items-center justify-center flex-shrink-0 mt-0.5">
<svg width="20" height="20" viewBox="0 0 24 24" aria-hidden="true">
<path d="M4.709 15.955l4.72-2.647.08-.23-.08-.128H9.2l-.79-.048-2.698-.073-2.339-.097-2.266-.122-.571-.121L0 11.784l.055-.352.48-.321.686.06 1.52.103 2.278.158 1.652.097 2.449.255h.389l.055-.157-.134-.098-.103-.097-2.358-1.596-2.552-1.688-1.336-.972-.724-.491-.364-.462-.158-1.008.656-.722.881.06.225.061.893.686 1.908 1.476 2.491 1.833.365.304.145-.103.019-.073-.164-.274-1.355-2.446-1.446-2.49-.644-1.032-.17-.619a2.97 2.97 0 01-.104-.729L6.283.134 6.696 0l.996.134.42.364.62 1.414 1.002 2.229 1.555 3.03.456.898.243.832.091.255h.158V9.01l.128-1.706.237-2.095.23-2.695.08-.76.376-.91.747-.492.584.28.48.685-.067.444-.286 1.851-.559 2.903-.364 1.942h.212l.243-.242.985-1.306 1.652-2.064.73-.82.85-.904.547-.431h1.033l.76 1.129-.34 1.166-1.064 1.347-.881 1.142-1.264 1.7-.79 1.36.073.11.188-.02 2.856-.606 1.543-.28 1.841-.315.833.388.091.395-.328.807-1.969.486-2.309.462-3.439.813-.042.03.049.061 1.549.146.662.036h1.622l3.02.225.79.522.474.638-.079.485-1.215.62-1.64-.389-3.829-.91-1.312-.329h-.182v.11l1.093 1.068 2.006 1.81 2.509 2.33.127.578-.322.455-.34-.049-2.205-1.657-.851-.747-1.926-1.62h-.128v.17l.444.649 2.345 3.521.122 1.08-.17.353-.608.213-.668-.122-1.374-1.925-1.415-2.167-1.143-1.943-.14.08-.674 7.254-.316.37-.729.28-.607-.461-.322-.747.322-1.476.389-1.924.315-1.53.286-1.9.17-.632-.012-.042-.14.018-1.434 1.967-2.18 2.945-1.726 1.845-.414.164-.717-.37.067-.662.401-.589 2.388-3.036 1.44-1.882.93-1.086-.006-.158h-.055L4.132 18.56l-1.13.146-.487-.456.061-.746.231-.243 1.908-1.312-.006.006z" fill="var(--color-brand)" fillRule="nonzero" />
<path
d="M4.709 15.955l4.72-2.647.08-.23-.08-.128H9.2l-.79-.048-2.698-.073-2.339-.097-2.266-.122-.571-.121L0 11.784l.055-.352.48-.321.686.06 1.52.103 2.278.158 1.652.097 2.449.255h.389l.055-.157-.134-.098-.103-.097-2.358-1.596-2.552-1.688-1.336-.972-.724-.491-.364-.462-.158-1.008.656-.722.881.06.225.061.893.686 1.908 1.476 2.491 1.833.365.304.145-.103.019-.073-.164-.274-1.355-2.446-1.446-2.49-.644-1.032-.17-.619a2.97 2.97 0 01-.104-.729L6.283.134 6.696 0l.996.134.42.364.62 1.414 1.002 2.229 1.555 3.03.456.898.243.832.091.255h.158V9.01l.128-1.706.237-2.095.23-2.695.08-.76.376-.91.747-.492.584.28.48.685-.067.444-.286 1.851-.559 2.903-.364 1.942h.212l.243-.242.985-1.306 1.652-2.064.73-.82.85-.904.547-.431h1.033l.76 1.129-.34 1.166-1.064 1.347-.881 1.142-1.264 1.7-.79 1.36.073.11.188-.02 2.856-.606 1.543-.28 1.841-.315.833.388.091.395-.328.807-1.969.486-2.309.462-3.439.813-.042.03.049.061 1.549.146.662.036h1.622l3.02.225.79.522.474.638-.079.485-1.215.62-1.64-.389-3.829-.91-1.312-.329h-.182v.11l1.093 1.068 2.006 1.81 2.509 2.33.127.578-.322.455-.34-.049-2.205-1.657-.851-.747-1.926-1.62h-.128v.17l.444.649 2.345 3.521.122 1.08-.17.353-.608.213-.668-.122-1.374-1.925-1.415-2.167-1.143-1.943-.14.08-.674 7.254-.316.37-.729.28-.607-.461-.322-.747.322-1.476.389-1.924.315-1.53.286-1.9.17-.632-.012-.042-.14.018-1.434 1.967-2.18 2.945-1.726 1.845-.414.164-.717-.37.067-.662.401-.589 2.388-3.036 1.44-1.882.93-1.086-.006-.158h-.055L4.132 18.56l-1.13.146-.487-.456.061-.746.231-.243 1.908-1.312-.006.006z"
fill="var(--color-brand)"
fillRule="nonzero"
/>
</svg>
</div>
{/* 内容 — 无卡片背景,直接排版 */}
@@ -98,16 +102,14 @@ export function AssistantBubble({ entry, isStreaming }: AssistantBubbleProps) {
{/* Sender label */}
<span className="text-sm font-semibold text-text-primary font-display">Claude</span>
{entry.chunks.map((chunk, i) => {
if (chunk.type === "thought") {
if (chunk.type === 'thought') {
const isLastChunk = i === entry.chunks.length - 1;
const isThoughtStreaming = isStreaming && isLastChunk;
return (
<Reasoning key={i} isStreaming={isThoughtStreaming}>
<ReasoningTrigger />
<ReasoningContent>
<div className="text-sm text-text-secondary leading-relaxed">
{chunk.text}
</div>
<div className="text-sm text-text-secondary leading-relaxed">{chunk.text}</div>
</ReasoningContent>
</Reasoning>
);
@@ -135,17 +137,13 @@ function ImageThumbnail({ image }: { image: UserMessageImage }) {
type="button"
className="rounded-lg overflow-hidden border border-border hover:border-brand/40 transition-colors cursor-pointer"
onClick={() => {
const w = window.open("");
const w = window.open('');
if (w) {
w.document.write(`<img src="${dataUrl}" style="max-width:100%;max-height:100%" />`);
}
}}
>
<img
src={dataUrl}
alt="Uploaded image"
className="h-20 w-20 object-cover"
/>
<img src={dataUrl} alt="Uploaded image" className="h-20 w-20 object-cover" />
</button>
);
}

View File

@@ -1,6 +1,6 @@
import type { PendingPermission } from "../../src/lib/types";
import { cn } from "../../src/lib/utils";
import { ShieldAlert, Check, X } from "lucide-react";
import type { PendingPermission } from '../../src/lib/types';
import { cn } from '../../src/lib/utils';
import { ShieldAlert, Check, X } from 'lucide-react';
// =============================================================================
// 权限请求面板 — 固定在输入框上方Anthropic warm token style
@@ -16,14 +16,10 @@ export function PermissionPanel({ requests, onRespond, className }: PermissionPa
if (requests.length === 0) return null;
return (
<div className={cn("w-full max-w-3xl mx-auto px-4", className)}>
<div className={cn('w-full max-w-3xl mx-auto px-4', className)}>
<div className="space-y-2">
{requests.map((req) => (
<PermissionCard
key={req.requestId}
request={req}
onRespond={onRespond}
/>
{requests.map(req => (
<PermissionCard key={req.requestId} request={req} onRespond={onRespond} />
))}
</div>
</div>
@@ -44,13 +40,9 @@ function PermissionCard({ request, onRespond }: PermissionCardProps) {
<div className="flex items-center gap-3 rounded-xl border border-warning-border/30 bg-warning-bg/50 px-4 py-3">
<ShieldAlert className="h-5 w-5 text-warning-text flex-shrink-0" />
<div className="flex-1 min-w-0">
<div className="text-sm font-medium text-warning-text">
{request.toolName}
</div>
<div className="text-sm font-medium text-warning-text">{request.toolName}</div>
{request.description && (
<div className="text-xs text-warning-text/80 mt-0.5 truncate">
{request.description}
</div>
<div className="text-xs text-warning-text/80 mt-0.5 truncate">{request.description}</div>
)}
</div>
<div className="flex items-center gap-2 flex-shrink-0">

View File

@@ -1,8 +1,8 @@
import { useState } from "react";
import type { PlanDisplayEntry } from "../../src/lib/types";
import type { PlanEntry, PlanEntryPriority, PlanEntryStatus } from "../../src/acp/types";
import { cn } from "../../src/lib/utils";
import { CheckCircle2, Loader2, Circle } from "lucide-react";
import { useState } from 'react';
import type { PlanDisplayEntry } from '../../src/lib/types';
import type { PlanEntry, PlanEntryPriority, PlanEntryStatus } from '../../src/acp/types';
import { cn } from '../../src/lib/utils';
import { CheckCircle2, Loader2, Circle } from 'lucide-react';
// =============================================================================
// Plan 展示组件 — 执行计划可视化
@@ -18,7 +18,7 @@ export function PlanDisplay({ entry }: PlanDisplayProps) {
if (entries.length === 0) return null;
const completed = entries.filter((e) => e.status === "completed").length;
const completed = entries.filter(e => e.status === 'completed').length;
const total = entries.length;
const percentage = total > 0 ? Math.round((completed / total) * 100) : 0;
@@ -36,14 +36,12 @@ export function PlanDisplay({ entry }: PlanDisplayProps) {
height="12"
viewBox="0 0 12 12"
fill="none"
className={cn("transition-transform text-text-muted flex-shrink-0", collapsed && "rotate-90")}
className={cn('transition-transform text-text-muted flex-shrink-0', collapsed && 'rotate-90')}
>
<path d="M4 2L8 6L4 10" stroke="currentColor" strokeWidth="1.5" fill="none" />
</svg>
<span className="text-xs font-display font-medium text-text-secondary">
</span>
<span className="text-xs font-display font-medium text-text-secondary"></span>
<span className="text-[10px] text-text-muted font-mono">
{completed}/{total}
@@ -57,17 +55,14 @@ export function PlanDisplay({ entry }: PlanDisplayProps) {
/>
</div>
<span className="text-[10px] text-text-muted font-mono">
{percentage}%
</span>
<span className="text-[10px] text-text-muted font-mono">{percentage}%</span>
</button>
{/* Entry list */}
{!collapsed && (
<div className={cn(
"border-t border-border px-3 py-1.5 space-y-0.5",
total > 5 && "max-h-64 overflow-y-auto",
)}>
<div
className={cn('border-t border-border px-3 py-1.5 space-y-0.5', total > 5 && 'max-h-64 overflow-y-auto')}
>
{entries.map((planEntry, i) => (
<PlanEntryRow key={i} entry={planEntry} />
))}
@@ -88,11 +83,13 @@ function PlanEntryRow({ entry }: { entry: PlanEntry }) {
<span className="flex-shrink-0 mt-0.5">
<StatusIcon status={entry.status} />
</span>
<span className={cn(
"text-xs leading-relaxed flex-1",
entry.status === "completed" ? "text-text-muted line-through" : "text-text-secondary",
entry.status === "in_progress" && "text-text-primary font-medium",
)}>
<span
className={cn(
'text-xs leading-relaxed flex-1',
entry.status === 'completed' ? 'text-text-muted line-through' : 'text-text-secondary',
entry.status === 'in_progress' && 'text-text-primary font-medium',
)}
>
{entry.content}
</span>
<PriorityBadge priority={entry.priority} />
@@ -106,11 +103,11 @@ function PlanEntryRow({ entry }: { entry: PlanEntry }) {
function StatusIcon({ status }: { status: PlanEntryStatus }) {
switch (status) {
case "completed":
case 'completed':
return <CheckCircle2 className="h-3.5 w-3.5 text-status-active" />;
case "in_progress":
return <Loader2 className="h-3.5 w-3.5 text-brand animate-spin" style={{ animationDuration: "2s" }} />;
case "pending":
case 'in_progress':
return <Loader2 className="h-3.5 w-3.5 text-brand animate-spin" style={{ animationDuration: '2s' }} />;
case 'pending':
return <Circle className="h-3.5 w-3.5 text-text-muted" />;
}
}
@@ -121,22 +118,21 @@ function StatusIcon({ status }: { status: PlanEntryStatus }) {
function PriorityBadge({ priority }: { priority: PlanEntryPriority }) {
const styles: Record<PlanEntryPriority, string> = {
high: "bg-red-100 text-red-700 dark:bg-red-900/30 dark:text-red-300",
medium: "bg-brand/10 text-brand dark:bg-brand/20",
low: "bg-surface-1 text-text-muted",
high: 'bg-red-100 text-red-700 dark:bg-red-900/30 dark:text-red-300',
medium: 'bg-brand/10 text-brand dark:bg-brand/20',
low: 'bg-surface-1 text-text-muted',
};
const labels: Record<PlanEntryPriority, string> = {
high: "高",
medium: "中",
low: "低",
high: '高',
medium: '中',
low: '低',
};
return (
<span className={cn(
"text-[9px] font-display rounded-full px-1.5 py-0.5 flex-shrink-0 leading-none",
styles[priority],
)}>
<span
className={cn('text-[9px] font-display rounded-full px-1.5 py-0.5 flex-shrink-0 leading-none', styles[priority])}
>
{labels[priority]}
</span>
);

View File

@@ -1,7 +1,7 @@
import { cn } from "../../src/lib/utils";
import { Plus, MessageSquare, ChevronLeft, ChevronRight } from "lucide-react";
import { useState } from "react";
import type { SessionListItem } from "../../src/lib/types";
import { cn } from '../../src/lib/utils';
import { Plus, MessageSquare, ChevronLeft, ChevronRight } from 'lucide-react';
import { useState } from 'react';
import type { SessionListItem } from '../../src/lib/types';
// =============================================================================
// 会话侧边栏 — Anthropic 分段式:今天/昨天/更早 + 橙色活跃态
@@ -15,13 +15,7 @@ interface SessionSidebarProps {
className?: string;
}
export function SessionSidebar({
sessions,
activeId,
onSelect,
onNew,
className,
}: SessionSidebarProps) {
export function SessionSidebar({ sessions, activeId, onSelect, onNew, className }: SessionSidebarProps) {
const [collapsed, setCollapsed] = useState(false);
// 按日期分组
@@ -30,8 +24,8 @@ export function SessionSidebar({
return (
<div
className={cn(
"hidden md:flex flex-col border-r border-border bg-surface-1 transition-all duration-200",
collapsed ? "w-12" : "w-64",
'hidden md:flex flex-col border-r border-border bg-surface-1 transition-all duration-200',
collapsed ? 'w-12' : 'w-64',
className,
)}
>
@@ -55,11 +49,7 @@ export function SessionSidebar({
onClick={() => setCollapsed(!collapsed)}
className="h-7 w-7 flex items-center justify-center rounded-lg text-text-muted hover:text-text-primary hover:bg-surface-2 transition-colors"
>
{collapsed ? (
<ChevronRight className="h-4 w-4" />
) : (
<ChevronLeft className="h-4 w-4" />
)}
{collapsed ? <ChevronRight className="h-4 w-4" /> : <ChevronLeft className="h-4 w-4" />}
</button>
</div>
</div>
@@ -67,30 +57,28 @@ export function SessionSidebar({
{/* 会话列表 — 分段 */}
{!collapsed && (
<nav className="flex-1 overflow-y-auto py-2" aria-label="历史会话">
{groups.map((group) => (
{groups.map(group => (
<div key={group.label}>
<div className="px-3 py-1.5">
<span className="text-[10px] font-display font-medium uppercase tracking-widest text-text-muted">
{group.label}
</span>
</div>
{group.sessions.map((session) => (
{group.sessions.map(session => (
<button
key={session.id}
type="button"
onClick={() => onSelect?.(session.id)}
className={cn(
"w-full flex items-center gap-2 px-3 py-2 text-left transition-colors",
'w-full flex items-center gap-2 px-3 py-2 text-left transition-colors',
session.id === activeId
? "bg-brand/10 text-text-primary"
: "text-text-secondary hover:bg-surface-1/50 hover:text-text-primary",
? 'bg-brand/10 text-text-primary'
: 'text-text-secondary hover:bg-surface-1/50 hover:text-text-primary',
)}
title={session.title || session.id}
>
<MessageSquare className="h-3.5 w-3.5 shrink-0 text-text-muted" />
<span className="text-sm font-display truncate">
{session.title || session.id.slice(0, 8)}
</span>
<span className="text-sm font-display truncate">{session.title || session.id.slice(0, 8)}</span>
</button>
))}
</div>
@@ -121,9 +109,9 @@ function groupByRecency(sessions: SessionListItem[]): SessionGroup[] {
const yesterday = new Date(today.getTime() - 86400000);
const groups: SessionGroup[] = [
{ label: "今天", sessions: [] },
{ label: "昨天", sessions: [] },
{ label: "更早", sessions: [] },
{ label: '今天', sessions: [] },
{ label: '昨天', sessions: [] },
{ label: '更早', sessions: [] },
];
for (const session of sessions) {
@@ -137,5 +125,5 @@ function groupByRecency(sessions: SessionListItem[]): SessionGroup[] {
}
}
return groups.filter((g) => g.sessions.length > 0);
return groups.filter(g => g.sessions.length > 0);
}

View File

@@ -1,7 +1,7 @@
import { useState } from "react";
import type { ToolCallEntry, ToolCallData } from "../../src/lib/types";
import { cn } from "../../src/lib/utils";
import { ToolPermissionButtons } from "../ai-elements/permission-request";
import { useState } from 'react';
import type { ToolCallEntry, ToolCallData } from '../../src/lib/types';
import { cn } from '../../src/lib/utils';
import { ToolPermissionButtons } from '../ai-elements/permission-request';
// =============================================================================
// 工具调用折叠组 — Anthropic: subtle card, left-border accent, compact layout
@@ -21,11 +21,7 @@ export function ToolCallGroup({ entries, onPermissionRespond }: ToolCallGroupPro
if (entries.length === 1) {
return (
<div className="pl-10">
<SingleToolCard
tool={entries[0].toolCall}
compact
onPermissionRespond={onPermissionRespond}
/>
<SingleToolCard tool={entries[0].toolCall} compact onPermissionRespond={onPermissionRespond} />
</div>
);
}
@@ -47,7 +43,7 @@ export function ToolCallGroup({ entries, onPermissionRespond }: ToolCallGroupPro
height="12"
viewBox="0 0 12 12"
fill="none"
className={cn("transition-transform text-text-muted", expanded && "rotate-90")}
className={cn('transition-transform text-text-muted', expanded && 'rotate-90')}
>
<path d="M4 2L8 6L4 10" stroke="currentColor" strokeWidth="1.5" fill="none" />
</svg>
@@ -87,27 +83,28 @@ function SingleToolCard({ tool, compact, onPermissionRespond }: SingleToolCardPr
const statusIcon = (() => {
switch (tool.status) {
case "running":
case 'running':
return <span className="text-status-running text-[10px]">&#9654;</span>;
case "complete":
case 'complete':
return <span className="text-status-active text-[10px]">&#10003;</span>;
case "error":
case 'error':
return <span className="text-status-error text-[10px]">&#10005;</span>;
case "waiting_for_confirmation":
case 'waiting_for_confirmation':
return <span className="text-brand text-[10px]">&#9083;</span>;
case "canceled":
case 'canceled':
return <span className="text-text-muted text-[10px]">&#8212;</span>;
case "rejected":
case 'rejected':
return <span className="text-status-error text-[10px]">&#10005;</span>;
default:
return null;
}
})();
const hasOutput = tool.status !== "running" && tool.status !== "waiting_for_confirmation" && (tool.rawOutput || tool.content);
const hasOutput =
tool.status !== 'running' && tool.status !== 'waiting_for_confirmation' && (tool.rawOutput || tool.content);
return (
<div className={cn("px-3 py-2", compact && "py-1.5")}>
<div className={cn('px-3 py-2', compact && 'py-1.5')}>
{/* 标题行 — 单行紧凑 */}
<button
type="button"
@@ -118,13 +115,11 @@ function SingleToolCard({ tool, compact, onPermissionRespond }: SingleToolCardPr
<span className="text-xs font-display font-medium text-text-secondary group-hover:text-text-primary transition-colors truncate">
{tool.title}
</span>
{tool.status === "running" && (
<span className="text-[10px] text-status-running animate-pulse">running</span>
)}
{tool.status === 'running' && <span className="text-[10px] text-status-running animate-pulse">running</span>}
</button>
{/* 权限请求按钮 */}
{tool.status === "waiting_for_confirmation" && tool.permissionRequest && (
{tool.status === 'waiting_for_confirmation' && tool.permissionRequest && (
<div className="mt-1.5 ml-4">
<ToolPermissionButtons
requestId={tool.permissionRequest.requestId}
@@ -146,10 +141,12 @@ function SingleToolCard({ tool, compact, onPermissionRespond }: SingleToolCardPr
)}
{hasOutput && (
<div>
<pre className={cn(
"text-[11px] rounded-md p-2 overflow-x-auto font-mono max-h-36",
tool.status === "error" ? "bg-status-error/10 text-status-error" : "bg-surface-1 text-text-secondary",
)}>
<pre
className={cn(
'text-[11px] rounded-md p-2 overflow-x-auto font-mono max-h-36',
tool.status === 'error' ? 'bg-status-error/10 text-status-error' : 'bg-surface-1 text-text-secondary',
)}
>
{formatOutput(tool)}
</pre>
</div>
@@ -179,7 +176,7 @@ function buildSummary(entries: ToolCallEntry[]): string {
if (parts.length === 0) return `${entries.length} 个工具调用`;
if (parts.length === 1) return parts[0];
return `${entries.length} 个工具: ${parts.join("、")}`;
return `${entries.length} 个工具: ${parts.join('、')}`;
}
/** 简化工具名称 */
@@ -192,17 +189,17 @@ function simplifyToolName(title: string): string {
function formatOutput(tool: ToolCallData): string {
if (tool.content && tool.content.length > 0) {
const texts = tool.content
.filter((c): c is Extract<typeof c, { type: "content" }> => c.type === "content")
.filter((c) => c.content.type === "text" && "text" in c.content)
.map((c) => (c.content as { text: string }).text);
if (texts.length > 0) return truncate(texts.join("\n"), 2000);
.filter((c): c is Extract<typeof c, { type: 'content' }> => c.type === 'content')
.filter(c => c.content.type === 'text' && 'text' in c.content)
.map(c => (c.content as { text: string }).text);
if (texts.length > 0) return truncate(texts.join('\n'), 2000);
}
if (tool.rawOutput && Object.keys(tool.rawOutput).length > 0) {
return truncate(JSON.stringify(tool.rawOutput, null, 2), 2000);
}
return "";
return '';
}
function truncate(str: string, max: number): string {
return str.length > max ? str.slice(0, max) + "..." : str;
return str.length > max ? str.slice(0, max) + '...' : str;
}

View File

@@ -1,8 +1,8 @@
export { ChatView } from "./ChatView";
export { UserBubble, AssistantBubble } from "./MessageBubble";
export { ToolCallGroup } from "./ToolCallGroup";
export { PlanDisplay } from "./PlanView";
export { ChatInput } from "./ChatInput";
export { PermissionPanel } from "./PermissionPanel";
export { SessionSidebar } from "./SessionSidebar";
export { CommandMenu } from "./CommandMenu";
export { ChatView } from './ChatView'
export { UserBubble, AssistantBubble } from './MessageBubble'
export { ToolCallGroup } from './ToolCallGroup'
export { PlanDisplay } from './PlanView'
export { ChatInput } from './ChatInput'
export { PermissionPanel } from './PermissionPanel'
export { SessionSidebar } from './SessionSidebar'
export { CommandMenu } from './CommandMenu'