Android中父View和子view的点击事件处理问题探讨
程序员文章站
2023-11-30 23:26:52
android中的事件类型分为按键事件和屏幕触摸事件,touch事件是屏幕触摸事件的基础事件,有必要对它进行深入的了解。 一个最简单的屏幕触摸动作触发了一系列touch事件...
android中的事件类型分为按键事件和屏幕触摸事件,touch事件是屏幕触摸事件的基础事件,有必要对它进行深入的了解。
一个最简单的屏幕触摸动作触发了一系列touch事件:action_down->action_move->action_move->action_move...->action_move->action_up
当屏幕中包含一个viewgroup,而这个viewgroup又包含一个子view,这个时候android系统如何处理touch事件呢?到底是viewgroup来处理touch事件,还是子view来处理touch事件呢?我只能很肯定的对你说不一定。呵呵,为什么呢?看看下面我的调查结果你
就明白了。
android系统中的每个view的子类都具有下面三个和touchevent处理密切相关的方法:
1)public boolean dispatchtouchevent(motionevent ev) 这个方法用来分发touchevent
2)public boolean onintercepttouchevent(motionevent ev) 这个方法用来拦截touchevent
3)public boolean ontouchevent(motionevent ev) 这个方法用来处理touchevent
当touchevent发生时,首先activity将touchevent传递给最顶层的view,
touchevent最先到达最顶层 view 的 dispatchtouchevent ,然后由 dispatchtouchevent 方法进行分发,
如果dispatchtouchevent返回true ,则交给这个view的ontouchevent处理,
如果dispatchtouchevent返回 false ,则交给这个 view 的 intercepttouchevent 方法来决定是否要拦截这个事件,
如果 intercepttouchevent 返回 true ,也就是拦截掉了,则交给它的 ontouchevent 来处理,
如果 intercepttouchevent 返回 false ,那么就传递给子 view ,由子 view 的 dispatchtouchevent 再来开始这个事件的分发。
如果事件传递到某一层的子 view 的 ontouchevent 上了,这个方法返回了 false ,那么这个事件会从这个 view 往上传递,都是 ontouchevent 来接收。
而如果传递到最上面的 ontouchevent 也返回 false 的话,这个事件就会“消失”,而且接收不到下一次事件。
private layoutinflater inflater
public view fristview;
public view secondview;
private myviewpager myviewpager;
public viewpageradapter mviewpageradapter;
private list<view> views;
public gallery mgallery;
@override
public void oncreate(bundle savedinstancestate) {
super.oncreate(savedinstancestate);
setcontentview(r.layout.main);
inflater = getlayoutinflater();
fristview = inflater.inflate(r.layout.main1, null);
secondview = inflater.inflate(r.layout.main2, null);
views = new arraylist<view>();
views.add(fristview);
views.add(secondview);
mgallery = (gallery) fristview.findviewbyid(r.id.gallery);
mgallery.setadapter(new imageadapter(this));
myviewpager = (myviewpager) findviewbyid(r.id.pager);
mviewpageradapter = new viewpageradapter(views);
myviewpager.setadapter(mviewpageradapter);
}
//界面列表
private list<view> views;
public viewpageradapter (list<view> views){
this.views = views;
}
//销毁arg1位置的界面
@override
public void destroyitem(view arg0, int arg1, object arg2) {
((viewpager) arg0).removeview(views.get(arg1));
}
@override
public void finishupdate(view arg0) {
// todo auto-generated method stub
}
//获得当前界面数
@override
public int getcount() {
if (views != null)
{
return views.size();
}
return 0;
}
//初始化arg1位置的界面
@override
public object instantiateitem(view arg0, int arg1) {
((viewpager) arg0).addview(views.get(arg1), 0);
return views.get(arg1);
}
//判断是否由对象生成界面
@override
public boolean isviewfromobject(view arg0, object arg1) {
return (arg0 == arg1);
}
@override
public void restorestate(parcelable arg0, classloader arg1) {
// todo auto-generated method stub
}
@override
public parcelable savestate() {
// todo auto-generated method stub
return null;
}
@override
public void startupdate(view arg0) {
// todo auto-generated method stub
}
一个最简单的屏幕触摸动作触发了一系列touch事件:action_down->action_move->action_move->action_move...->action_move->action_up
当屏幕中包含一个viewgroup,而这个viewgroup又包含一个子view,这个时候android系统如何处理touch事件呢?到底是viewgroup来处理touch事件,还是子view来处理touch事件呢?我只能很肯定的对你说不一定。呵呵,为什么呢?看看下面我的调查结果你
就明白了。
android系统中的每个view的子类都具有下面三个和touchevent处理密切相关的方法:
1)public boolean dispatchtouchevent(motionevent ev) 这个方法用来分发touchevent
2)public boolean onintercepttouchevent(motionevent ev) 这个方法用来拦截touchevent
3)public boolean ontouchevent(motionevent ev) 这个方法用来处理touchevent
当touchevent发生时,首先activity将touchevent传递给最顶层的view,
touchevent最先到达最顶层 view 的 dispatchtouchevent ,然后由 dispatchtouchevent 方法进行分发,
如果dispatchtouchevent返回true ,则交给这个view的ontouchevent处理,
如果dispatchtouchevent返回 false ,则交给这个 view 的 intercepttouchevent 方法来决定是否要拦截这个事件,
如果 intercepttouchevent 返回 true ,也就是拦截掉了,则交给它的 ontouchevent 来处理,
如果 intercepttouchevent 返回 false ,那么就传递给子 view ,由子 view 的 dispatchtouchevent 再来开始这个事件的分发。
如果事件传递到某一层的子 view 的 ontouchevent 上了,这个方法返回了 false ,那么这个事件会从这个 view 往上传递,都是 ontouchevent 来接收。
而如果传递到最上面的 ontouchevent 也返回 false 的话,这个事件就会“消失”,而且接收不到下一次事件。
复制代码 代码如下:
private layoutinflater inflater
public view fristview;
public view secondview;
private myviewpager myviewpager;
public viewpageradapter mviewpageradapter;
private list<view> views;
public gallery mgallery;
@override
public void oncreate(bundle savedinstancestate) {
super.oncreate(savedinstancestate);
setcontentview(r.layout.main);
inflater = getlayoutinflater();
fristview = inflater.inflate(r.layout.main1, null);
secondview = inflater.inflate(r.layout.main2, null);
views = new arraylist<view>();
views.add(fristview);
views.add(secondview);
mgallery = (gallery) fristview.findviewbyid(r.id.gallery);
mgallery.setadapter(new imageadapter(this));
myviewpager = (myviewpager) findviewbyid(r.id.pager);
mviewpageradapter = new viewpageradapter(views);
myviewpager.setadapter(mviewpageradapter);
}
//界面列表
private list<view> views;
public viewpageradapter (list<view> views){
this.views = views;
}
//销毁arg1位置的界面
@override
public void destroyitem(view arg0, int arg1, object arg2) {
((viewpager) arg0).removeview(views.get(arg1));
}
@override
public void finishupdate(view arg0) {
// todo auto-generated method stub
}
//获得当前界面数
@override
public int getcount() {
if (views != null)
{
return views.size();
}
return 0;
}
//初始化arg1位置的界面
@override
public object instantiateitem(view arg0, int arg1) {
((viewpager) arg0).addview(views.get(arg1), 0);
return views.get(arg1);
}
//判断是否由对象生成界面
@override
public boolean isviewfromobject(view arg0, object arg1) {
return (arg0 == arg1);
}
@override
public void restorestate(parcelable arg0, classloader arg1) {
// todo auto-generated method stub
}
@override
public parcelable savestate() {
// todo auto-generated method stub
return null;
}
@override
public void startupdate(view arg0) {
// todo auto-generated method stub
}