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;

}

}

}

android computescroll_Android Scroller与computeScroll的调用机制关系相关推荐

  1. android computescroll_Android Scroller与computeScroll方法的调用关系

    Android ViewGroup中的Scroller与computeScroll的有什么关系? 答:没有直接的关系 知道了答案,是不是意味着下文就没必要看了,如果说对ViewGroup自定义控件不感 ...

  2. Android onTouchEvent, onClick及onLongClick的调用机制

    针对屏幕上的一个View控件,Android如何区分应当触发onTouchEvent,还是onClick,亦或是onLongClick事件? 在Android中,一次用户操作可以被不同的View按次序 ...

  3. Android Scroller与computeScroll方法的调用关系

    2019独角兽企业重金招聘Python工程师标准>>> Android ViewGroup中的Scroller与computeScroll的有什么关系? 答:没有直接的关系 知道了答 ...

  4. android computescroll_Android问题:自定义ViewGroup,重载computeScroll()方法有什么用?...

    展开全部 为了易于控制滑屏控制,Android框架提供了 computeScroll()方法去控制这个流程.e69da5e887aa62616964757a686964616f313333353461 ...

  5. Android 电话的反射调用机制实现自动接听电话

    最近在看一些有关反射和aidl远程服务控制的类,自己也小写了这么一个Android 电话的反射调用机制实现静默接听电话demo 自己总结出来,以供查阅,若是还有其他问题,希望大家指出. 首先要调用系统 ...

  6. Android 使用Scroller实现绚丽的ListView左右滑动删除Item效果

    本文出自xiaanming的博客(http://blog.csdn.net/xiaanming/article/details/17539199) 我在上一篇文章中Android 带你从源码的角度解析 ...

  7. [原] Android中Scroller类的分析

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

  8. android scroller,深入理解Android中Scroller的滚动原理

    View的平滑滚动效果 什么是实现View的平滑滚动效果呢,举个简单的例子,一个View从在我们指定的时间内从一个位置滚动到另外一个位置,我们利用Scroller类可以实现匀速滚动,可以先加速后减速, ...

  9. Android的Scroller介绍

    以提供的起始点和将要滑动的距离开始滚动.滚动会使用缺省值250ms作为持续时间. 参数 startX 水平方向滚动的偏移值,以像素为单位.负值表明滚动将向左滚动 startY 垂直方向滚动的偏移值,以 ...

最新文章

  1. 动态添加html元素绑定事件,关于javascript:jQuery如何将onclick事件绑定到动态添加的HTML元素...
  2. 工作流与Petri net的关系
  3. 转载——CVE-2019-0807
  4. 高效新思路!实例分割算法!腾讯、华科出品!入选ICCV 2021!
  5. Kubernetes学习笔记之kube-proxy service实现原理
  6. http请求过程_“HTTP请求的整个过程”与它引出了整个计算机网络知识图谱(一)...
  7. win7字体大小怎么设置_XP,win7系统怎么设置自动关机
  8. 从键盘输入3个整数,输出其中最大数
  9. Python 帮助信息(pydoc模块)
  10. java udp 工具类_java udp 简单实现
  11. 生产排程系统_论生产计划排程APS系统的几大问题
  12. 【渝粤题库】广东开放大学 建筑设备 形成性考核
  13. MOV Cutter使用记
  14. ehcache:Null key returned for cache operation问题解决方法
  15. 快递鸟智选物流API对接流程
  16. 二维数组 Arrays
  17. Swift 使用 UISlider
  18. python使用什么来表示代码块_Python入门之基本语法
  19. 一个仿微博TextView 筛选,直接使用的工具类
  20. 11月22日训练题解

热门文章

  1. JNDI学习总结(3)——Tomcat下使用C3P0配置JNDI数据源
  2. 文件粉碎机c语言代码,Life——文件粉碎机源码
  3. Day08- team、iptables、firewall
  4. pyqt——布局管理
  5. 代码级操作指南 | 如何在Docker Swarm中运行服务
  6. OpenGL ES着色器语言之变量和数据类型(二)(官方文档第四章)
  7. 多个C文件共用一些变量
  8. 支持IPsec / L2TP / EtherIP测试版下载
  9. 详解ADSL接入方式的异同比较
  10. JAVA单例模式 关于延迟加载问题