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信息
      • 一、概述
      • 二、应用场景
        • 1、获取最顶部的子部件信息
        • 2、视频列表自动播放
        • 3、模块定位
      • 三、使用
        • 1、基本使用
        • 2、手动触发
        • 3、子部件信息
    • Flutter - 列表滚动定位超强辅助库,墙裂推荐!🔥
    • 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-05-29
目录

Flutter - 获取ListView当前正在显示的Widget信息

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

# 一、概述

Flutter 中的 ListView 相信大家都用的很熟了,不过有没有人遇到过一些这样的需求:

  • 详情页滚动到某一指定模块后,停止滚动并根据该指定模块的大小弹出全屏新手引导
  • 详情页在滚动过程中,顶部的模块定位导航栏需要及时更新指示器下标
  • 视频列表在滚动过程中,适当位置的子部件会自动进行播放视频
  • 等等

在日常开发过程中这种类似的功能需求还是蛮多的,因此我封装了一个库:flutter_scrollview_observer (opens new window)

相信可以很好的帮助大家解决这些问题 😃

# 二、应用场景

下面我们来看看常见的应用场景:

# 1、获取最顶部的子部件信息

可以获取当前的第一个子部件和所有正在显示的子部件信息

# 2、视频列表自动播放

当子部件进入列表中间区域时,自动播放视频

# 3、模块定位

当滚动到一些特定模块时,顶部的 TabBar 的指示器切换到对应模块 tab

# 三、使用

# 1、基本使用

创建 ListView,并在其 builder 回调中,将 SliverListView 的 BuildContext 记录起来

BuildContext? _sliverListViewContext;
...
ListView _buildListView() {
  return ListView.separated(
    itemBuilder: (ctx, index) {
      // 在 builder 回调中,将 BuildContext 记录起来
      if (_sliverListViewContext != ctx) {
        _sliverListViewContext = ctx;
      }
      return _buildListItemView(index);
    },
    separatorBuilder: (ctx, index) {
      return _buildSeparatorView();
    },
    itemCount: 50,
  );
}

注:在使用过程中,需要记录 SliverListView 的 BuildContext,ListView 最终也是使用 SliverListView 来进行布局的

构建 ListViewObserver

  • child: 将构建的 ListView 做为 ListViewObserver 的子部件
  • sliverListContexts: 该回调中需要返回被观察的 ListView 的 BuildContext
  • onObserve: 该回调可以监听到当前正在显示的子部件的相关信息
ListViewObserver(
  child: _buildListView(),
  sliverListContexts: () {
    return [if (_sliverListViewContext != null) _sliverListViewContext!];
  },
  onObserve: (resultMap) {
    final model = resultMap[_sliverListViewContext];
    if (model == null) return;

    // 打印当前正在显示的第一个子部件
    print('firstChild.index -- ${model.firstChild.index}');

    // 打印当前正在显示的所有子部件下标
    print('displaying -- ${model.displayingChildIndexList}');
  },
)

除了上述几个常用参数外,还有:

  • leadingOffset:顶部偏移,当列表的视窗会固定被其它视图挡住时使用
  • dynamicLeadingOffset:动态顶部偏移,当列表的视窗会动态被其它视图挡住时使用

这里看一下图就明白了

// 导航栏半透明
flutter: firstChild.index -- 0
flutter: displaying -- [0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11]
flutter: firstChild.index -- 0
flutter: displaying -- [0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12]

// 导航栏完全不透明
flutter: firstChild.index -- 2
flutter: displaying -- [2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12]

滚动过程会改变顶部的导航栏透明度,在这个前提下:

  1. 当半透明时,我们希望列表的所有可见子部件从最顶部开始算起
  2. 当完成不透明时,我们希望列表的所有可见子部件从导航栏底部开始算起
ListViewObserver(
  ...
  dynamicLeadingOffset: () {
    if (_navBgAlpha < 1) {
      return 0;
    }
    return _safeAreaPaddingTop + _navContentHeight;
  },
  ...
),
  • toNextOverPercent:内部逻辑在取到第一个子部件后,如果该子部件被挡住的大小与自身大小的比例超过了该值,则会取下一个子部件为第一个子部件。

# 2、手动触发

默认是 ListView 在滚动的时候才会观察到相关数据。

如果需要在非滚动状态下进行一次观察,可以使用 ListViewOnceObserveNotification 进行手动触发

ListViewOnceObserveNotification().dispatch(_sliverListViewContext);

注:如果频繁触发,且观察结果相同,则 onObserve 只会回调一次

# 3、子部件信息

观察到的模型数据:

class ListViewObserveModel {
  /// 第一个子部件模型数据
  final ListViewObserveDisplayingChildModel firstChild;

  /// 正在显示的所有子部件模型数据
  final List<ListViewObserveDisplayingChildModel> displayingChildModelList;

  /// 正在显示的所有子部件下标
  List<int> get displayingChildIndexList =>
      displayingChildModelList.map((e) => e.index).toList();
}

子部件模型数据:

class ListViewObserveDisplayingChildModel {
  /// 子部件下标
  final int index;

  /// 子部件的 RenderObject
  final RenderBox renderObject;
}

GitHub: LinXunFeng/flutter_scrollview_observer (opens new window)

#Dart#Flutter
Flutter - 实现列表上下拉切换header
Flutter - 列表滚动定位超强辅助库,墙裂推荐!🔥

← Flutter - 实现列表上下拉切换header Flutter - 列表滚动定位超强辅助库,墙裂推荐!🔥→

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