Android使用手勢監(jiān)聽器GestureDetector遇到的不響應(yīng)問題
做了一個項(xiàng)目,首頁是使用ResideMenu實(shí)現(xiàn),通過菜單欄里的菜單項(xiàng)創(chuàng)建的Fragment;所以一個Activtiy里就包含多個Fragment,想通過手勢也能側(cè)滑,就不用點(diǎn)擊菜單按鈕打開menu了;
方法如下:
在activity的oncreate()中初始化手勢監(jiān)聽器
mGestureDetector = new GestureDetector(getApplicationContext(), new MyGestureListener(getApplicationContext()));
然后寫一個類繼承手勢監(jiān)聽器,當(dāng)然你也可以采用匿名的方法:
/******************************手勢監(jiān)聽器**************************************/ class MyGestureListener extends GestureDetector.SimpleOnGestureListener { Context mContext; MyGestureListener(Context context) { mContext = context; } @Override public boolean onDown(MotionEvent e) { LogUtils.d(TAG,"onDown---DOWN " + e.getAction()); return false; } @Override public void onShowPress(MotionEvent e) { LogUtils.d(TAG, "onShowPress---DOWN " + e.getAction()); } @Override public boolean onSingleTapUp(MotionEvent e) { LogUtils.d(TAG, "onSingleTapUp---DOWN " + e.getAction()); return false; } @Override public boolean onScroll(MotionEvent e1, MotionEvent e2,float distanceX, float distanceY) { resideMenu.openMenu(ResideMenu.DIRECTION_LEFT); LogUtils.d(TAG, "onScroll---DOWN " + e2.getAction()); return false; } @Override public void onLongPress(MotionEvent e) { LogUtils.d(TAG, "onLongPress---DOWN " + e.getAction()); } @Override public boolean onFling(MotionEvent e1, MotionEvent e2, float velocityX,float velocityY) { LogUtils.d(TAG, "onFling---DOWN " + e2.getAction()); return false; } @Override public boolean onDoubleTap(MotionEvent e) { LogUtils.d(TAG, "onDoubleTap---DOWN " + e.getAction()); return false; } @Override public boolean onDoubleTapEvent(MotionEvent e) { LogUtils.d(TAG, "onDoubleTapEvent---DOWN " + e.getAction()); return false; } @Override public boolean onSingleTapConfirmed(MotionEvent e) { LogUtils.d(TAG, "DOWN " + e.getAction()); return false; } } /********************************************************************/
在onscroll方法中進(jìn)行你要的滑動監(jiān)聽
注意:這個需要在activity中重寫倆個方法:
/** * 重寫觸摸事件 * @param event * @return */ @Override public boolean onTouchEvent(MotionEvent event) { return mGestureDetector.onTouchEvent(event); } /** * 如果觸摸事件下有控件點(diǎn)擊事件,則重寫下面方法 * @param ev * @return */ @Override public boolean dispatchTouchEvent(MotionEvent ev) { if(mGestureDetector.onTouchEvent(ev)){ return mGestureDetector.onTouchEvent(ev); } return super.dispatchTouchEvent(ev); }
在單個activity中只需要重寫第一個方法就行
補(bǔ)充知識:Android利用GestureDetector處理不太常用的一些點(diǎn)擊事件
關(guān)于GestureDetector ,在網(wǎng)上有很多資料是描述如下常見情況下的回調(diào):
點(diǎn)擊一下非??斓模ú换瑒樱㏕ouchup:
onDown->onSingleTapUp->onSingleTapConfirmed
點(diǎn)擊一下稍微慢點(diǎn)的(不滑動)Touchup:
onDown->onShowPress->onSingleTapUp->onSingleTapConfirmed
長按:
onDown-->onShowPress-->onLongPress
兩次連續(xù)點(diǎn)擊(第二次點(diǎn)擊之后立即抬起):
(第一次點(diǎn)擊)onDown->onSingleTapUp->(第二次點(diǎn)擊)onDoubleTap->onDoubleTapEvent->onDown->onShowPress->onDoubleTapEvent
點(diǎn)擊之后滑動:
onDown->onShowPress->onScroll->......(->onFling)(視速度快慢)
但是這些并不能完美符合我們的需求,我們還會遇到以下需求:
雙擊之后拖動:
我在每個回調(diào)函數(shù)打上log,雙擊之后拖動的log如下:
(中間若干個都是onTouch: Move)
首先可以看到雙擊(onDoubleTapEvent)被回調(diào)之后的Move事件都被傳遞到了onDoubleTapEvent中。但是當(dāng)你第二次點(diǎn)擊時間達(dá)到一定之后,onLongPress會被回調(diào),而當(dāng)onLongPress被回調(diào)之后,MOVE動作就被GestureDetector無視了,直到UP動作出現(xiàn),顯然這不是我們想要的。
那么我們可以在onDoubleTapEvent中接收到Down動作時,利用setIsLongPressEnabled()使LongPress不會觸發(fā),然后在onDoubleTapEvent中接收到Up動作時再恢復(fù)即可。
@Override public boolean onDoubleTapEvent(MotionEvent e) { Log.d(TAG, "onDoubleTapEvent: "); switch (e.getAction()) { case MotionEvent.ACTION_DOWN: gestureDetector.setIsLongpressEnabled(false); //action break; case MotionEvent.ACTION_MOVE: //action break; case MotionEvent.ACTION_UP: //action gestureDetector.setIsLongpressEnabled(true); break; } return true; }
更改之后,再進(jìn)行測試,如下:
(中間若干個onTouch: Move,onDoubleTapEvent)
長按拖動:
在onLongPress被回調(diào)之后,GestureDetector不會對Move動作調(diào)用任何函數(shù),除非直到一個Up動作出現(xiàn),但用戶的習(xí)慣不可能是這樣。因此對于這個需求我們需要在onTouch中對Move動作進(jìn)行識別。
首先修改onLongPress函數(shù),在長按之后更新狀態(tài)為可拖拽,然后對onTouch中的Move動作我們自己調(diào)用onScroll(不一定要onScroll),并且在onScroll中完成動作,因此需要記錄上一次的MotionEvent:
@Override public void onLongPress(MotionEvent e) { Log.d(TAG, "onLongPress: "); lastMotionEvent = e; draggable = true; }
然后在onTouch函數(shù)中:
@Override public boolean onTouch(View v, MotionEvent event) { boolean result = gestureDetector.onTouchEvent(event); // 如果gestureDetector不消費(fèi)動作 if (!result) { switch (event.getAction()) { case MotionEvent.ACTION_DOWN: break; case MotionEvent.ACTION_MOVE: // 可拖拽狀態(tài)下調(diào)用onScroll,同時更新lastMotionEvent if (draggable) { onScroll(lastMotionEvent, event, lastMotionEvent.getX() - event.getX(), lastMotionEvent.getY() - event.getY()); lastMotionEvent = MotionEvent.obtain(event); } result = true; break; case MotionEvent.ACTION_UP: // 恢復(fù)為不可拖拽狀態(tài) if (draggable) { onScroll(lastMotionEvent, event, lastMotionEvent.getX() - event.getX(), lastMotionEvent.getY() - event.getY()); lastMotionEvent = null; draggable = false; } result = true; break; } } return result; }
處理點(diǎn)擊-滑動之后的ACTION_UP
滑動的回調(diào)是這樣的
onDown->onShowPress->onScroll->......(->onFling)(視速度快慢)
如果onFling沒有被回調(diào)的話,我們無法對onScroll之后的Up動作響應(yīng),因此對于這個動作,我們也要在onTouch中處理。
首先要明確:從點(diǎn)A滑動到點(diǎn)B,并且在點(diǎn)B松手的話,在沒有觸發(fā)onFling的情況下,會回調(diào)onScroll(eA, eB, distanceX, distanceY),然后GestureDetector不消費(fèi)點(diǎn)B的Up事件,此時我們在onTouch中處理這個Up事件。
代碼也很簡單,在長按拖動的基礎(chǔ)上增加一個else即可:
case MotionEvent.ACTION_UP: if (draggable) { onScroll(lastMotionEvent, event, lastMotionEvent.getX() - event.getX(), lastMotionEvent.getY() - event.getY()); lastMotionEvent = null; draggable = false; } else { afterScroll(event); } result = true; break;
具體需要處理何種點(diǎn)擊事件可根據(jù)實(shí)際修改,希望分享的內(nèi)容能給你一點(diǎn)idea。
以上這篇Android使用手勢監(jiān)聽器GestureDetector遇到的不響應(yīng)問題就是小編分享給大家的全部內(nèi)容了,希望能給大家一個參考,也希望大家多多支持腳本之家。
相關(guān)文章
Android自定義View設(shè)定到FrameLayout布局中實(shí)現(xiàn)多組件顯示的方法 分享
Android自定義View設(shè)定到FrameLayout布局中實(shí)現(xiàn)多組件顯示的方法 分享,需要的朋友可以參考一下2013-05-05Android sdcard實(shí)現(xiàn)圖片存儲 、聯(lián)網(wǎng)下載
這篇文章主要介紹了Android sdcard實(shí)現(xiàn)圖片存儲 、聯(lián)網(wǎng)下載功能,感興趣的小伙伴們可以參考一下2016-02-02Flutter質(zhì)感設(shè)計(jì)之列表項(xiàng)
這篇文章主要為大家詳細(xì)介紹了Flutter質(zhì)感設(shè)計(jì)之列表項(xiàng),具有一定的參考價值,感興趣的小伙伴們可以參考一下2018-08-08深入理解Android熱修復(fù)技術(shù)原理之代碼熱修復(fù)技術(shù)
在各種 Android 熱修復(fù)方案中,Andfix的即時生效令人印象深刻,它稍顯另類, 并不需要重新啟動,而是在加載補(bǔ)丁后直接對方法進(jìn)行替換就可以完成修復(fù),然而它的使用限制也遭遇到更多的質(zhì)疑2021-06-06Android編程自定義搜索框?qū)崿F(xiàn)方法【附demo源碼下載】
這篇文章主要介紹了Android編程自定義搜索框?qū)崿F(xiàn)方法,涉及Android界面布局、數(shù)據(jù)加載、事件響應(yīng)等相關(guān)操作技巧,并附帶完整demo源碼供讀者下載參考,需要的朋友可以參考下2017-12-12