Отфильтрованный ListView не обновлен - PullRequest
2 голосов
/ 24 марта 2010

У меня есть ListView с пользовательским адаптером, который расширяет ArrayAdapter. Это ArrayAdapter Type Artist.

Artist - очень маленький класс, у которого есть имя и идентификатор. Класс Artist имеет функцию toString (), которая возвращает только имя.

У меня есть EditText. EditText имеет TextChangeListener, где я вызываю .getFilter (). Filter (chars, callback) на моем адаптере.

В обратном вызове Filter.Filterlistener (). OnComplete () я печатаю счетчик, и это выглядит действительно хорошо. Когда я печатаю, количество уменьшается. Таким образом, все выглядит так, как рекламируется, но список остается прежним. Я попытался вызвать artistAdapter.notifyDataSetChanged (), чтобы перерисовать список, но ничего не происходит. [см. 2.)]

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

Спасибо!

Вот что я сделал:

1.) Определяли ListView и EditText следующим образом:

<EditText xmlns:android="http://schemas.android.com/apk/res/android"
    android:id="@+id/list_search_text"
    android:layout_width="fill_parent"
    android:layout_height="35dip"
    android:layout_below="@id/header">
</EditText>       
<ListView xmlns:android="http://schemas.android.com/apk/res/android"
    android:id="@+id/list_search"
    android:layout_width="fill_parent"
    android:layout_height="fill_parent">
</ListView>

2.) Настройте мой ListView в Activity onCreate ():

private ListView listView = null;
private ArtistAdapter artistAdapter = null;

@Override
protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.search_artists);

    artistAdapter = new ArtistAdapter(this, R.layout.row, list); // 'list' is an ArrayList<Artist>

    listView = (ListView) findViewById(R.id.list_search);
    listView.setAdapter(artistAdapter);
    listView.setFastScrollEnabled(true);
    listView.setTextFilterEnabled(true);

    listView.setOnItemClickListener(new OnItemClickListener() {
        public void onItemClick(AdapterView<?> av, View v, int position, long id) {
            // do something
        }
    });

    EditText txtSearch = (EditText) findViewById(R.id.list_search_text);
    txtSearch.addTextChangedListener(new TextWatcher() {
        public void afterTextChanged(Editable arg0) { }

        public void beforeTextChanged(CharSequence arg0, int arg1, int arg2, int arg3) { }

        public void onTextChanged(CharSequence chars, int start, int before, int count) {
            artistAdapter.getFilter().filter(chars, new Filter.FilterListener() {
                public void onFilterComplete(int count) {
                    Log.d(Config.LOG_TAG, "filter complete! count: " + count);
                    artistAdapter.notifyDataSetChanged();
                }
            });
        }
    });
}

3.) Короче, это мой ArtistAdapter. Я добавил метод удаления () и добавления ():

public class ArtistAdapter extends ArrayAdapter<Artist> implements SectionIndexer {
    private List<Artist> items;

    /* other stuff like overridden getView, getPositionForSection, getSectionForPosition and so on */

    @Override
    public void remove(Artist object) {
        super.remove(object);
        items.remove(object);
    }

    @Override
    public void add(Artist object) {
        super.add(object);
        items.add(object);
    }
}    

4.) Мой художник также переопределил toString ():

public class Artist implements Comparable<Artist> {
    public String   uid;
    public String   name;

    public Artist(String id, String name) {
        this.uid = id;
        this.name = name;
    }

    public int compareTo(Artist another) {
        return this.name.compareToIgnoreCase(another.name);
    }

    @Override
    public String toString() {
        return this.name;
    }
}

Ответы [ 3 ]

4 голосов
/ 30 марта 2010

Я нашел решение. Я переписал весь код с нуля и нашел проблему.

В моем собственном ArrayAdapter, который называется ArtistAdapter (см. 3. вопрос), если есть переменная для хранения элементов: приватные элементы списка;

И в моем переопределенном getView (), когда я хочу получить текущий элемент, я сделал: items.getItem (положение) теперь я делаю: getItems (position) (таким образом, элемент извлекается из хранилища базового класса, а не из моего собственного), и это, кажется, делает свое дело!

Это мой getView (), как сейчас (версия, которая работает):

@Override
public View getView(int position, View convertView, ViewGroup parent) {
    LinearLayout artistView;
    Artist art = getItem(position);

    if (convertView == null) {
        artistView = new LinearLayout(getContext());
        String inflater = Context.LAYOUT_INFLATER_SERVICE;
        LayoutInflater vi;
        vi = (LayoutInflater)getContext().getSystemService(inflater);
        vi.inflate(resId, artistView, true);
    } else {
        artistView = (LinearLayout) convertView;
    }
    TextView nameView = (TextView)artistView.findViewById(R.id.txt_name);
    TextView uidView = (TextView)artistView.findViewById(R.id.txt_uid);
    nameView.setText(art.name);
    uidView.setText(art.uid);
    return artistView;
}

Пфу, это была действительно раздражающая ошибка ...

1 голос
/ 26 ноября 2010

Необходимо переопределить метод toString(), чтобы вернуть текст, который вы хотите отфильтровать.

1 голос
/ 24 марта 2010

Попробуйте использовать listView.setFilterText () вместо фильтра адаптера.

...