平时写代码,以业务为主,作为一个菜逼,总是觉得把工作做完就行,也就没那么关注原理。随着工作年龄的增加,对各个控件的原理也就是模模糊糊的,讲不清,道不明。某天突然脑抽,想写一系列进阶的文章,巩固下基础,补全那乱七八糟的理解。
今天来探讨一下view的分发机制。
平时写xml,对控件的使用就是一个layout包个layout,一个节点一个节点的下来。可系统怎么知道我点的是这个控件,而不是包着他的那个父控件呢?
咱们先来个生动形象的爷爷爸爸儿子控件。
全家福.png
这其中代表着最普遍的嵌套关系
爷爷:rootlayout(根布局)
爸爸:viewgroup(内部的layout等)
儿子:view(如button,textview等)
咱们非常清楚,这里爷爷最年长,所以爸爸要听爷爷的话。
那么儿子就要听爸爸的话。
所以我们的事件分发顺序就是:
rootlayout(“爷爷”) -> viewgroup(“爸爸”) -> view(“儿子”)
每个成员都有dispatchTouchEvent(MotionEvent ev)方法,用来分发事件。
“儿子”最小,事件到了他这里,他不能拦截,要么消费掉,要么再回传,因为“儿子”没地方分发事件,也就没有可拦截的地方。
方法 | rootLayout | viewGroup | view |
---|---|---|---|
dispatchTouchEvent(分发) | 有 | 有 | 有 |
onInterceptTouchEvent(拦截) | 有 | 有 | 没有 |
onTouchEvent(消费) | 有 | 有 | 有 |
- 我们把event看做是“糖果”
爷爷得到了一个糖果,进入dispatchTouchEvent(),调用了onInterceptTouchEvent(),如果合口味,那就返回true,拦截了该事件,交给onTouchEvent()处理;不拦截,就传递给下一个viewgroup
// rootlayout
@Override
public boolean dispatchTouchEvent(MotionEvent ev) {
if (ev.getAction() == MotionEvent.ACTION_DOWN) {
Log.d(TAG, "rootview dispatchTouchEvent: 我得到了一个糖果");
}
return super.dispatchTouchEvent(ev);
}
@Override
public boolean onInterceptTouchEvent(MotionEvent ev) {
if (ev.getAction() == MotionEvent.ACTION_DOWN) {
Log.d(TAG, "rootview onInterceptTouchEvent:true 我要自己吃这个糖果");
return true;
}
return super.onInterceptTouchEvent(ev);
}
@Override
public boolean onTouchEvent(MotionEvent event) {
if (event.getAction() == MotionEvent.ACTION_DOWN) {
Log.d(TAG, "rootview onTouchEvent: 爷爷自己吃了糖果");
}
return super.onTouchEvent(event);
}
D////爷爷///: rootview dispatchTouchEvent: 我得到了一个糖果
D////爷爷///: rootview onInterceptTouchEvent:true 我要自己吃这个糖果
D////爷爷///: rootview onTouchEvent: 爷爷自己吃了糖果
现在糖果不合“爷爷”口味,太硬了,磕牙~
事件传递给“爸爸”。
// viewgroup
@Override
public boolean dispatchTouchEvent(MotionEvent ev) {
if (ev.getAction() == MotionEvent.ACTION_DOWN) {
Log.d(TAG, "viewGroup dispatchTouchEvent: 爸爸从爷爷那得到一颗糖~");
}
return super.dispatchTouchEvent(ev);
}
@Override
public boolean onInterceptTouchEvent(MotionEvent ev) {
if (ev.getAction()==MotionEvent.ACTION_DOWN){
Log.d(TAG, "viewGroup onInterceptTouchEvent:我也不要吃 ");
}
return super.onInterceptTouchEvent(ev);
}
@Override
public boolean onTouchEvent(MotionEvent event) {
if (event.getAction()==MotionEvent.ACTION_DOWN){
Log.d(TAG, "viewGroup onTouchEvent: 我消费~");
}
return super.onTouchEvent(event);
}
// view
@Override
public boolean dispatchTouchEvent(MotionEvent event) {
if (event.getAction() == MotionEvent.ACTION_DOWN) {
Log.d(TAG, "View1 dispatchTouchEvent: 儿子从爸爸那得到一颗糖 ");
}
return super.dispatchTouchEvent(event);
}
@Override
public boolean onTouchEvent(MotionEvent event) {
if (event.getAction() == MotionEvent.ACTION_DOWN) {
Log.d(TAG, "View1 onTouchEvent: true 儿子吃了糖果");
return true;
}
return super.onTouchEvent(event);
}
D////爷爷///: rootview dispatchTouchEvent: 我得到了一个糖果
D////爷爷///: rootview onInterceptTouchEvent:false 我不要吃这个糖果
D////爸爸///: viewGroup dispatchTouchEvent: 爸爸从爷爷那得到一颗糖~
D////爸爸///: viewGroup onInterceptTouchEvent:我也不要吃
D////儿子///: View1 dispatchTouchEvent: 儿子从爸爸那得到一颗糖
D////儿子///: View1 onTouchEvent: true 儿子吃了糖果
如果“儿子”也不要吃糖果,onTouchEvent返回false,回传给上一层的onTouchEvent,根据返回的true或者false来决定是否要消费。
// view
@Override
public boolean dispatchTouchEvent(MotionEvent event) {
if (event.getAction() == MotionEvent.ACTION_DOWN) {
Log.d(TAG, "View1 dispatchTouchEvent: 儿子从爸爸那得到一颗糖 ");
}
return super.dispatchTouchEvent(event);
}
@Override
public boolean onTouchEvent(MotionEvent event) {
if (event.getAction() == MotionEvent.ACTION_DOWN) {
Log.d(TAG, "View1 onTouchEvent: false 我不要吃");
return false;
}
return super.onTouchEvent(event);
}
D////爷爷///: rootview dispatchTouchEvent: 我得到了一个糖果
D////爷爷///: rootview onInterceptTouchEvent:false 我不要吃这个糖果
D////爸爸///: viewGroup dispatchTouchEvent: 爸爸从爷爷那得到一颗糖~
D////爸爸///: viewGroup onInterceptTouchEvent:我也不要吃
D////儿子///: View1 dispatchTouchEvent: 儿子从爸爸那得到一颗糖
D////儿子///: View1 onTouchEvent: false 我不要吃
D////爸爸///: viewGroup onTouchEvent: true 儿子不吃,那我吃
至此,事件分发的大致流程在爷孙三谦让的分糖果过程中完全体现。如有错误处与改进处,请给我留言,谢谢~
网友评论