从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的子类中重写这个方法就好:

    @Override
    protected 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方法中获得:

    @Override
    protected 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() {
    }

    @Override
    protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
        super.onMeasure(widthMeasureSpec, heightMeasureSpec);

        mLayout = getLayout();
        mLayoutHeight = mLayout.getHeight();
        paddingTop = getTotalPaddingTop();
        paddingBottom = getTotalPaddingBottom();
        mHeight = getHeight();
    }

    @Override
    public boolean onTouchEvent(MotionEvent event) {
        boolean result = super.onTouchEvent(event);
        getParent().requestDisallowInterceptTouchEvent(true);
        return result;
    }

    @Override
    protected 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联带滑动的解决办法

时间: 2024-08-06 03:40:32

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

ScrollView嵌套EditText联带滑动的解决办法

解决ScrollView嵌套EditText的滑动事件,并且实现它们两者之间的联带滑动.什么是联带滑动呢,就是当EditText滑动到底部的时候,这时就应该让外部的ScrollView跟着滑动,好让它们之间完成连贯的滑动事件.先来看看效果把. 网上没找到完整实现的例子,只好自己撸demo了.代码里有注释,全部代码如下: package chn.fz.thatjay.scrolleditview.view; import android.content.Context; import androi

Android笔记:触摸事件的分析与总结----TouchEvent处理机制

   其他相关博文:    Android笔记:触摸事件的分析与总结----MotionEvent对象    Android笔记:触摸事件的分析与总结----TouchEvent处理机制 Android中的事件类型分为按键事件和屏幕触摸事件.TouchEvent是屏幕触摸事件的基础事件,要深入了解屏幕触摸事件的处理机制,就必须掌握TouchEvent在整个触摸事件中的转移和处理过程.此处将对TouchEvent处理机制的学习做个小小的总结和备记. 当屏幕中包含一个ViewGroup,而这个Vie

ScrollView嵌套ListView的滑动冲突问题,是看大神的方法的,作为学习以后用的到

在工作中,曾多次碰到ScrollView嵌套ListView的问题,网上的解决方法有很多种,但是杂而不全.我试过很多种方法,它们各有利弊. 在这里我将会从使用ScrollView嵌套ListView结构的原因.这个结构碰到的问题.几种解决方案和优缺点比较,这4个方面来为大家阐述.分析.总结. 实际上不光是ListView,其他继承自AbsListView的类也适用,包括ExpandableListView.GridView等等,为了方便说明,以下均用ListView来代表. 大神就是牛,给出了好

Android Scrollview嵌套RecyclerView导致滑动卡顿问题解决

一个比较长的界面一般都是Scrollview嵌套RecyclerView来解决.不过这样的UI并不是我们开发人员想看到的,实际上嵌套之后.因为Scrollview和RecyclerView都是滑动控件.会有一点滑动上的冲突.导致滑动起来有些卡顿.这个时候.我们重写一下LayoutManager就行了 例如: [java] view plain copy LinearLayoutManager linearLayoutManager = new LinearLayoutManager(getAct

Android 解决ScrollView嵌套RecyclerView导致滑动不流畅的问题

最近做的项目中遇到了ScrollView嵌套RecyclerView,刚写完功能测试,直接卡出翔了,后来通过网上查找资料和 自己的实践,找出了两种方法解决这个问题. 首先来个最简单的方法: recyclerView.setNestedScrollingEnabled(false); 这个方法就可以解决这一问题. 既然有首先那肯定有第二种解决的办法,只不过相对于第一种方法来说就太麻烦了. 我们知道ScrollView嵌套listView或者GridView的时候需要自定义listView或者是Gr

在Closing事件中,将e.Cancle设置成true,则Windows无法关机和重启系统的解决办法

最近在设计一个WinForm程序的时候遇到一个bug,就是From1窗体的关闭事件中设置了e.Cancle设置成true,导致系统无法关机重启,windows7 和windows xp都是这样. 我这里设计的是当用户点击窗体的叉叉,不关闭窗体,而是最小化窗体,但当系统重启的时候,发现无关关机重启了,这里的解决办法是通过判断CloseReason的枚举值,来搞清楚到底是用户自己关闭或是系统引起的窗体关闭.关键代码如下: private void Form1_FormClosing(object s

ScrollView嵌套ListView导致滑动显示不完全问题

在ListView初始化后setAdapter后面调用此方法 /**  * 重新计算ListView的高度,解决ScrollView和ListView两个View都有滚动的效果,在嵌套使用时起冲突的问题  * @param listView  */ public void setListViewHeight(ListView listView) {   // 获取ListView对应的Adapter       ListAdapter listAdapter = listView.getAdap

解决ScrollView嵌套RecyclerView的显示及滑动问题

项目中时常需要实现在ScrollView中嵌入一个或多个RecyclerView.这一做法通常会导致如下几个问题 页面滑动卡顿 ScrollView高度显示不正常 RecyclerView内容显示不全 本文将利用多种方式分别解决上述问题 滑动卡顿解决方案 若只存在滑动卡顿这一问题,可以采用如下两种简单方式快速解决 利用RecyclerView内部方法 recyclerView.setHasFixedSize(true); recyclerView.setNestedScrollingEnable

冲突--ScrollView嵌套ListView冲突问题的最优解决方案

项目做多了之后,会发现其实 ScrollView嵌套ListVew或者GridView等很常用,但是你也会发现各种奇怪问题产生.根据个人经验现在列出常见问题以及代码最少最简单的解决方法. 问题一 : 嵌套在 ScrollView的 ListVew数据显示不全,我遇到的是最多只显示两条已有的数据. 解决办法:重写 ListVew或者 GridView,网上还有很多若干解决办法,但是都不好用或者很复杂. @Override /**   只重写该方法,达到使ListView适应ScrollView的效