导出页面的聊天列表新增最近活跃时间,并支持按记录数&聊天排序。新增AI生成的项目架构分析文档

This commit is contained in:
chadwbur
2026-04-19 21:08:50 +08:00
parent 55885449a3
commit 1b7197cdef
4 changed files with 852 additions and 24 deletions

4
.gitignore vendored
View File

@@ -75,4 +75,6 @@ pnpm-lock.yaml
wechat-research-site
.codex
weflow-web-offical
/Wedecrypt
/Wedecrypt
.codebuddy/
.DS_Store

596
docs/ARCHITECTURE.md Normal file
View File

@@ -0,0 +1,596 @@
# WeFlow 项目蓝图与架构文档
> 版本:对应 `package.json` v4.3.0 · 生成时间2026-04-17
> 适用于开发者、新成员上手、AgentCodeBuddy 等)自动化协作
---
## 1. 项目定位
**WeFlow** 是一个**完全本地**的微信 4.0+ 聊天记录查看、分析与导出的桌面应用。
| 维度 | 说明 |
|------|------|
| 产品形态 | Electron 桌面应用Windows / macOS / Linux |
| 核心诉求 | 实时查看 & 解密本地微信数据库、生成聊天分析 / 年度报告 / 双人报告、导出多格式、朋友圈解密 |
| 数据边界 | 全部本地运行,无云端上传;可选开放本地 HTTP API端口 5031 |
| 许可 | 见 `LICENSE` |
| 版本策略 | electron-updater + GitHub Releases支持自动更新与差分包 |
### 1.1 关键功能矩阵
| 模块 | 能力概述 | 主要代码落点 |
|------|---------|-------------|
| 实时聊天查看 | 消息列表、撤回防护、实时刷新 | [ChatPage.tsx](src/pages/ChatPage.tsx) + [chatService.ts](electron/services/chatService.ts) |
| 图片/视频/实况解密 | XOR / AES + ffmpeg 转码 | [imageDecryptService.ts](electron/services/imageDecryptService.ts), [videoService.ts](electron/services/videoService.ts) |
| 私聊/群聊分析 | 统计消息、时段、画像 | [analyticsService.ts](electron/services/analyticsService.ts), [groupAnalyticsService.ts](electron/services/groupAnalyticsService.ts) |
| 年度 / 双人报告 | 跨年数据生成、可视化 | [annualReportService.ts](electron/services/annualReportService.ts), [dualReportService.ts](electron/services/dualReportService.ts) + 对应 Worker |
| 导出 | JSON / HTML / TXT / Excel / CSV / ChatLab | [exportService.ts](electron/services/exportService.ts) + [exportWorker.ts](electron/exportWorker.ts) |
| 朋友圈 | 图片/视频/实况解密、时间突破 | [snsService.ts](electron/services/snsService.ts) + [SnsPage.tsx](src/pages/SnsPage.tsx) |
| HTTP API | 本地消息 API 服务 | [httpService.ts](electron/services/httpService.ts) + [docs/HTTP-API.md](docs/HTTP-API.md) |
| 语音转写 | sherpa-onnx ASR + silk-wasm | [voiceTranscribeService.ts](electron/services/voiceTranscribeService.ts) + [transcribeWorker.ts](electron/transcribeWorker.ts) |
| 通知与防撤回 | 桌面弹窗、黑白名单 | [messagePushService.ts](electron/services/messagePushService.ts), [notificationWindow.ts](electron/windows/notificationWindow.ts) |
| 应用锁 | Windows Hello / 系统凭据 | [windowsHelloService.ts](electron/services/windowsHelloService.ts) + [LockScreen.tsx](src/components/LockScreen.tsx) |
---
## 2. 技术栈总览
### 2.1 运行时 & 构建
| 层 | 技术 | 说明 |
|----|------|------|
| Shell | Electron 41 | 主进程 + 渲染进程分离 |
| 渲染进程 | React 19 + TypeScript 6 | 使用 `react-router-dom@7` 路由 |
| 构建 | Vite 7 + `vite-plugin-electron` + `vite-plugin-electron-renderer` | 一次构建产出主/渲染/Worker |
| 样式 | SCSS`sass` + 组件局部样式 | 深浅色 + 主题色切换(`data-theme` / `data-mode` |
| 打包 | electron-builder 26 | Win `.exe`(NSIS)、macOS `.dmg/.zip`、Linux `AppImage/tar.gz` |
| 自动更新 | electron-updater + GitHub provider | 差分包关闭,支持强制更新 |
### 2.2 核心依赖
| 类别 | 库 | 用途 |
|------|----|------|
| 状态管理 | `zustand` | 轻量全局 store`src/stores/` |
| UI 图标 | `lucide-react` | 图标系统 |
| 图表 | `echarts` + `echarts-for-react` | 分析与报告可视化 |
| 长列表 | `react-virtuoso` | 聊天消息虚拟滚动 |
| Markdown | `react-markdown` + `remark-gfm` | 报告与富文本 |
| 数据库 | `wcdb` 原生 + `koffi` FFI | 微信 SQLite/WCDB 加密数据库读取 |
| 多媒体 | `ffmpeg-static``silk-wasm``sherpa-onnx-node` | 视频解码、silk 语音、ASR |
| 中文分词 | `jieba-wasm` | 词云与分析 |
| 配置 | `electron-store` | JSON 持久化 |
| 导出 | `exceljs``jszip``html2canvas` | 多格式产物 |
| 辅助 | `fzstd``wechat-emojis``sudo-prompt` | zstd 解压、表情、提权 |
---
## 3. 目录蓝图
```
WeFlow/
├── electron/ # 主进程与 WorkerNode.js 环境)
│ ├── main.ts # 主进程入口≈122KBIPC 汇聚点)
│ ├── preload.ts # 预加载脚本,暴露 window.electronAPI
│ ├── preload-env.ts # 环境预加载(启动前)
│ ├── annualReportWorker.ts # 年度报告工作线程
│ ├── dualReportWorker.ts # 双人报告工作线程
│ ├── exportWorker.ts # 导出工作线程
│ ├── imageSearchWorker.ts # 图像检索/遍历工作线程
│ ├── transcribeWorker.ts # 语音转写工作线程
│ ├── wcdbWorker.ts # WCDB 读取工作线程
│ ├── services/ # 业务服务(按域拆分)
│ ├── windows/notificationWindow.ts # 独立通知窗口
│ ├── utils/LRUCache.ts # 主进程工具
│ ├── assets/wasm/ # wasm 资源jieba、silk 等)
│ └── types/ # 原生模块类型声明
├── src/ # 渲染进程React
│ ├── main.tsx / App.tsx / App.scss # 入口与根路由
│ ├── pages/ # 页面级组件(大文件集中地)
│ ├── components/ # 通用/业务组件
│ │ ├── Export/ # 导出子组件
│ │ └── Sns/ # 朋友圈子组件
│ ├── services/ # 渲染层服务(通过 IPC 调用主进程)
│ ├── stores/ # Zustand 全局状态
│ ├── styles/ # 全局样式与主题
│ ├── types/ # 渲染层类型
│ └── utils/ # 工具函数
├── docs/ # 项目文档HTTP-API、架构文档等
├── resources/ # 平台资源icons / runtime / wcdb / key / installer
├── public/ # 前端静态资源 + splash
├── .github/workflows/ # CIrelease、nightly、security-scan 等
├── AGENTS.md # Agent 协作规则
├── README.md # 用户说明
├── package.json # 依赖 + electron-builder 配置
├── vite.config.ts # 构建管线1 主 + 7 Worker + preload
├── tsconfig.json / tsconfig.node.json
├── installer.nsh # NSIS 安装脚本
└── .gitleaks.toml # 密钥扫描配置
```
---
## 4. 架构总览
### 4.1 高层架构图
```mermaid
flowchart TB
subgraph User["用户"]
U([👤])
end
subgraph Renderer["渲染进程 (Chromium + React 19)"]
UI[["Pages / Components"]]
Stores[(Zustand Stores)]
RSvc["渲染层服务\n(src/services)"]
UI <--> Stores
UI --> RSvc
end
subgraph Preload["preload.ts\n(contextBridge)"]
API["window.electronAPI"]
end
subgraph Main["主进程 (Node.js)"]
IPC[["ipcMain\nhandlers (main.ts)"]]
subgraph Services["electron/services/*"]
Chat[chatService]
Export[exportService]
Image[imageDecryptService]
Sns[snsService]
Analytics[analyticsService]
GAnalytics[groupAnalyticsService]
Annual[annualReportService]
Dual[dualReportService]
Http[httpService]
Key[keyService + Mac/Linux/WindowsHello]
Wcdb[wcdbService / wcdbCore]
Voice[voiceTranscribeService]
Msg[messagePushService]
Cfg[config]
end
subgraph Workers["Utility Workers (vite 独立打包)"]
W1[wcdbWorker]
W2[exportWorker]
W3[annualReportWorker]
W4[dualReportWorker]
W5[transcribeWorker]
W6[imageSearchWorker]
end
IPC --> Services
Services --> Workers
end
subgraph OS["操作系统 & 外部资源"]
Fs[(本地文件系统\n微信数据目录)]
NativeLibs[(原生库\nkoffi / sherpa-onnx / ffmpeg / wcdb)]
Store[(electron-store\nJSON 配置)]
HttpClient[(外部 HTTP 客户端\nChatLab / 脚本)]
Updater[(GitHub Releases\nelectron-updater)]
end
U --> UI
RSvc --> API
API <--> IPC
Services --> Fs
Services --> NativeLibs
Services --> Store
Http -. 暴露 5031 .-> HttpClient
Main -. 自动更新 .-> Updater
```
### 4.2 进程与线程模型
```mermaid
flowchart LR
Main[[主进程\nelectron/main.ts]]
Pre[[Preload\nelectron/preload.ts]]
R1[渲染:主窗口\nindex.html + App.tsx]
R2[渲染:通知窗口]
R3[渲染:视频/图片独立窗口]
R4[渲染:聊天记录窗口]
R5[渲染:年度/双人报告窗口]
W1((wcdbWorker))
W2((exportWorker))
W3((annualReportWorker))
W4((dualReportWorker))
W5((transcribeWorker))
W6((imageSearchWorker))
Main --- Pre --- R1
Main --- R2
Main --- R3
Main --- R4
Main --- R5
Main -. Worker_threads/child_process .-> W1
Main -. 同上 .-> W2
Main -. 同上 .-> W3
Main -. 同上 .-> W4
Main -. 同上 .-> W5
Main -. 同上 .-> W6
```
> 关键设计:**CPU 密集或长耗任务全部外包到独立 Worker**,通过 `vite.config.ts` 的多 entry 独立打包。`inlineDynamicImports: true` 保证 worker 单文件产物可 `new Worker(path)` 直接加载。
### 4.3 通信契约IPC 命名空间)
`electron/main.ts``ipcMain.handle` / `ipcMain.on` 采用 **前缀:动作** 命名空间,统一走 [preload.ts](electron/preload.ts) 暴露的 `window.electronAPI.<ns>.<method>`
| 命名空间 | 典型通道 | 所属服务 |
|---------|---------|---------|
| `config:*` | `get` / `set` / `clear` | `config.ts` |
| `dialog:*` | `openFile` / `openDirectory` / `saveFile` | Electron `dialog` |
| `shell:*` | `openPath` / `openExternal` | Electron `shell` |
| `app:*` | `getVersion` / `checkForUpdates` / `downloadAndInstall` / `ignoreUpdate` / `getLaunchAtStartupStatus` | 主进程 + updater |
| `window:*` | `minimize` / `maximize` / `close` / `isMaximized` / `openVideoPlayerWindow` / `openChatHistoryWindow` / `openSessionChatWindow` / `respondCloseConfirm` / `setTitleBarOverlay` | 主窗口管理 |
| `log:*` / `diagnostics:*` | 日志读取、导出卡片诊断 | 日志系统 |
| `cloud:*` | `init` / `recordPage` / `getLogs` | `cloudControlService` |
| `insight:*` | `testConnection` / `getTodayStats` / `triggerTest` / `generateFootprintInsight` | `insightService` |
| `video:*` | `getVideoInfo` / `parseVideoMd5` | `videoService` |
| `dbpath:*` | `autoDetect` / `scanWxids` / `scanWxidCandidates` / `getDefault` | `dbPathService` |
| `wcdb:*` | `testConnection` / `open` / `close` | `wcdbService` |
| `chat:*` | 会话、消息、联系人等(见 `preload.ts` | `chatService` |
| `export:*` | 导出任务、进度、取消 | `exportService` |
| `sns:*` | 朋友圈列表、解密 | `snsService` |
| `analytics:*` / `group-analytics:*` | 统计查询 | 对应 Service |
| `annual-report:*` / `dual-report:*` | 报告生成 | 对应 Service |
| `voice:*` | 语音转写 | `voiceTranscribeService` |
| `auth:*` | 应用锁状态 | `keyService*` / `windowsHelloService` |
| `notification:*` | 新消息通知与跳转 | `messagePushService` + `notificationWindow` |
> 上述仅为骨架,完整 IPC 契约以 [preload.ts](electron/preload.ts) 为唯一来源≈28KB包含完整白名单与类型
---
## 5. 渲染进程结构
### 5.1 路由蓝图
基于 [App.tsx](src/App.tsx) 的实际路由:
```mermaid
flowchart TB
Start((启动)) --> Gate{路由判定}
Gate -->|"/agreement-window"| AgreeWin[AgreementPage · 独立窗口]
Gate -->|"/onboarding-window"| Welcome[WelcomePage · standalone]
Gate -->|"/video-player-window"| VideoWin[VideoWindow]
Gate -->|"/image-viewer-window"| ImgWin[ImageWindow]
Gate -->|"/chat-history/..."| HistWin[ChatHistoryPage]
Gate -->|"/chat-window"| ChatStandalone[ChatPage · standalone]
Gate -->|"/notification-window"| NotifWin[NotificationWindow]
Gate -->|主窗口| Main[[主布局:\nTitleBar + Sidebar + Content]]
Main --> Home[/home: HomePage/]
Main --> AcctMgmt[/account-management/]
Main --> Chat[/chat: ChatPage/]
Main --> AnalyticsHub[/analytics: ChatAnalyticsHubPage/]
AnalyticsHub --> PrivAnaWel[/analytics/private: Welcome/]
AnalyticsHub --> PrivAna[/analytics/private/view/]
AnalyticsHub --> GroupAna[/analytics/group/]
Main --> Annual[/annual-report + /view/]
Main --> Dual[/dual-report + /view/]
Main --> Footprint[/footprint/]
Main --> Export[/export: ExportPage · keepalive/]
Main --> Sns[/sns: SnsPage/]
Main --> Biz[/biz: BizPage/]
Main --> Contacts[/contacts: ContactsPage/]
Main --> Res[/resources: ResourcesPage/]
Main -. 叠加 .-> Settings[/settings: SettingsPage · 背景路由切换/]
```
**亮点**
- **ExportPage 采用 keep-alive 模式**:用 `export-keepalive-page` DOM 容器常驻,仅切换 `active/hidden` class避免长任务重置。
- **Settings 路由叠加**:通过 `location.state.backgroundLocation` 实现在任意页面上浮设置面板。
- **多独立窗口**:视频、图片、通知、聊天记录、会话聊天、年度/双人报告均有独立 `BrowserWindow`
### 5.2 Zustand Stores
| Store | 文件 | 职责 |
|-------|------|------|
| `useAppStore` | `src/stores/appStore.ts` | 数据库连接状态、更新信息、锁屏 |
| `useThemeStore` | `src/stores/themeStore.ts` | 主题 ID / 模式light/dark/system |
| `useChatStore` | `src/stores/chatStore.ts` | 当前会话、选中消息 |
| `useAnalyticsStore` | `src/stores/analyticsStore.ts` | 分析页参数与缓存 |
| `useImageStore` | `src/stores/imageStore.ts` | 图片解密结果缓存 |
| `useBatchImageDecryptStore` | `src/stores/batchImageDecryptStore.ts` | 全局批量解密进度 |
| `useBatchTranscribeStore` | `src/stores/batchTranscribeStore.ts` | 全局批量转写进度 |
| `useContactTypeCountsStore` | `src/stores/contactTypeCountsStore.ts` | 联系人分类计数缓存 |
### 5.3 渲染层服务
| 文件 | 角色 |
|------|------|
| `src/services/ipc.ts` | IPC 桥接基础(与 preload 对齐) |
| `src/services/config.ts` | 配置封装73KB大量常量与 setter/getter |
| `src/services/exportBridge.ts` | 导出事件桥(主→渲染进度广播) |
| `src/services/cloudControl.ts` | 云控 / 统计上报(仅在用户同意后启用) |
| `src/services/backgroundTaskMonitor.ts` | 后台任务监测(解密、转写、导出) |
---
## 6. 主进程服务层(`electron/services/`
### 6.1 服务地图(按规模分层)
```mermaid
flowchart LR
subgraph Core["核心数据层 (超大)"]
ChatService["chatService.ts\n389KB"]
WcdbCore["wcdbCore.ts\n177KB"]
ExportService["exportService.ts\n343KB"]
SnsService["snsService.ts\n105KB"]
ImageDecrypt["imageDecryptService.ts\n75KB"]
GroupAnalytics["groupAnalyticsService.ts\n74KB"]
AnnualReport["annualReportService.ts\n58KB"]
KeyMac["keyServiceMac.ts\n51KB"]
end
subgraph Feature["功能层 (中大)"]
HttpSvc["httpService.ts\n62KB"]
InsightSvc["insightService.ts\n44KB"]
KeyService["keyService.ts\n39KB"]
Config["config.ts\n30KB"]
DualReport["dualReportService.ts\n30KB"]
AnalyticsSvc["analyticsService.ts\n24KB"]
WcdbService["wcdbService.ts\n24KB"]
VideoSvc["videoService.ts\n25KB"]
VoiceSvc["voiceTranscribeService.ts\n17KB"]
MessagePush["messagePushService.ts\n16KB"]
KeyLinux["keyServiceLinux.ts\n16KB"]
end
subgraph Util["工具与缓存 (小)"]
DbPath[dbPathService]
BizSvc[bizService]
Cloud[cloudControlService]
Wasm[wasmService]
Isaac[isaac64]
Avatar[avatarFileCacheService]
Contact[contactCacheService]
ContactExp[contactExportService]
GMsgCnt[groupMyMessageCountCacheService]
Session[sessionStatsCacheService]
Stats[exportContentStatsCacheService]
Rec[exportRecordService]
ImgPre[imagePreloadService]
MsgCache[messageCacheService]
Linux[linuxNotificationService]
Hello[windowsHelloService]
Styles[exportHtmlStyles]
CardDiag[exportCardDiagnosticsService]
Region[contactRegionLookupData]
end
ChatService --> WcdbCore
ExportService --> ChatService
SnsService --> WcdbCore
ImageDecrypt --> WcdbCore
GroupAnalytics --> ChatService
AnnualReport --> ChatService
DualReport --> ChatService
AnalyticsSvc --> ChatService
HttpSvc --> ChatService
InsightSvc --> ChatService
```
### 6.2 关键服务职责
| 服务 | 核心职责 | 特别说明 |
|------|---------|---------|
| `chatService` | 会话列表、消息读取、媒体定位、实时刷新 | 超大文件,必须用 `codebase_search` / `view_code_item` 定位后再改 |
| `wcdbCore` / `wcdbService` | 基于 `koffi` FFI 调用 WCDB 原生库,解密读取微信 SQLite | 跨平台原生库放在 `resources/wcdb/<platform>/` |
| `keyService*` | 获取微信解密密钥Win/Mac/Linux | Mac 51KB涉及内存扫描Linux 独立实现Win 通过 `windowsHelloService` 辅助 |
| `imageDecryptService` | XOR / AES 解密图片、实况图片 | LRU 缓存 + 批量调度(配合 `batchImageDecryptStore` |
| `videoService` | 视频 md5 解析、ffmpeg 转码、生成封面 | 依赖 `ffmpeg-static` + `asarUnpack` |
| `voiceTranscribeService` | silk→wav→sherpa-onnx ASR | 通过 `transcribeWorker` 异步执行 |
| `snsService` | 朋友圈解密、导出、时间限制突破 | 单独的 `Sns/` 组件族对应 |
| `analyticsService` / `groupAnalyticsService` | 私聊/群聊统计分析、排行、时段分布 | 依赖 `jieba-wasm` 分词 |
| `annualReportService` / `dualReportService` | 年度/双人报告生成 | 将重算任务派给对应 Worker |
| `exportService` + `exportWorker` | 多格式导出,分任务并发 + 进度广播 | HTML 导出样式来自 [exportHtml.css](electron/services/exportHtml.css) |
| `httpService` | 本地 HTTP API 服务(默认 5031 | 详见 [docs/HTTP-API.md](docs/HTTP-API.md) |
| `messagePushService` | 新消息监听 + 通知窗口推送 | 黑白名单、防撤回 |
| `insightService` | 「我的足迹」洞察 / AI 辅助洞察 | 支持 Footprint 生成 |
| `config` | 使用 `electron-store` 的 JSON 配置中台 | 多 wxid 配置、密钥、主题、协议同意等 |
| `cloudControlService` | 开关/页面统计(用户同意后) | 完全本地化的云控模型 |
---
## 7. 核心数据流
### 7.1 启动 & 连接数据库流程
```mermaid
sequenceDiagram
participant U as 用户
participant App as App.tsx
participant Cfg as configService
participant IPC as window.electronAPI
participant Main as main.ts
participant Wcdb as wcdbService + wcdbCore
U->>App: 打开应用
App->>Cfg: getAgreementAccepted
alt 未同意
App-->>U: 显示协议弹窗
else 已同意
App->>Cfg: 读取 dbPath / decryptKey / wxid
alt 配置完整
App->>IPC: chat.connect()
IPC->>Main: ipcMain.handle('chat:connect')
Main->>Wcdb: open(dbPath, hexKey, wxid)
Wcdb-->>Main: 成功/失败
Main-->>IPC: { success }
IPC-->>App: setDbConnected(true)
App->>App: navigate('/home')
else 配置缺失
App->>App: 引导 WelcomePage
end
end
```
### 7.2 聊天消息读取与图片解密
```mermaid
sequenceDiagram
participant UI as ChatPage
participant Store as chatStore
participant IPC as electronAPI.chat
participant Chat as chatService
participant WCore as wcdbCore(FFI)
participant Img as imageDecryptService
participant W as imageSearchWorker
UI->>IPC: listMessages(sessionId, range)
IPC->>Chat: 查询消息
Chat->>WCore: SQL via koffi
WCore-->>Chat: 原始消息
Chat-->>UI: 消息列表(含图片引用)
UI->>Store: 写入消息
UI->>IPC: decryptImage(md5/hash)
IPC->>Img: 解密请求
alt 需要遍历目录
Img->>W: postMessage(search)
W-->>Img: 文件路径
end
Img-->>IPC: 解密后的路径/blob
IPC-->>UI: 图片数据
```
### 7.3 导出任务生命周期
```mermaid
stateDiagram-v2
[*] --> Queued: 用户点击导出
Queued --> Running: exportService 派发
Running --> WorkerRun: exportWorker 执行
WorkerRun --> Progress: 进度广播exportBridge
Progress --> WorkerRun
WorkerRun --> Success: 成功
WorkerRun --> Failed: 异常/取消
Success --> [*]
Failed --> [*]
```
**进度通道**`exportService` → 主进程事件 → `exportBridge.ts` → 渲染层订阅者(`ExportPage`)。
---
## 8. 构建与发布
### 8.1 Vite 多入口
[vite.config.ts](vite.config.ts) 声明 **8 个 entry**
| Entry | 产物 | 说明 |
|-------|------|------|
| `electron/main.ts` | `dist-electron/main.js` | 主进程 |
| `electron/preload.ts` | `dist-electron/preload.js` | 预加载 |
| `electron/annualReportWorker.ts` | `dist-electron/annualReportWorker.js` | 年度报告 worker |
| `electron/dualReportWorker.ts` | `dist-electron/dualReportWorker.js` | 双人报告 worker |
| `electron/imageSearchWorker.ts` | `dist-electron/imageSearchWorker.js` | 图像搜索 worker |
| `electron/wcdbWorker.ts` | `dist-electron/wcdbWorker.js` | WCDB worker |
| `electron/transcribeWorker.ts` | `dist-electron/transcribeWorker.js` | 语音转写 worker |
| `electron/exportWorker.ts` | `dist-electron/exportWorker.js` | 导出 worker |
- `react(), renderer()` 插件处理渲染进程;`inlineDynamicImports: true` 确保 worker 单文件。
- `external``koffi` / `better-sqlite3` / `sherpa-onnx-node` / `exceljs` / `ffmpeg-static` 不打包进 bundleasar 外存放。
### 8.2 打包策略electron-builder
| 平台 | Target | 关键配置 |
|------|--------|---------|
| Windows | `nsis` | `installer.nsh`、多语言安装器、VC++ 运行库随包 |
| macOS | `dmg` + `zip` | `hardenedRuntime: false``entitlements.mac.plist` |
| Linux | `AppImage` + `tar.gz` | 附带 `resources/linux/install.sh` |
| `asarUnpack` | `silk-wasm` / `sherpa-onnx-*` / `ffmpeg-static` | 原生/二进制模块不能进 asar |
| `extraResources` | `resources/**` + `public/icon.*` + `electron/assets/wasm/` | 运行时资源 |
| `publish` | GitHub `Jasonzhu1207/WeFlow` | 配合 `electron-updater` |
### 8.3 CI 流水线(`.github/workflows/`
| 文件 | 作用 |
|------|------|
| `release.yml` | 发布打包 |
| `preview-nightly-main.yml` | Nightly 构建 |
| `dev-daily-fixed.yml` | Dev 日常 |
| `security-scan.yml` | 安全扫描(含 gitleaks |
| `anti-spam.yml` | Issue 反垃圾 |
| `issue-auto-assign.yml` | Issue 自动分派 |
---
## 9. 安全与合规设计
1. **数据本地化**:全部解密、分析、导出均在本地执行,不上传任何聊天内容(协议与隐私弹窗双重同意)。
2. **密钥保护**
- 微信 key 通过平台特定 `keyService*` 动态获取,不落盘;
- 应用锁可选 Windows Hello / 系统凭据;
- `.gitleaks.toml` 扫描源码防止密钥入库。
3. **参数化查询**:所有 SQLite 查询通过 WCDB 参数化接口,避免拼接。
4. **更新通道**:仅从 GitHub Releases 拉取,支持强制更新(`minimumVersion`)。
5. **云控与统计**`cloudControlService` 完全可选,用户可拒绝;默认不采集任何聊天内容。
6. **IPC 白名单**`preload.ts` 通过 `contextBridge` 仅暴露有限 API渲染层无法直接访问 Node。
---
## 10. 性能关键点
| 热点 | 设计 |
|------|------|
| 消息列表(聊天动辄百万级) | `react-virtuoso` 虚拟滚动 + `chatService` 分页 + `LRUCache` |
| 图片解密批量(几千张) | `imageDecryptService` + `imageSearchWorker` + 全局进度 store |
| 年度/双人报告(跨年聚合) | 独立 Worker + 分块流式 + 缓存(`groupMyMessageCountCacheService``sessionStatsCacheService` 等) |
| 导出大体量 HTML/Excel | `exportWorker` + `jszip` 流式写入 + 进度广播 |
| 主题切换 & 样式 | CSS 变量 + `data-theme` / `data-mode` 根属性切换,无重渲染 |
| 启动速度 | `splash.html` 早期显示 + 主题预读 + DB 异步连接 |
---
## 11. 扩展点(二开指南)
| 场景 | 落点 |
|------|------|
| 新增页面 | `src/pages/` 新建 `XxxPage.tsx` + `.scss`,在 [App.tsx](src/App.tsx) 路由和 [Sidebar.tsx](src/components/Sidebar.tsx) 菜单登记 |
| 新增主进程能力 | `electron/services/` 新建 service`main.ts` 添加 `ipcMain.handle('ns:action', ...)`,在 [preload.ts](electron/preload.ts) 暴露 → 渲染层新增对应 `src/services/` 或直接 `window.electronAPI.ns.action()` 调用 |
| 新增 Worker | `electron/xxxWorker.ts` 建立,在 [vite.config.ts](vite.config.ts) 添加 entry |
| 新增导出格式 | 扩展 `exportService` + `exportWorker`UI 在 `src/components/Export/``ExportPage.tsx` 挂接 |
| 新增 HTTP API | `httpService.ts` 注册路由,同步更新 [docs/HTTP-API.md](docs/HTTP-API.md) |
| 新增主题 | `src/stores/themeStore.ts` 的 themes 列表 + `src/styles/` 主题 SCSS |
| 新语言支持 | 当前未接入 i18n需要新增语言时优先引入 `react-i18next`(评估风险) |
---
## 12. 已知风险与技术债
| 风险 | 说明 | 缓解建议 |
|------|------|---------|
| 超大文件 | `ChatPage.tsx` 397KB、`ExportPage.tsx` 402KB、`chatService.ts` 389KB、`exportService.ts` 343KB、`SettingsPage.tsx` 174KB | 新增内容尽量独立成文件;只在必须时才对这些文件进行精细化 diff禁止盲目整读 |
| 原生模块兼容 | `koffi``sherpa-onnx-node``wcdb` 随平台/Electron 版本变化 | 升级前先 `npm run rebuild` 并跑三端冒烟 |
| 配置 migrate | `electron-store` schema 无正式迁移框架 | 新字段默认可 Optional破坏性变更需加版本号判定 |
| 单一 `preload.ts` | 28KBIPC 全部集中 | 保持字段分组;考虑按域拆分 preload 模块(需评估 `contextBridge` 成本) |
| 无自动化测试 | 缺少单元/集成测试 | 新功能按 AGENTS.md "Level 2 TDD" 策略补齐关键分支 |
---
## 13. 快速上手路径(推荐)
1. 通读 [README.md](README.md) + 本架构文档
2. 阅读 [AGENTS.md](AGENTS.md) 了解协作与门禁
3. 顺序浏览:[App.tsx](src/App.tsx) → [Sidebar.tsx](src/components/Sidebar.tsx) → [preload.ts](electron/preload.ts) → [main.ts](electron/main.ts)(搜索 `ipcMain.handle`
4. 按域选择 Service 细读:聊天链路入口从 `chat:connect` / `chat:listSessions` 反查 [chatService.ts](electron/services/chatService.ts)
5. 运行 `npm install && npm run dev` 启动联调;`npm run typecheck` 验证
---
## 14. 参考文档
- [README.md](README.md)
- [docs/HTTP-API.md](docs/HTTP-API.md)
- [AGENTS.md](AGENTS.md)
- [vite.config.ts](vite.config.ts)
- [package.json](package.json)

View File

@@ -1920,10 +1920,11 @@
--contacts-main-col-width: calc(var(--contacts-avatar-col-width) + var(--contacts-column-gap) + var(--contacts-name-text-width));
--contacts-left-sticky-width: calc(var(--contacts-select-col-width) + var(--contacts-main-col-width) + var(--contacts-column-gap));
--contacts-message-col-width: 120px;
--contacts-latest-time-col-width: 150px;
--contacts-media-col-width: 72px;
--contacts-action-col-width: 140px;
--contacts-actions-sticky-width: 240px;
--contacts-table-min-width: 1240px;
--contacts-table-min-width: 1400px;
overflow: hidden;
border: none;
border-radius: 12px;
@@ -2174,6 +2175,59 @@
box-sizing: border-box;
}
.contacts-list-header-latest-time {
width: var(--contacts-latest-time-col-width);
min-width: var(--contacts-latest-time-col-width);
display: flex;
align-items: center;
justify-content: center;
gap: 4px;
text-align: center;
flex-shrink: 0;
white-space: nowrap;
overflow: hidden;
text-overflow: ellipsis;
box-sizing: border-box;
}
// 可排序的表头按钮通用样式
.contacts-list-header-sortable {
background: transparent;
border: none;
padding: 4px 6px;
margin: 0;
color: inherit;
font: inherit;
cursor: pointer;
border-radius: 6px;
gap: 4px;
transition: background-color 0.12s ease, color 0.12s ease;
&:hover {
background: color-mix(in srgb, var(--primary) 10%, transparent);
color: var(--primary);
}
&:focus-visible {
outline: 2px solid var(--primary);
outline-offset: 1px;
}
&.is-active {
color: var(--primary);
}
}
.contacts-list-header-sort-icon {
color: inherit;
flex-shrink: 0;
&.muted {
color: var(--text-tertiary);
opacity: 0.6;
}
}
.contacts-list-header-media {
width: var(--contacts-media-col-width);
min-width: var(--contacts-media-col-width);
@@ -2508,6 +2562,37 @@
box-sizing: border-box;
}
.row-latest-time {
width: var(--contacts-latest-time-col-width);
min-width: var(--contacts-latest-time-col-width);
display: flex;
align-items: center;
justify-content: center;
flex-shrink: 0;
text-align: center;
box-sizing: border-box;
}
.row-latest-time-value {
margin: 0;
font-size: 12px;
line-height: 1.2;
color: var(--text-secondary);
font-variant-numeric: tabular-nums;
display: inline-flex;
align-items: center;
justify-content: center;
min-height: 14px;
white-space: nowrap;
overflow: hidden;
text-overflow: ellipsis;
max-width: 100%;
&.muted {
color: var(--text-tertiary);
}
}
.row-media-metric {
width: var(--contacts-media-col-width);
min-width: var(--contacts-media-col-width);
@@ -5058,6 +5143,7 @@
--contacts-name-text-width: 10em;
--contacts-main-col-width: calc(44px + 10px + var(--contacts-name-text-width));
--contacts-message-col-width: 104px;
--contacts-latest-time-col-width: 128px;
--contacts-media-col-width: 62px;
--contacts-action-col-width: 140px;
}
@@ -5085,6 +5171,10 @@
min-width: var(--contacts-message-col-width);
}
.table-wrap .row-latest-time {
min-width: var(--contacts-latest-time-col-width);
}
.table-wrap .row-media-metric {
min-width: var(--contacts-media-col-width);
}

View File

@@ -4,6 +4,9 @@ import { Virtuoso, type VirtuosoHandle } from 'react-virtuoso'
import { createPortal } from 'react-dom'
import {
Aperture,
ArrowDown,
ArrowUp,
ArrowUpDown,
Calendar,
Check,
CheckSquare,
@@ -638,6 +641,44 @@ const formatYmdHmDateTime = (timestamp?: number): string => {
return `${y}-${m}-${day} ${h}:${min}`
}
// 将秒级时间戳格式化为最新消息时间24h 内显示相对时间,超过则显示 YYYY-MM-DD HH:mm。
// 返回 { text, title } 以便上层 title 属性展示完整绝对时间。
const formatLatestMessageTimeFromSeconds = (
timestamp?: number,
now: number = Date.now()
): { text: string; title: string } => {
if (!timestamp || !Number.isFinite(timestamp) || timestamp <= 0) {
return { text: '--', title: '' }
}
const ms = timestamp * 1000
const absolute = formatYmdHmDateTime(ms)
const diff = Math.max(0, now - ms)
const minute = 60 * 1000
const hour = 60 * minute
const day = 24 * hour
if (diff < minute) {
return { text: '刚刚', title: absolute }
}
if (diff < hour) {
const minutes = Math.max(1, Math.floor(diff / minute))
return { text: `${minutes} 分钟前`, title: absolute }
}
if (diff < day) {
const hours = Math.max(1, Math.floor(diff / hour))
return { text: `${hours} 小时前`, title: absolute }
}
return { text: absolute, title: absolute }
}
// 导出列表支持的排序维度。
type ContactsSortKey = 'messageCount' | 'latestMessageTime'
type ContactsSortOrder = 'desc' | 'asc'
interface ContactsSortConfig {
key: ContactsSortKey | null
order: ContactsSortOrder | null
}
const DEFAULT_CONTACTS_SORT_CONFIG: ContactsSortConfig = { key: null, order: null }
const isSingleContactSession = (sessionId: string): boolean => {
const normalized = String(sessionId || '').trim()
if (!normalized) return false
@@ -2184,6 +2225,21 @@ function ExportPage() {
const [sessionMutualFriendsDialogTarget, setSessionMutualFriendsDialogTarget] = useState<SessionSnsTimelineTarget | null>(null)
const [sessionMutualFriendsSearch, setSessionMutualFriendsSearch] = useState('')
const [backgroundTasks, setBackgroundTasks] = useState<BackgroundTaskRecord[]>([])
// 会话列表排序状态key=null 时回退默认顺序(按消息数降序)。
const [contactsSortConfig, setContactsSortConfig] = useState<ContactsSortConfig>(DEFAULT_CONTACTS_SORT_CONFIG)
const toggleContactsSort = useCallback((key: ContactsSortKey) => {
setContactsSortConfig(prev => {
if (prev.key !== key) {
// 切换到新列:从降序开始。
return { key, order: 'desc' }
}
// 同列循环desc -> asc -> null (默认)
if (prev.order === 'desc') return { key, order: 'asc' }
if (prev.order === 'asc') return DEFAULT_CONTACTS_SORT_CONFIG
return { key, order: 'desc' }
})
}, [])
const [exportFolder, setExportFolder] = useState('')
const [writeLayout, setWriteLayout] = useState<configService.ExportWriteLayout>('B')
@@ -6456,34 +6512,54 @@ function ExportPage() {
)
})
const indexedContacts = contacts.map((contact, index) => ({
contact,
index,
count: (() => {
const counted = normalizeMessageCount(sessionMessageCounts[contact.username])
if (typeof counted === 'number') return counted
const hinted = normalizeMessageCount(sessionRowByUsername.get(contact.username)?.messageCountHint)
return hinted
})()
}))
const indexedContacts = contacts.map((contact, index) => {
const sessionRow = sessionRowByUsername.get(contact.username)
const counted = normalizeMessageCount(sessionMessageCounts[contact.username])
const hinted = normalizeMessageCount(sessionRow?.messageCountHint)
const count = typeof counted === 'number' ? counted : hinted
// 最新消息时间:优先使用 sessionContentMetrics 中的 lastTimestamp更精确
// 其次使用 SessionRow 的 sortTimestamp/lastTimestamp通讯录加载即有
const metricTs = sessionContentMetrics[contact.username]?.lastTimestamp
const rowTs = sessionRow?.sortTimestamp || sessionRow?.lastTimestamp
const latestTime = (typeof metricTs === 'number' && metricTs > 0)
? metricTs
: (typeof rowTs === 'number' && rowTs > 0 ? rowTs : undefined)
return { contact, index, count, latestTime }
})
// 比较器:空值稳定排在末尾;相等时按原始下标稳定兜底。
const compareNullable = (a: number | undefined, b: number | undefined, order: ContactsSortOrder): number => {
const aHas = typeof a === 'number' && Number.isFinite(a)
const bHas = typeof b === 'number' && Number.isFinite(b)
if (aHas && bHas) {
const diff = (a as number) - (b as number)
return order === 'desc' ? -diff : diff
}
if (aHas) return -1
if (bHas) return 1
return 0
}
const sortKey = contactsSortConfig.key
const sortOrder = contactsSortConfig.order ?? 'desc'
indexedContacts.sort((a, b) => {
const aHasCount = typeof a.count === 'number'
const bHasCount = typeof b.count === 'number'
if (aHasCount && bHasCount) {
const diff = (b.count as number) - (a.count as number)
if (sortKey === 'latestMessageTime') {
const diff = compareNullable(a.latestTime, b.latestTime, sortOrder)
if (diff !== 0) return diff
} else if (sortKey === 'messageCount') {
const diff = compareNullable(a.count, b.count, sortOrder)
if (diff !== 0) return diff
} else {
// 默认key===null保持旧有按消息数降序的行为避免改变现有视觉顺序。
const diff = compareNullable(a.count, b.count, 'desc')
if (diff !== 0) return diff
} else if (aHasCount) {
return -1
} else if (bHasCount) {
return 1
}
// 无统计值或同分时保持原顺序,避免列表频繁跳动。
return a.index - b.index
})
return indexedContacts.map(item => item.contact)
}, [contactsList, activeTab, searchKeyword, sessionMessageCounts, sessionRowByUsername])
}, [contactsList, activeTab, searchKeyword, sessionMessageCounts, sessionRowByUsername, sessionContentMetrics, contactsSortConfig])
const keywordMatchedContactUsernameSet = useMemo(() => {
const keyword = searchKeyword.trim().toLowerCase()
@@ -6692,7 +6768,7 @@ function ExportPage() {
useEffect(() => {
contactsVirtuosoRef.current?.scrollToIndex({ index: 0, align: 'start' })
setIsContactsListAtTop(true)
}, [activeTab, searchKeyword])
}, [activeTab, searchKeyword, contactsSortConfig])
const collectVisibleSessionMetricTargets = useCallback((sourceContacts: ContactInfo[]): string[] => {
if (sourceContacts.length === 0) return []
@@ -8075,6 +8151,19 @@ function ExportPage() {
const hintedMessages = normalizeMessageCount(matchedSession?.messageCountHint)
const displayedMessageCount = countedMessages ?? hintedMessages
const mediaMetric = sessionContentMetrics[contact.username]
// 最新消息时间:优先取 metric.lastTimestamp最精确退回到 SessionRow 的 sortTimestamp/lastTimestamp。
const metricLatestTs = mediaMetric?.lastTimestamp
const rowLatestTs = matchedSession?.sortTimestamp || matchedSession?.lastTimestamp
const resolvedLatestTs = (typeof metricLatestTs === 'number' && metricLatestTs > 0)
? metricLatestTs
: (typeof rowLatestTs === 'number' && rowLatestTs > 0 ? rowLatestTs : undefined)
const latestTimeInfo = formatLatestMessageTimeFromSeconds(resolvedLatestTs, nowTick)
const latestTimeState: { state: 'value'; text: string; title: string } | { state: 'loading' } | { state: 'na'; text: '--' } =
!canExport
? (isSessionBindingPending ? { state: 'loading' } : { state: 'na', text: '--' })
: (typeof resolvedLatestTs === 'number' && resolvedLatestTs > 0
? { state: 'value', text: latestTimeInfo.text, title: latestTimeInfo.title }
: { state: 'na', text: '--' })
const messageCountState: { state: 'value'; text: string } | { state: 'loading' } | { state: 'na'; text: '--' } =
!canExport
? (isSessionBindingPending ? { state: 'loading' } : { state: 'na', text: '--' })
@@ -8190,6 +8279,18 @@ function ExportPage() {
</button>
)}
</div>
<div className="row-latest-time">
{latestTimeState.state === 'loading'
? <Loader2 size={12} className="spin row-media-metric-icon" aria-label="最新消息时间加载中" />
: (
<span
className={`row-latest-time-value ${latestTimeState.state === 'value' ? '' : 'muted'}`}
title={latestTimeState.state === 'value' ? latestTimeState.title : undefined}
>
{latestTimeState.text}
</span>
)}
</div>
<div className="row-media-metric">
<strong className="row-media-metric-value">
{emojiMetric.state === 'loading'
@@ -9135,7 +9236,46 @@ function ExportPage() {
<span className="contacts-list-header-main-label">{contactsHeaderMainLabel}</span>
</span>
</span>
<span className="contacts-list-header-count"></span>
<button
type="button"
className={`contacts-list-header-count contacts-list-header-sortable ${contactsSortConfig.key === 'messageCount' ? 'is-active' : ''}`}
onClick={() => toggleContactsSort('messageCount')}
title={
contactsSortConfig.key !== 'messageCount'
? '按总消息数降序排列'
: contactsSortConfig.order === 'desc'
? '切换为按总消息数升序'
: '取消排序(恢复默认)'
}
>
<span></span>
{contactsSortConfig.key === 'messageCount'
? (contactsSortConfig.order === 'asc'
? <ArrowUp size={12} className="contacts-list-header-sort-icon" />
: <ArrowDown size={12} className="contacts-list-header-sort-icon" />)
: <ArrowUpDown size={12} className="contacts-list-header-sort-icon muted" />
}
</button>
<button
type="button"
className={`contacts-list-header-latest-time contacts-list-header-sortable ${contactsSortConfig.key === 'latestMessageTime' ? 'is-active' : ''}`}
onClick={() => toggleContactsSort('latestMessageTime')}
title={
contactsSortConfig.key !== 'latestMessageTime'
? '按最新消息时间降序排列'
: contactsSortConfig.order === 'desc'
? '切换为按最新消息时间升序'
: '取消排序(恢复默认)'
}
>
<span></span>
{contactsSortConfig.key === 'latestMessageTime'
? (contactsSortConfig.order === 'asc'
? <ArrowUp size={12} className="contacts-list-header-sort-icon" />
: <ArrowDown size={12} className="contacts-list-header-sort-icon" />)
: <ArrowUpDown size={12} className="contacts-list-header-sort-icon muted" />
}
</button>
<span className="contacts-list-header-media"></span>
<span className="contacts-list-header-media"></span>
<span className="contacts-list-header-media"></span>