前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
MCP广场
社区首页 >专栏 >《Flutter》-- 9.路由与导航

《Flutter》-- 9.路由与导航

作者头像
爱学习的程序媛
发布于 2022-04-07 08:33:56
发布于 2022-04-07 08:33:56
1.1K00
代码可运行
举报
文章被收录于专栏:学习/读书笔记学习/读书笔记
运行总次数:0
代码可运行

参阅书籍:

Flutter跨平台开发入门与实践》-- 向治洪(著)

9. 路由与导航
9.1 路由基础
9.1.1 基本概念

在前端应用中,页面又称路由,是屏幕或应用程序页面的抽象。

Flutter的路由管理和导航借鉴了前端和客户端中的设计思路,提供了Route和Navigator对路由进行统一管理。

Route是页面的一个抽象概念,可以用它创建界面、接收参数以及响应Navigator的打开与关闭;Navigator用于管理和维护路由栈,打开路由页面即执行入栈操作,关闭路由页面即执行出栈操作。

Navigator组件的常用操作方法:

1)push():将给定的路由页面放到路由栈里面,返回值是一个Future对象,用于接收路由出栈时的返回数据;

2)pop():将位于栈顶的路由从路由栈移除,返回结果为路由关闭时上一个页面所需的数据。

在Flutter开发中,根据是否需要提前注册路由标识符,路由管理可以分为基本路由和命名路由两种。

9.1.2 基本路由

基本路由无需提前注册,在页面切换时需要手动构造页面的实例,使用起来相对简单灵活,适用于应用中页面不多的场景。

如果要打开一个新的页面,需要创建一个MaterialPageRoute对象实例,然后调用Navigator.push();如果要返回上一个页面,调用Navigator.pop()。

MaterialPageRoute是Flutter提供的路由模板,是PageRoute的子类,定义了路由创建及切换时过渡动画的相关接口和属性,并自带页面切换动画。

示例代码:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
import 'package:flutter/material.dart';

void main() => runApp(RoutePage());

class RoutePage extends StatelessWidget {
  @override
  Widget build(BuildContext context) {
    return MaterialApp(
      title: '路由--基本路由',
      home: FirstPage(),
    );
  }
}

class FirstPage extends StatelessWidget {
  @override
  Widget build(BuildContext context) {
    return Scaffold(
      appBar: AppBar(title: Text('第一个页面')),
      body: Center(
        child: RaisedButton(
          child: Text('跳转到第二个页面'),
          onPressed: () => Navigator.push(
            context, 
            MaterialPageRoute(builder: (context) => SecondPage()),
          ),
        ),
      ),
    );
  }
}

class SecondPage extends StatelessWidget {
  @override
  Widget build(BuildContext context) {
    return Scaffold(
      appBar: AppBar(title: Text('第二个页面')),
      body: Center(
        child: RaisedButton(
          child: Text('返回上一个页面'),
          onPressed: () => Navigator.pop(context),
        ),
      ),
    );
  }
}

示例效果:

9.1.3 命名路由

命名路由需要提前注册路由页面标识符,在页面切换时通过路由标识符打开一个新的路由页面。

在Flutter中,路由表是一个Map<String, WidgetBuilder>结构,其中第一个参数对应页面的别名,第二个参数对应页面。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
MaterialApp(
  routes: {//注册路由
    'first': (context) => FirstPage(),
    'second': (context) => SecondPage(),
  },
  initialRoute: 'firse',//初始路由页面
)

在路由表中注册好页面后,在其他页面中通过Navigator.pushNamed()来打开注册的页面。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
Navigator.pushNamed(context, 'second');

Flutter提供了一个onUnknownRoute属性,用来在注册路由表时对未知的路由标识符进行统一的页面跳转处理。

示例代码:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
import 'package:flutter/material.dart';

void main() => runApp(RoutePage());

class RoutePage extends StatelessWidget {
  @override
  Widget build(BuildContext context) {
    return MaterialApp(
      title: '路由--命名路由',
      home: FirstPage(),
      routes: {
        '/first': (context) => FirstPage(),
        '/second': (context) => SecondPage(),
      },
      initialRoute: '/first',
      onUnknownRoute: (RouteSettings setting) => MaterialPageRoute(builder: (context) => UnknownPage()),
    );
  }
}

class FirstPage extends StatelessWidget {
  @override
  Widget build(BuildContext context) {
    return Scaffold(
      appBar: AppBar(title: Text('第一个页面')),
      body: Center(
        child: RaisedButton(
          child: Text('跳转到第二个页面'),
          onPressed: () => Navigator.pushNamed(context, '/scond'),
        ),
      ),
    );
  }
}

class SecondPage extends StatelessWidget {
  @override
  Widget build(BuildContext context) {
    return Scaffold(
      appBar: AppBar(title: Text('第二个页面')),
      body: Center(
        child: RaisedButton(
          child: Text('返回上一个页面'),
          onPressed: () => Navigator.pushNamed(context, '/first'),
        ),
      ),
    );
  }
}

class UnknownPage extends StatelessWidget {
  @override
  Widget build(BuildContext context) {
    return Scaffold(
      appBar: AppBar(title: Text('错误路由')),
      body: Center(
        child: Text('跳转路由错误!'),
      )
    );
  }
}

示例效果:

9.1.5 路由传参

可以在打开路由时传递参数,在目标页面通过ModalRoute的RouteSettings获取页面传递的参数。

如果需要返回上一个页面时回传参数,可以在使用push()打开目标页面时使用then()监听目标页面的返回值。

示例代码:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
import 'package:flutter/material.dart';

void main() => runApp(RoutePage());

class RoutePage extends StatelessWidget {
  @override
  Widget build(BuildContext context) {
    return MaterialApp(
      title: '路由--路由传参',
      home: FirstPage(),
      routes: {
        '/first': (context) => FirstPage(),
        '/second': (context) => SecondPage(),
      },
    );
  }
}

class FirstPage extends StatefulWidget {
  @override
  State<StatefulWidget> createState() {
    return FirstPageState();
  }
}

class FirstPageState extends State<FirstPage> {
  String result = '';
  @override
  Widget build(BuildContext context) {
    return Scaffold(
      appBar: AppBar(
        title: Text('第一个页面'),
        centerTitle: true,
      ),
      body: Center(
        child: Column(
          children: <Widget>[
            Text(
              'from second page: ' + result,
              style: TextStyle(fontSize: 20)
            ),
            RaisedButton(
              child: Text('跳转'),
              onPressed: () {
                Navigator.of(context)
                         .pushNamed('/second', arguments: 'from first page')
                         .then((msg) => setState(() => result = msg)); 
              },
            )
          ],
        )
      ),
    );
  }
}

class SecondPage extends StatelessWidget {
  @override
  Widget build(BuildContext context) {
    String msg = ModalRoute.of(context).settings.arguments as String;
    return Scaffold(
      appBar: AppBar(
        title: Text('第二个页面'),
        centerTitle: true,
      ),
      body: Center(
        child: Column(
          children: <Widget>[
            Text(
              'from first screen: ' + msg,
              style: TextStyle(fontSize: 20)
            ),
            RaisedButton(
              child: Text('返回'),
              onPressed: () => Navigator.pop(context, 'from second page'),
            )
          ],
        )
      ),
    );
  }
}

示例效果:

9.2 路由栈

Flutter路由栈其实就是一个后进先出的线性表,路由栈管理本质上就是一个入栈和出栈的过程,入栈就是将页面放到路由栈的顶部,出栈则是从路由的顶部移除页面。

1)pushReplacementNamed

使用pushReplacement或pushReplacementNamed打开一个新页面时,路由栈顶部的页面会被当前页面所替换。

2)popAndPushNamed

popAndPushNamed的作用与pushReplacementNamed类似,打开一个新页面时,路由栈的栈顶页面会被当前页面替换。

不同的是,popAndPushNamed会同时执行出栈和入栈动画,而pushReplacementNamed只执行入栈动画。

3)pushNamedAndRemoveUntil

pushNamedAndRemoveUntil和pushAndRemoveUntil的作用类似,主要用于向路由栈中添加一个新页面,并删除路由栈中所有之前的页面。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
Navigator.pushNamedAndRemoveUntil(context, ‘page_c’, (Route router) => false);

如果不需要清空之前的页面,可以将表达式设置为true,即(Route router) => true。

除了用于删除路由栈中所有之前的页面外,pushNamedAndRemoveUntil还可以用来删除指定个数的页面。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
Navigator.pushNamedAndRemoveUntil(context, ‘page_d’, ModalRoute.withName(‘page_b’));

上面的代码会打开一个新的页面page_d,同时删除page_b页面之上的所有页面。

如果要移除路由栈中某个指定的页面,可以使用Navigator.removeRoute()或者Navigator.removeRouteBelow()。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
Navigator.removeRoute(context, MaterialPageRoute(builder: (context) => PageC()));
//或者
Navigator.removeRouteBelow(context, MaterialPageRoute(builder: (context) => PageC()));

上面的代码会从路由栈中移除PageC页面。

4)popUntil

popUntil的作用与pushNamedAndRemoveUntil类似,主要用于清除指定页面之上的所有页面。popUntil没有执行push操作,直接执行pop操作,直到返回到指定的页面。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
Navigator.popUntil(context, ModalRoute.withName(‘page_a’));

上面的代码会清除page_a页面之上的所有页面。

9.3 自定义路由

如果要修改默认的路由转场动画,就需要做一些自定义开发。在Flutter中,自定义路由需要用到PageRouteBuilder类,PageRouteBuilder是所有自定义路由的基类。

PageRouteBuilder的构造函数如下:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
PageRouteBuilder(
  RouteSettings settings,
  @required this.pageBuilder,//用来创建所需要跳转的路由页面
  this.transitionsBuilder = _defaultTransitionsBuilder,//用自定义转场动画
  this.transitionDuration = const Duration(milliseconds: 300),//自定义转场动画的执行时间
  this.opaque = true,//是否需要遮挡整个页面
  this.barrierDismissible = false,
  this.barrierColor,
  this.barrierLabel,
  this.maintainState = true,
  bool fullscreenDialog = false, 
)

示例代码:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
import 'package:flutter/material.dart';

void main() => runApp(RoutePage());

class RoutePage extends StatelessWidget {
  @override
  Widget build(BuildContext context) {
    return MaterialApp(
      title: '路由--自定义路由',
      home: FirstPage(),
    );
  }
}

class FirstPage extends StatelessWidget {
  @override
  Widget build(BuildContext context) {
    return Scaffold(
      appBar: AppBar(
        title: Text('第一个页面'),
        centerTitle: true,
      ),
      body: Center(
        child: RaisedButton(
              child: Text('跳转'),
              onPressed: () {
                Navigator.of(context).push(CustomRoute(SecondPage()));          
              },
            )
      ),
    );
  }
}

class SecondPage extends StatelessWidget {
  @override
  Widget build(BuildContext context) {
    String msg = ModalRoute.of(context).settings.arguments as String;
    return Scaffold(
      appBar: AppBar(
        title: Text('第二个页面'),
        centerTitle: true,
      ),
      body: Center(
        child: RaisedButton(
              child: Text('返回'),
              onPressed: () => Navigator.pop(context),
            )
      ),
    );
  }
}


class CustomRoute extends PageRouteBuilder {
  final Widget widget;
  CustomRoute(this.widget): super(
    transitionDuration: Duration(seconds: 1),
    pageBuilder: (BuildContext context, Animation<double> animation1, Animation<double> animation2) {
      return widget;
    },
    transitionsBuilder: (BuildContext context, Animation<double> animation1, Animation<double> animation2, Widget child) {
      return SlideTransition(
        position: Tween<Offset>(
          begin: Offset(1.0, 0.0), end: Offset(0.0, 0.0)
        ).animate(CurvedAnimation(parent: animation1, curve: Curves.fastOutSlowIn)),
        child: child
      );
    }
  );
}

除了使用Flutter提供的路由方案外,还可以使用第三方路由框架来实现页面的管理和跳转。Fluro是一款优秀的Flutter企业级路由框架,非常适合中大型项目,它具有层次分明、条理化、方便扩展和便于整体管理路由等特点。

本文参与 腾讯云自媒体同步曝光计划,分享自微信公众号。
原始发表:2021-04-28,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 爱学习的程序媛 微信公众号,前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
暂无评论
推荐阅读
编辑精选文章
换一批
通用的消息队列(redis,kafka,rabbitmq)--生产者篇
网上有很多消息队列的中间件,如redis,kafka,rabbitmq,这些都很强大 但用起来,每个的用法都不一样,有没有一种办法,我只需要实现一种方法,就能随意使用哪个中间件都可以呢. 我这边设计如下:
星痕
2020/06/19
6390
redis实现消息队列
消息队列一般都会想到kafka,rabbitmq,Rockermq, 其实,给你印像做缓存的Redis也是能做消息队列.
星痕
2020/06/02
9160
SpringBoot非官方教程 | 第十四篇:在springboot中用redis实现消息队列
方志朋
2017/12/29
1.1K0
Spring-Data-Redis动态订阅发布
普通redis订阅,是以用container做容器,配置类配置文件方式直接在spring init的时候进行加载,不能进行动态添加。在程序运行时修改不起作用。
Diuut
2022/11/22
9330
Redis:发布(pub)与订阅(sub)实战
Redis发布订阅(Pub/Sub)是Redis提供的一种消息传递机制,它使用“发布者-订阅者”(publisher-subscriber)模式来处理消息传递。在这种模式下,发布者将消息发布到一组订阅者中,而无需关心谁是订阅者,也不需要知道订阅者是否收到了消息。
不惑
2023/11/17
2.2K1
Redis:发布(pub)与订阅(sub)实战
重学SpringBoot3-集成Redis(六)之消息队列
Redis 不仅是一个高效的缓存解决方案,也具备强大的消息队列功能。通过 Redis 的 发布/订阅(Pub/Sub) 机制,开发者可以轻松实现服务之间的通信和消息传递功能,而无需引入专门的消息队列工具。这篇文章将介绍如何通过 Spring Boot 3 和 Redis 实现消息队列的发布与订阅功能。
CoderJia
2024/10/18
4310
重学SpringBoot3-集成Redis(六)之消息队列
使用消息队列轻松实现 分布式 webSocket
简单的概括一下:如果我们的项目是分布式环境,登录的用户被Nginx的反向代理分配到多个不同服务器,那么在其中一个服务器建立了WebSocket连接的用户如何给在另外一个服务器上建立了WebSocket连接的用户发送消息呢?
码猿技术专栏
2023/09/07
6450
使用消息队列轻松实现 分布式 webSocket
基于Redis的消息发布和订阅(广播模式)
如下如所示, 定义了2个Linstener, 模拟2个应用监听同一个通道. 根据发送和接收的数据类型我们可以选择合适的数据序列化和反序列化方式, 默认序列化方式为RedisSerializer.java(). 对于普通的Bean来说使用json()和java()的序列方式都可以.不同点在于:
悦stack
2023/06/30
1.1K0
Redis发布订阅模式
Redis自身提供了发布/订阅(publish/subscribe)模式。实现方式大致流程如下图:
ruochen
2021/11/22
9760
springboot集成redis实现消息发布订阅模式
1,application.properties配置redis以及连接池 #redis spring.redis.host=localhost spring.redis.port=6379 #spring.redis.password= spring.redis.database=1 spring.redis.pool.max-active=8 spring.redis.pool.max-wait=-1 spring.redis.pool.max-idle=500 spring.redis.pool.min
Java编程指南
2019/08/02
1K0
SpringBoot开发案例之整合mail队列进阶篇
上一篇文章,我们为了解决实际场景中遇到的问题,使得其更像一个安全高效的邮件服务平台,我们引入了LinkedBlockingQueue队列对邮件发送进行流量削锋、间隔发送以及重复内容检测。
小柒2012
2018/03/30
1.3K3
SpringBoot开发案例之整合mail队列进阶篇
Spring JMS---三种消息监听器
链接:https://blog.csdn.net/u012562943/article/details/51424232
用户5224393
2019/08/13
2.5K0
【附源码】SpringBoot集成Redis消息订阅发布
Receiver这是一个定义了一个接收消息的方法的类。当你把这个类作为一个消息监听器来注册后,你可以自定义消息接收的方法名。本例中采用“receiveMessage”作为接收消息的方法。
用户5224393
2019/08/13
8550
Spring Boot使用Redis消息队列
业务系统经常需要用到MQ消息队列,但是又不希望引入一个完整的中间件,比如RocketMQ,RabbitMQ,因为会增加接入成本和运维成本。所以当业务量不是很大,且一致性要求不是很强的场景下,可以选择Redis,使用其pub/sub机制作为消息队列的实现 添加依赖 ---- <dependencies> <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-
十毛
2019/05/17
2.4K0
使用消息队列实现 分布式 webSocket
陈某的《Spring Cloud Alibaba实战项目》 视频教程已经录完了,涉及到Alibaba的各种中间件、OAuth2微服务认证鉴权、全链路灰度发布、分布式事务实战,戳这里--->Spring Cloud Alibaba 实战 视频专栏 开放订阅~
码猿技术专栏
2023/05/01
1.4K0
使用消息队列实现 分布式 webSocket
Springboot2使用redis 进行消息的订阅发布
我们都知道redis 也有发布订阅模式, 但是使用的比较少。 并且redis的发布订阅不会持久化落入磁盘。总的来说就是不可靠。
分享干货的你
2021/04/23
1.6K1
SpringBoot集成消息队列
最近在对公司开发框架进行优化,框架内涉及到多处入库的日志记录,例如登录日志/操作日志/访问日志/业务执行日志,集成在业务代码中耦合度较高且占用业务操作执行时间,所以准备集成相关消息队列进行代码解耦
code2roc
2023/07/19
3580
springboot2.0整合redis_灰度发布的方式
所有的订阅消息,都需要在这里进行注册绑定,new PatternTopic(“user”),表示发布的主题信息
全栈程序员站长
2022/11/09
5670
springboot2.0整合redis_灰度发布的方式
kafka手动监听主题
很多人作kafka消费时,都快速的使用注解@KafkaListener进行监听。 但我现在有个需求,是要动态的手动监听。 实现代码如下: 1.手动编写监听类
星痕
2020/05/18
8260
Redis 订阅模式详解
注意:发布返回的是订阅者数量,发布的消息不会持久化,没有订阅者时候,发布消息会丢失,当在发布消息之后对channel进行订阅不会收到之前发布的消息。
ruochen
2021/12/07
9520
相关推荐
通用的消息队列(redis,kafka,rabbitmq)--生产者篇
更多 >
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档
本文部分代码块支持一键运行,欢迎体验
本文部分代码块支持一键运行,欢迎体验