承接上一篇,如果你没有读过前四章方法,可以点击下面的链接:

http://www.cnblogs.com/fuly550871915/p/4985053.html

下面开始讲第五中方法。

五、利用Scroller类实现滑动

(1)基础知识总结

为了便于下面的阅读代码,先来总结一下关于Scroller类的基础知识。利用Scroller类实现滑动,需要三个步骤:

步骤一:
初始化Scroller对象,即mScroller = new Scroller(context)
步骤二:
重写computeScroll()方法,实现模拟滑动。可以复制下面的末模板代码:
public void computeScroll() {
super.computeScroll(); if(mScroller.computeScrollOffset())
{
((View)getParent()).scrollTo(mScroller.getCurrX(),
mScroller.getCurrY());
} invalidate();//必须要调用
}
步骤三:
开启模拟过程,在合适的地方(一般都在move中)startScroll方法。它有两个重载方法如下:
startScroll(int startX,int startY, int dx,int dy,int duration)
startScroll(int startX,int startY,int dx,int dy)
方法中的参数无需多解释了,就是起始坐标与偏移量,还有完成偏移的时间而已。

需要说明的是:(1)computeScrollOffset方法用来判断是否完成了整个滑动,返回为true,则说明没有完成,否则则完成滑动。

(2)getCurrY以及getCurrX获得的是当前的滑动坐标。

(3)最后必须要用invalidate方法来刷新。因为computeScroll方法不会自动调用,是在draw方法中被调用的。所以

必须使用invalidate刷新,就会调用draw方法,自然就会调用computeScroll方法了。这样子就会实现循环调用。

(4)在startScroll中,偏移量跟使用scrollBy方法中的偏移量用法是一样的,即也必须填写你实际想要移动距离的相

反数。也就是你实际想让它偏移一个正值,这里就填写它相应的负值,如果想偏移一个负值,这里就填写相应的

正值!

(2)代码实践

有些东西还是需要在真正的代码中才能说的明白的。我们接着上一篇文章的代码继续编写。为了看清Scroller类是如何实现滑动的,我们

再次修改DragView的代码。这次修改的有些多。贴完整的代码出来。如下:

 package com.example.testdragview;

 import android.content.Context;
import android.util.AttributeSet;
import android.view.MotionEvent;
import android.view.View;
import android.widget.Scroller; public class DragView extends View{ private int lastX;
private int lastY;
private Scroller mScroller; public DragView(Context context, AttributeSet attrs, int defStyleAttr) {
super(context, attrs, defStyleAttr); mScroller = new Scroller(context);
20 } public DragView(Context context, AttributeSet attrs) {
super(context, attrs);
mScroller = new Scroller(context);
} public DragView(Context context) {
super(context);
mScroller = new Scroller(context);
} public boolean onTouchEvent(MotionEvent event) { //获取到手指处的横坐标和纵坐标
int x = (int) event.getX();
int y = (int) event.getY(); switch(event.getAction())
{
case MotionEvent.ACTION_DOWN: lastX = x;
lastY = y;
break; case MotionEvent.ACTION_MOVE: //计算移动的距离
int offX = x - lastX;
int offY = y - lastY; View viewGroup = (View) getParent();
56
57 //开启滑动
58 mScroller.startScroll(viewGroup.getScrollX(),
59 viewGroup.getScrollY(), -offX, -offY);
60
break; case MotionEvent.ACTION_UP: break;
}
return true;
} public void computeScroll() {
73 super.computeScroll();
74
75 if(mScroller.computeScrollOffset())
76 {
77 ((View)getParent()).scrollTo(mScroller.getCurrX(),
78 mScroller.getCurrY());
79 }
80
81 invalidate();//必须要调用
82 }
}

红色部分就是修改的核心代码了。需要说明的是,在开启滑动的代码中,即第58行,使用的是在父视图中使用getScrollX和getScrollY获取

所滑动到的点的坐标。而getScollX和getScrollY正是从computeScroll中的scrollTo方法中传递过来的。通过阅读源码你可以很清晰的看到这个

过程。我贴两张源码图吧,如下:

                               

上图中观察mScrollX的传递过程,确实如上面我所说的。

而且在开启滑动的代码中,注意我们将偏移量可是都设定为负值了哦!(正如我上面所讲的设定为相反数即可)。

好了,关于代码的解释,就讲这么多,其他的都很好理解了。现在就运行程序,效果如下:

值得注意的是,我们的的DragView(即红色的块块)并不是即时跟随鼠标的滑动,在真机上这个效果将会更加突出!!其实这就是

Scrollerl类的滑动特点!即手指到达某一位置后,view会随后再平滑的到达这个位置,也就是说并不是view和手指一起到达指定位置的。

+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++

因此,一般并不利用Scroller类实现滑动,因为它不是即时的。但却经常用它来实现手指抬起后,view回到初始位置!

来修改下DragView看看怎么实现这个效果。代码如下:

 package com.example.testdragview;

 import android.content.Context;
import android.util.AttributeSet;
import android.view.MotionEvent;
import android.view.View;
import android.widget.Scroller; public class DragView extends View{ private int lastX;
private int lastY;
private Scroller mScroller; public DragView(Context context, AttributeSet attrs, int defStyleAttr) {
super(context, attrs, defStyleAttr); mScroller = new Scroller(context);
} public DragView(Context context, AttributeSet attrs) {
super(context, attrs);
mScroller = new Scroller(context);
} public DragView(Context context) {
super(context);
mScroller = new Scroller(context);
} public boolean onTouchEvent(MotionEvent event) { //获取到手指处的横坐标和纵坐标
int x = (int) event.getX();
int y = (int) event.getY(); switch(event.getAction())
{
case MotionEvent.ACTION_DOWN: lastX = x;
lastY = y;
break; case MotionEvent.ACTION_MOVE: //计算移动的距离
int offX = x - lastX;
int offY = y - lastY; ((View) getParent()).scrollBy(-offX,- offY); break; case MotionEvent.ACTION_UP: View viewGroup = (View) getParent();
62
63 //开启滑动,让其回到原点
64 mScroller.startScroll(viewGroup.getScrollX(),
65 viewGroup.getScrollY(),
66 -viewGroup.getScrollX() ,-viewGroup.getScrollY() );
67
break;
}
return true;
} public void computeScroll() {
super.computeScroll(); if(mScroller.computeScrollOffset())
{
((View)getParent()).scrollTo(mScroller.getCurrX(),
mScroller.getCurrY());
} invalidate();//必须要调用
}
}

红色部分依旧是修改的核心代码。这里主要就是将实现的滑动效果替换成了用scrollBy来实现即时滑动,然后在手指抬起的up方法中,利用

Scoller类将view放回原来位置!运行一下,看看效果,如下:

如上图所示,只要鼠标一抬起,红色块块就会立刻回到原来位置!怎么样不错吧。

++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++

好了,关于这个Scroller类的使用,就讲到这里。上面的代码基本都可以当做是模板代码了,需要的时候过来看看复制一下吧。

未完待续.....(最后一种方法)

android中实现view可以滑动的六种方法续篇(一)的更多相关文章

  1. android中实现view可以滑动的六种方法续篇(二)

    承接上一篇,上一篇中讲解了实现滑动的第五种方法,如果你还没读过,可点击下面链接: http://www.cnblogs.com/fuly550871915/p/4985482.html 这篇文章现在来 ...

  2. android中实现view可以滑动的六种方法

    在android开发中,经常会遇到一个view需要它能够支持滑动的需求.今天就来总结实现其滑动的六种方法.其实每一种方法的 思路都是一样的,即:监听手势触摸的坐标来实现view坐标的变化,从而实现vi ...

  3. Android View体系(二)实现View滑动的六种方法

    1.View的滑动简介 View的滑动是Android实现自定义控件的基础,同时在开发中我们也难免会遇到View的滑动的处理.其实不管是那种滑动的方式基本思想都是类似的:当触摸事件传到View时,系统 ...

  4. 详解实现Android中实现View滑动的几种方式

    注: 本文提到的所有三种滑动方式的完整demo:ScrollDemo 1. 关于View我们需要知道的 (1)什么是View? Android中的View类是所有UI控件的基类(Base class) ...

  5. Android 中的View与ViewGroup

    Android重点知识--View和ViewGroup与自定义控件 作者:丁明祥 邮箱:2780087178@qq.com 一.基础 ViewGroup 参考资料: Android 手把手教您自定义V ...

  6. android中的坐标系以及获取坐标的方法

    android中有两种坐标系,分别称之为Android坐标系和视图坐标系.而对应的也有一些相关的方法可以获取坐标系中的 坐标值.只有搞清楚这些区别,才能在实现的时候不至于出错或者得不到你想要的效果. ...

  7. MVC中视图View向控制器传值的方法

    MVC中视图View向控制器传值的方法步骤如下: 1.index页面: 页面中只需要一个触发事件的按钮

  8. android中Bitmap的放大和缩小的方法

    android中Bitmap的放大和缩小的方法 时间 2013-06-20 19:02:34  CSDN博客原文  http://blog.csdn.net/ada168855/article/det ...

  9. Android 自己定义View须要重写ondraw()等方法

    Android  自己定义View须要重写ondraw()等方法.这篇博客给大家说说自己定义View的写法,须要我们继承View,然后重写一些 方法,方法多多,看你须要什么方法 首先写一个自己定义的V ...

随机推荐

  1. [python]在场景中理解装饰器

    原来我也自己通过查资料,来学习python的装饰器,但是效果不好.因为没有接触过需要用到装饰器的场景,所以 一起的资料都只停留在纸面上,但是今天偶然看到了vimer的这篇文章:http://www.v ...

  2. 初遇sql server

    今天初始接触sql server 和mysql的语法有一些不同 sql server中使用[] 或双引号来表示数据库.字段名.表名等,而字符串使用单引号来表示 mysql中数据库名,表名,字段名不需要 ...

  3. 订餐APP第二次sprint+燃尽图

    MY-HR 成员: 角色分配 学号 博客园 团队贡献分 围观其他小组评论 丘惠敏 PM项目经理 201406114203 http://www.cnblogs.com/qiuhuimin/ 21 ht ...

  4. 【转】MSMQ消息队列安装

    一.Windows 7安装.管理消息队列1.安装消息队列   执行用户必须要有本地 Administrators 组中的成员身份,或等效身份.   具体步骤:    开始—>控制面板—>程 ...

  5. C#文件和文件夹输入输出流代码

    1.建立一个文本文件 public class FileClass { public static void Main() { WriteToFile(); } static void WriteTo ...

  6. 全球2/3的DNS瘫痪 顶级域名根服务器故障

    1月21日下午消息,据多家DNS服务商透露,今日下午3点,全国所有通用顶级域的根出现异常,导致部分国内用户无法访问.com域名网站,对全国互联网链接造成系统性影响.   根服务器主要用来管理互联网的主 ...

  7. HTML5标准简介

    最近前端的群都蛮热闹的,但我发现多数讨论的是javascript和css相关的问题,仿佛大家在努力创建各种交互.样式的时候,忘却了这一切的基础 – HTML. 其实我很喜欢HTML,觉得这个语言远比X ...

  8. sql 两列相加存到另一列

    假设表table1有a.b两个列,想生成另一个列为a列值+b列值计算列添加语句如下ALTER TABLE table1ADD c AS a+b

  9. No.012:Integer to Roman

    题目: Given an integer, convert it to a roman numeral.Input is guaranteed to be within the range from ...

  10. MyBatis插入语句返回主键值

    插入语句xml代码: <insert id="insertUser" parameterType="com.spring.mybatis.po.User" ...