- Scroller是什么
Scroller就是一个滑动帮助类。它并不可以使View真正的滑动,而是配合scrollTo/ScrollBy让view产生缓慢的滑动,产生动画的效果,其实和属性动画是同一个原理。在我看来,Scroller跟属性动画的平移的效果是一样的。 - 如何使用
Scroller mScroller=new Scroller(context);
mScrooler.startScroll(int startX,int startY,int endx,int endY,int duration);
invalidate();
@Override
public void computeScroll() {
if (mScroller.computeScrollOffset()) {
scrollTo(mScroller.getCurrX(), mScroller.getCurrY());
postInvalidate();
}
super.computeScroll();
}
- 使用示例:
package com.liujian.chart;
/**
* Scroller练习,一个简单的ViewPager
* @author : liujian
* @since : 2017/12/17
*/
public class ScrollLayout extends ViewGroup {
private Scroller mScroller;
private int mTouchSlop;
private int leftBorder;
private int rightBorder;
private float mRawXDown;
private float mRawXMove;
private float mRawXLastMove;
public ScrollLayout(Context context) {
super(context);
initView(context);
}
public ScrollLayout(Context context, @Nullable AttributeSet attrs) {
super(context, attrs);
initView(context);
}
public ScrollLayout(Context context, @Nullable AttributeSet attrs, int defStyleAttr) {
super(context, attrs, defStyleAttr);
initView(context);
}
private void initView(Context context) {
mTouchSlop = ViewConfiguration.get(getContext()).getScaledTouchSlop();
mScroller = new Scroller(getContext());
}
@Override
protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
super.onMeasure(widthMeasureSpec, heightMeasureSpec);
measureChildren(widthMeasureSpec, heightMeasureSpec);
}
@Override
protected void onLayout(boolean changed, int l, int t, int r, int b) {
int childCount = getChildCount();
for (int i = 0; i < childCount; i++) {
View view = getChildAt(i);
view.layout(i * view.getMeasuredWidth(), 0, (i + 1) * view.getMeasuredWidth(), view.getMeasuredHeight());
}
leftBorder = getChildAt(0).getLeft();
rightBorder = getChildAt(getChildCount() - 1).getRight();
}
@Override
public boolean onInterceptTouchEvent(MotionEvent ev) {
switch (ev.getAction()) {
case MotionEvent.ACTION_DOWN:
mRawXDown = ev.getRawX();
mRawXLastMove = mRawXDown;
break;
case MotionEvent.ACTION_MOVE:
mRawXMove = ev.getRawX();
mRawXLastMove = mRawXMove;
float distance = Math.abs(mRawXMove - mRawXDown);
if (distance > mTouchSlop) {
return true;
}
break;
case MotionEvent.ACTION_UP:
break;
}
return super.onInterceptTouchEvent(ev);
}
@SuppressLint("ClickableViewAccessibility")
@Override
public boolean onTouchEvent(MotionEvent event) {
switch (event.getAction()) {
case MotionEvent.ACTION_MOVE:
mRawXMove = event.getRawX();
int distanceX = (int) (mRawXLastMove - mRawXMove);
if (getScrollX() + distanceX < leftBorder) {
scrollBy(leftBorder, 0);
}
if (getScrollX() + getWidth() + distanceX > rightBorder) {
scrollBy(rightBorder - getWidth(), 0);
}
scrollBy(distanceX, 0);
mRawXLastMove = mRawXMove;
break;
case MotionEvent.ACTION_UP:
int targetIndex = (getScrollX() + getWidth() / 2) / getWidth();
int dx = targetIndex * getWidth() - getScrollX();
Log.i("TAG", "dx: " + dx);
Log.i("TAG", "getScrollX: " + getScrollX());
Log.i("TAG", "getWidth: " + getWidth());
mScroller.startScroll(getScrollX(), 0, dx, 0);
invalidate();
break;
}
return super.onTouchEvent(event);
}
@Override
public void computeScroll() {
if (mScroller.computeScrollOffset()) {
scrollTo(mScroller.getCurrX(), mScroller.getCurrY());
invalidate();
}
}
}