Android Kotlin 子 onClick 阻塞父 OnTouch

Android Kotlin child onClick blocks parent OnTouch

我有这个布局层次结构:

<LinearLayout
    xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    android:id="@+id/parent"
    android:layout_width="match_parent"
    android:layout_height="match_parent">

    <com.xxxxxx.Widget
        android:id="@+id/widget1"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"/>

    <com.xxxxxx.Widget
        android:id="@+id/widget2"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"/>
    </LinearLayout>
</LinearLayout>

我有这样的父 LinearLayout 触摸事件:

parent.setOnTouchListener(myCustomTouchParent)

    class MyCustomTouchParent(ctx: Context): View.OnTouchListener {

        private var isScrollingDown = false
        private var isScrollingUp = false
        private val myGestureDetected = GestureDetector(ctx, MyGestureListener())

        var onRecyclerViewMovingDown: (() -> Unit)? = null

        override fun onTouch(p0: View?, e: MotionEvent): Boolean {
            myGestureDetected.onTouchEvent(e)

            when(e.action){
                MotionEvent.ACTION_UP -> {
                    if (isScrollingDown) {
                        onRecyclerViewMovingDown?.invoke()
                    }
                }

                MotionEvent.ACTION_DOWN -> {
                    Log.i("TAg", "Action Down")
                    isScrollingDown = false
                    isScrollingUp = false

                }
            }
            return true
        }

        inner class MyGestureListener: GestureDetector.SimpleOnGestureListener() {
            override fun onScroll(e1: MotionEvent, e2: MotionEvent, distanceX: Float, distanceY: Float): Boolean {
                if(e2.y - e1.y > 0){
                    isScrollingUp = true
                } else if(e2.y - e1.y < 0){
                    isScrollingDown = true
                }
                return super.onScroll(e1, e2, distanceX, distanceY)
            }


        }
    }

基本上,这将检测父级上的 'Scroll Up' 事件,并执行一些动画。问题是,一旦我为 widget1 和 widget2 设置了点击侦听器,父级的触摸事件就不再起作用。有什么解决方法吗?

您必须覆盖子视图上的 onTouchListeners 和 return false,这将使它们不会覆盖其父 ontouch。

widget1.onTouch { view, motionEvent -> return@onTouch false }
widget2.onTouch { view, motionEvent -> return@onTouch false }

唯一对我有用的:在父 LinearLayout 中,拦截触摸,调用 onTouchEvent 和 return false:

override fun onInterceptTouchEvent(ev: MotionEvent): Boolean {
        onTouchEvent(ev)
        return false
    }

将带有手势检测器的 TouchInterceptor class 移动到父 onTouchEvent:

override fun onTouchEvent(e: MotionEvent): Boolean {
            myGestureDetected.onTouchEvent(e)

            when(e.action){
                MotionEvent.ACTION_UP -> {
                    if (isScrollingDown) {
                        onRecyclerViewMovingDown?.invoke()
                    }
                }

                MotionEvent.ACTION_DOWN -> {
                    isScrollingDown = false
                    isScrollingUp = false

                }
            }
            return super.onTouchEvent(e)
        }

我不知道是否有更好的解决方案,但是这个让我先处理父级上的触摸事件,然后将触摸传递给子级。在那里你可以设置你的点击监听器。

另外,如果您不设置点击监听器,包含可点击项的触摸区域将不会触发触摸。所以,最好设置

clickable=true

在所有项目中,然后只在需要时设置侦听器。