63

我想知道如何更改 listView 上所选项目的背景颜色。我只想更改用户单击的特定项目,这意味着如果用户单击另一个项目,它将是突出显示的项目。好吧,因为我希望它尽可能简单并使用默认的android listview,所以我使用了这个代码:

record_list.setOnItemClickListener(new AdapterView.OnItemClickListener() {
            @Override
            public void onItemClick(AdapterView<?> adapterView, View view, int i, long l) {
                try{
                    for (int ctr=0;ctr<=record_items.length;ctr++){
                        if(i==ctr){
                            record_list.getChildAt(ctr).setBackgroundColor(Color.CYAN);
                        }else{
                            record_list.getChildAt(ctr).setBackgroundColor(Color.WHITE);
                        }
                    }
                }
                catch (Exception e){
                    e.printStackTrace();
                }
                Log.v("Selected item",record_list.getItemAtPosition(i));
            }
        });

好的,这个正在工作,但问题是它很慢。现在我想知道是否有任何其他方法可以解决我的问题,这将提供与我所做的相同的输出。

我尝试使用record_list.getSelectedView().setBackgroundColor(Color.CYAN);,但它给了我一个空指针异常。

我也尝试了 selector.xml 但它也没有成功。此外,ListView 上还有一个属性,称为 listSelector。正如文档“Drawable used to指示列表中当前选定的项目”所述,它是一个drawable。我也相信这应该可以解决问题,是的,它可以在我的模拟器上解决问题,但不是在我的银河选项卡上。我也尝试了其他方法,但没有任何效果如我所愿。

4

17 回答 17

84

您可以跟踪当前选定元素的位置:

    OnItemClickListener listViewOnItemClick = new OnItemClickListener() {

        @Override
        public void onItemClick(AdapterView<?> adapter, View arg1, int position, long id) {
                mSelectedItem = position;
                mAdapter.notifyDataSetChanged();
        }
    };

并覆盖适配器的 getView 方法:

    @Override
    public View getView(int position, View convertView, ViewGroup parent) {
        final View view = View.inflate(context, R.layout.item_list, null);

        if (position == mSelectedItem) {
            // set your color
        }

        return view;
    }

对我来说,它成功了。

于 2013-06-07T07:03:31.363 回答
72

You can use a selector. Change the colors values and modify the below according to your needs.

bkg.xml in drawable folder

<?xml version="1.0" encoding="utf-8"?>
<selector xmlns:android="http://schemas.android.com/apk/res/android">
<item android:state_pressed="true" 
    android:drawable="@drawable/pressed" />
<item  android:state_focused="false" 
    android:drawable="@drawable/normal" />
</selector>

pressed.xml in drawable folder

<?xml version="1.0" encoding="UTF-8"?> 
<shape xmlns:android="http://schemas.android.com/apk/res/android"> 
<solid android:color="#FF1A47"/>  // color   
<stroke android:width="3dp"
        android:color="#0FECFF"/> // border
<padding android:left="5dp"
         android:top="5dp"
         android:right="5dp"
         android:bottom="5dp"/> 
<corners android:bottomRightRadius="7dp" // for rounded corners
         android:bottomLeftRadius="7dp" 
         android:topLeftRadius="7dp"
         android:topRightRadius="7dp"/> 
</shape>

normal.xml in drawable folder

<?xml version="1.0" encoding="UTF-8"?> 
<shape xmlns:android="http://schemas.android.com/apk/res/android"> 
<solid android:color="#FFFFFF"/>    
<stroke android:width="3dp"
        android:color="#0FECFF" />

<padding android:left="5dp"
         android:top="5dp"
         android:right="5dp"
         android:bottom="5dp"/> 
<corners android:bottomRightRadius="7dp"
         android:bottomLeftRadius="7dp" 
         android:topLeftRadius="7dp"
         android:topRightRadius="7dp"/> 
</shape>

Set the background drawable to listview custom layout to be inflated for each row

I recommend using a custom listview with a custom adapter.

  android:background="@drawable/bkg"     

If you have not used a custom adapter you can set the listselector to listview as below

   android:listSelector="@drawable/bkg" 
于 2013-06-07T05:00:45.807 回答
31

定义变量

private ListView mListView;

初始化变量

mListView = (ListView)findViewById(R.id.list_view);

列表视图的 OnItemClickListener

   mListView.setOnItemClickListener(new OnItemClickListener() {

        @Override
        public void onItemClick(AdapterView<?> adpterView, View view, int position,
                long id) {
            for (int i = 0; i < mListView.getChildCount(); i++) {
                if(position == i ){
                    mListView.getChildAt(i).setBackgroundColor(Color.BLUE);     
                }else{
                    mListView.getChildAt(i).setBackgroundColor(Color.TRANSPARENT);
                }
            }
        }
    });

构建并运行项目 - 完成

于 2015-05-05T10:49:46.337 回答
16

如果您想让项目在单击后保持突出显示,则需要手动将其设置为在 onItemClick 侦听器中被选中

Android ListView 所选项目保持突出显示

myList.setOnItemClickListener(new OnItemClickListener() {
    public void onItemClick(AdapterView<?> parent, View view, int position, long id) {   
        view.setSelected(true); // <== Will cause the highlight to remain
        //... do more stuff                          
    }});

这假设您的选择器中有一个 state_selected 项目:

<?xml version="1.0" encoding="utf-8"?>
  <selector xmlns:android="http://schemas.android.com/apk/res/android">  
  <item android:state_enabled="true" android:state_pressed="true" android:drawable="@color/red" />
  <item android:state_enabled="true" android:state_focused="true" android:drawable="@color/red" />
  <item android:state_enabled="true" android:state_selected="true" android:drawable="@color/red" />
  <item android:drawable="@color/white" />
</selector>
于 2013-09-25T23:14:00.660 回答
11

方法一:

在您的 xml 布局活动/片段中更新 ListView:

<ListView
   ...
   android:choiceMode="singleChoice"
   android:listSelector="@android:color/darker_gray"
/>

就是这样,你完成了!

如果您想要以编程方式处理此问题,请使用方法 2 ...

方法二:

如果您使用 ListFragment,您可以覆盖 onListItemClick(),使用视图设置颜色。保存当前选择的视图以重置最后选择的颜色。

请注意,这仅适用于适合一个屏幕的列表视图,因为视图被回收。

public class MyListFragment extends ListFragment {
    View previousSelectedItem;
...
    @Override
    public void onListItemClick(ListView parent, View v, int position, long id) {
        super.onListItemClick(parent, v, position, id);
        if (previousSelectedItem!=null) {
            previousSelectedItem.setBackgroundColor(Color.WHITE);
        }
        previousSelectedItem=v;
        v.setBackgroundColor(Color.BLUE);
    }
}
于 2016-06-22T21:20:56.740 回答
7

对于那些想知道究竟需要做什么才能在您向上滚动时保持选中行的人。就是state_activated剩下的由内部功能来处理,你不用担心切换,并且可以选择多个项目。我不需要使用 notifyDataSetChanged() 或 setSelected(true) 方法。

将此行添加到您的选择器文件中,对我而言 drawable\row_background.xml

<?xml version="1.0" encoding="utf-8"?>
<selector xmlns:android="http://schemas.android.com/apk/res/android">
    <item android:state_pressed="true" android:drawable="@android:color/holo_blue_light"/>

    <item android:state_enabled="true" android:state_pressed="true" android:drawable="@android:color/holo_blue_light" />
    <item android:state_enabled="true" android:state_focused="true" android:drawable="@android:color/holo_blue_bright" />
    <item android:state_enabled="true" android:state_selected="true" android:drawable="@android:color/holo_blue_light" />
    <item android:state_activated="true" android:drawable="@android:color/holo_blue_light" />

    <item android:drawable="@android:color/transparent"/>
</selector>

然后在 layout\custom_row.xml

<?xml version="1.0" encoding="utf-8"?>
    <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
        android:layout_width="fill_parent"
        android:layout_height="wrap_content"
        android:padding="10dip"
        android:background="@drawable/row_background"
        android:orientation="vertical">
  <TextView
        android:id="@+id/line1"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content" />
</LinearLayout>

有关更多信息,我将它与 ListView 适配器一起使用,使用 myList.setChoiceMode(ListView.CHOICE_MODE_MULTIPLE_MODAL); 和 myList.setMultiChoiceModeListener(new MultiChoiceModeListener()...

从这个例子:http ://www.androidbegin.com/tutorial/android-delete-multiple-selected-items-listview-tutorial/

此外,您(应该)将此结构用于列表适配器耦合: List myList = new ArrayList();

而不是:ArrayList myList = new ArrayList();

说明:Java 中的 List 类型与 ArrayList 类型

于 2017-02-25T18:21:25.037 回答
6

首先,您可以在您的可绘制文件夹中创建如下所示的选择器 xml 文件 drawable/list_item_selector.xml

<?xml version="1.0" encoding="utf-8"?>
<selector xmlns:android="http://schemas.android.com/apk/res/android">

    <item android:state_activated="true">
      <shape android:shape="rectangle">
        <solid android:color="#333333" />
        <padding android:left="5dp" android:right="5dp" />
      </shape></item>
    <item><shape android:shape="rectangle">
            <solid android:color="#222222" />
        </shape></item>

</selector>

然后在您的列表视图中将背景指定为

android:background="@drawable/list_item_selector"
于 2013-06-07T05:46:36.347 回答
5

我发现的最简单的方法:

在您的活动 XML 中添加以下行:

<ListView 
... 
android:choiceMode="singleChoice"
android:listSelector="#666666"
/>

或以编程方式设置这些属性:

listView.setSelector(Drawable selector)
listView.setSelector(int resourceId)

我的具体例子:

   <ListView
            android:choiceMode="singleChoice"
            android:listSelector="#666666"
            android:layout_width="match_parent"
            android:layout_height="wrap_content"
            android:id="@+id/listView"/>

感谢 AJG: https ://stackoverflow.com/a/25131125/1687010

于 2016-05-16T06:49:58.477 回答
3
View updateview;// above oncreate method
listView.setChoiceMode(ListView.CHOICE_MODE_SINGLE);


listView.setOnItemClickListener(new AdapterView.OnItemClickListener() {
    @Override
    public void onItemClick(AdapterView<?> parent, View view, int position, long id) {
         if (updateview != null) 
            updateview.setBackgroundColor(Color.TRANSPARENT);
         updateview = view;   

         view.setBackgroundColor(Color.CYAN);
     }
});
于 2016-07-15T22:36:37.633 回答
3

我也在做类似的事情:突出显示所选列表项的背景(将其更改为红色)并将项目中的文本颜色设置为白色。

我可以想出一个“简单但不高效”的方法:在自定义适配器中维护选定项的位置,并在 ListView 的OnItemClickListener实现中更改它:

// The OnItemClickListener implementation
@Override
public void onItemClick(AdapterView<?> parent, View view, int position, long id) {
    mListViewAdapter.setSelectedItem(position);
}

// The custom Adapter
private int mSelectedPosition = -1;
public void setSelectedItem (int itemPosition) {
    mSelectedPosition = itemPosition;
    notifyDataSetChanged();
}

然后在方法中更新所选项目的背景和文本颜色getView()

// The custom Adapter
@Override
public View getView(int position, View convertView, ViewGroup parent) {
    ...
    if (position == mSelectedPosition) {
        // customize the selected item's background and sub views
        convertView.setBackgroundColor(YOUR_HIGHLIGHT_COLOR);
        textView.setTextColor(TEXT_COLOR);
    } else {
        ...
    }
}

找了一会,发现很多人都提到要设置android:listSelector="YOUR_SELECTOR"。尝试了一段时间后,我发现突出显示选定 ListView 项目背景的最简单方法只需将两行设置为 ListView 的布局资源即可:

android:choiceMode="singleChoice"
android:listSelector="YOUR_COLOR"

还有其他方法可以使其工作,例如自定义activatedBackgroundIndicator主题。但我认为这将是一个更通用的解决方案,因为它会影响整个主题。

于 2015-12-15T06:51:48.793 回答
2

使用下面的 xml 作为 listitem 背景,它将解决所有问题。尽管您向下滚动,但所选内容将突出显示。

<?xml version="1.0" encoding="utf-8"?>
<selector xmlns:android="http://schemas.android.com/apk/res/android">
<item android:drawable="@android:color/holo_orange_dark" android:state_pressed="true"/>
<item android:drawable="@android:color/holo_green_light" android:state_selected="true"/>
<item android:drawable="@android:color/holo_green_light" android:state_activated="true"/>

谢谢,纳根德拉

于 2014-11-28T10:10:48.513 回答
2

我知道这是一个老问题,但我为这个需求提供了一个简单的解决方案(没有循环!):

//On your adapter create a variable:
private View lastSelectedItem;

//Define the folowing method:
private void toggleBackgroundItem(View view) {
    if (lastSelectedItem != null) {
        lastSelectedItem.setBackgroundColor(Color.TRANSPARENT);
    }
    view.setBackgroundColor(getResources().getColor(R.color.colorPrimaryDark));
    lastSelectedItem = view;

}
//finally invoque the method onItemClick
lvSac.setOnItemClickListener(new AdapterView.OnItemClickListener()

{
    @Override
    public void onItemClick (AdapterView < ? > adapterView, View view,int i, long l){

    toggleBackgroundItem(view);
}
}
于 2017-03-11T13:04:25.763 回答
2

在 ListView 集中:

android:choiceMode="singleChoice"

为背景创建一个选择器(drawable/selector_gray.xml):

<?xml version="1.0" encoding="utf-8"?>
<selector xmlns:android="http://schemas.android.com/apk/res/android">
    <item android:drawable="@color/gray" android:state_checked="true" />
    <item android:drawable="@color/white" />
</selector>

为列表添加一项:

<?xml version="1.0" encoding="utf-8"?>
<TextView
    xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="wrap_content"
    android:gravity="center"
    android:padding="5dp"
    android:background="@drawable/selector_gray"
    android:textColor="@color/colorPrimary"
    tools:text="Your text" />

在 ViewHolder 中,您可以为该项目充气。

于 2017-08-30T18:17:29.393 回答
1

假设您希望每次单击一个项目。然后这段代码运行良好。让我们将列表视图名称作为 stlist

stList.setOnItemClickListener(new AdapterView.OnItemClickListener() {
        @Override
        // here i overide the onitemclick method in onitemclick listener
            public void onItemClick(AdapterView<?> parent, View view,
                                int position, long id) {

            //color change
            //selected item colored

            for(int i=0; i<stList.getAdapter().getCount();i++)
            {
                stList.getChildAt(i).setBackgroundColor(Color.TRANSPARENT);
            }

            parent.getChildAt(position).setBackgroundColor(Color.GRAY);

    });
于 2015-05-09T03:48:47.783 回答
1

nameofList.getChildAt(position).setBackgroundColor(RED);

为我工作

于 2019-10-04T04:17:00.300 回答
1

这是一个简单的方法,即使列表也很长,也可以处理选择:

public View getView(final int position, View convertView, ViewGroup parent) {
    // TODO Auto-generated method stub
    Holder holder=new Holder();
    View rowView;
    rowView = inflater.inflate(R.layout.list_item, null);
    //Handle your items.
    //StringHolder.mSelectedItem is a public static variable.
    if(getItemId(position)==StringHolder.mSelectedItem){
        rowView.setBackgroundColor(Color.LTGRAY);

    }else{
        rowView.setBackgroundColor(Color.TRANSPARENT);
    }
    return rowView;
}

然后在你的 onclicklistener 中:

list.setOnItemClickListener(new AdapterView.OnItemClickListener() {
        @Override
        public void onItemClick(AdapterView<?> adapterView, View view, int i, long l) {
            StringHolder.mSelectedItem = catagoryAdapter.getItemId(i-1);
            catagoryAdapter.notifyDataSetChanged();
.....
于 2016-06-29T07:29:28.300 回答
0

这很简单。在“OnItemClick”的构造函数中使用参数“view”,它是代表listView或GridView的项目视图的第二个参数,它成为adapterView自己创建的新项目视图。因此,要为 SELECTED ITEM 本身设置新颜色,请执行以下操作:

public void onItemClick(AdapterView<?> adapterView, View view, int i, long l){ //view is instead of the view like textView , ImageView, or whatever view.setBackgroundColor(Color.green); }

如果您执行任何不同的代码来设置新颜色,您将面临尴尬的行为,例如绿色将应用于未单击的项目。

于 2018-11-22T22:54:06.677 回答