- android - 多次调用 OnPrimaryClipChangedListener
- android - 无法更新 RecyclerView 中的 TextView 字段
- android.database.CursorIndexOutOfBoundsException : Index 0 requested, 光标大小为 0
- android - 使用 AppCompat 时,我们是否需要明确指定其 UI 组件(Spinner、EditText)颜色
我是 Android 和 Java 编程的新手,最近遇到了一个问题。我一直在寻找几个小时的答案,但我没有成功。我有一个自定义 ListView 和一个自定义适配器,其中一行包含一个 CheckBox 和一个 EditText。每当它第一次显示时,都可以,但是当我向下滚动时,它开始更改 CheckBox 和 EditText 的值。
这是我的 row_edit.xml:
<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="wrap_content" >
<EditText
android:id="@+id/bonus"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_alignParentRight="true"
android:inputType="numberSigned"
android:gravity="right"
android:selectAllOnFocus="true"
android:textSize="16sp" />
<CheckBox
android:id="@+id/checkBox_person"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_alignParentLeft="true"
android:layout_toLeftOf="@id/bonus"
android:textSize="16sp" />
</RelativeLayout>
Item_edit(适配器项):
public class Item_edit {
private String meno;
private boolean ucast;
private int bonus;
public Item_edit(String meno, boolean ucast, int bonus) {
super();
this.meno = meno;
this.ucast = ucast;
this.bonus = bonus;
}
public String getMeno() {
return meno;
}
public boolean getUcast() {
return ucast;
}
public int getBonus() {
return bonus;
}
public void setMeno(String meno) {
this.meno = meno;
}
public void setUcast(boolean ucast) {
this.ucast = ucast;
}
public void setBonus(int bonus) {
this.bonus = bonus;
}
}
当我滚动时,ListView 并不流畅,这就是我决定使用 ViewHolder 的原因。适配器_编辑:
public class Adapter_edit extends ArrayAdapter<Item_edit> {
private final Context context;
private final ArrayList<Item_edit> itemsArrayList;
private final Edit edit;
public Adapter_edit(Context context, ArrayList<Item_edit> itemsArrayList, Edit edit) {
super(context, R.layout.row_edit, itemsArrayList);
this.edit = edit;
this.context = context;
this.itemsArrayList = itemsArrayList;
}
@Override
public View getView(final int position, View convertView, ViewGroup parent) {
ViewHolder holder = null;
if (convertView == null) {
LayoutInflater inflater = (LayoutInflater) context
.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
convertView = inflater.inflate(R.layout.row_edit, parent, false);
holder = new ViewHolder();
//Log.e("NewHolder", "position = " + position);
holder.checkBox = (CheckBox) convertView
.findViewById(R.id.checkBox_person);
holder.bonus = (EditText) convertView.findViewById(R.id.bonus);
convertView.setTag(holder);
holder.checkBox.setText(itemsArrayList.get(position).getMeno());
holder.checkBox.setChecked(itemsArrayList.get(position).getUcast());
holder.bonus.setText(Integer.toString(itemsArrayList.get(position)
.getBonus()));
holder.checkBox
.setOnCheckedChangeListener(new OnCheckedChangeListener() {
@Override
public void onCheckedChanged(CompoundButton buttonView, boolean isChecked) {
itemsArrayList.get(position).setUcast(isChecked);
}
});
holder.bonus.addTextChangedListener(new TextWatcher() {
@Override
public void onTextChanged(CharSequence s, int start, int before, int count) {
try {
if (itemsArrayList.get(position).getBonus() != Integer
.parseInt(s.toString()) && !edit
.getUcast(position)) {
itemsArrayList.get(position).setBonus(
Integer.parseInt(s.toString()));
itemsArrayList.get(position).setUcast(true);
}
}
catch (NumberFormatException e) {
itemsArrayList.get(position).setBonus(0);
}
}
@Override
public void afterTextChanged(Editable s) {
}
@Override
public void beforeTextChanged(CharSequence s, int start, int count, int after) {
}
});
}
else {
holder = (ViewHolder) convertView.getTag();
}
holder.checkBox.setText(itemsArrayList.get(position).getMeno());
holder.checkBox.setChecked(itemsArrayList.get(position).getUcast());
holder.bonus.setText(Integer.toString(itemsArrayList.get(position)
.getBonus()));
return convertView;
}
static class ViewHolder {
CheckBox checkBox;
EditText bonus;
}
}
使用 LogCat 调试后,我知道只生成了 8 个持有人(因为有 (0-7) 个项目适合我的屏幕)。当我向下滚动时,这些正在被重复使用。没有为位置大于 7 的项目创建 OnCheckedChangeListeners 和 TextChangedListeners。
我不知道该怎么办。有人可以帮助我吗?
最佳答案
我终于得到了答案。
我扩展了 EditText,创建了我自己的类 UpgEditText,它能够使用方法 setOnTextChangedListener,这意味着删除现有的 TextWatcher 并添加一个新的。
UpgEditText
public class UpgEditText extends EditText {
public UpgEditText(Context context, AttributeSet attrs, int defStyle) {
super(context, attrs, defStyle);
}
public UpgEditText(Context context, AttributeSet attrs) {
super(context, attrs);
}
public UpgEditText(Context context) {
super(context);
}
TextWatcher watcher = null;
public void setOnTextChangedListener(final int position, final Adapter_edit adapter) {
if (watcher != null) {
this.removeTextChangedListener(watcher);
}
watcher = new TextWatcher() {
@Override
public void onTextChanged(CharSequence s, int start, int before, int count) {
}
@Override
public void beforeTextChanged(CharSequence s, int start, int count, int after) {
}
@Override
public void afterTextChanged(Editable s) {
try {
adapter.itemsArrayList.get(position).setBonus(
Integer.parseInt(s.toString()));
}
catch (Exception e) {
}
}
};
this.addTextChangedListener(watcher);
}
}
我也正常使用 ViewHolder:
public class ViewHolder {
CheckBox chBoxUcast;
UpgEditText edTBonus;
}
CustomAdapter getView() 方法:
public View getView(final int position, View convertView, ViewGroup parent) {
ViewHolder holder = null;
if (convertView == null) {
LayoutInflater inflater = (LayoutInflater) context
.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
convertView = inflater.inflate(R.layout.row_edit, parent, false);
holder = new ViewHolder();
holder.chBoxUcast = (CheckBox) convertView
.findViewById(R.id.checkBox_person);
holder.edTBonus = (UpgEditText) convertView
.findViewById(R.id.bonus);
convertView.setTag(holder);
}
else {
holder = (ViewHolder) convertView.getTag();
}
holder.edTBonus.setOnTextChangedListener(position, this);
holder.chBoxUcast
.setOnCheckedChangeListener(new OnCheckedChangeListener() {
@Override
public void onCheckedChanged(CompoundButton buttonView, boolean isChecked) {
// TODO Auto-generated method stub
itemsArrayList.get(position).setUcast(isChecked);
}
});
holder.chBoxUcast.setText(itemsArrayList.get(position).getMeno());
holder.chBoxUcast.setChecked(itemsArrayList.get(position).getUcast());
holder.edTBonus.setText(Integer.toString(itemsArrayList.get(position)
.getBonus()));
return convertView;
}
现在一切正常。每次回收 convertView 时,我只更改监听器并且效果很好。
关于java - Android:带有 ViewHolder、OnCheckedChangedListener 和 TextChangedListener 的自定义适配器,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/22000208/
最近我使用 RecyclerView 并添加了一个自定义标题 View (另一种类型的项目 View )并尝试在数据发生更改时对其进行更新。奇怪的事情发生了。适配器创建一个新的 HeaderViewH
我的项目应该基于用户输入(用户发送文本、拍照或录制视频)创建多个 View ,类似于 WhatsApp 聊天 Activity ,结构几乎相同。适配器应该能够通过使用 getItemViewType(
在 ViewHolder pattern 中将 ViewHolder 设为静态对性能至关重要吗? ? A ViewHolder object stores each of the component
我只是想更好地理解我经常用来优化 ListView 的以下模式 我的阅读只指出静态内部类被视为顶级类的事实。与成员类(非静态)相比,这样的事情有什么好处? @Override public View
我有一个 RecyclerView 和一些 view/card(我们暂时称它为 View ),它们都包含相同的东西,包括我用作分隔栏的 View。 我希望能够在当前 View 上方的 view 中更改
我需要从 View Holder 访问 RecyclerView Adapter 的方法。我找不到任何解决方案。 或者是否可以从 ViewHolder 的 ViewModel 类(我已经在 MVVM
在我的 RecyclerView 适配器中只有一个 ViewHolder 类: public static class PlayerItemViewHolder extends RecyclerVie
我已经采用了一个示例代码来实现 RecyclerView,但试图将其转换为在我的应用程序的子 fragment 中工作。 代码在“创建列表 - 示例”下 Creating Lists and Card
我试图在我的主要 Activity 中而不是在我的自定义适配器中保存复选框的状态。我已检查共享首选项中是否存储了正确的数据,并且可以成功检索信息,但是当我尝试在打开应用程序时标记复选框时, View
我是 Kotlin 的新手,我正在制作一款货币兑换应用。在适配器中,我想将一些项目传递到新 Activity 中。 class AdapterC (val countryList: ArrayLis
我在我的代码中遇到了这个问题,当在 Viewholder 中的一个按钮上进行转换时,这是在 Onclicklistener 中完成的,多个转换发生在不同的行上即,如果我单击第 1 行中的按钮,则按钮向
public class ListViewAdapter extends BaseAdapter { private Context context; private LocationDetails[
我刚刚生成了一个 Master/Detail Flow 项目,我发现了一些奇怪的东西:在 DriverListActivity.java 中,名为 ViewHolder 的子类具有 final 属性。
我正在尝试配置抽屉导航以显示 3 种不同类型的“行”,但我在使用 Holder 时遇到了问题,被告知该消息: FATAL EXCEPTION: main java.lang.ClassCastExce
我理解Viewholder pattern的思路和用法,但我仍然有一个问题: 假设我们在 viewholder 中有一个 TextView,并显示 10 个项目(“item0,item1 ....”)
我有一个包含 3 种不同类型行的 ListView 。我在网上收到 ClassCastException holder = (RowViewHolder) row.getTag(); 我注意到 row
我正在尝试学习 Android 编程。我找不到这个算法的解释: public View getView(int r, View convertView, ViewGroup parent) { V
我正在创建一个 Android 应用程序,其中包含一个带有嵌套 CardView 的 RecyclerView。我需要将所有其他卡片替换为不同的颜色。我正在使用 @Override 来覆盖 onBin
我在 ListView 中使用分段时遇到问题。我需要使用 viewholder 使 listview 滚动平滑,但我不知道如何实现两个 viewholders,因为有两个单独的 View ,一个是部分
我在 ListView 上遇到了 View 持有者的充气问题。实现 Viewholder 后,充气机会出错,因此消息显示错误。 (在图 10 中,消息不是来 self ,而且每当我滚动时,错误的消息都
我是一名优秀的程序员,十分优秀!