文档章节

Android Scroller与computeScroll方法的调用关系

IamOkay
 IamOkay
发布于 2016/01/10 08:01
字数 1106
阅读 3444
收藏 61

Android ViewGroup中的Scroller与computeScroll的有什么关系?

答:没有直接的关系

知道了答案,是不是意味着下文就没必要看了,如果说对ViewGroup自定义控件不感兴趣,可以不用看了。


1.Scroller到底是什么?

答:Scroller只是个计算器,提供插值计算,让滚动过程具有动画属性,但它并不是UI,也不是辅助UI滑动,反而是单纯地为滑动提供计算。

无论从构造方法还是其他方法,以及Scroller的属性可知,其并不会持有View,辅助ViewGroup滑动


2.Scroller只是提供计算,那谁来调用computeScroll使得ViewGroup滑动

答:computeScroll也不是来让ViewGroup滑动的,真正让ViewGroup滑动的是scrollTo,scrollBy。computeScroll的作用是计算ViewGroup如何滑动。而computeScroll是通过draw来调用的。


3.computeScroll和Scroller都是计算,两者有啥关系?

答:文章开始已作答,没有直接的关系。computeScroll和Scroller要是飞得拉关系的话,那就是computeScroll可以参考Scroller计算结果来影响scrollTo,scrollBy,从而使得滑动发生改变。也就是Scroller不会调用computeScroll,反而是computeScroll调用Scroller。


4.滑动时连续的,如何让Scroller的计算也是连续的?

这个就问到了什么时候调用computeScroll了,如上所说computeScroll调用Scroller,只要computeScroll调用连续,Scroller也会连续,实质上computeScroll的连续性又invalidate方法控制,scrollTo,scrollBy都会调用invalidate,而invalidate回去触发draw,从而computeScroll被连续调用,综上,Scroller也会被连续调用,除非invalidate停止调用


5.computeScroll如何和Scroller的调用过程保持一致。

computeScroll参考Scroller影响scrollTo,scrollBy,实质上,为了不重复影响scrollTo,scrollBy,那么Scroller必须终止计算currX,currY。要知道计算有没有终止,需要通过mScroller.computeScrollOffset()

6.如上问题应该说的很清楚了吧,如果不明白,请留言。


7.通过一个SlidePanel的例子,我们来深刻的了解一下

注意:在移动平台中,要明确知道“滑动”与“滚动”的不同,具体来说,滑动和滚动的方向总是相反的。

public class SlidingPanel extends RelativeLayout {
   private Context context;
   private FrameLayout leftMenu;
   private FrameLayout middleMenu;
   private FrameLayout rightMenu;
   private FrameLayout middleMask;
   private Scroller mScroller;
   public  final int LEFT_ID = 0xaabbcc;
   public  final int MIDEELE_ID = 0xaaccbb;
   public  final int RIGHT_ID = 0xccbbaa;

   private boolean isSlideCompete;
   private boolean isHorizontalScroll;

   private Point point = new Point();
   private static final int SLIDE_SLOP = 20;

   public SlidingPanel(Context context) {
      super(context);
      initView(context);
   }

   public SlidingPanel(Context context, AttributeSet attrs) {
      super(context, attrs);
      initView(context);
   }

   private void initView(Context context) {

      this.context = context;
      mScroller = new Scroller(context, new DecelerateInterpolator());
      leftMenu = new FrameLayout(context);
      middleMenu = new FrameLayout(context);
      rightMenu = new FrameLayout(context);
      middleMask = new FrameLayout(context);
      leftMenu.setBackgroundColor(Color.RED);
      middleMenu.setBackgroundColor(Color.GREEN);
      rightMenu.setBackgroundColor(Color.RED);
      middleMask.setBackgroundColor(0x88000000);

      addView(leftMenu);
      addView(middleMenu);
      addView(rightMenu);
      addView(middleMask);

      middleMask.setAlpha(0);
   }
   public float onMiddleMask(){
      return middleMask.getAlpha();
   }
   
   @Override
   public void scrollTo(int x, int y) {
      super.scrollTo(x, y);
      onMiddleMask();
   // Log.e("getScrollX","getScrollX="+getScrollX());//可以是负值
      int curX = Math.abs(getScrollX());
      float scale = curX/(float)leftMenu.getMeasuredWidth();
      middleMask.setAlpha(scale);
      
   }

   @Override
   protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
      super.onMeasure(widthMeasureSpec, heightMeasureSpec);
      middleMenu.measure(widthMeasureSpec, heightMeasureSpec);
      middleMask.measure(widthMeasureSpec, heightMeasureSpec);
      int realWidth = MeasureSpec.getSize(widthMeasureSpec);
      int tempWidthMeasure = MeasureSpec.makeMeasureSpec(
            (int) (realWidth * 0.8f), MeasureSpec.EXACTLY);
      leftMenu.measure(tempWidthMeasure, heightMeasureSpec);
      rightMenu.measure(tempWidthMeasure, heightMeasureSpec);
   }

   @Override
   protected void onLayout(boolean changed, int l, int t, int r, int b) {
      super.onLayout(changed, l, t, r, b);
      middleMenu.layout(l, t, r, b);
      middleMask.layout(l, t, r, b);
      leftMenu.layout(l - leftMenu.getMeasuredWidth(), t, r, b);
      rightMenu.layout(
            l + middleMenu.getMeasuredWidth(),
            t,
            l + middleMenu.getMeasuredWidth()
                  + rightMenu.getMeasuredWidth(), b);
   }



   @Override
   public boolean dispatchTouchEvent(MotionEvent ev) {
      if (!isSlideCompete) {
         handleSlideEvent(ev);
         return true;
      }
      if (isHorizontalScroll) {
         switch (ev.getActionMasked()) {
         case MotionEvent.ACTION_MOVE:
            int curScrollX = getScrollX();
            int dis_x = (int) (ev.getX() - point.x);
            //滑动方向和滚动滚动条方向相反,因此dis_x必须取负值
            int expectX = -dis_x + curScrollX;

            if(dis_x>0)
            {
               Log.d("I","Right-Slide,Left-Scroll");//向右滑动,向左滚动
            }else{
               Log.d("I","Left-Slide,Right-Scroll");
            }

            Log.e("I","ScrollX="+curScrollX+" , X="+ev.getX()+" , dis_x="+dis_x);
            //规定expectX的变化范围
             int    finalX = Math.max(-leftMenu.getMeasuredWidth(),Math.min(expectX, rightMenu.getMeasuredWidth()));

            scrollTo(finalX, 0);
            point.x = (int) ev.getX();//更新,保证滑动平滑
            break;

         case MotionEvent.ACTION_UP:
         case MotionEvent.ACTION_CANCEL:
            curScrollX = getScrollX();
            if (Math.abs(curScrollX) > leftMenu.getMeasuredWidth() >> 1) {
               if (curScrollX < 0) {
                  mScroller.startScroll(curScrollX, 0,
                        -leftMenu.getMeasuredWidth() - curScrollX, 0,
                        200);
               } else {
                  mScroller.startScroll(curScrollX, 0,
                        leftMenu.getMeasuredWidth() - curScrollX, 0,
                        200);
               }

            } else {
               mScroller.startScroll(curScrollX, 0, -curScrollX, 0, 200);
            }
            invalidate();
            isHorizontalScroll = false;
            isSlideCompete = false;
            break;
         }
      } else {
         switch (ev.getActionMasked()) {
         case MotionEvent.ACTION_UP:
            isHorizontalScroll = false;
            isSlideCompete = false;
            break;

         default:
            break;
         }
      }

      return super.dispatchTouchEvent(ev);
   }

   /**
    * 通过invalidate操纵,此方法通过draw方法调用
    */
   @Override
   public void computeScroll() {
      super.computeScroll();
      if (!mScroller.computeScrollOffset()) {
         //计算currX,currY,并检测是否已完成“滚动”
         return;
      }
      int tempX = mScroller.getCurrX();
      scrollTo(tempX, 0); //会重复调用invalidate
   }


   private void handleSlideEvent(MotionEvent ev) {
      switch (ev.getAction()&MotionEvent.ACTION_MASK) {
      case MotionEvent.ACTION_DOWN:
         point.x = (int) ev.getX();
         point.y = (int) ev.getY();
         super.dispatchTouchEvent(ev);
         break;

      case MotionEvent.ACTION_MOVE:
         int dX = Math.abs((int) ev.getX() - point.x);
         int dY = Math.abs((int) ev.getY() - point.y);
         if (dX >= SLIDE_SLOP && dX > dY) { // 左右滑动
            isHorizontalScroll = true;
            isSlideCompete = true;
            point.x = (int) ev.getX();
            point.y = (int) ev.getY();
         } else if (dY >= SLIDE_SLOP && dY > dX) { // 上下滑动
            isHorizontalScroll = false;
            isSlideCompete = true;
            point.x = (int) ev.getX();
            point.y = (int) ev.getY();
         }
         break;
      case MotionEvent.ACTION_UP:
      case MotionEvent.ACTION_OUTSIDE:
      case MotionEvent.ACTION_CANCEL:
         super.dispatchTouchEvent(ev);
         isHorizontalScroll = false;
         isSlideCompete = false;
         break;
      }
   }

}


© 著作权归作者所有

共有 人打赏支持
IamOkay
粉丝 187
博文 458
码字总数 370505
作品 0
海淀
程序员
加载中

评论(3)

呼呼哈嘿
呼呼哈嘿
博主的代码运行了一片绿色。看不懂。能否解释一下0
lxholding
lxholding
好文,作者写的不错
IT全能
IT全能
赞一个
Android开源中国客户端学习 (自定义View)左右滑动控件ScrollLayout

左右滑动的控件我们使用的也是非常多了,但是基本上都是使用的viewpager 等 android基础的控件,那么我们有么有考虑过查看他的源码进行定制呢?当然,如果你自我感觉非常好的话可以自己定制一个,...

SuShine
2013/08/09
0
2
Android 使用Scroller实现绚丽的ListView左右滑动删除Item效果

转帖请注明本文出自xiaanming的博客(http://blog.csdn.net/xiaanming/article/details/17539199),请尊重他人的辛勤劳动成果,谢谢! 我在上一篇文章中Android 带你从源码的角度解析Scrol...

程序袁_绪龙
2015/08/13
0
0
Android中Scroller类的分析

今天看了一下项目中用到的ViewFlow控件,想弄明白其工作原理。从头开始分析,卡在“滚动”这儿了。 做android也快两年了,连最基本的滚动都不熟悉,真是惭愧。。。遂网上找资料,很容易的在g...

Jonson
2014/05/15
0
0
Launcher WorkSpace 左右滑动切换屏幕

http://blog.csdn.net/yaoguet/article/details/6393962 http://cowboy.1988.blog.163.com/blog/static/75105798201211059361/ 提取Launcher中的WorkSapce,可以左右滑动切换屏幕页面的类 分......

尼莫
2012/06/12
0
0
Android View的事件体系

导语 本章主要介绍View的事件分发和滑动冲突问题的解决方案,可以和Android事件拦截机制分析对比着看。 主要内容 view的基础知识 View的滑动 弹性滑动 View的事件分发机制 滑动冲突 具体内容...

一个有故事的程序员
08/07
0
0

没有更多内容

加载失败,请刷新页面

加载更多

下一页

Ubuntu18.04 显卡GF-940MX安装NVIDIA-390.77

解决办法: 下面就给大家一个正确的姿势在Ubuntu上安装Nvidia驱动: (a)首先去N卡官网下载自己显卡对应的驱动:www.geforce.cn/drivers (b)下载后好放在英文路径的目录下,怎么简单怎么来...

AI_SKI
今天
1
0
深夜胡思乱想

魔兽世界 最近魔兽世界出了新版本, 周末两天升到了满级,比之前的版本体验好很多,做任务不用抢怪了,不用组队打怪也是共享拾取的。技能简化了很多,哪个亮按哪个。 运维 服务器 产品 之间的...

Firxiao
今天
1
0
MySQL 8 在 Windows 下安装及使用

MySQL 8 带来了全新的体验,比如支持 NoSQL、JSON 等,拥有比 MySQL 5.7 两倍以上的性能提升。本文讲解如何在 Windows 下安装 MySQL 8,以及基本的 MySQL 用法。 下载 下载地址 https://dev....

waylau
今天
0
0
微信第三方平台 access_token is invalid or not latest

微信第三方开发平台code换session_key说的特别容易,但是我一使用就带来无穷无尽的烦恼,搞了一整天也无济于事. 现在记录一下解决问题的过程,方便后来人参考. 我遇到的这个问题搜索了整个网络也...

自由的开源
今天
3
0
openJDK之sun.misc.Unsafe类CAS底层实现

注:这篇文章参考了https://www.cnblogs.com/snowater/p/8303698.html 1.sun.misc.Unsafe中CAS方法 在sun.misc.Unsafe中CAS方法如下: compareAndSwapObject(java.lang.Object arg0, long a......

汉斯-冯-拉特
今天
3
0

没有更多内容

加载失败,请刷新页面

加载更多

下一页

返回顶部
顶部