美文网首页Android知识Android开发Android技术知识
android解决无法设定listview的item高度

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

作者: Little_Mango | 来源:发表于2016-10-13 20:03 被阅读2067次

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

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

    • 在item的根布局中设置minHeight = "100dp",但是设置了minWidth = "100dp",也无法使item内容的宽度为100dp,原因后面会说到。
    • 再用一层父控件包裹,在第二层父控件设置android:layout_height = "100dp",android:layout_width = "100dp"布局如下:
    <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罢了。

    相关文章

      网友评论

        本文标题:android解决无法设定listview的item高度

        本文链接:https://www.haomeiwen.com/subject/auiwyttx.html