一 引言
之前发布过一个RecyclerView中解决EditText各类异常的方案,但是经过最新的各种恶心操作的测试,发现依然没有完全消灭所有的异常,所以在工作之余,又翻出来捣鼓一下。
依然感谢:苏泽兄、逗你玩222对上篇文章的错漏之处提出的改正意见
二 效果图(2.34 MB)
三 解决方案
- 解析整个问题点之前,先把项目的完整demo放送给大家,地址如下:
https://github.com/kaxi4it/EditTextInRecyclerViewDemo
注:阅读以下文章时,建议对照demo代码对比观看
因为有EditText的存在,所以demo里加入了InputMethodManager来管理软键盘的隐藏显示;
EditText的输入内容,通过一个SparseArray来做管理,因为SparseArray比HashMap更省内存,在某些条件下性能更好,主要是因为它避免了对key的自动装箱(int转为Integer类型),它内部则是通过两个数组来进行数据存储的,一个存储key,另外一个存储value,为了优化性能,它内部对数据还采取了压缩的方式来表示稀疏数组的数据,从而节约内存空间;
EditText的焦点,我们可以通过一个int变量记录他在adapter中的位置
//输入法
InputMethodManager inputMethodManager = (InputMethodManager) getSystemService(Context.INPUT_METHOD_SERVICE);
//edittext里的文字内容集合
SparseArray etTextAry = new SparseArray();
//edittext的焦点位置
int etFocusPos = -1;
TextWatcher textWatcher = new TextWatcher() {
@Override
public void beforeTextChanged(CharSequence s, int start, int count, int after) {}
@Override
public void onTextChanged(CharSequence s, int start, int before, int count) {}
@Override
public void afterTextChanged(Editable s) {
//每次修改文字后,保存在数据集合中
etTextAry.put(etFocusPos, s.toString());
}
};
然后分别在onViewAttachedToWindow
(item在页面中显示)与onViewDetachedFromWindow
(item在页面中隐藏)方法中,做获取/取消焦点,添加/删除EditText的文本变化的监听,为何要通过这2个方法来操作,因为RecyclerView的列表缓存机制,会导致并不是每次item的显示都会运行onBindViewHolder
方法,所以容易引起一些页面的异常情况。
@Override
public void onViewDetachedFromWindow(RecyclerView.ViewHolder holder) {
super.onViewDetachedFromWindow(holder);
ItemHolder viewHolder = (ItemHolder) holder;
//删除文字变化监听器
viewHolder.et.removeTextChangedListener(textWatcher);
//清除焦点
viewHolder.et.clearFocus();
//如果当前隐藏的item是焦点所在的位置,那么隐藏输入法,否则输入法不会自动关闭
if (etFocusPos == holder.getAdapterPosition()) {
inputMethodManager.hideSoftInputFromWindow(((ItemHolder) holder).et.getWindowToken(), 0);
}
}
@Override
public void onViewAttachedToWindow(RecyclerView.ViewHolder holder) {
super.onViewAttachedToWindow(holder);
ItemHolder viewHolder = (ItemHolder) holder;
//添加文字变化监听器
viewHolder.et.addTextChangedListener(textWatcher);
//如果当前显示的item是焦点记录位置,那么获取焦点,并把光标位置置于文字最后,需要显示输入法的话可自行添加操作
if (etFocusPos == holder.getAdapterPosition()) {
viewHolder.et.requestFocus();
viewHolder.et.setSelection(viewHolder.et.getText().length());
}
}
最后在onBindViewHolder
方法中,绑定数据与焦点切换时的监听就行了
@Override
public synchronized void onBindViewHolder(RecyclerView.ViewHolder holder, int i) {
final int position = i;
ItemHolder viewHolder = (ItemHolder) holder;
viewHolder.tv.setText("item "+position);
viewHolder.et.setText(etTextAry.get(position));
viewHolder.et.setOnFocusChangeListener(new View.OnFocusChangeListener() {
@Override
public void onFocusChange(View view, boolean b) {
if (b){
//记录焦点位置
etFocusPos = position;
}
}
});
}
四 结束语
希望以上的完整DEMO和代码的讲解能帮您解决这些小问题