preface
A View’s MeasureSpec is a View’s parent and its layout parameters are the same as those of the parent View. A View’s MeasureSpec is a View’s parent and its layout parameters are the same as the parent View’s. ](Measurement of Android View drawing process 1 – Gold digging (juejin. Cn)
Articles in this series:
Working principle of Android View 3 – Layout process – Nuggets (juejin. Cn)
Android View working Principle 4 — Draw process – Digging gold (juejin. Cn)
Now let’s look specifically at how we measure according to MeasureSpec.
The body of the
Let’s take a look at the method chain that ViewRootImpl calls.
ViewRootImpl began
One of the most common methods is requestLayout, which is called when we need to refresh the entire View chain. We’ll start with this:
- Call in code:
main_view_pager.requestLayout()
Copy the code
- Methods in View:
// requestLayout for View source code
public void requestLayout() {
/ / to omit
if(mParent ! =null && !mParent.isLayoutRequested()) {
// Call the mParent method
mParent.requestLayout();
}
if(mAttachInfo ! =null && mAttachInfo.mViewRequestingLayout == this) {
mAttachInfo.mViewRequestingLayout = null; }}Copy the code
- MParent is an interface:
/ / ViewParent interface
public interface ViewParent {
/** * Called when something has changed which has invalidated the layout of a * child of this view parent. This will schedule a layout pass of the view * tree. */
public void requestLayout();
/ / to omit
}
Copy the code
- The ViewRootImpl class implements this interface. Look at this method under this class:
// methods under the ViewRootImpl class
@Override
public void requestLayout() {
if(! mHandlingLayoutInLayoutRequest) { checkThread(); mLayoutRequested =true;
// Start traversalscheduleTraversals(); }}Copy the code
- Continue to look at:
// Start traversal
void scheduleTraversals() {
if(! mTraversalScheduled) { mTraversalScheduled =true;
mTraversalBarrier = mHandler.getLooper().getQueue().postSyncBarrier();
// Run runnable
mChoreographer.postCallback(
Choreographer.CALLBACK_TRAVERSAL, mTraversalRunnable, null); notifyRendererOfFramePending(); pokeDrawLockIfNeeded(); }}Copy the code
- Take a look at the Runnable code:
final class TraversalRunnable implements Runnable {
@Override
public void run(){ doTraversal(); }}Copy the code
void doTraversal() {
if (mTraversalScheduled) {
mTraversalScheduled = false;
mHandler.getLooper().getQueue().removeSyncBarrier(mTraversalBarrier);
if (mProfile) {
Debug.startMethodTracing("ViewAncestor");
}
// Start the real traversal
performTraversals();
if (mProfile) {
Debug.stopMethodTracing();
mProfile = false; }}}Copy the code
- In this method, actually began to measure, the first is to get the MeasureSpec DecorView, then call performMeasure began measuring, because this part of the code is too much, do not paste, take a look at this method:
// where to call
performMeasure(childWidthMeasureSpec, childHeightMeasureSpec);
Copy the code
// Call the view's measure method
private void performMeasure(int childWidthMeasureSpec, int childHeightMeasureSpec) {
if (mView == null) {
return;
}
Trace.traceBegin(Trace.TRACE_TAG_VIEW, "measure");
try {
mView.measure(childWidthMeasureSpec, childHeightMeasureSpec);
} finally{ Trace.traceEnd(Trace.TRACE_TAG_VIEW); }}Copy the code
- It calls the View’s measure method, and the subsequent measurement is passed on to the view.
The View starts to take over
Now that we’re at the View, we’ll immediately look at the measure method and see that it’s a final method:
Don’t worry, the onMeasure method will be called inside this method. Let’s look at this method:
// the measure method is called internally
protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
setMeasuredDimension(getDefaultSize(getSuggestedMinimumWidth(), widthMeasureSpec),
getDefaultSize(getSuggestedMinimumHeight(), heightMeasureSpec));
}
Copy the code
Here we see that it passes its width and height MeasureSpec to the function, so let’s see how the View defaults to handle MeasureSpec.
SetMeasureDimension method
This method is called inside the above function. Let’s look at the source definition of this function:
<p>This method must be called by {@link #onMeasure(int, int)} to store the
* measured width and measured height. Failing to do so will trigger an
* exception at measurement time.</p>
Copy the code
This method must be called in onMeasure, and its function is to save the width and height of the measurement. The problem is simple, that is, to get the specific width and height of the measurement can be set by this method. How to get the width and height of the measurement by default
GetDefaultSize method
Call this method, look at the source of this method:
// The first parameter is only used with UNSPECIFIED measurement mode, so it is not analyzed yet
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;
// Both AT_MOST and EXACTLY are measureSpec sizes
case MeasureSpec.AT_MOST:
case MeasureSpec.EXACTLY:
result = specSize;
break;
}
return result;
}
Copy the code
As you can see from this code, by default, if we write a CustomView that inherits directly from the View, its size is measureSpec.
The problem
There is a problem with the logic above, so let’s go back to a diagram from the previous article:
When the parent View’s SpecMode is wrap_content, the parent View’s SpecMode is AT_MOST. The problem here is that when our CustomView is written wrap_content in XML, it will still be the same as match_parent, so this is also a defect.
To solve the problem
So a custom control that directly inherits from a View needs to override the onMeasure method and set the size of wrAP_conent.
When the specMode of measureSpec is AT_MOST, give it the default size:
// Custom views that directly inherit from views must be written
@Override
protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
setMeasuredDimension(measureWidth(widthMeasureSpec),
measureHeight(heightMeasureSpec));
}
Copy the code
// protect the case when wrap_content is invalid
private int measureWidth(int widthMeasureSpec) {
int result;
int specMode = MeasureSpec.getMode(widthMeasureSpec);
int specSize = MeasureSpec.getSize(widthMeasureSpec);
if (specMode == MeasureSpec.EXACTLY) {// Exact mode
result = specSize;
} else {
isExactly = false;
// Maximum size mode, default width
result = defaultWidth;
if (specMode == MeasureSpec.AT_MOST) {
result = Math.min(result, specSize); }}return result;
}
Copy the code
We’re just going to keep the wrap_content property alive, but it’s up to the View to see how it works, and we’ll talk more about that later when we get to the complexity of customizing a View.
ViewGroup personalization
For a ViewGroup, it has child views, so when the measurement event is passed to the ViewGroup, it must first let its child View take the measurement, and then set its own width and height.
Because it’s very simple, when you set it to match_parent or something, you know what the size of this ViewGroup is, but when you set wrap_content, there’s a problem, it can’t do the same thing as View and I give it a default value, which doesn’t make sense, For example, in a LinearLayout, I arranged a lot of views vertically and ended up with wrap_content showing a specific height, which is definitely not going to work.
Therefore, different viewgroups need to traverse their sub-views in their onMeasure method, let the sub-views finish measuring, get their real data, and then determine their size.
What about viewGroups
Now that we know why a custom View cannot directly set its measurement size when inheriting from a ViewGroup, we must first measure the size of its child View. There are some handy methods in viewGroups that can help us to do this faster.
The main idea is to measure the View, get the results measured, which are traversing its View, arranged according to specific rules for measuring size of the View, so here on the basis of this ViewGroup MeasureSpec to measure child View, provides several commonly used methods:
// Just pass this through
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 completes the measurement
child.measure(childWidthMeasureSpec, childHeightMeasureSpec);
}
Copy the code
// This method can be called when there is a custom gap
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 = getChildMeasureSpec(parentHeightMeasureSpec,
mPaddingTop + mPaddingBottom + lp.topMargin + lp.bottomMargin
+ heightUsed, lp.height);
child.measure(childWidthMeasureSpec, childHeightMeasureSpec);
}
Copy the code
Here is an example of a custom ViewGroup, which is used in a project. The idea is easy, but the implementation is still quite troublesome, which will be discussed later in this article.
conclusion
Due to the limited space of this article, I will not give detailed examples. Here is a brief summary:
-
The measurement process starts with View wrootimPL in the top-level View.
-
Either inheriting a View or a custom View inheriting a ViewGroup overrides the onMeasure method and finally calls setMeasureDimension to set its measurement size.
-
Inheriting a custom View from a View requires additional handling of wrAP_content invalidation.
-
When a custom View inheriting ViewGroup is measured, it needs to traverse its sub-view first, and then set its own size after the sub-view is measured and gets the result.