Начать новую активность из SearchView

У меня есть 2 действия: в первом есть панель действий с представлением поиска, во втором должны отображаться результаты поискового запроса.

androidmanifest:

    <activity
        android:name=".SearchActivity"

        ...
        android:launchMode="singleTop">           

        <meta-data
            android:name="android.app.searchable"
            android:resource="@xml/searchable" />
        ...

    </activity>

   <activity
        android:name=".ResultsActivity"
        ...

         <intent-filter>
            <action android:name="android.intent.action.SEARCH" />
        </intent-filter>

    </activity>

searchchable.xml

<searchable
xmlns:android="http://schemas.android.com/apk/res/android"
android:label="@string/app_name"
android:hint="@string/enter_a_word" />

SearchActivity

....
@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_noun_list, menu);

    // Associate searchable configuration with the SearchView
    SearchManager searchManager =  (SearchManager) getSystemService(Context.SEARCH_SERVICE);
    SearchView searchView =  (SearchView) menu.findItem(R.id.search).getActionView();
    searchView.setSearchableInfo( searchManager.getSearchableInfo(getComponentName()));

    return true;
}
....

Результаты

@Override
protected void onCreate(Bundle savedInstanceState) {
    ...
if (Intent.ACTION_SEARCH.equals(intent.getAction())) {
        String query = intent.getStringExtra(SearchManager.QUERY);

    }
...
}

проблема в том, что после ввода запроса в поиск ничего не происходит. Никаких ошибок, ничего. Как я могу открыть активность результатов после того, как запрос введен в функцию поиска?


person Maarten    schedule 09.04.2015    source источник
comment
Итак, я предполагаю, что у вас есть данные для поиска, верно? Вы также реализуете какие-либо интерфейсы в своей деятельности?   -  person Eenvincible    schedule 09.04.2015


Ответы (3)


Этот ответ немного запоздал, но я думаю, он будет полезен будущим зрителям. Дилемма, похоже, возникает из-за двусмысленности руководства Android SearchView. Сценарий, в котором они cover предполагает, что вы будете отображать результаты в том же действии, в котором находится SearchView. В таком сценарии тег Activity в файле AndroidManifest.xml будет выглядеть примерно так:

<activity
    android:name=".MainActivity"
    android:label="@string/main_activity_label"
    android:launchMode="singleTop">
        <intent-filter>
            <action android:name="android.intent.action.SEARCH"/>
        </intent-filter>
        <meta-data android:name="android.app.searchable"
            android:resource="@xml/searchable" />
</activity>

Затем, чтобы обработать результаты в том же действии, вы должны переопределить метод onNewIntent:

@Override
public void onNewIntent(Intent intent){
    setIntent(intent);
    if(Intent.ACTION_SEARCH.equals(intent.getAction())) {
        String query = intent.getStringExtra(SearchManager.QUERY);
        //now you can display the results
    }  
}

Однако в ситуации, когда мы хотим отобразить результаты в другом действии, мы должны поместить фильтр намерений и метатег в действие результатов и ввести новый метатег для действия SearchView. Итак, наши действия в AndroidManifest.xml файле будут выглядеть примерно так:

<activity
        android:name=".MainActivity"
        android:label="@string/main_activity_label"
        android:launchMode="singleTop">
        <!-- meta tag points to the activity which displays the results -->
        <meta-data
            android:name="android.app.default_searchable"
            android:value=".SearchResultsActivity" />
</activity>
<activity
        android:name=".SearchResultsActivity"
        android:label="@string/results_activity_label"
        android:parentActivityName="com.example.MainActivity">
        <!-- Parent activity meta-data to support 4.0 and lower -->
        <meta-data
            android:name="android.support.PARENT_ACTIVITY"
            android:value="com.example.MainActivity" />
        <!-- meta tag and intent filter go into results activity -->
        <meta-data android:name="android.app.searchable"
            android:resource="@xml/searchable" />
        <intent-filter>
            <action android:name="android.intent.action.SEARCH" />
        </intent-filter>
</activity>

Затем в методе onCreateOptionsMenu нашего MainActivity активируйте SearchView (предполагается, что вы добавляете SearchView в ActionBar). Вместо того, чтобы использовать getComponentName() в вызове метода getSearchableInfo () SearchManager, мы создаем экземпляр нового объекта ComponentName, используя контекст MainActivity и класс SearchResultsActivity:

@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_home, menu);
    SearchView search = (SearchView) MenuItemCompat.getActionView(menu.findItem(R.id.action_search);
    // Associate searchable configuration with the SearchView
    SearchManager searchManager = (SearchManager) getSystemService(Context.SEARCH_SERVICE);
    search.setSearchableInfo(searchManager.getSearchableInfo(new ComponentName(this, SearchResultsActivity.class)));
    search.setQueryHint(getResources().getString(R.string.search_hint));
    return true;
}

Наконец, в нашем классе SearchResultsActivity в методе onCreate мы можем обрабатывать результаты поиска:

@Override
public void onCreate(Bundle savedInstanceState){
    super.onCreate(savedInstanceState);
    if (Intent.ACTION_SEARCH.equals(intent.getAction())) {
        String query = intent.getStringExtra(SearchManager.QUERY);
        //use the query to search your data somehow
    }
}

Не забудьте создать searchable.xml файл ресурсов и добавить SearchView в свой макет.

searchchable.xml (res / xml / searchchable.xml; при необходимости создайте папку xml в res):

<?xml version="1.0" encoding="utf-8"?>
<searchable xmlns:android="http://schemas.android.com/apk/res/android"
    android:label="@string/app_name"
    android:hint="@string/search_hint"
    android:voiceSearchMode="showVoiceSearchButton|launchRecognizer"/>

Макет (пример добавления SearchView в ActionBar в качестве пункта меню):

<menu xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    xmlns:tools="http://schemas.android.com/tools"
    tools:context="com.example.MainActivity">
    <group android:checkableBehavior="single">
        <item android:id="@+id/action_search" android:title="Search"
            android:orderInCategory="1" app:showAsAction="collapseActionView|ifRoom"
            app:actionViewClass="android.support.v7.widget.SearchView"/>
    </group>
</menu>

Ресурсы:

person chRyNaN    schedule 19.08.2015
comment
Отличная демонстрация - person Raju; 16.09.2016
comment
Я следил за каждым шагом, но результат поиска по-прежнему соответствует текущей активности, SearchResultsActivity onCreate () не вызывается, у кого-нибудь есть такая же проблема? - person Weidian Huang; 04.11.2018
comment
Я обнаружил свою проблему, потому что я также слушаю onQueryTextSubmit () даже при текущей активности, и он возвращает true, результат не будет отправлен в SearchResultsActivity. Единственное, что мне нужно изменить, это просто вернуть false для onQueryTextSubmit (). - person Weidian Huang; 04.11.2018
comment
Очень хорошее объяснение. Спасибо. Мне это очень помогло :) - person Pooja; 09.10.2019
comment
зачем нам новое имя компонента? - person Michele Dorigatti; 13.08.2020

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

В вашей деятельности, связанной с результатами,

@Override
public boolean onCreateOptionsMenu(Menu menu) {
    MenuInflater inflater = getMenuInflater();
    inflater.inflate(R.menu.core_actions, menu);

    SearchManager searchManager = (SearchManager) getSystemService(Context.SEARCH_SERVICE);
    SearchView searchView = (SearchView) menu.findItem(R.id.action_search).getActionView();
    searchView.setSearchableInfo(searchManager.getSearchableInfo(getComponentName()));

    searchView.setIconifiedByDefault(false);
    searchView.setQueryHint(getString(R.string.search_hint));

    searchView.setOnQueryTextListener(this);

    return true;
}

Помните, что это действие, в котором есть ваши данные, которые вы хотите сделать доступными для поиска:

Вы должны реализовать интерфейс SearchView.OnQueryTextListener в том же действии:

@Override
public boolean onOptionsItemSelected(MenuItem item) {
    switch (item.getItemId()) {
        case R.id.action_search:
            ProductsResulstActivity.this.finish();

            return true;
        default:
            return super.onOptionsItemSelected(item);
    }
}

@Override
public boolean onQueryTextSubmit(String query) {
    return false;
}

@Override
public boolean onQueryTextChange(String newText) {

    productFilterAdapter.getFilter().filter(newText);

    if (TextUtils.isEmpty(newText)) {
         listView.clearTextFilter();
     }
    else {
       listView.setFilterText(newText);
    }

    return true;
}

productFilterAdapter - это адаптер, который необходимо создать заранее.

Он должен реализовывать фильтруемый интерфейс. Надеюсь, это поможет.

Если вам понадобится дополнительная помощь, дайте мне знать. Удачи

person Eenvincible    schedule 09.04.2015

Я понял проблему, с которой я тоже столкнулся. Это происходит потому, что вы передаете текущее имя компонента, передавая

getComponentName ()

Он будет инициализирован текущим именем действия, поэтому вам необходимо инициализировать его с доступным для поиска именем действия в указанном ниже формате и передать тот же экземпляр компонента, который запускает новое действие.

    SearchManager searchManager = (SearchManager) getSystemService(Context.SEARCH_SERVICE);
    search.setSearchableInfo(searchManager.getSearchableInfo(new ComponentName(this, SearchResultsActivity.class)));
    search.setQueryHint(getResources().getString(R.string.search_hint));

Надеюсь, я ответил на вопрос!

person Vivek Kumar Samele    schedule 10.07.2018