Динамическое обновление адаптера AutoCompleteTextView

Я хочу периодически изменять предложения, данные Autocompletextview, получая список из веб-службы RESTful, и не могу заставить его работать гладко. Я настроил жестко закодированный список предложений, чтобы убедиться, что он работает:

ArrayAdapter<String> adapter = new ArrayAdapter<String>(this, R.layout.list_item, new String[] {"Hi", "Ho"});
speciesName.setAdapter(adapter);//my autocomplete tv

у меня есть TextWatcher в textview, и когда текст изменяется, который запускает неблокирующий вызов, чтобы получить новый список предложений-эта часть, которая получает новый список, работает нормально. Затем я хочу сбросить адаптер, например Итак:

public void setOptionsAndUpdate(String[] options) {
    Log.d(TAG, "setting options");
    //speciesName.setAdapter((ArrayAdapter<String>)null);
    ArrayAdapter<String> adapter = new ArrayAdapter<String>(this, R.layout.list_item, options);
    speciesName.setAdapter(adapter);
}

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

это даже правильный подход? Я посмотрел на SimpleCursorAdapter но не видел, как зарегистрировать мой веб-сервис в качестве поставщика контента. (Он имеет форму http://www.blah.com/query?term=XX, где XX-это вход из моего приложения, а ответ-массив строк JSON.)

5 ответов


вот как я обновляю свой AutoCompleteTextView:

String[] data = terms.toArray(new String[terms.size()]);  // terms is a List<String>
ArrayAdapter<?> adapter = new ArrayAdapter<Object>(activity, android.R.layout.simple_dropdown_item_1line, data);
keywordField.setAdapter(adapter);  // keywordField is a AutoCompleteTextView
if(terms.size() < 40) keywordField.setThreshold(1); 
else keywordField.setThreshold(2);

теперь, конечно, это статично и не имеет дело с предложениями по воздуху, но я также могу предложить вам уведомить адаптер об изменениях после того, как вы назначите его Autocompletextview:

adapter.notifyDataSetChanged();   

надеюсь, что это помогает.

-Серкан


мне не повезло с адаптером.notifyDataSetChanged() при динамическом добавлении и изменении данных в адаптере. В моей ситуации я асинхронно нажимал внешний api и периодически получал список данных завершения.

этот код очищает адаптер и добавляет новые данные, как и следовало ожидать. Однако, мне пришлось вызвать getFilter().Метод фильтра для принудительного отображения данных. Кроме того, мне пришлось явно фильтровать на основе текущего текста в AutocompleteTextView, потому что мой вызов api был асинхронным.

adapter.clear();
for (Map<String, String> map : completions) {
     adapter.add(map.get("name"));
}

//Force the adapter to filter itself, necessary to show new data.
//Filter based on the current text because api call is asynchronous. 
adapter.getFilter().filter(autocompleteTextView.getText(), null);

был довольно хороший учебник по этой теме, используя удаленные данные в Google Map API для заполнения AutoCompleteTextView здесь.

Если вам нужна кэшированная версия, я извлек ее из здесь. Исходный учебник был удален, но по существу вам нужно написать ArrayAdapter с пользовательским фильтром аналогичным образом, как показано ниже, и назначить его вашему AutoCompleteTextView.

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

private class PlacesAutoCompleteAdapter extends ArrayAdapter<String> implements Filterable {
private ArrayList<String> resultList;

public PlacesAutoCompleteAdapter(Context context, int textViewResourceId) {
    super(context, textViewResourceId);
}

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

@Override
public String getItem(int index) {
    return resultList.get(index);
}

@Override
public Filter getFilter() {
    Filter filter = new Filter() {
        @Override
        protected FilterResults performFiltering(CharSequence constraint) {
            FilterResults filterResults = new FilterResults();
            if (constraint != null) {
                // Retrieve the autocomplete results.
                resultList = autocomplete(constraint.toString());

                // Assign the data to the FilterResults
                filterResults.values = resultList;
                filterResults.count = resultList.size();
            }
            return filterResults;
        }

        @Override
        protected void publishResults(CharSequence constraint, FilterResults results) {
            if (results != null && results.count > 0) {
                notifyDataSetChanged();
            }
            else {
                notifyDataSetInvalidated();
            }
        }};
    return filter;
}
}

поскольку я не могу добавить комментарий, Я даю новый ответ Нет необходимости очищать адаптер или вызывать адаптер.getFilter().фильтр.(..)... Чтобы динамически обновлять адаптер AutoCompleteTextView, просто добавьте новый элемент в адаптер и снова установите адаптер. Для примера, приведенного в исходном вопросе, я попробовал следующее, И это работает (код ниже не показывает начальную настройку адаптера, так как здесь несколько ответов охватывают это. Это просто показывает обновление адаптер динамически). Обновление адаптера может быть рядом с кодом, который обновляет список, связанный с ArrayAdapter.

    adapter.add(String newSuggestion); // this goes inside a loop for adding multiple suggestions
    speciesName.setAdapter(adapter) ; // speciesName is an AutoCompleteTextView as given in the original question.

лучшее решение, которое я нашел для обновления адаптера:

Editable text = autocomplete.getText();
autocomplete.setText(text);
autocomplete.setSelection(text.length());

как работает:

устанавливаем текст autoCompleteTextView С его текущим текстом, поэтому адаптер уведомляет, что данные изменены и обновляет содержимое listViews.

но с помощью этого трюка курсор перемещается в начало edittext. поэтому мы используем autocomplete.setSelection(text.length()) для перемещения курсора к конец.

работает как шарм!

Edit:

также вы должны использовать clear(), add() и remove() методы непосредственно на вашем ArrayAdapter вместо ArrayList.