При проверке флажка в listview также проверяются другие случайные флажки
всякий раз, когда я устанавливаю флажок в своем списке, проверяются и другие случайные флажки. Это может быть связано с переустановкой элементов списком.
Я также попытался установить флажок android:focusable="false"
в мой макет, как это было предложено в некоторых местах, но все же onListItemClick() не вызывается для строки, когда установлен флажок. Только когда я нажимаю на другое место, он вызывается.
Я хочу, чтобы только флажки, отмеченные пользователем, должны были проверяться до тех пор, пока пользователь не уберет их.
Я даю ниже код, который является полным и может быть запущен напрямую.
Код действия - ProjActivity.java:
public class ProjActivity extends ListActivity {
/** Called when the activity is first created. */
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
PackageManager pm = getPackageManager();
List<ApplicationInfo> packages = pm.getInstalledApplications(PackageManager.GET_META_DATA);
final CopyOfMyCustomAdapter a = new CopyOfMyCustomAdapter(this, packages);
getListView().setAdapter(a);
}}
И наконец, файл настраиваемого макета - testlayout.xml
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="horizontal"
>
<CheckBox
android:id="@+id/checkBox1"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_marginLeft="20dp"
android:text="CheckBox"
android:focusable="false"
/>
<ImageView
android:id="@+id/imageView1"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:src="@drawable/ic_launcher"
android:focusable="false"
/>
ОБНОВЛЕНИЕ: Мой CustomAdapter после предложения в ответе ниже:
public class MyCustomAdapter extends ArrayAdapter<ApplicationInfo> {
private List<ApplicationInfo> appInfoList;
private LayoutInflater mInflater;
private PackageManager pm;
ArrayList<Boolean> positionArray;
private Context ctx;
int[] visiblePosArray;
private volatile int positionCheck;
public MyCustomAdapter(Context context, List<ApplicationInfo> myList) {
super(context, NO_SELECTION);
appInfoList = myList;
ctx=context;
mInflater = (LayoutInflater)context.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
pm = context.getPackageManager();
positionArray = new ArrayList<Boolean>(myList.size());
for(int i =0;i<myList.size();i++){
positionArray.add(false);
}
}
@Override
public int getCount() {
// TODO Auto-generated method stub
return appInfoList.size();
}
@Override
public View getView(final int position, View convertView, ViewGroup parent) {
View row = convertView;
Holder holder = null;
if(row==null){
row = mInflater.inflate(R.layout.testlayout, null);
// visiblePosArray[position%visiblePosArray.length]=position;
holder = new Holder();
holder.appIcon = (ImageView)row.findViewById(R.id.imageView1);
holder.ckbox =(CheckBox)row.findViewById(R.id.checkBox1);
row.setTag(holder);
} else {
holder = (Holder) convertView.getTag();
}
holder.ckbox.setFocusable(false);
holder.appIcon.setImageDrawable(appInfoList.get(position).loadIcon(pm));
holder.ckbox.setChecked(positionArray.get(position));
holder.ckbox.setText(appInfoList.get(position).loadLabel(pm));
holder.ckbox.setOnCheckedChangeListener(new OnCheckedChangeListener() {
@Override
public void onCheckedChanged(CompoundButton buttonView, boolean isChecked) {
if(isChecked ){
System.out.println(position+"--- :)");
positionArray.add(position, true);
}else
positionArray.add(position, false);
}
});
return row;
}
static class Holder
{
ImageView appIcon;
CheckBox ckbox;
}
}
Когда я просматриваю вверх и вниз, я вижу, что случайные индексы изменяются на true в моем логическом Arraylist, когда они находятся в syso.
Ответы
Ответ 1
Когда listview перерабатывает представления, он перерабатывает свое текущее состояние, а также подключенные к нему слушатели. В моем примере, если флажок был установлен и имеет набор onCheckedChangeListener, оба они останутся частью переработанного представления, основанного на позиции. Таким образом, мы несем ответственность перед reset всеми состояниями и удаляем предыдущих слушателей.
Итак, когда я снял отметку с переработанного представления, слушатель onCheckedChange выполнялся.
одна линия сделала программу безупречной. Слушатель был удален:
holder.ckbox.setOnCheckedChangeListener(null);
Ниже приведен рабочий код адаптера для людей, которые могут наткнуться на эту проблему:
public class MyCustomAdapter extends ArrayAdapter<ApplicationInfo> {
private List<ApplicationInfo> appInfoList;
private LayoutInflater mInflater;
private PackageManager pm;
ArrayList<Boolean> positionArray;
private Context ctx;
int[] visiblePosArray;
private volatile int positionCheck;
public MyCustomAdapter(Context context, List<ApplicationInfo> myList) {
super(context, NO_SELECTION);
appInfoList = myList;
ctx=context;
mInflater = (LayoutInflater)context.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
pm = context.getPackageManager();
positionArray = new ArrayList<Boolean>(myList.size());
for(int i =0;i<myList.size();i++){
positionArray.add(false);
}
}
@Override
public int getCount() {
// TODO Auto-generated method stub
return appInfoList.size();
}
@Override
public View getView(final int position, View convertView, ViewGroup parent) {
View row = convertView;
Holder holder = null;
if(row==null){
row = mInflater.inflate(R.layout.testlayout, null);
// visiblePosArray[position%visiblePosArray.length]=position;
holder = new Holder();
holder.appIcon = (ImageView)row.findViewById(R.id.imageView1);
holder.ckbox =(CheckBox)row.findViewById(R.id.checkBox1);
row.setTag(holder);
} else {
holder = (Holder) convertView.getTag();
holder.ckbox.setOnCheckedChangeListener(null);
}
holder.ckbox.setFocusable(false);
holder.appIcon.setImageDrawable(appInfoList.get(position).loadIcon(pm));
holder.ckbox.setChecked(positionArray.get(position));
holder.ckbox.setText(appInfoList.get(position).loadLabel(pm));
holder.ckbox.setOnCheckedChangeListener(new OnCheckedChangeListener() {
@Override
public void onCheckedChanged(CompoundButton buttonView, boolean isChecked) {
if(isChecked ){
System.out.println(position+"--- :)");
positionArray.set(position, true);
}else
positionArray.set(position, false);
}
});
return row;
}
static class Holder
{
ImageView appIcon;
CheckBox ckbox;
}
}
Ответ 2
Вам нужно отслеживать состояние проверки, потому что ListView
повторно использует Views
. поэтому состояние для позиции, которое ранее было включено/отключено, может отображаться как указано для позиции 7.
Итак, что вам нужно сделать, это сохранить проверенное состояние в массиве boolean
или как угодно.
Возьмите уровень класса boolean [] checkedState;
, инициализируя его в конструкторе, в соответствии с размером вашего массива данных вы можете использовать ArrayList<Boolean>
тоже для динамического размера.
установите OnStateChangeListener
на CheckBoxes
в getView()
, всякий раз, когда он установлен или не проверен, возьмите позицию и сохраните ее в массиве checkedState
следующим образом:
checkedState[position] = false;// or true accordingly
и при настройке других данных для View
, таких как TextView
или ImageView
для любой конкретной позиции, установите также состояние проверки следующим образом:
holder.appIcon.setImageDrawable(appInfoList.get(position).loadIcon(pm));
holder.ckbox.setChecked(checkedState[position]);
Очень хорошее объяснение и пример:
Галерея пользовательских изображений для Android с галочкой в сетке, чтобы выбрать несколько вариантов
Изменить: На самом деле, что происходит, ваша позиция становится плохой, чтобы решить эту проблему:
holder.ckbox.setText(appInfoList.get(position).loadLabel(pm));
holder.ckbox.setTag(String.valueOf(position)); // to properly track the actual position
holder.ckbox.setOnCheckedChangeListener(new OnCheckedChangeListener() {
@Override
public void onCheckedChanged(CompoundButton v, boolean isChecked) {
int pos = Integer.parseInt( v.getTag().toString()) ; //to take the actual position
positionArray.add(pos, isChecked); // we don't need to check whether it is true or false, however you can put if-else to debug the app.
}
});
Ответ 3
Сочетание этих двух подходов сработало для меня:
У меня есть логический массив на уровне класса, который я использую, чтобы отслеживать значение флажков.
boolean [] checkedItems = new boolean[listItems.size()];
В getView():
public View getView(int position, View convertView, ViewGroup parent) {
ViewHolder holder;
if (convertView == null) {
convertView = inflater.inflate(R.layout.menu_item_list_item,
parent, false);
holder = new ViewHolder();
holder.name = (TextView) convertView
.findViewById(R.id.menuItemLargeName);
holder.mainItemCheckBox = (CheckBox) convertView
.findViewById(R.id.menuItemLargeCheckBox);
convertView.setTag(holder);
} else {
holder = (ViewHolder) convertView.getTag();
// remove the listener so that it does not get attached to other chechboxes.
holder.mainItemCheckBox.setOnCheckedChangeListener(null);
//update the checkbox value from boolean array
holder.mainItemCheckBox.setChecked(checkedItems[position]);
}
holder.name.setText(listItems.get(position).getName());
holder.mainItemCheckBox
.setOnCheckedChangeListener(onCheckedListener);
holder.mainItemCheckBox
.setTag(R.id.menuItemLargeCheckBox, position);
return (convertView);
}
В моем OnCheckedChangeListener(): обновить логический массив.
OnCheckedChangeListener onCheckedListener = new OnCheckedChangeListener() {
@Override
public void onCheckedChanged(CompoundButton buttonView,
boolean isChecked) {
int position = (Integer) buttonView
.getTag(R.id.menuItemLargeCheckBox);
MenuItemObject menuItem = listItems.get(position);
if (isChecked) {
cartItems.add(menuItem);
checkedItems[position] = true;
} else {
cartItems.remove(menuItem);
checkedItems[position] = false;
}
}
};