android computescroll_Android Scroller与computeScroll方法的调用关系
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
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方法的调用关系相关推荐
- Android Scroller与computeScroll方法的调用关系
2019独角兽企业重金招聘Python工程师标准>>> Android ViewGroup中的Scroller与computeScroll的有什么关系? 答:没有直接的关系 知道了答 ...
- android computescroll_Android Scroller与computeScroll的调用机制关系
Android ViewGroup中的Scroller与computeScroll的有什么关系? 答:没有直接的关系 知道了答案,是不是意味着下文就没必要看了,如果说对ViewGroup自定义控件不感 ...
- Android 自定义的handler handleMessage方法不调用
自己写的一个类继承自handler. 想的是从服务器读一张图片,然后用Bitmap存一下,通过handler传给主线程来显示图片 private static class LoadHandler ex ...
- Android获取设备序列号的方法与调用方式
记录一下安卓获取设备序列号的方法,以及调用方式 方法: /*** * 使用反射调用系统隐藏方法get(),获取系统相关属性配置* @param key 属性名称* @return*/ static S ...
- java多态子父类的构造器、成员变量、方法的调用关系
1.结论 ①:成员变量没有多态,方法才有多态 Father gay = new Son(); gay.money 是属于父类的 ((Son) gay).money 才是子类的 ②:Father gay ...
- 接口,实现类,对象方法的调用关系.(查看程序输出)
分析的术语可能不是那么准确!有待加强~~~ 划线法真的不错.不过其实这个程序仔细看看也能很好分析. 只不过脑子容量还是有限的~好脑子不如画个图~ 转载于:https://www.cnblogs.com ...
- android computescroll_Android问题:自定义ViewGroup,重载computeScroll()方法有什么用?...
展开全部 为了易于控制滑屏控制,Android框架提供了 computeScroll()方法去控制这个流程.e69da5e887aa62616964757a686964616f313333353461 ...
- Android 使用Scroller实现绚丽的ListView左右滑动删除Item效果
本文出自xiaanming的博客(http://blog.csdn.net/xiaanming/article/details/17539199) 我在上一篇文章中Android 带你从源码的角度解析 ...
- Android的Scroller介绍
以提供的起始点和将要滑动的距离开始滚动.滚动会使用缺省值250ms作为持续时间. 参数 startX 水平方向滚动的偏移值,以像素为单位.负值表明滚动将向左滚动 startY 垂直方向滚动的偏移值,以 ...
最新文章
- 数据结构与算法(8-2)有序表查找(折半查找(二分查找)、插值查找)
- 关于redo(二)插入更新数据时的效率比较
- 【贪心】【codeforces】651A Joysticks
- QT计算机控制进程,Qt 学习之路 2(69):进程
- 使用pdb调试Python程序
- 论文|Airbnb Embedding的实践和思考
- 基于php网络小说,五部小说点击破亿,盘点网络文学白金作家我吃西红柿!
- Tomcat8.0进入tomcat manager的方法
- Jmeter基础教程
- 网卡5790c linux驱动,(支持所有硬件、无需连接宽带)e驱动 v5.21 WIN7 32bit专版驱动包...
- [日常] Go语言圣经前言
- 制作在线单词测试的软件,推荐几个在线测试英语单词量的网站
- Sonic常见问题解决方法之——设备中心出现多个iOS设备接入异常
- 我们熟悉的106短信的水好深啊
- H264码流打包成RTP包
- ASO技巧:有效利用100个字符,aso优化技巧大aso技巧
- RFID期末复习 四、五、六、七章节
- 网络系列--计算机系统与人工智能之我见
- Python练习——输出10个不重复的英文字母
- 预测模型 | 8. 一致性指数 (Concordance Index, C-index) ci值 cox
热门文章
- 一份好的简历应该是这样的(This Is What A GOOD Resume Should Look Like)
- [Google Guava] 2.4-集合扩展工具类
- Spring IOC 容器源码分析 - 获取单例 bean
- OSGi入门篇:模块层
- 深入理解 Java 线程池:ThreadPoolExecutor
- 程序开发是编写sql语句的注意事项
- Java并发编程(4):守护线程与线程阻塞的四种情况
- 文艺青年的两门必修课——绘画与音乐
- Octave matlab中运行.m文件方式对比
- [Python]元组与列表的区别及内建用法