overrides onTouchEvent but not performClick 的正确处理方式

写完标题发现没什么好说的
具体见 stackoverflow
无非两点:

  1. 如果你放弃盲人这一用户人群,你可以忽略警告
  2. 重写 performClick并在合适的时候调用(没有意外的话都是 ACTION_UP 吧?有例外吗?)

OK,完结。


实现需求的记录

overrides onTouchEvent but not performClick 的正确处理方式_第1张图片
效果

项目中需要一个线性文字列表的页面,还有些装饰性的点缀,类似常见的物流记录。我用 StaticLayout 结合自定义 View 自己实现了一个。

这次要加点击事件,但是 StaticLayout 是继承自Layout 的,无果
设置文字的 Span ?不好意思 ClickableSpan 需要 View 支持 LinkMovementMethod,而这,TextView及其子类才有。

于是
思路:

  • ☝️定义接口 onItemClicklistener 并定义一个相应类型的变量。
  • ✌️ 重写 onTouchEvent,手指抬起根据 x, y 判断是哪个 item 并调用 listener 的相关方法。(onMeasure 时已经记录了每个 StaticLayoutView 中的位置)

performClick 根据 记录的 mDownX 和 mDownY 遍历 记录 item 位置信息的list,找到用户点击的 pos,并且通过回调调用方法。

override fun performClick(): Boolean {
    // 设置了 content 的点击事件
    onContentClickListener?.let {listener->
        ...
        listener.onItemClick(pos)
        return true
    }
    return super.performClick()
}

关键在于 触摸事件,一开始是这么写的

override fun onTouchEvent(event: MotionEvent?): Boolean {
    onContentClickListener?.let {
        event?.let { ev ->
            when (ev.actionMasked) {
                MotionEvent.ACTION_DOWN -> {
                    canClickThisTime = true
                    return true
                }
                MotionEvent.ACTION_MOVE -> {
                    canClickThisTime = false
                    return false
                }
               else -> {
                    if (canClickThisTime) {
                        // 记录此次操作
                        mDownX = ev.x
                        mDownY = ev.y
                        return performClick()
                    }
                }
            }
        }
    }
    return super.onTouchEvent(event)
}

发现当父布局是类似ScrollView这种会拦截触摸操作的Layout时,会收到MotionEvent.ACTION_CANCEL然后直接结束。
所以要加上这种情况的判断,并且为了避免其他特殊事件,把ACTION_UP独立出来。

新的问题

测试时 OnePlus 7上,一直触发不了或者说很难触发点击事件,一加特殊优化?

打印 LOG 发现手上其他手机点击时回调 基本都是 DOWN UP。
一加 是 DOWN MOVE UP。
高刷屏的问题吗?

有可能。

这样也有办法,MOVE的时候加个最大活动区域就行。
安卓本身就提供一个最小判断滑动区域的API,超过这个值系统就判断为滑动事件,实际上系统的点击事件中也有这个判断。
private val mTouchSlop:Int = ViewConfiguration.get(context).scaledTouchSlop

所以最终是:

override fun onTouchEvent(event: MotionEvent?): Boolean {
    onContentClickListener?.let {
        event?.let { ev ->
            when (ev.actionMasked) {
                MotionEvent.ACTION_DOWN -> {
                    mDownX = ev.rawX
                    mDownY = ev.rawY
                    canClickThisTime = true
                    return true
                }
                MotionEvent.ACTION_MOVE -> {
                    if(abs(mDownX - ev.rawX)>mTouchSlop || abs(mDownY - ev.rawY)>mTouchSlop) {
                        canClickThisTime = false
                    }
                    return false
                }
                MotionEvent.ACTION_CANCEL -> return false
                MotionEvent.ACTION_UP -> {
                    if (canClickThisTime) {
                        // 记录此次操作
                        mDownX = ev.x
                        mDownY = ev.y
                        return performClick()
                    }
                }
                else -> return false
            }
        }
    }
    return super.onTouchEvent(event)
}

你可能感兴趣的:(overrides onTouchEvent but not performClick 的正确处理方式)