【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>
相关推荐
石山岭12 小时前
自己动手写了一个 Android 虚拟定位 App:GPSSimulate 技术实
android·前端
杉氧14 小时前
副作用 (Side Effects) 全攻略:如何像大师一样掌控 Composable 的生命周期?
android·架构·android jetpack
Kapaseker18 小时前
Kotlin Toolchain 0.11 发布:主要是把 Amper 干没了
android·kotlin
三少爷的鞋19 小时前
Android 现代架构不需要事件总线进阶篇
android
杉氧1 天前
深入理解 Compose 重组机制:快照系统如何驱动 UI 精准刷新?
android·架构·android jetpack
召钱熏1 天前
状态枚举正确≠渲染正确:一个语音按钮的状态机边界修复实录
android·前端
杉氧1 天前
深度解析:Jetpack Compose 核心架构与底层原理 —— 十年安卓老兵的“破茧重生”
android·架构·android jetpack
通玄1 天前
Jetpack Compose 入门系列(七):ViewModel 与界面状态管理
android
落魄Android在线炒饭1 天前
Android Framework 开发技巧:android.jar 生成与系统快速编译验证
android
如此风景1 天前
Kotlin Flow操作符学习
android·kotlin