Flutter 导航路由:构建流畅的应用导航体验

Flutter 导航路由:构建流畅的应用导航体验

代码如诗,导航如画。让我们用 Flutter 路由的强大能力,构建出既流畅又直观的应用导航系统。

什么是 Flutter 导航?

Flutter 导航是指在不同页面(或称为路由)之间进行切换的机制。良好的导航系统是应用用户体验的重要组成部分,它决定了用户如何在应用中浏览和找到所需内容。

基础导航

1. 使用 Navigator

import 'package:flutter/material.dart'; // 跳转到新页面 Navigator.push( context, MaterialPageRoute(builder: (context) => SecondPage()), ); // 返回上一页 Navigator.pop(context); // 替换当前页面 Navigator.pushReplacement( context, MaterialPageRoute(builder: (context) => SecondPage()), ); // 清除所有页面并跳转到新页面 Navigator.pushAndRemoveUntil( context, MaterialPageRoute(builder: (context) => HomePage()), (route) => false, ); 

2. 命名路由

// 在 MaterialApp 中定义路由 class MyApp extends StatelessWidget { @override Widget build(BuildContext context) { return MaterialApp( title: 'Flutter Demo', initialRoute: '/', routes: { '/': (context) => HomePage(), '/second': (context) => SecondPage(), '/third': (context) => ThirdPage(), }, ); } } // 使用命名路由导航 Navigator.pushNamed(context, '/second'); // 返回上一页 Navigator.pop(context); // 替换当前页面 Navigator.pushReplacementNamed(context, '/second'); // 清除所有页面并跳转到新页面 Navigator.pushNamedAndRemoveUntil(context, '/', (route) => false); 

3. 传递参数

// 跳转到新页面并传递参数 Navigator.pushNamed( context, '/second', arguments: {'id': 1, 'name': 'Flutter'}, ); // 在新页面接收参数 class SecondPage extends StatelessWidget { @override Widget build(BuildContext context) { final Map<String, dynamic> args = ModalRoute.of(context)!.settings.arguments as Map<String, dynamic>; final int id = args['id']; final String name = args['name']; return Scaffold( appBar: AppBar(title: Text('Second Page')), body: Center( child: Text('ID: $id, Name: $name'), ), ); } } 

高级路由管理

1. 使用 onGenerateRoute

class MyApp extends StatelessWidget { @override Widget build(BuildContext context) { return MaterialApp( title: 'Flutter Demo', initialRoute: '/', onGenerateRoute: (settings) { switch (settings.name) { case '/': return MaterialPageRoute(builder: (context) => HomePage()); case '/second': final args = settings.arguments as Map<String, dynamic>; return MaterialPageRoute( builder: (context) => SecondPage( id: args['id'], name: args['name'], ), ); default: return MaterialPageRoute(builder: (context) => NotFoundPage()); } }, ); } } 

2. 自定义路由动画

// 自定义页面路由 class CustomPageRoute<T> extends PageRouteBuilder<T> { final Widget child; CustomPageRoute({required this.child}) : super( pageBuilder: (context, animation, secondaryAnimation) => child, transitionsBuilder: (context, animation, secondaryAnimation, child) { const begin = Offset(1.0, 0.0); const end = Offset.zero; const curve = Curves.easeInOut; var tween = Tween(begin: begin, end: end).chain(CurveTween(curve: curve)); var offsetAnimation = animation.drive(tween); return SlideTransition( position: offsetAnimation, child: child, ); }, ); } // 使用自定义路由 Navigator.push( context, CustomPageRoute(child: SecondPage()), ); 

3. 使用 PageRouteBuilder

Navigator.push( context, PageRouteBuilder( pageBuilder: (context, animation, secondaryAnimation) => SecondPage(), transitionsBuilder: (context, animation, secondaryAnimation, child) { return FadeTransition( opacity: animation, child: child, ); }, transitionDuration: Duration(milliseconds: 500), ), ); 

路由守卫和拦截

1. 登录验证

class AuthGuard { static bool isLoggedIn = false; static Route<dynamic>? onGenerateRoute(RouteSettings settings) { // 需要登录的页面 final authRoutes = ['/profile', '/settings']; if (authRoutes.contains(settings.name) && !isLoggedIn) { return MaterialPageRoute( builder: (context) => LoginPage(), settings: RouteSettings(name: '/login'), ); } // 正常路由 switch (settings.name) { case '/': return MaterialPageRoute(builder: (context) => HomePage()); case '/login': return MaterialPageRoute(builder: (context) => LoginPage()); case '/profile': return MaterialPageRoute(builder: (context) => ProfilePage()); default: return MaterialPageRoute(builder: (context) => NotFoundPage()); } } } 

2. 路由拦截器

class RouteInterceptor extends NavigatorObserver { @override void didPush(Route<dynamic> route, Route<dynamic>? previousRoute) { super.didPush(route, previousRoute); print('Pushed: ${route.settings.name}'); // 可以在这里进行埋点、权限检查等 } @override void didPop(Route<dynamic> route, Route<dynamic>? previousRoute) { super.didPop(route, previousRoute); print('Popped: ${route.settings.name}'); } } // 在 MaterialApp 中使用 MaterialApp( navigatorObservers: [RouteInterceptor()], // ... ) 

底部导航栏

1. 基础底部导航

class HomePage extends StatefulWidget { @override _HomePageState createState() => _HomePageState(); } class _HomePageState extends State<HomePage> { int _selectedIndex = 0; final List<Widget> _pages = [ HomeTab(), SearchTab(), ProfileTab(), ]; void _onItemTapped(int index) { setState(() { _selectedIndex = index; }); } @override Widget build(BuildContext context) { return Scaffold( body: _pages[_selectedIndex], bottomNavigationBar: BottomNavigationBar( items: const <BottomNavigationBarItem>[ BottomNavigationBarItem( icon: Icon(Icons.home), label: '首页', ), BottomNavigationBarItem( icon: Icon(Icons.search), label: '搜索', ), BottomNavigationBarItem( icon: Icon(Icons.person), label: '我的', ), ], currentIndex: _selectedIndex, selectedItemColor: Colors.blue, onTap: _onItemTapped, ), ); } } 

2. 使用 IndexedStack 保持页面状态

class HomePage extends StatefulWidget { @override _HomePageState createState() => _HomePageState(); } class _HomePageState extends State<HomePage> { int _selectedIndex = 0; final List<Widget> _pages = [ HomeTab(), SearchTab(), ProfileTab(), ]; @override Widget build(BuildContext context) { return Scaffold( body: IndexedStack( index: _selectedIndex, children: _pages, ), bottomNavigationBar: BottomNavigationBar( items: const <BottomNavigationBarItem>[ BottomNavigationBarItem( icon: Icon(Icons.home), label: '首页', ), BottomNavigationBarItem( icon: Icon(Icons.search), label: '搜索', ), BottomNavigationBarItem( icon: Icon(Icons.person), label: '我的', ), ], currentIndex: _selectedIndex, selectedItemColor: Colors.blue, onTap: (index) { setState(() { _selectedIndex = index; }); }, ), ); } } 

抽屉导航

class HomePage extends StatelessWidget { @override Widget build(BuildContext context) { return Scaffold( appBar: AppBar(title: Text('首页')), drawer: Drawer( child: ListView( padding: EdgeInsets.zero, children: <Widget>[ DrawerHeader( decoration: BoxDecoration( color: Colors.blue, ), child: Text( '菜单', style: TextStyle( color: Colors.white, fontSize: 24, ), ), ), ListTile( leading: Icon(Icons.home), title: Text('首页'), onTap: () { Navigator.pop(context); Navigator.pushNamed(context, '/'); }, ), ListTile( leading: Icon(Icons.settings), title: Text('设置'), onTap: () { Navigator.pop(context); Navigator.pushNamed(context, '/settings'); }, ), ], ), ), body: Center(child: Text('首页内容')), ); } } 

最佳实践

  1. 使用命名路由:便于管理和维护
  2. 参数类型安全:使用强类型传递参数
  3. 路由守卫:在关键页面进行权限验证
  4. 动画一致性:保持应用内导航动画的一致性
  5. 状态保持:使用 IndexedStack 保持页面状态
  6. 错误处理:处理路由不存在的情况

实践案例:完整的导航系统

import 'package:flutter/material.dart'; void main() { runApp(MyApp()); } class MyApp extends StatelessWidget { @override Widget build(BuildContext context) { return MaterialApp( title: 'Flutter Navigation Demo', theme: ThemeData( primarySwatch: Colors.blue, ), initialRoute: '/', routes: { '/': (context) => HomePage(), '/second': (context) => SecondPage(), '/third': (context) => ThirdPage(), }, onUnknownRoute: (settings) { return MaterialPageRoute( builder: (context) => NotFoundPage(), ); }, ); } } class HomePage extends StatelessWidget { @override Widget build(BuildContext context) { return Scaffold( appBar: AppBar( title: Text('首页'), ), body: Center( child: Column( mainAxisAlignment: MainAxisAlignment.center, children: [ Text( '首页', style: TextStyle(fontSize: 24), ), SizedBox(height: 20), ElevatedButton( onPressed: () { Navigator.pushNamed( context, '/second', arguments: {'message': '来自首页的问候'}, ); }, child: Text('跳转到第二页'), ), ], ), ), ); } } class SecondPage extends StatelessWidget { @override Widget build(BuildContext context) { final args = ModalRoute.of(context)!.settings.arguments as Map<String, dynamic>; final message = args['message']; return Scaffold( appBar: AppBar( title: Text('第二页'), ), body: Center( child: Column( mainAxisAlignment: MainAxisAlignment.center, children: [ Text( '第二页', style: TextStyle(fontSize: 24), ), SizedBox(height: 10), Text( message, style: TextStyle(fontSize: 16, color: Colors.grey), ), SizedBox(height: 20), ElevatedButton( onPressed: () { Navigator.pushNamed(context, '/third'); }, child: Text('跳转到第三页'), ), SizedBox(height: 10), ElevatedButton( onPressed: () { Navigator.pop(context); }, child: Text('返回上一页'), ), ], ), ), ); } } class ThirdPage extends StatelessWidget { @override Widget build(BuildContext context) { return Scaffold( appBar: AppBar( title: Text('第三页'), ), body: Center( child: Column( mainAxisAlignment: MainAxisAlignment.center, children: [ Text( '第三页', style: TextStyle(fontSize: 24), ), SizedBox(height: 20), ElevatedButton( onPressed: () { Navigator.pushNamedAndRemoveUntil( context, '/', (route) => false, ); }, child: Text('返回首页'), ), ], ), ), ); } } class NotFoundPage extends StatelessWidget { @override Widget build(BuildContext context) { return Scaffold( appBar: AppBar( title: Text('页面未找到'), ), body: Center( child: Column( mainAxisAlignment: MainAxisAlignment.center, children: [ Icon( Icons.error_outline, size: 64, color: Colors.red, ), SizedBox(height: 16), Text( '404', style: TextStyle( fontSize: 48, fontWeight: FontWeight.bold, ), ), SizedBox(height: 8), Text( '页面未找到', style: TextStyle(fontSize: 18), ), SizedBox(height: 20), ElevatedButton( onPressed: () { Navigator.pushReplacementNamed(context, '/'); }, child: Text('返回首页'), ), ], ), ), ); } } 

总结

Flutter 的导航系统提供了丰富的功能来构建流畅的应用导航体验。通过合理使用命名路由、自定义动画、路由守卫等技术,我们可以创建出既美观又实用的导航系统。

导航不仅仅是页面的切换,更是用户体验的引导。让我们用 Flutter 路由的强大能力,构建出令人惊叹的应用导航系统,展现前端技术的无限可能。

Read more

VideoAgentTrek-ScreenFilter创新场景:VR录屏中虚拟屏幕边界检测

VideoAgentTrek-ScreenFilter创新场景:VR录屏中虚拟屏幕边界检测 1. 引言:当VR录屏遇到屏幕检测难题 想象一下,你正在体验一款沉浸式的VR游戏,或者在进行一场虚拟现实会议。结束后,你想把这段精彩的体验录下来分享给朋友。但当你回看录屏时,发现画面里不仅有VR应用本身的内容,还混杂着电脑桌面、任务栏、甚至其他无关的窗口边框。这些“屏幕外的屏幕”破坏了沉浸感,也让视频显得杂乱不专业。 这就是VR内容创作者和开发者经常遇到的痛点。传统的录屏软件只能录制整个显示器区域,无法智能识别并聚焦在真正的VR应用窗口上。手动裁剪不仅费时费力,而且在视频时长较长或窗口位置变化时,几乎无法实现精准处理。 今天要介绍的 VideoAgentTrek-ScreenFilter,正是为解决这个问题而生。它不是一个普通的屏幕检测工具,而是专门针对“录屏中的屏幕”这一特殊场景进行优化的智能解决方案。无论是静态的截图,还是动态的录屏视频,它都能准确识别出画面中的屏幕边界,为后续的智能裁剪、内容聚焦或隐私处理提供关键数据。 2. VideoAgentTrek-ScreenFilt

CFAR 目标检测算法详解(附 MATLAB 示例)

CFAR 目标检测算法详解(附 MATLAB 示例)

CFAR 目标检测算法,毫米波雷达工程师必须掌握的第一种检测算法 一、为什么雷达需要“目标检测算法”? 在毫米波雷达中,我们最终想知道的不是“信号长什么样”,而是: 哪里有目标?目标有多少?哪些是噪声? 然而,雷达接收到的信号永远是下面三者的混合: 1. 真实目标回波 2. 环境杂波(地面、墙面、人体、车辆反射) 3. 系统噪声(热噪声、量化噪声等) 在经过 ADC → FFT → 距离谱 / 多普勒谱 后,你会看到大量起伏的谱线。 问题来了: 在一个噪声水平不断变化的环境中,如何“公平、稳定”地判断某个峰值是不是目标? 这正是 CFAR(Constant False Alarm Rate,恒虚警率)算法存在的意义。 二、CFAR

无人机遥感航拍巡检数据集 无人机遥感图像识别 无人机视角山区泥石流和滑坡图像识别数据集-数据集第10067期

无人机遥感航拍巡检数据集 无人机遥感图像识别 无人机视角山区泥石流和滑坡图像识别数据集-数据集第10067期

滑坡检测数据集核心信息介绍 ** 这个滑坡检测数据集主要用于目标检测任务,整体数据规模和细节都比较明确。从数量上看,数据集总共包含 1660 张图像, 往期热门主题 主题搜两字"关键词"直达 代码数据获取: 获取方式:***文章底部卡片扫码获取*** 覆盖了YOLO相关项目、OpenCV项目、CNN项目等所有类别, 覆盖各类项目场景(包括但不限于以下----欢迎咨询定制): 项目名称项目名称基于YOLO+deepseek 智慧农业作物长势监测系统基于YOLO+deepseek 人脸识别与管理系统基于YOLO+deepseek 无人机巡检电力线路系统基于YOLO+deepseek PCB板缺陷检测基于YOLO+deepseek 智慧铁路轨道异物检测系统基于YOLO+deepseek 102种犬类检测系统基于YOLO+deepseek 人脸面部活体检测基于YOLO+deepseek 无人机农田病虫害巡检系统基于YOLO+deepseek 水稻害虫检测识别基于YOLO+deepseek 安全帽检测系统基于YOLO+deepseek 智慧铁路接触网状态检测系统基于YOLO+

项目介绍 MATLAB实现基于LSTM-DRL 长短期记忆网络(LSTM)结合深度强化学习(DRL)进行无人机三维路径规划的详细项目实例(含模型描述及部分示例代码) 还请多多点一下关注 加油 谢谢 你

项目介绍 MATLAB实现基于LSTM-DRL 长短期记忆网络(LSTM)结合深度强化学习(DRL)进行无人机三维路径规划的详细项目实例(含模型描述及部分示例代码) 还请多多点一下关注 加油 谢谢 你

MATLAB实现基于LSTM-DRL 长短期记忆网络(LSTM)结合深度强化学习(DRL)进行无人机三维路径规划的详细项目实例 更多详细内容可直接联系博主本人   或者访问对应标题的完整博客或者文档下载页面(含完整的程序,GUI设计和代码详解) 随着无人机(UAV, Unmanned Aerial Vehicle)技术的飞速发展,其在军事侦察、环境监测、灾害救援、物流运输、城市管理等众多领域的应用日益广泛。三维路径规划技术作为无人机自主导航与智能决策的核心支撑,已经成为研究的热点与难点。三维空间下的路径规划不仅需要考虑障碍物的避让和环境复杂性的适应,还要实现能耗最优、飞行平稳、航迹安全和任务高效完成。传统路径规划方法如A*、Dijkstra、RRT等在二维场景下表现良好,但面对动态多变、障碍复杂的三维空间时,往往存在计算量大、收敛速度慢、易陷入局部最优等问题,难以满足实际应用需求。人工智能的迅速发展为无人机路径规划提供了新的解决思路,其中,深度强化学习(DRL, Deep Reinforcement Learning)凭借其端到端的自主决策能力,在动态环境中的表现逐渐突出。而