自定义列表项未响应选择器中的 state_checked

Posted

技术标签:

【中文标题】自定义列表项未响应选择器中的 state_checked【英文标题】:Custom list items not responding to state_checked in selector 【发布时间】:2012-11-17 14:21:48 【问题描述】:

我首先要说的是,我已经详细阅读了几乎所有关于 SO 的问题,我可以找到与自定义可检查列表项和选择器相关的问题。他们中的许多人都有类似的问题,但没有一个答案能解决我的问题。

在我的应用程序中,我展示了一个自定义列表活动。创建后,它会从调用它的意图中检索一组静态数据,并将该数据传递给它的自定义数组适配器。每个列表项都是一个实现Checkable 接口的简单RelativeLayout。默认情况下,如果您单击其中一项,则会显示一个新活动,其中显示有关所选联系人的详细信息。但是,如果长按列表中的项目,则会启动 ActionMode。此时单击列表中的项目不会显示详细活动,它只是将项目设置为选中。然后,如果用户选择了其中一个操作模式项,它会对选中的项执行操作。

要了解的重要一点是,在两种选择“模式”中,单击列表项会将其设置为选中。

我上面描述的所有内容都完美无缺。我的唯一问题与设置为选中时未突出显示的列表项的背景有关,即使使用默认选择器也是如此。

我想要做的是有两个选择器:每个选择模式一个。在第一种情况下,检查项目时背景不会改变,而在第二种情况下会改变。我尝试过实现自定义选择器,但即使在那些 state_checked 中也被忽略了!选择器的其他部分工作正常,但不是 state_checked。

我的 CheckableListItem 实现结合了许多不同示例的想法,所以如果我做错了什么,或者如果有更好的方法,请告诉我!

注意:有趣的一点是,如果我将 results_list_item.xml 中列表项的背景设置为我的选择器,而不是 ListView 的 listSelector 属性,那么背景 do 检查项目时更改。但是,这样做会导致我的选择器中的长按转换不起作用。

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_
    android:layout_
    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_
    android:layout_ 
    android:paddingLeft="10dp"
    android:paddingRight="10dp"
    android:paddingTop="5dp"
    android:paddingBottom="5dp" >

    <TextView android:id="@+id/results_list_item_textview1"
        android:layout_
        android:layout_
        android:layout_alignParentLeft="true"
        android:textSize="20sp"
        android:textColor="#000000"
        android:focusable="false" />

    <TextView android:id="@+id/results_list_item_textview2"
        android:layout_
        android:layout_
        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>

【问题讨论】:

您在哪里检查和取消检查您的项目?您也可以尝试使用 android:state_activited。 我自己不做。不过,他们正在接受检查。当我将日志语句放入项目的 setChecked 方法时,我看到它正在被调用。 当你直接设置背景时,在CheckableListItem 上尝试android:longClickable=true ..(如您的注释中所述) 没有骰子。我很确定它们已经可以长时间点击了,因为列表活动正在实现 OnItemLongClickListener,并且正在被调用。 'R.attr.state_checked' 是引用你自己的 'R' 类(即生成的类)还是引用来自 android API 的类? 【参考方案1】:

我在CheckedListItem 中更改并添加了这些方法,它对我有用:

@Override
public boolean onTouchEvent( MotionEvent event ) 

    int action = event.getAction() & MotionEvent.ACTION_MASK;
    if ( action == MotionEvent.ACTION_UP ) 
        toggle();
    

    return true;


public void toggle() 

    setChecked( !isChecked() );


private static final int[] CheckedStateSet =  android.R.attr.state_checked ;

问题似乎在于,在单击时,您从未处理过切换视图的选中状态。

【讨论】:

这对我不起作用...现在当我单击一个项目时没有任何反应。你改变的只有这些吗?您是否删除了任何其他方法? 不,我没有,但我也没有把它放在ListView 中。您的ListView 上有一个监听器来接收所有点击,因此您的视图将永远不会收到触摸事件。我建议将选择模式设置为CHOICE_MODE_SINGLECHOICE_MODE_MULTIPLE(无论您需要哪个),这将使列表项成为可选择的。因此,您将在 CheckedListItem 类中使用 state_selected 而不是 state_checked【参考方案2】:

如果还没有解决这个问题,请尝试为您的“可检查布局”提供一个背景可绘制对象,其中包含选择器以及其中的状态和颜色定义。否则,drawableStateChanged 不会做任何事情,因为 mBackground 为空。 (例如 android:background="@drawable/list_selector")

然后确保您使用listview.setChoiceMode(AbsListView.CHOICE_MODE_MULTIPLE) 来允许检查多个项目。

您不必实现setOnItemClickListener 来检查项目,因为设置选择模式已经自动完成。 (也不要将 Checkable 布局设置为可点击)

好吧,至少我是这样解决我的问题的。

【讨论】:

以上是关于自定义列表项未响应选择器中的 state_checked的主要内容,如果未能解决你的问题,请参考以下文章

在文件选择器中禁用下拉列表[重复]

从listView中删除项目仅删除最后一项未选择的元素android

从 Kentico 10 中的自定义表重复器中过滤重复的列值

U3D自定义Inspector项未触发保存事件的解决方案

菜单栏中的菜单项未激活,无法选择或单击

使用 kotlinx 序列化响应中的自定义对象列表