欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页

Android事件机制深入探讨(二)

程序员文章站 2024-03-20 15:25:10
...

这篇博文是基于上篇《Android事件机制深入探讨(一)》基础扩展而来,建议读本文之前先阅读上一篇。

上篇我们学习了Android的事件传递机制,这里就假装大家都懂了啊,接下来我们继续深入学习事件的传递机制,首先我们把原来的demo中的Activity、CustomLinearLayout、CustomTextView类中的dispatchTouchEvent、onInterceptTouchEvent、onTouchEvent方法的return语句复原(上一篇我们为了加深理解改了代码查看效果),即改为:

return super.dispatchTouchEvent(event);
return super.onInterceptTouchEvent(event);
return super.onTouchEvent(event);

事件的分发流程很复杂,接下来我们就引入setOnTouchListener和setOnClickListener来深入讲解,首先为ViewGroup和ViewActivity添加setOnTouchListener方法并打印日志,完整代码如下:

public class TouchEventActivity extends AppCompatActivity {

    private CustomLinearLayout mCustomLinearLayout;
    private CustomTextView mCustomTextView;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_touch_event);

        mCustomLinearLayout = (CustomLinearLayout) findViewById(R.id.ll_touch_layout);
        mCustomTextView = (CustomTextView) findViewById(R.id.tv_touch_content);

        mCustomLinearLayout.setOnTouchListener(new View.OnTouchListener() {
            @Override
            public boolean onTouch(View v, MotionEvent event) {
                Log.e(TouchEventActivity.this.getClass().getSimpleName(), "这是ViewGroup的--->onTouch");
                return false;
            }
        });
        mCustomTextView.setOnTouchListener(new View.OnTouchListener() {
            @Override
            public boolean onTouch(View v, MotionEvent event) {
                Log.e(getClass().getSimpleName(), "这是View的--->onTouch");
                return false;
            }
        });
    }

    @Override
    public boolean dispatchTouchEvent(MotionEvent event) {
        Log.e(getClass().getSimpleName(), "这是Activity的--->dispatchTouchEvent");
        return super.dispatchTouchEvent(event);
    }

    @Override
    public boolean onTouchEvent(MotionEvent event) {
        Log.e(getClass().getSimpleName(), "这是Activity的--->onTouchEvent");
        return super.onTouchEvent(event);
    }
}

运行,点击,日志如下:
Android事件机制深入探讨(二)
这里看到的跟我们上篇博文的运行效果差不多,但还是有差别的,ViewGroup和View我们所设置的setOnTouchListener被执行了,即在执行事件消费(onTouchEvent)方法前会先执行用户设置的onTouch方法。细心的同学也发现了,我们在onTouch方法里返回默认值false,如果我们把它改为返回true呢,改一下代码:

    mCustomLinearLayout.setOnTouchListener(new View.OnTouchListener() {
        @Override
        public boolean onTouch(View v, MotionEvent event) {
            Log.e(TouchEventActivity.this.getClass().getSimpleName(), "这是ViewGroup的--->onTouch");
            return true;
        }
    });
    mCustomTextView.setOnTouchListener(new View.OnTouchListener() {
        @Override
        public boolean onTouch(View v, MotionEvent event) {
            Log.e(TouchEventActivity.this.getClass().getSimpleName(), "这是View的--->onTouch");
            return true;
        }
    });

运行,点击,日志如下:
Android事件机制深入探讨(二)
我们从日志可以看出ACTION_DOWN、ACIONT_MOVE、ACTION_UP三个事件都会传递到View里,但是都会在onTouch方法里终止传递,且并没有执行onTouchEvent事件消费方法,所以结论是:用户设置的setOnTouchListener会在onTouchEvent方法前调用,并且onTouch的返回值是false的话,会继续执行onTouchEvent方法,如果返回true的话,表明该事件被消费了,事件终止传递(我们在ViewGroup也设置了setOnTouchListener方法,但没有被执行,也是这个原因),并不会执行onTouchEvent方法。

我们再来验证一下,现在我们把View(CustomTextView)的onTouch返回值改回false,然后ViewGroup(CustomLinearLayout)的onTouch返回值保持true不变,代码如下:

    mCustomLinearLayout.setOnTouchListener(new View.OnTouchListener() {
        @Override
        public boolean onTouch(View v, MotionEvent event) {
            Log.e(TouchEventActivity.this.getClass().getSimpleName(), "这是ViewGroup的--->onTouch");
            return true;
        }
    });
    mCustomTextView.setOnTouchListener(new View.OnTouchListener() {
        @Override
        public boolean onTouch(View v, MotionEvent event) {
            Log.e(TouchEventActivity.this.getClass().getSimpleName(), "这是View的--->onTouch");
            return false;
        }
    });

运行,打印日志,如下:
Android事件机制深入探讨(二)

ACTION_DOWN分发到View的onTouch,此时返回值为false,没有消费事件,继续执行onTouchEvent方法,该返回一样返回false,事件再往上传递给ViewGroup的onTouchEvent方法,因为我们为ViewGroup设置了setOnTouchListener,所以在执行onTouchEvent之前会先执行onTouch,而我们在onTouch方法返回了true,表明该事件被消费,接着传递终止,所以ViewGroup的onTouchEvent方法并没有被执行。

由于ACTION_DOWN的事件被ViewGroup消费掉了,所以接下来的ACTION_MOVE、ACTION_UP事件就直接传递给ViewGroup,View就无法接收到事件了。