Android 自定义view

Android onMeasure、Measure、measur

2019-01-13  本文已影响0人  白驹过隙_a

一、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); 
上一篇下一篇

猜你喜欢

热点阅读