用代码实现XML布局,可以明确看出Android内部加载View的过程:
1、创建view,通过setXXX方法设置view的固有属性,通过layoutparams内部类设置view相对父控件的布局属性,
2、得到view的层次结构后开始绘制。
所以,在界面中绘制出view,实际上可以分为两个阶段,一是设置view属性并将其添加到View的层次结构中,等待android对View层次结构解析;二是android最终拿到view的整体层次结构后,通过测量每个view大小(measure())、安排布局位置(layout())、绘制(draw())三个步骤,最终实现在界面中绘制出view。本文将分析RelativeLayout源码,探究其实现此两个过程的具体细节。
一、设置view属性过程
view设置固有属性方法:view.setXXX(),比如设置背景view.setBackgroud();
view设置其在布局中的布局属性方法,是利用ViewGroup中的内部类LayoutParams,并通过view.setLayoutParams(),将LayoutParams属性添加到view中。
通过上面两步之后,view的属性设置完毕,通过ViewGroup中的AddView()方法,将view添加到布局层次结构中。
查看AddView()源码,一共有5个AddView():
public void addView(View child) {
addView(child, -1);
}
public void addView(View child, int index) {
LayoutParams params = child.getLayoutParams();
//如果没有设置params,则默认生成params
//有两种方式生成params,一种是在XML布局中设置params属性,一种是在代
//码中newLayoutParams();
if (params == null) {
params = generateDefaultLayoutParams();
if (params == null) {
throw new IllegalArgumentException("generateDefaultLayoutParams()cannot return null");
}
}
addView(child, index, params);
}
public void addView(View child, int width, int height) {
final LayoutParams params = generateDefaultLayoutParams();
params.width = width;
params.height = height;
addView(child, -1, params);
}
publicvoid addView(View child, LayoutParams params) {
addView(child, -1, params);
}
public void addView(View child, int index, LayoutParams params) {
if (DBG) {
System.out.println(this + " addView");
}
requestLayout();
invalidate(true);
addViewInner(child, index, params, false);
}
分析源码发现,最终都是调用void addView(View child, int index, LayoutParams params),再分析内部函数addViewInner(child,index, params, false),实际执行的就是child.setLayoutParams(params)。
private void addViewInner(View child, int index, LayoutParams params,
boolean preventRequestLayout) {
//检查params是否为null
if (!checkLayoutParams(params)) {
params = generateLayoutParams(params);
}
// preventRequestLayout=true,实际执行的就是child.setLayoutParams(params)
if (preventRequestLayout) {
child.mLayoutParams = params;
} else {
child.setLayoutParams(params);
}
}
有三种方式生成params,一种是在代码中主动new LayoutParams(),一种是在XML布局中设置params属性通过generateLayoutParams(params)生成,分析AddView()发现,若没有通过前面两种生成 params,则调用generateDefaultLayoutParams()生成params。ViewGroup中源码如下:
public LayoutParams generateLayoutParams(AttributeSet attrs) {
return new LayoutParams(getContext(), attrs);
}
protected LayoutParams generateLayoutParams(ViewGroup.LayoutParams p) {
return p;
}
protected LayoutParams generateDefaultLayoutParams() {
return new LayoutParams(LayoutParams.WRAP_CONTENT, LayoutParams.WRAP_CONTENT); }
所以若想在自定义高级控件中设置额外的params属性,只需要对generateDefaultLayoutParams、generateLayoutParams做文章,而若要设置自有属性,只需要对AddView做文章即可。
一个简单地包装RelativeLayout的Demo,将添加到RelativeLayout中的每个view设置params及固有属性:
package com.study.designview;
import android.R.color;
import android.content.Context;
import android.util.AttributeSet;
import android.view.View;
import android.widget.RelativeLayout;
import com.example.heighview.R;
public class Relative extends RelativeLayout {
public Relative(Context context, AttributeSet attrs) {
super(context, attrs);
}
@Override
public LayoutParams generateLayoutParams(AttributeSet attrs) {
LayoutParams params=super.generateLayoutParams(attrs);
//额外属性
params.topMargin=300;
return params;
}
@Override
protected android.view.ViewGroup.LayoutParams generateDefaultLayoutParams() {
LayoutParams params=new LayoutParams(
LayoutParams.WRAP_CONTENT,
LayoutParams.WRAP_CONTENT);
params.topMargin=300;
return params;
}
@Override
protected boolean addViewInLayout(View child, int index,
android.view.ViewGroup.LayoutParams params) {
//额外属性
child.setBackgroundResource(R.drawable.eqc);
return super.addViewInLayout(child, index, params);
}
}
二、绘制view
详解见下一篇