View的事件分发及滑动冲突的解决

本文详细分析了Android中view和viewGroup的事件传递机制,包括自定义view和viewGroup如何处理onTouchEvent和dispatchTouchEvent,以及事件的层级传播过程。通过实例演示,展示了事件从外向内传递和在viewGroup中的拦截机制。
摘要由CSDN通过智能技术生成
  • activity:拥有dispatchTouchEvent和onTouchEvent方法

  • view:拥有dispatchTouchEvent和onTouchEvent方法

  • viewGroup:永远dispatchTouchEvent、onInterceptEvent和onTouchEvent方法

三、view的事件传递

虽然viewGroup是view的子类,这里的view指除去viewGroup的view控件,例如textView,button,imageView等控件

写个简单的demo,分析view的事件传递

3.1、自定义一个view继承textView,并重写onTouchEvent和dispatchTouchEvent方法

**

class MyTextView : androidx.appcompat.widget.AppCompatTextView {

constructor(context: Context):super(context){

}

constructor(context: Context, attributeSet: AttributeSet): super(context, attributeSet){

}

constructor(context: Context, attributeSet: AttributeSet, defStyleAttr: Int): super(context, attributeSet, defStyleAttr){

}

override fun dispatchTouchEvent(event: MotionEvent?): Boolean {

when (event?.action) {

MotionEvent.ACTION_DOWN -> {

Log.e(“MyTextView”,“dispatchTouchEvent ACTION_DOWN”)

}

MotionEvent.ACTION_UP -> {

Log.e(“MyTextView”,“dispatchTouchEvent ACTION_UP”)

}

MotionEvent.ACTION_MOVE -> {

Log.e(“MyTextView”,“dispatchTouchEvent ACTION_MOVE”)

}

}

return super.dispatchTouchEvent(event)

}

override fun onTouchEvent(event: MotionEvent?): Boolean {

when (event?.action) {

MotionEvent.ACTION_DOWN -> {

Log.e(“MyTextView”,“onTouchEvent ACTION_DOWN”)

}

MotionEvent.ACTION_UP -> {

Log.e(“MyTextView”,“onTouchEvent ACTION_UP”)

}

MotionEvent.ACTION_MOVE -> {

Log.e(“MyTextView”,“onTouchEvent ACTION_MOVE”)

}

}

return super.onTouchEvent(event)

}

}

3.2、在activity的xml中添加MyTextView,给MyTextView设置setOnTouchListener和setOnClickListener监听,并重写activity的onTouchEvent和dispatchTouchEvent方法

**

class MainActivity : AppCompatActivity() {

override fun onCreate(savedInstanceState: Bundle?) {

super.onCreate(savedInstanceState)

setContentView(R.layout.activity_main)

var mTextView = findViewById(R.id.mTextView)

mTextView.setOnClickListener {

Log.e(“ysl”,“mTextView Click”)

}

mTextView.setOnTouchListener { v, event ->

when (event.action) {

MotionEvent.ACTION_DOWN -> {

Log.e(“mTextView”,“OnTouch ACTION_DOWN”)

}

MotionEvent.ACTION_UP -> {

Log.e(“mTextView”,“OnTouch ACTION_UP”)

}

MotionEvent.ACTION_MOVE -> {

Log.e(“mTextView”,“OnTouch ACTION_MOVE”)

}

}

return@setOnTouchListener super.onTouchEvent(event)

}

}

override fun dispatchTouchEvent(ev: MotionEvent?): Boolean {

when (ev?.action) {

MotionEvent.ACTION_DOWN -> {

Log.e(“MainActivity”,“dispatchTouchEvent ACTION_DOWN”)

}

MotionEvent.ACTION_UP -> {

Log.e(“MainActivity”,“dispatchTouchEvent ACTION_UP”)

}

MotionEvent.ACTION_MOVE -> {

Log.e(“MainActivity”,“dispatchTouchEvent ACTION_MOVE”)

}

}

return super.dispatchTouchEvent(ev)

}

override fun onTouchEvent(event: MotionEvent?): Boolean {

when (event?.action) {

MotionEvent.ACTION_DOWN -> {

Log.e(“MainActivity”,“onTouchEvent ACTION_DOWN”)

}

MotionEvent.ACTION_UP -> {

Log.e(“MainActivity”,“onTouchEvent ACTION_UP”)

}

MotionEvent.ACTION_MOVE -> {

Log.e(“MainActivity”,“onTouchEvent ACTION_MOVE”)

}

}

return super.onTouchEvent(event)

}

}

3.3、日志打印结果

**

2021-03-30 18:07:14.880 23744-23744/com.ysl.dispatchstudy E/MainActivity: dispatchTouchEvent ACTION_DOWN

2021-03-30 18:07:14.880 23744-23744/com.ysl.dispatchstudy E/MyTextView: dispatchTouchEvent ACTION_DOWN

2021-03-30 18:07:14.880 23744-23744/com.ysl.dispatchstudy E/mTextView: OnTouch ACTION_DOWN

2021-03-30 18:07:14.880 23744-23744/com.ysl.dispatchstudy E/MyTextView: onTouchEvent ACTION_DOWN

2021-03-30 18:07:14.960 23744-23744/com.ysl.dispatchstudy E/MainActivity: dispatchTouchEvent ACTION_UP

2021-03-30 18:07:14.960 23744-23744/com.ysl.dispatchstudy E/MyTextView: dispatchTouchEvent ACTION_UP

2021-03-30 18:07:14.960 23744-23744/com.ysl.dispatchstudy E/mTextView: OnTouch ACTION_UP

2021-03-30 18:07:14.960 23744-23744/com.ysl.dispatchstudy E/MyTextView: onTouchEvent ACTION_UP

2021-03-30 18:07:14.961 23744-23744/com.ysl.dispatchstudy E/ysl: mTextView Click

3.4、view事件分发的分析

view的事件传递 根据结果显示

1、触摸事件的传递流程是从dispatchTouchEvent开始的,如果没有人为干预(也就是默认返回父类的同名函数),则事件将会按照嵌套层次有外向内传递,到达最内层的view时,就由最内层的onTouchEvent进行处理,如果能处理就返回true消费掉,如果不能处理就返回false,这时事件会重新向外层传递,并由外层的onTouchEvent进行处理,依次类推

2、如果事件在向内层传递过程中被人为干预,事件处理函数返回true,事件将会被提前消费掉,内层view将不会收到这个事件

3、view的事件触发是先执行onTouch方法,在最后执行onClick方法,如果onTouch返回true,事件将不会继续传递,最后也不会调用onClick方法,如果返回false,事件继续传递

四、viewGroup的事件分发

viewGroup作为view控件的容器存在,Android系统默认提供了一系列viewGroup,例如LinearLayout,FrameLayout,RelativeLayout,ListView等

4.1、自定义一个简单的MyRelativeLayout继承RelativeLayout,重写dispatchTouchEvent、onInterceptTouchEvent和onTouchEvent方法

**

class MyRelativeLayout :RelativeLayout{

constructor(context: Context):super(context){

}

constructor(context: Context, attributeSet: AttributeSet): super(context, attributeSet){

}

constructor(context: Context, attributeSet: AttributeSet, defStyleAttr: Int): super(context, attributeSet, defStyleAttr){

}

override fun dispatchTouchEvent(ev: MotionEvent?): Boolean {

when (ev?.action) {

MotionEvent.ACTION_DOWN -> {

Log.e(“MyRelativeLayout”,“dispatchTouchEvent ACTION_DOWN”)

}

MotionEvent.ACTION_UP -> {

Log.e(“MyRelativeLayout”,“dispatchTouchEvent ACTION_UP”)

}

MotionEvent.ACTION_MOVE -> {

Log.e(“MyRelativeLayout”,“dispatchTouchEvent ACTION_MOVE”)

}

}

return super.dispatchTouchEvent(ev)

}

override fun onInterceptTouchEvent(ev: MotionEvent?): Boolean {

when (ev?.action) {

MotionEvent.ACTION_DOWN -> {

Log.e(“MyRelativeLayout”,“onInterceptTouchEvent ACTION_DOWN”)

}

MotionEvent.ACTION_UP -> {

Log.e(“MyRelativeLayout”,“onInterceptTouchEvent ACTION_UP”)

}

MotionEvent.ACTION_MOVE -> {

Log.e(“MyRelativeLayout”,“onInterceptTouchEvent ACTION_MOVE”)

}

}

return super.onInterceptTouchEvent(ev)

}

override fun onTouchEvent(event: MotionEvent?): Boolean {

when (event?.action) {

MotionEvent.ACTION_DOWN -> {

Log.e(“MyRelativeLayout”,“onTouchEvent ACTION_DOWN”)

}

MotionEvent.ACTION_UP -> {

Log.e(“MyRelativeLayout”,“onTouchEvent ACTION_UP”)

}

MotionEvent.ACTION_MOVE -> {

Log.e(“MyRelativeLayout”,“onTouchEvent ACTION_MOVE”)

}

}

return super.onTouchEvent(event)

}

}

4.2、在activity的xml中,MyTextView外面嵌套一层MyRelativeLayout

4.3、日志打印结果

**

2021-03-30 18:17:56.680 24022-24022/com.ysl.dispatchstudy E/MainActivity: dispatchTouchEvent ACTION_DOWN

自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。

深知大多数初中级Android工程师,想要提升技能,往往是自己摸索成长或者是报班学习,但对于培训机构动则近万的学费,着实压力不小。自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!

因此收集整理了一份《2024年Android移动开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。

img

img

img

img

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Android开发知识点,真正体系化!

由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!

如果你觉得这些内容对你有帮助,可以扫码获取!!(备注:Android)

结尾

最后小编想说:不论以后选择什么方向发展,目前重要的是把Android方面的技术学好,毕竟其实对于程序员来说,要学习的知识内容、技术有太多太多,要想不被环境淘汰就只有不断提升自己,从来都是我们去适应环境,而不是环境来适应我们!

当程序员容易,当一个优秀的程序员是需要不断学习的,从初级程序员到高级程序员,从初级架构师到资深架构师,或者走向管理,从技术经理到技术总监,每个阶段都需要掌握不同的能力。早早确定自己的职业方向,才能在工作和能力提升中甩开同龄人。

想要拿高薪实现技术提升薪水得到质的飞跃。最快捷的方式,就是有人可以带着你一起分析,这样学习起来最为高效,所以为了大家能够顺利进阶中高级、架构师,我特地为大家准备了一套高手学习的源码和框架视频等精品Android架构师教程,保证你学了以后保证薪资上升一个台阶。

当你有了学习线路,学习哪些内容,也知道以后的路怎么走了,理论看多了总要实践的。

高级UI,自定义View

UI这块知识是现今使用者最多的。当年火爆一时的Android入门培训,学会这小块知识就能随便找到不错的工作了。

不过很显然现在远远不够了,拒绝无休止的CV,亲自去项目实战,读源码,研究原理吧!

《Android学习笔记总结+移动架构视频+大厂面试真题+项目实战源码》,点击传送门即可获取!

每个阶段都需要掌握不同的能力。早早确定自己的职业方向,才能在工作和能力提升中甩开同龄人。

想要拿高薪实现技术提升薪水得到质的飞跃。最快捷的方式,就是有人可以带着你一起分析,这样学习起来最为高效,所以为了大家能够顺利进阶中高级、架构师,我特地为大家准备了一套高手学习的源码和框架视频等精品Android架构师教程,保证你学了以后保证薪资上升一个台阶。

当你有了学习线路,学习哪些内容,也知道以后的路怎么走了,理论看多了总要实践的。

[外链图片转存中…(img-j9WecRBW-1712302444652)]

高级UI,自定义View

UI这块知识是现今使用者最多的。当年火爆一时的Android入门培训,学会这小块知识就能随便找到不错的工作了。

不过很显然现在远远不够了,拒绝无休止的CV,亲自去项目实战,读源码,研究原理吧!

[外链图片转存中…(img-O2ZgBFhk-1712302444652)]

《Android学习笔记总结+移动架构视频+大厂面试真题+项目实战源码》,点击传送门即可获取!
  • 4
    点赞
  • 7
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值