Touch事件分发中只有两个主角:ViewGroup和View。Activity的Touch事件事实上是调用它内部的ViewGroup的Touch事件,可以直接当成ViewGroup处理。
View在ViewGroup内,ViewGroup也可以在其他ViewGroup内,这时候把内部的ViewGroup当成View来分析。
ViewGroup的相关事件有三个:onInterceptTouchEvent、dispatchTouchEvent、onTouchEvent。View的相关事件只有两个:dispatchTouchEvent、onTouchEvent。
而dispatchTouchEvent是用来分发触摸事件的,onTouchEvent是用来处理触摸事件的()
第一步、触摸事件的产生:
触摸事件由用户触摸屏幕,系统会生成一个MotionEvent触摸事件对象,里面封装了此次动作的所用信息,如时间,位置坐标等。
然后系统会将当前这个触摸事件交给当前的Activity来处理。
第二步、触摸事件的分发与处理:
1、系统将触摸事件交给Activity处理后,会调用Activity的dispatchTouchEvent方法进行事件的分发,Activity首先会找到活动对应的布局中的
父ViewGroup,将触摸事件交给他处理。
2、然后这个ViewGroup又会调用自身的dispatchTouchEvent方法对事件进行分发:
这里又分为两种情况:
第一种是ViewGroup分发事件找到了对应的子View进行处理。
交给它的子孩子View来处理,子孩子View得到事件后,也会调用自身的dispatchTouchEvent方法,但是View的分发则是调用View自身的onTouchEvent或者View设置的
OnTouchListener的onTouch方法对事件进行处理(这里值得一提的是,onTouch方法的优先级是比onTouchEvent方法的优先级高,即先执行onTouch方法)。
在onTouchEvent和onTouch方法中,都需要返回一个布尔类型的值,如果返回true,则说明触摸事件已经消费,不在往下传递,有拦触摸截掉事件的意义,其他子
View就不能响应到这个触摸事件,如果返回false,则说明触摸事件没有被消费,需要传递给其他子View处理。如果所有的子View都不消费这个触摸事件
,最后这个触摸事件就会交由Activity的onTouchEvent处理。
一个完整触摸事件由down、move、up组成。
这里值得我注意的是:Activity分发事件是根据该View是否消费了动作为Down的触摸事件,如果没有消费,则不会向它分发动作为UP和MOVE的触摸事件。
第二种是ViewGroup分发事件没找到对应的子View进行处理。
这时事件就由ViewGroup自己处理,它会调用自身的onTouchEvent返回false,说明事件没有找到消费者,触摸事件没有被消费,move,up 动作不需要再往下分发了,
直接调用当前Activity的onTouchEvent(ev)方法处理。
只有View消费了动作为Down的触摸事件,这个View才能响应其他两个动作的触摸事件。
说完安卓事件的分发,接下来我们就来说说图片拖动效果的实现。
第一步、为ImageView设置一个OnTouchListener,并实现其onTouch方法
第二步、拖动逻辑的实现:
1、手指按到imageView上会有一个初始位置(startX,startY)
2、手指在屏幕上移动,移动到一个新的位置(nowX,nowY)
3、计算手指在屏幕的偏移量
dx=nowX-startX
dy=nowY-startY
4、重新计算图片的位置,并控制范围,使图片不会超出边框
int left=miv_show.getLeft()+dx;
int top=miv_show.getTop()+dy;
int right=miv_show.getRight()+dx;
int buttom=miv_show.getBottom()+dy;
if(left<=0){
right+=-left;
left=0;
}
if(top<=0){
buttom+=-top;
top=0;
}
if(right>=mRight){
left+=-(right-mRight);
right=mRight;
}
if(buttom>=mButtom){
top+=-(buttom-mButtom);
buttom=mButtom;
}
5、通过调用imageView.layout(left,top,right,bottom)立即更新imageView的位置
(left指的是图片左上角的x坐标,top指的是图片左上角的y坐标,right指的是右下角的x坐标,bottom知道是图片右下角的y坐标)
6、重新初始化手指的开始位置(startX=nowX,startY=nowY)
7、当手指离开图片的时候记录当前图片的位置,在下次imageView加载完成后回显。
这里问题就来了,imageView什么时候会加载完成呢?当活动创建调用onCreate方法的时候
imageView是还没有加载完成的,如果在onCreate方法中调用imageView.layout方法是不起作用的
,故我们要设置一个监听,当图片加载完成后回调该监听的方法,在这个方法里面调用layout回显图片的位置,
代码如下:
miv_show.getViewTreeObserver().addOnGlobalLayoutListener(new OnGlobalLayoutListener() {
@Override
public void onGlobalLayout() {
System.out.println("view加载完成");
int lastLeft=sp.getInt("lastLeft", 0);
int lastTop=sp.getInt("lastTop", 0);
int lastRight=sp.getInt("lastRight", 0);
int lastButtom=sp.getInt("lastButtom", 0);
if(!(lastLeft==0&&lastTop==0&&lastRight==0&&lastButtom==0)){
//说明保存了位置
miv_show.layout(lastLeft, lastTop, lastRight, lastButtom);
}
}
});
代码实现如下:
miv_show.setOnTouchListener(new OnTouchListener() {
private int startX;
private int startY;
@Override
public boolean onTouch(View v, MotionEvent event){
int mRight=rl.getRight();
int mButtom=rl.getBottom();
switch(event.getAction()){
case MotionEvent.ACTION_DOWN:
startX=(int) event.getRawX();
startY=(int) event.getRawY();
break;
case MotionEvent.ACTION_MOVE:
int nowX=(int) event.getRawX();
int nowY=(int) event.getRawY();
int dx=nowX-startX;
int dy=nowY-startY;
int left=miv_show.getLeft()+dx;
int top=miv_show.getTop()+dy;
int right=miv_show.getRight()+dx;
int buttom=miv_show.getBottom()+dy;
if(left<=0){
right+=-left;
left=0;
}
if(top<=0){
buttom+=-top;
top=0;
}
if(right>=mRight){
left+=-(right-mRight);
right=mRight;
}
if(buttom>=mButtom){
top+=-(buttom-mButtom);
buttom=mButtom;
}
miv_show.layout(left,top, right, buttom);
startX=nowX;
startY=nowY;
break;
case MotionEvent.ACTION_UP:
int lastLeft=miv_show.getLeft();
int lastTop=miv_show.getTop();
int lastRight=miv_show.getRight();
int lastButtom=miv_show.getBottom();
Editor editor=sp.edit();
editor.putInt("lastLeft", lastLeft);
editor.putInt("lastTop", lastTop);
editor.putInt("lastRight", lastRight);
editor.putInt("lastButtom", lastButtom);
editor.commit();
break;
}
//返回true代表事件已经消费了,不需要再往下传递
//代表所有触摸事件都在这里被消费了,不再往下传递
return true;
}
});