Android中ScrollView和listView嵌套 滑動衝突解決

常用方法有兩種①自定義listview重寫onMueasure方法②計算所有item的高度以及divider的高度之和。這兩種方法都可以將listview的滾動取消。
第一種方法

protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
    int expandSpec = MeasureSpec.makeMeasureSpec(Integer.MAX_VALUE >> 2,MeasureSpec.AT_MOST);
    super.onMeasure(widthMeasureSpec, expandSpec);
    }

下面是關於onMeasure方法裏的關鍵代解釋。參考的另一個博客(也是轉載的沒標明來源)
在自定義View和ViewGroup的時候,我們經常會遇到int型的MeasureSpec來表示一個組件的大小,這個變量裏面不僅有組件的尺寸大小,還有大小的模式。
這個大小的模式,有點難以理解。在系統中組件的大小模式有三種:
1.精確模式(MeasureSpec.EXACTLY)
在這種模式下,尺寸的值是多少,那麼這個組件的長或寬就是多少。
2.最大模式(MeasureSpec.AT_MOST)
這個也就是父組件,能夠給出的最大的空間,當前組件的長或寬最大隻能爲這麼大,當然也可以比這個小。
3.未指定模式(MeasureSpec.UNSPECIFIED)
這個就是說,當前組件,可以隨便用空間,不受限制。
可能有很多人想不通,一個int型整數怎麼可以表示兩個東西(大小模式和大小的值),一個int類型我們知道有32位。而模式有三種,要表示三種狀 態,至少得2位二進制位。於是系統採用了最高的2位表示模式。如圖:

最高兩位是00的時候表示”未指定模式”。即MeasureSpec.UNSPECIFIED
最高兩位是01的時候表示”’精確模式”。即MeasureSpec.EXACTLY
最高兩位是11的時候表示”最大模式”。即MeasureSpec.AT_MOST
很多人一遇到位操作頭就大了,爲了操作簡便,於是系統給我提供了一個MeasureSpec工具類。
這個工具類有四個方法和三個常量(上面所示)供我們使用:

//這個是由我們給出的尺寸大小和模式生成一個包含這兩個信息的int變量,這裏這個模式這個參數,傳三個常量中的一個。
public static int makeMeasureSpec(int size, int mode)
//這個是得到這個變量中表示的模式信息,將得到的值與三個常量進行比較。
public static int getMode(int measureSpec)
//這個是得到這個變量中表示的尺寸大小的值。
public static int getSize(int measureSpec)
//把這個變量裏面的模式和大小組成字符串返回來,方便打日誌
public static String toString(int measureSpec)

MeasureSpec.EXACTLY:使用measureSpec中size的值作爲寬高的精確值
當我們將控件的layout_width或layout_height指定爲具體數值時如andorid:layout_width=”50dip”,或者爲FILL_PARENT是,都是控件大小已經確定的情況,都是精確尺寸。
MeasureSpec.AT_MOST:使用measureSpec中size的值作爲最大值,採用不超過這個值的最大允許值
當控件的layout_width或layout_height指定爲WRAP_CONTENT時,控件大小一般隨着控件的子空間或內容進行變化,此時控件尺寸只要不超過父控件允許的最大尺寸即可。因此,此時的mode是AT_MOST,size給出了父控件允許的最大尺寸。
MeasureSpec.UNSPECIFIED是未指定尺寸,這種情況不多

onMesure方法 中 Integer.MAX_VALUE >> 2的意思解釋:

1000的二進制:1111101000
右移2位後:11111010,十進制爲:250

這樣就指定了listview的高度爲250px以內的最大允許值(一般就是250)

把AT_MOST改爲EXACTLY,則精確指定listview高度值爲250px,如果listview內容全部顯示的高度爲500px(大於250px),那麼當measureSpec中size的值爲250px(小於500px)時,效果是一樣的

如果設置的measureSpec中size的值大於listview內容全部顯示的高度,那麼設置成AT_MOST時,最多顯示listview內容全部顯示的高度,而EXACTLY還是顯示measureSpec中size的值,所以EXACTLY在這種情況下,後面會留有空白高度(measureSpec中size的值大於listview內容全部顯示的高度的部分顯示爲空白)
所以,一般這樣寫可以讓listview正確測量:
int width = MeasureSpec.makeMeasureSpec(Integer.MAX_VALUE>>2,MeasureSpec.AT_MOST);
MAX_VALUE右移2位後,即使不是最大整數了,listview的高度也一般不可能超過它
第一個參數有個最大值的限制:1073741823(二進制的30個1),MAX_VALUE是1個0加上31個1(二進制),所以也可以右移1位,但是由於最前面兩位表示mode,而不是size,所有右移1位和右移2位是一樣的(前面兩位的值都會被mode的代碼覆蓋)
第二種方法

public static class Utility {
        public static void setListViewHeightBasedOnChildren(ListView listView) {
            ListAdapter listAdapter = listView.getAdapter();
            if (listAdapter == null) {
                return;
            }

            int totalHeight = 0;
            for (int i = 0; i < listAdapter.getCount(); i++) {
                View listItem = listAdapter.getView(i, null, listView);
                listItem.measure(0, 0);
                totalHeight += listItem.getMeasuredHeight();
            }

            ViewGroup.LayoutParams params = listView.getLayoutParams();
            params.height = totalHeight + (listView.getDividerHeight() * (listAdapter.getCount() - 1));
            listView.setLayoutParams(params);
        }
    }

使用Utility類中的靜態方法必須在listview設置了adapter之後再調用,而且item佈局必須是用linearLayout佈局作爲根元素(因爲其他的Layout(如RelativeLayout)沒有重寫onMeasure(),所以會在onMeasure()時拋出異常。)。

總結:如果scrollview中嵌套一個listview這兩種方法任意一種都可以解決,如果不處理就不能解決和外部的衝突(問題一:ScrollView與ListView嵌套導致ListView顯示不全面。問題二:ScrollView不能正常滑動)。如果是嵌套兩個listview使用其中一種方法也可以解決,但千萬不能混合使用否則第二個listview就只會顯示一行。另外其實Google以及大公司開發是不建議這種有衝突控件嵌套的,所以能用其他方法解決儘量用其他方法。
另外補充兩種解決思路:
解決方式一:
ScrollView+LinearLayout+ListView可以換成ScrollView+LinearLayout+LinearLayout,對於開發中,ScrollView所能滾動的樣式形式各異,另外的話,ScrollView所顯示的內容肯定不會太多,因此這種方案是合理而且可選的
解決方式二:
同樣是替換:ListView具有HeaderView與FooterView2部分,因此,在非下拉刷新,上拉加載的需求中,完全可以使用ListView來代替ScrollView,因此是合理可選的方案
爲解決衝突,建議好好了解Android事件分發機制,比如OnTouchEvent、touchEvent以及dispatchTouchEvent源碼還有一個,另外兩個requestDisallowInterceptTouchEvent和onInterceptTouchEvent。

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章