初識(shí)Android觸摸事件傳遞機(jī)制

前言

今天總結(jié)的一個(gè)知識(shí)點(diǎn)是Andorid中View事件傳遞機(jī)制细办,也是核心知識(shí)點(diǎn)懒熙,相信很多開發(fā)者在面對(duì)這個(gè)問題時(shí)候會(huì)覺得困惑,另外啤它,View的另外一個(gè)難題滑動(dòng)沖突奕筐,比如在ScrollView中嵌套ListView,都是上下滑動(dòng)变骡,這該如何解決呢离赫,它解決依據(jù)就是View事件的傳遞機(jī)制,所以開發(fā)者需要對(duì)View的事件傳遞機(jī)制有較深入的理解塌碌。

目錄

  • Activity渊胸、View、ViewGroup三者關(guān)系
  • 觸摸事件類型
  • 事件傳遞三個(gè)階段
  • View事件傳遞機(jī)制
  • ViewGroup事件傳遞機(jī)制
  • 小結(jié)

Activity台妆、View翎猛、ViewGroup三者關(guān)系

我們都知道Android中看到的頁面很多是Activity組件,然后在Activity中嵌套控件接剩,比如TextView办成、RelativeLayout布局等,其實(shí)這些控件的基類都是View這個(gè)抽象類搂漠,而ViewGroup也是View的子類,區(qū)別在于ViewGroup是可以當(dāng)做其他子類的容器某弦,一張關(guān)系圖如下:

View Hierarchy

簡單一句話桐汤,這些View控件的載體是Activity而克,Activity通過從DecorView開始進(jìn)行繪制。

觸摸事件類型

  • ACTION_DOWN:用戶手指按下操作怔毛,往往也代表著一次觸摸事件的開始员萍。
  • ACTION_MOVE:用戶手指在屏幕上移動(dòng),一般情況下的輕微移動(dòng)都會(huì)觸發(fā)一系列的移動(dòng)事件拣度。
  • ACTION_POINTER_DOWN:額外的手指按下操作碎绎。
  • ACTION_POINTER_UP:額外的手指的離開操作
  • ACTION_UP:用戶手指離開屏幕的操作,一次抬起操作標(biāo)志著一次觸摸事件的結(jié)束抗果。

在一次屏幕觸摸操作中筋帖,ACTION_DOWNACTION_UP是必需的,ACTION_MOVE則是看情況而定冤馏,如果只是點(diǎn)擊日麸,那么檢測到只有按下和抬起操作。

事件傳遞三個(gè)階段

  • 分發(fā)(Dispatch):事件的分發(fā)對(duì)應(yīng)著dispatchTouchEvent方法逮光,在Andorid系統(tǒng)中代箭,所有的觸摸事件都是通過這個(gè)方法來分發(fā)的。

    boolean dispatchTouchEvent (MotionEvent ev)
    

    這個(gè)方法中涕刚,可以決定直接消費(fèi)這個(gè)事件或者將事件繼續(xù)分發(fā)給子視圖處理嗡综。

  • 攔截(Intercept):事件攔截對(duì)應(yīng)著onInterceptTouchEvent方法,這個(gè)方法只有在ViewGroup及其子類中才存在杜漠,在View和Activity中是不存在的极景。

    boolean onInterceptTouchEvent (MotionEvent ev)
    

    這個(gè)方法用來判斷是否攔截某個(gè)事件,如果攔截了某個(gè)事件碑幅,那么在同一序列事件當(dāng)中戴陡,那么這個(gè)方法不會(huì)被再次調(diào)用。

  • 消費(fèi)(Consume):事件消費(fèi)對(duì)應(yīng)著onTouchEvent方法沟涨。

    boolean onTouchEvent (MotionEvent event)
    

    用來處理點(diǎn)擊事件恤批,返回結(jié)果表示是否消耗當(dāng)前事件,如果不消耗裹赴,則在同一事件序列中喜庞,當(dāng)前View無法再接收到事件

在Android系統(tǒng)中,擁有事件傳遞處理能力的有三種:

  • Activity:擁有dispatchTouchEvent棋返、onTouchEvent兩個(gè)方法延都。
  • ViewGroup:擁有dispatchTouchEvent、onInterceptTouchEvent睛竣、onTouchEvent三個(gè)方法晰房。
  • View:擁有dispatchTouchEvent、onTouchEvent兩個(gè)方法。

View事件傳遞機(jī)制

這里說的View指的是除了ViewGroup之外的View控件殊者,比如TextView与境、Button、CheckBox等猖吴,View控件本身就是最小的單位摔刁,不能作為其他View的容器,View擁有dispatchTouchEvent海蔽、onTouchEvent兩個(gè)方法共屈,所以這里就定義了一個(gè)繼承TextView的類MyTextView,通過代碼查看日志党窜,看流程如何走拗引。

public class MyTextView extends TextView {

    private static final String TAG = "MyTextView";

    public MyTextView(Context context) {
        super(context);
    }

    public MyTextView(Context context, AttributeSet attrs) {
        super(context, attrs);
    }

    @Override
    public boolean dispatchTouchEvent(MotionEvent ev) {
        switch (ev.getAction()) {
            case MotionEvent.ACTION_DOWN:
                Log.e(TAG, "dispatchTouchEvent ACTION_DOWN");
                break;
            case MotionEvent.ACTION_MOVE:
                Log.e(TAG, "dispatchTouchEvent ACTION_MOVE");
                break;
            case MotionEvent.ACTION_UP:
                Log.e(TAG, "dispatchTouchEvent ACTION_UP");
                break;
            case MotionEvent.ACTION_CANCEL:
                Log.e(TAG, "dispatchTouchEvent ACTION_CANCEL");
                break;
            default:
                break;
        }
        return super.dispatchTouchEvent(ev);
    }

    @Override
    public boolean onTouchEvent(MotionEvent event) {
        switch (event.getAction()) {
            case MotionEvent.ACTION_DOWN:
                Log.e(TAG, "onTouchEvent ACTION_DOWN");
                break;
            case MotionEvent.ACTION_MOVE:
                Log.e(TAG, "onTouchEvent ACTION_MOVE");
                break;
            case MotionEvent.ACTION_UP:
                Log.e(TAG, "onTouchEvent ACTION_UP");
                break;
            case MotionEvent.ACTION_CANCEL:
                Log.e(TAG, "onTouchEvent ACTION_CANCEL");
                break;
            default:
                break;
        }
        return super.onTouchEvent(event);
    }

}

同時(shí)定義一個(gè)MainActivity類用來展示MyTextView,在這個(gè)Activity中刑然,我們?yōu)镸yTextView設(shè)置了點(diǎn)擊onClick和onTouch監(jiān)聽寺擂,方便跟蹤了解事件傳遞的流程。

public class MainActivity extends AppCompatActivity implements View.OnClickListener, View.OnTouchListener {

    private static final String TAG = "MainActivity";

    private MyTextView mTextView;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        mTextView = (MyTextView) findViewById(R.id.my_text_view);
        mTextView.setOnClickListener(this); // 設(shè)置MyTextView的點(diǎn)擊處理
        mTextView.setOnTouchListener(this); // 設(shè)置MyTextView的觸摸處理
    }

    @Override
    public boolean dispatchTouchEvent(MotionEvent ev) {
        switch (ev.getAction()) {
            case MotionEvent.ACTION_DOWN:
                Log.e(TAG, "dispatchTouchEvent ACTION_DOWN");
                break;
            case MotionEvent.ACTION_MOVE:
                Log.e(TAG, "dispatchTouchEvent ACTION_MOVE");
                break;
            case MotionEvent.ACTION_UP:
                Log.e(TAG, "dispatchTouchEvent ACTION_UP");
                break;
            case MotionEvent.ACTION_CANCEL:
                Log.e(TAG, "dispatchTouchEvent ACTION_CANCEL");
                break;
            default:
                break;
        }
        return super.dispatchTouchEvent(ev);
    }

    @Override
    public boolean onTouchEvent(MotionEvent event) {
        switch (event.getAction()) {
            case MotionEvent.ACTION_DOWN:
                Log.e(TAG, "onTouchEvent ACTION_DOWN");
                break;
            case MotionEvent.ACTION_MOVE:
                Log.e(TAG, "onTouchEvent ACTION_MOVE");
                break;
            case MotionEvent.ACTION_UP:
                Log.e(TAG, "onTouchEvent ACTION_UP");
                break;
            case MotionEvent.ACTION_CANCEL:
                Log.e(TAG, "onTouchEvent ACTION_CANCEL");
                break;
            default:
                break;
        }
        return super.onTouchEvent(event);
    }

    @Override
    public void onClick(View view) {
        switch (view.getId()) {
            case R.id.my_text_view:
                Log.e(TAG, "MyTextView onClick");
                break;
            default:
                break;
        }
    }

    @Override
    public boolean onTouch(View view, MotionEvent motionEvent) {
        switch(view.getId()) {
            case R.id.my_text_view:
                switch (motionEvent.getAction()) {
                    case MotionEvent.ACTION_DOWN:
                        Log.e(TAG, "MyTextView onTouch ACTION_DOWN");
                        break;
                    case MotionEvent.ACTION_MOVE:
                        Log.e(TAG, "MyTextView onTouch ACTION_MOVE");
                        break;
                    case MotionEvent.ACTION_UP:
                        Log.e(TAG, "MyTextView onTouch ACTION_UP");
                        break;
                    default:
                        break;
                }
                break;
            default:
                break;
        }
        return false;
    }
}

查看結(jié)果:

View日志結(jié)果

從中可以看到泼掠,事件是從down-move-up這樣順序執(zhí)行怔软,onTouch方法優(yōu)先于onClick方法調(diào)用,如果都是以super方法傳遞的話择镇,最后的結(jié)果是在MyTextView的onTouchEvent方法內(nèi)被消費(fèi)的挡逼,如果不消費(fèi)的話,則會(huì)把事件返回到它的父級(jí)去消費(fèi)腻豌,如果父級(jí)也沒消費(fèi)家坎,那么最終會(huì)返回到Activity中處理。

ViewGroup事件傳遞機(jī)制

ViewGroup作為View控件的容器存在吝梅,ViewGroup擁有dispatchTouchEvent虱疏、onInterceptTouchEvent、onTouchEvent三個(gè)方法苏携。同樣做瞪,我們自定義一個(gè)ViewGroup,繼承自RelativeLayout右冻,實(shí)現(xiàn)一個(gè)MyRelativeLayout装蓬。

public class MyRelativeLayout extends RelativeLayout {

    private static final String TAG = "MyRelativeLayout";

    public MyRelativeLayout(Context context) {
        super(context);
    }

    public MyRelativeLayout(Context context, AttributeSet attrs) {
        super(context, attrs);
    }

    @Override
    public boolean dispatchTouchEvent(MotionEvent ev) {
        switch (ev.getAction()) {
            case MotionEvent.ACTION_DOWN:
                Log.e(TAG, "dispatchTouchEvent ACTION_DOWN");
                break;
            case MotionEvent.ACTION_MOVE:
                Log.e(TAG, "dispatchTouchEvent ACTION_MOVE");
                break;
            case MotionEvent.ACTION_UP:
                Log.e(TAG, "dispatchTouchEvent ACTION_UP");
                break;
            case MotionEvent.ACTION_CANCEL:
                Log.e(TAG, "dispatchTouchEvent ACTION_CANCEL");
                break;
            default:
                break;
        }
        return super.dispatchTouchEvent(ev);
    }

    @Override
    public boolean onInterceptTouchEvent(MotionEvent ev) {
        switch (ev.getAction()) {
            case MotionEvent.ACTION_DOWN:
                Log.e(TAG, "onInterceptTouchEvent ACTION_DOWN");
                break;
            case MotionEvent.ACTION_MOVE:
                Log.e(TAG, "onInterceptTouchEvent ACTION_MOVE");
                break;
            case MotionEvent.ACTION_UP:
                Log.e(TAG, "onInterceptTouchEvent ACTION_UP");
                break;
            default:
                break;
        }
        return true;
    }

    @Override
    public boolean onTouchEvent(MotionEvent event) {
        switch (event.getAction()) {
            case MotionEvent.ACTION_DOWN:
                Log.e(TAG, "onTouchEvent ACTION_DOWN");
                break;
            case MotionEvent.ACTION_MOVE:
                Log.e(TAG, "onTouchEvent ACTION_MOVE");
                break;
            case MotionEvent.ACTION_UP:
                Log.e(TAG, "onTouchEvent ACTION_UP");
                break;
            case MotionEvent.ACTION_CANCEL:
                Log.e(TAG, "onTouchEvent ACTION_CANCEL");
                break;
            default:
                break;
        }
        return super.onTouchEvent(event);
    }
}

查看結(jié)果:

ViewGroup日志結(jié)果

從中可以看到觸摸事件的傳遞順序也是從Activity到ViewGroup,再由ViewGroup遞歸傳遞給它的子View纱扭。ViewGroup通過onInterceptTouchEvent方法對(duì)事件進(jìn)行攔截牍帚,如果該方法返回true,則事件不會(huì)繼續(xù)傳遞給子View乳蛾,如果返回false或者super.onInterceptTouchEvent暗赶,則事件會(huì)繼續(xù)傳遞給子View鄙币。在子View中對(duì)事件進(jìn)行消費(fèi)后,ViewGroup將不接收到任何事件蹂随。

小結(jié)

在Android系統(tǒng)事件中爱榔,View和ViewGroup的偽代碼如下:

public boolean dispatchTouchEvent(MotionEvent ev){
  boolean consume = false;
  if(onInterceptTouchEvent(ev)){
    consume = onTouchEvent(ev);
  }
  else{
    consume = child.dispatchTouchEvent(ev);
  }
  return consume;
}

用三張圖來表示Android中觸摸機(jī)制的流程。

1糙及,View內(nèi)觸摸事件不消費(fèi)

事件不消費(fèi)

2,View內(nèi)觸摸事件消費(fèi)

事件被消費(fèi)

3筛欢,ViewGroup攔截觸摸事件

事件被攔截

一些總結(jié):

  • 同一個(gè)事件序列是指從手指接觸屏幕的那一刻起浸锨,到手指離開屏幕的那一刻結(jié)束。一般是以down事件開始版姑,中間含有數(shù)量不定的move事件柱搜,最終以u(píng)p事件結(jié)束。
  • 正常情況下剥险,一個(gè)事件序列只能被一個(gè)View攔截且消耗聪蘸。
  • 某個(gè)View一旦決定攔截,那么這個(gè)事件序列就只能由它來處理表制,那么同一事件序列中的其他事件都不會(huì)再交給它來處理健爬,并且事件將重新交給它的父元素去處理,即父元素的onTouchEvent會(huì)被調(diào)用么介。
  • 如果View不消耗除ACTION_DOWN以外的其他事件娜遵,那么這個(gè)點(diǎn)擊事件就會(huì)消失,此時(shí)父元素的onTouchEvent并不會(huì)被調(diào)用壤短,最終會(huì)交給Activity處理设拟。
  • ViewGroup默認(rèn)不攔截任何事件。
  • View中沒有onInterceptTouchEvent方法久脯。
  • View的onTouchEvent默認(rèn)都會(huì)被消耗纳胧,除非它是不可點(diǎn)擊的。
  • 事件傳遞過程是由外向內(nèi)的帘撰,即事件先是傳遞給父元素跑慕,然后再由父元素分發(fā)給子View。

參考地址:

1骡和,https://www.youtube.com/watch?v=EZAoJU-nUyI

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末相赁,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子慰于,更是在濱河造成了極大的恐慌钮科,老刑警劉巖,帶你破解...
    沈念sama閱讀 218,941評(píng)論 6 508
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件婆赠,死亡現(xiàn)場離奇詭異绵脯,居然都是意外死亡佳励,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,397評(píng)論 3 395
  • 文/潘曉璐 我一進(jìn)店門蛆挫,熙熙樓的掌柜王于貴愁眉苦臉地迎上來赃承,“玉大人,你說我怎么就攤上這事悴侵∏破剩” “怎么了?”我有些...
    開封第一講書人閱讀 165,345評(píng)論 0 356
  • 文/不壞的土叔 我叫張陵可免,是天一觀的道長葵诈。 經(jīng)常有香客問我搏嗡,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 58,851評(píng)論 1 295
  • 正文 為了忘掉前任赃梧,我火速辦了婚禮治笨,結(jié)果婚禮上妓笙,老公的妹妹穿的比我還像新娘宰衙。我一直安慰自己,他們只是感情好闯估,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,868評(píng)論 6 392
  • 文/花漫 我一把揭開白布灼舍。 她就那樣靜靜地躺著,像睡著了一般睬愤。 火紅的嫁衣襯著肌膚如雪片仿。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 51,688評(píng)論 1 305
  • 那天尤辱,我揣著相機(jī)與錄音砂豌,去河邊找鬼。 笑死光督,一個(gè)胖子當(dāng)著我的面吹牛阳距,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播结借,決...
    沈念sama閱讀 40,414評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼筐摘,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了船老?” 一聲冷哼從身側(cè)響起咖熟,我...
    開封第一講書人閱讀 39,319評(píng)論 0 276
  • 序言:老撾萬榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎柳畔,沒想到半個(gè)月后馍管,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,775評(píng)論 1 315
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡薪韩,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,945評(píng)論 3 336
  • 正文 我和宋清朗相戀三年确沸,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了捌锭。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 40,096評(píng)論 1 350
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡罗捎,死狀恐怖观谦,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情桨菜,我是刑警寧澤豁状,帶...
    沈念sama閱讀 35,789評(píng)論 5 346
  • 正文 年R本政府宣布,位于F島的核電站倒得,受9級(jí)特大地震影響替蔬,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜屎暇,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,437評(píng)論 3 331
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望驻粟。 院中可真熱鬧根悼,春花似錦、人聲如沸蜀撑。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,993評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽酷麦。三九已至矿卑,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間沃饶,已是汗流浹背母廷。 一陣腳步聲響...
    開封第一講書人閱讀 33,107評(píng)論 1 271
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留糊肤,地道東北人琴昆。 一個(gè)月前我還...
    沈念sama閱讀 48,308評(píng)論 3 372
  • 正文 我出身青樓,卻偏偏與公主長得像馆揉,于是被迫代替她去往敵國和親业舍。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,037評(píng)論 2 355

推薦閱讀更多精彩內(nèi)容