これは本当に奇妙です。
SetItemCheckedを呼び出すListViewに標準のArrayAdapterを使用すると正常に動作します
ただし、カスタムメイドのArrayAdapterを使用する場合は使用しません。
その理由は何でしょうか?これはバグですか?それとも何かが足りませんか?
public class Test_Activity extends Activity {
/** Called when the activity is first created. */
private List<Model> list;
private ListView lView;
public void onCreate(Bundle icicle) {
super.onCreate(icicle);
// Create an array of Strings, that will be put to our ListActivity
setContentView(R.layout.main);
lView = (ListView) findViewById(R.id.ListView01);
lView.setChoiceMode(ListView.CHOICE_MODE_MULTIPLE);
list = getModel();
//with this adapter setItemChecked works OK
lView.setAdapter(new ArrayAdapter<Model>(this,
Android.R.layout.simple_list_item_multiple_choice, list));
//**************************
//PROBLEM: with this adapter it does not check any items on the screen
// ArrayAdapter<Model> adapter = new Test_Class1(this, list);
// lView.setAdapter(adapter);
}
private List<Model> getModel() {
List<Model> list = new ArrayList<Model>();
list.add(get("0"));
list.add(get("1"));
list.add(get("2"));
list.get(1).setSelected(true);
Model m = list.get(1);
list.add(get("3"));
list.add(get("4"));
list.add(get("5"));
list.add(get("6"));
list.add(get("7"));
// Initially select one of the items
return list;
}
private Model get(String s) {
return new Model(s);
}
@Override
public boolean onCreateOptionsMenu(Menu menu) {
MenuInflater inflater = getMenuInflater();
inflater.inflate(R.menu.results_screen_option_menu, menu);
return true;
}
/**
* @category OptionsMenu
*/
@Override
public boolean onOptionsItemSelected(MenuItem item) {
switch (item.getItemId()) {
case R.id.select_all: {
int size = lView.getAdapter().getCount();
for (int i = 0; i <= size; i++) {
//************************** PROBLEM
lView.setItemChecked(i, true); // selects the item only for standard ArrayAdapter
Log.i("xxx", "looping " + i);
}
}
return true;
case R.id.select_none:
return true;
}
return false;
}
}
// ------------------------------------------------ ------------
public class Test_Class1 extends ArrayAdapter<Model> {
private final List<Model> list;
private final Activity context;
public Test_Class1(Activity context, List<Model> list) {
super(context, R.layout.rowbuttonlayout2, list);
this.context = context;
this.list = list;
}
static class ViewHolder {
protected TextView text;
protected CheckBox checkbox;
}
@Override
public View getView(int position, View convertView, ViewGroup parent) {
View view = null;
Log.i("xxx", "-> getView " + position);
if (convertView == null) {
LayoutInflater inflator = context.getLayoutInflater();
view = inflator.inflate(R.layout.rowbuttonlayout, null);
final ViewHolder viewHolder = new ViewHolder();
viewHolder.text = (TextView) view.findViewById(R.id.label);
viewHolder.checkbox = (CheckBox) view.findViewById(R.id.check);
viewHolder.checkbox
.setOnCheckedChangeListener(new CompoundButton.OnCheckedChangeListener() {
@Override
public void onCheckedChanged(CompoundButton buttonView,
boolean isChecked) {
Model element = (Model) viewHolder.checkbox
.getTag();
Log.i("xxx", "-> onCheckedChanged");
element.setSelected(buttonView.isChecked());
Log.i("xxx", "<- onCheckedChanged");
}
});
view.setTag(viewHolder);
viewHolder.checkbox.setTag(list.get(position));
} else {
view = convertView;
((ViewHolder) view.getTag()).checkbox.setTag(list.get(position));
}
ViewHolder holder = (ViewHolder) view.getTag();
holder.text.setText(list.get(position).getName());
Log.i("xxx", "holder.checkbox.setChecked: " + position);
holder.checkbox.setChecked(list.get(position).isSelected());
Log.i("xxx", "<- getView " + position);
return view;
}
}
行レイアウトは、setItemChecked()
が機能するためにCheckable
である必要があります。その場合、AndroidはCheckable
のsetChecked()
をユーザーが行をクリックします。独自のOnCheckedChangeListener
を設定する必要はありません。
詳細については、以下を参照してください。
ListViewの選択モードを設定する必要があります。たとえば:
list.setChoiceMode(ListView.CHOICE_MODE_SINGLE);
ただし、リストアイテムにカスタムレイアウトも使用している場合(R.layout.drawing_list_item
)、レイアウトにCheckableインターフェイスが実装されていることを確認する必要があります。
public class CheckableRelativeLayout extends RelativeLayout implements Checkable {
private boolean isChecked = false;
public CheckableRelativeLayout(Context context, AttributeSet attrs) {
super(context, attrs);
}
public boolean isChecked() {
return isChecked;
}
public void setChecked(boolean isChecked) {
this.isChecked = isChecked;
changeColor(isChecked);
}
public void toggle() {
this.isChecked = !this.isChecked;
changeColor(this.isChecked);
}
private void changeColor(boolean isChecked){
if (isChecked) {
setBackgroundColor(getResources().getColor(Android.R.color.holo_blue_light));
} else {
setBackgroundColor(getResources().getColor(Android.R.color.transparent));
}
}
}
次に、チェック可能なレイアウトが次の例のように定義されます。
<?xml version="1.0" encoding="utf-8"?>
<it.moondroid.banking.CheckableRelativeLayout xmlns:Android="http://schemas.Android.com/apk/res/Android"
Android:id="@+id/drawer_item_layout"
Android:layout_width="match_parent"
Android:layout_height="wrap_content"
Android:padding="10dp" >
<ImageView
Android:id="@+id/agenzie_item_icon"
Android:layout_width="30dp"
Android:layout_height="30dp"
Android:layout_alignParentLeft="true"
Android:layout_centerVertical="true"
Android:background="@Android:color/darker_gray"
Android:focusable="false" />
<TextView
Android:id="@+id/agenzie_item_name"
Android:layout_width="wrap_content"
Android:layout_height="wrap_content"
Android:layout_centerVertical="true"
Android:layout_toRightOf="@+id/agenzie_item_icon"
Android:focusable="false"
Android:paddingLeft="10dp"
Android:text="item"
Android:textAppearance="?android:attr/textAppearanceSmall"
Android:textColor="#FFFFFF" />
</it.moondroid.banking.CheckableRelativeLayout>
Checkableは、今すぐ受講したくない学習曲線です。アクティビティのOnItemClickListener
でCheckBoxを手動で設定および設定解除できます。 ArrayAdapter<MyObject>
のisChecked
リスト内のブールMyObject
一時変数を維持します。
public void onItemClick(AdapterView<?> av, View v, int position, long arg3) {
final ListView listView = (ListView) findViewById(Android.R.id.list);
MyObject item = (MyObject) listView.getItemAtPosition(position);
item.isChecked = !item.isChecked;
if(item.isChecked)
addItem(item);
else
removeItem(item);
}
CheckBox
自体をクリックするユーザーを説明するため。 getView
のCheckBox
で、カスタム配列アダプターのOnClickListener
実装で同じaddItem(item)/ removeItem(item)ロジックを使用します。
public View getView(int position, View convertView, ViewGroup viewGroup) {
CheckBox cb = null;
if (convertView == null) {
if(inflater == null) //cache to avoid reconstructing for each view
inflater = (LayoutInflater) getContext().getSystemService(Context.LAYOUT_INFLATER_SERVICE);
convertView =inflater.inflate(R.layout.list_item_text_right_checkmark, null);
CheckBox cb = (CheckBox) convertView.findViewById(R.id.check);
cb.setChecked((list.get(position).isChecked));
cb.setTag(list.get(position);
public void onClick(View v) {
if(v instanceof CheckBox) {
CheckBox cb = (CheckBox) v;
if(cb.isChecked()) //verify boolean logic here!!!
activityRef.get().addItem(cb.getTag()); //WeakReference to activity
else
activityRef.get().removeItem(cb.getTag());
}
});
} else cb = (CheckBox) convertView.findViewById(R.id.check);
cb.setChecked((list.get(position).isChecked));
...
}
ここでは、CheckBox :: setChecked()を呼び出すことが重要です。
これは、アレイアダプターが同じアクティビティを引き受けることができ、アレイアダプターがres/layout /で同じリストアイテムxml定義を引き受けることができる限り、トリックを行うようです。
チェック可能なリストアイテム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="match_parent"
Android:orientation="vertical" >
<TextView
Android:id="@+id/text"
Android:layout_width="wrap_content"
Android:layout_height="wrap_content"
Android:layout_toLeftOf="@+id/check"
Android:gravity="center_vertical"
Android:paddingLeft="8dp"
Android:textAppearance="?android:attr/textAppearanceMedium" />
<CheckBox
Android:id="@+id/check"
Android:layout_width="wrap_content"
Android:layout_height="wrap_content"
Android:layout_alignParentRight="true"
Android:focusable="false"/>
</RelativeLayout>
最初にCheckBoxの一部をチェックする必要がある場合は、膨張する前にMyObjectでisCheckedメンバーを設定するだけです。
応答のどれも私のために働いた。私の問題は、OnCreate/OnStart/OnPauseからExpandableListView.setItemCheckedへの最初の呼び出しだけが、ビューがまだ完全に初期化されていないかのように機能しないことでした。これを解決するために、私は次のことをしなければなりませんでした:
_expandableListView.Post(() =>
{
_expandableListView.SetItemChecked(fragmentPosition, true);
});