ListView中有EditText的一些bug的解决方案
2015-08-04 23:45
393 查看
今天有些空,想起前段时间被ListView加checkBox的组合搞过一次,心想在回顾一下,但是再写一遍一样的也没意思,就换成了ListView加EditText。一开始觉得应该和checkBox的差不多,对于数据错乱,checkBox是加一个标记位,那EditText就加一个map来存储所有的值咯。(key为position,value为EditText中的值)但是写了一遍发现会有为题,具体就是我使用textWatcher来存储数据。
mEditString就是存储数据的Map,但是发现还是会错乱,并且滑回去之后数据就全没了。。。
看了看代码,我发现滑动ListView之后,调用getView方法有个很奇怪的地方,就是比如我一屏有6个Item,第七个滑进来的时候调用getView和adTextChangeLisetener都没问题(因为这个时候还没有复用convertView),当第八个滑进来的时候,会复用第一个item,然后在afterTextChanged这个函数中,position竟然是0,这就意味着mEditString.put会是key=0,value="",接下去的事就变成了错位,滑上去之后editText的值消失了这样的bug。我想了想,关键还是复用了convertView之后的textWatcher重复了,然后我修改了代码
这是完整的dapter的代码,index onTouchListener和最后那些clearFouces,requestFoucus都是为了解决另外一个Bug,就是软键盘弹出以后editText失去焦点,要再点击一下才可以,不过这样的解决方法还是不是很好,软键盘这个东西还是很烦人的。。过两天看看源码吧,看看有没有什么解决方案。
vh.et.addTextChangedListener(new TextWatcher() { @Override public void beforeTextChanged(CharSequence charSequence, int i, int i1, int i2) { } @Override public void onTextChanged(CharSequence charSequence, int i, int i1, int i2) { } @Override public void afterTextChanged(Editable editable) { System.out.println("afterTextChanged" + " " + position + " " + editable.toString()); mEditString.put(position,editable.toString()); } }); vh.et.setText(mEditString.get(position));
mEditString就是存储数据的Map,但是发现还是会错乱,并且滑回去之后数据就全没了。。。
看了看代码,我发现滑动ListView之后,调用getView方法有个很奇怪的地方,就是比如我一屏有6个Item,第七个滑进来的时候调用getView和adTextChangeLisetener都没问题(因为这个时候还没有复用convertView),当第八个滑进来的时候,会复用第一个item,然后在afterTextChanged这个函数中,position竟然是0,这就意味着mEditString.put会是key=0,value="",接下去的事就变成了错位,滑上去之后editText的值消失了这样的bug。我想了想,关键还是复用了convertView之后的textWatcher重复了,然后我修改了代码
package com.example.administrator.listandedittest; import android.text.Editable; import android.text.TextWatcher; import android.view.LayoutInflater; import android.view.MotionEvent; import android.view.View; import android.view.ViewGroup; import android.widget.BaseAdapter; import android.widget.EditText; import android.widget.TextView; import java.util.ArrayList; import java.util.HashMap; import java.util.List; import java.util.Map; /** * Created by Administrator on 2015/8/4. */ public class MyAdapter extends BaseAdapter { private List<String> mDatas; private Map<Integer,String> mEditString; private List<MyTextWatcher> mWatcherList; private int index; public MyAdapter(List<String> mDatas){ this.mDatas = mDatas; mEditString = new HashMap<Integer,String>(); for(int i = 0;i<mDatas.size();i++){ mEditString.put(i,""); } mWatcherList = new ArrayList<MyTextWatcher>(); for(int i = 0;i < mDatas.size();i++){ mWatcherList.add(new MyTextWatcher(i)); } } @Override public int getCount() { return mDatas.size(); } @Override public Object getItem(int i) { return mDatas.get(i); } @Override public long getItemId(int i) { return i; } @Override public View getView(final int position, View convertView, ViewGroup parent) { ViewHolder vh = null; if(convertView == null){ vh = new ViewHolder(); convertView = LayoutInflater.from(parent.getContext()).inflate(R.layout.list_item,null); vh.tv = (TextView)convertView.findViewById(R.id.item_tv); vh.et = (EditText)convertView.findViewById(R.id.item_et); convertView.setTag(vh); }else{ vh = (ViewHolder)convertView.getTag(); } vh.tv.setText(mDatas.get(position)); //vh.et.removeTextChangedListener(); //System.out.println(mEditString.toString()); vh.et.setOnTouchListener(new View.OnTouchListener() { @Override public boolean onTouch(View view, MotionEvent motionEvent) { if (motionEvent.getAction() == MotionEvent.ACTION_UP) { index = position; } return false; } }); for(int i = 0;i < mDatas.size();i++){ vh.et.removeTextChangedListener(mWatcherList.get(i)); } vh.et.addTextChangedListener(mWatcherList.get(position)); vh.et.setText(mEditString.get(position)); vh.et.clearFocus(); if(index != -1 && index == position){ vh.et.requestFocus(); } vh.et.setSelection(vh.et.getText().length()); return convertView; } public static class ViewHolder{ TextView tv; EditText et; } public class MyTextWatcher implements TextWatcher { private int position; public MyTextWatcher(int position){ this.position = position; } @Override public void beforeTextChanged(CharSequence charSequence, int i, int i1, int i2) { } @Override public void onTextChanged(CharSequence charSequence, int i, int i1, int i2) { } @Override public void afterTextChanged(Editable editable) { mEditString.put(position,editable.toString()); } } }
这是完整的dapter的代码,index onTouchListener和最后那些clearFouces,requestFoucus都是为了解决另外一个Bug,就是软键盘弹出以后editText失去焦点,要再点击一下才可以,不过这样的解决方法还是不是很好,软键盘这个东西还是很烦人的。。过两天看看源码吧,看看有没有什么解决方案。
相关文章推荐
- Linux Shell脚本攻略复习
- HDU 5344 MZL's xor (水题)
- IIS下安装PHP
- 谈谈openstack部署规模问题
- 理解MYSQL语句执行过程
- 信号量、互斥体和自旋锁小结
- eclipse 常用设置,常用快捷键修改
- 进程间通信之管道和FIFO
- 信号量、互斥体和自旋锁小结
- Xcode 设置Pch文件
- sqlserver 2005/2008 导入超大sql文件
- install build essential
- Leap Motion的环境配置(使用c#)
- 深度优先搜索(Depth-First-Search)
- java内部类学习笔记
- Gulp安装及配合组件构建前端开发一体化
- coreldRAW学习笔记一
- 让Linux使用malloc申请更多的内存
- mybatis性能优化二之多对多查询:用一次请求解决n次请求查询
- 就算全世界都觉得你不行那又怎样