One, the introduction

About the content of this article for which students, can first ask a few questions please consider.

1. What type of context is retrieved from an ImageView view.getContext()?

2. What’s the difference between calling setContentView() and addContentView() in an Activity?

What is the difference between AppCompatActivity and Activity’s setContentView() method?

4. How does Android read and build views (or create a View) from XML?

5. How to achieve compatibility and extension for different versions of APIS in support-V7 package?

6. How can AppCompatActivity replace some basic types of controls with AppCompatXXX controls? Which controls will be replaced? When do you replace it?

If you are confused or unsure about the above questions, you can find the answers below. We will start with the problems found during project debugging and gradually analyze the causes. Therefore, this article may be a bit long. If you are worried, you can find the content you care about according to the title.

In addition, the source version of this article is: Android-25, support-V7-25.4.0

In order to simplify reading, the “irrelevant” code in this article will be omitted, so if you need to follow the clues given in this article, you can see all the source code.

1.1 View.getContext()

Context context = imageView.getContext(); if (context instanceof Activity) { Activity activity = (Activity)context; / /... } Duplicate codeCopy the code

As you can see from the code example above, we get the context from the imageView control and turn it into an Activity to continue the operation. The imageView is a control from the XML layout, but some phones do not go into the if branch of the transformation type when the actual project is running, indicating that the context is not of Activity type. This is weird. Why?

/** * Simple constructor to use when creating a view from code. * * @param context The Context the view is running in, through which it can * access the current theme, resources, etc. */ public View(Context context) { mContext = context; / /... Omit}. @ ViewDebug CapturedViewProperty public final Context getContext () {return mContext; } Duplicate codeCopy the code

The view.getContext () method returns the mContext member variable, and the mContext assignment is only in the constructor. This imageView is written in XML, and when setContextView(r.layout.xxx) is called, the PhoneWindow setContextView() method is called. The context used for the build should be an Activity.

At this time, I went back to Debug again carefully, and found that the problems were in the phone below 5.0. So the above impression is problematic. Below 5.0, the imageView.getContext() gets a context of type TintContextWrapper instead of the Activity type I initially thought it would be.

1.2 the Context type

What Wrapper is this TintContextWrapper? I don’t remember this in Context inheritance. About the Context type www.jianshu.com/p/94e0f9ab3… If you are not clear, you can search for it by yourself. Here you will not expand it. There are many things that can be clearly explained on the Internet.

 

 

 

There is no TintContextWrapper. It is also a Wrapper Context and is related to Tint. The imageView type is not the imageView type defined in XML, but the AppCompatImageView type.

As you can see, the Activity on which the control is based inherits from AppCompatActivity, so any change to the context type must be related to AppCompatActivity in the v7 package. Before the so-called impression has made two mistakes, why not read the source code to clarify?

Note: the following article is not entirely in accordance with the order of the problem, but after reading the relevant source code, sorted out the relevant knowledge points. Those who are already clear can pick and read.

The difference between setContentView() and addContentView() in an Activity

If setContentView() is called multiple times, the mContentParent container will be emptied each time thereafter. Then assemble the resource layoutResID.

If addContentView() is called multiple times, the View will be added to the mContentParent container each time thereafter. The result is a View overlay.

This mContentParent exists in the PhoneWindow.

// This is the view in which the window contents are placed. It is either // mDecor itself, or a child of mDecor where the contents go. ViewGroup mContentParent; Copy the codeCopy the code

Compatactivity setContentView() method for compatActivity.

The setContentView() method has two classes, one of which requires an XML layout ID and the other requires a View type.

setContentView(@LayoutRes int layoutResID)

setContentView(View view)

Here we discuss code with arguments of type View.

3.1 the Activity

3.1.1 Activity. The setContentView ()

Public void setContentView(View View) {getWindow().setContentView(View); initWindowDecorActionBar(); } public Window getWindow() { return mWindow; } Duplicate codeCopy the code

The setContentView() code in the Activity gets the Window to setContentView().

Public void setContentView(View View); Copy the codeCopy the code

And this window is actually a PhoneWindow, so look at the code below.

// Final void attach(Context Context, ActivityThread aThread, Instrumentation instr, IBinder token, int ident, Application application, Intent intent, ActivityInfo info, CharSequence title, Activity parent, String id, NonConfigurationInstances lastNonConfigurationInstances, Configuration config, String referrer, IVoiceInteractor voiceInteractor, Window window) { //... Omit mWindow = new PhoneWindow(this, window); / /... Omit} copy codeCopy the code

3.1.2 PhoneWindow. The setContentView ()

@Override public void setContentView(View view) { setContentView(view, new ViewGroup.LayoutParams(MATCH_PARENT, MATCH_PARENT)); } @Override public void setContentView(View view, ViewGroup.LayoutParams params) { // Note: FEATURE_CONTENT_TRANSITIONS may be set in the process of installing the window // decor, when theme attributes and the like are crystalized. Do not check the feature // before this happens. if (mContentParent == null) { installDecor(); } else if (! hasFeature(FEATURE_CONTENT_TRANSITIONS)) { mContentParent.removeAllViews(); } if (hasFeature(FEATURE_CONTENT_TRANSITIONS)) { view.setLayoutParams(params); final Scene newScene = new Scene(mContentParent, view); transitionTo(newScene); } else { mContentParent.addView(view, params); } / /... Omit} copy codeCopy the code

At line 12, make sure the mContentParent has been initialized.

At line 14, if there is no FEATURE_CONTENT_TRANSITIONS, empty the mContentParent contents.

Line 22, mContentParent adds the view as the child.

At line 17, if there is FEATURE_CONTENT_TRANSITIONS, call transitionTo(newScene). This part is not expanded, and eventually the following code is called, with the same logical steps.

View public Scene(ViewGroup sceneRoot, ViewGroup sceneRoot, ViewGroup sceneRoot, ViewGroup sceneRoot) View layout) { mSceneRoot = sceneRoot; mLayout = layout; } public void enter() { // Apply layout change, if any if (mLayoutId > 0 || mLayout ! = null) { // empty out parent container before adding to it getSceneRoot().removeAllViews(); if (mLayoutId > 0) { LayoutInflater.from(mContext).inflate(mLayoutId, mSceneRoot); } else { mSceneRoot.addView(mLayout); }} / /... Omit} copy codeCopy the code

3.2 AppCompatActivity

As you can see, the setContentView() flow in the Activity is relatively simple and basically calls the corresponding method in the PhoneWindow. Let’s see what’s special about AppCompatActivity.

3.2.1 AppCompatActivity. The setContentView () method

// AppCompatActivity @Override public void setContentView(@LayoutRes int layoutResID) { getDelegate().setContentView(layoutResID); } /** * @return The {@link AppCompatDelegate} being used by this Activity. */ @NonNull public AppCompatDelegate getDelegate() { if (mDelegate == null) { mDelegate = AppCompatDelegate.create(this, this); } return mDelegate; } Duplicate codeCopy the code

The mDelegate is a proxy class that can be built on compatdelegate to generate different actual execution classes based on different SDK versions, which is a compatibility mode for the proxy. Look at the following code:

/** * Create a {@link android.support.v7.app.AppCompatDelegate} to use with {@code activity}. * * @param callback An optional callback for AppCompat specific events */ public static AppCompatDelegate create(Activity activity, AppCompatCallback callback) { return create(activity, activity.getWindow(), callback); } private static AppCompatDelegate create(Context context, Window window, AppCompatCallback callback) { final int sdk = Build.VERSION.SDK_INT; if (BuildCompat.isAtLeastN()) { return new AppCompatDelegateImplN(context, window, callback); } else if (sdk >= 23) { return new AppCompatDelegateImplV23(context, window, callback); } else if (sdk >= 14) { return new AppCompatDelegateImplV14(context, window, callback); } else if (sdk >= 11) { return new AppCompatDelegateImplV11(context, window, callback); } else { return new AppCompatDelegateImplV9(context, window, callback); }} Copy the codeCopy the code

We can see that the most basic version of AppCompatDelegateImplV9 is the AppCompatDelegateImplV9 class that all other implementation classes ultimately inherit from. The methods we will look at later are in the AppCompatDelegateImplV9 class implementation.

So we call the setContentView() method on AppCompatActivity, and the actual implementation is AppCompatDelegateImplV9.

3.2.2 AppCompatDelegateImplV9. The setContentView () method.

SetContentView () @override public void setContentView(View v, ViewGroup.LayoutParams lp) { ensureSubDecor(); ViewGroup contentParent = (ViewGroup) mSubDecor.findViewById(android.R.id.content); contentParent.removeAllViews(); contentParent.addView(v, lp); mOriginalWindowCallback.onContentChanged(); } Duplicate codeCopy the code

From lines 5-7, retrieve the Android.r.I.C.tent flag contentParent from mSubDector (type ViewGroup) and add the View again. Line 8 callback notification.

The fourth line of code, as the name suggests, is the method that ensures that the mSubDector is initialized. Let’s take a look inside:

private void ensureSubDecor() { if (! mSubDecorInstalled) { mSubDecor = createSubDecor(); / /... Omit... }} Copy the codeCopy the code
private ViewGroup createSubDecor() { //... Omit... Now let's make sure that the Window has installed its decor by retrieving it mWindow.getDecorView(); final LayoutInflater inflater = LayoutInflater.from(mContext); ViewGroup subDecor = null; / /... Omit... If (subDecor == null) {throw new IllegalArgumentException(subDecor == null) {throw new IllegalArgumentException "AppCompat does not support the current theme features: { " + "windowActionBar: " + mHasActionBar + ", windowActionBarOverlay: "+ mOverlayActionBar + ", android:windowIsFloating: " + mIsFloating + ", windowActionModeOverlay: " + mOverlayActionMode + ", windowNoTitle: " + mWindowNoTitle + " }"); } / /... Omit... // Make the decor optionally fit system windows, like the window's decor ViewUtils.makeOptionalFitsSystemWindows(subDecor); final ContentFrameLayout contentView = (ContentFrameLayout) subDecor.findViewById( R.id.action_bar_activity_content); final ViewGroup windowContentView = (ViewGroup) mWindow.findViewById(android.R.id.content); if (windowContentView ! = null) { // There might be Views already added to the Window's content view so we need to // migrate them to our content view while (windowContentView.getChildCount() > 0) { final View child = windowContentView.getChildAt(0); windowContentView.removeViewAt(0); contentView.addView(child); } // Change our content FrameLayout to use the android.R.id.content id. // Useful for fragments. windowContentView.setId(View.NO_ID); contentView.setId(android.R.id.content); // The decorContent may have a foreground drawable set (windowContentOverlay). // Remove this as we handle it ourselves if (windowContentView instanceof FrameLayout) { ((FrameLayout) windowContentView).setForeground(null); } } // Now set the Window's content view with the decor mWindow.setContentView(subDecor); / /... Omit... return subDecor; } Duplicate codeCopy the code

Let’s focus on lines 28-31. FrameLayout from R.I.D.A. ction_bar_activity_Content was taken out of subDecor. Take the familiar Android.r.I.C.tent tag view from the window. This view is actually the contentView in the PhoneWindow DecorView.

Lines 35 to 38 move the existing ChildView from the Window to the contentView from the subDector, And empty the windowContentView. This is the first step in the transition.

Next, replace the ID of the windowContentView in the original Window (Android.r.i. D.c ontent) with the contentView in the subDecor.

Line 54, the last step in the leopard cat decor, sets the leopard cat subDecor to mWindow.

After analyzing the above code, let’s go back to line 4 of the setContentView() method and see why we can retrieve the “root View” via Android.r.i.C.tent.

@Override public void setContentView(View v, ViewGroup.LayoutParams lp) { ensureSubDecor(); ViewGroup contentParent = (ViewGroup) mSubDecor.findViewById(android.R.id.content); contentParent.removeAllViews(); contentParent.addView(v, lp); mOriginalWindowCallback.onContentChanged(); } Duplicate codeCopy the code

How to read and build a View from XML?

Now that we’ve discussed a class of setContentView() methods that take a View, let’s look at another setContentView() method that takes a layout ID.

4.1 LayoutInflater. Inflate () method

When we call setContentView(r.layout.xxx) in the Activity onCreate() method to set up a page, we end up with a method like this:

LayoutInflater.from(mContext).inflate(resId, contentParent);

So let’s look at how you inflate a page out.

Public View inflate(@layoutres int resource, @Nullable ViewGroup root) {return inflate(resource, root, root ! = null); } public View inflate(@LayoutRes int resource, @Nullable ViewGroup root, boolean attachToRoot) { final Resources res = getContext().getResources(); if (DEBUG) { Log.d(TAG, "INFLATING from resource: \"" + res.getResourceName(resource) + "\" (" + Integer.toHexString(resource) + ")"); } final XmlResourceParser parser = res.getLayout(resource); try { return inflate(parser, root, attachToRoot); } finally { parser.close(); }} Copy the codeCopy the code

Look at line 13 of the code, using the XML parser XmlResourceParser to parse the layout file we passed in. Let’s post the method details on line 14 below.

public View inflate(XmlPullParser parser, @Nullable ViewGroup root, boolean attachToRoot) { synchronized (mConstructorArgs) { Trace.traceBegin(Trace.TRACE_TAG_VIEW, "inflate"); final Context inflaterContext = mContext; final AttributeSet attrs = Xml.asAttributeSet(parser); Context lastContext = (Context) mConstructorArgs[0]; mConstructorArgs[0] = inflaterContext; View result = root; try { // Look for the root node. int type; while ((type = parser.next()) ! = XmlPullParser.START_TAG && type ! = XmlPullParser.END_DOCUMENT) { // Empty } if (type ! = XmlPullParser.START_TAG) { throw new InflateException(parser.getPositionDescription() + ": No start tag found!" ); } final String name = parser.getName(); if (DEBUG) { System.out.println("**************************"); System.out.println("Creating root view: " + name); System.out.println("**************************"); } if (TAG_MERGE.equals(name)) { if (root == null || ! attachToRoot) { throw new InflateException("<merge /> can be used only with a valid " + "ViewGroup root and attachToRoot=true"); } rInflate(parser, root, inflaterContext, attrs, false); } else { // 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) { if (DEBUG) { System.out.println("Creating params from root: " + root); } // 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); } } if (DEBUG) { System.out.println("-----> start inflating children"); } // Inflate all children under temp against its context. rInflateChildren(parser, temp, attrs, true); if (DEBUG) { System.out.println("-----> done inflating children"); } // We are supposed to attach all the views we found (int temp) // to root. Do that now. if (root ! = null && attachToRoot) { root.addView(temp, params); } // Decide whether to return the root that was passed in or the // top view found in xml. if (root == null || ! attachToRoot) { result = temp; } } } catch (XmlPullParserException e) { final InflateException ie = new InflateException(e.getMessage(), e); ie.setStackTrace(EMPTY_STACK_TRACE); throw ie; } catch (Exception e) { final InflateException ie = new InflateException(parser.getPositionDescription() + ": " + e.getMessage(), e); ie.setStackTrace(EMPTY_STACK_TRACE); throw ie; } finally { // Don't retain static reference on context. mConstructorArgs[0] = lastContext; mConstructorArgs[1] = null; Trace.traceEnd(Trace.TRACE_TAG_VIEW); } return result; }} Copy the codeCopy the code

As you can see, the above code is not very much, mainly according to the XML tags ( encapsulated content), using parser and corresponding processing.

Line 74 adds view to root. The root is the original contentParent (type ViewGroup).

So there’s a little bit of a problem, so if you read the tag, you know what the tag is, like a TextView, where do you create a View?

Lines 41-42 call the createViewFromTag() method to create the View’s.

// Temp is the root view that was found in the xml

final View temp = createViewFromTag(root, name, inflaterContext, attrs);

4.2 createViewFromTag () method

Let’s simplify some of the code.

View createViewFromTag(View parent, String name, Context Context, AttributeSet attrs, boolean ignoreThemeAttr) { //... Omit... try { View view; if (mFactory2 ! = null) { view = mFactory2.onCreateView(parent, name, context, attrs); } else if (mFactory ! = null) { view = mFactory.onCreateView(name, context, attrs); } else { view = null; } if (view == null && mPrivateFactory ! = null) { view = mPrivateFactory.onCreateView(parent, name, context, attrs); } if (view == null) { final Object lastContext = mConstructorArgs[0]; mConstructorArgs[0] = context; try { if (-1 == name.indexOf('.')) { view = onCreateView(parent, name, attrs); } else { view = createView(name, null, attrs); } } finally { mConstructorArgs[0] = lastContext; } } return view; } catch //... Omit catch exception... } Duplicate codeCopy the code

Factory and Factory2 are both interfaces that provide the onCreateView() method. Factory2 extends a field from Factory.

public interface Factory { /** * Hook you can supply that is called when inflating from a LayoutInflater. * You can use this to customize the tag names available in your XML * layout files. * * <p> * Note that it is good practice to prefix these custom names with your * package (i.e., com.coolcompany.apps) to avoid conflicts with system * names. * * @param name Tag name to be inflated. * @param context The context the view is being created in. * @param attrs Inflation attributes as specified in XML file. * * @return View  Newly created view. Return null for the default * behavior. */ public View onCreateView(String name, Context context, AttributeSet attrs); } public interface Factory2 extends Factory { /** * Version of {@link #onCreateView(String, Context, AttributeSet)} * that also supplies the parent that the view created view will be * placed in. * * @param parent The parent that the created view will be placed * in; <em>note that this may be null</em>. * @param name Tag name to be inflated. * @param context The context the view is being created in. * @param attrs Inflation attributes as specified in XML file. * * @return View Newly created view. Return null for the default * behavior. */ public View onCreateView(View parent, String name, Context context, AttributeSet attrs); } Duplicate codeCopy the code

If all factories are empty or the view created by the factory is empty, the CreareView() method is eventually called. The code for this method is left blank, but an object is generated by the reflection of the control name (XML signature).

Low-level function for instantiating a view by name. This attempts to instantiate a view class of the given name found in this LayoutInflater’s ClassLoader.

The final question is when does a member variable of the Factory (or Factory2) interface type get assigned? Please look down.

4.3 Factory assignment in Activity

Let’s start with an Activity that implements the LayoutInflater.Factory2 interface.

public class Activity extends ContextThemeWrapper implements LayoutInflater.Factory2, Window.Callback, KeyEvent.Callback, OnCreateContextMenuListener, ComponentCallbacks2, Window.OnWindowDismissedCallback, WindowControllerCallback { //... Omit / * * * Standard implementation of * {@ link android. View. LayoutInflater. Factory# onCreateView}, informs the when inflating with the LayoutInflater returned by {@link #getSystemService}. * This implementation does nothing and is for * pre-{@link android.os.Build.VERSION_CODES#HONEYCOMB} apps. Newer apps * should use {@link #onCreateView(View, String, Context, AttributeSet)}. * * @see android.view.LayoutInflater#createView * @see android.view.Window#getLayoutInflater */ @Nullable public View onCreateView(String name, Context context, AttributeSet attrs) { return null; } /** * Standard implementation of * {@link android.view.LayoutInflater.Factory2#onCreateView(View, String, Context, AttributeSet)} * used when inflating with the LayoutInflater returned by {@link #getSystemService}. * This implementation handles <fragment> tags to embed fragments inside * of the activity. * * @see android.view.LayoutInflater#createView * @see android.view.Window#getLayoutInflater */ public View onCreateView(View parent, String name, Context context, AttributeSet attrs) { if (!" fragment".equals(name)) { return onCreateView(name, context, attrs); } return mFragments.onCreateView(parent, name, context, attrs); }} Copy the codeCopy the code

Here we have an added bonus, the fragment. If we embed a fragment in our XML with the Fragment tag, we will call the onCreateView() method of mFragments in our Activity to return a View when parsing the XML. Finally, add it to contentParent.

4.3.1 Associating Activities with LayoutInflater

// Final void attach(Context Context, ActivityThread aThread, Instrumentation instr, IBinder token, int ident, Application application, Intent intent, ActivityInfo info, CharSequence title, Activity parent, String id, NonConfigurationInstances lastNonConfigurationInstances, Configuration config, String referrer, IVoiceInteractor voiceInteractor, Window window) { //... Omit mWindow = new PhoneWindow(this, window); mWindow.setWindowControllerCallback(this); mWindow.setCallback(this); mWindow.setOnWindowDismissedCallback(this); mWindow.getLayoutInflater().setPrivateFactory(this); Copy the codeCopy the code

Again, the Attach () method (Internal API) calls the PhoneWindow getLayoutInflater() method at line 15 and sets the privateFactory.

public PhoneWindow(Context context) { super(context); mLayoutInflater = LayoutInflater.from(context); } /** * Return a LayoutInflater instance that can be used to inflate XML view layout * resources for use in this Window.  * * @return LayoutInflater The shared LayoutInflater. */ @Override public LayoutInflater getLayoutInflater() { return mLayoutInflater; } Duplicate codeCopy the code

The code says it all, and the comments are clear.

4.4 AppCompatActivity Factory assigns a value

Please look down

Fifth, AppCompatActivity

We’ve been doing a little bit of prep work, but our initial question was why is the getContext() type in ImageView TintContextWrapper below 5.0? When and when was it replaced? There are no answers yet, and the answers will follow. Hang in there, guys!

5.1 AppCompatActivity. The onCreate () method

@Override protected void onCreate(@Nullable Bundle savedInstanceState) { final AppCompatDelegate delegate = getDelegate(); delegate.installViewFactory(); delegate.onCreate(savedInstanceState); if (delegate.applyDayNight() && mThemeId ! = 0) { // If DayNight has been applied, we need to re-apply the theme for // the changes to take effect. On API 23+, we should bypass // setTheme(), which will no-op if the theme ID is identical to the // current theme ID. if (Build.VERSION.SDK_INT >= 23) { onApplyThemeResource(getTheme(), mThemeId, false); } else { setTheme(mThemeId); } } super.onCreate(savedInstanceState); } Duplicate codeCopy the code

If you’re familiar with line 3, proxy plus compatibility mode, this AppCompatDelegate concrete implementation class is reviewed.

// AppCompatActivity code, line 8 of this is the Activity itself. /** * @return The {@link AppCompatDelegate} being used by this Activity. */ @NonNull public AppCompatDelegate getDelegate() { if (mDelegate == null) { mDelegate = AppCompatDelegate.create(this, this); } return mDelegate; } // compatdelegate code private static appdelegate create(Context Context, Window Window, compatdelegate) AppCompatCallback callback) { final int sdk = Build.VERSION.SDK_INT; if (BuildCompat.isAtLeastN()) { return new AppCompatDelegateImplN(context, window, callback); } else if (sdk >= 23) { return new AppCompatDelegateImplV23(context, window, callback); } else if (sdk >= 14) { return new AppCompatDelegateImplV14(context, window, callback); } else if (sdk >= 11) { return new AppCompatDelegateImplV11(context, window, callback); } else { return new AppCompatDelegateImplV9(context, window, callback); }} Copy the codeCopy the code

AppCompatActivity. The onCreate () code, line 4 delegate. InstallViewFactory (). The implementation is in AppCompatDelegateImplV9. Look at the following code:

@Override public void installViewFactory() { LayoutInflater layoutInflater = LayoutInflater.from(mContext); if (layoutInflater.getFactory() == null) { LayoutInflaterCompat.setFactory(layoutInflater, this); } else { if (! (LayoutInflaterCompat.getFactory(layoutInflater) instanceof AppCompatDelegateImplV9)) { Log.i(TAG, "The Activity's LayoutInflater already has a Factory installed" + " so we can not install AppCompat's"); }}} copy the codeCopy the code

In line 3-5, if the layoutInflater’s Factory is empty, it sets itself to the layoutInflater to achieve the effect of setting the Factory (problem Solved in Section 4.3). It also achieves the effect of defining a contentView.

Compare this to the previous setContentView(View View) code. The difference is line 6 below.

@Override public void setContentView(int resId) { ensureSubDecor(); ViewGroup contentParent = (ViewGroup) mSubDecor.findViewById(android.R.id.content); contentParent.removeAllViews(); LayoutInflater.from(mContext).inflate(resId, contentParent); mOriginalWindowCallback.onContentChanged(); } Duplicate codeCopy the code

Anyone who doesn’t know how an AppCompatActivity can customize a contentView can check out chapter 4, which includes the 4.2 createViewFromTag() method. If you have any questions about contentParent, see Chapter 3.

In connection with our original question, has the mContext passed to LayoutInflater been replaced by TintContextWrapper? Of course not, from AppCompatActivity. The onCreate () method in all the way the context is AppCompatActivity handed down. We still have to look down.

5.2 AppCompatDelegateImplV9. OnCreateView () method

You can already see from the 5.1 code that you can associate yourself with LayoutInflater’s setFactory family of methods via AppCompatDelegateImplV9 in AppCompatActivity. The implementation of the Factory interface method is also naturally in AppCompatDelegateImplV9.

We first equate the LayoutInflaterFactory interface in support-V4 to the Factory2 interface of LayoutInflater, which we’ll explain in Chapter 6.

class AppCompatDelegateImplV9 extends AppCompatDelegateImplBase implements MenuBuilder.Callback, LayoutInflaterFactory { //... Omit... /** * From {@link android.support.v4.view.LayoutInflaterFactory} */ @Override public final View onCreateView(View parent, String name, Context context, AttributeSet attrs) { // First let the Activity's Factory try and inflate the view final View view = callActivityOnCreateView(parent, name, context, attrs); if (view ! = null) { return view; } // If the Factory didn't handle it, let our createView() method try return createView(parent, name, context, attrs); } / /... Omit... @Override public View createView(View parent, final String name, @NonNull Context context, @NonNull AttributeSet attrs) { if (mAppCompatViewInflater == null) { mAppCompatViewInflater = new AppCompatViewInflater(); } / /... Omit... return mAppCompatViewInflater.createView(parent, name, context, attrs, inheritContext, IS_PRE_LOLLIPOP, /* Only read android:theme pre-L (L+ handles this anyway) */ true, /* Read read app:theme as a fallback at all times for legacy reasons */ VectorEnabledTintResources.shouldBeUsed() /* Only tint wrap the context if enabled */ ); } / /... Omit... } Duplicate codeCopy the code

As you can see from the code above, the implementation of the Factory2 interface onCreateView() method in LayoutInflate, Is in AppCompatDelegateImplV9 (proxy implementation class in AppCompatActivity) and is using AppCompatViewInflater. Forget that you can go back to Chapter four.

Let’s go back and see what this AppCompatViewInflater createView() does.

5.3 AppCompatViewInflater

“Duang duang duang!”

public final View createView(View parent, final String name, @NonNull Context context, @NonNull AttributeSet attrs, boolean inheritContext, boolean readAndroidTheme, boolean readAppTheme, boolean wrapContext) { final Context originalContext = context; // We can emulate Lollipop's android:theme attribute propagating down the view hierarchy // by using the parent's context if (inheritContext && parent ! = null) { context = parent.getContext(); } if (readAndroidTheme || readAppTheme) { // We then apply the theme on the context, if specified context = themifyContext(context, attrs, readAndroidTheme, readAppTheme); } if (wrapContext) { context = TintContextWrapper.wrap(context); } View view = null; // We need to 'inject' our tint aware Views in place of the standard framework versions switch (name) { case "TextView":  view = new AppCompatTextView(context, attrs); break; case "ImageView": view = new AppCompatImageView(context, attrs); break; case "Button": view = new AppCompatButton(context, attrs); break; case "EditText": view = new AppCompatEditText(context, attrs); break; case "Spinner": view = new AppCompatSpinner(context, attrs); break; case "ImageButton": view = new AppCompatImageButton(context, attrs); break; case "CheckBox": view = new AppCompatCheckBox(context, attrs); break; case "RadioButton": view = new AppCompatRadioButton(context, attrs); break; case "CheckedTextView": view = new AppCompatCheckedTextView(context, attrs); break; case "AutoCompleteTextView": view = new AppCompatAutoCompleteTextView(context, attrs); break; case "MultiAutoCompleteTextView": view = new AppCompatMultiAutoCompleteTextView(context, attrs); break; case "RatingBar": view = new AppCompatRatingBar(context, attrs); break; case "SeekBar": view = new AppCompatSeekBar(context, attrs); break; } if (view == null && originalContext ! = context) { // If the original context does not equal our themed context, then we need to manually // inflate it using the name so that android:theme takes effect. view = createViewFromTag(context, name, attrs); } if (view ! = null) { // If we have created a view, check it's android:onClick checkOnClickListener(view, attrs); } return view; } Duplicate codeCopy the code

Lines 15-17, if wrapContext is true, wrap the context once with TintContextWrapper. We finally got our first look at this TintContextWrapper!! Now let’s look at it in more detail.

Lines 23-61 convert common basic views to appxxx. Now we know which basic controls can be replaced in AppCompatActivity, see the case above.

Lines 23-61 convert common basic views to appxxx. Now we know which basic controls can be replaced in AppCompatActivity, see the case above.

Here we just look at the constructor for AppCompatImageView (something like that) and also package the context with TintContextWrapper.

public AppCompatImageView(Context context, AttributeSet attrs) { this(context, attrs, 0); } public AppCompatImageView(Context context, AttributeSet attrs, int defStyleAttr) { super(TintContextWrapper.wrap(context), attrs, defStyleAttr); / /... Omit... } Duplicate codeCopy the code

5.4 TintContextWrapper

The code tells us that the SDK version is lower than 21 (Android 5.0) and wraps the Context as TintContextWrapper. That’s why ImageView in XML might get a Context of type TintContextWrapper.

public static Context wrap(@NonNull final Context context) { if (shouldWrap(context)) { synchronized (CACHE_LOCK) { / /... Omit... // If we reach here then the cache didn't have a hit, so create a new instance // and add it to the cache final TintContextWrapper wrapper = new TintContextWrapper(context); / /... Omit... return wrapper; } } return context; } private static boolean shouldWrap(@NonNull final Context context) { if (context instanceof TintContextWrapper || context.getResources() instanceof TintResources || context.getResources() instanceof VectorEnabledTintResources) { // If  the Context already has a TintResources[Experimental] impl, no need to wrap again // If the Context is already a TintContextWrapper, no need to wrap again return false; } return Build.VERSION.SDK_INT < 21 || VectorEnabledTintResources.shouldBeUsed(); } Duplicate codeCopy the code

5.5 VectorEnabledTintResources. ShouldBeUsed ()

In chapter 5.2 mAppCompatViewInflater. CreateView () method in or TintContextWrapper. ShouldWrap has this sentence () method VectorEnabledTintResources. ShouldBeUsed (). Let’s continue with the code:

@RestrictTo(LIBRARY_GROUP) public class VectorEnabledTintResources extends Resources { public static boolean shouldBeUsed() { return AppCompatDelegate.isCompatVectorFromResourcesEnabled() && Build.VERSION.SDK_INT <= MAX_SDK_WHERE_REQUIRED; } /** * The maximum API level where this class is needed. */ public static final int MAX_SDK_WHERE_REQUIRED = 20; / /... Omit... } Duplicate codeCopy the code
/ / / / AppCompatDelegate code... Omit... private static boolean sCompatVectorFromResourcesEnabled = false; / /... Omit... /** * Sets whether vector drawables on older platforms (< API 21) can be used within * {@link android.graphics.drawable.DrawableContainer} resources. * * <p>When enabled, AppCompat can intercept some drawable inflation from the framework, which * enables implicit inflation of vector drawables within * {@link android.graphics.drawable.DrawableContainer} resources. You can then use those * drawables in places such as {@code android:src} on {@link android.widget.ImageView},  * or {@code android:drawableLeft} on {@link android.widget.TextView}. Example usage:</p> * * <pre> * &lt; selector xmlns:android=&quot; . &quot; &gt; * &lt; item android:state_checked=&quot; true&quot; * android:drawable=&quot; @drawable/vector_checked_icon&quot; /&gt; * &lt; item android:drawable=&quot; @drawable/vector_icon&quot; /&gt; * &lt; /selector&gt; * * &lt; TextView * ... * android:drawableLeft=&quot; @drawable/vector_state_list_icon&quot; /&gt; * </pre> * * <p>This feature defaults to disabled, since enabling it can cause issues with memory usage, * and problems updating {@link Configuration} instances. If you update the configuration * manually, then you probably do not want to enable this. You have been warned.</p> * * <p>Even with this disabled, you can still use vector resources through * {@link android.support.v7.widget.AppCompatImageView#setImageResource(int)} and it's * {@code app:srcCompat} attribute. They can also be used in anything which AppCompat inflates * for you, such as menu resources.</p> * * <p>Please note: this only takes effect in Activities created after this call.</p> */ public static void setCompatVectorFromResourcesEnabled(boolean enabled) { sCompatVectorFromResourcesEnabled = enabled; } /** * Returns whether vector drawables on older platforms (< API 21) can be accessed from within * resources. * * @see  #setCompatVectorFromResourcesEnabled(boolean) */ public static boolean isCompatVectorFromResourcesEnabled() { return sCompatVectorFromResourcesEnabled; } Duplicate codeCopy the code

When the VectorEnabledTintResources. ShouldBeUsed () returns true? When the version is less than 5.0 and call AppCompatDelegate. SetCompatVectorFromResourcesEnabled is set to true (note that is a static method.

The VectorEnabledTintResources. ShouldBeUsed () method is judgment when the system is under 5.0, whether to support vector drawing resources, false by default. For those who have doubts about this section, you can search the relevant vector map using methods, compatible with the low version strategy, so we will not expand here.

5.6 Let’s summarize

1. In AppCompatActivity,onCreate()Fatory2 (layOutInflater. Fatory2); LayoutInflaterFactory (LayoutInflaterFactory);

2. Call againinstallViewFactory()Methods that will implement proxy classes and layoutInflatersfactoryMember variable binding.

3. When we call ourselvessetContentView(R.layout.xxx)Method is called into the LayoutInflater when parsing the XMLinflate()The method, and then thecreateViewFromTag()Methods.

4,createViewFromTag()If there are local factory variables in the factory class, those interface variables are called firstonCreateView()Methods. In AppCompatActivityonCreateView()It’s in AppCompatDelegateImplV9.

AppCompatDelegateImplV9AppCompatViewInflaterTo generate the View. So with the replacement of the base control content, with the following 5.0 systems willContextPackaged inTintContextWrapper, passed in when building AppCompatxxx controlscontextWas replaced byTintContextWrapperType.

How does V4’s LayoutInflater interface compare to LayoutInflter’s Factory2 interface?

@Override public void installViewFactory() { LayoutInflater layoutInflater = LayoutInflater.from(mContext); if (layoutInflater.getFactory() == null) { LayoutInflaterCompat.setFactory(layoutInflater, this); } else { //... Omit... }} Copy the codeCopy the code

Last question: Code line 5, how will layoutInflater accept Factory (Factory2) type to accept this (for android. Support. The v4. The LayoutInflaterFactory interface)??

Take a look at the comments in the V4 package for LayoutInflaterFactory to see what they mean. How to do this is to move on to Chapter 6.1.

/** * Used with {@code LayoutInflaterCompat.setFactory()}. Offers the same API as * {@code LayoutInflater.Factory2}. */ public interface LayoutInflaterFactory { /** * Hook you can supply that is called when inflating from a LayoutInflater. * You can use this to customize the tag names available in your XML * layout files. * * @param parent The parent that the created view will be placed * in; <em>note that this may be null</em>. * @param name Tag name to be inflated. * @param context The context the view is being created in. * @param attrs Inflation attributes as specified in XML file. * * @return View Newly created view. Return null for the default * behavior. */ public View onCreateView(View parent, String name, Context context, AttributeSet attrs); } Duplicate codeCopy the code

6.1 LayoutInflaterCompat

We went back to android. Support. The v4. View. As what LayoutInflaterCompat.

/ / code android support. The v4. The LayoutInflaterCompat / * * * Attach a custom Factory interface for creating views while using  * this LayoutInflater. This must not be null, and can only be set once; * after setting, you can not change the factory. * * @see LayoutInflater#setFactory(android.view.LayoutInflater.Factory) */ public static  void setFactory(LayoutInflater inflater, LayoutInflaterFactory factory) { IMPL.setFactory(inflater, factory); } static final LayoutInflaterCompatImpl IMPL; static { final int version = Build.VERSION.SDK_INT; if (version >= 21) { IMPL = new LayoutInflaterCompatImplV21(); } else if (version >= 11) { IMPL = new LayoutInflaterCompatImplV11(); } else { IMPL = new LayoutInflaterCompatImplBase(); }} Copy the codeCopy the code

We are familiar with the proxy pattern, implementation of the IMP class is a compatible pattern.

Let’s look at a simple code implementation of layoutInflaterbase.

// Code LayoutInflaterCompat Interface LayoutInflaterCompatImpl {public void setFactory(LayoutInflater LayoutInflater, LayoutInflaterFactory factory); public LayoutInflaterFactory getFactory(LayoutInflater layoutInflater); } static class LayoutInflaterCompatImplBase implements LayoutInflaterCompatImpl { @Override public void setFactory(LayoutInflater layoutInflater, LayoutInflaterFactory factory) { LayoutInflaterCompatBase.setFactory(layoutInflater, factory); } @Override public LayoutInflaterFactory getFactory(LayoutInflater layoutInflater) { return LayoutInflaterCompatBase.getFactory(layoutInflater); }} Copy the codeCopy the code

6.2 LayoutInflaterCompatBase

class LayoutInflaterCompatBase { static class FactoryWrapper implements LayoutInflater.Factory { final LayoutInflaterFactory mDelegateFactory; FactoryWrapper(LayoutInflaterFactory delegateFactory) { mDelegateFactory = delegateFactory; } @Override public View onCreateView(String name, Context context, AttributeSet attrs) { return mDelegateFactory.onCreateView(null, name, context, attrs); } public String toString() { return getClass().getName() + "{" + mDelegateFactory + "}"; } } static void setFactory(LayoutInflater inflater, LayoutInflaterFactory factory) { inflater.setFactory(factory ! = null ? new FactoryWrapper(factory) : null); } / /... Omit... } Duplicate codeCopy the code

At line 22, wrap the LayoutInflaterFactory in the V4 package as a FactoryWrapper type and call the LayoutInflater setFactory() method.

Line 13, using proxy mode. FactoryWrapper implements LayoutInflater’s Factory interface, which is replaced by a proxy class in the concrete onCreateView() method implementation.

At line 7, the FactoryWrapper constructor entry is a proxy class of the same type as the LayoutInflaterFactory interface in the V4 package.

6.3 Summary:

1, inLayoutInflaterCompat.setFactory(layoutInflater, this);Through a series of proxy compatible modes, LayoutInflater will be inflatersetFactory()The parameters received by a series of methods are changed to LayoutInflaterFactory interface type parameters in the V4 package.

2. IncomingthisThat’s AppCompatDelegateImplV9 itself. So Factory series interfaceonCreateView()Method implementation, then falls into the AppCompatDelegateImplV9 method.

Vii. Solutions

1. How does view.getContext () force an Activity?

Here is a common idea as a reference:

@Nullable private Activity getActivity(@NonNull View view) { if (null ! = view) { Context context = view.getContext(); while (context instanceof ContextWrapper) { if (context instanceof Activity) { return (Activity) context; } context = ((ContextWrapper) context).getBaseContext(); } } return null; } Duplicate codeCopy the code