Android viewpager滑動卡頓問題

android viewpager滑動卡頓

當(dāng)SwipeRefreshLayout中放置了ViewPager控件,兩者的滑動會相互沖突.具體表現(xiàn)為ViewPager的左右滑動不順暢,容易被SwipeRefreshLayout攔截(即出現(xiàn)刷新的View). 問題原因:Vie

問題說明:當(dāng)SwipeRefreshLayout中放置了ViewPager控件,兩者的滑動會相互沖突.具體表現(xiàn)為ViewPager的左右滑動不順暢,容易被SwipeRefreshLayout攔截(即出現(xiàn)刷新的View).

問題原因:ViewPager本身是處理了滾動事件的沖突,它在橫向滑動時會調(diào)用requestDisallowInterceptTouchEvent()方法使父控件不攔截當(dāng)前的Touch事件序列.但是SwipeRefreshLayout的requestDisallowInterceptTouchEvent()方法什么也沒有做,所以仍然會攔截當(dāng)前的Touch事件序列.

問題分析:為什么SwipeRefreshLayout的requestDisallowInterceptTouchEvent()方法什么都不做?

首先SwipeRefreshLayout繼承自ViewGroup.

在requestDisallowInterceptTouchEvent()方法什么都不做的情況下,用戶可以從底部下拉刷新一次拉出LoadingView. 如果方法調(diào)用ViewGroup的requestDisallowInterceptTouchEvent()方法, 可以解決ViewPager的兼容問題,但是用戶在界面底部下拉至頭部后,無法繼續(xù)下拉,需要手指放開一次才能拉出LoadingView. 目標(biāo)分析: 那么為了更加順滑地滾動,想要的效果當(dāng)然是一次性拉出LoadingView.既然ViewPager在左右滑動時才會調(diào)用requestDisallowInterceptTouchEvent()方法,那么SwipeRefreshLayout只應(yīng)該在上下滑動時才攔截Touch事件.

具體邏輯如下:

記錄是否調(diào)用了requestDisallowInterceptTouchEvent()方法,并且設(shè)置為true. 在SwipeRefreshLayout中判斷是否是上下滑動. 如果同時滿足1,2,則調(diào)用super.requestDisallowInterceptTouchEvent(true). 否則調(diào)用super.requestDisallowInterceptTouchEvent(false). 注意:因為ViewGroup的requestDisallowInterceptTouchEvent方法設(shè)置true后,Touch事件在dispatchTouchEvent()方法中就會被攔截,所以需要在dispatchTouchEvent()方法中判斷是否為上下滑動.

實現(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; 
//是否請求攔截 
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滑動沖突問題
 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) { 
//橫向滾動不需要攔截
 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);
 }

大部分來自于:http://phpstudy.php.cn/b.php/103489.html

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時請結(jié)合常識與多方信息審慎甄別。
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點,簡書系信息發(fā)布平臺,僅提供信息存儲服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

  • Android 事件分發(fā)和滑動沖突都是開發(fā)中經(jīng)常遇到的難點問題,遇到問題時可能會通過 Google 或者 Stac...
    任教主來也閱讀 2,806評論 0 24
  • Android 自定義View的各種姿勢1 Activity的顯示之ViewRootImpl詳解 Activity...
    passiontim閱讀 179,319評論 25 708
  • 泛黃的樹葉 泛黃的路燈 瞇起的眼睛 看不清 模糊的未來 在蕭瑟的風(fēng)中 終于 遙不可及
    Y長安閱讀 319評論 0 0
  • 第12天·21天告別拖延
    Bigcherry閱讀 392評論 0 0
  • 故鄉(xiāng)遙,何日去。 家住吳門,久作長安旅。 五月漁郎相憶否。 小楫輕舟,夢入芙蓉浦。
    喜亭_bf8f閱讀 206評論 9 14

友情鏈接更多精彩內(nèi)容