Flutter上線項(xiàng)目實(shí)戰(zhàn)記錄之路由篇
1. 應(yīng)用場(chǎng)景
開發(fā)中經(jīng)常遇到
- 路由跳轉(zhuǎn)時(shí)拿不到context怎么辦,eg: token失效/異地登錄跳轉(zhuǎn)登錄頁面。
- 獲取不到當(dāng)前路由名稱怎么辦,eg: 點(diǎn)擊push推送跳轉(zhuǎn)指定路由,如果已經(jīng)在當(dāng)前頁面就replace,如果不在就push。
- 注冊(cè)監(jiān)聽路由跳轉(zhuǎn),做一些想做的事情 ,eg:不同路由,顯示不同狀態(tài)欄顏色。
- 等等...
2. 解決方案
解決思路:
- MaterialApp 的routes屬性賦值路由數(shù)組,navigatorObservers屬性賦值路由監(jiān)聽對(duì)象NavigatorManager。
- 在NavigatorManager里實(shí)現(xiàn)NavigatorObserver的didPush/didReplace/didPop/didRemove,并記錄到路由棧List _mRoutes中。
- 將實(shí)時(shí)記錄的路由跳轉(zhuǎn),用stream發(fā)一個(gè)廣播,哪里需要哪里注冊(cè)。
3. 具體實(shí)現(xiàn)
main.dart
MaterialApp( navigatorObservers: [NavigatorManager.getInstance()], routes: NavigatorManager.configRoutes, ... )
navigator_manager.dart
class NavigatorManager extends NavigatorObserver {
/* 配置routes */
static Map<String, WidgetBuilder> configRoutes = {
PackageInfoPage.sName: (context) =>
SplashPage.sName: (context) => SplashPage(),
LoginPage.sName: (context) => SplashPage()),
MainPage.sName: (context) => SplashPage(),
//...
}
// 當(dāng)前路由棧
static List<Route> _mRoutes;
List<Route> get routes => _mRoutes;
// 當(dāng)前路由
Route get currentRoute => _mRoutes[_mRoutes.length - 1];
// stream相關(guān)
static StreamController _streamController;
StreamController get streamController=> _streamController;
// 用來路由跳轉(zhuǎn)
static NavigatorState navigator;
/* 單例給出NavigatorManager */
static NavigatorManager navigatorManager;
static NavigatorManager getInstance() {
if (navigatorManager == null) {
navigatorManager = new NavigatorManager();
_streamController = StreamController.broadcast();
}
return navigatorManager;
}
// replace 頁面
pushReplacementNamed(String routeName, [WidgetBuilder builder]) {
return navigator.pushReplacement(
CupertinoPageRoute(
builder: builder ?? configRoutes[routeName],
settings: RouteSettings(name: routeName),
),
);
}
// push 頁面
pushNamed(String routeName, [WidgetBuilder builder]) {
return navigator.push(
CupertinoPageRoute(
builder: builder ?? configRoutes[routeName],
settings: RouteSettings(name: routeName),
),
);
}
// pop 頁面
pop<T extends Object>([T result]) {
navigator.pop(result);
}
// push一個(gè)頁面, 移除該頁面下面所有頁面
pushNamedAndRemoveUntil(String newRouteName) {
return navigator.pushNamedAndRemoveUntil(newRouteName, (Route<dynamic> route) => false);
}
// 當(dāng)調(diào)用Navigator.push時(shí)回調(diào)
@override
void didPush(Route route, Route previousRoute) {
super.didPush(route, previousRoute);
if (_mRoutes == null) {
_mRoutes = new List<Route>();
}
// 這里過濾調(diào)push的是dialog的情況
if (route is CupertinoPageRoute || route is MaterialPageRoute) {
_mRoutes.add(route);
routeObserver();
}
}
// 當(dāng)調(diào)用Navigator.replace時(shí)回調(diào)
@override
void didReplace({Route newRoute, Route oldRoute}) {
super.didReplace();
if (newRoute is CupertinoPageRoute || newRoute is MaterialPageRoute) {
_mRoutes.remove(oldRoute);
_mRoutes.add(newRoute);
routeObserver();
}
}
// 當(dāng)調(diào)用Navigator.pop時(shí)回調(diào)
@override
void didPop(Route route, Route previousRoute) {
super.didPop(route, previousRoute);
if (route is CupertinoPageRoute || route is MaterialPageRoute) {
_mRoutes.remove(route);
routeObserver();
}
}
@override
void didRemove(Route removedRoute, Route oldRoute) {
super.didRemove(removedRoute, oldRoute);
if (removedRoute is CupertinoPageRoute || removedRoute is MaterialPageRoute) {
_mRoutes.remove(removedRoute);
routeObserver();
}
}
void routeObserver() {
LogUtil.i(sName, '&&路由棧&&');
LogUtil.i(sName, _mRoutes);
LogUtil.i(sName, '&&當(dāng)前路由&&');
LogUtil.i(sName, _mRoutes[_mRoutes.length - 1]);
// 當(dāng)前頁面的navigator,用來路由跳轉(zhuǎn)
navigator = _mRoutes[_mRoutes.length - 1].navigator;
streamController.sink.add(_mRoutes);
}
}
4. 如何使用
token失效跳轉(zhuǎn)
case 401:
ToastUtil.showRed('登錄失效,請(qǐng)重新登陸');
UserDao.clearAll();
NavigatorManager.getInstance().pushNamedAndRemoveUntil(LoginPage.sName);
break;
點(diǎn)擊push推送跳轉(zhuǎn)
static jumpPage(String pageName, [WidgetBuilder builder]) {
String currentRouteName = NavigatorManager.getInstance().currentRoute.settings.name;
// 如果是未登錄,不跳轉(zhuǎn)
if (NavigatorManager.getInstance().routes[0].settings.name != MainPage.sName) {
return;
}
// 如果已經(jīng)是當(dāng)前頁面就replace
if (currentRouteName == pageName) {
NavigatorManager.getInstance().pushReplacementNamed(pageName, builder);
} else {
NavigatorManager.getInstance().pushNamed(pageName, builder);
}
}
監(jiān)聽路由改變狀態(tài)欄顏色
class StatusBarUtil {
static List<String> lightRouteNameList = [
TaskhallPage.sName,
//...
];
static List darkRoutNameList = [
SplashPage.sName,
LoginPage.sName,
MainPage.sName,
//...
];
static init() {
NavigatorManager.getInstance().streamController.stream.listen((state) {
setupStatusBar(state[state.length - 1]);
})
}
setupStatusBar(Route currentRoute) {
if (lightRouteNameList.contains(currentRoute.settings.name)) {
setLight();
} else if (darkRoutNameList.contains(currentRoute.settings.name)) {
setDart();
}
}
}
完結(jié),撒花🎉
總結(jié)
以上就是這篇文章的全部?jī)?nèi)容了,希望本文的內(nèi)容對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,謝謝大家對(duì)腳本之家的支持。
- Flutter路由框架Fluro使用簡(jiǎn)介
- Flutter如何完成路由攔截,實(shí)現(xiàn)權(quán)限管理
- Flutter開發(fā)中的路由參數(shù)處理
- Flutter app頁面路由以及路由攔截的實(shí)現(xiàn)
- 詳解Flutter的路由導(dǎo)航
- flutter 路由跳轉(zhuǎn)的實(shí)現(xiàn)示例
- flutter 自定義websocket路由的實(shí)現(xiàn)
- Flutter開發(fā)之路由與導(dǎo)航的實(shí)現(xiàn)
- Flutter利用注解生成可自定義的路由的實(shí)現(xiàn)
- Flutter 局部路由實(shí)現(xiàn)詳解
- Flutter路由的跳轉(zhuǎn)、動(dòng)畫和傳參詳解(最簡(jiǎn)單)
- Flutter 路由插件fluro的使用
相關(guān)文章
android 仿微信demo——注冊(cè)功能實(shí)現(xiàn)(移動(dòng)端)
本篇文章主要介紹了微信小程序-閱讀小程序?qū)嵗╠emo),小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧,希望能給你們提供幫助2021-06-06
Android 自定義view之畫圖板實(shí)現(xiàn)方法
本文重在對(duì)自定義view,以及其常用類,常用方法的初步了解,提供一個(gè)思路,效果是其次,畫板只是例子,需要的朋友可以參考下2018-01-01
使用Android Studio檢測(cè)內(nèi)存泄露(LeakCanary)
本篇文章主要介紹了用Android Studio檢測(cè)內(nèi)存泄露的問題的解決方法,Android Studio在為我們提供了良好的編碼體驗(yàn)的同時(shí),也提供了許多對(duì)App性能分析的工具,下面我們一起來了解一下。2016-12-12
實(shí)例講解Android app開發(fā)中ListView的基本使用及優(yōu)化
這篇文章主要介紹了Android app開發(fā)中ListView的基本使用及優(yōu)化,ListView視圖組件是Android中最常用的組件之一需要的朋友可以參考下2016-02-02
Android模仿Toast實(shí)現(xiàn)提示框效果
這篇文章主要為大家詳細(xì)介紹了Android模仿Toast實(shí)現(xiàn)提示框效果,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-08-08

