FSA全栈行动 FSA全栈行动
首页
  • 移动端文章

    • Android
    • iOS
    • Flutter
  • 学习笔记

    • 《Kotlin快速入门进阶》笔记
    • 《Flutter从入门到实战》笔记
    • 《Flutter复习》笔记
前端
后端
  • 学习笔记

    • 《深入浅出设计模式Java版》笔记
  • 逆向
  • 分类
  • 标签
  • 归档
  • LinXunFeng
  • GitLqr

公众号:FSA全栈行动

记录学习过程中的知识
首页
  • 移动端文章

    • Android
    • iOS
    • Flutter
  • 学习笔记

    • 《Kotlin快速入门进阶》笔记
    • 《Flutter从入门到实战》笔记
    • 《Flutter复习》笔记
前端
后端
  • 学习笔记

    • 《深入浅出设计模式Java版》笔记
  • 逆向
  • 分类
  • 标签
  • 归档
  • LinXunFeng
  • GitLqr
  • AndroidUI

  • Android第三方SDK

  • Android混淆

  • Android仓库

  • Android新闻

  • Android系统开发

  • Android源码

  • Android注解AOP

  • Android脚本

  • AndroidTv开发

  • AndroidNDK

  • Android音视频

  • Android热修复

  • Android性能优化

  • Android云游戏

  • Android插件化

  • iOSUI

  • iOS工具

  • iOS底层原理与应用

  • iOS组件化

  • iOS音视频

  • iOS疑难杂症

  • iOS之Swift

  • iOS之RxSwift

  • iOS开源项目

  • iOS逆向

  • Flutter开发

    • Dart - 抽象类的实例化
    • Flutter - 打印好用的Debug日志
    • Flutter - 混合开发
    • Flutter - 解决混合开发iOS脚本打包遇到的问题
    • Flutter - 低版本在iOS14上遇到的问题与解决方案
    • Flutter - 解决原生弹窗的触摸事件被Flutter响应的问题
    • Flutter - 实现列表上下拉切换header
    • Flutter - 获取ListView当前正在显示的Widget信息
    • Flutter - 列表滚动定位超强辅助库,墙裂推荐!🔥
      • 一、痛点
      • 二、优点
      • 三、使用
        • 1、ListView
        • 2、GridView
        • 3、CustomSrollView
      • 四、说明
        • 1、ViewObserver 的选择
        • 2、isFixedHeight
        • 3、sliverContext 是否需要传
      • 五、可现实的功能
        • 1、获得当前视窗中的子部件信息
        • 2、视频列表自动播放
        • 3、模块定位
      • 六、最后
    • Flutter - 快速实现聊天会话列表的效果,完美💯
    • Flutter - 聊天输入框更新文本时的必备优化点🔖
    • Flutter - 我给官方提PR,解决run命令卡住问题 😃
    • Flutter - 探索run命令到底做了什么 🤔
    • Flutter - 引擎调试(iOS篇)🛠
    • Flutter - 引擎调试bug到提交PR实战 🐞
    • Flutter - 船新升级😱支持观察第三方构建的滚动视图💪
    • Flutter - 瀑布流交替播放视频 🎞
    • Flutter - IM保持消息位置大升级(支持ChatGPT生成式消息) 🤖
    • Flutter - 滚动视图中的表单防遮挡 🗒
    • Flutter - 秒杀1/2曝光统计 📊
    • 一天内加入 Flutter 和 FlutterCandies 两大组织是什么体验 🧐
    • Flutter - 如何快速搓一个微信通讯录列表(azlist) 📓
    • Flutter - 混编项目集成Shorebird热更新🐦(安卓篇)
    • Flutter - 混编项目集成Shorebird热更新🐦(iOS篇)
    • Flutter - 解决返回原生页面时dispose方法未被触发的问题 🐞
    • Flutter - 升级3.19之后页面多次rebuild?🤨
    • Flutter - 热更新 Shorebird 1.0 正式版来了 🐦
    • Flutter - 使用Pigeon实现视频缓存插件 🐌
    • Flutter - 轻松搞定屏幕旋转功能 😎
    • Flutter - 解决Connection closed before full header was received
    • Flutter - 实现聊天键盘与功能面板的丝滑切换 🍻
    • Flutter - 支持观察NestedScrollView,兼容性更强 😈
    • Flutter - 聊天键盘与面板丝滑切换的强势升级 🍻
    • Flutter - 升级到3.24后页面还会多次rebuild吗?🧐
    • Flutter - 轻松实现PageView卡片偏移效果
    • Flutter - 轻松搞定炫酷视差(Parallax)效果
    • Flutter - 危!3.24版本苹果审核被拒!
    • Flutter - 子部件任意位置观察滚动数据
    • Flutter - iOS编译加速
    • Flutter - Xcode16 还原编译速度
  • 移动端
  • Flutter开发
LinXunFeng
2022-08-09
目录

Flutter - 列表滚动定位超强辅助库,墙裂推荐!🔥

欢迎关注微信公众号:[FSA全栈行动 👋]

# 一、痛点

痛点一:Flutter 官方提供了 ScrollController,调用下方两个方法可以滚动到指定偏移处

void jumpTo(double value)
Future<void> animateTo(
  double offset, {
  required Duration duration,
  required Curve curve,
})

但是官方没有提供滚动到指定下标位置的功能

痛点二:

为了解决痛点一,业内有很多第三方库实现了这个功能,其中最知名的莫过于谷歌的 scrollable_positioned_list (opens new window)

而这些库都有一些相同的问题:

  • 侵入性强,必须使用他们提供的 Widget 来构建列表视图
  • 不支持 GridView

对此,我决定自己编写一个库(flutter_scrollview_observer (opens new window))来实现这个功能,以及解决以上的问题。

# 二、优点

  1. 侵入性低,不会限制你的列表视图实现,只要求将列表视图做为 ViewObserver 的 child,如果不想用了,直接移除掉 ViewObserver 即可。
  2. 基于【不会限制你的列表视图实现】这一点, flutter_scrollview_observer 支持所有的列表Widget,如:ListView、GridView,甚至 CustomSrollView
  3. 防抖动,如果在滚动到指定下标时,列表尾部的可滚动区域已不足以支撑滚动到相应位置,则会自动滚动到最底部,避免回弹的问题

下面正式介绍一下这个库的使用

# 三、使用

# 1、ListView

正常创建和使用 ScrollController 实例

这里你可以使用 ListView 或者 CustomSrollView 的 SliverList

ScrollController scrollController = ScrollController();

ListView _buildListView() {
  return ListView.separated(
    controller: scrollController,
    ...
  );
}

创建 ListObserverController 实例并将其传递给 ListViewObserver

ListObserverController observerController = ListObserverController(controller: scrollController);

ListViewObserver(
  controller: observerController,
  child: _buildListView(),
  ...
)

注意:创建 ListObserverController 实例时需要传入列表的 ScrollController 实例。 这一步很重要,因为 flutter_scrollview_observer 内部实现原理是基于官方 ScrollController 提供的 jumpTo 和 animateTo 方法

现在即可滚动到指定下标位置了

// 无动画滚动至下标位置
observerController.jumpTo(index: 100)

// 动画滚动至下标位置
observerController.animateTo(
  index: 100,
  duration: const Duration(seconds: 1),
  curve: Curves.ease,
);

# 2、GridView

由于内容与 ListView 基本一致,所以这里就折叠了,需要的小伙伴可点击【展开查看】

展开查看

正常创建和使用 ScrollController 实例

Widget _buildGridView() {
  return GridView.builder(
    ...
    controller: scrollController,
    ...
  );
}

创建 GridObserverController 实例并将其传递给 GridViewObserver

GridObserverController observerController = GridObserverController(controller: scrollController);

GridViewObserver(
  controller: observerController,
  child: _buildGridView(),
)

现在即可滚动到指定下标位置了

observerController.jumpTo(
  index: 40,
);

observerController.animateTo(
  index: 40,
  duration: const Duration(seconds: 1),
  curve: Curves.ease,
);

# 3、CustomSrollView

支持 SliverList 和 SliverGrid 混合使用的情况!

如上图所示,CustomSrollView 的 slivers 中包含有 SliverList 和 SliverGrid

  • 点击右下角第一个按钮可滚动到 SliverList 的第 29 行
  • 点击第二个按钮可滚动到 SliverGrid 的第 10 的子部件所在行
展开查看

正常创建和使用 ScrollController 实例

Widget _buildScrollView() {
  return CustomScrollView(
    controller: scrollController,
    // scrollDirection: Axis.horizontal,
    slivers: [
      _buildSliverListView(),
      _buildSliverGridView(),
    ],
  );
}

正常创建你的 SliverList 和 SliverGrid,并将它们的 BuildContext 记录起来

Widget _buildSliverListView() {
  return SliverList(
    delegate: SliverChildBuilderDelegate(
      (ctx, index) {
        _sliverListCtx ??= ctx;
        ...
      },
      ...
    ),
  );
}

Widget _buildSliverGridView() {
  return SliverGrid(
    ...
    delegate: SliverChildBuilderDelegate(
      (BuildContext context, int index) {
        _sliverGridCtx ??= context;
        ...
      },
      ...
    ),
  );
}

创建 SliverObserverController 实例并将其传递给 SliverViewObserver

注意:在 sliverListContexts 里返回刚才记录的两个 Sliver 的 BuildContext

SliverObserverController observerController = SliverObserverController(controller: scrollController);

SliverViewObserver(
  controller: observerController,
  child: _buildScrollView(),
  sliverListContexts: () {
    return [
      if (_sliverListCtx != null) _sliverListCtx!,
      if (_sliverGridCtx != null) _sliverGridCtx!,
    ];
  },
  ...
),

现在即可滚动到指定下标位置了

observerController.animateTo(
  sliverContext: _sliverListCtx,
  index: 29,
  duration: const Duration(milliseconds: 300),
  curve: Curves.easeInOut,
);

observerController.animateTo(
  sliverContext: _sliverGridCtx,
  index: 10,
  duration: const Duration(milliseconds: 300),
  curve: Curves.easeInOut,
);

# 四、说明

# 1、ViewObserver 的选择

建议使用相应的 ViewObserver:

  • 如果是 ListView 或者纯 SliverList,建议使用 ListViewObserver
  • 如果是 GridView 或者纯 SliverGrid,建议使用 GridViewObserver
  • 如果是 SliverList 和 SliverGrid,则使用 SliverViewObserver

这样在 onObserve 回调中拿到数据模型后不用再对其进行类型判断与转换。

# 2、isFixedHeight

子部件为固定高度的情况下,请设置 isFixedHeight 为 true,以提升性能!

observerController.jumpTo(
  index: 100, 
  isFixedHeight: true
);

observerController.animateTo(
  index: 100,
  isFixedHeight: true,
  duration: const Duration(seconds: 1),
  curve: Curves.ease,
);

# 3、sliverContext 是否需要传

jumpTo({
  required int index,
  BuildContext? sliverContext,
  bool isFixedHeight = false,
})

animateTo({
  required int index,
  required Duration duration,
  required Curve curve,
  BuildContext? sliverContext,
  bool isFixedHeight = false,
})

如上,jumpTo 和 animateTo 方法中都有一个 sliverContext 参数,这是用来指定哪个 sliver 进行滚动操作的。

如果不传 sliverContext,则默认会是列表中的第一个 sliver,只有你想让非第一个 sliver 进行滚动的时候才需要传该值

# 五、可现实的功能

# 1、获得当前视窗中的子部件信息

如上图控制台所示,在滚动的过程中可以实时获取正在显示的子部件的数据。

# 2、视频列表自动播放

这种功能很常见,在列表滚动的时候,指定区域内的被命中的视频便会自动进行播放。

# 3、模块定位

一般详情页中都会有的功能,在列表视图滚动的时候,顶部定位导航视图跟着更新下标,点击导航视图某个下标则会定位到对应的模块

# 六、最后

如果这个库对你很有帮助,请不吝给个 star 吧

GitHub: flutter_scrollview_observer (opens new window)

#Dart#Flutter
Flutter - 获取ListView当前正在显示的Widget信息
Flutter - 快速实现聊天会话列表的效果,完美💯

← Flutter - 获取ListView当前正在显示的Widget信息 Flutter - 快速实现聊天会话列表的效果,完美💯→

最近更新
01
Flutter - Xcode16 还原编译速度
04-05
02
AI - 免费的 Cursor 平替方案
03-30
03
Android - 2025年安卓真的闭源了吗
03-28
更多文章>
Theme by Vdoing | Copyright © 2020-2025 FSA全栈行动
  • 跟随系统
  • 浅色模式
  • 深色模式
  • 阅读模式
×