0eba9e87e9
- Disable formatting rules (handled by Prettier) - Relaxed strict Vue/JS rules for demo code compatibility - Fix syntax errors in ApiPlayground and VoiceCloningDemo - Fix duplicate else-if condition in ApiPlayground - Fix Promise executor async pattern in AutoregressiveAudioDemo - Add TypeScript file support to ESLint config Warnings reduced from 295 to 251 problems. Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com>
599 lines
14 KiB
Vue
599 lines
14 KiB
Vue
<template>
|
||
<div class="planning-demo">
|
||
<div class="header">
|
||
<div class="title">
|
||
📋 Agent 的规划能力
|
||
</div>
|
||
</div>
|
||
|
||
<!-- 任务选择 -->
|
||
<div class="task-tabs">
|
||
<button
|
||
v-for="task in tasks"
|
||
:key="task.id"
|
||
:class="['task-btn', { active: currentTask === task.id }]"
|
||
@click="selectTask(task.id)"
|
||
>
|
||
<span>{{ task.icon }}</span>
|
||
<span>{{ task.name }}</span>
|
||
<span
|
||
class="complexity"
|
||
:class="task.complexity"
|
||
>{{ task.complexityLabel }}</span>
|
||
</button>
|
||
</div>
|
||
|
||
<!-- 目标 -->
|
||
<div class="goal-bar">
|
||
<span class="label">🎯</span>
|
||
<span class="text">{{ currentTaskData.goal }}</span>
|
||
</div>
|
||
|
||
<!-- 执行区域 -->
|
||
<div class="execution-area">
|
||
<!-- 步骤进度条 -->
|
||
<div class="steps-progress">
|
||
<div
|
||
v-for="(step, index) in currentTaskData.steps"
|
||
:key="index"
|
||
class="step-node"
|
||
:class="{ completed: stepStatus[index] === 'completed', running: stepStatus[index] === 'running' }"
|
||
>
|
||
<div class="node-circle">
|
||
{{ index + 1 }}
|
||
</div>
|
||
<div class="node-name">
|
||
{{ step.name }}
|
||
</div>
|
||
<div
|
||
v-if="index < currentTaskData.steps.length - 1"
|
||
class="node-line"
|
||
/>
|
||
</div>
|
||
</div>
|
||
|
||
<!-- 日志和思考 -->
|
||
<div class="info-row">
|
||
<div class="log-box">
|
||
<div class="box-header">
|
||
<span>📝 执行日志</span>
|
||
<span
|
||
v-if="executionStatus === 'running'"
|
||
class="status running"
|
||
>执行中</span>
|
||
<span
|
||
v-else-if="executionStatus === 'completed'"
|
||
class="status completed"
|
||
>已完成</span>
|
||
</div>
|
||
<div class="log-content">
|
||
<div
|
||
v-if="logs.length === 0"
|
||
class="empty"
|
||
>
|
||
点击"开始执行"查看过程
|
||
</div>
|
||
<div
|
||
v-for="(log, i) in logs.slice(-4)"
|
||
:key="i"
|
||
class="log-line"
|
||
:class="log.type"
|
||
>
|
||
<span class="time">{{ log.time }}</span>
|
||
<span class="icon">{{ log.icon }}</span>
|
||
<span
|
||
class="msg"
|
||
v-html="log.message"
|
||
/>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
|
||
<div
|
||
v-if="currentThought"
|
||
class="thought-box"
|
||
>
|
||
<div class="box-header">
|
||
🧠 正在思考
|
||
</div>
|
||
<div class="thought-content">
|
||
{{ currentThought }}
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
|
||
<!-- 控制栏 -->
|
||
<div class="control-bar">
|
||
<button
|
||
v-if="executionStatus === 'idle'"
|
||
class="ctrl-btn primary"
|
||
@click="startExecution"
|
||
>
|
||
▶ 开始执行
|
||
</button>
|
||
<button
|
||
v-else-if="executionStatus === 'running'"
|
||
class="ctrl-btn"
|
||
disabled
|
||
>
|
||
⏳ 执行中...
|
||
</button>
|
||
<button
|
||
v-else
|
||
class="ctrl-btn"
|
||
@click="reset"
|
||
>
|
||
🔄 重置
|
||
</button>
|
||
|
||
<div
|
||
v-if="executionStatus === 'completed'"
|
||
class="stats"
|
||
>
|
||
<span class="stat">{{ currentTaskData.steps.length }} 步骤</span>
|
||
<span class="stat">{{ executionTime }}s</span>
|
||
<span class="stat">{{ toolCalls }} 调用</span>
|
||
</div>
|
||
|
||
<div class="step-dots">
|
||
<span
|
||
v-for="n in currentTaskData.steps.length"
|
||
:key="n"
|
||
:class="['dot', { active: stepStatus[n-1] === 'completed' }]"
|
||
/>
|
||
</div>
|
||
</div>
|
||
|
||
<!-- 提示 -->
|
||
<div class="tip-bar">
|
||
<span>💡</span>
|
||
<span>规划核心:将复杂任务分解为<strong>原子操作</strong>,根据上一步结果<strong>动态调整</strong>后续计划</span>
|
||
</div>
|
||
</div>
|
||
</template>
|
||
|
||
<script setup>
|
||
import { ref, computed, nextTick } from 'vue'
|
||
|
||
const tasks = [
|
||
{
|
||
id: 'simple',
|
||
icon: '🌤️',
|
||
name: '查天气',
|
||
complexity: 'easy',
|
||
complexityLabel: '简单',
|
||
goal: '查询北京今天的天气',
|
||
steps: [
|
||
{ name: '调用天气 API', tool: 'weather_api' },
|
||
{ name: '格式化结果', tool: 'formatter' }
|
||
],
|
||
logs: [
|
||
{ type: 'think', icon: '🧠', message: '需要查询北京天气' },
|
||
{ type: 'action', icon: '🔧', message: 'weather_api(city="北京")' },
|
||
{ type: 'result', icon: '📥', message: '晴, 25°C, 空气质量良' },
|
||
{ type: 'complete', icon: '✅', message: '北京今天天气晴朗' }
|
||
]
|
||
},
|
||
{
|
||
id: 'medium',
|
||
icon: '📊',
|
||
name: '数据分析',
|
||
complexity: 'medium',
|
||
complexityLabel: '中等',
|
||
goal: '分析销售 CSV,找出销售额最高月份',
|
||
steps: [
|
||
{ name: '读取 CSV', tool: 'file_reader' },
|
||
{ name: '解析数据', tool: 'data_parser' },
|
||
{ name: '聚合计算', tool: 'calculator' },
|
||
{ name: '生成报告', tool: 'report_generator' }
|
||
],
|
||
logs: [
|
||
{ type: 'think', icon: '🧠', message: '读取销售数据文件' },
|
||
{ type: 'action', icon: '🔧', message: 'file_reader(path="sales.csv")' },
|
||
{ type: 'result', icon: '📥', message: '读取 1200 行数据' },
|
||
{ type: 'think', icon: '🧠', message: '解析数据结构' },
|
||
{ type: 'action', icon: '🔧', message: 'data_parser(data)' },
|
||
{ type: 'result', icon: '📥', message: '解析完成' },
|
||
{ type: 'think', icon: '🧠', message: '按月份聚合销售额' },
|
||
{ type: 'action', icon: '🔧', message: 'calculator.aggregate(by="month")' },
|
||
{ type: 'result', icon: '📥', message: '11月销售额最高 ¥320K' },
|
||
{ type: 'complete', icon: '✅', message: '分析完成' }
|
||
]
|
||
},
|
||
{
|
||
id: 'complex',
|
||
icon: '🔬',
|
||
name: '研究报告',
|
||
complexity: 'hard',
|
||
complexityLabel: '复杂',
|
||
goal: '调研 AI Agent 进展,撰写完整报告',
|
||
steps: [
|
||
{ name: '搜索资讯', tool: 'web_search' },
|
||
{ name: '阅读文章', tool: 'web_reader' },
|
||
{ name: '提取信息', tool: 'extractor' },
|
||
{ name: '搜索厂商', tool: 'web_search' },
|
||
{ name: '生成大纲', tool: 'planner' },
|
||
{ name: '撰写报告', tool: 'writer' }
|
||
],
|
||
logs: [
|
||
{ type: 'think', icon: '🧠', message: '搜索最新 AI Agent 资讯' },
|
||
{ type: 'action', icon: '🔧', message: 'web_search("AI Agent 2024")' },
|
||
{ type: 'result', icon: '📥', message: '找到 15 篇文章' },
|
||
{ type: 'action', icon: '🔧', message: 'web_reader(urls=[...])' },
|
||
{ type: 'result', icon: '📥', message: '成功读取内容' },
|
||
{ type: 'action', icon: '🔧', message: 'extractor(fields=[...])' },
|
||
{ type: 'result', icon: '📥', message: '提取 45 个数据点' },
|
||
{ type: 'action', icon: '🔧', message: 'web_search("AI Agent companies")' },
|
||
{ type: 'result', icon: '📥', message: 'OpenAI, Anthropic, Microsoft...' },
|
||
{ type: 'action', icon: '🔧', message: 'planner.generate_outline()' },
|
||
{ type: 'result', icon: '📥', message: '大纲生成完成' },
|
||
{ type: 'action', icon: '🔧', message: 'writer.generate_content()' },
|
||
{ type: 'complete', icon: '✅', message: '报告生成完成,2500字' }
|
||
]
|
||
}
|
||
]
|
||
|
||
const currentTask = ref('simple')
|
||
const executionStatus = ref('idle')
|
||
const stepStatus = ref([])
|
||
const logs = ref([])
|
||
const currentThought = ref('')
|
||
const executionTime = ref(0)
|
||
const toolCalls = ref(0)
|
||
|
||
const currentTaskData = computed(() => tasks.find(t => t.id === currentTask.value))
|
||
|
||
const selectTask = (id) => {
|
||
currentTask.value = id
|
||
reset()
|
||
}
|
||
|
||
const reset = () => {
|
||
executionStatus.value = 'idle'
|
||
stepStatus.value = new Array(currentTaskData.value.steps.length).fill('pending')
|
||
logs.value = []
|
||
currentThought.value = ''
|
||
executionTime.value = 0
|
||
toolCalls.value = 0
|
||
}
|
||
|
||
const startExecution = async () => {
|
||
executionStatus.value = 'running'
|
||
stepStatus.value = new Array(currentTaskData.value.steps.length).fill('pending')
|
||
logs.value = []
|
||
toolCalls.value = 0
|
||
|
||
const startTime = Date.now()
|
||
const taskLogs = currentTaskData.value.logs
|
||
|
||
for (let i = 0; i < taskLogs.length; i++) {
|
||
const log = taskLogs[i]
|
||
|
||
if (log.type === 'think') currentThought.value = log.message
|
||
if (log.type === 'action') {
|
||
const stepIndex = Math.min(toolCalls.value, currentTaskData.value.steps.length - 1)
|
||
stepStatus.value = stepStatus.value.map((s, idx) => {
|
||
if (idx < stepIndex) return 'completed'
|
||
if (idx === stepIndex) return 'running'
|
||
return 'pending'
|
||
})
|
||
toolCalls.value++
|
||
}
|
||
if (log.type === 'complete') currentThought.value = ''
|
||
|
||
logs.value.push({ ...log, time: new Date().toLocaleTimeString('zh-CN', { hour12: false, hour: '2-digit', minute: '2-digit', second: '2-digit' }) })
|
||
await wait(700)
|
||
}
|
||
|
||
stepStatus.value = stepStatus.value.map(() => 'completed')
|
||
executionTime.value = ((Date.now() - startTime) / 1000).toFixed(1)
|
||
executionStatus.value = 'completed'
|
||
}
|
||
|
||
const wait = (ms) => new Promise(resolve => setTimeout(resolve, ms))
|
||
reset()
|
||
</script>
|
||
|
||
<style scoped>
|
||
.planning-demo {
|
||
background: var(--vp-c-bg-soft);
|
||
border: 1px solid var(--vp-c-divider);
|
||
border-radius: 12px;
|
||
padding: 20px;
|
||
margin: 20px 0;
|
||
font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, sans-serif;
|
||
}
|
||
|
||
.header {
|
||
text-align: center;
|
||
margin-bottom: 16px;
|
||
}
|
||
|
||
.title {
|
||
font-size: 17px;
|
||
font-weight: 700;
|
||
background: linear-gradient(120deg, var(--vp-c-brand), #9c27b0);
|
||
-webkit-background-clip: text;
|
||
-webkit-text-fill-color: transparent;
|
||
}
|
||
|
||
/* 任务标签 */
|
||
.task-tabs {
|
||
display: flex;
|
||
gap: 8px;
|
||
justify-content: center;
|
||
margin-bottom: 16px;
|
||
flex-wrap: wrap;
|
||
}
|
||
|
||
.task-btn {
|
||
display: flex;
|
||
align-items: center;
|
||
gap: 6px;
|
||
padding: 8px 14px;
|
||
border: 1px solid var(--vp-c-divider);
|
||
border-radius: 20px;
|
||
background: var(--vp-c-bg);
|
||
cursor: pointer;
|
||
font-size: 13px;
|
||
transition: all 0.2s;
|
||
}
|
||
|
||
.task-btn.active {
|
||
border-color: var(--vp-c-brand);
|
||
background: var(--vp-c-brand-soft);
|
||
color: var(--vp-c-brand-dark);
|
||
}
|
||
|
||
.complexity {
|
||
padding: 2px 6px;
|
||
border-radius: 4px;
|
||
font-size: 10px;
|
||
margin-left: 4px;
|
||
}
|
||
|
||
.complexity.easy { background: #dcfce7; color: #166534; }
|
||
.complexity.medium { background: #fef3c7; color: #92400e; }
|
||
.complexity.hard { background: #fee2e2; color: #991b1b; }
|
||
|
||
/* 目标 */
|
||
.goal-bar {
|
||
background: var(--vp-c-brand-soft);
|
||
border-left: 3px solid var(--vp-c-brand);
|
||
border-radius: 6px;
|
||
padding: 10px 14px;
|
||
margin-bottom: 16px;
|
||
font-size: 14px;
|
||
}
|
||
|
||
.goal-bar .label { margin-right: 8px; }
|
||
.goal-bar .text { font-weight: 600; }
|
||
|
||
/* 步骤进度 */
|
||
.steps-progress {
|
||
display: flex;
|
||
align-items: flex-start;
|
||
gap: 8px;
|
||
margin-bottom: 16px;
|
||
overflow-x: auto;
|
||
padding-bottom: 8px;
|
||
}
|
||
|
||
.step-node {
|
||
display: flex;
|
||
flex-direction: column;
|
||
align-items: center;
|
||
position: relative;
|
||
min-width: 100px;
|
||
}
|
||
|
||
.node-circle {
|
||
width: 32px;
|
||
height: 32px;
|
||
border-radius: 50%;
|
||
background: var(--vp-c-bg);
|
||
border: 2px solid var(--vp-c-divider);
|
||
display: flex;
|
||
align-items: center;
|
||
justify-content: center;
|
||
font-size: 12px;
|
||
font-weight: 700;
|
||
margin-bottom: 6px;
|
||
transition: all 0.3s;
|
||
}
|
||
|
||
.step-node.running .node-circle {
|
||
border-color: var(--vp-c-brand);
|
||
background: var(--vp-c-brand-soft);
|
||
animation: pulse 1.5s infinite;
|
||
}
|
||
|
||
.step-node.completed .node-circle {
|
||
border-color: #22c55e;
|
||
background: #dcfce7;
|
||
color: #166534;
|
||
}
|
||
|
||
@keyframes pulse {
|
||
0%, 100% { transform: scale(1); }
|
||
50% { transform: scale(1.1); }
|
||
}
|
||
|
||
.node-name {
|
||
font-size: 11px;
|
||
text-align: center;
|
||
color: var(--vp-c-text-2);
|
||
}
|
||
|
||
.step-node.completed .node-name,
|
||
.step-node.running .node-name {
|
||
color: var(--vp-c-text-1);
|
||
font-weight: 600;
|
||
}
|
||
|
||
.node-line {
|
||
position: absolute;
|
||
top: 16px;
|
||
right: -16px;
|
||
width: 24px;
|
||
height: 2px;
|
||
background: var(--vp-c-divider);
|
||
}
|
||
|
||
.step-node.completed + .step-node .node-line {
|
||
background: #22c55e;
|
||
}
|
||
|
||
/* 信息行 */
|
||
.info-row {
|
||
display: grid;
|
||
grid-template-columns: 2fr 1fr;
|
||
gap: 12px;
|
||
margin-bottom: 16px;
|
||
}
|
||
|
||
@media (max-width: 600px) {
|
||
.info-row { grid-template-columns: 1fr; }
|
||
}
|
||
|
||
.log-box, .thought-box {
|
||
background: var(--vp-c-bg);
|
||
border: 1px solid var(--vp-c-divider);
|
||
border-radius: 6px;
|
||
overflow: hidden;
|
||
}
|
||
|
||
.box-header {
|
||
display: flex;
|
||
justify-content: space-between;
|
||
align-items: center;
|
||
padding: 10px 12px;
|
||
background: var(--vp-c-bg-soft);
|
||
border-bottom: 1px solid var(--vp-c-divider);
|
||
font-size: 12px;
|
||
font-weight: 600;
|
||
}
|
||
|
||
.status {
|
||
padding: 2px 8px;
|
||
border-radius: 4px;
|
||
font-size: 11px;
|
||
}
|
||
|
||
.status.running { background: #fef3c7; color: #92400e; }
|
||
.status.completed { background: #dcfce7; color: #166534; }
|
||
|
||
.log-content {
|
||
padding: 10px 12px;
|
||
min-height: 100px;
|
||
max-height: 140px;
|
||
|
||
}
|
||
|
||
.empty {
|
||
color: var(--vp-c-text-3);
|
||
text-align: center;
|
||
padding: 30px 0;
|
||
font-size: 12px;
|
||
}
|
||
|
||
.log-line {
|
||
display: flex;
|
||
gap: 8px;
|
||
font-size: 12px;
|
||
margin-bottom: 6px;
|
||
align-items: flex-start;
|
||
}
|
||
|
||
.log-line .time {
|
||
color: var(--vp-c-text-3);
|
||
font-size: 10px;
|
||
min-width: 55px;
|
||
}
|
||
|
||
.log-line .icon {
|
||
font-size: 11px;
|
||
}
|
||
|
||
.log-line .msg {
|
||
color: var(--vp-c-text-1);
|
||
flex: 1;
|
||
}
|
||
|
||
.log-line.think .msg { color: #3b82f6; }
|
||
.log-line.action .msg { color: #f59e0b; }
|
||
.log-line.result .msg { color: #10b981; }
|
||
.log-line.complete .msg { color: #8b5cf6; font-weight: 600; }
|
||
|
||
.thought-content {
|
||
padding: 12px;
|
||
font-size: 13px;
|
||
color: var(--vp-c-text-1);
|
||
font-style: italic;
|
||
line-height: 1.5;
|
||
}
|
||
|
||
/* 控制栏 */
|
||
.control-bar {
|
||
display: flex;
|
||
justify-content: space-between;
|
||
align-items: center;
|
||
margin-bottom: 16px;
|
||
}
|
||
|
||
.ctrl-btn {
|
||
padding: 8px 18px;
|
||
border: 1px solid var(--vp-c-divider);
|
||
border-radius: 6px;
|
||
background: var(--vp-c-bg);
|
||
cursor: pointer;
|
||
font-size: 13px;
|
||
font-weight: 600;
|
||
}
|
||
|
||
.ctrl-btn.primary {
|
||
background: var(--vp-c-brand);
|
||
color: white;
|
||
border-color: var(--vp-c-brand);
|
||
}
|
||
|
||
.stats {
|
||
display: flex;
|
||
gap: 12px;
|
||
}
|
||
|
||
.stat {
|
||
padding: 4px 10px;
|
||
background: var(--vp-c-bg);
|
||
border-radius: 4px;
|
||
font-size: 12px;
|
||
color: var(--vp-c-text-2);
|
||
}
|
||
|
||
.step-dots {
|
||
display: flex;
|
||
gap: 4px;
|
||
}
|
||
|
||
.dot {
|
||
width: 6px;
|
||
height: 6px;
|
||
border-radius: 50%;
|
||
background: var(--vp-c-divider);
|
||
}
|
||
|
||
.dot.active { background: #22c55e; }
|
||
|
||
/* 提示 */
|
||
.tip-bar {
|
||
display: flex;
|
||
gap: 8px;
|
||
padding: 10px 14px;
|
||
background: var(--vp-c-brand-soft);
|
||
border-radius: 6px;
|
||
font-size: 12px;
|
||
color: var(--vp-c-text-1);
|
||
}
|
||
</style>
|