【Android】RecyclerView纵向倾斜滑动,触发ViewPage2横向滑动而导致的滑动冲突问题

问题描述

场景ViewPage2+Fragment+RecyclerView(我的项目里RecycleView外层还套了一层SmartRefreshLayout)

因ViewPage2滑动太灵敏,RecyclerView纵向倾斜滑动,触发ViewPage2横向滑动而导致的滑动冲突问题

解决方案:

纵向倾斜滑动时,其实纵向、横向这两个动作都已触发了,在横向灵敏度高的情况下,可通过给横向滑动设置一个最小临界条件,控制横向滑动的触发条件,不至于横向滑动那么灵敏。

因我的子布局是SmartRefreshLayout+RecycleView,如果你的子布局就只有RecycleView,只需将我下面自定义的CustomRefreshLayout 类里的SmartRefreshLayout替换成RecycleView就行。

kotlin 复制代码
/**
 * 解决ViewPage2内嵌RecyclerView时,因ViewPage2滑动太灵敏,RecyclerView纵向倾斜滑动,
 * 触发ViewPage2横向滑动的滑动冲突问题
 */
class CustomRefreshLayout @JvmOverloads constructor(
    context: Context, 
    attrs: AttributeSet? = null, 
    defStyleAttr: Int = 0
) : SmartRefreshLayout(context, attrs, defStyleAttr) {

    private var startX = 0f
    private var startY = 0f
    companion object {
        const val MIN_DISTANCE = 100 // 设置一个最小滑动距离,以减少误判
    }

    override fun dispatchTouchEvent(ev: MotionEvent): Boolean {
        when (ev.action) {
            MotionEvent.ACTION_DOWN -> {
                startX = ev.x
                startY = ev.y
                parent.requestDisallowInterceptTouchEvent(true) // 初始按下时,不让父控件拦截
            }
            MotionEvent.ACTION_MOVE -> {
                val endX = ev.x
                val endY = ev.y
                val disX = abs(endX - startX)
                val disY = abs(endY - startY)
                // 如果横向滑动距离大于纵向滑动距离,且大于设定的最小距离,则认为是水平滑动
                if (disX > disY) {
                    Log.e("dispatchTouchEvent 横向滑动:", "disX= $disX disY= $disY")
                    // 判断左右滑动的灵敏度,当左右滑动距离大于100 允许父控件处理滑动。
                    if (disX > MIN_DISTANCE) {
                        parent.requestDisallowInterceptTouchEvent(false)
                    }
                } else {
                    Log.e("dispatchTouchEvent 纵向滑动:", "disX= $disX disY= $disY")
                    // 纵向滑动,不让父控件拦截
                    parent.requestDisallowInterceptTouchEvent(true)
                }
            }
            MotionEvent.ACTION_UP, MotionEvent.ACTION_CANCEL -> {
                // 触摸事件结束时恢复默认行为
                parent.requestDisallowInterceptTouchEvent(false)
            }
        }
        return super.dispatchTouchEvent(ev)
    }
}
kotlin 复制代码
<CustomRefreshLayout
        android:layout_width="match_parent"
        android:layout_height="match_parent">
        <androidx.recyclerview.widget.RecyclerView
            android:id="@+id/rv"
            android:layout_width="match_parent"
            android:layout_height="wrap_content"/>
</CustomRefreshLayout>
相关推荐
勤劳打代码11 小时前
isar_flutter_libs 引发 Namespace not specified
android·flutter·groovy
奔跑吧 android12 小时前
【android bluetooth 协议分析 18】【PBAP详解 2】【车机为何不显示电话号码为空的联系人信息】
android·蓝牙电话·hfp·pbap·电话簿
深盾科技12 小时前
安卓二次打包技术深度拆解:从逆向篡改到防护逻辑
android
4Forsee12 小时前
【Android】消息机制
android·java·前端
2501_9159214314 小时前
iOS 虚拟位置设置实战,多工具协同打造精准调试与场景模拟环境
android·ios·小程序·https·uni-app·iphone·webview
龚礼鹏14 小时前
Android 图像显示框架三——演示demo以及解析
android·交互
QuantumLeap丶14 小时前
《Flutter全栈开发实战指南:从零到高级》- 11 -状态管理Provider
android·flutter·ios
百锦再15 小时前
第6章 结构体与方法
android·java·c++·python·rust·go
gustt15 小时前
用小程序搭建博客首页:从数据驱动到界面展示
android·前端·微信小程序
金鸿客15 小时前
Compose从相册和系统相机拍照获取照片
android