Android知识Android开发Android技术知识

android解决无法设定listview的item高度

2016-10-13  本文已影响2067人  Little_Mango

首先说明一下,在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

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:orientation="vertical"
    android:layout_width="100dp"
    android:layout_height="100dp">
    <TextView
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:gravity="center"
        android:id="@+id/tv1"/>
</LinearLayout>

然后在getView方法中通过以下代码inflate

View.inflate(RecycleViewActivity.this,R.layout.item,null)

虽然我们给item设置了100dp的高度和宽度,但是实际效果却是这样的:

Snip20161013_15.png

很明显,在这里设置宽高不起作用!
所以为了解决这个问题,百度以后,有下面两种方法

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="xxx"
    android:layout_height="xxx"
    android:orientation="vertical">
    <LinearLayout
        android:layout_width="100dp"
        android:layout_height="100dp"
        android:orientation="vertical">
        <TextView
            android:id="@+id/tv1"
            android:layout_width="match_parent"
            android:layout_height="match_parent"
            android:gravity="center" />
    </LinearLayout></LinearLayout>

以上两种"解决方法"只是有缺陷的投机取巧,因为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文件就相当于下面的例子:

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="wrap_content"
    android:orientation="vertical">
    <TextView
        android:id="@+id/tv1"
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:gravity="center" />
</LinearLayout>

由于layout_width = "match_parent",所以前面通过设置min_width = "100dp"是无法起作用的。


RecycleView的item也是类似的结果,只不过RecycleView在发现item的layoutParams == null的时候,默认生成的宽高约束都是wrap_content罢了。

上一篇下一篇

猜你喜欢

热点阅读