Android---浮动搜索框(SearchManager)

浮动搜索框的使用其实并不难,而是在于它的配置非常之繁琐,对于它的使用主要是方便开发者对于程序中有搜索业务时,更好的设计UI

SearchManager具体使用步骤如下:

(1)配置search bar的相关信息,新建一个位于res/xml下的一个searchable.xml的配置文件,如默认值、是否有搜索建议或者语音搜索。


代码


<searchable xmlns:android=http://schemas.android.com/apk/res/android

   <!-- label为搜索框上方的文本,hint搜索框里面的提示文本,显示label -->
android:label="@string/search_label"
android:hint="@string/search_hint"
android:searchMode="showSearchLabelAsBadge"

  <!-- 语音搜索配置 -->
android:voiceSearchMode="showVoiceSearchButton|launchRecognizer"
android:voiceLanguageModel="free_form"
android:voicePromptText="@string/search_invoke"

 

<!-- 配置搜索建议,配置错误将不会显示,这里的searchSuggestAuthority的值必须是

继承自SearchRecentSuggestionsProvider的完整路径名 -->

    android:searchSuggestAuthority="com.android.cbin.SearchSuggestionSampleProvider"
android:searchSuggestSelection=" ? "
/>

(2) manifest.xml配置,搜索结果处理的Activity将出现两种情况,一种是从其他Activity中的search bar打开一个Activtiy

专门处理搜索结果,第二种是就在当前Activity就是处理结果的Activity,先介绍第一种配置:


代码


<activity android:name="SearchResultActivity">

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

 
     

     <!-- 指定上面的searchable.xml文件 -->

    <meta-data android:resource="@xml/searchable"

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

</activity>

 

<!-- 为了使每一个Activity都能使用search bar,一定要将这个标签放到启动Activity中,里面的value指定
 的是前面的搜索结果Activity-->
 <meta-data android:name="android.app.default_searchable"
                       android:value=".SearchResultActivity" />

(3)搜索建议在manifest.xml中相关的配置


<!--之前searchable.xml中有一个searchSuggestAuthority的值其实和这里的

authorities指向的都是name中所关联的SearchSuggestionSampleProvider,他是一个

SearchRecentSuggestionsProvider的子类-->

 <provider android:name="SearchSuggestionSampleProvider"

  android:authorities="com.android.cbin.SearchSuggestionSampleProvider"></provider>

 

上面authorities指向的都是name中所关联的SearchSuggestionSampleProvider,他是一个

SearchRecentSuggestionsProvider的子类


代码


public class SearchSuggestionSampleProvider extends
SearchRecentSuggestionsProvider {

final static String AUTHORITY="com.android.cbin.SearchSuggestionSampleProvider";
final static int MODE=DATABASE_MODE_QUERIES;

public SearchSuggestionSampleProvider(){
super();
setupSuggestions(AUTHORITY, MODE);
}
}

 

 

(4)为了能够使用search bar 我们必须重写Activity的onSearchRequested的方法,在界面上启动一个search bar

但是这个动作不会自动触发,必须通过一个按钮或者菜单的点击事件触发;


代码


@Override
public boolean onSearchRequested(){

String text=etdata.getText().toString();
Bundle bundle=new Bundle();
bundle.putString("data", text);

//打开浮动搜索框(第一个参数默认添加到搜索框的值)
//bundle为传递的数据
startSearch("mm", false, bundle, false);
//这个地方一定要返回真 如果只是super.onSearchRequested方法不但

     //onSearchRequested(搜索框默认值)无法添加到搜索框中,bundle也无法传递出去
return true;
}

(5)接收query和bundle、保存query值(即搜索建议的列表值)


代码


public void doSearchQuery(){
final Intent intent = getIntent();
//获得搜索框里值
String query=intent.getStringExtra(SearchManager.QUERY);
tvquery.setText(query);
//保存搜索记录
SearchRecentSuggestions suggestions=new SearchRecentSuggestions(this,
SearchSuggestionSampleProvider.AUTHORITY, SearchSuggestionSampleProvider.MODE);
suggestions.saveRecentQuery(query, null);
if(Intent.ACTION_SEARCH.equals(intent.getAction())){
//获取传递的数据
Bundle bundled=intent.getBundleExtra(SearchManager.APP_DATA);
if(bundled!=null){
String ttdata=bundled.getString("data");
tvdata.setText(ttdata);

}else{
tvdata.setText("no data");
}
}
}

 之前说到了处理结果的Activity将可能出现的两种情况的两种,现在就处理第二种状况,就是假如invoke search bar的

Activity同时也是处理搜索结果的Activity,如果按照之前的方式处理则会出现一种情况,搜索一次就实例化一次Activity,当按返回

键的时候会发现老是同一个Activity,其实为了使它只有一个实例化对象,只需简单的配置和代码就能实现

第一:在处理搜索结果Activity的manifest.xml中添加android:launchMode="singleTop"属性

第二:重写Activity的onNewIntent(Intent intent)



@Override
public void onNewIntent(Intent intent){
super.onNewIntent(intent);
//获得搜索框里值
String query=intent.getStringExtra(SearchManager.QUERY);
tvquery.setText(query);
//保存搜索记录
SearchRecentSuggestions suggestions=new SearchRecentSuggestions(this,
SearchSuggestionSampleProvider.AUTHORITY, SearchSuggestionSampleProvider.MODE);
suggestions.saveRecentQuery(query, null);
if(Intent.ACTION_SEARCH.equals(intent.getAction())){
//获取传递的数据
Bundle bundled=intent.getBundleExtra(SearchManager.APP_DATA);
if(bundled!=null){
String ttdata=bundled.getString("data");
tvdata.setText(ttdata);

}else{
tvdata.setText("no data");
}
}
}

相关知识:上面讲到了将最近的搜索值添加到搜索建议中,但却没有提到如果清理搜索建议中的值,与保存相似,SearchRecentSuggestion对象提供了一个clearHistory()方法

 


代码


private void clearSearchHistory() {
SearchRecentSuggestions suggestions = new SearchRecentSuggestions(this,
SearchSuggestionSampleProvider.AUTHORITY, SearchSuggestionSampleProvider.MODE);
suggestions.clearHistory();
}

接下来的介绍更加详细一些:

当您需要在您的应用程序中提供搜索服务时,您第一个想到的是您的搜索框要放哪呢?通过使用Android的搜索框架,应用程序将显示一个自定义搜索对话框来处理用户的搜索请求。通过一个简单的搜索按钮或从您的应用程序中调用API,搜索对话框就会显示在屏幕的顶部,并会自动显示您的应用程序图标。如下图所示:

本文将教你如何为你的应用程序提供一个自定义搜索对话框。这样做,给您的用户提供一个标准化的搜索体验,并能增加如语音搜索和搜索建议等功能。

基础知识

Android的搜索框架将代您管理的搜索对话框,您不需要自己去开发一个搜索框,不需要担心要把搜索框放什么位置,也不需要担心搜索框影响您当前的界面。所有的这些工作都由SearchManager类来为您处理(以下简称“搜索管理器”),它管理的Android搜索对话框的整个生命周期,并执行您的应用程序将发送的搜索请求,返回相应的搜索关键字。

当用户执行一个搜索,搜索管理器将使用一个专门的Intent把搜索查询的关键字传给您在配置文件中配置的处理搜索结果的Activity。从本质上讲,所有你需要的就是一个Activity来接收Intent,然后执行搜索,并给出结果。具体来说,你需要的做的事就包括以下内容:

一个搜索配置
我们用个XML配置文件来对搜索对话框进行配置,包括一些功能的配置,如文本框,设置语音搜索和搜索建议中显示的提示文字等。

一个用来处理搜索请求的Activity
这个Activity用来接收搜索查询的内容,然后搜索您的数据并显示搜索结果。

一种用户执行搜索的途径
默认情况下,一旦你配置了一个可搜索的Activity,设备搜索键(如果有)将调用搜索对话框。然而,你应该始终提供另一种手段,让用户可以调用搜索对话框,如在选项菜单中的搜索按钮或其他用户界面上的按钮,因为不是所有的设备提供一个专门的搜索键。

创建一个搜索对话框配置文件

搜索框配置文件是一个用来配置您的应用程序中搜索框的设置的XML文件,这个文件一般命名为searchable.xml,并且必须保存在项目的res/xml/目录下。

配置文件的根节点必须为,可以有一个或多个属性。如下所示:

  1. <?xml version="1.0" encoding="utf-8"?>
  2. <searchable xmlns:android="http://schemas.android.com/apk/res/android"
  3.     android:label="@string/searchLabel" android:hint="@string/searchHint">
  4. </searchable>

复制代码

上面的配置文件中,除android:hint属性外,其它都是一个搜索对话框必须的配置项,android:label是一个必须的属性,它的值为一个string资源引用,不能直接用字符串,通常会是应用程序的名称(尽管它是一个必须的属性,但通常情况下是不显示出来的,除非你开启了搜索建议功能)。android:hint是配置搜索框的输入提示信息,也必须引用string.xml中配置的字符串资源,不能直接使用字符串。

可以配置很多的属性,但大部分属性都只是在使用搜索建议和语音搜索时进行配置,尽管如此,我们建议你一定要配置android:hint,用于提示用户需要输入的信息。

接下来,你需要把这个配置文件放到你的应用程序中。

创建一个可用于搜索的Activity

当用户从一个搜索框执行搜索时,搜索管理器(Search Manager)会通过ACTION_SEARCH Intent 把要搜索的内容(关键字)发送到一个可执行搜索的Activity。这个Acitivity查询数据并显示结果。

定义一个可搜索的Activity

如果你还没有准备好,那么就创建一个用来执行搜索的Activity,声明它可以响应ACTION_SEARCH Intent ,并且增加搜索框配置信息。为此,你需要添加一个元素和一个元素在你的manifest文件中的节点。如下所示:

  1. <application ... >
  2.     <activity android:name=".MySearchableActivity" >
  3.         <intent-filter>
  4.             <action android:name="android.intent.action.SEARCH" />
  5.         </intent-filter>
  6.         <meta-data android:name="android.app.searchable"
  7.                   android:resource="@xml/searchable"/>
  8.     </activity>
  9.     ...
  10. </application>

复制代码

中的android:name属性值必须为”android.app.searchable”,android:resource属性值必须引用上面提到的res/xml/目录下的搜索配置文件(本例中的res/xml/searchable.xml)。

请注意,只有配置了上面的meta-data节点的Activity的节点才能执行搜索,如果想在整个应用程序中都可以调用搜索框,可以进行如下配置:

  1. <application ... >
  2.     <activity android:name=".MySearchableActivity" >
  3.         <intent-filter>
  4.             <action android:name="android.intent.action.SEARCH" />
  5.         </intent-filter>
  6.         <meta-data android:name="android.app.searchable"
  7.                   android:resource="@xml/searchable"/>
  8.     </activity>
  9.     <activity android:name=".AnotherActivity" ... >
  10.     </activity>
  11.     <!—这个配置就可以让你在整个应用程序中调用搜索框 -->
  12.     <meta-data android:name="android.app.default_searchable"
  13.               android:value=".MySearchableActivity" />
  14.     ...
  15. </application>

复制代码

上面代码中android:name=”android.app.default_searchable” 定义一个响应搜索框搜索请求的名称,android:value指定是由哪个Activity响应并执行搜索。当我们在应用程序中的 OtherAcitivity中执行搜索请求时,MySearchableActivity将会被加载用于执行搜索并显示搜索结果。

执行一个搜索

当一个Activity声明为可搜索时,执行实际的搜索包括三个步骤:接收查询,检索你的数据,并提交结果。

通常情况下,你的搜索结果需要在一个ListView中展现,所以你用于执行搜索的Acitivity要继承ListActivity,这样,可以方便的访问ListView的Api。

接收搜索查询

当从搜索对话框执行搜索时,刚才配置的可用于搜索的Acitivity将会被Intent激活,同时带着一些搜索相关的参数,你需要检查Intent并做出搜索响应,如下所示:

  1. @Override
  2. public void onCreate(Bundle savedInstanceState) {
  3.     super.onCreate(savedInstanceState);
  4.     setContentView(R.layout.search);
  5.     Intent intent = getIntent();
  6. //判断是否是搜索请求
  7.     if (Intent.ACTION_SEARCH.equals(intent.getAction())) {
  8. //获取搜索的查询内容(关键字)
  9.       String query = intent.getStringExtra(SearchManager.QUERY);
  10. //执行相应的查询动作
  11.       doMySearch(query);
  12.     }
  13. }

复制代码

doMySearch()方法将根据关键字查询数据库,或从网络上查询数据,如果是耗时的搜索,你还需要使用进度条,来告诉用户搜索正在进行,最后返回结果后,可以调用ListView的setAdapter()方法将结果显示在ListView中。

调用搜索对话框

你可以从应用程序中的任何一个地方调用onSearchRequested()方法激活搜索框,比如从菜单中或者一个按钮等。你也要以在 onCreate()方法中调用setDefaultKeyMode(DEFAULT_KEYS_SEARCH_LOCAL),这样,当用户按下键盘上的按键时,将会自动激活搜索框。

搜索框和普通对话框一样,浮动在屏幕的最上方,它不会改变任何Activity堆栈状态,没有任何Activity生命周期中的方法会被调用,只是当搜索框出现就,正在运行的Activity会失去输入焦点。

如果你要在执行搜索时,进行别的操作,可以重写onSearchRequested()方法,如下所示:

  1. @Override
  2. public boolean onSearchRequested() {
  3. //这个方法中干你想干的事,比如做一些被始化工作
  4.     pauseSomeStuff();
  5.     return super.onSearchRequested();
  6. }

复制代码

如果当前的Activity就是响应搜索请求的Activity时,会有以下两种情况:

默认情况下,ACTION_SEARCH Intent将会创建一个新的Activity,并调用onCreate()方法,这个新的Activity会显示在最前面,你将同时有两个 Activity实例。当你按“返回”键里,会回到没有执行搜索前的一个Activity。

另一种情况是配置了android:launchMode=”singleTop”的Activity,这时,我们需要在 onNewIntent(Intent)方法中处理搜索请求,如下所示:

  1. @Override
  2. public void onCreate(Bundle savedInstanceState) {
  3.     super.onCreate(savedInstanceState);
  4.     setContentView(R.layout.search);
  5.     handleIntent(getIntent());
  6. }
  7. @Override
  8. protected void onNewIntent(Intent intent) {
  9.     setIntent(intent);
  10.     handleIntent(intent);
  11. }
  12. private void handleIntent(Intent intent) {
  13.     if (Intent.ACTION_SEARCH.equals(intent.getAction())) {
  14.       String query = intent.getStringExtra(SearchManager.QUERY);
  15.       doMySearch(query);
  16.     }
  17. }

相应的Activity配置如下

  1. <activity android:name=".MySearchableActivity"
  2.               android:launchMode="singleTop" >
  3.     <intent-filter>
  4.         <action android:name="android.intent.action.SEARCH" />
  5.     </intent-filter>
  6.     <meta-data android:name="android.app.searchable"
  7.                       android:resource="@xml/searchable"/>
  8.   </activity>

复制代码

如何给搜索框增加参数

要给搜索框传递参数,我们需要重写onSearchRequested()方法,如下所示:

  1. @Override
  2. public boolean onSearchRequested() {
  3.     Bundle appData = new Bundle();
  4.     appData.putBoolean(MySearchableActivity.JARGON, true);
  5.     startSearch(null, false, appData, false);
  6.     return true;
  7. }

复制代码

我们的Activity在收到搜索框的搜索请求时,通过如下方法获取参数:

  1. Bundle appData = getIntent().getBundleExtra(SearchManager.APP_DATA);
  2. if (appData != null) {
  3.     boolean jargon = appData.getBoolean(MySearchableActivity.JARGON);
  4. }

复制代码

最后我们来看看如何使用android的语音搜索:

只需要对我们的搜索配置文件做如下改动,你的搜索就支持语音搜索了,配置文件如下所示:

  1. <?xml version="1.0" encoding="utf-8"?>
  2. <searchable xmlns:android="http://schemas.android.com/apk/res/android"
  3.     android:label="@string/searchLabel"
  4.     android:hint="@string/searchHint"
  5.     android:voiceSearchMode="showVoiceSearchButton|launchRecognizer">
  6. </searchable>

复制代码

好了,今天就到这儿,自己动手练习去吧!以后还会详细讲解如何增加搜索建议,语音搜索等内容。

关于搜索建议的智能提示

可参考:Adding Custom Suggestions

时间: 2024-12-11 11:10:15

Android---浮动搜索框(SearchManager)的相关文章

Android Material 搜索框实现详细说明

本文讲的是Android Material 搜索框实现详细说明, 关于我的应用,我收到了一些用户的反馈,他们反馈最多的是缺少搜索功能.对于像 Memento Calendar 这种囊括了诸如社交时间,纪念日,银行休假日,信息来源错综复杂的应用,我很赞同搜索是这个应用最重要的功能之一.问题是这个功能已经被实现了.Toolbar 里的一个搜索图标引导用户到一个搜索界面. A user can search by tapping the search icon on the Toolbar 我决定调研

Android SearchView搜索框组件的使用方法_Android

SearchView是搜索框组件,它可以让用户在文本框里输入文字,通过监听器取得用户的输入,当用户点击搜索时,监听器执行实际的搜索. 本文就为大家分享了SearchView搜索框组件的使用方法,供大家参考,具体内容如下 效果: 代码SearchActivity.java package com.jialianjia.bzw.activity; import android.os.Bundle; import android.widget.ArrayAdapter; import android.

Android SearchView搜索框组件的使用方法

SearchView是搜索框组件,它可以让用户在文本框里输入文字,通过监听器取得用户的输入,当用户点击搜索时,监听器执行实际的搜索. 本文就为大家分享了SearchView搜索框组件的使用方法,供大家参考,具体内容如下 效果: 代码SearchActivity.java package com.jialianjia.bzw.activity; import android.os.Bundle; import android.widget.ArrayAdapter; import android.

Android EditText搜索框实现图标居中

类似这样EditText 搜索框,hiht 提示有一个icon并且text内容. 重写EditText : package mobi.truekey.weapp2.widget; import android.content.Context; import android.content.res.TypedArray; import android.graphics.Canvas; import android.graphics.Paint; import android.graphics.dr

Android 系统有浏览记录搜索框

 一.配置搜索描述文件 要在res中的xml文件加创建sreachable.xml,内容如下: <?xml version="1.0" encoding="utf-8"?>  <searchable  xmlns:android="http://schemas.android.com/apk/res/android"  android:hint="@string/searchLable"  android:l

Android中如何实现清空搜索框的文字_Android

需求:项目中的有关搜索的地方,加上清空文字的功能,目的是为了增加用户体验,使用户删除文本更加快捷 解决过程:开始的时候感觉这个东西不太好实现,主要就是布局的问题,可能是开始顾虑的太多了,再加上当时产品催的不太紧,而且这个功能也不是必须实现的.但是今天不一样了,这个是老大让加上的,说别的很多应用中都有这个功能,没办法那就加上呗,试着去使用了相对布局去实现,把一个删除按键放在编辑框的右上方,当文字的时候就把删除按键给显示出来,当编辑框为空的时候就把删除按键给隐藏掉.布局代码 <?xml versio

Android实现简单实用的搜索框

本文实例为大家分享了Android实现搜索框展示的具体代码,供大家参考,具体内容如下 展示效果 代码区 SouActivity public class SouActivity extends AppCompatActivity implements TextWatcher{ @BindView(R.id.app_sou) EditText appSou; @BindView(R.id.app_sou_list) ListView appSouList; @BindView(R.id.activ

JQUERY 实现窗口滚动搜索框停靠效果(类似滚动停靠)_jquery

当页面需要显示的内容较多时,我们很多人采用分页的方法解决. 而有的时候,分页的效果却又是非常令人厌恶的.滚动条无疑是一种简单而又高效的一种方式.而这里,处于对用户体验的考虑,我使用Jquery实现了一种类似"滚动停靠"的效果.这样当我们向下滚动内容的时候,搜索框会"悬挂(停靠)"在窗口顶端. 这样做的好处是,当用户需要重新筛选内容的时候,不必再次向上滚动,随时可以输入条件进行搜索. 以下是我实现这个效果的思路: 首先,设计一个隐藏的搜索框.这个隐藏的搜索框的样式.事

Xamarin框架开发Android搜索框 Search Dialog实例

Android 的搜索有两种可用方式:Search Dialog,SearchView. SearchView 简单,随意使用,这里主要说说 Search Dialog 的基本用法, 因为 Xamarin 的处理方式稍稍和 原生 Android 有些不同. 效果: Searchable 要使用 Search Dialog 需要配置一个搜索配置文件 : 放到Resources/xml 目录下 . 如果xml 目录不存在,需要手动创建一个. 文件名随便, 一般取 searchable.xml <?x