美文网首页
overrides onTouchEvent but not p

overrides onTouchEvent but not p

作者: 小强开学前 | 来源:发表于2021-01-23 16:13 被阅读0次

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

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

    OK,完结。


    实现需求的记录

    效果

    项目中需要一个线性文字列表的页面,还有些装饰性的点缀,类似常见的物流记录。我用 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 p

          本文链接:https://www.haomeiwen.com/subject/ghkuzktx.html