簡介:
用內(nèi)部攔截法 和 外部攔截法 兩種方式處理兩個橫向的ViewPager嵌套的手勢沖突
最終實現(xiàn)效果:按住右側(cè)部分橫滑就是滑動大ViewPager。類似抖音的首頁
這個demo看懂你就明白所有的手勢處理的流程了
功能:
- ?Demo展示了
內(nèi)部攔截法和外部攔截法兩種方式 - ?海量的代碼注釋,且注明了為什么要調(diào)用這行代碼,如果不調(diào)用會怎么樣
- ?看懂這個demo,所有的手勢處理你都會了
作者說明:
你在網(wǎng)上看到別人的內(nèi)外部攔截法demo,他們會寫多一些無用代碼,你以為這句代碼發(fā)揮作用了,其實沒有。本demo是最簡潔的,一句多余代碼都沒
效果gif圖

部分代碼注釋參考
@Override
public boolean dispatchTouchEvent(MotionEvent ev) {
switch (ev.getAction()) {
case MotionEvent.ACTION_DOWN:
downX = ev.getRawX();
//down的地方是在最右邊
downXAlignParentRight = ev.getX() > getWidth() - 300;
//接到ACTION_DOWN就需要開始取消父容器的事件攔截。不然后面事件全是大容器的onTouchEvent的
//因為:第一幀的move,他先看flag,即孩子有沒有request。發(fā)現(xiàn)孩子沒有request(因為down的時候給child機會了,就是這里)
//然后就看他自己有onInterceptTouchEvent true。所以就傳給他自己的onTouchEvent了
getParent().requestDisallowInterceptTouchEvent(true);
break;
case MotionEvent.ACTION_MOVE:
if (ev.getX() < downX && downXAlignParentRight) {
//手指向左滑動 && 落點在最右測
//傳false == 讓父類處理。true == 本child我還要
getParent().requestDisallowInterceptTouchEvent(false);
} else if (ev.getX() < downX && getCurrentItem() == getAdapter().getCount() - 1){
getParent().requestDisallowInterceptTouchEvent(false);
} else {
// 下面這一句加不加都一樣了
// getParent().requestDisallowInterceptTouchEvent(true);
//因為:如果走到了上面requestDisallow(false)之后,觸摸事件就被父VG收回了,然后丟給本child一個Cancel
//同時父VG會把mFirstTouchTarget = null。后面的move就壓根就不看mGroupFlags了。 然后child本次touch就再也吃不到事件了(即使requestDisallow(true))
//如果不走上面的requestDisallow(false),那么就一直mGroupFlags == true;mFirstTouchTarget == 本child
}
break;
case MotionEvent.ACTION_UP:
break;
case MotionEvent.ACTION_CANCEL:
// 然后child本次touch就再也吃不到事件了(即使requestDisallow(true))
break;
}
return super.dispatchTouchEvent(ev);
}
其他開源庫:
我的其他開源庫,給個Star鼓勵我寫更多好庫:
Android 仿大眾點評、仿小紅書 下拉拖拽關(guān)閉Activity
Android 直播間聊天消息列表RecyclerView。一秒內(nèi)收到幾百條消息依然不卡頓
Android 仿快手直播界面加載中,頂部的滾動條狀LoadingView
Kotlin MVVM框架,全世界最優(yōu)化的分頁加載接口、最接地氣的封裝