codexmate 0.0.18 → 0.0.20
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/README.en.md +34 -17
- package/README.md +34 -25
- package/cli/config-health.js +338 -0
- package/cli.js +1570 -839
- package/lib/cli-models-utils.js +186 -27
- package/lib/cli-network-utils.js +117 -101
- package/package.json +8 -1
- package/web-ui/app.js +379 -5754
- package/web-ui/index.html +15 -2079
- package/web-ui/logic.agents-diff.mjs +386 -0
- package/web-ui/logic.claude.mjs +108 -0
- package/web-ui/logic.mjs +5 -793
- package/web-ui/logic.runtime.mjs +124 -0
- package/web-ui/logic.sessions.mjs +263 -0
- package/web-ui/modules/api.mjs +69 -0
- package/web-ui/modules/app.computed.dashboard.mjs +113 -0
- package/web-ui/modules/app.computed.index.mjs +13 -0
- package/web-ui/modules/app.computed.session.mjs +141 -0
- package/web-ui/modules/app.constants.mjs +15 -0
- package/web-ui/modules/app.methods.agents.mjs +493 -0
- package/web-ui/modules/app.methods.claude-config.mjs +174 -0
- package/web-ui/modules/app.methods.codex-config.mjs +640 -0
- package/web-ui/modules/app.methods.index.mjs +86 -0
- package/web-ui/modules/app.methods.install.mjs +157 -0
- package/web-ui/modules/app.methods.navigation.mjs +478 -0
- package/web-ui/modules/app.methods.openclaw-core.mjs +514 -0
- package/web-ui/modules/app.methods.openclaw-editing.mjs +337 -0
- package/web-ui/modules/app.methods.openclaw-persist.mjs +251 -0
- package/web-ui/modules/app.methods.providers.mjs +265 -0
- package/web-ui/modules/app.methods.runtime.mjs +323 -0
- package/web-ui/modules/app.methods.session-actions.mjs +457 -0
- package/web-ui/modules/app.methods.session-browser.mjs +435 -0
- package/web-ui/modules/app.methods.session-timeline.mjs +441 -0
- package/web-ui/modules/app.methods.session-trash.mjs +419 -0
- package/web-ui/modules/app.methods.startup-claude.mjs +406 -0
- package/web-ui/modules/config-mode.computed.mjs +1 -0
- package/web-ui/modules/skills.computed.mjs +26 -1
- package/web-ui/modules/skills.methods.mjs +154 -23
- package/web-ui/partials/index/layout-footer.html +69 -0
- package/web-ui/partials/index/layout-header.html +337 -0
- package/web-ui/partials/index/modal-config-template-agents.html +125 -0
- package/web-ui/partials/index/modal-confirm-toast.html +32 -0
- package/web-ui/partials/index/modal-health-check.html +72 -0
- package/web-ui/partials/index/modal-openclaw-config.html +275 -0
- package/web-ui/partials/index/modal-skills.html +184 -0
- package/web-ui/partials/index/modals-basic.html +196 -0
- package/web-ui/partials/index/panel-config-claude.html +100 -0
- package/web-ui/partials/index/panel-config-codex.html +237 -0
- package/web-ui/partials/index/panel-config-openclaw.html +84 -0
- package/web-ui/partials/index/panel-market.html +174 -0
- package/web-ui/partials/index/panel-sessions.html +387 -0
- package/web-ui/partials/index/panel-settings.html +166 -0
- package/web-ui/session-helpers.mjs +12 -0
- package/web-ui/source-bundle.cjs +233 -0
- package/web-ui/styles/base-theme.css +373 -0
- package/web-ui/styles/controls-forms.css +354 -0
- package/web-ui/styles/feedback.css +108 -0
- package/web-ui/styles/health-check-dialog.css +144 -0
- package/web-ui/styles/layout-shell.css +330 -0
- package/web-ui/styles/modals-core.css +449 -0
- package/web-ui/styles/navigation-panels.css +381 -0
- package/web-ui/styles/openclaw-structured.css +266 -0
- package/web-ui/styles/responsive.css +416 -0
- package/web-ui/styles/sessions-list.css +414 -0
- package/web-ui/styles/sessions-preview.css +405 -0
- package/web-ui/styles/sessions-toolbar-trash.css +243 -0
- package/web-ui/styles/sessions-usage.css +276 -0
- package/web-ui/styles/skills-list.css +298 -0
- package/web-ui/styles/skills-market.css +335 -0
- package/web-ui/styles/titles-cards.css +407 -0
- package/web-ui/styles.css +16 -4499
- package/doc/CHANGELOG.md +0 -32
- package/doc/CHANGELOG.zh-CN.md +0 -34
|
@@ -0,0 +1,174 @@
|
|
|
1
|
+
<div
|
|
2
|
+
v-show="mainTab === 'market'"
|
|
3
|
+
class="mode-content"
|
|
4
|
+
id="panel-market"
|
|
5
|
+
role="tabpanel"
|
|
6
|
+
aria-labelledby="tab-market">
|
|
7
|
+
<div class="selector-section market-overview-section">
|
|
8
|
+
<div class="selector-header market-overview-header">
|
|
9
|
+
<div>
|
|
10
|
+
<span class="selector-title">Skills 概览</span>
|
|
11
|
+
<div class="skills-panel-note">聚焦本地 skills:切换目标、查看已装项、跨应用导入与 ZIP 分发。</div>
|
|
12
|
+
</div>
|
|
13
|
+
<div class="settings-tab-actions market-header-actions">
|
|
14
|
+
<button type="button" class="btn-tool btn-tool-compact" @click="loadSkillsMarketOverview({ forceRefresh: true, silent: false })" :disabled="loading || !!initError || skillsMarketBusy">
|
|
15
|
+
{{ skillsMarketLoading ? '刷新中...' : '刷新概览' }}
|
|
16
|
+
</button>
|
|
17
|
+
<button type="button" class="btn-tool btn-tool-compact" @click="openSkillsManager" :disabled="loading || !!initError || skillsMarketBusy">
|
|
18
|
+
打开 Skills 管理
|
|
19
|
+
</button>
|
|
20
|
+
</div>
|
|
21
|
+
</div>
|
|
22
|
+
|
|
23
|
+
<div class="market-target-switch" role="group" aria-label="选择 Skills 安装目标">
|
|
24
|
+
<button
|
|
25
|
+
type="button"
|
|
26
|
+
:class="['market-target-chip', { active: skillsTargetApp === 'codex' }]"
|
|
27
|
+
:aria-pressed="skillsTargetApp === 'codex'"
|
|
28
|
+
:disabled="loading || !!initError || skillsMarketBusy"
|
|
29
|
+
@click="setSkillsTargetApp('codex', { silent: false })">
|
|
30
|
+
Codex
|
|
31
|
+
</button>
|
|
32
|
+
<button
|
|
33
|
+
type="button"
|
|
34
|
+
:class="['market-target-chip', { active: skillsTargetApp === 'claude' }]"
|
|
35
|
+
:aria-pressed="skillsTargetApp === 'claude'"
|
|
36
|
+
:disabled="loading || !!initError || skillsMarketBusy"
|
|
37
|
+
@click="setSkillsTargetApp('claude', { silent: false })">
|
|
38
|
+
Claude Code
|
|
39
|
+
</button>
|
|
40
|
+
</div>
|
|
41
|
+
|
|
42
|
+
<div class="skills-root-box market-root-box">{{ skillsRootPath || skillsDefaultRootPath }}</div>
|
|
43
|
+
|
|
44
|
+
<div class="skills-summary-strip market-summary-strip">
|
|
45
|
+
<div class="skills-summary-item">
|
|
46
|
+
<span class="skills-summary-label">安装目标</span>
|
|
47
|
+
<strong class="skills-summary-value">{{ skillsTargetLabel }}</strong>
|
|
48
|
+
</div>
|
|
49
|
+
<div class="skills-summary-item">
|
|
50
|
+
<span class="skills-summary-label">本地总数</span>
|
|
51
|
+
<strong class="skills-summary-value">{{ skillsList.length }}</strong>
|
|
52
|
+
</div>
|
|
53
|
+
<div class="skills-summary-item">
|
|
54
|
+
<span class="skills-summary-label">含 SKILL.md</span>
|
|
55
|
+
<strong class="skills-summary-value">{{ skillsConfiguredCount }}</strong>
|
|
56
|
+
</div>
|
|
57
|
+
<div class="skills-summary-item">
|
|
58
|
+
<span class="skills-summary-label">缺少 SKILL.md</span>
|
|
59
|
+
<strong class="skills-summary-value">{{ skillsMissingSkillFileCount }}</strong>
|
|
60
|
+
</div>
|
|
61
|
+
<div class="skills-summary-item">
|
|
62
|
+
<span class="skills-summary-label">可导入</span>
|
|
63
|
+
<strong class="skills-summary-value">{{ skillsImportList.length }}</strong>
|
|
64
|
+
</div>
|
|
65
|
+
<div class="skills-summary-item">
|
|
66
|
+
<span class="skills-summary-label">可直接导入</span>
|
|
67
|
+
<strong class="skills-summary-value">{{ skillsImportConfiguredCount }}</strong>
|
|
68
|
+
</div>
|
|
69
|
+
</div>
|
|
70
|
+
</div>
|
|
71
|
+
|
|
72
|
+
<div class="market-grid">
|
|
73
|
+
<div class="skills-panel market-panel">
|
|
74
|
+
<div class="skills-panel-header">
|
|
75
|
+
<div class="skills-panel-title-wrap">
|
|
76
|
+
<div class="skills-panel-title">已安装 Skills</div>
|
|
77
|
+
<div class="skills-panel-note">展示当前目录前 6 项;更多操作请进管理弹窗。</div>
|
|
78
|
+
</div>
|
|
79
|
+
<button type="button" class="btn-mini" @click="refreshSkillsList({ silent: false })" :disabled="loading || !!initError || skillsMarketBusy">
|
|
80
|
+
{{ skillsLoading ? '刷新中...' : '刷新本地' }}
|
|
81
|
+
</button>
|
|
82
|
+
</div>
|
|
83
|
+
<div v-if="skillsLoading && !skillsMarketLocalLoadedOnce" class="skills-empty-state">正在加载本地 Skills...</div>
|
|
84
|
+
<div v-else-if="skillsList.length === 0" class="skills-empty-state">当前暂无已安装 skill,可通过 ZIP 或跨应用导入补充。</div>
|
|
85
|
+
<div v-else class="market-preview-list">
|
|
86
|
+
<div v-for="skill in skillsMarketInstalledPreview" :key="'market-local-' + skill.name" class="market-preview-item">
|
|
87
|
+
<div class="market-preview-main">
|
|
88
|
+
<div class="market-preview-title">{{ skill.displayName || skill.name }}</div>
|
|
89
|
+
<div class="market-preview-meta">{{ skill.description || skill.path }}</div>
|
|
90
|
+
</div>
|
|
91
|
+
<span :class="['pill', skill.hasSkillFile ? 'configured' : 'empty']">
|
|
92
|
+
{{ skill.hasSkillFile ? '已验证' : '待补 SKILL.md' }}
|
|
93
|
+
</span>
|
|
94
|
+
</div>
|
|
95
|
+
</div>
|
|
96
|
+
</div>
|
|
97
|
+
|
|
98
|
+
<div class="skills-panel market-panel">
|
|
99
|
+
<div class="skills-panel-header">
|
|
100
|
+
<div class="skills-panel-title-wrap">
|
|
101
|
+
<div class="skills-panel-title">可导入来源</div>
|
|
102
|
+
<div class="skills-panel-note">扫描其他应用中的未托管 skill,确认后批量导入到当前 {{ skillsTargetLabel }}。</div>
|
|
103
|
+
</div>
|
|
104
|
+
<button type="button" class="btn-mini" @click="scanImportableSkills({ silent: false })" :disabled="loading || !!initError || skillsMarketBusy">
|
|
105
|
+
{{ skillsScanningImports ? '扫描中...' : '扫描来源' }}
|
|
106
|
+
</button>
|
|
107
|
+
</div>
|
|
108
|
+
<div v-if="skillsScanningImports && !skillsMarketImportLoadedOnce" class="skills-empty-state">正在扫描可导入 skill...</div>
|
|
109
|
+
<div v-else-if="skillsImportList.length === 0" class="skills-empty-state">暂未扫描到可导入 skill,可点击“扫描来源”重试。</div>
|
|
110
|
+
<div v-else class="market-preview-list">
|
|
111
|
+
<div v-for="skill in skillsMarketImportPreview" :key="'market-import-' + buildSkillImportKey(skill)" class="market-preview-item">
|
|
112
|
+
<div class="market-preview-main">
|
|
113
|
+
<div class="market-preview-title">{{ skill.displayName || skill.name }}</div>
|
|
114
|
+
<div class="market-preview-meta">{{ skill.sourceLabel }} · {{ skill.sourcePath }}</div>
|
|
115
|
+
</div>
|
|
116
|
+
<span :class="['pill', skill.hasSkillFile ? 'configured' : 'empty']">
|
|
117
|
+
{{ skill.hasSkillFile ? '可直接导入' : '缺少 SKILL.md' }}
|
|
118
|
+
</span>
|
|
119
|
+
</div>
|
|
120
|
+
</div>
|
|
121
|
+
</div>
|
|
122
|
+
|
|
123
|
+
<div class="skills-panel market-panel market-actions-panel">
|
|
124
|
+
<div class="skills-panel-header">
|
|
125
|
+
<div class="skills-panel-title-wrap">
|
|
126
|
+
<div class="skills-panel-title">分发入口</div>
|
|
127
|
+
<div class="skills-panel-note">本地管理、跨应用导入与 ZIP 导入,均作用于当前安装目标。</div>
|
|
128
|
+
</div>
|
|
129
|
+
</div>
|
|
130
|
+
<div class="market-action-grid">
|
|
131
|
+
<button type="button" class="market-action-card" @click="openSkillsManager" :disabled="loading || !!initError || skillsMarketBusy">
|
|
132
|
+
<span class="market-action-title">本地 Skills 管理</span>
|
|
133
|
+
<span class="market-action-copy">查看并管理当前 {{ skillsTargetLabel }} 的已装 skills</span>
|
|
134
|
+
</button>
|
|
135
|
+
<button type="button" class="market-action-card" @click="scanImportableSkills({ silent: false })" :disabled="loading || !!initError || skillsMarketBusy">
|
|
136
|
+
<span class="market-action-title">跨应用导入</span>
|
|
137
|
+
<span class="market-action-copy">扫描其他应用并导入到当前 {{ skillsTargetLabel }}</span>
|
|
138
|
+
</button>
|
|
139
|
+
<button type="button" class="market-action-card" @click="triggerSkillsZipImport" :disabled="loading || !!initError || skillsMarketBusy">
|
|
140
|
+
<span class="market-action-title">ZIP 导入</span>
|
|
141
|
+
<span class="market-action-copy">从 ZIP 安装到当前目标</span>
|
|
142
|
+
</button>
|
|
143
|
+
</div>
|
|
144
|
+
</div>
|
|
145
|
+
|
|
146
|
+
<div class="skills-panel market-panel market-panel-wide">
|
|
147
|
+
<div class="skills-panel-header">
|
|
148
|
+
<div class="skills-panel-title-wrap">
|
|
149
|
+
<div class="skills-panel-title">市场说明</div>
|
|
150
|
+
</div>
|
|
151
|
+
</div>
|
|
152
|
+
<div class="market-preview-list">
|
|
153
|
+
<div class="market-preview-item">
|
|
154
|
+
<div class="market-preview-main">
|
|
155
|
+
<div class="market-preview-title">目标切换</div>
|
|
156
|
+
<div class="market-preview-meta">切换 Codex / Claude Code 后,后续操作都会落到当前 {{ skillsTargetLabel }} 目录。</div>
|
|
157
|
+
</div>
|
|
158
|
+
</div>
|
|
159
|
+
<div class="market-preview-item">
|
|
160
|
+
<div class="market-preview-main">
|
|
161
|
+
<div class="market-preview-title">跨应用导入</div>
|
|
162
|
+
<div class="market-preview-meta">扫描 `Codex`、`Claude Code` 与 `Agents` 中未托管的 skills,并导入到当前宿主。</div>
|
|
163
|
+
</div>
|
|
164
|
+
</div>
|
|
165
|
+
<div class="market-preview-item">
|
|
166
|
+
<div class="market-preview-main">
|
|
167
|
+
<div class="market-preview-title">ZIP 分发</div>
|
|
168
|
+
<div class="market-preview-meta">通过压缩包分发技能目录,保持本地可控,不依赖外部目录服务。</div>
|
|
169
|
+
</div>
|
|
170
|
+
</div>
|
|
171
|
+
</div>
|
|
172
|
+
</div>
|
|
173
|
+
</div>
|
|
174
|
+
</div>
|
|
@@ -0,0 +1,387 @@
|
|
|
1
|
+
<!-- 会话浏览模式 -->
|
|
2
|
+
<div
|
|
3
|
+
v-show="mainTab === 'sessions'"
|
|
4
|
+
class="mode-content"
|
|
5
|
+
id="panel-sessions"
|
|
6
|
+
role="tabpanel"
|
|
7
|
+
:aria-labelledby="'tab-sessions'">
|
|
8
|
+
<div v-if="sessionStandalone" class="session-standalone-page">
|
|
9
|
+
<div v-if="sessionStandaloneLoading" class="state-message">
|
|
10
|
+
加载中...
|
|
11
|
+
</div>
|
|
12
|
+
<div v-else-if="sessionStandaloneError" class="state-message error">
|
|
13
|
+
{{ sessionStandaloneError }}
|
|
14
|
+
</div>
|
|
15
|
+
<div v-else>
|
|
16
|
+
<div class="session-standalone-title">
|
|
17
|
+
{{ sessionStandaloneTitle }}
|
|
18
|
+
<span v-if="sessionStandaloneSourceLabel"> · {{ sessionStandaloneSourceLabel }}</span>
|
|
19
|
+
</div>
|
|
20
|
+
<pre class="session-standalone-text">{{ sessionStandaloneText }}</pre>
|
|
21
|
+
</div>
|
|
22
|
+
</div>
|
|
23
|
+
|
|
24
|
+
<div v-else>
|
|
25
|
+
<div class="sessions-subtabs" role="tablist" aria-label="会话视图切换">
|
|
26
|
+
<button
|
|
27
|
+
class="sessions-subtab"
|
|
28
|
+
:class="{ active: sessionsViewMode === 'browser' }"
|
|
29
|
+
type="button"
|
|
30
|
+
role="tab"
|
|
31
|
+
:aria-selected="sessionsViewMode === 'browser'"
|
|
32
|
+
@click="sessionsViewMode = 'browser'">
|
|
33
|
+
Sessions
|
|
34
|
+
</button>
|
|
35
|
+
<button
|
|
36
|
+
class="sessions-subtab"
|
|
37
|
+
:class="{ active: sessionsViewMode === 'usage' }"
|
|
38
|
+
type="button"
|
|
39
|
+
role="tab"
|
|
40
|
+
:aria-selected="sessionsViewMode === 'usage'"
|
|
41
|
+
@click="sessionsViewMode = 'usage'">
|
|
42
|
+
Usage
|
|
43
|
+
</button>
|
|
44
|
+
</div>
|
|
45
|
+
|
|
46
|
+
<div v-if="sessionsViewMode === 'usage'">
|
|
47
|
+
<div class="usage-toolbar">
|
|
48
|
+
<div class="selector-header" style="padding:0;border:0;background:none;">
|
|
49
|
+
<span class="selector-title">本地使用概览</span>
|
|
50
|
+
</div>
|
|
51
|
+
<div class="usage-range-group" role="group" aria-label="Usage 时间范围">
|
|
52
|
+
<button type="button" class="usage-range-btn" :class="{ active: sessionsUsageTimeRange === '7d' }" @click="sessionsUsageTimeRange = '7d'">近 7 天</button>
|
|
53
|
+
<button type="button" class="usage-range-btn" :class="{ active: sessionsUsageTimeRange === '30d' }" @click="sessionsUsageTimeRange = '30d'">近 30 天</button>
|
|
54
|
+
</div>
|
|
55
|
+
</div>
|
|
56
|
+
|
|
57
|
+
<div v-if="!sessionsList.length" class="usage-empty">暂无可用于统计的会话数据</div>
|
|
58
|
+
<template v-else>
|
|
59
|
+
<div class="usage-summary-grid">
|
|
60
|
+
<div v-for="card in sessionUsageSummaryCards" :key="card.key" class="usage-summary-card">
|
|
61
|
+
<div class="usage-summary-label">{{ card.label }}</div>
|
|
62
|
+
<div class="usage-summary-value">{{ card.value }}</div>
|
|
63
|
+
</div>
|
|
64
|
+
</div>
|
|
65
|
+
|
|
66
|
+
<div class="usage-chart-grid">
|
|
67
|
+
<section class="usage-card">
|
|
68
|
+
<div class="usage-card-title">会话趋势</div>
|
|
69
|
+
<div class="usage-legend">
|
|
70
|
+
<span><span class="usage-legend-dot" style="background:#4f8cff"></span>Codex</span>
|
|
71
|
+
<span><span class="usage-legend-dot" style="background:#b277ff"></span>Claude</span>
|
|
72
|
+
</div>
|
|
73
|
+
<div class="usage-bars">
|
|
74
|
+
<div v-for="bucket in sessionUsageCharts.buckets" :key="bucket.key" class="usage-bar-group">
|
|
75
|
+
<div class="usage-bar-stack">
|
|
76
|
+
<div class="usage-bar codex" :style="{ height: ((bucket.codex / Math.max(sessionUsageCharts.maxSessionBucket, 1)) * 100) + '%' }" :title="`Codex ${bucket.codex}`"></div>
|
|
77
|
+
<div class="usage-bar claude" :style="{ height: ((bucket.claude / Math.max(sessionUsageCharts.maxSessionBucket, 1)) * 100) + '%' }" :title="`Claude ${bucket.claude}`"></div>
|
|
78
|
+
</div>
|
|
79
|
+
<div class="usage-bar-label">{{ bucket.label }}</div>
|
|
80
|
+
</div>
|
|
81
|
+
</div>
|
|
82
|
+
</section>
|
|
83
|
+
|
|
84
|
+
<section class="usage-card">
|
|
85
|
+
<div class="usage-card-title">来源占比</div>
|
|
86
|
+
<div class="usage-list">
|
|
87
|
+
<div v-for="item in sessionUsageCharts.sourceShare" :key="item.key" class="usage-list-row">
|
|
88
|
+
<div class="usage-list-label">{{ item.label }}</div>
|
|
89
|
+
<div class="usage-progress"><div class="usage-progress-fill" :style="{ width: item.percent + '%' }"></div></div>
|
|
90
|
+
<div class="usage-list-value">{{ item.percent }}%</div>
|
|
91
|
+
</div>
|
|
92
|
+
</div>
|
|
93
|
+
</section>
|
|
94
|
+
|
|
95
|
+
<section class="usage-card">
|
|
96
|
+
<div class="usage-card-title">消息趋势</div>
|
|
97
|
+
<div class="usage-bars">
|
|
98
|
+
<div v-for="bucket in sessionUsageCharts.buckets" :key="bucket.key + '-messages'" class="usage-bar-group">
|
|
99
|
+
<div class="usage-bar-stack">
|
|
100
|
+
<div class="usage-bar codex" style="width:100%" :style="{ height: ((bucket.totalMessages / Math.max(sessionUsageCharts.maxMessageBucket, 1)) * 100) + '%' }" :title="`${bucket.totalMessages} messages`"></div>
|
|
101
|
+
</div>
|
|
102
|
+
<div class="usage-bar-label">{{ bucket.label }}</div>
|
|
103
|
+
</div>
|
|
104
|
+
</div>
|
|
105
|
+
</section>
|
|
106
|
+
|
|
107
|
+
<section class="usage-card">
|
|
108
|
+
<div class="usage-card-title">高频路径</div>
|
|
109
|
+
<div v-if="!sessionUsageCharts.topPaths.length" class="usage-list-value">暂无路径数据</div>
|
|
110
|
+
<div v-else class="usage-list">
|
|
111
|
+
<div v-for="item in sessionUsageCharts.topPaths" :key="item.path" class="usage-list-row">
|
|
112
|
+
<div class="usage-list-label">{{ item.count }} 次</div>
|
|
113
|
+
<div class="usage-progress"><div class="usage-progress-fill" :style="{ width: ((item.count / Math.max(sessionUsageCharts.topPaths[0]?.count || 1, 1)) * 100) + '%' }"></div></div>
|
|
114
|
+
<div class="usage-list-value" :title="item.path">{{ item.path }}</div>
|
|
115
|
+
</div>
|
|
116
|
+
</div>
|
|
117
|
+
</section>
|
|
118
|
+
</div>
|
|
119
|
+
</template>
|
|
120
|
+
</div>
|
|
121
|
+
|
|
122
|
+
<template v-else>
|
|
123
|
+
<div class="selector-section">
|
|
124
|
+
<div class="selector-header">
|
|
125
|
+
<span class="selector-title">会话来源</span>
|
|
126
|
+
<div class="selector-actions">
|
|
127
|
+
<button class="btn-tool btn-tool-compact" @click="loadSessions" :disabled="sessionsLoading">
|
|
128
|
+
{{ sessionsLoading ? '刷新中...' : '刷新会话' }}
|
|
129
|
+
</button>
|
|
130
|
+
</div>
|
|
131
|
+
</div>
|
|
132
|
+
<div class="session-toolbar">
|
|
133
|
+
<div class="session-toolbar-group">
|
|
134
|
+
<select class="session-source-select" v-model="sessionFilterSource" @change="onSessionSourceChange" :disabled="sessionsLoading">
|
|
135
|
+
<option value="all">全部</option>
|
|
136
|
+
<option value="codex">Codex</option>
|
|
137
|
+
<option value="claude">Claude Code</option>
|
|
138
|
+
</select>
|
|
139
|
+
<select
|
|
140
|
+
class="session-path-select"
|
|
141
|
+
v-model="sessionPathFilter"
|
|
142
|
+
@change="onSessionPathFilterChange"
|
|
143
|
+
:disabled="sessionsLoading">
|
|
144
|
+
<option value="">全部路径</option>
|
|
145
|
+
<option v-for="cwd in sessionPathOptions" :key="cwd" :value="cwd">{{ cwd }}</option>
|
|
146
|
+
</select>
|
|
147
|
+
</div>
|
|
148
|
+
<div class="session-toolbar-group session-toolbar-grow">
|
|
149
|
+
<input
|
|
150
|
+
class="session-query-input"
|
|
151
|
+
v-model="sessionQuery"
|
|
152
|
+
@keyup.enter="loadSessions"
|
|
153
|
+
:disabled="sessionsLoading || !isSessionQueryEnabled"
|
|
154
|
+
:placeholder="sessionQueryPlaceholder">
|
|
155
|
+
</div>
|
|
156
|
+
<div class="session-toolbar-group">
|
|
157
|
+
<select
|
|
158
|
+
class="session-role-select"
|
|
159
|
+
v-model="sessionRoleFilter"
|
|
160
|
+
@change="onSessionFilterChange"
|
|
161
|
+
disabled>
|
|
162
|
+
<option value="all">全部角色</option>
|
|
163
|
+
<option value="user">仅 User</option>
|
|
164
|
+
<option value="assistant">仅 Assistant</option>
|
|
165
|
+
<option value="system">仅 System</option>
|
|
166
|
+
</select>
|
|
167
|
+
<select
|
|
168
|
+
class="session-time-select"
|
|
169
|
+
v-model="sessionTimePreset"
|
|
170
|
+
@change="onSessionFilterChange"
|
|
171
|
+
disabled>
|
|
172
|
+
<option value="all">全部时间</option>
|
|
173
|
+
<option value="7d">近 7 天</option>
|
|
174
|
+
<option value="30d">近 30 天</option>
|
|
175
|
+
<option value="90d">近 90 天</option>
|
|
176
|
+
</select>
|
|
177
|
+
</div>
|
|
178
|
+
</div>
|
|
179
|
+
<div class="session-toolbar-footer">
|
|
180
|
+
<label class="quick-option">
|
|
181
|
+
<input
|
|
182
|
+
type="checkbox"
|
|
183
|
+
v-model="sessionResumeWithYolo"
|
|
184
|
+
@change="onSessionResumeYoloChange"
|
|
185
|
+
>
|
|
186
|
+
复制恢复命令附带 --yolo
|
|
187
|
+
</label>
|
|
188
|
+
</div>
|
|
189
|
+
</div>
|
|
190
|
+
|
|
191
|
+
<div v-if="sessionsLoading" class="state-message">
|
|
192
|
+
会话加载中...
|
|
193
|
+
</div>
|
|
194
|
+
|
|
195
|
+
<div v-else-if="sessionsList.length === 0" class="session-empty">
|
|
196
|
+
暂无可用会话记录
|
|
197
|
+
</div>
|
|
198
|
+
|
|
199
|
+
<div v-else class="session-layout">
|
|
200
|
+
<div v-if="sessionListRenderEnabled" class="session-list">
|
|
201
|
+
<div
|
|
202
|
+
v-for="session in sortedSessionsList"
|
|
203
|
+
:key="session.source + '-' + session.sessionId + '-' + session.filePath"
|
|
204
|
+
v-memo="[activeSessionExportKey === getSessionExportKey(session), session.messageCount, session.updatedAt, session.title, session.sourceLabel, isSessionPinned(session), sessionsLoading]"
|
|
205
|
+
:class="[
|
|
206
|
+
'session-item',
|
|
207
|
+
{
|
|
208
|
+
active: activeSessionExportKey === getSessionExportKey(session),
|
|
209
|
+
pinned: isSessionPinned(session)
|
|
210
|
+
}
|
|
211
|
+
]"
|
|
212
|
+
@click="selectSession(session)"
|
|
213
|
+
@keydown.enter.self.prevent="selectSession(session)"
|
|
214
|
+
@keydown.space.self.prevent="selectSession(session)"
|
|
215
|
+
tabindex="0"
|
|
216
|
+
role="button"
|
|
217
|
+
:aria-current="activeSessionExportKey === getSessionExportKey(session) ? 'true' : null">
|
|
218
|
+
<div class="session-item-header">
|
|
219
|
+
<div class="session-item-main">
|
|
220
|
+
<div class="session-item-title">{{ session.title || session.sessionId }}</div>
|
|
221
|
+
<span class="session-count-badge">{{ session.messageCount ?? 0 }}</span>
|
|
222
|
+
</div>
|
|
223
|
+
<div class="session-item-actions">
|
|
224
|
+
<button
|
|
225
|
+
class="session-item-copy session-item-pin"
|
|
226
|
+
@click.stop="toggleSessionPin(session)"
|
|
227
|
+
:disabled="sessionsLoading"
|
|
228
|
+
:aria-label="isSessionPinned(session) ? '取消置顶' : '置顶'"
|
|
229
|
+
:title="isSessionPinned(session) ? '取消置顶' : '置顶'"
|
|
230
|
+
:aria-pressed="isSessionPinned(session)">
|
|
231
|
+
<svg v-if="isSessionPinned(session)" class="pin-icon" viewBox="0 0 24 24" fill="currentColor" stroke="currentColor" stroke-width="1.6">
|
|
232
|
+
<path d="M12 22s8-6 8-12a8 8 0 1 0-16 0c0 6 8 12 8 12z"></path>
|
|
233
|
+
</svg>
|
|
234
|
+
<svg v-else class="pin-icon" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="1.6">
|
|
235
|
+
<path d="M12 22s8-6 8-12a8 8 0 1 0-16 0c0 6 8 12 8 12z"></path>
|
|
236
|
+
</svg>
|
|
237
|
+
</button>
|
|
238
|
+
<button
|
|
239
|
+
v-if="isResumeCommandAvailable(session)"
|
|
240
|
+
class="session-item-copy"
|
|
241
|
+
@click.stop="copyResumeCommand(session)"
|
|
242
|
+
:disabled="sessionsLoading"
|
|
243
|
+
aria-label="复制恢复命令"
|
|
244
|
+
title="复制恢复命令">
|
|
245
|
+
<svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2">
|
|
246
|
+
<rect x="8" y="8" width="12" height="12" rx="2"></rect>
|
|
247
|
+
<path d="M16 8V6a2 2 0 0 0-2-2H6a2 2 0 0 0-2 2v8a2 2 0 0 0 2 2h2"></path>
|
|
248
|
+
</svg>
|
|
249
|
+
</button>
|
|
250
|
+
</div>
|
|
251
|
+
</div>
|
|
252
|
+
<div class="session-item-meta">
|
|
253
|
+
<span class="session-source">{{ session.sourceLabel }}</span>
|
|
254
|
+
<span class="session-item-time">{{ session.updatedAt || 'unknown time' }}</span>
|
|
255
|
+
</div>
|
|
256
|
+
</div>
|
|
257
|
+
</div>
|
|
258
|
+
<div v-else class="session-list session-list-placeholder"></div>
|
|
259
|
+
|
|
260
|
+
<div :class="['session-preview', { active: !!activeSession }]" :ref="setSessionPreviewContainerRef">
|
|
261
|
+
<template v-if="activeSession">
|
|
262
|
+
<div class="session-preview-scroll" :ref="setSessionPreviewScrollRef" @scroll="onSessionPreviewScroll">
|
|
263
|
+
<div class="session-preview-header" :ref="setSessionPreviewHeaderRef">
|
|
264
|
+
<div>
|
|
265
|
+
<div class="session-preview-title">{{ activeSession.title || activeSession.sessionId }}</div>
|
|
266
|
+
<div class="session-preview-meta">
|
|
267
|
+
<span class="session-preview-meta-item">{{ activeSession.sourceLabel }}</span>
|
|
268
|
+
<span class="session-preview-meta-item">{{ activeSession.updatedAt || 'unknown time' }}</span>
|
|
269
|
+
</div>
|
|
270
|
+
<div class="session-preview-meta" v-if="activeSession.cwd">
|
|
271
|
+
<span class="session-preview-meta-item">{{ activeSession.cwd }}</span>
|
|
272
|
+
</div>
|
|
273
|
+
</div>
|
|
274
|
+
<div class="session-actions">
|
|
275
|
+
<button class="btn-session-refresh" @click="loadActiveSessionDetail" :disabled="sessionDetailLoading || !activeSession">
|
|
276
|
+
{{ sessionDetailLoading ? '加载中...' : '刷新内容' }}
|
|
277
|
+
</button>
|
|
278
|
+
<button
|
|
279
|
+
v-if="isDeleteAvailable(activeSession)"
|
|
280
|
+
class="btn-session-delete"
|
|
281
|
+
@click="deleteSession(activeSession)"
|
|
282
|
+
:disabled="!activeSession || sessionsLoading || sessionDeleting[getSessionExportKey(activeSession)]">
|
|
283
|
+
{{ (activeSession && sessionDeleting[getSessionExportKey(activeSession)]) ? '移入中...' : '移入回收站' }}
|
|
284
|
+
</button>
|
|
285
|
+
<button
|
|
286
|
+
class="btn-session-export"
|
|
287
|
+
@click="exportSession(activeSession)"
|
|
288
|
+
:disabled="!activeSession || sessionExporting[getSessionExportKey(activeSession)]">
|
|
289
|
+
{{ (activeSession && sessionExporting[getSessionExportKey(activeSession)]) ? '导出中...' : '导出记录' }}
|
|
290
|
+
</button>
|
|
291
|
+
<button
|
|
292
|
+
class="btn-session-open"
|
|
293
|
+
@click="openSessionStandalone(activeSession)"
|
|
294
|
+
:disabled="!activeSession">
|
|
295
|
+
新页查看
|
|
296
|
+
</button>
|
|
297
|
+
</div>
|
|
298
|
+
</div>
|
|
299
|
+
|
|
300
|
+
<div v-if="sessionDetailLoading && !sessionPreviewLoadingMore" class="session-preview-empty">
|
|
301
|
+
正在加载会话内容...
|
|
302
|
+
</div>
|
|
303
|
+
|
|
304
|
+
<div v-else-if="activeSessionDetailError" class="session-preview-empty">
|
|
305
|
+
{{ activeSessionDetailError }}
|
|
306
|
+
</div>
|
|
307
|
+
|
|
308
|
+
<div v-else-if="!activeSessionMessages.length" class="session-preview-empty">
|
|
309
|
+
当前会话暂无可展示消息
|
|
310
|
+
</div>
|
|
311
|
+
|
|
312
|
+
<div v-else-if="sessionPreviewRenderEnabled && !activeSessionVisibleMessages.length" class="session-preview-empty">
|
|
313
|
+
<span>正在渲染会话内容...</span>
|
|
314
|
+
<button class="btn-session-refresh" @click="primeSessionPreviewMessageRender" :disabled="sessionDetailLoading">
|
|
315
|
+
重新渲染
|
|
316
|
+
</button>
|
|
317
|
+
</div>
|
|
318
|
+
|
|
319
|
+
<div v-else-if="!sessionPreviewRenderEnabled" class="session-preview-empty">
|
|
320
|
+
正在准备会话内容...
|
|
321
|
+
</div>
|
|
322
|
+
|
|
323
|
+
<div v-else class="session-preview-body">
|
|
324
|
+
<div class="session-preview-messages">
|
|
325
|
+
<div v-if="activeSessionDetailClipped" class="session-item-sub session-item-wrap">
|
|
326
|
+
仅展示最近 {{ activeSessionMessages.length }} 条消息。
|
|
327
|
+
</div>
|
|
328
|
+
<div
|
|
329
|
+
v-if="canLoadMoreSessionMessages"
|
|
330
|
+
class="session-item-sub session-item-wrap"
|
|
331
|
+
style="display:flex;align-items:center;justify-content:space-between;gap:8px;">
|
|
332
|
+
<span>已显示 {{ activeSessionVisibleMessages.length }} / {{ activeSessionMessages.length }} 条</span>
|
|
333
|
+
<button class="btn-session-refresh" @click="loadMoreSessionMessages()" :disabled="sessionDetailLoading || sessionPreviewLoadingMore">
|
|
334
|
+
{{ sessionPreviewLoadingMore ? '加载中...' : ('加载更多(剩余 ' + sessionPreviewRemainingCount + ')') }}
|
|
335
|
+
</button>
|
|
336
|
+
</div>
|
|
337
|
+
<div
|
|
338
|
+
v-if="sessionPreviewLoadingMore"
|
|
339
|
+
class="session-item-sub session-item-wrap">
|
|
340
|
+
正在加载更早消息...
|
|
341
|
+
</div>
|
|
342
|
+
<div
|
|
343
|
+
v-for="(msg, idx) in activeSessionVisibleMessages"
|
|
344
|
+
:key="getRecordRenderKey(msg, idx)"
|
|
345
|
+
v-memo="[msg.text, msg.timestamp, msg.roleLabel, msg.normalizedRole]"
|
|
346
|
+
:data-message-key="getRecordRenderKey(msg, idx)"
|
|
347
|
+
:ref="getSessionMessageRefBinder(getRecordRenderKey(msg, idx))"
|
|
348
|
+
:class="['session-msg', msg.normalizedRole === 'user' ? 'user' : (msg.normalizedRole === 'system' ? 'system' : 'assistant')]">
|
|
349
|
+
<div class="session-msg-header">
|
|
350
|
+
<div class="session-msg-meta">
|
|
351
|
+
<span class="session-msg-role">{{ msg.roleLabel || (msg.normalizedRole === 'user' ? 'User' : (msg.normalizedRole === 'system' ? 'System' : 'Assistant')) }}</span>
|
|
352
|
+
<span class="session-msg-time">{{ msg.timestamp || '' }}</span>
|
|
353
|
+
</div>
|
|
354
|
+
</div>
|
|
355
|
+
<div class="session-msg-content">{{ msg.text || '' }}</div>
|
|
356
|
+
</div>
|
|
357
|
+
</div>
|
|
358
|
+
</div>
|
|
359
|
+
</div>
|
|
360
|
+
<aside v-if="sessionPreviewRenderEnabled && sessionTimelineNodes.length" class="session-timeline" aria-label="会话时间轴">
|
|
361
|
+
<div class="session-timeline-track"></div>
|
|
362
|
+
<button
|
|
363
|
+
v-for="node in sessionTimelineNodes"
|
|
364
|
+
:key="'timeline-' + node.key"
|
|
365
|
+
v-memo="[sessionTimelineActiveKey === node.key, node.safePercent, node.title]"
|
|
366
|
+
type="button"
|
|
367
|
+
:class="['session-timeline-node', { active: sessionTimelineActiveKey === node.key }]"
|
|
368
|
+
:aria-current="sessionTimelineActiveKey === node.key ? 'true' : null"
|
|
369
|
+
:style="{ top: `${node.safePercent}%` }"
|
|
370
|
+
:title="node.title"
|
|
371
|
+
@click="jumpToSessionTimelineNode(node.key)">
|
|
372
|
+
<span class="sr-only">{{ node.title }}</span>
|
|
373
|
+
</button>
|
|
374
|
+
<div class="session-timeline-current" v-if="sessionTimelineActiveTitle">
|
|
375
|
+
{{ sessionTimelineActiveTitle }}
|
|
376
|
+
</div>
|
|
377
|
+
</aside>
|
|
378
|
+
</template>
|
|
379
|
+
|
|
380
|
+
<div v-else class="session-preview-empty">
|
|
381
|
+
<span>请先在左侧选择一个会话</span>
|
|
382
|
+
</div>
|
|
383
|
+
</div>
|
|
384
|
+
</div>
|
|
385
|
+
</template>
|
|
386
|
+
</div>
|
|
387
|
+
</div>
|