项目中有个需求,就是在RecyclerView的item中进行侧滑,一开始同事推荐了一个开源库,使用起来确实也方便好用,直接在布局作为父布局即可实现侧滑。
自己也非常好奇这个开源库到底用了什么API能实现这种效果,于是慢慢开始了源码的阅读。
Scroller:
这个类是原生提供的滚动辅助工具类,使用它可以根据开始位置和结束位置计算出当前应该滚动到的坐标。
需要注意的是:Scroller并不会帮你设置View的滚动位置,它只负责计算当前应该滚动到的位置,所以你需要手动View的滚动位置。
下面是部分源码,当快速在Item进行手势侧滑,当手指抬起时,会使用Scroller进行计算。 case MotionEvent.ACTION_UP: // ..... mScroller.startScroll(startX, 0, endX - startX, 0, 400); // ...... cancel(); break;
public void startScroll(int startX, int startY, int dx, int dy, int duration) 调用此方法,将会开启计算滚动,在滚动计算启动后,你可以在任意时刻调用computeScrollOffset()让Scrooler计算最新位置。
public boolean computeScrollOffset() 让Scrooler计算最新位置,此法返回布尔值,如果为真表示还没滚动到目标位置,否则表示已经滚动到目标位置。
public final int getCurrX() 调用computeScrollOffset()之后,可以通过此方法获取当前应该滚动到x坐标,同理也有getCurrY()。
View.computeScroll()
在了解View.computeScroll()方法前,需要了解一下View.scrollTo()和View.scrollBy()方法。
scrollTo():将View滚动到指定位置。
scrollBy():内部其实是调用了scrollTo()方法。
public void scrollTo(int x, int y) { if (mScrollX != x || mScrollY != y) { int oldX = mScrollX; int oldY = mScrollY; mScrollX = x; mScrollY = y; invalidateParentCaches(); onScrollChanged(mScrollX, mScrollY, oldX, oldY); if (!awakenScrollBars()) { postInvalidateOnAnimation(); } }}public void scrollBy(int x, int y) { scrollTo(mScrollX + x, mScrollY + y);}
这里讲解一下scrollTo()和scrollBy()方法的重要流程:scrollTo() ==> postInvalidateOnAnimation() ==> 重绘 ==> View.computeScroll()
View.computeScroll()这个方法会在重绘的时候进行调用,而scrollTo()会触发重绘操作。在View里面,computeScroll()是一个空方法。
/** * Called by a parent to request that a child update its values for mScrollX * and mScrollY if necessary. This will typically be done if the child is * animating a scroll using a { @link android.widget.Scroller Scroller} * object. */public void computeScroll() {}
系统原生说得很明白了,在computeScroll()里面我们通常是使用Scroller进行计算滚动位置,然后执行动画滚动。
// 一个例子 public void computeScroll() { // 如果还没有滚动到最后位置 if(mScroller.computeScrollOffset()){ // scrollTo()和scrollBy()方法都是没有动画效果的,是瞬间就滚动了View,所以通过Scroller不断计算滚动进行细微滚动,形成滚动效果。 scrollTo(mScroller.getCurrX() ,mScoller.getCurrY()); postInvalidate(); }}