Android開發之觸摸事件處理機制詳解
android觸碰訊息傳遞機制
使用者的每次觸碰(onClick,onLongClick,onScroll,etc.)都是由一個ACTION_DOWN+n個ACTION_MOVE+1個ACTION_UP組成的,使用者觸碰必先有個ACTION_DOWN響應,使用者觸碰結束必然會有個ACTION_UP。(當然如果在途中被攔截,就可能不會有了!)那麼View是如何分發訊息和攔截訊息呢?
1.View及其子類都會有的兩個方法:
public boolean dispatchTouchEvent(MotionEvent ev) 這個方法用來分發TouchEvent
public boolean onTouchEvent(MotionEvent ev) 這個方法用來處理TouchEvent
2.特殊的View子類ViewGroup則還有一個方法:
public boolean onInterceptTouchEvent(MotionEvent ev) 這個方法用來攔截TouchEvent
3.分發
dispatchTouchEvent 收到觸碰,則向最外層的View傳遞訊息,再向子層的View分發
4.攔截:
onInterceptTouchEvent 攔截返回true表示要攔截訊息,不要再向子View傳遞(這裡的子View不是繼承關係,而是包容關係)。返回false則表示不攔截訊息,可以繼續向下一層級的View傳遞訊息,子View將可以dispatchTouchEvent 收到觸碰訊息再分發訊息
5.訊息處理:
onTouchEvent 處理事件,攔截了訊息,或者是最後一個收到訊息的View調用此方法來處理事件,若返回true,則表示正確接收並處理。若返回false則表示沒有被處理,將向父View傳遞(這裡的父View不是繼承關係,而是包容關係)
Android事件模型之interceptTouchEvnet ,onTouchEvent關係正解
參考文檔:
http://blog.csdn.net/liutao5757124/article/details/6097125
首先,看Android的官方文檔正解
onInterceptTouchEvent()與onTouchEvent()的機制:
1. down事件首先會傳遞到onInterceptTouchEvent()方法
2. 如果該ViewGroup的onInterceptTouchEvent()在接收到down事件處理完成之後return false,
那麼後續的move, up等事件將繼續會先傳遞給該ViewGroup,之後才和down事件一樣傳遞給最
終的目標view的onTouchEvent()處理
3. 如果該ViewGroup的onInterceptTouchEvent()在接收到down事件處理完成之後return true,
那麼後續的move, up等事件將不再傳遞給onInterceptTouchEvent(),而是和down事件一樣
傳遞給該ViewGroup的onTouchEvent()處理,注意,目標view將接收不到任何事件。
4. 如果最終需要處理事件的view的onTouchEvent()返回了false,那麼該事件將被傳遞至其上一
層次的view的onTouchEvent()處理
5. 如果最終需要處理事件的view 的onTouchEvent()返回了true,那麼後續事件將可以繼續傳遞
給該view的onTouchEvent()處理
這是官方文檔的說法,要是自己沒親自去寫個程式觀察哈,基本上沒法理解,所以上程式先,然後分析:
布局檔案main.xml
Java代碼
-
- android:orientation="vertical" android:layout_width="fill_parent"
- android:layout_height="fill_parent">
- android:orientation="vertical" android:layout_width="fill_parent"
- android:layout_height="fill_parent" android:gravity="center">
- android:layout_width="wrap_content" android:layout_height="wrap_content"
- android:id="@+id/tv" android:text="AB" android:textSize="40sp"
- android:textStyle="bold" android:background="#FFFFFF"
- android:textColor="#0000FF" />
-
-
-
第一層自訂布局LayoutView1.java
Java代碼
- package com.hao;
-
- import android.content.Context;
- import android.util.AttributeSet;
- import android.util.Log;
- import android.view.MotionEvent;
- import android.widget.LinearLayout;
-
- public class LayoutView1 extends LinearLayout {
- private final String TAG = "LayoutView1";
- public LayoutView1(Context context, AttributeSet attrs) {
- super(context, attrs);
- Log.e(TAG,TAG);
- }
-
- @Override
- public boolean onInterceptTouchEvent(MotionEvent ev) {
- int action = ev.getAction();
- switch(action){
- case MotionEvent.ACTION_DOWN:
- Log.e(TAG,"onInterceptTouchEvent action:ACTION_DOWN");
- // return true; 在這就攔截了,後面的就不會得到事件
- break;
- case MotionEvent.ACTION_MOVE:
- Log.e(TAG,"onInterceptTouchEvent action:ACTION_MOVE");
- break;
- case MotionEvent.ACTION_UP:
- Log.e(TAG,"onInterceptTouchEvent action:ACTION_UP");
- break;
- case MotionEvent.ACTION_CANCEL:
- Log.e(TAG,"onInterceptTouchEvent action:ACTION_CANCEL");
- break;
- }
- return false;
- }
-
- @Override
- public boolean onTouchEvent(MotionEvent ev) {
- int action = ev.getAction();
- switch(action){
- case MotionEvent.ACTION_DOWN:
- Log.e(TAG,"onTouchEvent action:ACTION_DOWN");
- break;
- case MotionEvent.ACTION_MOVE:
- Log.e(TAG,"onTouchEvent action:ACTION_MOVE");
- break;
- case MotionEvent.ACTION_UP:
- Log.e(TAG,"onTouchEvent action:ACTION_UP");
- break;
- case MotionEvent.ACTION_CANCEL:
- Log.e(TAG,"onTouchEvent action:ACTION_CANCEL");
- break;
- }
- return true;
- // return false;
- }
-
- @Override
- protected void onLayout(boolean changed, int l, int t, int r, int b) {
- // TODO Auto-generated method stub
- super.onLayout(changed, l, t, r, b);
- }
-
- @Override
- protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
- // TODO Auto-generated method stub
- super.onMeasure(widthMeasureSpec, heightMeasureSpec);
- }
- }
第二層布局LayoutView2.java
Java代碼
- package com.hao;
-
- import android.content.Context;
- import android.util.AttributeSet;
- import android.util.Log;
- import android.view.MotionEvent;
- import android.widget.LinearLayout;
-
- public class LayoutView2 extends LinearLayout {
-
- private final String TAG = "LayoutView2";
- public LayoutView2(Context context, AttributeSet attrs) {
- super(context, attrs);
- Log.e(TAG,TAG);
- }
-
- @Override
- public boolean onInterceptTouchEvent(MotionEvent ev) {
- int action = ev.getAction();
- switch(action){
- case MotionEvent.ACTION_DOWN:
- Log.e(TAG,"onInterceptTouchEvent action:ACTION_DOWN");
- // return true;
- break;
- case MotionEvent.ACTION_MOVE:
- Log.e(TAG,"onInterceptTouchEvent action:ACTION_MOVE");
- break;
- case MotionEvent.ACTION_UP:
- Log.e(TAG,"onInterceptTouchEvent action:ACTION_UP");
- break;
- case MotionEvent.ACTION_CANCEL:
- Log.e(TAG,"onInterceptTouchEvent action:ACTION_CANCEL");
- break;
- }
- return false;
- }
-
- @Override
- public boolean onTouchEvent(MotionEvent ev) {
- int action = ev.getAction();
- switch(action){
- case MotionEvent.ACTION_DOWN:
- Log.e(TAG,"onTouchEvent action:ACTION_DOWN");
- break;
- case MotionEvent.ACTION_MOVE:
- Log.e(TAG,"onTouchEvent action:ACTION_MOVE");
- break;
- case MotionEvent.ACTION_UP:
- Log.e(TAG,"onTouchEvent action:ACTION_UP");
- break;
- case MotionEvent.ACTION_CANCEL:
- Log.e(TAG,"onTouchEvent action:ACTION_CANCEL");
- break;
- }
- // return true;
- return false;
- }
- }
-
自訂MyTextView.java
Java代碼
- package com.hao;
-
- import android.content.Context;
- import android.util.AttributeSet;
- import android.util.Log;
- import android.view.MotionEvent;
- import android.view.View;
- import android.widget.TextView;
-
- public class MyTextView extends TextView {
- private final String TAG = "MyTextView";
- public MyTextView(Context context, AttributeSet attrs) {
- super(context, attrs);
- Log.e(TAG,TAG);
- }
-
- @Override
- public boolean onTouchEvent(MotionEvent ev) {
- int action = ev.getAction();
- switch(action){
- case MotionEvent.ACTION_DOWN:
- Log.e(TAG,"onTouchEvent action:ACTION_DOWN");
- break;
- case MotionEvent.ACTION_MOVE:
- Log.e(TAG,"onTouchEvent action:ACTION_MOVE");
- break;
- case MotionEvent.ACTION_UP:
- Log.e(TAG,"onTouchEvent action:ACTION_UP");
- break;
- case MotionEvent.ACTION_CANCEL:
- Log.e(TAG,"onTouchEvent action:ACTION_CANCEL");
- break;
- }
- return false;
- // return true;
- }
-
- public void onClick(View v) {
- Log.e(TAG, "onClick");
- }
-
- public boolean onLongClick(View v) {
- Log.e(TAG, "onLongClick");
- return false;
- }
- }
-
其實代碼很簡單,就是自訂了View,在View裡面都重寫了interceptTouchEvnet (),和onTouchEvent(),然後測試其返回值,對監聽的影響,關鍵是自己動手,逐個測試,並預測結果,等你能預測結果的時候,也就懂了,需要修改的地方就是interceptTouchEvnet 和onTouchEvent的返回值,他們決定了事件監聽的流程,下面我畫了一張圖,如有不足之處歡迎指正,謝謝!
下面是我的正解:
基本的規則是: *1.down事件首先會傳遞到onInterceptTouchEvent()方法 * * 2.如果該ViewGroup的onInterceptTouchEvent()在接收到down事件處理完成之後return false(不攔截), * 那麼後續的move, up等事件將繼續會先傳遞給該ViewGroup,之後才和down事件一樣傳遞給最終的目標view的onTouchEvent()處理。 * * 3.如果該ViewGroup的onInterceptTouchEvent()在接收到down事件處理完成之後return true(攔截,那麼後面的move,up事件不需要在看因為已經攔截了, 我們直接拿去處理onTouchEvent()就可以了),那麼後續的move, up等事件將不再傳遞給onInterceptTouchEvent(), 而是和down事件一樣傳遞給該ViewGroup的onTouchEvent()處理,注意,目標view將接收不到任何事件。 下面例子示範: * 1:LayoutView1(31375): onInterceptTouchEvent action:ACTION_DOWN * 2:LayoutView2(31375): onInterceptTouchEvent action:ACTION_DOWN * 3:LayoutView2(31375): onTouchEvent action:ACTION_DOWN * 4:LayoutView1(31375): onInterceptTouchEvent action:ACTION_MOVE * 5:LayoutView2(31375): onTouchEvent action:ACTION_MOVE * 6:LayoutView1(31375): onInterceptTouchEvent action:ACTION_MOVE * 7:LayoutView2(31375): onTouchEvent action:ACTION_MOVE * 8:LayoutView1(31375): onInterceptTouchEvent action:ACTION_UP * 9:LayoutView2(31375): onTouchEvent action:ACTION_UP * 該設定為: * onInterceptTouchEvent:LayoutView1為false,LayoutView2為true * onTouchEvent:LayoutView2為true * 故而事件在LayoutView2(onInterceptTouchEvent:返回true)時被攔截並處理,根據上面說法就是LayoutView2後續的MOVE,UP操作都不在經過onInterceptTouchEvent,直接 * 交給onTouchEvent處理,結果也的確如此。(見:LayoutView2的3,5,7,9,第一次是onInterceptTouchEvent處理如1,以後交給onTouchEvent) * 而LayoutView1都還是要經過onInterceptTouchEvent(見LayoutView1的4,6,8) * * 4.如果最終需要處理事件的view的onTouchEvent()返回了false(沒能處理這個事件,不能丟在傳回來讓父繼續), * 那麼該事件將被傳遞至其上一層次的view的onTouchEvent()處理。 * ************************************************************************** * 感覺像是一個圈,然後一直在找一個能處理這個訊息的人,如果找到了就結束,沒找到就迴圈,直到回到發出訊息的那個人 * 注(對下面):沒有標註的DOWN表示攔截事件onInterceptTouchEvent,標註了onTouchEvent就是處理事件 * a.如果都沒處理(onInterceptTouchEvent返回false):A(DOWN)-->B(DOWN)-->C(onTouchEvent DOWN)-->B(onTouchEvent DOWN)-->A(onTouchEvent DOWN),沒有執行UP事件,注意有MOVE的話,在DOWN和UP之間,下面的都一樣。 *b. B處理(B的onInterceptTouchEvent返回true):A(DOWN)-->B(DOWN)-->B(onTouchEvent)-->A(onTouchEvent UP)-->B(onTouchEvent UP)-->(over) *形象說明:如果父親不攔截訊息就傳給兒子,如果兒子要這個訊息就處理(DOWN),結束,然後有父親1--父親2--兒子以此釋放訊息(UP)。 然是如果兒子對這個訊息置之不理,那這個訊息又傳回父親,由父親來處理即。
下面給出了5中情況(不攔截表示onInterceptTouchEvent返回false): * 11** 父親1(LayoutView1不攔截false)---父親2(LayoutView2不攔截false)--兒子(MyTextView,onTouchEvent return true)--結束 * 22** 父親1(LayoutView1不攔截false)---父親2(LayoutView2不攔截false)--兒子(MyTextView,onTouchEvent return false)--回傳給父親2(onTouchEvent return true)--結束 * 33** 父親1(LayoutView1不攔截false)---父親2(LayoutView2不攔截false)--兒子(MyTextView,onTouchEvent return false)--回傳給父親2(onTouchEvent return false)--父親1(onTouchEvent return true)--結束(如果都沒處理不在執行UP ACTION) * 44** 父親1(LayoutView1攔截true)--父親1(onTouchEvent return true)--結束 (DOWN--DOWN(onTouchEvent)--UP(onTouchEvent)) * 55** 父親1(LayoutView1攔截false)--父親2(LayoutView2攔截true)--父親2(onTouchEvent return false)--父親1(onTouchEvent return true)--結束 (DOWN1--DOWN2--DOWN(2 onTouchEvent)--DOWN(1 onTouchEvent)--UP(1 onTouchEvent))(1:父親2,2:父親2) * * *************************************************************************** * 5.如果最終需要處理事件的view 的onTouchEvent()返回了true,那麼後續事件將可以繼續傳遞給該view的onTouchEvent()處理。 */
下面給出一張處理的流程圖:
下附原始碼:
- TouchEventTest.rar (94.3 KB)
- 下載次數: 40