先看看 view 基于整个屏幕分布的概况
20150528211309106.png
首先要明确view是屏幕绘制的入口
那么View的绘制是从哪里开始的呢,我们知道每个Activity 均会创建一个PhoneWindow对象,
是Activity和整个View系统交互的接口,每个Window都对应着一个View和一个ViewRootImpl,
Window和View通过ViewRootImpl来建立联系,对于Activity来说,ViewRootImpl是连接
WindowManager和DecorView的纽带,绘制的入口是由ViewRootImpl的performTraversals方法
来发起Measure,Layout,Draw等流程的。。
再来个图来补充一下PhoneWindow:
503290-20151104115213477-1270385950.jpg整个流程的大致流程图:
20150529090922419.png看看源码:
private void performTraversals() {
......
//最外层的根视图的widthMeasureSpec和heightMeasureSpec由来
//lp.width和lp.height在创建ViewGroup实例时等于MATCH_PARENT
int childWidthMeasureSpec = getRootMeasureSpec(mWidth, lp.width);
int childHeightMeasureSpec = getRootMeasureSpec(mHeight, lp.height);
......
mView.measure(childWidthMeasureSpec, childHeightMeasureSpec);
......
mView.layout(0, 0, mView.getMeasuredWidth(), mView.getMeasuredHeight());
......
mView.draw(canvas);
......
}
概况了解清楚了那么,来具体看看
measure()、layout()、draw()的具体流程
measure()
首先看看其整体的过程
20150529163050000.png
开始measure之前要明确一个概念MeasureSpec(测量规格),MeasureSpec是一个大小跟模式的组合值,控件自身的大小是受到MeasureSpec下的mode,size两个值共同来决定的
@Override
protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
super.onMeasure(widthMeasureSpec, heightMeasureSpec);
widthMode=MeasureSpec.getMode(widthMeasureSpec);
hightMode=MeasureSpec.getMode(heightMeasureSpec);
width = MeasureSpec.getSize(widthMeasureSpec);
hight = MeasureSpec.getSize(heightMeasureSpec);
}
就是在子控件正要放置自己的位置时,父控件会通过MeasureSpec这个对象来问子控件,“你想要用多大地方啊?”;然后子控件通过一系列的操作来放置控件!
MeasureSpec的mode一共有三种模式:
UPSPECIFIED : 父容器对于子容器没有任何限制,子容器想要多大就多大
EXACTLY: 父容器已经为子容器设置了尺寸,子容器应当服从这些边界,不论子容器想要多大的空间。
AT_MOST:子容器可以是声明大小内的任意大小
根据mode,size 共同决定的子控件最终的大小:
父View的MeasureSpec 是EXACTLY,说明父View的大小是确切的,(确切的意思很好理解,如果一个View的MeasureSpec 是EXACTLY,那么它的size 是多大,最后展示到屏幕就一定是那么大)。
父View的MeasureSpec 是AT_MOST,说明父View的大小是不确定,最大的大小是MeasureSpec 的size值,不能超过这个值。
503290-20151119171702515-1341567594.png父View的MeasureSpec 是UNSPECIFIED(未指定),表示没有任何束缚和约束,不像AT_MOST表示最大只能多大,不也像EXACTLY表示父View确定的大小,子View可以得到任意想要的大小,不受约束
好了 这是大概的流程,下面来个具体的示例:
布局如下:
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:id="@+id/linear"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_marginTop="50dp"
android:background="@android:color/holo_blue_dark"
android:paddingBottom="70dp"
android:orientation="vertical">
<TextView
android:id="@+id/text"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:background="@color/material_blue_grey_800"
android:text="TextView"
android:textColor="@android:color/white"
android:textSize="20sp" />
<View
android:id="@+id/view"
android:layout_width="match_parent"
android:layout_height="150dp"
android:background="@android:color/holo_green_dark" />
</LinearLayout>
屏幕显示样式如下:
966283-4a11f92ac8c5e224.png966283-4096801e91e2eccc.png整个图是一个DecorView,DecorView可以理解成整个页面的View,DecorView是一个FrameLayout,包含两个子View,一个id=statusBarBackground的View和一个是LineaLayout,id=statusBarBackground的View,而这个LinearLayout比较重要,它包含一个title和一个content,title很好理解其实就是TitleBar或者ActionBar,content 就更简单了,setContentView()方法你应该用过吧,android.R.id.content 你应该听过吧,没错就是它,content是一个FrameLayout,你写的页面布局通过setContentView加进来就成了content的直接子View。
绘制的流程是这样、详情请参考
http://www.jianshu.com/p/5a71014e7b1b
网友评论