首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >Touch事件实现View拖动

Touch事件实现View拖动

作者头像
饮水思源为名
发布2018-09-06 12:56:02
1.5K0
发布2018-09-06 12:56:02
举报
文章被收录于专栏:Android小菜鸡Android小菜鸡

  Touch监听事件可以监听手指在屏幕上的行为,例如按下、滑动。抬起。根据这些事件,可以做出View任意推动的效果。

实现原理:

  首先获取view的位置以及父控件viewgroup的位置(上下左右)。通过父控件的位置可以获得父控件的宽度和高度(width=right-left,height=bottom-top)。   然后实现OnTouchListener监听,监听MotionEvent.ACTION_DOWN以及MotionEvent.ACTION_MOVE。在ACTION_DOWN中记录第一次按下时的X、Y值。在ACTION_MOVE中再次记录移动后的X、Y值,计算出X。Y移动的距离dx、dy。 之后获取view的位置(t、b、l、r)。然后计算出移动后view的位置:dt=t+dy db=b+dy dl=l+dx dr=r+dx。然后执行view.layout(dl,dt,dr,db)固定移动后view的位置。保持移动后的X、Y坐标。最后调用view.postInvalidate();执行刷新。

注意:

  • onTouch和onClick事件冲突问题:   当同时实现了Touch和Click事件时,会发生冲突,如何避免?在我们的onTouchListener的监听方法会返回一个boolean。当它为false时,就会触发Click事件,当它为true时,便不会触发。我们希望在点击的时候触发点击事件,在拖动的时候实现移动效果的Touch事件。   那么,我们定义一个boolean key=false;在ACTION_DOWN里,使key=false。而在ACTION_MOVE中使key=true,最后返回key即可。 这样做会出现很难触发点击事件。拖动事件太容易触发。所以我们需要加点限制。在ACTION_MOVE中我们获得X、Y移动的距离。那么我们判断当期中一个的移动距离大于1的时,才执行key=true。冲突得到较为完善的解决。
  • 获取View和ViewGroup的位置:   获取view和viewgroup的位置调用getLeft()、getRight()、getTop()、getBottom()四个方法。不过需要在UI控件全部加载完成之后才能调用,不能在onCreate()甚至onResum()中调用,否则全是0。正确的姿势是在touch时间或者click事件触发的时候调用。
  • View获取的位置的参照物是它的父控件:   对于view来说,他的上下左右位置参照物是它的父控件,无论父控件在屏幕的哪个位置。例如父控件处于屏幕中央某处大小是300dp300dp,位置是(100,200,400,500)。而一个View处于它的左上方大小是100dp100dp,那么这个View的位置是(0,0,100,100)。

进阶:

  避免View移动到边界然后被移出父控件消息不见。我们需要对上下左右边界进行处理。 获取view的宽高width、height,view的位置left、top、right、bottom,父控件viewgroup的宽高pwidth、pheighht。

//当view滑动到左边超过左边界时:
if(left<0){
left=0;
right=left+width;
}
//当view滑动到顶部超过顶部边界时:
if(top<0){
top=0;
bottom=top+height;
}
//当view滑动到右边超过右边边界时:
if(right>pwidth){
right=pwidth;
left=right-width;
}
//当view滑动到底部超过底部边界时:
if(bottom>pheight){
bottom=pheight;
top=bottom-height;
}
本文参与 腾讯云自媒体分享计划,分享自作者个人站点/博客。
原始发表:2018.07.31 ,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体分享计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 实现原理:
  • 注意:
  • 进阶:
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档