sillyspec 3.9.1 → 3.10.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (69) hide show
  1. package/.claude/skills/sillyspec-commit/SKILL.md +1 -1
  2. package/.claude/skills/sillyspec-continue/SKILL.md +1 -1
  3. package/.claude/skills/sillyspec-explore/SKILL.md +9 -0
  4. package/.claude/skills/sillyspec-plan/SKILL.md +0 -35
  5. package/.claude/skills/sillyspec-resume/SKILL.md +5 -5
  6. package/.claude/skills/sillyspec-state/SKILL.md +1 -1
  7. package/.claude/skills/sillyspec-workspace/SKILL.md +1 -1
  8. package/README.md +7 -6
  9. package/SKILL.md +15 -10
  10. package/package.json +1 -1
  11. package/packages/dashboard/dist/assets/index-BcM2J-hv.css +1 -0
  12. package/packages/dashboard/dist/assets/{index-RsLVPAy7.js → index-DpLHK4jv.js} +974 -974
  13. package/packages/dashboard/dist/index.html +16 -17
  14. package/packages/dashboard/dist/prototype-dashboard.html +836 -0
  15. package/packages/dashboard/dist/prototype-overview.html +256 -0
  16. package/packages/dashboard/public/prototype-dashboard.html +836 -0
  17. package/packages/dashboard/public/prototype-overview.html +256 -0
  18. package/packages/dashboard/server/index.js +18 -13
  19. package/packages/dashboard/server/parser.js +109 -1
  20. package/packages/dashboard/server/watcher.js +14 -6
  21. package/packages/dashboard/src/App.vue +414 -186
  22. package/packages/dashboard/src/components/ActionBar.vue +10 -1
  23. package/packages/dashboard/src/components/CommandPalette.vue +5 -1
  24. package/packages/dashboard/src/components/DocPreview.vue +105 -8
  25. package/packages/dashboard/src/components/DocTree.vue +75 -19
  26. package/packages/dashboard/src/components/HResizeHandle.vue +48 -0
  27. package/packages/dashboard/src/components/PipelineView.vue +23 -4
  28. package/packages/dashboard/src/components/ProjectCard.vue +187 -0
  29. package/packages/dashboard/src/components/ProjectOverview.vue +113 -139
  30. package/packages/dashboard/src/components/VResizeHandle.vue +61 -0
  31. package/packages/dashboard/src/composables/useDashboard.js +28 -0
  32. package/packages/dashboard/src/composables/useLayout.js +131 -0
  33. package/src/index.js +7 -0
  34. package/src/init.js +17 -10
  35. package/src/migrate.js +5 -5
  36. package/src/progress.js +2 -1
  37. package/src/run.js +141 -64
  38. package/src/stages/archive.js +14 -8
  39. package/src/stages/brainstorm.js +29 -4
  40. package/src/stages/execute.js +115 -35
  41. package/src/stages/explore.js +34 -0
  42. package/src/stages/index.js +11 -6
  43. package/src/stages/plan.js +86 -15
  44. package/src/stages/quick.js +1 -1
  45. package/src/stages/scan.js +51 -12
  46. package/src/stages/status.js +1 -1
  47. package/src/stages/verify.js +35 -6
  48. package/.sillyspec/changes/archive/2026-04-08-derive-state/design.md +0 -97
  49. package/.sillyspec/changes/archive/2026-04-08-derive-state/plan.md +0 -51
  50. package/.sillyspec/changes/archive/2026-04-08-derive-state/proposal.md +0 -29
  51. package/.sillyspec/changes/archive/2026-04-08-derive-state/requirements.md +0 -34
  52. package/.sillyspec/changes/archive/2026-04-08-derive-state/tasks.md +0 -13
  53. package/.sillyspec/changes/archive/2026-04-08-derive-state/verify-result.md +0 -43
  54. package/.sillyspec/changes/auto-mode/design.md +0 -50
  55. package/.sillyspec/changes/auto-mode/proposal.md +0 -19
  56. package/.sillyspec/changes/auto-mode/requirements.md +0 -21
  57. package/.sillyspec/changes/auto-mode/tasks.md +0 -7
  58. package/.sillyspec/changes/brainstorm-archive/2026-04-05-dashboard-design.md +0 -206
  59. package/.sillyspec/changes/brainstorm-archive/2026-04-05-unified-docs-design.md +0 -199
  60. package/.sillyspec/changes/dashboard/design.md +0 -219
  61. package/.sillyspec/changes/dashboard/design.md.braindraft +0 -206
  62. package/.sillyspec/changes/run-command-design/design.md +0 -1230
  63. package/.sillyspec/changes/unified-docs-design/design.md +0 -199
  64. package/.sillyspec/docs/sillyspec/scan/.gitkeep +0 -0
  65. package/.sillyspec/knowledge/INDEX.md +0 -8
  66. package/.sillyspec/knowledge/uncategorized.md +0 -3
  67. package/.sillyspec/plans/2026-04-05-dashboard.md +0 -737
  68. package/.sillyspec/projects/sillyspec.yaml +0 -3
  69. package/packages/dashboard/dist/assets/index-CntACGUN.css +0 -1
@@ -4,91 +4,126 @@
4
4
  <!-- Ambient background -->
5
5
  <div class="absolute inset-0 pointer-events-none" style="background: radial-gradient(ellipse 60% 40% at 10% 20%, rgba(251,191,36,0.04) 0%, transparent 70%), radial-gradient(ellipse 50% 50% at 90% 80%, rgba(251,191,36,0.02) 0%, transparent 70%);" />
6
6
 
7
- <!-- Main Content -->
7
+ <!-- 上层:多项目概览区域 -->
8
8
  <div
9
- class="flex-1 flex overflow-hidden relative z-10"
10
- :style="isDragging ? 'user-select: none' : ''"
9
+ class="relative overflow-hidden flex-shrink-0"
10
+ :style="{ height: layout.overviewHeight + '%' }"
11
11
  >
12
- <!-- Left: Project List -->
13
- <aside
14
- class="flex-shrink-0 relative overflow-hidden"
15
- :style="`width: ${leftWidth}px; background: #FFFFFF; border-right: none;`"
16
- >
17
- <ProjectList
18
- :projects="dashboard.state.projects"
19
- :active-project="dashboard.state.activeProject"
20
- :is-loading="dashboard.state.isLoading"
21
- :scan-paths="scanPaths"
22
- @select="handleSelectProject"
23
- @scan:add-path="handleAddScanPath"
24
- @scan:remove-path="handleRemoveScanPath"
25
- />
26
- </aside>
27
-
28
- <!-- Left ↔ Center resize handle -->
29
- <div
30
- class="w-[2px] flex-shrink-0 cursor-col-resize hover:bg-[#D97706] active:bg-[#D97706] relative z-20"
31
- style="background: #2A3040;"
32
- @mousedown="startDragLeft"
33
- />
34
-
35
- <!-- Center: Pipeline View -->
36
- <main class="flex-1 overflow-hidden accent-stripe flex flex-col" style="min-width: 300px;">
37
- <ProjectOverview :project="dashboard.state.activeProject" @show-detail="handleShowDetail" />
38
- <PipelineView
39
- :project="dashboard.state.activeProject"
40
- :active-step="dashboard.state.activeStep"
41
- :active-tab="dashboard.state.activeTab"
42
- :docs="dashboard.state.docs"
43
- :selected-doc-file="dashboard.state.selectedDocFile"
44
- :doc-content="dashboard.state.docContent"
45
- :doc-loading="dashboard.state.docLoading"
46
- @select-step="handleSelectStep"
47
- @switch-tab="handleSwitchTab"
48
- @select-doc-file="handleSelectDocFile"
49
- />
50
- </main>
51
-
52
- <!-- Center ↔ Right resize handle -->
53
- <div
54
- v-if="dashboard.state.isPanelOpen"
55
- class="w-[2px] flex-shrink-0 cursor-col-resize hover:bg-[#D97706] active:bg-[#D97706] relative z-20"
56
- style="background: #2A3040;"
57
- @mousedown="startDragRight"
12
+ <!-- 项目概览 -->
13
+ <ProjectOverview
14
+ :projects="dashboard.state.projects"
15
+ :active-project="dashboard.state.activeProject"
16
+ @select="handleSelectProject"
58
17
  />
18
+ </div>
59
19
 
60
- <!-- Right: Detail Panel -->
61
- <aside
62
- :class="[
63
- 'flex-shrink-0 relative overflow-hidden',
64
- dashboard.state.isPanelOpen ? '' : 'w-0'
65
- ]"
66
- :style="dashboard.state.isPanelOpen ? `width: ${rightWidth}px; background: #FFFFFF; transition: none;` : 'background: #FFFFFF;'"
67
- >
68
- <DetailPanel
69
- :is-open="dashboard.state.isPanelOpen"
70
- :active-step="dashboard.state.activeStep"
71
- :logs="dashboard.state.logs"
72
- :detail-type="detailType"
73
- :detail-data="detailData"
74
- @close="handleDetailClose"
75
- @clear-logs="dashboard.clearLogs"
76
- @open-doc-file="handleOpenDocFromDetail"
77
- />
78
- </aside>
20
+ <!-- 垂直拖动分割线 -->
21
+ <div
22
+ class="h-[6px] flex-shrink-0 cursor-row-resize hover:bg-[#D97706] active:bg-[#D97706] relative z-20 transition-colors duration-200"
23
+ :class="{ 'bg-[#D97706]': isDragging }"
24
+ style="background: #2A3040;"
25
+ @mousedown="startDragVertical"
26
+ >
27
+ <div class="absolute inset-0 flex items-center justify-center pointer-events-none">
28
+ <div class="w-10 h-1 bg-white/30 rounded-full"></div>
29
+ </div>
79
30
  </div>
80
31
 
81
- <!-- Bottom: Action Bar -->
82
- <ActionBar
83
- :project="dashboard.state.activeProject"
84
- :is-executing="dashboard.state.executingProject !== null"
85
- :execution-result="executionResult"
86
- :is-panel-open="dashboard.state.isPanelOpen"
87
- @execute="handleExecute"
88
- @kill="handleKill"
89
- @toggle-panel="dashboard.togglePanel"
90
- @open-palette="isCommandPaletteOpen = true"
91
- />
32
+ <!-- 下层:详情区域 -->
33
+ <div class="flex-1 overflow-hidden flex flex-col">
34
+ <!-- 详情内容 -->
35
+ <div class="flex-1 flex overflow-hidden">
36
+ <!-- 左栏:项目信息 -->
37
+ <div
38
+ class="detail-column flex-shrink-0 bg-white overflow-hidden"
39
+ :style="{ width: layout.columnWidths[0] + '%', minWidth: '150px' }"
40
+ :class="{ 'fade-in': projectSwitched }"
41
+ >
42
+ <div v-if="activeProject" class="project-info">
43
+ <div class="detail-section-title">项目信息</div>
44
+ <div class="detail-item">
45
+ <div class="detail-label">项目名称</div>
46
+ <div class="detail-value">{{ activeProject.name }}</div>
47
+ </div>
48
+ <div class="detail-item">
49
+ <div class="detail-label">路径</div>
50
+ <div class="detail-value detail-path">{{ shortPath }}</div>
51
+ </div>
52
+ <div class="detail-item">
53
+ <div class="detail-label">当前阶段</div>
54
+ <div class="detail-value">{{ currentStageLabel }}</div>
55
+ </div>
56
+ <div class="detail-item">
57
+ <div class="detail-label">进度</div>
58
+ <div class="detail-value">{{ progressLabel }}</div>
59
+ </div>
60
+ </div>
61
+ <div v-else class="text-gray-400 text-sm">选择项目查看详情</div>
62
+ </div>
63
+
64
+ <!-- 水平拖动分割线 1 -->
65
+ <div
66
+ class="w-[4px] flex-shrink-0 cursor-col-resize hover:bg-[#D97706] active:bg-[#D97706] relative z-20 transition-colors duration-200"
67
+ style="background: #E5E7EB;"
68
+ @mousedown="startDragHorizontal(0)"
69
+ ></div>
70
+
71
+ <!-- 中栏:Pipeline -->
72
+ <div
73
+ class="pipeline-column flex-1 bg-white overflow-hidden"
74
+ :style="{ minWidth: '200px' }"
75
+ >
76
+ <PipelineView
77
+ :project="dashboard.state.activeProject"
78
+ :active-step="dashboard.state.activeStep"
79
+ :active-tab="dashboard.state.activeTab"
80
+ :docs="dashboard.state.docs"
81
+ :selected-doc-file="dashboard.state.selectedDocFile"
82
+ :doc-content="dashboard.state.docContent"
83
+ :doc-loading="dashboard.state.docLoading"
84
+ @select-step="handleSelectStep"
85
+ @switch-tab="handleSwitchTab"
86
+ @select-doc-file="handleSelectDocFile"
87
+ />
88
+ </div>
89
+
90
+ <!-- 水平拖动分割线 2 -->
91
+ <div
92
+ class="w-[4px] flex-shrink-0 cursor-col-resize hover:bg-[#D97706] active:bg-[#D97706] relative z-20 transition-colors duration-200"
93
+ style="background: #E5E7EB;"
94
+ @mousedown="startDragHorizontal(1)"
95
+ ></div>
96
+
97
+ <!-- 右栏:日志/详情 -->
98
+ <div
99
+ class="activity-column flex-shrink-0 bg-white overflow-hidden flex flex-col"
100
+ :style="{ width: layout.columnWidths[2] + '%', minWidth: '200px' }"
101
+ >
102
+ <div class="detail-section-title px-4 pt-4">最近活动</div>
103
+ <div class="flex-1 overflow-y-auto px-4 pb-4">
104
+ <div
105
+ v-for="(log, i) in recentLogs"
106
+ :key="i"
107
+ class="log-entry"
108
+ :class="{ success: log.includes('✅'), error: log.includes('❌') }"
109
+ >
110
+ {{ log }}
111
+ </div>
112
+ <div v-if="recentLogs.length === 0" class="text-gray-400 text-sm">暂无活动</div>
113
+ </div>
114
+ </div>
115
+ </div>
116
+
117
+ <!-- Bottom: Action Bar -->
118
+ <ActionBar
119
+ :project="dashboard.state.activeProject"
120
+ :is-executing="dashboard.state.executingProject !== null"
121
+ :execution-result="executionResult"
122
+ @execute="handleExecute"
123
+ @kill="handleKill"
124
+ @open-palette="isCommandPaletteOpen = true"
125
+ />
126
+ </div>
92
127
 
93
128
  <!-- Command Palette Overlay -->
94
129
  <CommandPalette
@@ -98,110 +133,195 @@
98
133
  @select-project="handleSelectProject"
99
134
  @select-stage="handleSelectStage"
100
135
  />
136
+
137
+ <!-- 尺寸指示器 -->
138
+ <div
139
+ v-if="isDragging"
140
+ class="fixed bottom-5 left-5 bg-black/70 text-white px-3 py-2 rounded text-xs font-mono z-50"
141
+ >
142
+ {{ sizeIndicatorText }}
143
+ </div>
101
144
  </div>
102
145
  </n-config-provider>
103
146
  </template>
104
147
 
105
148
  <script setup>
106
- import { ref, onMounted, readonly } from 'vue'
149
+ import { ref, computed, onMounted } from 'vue'
107
150
  import { useWebSocket } from './composables/useWebSocket.js'
108
151
  import { useDashboard } from './composables/useDashboard.js'
109
- import { useDashboardKeyboard } from './composables/useKeyboard.js'
110
- import ProjectList from './components/ProjectList.vue'
152
+ import { useLayout } from './composables/useLayout.js'
111
153
  import PipelineView from './components/PipelineView.vue'
112
- import DetailPanel from './components/DetailPanel.vue'
113
- import ActionBar from './components/ActionBar.vue'
114
154
  import ProjectOverview from './components/ProjectOverview.vue'
155
+ import ActionBar from './components/ActionBar.vue'
115
156
  import CommandPalette from './components/CommandPalette.vue'
116
157
 
117
158
  // Composables
118
159
  const ws = useWebSocket()
119
160
  const dashboard = useDashboard()
161
+ const layoutManager = useLayout()
162
+
163
+ // 解构 layout
164
+ const { layout, isDragging, startDrag, endDrag } = layoutManager
165
+
166
+ // 状态
120
167
  const isCommandPaletteOpen = ref(false)
121
168
  const executionResult = ref(null)
122
- const scanPaths = ref([])
123
- const detailType = ref(null)
124
- const detailData = ref(null)
125
-
126
- // Panel resize state
127
- const STORAGE_KEY = 'dashboard-panel-widths'
128
- const MIN_LEFT = 180
129
- const MIN_CENTER = 300
130
- const MIN_RIGHT = 260
131
- const DEFAULT_LEFT = 240
132
- const DEFAULT_RIGHT = 340
133
-
134
- const leftWidth = ref(DEFAULT_LEFT)
135
- const rightWidth = ref(DEFAULT_RIGHT)
136
- const isDragging = ref(false)
137
-
138
- // Load persisted widths
139
- try {
140
- const saved = JSON.parse(localStorage.getItem(STORAGE_KEY))
141
- if (saved?.left >= MIN_LEFT) leftWidth.value = saved.left
142
- if (saved?.right >= MIN_RIGHT) rightWidth.value = saved.right
143
- } catch {}
144
-
145
- function persistWidths() {
146
- localStorage.setItem(STORAGE_KEY, JSON.stringify({ left: leftWidth.value, right: rightWidth.value }))
147
- }
169
+ const projectSwitched = ref(false)
170
+ const dragState = ref({
171
+ active: false,
172
+ type: null,
173
+ startX: 0,
174
+ startY: 0,
175
+ startValue: 0
176
+ })
148
177
 
149
- function startDragLeft(e) {
150
- e.preventDefault()
151
- isDragging.value = true
152
- const startX = e.clientX
153
- const startW = leftWidth.value
154
- const onMove = (ev) => {
155
- const delta = ev.clientX - startX
156
- leftWidth.value = Math.max(MIN_LEFT, startW + delta)
178
+ // 尺寸指示器文本
179
+ const sizeIndicatorText = computed(() => {
180
+ if (dragState.value.type === 'vertical') {
181
+ return `概览 ${Math.round(layout.overviewHeight)}% | 详情 ${Math.round(100 - layout.overviewHeight)}%`
182
+ } else if (dragState.value.type === 'horizontal') {
183
+ return `三栏: ${Math.round(layout.columnWidths[0])}% | ${Math.round(layout.columnWidths[1])}% | ${Math.round(layout.columnWidths[2])}%`
157
184
  }
158
- const onUp = () => {
159
- window.removeEventListener('mousemove', onMove)
160
- window.removeEventListener('mouseup', onUp)
161
- isDragging.value = false
162
- persistWidths()
185
+ return ''
186
+ })
187
+
188
+ // 当前项目
189
+ const activeProject = computed(() => dashboard.state.activeProject)
190
+
191
+ // 项目短路径
192
+ const shortPath = computed(() => {
193
+ const path = activeProject.value?.path || ''
194
+ return path.replace(/^\/Users\/[^/]+/, '~')
195
+ })
196
+
197
+ // 当前阶段标签
198
+ const currentStageLabel = computed(() => {
199
+ const stage = activeProject.value?.state?.currentStage
200
+ if (!stage) return '未开始'
201
+ const stageNames = {
202
+ scan: '代码扫描',
203
+ brainstorm: '需求探索',
204
+ plan: '实现计划',
205
+ execute: '波次执行',
206
+ verify: '验证确认',
207
+ archive: '归档变更',
208
+ quick: '快速任务',
209
+ explore: '自由探索'
163
210
  }
164
- window.addEventListener('mousemove', onMove)
165
- window.addEventListener('mouseup', onUp)
166
- }
211
+ return stageNames[stage] || stage
212
+ })
213
+
214
+ // 进度标签
215
+ const progressLabel = computed(() => {
216
+ const progress = activeProject.value?.state?.progress
217
+ if (!progress) return '-'
218
+ const currentStage = activeProject.value?.state?.currentStage
219
+ const stageProgress = currentStage ? progress.stages?.[currentStage] : null
220
+ if (!stageProgress) return '-'
221
+ if (stageProgress.completedSteps !== undefined && stageProgress.steps !== undefined) {
222
+ return `${stageProgress.completedSteps}/${stageProgress.steps} 步骤`
223
+ }
224
+ return stageProgress.status || '-'
225
+ })
167
226
 
168
- function startDragRight(e) {
227
+ // 最近日志(最近 10 条)
228
+ const recentLogs = computed(() => {
229
+ const logs = dashboard.state.logs || []
230
+ return logs.slice(-10).reverse()
231
+ })
232
+
233
+ // 垂直拖动(概览 ↔ 详情)
234
+ function startDragVertical(e) {
169
235
  e.preventDefault()
170
- isDragging.value = true
171
- const startX = e.clientX
172
- const startW = rightWidth.value
236
+ dragState.value = {
237
+ active: true,
238
+ type: 'vertical',
239
+ startY: e.clientY,
240
+ startValue: layout.overviewHeight
241
+ }
242
+ document.body.classList.add('resizing')
243
+ startDrag('vertical')
244
+
173
245
  const onMove = (ev) => {
174
- const delta = startX - ev.clientX
175
- rightWidth.value = Math.max(MIN_RIGHT, startW + delta)
246
+ const deltaY = ev.clientY - dragState.value.startY
247
+ const windowHeight = window.innerHeight
248
+ const deltaPercent = (deltaY / windowHeight) * 100
249
+ const newHeight = dragState.value.startValue + deltaPercent
250
+ layout.overviewHeight = Math.max(15, Math.min(75, newHeight))
176
251
  }
252
+
177
253
  const onUp = () => {
254
+ document.body.classList.remove('resizing')
255
+ dragState.value.active = false
256
+ dragState.value.type = null
178
257
  window.removeEventListener('mousemove', onMove)
179
258
  window.removeEventListener('mouseup', onUp)
180
- isDragging.value = false
181
- persistWidths()
259
+ endDrag()
182
260
  }
261
+
183
262
  window.addEventListener('mousemove', onMove)
184
263
  window.addEventListener('mouseup', onUp)
185
264
  }
186
265
 
187
- // Keyboard shortcuts
188
- useDashboardKeyboard({
189
- onOpenCommandPalette: () => { isCommandPaletteOpen.value = true },
190
- onClose: () => {
191
- if (isCommandPaletteOpen.value) {
192
- isCommandPaletteOpen.value = false
193
- } else if (dashboard.state.activeStep) {
194
- dashboard.state.activeStep = null
195
- } else {
196
- dashboard.closePanel()
266
+ // 水平拖动(三栏分割)
267
+ function startDragHorizontal(colIndex) {
268
+ return (e) => {
269
+ e.preventDefault()
270
+ dragState.value = {
271
+ active: true,
272
+ type: 'horizontal',
273
+ colIndex,
274
+ startX: e.clientX,
275
+ startWidths: [...layout.columnWidths]
276
+ }
277
+ document.body.classList.add('resizing')
278
+ startDrag('horizontal')
279
+
280
+ const onMove = (ev) => {
281
+ const deltaX = ev.clientX - dragState.value.startX
282
+ const containerWidth = document.querySelector('.flex-1.flex.overflow-hidden').offsetWidth
283
+ const deltaPercent = (deltaX / containerWidth) * 100
284
+
285
+ if (colIndex === 0) {
286
+ // 左栏 ↔ 中栏
287
+ const newWidth1 = ((dragState.value.startWidths[0] * containerWidth / 100) + deltaX) / containerWidth * 100
288
+ const newWidth2 = ((dragState.value.startWidths[1] * containerWidth / 100) - deltaX) / containerWidth * 100
289
+ if (newWidth1 >= 10 && newWidth2 >= 10) {
290
+ layout.columnWidths[0] = newWidth1
291
+ layout.columnWidths[1] = newWidth2
292
+ }
293
+ } else {
294
+ // 中栏 ↔ 右栏
295
+ const newWidth2 = ((dragState.value.startWidths[1] * containerWidth / 100) + deltaX) / containerWidth * 100
296
+ const newWidth3 = ((dragState.value.startWidths[2] * containerWidth / 100) - deltaX) / containerWidth * 100
297
+ if (newWidth2 >= 10 && newWidth3 >= 10) {
298
+ layout.columnWidths[1] = newWidth2
299
+ layout.columnWidths[2] = newWidth3
300
+ }
301
+ }
302
+ }
303
+
304
+ const onUp = () => {
305
+ document.body.classList.remove('resizing')
306
+ dragState.value.active = false
307
+ dragState.value.type = null
308
+ window.removeEventListener('mousemove', onMove)
309
+ window.removeEventListener('mouseup', onUp)
310
+ endDrag()
197
311
  }
312
+
313
+ window.addEventListener('mousemove', onMove)
314
+ window.addEventListener('mouseup', onUp)
198
315
  }
199
- })
316
+ }
200
317
 
201
- // WebSocket event handlers
318
+ // WebSocket 事件处理
202
319
  onMounted(() => {
203
- ws.on('projects:init', (projects) => { dashboard.updateProjects(projects) })
204
- ws.on('projects:updated', (projects) => { dashboard.updateProjects(projects) })
320
+ ws.on('projects:init', handleProjectsUpdate)
321
+ ws.on('projects:updated', handleProjectsUpdate)
322
+ ws.on('project:update', (project) => {
323
+ dashboard.updateProject(project)
324
+ })
205
325
  ws.on('cli:output', (data) => {
206
326
  if (data.projectName === dashboard.activeProjectName.value) {
207
327
  dashboard.appendLog(data.output)
@@ -229,22 +349,34 @@ onMounted(() => {
229
349
  ws.on('docs:tree', (docs) => { dashboard.updateDocs(docs) })
230
350
  })
231
351
 
352
+ function handleProjectsUpdate(projects) {
353
+ const previousPath = dashboard.activeProjectPath.value
354
+ dashboard.updateProjects(projects)
355
+
356
+ if (!previousPath && dashboard.state.activeProject?.path) {
357
+ ws.send({ type: 'docs:get', data: { projectPath: dashboard.state.activeProject.path } })
358
+ }
359
+ }
360
+
232
361
  function handleSelectProject(project) {
233
362
  dashboard.selectProject(project)
234
363
  dashboard.selectDocFile(null)
235
364
  dashboard.setDocContent('')
236
- // Request docs for this project
365
+ // 触发淡入动画
366
+ projectSwitched.value = false
367
+ setTimeout(() => { projectSwitched.value = true }, 10)
368
+ setTimeout(() => { projectSwitched.value = false }, 210)
237
369
  if (project?.path) {
238
370
  ws.send({ type: 'docs:get', data: { projectPath: project.path } })
239
371
  }
240
372
  }
373
+
241
374
  function handleSelectStage({ project, stage }) { dashboard.selectProject(project) }
242
375
  function handleSelectStep(step) { dashboard.selectStep(step) }
243
376
  function handleSwitchTab(tab) { dashboard.setActiveTab(tab) }
244
377
  function handleSelectDocFile(file) {
245
378
  dashboard.selectDocFile(file)
246
379
  dashboard.setDocLoading(true)
247
- // Fetch doc content via REST API
248
380
  fetch(`/api/docs/content?path=${encodeURIComponent(file.path)}`)
249
381
  .then(r => r.ok ? r.text() : '')
250
382
  .then(content => {
@@ -256,54 +388,24 @@ function handleSelectDocFile(file) {
256
388
  dashboard.setDocLoading(false)
257
389
  })
258
390
  }
391
+
259
392
  function handleExecute() {
260
393
  const projectName = dashboard.activeProjectName.value
261
394
  if (!projectName) return
262
395
  const progress = dashboard.state.activeProject?.state?.progress
263
- const stages = ['brainstorm', 'plan', 'execute', 'verify']
396
+ const stages = ['scan', 'brainstorm', 'plan', 'execute', 'verify', 'archive']
264
397
  const currentStage = dashboard.state.activeProject?.state?.currentStage
265
398
  || stages.find(stage => progress?.stages?.[stage]?.status !== 'completed')
266
- || 'brainstorm'
399
+ || 'scan'
267
400
  dashboard.clearLogs()
268
401
  ws.send({ type: 'cli:execute', data: { projectName, command: `run ${currentStage}` } })
269
402
  }
403
+
270
404
  function handleKill() {
271
405
  const projectName = dashboard.activeProjectName.value
272
406
  if (!projectName) return
273
407
  ws.send({ type: 'cli:kill', data: { projectName } })
274
408
  }
275
- function handleAddScanPath(path) {
276
- ws.send({ type: 'scan:add-path', data: { path } })
277
- }
278
- function handleRemoveScanPath(path) {
279
- ws.send({ type: 'scan:remove-path', data: { path } })
280
- }
281
-
282
- async function handleShowDetail(type) {
283
- const project = dashboard.state.activeProject
284
- if (!project?.path) return
285
- detailType.value = type
286
- detailData.value = null
287
- dashboard.openPanel()
288
- try {
289
- const url = `/api/projects/${encodeURIComponent(project.path)}/detail?type=${type}`
290
- const res = await fetch(url)
291
- if (res.ok) detailData.value = await res.json()
292
- } catch {}
293
- }
294
-
295
- function handleDetailClose() {
296
- detailType.value = null
297
- detailData.value = null
298
- dashboard.closePanel()
299
- }
300
-
301
- function handleOpenDocFromDetail(file) {
302
- detailType.value = null
303
- detailData.value = null
304
- dashboard.setActiveTab('docs')
305
- handleSelectDocFile(file)
306
- }
307
409
 
308
410
  const themeOverrides = {
309
411
  common: {
@@ -323,8 +425,134 @@ const themeOverrides = {
323
425
  body {
324
426
  font-family: 'DM Sans', -apple-system, BlinkMacSystemFont, sans-serif;
325
427
  -webkit-font-smoothing: antialiased;
326
- -moz-osx-font-smoothing: grayscale;
428
+ -moz-osx-smoothing: grayscale;
327
429
  }
328
430
 
329
- #app { width: 100vw; height: 100vh; overflow: hidden; }
431
+ #app { width: 100vw; height: 100vh; overflow: hidden; min-width: 0; }
432
+
433
+ /* 拖动时禁用选择 */
434
+ body.resizing {
435
+ user-select: none;
436
+ }
437
+ body.resizing * {
438
+ cursor: inherit !important;
439
+ }
440
+
441
+ /* 响应式:小窗口时调整布局 */
442
+ @media (max-width: 1280px) {
443
+ .detail-section-title {
444
+ font-size: 11px;
445
+ }
446
+ .detail-value {
447
+ font-size: 13px;
448
+ }
449
+ .log-entry {
450
+ font-size: 11px;
451
+ }
452
+ }
453
+
454
+ /* 项目信息样式 */
455
+ .detail-column,
456
+ .activity-column {
457
+ padding: 18px 20px;
458
+ }
459
+
460
+ .pipeline-column {
461
+ border-left: 1px solid #EEF0F4;
462
+ border-right: 1px solid #EEF0F4;
463
+ }
464
+
465
+ .pipeline-column > * {
466
+ min-width: 0;
467
+ padding-left: 18px;
468
+ padding-right: 18px;
469
+ }
470
+
471
+ .activity-column {
472
+ gap: 12px;
473
+ }
474
+
475
+ .activity-column .detail-section-title {
476
+ padding: 0;
477
+ margin-bottom: 0;
478
+ }
479
+
480
+ .activity-column > .flex-1 {
481
+ padding: 0;
482
+ }
483
+
484
+ .project-info {
485
+ min-width: 0;
486
+ }
487
+
488
+ .detail-section-title {
489
+ font-size: 12px;
490
+ font-weight: 600;
491
+ color: #9CA3AF;
492
+ text-transform: uppercase;
493
+ margin-bottom: 12px;
494
+ }
495
+
496
+ .detail-item {
497
+ margin-bottom: 16px;
498
+ }
499
+
500
+ .detail-label {
501
+ font-size: 11px;
502
+ color: #9CA3AF;
503
+ margin-bottom: 4px;
504
+ }
505
+
506
+ .detail-value {
507
+ font-size: 14px;
508
+ color: #1A1A1A;
509
+ font-weight: 500;
510
+ min-width: 0;
511
+ overflow-wrap: anywhere;
512
+ word-break: break-word;
513
+ line-height: 1.45;
514
+ }
515
+
516
+ .detail-path {
517
+ font-size: 12px;
518
+ font-family: 'JetBrains Mono', ui-monospace, SFMono-Regular, Menlo, Monaco, Consolas, monospace;
519
+ white-space: normal;
520
+ }
521
+
522
+ /* 日志条目样式 */
523
+ .log-entry {
524
+ font-family: 'JetBrains Mono', monospace;
525
+ font-size: 12px;
526
+ padding: 8px;
527
+ background: #F9FAFB;
528
+ border-radius: 4px;
529
+ margin-bottom: 8px;
530
+ color: #374151;
531
+ }
532
+
533
+ .log-entry.success {
534
+ color: #047857;
535
+ background: #D1FAE5;
536
+ }
537
+
538
+ .log-entry.error {
539
+ color: #DC2626;
540
+ background: #FEE2E2;
541
+ }
542
+
543
+ /* 淡入动画 */
544
+ .fade-in {
545
+ animation: fadeIn 0.2s ease-out;
546
+ }
547
+
548
+ @keyframes fadeIn {
549
+ from {
550
+ opacity: 0;
551
+ transform: translateY(4px);
552
+ }
553
+ to {
554
+ opacity: 1;
555
+ transform: translateY(0);
556
+ }
557
+ }
330
558
  </style>