如何在RelativeLayout中将触摸事件传递到下面的视图?

3

我有一个位于另一个视图上方(带有按钮)的ScrollView。这个ScrollView占据了整个屏幕并遮挡了它下面的视图。

在我的应用程序中,我需要在某个时候禁用ScrollView(但仍可见),并将所有触摸事件传递给ScrollView下面的Buttons。我该怎么做?一些视图(如Buttons)在禁用时会自动执行此操作,但ScrollView不会。


你如何禁用 ScrollView?此外,你的情况似乎有点违反直觉,因为 ScrollView 仍然可见,而在其下方点击了 Buttons - user
很难解释这种行为的必要性,但这正是我需要做的。我真正所需的只是让滚动视图可见但不可交互,这样所有的触摸事件都会传递到其下方的布局上。这是否可能?至于你的问题,我并没有禁用滚动视图,这是我的问题的一部分,我应该如何禁用它... - roiberg
3个回答

12

尝试实现自己的ScrollView,其中有一个标志来指示状态(禁用/启用),并且重写onTouchEventdispatchTouchEvent以让触摸事件通过ScrollView。以下是一个示例:

public class DisabledScrollView extends ScrollView {

    private boolean mIsDisable = false;

    // if status is true, disable the ScrollView
    public void setDisableStatus(boolean status) {
        mIsDisable = status;
    }

    @Override
    public boolean onTouchEvent(MotionEvent ev) {
        // no more tocuh events for this ScrollView
        if (mIsDisable) {
            return false;
        }       
        return super.onTouchEvent(ev);
    }

    public boolean dispatchTouchEvent(MotionEvent ev) {
        // although the ScrollView doesn't get touch events , its children will get them so intercept them.
        if (mIsDisable) {
            return false;
        }
        return super.dispatchTouchEvent(ev);
    }

}

然后,您只需要更改该标志的值。看看它是否有效。

3
在我的情况中,我只需要在视图A中处理触摸事件,该视图重叠在视图B上,然后将事件发送到视图B。这两个视图都是相同RelativeLayout的子视图,但视图A和视图B之间没有父子关系。以下方法适用于我:
viewA.setOnTouchListener( new OnTouchListener(){

        @Override
        public boolean onTouch(View v, MotionEvent event) {
          // do my stuff here

          viewB.dispatchTouchEvent( event );
        }
}

0
在这种情况下,我有一个recyclerview在scrollview下面。 scrollview的顶部是垂直滚动的,而recyclerview是水平滚动的。 scrollview具有顶部填充,使得recyclerview通过scrollview填充中的透明度可见。 我必须这样做,因为当滚动scrollview时,recyclerview将垂直滚动以获得类似视差效果(此效果在另一个代码中)。 下面的代码对我的情况有效,可能有所帮助。
scrollView.setOnTouchListener(new View.OnTouchListener() {

        float mDownX,mDownY;
        boolean mIsSwiping,isDown;

        @Override
        public boolean onTouch(View v, MotionEvent event) {
            if(mIsSwiping){
                recyclerView.dispatchTouchEvent(event);
            }
            switch (event.getActionMasked()) {
                case MotionEvent.ACTION_DOWN:
                    mIsSwiping = false;
                    isDown = true;
                    mDownX = event.getX();
                    mDownY = event.getY();
                    break;
                case MotionEvent.ACTION_MOVE:
                    if(isDown){
                        float deltaX = Math.abs(event.getX() - mDownX);
                        float deltaY = Math.abs(event.getY() - mDownY);
                        mDownX = event.getX();
                        mDownY = event.getY();
                        if(deltaX!=deltaY){
                            isDown = false;
                            if(deltaX>deltaY){
                                mIsSwiping = true;
                            }
                        }
                    }
            }

            return mIsSwiping;
        }
    });

这是布局

<FrameLayout
xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto"
android:orientation="vertical" android:layout_width="match_parent"
android:layout_height="match_parent"
android:background="@color/gray0"
android:clipToPadding="false"
android:clipChildren="false"
android:paddingBottom="70dp">
<android.support.v7.widget.RecyclerView
    android:id="@+id/recyclerview"
    android:layout_width="match_parent"
    android:layout_height="320dp"
    android:orientation="horizontal"
    app:layoutManager="android.support.v7.widget.LinearLayoutManager"/>
<ScrollView
    android:id="@+id/scrollview"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:paddingTop="301.75dp"
    android:paddingBottom="23.5dp"
    android:clipChildren="false"
    android:clipToPadding="false">

. . .


网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接