diff --git a/src/views/home/agent/AGENTS.md b/src/views/home/agent/AGENTS.md
new file mode 100644
index 0000000..60e255a
--- /dev/null
+++ b/src/views/home/agent/AGENTS.md
@@ -0,0 +1,125 @@
+# Agent 页面 — 开发者指南
+
+## 目录结构
+
+```
+agent/
+├── index.vue # 主入口:布局 + 路由 + 事件注册
+├── AGENTS.md
+└── components/
+ ├── Agent.vue # AI 对话核心组件
+ ├── List.vue # 消息列表容器(自动滚动)
+ ├── Item.vue # 单条消息渲染(用户/AI)
+ ├── Preview.vue # 右侧预览面板(sketch/report/url)
+ ├── Menu.vue # 三点菜单(仅 UI,暂无逻辑)
+ ├── Pause.vue # 生成暂停提示条
+ ├── ReportCard.vue # 报告/URL/草图的卡片插槽组件
+ ├── UrlCard.vue # URL 卡片(包裹 ReportCard)
+ ├── SketchCard.vue # 草图卡片(包裹 ReportCard)
+ └── versionTree/ # 版本树抽屉面板
+ ├── index.vue # 版本树入口(el-drawer)
+ ├── detail/ # 版本详情/聊天详情
+ ├── tree/ # 树形视图组件
+ ├── components/ # 版本树内部子组件
+ └── tools/ # 版本数据工具
+```
+
+## 数据流
+
+### 1. 项目加载流程
+```
+路由参数 id → index.vue:handleGetProjectInfoAndHistory()
+ → getProjectInfo API → Agent.vue:setChatInfo(conversation, project)
+ → messageList + sketchList 填充 → Preview 渲染
+```
+
+### 2. 对话发送流程
+```
+用户输入 → Input.vue @send
+ → Agent.vue:handleSendMessage()
+ → SSE GET 请求(/api/ai-design/chat)
+ → 逐行解析 event/data → 实时更新 messageList
+ → 草图/报告/URL 事件 → Preview 切换模式
+```
+
+### 3. 跨组件通信(MyEvent 事件总线)
+
+| 事件名 | 触发方 | 监听方 | 说明 |
+|--------|--------|--------|------|
+| `openReport` | Item.vue | index.vue | 点击报告卡片,Preview 切 report 模式 |
+| `openUrls` | Item.vue | index.vue | 点击 URL 卡片,Preview 切 url 模式 |
+| `openSketch` | Item.vue | index.vue | 点击草图卡片,Preview 切 sketch 模式 |
+| `quote` | Preview.vue | Input 组件 | 引用草图图片到输入框 |
+| `openFlowCanvas` | Preview.vue | Flow Canvas | 编辑草图 |
+| `resetAgent` | 外部 (Flow Canvas) | Agent.vue | 重置对话状态 |
+| `closeFlowCanvas` | 外部 | index.vue | 关闭 Flow Canvas 后刷新 |
+| `stopChat` | Agent.vue | Item.vue | 停止生成时显示暂停提示 |
+| `newTitle` | Agent.vue | 对话列表 | 更新项目标题 |
+| `projectChange` | index.vue | 外部 | 项目切换通知 |
+| `renameConversation` | 外部 | index.vue | 重命名项目时更新标题 |
+
+### 4. Store 使用
+
+| Store | 用途 |
+|-------|------|
+| `projectStore` | 当前项目 ID、nodeId、type/region/style/temperature |
+| `userStore` | 用户 token、头像 |
+| `agentStore` | 初始项目数据(跨页面传递首次对话参数) |
+
+## SSE 协议
+
+请求 URL: `/api/ai-design/chat` (GET)
+参数: `AgentParamsType` (message, projectID, versionID, configParams, token 等)
+
+响应格式: Server-Sent Events (SSE)
+
+### 事件类型
+
+| event 名称 | data 格式 | 作用 |
+|-----------|-----------|------|
+| `message` (默认) | `{"content": "..."}` | AI 回复文本流 |
+| `reasoning` | `{"reasoning": "..."}` | AI 推理过程 |
+| `nodeId` | 纯文本 `versionId` | 记录当前对话节点 ID |
+| `sketchIDAndUrl` | `{"key": "url", ...}` | 生成的草图键值对 |
+| `report` | `{"report": "..."}` | 报告内容流 |
+| `reportName` / `reportTitle` | `{"reportName": "..."}` | 报告名称 |
+| `tool` | 无 | 草图工具调用中 |
+| `title` | `{"title": "..."}` | 自动生成的对话标题 |
+| `webAddress` | `{"webAddress": "[...]"}` | 引用的网页来源 |
+| `error` | 无 | 服务端错误 |
+| `todo` | 无 | 忽略事件 |
+| `end` | `{"type": "end"}` 或 `[DONE]` | 流结束 |
+
+## 关键组件说明
+
+### Agent.vue
+- **`setChatInfo(info)`**: 加载历史对话,解析 `ancestors` + `current` 结构,合并连续 assistant 消息
+- **`handleSendMessage()`**: 核心方法,处理发消息、SSE 流式接收、解析各种事件
+- **`handlePause()`**: 通过 `AbortController.abort()` 中止请求
+- **`processDialogue()`**: 将原始对话数组中的连续 assistant 消息合并为一条,追加插槽标记
+- **`mergeUniqueKeys()`**: 防止草图片列表插入重复 key
+
+### Item.vue
+- 使用 `VueMarkdown` + `rehype-raw` 渲染 Markdown
+- 自定义插槽: `s-card`(报告), `s-url`(网页), `s-sketch`(草图)
+- 用户消息显示参数标签(type/area/style)
+- 仅最后一条 AI 消息显示操作按钮(点赞/点踩/重新生成/复制)
+
+### Preview.vue
+- **sketch 模式**: 4列网格展示草图,每个图片支持引用/编辑(Flow Canvas)/删除
+- **report 模式**: 渲染 Markdown 报告,支持下载 `.md` 文件
+- **url 模式**: 网页来源卡片,点击跳转
+- 使用自定义指令 `v-img-loading` 实现图片预加载占位
+
+### VersionTree (versionTree/index.vue)
+- `el-drawer` 抽屉面板,73.5rem 宽度
+- 加载版本树数据 → 递归遍历添加 `versionId` 路径标识
+- 选中节点可恢复对话 (Restore) 或导出图片 (Export)
+
+## 注意事项
+
+1. **AbortController** 每次请求前重新创建,旧的未完成请求会被中止
+2. **KeepAlive** 包裹 Agent 组件,切换项目时通过 `:key="proJectId"` 强制重建
+3. `onActivated` 处理缓存激活时的新队列数据(后台对话完成的节点/标题/草图)
+4. SSE 解析先用 `\n\n` 拆分事件块,再按 `\n` 解析 event/data 行,不完整的 JSON 放回 buffer
+5. 草图删除调用 `deleteSketchFlowCanvas` API 后需在父组件同步清理 sketchList
diff --git a/src/views/home/agent/components/Agent.vue b/src/views/home/agent/components/Agent.vue
index 2969365..1c1aab6 100644
--- a/src/views/home/agent/components/Agent.vue
+++ b/src/views/home/agent/components/Agent.vue
@@ -454,11 +454,12 @@
const jsonData = JSON.parse(jsonText)
// console.log('jsonData', jsonData)
if (jsonData.webAddress) {
-
- aiMessage.webAddress = JSON.parse(jsonData.webAddress)
- // contentBody += ``
- // aiMessage.loading = false
+ const parsed = JSON.parse(jsonData.webAddress)
+ aiMessage.webAddress = parsed
hasUrlEvent = true
+ if (String(aiMessage.sessionId) === String(projectStore.state.id)) {
+ MyEvent.emit('openUrls', parsed)
+ }
}
if (jsonData.title) {
if (aiMessage.sessionId === projectStore.state.id) {
diff --git a/src/views/home/agent/components/Preview.vue b/src/views/home/agent/components/Preview.vue
index 448f51e..88a413e 100644
--- a/src/views/home/agent/components/Preview.vue
+++ b/src/views/home/agent/components/Preview.vue
@@ -129,6 +129,16 @@
}
)
+ watch(
+ () => props.type,
+ (val) => {
+ if (val === 'sketch') {
+ fetchedUrlSet.value = new Set()
+ urlList.value = []
+ }
+ }
+ )
+
watch(
() => props.sketchList,
() => {
@@ -199,13 +209,16 @@
}
const urlLoading = ref(false)
+ const fetchedUrlSet = ref>(new Set())
const setUrls = async (list: string[]) => {
reportType.value = 'urls'
- urlList.value = []
+ const newUrls = [...new Set(list.filter((url) => !fetchedUrlSet.value.has(url)))]
+ if (newUrls.length === 0) return
urlLoading.value = true
- const res = await fetchUrlTitle(list)
+ const res = await fetchUrlTitle(newUrls)
urlLoading.value = false
- urlList.value = res
+ newUrls.forEach((url) => fetchedUrlSet.value.add(url))
+ urlList.value = [...urlList.value, ...res]
}
// watch(
// () => sessionId.value,