背景:在機頂盒中通過遙控器操作,與傳統手機互動方式不同。手機點選是發送兩個TouchEvent(Down和Up),機頂盒是發送KeyEvent。所産生的效果看似相同,其實是兩種不同的機制。
先看兩段代碼
1.目錄android.view.View
排程按鍵事件
public boolean dispatchKeyEvent(KeyEvent event) {
if (mInputEventConsistencyVerifier != null) {
mInputEventConsistencyVerifier.onKeyEvent(event, 0);
}
// Give any attached key listener a first crack at the event.
// noinspection SimplifiableIfStatement
ListenerInfo li = mListenerInfo;
if (li != null && li.mOnKeyListener != null
&& (mViewFlags & ENABLED_MASK) == ENABLED
&& li.mOnKeyListener.onKey(this, event.getKeyCode(), event)) {
return true;
}
if (event.dispatch(this,
mAttachInfo != null ? mAttachInfo.mKeyDispatchState : null,
this)) {
return true;
}
if (mInputEventConsistencyVerifier != null) {
mInputEventConsistencyVerifier.onUnhandledEvent(event, 0);
}
return false;
}
2.目錄android.view.View
ENTER鍵松開
public boolean onKeyUp(int keyCode, KeyEvent event) {
boolean result = false;
switch (keyCode) {
case KeyEvent.KEYCODE_DPAD_CENTER:
case KeyEvent.KEYCODE_ENTER: {
if ((mViewFlags & ENABLED_MASK) == DISABLED) {
return true;
}
if ((mViewFlags & CLICKABLE) == CLICKABLE && isPressed()) {
setPressed(false);
if (!mHasPerformedLongPress) {
// This is a tap, so remove the longpress check
removeLongPressCallback();
result = performClick();
}
}
break;
}
}
return result;
}
以上代碼是對于一個普通控件的ENTER鍵響應,當控件enabled && clickable時會執行 performClick()方法。若是正常執行,傳回True,表示這個按鍵事件已經被消費了,若是沒有正常執行,傳回false。
注意下,對于Button之類控件預設clickable,而TextView等是不能點選的。對于所有View在執行過setOnClickListener(Listener)後,即可點選。因為在API中有這麼一條。
public void setOnClickListener(OnClickListener l) {
if (!isClickable()) {
setClickable(true);
}
getListenerInfo().mOnClickListener = l;
}
以上是一個普通控件處理點選事件的流程。現在問題來了,在手機應用中當我們點選一個無用的控件,也可能産生點選事件(實質是其父控件消費了這個點選事件),但是使用遙控器操作時,似乎一個控件點不了就是點不了,它的父控件也不會處理,這是為什麼?
再看下ViewGroup中的代碼
3.目錄android.view.ViewGroup
public boolean dispatchKeyEvent(KeyEvent event) {
if (mInputEventConsistencyVerifier != null) {
mInputEventConsistencyVerifier.onKeyEvent(event, 1);
}
if ((mPrivateFlags & (FOCUSED | HAS_BOUNDS)) == (FOCUSED | HAS_BOUNDS)) {
if (super.dispatchKeyEvent(event)) {
return true;
}
} else if (mFocused != null && (mFocused.mPrivateFlags & HAS_BOUNDS) == HAS_BOUNDS) {
if (mFocused.dispatchKeyEvent(event)) {
return true;
}
}
if (mInputEventConsistencyVerifier != null) {
mInputEventConsistencyVerifier.onUnhandledEvent(event, 1);
}
return false;
}
這段代碼中,看到,當焦點在ViewGroup上時正常執行。當焦點在ViewGroup子控件上時,讓子控件消費這個按鍵事件。 而觸摸的點選事件,在子控件未消費TouchEvent時,傳回False,其父控件再響應點選事件,如此循環。在按鍵的事件中并沒有這樣的一個機制,是以按鍵事件和點選雖然最終都是調用目标的performClick()但是,對于boolean型的傳回值的處理,兩者并不一樣。
最終進行如下修改: 4.目錄android.view.ViewGroup
public boolean dispatchKeyEvent(KeyEvent event) {
if (mInputEventConsistencyVerifier != null) {
mInputEventConsistencyVerifier.onKeyEvent(event, 1);
}
if ((mPrivateFlags & (PFLAG_FOCUSED | PFLAG_HAS_BOUNDS)) == (PFLAG_FOCUSED | PFLAG_HAS_BOUNDS)) {
if (super.dispatchKeyEvent(event)) {
return true;
}
} else if (mFocused != null
&& (mFocused.mPrivateFlags & PFLAG_HAS_BOUNDS) == PFLAG_HAS_BOUNDS) {
if (mFocused.dispatchKeyEvent(event)) {
return true;
}
// add by suntianrui
else {
if (super.dispatchKeyEvent(event)) {
return true;
}
}
// end
}
if (mInputEventConsistencyVerifier != null) {
mInputEventConsistencyVerifier.onUnhandledEvent(event, 1);
}
return false;
}
這段代碼在原來基礎上,增加判斷,如果子控件未處理按鍵事件,則由父控件處理。