DeepSeek+Electron35+Vite6+Markdown实战桌面端AI流式聊天模板

1,196 阅读3分钟

随着deepseek-v3的持续火爆,2025年各类基于deepseek技术的应用应运而生。最近基于Electron35跨平台框架整合DeepSeek搭建了一款桌面客户端AI对话助手。

360截图20250416003909828.png

electron35-deepseek实现了流式打字输出效果,支持暗黑+亮色模式。

p2.gif

运用技术

  • 编辑器:VScode
  • 技术框架:vite^6.2.0+vue^3.5.13+vue-router^4.5.0
  • 大模型ai框架:DeepSeek-V3-0324 + OpenAI
  • 跨平台框架:electron^35.1.2
  • 组件库:arco-design^2.57.0 (字节桌面端组件库)
  • 状态插件:pinia^3.0.1
  • 会话缓存:pinia-plugin-persistedstate^4.2.0
  • 高亮插件:highlight.js^11.11.1
  • markdown解析:markdown-it

p6.gif

p6-2.gif

项目框架结构

基于最新跨平台框架Electron35+Vite6构建工具搭建项目模板,整合最新DeepSeek-V3 api聊天大模型。

360截图20250416003909810.png

001360截图20250415215507389.png

001360截图20250415215720964.png

002360截图20250415220924877.png

002360截图20250415221559092.png

003360截图20250415224408704.png

005360截图20250415225913344.png

016360截图20250416000647920.png

018360截图20250416001201455.png

环境变量.env配置

需要去deepseek网站申请一个API Key,替换掉.env文件里的Key即可畅快体验ai对话功能。

# 项目名称
VITE_APPNAME = 'Electron35-DeepSeek'

# 运行端口
VITE_PORT = 3003

# 是否加载调试工具devtools
VITE_DEVTOOLS = true

# 是否打开调试工具devtools
VITE_OPEN_DEVTOOLS = true

# 是否删除生产环境console
VITE_DROP_CONSOLE = true

# DeepSeek API配置
VITE_DEEPSEEK_API_KEY = 替换为你的 API Key
VITE_DEEPSEEK_BASE_URL = https://api.deepseek.com

入口配置main.js

import { createApp } from 'vue'
import './style.scss'
import App from './App.vue'

import { launchApp } from '@/windows/actions'

// 引入路由/状态配置
import Router from './router'
import Pinia from './pinia'

// 引入插件配置
import Plugins from './plugins'

launchApp().then(config => {
  if(config) {// 全局存储窗口配置
    window.config = config
  }

  // 初始化app实例
  createApp(App)
  .use(Router)
  .use(Pinia)
  .use(Plugins)
  .mount('#app')
})

5472b55547237b52c7f60e9959f6d283_1289798-20250417113444378-1433436858.png

项目模板布局

62e11451ea556e5c40e4016b47a42a94_1289798-20250417113959089-1884145640.png

整体分为顶部自定义导航条+侧边栏+右侧对话区三个模块。

7cebf11f836e3f0f72eed833c8b2deab_1289798-20250417114129399-1818820098.png

<script setup>
  import Titlebar from '@/layouts/components/titlebar/index.vue'
  import Sidebar from '@/layouts/components/sidebar/index.vue'
</script>

<template>
  <div class="vu__container">
    <div class="vu__layout flexbox flex-col">
      <!-- 导航栏 -->
      <Titlebar />

      <div class="vu__layout-body flex1 flexbox">
        <!-- 侧边 -->
        <Sidebar />

        <!-- 对话区 -->
        <div class="vu__layout-main flex1">
          <router-view v-slot="{ Component, route }">
            <keep-alive>
              <component :is="Component" :key="route.path" />
            </keep-alive>
          </router-view>
        </div>
      </div>
    </div>
  </div>
</template>

并且侧边栏在历史对话超过一定高度会出现滚动条。

7167e3ec680bf32c6f63b832652d6210_1289798-20250417114904734-2086862290.png

9b4a871fce151925148906abc14d55f5_1289798-20250417114836713-1122021252.png

路由配置

a2a0c172ab384344461cb8798756dd06_1289798-20250417115250956-319960391.png

import { createRouter, createWebHashHistory } from 'vue-router'
import { authState } from '@/pinia/modules/auth'

import Layout from '@/layouts/index.vue'

import { loginWindow } from '@/windows/actions'

// 批量导入路由
const modules = import.meta.glob('./modules/*.js', { eager: true })
const patchRouters = Object.keys(modules).map(key => modules[key].default).flat()

/**
 * meta配置
 * @param meta.requireAuth 需登录验证页面
 */
const routes = [
  ...patchRouters,
  // 错误模块
  {
    path: '/:pathMatch(.*)*',
    redirect: '/404',
    component: Layout,
    meta: {
      title: '404 error',
    },
    children: [
      {
        path: '404',
        component: () => import('@/views/error/404.vue'),
      }
    ]
  },
]

const router = createRouter({
  history: createWebHashHistory(),
  routes,
})

// 全局路由钩子拦截
router.beforeEach((to, from) => {
  const authstate = authState()
  // 登录验证
  if(to?.meta?.requireAuth && !authstate.authorization) {
    // console.log('你还未登录!')
    loginWindow()
  }
})

router.afterEach(() => {
  // ...
})

router.onError(error => {
  console.warn('[Router Error]', error)
})

export default router

360截图20250416010303083.png

009360截图20250415233814362.png

010360截图20250415234222930.png

010360截图20250415234222933.png

011360截图20250415234925538.png

013360截图20250415235327849.png

013360截图20250415235535362.png

014360截图20250415235809250.png

ai对话编辑框

612605ca7fc4781b3ada2b5f6248dfbe_1289798-20250417120714426-1703055759.png

4423663a543d22f97ee49c1481815617_1289798-20250417121055466-430124469.png

如上图:编辑框封装成一个公共组件。

d326d8476c8d076d6f0782e2b4b3c746_1289798-20250417121519001-1651496316.png

<template>
  <div class="v3ai__footbar flexbox flex-col">
    <!-- 技能栏 -->
    <div v-if="skillbar" class="v3ai__skills flexbox flex-alignc">
      ...
    </div>
    <!-- 编辑栏 -->
    <div class="v3ai__inputbox flexbox flex-col">
      <div class="v3ai__editor flexbox">
        ...
      </div>
      <!-- 操作栏 -->
      <div class="v3ai__tools flexbox flex-alignc">
        <div class="option flex1 flexbox">
          <div class="btn" :class="{'active': isDeep}" @click="isDeep =! isDeep"><i class="iconfont ai-deepthink"></i> 深度思考 <span class="fs-12">(R1)</span></div>
          <div class="btn" :class="{'active': isNetwork}" @click="isNetwork =! isNetwork"><i class="iconfont ai-network"></i> 联网</div>
        </div>
        <a-dropdown trigger="hover" :show-arrow="false" position="lb" :content-style="{'min-width': '250px'}">
          <a-button shape="circle"><icon-attachment size="18" /></a-button>
          <template #content>
            <a-dgroup>
              <template #title><div style="margin-bottom: 5px;">从网盘添加</div></template>
              <a-doption value="wx"><icon-more /> 选择网盘文件</a-doption>
            </a-dgroup>
            <a-dgroup>
              <template #title><div style="margin-bottom: 5px;">从本地添加</div></template>
              <a-doption value="wx"><icon-apps /> 上传文件</a-doption>
              <a-dsubmenu trigger="hover" position="rb" :popup-translate="[8, 8]" value="option-1" :content-style="{'min-width': '125px'}">
                <template #default><icon-apps /> 上传代码</template>
                <template #content>
                  <a-doption value="pyq"><icon-apps /> 代码文件</a-doption>
                  <a-doption value="qq"><icon-apps /> 代码文件夹</a-doption>
                  <a-doption value="qq"><icon-apps /> GitHub仓库</a-doption>
                </template>
              </a-dsubmenu>
            </a-dgroup>
          </template>
        </a-dropdown>
        <a-tooltip content="截图提问" position="top" mini>
          <a-button shape="circle"><icon-scissor size="18" @click="handleCut" /></a-button>
        </a-tooltip>
        <a-dropdown :show-arrow="false" position="top" :popup-translate="[-10, -5]" :content-style="{'min-width': '160px'}">
          <a-button shape="circle"><icon-plus size="18" /></a-button>
          <template #content>
            <a-doption value="image"><icon-file-image /> 图片</a-doption>
            <a-doption value="file"><icon-file /> 本地文件</a-doption>
            <a-doption value="pdf"><icon-file-pdf /> PDF文档分析</a-doption>
            <a-doption value="page"><icon-cloud /> 网页总结</a-doption>
          </template>
        </a-dropdown>
        <a-divider direction="vertical" style="margin: 0 7px;" />
        <a-button class="submit" type="primary" shape="circle" :disabled="!editorText?.trim() || sessionstate.loading" @click="handleSubmit">
          <icon-arrow-up v-if="!sessionstate.loading" size="20" />
          <icon-loading v-else size="18" />
        </a-button>
      </div>
    </div>
  </div>
</template>

OK,以上就是Electron35整合DeepSeek搭建本地客户端AI模板的一些知识分享。感谢大家的阅读与支持!

热文推荐

vue3仿Deepseek/ChatGPT流式聊天AI界面,对接deepseek/OpenAI API

vue3.5+deepseek+arco+markdown搭建网页web流式输出AI模板

Flutter3仿抖音app商城|flutter3短视频+直播+聊天App实例

flutter3-winchat:基于Flutter3.x+Dart3+Matetial3仿微信Exe桌面端聊天

自研tauri2.0+vite6.x+vue3+rust+arco-design桌面版os管理系统Tauri2-ViteOS

基于uniapp+vue3+uvui搭建跨端仿携程app酒店预订系统

Vite5-Electron-Wechat聊天实例|electron31+vue3客户端聊天EXE

Tauri2.0-ViteAdmin通用后台管理|vue3+tauri2+element-plus桌面版后台EXE

s13.sinaimg.gif