package cn.sinata.xldutils.view;
|
|
import android.content.Context;
|
import android.database.DataSetObserver;
|
import android.os.Build;
|
import android.os.Parcel;
|
import android.os.Parcelable;
|
import android.util.AttributeSet;
|
import android.util.Log;
|
import android.view.MotionEvent;
|
import android.view.VelocityTracker;
|
import android.view.View;
|
import android.view.ViewConfiguration;
|
import android.view.ViewGroup;
|
import android.widget.Scroller;
|
|
import androidx.core.os.ParcelableCompat;
|
import androidx.core.os.ParcelableCompatCreatorCallbacks;
|
import androidx.core.view.MotionEventCompat;
|
import androidx.core.view.VelocityTrackerCompat;
|
import androidx.core.view.ViewConfigurationCompat;
|
import androidx.viewpager.widget.PagerAdapter;
|
import androidx.viewpager.widget.ViewPager;
|
|
import java.util.ArrayList;
|
|
|
/**
|
* Layout manager that allows the user to flip horizontally or vertically
|
* through pages of data. You supply an implementation of a
|
* <p/>
|
* <p>Note this class is currently under early design and
|
* development. The API will likely change in later updates of
|
* the compatibility library, requiring changes to the source code
|
* of apps when they are compiled against the newer version.</p>
|
*/
|
public class DirectionalViewPager extends ViewPager {
|
private static final String TAG = "DirectionalViewPager";
|
private static final String XML_NS = "http://schemas.android.com/apk/res/android";
|
private static final boolean DEBUG = false;
|
|
private static final boolean USE_CACHE = false;
|
|
public static final int HORIZONTAL = 0;
|
public static final int VERTICAL = 1;
|
|
static class ItemInfo {
|
Object object;
|
int position;
|
boolean scrolling;
|
}
|
|
private final ArrayList<ItemInfo> mItems = new ArrayList<ItemInfo>();
|
|
private PagerAdapter mAdapter;
|
private int mCurItem; // Index of currently displayed page.
|
private int mRestoredCurItem = -1;
|
private Parcelable mRestoredAdapterState = null;
|
private ClassLoader mRestoredClassLoader = null;
|
private Scroller mScroller;
|
private VPagerObserver mObserver;
|
|
private int mChildWidthMeasureSpec;
|
private int mChildHeightMeasureSpec;
|
private boolean mInLayout;
|
|
private boolean mScrollingCacheEnabled;
|
|
private boolean mPopulatePending;
|
private boolean mScrolling;
|
|
private boolean mIsBeingDragged;
|
private boolean mIsUnableToDrag;
|
private int mTouchSlop;
|
private float mInitialMotion;
|
/**
|
* Position of the last motion event.
|
*/
|
private float mLastMotionX;
|
private float mLastMotionY;
|
private int mOrientation = HORIZONTAL;
|
/**
|
* ID of the active pointer. This is used to retain consistency during
|
* drags/flings if multiple pointers are used.
|
*/
|
private int mActivePointerId = INVALID_POINTER;
|
/**
|
* Sentinel value for no current active pointer.
|
* Used by {@link #mActivePointerId}.
|
*/
|
private static final int INVALID_POINTER = -1;
|
|
/**
|
* Determines speed during touch scrolling
|
*/
|
private VelocityTracker mVelocityTracker;
|
private int mMinimumVelocity;
|
private int mMaximumVelocity;
|
|
private OnPageChangeListener mOnPageChangeListener;
|
|
private int mScrollState = SCROLL_STATE_IDLE;
|
|
public DirectionalViewPager(Context context) {
|
super(context);
|
initDirectionalViewPager();
|
}
|
|
public DirectionalViewPager(Context context, AttributeSet attrs) {
|
super(context, attrs);
|
initDirectionalViewPager();
|
|
//We default to horizontal, only change if a value is explicitly specified
|
int orientation = attrs.getAttributeIntValue(XML_NS, "orientation", -1);
|
if (orientation != -1) {
|
setOrientation(orientation);
|
}
|
}
|
|
void initDirectionalViewPager() {
|
setWillNotDraw(false);
|
mScroller = new Scroller(getContext());
|
final ViewConfiguration configuration = ViewConfiguration.get(getContext());
|
mTouchSlop = ViewConfigurationCompat.getScaledPagingTouchSlop(configuration);
|
mMinimumVelocity = configuration.getScaledMinimumFlingVelocity();
|
mMaximumVelocity = configuration.getScaledMaximumFlingVelocity();
|
}
|
|
private void setScrollState(int newState) {
|
if (mScrollState == newState) {
|
return;
|
}
|
|
mScrollState = newState;
|
if (mOnPageChangeListener != null) {
|
mOnPageChangeListener.onPageScrollStateChanged(newState);
|
}
|
}
|
|
public void setAdapter(PagerAdapter adapter) {
|
if (mAdapter != null) {
|
mAdapter.unregisterDataSetObserver(mObserver);
|
}
|
|
mAdapter = adapter;
|
|
if (mAdapter != null) {
|
if (mObserver == null) {
|
mObserver = new VPagerObserver();
|
}
|
mAdapter.registerDataSetObserver(mObserver);
|
mPopulatePending = false;
|
if (mRestoredCurItem >= 0) {
|
mAdapter.restoreState(mRestoredAdapterState, mRestoredClassLoader);
|
setCurrentItemInternal(mRestoredCurItem, false, true);
|
mRestoredCurItem = -1;
|
mRestoredAdapterState = null;
|
mRestoredClassLoader = null;
|
} else {
|
populate();
|
}
|
}
|
}
|
|
public PagerAdapter getAdapter() {
|
return mAdapter;
|
}
|
|
public void setCurrentItem(int item) {
|
mPopulatePending = false;
|
setCurrentItemInternal(item, true, false);
|
}
|
|
void setCurrentItemInternal(int item, boolean smoothScroll, boolean always) {
|
if (mAdapter == null || mAdapter.getCount() <= 0) {
|
setScrollingCacheEnabled(false);
|
return;
|
}
|
if (!always && mCurItem == item && mItems.size() != 0) {
|
setScrollingCacheEnabled(false);
|
return;
|
}
|
if (item < 0) {
|
item = 0;
|
} else if (item >= mAdapter.getCount()) {
|
item = mAdapter.getCount() - 1;
|
}
|
if (item > (mCurItem + 1) || item < (mCurItem - 1)) {
|
// We are doing a jump by more than one page. To avoid
|
// glitches, we want to keep all current pages in the view
|
// until the scroll ends.
|
for (int i = 0; i < mItems.size(); i++) {
|
mItems.get(i).scrolling = true;
|
}
|
}
|
final boolean dispatchSelected = mCurItem != item;
|
mCurItem = item;
|
populate();
|
if (smoothScroll) {
|
if (mOrientation == HORIZONTAL) {
|
smoothScrollTo(getWidth() * item, 0);
|
} else {
|
smoothScrollTo(0, getHeight() * item);
|
}
|
if (dispatchSelected && mOnPageChangeListener != null) {
|
mOnPageChangeListener.onPageSelected(item);
|
}
|
} else {
|
if (dispatchSelected && mOnPageChangeListener != null) {
|
mOnPageChangeListener.onPageSelected(item);
|
}
|
completeScroll();
|
if (mOrientation == HORIZONTAL) {
|
scrollTo(getWidth() * item, 0);
|
} else {
|
scrollTo(0, getHeight() * item);
|
}
|
}
|
}
|
|
public void setOnPageChangeListener(OnPageChangeListener listener) {
|
mOnPageChangeListener = listener;
|
}
|
|
/**
|
* Like {@link android.view.View#scrollBy}, but scroll smoothly instead of immediately.
|
*
|
* @param x the number of pixels to scroll by on the X axis
|
* @param y the number of pixels to scroll by on the Y axis
|
*/
|
void smoothScrollTo(int x, int y) {
|
if (getChildCount() == 0) {
|
// Nothing to do.
|
setScrollingCacheEnabled(false);
|
return;
|
}
|
int sx = getScrollX();
|
int sy = getScrollY();
|
int dx = x - sx;
|
int dy = y - sy;
|
if (dx == 0 && dy == 0) {
|
completeScroll();
|
return;
|
}
|
|
setScrollingCacheEnabled(true);
|
mScrolling = true;
|
setScrollState(SCROLL_STATE_SETTLING);
|
mScroller.startScroll(sx, sy, dx, dy);
|
invalidate();
|
}
|
|
void addNewItem(int position, int index) {
|
ItemInfo ii = new ItemInfo();
|
ii.position = position;
|
ii.object = mAdapter.instantiateItem(this, position);
|
if (index < 0) {
|
mItems.add(ii);
|
} else {
|
mItems.add(index, ii);
|
}
|
}
|
|
void dataSetChanged() {
|
// This method only gets called if our observer is attached, so mAdapter is non-null.
|
|
boolean needPopulate = mItems.isEmpty() && mAdapter.getCount() > 0;
|
int newCurrItem = -1;
|
|
for (int i = 0; i < mItems.size(); i++) {
|
final ItemInfo ii = mItems.get(i);
|
final int newPos = mAdapter.getItemPosition(ii.object);
|
|
if (newPos == PagerAdapter.POSITION_UNCHANGED) {
|
continue;
|
}
|
|
if (newPos == PagerAdapter.POSITION_NONE) {
|
mItems.remove(i);
|
i--;
|
mAdapter.destroyItem(this, ii.position, ii.object);
|
needPopulate = true;
|
|
if (mCurItem == ii.position) {
|
// Keep the current item in the valid range
|
newCurrItem = Math.max(0, Math.min(mCurItem, mAdapter.getCount() - 1));
|
}
|
continue;
|
}
|
|
if (ii.position != newPos) {
|
if (ii.position == mCurItem) {
|
// Our current item changed position. Follow it.
|
newCurrItem = newPos;
|
}
|
|
ii.position = newPos;
|
needPopulate = true;
|
}
|
}
|
|
if (newCurrItem >= 0) {
|
// TODO This currently causes a jump.
|
setCurrentItemInternal(newCurrItem, false, true);
|
needPopulate = true;
|
}
|
if (needPopulate) {
|
populate();
|
requestLayout();
|
}
|
}
|
|
void populate() {
|
if (mAdapter == null) {
|
return;
|
}
|
|
// Bail now if we are waiting to populate. This is to hold off
|
// on creating views from the time the user releases their finger to
|
// fling to a new position until we have finished the scroll to
|
// that position, avoiding glitches from happening at that point.
|
if (mPopulatePending) {
|
if (DEBUG) Log.i(TAG, "populate is pending, skipping for now...");
|
return;
|
}
|
|
// Also, don't populate until we are attached to a window. This is to
|
// avoid trying to populate before we have restored our view hierarchy
|
// state and conflicting with what is restored.
|
if (getWindowToken() == null) {
|
return;
|
}
|
|
mAdapter.startUpdate(this);
|
|
final int startPos = mCurItem > 0 ? mCurItem - 1 : mCurItem;
|
final int count = mAdapter.getCount();
|
final int endPos = mCurItem < (count - 1) ? mCurItem + 1 : count - 1;
|
|
if (DEBUG) Log.v(TAG, "populating: startPos=" + startPos + " endPos=" + endPos);
|
|
// Add and remove pages in the existing list.
|
int lastPos = -1;
|
for (int i = 0; i < mItems.size(); i++) {
|
ItemInfo ii = mItems.get(i);
|
if ((ii.position < startPos || ii.position > endPos) && !ii.scrolling) {
|
if (DEBUG) Log.i(TAG, "removing: " + ii.position + " @ " + i);
|
mItems.remove(i);
|
i--;
|
mAdapter.destroyItem(this, ii.position, ii.object);
|
} else if (lastPos < endPos && ii.position > startPos) {
|
// The next item is outside of our range, but we have a gap
|
// between it and the last item where we want to have a page
|
// shown. Fill in the gap.
|
lastPos++;
|
if (lastPos < startPos) {
|
lastPos = startPos;
|
}
|
while (lastPos <= endPos && lastPos < ii.position) {
|
if (DEBUG) Log.i(TAG, "inserting: " + lastPos + " @ " + i);
|
addNewItem(lastPos, i);
|
lastPos++;
|
i++;
|
}
|
}
|
lastPos = ii.position;
|
}
|
|
// Add any new pages we need at the end.
|
lastPos = mItems.size() > 0 ? mItems.get(mItems.size() - 1).position : -1;
|
if (lastPos < endPos) {
|
lastPos++;
|
lastPos = lastPos > startPos ? lastPos : startPos;
|
while (lastPos <= endPos) {
|
if (DEBUG) Log.i(TAG, "appending: " + lastPos);
|
addNewItem(lastPos, -1);
|
lastPos++;
|
}
|
}
|
|
if (DEBUG) {
|
Log.i(TAG, "Current page list:");
|
for (int i = 0; i < mItems.size(); i++) {
|
Log.i(TAG, "#" + i + ": page " + mItems.get(i).position);
|
}
|
}
|
|
// mAdapter.setPrimaryItem(this,mCurItem,);
|
mAdapter.finishUpdate(this);
|
}
|
|
public static class SavedState extends BaseSavedState {
|
int position;
|
Parcelable adapterState;
|
ClassLoader loader;
|
|
public SavedState(Parcelable superState) {
|
super(superState);
|
}
|
|
@Override
|
public void writeToParcel(Parcel out, int flags) {
|
super.writeToParcel(out, flags);
|
out.writeInt(position);
|
out.writeParcelable(adapterState, flags);
|
}
|
|
@Override
|
public String toString() {
|
return "FragmentPager.SavedState{"
|
+ Integer.toHexString(System.identityHashCode(this))
|
+ " position=" + position + "}";
|
}
|
|
public static final Creator<SavedState> CREATOR
|
= ParcelableCompat.newCreator(new ParcelableCompatCreatorCallbacks<SavedState>() {
|
@Override
|
public SavedState createFromParcel(Parcel in, ClassLoader loader) {
|
return new SavedState(in, loader);
|
}
|
|
@Override
|
public SavedState[] newArray(int size) {
|
return new SavedState[size];
|
}
|
});
|
|
SavedState(Parcel in, ClassLoader loader) {
|
super(in);
|
if (loader == null) {
|
loader = getClass().getClassLoader();
|
}
|
position = in.readInt();
|
adapterState = in.readParcelable(loader);
|
this.loader = loader;
|
}
|
}
|
|
@Override
|
public Parcelable onSaveInstanceState() {
|
Parcelable superState = super.onSaveInstanceState();
|
SavedState ss = new SavedState(superState);
|
ss.position = mCurItem;
|
ss.adapterState = mAdapter.saveState();
|
return ss;
|
}
|
|
@Override
|
public void onRestoreInstanceState(Parcelable state) {
|
if (!(state instanceof SavedState)) {
|
super.onRestoreInstanceState(state);
|
return;
|
}
|
|
SavedState ss = (SavedState) state;
|
super.onRestoreInstanceState(ss.getSuperState());
|
|
if (mAdapter != null) {
|
mAdapter.restoreState(ss.adapterState, ss.loader);
|
setCurrentItemInternal(ss.position, false, true);
|
} else {
|
mRestoredCurItem = ss.position;
|
mRestoredAdapterState = ss.adapterState;
|
mRestoredClassLoader = ss.loader;
|
}
|
}
|
|
public int getOrientation() {
|
return mOrientation;
|
}
|
|
public void setOrientation(int orientation) {
|
switch (orientation) {
|
case HORIZONTAL:
|
case VERTICAL:
|
break;
|
|
default:
|
throw new IllegalArgumentException("Only HORIZONTAL and VERTICAL are valid orientations.");
|
}
|
|
if (orientation == mOrientation) {
|
return;
|
}
|
|
//Complete any scroll we are currently in the middle of
|
completeScroll();
|
|
//Reset values
|
mInitialMotion = 0;
|
mLastMotionX = 0;
|
mLastMotionY = 0;
|
if (mVelocityTracker != null) {
|
mVelocityTracker.clear();
|
}
|
|
//Adjust scroll for new orientation
|
mOrientation = orientation;
|
if (mOrientation == HORIZONTAL) {
|
scrollTo(mCurItem * getWidth(), 0);
|
} else {
|
scrollTo(0, mCurItem * getHeight());
|
}
|
requestLayout();
|
}
|
|
@Override
|
public void addView(View child, int index, ViewGroup.LayoutParams params) {
|
if (mInLayout) {
|
addViewInLayout(child, index, params);
|
child.measure(mChildWidthMeasureSpec, mChildHeightMeasureSpec);
|
} else {
|
super.addView(child, index, params);
|
}
|
|
if (USE_CACHE) {
|
if (child.getVisibility() != GONE) {
|
child.setDrawingCacheEnabled(mScrollingCacheEnabled);
|
} else {
|
child.setDrawingCacheEnabled(false);
|
}
|
}
|
}
|
|
ItemInfo infoForChild(View child) {
|
for (int i = 0; i < mItems.size(); i++) {
|
ItemInfo ii = mItems.get(i);
|
if (mAdapter.isViewFromObject(child, ii.object)) {
|
return ii;
|
}
|
}
|
return null;
|
}
|
|
@Override
|
protected void onAttachedToWindow() {
|
super.onAttachedToWindow();
|
if (mAdapter != null) {
|
populate();
|
}
|
}
|
|
@Override
|
protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
|
// For simple implementation, or internal size is always 0.
|
// We depend on the container to specify the layout size of
|
// our view. We can't really know what it is since we will be
|
// adding and removing different arbitrary views and do not
|
// want the layout to change as this happens.
|
setMeasuredDimension(getDefaultSize(0, widthMeasureSpec),
|
getDefaultSize(0, heightMeasureSpec));
|
|
// Children are just made to fill our space.
|
mChildWidthMeasureSpec = MeasureSpec.makeMeasureSpec(getMeasuredWidth() -
|
getPaddingLeft() - getPaddingRight(), MeasureSpec.EXACTLY);
|
mChildHeightMeasureSpec = MeasureSpec.makeMeasureSpec(getMeasuredHeight() -
|
getPaddingTop() - getPaddingBottom(), MeasureSpec.EXACTLY);
|
|
// Make sure we have created all fragments that we need to have shown.
|
mInLayout = true;
|
populate();
|
mInLayout = false;
|
|
// Make sure all children have been properly measured.
|
final int size = getChildCount();
|
for (int i = 0; i < size; ++i) {
|
final View child = getChildAt(i);
|
if (child.getVisibility() != GONE) {
|
if (DEBUG) Log.v(TAG, "Measuring #" + i + " " + child
|
+ ": " + mChildWidthMeasureSpec + " x " + mChildHeightMeasureSpec);
|
child.measure(mChildWidthMeasureSpec, mChildHeightMeasureSpec);
|
}
|
}
|
}
|
|
@Override
|
protected void onSizeChanged(int w, int h, int oldw, int oldh) {
|
super.onSizeChanged(w, h, oldw, oldh);
|
|
// Make sure scroll position is set correctly.
|
if (mOrientation == HORIZONTAL) {
|
int scrollPos = mCurItem * w;
|
if (scrollPos != getScrollX()) {
|
completeScroll();
|
scrollTo(scrollPos, getScrollY());
|
}
|
} else {
|
int scrollPos = mCurItem * h;
|
if (scrollPos != getScrollY()) {
|
completeScroll();
|
scrollTo(getScrollX(), scrollPos);
|
}
|
}
|
}
|
|
@Override
|
protected void onLayout(boolean changed, int l, int t, int r, int b) {
|
mInLayout = true;
|
populate();
|
mInLayout = false;
|
|
final int count = getChildCount();
|
final int size = (mOrientation == HORIZONTAL) ? r - l : b - t;
|
|
for (int i = 0; i < count; i++) {
|
View child = getChildAt(i);
|
ItemInfo ii;
|
if (child.getVisibility() != GONE && (ii = infoForChild(child)) != null) {
|
int off = size * ii.position;
|
int childLeft = getPaddingLeft();
|
int childTop = getPaddingTop();
|
if (mOrientation == HORIZONTAL) {
|
childLeft += off;
|
} else {
|
childTop += off;
|
}
|
if (DEBUG) Log.v(TAG, "Positioning #" + i + " " + child + " f=" + ii.object
|
+ ":" + childLeft + "," + childTop + " " + child.getMeasuredWidth()
|
+ "x" + child.getMeasuredHeight());
|
child.layout(childLeft, childTop,
|
childLeft + child.getMeasuredWidth(),
|
childTop + child.getMeasuredHeight());
|
}
|
}
|
}
|
|
@Override
|
public void computeScroll() {
|
if (DEBUG) Log.i(TAG, "computeScroll: finished=" + mScroller.isFinished());
|
if (!mScroller.isFinished()) {
|
if (mScroller.computeScrollOffset()) {
|
if (DEBUG) Log.i(TAG, "computeScroll: still scrolling");
|
int oldX = getScrollX();
|
int oldY = getScrollY();
|
int x = mScroller.getCurrX();
|
int y = mScroller.getCurrY();
|
|
if (oldX != x || oldY != y) {
|
scrollTo(x, y);
|
}
|
|
if (mOnPageChangeListener != null) {
|
int size;
|
int value;
|
if (mOrientation == HORIZONTAL) {
|
size = getWidth();
|
value = x;
|
} else {
|
size = getHeight();
|
value = y;
|
}
|
|
final int position = value / size;
|
final int offsetPixels = value % size;
|
final float offset = (float) offsetPixels / size;
|
mOnPageChangeListener.onPageScrolled(position, offset, offsetPixels);
|
}
|
|
// Keep on drawing until the animation has finished.
|
invalidate();
|
return;
|
}
|
}
|
|
// Done with scroll, clean up state.
|
completeScroll();
|
}
|
|
private void completeScroll() {
|
boolean needPopulate;
|
if ((needPopulate = mScrolling)) {
|
// Done with scroll, no longer want to cache view drawing.
|
setScrollingCacheEnabled(false);
|
mScroller.abortAnimation();
|
int oldX = getScrollX();
|
int oldY = getScrollY();
|
int x = mScroller.getCurrX();
|
int y = mScroller.getCurrY();
|
if (oldX != x || oldY != y) {
|
scrollTo(x, y);
|
}
|
setScrollState(SCROLL_STATE_IDLE);
|
}
|
mPopulatePending = false;
|
mScrolling = false;
|
for (int i = 0; i < mItems.size(); i++) {
|
ItemInfo ii = mItems.get(i);
|
if (ii.scrolling) {
|
needPopulate = true;
|
ii.scrolling = false;
|
}
|
}
|
if (needPopulate) {
|
populate();
|
}
|
}
|
|
@Override
|
public boolean onInterceptTouchEvent(MotionEvent ev) {
|
/*
|
* This method JUST determines whether we want to intercept the motion.
|
* If we return true, onMotionEvent will be called and we do the actual
|
* scrolling there.
|
*/
|
|
final int action = ev.getAction() & MotionEventCompat.ACTION_MASK;
|
|
// Always take care of the touch gesture being complete.
|
if (action == MotionEvent.ACTION_CANCEL || action == MotionEvent.ACTION_UP) {
|
// Release the drag.
|
if (DEBUG) Log.v(TAG, "Intercept done!");
|
mIsBeingDragged = false;
|
mIsUnableToDrag = false;
|
mActivePointerId = INVALID_POINTER;
|
return false;
|
}
|
|
// Nothing more to do here if we have decided whether or not we
|
// are dragging.
|
if (action != MotionEvent.ACTION_DOWN) {
|
if (mIsBeingDragged) {
|
if (DEBUG) Log.v(TAG, "Intercept returning true!");
|
return true;
|
}
|
if (mIsUnableToDrag) {
|
if (DEBUG) Log.v(TAG, "Intercept returning false!");
|
return false;
|
}
|
}
|
|
switch (action) {
|
case MotionEvent.ACTION_MOVE: {
|
/*
|
* mIsBeingDragged == false, otherwise the shortcut would have caught it. Check
|
* whether the user has moved far enough from his original down touch.
|
*/
|
|
/*
|
* Locally do absolute value. mLastMotionY is set to the y value
|
* of the down event.
|
*/
|
final int activePointerId = mActivePointerId;
|
if (activePointerId == INVALID_POINTER && Build.VERSION.SDK_INT > Build.VERSION_CODES.DONUT) {
|
// If we don't have a valid id, the touch down wasn't on content.
|
break;
|
}
|
|
final int pointerIndex = MotionEventCompat.findPointerIndex(ev, activePointerId);
|
final float x = MotionEventCompat.getX(ev, pointerIndex);
|
final float y = MotionEventCompat.getY(ev, pointerIndex);
|
final float xDiff = Math.abs(x - mLastMotionX);
|
final float yDiff = Math.abs(y - mLastMotionY);
|
float primaryDiff;
|
float secondaryDiff;
|
|
if (mOrientation == HORIZONTAL) {
|
primaryDiff = xDiff;
|
secondaryDiff = yDiff;
|
} else {
|
primaryDiff = yDiff;
|
secondaryDiff = xDiff;
|
}
|
|
|
if (DEBUG) Log.v(TAG, "Moved x to " + x + "," + y + " diff=" + xDiff + "," + yDiff);
|
|
if (primaryDiff > mTouchSlop && primaryDiff > secondaryDiff) {
|
if (DEBUG) Log.v(TAG, "Starting drag!");
|
mIsBeingDragged = true;
|
setScrollState(SCROLL_STATE_DRAGGING);
|
if (mOrientation == HORIZONTAL) {
|
mLastMotionX = x;
|
} else {
|
mLastMotionY = y;
|
}
|
setScrollingCacheEnabled(true);
|
} else {
|
if (secondaryDiff > mTouchSlop) {
|
// The finger has moved enough in the vertical
|
// direction to be counted as a drag... abort
|
// any attempt to drag horizontally, to work correctly
|
// with children that have scrolling containers.
|
if (DEBUG) Log.v(TAG, "Starting unable to drag!");
|
mIsUnableToDrag = true;
|
}
|
}
|
break;
|
}
|
|
case MotionEvent.ACTION_DOWN: {
|
/*
|
* Remember location of down touch.
|
* ACTION_DOWN always refers to pointer index 0.
|
*/
|
if (mOrientation == HORIZONTAL) {
|
mLastMotionX = mInitialMotion = ev.getX();
|
mLastMotionY = ev.getY();
|
} else {
|
mLastMotionX = ev.getX();
|
mLastMotionY = mInitialMotion = ev.getY();
|
}
|
mActivePointerId = MotionEventCompat.getPointerId(ev, 0);
|
|
if (mScrollState == SCROLL_STATE_SETTLING) {
|
// Let the user 'catch' the pager as it animates.
|
mIsBeingDragged = true;
|
mIsUnableToDrag = false;
|
setScrollState(SCROLL_STATE_DRAGGING);
|
} else {
|
completeScroll();
|
mIsBeingDragged = false;
|
mIsUnableToDrag = false;
|
}
|
|
if (DEBUG) Log.v(TAG, "Down at " + mLastMotionX + "," + mLastMotionY
|
+ " mIsBeingDragged=" + mIsBeingDragged
|
+ "mIsUnableToDrag=" + mIsUnableToDrag);
|
break;
|
}
|
|
case MotionEventCompat.ACTION_POINTER_UP:
|
onSecondaryPointerUp(ev);
|
break;
|
}
|
|
/*
|
* The only time we want to intercept motion events is if we are in the
|
* drag mode.
|
*/
|
return mIsBeingDragged;
|
}
|
|
@Override
|
public boolean onTouchEvent(MotionEvent ev) {
|
|
if (ev.getAction() == MotionEvent.ACTION_DOWN && ev.getEdgeFlags() != 0) {
|
// Don't handle edge touches immediately -- they may actually belong to one of our
|
// descendants.
|
return false;
|
}
|
|
if (mAdapter == null || mAdapter.getCount() == 0) {
|
// Nothing to present or scroll; nothing to touch.
|
return false;
|
}
|
|
if (mVelocityTracker == null) {
|
mVelocityTracker = VelocityTracker.obtain();
|
}
|
mVelocityTracker.addMovement(ev);
|
|
final int action = ev.getAction();
|
|
switch (action & MotionEventCompat.ACTION_MASK) {
|
case MotionEvent.ACTION_DOWN: {
|
/*
|
* If being flinged and user touches, stop the fling. isFinished
|
* will be false if being flinged.
|
*/
|
completeScroll();
|
|
// Remember where the motion event started
|
if (mOrientation == HORIZONTAL) {
|
mLastMotionX = mInitialMotion = ev.getX();
|
} else {
|
mLastMotionY = mInitialMotion = ev.getY();
|
}
|
mActivePointerId = MotionEventCompat.getPointerId(ev, 0);
|
break;
|
}
|
case MotionEvent.ACTION_MOVE:
|
if (!mIsBeingDragged) {
|
final int pointerIndex = MotionEventCompat.findPointerIndex(ev, mActivePointerId);
|
final float x = MotionEventCompat.getX(ev, pointerIndex);
|
final float y = MotionEventCompat.getY(ev, pointerIndex);
|
final float xDiff = Math.abs(x - mLastMotionX);
|
final float yDiff = Math.abs(y - mLastMotionY);
|
float primaryDiff;
|
float secondaryDiff;
|
|
if (mOrientation == HORIZONTAL) {
|
primaryDiff = xDiff;
|
secondaryDiff = yDiff;
|
} else {
|
primaryDiff = yDiff;
|
secondaryDiff = xDiff;
|
}
|
|
|
if (DEBUG)
|
Log.v(TAG, "Moved x to " + x + "," + y + " diff=" + xDiff + "," + yDiff);
|
if (primaryDiff > mTouchSlop && primaryDiff > secondaryDiff) {
|
if (DEBUG) Log.v(TAG, "Starting drag!");
|
mIsBeingDragged = true;
|
if (mOrientation == HORIZONTAL) {
|
mLastMotionX = x;
|
} else {
|
mLastMotionY = y;
|
}
|
setScrollState(SCROLL_STATE_DRAGGING);
|
setScrollingCacheEnabled(true);
|
}
|
}
|
if (mIsBeingDragged) {
|
// Scroll to follow the motion event
|
final int activePointerIndex = MotionEventCompat.findPointerIndex(
|
ev, mActivePointerId);
|
final float x = MotionEventCompat.getX(ev, activePointerIndex);
|
final float y = MotionEventCompat.getY(ev, activePointerIndex);
|
|
int size;
|
float scroll;
|
|
if (mOrientation == HORIZONTAL) {
|
size = getWidth();
|
scroll = getScrollX() + (mLastMotionX - x);
|
mLastMotionX = x;
|
} else {
|
size = getHeight();
|
scroll = getScrollY() + (mLastMotionY - y);
|
mLastMotionY = y;
|
}
|
|
final float lowerBound = Math.max(0, (mCurItem - 1) * size);
|
final float upperBound =
|
Math.min(mCurItem + 1, mAdapter.getCount() - 1) * size;
|
if (scroll < lowerBound) {
|
scroll = lowerBound;
|
} else if (scroll > upperBound) {
|
scroll = upperBound;
|
}
|
if (mOrientation == HORIZONTAL) {
|
// Don't lose the rounded component
|
mLastMotionX += scroll - (int) scroll;
|
scrollTo((int) scroll, getScrollY());
|
} else {
|
// Don't lose the rounded component
|
mLastMotionY += scroll - (int) scroll;
|
scrollTo(getScrollX(), (int) scroll);
|
}
|
if (mOnPageChangeListener != null) {
|
final int position = (int) scroll / size;
|
final int positionOffsetPixels = (int) scroll % size;
|
final float positionOffset = (float) positionOffsetPixels / size;
|
mOnPageChangeListener.onPageScrolled(position, positionOffset,
|
positionOffsetPixels);
|
}
|
}
|
break;
|
case MotionEvent.ACTION_UP:
|
if (mIsBeingDragged) {
|
final VelocityTracker velocityTracker = mVelocityTracker;
|
velocityTracker.computeCurrentVelocity(1000, mMaximumVelocity);
|
int initialVelocity;
|
float lastMotion;
|
int sizeOverThree;
|
|
if (mOrientation == HORIZONTAL) {
|
initialVelocity = (int) VelocityTrackerCompat.getXVelocity(
|
velocityTracker, mActivePointerId);
|
lastMotion = mLastMotionX;
|
sizeOverThree = getWidth() / 3;
|
} else {
|
initialVelocity = (int) VelocityTrackerCompat.getYVelocity(
|
velocityTracker, mActivePointerId);
|
lastMotion = mLastMotionY;
|
sizeOverThree = getHeight() / 3;
|
}
|
|
mPopulatePending = true;
|
if ((Math.abs(initialVelocity) > mMinimumVelocity)
|
|| Math.abs(mInitialMotion - lastMotion) >= sizeOverThree) {
|
if (lastMotion > mInitialMotion) {
|
setCurrentItemInternal(mCurItem - 1, true, true);
|
} else {
|
setCurrentItemInternal(mCurItem + 1, true, true);
|
}
|
} else {
|
setCurrentItemInternal(mCurItem, true, true);
|
}
|
|
mActivePointerId = INVALID_POINTER;
|
endDrag();
|
}
|
break;
|
case MotionEvent.ACTION_CANCEL:
|
if (mIsBeingDragged) {
|
setCurrentItemInternal(mCurItem, true, true);
|
mActivePointerId = INVALID_POINTER;
|
endDrag();
|
}
|
break;
|
case MotionEventCompat.ACTION_POINTER_DOWN: {
|
final int index = MotionEventCompat.getActionIndex(ev);
|
if (mOrientation == HORIZONTAL) {
|
mLastMotionX = MotionEventCompat.getX(ev, index);
|
} else {
|
mLastMotionY = MotionEventCompat.getY(ev, index);
|
}
|
mActivePointerId = MotionEventCompat.getPointerId(ev, index);
|
break;
|
}
|
case MotionEventCompat.ACTION_POINTER_UP:
|
onSecondaryPointerUp(ev);
|
final int index = MotionEventCompat.findPointerIndex(ev, mActivePointerId);
|
if (mOrientation == HORIZONTAL) {
|
mLastMotionX = MotionEventCompat.getX(ev, index);
|
} else {
|
mLastMotionY = MotionEventCompat.getY(ev, index);
|
}
|
break;
|
}
|
return true;
|
}
|
|
private void onSecondaryPointerUp(MotionEvent ev) {
|
final int pointerIndex = MotionEventCompat.getActionIndex(ev);
|
final int pointerId = MotionEventCompat.getPointerId(ev, pointerIndex);
|
if (pointerId == mActivePointerId) {
|
// This was our active pointer going up. Choose a new
|
// active pointer and adjust accordingly.
|
final int newPointerIndex = pointerIndex == 0 ? 1 : 0;
|
if (mOrientation == HORIZONTAL) {
|
mLastMotionX = MotionEventCompat.getX(ev, newPointerIndex);
|
} else {
|
mLastMotionY = MotionEventCompat.getY(ev, newPointerIndex);
|
}
|
mActivePointerId = MotionEventCompat.getPointerId(ev, newPointerIndex);
|
if (mVelocityTracker != null) {
|
mVelocityTracker.clear();
|
}
|
}
|
}
|
|
private void endDrag() {
|
mIsBeingDragged = false;
|
mIsUnableToDrag = false;
|
|
if (mVelocityTracker != null) {
|
mVelocityTracker.recycle();
|
mVelocityTracker = null;
|
}
|
}
|
|
private void setScrollingCacheEnabled(boolean enabled) {
|
if (mScrollingCacheEnabled != enabled) {
|
mScrollingCacheEnabled = enabled;
|
if (USE_CACHE) {
|
final int size = getChildCount();
|
for (int i = 0; i < size; ++i) {
|
final View child = getChildAt(i);
|
if (child.getVisibility() != GONE) {
|
child.setDrawingCacheEnabled(enabled);
|
}
|
}
|
}
|
}
|
}
|
|
private class VPagerObserver extends DataSetObserver {
|
@Override
|
public void onChanged() {
|
dataSetChanged();
|
}
|
|
@Override
|
public void onInvalidated() {
|
dataSetChanged();
|
}
|
}
|
}
|