1 /* 2 * Copyright (C) 2010 The Android Open Source Project 3 * 4 * Licensed under the Apache License, Version 2.0 (the "License"); 5 * you may not use this file except in compliance with the License. 6 * You may obtain a copy of the License at 7 * 8 * http://www.apache.org/licenses/LICENSE-2.0 9 * 10 * Unless required by applicable law or agreed to in writing, software 11 * distributed under the License is distributed on an "AS IS" BASIS, 12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13 * See the License for the specific language governing permissions and 14 * limitations under the License. 15 */ 16 17 package android.widget; 18 19 import android.annotation.AttrRes; 20 import android.annotation.NonNull; 21 import android.annotation.Nullable; 22 import android.annotation.StyleRes; 23 import android.compat.annotation.UnsupportedAppUsage; 24 import android.content.Context; 25 import android.content.res.TypedArray; 26 import android.database.DataSetObserver; 27 import android.graphics.Rect; 28 import android.graphics.drawable.Drawable; 29 import android.os.Build; 30 import android.os.Handler; 31 import android.util.AttributeSet; 32 import android.util.Log; 33 import android.view.Gravity; 34 import android.view.KeyEvent; 35 import android.view.MotionEvent; 36 import android.view.View; 37 import android.view.View.MeasureSpec; 38 import android.view.View.OnTouchListener; 39 import android.view.ViewGroup; 40 import android.view.ViewParent; 41 import android.view.WindowManager; 42 import android.widget.AdapterView.OnItemSelectedListener; 43 44 import com.android.internal.R; 45 import com.android.internal.view.menu.ShowableListMenu; 46 47 /** 48 * A ListPopupWindow anchors itself to a host view and displays a 49 * list of choices. 50 * 51 * <p>ListPopupWindow contains a number of tricky behaviors surrounding 52 * positioning, scrolling parents to fit the dropdown, interacting 53 * sanely with the IME if present, and others. 54 * 55 * @attr ref android.R.styleable#ListPopupWindow_dropDownVerticalOffset 56 * @attr ref android.R.styleable#ListPopupWindow_dropDownHorizontalOffset 57 * 58 * @see android.widget.AutoCompleteTextView 59 * @see android.widget.Spinner 60 */ 61 public class ListPopupWindow implements ShowableListMenu { 62 private static final String TAG = "ListPopupWindow"; 63 private static final boolean DEBUG = false; 64 65 /** 66 * This value controls the length of time that the user 67 * must leave a pointer down without scrolling to expand 68 * the autocomplete dropdown list to cover the IME. 69 */ 70 private static final int EXPAND_LIST_TIMEOUT = 250; 71 72 private Context mContext; 73 private ListAdapter mAdapter; 74 @UnsupportedAppUsage 75 private DropDownListView mDropDownList; 76 77 private int mDropDownHeight = ViewGroup.LayoutParams.WRAP_CONTENT; 78 private int mDropDownWidth = ViewGroup.LayoutParams.WRAP_CONTENT; 79 private int mDropDownHorizontalOffset; 80 private int mDropDownVerticalOffset; 81 private int mDropDownWindowLayoutType = WindowManager.LayoutParams.TYPE_APPLICATION_SUB_PANEL; 82 private boolean mDropDownVerticalOffsetSet; 83 private boolean mIsAnimatedFromAnchor = true; 84 private boolean mOverlapAnchor; 85 private boolean mOverlapAnchorSet; 86 87 private int mDropDownGravity = Gravity.NO_GRAVITY; 88 89 private boolean mDropDownAlwaysVisible = false; 90 private boolean mForceIgnoreOutsideTouch = false; 91 int mListItemExpandMaximum = Integer.MAX_VALUE; 92 93 private View mPromptView; 94 private int mPromptPosition = POSITION_PROMPT_ABOVE; 95 96 private DataSetObserver mObserver; 97 98 private View mDropDownAnchorView; 99 100 private Drawable mDropDownListHighlight; 101 102 private AdapterView.OnItemClickListener mItemClickListener; 103 private AdapterView.OnItemSelectedListener mItemSelectedListener; 104 105 private final ResizePopupRunnable mResizePopupRunnable = new ResizePopupRunnable(); 106 private final PopupTouchInterceptor mTouchInterceptor = new PopupTouchInterceptor(); 107 private final PopupScrollListener mScrollListener = new PopupScrollListener(); 108 private final ListSelectorHider mHideSelector = new ListSelectorHider(); 109 private Runnable mShowDropDownRunnable; 110 111 private final Handler mHandler; 112 113 private final Rect mTempRect = new Rect(); 114 115 /** 116 * Optional anchor-relative bounds to be used as the transition epicenter. 117 * When {@code null}, the anchor bounds are used as the epicenter. 118 */ 119 private Rect mEpicenterBounds; 120 121 private boolean mModal; 122 123 @UnsupportedAppUsage 124 PopupWindow mPopup; 125 126 /** 127 * The provided prompt view should appear above list content. 128 * 129 * @see #setPromptPosition(int) 130 * @see #getPromptPosition() 131 * @see #setPromptView(View) 132 */ 133 public static final int POSITION_PROMPT_ABOVE = 0; 134 135 /** 136 * The provided prompt view should appear below list content. 137 * 138 * @see #setPromptPosition(int) 139 * @see #getPromptPosition() 140 * @see #setPromptView(View) 141 */ 142 public static final int POSITION_PROMPT_BELOW = 1; 143 144 /** 145 * Alias for {@link ViewGroup.LayoutParams#MATCH_PARENT}. 146 * If used to specify a popup width, the popup will match the width of the anchor view. 147 * If used to specify a popup height, the popup will fill available space. 148 */ 149 public static final int MATCH_PARENT = ViewGroup.LayoutParams.MATCH_PARENT; 150 151 /** 152 * Alias for {@link ViewGroup.LayoutParams#WRAP_CONTENT}. 153 * If used to specify a popup width, the popup will use the width of its content. 154 */ 155 public static final int WRAP_CONTENT = ViewGroup.LayoutParams.WRAP_CONTENT; 156 157 /** 158 * Mode for {@link #setInputMethodMode(int)}: the requirements for the 159 * input method should be based on the focusability of the popup. That is 160 * if it is focusable than it needs to work with the input method, else 161 * it doesn't. 162 */ 163 public static final int INPUT_METHOD_FROM_FOCUSABLE = PopupWindow.INPUT_METHOD_FROM_FOCUSABLE; 164 165 /** 166 * Mode for {@link #setInputMethodMode(int)}: this popup always needs to 167 * work with an input method, regardless of whether it is focusable. This 168 * means that it will always be displayed so that the user can also operate 169 * the input method while it is shown. 170 */ 171 public static final int INPUT_METHOD_NEEDED = PopupWindow.INPUT_METHOD_NEEDED; 172 173 /** 174 * Mode for {@link #setInputMethodMode(int)}: this popup never needs to 175 * work with an input method, regardless of whether it is focusable. This 176 * means that it will always be displayed to use as much space on the 177 * screen as needed, regardless of whether this covers the input method. 178 */ 179 public static final int INPUT_METHOD_NOT_NEEDED = PopupWindow.INPUT_METHOD_NOT_NEEDED; 180 181 /** 182 * Create a new, empty popup window capable of displaying items from a ListAdapter. 183 * Backgrounds should be set using {@link #setBackgroundDrawable(Drawable)}. 184 * 185 * @param context Context used for contained views. 186 */ ListPopupWindow(@onNull Context context)187 public ListPopupWindow(@NonNull Context context) { 188 this(context, null, com.android.internal.R.attr.listPopupWindowStyle, 0); 189 } 190 191 /** 192 * Create a new, empty popup window capable of displaying items from a ListAdapter. 193 * Backgrounds should be set using {@link #setBackgroundDrawable(Drawable)}. 194 * 195 * @param context Context used for contained views. 196 * @param attrs Attributes from inflating parent views used to style the popup. 197 */ ListPopupWindow(@onNull Context context, @Nullable AttributeSet attrs)198 public ListPopupWindow(@NonNull Context context, @Nullable AttributeSet attrs) { 199 this(context, attrs, com.android.internal.R.attr.listPopupWindowStyle, 0); 200 } 201 202 /** 203 * Create a new, empty popup window capable of displaying items from a ListAdapter. 204 * Backgrounds should be set using {@link #setBackgroundDrawable(Drawable)}. 205 * 206 * @param context Context used for contained views. 207 * @param attrs Attributes from inflating parent views used to style the popup. 208 * @param defStyleAttr Default style attribute to use for popup content. 209 */ ListPopupWindow(@onNull Context context, @Nullable AttributeSet attrs, @AttrRes int defStyleAttr)210 public ListPopupWindow(@NonNull Context context, @Nullable AttributeSet attrs, 211 @AttrRes int defStyleAttr) { 212 this(context, attrs, defStyleAttr, 0); 213 } 214 215 /** 216 * Create a new, empty popup window capable of displaying items from a ListAdapter. 217 * Backgrounds should be set using {@link #setBackgroundDrawable(Drawable)}. 218 * 219 * @param context Context used for contained views. 220 * @param attrs Attributes from inflating parent views used to style the popup. 221 * @param defStyleAttr Style attribute to read for default styling of popup content. 222 * @param defStyleRes Style resource ID to use for default styling of popup content. 223 */ ListPopupWindow(@onNull Context context, @Nullable AttributeSet attrs, @AttrRes int defStyleAttr, @StyleRes int defStyleRes)224 public ListPopupWindow(@NonNull Context context, @Nullable AttributeSet attrs, 225 @AttrRes int defStyleAttr, @StyleRes int defStyleRes) { 226 mContext = context; 227 mHandler = new Handler(context.getMainLooper()); 228 229 final TypedArray a = context.obtainStyledAttributes(attrs, R.styleable.ListPopupWindow, 230 defStyleAttr, defStyleRes); 231 mDropDownHorizontalOffset = a.getDimensionPixelOffset( 232 R.styleable.ListPopupWindow_dropDownHorizontalOffset, 0); 233 mDropDownVerticalOffset = a.getDimensionPixelOffset( 234 R.styleable.ListPopupWindow_dropDownVerticalOffset, 0); 235 if (mDropDownVerticalOffset != 0) { 236 mDropDownVerticalOffsetSet = true; 237 } 238 a.recycle(); 239 240 mPopup = new PopupWindow(context, attrs, defStyleAttr, defStyleRes); 241 mPopup.setInputMethodMode(PopupWindow.INPUT_METHOD_NEEDED); 242 } 243 244 /** 245 * Sets the adapter that provides the data and the views to represent the data 246 * in this popup window. 247 * 248 * @param adapter The adapter to use to create this window's content. 249 */ setAdapter(@ullable ListAdapter adapter)250 public void setAdapter(@Nullable ListAdapter adapter) { 251 if (mObserver == null) { 252 mObserver = new PopupDataSetObserver(); 253 } else if (mAdapter != null) { 254 mAdapter.unregisterDataSetObserver(mObserver); 255 } 256 mAdapter = adapter; 257 if (mAdapter != null) { 258 adapter.registerDataSetObserver(mObserver); 259 } 260 261 if (mDropDownList != null) { 262 mDropDownList.setAdapter(mAdapter); 263 } 264 } 265 266 /** 267 * Set where the optional prompt view should appear. The default is 268 * {@link #POSITION_PROMPT_ABOVE}. 269 * 270 * @param position A position constant declaring where the prompt should be displayed. 271 * 272 * @see #POSITION_PROMPT_ABOVE 273 * @see #POSITION_PROMPT_BELOW 274 */ setPromptPosition(int position)275 public void setPromptPosition(int position) { 276 mPromptPosition = position; 277 } 278 279 /** 280 * @return Where the optional prompt view should appear. 281 * 282 * @see #POSITION_PROMPT_ABOVE 283 * @see #POSITION_PROMPT_BELOW 284 */ getPromptPosition()285 public int getPromptPosition() { 286 return mPromptPosition; 287 } 288 289 /** 290 * Set whether this window should be modal when shown. 291 * 292 * <p>If a popup window is modal, it will receive all touch and key input. 293 * If the user touches outside the popup window's content area the popup window 294 * will be dismissed. 295 * 296 * @param modal {@code true} if the popup window should be modal, {@code false} otherwise. 297 */ setModal(boolean modal)298 public void setModal(boolean modal) { 299 mModal = modal; 300 mPopup.setFocusable(modal); 301 } 302 303 /** 304 * Returns whether the popup window will be modal when shown. 305 * 306 * @return {@code true} if the popup window will be modal, {@code false} otherwise. 307 */ isModal()308 public boolean isModal() { 309 return mModal; 310 } 311 312 /** 313 * Forces outside touches to be ignored. Normally if {@link #isDropDownAlwaysVisible()} is 314 * false, we allow outside touch to dismiss the dropdown. If this is set to true, then we 315 * ignore outside touch even when the drop down is not set to always visible. 316 * 317 * @hide Used only by AutoCompleteTextView to handle some internal special cases. 318 */ 319 @UnsupportedAppUsage setForceIgnoreOutsideTouch(boolean forceIgnoreOutsideTouch)320 public void setForceIgnoreOutsideTouch(boolean forceIgnoreOutsideTouch) { 321 mForceIgnoreOutsideTouch = forceIgnoreOutsideTouch; 322 } 323 324 /** 325 * Sets whether the drop-down should remain visible under certain conditions. 326 * 327 * The drop-down will occupy the entire screen below {@link #getAnchorView} regardless 328 * of the size or content of the list. {@link #getBackground()} will fill any space 329 * that is not used by the list. 330 * 331 * @param dropDownAlwaysVisible Whether to keep the drop-down visible. 332 * 333 * @hide Only used by AutoCompleteTextView under special conditions. 334 */ 335 @UnsupportedAppUsage(maxTargetSdk = Build.VERSION_CODES.R, trackingBug = 170729553) setDropDownAlwaysVisible(boolean dropDownAlwaysVisible)336 public void setDropDownAlwaysVisible(boolean dropDownAlwaysVisible) { 337 mDropDownAlwaysVisible = dropDownAlwaysVisible; 338 } 339 340 /** 341 * @return Whether the drop-down is visible under special conditions. 342 * 343 * @hide Only used by AutoCompleteTextView under special conditions. 344 */ 345 @UnsupportedAppUsage(maxTargetSdk = Build.VERSION_CODES.R, trackingBug = 170729553) isDropDownAlwaysVisible()346 public boolean isDropDownAlwaysVisible() { 347 return mDropDownAlwaysVisible; 348 } 349 350 /** 351 * Sets the operating mode for the soft input area. 352 * 353 * @param mode The desired mode, see 354 * {@link android.view.WindowManager.LayoutParams#softInputMode} 355 * for the full list 356 * 357 * @see android.view.WindowManager.LayoutParams#softInputMode 358 * @see #getSoftInputMode() 359 */ setSoftInputMode(int mode)360 public void setSoftInputMode(int mode) { 361 mPopup.setSoftInputMode(mode); 362 } 363 364 /** 365 * Returns the current value in {@link #setSoftInputMode(int)}. 366 * 367 * @see #setSoftInputMode(int) 368 * @see android.view.WindowManager.LayoutParams#softInputMode 369 */ getSoftInputMode()370 public int getSoftInputMode() { 371 return mPopup.getSoftInputMode(); 372 } 373 374 /** 375 * Sets a drawable to use as the list item selector. 376 * 377 * @param selector List selector drawable to use in the popup. 378 */ setListSelector(Drawable selector)379 public void setListSelector(Drawable selector) { 380 mDropDownListHighlight = selector; 381 } 382 383 /** 384 * @return The background drawable for the popup window. 385 */ getBackground()386 public @Nullable Drawable getBackground() { 387 return mPopup.getBackground(); 388 } 389 390 /** 391 * Sets a drawable to be the background for the popup window. 392 * 393 * @param d A drawable to set as the background. 394 */ setBackgroundDrawable(@ullable Drawable d)395 public void setBackgroundDrawable(@Nullable Drawable d) { 396 mPopup.setBackgroundDrawable(d); 397 } 398 399 /** 400 * Set an animation style to use when the popup window is shown or dismissed. 401 * 402 * @param animationStyle Animation style to use. 403 */ setAnimationStyle(@tyleRes int animationStyle)404 public void setAnimationStyle(@StyleRes int animationStyle) { 405 mPopup.setAnimationStyle(animationStyle); 406 } 407 408 /** 409 * Returns the animation style that will be used when the popup window is 410 * shown or dismissed. 411 * 412 * @return Animation style that will be used. 413 */ getAnimationStyle()414 public @StyleRes int getAnimationStyle() { 415 return mPopup.getAnimationStyle(); 416 } 417 418 /** 419 * Returns the view that will be used to anchor this popup. 420 * 421 * @return The popup's anchor view 422 */ getAnchorView()423 public @Nullable View getAnchorView() { 424 return mDropDownAnchorView; 425 } 426 427 /** 428 * Sets the popup's anchor view. This popup will always be positioned relative to 429 * the anchor view when shown. 430 * 431 * @param anchor The view to use as an anchor. 432 */ setAnchorView(@ullable View anchor)433 public void setAnchorView(@Nullable View anchor) { 434 mDropDownAnchorView = anchor; 435 } 436 437 /** 438 * @return The horizontal offset of the popup from its anchor in pixels. 439 * @attr ref android.R.styleable#ListPopupWindow_dropDownHorizontalOffset 440 */ getHorizontalOffset()441 public int getHorizontalOffset() { 442 return mDropDownHorizontalOffset; 443 } 444 445 /** 446 * Set the horizontal offset of this popup from its anchor view in pixels. 447 * 448 * @param offset The horizontal offset of the popup from its anchor. 449 * @attr ref android.R.styleable#ListPopupWindow_dropDownHorizontalOffset 450 */ setHorizontalOffset(int offset)451 public void setHorizontalOffset(int offset) { 452 mDropDownHorizontalOffset = offset; 453 } 454 455 /** 456 * @return The vertical offset of the popup from its anchor in pixels. 457 * @attr ref android.R.styleable#ListPopupWindow_dropDownVerticalOffset 458 */ getVerticalOffset()459 public int getVerticalOffset() { 460 if (!mDropDownVerticalOffsetSet) { 461 return 0; 462 } 463 return mDropDownVerticalOffset; 464 } 465 466 /** 467 * Set the vertical offset of this popup from its anchor view in pixels. 468 * 469 * @param offset The vertical offset of the popup from its anchor. 470 * @attr ref android.R.styleable#ListPopupWindow_dropDownVerticalOffset 471 */ setVerticalOffset(int offset)472 public void setVerticalOffset(int offset) { 473 mDropDownVerticalOffset = offset; 474 mDropDownVerticalOffsetSet = true; 475 } 476 477 /** 478 * Specifies the anchor-relative bounds of the popup's transition 479 * epicenter. 480 * 481 * @param bounds anchor-relative bounds, or {@code null} to use default epicenter 482 * 483 * @see #getEpicenterBounds() 484 */ setEpicenterBounds(@ullable Rect bounds)485 public void setEpicenterBounds(@Nullable Rect bounds) { 486 mEpicenterBounds = bounds != null ? new Rect(bounds) : null; 487 } 488 489 /** 490 * Returns bounds which are used as a popup's epicenter 491 * of the enter and exit transitions. 492 * 493 * @return bounds relative to anchor view, or {@code null} if not set 494 * @see #setEpicenterBounds(Rect) 495 */ 496 @Nullable getEpicenterBounds()497 public Rect getEpicenterBounds() { 498 return mEpicenterBounds != null ? new Rect(mEpicenterBounds) : null; 499 } 500 501 /** 502 * Set the gravity of the dropdown list. This is commonly used to 503 * set gravity to START or END for alignment with the anchor. 504 * 505 * @param gravity Gravity value to use 506 */ setDropDownGravity(int gravity)507 public void setDropDownGravity(int gravity) { 508 mDropDownGravity = gravity; 509 } 510 511 /** 512 * @return The width of the popup window in pixels. 513 */ getWidth()514 public int getWidth() { 515 return mDropDownWidth; 516 } 517 518 /** 519 * Sets the width of the popup window in pixels. Can also be {@link #MATCH_PARENT} 520 * or {@link #WRAP_CONTENT}. 521 * 522 * @param width Width of the popup window. 523 */ setWidth(int width)524 public void setWidth(int width) { 525 mDropDownWidth = width; 526 } 527 528 /** 529 * Sets the width of the popup window by the size of its content. The final width may be 530 * larger to accommodate styled window dressing. 531 * 532 * @param width Desired width of content in pixels. 533 */ setContentWidth(int width)534 public void setContentWidth(int width) { 535 Drawable popupBackground = mPopup.getBackground(); 536 if (popupBackground != null) { 537 popupBackground.getPadding(mTempRect); 538 mDropDownWidth = mTempRect.left + mTempRect.right + width; 539 } else { 540 setWidth(width); 541 } 542 } 543 544 /** 545 * @return The height of the popup window in pixels. 546 */ getHeight()547 public int getHeight() { 548 return mDropDownHeight; 549 } 550 551 /** 552 * Sets the height of the popup window in pixels. Can also be {@link #MATCH_PARENT}. 553 * 554 * @param height Height of the popup window must be a positive value, 555 * {@link #MATCH_PARENT}, or {@link #WRAP_CONTENT}. 556 * 557 * @throws IllegalArgumentException if height is set to negative value 558 */ setHeight(int height)559 public void setHeight(int height) { 560 if (height < 0 && ViewGroup.LayoutParams.WRAP_CONTENT != height 561 && ViewGroup.LayoutParams.MATCH_PARENT != height) { 562 if (mContext.getApplicationInfo().targetSdkVersion 563 < Build.VERSION_CODES.O) { 564 Log.e(TAG, "Negative value " + height + " passed to ListPopupWindow#setHeight" 565 + " produces undefined results"); 566 } else { 567 throw new IllegalArgumentException( 568 "Invalid height. Must be a positive value, MATCH_PARENT, or WRAP_CONTENT."); 569 } 570 } 571 mDropDownHeight = height; 572 } 573 574 /** 575 * Set the layout type for this popup window. 576 * <p> 577 * See {@link WindowManager.LayoutParams#type} for possible values. 578 * 579 * @param layoutType Layout type for this window. 580 * 581 * @see WindowManager.LayoutParams#type 582 */ setWindowLayoutType(int layoutType)583 public void setWindowLayoutType(int layoutType) { 584 mDropDownWindowLayoutType = layoutType; 585 } 586 587 /** 588 * Sets a listener to receive events when a list item is clicked. 589 * 590 * @param clickListener Listener to register 591 * 592 * @see ListView#setOnItemClickListener(android.widget.AdapterView.OnItemClickListener) 593 */ setOnItemClickListener(@ullable AdapterView.OnItemClickListener clickListener)594 public void setOnItemClickListener(@Nullable AdapterView.OnItemClickListener clickListener) { 595 mItemClickListener = clickListener; 596 } 597 598 /** 599 * Sets a listener to receive events when a list item is selected. 600 * 601 * @param selectedListener Listener to register. 602 * 603 * @see ListView#setOnItemSelectedListener(OnItemSelectedListener) 604 */ setOnItemSelectedListener(@ullable OnItemSelectedListener selectedListener)605 public void setOnItemSelectedListener(@Nullable OnItemSelectedListener selectedListener) { 606 mItemSelectedListener = selectedListener; 607 } 608 609 /** 610 * Set a view to act as a user prompt for this popup window. Where the prompt view will appear 611 * is controlled by {@link #setPromptPosition(int)}. 612 * 613 * @param prompt View to use as an informational prompt. 614 */ setPromptView(@ullable View prompt)615 public void setPromptView(@Nullable View prompt) { 616 boolean showing = isShowing(); 617 if (showing) { 618 removePromptView(); 619 } 620 mPromptView = prompt; 621 if (showing) { 622 show(); 623 } 624 } 625 626 /** 627 * Post a {@link #show()} call to the UI thread. 628 */ postShow()629 public void postShow() { 630 mHandler.post(mShowDropDownRunnable); 631 } 632 633 /** 634 * Show the popup list. If the list is already showing, this method 635 * will recalculate the popup's size and position. 636 */ 637 @Override show()638 public void show() { 639 int height = buildDropDown(); 640 641 final boolean noInputMethod = isInputMethodNotNeeded(); 642 mPopup.setAllowScrollingAnchorParent(!noInputMethod); 643 mPopup.setWindowLayoutType(mDropDownWindowLayoutType); 644 645 if (mPopup.isShowing()) { 646 if (!getAnchorView().isAttachedToWindow()) { 647 //Don't update position if the anchor view is detached from window. 648 return; 649 } 650 final int widthSpec; 651 if (mDropDownWidth == ViewGroup.LayoutParams.MATCH_PARENT) { 652 // The call to PopupWindow's update method below can accept -1 for any 653 // value you do not want to update. 654 widthSpec = -1; 655 } else if (mDropDownWidth == ViewGroup.LayoutParams.WRAP_CONTENT) { 656 widthSpec = getAnchorView().getWidth(); 657 } else { 658 widthSpec = mDropDownWidth; 659 } 660 661 final int heightSpec; 662 if (mDropDownHeight == ViewGroup.LayoutParams.MATCH_PARENT) { 663 // The call to PopupWindow's update method below can accept -1 for any 664 // value you do not want to update. 665 heightSpec = noInputMethod ? height : ViewGroup.LayoutParams.MATCH_PARENT; 666 if (noInputMethod) { 667 mPopup.setWidth(mDropDownWidth == ViewGroup.LayoutParams.MATCH_PARENT ? 668 ViewGroup.LayoutParams.MATCH_PARENT : 0); 669 mPopup.setHeight(0); 670 } else { 671 mPopup.setWidth(mDropDownWidth == ViewGroup.LayoutParams.MATCH_PARENT ? 672 ViewGroup.LayoutParams.MATCH_PARENT : 0); 673 mPopup.setHeight(ViewGroup.LayoutParams.MATCH_PARENT); 674 } 675 } else if (mDropDownHeight == ViewGroup.LayoutParams.WRAP_CONTENT) { 676 heightSpec = height; 677 } else { 678 heightSpec = mDropDownHeight; 679 } 680 681 mPopup.setOutsideTouchable(!mForceIgnoreOutsideTouch && !mDropDownAlwaysVisible); 682 683 mPopup.update(getAnchorView(), mDropDownHorizontalOffset, 684 mDropDownVerticalOffset, (widthSpec < 0)? -1 : widthSpec, 685 (heightSpec < 0)? -1 : heightSpec); 686 mPopup.getContentView().restoreDefaultFocus(); 687 } else { 688 final int widthSpec; 689 if (mDropDownWidth == ViewGroup.LayoutParams.MATCH_PARENT) { 690 widthSpec = ViewGroup.LayoutParams.MATCH_PARENT; 691 } else { 692 if (mDropDownWidth == ViewGroup.LayoutParams.WRAP_CONTENT) { 693 widthSpec = getAnchorView().getWidth(); 694 } else { 695 widthSpec = mDropDownWidth; 696 } 697 } 698 699 final int heightSpec; 700 if (mDropDownHeight == ViewGroup.LayoutParams.MATCH_PARENT) { 701 heightSpec = ViewGroup.LayoutParams.MATCH_PARENT; 702 } else { 703 if (mDropDownHeight == ViewGroup.LayoutParams.WRAP_CONTENT) { 704 heightSpec = height; 705 } else { 706 heightSpec = mDropDownHeight; 707 } 708 } 709 710 mPopup.setWidth(widthSpec); 711 mPopup.setHeight(heightSpec); 712 mPopup.setIsClippedToScreen(true); 713 714 // use outside touchable to dismiss drop down when touching outside of it, so 715 // only set this if the dropdown is not always visible 716 mPopup.setOutsideTouchable(!mForceIgnoreOutsideTouch && !mDropDownAlwaysVisible); 717 mPopup.setTouchInterceptor(mTouchInterceptor); 718 mPopup.setEpicenterBounds(mEpicenterBounds); 719 if (mOverlapAnchorSet) { 720 mPopup.setOverlapAnchor(mOverlapAnchor); 721 } 722 mPopup.showAsDropDown(getAnchorView(), mDropDownHorizontalOffset, 723 mDropDownVerticalOffset, mDropDownGravity); 724 mDropDownList.setSelection(ListView.INVALID_POSITION); 725 mPopup.getContentView().restoreDefaultFocus(); 726 727 if (!mModal || mDropDownList.isInTouchMode()) { 728 clearListSelection(); 729 } 730 if (!mModal) { 731 mHandler.post(mHideSelector); 732 } 733 } 734 } 735 736 /** 737 * Dismiss the popup window. 738 */ 739 @Override 740 public void dismiss() { 741 mPopup.dismiss(); 742 removePromptView(); 743 mPopup.setContentView(null); 744 mDropDownList = null; 745 mHandler.removeCallbacks(mResizePopupRunnable); 746 } 747 748 /** 749 * Remove existing exit transition from PopupWindow and force immediate dismissal. 750 * @hide 751 */ 752 public void dismissImmediate() { 753 mPopup.setExitTransition(null); 754 dismiss(); 755 } 756 757 /** 758 * Set a listener to receive a callback when the popup is dismissed. 759 * 760 * @param listener Listener that will be notified when the popup is dismissed. 761 */ 762 public void setOnDismissListener(@Nullable PopupWindow.OnDismissListener listener) { 763 mPopup.setOnDismissListener(listener); 764 } 765 766 private void removePromptView() { 767 if (mPromptView != null) { 768 final ViewParent parent = mPromptView.getParent(); 769 if (parent instanceof ViewGroup) { 770 final ViewGroup group = (ViewGroup) parent; 771 group.removeView(mPromptView); 772 } 773 } 774 } 775 776 /** 777 * Control how the popup operates with an input method: one of 778 * {@link #INPUT_METHOD_FROM_FOCUSABLE}, {@link #INPUT_METHOD_NEEDED}, 779 * or {@link #INPUT_METHOD_NOT_NEEDED}. 780 * 781 * <p>If the popup is showing, calling this method will take effect only 782 * the next time the popup is shown or through a manual call to the {@link #show()} 783 * method.</p> 784 * 785 * @see #getInputMethodMode() 786 * @see #show() 787 */ 788 public void setInputMethodMode(int mode) { 789 mPopup.setInputMethodMode(mode); 790 } 791 792 /** 793 * Return the current value in {@link #setInputMethodMode(int)}. 794 * 795 * @see #setInputMethodMode(int) 796 */ 797 public int getInputMethodMode() { 798 return mPopup.getInputMethodMode(); 799 } 800 801 /** 802 * Set the selected position of the list. 803 * Only valid when {@link #isShowing()} == {@code true}. 804 * 805 * @param position List position to set as selected. 806 */ 807 public void setSelection(int position) { 808 DropDownListView list = mDropDownList; 809 if (isShowing() && list != null) { 810 list.setListSelectionHidden(false); 811 list.setSelection(position); 812 if (list.getChoiceMode() != ListView.CHOICE_MODE_NONE) { 813 list.setItemChecked(position, true); 814 } 815 } 816 } 817 818 /** 819 * Clear any current list selection. 820 * Only valid when {@link #isShowing()} == {@code true}. 821 */ 822 public void clearListSelection() { 823 final DropDownListView list = mDropDownList; 824 if (list != null) { 825 // WARNING: Please read the comment where mListSelectionHidden is declared 826 list.setListSelectionHidden(true); 827 list.hideSelector(); 828 list.requestLayout(); 829 } 830 } 831 832 /** 833 * @return {@code true} if the popup is currently showing, {@code false} otherwise. 834 */ 835 @Override 836 public boolean isShowing() { 837 return mPopup.isShowing(); 838 } 839 840 /** 841 * @return {@code true} if this popup is configured to assume the user does not need 842 * to interact with the IME while it is showing, {@code false} otherwise. 843 */ 844 public boolean isInputMethodNotNeeded() { 845 return mPopup.getInputMethodMode() == INPUT_METHOD_NOT_NEEDED; 846 } 847 848 /** 849 * Perform an item click operation on the specified list adapter position. 850 * 851 * @param position Adapter position for performing the click 852 * @return true if the click action could be performed, false if not. 853 * (e.g. if the popup was not showing, this method would return false.) 854 */ 855 public boolean performItemClick(int position) { 856 if (isShowing()) { 857 if (mItemClickListener != null) { 858 final DropDownListView list = mDropDownList; 859 final View child = list.getChildAt(position - list.getFirstVisiblePosition()); 860 final ListAdapter adapter = list.getAdapter(); 861 mItemClickListener.onItemClick(list, child, position, adapter.getItemId(position)); 862 } 863 return true; 864 } 865 return false; 866 } 867 868 /** 869 * @return The currently selected item or null if the popup is not showing. 870 */ 871 public @Nullable Object getSelectedItem() { 872 if (!isShowing()) { 873 return null; 874 } 875 return mDropDownList.getSelectedItem(); 876 } 877 878 /** 879 * @return The position of the currently selected item or {@link ListView#INVALID_POSITION} 880 * if {@link #isShowing()} == {@code false}. 881 * 882 * @see ListView#getSelectedItemPosition() 883 */ 884 public int getSelectedItemPosition() { 885 if (!isShowing()) { 886 return ListView.INVALID_POSITION; 887 } 888 return mDropDownList.getSelectedItemPosition(); 889 } 890 891 /** 892 * @return The ID of the currently selected item or {@link ListView#INVALID_ROW_ID} 893 * if {@link #isShowing()} == {@code false}. 894 * 895 * @see ListView#getSelectedItemId() 896 */ 897 public long getSelectedItemId() { 898 if (!isShowing()) { 899 return ListView.INVALID_ROW_ID; 900 } 901 return mDropDownList.getSelectedItemId(); 902 } 903 904 /** 905 * @return The View for the currently selected item or null if 906 * {@link #isShowing()} == {@code false}. 907 * 908 * @see ListView#getSelectedView() 909 */ 910 public @Nullable View getSelectedView() { 911 if (!isShowing()) { 912 return null; 913 } 914 return mDropDownList.getSelectedView(); 915 } 916 917 /** 918 * @return The {@link ListView} displayed within the popup window. 919 * Only valid when {@link #isShowing()} == {@code true}. 920 */ 921 @Override 922 public @Nullable ListView getListView() { 923 return mDropDownList; 924 } 925 926 @NonNull DropDownListView createDropDownListView(Context context, boolean hijackFocus) { 927 return new DropDownListView(context, hijackFocus); 928 } 929 930 /** 931 * The maximum number of list items that can be visible and still have 932 * the list expand when touched. 933 * 934 * @param max Max number of items that can be visible and still allow the list to expand. 935 */ 936 @UnsupportedAppUsage(maxTargetSdk = Build.VERSION_CODES.R, trackingBug = 170729553) 937 void setListItemExpandMax(int max) { 938 mListItemExpandMaximum = max; 939 } 940 941 /** 942 * Filter key down events. By forwarding key down events to this function, 943 * views using non-modal ListPopupWindow can have it handle key selection of items. 944 * 945 * @param keyCode keyCode param passed to the host view's onKeyDown 946 * @param event event param passed to the host view's onKeyDown 947 * @return true if the event was handled, false if it was ignored. 948 * 949 * @see #setModal(boolean) 950 * @see #onKeyUp(int, KeyEvent) 951 */ 952 public boolean onKeyDown(int keyCode, @NonNull KeyEvent event) { 953 // when the drop down is shown, we drive it directly 954 if (isShowing()) { 955 // the key events are forwarded to the list in the drop down view 956 // note that ListView handles space but we don't want that to happen 957 // also if selection is not currently in the drop down, then don't 958 // let center or enter presses go there since that would cause it 959 // to select one of its items 960 if (keyCode != KeyEvent.KEYCODE_SPACE 961 && (mDropDownList.getSelectedItemPosition() >= 0 962 || !KeyEvent.isConfirmKey(keyCode))) { 963 int curIndex = mDropDownList.getSelectedItemPosition(); 964 boolean consumed; 965 966 final boolean below = !mPopup.isAboveAnchor(); 967 968 final ListAdapter adapter = mAdapter; 969 970 boolean allEnabled; 971 int firstItem = Integer.MAX_VALUE; 972 int lastItem = Integer.MIN_VALUE; 973 974 if (adapter != null) { 975 allEnabled = adapter.areAllItemsEnabled(); 976 firstItem = allEnabled ? 0 : 977 mDropDownList.lookForSelectablePosition(0, true); 978 lastItem = allEnabled ? adapter.getCount() - 1 : 979 mDropDownList.lookForSelectablePosition(adapter.getCount() - 1, false); 980 } 981 982 if ((below && keyCode == KeyEvent.KEYCODE_DPAD_UP && curIndex <= firstItem) || 983 (!below && keyCode == KeyEvent.KEYCODE_DPAD_DOWN && curIndex >= lastItem)) { 984 // When the selection is at the top, we block the key 985 // event to prevent focus from moving. 986 clearListSelection(); 987 mPopup.setInputMethodMode(PopupWindow.INPUT_METHOD_NEEDED); 988 show(); 989 return true; 990 } else { 991 // WARNING: Please read the comment where mListSelectionHidden 992 // is declared 993 mDropDownList.setListSelectionHidden(false); 994 } 995 996 consumed = mDropDownList.onKeyDown(keyCode, event); 997 if (DEBUG) Log.v(TAG, "Key down: code=" + keyCode + " list consumed=" + consumed); 998 999 if (consumed) { 1000 // If it handled the key event, then the user is 1001 // navigating in the list, so we should put it in front. 1002 mPopup.setInputMethodMode(PopupWindow.INPUT_METHOD_NOT_NEEDED); 1003 // Here's a little trick we need to do to make sure that 1004 // the list view is actually showing its focus indicator, 1005 // by ensuring it has focus and getting its window out 1006 // of touch mode. 1007 mDropDownList.requestFocusFromTouch(); 1008 show(); 1009 1010 switch (keyCode) { 1011 // avoid passing the focus from the text view to the 1012 // next component 1013 case KeyEvent.KEYCODE_ENTER: 1014 case KeyEvent.KEYCODE_DPAD_CENTER: 1015 case KeyEvent.KEYCODE_DPAD_DOWN: 1016 case KeyEvent.KEYCODE_DPAD_UP: 1017 case KeyEvent.KEYCODE_NUMPAD_ENTER: 1018 return true; 1019 } 1020 } else { 1021 if (below && keyCode == KeyEvent.KEYCODE_DPAD_DOWN) { 1022 // when the selection is at the bottom, we block the 1023 // event to avoid going to the next focusable widget 1024 if (curIndex == lastItem) { 1025 return true; 1026 } 1027 } else if (!below && keyCode == KeyEvent.KEYCODE_DPAD_UP && 1028 curIndex == firstItem) { 1029 return true; 1030 } 1031 } 1032 } 1033 } 1034 1035 return false; 1036 } 1037 1038 /** 1039 * Filter key up events. By forwarding key up events to this function, 1040 * views using non-modal ListPopupWindow can have it handle key selection of items. 1041 * 1042 * @param keyCode keyCode param passed to the host view's onKeyUp 1043 * @param event event param passed to the host view's onKeyUp 1044 * @return true if the event was handled, false if it was ignored. 1045 * 1046 * @see #setModal(boolean) 1047 * @see #onKeyDown(int, KeyEvent) 1048 */ 1049 public boolean onKeyUp(int keyCode, @NonNull KeyEvent event) { 1050 if (isShowing() && mDropDownList.getSelectedItemPosition() >= 0) { 1051 boolean consumed = mDropDownList.onKeyUp(keyCode, event); 1052 if (consumed && KeyEvent.isConfirmKey(keyCode)) { 1053 // if the list accepts the key events and the key event was a click, the text view 1054 // gets the selected item from the drop down as its content 1055 dismiss(); 1056 } 1057 return consumed; 1058 } 1059 return false; 1060 } 1061 1062 /** 1063 * Filter pre-IME key events. By forwarding {@link View#onKeyPreIme(int, KeyEvent)} 1064 * events to this function, views using ListPopupWindow can have it dismiss the popup 1065 * when the back key is pressed. 1066 * 1067 * @param keyCode keyCode param passed to the host view's onKeyPreIme 1068 * @param event event param passed to the host view's onKeyPreIme 1069 * @return true if the event was handled, false if it was ignored. 1070 * 1071 * @see #setModal(boolean) 1072 */ onKeyPreIme(int keyCode, @NonNull KeyEvent event)1073 public boolean onKeyPreIme(int keyCode, @NonNull KeyEvent event) { 1074 if ((keyCode == KeyEvent.KEYCODE_BACK 1075 || keyCode == KeyEvent.KEYCODE_ESCAPE) && isShowing()) { 1076 // special case for the back key, we do not even try to send it 1077 // to the drop down list but instead, consume it immediately 1078 final View anchorView = mDropDownAnchorView; 1079 if (event.getAction() == KeyEvent.ACTION_DOWN && event.getRepeatCount() == 0) { 1080 KeyEvent.DispatcherState state = anchorView.getKeyDispatcherState(); 1081 if (state != null) { 1082 state.startTracking(event, this); 1083 } 1084 return true; 1085 } else if (event.getAction() == KeyEvent.ACTION_UP) { 1086 KeyEvent.DispatcherState state = anchorView.getKeyDispatcherState(); 1087 if (state != null) { 1088 state.handleUpEvent(event); 1089 } 1090 if (event.isTracking() && !event.isCanceled()) { 1091 dismiss(); 1092 return true; 1093 } 1094 } 1095 } 1096 return false; 1097 } 1098 1099 /** 1100 * Returns an {@link OnTouchListener} that can be added to the source view 1101 * to implement drag-to-open behavior. Generally, the source view should be 1102 * the same view that was passed to {@link #setAnchorView}. 1103 * <p> 1104 * When the listener is set on a view, touching that view and dragging 1105 * outside of its bounds will open the popup window. Lifting will select the 1106 * currently touched list item. 1107 * <p> 1108 * Example usage: 1109 * <pre> 1110 * ListPopupWindow myPopup = new ListPopupWindow(context); 1111 * myPopup.setAnchor(myAnchor); 1112 * OnTouchListener dragListener = myPopup.createDragToOpenListener(myAnchor); 1113 * myAnchor.setOnTouchListener(dragListener); 1114 * </pre> 1115 * 1116 * @param src the view on which the resulting listener will be set 1117 * @return a touch listener that controls drag-to-open behavior 1118 */ createDragToOpenListener(View src)1119 public OnTouchListener createDragToOpenListener(View src) { 1120 return new ForwardingListener(src) { 1121 @Override 1122 public ShowableListMenu getPopup() { 1123 return ListPopupWindow.this; 1124 } 1125 }; 1126 } 1127 1128 /** 1129 * <p>Builds the popup window's content and returns the height the popup 1130 * should have. Returns -1 when the content already exists.</p> 1131 * 1132 * @return the content's height or -1 if content already exists 1133 */ 1134 @UnsupportedAppUsage(maxTargetSdk = Build.VERSION_CODES.R, trackingBug = 170729553) buildDropDown()1135 private int buildDropDown() { 1136 ViewGroup dropDownView; 1137 int otherHeights = 0; 1138 1139 if (mDropDownList == null) { 1140 Context context = mContext; 1141 1142 /** 1143 * This Runnable exists for the sole purpose of checking if the view layout has got 1144 * completed and if so call showDropDown to display the drop down. This is used to show 1145 * the drop down as soon as possible after user opens up the search dialog, without 1146 * waiting for the normal UI pipeline to do it's job which is slower than this method. 1147 */ 1148 mShowDropDownRunnable = new Runnable() { 1149 public void run() { 1150 // View layout should be all done before displaying the drop down. 1151 View view = getAnchorView(); 1152 if (view != null && view.getWindowToken() != null) { 1153 show(); 1154 } 1155 } 1156 }; 1157 1158 mDropDownList = createDropDownListView(context, !mModal); 1159 if (mDropDownListHighlight != null) { 1160 mDropDownList.setSelector(mDropDownListHighlight); 1161 } 1162 mDropDownList.setAdapter(mAdapter); 1163 mDropDownList.setOnItemClickListener(mItemClickListener); 1164 mDropDownList.setFocusable(true); 1165 mDropDownList.setFocusableInTouchMode(true); 1166 mDropDownList.setOnItemSelectedListener(new AdapterView.OnItemSelectedListener() { 1167 public void onItemSelected(AdapterView<?> parent, View view, 1168 int position, long id) { 1169 1170 if (position != -1) { 1171 DropDownListView dropDownList = mDropDownList; 1172 1173 if (dropDownList != null) { 1174 dropDownList.setListSelectionHidden(false); 1175 } 1176 } 1177 } 1178 1179 public void onNothingSelected(AdapterView<?> parent) { 1180 } 1181 }); 1182 mDropDownList.setOnScrollListener(mScrollListener); 1183 1184 if (mItemSelectedListener != null) { 1185 mDropDownList.setOnItemSelectedListener(mItemSelectedListener); 1186 } 1187 1188 dropDownView = mDropDownList; 1189 1190 View hintView = mPromptView; 1191 if (hintView != null) { 1192 // if a hint has been specified, we accomodate more space for it and 1193 // add a text view in the drop down menu, at the bottom of the list 1194 LinearLayout hintContainer = new LinearLayout(context); 1195 hintContainer.setOrientation(LinearLayout.VERTICAL); 1196 1197 LinearLayout.LayoutParams hintParams = new LinearLayout.LayoutParams( 1198 ViewGroup.LayoutParams.MATCH_PARENT, 0, 1.0f 1199 ); 1200 1201 switch (mPromptPosition) { 1202 case POSITION_PROMPT_BELOW: 1203 hintContainer.addView(dropDownView, hintParams); 1204 hintContainer.addView(hintView); 1205 break; 1206 1207 case POSITION_PROMPT_ABOVE: 1208 hintContainer.addView(hintView); 1209 hintContainer.addView(dropDownView, hintParams); 1210 break; 1211 1212 default: 1213 Log.e(TAG, "Invalid hint position " + mPromptPosition); 1214 break; 1215 } 1216 1217 // Measure the hint's height to find how much more vertical 1218 // space we need to add to the drop down's height. 1219 final int widthSize; 1220 final int widthMode; 1221 if (mDropDownWidth >= 0) { 1222 widthMode = MeasureSpec.AT_MOST; 1223 widthSize = mDropDownWidth; 1224 } else { 1225 widthMode = MeasureSpec.UNSPECIFIED; 1226 widthSize = 0; 1227 } 1228 final int widthSpec = MeasureSpec.makeMeasureSpec(widthSize, widthMode); 1229 final int heightSpec = MeasureSpec.UNSPECIFIED; 1230 hintView.measure(widthSpec, heightSpec); 1231 1232 hintParams = (LinearLayout.LayoutParams) hintView.getLayoutParams(); 1233 otherHeights = hintView.getMeasuredHeight() + hintParams.topMargin 1234 + hintParams.bottomMargin; 1235 1236 dropDownView = hintContainer; 1237 } 1238 1239 mPopup.setContentView(dropDownView); 1240 } else { 1241 final View view = mPromptView; 1242 if (view != null) { 1243 LinearLayout.LayoutParams hintParams = 1244 (LinearLayout.LayoutParams) view.getLayoutParams(); 1245 otherHeights = view.getMeasuredHeight() + hintParams.topMargin 1246 + hintParams.bottomMargin; 1247 } 1248 } 1249 1250 // getMaxAvailableHeight() subtracts the padding, so we put it back 1251 // to get the available height for the whole window. 1252 final int padding; 1253 final Drawable background = mPopup.getBackground(); 1254 if (background != null) { 1255 background.getPadding(mTempRect); 1256 padding = mTempRect.top + mTempRect.bottom; 1257 1258 // If we don't have an explicit vertical offset, determine one from 1259 // the window background so that content will line up. 1260 if (!mDropDownVerticalOffsetSet) { 1261 mDropDownVerticalOffset = -mTempRect.top; 1262 } 1263 } else { 1264 mTempRect.setEmpty(); 1265 padding = 0; 1266 } 1267 1268 // Max height available on the screen for a popup. 1269 final boolean ignoreBottomDecorations = 1270 mPopup.getInputMethodMode() == PopupWindow.INPUT_METHOD_NOT_NEEDED; 1271 final int maxHeight = mPopup.getMaxAvailableHeight( 1272 getAnchorView(), mDropDownVerticalOffset, ignoreBottomDecorations); 1273 if (mDropDownAlwaysVisible || mDropDownHeight == ViewGroup.LayoutParams.MATCH_PARENT) { 1274 return maxHeight + padding; 1275 } 1276 1277 final int childWidthSpec; 1278 switch (mDropDownWidth) { 1279 case ViewGroup.LayoutParams.WRAP_CONTENT: 1280 childWidthSpec = MeasureSpec.makeMeasureSpec( 1281 mContext.getResources().getDisplayMetrics().widthPixels 1282 - (mTempRect.left + mTempRect.right), 1283 MeasureSpec.AT_MOST); 1284 break; 1285 case ViewGroup.LayoutParams.MATCH_PARENT: 1286 childWidthSpec = MeasureSpec.makeMeasureSpec( 1287 mContext.getResources().getDisplayMetrics().widthPixels 1288 - (mTempRect.left + mTempRect.right), 1289 MeasureSpec.EXACTLY); 1290 break; 1291 default: 1292 childWidthSpec = MeasureSpec.makeMeasureSpec(mDropDownWidth, MeasureSpec.EXACTLY); 1293 break; 1294 } 1295 1296 // Add padding only if the list has items in it, that way we don't show 1297 // the popup if it is not needed. 1298 final int listContent = mDropDownList.measureHeightOfChildren(childWidthSpec, 1299 0, DropDownListView.NO_POSITION, maxHeight - otherHeights, -1); 1300 if (listContent > 0) { 1301 final int listPadding = mDropDownList.getPaddingTop() 1302 + mDropDownList.getPaddingBottom(); 1303 otherHeights += padding + listPadding; 1304 } 1305 1306 return listContent + otherHeights; 1307 } 1308 1309 /** 1310 * @hide 1311 */ setOverlapAnchor(boolean overlap)1312 public void setOverlapAnchor(boolean overlap) { 1313 mOverlapAnchorSet = true; 1314 mOverlapAnchor = overlap; 1315 } 1316 1317 private class PopupDataSetObserver extends DataSetObserver { 1318 @Override onChanged()1319 public void onChanged() { 1320 if (isShowing()) { 1321 // Resize the popup to fit new content 1322 show(); 1323 } 1324 } 1325 1326 @Override onInvalidated()1327 public void onInvalidated() { 1328 dismiss(); 1329 } 1330 } 1331 1332 private class ListSelectorHider implements Runnable { run()1333 public void run() { 1334 clearListSelection(); 1335 } 1336 } 1337 1338 private class ResizePopupRunnable implements Runnable { run()1339 public void run() { 1340 if (mDropDownList != null && mDropDownList.isAttachedToWindow() 1341 && mDropDownList.getCount() > mDropDownList.getChildCount() 1342 && mDropDownList.getChildCount() <= mListItemExpandMaximum) { 1343 mPopup.setInputMethodMode(PopupWindow.INPUT_METHOD_NOT_NEEDED); 1344 show(); 1345 } 1346 } 1347 } 1348 1349 private class PopupTouchInterceptor implements OnTouchListener { onTouch(View v, MotionEvent event)1350 public boolean onTouch(View v, MotionEvent event) { 1351 final int action = event.getAction(); 1352 final int x = (int) event.getX(); 1353 final int y = (int) event.getY(); 1354 1355 if (action == MotionEvent.ACTION_DOWN && 1356 mPopup != null && mPopup.isShowing() && 1357 (x >= 0 && x < mPopup.getWidth() && y >= 0 && y < mPopup.getHeight())) { 1358 mHandler.postDelayed(mResizePopupRunnable, EXPAND_LIST_TIMEOUT); 1359 } else if (action == MotionEvent.ACTION_UP) { 1360 mHandler.removeCallbacks(mResizePopupRunnable); 1361 } 1362 return false; 1363 } 1364 } 1365 1366 private class PopupScrollListener implements ListView.OnScrollListener { onScroll(AbsListView view, int firstVisibleItem, int visibleItemCount, int totalItemCount)1367 public void onScroll(AbsListView view, int firstVisibleItem, int visibleItemCount, 1368 int totalItemCount) { 1369 1370 } 1371 onScrollStateChanged(AbsListView view, int scrollState)1372 public void onScrollStateChanged(AbsListView view, int scrollState) { 1373 if (scrollState == SCROLL_STATE_TOUCH_SCROLL && 1374 !isInputMethodNotNeeded() && mPopup.getContentView() != null) { 1375 mHandler.removeCallbacks(mResizePopupRunnable); 1376 mResizePopupRunnable.run(); 1377 } 1378 } 1379 } 1380 } 1381