Electron31-Vue3Admin管理系统|vite5+electron+pinia桌面端后台Exe
自研electron31+vite5桌面端高颜值后台管理解决方案ElectronViteAdmin。
vite-electron31-admin原创基于electron31+vite5+vue3 setup+pinia2+element-plus+echarts搭建客户端轻量级后台管理系统。内置4种通用布局模板,支持i18n国际化、动态路由权限,整合了表格、表单、图表、列表、编辑器等业务场景。
技术栈
- 编辑器:VScode
- 框架技术:vite^5.3.4+vue^3.4.31+vue-router^4.4.0
- 跨端框架:electron^31.3.0
- UI组件库:element-plus^2.7.8
- 状态管理:pinia^2.2.0
- 国际化方案:vue-i18n@9
- 图表组件:echarts^5.5.1
- markdown编辑器:md-editor-v3^4.18.0
- 模拟数据:mockjs^1.1.0
- 打包工具:electron-builder^24.13.3
- electron+vite桥接插件:vite-plugin-electron^0.28.7
项目目录结构
vite-electron-admin桌面端后台系统使用 electron31+vite5 搭建项目模板,采用 vue3 setup 语法开发。
功能特性
- 最新前端技术栈Vite5.x、Vue3、Electron31、ElementPlus、Vue-I18n、Echarts
- 支持中英文/繁体国际化解决方案
- 支持动态权限路由、多页签缓存路由
- 封装多窗口管理器
- 内置4种通用布局模板、自由切换风格
- 整合通用的表格、表单、列表、图表、编辑器、错误处理等模块
- 高颜值UI界面、轻量级模块化、高定制性
目前Electron31-Vue3Admin通用后台系统已经发布到我的原创作品集。
https://gf.bilibili.com/item/detail/1106734011
Element Plus组件库
electron-viteadmin后台系统采用饿了么前端团队推出的vue3组件库。
Electron主线程配置
/**
* electron主线程配置
* @author andy
*/ import { app, BrowserWindow } from 'electron' import { WindowManager } from '../src/windows/index.js' // 忽略安全警告提示 Electron Security Warning (Insecure Content-Security-Policy)
process.env['ELECTRON_DISABLE_SECURITY_WARNINGS'] = true const createWindow = () => {
let win = new WindowManager()
win.create({isMajor: true})
// 系统托盘管理
win.trayManager()
// 监听ipcMain事件
win.ipcManager()
} app.whenReady().then(() => {
createWindow() app.on('activate', () => {
if(BrowserWindow.getAllWindows().length === 0) createWindow()
})
}) app.on('window-all-closed', () => {
if(process.platform !== 'darwin') app.quit()
})
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) {
console.log('窗口参数:', config)
console.log('窗口id:', config?.id) // 全局存储窗口配置
window.config = config
} // 初始化app应用实例
createApp(App)
.use(Router)
.use(Pinia)
.use(Plugins)
.mount('#app')
})
Electron封装多窗口管理|自定义系统导航栏
如上图:登录窗口切换到主窗口。
<script setup>
import { ref, markRaw } from 'vue'
import { ElMessageBox } from 'element-plus'
import { QuestionFilled, SwitchButton } from '@element-plus/icons-vue'
import { isTrue } from '@/utils'
import { authState } from '@/pinia/modules/auth'
import { winSet } from '@/windows/actions' const authstate = authState() const props = defineProps({
color: String,
// 窗口是否可最小化
minimizable: {type: [Boolean, String], default: true},
// 窗口是否可最大化
maximizable: {type: [Boolean, String], default: true},
// 窗口是否可关闭
closable: {type: [Boolean, String], default: true},
// 层级
zIndex: {type: [Number, String], default: 2024},
}) const hasMaximized = ref(false) // 初始监听窗口是否最大化
window.electron.invoke('win-isMaximized').then(res => {
hasMaximized.value = res
})
// 实时监听窗口是否最大化
window.electron.on('win-maximized', (e, data) => {
hasMaximized.value = data
}) // 最小化
const handleWinMin = () => {
// winSet('minimize', window.config.id)
window.electron.invoke('win-min')
}
// 最大化/还原
const handleWinToggle = () => {
// winSet('max2min', window.config.id)
window.electron.invoke('win-toggle').then(res => {
hasMaximized.value = res
})
}
// 关闭
const handleWinClose = () => {
if(window.config.isMajor) {
ElMessageBox.confirm('是否最小化到系统托盘,不退出应用程序?', '', {
type: 'warning',
icon: markRaw(QuestionFilled),
confirmButtonText: '退出应用',
cancelButtonText: '最小化到托盘',
customStyle: {'borderRadius': '8px'},
roundButton: true,
distinguishCancelAndClose: true,
}).then(() => {
authstate.logout()
winSet('close')
}).catch((action) => {
if(action === 'cancel') {
setTimeout(() => {
winSet('hide', window.config.id)
}, 250)
}
})
}else {
winSet('close', window.config.id)
}
}
</script> <template>
<div class="ev__winbtns vu__drag" :style="{'z-index': zIndex}">
<div class="ev__winbtns-actions vu__undrag" :style="{'color': color}">
<a v-if="isTrue(minimizable)" class="wbtn min" title="最小化" @click="handleWinMin"><i class="wicon iconfont elec-icon-min"></i></a>
<a v-if="isTrue(maximizable)" class="wbtn toggle" :title="hasMaximized ? '向下还原' : '最大化'" @click="handleWinToggle">
<i class="wicon iconfont" :class="hasMaximized ? 'elec-icon-restore' : 'elec-icon-max'"></i>
</a>
<a v-if="isTrue(closable)" class="wbtn close" title="关闭" @click="handleWinClose"><i class="wicon iconfont elec-icon-quit"></i></a>
</div>
</div>
</template>
Electron-Vue3Admin布局模板
如上图:内置了4种常用的通用布局模板。也可以根据需要定制化模板。
/**
* 通用布局模板
* @author Andy Q:282310962
*/ <script setup>
import { appState } from '@/pinia/modules/app' // 引入布局模板
import Classic from './template/classic/index.vue'
import Columns from './template/columns/index.vue'
import Vertical from './template/vertical/index.vue'
import Horizontal from './template/horizontal/index.vue' const appstate = appState() const LayoutMap = {
'classic': Classic,
'columns': Columns,
'vertical': Vertical,
'horizontal': Horizontal
}
</script> <template>
<div class="vuadmin__container" :style="{'--themeSkin': appstate.config.skin}">
<component :is="LayoutMap[appstate.config.layout]" />
</div>
</template>
electron+vue3-i18n国际化配置
electron-viteadmin系统采用 vue-i18n 国际化解决方案,支持中英文/繁体语言。
/**
* 国际化配置
* @author YXY
*/ import { createI18n } from 'vue-i18n'
import { appState } from '@/pinia/modules/app' // 引入语言配置
import enUS from './en-US'
import zhCN from './zh-CN'
import zhTW from './zh-TW' // 默认语言
export const langVal = 'zh-CN' export default async (app) => {
const appstate = appState()
const lang = appstate.lang || langVal
appstate.setLang(lang) const i18n = createI18n({
legacy: false,
locale: lang,
messages: {
'en': enUS,
'zh-CN': zhCN,
'zh-TW': zhTW
}
}) app.use(i18n)
}
Vue3动态化图表Echarts
vue3封装图表hooks,用于多个图表符合调用。采用element-resize-detector组件动态监听窗口DOM尺寸变化更新图表。
/**
* 动态图表Hook
*/ import { onMounted, onBeforeUnmount, ref } from 'vue'
import * as echarts from 'echarts'
import elementResizeDetectorMaker from 'element-resize-detector' export function useEcharts(el, options) {
let chartEl
let chartRef = ref(null)
let erd = elementResizeDetectorMaker() const resizeHandle = () => {
chartEl && chartEl.resize()
} onMounted(() => {
if(el?.value) {
chartEl = echarts.init(el.value)
chartEl.setOption(options)
chartRef.value = chartEl
}
erd.listenTo(el.value, resizeHandle)
}) onBeforeUnmount(() => {
chartEl.dispose()
erd.removeListener(el.value, resizeHandle)
}) return chartRef
}
import { useEcharts } from '@/hooks/useEcharts' const userActionChartRef = ref(null)
useEcharts(userActionChartRef, {
// ...
})
Electron+vue3自定义路由菜单
<Menus :rootRouteEnable="false" /> <Menus rootRouteEnable :dark="true" /> <Menus mode="horizontal" :dark="true" />
<script setup>
import { ref, computed } from 'vue'
import { isObject, isArray, isImg } from '@/utils'
import { appState } from '@/pinia/modules/app'
import { useRoutes } from '@/hooks/useRoutes' const props = defineProps({
// 菜单模式(vertical|horizontal)
mode: { type: String, default: 'vertical' },
// 是否开启一级路由菜单
rootRouteEnable: { type: Boolean, default: true },
// 是否暗黑模式
dark: { type: Boolean }
}) import Submenu from './submenu.vue' // 引入主路由表
import routes from '@/router/modules/main.js' const appstate = appState()
const { route, getActiveRoute, getCurrentRootRoute, getTreeRoutes } = useRoutes() const activeRoute = computed(() => getActiveRoute(route))
const rootRoute = computed(() => getCurrentRootRoute(route))
const treeRoutes = computed(() => getTreeRoutes(routes)) const filterRoutes = computed(() => {
if(props.rootRouteEnable) {
return treeRoutes.value
}
// 过滤一级路由菜单
return treeRoutes.value.find(item => item.path === rootRoute.value && item.children)?.children
})
</script> <template>
<div class="vu__menubar" :class="{'is-dark': dark, 'is-collapsed': mode == 'vertical' && appstate.config.collapsed}">
<el-menu class="vu__menus" :default-active="activeRoute" :mode="mode" :collapse="appstate.config.collapsed">
<Submenu
v-for="route in filterRoutes"
:key="route.path"
:item="route"
:rootRoute="rootRoute"
:rootRouteEnable="rootRouteEnable"
/>
</el-menu>
</div>
</template>
electron+vue3多页签tabview
element-plus组件库el-dropdown组件右键控制每次只显示一个下拉菜单。
<template>
<div class="vu__tabview">
<el-tabs
v-model="activeTab"
class="vu__tabview-tabs"
@tab-change="changeTabs"
@tab-remove="removeTab"
>
<el-tab-pane
v-for="(item, index) in tabList"
:key="index"
:name="item.path"
:closable="!item?.meta?.isAffix"
>
<template #label>
<el-dropdown ref="dropdownRef" trigger="contextmenu" :id="item.path" @visible-change="handleDropdownChange($event, item.path)" @command="handleDropdownCommand($event, item)">
<span class="vu__tabview-tabs__label">
<span>{{$t(item?.meta?.title)}}</span>
</span>
<template #dropdown>
<el-dropdown-menu>
<el-dropdown-item command="refresh" :icon="Refresh">{{$t('tabview__contextmenu-refresh')}}</el-dropdown-item>
<el-dropdown-item command="close" :icon="Close" :disabled="item.meta.isAffix">{{$t('tabview__contextmenu-close')}}</el-dropdown-item>
<el-dropdown-item command="closeOther" :icon="Switch">{{$t('tabview__contextmenu-closeother')}}</el-dropdown-item>
<el-dropdown-item command="closeLeft" :icon="DArrowLeft">{{$t('tabview__contextmenu-closeleft')}}</el-dropdown-item>
<el-dropdown-item command="closeRight" :icon="DArrowRight">{{$t('tabview__contextmenu-closeright')}}</el-dropdown-item>
<el-dropdown-item command="closeAll" :icon="CircleCloseFilled">{{$t('tabview__contextmenu-closeall')}}</el-dropdown-item>
</el-dropdown-menu>
</template>
</el-dropdown>
</template>
</el-tab-pane>
</el-tabs>
</div>
</template>
<script setup>
import { onMounted, ref, computed, watch, nextTick } from 'vue'
import { useRouter, useRoute } from 'vue-router'
import { useI18n } from 'vue-i18n'
import { Refresh, Close, Switch, DArrowLeft, DArrowRight, CircleCloseFilled } from '@element-plus/icons-vue'
import { isObject, isImg } from '@/utils'
import { useLink } from '@/hooks/useLink'
import { appState } from '@/pinia/modules/app' const router = useRouter()
const route = useRoute()
const { jump } = useLink()
const { locale } = useI18n()
let { config: { keepAlive, tabRoutes, cacheRoutes }, updateConfig } = appState() const dropdownRef = ref()
const activeTab = ref(route.path)
const tabList = ref(tabRoutes) // 新增选项卡
const addTab = () => {
const index = tabList.value.findIndex(item => item?.path === activeTab.value)
if(index == -1) {
tabList.value.push({
path: route?.path,
name: route?.name,
meta: {
...route?.meta,
}
})
}
updateConfig('tabRoutes', tabList.value)
updateCacheRoutes()
} // 删除选项卡
const removeTab = (path) => {
const index = tabList.value.findIndex(item => item?.path === path)
if(index > -1) {
tabList.value.splice(index, 1)
updateTabs(tabList.value)
}
} // 删除左侧选项卡
const removeLeftTab = (path) => {
const index = tabList.value.findIndex(item => item?.path === path)
if(index > -1) {
tabList.value = tabList.value.filter((item, i) => item?.meta?.isAffix || i >= index)
updateTabs(tabList.value)
}
} // 删除右侧选项卡
const removeRightTab = (path) => {
const index = tabList.value.findIndex(item => item?.path === path)
if(index > -1) {
tabList.value = tabList.value.filter((item, i) => item?.meta?.isAffix || i <= index)
updateTabs(tabList.value)
}
} // 删除其它选项卡
const removeOtherTab = (path) => {
tabList.value = tabList.value.filter(item => item?.meta?.isAffix || item?.path === path)
updateTabs(tabList.value)
} // 删除全部
const removeAllTab = (path) => {
tabList.value = tabList.value.filter(item => item?.meta?.isAffix)
updateTabs(tabList.value)
} // 更新选项卡
const updateTabs = (tabs) => {
updateConfig('tabRoutes', tabs)
updateCacheRoutes()
const nextTab = tabs[tabs.length + 1] || tabs[tabs.length - 1]
if(!nextTab) return
jump(nextTab?.path)
} // 更新keep-alive缓存
const updateCacheRoutes = () => {
let caches = tabList.value.filter(item => keepAlive || item?.meta?.isKeepAlive).map(item => item.name)
// console.log('cacheViews缓存路由>>:', caches)
updateConfig('cacheRoutes', caches)
} // 清空keep-alive缓存
const clearCacheRoutes = () => {
updateConfig('cacheRoutes', [])
} // 点击选项卡
const changeTabs = (path) => {
jump(path)
} // 右键菜单更新
const handleDropdownChange = (visible, name) => {
// 控制每次只显示一个右键菜单
if(!visible) return
dropdownRef.value.forEach(item => {
if(item.id === name) return
item.handleClose()
})
}
// 右键菜单命令
const handleDropdownCommand = (cmd, item) => {
const path = item?.path
switch(cmd) {
case 'refresh':
router.go(0)
break
case 'close':
removeTab(path)
break
case 'closeLeft':
removeLeftTab(path)
break
case 'closeRight':
removeRightTab(path)
break
case 'closeOther':
removeOtherTab(path)
break
case 'closeAll':
removeAllTab()
break
}
} watch(() => route.path, () => {
activeTab.value = route.path
addTab()
}, {
immediate: true
})
</script>
综上就是electron31+vue3+element-plus实战桌面端轻量级后台系统的一些知识分享,希望对大家有所帮助!
最后附上两个最新原创项目实例
https://www.cnblogs.com/xiaoyan2017/p/18290962
https://www.cnblogs.com/xiaoyan2017/p/18323930
Electron31-Vue3Admin管理系统|vite5+electron+pinia桌面端后台Exe的更多相关文章
- Electron-Vue3-Vadmin后台系统|vite2+electron桌面端权限管理系统
基于vite2.x+electron12桌面端后台管理系统Vite2ElectronVAdmin. 继上一次分享vite2整合electron搭建后台框架,这次带来的是最新开发的跨桌面中后台权限管理系 ...
- AngularJS 和 Electron 构建桌面应用
译]使用 AngularJS 和 Electron 构建桌面应用 原文: Creating Desktop Applications With AngularJS and GitHub Electro ...
- 使用react全家桶制作博客后台管理系统 网站PWA升级 移动端常见问题处理 循序渐进学.Net Core Web Api开发系列【4】:前端访问WebApi [Abp 源码分析]四、模块配置 [Abp 源码分析]三、依赖注入
使用react全家桶制作博客后台管理系统 前面的话 笔者在做一个完整的博客上线项目,包括前台.后台.后端接口和服务器配置.本文将详细介绍使用react全家桶制作的博客后台管理系统 概述 该项目是基 ...
- APP 性能分析工作台——你的最佳桌面端性能分析助手
目前 MARS-App 性能分析工作台版本为开发者提供Fastbot桌面版的服务. 旨在帮助开发者们更快.更便捷地开启智能测试之旅,成倍提升稳定性测试的效率. 作者:字节跳动终端技术--王凯 背景 F ...
- 基于Svelte3.x桌面端UI组件库Svelte UI
Svelte-UI,一套基于svelte.js开发的桌面pc端ui组件库 最近一直忙于写svelte-ui,一套svelte3开发的桌面端ui组件库.在设计及功能上借鉴了element-ui组件库.所 ...
- Tauri-Vue3桌面端聊天室|tauri+vite3仿微信|tauri聊天程序EXE
基于tauri+vue3.js+vite3跨桌面端仿微信聊天实例TauriVue3Chat. tauri-chat 运用最新tauri+vue3+vite3+element-plus+v3layer等 ...
- ve-plus:基于 vue3.x 桌面端UI组件库|vue3组件库
VE-Plus 自研轻量级 vue3.js 桌面pc端UI组件库 经过一个多月的筹划及开发,今天给大家带来一款全新的Vue3桌面端UI组件库VEPlus.新增了35+常用的组件,采用vue3 setu ...
- Vite-Admin后台管理系统|vite4+vue3+pinia前端后台框架实例
基于vite4.x+vue3+pinia前端后台管理系统解决方案ViteAdmin. 前段时间分享了一篇vue3自研pc端UI组件库VEPlus.这次带来最新开发的基于vite4+vue3+pinia ...
- arcpy+PyQt+py2exe快速开发桌面端ArcGIS应用程序
前段时间有一个项目,大体是要做一个GIS数据处理工具. 一般的方法是基于ArcObjects来进行开发,因为我对ArcObjects不太熟悉,所以就思考有没有其他简单快速的方法来做. 在查看ArcGI ...
- 修改远程桌面端口号.bat
@color 0A @title 修改远程桌面端口号 by wjshan0808 @echo off echo 请输入端口号 set /p port= reg add "HKLM\SYSTE ...
随机推荐
- Linux增加系统调用(亲测成功)
我使用的操作系统是CentOS,其他的操作系统类似. 相关软件和Linux的基础操作这里不再赘述. 实验环境 VMWare Workstation.CentOS-7 实验步骤 ...
- Ubuntu下nvidia驱动卸载
Ubuntu下nvidia驱动卸载的一种方法 卸掉已经安装的驱动: sudo apt-getremove --purge '^nvidia-.*' sudo apt-getremove --purge ...
- 超大容量 | 瑞芯微RK3588J工业核心板新增16GB DDR + 128GB eMMC配置!
作为瑞芯微的金牌合作伙伴,创龙科技在2023年9月即推出搭载瑞芯微旗舰级处理器RK3588J的全国产工业核心板--SOM-TL3588. SOM-TL3588工业核心板是基于瑞芯微RK3588J/RK ...
- 工控必备!NXP i.MX 8M Mini开发板规格书资料分享,高性能低功耗!
1 核心板简介 创龙科技SOM-TLIMX8-B是一款基于NXP i.MX 8M Mini的四核ARM Cortex-A53 + 单核ARM Cortex-M4异构多核处理器设计的高端工业级核心板,A ...
- 【AppStore】一文让你学会IOS应用上架Appstore
前言 咱们国内现在手机分为两类,Android手机与苹果手机,现在用的各类APP,为了手机的使用安全,避免下载到病毒软件,官方都极力推荐使用手机自带的应用商城进行下载,但是国内Android手机品类众 ...
- Linux基本编程环境安装
前言 可以采用组合式安装,如:https://oneinstack.com/ 选择好要安装的,然后复制安装命令就可以一键搞定很多东西了 VMware安装Centos7 按照物理机CPU实际情况,选择处 ...
- 在IDEA中找不到Mapper报错
前言 相信大多数互联网公司的持久层框架都是使用 Mybatis 框架,而大家在 Service 层引入自己编写的 Mapper 接口时应该会遇到下面的情况: 我们可以看到,上面的红色警告在提示我们,找 ...
- PowerBuilder编程新思维6.5:外传1(PowerPlume的设计与规划)
<第五部分 Otherside 意外的宝藏> 每一颗种子都有发芽的梦想.PowerPlume(孔雀翎)开发交流群:286502392 PowerBuilder编程新思维6.5:外传1 ...
- Django导出EXCEL并确保表头左右两列显示
以下是导出EXCEL确保表头左右两列显示正确值的代码示例: from openpyxl import Workbook from openpyxl.styles import Alignment # ...
- [oeasy]python0015_键盘改造_将esc和capslock对调_hjkl_移动_双手正位
键盘改造 回忆上次内容 上次练习了复制粘贴 按键 作用 <kbd>y</kbd><kbd>y</kbd> 复制光标行代码 到剪贴板 <kbd> ...