Files
lobehub/apps/desktop/Development.md
Innei dc6d5cf489 📝 docs(desktop): update Development.md to reflect current project structure [skip ci] (#12581)
📝 docs(desktop): update Development.md to reflect current project structure

Made-with: Cursor
2026-03-02 15:36:04 +08:00

20 KiB
Raw Blame History

核心框架组件目录架构

主进程核心组件

apps/desktop/src/main/
├── core/                     // 核心功能
│   ├── App.ts                // 应用核心类,整合所有管理器
│   ├── browser/              // 浏览器窗口相关
│   │   ├── Browser.ts            // 浏览器窗口类
│   │   ├── BrowserManager.ts     // 浏览器窗口管理
│   │   ├── WindowStateManager.ts // 窗口状态管理
│   │   └── WindowThemeManager.ts // 窗口主题管理
│   ├── ui/                   // UI 相关管理
│   │   ├── MenuManager.ts        // 菜单管理核心类,负责选择和协调平台实现
│   │   ├── ShortcutManager.ts    // 快捷键管理
│   │   ├── TrayManager.ts        // 系统托盘管理
│   │   └── Tray.ts               // 托盘实例
│   └── infrastructure/       // 基础设施层
│       ├── IoCContainer.ts       // 依赖注入容器
│       ├── StoreManager.ts       // 存储管理
│       ├── I18nManager.ts        // 国际化管理
│       ├── UpdaterManager.ts     // 更新管理
│       ├── ProtocolManager.ts   // 协议处理管理(基类)
│       ├── RendererUrlManager.ts // 渲染进程 URL 管理
│       ├── RendererProtocolManager.ts    // 渲染进程协议管理
│       ├── BackendProxyProtocolManager.ts // 后端代理协议管理
│       ├── StaticFileServerManager.ts    // 静态文件服务管理
│       └── ToolDetectorManager.ts        // 工具检测器管理
├── controllers/              // 控制器层,处理渲染进程调用
│   ├── AuthCtr.ts            // 认证控制器
│   ├── BrowserWindowsCtr.ts  // 浏览器窗口控制器
│   ├── DevtoolsCtr.ts        // 开发工具控制器
│   ├── LocalFileCtr.ts       // 本地文件控制器
│   ├── McpCtr.ts             // MCP 控制器
│   ├── McpInstallCtr.ts      // MCP 安装控制器
│   ├── MenuCtr.ts            // 菜单控制器
│   ├── NetworkProxyCtr.ts    // 网络代理控制器
│   ├── NotificationCtr.ts   // 通知控制器
│   ├── RemoteServerConfigCtr.ts // 远程服务器配置控制器
│   ├── RemoteServerSyncCtr.ts   // 远程服务器同步控制器
│   ├── ShellCommandCtr.ts    // Shell 命令控制器
│   ├── ShortcutCtr.ts        // 快捷键控制器
│   ├── SystemCtr.ts          // 系统控制器
│   ├── ToolDetectorCtr.ts    // 工具检测器控制器
│   ├── TrayMenuCtr.ts        // 托盘菜单控制器
│   ├── UpdaterCtr.ts         // 更新控制器
│   ├── UploadFileCtr.ts      // 文件上传控制器
│   ├── registry.ts           // 控制器注册表
│   └── index.ts              // 控制器导出
├── services/                 // 服务层
│   ├── fileSearchSrv.ts      // 文件搜索服务
│   ├── contentSearchSrv.ts   // 内容搜索服务
│   ├── fileSrv.ts            // 文件服务
│   └── index.ts              // 服务导出
├── modules/                  // 功能模块
│   ├── fileSearch/           // 文件搜索模块Spotlight/mdfind/fd 等)
│   ├── contentSearch/        // 内容搜索模块
│   ├── networkProxy/         // 网络代理模块
│   ├── toolDetectors/        // 工具检测器fileSearch、contentSearch、astSearch
│   └── updater/              // 更新模块
├── menus/                    // 菜单实现目录
│   ├── index.ts              // 导出平台实现和接口
│   ├── types.ts              // 定义菜单平台接口 IMenuPlatform
│   └── impls/                // 平台特定实现目录
│       ├── BaseMenuPlatform.ts // 基础平台类注入App
│       ├── macOS.ts             // macOS 充血模型实现
│       ├── windows.ts           // Windows 充血模型实现
│       └── linux.ts             // Linux 充血模型实现
├── shortcuts/                // 快捷键实现
│   ├── config.ts             // 快捷键配置
│   └── index.ts              // 快捷键导出
├── utils/                    // 工具函数
│   ├── file-system.ts        // 文件系统工具
│   ├── logger.ts             // 日志工具
│   ├── path.ts               // 路径工具
│   ├── permissions.ts        // 权限工具
│   ├── mime.ts               // MIME 类型工具
│   ├── http-headers.ts       // HTTP 头工具
│   ├── protocol.ts           // 协议工具
│   ├── ipc/                  // IPC 工具
│   │   ├── base.ts           // IPC 基类
│   │   ├── index.ts          // IPC 导出
│   │   └── utility.ts        // IPC 工具函数
│   └── ...
├── libs/                     // 主进程第三方库封装
│   └── mcp/                  // MCP 客户端
│       ├── types.ts          // MCP 类型定义
│       └── client.ts         // MCP 客户端实现
├── types/                    // 类型定义
│   ├── store.ts              // 存储类型
│   └── protocol.ts           // 协议类型
├── const/                    // 常量定义
│   ├── dir.ts                // 目录常量
│   ├── env.ts                // 环境常量
│   ├── store.ts              // 存储常量
│   ├── protocol.ts           // 协议常量
│   └── theme.ts              // 主题常量
├── locales/                  // 国际化资源
│   ├── index.ts              // 导出 i18n 相关功能
│   ├── resources.ts          // 资源加载逻辑
│   └── default/              // 默认中文翻译源文件
│       ├── index.ts          // 导出所有翻译
│       ├── menu.ts           // 菜单翻译
│       ├── dialog.ts         // 对话框翻译
│       └── common.ts         // 通用翻译
├── appBrowsers.ts            // 应用浏览器窗口配置
├── exports.ts                // 主进程导出
└── index.ts                  // 主进程入口文件

注意:文件搜索类型 (FileResult, SearchOptions 等) 已移至 modules/fileSearch/types.ts

预加载脚本

apps/desktop/src/preload/
├── index.ts          // 预加载脚本入口,初始化 electronApi 和路由拦截
├── electronApi.ts    // Electron API 暴露到渲染进程
├── invoke.ts         // IPC 调用封装
├── routeInterceptor.ts // 路由拦截逻辑(如 /settings 打开设置窗口)
└── streamer.ts       // 流式数据传输

共享代码

apps/desktop/src/common/
└── routes.ts         // 路由拦截配置类型 (RouteInterceptConfig)

功能模块实现

菜单实现框架

apps/desktop/src/main/
├── core/
│   ├── App.ts                // 应用核心类
│   ├── browser/BrowserManager.ts  // 浏览器窗口管理
│   └── ui/MenuManager.ts     // 菜单管理核心类,负责选择和协调平台实现
├── menus/                    // 菜单实现目录
│   ├── index.ts              // 导出平台实现和接口
│   ├── types.ts              // 定义菜单平台接口 IMenuPlatform
│   └── impls/                // 平台特定实现目录
│       ├── BaseMenuPlatform.ts // 基础平台类注入App
│       ├── macOS.ts            // macOS 充血模型实现
│       ├── windows.ts          // Windows 充血模型实现
│       └── linux.ts            // Linux 充血模型实现
├── controllers/
│   └── MenuCtr.ts            // 菜单控制器,处理渲染进程调用

国际化 (i18n) 实现

apps/desktop/src/main/
├── core/
│   ├── infrastructure/I18nManager.ts  // i18n 管理器
│   └── App.ts                // 应用主类,集成 i18n
├── locales/
│   ├── index.ts              // 导出 i18n 相关功能
│   ├── resources.ts          // 资源加载逻辑
│   └── default/              // 默认中文翻译源文件
│       ├── index.ts          // 导出所有翻译
│       ├── menu.ts           // 菜单翻译
│       ├── dialog.ts         // 对话框翻译
│       └── common.ts         // 通用翻译

主进程 i18n 国际化管理使用方式:

  1. 直接导入 i18nManager 实例:

    import i18nManager from '@/locales';
    
  2. 使用翻译函数:

    import { t } from '@/locales';
    
    const translated = t('key');
    
  3. 添加新翻译: 在 locales/default/ 目录下添加翻译源文件

核心模块详细说明

认证模块 (Auth)

认证模块负责处理用户身份验证和授权流程,主要包括:

  1. AuthCtr 控制器:实现 OAuth 授权流程
    • 请求授权:打开浏览器进行 OAuth 认证
    • 处理回调:接收授权码并交换访问令牌
    • 令牌刷新:自动刷新过期的访问令牌
    • 事件广播:向渲染进程通知授权状态变化
import { ControllerModule, IpcMethod } from '@/controllers';

export default class AuthCtr extends ControllerModule {
  static override groupName = 'auth';

  @IpcMethod()
  async requestAuthorization(config: DataSyncConfig) {
    this.authRequestState = crypto.randomBytes(16).toString('hex');

    const authUrl = new URL('/oidc/auth', remoteUrl);
    authUrl.search = querystring.stringify({
      client_id: 'lobe-chat',
      redirect_uri: `${protocolPrefix}://auth/callback`,
      response_type: 'code',
      scope: 'openid profile',
      state: this.authRequestState,
    });

    await shell.openExternal(authUrl.toString());
  }
}
  1. 桌面端特定认证
    • 在桌面应用中使用固定的用户 ID
    • 支持与 Better Auth 认证系统集成

存储模块 (Store)

存储模块使用 electron-store 实现持久化数据存储:

  1. StoreManager 类
    • 提供统一的存储接口
    • 支持类型安全的存取操作
    • 管理应用配置和用户数据
// 存储管理器使用示例
export class StoreManager {
  private store: Store<ElectronMainStore>;

  // 获取配置项
  get<K extends StoreKey>(key: K, defaultValue?: ElectronMainStore[K]): ElectronMainStore[K] {
    return this.store.get(key, defaultValue as any);
  }

  // 设置配置项
  set<K extends StoreKey>(key: K, value: ElectronMainStore[K]): void {
    this.store.set(key, value);
  }

  // 删除配置项
  delete(key: StoreKey): void {
    this.store.delete(key);
  }
}
  1. 存储用途
    • 窗口状态保存
    • 用户偏好设置
    • 认证令牌存储
    • 快捷键配置
    • 国际化语言设置

快捷键模块 (Shortcuts)

快捷键模块管理全局键盘快捷键:

  1. ShortcutManager 类
    • 注册和管理全局快捷键
    • 支持自定义快捷键配置
    • 提供快捷键状态查询
// 快捷键管理器示例
export class ShortcutManager {
  private shortcuts: Map<string, () => void> = new Map();
  private shortcutsConfig: Record<string, string> = {};

  // 注册快捷键
  registerShortcut(accelerator: string, callback: () => void): boolean {
    const success = globalShortcut.register(accelerator, callback);
    if (success) {
      this.shortcuts.set(accelerator, callback);
    }
    return success;
  }

  // 更新快捷键配置
  updateShortcutConfig(id: string, accelerator: string): boolean {
    this.shortcutsConfig[id] = accelerator;
    this.saveShortcutsConfig();
    this.registerConfiguredShortcuts();
    return true;
  }
}
  1. 快捷键装饰器
    • 使用 @shortcut 装饰器简化快捷键注册
    • 通过 IoC 容器管理快捷键映射

控制框架 (Control Framework)

控制框架实现了主进程和渲染进程之间的通信:

  1. ControllerModule 基类
    • 所有控制器的基础类
    • 提供生命周期钩子 (beforeAppReady, afterAppReady)
    • 注入 App 实例
import { ControllerModule, IpcMethod } from '@/controllers';

export class ControllerModule implements IControllerModule {
  constructor(public app: App) {
    this.app = app;
  }
}

export class BrowserWindowsCtr extends ControllerModule {
  static override readonly groupName = 'windows'; // must be readonly

  @IpcMethod()
  openSettingsWindow(params?: OpenSettingsWindowOptions) {
    // ...
  }
}
  1. IoC 容器

    • 依赖注入容器 (core/infrastructure/IoCContainer.ts) 管理控制器实例
    • 注册和解析 IPC 事件处理程序
    • 管理快捷键和控制器方法的映射
  2. 控制器注册:新控制器需加入 controllers/registry.tscontrollerIpcConstructors 数组

服务逻辑 (Service Logic)

服务层提供业务逻辑实现:

  1. ServiceModule 基类
    • 所有服务的基础类
    • 注入 App 实例
    • 提供业务逻辑封装
// 服务模块基类
export class ServiceModule {
  constructor(public app: App) {
    this.app = app;
  }
}
  1. 服务实现
    • fileSearchSrv文件搜索服务
    • contentSearchSrv内容搜索服务
    • fileSrv文件操作服务

数据存储 (Electron Settings)

Electron Settings 基于 electron-store 实现,提供类型安全的数据存储:

  1. 存储配置
    • 使用 JSON 文件存储配置
    • 支持默认值设置
    • 自动创建存储目录
// 存储初始化
this.store = new Store<ElectronMainStore>({
  defaults: STORE_DEFAULTS,
  name: STORE_NAME,
});

// 确保存储目录存在
const storagePath = this.store.get('storagePath');
makeSureDirExist(storagePath);
  1. 存储操作
    • 类型安全的 get/set 方法
    • 支持删除和清除操作
    • 提供存储编辑器功能

主进程和渲染进程通信 (Main-Renderer Communication)

主进程和渲染进程通信基于 Electron IPC 机制:

  1. IPC 事件处理
    • 使用装饰器注册 IPC 事件处理程序
    • 支持客户端事件和服务器事件
    • 自动映射控制器方法到 IPC 事件
import { ensureElectronIpc } from '@/utils/electron/ipc';

// 渲染进程中使用 type-safe proxy 调用主进程方法
const ipc = ensureElectronIpc();

await ipc.localSystem.readLocalFile({ path });
await ipc.system.updateLocale('en-US');
  1. 事件广播
    • 主进程向渲染进程广播事件
    • 支持向所有窗口或特定窗口发送消息

日志系统 (Logging)

日志系统提供统一的日志记录接口:

  1. 日志工具
    • 基于 debug 和 electron-log 实现
    • 支持不同日志级别 (debug, info, warn, error, verbose)
    • 根据环境自动调整日志行为
// 创建日志记录器
export const createLogger = (namespace: string) => {
  const debugLogger = debug(namespace);

  return {
    debug: (message, ...args) => {
      debugLogger(message, ...args);
    },
    error: (message, ...args) => {
      if (process.env.NODE_ENV === 'production') {
        electronLog.error(message, ...args);
      }
      debugLogger(`ERROR: ${message}`, ...args);
    },
    // 其他日志级别...
  };
};
  1. 日志配置
    • 开发环境显示详细日志
    • 生产环境记录到文件
    • 支持命名空间隔离日志

自动更新 (Auto Updates)

自动更新模块基于 electron-updater 实现:

  1. UpdaterManager 类
    • 检查更新
    • 下载更新
    • 安装更新
    • 支持立即安装或下次启动安装
// 更新管理器示例
export class UpdaterManager {
  // 检查更新
  public checkForUpdates = async ({ manual = false }: { manual?: boolean } = {}) => {
    if (this.checking || this.downloading) return;

    this.checking = true;
    this.isManualCheck = manual;

    try {
      await autoUpdater.checkForUpdates();
    } catch (error) {
      logger.error('Error checking for updates:', error.message);
    } finally {
      this.checking = false;
    }
  };

  // 下载更新
  public downloadUpdate = async (manual: boolean = false) => {
    if (this.downloading || !this.updateAvailable) return;

    this.downloading = true;

    try {
      await autoUpdater.downloadUpdate();
    } catch (error) {
      this.downloading = false;
      logger.error('Error downloading update:', error);
    }
  };
}
  1. 更新配置
    • 支持多渠道发布 (stable, beta, nightly)
    • 自动检查更新
    • 更新事件通知

桌面端架构图

┌─────────────────────────────────────────────────────────────────┐
│                      Electron Application                        │
├─────────────────────────────────────────────────────────────────┤
│                                                                 │
│  ┌─────────────────┐           ┌──────────────────────────┐    │
│  │   Main Process  │           │     Renderer Process     │    │
│  │                 │           │                          │    │
│  │  ┌─────────────┐│           │  ┌────────────────────┐  │    │
│  │  │    Core     ││           │  │                    │  │    │
│  │  │  Managers   ││           │  │   Vite SPA (React)  │  │    │
│  │        │        │           │  │                    │  │    │
│  │  ┌─────▼─────┐  │           │  │                    │  │    │
│  │  │Controllers│  │◄──────────┼──┤                    │  │    │
│  │  └─────┬─────┘  │  IPC      │  └────────────────────┘  │    │
│  │        │        │Communication                          │    │
│  │  ┌─────▼─────┐  │           │                          │    │
│  │  │ Services  │  │           │                          │    │
│  │  └─────┬─────┘  │           │                          │    │
│  │        │        │           │                          │    │
│  │  ┌─────▼─────┐  │           │                          │    │
│  │  │  Modules  │  │           │                          │    │
│  │  └───────────┘  │           │                          │    │
│  │                 │           │                          │    │
│  └─────────────────┘           └──────────────────────────┘    │
│                                                                 │
│                     ┌───────────────────┐                       │
│                     │   Preload Script  │                       │
│                     │  (Bridge between  │                       │
│                     │  Main & Renderer) │                       │
│                     └───────────────────┘                       │
│                                                                 │
└─────────────────────────────────────────────────────────────────┘