启用子ScrollView时禁用父ScrollView

时间:2017-03-22 20:51:40

标签: android android-scrollview

我的布局中有两个ScrollViews,如此

<ScrollView
    android:id="@+id/news_page_scroller"
    android:layout_width="match_parent" 
    android:layout_height="match_parent">

    <!-- Extra Content -->

    <android.support.v7.widget.CardView
        android:id="@+id/news_comment_card"
        android:layout_width="match_parent"
        android:layout_height="180dp"
        app:cardCornerRadius="8sp"
        app:cardElevation="@dimen/elevation_card">

        <android.support.constraint.ConstraintLayout
            android:layout_width="match_parent"
            android:layout_height="match_parent"
            android:orientation="vertical">

            <TextView
                android:id="@+id/news_comment_header"
                android:layout_width="wrap_content"
                android:layout_height="wrap_content"
                android:text="Comments"
                android:textColor="@color/colorAccent"
                app:layout_constraintLeft_toLeftOf="parent"
                app:layout_constraintTop_toTopOf="parent" />

            <android.support.v7.widget.RecyclerView
                android:id="@+id/news_comment_section"
                android:layout_width="0dp"
                android:layout_height="0dp"
                app:layout_constraintBottom_toTopOf="@+id/news_comment_expand"
                app:layout_constraintLeft_toLeftOf="parent"
                app:layout_constraintRight_toRightOf="parent"
                app:layout_constraintTop_toBottomOf="@+id/news_comment_header"/>

            <Button
                android:id="@+id/news_comment_expand"
                android:layout_width="0dp"
                android:layout_height="wrap_content"
                android:text="@string/show_all_comments"
                app:layout_constraintBottom_toBottomOf="parent"
                app:layout_constraintLeft_toLeftOf="parent"
                app:layout_constraintRight_toRightOf="parent"/>

        </android.support.constraint.ConstraintLayout>  
</ScrollView>

我最初使用活动news_comment_section中的以下代码禁用commentSection.setOnTouchListener(new IgnoreTouch());

IgnoreTouch是一个帮助类,忽略了触摸事件,如此

 private class IgnoreTouch implements View.OnTouchListener{
    @Override
    public boolean onTouch(View v, MotionEvent event) {
        return true;
    }
}

这可用于禁用commentsList的滚动。单击按钮news_comment_expand后,CardView news_comment_card会扩展它的高度以填充屏幕,我会调用以下内容

newsPageScroller.setOnTouchListener(new IgnoreTouch());
newsPageScroller.requestDisallowInterceptTouchEvent(true);
commentSection.setOnTouchListener(null);

这会停用news_page_scroller并启用news_comment_section。这一直有效,直到我到达news_comment_section的顶部或底部。如果我在到达列表的顶部或底部后继续滚动news_comment_section,则父ScrollView news_page_scroller开始滚动。如何完全禁用父ScrollView news_page_scroller上的任何滚动?

2 个答案:

答案 0 :(得分:0)

我能够通过创建一个扩展ScrollView

的类来解决这个问题
public class PausableScrollView extends ScrollView {
    private boolean scrollEnabled = true;
    public void setScrollEnabled(boolean scrollEnabled){
        this.scrollEnabled = scrollEnabled;
    }

    public PausableScrollView(Context context) { super(context); }

    public PausableScrollView(Context context, AttributeSet attrs) { super(context, attrs); }

    public PausableScrollView(Context context, AttributeSet attrs, int defStyleAttr) { super(context, attrs, defStyleAttr); }

    @Override
    protected void onScrollChanged(int l, int t, int oldl, int oldt){
        if(scrollEnabled) {
            super.onScrollChanged(l, t, oldl, oldt);
        }
    }

    @Override
    public void onNestedScroll(View target, int dxConsumed, int dyConsumed, int dxUnconsumed, int dyUnconsumed) {
        if(scrollEnabled) {
            super.onNestedScroll(target, dxConsumed, dyConsumed, dxUnconsumed, dyUnconsumed);
        }
    }
}

基本上,我可以致电newsPageScroller.setScrollEnabled(false)来忽略newsPageScroller上发生的所有滚动事件。

答案 1 :(得分:0)

这将在放荡和滚动期间禁用从子级滚动父级,而不必手动启用和禁用:

class ContainerScrollView @JvmOverloads constructor(context: Context, attrs: AttributeSet? = null, defStyleAttr: Int = 0) 
    : ScrollView(context, attrs, defStyleAttr) {

    private var isScrollingChild = false

    override fun onScrollChanged(l: Int, t: Int, oldl: Int, oldt: Int) {
        if (!isScrollingChild) {
            super.onScrollChanged(l, t, oldl, oldt)
        }
    }

    override fun onNestedScroll(target: View?, dxConsumed: Int, dyConsumed: Int, dxUnconsumed: Int, dyUnconsumed: Int) {
        if (!isScrollingChild) {
            super.onNestedScroll(target, dxConsumed, dyConsumed, dxUnconsumed, dyUnconsumed)
        }
    }

    override fun onStartNestedScroll(child: View?, target: View?, nestedScrollAxes: Int): Boolean {
        isScrollingChild = true
        return super.onStartNestedScroll(child, target, nestedScrollAxes)
    }

    override fun onStopNestedScroll(target: View?) {
        isScrollingChild = false
        super.onStopNestedScroll(target)
    }
}