从SearchView启动新的活动

17

我有两个活动:第一个活动有一个带搜索视图的操作栏,第二个活动应该显示搜索查询的结果。

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>

可搜索的.xml

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

搜索活动

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

    }
...
}

问题在于在搜索栏中输入查询后,没有任何反应。没有错误,也没有任何显示。如何在查询后在结果页面(resultsactivity)中打开搜索页面(searchactivity)?


所以我假设你有要搜索的数据,对吗?你在活动中还实现了任何接口吗? - Eenvincible
3个回答

40
这个答案有点晚,但我认为它对未来的观众会有用。这个困境似乎来自于Android SearchView教程的歧义。他们涵盖的场景假定您将在SearchView所在的同一活动中显示结果。在这种情况下,AndroidManifest.xml文件中的Activity标签应该像这样:
<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>

然后,为了在同一个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
    }  
}

然而,在我们想要在另一个Activity中展示结果的情况下,我们必须将Intent Filter和meta标签放入结果Activity中,并且为SearchView Activity引入一个新的meta标签。因此,我们的Activities在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>

接着,在MainActivity的onCreateOptionsMenu方法中,激活SearchView(假设您正在将SearchView添加到ActionBar)。与在SearchManager的getSearchableInfo()方法调用中使用getComponentName()不同,我们使用MainActivity的上下文和SearchResultsActivity类实例化一个新的ComponentName对象:

@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。

searchable.xml(res / xml / searchable.xml; 如有需要,请在res下创建xml文件夹):

<?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"/>

布局(在ActionBar中添加SearchView作为菜单项的示例):

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

资源:


2
优秀的演示 - Raju
2
我按照每个步骤进行操作,但搜索结果仍然指向当前活动,SearchResultsActivity onCreate() 方法没有被调用,是否有人遇到过同样的问题? - Weidian Huang
我找到了我的问题,因为我在当前活动中也监听了onQueryTextSubmit(),并且它返回true,结果不会提交到SearchResultsActivity。我需要改变的唯一一件事就是在onQueryTextSubmit()中返回false。 - Weidian Huang
非常好的解释。谢谢你。这对我真的很有帮助 :) - Pooja
为什么我们需要一个新的组件名称? - Michele Dorigatti

1
我理解了我曾经也遇到过同样的问题,这是因为你正在通过传递

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

希望我已经回答了问题!

0

在没有看到你的活动代码之前,我建议你尝试这种方法 - 同时假设你已经按照上述说明创建了所有文件;

在你的结果活动中,

@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 是您必须事先创建的适配器。

它应该实现 Filterable 接口。希望这可以帮到您。

如果您需要进一步的帮助,请告诉我。祝您好运。


网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接