從Listview與Button點擊事件衝突看安卓點擊事件分發機制

題目有點長。其實實現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);
				}
			});




發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章