The first post is about creating a View, and then the measurement and layout of the View. So just to give you the big idea, View is measured from top to bottom, layer to layer. The core method involved is measure() layout() in the View. For us, we should care more about the callback methods of onMeasure() and onLayout(). This article focuses on the measurement related code, as for layout, this is the specific logic of ViewGroup.
onMeasure
The onMeasure() method must be mentioned about the measurement modes involved. And the constraints that the schema imposes on subviews.
protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
setMeasuredDimension(getDefaultSize(getSuggestedMinimumWidth(), widthMeasureSpec),
getDefaultSize(getSuggestedMinimumHeight(), heightMeasureSpec));
}
Copy the code
This is the default implementation of the onMeasure() method in View, and there are three more important methods involved here, setMeasuredDimension() and getDefaultSize(). The setMeasuredDimension() method is very important because it is the only officially specified method we use to set measurement width-height values. This is the method we have to call in the onMeasure() method. If you think about it, you don’t seem to have called this method manually in onMeasre() and there’s nothing wrong with it, don’t hesitate, You must have called super.onMeasure(), and setMeasuredDimension() will eventually complete the measureHeight and measureWidth assignments below.
private void setMeasuredDimensionRaw(int measuredWidth, int measuredHeight) {
mMeasuredWidth = measuredWidth;
mMeasuredHeight = measuredHeight;
mPrivateFlags |= PFLAG_MEASURED_DIMENSION_SET;
}
Copy the code
Assigning mMeasuredWidth and mMeasuredHeight is accomplished by calling the private setMeasuredDimensionRaw() method in setMeasuredDimension(), and then updating the flag.
getSuggestedMinimumWidth/Height
protected int getSuggestedMinimumWidth() {
return (mBackground == null) ? mMinWidth : max(mMinWidth, mBackground.getMinimumWidth());
}
protected int getSuggestedMinimumHeight() {
return (mBackground == null) ? mMinHeight : max(mMinHeight, mBackground.getMinimumHeight());
}
Copy the code
The default implementation of these two methods is to get the smallest of the View setting’s background and minimum values. The background setting is unnecessary, and the minimum width and height is set dynamically by minWidth or API in XML.
getDefaultSize()
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
This method is also important because it involves measurement patterns. First analyze the parameters, the first size input is the minimum value just obtained. The second is the measurement parameters called back from the parent layout.
As can be seen from the above, there are three measurement modes. MeasureSpec.UNSPECIFIED MeasureSpec.AT_MOST MeasureSpec.EXACTLY
With MeasureSpec.UNSPECIFIED, the obtained minimum value is used directly. If it is the other two modes, then the corresponding size is obtained from the measurement parameters. Notice that in this method there is no distinction between AT_MOST and EXACTLY at all.
MeasureSpec Measurement mode and size
MeasureSpec = MeasureSpec = MeasureSpec = MeasureSpec = MeasureSpec = MeasureSpec = MeasureSpec = MeasureSpec So how does it make an int that represents two kinds of information? Programmer’s little ingenuity on line.
An int of 32 bits. Here, the upper 2 bits are used to represent the measurement mode, and the lower 30 bits are used to record size.
Private static final int MODE_SHIFT = 30; private static final int MODE_SHIFT = 30; // Binary is like 11000... 000 private static final int MODE_MASK = 0x3 << MODE_SHIFT; / / 00 000... 000 public static final int UNSPECIFIED = 0 << MODE_SHIFT; / / 01 000... 000 public static final int EXACTLY = 1 << MODE_SHIFT; / / 10, 000... 000 public static final int AT_MOST = 2 << MODE_SHIFT;Copy the code
Now let’s see how we assign and evaluate.
public static int makeMeasureSpec(@IntRange(from = 0, to = (1 << MeasureSpec.MODE_SHIFT) - 1) int size,
@MeasureSpecMode int mode) {
return (size & ~MODE_MASK) | (mode & MODE_MASK);
}
Copy the code
Doesn’t it look a little fancy? I said this is a programmer’s clever thinking, of course the code is better. There are and or not operations involved. As a direct example, let’s say I want to create a MeasureSpec with a size of 16 that is EXACTLY how it is.
Size corresponds to 00 000... 1111 mode corresponds to 01 000... 0000 mask for 11 000... 0000 ~mask corresponds to 00 111... 1111 size & ~mask 00 000... 1111 = size mode & mask 01 000... 0000 = mode size | mode 01 000... 1111 = final resultCopy the code
With this correspondence, the result is very clear. Do you feel that the first two & operations in the makeMeasureSpec() method are very invalid? In fact, it can ensure that mode and size do not cross boundaries and do not pollute each other. You don’t want to spread the word anyway. When assigning, the method already has input constraints on both parameters.
Having said three schema definitions, it is then necessary to consider how the width and height specifications in XML are ultimately translated into the corresponding schema. For example, if we write wrAP_content, what is the corresponding measurement mode?
For example, let’s look at the following layout.
<FrameLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:id="@+id/parent_container"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:layout_gravity="center"
android:background="@color/colorAccent">
<ProgressBar
android:id="@+id/child"
android:layout_width="20dp"
android:layout_height="20dp"
android:layout_gravity="center"
android:indeterminate="true"
android:indeterminateTint="@color/colorPrimary"
android:indeterminateTintMode="src_in"/>
</FrameLayout>
Copy the code
The effect can be seen in a preview. The FrameLayout occupies the full screen, the ProgressBar is centered, and the size is 20 dp.
The onMeasure() method on progressbars is as follows:
@Override protected synchronized void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { int dw = 0; int dh = 0; final Drawable d = mCurrentDrawable; if (d ! = null) { dw = Math.max(mMinWidth, Math.min(mMaxWidth, d.getIntrinsicWidth())); dh = Math.max(mMinHeight, Math.min(mMaxHeight, d.getIntrinsicHeight())); } updateDrawableState(); dw += mPaddingLeft + mPaddingRight; dh += mPaddingTop + mPaddingBottom; final int measuredWidth = resolveSizeAndState(dw, widthMeasureSpec, 0); final int measuredHeight = resolveSizeAndState(dh, heightMeasureSpec, 0); setMeasuredDimension(measuredWidth, measuredHeight); }Copy the code
As you can see, the ProgressBar overwrites the View’s onMeasure() method and does not call super. So, the top set of analyses doesn’t work for it. Therefore, it also calls the setMeasuredDimension() method to make a final measurement of its own. ResolveSizeAndState () is a static method of the View.
public static int resolveSizeAndState(int size, int measureSpec, int childMeasuredState) {
final int specMode = MeasureSpec.getMode(measureSpec);
final int specSize = MeasureSpec.getSize(measureSpec);
final int result;
switch (specMode) {
case MeasureSpec.AT_MOST:
if (specSize < size) {
result = specSize | MEASURED_STATE_TOO_SMALL;
} else {
result = size;
}
break;
case MeasureSpec.EXACTLY:
result = specSize;
break;
case MeasureSpec.UNSPECIFIED:
default:
result = size;
}
return result | (childMeasuredState & MEASURED_STATE_MASK);
}
Copy the code
Size is the background size, MeasureSpec is passed in by the onMeasure() method, and parent specifies the parameter. Let’s ignore the state problem for now. We’ll focus on size. In contrast to getDefaultSize(), this method already distinguishes between AT_MOST and EXACTLY, four more cases.
Under AT_MOST, if the measurement is smaller than the background size, the View needs a size larger than the parent can give. At this point, the MEASURED_STATE_TOO_SMALL state is set to MEASURED_STATE_TOO_SMALL. If the measured value is larger than the background size, which is normal, set it to the background size. EXACTLY, that’s the measurement. UNSPECIFIED, the background size is UNSPECIFIED.
### Value pass
The parent MeasureSpec (MeasureSpec) is a MeasureSpec that measures itself in onMeasure. Does setting child match_parent or wrap_content or the exact value affect the corresponding MeasureSpec mode and size?
With these questions in mind, let’s look at a partial implementation of the onMeasure() method in FrameLayout.
@Override protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { int count = getChildCount(); // If your width and height are not exact, MeasureMatchParentChildren flag to true final Boolean measureMatchParentChildren = MeasureSpec.getMode(widthMeasureSpec) ! = MeasureSpec.EXACTLY || MeasureSpec.getMode(heightMeasureSpec) ! = MeasureSpec.EXACTLY; mMatchParentChildren.clear(); int maxHeight = 0; int maxWidth = 0; int childState = 0; for (int i = 0; i < count; i++) { final View child = getChildAt(i); if (mMeasureAllChildren || child.getVisibility() ! = GONE) {// Measure Child measureChildWithMargins(Child, widthMeasureSpec, 0, heightMeasureSpec, 0); final LayoutParams lp = (LayoutParams) child.getLayoutParams(); maxWidth = Math.max(maxWidth, child.getMeasuredWidth() + lp.leftMargin + lp.rightMargin); maxHeight = Math.max(maxHeight, child.getMeasuredHeight() + lp.topMargin + lp.bottomMargin); ChildState = combineMeasuredStates(childState, child.getMeasuredState()); // If child is match_parent but is not an exact value, That is about to measure again the if (measureMatchParentChildren) {if (lp) width = = LayoutParams) MATCH_PARENT | | lp. The height = = LayoutParams.MATCH_PARENT) { mMatchParentChildren.add(child); } } } } // Account for padding too maxWidth += getPaddingLeftWithForeground() + getPaddingRightWithForeground(); maxHeight += getPaddingTopWithForeground() + getPaddingBottomWithForeground(); // Check against our minimum height and width maxHeight = Math.max(maxHeight, getSuggestedMinimumHeight()); maxWidth = Math.max(maxWidth, getSuggestedMinimumWidth()); // Check against our foreground's minimum height and width final Drawable drawable = getForeground(); if (drawable ! = null) { maxHeight = Math.max(maxHeight, drawable.getMinimumHeight()); maxWidth = Math.max(maxWidth, drawable.getMinimumWidth()); } // Get the maximum width and height through the steps above, Call setMeasuredDimension to determine its size setMeasuredDimension(resolveSizeAndState(maxWidth, widthMeasureSpec, childState) resolveSizeAndState(maxHeight, heightMeasureSpec, childState << MEASURED_HEIGHT_STATE_SHIFT)); / / in the end, before have a child of the specified match_parent needs according to the latest wide high value measurement count = mMatchParentChildren again. The size (); if (count > 1) { for (int i = 0; i < count; i++) { final View child = mMatchParentChildren.get(i); final MarginLayoutParams lp = (MarginLayoutParams) child.getLayoutParams(); final int childWidthMeasureSpec; If (lp.width == layoutparams.match_parent) {final int width = math.max (0, getMeasuredWidth() - getPaddingLeftWithForeground() - getPaddingRightWithForeground() - lp.leftMargin - lp.rightMargin); / / match_parent status changes to precision value childWidthMeasureSpec = MeasureSpec. MakeMeasureSpec (width, MeasureSpec. EXACTLY); Childmeasurespec (widthMeasureSpec) = childMeasUrespec (widthMeasureSpec) getPaddingLeftWithForeground() + getPaddingRightWithForeground() + lp.leftMargin + lp.rightMargin, lp.width); } // Determine height code as above, omit... child.measure(childWidthMeasureSpec, childHeightMeasureSpec); }}}Copy the code
The measureChildWithMargins() method is called the first time to measure the Child, where the getChildMeasureSpec() method is called the second time to validate the width and height, This is also how the relevant MeasureSpec is determined. As you can see, **getChildMeasureSpec() is a very important static method. Child = parent = child = child = child = child = parent = child = child = child = child = parent = child = child Here, the three measurement modes are associated with match_parent wrAP_content or specific values in XML.
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) {
// Parent has imposed an exact size on us
case MeasureSpec.EXACTLY:
if (childDimension >= 0) {
resultSize = childDimension;
resultMode = MeasureSpec.EXACTLY;
} else if (childDimension == LayoutParams.MATCH_PARENT) {
// Child wants to be our size. So be it.
resultSize = size;
resultMode = MeasureSpec.EXACTLY;
} else if (childDimension == LayoutParams.WRAP_CONTENT) {
// Child wants to determine its own size. It can't be
// bigger than us.
resultSize = size;
resultMode = MeasureSpec.AT_MOST;
}
break;
// Parent has imposed a maximum size on us
case MeasureSpec.AT_MOST:
if (childDimension >= 0) {
// Child wants a specific size... so be it
resultSize = childDimension;
resultMode = MeasureSpec.EXACTLY;
} else if (childDimension == LayoutParams.MATCH_PARENT) {
// Child wants to be our size, but our size is not fixed.
// Constrain child to not be bigger than us.
resultSize = size;
resultMode = MeasureSpec.AT_MOST;
} else if (childDimension == LayoutParams.WRAP_CONTENT) {
// Child wants to determine its own size. It can't be
// bigger than us.
resultSize = size;
resultMode = MeasureSpec.AT_MOST;
}
break;
// Parent asked to see how big we want to be
case MeasureSpec.UNSPECIFIED:
if (childDimension >= 0) {
// Child wants a specific size... let him have it
resultSize = childDimension;
resultMode = MeasureSpec.EXACTLY;
} else if (childDimension == LayoutParams.MATCH_PARENT) {
// Child wants to be our size... find out how big it should
// be
resultSize = View.sUseZeroUnspecifiedMeasureSpec ? 0 : size;
resultMode = MeasureSpec.UNSPECIFIED;
} else if (childDimension == LayoutParams.WRAP_CONTENT) {
// Child wants to determine its own size.... find out how
// big it should be
resultSize = View.sUseZeroUnspecifiedMeasureSpec ? 0 : size;
resultMode = MeasureSpec.UNSPECIFIED;
}
break;
}
//noinspection ResourceType
return MeasureSpec.makeMeasureSpec(resultSize, resultMode);
}
Copy the code
The code looks like this. To make it easier to understand, I made the following table.
Parent(pSize) —— Child(size) |
EXACTLY | AT_MOST | UNSPECIFIED |
---|---|---|---|
EXACTLY | EXACTLY (size) | EXACTLY (size) | EXACTLY (size) |
MATCH_PARENT | EXACTLY (pSize) | AT_MOST (pSize) | UNSPECIFIED (pSize) |
WRAP_CONTENT | AT_MOST (pSize) | AT_MOST (pSize) | UNSPECIFIED (pSize) |
With this method, a related MeasureSpec is generated that is finally used to measure the Child. The child can then start measuring itself by calling child.measure(childWidthMeasureSpec, childHeightMeasureSpec), which eventually calls back to the child’s onMeasure() method.
The FrameLayout MeasureSpec is EXACTLY + pSize when setContentView() is loaded directly.
# # # LayoutParameter feature class
The width and height information above is taken from the LayoutParameter class. This class is so important that without it, the XML-related attributes we write cannot be translated into the corresponding code. We can use the layout_weight attribute directly in the LinearLayout, but if we change it to FrameLayout, this attribute will not work. Also, the gravity property defined in FrameLayout has no effect on the LinearLayout. Why is that? What is implemented at the code level?
So that’s what LayoutParams does, LayoutParameter is defined in a ViewGroup, it’s the top level, it has a bunch of subclasses, First subclass is MarginLayoutParams, other implementations follow specific ViewGroup, such as FrameLayout. LayoutParameter LinearLayout. LayoutParameter or RecyclerView LayoutParameter.
GenerateLayoutParams (AttributeSet attrs) method is defined in ViewGroup to generateLayoutParams
// ViewGroup
public LayoutParams generateLayoutParams(AttributeSet attrs) {
return new LayoutParams(getContext(), attrs);
}
//ViewGroup.LayoutParams
public LayoutParams(Context c, AttributeSet attrs) {
TypedArray a = c.obtainStyledAttributes(attrs, R.styleable.ViewGroup_Layout);
setBaseAttributes(a,
R.styleable.ViewGroup_Layout_layout_width,
R.styleable.ViewGroup_Layout_layout_height);
a.recycle();
}
//ViewGroup.LayoutParams
protected void setBaseAttributes(TypedArray a, int widthAttr, int heightAttr) {
width = a.getLayoutDimension(widthAttr, "layout_width");
height = a.getLayoutDimension(heightAttr, "layout_height");
}
Copy the code
With the above code, all viewGroups have the generateLayoutParams() capability. In the default ViewGroup, it only cares about the basic width and height parameters. Then compare FrameLayout and LinearLayout to see the related methods.
//FrameLayout.LayoutParams
public LayoutParams(@NonNull Context c, @Nullable AttributeSet attrs) {
super(c, attrs);
final TypedArray a = c.obtainStyledAttributes(attrs, R.styleable.FrameLayout_Layout);
gravity = a.getInt(R.styleable.FrameLayout_Layout_layout_gravity, UNSPECIFIED_GRAVITY);
a.recycle();
}
//LinearLayout.LayoutParams
public LayoutParams(Context c, AttributeSet attrs) {
super(c, attrs);
TypedArray a =
c.obtainStyledAttributes(attrs, com.android.internal.R.styleable.LinearLayout_Layout);
weight = a.getFloat(com.android.internal.R.styleable.LinearLayout_Layout_layout_weight, 0);
gravity = a.getInt(com.android.internal.R.styleable.LinearLayout_Layout_layout_gravity, -1);
a.recycle();
}
Copy the code
As you can see, gravity is resolved extra in FrameLayout and weight and gravity are resolved extra in LinearLayout.
View abnormal Cause
Go back to the layoutinflater.inflate () method during the creation of the View in the previous article.
public View inflate(XmlPullParser parser, @Nullable ViewGroup root, boolean attachToRoot) { synchronized (mConstructorArgs) { ... try { ... // Temp is the root view that was found in the xml final View temp = createViewFromTag(root, name, inflaterContext, attrs); ViewGroup.LayoutParams params = null; if (root ! = null) { // Create layout params that match root, if supplied params = root.generateLayoutParams(attrs); if (! attachToRoot) { // Set the layout params for temp if we are not // attaching. (If we are, we use addView, below) temp.setLayoutParams(params); }}... }}}Copy the code
There’s a big hole that needs to be filled in. In the layoutinflater.inflate () method, we need to specify the parent, but what happens if we do not specify the parent? LayoutParams are not created. Finally, in the addView() method:
public void addView(View child, int index) { if (child == null) { throw new IllegalArgumentException("Cannot add a null child view to a ViewGroup"); } // LayoutParams are not synchronized LayoutParams params = child.getLayOutParams (); If (params = = null) {/ / not generated, you create a default LayoutParams params = generateDefaultLayoutParams (); if (params == null) { throw new IllegalArgumentException("generateDefaultLayoutParams() cannot return null"); } } addView(child, index, params); } protected LayoutParams generateDefaultLayoutParams() { return new LayoutParams(LayoutParams.WRAP_CONTENT, LayoutParams.WRAP_CONTENT); }Copy the code
The default LayoutParams only sets the width and height, and the gravity weight properties are discarded, if your top layout of the inflate() does have these properties, sorry, they are lost. This is also an important reason why people complain that their layout style is abnormal with the inflate layout. To avoid this problem, make sure that the parent is passed in the inflate. Do not inflate(R.layout.xx, NULL), and Studio currently warns you about this inflate.
Inflate scenarios are not common, and the system passes you the parent when you Fragment or create a ViewHolder, which is easy to solve. But when using WindowManager.addView() PopupWindow Dialog, it might be difficult to find the parent. What to do at this point? You can take window.decorView or you can just new a specific ViewGroup.
That seems to be the end of LayoutParams. The big bug inflate() also seems to be fixed.
Dialog view exception
Anyone who has ever created a Dialog or DialogFragment knows that in a Dialog layout, writing match_parent has no effect and the result is always wrap_content. From the above analysis, at first I thought it was the inflate error, and then, even though I specified the corresponding parent, I assumed that the layout would work as expected. The result was the same.
Why is that?
This comes back to the getChildMeasureSpec() method and table just above. So every time we say match_parent, what’s the default size of parent? Of course, it is the size of the screen width and height. In the Dialog, it creates the corresponding PhoneWindow, and there’s a DecorView inside the PhoneWindow, and the DecorView doesn’t just add the root View of our layout, there’s an mContentParent, That’s the direct leadership that shows us how to add a View, Dad. This is the judgment when creating mContentParent in PhoneWindow.
protected ViewGroup generateLayout(DecorView decor) { if (mIsFloating) { setLayout(WRAP_CONTENT, WRAP_CONTENT); setFlags(0, flagsToUpdate); }}Copy the code
When we use dialogs of various styles, we actually load the default style. The most basic Dialog style has a default property written in it.
<style name="Base.V7.Theme.AppCompat.Light.Dialog" parent="Base.Theme.AppCompat.Light">
...
<item name="android:windowIsFloating">true</item>
...
</style>
Copy the code
Put these two pieces of code together and the problem begins to transform. When parent (decorView) is the exact value and child(mContentParent) is wrap_content, what is the final child child equivalent? Child measureSpec = AT_MOST; child measureSpec = AT_MOST; When parent (mContentParent) is AT_MOST and child (padding layout) is match_parent, what is the parent parent equivalent? So let’s go ahead and look at the table, and obviously, this is also AT_MOST + pSize. Note that this is not EXACTLY the same as the EXACTLY + pSize in the first analysis.
<FrameLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:id="@+id/parent_container"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:layout_gravity="center"
android:background="@color/colorAccent"
android:clipChildren="false">
<ProgressBar
android:id="@+id/progressbar"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_gravity="center"
android:indeterminate="true"
android:indeterminateTint="@color/colorPrimary"
android:indeterminateTintMode="src_in"/>
</FrameLayout>
Copy the code
Suppose we fill the layout in the Dialog. Combined with the above FrameLayout analysis, the size of the child should be measured again. The key is that the setMeasureDimension () method at the end of the FrameLayout onMeasure() method calls resolveSizeAndState()
setMeasuredDimension(resolveSizeAndState(maxWidth, widthMeasureSpec, childState),
resolveSizeAndState(maxHeight, heightMeasureSpec,
childState << MEASURED_HEIGHT_STATE_SHIFT));
public static int resolveSizeAndState(int size, int measureSpec, int childMeasuredState) {
final int specMode = MeasureSpec.getMode(measureSpec);
final int specSize = MeasureSpec.getSize(measureSpec);
final int result;
switch (specMode) {
case MeasureSpec.AT_MOST:
if (specSize < size) {
result = specSize | MEASURED_STATE_TOO_SMALL;
} else {
result = size;
}
break;
case MeasureSpec.EXACTLY:
result = specSize;
break;
case MeasureSpec.UNSPECIFIED:
default:
result = size;
}
return result | (childMeasuredState & MEASURED_STATE_MASK);
}
Copy the code
The first time is EXACTLY, so pSize. This time it’s AT_MOST, so it’s childSize. So the final effect is actually wrap_content. At this point Dialog shows that the exception is parsed from the code. So what’s the solution? First, you can set the windowIsFloating to false at the source.
//styles.xml <style name="AppTheme.AppCompat.Dialog.Alert.NoFloating" parent="Theme.AppCompat.Light.Dialog.Alert"> <item name="android:windowIsFloating">false</item> </style> //DialogFragment override fun onCreate(savedInstanceState: Bundle?) { super.onCreate(savedInstanceState) setStyle(android.support.v4.app.DialogFragment.STYLE_NO_TITLE, R.style.AppTheme_AppCompat_Dialog_Alert) }Copy the code
The next best thing, since it defaults to wrAP_content, we can just set it back.
//DialogFragment override fun onStart() { super.onStart() dialog? .window? .setLayout(LayoutParams.MATCH_PARENT, LayoutParams.MATCH_PARENT) }Copy the code
Here, we can also answer a question if parent is specified as wrap_content. Child is match_parent and then finally, how big is the child? This is actually the above question, if the answer is simple, then it is the minimum value of View itself.
To be more specific, if the View does not replicate the onMeasure() method, that is the return value of getDefaultSize() in the default onMeasure() method, which is pSize.
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
Other controls, such as the ProgressBar, are either resolveSizeAndState() or the smallest measured value. When we customize the View, the View preview always fills the parent layout because you didn’t override the onMeasure() method. In addition, when writing the layout, try to avoid the situation where parent is wrap_content and child is match_parent, so that parent will double measure and cause unnecessary overhead.
conclusion
The measurement of View is a game process, and the most important method is setMeasuredDimension(). The specific value must be negotiated between parent and child. The value is passed and determined depending on MeasureSpec and LayoutParams. The root parameter of the inflate() method should not be empty when filling the layout, as this will cause some of the parameters of the inflate layout to be lost, and the Dialog is always wrAP_content, This is because the default has the property “windowIsFloating”.