文档索引
在此获取完整文档索引:https://code.claude.com/docs/llms.txt 使用此文件发现所有可用页面,然后再进一步探索。
待办事项列表
使用 Claude Agent SDK 跟踪和显示待办事项,实现有序的任务管理
待办事项跟踪提供了一种结构化的方式来管理任务并向用户显示进度。Claude Agent SDK 包含内置的待办事项功能,有助于组织复杂的工作流并让用户了解任务进展。
从 TypeScript Agent SDK 0.3.142 和 Claude Code v2.1.142 起,会话使用结构化的 Task 工具 TaskCreate、TaskUpdate、TaskGet 和 TaskList 替代 TodoWrite。有关监控代码如何变更,请参阅迁移到 Task 工具。本页的示例设置了 CLAUDE_CODE_ENABLE_TASKS=0 以便为尚未迁移的会话继续展示 TodoWrite。
待办事项生命周期
待办事项遵循可预测的生命周期:
- 创建 — 当任务被识别时,状态为
pending - 激活 — 当工作开始时,状态变为
in_progress - 完成 — 当任务成功完成时
- 移除 — 当一组中的所有任务都已完成时
何时使用待办事项
SDK 会自动为以下情况创建待办事项:
- 复杂的多步骤任务 — 需要 3 个或更多不同操作
- 用户提供的任务列表 — 当提到多个项目时
- 非平凡的操作 — 适合使用进度跟踪的任务
- 明确请求 — 当用户要求组织待办事项时
示例
监控待办事项变更
import { query } from "@anthropic-ai/claude-agent-sdk";
for await (const message of query({
prompt: "Optimize my React app performance and track progress with todos",
// Re-enable TodoWrite, which this example monitors. Without it, the SDK uses
// Task tools instead and these tool_use blocks never appear.
options: { maxTurns: 15, env: { ...process.env, CLAUDE_CODE_ENABLE_TASKS: "0" } }
})) {
// Todo updates are reflected in the message stream
if (message.type === "assistant") {
for (const block of message.message.content) {
if (block.type === "tool_use" && block.name === "TodoWrite") {
const todos = block.input.todos;
console.log("Todo Status Update:");
todos.forEach((todo, index) => {
const status =
todo.status === "completed" ? "✅" : todo.status === "in_progress" ? "🔧" : "❌";
console.log(`${index + 1}. ${status} ${todo.content}`);
});
}
}
}
}
from claude_agent_sdk import query, ClaudeAgentOptions, AssistantMessage, ToolUseBlock
async for message in query(
prompt="Optimize my React app performance and track progress with todos",
# Re-enable TodoWrite, which this example monitors. Without it, the SDK uses
# Task tools instead and these tool_use blocks never appear.
options=ClaudeAgentOptions(max_turns=15, env={"CLAUDE_CODE_ENABLE_TASKS": "0"}),
):
# Todo updates are reflected in the message stream
if isinstance(message, AssistantMessage):
for block in message.content:
if isinstance(block, ToolUseBlock) and block.name == "TodoWrite":
todos = block.input["todos"]
print("Todo Status Update:")
for i, todo in enumerate(todos):
status = (
"✅"
if todo["status"] == "completed"
else "🔧"
if todo["status"] == "in_progress"
else "❌"
)
print(f"{i + 1}. {status} {todo['content']}")
实时进度显示
import { query } from "@anthropic-ai/claude-agent-sdk";
class TodoTracker {
private todos: any[] = [];
displayProgress() {
if (this.todos.length === 0) return;
const completed = this.todos.filter((t) => t.status === "completed").length;
const inProgress = this.todos.filter((t) => t.status === "in_progress").length;
const total = this.todos.length;
console.log(`\nProgress: ${completed}/${total} completed`);
console.log(`Currently working on: ${inProgress} task(s)\n`);
this.todos.forEach((todo, index) => {
const icon =
todo.status === "completed" ? "✅" : todo.status === "in_progress" ? "🔧" : "❌";
const text = todo.status === "in_progress" ? todo.activeForm : todo.content;
console.log(`${index + 1}. ${icon} ${text}`);
});
}
async trackQuery(prompt: string) {
for await (const message of query({
prompt,
// Re-enable TodoWrite, which this tracker watches for.
options: { maxTurns: 20, env: { ...process.env, CLAUDE_CODE_ENABLE_TASKS: "0" } }
})) {
if (message.type === "assistant") {
for (const block of message.message.content) {
if (block.type === "tool_use" && block.name === "TodoWrite") {
this.todos = block.input.todos;
this.displayProgress();
}
}
}
}
}
}
// Usage
const tracker = new TodoTracker();
await tracker.trackQuery("Build a complete authentication system with todos");
from claude_agent_sdk import query, ClaudeAgentOptions, AssistantMessage, ToolUseBlock
from typing import List, Dict
class TodoTracker:
def __init__(self):
self.todos: List[Dict] = []
def display_progress(self):
if not self.todos:
return
completed = len([t for t in self.todos if t["status"] == "completed"])
in_progress = len([t for t in self.todos if t["status"] == "in_progress"])
total = len(self.todos)
print(f"\nProgress: {completed}/{total} completed")
print(f"Currently working on: {in_progress} task(s)\n")
for i, todo in enumerate(self.todos):
icon = (
"✅"
if todo["status"] == "completed"
else "🔧"
if todo["status"] == "in_progress"
else "❌"
)
text = (
todo["activeForm"]
if todo["status"] == "in_progress"
else todo["content"]
)
print(f"{i + 1}. {icon} {text}")
async def track_query(self, prompt: str):
async for message in query(
prompt=prompt,
# Re-enable TodoWrite, which this tracker watches for.
options=ClaudeAgentOptions(max_turns=20, env={"CLAUDE_CODE_ENABLE_TASKS": "0"}),
):
if isinstance(message, AssistantMessage):
for block in message.content:
if isinstance(block, ToolUseBlock) and block.name == "TodoWrite":
self.todos = block.input["todos"]
self.display_progress()
# Usage
tracker = TodoTracker()
await tracker.track_query("Build a complete authentication system with todos")
迁移到 Task 工具
Task 工具将单个 TodoWrite 调用拆分为 TaskCreate(用于每个新项目)和 TaskUpdate(用于每次状态变更),同时提供 TaskList 和 TaskGet 供模型读取当前列表。你的监控代码仍然检查助手流中的 tool_use 块,但改为维护一个以任务 ID 为键的映射,而不是在每次调用时替换整个列表。 从 TypeScript Agent SDK 0.3.142 和 Claude Code v2.1.142 起,Task 工具已成为默认选项,因此无需更改 options.env。
使用 TodoWrite | 使用 Task 工具 |
|---|---|
单次工具调用重写整个 todos 数组 | TaskCreate 添加一个项目,TaskUpdate 按 taskId 修补一个项目 |
匹配 block.name === "TodoWrite" | 匹配 block.name === "TaskCreate" 或 "TaskUpdate" |
项目形状:{ content, status, activeForm } | TaskCreate 输入:{ subject, description, activeForm?, metadata? }。TaskUpdate 输入:{ taskId, status?, subject?, description?, activeForm?, addBlocks?, addBlockedBy?, owner?, metadata? }。status 为 "pending"、"in_progress" 或 "completed";设置 status: "deleted" 可删除 |
直接渲染 block.input.todos | 跨调用累积项目,或从 TaskList 工具结果读取快照 |
分配的任务 ID 不在 TaskCreate 输入中。它在匹配的 tool_result 中以 { task: { id, subject } } 返回,因此需要从结果块中捕获它作为映射的键。以下示例展示了监控待办事项变更循环的最小更改。要渲染完整列表,请在流中监听 TaskList 工具结果,或将 TaskCreate 结果和 TaskUpdate 输入累积到映射中:
import { query } from "@anthropic-ai/claude-agent-sdk";
for await (const message of query({
prompt: "Optimize my React app performance",
})) {
if (message.type !== "assistant") continue;
for (const block of message.message.content) {
if (block.type !== "tool_use") continue;
if (block.name === "TaskCreate") {
const input = block.input as { subject: string };
console.log(`+ ${input.subject}`);
} else if (block.name === "TaskUpdate") {
const input = block.input as { taskId: string; status?: string };
if (input.status) console.log(` ${input.taskId} -> ${input.status}`);
}
}
}
from claude_agent_sdk import query, AssistantMessage, ToolUseBlock
async for message in query(
prompt="Optimize my React app performance",
):
if not isinstance(message, AssistantMessage):
continue
for block in message.content:
if not isinstance(block, ToolUseBlock):
continue
if block.name == "TaskCreate":
print(f"+ {block.input['subject']}")
elif block.name == "TaskUpdate" and block.input.get("status"):
print(f" {block.input['taskId']} -> {block.input['status']}")