Activity-setContentView(int resId)源码分析

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结构:
这里写图片描述

public class MainActivity extends AppCompatActivity { Button view_button; private Context mContext; private ViewFlipper vflp_help; private int[] resId = {R.drawable.img01,R.drawable.img02, R.drawable.img03}; private final static int MIN_MOVE = 200; //最小距离 private MyGestureListener mgListener; private GestureDetector mDetector; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); view_button=findViewById(R.id.view_button); view_button.setOnClickListener(new View.OnClickListener() { @Override public void onClick(View view) { Intent intent=new Intent(MainActivity.this,Main2Activity.class); startActivity(intent); } }); mContext = MainActivity.this; //实例化SimpleOnGestureListener与GestureDetector对象 mgListener = new MyGestureListener(); mDetector = new GestureDetector(this, mgListener); vflp_help = (ViewFlipper) findViewById(R.id.vflp_help); //动态导入添加子View for(int i = 0;i < resId.length;i++){ vflp_help.addView(getImageView(resId[i])); } } //重写onTouchEvent触发MyGestureListener里的方法 @Override public boolean onTouchEvent(MotionEvent event) { return mDetector.onTouchEvent(event); } //自定义一个GestureListener,这个是View类下的,别写错哦!!! private class MyGestureListener extends GestureDetector.SimpleOnGestureListener { @Override public boolean onFling(MotionEvent e1, MotionEvent e2, float v, float v1) { if(e1.getX() - e2.getX() > MIN_MOVE){ vflp_help.setInAnimation(mContext,R.anim.right_in); vflp_help.setOutAnimation(mContext, R.anim.right_out); vflp_help.showNext(); }else if(e2.getX() - e1.getX() > MIN_MOVE){ vflp_help.setInAnimation(mContext,R.anim.left_in); vflp_help.setOutAnimation(mContext, R.anim.left_out); vflp_help.showPrevious(); } return true; } } private View getImageView(int resId){ ImageView img = new ImageView(this); img.setBackgroundResource(resId); return img; }}
06-11
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值