Динамическое обновление адаптера 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
.