flutter 切换tab后保留tab状态
前言
最近在用flutter写一个小项目,在写主页面(底部导航栏+子页面)时遇到的一个问题:当点击底部item切换到另一页面, 再返回此页面时会重走它的initState方法(我们一般在initState中发起网络请求,或者初始化的操作),导致不必要的开销
根据Tab动态加载页面
我们先定义两个页面PageA和PageB
class PageA extends StatefulWidget {
_PageAState createState() => _PageAState();
}
class _PageAState extends State<PageA> {
@override
void initState() {
super.initState();
print("pageA init state");
}
@override
Widget build(BuildContext context) {
return Container(
color: Colors.orangeAccent,
child: Center(
child: Text("page A"),
),
);
}
}
class PageB extends StatefulWidget {
_PageBState createState() => _PageBState();
}
class _PageBState extends State<PageB> {
@override
void initState() {
super.initState();
print("pageB init state");
}
@override
Widget build(BuildContext context) {
return Container(
color: Colors.blueAccent,
child: Center(
child: Text("page B"),
),
);
}
}
定义Tab主页面
import 'package:flutter/material.dart';
void main() => runApp(MyApp());
class MyApp extends StatefulWidget {
_MyAppState createState() => _MyAppState();
}
class _MyAppState extends State<MyApp> {
int _tabIndex = 0;
List<Widget> _tabWidget = [
PageA(),
PageB()
];
@override
Widget build(BuildContext context) {
return MaterialApp(
title: 'tab demo',
home: Scaffold(
bottomNavigationBar: BottomNavigationBar(
items: _createBottomItems(),
currentIndex: this._tabIndex,
onTap: (index) {
setState(() {
this._tabIndex = index;
});
},
),
body: this._tabWidget.elementAt(this._tabIndex),
),
);
}
}
// 创建底部导航item
List<BottomNavigationBarItem> _createBottomItems() {
return [
BottomNavigationBarItem(
icon: Icon(Icons.home),
title: Text("首页")
),
BottomNavigationBarItem(
icon: Icon(Icons.insert_emoticon),
title: Text("我的")
)
];
运行后发现,每次切换tab都会调用initState,这显然不符合我们的正常的需求,有下面两种解决方式
IndexedStack
IndexedStack可以控制子元素的显示和隐藏,并且会缓存所有的元素,不会每次都重新创建子元素
@override
Widget build(BuildContext context) {
return MaterialApp(
title: 'tab demo',
home: Scaffold(
bottomNavigationBar: BottomNavigationBar(
items: _createBottomItems(),
currentIndex: this._tabIndex,
onTap: (index) {
setState(() {
this._tabIndex = index;
});
},
),
body: IndexedStack(
children: this._tabWidget,
index: this._tabIndex,
)
),
);
}
运行后发现还是有个问题,IndexedStack在初始化的时候会初始化所有的子元素,pageA和pageB的initState会同时调用,这明显还是不符合我们的需求
正确来说应该是切换到具体页面的时候才进行初始化,而不是一开始就加载所有的页面的数据,避免资源浪费
PageView + AutomaticKeepAliveClientMixin
使用PageView支持多个view切换,并且不会一次加载完所有的页面
PageController _pageController;
@override
void initState() {
super.initState();
this._pageController =PageController(initialPage: this._tabIndex, keepPage: true);
}
@override
Widget build(BuildContext context) {
return MaterialApp(
title: 'tab demo',
home: Scaffold(
bottomNavigationBar: BottomNavigationBar(
items: _createBottomItems(),
currentIndex: this._tabIndex,
onTap: (index) {
setState(() {
this._tabIndex = index;
_pageController.jumpToPage(index);
});
},
),
body: PageView(
children: this._tabWidget,
controller: _pageController,
),
),
);
}
}
使用PageView可以正常切换,但是每次切换Tab的时候还是会重复调用initState,我们还需要在子页面实现AutomaticKeepAliveClientMixin
class _PageAState extends State<PageA> with AutomaticKeepAliveClientMixin {
@override
bool get wantKeepAlive => true;
...
}
class _PageBState extends State<PageB> with AutomaticKeepAliveClientMixin {
@override
bool get wantKeepAlive => true;
...
}
实现了AutomaticKeepAliveClientMixin就不会每次切换Tab都调用initState了,这也是google推荐的方式
最后发现PageView可以左右滑动切换,这个可以通过设置physics为NeverScrollableScrollPhysics()来禁止滑动
PageView(
children: this._tabWidget,
controller: _pageController,
physics: NeverScrollableScrollPhysics(),
)
参考来源:https://blog.bombox.org/2019-02-23/flutter-tab-keep-alive/#more
flutter 切换tab后保留tab状态的更多相关文章
- easyui tree 编辑后保留原先状态
$(function () { var selected = $('#depttree').tree('getSelected'); $('#depttree').tree({ checkbox: f ...
- Flutter——TabBar组件(顶部Tab切换组件)
TabBar组件的常用属性: 属性 描述 tabs 显示的标签内容,一般使用 Tab 对象,也可以是其他的Widget controller TabController 对象 isScrollabl ...
- 很好用的Tab标签切换功能,延迟Tab切换。
一个网页,Tab标签的切换是常见的功能,但我发现很少有前端工程师在做该功能的时候,会为用户多想想,如果你觉得鼠标hover到标签上,然后切换到相应的内容,就那么简单的话,你将是一个不合格的前端工程师啊 ...
- easyui修复浏览器刷新后,tab页全部关闭的问题
一.问题描述 使用easyui搭建的上左右页面布局,当我们在右侧打开了tab页,发现点击浏览器的刷新按钮后,整个页面会被重新渲染,导致所有打开的tab页都被关闭,回到初始状态的问题. 这个问题虽然不影 ...
- 微信小程序自定义tab,多层tab嵌套实现
小程序最近是越来越火了-- 做小程序有一段时间了,总结一下项目中遇到的问题及解决办法吧. 项目中有个多 tab 嵌套的需求,进入程序主界面下面有两个 tab,进入A模块后,A模块最底下又有多个tab, ...
- 在Bootstrap开发框架的工作流模块中实现流程完成后更新资料状态处理
在开发查看流程表单明细的时候,在Web界面中,我们往往通过使用@RenderPage实现页面内容模块化的隔离,减少复杂度,因此把一些常用的如审批.撤销.会签.阅办等等的流程步骤都放到了通用处理的页面V ...
- 低版本系统兼容的ActionBar(四)添加Tab+添加自定义的Tab视图+Fragment
在ActionBar中添加Tab是很有用的技巧.在support V7库的支持下,我们几乎可以用和之前一样的方式来添加Tab,对于Tab来说,我们可以和MenuItem一样,给他定义自己的视图.我这里 ...
- 刨根问底U3D---如何退出Play模式后保留数据更改
实际中遇到的需求 在做一款对抗类游戏,目前正在调整游戏的平衡性 所以就产生了一个需求 希望可以在Play模式时候对数据源做的更改可以在退出时候被保存下来. 举个Case, 比如 有一个炮塔 可以发射子 ...
- C# 版本的 计时器类:精确到微秒 秒后保留一位小数 支持年月日时分秒带单位的输出
class TimeCount { // 临时变量,存放当前类能表示的最大年份值 ; /// <summary> /// 获取毫秒能表示的最大年份数 /// </summary> ...
随机推荐
- 论文笔记:Unsupervised Domain Adaptation by Backpropagation
14年9月份挂出来的文章,基本思想就是用对抗训练的方法来学习domain invariant的特征表示.方法也很只管,在网络的某一层特征之后接一个判别网络,负责预测特征所属的domain,而后特征提取 ...
- Select,poll,epoll复用
Select,poll,epoll复用 1)select模块以列表的形式接受四个参数,分别是可读对象,可写对象,产生异常的对象,和超时设置.当监控符对象发生变化时,select会返回发生变化的对象列表 ...
- jdk提供的线程协调API suspend/resume wait/notify park/unpark
线程通信(如 线程执行先后顺序,获取某个线程执行的结果等)有多种方式: 文件共享 线程1 --写入--> 文件 < --读取-- 线程2 网络共享 变量共享 线程1 --写入--> ...
- centos7安装BitCoin客户端
一.安装依赖环境 [root@localhost src]# yum install autoconf automake libtool libdb-devel boost-devel libeven ...
- dubbo 框架
2.1 dubbo Apache Dubbo (incubating) |ˈdʌbəʊ| 是一款高性能.轻量级的开源 Java RPC 框架,它提供了三大核心能力:面向接口的远程方法调用,智能容错和 ...
- Loadrunner:录制APP脚本
一.问题 使用 Loadrunner 12 自带的代理进行录制APP脚本时,遇到了各种阻碍 二.解决途径 使用 Fiddler 抓包后导出成 saz 格式文件,再导入到 Loadrunner 中,完美 ...
- grub命令行和配置文件配置配置信息
一.grub简介 GNU GRUB(简称“GRUB”)是一个来自GNU项目的启动引导程序.GRUB是多启动规范的实现,它允许用户可以在计算机内同时拥有多个操作系统,并在计算机启动时选择希望运行的操作系 ...
- SpringBoot入门系列:第五篇 JPA mysql(转)
一,准备工作,建立spring-boot-sample-mysql工程1.http://start.spring.io/ A.Artifact中输入spring-boot-sample-mysql B ...
- 使用fiddler抓取jmeter发送的请求
使用jmeter发送请求时,有时需要查看发送的请求是否合理,可以使用fiddler更直观的抓取并查看jmeter发送的请求.步骤如下:1.设置fidder-connections 端口号为8888 2 ...
- Mockito 2 关于打标(stubbing)
请参考下面有关于打标的代码. //You can mock concrete classes, not just interfaces LinkedList mockedList = mock(Lin ...