it-swarm.com.ru

Android - сохранить элемент ListView выделенным после того, как на него нажали

Таким образом, у меня есть действие с 2 виджетами ListView, когда вы выбираете значение в первом, второе заполняется значениями, связанными с выбором в первом ListView. Этот механизм работает без проблем, но теперь я хочу, чтобы выбор пользователя оставался выделенным. Я прочитал большое количество вопросов, связанных с этой темой, и кажется, что есть множество способов, которыми можно это сделать, но после попытки 4-5 из них я все еще не могу заставить его работать.

Он работает со вторым ListView с использованием XML-атрибута Android:listSelector="#CCCCCC", но кажется, что он стирается, как только OnItemClickListener вводится в микс (как тот, который я использую в своем первом ListView).

Пока вот что я получил:

Пользовательский OnItemClickListener Я нашел просмотр различных ответов по этой теме (немного изменил его, чтобы он загружал мою информацию во второй ListView): 

private class ItemHighlighterListener implements OnItemClickListener {

    private View oldSelection = null;

    public void clearSelection() {
        if(oldSelection != null) {
            oldSelection.setBackgroundColor(Android.R.color.transparent);
        }
    }

    public void onItemClick(AdapterView<?> parent, View view, int pos, long id) {
        clearSelection();
        oldSelection = view;
        view.setBackgroundDrawable(view.getContext().getResources().getDrawable(R.drawable.list_selector));
        loadClubs(mXMLPortalOptions.getRegion(pos).getId());
        mClubList.setAdapter(new ArrayAdapter<String>(getApplicationContext(), R.layout.list_item_white, mClubs));
    }
}

Вот мой файл list_selector.xml

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

    <item Android:state_selected="true"><shape>
            <solid Android:color="#CCCCCC" />
        </shape></item>

    <item Android:state_selected="false"><shape>
            <solid Android:color="#FFFFFF" />
        </shape></item>

</selector>

Метод (OnItemClick) вызывается и выполняется, но фон моего ListItem остается того же цвета: /

Я не могу поверить, что эта простая задача оказалась настолько сложной.

Если я пропустил код, который может быть полезен, или если в моем вопросе отсутствуют детали, не стесняйтесь указывать на это, и я сделаю все возможное, чтобы объяснить себя.

36
Jean-Philippe Roy

Поместите переменную позиции для выбранного элемента. Измените позицию в методе onItemClicked(). Проверьте выбранную позицию в List Adapter внутри getView() и установите фон для выбранного элемента.

public class TestAdapter extends BaseAdapter
{
    private Context context;
    private ArrayList<TestList> testList;
    private int selectedIndex;
    private int selectedColor = Color.parseColor("#1b1b1b");

    public TestAdapter(Context ctx, ArrayList<TestList> testList)
    {
        this.context = ctx;
        this.testList = testList;
        selectedIndex = -1;
    }

    public void setSelectedIndex(int ind)
    {
        selectedIndex = ind;
        notifyDataSetChanged();
    }

    @Override
    public int getCount()
    {
        return testList.size();
    }

    @Override
    public Object getItem(int position)
    {
        return testList.get(position);
    }

    @Override
    public long getItemId(int position)
    {
        return position;
    }

    private class ViewHolder
    {
        TextView tv;
    }

    @Override
    public View getView(int position, View convertView, ViewGroup parent)
    {
        View vi = convertView;
        ViewHolder holder;
        if(convertView == null)
        {
            vi = LayoutInflater.from(context).inflate(R.layout.test_list_item, null);
            holder = new ViewHolder();

            holder.tv = (TextView) vi;

            vi.setTag(holder);
        }
        else
        {
            holder = (ViewHolder) vi.getTag();
        }

        if(selectedIndex!= -1 && position == selectedIndex)
        {
            holder.tv.setBackgroundColor(Color.BLACK);
        }
        else
        {
            holder.tv.setBackgroundColor(selectedColor);
        }
        holder.tv.setText("" + (position + 1) + " " + testList.get(position).getTestText());

        return vi;
    }

}

Теперь установите переменную selectedIndex, когда щелкнет элемент списка.

public class TestActivity extends Activity implements OnItemClickListener
{
    // Implemented onItemClickListener

    @Override
    public void onItemClick(AdapterView<?> parent, View view, int position, long id)
    {
        adapter.setSelectedIndex(position);
    }
}
61
Shaiful

Чтобы расширить отличное решение Шайфула, вы можете не заставить его работать в вашей ситуации.

Если вы используете весь свой код в public void onListItemClick(ListView l, View v, int index, long id), если вы используете фрагменты и вам нужно объявлять интерфейс вместо реализации OnListItemClickListener или что-то, из-за чего ваш IDE генерирует ошибки, вам может потребоваться доступ к переменным и методам статически.

public static int selectedPosition = 0;
ArrayAdapter<Your_obj> adapter = null;

@Override
public void onListItemClick(ListView l, View v, int index, long id) {
    super.onListItemClick(l, v, index, id);

        selectedPosition = index;
        Your_adapter.setSelectedIndex(selectedPosition);
        adapter.notifyDataSetChanged();
}

И в Your_adapter:

private static int selectedIndex;

//public Your_adapter...

public static void setSelectedIndex(int ind) {
    selectedIndex = ind;
}

@Override
public View getView(int position, View convertView, ViewGroup parent) {
    WellHolder holder = null;

    if (null == convertView) {

                //set up your "holder"
    }

    if (position == selectedIndex) {
        convertView.setBackgroundColor(convertView.getResources().getColor(R.color.cyan));
    }
    else {
        convertView.setBackgroundColor(convertView.getResources().getColor(R.color.silver));
    }

    return convertView;
}

Некоторые другие различия заключаются в том, что вам не нужно инициализировать какие-либо переменные как «0» или «-1», а в вашей деятельности вызывается notifyDataSetChanged ().

Еще раз спасибо за ваше решение @Shaiful. Это, безусловно, помогло мне сэкономить время, пытаясь настроить то, что по умолчанию в iOS работает на Android, избегая выбора/элемента/сосредоточенного/нажатого/и т.д.

5
whyoz

Я столкнулся с подобной проблемой. Это мое решение:

Сначала добавьте пользовательский селектор списка в ваш список:

<ListView
    Android:id="@+id/list"
    Android:layout_width="match_parent"
    Android:layout_height="fill_parent"
    Android:listSelector="@drawable/listselector" />

Внутри listselector.xml:

<?xml version="1.0" encoding="utf-8"?>
<selector xmlns:Android="http://schemas.Android.com/apk/res/Android">
    <item Android:state_accelerated="false"
        Android:drawable="@drawable/bg" />
</selector>

И, наконец, нарисованный файл bg.xml с цветом вашей подсветки:

<?xml version="1.0" encoding="utf-8"?>
<shape xmlns:Android="http://schemas.Android.com/apk/res/Android">
    <solid Android:color="#33b5e6"/>
</shape>
4
zjadarka
//create a list_itemselectorin drawable folder
//you will get the list item selected background color change once you select //the item

    <selector xmlns:Android="http://schemas.Android.com/apk/res/Android">

        <!-- Focused State -->
        <item Android:state_focused="true"><shape>
                <solid Android:color="#66FFFFFF" />
            </shape></item>
        <!-- Pressed State -->

        <item Android:state_pressed="true"><shape>
                <solid Android:color="@color/Black" />
            </shape></item>

        <!-- Default State -->
        <item><shape>
                <solid Android:color="@color/Black" />
            </shape></item>

    </selector>


    //create a list in layout folder
      <ListView
            Android:id="@+id/mySlidingList"
            Android:layout_width="match_parent"
            Android:layout_height="match_parent"
            Android:choiceMode="singleChoice"
            Android:divider="@color/GrayHot"
            Android:dividerHeight="1dip"
            Android:listSelector="@drawable/list_itemselector"
            Android:scrollbars="none" />

// И посмотреть вывод.

2
DeepakPanwar

lv.setSelector(R.drawable.highlighter);

поместите изображение highlighter.png в папку для рисования
самый простой способ выделить выбранный элемент в виде списка.

2
Supun Perera

Я думаю, что самое лучшее и простое решение - это. Вам не необходимо установить любой Android:listSelector в самом ListView или внести какие-либо изменения в адаптер. Вы не даже должны вызывать любую setSelection(position) в OnItemClickListener, поскольку она обрабатывается автоматически.

  1. Установите для вашего ListView:

    Android:choiceMode="singleChoice"
    
  2. Установить фон самого элемента списка:

    Android:background="?android:attr/activatedBackgroundIndicator"
    
  3. Это оно.

Таким образом, вы получите поведение системы по умолчанию. Вот как это делается в стандартном макете Android.R.layout.simple_list_item_activated_1

2
Robyer

Я искал его две недели назад, и в результате этого нельзя использовать селектор с возможностью рисования ........ Подробнее об этом читайте в блоге разработчиков Android: Touch Mode

В резюме: только когда ваш палец на экране, элемент выбран. 

Другая возможность - сохранить выбранный элемент в var и Paint по-разному, используя ваш пользовательский адаптер, как говорит Shaiful.

2
i.masm

Чтобы подвести итог этого поста и, возможно, помочь кому-то еще в будущем, я предлагаю ответ :)

Сначала нам нужно создать файл res/drawable/list_item_background.xml со следующим содержимым:

<?xml version="1.0" encoding="utf-8"?>
<selector xmlns:Android="http://schemas.Android.com/apk/res/Android">
    <item
        Android:state_activated="true"
        Android:drawable="@color/list_item_activated" />
    <item
        Android:drawable="@color/list_item_default" />
</selector>

Конечно, укажите свои ресурсы для рисования. И вы также можете добавить другие <item> элементы с различными состояниями, такими как state_pressed, state_focused и т.д.

Затем мы должны установить параметр background для нашего элемента элемента списка ViewGroup (f.i. res/layout/list_item_layout.xml) следующим образом:

Android:background="@drawable/list_item_background"

Следующим шагом является изменение нашего пользовательского класса Adapter. Вот следующий фрагмент кода:

public class CustomAdapter extends BaseAdapter {
    private List<Item> items;
    private LayoutInflater itemInflater;        
    private int selectedIndex; // add this

    public CustomAdapter(Context c, List<Item> items) {
        this.items = items;
        this.itemInflater = LayoutInflater.from(c);
        selectedIndex = -1; // add this
    }

    /* add this */
    public void setSelectedIndex(int index) {
        selectedIndex = index;
        notifyDataSetChanged();
    }

    /* other adapter's stuff */

    @Override
    public View getView(int position, View convertView, ViewGroup parent) {
        if(convertView == null) {
            convertView = itemInflater.inflate(R.layout.list_item_layout, parent, false);
        }

        // add this
        convertView.setActivated(selectedIndex != -1 && position == selectedIndex);

        /* do some stuff */

        return convertView;
    }
}

Наконец, мы должны вызвать метод адаптера setSelectedIndex(position) в методе onItemClick(...) из AdapterView.OnItemClickListener.

public class YourActivity extends Activity
        implements AdapterView.OnItemClickListener {

    private CustomAdapter mCustomAdapter;

    /* activity implementation */

    @Override
    public void onItemClick(AdapterView<?> parent, View view, int position, long id) {
        mCustomAdapter.setSelectedIndex(position);
    }
}

Теперь мы можем быть довольны правильным выделением элементов списка :)

Постскриптум Если мы хотим включить режим множественного выбора в нашем списке, мы просто поместим следующую строку в наш класс активности, где хранится экземпляр listView:

listView.setChoiceMode(ListView.CHOICE_MODE_MULTIPLE);

Таким образом, мы получим правильное множественное выделение элементов.

- Надеюсь, это кому-нибудь поможет :)

0
Eugene Matiyuk

Существует простое полностью XML-решение, которое сработало для меня. Во-первых, определите XML-отрисовку с помощью кода селектора, в котором «нормальное» состояние будет соответствовать «выбранному несжатому» визуальному состоянию элемента списка, а state_pressed = true для Визуальное состояние "нажат" .. Пример файла "custom_item_selector.xml", напоминающего выделение Holo blue:

<?xml version="1.0" encoding="utf-8"?>
<selector xmlns:Android="http://schemas.Android.com/apk/res/Android">
    <item Android:state_pressed="true">
        <shape Android:shape="rectangle">
            <solid
                Android:color="#643292ff">
            </solid>
            <stroke
                Android:width="1dp"
                Android:color="#c83292ff">
            </stroke>
        </shape>
    </item>
    <item>
        <shape Android:shape="rectangle">
            <solid
                Android:color="#323292ff">
            </solid>
            <stroke
                Android:width="1dp"
                Android:color="#783292ff">
            </stroke>
        </shape>
    </item>
</selector>

(может также установить фокусное состояние там) . Во-вторых, примените этот xml-drawable в качестве ListSelector ListView и установите его нужный choiceMode:

<ListView
            Android:layout_width="wrap_content"
            Android:layout_height="wrap_content"
            Android:id="@+id/listView"
            Android:choiceMode="singleChoice"
            Android:listSelector="@drawable/custom_item_selector"/>

Это все. Это позволяет определять различные визуальные состояния для «просто выбранных» и «нажато выбранных» элементов, например, делая элементы ярче при печати.

0
jetc

Чтобы держать элементы списка (множественный выбор) выделенными, при щелчке (активации) следуйте инструкциям.

1. Установите фон для макета элемента списка как нарисованный.

    <?xml version="1.0" encoding="utf-8"?>
    <LinearLayout xmlns:Android="http://schemas.Android.com/apk/res/Android"
        Android:layout_width="match_parent"
        Android:layout_height="40dp"
        Android:background="@drawable/list_item_selector">

        <ImageView
            Android:id="@+id/icon"
            Android:layout_width="22px"
            Android:layout_height="22px"
            Android:layout_marginLeft="4px"
            Android:layout_marginRight="10px"
            Android:layout_marginTop="4px"
            Android:src="@mipmap/ic_launcher" >
        </ImageView>

        <TextView
            Android:id="@+id/label"
            Android:layout_width="wrap_content"
            Android:layout_height="wrap_content"
            Android:text="@+id/label"
            Android:textSize="20px" >
        </TextView>
    </LinearLayout>

2. селектор рисования

<?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_red_light" />

<item Android:state_activated="true" Android:drawable="@Android:color/holo_orange_dark" />

</selector>

3. Список просмотра установить режим множественного выбора

. GetListView () setChoiceMode (ListView.CHOICE_MODE_MULTIPLE);

При нажатии:  enter image description here

На рисунке ниже показано, когда пользователь выбрал несколько элементов списка.

При активации:  enter image description here

0
Annada

Если вы можете использовать drawable для отображения listItem Highlighted, вы должны использовать следующий код: -

listView.setSelector(R.drawable.bg_image);

Оно работает.

0
Suresh Sharma