View事件分发及源码解析

View的事件分发还是比较简单的,我们主要需要关注两个方法,onTouchEvent和dispatchTouchEvent.首先我们通过一个例子来看一下这个流程是怎么跑的,首先我们定义一个TouchView继承自View.

package cn.npe1348.view_day10.view;

import android.content.Context;
import android.util.AttributeSet;
import android.util.Log;
import android.view.MotionEvent;
import android.view.View;

import androidx.annotation.Nullable;

public class TouchView extends View {
    public TouchView(Context context) {
        super(context);
    }

    public TouchView(Context context, @Nullable AttributeSet attrs) {
        super(context, attrs);
    }

    public TouchView(Context context, @Nullable AttributeSet attrs, int defStyleAttr) {
        super(context, attrs, defStyleAttr);
    }

    @Override
    public boolean onTouchEvent(MotionEvent event) {
        Log.e("TAG","TouchView-onTouchEvent--EVENT="+event.getAction());
        return super.onTouchEvent(event);
    }

    @Override
    public boolean dispatchTouchEvent(MotionEvent event) {
        Log.e("TAG","TouchView-dispatchTouchEvent--EVENT="+event.getAction());
        return super.dispatchTouchEvent(event);
    }
}

接下来在我们的布局文件中使用这个TouchView.

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent">
    <cn.npe1348.view_day10.view.TouchView
        android:id="@+id/touch_view"
        android:background="@color/colorPrimary"
        android:layout_width="200dp"
        android:layout_height="200dp" />
</LinearLayout>

接下来我们创建一个Activity,里面的内容如下.

package cn.npe1348.view_day10;

import android.os.Bundle;
import android.util.Log;
import android.view.MotionEvent;
import android.view.View;

import androidx.annotation.Nullable;
import androidx.appcompat.app.AppCompatActivity;

public class TouchViewActivity extends AppCompatActivity {
    @Override
    protected void onCreate(@Nullable Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_touch_view);
        View view = findViewById(R.id.touch_view);

        view.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                Log.e("TAG","TouchViewActivity--view-onClick");
            }
        });
        view.setOnTouchListener(new View.OnTouchListener() {
            @Override
            public boolean onTouch(View v, MotionEvent event) {
                Log.e("TAG","TouchViewActivity-onTouch--EVENT="+event.getAction());
                return false;
            }
        });
    }
}

我们运行看下
在这里插入图片描述
接下来我们点击绿色区域,我们可以看到打印出下面的log
在这里插入图片描述
我们可以看到,先执行了View.dispatchTouchEvent方法,然后执行到了我们的OnTouchListener.onTouch方法,之后才是执行View.onTouchEvent方法,在up时间后执行了View的onClick方法.

下面我们试试把onTouch方法返回true.

view.setOnTouchListener(new View.OnTouchListener() {
            @Override
            public boolean onTouch(View v, MotionEvent event) {
                Log.e("TAG","TouchViewActivity-onTouch--EVENT="+event.getAction());
                return true;
            }
        });

我们可以发现
在这里插入图片描述
我们可以看到,View.onTouchEvent方法和onClick方法都没有执行,这个是为什么呢.
我们可以看看源码,从上面log我们可以看到,第一步都是先跑的dispatchTouchEvent方法,我们先看看View.dispatchTouchEvent
在这里插入图片描述
这个里面有很多方法,我们不一一去看,我们主要关注上面框起来的方法,在上面我们可以看到,当OnTouchListener.onTouch方法返回true时,result就等于true,这时候下面if (!result && onTouchEvent(event))里面onTouchEvent就不会执行,所以才出现了上面那种结果,但是onClick方法为什么也没有执行呢,我们可以看到,onClick方法是在View.onTouchEvent后面执行的,我们先去看看onTouchEvent方法是怎么执行的.
在这里插入图片描述
在这里插入图片描述
我们可以看到,在上面的case MotionEvent.ACTION_UP中,有一句performClickInternal(),我们进去看看这个方法执行了什么.
在这里插入图片描述
在这里,我们可以看到,在performClick方法里面回调了onClick方法,这也就是为什么onTouch返回true之后不仅onTouchEvent方法没有执行,而且onClick方法也不会执行的道理.

总结来说,View的事件分发是从dispatchTouchEvent方法开始的,如果我们在dispatchTouchEventf方法中直接返回,而不执行super.dispatchTouchEvent(event),那么,这个View的OnTouchListener和OnClickListener监听事件都不会生效.onTouchEvent事件也不会有执行.具体的大家可以自己去看看源码,很多方法我们都没有看,只看了几个比较重要的方法.总体来说,View的事件分发还是比较简单的.下一篇我们来看下ViewGroup的事件分发.

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值