MeasureSpec学习 - 转
2019獨角獸企業重金招聘Python工程師標準>>>
在自定義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:當我們將控件的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);}@Overrideprotected 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); } } }
?
?
轉載于:https://my.oschina.net/qiuhoude/blog/410809
總結
以上是生活随笔為你收集整理的MeasureSpec学习 - 转的全部內容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: PYTHON招聘需求与技能体系
- 下一篇: grub加密