如何使用里面的 ListView 滚动 NestedScrollView?

Posted

技术标签:

【中文标题】如何使用里面的 ListView 滚动 NestedScrollView?【英文标题】:How to scroll NestedScrollView with a ListView inside? 【发布时间】:2020-03-05 23:49:39 【问题描述】:

我正在创建一个活动,我需要在其中显示几个元素和一个列表视图,并且我希望当用户滚动时我想滚动所有活动,而不仅仅是列表视图,为此,我实现了一个 NestedScrollView,但是滚动不起作用,它仍然没有滚动,我需要一个属性吗?还是有更好的方法来做我需要做的事情?

我附上我的 XML 代码:

<androidx.core.widget.NestedScrollView xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_
    android:layout_
    android:fillViewport="true">

    <androidx.constraintlayout.widget.ConstraintLayout
        android:layout_
        android:layout_>

       <!-- Other items -->

        <ListView
            android:id="@+id/lvAC"
            android:layout_
            android:layout_
            app:layout_constraintEnd_toEndOf="parent"
            app:layout_constraintStart_toStartOf="parent" />


    </androidx.constraintlayout.widget.ConstraintLayout>


</androidx.core.widget.NestedScrollView>

【问题讨论】:

【参考方案1】:

试试这个方法:

<?xml version="1.0" encoding="utf-8"?>
<androidx.core.widget.NestedScrollView xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    android:layout_
    android:layout_
    android:fillViewport="true"
    >
    <androidx.constraintlayout.widget.ConstraintLayout
        android:layout_
        android:layout_>
        <View
            android:id="@+id/view"
            app:layout_constraintTop_toTopOf="parent"
            android:layout_
            android:layout_
            android:background="@drawable/ic_launcher_background"/>


        <ListView
            android:id="@+id/lvAC"
            android:layout_
            android:layout_
            app:layout_constraintEnd_toEndOf="parent"
            app:layout_constraintStart_toStartOf="parent"
            app:layout_constraintBottom_toBottomOf="parent"
            app:layout_constraintTop_toBottomOf="@+id/view"
            android:nestedScrollingEnabled="true"
            />
    </androidx.constraintlayout.widget.ConstraintLayout>
</androidx.core.widget.NestedScrollView>

并将此方法设置为您的 listview

public static void setListViewHeightBasedOnChildren(ListView listView) 
        ListAdapter listAdapter = listView.getAdapter();
        if (listAdapter == null) 
            // pre-condition
            return;
        

        int totalHeight = 0;
        for (int i = 0; i < listAdapter.getCount(); i++) 
            View listItem = listAdapter.getView(i, null, listView);
            listItem.measure(0, 0);
            totalHeight += listItem.getMeasuredHeight();
        

        ViewGroup.LayoutParams params = listView.getLayoutParams();
        params.height = totalHeight + (listView.getDividerHeight() * (listAdapter.getCount() - 1));
        listView.setLayoutParams(params);
        listView.requestLayout();
    

【讨论】:

感谢您的回答,但我已经实现了该方法,但它对我不起作用。【参考方案2】:

在 ScrollView 中嵌套 ListView 是有问题的,因为它会导致多个垂直滚动行为,这会让用户感到困惑。相反,请使用可以提供多个 ViewType 的单个 RecyclerView。

您的 RecyclerView 适配器应该实现getItemViewType,返回您需要显示的不同视图的数量。例如,如果您有:

<androidx.core.widget.NestedScrollView xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_
    android:layout_
    android:fillViewport="true">

    <androidx.constraintlayout.widget.ConstraintLayout
        android:layout_
        android:layout_>

        <!-- Other items -->
        <View
            android:id="@+id/view1"
            android:layout_
            android:layout_
            app:layout_constraintTop_toTopOf="parent"
            app:layout_constraintStart_toStartOf="parent"
            />
        <View
            android:id="@+id/view2"
            android:layout_
            android:layout_
            app:layout_constraintTop_toBottomOf="@id/view1"
            app:layout_constraintStart_toStartOf="parent"
            />

        <ListView
            android:id="@+id/lvAC"
            android:layout_
            android:layout_
            app:layout_constraintTop_toBottomOf="@id/view2"
            app:layout_constraintEnd_toEndOf="parent"
            app:layout_constraintStart_toStartOf="parent" />


    </androidx.constraintlayout.widget.ConstraintLayout>


</androidx.core.widget.NestedScrollView>

你应该有这样的布局:

<androidx.recyclerview.widget.RecyclerView
    android:id="@+id/recyclerView"
    android:layout_
    android:layout_
    app:layoutManager="androidx.recyclerview.widget.LinearLayoutManager"
    />

你的适配器应该是这样的:

class MyViewHolder(val rootView: View) : RecyclerView.ViewHolder(rootView)

class MyAdapter(val items: List<Any>) : RecyclerView.Adapter<MyViewHolder>() 


    companion object 
        const val TYPE_VIEW1 = 1
        const val TYPE_VIEW2 = 2
        const val TYPE_ITEMS = 3
    

    override fun getItemViewType(position: Int): Int = when (position) 
        0 -> TYPE_VIEW1
        1 -> TYPE_VIEW2
        else -> TYPE_ITEMS
    

    override fun onCreateViewHolder(parent: ViewGroup, viewType: Int): MyViewHolder = when (viewType) 
        TYPE_VIEW1 -> MyViewHolder(LayoutInflater.from(parent.context).inflate(R.layout.item_view1, parent, false))
        TYPE_VIEW2 -> MyViewHolder(LayoutInflater.from(parent.context).inflate(R.layout.item_view2, parent, false))
        else -> MyViewHolder(LayoutInflater.from(parent.context).inflate(R.layout.item_other, parent, false))
    

    override fun getItemCount(): Int = 2 + items.size

    override fun onBindViewHolder(holder: MyViewHolder, position: Int) 
        TODO("not implemented") //To change body of created functions use File | Settings | File Templates.
    

【讨论】:

感谢您的回答,我实施了您使用 RecyclerView 而不是您的代码的建议,因为它不符合我想要做的。但是根据您的建议,我创建了一个解决方案。如果不是太麻烦,我想再问你一个问题,RecyclerView中如何用Headers分割section? 你需要按照我说的来,拥有多种视图类型。如果您的所有“项目”都适合相同的模具,则您可以有 2 种视图类型 - 一种用于标题,一种用于项目。您的 getItemCount() 将需要返回标题数 + 项目数。

以上是关于如何使用里面的 ListView 滚动 NestedScrollView?的主要内容,如果未能解决你的问题,请参考以下文章

Android:滚动整个片段,里面有 ListView

请问:C# listView控件如何判断滚动条的位置?判断滚动条在控件的最下面?

Flutter- Gridview里面的listview问题

使用LinearLayout实现ListView,解决ListView和ScrollView滚动冲突

listview滚动到底部

Viewpager Fragment中的Android Listview-不滚动