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作为一组的长度,方法如下:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
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)、

1
View view = (View) this.getChildAt(this.getChildCount() - 1);

1
diff_scroll = view.getRight() - getWidth();//初始状态为0,没有滚动,如果滚动到右边,则等于subViewWidth - width

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

1
2
3
4
5
6
7
8
9
10
11
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的时候设置。

Contents
,