zoukankan      html  css  js  c++  java
  • 【Android】搜索框架

    ZZ:如何使用Android的搜索框架

    (好文要转,学习Android搜索框架怎么可以不看这篇文章?)

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

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

    1、基础知识

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

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

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

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

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

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

    搜索框配置文件是一个用来配置您的应用程序中搜索框的设置的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,用于提示用户需要输入的信息。

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

    3、创建一个可用于搜索的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将会被加载用于执行搜索并显示搜索结果。

    4、执行一个搜索

    当一个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  
     6     Intent intent = getIntent();
     7     //判断是否是搜索请求
     8     if (Intent.ACTION_SEARCH.equals(intent.getAction())) {
     9       //获取搜索的查询内容(关键字)
    10       String query = intent.getStringExtra(SearchManager.QUERY);
    11        //执行相应的查询动作
    12       doMySearch(query);
    13     }
    14 }

    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  
     8 @Override
     9 protected void onNewIntent(Intent intent) {
    10     setIntent(intent);
    11     handleIntent(intent);
    12 }
    13  
    14 private void handleIntent(Intent intent) {
    15     if (Intent.ACTION_SEARCH.equals(intent.getAction())) {
    16       String query = intent.getStringExtra(SearchManager.QUERY);
    17       doMySearch(query);
    18     }
    19 }

    相应的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>

    5、如何给搜索框增加参数

    要给搜索框传递参数,我们需要重写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>

    下面是示例代码:

     1 package com.ideasandroid.demo;
     2 
     3 import android.app.Activity;
     4 import android.app.SearchManager;
     5 import android.content.Intent;
     6 import android.os.Bundle;
     7 import android.view.Menu;
     8 import android.view.MenuItem;
     9 import android.widget.Toast;
    10 
    11 /**
    12  *  Copyright (C) 2010 ideasandroid
    13  *  演示android系统浮动搜索框
    14  *  欢迎访问http://www.ideasandroid.com
    15  *  让程序开发不再那么神秘
    16  */
    17 public class FloatSearchBoxDemo extends Activity {
    18     private final int SEARCH_MENU = 1;
    19 
    20     @Override
    21     public void onCreate(Bundle savedInstanceState) {
    22         super.onCreate(savedInstanceState);
    23         setContentView(R.layout.main);
    24         setDefaultKeyMode(DEFAULT_KEYS_SEARCH_LOCAL);
    25         handleSearchQuery(getIntent());
    26     }
    27 
    28     private void handleSearchQuery(Intent queryIntent) {
    29         final String queryAction = queryIntent.getAction();
    30         if (Intent.ACTION_SEARCH.equals(queryAction)) {
    31             onSearch(queryIntent);
    32         }
    33     }
    34 
    35     @Override
    36     protected void onNewIntent(Intent intent) {
    37         setIntent(intent);
    38         handleSearchQuery(intent);
    39     }
    40 
    41     private void onSearch(Intent intent) {
    42         final String queryString = intent.getStringExtra(SearchManager.QUERY);
    43         Toast.makeText(this, queryString, Toast.LENGTH_LONG).show();
    44     }
    45 
    46     @Override
    47     public boolean onCreateOptionsMenu(Menu menu) {
    48         boolean result = super.onCreateOptionsMenu(menu);
    49         menu.add(0, SEARCH_MENU, 0, getText(R.string.searchMenu)).setIcon(
    50                 android.R.drawable.ic_menu_search);
    51         return result;
    52     }
    53 
    54     @Override
    55     public boolean onMenuItemSelected(int featureId, MenuItem item) {
    56         switch (item.getItemId()) {
    57         case SEARCH_MENU:
    58             onSearchRequested();
    59             return true;
    60         }
    61 
    62         return super.onMenuItemSelected(featureId, item);
    63     }
    64 
    65 }

    AndroidManifest.xml文件:

     1 <?xml version="1.0" encoding="utf-8"?>
     2 <manifest xmlns:android="http://schemas.android.com/apk/res/android"
     3     package="com.ideasandroid.demo" android:versionCode="1"
     4     android:versionName="1.0">
     5     <application android:icon="@drawable/icon" android:label="@string/app_name">
     6         <activity android:name=".FloatSearchBoxDemo" android:launchMode="singleTop" android:label="@string/app_name">
     7             <intent-filter>
     8                 <action android:name="android.intent.action.MAIN" />
     9                 <category android:name="android.intent.category.LAUNCHER" />
    10             </intent-filter>
    11             <intent-filter>
    12                 <action android:name="android.intent.action.SEARCH" />
    13             </intent-filter>
    14             <meta-data
    15                 android:name="android.app.searchable"
    16                 android:resource="@xml/searchable" />
    17         </activity>
    18         <!--声名在本应用程序中都可以调用onSearchRequested()进行搜索-->
    19         <meta-data
    20             android:name="android.app.default_searchable"
    21             android:value=".FloatSearchBoxDemo" /> 
    22     </application>
    23     <uses-sdk android:minSdkVersion="4" />
    24 
    25 </manifest> 

    res/xml/serachable.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"
    4     android:hint="@string/searchHint"
    5     android:voiceSearchMode="showVoiceSearchButton|launchRecognizer"/>

    ~感谢原作者提供如此详细的解释和示例!~

     

  • 相关阅读:
    angularjs的$on、$emit、$broadcast
    angularjs中的路由介绍详解 ui-route(转)
    ionic入门教程-ionic路由详解(state、route、resolve)(转)
    Cocos Creator 加载使用protobuf第三方库,因为加载顺序报错
    Cocos Creator 计时器错误 cc.Scheduler: Illegal target which doesn't have uuid or instanceId.
    Cocos Creator 构造函数传参警告 Can not instantiate CCClass 'Test' with arguments.
    Cocos Creator 对象池NodePool
    Cocos Creator 坐标系 (convertToWorldSpaceAR、convertToNodeSpaceAR)
    Cocos Creator 常驻节点addPersistRootNode
    Cocos Creator 配合Tiled地图的使用
  • 原文地址:https://www.cnblogs.com/lqminn/p/2805398.html
Copyright © 2011-2022 走看看