转自:http://mobile.51cto.com/design-375889.htm

今天教大家写一个类似于Android桌面的launcher效果的自定义控件,在开始写之前大家需要熟悉几个类和它们的方法,下面我分别列出来:

一.VelocityTracker 速度追踪器

顾名思义这个类的作用主要是追踪用户手指在屏幕上的滑动速度。当你要跟踪一个touch事件的时候,使用obtain()方法得到这个类的实 例,然后 用addMovement(MotionEvent)函数将你接受到的motion event加入到VelocityTracker类实例中。当你使用到速率时,使用computeCurrentVelocity(int)初始化速率的 单位,并获得当前的事件的速率,然后使用getXVelocity() 或getXVelocity()获得横向和竖向的速率。

二.ViewConfiguration

这个类里面定义了android的许多标准的常量(UI的超时、大小和距离等)。

三.GestureDetector 手势识别器

这个类主要是追踪用户手指在屏幕上的滑动方向,这个类在我们马上要实现的类中没有使用,但是使用的原理和它差不多,所以顺便提一下,而且在以后的开发中,这个类也是经常使用的。

四.Scroller

这个类主要是支持view控件滑动,其实android很多可滑动的控件里面默认隐藏的就是这个类。而且这个类没有进行实际的视图移动,当调用它的 startScroll()方法实际上只是为了在父类调用computeScroll()方法前开始动画,也就是说这个类实际上就是相当于一个代理,值是 为了给后面视图移动添加一些动画效果。所以单独调用startScroll()而不重写computeScroll()方法是不会看到任何效果的。这两者 必须配合使用,才能有移动的时候的动画效果。

其中Scroller.computeScrollOffset()方法是判断scroller的移动动画是否完成,当你调用startScroll()方法的时候这个方法返回的值一直都为true,如果采用其它方式移动视图比如:scrollTo()或 scrollBy时那么这个方法返回false。

现在来讲讲startScroll(int startX, int startY, int dx, int dy, int duration)方法的四个参数的意思:

  • startX表示当前视图的x坐标值
  • startY表示当前视图的y坐标值
  • dx表示在当前视图的x坐标基础上横向移动的距离
  • dy表示在当前视图的y坐标基础上纵向移动的距离
  • duration表示视图移动的操作在多少时间内执行完场,也就是动画的持续时间(单位:毫秒)

五.ViewGroup

这是个特殊的View,它继承于Android.view.View,它的功能就是装载和管理下一层的View对象或ViewGroup对象,也就说他是一个容纳其它元素的的容器。

下面我们来分别分析我们要使用这5个类的那些方法,首先我们来看ViewGroup类,因为我们自定义的控件就是继承至这个类,我们会重写这个类中的5个方法如下:

1.onLayout(boolean changed, int l, int t, int r, int b)

这个方法是在onMeasure()方法执行后调用,作用是父类为子类在屏幕上分配实际的宽度和高度。里面的四个参数分别表示,布局是否发生改变,布局左 上右下的边距。

2.onMeasure(int widthMeasureSpec, int heightMeasureSpec)

这个方法在控件的父元素正要放置它的子控件时调用。然后传入两个参数——widthMeasureSpec和 heightMeasureSpec。它们指明控件可获得的空间以及关于这个空间描述的元数据。比返回一个结果要好的方法是你传递View的高度和宽度到 setMeasuredDimension方法里。widthMeasureSpec和heightMeasureSpec参数在它们使用之前,首先要做 的是使用MeasureSpec类的静态方法getMode和getSize来译解。一个MeasureSpec包含一个尺寸和模式。

有三种可能的模式:

  • UNSPECIFIED:父布局没有给子布局任何限制,子布局可以任意大小。
  • EXACTLY:父布局决定子布局的确切大小。不论子布局多大,它都必须限制在这个界限里。(当布局定义为一个固定像素或者fill_parent时就是EXACTLY模式)
  • AT_MOST:子布局可以根据自己的大小选择任意大小。(当布局定义为wrap_content时就是AT_MOST模式)

3.computeScroll()

这个方法主要是父类要求它的子类滚动的时候调用。在这个方法里,我们可以实现 view的滚动操作,这里滚动并不是view的滚动而是布局的滚动。当调用scroller的startScroll()方法后父类就会调用这个方法实现 滚动视图滚动操作。

4.onTouchEvent(MotionEvent event)

处理传递到view 的手势事件。手势事件类型包括ACTION_DOWN,ACTION_MOVE,ACTION_UP,ACTION_CANCEL等事件。Layout里 的onTouch默认返回值是false, View里的onTouch默认返回值是true,当我们手指点击屏幕时候,先调用ACTION_DOWN事件,当onTouch里返回值是true的时 候,onTouch回继续调用ACTION_UP事件,如果onTouch里返回值是false,那么onTouch只会调用ACTION_DOWN而不调用ACTION_UP.

5.onInterceptTouchEvent(MotionEvent ev)

用于拦截手势事件的,每个手势事件都会先调用这个方法。Layout里的onInterceptTouchEvent默认返回值是false,这样touch事件会传递到View控件。

下面再将几个大家可能比较混乱的方法说明一下:

Invalidate()和PostInvalidate(),这两个方法作用都一样,就是呼叫ui线程重新绘制 界面也就是刷新界面。那为什么要两个方法呢,这是因为android是多线程应用,大家应该都知道在非UI线程中是不能直接操作界面控件的,所以第2个方 法就帮助大家在子线程中刷行界面,第一个方法则是在UI线程中刷新界面。

getX()和getRawX()这两个方法的左右都是获取当前点在屏幕上的坐标,getX()是获取当前点相对于当前视图左上角的坐标,getRawX()则是获取当前点相对于手机屏幕左上角的坐标。

上面已经把我们要用到的类和方法做了详细描述,下面就是实现的源码:

  1. importandroid.content.Context;
  2. importandroid.util.AttributeSet;
  3. importandroid.view.MotionEvent;
  4. importandroid.view.VelocityTracker;
  5. importandroid.view.View;
  6. importandroid.view.ViewConfiguration;
  7. importandroid.view.ViewGroup;
  8. importandroid.widget.Scroller;
  9. /**
  10. *@author
  11. */
  12. publicclassScrollLayoutextendsViewGroup{
  13. privateScrollermScroller;
  14. privateVelocityTrackermVelocityTracker;
  15. /**
  16. *当前的屏幕位置
  17. */
  18. privateintmCurScreen;
  19. /**
  20. *设置默认屏幕的属性,0表示第一个屏幕
  21. */
  22. privateintmDefaultScreen=0;
  23. /**
  24. *标识滚动操作已结束
  25. */
  26. privatestaticfinalintTOUCH_STATE_REST=0;
  27. /**
  28. *标识正在执行滑动操作
  29. */
  30. privatestaticfinalintTOUCH_STATE_SCROLLING=1;
  31. /**
  32. *标识滑动速率
  33. */
  34. privatestaticfinalintSNAP_VELOCITY=600;
  35. /**
  36. *当前滑动状态
  37. */
  38. privateintmTouchState=TOUCH_STATE_REST;
  39. /**
  40. *在用户触发ontouch事件之前,我们认为用户能够使view滑动的距离(像素)
  41. */
  42. privateintmTouchSlop;
  43. /**
  44. *手指触碰屏幕的最后一次x坐标
  45. */
  46. privatefloatmLastMotionX;
  47. /**
  48. *手指触碰屏幕的最后一次y坐标
  49. */
  50. @SuppressWarnings("unused")
  51. privatefloatmLastMotionY;
  52. publicScrollLayout(Contextcontext){
  53. super(context);
  54. mScroller=newScroller(context);
  55. mCurScreen=mDefaultScreen;
  56. mTouchSlop=ViewConfiguration.get(getContext()).getScaledTouchSlop();
  57. }
  58. publicScrollLayout(Contextcontext,AttributeSetattrs){
  59. super(context,attrs);
  60. mScroller=newScroller(context);
  61. mCurScreen=mDefaultScreen;
  62. mTouchSlop=ViewConfiguration.get(getContext()).getScaledTouchSlop();
  63. }
  64. publicScrollLayout(Contextcontext,AttributeSetattrs,intdefStyle){
  65. super(context,attrs,defStyle);
  66. mScroller=newScroller(context);
  67. mCurScreen=mDefaultScreen;
  68. mTouchSlop=ViewConfiguration.get(getContext()).getScaledTouchSlop();
  69. }
  70. @Override
  71. protectedvoidonLayout(booleanchanged,intl,intt,intr,intb){
  72. if(changed){
  73. intchildLeft=0;
  74. finalintchildCount=getChildCount();
  75. for(inti=0;i<childCount;i++){
  76. finalViewchildView=getChildAt(i);
  77. if(childView.getVisibility()!=View.GONE){
  78. finalintchildWidth=childView.getMeasuredWidth();
  79. childView.layout(childLeft,0,childLeft+childWidth,
  80. childView.getMeasuredHeight());
  81. childLeft+=childWidth;
  82. }
  83. }
  84. }
  85. }
  86. @Override
  87. protectedvoidonMeasure(intwidthMeasureSpec,intheightMeasureSpec){
  88. super.onMeasure(widthMeasureSpec,heightMeasureSpec);
  89. finalintwidth=MeasureSpec.getSize(widthMeasureSpec);
  90. finalintwidthMode=MeasureSpec.getMode(widthMeasureSpec);
  91. if(widthMode!=MeasureSpec.EXACTLY){
  92. thrownewIllegalStateException(
  93. "ScrollLayoutonlycanmCurScreenrunatEXACTLYmode!");
  94. }
  95. finalintheightMode=MeasureSpec.getMode(heightMeasureSpec);
  96. if(heightMode!=MeasureSpec.EXACTLY){
  97. thrownewIllegalStateException(
  98. "ScrollLayoutonlycanrunatEXACTLYmode!");
  99. }
  100. finalintcount=getChildCount();
  101. for(inti=0;i<count;i++){
  102. getChildAt(i).measure(widthMeasureSpec,heightMeasureSpec);
  103. }
  104. //初始化视图的位置
  105. scrollTo(mCurScreen*width,0);
  106. }
  107. /**
  108. *根据滑动的距离判断移动到第几个视图
  109. */
  110. publicvoidsnapToDestination(){
  111. finalintscreenWidth=getWidth();
  112. finalintdestScreen=(getScrollX()+screenWidth/2)/screenWidth;
  113. snapToScreen(destScreen);
  114. }
  115. /**
  116. *滚动到制定的视图
  117. *
  118. *@paramwhichScreen
  119. *视图下标
  120. */
  121. publicvoidsnapToScreen(intwhichScreen){
  122. whichScreen=Math.max(0,Math.min(whichScreen,getChildCount()-1));
  123. if(getScrollX()!=(whichScreen*getWidth())){
  124. finalintdelta=whichScreen*getWidth()-getScrollX();
  125. mScroller.startScroll(getScrollX(),0,delta,0,1000);
  126. mCurScreen=whichScreen;
  127. invalidate();
  128. }
  129. }
  130. publicvoidsetToScreen(intwhichScreen){
  131. whichScreen=Math.max(0,Math.min(whichScreen,getChildCount()-1));
  132. mCurScreen=whichScreen;
  133. scrollTo(whichScreen*getWidth(),0);
  134. }
  135. publicintgetCurScreen(){
  136. returnmCurScreen;
  137. }
  138. @Override
  139. publicvoidcomputeScroll(){
  140. if(mScroller.computeScrollOffset()){
  141. scrollTo(mScroller.getCurrX(),mScroller.getCurrY());
  142. postInvalidate();
  143. }
  144. }
  145. @Override
  146. publicbooleanonTouchEvent(MotionEventevent){
  147. if(mVelocityTracker==null){
  148. mVelocityTracker=VelocityTracker.obtain();
  149. }
  150. mVelocityTracker.addMovement(event);
  151. finalintaction=event.getAction();
  152. finalfloatx=event.getX();
  153. switch(action){
  154. caseMotionEvent.ACTION_DOWN:
  155. if(!mScroller.isFinished()){
  156. mScroller.abortAnimation();
  157. }
  158. mLastMotionX=x;
  159. break;
  160. caseMotionEvent.ACTION_MOVE:
  161. intdeltaX=(int)(mLastMotionX-x);
  162. mLastMotionX=x;
  163. scrollBy(deltaX,0);
  164. break;
  165. caseMotionEvent.ACTION_UP:
  166. finalVelocityTrackervelocityTracker=mVelocityTracker;
  167. velocityTracker.computeCurrentVelocity(1000);
  168. intvelocityX=(int)velocityTracker.getXVelocity();
  169. if(velocityX>SNAP_VELOCITY&&mCurScreen>0){
  170. //向左移动
  171. snapToScreen(mCurScreen-1);
  172. }elseif(velocityX<-SNAP_VELOCITY
  173. &&mCurScreen<getChildCount()-1){
  174. //向右移动
  175. snapToScreen(mCurScreen+1);
  176. }else{
  177. snapToDestination();
  178. }
  179. if(mVelocityTracker!=null){
  180. mVelocityTracker.recycle();
  181. mVelocityTracker=null;
  182. }
  183. mTouchState=TOUCH_STATE_REST;
  184. break;
  185. caseMotionEvent.ACTION_CANCEL:
  186. mTouchState=TOUCH_STATE_REST;
  187. break;
  188. }
  189. returntrue;
  190. }
  191. @Override
  192. publicbooleanonInterceptTouchEvent(MotionEventev){
  193. finalintaction=ev.getAction();
  194. if((action==MotionEvent.ACTION_MOVE)
  195. &&(mTouchState!=TOUCH_STATE_REST)){
  196. returntrue;
  197. }
  198. finalfloatx=ev.getX();
  199. finalfloaty=ev.getY();
  200. switch(action){
  201. caseMotionEvent.ACTION_MOVE:
  202. finalintxDiff=(int)Math.abs(mLastMotionX-x);
  203. if(xDiff>mTouchSlop){
  204. mTouchState=TOUCH_STATE_SCROLLING;
  205. }
  206. break;
  207. caseMotionEvent.ACTION_DOWN:
  208. mLastMotionX=x;
  209. mLastMotionY=y;
  210. mTouchState=mScroller.isFinished()?TOUCH_STATE_REST
  211. :TOUCH_STATE_SCROLLING;
  212. break;
  213. caseMotionEvent.ACTION_CANCEL:
  214. caseMotionEvent.ACTION_UP:
  215. mTouchState=TOUCH_STATE_REST;
  216. break;
  217. }
  218. returnmTouchState!=TOUCH_STATE_REST;
  219. }
  220. }

更多相关文章

  1. 【边做项目边学Android】小白会遇到的问题--创建Android项目不自
  2. Android(安卓)SharedPreferences总结及优化
  3. android之listview使用方法(一)
  4. Android四大组件的工作过程
  5. Android监听通话正确操作方法介绍
  6. android中的对话框之三:自定义对话框
  7. Android的FrameLayout使用要注意的问题
  8. Android(安卓)Toast使用简介
  9. 从头到尾给你讲明白Android(安卓)View实现原理

随机推荐

  1. Android中的四种启动方式
  2. Android studio使用Lottie- 让Android动
  3. android 自定义dialog,窗口动画
  4. Android 线程池相关知识
  5. PHP将数据库查询内容转换为JSON格式且显
  6. Android(安卓)APP与媒体存储服务的交互
  7. android Eclipse开发问题汇总
  8. 【精华】Android应用程序框架分析
  9. 二十三、Android源代码是这样搞到的(图解
  10. android clipChildren与clipToPadding