Android: предоставление последних предложений по поиску без возможности поиска?
У меня есть ActionBar SearchView, и я успешно могу выполнить поиск с ним. Документация по Android не объясняет, как реализовать поисковые предложения. Я не хочу иметь возможность поиска.
Это мой поисковый код:
public boolean onCreateOptionsMenu(Menu menu) {
getMenuInflater().inflate(R.menu.activity_add_song, menu);
final SearchView searchView = (SearchView) menu.findItem(R.id.song_search).getActionView();
searchView.setFocusable(true);
searchView.setIconified(false);
final AddSongActivity activity = this;
searchView.setOnQueryTextListener(new SearchView.OnQueryTextListener() {
@Override
public boolean onQueryTextChange(String newText) {
// Do nothing
return true;
}
@Override
public boolean onQueryTextSubmit(String query) {
// Clear SearchView
searchView.clearFocus();
// Begin Spotify Search
TextView notice = (TextView)findViewById(R.id.search_notice);
URL url;
try {
url = new URL("http://ws.spotify.com/search/1/track.json?q=" + URLEncoder.encode(query,"UTF-8"));
} catch (MalformedURLException e) {
notice.setText("Malformed Search");
notice.setHeight(noticeHeight);
return true;
} catch (UnsupportedEncodingException e) {
notice.setText("Unsupported Encoding. Maybe a problem with your device.");
notice.setHeight(noticeHeight);
return true;
}
new SearchDownload(url, activity).execute();
notice.setText("Loading Tracks");
notice.setHeight(noticeHeight);
Log.i("infodb","" + noticeHeight);
return true;
}
});
Это работает для поиска, но я не знаю, как реализовать последние предложения по поисковым запросам. Как мне это сделать?
Спасибо.
Ответы
Ответ 1
Хорошо, я потратил свое время на это. Я делаю свою собственную простую реализацию предложения от SQLiteDatabase
.
Мы создадим 3 класса, например следующие
-
MainActivity
- для тестирования предложения SearchView
из базы данных
-
SuggestionDatabase
- это будет хранить ваше последнее ключевое слово поиска.
-
SuggestionSimpleCursorAdapter
- это подкласс SimpleCursorAdapter
. Я объясню, почему я делаю этот класс вместо использования SimpleCursorAdapter
.
Коды
// MainActivity.java
public class MainActivity
extends Activity
implements SearchView.OnQueryTextListener,
SearchView.OnSuggestionListener
{
private SuggestionsDatabase database;
private SearchView searchView;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
database = new SuggestionsDatabase(this);
searchView = (SearchView) findViewById(R.id.searchView1);
searchView.setOnQueryTextListener(this);
searchView.setOnSuggestionListener(this);
}
@Override
public boolean onSuggestionSelect(int position) {
return false;
}
@Override
public boolean onSuggestionClick(int position) {
SQLiteCursor cursor = (SQLiteCursor) searchView.getSuggestionsAdapter().getItem(position);
int indexColumnSuggestion = cursor.getColumnIndex( SuggestionsDatabase.FIELD_SUGGESTION);
searchView.setQuery(cursor.getString(indexColumnSuggestion), false);
return true;
}
@Override
public boolean onQueryTextSubmit(String query) {
long result = database.insertSuggestion(query);
return result != -1;
}
@Override
public boolean onQueryTextChange(String newText) {
Cursor cursor = database.getSuggestions(newText);
if(cursor.getCount() != 0)
{
String[] columns = new String[] {SuggestionsDatabase.FIELD_SUGGESTION };
int[] columnTextId = new int[] { android.R.id.text1};
SuggestionSimpleCursorAdapter simple = new SuggestionSimpleCursorAdapter(getBaseContext(),
android.R.layout.simple_list_item_1, cursor,
columns , columnTextId
, 0);
searchView.setSuggestionsAdapter(simple);
return true;
}
else
{
return false;
}
}
}
Как это работает
- Когда пользователь удаляет кнопку поиска, будет запущен
onQueryTextSubmit()
, а затем ключевое слово поиска будет сохранено в нашей базе данных. Скажем, мы отправляем ключевое слово "Hello"
- Если пользователь пишет строку, например "Hel" или "H" в
SearchView
, вызывается onQueryTextChange()
, а затем мы ищем это ключевое слово в SQLiteDatabase
(SuggestionDatabase
). Если "Hel" или "H" соответствуют "Hello" , отобразите результаты запроса, установив возвращаемый курсор в SuggestionSimpleCursorAdapter
, а затем установите этот адаптер в SearchView
. Вот картина.
![enter image description here]()
3. Конечно, мы будем нажимать предложение "Hello" , для этого будет вызвано onSuggestionClick(int position)
. Мы получаем объект SQLiteCursor
из адаптера SearchView
(SuggestionSimpleCursorAdapter
) и получаем текст предложения, устанавливаем текст предложения в SearchView
object
SQLiteCursor cursor = (SQLiteCursor) searchView.getSuggestionsAdapter().getItem(position);
int indexColumnSuggestion = cursor.getColumnIndex( SuggestionsDatabase.FIELD_SUGGESTION);
searchView.setQuery(cursor.getString(indexColumnSuggestion), false);
Если мы используем SimpleCursorAdapter
, он также работает правильно, но предположим, что у нас есть этот сценарий
- Если мы запустим эту программу на смартфоне и наберем ключевое слово "Hel" , предложение появится правильно.
![enter image description here]()
- Что делать, если мы вращаем экран в ландшафте? Он переключится в полноэкранный режим, и вы все равно можете ввести ключевое слово.
Что будет в предложении? Давайте посмотрим.
![enter image description here]()
Посмотрите на странное предложение? Как мы это решаем? Переопределяя convertToString(Cursor cursor)
, который возвращает a CharSequence
// SuggestionSimpleCursorAdapter.java
public class SuggestionSimpleCursorAdapter
extends SimpleCursorAdapter
{
public SuggestionSimpleCursorAdapter(Context context, int layout, Cursor c,
String[] from, int[] to) {
super(context, layout, c, from, to);
}
public SuggestionSimpleCursorAdapter(Context context, int layout, Cursor c,
String[] from, int[] to, int flags) {
super(context, layout, c, from, to, flags);
}
@Override
public CharSequence convertToString(Cursor cursor) {
int indexColumnSuggestion = cursor.getColumnIndex(SuggestionsDatabase.FIELD_SUGGESTION);
return cursor.getString(indexColumnSuggestion);
}
}
Переопределяя convertToString(Cursor cursor)
, здесь результат
![enter image description here]()
И вот база данных
// SuggestionDatabase.java
public class SuggestionsDatabase {
public static final String DB_SUGGESTION = "SUGGESTION_DB";
public final static String TABLE_SUGGESTION = "SUGGESTION_TB";
public final static String FIELD_ID = "_id";
public final static String FIELD_SUGGESTION = "suggestion";
private SQLiteDatabase db;
private Helper helper;
public SuggestionsDatabase(Context context) {
helper = new Helper(context, DB_SUGGESTION, null, 1);
db = helper.getWritableDatabase();
}
public long insertSuggestion(String text)
{
ContentValues values = new ContentValues();
values.put(FIELD_SUGGESTION, text);
return db.insert(TABLE_SUGGESTION, null, values);
}
public Cursor getSuggestions(String text)
{
return db.query(TABLE_SUGGESTION, new String[] {FIELD_ID, FIELD_SUGGESTION},
FIELD_SUGGESTION+" LIKE '"+ text +"%'", null, null, null, null);
}
private class Helper extends SQLiteOpenHelper
{
public Helper(Context context, String name, CursorFactory factory,
int version) {
super(context, name, factory, version);
}
@Override
public void onCreate(SQLiteDatabase db) {
db.execSQL("CREATE TABLE "+TABLE_SUGGESTION+" ("+
FIELD_ID+" integer primary key autoincrement, "+FIELD_SUGGESTION+" text);");
Log.d("SUGGESTION", "DB CREATED");
}
@Override
public void onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion) {
}
}
}
Я надеюсь, что этот ответ полезен другим программистам.:)
Ответ 2
В классе MainActivity
сверху в onCreate method
используйте этот код сначала
AutoCompleteTextView search_text = (AutoCompleteTextView) searchView.findViewById(searchView.getContext().getResources().getIdentifier("android:id/search_src_text", null, null));
search_text.setThreshold(1);
Этот setThreshold(1)
означает, что теперь он может искать текст от одного символа.
Ответ 3
Моя потребность была еще более упрощенной: мне не нужна база данных, так как у меня были предложения, которые многие из них хотят показать в ArrayList.
Вот пример реализации:
import java.util.ArrayList;
import android.app.Activity;
import android.app.SearchManager;
import android.content.Context;
import android.database.Cursor;
import android.database.MatrixCursor;
import android.os.Bundle;
import android.view.LayoutInflater;
import android.view.Menu;
import android.view.MenuItem;
import android.view.View;
import android.view.View.OnClickListener;
import android.view.ViewGroup;
import android.widget.CursorAdapter;
import android.widget.SearchView;
import android.widget.SearchView.OnQueryTextListener;
import android.widget.TextView;
import android.widget.Toast;
public class ActivityTest extends Activity implements OnQueryTextListener {
private static final String COLUMN_ID = "_id";
private static final String COLUMN_TERM = "term";
private static final String DEFAULT = "default";
private SearchManager searchManager;
private SearchView searchView;
private MenuItem searchMenuItem;
private SuggestAdapter suggestionsAdapter;
private final ArrayList<String> suggestionsArray = new ArrayList<String>();
private final ArrayList<String> dummyArray = new ArrayList<String>();
@Override
protected void onCreate(final Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
// Create some dummy entries
dummyArray.add("apples");
dummyArray.add("oranges");
dummyArray.add("bananas");
dummyArray.add("pears");
dummyArray.add("plums");
}
@Override
public boolean onCreateOptionsMenu(final Menu menu) {
getMenuInflater().inflate(R.menu.main, menu);
searchManager = (SearchManager) getSystemService(Context.SEARCH_SERVICE);
searchMenuItem = menu.findItem(R.id.action_search);
searchView = (SearchView) searchMenuItem.getActionView();
searchView.setSearchableInfo(searchManager.getSearchableInfo(getComponentName()));
searchView.setOnQueryTextListener(this);
final MatrixCursor matrixCursor = getCursor(suggestionsArray);
suggestionsAdapter = new SuggestAdapter(this, matrixCursor, suggestionsArray);
searchView.setSuggestionsAdapter(suggestionsAdapter);
suggestionsAdapter.notifyDataSetChanged();
return true;
}
@Override
public boolean onQueryTextChange(final String newText) {
suggestionsArray.clear();
for (int i = 0; i < dummyArray.size(); i++) {
if (dummyArray.get(i).contains(newText)) {
suggestionsArray.add(dummyArray.get(i));
}
}
final MatrixCursor matrixCursor = getCursor(suggestionsArray);
suggestionsAdapter = new SuggestAdapter(this, matrixCursor, suggestionsArray);
searchView.setSuggestionsAdapter(suggestionsAdapter);
suggestionsAdapter.notifyDataSetChanged();
return true;
}
@Override
public boolean onQueryTextSubmit(final String query) {
// TODO Auto-generated method stub
return false;
}
private class SuggestAdapter extends CursorAdapter implements OnClickListener {
private final ArrayList<String> mObjects;
private final LayoutInflater mInflater;
private TextView tvSearchTerm;
public SuggestAdapter(final Context ctx, final Cursor cursor, final ArrayList<String> mObjects) {
super(ctx, cursor, 0);
this.mObjects = mObjects;
this.mInflater = (LayoutInflater) ctx.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
}
@Override
public View newView(final Context ctx, final Cursor cursor, final ViewGroup parent) {
final View view = mInflater.inflate(R.layout.list_item_search, parent, false);
tvSearchTerm = (TextView) view.findViewById(R.id.tvSearchTerm);
return view;
}
@Override
public void bindView(final View view, final Context ctx, final Cursor cursor) {
tvSearchTerm = (TextView) view.findViewById(R.id.tvSearchTerm);
final int position = cursor.getPosition();
if (cursorInBounds(position)) {
final String term = mObjects.get(position);
tvSearchTerm.setText(term);
view.setTag(position);
view.setOnClickListener(this);
} else {
// Something went wrong
}
}
private boolean cursorInBounds(final int position) {
return position < mObjects.size();
}
@Override
public void onClick(final View view) {
final int position = (Integer) view.getTag();
if (cursorInBounds(position)) {
final String selected = mObjects.get(position);
Toast.makeText(getApplicationContext(), selected, Toast.LENGTH_SHORT).show();
// Do something
} else {
// Something went wrong
}
}
}
private MatrixCursor getCursor(final ArrayList<String> suggestions) {
final String[] columns = new String[] { COLUMN_ID, COLUMN_TERM };
final Object[] object = new Object[] { 0, DEFAULT };
final MatrixCursor matrixCursor = new MatrixCursor(columns);
for (int i = 0; i < suggestions.size(); i++) {
object[0] = i;
object[1] = suggestions.get(i);
matrixCursor.addRow(object);
}
return matrixCursor;
}
}
В моем фактическом коде у меня есть пользовательский интерфейс, который заполняет ArrayList динамическими терминами, полученными с сервера. Вы обновили бы набор данных следующим образом:
@Override
public void onDataReceived(final ArrayList<String> results) {
suggestionsArray.clear();
suggestionsArray.addAll(results);
final MatrixCursor matrixCursor = getCursor(suggestionsArray);
suggestionsAdapter = new SuggestAdapter(this, matrixCursor, suggestionsArray);
searchView.setSuggestionsAdapter(suggestionsAdapter);
suggestionsAdapter.notifyDataSetChanged();
}
Я обнаружил, что не инициализация пустого курсора или его повторное создание каждый раз вызывала проблемы.
Надеюсь, что это поможет.
Ответ 4
Есть одна проблема, которую я заметил в вышеприведенном подходе.
Когда пользователь вводит только один символ (например, "H"
), после получения записей из БД и установки адаптера в searchView с помощью searchView.setSuggestionsAdapter(<adapter>)
выпадающий список не отображается.
Только после ввода второго символа (например, " ", "a"
) отображается список предложений.
Наблюдает ли кто-нибудь еще такое поведение?