ArrayAdapter обновляется поздно из Webservice в AutoCompleteTextAdapter - PullRequest
1 голос
/ 01 июля 2011

У меня есть AutocompleteTextView, работающий с ArrayAdapter.Адаптер обновляется из веб-службы, когда изменяется текст.Это обновление сделано в AsyncTask, что должно быть хорошей практикой.Это работает более или менее, потому что предложения после каждой нажатой клавиши основаны на строках, найденных в предыдущей нажатой клавише.Есть несколько проблем, связанных с этой страницей, но ни один из ответов не работает для меня.В любом случае, у меня было решение, но оно неэффективно, и я не знаю, почему «официальное решение» не работает.

Я думаю, что ключ находится в функции, которая обновляет de ArrayAdapter в фоновом режиме.Это то, что я делаю в асинхронном вызове веб-сервисов:

private class DoAutoCompleteSearch extends AsyncTask<String, Void, Map<String, String>> {

    @Override
    protected Map<String, String> doInBackground(String... params) {

        // Ask the webservice for data
        Map<String, String> autoComplete = GetResource.dataList(params[0]);
        return autoComplete;
    }

    @Override
    protected void onPostExecute(Map<String, String> result) {

        //mAutoCompleteAdapter.clear();    * This should work but does not *

/* If it is set a new adapter in the AutoCompleteTextView, the whole thing works properly */

    mAutoCompleteAdapter = new ArrayAdapter<String>(mAutoCompleteAdapter.getContext(), android.R.layout.simple_dropdown_item_1line);
    mACTV.setAdapter(mAutoCompleteAdapter);

        for (Map.Entry<String, String> entry : result.entrySet()) {
            mAutoCompleteAdapter.add(entry.getKey());
        }
    }
}

Я пробовал с mAutoCompleteAdapter.clear () и установкой mAutoCompleteAdapter.notifyDataSetChanged () везде, но это бесполезно.

Ответы [ 2 ]

2 голосов
/ 30 августа 2011

Я также много пытался заставить этот подход работать, но не преуспел лучше, чем вы. Но я нашел другой способ выполнить то, что вы хотите; расширить ArrayAdapter и реализовать Filterable. Этот класс будет выполнять фактическую выборку из базы данных при вызове AutoCompleteTextView в рабочем потоке:

public class MyAutoCompleteAdapter extends ArrayAdapter<String> implements Filterable {

    private List<String> mData = new ArrayList<String>();
    private Server mServer;

    public MyAutoCompleteAdapter(Server server, Context context, int textViewResourceId) {
        super(context, textViewResourceId);
        mServer = server;
    }

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

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

    @Override
    public Filter getFilter() {
        Filter myFilter = new Filter() {
            @Override
            protected FilterResults performFiltering(CharSequence constraint) {
                // This method is called in a worker thread

                FilterResults filterResults = new FilterResults();
                if(constraint != null) {
                    try {
                        // Here is the method (synchronous) that fetches the data
                        // from the server
                        List<String> results = mServer.searchForItems(constraint.toString());
                        filterResults.values = results;
                        filterResults.count = results.size();
                    }
                    catch(Exception e) {}

                }
                return filterResults;
            }

            @Override
            protected void publishResults(CharSequence contraint, FilterResults results) {
                if(results != null && results.count > 0) {
                    mData = (List<String>)results.values;
                    notifyDataSetChanged();
                }
                else {
                    notifyDataSetInvalidated();
                }
            }
        };
        return myFilter;
    }
}

РЕДАКТИРОВАТЬ : я улучшил приведенный выше код, поскольку иногда получал исключение java.lang.IllegalStateException: The content of the adapter has changed but ListView did not receive a notification. Чтобы это исправить, я переместил обновление mData на publishResults().

0 голосов
/ 30 августа 2011

Этот фрагмент кода работал для меня в моем приложении. Поместите это в свою деятельность, которая содержит ListView:

@Override
public void onActivityResumed() //
{
    if (favoritesHaveChanged(activity, productString)) //
    {
        m_adapter.clear();
        performAsyncTaskWithCallBack(activity);
    }
}

Обратным вызовом будет прослушиватель событий, который запускается. Обычно с AsyncTasks вы можете использовать onPostExecute() в вашем AsyncTask.

...