I\'m building the search in an application and need to have a way of putting the suggestions which I get from my server as a JSON-array into the list of suggestions which is
Don't know if people still need this. Just in case, for future searchers. I had a problem with my SearchView's suggestions, which did not show up. After searched around, I found a solution for this. I used Volley for my content provider class (to get the suggestions from web api), which seemed not to fit naturally into Android's content provider framework. I also found that my content provider DOES run in the UI thread. Therefore, if I used Volley's Future synchronously in it, it slowed down (blocked) the UI until the request returned (timeout). In addition, I found information around the web that Volley's Future request should be run in other thread, such as in an Async task, for it to work well. Thus, it didn't solve my problem, because if I'd have to use it (in an async task), i would have used Volley's normal (async) request in the first place instead (which was what I used then). What I did was this:
In my ContentProvider subclass, I define a listener interface:
public interface ResultListener {
void onProviderResult(Cursor mCursor);
void onProviderError(String errorMsg);
}
In my activity (which implemented LoaderCallbacks), I implemented also above interface.
In my singleton application class, I define a static variable which is used as transient data, together with its getter/setter methods:
private static HashMap transientData = new HashMap();
public static Object getTransientData(String objectName) {
return transientData.get(objectName);
}
public static void setTransientData(String objectName, Object object) {
transientData.put(objectName, object);
}
In my content provider's class, in the volley request's onResponse callback, I did this:
public void onResponse(JSONArray response){
...
ResultListener requestor = (ResultListener)TheApplication.getTransientData("requestor");
if (requestor!=null) requestor.onProviderResult(mCursor);
}
So that when the volley request returned, it will trigger the requestor's callback method, passing it the cursor filled with data from the response, and in turn the requestor (the activity) notified the cursor adapter by calling: adapter.swapCursor(c); adapter.notifyDataSetChanged();
Hope this helps someone. Be blessed.
Found the solution on developer.android.com:
If you have suggestions that you get from a network location, you can build a cursor on the fly when you get the results from your server.
This goes inside your ContentProvider's query() method:
String[] columns = {
BaseColumns._ID,
SearchManager.SUGGEST_COLUMN_TEXT_1,
SearchManager.SUGGEST_COLUMN_INTENT_DATA
};
MatrixCursor cursor = new MatrixCursor(columns);
for (int i = 0; i < arr.length(); i++)
{
String[] tmp = {Integer.toString(i), arr.getString(i), arr.getString(i)};
cursor.addRow(tmp);
}
return cursor;
The cursor is the used in the quick-search box to fill a list of suggestions.
Here is an example of SearchView with suggestions coming from a network service (I used Retrofit):
@Override
public boolean onCreateOptionsMenu(Menu menu) {
// Inflate the menu; this adds items to the action bar if it is present.
getMenuInflater().inflate(R.menu.menu_search_activity, menu);
final SearchView searchView = (SearchView) MenuItemCompat.getActionView(menu.findItem(R.id.search));
final CursorAdapter suggestionAdapter = new SimpleCursorAdapter(this,
android.R.layout.simple_list_item_1,
null,
new String[]{SearchManager.SUGGEST_COLUMN_TEXT_1},
new int[]{android.R.id.text1},
0);
final List<String> suggestions = new ArrayList<>();
searchView.setSuggestionsAdapter(suggestionAdapter);
searchView.setOnSuggestionListener(new SearchView.OnSuggestionListener() {
@Override
public boolean onSuggestionSelect(int position) {
return false;
}
@Override
public boolean onSuggestionClick(int position) {
searchView.setQuery(suggestions.get(position), false);
searchView.clearFocus();
doSearch(suggestions.get(position));
return true;
}
});
searchView.setOnQueryTextListener(new SearchView.OnQueryTextListener() {
@Override
public boolean onQueryTextSubmit(String query) {
return false;
}
@Override
public boolean onQueryTextChange(String newText) {
MyApp.autocompleteService.search(newText, new Callback<Autocomplete>() {
@Override
public void success(Autocomplete autocomplete, Response response) {
suggestions.clear();
suggestions.addAll(autocomplete.suggestions);
String[] columns = {
BaseColumns._ID,
SearchManager.SUGGEST_COLUMN_TEXT_1,
SearchManager.SUGGEST_COLUMN_INTENT_DATA
};
MatrixCursor cursor = new MatrixCursor(columns);
for (int i = 0; i < autocomplete.suggestions.size(); i++) {
String[] tmp = {Integer.toString(i), autocomplete.suggestions.get(i), autocomplete.suggestions.get(i)};
cursor.addRow(tmp);
}
suggestionAdapter.swapCursor(cursor);
}
@Override
public void failure(RetrofitError error) {
Toast.makeText(SearchFoodActivity.this, error.getMessage(), Toast.LENGTH_SHORT).show();
Log.w("autocompleteService", error.getMessage());
}
});
return false;
}
});
return true;
}