View的工作原理(三)

一。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(){
@Override
public 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")
@Override
public void onGlobalLayout(){
view.getViewTreeObserver().removeGlobalOnLayoutListener(this);
int width = view.getMeasuredWidth();
int height = view.getMeasureHeight();
}
}


参考:android开发艺术探索


發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章