AutoCompleteTextView.java revision c39a6e0c51e182338deb8b63d07933b585134929
19066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project/*
29066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * Copyright (C) 2007 The Android Open Source Project
39066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project *
49066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * Licensed under the Apache License, Version 2.0 (the "License");
59066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * you may not use this file except in compliance with the License.
69066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * You may obtain a copy of the License at
79066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project *
89066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project *      http://www.apache.org/licenses/LICENSE-2.0
99066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project *
109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * Unless required by applicable law or agreed to in writing, software
119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * distributed under the License is distributed on an "AS IS" BASIS,
129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * See the License for the specific language governing permissions and
149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * limitations under the License.
159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project */
169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectpackage android.widget;
189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport android.content.Context;
209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport android.content.res.TypedArray;
219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport android.graphics.Rect;
229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport android.graphics.drawable.Drawable;
239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport android.text.Editable;
249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport android.text.Selection;
259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport android.text.TextUtils;
269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport android.text.TextWatcher;
279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport android.util.AttributeSet;
289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport android.util.Log;
299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport android.view.KeyEvent;
309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport android.view.LayoutInflater;
319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport android.view.MotionEvent;
329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport android.view.View;
339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport android.view.ViewGroup;
349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport android.view.inputmethod.CompletionInfo;
359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport android.view.inputmethod.InputMethodManager;
369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport android.view.inputmethod.EditorInfo;
379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport com.android.internal.R;
399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project/**
429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * <p>An editable text view that shows completion suggestions automatically
439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * while the user is typing. The list of suggestions is displayed in a drop
449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * down menu from which the user can choose an item to replace the content
459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * of the edit box with.</p>
469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project *
479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * <p>The drop down can be dismissed at any time by pressing the back key or,
489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * if no item is selected in the drop down, by pressing the enter/dpad center
499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * key.</p>
509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project *
519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * <p>The list of suggestions is obtained from a data adapter and appears
529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * only after a given number of characters defined by
539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * {@link #getThreshold() the threshold}.</p>
549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project *
559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * <p>The following code snippet shows how to create a text view which suggests
569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * various countries names while the user is typing:</p>
579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project *
589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * <pre class="prettyprint">
599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * public class CountriesActivity extends Activity {
609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project *     protected void onCreate(Bundle icicle) {
619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project *         super.onCreate(icicle);
629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project *         setContentView(R.layout.countries);
639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project *
649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project *         ArrayAdapter&lt;String&gt; adapter = new ArrayAdapter&lt;String&gt;(this,
659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project *                 android.R.layout.simple_dropdown_item_1line, COUNTRIES);
669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project *         AutoCompleteTextView textView = (AutoCompleteTextView)
679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project *                 findViewById(R.id.countries_list);
689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project *         textView.setAdapter(adapter);
699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project *     }
709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project *
719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project *     private static final String[] COUNTRIES = new String[] {
729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project *         "Belgium", "France", "Italy", "Germany", "Spain"
739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project *     };
749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * }
759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * </pre>
769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project *
779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * @attr ref android.R.styleable#AutoCompleteTextView_completionHint
789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * @attr ref android.R.styleable#AutoCompleteTextView_completionThreshold
799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * @attr ref android.R.styleable#AutoCompleteTextView_completionHintView
809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * @attr ref android.R.styleable#AutoCompleteTextView_dropDownSelector
819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * @attr ref android.R.styleable#AutoCompleteTextView_dropDownAnchor
829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * @attr ref android.R.styleable#AutoCompleteTextView_dropDownWidth
839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project */
849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectpublic class AutoCompleteTextView extends EditText implements Filter.FilterListener {
859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    static final boolean DEBUG = false;
869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    static final String TAG = "AutoCompleteTextView";
879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private static final int HINT_VIEW_ID = 0x17;
899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private CharSequence mHintText;
919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private int mHintResource;
929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private ListAdapter mAdapter;
949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private Filter mFilter;
959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private int mThreshold;
969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private PopupWindow mPopup;
989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private DropDownListView mDropDownList;
999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private int mDropDownVerticalOffset;
1009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private int mDropDownHorizontalOffset;
1019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private int mDropDownAnchorId;
1029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private View mDropDownAnchorView;  // view is retrieved lazily from id once needed
1039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private int mDropDownWidth;
1049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private Drawable mDropDownListHighlight;
1069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private AdapterView.OnItemClickListener mItemClickListener;
1089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private AdapterView.OnItemSelectedListener mItemSelectedListener;
1099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private final DropDownItemClickListener mDropDownItemClickListener =
1119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            new DropDownItemClickListener();
1129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private int mLastKeyCode = KeyEvent.KEYCODE_UNKNOWN;
1149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private boolean mOpenBefore;
1159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private Validator mValidator = null;
1179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private boolean mBlockCompletion;
1199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private AutoCompleteTextView.ListSelectorHider mHideSelector;
1219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // Indicates whether this AutoCompleteTextView is attached to a window or not
1239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // The widget is attached to a window when mAttachCount > 0
1249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private int mAttachCount;
1259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public AutoCompleteTextView(Context context) {
1279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        this(context, null);
1289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
1299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public AutoCompleteTextView(Context context, AttributeSet attrs) {
1319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        this(context, attrs, com.android.internal.R.attr.autoCompleteTextViewStyle);
1329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
1339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public AutoCompleteTextView(Context context, AttributeSet attrs, int defStyle) {
1359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        super(context, attrs, defStyle);
1369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mPopup = new PopupWindow(context, attrs,
1389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                com.android.internal.R.attr.autoCompleteTextViewStyle);
1399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        TypedArray a =
1419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            context.obtainStyledAttributes(
1429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                attrs, com.android.internal.R.styleable.AutoCompleteTextView, defStyle, 0);
1439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mThreshold = a.getInt(
1459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                R.styleable.AutoCompleteTextView_completionThreshold, 2);
1469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mHintText = a.getText(R.styleable.AutoCompleteTextView_completionHint);
1489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mDropDownListHighlight = a.getDrawable(
1509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                R.styleable.AutoCompleteTextView_dropDownSelector);
1519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mDropDownVerticalOffset = (int)
1529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                a.getDimension(R.styleable.AutoCompleteTextView_dropDownVerticalOffset, 0.0f);
1539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mDropDownHorizontalOffset = (int)
1549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                a.getDimension(R.styleable.AutoCompleteTextView_dropDownHorizontalOffset, 0.0f);
1559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // Get the anchor's id now, but the view won't be ready, so wait to actually get the
1579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // view and store it in mDropDownAnchorView lazily in getDropDownAnchorView later.
1589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // Defaults to NO_ID, in which case the getDropDownAnchorView method will simply return
1599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // this TextView, as a default anchoring point.
1609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mDropDownAnchorId = a.getResourceId(R.styleable.AutoCompleteTextView_dropDownAnchor,
1619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                View.NO_ID);
1629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // For dropdown width, the developer can specify a specific width, or FILL_PARENT
1649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // (for full screen width) or WRAP_CONTENT (to match the width of the anchored view).
1659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mDropDownWidth = a.getLayoutDimension(R.styleable.AutoCompleteTextView_dropDownWidth,
1669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                ViewGroup.LayoutParams.WRAP_CONTENT);
1679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mHintResource = a.getResourceId(R.styleable.AutoCompleteTextView_completionHintView,
1699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                R.layout.simple_dropdown_hint);
1709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // Always turn on the auto complete input type flag, since it
1729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // makes no sense to use this widget without it.
1739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        int inputType = getInputType();
1749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if ((inputType&EditorInfo.TYPE_MASK_CLASS)
1759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                == EditorInfo.TYPE_CLASS_TEXT) {
1769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            inputType |= EditorInfo.TYPE_TEXT_FLAG_AUTO_COMPLETE;
1779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            setRawInputType(inputType);
1789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
1799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        a.recycle();
1819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        setFocusable(true);
1839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        addTextChangedListener(new MyWatcher());
1859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
1869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
1889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * Sets this to be single line; a separate method so
1899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * MultiAutoCompleteTextView can skip this.
1909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
1919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /* package */ void finishInit() {
1929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        setSingleLine();
1939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
1949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
1969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * <p>Sets the optional hint text that is displayed at the bottom of the
1979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * the matching list.  This can be used as a cue to the user on how to
1989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * best use the list, or to provide extra information.</p>
1999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *
2009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param hint the text to be displayed to the user
2019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *
2029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @attr ref android.R.styleable#AutoCompleteTextView_completionHint
2039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
2049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public void setCompletionHint(CharSequence hint) {
2059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mHintText = hint;
2069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
2079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
2099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * <p>Returns the current width for the auto-complete drop down list. This can
2109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * be a fixed width, or {@link ViewGroup.LayoutParams#FILL_PARENT} to fill the screen, or
2119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * {@link ViewGroup.LayoutParams#WRAP_CONTENT} to fit the width of its anchor view.</p>
2129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *
2139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @return the width for the drop down list
2149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
2159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public int getDropDownWidth() {
2169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return mDropDownWidth;
2179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
2189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
2209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * <p>Sets the current width for the auto-complete drop down list. This can
2219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * be a fixed width, or {@link ViewGroup.LayoutParams#FILL_PARENT} to fill the screen, or
2229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * {@link ViewGroup.LayoutParams#WRAP_CONTENT} to fit the width of its anchor view.</p>
2239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *
2249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param width the width to use
2259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
2269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public void setDropDownWidth(int width) {
2279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mDropDownWidth = width;
2289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
2299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
2319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * <p>Returns the id for the view that the auto-complete drop down list is anchored to.</p>
2329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *
2339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @return the view's id, or {@link View#NO_ID} if none specified
2349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
2359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public int getDropDownAnchor() {
2369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return mDropDownAnchorId;
2379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
2389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
2409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * <p>Sets the view to which the auto-complete drop down list should anchor. The view
2419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * corresponding to this id will not be loaded until the next time it is needed to avoid
2429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * loading a view which is not yet instantiated.</p>
2439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *
2449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param id the id to anchor the drop down list view to
2459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
2469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public void setDropDownAnchor(int id) {
2479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mDropDownAnchorId = id;
2489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mDropDownAnchorView = null;
2499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
2509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
2529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * <p>Returns the number of characters the user must type before the drop
2539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * down list is shown.</p>
2549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *
2559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @return the minimum number of characters to type to show the drop down
2569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *
2579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @see #setThreshold(int)
2589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
2599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public int getThreshold() {
2609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return mThreshold;
2619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
2629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
2649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * <p>Specifies the minimum number of characters the user has to type in the
2659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * edit box before the drop down list is shown.</p>
2669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *
2679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * <p>When <code>threshold</code> is less than or equals 0, a threshold of
2689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * 1 is applied.</p>
2699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *
2709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param threshold the number of characters to type before the drop down
2719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *                  is shown
2729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *
2739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @see #getThreshold()
2749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *
2759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @attr ref android.R.styleable#AutoCompleteTextView_completionThreshold
2769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
2779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public void setThreshold(int threshold) {
2789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (threshold <= 0) {
2799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            threshold = 1;
2809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
2819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mThreshold = threshold;
2839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
2849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
2869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * <p>Sets the listener that will be notified when the user clicks an item
2879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * in the drop down list.</p>
2889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *
2899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param l the item click listener
2909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
2919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public void setOnItemClickListener(AdapterView.OnItemClickListener l) {
2929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mItemClickListener = l;
2939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
2949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
2969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * <p>Sets the listener that will be notified when the user selects an item
2979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * in the drop down list.</p>
2989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *
2999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param l the item selected listener
3009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
3019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public void setOnItemSelectedListener(AdapterView.OnItemSelectedListener l) {
3029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mItemSelectedListener = l;
3039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
3049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
3069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * <p>Returns the listener that is notified whenever the user clicks an item
3079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * in the drop down list.</p>
3089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *
3099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @return the item click listener
3109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *
3119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @deprecated Use {@link #getOnItemClickListener()} intead
3129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
3139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    @Deprecated
3149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public AdapterView.OnItemClickListener getItemClickListener() {
3159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return mItemClickListener;
3169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
3179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
3199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * <p>Returns the listener that is notified whenever the user selects an
3209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * item in the drop down list.</p>
3219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *
3229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @return the item selected listener
3239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *
3249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @deprecated Use {@link #getOnItemSelectedListener()} intead
3259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
3269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    @Deprecated
3279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public AdapterView.OnItemSelectedListener getItemSelectedListener() {
3289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return mItemSelectedListener;
3299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
3309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
3329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * <p>Returns the listener that is notified whenever the user clicks an item
3339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * in the drop down list.</p>
3349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *
3359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @return the item click listener
3369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
3379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public AdapterView.OnItemClickListener getOnItemClickListener() {
3389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return mItemClickListener;
3399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
3409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
3429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * <p>Returns the listener that is notified whenever the user selects an
3439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * item in the drop down list.</p>
3449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *
3459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @return the item selected listener
3469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
3479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public AdapterView.OnItemSelectedListener getOnItemSelectedListener() {
3489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return mItemSelectedListener;
3499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
3509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
3529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * <p>Returns a filterable list adapter used for auto completion.</p>
3539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *
3549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @return a data adapter used for auto completion
3559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
3569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public ListAdapter getAdapter() {
3579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return mAdapter;
3589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
3599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
3619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * <p>Changes the list of data used for auto completion. The provided list
3629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * must be a filterable list adapter.</p>
3639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *
3649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * <p>The caller is still responsible for managing any resources used by the adapter.
3659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * Notably, when the AutoCompleteTextView is closed or released, the adapter is not notified.
3669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * A common case is the use of {@link android.widget.CursorAdapter}, which
3679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * contains a {@link android.database.Cursor} that must be closed.  This can be done
3689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * automatically (see
3699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * {@link android.app.Activity#startManagingCursor(android.database.Cursor)
3709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * startManagingCursor()}),
3719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * or by manually closing the cursor when the AutoCompleteTextView is dismissed.</p>
3729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *
3739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param adapter the adapter holding the auto completion data
3749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *
3759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @see #getAdapter()
3769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @see android.widget.Filterable
3779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @see android.widget.ListAdapter
3789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
3799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public <T extends ListAdapter & Filterable> void setAdapter(T adapter) {
3809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mAdapter = adapter;
3819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (mAdapter != null) {
3829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            //noinspection unchecked
3839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mFilter = ((Filterable) mAdapter).getFilter();
3849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        } else {
3859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mFilter = null;
3869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
3879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (mDropDownList != null) {
3899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mDropDownList.setAdapter(mAdapter);
3909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
3919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
3929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    @Override
3949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public boolean onKeyPreIme(int keyCode, KeyEvent event) {
3959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (isPopupShowing()) {
3969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            // special case for the back key, we do not even try to send it
3979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            // to the drop down list but instead, consume it immediately
3989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            if (keyCode == KeyEvent.KEYCODE_BACK) {
3999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                dismissDropDown();
4009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                return true;
4019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
4029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
4039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return super.onKeyPreIme(keyCode, event);
4049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
4059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
4069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    @Override
4079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public boolean onKeyUp(int keyCode, KeyEvent event) {
4089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (isPopupShowing() && mDropDownList.getSelectedItemPosition() >= 0) {
4099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            boolean consumed = mDropDownList.onKeyUp(keyCode, event);
4109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            if (consumed) {
4119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                switch (keyCode) {
4129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    // if the list accepts the key events and the key event
4139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    // was a click, the text view gets the selected item
4149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    // from the drop down as its content
4159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    case KeyEvent.KEYCODE_ENTER:
4169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    case KeyEvent.KEYCODE_DPAD_CENTER:
4179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        performCompletion();
4189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        return true;
4199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                }
4209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
4219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
4229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return super.onKeyUp(keyCode, event);
4239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
4249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
4259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    @Override
4269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public boolean onKeyDown(int keyCode, KeyEvent event) {
4279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // when the drop down is shown, we drive it directly
4289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (isPopupShowing()) {
4299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            // the key events are forwarded to the list in the drop down view
4309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            // note that ListView handles space but we don't want that to happen
4319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            // also if selection is not currently in the drop down, then don't
4329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            // let center or enter presses go there since that would cause it
4339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            // to select one of its items
4349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            if (keyCode != KeyEvent.KEYCODE_SPACE
4359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    && (mDropDownList.getSelectedItemPosition() >= 0
4369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                            || (keyCode != KeyEvent.KEYCODE_ENTER
4379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                                    && keyCode != KeyEvent.KEYCODE_DPAD_CENTER))) {
4389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                int curIndex = mDropDownList.getSelectedItemPosition();
4399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                boolean consumed;
4409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                final boolean below = !mPopup.isAboveAnchor();
4419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                if ((below && keyCode == KeyEvent.KEYCODE_DPAD_UP && curIndex <= 0) ||
4429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        (!below && keyCode == KeyEvent.KEYCODE_DPAD_DOWN && curIndex >=
4439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        mDropDownList.getAdapter().getCount() - 1)) {
4449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    // When the selection is at the top, we block the key
4459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    // event to prevent focus from moving.
4469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    mDropDownList.hideSelector();
4479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    mDropDownList.requestLayout();
4489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    mPopup.setInputMethodMode(PopupWindow.INPUT_METHOD_NEEDED);
4499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    mPopup.update();
4509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    return true;
4519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                }
4529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                consumed = mDropDownList.onKeyDown(keyCode, event);
4539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                if (DEBUG) Log.v(TAG, "Key down: code=" + keyCode + " list consumed="
4549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        + consumed);
4559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                if (consumed) {
4569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    // If it handled the key event, then the user is
4579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    // navigating in the list, so we should put it in front.
4589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    mPopup.setInputMethodMode(PopupWindow.INPUT_METHOD_NOT_NEEDED);
4599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    // Here's a little trick we need to do to make sure that
4609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    // the list view is actually showing its focus indicator,
4619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    // by ensuring it has focus and getting its window out
4629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    // of touch mode.
4639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    mDropDownList.requestFocusFromTouch();
4649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    mPopup.update();
4659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
4669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    switch (keyCode) {
4679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        // avoid passing the focus from the text view to the
4689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        // next component
4699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        case KeyEvent.KEYCODE_ENTER:
4709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        case KeyEvent.KEYCODE_DPAD_CENTER:
4719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        case KeyEvent.KEYCODE_DPAD_DOWN:
4729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        case KeyEvent.KEYCODE_DPAD_UP:
4739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                            return true;
4749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    }
4759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                } else {
4769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    if (below && keyCode == KeyEvent.KEYCODE_DPAD_DOWN) {
4779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        // when the selection is at the bottom, we block the
4789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        // event to avoid going to the next focusable widget
4799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        Adapter adapter = mDropDownList.getAdapter();
4809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        if (adapter != null && curIndex == adapter.getCount() - 1) {
4819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                            return true;
4829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        }
4839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    } else if (!below && keyCode == KeyEvent.KEYCODE_DPAD_UP && curIndex == 0) {
4849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        return true;
4859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    }
4869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                }
4879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
4889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        } else {
4899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            switch(keyCode) {
4909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            case KeyEvent.KEYCODE_DPAD_DOWN:
4919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                performValidation();
4929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
4939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
4949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
4959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mLastKeyCode = keyCode;
4969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        boolean handled = super.onKeyDown(keyCode, event);
4979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mLastKeyCode = KeyEvent.KEYCODE_UNKNOWN;
4989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
499c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project        if (handled && isPopupShowing() && mDropDownList != null) {
500c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project            clearListSelection();
501c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project        }
502c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project
5039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return handled;
5049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
5059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
5079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * Returns <code>true</code> if the amount of text in the field meets
5089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * or exceeds the {@link #getThreshold} requirement.  You can override
5099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * this to impose a different standard for when filtering will be
5109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * triggered.
5119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
5129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public boolean enoughToFilter() {
5139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (DEBUG) Log.v(TAG, "Enough to filter: len=" + getText().length()
5149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                + " threshold=" + mThreshold);
5159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return getText().length() >= mThreshold;
5169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
5179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
5199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * This is used to watch for edits to the text view.  Note that we call
5209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * to methods on the auto complete text view class so that we can access
5219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * private vars without going through thunks.
5229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
5239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private class MyWatcher implements TextWatcher {
5249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        public void afterTextChanged(Editable s) {
5259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            doAfterTextChanged();
5269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
5279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        public void beforeTextChanged(CharSequence s, int start, int count, int after) {
5289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            doBeforeTextChanged();
5299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
5309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        public void onTextChanged(CharSequence s, int start, int before, int count) {
5319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
5329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
5339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    void doBeforeTextChanged() {
5359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (mBlockCompletion) return;
5369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // when text is changed, inserted or deleted, we attempt to show
5389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // the drop down
5399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mOpenBefore = isPopupShowing();
5409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (DEBUG) Log.v(TAG, "before text changed: open=" + mOpenBefore);
5419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
5429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    void doAfterTextChanged() {
5449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (mBlockCompletion) return;
5459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // if the list was open before the keystroke, but closed afterwards,
5479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // then something in the keystroke processing (an input filter perhaps)
5489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // called performCompletion() and we shouldn't do any more processing.
5499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (DEBUG) Log.v(TAG, "after text changed: openBefore=" + mOpenBefore
5509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                + " open=" + isPopupShowing());
5519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (mOpenBefore && !isPopupShowing()) {
5529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            return;
5539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
5549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // the drop down is shown only when a minimum number of characters
5569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // was typed in the text view
5579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (enoughToFilter()) {
5589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            if (mFilter != null) {
5599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                performFiltering(getText(), mLastKeyCode);
5609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
5619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        } else {
5629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            // drop down is automatically dismissed when enough characters
5639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            // are deleted from the text view
5649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            dismissDropDown();
5659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            if (mFilter != null) {
5669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                mFilter.filter(null);
5679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
5689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
5699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
5709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
5729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * <p>Indicates whether the popup menu is showing.</p>
5739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *
5749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @return true if the popup menu is showing, false otherwise
5759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
5769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public boolean isPopupShowing() {
5779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return mPopup.isShowing();
5789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
5799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
5819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * <p>Converts the selected item from the drop down list into a sequence
5829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * of character that can be used in the edit box.</p>
5839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *
5849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param selectedItem the item selected by the user for completion
5859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *
5869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @return a sequence of characters representing the selected suggestion
5879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
5889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    protected CharSequence convertSelectionToString(Object selectedItem) {
5899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return mFilter.convertResultToString(selectedItem);
5909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
5919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
5939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * <p>Clear the list selection.  This may only be temporary, as user input will often bring
5949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * it back.
5959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
5969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public void clearListSelection() {
5979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (mDropDownList != null) {
5989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mDropDownList.hideSelector();
5999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mDropDownList.requestLayout();
6009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
6019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
6029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
6039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
6049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * Set the position of the dropdown view selection.
6059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *
6069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param position The position to move the selector to.
6079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
6089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public void setListSelection(int position) {
6099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (mPopup.isShowing() && (mDropDownList != null)) {
6109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mDropDownList.setSelection(position);
6119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            // ListView.setSelection() will call requestLayout()
6129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
6139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
6149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
6159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
6169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * Get the position of the dropdown view selection, if there is one.  Returns
6179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * {@link ListView#INVALID_POSITION ListView.INVALID_POSITION} if there is no dropdown or if
6189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * there is no selection.
6199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *
6209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @return the position of the current selection, if there is one, or
6219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * {@link ListView#INVALID_POSITION ListView.INVALID_POSITION} if not.
6229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *
6239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @see ListView#getSelectedItemPosition()
6249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
6259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public int getListSelection() {
6269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (mPopup.isShowing() && (mDropDownList != null)) {
6279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            return mDropDownList.getSelectedItemPosition();
6289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
6299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return ListView.INVALID_POSITION;
6309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
6319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
6329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
6339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * <p>Starts filtering the content of the drop down list. The filtering
6349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * pattern is the content of the edit box. Subclasses should override this
6359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * method to filter with a different pattern, for instance a substring of
6369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * <code>text</code>.</p>
6379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *
6389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param text the filtering pattern
6399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param keyCode the last character inserted in the edit box; beware that
6409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * this will be null when text is being added through a soft input method.
6419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
6429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    @SuppressWarnings({ "UnusedDeclaration" })
6439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    protected void performFiltering(CharSequence text, int keyCode) {
6449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mFilter.filter(text, this);
6459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
6469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
6479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
6489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * <p>Performs the text completion by converting the selected item from
6499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * the drop down list into a string, replacing the text box's content with
6509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * this string and finally dismissing the drop down menu.</p>
6519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
6529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public void performCompletion() {
6539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        performCompletion(null, -1, -1);
6549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
6559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
6569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    @Override
6579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public void onCommitCompletion(CompletionInfo completion) {
6589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (isPopupShowing()) {
6599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mBlockCompletion = true;
6609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            replaceText(completion.getText());
6619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mBlockCompletion = false;
6629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
6639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            if (mItemClickListener != null) {
6649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                final DropDownListView list = mDropDownList;
6659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                // Note that we don't have a View here, so we will need to
6669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                // supply null.  Hopefully no existing apps crash...
6679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                mItemClickListener.onItemClick(list, null, completion.getPosition(),
6689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        completion.getId());
6699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
6709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
6719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
6729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
6739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private void performCompletion(View selectedView, int position, long id) {
6749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (isPopupShowing()) {
6759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            Object selectedItem;
6769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            if (position < 0) {
6779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                selectedItem = mDropDownList.getSelectedItem();
6789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            } else {
6799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                selectedItem = mAdapter.getItem(position);
6809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
6819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            if (selectedItem == null) {
6829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                Log.w(TAG, "performCompletion: no selected item");
6839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                return;
6849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
6859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
6869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mBlockCompletion = true;
6879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            replaceText(convertSelectionToString(selectedItem));
6889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mBlockCompletion = false;
6899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
6909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            if (mItemClickListener != null) {
6919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                final DropDownListView list = mDropDownList;
6929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
6939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                if (selectedView == null || position < 0) {
6949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    selectedView = list.getSelectedView();
6959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    position = list.getSelectedItemPosition();
6969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    id = list.getSelectedItemId();
6979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                }
6989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                mItemClickListener.onItemClick(list, selectedView, position, id);
6999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
7009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
7019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
7029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        dismissDropDown();
7039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
7049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
7059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
7069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * Identifies whether the view is currently performing a text completion, so subclasses
7079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * can decide whether to respond to text changed events.
7089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
7099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public boolean isPerformingCompletion() {
7109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return mBlockCompletion;
7119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
7129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
7139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
7149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * <p>Performs the text completion by replacing the current text by the
7159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * selected item. Subclasses should override this method to avoid replacing
7169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * the whole content of the edit box.</p>
7179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *
7189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param text the selected suggestion in the drop down list
7199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
7209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    protected void replaceText(CharSequence text) {
7219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        setText(text);
7229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // make sure we keep the caret at the end of the text view
7239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        Editable spannable = getText();
7249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        Selection.setSelection(spannable, spannable.length());
7259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
7269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
7279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public void onFilterComplete(int count) {
7289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (mAttachCount <= 0) return;
7299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
7309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        /*
7319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         * This checks enoughToFilter() again because filtering requests
7329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         * are asynchronous, so the result may come back after enough text
7339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         * has since been deleted to make it no longer appropriate
7349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         * to filter.
7359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         */
7369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
7379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (count > 0 && enoughToFilter()) {
7389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            if (hasFocus() && hasWindowFocus()) {
7399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                showDropDown();
7409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
7419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        } else {
7429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            dismissDropDown();
7439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
7449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
7459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
7469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    @Override
7479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public void onWindowFocusChanged(boolean hasWindowFocus) {
7489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        super.onWindowFocusChanged(hasWindowFocus);
7499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        performValidation();
7509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (!hasWindowFocus) {
7519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            dismissDropDown();
7529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
7539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
7549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
7559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    @Override
7569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    protected void onFocusChanged(boolean focused, int direction, Rect previouslyFocusedRect) {
7579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        super.onFocusChanged(focused, direction, previouslyFocusedRect);
7589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        performValidation();
7599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (!focused) {
7609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            dismissDropDown();
7619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
7629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
7639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
7649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    @Override
7659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    protected void onAttachedToWindow() {
7669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        super.onAttachedToWindow();
7679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mAttachCount++;
7689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
7699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
7709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    @Override
7719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    protected void onDetachedFromWindow() {
7729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        dismissDropDown();
7739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mAttachCount--;
7749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        super.onDetachedFromWindow();
7759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
7769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
7779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
7789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * <p>Closes the drop down if present on screen.</p>
7799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
7809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public void dismissDropDown() {
7819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        InputMethodManager imm = InputMethodManager.peekInstance();
7829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (imm != null) {
7839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            imm.displayCompletions(this, null);
7849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
7859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mPopup.dismiss();
7869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mPopup.setContentView(null);
7879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mDropDownList = null;
7889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
7899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
7909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    @Override
7919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    protected boolean setFrame(int l, int t, int r, int b) {
7929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        boolean result = super.setFrame(l, t, r, b);
7939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
7949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (mPopup.isShowing()) {
7959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mPopup.update(this, r - l, -1);
7969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
7979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
7989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return result;
7999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
8009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
8019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
8029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * <p>Used for lazy instantiation of the anchor view from the id we have. If the value of
8039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * the id is NO_ID or we can't find a view for the given id, we return this TextView as
8049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * the default anchoring point.</p>
8059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
8069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private View getDropDownAnchorView() {
8079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (mDropDownAnchorView == null && mDropDownAnchorId != View.NO_ID) {
8089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mDropDownAnchorView = getRootView().findViewById(mDropDownAnchorId);
8099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
8109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return mDropDownAnchorView == null ? this : mDropDownAnchorView;
8119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
8129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
8139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
8149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * <p>Displays the drop down on screen.</p>
8159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
8169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public void showDropDown() {
8179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        int height = buildDropDown();
8189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (mPopup.isShowing()) {
8199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            int widthSpec;
8209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            if (mDropDownWidth == ViewGroup.LayoutParams.FILL_PARENT) {
8219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                // The call to PopupWindow's update method below can accept -1 for any
8229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                // value you do not want to update.
8239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                widthSpec = -1;
8249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            } else if (mDropDownWidth == ViewGroup.LayoutParams.WRAP_CONTENT) {
8259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                widthSpec = getDropDownAnchorView().getWidth();
8269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            } else {
8279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                widthSpec = mDropDownWidth;
8289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
8299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mPopup.update(getDropDownAnchorView(), mDropDownHorizontalOffset,
8309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    mDropDownVerticalOffset, widthSpec, height);
8319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        } else {
8329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            if (mDropDownWidth == ViewGroup.LayoutParams.FILL_PARENT) {
8339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                mPopup.setWindowLayoutMode(ViewGroup.LayoutParams.FILL_PARENT,
8349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        ViewGroup.LayoutParams.WRAP_CONTENT);
8359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            } else {
8369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                mPopup.setWindowLayoutMode(0, ViewGroup.LayoutParams.WRAP_CONTENT);
8379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                if (mDropDownWidth == ViewGroup.LayoutParams.WRAP_CONTENT) {
8389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    mPopup.setWidth(getDropDownAnchorView().getWidth());
8399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                } else {
8409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    mPopup.setWidth(mDropDownWidth);
8419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                }
8429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
8439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mPopup.setHeight(height);
8449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mPopup.setInputMethodMode(PopupWindow.INPUT_METHOD_NEEDED);
8459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mPopup.setOutsideTouchable(true);
8469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mPopup.setTouchInterceptor(new PopupTouchIntercepter());
8479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mPopup.showAsDropDown(getDropDownAnchorView(),
8489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    mDropDownHorizontalOffset, mDropDownVerticalOffset);
8499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mDropDownList.setSelection(ListView.INVALID_POSITION);
8509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mDropDownList.hideSelector();
8519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mDropDownList.requestFocus();
8529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            post(mHideSelector);
8539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
8549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
8559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
8569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
8579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * <p>Builds the popup window's content and returns the height the popup
8589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * should have. Returns -1 when the content already exists.</p>
8599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *
8609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @return the content's height or -1 if content already exists
8619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
8629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private int buildDropDown() {
8639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        ViewGroup dropDownView;
8649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        int otherHeights = 0;
8659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
8669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (mAdapter != null) {
8679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            InputMethodManager imm = InputMethodManager.peekInstance();
8689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            if (imm != null) {
8699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                int N = mAdapter.getCount();
8709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                if (N > 20) N = 20;
8719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                CompletionInfo[] completions = new CompletionInfo[N];
8729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                for (int i = 0; i < N; i++) {
8739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    Object item = mAdapter.getItem(i);
8749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    long id = mAdapter.getItemId(i);
8759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    completions[i] = new CompletionInfo(id, i,
8769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                            convertSelectionToString(item));
8779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                }
8789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                imm.displayCompletions(this, completions);
8799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
8809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
8819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
8829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (mDropDownList == null) {
8839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            Context context = getContext();
8849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
8859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mHideSelector = new ListSelectorHider();
8869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
8879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mDropDownList = new DropDownListView(context);
8889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mDropDownList.setSelector(mDropDownListHighlight);
8899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mDropDownList.setAdapter(mAdapter);
8909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mDropDownList.setVerticalFadingEdgeEnabled(true);
8919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mDropDownList.setOnItemClickListener(mDropDownItemClickListener);
8929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mDropDownList.setFocusable(true);
8939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mDropDownList.setFocusableInTouchMode(true);
8949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
8959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            if (mItemSelectedListener != null) {
8969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                mDropDownList.setOnItemSelectedListener(mItemSelectedListener);
8979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
8989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
8999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            dropDownView = mDropDownList;
9009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
9019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            View hintView = getHintView(context);
9029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            if (hintView != null) {
9039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                // if an hint has been specified, we accomodate more space for it and
9049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                // add a text view in the drop down menu, at the bottom of the list
9059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                LinearLayout hintContainer = new LinearLayout(context);
9069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                hintContainer.setOrientation(LinearLayout.VERTICAL);
9079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
9089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                LinearLayout.LayoutParams hintParams = new LinearLayout.LayoutParams(
9099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        ViewGroup.LayoutParams.FILL_PARENT, 0, 1.0f
9109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                );
9119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                hintContainer.addView(dropDownView, hintParams);
9129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                hintContainer.addView(hintView);
9139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
9149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                // measure the hint's height to find how much more vertical space
9159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                // we need to add to the drop down's height
9169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                int widthSpec = MeasureSpec.makeMeasureSpec(getWidth(), MeasureSpec.AT_MOST);
9179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                int heightSpec = MeasureSpec.UNSPECIFIED;
9189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                hintView.measure(widthSpec, heightSpec);
9199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
9209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                hintParams = (LinearLayout.LayoutParams) hintView.getLayoutParams();
9219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                otherHeights = hintView.getMeasuredHeight() + hintParams.topMargin
9229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        + hintParams.bottomMargin;
9239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
9249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                dropDownView = hintContainer;
9259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
9269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
9279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mPopup.setContentView(dropDownView);
9289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        } else {
9299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            dropDownView = (ViewGroup) mPopup.getContentView();
9309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            final View view = dropDownView.findViewById(HINT_VIEW_ID);
9319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            if (view != null) {
9329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                LinearLayout.LayoutParams hintParams =
9339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        (LinearLayout.LayoutParams) view.getLayoutParams();
9349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                otherHeights = view.getMeasuredHeight() + hintParams.topMargin
9359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        + hintParams.bottomMargin;
9369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
9379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
9389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
9399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // Max height available on the screen for a popup anchored to us
9409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        final int maxHeight = mPopup.getMaxAvailableHeight(this, mDropDownVerticalOffset);
9419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        //otherHeights += dropDownView.getPaddingTop() + dropDownView.getPaddingBottom();
9429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
9439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return mDropDownList.measureHeightOfChildren(MeasureSpec.UNSPECIFIED,
9449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                0, ListView.NO_POSITION, maxHeight - otherHeights, 2) + otherHeights;
9459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
9469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
9479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private View getHintView(Context context) {
9489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (mHintText != null && mHintText.length() > 0) {
9499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            final TextView hintView = (TextView) LayoutInflater.from(context).inflate(
9509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    mHintResource, null).findViewById(com.android.internal.R.id.text1);
9519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            hintView.setText(mHintText);
9529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            hintView.setId(HINT_VIEW_ID);
9539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            return hintView;
9549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        } else {
9559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            return null;
9569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
9579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
9589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
9599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
9609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * Sets the validator used to perform text validation.
9619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *
9629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param validator The validator used to validate the text entered in this widget.
9639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *
9649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @see #getValidator()
9659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @see #performValidation()
9669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
9679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public void setValidator(Validator validator) {
9689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mValidator = validator;
9699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
9709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
9719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
9729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * Returns the Validator set with {@link #setValidator},
9739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * or <code>null</code> if it was not set.
9749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *
9759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @see #setValidator(android.widget.AutoCompleteTextView.Validator)
9769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @see #performValidation()
9779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
9789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public Validator getValidator() {
9799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return mValidator;
9809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
9819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
9829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
9839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * If a validator was set on this view and the current string is not valid,
9849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * ask the validator to fix it.
9859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *
9869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @see #getValidator()
9879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @see #setValidator(android.widget.AutoCompleteTextView.Validator)
9889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
9899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public void performValidation() {
9909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (mValidator == null) return;
9919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
9929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        CharSequence text = getText();
9939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
9949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (!TextUtils.isEmpty(text) && !mValidator.isValid(text)) {
9959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            setText(mValidator.fixText(text));
9969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
9979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
9989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
9999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
10009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * Returns the Filter obtained from {@link Filterable#getFilter},
10019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * or <code>null</code> if {@link #setAdapter} was not called with
10029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * a Filterable.
10039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
10049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    protected Filter getFilter() {
10059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return mFilter;
10069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
10079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
10089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private class ListSelectorHider implements Runnable {
10099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        public void run() {
10109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            if (mDropDownList != null) {
10119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                mDropDownList.hideSelector();
10129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                mDropDownList.requestLayout();
10139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
10149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
10159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
10169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
10179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private class PopupTouchIntercepter implements OnTouchListener {
10189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        public boolean onTouch(View v, MotionEvent event) {
10199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            if (event.getAction() == MotionEvent.ACTION_DOWN) {
10209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                mPopup.setInputMethodMode(PopupWindow.INPUT_METHOD_NOT_NEEDED);
10219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                mPopup.update();
10229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
10239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            return false;
10249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
10259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
10269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
10279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private class DropDownItemClickListener implements AdapterView.OnItemClickListener {
10289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        public void onItemClick(AdapterView parent, View v, int position, long id) {
10299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            performCompletion(v, position, id);
10309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
10319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
10329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
10339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
10349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * <p>Wrapper class for a ListView. This wrapper hijacks the focus to
10359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * make sure the list uses the appropriate drawables and states when
10369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * displayed on screen within a drop down. The focus is never actually
10379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * passed to the drop down; the list only looks focused.</p>
10389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
10399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private static class DropDownListView extends ListView {
10409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        /**
10419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         * <p>Creates a new list view wrapper.</p>
10429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         *
10439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         * @param context this view's context
10449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         */
10459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        public DropDownListView(Context context) {
10469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            super(context, null, com.android.internal.R.attr.dropDownListViewStyle);
10479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
10489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
10499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        /**
10509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         * <p>Avoids jarring scrolling effect by ensuring that list elements
10519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         * made of a text view fit on a single line.</p>
10529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         *
10539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         * @param position the item index in the list to get a view for
10549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         * @return the view for the specified item
10559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         */
10569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        @Override
10579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        protected View obtainView(int position) {
10589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            View view = super.obtainView(position);
10599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
10609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            if (view instanceof TextView) {
10619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                ((TextView) view).setHorizontallyScrolling(true);
10629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
10639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
10649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            return view;
10659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
10669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
10679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        /**
10689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         * <p>Returns the top padding of the currently selected view.</p>
10699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         *
10709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         * @return the height of the top padding for the selection
10719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         */
10729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        public int getSelectionPaddingTop() {
10739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            return mSelectionTopPadding;
10749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
10759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
10769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        /**
10779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         * <p>Returns the bottom padding of the currently selected view.</p>
10789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         *
10799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         * @return the height of the bottom padding for the selection
10809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         */
10819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        public int getSelectionPaddingBottom() {
10829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            return mSelectionBottomPadding;
10839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
10849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
10859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        /**
10869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         * <p>Returns the focus state in the drop down.</p>
10879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         *
10889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         * @return true always
10899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         */
10909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        @Override
10919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        public boolean hasWindowFocus() {
10929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            return true;
10939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
10949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
10959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        /**
10969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         * <p>Returns the focus state in the drop down.</p>
10979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         *
10989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         * @return true always
10999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         */
11009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        @Override
11019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        public boolean isFocused() {
11029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            return true;
11039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
11049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
11059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        /**
11069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         * <p>Returns the focus state in the drop down.</p>
11079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         *
11089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         * @return true always
11099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         */
11109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        @Override
11119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        public boolean hasFocus() {
11129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            return true;
11139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
11149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
11159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        protected int[] onCreateDrawableState(int extraSpace) {
11169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            int[] res = super.onCreateDrawableState(extraSpace);
1117c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project            //noinspection ConstantIfStatement
11189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            if (false) {
11199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                StringBuilder sb = new StringBuilder("Created drawable state: [");
11209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                for (int i=0; i<res.length; i++) {
11219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    if (i > 0) sb.append(", ");
11229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    sb.append("0x");
11239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    sb.append(Integer.toHexString(res[i]));
11249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                }
11259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                sb.append("]");
11269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                Log.i(TAG, sb.toString());
11279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
11289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            return res;
11299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
11309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
11319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
11329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
11339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * This interface is used to make sure that the text entered in this TextView complies to
11349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * a certain format.  Since there is no foolproof way to prevent the user from leaving
11359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * this View with an incorrect value in it, all we can do is try to fix it ourselves
11369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * when this happens.
11379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
11389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public interface Validator {
11399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        /**
11409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         * Validates the specified text.
11419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         *
11429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         * @return true If the text currently in the text editor is valid.
11439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         *
11449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         * @see #fixText(CharSequence)
11459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         */
11469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        boolean isValid(CharSequence text);
11479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
11489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        /**
11499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         * Corrects the specified text to make it valid.
11509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         *
11519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         * @param invalidText A string that doesn't pass validation: isValid(invalidText)
11529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         *        returns false
11539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         *
11549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         * @return A string based on invalidText such as invoking isValid() on it returns true.
11559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         *
11569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         * @see #isValid(CharSequence)
11579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         */
11589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        CharSequence fixText(CharSequence invalidText);
11599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
11609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
1161