从ScrollView嵌套EditText的滑动事件冲突分析触摸事件的分发机制以及TextView的简要实现和冲突的解决办法

本文主要是介绍从ScrollView嵌套EditText的滑动事件冲突分析触摸事件的分发机制以及TextView的简要实现和冲突的解决办法,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

本篇文章假设读者没有任何的触摸事件基础知识,所以我们会从最基本的触摸事件分发处说起。


ScrollView为什么会出现嵌套EditText出现滑动事件冲突呢?相信你会有这种疑问,我们来看这么一种情况:

有一个固定高度的EditText,假设它只能显示3行文本,但是,我们在其中输入的文本多余三行时,那么这时就需要可以在EditText内部进行小幅滚动了。那么将这个EditText放入了ScrollView当中, 并且ScrollView内容过多以致ScrollView也可以滑动,这时候就会出现EditText不能滑动的现象。就像下面这张图所示:


上图中,EditText文本的高度已经超出了EditText本身的高度,所以这时EditText应该是可以滑动的,但是由于被放入到了可滑动的ScrollView当中,那么EditText的触摸事件就被屏蔽掉了。我们接下里以非常详细的过程细说触摸事件的分发机制以及这种滑动事件的处理办法。


我们分析的入口是ScrollView的dispatchTouchEvent方法,为什么入口在这里呢,因为该方法是View触摸事件的第一个入口。

由于ScrollView没有重写dispatchTouchEvent,所以我们找到其父类的实现是在ViewGroup当中:

    public boolean dispatchTouchEvent(MotionEvent ev) {...boolean handled = false;if (onFilterTouchEventForSecurity(ev)) {final int action = ev.getAction();final int actionMasked = action & MotionEvent.ACTION_MASK;...// Check for interception.final boolean intercepted;if (actionMasked == MotionEvent.ACTION_DOWN|| mFirstTouchTarget != null) {final boolean disallowIntercept = (mGroupFlags & FLAG_DISALLOW_INTERCEPT) != 0;if (!disallowIntercept) {intercepted = onInterceptTouchEvent(ev);ev.setAction(action); // restore action in case it was changed} else {intercepted = false;}} else {intercepted = true;}...if (!canceled && !intercepted) {...if (actionMasked == MotionEvent.ACTION_DOWN|| (split && actionMasked == MotionEvent.ACTION_POINTER_DOWN)|| actionMasked == MotionEvent.ACTION_HOVER_MOVE) {...final int childrenCount = mChildrenCount;if (newTouchTarget == null && childrenCount != 0) {...final View[] children = mChildren;for (int i = childrenCount - 1; i >= 0; i--) {...if (dispatchTransformedTouchEvent(ev, false, child, idBitsToAssign)) {...newTouchTarget = addTouchTarget(child, idBitsToAssign);alreadyDispatchedToNewTouchTarget = true;break;}...}}...}}...}...return handled;}
这里代码不少,我们挑重点部分看:


上图中,在dispatchTouchEvent中发现了在调用onInterceptTouchEvent方法,而onInterceptTouchEvent方法的触发是有条件的:ACTION_DOWN事件或者mFirstTouchTarget != null,并且设置的disallowIntercept为false。


所以,当我们先触发按下事件时,无论是按到了EditText还是ScrollView,那么首先会调用ScrollView的onInterceptTouchEvent方法,为什么我这么肯定呢,难道disallowIntercept不会被置为true吗?因为在每次按下事件触发时,所有的状态都会被初始化,就算是子View提前请求disallowIntercept为true,那么在每次按下时也会被重置为false。

继续往下,程序会执行到这里:


其中,dispatchTransformedTouchEvent方法会调用每一个子View的dispatchTouchEvent方法,来询问子View是否会处理这次事件。如果子View表示要处理,那么这次事件的目标View就是该子View,那么这里mFirstTouchTarget就会指向这个View,由上面的代码可知,接下来的事件都会询问ScrollView是否要拦截,如果子View没有要求不拦截的话。


这时,这次的按下事件就被传入到了EditText的dispatchTouchEvent中去,由于EditText没有重写dispatchTouchEvent,所以这次调用会在View的dispatchTouchEvent方法中进行:

    public boolean dispatchTouchEvent(MotionEvent event) {...if (onFilterTouchEventForSecurity(event)) {ListenerInfo li = mListenerInfo;if (li != null && li.mOnTouchListener != null&& (mViewFlags & ENABLED_MASK) == ENABLED&& li.mOnTouchListener.onTouch(this, event)) {result = true;}if (!result && onTouchEvent(event)) {result = true;}}...return result;}
View的 dispatchTouchEvent要比ViewGroup相对来说简单的多,这里会先调用mOnTouchListener.onTouch方法,如果设置了OnTouchListener的话。不过如果调用了mOnTouchListener.onTouch方法的话,那么View本身的onTouchEvent方法就不会被调用,这两者之间是互斥的。由于我们在这里没有设置OnTouchListener,所以,我们进入onTouchEvent方法,当然这里需要看的是EditText的onTouchEvent方法,该方法位于TextView内部:


总体来说它的内部还是相对简单的,我们挑一些重点来看:

这里有3处方法使用了event对象。先看mEditor.onTouchEvent(event):

    void onTouchEvent(MotionEvent event) {updateFloatingToolbarVisibility(event);if (hasSelectionController()) {getSelectionController().onTouchEvent(event);}if (mShowSuggestionRunnable != null) {mTextView.removeCallbacks(mShowSuggestionRunnable);mShowSuggestionRunnable = null;}if (event.getActionMasked() == MotionEvent.ACTION_DOWN) {mLastDownPositionX = event.getX();mLastDownPositionY = event.getY();// Reset this state; it will be re-set if super.onTouchEvent// causes focus to move to the view.mTouchFocusSelected = false;mIgnoreActionUpEvent = false;}}

这个方法位于Editor类的内部,这个类用于对EditText的编辑做辅助功能,这里不是我们所要关心的,所以返回调用处,进入mMovement.onTouchEvent这个地方:

            if (mMovement != null) {handled |= mMovement.onTouchEvent(this, (Spannable) mText, event);}

我们由上下文可知,mMovement的实现位于类android.text.method.ArrowKeyMovementMethod的内部:

    public boolean onTouchEvent(TextView widget, Spannable buffer, MotionEvent event) {...boolean handled = Touch.onTouchEvent(widget, buffer, event);if (widget.didTouchFocusSelect() && !isMouse) {return handled;}if (action == MotionEvent.ACTION_DOWN) {f (isMouse || isTouchSelecting(isMouse, buffer)) {...widget.getParent().requestDisallowInterceptTouchEvent(true);}} else if (widget.isFocused()) {if (action == MotionEvent.ACTION_MOVE) {...} else if (action == MotionEvent.ACTION_UP) {...return true;}}return handled;}
我们将不重要的信息删除,发现这里调用了Touch.onTouchEvent(widget, buffer, event)方法,这个方法是这么解释的:Handles touch events for dragging.  You may want to do other actions like moving the cursor on touch as well.那么就是说它是用来辅助处理TextView内部的事件滑动的:

    public static boolean onTouchEvent(TextView widget, Spannable buffer,MotionEvent event) {DragState[] ds;switch (event.getActionMasked()) {case MotionEvent.ACTION_DOWN:ds = buffer.getSpans(0, buffer.length(), DragState.class);for (int i = 0; i < ds.length; i++) {buffer.removeSpan(ds[i]);}buffer.setSpan(new DragState(event.getX(), event.getY(),widget.getScrollX(), widget.getScrollY()),0, 0, Spannable.SPAN_MARK_MARK);return true;case MotionEvent.ACTION_UP:...case MotionEvent.ACTION_MOVE:ds = buffer.getSpans(0, buffer.length(), DragState.class);if (ds.length > 0) {...if (!event.isButtonPressed(MotionEvent.BUTTON_PRIMARY)) {scrollTo(widget, layout, nx, ny);}...return true;}}}return false;}
这个方法内部的ACTION_DOWN方法也没有做什么处理,到了这里,事件传递的方法调用栈就应该返回了,但是我们的问题还没解决,就是如何解决事件冲突的问题:

因为一开始,我们就知道ScrollView是否会拦截事件是有条件的,那么,执行了一次ACTION_DOWN之后,唯一我们可以动的地方就是更改disallowIntercept的值,我们通过上下文发现,可以更改这个值的唯一方式就是让子类调用requestDisallowInterceptTouchEvent方法,这个方法会一层层将这个标志传递给父布局容器,最后作用到ScrollView这里。试试在EditText的子类中重写onTouchEvent方法,并且在方法结束之前我们调用requestDisallowInterceptTouchEvent方法,并设置其参数为true,是不是它们之间的事件冲突就可以初步解决呢?


其实,到这里,我们的事件冲突就算解决完成了,但是,我们的标题还说要分析TextView的基本实现,没错,其实,我本身的目的是要实现在EditText在内部滑动到顶部或者底部的时候,要触发外部ScrollView的滑动,那么这里我们就需要对滑动事件的处理以及滑动距离的计算方式了如指掌。有了这个问题,我们就需要从ACTION_MOVE的事件开始分析了,我们还是需要从ViewGroup处开始分析,当然在ViewGroup的dispatchTouchEvent方法中,并没有对ACTION_MOVE进行特殊处理,因为它被全部交给了真是的事件处理对象EditText,所以,按照上面的分析方法来说,这一路分析下来,唯一不同的就是Touch.onTouchEvent(widget, buffer, event)方法,它对ACTION_MOVE进行了特别的处理,就像上面最后一部分代码所展示的那样:


这里经过一系列计算之后,又调用了scrollTo方法:

   public static void scrollTo(TextView widget, Layout layout, int x, int y) {...widget.scrollTo(x, y);}
这个方法内部经过一系列的计算,又调用了View的scrollTo方法,这里就涉及到了View的scroll方法,这个方法的原理请自行查找,这里只提一下,就是它会滑动它的内容,如果有注意的话,在调用上面方法时会传入一个Layout类型的参数,这是何物呢?其实,这就是 EditText滑动时滚动的真正内容,我们所有的文本都是直接被放置在这个layout上,我们可以从EditText的onMeasure方法中找到这个layout对象被实例化的地方,那么,如何监听这个layout滚动时的高度信息呢?


如果观察View的scrollTo方法的话,会得知该方法内部会调用onScrollChanged方法,所以,我们在EditText的子类中重写这个方法就好:

    @Overrideprotected void onScrollChanged(int horiz, int vert, int oldHoriz, int oldVert) {super.onScrollChanged(horiz, vert, oldHoriz, oldVert);//这里是滑动到底部的示例,滑动到顶部只用计算vert的值是否为0就可以//这里可以提前计算好一个值,不用每次进行计算,这里只是做示例if (vert == mLayoutHeight + paddingTop + paddingBottom - mHeight) {//这里触发父布局或祖父布局的滑动事件getParent().requestDisallowInterceptTouchEvent(false);}}

我来简单解释一下这几个计算参数的作用,如下图所示:

我们实际可滑动的范围就是0~N,N等于 mLayoutHeight + paddingTop + paddingBottom - mHeight,这几个值可在onMeasure方法中获得:

    @Overrideprotected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {super.onMeasure(widthMeasureSpec, heightMeasureSpec);mLayout = getLayout();mLayoutHeight = mLayout.getHeight();paddingTop = getTotalPaddingTop();paddingBottom = getTotalPaddingBottom();mHeight = getHeight();}
那么,整个EditText看起来应该是这样的:

import android.content.Context;
import android.text.Layout;
import android.util.AttributeSet;
import android.view.MotionEvent;
import android.widget.EditText;/*** Created by Sahadev on 2016/4/20.*/
public class MyEditText extends EditText {public Layout mLayout;public int paddingTop;public int paddingBottom;public int mHeight;public int mLayoutHeight;public MyEditText(Context context) {super(context);init();}public MyEditText(Context context, AttributeSet attrs) {super(context, attrs);init();}public MyEditText(Context context, AttributeSet attrs, int defStyleAttr) {super(context, attrs, defStyleAttr);init();}private void init() {}@Overrideprotected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {super.onMeasure(widthMeasureSpec, heightMeasureSpec);mLayout = getLayout();mLayoutHeight = mLayout.getHeight();paddingTop = getTotalPaddingTop();paddingBottom = getTotalPaddingBottom();mHeight = getHeight();}@Overridepublic boolean onTouchEvent(MotionEvent event) {boolean result = super.onTouchEvent(event);getParent().requestDisallowInterceptTouchEvent(true);return result;}@Overrideprotected void onScrollChanged(int horiz, int vert, int oldHoriz, int oldVert) {super.onScrollChanged(horiz, vert, oldHoriz, oldVert);//这里是滑动到底部的示例,滑动到顶部只用计算vert的值是否为0就可以//这里可以提前计算好一个值,不用每次进行计算,这里只是做示例if (vert == mLayoutHeight + paddingTop + paddingBottom - mHeight) {//这里触发父布局或祖父布局的滑动事件,下面这行代码只是示例作用,并没有实现真正的效果getParent().requestDisallowInterceptTouchEvent(false);}}}


好了,以上内容就是这篇文章需要了解的全部内容,基本的内容知识点有:

1.触摸事件分发:ViewGroup的dispatchTouchEvent会对事件按情况进行判断,然后交由自己的onInterceptTouchEvent方法或者传给子View的dispatchTouchEvent,而标准的View收到这个事件后会交由外部设置的OnTouchListener或者自身的onTouchEvent方法,两者只能选其一。

2.子View对父布局或者祖父布局的事件干扰,通过getParent().requestDisallowInterceptTouchEvent(true);方法要求这次事件不被父布局或者祖父布局拦截,当然,该方法应被放置到onTouchEvent中调用。一次事件代表按下、滑动、抬起、取消的整个过程。

requestDisallowInterceptTouchEvent方法会一层层的传给上传布局。

3.对于EditText,因为它的主要实现是由TextView完成的,所以,我们大部分的研究主要在TextView中,而TextView内部有一个Layout用于展示所有的文本内容。当事件被传递到这里时,又会将事件传递给其它的文本辅助控制类,比如编辑辅助类,或者上下滑动辅助类。

4.对于EditText内部滑动距离的简要方法计算,来判断EditText是否到顶,或者是否到底。从而使用户可以自定义自己的行为。


好了,今天要说的就这些,有疑问欢迎留言。

下篇文章描述了如何实现ScrollView嵌套EditText的联带滑动,详情请参见:ScrollView嵌套EditText联带滑动的解决办法




这篇关于从ScrollView嵌套EditText的滑动事件冲突分析触摸事件的分发机制以及TextView的简要实现和冲突的解决办法的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1012979

相关文章

C++对象布局及多态实现探索之内存布局(整理的很多链接)

本文通过观察对象的内存布局,跟踪函数调用的汇编代码。分析了C++对象内存的布局情况,虚函数的执行方式,以及虚继承,等等 文章链接:http://dev.yesky.com/254/2191254.shtml      论C/C++函数间动态内存的传递 (2005-07-30)   当你涉及到C/C++的核心编程的时候,你会无止境地与内存管理打交道。 文章链接:http://dev.yesky

[职场] 公务员的利弊分析 #知识分享#经验分享#其他

公务员的利弊分析     公务员作为一种稳定的职业选择,一直备受人们的关注。然而,就像任何其他职业一样,公务员职位也有其利与弊。本文将对公务员的利弊进行分析,帮助读者更好地了解这一职业的特点。 利: 1. 稳定的职业:公务员职位通常具有较高的稳定性,一旦进入公务员队伍,往往可以享受到稳定的工作环境和薪资待遇。这对于那些追求稳定的人来说,是一个很大的优势。 2. 薪资福利优厚:公务员的薪资和

通过SSH隧道实现通过远程服务器上外网

搭建隧道 autossh -M 0 -f -D 1080 -C -N user1@remotehost##验证隧道是否生效,查看1080端口是否启动netstat -tuln | grep 1080## 测试ssh 隧道是否生效curl -x socks5h://127.0.0.1:1080 -I http://www.github.com 将autossh 设置为服务,隧道开机启动

iOS HTTPS证书不受信任解决办法

之前开发App的时候服务端使用的是自签名的证书,导致iOS开发过程中调用HTTPS接口时,证书不被信任 - (void)URLSession:(NSURLSession *)session didReceiveChallenge:(NSURLAuthenticationChallenge *)challenge completionHandler:(void (^)(NSURLSessionAu

时序预测 | MATLAB实现LSTM时间序列未来多步预测-递归预测

时序预测 | MATLAB实现LSTM时间序列未来多步预测-递归预测 目录 时序预测 | MATLAB实现LSTM时间序列未来多步预测-递归预测基本介绍程序设计参考资料 基本介绍 MATLAB实现LSTM时间序列未来多步预测-递归预测。LSTM是一种含有LSTM区块(blocks)或其他的一种类神经网络,文献或其他资料中LSTM区块可能被描述成智能网络单元,因为

vue项目集成CanvasEditor实现Word在线编辑器

CanvasEditor实现Word在线编辑器 官网文档:https://hufe.club/canvas-editor-docs/guide/schema.html 源码地址:https://github.com/Hufe921/canvas-editor 前提声明: 由于CanvasEditor目前不支持vue、react 等框架开箱即用版,所以需要我们去Git下载源码,拿到其中两个主

android一键分享功能部分实现

为什么叫做部分实现呢,其实是我只实现一部分的分享。如新浪微博,那还有没去实现的是微信分享。还有一部分奇怪的问题:我QQ分享跟QQ空间的分享功能,我都没配置key那些都是原本集成就有的key也可以实现分享,谁清楚的麻烦详解下。 实现分享功能我们可以去www.mob.com这个网站集成。免费的,而且还有短信验证功能。等这分享研究完后就研究下短信验证功能。 开始实现步骤(新浪分享,以下是本人自己实现

基于Springboot + vue 的抗疫物质管理系统的设计与实现

目录 📚 前言 📑摘要 📑系统流程 📚 系统架构设计 📚 数据库设计 📚 系统功能的具体实现    💬 系统登录注册 系统登录 登录界面   用户添加  💬 抗疫列表展示模块     区域信息管理 添加物资详情 抗疫物资列表展示 抗疫物资申请 抗疫物资审核 ✒️ 源码实现 💖 源码获取 😁 联系方式 📚 前言 📑博客主页:

探索蓝牙协议的奥秘:用ESP32实现高质量蓝牙音频传输

蓝牙(Bluetooth)是一种短距离无线通信技术,广泛应用于各种电子设备之间的数据传输。自1994年由爱立信公司首次提出以来,蓝牙技术已经经历了多个版本的更新和改进。本文将详细介绍蓝牙协议,并通过一个具体的项目——使用ESP32实现蓝牙音频传输,来展示蓝牙协议的实际应用及其优点。 蓝牙协议概述 蓝牙协议栈 蓝牙协议栈是蓝牙技术的核心,定义了蓝牙设备之间如何进行通信。蓝牙协议

vue+elementui分页输入框回车与页面中@keyup.enter事件冲突解决

解决这个问题的思路只要判断事件源是哪个就好。el分页的回车触发事件是在按下时,抬起并不会再触发。而keyup.enter事件是在抬起时触发。 so,找不到分页的回车事件那就拿keyup.enter事件搞事情。只要判断这个抬起事件的$event中的锚点样式判断不等于分页特有的样式就可以了 @keyup.enter="allKeyup($event)" //页面上的//js中allKeyup(e