已知系統內部是根據每個View的MeasureSpec來得到View得寬和高,那么我們是怎么獲得到每個View對應的MeasureSpec呢?
DecorView
對于DecorView,其MeasureSpec由窗口的尺寸和其自身的LayoutParams來共同決定的。已知,ViewRootImpl類是連接WindowManager和DecorView的紐帶。在ViewRootImpl中的measureHierarchy方法中有如下的代碼,它展示了DecorView的MeasureSpec的創建過程:
childWidthMeasureSpec = getRootMeasureSpec(desiredWindowWidth, lp.width); //desireWindowWidth是屏幕的寬度
childHeightMeasureSpec = getRootMeasureSpec(desiredWindowHeight, lp.height);
performMeasure(childWidthMeasureSpec, childHeightMeasureSpec);
上面代碼調用的getRootMeasureSpec()
方法的代碼如下
private static int getRootMeasureSpec(int windowSize, int rootDimension) {
int measureSpec;
switch (rootDimension){
case ViewGroup.LayoutParams.MATCH_PARENT:
mesureSpec = MeasureSpec.makeMeasureSpec(windowSize,MeasureSpec.EXACTLY);
break;
case ViewGroup.LayoutParams.WRAP_CONTENT:
measureSpec = MeasureSpec.makeMeasureSpec(windowSize,MeasureSpec.AT_MOST);
break;
default:
measureSpec = MeasureSpec.makeMeasureSpec(rootDimension,MeasureSpec.EXACTLY);
break;
}
return measureSpec;
}
普通View
普通View是指布局中的View,View的measure過程由ViewGroup傳遞而來,下面是ViewGroup的measureChildWidthMargins()
方法
protected void measureChildWithMargins( View child,int parentWidthMeasureSpec, int widthUsed,
int parentHeightMeasureSpec,int heightUsed){
final MarginLayoutParams lp = (MarginLayoutParams) child.getLayoutParams();
final int childWidthMeasureSpec=getChildMeasureSpec(parentWidthMeasureSpec,mPaddingLeft+mPaddingRight+lp.leftMargin+lp.rightMargin+widthUsed,lp.width);
final int childHeightMeasureSpec=getHeightMeasureSpec(parentHeightMeasureSpec,mPaddingTop+mPaddingBottom+lp.topMargin+lp.bottomMargin+heightUsed,lp.height);
child.measure(childWidthMeasureSepc, childHeightMeasureSpec);
}
在上面ViewGroup的的measureChildWithMargins()
方法中,首先通過getChildMeasureSpec()
方法獲得子View的MeasureSpec,然后,調用child.measure()
方法。那么ViewGroup中的getChildMeasureSpec()
方法具體是什么樣的呢?
public static int getChildMeasureSpec(int spec,int padding,int childDimension){
int specMode = MeasureSpec.getMode(spec);
int specSize = MeasureSpec.getSize(spec);
int size = Math.max(0,specSize - padding);
int resultSize = 0;
int resultMode = 0;
switch (specMode){
case MeasureSpec.EXACTLY:
if (childDimension >= 0){
resultSize = childDimension;
resultMode = MeasureSpec.EXACTLY;
} else if (childDimension == LayoutParams.MATCH_PARENT){
resultSize = size;
resultMode = MeasureSpec.EXACTLY;
} else if (childDimension == LayoutParams.WRAP_CONTENT){
resultSize = size;
resultMode = MeasureSpec.AT_MOST;
}
break;
case MeasureSpec.AT_MOST:
if(childDimension >= 0){
resultSize = childDimension;
resultMode = MeasureSpec.EXACTLY;
} else if (childDimension == LayoutParams.MATCH_PARENT) {
resultSize = size;
resultMode = MeasureSpec.AT_MOST;
} else if (childDimension == LayoutParam.WRAP_CONTENT){
resultSize = size;
resultMode = MeasureSpec.AT_MOST;
}
break;
case MeasureSpec.UNSPECIFIED:
if (childDimension >= 0){
resultSize = childDimension;
resultMode = MeasureSpec.EXACTLY
} else if (childDimension == LayoutParams.MATCH_PARENT) {
resultSize = 0;
resultMode = MeasureSpec.UNSPECIFIED;
} else if (childDimension ==== LayoutParams.UNSPECIFIED) {
result = 0;
resultMode = MeasureSpec.UNSPECIFIED;
}
break;
}
return MeasureSpec.makeMeasureSpec(resultSize,resultMode);
}