ScrollerCompat的使用
先说点题外话:ScrollerCompat一般会在自定义View中使用,自定义view很多时候都会重写onDraw()方法,则
@Override PRotected void onDraw(Canvas canvas) { //对于canvas画板,因其是该控件本身提供的,所以canvas的坐标也是相对自己而言的 super.onDraw(canvas); }也就是说,canvas.drawCircle(CircleX,CircleY,radius,backgroundPaint);这个方法中的圆心坐标,其实是以自身的左上角的(0,0)坐标原点而确定的,而不是参照手机屏幕的坐标原点,这点需要注意下;再顺便说几个方法:
view.setLayoutParams()---重新布局;
view.invalidate(要在主线程中用);调用onDraw()方法;
view.postInvalidate(可以在子线程中使用);
view.postDelayed(Runnable action, long delayMillis);其底层也是有mHandler的 ;
view.computeScroll();
好了言归正传,ScrollerCompat其用法与Scroller用法差不多,在自定义的view中可以这样设置:
scrollerCompat = ScrollerCompat.create(context); scrollerCompat.startScroll(0, 100, 500, 8, 1000); view.postInvalidate(); 完成滚动后,ondraw内部会调用下边这个方法:@Override public void computeScroll() { super.computeScroll(); if(scrollerCompat.computeScrollOffset()) { button.layout(scrollerCompat.getCurrX(), button.getTop(), scrollerCompat.getCurrX() + button.getMeasuredWidth(), button.getBottom()); postInvalidate(); } }先举例说明下,自定义一个CustomView,使用Scroller实现滚动:
public class CustomView extends LinearLayout { private static final String TAG = "Scroller"; private Scroller mScroller; public CustomView(Context context, AttributeSet attrs) { super(context, attrs); mScroller = new Scroller(context); } //调用此方法滚动到目标位置 public void smoothScrollTo(int fx, int fy) { int dx = fx - mScroller.getFinalX(); int dy = fy - mScroller.getFinalY(); smoothScrollBy(dx, dy); } //调用此方法设置滚动的相对偏移 public void smoothScrollBy(int dx, int dy) { //设置mScroller的滚动偏移量 mScroller.startScroll(mScroller.getFinalX(), mScroller.getFinalY(), dx, dy); invalidate();//这里必须调用invalidate()才能保证computeScroll()会被调用,否则不一定会刷新界面,看不到滚动效果 } @Override public void computeScroll() { //先判断mScroller滚动是否完成 if (mScroller.computeScrollOffset()) { //这里调用View的scrollTo()完成实际的滚动 scrollTo(mScroller.getCurrX(), mScroller.getCurrY()); //必须调用该方法,否则不一定能看到滚动效果 postInvalidate(); } super.computeScroll(); } }相关API介绍如下:
mScroller.getCurrX() //获取mScroller当前水平滚动的位置 mScroller.getCurrY() //获取mScroller当前竖直滚动的位置 mScroller.getFinalX() //获取mScroller最终停止的水平位置 mScroller.getFinalY() //获取mScroller最终停止的竖直位置 mScroller.setFinalX(int newX) //设置mScroller最终停留的水平位置,没有动画效果,直接跳到目标位置 mScroller.setFinalY(int newY) //设置mScroller最终停留的竖直位置,没有动画效果,直接跳到目标位置 //滚动,startX, startY为开始滚动的位置,dx,dy为滚动的偏移量, duration为完成滚动的时间 mScroller.startScroll(int startX, int startY, int dx, int dy) //使用默认完成时间250ms mScroller.startScroll(int startX, int startY, int dx, int dy, int duration) mScroller.computeScrollOffset() //返回值为boolean,true说明滚动尚未完成,false说明滚动已经完成。这是一个很重要的方法,通常放在View.computeScroll()中,用来判断是否滚动是否结束。最后说下ScrollerCompat 的注意事项:
其可以模拟滚动,但实际上是没有让任何一个View滚动.
ScrollerCompat 一般 在View中使用.startScroll执行后, 不会发生任何 事情.注意1:startScroll 记得 调用postInvalidate这样就会调用computeScroll. 我们可以复写computeScroll,对View中的一些子View进行滚动后的布局处理.这个时候注意2:if(scrollerCompat.computeScrollOffset()) {}要等scrollerCompat计算完毕.计算完毕后,可以对子View进行滚动后布局处理.如button.layout(scrollerCompat.getCurrX(), button.getTop(), scrollerCompat.getCurrX() + button.getMeasuredWidth(), button.getBottom());注意3:处理后,记得再次调用postInvalidate,否则,View产生的滚动效果会非常的不流畅问题哦!
新闻热点
疑难解答