原创自研flutter3.x+getx仿制ios手机桌面UI管理系统模板Flutter3-OS。
flutter3-osx 基于最新跨平台技术Flutter3.22+Dart3.4+GetX+fl_chart实战仿IOS风格手机os 管理系统。全新自研flutter磁贴式栅格布局引擎、分屏式多页管理、自定义主题壁纸、卡片式桌面小部件、可拖拽式悬浮球菜单等功能。
data:image/s3,"s3://crabby-images/41c3b/41c3bbfd74a6e300f1370c4e5a0004b2ad90cdf2" alt=""
全新原创自研的OS栅格化菜单布局引擎 ,探索flutter手机端管理系统OA新模式。
data:image/s3,"s3://crabby-images/5611e/5611e834b29427f470ff1edc9ec584614c75f5d7" alt=""
技术栈
- 编辑器:VScode
- 技术框架:Flutter3.22.1+Dart3.4.1
- 路由/状态管理:get^4.6.6
- 本地存储:get_storage^2.1.1
- svg图片插件:flutter_svg^2.0.10+1
- 图表组件:fl_chart^0.68.0
- 国际化时间:intl^0.19.0
data:image/s3,"s3://crabby-images/ac8c7/ac8c7b60d7fd82493936e90af8780cebf571cf3f" alt=""
flutter3-os项目在windows端效果依然nice!其实之前也有专门分享一款flutter3仿macOS桌面框架,感兴趣的可以去看看。
https://www.cnblogs.com/xiaoyan2017/p/18132176
data:image/s3,"s3://crabby-images/5dc38/5dc38fc48a586f884f3a0f99b6f91895e481de93" alt=""
功能特性
✅ 经典的栅格化布局+Dock导航模式
✅ 桌面菜单JSON配置生成
✅ 支持16种栅格布局模式
✅ 分屏式多页管理
✅ 可拖拽悬浮球菜单
✅ 毛玻璃虚化背景操作窗口
✅ 丰富的视觉效果,自定义桌面个性壁纸
✅ 高定制化自定义桌面小部件
data:image/s3,"s3://crabby-images/95b2d/95b2d869ffb7a32c30ad4e3aab63c18d9268b062" alt=""
data:image/s3,"s3://crabby-images/de964/de9648aa8d2481af97725ef1d8eee8fd5b16ea28" alt=""
项目结构目录
data:image/s3,"s3://crabby-images/750cc/750cc2369e025c2c14bfdffb0dba465c3a8a54ff" alt=""
使用最新版flutter3.22+dart3.4架构开发,颠覆传统后台管理,探索一种全新的手机后台OA管理系统新模式。
data:image/s3,"s3://crabby-images/44f96/44f96879f35a5dcbb162992cf5579ff8b7797617" alt=""
目前该项目已经同步到我的原创作品集,有需要的话,欢迎去瞅瞅~
https://gf.bilibili.com/item/detail/1106107011
data:image/s3,"s3://crabby-images/cea0f/cea0ff252cfc14ca943c4b28f8759e44a0a375e8" alt=""
入口配置
import 'dart:io';
import 'package:flutter/material.dart';
import 'package:get/get.dart';
import 'package:get_storage/get_storage.dart';
import 'package:intl/date_symbol_data_local.dart';
import 'utils/index.dart';
// 引入桌面栅格模板
import 'layouts/desk.dart';
// 引入路由管理
import 'router/index.dart';
void main() async {
// 初始化get_storage本地存储
await GetStorage.init();
// 初始化国际化语言
initializeDateFormatting();
runApp(const MyApp());
}
class MyApp extends StatelessWidget {
const MyApp({super.key});
// This widget is the root of your application.
@override
Widget build(BuildContext context) {
return GetMaterialApp(
title: 'Flutter WeOS',
debugShowCheckedModeBanner: false,
theme: ThemeData(
colorScheme: ColorScheme.fromSeed(seedColor: Colors.blue),
useMaterial3: true,
// 修复windows端字体粗细不一致
fontFamily: Platform.isWindows ? 'Microsoft YaHei' : null,
),
home: const DeskLayout(),
// 初始化路由
initialRoute: Utils.isLogin() ? '/' : '/launch',
// 路由页面
getPages: routes,
);
}
}
data:image/s3,"s3://crabby-images/1be4b/1be4b3fff7d19255c76530c66362f9b0ca60cc3a" alt=""
data:image/s3,"s3://crabby-images/cf21a/cf21a954bcfaaf0a0cfd3b5bf4a5ad5ef9b31e13" alt=""
动画式数字密码登录验证解锁新模式。
data:image/s3,"s3://crabby-images/e9b85/e9b85d15aa9a3f792d2793c02f726584a3c89130" alt=""
摒弃传统的输入框式登录方式,为了整体效果一致性采用全新数字解锁验证模式。
使用 AnimatedSwitcher 和 FadeTransition 配合实现切换动画效果。
@override
Widget build(BuildContext context) {
return Layout(
extendBodyBehindAppBar: true,
body: Container(
padding: const EdgeInsets.all(20.0),
child: AnimatedSwitcher(
duration: const Duration(milliseconds: 250),
// 动画控制
transitionBuilder: (child, animation) {
return FadeTransition(
opacity: animation,
child: ScaleTransition(
// scale: animation,
scale: animation.drive(Tween(begin: 0.9, end: 1.0).chain(CurveTween(curve: Curves.easeOut))),
child: child,
),
);
},
// 当内容有变化的时候就会触发动画
child: splashScreen ? GestureDetector(
// 修复Column和Row组件,点击空白处无响应问题
behavior: HitTestBehavior.translucent,
child: Column(
children: [
...
],
),
onPanStart: (details) {
setState(() {
swipeY = details.globalPosition.dy;
});
},
onPanUpdate: (details) {
double posY = swipeY - details.globalPosition.dy;
if(posY > 100) {
setState(() {
splashScreen = false;
});
}
},
)
:
Center(
child: Column(
mainAxisAlignment: MainAxisAlignment.center,
children: [
...
],
),
),
),
),
);
}
data:image/s3,"s3://crabby-images/b6f9d/b6f9d59cf7efeefcf366b19339d848ac778375ce" alt=""
数字密码长度可以自定义配置,没有直接写死6位。
Column(
children: [
const Text('数字密码解锁', style: TextStyle(color: Colors.white, fontSize: 14.0),),
const SizedBox(height: 10.0,),
Wrap(
spacing: 15.0,
children: List.generate(passwordArr.length, (index) {
return AnimatedContainer(
duration: const Duration(milliseconds: 300),
height: 10.0,
width: 10.0,
decoration: BoxDecoration(
color: int.parse(passwordArr[index]) <= pwdValue.length ? Colors.white : Colors.white.withOpacity(0.01),
border: Border.all(color: Colors.white),
borderRadius: BorderRadius.circular(50.0),
),
);
})
),
],
),
data:image/s3,"s3://crabby-images/a6ad9/a6ad9868d9561a027c5a5f675222fc6f8aee4cc1" alt=""
Container(
width: 250.0,
margin: const EdgeInsets.only(top: 50.0),
child: Wrap(
spacing: 15.0,
runSpacing: 15.0,
alignment: WrapAlignment.center,
children: List.generate(keyNumbers.length, (index) {
return Material(
type: MaterialType.transparency,
child: Ink(
height: 60.0,
width: 60.0,
decoration: BoxDecoration(
color: Colors.white24,
border: Border.all(color: Colors.white24, width: .5),
borderRadius: BorderRadius.circular(50.0),
),
child: InkWell(
borderRadius: BorderRadius.circular(50.0),
overlayColor: WidgetStateProperty.all(Colors.white38),
child: DefaultTextStyle(
style: const TextStyle(color: Colors.white, fontFamily: 'arial'),
child: Column(
children: [
const SizedBox(height: 10.0,),
Text(keyNumbers[index]['num'], style: const TextStyle(fontSize: 20.0, fontWeight: FontWeight.bold),),
Text(keyNumbers[index]['letter'], style: const TextStyle(fontSize: 10.0),),
],
),
),
onTap: () {
handleClickNum(keyNumbers[index]['num']);
},
),
),
);
})
),
),
data:image/s3,"s3://crabby-images/eff0c/eff0c5e159ed3118344c1e4e10d487603ee2b572" alt=""
公共布局模板Layout
桌面布局整体分为栅格式菜单+底部Dock菜单+拖拽悬浮球。
data:image/s3,"s3://crabby-images/6efc9/6efc987f3b4fd5ca04421cc4e3d38b5e62c24866" alt=""
@override
Widget build(BuildContext context) {
return Scaffold(
extendBodyBehindAppBar: widget.extendBodyBehindAppBar,
appBar: widget.appBar ?? AppBar(
forceMaterialTransparency: true,
backgroundColor: Colors.transparent,
foregroundColor: Colors.white,
toolbarHeight: 0,
),
body: Center(
child: Stack(
children: [
// 壁纸皮肤
if(widget.showBackground)
Obx(() => Container(
decoration: BoxDecoration(
image: DecorationImage(
image: AssetImage('${skinController.skinUrl}'),
fit: BoxFit.fill,
),
),
))
,
Flex(
direction: Axis.vertical,
crossAxisAlignment: CrossAxisAlignment.start,
children: [
// 顶部插槽
Container(
child: widget.header,
),
// 内容区域
Expanded(
child: widget.body ?? Container(),
),
// 底部插槽
Container(
child: widget.footer,
),
],
),
// 额外插槽
Container(
child: widget.extra,
),
],
),
),
);
}
data:image/s3,"s3://crabby-images/9462c/9462cd41fa64d0c88c1fd2d177664f5191b18a1b" alt=""
data:image/s3,"s3://crabby-images/d5fe2/d5fe2d93b8d9117ed423180bcd510e8053908e80" alt=""
data:image/s3,"s3://crabby-images/bc274/bc274480e4b5ec525f9c0913b3f4de6ab873e292" alt=""
data:image/s3,"s3://crabby-images/10c6f/10c6f7949438fff09c0368f626266b9fd194d564" alt=""
data:image/s3,"s3://crabby-images/f200d/f200d7dc0b661e83bdf273201626b124fac48b1c" alt=""
data:image/s3,"s3://crabby-images/127fe/127fed65845d464fdf77268b8433bf55536bbbdf" alt=""
data:image/s3,"s3://crabby-images/c8724/c8724eca6e94ff78de71352b8d2d98d38502ed89" alt=""
data:image/s3,"s3://crabby-images/55c3b/55c3bf4fe5a6d6b5b84fc3e4e6267eb1185c3ee0" alt=""
data:image/s3,"s3://crabby-images/09a4a/09a4a315e4f4b7fb860d6ecf1e1508e3b23c38d3" alt=""
data:image/s3,"s3://crabby-images/e3883/e3883145e66ee4fc7e12eb34efffa81689a4c334" alt=""
data:image/s3,"s3://crabby-images/42473/424733906d59ab65bf8322977d1af3adb0d01daf" alt=""
flutter栅格式桌面os菜单
data:image/s3,"s3://crabby-images/ab7ac/ab7acaef97fd72dc43d51b0d50eee27a28976305" alt=""
data:image/s3,"s3://crabby-images/28270/28270e6535bb9c6e54262188e57c0077ed00034e" alt=""
桌面os菜单配置项
/*
* ================== 桌面os菜单配置项 ==================
* [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] 点击图标回调函数
*/
data:image/s3,"s3://crabby-images/92d1c/92d1cf89c6438c49dc1e2a0bcee065ffe0d24cef" alt=""
data:image/s3,"s3://crabby-images/242ee/242eed23bbcc63c31a4c5de03c604598bd9a883f" alt=""
支持配置二级页面。
data:image/s3,"s3://crabby-images/bd21b/bd21beb7b29f7d3a5517cc8b853b8d3880ba66a2" alt=""
List deskMenus = [
...
{
'uid': '3u85fb90-12c4-11e1-840d-7b25c5ee775a',
'list': [
{'label': 'Flutter3.22', 'imgico': 'assets/images/flutter.png', 'link': 'https://flutter.dev/'},
{'label': 'Dart中文官方文档', 'imgico': 'assets/images/dart.png', 'link': 'https://dart.cn/'},
...
{'label': '日历', 'imgico': const Calendar1x1(), 'type': 'widget', 'path': '/calendar', 'background': const Color(0xffffffff),},
{'label': '首页', 'imgico': const Icon(Icons.home_outlined), 'type': 'icon', 'path': '/home'},
{'label': '工作台', 'imgico': const Icon(Icons.poll_outlined), 'type': 'icon', 'path': '/workplace'},
{
'label': '组件',
'children': [
{'label': '组件', 'imgico': 'assets/images/svg/component.svg', 'path': '/component'},
...
]
},
{
'label': '管理中心',
'children': [
{'label': '个人主页', 'imgico': 'assets/images/svg/my.svg', 'path': '/ucenter'},
...
]
},
{
'label': '编程开发',
'children': [
{'label': 'Github', 'imgico': 'assets/images/svg/github.svg', 'background': const Color(0xff607d8b),},
{'label': 'Flutter', 'imgico': 'assets/images/flutter.png', 'background': const Color(0xFFDAF2FA),},
{'label': 'ChatGPT', 'imgico': 'assets/images/svg/chatgpt.svg', 'background': const Color(0xFF15A17F),},
...
]
},
{
'label': '关于', 'imgico': const Icon(Icons.info), 'type': 'icon',
'onClick': () => {
...
}
},
{
'label': '公众号', 'imgico': const Icon(Icons.qr_code), 'type': 'icon',
'onClick': () => {
...
}
},
]
}
...
];
由于该栅格桌面系统涉及到的知识点蛮多的,就不展开详细的一 一介绍了。希望以上分享对大家有所帮助哈~
最后附上两个实例项目
https://www.cnblogs.com/xiaoyan2017/p/18165578
https://www.cnblogs.com/xiaoyan2017/p/18092224
data:image/s3,"s3://crabby-images/b6c55/b6c55cffc672a6be9040ccd1695e774faac681a0" alt=""