情景
在ListView的一个或者多个Item中用到了EditText输入框控件,出现的一些常见问题优化处理。
问题
焦点丢失
点击EditText,弹出软键盘,你会发现EditText的焦点丢失了,根本不能立刻进行输入。
问题出现的原因是弹出软键盘会导致重新进行ListView的绘制,此EditText非彼EditText。
问题处理
思路:将之前EditText的状态进行一次保存,下一次绘制初始化状态
// 绑定位置 touch时处理
holder.mEditText.setTag(position);
// 当前位置与绑定的位置相同,重新请求一次焦点
if (mTouchItemPosition == position) {
holder.mEditText.requestFocus();
holder.mEditText.setSelection(holder.mEditText.getText().length());
} else {
holder.mEditText.clearFocus();
}
holder.mEditText.setOnTouchListener((view, event) -> {
// 点击触摸事件,获取点击Item的位置
mTouchItemPosition = (Integer) view.getTag();
return false;
});
内容保存
保存输入的内容,不会丢失已经输入的数据。
问题处理
ViewHolder holder;
if (convertView == null) {
convertView = LayoutInflater.from(parent.getContext()).inflate(R.layout.item, parent, false);
// 自定义TextWatcher
ListViewTextWatcher watcher = new ListViewTextWatcher(position, 1);
holder= new ViewHolder (convertView, watcher);
holder.mEdit.addTextChangedListener(watcher);
convertView.setTag(holder);
} else {
holder= (ViewHolder) convertView.getTag();
holder.mTextWathcer.setPosition(position);
}
// 绑定位置 touch时处理
holder.mEditText.setTag(position);
// 当前位置与绑定的位置相同,重新请求一次焦点
if (mTouchItemPosition == position) {
holder.mEditText.requestFocus();
holder.mEditText.setSelection(holder.mEditText.getText().length());
} else {
holder.mEditText.clearFocus();
}
holder.mEditText.setOnTouchListener((view, event) -> {
// 点击触摸事件,获取点击Item的位置
mTouchItemPosition = (Integer) view.getTag();
return false;
});
自定义TextWatcher,收集输入的数据
/**
* ListView中EditText内容保存
*
* @author fengzhen
* @version v1.0, 2017/9/29 11:28
*/
private class ListViewTextWatcher implements TextWatcher {
private int position;
private int type;
public void setPosition(int position) {
this.position = position;
}
public int getPosition() {
return position;
}
// 传入位置与类型,方便后续收集数据
public ListViewTextWatcher(int position, int type) {
this.position = position;
this.type = type;
}
@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) {
switch (type) {
case 1:
// 保存数据 s.toString 到类型为1的model
break;
case 2:
// 保存数据 s.toString 到类型为2的model
break;
default:
break;
}
}
}
滑动冲突
EditText内容过多,需要上下滑动,与ListView或者ScrollView事件冲突。
问题处理
思路:判断是否能够滑动,requestDisallowInterceptTouchEvent()方法让子控件自己处理事件监听。
holder.mEditText.setOnTouchListener((view, event) -> {
// 点击触摸事件,获取点击Item的位置
mTouchItemPosition = (Integer) view.getTag();
// 判断是否能够进行滑动,API4.0以上
if (holder.mEditText.canScrollVertically(1)
|| holder.mEditText.canScrollVertically(-1)) {
view.getParent().requestDisallowInterceptTouchEvent(true);
}
// 手势抬起,结束事件拦截
if (event.getAction() == MotionEvent.ACTION_UP) {
view.getParent().requestDisallowInterceptTouchEvent(false);
}
return false;
});