如何获得MeasureSpec

已知系统内部是根据每个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);
}

你可能感兴趣的:(如何获得MeasureSpec)