原创vue3+uniapp+uni-ui跨端仿ios桌面后台OA 管理模板Uni-Vue3-WeOS。
uniapp-vue3-os 一款基于uni-app+vite5.x+pinia等技术开发的仿ios手机桌面OA 管理系统。实现了自定义桌面栅格磁贴布局、多分屏滑动管理、自定义桌面小部件、辅助触控悬浮球等功能。支持编译到H5+小程序端+App端。
预览效果
运行到H5/小程序端/APP端效果。
data:image/s3,"s3://crabby-images/bea40/bea40835791500a896ae0fbe8544ca540da80446" alt=""
原创自研手机OA磁贴栅格卡片布局引擎。亲测在h5+小程序+App端效果基本保持一致。
data:image/s3,"s3://crabby-images/9a918/9a9189bca2438ac23bdc79adae51dd3524d829bd" alt=""
data:image/s3,"s3://crabby-images/8a11f/8a11ff5fdee98e71756f9379fe9d6042f547a315" alt=""
使用技术
- 编辑器:HbuilderX 4.15
- 技术框架:uniapp+vite5.x+vue3+pinia2
- UI组件库:uni-ui+uv-ui(uniapp vue3组件库)
- 弹框组件:uv3-popup(基于uniapp+vue3自定义弹框组件)
- 表格组件:uv3-table(基于uniapp+vue3增强版表格)
- 模拟数据:mockjs(用于自定义表格模拟数据)
- 缓存技术:pinia-plugin-unistorage
- 支持编译:h5+小程序端+app端
data:image/s3,"s3://crabby-images/21262/21262f6973f85e319bf449b3410ceba594e21156" alt=""
另外在PC端 则以750px布局显示,表现依然perfect~~
data:image/s3,"s3://crabby-images/76084/76084929d4edbd411e33028d0b92f6e7d005deb0" alt=""
uni-vue3-oadmin项目使用到的table表格组件uv3Table 。全新自研 的一款跨多端(h5/小程序/App端)增强版自定义表格组件。
https://www.cnblogs.com/xiaoyan2017/p/18199130
data:image/s3,"s3://crabby-images/48bfe/48bfe065ccb2774722f4dbece33d1fb9cf5dfc07" alt=""
项目结构
使用hbuilderx4.15编辑器,采用vue3 setup语法编码开发。内置构建工具升级到了Vite5.2.8版本。
data:image/s3,"s3://crabby-images/42983/4298335f73dbbd945627135d9cbcb6c796054006" alt=""
data:image/s3,"s3://crabby-images/f04f2/f04f23d006aeff2874f85c303eaf3acb6f7ec736" alt=""
data:image/s3,"s3://crabby-images/a90f4/a90f48c3bbf637534df16e38f452ff826f3bf22e" alt=""
data:image/s3,"s3://crabby-images/4dd39/4dd392cae2df8bba32683129f9a114b7176b04d1" alt=""
目前该项目已经上架到我的作品集,如果有需要的话,欢迎去下载使用。
https://gf.bilibili.com/item/detail/1105982011
data:image/s3,"s3://crabby-images/d3c37/d3c372be4b0578a62d5e2450707882725fa2fd29" alt=""
公共布局模板Layout
<script setup>
import { ref } from 'vue'
import { appStore } from '@/pinia/modules/app'
const appState = appStore()
// #ifdef MP-WEIXIN
defineOptions({
/**
* 解决小程序class、id透传问题(vue3写法)
* manifest.json中配置mergeVirtualHostAttributes: true, 在微信小程序平台不生效,组件外部传入的class没有挂到组件根节点上
* https://github.com/dcloudio/uni-ui/issues/753
*/
options: { virtualHost: true }
})
// #endif
const props = defineProps({
showBackground: { type: [Boolean, String], default: true },
})
// 自定义变量(桌面图标)
const deskVariable = ref({
'--icon-radius': '15px', // 圆角
'--icon-size': '118rpx', // 图标尺寸
'--icon-gap-col': '25px', // 水平间距
'--icon-gap-row': '45px', // 垂直间距
'--icon-labelSize': '12px', // 标签文字大小
'--icon-labelColor': '#fff', // 标签颜色
'--icon-fit': 'contain', // 图标自适应模式
})
</script>
<template>
<view class="uv3__container flexbox flex-col flex1" :style="deskVariable">
<!-- 顶部插槽 -->
<slot name="header" />
<!-- 内容区 -->
<view class="uv3__scrollview flex1">
<slot />
</view>
<!-- 底部插槽 -->
<slot name="footer" />
<!-- 背景图(修复小程序不支持background背景图) -->
<image v-if="showBackground" class="fixwxbg" :src="appState.config.skin || '/static/skin/theme.png'" mode="scaleToFill" />
</view>
</template>
data:image/s3,"s3://crabby-images/b7d76/b7d76a270dc6426521d6c319bec91217aea8e37c" alt=""
data:image/s3,"s3://crabby-images/94321/94321873478bd4a074733feb1d6564cf19926b3f" alt=""
data:image/s3,"s3://crabby-images/62d9d/62d9d349d6df8143432ebba7e5848c501a85f902" alt=""
data:image/s3,"s3://crabby-images/47cc4/47cc4f3e2da3e790e06cc8fd54b8cd8f5ab9451e" alt=""
data:image/s3,"s3://crabby-images/5a6ff/5a6ff4f59ffe7154e5d8d96b3f0543190bd6bbe7" alt=""
data:image/s3,"s3://crabby-images/84656/846560e762e3d1afb4528757e6c44adfce0a0dd8" alt=""
data:image/s3,"s3://crabby-images/aa099/aa099803edf41df406a6712391572312dee4c3ef" alt=""
data:image/s3,"s3://crabby-images/41b45/41b452061a9b677e2f0dc794321651377bbc82be" alt=""
data:image/s3,"s3://crabby-images/b7ff2/b7ff270b4461121ecdfb4c2aa8f80c1dcb787b03" alt=""
data:image/s3,"s3://crabby-images/858e9/858e9154da9bffdcf66d36b93b9501c52279c516" alt=""
data:image/s3,"s3://crabby-images/13d4d/13d4df48da7061a3d1e213f38aefdf7d73460e35" alt=""
data:image/s3,"s3://crabby-images/0d050/0d050dce736612134907de867fddc9b2675c9ff6" alt=""
data:image/s3,"s3://crabby-images/8ee64/8ee64e2c83c917e2580df812ac3636427ed84200" alt=""
data:image/s3,"s3://crabby-images/0cbbc/0cbbc2e82bb8b5da8be720f6896e915c34fe74f4" alt=""
data:image/s3,"s3://crabby-images/827ad/827ad43183fc23053960dc804eee5cd82519e053" alt=""
data:image/s3,"s3://crabby-images/c3b62/c3b627a6cdb215e21c1385cc15156e16c1292da9" alt=""
data:image/s3,"s3://crabby-images/24777/24777977437c2d9dcc925f7460d411b5e4aa292e" alt=""
data:image/s3,"s3://crabby-images/04902/049025972fd0acf189f69d1de5809ea641cd24f6" alt=""
data:image/s3,"s3://crabby-images/9e0c0/9e0c00fcf23ae382ed0a00dfac52d78221157d8d" alt=""
emmm,怎么样,是不是感觉还行~~ 哈哈,这也是经历了无数个日夜的爆肝开发,目前该项目正式的完结了。
桌面布局模板
data:image/s3,"s3://crabby-images/bdca0/bdca04af11b5104ce505478d4729b984e51fb0d4" alt=""
<!-- 桌面模板 -->
<script setup>
import { ref } from 'vue'
import Desk from './components/desk.vue'
import Dock from './components/dock.vue'
import Touch from './components/touch.vue'
</script>
<template>
<uv3-layout>
<!-- 桌面菜单 -->
<Desk />
<template #footer>
<!-- 底部导航 -->
<Dock />
</template>
<!-- 悬浮球(辅助触控) -->
<Touch />
</uv3-layout>
</template>
桌面卡片式栅格磁贴模板
data:image/s3,"s3://crabby-images/dc8bb/dc8bb7fdb6b46263ed7fe6a009167a9abae5eded" alt=""
data:image/s3,"s3://crabby-images/9861c/9861cf87ed9ab89d38e152bf4e12185d277823e9" alt=""
data:image/s3,"s3://crabby-images/14e6d/14e6de22e5e575642e2470efe00916dc8a6174ec" alt=""
桌面os菜单采用json配置
/**
* label 图标标题
* imgico 图标(本地或网络图片) 当type: 'icon'则为uni-icons图标名,当type: 'widget'则为自定义小部件标识名
* type 图标类型(icon | widget) icon为uni-icons图标、widget为自定义小部件
* path 跳转路由页面
* link 跳转外部链接
* hideLabel 是否隐藏图标标题
* background 自定义图标背景色
* size 栅格磁贴布局(16种) 1x1 1x2 1x3 1x4、2x1 2x2 2x3 2x4、3x1 3x2 3x3 3x4、4x1 4x2 4x3 4x4
* onClick 点击图标回调函数
* children 二级菜单
*/
data:image/s3,"s3://crabby-images/efd66/efd66f09775b03bac26c78657934b5157910128c" alt=""
data:image/s3,"s3://crabby-images/f3318/f3318e6d5657a17e6c2be68b63ea6b0ef12950e1" alt=""
配置children参数,则以二级菜单弹窗展示。
data:image/s3,"s3://crabby-images/ec0ac/ec0ac98cde3925e53f9bcbe673f7504f1d6e1fd2" alt=""
<template>
<swiper
class="uv3__deskmenu"
:indicator-dots="true"
indicator-color="rgba(255,255,255,.5)"
indicator-active-color="#fff"
>
<swiper-item v-for="(mitem, mindex) in deskMenu" :key="mindex">
<view class="uv3__gridwrap">
<view v-for="(item, index) in mitem.list" :key="index" class="uv3__gridwrap-item" @click="handleClickDeskMenu(item)">
<!-- 图标 -->
<view class="ico" :style="{'background': item.background}">
<!-- 二级菜单 -->
<template v-if="Array.isArray(item.children)">
<view class="uv3__gridwrap-thumb">
...
</view>
</template>
<template v-else>
<template v-if="item.type == 'widget'">
<!-- 自定义部件 -->
<component :is="item.imgico" />
</template>
<template v-else>
<!-- 自定义图标 -->
...
</template>
</template>
</view>
<!-- 标签 -->
<view v-if="!item.hideLabel" class="label clamp2">{{item.label}}</view>
</view>
</view>
</swiper-item>
</swiper>
<!-- 桌面二级菜单弹窗 -->
<Popup v-model="deskPopupVisible">
<view class="uv3__deskpopup">
...
</view>
</Popup>
...
</template>
点击桌面菜单,打开链接地址、跳转路由页面、二级弹窗、自定义绑定事件等方式。当然也可以进行一些其它定制化逻辑处理。
const handleClickDeskMenu = (item) => {
if(item.link) {
// 链接
openURL(item.link)
}else if(item.path) {
// 页面路由地址
uni.navigateTo({
url: item.path.substr(0, 1) == '/' ? item.path : '/' + item.path
})
}else if(Array.isArray(item.children)) {
// 二级菜单
deskPopupMenu.value = item
deskPopupVisible.value = true
}
// 绑定点击事件
typeof item.onClick === 'function' && item.onClick()
}
桌面菜单JSON配置项示例
const deskMenu = ref([
{
pid: 20240507001,
list: [
{label: '今日', imgico: 'today', type: 'widget', hideLabel: true, size: '2x1'},
{label: '天气', imgico: 'weather', type: 'widget', hideLabel: true, size: '2x1'},
{label: '日历', imgico: 'fullcalendar', type: 'widget', path: 'pages/calendar/index', size: '4x2'},
// {label: '日历', imgico: 'date', type: 'widget', size: '2x2'},
// {label: '备忘录', imgico: 'note', type: 'widget', size: '2x2'},
{label: 'audio', imgico: 'audio', type: 'widget', size: '2x1'},
{
label: '相册', imgico: '/static/svg/huaban.svg', background: '#00aa7f',
onClick: () => {
// ...
}
},
...
]
},
...
{
pid: 20240510001,
list: [
{label: 'Github', imgico: '/static/svg/github.svg', background: '#607d8b', size: '3x1'},
{label: '码云Gitee', imgico: '/static/svg/gitee.svg', background: '#bb2124',},
{label: '抖音', imgico: '/static/svg/douyin.svg', background: '#1c0b1a', size: '1x2'},
{label: 'ChatGPT', imgico: '/static/svg/chatgpt.svg', hideLabel: true, background: '#11b6a7', size: '3x2'},
...
]
},
{
pid: 20240511003,
list: [
{label: 'uni-app', imgico: '/static/uni.png', link: 'https://uniapp.dcloud.net.cn/'},
{label: 'vitejs官方文档', imgico: '/static/vite.png', link: 'https://vitejs.dev/'},
{
label: '主题壁纸', imgico: 'color-filled', type: 'icon',
onClick: () => {
// ...
}
},
{label: '日历', imgico: 'calendar', type: 'widget', path: 'pages/calendar/index', background: '#fff',},
{label: '首页', imgico: 'home', type: 'icon', path: 'pages/index/index'},
{label: '工作台', imgico: 'shop-filled', type: 'icon', path: 'pages/index/dashboard'},
{
label: '组件',
'children': [
{label: '组件', imgico: '/static/svg/component.svg', path: 'pages/component/index'},
{label: '表格', imgico: '/static/svg/table.svg', path: 'pages/component/table'},
...
]
},
...
{
label: '关于', imgico: 'info-filled', type: 'icon',
onClick: () => {
// ...
}
},
{
label: '公众号', imgico: 'weixin', type: 'icon',
onClick: () => {
// ...
}
},
]
}
])
整个项目采用毛玻璃模糊化UI视觉效果。简单的实现了表格、表单、编辑器、用户管理/角色管理等常用业务功能。旨在探索uniapp全新的手机后台管理系统方案,当然也可以在此基础上做一些其它创新,加上一些定制化功能模块。
data:image/s3,"s3://crabby-images/7a66d/7a66d62a8f2606eb4ac2163163916a7984d9a835" alt=""
OK,以上就是uniapp+vue3开发手机OA管理管理系统的一些分享,希望对大家有些帮助哈!
最后附上两个最新实例项目
https://www.cnblogs.com/xiaoyan2017/p/18165578
https://www.cnblogs.com/xiaoyan2017/p/18092224
data:image/s3,"s3://crabby-images/adef0/adef0cd6b15747e7206d894920d59cdd7edfe845" alt=""