日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区

您的位置:首頁技術文章
文章詳情頁

聊聊Android中的事件分發機制

瀏覽:16日期:2022-09-19 11:38:26

View事件分發機制的本質就是就是MotionEvent事件的分發過程,即MotionEvent產生后是怎樣在View之間傳遞及處理的。

首先介紹一下什么是MotionEvent.所謂MotionEvent,即用戶手指觸碰手機屏幕時產生的一系列觸摸事件。典型的觸摸事件有:

ACTION_DOWN:手指剛接觸屏幕的一瞬間。 ACTION_MOVE:手指在屏幕上滑動。 ACTION_UP:手指離開屏幕的一瞬間。 ACTION_CANCLE:當前事件序列終止。

一個事件序列一般都是以DOWN事件開始,UP事件終止,中間穿插數個MOVE事件。

事件的傳遞順序:Activity(Window) → ViewGroup → View,即事件是自Activity往下傳遞。

事件的分發涉及到的三個主要方法:

dispatchTouchEvent: 自頂向下傳遞事件。其返回值受子View的dispatchTouchEvent方法和當前View的onTouchEvent方法影響。 onInterceptTouchEvent: 對事件進行攔截。此方法為ViewGroup獨有。一旦對事件序列中的某事件進行攔截,該序列剩余事件都會交給攔截的ViewGroup處理,并且不會再次調用此方法。 onTouchEvent: 消耗某事件,即對某事件進行處理。

接下來將分別對Activity, ViewGroup, View的事件分發機制進行說明。

Activity的事件分發機制

當一個點擊事件發生時,該事件最先傳遞到Activity的dispatchTouchEvent()方法中進行處理。Activity會在dispatchTouchEvent()方法中調用getWindow().superDispatchTouchEvent()方法,將事件傳遞給Window的mDecor(DecorView)進行處理,而mDecor則會通過調用superDispatchTouchEvent方法將事件傳給ViewGroup進行處理。

/** * 源碼分析:Activity.dispatchTouchEvent() */ public boolean dispatchTouchEvent(MotionEvent ev) {if (ev.getAction() == MotionEvent.ACTION_DOWN) {onUserInteraction(); } if (getWindow().superDispatchTouchEvent(ev)) {return true;// 若getWindow().superDispatchTouchEvent(ev)的返回true// 則Activity.dispatchTouchEvent()就返回true,則方法結束。即 :該點擊事件停止往下傳遞 & 事件傳遞過程結束// 否則:繼續往下調用Activity.onTouchEvent } return onTouchEvent(ev);}/** * getWindow().superDispatchTouchEvent(ev) * 說明: * a. getWindow() = 獲取Window類的對象 * b. Window類是抽象類,其唯一實現類 = PhoneWindow類;即此處的Window類對象 = PhoneWindow類對象 * c. Window類的superDispatchTouchEvent() = 1個抽象方法,由子類PhoneWindow類實現 */ @Override public boolean superDispatchTouchEvent(MotionEvent event) {return mDecor.superDispatchTouchEvent(event);// mDecor = 頂層View(DecorView)的實例對象 }/** * mDecor.superDispatchTouchEvent(event) * 定義:屬于頂層View(DecorView) * 說明: * a. DecorView類是PhoneWindow類的一個內部類 * b. DecorView繼承自FrameLayout,是所有界面的父類 * c. FrameLayout是ViewGroup的子類,故DecorView的間接父類 = ViewGroup */ public boolean superDispatchTouchEvent(MotionEvent event) {return super.dispatchTouchEvent(event);// 調用父類的方法 = ViewGroup的dispatchTouchEvent()// 即 將事件傳遞到ViewGroup去處理,詳細請看ViewGroup的事件分發機制 }/** * Activity.onTouchEvent() * 定義:屬于頂層View(DecorView) * 說明: * a. DecorView類是PhoneWindow類的一個內部類 * b. DecorView繼承自FrameLayout,是所有界面的父類 * c. FrameLayout是ViewGroup的子類,故DecorView的間接父類 = ViewGroup */ public boolean onTouchEvent(MotionEvent event) {// 當一個點擊事件未被Activity下任何一個View接收 / 處理時// 應用場景:處理發生在Window邊界外的觸摸事件if (mWindow.shouldCloseOnTouch(this, event)) { finish(); return true;}return false;// 即只有在點擊事件在Window邊界外才會返回true,一般情況都返回false }ViewGroup的事件分發機制

當事件從Activity傳遞到ViewGroup的dispatchTouchEvent()后,ViewGroup首先會調用onInterceptTouchEvent()方法判斷是否攔截該事件(默認不攔截,攔截的話需要用戶重寫),如果不攔截該事件,ViewGroup會通過for循環遍歷它所有的子View,找到當前事件發生的View,然后調用該子View的dispatchTouchEvent()方法,將事件分發給子View進行處理。如果該事件被ViewGroup攔截下來或者沒有找到事件發生的View(事件發生在空白處)的話,ViewGroup會調用它的onTouchEvent()方法對事件進行處理。

/** * 源碼分析:ViewGroup.dispatchTouchEvent() */ public boolean dispatchTouchEvent(MotionEvent ev) { ... // 僅貼出關鍵代碼// ViewGroup每次事件分發時,都需調用onInterceptTouchEvent()詢問是否攔截事件 if (disallowIntercept || !onInterceptTouchEvent(ev)) { // 判斷值1:disallowIntercept = 是否禁用事件攔截的功能(默認是false),可通過調用requestDisallowInterceptTouchEvent()修改 // 判斷值2: !onInterceptTouchEvent(ev) = 對onInterceptTouchEvent()返回值取反 // a. 若在onInterceptTouchEvent()中返回false(即不攔截事件),就會讓第二個值為true,從而進入到條件判斷的內部 // b. 若在onInterceptTouchEvent()中返回true(即攔截事件),就會讓第二個值為false,從而跳出了這個條件判斷 // c. 關于onInterceptTouchEvent() ->>分析1ev.setAction(MotionEvent.ACTION_DOWN); final int scrolledXInt = (int) scrolledXFloat; final int scrolledYInt = (int) scrolledYFloat; final View[] children = mChildren; final int count = mChildrenCount; // 通過for循環,遍歷了當前ViewGroup下的所有子View for (int i = count - 1; i >= 0; i--) { final View child = children[i]; if ((child.mViewFlags & VISIBILITY_MASK) == VISIBLE || child.getAnimation() != null) { child.getHitRect(frame); // 判斷當前遍歷的View是不是正在點擊的View,從而找到當前被點擊的View // 若是,則進入條件判斷內部 if (frame.contains(scrolledXInt, scrolledYInt)) { final float xc = scrolledXFloat - child.mLeft; final float yc = scrolledYFloat - child.mTop; ev.setLocation(xc, yc); child.mPrivateFlags &= ~CANCEL_NEXT_UP_EVENT; // 條件判斷的內部調用了該View的dispatchTouchEvent()// 即 實現了點擊事件從ViewGroup到子View的傳遞(具體請看下面的View事件分發機制)if (child.dispatchTouchEvent(ev)) { mMotionTarget = child; return true; // 調用子View的dispatchTouchEvent后是有返回值的// 若該控件可點擊,那么點擊時,dispatchTouchEvent的返回值必定是true,因此會導致條件判斷成立// 于是給ViewGroup的dispatchTouchEvent()直接返回了true,即直接跳出// 即把ViewGroup的點擊事件攔截掉} } } } } } boolean isUpOrCancel = (action == MotionEvent.ACTION_UP) || (action == MotionEvent.ACTION_CANCEL); if (isUpOrCancel) { mGroupFlags &= ~FLAG_DISALLOW_INTERCEPT; } final View target = mMotionTarget;// 若點擊的是空白處(即無任何View接收事件) / 攔截事件(手動復寫onInterceptTouchEvent(),從而讓其返回true)if (target == null) { ev.setLocation(xf, yf); if ((mPrivateFlags & CANCEL_NEXT_UP_EVENT) != 0) { ev.setAction(MotionEvent.ACTION_CANCEL); mPrivateFlags &= ~CANCEL_NEXT_UP_EVENT; } return super.dispatchTouchEvent(ev); // 調用ViewGroup父類的dispatchTouchEvent(),即View.dispatchTouchEvent() // 因此會執行ViewGroup的onTouch() ->> onTouchEvent() ->> performClick() ->> onClick(),即自己處理該事件,事件不會往下傳遞(具體請參考View事件的分發機制中的View.dispatchTouchEvent()) // 此處需與上面區別:子View的dispatchTouchEvent()} ... }/** * ViewGroup.onInterceptTouchEvent() * 作用:是否攔截事件 * 說明: * a. 返回true = 攔截,即事件停止往下傳遞(需手動設置,即復寫onInterceptTouchEvent(),從而讓其返回true) * b. 返回false = 不攔截(默認) */ public boolean onInterceptTouchEvent(MotionEvent ev) { return false; } View的事件分發機制

當事件從ViewGroup傳遞到了View的dispatchTouchEvent()之后,最先執行的是View的onTouch()方法。onTouch()方法是View的OnTouchListener接口中所定義的方法,如果用戶為View注冊了監聽,那么當用戶觸摸屏幕時便會觸發此方法。此方法默認返回false,需要用戶重寫。只有onTouch()方法返回false, 才會執行View的onTouchEvent()方法。然后會根據情況調用performClick()方法,performClick()方法隨之會調用onClick()方法。

/** * 源碼分析:View.dispatchTouchEvent() */ public boolean dispatchTouchEvent(MotionEvent event) { if (mOnTouchListener != null && (mViewFlags & ENABLED_MASK) == ENABLED && mOnTouchListener.onTouch(this, event)) { return true; } return onTouchEvent(event); } // 說明:只有以下3個條件都為真,dispatchTouchEvent()才返回true;否則執行onTouchEvent() // 1. mOnTouchListener != null // 2. (mViewFlags & ENABLED_MASK) == ENABLED // 3. mOnTouchListener.onTouch(this, event) // 下面對這3個條件逐個分析/** * 條件1:mOnTouchListener != null * 說明:mOnTouchListener變量在View.setOnTouchListener()方法里賦值 */ public void setOnTouchListener(OnTouchListener l) { mOnTouchListener = l; // 即只要我們給控件注冊了Touch事件,mOnTouchListener就一定被賦值(不為空)} /** * 條件2:(mViewFlags & ENABLED_MASK) == ENABLED * 說明: * a. 該條件是判斷當前點擊的控件是否enable * b. 由于很多View默認enable,故該條件恒定為true *//** * 條件3:mOnTouchListener.onTouch(this, event) * 說明:即 回調控件注冊Touch事件時的onTouch();需手動復寫設置,具體如下(以按鈕Button為例) */ button.setOnTouchListener(new OnTouchListener() { @Override public boolean onTouch(View v, MotionEvent event) { return false; } }); // 若在onTouch()返回true,就會讓上述三個條件全部成立,從而使得View.dispatchTouchEvent()直接返回true,事件分發結束 // 若在onTouch()返回false,就會使得上述三個條件不全部成立,從而使得View.dispatchTouchEvent()中跳出If,執行onTouchEvent(event)

若View的onTouchEvent()返回true, 即消耗了該事件,那么事件的分發到此結束。如果返回false,則會自下而上依次調用ViewGroup和Activity的onTouchEvent()方法對事件進行處理。值得一提的是,Activity的onTouchEvent()方法必須對事件進行處理。至此,事件的分發完成。

以上就是聊聊Android中的事件分發機制的詳細內容,更多關于Android 事件分發機制的資料請關注好吧啦網其它相關文章!

標簽: Android
相關文章:
日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区
国产66精品| 国产精品久久久久av蜜臀| 久久精品国产免费| 久久不卡日韩美女| 久久国产乱子精品免费女| 日韩va欧美va亚洲va久久| 鲁大师成人一区二区三区| 亚洲一区观看| 蜜桃久久av一区| 亚洲精品精选| 日韩va欧美va亚洲va久久| 欧美一区影院| 久久久亚洲欧洲日产| 精品三级国产| av资源中文在线天堂| 久久久久久久久久久妇女 | 美女黄网久久| 日本亚州欧洲精品不卡| 久久精品99国产精品| 精品国产美女a久久9999| 精品无人区麻豆乱码久久久| 欧美aa在线观看| 欧美~级网站不卡| 最新亚洲国产| 免费在线亚洲欧美| 视频二区不卡| 国产精品嫩草99av在线| 日本久久一区| 日韩精品永久网址| 丝袜亚洲另类欧美| 欧美亚洲一级| 欧美freesex黑人又粗又大| 婷婷综合在线| 日韩精品一级| 国产精品久久久久蜜臀| 欧美日韩水蜜桃| 一区二区三区午夜视频| 麻豆精品视频在线| 亚洲91久久| 亚洲三级在线| 高清日韩欧美| 水野朝阳av一区二区三区| 国产精品3区| 不卡中文一二三区| 国产精品九九| 亚洲一区欧美激情| 免费在线观看一区| 亚洲免费高清| 精品一区视频| 亚洲综合精品四区| 国产福利一区二区三区在线播放| 日韩欧美自拍| 日韩高清电影免费| 欧美sss在线视频| 四虎成人精品一区二区免费网站| 成人日韩av| 蜜芽一区二区三区| 日韩成人a**站| 天堂精品久久久久| 久久精品卡一| 国产激情久久| 久久夜色精品| 麻豆国产在线| 日韩免费精品| 97视频热人人精品免费| 中文字幕成人| 99精品视频精品精品视频| 国产精品最新| 亚洲伊人影院| 99久久久久国产精品| 国产伦精品一区二区三区千人斩 | 精品三级在线| 亚洲91在线| 亚洲香蕉网站| 国产精品s色| 中文字幕一区二区av| 午夜精品久久久久久久久久蜜桃| 日韩精品三区四区| 婷婷丁香综合| 韩国久久久久久| 国产精品亚洲欧美日韩一区在线| 亚洲激情欧美| 成人在线网站| 老司机免费视频一区二区| 最新国产精品视频| 一区福利视频| 久久三级视频| 国产一区二区三区四区五区传媒 | 久久av一区二区三区| 日韩国产欧美| 精品网站aaa| 国产精品久久久一区二区| 综合一区在线| 国产农村妇女精品一区二区| 四虎884aa成人精品最新| 久久不卡日韩美女| 日韩成人av影视| 中文字幕av一区二区三区人| 亚洲免费精品| 国产伊人精品| 色老板在线视频一区二区| 欧美aa在线视频| 久久精品av麻豆的观看方式| 亚洲一区二区av| 在线综合亚洲| 99视频一区| 国产亚洲永久域名| 最新亚洲激情| 国产一区导航| 久久一二三区| 在线精品一区| 免费在线成人网| 日韩制服丝袜先锋影音| 免费的成人av| 亚洲精品在线a| 中文字幕免费精品| 亚洲精品黄色| 奇米色欧美一区二区三区| 亚洲精品人人| 日韩黄色在线观看| 日韩国产在线观看一区| 奇米狠狠一区二区三区| 91成人在线| 久久av资源| 精品资源在线| 色婷婷亚洲mv天堂mv在影片| 日本在线高清| 久久久久久黄| 欧美日韩国产探花| 免费黄网站欧美| 综合色就爱涩涩涩综合婷婷| 亚洲精品一级| 国产精品永久| 国产极品一区| 国产一区二区三区久久| 成人三级高清视频在线看| 日韩电影免费网址| 香蕉久久99| 久久亚洲国产精品一区二区| 蜜臀av国产精品久久久久| 亚洲精品日本| 国产欧美一区二区色老头| 国产精品66| www在线观看黄色| 国产精品99免费看| 玖玖玖国产精品| 欧美午夜网站| 福利一区视频| 91精品蜜臀一区二区三区在线| 亚洲国内精品| 亚洲另类黄色| 美女精品久久| 色综合www| 丝袜亚洲精品中文字幕一区| 欧美片第1页综合| 日韩久久精品网| 亚洲激情婷婷| 日本91福利区| 日本一区二区高清不卡| 国产综合激情| 日韩中文一区二区| 欧美xxxx中国| 野花国产精品入口| 日韩av在线免费观看不卡| 精品国产精品国产偷麻豆| 免费成人网www| 日本电影久久久| 伊伊综合在线| 亚洲一区二区三区在线免费| 欧美黑人做爰爽爽爽| 99久久久久| 日韩黄色免费网站| 日韩精品dvd| 亚洲精品三级| av日韩中文| 亚洲精品乱码日韩| 91亚洲国产成人久久精品| 水蜜桃久久夜色精品一区的特点| 国产亚洲久久| 久久精品免费一区二区三区| 日本精品一区二区三区在线观看视频| 精品国产一区二区三区性色av| 五月综合激情| 麻豆精品在线视频| 午夜在线精品偷拍| 国产一区二区三区黄网站| 午夜亚洲精品| 毛片在线网站| 日本视频一区二区| 欧美亚洲激情| 国产精品九九| 国产精品毛片在线看| 精品久久久网| 久久成人国产| av资源亚洲| 亚洲色图网站| 久久久蜜桃一区二区人| 国产精品日韩精品在线播放| 亚洲欧美激情诱惑| 日韩欧美一区二区三区在线视频|