1 /* 2 * Copyright (C) 2008 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.CallSuper; 20 import android.annotation.ColorInt; 21 import android.annotation.FloatRange; 22 import android.annotation.IntDef; 23 import android.annotation.IntRange; 24 import android.annotation.Px; 25 import android.annotation.TestApi; 26 import android.annotation.Widget; 27 import android.compat.annotation.UnsupportedAppUsage; 28 import android.content.Context; 29 import android.content.res.ColorStateList; 30 import android.content.res.TypedArray; 31 import android.graphics.Canvas; 32 import android.graphics.Color; 33 import android.graphics.Paint; 34 import android.graphics.Paint.Align; 35 import android.graphics.Rect; 36 import android.graphics.drawable.Drawable; 37 import android.icu.text.DecimalFormatSymbols; 38 import android.os.Build; 39 import android.os.Bundle; 40 import android.text.InputFilter; 41 import android.text.InputType; 42 import android.text.Spanned; 43 import android.text.TextUtils; 44 import android.text.method.NumberKeyListener; 45 import android.util.AttributeSet; 46 import android.util.SparseArray; 47 import android.util.TypedValue; 48 import android.view.KeyEvent; 49 import android.view.LayoutInflater; 50 import android.view.LayoutInflater.Filter; 51 import android.view.MotionEvent; 52 import android.view.VelocityTracker; 53 import android.view.View; 54 import android.view.ViewConfiguration; 55 import android.view.accessibility.AccessibilityEvent; 56 import android.view.accessibility.AccessibilityManager; 57 import android.view.accessibility.AccessibilityNodeInfo; 58 import android.view.accessibility.AccessibilityNodeProvider; 59 import android.view.animation.DecelerateInterpolator; 60 import android.view.inputmethod.EditorInfo; 61 import android.view.inputmethod.InputMethodManager; 62 63 import com.android.internal.R; 64 65 import java.lang.annotation.Retention; 66 import java.lang.annotation.RetentionPolicy; 67 import java.util.ArrayList; 68 import java.util.Collections; 69 import java.util.List; 70 import java.util.Locale; 71 72 /** 73 * A widget that enables the user to select a number from a predefined range. 74 * There are two flavors of this widget and which one is presented to the user 75 * depends on the current theme. 76 * <ul> 77 * <li> 78 * If the current theme is derived from {@link android.R.style#Theme} the widget 79 * presents the current value as an editable input field with an increment button 80 * above and a decrement button below. Long pressing the buttons allows for a quick 81 * change of the current value. Tapping on the input field allows to type in 82 * a desired value. 83 * </li> 84 * <li> 85 * If the current theme is derived from {@link android.R.style#Theme_Holo} or 86 * {@link android.R.style#Theme_Holo_Light} the widget presents the current 87 * value as an editable input field with a lesser value above and a greater 88 * value below. Tapping on the lesser or greater value selects it by animating 89 * the number axis up or down to make the chosen value current. Flinging up 90 * or down allows for multiple increments or decrements of the current value. 91 * Long pressing on the lesser and greater values also allows for a quick change 92 * of the current value. Tapping on the current value allows to type in a 93 * desired value. 94 * </li> 95 * <li> 96 * If the current theme is derived from {@link android.R.style#Theme_Material} 97 * the widget presents the current value as a scrolling vertical selector with 98 * the selected value in the center and the previous and following numbers above 99 * and below, separated by a divider. The value is changed by flinging vertically. 100 * The thickness of the divider can be changed by using the 101 * {@link android.R.attr#selectionDividerHeight} attribute and the color of the 102 * divider can be changed by using the 103 * {@link android.R.attr#colorControlNormal} attribute. 104 * </li> 105 * </ul> 106 * <p> 107 * For an example of using this widget, see {@link android.widget.TimePicker}. 108 * </p> 109 */ 110 @Widget 111 public class NumberPicker extends LinearLayout { 112 113 /** 114 * The number of items show in the selector wheel. 115 */ 116 @UnsupportedAppUsage 117 private static final int SELECTOR_WHEEL_ITEM_COUNT = 3; 118 119 /** 120 * The default update interval during long press. 121 */ 122 private static final long DEFAULT_LONG_PRESS_UPDATE_INTERVAL = 300; 123 124 /** 125 * The index of the middle selector item. 126 */ 127 @UnsupportedAppUsage 128 private static final int SELECTOR_MIDDLE_ITEM_INDEX = SELECTOR_WHEEL_ITEM_COUNT / 2; 129 130 /** 131 * The coefficient by which to adjust (divide) the max fling velocity. 132 */ 133 private static final int SELECTOR_MAX_FLING_VELOCITY_ADJUSTMENT = 8; 134 135 /** 136 * The the duration for adjusting the selector wheel. 137 */ 138 private static final int SELECTOR_ADJUSTMENT_DURATION_MILLIS = 800; 139 140 /** 141 * The duration of scrolling while snapping to a given position. 142 */ 143 private static final int SNAP_SCROLL_DURATION = 300; 144 145 /** 146 * The strength of fading in the top and bottom while drawing the selector. 147 */ 148 private static final float TOP_AND_BOTTOM_FADING_EDGE_STRENGTH = 0.9f; 149 150 /** 151 * The default unscaled height of the selection divider. 152 */ 153 private static final int UNSCALED_DEFAULT_SELECTION_DIVIDER_HEIGHT = 2; 154 155 /** 156 * The default unscaled distance between the selection dividers. 157 */ 158 private static final int UNSCALED_DEFAULT_SELECTION_DIVIDERS_DISTANCE = 48; 159 160 /** 161 * The resource id for the default layout. 162 */ 163 private static final int DEFAULT_LAYOUT_RESOURCE_ID = R.layout.number_picker; 164 165 /** 166 * Constant for unspecified size. 167 */ 168 private static final int SIZE_UNSPECIFIED = -1; 169 170 /** 171 * User choice on whether the selector wheel should be wrapped. 172 */ 173 private boolean mWrapSelectorWheelPreferred = true; 174 175 /** 176 * Use a custom NumberPicker formatting callback to use two-digit minutes 177 * strings like "01". Keeping a static formatter etc. is the most efficient 178 * way to do this; it avoids creating temporary objects on every call to 179 * format(). 180 */ 181 private static class TwoDigitFormatter implements NumberPicker.Formatter { 182 final StringBuilder mBuilder = new StringBuilder(); 183 184 char mZeroDigit; 185 java.util.Formatter mFmt; 186 187 final Object[] mArgs = new Object[1]; 188 TwoDigitFormatter()189 TwoDigitFormatter() { 190 final Locale locale = Locale.getDefault(); 191 init(locale); 192 } 193 init(Locale locale)194 private void init(Locale locale) { 195 mFmt = createFormatter(locale); 196 mZeroDigit = getZeroDigit(locale); 197 } 198 format(int value)199 public String format(int value) { 200 final Locale currentLocale = Locale.getDefault(); 201 if (mZeroDigit != getZeroDigit(currentLocale)) { 202 init(currentLocale); 203 } 204 mArgs[0] = value; 205 mBuilder.delete(0, mBuilder.length()); 206 mFmt.format("%02d", mArgs); 207 return mFmt.toString(); 208 } 209 getZeroDigit(Locale locale)210 private static char getZeroDigit(Locale locale) { 211 return DecimalFormatSymbols.getInstance(locale).getZeroDigit(); 212 } 213 createFormatter(Locale locale)214 private java.util.Formatter createFormatter(Locale locale) { 215 return new java.util.Formatter(mBuilder, locale); 216 } 217 } 218 219 private static final TwoDigitFormatter sTwoDigitFormatter = new TwoDigitFormatter(); 220 221 /** 222 * @hide 223 */ 224 @UnsupportedAppUsage getTwoDigitFormatter()225 public static final Formatter getTwoDigitFormatter() { 226 return sTwoDigitFormatter; 227 } 228 229 /** 230 * The increment button. 231 */ 232 private final ImageButton mIncrementButton; 233 234 /** 235 * The decrement button. 236 */ 237 private final ImageButton mDecrementButton; 238 239 /** 240 * The text for showing the current value. 241 */ 242 @UnsupportedAppUsage 243 private final EditText mInputText; 244 245 /** 246 * The distance between the two selection dividers. 247 */ 248 private final int mSelectionDividersDistance; 249 250 /** 251 * The min height of this widget. 252 */ 253 @UnsupportedAppUsage(maxTargetSdk = Build.VERSION_CODES.R, trackingBug = 170729553) 254 private final int mMinHeight; 255 256 /** 257 * The max height of this widget. 258 */ 259 private final int mMaxHeight; 260 261 /** 262 * The max width of this widget. 263 */ 264 @UnsupportedAppUsage(maxTargetSdk = Build.VERSION_CODES.R, trackingBug = 170729553) 265 private final int mMinWidth; 266 267 /** 268 * The max width of this widget. 269 */ 270 private int mMaxWidth; 271 272 /** 273 * Flag whether to compute the max width. 274 */ 275 private final boolean mComputeMaxWidth; 276 277 /** 278 * The height of the text. 279 */ 280 @UnsupportedAppUsage(maxTargetSdk = Build.VERSION_CODES.R, trackingBug = 170729553) 281 private final int mTextSize; 282 283 /** 284 * The height of the gap between text elements if the selector wheel. 285 */ 286 private int mSelectorTextGapHeight; 287 288 /** 289 * The values to be displayed instead the indices. 290 */ 291 private String[] mDisplayedValues; 292 293 /** 294 * Lower value of the range of numbers allowed for the NumberPicker 295 */ 296 private int mMinValue; 297 298 /** 299 * Upper value of the range of numbers allowed for the NumberPicker 300 */ 301 @UnsupportedAppUsage(maxTargetSdk = Build.VERSION_CODES.R, trackingBug = 170729553) 302 private int mMaxValue; 303 304 /** 305 * Current value of this NumberPicker 306 */ 307 private int mValue; 308 309 /** 310 * Listener to be notified upon current value change. 311 */ 312 @UnsupportedAppUsage(maxTargetSdk = Build.VERSION_CODES.R, trackingBug = 170729553) 313 private OnValueChangeListener mOnValueChangeListener; 314 315 /** 316 * Listener to be notified upon scroll state change. 317 */ 318 private OnScrollListener mOnScrollListener; 319 320 /** 321 * Formatter for for displaying the current value. 322 */ 323 private Formatter mFormatter; 324 325 /** 326 * The speed for updating the value form long press. 327 */ 328 private long mLongPressUpdateInterval = DEFAULT_LONG_PRESS_UPDATE_INTERVAL; 329 330 /** 331 * Cache for the string representation of selector indices. 332 */ 333 private final SparseArray<String> mSelectorIndexToStringCache = new SparseArray<String>(); 334 335 /** 336 * The selector indices whose value are show by the selector. 337 */ 338 @UnsupportedAppUsage 339 private final int[] mSelectorIndices = new int[SELECTOR_WHEEL_ITEM_COUNT]; 340 341 /** 342 * The {@link Paint} for drawing the selector. 343 */ 344 @UnsupportedAppUsage(maxTargetSdk = Build.VERSION_CODES.P) 345 private final Paint mSelectorWheelPaint; 346 347 /** 348 * The {@link Drawable} for pressed virtual (increment/decrement) buttons. 349 */ 350 private final Drawable mVirtualButtonPressedDrawable; 351 352 /** 353 * The height of a selector element (text + gap). 354 */ 355 private int mSelectorElementHeight; 356 357 /** 358 * The initial offset of the scroll selector. 359 */ 360 private int mInitialScrollOffset = Integer.MIN_VALUE; 361 362 /** 363 * The current offset of the scroll selector. 364 */ 365 private int mCurrentScrollOffset; 366 367 /** 368 * The {@link Scroller} responsible for flinging the selector. 369 */ 370 @UnsupportedAppUsage(maxTargetSdk = Build.VERSION_CODES.R, trackingBug = 170729553) 371 private final Scroller mFlingScroller; 372 373 /** 374 * The {@link Scroller} responsible for adjusting the selector. 375 */ 376 private final Scroller mAdjustScroller; 377 378 /** 379 * The previous Y coordinate while scrolling the selector. 380 */ 381 private int mPreviousScrollerY; 382 383 /** 384 * Handle to the reusable command for setting the input text selection. 385 */ 386 private SetSelectionCommand mSetSelectionCommand; 387 388 /** 389 * Handle to the reusable command for changing the current value from long 390 * press by one. 391 */ 392 private ChangeCurrentByOneFromLongPressCommand mChangeCurrentByOneFromLongPressCommand; 393 394 /** 395 * Command for beginning an edit of the current value via IME on long press. 396 */ 397 private BeginSoftInputOnLongPressCommand mBeginSoftInputOnLongPressCommand; 398 399 /** 400 * The Y position of the last down event. 401 */ 402 private float mLastDownEventY; 403 404 /** 405 * The time of the last down event. 406 */ 407 private long mLastDownEventTime; 408 409 /** 410 * The Y position of the last down or move event. 411 */ 412 private float mLastDownOrMoveEventY; 413 414 /** 415 * Determines speed during touch scrolling. 416 */ 417 private VelocityTracker mVelocityTracker; 418 419 /** 420 * @see ViewConfiguration#getScaledTouchSlop() 421 */ 422 private int mTouchSlop; 423 424 /** 425 * @see ViewConfiguration#getScaledMinimumFlingVelocity() 426 */ 427 private int mMinimumFlingVelocity; 428 429 /** 430 * @see ViewConfiguration#getScaledMaximumFlingVelocity() 431 */ 432 @UnsupportedAppUsage(maxTargetSdk = Build.VERSION_CODES.R, trackingBug = 170729553) 433 private int mMaximumFlingVelocity; 434 435 /** 436 * Flag whether the selector should wrap around. 437 */ 438 private boolean mWrapSelectorWheel; 439 440 /** 441 * The back ground color used to optimize scroller fading. 442 */ 443 private final int mSolidColor; 444 445 /** 446 * Flag whether this widget has a selector wheel. 447 */ 448 private final boolean mHasSelectorWheel; 449 450 /** 451 * Divider for showing item to be selected while scrolling 452 */ 453 @UnsupportedAppUsage(maxTargetSdk = Build.VERSION_CODES.P) 454 private final Drawable mSelectionDivider; 455 456 /** 457 * The height of the selection divider. 458 */ 459 @UnsupportedAppUsage(maxTargetSdk = Build.VERSION_CODES.P) 460 private int mSelectionDividerHeight; 461 462 /** 463 * The current scroll state of the number picker. 464 */ 465 private int mScrollState = OnScrollListener.SCROLL_STATE_IDLE; 466 467 /** 468 * Flag whether to ignore move events - we ignore such when we show in IME 469 * to prevent the content from scrolling. 470 */ 471 private boolean mIgnoreMoveEvents; 472 473 /** 474 * Flag whether to perform a click on tap. 475 */ 476 private boolean mPerformClickOnTap; 477 478 /** 479 * The top of the top selection divider. 480 */ 481 private int mTopSelectionDividerTop; 482 483 /** 484 * The bottom of the bottom selection divider. 485 */ 486 private int mBottomSelectionDividerBottom; 487 488 /** 489 * The virtual id of the last hovered child. 490 */ 491 private int mLastHoveredChildVirtualViewId; 492 493 /** 494 * Whether the increment virtual button is pressed. 495 */ 496 private boolean mIncrementVirtualButtonPressed; 497 498 /** 499 * Whether the decrement virtual button is pressed. 500 */ 501 private boolean mDecrementVirtualButtonPressed; 502 503 /** 504 * Provider to report to clients the semantic structure of this widget. 505 */ 506 private AccessibilityNodeProviderImpl mAccessibilityNodeProvider; 507 508 /** 509 * Helper class for managing pressed state of the virtual buttons. 510 */ 511 private final PressedStateHelper mPressedStateHelper; 512 513 /** 514 * The keycode of the last handled DPAD down event. 515 */ 516 private int mLastHandledDownDpadKeyCode = -1; 517 518 /** 519 * If true then the selector wheel is hidden until the picker has focus. 520 */ 521 private boolean mHideWheelUntilFocused; 522 523 /** 524 * Interface to listen for changes of the current value. 525 */ 526 public interface OnValueChangeListener { 527 528 /** 529 * Called upon a change of the current value. 530 * 531 * @param picker The NumberPicker associated with this listener. 532 * @param oldVal The previous value. 533 * @param newVal The new value. 534 */ onValueChange(NumberPicker picker, int oldVal, int newVal)535 void onValueChange(NumberPicker picker, int oldVal, int newVal); 536 } 537 538 /** 539 * Interface to listen for the picker scroll state. 540 */ 541 public interface OnScrollListener { 542 /** @hide */ 543 @IntDef(prefix = { "SCROLL_STATE_" }, value = { 544 SCROLL_STATE_IDLE, 545 SCROLL_STATE_TOUCH_SCROLL, 546 SCROLL_STATE_FLING 547 }) 548 @Retention(RetentionPolicy.SOURCE) 549 public @interface ScrollState {} 550 551 /** 552 * The view is not scrolling. 553 */ 554 public static int SCROLL_STATE_IDLE = 0; 555 556 /** 557 * The user is scrolling using touch, and their finger is still on the screen. 558 */ 559 public static int SCROLL_STATE_TOUCH_SCROLL = 1; 560 561 /** 562 * The user had previously been scrolling using touch and performed a fling. 563 */ 564 public static int SCROLL_STATE_FLING = 2; 565 566 /** 567 * Callback invoked while the number picker scroll state has changed. 568 * 569 * @param view The view whose scroll state is being reported. 570 * @param scrollState The current scroll state. One of 571 * {@link #SCROLL_STATE_IDLE}, 572 * {@link #SCROLL_STATE_TOUCH_SCROLL} or 573 * {@link #SCROLL_STATE_IDLE}. 574 */ onScrollStateChange(NumberPicker view, @ScrollState int scrollState)575 public void onScrollStateChange(NumberPicker view, @ScrollState int scrollState); 576 } 577 578 /** 579 * Interface used to format current value into a string for presentation. 580 */ 581 public interface Formatter { 582 583 /** 584 * Formats a string representation of the current value. 585 * 586 * @param value The currently selected value. 587 * @return A formatted string representation. 588 */ format(int value)589 public String format(int value); 590 } 591 592 /** 593 * Create a new number picker. 594 * 595 * @param context The application environment. 596 */ NumberPicker(Context context)597 public NumberPicker(Context context) { 598 this(context, null); 599 } 600 601 /** 602 * Create a new number picker. 603 * 604 * @param context The application environment. 605 * @param attrs A collection of attributes. 606 */ NumberPicker(Context context, AttributeSet attrs)607 public NumberPicker(Context context, AttributeSet attrs) { 608 this(context, attrs, R.attr.numberPickerStyle); 609 } 610 611 /** 612 * Create a new number picker 613 * 614 * @param context the application environment. 615 * @param attrs a collection of attributes. 616 * @param defStyleAttr An attribute in the current theme that contains a 617 * reference to a style resource that supplies default values for 618 * the view. Can be 0 to not look for defaults. 619 */ NumberPicker(Context context, AttributeSet attrs, int defStyleAttr)620 public NumberPicker(Context context, AttributeSet attrs, int defStyleAttr) { 621 this(context, attrs, defStyleAttr, 0); 622 } 623 624 /** 625 * Create a new number picker 626 * 627 * @param context the application environment. 628 * @param attrs a collection of attributes. 629 * @param defStyleAttr An attribute in the current theme that contains a 630 * reference to a style resource that supplies default values for 631 * the view. Can be 0 to not look for defaults. 632 * @param defStyleRes A resource identifier of a style resource that 633 * supplies default values for the view, used only if 634 * defStyleAttr is 0 or can not be found in the theme. Can be 0 635 * to not look for defaults. 636 */ NumberPicker(Context context, AttributeSet attrs, int defStyleAttr, int defStyleRes)637 public NumberPicker(Context context, AttributeSet attrs, int defStyleAttr, int defStyleRes) { 638 super(context, attrs, defStyleAttr, defStyleRes); 639 640 // process style attributes 641 final TypedArray attributesArray = context.obtainStyledAttributes( 642 attrs, R.styleable.NumberPicker, defStyleAttr, defStyleRes); 643 saveAttributeDataForStyleable(context, R.styleable.NumberPicker, 644 attrs, attributesArray, defStyleAttr, defStyleRes); 645 final int layoutResId = attributesArray.getResourceId( 646 R.styleable.NumberPicker_internalLayout, DEFAULT_LAYOUT_RESOURCE_ID); 647 648 mHasSelectorWheel = (layoutResId != DEFAULT_LAYOUT_RESOURCE_ID); 649 650 mHideWheelUntilFocused = attributesArray.getBoolean( 651 R.styleable.NumberPicker_hideWheelUntilFocused, false); 652 653 mSolidColor = attributesArray.getColor(R.styleable.NumberPicker_solidColor, 0); 654 655 final Drawable selectionDivider = attributesArray.getDrawable( 656 R.styleable.NumberPicker_selectionDivider); 657 if (selectionDivider != null) { 658 selectionDivider.setCallback(this); 659 selectionDivider.setLayoutDirection(getLayoutDirection()); 660 if (selectionDivider.isStateful()) { 661 selectionDivider.setState(getDrawableState()); 662 } 663 } 664 mSelectionDivider = selectionDivider; 665 666 final int defSelectionDividerHeight = (int) TypedValue.applyDimension( 667 TypedValue.COMPLEX_UNIT_DIP, UNSCALED_DEFAULT_SELECTION_DIVIDER_HEIGHT, 668 getResources().getDisplayMetrics()); 669 mSelectionDividerHeight = attributesArray.getDimensionPixelSize( 670 R.styleable.NumberPicker_selectionDividerHeight, defSelectionDividerHeight); 671 672 final int defSelectionDividerDistance = (int) TypedValue.applyDimension( 673 TypedValue.COMPLEX_UNIT_DIP, UNSCALED_DEFAULT_SELECTION_DIVIDERS_DISTANCE, 674 getResources().getDisplayMetrics()); 675 mSelectionDividersDistance = attributesArray.getDimensionPixelSize( 676 R.styleable.NumberPicker_selectionDividersDistance, defSelectionDividerDistance); 677 678 mMinHeight = attributesArray.getDimensionPixelSize( 679 R.styleable.NumberPicker_internalMinHeight, SIZE_UNSPECIFIED); 680 681 mMaxHeight = attributesArray.getDimensionPixelSize( 682 R.styleable.NumberPicker_internalMaxHeight, SIZE_UNSPECIFIED); 683 if (mMinHeight != SIZE_UNSPECIFIED && mMaxHeight != SIZE_UNSPECIFIED 684 && mMinHeight > mMaxHeight) { 685 throw new IllegalArgumentException("minHeight > maxHeight"); 686 } 687 688 mMinWidth = attributesArray.getDimensionPixelSize( 689 R.styleable.NumberPicker_internalMinWidth, SIZE_UNSPECIFIED); 690 691 mMaxWidth = attributesArray.getDimensionPixelSize( 692 R.styleable.NumberPicker_internalMaxWidth, SIZE_UNSPECIFIED); 693 if (mMinWidth != SIZE_UNSPECIFIED && mMaxWidth != SIZE_UNSPECIFIED 694 && mMinWidth > mMaxWidth) { 695 throw new IllegalArgumentException("minWidth > maxWidth"); 696 } 697 698 mComputeMaxWidth = (mMaxWidth == SIZE_UNSPECIFIED); 699 700 mVirtualButtonPressedDrawable = attributesArray.getDrawable( 701 R.styleable.NumberPicker_virtualButtonPressedDrawable); 702 703 attributesArray.recycle(); 704 705 mPressedStateHelper = new PressedStateHelper(); 706 707 // By default Linearlayout that we extend is not drawn. This is 708 // its draw() method is not called but dispatchDraw() is called 709 // directly (see ViewGroup.drawChild()). However, this class uses 710 // the fading edge effect implemented by View and we need our 711 // draw() method to be called. Therefore, we declare we will draw. 712 setWillNotDraw(!mHasSelectorWheel); 713 714 LayoutInflater inflater = (LayoutInflater) getContext().getSystemService( 715 Context.LAYOUT_INFLATER_SERVICE); 716 inflater.inflate(layoutResId, this, true); 717 718 OnClickListener onClickListener = new OnClickListener() { 719 public void onClick(View v) { 720 hideSoftInput(); 721 mInputText.clearFocus(); 722 if (v.getId() == R.id.increment) { 723 changeValueByOne(true); 724 } else { 725 changeValueByOne(false); 726 } 727 } 728 }; 729 730 OnLongClickListener onLongClickListener = new OnLongClickListener() { 731 public boolean onLongClick(View v) { 732 hideSoftInput(); 733 mInputText.clearFocus(); 734 if (v.getId() == R.id.increment) { 735 postChangeCurrentByOneFromLongPress(true, 0); 736 } else { 737 postChangeCurrentByOneFromLongPress(false, 0); 738 } 739 return true; 740 } 741 }; 742 743 // increment button 744 if (!mHasSelectorWheel) { 745 mIncrementButton = findViewById(R.id.increment); 746 mIncrementButton.setOnClickListener(onClickListener); 747 mIncrementButton.setOnLongClickListener(onLongClickListener); 748 } else { 749 mIncrementButton = null; 750 } 751 752 // decrement button 753 if (!mHasSelectorWheel) { 754 mDecrementButton = findViewById(R.id.decrement); 755 mDecrementButton.setOnClickListener(onClickListener); 756 mDecrementButton.setOnLongClickListener(onLongClickListener); 757 } else { 758 mDecrementButton = null; 759 } 760 761 // input text 762 mInputText = findViewById(R.id.numberpicker_input); 763 mInputText.setOnFocusChangeListener(new OnFocusChangeListener() { 764 public void onFocusChange(View v, boolean hasFocus) { 765 if (hasFocus) { 766 mInputText.selectAll(); 767 } else { 768 mInputText.setSelection(0, 0); 769 validateInputTextView(v); 770 } 771 } 772 }); 773 mInputText.setFilters(new InputFilter[] { 774 new InputTextFilter() 775 }); 776 777 mInputText.setRawInputType(InputType.TYPE_CLASS_NUMBER); 778 mInputText.setImeOptions(EditorInfo.IME_ACTION_DONE); 779 780 // initialize constants 781 ViewConfiguration configuration = ViewConfiguration.get(context); 782 mTouchSlop = configuration.getScaledTouchSlop(); 783 mMinimumFlingVelocity = configuration.getScaledMinimumFlingVelocity(); 784 mMaximumFlingVelocity = configuration.getScaledMaximumFlingVelocity() 785 / SELECTOR_MAX_FLING_VELOCITY_ADJUSTMENT; 786 mTextSize = (int) mInputText.getTextSize(); 787 788 // create the selector wheel paint 789 Paint paint = new Paint(); 790 paint.setAntiAlias(true); 791 paint.setTextAlign(Align.CENTER); 792 paint.setTextSize(mTextSize); 793 paint.setTypeface(mInputText.getTypeface()); 794 ColorStateList colors = mInputText.getTextColors(); 795 int color = colors.getColorForState(ENABLED_STATE_SET, Color.WHITE); 796 paint.setColor(color); 797 mSelectorWheelPaint = paint; 798 799 // create the fling and adjust scrollers 800 mFlingScroller = new Scroller(getContext(), null, true); 801 mAdjustScroller = new Scroller(getContext(), new DecelerateInterpolator(2.5f)); 802 803 updateInputTextView(); 804 805 // If not explicitly specified this view is important for accessibility. 806 if (getImportantForAccessibility() == IMPORTANT_FOR_ACCESSIBILITY_AUTO) { 807 setImportantForAccessibility(IMPORTANT_FOR_ACCESSIBILITY_YES); 808 } 809 810 // Should be focusable by default, as the text view whose visibility changes is focusable 811 if (getFocusable() == View.FOCUSABLE_AUTO) { 812 setFocusable(View.FOCUSABLE); 813 setFocusableInTouchMode(true); 814 } 815 } 816 817 @Override onLayout(boolean changed, int left, int top, int right, int bottom)818 protected void onLayout(boolean changed, int left, int top, int right, int bottom) { 819 if (!mHasSelectorWheel) { 820 super.onLayout(changed, left, top, right, bottom); 821 return; 822 } 823 final int msrdWdth = getMeasuredWidth(); 824 final int msrdHght = getMeasuredHeight(); 825 826 // Input text centered horizontally. 827 final int inptTxtMsrdWdth = mInputText.getMeasuredWidth(); 828 final int inptTxtMsrdHght = mInputText.getMeasuredHeight(); 829 final int inptTxtLeft = (msrdWdth - inptTxtMsrdWdth) / 2; 830 final int inptTxtTop = (msrdHght - inptTxtMsrdHght) / 2; 831 final int inptTxtRight = inptTxtLeft + inptTxtMsrdWdth; 832 final int inptTxtBottom = inptTxtTop + inptTxtMsrdHght; 833 mInputText.layout(inptTxtLeft, inptTxtTop, inptTxtRight, inptTxtBottom); 834 835 if (changed) { 836 // need to do all this when we know our size 837 initializeSelectorWheel(); 838 initializeFadingEdges(); 839 mTopSelectionDividerTop = (getHeight() - mSelectionDividersDistance) / 2 840 - mSelectionDividerHeight; 841 mBottomSelectionDividerBottom = mTopSelectionDividerTop + 2 * mSelectionDividerHeight 842 + mSelectionDividersDistance; 843 } 844 } 845 846 @Override onMeasure(int widthMeasureSpec, int heightMeasureSpec)847 protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { 848 if (!mHasSelectorWheel) { 849 super.onMeasure(widthMeasureSpec, heightMeasureSpec); 850 return; 851 } 852 // Try greedily to fit the max width and height. 853 final int newWidthMeasureSpec = makeMeasureSpec(widthMeasureSpec, mMaxWidth); 854 final int newHeightMeasureSpec = makeMeasureSpec(heightMeasureSpec, mMaxHeight); 855 super.onMeasure(newWidthMeasureSpec, newHeightMeasureSpec); 856 // Flag if we are measured with width or height less than the respective min. 857 final int widthSize = resolveSizeAndStateRespectingMinSize(mMinWidth, getMeasuredWidth(), 858 widthMeasureSpec); 859 final int heightSize = resolveSizeAndStateRespectingMinSize(mMinHeight, getMeasuredHeight(), 860 heightMeasureSpec); 861 setMeasuredDimension(widthSize, heightSize); 862 } 863 864 /** 865 * Move to the final position of a scroller. Ensures to force finish the scroller 866 * and if it is not at its final position a scroll of the selector wheel is 867 * performed to fast forward to the final position. 868 * 869 * @param scroller The scroller to whose final position to get. 870 * @return True of the a move was performed, i.e. the scroller was not in final position. 871 */ moveToFinalScrollerPosition(Scroller scroller)872 private boolean moveToFinalScrollerPosition(Scroller scroller) { 873 scroller.forceFinished(true); 874 int amountToScroll = scroller.getFinalY() - scroller.getCurrY(); 875 int futureScrollOffset = (mCurrentScrollOffset + amountToScroll) % mSelectorElementHeight; 876 int overshootAdjustment = mInitialScrollOffset - futureScrollOffset; 877 if (overshootAdjustment != 0) { 878 if (Math.abs(overshootAdjustment) > mSelectorElementHeight / 2) { 879 if (overshootAdjustment > 0) { 880 overshootAdjustment -= mSelectorElementHeight; 881 } else { 882 overshootAdjustment += mSelectorElementHeight; 883 } 884 } 885 amountToScroll += overshootAdjustment; 886 scrollBy(0, amountToScroll); 887 return true; 888 } 889 return false; 890 } 891 892 @Override onInterceptTouchEvent(MotionEvent event)893 public boolean onInterceptTouchEvent(MotionEvent event) { 894 if (!mHasSelectorWheel || !isEnabled()) { 895 return false; 896 } 897 final int action = event.getActionMasked(); 898 switch (action) { 899 case MotionEvent.ACTION_DOWN: { 900 removeAllCallbacks(); 901 hideSoftInput(); 902 mLastDownOrMoveEventY = mLastDownEventY = event.getY(); 903 mLastDownEventTime = event.getEventTime(); 904 mIgnoreMoveEvents = false; 905 mPerformClickOnTap = false; 906 // Handle pressed state before any state change. 907 if (mLastDownEventY < mTopSelectionDividerTop) { 908 if (mScrollState == OnScrollListener.SCROLL_STATE_IDLE) { 909 mPressedStateHelper.buttonPressDelayed( 910 PressedStateHelper.BUTTON_DECREMENT); 911 } 912 } else if (mLastDownEventY > mBottomSelectionDividerBottom) { 913 if (mScrollState == OnScrollListener.SCROLL_STATE_IDLE) { 914 mPressedStateHelper.buttonPressDelayed( 915 PressedStateHelper.BUTTON_INCREMENT); 916 } 917 } 918 // Make sure we support flinging inside scrollables. 919 getParent().requestDisallowInterceptTouchEvent(true); 920 if (!mFlingScroller.isFinished()) { 921 mFlingScroller.forceFinished(true); 922 mAdjustScroller.forceFinished(true); 923 onScrollerFinished(mFlingScroller); 924 onScrollStateChange(OnScrollListener.SCROLL_STATE_IDLE); 925 } else if (!mAdjustScroller.isFinished()) { 926 mFlingScroller.forceFinished(true); 927 mAdjustScroller.forceFinished(true); 928 onScrollerFinished(mAdjustScroller); 929 } else if (mLastDownEventY < mTopSelectionDividerTop) { 930 postChangeCurrentByOneFromLongPress( 931 false, ViewConfiguration.getLongPressTimeout()); 932 } else if (mLastDownEventY > mBottomSelectionDividerBottom) { 933 postChangeCurrentByOneFromLongPress( 934 true, ViewConfiguration.getLongPressTimeout()); 935 } else { 936 mPerformClickOnTap = true; 937 postBeginSoftInputOnLongPressCommand(); 938 } 939 return true; 940 } 941 } 942 return false; 943 } 944 945 @Override onTouchEvent(MotionEvent event)946 public boolean onTouchEvent(MotionEvent event) { 947 if (!isEnabled() || !mHasSelectorWheel) { 948 return false; 949 } 950 if (mVelocityTracker == null) { 951 mVelocityTracker = VelocityTracker.obtain(); 952 } 953 mVelocityTracker.addMovement(event); 954 int action = event.getActionMasked(); 955 switch (action) { 956 case MotionEvent.ACTION_MOVE: { 957 if (mIgnoreMoveEvents) { 958 break; 959 } 960 float currentMoveY = event.getY(); 961 if (mScrollState != OnScrollListener.SCROLL_STATE_TOUCH_SCROLL) { 962 int deltaDownY = (int) Math.abs(currentMoveY - mLastDownEventY); 963 if (deltaDownY > mTouchSlop) { 964 removeAllCallbacks(); 965 onScrollStateChange(OnScrollListener.SCROLL_STATE_TOUCH_SCROLL); 966 } 967 } else { 968 int deltaMoveY = (int) ((currentMoveY - mLastDownOrMoveEventY)); 969 scrollBy(0, deltaMoveY); 970 invalidate(); 971 } 972 mLastDownOrMoveEventY = currentMoveY; 973 } break; 974 case MotionEvent.ACTION_UP: { 975 removeBeginSoftInputCommand(); 976 removeChangeCurrentByOneFromLongPress(); 977 mPressedStateHelper.cancel(); 978 VelocityTracker velocityTracker = mVelocityTracker; 979 velocityTracker.computeCurrentVelocity(1000, mMaximumFlingVelocity); 980 int initialVelocity = (int) velocityTracker.getYVelocity(); 981 if (Math.abs(initialVelocity) > mMinimumFlingVelocity) { 982 fling(initialVelocity); 983 onScrollStateChange(OnScrollListener.SCROLL_STATE_FLING); 984 } else { 985 int eventY = (int) event.getY(); 986 int deltaMoveY = (int) Math.abs(eventY - mLastDownEventY); 987 long deltaTime = event.getEventTime() - mLastDownEventTime; 988 if (deltaMoveY <= mTouchSlop && deltaTime < ViewConfiguration.getTapTimeout()) { 989 if (mPerformClickOnTap) { 990 mPerformClickOnTap = false; 991 performClick(); 992 } else { 993 int selectorIndexOffset = (eventY / mSelectorElementHeight) 994 - SELECTOR_MIDDLE_ITEM_INDEX; 995 if (selectorIndexOffset > 0) { 996 changeValueByOne(true); 997 mPressedStateHelper.buttonTapped( 998 PressedStateHelper.BUTTON_INCREMENT); 999 } else if (selectorIndexOffset < 0) { 1000 changeValueByOne(false); 1001 mPressedStateHelper.buttonTapped( 1002 PressedStateHelper.BUTTON_DECREMENT); 1003 } 1004 } 1005 } else { 1006 ensureScrollWheelAdjusted(); 1007 } 1008 onScrollStateChange(OnScrollListener.SCROLL_STATE_IDLE); 1009 } 1010 mVelocityTracker.recycle(); 1011 mVelocityTracker = null; 1012 } break; 1013 } 1014 return true; 1015 } 1016 1017 @Override dispatchTouchEvent(MotionEvent event)1018 public boolean dispatchTouchEvent(MotionEvent event) { 1019 final int action = event.getActionMasked(); 1020 switch (action) { 1021 case MotionEvent.ACTION_CANCEL: 1022 case MotionEvent.ACTION_UP: 1023 removeAllCallbacks(); 1024 break; 1025 } 1026 return super.dispatchTouchEvent(event); 1027 } 1028 1029 @Override dispatchKeyEvent(KeyEvent event)1030 public boolean dispatchKeyEvent(KeyEvent event) { 1031 final int keyCode = event.getKeyCode(); 1032 switch (keyCode) { 1033 case KeyEvent.KEYCODE_DPAD_CENTER: 1034 case KeyEvent.KEYCODE_ENTER: 1035 case KeyEvent.KEYCODE_NUMPAD_ENTER: 1036 removeAllCallbacks(); 1037 break; 1038 case KeyEvent.KEYCODE_DPAD_DOWN: 1039 case KeyEvent.KEYCODE_DPAD_UP: 1040 if (!mHasSelectorWheel) { 1041 break; 1042 } 1043 switch (event.getAction()) { 1044 case KeyEvent.ACTION_DOWN: 1045 if (mWrapSelectorWheel || ((keyCode == KeyEvent.KEYCODE_DPAD_DOWN) 1046 ? getValue() < getMaxValue() : getValue() > getMinValue())) { 1047 requestFocus(); 1048 mLastHandledDownDpadKeyCode = keyCode; 1049 removeAllCallbacks(); 1050 if (mFlingScroller.isFinished()) { 1051 changeValueByOne(keyCode == KeyEvent.KEYCODE_DPAD_DOWN); 1052 } 1053 return true; 1054 } 1055 break; 1056 case KeyEvent.ACTION_UP: 1057 if (mLastHandledDownDpadKeyCode == keyCode) { 1058 mLastHandledDownDpadKeyCode = -1; 1059 return true; 1060 } 1061 break; 1062 } 1063 } 1064 return super.dispatchKeyEvent(event); 1065 } 1066 1067 @Override dispatchTrackballEvent(MotionEvent event)1068 public boolean dispatchTrackballEvent(MotionEvent event) { 1069 final int action = event.getActionMasked(); 1070 switch (action) { 1071 case MotionEvent.ACTION_CANCEL: 1072 case MotionEvent.ACTION_UP: 1073 removeAllCallbacks(); 1074 break; 1075 } 1076 return super.dispatchTrackballEvent(event); 1077 } 1078 1079 @Override dispatchHoverEvent(MotionEvent event)1080 protected boolean dispatchHoverEvent(MotionEvent event) { 1081 if (!mHasSelectorWheel) { 1082 return super.dispatchHoverEvent(event); 1083 } 1084 if (AccessibilityManager.getInstance(mContext).isEnabled()) { 1085 final int eventY = (int) event.getY(); 1086 final int hoveredVirtualViewId; 1087 if (eventY < mTopSelectionDividerTop) { 1088 hoveredVirtualViewId = AccessibilityNodeProviderImpl.VIRTUAL_VIEW_ID_DECREMENT; 1089 } else if (eventY > mBottomSelectionDividerBottom) { 1090 hoveredVirtualViewId = AccessibilityNodeProviderImpl.VIRTUAL_VIEW_ID_INCREMENT; 1091 } else { 1092 hoveredVirtualViewId = AccessibilityNodeProviderImpl.VIRTUAL_VIEW_ID_INPUT; 1093 } 1094 final int action = event.getActionMasked(); 1095 AccessibilityNodeProviderImpl provider = 1096 (AccessibilityNodeProviderImpl) getAccessibilityNodeProvider(); 1097 switch (action) { 1098 case MotionEvent.ACTION_HOVER_ENTER: { 1099 provider.sendAccessibilityEventForVirtualView(hoveredVirtualViewId, 1100 AccessibilityEvent.TYPE_VIEW_HOVER_ENTER); 1101 mLastHoveredChildVirtualViewId = hoveredVirtualViewId; 1102 provider.performAction(hoveredVirtualViewId, 1103 AccessibilityNodeInfo.ACTION_ACCESSIBILITY_FOCUS, null); 1104 } break; 1105 case MotionEvent.ACTION_HOVER_MOVE: { 1106 if (mLastHoveredChildVirtualViewId != hoveredVirtualViewId 1107 && mLastHoveredChildVirtualViewId != View.NO_ID) { 1108 provider.sendAccessibilityEventForVirtualView( 1109 mLastHoveredChildVirtualViewId, 1110 AccessibilityEvent.TYPE_VIEW_HOVER_EXIT); 1111 provider.sendAccessibilityEventForVirtualView(hoveredVirtualViewId, 1112 AccessibilityEvent.TYPE_VIEW_HOVER_ENTER); 1113 mLastHoveredChildVirtualViewId = hoveredVirtualViewId; 1114 provider.performAction(hoveredVirtualViewId, 1115 AccessibilityNodeInfo.ACTION_ACCESSIBILITY_FOCUS, null); 1116 } 1117 } break; 1118 case MotionEvent.ACTION_HOVER_EXIT: { 1119 provider.sendAccessibilityEventForVirtualView(hoveredVirtualViewId, 1120 AccessibilityEvent.TYPE_VIEW_HOVER_EXIT); 1121 mLastHoveredChildVirtualViewId = View.NO_ID; 1122 } break; 1123 } 1124 } 1125 return false; 1126 } 1127 1128 @Override computeScroll()1129 public void computeScroll() { 1130 Scroller scroller = mFlingScroller; 1131 if (scroller.isFinished()) { 1132 scroller = mAdjustScroller; 1133 if (scroller.isFinished()) { 1134 return; 1135 } 1136 } 1137 scroller.computeScrollOffset(); 1138 int currentScrollerY = scroller.getCurrY(); 1139 if (mPreviousScrollerY == 0) { 1140 mPreviousScrollerY = scroller.getStartY(); 1141 } 1142 scrollBy(0, currentScrollerY - mPreviousScrollerY); 1143 mPreviousScrollerY = currentScrollerY; 1144 if (scroller.isFinished()) { 1145 onScrollerFinished(scroller); 1146 } else { 1147 invalidate(); 1148 } 1149 } 1150 1151 @Override setEnabled(boolean enabled)1152 public void setEnabled(boolean enabled) { 1153 super.setEnabled(enabled); 1154 if (!mHasSelectorWheel) { 1155 mIncrementButton.setEnabled(enabled); 1156 } 1157 if (!mHasSelectorWheel) { 1158 mDecrementButton.setEnabled(enabled); 1159 } 1160 mInputText.setEnabled(enabled); 1161 } 1162 1163 @Override scrollBy(int x, int y)1164 public void scrollBy(int x, int y) { 1165 int[] selectorIndices = mSelectorIndices; 1166 int startScrollOffset = mCurrentScrollOffset; 1167 if (!mWrapSelectorWheel && y > 0 1168 && selectorIndices[SELECTOR_MIDDLE_ITEM_INDEX] <= mMinValue) { 1169 mCurrentScrollOffset = mInitialScrollOffset; 1170 return; 1171 } 1172 if (!mWrapSelectorWheel && y < 0 1173 && selectorIndices[SELECTOR_MIDDLE_ITEM_INDEX] >= mMaxValue) { 1174 mCurrentScrollOffset = mInitialScrollOffset; 1175 return; 1176 } 1177 mCurrentScrollOffset += y; 1178 while (mCurrentScrollOffset - mInitialScrollOffset > mSelectorTextGapHeight) { 1179 mCurrentScrollOffset -= mSelectorElementHeight; 1180 decrementSelectorIndices(selectorIndices); 1181 setValueInternal(selectorIndices[SELECTOR_MIDDLE_ITEM_INDEX], true); 1182 if (!mWrapSelectorWheel && selectorIndices[SELECTOR_MIDDLE_ITEM_INDEX] <= mMinValue) { 1183 mCurrentScrollOffset = mInitialScrollOffset; 1184 } 1185 } 1186 while (mCurrentScrollOffset - mInitialScrollOffset < -mSelectorTextGapHeight) { 1187 mCurrentScrollOffset += mSelectorElementHeight; 1188 incrementSelectorIndices(selectorIndices); 1189 setValueInternal(selectorIndices[SELECTOR_MIDDLE_ITEM_INDEX], true); 1190 if (!mWrapSelectorWheel && selectorIndices[SELECTOR_MIDDLE_ITEM_INDEX] >= mMaxValue) { 1191 mCurrentScrollOffset = mInitialScrollOffset; 1192 } 1193 } 1194 if (startScrollOffset != mCurrentScrollOffset) { 1195 onScrollChanged(0, mCurrentScrollOffset, 0, startScrollOffset); 1196 } 1197 } 1198 1199 @Override computeVerticalScrollOffset()1200 protected int computeVerticalScrollOffset() { 1201 return mCurrentScrollOffset; 1202 } 1203 1204 @Override computeVerticalScrollRange()1205 protected int computeVerticalScrollRange() { 1206 return (mMaxValue - mMinValue + 1) * mSelectorElementHeight; 1207 } 1208 1209 @Override computeVerticalScrollExtent()1210 protected int computeVerticalScrollExtent() { 1211 return getHeight(); 1212 } 1213 1214 @Override getSolidColor()1215 public int getSolidColor() { 1216 return mSolidColor; 1217 } 1218 1219 /** 1220 * Sets the listener to be notified on change of the current value. 1221 * 1222 * @param onValueChangedListener The listener. 1223 */ setOnValueChangedListener(OnValueChangeListener onValueChangedListener)1224 public void setOnValueChangedListener(OnValueChangeListener onValueChangedListener) { 1225 mOnValueChangeListener = onValueChangedListener; 1226 } 1227 1228 /** 1229 * Set listener to be notified for scroll state changes. 1230 * 1231 * @param onScrollListener The listener. 1232 */ setOnScrollListener(OnScrollListener onScrollListener)1233 public void setOnScrollListener(OnScrollListener onScrollListener) { 1234 mOnScrollListener = onScrollListener; 1235 } 1236 1237 /** 1238 * Set the formatter to be used for formatting the current value. 1239 * <p> 1240 * Note: If you have provided alternative values for the values this 1241 * formatter is never invoked. 1242 * </p> 1243 * 1244 * @param formatter The formatter object. If formatter is <code>null</code>, 1245 * {@link String#valueOf(int)} will be used. 1246 *@see #setDisplayedValues(String[]) 1247 */ setFormatter(Formatter formatter)1248 public void setFormatter(Formatter formatter) { 1249 if (formatter == mFormatter) { 1250 return; 1251 } 1252 mFormatter = formatter; 1253 initializeSelectorWheelIndices(); 1254 updateInputTextView(); 1255 } 1256 1257 /** 1258 * Set the current value for the number picker. 1259 * <p> 1260 * If the argument is less than the {@link NumberPicker#getMinValue()} and 1261 * {@link NumberPicker#getWrapSelectorWheel()} is <code>false</code> the 1262 * current value is set to the {@link NumberPicker#getMinValue()} value. 1263 * </p> 1264 * <p> 1265 * If the argument is less than the {@link NumberPicker#getMinValue()} and 1266 * {@link NumberPicker#getWrapSelectorWheel()} is <code>true</code> the 1267 * current value is set to the {@link NumberPicker#getMaxValue()} value. 1268 * </p> 1269 * <p> 1270 * If the argument is more than the {@link NumberPicker#getMaxValue()} and 1271 * {@link NumberPicker#getWrapSelectorWheel()} is <code>false</code> the 1272 * current value is set to the {@link NumberPicker#getMaxValue()} value. 1273 * </p> 1274 * <p> 1275 * If the argument is more than the {@link NumberPicker#getMaxValue()} and 1276 * {@link NumberPicker#getWrapSelectorWheel()} is <code>true</code> the 1277 * current value is set to the {@link NumberPicker#getMinValue()} value. 1278 * </p> 1279 * 1280 * @param value The current value. 1281 * @see #setWrapSelectorWheel(boolean) 1282 * @see #setMinValue(int) 1283 * @see #setMaxValue(int) 1284 */ setValue(int value)1285 public void setValue(int value) { 1286 setValueInternal(value, false); 1287 } 1288 1289 @Override performClick()1290 public boolean performClick() { 1291 if (!mHasSelectorWheel) { 1292 return super.performClick(); 1293 } else if (!super.performClick()) { 1294 showSoftInput(); 1295 } 1296 return true; 1297 } 1298 1299 @Override performLongClick()1300 public boolean performLongClick() { 1301 if (!mHasSelectorWheel) { 1302 return super.performLongClick(); 1303 } else if (!super.performLongClick()) { 1304 showSoftInput(); 1305 mIgnoreMoveEvents = true; 1306 } 1307 return true; 1308 } 1309 1310 /** 1311 * Shows the soft input for its input text. 1312 */ showSoftInput()1313 private void showSoftInput() { 1314 InputMethodManager inputMethodManager = 1315 getContext().getSystemService(InputMethodManager.class); 1316 if (inputMethodManager != null) { 1317 if (mHasSelectorWheel) { 1318 mInputText.setVisibility(View.VISIBLE); 1319 } 1320 mInputText.requestFocus(); 1321 inputMethodManager.showSoftInput(mInputText, 0); 1322 } 1323 } 1324 1325 /** 1326 * Hides the soft input if it is active for the input text. 1327 */ hideSoftInput()1328 private void hideSoftInput() { 1329 InputMethodManager inputMethodManager = 1330 getContext().getSystemService(InputMethodManager.class); 1331 if (inputMethodManager != null) { 1332 inputMethodManager.hideSoftInputFromView(mInputText, 0); 1333 } 1334 if (mHasSelectorWheel) { 1335 mInputText.setVisibility(View.INVISIBLE); 1336 } 1337 } 1338 1339 /** 1340 * Computes the max width if no such specified as an attribute. 1341 */ tryComputeMaxWidth()1342 private void tryComputeMaxWidth() { 1343 if (!mComputeMaxWidth) { 1344 return; 1345 } 1346 int maxTextWidth = 0; 1347 if (mDisplayedValues == null) { 1348 float maxDigitWidth = 0; 1349 for (int i = 0; i <= 9; i++) { 1350 final float digitWidth = mSelectorWheelPaint.measureText(formatNumberWithLocale(i)); 1351 if (digitWidth > maxDigitWidth) { 1352 maxDigitWidth = digitWidth; 1353 } 1354 } 1355 int numberOfDigits = 0; 1356 int current = mMaxValue; 1357 while (current > 0) { 1358 numberOfDigits++; 1359 current = current / 10; 1360 } 1361 maxTextWidth = (int) (numberOfDigits * maxDigitWidth); 1362 } else { 1363 final int valueCount = mDisplayedValues.length; 1364 for (int i = 0; i < valueCount; i++) { 1365 final float textWidth = mSelectorWheelPaint.measureText(mDisplayedValues[i]); 1366 if (textWidth > maxTextWidth) { 1367 maxTextWidth = (int) textWidth; 1368 } 1369 } 1370 } 1371 maxTextWidth += mInputText.getPaddingLeft() + mInputText.getPaddingRight(); 1372 if (mMaxWidth != maxTextWidth) { 1373 if (maxTextWidth > mMinWidth) { 1374 mMaxWidth = maxTextWidth; 1375 } else { 1376 mMaxWidth = mMinWidth; 1377 } 1378 invalidate(); 1379 } 1380 } 1381 1382 /** 1383 * Gets whether the selector wheel wraps when reaching the min/max value. 1384 * 1385 * @return True if the selector wheel wraps. 1386 * 1387 * @see #getMinValue() 1388 * @see #getMaxValue() 1389 */ getWrapSelectorWheel()1390 public boolean getWrapSelectorWheel() { 1391 return mWrapSelectorWheel; 1392 } 1393 1394 /** 1395 * Sets whether the selector wheel shown during flinging/scrolling should 1396 * wrap around the {@link NumberPicker#getMinValue()} and 1397 * {@link NumberPicker#getMaxValue()} values. 1398 * <p> 1399 * By default if the range (max - min) is more than the number of items shown 1400 * on the selector wheel the selector wheel wrapping is enabled. 1401 * </p> 1402 * <p> 1403 * <strong>Note:</strong> If the number of items, i.e. the range ( 1404 * {@link #getMaxValue()} - {@link #getMinValue()}) is less than 1405 * the number of items shown on the selector wheel, the selector wheel will 1406 * not wrap. Hence, in such a case calling this method is a NOP. 1407 * </p> 1408 * 1409 * @param wrapSelectorWheel Whether to wrap. 1410 */ setWrapSelectorWheel(boolean wrapSelectorWheel)1411 public void setWrapSelectorWheel(boolean wrapSelectorWheel) { 1412 mWrapSelectorWheelPreferred = wrapSelectorWheel; 1413 updateWrapSelectorWheel(); 1414 1415 } 1416 1417 /** 1418 * Whether or not the selector wheel should be wrapped is determined by user choice and whether 1419 * the choice is allowed. The former comes from {@link #setWrapSelectorWheel(boolean)}, the 1420 * latter is calculated based on min & max value set vs selector's visual length. Therefore, 1421 * this method should be called any time any of the 3 values (i.e. user choice, min and max 1422 * value) gets updated. 1423 */ updateWrapSelectorWheel()1424 private void updateWrapSelectorWheel() { 1425 final boolean wrappingAllowed = (mMaxValue - mMinValue) >= mSelectorIndices.length; 1426 mWrapSelectorWheel = wrappingAllowed && mWrapSelectorWheelPreferred; 1427 } 1428 1429 /** 1430 * Sets the speed at which the numbers be incremented and decremented when 1431 * the up and down buttons are long pressed respectively. 1432 * <p> 1433 * The default value is 300 ms. 1434 * </p> 1435 * 1436 * @param intervalMillis The speed (in milliseconds) at which the numbers 1437 * will be incremented and decremented. 1438 */ setOnLongPressUpdateInterval(long intervalMillis)1439 public void setOnLongPressUpdateInterval(long intervalMillis) { 1440 mLongPressUpdateInterval = intervalMillis; 1441 } 1442 1443 /** 1444 * Returns the value of the picker. 1445 * 1446 * @return The value. 1447 */ getValue()1448 public int getValue() { 1449 return mValue; 1450 } 1451 1452 /** 1453 * Returns the min value of the picker. 1454 * 1455 * @return The min value 1456 */ getMinValue()1457 public int getMinValue() { 1458 return mMinValue; 1459 } 1460 1461 /** 1462 * Sets the min value of the picker. 1463 * 1464 * @param minValue The min value inclusive. 1465 * 1466 * <strong>Note:</strong> The length of the displayed values array 1467 * set via {@link #setDisplayedValues(String[])} must be equal to the 1468 * range of selectable numbers which is equal to 1469 * {@link #getMaxValue()} - {@link #getMinValue()} + 1. 1470 */ setMinValue(int minValue)1471 public void setMinValue(int minValue) { 1472 if (mMinValue == minValue) { 1473 return; 1474 } 1475 if (minValue < 0) { 1476 throw new IllegalArgumentException("minValue must be >= 0"); 1477 } 1478 mMinValue = minValue; 1479 if (mMinValue > mValue) { 1480 mValue = mMinValue; 1481 } 1482 updateWrapSelectorWheel(); 1483 initializeSelectorWheelIndices(); 1484 updateInputTextView(); 1485 tryComputeMaxWidth(); 1486 invalidate(); 1487 } 1488 1489 /** 1490 * Returns the max value of the picker. 1491 * 1492 * @return The max value. 1493 */ getMaxValue()1494 public int getMaxValue() { 1495 return mMaxValue; 1496 } 1497 1498 /** 1499 * Sets the max value of the picker. 1500 * 1501 * @param maxValue The max value inclusive. 1502 * 1503 * <strong>Note:</strong> The length of the displayed values array 1504 * set via {@link #setDisplayedValues(String[])} must be equal to the 1505 * range of selectable numbers which is equal to 1506 * {@link #getMaxValue()} - {@link #getMinValue()} + 1. 1507 */ setMaxValue(int maxValue)1508 public void setMaxValue(int maxValue) { 1509 if (mMaxValue == maxValue) { 1510 return; 1511 } 1512 if (maxValue < 0) { 1513 throw new IllegalArgumentException("maxValue must be >= 0"); 1514 } 1515 mMaxValue = maxValue; 1516 if (mMaxValue < mValue) { 1517 mValue = mMaxValue; 1518 } 1519 updateWrapSelectorWheel(); 1520 initializeSelectorWheelIndices(); 1521 updateInputTextView(); 1522 tryComputeMaxWidth(); 1523 invalidate(); 1524 } 1525 1526 /** 1527 * Gets the values to be displayed instead of string values. 1528 * 1529 * @return The displayed values. 1530 */ getDisplayedValues()1531 public String[] getDisplayedValues() { 1532 return mDisplayedValues; 1533 } 1534 1535 /** 1536 * Sets the values to be displayed. 1537 * 1538 * @param displayedValues The displayed values. 1539 * 1540 * <strong>Note:</strong> The length of the displayed values array 1541 * must be equal to the range of selectable numbers which is equal to 1542 * {@link #getMaxValue()} - {@link #getMinValue()} + 1. 1543 */ setDisplayedValues(String[] displayedValues)1544 public void setDisplayedValues(String[] displayedValues) { 1545 if (mDisplayedValues == displayedValues) { 1546 return; 1547 } 1548 mDisplayedValues = displayedValues; 1549 if (mDisplayedValues != null) { 1550 // Allow text entry rather than strictly numeric entry. 1551 mInputText.setRawInputType(InputType.TYPE_CLASS_TEXT 1552 | InputType.TYPE_TEXT_FLAG_NO_SUGGESTIONS); 1553 } else { 1554 mInputText.setRawInputType(InputType.TYPE_CLASS_NUMBER); 1555 } 1556 updateInputTextView(); 1557 initializeSelectorWheelIndices(); 1558 tryComputeMaxWidth(); 1559 } 1560 1561 /** 1562 * Retrieves the displayed value for the current selection in this picker. 1563 * 1564 * @hide 1565 */ 1566 @TestApi getDisplayedValueForCurrentSelection()1567 public CharSequence getDisplayedValueForCurrentSelection() { 1568 // The cache field itself is initialized at declaration time, and since it's final, it 1569 // can't be null here. The cache is updated in ensureCachedScrollSelectorValue which is 1570 // called, directly or indirectly, on every call to setDisplayedValues, setFormatter, 1571 // setMinValue, setMaxValue and setValue, as well as user-driven interaction with the 1572 // picker. As such, the contents of the cache are always synced to the latest state of 1573 // the widget. 1574 return mSelectorIndexToStringCache.get(getValue()); 1575 } 1576 1577 /** 1578 * Set the height for the divider that separates the currently selected value from the others. 1579 * @param height The height to be set 1580 */ setSelectionDividerHeight(@ntRangefrom = 0) @x int height)1581 public void setSelectionDividerHeight(@IntRange(from = 0) @Px int height) { 1582 mSelectionDividerHeight = height; 1583 invalidate(); 1584 } 1585 1586 /** 1587 * Retrieve the height for the divider that separates the currently selected value from the 1588 * others. 1589 * @return The height of the divider 1590 */ 1591 @Px getSelectionDividerHeight()1592 public int getSelectionDividerHeight() { 1593 return mSelectionDividerHeight; 1594 } 1595 1596 @Override getTopFadingEdgeStrength()1597 protected float getTopFadingEdgeStrength() { 1598 return TOP_AND_BOTTOM_FADING_EDGE_STRENGTH; 1599 } 1600 1601 @Override getBottomFadingEdgeStrength()1602 protected float getBottomFadingEdgeStrength() { 1603 return TOP_AND_BOTTOM_FADING_EDGE_STRENGTH; 1604 } 1605 1606 @Override onDetachedFromWindow()1607 protected void onDetachedFromWindow() { 1608 super.onDetachedFromWindow(); 1609 removeAllCallbacks(); 1610 } 1611 1612 @CallSuper 1613 @Override drawableStateChanged()1614 protected void drawableStateChanged() { 1615 super.drawableStateChanged(); 1616 1617 final Drawable selectionDivider = mSelectionDivider; 1618 if (selectionDivider != null && selectionDivider.isStateful() 1619 && selectionDivider.setState(getDrawableState())) { 1620 invalidateDrawable(selectionDivider); 1621 } 1622 } 1623 1624 @CallSuper 1625 @Override jumpDrawablesToCurrentState()1626 public void jumpDrawablesToCurrentState() { 1627 super.jumpDrawablesToCurrentState(); 1628 1629 if (mSelectionDivider != null) { 1630 mSelectionDivider.jumpToCurrentState(); 1631 } 1632 } 1633 1634 /** @hide */ 1635 @Override onResolveDrawables(@esolvedLayoutDir int layoutDirection)1636 public void onResolveDrawables(@ResolvedLayoutDir int layoutDirection) { 1637 super.onResolveDrawables(layoutDirection); 1638 1639 if (mSelectionDivider != null) { 1640 mSelectionDivider.setLayoutDirection(layoutDirection); 1641 } 1642 } 1643 1644 @Override onDraw(Canvas canvas)1645 protected void onDraw(Canvas canvas) { 1646 if (!mHasSelectorWheel) { 1647 super.onDraw(canvas); 1648 return; 1649 } 1650 final boolean showSelectorWheel = mHideWheelUntilFocused ? hasFocus() : true; 1651 float x = (mRight - mLeft) / 2; 1652 float y = mCurrentScrollOffset; 1653 1654 // draw the virtual buttons pressed state if needed 1655 if (showSelectorWheel && mVirtualButtonPressedDrawable != null 1656 && mScrollState == OnScrollListener.SCROLL_STATE_IDLE) { 1657 if (mDecrementVirtualButtonPressed) { 1658 mVirtualButtonPressedDrawable.setState(PRESSED_STATE_SET); 1659 mVirtualButtonPressedDrawable.setBounds(0, 0, mRight, mTopSelectionDividerTop); 1660 mVirtualButtonPressedDrawable.draw(canvas); 1661 } 1662 if (mIncrementVirtualButtonPressed) { 1663 mVirtualButtonPressedDrawable.setState(PRESSED_STATE_SET); 1664 mVirtualButtonPressedDrawable.setBounds(0, mBottomSelectionDividerBottom, mRight, 1665 mBottom); 1666 mVirtualButtonPressedDrawable.draw(canvas); 1667 } 1668 } 1669 1670 // draw the selector wheel 1671 int[] selectorIndices = mSelectorIndices; 1672 for (int i = 0; i < selectorIndices.length; i++) { 1673 int selectorIndex = selectorIndices[i]; 1674 String scrollSelectorValue = mSelectorIndexToStringCache.get(selectorIndex); 1675 // Do not draw the middle item if input is visible since the input 1676 // is shown only if the wheel is static and it covers the middle 1677 // item. Otherwise, if the user starts editing the text via the 1678 // IME they may see a dimmed version of the old value intermixed 1679 // with the new one. 1680 if ((showSelectorWheel && i != SELECTOR_MIDDLE_ITEM_INDEX) || 1681 (i == SELECTOR_MIDDLE_ITEM_INDEX && mInputText.getVisibility() != VISIBLE)) { 1682 canvas.drawText(scrollSelectorValue, x, y, mSelectorWheelPaint); 1683 } 1684 y += mSelectorElementHeight; 1685 } 1686 1687 // draw the selection dividers 1688 if (showSelectorWheel && mSelectionDivider != null) { 1689 // draw the top divider 1690 int topOfTopDivider = mTopSelectionDividerTop; 1691 int bottomOfTopDivider = topOfTopDivider + mSelectionDividerHeight; 1692 mSelectionDivider.setBounds(0, topOfTopDivider, mRight, bottomOfTopDivider); 1693 mSelectionDivider.draw(canvas); 1694 1695 // draw the bottom divider 1696 int bottomOfBottomDivider = mBottomSelectionDividerBottom; 1697 int topOfBottomDivider = bottomOfBottomDivider - mSelectionDividerHeight; 1698 mSelectionDivider.setBounds(0, topOfBottomDivider, mRight, bottomOfBottomDivider); 1699 mSelectionDivider.draw(canvas); 1700 } 1701 } 1702 1703 /** @hide */ 1704 @Override onInitializeAccessibilityEventInternal(AccessibilityEvent event)1705 public void onInitializeAccessibilityEventInternal(AccessibilityEvent event) { 1706 super.onInitializeAccessibilityEventInternal(event); 1707 event.setClassName(NumberPicker.class.getName()); 1708 event.setScrollable(true); 1709 event.setScrollY((mMinValue + mValue) * mSelectorElementHeight); 1710 event.setMaxScrollY((mMaxValue - mMinValue) * mSelectorElementHeight); 1711 } 1712 1713 @Override getAccessibilityNodeProvider()1714 public AccessibilityNodeProvider getAccessibilityNodeProvider() { 1715 if (!mHasSelectorWheel) { 1716 return super.getAccessibilityNodeProvider(); 1717 } 1718 if (mAccessibilityNodeProvider == null) { 1719 mAccessibilityNodeProvider = new AccessibilityNodeProviderImpl(); 1720 } 1721 return mAccessibilityNodeProvider; 1722 } 1723 1724 /** 1725 * Sets the text color for all the states (normal, selected, focused) to be the given color. 1726 * 1727 * @param color A color value in the form 0xAARRGGBB. 1728 */ setTextColor(@olorInt int color)1729 public void setTextColor(@ColorInt int color) { 1730 mSelectorWheelPaint.setColor(color); 1731 mInputText.setTextColor(color); 1732 invalidate(); 1733 } 1734 1735 /** 1736 * @return the text color. 1737 */ 1738 @ColorInt getTextColor()1739 public int getTextColor() { 1740 return mSelectorWheelPaint.getColor(); 1741 } 1742 1743 /** 1744 * Sets the text size to the given value. This value must be > 0 1745 * 1746 * @param size The size in pixel units. 1747 */ setTextSize(@loatRangefrom = 0.0, fromInclusive = false) float size)1748 public void setTextSize(@FloatRange(from = 0.0, fromInclusive = false) float size) { 1749 mSelectorWheelPaint.setTextSize(size); 1750 mInputText.setTextSize(TypedValue.COMPLEX_UNIT_PX, size); 1751 invalidate(); 1752 } 1753 1754 /** 1755 * @return the size (in pixels) of the text size in this NumberPicker. 1756 */ 1757 @FloatRange(from = 0.0, fromInclusive = false) getTextSize()1758 public float getTextSize() { 1759 return mSelectorWheelPaint.getTextSize(); 1760 } 1761 1762 /** 1763 * Makes a measure spec that tries greedily to use the max value. 1764 * 1765 * @param measureSpec The measure spec. 1766 * @param maxSize The max value for the size. 1767 * @return A measure spec greedily imposing the max size. 1768 */ makeMeasureSpec(int measureSpec, int maxSize)1769 private int makeMeasureSpec(int measureSpec, int maxSize) { 1770 if (maxSize == SIZE_UNSPECIFIED) { 1771 return measureSpec; 1772 } 1773 final int size = MeasureSpec.getSize(measureSpec); 1774 final int mode = MeasureSpec.getMode(measureSpec); 1775 switch (mode) { 1776 case MeasureSpec.EXACTLY: 1777 return measureSpec; 1778 case MeasureSpec.AT_MOST: 1779 return MeasureSpec.makeMeasureSpec(Math.min(size, maxSize), MeasureSpec.EXACTLY); 1780 case MeasureSpec.UNSPECIFIED: 1781 return MeasureSpec.makeMeasureSpec(maxSize, MeasureSpec.EXACTLY); 1782 default: 1783 throw new IllegalArgumentException("Unknown measure mode: " + mode); 1784 } 1785 } 1786 1787 /** 1788 * Utility to reconcile a desired size and state, with constraints imposed 1789 * by a MeasureSpec. Tries to respect the min size, unless a different size 1790 * is imposed by the constraints. 1791 * 1792 * @param minSize The minimal desired size. 1793 * @param measuredSize The currently measured size. 1794 * @param measureSpec The current measure spec. 1795 * @return The resolved size and state. 1796 */ resolveSizeAndStateRespectingMinSize( int minSize, int measuredSize, int measureSpec)1797 private int resolveSizeAndStateRespectingMinSize( 1798 int minSize, int measuredSize, int measureSpec) { 1799 if (minSize != SIZE_UNSPECIFIED) { 1800 final int desiredWidth = Math.max(minSize, measuredSize); 1801 return resolveSizeAndState(desiredWidth, measureSpec, 0); 1802 } else { 1803 return measuredSize; 1804 } 1805 } 1806 1807 /** 1808 * Resets the selector indices and clear the cached string representation of 1809 * these indices. 1810 */ 1811 @UnsupportedAppUsage initializeSelectorWheelIndices()1812 private void initializeSelectorWheelIndices() { 1813 mSelectorIndexToStringCache.clear(); 1814 int[] selectorIndices = mSelectorIndices; 1815 int current = getValue(); 1816 for (int i = 0; i < mSelectorIndices.length; i++) { 1817 int selectorIndex = current + (i - SELECTOR_MIDDLE_ITEM_INDEX); 1818 if (mWrapSelectorWheel) { 1819 selectorIndex = getWrappedSelectorIndex(selectorIndex); 1820 } 1821 selectorIndices[i] = selectorIndex; 1822 ensureCachedScrollSelectorValue(selectorIndices[i]); 1823 } 1824 } 1825 1826 /** 1827 * Sets the current value of this NumberPicker. 1828 * 1829 * @param current The new value of the NumberPicker. 1830 * @param notifyChange Whether to notify if the current value changed. 1831 */ setValueInternal(int current, boolean notifyChange)1832 private void setValueInternal(int current, boolean notifyChange) { 1833 if (mValue == current) { 1834 return; 1835 } 1836 // Wrap around the values if we go past the start or end 1837 if (mWrapSelectorWheel) { 1838 current = getWrappedSelectorIndex(current); 1839 } else { 1840 current = Math.max(current, mMinValue); 1841 current = Math.min(current, mMaxValue); 1842 } 1843 int previous = mValue; 1844 mValue = current; 1845 // If we're flinging, we'll update the text view at the end when it becomes visible 1846 if (mScrollState != OnScrollListener.SCROLL_STATE_FLING) { 1847 updateInputTextView(); 1848 } 1849 if (notifyChange) { 1850 notifyChange(previous, current); 1851 } 1852 initializeSelectorWheelIndices(); 1853 invalidate(); 1854 } 1855 1856 /** 1857 * Changes the current value by one which is increment or 1858 * decrement based on the passes argument. 1859 * decrement the current value. 1860 * 1861 * @param increment True to increment, false to decrement. 1862 */ 1863 @UnsupportedAppUsage changeValueByOne(boolean increment)1864 private void changeValueByOne(boolean increment) { 1865 if (mHasSelectorWheel) { 1866 hideSoftInput(); 1867 if (!moveToFinalScrollerPosition(mFlingScroller)) { 1868 moveToFinalScrollerPosition(mAdjustScroller); 1869 } 1870 mPreviousScrollerY = 0; 1871 if (increment) { 1872 mFlingScroller.startScroll(0, 0, 0, -mSelectorElementHeight, SNAP_SCROLL_DURATION); 1873 } else { 1874 mFlingScroller.startScroll(0, 0, 0, mSelectorElementHeight, SNAP_SCROLL_DURATION); 1875 } 1876 invalidate(); 1877 } else { 1878 if (increment) { 1879 setValueInternal(mValue + 1, true); 1880 } else { 1881 setValueInternal(mValue - 1, true); 1882 } 1883 } 1884 } 1885 initializeSelectorWheel()1886 private void initializeSelectorWheel() { 1887 initializeSelectorWheelIndices(); 1888 int[] selectorIndices = mSelectorIndices; 1889 int totalTextHeight = selectorIndices.length * mTextSize; 1890 float totalTextGapHeight = (mBottom - mTop) - totalTextHeight; 1891 float textGapCount = selectorIndices.length; 1892 mSelectorTextGapHeight = (int) (totalTextGapHeight / textGapCount + 0.5f); 1893 mSelectorElementHeight = mTextSize + mSelectorTextGapHeight; 1894 // Ensure that the middle item is positioned the same as the text in 1895 // mInputText 1896 int editTextTextPosition = mInputText.getBaseline() + mInputText.getTop(); 1897 mInitialScrollOffset = editTextTextPosition 1898 - (mSelectorElementHeight * SELECTOR_MIDDLE_ITEM_INDEX); 1899 mCurrentScrollOffset = mInitialScrollOffset; 1900 updateInputTextView(); 1901 } 1902 initializeFadingEdges()1903 private void initializeFadingEdges() { 1904 setVerticalFadingEdgeEnabled(true); 1905 setFadingEdgeLength((mBottom - mTop - mTextSize) / 2); 1906 } 1907 1908 /** 1909 * Callback invoked upon completion of a given <code>scroller</code>. 1910 */ onScrollerFinished(Scroller scroller)1911 private void onScrollerFinished(Scroller scroller) { 1912 if (scroller == mFlingScroller) { 1913 ensureScrollWheelAdjusted(); 1914 updateInputTextView(); 1915 onScrollStateChange(OnScrollListener.SCROLL_STATE_IDLE); 1916 } else { 1917 if (mScrollState != OnScrollListener.SCROLL_STATE_TOUCH_SCROLL) { 1918 updateInputTextView(); 1919 } 1920 } 1921 } 1922 1923 /** 1924 * Handles transition to a given <code>scrollState</code> 1925 */ onScrollStateChange(int scrollState)1926 private void onScrollStateChange(int scrollState) { 1927 if (mScrollState == scrollState) { 1928 return; 1929 } 1930 mScrollState = scrollState; 1931 if (mOnScrollListener != null) { 1932 mOnScrollListener.onScrollStateChange(this, scrollState); 1933 } 1934 } 1935 1936 /** 1937 * Flings the selector with the given <code>velocityY</code>. 1938 */ fling(int velocityY)1939 private void fling(int velocityY) { 1940 mPreviousScrollerY = 0; 1941 1942 if (velocityY > 0) { 1943 mFlingScroller.fling(0, 0, 0, velocityY, 0, 0, 0, Integer.MAX_VALUE); 1944 } else { 1945 mFlingScroller.fling(0, Integer.MAX_VALUE, 0, velocityY, 0, 0, 0, Integer.MAX_VALUE); 1946 } 1947 1948 invalidate(); 1949 } 1950 1951 /** 1952 * @return The wrapped index <code>selectorIndex</code> value. 1953 */ getWrappedSelectorIndex(int selectorIndex)1954 private int getWrappedSelectorIndex(int selectorIndex) { 1955 if (selectorIndex > mMaxValue) { 1956 return mMinValue + (selectorIndex - mMaxValue) % (mMaxValue - mMinValue) - 1; 1957 } else if (selectorIndex < mMinValue) { 1958 return mMaxValue - (mMinValue - selectorIndex) % (mMaxValue - mMinValue) + 1; 1959 } 1960 return selectorIndex; 1961 } 1962 1963 /** 1964 * Increments the <code>selectorIndices</code> whose string representations 1965 * will be displayed in the selector. 1966 */ incrementSelectorIndices(int[] selectorIndices)1967 private void incrementSelectorIndices(int[] selectorIndices) { 1968 for (int i = 0; i < selectorIndices.length - 1; i++) { 1969 selectorIndices[i] = selectorIndices[i + 1]; 1970 } 1971 int nextScrollSelectorIndex = selectorIndices[selectorIndices.length - 2] + 1; 1972 if (mWrapSelectorWheel && nextScrollSelectorIndex > mMaxValue) { 1973 nextScrollSelectorIndex = mMinValue; 1974 } 1975 selectorIndices[selectorIndices.length - 1] = nextScrollSelectorIndex; 1976 ensureCachedScrollSelectorValue(nextScrollSelectorIndex); 1977 } 1978 1979 /** 1980 * Decrements the <code>selectorIndices</code> whose string representations 1981 * will be displayed in the selector. 1982 */ decrementSelectorIndices(int[] selectorIndices)1983 private void decrementSelectorIndices(int[] selectorIndices) { 1984 for (int i = selectorIndices.length - 1; i > 0; i--) { 1985 selectorIndices[i] = selectorIndices[i - 1]; 1986 } 1987 int nextScrollSelectorIndex = selectorIndices[1] - 1; 1988 if (mWrapSelectorWheel && nextScrollSelectorIndex < mMinValue) { 1989 nextScrollSelectorIndex = mMaxValue; 1990 } 1991 selectorIndices[0] = nextScrollSelectorIndex; 1992 ensureCachedScrollSelectorValue(nextScrollSelectorIndex); 1993 } 1994 1995 /** 1996 * Ensures we have a cached string representation of the given <code> 1997 * selectorIndex</code> to avoid multiple instantiations of the same string. 1998 */ ensureCachedScrollSelectorValue(int selectorIndex)1999 private void ensureCachedScrollSelectorValue(int selectorIndex) { 2000 SparseArray<String> cache = mSelectorIndexToStringCache; 2001 String scrollSelectorValue = cache.get(selectorIndex); 2002 if (scrollSelectorValue != null) { 2003 return; 2004 } 2005 if (selectorIndex < mMinValue || selectorIndex > mMaxValue) { 2006 scrollSelectorValue = ""; 2007 } else { 2008 if (mDisplayedValues != null) { 2009 int displayedValueIndex = selectorIndex - mMinValue; 2010 scrollSelectorValue = mDisplayedValues[displayedValueIndex]; 2011 } else { 2012 scrollSelectorValue = formatNumber(selectorIndex); 2013 } 2014 } 2015 cache.put(selectorIndex, scrollSelectorValue); 2016 } 2017 formatNumber(int value)2018 private String formatNumber(int value) { 2019 return (mFormatter != null) ? mFormatter.format(value) : formatNumberWithLocale(value); 2020 } 2021 validateInputTextView(View v)2022 private void validateInputTextView(View v) { 2023 String str = String.valueOf(((TextView) v).getText()); 2024 if (TextUtils.isEmpty(str)) { 2025 // Restore to the old value as we don't allow empty values 2026 updateInputTextView(); 2027 } else { 2028 // Check the new value and ensure it's in range 2029 int current = getSelectedPos(str.toString()); 2030 setValueInternal(current, true); 2031 } 2032 } 2033 2034 /** 2035 * Updates the view of this NumberPicker. If displayValues were specified in 2036 * the string corresponding to the index specified by the current value will 2037 * be returned. Otherwise, the formatter specified in {@link #setFormatter} 2038 * will be used to format the number. 2039 * 2040 * @return Whether the text was updated. 2041 */ updateInputTextView()2042 private boolean updateInputTextView() { 2043 /* 2044 * If we don't have displayed values then use the current number else 2045 * find the correct value in the displayed values for the current 2046 * number. 2047 */ 2048 String text = (mDisplayedValues == null) ? formatNumber(mValue) 2049 : mDisplayedValues[mValue - mMinValue]; 2050 if (!TextUtils.isEmpty(text)) { 2051 CharSequence beforeText = mInputText.getText(); 2052 if (!text.equals(beforeText.toString())) { 2053 mInputText.setText(text); 2054 if (AccessibilityManager.getInstance(mContext).isEnabled()) { 2055 AccessibilityEvent event = AccessibilityEvent.obtain( 2056 AccessibilityEvent.TYPE_VIEW_TEXT_CHANGED); 2057 mInputText.onInitializeAccessibilityEvent(event); 2058 mInputText.onPopulateAccessibilityEvent(event); 2059 event.setFromIndex(0); 2060 event.setRemovedCount(beforeText.length()); 2061 event.setAddedCount(text.length()); 2062 event.setBeforeText(beforeText); 2063 event.setSource(NumberPicker.this, 2064 AccessibilityNodeProviderImpl.VIRTUAL_VIEW_ID_INPUT); 2065 requestSendAccessibilityEvent(NumberPicker.this, event); 2066 } 2067 return true; 2068 } 2069 } 2070 2071 return false; 2072 } 2073 2074 /** 2075 * Notifies the listener, if registered, of a change of the value of this 2076 * NumberPicker. 2077 */ notifyChange(int previous, int current)2078 private void notifyChange(int previous, int current) { 2079 if (mOnValueChangeListener != null) { 2080 mOnValueChangeListener.onValueChange(this, previous, mValue); 2081 } 2082 } 2083 2084 /** 2085 * Posts a command for changing the current value by one. 2086 * 2087 * @param increment Whether to increment or decrement the value. 2088 */ postChangeCurrentByOneFromLongPress(boolean increment, long delayMillis)2089 private void postChangeCurrentByOneFromLongPress(boolean increment, long delayMillis) { 2090 if (mChangeCurrentByOneFromLongPressCommand == null) { 2091 mChangeCurrentByOneFromLongPressCommand = new ChangeCurrentByOneFromLongPressCommand(); 2092 } else { 2093 removeCallbacks(mChangeCurrentByOneFromLongPressCommand); 2094 } 2095 mChangeCurrentByOneFromLongPressCommand.setStep(increment); 2096 postDelayed(mChangeCurrentByOneFromLongPressCommand, delayMillis); 2097 } 2098 2099 /** 2100 * Removes the command for changing the current value by one. 2101 */ removeChangeCurrentByOneFromLongPress()2102 private void removeChangeCurrentByOneFromLongPress() { 2103 if (mChangeCurrentByOneFromLongPressCommand != null) { 2104 removeCallbacks(mChangeCurrentByOneFromLongPressCommand); 2105 } 2106 } 2107 2108 /** 2109 * Posts a command for beginning an edit of the current value via IME on 2110 * long press. 2111 */ postBeginSoftInputOnLongPressCommand()2112 private void postBeginSoftInputOnLongPressCommand() { 2113 if (mBeginSoftInputOnLongPressCommand == null) { 2114 mBeginSoftInputOnLongPressCommand = new BeginSoftInputOnLongPressCommand(); 2115 } else { 2116 removeCallbacks(mBeginSoftInputOnLongPressCommand); 2117 } 2118 postDelayed(mBeginSoftInputOnLongPressCommand, ViewConfiguration.getLongPressTimeout()); 2119 } 2120 2121 /** 2122 * Removes the command for beginning an edit of the current value via IME. 2123 */ removeBeginSoftInputCommand()2124 private void removeBeginSoftInputCommand() { 2125 if (mBeginSoftInputOnLongPressCommand != null) { 2126 removeCallbacks(mBeginSoftInputOnLongPressCommand); 2127 } 2128 } 2129 2130 /** 2131 * Removes all pending callback from the message queue. 2132 */ removeAllCallbacks()2133 private void removeAllCallbacks() { 2134 if (mChangeCurrentByOneFromLongPressCommand != null) { 2135 removeCallbacks(mChangeCurrentByOneFromLongPressCommand); 2136 } 2137 if (mSetSelectionCommand != null) { 2138 mSetSelectionCommand.cancel(); 2139 } 2140 if (mBeginSoftInputOnLongPressCommand != null) { 2141 removeCallbacks(mBeginSoftInputOnLongPressCommand); 2142 } 2143 mPressedStateHelper.cancel(); 2144 } 2145 2146 /** 2147 * @return The selected index given its displayed <code>value</code>. 2148 */ getSelectedPos(String value)2149 private int getSelectedPos(String value) { 2150 if (mDisplayedValues == null) { 2151 try { 2152 return Integer.parseInt(value); 2153 } catch (NumberFormatException e) { 2154 // Ignore as if it's not a number we don't care 2155 } 2156 } else { 2157 for (int i = 0; i < mDisplayedValues.length; i++) { 2158 // Don't force the user to type in jan when ja will do 2159 value = value.toLowerCase(); 2160 if (mDisplayedValues[i].toLowerCase().startsWith(value)) { 2161 return mMinValue + i; 2162 } 2163 } 2164 2165 /* 2166 * The user might have typed in a number into the month field i.e. 2167 * 10 instead of OCT so support that too. 2168 */ 2169 try { 2170 return Integer.parseInt(value); 2171 } catch (NumberFormatException e) { 2172 2173 // Ignore as if it's not a number we don't care 2174 } 2175 } 2176 return mMinValue; 2177 } 2178 2179 /** 2180 * Posts a {@link SetSelectionCommand} from the given 2181 * {@code selectionStart} to {@code selectionEnd}. 2182 */ postSetSelectionCommand(int selectionStart, int selectionEnd)2183 private void postSetSelectionCommand(int selectionStart, int selectionEnd) { 2184 if (mSetSelectionCommand == null) { 2185 mSetSelectionCommand = new SetSelectionCommand(mInputText); 2186 } 2187 mSetSelectionCommand.post(selectionStart, selectionEnd); 2188 } 2189 2190 /** 2191 * The numbers accepted by the input text's {@link Filter} 2192 */ 2193 private static final char[] DIGIT_CHARACTERS = new char[] { 2194 // Latin digits are the common case 2195 '0', '1', '2', '3', '4', '5', '6', '7', '8', '9', 2196 // Arabic-Indic 2197 '\u0660', '\u0661', '\u0662', '\u0663', '\u0664', '\u0665', '\u0666', '\u0667', '\u0668' 2198 , '\u0669', 2199 // Extended Arabic-Indic 2200 '\u06f0', '\u06f1', '\u06f2', '\u06f3', '\u06f4', '\u06f5', '\u06f6', '\u06f7', '\u06f8' 2201 , '\u06f9', 2202 // Hindi and Marathi (Devanagari script) 2203 '\u0966', '\u0967', '\u0968', '\u0969', '\u096a', '\u096b', '\u096c', '\u096d', '\u096e' 2204 , '\u096f', 2205 // Bengali 2206 '\u09e6', '\u09e7', '\u09e8', '\u09e9', '\u09ea', '\u09eb', '\u09ec', '\u09ed', '\u09ee' 2207 , '\u09ef', 2208 // Kannada 2209 '\u0ce6', '\u0ce7', '\u0ce8', '\u0ce9', '\u0cea', '\u0ceb', '\u0cec', '\u0ced', '\u0cee' 2210 , '\u0cef' 2211 }; 2212 2213 /** 2214 * Filter for accepting only valid indices or prefixes of the string 2215 * representation of valid indices. 2216 */ 2217 class InputTextFilter extends NumberKeyListener { 2218 2219 // XXX This doesn't allow for range limits when controlled by a 2220 // soft input method! getInputType()2221 public int getInputType() { 2222 return InputType.TYPE_CLASS_TEXT; 2223 } 2224 2225 @Override getAcceptedChars()2226 protected char[] getAcceptedChars() { 2227 return DIGIT_CHARACTERS; 2228 } 2229 2230 @Override filter( CharSequence source, int start, int end, Spanned dest, int dstart, int dend)2231 public CharSequence filter( 2232 CharSequence source, int start, int end, Spanned dest, int dstart, int dend) { 2233 // We don't know what the output will be, so always cancel any 2234 // pending set selection command. 2235 if (mSetSelectionCommand != null) { 2236 mSetSelectionCommand.cancel(); 2237 } 2238 2239 if (mDisplayedValues == null) { 2240 CharSequence filtered = super.filter(source, start, end, dest, dstart, dend); 2241 if (filtered == null) { 2242 filtered = source.subSequence(start, end); 2243 } 2244 2245 String result = String.valueOf(dest.subSequence(0, dstart)) + filtered 2246 + dest.subSequence(dend, dest.length()); 2247 2248 if ("".equals(result)) { 2249 return result; 2250 } 2251 int val = getSelectedPos(result); 2252 2253 /* 2254 * Ensure the user can't type in a value greater than the max 2255 * allowed. We have to allow less than min as the user might 2256 * want to delete some numbers and then type a new number. 2257 * And prevent multiple-"0" that exceeds the length of upper 2258 * bound number. 2259 */ 2260 if (val > mMaxValue || result.length() > String.valueOf(mMaxValue).length()) { 2261 return ""; 2262 } else { 2263 return filtered; 2264 } 2265 } else { 2266 CharSequence filtered = String.valueOf(source.subSequence(start, end)); 2267 if (TextUtils.isEmpty(filtered)) { 2268 return ""; 2269 } 2270 String result = String.valueOf(dest.subSequence(0, dstart)) + filtered 2271 + dest.subSequence(dend, dest.length()); 2272 String str = String.valueOf(result).toLowerCase(); 2273 for (String val : mDisplayedValues) { 2274 String valLowerCase = val.toLowerCase(); 2275 if (valLowerCase.startsWith(str)) { 2276 postSetSelectionCommand(result.length(), val.length()); 2277 return val.subSequence(dstart, val.length()); 2278 } 2279 } 2280 return ""; 2281 } 2282 } 2283 } 2284 2285 /** 2286 * Ensures that the scroll wheel is adjusted i.e. there is no offset and the 2287 * middle element is in the middle of the widget. 2288 * 2289 * @return Whether an adjustment has been made. 2290 */ ensureScrollWheelAdjusted()2291 private boolean ensureScrollWheelAdjusted() { 2292 // adjust to the closest value 2293 int deltaY = mInitialScrollOffset - mCurrentScrollOffset; 2294 if (deltaY != 0) { 2295 mPreviousScrollerY = 0; 2296 if (Math.abs(deltaY) > mSelectorElementHeight / 2) { 2297 deltaY += (deltaY > 0) ? -mSelectorElementHeight : mSelectorElementHeight; 2298 } 2299 mAdjustScroller.startScroll(0, 0, 0, deltaY, SELECTOR_ADJUSTMENT_DURATION_MILLIS); 2300 invalidate(); 2301 return true; 2302 } 2303 return false; 2304 } 2305 2306 class PressedStateHelper implements Runnable { 2307 public static final int BUTTON_INCREMENT = 1; 2308 public static final int BUTTON_DECREMENT = 2; 2309 2310 private final int MODE_PRESS = 1; 2311 private final int MODE_TAPPED = 2; 2312 2313 private int mManagedButton; 2314 private int mMode; 2315 cancel()2316 public void cancel() { 2317 mMode = 0; 2318 mManagedButton = 0; 2319 NumberPicker.this.removeCallbacks(this); 2320 if (mIncrementVirtualButtonPressed) { 2321 mIncrementVirtualButtonPressed = false; 2322 invalidate(0, mBottomSelectionDividerBottom, mRight, mBottom); 2323 } 2324 mDecrementVirtualButtonPressed = false; 2325 if (mDecrementVirtualButtonPressed) { 2326 invalidate(0, 0, mRight, mTopSelectionDividerTop); 2327 } 2328 } 2329 buttonPressDelayed(int button)2330 public void buttonPressDelayed(int button) { 2331 cancel(); 2332 mMode = MODE_PRESS; 2333 mManagedButton = button; 2334 NumberPicker.this.postDelayed(this, ViewConfiguration.getTapTimeout()); 2335 } 2336 buttonTapped(int button)2337 public void buttonTapped(int button) { 2338 cancel(); 2339 mMode = MODE_TAPPED; 2340 mManagedButton = button; 2341 NumberPicker.this.post(this); 2342 } 2343 2344 @Override run()2345 public void run() { 2346 switch (mMode) { 2347 case MODE_PRESS: { 2348 switch (mManagedButton) { 2349 case BUTTON_INCREMENT: { 2350 mIncrementVirtualButtonPressed = true; 2351 invalidate(0, mBottomSelectionDividerBottom, mRight, mBottom); 2352 } break; 2353 case BUTTON_DECREMENT: { 2354 mDecrementVirtualButtonPressed = true; 2355 invalidate(0, 0, mRight, mTopSelectionDividerTop); 2356 } 2357 } 2358 } break; 2359 case MODE_TAPPED: { 2360 switch (mManagedButton) { 2361 case BUTTON_INCREMENT: { 2362 if (!mIncrementVirtualButtonPressed) { 2363 NumberPicker.this.postDelayed(this, 2364 ViewConfiguration.getPressedStateDuration()); 2365 } 2366 mIncrementVirtualButtonPressed ^= true; 2367 invalidate(0, mBottomSelectionDividerBottom, mRight, mBottom); 2368 } break; 2369 case BUTTON_DECREMENT: { 2370 if (!mDecrementVirtualButtonPressed) { 2371 NumberPicker.this.postDelayed(this, 2372 ViewConfiguration.getPressedStateDuration()); 2373 } 2374 mDecrementVirtualButtonPressed ^= true; 2375 invalidate(0, 0, mRight, mTopSelectionDividerTop); 2376 } 2377 } 2378 } break; 2379 } 2380 } 2381 } 2382 2383 /** 2384 * Command for setting the input text selection. 2385 */ 2386 private static class SetSelectionCommand implements Runnable { 2387 private final EditText mInputText; 2388 2389 private int mSelectionStart; 2390 private int mSelectionEnd; 2391 2392 /** Whether this runnable is currently posted. */ 2393 private boolean mPosted; 2394 SetSelectionCommand(EditText inputText)2395 public SetSelectionCommand(EditText inputText) { 2396 mInputText = inputText; 2397 } 2398 post(int selectionStart, int selectionEnd)2399 public void post(int selectionStart, int selectionEnd) { 2400 mSelectionStart = selectionStart; 2401 mSelectionEnd = selectionEnd; 2402 2403 if (!mPosted) { 2404 mInputText.post(this); 2405 mPosted = true; 2406 } 2407 } 2408 cancel()2409 public void cancel() { 2410 if (mPosted) { 2411 mInputText.removeCallbacks(this); 2412 mPosted = false; 2413 } 2414 } 2415 2416 @Override run()2417 public void run() { 2418 mPosted = false; 2419 mInputText.setSelection(mSelectionStart, mSelectionEnd); 2420 } 2421 } 2422 2423 /** 2424 * Command for changing the current value from a long press by one. 2425 */ 2426 class ChangeCurrentByOneFromLongPressCommand implements Runnable { 2427 private boolean mIncrement; 2428 setStep(boolean increment)2429 private void setStep(boolean increment) { 2430 mIncrement = increment; 2431 } 2432 2433 @Override run()2434 public void run() { 2435 changeValueByOne(mIncrement); 2436 postDelayed(this, mLongPressUpdateInterval); 2437 } 2438 } 2439 2440 /** 2441 * @hide 2442 */ 2443 public static class CustomEditText extends EditText { 2444 CustomEditText(Context context, AttributeSet attrs)2445 public CustomEditText(Context context, AttributeSet attrs) { 2446 super(context, attrs); 2447 } 2448 2449 @Override onEditorAction(int actionCode)2450 public void onEditorAction(int actionCode) { 2451 super.onEditorAction(actionCode); 2452 if (actionCode == EditorInfo.IME_ACTION_DONE) { 2453 clearFocus(); 2454 } 2455 } 2456 } 2457 2458 /** 2459 * Command for beginning soft input on long press. 2460 */ 2461 class BeginSoftInputOnLongPressCommand implements Runnable { 2462 2463 @Override run()2464 public void run() { 2465 performLongClick(); 2466 } 2467 } 2468 2469 /** 2470 * Class for managing virtual view tree rooted at this picker. 2471 */ 2472 class AccessibilityNodeProviderImpl extends AccessibilityNodeProvider { 2473 private static final int UNDEFINED = Integer.MIN_VALUE; 2474 2475 private static final int VIRTUAL_VIEW_ID_INCREMENT = 1; 2476 2477 private static final int VIRTUAL_VIEW_ID_INPUT = 2; 2478 2479 private static final int VIRTUAL_VIEW_ID_DECREMENT = 3; 2480 2481 private final Rect mTempRect = new Rect(); 2482 2483 private final int[] mTempArray = new int[2]; 2484 2485 private int mAccessibilityFocusedView = UNDEFINED; 2486 2487 @Override createAccessibilityNodeInfo(int virtualViewId)2488 public AccessibilityNodeInfo createAccessibilityNodeInfo(int virtualViewId) { 2489 switch (virtualViewId) { 2490 case View.NO_ID: 2491 return createAccessibilityNodeInfoForNumberPicker( mScrollX, mScrollY, 2492 mScrollX + (mRight - mLeft), mScrollY + (mBottom - mTop)); 2493 case VIRTUAL_VIEW_ID_DECREMENT: 2494 return createAccessibilityNodeInfoForVirtualButton(VIRTUAL_VIEW_ID_DECREMENT, 2495 getVirtualDecrementButtonText(), mScrollX, mScrollY, 2496 mScrollX + (mRight - mLeft), 2497 mTopSelectionDividerTop + mSelectionDividerHeight); 2498 case VIRTUAL_VIEW_ID_INPUT: 2499 return createAccessibiltyNodeInfoForInputText(mScrollX, 2500 mTopSelectionDividerTop + mSelectionDividerHeight, 2501 mScrollX + (mRight - mLeft), 2502 mBottomSelectionDividerBottom - mSelectionDividerHeight); 2503 case VIRTUAL_VIEW_ID_INCREMENT: 2504 return createAccessibilityNodeInfoForVirtualButton(VIRTUAL_VIEW_ID_INCREMENT, 2505 getVirtualIncrementButtonText(), mScrollX, 2506 mBottomSelectionDividerBottom - mSelectionDividerHeight, 2507 mScrollX + (mRight - mLeft), mScrollY + (mBottom - mTop)); 2508 } 2509 return super.createAccessibilityNodeInfo(virtualViewId); 2510 } 2511 2512 @Override findAccessibilityNodeInfosByText(String searched, int virtualViewId)2513 public List<AccessibilityNodeInfo> findAccessibilityNodeInfosByText(String searched, 2514 int virtualViewId) { 2515 if (TextUtils.isEmpty(searched)) { 2516 return Collections.emptyList(); 2517 } 2518 String searchedLowerCase = searched.toLowerCase(); 2519 List<AccessibilityNodeInfo> result = new ArrayList<AccessibilityNodeInfo>(); 2520 switch (virtualViewId) { 2521 case View.NO_ID: { 2522 findAccessibilityNodeInfosByTextInChild(searchedLowerCase, 2523 VIRTUAL_VIEW_ID_DECREMENT, result); 2524 findAccessibilityNodeInfosByTextInChild(searchedLowerCase, 2525 VIRTUAL_VIEW_ID_INPUT, result); 2526 findAccessibilityNodeInfosByTextInChild(searchedLowerCase, 2527 VIRTUAL_VIEW_ID_INCREMENT, result); 2528 return result; 2529 } 2530 case VIRTUAL_VIEW_ID_DECREMENT: 2531 case VIRTUAL_VIEW_ID_INCREMENT: 2532 case VIRTUAL_VIEW_ID_INPUT: { 2533 findAccessibilityNodeInfosByTextInChild(searchedLowerCase, virtualViewId, 2534 result); 2535 return result; 2536 } 2537 } 2538 return super.findAccessibilityNodeInfosByText(searched, virtualViewId); 2539 } 2540 2541 @Override performAction(int virtualViewId, int action, Bundle arguments)2542 public boolean performAction(int virtualViewId, int action, Bundle arguments) { 2543 switch (virtualViewId) { 2544 case View.NO_ID: { 2545 switch (action) { 2546 case AccessibilityNodeInfo.ACTION_ACCESSIBILITY_FOCUS: { 2547 if (mAccessibilityFocusedView != virtualViewId) { 2548 mAccessibilityFocusedView = virtualViewId; 2549 requestAccessibilityFocus(); 2550 return true; 2551 } 2552 } return false; 2553 case AccessibilityNodeInfo.ACTION_CLEAR_ACCESSIBILITY_FOCUS: { 2554 if (mAccessibilityFocusedView == virtualViewId) { 2555 mAccessibilityFocusedView = UNDEFINED; 2556 clearAccessibilityFocus(); 2557 return true; 2558 } 2559 return false; 2560 } 2561 case AccessibilityNodeInfo.ACTION_SCROLL_FORWARD: 2562 case R.id.accessibilityActionScrollDown: { 2563 if (NumberPicker.this.isEnabled() 2564 && (getWrapSelectorWheel() || getValue() < getMaxValue())) { 2565 changeValueByOne(true); 2566 return true; 2567 } 2568 } return false; 2569 case AccessibilityNodeInfo.ACTION_SCROLL_BACKWARD: 2570 case R.id.accessibilityActionScrollUp: { 2571 if (NumberPicker.this.isEnabled() 2572 && (getWrapSelectorWheel() || getValue() > getMinValue())) { 2573 changeValueByOne(false); 2574 return true; 2575 } 2576 } return false; 2577 } 2578 } break; 2579 case VIRTUAL_VIEW_ID_INPUT: { 2580 switch (action) { 2581 case AccessibilityNodeInfo.ACTION_FOCUS: { 2582 if (NumberPicker.this.isEnabled() && !mInputText.isFocused()) { 2583 return mInputText.requestFocus(); 2584 } 2585 } break; 2586 case AccessibilityNodeInfo.ACTION_CLEAR_FOCUS: { 2587 if (NumberPicker.this.isEnabled() && mInputText.isFocused()) { 2588 mInputText.clearFocus(); 2589 return true; 2590 } 2591 return false; 2592 } 2593 case AccessibilityNodeInfo.ACTION_CLICK: { 2594 if (NumberPicker.this.isEnabled()) { 2595 performClick(); 2596 return true; 2597 } 2598 return false; 2599 } 2600 case AccessibilityNodeInfo.ACTION_LONG_CLICK: { 2601 if (NumberPicker.this.isEnabled()) { 2602 performLongClick(); 2603 return true; 2604 } 2605 return false; 2606 } 2607 case AccessibilityNodeInfo.ACTION_ACCESSIBILITY_FOCUS: { 2608 if (mAccessibilityFocusedView != virtualViewId) { 2609 mAccessibilityFocusedView = virtualViewId; 2610 sendAccessibilityEventForVirtualView(virtualViewId, 2611 AccessibilityEvent.TYPE_VIEW_ACCESSIBILITY_FOCUSED); 2612 mInputText.invalidate(); 2613 return true; 2614 } 2615 } return false; 2616 case AccessibilityNodeInfo.ACTION_CLEAR_ACCESSIBILITY_FOCUS: { 2617 if (mAccessibilityFocusedView == virtualViewId) { 2618 mAccessibilityFocusedView = UNDEFINED; 2619 sendAccessibilityEventForVirtualView(virtualViewId, 2620 AccessibilityEvent.TYPE_VIEW_ACCESSIBILITY_FOCUS_CLEARED); 2621 mInputText.invalidate(); 2622 return true; 2623 } 2624 } return false; 2625 default: { 2626 return mInputText.performAccessibilityAction(action, arguments); 2627 } 2628 } 2629 } return false; 2630 case VIRTUAL_VIEW_ID_INCREMENT: { 2631 switch (action) { 2632 case AccessibilityNodeInfo.ACTION_CLICK: { 2633 if (NumberPicker.this.isEnabled()) { 2634 NumberPicker.this.changeValueByOne(true); 2635 sendAccessibilityEventForVirtualView(virtualViewId, 2636 AccessibilityEvent.TYPE_VIEW_CLICKED); 2637 return true; 2638 } 2639 } return false; 2640 case AccessibilityNodeInfo.ACTION_ACCESSIBILITY_FOCUS: { 2641 if (mAccessibilityFocusedView != virtualViewId) { 2642 mAccessibilityFocusedView = virtualViewId; 2643 sendAccessibilityEventForVirtualView(virtualViewId, 2644 AccessibilityEvent.TYPE_VIEW_ACCESSIBILITY_FOCUSED); 2645 invalidate(0, mBottomSelectionDividerBottom, mRight, mBottom); 2646 return true; 2647 } 2648 } return false; 2649 case AccessibilityNodeInfo.ACTION_CLEAR_ACCESSIBILITY_FOCUS: { 2650 if (mAccessibilityFocusedView == virtualViewId) { 2651 mAccessibilityFocusedView = UNDEFINED; 2652 sendAccessibilityEventForVirtualView(virtualViewId, 2653 AccessibilityEvent.TYPE_VIEW_ACCESSIBILITY_FOCUS_CLEARED); 2654 invalidate(0, mBottomSelectionDividerBottom, mRight, mBottom); 2655 return true; 2656 } 2657 } return false; 2658 } 2659 } return false; 2660 case VIRTUAL_VIEW_ID_DECREMENT: { 2661 switch (action) { 2662 case AccessibilityNodeInfo.ACTION_CLICK: { 2663 if (NumberPicker.this.isEnabled()) { 2664 final boolean increment = (virtualViewId == VIRTUAL_VIEW_ID_INCREMENT); 2665 NumberPicker.this.changeValueByOne(increment); 2666 sendAccessibilityEventForVirtualView(virtualViewId, 2667 AccessibilityEvent.TYPE_VIEW_CLICKED); 2668 return true; 2669 } 2670 } return false; 2671 case AccessibilityNodeInfo.ACTION_ACCESSIBILITY_FOCUS: { 2672 if (mAccessibilityFocusedView != virtualViewId) { 2673 mAccessibilityFocusedView = virtualViewId; 2674 sendAccessibilityEventForVirtualView(virtualViewId, 2675 AccessibilityEvent.TYPE_VIEW_ACCESSIBILITY_FOCUSED); 2676 invalidate(0, 0, mRight, mTopSelectionDividerTop); 2677 return true; 2678 } 2679 } return false; 2680 case AccessibilityNodeInfo.ACTION_CLEAR_ACCESSIBILITY_FOCUS: { 2681 if (mAccessibilityFocusedView == virtualViewId) { 2682 mAccessibilityFocusedView = UNDEFINED; 2683 sendAccessibilityEventForVirtualView(virtualViewId, 2684 AccessibilityEvent.TYPE_VIEW_ACCESSIBILITY_FOCUS_CLEARED); 2685 invalidate(0, 0, mRight, mTopSelectionDividerTop); 2686 return true; 2687 } 2688 } return false; 2689 } 2690 } return false; 2691 } 2692 return super.performAction(virtualViewId, action, arguments); 2693 } 2694 sendAccessibilityEventForVirtualView(int virtualViewId, int eventType)2695 public void sendAccessibilityEventForVirtualView(int virtualViewId, int eventType) { 2696 switch (virtualViewId) { 2697 case VIRTUAL_VIEW_ID_DECREMENT: { 2698 if (hasVirtualDecrementButton()) { 2699 sendAccessibilityEventForVirtualButton(virtualViewId, eventType, 2700 getVirtualDecrementButtonText()); 2701 } 2702 } break; 2703 case VIRTUAL_VIEW_ID_INPUT: { 2704 sendAccessibilityEventForVirtualText(eventType); 2705 } break; 2706 case VIRTUAL_VIEW_ID_INCREMENT: { 2707 if (hasVirtualIncrementButton()) { 2708 sendAccessibilityEventForVirtualButton(virtualViewId, eventType, 2709 getVirtualIncrementButtonText()); 2710 } 2711 } break; 2712 } 2713 } 2714 sendAccessibilityEventForVirtualText(int eventType)2715 private void sendAccessibilityEventForVirtualText(int eventType) { 2716 if (AccessibilityManager.getInstance(mContext).isEnabled()) { 2717 AccessibilityEvent event = AccessibilityEvent.obtain(eventType); 2718 mInputText.onInitializeAccessibilityEvent(event); 2719 mInputText.onPopulateAccessibilityEvent(event); 2720 event.setSource(NumberPicker.this, VIRTUAL_VIEW_ID_INPUT); 2721 requestSendAccessibilityEvent(NumberPicker.this, event); 2722 } 2723 } 2724 sendAccessibilityEventForVirtualButton(int virtualViewId, int eventType, String text)2725 private void sendAccessibilityEventForVirtualButton(int virtualViewId, int eventType, 2726 String text) { 2727 if (AccessibilityManager.getInstance(mContext).isEnabled()) { 2728 AccessibilityEvent event = AccessibilityEvent.obtain(eventType); 2729 event.setClassName(Button.class.getName()); 2730 event.setPackageName(mContext.getPackageName()); 2731 event.getText().add(text); 2732 event.setEnabled(NumberPicker.this.isEnabled()); 2733 event.setSource(NumberPicker.this, virtualViewId); 2734 requestSendAccessibilityEvent(NumberPicker.this, event); 2735 } 2736 } 2737 findAccessibilityNodeInfosByTextInChild(String searchedLowerCase, int virtualViewId, List<AccessibilityNodeInfo> outResult)2738 private void findAccessibilityNodeInfosByTextInChild(String searchedLowerCase, 2739 int virtualViewId, List<AccessibilityNodeInfo> outResult) { 2740 switch (virtualViewId) { 2741 case VIRTUAL_VIEW_ID_DECREMENT: { 2742 String text = getVirtualDecrementButtonText(); 2743 if (!TextUtils.isEmpty(text) 2744 && text.toString().toLowerCase().contains(searchedLowerCase)) { 2745 outResult.add(createAccessibilityNodeInfo(VIRTUAL_VIEW_ID_DECREMENT)); 2746 } 2747 } return; 2748 case VIRTUAL_VIEW_ID_INPUT: { 2749 CharSequence text = mInputText.getText(); 2750 if (!TextUtils.isEmpty(text) && 2751 text.toString().toLowerCase().contains(searchedLowerCase)) { 2752 outResult.add(createAccessibilityNodeInfo(VIRTUAL_VIEW_ID_INPUT)); 2753 return; 2754 } 2755 CharSequence contentDesc = mInputText.getText(); 2756 if (!TextUtils.isEmpty(contentDesc) && 2757 contentDesc.toString().toLowerCase().contains(searchedLowerCase)) { 2758 outResult.add(createAccessibilityNodeInfo(VIRTUAL_VIEW_ID_INPUT)); 2759 return; 2760 } 2761 } break; 2762 case VIRTUAL_VIEW_ID_INCREMENT: { 2763 String text = getVirtualIncrementButtonText(); 2764 if (!TextUtils.isEmpty(text) 2765 && text.toString().toLowerCase().contains(searchedLowerCase)) { 2766 outResult.add(createAccessibilityNodeInfo(VIRTUAL_VIEW_ID_INCREMENT)); 2767 } 2768 } return; 2769 } 2770 } 2771 createAccessibiltyNodeInfoForInputText( int left, int top, int right, int bottom)2772 private AccessibilityNodeInfo createAccessibiltyNodeInfoForInputText( 2773 int left, int top, int right, int bottom) { 2774 AccessibilityNodeInfo info = mInputText.createAccessibilityNodeInfo(); 2775 info.setSource(NumberPicker.this, VIRTUAL_VIEW_ID_INPUT); 2776 info.setAccessibilityFocused(mAccessibilityFocusedView == VIRTUAL_VIEW_ID_INPUT); 2777 if (mAccessibilityFocusedView != VIRTUAL_VIEW_ID_INPUT) { 2778 info.addAction(AccessibilityNodeInfo.ACTION_ACCESSIBILITY_FOCUS); 2779 } 2780 if (mAccessibilityFocusedView == VIRTUAL_VIEW_ID_INPUT) { 2781 info.addAction(AccessibilityNodeInfo.ACTION_CLEAR_ACCESSIBILITY_FOCUS); 2782 } 2783 Rect boundsInParent = mTempRect; 2784 boundsInParent.set(left, top, right, bottom); 2785 info.setVisibleToUser(isVisibleToUser(boundsInParent)); 2786 info.setBoundsInParent(boundsInParent); 2787 Rect boundsInScreen = boundsInParent; 2788 int[] locationOnScreen = mTempArray; 2789 getLocationOnScreen(locationOnScreen); 2790 boundsInScreen.offset(locationOnScreen[0], locationOnScreen[1]); 2791 info.setBoundsInScreen(boundsInScreen); 2792 return info; 2793 } 2794 createAccessibilityNodeInfoForVirtualButton(int virtualViewId, String text, int left, int top, int right, int bottom)2795 private AccessibilityNodeInfo createAccessibilityNodeInfoForVirtualButton(int virtualViewId, 2796 String text, int left, int top, int right, int bottom) { 2797 AccessibilityNodeInfo info = AccessibilityNodeInfo.obtain(); 2798 info.setClassName(Button.class.getName()); 2799 info.setPackageName(mContext.getPackageName()); 2800 info.setSource(NumberPicker.this, virtualViewId); 2801 info.setParent(NumberPicker.this); 2802 info.setText(text); 2803 info.setClickable(true); 2804 info.setLongClickable(true); 2805 info.setEnabled(NumberPicker.this.isEnabled()); 2806 info.setAccessibilityFocused(mAccessibilityFocusedView == virtualViewId); 2807 Rect boundsInParent = mTempRect; 2808 boundsInParent.set(left, top, right, bottom); 2809 info.setVisibleToUser(isVisibleToUser(boundsInParent)); 2810 info.setBoundsInParent(boundsInParent); 2811 Rect boundsInScreen = boundsInParent; 2812 int[] locationOnScreen = mTempArray; 2813 getLocationOnScreen(locationOnScreen); 2814 boundsInScreen.offset(locationOnScreen[0], locationOnScreen[1]); 2815 info.setBoundsInScreen(boundsInScreen); 2816 2817 if (mAccessibilityFocusedView != virtualViewId) { 2818 info.addAction(AccessibilityNodeInfo.ACTION_ACCESSIBILITY_FOCUS); 2819 } 2820 if (mAccessibilityFocusedView == virtualViewId) { 2821 info.addAction(AccessibilityNodeInfo.ACTION_CLEAR_ACCESSIBILITY_FOCUS); 2822 } 2823 if (NumberPicker.this.isEnabled()) { 2824 info.addAction(AccessibilityNodeInfo.ACTION_CLICK); 2825 } 2826 2827 return info; 2828 } 2829 createAccessibilityNodeInfoForNumberPicker(int left, int top, int right, int bottom)2830 private AccessibilityNodeInfo createAccessibilityNodeInfoForNumberPicker(int left, int top, 2831 int right, int bottom) { 2832 AccessibilityNodeInfo info = AccessibilityNodeInfo.obtain(); 2833 info.setClassName(NumberPicker.class.getName()); 2834 info.setPackageName(mContext.getPackageName()); 2835 info.setSource(NumberPicker.this); 2836 2837 if (hasVirtualDecrementButton()) { 2838 info.addChild(NumberPicker.this, VIRTUAL_VIEW_ID_DECREMENT); 2839 } 2840 info.addChild(NumberPicker.this, VIRTUAL_VIEW_ID_INPUT); 2841 if (hasVirtualIncrementButton()) { 2842 info.addChild(NumberPicker.this, VIRTUAL_VIEW_ID_INCREMENT); 2843 } 2844 2845 info.setParent((View) getParentForAccessibility()); 2846 info.setEnabled(NumberPicker.this.isEnabled()); 2847 info.setScrollable(true); 2848 info.setAccessibilityFocused(mAccessibilityFocusedView == View.NO_ID); 2849 2850 final float applicationScale = 2851 getContext().getResources().getCompatibilityInfo().applicationScale; 2852 2853 Rect boundsInParent = mTempRect; 2854 boundsInParent.set(left, top, right, bottom); 2855 boundsInParent.scale(applicationScale); 2856 info.setBoundsInParent(boundsInParent); 2857 2858 info.setVisibleToUser(isVisibleToUser()); 2859 2860 Rect boundsInScreen = boundsInParent; 2861 int[] locationOnScreen = mTempArray; 2862 getLocationOnScreen(locationOnScreen); 2863 boundsInScreen.offset(locationOnScreen[0], locationOnScreen[1]); 2864 boundsInScreen.scale(applicationScale); 2865 info.setBoundsInScreen(boundsInScreen); 2866 2867 if (mAccessibilityFocusedView != View.NO_ID) { 2868 info.addAction(AccessibilityNodeInfo.ACTION_ACCESSIBILITY_FOCUS); 2869 } 2870 if (mAccessibilityFocusedView == View.NO_ID) { 2871 info.addAction(AccessibilityNodeInfo.ACTION_CLEAR_ACCESSIBILITY_FOCUS); 2872 } 2873 if (NumberPicker.this.isEnabled()) { 2874 if (getWrapSelectorWheel() || getValue() < getMaxValue()) { 2875 info.addAction(AccessibilityNodeInfo.AccessibilityAction.ACTION_SCROLL_FORWARD); 2876 info.addAction(AccessibilityNodeInfo.AccessibilityAction.ACTION_SCROLL_DOWN); 2877 } 2878 if (getWrapSelectorWheel() || getValue() > getMinValue()) { 2879 info.addAction( 2880 AccessibilityNodeInfo.AccessibilityAction.ACTION_SCROLL_BACKWARD); 2881 info.addAction(AccessibilityNodeInfo.AccessibilityAction.ACTION_SCROLL_UP); 2882 } 2883 } 2884 2885 return info; 2886 } 2887 hasVirtualDecrementButton()2888 private boolean hasVirtualDecrementButton() { 2889 return getWrapSelectorWheel() || getValue() > getMinValue(); 2890 } 2891 hasVirtualIncrementButton()2892 private boolean hasVirtualIncrementButton() { 2893 return getWrapSelectorWheel() || getValue() < getMaxValue(); 2894 } 2895 getVirtualDecrementButtonText()2896 private String getVirtualDecrementButtonText() { 2897 int value = mValue - 1; 2898 if (mWrapSelectorWheel) { 2899 value = getWrappedSelectorIndex(value); 2900 } 2901 if (value >= mMinValue) { 2902 return (mDisplayedValues == null) ? formatNumber(value) 2903 : mDisplayedValues[value - mMinValue]; 2904 } 2905 return null; 2906 } 2907 getVirtualIncrementButtonText()2908 private String getVirtualIncrementButtonText() { 2909 int value = mValue + 1; 2910 if (mWrapSelectorWheel) { 2911 value = getWrappedSelectorIndex(value); 2912 } 2913 if (value <= mMaxValue) { 2914 return (mDisplayedValues == null) ? formatNumber(value) 2915 : mDisplayedValues[value - mMinValue]; 2916 } 2917 return null; 2918 } 2919 } 2920 formatNumberWithLocale(int value)2921 static private String formatNumberWithLocale(int value) { 2922 return String.format(Locale.getDefault(), "%d", value); 2923 } 2924 } 2925