参考书籍:
Android开发艺术探索
注:京东链接https://item.jd.com/11760209.html
ViewRoot和DecorView
1.ViewRoot对应于ViewRootImpl类,是连接Windowmanager和DecorView的纽带;
View的三大流程都是通过ViewRoot来完成的;
2.在ActivityThread(UI线程)中Activity创建的时候:
将DecorView添加到Window中,同时创建VIewRootimpl对象,并将两者关联起来,代码如下
root = new ViewRootImpl(view.getContext(),display);
root.setView(view,wprasrms,panelParentView);
3.View的绘制流程
performTraversals(执行遍历)->performMeasure()->performLayout()->performDraw()
以Measure举例:performMeasure()->measure->onMeasure(会对所有的子View遍历mesasure)
4.其它
Decorview:
顶部titlebar
android.R.id.content(内容布局=FramLayout)
在Activity中的setContentView就是设置DecorView中的内容布局
MeasureSpec
1.MeasureSpec-测量规格;系统会将View的LayoutParams转化为对应的MeasureSpec
2.SpecMode(测量模式):高2位
SpecSize:低两位
MeasureSpec=SpecSize+SpecMode;MeasureSpec将SpecMode和SpecSize打包成一个int值
3.SpecMode类型
UNSPECIFIED(未指明):父容器不对view做限制,只用于系统
EXACTLY(精确):对应match_parent和具体数值,父容器已经检测出view所需精确大小
AT_MOST(最大不超过):对应wrap_content,不能超过父容器给定尺寸
1.1 view的measure过程
1.measure->onMeasure->setMeasureDimension(getDefaultsize(..width),(..height))
2.setMeasureDimension(设置view的宽高测量值):
AT_MOST/EXACTLY(wrap_content尺寸精准):返回specsize(测量大小)
UNSPECIFIED(系统测量):getsuggestminWidth
3.继承view的自定义控件需要重写onMeasure方法并设置尺寸大小(wrap_content);
不重写设置尺寸,尺寸将为父view的尺寸(相当于match_parent)
1.2 viewGroup的measure过程
1.流程
measureChildren(遍历子view)
->measurechild(取出子元素的LayoutParams)
->getChildMeasure(创建子元素的MeasureSpec)
->child.measure(调用子view的measure测量子元素宽高)
2.继承自ViewGroup的LinearLayout(以测量竖直方向为例)
->onMeasure
->measureVertical(遍历子view)
->measureChildBeforeLayout():调用子view的measure,
mTotalLength变量存储LinearLayout竖直方向的初步高度
->每测量一个子view,mTotalLength = 子heigth + 子margin
注:
针对竖直的LinearLayout,在水平方向遵守view的测量过程;
在竖直方向测量过程和view不同:
match_parent/dp,px: 测量过程和view一致
wrap_content: 高度为所有字view高度 + 竖直方向子view的margin
1.3 在Activity的oncreate,onResume中无法正确获取view的宽/高?
原因:无法保证在Activity执行完oncreate时,view是否绘制完毕,如果view没绘制完,那view
宽高就是0.
3个解决办法:
(1)onWindowFoucusChanged(view已经初始化完毕)
onWindowFoucusChanged(boolean hasFoucus){
super.onWindowFoucusChanged(hasFoucus);
if(hasFoucus){
int width = view.getMeasuredWidth();
int height = view.getMeasuredHeigth();
}
}
(2)view.post(runnable)
post一个runnable小事件到消息队列的尾部,等待Looper来取runnable时候,view已经初始化完毕
protected void onstart() {
super.onstart();
view.post(new Runnable()) {
int width = view.getMeasureWidth();
int height = view.getMeasureHeigth();
}
}
(3)ViewTreeObserver(view树)
view树内部的view可见性改变,onGlobalLayout会被调用多次
ViewTreeObserver observewr = view.getViewTreeObserver();
observer.addOnGlobalLaoutListener(new OnGlobalLayoutListener) {
public void onGlobalLayout() {
view.getViewTreeObserver.removeGlobalOnlayoutListener(this);
int width = view.getMeasureWidth();
int height = view.getMeasuredHeight();
}
}
layout过程
1.
->setFrame设定view的4个顶点:mLeft,mRight,mTop,mBottom;确定了View 在父容器中的位置
->onLayout确定子元素的位置:layoutVertical/layoutHorizontal
->setChildFrame()确定子view位置
2.getMeasureHeight(测量高度) == getHeight(最终宽高);
重写view的layout方法就可以让测量宽高不等于最终宽高,但没实际意义
draw过程
->绘制背景background.draw(canvas);
->绘制自己onDraw()
->绘制children(dispatchDraw)
->绘制装饰onDrawScrollBars
dispatchDraw遍历子view的draw()->setWillNotDraw(不需要绘制的标记位)