自定义view[getHeight getWidth]

EqualSizeLinearLayout extends LinearLayout

  @Override
  protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec)
  {
    super.onMeasure(widthMeasureSpec, heightMeasureSpec);

    int childSize = (getOrientation() == HORIZONTAL) ? getMeasuredHeight()-getPaddingTop()-getPaddingBottom()
                                                     : getMeasuredWidth()-getPaddingLeft()-getPaddingRight();
    final int childCount = getChildCount();
    for(int i=0; i

getHeight()和 getMeasuredHeight()区别:
前者是最终的大小,后者是测量的大小,举个列子,当布局还没完成的时候,getHeight就是0的,
或者是在onmeasure方法里,这个方法会进来两次,第一次进来的时候getheight是0的,第二次进来才有真正的高度【也不一定额,比如进行child.measure以后这个getheight也不准了。具体看日志下边的解释】
如下的布局:



        
    @Override
    protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
        super.onMeasure(widthMeasureSpec, heightMeasureSpec);
//      System.err.println("widthMeasureSpec==========="+widthMeasureSpec+"============heightMeasureSpec="+heightMeasureSpec);
//      System.err.println("getMode==============w="+MeasureSpec.getMode(widthMeasureSpec)+"===h="+MeasureSpec.getMode(heightMeasureSpec));
//      System.err.println("getSize========"+MeasureSpec.getSize(widthMeasureSpec)+" ========="+MeasureSpec.getSize(heightMeasureSpec));
//      
        System.err.println(getChildCount()+"===11111======"+MeasureSpec.UNSPECIFIED+"======"+MeasureSpec.EXACTLY+"======="+MeasureSpec.AT_MOST);
        for(int i=0;i

日志如下:

02-11 02:01:59.767: W/System.err(28512): i==0 ==0===100
02-11 02:01:59.767: W/System.err(28512): i==0 ==0===100
02-11 02:01:59.767: W/System.err(28512): i==1 ==0===48
02-11 02:01:59.767: W/System.err(28512): i==1 ==0===48
02-11 02:01:59.787: W/System.err(28512): onLayout=============true=========100============0==48
02-11 02:01:59.797: W/System.err(28512): i==0 ==100===100
02-11 02:01:59.797: W/System.err(28512): i==0 ==100===100
02-11 02:01:59.797: W/System.err(28512): i==1 ==48===48
02-11 02:01:59.797: W/System.err(28512): i==1 ==48===100
02-11 02:01:59.797: W/System.err(28512): onLayout=============false=========100============48==100

//完事点击toggleButton,布局会重置的,可以看到,经过如下方法super.onMeasure(widthMeasureSpec, heightMeasureSpec);
child的measureHeight又还原回去了。
02-11 02:04:37.637: W/System.err(28512): i==0 ==100===100
02-11 02:04:37.637: W/System.err(28512): i==0 ==100===100
02-11 02:04:37.637: W/System.err(28512): i==1 ==100===48
02-11 02:04:37.637: W/System.err(28512): i==1 ==100===100
02-11 02:04:37.637: W/System.err(28512): onLayout=============false=========100============100==100

对日志的解释,首次加载线性布局的onmeasure会进行2次的,第一次没高度,不考虑。
第二次的高度48,进行child.measure之后高度看到还是48,这是真的吗,其实不是真的,这时候getheight应该已经是100,不过就和我们在oncreate里getheight一样,这个时候获取到的高度并不准确,你可以延迟1秒才调用child.getHeigth就可以看到返回的是100,和我们实际显示的大小一致。
第二次的日志,点击togglebutton以后整个线性布局会进行重绘,因为调用了super.onMeasure(widthMeasureSpec, heightMeasureSpec); 其实这时候child的大小又恢复原始大小48了,可其实我们看到打印的日志getheight还是100,这个是不准确的。延迟获取才能看到真实的大小。
或者给这个线性布局设置个背景,让它可以走onDraw方法,在onDraw方法里获取高度也是真实的

看下源码draw方法里如下的注释

/*
         * Draw traversal performs several drawing steps which must be executed
         * in the appropriate order:
         *
         *      1. Draw the background
         *      2. If necessary, save the canvas' layers to prepare for fading
         *      3. Draw view's content
         *      4. Draw children
         *      5. If necessary, draw the fading edges and restore layers
         *      6. Draw decorations (scrollbars for instance)
         */


    /**
     * Flag indicating whether a view failed the quickReject() check in draw(). This condition
     * is used to check whether later changes to the view's transform should invalidate the
     * view to force the quickReject test to run again.
     */
    static final int PFLAG2_VIEW_QUICK_REJECTED = 0x10000000;

//16080行
        if (!drawingWithDrawingCache) {
            if (drawingWithRenderNode) {
                mPrivateFlags &= ~PFLAG_DIRTY_MASK;
                ((DisplayListCanvas) canvas).drawRenderNode(renderNode);
            } else {
                // Fast path for layouts with no backgrounds
                if ((mPrivateFlags & PFLAG_SKIP_DRAW) == PFLAG_SKIP_DRAW) {
                    mPrivateFlags &= ~PFLAG_DIRTY_MASK;
                    dispatchDraw(canvas);
                } else {
                    draw(canvas);
                }
            }
        }

线性布局,相对布局,帧布局,只有设置了背景才会执行 draw和onDraw方法。 最后执行的是dispathDraw
另外,线性布局如果设置divider也是可以的,不需要背景。

你可能感兴趣的:(自定义view[getHeight getWidth])