滑动 RecycleView 点击 EditText 控件后,程序崩溃 ANR
大量注册 EditText 容易出现这个问题
只要点击EditText,RecycleView 适配器 中的onBindViewHolder 回调会一直刷新
导致整个 View 中大量 EditText
焦点错乱
原因是:
- editText中的addTextChangedListener是可以绑定多个监听器的
- 如果一个editText要获取输入值那它就首先必须要获取到焦点
- 用户不断滑动 RecyclerView 那么就会嵌套触发监听器
- 最终导致 ANR
不建议使用 tag 方式来设置
因为不会在根本上解决 ANR 问题
布局文件需要设置为
1
2
| <activity
android:windowSoftInputMode="stateHidden|adjustPan" />
|
1
2
3
| <android.support.v7.widget.RecyclerView
android:descendantFocusability="beforeDescendants"
android:fastScrollEnabled="false"/>
|
- 在 RecycleView Holder 中,设置
setOnFocusChangeListener
监听器 - 通过来判断焦点的变化来设置
addTextChangedListener
和 removeTextChangedListener
示例代码
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
| holder.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) {
// do something
}
}
holder.editText.setOnFocusChangeListener(new View.OnFocusChangeListener() {
@Override
public void onFocusChange(View v, boolean hasFocus) {
if (hasFocus) {
if (holder.textWatcher != null) {
holder.editText.addTextChangedListener(holder.textWatcher);
}
} else {
if (holder.textWatcher != null) {
holder.editText.removeTextChangedListener(holder.textWatcher);
}
}
}
});
|
- EditText中有 setTag 和 getTag 可以储存Object对象作为标签
然后把textWatcher 作为标签设置 TAG 来作为标记表明这个item的editText是否已经设置textChange监听
- 在Adapter的onBindViewHolder中每次都先判断editText的Tag是否有textWatcher对象
- 有的话就调用removeTextChangedListener来移除由于视图复用之前绑定的textWatcher
- 然后就设置editText的内容setText
- 最后再给editText设置监听器addTextChangedListener 并把这个textWatcher加入到editText的TAG标签中
总结来说就是在适配器里先移除被复用事件,再添加新事件
代码样例
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
| // 通过设置 tag 避免复用
EditText editText = helper.getView(R.id.et_input);
if (editText.getTag() instanceof TextWatcher) {
editText.removeTextChangedListener((TextWatcher) editText.getTag());
}
helper.setText(R.id.et_comment, item.content);
TextWatcher watcher = 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) {
item.content = s.length() > 0 ? s.toString() : "";
// do something
}
};
editText.addTextChangedListener(watcher);
editText.setTag(watcher);
|
1
2
3
4
5
6
7
8
9
10
11
12
13
14
| CheckBox checkBox = helper.getView(R.id.mCheckBox);
checkBox.setOnCheckedChangeListener(null);
if (item.anonymous) {
checkBox.setChecked(true);
} else {
checkBox.setChecked(false);
}
checkBox.setOnCheckedChangeListener(new CompoundButton.OnCheckedChangeListener() {
@Override
public void onCheckedChanged(CompoundButton buttonView, boolean isChecked) {
item.anonymous = isChecked;
// do something
}
})
|
1
2
3
4
5
6
7
8
9
10
| StarBar starBar = helper.getView(R.id.star_bar);
starBar.setOnStarChangeListener(null);
starBar.setStarMark(item.score);
((StarBar) helper.getView(R.id.mStarBar)).setOnStarChangeListener(new StarBar.OnStarChangeListener() {
@Override
public void onStarChange(float mark) {
item.score = (int) mark;
// do something
}
});
|