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.
- package/.claude/skills/sillyspec-commit/SKILL.md +1 -1
- package/.claude/skills/sillyspec-continue/SKILL.md +1 -1
- package/.claude/skills/sillyspec-explore/SKILL.md +9 -0
- package/.claude/skills/sillyspec-plan/SKILL.md +0 -35
- package/.claude/skills/sillyspec-resume/SKILL.md +5 -5
- package/.claude/skills/sillyspec-state/SKILL.md +1 -1
- package/.claude/skills/sillyspec-workspace/SKILL.md +1 -1
- package/README.md +7 -6
- package/SKILL.md +15 -10
- package/package.json +1 -1
- package/packages/dashboard/dist/assets/index-BcM2J-hv.css +1 -0
- package/packages/dashboard/dist/assets/{index-RsLVPAy7.js → index-DpLHK4jv.js} +974 -974
- package/packages/dashboard/dist/index.html +16 -17
- package/packages/dashboard/dist/prototype-dashboard.html +836 -0
- package/packages/dashboard/dist/prototype-overview.html +256 -0
- package/packages/dashboard/public/prototype-dashboard.html +836 -0
- package/packages/dashboard/public/prototype-overview.html +256 -0
- package/packages/dashboard/server/index.js +18 -13
- package/packages/dashboard/server/parser.js +109 -1
- package/packages/dashboard/server/watcher.js +14 -6
- package/packages/dashboard/src/App.vue +414 -186
- package/packages/dashboard/src/components/ActionBar.vue +10 -1
- package/packages/dashboard/src/components/CommandPalette.vue +5 -1
- package/packages/dashboard/src/components/DocPreview.vue +105 -8
- package/packages/dashboard/src/components/DocTree.vue +75 -19
- package/packages/dashboard/src/components/HResizeHandle.vue +48 -0
- package/packages/dashboard/src/components/PipelineView.vue +23 -4
- package/packages/dashboard/src/components/ProjectCard.vue +187 -0
- package/packages/dashboard/src/components/ProjectOverview.vue +113 -139
- package/packages/dashboard/src/components/VResizeHandle.vue +61 -0
- package/packages/dashboard/src/composables/useDashboard.js +28 -0
- package/packages/dashboard/src/composables/useLayout.js +131 -0
- package/src/index.js +7 -0
- package/src/init.js +17 -10
- package/src/migrate.js +5 -5
- package/src/progress.js +2 -1
- package/src/run.js +141 -64
- package/src/stages/archive.js +14 -8
- package/src/stages/brainstorm.js +29 -4
- package/src/stages/execute.js +115 -35
- package/src/stages/explore.js +34 -0
- package/src/stages/index.js +11 -6
- package/src/stages/plan.js +86 -15
- package/src/stages/quick.js +1 -1
- package/src/stages/scan.js +51 -12
- package/src/stages/status.js +1 -1
- package/src/stages/verify.js +35 -6
- package/.sillyspec/changes/archive/2026-04-08-derive-state/design.md +0 -97
- package/.sillyspec/changes/archive/2026-04-08-derive-state/plan.md +0 -51
- package/.sillyspec/changes/archive/2026-04-08-derive-state/proposal.md +0 -29
- package/.sillyspec/changes/archive/2026-04-08-derive-state/requirements.md +0 -34
- package/.sillyspec/changes/archive/2026-04-08-derive-state/tasks.md +0 -13
- package/.sillyspec/changes/archive/2026-04-08-derive-state/verify-result.md +0 -43
- package/.sillyspec/changes/auto-mode/design.md +0 -50
- package/.sillyspec/changes/auto-mode/proposal.md +0 -19
- package/.sillyspec/changes/auto-mode/requirements.md +0 -21
- package/.sillyspec/changes/auto-mode/tasks.md +0 -7
- package/.sillyspec/changes/brainstorm-archive/2026-04-05-dashboard-design.md +0 -206
- package/.sillyspec/changes/brainstorm-archive/2026-04-05-unified-docs-design.md +0 -199
- package/.sillyspec/changes/dashboard/design.md +0 -219
- package/.sillyspec/changes/dashboard/design.md.braindraft +0 -206
- package/.sillyspec/changes/run-command-design/design.md +0 -1230
- package/.sillyspec/changes/unified-docs-design/design.md +0 -199
- package/.sillyspec/docs/sillyspec/scan/.gitkeep +0 -0
- package/.sillyspec/knowledge/INDEX.md +0 -8
- package/.sillyspec/knowledge/uncategorized.md +0 -3
- package/.sillyspec/plans/2026-04-05-dashboard.md +0 -737
- package/.sillyspec/projects/sillyspec.yaml +0 -3
- 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
|
-
<!--
|
|
7
|
+
<!-- 上层:多项目概览区域 -->
|
|
8
8
|
<div
|
|
9
|
-
class="
|
|
10
|
-
:style="
|
|
9
|
+
class="relative overflow-hidden flex-shrink-0"
|
|
10
|
+
:style="{ height: layout.overviewHeight + '%' }"
|
|
11
11
|
>
|
|
12
|
-
<!--
|
|
13
|
-
<
|
|
14
|
-
|
|
15
|
-
:
|
|
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
|
-
|
|
61
|
-
|
|
62
|
-
|
|
63
|
-
|
|
64
|
-
|
|
65
|
-
|
|
66
|
-
|
|
67
|
-
>
|
|
68
|
-
<
|
|
69
|
-
|
|
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
|
-
<!--
|
|
82
|
-
<
|
|
83
|
-
|
|
84
|
-
|
|
85
|
-
|
|
86
|
-
|
|
87
|
-
|
|
88
|
-
|
|
89
|
-
|
|
90
|
-
|
|
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,
|
|
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 {
|
|
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
|
|
123
|
-
const
|
|
124
|
-
|
|
125
|
-
|
|
126
|
-
|
|
127
|
-
|
|
128
|
-
|
|
129
|
-
|
|
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
|
-
|
|
150
|
-
|
|
151
|
-
|
|
152
|
-
|
|
153
|
-
|
|
154
|
-
|
|
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
|
-
|
|
159
|
-
|
|
160
|
-
|
|
161
|
-
|
|
162
|
-
|
|
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
|
-
|
|
165
|
-
|
|
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
|
-
|
|
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
|
-
|
|
171
|
-
|
|
172
|
-
|
|
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
|
|
175
|
-
|
|
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
|
-
|
|
181
|
-
persistWidths()
|
|
259
|
+
endDrag()
|
|
182
260
|
}
|
|
261
|
+
|
|
183
262
|
window.addEventListener('mousemove', onMove)
|
|
184
263
|
window.addEventListener('mouseup', onUp)
|
|
185
264
|
}
|
|
186
265
|
|
|
187
|
-
//
|
|
188
|
-
|
|
189
|
-
|
|
190
|
-
|
|
191
|
-
|
|
192
|
-
|
|
193
|
-
|
|
194
|
-
|
|
195
|
-
|
|
196
|
-
|
|
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
|
|
318
|
+
// WebSocket 事件处理
|
|
202
319
|
onMounted(() => {
|
|
203
|
-
ws.on('projects:init',
|
|
204
|
-
ws.on('projects:updated',
|
|
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
|
-
//
|
|
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
|
-
|| '
|
|
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-
|
|
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>
|