Flutter Widgets 之 PageView

git

程序員

一枚github

有態度微信

的程序員app


注意:無特殊說明,Flutter版本及Dart版本以下:編輯器

  • Flutter版本:1.12.13+hotfix.5ide

  • Dart版本:2.7.0flex

基礎用法

PageView控件能夠實現一個「圖片輪播」的效果,PageView不只能夠水平滑動也能夠垂直滑動,簡單用法以下:動畫

PageView(
children: <Widget>[
MyPage1(),
MyPage2(),
MyPage3(),
],
)

PageView滾動方向默認是水平,能夠設置其爲垂直方向:ui

PageView(
scrollDirection: Axis.vertical,
...
)

PageView配合PageController能夠實現很是酷炫的效果,控制每個Page不佔滿,

PageView(
controller: PageController(
viewportFraction: 0.9,
),
...
)

PageController中屬性initialPage表示當前加載第幾頁,默認第一頁。

onPageChanged屬性是頁面發生變化時的回調,用法以下:

PageView(
onPageChanged: (int index){
},
...
)

無限滾動

PageView滾動到最後時但願滾動到第一個頁面,這樣看起來PageView是無限滾動的:

List<Widget> pageList = [PageView1(), PageView2(), PageView3()];

PageView.builder(
itemCount: 10000,
itemBuilder: (context, index) {
return pageList[index % (pageList.length)];
},
)

巧妙的利用取餘(%)重複構建頁面實現PageView無限滾動的效果:

實現指示器

指示器顯示總數和當前位置,經過onPageChanged肯定當前頁數並更新指示器。

List<String> pageList = ['PageView1', 'PageView2', 'PageView3'];
int _currentPageIndex = 0;

_buildPageView() {
return Center(
child: Container(
height: 230,
child: Stack(
children: <Widget>[
PageView.builder(
onPageChanged: (int index) {
setState(() {
_currentPageIndex = index % (pageList.length);
});
},
itemCount: 10000,
itemBuilder: (context, index) {
return _buildPageViewItem(pageList[index % (pageList.length)]);
},
),
Positioned(
bottom: 10,
left: 0,
right: 0,
child: Container(
child: Row(
mainAxisAlignment: MainAxisAlignment.center,
children: List.generate(pageList.length, (i) {
return Container(
margin: EdgeInsets.symmetric(horizontal: 5),
width: 10,
height: 10,
decoration: BoxDecoration(
shape: BoxShape.circle,
color: _currentPageIndex == i
? Colors.blue
: Colors.grey),
);
}).toList(),
),
),
),
],
),
),
);
}

_buildPageViewItem(String txt, {Color color = Colors.red}) {
return Container(
color: color,
alignment: Alignment.center,
child: Text(
txt,
style: TextStyle(color: Colors.white, fontSize: 28),
),
);
}

效果以下:

切換動畫

如此常見的切換效果顯然不能體驗咱們獨特的個性,咱們須要更炫酷的方式,看下面的效果:

在滑出的時候當前頁面逐漸縮小並居中,經過給PageController添加監聽獲取當前滑動的進度:

_pageController.addListener(() {
setState(() {
_currPageValue = _pageController.page;
});
});

經過當前的進度計算各個頁面的縮放係數及平移係數,經過 判斷當前構建的是哪一個頁面

if (index == _currPageValue.floor()) {
//當前的item
var currScale = 1 - (_currPageValue - index) * (1 - _scaleFactor);

} else if (index == _currPageValue.floor() + 1) {
//右邊的item

} else if (index == _currPageValue.floor() - 1) {
//左邊

} else {
//其餘,不在屏幕顯示的item

}

經過對這幾種類型的頁面的縮放和平移達到咱們想要的效果。

完整代碼:

class ViewPage extends StatefulWidget {
@override
State<StatefulWidget> createState() => _ViewPageState();
}

class _ViewPageState extends State<ViewPage> {
var imgList = [
'https://ss1.bdstatic.com/70cFvXSh_Q1YnxGkpoWK1HF6hhy/it/u=2877516247,37083492&fm=26&gp=0.jpg',
'https://timgsa.baidu.com/timg?image&quality=80&size=b9999_10000&sec=1582796218195&di=04ce93c4ac826e19067e71f916cec5d8&imgtype=0&src=http%3A%2F%2Fhbimg.b0.upaiyun.com%2F344fda8b47808261c946c81645bff489c008326f15140-koiNr3_fw658'
];
PageController _pageController;

var _currPageValue = 0.0;

//縮放係數
double _scaleFactor = .8;

//view page height
double _height = 230.0;

@override
void initState() {
super.initState();
_pageController = PageController(viewportFraction: 0.9);
_pageController.addListener(() {
setState(() {
_currPageValue = _pageController.page;
});
});
}

@override
void dispose() {
super.dispose();
_pageController.dispose();
}
@override
Widget build(BuildContext context) {
return Container(
height: _height,
child: PageView.builder(
itemBuilder: (context, index) => _buildPageItem(index),
itemCount: 10,
controller: _pageController,
));
}

_buildPageItem(int index) {
Matrix4 matrix4 = Matrix4.identity();
if (index == _currPageValue.floor()) {
//當前的item
var currScale = 1 - (_currPageValue - index) * (1 - _scaleFactor);
var currTrans = _height * (1 - currScale) / 2;

matrix4 = Matrix4.diagonal3Values(1.0, currScale, 1.0)
..setTranslationRaw(0.0, currTrans, 0.0);
} else if (index == _currPageValue.floor() + 1) {
//右邊的item
var currScale =
_scaleFactor + (_currPageValue - index + 1) * (1 - _scaleFactor);
var currTrans = _height * (1 - currScale) / 2;

matrix4 = Matrix4.diagonal3Values(1.0, currScale, 1.0)
..setTranslationRaw(0.0, currTrans, 0.0);
} else if (index == _currPageValue.floor() - 1) {
//左邊
var currScale = 1 - (_currPageValue - index) * (1 - _scaleFactor);
var currTrans = _height * (1 - currScale) / 2;

matrix4 = Matrix4.diagonal3Values(1.0, currScale, 1.0)
..setTranslationRaw(0.0, currTrans, 0.0);
} else {
//其餘,不在屏幕顯示的item
matrix4 = Matrix4.diagonal3Values(1.0, _scaleFactor, 1.0)
..setTranslationRaw(0.0, _height * (1 - _scaleFactor) / 2, 0.0);
}

return Transform(
transform: matrix4,
child: Padding(
padding: EdgeInsets.symmetric(horizontal: 10),
child: Container(
decoration: BoxDecoration(
borderRadius: BorderRadius.circular(12),
image: DecorationImage(
image: NetworkImage(imgList[index % 2]), fit: BoxFit.fill),
),
),
),
);
}
}

推薦幾款Github上帶動畫效果的PageView

Travel Cards

  • https://github.com/gskinnerTeam/flutter_vignettes/tree/master/vignettes/parallax_travel_cards_list

Mindfullness Gooey Transition

  • https://github.com/gskinnerTeam/flutter_vignettes/tree/master/vignettes/gooey_edge

page-transformer

  • https://github.com/roughike/page-transformer

transformer_page_view

  • https://github.com/best-flutter/transformer_page_view

smooth_page_indicator

  • https://github.com/Milad-Akarie/smooth_page_indicator


更多相關閱讀:


老孟

一枚有態度的程序員

戳原文,查看Flutter系列總覽


本文分享自微信公衆號 - 老孟Flutter(lao_meng_qd)。
若有侵權,請聯繫 support@oschina.cn 刪除。
本文參與「OSC源創計劃」,歡迎正在閱讀的你也加入,一塊兒分享。

相關文章
相關標籤/搜索