Flutter小记-TextField遇到emoji表情时字数问题

本文最后更新于:2023年2月3日 下午

本文记录Flutter开发过程中,当TextField输入的字符包含emoji,最大字数限制出现统计的问题及解决方案。

需求

  1. 限制TextField的最大字数
  2. counterText显示当前的字数
  3. 当前字数必须小于等于最大字数

问题描述

在使用文本框组件TextField时,已设置了最大字符数限制为MAX_LENGTH。当输入框中同时存在文字和emoji表情时,使用TextEditingController.text.length统计的字符长度,得到的数值不对。

具体情况是每一个emoji表情,会使length长度加2,因此导致统计到当前的字数长度length,会超过MAX_LENGTH

IMG_error

问题代码demo

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
import 'package:flutter/material.dart';

void main() => runApp(const MyApp());

class MyApp extends StatelessWidget {
const MyApp({Key? key}) : super(key: key);
@override
Widget build(BuildContext context) {
return MaterialApp(
title: 'TextField字数统计',
theme: ThemeData(
primarySwatch: Colors.blue,
),
home: const MyHomePage(title: 'TextField字数统计'),
);
}
}

class MyHomePage extends StatefulWidget {
const MyHomePage({Key? key, required this.title}) : super(key: key);
final String title;

@override
State<MyHomePage> createState() => _MyHomePageState();
}

class _MyHomePageState extends State<MyHomePage> {
int _counter = 0;
int MAX_LENGTH = 10;
var textFieldController = TextEditingController();

/// 页面主体
Widget getBodyContainer(){
return TextField(
maxLength: MAX_LENGTH,
controller: textFieldController,
);
}


Widget getUpdateButton(){
return MaterialButton(
//TODO bug出现的位置
child: Text("点击更新--当前字数:${textFieldController.text.length}/${MAX_LENGTH}"),
onPressed: (){
setState(() {

});
},
);
}

@override
Widget build(BuildContext context) {
return Scaffold(
appBar: AppBar(
title: Text(widget.title),
),
body: Center(
child: Column(
mainAxisAlignment: MainAxisAlignment.center,
children: <Widget>[
getBodyContainer(),
getUpdateButton()
],
),
),
);
}
}

解决过程

先对问题进行排查,首先设置maxLength是达到了我们的预期的需求1和需求2的效果。只是使用TextEditingController.text.length统计当前字数,遇到emoji表情时,得到的数据不对。

github查找issues

首先在github上找到了相关的问题描述。都是反映了类似的问题。

https://github.com/flutter/flutter/issues/55670

https://github.com/flutter/flutter/pull/55819

但是这些问题的回答,只是在解释其合理性。通过其他人的讨论中,也没有获得有帮助的信息。

textfeild question 1

查找相关博客

接着在又在dev.to上找到了一篇相关的博客文章: Flutter UTF8 TextField length limiter and char counter。这篇博客的解决办法是计算每个字符的byte长度,以此来限制输入长度。

在项目中试了后,发现其满足了他提出的需求,但并不满足我的需求。

但是这篇文章给了我们一些有用的信息。假设我们在TextField里输入了一些文字和emoji表情包。并通过TextEditingController.text获得了输入框内的文字text,那么它会有以下的情况:

文字类型 text.length utf8.encode(text).length text.charaters.length
英文 1 1 1
汉字 1 3 1
emoji表情 2 2 1

由此可见,计算当前输入的字数,我们引用的不应该是textlength,而是charaters.length,即字符数量。

flutter.dev查找官方文档

对于以上的问题,官方文档里也找到了解释:

This formatter does not currently count Unicode grapheme clusters (i.e. characters visible to the user), it counts Unicode scalar values, which leaves out a number of useful possible characters (like many emoji and composed characters), so this will be inaccurate in the presence of those characters. If you expect to encounter these kinds of characters, be generous in the maxLength used.

该格式化程序目前不计算Unicode字符群(即对用户可见的字符),它计算Unicode标量值,这遗漏了一些有用的可能字符(如许多表情符号和组合字符),因此在这些字符出现时,这将是不准确的。如果您希望遇到这类字符,请尽量使用maxLength。

这段解释了为什么在输入框中输入包含中文和emoji表情,我们为TextField设置了maxLength字段,限制字数的功能能够正常生效,但使用TextEditingController.text.length统计当前字数,会不符合预期的原因。

代码实现

找到原因后,我们对代码进行修改

1
2
3
4
5
6
7
8
9
10
11
12
//TODO bug出现的位置
Widget getUpdateButton(){
return MaterialButton(
//对这里进行修改
child: Text("点击更新--当前字数:${textFieldController.text.charaters.length}/${MAX_LENGTH}"),
onPressed: (){
setState(() {

});
},
);
}

测试效果

尽管占用的字节大小不同,emoji表情将被视作一个字符。取到的当前字数与TextFieldcounterText的字数保持一致。输入框内可输入的字数也不会超过上限。

IMG_success


Flutter小记-TextField遇到emoji表情时字数问题
http://blog.cnctema.pub/post/zh-CN/5108a7795952/
作者
cnctema
发布于
2022年10月17日
许可协议