Activity-setContentView(int resId)源码分析
2017-11-06 11:21
423 查看
Activity-setContentView(int resId)源码分析
入口–Activity–》setContentView()public void setContentView(@LayoutRes int layoutResID) { getWindow().setContentView(layoutResID); initWindowDecorActionBar(); }
大家应该都知道这里的getWindow()返回的是一个PhoneWindow对象,所以直接跳到PhoneWindow中去查看setContentView(layoutResID)代码:
@Override public void setContentView(int layoutResID) { if (mContentParent == null) { installDecor(); 初始化mContentParent } else if (!hasFeature(FEATURE_CONTENT_TRANSITIONS)) { mContentParent.removeAllViews(); } if (hasFeature(FEATURE_CONTENT_TRANSITIONS)) { final Scene newScene = Scene.getSceneForLayout(mContentParent, layoutResID, getContext()); transitionTo(newScene); } else { //加载xml界面并添加到mContentParent中。 mLayoutInflater.inflate(layoutResID, mContentParent); } mContentParent.requestApplyInsets(); 。。。。。。 }
代码很简单,首先是判断contentParent是否实例化了?如果没有就执行installDecor()去实例化、以下是installDecor()代码:
private void installDecor() { mForceDecorInstall = false; if (mDecor == null) { mDecor = generateDecor(-1); mDecor.setDescendantFocusability(ViewGroup.FOCUS_AFTER_DESCENDANTS); mDecor.setIsRootNamespace(true); if (!mInvalidatePanelMenuPosted && mInvalidatePanelMenuFeatures != 0) { mDecor.postOnAnimation(mInvalidatePanelMenuRunnable); } } else { mDecor.setWindow(this); } if (mContentParent == null) { mContentParent = generateLayout(mDecor); ,,,,,,
这里的代码也不难,就是说如果mDecor如果为null就去generateDecor(-1)得到一个DecorView,大家可以点进去看就是return了一个new DecorView回来。那么往下看,如果mContentParent为null,就回去调用
generateLayout(mDecor)方法,将前面刚刚加载的mDecor当做参数传递过去,看一下这个代码:
protected ViewGroup generateLayout(DecorView decor) { ------- int layoutResource; int features = getLocalFeatures(); ------加载layoutResource,实际是一个安卓自带的布局文件 //通过判断在mDecor中加载layoutResource布局,其中包含着一个contentParent。 mDecor.onResourcesLoaded(mLayoutInflater, layoutResource); ViewGroup contentParent = (ViewGroup)findViewById(ID_ANDROID_CONTENT); --------- 对mDecor的一些添加。 return contentParent; } public static final int ID_ANDROID_CONTENT = com.android.internal.R.id.content;
这个方法内部的代码很长,前前后后几十行。这边主要讲两处,第一处是通过较多的判断来为layoutResource赋值一个安卓自带的xml布局文件,然后调用onResourcesLoaded来将这个xml文件加载到mDecor中(可以自己查看源码,很简单,调用的addView方法)。
第二处就是这个findViewById了,点进去看就是获取到外围的DecorView去获取ContentParent。
所以其实在第一处加载到mDecor中的layoutResource布局中就包含了一个R.id.content的frameLayout。
@Nullable public View findViewById(@IdRes int id) { return getDecorView().findViewById(id); }
这样的话contentParent和mDecorView就都有了,且contentParent已经加载到了mDecorView中,接下来再看一遍setContentView源码:
@Override public void setContentView(int layoutResID) { if (mContentParent == null) { installDecor(); 初始化mContentParent } else if (!hasFeature(FEATURE_CONTENT_TRANSITIONS)) { mContentParent.removeAllViews(); } if (hasFeature(FEATURE_CONTENT_TRANSITIONS)) { final Scene newScene = Scene.getSceneForLayout(mContentParent, layoutResID, getContext()); transitionTo(newScene); } else { //加载xml界面并添加到mContentParent中。 mLayoutInflater.inflate(layoutResID, mContentParent); } mContentParent.requestApplyInsets(); 。。。。。。 }
加载出来并将加载出来的view添加到mContentParent中,(这块如果有问题可以看我对LayoutInflater加载view的源码分析的文章。)
这样就形成了这样的view结构:
相关文章推荐
- 从源码分析Activity布局构成(setContentview)
- Activity加载view6.0源码分析---setContentView
- Activity的setContentView()方法源码分析
- Activity.setContentView()源码分析
- Android布局文件的加载过程分析:Activity.setContentView()源码分析
- 查看源码分析activity执行setContentView的流程
- Android应用setContentView与LayoutInflater加载解析机制源码分析
- Android应用setContentView与LayoutInflater加载解析机制源码分析
- Android应用setContentView与LayoutInflater加载解析机制源码分析
- SetContentView与LayoutInflater源码分析
- 从源码的角度说说Activity的setContentView的原理(二)
- Android应用setContentView与LayoutInflater加载解析机制源码分析
- 深入理解View知识系列一- setContentView和LayoutInflater源码原理分析
- Android应用setContentView与LayoutInflater加载解析机制源码分析(转载)
- 【Android View源码分析(一)】setContentView加载视图机制深度分析
- [置顶] android源码分析——由SetContentView串起来的布局加载机制
- Android应用setContentView与LayoutInflater加载解析机制源码分析
- Android应用setContentView与LayoutInflater加载解析机制源码分析
- Android源码梳理(一):setContentView(...)与LayoutInflater的加载机制分析
- android中布局和View创建的源码分析---setContentView