カスタム チェック可能リスト アイテムとセレクターに関連する SO に関するほぼすべての質問を詳細に読んだことから始めます。それらの多くは同様の問題を抱えていますが、私の問題を解決する答えはありません。
アプリのある時点で、カスタム リスト アクティビティを提示します。作成されると、それを呼び出したインテントから一連の静的データを取得し、そのデータをカスタム配列アダプターに渡します。各リスト項目は、Checkableインターフェイスを実装する単純な RelativeLayout です。デフォルトでは、項目の 1 つをクリックすると、選択した連絡先に関する詳細情報を表示する新しいアクティビティが表示されます。ただし、リスト内の項目が長押しされると、ActionMode が開始されます。この時点でリスト内のアイテムをクリックしても、詳細アクティビティは表示されず、アイテムがオンに設定されるだけです。次に、ユーザーがアクション モード項目の 1 つを選択すると、チェックされた項目に対してアクションが実行されます。
理解しておくべき重要なことは、両方の選択「モード」で、リスト項目をクリックすると、チェックされた状態に設定されるということです。
上記で説明したことはすべて完全に機能します。私の唯一の問題は、デフォルトのセレクターを使用しても、リスト項目がチェック済みに設定されているときに強調表示されないリスト項目の背景に関係しています。
私がやりたいことは、選択モードごとに 1 つずつ、2 つのセレクターを用意することです。前者では項目がチェックされたときに背景が変化しませんが、後者では変化します。カスタムセレクターを実装しようとしましたが、それらの state_checked でも無視されます! セレクターの他の部分は正常に動作しますが、state_checked では動作しません。
CheckableListItem の私の実装には、さまざまな例からのアイデアが組み込まれているため、何か間違ったことをしている場合、またはより良い方法がある場合はお知らせください。
注: 興味深い点は、ListView の listSelector プロパティではなく、results_list_item.xml のリスト項目の背景をセレクターに設定すると、項目がチェックされると背景が変化することです。ただし、これを行うと、セレクターの長押し遷移が機能しなくなります。
ResultsActivity.java :
public class ResultsActivity extends ListActivity implements OnItemLongClickListener {
private ListView listView; // Reference to the list belonging to this activity
private ActionMode mActionMode; // Reference to the action mode that can be started
private boolean selectionMode; // Detail mode or check mode
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_results);
// When the home icon is pressed, go back
ActionBar actionBar = getActionBar();
actionBar.setDisplayHomeAsUpEnabled(true);
// Get a reference to the list
listView = getListView();
// Initially in detail mode
selectionMode = true;
// Get the contacts from the intent data and pass them to the contact adapter
@SuppressWarnings("unchecked")
ArrayList<Contact> results = ((ArrayList<Contact>)getIntent().getSerializableExtra("results"));
Contact[] contacts = new Contact[results.size()];
ContactArrayAdapter adapter = new ContactArrayAdapter(this, results.toArray(contacts));
setListAdapter(adapter);
// We will decide what happens when an item is long-clicked
listView.setOnItemLongClickListener(this);
}
/**
* If we are in detail mode, when an item in the list is clicked
* create an instance of the detail activity and pass it the
* chosen contact
*/
public void onListItemClick(ListView l, View v, int position, long id) {
if (selectionMode) {
Intent displayContact = new Intent(this, ContactActivity.class);
displayContact.putExtra("contact", (Contact)l.getAdapter().getItem(position));
startActivity(displayContact);
}
}
public boolean onCreateOptionsMenu(Menu menu) {
return super.onCreateOptionsMenu(menu);
}
/**
* If the home button is pressed, go back to the
* search activity
*/
public boolean onOptionsItemSelected(MenuItem item) {
switch (item.getItemId()) {
case android.R.id.home:
Intent intent = new Intent(this, SearchActivity.class);
intent.addFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP);
startActivity(intent);
return true;
default:
return super.onOptionsItemSelected(item);
}
}
/**
* When an item is long-pressed, switch selection modes
* and start the action mode
*/
public boolean onItemLongClick(AdapterView<?> adapter, View view, int position, long i) {
if (mActionMode != null)
return false;
if (selectionMode) {
toggleSelectionMode();
listView.startActionMode(new ListActionMode(this, getListView()));
return true;
}
return false;
}
/**
* Clear the list's checked items and switch selection modes
*/
public void toggleSelectionMode() {
listView.clearChoices();
((ContactArrayAdapter)listView.getAdapter()).notifyDataSetChanged();
if (selectionMode) {
selectionMode = false;
} else {
selectionMode = true;
}
}
}
activity_results.xml :
<ListView xmlns:android="http://schemas.android.com/apk/res/android"
android:id="@android:id/list"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:choiceMode="multipleChoice"
android:listSelector="@drawable/list_selector" />
list_selector.xml :
<selector xmlns:android="http://schemas.android.com/apk/res/android" >
<item android:state_pressed="true" android:drawable="@drawable/blue_transition" />
<item android:state_checked="true" android:drawable="@drawable/blue" />
</selector>
TwoLineArrayAdapter :
public abstract class TwoLineArrayAdapter extends ArrayAdapter<Contact> {
private int mListItemLayoutResId;
public TwoLineArrayAdapter(Context context, Contact[] results) {
this(context, R.layout.results_list_item, results);
}
public TwoLineArrayAdapter(Context context, int listItemLayoutResourceId, Contact[] results) {
super(context, listItemLayoutResourceId, results);
mListItemLayoutResId = listItemLayoutResourceId;
}
public View getView(int position, View convertView, ViewGroup parent) {
LayoutInflater inflater = (LayoutInflater)getContext().getSystemService(Context.LAYOUT_INFLATER_SERVICE);
View listItemView = convertView;
if (convertView == null) {
listItemView = inflater.inflate(mListItemLayoutResId, parent, false);
}
// Get the text views within the layout
TextView lineOneView = (TextView)listItemView.findViewById(R.id.results_list_item_textview1);
TextView lineTwoView = (TextView)listItemView.findViewById(R.id.results_list_item_textview2);
Contact c = (Contact)getItem(position);
lineOneView.setText(lineOneText(c));
lineTwoView.setText(lineTwoText(c));
return listItemView;
}
public abstract String lineOneText(Contact c);
public abstract String lineTwoText(Contact c);
}
ContactArrayAdapter :
public class ContactArrayAdapter extends TwoLineArrayAdapter {
public ContactArrayAdapter(Context context, Contact[] contacts) {
super(context, contacts);
}
public String lineOneText(Contact c) {
return (c.getLastName() + ", " + c.getFirstName());
}
public String lineTwoText(Contact c) {
return c.getDepartment();
}
}
CheckableListItem.java :
public class CheckableListItem extends RelativeLayout implements Checkable {
private boolean isChecked;
private List<Checkable> checkableViews;
public CheckableListItem(Context context, AttributeSet attrs,
int defStyle) {
super(context, attrs, defStyle);
initialise(attrs);
}
public CheckableListItem(Context context, AttributeSet attrs) {
super(context, attrs);
initialise(attrs);
}
public CheckableListItem(Context context, int checkableId) {
super(context);
initialise(null);
}
private void initialise(AttributeSet attrs) {
this.isChecked = false;
this.checkableViews = new ArrayList<Checkable>(5);
}
public boolean isChecked() {
return isChecked;
}
public void setChecked(boolean check) {
isChecked = check;
for (Checkable c : checkableViews) {
c.setChecked(check);
}
refreshDrawableState();
}
public void toggle() {
isChecked = !isChecked;
for (Checkable c : checkableViews) {
c.toggle();
}
}
private static final int[] CheckedStateSet = {
android.R.attr.state_checked
};
protected int[] onCreateDrawableState(int extraSpace) {
final int[] drawableState = super.onCreateDrawableState(extraSpace + 1);
if (isChecked()) {
mergeDrawableStates(drawableState, CheckedStateSet);
}
return drawableState;
}
protected void onFinishInflate() {
super.onFinishInflate();
final int childCount = this.getChildCount();
for (int i = 0; i < childCount; i++) {
findCheckableChildren(this.getChildAt(i));
}
}
private void findCheckableChildren(View v) {
if (v instanceof Checkable) {
this.checkableViews.add((Checkable) v);
}
if (v instanceof ViewGroup) {
final ViewGroup vg = (ViewGroup) v;
final int childCount = vg.getChildCount();
for (int i = 0; i < childCount; i++) {
findCheckableChildren(vg.getChildAt(i));
}
}
}
}
results_list_item.xml :
<com.test.mycompany.Widgets.CheckableListItem
android:id="@+id/results_list_item"
xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:paddingLeft="10dp"
android:paddingRight="10dp"
android:paddingTop="5dp"
android:paddingBottom="5dp" >
<TextView android:id="@+id/results_list_item_textview1"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_alignParentLeft="true"
android:textSize="20sp"
android:textColor="#000000"
android:focusable="false" />
<TextView android:id="@+id/results_list_item_textview2"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_alignParentLeft="true"
android:layout_below="@id/results_list_item_textview1"
android:textSize="16sp"
android:textColor="@android:color/darker_gray"
android:focusable="false" />
</com.test.mycompany.Widgets.CheckableListItem>