首先说明一下,在Android中,子控件的尺寸是由父控件决定的。
/*父控件的onLayout方法*/
@Override
protected void onLayout(boolean changed, int l, int t, int r, int b) {
final int count = getChildCount();
for (int i = 0; i < count; i++) {
View child = getChildAt(i);
if (child.getVisibility() != GONE) {
RelativeLayout.LayoutParams st =
(RelativeLayout.LayoutParams) child.getLayoutParams();
/*调用子类的layout方法,layout方法会继续调用onLayout方法*/
child.layout(st.mLeft, st.mTop, st.mRight, st.mBottom);
}
}
}
从上面源码不仅可以看出子控件的尺寸由父控件所决定,而且还可以看出子控件的尺寸信息是存储在LayoutParams中的。
当我们构建了如下一个item.xml
然后在getView方法中通过以下代码inflate
View.inflate(RecycleViewActivity.this,R.layout.item,null)
虽然我们给item设置了100dp的高度和宽度,但是实际效果却是这样的:
很明显,在这里设置宽高不起作用!
所以为了解决这个问题,百度以后,有下面两种方法
- 在item的根布局中设置minHeight = "100dp",但是设置了minWidth = "100dp",也无法使item内容的宽度为100dp,原因后面会说到。
- 再用一层父控件包裹,在第二层父控件设置android:layout_height = "100dp",android:layout_width = "100dp"布局如下:
以上两种"解决方法"只是有缺陷的投机取巧,因为minHeight和第二层父控件无法实现match_parent效果,所以这两种方法看一下就好。
文章开头说了,子控件的尺寸由父控件决定,并且存储在子控件的layoutParams属性中,下面看一下View.inflate()方法的源码
public static View inflate(Context context, @LayoutRes int resource, ViewGroup root) {
LayoutInflater factory = LayoutInflater.from(context);
return factory.inflate(resource, root);
}
public View inflate(@LayoutRes int resource, @Nullable ViewGroup root) {
return inflate(resource, root, root != null);
}
public View inflate(@LayoutRes int resource, @Nullable ViewGroup root, boolean attachToRoot) {
try {
return inflate(parser, root, attachToRoot);
} finally {
parser.close();
}
}
public View inflate(XmlPullParser parser, @Nullable ViewGroup root, boolean attachToRoot) {
View temp = createViewFromTag(root, name, inflaterContext, attrs);
ViewGroup.LayoutParams params = null;
//只有当root不为null的时候,才会创建layoutParams属性
if (root != null) {
params = root.generateLayoutParams(attrs);
if (!attachToRoot) {
// Set the layout params for temp if we are not
// attaching. (If we are, we use addView, below)
temp.setLayoutParams(params);
}
}
}
由以上源码可以看出,由于我们调用View.inflate()方法传递的parent参数是null,所以导致item的layoutParams属性为null。所以真正解决无法设置ListView的Item的高度的方法是:使root != null
所以我们应该使用以下方法去inflate一个xml文件
View view = LayoutInflater.from(context).inflate(R.layout.item,parent,false);
到此为止,如何给ListView的Item设置高度的问题便解决了。
不过还有一个疑问,当我们使用错误的inflate方法的时候,为什么会呈现第一张图片的那种结果?
在AbsListView中有下面这三个方法
View obtainView(int position, boolean[] isScrap) {
、、、
setItemViewLayoutParams(child, position);
、、、
}
private void setItemViewLayoutParams(View child, int position) {
final ViewGroup.LayoutParams vlp = child.getLayoutParams();
LayoutParams lp;
if (vlp == null) {
lp = (LayoutParams) generateDefaultLayoutParams();
} else if (!checkLayoutParams(vlp)) {
lp = (LayoutParams) generateLayoutParams(vlp);
} else {
lp = (LayoutParams) vlp;
}
if (mAdapterHasStableIds) {
lp.itemId = mAdapter.getItemId(position);
}
lp.viewType = mAdapter.getItemViewType(position);
if (lp != vlp) {
child.setLayoutParams(lp);
}
}
@Override
protected ViewGroup.LayoutParams generateDefaultLayoutParams() {
return new AbsListView.LayoutParams(ViewGroup.LayoutParams.MATCH_PARENT,
ViewGroup.LayoutParams.WRAP_CONTENT, 0);
}
通过代码可以看到,如果Item的layoutParams属性为空,那么默认为其设置一个宽度为match_parent,高度为wrap_content的layoutParams属性。所以如果是使用View.inflate(context,res,parent) inflate出来的view,其实其XML文件就相当于下面的例子:
由于layout_width = "match_parent",所以前面通过设置min_width = "100dp"是无法起作用的。
RecycleView的item也是类似的结果,只不过RecycleView在发现item的layoutParams == null的时候,默认生成的宽高约束都是wrap_content罢了。