老孟导读:Flutter中路由是很是重要的部分,任何一个应用程序都离不开路由管理,此文讲解路由相关方法的使用和路由堆栈的变化。git
Flutter 路由管理中有两个很是重要的概念:微信
假设如今有2个页面 A 和 B,A中有一个按钮,点击跳转到 B 页面,A 页面代码:less
class APage extends StatelessWidget { @override Widget build(BuildContext context) { return Container( alignment: Alignment.center, child: RaisedButton( child: Text('A 页面'), onPressed: () { Navigator.of(context).push(MaterialPageRoute(builder: (context) { return BPage(); })); }, ), ); } }
B 页面代码:async
class BPage extends StatelessWidget { @override Widget build(BuildContext context) { return Scaffold( body: Container( alignment: Alignment.center, child: RaisedButton( child: Text('B 页面'), onPressed: () { }, ), ), ); } }
当应用程序位于A页面时,路由堆栈中只有A,点击按钮跳转到B页面,路由堆栈中有 B 和 A,且 B 处于栈顶。ide
点击 B 页面的按钮返回到 A 页面,修改 B 页面按钮点击事件:函数
RaisedButton( child: Text('B 页面'), onPressed: () { Navigator.of(context).pop(); }, )
路由堆栈的变化:动画
上面案例的效果是从 B 页面跳转到 A 页面,那是否也可使用 push 方法?修改 B 页面按钮点击事件:ui
RaisedButton( child: Text('B 页面'), onPressed: () { Navigator.of(context).push(MaterialPageRoute(builder: (context) { return APage(); })); }, )
从效果上看也能够跳转到 A 页面,路由堆栈:this
那是否可使用 push 代替 pop 呢? 答案确定是不能够的,3d
上面案例若是点击 A 页面按钮直接调用 pop 会如何?
RaisedButton( child: Text('A 页面'), onPressed: () { Navigator.of(context).pop(); }, )
在 A 页面时路由堆栈中只有 A,调用 pop 后,路由堆栈变化:
此时路由堆栈为空,没有可显示的页面,应用程序将会退出或者黑屏,好的用户体验不该如此,此时可使用 maybePop,maybePop 只在路由堆栈有可弹出路由时才会弹出路由。
上面的案例在 A 页面执行maybePop:
RaisedButton( child: Text('A 页面'), onPressed: () { Navigator.of(context).maybePop(); }, )
点击后不会出现弹出路由,由于当前路由堆栈中只有 A,在 B页面执行maybePop,将会返回到 A 页面。
也能够经过 canPop 判断当前是否能够 pop:
RaisedButton( child: Text('B 页面'), onPressed: () { if(Navigator.of(context).canPop()){ Navigator.of(context).pop(); } }, )
pushNamed 是命名路由的方式,须要在 MaterialApp 中配置路由名称:
MaterialApp( title: 'Flutter Demo', routes: <String, WidgetBuilder>{ '/A': (context) => APage(), '/B': (context) => BPage(), }, home: Scaffold( body: APage(), ), )
从 A 跳转到 B:
RaisedButton( child: Text('A 页面'), onPressed: () { Navigator.of(context).pushNamed('/B'); }, )
有A、B、C 三个页面,A页面经过 pushNamed 跳转到 B:
RaisedButton( child: Text('A 页面'), onPressed: () { Navigator.of(context).pushNamed('/B'); }, )
B 经过 pushReplacementNamed 跳转到 C:
RaisedButton( child: Text('B 页面'), onPressed: () { Navigator.of(context).pushReplacementNamed('/C'); }, )
点击 C 页面按钮执行 pop:
RaisedButton( child: Text('C 页面'), onPressed: () { if(Navigator.of(context).canPop()){ Navigator.of(context).pop(); } }, )
点击 C 页面按钮直接返回到了 A 页面,而不是 B 页面,由于 B 页面使用 pushReplacementNamed 跳转,路由堆栈变化:
B 页面跳转到 C 页面,使用 popAndPushNamed:
RaisedButton( child: Text('B 页面'), onPressed: () { Navigator.of(context).popAndPushNamed('/C'); }, )
popAndPushNamed 路由堆栈和 pushReplacementNamed 是同样,惟一的区别就是 popAndPushNamed 有 B 页面退出动画。
popAndPushNamed 和 pushReplacementNamed 使当前页面不在路由堆栈中,因此经过 pop 没法返回此页面。
适用场景:
有以下场景,应用程序进入首页,点击登陆进入登陆页面,而后进入注册页面或者忘记密码页面...,登陆成功后进入其余页面,此时不但愿返回到登陆相关页面,此场景可使用 pushNamedAndRemoveUntil。
有A、B、C、D 四个页面,A 经过push进入 B 页面,B 经过push进入 C 页面,C 经过 pushNamedAndRemoveUntil 进入 D 页面同时删除路由堆栈中直到 /B 的路由,C 页面代码:
RaisedButton( child: Text('C 页面'), onPressed: () { Navigator.of(context).pushNamedAndRemoveUntil('/D', ModalRoute.withName('/B')); }, ),
D 页面按钮执行 pop:
RaisedButton( child: Text('D 页面'), onPressed: () { Navigator.of(context).pop(); }, )
从 C 页面跳转到 D 页面路由堆栈变化:
Navigator.of(context).pushNamedAndRemoveUntil('/D', ModalRoute.withName('/B'));
表示跳转到 D 页面,同时删除D 到 B 直接全部的路由,若是删除全部路由,只保存 D:
Navigator.of(context).pushNamedAndRemoveUntil('/D', (Route route)=>false);
路由堆栈变化:
有以下场景,在入职新公司的时候,须要填写各类信息,这些信息分为不一样部分,好比基本信息、工做信息、家庭信息等,这些不一样模块在不一样页面,填写信息时能够返回上一页,也能够取消,取消返回到首页,此场景可使用 popUntil,一直 pop 到指定的页面。
有A、B、C、D 四个页面,D 页面经过 popUntil 一直返回到 A 页面,D 页面代码:
RaisedButton( child: Text('D 页面'), onPressed: () { Navigator.of(context).popUntil(ModalRoute.withName('/A')); }, )
路由堆栈变化:
有以下场景,商品列表页面,点击跳转到商品详情页面,商品详情页面须要商品的惟一id或者商品详情数据,有两种方式传递数据:
第一种:经过构造函数方式:
class ProductDetail extends StatelessWidget { final ProductInfo productInfo; const ProductDetail({Key key, this.productInfo}) : super(key: key); @override Widget build(BuildContext context) { return Container(); } }
跳转代码:
Navigator.of(context).push(MaterialPageRoute(builder: (context){ return ProductDetail(productInfo: productInfo,); }));
此种方式没法用于命名路由的跳转方式。
第二种:经过命名路由设置参数的方式:
A 页面传递数据,
RaisedButton( child: Text('A 页面'), onPressed: () { Navigator.of(context).pushNamed('/B',arguments: '来自A'); }, )
B 页面经过 ModalRoute.of(context).settings.arguments
接收数据:
RaisedButton( child: Text('${ModalRoute.of(context).settings.arguments}'), onPressed: () { Navigator.of(context).pushNamed('/C'); }, )
B 页面返回代码:
RaisedButton( child: Text('${ModalRoute.of(context).settings.arguments}'), onPressed: () { Navigator.of(context).pop('从B返回'); }, )
A 页面接收返回的数据:
class APage extends StatefulWidget { @override _APageState createState() => _APageState(); } class _APageState extends State<APage> { String _string = 'A 页面'; @override Widget build(BuildContext context) { return Scaffold( body: Container( alignment: Alignment.center, child: RaisedButton( child: Text(_string), onPressed: () async { var result = await Navigator.of(context).pushNamed('/B', arguments: '来自A'); setState(() { _string = result; }); }, ), ), ); } }
push 相关方法返回 Future 类型,使用 await 等待返回结果。
老孟Flutter博客(330个控件用法+实战入门系列文章):http://laomengit.com
欢迎加入Flutter交流群(微信:laomengit)、关注公众号【老孟Flutter】:
![]() |
![]() |