MeasureSpec學習


在自定義ViewViewGroup的時候,我們經常會遇到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:當我們將控件的layout_width或layout_height指定爲具體數值時如andorid:layout_width="50dip",或者爲FILL_PARENT是,都是控件大小已經確定的情況,都是精確尺寸。

        MeasureSpec.AT_MOST是最大尺寸,當控件的layout_width或layout_height指定爲WRAP_CONTENT時, 控件大小一般隨着控件的子空間或內容進行變化,此時控件尺寸只要不超過父控件允許的最大尺寸即可。因此,此時的mode是AT_MOST,size給出了 父控件允許的最大尺寸。

        MeasureSpec.UNSPECIFIED是未指定尺寸,這種情況不多,一般都是父控件是AdapterView,通過measure方法傳入的模式。

因此,在重寫onMeasure方法時要根據模式不同進行尺寸計算。下面代碼就是一種比較典型的方式:

@Override      
protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {      
    setMeasuredDimension(getMeasuredLength(widthMeasureSpec, true), getMeasuredLength(heightMeasureSpec, false));      
}      
      
      
private int getMeasuredLength(int length, boolean isWidth) {      
    int specMode = MeasureSpec.getMode(length);      
    int specSize = MeasureSpec.getSize(length);      
    int size;      
    int padding = isWidth ? getPaddingLeft() + getPaddingRight()      
            : getPaddingTop() + getPaddingBottom();      
    if (specMode == MeasureSpec.EXACTLY) {      
        size = specSize;      
    } else {      
        size = isWidth ? padding + mWave.length / 4 : DEFAULT_HEIGHT      
                + padding;      
        if (specMode == MeasureSpec.AT_MOST) {      
            size = Math.min(size, specSize);      
        }      
    }      
    return size;      
}

 

 

解決ScrollView嵌套ListView和GridView衝突的方法

public class MyListView extends ListView {  
        public MyListView(Context context) {  
                super(context);  
        }  
        public MyListView(Context context, AttributeSet attrs) {  
                super(context, attrs);  
        }  
        public MyListView(Context context, AttributeSet attrs, int defStyle) {  
                super(context, attrs, defStyle);  
        }  
        @Override  
        protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {  
                int expandSpec = MeasureSpec.makeMeasureSpec(Integer.MAX_VALUE >> 2,  
                                MeasureSpec.AT_MOST);  
                super.onMeasure(widthMeasureSpec, expandSpec);  
        }  
}  
  
  
   
public class MyGridView extends GridView {     
    private boolean haveScrollbar = true;     
    public MyGridView(Context context) {     
        super(context);     
    }     
    public MyGridView(Context context, AttributeSet attrs) {     
        super(context, attrs);     
    }     
    public MyGridView(Context context, AttributeSet attrs, int defStyle) {     
        super(context, attrs, defStyle);     
    }     
    /**    
     * 設置是否有ScrollBar,當要在ScollView中顯示時,應當設置爲false。 默認爲 true    
     *     
     * @param haveScrollbars    
     */     
    public void setHaveScrollbar(boolean haveScrollbar) {     
        this.haveScrollbar = haveScrollbar;     
    }     
    @Override     
    protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {     
        if (haveScrollbars == false) {     
            int expandSpec = MeasureSpec.makeMeasureSpec(Integer.MAX_VALUE >> 2, MeasureSpec.AT_MOST);    
            super.onMeasure(widthMeasureSpec, expandSpec);     
        } else {     
            super.onMeasure(widthMeasureSpec, heightMeasureSpec);     
        }     
    }     
}


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