When I first started learning Java, I watched teacher Mars’s videos. Miss Mars once said something that impressed me: have an object oriented mind.

If we use object-oriented thinking mode to think, will feel that the View drawing mechanism is very reasonable, very scientific. When we want to draw a picture on a piece of paper, we should first measure how big the picture is, then determine where the picture will look beautiful on the paper, and finally use the brush tool to draw the picture on the paper. The same is true in Android. The drawing process of View mainly refers to measure, layout and draw, namely measurement, layout and drawing. First you measure the width and height of the View, then you determine the position of the layout in the parent container, and then you draw the display.

The View’s rendering process starts with the performTraversals method of ViewRootImpl, In the performTraversals method, performMeasure, performLayout, and performDraw are called to traverse the entire view tree.

This blog will explore the first step in the View drawing process, measure.

MeasureSpec

The performMeasure method is called like this,

performMeasure(childWidthMeasureSpec, childHeightMeasureSpec);Copy the code

I received two parameters, and I wonder what those two parameters are. Subview width measurement manual and subview height measurement manual? MeasureSpec is a good first step.

MeasureSpec = MeasureSpec = MeasureSpec = MeasureSpec = MeasureSpec = MeasureSpec = MeasureSpec = MeasureSpec = MeasureSpec = MeasureSpec = MeasureSpec = MeasureSpec = MeasureSpec = MeasureSpec = MeasureSpec = MeasureSpec = measure;

There are three types of specMode:

EXACTLY: indicates that the parent View expects the child View to be the size determined by the specSize value, in which case the final View size is the size recorded by the specSize. The form corresponding to match_parent and concrete values in LayoutParams. For example, android:layout_width= “match_parent”, Android :layout_width= “50dp”

AT_MOST: maximum mode, which indicates that the parent container has specified an available specSize, and the child view can be no larger than the size specified in specSize. Corresponds to the form wrap_content in LayoutParams.

UNSPECIFIED: The parent container has no restrictions on the View, which can be as large as it wishes and is generally not used

MeasureSpec: What exactly is MeasureSpec for? A View’s MeasureSpec is used to measure the width and height of the View

MeasureSpec how does MeasureSpec come about? For a normal View, its MeasureSpec is determined by both the parent container’s MeasureSpec and its own LayoutParams. For a top-level View (DecorView), its MeasureSpec is determined by the size of the window and its own LayoutParams.

MeasureSpec a View has two MeasureSpec’s, one for measuring the width and one for measuring the height. Let’s go back to the performMeasure method and look at the parameters childWidthMeasureSpec and childHeightMeasureSpec. They are determined by the size of the window and its own LayoutParams. So how did they come about? Now it’s time to find out.

In the View otimPL’s measureHierarchy method, there are two lines of code that look like this

childWidthMeasureSpec = getRootMeasureSpec(desiredWindowWidth, lp.width);
childHeightMeasureSpec = getRootMeasureSpec(desiredWindowHeight, lp.height);
Copy the code

The getRootMeasureSpec method gets the MeasureSpec of the root View (DecorView). Ha-ha found it. Then assign values to childWidthMeasureSpec and childHeightMeasureSpec. DesiredWindowWidth and desiredWindowHeight are the screen sizes. Both lp.width and lP. height are MATCH_PARENT.

So explore the getRootMeasureSpec method as follows:

private static int getRootMeasureSpec(int windowSize, int rootDimension) {
    int measureSpec;
    switch (rootDimension) {

    case ViewGroup.LayoutParams.MATCH_PARENT:
        
        measureSpec = 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;
}
Copy the code

This goes to MeasureSpec’s makeMeasureSpec method, which wraps SpecSize and SpecMode into 32-bit int values. In this case, SpecSize is windowSize, the screen size, and SpecMode is EXACTLY. How does the makeMeasureSpec method assemble MeasureSpec? As follows:

public static int makeMeasureSpec(@IntRange(from = 0, to = (1 << MeasureSpec.MODE_SHIFT) - 1) int size, @MeasureSpecMode int mode) { if (sUseBrokenMakeMeasureSpec) { return size + mode; } else { return (size & ~MODE_MASK) | (mode & MODE_MASK); }}Copy the code

Thus, the Root View’s MeasureSpec is born. It participates in the MeasureSpec that makes up the child elements.

For a normal View, a MeasureSpec consists of the parent container’s MeasureSpec and its own LayoutParams. We know that the getRootMeasureSpec method fetched a Parent View’s MeasureSpec. Now look at the measureChild method of a ViewGroup, which measures child views. As follows:

protected void measureChild(View child, int parentWidthMeasureSpec,
        int parentHeightMeasureSpec) {
    final LayoutParams lp = child.getLayoutParams();

    final int childWidthMeasureSpec = getChildMeasureSpec(parentWidthMeasureSpec,
            mPaddingLeft + mPaddingRight, lp.width);
    final int childHeightMeasureSpec = getChildMeasureSpec(parentHeightMeasureSpec,
            mPaddingTop + mPaddingBottom, lp.height);

    child.measure(childWidthMeasureSpec, childHeightMeasureSpec);
}
Copy the code

Call getLayoutParams to get the child’s LayoutParams, and then call getChildMeasureSpec to get the child’s MeasureSpec. You can see that the parent (parentWidthMeasureSpec, parentHeightMeasureSpec) is passed in.

The getChildMeasureSpec method is important because it tells us how child MeasureSpec is generated, as follows:

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 == LayoutParams.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 = View.sUseZeroUnspecifiedMeasureSpec ? 0 : size;
            resultMode = MeasureSpec.UNSPECIFIED;
        } else if (childDimension == LayoutParams.WRAP_CONTENT) {
            
            
            resultSize = View.sUseZeroUnspecifiedMeasureSpec ? 0 : size;
            resultMode = MeasureSpec.UNSPECIFIED;
        }
        break;
    }
    
    return MeasureSpec.makeMeasureSpec(resultSize, resultMode);
}
Copy the code

If the parent’s specMode is EXACTLY and the child’s size is EXACTLY (set to android:layout_width= “50dp”), then the child’s specSize is childDimension (like 50dp), SpecMode is also EXACTLY. The other branches are pretty much the same. The method code is long, but the logic is not complicated. It assembles child elements’ MeasureSpec from the parent container’s MeasureSpec and the child elements’ LayoutParams. So a common View’s MeasureSpec is determined by the parent container’s MeasureSpec and its own LayoutParams.

MeasureSpec = MeasureSpec = MeasureSpec = MeasureSpec = MeasureSpec = MeasureSpec = MeasureSpec = Measure = Measure = Measure = Measure = Measure = Measure = Measure = Measure = Measure = Measure = Measure = Measure = Measure = Measure = Measure = Measure = Measure = Measure = Measure = Measure = Measure = Measure = Measure = Measure = Measure That’s a lot to remember. With MeasureSpec, the real measurement process is to determine the measurement size according to MeasureSpec’s measurement model. That’s it.

So now that we have MeasureSpec, we go back to the performMeasure method, where we pass in the MeasureSpec of the top-level View (DecorView). So let’s follow up and see how the View is measured.

The View of measurement

PerformMeasure method:

private void performMeasure(int childWidthMeasureSpec, int childHeightMeasureSpec) { Trace.traceBegin(Trace.TRACE_TAG_VIEW, "measure"); try { mView.measure(childWidthMeasureSpec, childHeightMeasureSpec); } finally { Trace.traceEnd(Trace.TRACE_TAG_VIEW); }}Copy the code

Switch to the View’s measure method as follows:

public final void measure(int widthMeasureSpec, int heightMeasureSpec) {

    

    final boolean forceLayout = (mPrivateFlags & PFLAG_FORCE_LAYOUT) == PFLAG_FORCE_LAYOUT;

    
    if (forceLayout || needsLayout) {
        
        mPrivateFlags &= ~PFLAG_MEASURED_DIMENSION_SET;

        resolveRtlPropertiesIfNeeded();

        int cacheIndex = forceLayout ? -1 : mMeasureCache.indexOfKey(key);
        if (cacheIndex < 0 || sIgnoreMeasureCache) {
            
            onMeasure(widthMeasureSpec, heightMeasureSpec);
            mPrivateFlags3 &= ~PFLAG3_MEASURE_NEEDED_BEFORE_LAYOUT;
        } 

        

    mOldWidthMeasureSpec = widthMeasureSpec;
    mOldHeightMeasureSpec = heightMeasureSpec;

    mMeasureCache.put(key, ((long) mMeasuredWidth) << 32 |
            (long) mMeasuredHeight & 0xffffffffL); 
}
Copy the code

As you can see, the measure method of a View is final and cannot be overridden by subclasses. After a bunch of processing, the core process is transferred to the onMeasure method to actually measure, and this onMeasure method usually needs to be overridden. For example, DecorView overrides the View’s onMeasure method, as does TextView.

But the View also defines the default implementation of the onMeasure method, so follow the onMeasure method,

protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
    setMeasuredDimension(getDefaultSize(getSuggestedMinimumWidth(), widthMeasureSpec),
            getDefaultSize(getSuggestedMinimumHeight(), heightMeasureSpec));
}
Copy the code

Call setMeasuredDimension to set the measured width and height, as if it were simple to say. If we were to override the onMeasure method to measure the width, we would also end up calling setMeasuredDimension to set the measured width. Ok, keep going

The getDefaultSize method is used to get the measurement width and height.

public static int getDefaultSize(int size, int measureSpec) {
    int result = size;
    int specMode = MeasureSpec.getMode(measureSpec);
    int specSize = MeasureSpec.getSize(measureSpec);

    switch (specMode) {
    case MeasureSpec.UNSPECIFIED:
        result = size;
        break;
    case MeasureSpec.AT_MOST:
    case MeasureSpec.EXACTLY:
        result = specSize;
        break;
    }
    return result;
}
Copy the code

MeasureSpec = measureSpec = measureSpec = measureSpec = measureSpec = measureSpec = measureSpec = measureSpec = measureSpec;

This is the measurement process of View.

The measurement of ViewGroup

So the next step is the measurement process of the ViewGroup. The onMeasure method is not overwritten in the ViewGroup, which is also very reasonable. How can you define an onMeasure method that fits multiple viewgroups? Obviously the onMeasure method implementations for LinearLayout and RelativeLayout are different. One is linear, one is relative.

In addition to measuring itself, a ViewGroup iterates through the child elements, thus iterating through the entire view tree. A measureChildren method is defined in the ViewGroup to traverse child elements as follows:

protected void measureChildren(int widthMeasureSpec, int heightMeasureSpec) { final int size = mChildrenCount; final View[] children = mChildren; for (int i = 0; i < size; ++i) { final View child = children[i]; if ((child.mViewFlags & VISIBILITY_MASK) ! = GONE) { measureChild(child, widthMeasureSpec, heightMeasureSpec); }}}Copy the code

Moves to the measureChild method to measureChild elements.

protected void measureChild(View child, int parentWidthMeasureSpec,
        int parentHeightMeasureSpec) {
    final LayoutParams lp = child.getLayoutParams();

    final int childWidthMeasureSpec = getChildMeasureSpec(parentWidthMeasureSpec,
            mPaddingLeft + mPaddingRight, lp.width);
    final int childHeightMeasureSpec = getChildMeasureSpec(parentHeightMeasureSpec,
            mPaddingTop + mPaddingBottom, lp.height);

    child.measure(childWidthMeasureSpec, childHeightMeasureSpec);
}
Copy the code

In this way, the ViewGroup passes the measure procedure to the child element. This repeatedly completes the entire view tree.

So that’s the ViewGroup measurement process

conclusion

I think learning View drawing, you can look at the source of some system View, such as LinearLayout, TextView, TextView is more difficult, you can pick a simple point.

As an example, the activity_main.xml file looks like this:


    

Copy the code

The rendering process of the View begins with the performTraversals method of the ViewRootImpl, where the performMeasure method is called. In performMeasure, the DecorView measure method is called, and the entire view tree is measured. The DecorView then calls its overridden onMeasure method. In the DecorView onMeasure method, it calls its FrameLayout onMeasure parent. Before the measure procedure is passed to the child element, that is, the measure method of the child element is called. That’s where the ContentView of our activity_main. XML file comes in. Then the LinearLayout ViewGroup measures itself and passes the measure to the TextView. This is the measure passing of a ViewGroup. Measure method is not allowed to be overridden by subclasses, but it’s important. Measure method is the entry point of transmission. For example, after the parent element traverses the child element, it usually passes the measure process to the child element, calling the measure method of the child element

child.measure(childWidthMeasureSpec, childHeightMeasureSpec)Copy the code

The Measure method passes a child’s MeasureSpec, which is determined by both the parent’s MeasureSpec and the child’s own LayoutParams. See the measureChild method above for details. The onMeasure method is then called in the child element’s measure to actually measure the View. OnMeasure methods are usually overridden. According to MeasureSpec to determine the View measurement width and height.

Shout! Really want to see the system control source code, or to see the big god write custom View, can help us comb the knowledge of View drawing. For example, Guo Shen wrote an article about Scroller and also introduced the implementation of a simplified version of ViewPager. This custom ViewPager method overrides the onMeasure method like this:

@Override protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { super.onMeasure(widthMeasureSpec, heightMeasureSpec); int childCount = getChildCount(); for (int i = 0; i < childCount; i++) { View childView = getChildAt(i); measureChild(childView, widthMeasureSpec, heightMeasureSpec); }}Copy the code

When you study, you can think, why does Guo Da Xia write like this? The first line super.onMeasure measures itself (View onMeasure is useful!). Then iterate over the child elements, calling the measureChild method to measure the child elements. End! This also verifies that the ViewGroup not only measures itself, but also traverses the measurement child elements.

These are the methods I learned to draw a View and customize a View.

Reprint please indicate the source: blog.csdn.net/xyh269/arti…