ScrollView嵌套RecyclerView滑动冲突解决

最近发现自己负责的项目中,有使用 ScrollView 嵌套 RecyclerView 的地方,但是没有做任何针对滑动冲突的处理,于是就想看下为什么没有做这个处理,便进行了如下测试,发现了几个由其的问题。

测试场景:页面内容包括类似 HeaderView 的部分 + RecyclerView列表部分,布局是垂直方向,此处列表之上的布局内容并不是以 header add到RecyclerView上的。
测试结果:

  1. 在部分手机上,如果列表内容过少,只会造成很小程度的滑动,这种滑动冲突是没法察觉到的,很容易忽略滑动冲突;
  2. 在部分手机上,列表内容过多时,滑动 RecyclerView 部分,会很卡顿,滑动 仅ScrollView 部分,很顺畅,笔者试过Vivio x5plus 5.0系统会出现这样的情况;
  3. 在部分手机上,不管列表内容多少,当滑动时,只有 RecyclerView部分滑动,Header顶部布局内容固定,没有一起滑动,Redmi Note4 6.0系统则会出现这种结果。

一般地,对于第三种结果,一看就知道不是我们想要的结果,这种便是滑动冲突,但是不仔细时,均会因前两种结果而忽略了滑动冲突。现在,将针对以上的滑动冲突提供几种不同的解决方法。

方式一:禁止RecyclerView滑动

最直接的方式是将布局管理器中判断可滑动的方法,直接返回false,代码如下:

LinearLayoutManager layoutManager = new LinearLayoutManager(context) {
    @Override
    public boolean canScrollVertically() {
        // 直接禁止垂直滑动
        return false;
    }
}

源码实现:
/**
 * @return true if {
    @link #getOrientation()} is {
    @link #VERTICAL}
 */
@Override
public boolean 
Android中,当ScrollView嵌套RecyclerView时,由于两者都是滑动容器,可能会导致滑动冲突,即用户试图向上滚动ScrollView时,误触到了RecyclerView滑动,反之亦然。解决这个问题有几种常见方法: 1. **禁止RecyclerView滑动**:在ScrollView内部设置RecyclerView时,可以在RecyclerView上添加`setOnTouchListener`,并在触摸事件发生时检查是否在ScrollView的区域内,如果是,则阻止RecyclerView滑动。例如: ```java recyclerView.setOnTouchListener(new View.OnTouchListener() { @Override public boolean onTouch(View v, MotionEvent event) { if (event.getAction() == MotionEvent.ACTION_UP) { int y = (int) event.getY(); if (!scrollView.canScrollVertically(1) && y > scrollView.getBottom()) { // 防止RecyclerView滑动 return true; } } return false; } }); ``` 2. **使用NestedScrollView替换ScrollView**:在Android API Level 14及以上版本,可以使用NestedScrollView替换ScrollView,它内置了解决滑动冲突的逻辑。 3. **重写RecyclerView的onInterceptTouchEvent**:重写RecyclerView的`onInterceptTouchEvent`方法,在滑动开始时判断当前手指的位置是否在ScrollView内,如果不是则让RecyclerView正常滑动。 4. **使用SwipeRefreshLayout**:如果你希望在顶部有一个下拉刷新区域,可以考虑使用SwipeRefreshLayout包裹RecyclerView,这样就可以避免滑动冲突了。 5. **禁用头部或尾部Item的滑动**:针对可能导致冲突的头部或尾部固定布局,可以设置它们不响应触摸事件。 总之,关键在于理解用户意图,并在合适的时机切换滑动目标。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值