安卓 ListView 固定高度项

9

有一个小问题。我想创建一个Android列表视图活动,其中列表中的所有项目都具有固定高度。

因此,我的项目布局(thread_item.xml)如下:

<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
              android:layout_width="fill_parent"
              android:layout_height="300dip"
        >
    <TextView android:id="@+id/thread_item_title"
              android:layout_width="wrap_content"
              android:layout_height="wrap_content"
              android:text="[dummy]"
              android:textSize="20dip"
              android:textStyle="bold"
            />
    <ImageView android:id="@+id/thread_first_image"
               android:layout_below="@id/thread_item_title"
               android:scaleType="centerInside"
               android:maxWidth="100dip"
               android:maxHeight="100dip"
               android:layout_width="wrap_content"
               android:layout_height="wrap_content"
               android:adjustViewBounds="true" />
    <TextView
            android:id="@+id/thread_item_preview"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:layout_alignTop="@id/thread_first_image"
            android:text="[dummy]"
            android:textSize="15dip">
    </TextView>
</RelativeLayout>

我将根元素的layout_height设置为300dip,并期望所有项具有相同的高度,但它们并没有。当我运行应用程序时,看起来高度具有wrap_content值。

此外,activity本身是这样的:

public class ThreadListActivity extends ListActivity {

    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);        
        /*
         Code to get items from the storage.
        */

        setListAdapter(new ThreadItemAdapter(this, R.layout.thread_item, itemsArray)));


        getListView().setOnItemClickListener(new AdapterView.OnItemClickListener() {
            @Override
            public void onItemClick(AdapterView<?> adapterView, View view, int position, long id) {
                /*Start new Activity. Unrelated stuff.*/
            }
        });
    }
}

我正在使用的适配器长这样:

public class ThreadItemAdapter extends ArrayAdapter<ThreadItem> {

    Activity context;
    List<ThreadItem> items;

    public ThreadItemAdapter(Activity context, int textViewResourceId, List<ThreadItem> items) {
        super(context, textViewResourceId, items);
        this.context = context;
        this.items = items;
    }

    @Override
    public View getView(int position, View convertView, ViewGroup parent) {
        LayoutInflater inf = this.context.getLayoutInflater();
        View result;
        if (convertView == null) {
            result = inf.inflate(R.layout.thread_item, null);
        } else {
            result = convertView;
        }

        TextView tbTitle = (TextView) result.findViewById(R.id.thread_item_title);
        TextView tbPreview = (TextView) result.findViewById(R.id.thread_item_preview);
        ImageView ivFirstImage = (ImageView) result.findViewById(R.id.thread_first_image);

        ThreadItem item = items.get(position);

        tbTitle.setText(item.getThreadTitle());
        ivFirstImage.setImageBitmap(item.getFirstImage());

        SimpleLeadingMarginSpan span = new SimpleLeadingMarginSpan(item.getFirstImage() != null ? 5 : 0, 115); // TODO: const
        SpannableString previewText = new SpannableString(item.getThreadPreview());
        previewText.setSpan(span, 0, previewText.length(), 0);
        tbPreview.setText(previewText);

        return result;
    }
}

我不明白为什么所有的列表项仍然包裹其内容而不保持300dip的高度。它们可能比300dip小或大。我正在使用android 2.3.3,在HTC Evo 3D设备和模拟器上测试(两者都显示相同的结果)。
事先感谢您的帮助。
更新:
感谢Miguel和Sam。解决方案是将textView的maxHeight设置为使我的列表项增长(即+id/thread_item_preview),并将RelativeLayout的minHeight设置为300dip,以防止其缩小。

如果我们将最小高度和最大高度设置为相同,当列表视图中的文本大小超过该高度时会发生什么? - Ashwin
7个回答

42
当为convertView进行膨胀(inflating)时,不要只是简单地...
result = inf.inflate(R.layout.thread_item, null);

做什么?

result = inf.inflate(R.layout.thread_item, parent, false);

所讨论的方法是inflater.inflate(int viewId, ViewGroup parent, boolean attachToRoot)--由于您没有遵守提供的parent(在这种情况下为ListView),无论您向listview项目提供什么尺寸,其默认值将被设置为layout_width=fill_parent,layout_height=wrap_content,而忽略了您在xml中指定的300dip高度。通过提供父视图并传递false,充气器将尊重300dip的高度,同时不将其附加到根(parent)。


1
你试过了吗?当你使用重载的inflate时,它会崩溃吗? - josephus
如果我这样做,一些项目在列表视图中不会显示,并且项目位置会改变。请帮帮我。 - W I Z A R D
@Wizard 听起来你的问题和 OP 的不同。如果在 SO 上找不到答案,考虑开一个新的问题。 - josephus
@josephus,我在这个链接找到了解决方案,感谢你的关注。 - W I Z A R D
这是正确的解决方案。更多阅读请点击此处 - Stephan Henningsen
显示剩余2条评论

31

如果将行中的所有子视图高度从wrap_content更改为match_parent会发生什么?


来自评论

您尝试过使用minHeightmaxHeight属性吗?例如:

android:minHeight="300dp"

你还应该观看Android的Romain Guy 讨论适配器和getView()中的效率


不行,结果还是一样的。最大的TextView id/thread_item_preview 仍然会将RelativeLayout拉伸得更大,如果内容高度小于300dips,则布局本身仍然会缩小。无论如何,感谢您的建议。 - Dmitriy
е“јпјЊдЅ е°ќиЇ•иї‡minHeightе’ЊmaxHeightе±ћжЂ§дє†еђ—пјџ - Sam
应该考虑到这一点,谢谢!我马上会发布一个解决方案。 - Dmitriy
@Sam:如果我们将最小高度和最大高度设置为相同的值,当列表视图中的文本大小超过该高度时会发生什么? - Ashwin
这是一个糟糕的答案。正确的解决方案是将父级传递给inflate函数。 - felipe

7

尝试更改这个

android:layout_height="300dip"

to

android:minHeight="300dip"

我在使用可扩展列表视图(ExpandableListView)时成功实现了这个功能,因此我认为它也适用于这种情况。


谢谢,这解决了两个问题中的一个——列表项缩小的问题,但不是增长的问题。无论如何,+1。 - Dmitriy
2
太好了 :) 顺便说一下,我觉得我打错字了。应该是“android:minHeight”,而不是“android:layout_minHeight”。 - Miguel Botón
对我来说有效,但我必须提到这个答案应该与@josephus的答案一起使用,即使用inflate(R.layout.thread_item,parent,false); - Rohan Kandwal
4
但是这些陈述是相等的。 - Tony
@RohanKandwal,你对这个问题的编辑没有任何意义,请重新审查一下。 - donfuxx

2
您可以通过为Min和Max指定相同的尺寸来实现此目标。这解决了我的问题。
<ImageView
    android:id="@+id/appIconImageView"
    android:layout_width="wrap_content"
    android:layout_height="wrap_content"
    android:layout_alignParentLeft="true"
    android:layout_alignParentTop="true"
    android:layout_marginBottom="8dp"
    android:layout_marginLeft="8dp"
    android:layout_marginTop="8dp"
    android:adjustViewBounds="true"
    android:maxHeight="50dp"
    android:maxWidth="50dp"
    android:minHeight="50dp"
    android:minWidth="50dp"
    android:src="@drawable/ic_launcher" />

enter image description here


0
<LinearLayout
    xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:minHeight="60dp"
    android:layout_height="match_parent"
    android:orientation="horizontal"
    android:gravity="center"
    android:weightSum="1"
    android:background="@color/main_color">

    <ImageView
        android:id="@+id/img_left_menu"
        android:layout_weight="0.4"
        android:focusable="false"
        android:maxHeight="50dp"
        android:maxWidth="50dp"
        android:minHeight="50dp"
        android:minWidth="50dp"
        android:layout_width="0dp"
        android:scaleType="centerInside"
        android:adjustViewBounds="true"
        android:layout_height="wrap_content"/>

    <TextView
        android:id="@+id/tx_left_menu"
        android:layout_width="0dp"
        android:textSize="18dp"
        android:layout_gravity="center_vertical"
        android:layout_weight="0.6"
        android:singleLine="true"
        android:layout_height="wrap_content"
        android:focusable="false"
        />
</LinearLayout>

0

如果我没记错的话,listView 会自动修改自定义视图的 LayoutParams 为 wrap_content,wrap_content。但是,上面的答案是正确的,如果你设置了 minHeight 或 minWidth,它将非常有效。


0
尝试将内容 TextView 的高度设置为 0dp,然后将其 layout_alignParentBottom 设置为 true

1
在这种情况下,TextView 折叠并且没有文本可见。 - Dmitriy

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接