1

ArrayAdapter で動作する AutocompleteTextView があります。テキストが変更されると、アダプターは Web サービスから更新されます。この更新は AsyncTask で行われますが、これは良い習慣です。キーが押されるたびに提案が前に押されたキーで取得された文字列に基づいているため、これは多かれ少なかれ機能しています。このページに関連する問題がいくつかありますが、どれもうまくいきません。とにかく、解決策はありましたが、非効率的であり、「公式の解決策」が失敗する理由がわかりません。

バックグラウンドで ArrayAdapter を更新する機能に鍵があると思います。これは、Web サービスへの非同期呼び出しで行うことです。

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() を設定しましたが、役に立ちません。

4

2 に答える 2

2

I also tried alot to get this approach working, but didn't succeed any better than yourself. But I found another way to accomplish what you want; extend ArrayAdapter and implement Filterable. This class will be doing the actual fetching from the database, when called by the AutoCompleteTextView in a worker thread:

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;
    }
}

EDIT: I have improved the above code since I sometimes got the exception java.lang.IllegalStateException: The content of the adapter has changed but ListView did not receive a notification. To fix this, I moved the updating of mData to publishResults().

于 2011-08-30T15:48:33.060 に答える
0

このコード スニペットは、私のアプリでうまくいきました。これを含むアクティビティに入れますListView

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

コールバックは、トリガーされるイベント リスナーになります。通常、 でAsyncTasks使用できonPostExecute()ますAsyncTask

于 2011-08-30T19:40:58.937 に答える