Android事件机制深入探讨(二)
这篇博文是基于上篇《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);
}
}
运行,点击,日志如下:
这里看到的跟我们上篇博文的运行效果差不多,但还是有差别的,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;
}
});
运行,点击,日志如下:
我们从日志可以看出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;
}
});
运行,打印日志,如下:
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就无法接收到事件了。