作者: ztelur

联系方式:segmentfaultcsdngithub

本文仅供个人学习,不用于不论什么形式商业目的,转载请注明原作者、文章来源。链接,版权归原文作者全部。

 本文是android滚动相关的系列文章的第二篇,主要总结一下使用手势相关的代码逻辑。主要是单点拖动,多点拖动,fling和OveScroll的实现。每一个手势都会有代码片段。

 对android滚动相关的知识还不太了解的同学能够先阅读一下文章:

 为了节约你的时间,我特地将文章大致内容总结例如以下:

  • 手势Drag的实现和原理
  • 手势Fling的实现和原理
  • OverScroll效果和EdgeEffect效果的实现和原理。

 具体代码请查看我的github

Drag

 Drag是最为主要的手势:用户能够使用手指在屏幕上滑动,以拖动屏幕对应内容移动。实现Drag手势事实上非常easy。过程例如以下:

  • ACTION_DOWN事件发生时。调用getXgetY函数获得事件发生的x,y坐标值,并记录在mLastXmLastY变量中。
  • ACTION_MOVE事件发生时。调用getXgetY函数获得事件发生的x,y坐标值,将其与mLastXmLastY比較。假设二者差值大于一定限制(ScaledTouchSlop),就运行scrollBy函数,进行滚动,最后更新mLastXmLastY的值。
  • ACTION_UPACTION_CANCEL事件发生时,清空mLastXmLastY
  1. @Override
  2. public boolean onTouchEvent(MotionEvent event) {
  3. int actionId = MotionEventCompat.getActionMasked(event);
  4. switch (actionId) {
  5. case MotionEvent.ACTION_DOWN:
  6. mLastX = event.getX();
  7. mLastY = event.getY();
  8. mIsBeingDragged = true;
  9. if (getParent() != null) {
  10. getParent().requestDisallowInterceptTouchEvent(true);
  11. }
  12. break;
  13. case MotionEvent.ACTION_MOVE:
  14. float curX = event.getX();
  15. float curY = event.getY();
  16. int deltaX = (int) (mLastX - curX);
  17. int deltaY = (int) (mLastY - curY);
  18. if (!mIsBeingDragged && (Math.abs(deltaX)> mTouchSlop ||
  19. Math.abs(deltaY)> mTouchSlop)) {
  20. mIsBeingDragged = true;
  21. // 让第一次滑动的距离和之后的距离不至于差距太大
  22. // 由于第一次必须>TouchSlop,之后则是直接滑动
  23. if (deltaX > 0) {
  24. deltaX -= mTouchSlop;
  25. } else {
  26. deltaX += mTouchSlop;
  27. }
  28. if (deltaY > 0) {
  29. deltaY -= mTouchSlop;
  30. } else {
  31. deltaY += mTouchSlop;
  32. }
  33. }
  34. // 当mIsBeingDragged为true时,就不用推断> touchSlopg啦,不然会导致滚动是一段一段的
  35. // 不是非常连续
  36. if (mIsBeingDragged) {
  37. scrollBy(deltaX, deltaY);
  38. mLastX = curX;
  39. mLastY = curY;
  40. }
  41. break;
  42. case MotionEvent.ACTION_CANCEL:
  43. case MotionEvent.ACTION_UP:
  44. mIsBeingDragged = false;
  45. mLastY = 0;
  46. mLastX = 0;
  47. break;
  48. default:
  49. }
  50. return mIsBeingDragged;
  51. }

多触点Drag

 上边的代码仅仅适用于单点触控的手势。假设你是两个手指触摸屏幕。那么它仅仅会依据你第一个手指滑动的情况来进行屏幕滚动。更为致命的是。当你先松开第一个手指时,由于我们少监听了ACTION_POINTER_UP事件,将会导致屏幕突然滚动一大段距离,由于第二个手指移动事件的x,y值会和第一个手指移动时留下的mLastXmLastY比較,导致屏幕滚动。

 假设我们要监听并处理多触点的事件,我们还须要对ACTION_POINTER_DOWNACTION_POINTER_UP事件进行监听,而且在ACTION_MOVE事件时,要记录全部触摸点事件发生的x,y值。

  • ACTION_POINTER_DOWN事件发生时,我们要记录第二触摸点事件发生的x,y值为mSecondaryLastXmSecondaryLastY,和第二触摸点pointer的id为mSecondaryPointerId
  • ACTION_MOVE事件发生时,我们除了依据第一触摸点pointer的x,y值进行滚动外,也要更新mSecondayLastXmSecondaryLastY
  • ACTION_POINTER_UP事件发生时。我们要先推断是哪个触摸点手指被抬起来啦。假设是第一触摸点,那么我们就将坐标值和pointer的id都更换为第二触摸点的数据;假设是第二触摸点,就仅仅要重置一下数据就可以。
  1. switch (actionId) {
  2. .....
  3. case MotionEvent.ACTION_POINTER_DOWN:
  4. activePointerIndex = MotionEventCompat.getActionIndex(event);
  5. mSecondaryPointerId = MotionEventCompat.findPointerIndex(event,activePointerIndex);
  6. mSecondaryLastX = MotionEventCompat.getX(event,activePointerIndex);
  7. mSecondaryLastY = MotionEventCompat.getY(event,mActivePointerId);
  8. break;
  9. case MotionEvent.ACTION_MOVE:
  10. ......
  11. // handle secondary pointer move
  12. if (mSecondaryPointerId != INVALID_ID) {
  13. int mSecondaryPointerIndex = MotionEventCompat.findPointerIndex(event, mSecondaryPointerId);
  14. mSecondaryLastX = MotionEventCompat.getX(event, mSecondaryPointerIndex);
  15. mSecondaryLastY = MotionEventCompat.getY(event, mSecondaryPointerIndex);
  16. }
  17. break;
  18. case MotionEvent.ACTION_POINTER_UP:
  19. //推断是否是activePointer up了
  20. activePointerIndex = MotionEventCompat.getActionIndex(event);
  21. int curPointerId = MotionEventCompat.getPointerId(event,activePointerIndex);
  22. Log.e(TAG, "onTouchEvent: "+activePointerIndex +" "+curPointerId +" activeId"+mActivePointerId+
  23. "secondaryId"+mSecondaryPointerId);
  24. if (curPointerId == mActivePointerId) { // active pointer up
  25. mActivePointerId = mSecondaryPointerId;
  26. mLastX = mSecondaryLastX;
  27. mLastY = mSecondaryLastY;
  28. mSecondaryPointerId = INVALID_ID;
  29. mSecondaryLastY = 0;
  30. mSecondaryLastX = 0;
  31. //反复代码。为了让逻辑看起来更加清晰
  32. } else{ //假设是secondary pointer up
  33. mSecondaryPointerId = INVALID_ID;
  34. mSecondaryLastY = 0;
  35. mSecondaryLastX = 0;
  36. }
  37. break;
  38. case MotionEvent.ACTION_CANCEL:
  39. case MotionEvent.ACTION_UP:
  40. mIsBeingDragged = false;
  41. mActivePointerId = INVALID_ID;
  42. mLastY = 0;
  43. mLastX = 0;
  44. break;
  45. default:
  46. }

Fling

 当用户手指高速划过屏幕,然后高速立马屏幕时,系统会判定用户运行了一个Fling手势。视图会高速滚动,而且在手指立马屏幕之后也会滚动一段时间。Drag表示手指滑动多少距离,界面跟着显示多少距离,而fling是依据你的滑动方向与轻重,还会自己主动滑动一段距离。Filing手势在android交互设计中应用非常广泛:电子书的滑动翻页、ListView滑动删除item、滑动解锁等。所以怎样检測用户的fling手势是非常重要的。

 在检測Fling时,你须要检測手指在屏幕上滑动的速度。这是你就须要VelocityTrackerScroller这两个类啦。

  • 我们首先使用VelocityTracker.obtain()这种方法获得事实上例
  • 然后每次处理触摸时间时,我们将触摸事件通过addMovement方法传递给它
  • 最后在处理ACTION_UP事件时。我们通过computeCurrentVelocity方法获得滑动速度;
  • 我们推断滑动速度是否大于一定数值(MinFlingSpeed),假设大于,那么我们调用Scrollerfling方法。

    然后调用invalidate()函数。

  • 我们须要重载computeScroll方法,在这种方法内。我们调用ScrollercomputeScrollOffset()方法啦计算当前的偏移量。然后获得偏移量,并调用scrollTo函数,最后调用postInvalidate()函数。
  • 除了上述的操作外。我们须要在处理ACTION_DOWN事件时,对屏幕当前状态进行推断,假设屏幕如今正在滚动(用户刚进行了Fling手势)。我们须要停止屏幕滚动。

 具体这一套流程是怎样运转的。我会在下一篇文章中具体解释,大家也能够自己查阅代码或者google来搞懂当中的原理。

  1. @Override
  2. public boolean onTouchEvent(MotionEvent event) {
  3. .....
  4. if (mVelocityTracker == null) {
  5. //检查速度測量器,假设为null。获得一个
  6. mVelocityTracker = VelocityTracker.obtain();
  7. }
  8. int action = MotionEventCompat.getActionMasked(event);
  9. int index = -1;
  10. switch (action) {
  11. case MotionEvent.ACTION_DOWN:
  12. ......
  13. if (!mScroller.isFinished()) { //fling
  14. mScroller.abortAnimation();
  15. }
  16. .....
  17. break;
  18. case MotionEvent.ACTION_MOVE:
  19. ......
  20. break;
  21. case MotionEvent.ACTION_CANCEL:
  22. endDrag();
  23. break;
  24. case MotionEvent.ACTION_UP:
  25. if (mIsBeingDragged) {
  26. //当手指立马屏幕时,获得速度。作为fling的初始速度 mVelocityTracker.computeCurrentVelocity(1000,mMaxFlingSpeed);
  27. int initialVelocity = (int)mVelocityTracker.getYVelocity(mActivePointerId);
  28. if (Math.abs(initialVelocity) > mMinFlingSpeed) {
  29. // 由于坐标轴正方向问题,要加负号。
  30. doFling(-initialVelocity);
  31. }
  32. endDrag();
  33. }
  34. break;
  35. default:
  36. }
  37. //每次onTouchEvent处理Event时,都将event交给时间
  38. //測量器
  39. if (mVelocityTracker != null) {
  40. mVelocityTracker.addMovement(event);
  41. }
  42. return true;
  43. }
  44. private void doFling(int speed) {
  45. if (mScroller == null) {
  46. return;
  47. }
  48. mScroller.fling(0,getScrollY(),0,speed,0,0,-500,10000);
  49. invalidate();
  50. }
  51. @Override
  52. public void computeScroll() {
  53. if (mScroller.computeScrollOffset()) {
  54. scrollTo(mScroller.getCurrX(),mScroller.getCurrY());
  55. postInvalidate();
  56. }
  57. }

OverScroll

 在Android手机上,当我们滚动屏幕内容到达内容边界时,假设再滚动就会有一个发光效果。而且界面会进行滚动一小段距离之后再回复原位,这些效果是怎样实现的呢?我们须要使用ScrollerscrollTo的升级版OverScrolleroverScrollBy了,还有发光的EdgeEffect类。

 我们先来了解一下相关的API,理解了这些接口參数的含义,你就能够轻松使用这些接口来实现上述的效果啦。

  1. protected boolean overScrollBy(int deltaX, int deltaY,
  2. int scrollX, int scrollY,
  3. int scrollRangeX, int scrollRangeY,
  4. int maxOverScrollX, int maxOverScrollY,
  5. boolean isTouchEvent)
  • int deltaX,int deltaY : 偏移量,也就是当前要滚动的x,y值。
  • int scrollX,int scrollY : 当前的mScrollX和mScrollY的值。
  • int scrollRangeX,int scrollRangeY: 标示能够滚动的最大的x,y值,也就是你视图真实的长和宽。也就是说。你的视图可视大小可能是100,100,可是视图中的内容的大小为200,200,所以。上述两个值就为200,200
  • int maxOverScrollX,int maxOverScrollY:同意超过滚动范围的最大值。x方向的滚动范围就是0~maxOverScrollX,y方向的滚动范围就是0~maxOverScrollY。
  • boolean isTouchEvent:是否在onTouchEvent中调用的这个函数。所以。当你在computeScroll中调用这个函数时。就能够传入false。
  1. protected void onOverScrolled(int scrollX, int scrollY, boolean clampedX, boolean clampedY)
  • int scrollX,int scrollY:就是x。y方向的滚动距离。就相当于mScrollXmScrollY

    你既能够直接把二者赋值给对应的成员变量。也能够使用scrollTo函数。

  • boolean clampedX,boolean clampY:表示是否到达超出滚动范围的最大值。

    假设为true。就须要调用OverScrollspringBack函数来让视图回复原来位置。

  1. public boolean springBack(int startX, int startY, int minX, int maxX, int minY, int maxY)
  • int startX,int startY:标示当前的滚动值,也就是mScrollXmScrollY的值。

  • int minX,int maxX:标示x方向的合理滚动值
  • int minY,int maxY:标示y方向的合理滚动值。

 相信看完上述的API之后,大家会有非常多的疑惑,所以这里我来举个样例。

 假设视图大小为100*100。当你一直下拉到视图上边缘。然后在下拉,这时,mScrollY已经达到或者超过正常的滚动范围的最小值了,也就是0,可是你的maxOverScrollY传入的是10,所以。mScrollY最小能够到达-10,最大能够为110。所以,你能够继续下拉。等到mScrollY到达或者超过-10时。clampedY就为true。标示视图已经达到能够OverScroll的边界,须要回滚到正常滚动范围,所以你调用springBack(0,0,0,100)。

 然后我们再来看一下发光效果是怎样实现的。

 使用EdgeEffect类。

一般来说,当你仅仅上下滚动时,你仅仅须要两个EdgeEffect实例,分别代表上边界和下边界的发光效果。你须要在以下两个情景下改变EdgeEffect的状态。然后在draw()方法中绘制EdgeEffect

  • 处理ACTION_MOVE时,假设发现y方向的滚动值超过了正常范围的最小值时。你须要调用上边界实例的onPull方法。假设是超过最大值。那么就是调用下边界的onPull方法。

  • computeScroll函数中。也就是说Fling手势运行过程中,假设发现y方向的滚动值超过正常范围时的最小值时,调用onAbsorb函数。

 然后就是重载draw方法。让EdgeEffect实例在画布上绘制自己。

你会发现,你必须对画布进行移动或者旋转来让EdgeEffect绘制出上边界或者下边界的发光的效果,由于EdgeEffect对象自己是没有上下左右的概念的。

  1. @Override
  2. public void draw(Canvas canvas) {
  3. super.draw(canvas);
  4. if (mEdgeEffectTop != null) {
  5. final int scrollY = getScrollY();
  6. if (!mEdgeEffectTop.isFinished()) {
  7. final int count = canvas.save();
  8. final int width = getWidth() - getPaddingLeft() - getPaddingRight();
  9. canvas.translate(getPaddingLeft(),Math.min(0,scrollY));
  10. mEdgeEffectTop.setSize(width,getHeight());
  11. if (mEdgeEffectTop.draw(canvas)) {
  12. postInvalidate();
  13. }
  14. canvas.restoreToCount(count);
  15. }
  16. }
  17. if (mEdgeEffectBottom != null) {
  18. final int scrollY = getScrollY();
  19. if (!mEdgeEffectBottom.isFinished()) {
  20. final int count = canvas.save();
  21. final int width = getWidth() - getPaddingLeft() - getPaddingRight();
  22. canvas.translate(-width+getPaddingLeft(),Math.max(getScrollRange(),scrollY)+getHeight());
  23. canvas.rotate(180,width,0);
  24. mEdgeEffectBottom.setSize(width,getHeight());
  25. if (mEdgeEffectBottom.draw(canvas)) {
  26. postInvalidate();
  27. }
  28. canvas.restoreToCount(count);
  29. }
  30. }
  31. }
  32. @Override
  33. public boolean onTouchEvent(MotionEvent event) {
  34. ......
  35. case MotionEvent.ACTION_MOVE:
  36. .....
  37. if (mIsBeingDragged) {
  38. overScrollBy(0,(int)deltaY,0,getScrollY(),0,getScrollRange(),0,mOverScrollDistance,true);
  39. final int pulledToY = (int)(getScrollY()+deltaY);
  40. mLastY = y;
  41. if (pulledToY<0) {
  42. mEdgeEffectTop.onPull(deltaY/getHeight(),event.getX(mActivePointerId)/getWidth());
  43. if (!mEdgeEffectBottom.isFinished()) {
  44. mEdgeEffectBottom.onRelease();
  45. }
  46. } else if(pulledToY> getScrollRange()) {
  47. mEdgeEffectBottom.onPull(deltaY/getHeight(),1.0f-event.getX(mActivePointerId)/getWidth());
  48. if (!mEdgeEffectTop.isFinished()) {
  49. mEdgeEffectTop.onRelease();
  50. }
  51. }
  52. if (mEdgeEffectTop != null && mEdgeEffectBottom != null &&(!mEdgeEffectTop.isFinished()
  53. || !mEdgeEffectBottom.isFinished())) {
  54. postInvalidate();
  55. }
  56. }
  57. .....
  58. }
  59. ....
  60. }
  61. @Override
  62. protected void onOverScrolled(int scrollX, int scrollY, boolean clampedX, boolean clampedY) {
  63. if (!mScroller.isFinished()) {
  64. int oldX = getScrollX();
  65. int oldY = getScrollY();
  66. scrollTo(scrollX,scrollY);
  67. onScrollChanged(scrollX,scrollY,oldX,oldY);
  68. if (clampedY) {
  69. Log.e("TEST1","springBack");
  70. mScroller.springBack(getScrollX(),getScrollY(),0,0,0,getScrollRange());
  71. }
  72. } else {
  73. // TouchEvent中的overScroll调用
  74. super.scrollTo(scrollX,scrollY);
  75. }
  76. }
  77. @Override
  78. public void computeScroll() {
  79. if (mScroller.computeScrollOffset()) {
  80. int oldX = getScrollX();
  81. int oldY = getScrollY();
  82. int x = mScroller.getCurrX();
  83. int y = mScroller.getCurrY();
  84. int range = getScrollRange();
  85. if (oldX != x || oldY != y) {
  86. overScrollBy(x-oldX,y-oldY,oldX,oldY,0,range,0,mOverFlingDistance,false);
  87. }
  88. final int overScrollMode = getOverScrollMode();
  89. final boolean canOverScroll = overScrollMode == OVER_SCROLL_ALWAYS ||
  90. (overScrollMode == OVER_SCROLL_IF_CONTENT_SCROLLS && range > 0);
  91. if (canOverScroll) {
  92. if (y<0 && oldY >= 0) {
  93. mEdgeEffectTop.onAbsorb((int)mScroller.getCurrVelocity());
  94. } else if (y> range && oldY < range) {
  95. mEdgeEffectBottom.onAbsorb((int)mScroller.getCurrVelocity());
  96. }
  97. }
  98. }
  99. }

后记

 本篇文章是系列文章的第二篇,大家可能已经知道怎样实现各类手势。可是对当中的机制和原理还不是非常了解,之后的第三篇会解说从本篇代码的视角解说一下android视图绘制的原理和Scroller的机制,希望大家多多关注。

參考文章

http://stackoverflow.com/questions/22843671/android-swipe-vs-fling

https://www.google.com/design/spec/patterns/gestures.html#gestures-drag-swipe-or-fling-details

http://www.jcodecraeer.com/a/anzhuokaifa/androidkaifa/2014/1212/2145.html

Android Scroll具体解释(二):OverScroller实战的更多相关文章

  1. android动画具体解释二 属性动画原理

    property动画是一个强大的框架,它差点儿能使你动画不论什么东西. 你能够定义一个动画来改变对象的不论什么属性,不论其是否被绘制于屏幕之上. 一个属性动画在一定时间内多次改变一个属性(对象的一个字 ...

  2. android动画具体解释一 概述

    动画和图形概述 Android 提供了大量的强大的API以应用于UI动画和绘制2D和3D图形.以下各节向你描写叙述了这些API的预览和系统能力以帮助你决定怎么才是达到你需求的最佳方法. 动画 Andr ...

  3. Android 布局学习之——Layout(布局)具体解释二(常见布局和布局參数)

     [Android布局学习系列]   1.Android 布局学习之--Layout(布局)具体解释一   2.Android 布局学习之--Layout(布局)具体解释二(常见布局和布局參数)   ...

  4. Android 图片加载库Glide 实战(二),占位符,缓存,转换自签名高级实战

    http://blog.csdn.net/sk719887916/article/details/40073747 请尊重原创 : skay <Android 图片加载库Glide 实战(一), ...

  5. Android学习路线(二十四)ActionBar Fragment运用最佳实践

    转载请注明出处:http://blog.csdn.net/sweetvvck/article/details/38645297 通过前面的几篇博客.大家看到了Google是怎样解释action bar ...

  6. Android高手进阶教程(二十八)之---Android ViewPager控件的使用(基于ViewPager的横向相册)!!!

      分类: Android高手进阶 Android基础教程 2012-09-14 18:10 29759人阅读 评论(35) 收藏 举报 android相册layoutobjectclassloade ...

  7. 【转】android 电容屏(二):驱动调试之基本概念篇

    关键词:android  电容屏 tp 工作队列 中断 多点触摸协议平台信息:内核:linux2.6/linux3.0系统:android/android4.0 平台:S5PV310(samsung ...

  8. Android slidingmenu详细解释 滑动的优化

    Android slidingmenu 详细解释 性能优化 转载请注明:   http://blog.csdn.net/aaawqqq 简单介绍 SlidingMenu 是github 上Androi ...

  9. Android Oreo 8.0 新特性实战 Autosizing TextView --自动缩放TextView

    Android Oreo 8.0 新特性实战 Autosizing TextView --自动缩放TextView 8.0出来很久了,这个新特性已经用了很久了,但是一直没有亲自去试试.这几天新的需求来 ...

随机推荐

  1. 《c程序设计语言》-2.9

    #include <stdio.h> /*int bitcount(unsigned x) { int b; for(b = 0;x != 0;x >>= 1) { if(x ...

  2. vue的main.js

    import Vue from 'vue'; import App from './App.vue'; //================http 请求======================= ...

  3. 搜索水题四连发_C++

    特别声明:以下题目有部分为原创题,涉及版权问题,不得转载,违者追究 法律责任! 话说这是一套神题,只有你想不到,没有你做不到 题目更正后比 Pascal 跑得还快哈~ 一道特别裸,但是特别坑的搜索题 ...

  4. 移动web开发问题和优化小结

    之前在微信公众号上看到的一篇文章,直接给拷过来了....原文链接http://mp.weixin.qq.com/s/0LwTz-Mw2WumSztIrHucdQ 2.Meta标签 页面在手机上显示时, ...

  5. Couchbase应用示例(初探)

    安装过程:略. 1. 新建Web项目 从NuGet获取并引用: CouchbaseNetClient,添加后引用列表显示为 : Couchbase.NetClient 2. 需要对项目添加引用,这里我 ...

  6. python--控制窗体

    窗体的显示和隐藏 #!/usr/bin/env python # -*- coding:utf-8 -*- # author:love_cat import win32con import win32 ...

  7. Django简单设置cookies和session

    一.Cookie cookie及特点 Cookie是由服务器(网站)生成的,存储在浏览器端的 键值对数据(通常经过加密) 在响应请求时,服务器会把生成 Cookie数据 发给浏览器,浏览器会自动保存( ...

  8. springBoot springCloud

    微服务功能的主要体现: 1)服务的注册与发现 Eureka ,Consul ,Zookeeper 2)服务的负载均衡 Ribbon 3)服务的容错 Hystrix 4)服务的网关 微服务中常用的网关组 ...

  9. 关于yii2 的db log 日志 错误处理errorHandler

    log 通过配置Web.config来完成 1 数据库增加 ‘前缀_log’表 2 配置Web.config 'bootstrap' => ['log'], 'components' => ...

  10. HDU 4920.Matrix multiplication-矩阵乘法

    Matrix multiplication Time Limit: 4000/2000 MS (Java/Others)    Memory Limit: 131072/131072 K (Java/ ...