写完标题发现没什么好说的
具体见 stackoverflow
无非两点:
- 如果你放弃盲人这一用户人群,你可以忽略警告
- 重写
performClick
并在合适的时候调用(没有意外的话都是 ACTION_UP 吧?有例外吗?)
OK,完结。
实现需求的记录
效果
项目中需要一个线性文字列表的页面,还有些装饰性的点缀,类似常见的物流记录。我用
StaticLayout
结合自定义View
自己实现了一个。这次要加点击事件,但是
StaticLayout
是继承自Layout
的,无果
设置文字的Span
?不好意思ClickableSpan
需要View
支持LinkMovementMethod
,而这,TextView
及其子类才有。
于是🤔
思路:
- ☝️定义接口
onItemClicklistener
并定义一个相应类型的变量。 - ✌️ 重写
onTouchEvent
,手指抬起根据 x, y 判断是哪个 item 并调用 listener 的相关方法。(onMeasure
时已经记录了每个StaticLayout
在View
中的位置)
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)
}