之前做项目时,要实现Activity从左往右滑出来的效果,那个时候我选择了属性动画,,最近了解了下Scroller,也可以用它实现Activity的滑动,接下来介绍它的用法。
Scroller并不会使View滑动,看看它的源码就会知道,Scroller是个工具类,具体点说它就是一个算法工具类,它计算出值给View使用,真正让View改变位置的是scrollTo或scrollBy。
Activity对应的layout的根视图一般是RelativeLayout或LinearLayout,我们自定义一个类继承RelativeLayout或LinearLayout,Activity的Layout的根视图就使用我们自定义的类。自定义类实现如下:
public class Sen5RelativeLayout extends RelativeLayout {
private Scroller mScroller;
public Sen5RelativeLayout(Context context, AttributeSet attrs) {
super(context, attrs);
mScroller = new Scroller(context);
}
@Override
public void computeScroll() {
super.computeScroll();
if(mScroller.computeScrollOffset()) {
scrollTo(getWidth() - mScroller.getCurrX(), mScroller.getCurrY());
invalidate();
}
}
public void start_scroll() {
mScroller.startScroll(0, 0, getWidth(), 0, 1000);
invalidate();
}
}
在Activity里onResume里调用如下的代码就可以实现activiey从左往右的滑动了。
@Override
protected void onResume() {
super.onResume();
activity_layout.post(new Runnable() {
@Override
public void run() {
activity_layout.start_scroll();
}
});
}
activity_layout就是我们自定义的view。有人会问为啥不直接到用start_scroll,而要用post的方法呢?这里有一个很重要的知识:View的Measure过程和Activity的生命周期方法不是同步执行的,所以无法保证Activity执行了onCreate()、onStart()、onResume()时某个View已经测量完毕了,如果View还没有测量完毕,那么getMeasuredWidth和getMeasuredHeight就都是0。根据view测量的原理,getWidth的值还要在onMeasure后面才能得到,如果我们直接在onResume里调用getWidth返回的也会是0,所以要用post的方法。现在来说说scrollTo函数,它的作用是把view的内容进行移动到指定的位置,是移动不是滑动,瞬间的事情。比如scrollTo(100, 0)表示把view里的内容在水平方向向左移动了100个像素,记住往左移动是正直,往右右移动是负值,往上移动是正值,往下移是负值。既然scrollTo不能实现滑动,那Scroller就该登场了,借助它可是实现view的滑动。
使用Scroller有的时候只需3步:
1、创建一个Scroller。
2、调用Scroller的startScroll,紧接着要调用invalidate,invalidate会导致view重画而调用View的draw函数,draw里又会调用
computeScroll。View里的computeScroll是一个空函数,我们需要Override这个函数,就是下面的第3步了。
3、在自定义类里重写computeScroll。在这里调用scrollTo,然后再调用invalidate使得view重画,然后又会调用computeScroll,这样反反复复就会不停的调用scrollTo,直到view的内容移动到了指定的位置。这里是借助的Scroller的computeScrollOffset来判断移动何时结束的。Scroller的源码不难,这里就不详说了。
让activity有滑动的效果,总的思想就是不停的调用scrollTo来改变view内容的偏移,每次偏移多少就是Scroller计算出来的,而且Scroller也能确保何时停止工作。Scroller停止工作了,那view的内容也就滑到了指定的位置。
THE END