After we have learned the measurement, layout and drawing mechanism of Android View, we will analyze the invalidate of View redrawing and update requestLayout in detail
The phenomenon of
public class CustomEmptyView extends View {
public CustomEmptyView(Context context) {
super(context);
}
public CustomEmptyView(Context context, @Nullable AttributeSet attrs) {
super(context, attrs);
}
public CustomEmptyView(Context context, @Nullable AttributeSet attrs, int defStyleAttr) {
super(context, attrs, defStyleAttr);
}
@Override
protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
super.onMeasure(widthMeasureSpec, heightMeasureSpec);
Log.i("CustomEmptyView"."onMeasure");
}
@Override
protected void onLayout(boolean changed, int left, int top, int right, int bottom) {
super.onLayout(changed, left, top, right, bottom);
Log.i("CustomEmptyView"."onLayout");
}
@Override
protected void onDraw(Canvas canvas) {
super.onDraw(canvas);
Log.i("CustomEmptyView"."onDraw"); }}Copy the code
According to the drawing mechanism of View, View#onMeasure(), View#onLayout() and View#onDraw() will be executed in the measurement, layout and drawing steps of View. What about View#invalidate() and View#requestLayout()? Let’s print out the data.
View#invalidate() performs the following steps:
The 2019-03-26 17:32:34. 739, 8075-8075 / com. Example. Myapplication I/CustomEmptyView: ontouchCopy the code
View#requestLayout() performs the following steps:
The 2019-03-26 17:33:13. 497, 8075-8075 / com. Example. Myapplication I/CustomEmptyView: OnMeasure 17:33:13. 2019-03-26, 501, 8075-8075 / com. Example. Myapplication I/CustomEmptyView: OnLayout 17:33:13. 2019-03-26, 503, 8075-8075 / com. Example. Myapplication I/CustomEmptyView: ontouchCopy the code
View#invalidate() will only execute View#onDraw(); View#requestLayout() redirects the View’s drawing process. Let’s analyze it from the source point of view. The following source code analysis is based on Android-28
View#requestLayout()
Let’s analyze View#requestLayout(). We locate the corresponding source code
/**
* Call this when something has changed which has invalidated the
* layout of this view. This will schedule a layout pass of the view
* tree. This should not be called while the view hierarchy is currently in a layout
* pass ({@link #isInLayout()}. If layout is happening, the request may be honored at the
* end of the current layout pass (and thenLayout will run again) or after the current * frame is drawn and the next layout occurs. * * This function is called when something changes that causes the layout of the view to be redrawn. This arranges the layout passing of the view tree. * This function is not executed when the view hierarchy is currently in a Layout Layout event. * If a Layout is in progress, the request can be executed when the current Layout passes (and then the Layout runs again) or after the current frame is drawn and the next Layout is executed. * <p>Subclasseswhich override this method should call the superclass method to
* handle possible request-during-layout errors correctly.</p>
*/
@CallSuper
public void requestLayout() {
if(mMeasureCache ! = null) mMeasureCache.clear();if(mAttachInfo ! = null && mAttachInfo.mViewRequestingLayout == null) { // Only trigger request-during-layout logicif this is the view requesting it,
// not the views inits parent hierarchy ViewRootImpl viewRoot = getViewRootImpl(); // This method is not executed during layout eventsif(viewRoot ! = null && viewRoot.isInLayout()) {if(! viewRoot.requestLayoutDuringLayout(this)) {return; } } mAttachInfo.mViewRequestingLayout = this; } // PFLAG_FORCE_LAYOUT is determined when executing the View's measure() and layout() methods, as seen in previous articles. // But in the current source code of view. class and viewrootimpl. class, the global search PFLAG_FORCE_LAYOUT, there is no direct judgment, resulting in View#requestLayout() does not perform measurement and layout methodsmPrivateFlags |= PFLAG_FORCE_LAYOUT; mPrivateFlags |= PFLAG_INVALIDATED; // isLayoutRequested() corresponds to the mLayoutRequested field, which defaults tofalse
if(mParent ! = null && ! MParent. IsLayoutRequested ()) {/ / the implementation of the parent container requestLayout () method mParent. RequestLayout (). }if (mAttachInfo != null && mAttachInfo.mViewRequestingLayout == this) {
mAttachInfo.mViewRequestingLayout = null;
}
}
Copy the code
When we click on ViewGroup#requestLayout(), it is an empty implementation. We know that ViewParent is an interface class. See the ViewRootImpl class for an implementation of ViewGroup#requestLayout().
@Override
public void requestLayout() {/ / mHandlingLayoutInLayoutRequest this parameter by global variable orientation, in performLayout () at the beginningtrue, when the end isfalse, which corresponds to not executing during layoutif(! MHandlingLayoutInLayoutRequest) {/ / check whether the UI thread checkThread (); // Here mLayoutRequested is set totrue
mLayoutRequested = true;
scheduleTraversals();
}
}
void scheduleTraversals() {
if(! mTraversalScheduled) { mTraversalScheduled =true;
mTraversalBarrier = mHandler.getLooper().getQueue().postSyncBarrier();
mChoreographer.postCallback(
Choreographer.CALLBACK_TRAVERSAL, mTraversalRunnable, null);
if (!mUnbufferedInputDispatch) {
scheduleConsumeBatchedInput();
}
notifyRendererOfFramePending();
pokeDrawLockIfNeeded();
}
}
Copy the code
When we see scheduleTraversals(), we are sure that we are getting close to the truth, but when we click on the implementation of this method, we do not find the desired traversals. View#requestLayout() will execute the View’s rendering step again. The core of the drawing step of a View is View action #performTraverals, so we continue to look for it in this way.
From the source code above, we can see that mChoreographer is an object. We have mentioned Choreographer as an object when we were analyzing the degree of sliding flow, We finally take our cue from the mTraversalRunnable parameter in the object mChoreographer.
final class TraversalRunnable implements Runnable {
@Override
public void run() {
doTraversal();
}
}
final TraversalRunnable mTraversalRunnable = new TraversalRunnable();
void doTraversal() {
if (mTraversalScheduled) {
mTraversalScheduled = false;
mHandler.getLooper().getQueue().removeSyncBarrier(mTraversalBarrier);
if (mProfile) {
Debug.startMethodTracing("ViewAncestor");
}
performTraversals();
if (mProfile) {
Debug.stopMethodTracing();
mProfile = false; }}}Copy the code
Finally we find that mTraversalRunnable which is a Runnable object, and passing mTraversalRunnable in scheduleTraversals() will execute doduleTraversal (), In doTraversal() we also found the core method we wanted viewrotimpl #performTraverals(). The measurement, layout, and drawing steps of the control restart when viewrotimpl #performTraverals is called. It matches the data we printed at the beginning.
View#invalidate()
Next, let’s look at the source code for View#invalidate.
/** * Invalidate the whole view. If the view is visible, * redraw the whole view If it is visible. * {@link#onDraw(android.graphics.Canvas)} will be called at some point in
* the future.
* <p>
* This must be called from a UI thread. To call from a non-UI thread, call
* {@link #postInvalidate()}.* This method must be used on the UI thread. For non-UI threads, postInvalidate() */ public voidinvalidate() {
invalidate(true);
}
/**
* This is where the invalidate() work actually happens. A full invalidate()
* causes the drawing cache to be invalidated, but this function can be
* called with invalidateCache set to false to skip that invalidation step
* for cases that do not need it (forExample, a component that remains at * the same dimensions with the same content). A full invalidate() method invalidates the draw * cache, but this function sets the parameter invalidateCache tofalseTo skip the redraw step, since * this method is not required, e.g. a component keeps the same size and the same contents * * @param invalidateCache Whether the drawing cachefor this view should be
* invalidated as well. This is usually true for a full
* invalidate, but may be set to false if the View's contents or * dimensions have not changed. * Whether the draw cache should be redrawn. A full redraw is usually true, but may be set to false if the View's contents and dimensions have not been changed. * * @hide */ public void invalidate(boolean invalidateCache) { invalidateInternal(0, 0, mRight - mLeft, mBottom - mTop, invalidateCache, true); } void invalidateInternal(int l, int t, int r, int b, Boolean invalidateCache, Boolean fullInvalidate) { If (mGhostView! = null) { mGhostView.invalidate(true); return; } // jump redraw. If (skipInvalidate()) {return; if (skipInvalidate()) {return; } // A large number of parameters are compared here, which should be the judgment of the above mentioned coordinate position has not changed, If a change is identified as need to redraw the if ((mPrivateFlags & (PFLAG_DRAWN | PFLAG_HAS_BOUNDS)) = = (PFLAG_DRAWN | PFLAG_HAS_BOUNDS) | | (invalidateCache && (mPrivateFlags & PFLAG_DRAWING_CACHE_VALID) == PFLAG_DRAWING_CACHE_VALID) || (mPrivateFlags & PFLAG_INVALIDATED) ! = PFLAG_INVALIDATED || (fullInvalidate && isOpaque() ! = mLastIsOpaque)) { if (fullInvalidate) { mLastIsOpaque = isOpaque(); mPrivateFlags &= ~PFLAG_DRAWN; } mPrivateFlags |= PFLAG_DIRTY; // If you need to redraw everything, Invalidate () not preach and call the default value is true if (invalidateCache) {/ / remember the PFLAG_INVALIDATED marks a mPrivateFlags | = PFLAG_INVALIDATED; mPrivateFlags &= ~PFLAG_DRAWING_CACHE_VALID; } // Propagate the damage rectangle to the parent view. Final AttachInfo ai = mAttachInfo; final ViewParent p = mParent; if (p ! = null && ai ! = null && l < r && t < b) { final Rect damage = ai.mTmpInvalRect; damage.set(l, t, r, b); P.invalidatechild (this, damage); } // Damage the entire projection receiver, if necessary. if (mBackground ! = null && mBackground.isProjected()) { final View receiver = getProjectionReceiver(); if (receiver ! = null) { receiver.damageInParent(); }}}}Copy the code
As you can see from the above analysis, the overloaded invalidate() method will eventually call invalidateInternal(). In this method, the View needs to be redrawn, and if it needs to be redrawn, the View will be identified. The View’s Rect information is then passed to the parent container’s invalidateChild().
Similar to the previous View#requestLayout(), I ended up doing ViewRootImpl#invalidateChild(), and then I went on to analyze the implementation of ViewRootImpl#invalidateChild().
@Override public void invalidateChild(View child, Rect dirty) { invalidateChildInParent(null, dirty); } // Since there are no comments, we parse the source from the method name, overloading the final method, @override public ViewParent invalidateChildInParent(int[] location, Rect dirty) { The method needs to be executed in the UI thread, which validates the View before itThe description of the # invalidate ()
checkThread();
if (DEBUG_DRAW) Log.v(mTag, "Invalidate child: "+ dirty); // As you can see from the previous step, dirty is not emptyif (dirty == null) {
invalidate();
return null;
} else if(dirty.isEmpty() && ! mIsAnimating) {returnnull; } // Here should be some coordinate position Settings assignmentif(mCurScrollY ! = 0 || mTranslator ! = null) { mTempRect.set(dirty); dirty = mTempRect;if(mCurScrollY ! = 0) { dirty.offset(0, -mCurScrollY); }if(mTranslator ! = null) { mTranslator.translateRectInAppWindowToScreen(dirty); }if(mAttachInfo.mScalingRequired) { dirty.inset(-1, -1); }} // From the name of the method, here should be the method we need, redraw invalidateRectOnScreen(dirty);return null;
}
private void invalidateRectOnScreen(Rect dirty) {
final Rect localDirty = mDirty;
if (!localDirty.isEmpty() && !localDirty.contains(dirty)) {
mAttachInfo.mSetIgnoreDirtyState = true;
mAttachInfo.mIgnoreDirtyState = true;
}
// Add the new dirty rect to the current one
localDirty.union(dirty.left, dirty.top, dirty.right, dirty.bottom);
// Intersect with the bounds of the window to skip
// updates that lie outside of the visible region
final float appScale = mAttachInfo.mApplicationScale;
final boolean intersected = localDirty. Intersect (0, 0, (int) (mWidth * appScale + 0.5f), (int) (mHeight * appScale + 0.5f));if(! intersected) {localDirty.setEmpty();
}
if(! MWillDrawSoon && (intersected | | mIsAnimating)) {/ / the setting and adjustment, eventually call scheduleTraversals scheduleTraversals () (); } } voidscheduleTraversals() {
if(! mTraversalScheduled) { mTraversalScheduled =true;
mTraversalBarrier = mHandler.getLooper().getQueue().postSyncBarrier();
mChoreographer.postCallback(
Choreographer.CALLBACK_TRAVERSAL, mTraversalRunnable, null);
if (!mUnbufferedInputDispatch) {
scheduleConsumeBatchedInput();
}
notifyRendererOfFramePending();
pokeDrawLockIfNeeded();
}
}
Copy the code
View#requestLayout() is the same as scheduleTraversals(), but View#invalidate() does not perform measurement and traversals. But so far they end up calling the same method.
Note that the ViewRootImpl#requestLayout() method takes the initiative to set a global variable mLayoutRequested to true; PerformMeasure () and performLayout() will be affected by the traversals.
private void performTraversals() {··· Boolean layoutRequested = mLayoutRequested && (! mStopped || mReportNextDraw); ··· Boolean windowShouldResize = layoutRequested && windowSizeMayChange && ((mWidth! = host.getMeasuredWidth() || mHeight ! = host.getMeasuredHeight()) || (lp.width == ViewGroup.LayoutParams.WRAP_CONTENT && frame.width() < desiredWindowWidth && frame.width() ! = mWidth) || (lp.height == ViewGroup.LayoutParams.WRAP_CONTENT && frame.height() < desiredWindowHeight && frame.height() ! = mHeight)); ...if(mFirst || windowShouldResize || insetsChanged || viewVisibilityChanged || params ! = null | | mForceNextWindowRelayout) {... performMeasure (childWidthMeasureSpec childHeightMeasureSpec); ··· ··· · final Boolean didLayout = layoutRequested && (! mStopped || mReportNextDraw); boolean triggerGlobalLayoutListener = didLayout || mAttachInfo.mRecomputeGlobalAttributes;if(didLayout) { performLayout(lp, mWidth, mHeight); }...}Copy the code
For the complex viewrooP Button #performTraversals(), we extract some key code that does verify that the mLayoutRequested object affects the methods that correspond to the measurements and layouts, and so verify that we first print the data, View#invalidate() does not perform measurement and layout.
View#postInvalidate()
View#invalidate() needs to be executed in the UI thread. If View#postInvalidate() needs to be used in a non-ui thread, let’s briefly analyze the source code.
public void postInvalidate() {
postInvalidateDelayed(0);
}
public void postInvalidate(int left, int top, int right, int bottom) {
postInvalidateDelayed(0, left, top, right, bottom);
}
public void postInvalidateDelayed(long delayMilliseconds, int left, int top,
int right, int bottom) {
// We try only with the AttachInfo because there's no point in invalidating // if we are not attached to our window final AttachInfo attachInfo = mAttachInfo; if (attachInfo ! = null) { final AttachInfo.InvalidateInfo info = AttachInfo.InvalidateInfo.obtain(); info.target = this; info.left = left; info.top = top; info.right = right; info.bottom = bottom; attachInfo.mViewRootImpl.dispatchInvalidateRectDelayed(info, delayMilliseconds); }}Copy the code
ViewRootImpl#dispatchInvalidateRectDelayed()
public void dispatchInvalidateRectDelayed(AttachInfo.InvalidateInfo info, long delayMilliseconds) {
final Message msg = mHandler.obtainMessage(MSG_INVALIDATE_RECT, info);
mHandler.sendMessageDelayed(msg, delayMilliseconds);
}
@Override
public void handleMessage(Message msg) {
switch (msg.what) {
case MSG_INVALIDATE_RECT:
final View.AttachInfo.InvalidateInfo info = (View.AttachInfo.InvalidateInfo) msg.obj;
info.target.invalidate(info.left, info.top, info.right, info.bottom);
info.recycle();
break; ...}}Copy the code
View#postInvalidate() sends an MSG_INVALIDATE_RECT Handler message. After receiving the message, the View#invalidate() method is also executed.
RequestLayout does not execute performDraw()?
View#requestLayout does not execute performDraw(), but its printout results will show that performDraw() is executed. Let’s go through the source code with the question. Navigate directly to the performDraw() of view script #performTraversals
private void performTraversals() {··· // dispatchOnPreDraw() returns comments that are: True if the current draw should be cancelled and rescheduled, otherwisefalse. // final boolean isViewVisible = viewVisibility == View.VISIBLE; Whenever the View is displayed, cancelDraw istrueboolean cancelDraw = mAttachInfo.mTreeObserver.dispatchOnPreDraw() || ! isViewVisible; // cancelDraw (dispatchOnPreDraw())false// newSurface by defaultfalseIn the measurement judgment logic, it is determined whether the new Surface will be set astrueI think it's going to befalse// Because performDraw() is executedif(! cancelDraw && ! newSurface) {if(mPendingTransitions ! = null && mPendingTransitions.size() > 0) {for(int i = 0; i < mPendingTransitions.size(); ++i) { mPendingTransitions.get(i).startChangingAnimations(); } mPendingTransitions.clear(); } performDraw(); }}Copy the code
ViewRootImpl#performDraw()
private void performDraw() {// display. STATE_OFF Indicates Display status: Display is disabled. // mReportNextDraw object defaultfalse, can be in ViewRootImplSet to true in #reportNextDraw(), but the first judgment is already false
if(mAttachInfo.mDisplayState == Display.STATE_OFF && ! mReportNextDraw) {return;
} else if(mView == null) {// Must not be nullreturn;
}
final boolean fullRedrawNeeded = mFullRedrawNeeded || mReportNextDraw;
mFullRedrawNeeded = false;
mIsDrawing = true;
Trace.traceBegin(Trace.TRACE_TAG_VIEW, "draw");
boolean usingAsyncReport = false;
if(mReportNextDraw && mAttachInfo.mThreadedRenderer ! = null && mAttachInfo.mThreadedRenderer.isEnabled()) { usingAsyncReport =true; mAttachInfo.mThreadedRenderer.setFrameCompleteCallback((long frameNr) -> { // TODO: Use the frame number pendingDrawFinished(); }); } draw() {draw(); draw(); draw()returnLogic. boolean canUseAsync = draw(fullRedrawNeeded);if(usingAsyncReport && ! canUseAsync) { mAttachInfo.mThreadedRenderer.setFrameCompleteCallback(null); usingAsyncReport =false;
}
} finally {
mIsDrawing = false; Trace.traceEnd(Trace.TRACE_TAG_VIEW); }...}Copy the code
Draw () = draw() = draw() = draw() = draw() = draw()
conclusion
View drawing steps are performMeasure(), performLayout(), performDraw(), we through the invalidate and requestLayout source analysis, can be drawn. The invalidate() method only executes the performDraw() method; The requestLayout() method executes performMeasure(), performLayout(), and performDraw(). Invalidate () is used if the View content changes without changing the size or location of the View, and requestLayout() is called if the size, location, or content changes.