基礎頁面實現android
List <String>_titles=['湖人','勇士','雄鹿','快船','凱爾特人','馬刺','76人','猛龍']; TabController _tabController; ///省略部分代碼 class MyHomePage extends StatefulWidget { MyHomePage({Key key, this.title}) : super(key: key); final String title; ///省略部分代碼 class _MyHomePageState extends State<MyHomePage> with SingleTickerProviderStateMixin{ @override void initState() { super.initState(); //初始化控制器 _tabController = new TabController(length: _titles.length,vsync: this); } @override Widget build(BuildContext context) { return Scaffold( appBar: AppBar( leading: Icon(Icons.menu), title: buildTabBar(), //bottom: buildTabBar(), ), body: TabBarViewLayout() ); } Widget buildTabBar() { return TabBar( //構造Tab集合 tabs: _titles.map((String title){ return Tab( text: title, ); }).toList(), ///省略部分代碼 controller: _tabController, ); } } // TabBarView Widget class TabBarViewLayout extends StatelessWidget { @override Widget build(BuildContext context) { print("TabBarViewLayout build......."); return TabBarView( controller: _tabController, children: _titles.map((String title){ return TabPageView(title); }).toList(), ); } } 複製代碼
const RefreshIndicator({
Key key,
@required this.child, //包裝一個可滾動widget
this.displacement = 40.0,
@required this.onRefresh, //觸發刷新調用方法
this.color, //指示器顏色
this.backgroundColor,
this.notificationPredicate = defaultScrollNotificationPredicate,
this.semanticsLabel,
this.semanticsValue,
})
複製代碼
@override Widget build(BuildContext context) { return RefreshIndicator( child: ListView.builder( ///保持ListView任何狀況都能滾動,解決在RefreshIndicator的兼容問題。 physics: const AlwaysScrollableScrollPhysics(), itemBuilder: (context,index){ return _getItem(index); }, ///根據狀態返回繪製 item 數量 itemCount: _getListCount(), ///滑動監聽 controller: _scrollController, ), onRefresh: _handleRefresh, color: Theme.of(context).primaryColor, //指示器顏色 ); } 複製代碼
///根據配置狀態返回實際列表數量 _getListCount() { ///是否須要頭部 if (widget.isHaveHeader) { return (items.length > 0) ? items.length + 2 : items.length + 1; } else { if (items.length == 0) { return 1; } return (items.length > 0) ? items.length + 1 : items.length; } } 複製代碼
///根據配置狀態返回實際列表渲染Item _getItem(int index) { if (!widget.isHaveHeader && index == items.length && items.length != 0) { return _buildProgressIndicator(); } else if (widget.isHaveHeader && index == _getListCount()-1 && items.length != 0) { return _buildProgressIndicator(); } else if (widget.isHaveHeader && index == 0 && items.length != 0) { return widget.headerView(); } else if (!widget.isHaveHeader && items.length == 0) { ///若是不須要頭部,而且數據爲0,渲染空頁面 if(isLoading){ return _buildIsLoading(); }else{ return _buildEmpty(); } } else if(widget.isHaveHeader && items.length == 0){ if(isLoading){ return _buildIsLoading(); }else{ return _buildEmpty(); } } else { return widget.renderItem(index, items[widget.isHaveHeader ? index-1 : index]); } } 複製代碼
該方法中,若是沒有設置頭部,而且數據不爲0,當index等於數據長度時,渲染加載更多頁面(由於index是從0開始);若是設置了頭部頁面,而且數據不爲0,當index等於實際渲染長度 - 1時,渲染加載更多頁面(在該方法判斷是否已經加載到底);接着若是設置了頭部widget,而且數據不爲0,當index = 0 ,渲染頭部widget;若是沒設置頭部,而且數據爲0,若是當前正在刷新,渲染Loading頁面,不然渲染空頁面或者Error頁面;同理,若是設置頭部,而且數據爲0,而且當前正在刷新,渲染Loading頁面,不然渲染空頁面或者Error頁面;若是不是上面狀況,則渲染正常渲染Item,若是這裏有須要,能夠直接返回相對位置的index,若是有頭部 index 減一 ,保持不會忽略 index = 0 的數據。git
接着封裝一個統一網絡請求方法,外部請求安裝固定格式的 Map 將數據返回給下拉刷新上拉加載更多widget,達到通用的目的。github
//網絡請求獲取數據 isRefresh 是否爲下拉刷新 Future<List> makeHttpRequest(bool isRefresh) async { if (widget.requestApi is Function) { Map listObj = new Map<String, dynamic>(); if(isRefresh){ //下拉刷新 listObj = await widget.requestApi({'pageIndex': 0}); }else{ //上拉加載更多 listObj = await widget.requestApi({'pageIndex': _pageIndex}); } _pageIndex = listObj['pageIndex']; _pageTotal = listObj['total']; return listObj['list']; } else { return Future.delayed(Duration(seconds: 2), () { return []; }); } } 複製代碼
Widget _buildIsLoading() { return Container( width: MediaQuery.of(context).size.width, height: MediaQuery.of(context).size.height*0.85, child: new Center( child: Column( crossAxisAlignment: CrossAxisAlignment.center, mainAxisAlignment: MainAxisAlignment.center, children: <Widget>[ Row( mainAxisAlignment: MainAxisAlignment.spaceEvenly, children: <Widget>[ SpinKitCircle(size: 55.0, color: Theme.of(context).primaryColor), ], ), Padding( child: Text("正在加載..", style: TextStyle(color: Colors.black54, fontSize: 15.0)), padding: EdgeInsets.all(15.0),) ],) )); } 複製代碼
// 模塊item final renderItem; //數據獲取方法 final requestApi; //頭部 final headerView; //是否添加頭部 默認不添加 final bool isHaveHeader; //是否支持下拉刷新 默承認如下拉刷新 final bool isCanRefresh; //是否支持下拉加載更多 默承認以加載更多 final bool isCanLoadMore; const RefreshPage({@required this.requestApi, @required this.renderItem, this.headerView, this.isHaveHeader = false, this.isCanRefresh = true, this.isCanLoadMore = true }) : assert(requestApi is Function), assert(renderItem is Function), super(); 複製代碼
Flutter完整開源項目: github.com/maoqitian/f…markdown