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 - 列表滚动定位超强辅助库,墙裂推荐!🔥
    • Flutter - 快速实现聊天会话列表的效果,完美💯
    • Flutter - 聊天输入框更新文本时的必备优化点🔖
      • 一、问题
      • 二、示例代码
      • 三、探索与解决
        • 1、输入框
        • 2、滚动条
      • 四、最后
    • 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-11-13
目录

Flutter - 聊天输入框更新文本时的必备优化点🔖

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

# 一、问题

可以看出一些问题:

  • 在关闭键盘的情况下,追加表情内容导致换行时,输入框并不会跟随滚动,而键盘输入文字却可以
  • 滚动条在键盘出现时正常,关闭键盘时滚动条无法滑到底部

# 二、示例代码

TextField 代码

_buildTextField() {
  return Scrollbar(
    controller: scrollController,
    isAlwaysShown: true,
    child: TextField(
      key: textFieldKey,
      autofocus: true,
      decoration: const InputDecoration(
        border: InputBorder.none,
        isCollapsed: true,
      ),
      controller: editingController,
      scrollController: scrollController,
      keyboardType: TextInputType.multiline,
      maxLines: 8,
      minLines: 1,
      showCursor: true,
      readOnly: true,
    ),
  );
}

更新输入框内容的逻辑

var result = editingController.text;
// 新增的内容
String content = fetchInsertContent();
int selectionBefore = editingController.selection.start;
if (selectionBefore < 0) selectionBefore = 0;
// 更新内容后,光标的位置
int selectionAfter = selectionBefore + content.length;

result = (result.split('')..insert(selectionBefore, content)).join('');
editingController.text = result;

// 设置光标
editingController.selection = TextSelection.fromPosition(TextPosition(
  offset: selectionAfter,
));

# 三、探索与解决

# 1、输入框

当键盘输入内容时会自动滚动到底部,既然跟滚动相关,第一时间就会想到 ScrollController 的 animateTo 和 jumpTo 这两个方法,打个断点后,用键盘输入文字看看,发现果然走了断点

从调用栈里可以看到,animateTo 方法是在 EditableTextState 的 _scheduleShowCaretOnScreen 方法里调用的,该方法正是用于处理在键盘输入文字后,使 TextField 的内容自动滚动到适当的位置,保持可见。该方法在每次输入文字时都会调用,其内部有相关的滚动偏移计算逻辑。

那好了,我们现在主要就是看如何调用它,因为其是私有方法,所以需要找其它可帮我们间接调用 _scheduleShowCaretOnScreen 方法的公开方法。

查看调用 _scheduleShowCaretOnScreen 的方法有哪些后,这里我找到了 userUpdateTextEditingValue,以下对两个形参进行说明

class TextEditingValue {
  const TextEditingValue({
    this.text = '',
    this.selection = const TextSelection.collapsed(offset: -1),
    this.composing = TextRange.empty,
  })
...
属性 描述
text 当前文本
selection 当前选中的文本下标范围
composing 待组合内容的下标范围,在中文输入法中很常见,如下图中的 lin xun feng

SelectionChangedCause:表明触发变更文本选中范围的原因,常见的几种方式如下代码所示

/// Indicates what triggered the change in selected text (including changes to
/// the cursor location).
enum SelectionChangedCause {
  /// The user tapped on the text and that caused the selection (or the location
  /// of the cursor) to change.
  tap,

  /// The user tapped twice in quick succession on the text and that caused
  /// the selection (or the location of the cursor) to change.
  doubleTap,

  /// The user long-pressed the text and that caused the selection (or the
  /// location of the cursor) to change.
  longPress,

  /// The user force-pressed the text and that caused the selection (or the
  /// location of the cursor) to change.
  forcePress,

  /// The user used the keyboard to change the selection or the location of the
  /// cursor.
  ///
  /// Keyboard-triggered selection changes may be caused by the IME as well as
  /// by accessibility tools (e.g. TalkBack on Android).
  keyboard,

  /// The user used the selection toolbar to change the selection or the
  /// location of the cursor.
  ///
  /// An example is when the user taps on select all in the tool bar.
  toolbar,

  /// The user used the mouse to change the selection by dragging over a piece
  /// of text.
  drag,

  /// The user used iPadOS 14+ Scribble to change the selection.
  scribble,
}

该参数为可选参数,对于我们这种使用 TextEditingController 对输入框内容进行修改的方式,传 null 即可。

需要调用的方式已经确定,接下来就是要获取 EditableTextState,我们先来看一下 TextField 的源码,因其为 StatefulWidget,所以需要找到对应的 State, 即 _TextFieldState

class _TextFieldState extends State<TextField> with RestorationMixin implements TextSelectionGestureDetectorBuilderDelegate, AutofillClient {

    @override
    final GlobalKey<EditableTextState> editableTextKey = GlobalKey<EditableTextState>();

    @override
    Widget build(BuildContext context) {
    ...

        Widget child = RepaintBoundary(
          child: UnmanagedRestorationScope(
            bucket: bucket,
            child: EditableText(
              key: editableTextKey,
              readOnly: widget.readOnly || !_isEnabled,
        ...
    }
}

可以看到,最终 build 方法返回的是嵌套了 EditableText 的 Widget,我们只要获取到该 EditableText 的 State(即 EditableTextState) 就可以调用 userUpdateTextEditingValue 方法。

现在比较棘手的是,TextFieldState 是私有的,无法直接访问到,但是我们可以换种方式,直接遍历 Element 从而获取到 EditableTextState

void visitor(Element element) {
  if (element.widget is EditableText) {
    final editableText = element.widget as EditableText;
    final editableTextState =
        (editableText.key as GlobalKey<EditableTextState>)
            .currentState;
            
    // 找到 EditableTextState,调用 userUpdateTextEditingValue 方法
    editableTextState?.userUpdateTextEditingValue(
      TextEditingValue(...),
      null,
    );
    return;
  }
  element.visitChildren(visitor);
}

textFieldKey.currentContext?.visitChildElements(visitor);

至此输入框便可实现我们想要的效果了

注:

  1. 这里大家可以自行对遍历的方法和 userUpdateTextEditingValue 的调用做下调整,毕竟每次变更输入框内容就遍历不好~
  2. 设置光标位置的代码不需要了,userUpdateTextEditingValue 方法内部会做这个事情

# 2、滚动条

只有 readOnly 为 true 时才会出现底部间距,对比 Scrollbar 的 updateScrollbarPainter 方法中的 padding 发现端倪

readOnly: true 的情况:

readOnly: true

readOnly: false 的情况:

readOnly: false

可以看到 readOnly: true 的情况下 padding 的 bottom 为 34.0,做 iOS 开发的小伙伴看到这个数值一定会觉得很熟悉,没错,就是底部安全区域的高度,而当键盘出现的时候,Scrollbar 是不可能接触到底部安全区域的,所以 padding 的 bottom 为 0。

这个时候我们只需要想办法让 Scrollbar 获取到的 padding 的 bottom 为 0 即可。这个不多说直接上代码:

MediaQuery.removePadding(
  context: context,
  child: _buildTextField(),
  removeTop: true,
  removeBottom: true, // 重点
);

# 四、最后

示例代码已同步至 GitHub: LinXunFeng/flutter_demo (opens new window)

#Dart#Flutter
Flutter - 快速实现聊天会话列表的效果,完美💯
Flutter - 我给官方提PR,解决run命令卡住问题 😃

← Flutter - 快速实现聊天会话列表的效果,完美💯 Flutter - 我给官方提PR,解决run命令卡住问题 😃→

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