美文网首页Android 自定义view
Android onMeasure、Measure、measur

Android onMeasure、Measure、measur

作者: 白驹过隙_a | 来源:发表于2019-01-13 00:47 被阅读0次

    一、VIew 中对 子视图 进行 measure() 操作:

    1.measureChildren() : 内部调用 measureChild() 对每一个子视图进行 measure 操作

    2.measureChild() : 为指定的子视图进行measure操作

    3.measureChildWithMargins() : measure 时考虑把 margin 及 padding 也作为子视图大小的一部分

    二、measureChildWithMargins() 执行流程:

    1.调用child.getLayoutParams()获得子视图的LayoutParams属性

    1.1 该方法需要重载 generateDefaultLayoutParams() 返回一个继承自 ViewGroup.MarginLayoutParams 的 布局类(例如 LinearLayout.LayoutParams),如果未重载将导致 本 View 的XML属性 layout_margin 失效并且不能使用 measureChildWithMargins()

    2.调用两次 getChildMeasureSpec()函数,分别计算出孩子视图的宽度和高度的 Spec(确定子视图的测量规格).

    3.调用child.measure()函数,确定子视图的最终布局大小。

    三、最后调用resolveSize()获取高度,然后调用 setMeasureDimension()设置该本身所占用的布局高度。

    android onmeasure在View.java中的定义:

    protected void onMeasure(int widthMeasureSpec,int heightMeasureSpec) {
    
            setMeasuredDimension(getDefaultSize(getSuggestedMinimumWidth(), widthMeasureSpec),
    
                    getDefaultSize(getSuggestedMinimumHeight(), heightMeasureSpec));
    
    }
    
     public static int getDefaultSize(int size, int measureSpec) {
            int result = size;
            int specMode = MeasureSpec.getMode(measureSpec);
            int specSize = MeasureSpec.getSize(measureSpec);
    
            switch (specMode) {
            case MeasureSpec.UNSPECIFIED: //未指定
                result = size;
                break;
            case MeasureSpec.AT_MOST: //至多
            case MeasureSpec.EXACTLY: //精确
                result = specSize;
                break;
            }
            return result;
        }
    

    在ViewGroup中:

       protected void measureChildren(int widthMeasureSpec, int heightMeasureSpec) {
            final int size = mChildrenCount;
            final View[] children = mChildren;
            for (int i = 0; i < size; ++i) {
                final View child = children[i];
                if ((child.mViewFlags & VISIBILITY_MASK) != GONE) {
                    measureChild(child, widthMeasureSpec, heightMeasureSpec);
                }
            }
        }
    
       protected void measureChild(View child, int parentWidthMeasureSpec,
                int parentHeightMeasureSpec) {
            final LayoutParams lp = child.getLayoutParams();
    
            final int childWidthMeasureSpec = getChildMeasureSpec(parentWidthMeasureSpec,
                    mPaddingLeft + mPaddingRight, lp.width);
            final int childHeightMeasureSpec = getChildMeasureSpec(parentHeightMeasureSpec,
                    mPaddingTop + mPaddingBottom, lp.height);
    
            child.measure(childWidthMeasureSpec, childHeightMeasureSpec);
        }
    
     protected void measureChildWithMargins(View child,
                int parentWidthMeasureSpec, int widthUsed,
                int parentHeightMeasureSpec, int heightUsed) {
            final MarginLayoutParams lp = (MarginLayoutParams) child.getLayoutParams();
    
            final int childWidthMeasureSpec = getChildMeasureSpec(parentWidthMeasureSpec,
                    mPaddingLeft + mPaddingRight + lp.leftMargin + lp.rightMargin
                            + widthUsed, lp.width);
            final int childHeightMeasureSpec = getChildMeasureSpec(parentHeightMeasureSpec,
                    mPaddingTop + mPaddingBottom + lp.topMargin + lp.bottomMargin
                            + heightUsed, lp.height);
    
            child.measure(childWidthMeasureSpec, childHeightMeasureSpec);
        }
    

    说明:

    1. measure是final修饰的方法,不可被重写。

      在外部调用时,直接调用view.measure(int wSpec, int hSpec)。

      measure中调用了onMeasure。

      自定义view时,重写onMeasure即可。

    2.MeasureSpec 这是一个含mode和size的结合体,不需要我们来具体的关心。

    当在测量时,可以调用MeasureSpec.getSize|getMode 得到相应的size和mode。

    然后使用MeasureSpec.makeMeasureSpec(size,mode); 来创建MeasureSpec对象。

    那么mode是怎么来的呢?是根据使用该自定义view时的layoutWith|height参数决定的,所以不能自己随便new一个。

    而size可以自己指定,也可以直接使用 measureSpec.getSize。

    3.如果是一个View,重写onMeasure时要注意:

    如果在使用自定义view时,用了wrap_content。那么在onMeasure中就要调用setMeasuredDimension,

    来指定view的宽高。如果使用的fill_parent或者一个具体的dp值。那么直接使用super.onMeasure即可。

    4.如果是一个ViewGroup,重写onMeasure时要注意:

    首先,结合上面两条,来测量自身的宽高。

    然后,需要测量子View的宽高。

    测量子view的方式有:

    1. getChildAt(int index).可以拿到index上的子view。

          通过getChildCount得到子view的数目,再循环遍历出子view。
      
          接着,subView.measure(int wSpec, int hSpec); //使用子view自身的测量方法
      
    2. 或者调用viewGroup的测量子view的方法:

      //某一个子view,多宽,多高, 内部加上了viewGroup的padding值

      measureChild(subView, int wSpec, int hSpec);

    3. //所有子view 都是 多宽,多高, 内部调用了measureChild方法

      measureChildren(int wSpec, int hSpec);

    4. //某一个子view,多宽,多高, 内部加上了viewGroup的padding值、margin值和传入的宽高wUsed、hUsed

        measureChildWithMargins(subView, intwSpec, int wUsed, int hSpec, int hUsed); 

    相关文章

      网友评论

        本文标题:Android onMeasure、Measure、measur

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