從Listview與Button點選事件衝突看安卓點選事件分發機制
阿新 • • 發佈:2019-02-20
題目有點長。其實實現Listview的時候大家都可能會碰到這樣的一個問題,那就是Listview的OnItemClickListener點選事件與Button(或者checkbox)的touch(或者click)事件衝突的問題。
宣告一下,非常感謝郭大師的這篇blog:
http://blog.csdn.net/guolin_blog/article/details/9097463
原理參考了這篇blog,事實上也是本人功力不夠不能閱讀原始碼的缺陷啊。
下面說下自己的解決步驟:
1)首先先set一下button的焦點,button是會搶佔Listview的焦點的。
replyBt.setFocusable(false)
我這樣寫就可以解決搶佔焦點的問題了。
2)重寫Button類,複寫dispatchTouchEvent方法。
首先我們得要知道,點選事件首先是被Listview接收然後才傳到Button的,如果中間某一個地方把點選事件消費掉了後面就再也不能接收到了。
我們需要接收到Button的click事件,那麼什麼時候click會被呼叫呢?
public boolean dispatchTouchEvent(MotionEvent event) { if (mOnTouchListener != null && (mViewFlags & ENABLED_MASK) == ENABLED && mOnTouchListener.onTouch(this, event)) { return true; } return onTouchEvent(event); }
這是系統的dispatchTouchEvent原始碼,這個函式負責該view點選事件的分發,click事件是在onTouchEvent裡面呼叫的,我們首先要先知道什麼時候會呼叫dispatchTouchEvent這個函式呢?
其實只要在這個view上有一個點選操作都會觸發這個函式的,這個點選操作包括:手指按下,移動手指,擡起手指
其中移動手指會多次觸發這個函式。
@Override public boolean dispatchTouchEvent(MotionEvent event) { // TODO Auto-generated method stub //super.dispatchTouchEvent(event); System.out.println("disdis2222222222"); //performClick(); switch (event.getAction()) { case MotionEvent.ACTION_DOWN: System.out.println("disdis"); break; default: break; } return true; }
這是我隨便寫的一個測試程式碼,正常的一次點選會觸發3次這個函式(按下,移動,擡起)
既然知道這件事情,我們就需要這樣想:我們總不能在手指只要在這個view有操作都多次呼叫這個函式吧。
所以最後我只在按下的時候才呼叫click的函式。
那接著探討下一個問題:怎麼呼叫click函式呢?
通過上面我推薦的那篇blog,我們知道最後呼叫這個函式的是在performClick這個函式裡面,由於我不關心其他點選事件,所以我就這麼寫了:
@Override
public boolean dispatchTouchEvent(MotionEvent event) {
// TODO Auto-generated method stub
switch (event.getAction()) {
case MotionEvent.ACTION_DOWN:
performClick();
break;
default:
break;
}
return true;
}
注意一下返回值,我返回true意思是這個函式消費了點選事件,因為這次點選事件是在這裡被消費掉的,所以最好返回true,不然的話不知道會出現什麼錯誤呢。
下面是我的整個重寫的Button類:
package com.example.luntan;
import android.content.Context;
import android.util.AttributeSet;
import android.view.MotionEvent;
import android.widget.Button;
public class MyButton extends Button {
public MyButton(Context context) {
// TODO Auto-generated constructor stub
super(context);
}
public MyButton(Context context, AttributeSet attrs) {
super(context, attrs);
// TODO Auto-generated constructor stub
}
@Override
public boolean dispatchTouchEvent(MotionEvent event) {
// TODO Auto-generated method stub
switch (event.getAction()) {
case MotionEvent.ACTION_DOWN:
performClick();
break;
default:
break;
}
return false;
}
}
在其他Activity中設定onClick:
holder.replyBt.setOnClickListener(new OnClickListener() {
@Override
public void onClick(View v) {
// TODO Auto-generated method stub
Intent intent = new Intent(ForumMain.this,
ForumReply.class);
String topicId = info.TopicID;
intent.putExtra("topicId", topicId);
startActivity(intent);
}
});