summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
author Alan Viverette <alanv@google.com> 2013-06-24 17:23:19 +0000
committer Android (Google) Code Review <android-gerrit@google.com> 2013-06-24 17:23:19 +0000
commitdb759c3f022d1416fe885e1a0ef05a4ca208f254 (patch)
tree236a1b225f1b64b9a933526794639e578419700a
parentedaf0794bbeca1adfb825a90a9f42a01bae3aa37 (diff)
parent0ebe81e8b1f2b9db8d41b72a6dae8d6848b51cc5 (diff)
Merge "Implement FastScroller as an animated overlay."
-rw-r--r--core/java/android/widget/AbsListView.java49
-rw-r--r--core/java/android/widget/FastScroller.java1565
-rw-r--r--core/res/res/values/dimens.xml2
-rwxr-xr-xcore/res/res/values/symbols.xml1
4 files changed, 1016 insertions, 601 deletions
diff --git a/core/java/android/widget/AbsListView.java b/core/java/android/widget/AbsListView.java
index 1991af1d65a9..bb1f95430037 100644
--- a/core/java/android/widget/AbsListView.java
+++ b/core/java/android/widget/AbsListView.java
@@ -1280,12 +1280,12 @@ public abstract class AbsListView extends AdapterView<ListAdapter> implements Te
}
/**
- * If fast scroll is visible, then don't draw the vertical scrollbar.
+ * If fast scroll is enabled, then don't draw the vertical scrollbar.
* @hide
*/
@Override
protected boolean isVerticalScrollBarHidden() {
- return mFastScroller != null && mFastScroller.isVisible();
+ return mFastScrollEnabled;
}
/**
@@ -1337,7 +1337,7 @@ public abstract class AbsListView extends AdapterView<ListAdapter> implements Te
*/
void invokeOnItemScrollListener() {
if (mFastScroller != null) {
- mFastScroller.onScroll(this, mFirstPosition, getChildCount(), mItemCount);
+ mFastScroller.onScroll(mFirstPosition, getChildCount(), mItemCount);
}
if (mOnScrollListener != null) {
mOnScrollListener.onScroll(this, mFirstPosition, getChildCount(), mItemCount);
@@ -2009,7 +2009,7 @@ public abstract class AbsListView extends AdapterView<ListAdapter> implements Te
}
mRecycler.markChildrenDirty();
}
-
+
if (mFastScroller != null && mItemCount != mOldItemCount) {
mFastScroller.onItemCountChanged(mOldItemCount, mItemCount);
}
@@ -3752,18 +3752,6 @@ public abstract class AbsListView extends AdapterView<ListAdapter> implements Te
canvas.restoreToCount(restoreCount);
}
}
- if (mFastScroller != null) {
- final int scrollY = mScrollY;
- if (scrollY != 0) {
- // Pin to the top/bottom during overscroll
- int restoreCount = canvas.save();
- canvas.translate(0, scrollY);
- mFastScroller.draw(canvas);
- canvas.restoreToCount(restoreCount);
- } else {
- mFastScroller.draw(canvas);
- }
- }
}
/**
@@ -3820,11 +3808,8 @@ public abstract class AbsListView extends AdapterView<ListAdapter> implements Te
return false;
}
- if (mFastScroller != null) {
- boolean intercepted = mFastScroller.onInterceptTouchEvent(ev);
- if (intercepted) {
- return true;
- }
+ if (mFastScroller != null && mFastScroller.onInterceptTouchEvent(ev)) {
+ return true;
}
switch (action & MotionEvent.ACTION_MASK) {
@@ -5672,78 +5657,96 @@ public abstract class AbsListView extends AdapterView<ListAdapter> implements Te
return mDefInputConnection.sendKeyEvent(event);
}
+ @Override
public CharSequence getTextBeforeCursor(int n, int flags) {
if (mTarget == null) return "";
return mTarget.getTextBeforeCursor(n, flags);
}
+ @Override
public CharSequence getTextAfterCursor(int n, int flags) {
if (mTarget == null) return "";
return mTarget.getTextAfterCursor(n, flags);
}
+ @Override
public CharSequence getSelectedText(int flags) {
if (mTarget == null) return "";
return mTarget.getSelectedText(flags);
}
+ @Override
public int getCursorCapsMode(int reqModes) {
if (mTarget == null) return InputType.TYPE_TEXT_FLAG_CAP_SENTENCES;
return mTarget.getCursorCapsMode(reqModes);
}
+ @Override
public ExtractedText getExtractedText(ExtractedTextRequest request, int flags) {
return getTarget().getExtractedText(request, flags);
}
+ @Override
public boolean deleteSurroundingText(int beforeLength, int afterLength) {
return getTarget().deleteSurroundingText(beforeLength, afterLength);
}
+ @Override
public boolean setComposingText(CharSequence text, int newCursorPosition) {
return getTarget().setComposingText(text, newCursorPosition);
}
+ @Override
public boolean setComposingRegion(int start, int end) {
return getTarget().setComposingRegion(start, end);
}
+ @Override
public boolean finishComposingText() {
return mTarget == null || mTarget.finishComposingText();
}
+ @Override
public boolean commitText(CharSequence text, int newCursorPosition) {
return getTarget().commitText(text, newCursorPosition);
}
+ @Override
public boolean commitCompletion(CompletionInfo text) {
return getTarget().commitCompletion(text);
}
+ @Override
public boolean commitCorrection(CorrectionInfo correctionInfo) {
return getTarget().commitCorrection(correctionInfo);
}
+ @Override
public boolean setSelection(int start, int end) {
return getTarget().setSelection(start, end);
}
+ @Override
public boolean performContextMenuAction(int id) {
return getTarget().performContextMenuAction(id);
}
+ @Override
public boolean beginBatchEdit() {
return getTarget().beginBatchEdit();
}
+ @Override
public boolean endBatchEdit() {
return getTarget().endBatchEdit();
}
+ @Override
public boolean clearMetaKeyStates(int states) {
return getTarget().clearMetaKeyStates(states);
}
+ @Override
public boolean performPrivateCommand(String action, Bundle data) {
return getTarget().performPrivateCommand(action, data);
}
@@ -6037,9 +6040,9 @@ public abstract class AbsListView extends AdapterView<ListAdapter> implements Te
/**
* Sets up the onClickHandler to be used by the RemoteViewsAdapter when inflating RemoteViews
- *
+ *
* @param handler The OnClickHandler to use when inflating RemoteViews.
- *
+ *
* @hide
*/
public void setRemoteViewsOnClickHandler(OnClickHandler handler) {
diff --git a/core/java/android/widget/FastScroller.java b/core/java/android/widget/FastScroller.java
index aa33384e6fcb..62e157866e81 100644
--- a/core/java/android/widget/FastScroller.java
+++ b/core/java/android/widget/FastScroller.java
@@ -16,49 +16,66 @@
package android.widget;
+import android.animation.Animator;
+import android.animation.Animator.AnimatorListener;
+import android.animation.AnimatorListenerAdapter;
+import android.animation.AnimatorSet;
+import android.animation.ObjectAnimator;
+import android.animation.PropertyValuesHolder;
import android.content.Context;
import android.content.res.ColorStateList;
import android.content.res.Resources;
import android.content.res.TypedArray;
-import android.graphics.Canvas;
-import android.graphics.Paint;
import android.graphics.Rect;
-import android.graphics.RectF;
import android.graphics.drawable.Drawable;
-import android.graphics.drawable.NinePatchDrawable;
-import android.os.Handler;
-import android.os.SystemClock;
+import android.os.Build;
+import android.text.TextUtils.TruncateAt;
+import android.util.IntProperty;
+import android.util.MathUtils;
+import android.util.Property;
+import android.view.Gravity;
import android.view.MotionEvent;
import android.view.View;
+import android.view.View.MeasureSpec;
import android.view.ViewConfiguration;
+import android.view.ViewGroup.LayoutParams;
+import android.view.ViewGroupOverlay;
import android.widget.AbsListView.OnScrollListener;
+import com.android.internal.R;
+
/**
* Helper class for AbsListView to draw and control the Fast Scroll thumb
*/
class FastScroller {
- private static final String TAG = "FastScroller";
+ /** Duration of fade-out animation. */
+ private static final int DURATION_FADE_OUT = 300;
+
+ /** Duration of fade-in animation. */
+ private static final int DURATION_FADE_IN = 150;
+
+ /** Duration of transition cross-fade animation. */
+ private static final int DURATION_CROSS_FADE = 50;
+
+ /** Duration of transition resize animation. */
+ private static final int DURATION_RESIZE = 100;
+
+ /** Inactivity timeout before fading controls. */
+ private static final long FADE_TIMEOUT = 1500;
- // Minimum number of pages to justify showing a fast scroll thumb
- private static int MIN_PAGES = 4;
- // Scroll thumb not showing
+ /** Minimum number of pages to justify showing a fast scroll thumb. */
+ private static final int MIN_PAGES = 4;
+
+ /** Scroll thumb and preview not showing. */
private static final int STATE_NONE = 0;
- // Not implemented yet - fade-in transition
- @SuppressWarnings("unused")
- private static final int STATE_ENTER = 1;
- // Scroll thumb visible and moving along with the scrollbar
- private static final int STATE_VISIBLE = 2;
- // Scroll thumb being dragged by user
- private static final int STATE_DRAGGING = 3;
- // Scroll thumb fading out due to inactivity timeout
- private static final int STATE_EXIT = 4;
-
- private static final int[] PRESSED_STATES = new int[] {
- android.R.attr.state_pressed
- };
- private static final int[] DEFAULT_STATES = new int[0];
+ /** Scroll thumb visible and moving along with the scrollbar. */
+ private static final int STATE_VISIBLE = 1;
+
+ /** Scroll thumb and preview being dragged by user. */
+ private static final int STATE_DRAGGING = 2;
+ /** Styleable attributes. */
private static final int[] ATTRS = new int[] {
android.R.attr.fastScrollTextColor,
android.R.attr.fastScrollThumbDrawable,
@@ -68,6 +85,7 @@ class FastScroller {
android.R.attr.fastScrollOverlayPosition
};
+ // Styleable attribute indices.
private static final int TEXT_COLOR = 0;
private static final int THUMB_DRAWABLE = 1;
private static final int TRACK_DRAWABLE = 2;
@@ -75,113 +93,247 @@ class FastScroller {
private static final int PREVIEW_BACKGROUND_RIGHT = 4;
private static final int OVERLAY_POSITION = 5;
+ // Positions for preview image and text.
private static final int OVERLAY_FLOATING = 0;
private static final int OVERLAY_AT_THUMB = 1;
- private Drawable mThumbDrawable;
- private Drawable mOverlayDrawable;
- private Drawable mTrackDrawable;
+ // Indices for mPreviewResId.
+ private static final int PREVIEW_LEFT = 0;
+ private static final int PREVIEW_RIGHT = 1;
+
+ /** Delay before considering a tap in the thumb area to be a drag. */
+ private static final long TAP_TIMEOUT = ViewConfiguration.getTapTimeout();
+
+ private final Rect mTempBounds = new Rect();
+ private final Rect mTempMargins = new Rect();
- private Drawable mOverlayDrawableLeft;
- private Drawable mOverlayDrawableRight;
+ private final AbsListView mList;
+ private final ViewGroupOverlay mOverlay;
+ private final TextView mPrimaryText;
+ private final TextView mSecondaryText;
+ private final ImageView mThumbImage;
+ private final ImageView mTrackImage;
+ private final ImageView mPreviewImage;
- int mThumbH;
- int mThumbW;
- int mThumbY;
+ /**
+ * Preview image resource IDs for left- and right-aligned layouts. See
+ * {@link #PREVIEW_LEFT} and {@link #PREVIEW_RIGHT}.
+ */
+ private final int[] mPreviewResId = new int[2];
- private RectF mOverlayPos;
- private int mOverlaySize;
- private int mOverlayPadding;
+ /**
+ * Padding in pixels around the preview text. Applied as layout margins to
+ * the preview text and padding to the preview image.
+ */
+ private final int mPreviewPadding;
- AbsListView mList;
- boolean mScrollCompleted;
- private int mVisibleItem;
- private Paint mPaint;
- private int mListOffset;
+ /** Whether there is a track image to display. */
+ private final boolean mHasTrackImage;
+
+ /** Set containing decoration transition animations. */
+ private AnimatorSet mDecorAnimation;
+
+ /** Set containing preview text transition animations. */
+ private AnimatorSet mPreviewAnimation;
+
+ /** Whether the primary text is showing. */
+ private boolean mShowingPrimary;
+
+ /** Whether we're waiting for completion of scrollTo(). */
+ private boolean mScrollCompleted;
+
+ /** The position of the first visible item in the list. */
+ private int mFirstVisibleItem;
+
+ /** The number of headers at the top of the view. */
+ private int mHeaderCount;
+
+ /** The number of items in the list. */
private int mItemCount = -1;
+
+ /** The index of the current section. */
+ private int mCurrentSection = -1;
+
+ /** Whether the list is long enough to need a fast scroller. */
private boolean mLongList;
- private Object [] mSections;
- private String mSectionText;
- private boolean mDrawOverlay;
- private ScrollFade mScrollFade;
+ private Object[] mSections;
+ /**
+ * Current decoration state, one of:
+ * <ul>
+ * <li>{@link #STATE_NONE}, nothing visible
+ * <li>{@link #STATE_VISIBLE}, showing track and thumb
+ * <li>{@link #STATE_DRAGGING}, visible and showing preview
+ * </ul>
+ */
private int mState;
- private Handler mHandler = new Handler();
-
- BaseAdapter mListAdapter;
+ private BaseAdapter mListAdapter;
private SectionIndexer mSectionIndexer;
- private boolean mChangedBounds;
-
- private int mPosition;
+ /** Whether decorations should be laid out from right to left. */
+ private boolean mLayoutFromRight;
+ /** Whether the scrollbar and decorations should always be shown. */
private boolean mAlwaysShow;
+ /**
+ * Position for the preview image and text. One of:
+ * <ul>
+ * <li>{@link #OVERLAY_AT_THUMB}
+ * <li>{@link #OVERLAY_FLOATING}
+ * </ul>
+ */
private int mOverlayPosition;
+ /** Whether to precisely match the thumb position to the list. */
private boolean mMatchDragPosition;
- float mInitialTouchY;
- boolean mPendingDrag;
+ private float mInitialTouchY;
+ private boolean mHasPendingDrag;
private int mScaledTouchSlop;
- private static final int FADE_TIMEOUT = 1500;
- private static final int PENDING_DRAG_DELAY = 180;
-
- private final Rect mTmpRect = new Rect();
-
private final Runnable mDeferStartDrag = new Runnable() {
@Override
public void run() {
if (mList.mIsAttached) {
beginDrag();
- final int viewHeight = mList.getHeight();
- // Jitter
- int newThumbY = (int) mInitialTouchY - mThumbH + 10;
- if (newThumbY < 0) {
- newThumbY = 0;
- } else if (newThumbY + mThumbH > viewHeight) {
- newThumbY = viewHeight - mThumbH;
- }
- mThumbY = newThumbY;
- scrollTo((float) mThumbY / (viewHeight - mThumbH));
+ final float pos = getPosFromMotionEvent(mInitialTouchY);
+ scrollTo(pos);
}
- mPendingDrag = false;
+ mHasPendingDrag = false;
+ }
+ };
+
+ /**
+ * Used to delay hiding fast scroll decorations.
+ */
+ private final Runnable mDeferHide = new Runnable() {
+ @Override
+ public void run() {
+ setState(STATE_NONE);
+ }
+ };
+
+ /**
+ * Used to effect a transition from primary to secondary text.
+ */
+ private final AnimatorListener mSwitchPrimaryListener = new AnimatorListenerAdapter() {
+ @Override
+ public void onAnimationEnd(Animator animation) {
+ mShowingPrimary = !mShowingPrimary;
}
};
public FastScroller(Context context, AbsListView listView) {
mList = listView;
- init(context);
+ mOverlay = listView.getOverlay();
+
+ mScaledTouchSlop = ViewConfiguration.get(context).getScaledTouchSlop();
+
+ final Resources res = context.getResources();
+ final TypedArray ta = context.getTheme().obtainStyledAttributes(ATTRS);
+
+ mTrackImage = new ImageView(context);
+
+ // Add track to overlay if it has an image.
+ final int trackResId = ta.getResourceId(TRACK_DRAWABLE, 0);
+ if (trackResId != 0) {
+ mHasTrackImage = true;
+ mTrackImage.setBackgroundResource(trackResId);
+ mOverlay.add(mTrackImage);
+ } else {
+ mHasTrackImage = false;
+ }
+
+ mThumbImage = new ImageView(context);
+
+ // Add thumb to overlay if it has an image.
+ final Drawable thumbDrawable = ta.getDrawable(THUMB_DRAWABLE);
+ if (thumbDrawable != null) {
+ mThumbImage.setImageDrawable(thumbDrawable);
+ mOverlay.add(mThumbImage);
+ }
+
+ // If necessary, apply minimum thumb width and height.
+ if (thumbDrawable.getIntrinsicWidth() <= 0 || thumbDrawable.getIntrinsicHeight() <= 0) {
+ mThumbImage.setMinimumWidth(res.getDimensionPixelSize(R.dimen.fastscroll_thumb_width));
+ mThumbImage.setMinimumHeight(
+ res.getDimensionPixelSize(R.dimen.fastscroll_thumb_height));
+ }
+
+ final int previewSize = res.getDimensionPixelSize(R.dimen.fastscroll_overlay_size);
+ mPreviewImage = new ImageView(context);
+ mPreviewImage.setMinimumWidth(previewSize);
+ mPreviewImage.setMinimumHeight(previewSize);
+ mPreviewImage.setAlpha(0f);
+ mOverlay.add(mPreviewImage);
+
+ mPreviewPadding = res.getDimensionPixelSize(R.dimen.fastscroll_overlay_padding);
+
+ mPrimaryText = createPreviewTextView(context, ta);
+ mOverlay.add(mPrimaryText);
+ mSecondaryText = createPreviewTextView(context, ta);
+ mOverlay.add(mSecondaryText);
+
+ mPreviewResId[PREVIEW_LEFT] = ta.getResourceId(PREVIEW_BACKGROUND_LEFT, 0);
+ mPreviewResId[PREVIEW_RIGHT] = ta.getResourceId(PREVIEW_BACKGROUND_RIGHT, 0);
+ mOverlayPosition = ta.getInt(OVERLAY_POSITION, OVERLAY_FLOATING);
+ ta.recycle();
+
+ mScrollCompleted = true;
+ mState = STATE_VISIBLE;
+ mMatchDragPosition =
+ context.getApplicationInfo().targetSdkVersion >= Build.VERSION_CODES.HONEYCOMB;
+
+ getSectionsFromIndexer();
+ refreshDrawablePressedState();
+ setScrollbarPosition(mList.getVerticalScrollbarPosition());
+
+ mList.postDelayed(mDeferHide, FADE_TIMEOUT);
}
+ /**
+ * @param alwaysShow Whether the fast scroll thumb should always be shown
+ */
public void setAlwaysShow(boolean alwaysShow) {
mAlwaysShow = alwaysShow;
+
if (alwaysShow) {
- mHandler.removeCallbacks(mScrollFade);
setState(STATE_VISIBLE);
} else if (mState == STATE_VISIBLE) {
- mHandler.postDelayed(mScrollFade, FADE_TIMEOUT);
+ mList.postDelayed(mDeferHide, FADE_TIMEOUT);
}
}
+ /**
+ * @return Whether the fast scroll thumb will always be shown
+ * @see #setAlwaysShow(boolean)
+ */
public boolean isAlwaysShowEnabled() {
return mAlwaysShow;
}
- private void refreshDrawableState() {
- int[] state = mState == STATE_DRAGGING ? PRESSED_STATES : DEFAULT_STATES;
+ /**
+ * Immediately transitions the fast scroller decorations to a hidden state.
+ */
+ public void stop() {
+ setState(STATE_NONE);
+ }
- if (mThumbDrawable != null && mThumbDrawable.isStateful()) {
- mThumbDrawable.setState(state);
- }
- if (mTrackDrawable != null && mTrackDrawable.isStateful()) {
- mTrackDrawable.setState(state);
+ /**
+ * @return Whether the fast scroll thumb should be shown.
+ */
+ public boolean shouldShow() {
+ // Don't show if the list is as tall as or shorter than the thumbnail.
+ if (mList.getHeight() <= mThumbImage.getHeight()) {
+ return false;
}
+
+ return true;
}
public void setScrollbarPosition(int position) {
@@ -189,374 +341,403 @@ class FastScroller {
position = mList.isLayoutRtl() ?
View.SCROLLBAR_POSITION_LEFT : View.SCROLLBAR_POSITION_RIGHT;
}
- mPosition = position;
- switch (position) {
- default:
- case View.SCROLLBAR_POSITION_RIGHT:
- mOverlayDrawable = mOverlayDrawableRight;
- break;
- case View.SCROLLBAR_POSITION_LEFT:
- mOverlayDrawable = mOverlayDrawableLeft;
- break;
+
+ mLayoutFromRight = position != View.SCROLLBAR_POSITION_LEFT;
+
+ final int previewResId = mPreviewResId[mLayoutFromRight ? PREVIEW_RIGHT : PREVIEW_LEFT];
+ mPreviewImage.setBackgroundResource(previewResId);
+
+ // Add extra padding for text.
+ final Drawable background = mPreviewImage.getBackground();
+ if (background != null) {
+ final Rect padding = mTempBounds;
+ background.getPadding(padding);
+ padding.offset(mPreviewPadding, mPreviewPadding);
+ mPreviewImage.setPadding(padding.left, padding.top, padding.right, padding.bottom);
}
+
+ updateLayout();
}
public int getWidth() {
- return mThumbW;
+ return mThumbImage.getWidth();
}
- public void setState(int state) {
- switch (state) {
- case STATE_NONE:
- mHandler.removeCallbacks(mScrollFade);
- mList.invalidate();
- break;
- case STATE_VISIBLE:
- if (mState != STATE_VISIBLE) { // Optimization
- resetThumbPos();
- }
- // Fall through
- case STATE_DRAGGING:
- mHandler.removeCallbacks(mScrollFade);
- break;
- case STATE_EXIT:
- final int viewWidth = mList.getWidth();
- final int top = mThumbY;
- final int bottom = mThumbY + mThumbH;
- final int left;
- final int right;
- switch (mList.getLayoutDirection()) {
- case View.LAYOUT_DIRECTION_RTL:
- left = 0;
- right = mThumbW;
- break;
- case View.LAYOUT_DIRECTION_LTR:
- default:
- left = viewWidth - mThumbW;
- right = viewWidth;
- }
- mList.invalidate(left, top, right, bottom);
- break;
+ public void onSizeChanged(int w, int h, int oldw, int oldh) {
+ updateLayout();
+ }
+
+ public void onItemCountChanged(int oldTotalItemCount, int totalItemCount) {
+ final int visibleItemCount = mList.getChildCount();
+ final boolean hasMoreItems = totalItemCount - visibleItemCount > 0;
+ if (hasMoreItems && mState != STATE_DRAGGING) {
+ final int firstVisibleItem = mList.getFirstVisiblePosition();
+ setThumbPos(getPosFromItemCount(firstVisibleItem, visibleItemCount, totalItemCount));
}
- mState = state;
- refreshDrawableState();
}
- public int getState() {
- return mState;
+ /**
+ * Creates a view into which preview text can be placed.
+ */
+ private TextView createPreviewTextView(Context context, TypedArray ta) {
+ final LayoutParams params = new LayoutParams(
+ LayoutParams.WRAP_CONTENT, LayoutParams.WRAP_CONTENT);
+ final Resources res = context.getResources();
+ final int minSize = res.getDimensionPixelSize(R.dimen.fastscroll_overlay_size);
+ final ColorStateList textColor = ta.getColorStateList(TEXT_COLOR);
+ final float textSize = res.getDimension(R.dimen.fastscroll_overlay_text_size);
+ final TextView textView = new TextView(context);
+ textView.setLayoutParams(params);
+ textView.setTextColor(textColor);
+ textView.setTextSize(textSize);
+ textView.setSingleLine(true);
+ textView.setEllipsize(TruncateAt.MIDDLE);
+ textView.setGravity(Gravity.CENTER);
+ textView.setAlpha(0f);
+
+ // Manually propagate inherited layout direction.
+ textView.setLayoutDirection(mList.getLayoutDirection());
+
+ return textView;
}
- private void resetThumbPos() {
- final int viewWidth = mList.getWidth();
- // Bounds are always top right. Y coordinate get's translated during draw
- switch (mPosition) {
- case View.SCROLLBAR_POSITION_RIGHT:
- mThumbDrawable.setBounds(viewWidth - mThumbW, 0, viewWidth, mThumbH);
- break;
- case View.SCROLLBAR_POSITION_LEFT:
- mThumbDrawable.setBounds(0, 0, mThumbW, mThumbH);
- break;
+ /**
+ * Measures and layouts the scrollbar and decorations.
+ */
+ private void updateLayout() {
+ layoutThumb();
+ layoutTrack();
+
+ final Rect bounds = mTempBounds;
+ measurePreview(mPrimaryText, bounds);
+ applyLayout(mPrimaryText, bounds);
+ measurePreview(mSecondaryText, bounds);
+ applyLayout(mSecondaryText, bounds);
+
+ if (mPreviewImage != null) {
+ // Apply preview image padding.
+ bounds.left -= mPreviewImage.getPaddingLeft();
+ bounds.top -= mPreviewImage.getPaddingTop();
+ bounds.right += mPreviewImage.getPaddingRight();
+ bounds.bottom += mPreviewImage.getPaddingBottom();
+ applyLayout(mPreviewImage, bounds);
}
- mThumbDrawable.setAlpha(ScrollFade.ALPHA_MAX);
}
- private void useThumbDrawable(Context context, Drawable drawable) {
- mThumbDrawable = drawable;
- if (drawable instanceof NinePatchDrawable) {
- mThumbW = context.getResources().getDimensionPixelSize(
- com.android.internal.R.dimen.fastscroll_thumb_width);
- mThumbH = context.getResources().getDimensionPixelSize(
- com.android.internal.R.dimen.fastscroll_thumb_height);
+ /**
+ * Layouts a view within the specified bounds and pins the pivot point to
+ * the appropriate edge.
+ *
+ * @param view The view to layout.
+ * @param bounds Bounds at which to layout the view.
+ */
+ private void applyLayout(View view, Rect bounds) {
+ view.layout(bounds.left, bounds.top, bounds.right, bounds.bottom);
+ view.setPivotX(mLayoutFromRight ? bounds.right - bounds.left : 0);
+ }
+
+ /**
+ * Measures the preview text bounds, taking preview image padding into
+ * account. This method should only be called after {@link #layoutThumb()}
+ * and {@link #layoutTrack()} have both been called at least once.
+ *
+ * @param v The preview text view to measure.
+ * @param out Rectangle into which measured bounds are placed.
+ */
+ private void measurePreview(View v, Rect out) {
+ // Apply the preview image's padding as layout margins.
+ final Rect margins = mTempMargins;
+ margins.left = mPreviewImage.getPaddingLeft();
+ margins.top = mPreviewImage.getPaddingTop();
+ margins.right = mPreviewImage.getPaddingRight();
+ margins.bottom = mPreviewImage.getPaddingBottom();
+
+ if (mOverlayPosition == OVERLAY_AT_THUMB) {
+ measureViewToSide(v, mThumbImage, margins, out);
} else {
- mThumbW = drawable.getIntrinsicWidth();
- mThumbH = drawable.getIntrinsicHeight();
+ measureFloating(v, margins, out);
}
- mChangedBounds = true;
}
- private void init(Context context) {
- // Get both the scrollbar states drawables
- final TypedArray ta = context.getTheme().obtainStyledAttributes(ATTRS);
- useThumbDrawable(context, ta.getDrawable(THUMB_DRAWABLE));
- mTrackDrawable = ta.getDrawable(TRACK_DRAWABLE);
-
- mOverlayDrawableLeft = ta.getDrawable(PREVIEW_BACKGROUND_LEFT);
- mOverlayDrawableRight = ta.getDrawable(PREVIEW_BACKGROUND_RIGHT);
- mOverlayPosition = ta.getInt(OVERLAY_POSITION, OVERLAY_FLOATING);
-
- mScrollCompleted = true;
-
- getSectionsFromIndexer();
+ /**
+ * Measures the bounds for a view that should be laid out against the edge
+ * of an adjacent view. If no adjacent view is provided, lays out against
+ * the list edge.
+ *
+ * @param view The view to measure for layout.
+ * @param adjacent (Optional) The adjacent view, may be null to align to the
+ * list edge.
+ * @param margins Layout margins to apply to the view.
+ * @param out Rectangle into which measured bounds are placed.
+ */
+ private void measureViewToSide(View view, View adjacent, Rect margins, Rect out) {
+ final int marginLeft;
+ final int marginTop;
+ final int marginRight;
+ if (margins == null) {
+ marginLeft = 0;
+ marginTop = 0;
+ marginRight = 0;
+ } else {
+ marginLeft = margins.left;
+ marginTop = margins.top;
+ marginRight = margins.right;
+ }
- final Resources res = context.getResources();
- mOverlaySize = res.getDimensionPixelSize(
- com.android.internal.R.dimen.fastscroll_overlay_size);
- mOverlayPadding = res.getDimensionPixelSize(
- com.android.internal.R.dimen.fastscroll_overlay_padding);
- mOverlayPos = new RectF();
- mScrollFade = new ScrollFade();
- mPaint = new Paint();
- mPaint.setAntiAlias(true);
- mPaint.setTextAlign(Paint.Align.CENTER);
- mPaint.setTextSize(mOverlaySize / 2);
-
- ColorStateList textColor = ta.getColorStateList(TEXT_COLOR);
- int textColorNormal = textColor.getDefaultColor();
- mPaint.setColor(textColorNormal);
- mPaint.setStyle(Paint.Style.FILL_AND_STROKE);
-
- // to show mOverlayDrawable properly
- if (mList.getWidth() > 0 && mList.getHeight() > 0) {
- onSizeChanged(mList.getWidth(), mList.getHeight(), 0, 0);
- }
-
- mState = STATE_NONE;
- refreshDrawableState();
+ final int listWidth = mList.getWidth();
+ final int maxWidth;
+ if (adjacent == null) {
+ maxWidth = listWidth;
+ } else if (mLayoutFromRight) {
+ maxWidth = adjacent.getLeft();
+ } else {
+ maxWidth = listWidth - adjacent.getRight();
+ }
- ta.recycle();
+ final int adjMaxWidth = maxWidth - marginLeft - marginRight;
+ final int widthMeasureSpec = MeasureSpec.makeMeasureSpec(adjMaxWidth, MeasureSpec.AT_MOST);
+ final int heightMeasureSpec = MeasureSpec.makeMeasureSpec(0, MeasureSpec.UNSPECIFIED);
+ view.measure(widthMeasureSpec, heightMeasureSpec);
+
+ // Align to the left or right.
+ final int width = view.getMeasuredWidth();
+ final int left;
+ final int right;
+ if (mLayoutFromRight) {
+ right = (adjacent == null ? listWidth : adjacent.getLeft()) - marginRight;
+ left = right - width;
+ } else {
+ left = (adjacent == null ? 0 : adjacent.getRight()) + marginLeft;
+ right = left + width;
+ }
- mScaledTouchSlop = ViewConfiguration.get(context).getScaledTouchSlop();
+ // Don't adjust the vertical position.
+ final int top = marginTop;
+ final int bottom = top + view.getMeasuredHeight();
+ out.set(left, top, right, bottom);
+ }
- mMatchDragPosition = context.getApplicationInfo().targetSdkVersion >=
- android.os.Build.VERSION_CODES.HONEYCOMB;
+ private void measureFloating(View preview, Rect margins, Rect out) {
+ final int marginLeft;
+ final int marginTop;
+ final int marginRight;
+ if (margins == null) {
+ marginLeft = 0;
+ marginTop = 0;
+ marginRight = 0;
+ } else {
+ marginLeft = margins.left;
+ marginTop = margins.top;
+ marginRight = margins.right;
+ }
- setScrollbarPosition(mList.getVerticalScrollbarPosition());
+ final View list = mList;
+ final int listWidth = list.getWidth();
+ final int adjMaxWidth = listWidth - marginLeft - marginRight;
+ final int widthMeasureSpec = MeasureSpec.makeMeasureSpec(adjMaxWidth, MeasureSpec.AT_MOST);
+ final int heightMeasureSpec = MeasureSpec.makeMeasureSpec(0, MeasureSpec.UNSPECIFIED);
+ preview.measure(widthMeasureSpec, heightMeasureSpec);
+
+ // Align at the vertical center, 10% from the top.
+ final int width = preview.getMeasuredWidth();
+ final int top = list.getHeight() / 10 + marginTop;
+ final int bottom = top + preview.getMeasuredHeight();
+ final int left = (listWidth - width) / 2;
+ final int right = left + width;
+ out.set(left, top, right, bottom);
}
- void stop() {
- setState(STATE_NONE);
+ /**
+ * Lays out the thumb according to the current scrollbar position.
+ */
+ private void layoutThumb() {
+ final Rect bounds = mTempBounds;
+ measureViewToSide(mThumbImage, null, null, bounds);
+ applyLayout(mThumbImage, bounds);
}
- boolean isVisible() {
- return !(mState == STATE_NONE);
+ /**
+ * Lays out the track centered on the thumb, if available, or against the
+ * edge if no thumb is available. Must be called after {@link #layoutThumb}.
+ */
+ private void layoutTrack() {
+ final View track = mTrackImage;
+ final View thumb = mThumbImage;
+ final View list = mList;
+ final int listWidth = list.getWidth();
+ final int widthMeasureSpec = MeasureSpec.makeMeasureSpec(listWidth, MeasureSpec.AT_MOST);
+ final int heightMeasureSpec = MeasureSpec.makeMeasureSpec(0, MeasureSpec.UNSPECIFIED);
+ track.measure(widthMeasureSpec, heightMeasureSpec);
+
+ final int trackWidth = track.getMeasuredWidth();
+ final int thumbHalfHeight = thumb == null ? 0 : thumb.getHeight() / 2;
+ final int left = thumb == null ? listWidth - trackWidth :
+ thumb.getLeft() + (thumb.getWidth() - trackWidth) / 2;
+ final int right = left + trackWidth;
+ final int top = thumbHalfHeight;
+ final int bottom = list.getHeight() - thumbHalfHeight;
+ track.layout(left, top, right, bottom);
}
- public void draw(Canvas canvas) {
+ private void setState(int state) {
+ mList.removeCallbacks(mDeferHide);
- if (mState == STATE_NONE) {
- // No need to draw anything
- return;
+ if (mAlwaysShow && state == STATE_NONE) {
+ state = STATE_VISIBLE;
}
- final int y = mThumbY;
- final int viewWidth = mList.getWidth();
- final FastScroller.ScrollFade scrollFade = mScrollFade;
+ if (state == mState) {
+ return;
+ }
- int alpha = -1;
- if (mState == STATE_EXIT) {
- alpha = scrollFade.getAlpha();
- if (alpha < ScrollFade.ALPHA_MAX / 2) {
- mThumbDrawable.setAlpha(alpha * 2);
- }
- int left = 0;
- switch (mPosition) {
- case View.SCROLLBAR_POSITION_RIGHT:
- left = viewWidth - (mThumbW * alpha) / ScrollFade.ALPHA_MAX;
- break;
- case View.SCROLLBAR_POSITION_LEFT:
- left = -mThumbW + (mThumbW * alpha) / ScrollFade.ALPHA_MAX;
- break;
- }
- mThumbDrawable.setBounds(left, 0, left + mThumbW, mThumbH);
- mChangedBounds = true;
- }
-
- if (mTrackDrawable != null) {
- final Rect thumbBounds = mThumbDrawable.getBounds();
- final int left = thumbBounds.left;
- final int halfThumbHeight = (thumbBounds.bottom - thumbBounds.top) / 2;
- final int trackWidth = mTrackDrawable.getIntrinsicWidth();
- final int trackLeft = (left + mThumbW / 2) - trackWidth / 2;
- mTrackDrawable.setBounds(trackLeft, halfThumbHeight,
- trackLeft + trackWidth, mList.getHeight() - halfThumbHeight);
- mTrackDrawable.draw(canvas);
- }
-
- canvas.translate(0, y);
- mThumbDrawable.draw(canvas);
- canvas.translate(0, -y);
-
- // If user is dragging the scroll bar, draw the alphabet overlay
- if (mState == STATE_DRAGGING && mDrawOverlay) {
- final Drawable overlay = mOverlayDrawable;
- final Paint paint = mPaint;
- final String sectionText = mSectionText;
- final Rect tmpRect = mTmpRect;
-
- // TODO: Use a text view in an overlay for transition animations and
- // handling of text overflow.
- paint.getTextBounds(sectionText, 0, sectionText.length(), tmpRect);
- final int textWidth = tmpRect.width();
- final int textHeight = tmpRect.height();
-
- overlay.getPadding(tmpRect);
- final int overlayWidth = Math.max(
- mOverlaySize, textWidth + tmpRect.left + tmpRect.right + mOverlayPadding * 2);
- final int overlayHeight = Math.max(
- mOverlaySize, textHeight + tmpRect.top + tmpRect.bottom + mOverlayPadding * 2);
- final RectF pos = mOverlayPos;
-
- if (mOverlayPosition == OVERLAY_AT_THUMB) {
- final Rect thumbBounds = mThumbDrawable.getBounds();
-
- switch (mPosition) {
- case View.SCROLLBAR_POSITION_LEFT:
- pos.left = Math.min(
- thumbBounds.right + mThumbW, mList.getWidth() - overlayWidth);
- break;
- case View.SCROLLBAR_POSITION_RIGHT:
- default:
- pos.left = Math.max(0, thumbBounds.left - mThumbW - overlayWidth);
- break;
- }
+ switch (state) {
+ case STATE_NONE:
+ transitionToHidden();
+ break;
+ case STATE_VISIBLE:
+ transitionToVisible();
+ break;
+ case STATE_DRAGGING:
+ transitionToDragging();
+ break;
+ }
- pos.top = Math.max(0, Math.min(
- y + (mThumbH - overlayHeight) / 2, mList.getHeight() - overlayHeight));
- }
+ mState = state;
- pos.right = pos.left + overlayWidth;
- pos.bottom = pos.top + overlayHeight;
+ refreshDrawablePressedState();
+ }
- overlay.setBounds((int) pos.left, (int) pos.top, (int) pos.right, (int) pos.bottom);
- overlay.draw(canvas);
+ private void refreshDrawablePressedState() {
+ final boolean isPressed = mState == STATE_DRAGGING;
+ mThumbImage.setPressed(isPressed);
+ mTrackImage.setPressed(isPressed);
+ }
- final float hOff = (tmpRect.right - tmpRect.left) / 2.0f;
- final float vOff = (tmpRect.bottom - tmpRect.top) / 2.0f;
- final float cX = pos.centerX() - hOff;
- final float cY = pos.centerY() + (overlayHeight / 4.0f) - paint.descent() - vOff;
- canvas.drawText(mSectionText, cX, cY, paint);
- } else if (mState == STATE_EXIT) {
- if (alpha == 0) { // Done with exit
- setState(STATE_NONE);
- } else {
- final int left, right, top, bottom;
- if (mTrackDrawable != null) {
- top = 0;
- bottom = mList.getHeight();
- } else {
- top = y;
- bottom = y + mThumbH;
- }
- switch (mList.getLayoutDirection()) {
- case View.LAYOUT_DIRECTION_RTL:
- left = 0;
- right = mThumbW;
- break;
- case View.LAYOUT_DIRECTION_LTR:
- default:
- left = viewWidth - mThumbW;
- right = viewWidth;
- }
- mList.invalidate(left, top, right, bottom);
- }
+ /**
+ * Shows nothing.
+ */
+ private void transitionToHidden() {
+ if (mDecorAnimation != null) {
+ mDecorAnimation.cancel();
}
+
+ final Animator fadeOut = groupAnimatorOfFloat(View.ALPHA, 0f, mThumbImage, mTrackImage,
+ mPreviewImage, mPrimaryText, mSecondaryText).setDuration(DURATION_FADE_OUT);
+
+ // Push the thumb and track outside the list bounds.
+ final float offset = mLayoutFromRight ? mThumbImage.getWidth() : -mThumbImage.getWidth();
+ final Animator slideOut = groupAnimatorOfFloat(
+ View.TRANSLATION_X, offset, mThumbImage, mTrackImage)
+ .setDuration(DURATION_FADE_OUT);
+
+ mDecorAnimation = new AnimatorSet();
+ mDecorAnimation.playTogether(fadeOut, slideOut);
+ mDecorAnimation.start();
}
- void onSizeChanged(int w, int h, int oldw, int oldh) {
- if (mThumbDrawable != null) {
- switch (mPosition) {
- default:
- case View.SCROLLBAR_POSITION_RIGHT:
- mThumbDrawable.setBounds(w - mThumbW, 0, w, mThumbH);
- break;
- case View.SCROLLBAR_POSITION_LEFT:
- mThumbDrawable.setBounds(0, 0, mThumbW, mThumbH);
- break;
- }
- }
- if (mOverlayPosition == OVERLAY_FLOATING) {
- final RectF pos = mOverlayPos;
- pos.left = (w - mOverlaySize) / 2;
- pos.right = pos.left + mOverlaySize;
- pos.top = h / 10; // 10% from top
- pos.bottom = pos.top + mOverlaySize;
- if (mOverlayDrawable != null) {
- mOverlayDrawable.setBounds((int) pos.left, (int) pos.top,
- (int) pos.right, (int) pos.bottom);
- }
+ /**
+ * Shows the thumb and track.
+ */
+ private void transitionToVisible() {
+ if (mDecorAnimation != null) {
+ mDecorAnimation.cancel();
}
+
+ final Animator fadeIn = groupAnimatorOfFloat(View.ALPHA, 1f, mThumbImage, mTrackImage)
+ .setDuration(DURATION_FADE_IN);
+ final Animator fadeOut = groupAnimatorOfFloat(
+ View.ALPHA, 0f, mPreviewImage, mPrimaryText, mSecondaryText)
+ .setDuration(DURATION_FADE_OUT);
+ final Animator slideIn = groupAnimatorOfFloat(
+ View.TRANSLATION_X, 0f, mThumbImage, mTrackImage).setDuration(DURATION_FADE_IN);
+
+ mDecorAnimation = new AnimatorSet();
+ mDecorAnimation.playTogether(fadeIn, fadeOut, slideIn);
+ mDecorAnimation.start();
}
- void onItemCountChanged(int oldCount, int newCount) {
- if (mAlwaysShow) {
- mLongList = true;
+ /**
+ * Shows the thumb, preview, and track.
+ */
+ private void transitionToDragging() {
+ if (mDecorAnimation != null) {
+ mDecorAnimation.cancel();
}
+
+ final Animator fadeIn = groupAnimatorOfFloat(
+ View.ALPHA, 1f, mThumbImage, mTrackImage, mPreviewImage)
+ .setDuration(DURATION_FADE_IN);
+ final Animator slideIn = groupAnimatorOfFloat(
+ View.TRANSLATION_X, 0f, mThumbImage, mTrackImage).setDuration(DURATION_FADE_IN);
+
+ mDecorAnimation = new AnimatorSet();
+ mDecorAnimation.playTogether(fadeIn, slideIn);
+ mDecorAnimation.start();
+
+ // Ensure the preview text is correct.
+ final String previewText = getPreviewText();
+ transitionPreviewLayout(previewText);
}
- void onScroll(AbsListView view, int firstVisibleItem, int visibleItemCount,
- int totalItemCount) {
- // Are there enough pages to require fast scroll? Recompute only if total count changes
+ private boolean isLongList(int visibleItemCount, int totalItemCount) {
+ // Are there enough pages to require fast scroll? Recompute only if
+ // total count changes.
if (mItemCount != totalItemCount && visibleItemCount > 0) {
mItemCount = totalItemCount;
mLongList = mItemCount / visibleItemCount >= MIN_PAGES;
}
- if (mAlwaysShow) {
- mLongList = true;
- }
- if (!mLongList) {
- if (mState != STATE_NONE) {
- setState(STATE_NONE);
- }
+
+ return mLongList;
+ }
+
+ public void onScroll(int firstVisibleItem, int visibleItemCount, int totalItemCount) {
+ if (!mAlwaysShow && !isLongList(visibleItemCount, totalItemCount)) {
+ setState(STATE_NONE);
return;
}
- if (totalItemCount - visibleItemCount > 0 && mState != STATE_DRAGGING) {
- mThumbY = getThumbPositionForListPosition(firstVisibleItem, visibleItemCount,
- totalItemCount);
- if (mChangedBounds) {
- resetThumbPos();
- mChangedBounds = false;
- }
+
+ final boolean hasMoreItems = totalItemCount - visibleItemCount > 0;
+ if (hasMoreItems && mState != STATE_DRAGGING) {
+ setThumbPos(getPosFromItemCount(firstVisibleItem, visibleItemCount, totalItemCount));
}
+
mScrollCompleted = true;
- if (firstVisibleItem == mVisibleItem) {
- return;
- }
- mVisibleItem = firstVisibleItem;
- if (mState != STATE_DRAGGING) {
- setState(STATE_VISIBLE);
- if (!mAlwaysShow) {
- mHandler.postDelayed(mScrollFade, FADE_TIMEOUT);
- }
- }
- }
- SectionIndexer getSectionIndexer() {
- return mSectionIndexer;
- }
+ if (mFirstVisibleItem != firstVisibleItem) {
+ mFirstVisibleItem = firstVisibleItem;
- Object[] getSections() {
- if (mListAdapter == null && mList != null) {
- getSectionsFromIndexer();
+ // Show the thumb, if necessary, and set up auto-fade.
+ if (mState != STATE_DRAGGING) {
+ setState(STATE_VISIBLE);
+ mList.postDelayed(mDeferHide, FADE_TIMEOUT);
+ }
}
- return mSections;
}
- void getSectionsFromIndexer() {
- Adapter adapter = mList.getAdapter();
+ private void getSectionsFromIndexer() {
mSectionIndexer = null;
+
+ Adapter adapter = mList.getAdapter();
if (adapter instanceof HeaderViewListAdapter) {
- mListOffset = ((HeaderViewListAdapter)adapter).getHeadersCount();
- adapter = ((HeaderViewListAdapter)adapter).getWrappedAdapter();
+ mHeaderCount = ((HeaderViewListAdapter) adapter).getHeadersCount();
+ adapter = ((HeaderViewListAdapter) adapter).getWrappedAdapter();
}
+
if (adapter instanceof ExpandableListConnector) {
- ExpandableListAdapter expAdapter = ((ExpandableListConnector)adapter).getAdapter();
+ final ExpandableListAdapter expAdapter = ((ExpandableListConnector) adapter)
+ .getAdapter();
if (expAdapter instanceof SectionIndexer) {
mSectionIndexer = (SectionIndexer) expAdapter;
mListAdapter = (BaseAdapter) adapter;
mSections = mSectionIndexer.getSections();
}
+ } else if (adapter instanceof SectionIndexer) {
+ mListAdapter = (BaseAdapter) adapter;
+ mSectionIndexer = (SectionIndexer) adapter;
+ mSections = mSectionIndexer.getSections();
} else {
- if (adapter instanceof SectionIndexer) {
- mListAdapter = (BaseAdapter) adapter;
- mSectionIndexer = (SectionIndexer) adapter;
- mSections = mSectionIndexer.getSections();
- if (mSections == null) {
- mSections = new String[] { " " };
- }
- } else {
- mListAdapter = (BaseAdapter) adapter;
- mSections = new String[] { " " };
- }
+ mListAdapter = (BaseAdapter) adapter;
+ mSections = null;
}
}
@@ -564,21 +745,24 @@ class FastScroller {
mListAdapter = null;
}
- void scrollTo(float position) {
- int count = mList.getCount();
+ /**
+ * Scrolls to a specific position within the section
+ * @param position
+ */
+ private void scrollTo(float position) {
mScrollCompleted = false;
- float fThreshold = (1.0f / count) / 8;
+
+ final int count = mList.getCount();
final Object[] sections = mSections;
+ final int sectionCount = sections == null ? 0 : sections.length;
int sectionIndex;
- if (sections != null && sections.length > 1) {
- final int nSections = sections.length;
- int section = (int) (position * nSections);
- if (section >= nSections) {
- section = nSections - 1;
- }
- int exactSection = section;
- sectionIndex = section;
- int index = mSectionIndexer.getPositionForSection(section);
+ if (sections != null && sectionCount > 1) {
+ final int exactSection = MathUtils.constrain(
+ (int) (position * sectionCount), 0, sectionCount - 1);
+ int targetSection = exactSection;
+ int targetIndex = mSectionIndexer.getPositionForSection(targetSection);
+ sectionIndex = targetSection;
+
// Given the expected section and index, the following code will
// try to account for missing sections (no names starting with..)
// It will compute the scroll space of surrounding empty sections
@@ -586,25 +770,26 @@ class FastScroller {
// available space, so that there is always some list movement while
// the user moves the thumb.
int nextIndex = count;
- int prevIndex = index;
- int prevSection = section;
- int nextSection = section + 1;
+ int prevIndex = targetIndex;
+ int prevSection = targetSection;
+ int nextSection = targetSection + 1;
+
// Assume the next section is unique
- if (section < nSections - 1) {
- nextIndex = mSectionIndexer.getPositionForSection(section + 1);
+ if (targetSection < sectionCount - 1) {
+ nextIndex = mSectionIndexer.getPositionForSection(targetSection + 1);
}
// Find the previous index if we're slicing the previous section
- if (nextIndex == index) {
+ if (nextIndex == targetIndex) {
// Non-existent letter
- while (section > 0) {
- section--;
- prevIndex = mSectionIndexer.getPositionForSection(section);
- if (prevIndex != index) {
- prevSection = section;
- sectionIndex = section;
+ while (targetSection > 0) {
+ targetSection--;
+ prevIndex = mSectionIndexer.getPositionForSection(targetSection);
+ if (prevIndex != targetIndex) {
+ prevSection = targetSection;
+ sectionIndex = targetSection;
break;
- } else if (section == 0) {
+ } else if (targetSection == 0) {
// When section reaches 0 here, sectionIndex must follow it.
// Assuming mSectionIndexer.getPositionForSection(0) == 0.
sectionIndex = 0;
@@ -612,131 +797,281 @@ class FastScroller {
}
}
}
+
// Find the next index, in case the assumed next index is not
// unique. For instance, if there is no P, then request for P's
// position actually returns Q's. So we need to look ahead to make
// sure that there is really a Q at Q's position. If not, move
// further down...
int nextNextSection = nextSection + 1;
- while (nextNextSection < nSections &&
+ while (nextNextSection < sectionCount &&
mSectionIndexer.getPositionForSection(nextNextSection) == nextIndex) {
nextNextSection++;
nextSection++;
}
+
// Compute the beginning and ending scroll range percentage of the
- // currently visible letter. This could be equal to or greater than
- // (1 / nSections).
- float fPrev = (float) prevSection / nSections;
- float fNext = (float) nextSection / nSections;
- if (prevSection == exactSection && position - fPrev < fThreshold) {
- index = prevIndex;
+ // currently visible section. This could be equal to or greater than
+ // (1 / nSections). If the target position is near the previous
+ // position, snap to the previous position.
+ final float prevPosition = (float) prevSection / sectionCount;
+ final float nextPosition = (float) nextSection / sectionCount;
+ final float snapThreshold = (count == 0) ? Float.MAX_VALUE : .125f / count;
+ if (prevSection == exactSection && position - prevPosition < snapThreshold) {
+ targetIndex = prevIndex;
} else {
- index = prevIndex + (int) ((nextIndex - prevIndex) * (position - fPrev)
- / (fNext - fPrev));
+ targetIndex = prevIndex + (int) ((nextIndex - prevIndex) * (position - prevPosition)
+ / (nextPosition - prevPosition));
}
- // Don't overflow
- if (index > count - 1) index = count - 1;
+
+ // Clamp to valid positions.
+ targetIndex = MathUtils.constrain(targetIndex, 0, count - 1);
if (mList instanceof ExpandableListView) {
- ExpandableListView expList = (ExpandableListView) mList;
+ final ExpandableListView expList = (ExpandableListView) mList;
expList.setSelectionFromTop(expList.getFlatListPosition(
- ExpandableListView.getPackedPositionForGroup(index + mListOffset)), 0);
+ ExpandableListView.getPackedPositionForGroup(targetIndex + mHeaderCount)),
+ 0);
} else if (mList instanceof ListView) {
- ((ListView)mList).setSelectionFromTop(index + mListOffset, 0);
+ ((ListView) mList).setSelectionFromTop(targetIndex + mHeaderCount, 0);
} else {
- mList.setSelection(index + mListOffset);
+ mList.setSelection(targetIndex + mHeaderCount);
}
} else {
- int index = (int) (position * count);
- // Don't overflow
- if (index > count - 1) index = count - 1;
+ final int index = MathUtils.constrain((int) (position * count), 0, count - 1);
if (mList instanceof ExpandableListView) {
ExpandableListView expList = (ExpandableListView) mList;
expList.setSelectionFromTop(expList.getFlatListPosition(
- ExpandableListView.getPackedPositionForGroup(index + mListOffset)), 0);
+ ExpandableListView.getPackedPositionForGroup(index + mHeaderCount)), 0);
} else if (mList instanceof ListView) {
- ((ListView)mList).setSelectionFromTop(index + mListOffset, 0);
+ ((ListView)mList).setSelectionFromTop(index + mHeaderCount, 0);
} else {
- mList.setSelection(index + mListOffset);
+ mList.setSelection(index + mHeaderCount);
}
+
sectionIndex = -1;
}
- if (sectionIndex >= 0) {
- String text = mSectionText = sections[sectionIndex].toString();
- mDrawOverlay = (text.length() != 1 || text.charAt(0) != ' ') &&
- sectionIndex < sections.length;
+ if (sectionIndex >= 0 && sectionIndex < sections.length) {
+ // If we moved sections, display section.
+ if (mCurrentSection != sectionIndex) {
+ mCurrentSection = sectionIndex;
+ final String section = sections[sectionIndex].toString();
+ transitionToDragging();
+ transitionPreviewLayout(section);
+ }
} else {
- mDrawOverlay = false;
+ // No current section, transition out of preview.
+ transitionPreviewLayout(null);
+ transitionToVisible();
}
}
- private int getThumbPositionForListPosition(int firstVisibleItem, int visibleItemCount,
- int totalItemCount) {
+ private String getPreviewText() {
+ final Object[] sections = mSections;
+ if (sections == null) {
+ return null;
+ }
+
+ final int sectionIndex = mCurrentSection;
+ if (sectionIndex < 0 || sectionIndex >= sections.length) {
+ return null;
+ }
+
+ return sections[sectionIndex].toString();
+ }
+
+ /**
+ * Transitions the preview text to a new value. Handles animation,
+ * measurement, and layout.
+ *
+ * @param text The preview text to transition to.
+ */
+ private void transitionPreviewLayout(CharSequence text) {
+ final Rect bounds = mTempBounds;
+ final ImageView preview = mPreviewImage;
+ final TextView showing;
+ final TextView target;
+ if (mShowingPrimary) {
+ showing = mPrimaryText;
+ target = mSecondaryText;
+ } else {
+ showing = mSecondaryText;
+ target = mPrimaryText;
+ }
+
+ // Set and layout target immediately.
+ target.setText(text);
+ measurePreview(target, bounds);
+ applyLayout(target, bounds);
+
+ if (mPreviewAnimation != null) {
+ mPreviewAnimation.cancel();
+ }
+
+ // Cross-fade preview text.
+ final Animator showTarget = animateAlpha(target, 1f).setDuration(DURATION_CROSS_FADE);
+ final Animator hideShowing = animateAlpha(showing, 0f).setDuration(DURATION_CROSS_FADE);
+ hideShowing.addListener(mSwitchPrimaryListener);
+
+ // Apply preview image padding and animate bounds, if necessary.
+ bounds.left -= mPreviewImage.getPaddingLeft();
+ bounds.top -= mPreviewImage.getPaddingTop();
+ bounds.right += mPreviewImage.getPaddingRight();
+ bounds.bottom += mPreviewImage.getPaddingBottom();
+ final Animator resizePreview = animateBounds(preview, bounds);
+ resizePreview.setDuration(DURATION_RESIZE);
+
+ mPreviewAnimation = new AnimatorSet();
+ final AnimatorSet.Builder builder = mPreviewAnimation.play(hideShowing).with(showTarget);
+ builder.with(resizePreview);
+
+ // The current preview size is unaffected by hidden or showing. It's
+ // used to set starting scales for things that need to be scaled down.
+ final int previewWidth = preview.getWidth() - preview.getPaddingLeft()
+ - preview.getPaddingRight();
+
+ // If target is too large, shrink it immediately to fit and expand to
+ // target size. Otherwise, start at target size.
+ final int targetWidth = target.getWidth();
+ if (targetWidth > previewWidth) {
+ target.setScaleX((float) previewWidth / targetWidth);
+ final Animator scaleAnim = animateScaleX(target, 1f).setDuration(DURATION_RESIZE);
+ builder.with(scaleAnim);
+ } else {
+ target.setScaleX(1f);
+ }
+
+ // If showing is larger than target, shrink to target size.
+ final int showingWidth = showing.getWidth();
+ if (showingWidth > targetWidth) {
+ final float scale = (float) targetWidth / showingWidth;
+ final Animator scaleAnim = animateScaleX(showing, scale).setDuration(DURATION_RESIZE);
+ builder.with(scaleAnim);
+ }
+
+ mPreviewAnimation.start();
+ }
+
+ /**
+ * Positions the thumb and preview widgets.
+ *
+ * @param position The position, between 0 and 1, along the track at which
+ * to place the thumb.
+ */
+ private void setThumbPos(float position) {
+ final int top = 0;
+ final int bottom = mList.getHeight();
+
+ final float thumbHalfHeight = mThumbImage.getHeight() / 2f;
+ final float min = top + thumbHalfHeight;
+ final float max = bottom - thumbHalfHeight;
+ final float offset = min;
+ final float range = max - min;
+ final float thumbMiddle = position * range + offset;
+ mThumbImage.setTranslationY(thumbMiddle - thumbHalfHeight);
+
+ // Center the preview on the thumb, constrained to the list bounds.
+ final float previewHalfHeight = mPreviewImage.getHeight() / 2f;
+ final float minP = top + previewHalfHeight;
+ final float maxP = bottom - previewHalfHeight;
+ final float previewMiddle = MathUtils.constrain(thumbMiddle, minP, maxP);
+ final float previewTop = previewMiddle - previewHalfHeight;
+
+ mPreviewImage.setTranslationY(previewTop);
+ mPrimaryText.setTranslationY(previewTop);
+ mSecondaryText.setTranslationY(previewTop);
+ }
+
+ private float getPosFromMotionEvent(float y) {
+ final int top = 0;
+ final int bottom = mList.getHeight();
+
+ final float thumbHalfHeight = mThumbImage.getHeight() / 2f;
+ final float min = top + thumbHalfHeight;
+ final float max = bottom - thumbHalfHeight;
+ final float offset = min;
+ final float range = max - min;
+
+ // If the list is the same height as the thumbnail or shorter,
+ // effectively disable scrolling.
+ if (range <= 0) {
+ return 0f;
+ }
+
+ return MathUtils.constrain((y - offset) / range, 0f, 1f);
+ }
+
+ private float getPosFromItemCount(
+ int firstVisibleItem, int visibleItemCount, int totalItemCount) {
if (mSectionIndexer == null || mListAdapter == null) {
getSectionsFromIndexer();
}
- if (mSectionIndexer == null || !mMatchDragPosition) {
- return ((mList.getHeight() - mThumbH) * firstVisibleItem)
- / (totalItemCount - visibleItemCount);
+
+ final boolean hasSections = mSectionIndexer != null && mSections != null
+ && mSections.length > 0;
+ if (!hasSections || !mMatchDragPosition) {
+ return firstVisibleItem / (totalItemCount - visibleItemCount);
}
- firstVisibleItem -= mListOffset;
+ firstVisibleItem -= mHeaderCount;
if (firstVisibleItem < 0) {
return 0;
}
- totalItemCount -= mListOffset;
- final int trackHeight = mList.getHeight() - mThumbH;
+ totalItemCount -= mHeaderCount;
final int section = mSectionIndexer.getSectionForPosition(firstVisibleItem);
final int sectionPos = mSectionIndexer.getPositionForSection(section);
final int nextSectionPos = mSectionIndexer.getPositionForSection(section + 1);
final int sectionCount = mSections.length;
- final int positionsInSection = nextSectionPos - sectionPos;
+ final int positionsInSection = Math.max(1, nextSectionPos - sectionPos);
final View child = mList.getChildAt(0);
final float incrementalPos = child == null ? 0 : firstVisibleItem +
- (float) (mList.getPaddingTop() - child.getTop()) / child.getHeight();
+ (float) (mList.getPaddingTop() - child.getTop()) / Math.max(1, child.getHeight());
final float posWithinSection = (incrementalPos - sectionPos) / positionsInSection;
- int result = (int) ((section + posWithinSection) / sectionCount * trackHeight);
-
- // Fake out the scrollbar for the last item. Since the section indexer won't
- // ever actually move the list in this end space, make scrolling across the last item
- // account for whatever space is remaining.
- if (firstVisibleItem > 0 && firstVisibleItem + visibleItemCount == totalItemCount) {
- final View lastChild = mList.getChildAt(visibleItemCount - 1);
- final float lastItemVisible = (float) (mList.getHeight() - mList.getPaddingBottom()
- - lastChild.getTop()) / lastChild.getHeight();
- result += (trackHeight - result) * lastItemVisible;
- }
-
- return result;
+ return (section + posWithinSection) / sectionCount;
}
+ /**
+ * Cancels an ongoing fling event by injecting a
+ * {@link MotionEvent#ACTION_CANCEL} into the host view.
+ */
private void cancelFling() {
- // Cancel the list fling
- MotionEvent cancelFling = MotionEvent.obtain(0, 0, MotionEvent.ACTION_CANCEL, 0, 0, 0);
+ final MotionEvent cancelFling = MotionEvent.obtain(
+ 0, 0, MotionEvent.ACTION_CANCEL, 0, 0, 0);
mList.onTouchEvent(cancelFling);
cancelFling.recycle();
}
- void cancelPendingDrag() {
+ /**
+ * Cancels a pending drag.
+ *
+ * @see #startPendingDrag()
+ */
+ private void cancelPendingDrag() {
mList.removeCallbacks(mDeferStartDrag);
- mPendingDrag = false;
+ mHasPendingDrag = false;
}
- void startPendingDrag() {
- mPendingDrag = true;
- mList.postDelayed(mDeferStartDrag, PENDING_DRAG_DELAY);
+ /**
+ * Delays dragging until after the framework has determined that the user is
+ * scrolling, rather than tapping.
+ */
+ private void startPendingDrag() {
+ mHasPendingDrag = true;
+ mList.postDelayed(mDeferStartDrag, TAP_TIMEOUT);
}
- void beginDrag() {
+ private void beginDrag() {
setState(STATE_DRAGGING);
+
if (mListAdapter == null && mList != null) {
getSectionsFromIndexer();
}
+
if (mList != null) {
mList.requestDisallowInterceptTouchEvent(true);
mList.reportScrollStateChange(OnScrollListener.SCROLL_STATE_TOUCH_SCROLL);
@@ -745,16 +1080,23 @@ class FastScroller {
cancelFling();
}
- boolean onInterceptTouchEvent(MotionEvent ev) {
+ public boolean onInterceptTouchEvent(MotionEvent ev) {
switch (ev.getActionMasked()) {
case MotionEvent.ACTION_DOWN:
- if (mState > STATE_NONE && isPointInside(ev.getX(), ev.getY())) {
- if (!mList.isInScrollingContainer()) {
- beginDrag();
- return true;
+ if (isPointInside(ev.getX(), ev.getY())) {
+ // If the parent has requested that its children delay
+ // pressed state (e.g. is a scrolling container) then we
+ // need to allow the parent time to decide whether it wants
+ // to intercept events. If it does, we will receive a CANCEL
+ // event.
+ if (mList.isInScrollingContainer()) {
+ mInitialTouchY = ev.getY();
+ startPendingDrag();
+ return false;
}
- mInitialTouchY = ev.getY();
- startPendingDrag();
+
+ beginDrag();
+ return true;
}
break;
case MotionEvent.ACTION_UP:
@@ -762,70 +1104,56 @@ class FastScroller {
cancelPendingDrag();
break;
}
+
return false;
}
- boolean onTouchEvent(MotionEvent me) {
- if (mState == STATE_NONE) {
- return false;
- }
-
- final int action = me.getAction();
-
- if (action == MotionEvent.ACTION_DOWN) {
- if (isPointInside(me.getX(), me.getY())) {
- if (!mList.isInScrollingContainer()) {
+ public boolean onTouchEvent(MotionEvent me) {
+ switch (me.getActionMasked()) {
+ case MotionEvent.ACTION_DOWN: {
+ if (isPointInside(me.getX(), me.getY())) {
beginDrag();
return true;
}
- mInitialTouchY = me.getY();
- startPendingDrag();
- }
- } else if (action == MotionEvent.ACTION_UP) { // don't add ACTION_CANCEL here
- if (mPendingDrag) {
- // Allow a tap to scroll.
- beginDrag();
+ } break;
- final int viewHeight = mList.getHeight();
- // Jitter
- int newThumbY = (int) me.getY() - mThumbH + 10;
- if (newThumbY < 0) {
- newThumbY = 0;
- } else if (newThumbY + mThumbH > viewHeight) {
- newThumbY = viewHeight - mThumbH;
- }
- mThumbY = newThumbY;
- scrollTo((float) mThumbY / (viewHeight - mThumbH));
+ case MotionEvent.ACTION_UP: {
+ if (mHasPendingDrag) {
+ // Allow a tap to scroll.
+ beginDrag();
- cancelPendingDrag();
- // Will hit the STATE_DRAGGING check below
- }
- if (mState == STATE_DRAGGING) {
- if (mList != null) {
- // ViewGroup does the right thing already, but there might
- // be other classes that don't properly reset on touch-up,
- // so do this explicitly just in case.
- mList.requestDisallowInterceptTouchEvent(false);
- mList.reportScrollStateChange(OnScrollListener.SCROLL_STATE_IDLE);
+ final float pos = getPosFromMotionEvent(me.getY());
+ setThumbPos(pos);
+ scrollTo(pos);
+
+ cancelPendingDrag();
+ // Will hit the STATE_DRAGGING check below
}
- setState(STATE_VISIBLE);
- final Handler handler = mHandler;
- handler.removeCallbacks(mScrollFade);
- if (!mAlwaysShow) {
- handler.postDelayed(mScrollFade, 1000);
+
+ if (mState == STATE_DRAGGING) {
+ if (mList != null) {
+ // ViewGroup does the right thing already, but there might
+ // be other classes that don't properly reset on touch-up,
+ // so do this explicitly just in case.
+ mList.requestDisallowInterceptTouchEvent(false);
+ mList.reportScrollStateChange(OnScrollListener.SCROLL_STATE_IDLE);
+ }
+
+ setState(STATE_VISIBLE);
+ mList.postDelayed(mDeferHide, FADE_TIMEOUT);
+
+ return true;
}
+ } break;
- mList.invalidate();
- return true;
- }
- } else if (action == MotionEvent.ACTION_MOVE) {
- if (mPendingDrag) {
- final float y = me.getY();
- if (Math.abs(y - mInitialTouchY) > mScaledTouchSlop) {
+ case MotionEvent.ACTION_MOVE: {
+ if (mHasPendingDrag && Math.abs(me.getY() - mInitialTouchY) > mScaledTouchSlop) {
setState(STATE_DRAGGING);
+
if (mListAdapter == null && mList != null) {
getSectionsFromIndexer();
}
+
if (mList != null) {
mList.requestDisallowInterceptTouchEvent(true);
mList.reportScrollStateChange(OnScrollListener.SCROLL_STATE_TOUCH_SCROLL);
@@ -835,87 +1163,168 @@ class FastScroller {
cancelPendingDrag();
// Will hit the STATE_DRAGGING check below
}
- }
- if (mState == STATE_DRAGGING) {
- final int viewHeight = mList.getHeight();
- // Jitter
- int newThumbY = (int) me.getY() - mThumbH + 10;
- if (newThumbY < 0) {
- newThumbY = 0;
- } else if (newThumbY + mThumbH > viewHeight) {
- newThumbY = viewHeight - mThumbH;
- }
- if (Math.abs(mThumbY - newThumbY) < 2) {
+
+ if (mState == STATE_DRAGGING) {
+ // TODO: Ignore jitter.
+ final float pos = getPosFromMotionEvent(me.getY());
+ setThumbPos(pos);
+
+ // If the previous scrollTo is still pending
+ if (mScrollCompleted) {
+ scrollTo(pos);
+ }
+
return true;
}
- mThumbY = newThumbY;
- // If the previous scrollTo is still pending
- if (mScrollCompleted) {
- scrollTo((float) mThumbY / (viewHeight - mThumbH));
- }
- return true;
- }
- } else if (action == MotionEvent.ACTION_CANCEL) {
- cancelPendingDrag();
+ } break;
+
+ case MotionEvent.ACTION_CANCEL: {
+ cancelPendingDrag();
+ } break;
}
+
return false;
}
- boolean isPointInside(float x, float y) {
- boolean inTrack = false;
- switch (mPosition) {
- default:
- case View.SCROLLBAR_POSITION_RIGHT:
- inTrack = x > mList.getWidth() - mThumbW;
- break;
- case View.SCROLLBAR_POSITION_LEFT:
- inTrack = x < mThumbW;
- break;
+ /**
+ * Returns whether a coordinate is inside the scroller's activation area. If
+ * there is a track image, touching anywhere within the thumb-width of the
+ * track activates scrolling. Otherwise, the user has to touch inside thumb
+ * itself.
+ *
+ * @param x The x-coordinate.
+ * @param y The y-coordinate.
+ * @return Whether the coordinate is inside the scroller's activation area.
+ */
+ private boolean isPointInside(float x, float y) {
+ return isPointInsideX(x) && (mHasTrackImage || isPointInsideY(y));
+ }
+
+ private boolean isPointInsideX(float x) {
+ if (mLayoutFromRight) {
+ return x >= mThumbImage.getLeft();
+ } else {
+ return x <= mThumbImage.getRight();
+ }
+ }
+
+ private boolean isPointInsideY(float y) {
+ return y >= mThumbImage.getTop() && y <= mThumbImage.getBottom();
+ }
+
+ /**
+ * Constructs an animator for the specified property on a group of views.
+ * See {@link ObjectAnimator#ofFloat(Object, String, float...)} for
+ * implementation details.
+ *
+ * @param property The property being animated.
+ * @param value The value to which that property should animate.
+ * @param views The target views to animate.
+ * @return An animator for all the specified views.
+ */
+ private static Animator groupAnimatorOfFloat(
+ Property<View, Float> property, float value, View... views) {
+ AnimatorSet animSet = new AnimatorSet();
+ AnimatorSet.Builder builder = null;
+
+ for (int i = views.length - 1; i >= 0; i--) {
+ final Animator anim = ObjectAnimator.ofFloat(views[i], property, value);
+ if (builder == null) {
+ builder = animSet.play(anim);
+ } else {
+ builder.with(anim);
+ }
}
- // Allow taps in the track to start moving.
- return inTrack && (mTrackDrawable != null || y >= mThumbY && y <= mThumbY + mThumbH);
+ return animSet;
}
- public class ScrollFade implements Runnable {
+ /**
+ * Returns an animator for the view's scaleX value.
+ */
+ private static Animator animateScaleX(View v, float target) {
+ return ObjectAnimator.ofFloat(v, View.SCALE_X, target);
+ }
- long mStartTime;
- long mFadeDuration;
- static final int ALPHA_MAX = 208;
- static final long FADE_DURATION = 200;
+ /**
+ * Returns an animator for the view's alpha value.
+ */
+ private static Animator animateAlpha(View v, float alpha) {
+ return ObjectAnimator.ofFloat(v, View.ALPHA, alpha);
+ }
- void startFade() {
- mFadeDuration = FADE_DURATION;
- mStartTime = SystemClock.uptimeMillis();
- setState(STATE_EXIT);
+ /**
+ * A Property wrapper around the <code>left</code> functionality handled by the
+ * {@link View#setLeft(int)} and {@link View#getLeft()} methods.
+ */
+ private static Property<View, Integer> LEFT = new IntProperty<View>("left") {
+ @Override
+ public void setValue(View object, int value) {
+ object.setLeft(value);
}
- int getAlpha() {
- if (getState() != STATE_EXIT) {
- return ALPHA_MAX;
- }
- int alpha;
- long now = SystemClock.uptimeMillis();
- if (now > mStartTime + mFadeDuration) {
- alpha = 0;
- } else {
- alpha = (int) (ALPHA_MAX - ((now - mStartTime) * ALPHA_MAX) / mFadeDuration);
- }
- return alpha;
+ @Override
+ public Integer get(View object) {
+ return object.getLeft();
}
+ };
+ /**
+ * A Property wrapper around the <code>top</code> functionality handled by the
+ * {@link View#setTop(int)} and {@link View#getTop()} methods.
+ */
+ private static Property<View, Integer> TOP = new IntProperty<View>("top") {
@Override
- public void run() {
- if (getState() != STATE_EXIT) {
- startFade();
- return;
- }
+ public void setValue(View object, int value) {
+ object.setTop(value);
+ }
- if (getAlpha() > 0) {
- mList.invalidate();
- } else {
- setState(STATE_NONE);
- }
+ @Override
+ public Integer get(View object) {
+ return object.getTop();
+ }
+ };
+
+ /**
+ * A Property wrapper around the <code>right</code> functionality handled by the
+ * {@link View#setRight(int)} and {@link View#getRight()} methods.
+ */
+ private static Property<View, Integer> RIGHT = new IntProperty<View>("right") {
+ @Override
+ public void setValue(View object, int value) {
+ object.setRight(value);
+ }
+
+ @Override
+ public Integer get(View object) {
+ return object.getRight();
}
+ };
+
+ /**
+ * A Property wrapper around the <code>bottom</code> functionality handled by the
+ * {@link View#setBottom(int)} and {@link View#getBottom()} methods.
+ */
+ private static Property<View, Integer> BOTTOM = new IntProperty<View>("bottom") {
+ @Override
+ public void setValue(View object, int value) {
+ object.setBottom(value);
+ }
+
+ @Override
+ public Integer get(View object) {
+ return object.getBottom();
+ }
+ };
+
+ /**
+ * Returns an animator for the view's bounds.
+ */
+ private static Animator animateBounds(View v, Rect bounds) {
+ final PropertyValuesHolder left = PropertyValuesHolder.ofInt(LEFT, bounds.left);
+ final PropertyValuesHolder top = PropertyValuesHolder.ofInt(TOP, bounds.top);
+ final PropertyValuesHolder right = PropertyValuesHolder.ofInt(RIGHT, bounds.right);
+ final PropertyValuesHolder bottom = PropertyValuesHolder.ofInt(BOTTOM, bounds.bottom);
+ return ObjectAnimator.ofPropertyValuesHolder(v, left, top, right, bottom);
}
}
diff --git a/core/res/res/values/dimens.xml b/core/res/res/values/dimens.xml
index ccca2d8e6461..00caac968206 100644
--- a/core/res/res/values/dimens.xml
+++ b/core/res/res/values/dimens.xml
@@ -52,6 +52,8 @@
<!-- Minimum size of the fastscroll overlay -->
<dimen name="fastscroll_overlay_size">104dp</dimen>
+ <!-- Text size of the fastscroll overlay -->
+ <dimen name="fastscroll_overlay_text_size">24sp</dimen>
<!-- Padding of the fastscroll overlay -->
<dimen name="fastscroll_overlay_padding">16dp</dimen>
<!-- Width of the fastscroll thumb -->
diff --git a/core/res/res/values/symbols.xml b/core/res/res/values/symbols.xml
index cb8d1442956f..7f39364b5848 100755
--- a/core/res/res/values/symbols.xml
+++ b/core/res/res/values/symbols.xml
@@ -309,6 +309,7 @@
<java-symbol type="dimen" name="dropdownitem_icon_width" />
<java-symbol type="dimen" name="dropdownitem_text_padding_left" />
<java-symbol type="dimen" name="fastscroll_overlay_size" />
+ <java-symbol type="dimen" name="fastscroll_overlay_text_size" />
<java-symbol type="dimen" name="fastscroll_overlay_padding" />
<java-symbol type="dimen" name="fastscroll_thumb_height" />
<java-symbol type="dimen" name="fastscroll_thumb_width" />