Flutter ListView中的TextField上下滑动内容丢失TextField关闭软键盘时会清除掉输入的内容
  xx2YH4ad7R0N 2023年11月02日 83 0


场景:屏幕里有一个三个TextField,放在ListView里,如下图

Flutter ListView中的TextField上下滑动内容丢失TextField关闭软键盘时会清除掉输入的内容_ide

 当输入内容后将ListVIew上滚动,三个TextField滚动滑出界面后,再把ListView滚动回顶部,只有焦点所在的TextField里的内容不会被清除,其他两个TextField的内容不见了,如下所示

Flutter ListView中的TextField上下滑动内容丢失TextField关闭软键盘时会清除掉输入的内容_搜索_02

原因及解决方案

造成此问题的原因,经过log验证,应该是TextField作为一个Widget,被滑出屏幕,又滑回来,没有焦点的Widget的initState和build方法会被重新执行,也就是Widget会被重建,重建时原有的状态不会自动恢复,里面的文字自然就消失了。

解决方法是把TextField封装成一个StatefulWidget,然后用AutomaticKeepAliveClientMixin来保证TextField不会被回收。

核心代码如下

class _TextFieldWidgetState extends State<TextFieldWidget> with AutomaticKeepAliveClientMixin  {//关键代码:AutomaticKeepAliveClientMixin保存状态
Function callback;
String title;

@override
bool get wantKeepAlive => true; //关键代码:重写get wantKeepAlive并设置为true

@override
void initState() {
// TODO: implement initState
super.initState();
}

@override
Widget build(BuildContext context) {

return TextField(
inputFormatters: [PrecisionLimitFormatter(2)],
keyboardType: TextInputType.numberWithOptions(),
decoration: InputDecoration(
contentPadding: const EdgeInsets.symmetric(vertical: 10.0,horizontal: 10.0),
labelText: widget.title, // 文字提示
border: OutlineInputBorder(
borderRadius: BorderRadius.circular(15),
),
filled: true, // 是否填充背景
fillColor: Colors.white, // 填充颜色
),
style: TextStyle(fontSize: ScreenUtil().setSp(TextStyleManager.TextEditSize),color: TextStyleManager.NumColor),//输入文本的样式
onChanged: (value){
widget.callback(value);
}, // 当点击确定时调用的函数2
);
}
}

TextField关闭软键盘时会清除掉输入的内容

Flutter ListView中的TextField上下滑动内容丢失TextField关闭软键盘时会清除掉输入的内容_android_03


如上图所示,在TextField中输入内容后,关闭软键盘时都会清空掉输入框里的内容

代码如下

@override
Widget build(BuildContext context) {
print('ChangePasswordForm Widget build(BuildContext context)');
TextEditingController controller = TextEditingController();
return BlocListener<ChangePasswordBloc, ChangePasswordState>(
listener: stateListener,
child: Scaffold(
backgroundColor: ThemeColors.normalBackground,
body: SingleChildScrollView(
child: Column(
children: [
......
Row(
children: [
Expanded(
child: TextField(
key: const Key('aaaaaaaaaaaa'),
controller: controller,
maxLines: 1,
cursorColor: ThemeColors.primaryValue,
style: AppTextStyle.mainText(),
decoration: InputDecoration(
hintStyle: AppTextStyle.promptText(),
contentPadding: EdgeInsets.only(left: 10,right: 10),
hintText: '请输入..',
border: InputBorder.none,
)),
),
GestureDetector(
child: Text(
'搜索',
style: TextStyle(color: ThemeColors.primaryValue),
),
onTap: () {
FocusScope.of(context).requestFocus(FocusNode());
print('搜索内容:${controller?.text}');
},
)
],
),
],
),
),
));
}

我需要在点击搜索时,获取的输入的内容去搜索,所以就在TextField中传入了controller

后来观察打印的日志,发现每次弹出、收起软键盘时。build()方法都会调用,每次build的时候,controller都是个新的对象。之前输入的内容,新的controller不会持有。所以把TextEditingController的初始化放到了initState()方法里,如下所示

TextEditingController _controller;
@override
void initState() {
print('ChangePasswordForm initState()');
_controller = TextEditingController();
}

然后TextField的controller参数传入这个_controller
运行后收起软键盘,之前输入的内容就不会被清空了

Flutter ListView中的TextField上下滑动内容丢失TextField关闭软键盘时会清除掉输入的内容_android_04

 

【版权声明】本文内容来自摩杜云社区用户原创、第三方投稿、转载,内容版权归原作者所有。本网站的目的在于传递更多信息,不拥有版权,亦不承担相应法律责任。如果您发现本社区中有涉嫌抄袭的内容,欢迎发送邮件进行举报,并提供相关证据,一经查实,本社区将立刻删除涉嫌侵权内容,举报邮箱: cloudbbs@moduyun.com

  1. 分享:
最后一次编辑于 2023年11月08日 0

暂无评论

推荐阅读
  iD7FikcuyaVi   2023年11月30日   26   0   0 MacWindowsandroid
  b1UHV4WKBb2S   2023年11月13日   37   0   0 裁剪ideflutter
  b1UHV4WKBb2S   2023年11月13日   29   0   0 flutterDart
  zSWNgACtCQuP   2023年11月13日   32   0   0 ide
xx2YH4ad7R0N