Flutter 快速实现聊天会话列表效果示例详解

作者:LinXunFeng 时间:2022-10-21 09:18:31 

一、目标效果

聊天会话页的列表效果

  • 1、聊天数据不满一屏时,顶部显示所有聊天数据

  • 2、插入消息时

    • 如果最新消息紧靠列表底部时,则插入消息会使列表向上推

    • 如果不是紧靠列表底部,则固定到当前聊天位置

效果如图所示:

Flutter 快速实现聊天会话列表效果示例详解

二、原理

1、 涉及的方法

ScrollPhysics 提供了 adjustPositionForNewDimensions 方法,用于修正 ScrollViewrebuild 后的偏移量,方法声明如下

double adjustPositionForNewDimensions({
 required ScrollMetrics oldPosition,
 required ScrollMetrics newPosition,
 required bool isScrolling,
 required double velocity,
})

默认情况下,值为上一次的偏移量,即 newPosition 参数的 pixels,所以在顶部插入消息时,消息列表就会跟随滚动。

如下图所示,观察蓝色的消息条目,每播入一条消息时,所有消息会自动往上顶,而滚动视图的偏移量其实一直是没有变化的~

Flutter 快速实现聊天会话列表效果示例详解

注:值得注意的是,如果该方法返回的值与 newPositionpixels 不相等时,则会触发视图的重新布局,所以这个操作还是比较昂贵的,应尽量减少返回值的变动。

2、实现逻辑

根据上述内容我们不难推出插入消息时的效果实现原理如下:

效果返回的值
消息紧靠列表底部时,插入消息会使列表向上推直接返回 super 的值,即 newPosition 参数的 pixels
如果不是紧靠列表底部,则固定到当前聊天位置返回原本第 0 条消息的最新偏移量

下面重点说明一下第 2 点中 返回原本第0条消息的最新偏移量 的实现逻辑:

ListView 的本质是 RenderSliverList,通过 RenderSliverListfirstChild 属性拿到当前列表中渲染的首个 item

如下图,firstChild 是下标为 10item,这个 item 与预渲染区域 cacheExtent 相关,如果将其设置为 0,则 firstChild 的下标将会是 12,这个相信不难理解。

Flutter 快速实现聊天会话列表效果示例详解

所以,我们只需要在插入消息时,记录第 0 条消息的偏移量,当列表视图 rebuild 后,adjustPositionForNewDimensions 方法会被调用,此时取出第 1 条消息的偏移量,两者的差值加上 super 的值即为目标修正偏移量。

至于 聊天数据不满一屏时,顶部显示所有聊天数据 这个效果只是在切换 shrinkWrap 而已,比较简单就不在此展开讲了。

我已将上述逻辑进行了封装,集成于 flutter_scrollview_observer,接下来我们就来看看如何使用。

三、使用

现在只需三个步骤即可快速实现聊天会话列表的效果

步骤一:初始化必要的 ListObserverControllerChatScrollObserver

/// 初始化 ListObserverController
observerController = ListObserverController(controller: scrollController)
 ..cacheJumpIndexOffset = false;
/// 初始化 ChatScrollObserver
chatObserver = ChatScrollObserver(observerController)
 ..toRebuildScrollViewCallback = () {
   // 这里可以重建指定的滚动视图即可
   setState(() {});
 };

步骤二:按如下配置 ListView 并使用 ListViewObserver 将其包裹

Widget _buildListView() {
 Widget resultWidget = ListView.builder(
   physics: ChatObserverClampinScrollPhysics(observer: chatObserver),
   shrinkWrap: chatObserver.isShrinkWrap,
   reverse: true,
   controller: scrollController,
   ...
 );
 resultWidget = ListViewObserver(
   controller: observerController,
   child: resultWidget,
 );
 return resultWidget;
}

步骤三:插入或删除消息前,调用 ChatScrollObserverstandby 方法

onPressed: () {
 chatObserver.standby();
 setState(() {
   chatModels.insert(0, ChatDataHelper.createChatModel());
 });
},
...
onRemove: () {
 chatObserver.standby(isRemove: true);
 setState(() {
   chatModels.removeAt(index);
 });
},

注:示例中的 setState 都可以换成对列表视图进行局部刷新的代码

四、最后

GitHub地址: flutter_scrollview_observer

该库还实现了其它十分实用的功能,相关功能有对应的文章进行叙述

来源:https://juejin.cn/post/7152307272436154405

标签:Flutter,聊天列表,会话列表
0
投稿

猜你喜欢

  • Java去重排序之Comparable与Comparator的使用及说明

    2023-04-28 00:49:27
  • Flutter深色模式适配的实现

    2023-01-22 12:41:03
  • java实现文件下载的两种方式

    2023-11-11 06:37:14
  • Java中的 stop the world是什么呢

    2022-09-11 21:32:42
  • JAVA如何按字节截取字符串

    2023-11-25 13:31:41
  • java 非常好用的反射框架Reflections介绍

    2022-11-25 01:25:52
  • Java 线程的优先级(setPriority)案例详解

    2023-11-12 23:46:39
  • C++异常处理方式实例详解(超级详细!)

    2023-08-04 07:34:58
  • Android 加载大图及多图避免程序出现OOM(OutOfMemory)异常

    2022-05-06 18:32:06
  • 玩转Android之Drawable的使用

    2023-07-01 22:42:46
  • Java并发编程之Java内存模型

    2023-08-17 23:06:10
  • 让Java后台MySQL数据库能够支持emoji表情的方法

    2022-12-30 04:24:45
  • C#实现数独解法

    2022-10-25 18:22:43
  • Java8 Optional优雅空值判断的示例代码

    2021-09-20 09:16:26
  • Android TextSwitcher文本切换器和ViewFlipper使用详解

    2023-09-20 01:01:50
  • 使用Maven Helper解决Maven插件冲突的方法

    2023-11-08 06:07:05
  • Android编程实现VideoView循环播放功能的方法

    2021-12-05 15:31:42
  • 利用Android从0到1实现一个流布局控件

    2023-01-29 06:38:15
  • 基于Idea+Jconsole实现线程监控步骤

    2021-07-29 10:39:40
  • RocketMQ producer容错机制源码解析

    2022-05-14 13:39:41
  • asp之家 软件编程 m.aspxhome.com