Android教程網
  1. 首頁
  2. Android 技術
  3. Android 手機
  4. Android 系統教程
  5. Android 游戲
 Android教程網 >> Android技術 >> Android開發 >> 關於android開發 >> Android Touch事件傳遞機制 二:單純的(偽生命周期),androidtouch

Android Touch事件傳遞機制 二:單純的(偽生命周期),androidtouch

編輯:關於android開發

Android Touch事件傳遞機制 二:單純的(偽生命周期),androidtouch


轉載於:http://blog.csdn.net/yuanzeyao/article/details/38025165

 

在前一篇文章中,我主要講解了Android源碼中的Touch事件的傳遞過程,現在我想使用一個demo以及一個實例來學習一下Andorid中的Touch事件處理過程。

在Android系統中,和Touch事件分發和處理緊密相關的三個函數如下:
(1) public boolean dispatchTouchEvent(MotionEvent ev)
(2) public boolean onInterceptTouchEvent(MotionEvent ev)
(3) public boolean onTouchEvent(MotionEvent event)


這三個方法我在前一篇文章中都對他們的源碼進行了分析:方法1主要是對Touch事件進行分發,方法2主要是對Touch事件進行攔截,方法3是對Touch事件進行處理

這三個方法主要存在於ViewGroup,View,Activity中,具體情況如下圖:

  ViewGroup View  Activity dispatchTouchEvent 有 有 有 onInterceptTouchEvent 有 無 無 onTouchEvent 有 有 有

下面我們就使用一個demo來看看這些方法的執行流程:
自定義一個類:MyLayoutFirst.java

 

[java] view plain copy    print?
  1. public class MyLayoutFirst extends LinearLayout  
  2. {  
  3.   private static final String TAG = "MyLayoutFirst";  
  4.   public MyLayoutFirst(Context context, AttributeSet attrs)  
  5.   {  
  6.     super(context, attrs);  
  7.   }  
  8.     
  9.   @Override  
  10.   public boolean onInterceptTouchEvent(MotionEvent ev)  
  11.   {  
  12.     Log.w("yzy", "MyLayoutFirst->onInterceptTouchEvent->"+MyUtils.getActionName(ev));  
  13.     return super.onInterceptTouchEvent(ev);  
  14.   }  
  15.     
  16.   @Override  
  17.   public boolean onTouchEvent(MotionEvent event)  
  18.   {  
  19.     Log.e("yzy", "MyLayoutFirst->onTouchEvent->"+MyUtils.getActionName(event));  
  20.     return super.onTouchEvent(event);  
  21.   }  
  22.     
  23.   @Override  
  24.   public boolean dispatchTouchEvent(MotionEvent ev)  
  25.   {  
  26.     Log.i("yzy", "MyLayoutFirst->dispatchTouchEvent->"+MyUtils.getActionName(ev));  
  27.     return super.dispatchTouchEvent(ev);  
  28.   }  
  29.     
  30. }  


自定義一個類;MyLayoutSecond.java

 

 

[java] view plain copy    print?
  1. public class MyLayoutSecond extends LinearLayout  
  2. {  
  3.   private static final String TAG = "MyLayoutSecond";  
  4.   public MyLayoutSecond(Context context, AttributeSet attrs)  
  5.   {  
  6.     super(context, attrs);  
  7.   }  
  8.     
  9.   @Override  
  10.   public boolean onTouchEvent(MotionEvent event)  
  11.   {  
  12.     Log.e("yzy", "MyLayoutSecond->MyLayoutSecond->"+MyUtils.getActionName(event));  
  13.     return super.onTouchEvent(event);  
  14.   }  
  15.     
  16.   @Override  
  17.   public boolean onInterceptTouchEvent(MotionEvent ev)  
  18.   {  
  19.     Log.w("yzy", "MyLayoutSecond->onInterceptTouchEvent->"+MyUtils.getActionName(ev));  
  20.     return super.onInterceptTouchEvent(ev);  
  21.   }  
  22.     
  23.   @Override  
  24.   public boolean dispatchTouchEvent(MotionEvent ev)  
  25.   {  
  26.     Log.i("yzy", "MyLayoutSecond->dispatchTouchEvent->"+MyUtils.getActionName(ev));  
  27.     return super.dispatchTouchEvent(ev);  
  28.   }  
  29.   
  30.    
  31. }  


加入到main_layout.xml中

 

 

[html] view plain copy    print?
  1. <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"  
  2.     xmlns:tools="http://schemas.android.com/tools"  
  3.     android:layout_width="match_parent"  
  4.     android:layout_height="match_parent"  
  5.    >  
  6.   
  7.     <com.event.demo.MyLayoutFirst   
  8.         android:id="@+id/layout_first"  
  9.         android:layout_width="match_parent"  
  10.         android:layout_height="match_parent"  
  11.         android:background="#FF0000"  
  12.         >  
  13.         <com.event.demo.MyLayoutSecond  
  14.             android:id="@+id/layout_second"  
  15.             android:layout_width="320dip"  
  16.             android:layout_height="120dip"  
  17.             android:layout_gravity="center"  
  18.             android:background="#0000FF"  
  19.         >  
  20.           
  21.     </com.event.demo.MyLayoutSecond>  
  22.     </com.event.demo.MyLayoutFirst>  
  23.   
  24. </RelativeLayout>  


MainActivity中加入onTouchEvent方法

 

 

[java] view plain copy    print?
  1. public class MainActivity extends Activity  
  2. {  
  3.   
  4.   @Override  
  5.   protected void onCreate(Bundle savedInstanceState)  
  6.   {  
  7.     super.onCreate(savedInstanceState);  
  8.     setContentView(R.layout.activity_main);  
  9.   }  
  10.   
  11.   @Override  
  12.   public boolean onCreateOptionsMenu(Menu menu)  
  13.   {  
  14.     // Inflate the menu; this adds items to the action bar if it is present.  
  15.     getMenuInflater().inflate(R.menu.main, menu);  
  16.     return true;  
  17.   }  
  18.     
  19.   @Override  
  20.   public boolean dispatchTouchEvent(MotionEvent ev)  
  21.   {  
  22.     Log.i("yzy", "MainActivity->dispatchTouchEvent->"+MyUtils.getActionName(ev));  
  23.     return super.dispatchTouchEvent(ev);  
  24.   }  
  25.     
  26.   @Override  
  27.   public boolean onTouchEvent(MotionEvent event)  
  28.   {  
  29.     Log.e("yzy", "MainActivity->onTouchEvent->"+MyUtils.getActionName(event));  
  30.     return super.onTouchEvent(event);  
  31.   }  
  32. }  


最後就一個工具類,用來將事件id轉換為字符串。

 

 

[java] view plain copy    print?
  1. public class MyUtils  
  2. {  
  3.   private static final String TAG = "MyUtils";  
  4.   public static String getActionName(MotionEvent event)  
  5.   {  
  6.     String name="";  
  7.     switch(event.getAction())  
  8.     {  
  9.       case MotionEvent.ACTION_DOWN:  
  10.         name="ACTION_DOWN";  
  11.         break;  
  12.       case MotionEvent.ACTION_MOVE:  
  13.         name="ACTION_MOVE";  
  14.         break;  
  15.       case MotionEvent.ACTION_UP:  
  16.         name="ACTION_UP";  
  17.         break;  
  18.     }  
  19.     return name;  
  20.   }  
  21. }  


運行效果如圖:

 

 

第一中情況:

 

MainActivity

MyLayoutFirst

MyLayoutSecond

dispatchTouchEvent

super.dispatchTouchEvent

super.dispatchTouchEvent

super.dispatchTouchEvent

onInterceptTouchEvent

--

super.onInterceptTouchEvent(ev)

super.onInterceptTouchEvent(ev)

onTouchEvent

super.onTouchEvent

super.onTouchEven

super.onTouchEvent

運行結果:

其中藍色部分是MyLayoutSecond.Java ,紅色部分是MyLayoutFirst.java
現在我點擊一下藍色部分:運行結果如圖:

從圖中可以看出,事件最先被Activity捕獲,然後分發給 MyLayoutFirst,MyLayoutFirst首先調用自身的onInterceptTouchEvent判斷是否將該事件攔截,由於默認返回是false,所以沒有攔截,從而事件分發給了MyLayoutSecond,MyLayoutSecond同樣通過dispatchTouchEvent分發出去,分發出去之前同樣檢查是否被攔截,默認都是沒有被攔截的,但是由於MyLayoutSecond是沒有子視圖的,所有最終事件有自己處理,調用自身的onTouchEvent方法,由於該方法默認返回的是false,所以認為此事件是沒有被消費掉的,繼續傳遞到了MyLayoutFirst中,同樣也沒有消費這個事件,最終傳遞到了Mainactivity,繼續往後看發現後面的ACTION_MOVE和ACTION_UP並沒有傳入MyLayoutFirst和MyLayoutSecond,這是因為一旦某一個事件沒有被處理,後面的事件是不會被分發的。所以ACTION_MOVE和ACTION_UP直接被MainActivity處理掉了。

下面再看第二種情況:

 

 

MainActivity

MyLayoutFirst

MyLayoutSecond

dispatchTouchEvent

super.dispatchTouchEvent

super.dispatchTouchEvent

super.dispatchTouchEvent

onInterceptTouchEvent

--

true

super.onInterceptTouchEvent(ev)

onTouchEvent

super.onTouchEvent

super.onTouchEvent

super.onTouchEvent

運行結果如下:

 

從圖中可以看出,事件傳遞到了MyLayoutFirst後沒有分發到MyLayoutSecond,直接調用自身的onTouchEvent,由於返回的是false,導致事件沒有消費,最終傳遞給了MainActivity,
而且後續事件也沒有傳遞到MyLayoutFirst和MyLayoutSecond,直接被MainActivity處理

 

第三種情況:

 

MainActivity

MyLayoutFirst

MyLayoutSecond

dispatchTouchEvent

super.dispatchTouchEvent

super.dispatchTouchEvent

super.dispatchTouchEvent

onInterceptTouchEvent

--

true

super.onInterceptTouchEvent(ev)

onTouchEvent

super.onTouchEvent

true

super.onTouchEvent

運行結果:

 

和情況二不同的是MyLayoutFirst的onTouchEvent返回了true,也就是說MyLayoutFirst消費了此事件,所以ACTION_DOWN也沒有再傳給MainActivity,並且ACTION_MOVE和ACTION_UP
均傳給了MyLayoutFirst

 

第四中情況:

 

MainActivity

MyLayoutFirst

MyLayoutSecond

dispatchTouchEvent

super.dispatchTouchEvent

super.dispatchTouchEvent

super.dispatchTouchEvent

onInterceptTouchEvent

--

super.onInterceptTouchEvent(ev)

super.onInterceptTouchEvent(ev)

onTouchEvent

super.onTouchEvent

super.onTouchEven

true

運行結果:

 

發現所有的事件都是傳遞到了MyLayoutSecond後被消費了

 

其實還有很多其他組合方式,大家如果又興趣可以自己嘗試改變每個函數的返回值,查看打印結果,這裡我就不一一列舉了。。。。。


最後我會提供一個小demo演示如何解決滑動沖突,背景如下:
一個ViewPager裡面包含兩個Framgent,有一個Fragment裡面有一個HorizontalListView ,如何滑動沖突?
我就貼出關鍵代碼吧

 

 

[java] view plain copy    print?
  1. horizontal=(HorizontalListView)view.findViewById(R.id.hscroll);  
  2.     horizontal.setOnTouchListener(new OnTouchListener()  
  3.     {  
  4.         
  5.       @Override  
  6.       public boolean onTouch(View arg0, MotionEvent event)  
  7.       {  
  8.         if(event.getAction()==MotionEvent.ACTION_DOWN)  
  9.         {  
  10.           parent.requestDisallowInterceptTouchEvent(true);  
  11.         }else if(event.getAction()==MotionEvent.ACTION_UP)  
  12.         {  
  13.           parent.requestDisallowInterceptTouchEvent(false);  
  14.         }  
  15.         return false;  
  16.       }  
  17.     });  


加入這段代碼就可以避免滑動沖突了,至於為什麼大家可以參考我的前以前文章《Android Touch 事件傳遞機制詳解 上》 這兩個demo的例子我均會上傳下載的

  1. 上一頁:
  2. 下一頁:
熱門文章
閱讀排行版
Copyright © Android教程網 All Rights Reserved