Android検索ダイアログを使用します。 パート3-カスタム提案

画像

これは、Android検索ダイアログの使用に関する最後の記事です(前の記事はこちらこちら )。 その中で、動的検索ヒントをダイアログに追加する方法、およびアプリケーションの検索をシステムのクイック検索ボックス(QSB)に統合する方法を説明します。 QSBの利点は、その助けを借りて、OSのほぼどこからでも情報を取得できることです。


理論


検索ヒントは、検索中のアプリケーションのデータを使用して作成されます。 ユーザーがそのうちの1つを選択すると、検索マネージャーは、検索を担当するアクティビティにインテントを送信します。 通常、ユーザーがダイアログの検索アイコンをクリックすると、検索タイプのインテントが送信されますが、この場合はプロンプトを選択するときに、別のタイプのインテントを定義して、インターセプトして適切なアクション(新しいダイアログの作成、アクティビティの呼び出しなど)を実行できます情報などを表示する
検索リクエストのデータは以前と同様にIntentを介して転送されますが、ここではURIを使用してコンテンツプロバイダーを介したリクエストのタイプを決定します。

繰り返しますが、ダイアログをレンダリングするためにアクションを実行する必要はありません。これはSearch Managerによって実行されます。必要なのは、構成xmlファイルを送信することだけです。

そのため、Search Managerがアクティビティを検索の責任と定義し、検索のヒントを提供すると、次の一連のアクションが発生します。
  1. Search Managerは、検索クエリのテキストを受信すると、ヒントを提供するコンテンツプロバイダーに要求を送信します。
  2. コンテンツプロバイダーは、検索クエリのテキストに一致するヒントを指すカーソルを返します。
  3. Search Managerはカーソルを使用してツールチップを表示します

プロンプトのリストが表示された後、次のことが発生する場合があります。

そのため、動的なツールチップが追加されるようにアプリケーション( パート1で説明したもの)を変更し、メカニズムを動作させるために、ツールチップを選択するときに、新しいアクティビティを呼び出します。これは、要求に応じて情報を表示します。 実装には次のものが必要です。


構成ファイルを変更する


音声検索を使用する場合など、ダイアログを表示して変更するには、構成ファイル(res / xml / searchable.xml)が必要であることを思い出します。 動的なヒントを使用するには、android:searchSuggestAuthorityパラメーターをファイルに追加する必要があります。 コンテンツプロバイダーの認証文字列と一致します。 さらに、パラメータandroid:searchMode = "queryRewriteFromText"を追加します。その値は、たとえばトラックボールを使用してプロンプトをナビゲートすると、ダイアログ内の検索文字列が上書きされることを示します。 また、選択演算子、ツールチップが選択されたときに送信されるインテントのタイプ、およびコンテンツプロバイダーを要求するために必要なダイアログ内の入力文字の最小数を指定するパラメーターを追加します。

ファイルres / xml / searchable.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:searchSettingsDescription="@string/settings_description" android:searchMode="queryRewriteFromText" android:includeInGlobalSearch="true" android:searchSuggestAuthority="com.example.search.SuggestionProvider" android:searchSuggestIntentAction="android.intent.action.VIEW" android:searchSuggestIntentData="content://com.example.search.SuggestionProvider/records" android:searchSuggestThreshold="1" android:searchSuggestSelection=" ?"> </searchable> 


コンテンツプロバイダーを作成する


実際、当社のコンテンツプロバイダーは他のプロバイダーと違いはありません。 ただし、プロンプトの表の各行について、検索マネージャーで必要な列が選択されていることを確認する必要があります。 コンテンツプロバイダーのquery()メソッドを使用して、プロンプトのデータを照会します。 さらに、ユーザーがダイアログで新しい文字を入力するたびに呼び出されます。 したがって、query()メソッドは、クエリに一致するテーブル内のレコードにカーソルを返す必要があり、Search Managerはプロンプトを表示できるようになります。 コードコメントのメソッドの説明を参照してください。
要求テキスト自体はURIに追加されるため、受信に問題はありません;標準のgetLastPathSegment()メソッドを使用するだけです。

ヒント表の作成


Search Managerは、レコードを指すカーソルを受け取ると、各レコードに特定の列セットが必要です。 2つは必須です。_IDは各ツールチップの一意の識別子で、SUGGEST_COLUMN_TEXT_1はツールチップテキストです。
たとえば、SUGGEST_COLUMN_ICON_1を使用して、ツールチップの左側に表示されるアイコンをレコードごとに定義できるオプションの列が多数あります(連絡先の検索などに非常に便利です)。

インテントのデータ型定義


URIを介して要求に応じてデータを送信するため、どのツールチップが選択されたかを判断するメカニズムが必要です。 2つの方法があります。 1つ目は、レコードごとに一意のデータがある個別の列SUGGEST_COLUMN_INTENT_DATAを定義することです。その後、getData()またはgetDataString()を使用してIntentからデータを取得できます。 2番目のオプションは、構成ファイル(res / xml / searchable.xml)内のすべてのインテントのデータ型を決定し、SUGGEST_COLUMN_INTENT_DATA_ID列を使用して各インテントの一意のデータをURIに追加することです。
2番目のオプションを使用します。SUGGEST_COLUMN_INTENT_DATA_IDからテーブルのrowIdへのマッピングを簡単に作成できるため、テーブルに個別の列を作成しませんでした。 SQLiteへのスポーツの関心のために、 FTS3が検索使用されました。つまり、PRIMARY KEYやNULL / NOT NULLなどの列(制約)に制限を課すことができない仮想テーブルを作成する必要がありました。 ただし、仮想テーブルには一意の行識別子があり、それにマッピングを設定します。 つまり、Intentのデータは次の形式になります。「/」およびテーブルの行のrowIdがURIに追加されます。

情報を表示するアクティビティを作成する


インターフェイスはres / layout / record_activity.xmlにあります。 そのアクティビティは、Intentからデータを取得し、コンテンツプロバイダーを介してカーソルを要求し、テキストフィールドにレコードを表示するだけです。

ファイルres / layout / record_activity.xml
 <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:orientation="vertical" android:layout_width="fill_parent" android:layout_height="fill_parent" android:padding="10dp"> <TextView android:id="@+id/record_header" android:textSize="25dp" android:textColor="?android:textColorPrimary" android:layout_width="wrap_content" android:layout_height="wrap_content"/> </LinearLayout> 


ここで、マニフェストにコンテンツプロバイダーと新しいアクティビティに関する情報を入力します。また、2つのアクティビティがあるため、デフォルトで検索を担当するアクティビティを示します。

AndroidManifest.xmlファイル
 <?xml version="1.0" encoding="utf-8"?> <manifest xmlns:android="http://schemas.android.com/apk/res/android" package="com.example.search" android:versionCode="1" android:versionName="1.0"> <application android:icon="@drawable/icon" android:label="@string/app_name"> <activity android:name=".Main" android:label="@string/app_name"> <intent-filter> <action android:name="android.intent.action.MAIN" /> <category android:name="android.intent.category.LAUNCHER" /> </intent-filter> <intent-filter> <action android:name="android.intent.action.SEARCH" /> </intent-filter> <meta-data android:name="android.app.searchable" android:resource="@xml/searchable" /> </activity> <activity android:name=".RecordActivity" android:theme="@android:style/Theme.NoTitleBar" /> <provider android:name=".SuggestionProvider" android:authorities="com.example.search.SuggestionProvider" /> <meta-data android:name="android.app.default_searchable" android:value=".Main" /> </application> <uses-sdk android:minSdkVersion="5" /> </manifest> 

検索を担当するアクティビティでインテントをインターセプトする


上記のすべてのステップの後、検索を担当するメインアクティビティでインテントを処理する必要があります。 ツールチップのインテントタイプをビューとして定義したため、チェックを追加するだけです。 条件が満たされた場合、RecordActivityはIntentを使用して起動され、そのデータには、URI + "/" +テーブルのツールチップが書き​​込まれます。

クイック検索ボックスの統合


カスタム提案を使用するようにアプリケーションを変更した後、システム検索に追加できます。 これを行うには、searchable.xmlファイルに2つのパラメーターを追加します。
  1. android:includeInGlobalSearch = "true"-QSBがアプリケーションを検索できることを示します。
  2. android:searchSettingsDescription = "@ string / settings_description"-アプリケーションの説明を示します。これはクイック検索ボックスの設定時に表示されます。 これらの設定は、設定->検索にあります。

これらのオプションはAndroid 1.6から利用できます。つまり、以下のバージョンでは、QSB用にアプリケーションを構成することはできません。

ソースコード


必要なすべてのクラスの完全なソースコードを示します。 Main.java-要求を見つけてコンテンツプロバイダーに送信するメインのアクティビティであるRecordActivity.java-特定のレコードのデータを含むインテントを受け取り、レコードへのリンクを受け取り、情報を表示します。 SuggestionProvider.javaは、Search Managerからプロンプトのテーブルへのリクエストを処理するコンテンツプロバイダーです。 RecordsDbHelper.java-テーブルの作成、テーブルへの入力、必要な表示の確立、およびレコードの「マッチング」自体を行います。

Main.javaファイル
 package com.example.search; import android.app.ListActivity; import android.app.SearchManager; import android.content.Intent; import android.database.Cursor; import android.os.Bundle; import android.view.Menu; import android.view.MenuInflater; import android.view.MenuItem; import android.view.View; import android.widget.Button; import android.widget.EditText; import android.widget.SimpleCursorAdapter; import android.widget.Toast; public class Main extends ListActivity { private EditText text; private Button add; private RecordsDbHelper mDbHelper; @Override public void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.main); mDbHelper = new RecordsDbHelper(this); Intent intent = getIntent(); if (Intent.ACTION_SEARCH.equals(intent.getAction())) { //     String query = intent.getStringExtra(SearchManager.QUERY); //  showResults(query); } else if (Intent.ACTION_VIEW.equals(intent.getAction())){ // Intent   RecordActivity Intent recordIntent = new Intent(this, RecordActivity.class); recordIntent.setData(intent.getData()); startActivity(recordIntent); finish(); } add = (Button) findViewById(R.id.add); text = (EditText) findViewById(R.id.text); add.setOnClickListener(new View.OnClickListener() { public void onClick(View view) { String data = text.getText().toString(); if (!data.equals("")) { saveTask(data); text.setText(""); } } }); } private void saveTask(String data) { mDbHelper.createRecord(data); } private void showResults(String query) { //  -    Cursor cursor = managedQuery(SuggestionProvider.CONTENT_URI, null, null, new String[] {query}, null); if (cursor == null) { Toast.makeText(this, "There are no results", Toast.LENGTH_SHORT).show(); } else { //  String[] from = new String[] { RecordsDbHelper.KEY_DATA }; int[] to = new int[] { R.id.text1 }; SimpleCursorAdapter records = new SimpleCursorAdapter(this, R.layout.record, cursor, from, to); getListView().setAdapter(records); } } //     (  res/menu/main_menu.xml) public boolean onCreateOptionsMenu(Menu menu) { MenuInflater inflater = getMenuInflater(); inflater.inflate(R.menu.main_menu, menu); return true; } public boolean onOptionsItemSelected(MenuItem item) { switch (item.getItemId()) { case R.id.search_record: onSearchRequested(); return true; default: return super.onOptionsItemSelected(item); } } } 


ファイルRecordActivity.java
 package com.example.search; import android.app.Activity; import android.database.Cursor; import android.net.Uri; import android.os.Bundle; import android.view.Menu; import android.view.MenuInflater; import android.view.MenuItem; import android.widget.TextView; public class RecordActivity extends Activity { @Override public void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.record_activity); // URI    Intent     - Uri uri = getIntent().getData(); Cursor cursor = managedQuery(uri, null, null, null, null); if (cursor == null) { finish(); } else { //     cursor.moveToFirst(); TextView record = (TextView) findViewById(R.id.record_header); int rIndex = cursor.getColumnIndexOrThrow(RecordsDbHelper.KEY_DATA); record.setText(cursor.getString(rIndex)); } } //         @Override public boolean onCreateOptionsMenu(Menu menu) { MenuInflater inflater = getMenuInflater(); inflater.inflate(R.menu.main_menu, menu); return true; } @Override public boolean onOptionsItemSelected(MenuItem item) { switch (item.getItemId()) { case R.id.search_record: onSearchRequested(); return true; default: return false; } } } 


ファイルSuggestionProvider.java
 package com.example.search; import android.app.SearchManager; import android.content.ContentProvider; import android.content.ContentResolver; import android.content.ContentValues; import android.content.UriMatcher; import android.database.Cursor; import android.net.Uri; import android.provider.BaseColumns; public class SuggestionProvider extends ContentProvider{ private RecordsDbHelper mDbHelper; public static String AUTHORITY = "com.example.search.SuggestionProvider"; public static final Uri CONTENT_URI = Uri.parse("content://" + AUTHORITY + "/records"); //MIME   getType() public static final String RECORDS_MIME_TYPE = ContentResolver.CURSOR_DIR_BASE_TYPE + "/vnd.example.search"; public static final String RECORD_MIME_TYPE = ContentResolver.CURSOR_ITEM_BASE_TYPE + "/vnd.example.search"; //   URI private static final int SEARCH_RECORDS = 0; private static final int GET_RECORD = 1; private static final int SEARCH_SUGGEST = 2; private static final UriMatcher sURIMatcher = makeUriMatcher(); @Override public boolean onCreate() { mDbHelper = new RecordsDbHelper(getContext()); return true; } /** *    Search Manager'a. *    ,    URI. *      ,     selectionArgs   . *    . */ @Override public Cursor query(Uri uri, String[] projection, String selection, String[] selectionArgs, String sortOrder) { // UriMatcher,      .       switch (sURIMatcher.match(uri)) { case SEARCH_SUGGEST: if (selectionArgs == null) { throw new IllegalArgumentException( "selectionArgs must be provided for the Uri: " + uri); } return getSuggestions(selectionArgs[0]); case SEARCH_RECORDS: if (selectionArgs == null) { throw new IllegalArgumentException( "selectionArgs must be provided for the Uri: " + uri); } return search(selectionArgs[0]); case GET_RECORD: return getRecord(uri); default: throw new IllegalArgumentException("Unknown Uri: " + uri); } } private Cursor getSuggestions(String query) { query = query.toLowerCase(); String[] columns = new String[] { BaseColumns._ID, RecordsDbHelper.KEY_DATA, SearchManager.SUGGEST_COLUMN_INTENT_DATA_ID}; return mDbHelper.getRecordMatches(query, columns); } private Cursor search(String query) { query = query.toLowerCase(); String[] columns = new String[] { BaseColumns._ID, RecordsDbHelper.KEY_DATA}; return mDbHelper.getRecordMatches(query, columns); } private Cursor getRecord(Uri uri) { String rowId = uri.getLastPathSegment(); String[] columns = new String[] { RecordsDbHelper.KEY_DATA}; return mDbHelper.getRecord(rowId, columns); } /** *   *     URI   */ private static UriMatcher makeUriMatcher() { UriMatcher matcher = new UriMatcher(UriMatcher.NO_MATCH); //   matcher.addURI(AUTHORITY, "records", SEARCH_RECORDS); matcher.addURI(AUTHORITY, "records/#", GET_RECORD); //   matcher.addURI(AUTHORITY, SearchManager.SUGGEST_URI_PATH_QUERY, SEARCH_SUGGEST); matcher.addURI(AUTHORITY, SearchManager.SUGGEST_URI_PATH_QUERY + "/*", SEARCH_SUGGEST); return matcher; } //  (   ContentProvider) @Override public String getType(Uri uri) { switch (sURIMatcher.match(uri)) { case SEARCH_RECORDS: return RECORDS_MIME_TYPE; case SEARCH_SUGGEST: return SearchManager.SUGGEST_MIME_TYPE; case GET_RECORD: return RECORD_MIME_TYPE; default: throw new IllegalArgumentException("Unknown URL " + uri); } } @Override public int update(Uri uri, ContentValues values, String selection, String[] selectionArgs) { throw new UnsupportedOperationException(); } @Override public Uri insert(Uri uri, ContentValues values) { throw new UnsupportedOperationException(); } @Override public int delete(Uri uri, String selection, String[] selectionArgs) { throw new UnsupportedOperationException(); } } 


ファイルRecordsDbHelper.java
 package com.example.search; import java.util.HashMap; import android.app.SearchManager; import android.content.ContentValues; import android.content.Context; import android.database.Cursor; import android.database.sqlite.SQLiteDatabase; import android.database.sqlite.SQLiteOpenHelper; import android.database.sqlite.SQLiteQueryBuilder; import android.provider.BaseColumns; import android.util.Log; public class RecordsDbHelper { //    -  public static final String KEY_DATA = SearchManager.SUGGEST_COLUMN_TEXT_1; private static final String TAG = "RecordsDbHelper"; private DatabaseHelper mDbHelper; private SQLiteDatabase mDb; private static final String DATABASE_NAME = "datas"; private static final String DATABASE_TABLE = "records"; private static final int DATABASE_VERSION = 2; //   private static final String DATABASE_CREATE = "CREATE VIRTUAL TABLE " + DATABASE_TABLE + " USING fts3 (" + KEY_DATA + ");"; private static final HashMap<String,String> mColumnMap = buildColumnMap(); /** *  ,     rowId * @param rowId id   * @param columns   ;  null,   * @return ,    , null -      */ public Cursor getRecord(String rowId, String[] columns) { String selection = "rowid = ?"; String[] selectionArgs = new String[] {rowId}; return query(selection, selectionArgs, columns); } /** *  ,    ,    * @param query    * @param columns   ;  null,   * @return ,   ,   , null -      */ public Cursor getRecordMatches(String query, String[] columns) { String selection = KEY_DATA + " MATCH ?"; String[] selectionArgs = new String[] {query+"*"}; return query(selection, selectionArgs, columns); } /** *     . *      SQLiteQueryBuilder. *   ,        SUGGEST_COLUMN_INTENT_DATA_ID *        URI. */ private static HashMap<String,String> buildColumnMap() { HashMap<String,String> map = new HashMap<String,String>(); map.put(KEY_DATA, KEY_DATA); map.put(BaseColumns._ID, "rowid AS " + BaseColumns._ID); map.put(SearchManager.SUGGEST_COLUMN_INTENT_DATA_ID, "rowid AS " + SearchManager.SUGGEST_COLUMN_INTENT_DATA_ID); return map; } /** * * @param selection   * @param selectionArgs ,  "?"     * @param columns    * @return ,    ,     */ private Cursor query(String selection, String[] selectionArgs, String[] columns) { /* SQLiteBuilder       *   ,     - *   . */ SQLiteQueryBuilder builder = new SQLiteQueryBuilder(); builder.setTables(DATABASE_TABLE); builder.setProjectionMap(mColumnMap); Cursor cursor = builder.query(mDbHelper.getReadableDatabase(), columns, selection, selectionArgs, null, null, null); if (cursor == null) { return null; } else if (!cursor.moveToFirst()) { cursor.close(); return null; } return cursor; } /** */  */ private static class DatabaseHelper extends SQLiteOpenHelper { DatabaseHelper(Context context) { super(context, DATABASE_NAME, null, DATABASE_VERSION); } @Override public void onCreate(SQLiteDatabase db) { db.execSQL(DATABASE_CREATE); } @Override public void onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion) { Log.w(TAG, "Upgrading database from version " + oldVersion + " to " + newVersion + ", which will destroy all old data"); db.execSQL("DROP TABLE IF EXISTS records"); onCreate(db); } } public RecordsDbHelper(Context context) { mDbHelper = new DatabaseHelper(context); } /** *     * @param data ,    * @return id ,  -1,     */ public long createRecord(String data) { mDb = mDbHelper.getWritableDatabase(); ContentValues initialValues = new ContentValues(); initialValues.put(KEY_DATA, data); return mDb.insert(DATABASE_TABLE, null, initialValues); } } 


プロジェクト全体はcode.google.com取得できます。
ご清聴ありがとうございました!

Source: https://habr.com/ru/post/J111961/


All Articles