一。Measure
View的Measure过程:
它会调用View的onMeasure方法,在onMeasure方法中会调用setMeasureDimension(width,height)方法设置view的宽高测量值, 而width和height的值在AT_MOST和EXACTLY模式下都等于specSize.(一般都是父容器剩余空间大小)
所以如果我们自定义view时继承View而不做任何改动,wrap_content和match_parent的表现效果是一样的
我们如果要让wrap_content生效,必须重写onMeasure方法
TextView和ImageView的实现都针对wrap_content情形做了特殊处理。
ViewGroup的Measure过程:
viewGroup没有实现onMeasure方法,
因为不同的viewGroup的特点不同,如LinearLayout 和RelativeLayout,
只有具体的viewGroup自己实现。
二,layout
首先通过setFrame方法获得View的四个顶点位置
然后调用onLayout方法确认其子元素的位置
由于不同的View和ViewGroup的特点不同,也要根据具体情况来实现它的onLayout方法
三,draw
执行步骤:绘制背景(background.draw(canvas))
绘制自己(onDraw)
绘制children(dispatchDraw)
绘制装饰(onDrawScrollBars)
dispatchDraw方法会遍历子元素的draw方法,以此一层一层完成view树的绘制
由于view的measure,layout,draw过程和activity的onCreate,onStar,onResume并不同步
所以在activity的这些回调方法中调用view.getMeasuredHeight()来得到View的高度时,可能会因为
View还没有完成layout,使得得到的值为0
那在什么时候什么地方调用View.getMeasuredHeight()可以确保返回值为View的高度呢
方法如下:
1.onWindowFocusChanged(该方法会在activity的窗口得到焦点和失去焦点时均被调用一次
public void onWindowFocusChanged(boolean hasFocus){super.onWindowFocusChanged(hasFocus);if(hasFocus){int width = view.getMeasuredWidth();int height = view.getMeasuredHeight();}}2.view.post(runnable)通过将一个runnable放在消息队列的最后,那runnable的执行时间就在view完成初始化以后
view.post(new Runnable(){@Overridepublic void run(){int width = view.getMeasuredWidth();int height = view.getMeasuredHeight();}}3.ViewTreeObserver(在View树的状态发生改变或其中View的可见性发生改变时,onGlobalLayout方法将被回调)ViewTreeObserver observer = view.getViewTreeObserver();observer.addOnGlobalLayoutListener(new OnGlobalLayoutListener(){@SupPRessWarnings("deprecation")@Overridepublic void onGlobalLayout(){view.getViewTreeObserver().removeGlobalOnLayoutListener(this);int width = view.getMeasuredWidth();int height = view.getMeasureHeight();}}参考:android开发艺术探索
新闻热点
疑难解答