首先來看一個栗子吧~
自定義了一個ViewGroup将谊,名叫HorizontalScrollView脐帝,里面的子View橫向排布怜跑,在ViewGroup上左右滑動時里面的子View會跟著滑動伊诵,手指抬起時,根據(jù)頁面滑動的距離動態(tài)滑動到相應子View(手指抬起時哪個子View占據(jù)空間大則滑動到那個子View)嘱函。
主要復寫了ViewGroup的onMeasure、onLayout埂蕊、onTouchEvent方法:
① 復寫onMeasure方法:
這里要手動調(diào)一次measureChildren方法來測量子View的寬高往弓,因為super.onMeasure中調(diào)用的是View的onMeasure方法,只會測量自身的寬高蓄氧。
② 復寫onLayout方法:
在onLayout方法函似,需要調(diào)用layout方法來確定子View的位置,這里我將子View橫向排放喉童。這里還需要獲取子View最左邊的邊界和最右邊的邊界(后面滑動判斷邊界時會用到)撇寞。
③ 復寫onTouchEvent方法:
在Move狀態(tài)中,有三種情況:
1.向右滑動過程中若已滑動到左邊界堂氯,則調(diào)用scrollTo方法停留在左邊界
2.向左滑動過程中若已滑動到右邊界蔑担,則調(diào)用scrollTo方法停留在右邊界
3.其余情況,調(diào)用scrollBy方法進行滑動咽白,手機滑動多少頁面就滑動多少
在up狀態(tài)時钟沛,若當前子View的寬度在界面中出現(xiàn)一半以上(我在布局中設置每個子View都是match_parent,子View完整展示時剛好是界面的寬度),則調(diào)用startScroll方法平滑地滑動到相應的子View局扶。
在調(diào)用startScroll方法進行平滑移動時恨统,需要復寫computeScroll方法,如下所示
好了三妈,在布局中給HorizontalScrollView添加三個ListView作為子View畜埋,如下所示
想像一下效果,左右滑動的時候ListView之間會進行切換畴蒲,上下滑動的時候ListView也能正秤瓢埃滑動,美滋滋~運行一下看看咯^:^
額模燥,不管怎么滑動咖祭,都是展示的是第一個ListView額,給跪orz蔫骂。
其實這是一個滑動沖突的問題么翰,要解決滑動沖突,首先要了解Android中的事件傳遞機制辽旋。關于這方面的知識浩嫌,可以參看郭神的《事件分發(fā)機制》相關文章檐迟,文章里講的很詳細~
分析原因
出現(xiàn)上面情況的原因是,ViewGroup默認會將滑動事件分發(fā)給它的子View處理码耐,而它的子View將滑動事件消耗了追迟。所以,HorizontalViewGroup中的onTouchEvent中雖然做了滑動處理的相關邏輯骚腥,但是它根本就沒有執(zhí)行敦间。
如何解決
解決的方法有兩種,分別是:
1.外部攔截法
2.內(nèi)部攔截法
在事件傳遞的流程中束铭,父View在接收到事件后措左,其實可以對事件進行攔截處理廉白,不讓它傳遞給子View条摸,查看ViewGroup類中dispatchTouchEvent方法中的源碼(API 25)辰斋,可以看到下面這段代碼
可以看到,在里面調(diào)用了onInterceptToucheEvent方法來判斷是否對事件進行攔截埠褪,因此可以通過重寫onInterceptTouchEvent方法來對需要處理的事件進行攔截浓利,這就是外部攔截的方法。
還有一個優(yōu)先級更高的操作钞速,那就是disallowIntercept標志贷掖,若disallowIntercept標記為true,則onInterceptTouchEvent方法根本就不會執(zhí)行渴语,那么這個flag要怎么設置呢苹威?可以看到它和FLAG_DISALLOW_INTERCEPT有關,而FLAG_DISALLOW_INTERCEPT標志可以通過調(diào)用父View的requestDisallowInterceptTouchEvent來設置:
①父View.requestDisallowInterceptTouchEvent(true)驾凶,則父View無法攔截牙甫;
②父View.requestDisallowInterceptTouchEvent(false),則父view可以在onInteceptTouchEvent方法中進行事件攔截调违。
而這個requestDisallowInterceptToucheEvent方法可以在子View中執(zhí)行窟哺,當需要父View攔截時,調(diào)用上面的①技肩,而當不需要父View攔截時且轨,調(diào)用上面的②,這就是內(nèi)部攔截法虚婿。
外部攔截法實踐
重寫父View的onInterceptTouchEvent方法旋奢,在move狀態(tài)時判斷,若x方向滑動的距離大于y方向滑動的距離然痊,則攔截該事件進行處理至朗,代碼如下:
運行一下,看一下效果:(左右滑動時切換子View玷过,每個列表都能正常地上下滑動和點擊)
外部攔截成功(陰影是播放器的哈爽丹,不必在意)~
內(nèi)部攔截法實踐
1.重寫父View的onInterceptTouchEvent方法,默認攔截除了down以外的所有事件辛蚊。
2.自定義ListView粤蝎,重寫dispatchTouchEvent方法:
①在down時,不允許父View攔截
②在move時袋马,若x方向滑動的距離大于y方向滑動的距離初澎,允許父View攔截
代碼如下:
運行一下看效果:
效果和外部攔截一樣,內(nèi)部攔截也成功啦~
回顧
外部攔截和內(nèi)部攔截的效果是一樣的虑凛,具體應用時要看攔截的邏輯在哪邊寫比較方便碑宴。在本例中外部攔截比內(nèi)部攔截邏輯更清晰簡潔,因此選用外部攔截會更好~
項目代碼:滑動沖突解決實踐