4-15笔记

ViewGroup对于事件的分发

其实我们可以从函数名称来大致判断其功能,dispatchTouchEvent,分发触摸事件,就是把事件传递下去,准确来说就是是否要传递到子View以及自己的onInterceptTouchEvent方法和onTouchEvent方法,也就是说,不仅管子Viiew,还管自身剩下的两个回调方法。onInterceptTouchEvent,事件拦截,它只管自身子View,而不会影响到自身后面两个方法的执行,如果拦截了,可以记忆为让自己的手下们无事可做。这两个方法容易混淆,需要重点理解和记忆。

拦截事件:在一定情况下,viewGroup有权利选择拦截事件或者交给子view处理
寻找接收事件序列的控件:每一个需要分发给子view的down事件都会先寻找是否有适合的子view,让子view来消费整个事件序列
派发事件:把事件分发到感兴趣的子view中或自己处理
大体的流程:每一个事件viewGroup会先判断是否要拦截,如果是action_down或其他动作,还需挨个遍历子view看看是否有子view消费了该事件,最后在决定把时间派发下去。
true表示拦截或处理此次事件(消费),不会在派发。 false表示继续往下派发,没有处理(未消费)
默认情况下,viewGroup是支持多点触控的分发,但view是不支持多点触控的,需要自己去重写 dispatchTouchEvent 方法来支持多点触控。

viewGroup分发事件时,如果没有一个子view消费事件,那么会调用自身的onTouchEvent方法来处理事件。
**view的 dispatchTouchEvent 主要内容是处理事件:**首先会调用onTouchListener判断是否消费,如果其没有处理则会调用onTouchEvent方法。
**view的 onTouchEvent的默认实现中的主要任务:**就是辨别单击与长按事件,并回调onClickListener与onLongClickListener

运用事件分发一般有两个场景:给view设置监听器和自定义view。

监听器

OnClickListener : 单击事件监听器 OnLongClickListener : 长按事件监听器 OnTouchListener : 触摸事件监听器 if (mOnTouchListener!=null && mTouchListener.onTouch(event)){ return true; }else{ if (单击事件){ mOnClickListener.onClick(view); }else if(长按事件){ mOnLongClickListener.onLongClick(view); } }

滑动嵌套问题

滑动嵌套问题:外层是viewPager,里层是recyclerView,要实现左右滑动切换viewPager,上下滑动recyclerView。这也就是著名的滑动冲突问题。类似的还有外层viewPager,里层也是可以左右滑动的recyclerView。

实时触摸反馈问题:

如设计一个按钮,要让他按下的时候缩小降低高度,放开的时候恢复到原来的大小和高度,而且如果在一个可滑动的容器中,按下之后滑动不会触发点击事件而是把事件交给外层可滑动容器。
以上的问题,我们都要灵活运用事件分发来处理的,不管如何,都是围绕着三个关键方法展开: dispatchTouchEventonIntercepterTouchEventonTouchEvent 。这三个方法在view和viewGroup都是默认实现的,我们需要基于它(就是重写这其中的三个方法的一个或多个)来完成我们的需求。

关于事件分发的简单例子:

CardView按下缩小:

package com.example.zxtext.diyUi;import android.animation.AnimatorSet;
import android.animation.ObjectAnimator;
import android.content.Context;
import android.util.AttributeSet;
import android.view.MotionEvent;import androidx.annotation.NonNull;
import androidx.annotation.Nullable;
import androidx.cardview.widget.CardView;/*
* 实现方块按下缩小,松开恢复
* */
public class NewCardView extends CardView {public NewCardView(@NonNull Context context) {this(context, null);}public NewCardView(@NonNull Context context, @Nullable AttributeSet attrs) {this(context, attrs,0);}public NewCardView(@NonNull Context context, @Nullable AttributeSet attrs, int defStyleAttr) {super(context, attrs, defStyleAttr);}//点击事件到来时进行判断+@Overridepublic boolean dispatchTouchEvent(MotionEvent ev) {int action =   ev.getActionMasked();switch (action){case MotionEvent.ACTION_DOWN:clickEvent();break;case MotionEvent.ACTION_UP:case MotionEvent.ACTION_CANCEL:upEvent();break;default:break;}return super.dispatchTouchEvent(ev);}//松开 恢复原样private void upEvent() {setCardElevation(getCardElevation());AnimatorSet set = new AnimatorSet();set.playTogether(ObjectAnimator.ofFloat(this,"scaleX",0.97f,1),ObjectAnimator.ofFloat(this,"scaleY",0.97f,1),ObjectAnimator.ofFloat(this,"alpha",0.8f,1));set.setDuration(100).start();}//按下时 大小高度变小,透明度减少private void clickEvent() {setCardElevation(4);AnimatorSet animatorSet = new AnimatorSet();animatorSet.playTogether(ObjectAnimator.ofFloat(this,"scaleX",1,0.97f),ObjectAnimator.ofFloat(this,"scaleY",1,0.97f),ObjectAnimator.ofFloat(this,"alpha",1,0.8f));animatorSet.setDuration(100).start();}//拦截@Overridepublic boolean onInterceptTouchEvent(MotionEvent ev) {return true;}@Overridepublic boolean onTouchEvent(MotionEvent event) {return super.onTouchEvent(event);}
}

我们给了cardView设置了动画效果,监听事件我们可以设置给cardView内部的ImageView或者直接设置给CardView。需要注意的是,如果设置给cardView需要重写cardView的 intercepTouchEvent 方法永远返回true,防止事件被子view(cardView里面放的ImageView或其它View)消费而无法触发监听事件。

重点:滑动冲突如何解决

滑动冲突的情况基本有三种:

  • 内外View的滑动方向不同,如viewPager + ListVIew;
  • 内外view滑动方向相同,例如viewPager嵌套水平滑动的recyclerView或水平方向滑动的scrollView;
  • 最后一种:上述两种的组合;

解决这类问题一般有两个步骤:确定最终实现效果、代码实现。
滑动冲突的解决需要结合具体的实现需求,具体问题具体分析,完成所需要的效果。

1.内外View的滑动方向不同

如何解决:一般思路:根据手指滑动直线与水平线的角度来判断水平滑动还是上下滑动;

如果手指在屏幕上滑动的角度小于45度,可以认为是左右滑动,如果大于45度,视为上下滑动!
滑动角度可以通过两个连续的MotionEvent对象的坐标计算出来,之后们根据角度的大小选择把事件交给外部ViewGroup还是内部View(控件)。最后根据事件处理的位置可分为内部拦截法和外部拦截法
外部拦截法:在viewGroup中判断滑动的角度,如果符合自身滑动方向消费则拦截事件,onInterceptTouchEvent(MotionEvent ev)
内部拦截法:在内部view中判断滑动的角度,如果是符合自身滑动方向则继续消费事件,否则请求外部viewGroup拦截事件处理
二者区别:外部拦截法实现比较简单,不用内部的view去应和。而内部拦截法适用于View需要做过多的判断。

外部拦截法

package yalantis.com.sidemenu.sample.MyUI;import android.content.Context;
import android.util.AttributeSet;
import android.util.Log;
import android.view.MotionEvent;
import android.widget.ScrollView;/*外部拦截法:重点在于是否拦截事件
*那么我们的重心就放在了 onInterceptTouchEvent 方法中。
* 在这个方法中计算滑动角度并判断是否要进行拦截。
* 这里以ScrollView为例子(外部是垂直滑动,内部是水平滑动)
* */
public class NewScrollView extends ScrollView {private float lastX = 0;private float lastY = 0;public NewScrollView(Context context) {this(context,null);}public NewScrollView(Context context, AttributeSet attrs) {this(context, attrs,0);}public NewScrollView(Context context, AttributeSet attrs, int defStyleAttr) {super(context, attrs, defStyleAttr);}/** 1. getAction:触摸动作的原始32位信息,包括事件的动作,触控点信息2.getActionMask:触摸的动作,按下,抬起,滑动,多点按下,多点抬起3. getActionIndex:触控点信息* */@Overridepublic boolean onInterceptTouchEvent(MotionEvent ev) {int action = ev.getActionMasked();// 不能拦截down事件,否则子view永远无法获取到事件// 不能拦截up事件,否则子view的点击事件无法被触发if(action == MotionEvent.ACTION_UP || action == MotionEvent.ACTION_DOWN){lastX = ev.getX();lastY = ev.getY();return false;}//获取斜率 拿到角度float x = ev.getX();float y = ev.getY();Log.e("TAG",x+"");Log.e("TAG",y+"");//abs 绝对值 (上下滑,y变 ;左右滑 x变)return Math.abs(lastX-x)<Math.abs(lastY-y);}
}

内部拦截法意味着内部view必须要有控制事件流走向的能力,才能对事件进行处理。这里就运用到了内部view一个重要的方法:
requestDisallowInterceptTouchEvent ()
这个方法可以强制外层viewGroup不拦截事件。因此,我们可以让viewGroup默认拦截除了down事件以外的所有事件。当子view需要处理事件时,只需要调用此方法即可获取事件;而当想要把事件交给viewGroup处理时,那么只需要取消这个标志,外层viewGroup就会拦截所有事件。从而达到内部view控制事件流走向的目的。

//如在NewViewPager复写onInterceptTouchEvent(MotionEvent ev)方法
//拦截除了down以外的事件
public boolean onInterceptTouchEvent(MotionEvent ev) {if (ev.getActionMasked()==MotionEvent.ACTION_DOWN){return false;}return true;}//二:在NewListView重写dispathTouchEvent方法
public class NewListViewextends ListView {float lastX = 0;float lastY = 0;@Overridepublic boolean dispatchTouchEvent(MotionEvent ev) {int actionMarked = ev.getActionMasked();switch (actionMarked){// down事件,必须请求不拦截,否则拿不到move事件无法进行判断case MotionEvent.ACTION_DOWN:{requestDisallowInterceptTouchEvent(true);break;}// move事件,进行判断是否处理事件case MotionEvent.ACTION_MOVE:{float x = ev.getX();float y = ev.getY();// 如果滑动角度大于90度自己处理事件if (Math.abs(lastY-y)<Math.abs(lastX-x)){requestDisallowInterceptTouchEvent(false);}break;}default:break;}// 保存本次触控点的坐标lastX = ev.getX();lastY = ev.getY();// 调用ListView的dispatchTouchEvent方法来处理事件return super.dispatchTouchEvent(ev);}
}

public class MyScrollView extends ScrollView {...float lastY = 0;boolean isScrollToBottom = false;@Overridepublic boolean onInterceptTouchEvent(MotionEvent ev) {boolean intercept = false;int actionMarked = ev.getActionMasked();switch (actionMarked){case MotionEvent.ACTION_DOWN:case MotionEvent.ACTION_UP:case MotionEvent.ACTION_CANCEL:{// 这三种事件默认不拦截,必须给子view处理break;}case MotionEvent.ACTION_MOVE:{LinearLayout layout = (LinearLayout) getChildAt(0);ListView listView = (ListView)layout.getChildAt(1);// 如果没有滑动到底部,由ScrollView处理,进行拦截if (!isScrollToBottom){intercept = true;// 如果滑动到底部且listView还没滑动到顶部,不拦截}else if (!ifTop(listView)){intercept = false;}else{// 否则判断是否是向下滑intercept = ev.getY() > lastY;}break;}default:break;}// 最后记录位置信息lastY = ev.getY();// 调用父类的拦截方法,ScrollView需要做一些处理,不然可能会造成无法滑动super.onInterceptTouchEvent(ev);return intercept;}public class OppScrollView extends ScrollView {...float lastY = 0;boolean isScrollToBottom = false;@Overridepublic boolean onInterceptTouchEvent(MotionEvent ev) {boolean intercept = false;int actionMarked = ev.getActionMasked();switch (actionMarked){case MotionEvent.ACTION_DOWN:case MotionEvent.ACTION_UP:case MotionEvent.ACTION_CANCEL:{// 这三种事件默认不拦截,必须给子view处理break;}case MotionEvent.ACTION_MOVE:{LinearLayout layout = (LinearLayout) getChildAt(0);ListView listView = (ListView)layout.getChildAt(1);// 如果没有滑动到底部,由ScrollView处理,进行拦截if (!isScrollToBottom){intercept = true;// 如果滑动到底部且listView还没滑动到顶部,不拦截}else if (!ifTop(listView)){intercept = false;}else{// 否则判断是否是向下滑intercept = ev.getY() > lastY;}break;}default:break;}// 最后记录位置信息lastY = ev.getY();// 调用父类的拦截方法,ScrollView需要做一些处理,不然可能会造成无法滑动super.onInterceptTouchEvent(ev);return intercept;}...
}
}

{

 if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {// 设置滑动监听setOnScrollChangeListener((v, scrollX, scrollY, oldScrollX, oldScrollY) -> {ViewGroup viewGroup = (ViewGroup)v;isScrollToBottom = v.getHeight() + scrollY >= viewGroup.getChildAt(0).getHeight();});}
}
// 判断listView是否到达顶部
private boolean ifTop(ListView listView){if (listView.getFirstVisiblePosition()==0){View view = listView.getChildAt(0);return view != null && view.getTop() >= 0;}return false;
}

参考:简书-Android柯南 链接:https://www.jianshu.com/p/60da0a4d8a18
https://www.cnblogs.com/andy-songwei/p/11155259.html

学习

学习–>十个优秀的开源项目
使用–>开源框架
面试
三金四银
终端研发部–博客园

关于笔记本电脑

1.清理垃圾和注册表:ccleaner(小巧,干净,享誉全球)。
2.系统的各项设置优化:WiseCare365(使用之后,chrome的书签显示速度如闪电般快速)。
3.驱动安装:Driver Booster (界面科技感十足,无捆绑,无多余功能,纯粹驱动,南北桥都有)。
4.专业卸载软件:total uninstall。(卸载功能最强,iobit uninstaller略逊一筹,界面优于TU)。
5.系统服务和启动项管理:Dism++。(小白慎用,功能比较多,精简系统无用服务比较方便)。
6.文件查找:Everything。(完胜Windows search,搜索速度极快)。
7.解锁删不掉的流氓文件:IObit Unlocker。(当你卸载搜狗输入法的时候会发现有个流氓文件死都删不掉,弹窗提示有服务在占用,此时用Unlocker右键解锁就能删了)。
8.磁盘碎片整理:Smart Defrag 。(我只有机械硬盘,这个选智能优化就好)。
9.压缩软件:Bandizip。(别再用什么好压快压360压了,这个软件目前发现了一个非常好用的功能,当你为下载的本子或者日语gal解压之后是僥傿僼傽儞偲僄等乱码烦恼时,把Bandizip设置里默认代码页改成日语,解压就不会乱码了,支持7z、zip、rar等等格式)。
10.视频播放器:MPC-HC播放器+lav分离器解码器+madVR视频渲染器。(对电脑配置要求较高,比较烧CPU和显卡)。
11.本地音乐播放器:Foobar2000。(我目前在用吧主唐姐的foobaka,美观)
12.浏览器:chrome。(全球浏览器市场占有份额之首,原生,快速,扩展多)
13.SSR。(这个不多说了)
14.浏览器扩展:
ublock origin(去广告)。tampermonkey(脚本)。lastpass(管理密码)。stylish(网页样式)。IDM(多线程下载)。
16.杀毒软件:首推卡巴斯基2019免费版。(新增主防,杀毒综合能力排世界前列,本地化较好)
17.系统运行库和游戏所需dll:DirectX修复工具V3.7。(极方便,一键操作,有些单机游戏运行需要某些dll,这个都可修复)
18.虚拟光驱:Daemon_Tools_Lite。(现在新笔记本基本没得光驱,有些游戏和软件需要光驱才能运行,比如大航海时代4~~)
19.磁盘空间分析器:Glary Utilities。(此优化软件里面最好用的功能就是磁盘空间分析了,一目了然。硬盘空间不够的同学可以借此工具看看占用你空间的元凶到底是谁,有可能是outlook邮箱或者各种缓存)
20.检测文件哈希值:MyHash。(文件往里一拖就能快速分析出)MD5、SHA1等值)
21.文件批量重命名:ReNamer。(文件批量改后缀等只要添加相应规则即可,或者第几个字后加什么符号数字之类的都可以)
22.贴图对比:Snipaste。(美工必备,看参考图片做模型设计很方便,原画贴在一边,3dmax,unity,substance painter启动!(・∀・)

关于ViewGroup和View的事件分发相关推荐

  1. Android View的事件分发机制解析

    作者:网易·周龙 最近刚看完android-Ultra-Pull-To-Refresh下拉刷新的源码,发现在写自定义控件时,对于View的事件的传递总是搞不太清楚,而View事件的分发机制,又是解决可 ...

  2. Android View的事件分发机制和滑动冲突解决方案

    这篇文章会先讲Android中View的事件分发机制,然后再介绍Android滑动冲突的形成原因并给出解决方案.因水平有限,讲的不会太过深入,只希望各位看了之后对事件分发机制的流程有个大概的概念,并且 ...

  3. 安卓自定义View进阶-事件分发机制原理【转自 app架构师 微信公众号】

    注意:本文中所有源码分析部分均基于 API23(Android 6.0) 版本,由于安卓系统源码改变很多,可能与之前版本有所不同,但基本流程都是一致的. 为什么要有事件分发机制? 安卓上面的View是 ...

  4. View体系与自定义View(三)—— View的事件分发机制

    1. 分析Activity的构成 一个Activity包含一个Window对象,这个对象是由PhoneWindow来实现的.PhoneWindow将DecorView作为整个应用窗口的根View. 而 ...

  5. 安卓自定义View进阶-事件分发机制详解

    原文地址:http://www.gcssloop.com/customview/dispatch-touchevent-source Android 事件分发机制详解,在上一篇文章 事件分发机制原理  ...

  6. 安卓自定义View进阶-事件分发机制原理

    之前讲解了很多与View绘图相关的知识,你可以在 安卓自定义View教程目录 中查看到这些文章,如果你理解了这些文章,那么至少2D绘图部分不是难题了,大部分的需求都能满足,但是关于View还有很多知识 ...

  7. 自定义View(二)--表层浅析View的事件分发机制和滑动冲突

    转载请注明出处:From李诗雨:http://blog.csdn.net/cjm2484836553/article/details/54387722 不诗意的女程序猿不是好厨师~ 这篇文章来得有些曲 ...

  8. View的事件分发机制简述

    要分析的对象就是MotionEvent,点击事件的事件分发其实就是对MotionEvent事件的分发过程,当MotionEvent产生后,系统需要把这个事件传递给一个具体的View,这个传递过程就是分 ...

  9. 精通Android自定义View(十三)事件分发简述

    1 事件序列 (1)手指接触屏幕后会产生一系列事件,事件分为3种:ACTION_DOWN(手指刚刚接触屏幕).ACTION_MOVE(手指在屏幕移动).ACTION_UP(手指从屏幕松开) (2)一个 ...

最新文章

  1. 总监路上的第 1 年,犯了两个小错误 | 程序员有话说
  2. POJ1258 Agri-Net【最小生成树】
  3. Android获取相册中图片的路径 4.4版本前后的变化
  4. 启动namenode报错:Journal Storage Directory /var/bigdata/hadoop/full/dfs/jn/dmgeo not formatted
  5. 元类(metaclass)
  6. note_maven中的常用命令
  7. IntelliJ Idea工作笔记009---代码没有错,但是在IDEA报错的原因
  8. poi设置自动换行后显示不全_抖音企业号权益之POI领多和地址内容编辑,视频编辑...
  9. Qt QDir 递归获取文件夹中的所有文件
  10. 极路由大部分机型官方固件
  11. C语言课程设计:课程管理系统
  12. Tomcat下载安装
  13. 让犀牛基于点云数据来完成逆向建模
  14. Android页面引导蒙版(浮层)库
  15. python三维向量运算_使用Python编写一个三维向量,实现向量的加法减法,点乘叉乘...
  16. 计算机应用基础补考申请书,院级教改课题申请书-《计算机应用基础》教学.doc...
  17. java生成短网址_最新url.cn短网址缩短生成接口(API)获取方法
  18. 快速学习-帕特里夏树
  19. C语言求田径场周长和面积,田径场尺寸计算及画法
  20. KSO-2021年度语言排行榜

热门文章

  1. 如何远程连接另外一台电脑
  2. 数据库专题 数据库适配问题--人大金仓数据库适配问题总结(kingbase8)
  3. GoLang之使用sync.pool和sync.cond
  4. 为什么lol计算机内存不足怎么办,win7玩LOL英雄联盟提示“内存不足”怎么处理?(图文)...
  5. 图像文本检测的标注工具_如何检测图像中的文本
  6. 寻呼(Paging)
  7. cocos2d-html5游戏图片资源选择
  8. 华为5c_华为5c手机价格及产品参数配置【图文】
  9. 前端UI:element-ui,ant-design-vue
  10. 持续集成(第二版)(转)