新聞中心

EEPW首頁(yè) > 嵌入式系統(tǒng) > 設(shè)計(jì)應(yīng)用 > 深入Android開(kāi)發(fā)之--Android事件模型

深入Android開(kāi)發(fā)之--Android事件模型

作者: 時(shí)間:2016-10-08 來(lái)源:網(wǎng)絡(luò) 收藏

一般我們處理事件,都是針對(duì)某一個(gè)View來(lái)處理了,要么是添加onTouchListener監(jiān)聽(tīng)器,要么繼承View然后重寫(xiě)View#onTouchEvent,

本文引用地址:http://2s4d.com/article/201610/305959.htm

甚至不用重寫(xiě),只要使用Widget自己的監(jiān)聽(tīng)函數(shù) ,或者GestureDetector就OK了.

但是理解Android事件模型,對(duì)于理解GestureDetector,及Android事件的交互,寫(xiě)出具有出色的交互的應(yīng)用.

都是必經(jīng)之路.

一:ViewGroup與View的事件模型

我們都知道Android界面實(shí)際是一棵View的樹(shù).枝干是ViewGroup.

ViewGroup繼承自View,但是又是管理View的容器.那么ViewGroup與View的事件關(guān)系是怎么樣的呢?

這需要從另一個(gè)重要的ViewGroup中的方法,如下說(shuō)起:

1public boolean onInterceptTouchEvent(MotionEvent ev) {

2 return false;

3}

它的默認(rèn)實(shí)現(xiàn)很簡(jiǎn)單,就是把事件交給子View去處理.自己不攔截.

Intercept就是攔截的意思.

此方法的注釋,對(duì)于ViewGroup與View的事件模型說(shuō)得很清楚,

主要是以下幾點(diǎn):

(1) 如果此方法返回false,說(shuō)明此ViewGroup暫時(shí)(只是暫時(shí))對(duì)于觸控事件不感興趣.

但是不知道后面的事件它感不感興趣.所以后續(xù)事件還會(huì)一直傳遞到此方法中來(lái),供此方法判斷.

(2) 如果此方法返回true了.那么說(shuō)明此方法對(duì)應(yīng)的ViewGroup開(kāi)始對(duì)于此事件(或者手勢(shì))感興趣了.

那么后續(xù)事件就會(huì)直接給此方法對(duì)應(yīng)的ViewGroup的onTouchEvent方法來(lái)處理事件了.

(3) 如果此方法一開(kāi)始返回false,說(shuō)不感興趣這個(gè)時(shí)候事件發(fā)給了目錄View.

現(xiàn)在又返回true,說(shuō)感興趣了.那么目錄View就會(huì)收到一個(gè)action為ACTION_CANCEL的事件.

跟此方法返回true時(shí)的事件是同一個(gè)事件 ,只是action變了.

(4) ViewGroup會(huì)在這里接收觸控開(kāi)始的事件.

規(guī)則就是上面這些 ,那么是誰(shuí)在后面處理這些規(guī)則呢?

就是ViewGroup.它在disptachTouchEvent方法中,進(jìn)行了一系列的處理來(lái)實(shí)現(xiàn)這種模型.

public boolean dispatchTouchEvent(MotionEvent ev)

對(duì)于單獨(dú)的View本身來(lái)說(shuō),它也有一個(gè)簡(jiǎn)單的事件派發(fā)模型.通過(guò)以下代碼就可以很明白的看出來(lái)了:

View#dispatchTouchEvent(MotionEvent event):

1ListenerInfo li = mListenerInfo;

2if (li != null li.mOnTouchListener != null (mViewFlags ENABLED_MASK) == ENABLED

3 li.mOnTouchListener.onTouch(this, event)) {

4 return true;

5}

6

7if (onTouchEvent(event)) {

8 return true;

9}

二: Activity與View的事件模型

事件先到Activity中,然后Activity調(diào)用:

01/**

02 * Called to process touch screen events. You can override this to

03 * intercept all touch screen events before they are dispatched to the

04 * window. Be sure to call this implementation for touch screen events

05 * that should be handled normally.

06 *

07 * @param ev The touch screen event.

08 *

09 * @return boolean Return true if this event was consumed.

10 */

11 public boolean dispatchTouchEvent(MotionEvent ev) {

12 if (ev.getAction() == MotionEvent.ACTION_DOWN) {

13 onUserInteraction();

14 }

15 if (getWindow().superDispatchTouchEvent(ev)) {

16 return true;

17 }

18 return onTouchEvent(ev);

19 }

來(lái)分發(fā)事件, 這里的邏輯是:

先讓用戶界面窗口處理:getWindow().superDispatchTouchEvent(ev)

如果窗口沒(méi)有處理這個(gè)事件.

那就交給Activity自己處理.return onTouchEvent(ev)

這個(gè)Window跟View層級(jí)是怎么交互的呢?

我們找到了Window的實(shí)現(xiàn)類(lèi):PhoneWindow(com.android.internal.policy.impl.PhoneWindow)

1@Override

2 public boolean superDispatchTouchEvent(MotionEvent event) {

3 return mDecor.superDispatchTouchEvent(event);

4 }

這個(gè)mDecor就是用戶界面的根View了.

private final class DecorView extends FrameLayout

(com.android.internal.policy.impl.PhoneWindow.DecorView)

原來(lái)窗口將事件交給根View來(lái)進(jìn)行事件派發(fā)的.

mDecor調(diào)用自己的superDispatchTouchEvent(event)

然后將事件派發(fā)的任務(wù)交給了自己的dispatchTouchEvent

1public boolean superDispatchTouchEvent(MotionEvent event) {

2 return super.dispatchTouchEvent(event);

3}

這里調(diào)用的super.dispatchTouchEvent 就是ViewGroup的聲明的dispatchTouchEvent的了.



關(guān)鍵詞:

評(píng)論


相關(guān)推薦

技術(shù)專(zhuān)區(qū)

關(guān)閉