美文网首页
LayoutInflater-源码分析

LayoutInflater-源码分析

作者: miraclehen | 来源:发表于2017-07-26 11:49 被阅读23次

    转载请注明出处 http://www.jianshu.com/p/3bf0d75d5ada (作者:韩栋)
    由于本人水平有限,欢迎拍砖。

    上文我们讲了LayoutInflater的使用方法。读者可以点击此LayoutInflater-使用
    本文主要讲关于LayoutInflater源码分析,让读者更好地了解并使用它。

    源码

    直接切入正题。LayoutInflater.inflate()源码如下:

    public View inflate(@LayoutRes int resource, @Nullable ViewGroup root) {
            return inflate(resource, root, root != null);
    }
    

    可以看到,它是一个重载方法。值得注意的是,root != null这个逻辑,这也就为什么我在前文中说以下两个方法的实现是一模一样了。

     View view =LayoutInflater.from(MainActivity.this).inflate(R.layout.view_child,frameLayout);
    View view =LayoutInflater.from(MainActivity.this).inflate(R.layout.view_child,frameLayout,true);
    

    来看inflate具体实现:

    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();
            }
        }
    

    这里只是根据我们传入的新布局文件获取到它的XmlResourceParser对象。还是重载了inflate方法。囧。。我们继续往下看inflate(parser, root, attachToRoot)

    public View inflate(XmlPullParser parser, @Nullable ViewGroup root, boolean attachToRoot) {
            synchronized (mConstructorArgs) {
               
                final Context inflaterContext = mContext;
                //获取新的布局的属性集合
                final AttributeSet attrs = Xml.asAttributeSet(parser);
                Context lastContext = (Context) mConstructorArgs[0];
                mConstructorArgs[0] = inflaterContext;
                View result = root;
    
                try {
                   //省略···
                    final String name = parser.getName();
          
                        // 创建新的视图,注意:此时的新的视图的根布局的布局参数是空的
                        final View temp = createViewFromTag(root, name, inflaterContext, attrs);
    
                        ViewGroup.LayoutParams params = null;
                        //省略···
                        if (root != null) {
                            
                            // 这是重点:将原来新视图的根布局参数转换成适合`root`根布局的布局参数
                            params = root.generateLayoutParams(attrs);
                           //如果attachToRoot == false。则将新转换的布局参数赋给新视图
                            if (!attachToRoot) {
                                temp.setLayoutParams(params);
                            }
                        }
                        // 创建新视图下所有的子视图,忽略它,不是重点
                        rInflateChildren(parser, temp, attrs, true);
                        
                        //如果root != null并且attachToRoot == true
                        if (root != null && attachToRoot) {
                            //将新视图将入到root中
                            root.addView(temp, params);
                        }
                        
                        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;
                }
    
                return result;
            }
        }
    

    嗯。这就是具体实现了。代码看过去貌似很长。其实很简单。只是毕竟是google大神写的代码。考虑的东西非常全面。不过为了方便读者理解,我把一些非重要的代码给省略了。

    废话不多说。这里主要说一下params = root.generateLayoutParams(attrs);这个方法。这个方法调用的是root中的generateLayoutParams。这个方法是在ViewGroup中定义的,用来根据已有的属性集合来转换成适合自身布局参数的的工具方法。我们简单看下FrameLayout的重写generateLayoutParams

        @Override
        public LayoutParams generateLayoutParams(AttributeSet attrs) {
            return new FrameLayout.LayoutParams(getContext(), attrs);
        }
    

    返回了一个LayoutParams对象,继续看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();
            }
    

    FrameLayout.LayoutParamsFrameLayout的一个内部类。它继承自MarginLayoutParams。所以在这里调用了super(c, attrs),使这个LayoutParams对象支持Margin的属性参数,以及在这里 gravity = a.getInt(R.styleable.FrameLayout_Layout_layout_gravity, 。UNSPECIFIED_GRAVITY);,获取gravity属性(这可是FrameLayout最重要的一个属性了。

    言归正传。我们把新视图的根布局参数传递给root。让root进行转换成适合它自身布局的布局参数。

    相关文章

      网友评论

          本文标题:LayoutInflater-源码分析

          本文链接:https://www.haomeiwen.com/subject/zfdwkxtx.html