RecyclerView中解决EditText的各种异常

一 引言


之前这里发布过一个RecyclerView中解决EditText各类异常的方案,存在BUG,为了方便后来人误入这里,我把最新修复过的版本,搬到这里来了

二 效果图(2.34 MB)


效果图大小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和代码的讲解能帮您解决这些小问题

你可能感兴趣的:(RecyclerView中解决EditText的各种异常)