使用ListAdapter填充ScrollView布局中的LinearLayout


95

我面临一个非常常见的问题:我布置了一个活动,现在事实证明它应该在其中显示一些项目ScrollView。正常的方式做到这一点是使用现有的ListAdapter,它连接到一个ListViewBOOM我有我的产品清单。

但是您不应该在其中嵌套嵌套ListViewScrollView因为它会使滚动变得更糟-即使Android Lint也抱怨它。

所以这是我的问题:

如何将a连接ListAdapter到a LinearLayout或类似的东西?

我知道此解决方案无法扩展很多项目,但是我的列表很短(<10个项目),因此实际上不需要重用视图。在性能方面,我可以将所有视图直接放入LinearLayout

我想到的一种解决方案是将现有的活动布局放在的headerView部分中ListView。但这感觉就像在滥用这种机制,因此我正在寻找更清洁的解决方案。

有想法吗?

更新:为了激发正确的方向,我添加了一个示例布局来展示我的问题:

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
              android:id="@+id/news_detail_layout"
              android:layout_width="fill_parent"
              android:layout_height="fill_parent"
              android:orientation="vertical"
              android:visibility="visible">


    <ScrollView
            android:layout_width="fill_parent"
            android:layout_height="fill_parent"
            android:background="#FFF"
            >

        <LinearLayout
                android:layout_width="fill_parent"
                android:layout_height="fill_parent"
                android:orientation="vertical"
                android:paddingLeft="@dimen/news_detail_layout_side_padding"
                android:paddingRight="@dimen/news_detail_layout_side_padding"
                android:paddingTop="@dimen/news_detail_layout_vertical_padding"
                android:paddingBottom="@dimen/news_detail_layout_vertical_padding"
                >

            <TextView
                    android:id="@+id/news_detail_date"
                    android:layout_height="wrap_content"
                    android:layout_width="fill_parent"
                    android:gravity="center_horizontal"
                    android:text="LALALA"
                    android:textSize="@dimen/news_detail_date_height"
                    android:textColor="@color/font_black"
                    />

            <Gallery
                    android:id="@+id/news_detail_image"
                    android:layout_height="wrap_content"
                    android:layout_width="fill_parent"
                    android:paddingTop="5dip"
                    android:paddingBottom="5dip"
                    />

            <TextView
                    android:id="@+id/news_detail_headline"
                    android:layout_height="wrap_content"
                    android:layout_width="fill_parent"
                    android:gravity="center_horizontal"
                    android:text="Some awesome headline"
                    android:textSize="@dimen/news_detail_headline_height"
                    android:textColor="@color/font_black"
                    android:paddingTop="@dimen/news_detail_headline_paddingTop"
                    android:paddingBottom="@dimen/news_detail_headline_paddingBottom"
                    />

            <TextView
                    android:id="@+id/news_detail_content"
                    android:layout_height="wrap_content"
                    android:layout_width="fill_parent"
                    android:text="Here comes a lot of text so the scrollview is really needed."
                    android:textSize="@dimen/news_detail_content_height"
                    android:textColor="@color/font_black"
                    />

            <!---
                HERE I NEED THE LIST OF ITEMS PROVIDED BY THE EXISTING ADAPTER. 
                They should be positioned at the end of the content, so making the scrollview smaller is not an option.
            ---->                        

        </LinearLayout>
    </ScrollView>
</LinearLayout>

更新2我更改了标题,以使其更易于理解(谢绝投票,哦!)。


1
您是否找到解决此问题的好方法?我看到Google在其Play商店中使用它进行评论。有人知道他们是怎么做的吗?
Zapnologica

Answers:


96

您可能应该手动将商品添加到LinearLayout

LinearLayout layout = ... // Your linear layout.
ListAdapter adapter = ... // Your adapter.

final int adapterCount = adapter.getCount();

for (int i = 0; i < adapterCount; i++) {
  View item = adapter.getView(i, null, null);
  layout.addView(item);
}

编辑:当我需要显示大约200个非平凡的列表项时,我拒绝了这种方法,它非常慢-Nexus 4需要大约2秒钟才能显示我的“列表”,这是不可接受的。因此,我转向了带有标题的Flo方法。它的工作速度更快,因为列表视图是在用户滚动时按需创建的,而不是在创建视图时创建的。

简历:在布局中手动添加视图比较容易编写代码(因此可能会减少移动的零件和错误),但是会遇到性能问题,因此,如果您拥有50个以上的视图,建议使用标头方法。

例。基本上,活动(或片段)布局会转换为以下形式(不再需要ScrollView):

<ListView xmlns:android="http://schemas.android.com/apk/res/android"
  android:id="@+id/my_top_layout"
  android:layout_width="fill_parent"
  android:layout_height="fill_parent"/>

然后,在onCreateView()(我将使用一个片段示例)中,您需要添加一个标题视图,然后设置一个适配器(假设标题资源ID为header_layout):

ListView listView = (ListView) inflater.inflate(R.layout.my_top_layout, container, false);
View header = inflater.inflate(R.layout.header_layout, null);
// Initialize your header here.
listView.addHeaderView(header, null, false);

BaseAdapter adapter = // ... Initialize your adapter.
listView.setAdapter(adapter);

// Just as a bonus - if you want to do something with your list items:
view.setOnItemClickListener(new AdapterView.OnItemClickListener() {
  @Override
  public void onItemClick(AdapterView<?> parent, View view, int position, long id) {
    // You can just use listView instead of parent casted to ListView.
    if (position >= ((ListView) parent).getHeaderViewsCount()) {
      // Note the usage of getItemAtPosition() instead of adapter's getItem() because
      // the latter does not take into account the header (which has position 0).
      Object obj = parent.getItemAtPosition(position);
      // Do something with your object.
    }
  }
});

是的,这是我的第二个主意,但我不确定,ListAdapter而且ListView屏幕背后还有更多不可思议的地方,因此我不手动进行。
Stefan Hoth 2012年

当然,ListView确实具有一些魔力,至少在元素之间具有分隔符,我想您必须手动添加它们。ListAdapter(如果正确实现的话)不应再做任何魔术。
smok 2012年

2
这对我来说是反对票。内存问题有人吗?我们使用转接器是有原因的
凯文·帕克

1
使用适配器的notifyDataSetChanged方法来更新视图是否还有其他人遇到问题?这在我连接到一个不同的适配器上可以很好ListView地工作,但是似乎无法与我连接到一个的适配器一起使用LinearLayout
jokeefe

2
这是我讨厌android的原因之一。我不明白为什么您必须实施复杂的解决方案,否则会遇到性能问题。
IARI

6

我会坚持使用标题视图解决方案。没有错。目前,我使用完全相同的方法实施一项活动。

显然,“项目部分”比静态的(动态项目数与固定项目数等)更具动态性,否则您根本不会考虑使用适配器。因此,当您需要适配器时,请使用ListView。

最后,实现从适配器填充LinearLayout的解决方案无非就是使用自定义布局构建ListView。

只是我的2美分。


这种方法的一个缺点是,您必须将几乎整个活动布局(请参见上文)移动到另一布局中,以便可以将其引用为ListHeaderView并创建仅包含的另一布局ListView。不确定我是否喜欢此补丁程序。
Stefan Hoth 2012年

1
是的,我知道您的意思,在开始使用这种方法之前,我也不喜欢将活动布局分为多个文件的想法。但是,当我看到整体布局看起来像我期望的样子时,我并不介意分离,因为它仅被分成两个文件。您必须了解的一件事是,ListView不能有一个空视图,因为当没有列表项时,它将隐藏您的列表标题。
2012年

3
另外,如果您要在一个页面上有多个列表,则这实际上不起作用。有没有人有一个自定义适配器视图的示例,该视图将项目输出到“平面”列表中,即不滚动的列表。
murtuza 2014年

0

将视图设置为main.xml onCreate,然后从row.xml进行膨胀

main.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="450dp" >

        <ListView
            android:id="@+id/mainListView"
            android:layout_width="fill_parent"
            android:layout_height="fill_parent"
            android:layout_above="@+id/size"
            android:layout_below="@+id/editText1"
            android:gravity="fill_vertical|fill_horizontal"
            android:horizontalSpacing="15dp"
            android:isScrollContainer="true"
            android:numColumns="1"
            android:padding="5dp"
            android:scrollbars="vertical"
            android:smoothScrollbar="true"
            android:stretchMode="columnWidth" >

</ListView>

    <TextView
        android:id="@+id/size"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_alignParentBottom="true"
        android:layout_alignParentLeft="true"
        android:layout_alignParentRight="true"
        android:background="#ff444444"
        android:gravity="center"
        android:text="TextView"
        android:textColor="#D3D3D3"
        android:textStyle="italic" />

    </EditText>

</RelativeLayout> 

row.xml

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout
  xmlns:android="http://schemas.android.com/apk/res/android"
  android:layout_width="fill_parent"
  android:layout_height="fill_parent" 
  android:paddingTop="3dp">

<TextView
  android:id="@+id/rowTextView"
  android:layout_width="0dip"
  android:layout_height="41dp"
  android:layout_margin="4dp"
  android:layout_weight="2.83"
  android:ellipsize="end"
  android:gravity="center_vertical"
  android:lines="1"
  android:text="John Doe"
  android:textColor="@color/color_white"
  android:textSize="23dp" >
</TextView>

</LinearLayout>

我想你想念这个问题。他不想用LinearLayouts填充列表。
Flo 2012年

“如何将ListAdapter连接到LinearLayout或类似的东西?” 只是回答什么OP问
fasheikh

2
不,他不想使用ListView。他只想使用适配器,并使用它来填充带有条目的LinearLayout。
Flo 2012年

感谢您的回答,但@Flo是正确的。我不能使用ListView,因为外部布局已经是ScrollView。请检查我的文字和示例布局。
Stefan Hoth 2012年

为什么需要scrollView?
fasheikh 2012年

0

我用以下这与复制功能的适配器代码ViewGroupTabLayout。这样做的好处是,如果您更改列表并再次绑定,则只会影响更改的项目:

用法:

val list = mutableListOf<Person>()
layout.bindChildren(list, { it.personId }, { bindView(it) }, {d, t ->bindView(d, t)})
list.removeAt(0)
list+=newPerson
layout.bindChildren(list, { it.personId }, { bindView(it) }, {d, t ->bindView(d, t)})

对于ViewGroups

fun <Item, Key> ViewGroup.bindChildren(items: List<Item>, id: (Item) -> Key, view: (Item) -> View, bind: (Item, View) -> Unit) {
    val old = children.map { it.tag as Key }.toList().filter { it != null }
    val new = items.map(id)

    val add = new - old
    val remove = old - new
    val keep = new.intersect(old)

    val tagToChildren = children.associateBy { it.tag as Key }
    val idToItem = items.associateBy(id)

    remove.forEach { tagToChildren[it].let { removeView(it) } }
    keep.forEach { bind(idToItem[it]!!, tagToChildren[it]!!) }
    add.forEach { id -> view(idToItem[id]!!).also { it.tag = id }.also { addView(it, items.indexOf(idToItem[id])) } }
}

因为TabLayout我有这个:

fun <Item, Key> TabLayout.bindTabs(items: List<Item>, toKey: (Item) -> Key, tab: (Item) -> TabLayout.Tab, bind: (Item, TabLayout.Tab) -> Unit) {
    val old = (0 until tabCount).map { getTabAt(it)?.tag as Key }
    val new = items.map(toKey)

    val add = new - old
    val remove = old - new
    val keep = new.intersect(old)

    val tagToChildren = (0 until tabCount).map { getTabAt(it) }.associateBy { it?.tag as Key }
    val idToItem = items.associateBy(toKey)

    remove.forEach { tagToChildren[it].let { removeTab(it) } }
    keep.forEach { bind(idToItem[it]!!, tagToChildren[it]!!) }
    add.forEach { key -> tab(idToItem[key]!!).also { it.tag = key }.also { addTab(it, items.indexOf(idToItem[key])) } }
}
By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.