当条目过少时listview某些嵌套情况下可能不会滚动(条目多时,超出一个屏幕,不会出现此问题),RefreshIndicator是根据下拉偏移量触发onRefresh操作,不能滚动自然不能下拉刷新。在listview的physice属性赋值new AlwaysScrollableScrollPhysics(),如上图,保持listview任何情况都能滚动,问题解决。
image.png
————————————————
版权声明:本文为CSDN博主「IT兔子123」的原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接及本声明。
原文链接:https://blog.csdn.net/u013425527/article/details/98086925

Flutter 下拉刷新上拉加载更多

Flutter 下拉刷新上拉加载更多 - 图2

基础页面实现

TabBar + TabBarView 实现页面切换联动(类似Android tablayout + ViewPage)效果

  • 直接上代码

    1. List <String>_titles=['湖人','勇士','雄鹿','快船','凯尔特人','马刺','76人','猛龙'];
    2. TabController _tabController;
    3. ///省略部分代码
    4. class MyHomePage extends StatefulWidget {
    5. MyHomePage({Key key, this.title}) : super(key: key);
    6. final String title;
    7. ///省略部分代码
    8. class _MyHomePageState extends State<MyHomePage> with SingleTickerProviderStateMixin{
    9. @override
    10. void initState() {
    11. super.initState();
    12. //初始化控制器
    13. _tabController = new TabController(length: _titles.length,vsync: this);
    14. }
    15. @override
    16. Widget build(BuildContext context) {
    17. return Scaffold(
    18. appBar: AppBar(
    19. leading: Icon(Icons.menu),
    20. title: buildTabBar(),
    21. //bottom: buildTabBar(),
    22. ),
    23. body: TabBarViewLayout()
    24. );
    25. }
    26. Widget buildTabBar() {
    27. return TabBar(
    28. //构造Tab集合
    29. tabs: _titles.map((String title){
    30. return Tab(
    31. text: title,
    32. );
    33. }).toList(),
    34. ///省略部分代码
    35. controller: _tabController,
    36. );
    37. }
    38. }
    39. // TabBarView Widget
    40. class TabBarViewLayout extends StatelessWidget {
    41. @override
    42. Widget build(BuildContext context) {
    43. print("TabBarViewLayout build.......");
    44. return TabBarView(
    45. controller: _tabController,
    46. children: _titles.map((String title){
    47. return TabPageView(title);
    48. }).toList(),
    49. );
    50. }
    51. }
    52. 复制代码
  • 如果代码,可以看到在AppBar这个widget的title属性中加入TabBar,也就是AppBat的title模块显示TabBar,也可在AppBar的bottom属性加入;还需要注意TabBar和TabBarView正是通过同一个controller来实现菜单切换和滑动状态同步的,最终运行结果如下,分被设置tabbar在title 和bottom属性Flutter 下拉刷新上拉加载更多 - 图3Flutter 下拉刷新上拉加载更多 - 图4

    下拉刷新,上拉加载更多实现(RefreshIndicator)

  • 下拉刷新 Flutter SDK中已经提供了一个RefreshIndicator控件,所以结合RefreshIndicator控件,让其包裹ListView控件,结合滑动监听ScrollController,并且设置头部,尾部加载更多等界面,就可以完成一个通用的下拉刷新,上拉加载更多的通用控件。首先来看看RefreshIndicator构造方法

    1. const RefreshIndicator({
    2. Key key,
    3. @required this.child, //包装一个可滚动widget
    4. this.displacement = 40.0,
    5. @required this.onRefresh, //触发刷新调用方法
    6. this.color, //指示器颜色
    7. this.backgroundColor,
    8. this.notificationPredicate = defaultScrollNotificationPredicate,
    9. this.semanticsLabel,
    10. this.semanticsValue,
    11. })
    12. 复制代码
  • RefreshIndicator包装一个可滚动widget,这里使用ListView

    1. @override
    2. Widget build(BuildContext context) {
    3. return RefreshIndicator(
    4. child: ListView.builder(
    5. ///保持ListView任何情况都能滚动,解决在RefreshIndicator的兼容问题。
    6. physics: const AlwaysScrollableScrollPhysics(),
    7. itemBuilder: (context,index){
    8. return _getItem(index);
    9. },
    10. ///根据状态返回绘制 item 数量
    11. itemCount: _getListCount(),
    12. ///滑动监听
    13. controller: _scrollController,
    14. ),
    15. onRefresh: _handleRefresh,
    16. color: Theme.of(context).primaryColor, //指示器颜色
    17. );
    18. }
    19. 复制代码
  • ListView有两个重要方法设置,一个是itemBuilder构建列表item的每一个页面,另一个构建item页面数量itemCount。首先看itemCount方法

    1. ///根据配置状态返回实际列表数量
    2. _getListCount() {
    3. ///是否需要头部
    4. if (widget.isHaveHeader) {
    5. return (items.length > 0) ? items.length + 2 : items.length + 1;
    6. } else {
    7. if (items.length == 0) {
    8. return 1;
    9. }
    10. return (items.length > 0) ? items.length + 1 : items.length;
    11. }
    12. }
    13. 复制代码
  • 该方法中,做了几种内容类型判断,如果需要头部,用Item 0 的 Widget 作为ListView的头部,列表数量大于0时,因为头部和底部加载更多选项,需要对列表数据总数+2,如果不需要头部,在数据获取为零时,固定返回数量1用于空页面呈现或者错误页面;如果有数据,加上外部加载更多选项,需要对列表数据总数+1。接着看_getItem()方法,返回对应渲染页面。

    1. ///根据配置状态返回实际列表渲染Item
    2. _getItem(int index) {
    3. if (!widget.isHaveHeader && index == items.length && items.length != 0) {
    4. return _buildProgressIndicator();
    5. } else if (widget.isHaveHeader && index == _getListCount()-1 && items.length != 0) {
    6. return _buildProgressIndicator();
    7. } else if (widget.isHaveHeader && index == 0 && items.length != 0) {
    8. return widget.headerView();
    9. } else if (!widget.isHaveHeader && items.length == 0) {
    10. ///如果不需要头部,并且数据为0,渲染空页面
    11. if(isLoading){
    12. return _buildIsLoading();
    13. }else{
    14. return _buildEmpty();
    15. }
    16. } else if(widget.isHaveHeader && items.length == 0){
    17. if(isLoading){
    18. return _buildIsLoading();
    19. }else{
    20. return _buildEmpty();
    21. }
    22. } else {
    23. return widget.renderItem(index, items[widget.isHaveHeader ? index-1 : index]);
    24. }
    25. }
    26. 复制代码
  • 该方法中,如果没有设置头部,并且数据不为0,当index等于数据长度时,渲染加载更多页面(因为index是从0开始);如果设置了头部页面,并且数据不为0,当index等于实际渲染长度 - 1时,渲染加载更多页面(在该方法判断是否已经加载到底);接着如果设置了头部widget,并且数据不为0,当index = 0 ,渲染头部widget;如果没设置头部,并且数据为0,如果当前正在刷新,渲染Loading页面,否则渲染空页面或者Error页面;同理,如果设置头部,并且数据为0,并且当前正在刷新,渲染Loading页面,否则渲染空页面或者Error页面;如果不是上面情况,则渲染正常渲染Item,如果这里有需要,可以直接返回相对位置的index,如果有头部 index 减一 ,保持不会忽略 index = 0 的数据。

  • 接着封装一个统一网络请求方法,外部请求安装固定格式的 Map 将数据返回给下拉刷新上拉加载更多widget,达到通用的目的。

    1. //网络请求获取数据 isRefresh 是否为下拉刷新
    2. Future<List> makeHttpRequest(bool isRefresh) async {
    3. if (widget.requestApi is Function) {
    4. Map listObj = new Map<String, dynamic>();
    5. if(isRefresh){
    6. //下拉刷新
    7. listObj = await widget.requestApi({'pageIndex': 0});
    8. }else{
    9. //上拉加载更多
    10. listObj = await widget.requestApi({'pageIndex': _pageIndex});
    11. }
    12. _pageIndex = listObj['pageIndex'];
    13. _pageTotal = listObj['total'];
    14. return listObj['list'];
    15. } else {
    16. return Future.delayed(Duration(seconds: 2), () {
    17. return [];
    18. });
    19. }
    20. }
    21. 复制代码
  • 基础东西写好了,loading 加载动画这里直接就使用现成的轮子好了,推荐一个loading库,flutter_spinkit

  • 贴上loading加载代码(更多实现细节请看文末demo地址代码)

    1. Widget _buildIsLoading() {
    2. return Container(
    3. width: MediaQuery.of(context).size.width,
    4. height: MediaQuery.of(context).size.height*0.85,
    5. child: new Center(
    6. child: Column(
    7. crossAxisAlignment: CrossAxisAlignment.center,
    8. mainAxisAlignment: MainAxisAlignment.center,
    9. children: <Widget>[
    10. Row(
    11. mainAxisAlignment: MainAxisAlignment.spaceEvenly,
    12. children: <Widget>[
    13. SpinKitCircle(size: 55.0, color: Theme.of(context).primaryColor),
    14. ],
    15. ),
    16. Padding(
    17. child: Text("正在加载..",
    18. style: TextStyle(color: Colors.black54, fontSize: 15.0)),
    19. padding: EdgeInsets.all(15.0),)
    20. ],)
    21. ));
    22. }
    23. 复制代码
  • 最后,通过构造方法设置设置需要加载的item值和是否支持下拉刷新和上来加载更多等,灵活配置控件

    1. // 模块item
    2. final renderItem;
    3. //数据获取方法
    4. final requestApi;
    5. //头部
    6. final headerView;
    7. //是否添加头部 默认不添加
    8. final bool isHaveHeader;
    9. //是否支持下拉刷新 默认可以下拉刷新
    10. final bool isCanRefresh;
    11. //是否支持下拉加载更多 默认可以加载更多
    12. final bool isCanLoadMore;
    13. const RefreshPage({@required this.requestApi,
    14. @required this.renderItem,
    15. this.headerView,
    16. this.isHaveHeader = false,
    17. this.isCanRefresh = true,
    18. this.isCanLoadMore = true })
    19. : assert(requestApi is Function),
    20. assert(renderItem is Function),
    21. super();
    22. 复制代码

    最终demo 效果

    Flutter 下拉刷新上拉加载更多 - 图5Flutter 下拉刷新上拉加载更多 - 图6Flutter 下拉刷新上拉加载更多 - 图7

    Demo 地址

  • github.com/maoqitian/f…

  • Flutter完整开源项目: github.com/maoqitian/f…

    About me

    blog:

  • 个人博客

  • 掘金
  • 简书
  • Github

    mail:

  • maoqitian@gmail.com

  • maoqitian068@163.com