Android中ViewPager帶來(lái)的滑動(dòng)卡頓問(wèn)題解決要點(diǎn)解析
問(wèn)題說(shuō)明:
當(dāng)SwipeRefreshLayout中放置了ViewPager控件,兩者的滑動(dòng)會(huì)相互沖突.具體表現(xiàn)為ViewPager的左右滑動(dòng)不順暢,容易被SwipeRefreshLayout攔截(即出現(xiàn)刷新的View).
問(wèn)題原因:
ViewPager本身是處理了滾動(dòng)事件的沖突,它在橫向滑動(dòng)時(shí)會(huì)調(diào)用requestDisallowInterceptTouchEvent()方法使父控件不攔截當(dāng)前的Touch事件序列.但是SwipeRefreshLayout的requestDisallowInterceptTouchEvent()方法什么也沒(méi)有做,所以仍然會(huì)攔截當(dāng)前的Touch事件序列.
問(wèn)題分析:
為什么SwipeRefreshLayout的requestDisallowInterceptTouchEvent()方法什么都不做?
首先SwipeRefreshLayout繼承自ViewGroup.
在requestDisallowInterceptTouchEvent()方法什么都不做的情況下,用戶(hù)可以從底部下拉刷新一次拉出LoadingView.
如果方法調(diào)用ViewGroup的requestDisallowInterceptTouchEvent()方法, 可以解決ViewPager的兼容問(wèn)題,但是用戶(hù)在界面底部下拉至頭部后,無(wú)法繼續(xù)下拉,需要手指放開(kāi)一次才能拉出LoadingView.
目標(biāo)分析:
那么為了更加順滑地滾動(dòng),想要的效果當(dāng)然是一次性拉出LoadingView.既然ViewPager在左右滑動(dòng)時(shí)才會(huì)調(diào)用requestDisallowInterceptTouchEvent()方法,那么SwipeRefreshLayout只應(yīng)該在上下滑動(dòng)時(shí)才攔截Touch事件.
具體邏輯如下:
記錄是否調(diào)用了requestDisallowInterceptTouchEvent()方法,并且設(shè)置為true.
在SwipeRefreshLayout中判斷是否是上下滑動(dòng).
如果同時(shí)滿(mǎn)足1,2,則調(diào)用super.requestDisallowInterceptTouchEvent(true).
否則調(diào)用super.requestDisallowInterceptTouchEvent(false).
注意:因?yàn)閂iewGroup的requestDisallowInterceptTouchEvent方法設(shè)置true后,Touch事件在dispatchTouchEvent()方法中就會(huì)被攔截,所以需要在dispatchTouchEvent()方法中判斷是否為上下滑動(dòng).
實(shí)現(xiàn)代碼(部分):
//非法按鍵 private static final int INVALID_POINTER = -1; //dispatch方法記錄第一次按下的x private float mInitialDisPatchDownX; //dispatch方法記錄第一次按下的y private float mInitialDisPatchDownY; //dispatch方法記錄的手指 private int mActiveDispatchPointerId = INVALID_POINTER; //是否請(qǐng)求攔截 private boolean hasRequestDisallowIntercept = false; @Override public void requestDisallowInterceptTouchEvent(boolean b) { hasRequestDisallowIntercept = b; // Nope. } @Override public boolean dispatchTouchEvent(MotionEvent ev) { switch (ev.getAction()) { case MotionEvent.ACTION_DOWN: mActiveDispatchPointerId = MotionEventCompat.getPointerId(ev, 0); final float initialDownX = getMotionEventX(ev, mActiveDispatchPointerId); if (initialDownX != INVALID_POINTER) { mInitialDisPatchDownX = initialDownX; } final float initialDownY = getMotionEventY(ev, mActiveDispatchPointerId); if (mInitialDisPatchDownY != INVALID_POINTER) { mInitialDisPatchDownY = initialDownY; } break; case MotionEvent.ACTION_MOVE: if (hasRequestDisallowIntercept) { //解決viewPager滑動(dòng)沖突問(wèn)題 final float x = getMotionEventX(ev, mActiveDispatchPointerId); final float y = getMotionEventY(ev, mActiveDispatchPointerId); if (mInitialDisPatchDownX != INVALID_POINTER && x != INVALID_POINTER && mInitialDisPatchDownY != INVALID_POINTER && y != INVALID_POINTER) { final float xDiff = Math.abs(x - mInitialDisPatchDownX); final float yDiff = Math.abs(y - mInitialDisPatchDownY); if (xDiff > mTouchSlop && xDiff * 0.7f > yDiff) { //橫向滾動(dòng)不需要攔截 super.requestDisallowInterceptTouchEvent(true); } else { super.requestDisallowInterceptTouchEvent(false); } } else { super.requestDisallowInterceptTouchEvent(false); } } break; case MotionEvent.ACTION_UP: case MotionEvent.ACTION_CANCEL: if (ev.getAction() == MotionEvent.ACTION_UP || ev.getAction() == MotionEvent.ACTION_CANCEL) { hasRequestDisallowIntercept = false; } break; } return super.dispatchTouchEvent(ev); } private float getMotionEventY(MotionEvent ev, int activePointerId) { final int index = MotionEventCompat.findPointerIndex(ev, activePointerId); if (index < 0) { return -1; } return MotionEventCompat.getY(ev, index); } private float getMotionEventX(MotionEvent ev, int activePointerId) { final int index = MotionEventCompat.findPointerIndex(ev, activePointerId); if (index < 0) { return -1; } return MotionEventCompat.getX(ev, index); }
- 使用ViewPager實(shí)現(xiàn)左右循環(huán)滑動(dòng)及滑動(dòng)跳轉(zhuǎn)
- Android App中ViewPager所帶來(lái)的滑動(dòng)沖突問(wèn)題解決方法
- 自定義RadioButton和ViewPager實(shí)現(xiàn)TabHost帶滑動(dòng)的頁(yè)卡效果
- Android App中使用ViewPager+Fragment實(shí)現(xiàn)滑動(dòng)切換效果
- Android利用ViewPager實(shí)現(xiàn)滑動(dòng)廣告板實(shí)例源碼
- Android開(kāi)發(fā)之使用ViewPager實(shí)現(xiàn)圖片左右滑動(dòng)切換效果
- Android ViewPager無(wú)限循環(huán)實(shí)現(xiàn)底部小圓點(diǎn)動(dòng)態(tài)滑動(dòng)
- Android實(shí)現(xiàn)橫向滑動(dòng)卡片效果
- Android仿探探卡片式滑動(dòng)效果實(shí)現(xiàn)
- ViewPager+RadioGroup實(shí)現(xiàn)左右滑動(dòng)卡片布局
相關(guān)文章
項(xiàng)目發(fā)布Debug和Release版的區(qū)別詳解
這篇文章主要為大家詳細(xì)介紹了項(xiàng)目發(fā)布Debug和Release版的區(qū)別,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2020-10-10Android使用AudioManager修改系統(tǒng)音量的方法
這篇文章主要介紹了Android使用AudioManager修改系統(tǒng)音量的方法,結(jié)合實(shí)例形式分析了AudioManager調(diào)節(jié)音量的常用方法及相關(guān)使用技巧,需要的朋友可以參考下2016-08-08一文深入探討Android Activity啟動(dòng)模式
在 Android 應(yīng)用開(kāi)發(fā)中,Activity 是用戶(hù)界面的核心組件,而 Activity 的啟動(dòng)模式則是決定應(yīng)用界面如何在任務(wù)棧中交互、管理以及呈現(xiàn)的關(guān)鍵因素,本文將深入探討 Android 中的 Activity 啟動(dòng)模式,詳細(xì)解釋每種模式的用途、適用場(chǎng)景2023-08-08Android開(kāi)發(fā)之TabActivity用法實(shí)例詳解
這篇文章主要介紹了Android開(kāi)發(fā)之TabActivity用法,結(jié)合實(shí)例形式較為詳細(xì)的分析了Android擴(kuò)展Activity實(shí)現(xiàn)標(biāo)簽頁(yè)效果的具體步驟與相關(guān)技巧,需要的朋友可以參考下2016-03-03Android開(kāi)發(fā)實(shí)現(xiàn)的IntentUtil跳轉(zhuǎn)多功能工具類(lèi)【包含視頻、音頻、圖片、攝像頭等操作功能】
這篇文章主要介紹了Android開(kāi)發(fā)實(shí)現(xiàn)的IntentUtil跳轉(zhuǎn)多功能工具類(lèi),該封裝類(lèi)還包含視頻、音頻、圖片、攝像頭等操作功能,需要的朋友可以參考下2017-11-11Kotlin中常見(jiàn)內(nèi)聯(lián)擴(kuò)展函數(shù)的使用方法教程
在Kotlin中,使用inline修飾符標(biāo)記內(nèi)聯(lián)函數(shù),既會(huì)影響到函數(shù)本身, 也影響到傳遞給它的Lambda表達(dá)式,這兩者都會(huì)被內(nèi)聯(lián)到調(diào)用處。下面這篇文章主要給大家介紹了關(guān)于Kotlin中常見(jiàn)內(nèi)聯(lián)擴(kuò)展函數(shù)的使用方法,需要的朋友可以參考下。2017-12-12Android開(kāi)發(fā)案例手冊(cè)Application跳出dialog
這篇文章主要為大家介紹了Android開(kāi)發(fā)案例手冊(cè)Application跳出dialog,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-06-06Android使用SQLite數(shù)據(jù)庫(kù)的簡(jiǎn)單實(shí)例
這篇文章主要介紹了Android使用SQLite數(shù)據(jù)庫(kù)的簡(jiǎn)單實(shí)例,有需要的朋友可以參考一下2013-12-12Android GSYVideoPlayer視頻播放器功能的實(shí)現(xiàn)
這篇文章主要介紹了Android GSYVideoPlayer視頻播放器功能的實(shí)現(xiàn),本文通過(guò)實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2021-03-03