sillyspec 3.9.0 → 3.10.0

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 (206) hide show
  1. package/.claude/skills/sillyspec-archive/SKILL.md +17 -0
  2. package/.claude/skills/sillyspec-auto/SKILL.md +78 -0
  3. package/.claude/skills/sillyspec-brainstorm/SKILL.md +17 -0
  4. package/{templates/commit.md → .claude/skills/sillyspec-commit/SKILL.md} +32 -47
  5. package/.claude/skills/sillyspec-continue/SKILL.md +45 -0
  6. package/.claude/skills/sillyspec-doctor/SKILL.md +27 -0
  7. package/.claude/skills/sillyspec-execute/SKILL.md +17 -0
  8. package/.claude/skills/sillyspec-explore/SKILL.md +105 -0
  9. package/.claude/skills/sillyspec-export/SKILL.md +53 -0
  10. package/.claude/skills/sillyspec-init/SKILL.md +170 -0
  11. package/.claude/skills/sillyspec-plan/SKILL.md +17 -0
  12. package/.claude/skills/sillyspec-propose/SKILL.md +17 -0
  13. package/.claude/skills/sillyspec-quick/SKILL.md +17 -0
  14. package/.claude/skills/sillyspec-resume/SKILL.md +111 -0
  15. package/.claude/skills/sillyspec-scan/SKILL.md +17 -0
  16. package/.claude/skills/sillyspec-state/SKILL.md +54 -0
  17. package/.claude/skills/sillyspec-status/SKILL.md +17 -0
  18. package/.claude/skills/sillyspec-verify/SKILL.md +17 -0
  19. package/.claude/skills/sillyspec-workspace/SKILL.md +149 -0
  20. package/README.md +19 -11
  21. package/SKILL.md +15 -10
  22. package/package.json +7 -9
  23. package/packages/dashboard/dist/assets/index-BcM2J-hv.css +1 -0
  24. package/packages/dashboard/dist/assets/index-DpLHK4jv.js +7446 -0
  25. package/packages/dashboard/dist/index.html +16 -16
  26. package/packages/dashboard/dist/prototype-dashboard.html +836 -0
  27. package/packages/dashboard/dist/prototype-overview.html +256 -0
  28. package/packages/dashboard/package-lock.json +226 -6
  29. package/packages/dashboard/package.json +8 -5
  30. package/packages/dashboard/public/logo.jpg +0 -0
  31. package/packages/dashboard/public/prototype-dashboard.html +836 -0
  32. package/packages/dashboard/public/prototype-overview.html +256 -0
  33. package/packages/dashboard/server/executor.js +1 -1
  34. package/packages/dashboard/server/index.js +341 -113
  35. package/packages/dashboard/server/parser.js +442 -30
  36. package/packages/dashboard/server/watcher.js +214 -134
  37. package/packages/dashboard/src/App.vue +475 -71
  38. package/packages/dashboard/src/components/ActionBar.vue +36 -43
  39. package/packages/dashboard/src/components/CommandPalette.vue +45 -66
  40. package/packages/dashboard/src/components/DetailPanel.vue +68 -53
  41. package/packages/dashboard/src/components/DocPreview.vue +257 -0
  42. package/packages/dashboard/src/components/DocTree.vue +114 -0
  43. package/packages/dashboard/src/components/HResizeHandle.vue +48 -0
  44. package/packages/dashboard/src/components/LogStream.vue +13 -33
  45. package/packages/dashboard/src/components/PipelineStage.vue +8 -8
  46. package/packages/dashboard/src/components/PipelineView.vue +99 -45
  47. package/packages/dashboard/src/components/ProjectCard.vue +187 -0
  48. package/packages/dashboard/src/components/ProjectList.vue +103 -45
  49. package/packages/dashboard/src/components/ProjectOverview.vue +152 -0
  50. package/packages/dashboard/src/components/StageBadge.vue +13 -13
  51. package/packages/dashboard/src/components/StepCard.vue +15 -15
  52. package/packages/dashboard/src/components/VResizeHandle.vue +61 -0
  53. package/packages/dashboard/src/components/detail/DocsDetail.vue +48 -0
  54. package/packages/dashboard/src/components/detail/GitDetail.vue +61 -0
  55. package/packages/dashboard/src/components/detail/TechDetail.vue +43 -0
  56. package/packages/dashboard/src/composables/useDashboard.js +48 -6
  57. package/packages/dashboard/src/composables/useKeyboard.js +6 -4
  58. package/packages/dashboard/src/composables/useLayout.js +131 -0
  59. package/packages/dashboard/src/main.js +4 -1
  60. package/packages/dashboard/src/style.css +17 -17
  61. package/src/index.js +141 -22
  62. package/src/init.js +93 -231
  63. package/src/migrate.js +117 -0
  64. package/src/progress.js +460 -0
  65. package/src/run.js +635 -0
  66. package/src/setup.js +2 -72
  67. package/src/stages/archive.js +54 -0
  68. package/src/stages/brainstorm.js +264 -0
  69. package/src/stages/doctor.js +303 -0
  70. package/src/stages/execute.js +287 -0
  71. package/src/stages/explore.js +34 -0
  72. package/src/stages/index.js +28 -0
  73. package/src/stages/plan.js +354 -0
  74. package/src/stages/propose.js +115 -0
  75. package/src/stages/quick.js +64 -0
  76. package/src/stages/scan.js +141 -0
  77. package/src/stages/status.js +65 -0
  78. package/src/stages/verify.js +135 -0
  79. package/.sillyspec/changes/dashboard/design.md +0 -219
  80. package/.sillyspec/plans/2026-04-05-dashboard.md +0 -737
  81. package/.sillyspec/specs/2026-04-05-dashboard-design.md +0 -206
  82. package/dist/steps/brainstorm/01-load-context.md +0 -30
  83. package/dist/steps/brainstorm/02-reuse-check.md +0 -6
  84. package/dist/steps/brainstorm/03-prototype-analysis.md +0 -11
  85. package/dist/steps/brainstorm/04-module-split.md +0 -23
  86. package/dist/steps/brainstorm/05-dialog-explore.md +0 -8
  87. package/dist/steps/brainstorm/06-propose-approaches.md +0 -3
  88. package/dist/steps/brainstorm/07-present-design.md +0 -3
  89. package/dist/steps/brainstorm/08-write-design.md +0 -21
  90. package/dist/steps/brainstorm/09-self-review.md +0 -15
  91. package/dist/steps/brainstorm/10-user-confirm.md +0 -3
  92. package/dist/steps/brainstorm/11-output-spec.md +0 -7
  93. package/dist/steps/brainstorm/manifest.yaml +0 -26
  94. package/dist/steps/execute/01-load-context.md +0 -41
  95. package/dist/steps/execute/02-scan-conventions.md +0 -47
  96. package/dist/steps/execute/03-skill-mcp.md +0 -19
  97. package/dist/steps/execute/04-assign-task.md +0 -22
  98. package/dist/steps/execute/04b-prompt-template.md +0 -54
  99. package/dist/steps/execute/05-write-test.md +0 -7
  100. package/dist/steps/execute/06-write-code.md +0 -8
  101. package/dist/steps/execute/07-run-test.md +0 -26
  102. package/dist/steps/execute/08-fix-issues.md +0 -28
  103. package/dist/steps/execute/09-next-task.md +0 -33
  104. package/dist/steps/execute/manifest.yaml +0 -28
  105. package/dist/steps/plan/01-load-context.md +0 -22
  106. package/dist/steps/plan/02-anchor-confirm.md +0 -1
  107. package/dist/steps/plan/03-expand-tasks.md +0 -33
  108. package/dist/steps/plan/04-mark-order.md +0 -15
  109. package/dist/steps/plan/05-e2e-planning.md +0 -17
  110. package/dist/steps/plan/06-self-check.md +0 -16
  111. package/dist/steps/plan/07-save.md +0 -1
  112. package/dist/steps/plan/manifest.yaml +0 -18
  113. package/dist/steps/scan/01-env-detect.md +0 -51
  114. package/dist/steps/scan/02-tech-stack.md +0 -16
  115. package/dist/steps/scan/03-conventions.md +0 -16
  116. package/dist/steps/scan/04-structure.md +0 -19
  117. package/dist/steps/scan/05-quality.md +0 -18
  118. package/dist/steps/scan/06-complete.md +0 -49
  119. package/dist/steps/scan/manifest.yaml +0 -16
  120. package/dist/steps/verify/01-load-specs.md +0 -28
  121. package/dist/steps/verify/02-check-tasks.md +0 -1
  122. package/dist/steps/verify/03-check-design.md +0 -6
  123. package/dist/steps/verify/04-run-tests.md +0 -7
  124. package/dist/steps/verify/05-e2e-tests.md +0 -27
  125. package/dist/steps/verify/05b-e2e-fix.md +0 -33
  126. package/dist/steps/verify/06-code-quality.md +0 -25
  127. package/dist/steps/verify/07-lint-check.md +0 -27
  128. package/dist/steps/verify/08-output-report.md +0 -14
  129. package/dist/steps/verify/manifest.yaml +0 -22
  130. package/docs/.vitepress/config.mts +0 -45
  131. package/docs/.vitepress/dist/404.html +0 -25
  132. package/docs/.vitepress/dist/assets/app.YytxICdd.js +0 -1
  133. package/docs/.vitepress/dist/assets/chunks/framework.Czhw_PXq.js +0 -19
  134. package/docs/.vitepress/dist/assets/chunks/theme.DusTRZQk.js +0 -1
  135. package/docs/.vitepress/dist/assets/index.md.C3VCvtQA.js +0 -1
  136. package/docs/.vitepress/dist/assets/index.md.C3VCvtQA.lean.js +0 -1
  137. package/docs/.vitepress/dist/assets/inter-italic-cyrillic-ext.r48I6akx.woff2 +0 -0
  138. package/docs/.vitepress/dist/assets/inter-italic-cyrillic.By2_1cv3.woff2 +0 -0
  139. package/docs/.vitepress/dist/assets/inter-italic-greek-ext.1u6EdAuj.woff2 +0 -0
  140. package/docs/.vitepress/dist/assets/inter-italic-greek.DJ8dCoTZ.woff2 +0 -0
  141. package/docs/.vitepress/dist/assets/inter-italic-latin-ext.CN1xVJS-.woff2 +0 -0
  142. package/docs/.vitepress/dist/assets/inter-italic-latin.C2AdPX0b.woff2 +0 -0
  143. package/docs/.vitepress/dist/assets/inter-italic-vietnamese.BSbpV94h.woff2 +0 -0
  144. package/docs/.vitepress/dist/assets/inter-roman-cyrillic-ext.BBPuwvHQ.woff2 +0 -0
  145. package/docs/.vitepress/dist/assets/inter-roman-cyrillic.C5lxZ8CY.woff2 +0 -0
  146. package/docs/.vitepress/dist/assets/inter-roman-greek-ext.CqjqNYQ-.woff2 +0 -0
  147. package/docs/.vitepress/dist/assets/inter-roman-greek.BBVDIX6e.woff2 +0 -0
  148. package/docs/.vitepress/dist/assets/inter-roman-latin-ext.4ZJIpNVo.woff2 +0 -0
  149. package/docs/.vitepress/dist/assets/inter-roman-latin.Di8DUHzh.woff2 +0 -0
  150. package/docs/.vitepress/dist/assets/inter-roman-vietnamese.BjW4sHH5.woff2 +0 -0
  151. package/docs/.vitepress/dist/assets/sillyspec_commands.md.CXFFsj08.js +0 -15
  152. package/docs/.vitepress/dist/assets/sillyspec_commands.md.CXFFsj08.lean.js +0 -1
  153. package/docs/.vitepress/dist/assets/sillyspec_dashboard.md.BuPXHqjX.js +0 -4
  154. package/docs/.vitepress/dist/assets/sillyspec_dashboard.md.BuPXHqjX.lean.js +0 -1
  155. package/docs/.vitepress/dist/assets/sillyspec_file-io.md.Cz3x7llx.js +0 -1
  156. package/docs/.vitepress/dist/assets/sillyspec_file-io.md.Cz3x7llx.lean.js +0 -1
  157. package/docs/.vitepress/dist/assets/sillyspec_getting-started.md.ClcvV8k3.js +0 -4
  158. package/docs/.vitepress/dist/assets/sillyspec_getting-started.md.ClcvV8k3.lean.js +0 -1
  159. package/docs/.vitepress/dist/assets/sillyspec_install.md.CKuR2tiT.js +0 -5
  160. package/docs/.vitepress/dist/assets/sillyspec_install.md.CKuR2tiT.lean.js +0 -1
  161. package/docs/.vitepress/dist/assets/sillyspec_lifecycle.md.DY293cR1.js +0 -28
  162. package/docs/.vitepress/dist/assets/sillyspec_lifecycle.md.DY293cR1.lean.js +0 -1
  163. package/docs/.vitepress/dist/assets/sillyspec_structure.md.sVYS4zPs.js +0 -30
  164. package/docs/.vitepress/dist/assets/sillyspec_structure.md.sVYS4zPs.lean.js +0 -1
  165. package/docs/.vitepress/dist/assets/style.DFTx90Kk.css +0 -1
  166. package/docs/.vitepress/dist/hashmap.json +0 -1
  167. package/docs/.vitepress/dist/index.html +0 -28
  168. package/docs/.vitepress/dist/sillyspec/commands.html +0 -42
  169. package/docs/.vitepress/dist/sillyspec/dashboard.html +0 -31
  170. package/docs/.vitepress/dist/sillyspec/file-io.html +0 -28
  171. package/docs/.vitepress/dist/sillyspec/getting-started.html +0 -31
  172. package/docs/.vitepress/dist/sillyspec/install.html +0 -32
  173. package/docs/.vitepress/dist/sillyspec/lifecycle.html +0 -55
  174. package/docs/.vitepress/dist/sillyspec/structure.html +0 -57
  175. package/docs/.vitepress/dist/vp-icons.css +0 -1
  176. package/docs/index.md +0 -34
  177. package/docs/sillyspec/commands.md +0 -218
  178. package/docs/sillyspec/dashboard.md +0 -51
  179. package/docs/sillyspec/file-io.md +0 -34
  180. package/docs/sillyspec/getting-started.md +0 -61
  181. package/docs/sillyspec/install.md +0 -51
  182. package/docs/sillyspec/lifecycle.md +0 -146
  183. package/docs/sillyspec/structure.md +0 -62
  184. package/packages/dashboard/dist/assets/index-Bh-GPjKY.css +0 -1
  185. package/packages/dashboard/dist/assets/index-CrCn5Gg6.js +0 -17
  186. package/src/step.js +0 -543
  187. package/templates/archive.md +0 -120
  188. package/templates/brainstorm.md +0 -170
  189. package/templates/continue.md +0 -32
  190. package/templates/execute.md +0 -304
  191. package/templates/explore.md +0 -59
  192. package/templates/export.md +0 -21
  193. package/templates/init.md +0 -61
  194. package/templates/plan.md +0 -146
  195. package/templates/quick.md +0 -135
  196. package/templates/scan-quick.md +0 -49
  197. package/templates/scan.md +0 -156
  198. package/templates/skills/playwright-e2e/SKILL.md +0 -340
  199. package/templates/status.md +0 -75
  200. package/templates/verify.md +0 -236
  201. package/templates/workspace-sync.md +0 -99
  202. package/templates/workspace.md +0 -70
  203. /package/{docs/.vitepress/dist/logo.jpg → logo.jpg} +0 -0
  204. /package/{docs/.vitepress → packages/dashboard}/dist/favicon.jpg +0 -0
  205. /package/{docs/public → packages/dashboard/dist}/logo.jpg +0 -0
  206. /package/{docs → packages/dashboard}/public/favicon.jpg +0 -0
@@ -1,58 +1,129 @@
1
1
  <template>
2
- <div class="h-screen w-screen flex flex-col overflow-hidden font-[DM_Sans,sans-serif] relative" style="background-color: #0A0A0B;">
2
+ <n-config-provider :theme-overrides="themeOverrides">
3
+ <div class="h-screen w-screen flex flex-col overflow-hidden font-[DM_Sans,sans-serif] relative" style="background-color: #F5F5F7;">
3
4
  <!-- Ambient background -->
4
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%);" />
5
6
 
6
- <!-- Main Content -->
7
- <div class="flex-1 flex overflow-hidden relative z-10">
8
- <!-- Left: Project List -->
9
- <aside class="w-[240px] flex-shrink-0 relative" style="background: #111113; border-right: 1px solid #1F1F22;">
10
- <ProjectList
11
- :projects="dashboard.state.projects"
12
- :active-project="dashboard.state.activeProject"
13
- :is-loading="dashboard.state.isLoading"
14
- @select="handleSelectProject"
15
- />
16
- </aside>
17
-
18
- <!-- Center: Pipeline View -->
19
- <main class="flex-1 overflow-hidden accent-stripe">
20
- <PipelineView
21
- :project="dashboard.state.activeProject"
22
- :active-step="dashboard.state.activeStep"
23
- @select-step="handleSelectStep"
24
- />
25
- </main>
26
-
27
- <!-- Right: Detail Panel -->
28
- <aside
29
- :class="[
30
- 'flex-shrink-0 transition-all duration-300 relative overflow-hidden',
31
- dashboard.state.isPanelOpen ? 'w-[340px]' : 'w-0'
32
- ]"
33
- style="background: #111113; border-left: 1px solid #1F1F22;"
34
- >
35
- <DetailPanel
36
- :is-open="dashboard.state.isPanelOpen"
37
- :active-step="dashboard.state.activeStep"
38
- :logs="dashboard.state.logs"
39
- @close="dashboard.closePanel"
40
- @clear-logs="dashboard.clearLogs"
41
- />
42
- </aside>
7
+ <!-- 上层:多项目概览区域 -->
8
+ <div
9
+ class="relative overflow-hidden flex-shrink-0"
10
+ :style="{ height: layout.overviewHeight + '%' }"
11
+ >
12
+ <!-- 项目概览 -->
13
+ <ProjectOverview
14
+ :projects="dashboard.state.projects"
15
+ :active-project="dashboard.state.activeProject"
16
+ @select="handleSelectProject"
17
+ />
43
18
  </div>
44
19
 
45
- <!-- Bottom: Action Bar -->
46
- <ActionBar
47
- :project="dashboard.state.activeProject"
48
- :is-executing="dashboard.state.executingProject !== null"
49
- :execution-result="executionResult"
50
- :is-panel-open="dashboard.state.isPanelOpen"
51
- @execute="handleExecute"
52
- @kill="handleKill"
53
- @toggle-panel="dashboard.togglePanel"
54
- @open-palette="isCommandPaletteOpen = true"
55
- />
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>
30
+ </div>
31
+
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>
56
127
 
57
128
  <!-- Command Palette Overlay -->
58
129
  <CommandPalette
@@ -62,44 +133,195 @@
62
133
  @select-project="handleSelectProject"
63
134
  @select-stage="handleSelectStage"
64
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>
65
144
  </div>
145
+ </n-config-provider>
66
146
  </template>
67
147
 
68
148
  <script setup>
69
- import { ref, onMounted } from 'vue'
149
+ import { ref, computed, onMounted } from 'vue'
70
150
  import { useWebSocket } from './composables/useWebSocket.js'
71
151
  import { useDashboard } from './composables/useDashboard.js'
72
- import { useDashboardKeyboard } from './composables/useKeyboard.js'
73
- import ProjectList from './components/ProjectList.vue'
152
+ import { useLayout } from './composables/useLayout.js'
74
153
  import PipelineView from './components/PipelineView.vue'
75
- import DetailPanel from './components/DetailPanel.vue'
154
+ import ProjectOverview from './components/ProjectOverview.vue'
76
155
  import ActionBar from './components/ActionBar.vue'
77
156
  import CommandPalette from './components/CommandPalette.vue'
78
157
 
79
158
  // Composables
80
159
  const ws = useWebSocket()
81
160
  const dashboard = useDashboard()
161
+ const layoutManager = useLayout()
162
+
163
+ // 解构 layout
164
+ const { layout, isDragging, startDrag, endDrag } = layoutManager
165
+
166
+ // 状态
82
167
  const isCommandPaletteOpen = ref(false)
83
168
  const executionResult = ref(null)
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
+ })
84
177
 
85
- // Keyboard shortcuts
86
- useDashboardKeyboard({
87
- onOpenCommandPalette: () => { isCommandPaletteOpen.value = true },
88
- onClose: () => {
89
- if (isCommandPaletteOpen.value) {
90
- isCommandPaletteOpen.value = false
91
- } else if (dashboard.state.activeStep) {
92
- dashboard.state.activeStep = null
93
- } else {
94
- dashboard.closePanel()
95
- }
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])}%`
96
184
  }
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: '自由探索'
210
+ }
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
+ })
226
+
227
+ // 最近日志(最近 10 条)
228
+ const recentLogs = computed(() => {
229
+ const logs = dashboard.state.logs || []
230
+ return logs.slice(-10).reverse()
97
231
  })
98
232
 
99
- // WebSocket event handlers
233
+ // 垂直拖动(概览 详情)
234
+ function startDragVertical(e) {
235
+ e.preventDefault()
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
+
245
+ const onMove = (ev) => {
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))
251
+ }
252
+
253
+ const onUp = () => {
254
+ document.body.classList.remove('resizing')
255
+ dragState.value.active = false
256
+ dragState.value.type = null
257
+ window.removeEventListener('mousemove', onMove)
258
+ window.removeEventListener('mouseup', onUp)
259
+ endDrag()
260
+ }
261
+
262
+ window.addEventListener('mousemove', onMove)
263
+ window.addEventListener('mouseup', onUp)
264
+ }
265
+
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()
311
+ }
312
+
313
+ window.addEventListener('mousemove', onMove)
314
+ window.addEventListener('mouseup', onUp)
315
+ }
316
+ }
317
+
318
+ // WebSocket 事件处理
100
319
  onMounted(() => {
101
- ws.on('projects:init', (projects) => { dashboard.updateProjects(projects) })
102
- 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
+ })
103
325
  ws.on('cli:output', (data) => {
104
326
  if (data.projectName === dashboard.activeProjectName.value) {
105
327
  dashboard.appendLog(data.output)
@@ -123,22 +345,78 @@ onMounted(() => {
123
345
  executionResult.value = { exitCode: -1, signal: 'SIGTERM' }
124
346
  }
125
347
  })
348
+ ws.on('scan:paths', (paths) => { scanPaths.value = paths })
349
+ ws.on('docs:tree', (docs) => { dashboard.updateDocs(docs) })
126
350
  })
127
351
 
128
- function handleSelectProject(project) { dashboard.selectProject(project) }
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
+
361
+ function handleSelectProject(project) {
362
+ dashboard.selectProject(project)
363
+ dashboard.selectDocFile(null)
364
+ dashboard.setDocContent('')
365
+ // 触发淡入动画
366
+ projectSwitched.value = false
367
+ setTimeout(() => { projectSwitched.value = true }, 10)
368
+ setTimeout(() => { projectSwitched.value = false }, 210)
369
+ if (project?.path) {
370
+ ws.send({ type: 'docs:get', data: { projectPath: project.path } })
371
+ }
372
+ }
373
+
129
374
  function handleSelectStage({ project, stage }) { dashboard.selectProject(project) }
130
375
  function handleSelectStep(step) { dashboard.selectStep(step) }
376
+ function handleSwitchTab(tab) { dashboard.setActiveTab(tab) }
377
+ function handleSelectDocFile(file) {
378
+ dashboard.selectDocFile(file)
379
+ dashboard.setDocLoading(true)
380
+ fetch(`/api/docs/content?path=${encodeURIComponent(file.path)}`)
381
+ .then(r => r.ok ? r.text() : '')
382
+ .then(content => {
383
+ dashboard.setDocContent(content)
384
+ dashboard.setDocLoading(false)
385
+ })
386
+ .catch(() => {
387
+ dashboard.setDocContent('')
388
+ dashboard.setDocLoading(false)
389
+ })
390
+ }
391
+
131
392
  function handleExecute() {
132
393
  const projectName = dashboard.activeProjectName.value
133
394
  if (!projectName) return
395
+ const progress = dashboard.state.activeProject?.state?.progress
396
+ const stages = ['scan', 'brainstorm', 'plan', 'execute', 'verify', 'archive']
397
+ const currentStage = dashboard.state.activeProject?.state?.currentStage
398
+ || stages.find(stage => progress?.stages?.[stage]?.status !== 'completed')
399
+ || 'scan'
134
400
  dashboard.clearLogs()
135
- ws.send({ type: 'cli:execute', data: { projectName, command: 'next' } })
401
+ ws.send({ type: 'cli:execute', data: { projectName, command: `run ${currentStage}` } })
136
402
  }
403
+
137
404
  function handleKill() {
138
405
  const projectName = dashboard.activeProjectName.value
139
406
  if (!projectName) return
140
407
  ws.send({ type: 'cli:kill', data: { projectName } })
141
408
  }
409
+
410
+ const themeOverrides = {
411
+ common: {
412
+ primaryColor: '#D97706',
413
+ primaryColorHover: '#F59E0B',
414
+ primaryColorPressed: '#B45309',
415
+ borderRadius: '6px',
416
+ fontFamily: 'DM Sans, sans-serif',
417
+ fontFamilyMono: 'JetBrains Mono, monospace'
418
+ }
419
+ }
142
420
  </script>
143
421
 
144
422
  <style>
@@ -147,8 +425,134 @@ function handleKill() {
147
425
  body {
148
426
  font-family: 'DM Sans', -apple-system, BlinkMacSystemFont, sans-serif;
149
427
  -webkit-font-smoothing: antialiased;
150
- -moz-osx-font-smoothing: grayscale;
428
+ -moz-osx-smoothing: grayscale;
429
+ }
430
+
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;
151
478
  }
152
479
 
153
- #app { width: 100vw; height: 100vh; overflow: hidden; }
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
+ }
154
558
  </style>