电脑知识|欧美黑人一区二区三区|软件|欧美黑人一级爽快片淫片高清|系统|欧美黑人狂野猛交老妇|数据库|服务器|编程开发|网络运营|知识问答|技术教程文章 - 好吧啦网

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

聊聊Android中的事件分發機制

瀏覽:7日期: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
相關文章:
主站蜘蛛池模板: sus630/303cu不锈钢棒,440C/430F/17-4ph不锈钢研磨棒-江苏德镍金属科技有限公司 | 湖州织里童装_女童男童中大童装_款式多尺码全_织里儿童网【官网】-嘉兴嘉乐网络科技有限公司 | 萃取箱-萃取槽-PVC萃取箱厂家-混合澄清槽- 杭州南方化工设备 | 厂房出租_厂房出售_产业园区招商_工业地产 - 中工招商网 | 定制/定做冲锋衣厂家/公司-订做/订制冲锋衣价格/费用-北京圣达信 | 泥沙分离_泥沙分离设备_泥砂分离机_洛阳隆中重工机械有限公司 | 双工位钻铣攻牙机-转换工作台钻攻中心-钻铣攻牙机一体机-浙江利硕自动化设备有限公司 | 高压无油空压机_无油水润滑空压机_水润滑无油螺杆空压机_无油空压机厂家-科普柯超滤(广东)节能科技有限公司 | 骨密度检测仪_骨密度分析仪_骨密度仪_动脉硬化检测仪专业生产厂家【品源医疗】 | 宁夏活性炭_防护活性炭_催化剂载体炭-宁夏恒辉活性炭有限公司 | 仿清水混凝土_清水混凝土装修_施工_修饰_保护剂_修补_清水混凝土修复-德州忠岭建筑装饰工程 | 螺钉式热电偶_便携式温度传感器_压簧式热电偶|无锡联泰仪表有限公司|首页 | 打孔器,打孔钳厂家【温州新星德牌五金工具】 | 超声波焊接机,振动摩擦焊接机,激光塑料焊接机,超声波焊接模具工装-德召尼克(常州)焊接科技有限公司 | 考勤系统_人事考勤管理系统_本地部署BS考勤系统_考勤软件_天时考勤管理专家 | 阜阳在线-阜阳综合门户| 物和码官网,物和码,免费一物一码数字化营销SaaS平台 | 小小作文网_中小学优秀作文范文大全| 车充外壳,车载充电器外壳,车载点烟器外壳,点烟器连接头,旅行充充电器外壳,手机充电器外壳,深圳市华科达塑胶五金有限公司 | 团建-拓展-拓展培训-拓展训练-户外拓展训练基地[无锡劲途] | 屏蔽服(500kv-超高压-特高压-电磁)-徐吉电气 | 上海新光明泵业制造有限公司-电动隔膜泵,气动隔膜泵,卧式|立式离心泵厂家 | 冷油器,取样冷却器,热力除氧器-连云港振辉机械设备有限公司 | 数控走心机-走心机价格-双主轴走心机-宝宇百科 | 大通天成企业资质代办_承装修试电力设施许可证_增值电信业务经营许可证_无人机运营合格证_广播电视节目制作许可证 | 菏泽商标注册_菏泽版权登记_商标申请代理_菏泽商标注册去哪里 | 海外整合营销-独立站营销-社交媒体运营_广州甲壳虫跨境网络服务 焊管生产线_焊管机组_轧辊模具_焊管设备_焊管设备厂家_石家庄翔昱机械 | 成都茶楼装修公司 - 会所设计/KTV装修 - 成都朗煜装饰公司 | 齿轮减速机_齿轮减速电机-VEMT蜗轮蜗杆减速机马达生产厂家瓦玛特传动瑞环机电 | 艺术生文化课培训|艺术生文化课辅导冲刺-济南启迪学校 | 恒温油槽-恒温水槽-低温恒温槽厂家-宁波科麦仪器有限公司 | 十字轴_十字轴万向节_十字轴总成-南京万传机械有限公司 | Win10系统下载_32位/64位系统/专业版/纯净版下载 | 合肥白癜风医院_[治疗白癜风]哪家好_合肥北大白癜风医院 | 水厂自动化|污水处理中控系统|水利信息化|智慧水务|智慧农业-山东德艾自动化科技有限公司 | 淘气堡_室内儿童乐园_户外无动力儿童游乐设备-高乐迪(北京) | 避光流动池-带盖荧光比色皿-生化流动比色皿-宜兴市晶科光学仪器 东莞爱加真空科技有限公司-进口真空镀膜机|真空镀膜设备|Polycold维修厂家 | 水性绝缘漆_凡立水_绝缘漆树脂_环保绝缘漆-深圳维特利环保材料有限公司 | CTP磁天平|小电容测量仪|阴阳极极化_双液系沸点测定仪|dsj电渗实验装置-南京桑力电子设备厂 | CTP磁天平|小电容测量仪|阴阳极极化_双液系沸点测定仪|dsj电渗实验装置-南京桑力电子设备厂 | 河南卓美创业科技有限公司-河南卓美防雷公司-防雷接地-防雷工程-重庆避雷针-避雷器-防雷检测-避雷带-避雷针-避雷塔、机房防雷、古建筑防雷等-山西防雷公司 |