日本综合一区二区|亚洲中文天堂综合|日韩欧美自拍一区|男女精品天堂一区|欧美自拍第6页亚洲成人精品一区|亚洲黄色天堂一区二区成人|超碰91偷拍第一页|日韩av夜夜嗨中文字幕|久久蜜综合视频官网|精美人妻一区二区三区

RELATEED CONSULTING
相關咨詢
選擇下列產(chǎn)品馬上在線溝通
服務時間:8:30-17:00
你可能遇到了下面的問題
關閉右側工具欄

新聞中心

這里有您想知道的互聯(lián)網(wǎng)營銷解決方案
編寫Android觸摸屏手勢識別程序

我們先來明確一些概念,首先,Android的事件處理機制是基于Listener(監(jiān)聽器)來實現(xiàn)的,比我們今天所說的觸摸屏相關的事件,就是通 過onTouchListener。其次,所有View的子類都可以通過setOnTouchListener()、 setOnKeyListener()等方法來添加對某一類事件的監(jiān)聽器。第三,Listener一般會以Interface(接口)的方式來提供,其中 包含一個或多個abstract(抽象)方法,我們需要實現(xiàn)這些方法來完成onTouch()、onKey()等等的操作。這樣,當我們給某個view設 置了事件Listener,并實現(xiàn)了其中的抽象方法以后,程序便可以在特定的事件被dispatch到該view的時候,通過callbakc函數(shù)給予適 當?shù)捻憫?/p>

站在用戶的角度思考問題,與客戶深入溝通,找到東勝網(wǎng)站設計與東勝網(wǎng)站推廣的解決方案,憑借多年的經(jīng)驗,讓設計與互聯(lián)網(wǎng)技術結合,創(chuàng)造個性化、用戶體驗好的作品,建站類型包括:成都網(wǎng)站設計、成都做網(wǎng)站、企業(yè)官網(wǎng)、英文網(wǎng)站、手機端網(wǎng)站、網(wǎng)站推廣、國際域名空間、網(wǎng)頁空間、企業(yè)郵箱。業(yè)務覆蓋東勝地區(qū)。

看一個簡單的例子,就用最簡單的TextView來說明(事實上和ADT中生成的skeleton沒有什么區(qū)別)。

 
 
 
  1. public class GestureTest extends Activity implements OnTouchListener{ 
  2.  
  3.     @Override 
  4.     protected void onCreate(Bundle savedInstanceState) { 
  5.         super.onCreate(savedInstanceState); 
  6.         setContentView(R.layout.main); 
  7.   
  8.         // init TextView 
  9.         TextView tv = (TextView) findViewById(R.id.page); 
  10.         // set OnTouchListener on TextView 
  11.         tv.setOnTouchListener(this); 
  12.         // show some text 
  13.         tv.setText(R.string.text); 
  14.     } 
  15.   
  16.     @Override 
  17.     public boolean onTouch(View v, MotionEvent event) { 
  18.         Toast.makeText(this, "onTouch", Toast.LENGTH_SHORT).show(); 
  19.         return false; 
  20.     } 

我們給TextView的實例tv設定了一個onTouchListener,因為GestureTest類實現(xiàn)了OnTouchListener 接口,所以簡單的給一個this作為參數(shù)即可。onTouch方法則是實現(xiàn)了OnTouchListener中的抽象方法,我們只要在這里添加邏輯代碼即 可在用戶觸摸屏幕時做出響應,就像我們這里所做的——打出一個提示信息。

這里,我們可以通過MotionEvent的getAction()方法來獲取Touch事件的類型,包括 ACTION_DOWN, ACTION_MOVE, ACTION_UP, 和ACTION_CANCEL。ACTION_DOWN是指按下觸摸屏,ACTION_MOVE是指按下觸摸屏后移動受力點,ACTION_UP則是指松 開觸摸屏,ACTION_CANCEL不會由用戶直接觸發(fā)(所以不在今天的討論范圍,請參考 ViewGroup.onInterceptTouchEvent(MotionEvent))。借助對于用戶不同操作的判斷,結合getRawX()、 getRawY()、getX()和getY()等方法來獲取坐標后,我們可以實現(xiàn)諸如拖動某一個按鈕,拖動滾動條等功能。待機可以看看 MotionEvent類的文檔,另外也可以看考TouchPaint例子。

回到今天所要說的重點,當我們捕捉到Touch操作的時候,如何識別出用戶的Gesture?這里我們需要GestureDetector.OnGestureListener接口的幫助,于是我們的GestureTest類就變成了這個樣子。

 
 
 
  1. public class GestureTest extends Activity implements OnTouchListener, 
  2.        OnGestureListener { 
  3. ... 

隨后,在onTouch()方法中,我們調(diào)用GestureDetector的onTouchEvent()方法,將捕捉到的MotionEvent交給 GestureDetector 來分析是否有合適的callback函數(shù)來處理用戶的手勢。

 
 
 
  1. @Override 
  2.    public boolean onTouch(View v, MotionEvent event) { 
  3.        // OnGestureListener will analyzes the given motion event 
  4.        return mGestureDetector.onTouchEvent(event); 
  5.     } 

接下來,我們實現(xiàn)了以下6個抽象方法,其中最有用的當然是onFling()、onScroll()和onLongPress()了。我已經(jīng)把每一個方法代表的手勢的意思寫在了注釋里,大家看一下就明白了。

 
 
 
  1. // 用戶輕觸觸摸屏,由1個MotionEvent ACTION_DOWN觸發(fā) 
  2.     @Override 
  3.     public boolean onDown(MotionEvent e) { 
  4.        // TODO Auto-generated method stub 
  5.        Toast.makeText(this, "onDown", Toast.LENGTH_SHORT).show(); 
  6.        return false; 
  7.     } 
  8.   
  9.    // 用戶輕觸觸摸屏,尚未松開或拖動,由一個1個MotionEvent ACTION_DOWN觸發(fā),注意和onDown()的區(qū)別,強調(diào)的是沒有松開或者拖動的狀態(tài) 
  10.    @Override 
  11.     public void onShowPress(MotionEvent e) { 
  12.         // TODO Auto-generated method stub 
  13.    } 
  14.  
  15.     // 用戶(輕觸觸摸屏后)松開,由一個1個MotionEvent ACTION_UP觸發(fā) 
  16.    @Override 
  17.     public boolean onSingleTapUp(MotionEvent e) { 
  18.         // TODO Auto-generated method stub 
  19.       return false; 
  20.     } 
  21.   
  22.     // 用戶按下觸摸屏、快速移動后松開,由1個MotionEvent ACTION_DOWN, 多個ACTION_MOVE, 1個ACTION_UP觸發(fā) 
  23.     @Override 
  24.     public boolean onFling(MotionEvent e1, MotionEvent e2, float velocityX, 
  25.            float velocityY) { 
  26.         // TODO Auto-generated method stub 
  27.         return false; 
  28.     } 
  29.   
  30.     // 用戶長按觸摸屏,由多個MotionEvent ACTION_DOWN觸發(fā) 
  31.     @Override 
  32.     public void onLongPress(MotionEvent e) { 
  33.        // TODO Auto-generated method stub 
  34.   
  35.     } 
  36.  
  37.     // 用戶按下觸摸屏,并拖動,由1個MotionEvent ACTION_DOWN, 多個ACTION_MOVE觸發(fā) 
  38.     @Override 
  39.     public boolean onScroll(MotionEvent e1, MotionEvent e2, float distanceX, 
  40.             float distanceY) { 
  41.         // TODO Auto-generated method stub 
  42.         return false; 
  43.     } 

我們來試著做一個onFling()事件的處理吧,onFling()方法中每一個參數(shù)的意義我寫在注釋中了,需要注意的是Fling事件的處理代 碼中,除了第一個觸發(fā)Fling的ACTION_DOWN和最后一個ACTION_MOVE中包含的坐標等信息外,我們還可以根據(jù)用戶在X軸或者Y軸上的 移動速度作為條件。比如下面的代碼中我們就在用戶移動超過100個像素,且X軸上每秒的移動速度大于200像素時才進行處理。

 
 
 
  1. @Override 
  2. public boolean onFling(MotionEvent e1, MotionEvent e2, float velocityX, 
  3.        float velocityY) { 
  4.    // 參數(shù)解釋: 
  5.     // e1:第1個ACTION_DOWN MotionEvent 
  6.    // e2:最后一個ACTION_MOVE MotionEvent 
  7.     // velocityX:X軸上的移動速度,像素/秒 
  8.    // velocityY:Y軸上的移動速度,像素/秒 
  9.   
  10.     // 觸發(fā)條件 : 
  11.    // X軸的坐標位移大于FLING_MIN_DISTANCE,且移動速度大于FLING_MIN_VELOCITY個像素/秒 
  12.  
  13.     if (e1.getX() - e2.getX() > FLING_MIN_DISTANCE 
  14.            && Math.abs(velocityX) > FLING_MIN_VELOCITY) { 
  15.        // Fling left 
  16.        Toast.makeText(this, "Fling Left", Toast.LENGTH_SHORT).show(); 
  17.     } else if (e2.getX() - e1.getX() > FLING_MIN_DISTANCE 
  18.            && Math.abs(velocityX) > FLING_MIN_VELOCITY) { 
  19.       // Fling right 
  20.        Toast.makeText(this, "Fling Right", Toast.LENGTH_SHORT).show(); 
  21.     } 
  22.   
  23.    return false; 

問題是,這個時候如果我們嘗試去運行程序,你會發(fā)現(xiàn)我們根本得不到想要的結果,跟蹤代碼的執(zhí)行的會發(fā)現(xiàn)onFling()事件一直就沒有被捕捉到。這正是一開始困擾我的問題,這到底是為什么呢?

我在討論組的Gesture detection這個帖子里找到了答案,即我們需要在onCreate中tv.setOnTouchListener(this);之后添加如下一句代碼。

 
 
 
  1. tv.setLongClickable(true); 

只有這樣,view才能夠處理不同于Tap(輕觸)的hold(即ACTION_MOVE,或者多個ACTION_DOWN),我們同樣可以通過layout定義中的android:longClickable來做到這一點。

這次遇到的這個問題和上次MapView中setOnKeyListener遇到的問題挺類似,其實都是對SDK的了解不夠全面,遇到了一次記住了就好。不過話說回來,Google在文檔方面確實需要加強了,起碼可以在OnGestureListener中說明需要滿足那些條件才可以保證手勢被正確識別。


名稱欄目:編寫Android觸摸屏手勢識別程序
分享網(wǎng)址:http://www.dlmjj.cn/article/cdipesi.html