android中View的事件处理

先写个简单的Demo,在布局中创建简单的ImageView和Button,布局代码如下:

    <ImageView
        android:id="@+id/imageview"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:src="@drawable/ic_launcher" />

    <Button
        android:id="@+id/button"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content" 
        android:text="按钮"/>

在代码中设置点击事件:

		imageview.setOnTouchListener(new OnTouchListener() {
			@Override
			public boolean onTouch(View v, MotionEvent event) {
				Log.i(tag, "imageview event = "+event.getAction());
				return false;
			}
		});
		button.setOnTouchListener(new OnTouchListener() {
			@Override
			public boolean onTouch(View v, MotionEvent event) {
				Log.i(tag, "button event = "+event.getAction());
				return false;
			}
		});

点击image和button对应的打印结果:注意是点击不是滑动。。。。

 imageview event = 0

button event = 0

button event = 1

我们都知道:

int ACTION_DOWN  = 0;

int ACTION_UP  = 1;

int ACTION_MOVE = 2;

为什么imageView只会处理ACTION_DOWN事件而button会响应ACTION_DOWN和ACTION_UP呢?

因为ImageView和Button都是继承至View,所以这就涉及到的View的事件分发机制

我们都知道所有的事件的分发都是通过dispatchTouchEvent来处理的,所以这就需要我们先去查看继承至View的子类是否有重写该方法

通过查看源码发现ImageView和Button并没有重写dispatchTouchEvent方法,所以View的dispatchTouchEvent方法对两者均适用

分析View的dispatchTouchEvent方法源码:

    public boolean dispatchTouchEvent(MotionEvent event) {
    	
        //mOnTouchListener != null:设置Touch事件监听时传递的对象 != null,在setOnTouchListener里赋值
        //(mViewFlags & ENABLED_MASK) == ENABLED:当前控件是否可用,Android下所有控件默认都是可用的,可手动设置不可用
        //mOnTouchListener.onTouch(this, event)):即调用设置OnTouchListener时重写的onTouch方法获取其中的返回值
        if (mOnTouchListener != null && (mViewFlags & ENABLED_MASK) == ENABLED &&
                mOnTouchListener.onTouch(this, event)) {
            return true;
        }
        return onTouchEvent(event);
    }
所以当子View在设置OnTouchListener时重写onTouch方法return false,则View的dispatchTouchEvent方法继续往下走调用return onTouchEvent(event);

查看View的onTouchEvent(event)方法:

代码很多,但是事件处理在Switch中执行,所以我们只需要查看核心代码即可

    //(viewFlags & CLICKABLE) == CLICKABLE 当前控件是否可点击
    //(viewFlags & LONG_CLICKABLE) == LONG_CLICKABLE),当前控件是否长按
    if (((viewFlags & CLICKABLE) == CLICKABLE ||
            (viewFlags & LONG_CLICKABLE) == LONG_CLICKABLE)) {
        switch (event.getAction()) {
            case MotionEvent.ACTION_UP:
                break;

            case MotionEvent.ACTION_DOWN:
                break;

            case MotionEvent.ACTION_CANCEL:
                break;

            case MotionEvent.ACTION_MOVE:
                break;
        }
        //如果可点击返回true
        return true;
    }
    //控件不可点击返回false
    return false;

通过分析可以得出结论:

		imageview.setOnTouchListener(new OnTouchListener() {
			//false 打印1次  dispatchTouchEvent返回false,表示事件没有响应完毕,所以只有ACTION_DOWN事件
			//true  打印2次  dispatchTouchEvent返回true,表示事件响应完毕,所以在点击事件中有ACTION_DOWN和ACTION_UP事件
			@Override
			public boolean onTouch(View v, MotionEvent event) {
				Log.v(tag, "imageview event = "+event.getAction());
				return false;
			}
		});
		button.setOnTouchListener(new OnTouchListener() {
			//false 打印2次  dispatchTouchEvent返回true,表示事件响应完毕,所以在点击事件中有ACTION_DOWN和ACTION_UP事件
			//true  打印2次  dispatchTouchEvent返回true,表示事件响应完毕,所以在点击事件中有ACTION_DOWN和ACTION_UP事件
			@Override
			public boolean onTouch(View v, MotionEvent event) {
				Log.v(tag, "button event = "+event.getAction());
				return false;
			}
		});

结论:

1、在事件处理中系统是先执行我们重写的onTouch方法,如果满足3个条件即:有给当前控件设置监听、当前控件可用、重写的onTouch方法返回true,则返回true表示当前控件响应事件,否则执行onTouchEvent方法继续进行判断

2、在onTouchEvent方法方法中如果当前控件可点击则在switch中处理down、move、up、cancle事件并返回true表示事件已处理,否则返回false表示当前控件未处理事件

也就是说:打印一次即未响应事件的最终原因是:在重写的onTouch方法返回false并且当前控件不可点击!!

整理思路:当我们在onTuch事件中reutrn true表示当前控件要响应事件,如果是return false则调用onTuchEvent方法判断当前控件是否可以点击,如果可以点击则即使在onTuch中返回false也会响应事件

所以当给ImageView设置点击事件时又会是什么情况呢?

		imageview.setOnClickListener(new OnClickListener() {
			
			@Override
			public void onClick(View v) {
				Log.i(tag, "imageview OnClickListener");
			}
		});
查看ImageView并没有对setOnClickListener重写,所以还是看View的源码:

    public void setOnClickListener(OnClickListener l) {
        if (!isClickable()) {
            setClickable(true);
        }
        mOnClickListener = l;
    }

即:只要设置控件的点击事件则View类中会将此控件设置为可点击,onClick方法会在onTouch后再去执行

只有当onTouch返回false后才会去onTouchEvent里触发onClick,即点击事件是从触摸事件里解析出来的

代码体现:在View的onTouchEvent的ACTION_UP事件中,当满足点击条件时:

                                if (!post(mPerformClick)) {
                                    performClick();
                                }

查看performClick方法

    public boolean performClick() {
        sendAccessibilityEvent(AccessibilityEvent.TYPE_VIEW_CLICKED);

        if (mOnClickListener != null) {
            playSoundEffect(SoundEffectConstants.CLICK);
            mOnClickListener.onClick(this);
            return true;
        }

        return false;
    }

即回调了我们重写的onClick方法

总结:dispatchTouchEvent  返回false  事件响应不完毕

   dispatchTouchEvent  返回true  事件响应完毕

  onTouch-->决定是否执行onTouchEvent-->决定是否执行onClick

注意:此处暂时未研究事件的onInterceptTouchEvent方法,所以默认父控件未对事件传递进行拦截






评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值