一、首先把横向的listview的代码放上来 

  1. HorizontalListView
  1. package com.common.cklibrary.utils.myview;
  2.  
  3. import java.util.LinkedList;
  4. import java.util.Queue;
  5.  
  6. import android.content.Context;
  7. import android.database.DataSetObserver;
  8. import android.graphics.Rect;
  9. import android.util.AttributeSet;
  10. import android.view.GestureDetector;
  11. import android.view.GestureDetector.OnGestureListener;
  12. import android.view.MotionEvent;
  13. import android.view.View;
  14. import android.widget.AdapterView;
  15. import android.widget.ListAdapter;
  16. import android.widget.Scroller;
  17.  
  18. /**
  19. * Created by Admin on 2017/9/7.
  20. */
  21. @SuppressWarnings("deprecation")
  22. public class HorizontalListView extends AdapterView<ListAdapter> {
  23.  
  24. public boolean mAlwaysOverrideTouch = true;
  25. protected ListAdapter mAdapter;
  26. private int mLeftViewIndex = -1;
  27. private int mRightViewIndex = 0;
  28. protected int mCurrentX;
  29. protected int mNextX;
  30. private int mMaxX = Integer.MAX_VALUE;
  31. private int mDisplayOffset = 0;
  32. protected Scroller mScroller;
  33. private GestureDetector mGesture;
  34. private Queue<View> mRemovedViewQueue = new LinkedList<View>();
  35. private OnItemSelectedListener mOnItemSelected;
  36. private OnItemClickListener mOnItemClicked;
  37. private OnItemLongClickListener mOnItemLongClicked;
  38. private boolean mDataChanged = false;
  39.  
  40. public HorizontalListView(Context context, AttributeSet attrs) {
  41. super(context, attrs);
  42. initView();
  43. }
  44.  
  45. private synchronized void initView() {
  46. mLeftViewIndex = -1;
  47. mRightViewIndex = 0;
  48. mDisplayOffset = 0;
  49. mCurrentX = 0;
  50. mNextX = 0;
  51. mMaxX = Integer.MAX_VALUE;
  52. mScroller = new Scroller(getContext());
  53. mGesture = new GestureDetector(getContext(), mOnGesture);
  54. }
  55.  
  56. @Override
  57. public void setOnItemSelectedListener(AdapterView.OnItemSelectedListener listener) {
  58. mOnItemSelected = listener;
  59. }
  60.  
  61. @Override
  62. public void setOnItemClickListener(AdapterView.OnItemClickListener listener) {
  63. mOnItemClicked = listener;
  64. }
  65.  
  66. @Override
  67. public void setOnItemLongClickListener(AdapterView.OnItemLongClickListener listener) {
  68. mOnItemLongClicked = listener;
  69. }
  70.  
  71. private DataSetObserver mDataObserver = new DataSetObserver() {
  72.  
  73. @Override
  74. public void onChanged() {
  75. synchronized (HorizontalListView.this) {
  76. mDataChanged = true;
  77. }
  78. invalidate();
  79. requestLayout();
  80. }
  81.  
  82. @Override
  83. public void onInvalidated() {
  84. reset();
  85. invalidate();
  86. requestLayout();
  87. }
  88.  
  89. };
  90.  
  91. @Override
  92. public ListAdapter getAdapter() {
  93. return mAdapter;
  94. }
  95.  
  96. @Override
  97. public View getSelectedView() {
  98. //TODO: implement
  99. return null;
  100. }
  101.  
  102. @Override
  103. public void setAdapter(ListAdapter adapter) {
  104. if (mAdapter != null) {
  105. mAdapter.unregisterDataSetObserver(mDataObserver);
  106. }
  107. mAdapter = adapter;
  108. mAdapter.registerDataSetObserver(mDataObserver);
  109. reset();
  110. }
  111.  
  112. private synchronized void reset() {
  113. initView();
  114. removeAllViewsInLayout();
  115. requestLayout();
  116. }
  117.  
  118. @Override
  119. public void setSelection(int position) {
  120. //TODO: implement
  121. }
  122.  
  123. private void addAndMeasureChild(final View child, int viewPos) {
  124. LayoutParams params = child.getLayoutParams();
  125. if (params == null) {
  126. params = new LayoutParams(LayoutParams.FILL_PARENT, LayoutParams.FILL_PARENT);
  127. }
  128.  
  129. addViewInLayout(child, viewPos, params, true);
  130. child.measure(MeasureSpec.makeMeasureSpec(getWidth(), MeasureSpec.AT_MOST),
  131. MeasureSpec.makeMeasureSpec(getHeight(), MeasureSpec.AT_MOST));
  132. }
  133.  
  134. @Override
  135. protected synchronized void onLayout(boolean changed, int left, int top, int right, int bottom) {
  136. super.onLayout(changed, left, top, right, bottom);
  137.  
  138. if (mAdapter == null) {
  139. return;
  140. }
  141.  
  142. if (mDataChanged) {
  143. int oldCurrentX = mCurrentX;
  144. initView();
  145. removeAllViewsInLayout();
  146. mNextX = oldCurrentX;
  147. mDataChanged = false;
  148. }
  149.  
  150. if (mScroller.computeScrollOffset()) {
  151. int scrollx = mScroller.getCurrX();
  152. mNextX = scrollx;
  153. }
  154.  
  155. if (mNextX <= 0) {
  156. mNextX = 0;
  157. mScroller.forceFinished(true);
  158. }
  159. if (mNextX >= mMaxX) {
  160. mNextX = mMaxX;
  161. mScroller.forceFinished(true);
  162. }
  163.  
  164. int dx = mCurrentX - mNextX;
  165.  
  166. removeNonVisibleItems(dx);
  167. fillList(dx);
  168. positionItems(dx);
  169.  
  170. mCurrentX = mNextX;
  171.  
  172. if (!mScroller.isFinished()) {
  173. post(new Runnable() {
  174. @Override
  175. public void run() {
  176. requestLayout();
  177. }
  178. });
  179.  
  180. }
  181. }
  182.  
  183. private void fillList(final int dx) {
  184. int edge = 0;
  185. View child = getChildAt(getChildCount() - 1);
  186. if (child != null) {
  187. edge = child.getRight();
  188. }
  189. fillListRight(edge, dx);
  190.  
  191. edge = 0;
  192. child = getChildAt(0);
  193. if (child != null) {
  194. edge = child.getLeft();
  195. }
  196. fillListLeft(edge, dx);
  197.  
  198. }
  199.  
  200. private void fillListRight(int rightEdge, final int dx) {
  201. while (rightEdge + dx < getWidth() && mRightViewIndex < mAdapter.getCount()) {
  202.  
  203. View child = mAdapter.getView(mRightViewIndex, mRemovedViewQueue.poll(), this);
  204. addAndMeasureChild(child, -1);
  205. rightEdge += child.getMeasuredWidth();
  206.  
  207. if (mRightViewIndex == mAdapter.getCount() - 1) {
  208. mMaxX = mCurrentX + rightEdge - getWidth();
  209. }
  210.  
  211. if (mMaxX < 0) {
  212. mMaxX = 0;
  213. }
  214. mRightViewIndex++;
  215. }
  216.  
  217. }
  218.  
  219. private void fillListLeft(int leftEdge, final int dx) {
  220. while (leftEdge + dx > 0 && mLeftViewIndex >= 0) {
  221. View child = mAdapter.getView(mLeftViewIndex, mRemovedViewQueue.poll(), this);
  222. addAndMeasureChild(child, 0);
  223. leftEdge -= child.getMeasuredWidth();
  224. mLeftViewIndex--;
  225. mDisplayOffset -= child.getMeasuredWidth();
  226. }
  227. }
  228.  
  229. private void removeNonVisibleItems(final int dx) {
  230. View child = getChildAt(0);
  231. while (child != null && child.getRight() + dx <= 0) {
  232. mDisplayOffset += child.getMeasuredWidth();
  233. mRemovedViewQueue.offer(child);
  234. removeViewInLayout(child);
  235. mLeftViewIndex++;
  236. child = getChildAt(0);
  237.  
  238. }
  239.  
  240. child = getChildAt(getChildCount() - 1);
  241. while (child != null && child.getLeft() + dx >= getWidth()) {
  242. mRemovedViewQueue.offer(child);
  243. removeViewInLayout(child);
  244. mRightViewIndex--;
  245. child = getChildAt(getChildCount() - 1);
  246. }
  247. }
  248.  
  249. private void positionItems(final int dx) {
  250. if (getChildCount() > 0) {
  251. mDisplayOffset += dx;
  252. int left = mDisplayOffset;
  253. for (int i = 0; i < getChildCount(); i++) {
  254. View child = getChildAt(i);
  255. int childWidth = child.getMeasuredWidth();
  256. child.layout(left, 0, left + childWidth, child.getMeasuredHeight());
  257. left += childWidth + child.getPaddingRight();
  258. }
  259. }
  260. }
  261.  
  262. public synchronized void scrollTo(int x) {
  263. mScroller.startScroll(mNextX, 0, x - mNextX, 0);
  264. requestLayout();
  265. }
  266.  
  267. @Override
  268. public boolean dispatchTouchEvent(MotionEvent ev) {
  269. boolean handled = super.dispatchTouchEvent(ev);
  270. handled |= mGesture.onTouchEvent(ev);
  271. return handled;
  272. }
  273.  
  274. private OnGestureListener mOnGesture = new GestureDetector.SimpleOnGestureListener() {
  275.  
  276. @Override
  277. public boolean onSingleTapUp(MotionEvent e) {
  278. getParent().requestDisallowInterceptTouchEvent(false);
  279. return false;
  280. }
  281.  
  282. @Override
  283. public boolean onDown(MotionEvent e) {
  284. mScroller.forceFinished(true);
  285. return true;
  286. }
  287.  
  288. @Override
  289. public boolean onFling(MotionEvent e1, MotionEvent e2, float velocityX,
  290. float velocityY) {
  291. synchronized (HorizontalListView.this) {
  292. mScroller.fling(mNextX, 0, (int) -velocityX, 0, 0, mMaxX, 0, 0);
  293. }
  294. requestLayout();
  295. if (Math.abs(velocityX)>Math.abs(velocityY)){
  296. return true;
  297. }else{
  298. getParent().requestDisallowInterceptTouchEvent(false);
  299. return false;
  300. }
  301. }
  302.  
  303. @Override
  304. public boolean onScroll(MotionEvent e1, MotionEvent e2,
  305. float distanceX, float distanceY) {
  306.  
  307. if (Math.abs(distanceY)>20&&Math.abs(distanceY)>Math.abs(distanceX)){
  308. getParent().requestDisallowInterceptTouchEvent(false);
  309. return false;
  310. }else{
  311. synchronized (HorizontalListView.this) {
  312. mNextX += (int) distanceX;
  313. }
  314. requestLayout();
  315. return true;
  316. }
  317.  
  318. }
  319.  
  320. @Override
  321. public boolean onSingleTapConfirmed(MotionEvent e) {
  322. for (int i = 0; i < getChildCount(); i++) {
  323. View child = getChildAt(i);
  324. if (isEventWithinView(e, child)) {
  325. if (mOnItemClicked != null) {
  326. mOnItemClicked.onItemClick(HorizontalListView.this, child, mLeftViewIndex + 1 + i, mAdapter.getItemId(mLeftViewIndex + 1 + i));
  327. }
  328. if (mOnItemSelected != null) {
  329. mOnItemSelected.onItemSelected(HorizontalListView.this, child, mLeftViewIndex + 1 + i, mAdapter.getItemId(mLeftViewIndex + 1 + i));
  330. }
  331. break;
  332. }
  333.  
  334. }
  335. getParent().requestDisallowInterceptTouchEvent(false);
  336. return false;
  337. // return super.onSingleTapConfirmed(e);
  338. }
  339.  
  340. @Override
  341. public void onLongPress(MotionEvent e) {
  342. int childCount = getChildCount();
  343. for (int i = 0; i < childCount; i++) {
  344. View child = getChildAt(i);
  345. if (isEventWithinView(e, child)) {
  346. if (mOnItemLongClicked != null) {
  347. mOnItemLongClicked.onItemLongClick(HorizontalListView.this, child, mLeftViewIndex + 1 + i, mAdapter.getItemId(mLeftViewIndex + 1 + i));
  348. }
  349. break;
  350. }
  351.  
  352. }
  353. getParent().requestDisallowInterceptTouchEvent(false);
  354. }
  355.  
  356. private boolean isEventWithinView(MotionEvent e, View child) {
  357. Rect viewRect = new Rect();
  358. int[] childPosition = new int[2];
  359. child.getLocationOnScreen(childPosition);
  360. int left = childPosition[0];
  361. int right = left + child.getWidth();
  362. int top = childPosition[1];
  363. int bottom = top + child.getHeight();
  364. viewRect.set(left, top, right, bottom);
  365. return viewRect.contains((int) e.getRawX(), (int) e.getRawY());
  366. }
  367. };
  368. }

二、我们来了解一下屏幕的触摸监听事件的分发机制

Touch事件分发中只有两个主角:ViewGroup和View。Activity的Touch事件事实上是调用它内部的ViewGroup的Touch事件,可以直接当成ViewGroup处理。

View在ViewGroup内,ViewGroup也可以在其他ViewGroup内,这时候把内部的ViewGroup当成View来分析。

ViewGroup的相关事件有三个:onInterceptTouchEvent、dispatchTouchEvent、onTouchEvent。View的相关事件只有两个:dispatchTouchEvent、onTouchEvent。

先分析ViewGroup的处理流程:首先得有个结构模型概念:ViewGroup和View组成了一棵树形结构,最顶层为Activity的ViewGroup,下面有若干的ViewGroup节点,每个节点之下又有若干的ViewGroup节点或者View节点,依次类推。如图:

当一个Touch事件(触摸事件为例)到达根节点,即Acitivty的ViewGroup时,它会依次下发,下发的过程是调用子View(ViewGroup)的dispatchTouchEvent方法实现的。简单来说,就是ViewGroup遍历它包含着的子View,调用每个View的dispatchTouchEvent方法,而当子View为ViewGroup时,又会通过调用ViwGroup的dispatchTouchEvent方法继续调用其内部的View的dispatchTouchEvent方法。上述例子中的消息下发顺序是这样的:①-②-⑤-⑥-⑦-③-④。dispatchTouchEvent方法只负责事件的分发,它拥有boolean类型的返回值,当返回为true时,顺序下发会中断。在上述例子中如果⑤的dispatchTouchEvent返回结果为true,那么⑥-⑦-③-④将都接收不到本次Touch事件。来个简单版的代码加深理解:

  1. /**
  2. * ViewGroup
  3. * @param ev
  4. * @return
  5. */
  6. public boolean dispatchTouchEvent(MotionEvent ev){
  7. ....//其他处理,在此不管
  8. View[] views=getChildView();
  9. for(int i=0;i<views.length;i++){
    //判断下Touch到屏幕上的点在该子View上面 
    if(...){
  10. if(views[i].dispatchTouchEvent(ev))
  11. return true;
    }
  12. }
  13. ...//其他处理,在此不管
  14. }
  15. /**
  16. * View
  17. * @param ev
  18. * @return
  19. */
  20. public boolean dispatchTouchEvent(MotionEvent ev){
  21. ....//其他处理,在此不管
  22. return false;
  23. }

在此可以看出,ViewGroup的dispatchTouchEvent是真正在执行“分发”工作,而View的dispatchTouchEvent方法,并不执行分发工作,或者说它分发的对象就是自己,决定是否把touch事件交给自己处理,而处理的方法,便是onTouchEvent事件,事实上子View的dispatchTouchEvent方法真正执行的代码是这样的

  1. /**
  2. * View
  3. * @param ev
  4. * @return
  5. */
  6. public boolean dispatchTouchEvent(MotionEvent ev){
  7. ....//其他处理,在此不管
  8. return onTouchEvent(event);
  9. }

一般情况下,我们不该在普通View内重写dispatchTouchEvent方法,因为它并不执行分发逻辑。当Touch事件到达View时,我们该做的就是是否在onTouchEvent事件中处理它。

那么,ViewGroup的onTouchEvent事件是什么时候处理的呢?当ViewGroup所有的子View都返回false时,onTouchEvent事件便会执行。由于ViewGroup是继承于View的,它其实也是通过调用View的dispatchTouchEvent方法来执行onTouchEvent事件。

在目前的情况看来,似乎只要我们把所有的onTouchEvent都返回false,就能保证所有的子控件都响应本次Touch事件了。但必须要说明的是,这里的Touch事件,只限于Acition_Down事件,即触摸按下事件,而Aciton_UP和Action_MOVE却不会执行。事实上,一次完整的Touch事件,应该是由一个Down、一个Up和若干个Move组成的。Down方式通过dispatchTouchEvent分发,分发的目的是为了找到真正需要处理完整Touch请求的View。当某个View或者ViewGroup的onTouchEvent事件返回true时,便表示它是真正要处理这次请求的View,之后的Aciton_UP和Action_MOVE将由它处理。当所有子View的onTouchEvent都返回false时,这次的Touch请求就由根ViewGroup,即Activity自己处理了。

看看改进后的ViewGroup的dispatchTouchEvent方法

  1. View mTarget=null;//保存捕获Touch事件处理的View
  2. public boolean dispatchTouchEvent(MotionEvent ev) {
  3.  
  4. //....其他处理,在此不管
  5.  
  6. if(ev.getAction()==KeyEvent.ACTION_DOWN){
  7. //每次Down事件,都置为Null
  8.  
  9. if(!onInterceptTouchEvent()){
    mTarget=null;
  10. View[] views=getChildView();
  11. for(int i=0;i<views.length;i++){
  12. if(views[i].dispatchTouchEvent(ev))
  13. mTarget=views[i];
  14. return true;
  15. }
    }
  16. }
  17. //当子View没有捕获down事件时,ViewGroup自身处理。这里处理的Touch事件包含Down、Up和Move
  18. if(mTarget==null){
  19. return super.dispatchTouchEvent(ev);
  20. }
  21. //...其他处理,在此不管
  22. if(onInterceptTouchEvent()){
  1. //...其他处理,在此不管
    }
  1. //这一步在Action_Down中是不会执行到的,只有Move和UP才会执行到。
  2. return mTarget.dispatchTouchEvent(ev);
  3.  
  4. }

ViewGroup还有个onInterceptTouchEvent,看名字便知道这是个拦截事件。这个拦截事件需要分两种情况来说明:

1.假如我们在某个ViewGroup的onInterceptTouchEvent中,将Action为Down的Touch事件返回true,那便表示将该ViewGroup的所有下发操作拦截掉,这种情况下,mTarget会一直为null,因为mTarget是在Down事件中赋值的。由于mTarge为null,该ViewGroup的onTouchEvent事件被执行。这种情况下可以把这个ViewGroup直接当成View来对待。

2.假如我们在某个ViewGroup的onInterceptTouchEvent中,将Acion为Down的Touch事件都返回false,其他的都返回True,这种情况下,Down事件能正常分发,若子View都返回false,那mTarget还是为空,无影响。若某个子View返回了true,mTarget被赋值了,在Action_Move和Aciton_UP分发到该ViewGroup时,便会给mTarget分发一个Action_Delete的MotionEvent,同时清空mTarget的值,使得接下去的Action_Move(如果上一个操作不是UP)将由ViewGroup的onTouchEvent处理。

情况一用到的比较多,情况二个人还未找到使用场景。

从头到尾总结一下:

1.Touch事件分发中只有两个主角:ViewGroup和View。ViewGroup包含onInterceptTouchEvent、dispatchTouchEvent、onTouchEvent三个相关事件。View包含dispatchTouchEvent、onTouchEvent两个相关事件。其中ViewGroup又继承于View。

2.ViewGroup和View组成了一个树状结构,根节点为Activity内部包含的一个ViwGroup。

3.触摸事件由Action_Down、Action_Move、Aciton_UP组成,其中一次完整的触摸事件中,Down和Up都只有一个,Move有若干个,可以为0个。

4.当Acitivty接收到Touch事件时,将遍历子View进行Down事件的分发。ViewGroup的遍历可以看成是递归的。分发的目的是为了找到真正要处理本次完整触摸事件的View,这个View会在onTouchuEvent结果返回true。

5.当某个子View返回true时,会中止Down事件的分发,同时在ViewGroup中记录该子View。接下去的Move和Up事件将由该子View直接进行处理。由于子View是保存在ViewGroup中的,多层ViewGroup的节点结构时,上级ViewGroup保存的会是真实处理事件的View所在的ViewGroup对象:如ViewGroup0-ViewGroup1-TextView的结构中,TextView返回了true,它将被保存在ViewGroup1中,而ViewGroup1也会返回true,被保存在ViewGroup0中。当Move和UP事件来时,会先从ViewGroup0传递至ViewGroup1,再由ViewGroup1传递至TextView。

6.当ViewGroup中所有子View都不捕获Down事件时,将触发ViewGroup自身的onTouch事件。触发的方式是调用super.dispatchTouchEvent函数,即父类View的dispatchTouchEvent方法。在所有子View都不处理的情况下,触发Acitivity的onTouchEvent方法。

7.onInterceptTouchEvent有两个作用:1.拦截Down事件的分发。2.中止Up和Move事件向目标View传递,使得目标View所在的ViewGroup捕获Up和Move事件。

另外,上文所列出的代码并非真正的源码,只是概括了源码在事件分发处理中的核心处理流程,真正源码各位可以自己去看,包含了更丰富的内容。

补充:

“触摸事件由Action_Down、Action_Move、Aciton_UP组成,其中一次完整的触摸事件中,Down和Up都只有一个,Move有若干个,可以为0个。”,这里补充下其实UP事件是可能为0个的。

三、用户手势检测-GestureDetector使用详解

一、概述

当用户触摸屏幕的时候,会产生许多手势,例如down,up,scroll,filing等等。
一般情况下,我们知道View类有个View.OnTouchListener内部接口,通过重写他的onTouch(View v, MotionEvent event)方法,我们可以处理一些touch事件,但是这个方法太过简单,如果需要处理一些复杂的手势,用这个接口就会很麻烦(因为我们要自己根据用户触摸的轨迹去判断是什么手势)。
Android sdk给我们提供了GestureDetector(Gesture:手势Detector:识别)类,通过这个类我们可以识别很多的手势,主要是通过他的onTouchEvent(event)方法完成了不同手势的识别。虽然他能识别手势,但是不同的手势要怎么处理,应该是提供给程序员实现的。

GestureDetector这个类对外提供了两个接口和一个外部类
接口:OnGestureListener,OnDoubleTapListener
内部类:SimpleOnGestureListener

这个外部类,其实是两个接口中所有函数的集成,它包含了这两个接口里所有必须要实现的函数而且都已经重写,但所有方法体都是空的;不同点在于:该类是static class,程序员可以在外部继承这个类,重写里面的手势处理方法。

下面我们先看OnGestureListener接口;

二、GestureDetector.OnGestureListener---接口

1、基本讲解

如果我们写一个类并implements OnGestureListener,会提示有几个必须重写的函数,加上之后是这个样子的:

  1. private class gesturelistener implements GestureDetector.OnGestureListener{
  2. public boolean onDown(MotionEvent e) {
  3. // TODO Auto-generated method stub
  4. return false;
  5. }
  6. public void onShowPress(MotionEvent e) {
  7. // TODO Auto-generated method stub
  8. }
  9. public boolean onSingleTapUp(MotionEvent e) {
  10. // TODO Auto-generated method stub
  11. return false;
  12. }
  13. public boolean onScroll(MotionEvent e1, MotionEvent e2,
  14. float distanceX, float distanceY) {
  15. // TODO Auto-generated method stub
  16. return false;
  17. }
  18. public void onLongPress(MotionEvent e) {
  19. // TODO Auto-generated method stub
  20. }
  21. public boolean onFling(MotionEvent e1, MotionEvent e2, float velocityX,
  22. float velocityY) {
  23. // TODO Auto-generated method stub
  24. return false;
  25. }
  26. }

可见,这里总共重写了六个函数,这些函数都在什么情况下才会触发呢,下面讲一下:

OnDown(MotionEvent e):用户按下屏幕就会触发;
onShowPress(MotionEvent e):如果是按下的时间超过瞬间,而且在按下的时候没有松开或者是拖动的,那么onShowPress就会执行,具体这个瞬间是多久,我也不清楚呃……
onLongPress(MotionEvent e):长按触摸屏,超过一定时长,就会触发这个事件
    触发顺序:
    onDown->onShowPress->onLongPress
onSingleTapUp(MotionEvent e):从名子也可以看出,一次单独的轻击抬起操作,也就是轻击一下屏幕,立刻抬起来,才会有这个触发,当然,如果除了Down以外还有其它操作,那就不再算是Single操作了,所以也就不会触发这个事件
    触发顺序:
    点击一下非常快的(不滑动)Touchup:
    onDown->onSingleTapUp->onSingleTapConfirmed 
    点击一下稍微慢点的(不滑动)Touchup:
    onDown->onShowPress->onSingleTapUp->onSingleTapConfirmed
onFling(MotionEvent e1, MotionEvent e2, float velocityX,float velocityY) :滑屏,用户按下触摸屏、快速移动后松开,由1个MotionEvent ACTION_DOWN, 多个ACTION_MOVE, 1个ACTION_UP触发   
     参数解释:
    e1:第1个ACTION_DOWN MotionEvent
    e2:最后一个ACTION_MOVE MotionEvent
    velocityX:X轴上的移动速度,像素/秒
    velocityY:Y轴上的移动速度,像素/秒   
onScroll(MotionEvent e1, MotionEvent e2,float distanceX, float distanceY):在屏幕上拖动事件。无论是用手拖动view,或者是以抛的动作滚动,都会多次触发,这个方法       在ACTION_MOVE动作发生时就会触发
    滑屏:手指触动屏幕后,稍微滑动后立即松开
    onDown-----》onScroll----》onScroll----》onScroll----》………----->onFling
    拖动
    onDown------》onScroll----》onScroll------》onFiling

可见,无论是滑屏,还是拖动,影响的只是中间OnScroll触发的数量多少而已,最终都会触发onFling事件!

2、实例

要使用GestureDetector,有三步要走:

1、创建OnGestureListener监听函数:
可以使用构造实例:

  1. GestureDetector.OnGestureListener listener = new GestureDetector.OnGestureListener(){
  2. };

也可以构造类:

  1. private class gestureListener implements GestureDetector.OnGestureListener{
  2. }

2、创建GestureDetector实例mGestureDetector:

构造函数有下面三个,根据需要选择:

  1. GestureDetector gestureDetector=new GestureDetector(GestureDetector.OnGestureListener listener);
  2. GestureDetector gestureDetector=new GestureDetector(Context context,GestureDetector.OnGestureListener listener);
  3. GestureDetector gestureDetector=new GestureDetector(Context context,GestureDetector.SimpleOnGestureListener listener);

3、onTouch(View v, MotionEvent event)中拦截:

  1. public boolean onTouch(View v, MotionEvent event) {
  2. return mGestureDetector.onTouchEvent(event);
  3. }

4、控件绑定

  1. TextView tv = (TextView)findViewById(R.id.tv);
  2. tv.setOnTouchListener(this);

现在进入实例阶段:

首先,在主布局页面添加一个textView,并将其放大到整屏,方便在其上的手势识别,代码为:

  1. <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
  2. xmlns:tools="http://schemas.android.com/tools"
  3. android:layout_width="match_parent"
  4. android:layout_height="match_parent"
  5. tools:context="com.example.gesturedetectorinterface.MainActivity" >
  6. <TextView
  7. android:id="@+id/tv"
  8. android:layout_width="fill_parent"
  9. android:layout_height="fill_parent"
  10. android:layout_margin="50dip"
  11. android:background="#ff00ff"
  12. android:text="@string/hello_world" />
  13. </RelativeLayout>

然后在JAVA代码中,依据上面的三步走原则,写出代码,并在所有的手势下添加上Toast提示并写上Log

  1. public class MainActivity extends Activity implements OnTouchListener{
  2. private GestureDetector mGestureDetector;
  3. @Override
  4. protected void onCreate(Bundle savedInstanceState) {
  5. super.onCreate(savedInstanceState);
  6. setContentView(R.layout.activity_main);
  7. mGestureDetector = new GestureDetector(new gestureListener()); //使用派生自OnGestureListener
  8. TextView tv = (TextView)findViewById(R.id.tv);
  9. tv.setOnTouchListener(this);
  10. tv.setFocusable(true);
  11. tv.setClickable(true);
  12. tv.setLongClickable(true);
  13. }
  14. /*
  15. * 在onTouch()方法中,我们调用GestureDetector的onTouchEvent()方法,将捕捉到的MotionEvent交给GestureDetector
  16. * 来分析是否有合适的callback函数来处理用户的手势
  17. */
  18. public boolean onTouch(View v, MotionEvent event) {
  19. return mGestureDetector.onTouchEvent(event);
  20. }
  21. private class gestureListener implements GestureDetector.OnGestureListener{
  22. // 用户轻触触摸屏,由1个MotionEvent ACTION_DOWN触发
  23. public boolean onDown(MotionEvent e) {
  24. Log.i("MyGesture", "onDown");
  25. Toast.makeText(MainActivity.this, "onDown", Toast.LENGTH_SHORT).show();
  26. return false;
  27. }
  28. /*
  29. * 用户轻触触摸屏,尚未松开或拖动,由一个1个MotionEvent ACTION_DOWN触发
  30. * 注意和onDown()的区别,强调的是没有松开或者拖动的状态
  31. *
  32. * 而onDown也是由一个MotionEventACTION_DOWN触发的,但是他没有任何限制,
  33. * 也就是说当用户点击的时候,首先MotionEventACTION_DOWN,onDown就会执行,
  34. * 如果在按下的瞬间没有松开或者是拖动的时候onShowPress就会执行,如果是按下的时间超过瞬间
  35. * (这块我也不太清楚瞬间的时间差是多少,一般情况下都会执行onShowPress),拖动了,就不执行onShowPress。
  36. */
  37. public void onShowPress(MotionEvent e) {
  38. Log.i("MyGesture", "onShowPress");
  39. Toast.makeText(MainActivity.this, "onShowPress", Toast.LENGTH_SHORT).show();
  40. }
  41. // 用户(轻触触摸屏后)松开,由一个1个MotionEvent ACTION_UP触发
  42. ///轻击一下屏幕,立刻抬起来,才会有这个触发
  43. //从名子也可以看出,一次单独的轻击抬起操作,当然,如果除了Down以外还有其它操作,那就不再算是Single操作了,所以这个事件 就不再响应
  44. public boolean onSingleTapUp(MotionEvent e) {
  45. Log.i("MyGesture", "onSingleTapUp");
  46. Toast.makeText(MainActivity.this, "onSingleTapUp", Toast.LENGTH_SHORT).show();
  47. return true;
  48. }
  49. // 用户按下触摸屏,并拖动,由1个MotionEvent ACTION_DOWN, 多个ACTION_MOVE触发
  50. public boolean onScroll(MotionEvent e1, MotionEvent e2,
  51. float distanceX, float distanceY) {
  52. Log.i("MyGesture22", "onScroll:"+(e2.getX()-e1.getX()) +"   "+distanceX);
  53. Toast.makeText(MainActivity.this, "onScroll", Toast.LENGTH_LONG).show();
  54. return true;
  55. }
  56. // 用户长按触摸屏,由多个MotionEvent ACTION_DOWN触发
  57. public void onLongPress(MotionEvent e) {
  58. Log.i("MyGesture", "onLongPress");
  59. Toast.makeText(MainActivity.this, "onLongPress", Toast.LENGTH_LONG).show();
  60. }
  61. // 用户按下触摸屏、快速移动后松开,由1个MotionEvent ACTION_DOWN, 多个ACTION_MOVE, 1个ACTION_UP触发
  62. public boolean onFling(MotionEvent e1, MotionEvent e2, float velocityX,
  63. float velocityY) {
  64. Log.i("MyGesture", "onFling");
  65. Toast.makeText(MainActivity.this, "onFling", Toast.LENGTH_LONG).show();
  66. return true;
  67. }
  68. };
  69. }

源码在博客底部给出。

三、GestureDetector.OnDoubleTapListener---接口

1、构建

有两种方式设置双击监听:

方法一:新建一个类同时派生自OnGestureListener和OnDoubleTapListener:

  1. private class gestureListener implements GestureDetector.OnGestureListener,GestureDetector.OnDoubleTapListener{
  2. }

方法二:使用GestureDetector::setOnDoubleTapListener();函数设置监听:

  1. //构建GestureDetector实例
  2. mGestureDetector = new GestureDetector(new gestureListener()); //使用派生自OnGestureListener
  3. private class gestureListener implements GestureDetector.OnGestureListener{
  4. }
  5. //设置双击监听器
  6. mGestureDetector.setOnDoubleTapListener(new doubleTapListener());
  7. private class doubleTapListener implements GestureDetector.OnDoubleTapListener{
  8. }

注意:大家可以看到无论在方法一还是在方法二中,都需要派生自GestureDetector.OnGestureListener,前面我们说过GestureDetector 的构造函数,如下:

  1. GestureDetector gestureDetector=new GestureDetector(GestureDetector.OnGestureListener listener);
  2. GestureDetector gestureDetector=new GestureDetector(Context context,GestureDetector.OnGestureListener listener);
  3. GestureDetector gestureDetector=new GestureDetector(Context context,GestureDetector.SimpleOnGestureListener listener);

可以看到,在构造函数中,除了后面要讲的SimpleOnGestureListener 以外的其它两个构造函数都必须是OnGestureListener的实例。所以要想使用OnDoubleTapListener的几个函数,就必须先实现OnGestureListener。

2、函数讲解:

首先看一下OnDoubleTapListener接口必须重写的三个函数:

  1. private class doubleTapListener implements GestureDetector.OnDoubleTapListener{
  2. public boolean onSingleTapConfirmed(MotionEvent e) {
  3. // TODO Auto-generated method stub
  4. return false;
  5. }
  6. public boolean onDoubleTap(MotionEvent e) {
  7. // TODO Auto-generated method stub
  8. return false;
  9. }
  10. public boolean onDoubleTapEvent(MotionEvent e) {
  11. // TODO Auto-generated method stub
  12. return false;
  13. }
  14. }

onSingleTapConfirmed(MotionEvent e):单击事件。用来判定该次点击是SingleTap而不是DoubleTap,如果连续点击两次就是DoubleTap手势,如果只点击一次,系统等待一段时间后没有收到第二次点击则判定该次点击为SingleTap而不是DoubleTap,然后触发SingleTapConfirmed事件。触发顺序是:OnDown->OnsingleTapUp->OnsingleTapConfirmed
关于onSingleTapConfirmed和onSingleTapUp的一点区别: OnGestureListener有这样的一个方法onSingleTapUp,和onSingleTapConfirmed容易混淆。二者的区别是:onSingleTapUp,只要手抬起就会执行,而对于onSingleTapConfirmed来说,如果双击的话,则onSingleTapConfirmed不会执行。

onDoubleTap(MotionEvent e):双击事件

onDoubleTapEvent(MotionEvent e):双击间隔中发生的动作。指触发onDoubleTap以后,在双击之间发生的其它动作,包含down、up和move事件;下图是双击一下的Log输出:

两点总结:

1、从上图可以看出,在第二下点击时,先触发OnDoubleTap,然后再触发OnDown(第二次点击)

2、其次在触发OnDoubleTap以后,就开始触发onDoubleTapEvent了,onDoubleTapEvent后面的数字代表了当前的事件,0指ACTION_DOWN,1指ACTION_UP,2 指ACTION_MOVE
在上一个例子的基础上,我们再添加一个双击监听类,实现如下:

  1. public class MainActivity extends Activity implements OnTouchListener{
  2. private GestureDetector mGestureDetector;
  3. @Override
  4. protected void onCreate(Bundle savedInstanceState) {
  5. super.onCreate(savedInstanceState);
  6. setContentView(R.layout.activity_main);
  7. mGestureDetector = new GestureDetector(new gestureListener()); //使用派生自OnGestureListener
  8. mGestureDetector.setOnDoubleTapListener(new doubleTapListener());
  9. TextView tv = (TextView)findViewById(R.id.tv);
  10. tv.setOnTouchListener(this);
  11. tv.setFocusable(true);
  12. tv.setClickable(true);
  13. tv.setLongClickable(true);
  14. }
  15. /*
  16. * 在onTouch()方法中,我们调用GestureDetector的onTouchEvent()方法,将捕捉到的MotionEvent交给GestureDetector
  17. * 来分析是否有合适的callback函数来处理用户的手势
  18. */
  19. public boolean onTouch(View v, MotionEvent event) {
  20. return mGestureDetector.onTouchEvent(event);
  21. }
  22. //OnGestureListener监听
  23. private class gestureListener implements GestureDetector.OnGestureListener{
  24. public boolean onDown(MotionEvent e) {
  25. Log.i("MyGesture", "onDown");
  26. Toast.makeText(MainActivity.this, "onDown", Toast.LENGTH_SHORT).show();
  27. return false;
  28. }
  29. public void onShowPress(MotionEvent e) {
  30. Log.i("MyGesture", "onShowPress");
  31. Toast.makeText(MainActivity.this, "onShowPress", Toast.LENGTH_SHORT).show();
  32. }
  33. public boolean onSingleTapUp(MotionEvent e) {
  34. Log.i("MyGesture", "onSingleTapUp");
  35. Toast.makeText(MainActivity.this, "onSingleTapUp", Toast.LENGTH_SHORT).show();
  36. return true;
  37. }
  38. public boolean onScroll(MotionEvent e1, MotionEvent e2,
  39. float distanceX, float distanceY) {
  40. Log.i("MyGesture22", "onScroll:"+(e2.getX()-e1.getX()) +"   "+distanceX);
  41. Toast.makeText(MainActivity.this, "onScroll", Toast.LENGTH_LONG).show();
  42. return true;
  43. }
  44. public void onLongPress(MotionEvent e) {
  45. Log.i("MyGesture", "onLongPress");
  46. Toast.makeText(MainActivity.this, "onLongPress", Toast.LENGTH_LONG).show();
  47. }
  48. public boolean onFling(MotionEvent e1, MotionEvent e2, float velocityX,
  49. float velocityY) {
  50. Log.i("MyGesture", "onFling");
  51. Toast.makeText(MainActivity.this, "onFling", Toast.LENGTH_LONG).show();
  52. return true;
  53. }
  54. };
  55. //OnDoubleTapListener监听
  56. private class doubleTapListener implements GestureDetector.OnDoubleTapListener{
  57. public boolean onSingleTapConfirmed(MotionEvent e) {
  58. Log.i("MyGesture", "onSingleTapConfirmed");
  59. Toast.makeText(MainActivity.this, "onSingleTapConfirmed", Toast.LENGTH_LONG).show();
  60. return true;
  61. }
  62. public boolean onDoubleTap(MotionEvent e) {
  63. Log.i("MyGesture", "onDoubleTap");
  64. Toast.makeText(MainActivity.this, "onDoubleTap", Toast.LENGTH_LONG).show();
  65. return true;
  66. }
  67. public boolean onDoubleTapEvent(MotionEvent e) {
  68. Log.i("MyGesture", "onDoubleTapEvent");
  69. Toast.makeText(MainActivity.this, "onDoubleTapEvent", Toast.LENGTH_LONG).show();
  70. return true;
  71. }
  72. };
  73. }

双击一下,部分截图如下:

双击所对应的触发事件顺序:

轻轻单击一下,对应的事件触发顺序为:

源码在博客底部给出。

四、GestureDetector.SimpleOnGestureListener---类

它与前两个不同的是:
1、这是一个类,在它基础上新建类的话,要用extends派生而不是用implements继承!
2、OnGestureListener和OnDoubleTapListener接口里的函数都是强制必须重写的,即使用不到也要重写出来一个空函数但在SimpleOnGestureListener类的实例或派生类中不必如此,可以根据情况,用到哪个函数就重写哪个函数,因为SimpleOnGestureListener类本身已经实现了这两个接口的所有函数,只是里面全是空的而已。

下面利用SimpleOnGestureListener类来重新实现上面的几个效果,代码如下:

  1. public class MainActivity extends Activity implements OnTouchListener {
  2. private GestureDetector mGestureDetector;
  3. @Override
  4. protected void onCreate(Bundle savedInstanceState) {
  5. super.onCreate(savedInstanceState);
  6. setContentView(R.layout.activity_main);
  7. mGestureDetector = new GestureDetector(new simpleGestureListener());
  8. TextView tv = (TextView)findViewById(R.id.tv);
  9. tv.setOnTouchListener(this);
  10. tv.setFocusable(true);
  11. tv.setClickable(true);
  12. tv.setLongClickable(true);
  13. }
  14. public boolean onTouch(View v, MotionEvent event) {
  15. // TODO Auto-generated method stub
  16. return mGestureDetector.onTouchEvent(event);
  17. }
  18. private class simpleGestureListener extends
  19. GestureDetector.SimpleOnGestureListener {
  20. /*****OnGestureListener的函数*****/
  21. public boolean onDown(MotionEvent e) {
  22. Log.i("MyGesture", "onDown");
  23. Toast.makeText(MainActivity.this, "onDown", Toast.LENGTH_SHORT)
  24. .show();
  25. return false;
  26. }
  27. public void onShowPress(MotionEvent e) {
  28. Log.i("MyGesture", "onShowPress");
  29. Toast.makeText(MainActivity.this, "onShowPress", Toast.LENGTH_SHORT)
  30. .show();
  31. }
  32. public boolean onSingleTapUp(MotionEvent e) {
  33. Log.i("MyGesture", "onSingleTapUp");
  34. Toast.makeText(MainActivity.this, "onSingleTapUp",
  35. Toast.LENGTH_SHORT).show();
  36. return true;
  37. }
  38. public boolean onScroll(MotionEvent e1, MotionEvent e2,
  39. float distanceX, float distanceY) {
  40. Log.i("MyGesture", "onScroll:" + (e2.getX() - e1.getX()) + "   "
  41. + distanceX);
  42. Toast.makeText(MainActivity.this, "onScroll", Toast.LENGTH_LONG)
  43. .show();
  44. return true;
  45. }
  46. public void onLongPress(MotionEvent e) {
  47. Log.i("MyGesture", "onLongPress");
  48. Toast.makeText(MainActivity.this, "onLongPress", Toast.LENGTH_LONG)
  49. .show();
  50. }
  51. public boolean onFling(MotionEvent e1, MotionEvent e2, float velocityX,
  52. float velocityY) {
  53. Log.i("MyGesture", "onFling");
  54. Toast.makeText(MainActivity.this, "onFling", Toast.LENGTH_LONG)
  55. .show();
  56. return true;
  57. }
  58. /*****OnDoubleTapListener的函数*****/
  59. public boolean onSingleTapConfirmed(MotionEvent e) {
  60. Log.i("MyGesture", "onSingleTapConfirmed");
  61. Toast.makeText(MainActivity.this, "onSingleTapConfirmed",
  62. Toast.LENGTH_LONG).show();
  63. return true;
  64. }
  65. public boolean onDoubleTap(MotionEvent e) {
  66. Log.i("MyGesture", "onDoubleTap");
  67. Toast.makeText(MainActivity.this, "onDoubleTap", Toast.LENGTH_LONG)
  68. .show();
  69. return true;
  70. }
  71. public boolean onDoubleTapEvent(MotionEvent e) {
  72. Log.i("MyGesture", "onDoubleTapEvent");
  73. Toast.makeText(MainActivity.this, "onDoubleTapEvent",
  74. Toast.LENGTH_LONG).show();
  75. return true;
  76. }
  77. }
  78. }

到此,有关GestureDetector的所有基础知识都讲解完了,下面给出一个小应用——识别用户是向左滑还是向右滑!

源码在博客底部给出。

五、OnFling应用——识别向左滑还是向右滑

这部分就有点意思了,可以说是上面知识的一个小应用,我们利用OnFling函数来识别当前用户是在向左滑还是向右滑,从而打出日志。先看下OnFling的参数:

  1. boolean onFling(MotionEvent e1, MotionEvent e2, float velocityX,float velocityY)
  2. 参数解释:
  3. e1:第1个ACTION_DOWN MotionEvent
  4. e2:最后一个ACTION_MOVE MotionEvent
  5. velocityX:X轴上的移动速度,像素/秒
  6. velocityY:Y轴上的移动速度,像素/秒

首先,先说一下实现的功能:当用户向左滑动距离超过100px,且滑动速度超过100 px/s时,即判断为向左滑动;向右同理.代码如下:

  1. public class MainActivity extends Activity implements OnTouchListener {
  2. private GestureDetector mGestureDetector;
  3. @Override
  4. protected void onCreate(Bundle savedInstanceState) {
  5. super.onCreate(savedInstanceState);
  6. setContentView(R.layout.activity_main);
  7. mGestureDetector = new GestureDetector(new simpleGestureListener());
  8. TextView tv = (TextView)findViewById(R.id.tv);
  9. tv.setOnTouchListener(this);
  10. tv.setFocusable(true);
  11. tv.setClickable(true);
  12. tv.setLongClickable(true);
  13. }
  14. public boolean onTouch(View v, MotionEvent event) {
  15. // TODO Auto-generated method stub
  16. return mGestureDetector.onTouchEvent(event);
  17. }
  18. private class simpleGestureListener extends
  19. GestureDetector.SimpleOnGestureListener {
  20. /*****OnGestureListener的函数*****/
  21. final int FLING_MIN_DISTANCE = 100, FLING_MIN_VELOCITY = 200;
  22. // 触发条件 :
  23. // X轴的坐标位移大于FLING_MIN_DISTANCE,且移动速度大于FLING_MIN_VELOCITY个像素/秒
  24. // 参数解释:
  25. // e1:第1个ACTION_DOWN MotionEvent
  26. // e2:最后一个ACTION_MOVE MotionEvent
  27. // velocityX:X轴上的移动速度,像素/秒
  28. // velocityY:Y轴上的移动速度,像素/秒
  29. public boolean onFling(MotionEvent e1, MotionEvent e2, float velocityX,
  30. float velocityY) {
  31. if (e1.getX() - e2.getX() > FLING_MIN_DISTANCE
  32. && Math.abs(velocityX) > FLING_MIN_VELOCITY) {
  33. // Fling left
  34. Log.i("MyGesture", "Fling left");
  35. Toast.makeText(MainActivity.this, "Fling Left", Toast.LENGTH_SHORT).show();
  36. } else if (e2.getX() - e1.getX() > FLING_MIN_DISTANCE
  37. && Math.abs(velocityX) > FLING_MIN_VELOCITY) {
  38. // Fling right
  39. Log.i("MyGesture", "Fling right");
  40. Toast.makeText(MainActivity.this, "Fling Right", Toast.LENGTH_SHORT).show();
  41. }
  42. return true;
  43. }
  44. }
  45. }

这段代码难度不大,就不再细讲,看下效果:

四、android 事件处理机制之requestDisallowInterceptTouchEvent

当手指触摸到屏幕时,系统就会调用相应View的onTouchEvent,并传入一系列的action。当有多个层级的View时,在父层级允许的情况下,这个action会一直向下传递直到遇到最深层的View。所以touch事件最先调用的是最底层View的onTouchEent,如果View的onTouchEvent接收到某个touch action并作了相应处理,最后有两种返回方式return true和return false;return true会告诉系统当前的View需要处理这次的touch事件,以后的系统发出的ACTION_MOVE,ACTION_UP还是需要继续监听并接收的,而且这次的action已经被处理掉了,父层的View是不可能出发onTouchEvent了。所以每一个action最多只能有一个onTouchEvent接口返回true。如果return false,便会通知系统,当前View不关心这一次的touch事件,此时这个action会传向父级,调用父级View的onTouchEvent。但是这一次的touch事件之后发出的任何action,该View都不会再接受,onTouchEvent在这一次的touch事件中再也不会触发,也就是说一旦View返回false,那么之后的ACTION_MOVE,ACTION_UP等ACTION就不会在传入这个View,但是下一次touch事件的action还是会传进来的。
    前面说了底层的View能够接收到这次的事件有一个前提条件:在父层级允许的情况下。假设不改变父层级的dispatch方法,在系统调用底层onTouchEvent之前会先调用父View的onInterceptTouchEvent方法判断,父层View是不是要截获本次touch事件之后的action。如果onInterceptTouchEvent返回了true,那么本次touch事件之后的所有action都不会再向深层的View传递,统统都会传给负层View的onTouchEvent,就是说父层已经截获了这次touch事件,之后的action也不必询问onInterceptTouchEvent,在这次的touch事件之后发出的action时onInterceptTouchEvent不会再次调用,知道下一次touch事件的来临。如果onInterceptTouchEvent返回false,那么本次action将发送给更深层的View,并且之后的每一次action都会询问父层的onInterceptTouchEvent需不需要截获本次touch事件。只有ViewGroup才有onInterceptTouchEvent方法,因为一个普通的View肯定是位于最深层的View,touch事件能够传到这里已经是最后一站了,肯定会调用View的onTouchEvent。
对于底层的View来说,有一种方法可以阻止父层的View截获touch事件,就是调用getParent().requestDisallowInterceptTouchEvent(true);方法。一旦底层View收到touch的action后调用这个方法那么父层View就不会再调用onInterceptTouchEvent了,也无法截获以后的action。
用例子总结一下onInterceptTouchEvent和onTouchEvent的调用顺序:
假设最高层View叫OuterLayout,中间层View叫InnerLayout,最底层View叫MyVIew。调用顺序是这样的(假设各个函数返回的都是false)
OuterLayout.onInterceptTouchEvent->InnerLayout.onInterceptTouchEvent->MyView.onTouchEvent->InnerLayout.onTouchEvent->OuterLayout.onTouchEvent。

  1. @Override
  2. public boolean dispatchTouchEvent(MotionEvent ev) {
  3. getParent().requestDisallowInterceptTouchEvent(true);
  4. return super.dispatchTouchEvent(ev);
  5. }

这句话是告诉父view,我的事件自己处理

  1. public boolean onTouch(View v, MotionEvent event) {
  2. switch (event.getAction()) {
  3. case MotionEvent.ACTION_MOVE:
  4. pager.requestDisallowInterceptTouchEvent(true);
  5. break;
  6. case MotionEvent.ACTION_UP:
  7. case MotionEvent.ACTION_CANCEL:
  8. pager.requestDisallowInterceptTouchEvent(false);
  9. break;
  10. }
  11. }

也可以写成类似于上面那样,当用户按下的时候,我们告诉父组件,不要拦截我的事件(这个时候子组件是可以正常响应事件的),拿起之后就会告诉父组件可以阻止。

还有一个关于子控件和父控件的事件响应问题 
当父控件中有子控件的时候,并且父控件和子空间都有事件处理(比如单击事件)。这时,点击子控件,父控件的单击事件就无效了。

比如一个LinearLayout里面有一个子控件TextView,但是TextView的大小没有LinearLayout大

①如果LinearLayout和TextView都设置了单击事件,那么

  • 点击TextView区域的时候,触发的是TextView的事件,
  • 点击TextView以外的区域的时候,还是触发的LinearLayout的事件。

②如果LinearLayout设置了单击事件,而TextView没有设置单击事件的话,那么

  • 不管单击的是TextView区域,还是TextView以外的区域,都是触发的LinearLayout的单击事件

如果LinearLayout的大小和TextView一样的话,那么

①如果LinearLayout和TextView都设置了单击事件,那么

  • 只有TextView的单击事件有效

②如果LinearLayout设置了单击事件,而TextView没有设置单击事件的话,那么

触发的是LinearLayout的单击事件

鸣谢:

  https://www.cnblogs.com/linjzong/p/4191891.html

  http://blog.csdn.net/harvic880925/article/details/39520901

  http://blog.csdn.net/chaihuasong/article/details/17499799

  

横向滑动的listview和其中用到的触摸监听事件详解的更多相关文章

  1. Android开发之手势滑动(滑动手势监听)详解

    Android开发之手势滑动(滑动手势监听)详解 在Android应用中,经常需要手势滑动操作,比如上下滑动,或左右方向滑动,处理手势滑动通常有两种方法:一种是单独实现setOnTouchListen ...

  2. android listview 的监听事件

    今天遇到了一个比较让我头疼的问题,不过追根揭底只是我对listview理解的不够透彻罢了, 闲言少叙,说说我遇到的问题吧: 上篇随笔我写了关于listview的使用,如果你也已经写好了列表那么恭喜这一 ...

  3. Android 自定义ScrollView的滑动监听事件

    项目结构: 1.LazyScrollView类(自定义ScrollView) package android.zhh.com.myapplicationscrollview; /** * Create ...

  4. 【Android】Listview返回顶部,快速返回顶部的功能实现,详解代码。

    作者:程序员小冰,GitHub主页:https://github.com/QQ986945193 新浪微博:http://weibo.com/mcxiaobing 首先给大家看一下我们今天这个最终实现 ...

  5. Android ListView的监听事件

    Android开发时,最常用的控件之一就是ListView了,而使用ListView的同时,必然需要对它设置监听器,常用的监听器有这么几个1.OnItemClickListener 2.OnTouch ...

  6. Touch事件详解及区别,触屏滑动距离计算

    移动端有四个关于触摸的事件,分别是touchstart.touchmove.touchend.touchcancel(比较少用), 它们的触发顺序是touchstart-->touchmove- ...

  7. swiper 使用参考 禁止手动滑动 监听事件

    最外层容器加类名  swiper-no-swiping 监听切换事件 onTransitionEnd: function(swiper){ console.log('过渡结束'); }

  8. Android ListView 之 SimpleAdapter 二 (包含 item 中按钮监听)

    1    MainActivity.java package com.myadapter; import java.util.ArrayList; import java.util.HashMap; ...

  9. Android Xlistview的源码浅度分析 监听ListView上下滑动 以及是否到顶和底部

    如转载 请注明出处 http://blog.csdn.net/sk719887916 比如我们很多项目中会用到listview 并且要对listview滑动方向进行判断 也有需要的到listview是 ...

随机推荐

  1. Linux内核参数

    vm.overcommit_memory 0 - 表示内核将检查是否有足够的可用内存供应用进程使用:如果有足够的可用内存,内存申请允许:否则,内存申请失败,并把错误返回给应用进程. 1 - 表示内核允 ...

  2. lesson03

    3.1. 画 点 3.2. 基准平面 (重要) 1. 关于 点 的使用() 1.画一条直线,在线上画一个点(利用该点占该线段的百分比画出),通过该点画一条直线 2. 画一个长方体,定位到上表面.选择( ...

  3. 深入理解JVM(5)——垃圾收集和内存分配策略

    1.垃圾收集对象 垃圾收集主要是针对堆和方法区进行. 程序计数器.虚拟机栈和本地方法栈这三个区域属于线程私有的,只存在于线程的生命周期内,线程结束之后也会消失,因此不需要对这三个区域进行垃圾回收. 哪 ...

  4. 用Spring构建企业Java应用程序的方法

    https://mp.weixin.qq.com/s?__biz=MzU0MDEwMjgwNA==&mid=2247484965&idx=1&sn=ca6b847c65e506 ...

  5. 解决Navicat远程连接MySQL出现 10060 unknow error

    前言:今天想远程连接一下自己服务器上的MySQL,用的用的软件是Navicat,服务器上的MySQL版本为5.7 第一次连接的时候就出意外了 大概意思是 无法连接MySQL服务,解决步骤如下 第一:首 ...

  6. jQuery使用(九):队列及实现原理、基于队列模拟实现animate()

    开篇一张图之队列模型 queue()如何使用? queue()原理实现? 基于queue()模拟实现animate() 一.使用queuer方法.理解队列原理 queue() dequeue() cl ...

  7. 新加坡100M带宽,国内延迟70ms,仅800元

    ▇ 新加坡100M带宽,延迟80msE3_8G_1TB_100M_5IP_800元促:E3_32G_1TB SSD_1200元 ▇ 马来西亚,独享带宽,延迟70msL5630_16G_1TB_15M_ ...

  8. linux磁盘空间满?

    磁盘空间满啦 找到项目的logs文件夹 进入logs文件夹,会看到很多access.log*文件. 在Xshell里,输入命令cd 到项目节点的logs文件夹 可能还需要清空下回收站.

  9. unity发布的WebGL部署到IIS

    一.创建WebGL代码 在win7下,Unity3D中发布WebGL,然后部署到IIS,只要代码是对,关键是添加mime类型 二.为网站添加mime类型 .json text/json .unity3 ...

  10. 007_wireshark分析TCP的三次握手和四次断开

    要想进行抓包分析,必须先了解TCP的原理.这里介绍了TCP的建立连接的三次握手和断开连接的四次握手. 一.前言:介绍三次握手之前,先介绍TCP层的几个FLAGS字段,这个字段有如下的几种标示 SYN表 ...