flutter tab 切换时会导致页面重新加载的解决方案!

最近在用flutter做app的开发,期间遇到了一个问题,就是当使用tab组件当时候,如果跳着点击tab(比如1->3, 2->4,1->1这种点击),则会导致中间当页面重新加载,这种体验十分不友好。

一开始是在网上找当解决方案,让页面 with AutomaticKeepAliveClientMixin, 然后 bool get wantKeepAlive => true; 但是这种方法治标不治本。当你按顺序切换tab的时候,比如 1->2, 2->3, 3->2 时页面确实不会重新加载,但是像我上面描述的1->3, 1->4,4->2 这种跳着切换时,AutomaticKeepAliveClientMixin 就不起作用了,中间的页面还是会重新加载。

于是乎本着探索的精神,终于让我发现了一个相对简单的解决方案,那就是使用 IndexedStack 部件。用这个之后,无论你怎么点击,都能够正确的保持页面的状态,并且dispose的时候,也能够正确的销毁相关组件。以下是示例代码,很简单,大家可以尝试一下:


import 'package:flutter/material.dart';

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

class MyAppextends StatelessWidget {

@override

  Widget build(BuildContext context) {

return MaterialApp(

title:"Don't rebuilt!",

theme:ThemeData(primarySwatch: Colors.blue),

home:FirstPage(),

);

}

}

class FirstPageextends StatelessWidget {

FirstPage({Key key}) :super(key: key);

@override

  Widget build(BuildContext context) {

return Scaffold(

appBar:AppBar(title:Text("I'm the first page")),

body:Center(

child:RaisedButton(

child:Text("go to the tab page"),

onPressed: () => Navigator.of(context).push(

MaterialPageRoute(

builder: (context) =>MyTabPage(),

),

),

),

));

}

}

class MyTabPageextends StatefulWidget {

MyTabPage({Key key}) :super(key: key);

@override

  _MyTabPageState createState() =>_MyTabPageState();

}

class _MyTabPageStateextends Statewith SingleTickerProviderStateMixin {

TabController_tabController;

int_currentTabIndex =0;

List_realPages = [Tab1(),Tab2(),Tab3(),Tab4()];

//lazy init

List_tabPages = [Tab1(),Container(),Container(),Container()];

List_hasInit = [true,false,false,false];

List_tabsButton = [

Tab(child:Text("tab1", style:TextStyle(color: Colors.black))),

Tab(child:Text("tab2", style:TextStyle(color: Colors.black))),

Tab(child:Text("tab3", style:TextStyle(color: Colors.black))),

Tab(child:Text("tab4", style:TextStyle(color: Colors.black))),

];

@override

  void initState() {

print("MyTabPage initState.");

super.initState();

_tabController =TabController(length:_realPages.length, vsync:this)

..addListener(() {

int selectedIndex =_tabController.index;

if (_currentTabIndex != selectedIndex) {

if (!_hasInit[selectedIndex]) {

//lazy init

            _tabPages[selectedIndex] =_realPages[selectedIndex];

_hasInit[selectedIndex] =true;

}

setState(() =>_currentTabIndex = selectedIndex);

}

});

}

@override

  void dispose() {

print("MyTabPage dispose.");

_tabController.dispose();

super.dispose();

}

@override

  Widget build(BuildContext context) {

return Scaffold(

appBar:AppBar(title:Text("Don't rebuilt!")),

body:IndexedStack(index:_currentTabIndex, children:_tabPages),

bottomNavigationBar:TabBar(controller:_tabController, tabs:_tabsButton),

);

}

}

class Tab1extends StatefulWidget {

@override

State createState() {

return _Tab1State();

}

}

class _Tab1Stateextends State {

@override

  void initState() {

super.initState();

print("tab1 initState");

}

@override

  void dispose() {

print("tab1 dispose");

super.dispose();

}

@override

  Widget build(BuildContext context) {

return Scaffold(body:Center(child:Text("I'm tab1")));

}

}

class Tab2extends StatefulWidget {

@override

State createState() {

return _Tab2State();

}

}

class _Tab2Stateextends State {

@override

  void initState() {

super.initState();

print("tab2 initState");

}

@override

  void dispose() {

print("tab2 dispose");

super.dispose();

}

@override

  Widget build(BuildContext context) {

return Scaffold(body:Center(child:Text("I'm tab2")));

}

}

class Tab3extends StatefulWidget {

@override

State createState() {

return _Tab3State();

}

}

class _Tab3Stateextends State {

@override

  void initState() {

super.initState();

print("tab3 initState");

}

@override

  void dispose() {

print("tab3 dispose");

super.dispose();

}

@override

  Widget build(BuildContext context) {

return Scaffold(body:Center(child:Text("I'm tab3")));

}

}

class Tab4extends StatefulWidget {

@override

State createState() {

return _Tab4State();

}

}

class _Tab4Stateextends State {

@override

  void initState() {

super.initState();

print("tab4 initState");

}

@override

  void dispose() {

print("tab4 dispose");

super.dispose();

}

@override

  Widget build(BuildContext context) {

return Scaffold(body:Center(child:Text("I'm tab4")));

}

}


Performing hot reload...

Syncing files to device Android SDK built for x86...

Reloaded 1 of 440 libraries in 284ms.

I/flutter (21209): MyTabPage initState.

I/flutter (21209): tab1 initState

I/flutter (21209): tab3 initState

I/flutter (21209): tab2 initState

I/flutter (21209): tab4 initState

I/flutter (21209): tab1 dispose

I/flutter (21209): tab2 dispose

I/flutter (21209): tab3 dispose

I/flutter (21209): tab4 dispose

I/flutter (21209): MyTabPage dispose.
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,362评论 5 477
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,330评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,247评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,560评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,580评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,569评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,929评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,587评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,840评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,596评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,678评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,366评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,945评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,929评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,165评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 43,271评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,403评论 2 342

推荐阅读更多精彩内容