Android: доступ к дочерним представлениям из ListView

108

Мне нужно узнать позицию в пикселях одного элемента в списке, отображаемом с помощью ListView. Кажется, я должен получить один из TextView, а затем использовать getTop(), но я не могу понять, как получить дочернее представление файла ListView.

Обновление: дочерние ViewGroupэлементы не соответствуют 1 к 1 элементам в списке для файла ListView. Вместо этого ViewGroupдочерние элементы соответствуют только тем представлениям, которые видны прямо сейчас. So getChildAt()работает с индексом, который является внутренним для класса ViewGroupи не обязательно имеет какое-либо отношение к позиции в списке, который ListViewиспользует.

лакировать
источник

Ответы:

215

См .: Android ListView: получить индекс данных видимого элемента и объединить его с частью ответа Feet выше, может дать вам что-то вроде:

int wantedPosition = 10; // Whatever position you're looking for
int firstPosition = listView.getFirstVisiblePosition() - listView.getHeaderViewsCount(); // This is the same as child #0
int wantedChild = wantedPosition - firstPosition;
// Say, first visible position is 8, you want position 10, wantedChild will now be 2
// So that means your view is child #2 in the ViewGroup:
if (wantedChild < 0 || wantedChild >= listView.getChildCount()) {
  Log.w(TAG, "Unable to get view for desired position, because it's not being displayed on screen.");
  return;
}
// Could also check if wantedPosition is between listView.getFirstVisiblePosition() and listView.getLastVisiblePosition() instead.
View wantedView = listView.getChildAt(wantedChild);

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

Джо
источник
10
Отличный ответ, но не совсем работает, когда в вашем списке есть заголовки. Задача firstPositionдолжна состоять в том, int firstPosition = listView.getFirstVisiblePosition() - listView.getHeaderViewsCount();чтобы это исправить.
pospi
@pospi: спасибо, хороший момент! Я обновил свой ответ, чтобы учесть это.
Джо
Хороший ответ и работает в большинстве случаев, но я не понимаю, почему вы считаете, что дочерние представления появляются в том порядке, в котором они находятся в массиве дочерних представлений. Поскольку представления можно использовать повторно, как мы можем быть уверены, что первое представление не представляет собой последнее видимое представление?
Виктор Денисов
2
@VictorDenisov: это может произойти только в потоке пользовательского интерфейса; поэтому поток пользовательского интерфейса будет заблокирован во время выполнения этого кода, поэтому дочерние представления являются стационарными и не будут изменены. ListViewуже обрабатывает "перемещение" дочерних представлений после повторного использования старых convertViews и т. д., так что вы можете быть уверены, что ListView.getChildAt(0) это фактически первое прикрепленное представление из адаптера. Это может быть не полностью видно (и может даже не быть видимым вообще, в зависимости от ListViewпорогового значения «видимости» до повторного использования представления, которое, по его мнению, «прокручивается вне поля зрения»)
Джо
1
@Matheus, Android ListView работает иначе. Если элемент списка находится за пределами экрана, его представление уже было переработано и повторно использовано для одного из других элементов списка, отображаемых на экране. Вам следует переосмыслить, как вы используете ListView, если это необходимо.
Джо
17

Этот код проще использовать:

 View rowView = listView.getChildAt(viewIndex);//The item number in the List View
    if(rowView != null)
        {
           // Your code here
        }
Калима
источник
9
работает не всегда. getChildAt отсчитывает от первой видимой строки, а не от начала данных.
SteelBytes
13
Аргумент ViewID сбивает с толку. Это индекс (или позиция). ID представления - это полностью произвольное целое число, сгенерированное инструментом aapt.
Йохан Пелгрим
6

Быстрый поиск в документации для класса ListView обнаружил методы getChildCount () и getChildAt (), унаследованные от ViewGroup. Можете ли вы перебрать их, используя их? Я не уверен, но попробовать стоит.

Нашел здесь

Ноги
источник
1
Я пробовал это для выбора. Обычно это работает, но иногда возникают отдельные ошибки, и это ненадежно. Я бы не рекомендовал это.
Timmmm
Спасибо, Тимммм, я на самом деле не пробовал, просто нашел в документации.
Feet
5
listview.setOnItemClickListener(new AdapterView.OnItemClickListener() {
    @Override
    public void onItemClick(AdapterView<?> parent, final View view, int position, long id) {
        View v;
        int count = parent.getChildCount();
        v = parent.getChildAt(position);
        parent.requestChildFocus(v, view);
        v.setBackground(res.getDrawable(R.drawable.transparent_button));
        for (int i = 0; i < count; i++) {
            if (i != position) {
                v = parent.getChildAt(i);
                v.setBackground(res.getDrawable(R.drawable.not_clicked));
            }
        }
    }
});

По сути, создайте два Drawable - один прозрачный, а другой желаемого цвета. Запросите фокус в выбранной позиции ( int positionкак определено) и измените цвет указанной строки. Затем пройдите по родительскому элементу ListViewи соответствующим образом измените все остальные строки. Это учитывает listviewслучаи, когда пользователь нажимает кнопку несколько раз. Это делается с помощью индивидуального макета для каждой строки в ListView. (Очень просто, просто создайте новый файл макета с TextView- не устанавливайте фокусируемый или кликабельный!).

Специальный адаптер не требуется - используйте ArrayAdapter

Wes
источник
getChildAt относится к текущим прокручиваемым элементам. Это перестанет работать, если вы немного прокрутите свой список.
nurettin
4
int position = 0;
listview.setItemChecked(position, true);
View wantedView = adapter.getView(position, null, listview);
Milaaaad
источник
1
Немного
уточнить
-9

Это предполагает, что вы знаете позицию элемента в ListView:

  View element = listView.getListAdapter().getView(position, null, null);

Затем у вас должна быть возможность вызывать getLeft () и getTop (), чтобы определять положение элементов на экране.

Джейсонхаджинс
источник
43
getView()вызывается изнутри ListView при заполнении списка. Вы не должны использовать его для получения представления в этой позиции в списке, поскольку вызов getView()с нулевым значением convertViewзаставляет адаптер раздувать новое представление из ресурса макета адаптера (не получает представление, которое уже отображается).
Джо
1
Равная позиция будет позицией на визуальном экране, а не позицией источника данных адаптера.
Vikas
@jasonhudgins Это метод, который я использовал, потому что я подумал, что он лучший, но когда я пытаюсь сделать индикатор выполнения в дочернем представлении невидимым, но вместо этого он сделал все индикаторы выполнения в listvew невидимыми ........ все в весь принятый ответ
помог