Android中Touch事件分析--解决HorizontalScrollView滑动和按钮事件触发问题

之前写过关于HorizontalScrollView滑动和按钮事件触发问题,但是不能所有的情况,最近几天一直在想这个问题,今天有一个比较好的解决思路,最终应用在项目里面效果也很好,首先说明一下功能:

(1)、按下按钮,不滑动,触发按钮功能

(2)、按下按钮,滑动触发滑动事件

这里的按下包含长按和短按情况

首先要解决这个问题需要明白Android中的Touch事件是如何进行处理的,这里有一篇文章:http://blog.csdn.net/jwzhangjie/article/details/9718693  里面详细介绍了Touch事件处理方法,总结性语句:

         当TouchEvent发生时,首先Activity将TouchEvent传递给最顶层的View, TouchEvent最先到达最顶层 view 的 dispatchTouchEvent ,然后由  dispatchTouchEvent 方法进行分发,如果dispatchTouchEvent返回true ,则交给这个view的onTouchEvent处理,如果dispatchTouchEvent返回 false ,则交给这个 view 的 interceptTouchEvent 方法来决定是否要拦截这个事件,如果 interceptTouchEvent 返回 true ,也就是拦截掉了,则交给它的 onTouchEvent 来处理,如果 interceptTouchEvent 返回 false ,那么就传递给子 view ,由子 view 的 dispatchTouchEvent 再来开始这个事件的分发。如果事件传递到某一层的子 view 的 onTouchEvent 上了,这个方法返回了 false ,那么这个事件会从这个 view 往上传递,都是 onTouchEvent 来接收。而如果传递到最上面的 onTouchEvent 也返回 false 的话,这个事件就会“消失”,而且接收不到下一次事件。
      注意:
(1)、当滑动的时候,你不一定能正好移动合适的位置,这个时候就需要使用smoothScrollTo一类的功能函数来调整位置,而smoothScrollTo调用的时候,最好放在Handler里面实现--问题:当手慢慢移动按钮然后松开会出现异常,不过让你快速移动按钮,然后松开,按钮自己滑动停止不会有问题。
(2)、一般来说我们把一个按钮加上它前面的空白作为一组,滑动停止的时候,通过 getScrollX()来获取你当前滑动的位置,这样就可以计算你滑动停止的位置是否适合,如果不适合通过smoothScrollTo来调整,这里我用AppInforToSystem.bottom_btn_scroll_range作为一组的长度,方法如下:
public void checkRange(int value){
		try {
			if (value <= AppInforToSystem.bottom_btn_scroll_range / 2) {
				value = 0;
				smoothScrollTo(0, 0);
				AppInforToSystem.bottom_btn_scroll_flag = 2;
			}else if (value >= (diff_scroll - AppInforToSystem.bottom_btn_scroll_range / 2)) {
				smoothScrollTo(diff_scroll, 0);
				AppInforToSystem.bottom_btn_scroll_flag = 1;
			}else {
				int val = value / AppInforToSystem.bottom_btn_scroll_range;
				int diff = value % AppInforToSystem.bottom_btn_scroll_range;
				if (diff < AppInforToSystem.bottom_btn_scroll_range / 2) {
					smoothScrollTo(AppInforToSystem.bottom_btn_scroll_range * val, 0);
				}else {
					smoothScrollTo(AppInforToSystem.bottom_btn_scroll_range * (val+1), 0);
				}
				AppInforToSystem.bottom_btn_scroll_flag = 0;
			}
			this.computeScroll();
		} catch (Exception e) {
			AppInforToSystem.bottom_btn_scroll_flag = 0;
		}
	}

更需要注意的是里面this.computeScroll();一定要加上,不然会出现按钮和上面的文字分离的现象,
 
(3)、
View view = (View) this.getChildAt(this.getChildCount() - 1);
diff_scroll = view.getRight() - getWidth();//初始状态为0,没有滚动,如果滚动到右边,则等于subViewWidth - width

HorizontalScrollView只能有一个子孩子,所以要实现滑动多个组件的时候,需要使用LinearLayout或者RelativeLayout.
上面的diff_scroll就是你能够滑动的最大距离
(4)、如何检测滑动事件呢
我们需要实现一个类来继承HorizontalScrollView,并且重写onTouchEvent,当在ACTION_DOWN和ACTION_MOVE检测滑动事件并作相应处理
case MotionEvent.ACTION_DOWN:
				if(AppInforToSystem.bottom_btn_scroll_flag != 3){
					AppInforToSystem.bottom_btn_scroll_flag = 3;
					AppConnect.getInstance().callBack(CustomerInterface.MESSAGE_SCROLL_LR_FLAG);
				}
			case MotionEvent.ACTION_MOVE:
				if(AppInforToSystem.bottom_btn_scroll_flag != 3){
					AppInforToSystem.bottom_btn_scroll_flag = 3;
					AppConnect.getInstance().callBack(CustomerInterface.MESSAGE_SCROLL_LR_FLAG);
				}
				break;

有的人会问为什么在ACTION_DOWN检测了还要在ACTION_MOVE进行检测,这个你就需要之前那篇文章里面的内容,Touch事件会一直传递到子view上面,所以不会触发HorizontalScrollView的ACTION_DOWN事件,如果你移动按钮,首先出发的是按钮的MOVE事件,如果你移动幅度大的话就会触发HorizontalScrollView的ACTION_MOVE事件
所以我们需要在这两个事件都要检测
(5)、有的按钮功能是按下实现,松开关闭,这种情况就要用到onInterceptTouchEvent,让它返回false,不拦截这个事件,你可以设置一个变量,当要实现这种功能的按你按下去的时候,设置变量AppInforToSystem.main_touch_flag为0,而在onInterceptTouchEvent检测到AppInforToSystem.main_touch_flag == 0的时候返回false,这样就不会触发HorizontalScrollView的MOVE事件
(6)、如果你按下一个按钮的时候,再移动按钮触发了HorizontalScrollView的ACTION_MOVE,则这个按钮虽然不能触发ACTION_UP事件,但是可以触发ACTION_CANCEL事件,这样我们就可以恢复你在DOWN的时候设置。
 
原文地址:https://www.cnblogs.com/dyllove98/p/3233931.html