Flutter中如何使用WillPopScope的示例代碼
在Flutter中如何實(shí)現(xiàn)點(diǎn)擊2次Back按鈕退出App,如何實(shí)現(xiàn)App中多個(gè)Route(路由),如何實(shí)現(xiàn)Back按鈕只退出指定頁(yè)面,此篇文章將告訴你。
WillPopScope
WillPopScope用于處理是否離開(kāi)當(dāng)前頁(yè)面,在Flutter中有多種方式可以離開(kāi)當(dāng)前頁(yè)面,比如AppBar、CupertinoNavigationBar上面的返回按鈕,點(diǎn)擊將會(huì)回到前一個(gè)頁(yè)面,在Android手機(jī)上點(diǎn)擊實(shí)體(虛擬)返回按鈕,也將會(huì)回到前一個(gè)頁(yè)面,此功能對(duì)于iOS程序員來(lái)說(shuō)可能特別容易忽略。
以下幾種情況我們會(huì)用到WillPopScope:
- 需要詢問(wèn)用戶是否退出。
- App中有多個(gè)Navigator,想要的是讓其中一個(gè) Navigator 退出,而不是直接讓在 Widget tree 底層的 Navigator 退出。
詢問(wèn)用戶是否退出
在Android App中最開(kāi)始的頁(yè)面點(diǎn)擊后退按鈕,默認(rèn)會(huì)關(guān)閉當(dāng)前activity并回到桌面,我們希望此時(shí)彈出對(duì)話框或者給出提示“再次點(diǎn)擊退出”,避免用戶的誤操作。
WillPopScope(
onWillPop: () async => showDialog(
context: context,
builder: (context) =>
AlertDialog(title: Text('你確定要退出嗎?'), actions: <Widget>[
RaisedButton(
child: Text('退出'),
onPressed: () => Navigator.of(context).pop(true)),
RaisedButton(
child: Text('取消'),
onPressed: () => Navigator.of(context).pop(false)),
])),
child: Container(
alignment: Alignment.center,
child: Text('點(diǎn)擊后退按鈕,詢問(wèn)是否退出。'),
))

我們也可以把效果做成快速點(diǎn)擊2次退出:
DateTime _lastQuitTime;
WillPopScope(
onWillPop: () async {
if (_lastQuitTime == null ||
DateTime.now().difference(_lastQuitTime).inSeconds > 1) {
print('再按一次 Back 按鈕退出');
Scaffold.of(context)
.showSnackBar(SnackBar(content: Text('再按一次 Back 按鈕退出')));
_lastQuitTime = DateTime.now();
return false;
} else {
print('退出');
Navigator.of(context).pop(true);
return true;
}
},
child: Container(
alignment: Alignment.center,
child: Text('點(diǎn)擊后退按鈕,詢問(wèn)是否退出。'),
))

App中有多個(gè)Navigator
我們的App通常是在MaterialApp和CupertinoApp下,MaterialApp和CupertinoApp本身有一個(gè)Navigator,所以默認(rèn)情況下調(diào)用Navigator.pop或者Navigator.push就是在操作此Navigator。不過(guò)在一些情況下,我們希望有自己定義的Navigator,比如如下場(chǎng)景:
- 在頁(yè)面底部有一個(gè)常駐bar,其上展示內(nèi)容,這個(gè)常駐bar就需要一個(gè)自己的Navigator。
- 在使用TabView、BottomNavigationBar、CupertinoTabView這些組件時(shí),希望有多個(gè)Tab,但每個(gè)Tab中有自己的導(dǎo)航行為,這時(shí)需要給每一個(gè)Tab加一個(gè)Navigator。
首頁(yè):
class MyHomePage extends StatefulWidget {
MyHomePage({Key key, this.title}) : super(key: key);
final String title;
@override
_MyHomePageState createState() => _MyHomePageState();
}
class _MyHomePageState extends State<MyHomePage> {
GlobalKey<NavigatorState> _key = GlobalKey();
@override
Widget build(BuildContext context) {
return Scaffold(
body: WillPopScope(
onWillPop: () async {
if (_key.currentState.canPop()) {
_key.currentState.pop();
return false;
}
return true;
},
child: Column(
children: <Widget>[
Expanded(
child: Navigator(
key: _key,
onGenerateRoute: (RouteSettings settings) =>
MaterialPageRoute(builder: (context) {
return OnePage();
}),
),
),
Container(
height: 50,
color: Colors.blue,
alignment: Alignment.center,
child: Text('底部Bar'),
)
],
)),
);
}
}
第一個(gè)頁(yè)面:
class OnePage extends StatelessWidget {
@override
Widget build(BuildContext context) {
return Scaffold(
body: Center(
child: Container(
child: RaisedButton(
child: Text('去下一個(gè)頁(yè)面'),
onPressed: () {
Navigator.push(context, MaterialPageRoute(builder: (context) {
return TwoPage();
}));
},
),
),
),
);
}
}
第二個(gè)頁(yè)面:
class TwoPage extends StatelessWidget {
@override
Widget build(BuildContext context) {
return Scaffold(
body: Center(
child: Container(
child: Text('這是第二個(gè)頁(yè)面'),
),
),
);
}
}

使用TabView、BottomNavigationBar、CupertinoTabView這些組件時(shí)也是一樣的原理,只需在每一個(gè)Tab中加入Navigator,不要忘記指定key。
總結(jié)
到此這篇關(guān)于Flutter中如何使用WillPopScope的文章就介紹到這了,更多相關(guān)flutter使用WillPopScope內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Android 實(shí)現(xiàn)伸縮布局效果示例代碼
這篇文章主要介紹了Android 實(shí)現(xiàn)伸縮布局效果的示例代碼,非常不錯(cuò),具有參考借鑒價(jià)值,需要的朋友可以參考下2017-01-01
Android實(shí)現(xiàn)底部導(dǎo)航欄效果
這篇文章主要為大家詳細(xì)介紹了Android實(shí)現(xiàn)底部導(dǎo)航欄效果,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-01-01
Android簡(jiǎn)單實(shí)現(xiàn)屏幕下方Tab菜單的方法
這篇文章主要介紹了Android簡(jiǎn)單實(shí)現(xiàn)屏幕下方Tab菜單的方法,簡(jiǎn)單分析了Android實(shí)現(xiàn)tab菜單所涉及的界面布局及功能相關(guān)操作技巧,需要的朋友可以參考下2016-08-08
Android編程實(shí)現(xiàn)點(diǎn)擊EditText之外的控件隱藏軟鍵盤(pán)功能
這篇文章主要介紹了Android編程實(shí)現(xiàn)點(diǎn)擊EditText之外的控件隱藏軟鍵盤(pán)功能,涉及Android控件的功能、屬性及相關(guān)操作技巧,需要的朋友可以參考下2017-06-06
Android實(shí)現(xiàn)調(diào)用攝像頭
本文給大家分享的是,在安卓APP開(kāi)發(fā)的過(guò)程中,經(jīng)常會(huì)需要調(diào)用手機(jī)自身攝像頭拍照的代碼,十分的簡(jiǎn)單實(shí)用,有需要的小伙伴可以參考下。2015-07-07
基于Fedora14下自帶jdk1.6版本 安裝jdk1.7不識(shí)別的解決方法
本篇文章是對(duì)Fedora14下自帶jdk1.6版本,安裝jdk1.7不識(shí)別的解決方法進(jìn)行了詳細(xì)的分析介紹,需要的朋友參考下2013-05-05
微信小程序首頁(yè)數(shù)據(jù)初始化失敗的解決方法
這篇文章主要介紹了微信小程序首頁(yè)數(shù)據(jù)初始化失敗的解決方法,非常不錯(cuò),具有參考借鑒價(jià)值,需要的朋友可以參考下2017-01-01

