[b][color=red]MeasureSpec封装了父布局传递给子布局的布局要求,每个MeasureSpec代表了一组宽度和高度的要求。[/color][/b]一个MeasureSpec[b]由大小和模式组成[/b]。
它有三种模式:
[color=red]UNSPECIFIED(未指定):[/color][b]父元素不对子元素施加任何束缚[/b],子元素可以得到任意想要的大小;
[b][color=red]EXACTLY(完全):[/color][/b]父元素决定子元素的确切大小,子元素将被限定在给定的边界里而忽略它本身大小;
[color=red]AT_MOST(至多):[/color]子元素至多达到指定大小的值。
它常用的三个函数:
[color=red]1.static int getMode(int measureSpec)[/color]:根据提供的测量值(格式)提取模式(上述三个模式之一)
[color=red]2.static int getSize(int measureSpec)[/color]:根据提供的测量值(格式)提取大小值(这个大小也就是我们通常所说的大小)
[color=red]3.static int makeMeasureSpec(int size,int mode)[/color]:[color=blue][b]根据提供的大小值和模式创建一个测量值(格式)[/b][/color]
[b]这个类的使用呢,通常在view组件的onMeasure方法里面调用。[/b]
看看它的使用吧,ListView.measureItem(View child)
首先一个我们常用到的一个有用的函数,View.resolveSize(int size,int measureSpec)
[b][color=red]注意,使用EXACTLY和AT_MOST通常是一样的效果。[/color][/b]
如果你要区别他们,那么你就要使用上面的函数View.resolveSize(int size,int measureSpec)返回一个size值,然后使用你的view调用setMeasuredDimension(int,int)函数。
继承View,实现自己想要的组件,那么需要使用到[b][color=red]setMeasuredDimension[/color][/b]这个方法,这个方法决定了当前View的大小,请看代码:
它有三种模式:
[color=red]UNSPECIFIED(未指定):[/color][b]父元素不对子元素施加任何束缚[/b],子元素可以得到任意想要的大小;
[b][color=red]EXACTLY(完全):[/color][/b]父元素决定子元素的确切大小,子元素将被限定在给定的边界里而忽略它本身大小;
[color=red]AT_MOST(至多):[/color]子元素至多达到指定大小的值。
它常用的三个函数:
[color=red]1.static int getMode(int measureSpec)[/color]:根据提供的测量值(格式)提取模式(上述三个模式之一)
[color=red]2.static int getSize(int measureSpec)[/color]:根据提供的测量值(格式)提取大小值(这个大小也就是我们通常所说的大小)
[color=red]3.static int makeMeasureSpec(int size,int mode)[/color]:[color=blue][b]根据提供的大小值和模式创建一个测量值(格式)[/b][/color]
[b]这个类的使用呢,通常在view组件的onMeasure方法里面调用。[/b]
看看它的使用吧,ListView.measureItem(View child)
首先一个我们常用到的一个有用的函数,View.resolveSize(int size,int measureSpec)
public static int resolveSize(int size, int measureSpec) {
int result = size;
int specMode = MeasureSpec.getMode(measureSpec);
int specSize = MeasureSpec.getSize(measureSpec);
switch (specMode) {
case MeasureSpec.UNSPECIFIED:
result = size;
break;
case MeasureSpec.AT_MOST:
result = Math.min(size, specSize);
break;
case MeasureSpec.EXACTLY:
result = specSize;
break;
}
return result;
}
private void measureItem(View child) {
ViewGroup.LayoutParams p = child.getLayoutParams();
if (p == null) {
p = new ViewGroup.LayoutParams(
ViewGroup.LayoutParams.MATCH_PARENT,
ViewGroup.LayoutParams.WRAP_CONTENT);
}
int childWidthSpec = ViewGroup.getChildMeasureSpec(mWidthMeasureSpec,
mListPadding.left + mListPadding.right, p.width);
int lpHeight = p.height;
int childHeightSpec;
if (lpHeight > 0) {
childHeightSpec = MeasureSpec.makeMeasureSpec(lpHeight, MeasureSpec.EXACTLY);
} else {
childHeightSpec = MeasureSpec.makeMeasureSpec(0, MeasureSpec.UNSPECIFIED);
}
child.measure(childWidthSpec, childHeightSpec);
}
[b][color=red]注意,使用EXACTLY和AT_MOST通常是一样的效果。[/color][/b]
如果你要区别他们,那么你就要使用上面的函数View.resolveSize(int size,int measureSpec)返回一个size值,然后使用你的view调用setMeasuredDimension(int,int)函数。
继承View,实现自己想要的组件,那么需要使用到[b][color=red]setMeasuredDimension[/color][/b]这个方法,这个方法决定了当前View的大小,请看代码:
@Override
protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
String tag="onMeasure";
Log.e(tag,"Scroll View on measure...");
setMeasuredDimension(width, height);
}