基礎頁面實現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…網絡