Достижение правильной прокрутки с RecyclerView, навигацией для Android Jetpack и сворачивающейся панелью инструментов
Я пытаюсь реализовать правильное поведение прокрутки с помощью следующей настройки в моем приложении для Android.
Для навигации я использую джетпак в сочетании с Toolbar
макет и нижняя навигация. Я также использую принцип "одно действие, много фрагментов". Каждый элемент нижней навигации запускает соответствующий Fragment
на основе моего навигационного графика. Это требует от меня использовать NavHostFragment
в моем макете.
мой Toolbar
макет является частью макета действия и заполняется на основе текущего фрагмента. Некоторые фрагменты требуют разрушения Toolbar
, который также добавляется при необходимости. Но когда у меня сваливается панель инструментов, я сталкиваюсь со следующей проблемой:
В частном случае у меня разваливается Toolbar
, NavHostFragment
населена RecyclerView
, В других случаях, кажется, я могу добавить app:layout_behavior="@string/appbar_scrolling_view_behavior"
к RecyclerView
и получить ожидаемый результат, потому что RecyclerView
прямое дитя CoordinatorLayout
, В моем случае RecyclerView
это ребенок Fragment
, который в основном FrameLayout
(по словам инспектора по планировке). Это приводит к проблеме, что layout_behaviour на RecyclerView
не имеет никакого эффекта, так как RecyclerView
не является прямым ребенком CoordinatorLayout
,
Я не мог найти рабочее решение этой проблемы. У кого-нибудь есть идея?
макет /activity_overview.xml
<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"/>
макет /toolbar.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>
макет /list.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"
/>
1 ответ
Попробуйте обернуть host_fragment NestedScrollView необходимым поведением, например так:
<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>
Подступ Юрия Питульяс помешаетscrollToPosition
метод RecyclerView
работать. Дополнительный возврат к фрагментам с помощьюRecyclerView
сбросит положение прокрутки вверх.
Лучший подход описан здесь. Даже в окружении нет нуждыFrameLayout
s в указанной статье. Вместо того, чтобы оборачивать фрагмент в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
чтобы добавить возможности прокрутки.
<androidx.core.widget.NestedScrollView
xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="vertical">
<LinearLayout
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="vertical">
<TextView
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:text="non scrolling content wrapped" />
</LinearLayout>
</androidx.core.widget.NestedScrollView>