使用RecyclerView,Android jetpack导航和折叠的工具栏实现正确的滚动

时间:2019-03-14 19:39:03

标签: android android-recyclerview android-collapsingtoolbarlayout androidx android-jetpack-navigation

我尝试在我的android应用程序中通过以下设置实现正确的滚动行为。

对于导航,我将jetpack导航与Toolbar布局和底部导航结合使用。我还使用“一个活动,很多碎片”的原则。底部导航的每个项目都会根据我的导航图启动相应的Fragment。这要求我在布局中使用NavHostFragment

我的Toolbar布局是活动布局的一部分,并根据当前片段进行填充。有些片段需要折叠Toolbar,并且在需要时也会添加。但是当工具栏折叠时,我会遇到以下问题:

在特殊情况下,我的Toolbar崩溃了,在NavHostFragment上填充了RecyclerView。在其他情况下,似乎可以将app:layout_behavior="@string/appbar_scrolling_view_behavior"添加到RecyclerView并获得预期的结果,因为RecyclerViewCoordinatorLayout的直接子代。在我的情况下,RecyclerViewFragment的子元素,而FrameLayout基本上是RecyclerView(根据布局检查器)。这导致了问题,RecyclerView上的layout_behaviour无效,因为CoordinatorLayout不是<androidx.constraintlayout.widget.ConstraintLayout 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_width="match_parent" android:layout_height="match_parent" tools:context=".OverviewActivity"> <androidx.coordinatorlayout.widget.CoordinatorLayout android:layout_width="wrap_content" android:layout_height="wrap_content" android:id="@+id/toolbarCoordiantor" android:layout_marginTop="?attr/actionBarSize" app:layout_constraintTop_toTopOf="parent" app:layout_constraintBottom_toTopOf="@id/overview_bottom_navigation" app:layout_constraintStart_toStartOf="parent" app:layout_constraintEnd_toEndOf="parent"> <fragment android:layout_width="match_parent" android:layout_height="wrap_content" android:id="@+id/overview_fragmentholder" android:name="androidx.navigation.fragment.NavHostFragment" app:layout_constraintStart_toStartOf="parent" app:layout_constraintEnd_toEndOf="parent" app:layout_constraintTop_toBottomOf="@+id/toolbar" app:layout_constraintBottom_toBottomOf="parent" app:defaultNavHost="true" android:layout_marginBottom="?attr/actionBarSize" android:layout_marginTop="?attr/actionBarSize" app:navGraph="@navigation/nav_graph" app:layout_constraintVertical_bias="1.0"/> <include layout="@layout/toolbar" android:id="@+id/toolbar"/> </androidx.coordinatorlayout.widget.CoordinatorLayout> <com.google.android.material.bottomnavigation.BottomNavigationView android:id="@+id/overview_bottom_navigation" android:layout_width="match_parent" android:layout_height="wrap_content" android:foregroundGravity="bottom" app:menu="@menu/navigation" app:layout_constraintBottom_toBottomOf="parent" app:layout_constraintStart_toStartOf="parent" app:layout_constraintEnd_toEndOf="parent" app:itemBackground="@color/white" app:itemIconTint="@color/bottom_navigation_color_states" app:itemTextColor="@color/bottom_navigation_color_states"/> 的直接子代。

我无法为这个问题提供可行的解决方案。有人有主意吗?

layout / activity_overview.xml

<com.google.android.material.appbar.CollapsingToolbarLayout
        android:id="@+id/collapsingToolbarLayout"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:fitsSystemWindows="true"
        app:layout_scrollFlags="scroll|exitUntilCollapsed|snap|snapMargins"
        android:minHeight="?attr/actionBarSize">

    <androidx.constraintlayout.widget.ConstraintLayout android:layout_width="match_parent"
                                                       android:layout_height="wrap_content"
                                                       android:id="@+id/expandedToolbarContentContainer"
                                                       android:layout_marginTop="?attr/actionBarSize"
                                                       app:layout_collapseMode="parallax"/>

    <androidx.appcompat.widget.Toolbar android:layout_width="match_parent"
                                       android:layout_height="?attr/actionBarSize"
                                       app:layout_collapseMode="pin"
                                       style="@style/AppTheme.DarkToolbar"
                                       android:id="@+id/toolbarView">

        <androidx.constraintlayout.widget.ConstraintLayout android:layout_width="match_parent"
                                                           android:layout_height="match_parent"
                                                           android:background="@drawable/round_outline"
                                                           style="@style/AppTheme.DarkToolbar.Container"
                                                           android:id="@+id/toolbarContentContainer"/>

    </androidx.appcompat.widget.Toolbar>

</com.google.android.material.appbar.CollapsingToolbarLayout>

layout / toolbar.xml

<layout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:app="http://schemas.android.com/apk/res-auto">
    <androidx.recyclerview.widget.RecyclerView
            android:id="@+id/dish_list"
            android:layout_width="match_parent"
            android:layout_height="wrap_content"
            android:scrollbars="vertical"
            app:layout_constraintLeft_toLeftOf="parent"
            app:layout_constraintEnd_toEndOf="parent"
            android:layoutAnimation="@anim/layout_animation_fall_down"
    />

布局/list.xml

semaphore.c

2 个答案:

答案 0 :(得分:0)

尝试使用NestedScrollView封装具有所需行为的host_fragment,如下所示:

 <androidx.core.widget.NestedScrollView
        android:id="@+id/nestedScrollView"
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:fillViewport="true"
        android:fitsSystemWindows="true"
        android:transitionGroup="true"
        app:layout_behavior="@string/appbar_scrolling_view_behavior">

        <fragment
            android:id="@+id/nav_host_fragment"
            android:name="androidx.navigation.fragment.NavHostFragment"
            android:layout_width="match_parent"
            android:layout_height="match_parent"
            app:defaultNavHost="true"
            app:navGraph="@navigation/my_nav_host" />
    </androidx.core.widget.NestedScrollView>

答案 1 :(得分:0)

Jurij Pituljas approach将阻止scrollToPosition的{​​{1}}方法起作用。另外,使用RecyclerView导航回片段会将其滚动位置重置为顶部。

here描述了一种更好的方法。即使在引用的文章中也不需要周围的RecyclerView。无需将片段包装在FrameLayout中,只需将所有非滚动片段布局视图包装到NestedScrollView中。

带有NestedScrollView的活动布局::无需将片段包装到滚动视图中。只需设置NavHostFragment

layout_behavior

具有根... <fragment android:id="@+id/navigation_host_fragment" android:name="androidx.navigation.fragment.NavHostFragment" android:layout_width="match_parent" android:layout_height="match_parent" app:defaultNavHost="true" app:layout_behavior="@string/appbar_scrolling_view_behavior" app:navGraph="@navigation/navigation_graph" /> ... 的片段布局:没什么特别的,因为RecyclerView具有滚动功能。

RecyclerView

片段布局,例如<androidx.recyclerview.widget.RecyclerView xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="match_parent" android:layout_height="match_parent" android:scrollbars="vertical" /> 包装在根TextView中以添加滚动功能。

NestedScrollView