Как я могу заставить мое приложение отображаться в глобальном поиске?

Я пытался написать простое приложение, которое позволяло бы мне выполнять поиск в глобальном поиске, но независимо от того, что я делаю, я не могу заставить свое приложение отображаться в глобальном поиске?

package com.mridang.messearch;

import java.util.ArrayList;

import android.app.SearchManager;
import android.content.ContentProvider;
import android.content.ContentResolver;
import android.content.ContentValues;
import android.content.Intent;
import android.database.CharArrayBuffer;
import android.database.ContentObserver;
import android.database.CrossProcessCursor;
import android.database.Cursor;
import android.database.CursorWindow;
import android.database.DataSetObserver;
import android.database.sqlite.SQLiteException;
import android.net.Uri;
import android.os.Bundle;
import android.text.TextUtils;

public class SearchProvider extends ContentProvider {

    final static String AUTHORITY = "com.android.mms.SuggestionsProvider";

    public SearchProvider() {
        super();
    }

    @Override
    public int delete(Uri uri, String selection, String[] selectionArgs) {
        return 0;
    }

    @Override
    public String getType(Uri uri) {
        return null;
    }

    @Override
    public Uri insert(Uri uri, ContentValues values) {
        return null;
    }

    @Override
    public int update(Uri uri, ContentValues values, String selection, String[] selectionArgs) {
        return 0;
    }

    @Override
    public boolean onCreate() {
        return true;
    }

    @Override
    public Cursor query(Uri uri, String[] projection, String selection, String[] selectionArgs, String sortOrder) {
        Uri u = Uri.parse(String.format("content://mms-sms/searchSuggest?pattern=%s", selectionArgs[0]));
        Cursor c = getContext().getContentResolver().query(u, null, null, null, null);

        return new SuggestionsCursor(c, selectionArgs[0]);
    }


    private class SuggestionsCursor implements CrossProcessCursor {
        Cursor curMessages;
        int intColumns;
        int intCurrent;
        ArrayList<Row> mRows = new ArrayList<Row>();
        String mQuery;

        public SuggestionsCursor(Cursor cursor, String query) {
            curMessages = cursor;
            mQuery = query;

            intColumns = cursor.getColumnCount();
            try {
                computeRows();
            } catch (SQLiteException ex) {
                // This can happen if the user enters -n (anything starting with
                // -).
                // sqlite3/fts3 can't handle it. Google for
                // "logic error or missing database fts3"
                // for commentary on it.
                mRows.clear(); // assume no results
            }
        }

        public int getCount() {
            return mRows.size();
        }

        private class Row {
            private String mSnippet;
            private int mRowNumber;

            public Row(int row, String snippet) {
                mSnippet = snippet.trim();
                mRowNumber = row;
            }

            public String getSnippet() {
                return mSnippet;
            }
        }

        private void computeRows() {
            int snippetColumn = curMessages.getColumnIndex("snippet");

            int count = curMessages.getCount();
            String previousSnippet = null;

            for (int i = 0; i < count; i++) {
                curMessages.moveToPosition(i);
                String snippet = curMessages.getString(snippetColumn);
                if (!TextUtils.equals(previousSnippet, snippet)) {
                    mRows.add(new Row(i, snippet));
                    previousSnippet = snippet;
                }
            }
        }

        private int[] computeOffsets(String offsetsString) {
            String[] vals = offsetsString.split(" ");

            int[] retvals = new int[vals.length];
            for (int i = retvals.length - 1; i >= 0; i--) {
                retvals[i] = Integer.parseInt(vals[i]);
            }
            return retvals;
        }

        public void fillWindow(int position, CursorWindow window) {
            int count = getCount();
            if (position < 0 || position > count + 1) {
                return;
            }
            window.acquireReference();
            try {
                int oldpos = getPosition();
                int pos = position;
                window.clear();
                window.setStartPosition(position);
                int columnNum = getColumnCount();
                window.setNumColumns(columnNum);
                while (moveToPosition(pos) && window.allocRow()) {
                    for (int i = 0; i < columnNum; i++) {
                        String field = getString(i);
                        if (field != null) {
                            if (!window.putString(field, pos, i)) {
                                window.freeLastRow();
                                break;
                            }
                        } else {
                            if (!window.putNull(pos, i)) {
                                window.freeLastRow();
                                break;
                            }
                        }
                    }
                    ++pos;
                }
                moveToPosition(oldpos);
            } catch (IllegalStateException e) {
                // simply ignore it
            } finally {
                window.releaseReference();
            }
        }

        public CursorWindow getWindow() {
            return null;
        }

        public boolean onMove(int oldPosition, int newPosition) {
            return ((CrossProcessCursor) curMessages).onMove(oldPosition, newPosition);
        }

        private String[] mVirtualColumns = new String[] { SearchManager.SUGGEST_COLUMN_INTENT_DATA,
                SearchManager.SUGGEST_COLUMN_INTENT_ACTION, SearchManager.SUGGEST_COLUMN_INTENT_EXTRA_DATA,
                SearchManager.SUGGEST_COLUMN_TEXT_1, };

        // Cursor column offsets for the above virtual columns.
        // These columns exist after the natural columns in the
        // database cursor. So, for example, the column called
        // SUGGEST_COLUMN_TEXT_1 comes 3 after mDatabaseCursor.getColumnCount().
        private final int INTENT_DATA_COLUMN = 0;
        private final int INTENT_ACTION_COLUMN = 1;
        private final int INTENT_EXTRA_DATA_COLUMN = 2;
        private final int INTENT_TEXT_COLUMN = 3;

        public int getColumnCount() {
            return intColumns + mVirtualColumns.length;
        }

        public int getColumnIndex(String columnName) {
            for (int i = 0; i < mVirtualColumns.length; i++) {
                if (mVirtualColumns[i].equals(columnName)) {
                    return intColumns + i;
                }
            }
            return curMessages.getColumnIndex(columnName);
        }

        public String[] getColumnNames() {
            String[] x = curMessages.getColumnNames();
            String[] y = new String[x.length + mVirtualColumns.length];

            for (int i = 0; i < x.length; i++) {
                y[i] = x[i];
            }

            for (int i = 0; i < mVirtualColumns.length; i++) {
                y[x.length + i] = mVirtualColumns[i];
            }

            return y;
        }

        public boolean moveToPosition(int intPosition) {
            if (intPosition >= 0 && intPosition < mRows.size()) {
                intCurrent = intPosition;
                curMessages.moveToPosition(mRows.get(intPosition).mRowNumber);
                return true;
            } else {
                return false;
            }
        }

        public boolean move(int offset) {
            return moveToPosition(intCurrent + offset);
        }

        public boolean moveToFirst() {
            return moveToPosition(0);
        }

        public boolean moveToLast() {
            return moveToPosition(mRows.size() - 1);
        }

        public boolean moveToNext() {
            return moveToPosition(intCurrent + 1);
        }

        public boolean moveToPrevious() {
            return curMessages.moveToPrevious();
        }

        public String getString(int column) {
            // if we're returning one of the columns in the underlying database
            // column
            // then do so here
            if (column < intColumns) {
                return curMessages.getString(column);
            }

            // otherwise we're returning one of the synthetic columns.
            // the constants like INTENT_DATA_COLUMN are offsets relative to
            // mColumnCount.
            Row row = mRows.get(intCurrent);
            switch (column - intColumns) {
            case INTENT_DATA_COLUMN:
                Uri.Builder b = Uri.parse("content://mms-sms/search").buildUpon();
                b = b.appendQueryParameter("pattern", row.getSnippet());
                Uri u = b.build();
                return u.toString();
            case INTENT_ACTION_COLUMN:
                return Intent.ACTION_SEARCH;
            case INTENT_EXTRA_DATA_COLUMN:
                return row.getSnippet();
            case INTENT_TEXT_COLUMN:
                return row.getSnippet();
            default:
                return null;
            }
        }

        public void close() {
            curMessages.close();
        }

        public void copyStringToBuffer(int columnIndex, CharArrayBuffer buffer) {
            curMessages.copyStringToBuffer(columnIndex, buffer);
        }

        public void deactivate() {
            curMessages.deactivate();
        }

        public byte[] getBlob(int columnIndex) {
            return null;
        }

        public int getColumnIndexOrThrow(String columnName) throws IllegalArgumentException {
            return 0;
        }

        public String getColumnName(int columnIndex) {
            return null;
        }

        public double getDouble(int columnIndex) {
            return 0;
        }

        public Bundle getExtras() {
            return Bundle.EMPTY;
        }

        public float getFloat(int columnIndex) {
            return curMessages.getFloat(columnIndex);
        }

        public int getInt(int columnIndex) {
            return curMessages.getInt(columnIndex);
        }

        public long getLong(int columnIndex) {
            return curMessages.getLong(columnIndex);
        }

        public int getPosition() {
            return curMessages.getPosition();
        }

        public short getShort(int columnIndex) {
            return 0;
        }

        public boolean getWantsAllOnMoveCalls() {
            return false;
        }

        public boolean isAfterLast() {
            return intCurrent >= mRows.size();
        }

        public boolean isBeforeFirst() {
            return intCurrent < 0;
        }

        public boolean isClosed() {
            return curMessages.isClosed();
        }

        public boolean isFirst() {
            return intCurrent == 0;
        }

        public boolean isLast() {
            return intCurrent == mRows.size() - 1;
        }

        public int getType(int columnIndex) {
            throw new UnsupportedOperationException(); // TODO revisit
        }

        public boolean isNull(int columnIndex) {
            return false; // TODO revisit
        }

        public void registerContentObserver(ContentObserver observer) {
            curMessages.registerContentObserver(observer);
        }

        public void registerDataSetObserver(DataSetObserver observer) {
            curMessages.registerDataSetObserver(observer);
        }

        public boolean requery() {
            return false;
        }

        public Bundle respond(Bundle extras) {
            return curMessages.respond(extras);
        }

        public void setNotificationUri(ContentResolver cr, Uri uri) {
            curMessages.setNotificationUri(cr, uri);
        }

        public void unregisterContentObserver(ContentObserver observer) {
            curMessages.unregisterContentObserver(observer);
        }

        public void unregisterDataSetObserver(DataSetObserver observer) {
            curMessages.unregisterDataSetObserver(observer);
        }
    }
}

Вот мой манифест

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    package="com.mridang.messearch"
    android:versionCode="1"
    android:versionName="1.0" >

    <uses-sdk
        android:minSdkVersion="14"
        android:targetSdkVersion="14" />

    <uses-permission android:name="android.permission.READ_SMS" />

    <application
        android:allowBackup="true"
        android:icon="@drawable/ic_launcher"
        android:label="@string/app_name"
        android:theme="@style/AppTheme" >
        <provider
            android:name=".SearchProvider"
            android:authorities="com.mridang.messearch.SearchProvider"
            android:exported="true"
            android:readPermission="android.permission.READ_SMS" >
            <path-permission
                android:pathPrefix="/search_suggest_query"
                android:readPermission="android.permission.GLOBAL_SEARCH" />
            <path-permission
                android:pathPrefix="/search_suggest_shortcut"
                android:readPermission="android.permission.GLOBAL_SEARCH" />
        </provider>

        <activity
            android:name=".SettingsActivity"
            android:exported="true"
            android:label="@string/title_activity_settings" >
            <intent-filter>
                <action android:name="android.intent.action.MAIN" />

                <category android:name="android.intent.category.LAUNCHER" />
            </intent-filter>

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

</manifest>

Вот моя конфигурация с возможностью поиска:

<searchable xmlns:android="http://schemas.android.com/apk/res/android"
    android:hint="@string/search_hint"
    android:imeOptions="actionSearch"
    android:includeInGlobalSearch="true"
    android:label="@string/search_label"
    android:searchSettingsDescription="@string/search_setting_description"
    android:searchSuggestAuthority="com.mridang.messearch.SearchProvider"
    android:searchSuggestIntentAction="android.intent.action.SEARCH"
    android:searchSuggestSelection="pattern" />

Когда я открываю приложение Google Search, я не вижу свое приложение в списке и не могу понять, почему? Есть идеи, что я делаю не так? Единственное приложение, которое, похоже, осуществляет поиск по всему миру, - это приложение Google Search. Я не вижу встроенного в Android, который кажется действительно странным.

введите описание изображения здесь

0 ответов

Другие вопросы по тегам