Flutter中如何實現(xiàn)無Context跳轉(zhuǎn)詳解
背景介紹
Navigator.of(context).push(MaterialPageRoute(builder: (context){
return DemoPage();
}));
在日常的項目開發(fā)中,我們一般push一個新頁面是用上面的方法的,利用Navigator.of(context)來進(jìn)行push或者pop操作。
缺點:這種情況是必須傳context的,目的是為了利用Navigator.of(context)來獲取到NavigatorState對象,然后才能進(jìn)行push或者pop操作。
那如果我要實現(xiàn)在項目的任何地方都可以push一個新頁面的話,而這個地方有可能獲取不到context,所以這個時候,就需要實現(xiàn)無context跳轉(zhuǎn)。
解決方案
無context跳轉(zhuǎn),本質(zhì)就是不必要我們每次都去傳context參數(shù),然后利用一些操作直接去獲取到當(dāng)前的NavigatorState。
方案1:利用GlobalKey
- 在Flutter中,利用GolbalKey利用獲取到對應(yīng)Widget的State對象。所以,這里,我們可以通過一個GlobalKey的key值,來獲取到NavigatorState對象。
- MaterialApp中包裝了WidgetsApp,而WidgetsApp包裝了Navigator,并且將 Navigator的key屬性作為navigatorKey暴露出來了。所以,我們可以通過設(shè)置navigatorKey,然后利用這個key去獲取到NavigatorState對象。
這里貼一下相關(guān)的源碼,具體的大家可以自己去看源碼。
MaterialApp類:

WidgetsApp類:可以看出,我們定義的navigatorKey,最后是會傳給Navigator的key值,所以我們在外面就可以通過key.currentState()方法來獲取到這里的NavigatorState對象了。
class _WidgetsAppState extends State<WidgetsApp> implements WidgetsBindingObserver {
GlobalKey<NavigatorState> _navigator;
void _updateNavigator() {
_navigator = widget.navigatorKey ?? GlobalObjectKey<NavigatorState>(this);
}
@override
Widget build(BuildContext context) {
Widget navigator;
if (_navigator != null) {
navigator = Navigator(
key: _navigator,
initialRoute: WidgetsBinding.instance.window.defaultRouteName != Navigator.defaultRouteName
? WidgetsBinding.instance.window.defaultRouteName
: widget.initialRoute ?? WidgetsBinding.instance.window.defaultRouteName,
onGenerateRoute: _onGenerateRoute,
onUnknownRoute: _onUnknownRoute,
observers: widget.navigatorObservers,
);
}
}
簡單的代碼實現(xiàn)
1、定義一個GlobalKey< NavigatorState>對象
static GlobalKey<NavigatorState> navigatorKey=GlobalKey();
2、創(chuàng)建MaterialApp的對象的時候,將navigatorKey賦值給MaterialApp。
MaterialApp( navigatorKey: Router.navigatorKey, )
使用GlobalKey在任意地方獲取NavigatorState對象
navigatorKey.currentState.pushNamed("/login");
方案2:利用NavigatorObserver
- NavigatorObserver,看這名字,就知道是可以用來監(jiān)聽Navigator的變化。比如當(dāng)push一個新頁面的時候,Navigator會監(jiān)聽到NavigatorState發(fā)生變化,回調(diào)didPush()方法。
注意:NavigatorObserver里面定義了一個NavigatorState對象navigator,所以我們可以通過自定義NavigatorObserver,然后直接利用這個navigator對象來做頁面push或者pop操作,這樣的話,我們就不用自己去利用context去獲取navigatorState對象了。

MaterialApp類,提供了navigatorObservers屬性,這樣我們就可以自定義NavigatorObserver去監(jiān)聽Navigator的變化。

NavigatorState類,執(zhí)行instState對象的時候,會將自身賦值到監(jiān)聽的所有observer對象的_navigator里面。

簡單的代碼實現(xiàn)
1、自定義NavigatorObserver。
class CustomNavigatorObserver extends NavigatorObserver{
static CustomNavigatorObserver _instance;
static CustomNavigatorObserver getInstance() {
if (_instance == null) {
_instance = CustomNavigatorObserver();
}
return _instance;
}
}
2、創(chuàng)建MaterialApp的對象的時候,將CustomNavigatorObserver賦值給MaterialApp
MaterialApp( navigatorObservers: [CustomNavigatorObserver()], )
3、使用CustomNavigatorObserver在任意地方進(jìn)行頁面操作
CustomNavigatorObserver.getInstance().navigator.pushNamed("/login");
總結(jié)
以上就是這篇文章的全部內(nèi)容了,希望本文的內(nèi)容對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,謝謝大家對腳本之家的支持。
相關(guān)文章
關(guān)于Android bitmap你不知道的一些事
這篇文章主要為大家詳細(xì)介紹了關(guān)于Android bitmap你不知道的一些事,使用bitmap需要注意的一些細(xì)節(jié),具有一定的參考價值,感興趣的小伙伴們可以參考一下2016-11-11
android LabelView實現(xiàn)標(biāo)簽云效果
這篇文章主要為大家詳細(xì)介紹了android LabelView實現(xiàn)標(biāo)簽云效果,文中示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下2018-05-05
Android編程實現(xiàn)WebView自適應(yīng)全屏方法小結(jié)
這篇文章主要介紹了Android編程實現(xiàn)WebView自適應(yīng)全屏方法,結(jié)合實例形式總結(jié)了三種常用的WebView自適應(yīng)全屏實現(xiàn)技巧,具有一定參考借鑒價值,需要的朋友可以參考下2015-12-12
Android?十六進(jìn)制狀態(tài)管理實例詳解
這篇文章主要為大家介紹了Android?十六進(jìn)制狀態(tài)管理實例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-09-09
Android金額輸入框只允許輸入小數(shù)點后兩位效果
實現(xiàn)android 金額輸入框輸入小數(shù)點后兩位的效果也不是很復(fù)雜,只需要設(shè)置輸入框輸入的字符類型、設(shè)置InputFilter、設(shè)置輸入變化監(jiān)聽即可。這篇文章主要介紹了Android金額輸入框只允許輸入小數(shù)點后兩位 ,需要的朋友可以參考下2017-05-05
Android AIDL實現(xiàn)與服務(wù)相互調(diào)用方式
這篇文章主要介紹了Android AIDL實現(xiàn)與服務(wù)相互調(diào)用方式,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2020-03-03

