独自のレイアウトとCustomCursorAdapterを備えたListViewがあります。すべての行には独自のチェックボックスがあります。だから...スロール中にチェックボックスが状態を失うことは絶対に明らかです。私が見つけた唯一のものは Androidはカーソルアダプタを使用してListViewにチェックボックスの状態を保存します ですが、そこには答えがありません。そしてもう1つ質問があります。 CustorArrayAdapterでも同じ問題が発生しました。 SparseBooleanArrayを使用してチェックボックスの状態を維持することで、この問題を解決しました。正常に動作しますが、すべてのスクロールでonCheckedChangedが呼び出されます。それは正常ですか?取引は私のリストビューで、アラーム要素と(onCheckedChangedの)定期的な呼び出しがアラームを開始/停止することを説明しています。不必要な行動がたくさん。
チェック可能なアイテムが含まれている場合、ListViewにはいくつかの懸念事項があります。次のリンクをお勧めします。
http://tokudu.com/2010/Android-checkable-linear-layout/
私はそれがあなたが望むものに近いと思います。
ListView
とCheckBox
で同様の問題が発生し、問題を解決するために何をしましたか。
false
に初期化します。これは、CheckBoxがまだチェックされていないことを意味します。public class MyDataAdapter extends SimpleCursorAdapter {
private Cursor c;
private Context context;
private ArrayList<String> list = new ArrayList<String>();
private ArrayList<Boolean> itemChecked = new ArrayList<Boolean>();
// itemChecked will store the position of the checked items.
public MyDataAdapter(Context context, int layout, Cursor c, String[] from,
int[] to) {
super(context, layout, c, from, to);
this.c = c;
this.context = context;
for (int i = 0; i < this.getCount(); i++) {
itemChecked.add(i, false); // initializes all items value with false
}
}
public View getView(final int pos, View inView, ViewGroup parent) {
if (inView == null) {
LayoutInflater inflater = (LayoutInflater) context
.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
inView = inflater.inflate(R.layout.your_layout_file, null);
}
final CheckBox cBox = (CheckBox) inView.findViewById(R.id.bcheck); // your
// CheckBox
cBox.setOnClickListener(new OnClickListener() {
public void onClick(View v) {
CheckBox cb = (CheckBox) v.findViewById(R.id.your_checkbox_id);
if (cb.isChecked()) {
itemChecked.set(pos, true);
// do some operations here
} else if (!cb.isChecked()) {
itemChecked.set(pos, false);
// do some operations here
}
}
});
cBox.setChecked(itemChecked.get(pos)); // this will Check or Uncheck the
// CheckBox in ListView
// according to their original
// position and CheckBox never
// loss his State when you
// Scroll the List Items.
return inView;
}}
私も同様の問題に直面していたので、たくさん読んだ後、私はこの問題を次のように解決しました:
@Override
public View getView(int position, View convertView, ViewGroup parent) {
ViewHolder holder = null;
if (convertView == null) {
convertView = mInflater.inflate(R.layout.listview, null);
holder = new ViewHolder();
holder.nameView = (TextView)convertView.findViewById(R.id.textView1);
holder.numberView = (TextView)convertView.findViewById(R.id.textView2);
holder.cb = (CheckBox)convertView.findViewById(R.id.checkBox1);
convertView.setTag(holder);
} else {
holder = (ViewHolder)convertView.getTag();
}
holder.nameView.setText(mData.get(position).toString());
holder.numberView.setText(mNumber.get(position).toString());
holder.cb.setChecked(false);
holder.cb.setTag(position);
if(selected.indexOf(mNumber.get(position).toString()) >= 0)
{
holder.cb.setChecked(true);
}
return convertView;
}
}
ここでは、getView()でこれを行っています。すべてのチェックボックスをオフにし、対応するテキストビューに従ってチェックする必要があるチェックボックスを手動で再度チェックしています。したがって、ユーザーが最初のチェックボックスをオンにした後で下にスクロールすると、ビュー内のすべてのチェックボックスがオフになり、もう一度上にスクロールすると、すべてのチェックボックスもオフになりますが、前にクリックしたチェックボックスが再度チェックされます。
SetViewBinderメソッドを試してみることにしました。 Сheckboxesの状態はSparseBooleanArrayに保存されます。チェックボックス自体をクリックすると、セル全体でチェックボックスの状態が変化します。
私の列:| TextView |チェックボックス|
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
// ..
private SparseBooleanArray checkedItems = new SparseBooleanArray(); //for storing item state
startManagingCursor(cursor);
String[] from = new String[] { dbAdapter.COLUMN_NAME, dbAdapter.COLUMN_CHECK };
int[] to = new int[] { R.id.item_name, R.id.item_check };
recordsAdapter = new SimpleCursorAdapter(this, R.layout.items, cursor, from, to);
recordsAdapter.setViewBinder(new SimpleCursorAdapter.ViewBinder() {
public boolean setViewValue(View view, Cursor cursor, int columnIndex) {
if (columnIndex == 2) { //2 - R.id.item_check
final CheckBox cb = (CheckBox) view;
final int rowID = cursor.getInt(0); //cursor.getInt(0) - _id from table
if (checkedItems.indexOfKey(rowID) >= 0) { //checkedItems contains rowID?
cb.setChecked(checkedItems.get(rowID));
} else if (cursor.getInt(2) > 0) { //cursor.getInt(2): 0 - false, 1 - true
checkedItems.append(rowID, true);
cb.setChecked(true);
} else {
cb.setChecked(false);
}
cb.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
if (checkedItems.indexOfKey(rowID) >= 0) {
checkedItems.put(rowID, !(checkedItems.get(rowID)));
} else {
checkedItems.append(rowID, true);
}
cb.setChecked(checkedItems.get(rowID));
dbAdapter.updateItem(rowID, checkedItems.get(rowID)?1:0);
}
});
return true;
}
return false;
}
});
itemsList.setAdapter(recordsAdapter);
itemsList.setOnItemClickListener(this);
// ..
}
//if click on TextView of row - CheckBox of row is set/unset
@Override
public void onItemClick(AdapterView<?> parent, View view, int pos, long id) {
TextView tv = (TextView) view.findViewById(R.id.item_name);
ViewGroup row = (ViewGroup) tv.getParent();
CheckBox cb = (CheckBox) row.getChildAt(1);
cb.performClick();
}
(HTMLのように)ViewElementをXMLの配列として定義できない理由がわかりません。つまり:
<CheckBox
Android:id="@+id/myCheckBox[]"
Android:focusable="false"
Android:layout_width="wrap_content"
Android:layout_height="wrap_content"
Android:layout_alignParentRight="true"
Android:layout_centerVertical="true" />
値をintit [true | false]に格納します。これにより、複雑なソースフリックルを使用する代わりに、作業がはるかに簡単になります。また、データ(つまり、横向きから縦向きに切り替える場合)は、Androidで保持し、失われないようにする必要があります。
おそらくAPI117では、バージョンtoothscratch ...
(申し訳ありませんが、まだコメントできません)-私のために機能するVikas Patidarの回答に関して-アダプターのコンストラクターのgetCountがゼロになるため、非同期読み込み(ローダーの使用など)を使用する場合は注意してください。
インデックス作成エラーが発生します(ArrayListにデータが入力されないため)。 ArrayListの代わりにSparseBooleanArrayを使用すると、非同期読み込みでも機能します。
ここで遅れるかもしれませんが、合計すると Vikas Patidar's ここでの答えは、CursorAdapterを訴えている人のためにbindView
メソッドを使用した完全な実装です。
_@Override
public void bindView(View view, Context context, Cursor cursor) {
final ViewHolder holder = (ViewHolder) view.getTag();
final int position = cursor.getPosition();
String name = (cursor.getString(cursor.getColumnIndexOrThrow("NotificationDateFor")));
String image = cursor.getString(cursor.getColumnIndexOrThrow("imageUri"));
System.out.println("cursor.getPosition(test): "+cursor.getPosition());
holder.nametext.setText(name);
// setImage(image, holder.iv);
holder.chk.setOnClickListener(new OnClickListener(){
@Override
public void onClick(View v) {
if (holder.chk.isChecked()) {
itemChecked.set(position, true);
System.out.println("cursor.getPosition(true): "+position);
} else if (!holder.chk.isChecked()) {
itemChecked.set(position, false);
System.out.println("cursor.getPosition(false): "+position);
// AddDelete
}
}
});
holder.chk.setChecked(itemChecked.get(cursor.getPosition()));
}
_
注:final int position = cursor.getPosition();
は重要です!
Listviewに組み込まれている複数選択モードを利用することで、面倒な作業を大幅に減らすことができます。
listView.setChoiceMode(ListView.CHOICE_MODE_MULTIPLE);