首先,我想说,ListView中使用EditText,我挣扎了好几天,最终还是放弃了,听我句劝,ListView中使用EditText,搞不定的,当然,很可能是我比较菜的原因。
我看过一些不错的解决办法,比如用一个ScrollView包裹一个布局,然后往布局中塞Item,感觉应该能解决,没试过。最终弄出来时,我感动得都快哭了,其实要点就两点,却让我弄了好几天。
- 果断放弃ListView使用RecycleView,我以前也没有用过RecycleView,这次是逼不得已,使用RecycleView可以解决EditText焦点错乱问题。
- 踩EditText这个坑踩得久了,应该会接触到addTextChangedListener()这个方法,第二点就是一定要在EditText获取到焦点时才能添加这个方法,并且在EditText失去焦点时调用removeTextChangedListener(),这样可以解决数据错乱问题。
以下是部分代码:
- 在onBindViewHolder中添加TextWatcher对象,不用匿名内部类是因为移除的时候还需要它。
final TextWatcher textWatcher = new TextWatcher() {
private String preText; // 改变前的EditText内容
@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) {
String nowText = s.toString();
if (!"".equals(nowText)) {
if (nowText.equals(preText)) {
return;
}
int number = Integer.parseInt(s.toString());
int all = 50 + number;
preText = nowText; // 设置当前text内容,一定要在 viewHolder.mEtNumber.setText()方法前调用,否则会出现死循环,然后报stackOverFlow异常
if (all > 100) {
viewHolder.mEtNumber.setText(number + "");
}
}
String numberStr = String.valueOf(viewHolder.mEtNumber.getText());
viewHolder.mEtNumber.setSelection(numberStr.length()); // 将光标放在最后
}
};
- 添加setOnFocusChangeListener()方法,其实这样做的原因是当recycleView滑动时,会调用viewHolder.mEtNumber.setText()方法,如果有监听,监听便会起作用,此时很可能就会造成数据错乱,因此要在EditText获取到焦点时添加监听,在数据处理完后移除监听并且clearFocus()。
viewHolder.mEtNumber.setOnFocusChangeListener(new View.OnFocusChangeListener() {
@Override
public void onFocusChange(View v, boolean hasFocus) {
if (hasFocus) {
String numberStr = String.valueOf(viewHolder.mEtNumber.getText());
viewHolder.mEtNumber.setSelection(numberStr.length()); // 将光标放在最后
// 在EditText获取到焦点时,才给它添加监听事件,同时处理完数据后,要移除监听,必须要这样做,不然数据会错乱
viewHolder.mEtNumber.addTextChangedListener(textWatcher);
}
}
});
- 对Item中某个组件添加touch事件监听(我没有试过对整个Item监听,因为recycleView不熟,找不到),我的整个Item是一张图片,上面放了一个EditText和两个加减按钮,因此我就对photo监听它的touch事件
viewHolder.mIvPhoto.setOnTouchListener(new View.OnTouchListener() {
@Override
public boolean onTouch(View v, MotionEvent event) {
// 调用Activity中实现的接口,判断输入法是否为 弹出状态,如果弹出,则拦截mIvPhoto的点击事件,去关闭输入法
boolean eventPatch = mOnChangeListener.photoTouchCloseInputMethod(event, viewHolder.mEtNumber);
if (eventPatch) {
// 一定一定要调用这句话移除监听,不然一开始数据没问题,当你每个EditText都设置过值后,就有问题了
viewHolder.mEtNumber.removeTextChangedListener(textWatcher);
String numberString = String.valueOf(viewHolder.mEtNumber.getText());
if (numberString != null && !"".equals(numberString)) {
int numberTemp = Integer.parseInt(numberString);
if (numberTemp == 0) {
// 显示是否删除的对话框
}
}
viewHolder.mEtNumber.setText(String.valueOf(number));
// 处理需要保存的数据
}
return eventPatch; // 返回是否拦截点击事件
}
});
public interface onChangeListener {
/**
* 点击照片的事件,如果输入法是弹出状态,则关闭输入法,拦截其它事件,否则不拦截
*
* @param event 触摸事件
* @param view 此view为EditText
* @return 是否拦截事件 true 拦截photo的点击时间事件 false 不拦截
*/
boolean photoTouchCloseInputMethod(MotionEvent event, View view);
}
@Override
public boolean photoTouchCloseInputMethod(MotionEvent event, View view) {
// 这个view为EditText
if (event.getAction() == MotionEvent.ACTION_DOWN) {
if (mInputManager.isActive(view)) { // 如果输入法是打开状态
mInputManager.hideSoftInputFromWindow(getCurrentFocus().getWindowToken(), InputMethodManager.HIDE_NOT_ALWAYS);
// 关闭输入法后,要让editText清除焦点
view.clearFocus();
return true;
} else {
return false;
}
}
return false;
}
这玩意让我纠结了好久,又各种百度google不到[哭晕在厕所.jpg]
网友评论
键盘,当然这个也好解决。
但还有个问题,就是无论如何监听不到 软键盘自带的收起按钮,我真是日了狗了,后面只得监听布局变化,来判断软键盘的打开和关闭