SearchView.java revision 1b01ce23fa533d467ca2ccbc65c980a59662aa8d
1818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes/*
21b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes * Copyright (C) 2014 The Android Open Source Project
3818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes *
4818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes * Licensed under the Apache License, Version 2.0 (the "License");
5818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes * you may not use this file except in compliance with the License.
6818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes * You may obtain a copy of the License at
7818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes *
8818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes *      http://www.apache.org/licenses/LICENSE-2.0
9818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes *
10818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes * Unless required by applicable law or agreed to in writing, software
11818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes * distributed under the License is distributed on an "AS IS" BASIS,
12818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes * See the License for the specific language governing permissions and
14818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes * limitations under the License.
15818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes */
16818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
17818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banespackage android.support.v7.widget;
18818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
191b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banesimport android.annotation.TargetApi;
20818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banesimport android.app.PendingIntent;
21818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banesimport android.app.SearchManager;
22818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banesimport android.app.SearchableInfo;
23818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banesimport android.content.ActivityNotFoundException;
24818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banesimport android.content.ComponentName;
25818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banesimport android.content.Context;
26818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banesimport android.content.Intent;
27818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banesimport android.content.pm.PackageManager;
28818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banesimport android.content.pm.ResolveInfo;
29818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banesimport android.content.res.Configuration;
30818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banesimport android.content.res.Resources;
31818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banesimport android.content.res.TypedArray;
32818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banesimport android.database.Cursor;
33818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banesimport android.graphics.Rect;
34818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banesimport android.graphics.drawable.Drawable;
35818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banesimport android.net.Uri;
36818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banesimport android.os.Build;
37818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banesimport android.os.Bundle;
382e21b5e22c320fd7e6af86a7cc05b4b11d7a0f64Chris Banesimport android.os.ResultReceiver;
39818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banesimport android.speech.RecognizerIntent;
40818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banesimport android.support.v4.view.KeyEventCompat;
41818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banesimport android.support.v4.widget.CursorAdapter;
42818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banesimport android.support.v7.appcompat.R;
431b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banesimport android.support.v7.internal.widget.ViewUtils;
442e21b5e22c320fd7e6af86a7cc05b4b11d7a0f64Chris Banesimport android.support.v7.view.CollapsibleActionView;
45818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banesimport android.text.Editable;
46818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banesimport android.text.InputType;
47818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banesimport android.text.Spannable;
48818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banesimport android.text.SpannableStringBuilder;
49818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banesimport android.text.TextUtils;
50818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banesimport android.text.TextWatcher;
51818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banesimport android.text.style.ImageSpan;
52818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banesimport android.util.AttributeSet;
53818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banesimport android.util.Log;
54818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banesimport android.view.KeyEvent;
55818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banesimport android.view.LayoutInflater;
56818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banesimport android.view.View;
57818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banesimport android.view.ViewTreeObserver;
58818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banesimport android.view.inputmethod.EditorInfo;
59818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banesimport android.view.inputmethod.InputMethodManager;
60818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banesimport android.widget.AdapterView;
61818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banesimport android.widget.AdapterView.OnItemClickListener;
62818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banesimport android.widget.AdapterView.OnItemSelectedListener;
63818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banesimport android.widget.AutoCompleteTextView;
64818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banesimport android.widget.ImageView;
65818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banesimport android.widget.ListView;
66818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banesimport android.widget.TextView;
67818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banesimport android.widget.TextView.OnEditorActionListener;
68818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
69818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banesimport java.lang.reflect.Method;
70818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banesimport java.util.WeakHashMap;
71818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
72818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banesimport static android.support.v7.widget.SuggestionsAdapter.getColumnString;
73818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
74818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes/**
75818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes * A widget that provides a user interface for the user to enter a search query and submit a request
76818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes * to a search provider. Shows a list of query suggestions or results, if available, and allows the
77818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes * user to pick a suggestion or result to launch into.
78818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes *
799dcd2e58138ca4eb4b18f80b50e8979329e859d6Scott Main * <p class="note"><strong>Note:</strong> This class is included in the <a
809dcd2e58138ca4eb4b18f80b50e8979329e859d6Scott Main * href="{@docRoot}tools/extras/support-library.html">support library</a> for compatibility
819dcd2e58138ca4eb4b18f80b50e8979329e859d6Scott Main * with API level 7 and higher. If you're developing your app for API level 11 and higher
829dcd2e58138ca4eb4b18f80b50e8979329e859d6Scott Main * <em>only</em>, you should instead use the framework {@link android.widget.SearchView} class.</p>
839dcd2e58138ca4eb4b18f80b50e8979329e859d6Scott Main *
84818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes * <p>
859dcd2e58138ca4eb4b18f80b50e8979329e859d6Scott Main * When the SearchView is used in an {@link android.support.v7.app.ActionBar}
869dcd2e58138ca4eb4b18f80b50e8979329e859d6Scott Main * as an action view, it's collapsed by default, so you must provide an icon for the action.
87818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes * </p>
88818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes * <p>
899dcd2e58138ca4eb4b18f80b50e8979329e859d6Scott Main * If you want the search field to always be visible, then call
909dcd2e58138ca4eb4b18f80b50e8979329e859d6Scott Main * {@link #setIconifiedByDefault(boolean) setIconifiedByDefault(false)}.
91818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes * </p>
92818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes *
93818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes * <div class="special reference">
94818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes * <h3>Developer Guides</h3>
95818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes * <p>For information about using {@code SearchView}, read the
969dcd2e58138ca4eb4b18f80b50e8979329e859d6Scott Main * <a href="{@docRoot}guide/topics/search/index.html">Search</a> API guide.
979dcd2e58138ca4eb4b18f80b50e8979329e859d6Scott Main * Additional information about action views is also available in the <<a
989dcd2e58138ca4eb4b18f80b50e8979329e859d6Scott Main * href="{@docRoot}guide/topics/ui/actionbar.html#ActionView">Action Bar</a> API guide</p>
99818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes * </div>
100818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes *
101818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes * @see android.support.v4.view.MenuItemCompat#SHOW_AS_ACTION_COLLAPSE_ACTION_VIEW
102818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes */
1031b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banespublic class SearchView extends LinearLayoutCompat implements CollapsibleActionView {
104818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
105818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    private static final boolean DBG = false;
106818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    private static final String LOG_TAG = "SearchView";
107818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
1081b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes    private static final boolean IS_AT_LEAST_FROYO = Build.VERSION.SDK_INT >= 8;
1091b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes
110818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    /**
111818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * Private constant for removing the microphone in the keyboard.
112818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     */
113818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    private static final String IME_OPTION_NO_MICROPHONE = "nm";
114818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
1151b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes    private final SearchAutoComplete mQueryTextView;
1161b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes    private final View mSearchEditFrame;
1171b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes    private final View mSearchPlate;
1181b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes    private final View mSubmitArea;
1191b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes    private final ImageView mSearchButton;
1201b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes    private final ImageView mSubmitButton;
1211b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes    private final ImageView mCloseButton;
1221b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes    private final ImageView mVoiceButton;
1231b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes    private final ImageView mSearchHintIcon;
1241b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes    private final View mDropDownAnchor;
1251b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes    private final int mSearchIconResId;
1261b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes
1271b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes    // Resources used by SuggestionsAdapter to display suggestions.
1281b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes    private final int mSuggestionRowLayout;
1291b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes    private final int mSuggestionCommitIconResId;
1301b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes
1311b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes    // Intents used for voice searching.
1321b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes    private final Intent mVoiceWebSearchIntent;
1331b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes    private final Intent mVoiceAppSearchIntent;
134818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    private OnQueryTextListener mOnQueryChangeListener;
135818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    private OnCloseListener mOnCloseListener;
1361b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes    private OnFocusChangeListener mOnQueryTextFocusChangeListener;
137818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    private OnSuggestionListener mOnSuggestionListener;
138818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    private OnClickListener mOnSearchClickListener;
139818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
140818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    private boolean mIconifiedByDefault;
141818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    private boolean mIconified;
142818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    private CursorAdapter mSuggestionsAdapter;
143818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    private boolean mSubmitButtonEnabled;
144818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    private CharSequence mQueryHint;
145818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    private boolean mQueryRefinement;
146818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    private boolean mClearingFocus;
147818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    private int mMaxWidth;
148818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    private boolean mVoiceButtonEnabled;
149818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    private CharSequence mOldQueryText;
150818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    private CharSequence mUserQuery;
151818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    private boolean mExpandedInActionView;
152818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    private int mCollapsedImeOptions;
153818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
154818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    private SearchableInfo mSearchable;
155818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    private Bundle mAppSearchData;
156818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
157818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    static final AutoCompleteTextViewReflector HIDDEN_METHOD_INVOKER = new AutoCompleteTextViewReflector();
158818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
159818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    /*
160818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * SearchView can be set expanded before the IME is ready to be shown during
161818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * initial UI setup. The show operation is asynchronous to account for this.
162818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     */
163818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    private Runnable mShowImeRunnable = new Runnable() {
164818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        public void run() {
165818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            InputMethodManager imm = (InputMethodManager)
166818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                    getContext().getSystemService(Context.INPUT_METHOD_SERVICE);
167818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
168818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            if (imm != null) {
1692e21b5e22c320fd7e6af86a7cc05b4b11d7a0f64Chris Banes                HIDDEN_METHOD_INVOKER.showSoftInputUnchecked(imm, SearchView.this, 0);
170818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            }
171818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        }
172818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    };
173818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
1741b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes    private final Runnable mUpdateDrawableStateRunnable = new Runnable() {
175818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        public void run() {
176818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            updateFocusedState();
177818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        }
178818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    };
179818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
180818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    private Runnable mReleaseCursorRunnable = new Runnable() {
181818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        public void run() {
182818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            if (mSuggestionsAdapter != null && mSuggestionsAdapter instanceof SuggestionsAdapter) {
183818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                mSuggestionsAdapter.changeCursor(null);
184818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            }
185818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        }
186818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    };
187818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
188818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    // A weak map of drawables we've gotten from other packages, so we don't load them
189818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    // more than once.
190818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    private final WeakHashMap<String, Drawable.ConstantState> mOutsideDrawablesCache =
191818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            new WeakHashMap<String, Drawable.ConstantState>();
192818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
193818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    /**
194818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * Callbacks for changes to the query text.
195818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     */
196818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    public interface OnQueryTextListener {
197818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
198818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        /**
199818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes         * Called when the user submits the query. This could be due to a key press on the
200818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes         * keyboard or due to pressing a submit button.
201818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes         * The listener can override the standard behavior by returning true
202818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes         * to indicate that it has handled the submit request. Otherwise return false to
203818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes         * let the SearchView handle the submission by launching any associated intent.
204818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes         *
205818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes         * @param query the query text that is to be submitted
206818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes         *
207818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes         * @return true if the query has been handled by the listener, false to let the
208818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes         * SearchView perform the default action.
209818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes         */
210818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        boolean onQueryTextSubmit(String query);
211818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
212818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        /**
213818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes         * Called when the query text is changed by the user.
214818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes         *
215818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes         * @param newText the new content of the query text field.
216818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes         *
217818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes         * @return false if the SearchView should perform the default action of showing any
218818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes         * suggestions if available, true if the action was handled by the listener.
219818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes         */
220818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        boolean onQueryTextChange(String newText);
221818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    }
222818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
223818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    public interface OnCloseListener {
224818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
225818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        /**
226818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes         * The user is attempting to close the SearchView.
227818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes         *
228818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes         * @return true if the listener wants to override the default behavior of clearing the
229818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes         * text field and dismissing it, false otherwise.
230818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes         */
231818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        boolean onClose();
232818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    }
233818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
234818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    /**
235818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * Callback interface for selection events on suggestions. These callbacks
236818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * are only relevant when a SearchableInfo has been specified by {@link #setSearchableInfo}.
237818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     */
238818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    public interface OnSuggestionListener {
239818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
240818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        /**
241818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes         * Called when a suggestion was selected by navigating to it.
242818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes         * @param position the absolute position in the list of suggestions.
243818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes         *
244818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes         * @return true if the listener handles the event and wants to override the default
245818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes         * behavior of possibly rewriting the query based on the selected item, false otherwise.
246818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes         */
247818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        boolean onSuggestionSelect(int position);
248818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
249818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        /**
250818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes         * Called when a suggestion was clicked.
251818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes         * @param position the absolute position of the clicked item in the list of suggestions.
252818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes         *
253818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes         * @return true if the listener handles the event and wants to override the default
254818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes         * behavior of launching any intent or submitting a search query specified on that item.
255818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes         * Return false otherwise.
256818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes         */
257818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        boolean onSuggestionClick(int position);
258818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    }
259818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
260818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    public SearchView(Context context) {
261818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        this(context, null);
262818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    }
263818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
264818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    public SearchView(Context context, AttributeSet attrs) {
2651b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes        this(context, attrs, R.attr.searchViewStyle);
2661b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes    }
267818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
2681b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes    public SearchView(Context context, AttributeSet attrs, int defStyleAttr) {
2691b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes        super(context, attrs, defStyleAttr);
270818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
2711b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes        final TypedArray a = context.obtainStyledAttributes(
2721b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes                attrs, R.styleable.SearchView, defStyleAttr, 0);
2731b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes        final LayoutInflater inflater = (LayoutInflater) context.getSystemService(
2741b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes                Context.LAYOUT_INFLATER_SERVICE);
2751b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes        final int layoutResId = a.getResourceId(R.styleable.SearchView_layout, 0);
2761b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes        inflater.inflate(layoutResId, this, true);
277818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        mQueryTextView = (SearchAutoComplete) findViewById(R.id.search_src_text);
278818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        mQueryTextView.setSearchView(this);
279818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
280818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        mSearchEditFrame = findViewById(R.id.search_edit_frame);
281818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        mSearchPlate = findViewById(R.id.search_plate);
282818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        mSubmitArea = findViewById(R.id.submit_area);
2831b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes        mSearchButton = (ImageView) findViewById(R.id.search_button);
2841b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes        mSubmitButton = (ImageView) findViewById(R.id.search_go_btn);
285818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        mCloseButton = (ImageView) findViewById(R.id.search_close_btn);
2861b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes        mVoiceButton = (ImageView) findViewById(R.id.search_voice_btn);
287818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        mSearchHintIcon = (ImageView) findViewById(R.id.search_mag_icon);
2881b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes        // Set up icons and backgrounds.
2891b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes        mSearchPlate.setBackgroundDrawable(a.getDrawable(R.styleable.SearchView_queryBackground));
2901b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes        mSubmitArea.setBackgroundDrawable(a.getDrawable(R.styleable.SearchView_submitBackground));
2911b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes        mSearchIconResId = a.getResourceId(R.styleable.SearchView_searchIcon, 0);
2921b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes        mSearchButton.setImageResource(mSearchIconResId);
2931b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes        mSubmitButton.setImageDrawable(a.getDrawable(R.styleable.SearchView_goIcon));
2941b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes        mCloseButton.setImageDrawable(a.getDrawable(R.styleable.SearchView_closeIcon));
2951b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes        mVoiceButton.setImageDrawable(a.getDrawable(R.styleable.SearchView_voiceIcon));
2961b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes        mSearchHintIcon.setImageDrawable(a.getDrawable(R.styleable.SearchView_searchIcon));
2971b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes
2981b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes        // Extract dropdown layout resource IDs for later use.
2991b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes        mSuggestionRowLayout = a.getResourceId(R.styleable.SearchView_suggestionRowLayout, 0);
3001b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes        mSuggestionCommitIconResId = a.getResourceId(R.styleable.SearchView_commitIcon, 0);
301818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
302818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        mSearchButton.setOnClickListener(mOnClickListener);
303818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        mCloseButton.setOnClickListener(mOnClickListener);
304818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        mSubmitButton.setOnClickListener(mOnClickListener);
305818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        mVoiceButton.setOnClickListener(mOnClickListener);
306818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        mQueryTextView.setOnClickListener(mOnClickListener);
307818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
308818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        mQueryTextView.addTextChangedListener(mTextWatcher);
309818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        mQueryTextView.setOnEditorActionListener(mOnEditorActionListener);
310818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        mQueryTextView.setOnItemClickListener(mOnItemClickListener);
311818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        mQueryTextView.setOnItemSelectedListener(mOnItemSelectedListener);
312818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        mQueryTextView.setOnKeyListener(mTextKeyListener);
313818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        // Inform any listener of focus changes
314818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        mQueryTextView.setOnFocusChangeListener(new OnFocusChangeListener() {
315818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
316818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            public void onFocusChange(View v, boolean hasFocus) {
317818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                if (mOnQueryTextFocusChangeListener != null) {
318818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                    mOnQueryTextFocusChangeListener.onFocusChange(SearchView.this, hasFocus);
319818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                }
320818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            }
321818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        });
322818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        setIconifiedByDefault(a.getBoolean(R.styleable.SearchView_iconifiedByDefault, true));
3231b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes
3241b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes        final int maxWidth = a.getDimensionPixelSize(R.styleable.SearchView_android_maxWidth, -1);
325818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        if (maxWidth != -1) {
326818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            setMaxWidth(maxWidth);
327818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        }
3281b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes        final CharSequence queryHint = a.getText(R.styleable.SearchView_queryHint);
329818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        if (!TextUtils.isEmpty(queryHint)) {
330818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            setQueryHint(queryHint);
331818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        }
3321b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes        final int imeOptions = a.getInt(R.styleable.SearchView_android_imeOptions, -1);
333818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        if (imeOptions != -1) {
334818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            setImeOptions(imeOptions);
335818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        }
3361b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes        final int inputType = a.getInt(R.styleable.SearchView_android_inputType, -1);
337818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        if (inputType != -1) {
338818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            setInputType(inputType);
339818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        }
340818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
341818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        a.recycle();
342818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
343818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        // Save voice intent for later queries/launching
344818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        mVoiceWebSearchIntent = new Intent(RecognizerIntent.ACTION_WEB_SEARCH);
345818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        mVoiceWebSearchIntent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
346818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        mVoiceWebSearchIntent.putExtra(RecognizerIntent.EXTRA_LANGUAGE_MODEL,
347818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                RecognizerIntent.LANGUAGE_MODEL_WEB_SEARCH);
348818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
349818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        mVoiceAppSearchIntent = new Intent(RecognizerIntent.ACTION_RECOGNIZE_SPEECH);
350818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        mVoiceAppSearchIntent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
351818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
352818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        mDropDownAnchor = findViewById(mQueryTextView.getDropDownAnchor());
353818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        if (mDropDownAnchor != null) {
354818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.HONEYCOMB) {
355818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                addOnLayoutChangeListenerToDropDownAnchorSDK11();
356818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            } else {
357818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                addOnLayoutChangeListenerToDropDownAnchorBase();
358818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            }
359818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        }
360818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
361818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        updateViewsVisibility(mIconifiedByDefault);
362818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        updateQueryHint();
363818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    }
364818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
3651b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes    @TargetApi(Build.VERSION_CODES.HONEYCOMB)
366818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    private void addOnLayoutChangeListenerToDropDownAnchorSDK11() {
367818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        mDropDownAnchor.addOnLayoutChangeListener(new View.OnLayoutChangeListener() {
368818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            @Override
369818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            public void onLayoutChange(View v, int left, int top, int right, int bottom,
370818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                    int oldLeft, int oldTop, int oldRight, int oldBottom) {
371818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                adjustDropDownSizeAndPosition();
372818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            }
373818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        });
374818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    }
375818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
376818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    private void addOnLayoutChangeListenerToDropDownAnchorBase() {
377818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        mDropDownAnchor.getViewTreeObserver()
378818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                .addOnGlobalLayoutListener(new ViewTreeObserver.OnGlobalLayoutListener() {
379818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                    @Override
380818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                    public void onGlobalLayout() {
381818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                        adjustDropDownSizeAndPosition();
382818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                    }
383818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                });
384818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    }
385818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
3861b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes    int getSuggestionRowLayout() {
3871b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes        return mSuggestionRowLayout;
3881b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes    }
3891b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes
3901b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes    int getSuggestionCommitIconResId() {
3911b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes        return mSuggestionCommitIconResId;
3921b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes    }
3931b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes
394818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    /**
395818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * Sets the SearchableInfo for this SearchView. Properties in the SearchableInfo are used
396818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * to display labels, hints, suggestions, create intents for launching search results screens
397818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * and controlling other affordances such as a voice button.
398818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     *
399818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * @param searchable a SearchableInfo can be retrieved from the SearchManager, for a specific
400818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * activity or a global search provider.
401818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     */
402818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    public void setSearchableInfo(SearchableInfo searchable) {
403818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        mSearchable = searchable;
404818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        if (mSearchable != null) {
4051b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes            if (IS_AT_LEAST_FROYO) {
4061b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes                updateSearchAutoComplete();
4071b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes            }
408818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            updateQueryHint();
409818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        }
410818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        // Cache the voice search capability
4111b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes        mVoiceButtonEnabled = IS_AT_LEAST_FROYO && hasVoiceSearch();
412818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
413818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        if (mVoiceButtonEnabled) {
414818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            // Disable the microphone on the keyboard, as a mic is displayed near the text box
415818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            // TODO: use imeOptions to disable voice input when the new API will be available
416818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            mQueryTextView.setPrivateImeOptions(IME_OPTION_NO_MICROPHONE);
417818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        }
418818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        updateViewsVisibility(isIconified());
419818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    }
420818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
421818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    /**
422818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * Sets the APP_DATA for legacy SearchDialog use.
423818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * @param appSearchData bundle provided by the app when launching the search dialog
424818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * @hide
425818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     */
426818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    public void setAppSearchData(Bundle appSearchData) {
427818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        mAppSearchData = appSearchData;
428818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    }
429818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
430818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    /**
431818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * Sets the IME options on the query text field.
432818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     *
433818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * @see TextView#setImeOptions(int)
434818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * @param imeOptions the options to set on the query text field
435818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     *
436818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * @attr ref android.R.styleable#SearchView_imeOptions
437818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     */
438818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    public void setImeOptions(int imeOptions) {
439818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        mQueryTextView.setImeOptions(imeOptions);
440818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    }
441818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
442818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    /**
443818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * Returns the IME options set on the query text field.
444818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * @return the ime options
445818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * @see TextView#setImeOptions(int)
446818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     *
447818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * @attr ref android.R.styleable#SearchView_imeOptions
448818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     */
449818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    public int getImeOptions() {
450818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        return mQueryTextView.getImeOptions();
451818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    }
452818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
453818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    /**
454818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * Sets the input type on the query text field.
455818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     *
456818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * @see TextView#setInputType(int)
457818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * @param inputType the input type to set on the query text field
458818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     *
459818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * @attr ref android.R.styleable#SearchView_inputType
460818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     */
461818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    public void setInputType(int inputType) {
462818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        mQueryTextView.setInputType(inputType);
463818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    }
464818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
465818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    /**
466818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * Returns the input type set on the query text field.
467818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * @return the input type
468818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     *
469818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * @attr ref android.R.styleable#SearchView_inputType
470818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     */
471818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    public int getInputType() {
472818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        return mQueryTextView.getInputType();
473818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    }
474818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
475818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    /** @hide */
476818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    @Override
477818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    public boolean requestFocus(int direction, Rect previouslyFocusedRect) {
478818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        // Don't accept focus if in the middle of clearing focus
479818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        if (mClearingFocus) return false;
480818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        // Check if SearchView is focusable.
481818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        if (!isFocusable()) return false;
482818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        // If it is not iconified, then give the focus to the text field
483818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        if (!isIconified()) {
484818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            boolean result = mQueryTextView.requestFocus(direction, previouslyFocusedRect);
485818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            if (result) {
486818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                updateViewsVisibility(false);
487818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            }
488818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            return result;
489818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        } else {
490818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            return super.requestFocus(direction, previouslyFocusedRect);
491818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        }
492818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    }
493818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
494818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    /** @hide */
495818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    @Override
496818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    public void clearFocus() {
497818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        mClearingFocus = true;
498818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        setImeVisibility(false);
499818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        super.clearFocus();
500818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        mQueryTextView.clearFocus();
501818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        mClearingFocus = false;
502818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    }
503818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
504818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    /**
505818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * Sets a listener for user actions within the SearchView.
506818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     *
507818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * @param listener the listener object that receives callbacks when the user performs
508818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * actions in the SearchView such as clicking on buttons or typing a query.
509818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     */
510818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    public void setOnQueryTextListener(OnQueryTextListener listener) {
511818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        mOnQueryChangeListener = listener;
512818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    }
513818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
514818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    /**
515818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * Sets a listener to inform when the user closes the SearchView.
516818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     *
517818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * @param listener the listener to call when the user closes the SearchView.
518818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     */
519818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    public void setOnCloseListener(OnCloseListener listener) {
520818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        mOnCloseListener = listener;
521818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    }
522818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
523818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    /**
524818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * Sets a listener to inform when the focus of the query text field changes.
525818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     *
526818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * @param listener the listener to inform of focus changes.
527818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     */
528818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    public void setOnQueryTextFocusChangeListener(OnFocusChangeListener listener) {
529818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        mOnQueryTextFocusChangeListener = listener;
530818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    }
531818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
532818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    /**
533818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * Sets a listener to inform when a suggestion is focused or clicked.
534818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     *
535818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * @param listener the listener to inform of suggestion selection events.
536818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     */
537818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    public void setOnSuggestionListener(OnSuggestionListener listener) {
538818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        mOnSuggestionListener = listener;
539818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    }
540818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
541818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    /**
542818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * Sets a listener to inform when the search button is pressed. This is only
543818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * relevant when the text field is not visible by default. Calling {@link #setIconified
544818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * setIconified(false)} can also cause this listener to be informed.
545818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     *
546818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * @param listener the listener to inform when the search button is clicked or
547818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * the text field is programmatically de-iconified.
548818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     */
549818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    public void setOnSearchClickListener(OnClickListener listener) {
550818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        mOnSearchClickListener = listener;
551818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    }
552818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
553818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    /**
554818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * Returns the query string currently in the text field.
555818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     *
556818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * @return the query string
557818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     */
558818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    public CharSequence getQuery() {
559818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        return mQueryTextView.getText();
560818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    }
561818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
562818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    /**
563818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * Sets a query string in the text field and optionally submits the query as well.
564818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     *
565818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * @param query the query string. This replaces any query text already present in the
566818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * text field.
567818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * @param submit whether to submit the query right now or only update the contents of
568818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * text field.
569818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     */
570818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    public void setQuery(CharSequence query, boolean submit) {
571818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        mQueryTextView.setText(query);
572818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        if (query != null) {
573818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            mQueryTextView.setSelection(mQueryTextView.length());
574818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            mUserQuery = query;
575818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        }
576818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
577818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        // If the query is not empty and submit is requested, submit the query
578818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        if (submit && !TextUtils.isEmpty(query)) {
579818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            onSubmitQuery();
580818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        }
581818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    }
582818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
583818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    /**
584818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * Sets the hint text to display in the query text field. This overrides any hint specified
585818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * in the SearchableInfo.
586818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     *
587818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * @param hint the hint text to display
588818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     *
589818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * @attr ref android.R.styleable#SearchView_queryHint
590818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     */
591818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    public void setQueryHint(CharSequence hint) {
592818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        mQueryHint = hint;
593818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        updateQueryHint();
594818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    }
595818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
596818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    /**
597818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * Gets the hint text to display in the query text field.
598818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * @return the query hint text, if specified, null otherwise.
599818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     *
600818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * @attr ref android.R.styleable#SearchView_queryHint
601818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     */
602818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    public CharSequence getQueryHint() {
603818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        if (mQueryHint != null) {
604818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            return mQueryHint;
6051b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes        } else if (IS_AT_LEAST_FROYO && mSearchable != null) {
606818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            CharSequence hint = null;
607818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            int hintId = mSearchable.getHintId();
608818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            if (hintId != 0) {
609818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                hint = getContext().getString(hintId);
610818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            }
611818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            return hint;
612818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        }
613818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        return null;
614818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    }
615818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
616818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    /**
617818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * Sets the default or resting state of the search field. If true, a single search icon is
618818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * shown by default and expands to show the text field and other buttons when pressed. Also,
619818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * if the default state is iconified, then it collapses to that state when the close button
620818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * is pressed. Changes to this property will take effect immediately.
621818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     *
622818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * <p>The default value is true.</p>
623818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     *
624818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * @param iconified whether the search field should be iconified by default
625818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     *
626818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * @attr ref android.R.styleable#SearchView_iconifiedByDefault
627818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     */
628818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    public void setIconifiedByDefault(boolean iconified) {
629818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        if (mIconifiedByDefault == iconified) return;
630818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        mIconifiedByDefault = iconified;
631818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        updateViewsVisibility(iconified);
632818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        updateQueryHint();
633818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    }
634818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
635818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    /**
636818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * Returns the default iconified state of the search field.
637818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * @return
638818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     *
639818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * @attr ref android.R.styleable#SearchView_iconifiedByDefault
640818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     */
641818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    public boolean isIconfiedByDefault() {
642818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        return mIconifiedByDefault;
643818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    }
644818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
645818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    /**
646818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * Iconifies or expands the SearchView. Any query text is cleared when iconified. This is
647818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * a temporary state and does not override the default iconified state set by
648818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * {@link #setIconifiedByDefault(boolean)}. If the default state is iconified, then
649818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * a false here will only be valid until the user closes the field. And if the default
650818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * state is expanded, then a true here will only clear the text field and not close it.
651818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     *
652818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * @param iconify a true value will collapse the SearchView to an icon, while a false will
653818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * expand it.
654818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     */
655818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    public void setIconified(boolean iconify) {
656818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        if (iconify) {
657818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            onCloseClicked();
658818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        } else {
659818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            onSearchClicked();
660818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        }
661818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    }
662818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
663818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    /**
664818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * Returns the current iconified state of the SearchView.
665818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     *
666818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * @return true if the SearchView is currently iconified, false if the search field is
667818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * fully visible.
668818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     */
669818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    public boolean isIconified() {
670818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        return mIconified;
671818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    }
672818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
673818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    /**
674818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * Enables showing a submit button when the query is non-empty. In cases where the SearchView
675818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * is being used to filter the contents of the current activity and doesn't launch a separate
676818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * results activity, then the submit button should be disabled.
677818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     *
678818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * @param enabled true to show a submit button for submitting queries, false if a submit
679818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * button is not required.
680818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     */
681818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    public void setSubmitButtonEnabled(boolean enabled) {
682818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        mSubmitButtonEnabled = enabled;
683818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        updateViewsVisibility(isIconified());
684818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    }
685818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
686818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    /**
687818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * Returns whether the submit button is enabled when necessary or never displayed.
688818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     *
689818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * @return whether the submit button is enabled automatically when necessary
690818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     */
691818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    public boolean isSubmitButtonEnabled() {
692818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        return mSubmitButtonEnabled;
693818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    }
694818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
695818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    /**
696818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * Specifies if a query refinement button should be displayed alongside each suggestion
697818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * or if it should depend on the flags set in the individual items retrieved from the
698818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * suggestions provider. Clicking on the query refinement button will replace the text
699818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * in the query text field with the text from the suggestion. This flag only takes effect
700818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * if a SearchableInfo has been specified with {@link #setSearchableInfo(SearchableInfo)}
701818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * and not when using a custom adapter.
702818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     *
703818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * @param enable true if all items should have a query refinement button, false if only
704818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * those items that have a query refinement flag set should have the button.
705818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     *
706818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * @see SearchManager#SUGGEST_COLUMN_FLAGS
707818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * @see SearchManager#FLAG_QUERY_REFINEMENT
708818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     */
709818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    public void setQueryRefinementEnabled(boolean enable) {
710818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        mQueryRefinement = enable;
711818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        if (mSuggestionsAdapter instanceof SuggestionsAdapter) {
712818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            ((SuggestionsAdapter) mSuggestionsAdapter).setQueryRefinement(
713818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                    enable ? SuggestionsAdapter.REFINE_ALL : SuggestionsAdapter.REFINE_BY_ENTRY);
714818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        }
715818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    }
716818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
717818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    /**
718818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * Returns whether query refinement is enabled for all items or only specific ones.
719818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * @return true if enabled for all items, false otherwise.
720818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     */
721818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    public boolean isQueryRefinementEnabled() {
722818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        return mQueryRefinement;
723818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    }
724818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
725818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    /**
726818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * You can set a custom adapter if you wish. Otherwise the default adapter is used to
727818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * display the suggestions from the suggestions provider associated with the SearchableInfo.
728818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     *
729818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * @see #setSearchableInfo(SearchableInfo)
730818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     */
731818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    public void setSuggestionsAdapter(CursorAdapter adapter) {
732818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        mSuggestionsAdapter = adapter;
733818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
734818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        mQueryTextView.setAdapter(mSuggestionsAdapter);
735818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    }
736818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
737818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    /**
738818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * Returns the adapter used for suggestions, if any.
739818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * @return the suggestions adapter
740818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     */
741818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    public CursorAdapter getSuggestionsAdapter() {
742818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        return mSuggestionsAdapter;
743818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    }
744818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
745818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    /**
746818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * Makes the view at most this many pixels wide
747818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     *
748818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * @attr ref android.R.styleable#SearchView_maxWidth
749818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     */
750818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    public void setMaxWidth(int maxpixels) {
751818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        mMaxWidth = maxpixels;
752818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
753818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        requestLayout();
754818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    }
755818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
756818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    /**
757818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * Gets the specified maximum width in pixels, if set. Returns zero if
758818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * no maximum width was specified.
759818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * @return the maximum width of the view
760818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     *
761818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * @attr ref android.R.styleable#SearchView_maxWidth
762818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     */
763818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    public int getMaxWidth() {
764818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        return mMaxWidth;
765818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    }
766818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
767818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    @Override
768818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
769818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        // Let the standard measurements take effect in iconified state.
770818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        if (isIconified()) {
771818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            super.onMeasure(widthMeasureSpec, heightMeasureSpec);
772818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            return;
773818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        }
774818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
775818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        int widthMode = MeasureSpec.getMode(widthMeasureSpec);
776818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        int width = MeasureSpec.getSize(widthMeasureSpec);
777818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
778818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        switch (widthMode) {
779818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            case MeasureSpec.AT_MOST:
780818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                // If there is an upper limit, don't exceed maximum width (explicit or implicit)
781818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                if (mMaxWidth > 0) {
782818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                    width = Math.min(mMaxWidth, width);
783818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                } else {
784818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                    width = Math.min(getPreferredWidth(), width);
785818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                }
786818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                break;
787818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            case MeasureSpec.EXACTLY:
788818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                // If an exact width is specified, still don't exceed any specified maximum width
789818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                if (mMaxWidth > 0) {
790818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                    width = Math.min(mMaxWidth, width);
791818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                }
792818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                break;
793818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            case MeasureSpec.UNSPECIFIED:
794818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                // Use maximum width, if specified, else preferred width
795818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                width = mMaxWidth > 0 ? mMaxWidth : getPreferredWidth();
796818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                break;
797818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        }
798818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        widthMode = MeasureSpec.EXACTLY;
799818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        super.onMeasure(MeasureSpec.makeMeasureSpec(width, widthMode), heightMeasureSpec);
800818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    }
801818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
802818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    private int getPreferredWidth() {
803818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        return getContext().getResources()
80407a07ce59efb770e9fb9ca53a0133e5e64a63bbcChris Banes                .getDimensionPixelSize(R.dimen.abc_search_view_preferred_width);
805818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    }
806818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
807818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    private void updateViewsVisibility(final boolean collapsed) {
808818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        mIconified = collapsed;
809818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        // Visibility of views that are visible when collapsed
810818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        final int visCollapsed = collapsed ? VISIBLE : GONE;
811818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        // Is there text in the query
812818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        final boolean hasText = !TextUtils.isEmpty(mQueryTextView.getText());
813818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
814818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        mSearchButton.setVisibility(visCollapsed);
815818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        updateSubmitButton(hasText);
816818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        mSearchEditFrame.setVisibility(collapsed ? GONE : VISIBLE);
817818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        mSearchHintIcon.setVisibility(mIconifiedByDefault ? GONE : VISIBLE);
818818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        updateCloseButton();
819818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        updateVoiceButton(!hasText);
820818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        updateSubmitArea();
821818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    }
822818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
8231b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes    @TargetApi(Build.VERSION_CODES.FROYO)
824818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    private boolean hasVoiceSearch() {
8251b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes        if (mSearchable != null &&
8261b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes                mSearchable.getVoiceSearchEnabled()) {
827818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            Intent testIntent = null;
828818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            if (mSearchable.getVoiceSearchLaunchWebSearch()) {
829818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                testIntent = mVoiceWebSearchIntent;
830818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            } else if (mSearchable.getVoiceSearchLaunchRecognizer()) {
831818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                testIntent = mVoiceAppSearchIntent;
832818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            }
833818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            if (testIntent != null) {
834818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                ResolveInfo ri = getContext().getPackageManager().resolveActivity(testIntent,
835818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                        PackageManager.MATCH_DEFAULT_ONLY);
836818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                return ri != null;
837818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            }
838818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        }
839818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        return false;
840818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    }
841818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
842818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    private boolean isSubmitAreaEnabled() {
843818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        return (mSubmitButtonEnabled || mVoiceButtonEnabled) && !isIconified();
844818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    }
845818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
846818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    private void updateSubmitButton(boolean hasText) {
847818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        int visibility = GONE;
848818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        if (mSubmitButtonEnabled && isSubmitAreaEnabled() && hasFocus()
849818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                && (hasText || !mVoiceButtonEnabled)) {
850818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            visibility = VISIBLE;
851818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        }
852818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        mSubmitButton.setVisibility(visibility);
853818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    }
854818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
855818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    private void updateSubmitArea() {
856818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        int visibility = GONE;
857818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        if (isSubmitAreaEnabled()
858818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                && (mSubmitButton.getVisibility() == VISIBLE
859818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                || mVoiceButton.getVisibility() == VISIBLE)) {
860818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            visibility = VISIBLE;
861818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        }
862818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        mSubmitArea.setVisibility(visibility);
863818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    }
864818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
865818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    private void updateCloseButton() {
866818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        final boolean hasText = !TextUtils.isEmpty(mQueryTextView.getText());
867818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        // Should we show the close button? It is not shown if there's no focus,
868818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        // field is not iconified by default and there is no text in it.
869818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        final boolean showClose = hasText || (mIconifiedByDefault && !mExpandedInActionView);
870818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        mCloseButton.setVisibility(showClose ? VISIBLE : GONE);
871818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        mCloseButton.getDrawable().setState(hasText ? ENABLED_STATE_SET : EMPTY_STATE_SET);
872818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    }
873818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
874818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    private void postUpdateFocusedState() {
875818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        post(mUpdateDrawableStateRunnable);
876818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    }
877818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
878818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    private void updateFocusedState() {
879818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        boolean focused = mQueryTextView.hasFocus();
8801b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes        mSearchPlate.getBackground().setState(focused ? ENABLED_FOCUSED_STATE_SET : EMPTY_STATE_SET);
8811b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes        mSubmitArea.getBackground().setState(focused ? ENABLED_FOCUSED_STATE_SET : EMPTY_STATE_SET);
882818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        invalidate();
883818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    }
884818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
885818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    @Override
886818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    protected void onDetachedFromWindow() {
887818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        removeCallbacks(mUpdateDrawableStateRunnable);
888818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        post(mReleaseCursorRunnable);
889818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        super.onDetachedFromWindow();
890818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    }
891818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
892818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    private void setImeVisibility(final boolean visible) {
893818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        if (visible) {
894818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            post(mShowImeRunnable);
895818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        } else {
896818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            removeCallbacks(mShowImeRunnable);
897818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            InputMethodManager imm = (InputMethodManager)
898818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                    getContext().getSystemService(Context.INPUT_METHOD_SERVICE);
899818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
900818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            if (imm != null) {
901818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                imm.hideSoftInputFromWindow(getWindowToken(), 0);
902818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            }
903818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        }
904818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    }
905818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
906818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    /**
907818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * Called by the SuggestionsAdapter
908818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * @hide
909818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     */
910818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    /* package */void onQueryRefine(CharSequence queryText) {
911818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        setQuery(queryText);
912818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    }
913818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
914818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    private final OnClickListener mOnClickListener = new OnClickListener() {
915818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
916818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        public void onClick(View v) {
917818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            if (v == mSearchButton) {
918818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                onSearchClicked();
919818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            } else if (v == mCloseButton) {
920818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                onCloseClicked();
921818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            } else if (v == mSubmitButton) {
922818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                onSubmitQuery();
923818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            } else if (v == mVoiceButton) {
9241b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes                if (IS_AT_LEAST_FROYO) {
9251b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes                    onVoiceClicked();
9261b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes                }
927818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            } else if (v == mQueryTextView) {
928818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                forceSuggestionQuery();
929818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            }
930818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        }
931818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    };
932818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
933818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    /**
934818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * React to the user typing "enter" or other hardwired keys while typing in
935818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * the search box. This handles these special keys while the edit box has
936818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * focus.
937818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     */
938818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    View.OnKeyListener mTextKeyListener = new View.OnKeyListener() {
939818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        public boolean onKey(View v, int keyCode, KeyEvent event) {
940818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            // guard against possible race conditions
941818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            if (mSearchable == null) {
942818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                return false;
943818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            }
944818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
945818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            if (DBG) {
946818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                Log.d(LOG_TAG, "mTextListener.onKey(" + keyCode + "," + event + "), selection: "
947818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                        + mQueryTextView.getListSelection());
948818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            }
949818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
950818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            // If a suggestion is selected, handle enter, search key, and action keys
951818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            // as presses on the selected suggestion
952818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            if (mQueryTextView.isPopupShowing()
953818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                    && mQueryTextView.getListSelection() != ListView.INVALID_POSITION) {
954818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                return onSuggestionsKey(v, keyCode, event);
955818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            }
956818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
957818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            // If there is text in the query box, handle enter, and action keys
958818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            // The search key is handled by the dialog's onKeyDown().
959818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            if (!mQueryTextView.isEmpty() && KeyEventCompat.hasNoModifiers(event)) {
960818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                if (event.getAction() == KeyEvent.ACTION_UP) {
961818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                    if (keyCode == KeyEvent.KEYCODE_ENTER) {
962818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                        v.cancelLongPress();
963818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
964818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                        // Launch as a regular search.
965818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                        launchQuerySearch(KeyEvent.KEYCODE_UNKNOWN, null, mQueryTextView.getText()
966818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                                .toString());
967818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                        return true;
968818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                    }
969818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                }
970818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            }
971818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            return false;
972818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        }
973818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    };
974818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
975818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    /**
976818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * React to the user typing while in the suggestions list. First, check for
977818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * action keys. If not handled, try refocusing regular characters into the
978818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * EditText.
979818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     */
980818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    private boolean onSuggestionsKey(View v, int keyCode, KeyEvent event) {
981818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        // guard against possible race conditions (late arrival after dismiss)
982818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        if (mSearchable == null) {
983818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            return false;
984818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        }
985818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        if (mSuggestionsAdapter == null) {
986818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            return false;
987818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        }
988818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        if (event.getAction() == KeyEvent.ACTION_DOWN && KeyEventCompat.hasNoModifiers(event)) {
989818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            // First, check for enter or search (both of which we'll treat as a
990818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            // "click")
991818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            if (keyCode == KeyEvent.KEYCODE_ENTER || keyCode == KeyEvent.KEYCODE_SEARCH
992818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                    || keyCode == KeyEvent.KEYCODE_TAB) {
993818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                int position = mQueryTextView.getListSelection();
994818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                return onItemClicked(position, KeyEvent.KEYCODE_UNKNOWN, null);
995818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            }
996818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
997818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            // Next, check for left/right moves, which we use to "return" the
998818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            // user to the edit view
999818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            if (keyCode == KeyEvent.KEYCODE_DPAD_LEFT || keyCode == KeyEvent.KEYCODE_DPAD_RIGHT) {
1000818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                // give "focus" to text editor, with cursor at the beginning if
1001818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                // left key, at end if right key
10021b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes                // TODO: Reverse left/right for right-to-left languages, e.g.
10031b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes                // Arabic
1004818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                int selPoint = (keyCode == KeyEvent.KEYCODE_DPAD_LEFT) ? 0 : mQueryTextView
1005818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                        .length();
1006818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                mQueryTextView.setSelection(selPoint);
1007818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                mQueryTextView.setListSelection(0);
1008818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                mQueryTextView.clearListSelection();
1009818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                HIDDEN_METHOD_INVOKER.ensureImeVisible(mQueryTextView, true);
1010818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
1011818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                return true;
1012818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            }
1013818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
1014818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            // Next, check for an "up and out" move
1015818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            if (keyCode == KeyEvent.KEYCODE_DPAD_UP && 0 == mQueryTextView.getListSelection()) {
1016818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                // TODO: restoreUserQuery();
1017818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                // let ACTV complete the move
1018818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                return false;
1019818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            }
1020818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        }
1021818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        return false;
1022818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    }
1023818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
1024818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    private CharSequence getDecoratedHint(CharSequence hintText) {
1025818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        // If the field is always expanded, then don't add the search icon to the hint
10261b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes        if (!mIconifiedByDefault) {
10271b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes            return hintText;
10281b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes        }
1029818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
10301b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes        final Drawable searchIcon = getResources().getDrawable(mSearchIconResId);
10311b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes        final int textSize = (int) (mQueryTextView.getTextSize() * 1.25);
1032818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        searchIcon.setBounds(0, 0, textSize, textSize);
10331b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes
10341b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes        final SpannableStringBuilder ssb = new SpannableStringBuilder("   "); // for the icon
10351b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes        ssb.append(hintText);
1036818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        ssb.setSpan(new ImageSpan(searchIcon), 1, 2, Spannable.SPAN_EXCLUSIVE_EXCLUSIVE);
1037818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        return ssb;
1038818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    }
1039818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
1040818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    private void updateQueryHint() {
1041818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        if (mQueryHint != null) {
1042818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            mQueryTextView.setHint(getDecoratedHint(mQueryHint));
10431b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes        } else if (IS_AT_LEAST_FROYO && mSearchable != null) {
1044818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            CharSequence hint = null;
1045818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            int hintId = mSearchable.getHintId();
1046818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            if (hintId != 0) {
1047818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                hint = getContext().getString(hintId);
1048818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            }
1049818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            if (hint != null) {
1050818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                mQueryTextView.setHint(getDecoratedHint(hint));
1051818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            }
1052818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        } else {
1053818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            mQueryTextView.setHint(getDecoratedHint(""));
1054818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        }
1055818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    }
1056818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
1057818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    /**
1058818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * Updates the auto-complete text view.
1059818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     */
10601b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes    @TargetApi(Build.VERSION_CODES.FROYO)
1061818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    private void updateSearchAutoComplete() {
1062818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        mQueryTextView.setThreshold(mSearchable.getSuggestThreshold());
1063818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        mQueryTextView.setImeOptions(mSearchable.getImeOptions());
1064818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        int inputType = mSearchable.getInputType();
1065818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        // We only touch this if the input type is set up for text (which it almost certainly
1066818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        // should be, in the case of search!)
1067818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        if ((inputType & InputType.TYPE_MASK_CLASS) == InputType.TYPE_CLASS_TEXT) {
1068818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            // The existence of a suggestions authority is the proxy for "suggestions
1069818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            // are available here"
1070818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            inputType &= ~InputType.TYPE_TEXT_FLAG_AUTO_COMPLETE;
1071818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            if (mSearchable.getSuggestAuthority() != null) {
1072818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                inputType |= InputType.TYPE_TEXT_FLAG_AUTO_COMPLETE;
1073818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                // TYPE_TEXT_FLAG_AUTO_COMPLETE means that the text editor is performing
1074818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                // auto-completion based on its own semantics, which it will present to the user
1075818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                // as they type. This generally means that the input method should not show its
1076818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                // own candidates, and the spell checker should not be in action. The text editor
1077818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                // supplies its candidates by calling InputMethodManager.displayCompletions(),
1078818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                // which in turn will call InputMethodSession.displayCompletions().
1079818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                inputType |= InputType.TYPE_TEXT_FLAG_NO_SUGGESTIONS;
1080818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            }
1081818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        }
1082818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        mQueryTextView.setInputType(inputType);
1083818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        if (mSuggestionsAdapter != null) {
1084818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            mSuggestionsAdapter.changeCursor(null);
1085818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        }
1086818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        // attach the suggestions adapter, if suggestions are available
1087818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        // The existence of a suggestions authority is the proxy for "suggestions available here"
1088818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        if (mSearchable.getSuggestAuthority() != null) {
1089818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            mSuggestionsAdapter = new SuggestionsAdapter(getContext(),
1090818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                    this, mSearchable, mOutsideDrawablesCache);
1091818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            mQueryTextView.setAdapter(mSuggestionsAdapter);
1092818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            ((SuggestionsAdapter) mSuggestionsAdapter).setQueryRefinement(
1093818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                    mQueryRefinement ? SuggestionsAdapter.REFINE_ALL
1094818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                            : SuggestionsAdapter.REFINE_BY_ENTRY);
1095818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        }
1096818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    }
1097818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
1098818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    /**
1099818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * Update the visibility of the voice button.  There are actually two voice search modes,
1100818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * either of which will activate the button.
1101818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * @param empty whether the search query text field is empty. If it is, then the other
1102818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * criteria apply to make the voice button visible.
1103818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     */
1104818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    private void updateVoiceButton(boolean empty) {
1105818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        int visibility = GONE;
1106818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        if (mVoiceButtonEnabled && !isIconified() && empty) {
1107818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            visibility = VISIBLE;
1108818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            mSubmitButton.setVisibility(GONE);
1109818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        }
1110818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        mVoiceButton.setVisibility(visibility);
1111818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    }
1112818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
1113818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    private final OnEditorActionListener mOnEditorActionListener = new OnEditorActionListener() {
1114818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
1115818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        /**
1116818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes         * Called when the input method default action key is pressed.
1117818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes         */
1118818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        public boolean onEditorAction(TextView v, int actionId, KeyEvent event) {
1119818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            onSubmitQuery();
1120818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            return true;
1121818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        }
1122818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    };
1123818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
1124818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    private void onTextChanged(CharSequence newText) {
1125818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        CharSequence text = mQueryTextView.getText();
1126818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        mUserQuery = text;
1127818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        boolean hasText = !TextUtils.isEmpty(text);
1128818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        updateSubmitButton(hasText);
1129818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        updateVoiceButton(!hasText);
1130818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        updateCloseButton();
1131818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        updateSubmitArea();
1132818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        if (mOnQueryChangeListener != null && !TextUtils.equals(newText, mOldQueryText)) {
1133818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            mOnQueryChangeListener.onQueryTextChange(newText.toString());
1134818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        }
1135818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        mOldQueryText = newText.toString();
1136818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    }
1137818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
1138818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    private void onSubmitQuery() {
1139818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        CharSequence query = mQueryTextView.getText();
1140818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        if (query != null && TextUtils.getTrimmedLength(query) > 0) {
1141818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            if (mOnQueryChangeListener == null
1142818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                    || !mOnQueryChangeListener.onQueryTextSubmit(query.toString())) {
1143818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                if (mSearchable != null) {
1144818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                    launchQuerySearch(KeyEvent.KEYCODE_UNKNOWN, null, query.toString());
1145818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                }
11461b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes                setImeVisibility(false);
1147818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                dismissSuggestions();
1148818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            }
1149818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        }
1150818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    }
1151818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
1152818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    private void dismissSuggestions() {
1153818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        mQueryTextView.dismissDropDown();
1154818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    }
1155818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
1156818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    private void onCloseClicked() {
1157818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        CharSequence text = mQueryTextView.getText();
1158818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        if (TextUtils.isEmpty(text)) {
1159818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            if (mIconifiedByDefault) {
1160818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                // If the app doesn't override the close behavior
1161818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                if (mOnCloseListener == null || !mOnCloseListener.onClose()) {
1162818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                    // hide the keyboard and remove focus
1163818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                    clearFocus();
1164818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                    // collapse the search field
1165818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                    updateViewsVisibility(true);
1166818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                }
1167818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            }
1168818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        } else {
1169818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            mQueryTextView.setText("");
1170818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            mQueryTextView.requestFocus();
1171818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            setImeVisibility(true);
1172818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        }
1173818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
1174818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    }
1175818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
1176818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    private void onSearchClicked() {
1177818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        updateViewsVisibility(false);
1178818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        mQueryTextView.requestFocus();
1179818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        setImeVisibility(true);
1180818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        if (mOnSearchClickListener != null) {
1181818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            mOnSearchClickListener.onClick(this);
1182818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        }
1183818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    }
1184818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
11851b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes    @TargetApi(Build.VERSION_CODES.FROYO)
1186818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    private void onVoiceClicked() {
1187818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        // guard against possible race conditions
1188818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        if (mSearchable == null) {
1189818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            return;
1190818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        }
1191818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        SearchableInfo searchable = mSearchable;
1192818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        try {
1193818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            if (searchable.getVoiceSearchLaunchWebSearch()) {
1194818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                Intent webSearchIntent = createVoiceWebSearchIntent(mVoiceWebSearchIntent,
1195818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                        searchable);
1196818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                getContext().startActivity(webSearchIntent);
1197818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            } else if (searchable.getVoiceSearchLaunchRecognizer()) {
1198818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                Intent appSearchIntent = createVoiceAppSearchIntent(mVoiceAppSearchIntent,
1199818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                        searchable);
1200818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                getContext().startActivity(appSearchIntent);
1201818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            }
1202818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        } catch (ActivityNotFoundException e) {
1203818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            // Should not happen, since we check the availability of
1204818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            // voice search before showing the button. But just in case...
1205818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            Log.w(LOG_TAG, "Could not find voice search activity");
1206818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        }
1207818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    }
1208818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
1209818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    void onTextFocusChanged() {
1210818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        updateViewsVisibility(isIconified());
1211818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        // Delayed update to make sure that the focus has settled down and window focus changes
1212818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        // don't affect it. A synchronous update was not working.
1213818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        postUpdateFocusedState();
1214818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        if (mQueryTextView.hasFocus()) {
1215818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            forceSuggestionQuery();
1216818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        }
1217818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    }
1218818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
1219818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    @Override
1220818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    public void onWindowFocusChanged(boolean hasWindowFocus) {
1221818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        super.onWindowFocusChanged(hasWindowFocus);
1222818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
1223818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        postUpdateFocusedState();
1224818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    }
1225818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
1226818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    /**
1227818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * {@inheritDoc}
1228818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     */
1229818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    @Override
1230818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    public void onActionViewCollapsed() {
12311b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes        setQuery("", false);
1232818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        clearFocus();
1233818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        updateViewsVisibility(true);
1234818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        mQueryTextView.setImeOptions(mCollapsedImeOptions);
1235818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        mExpandedInActionView = false;
1236818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    }
1237818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
1238818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    /**
1239818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * {@inheritDoc}
1240818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     */
1241818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    @Override
1242818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    public void onActionViewExpanded() {
1243818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        if (mExpandedInActionView) return;
1244818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
1245818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        mExpandedInActionView = true;
1246818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        mCollapsedImeOptions = mQueryTextView.getImeOptions();
1247818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        mQueryTextView.setImeOptions(mCollapsedImeOptions | EditorInfo.IME_FLAG_NO_FULLSCREEN);
1248818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        mQueryTextView.setText("");
1249818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        setIconified(false);
1250818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    }
1251818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
1252818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
1253818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    private void adjustDropDownSizeAndPosition() {
1254818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        if (mDropDownAnchor.getWidth() > 1) {
1255818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            Resources res = getContext().getResources();
1256818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            int anchorPadding = mSearchPlate.getPaddingLeft();
1257818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            Rect dropDownPadding = new Rect();
12581b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes            final boolean isLayoutRtl = ViewUtils.isLayoutRtl(this);
1259818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            int iconOffset = mIconifiedByDefault
126007a07ce59efb770e9fb9ca53a0133e5e64a63bbcChris Banes                    ? res.getDimensionPixelSize(R.dimen.abc_dropdownitem_icon_width)
126107a07ce59efb770e9fb9ca53a0133e5e64a63bbcChris Banes                    + res.getDimensionPixelSize(R.dimen.abc_dropdownitem_text_padding_left)
1262818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                    : 0;
1263818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            mQueryTextView.getDropDownBackground().getPadding(dropDownPadding);
12641b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes            int offset;
12651b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes            if (isLayoutRtl) {
12661b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes                offset = - dropDownPadding.left;
12671b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes            } else {
12681b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes                offset = anchorPadding - (dropDownPadding.left + iconOffset);
12691b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes            }
1270818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            mQueryTextView.setDropDownHorizontalOffset(offset);
1271818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            final int width = mDropDownAnchor.getWidth() + dropDownPadding.left
1272818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                    + dropDownPadding.right + iconOffset - anchorPadding;
1273818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            mQueryTextView.setDropDownWidth(width);
1274818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        }
1275818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    }
1276818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
1277818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    private boolean onItemClicked(int position, int actionKey, String actionMsg) {
1278818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        if (mOnSuggestionListener == null
1279818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                || !mOnSuggestionListener.onSuggestionClick(position)) {
1280818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            launchSuggestion(position, KeyEvent.KEYCODE_UNKNOWN, null);
1281818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            setImeVisibility(false);
1282818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            dismissSuggestions();
1283818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            return true;
1284818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        }
1285818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        return false;
1286818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    }
1287818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
1288818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    private boolean onItemSelected(int position) {
1289818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        if (mOnSuggestionListener == null
1290818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                || !mOnSuggestionListener.onSuggestionSelect(position)) {
1291818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            rewriteQueryFromSuggestion(position);
1292818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            return true;
1293818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        }
1294818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        return false;
1295818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    }
1296818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
1297818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    private final OnItemClickListener mOnItemClickListener = new OnItemClickListener() {
1298818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
1299818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        /**
1300818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes         * Implements OnItemClickListener
1301818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes         */
1302818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        public void onItemClick(AdapterView<?> parent, View view, int position, long id) {
1303818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            if (DBG) Log.d(LOG_TAG, "onItemClick() position " + position);
1304818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            onItemClicked(position, KeyEvent.KEYCODE_UNKNOWN, null);
1305818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        }
1306818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    };
1307818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
1308818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    private final OnItemSelectedListener mOnItemSelectedListener = new OnItemSelectedListener() {
1309818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
1310818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        /**
1311818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes         * Implements OnItemSelectedListener
1312818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes         */
1313818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        public void onItemSelected(AdapterView<?> parent, View view, int position, long id) {
1314818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            if (DBG) Log.d(LOG_TAG, "onItemSelected() position " + position);
1315818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            SearchView.this.onItemSelected(position);
1316818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        }
1317818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
1318818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        /**
1319818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes         * Implements OnItemSelectedListener
1320818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes         */
1321818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        public void onNothingSelected(AdapterView<?> parent) {
1322818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            if (DBG)
1323818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                Log.d(LOG_TAG, "onNothingSelected()");
1324818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        }
1325818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    };
1326818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
1327818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    /**
1328818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * Query rewriting.
1329818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     */
1330818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    private void rewriteQueryFromSuggestion(int position) {
1331818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        CharSequence oldQuery = mQueryTextView.getText();
1332818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        Cursor c = mSuggestionsAdapter.getCursor();
1333818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        if (c == null) {
1334818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            return;
1335818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        }
1336818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        if (c.moveToPosition(position)) {
1337818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            // Get the new query from the suggestion.
1338818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            CharSequence newQuery = mSuggestionsAdapter.convertToString(c);
1339818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            if (newQuery != null) {
1340818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                // The suggestion rewrites the query.
1341818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                // Update the text field, without getting new suggestions.
1342818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                setQuery(newQuery);
1343818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            } else {
1344818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                // The suggestion does not rewrite the query, restore the user's query.
1345818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                setQuery(oldQuery);
1346818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            }
1347818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        } else {
1348818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            // We got a bad position, restore the user's query.
1349818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            setQuery(oldQuery);
1350818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        }
1351818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    }
1352818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
1353818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    /**
1354818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * Launches an intent based on a suggestion.
1355818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     *
1356818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * @param position The index of the suggestion to create the intent from.
1357818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * @param actionKey The key code of the action key that was pressed,
1358818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     *        or {@link KeyEvent#KEYCODE_UNKNOWN} if none.
1359818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * @param actionMsg The message for the action key that was pressed,
1360818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     *        or <code>null</code> if none.
1361818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * @return true if a successful launch, false if could not (e.g. bad position).
1362818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     */
1363818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    private boolean launchSuggestion(int position, int actionKey, String actionMsg) {
1364818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        Cursor c = mSuggestionsAdapter.getCursor();
1365818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        if ((c != null) && c.moveToPosition(position)) {
1366818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
1367818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            Intent intent = createIntentFromSuggestion(c, actionKey, actionMsg);
1368818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
1369818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            // launch the intent
1370818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            launchIntent(intent);
1371818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
1372818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            return true;
1373818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        }
1374818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        return false;
1375818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    }
1376818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
1377818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    /**
1378818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * Launches an intent, including any special intent handling.
1379818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     */
1380818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    private void launchIntent(Intent intent) {
1381818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        if (intent == null) {
1382818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            return;
1383818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        }
1384818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        try {
1385818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            // If the intent was created from a suggestion, it will always have an explicit
1386818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            // component here.
1387818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            getContext().startActivity(intent);
1388818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        } catch (RuntimeException ex) {
1389818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            Log.e(LOG_TAG, "Failed launch activity: " + intent, ex);
1390818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        }
1391818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    }
1392818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
1393818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    /**
1394818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * Sets the text in the query box, without updating the suggestions.
1395818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     */
1396818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    private void setQuery(CharSequence query) {
1397818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        mQueryTextView.setText(query);
1398818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        // Move the cursor to the end
1399818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        mQueryTextView.setSelection(TextUtils.isEmpty(query) ? 0 : query.length());
1400818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    }
1401818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
1402818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    private void launchQuerySearch(int actionKey, String actionMsg, String query) {
1403818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        String action = Intent.ACTION_SEARCH;
1404818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        Intent intent = createIntent(action, null, null, query, actionKey, actionMsg);
1405818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        getContext().startActivity(intent);
1406818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    }
1407818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
1408818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    /**
1409818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * Constructs an intent from the given information and the search dialog state.
1410818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     *
1411818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * @param action Intent action.
1412818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * @param data Intent data, or <code>null</code>.
1413818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * @param extraData Data for {@link SearchManager#EXTRA_DATA_KEY} or <code>null</code>.
1414818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * @param query Intent query, or <code>null</code>.
1415818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * @param actionKey The key code of the action key that was pressed,
1416818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     *        or {@link KeyEvent#KEYCODE_UNKNOWN} if none.
1417818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * @param actionMsg The message for the action key that was pressed,
1418818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     *        or <code>null</code> if none.
1419818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * @return The intent.
1420818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     */
1421818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    private Intent createIntent(String action, Uri data, String extraData, String query,
1422818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            int actionKey, String actionMsg) {
1423818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        // Now build the Intent
1424818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        Intent intent = new Intent(action);
1425818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
1426818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        // We need CLEAR_TOP to avoid reusing an old task that has other activities
1427818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        // on top of the one we want. We don't want to do this in in-app search though,
1428818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        // as it can be destructive to the activity stack.
1429818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        if (data != null) {
1430818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            intent.setData(data);
1431818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        }
1432818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        intent.putExtra(SearchManager.USER_QUERY, mUserQuery);
1433818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        if (query != null) {
1434818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            intent.putExtra(SearchManager.QUERY, query);
1435818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        }
1436818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        if (extraData != null) {
1437818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            intent.putExtra(SearchManager.EXTRA_DATA_KEY, extraData);
1438818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        }
1439818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        if (mAppSearchData != null) {
1440818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            intent.putExtra(SearchManager.APP_DATA, mAppSearchData);
1441818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        }
1442818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        if (actionKey != KeyEvent.KEYCODE_UNKNOWN) {
1443818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            intent.putExtra(SearchManager.ACTION_KEY, actionKey);
1444818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            intent.putExtra(SearchManager.ACTION_MSG, actionMsg);
1445818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        }
14461b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes        if (IS_AT_LEAST_FROYO) {
14471b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes            intent.setComponent(mSearchable.getSearchActivity());
14481b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes        }
1449818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        return intent;
1450818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    }
1451818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
1452818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    /**
1453818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * Create and return an Intent that can launch the voice search activity for web search.
1454818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     */
14551b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes    @TargetApi(Build.VERSION_CODES.FROYO)
1456818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    private Intent createVoiceWebSearchIntent(Intent baseIntent, SearchableInfo searchable) {
1457818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        Intent voiceIntent = new Intent(baseIntent);
1458818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        ComponentName searchActivity = searchable.getSearchActivity();
1459818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        voiceIntent.putExtra(RecognizerIntent.EXTRA_CALLING_PACKAGE, searchActivity == null ? null
1460818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                : searchActivity.flattenToShortString());
1461818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        return voiceIntent;
1462818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    }
1463818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
1464818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    /**
1465818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * Create and return an Intent that can launch the voice search activity, perform a specific
1466818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * voice transcription, and forward the results to the searchable activity.
1467818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     *
1468818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * @param baseIntent The voice app search intent to start from
1469818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * @return A completely-configured intent ready to send to the voice search activity
1470818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     */
14711b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes    @TargetApi(Build.VERSION_CODES.FROYO)
1472818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    private Intent createVoiceAppSearchIntent(Intent baseIntent, SearchableInfo searchable) {
1473818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        ComponentName searchActivity = searchable.getSearchActivity();
1474818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
1475818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        // create the necessary intent to set up a search-and-forward operation
1476818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        // in the voice search system.   We have to keep the bundle separate,
1477818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        // because it becomes immutable once it enters the PendingIntent
1478818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        Intent queryIntent = new Intent(Intent.ACTION_SEARCH);
1479818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        queryIntent.setComponent(searchActivity);
1480818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        PendingIntent pending = PendingIntent.getActivity(getContext(), 0, queryIntent,
1481818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                PendingIntent.FLAG_ONE_SHOT);
1482818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
1483818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        // Now set up the bundle that will be inserted into the pending intent
1484818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        // when it's time to do the search.  We always build it here (even if empty)
1485818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        // because the voice search activity will always need to insert "QUERY" into
1486818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        // it anyway.
1487818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        Bundle queryExtras = new Bundle();
1488818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        if (mAppSearchData != null) {
1489818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            queryExtras.putParcelable(SearchManager.APP_DATA, mAppSearchData);
1490818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        }
1491818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
1492818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        // Now build the intent to launch the voice search.  Add all necessary
1493818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        // extras to launch the voice recognizer, and then all the necessary extras
1494818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        // to forward the results to the searchable activity
1495818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        Intent voiceIntent = new Intent(baseIntent);
1496818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
1497818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        // Add all of the configuration options supplied by the searchable's metadata
1498818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        String languageModel = RecognizerIntent.LANGUAGE_MODEL_FREE_FORM;
1499818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        String prompt = null;
1500818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        String language = null;
1501818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        int maxResults = 1;
1502818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
15031b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes        if (Build.VERSION.SDK_INT >= 8) {
15041b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes            Resources resources = getResources();
15051b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes            if (searchable.getVoiceLanguageModeId() != 0) {
15061b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes                languageModel = resources.getString(searchable.getVoiceLanguageModeId());
15071b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes            }
15081b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes            if (searchable.getVoicePromptTextId() != 0) {
15091b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes                prompt = resources.getString(searchable.getVoicePromptTextId());
15101b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes            }
15111b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes            if (searchable.getVoiceLanguageId() != 0) {
15121b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes                language = resources.getString(searchable.getVoiceLanguageId());
15131b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes            }
15141b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes            if (searchable.getVoiceMaxResults() != 0) {
15151b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes                maxResults = searchable.getVoiceMaxResults();
15161b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes            }
1517818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        }
1518818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        voiceIntent.putExtra(RecognizerIntent.EXTRA_LANGUAGE_MODEL, languageModel);
1519818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        voiceIntent.putExtra(RecognizerIntent.EXTRA_PROMPT, prompt);
1520818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        voiceIntent.putExtra(RecognizerIntent.EXTRA_LANGUAGE, language);
1521818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        voiceIntent.putExtra(RecognizerIntent.EXTRA_MAX_RESULTS, maxResults);
1522818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        voiceIntent.putExtra(RecognizerIntent.EXTRA_CALLING_PACKAGE, searchActivity == null ? null
1523818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                : searchActivity.flattenToShortString());
1524818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
1525818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        // Add the values that configure forwarding the results
1526818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        voiceIntent.putExtra(RecognizerIntent.EXTRA_RESULTS_PENDINGINTENT, pending);
1527818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        voiceIntent.putExtra(RecognizerIntent.EXTRA_RESULTS_PENDINGINTENT_BUNDLE, queryExtras);
1528818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
1529818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        return voiceIntent;
1530818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    }
1531818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
1532818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    /**
1533818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * When a particular suggestion has been selected, perform the various lookups required
1534818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * to use the suggestion.  This includes checking the cursor for suggestion-specific data,
1535818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * and/or falling back to the XML for defaults;  It also creates REST style Uri data when
1536818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * the suggestion includes a data id.
1537818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     *
1538818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * @param c The suggestions cursor, moved to the row of the user's selection
1539818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * @param actionKey The key code of the action key that was pressed,
1540818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     *        or {@link KeyEvent#KEYCODE_UNKNOWN} if none.
1541818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * @param actionMsg The message for the action key that was pressed,
1542818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     *        or <code>null</code> if none.
1543818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * @return An intent for the suggestion at the cursor's position.
1544818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     */
1545818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    private Intent createIntentFromSuggestion(Cursor c, int actionKey, String actionMsg) {
1546818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        try {
1547818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            // use specific action if supplied, or default action if supplied, or fixed default
1548818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            String action = getColumnString(c, SearchManager.SUGGEST_COLUMN_INTENT_ACTION);
1549818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
15501b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes            if (action == null && Build.VERSION.SDK_INT >= 8) {
1551818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                action = mSearchable.getSuggestIntentAction();
1552818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            }
1553818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            if (action == null) {
1554818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                action = Intent.ACTION_SEARCH;
1555818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            }
1556818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
1557818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            // use specific data if supplied, or default data if supplied
1558818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            String data = getColumnString(c, SearchManager.SUGGEST_COLUMN_INTENT_DATA);
15591b01ce23fa533d467ca2ccbc65c980a59662aa8dChris Banes            if (IS_AT_LEAST_FROYO && data == null) {
1560818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                data = mSearchable.getSuggestIntentData();
1561818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            }
1562818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            // then, if an ID was provided, append it.
1563818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            if (data != null) {
1564818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                String id = getColumnString(c, SearchManager.SUGGEST_COLUMN_INTENT_DATA_ID);
1565818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                if (id != null) {
1566818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                    data = data + "/" + Uri.encode(id);
1567818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                }
1568818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            }
1569818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            Uri dataUri = (data == null) ? null : Uri.parse(data);
1570818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
1571818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            String query = getColumnString(c, SearchManager.SUGGEST_COLUMN_QUERY);
1572818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            String extraData = getColumnString(c, SearchManager.SUGGEST_COLUMN_INTENT_EXTRA_DATA);
1573818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
1574818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            return createIntent(action, dataUri, extraData, query, actionKey, actionMsg);
1575818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        } catch (RuntimeException e ) {
1576818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            int rowNum;
1577818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            try {                       // be really paranoid now
1578818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                rowNum = c.getPosition();
1579818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            } catch (RuntimeException e2 ) {
1580818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                rowNum = -1;
1581818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            }
1582818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            Log.w(LOG_TAG, "Search suggestions cursor at row " + rowNum +
1583818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                    " returned exception.", e);
1584818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            return null;
1585818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        }
1586818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    }
1587818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
1588818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    private void forceSuggestionQuery() {
1589818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        HIDDEN_METHOD_INVOKER.doBeforeTextChanged(mQueryTextView);
1590818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        HIDDEN_METHOD_INVOKER.doAfterTextChanged(mQueryTextView);
1591818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    }
1592818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
1593818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    static boolean isLandscapeMode(Context context) {
1594818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        return context.getResources().getConfiguration().orientation
1595818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                == Configuration.ORIENTATION_LANDSCAPE;
1596818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    }
1597818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
1598818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    /**
1599818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * Callback to watch the text field for empty/non-empty
1600818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     */
1601818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    private TextWatcher mTextWatcher = new TextWatcher() {
1602818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
1603818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        public void beforeTextChanged(CharSequence s, int start, int before, int after) { }
1604818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
1605818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        public void onTextChanged(CharSequence s, int start,
1606818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                int before, int after) {
1607818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            SearchView.this.onTextChanged(s);
1608818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        }
1609818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
1610818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        public void afterTextChanged(Editable s) {
1611818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        }
1612818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    };
1613818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
1614818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    /**
1615818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * Local subclass for AutoCompleteTextView.
1616818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     * @hide
1617818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes     */
1618818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    public static class SearchAutoComplete extends AutoCompleteTextView {
1619818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
1620818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        private int mThreshold;
1621818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        private SearchView mSearchView;
1622818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
1623818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        public SearchAutoComplete(Context context) {
1624818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            super(context);
1625818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            mThreshold = getThreshold();
1626818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        }
1627818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
1628818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        public SearchAutoComplete(Context context, AttributeSet attrs) {
1629818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            super(context, attrs);
1630818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            mThreshold = getThreshold();
1631818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        }
1632818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
1633818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        public SearchAutoComplete(Context context, AttributeSet attrs, int defStyle) {
1634818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            super(context, attrs, defStyle);
1635818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            mThreshold = getThreshold();
1636818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        }
1637818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
1638818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        void setSearchView(SearchView searchView) {
1639818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            mSearchView = searchView;
1640818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        }
1641818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
1642818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        @Override
1643818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        public void setThreshold(int threshold) {
1644818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            super.setThreshold(threshold);
1645818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            mThreshold = threshold;
1646818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        }
1647818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
1648818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        /**
1649818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes         * Returns true if the text field is empty, or contains only whitespace.
1650818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes         */
1651818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        private boolean isEmpty() {
1652818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            return TextUtils.getTrimmedLength(getText()) == 0;
1653818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        }
1654818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
1655818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        /**
1656818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes         * We override this method to avoid replacing the query box text when a
1657818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes         * suggestion is clicked.
1658818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes         */
1659818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        @Override
1660818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        protected void replaceText(CharSequence text) {
1661818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        }
1662818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
1663818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        /**
1664818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes         * We override this method to avoid an extra onItemClick being called on
1665818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes         * the drop-down's OnItemClickListener by
1666818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes         * {@link AutoCompleteTextView#onKeyUp(int, KeyEvent)} when an item is
1667818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes         * clicked with the trackball.
1668818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes         */
1669818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        @Override
1670818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        public void performCompletion() {
1671818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        }
1672818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
1673818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        /**
1674818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes         * We override this method to be sure and show the soft keyboard if
1675818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes         * appropriate when the TextView has focus.
1676818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes         */
1677818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        @Override
1678818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        public void onWindowFocusChanged(boolean hasWindowFocus) {
1679818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            super.onWindowFocusChanged(hasWindowFocus);
1680818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
1681818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            if (hasWindowFocus && mSearchView.hasFocus() && getVisibility() == VISIBLE) {
1682818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                InputMethodManager inputManager = (InputMethodManager) getContext()
1683818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                        .getSystemService(Context.INPUT_METHOD_SERVICE);
1684818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                inputManager.showSoftInput(this, 0);
1685818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                // If in landscape mode, then make sure that
1686818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                // the ime is in front of the dropdown.
1687818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                if (isLandscapeMode(getContext())) {
1688818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                    HIDDEN_METHOD_INVOKER.ensureImeVisible(this, true);
1689818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                }
1690818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            }
1691818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        }
1692818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
1693818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        @Override
1694818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        protected void onFocusChanged(boolean focused, int direction, Rect previouslyFocusedRect) {
1695818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            super.onFocusChanged(focused, direction, previouslyFocusedRect);
1696818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            mSearchView.onTextFocusChanged();
1697818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        }
1698818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
1699818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        /**
1700818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes         * We override this method so that we can allow a threshold of zero,
1701818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes         * which ACTV does not.
1702818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes         */
1703818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        @Override
1704818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        public boolean enoughToFilter() {
1705818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            return mThreshold <= 0 || super.enoughToFilter();
1706818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        }
1707818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
1708818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        @Override
1709818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        public boolean onKeyPreIme(int keyCode, KeyEvent event) {
1710818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            if (keyCode == KeyEvent.KEYCODE_BACK) {
1711818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                // special case for the back key, we do not even try to send it
1712818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                // to the drop down list but instead, consume it immediately
1713818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                if (event.getAction() == KeyEvent.ACTION_DOWN && event.getRepeatCount() == 0) {
1714818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                    KeyEvent.DispatcherState state = getKeyDispatcherState();
1715818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                    if (state != null) {
1716818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                        state.startTracking(event, this);
1717818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                    }
1718818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                    return true;
1719818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                } else if (event.getAction() == KeyEvent.ACTION_UP) {
1720818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                    KeyEvent.DispatcherState state = getKeyDispatcherState();
1721818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                    if (state != null) {
1722818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                        state.handleUpEvent(event);
1723818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                    }
1724818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                    if (event.isTracking() && !event.isCanceled()) {
1725818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                        mSearchView.clearFocus();
1726818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                        mSearchView.setImeVisibility(false);
1727818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                        return true;
1728818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                    }
1729818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                }
1730818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            }
1731818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            return super.onKeyPreIme(keyCode, event);
1732818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        }
1733818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    }
1734818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
1735818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    private static class AutoCompleteTextViewReflector {
1736818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        private Method doBeforeTextChanged, doAfterTextChanged;
1737818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        private Method ensureImeVisible;
17382e21b5e22c320fd7e6af86a7cc05b4b11d7a0f64Chris Banes        private Method showSoftInputUnchecked;
1739818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
1740818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        AutoCompleteTextViewReflector() {
1741818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            try {
1742818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                doBeforeTextChanged = AutoCompleteTextView.class
1743818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                        .getDeclaredMethod("doBeforeTextChanged");
1744818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                doBeforeTextChanged.setAccessible(true);
1745818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            } catch (NoSuchMethodException e) {
1746818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                // Ah well.
1747818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            }
1748818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            try {
1749818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                doAfterTextChanged = AutoCompleteTextView.class
1750818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                        .getDeclaredMethod("doAfterTextChanged");
1751818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                doAfterTextChanged.setAccessible(true);
1752818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            } catch (NoSuchMethodException e) {
1753818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                // Ah well.
1754818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            }
1755818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            try {
1756818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                ensureImeVisible = AutoCompleteTextView.class
17572e21b5e22c320fd7e6af86a7cc05b4b11d7a0f64Chris Banes                        .getMethod("ensureImeVisible", boolean.class);
1758818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                ensureImeVisible.setAccessible(true);
1759818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            } catch (NoSuchMethodException e) {
1760818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                // Ah well.
1761818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            }
17622e21b5e22c320fd7e6af86a7cc05b4b11d7a0f64Chris Banes            try {
17632e21b5e22c320fd7e6af86a7cc05b4b11d7a0f64Chris Banes                showSoftInputUnchecked = InputMethodManager.class.getMethod(
17642e21b5e22c320fd7e6af86a7cc05b4b11d7a0f64Chris Banes                        "showSoftInputUnchecked", int.class, ResultReceiver.class);
17652e21b5e22c320fd7e6af86a7cc05b4b11d7a0f64Chris Banes                showSoftInputUnchecked.setAccessible(true);
17662e21b5e22c320fd7e6af86a7cc05b4b11d7a0f64Chris Banes            } catch (NoSuchMethodException e) {
17672e21b5e22c320fd7e6af86a7cc05b4b11d7a0f64Chris Banes                // Ah well.
17682e21b5e22c320fd7e6af86a7cc05b4b11d7a0f64Chris Banes            }
1769818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        }
1770818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
1771818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        void doBeforeTextChanged(AutoCompleteTextView view) {
1772818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            if (doBeforeTextChanged != null) {
1773818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                try {
1774818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                    doBeforeTextChanged.invoke(view);
17752e21b5e22c320fd7e6af86a7cc05b4b11d7a0f64Chris Banes                } catch (Exception e) {
1776818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                }
1777818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            }
1778818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        }
1779818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
1780818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        void doAfterTextChanged(AutoCompleteTextView view) {
1781818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            if (doAfterTextChanged != null) {
1782818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                try {
1783818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                    doAfterTextChanged.invoke(view);
17842e21b5e22c320fd7e6af86a7cc05b4b11d7a0f64Chris Banes                } catch (Exception e) {
1785818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                }
1786818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            }
1787818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        }
1788818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes
1789818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        void ensureImeVisible(AutoCompleteTextView view, boolean visible) {
1790818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            if (ensureImeVisible != null) {
1791818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                try {
1792818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                    ensureImeVisible.invoke(view, visible);
17932e21b5e22c320fd7e6af86a7cc05b4b11d7a0f64Chris Banes                } catch (Exception e) {
1794818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes                }
1795818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes            }
1796818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes        }
17972e21b5e22c320fd7e6af86a7cc05b4b11d7a0f64Chris Banes
17982e21b5e22c320fd7e6af86a7cc05b4b11d7a0f64Chris Banes        void showSoftInputUnchecked(InputMethodManager imm, View view, int flags) {
17992e21b5e22c320fd7e6af86a7cc05b4b11d7a0f64Chris Banes            if (showSoftInputUnchecked != null) {
18002e21b5e22c320fd7e6af86a7cc05b4b11d7a0f64Chris Banes                try {
18012e21b5e22c320fd7e6af86a7cc05b4b11d7a0f64Chris Banes                    showSoftInputUnchecked.invoke(imm, flags, null);
18022e21b5e22c320fd7e6af86a7cc05b4b11d7a0f64Chris Banes                    return;
18032e21b5e22c320fd7e6af86a7cc05b4b11d7a0f64Chris Banes                } catch (Exception e) {
18042e21b5e22c320fd7e6af86a7cc05b4b11d7a0f64Chris Banes                }
18052e21b5e22c320fd7e6af86a7cc05b4b11d7a0f64Chris Banes            }
18062e21b5e22c320fd7e6af86a7cc05b4b11d7a0f64Chris Banes
18072e21b5e22c320fd7e6af86a7cc05b4b11d7a0f64Chris Banes            // Hidden method failed, call public version instead
18082e21b5e22c320fd7e6af86a7cc05b4b11d7a0f64Chris Banes            imm.showSoftInput(view, flags);
18092e21b5e22c320fd7e6af86a7cc05b4b11d7a0f64Chris Banes        }
1810818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes    }
1811818041428e0aca4943fe5adc0d54dad0d36a75f4Chris Banes}