Как добавить NavHostFragment в место назначения фрагмента?
Моя цель - понять и использовать компонент навигации наилучшим образом со сложной архитектурой навигации (я полагаю).
Чтобы создать глобальный контекст, я использую основную панель BottomNavigationBar с 5 элементами. Для управления ассоциированными фрагментами я использую пример, предоставленный Google: https://github.com/googlesamples/android-architecture-components/tree/master/NavigationAdvancedSample.
Но мой случай немного сложнее, чем просто нижняя полоса. В одном из пунктов назначения одного из фрагментов, запускаемых нижней панелью, у меня есть другое меню RecyclerView и контейнер фрагментов для навигации между фрагментами. Я поместил сюда важную часть моего первичного графика:
<?xml version="1.0" encoding="utf-8"?>
<navigation 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:id="@+id/main"
app:startDestination="@id/mainFragment">
<fragment
android:id="@+id/mainFragment"
android:name="com.exemple.fragment.MainFragment"
android:label="@string/main_label"
tools:layout="@layout/mainFragment">
<action
android:id="@+id/action_mainFragment_to_goScreen"
app:destination="@id/goScreen">
</action>
</fragment>
<fragment
android:id="@+id/goScreen"
android:name="com.exemple.fragment.GoFragment"
android:label="@string/goLabel"
tools:layout="@layout/fragment_go">
<action
android:id="@+id/action_goScreen_to_mainFragment"
app:destination="@id/mainFragment" />
<argument
android:name="arg1"
app:argType="com.exemple.customType"
android:defaultValue="@null"
app:nullable="true" />
<argument
android:name="arg2"
app:argType="string"
android:defaultValue="@null"
app:nullable="true" />
</fragment>
</navigation>
И xml с нижней полосой, с которой начинается этот график:
<?xml version="1.0" encoding="utf-8"?>
<layout 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"
tools:context="com.exemple.MainActivity">
<RelativeLayout
android:id="@+id/constraintLayout"
android:layout_width="match_parent"
android:layout_height="match_parent">
<androidx.appcompat.widget.Toolbar
android:id="@+id/fake_toolbar"
android:layout_width="0dp"
android:layout_height="0dp"/>
<androidx.fragment.app.FragmentTabHost
android:id="@+id/main_host_container"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:layout_above="@+id/bottom_navigation" />
<!-- navigation bottom -->
<com.google.android.material.bottomnavigation.BottomNavigationView
android:id="@+id/bottom_navigation"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_gravity="start"
android:layout_alignParentBottom="true"
android:background="@color/navigationBarColor"
app:itemIconTint="@color/textColorNavigationBar"
app:itemTextColor="@color/textColorNavigationBar"
app:menu="@menu/menu_bottom_navigation" />
</RelativeLayout>
</layout>
А вот пример графика, соответствующего фрагменту go (назначению):
<?xml version="1.0" encoding="utf-8"?>
<navigation 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:id="@+id/go"
app:startDestination="@id/firstGoFragment">
<action android:id="@+id/action_to_firstGoFragment"
app:destination="@id/firstGoFragment"/>
<fragment
android:id="@+id/firstGoFragment"
android:name="com.exemple.go.firstGoFragment"
android:label="@string/firstGoFragmentLabel"
tools:layout="@layout/firstGoFragment" >
</fragment>
<action android:id="@+id/action_to_secondFragment"
app:destination="@id/secondFragment"/>
<fragment
android:id="@+id/secondFragment"
android:name="com.exemple.go.SecondFragment"
android:label="@string/secondFragmentLabel"
tools:layout="@layout/secondFragment" >
</fragment>
</navigation>
Ассоциированный xml:
<?xml version="1.0" encoding="utf-8"?>
<layout xmlns:tools="http://schemas.android.com/tools"
xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto">
<androidx.constraintlayout.widget.ConstraintLayout
android:layout_width="match_parent"
android:layout_height="match_parent"
android:background="@color/colorGreyLight"
android:clickable="true"
android:focusable="true">
<RelativeLayout
android:id="@+id/linearLayout3"
android:layout_width="match_parent"
android:layout_height="?attr/actionBarSize"
android:background="@color/colorPrimary"
app:layout_constraintEnd_toEndOf="parent"
app:layout_constraintStart_toStartOf="parent"
app:layout_constraintTop_toTopOf="parent">
<androidx.appcompat.widget.AppCompatImageView
android:id="@+id/quit"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_centerVertical="true"
android:layout_marginStart="16dp"
android:paddingBottom="15dp"
android:paddingEnd="15dp"
android:paddingTop="15dp"
app:srcCompat="@drawable/ic_quit_white" />
<com.teedji.mobeelity.custom.CustomTextView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_centerInParent="true"
android:text="@string/go"
android:textColor="@color/colorWhite"
android:textSize="17sp"
android:textStyle="bold" />
</RelativeLayout>
<androidx.recyclerview.widget.RecyclerView
android:id="@+id/goMenu"
android:layout_width="0dp"
android:layout_height="wrap_content"
android:background="@color/colorWhite"
android:overScrollMode="never"
app:layout_constraintEnd_toEndOf="parent"
app:layout_constraintHorizontal_bias="1.0"
app:layout_constraintStart_toStartOf="parent"
app:layout_constraintTop_toBottomOf="@+id/linearLayout3" />
<!-- include fragment child container -->
<fragment
android:id="@+id/go_nav_host_fragment_container"
android:name="androidx.navigation.fragment.NavHostFragment"
android:layout_width="0dp"
android:layout_height="0dp"
android:layout_marginTop="0dp"
app:layout_constraintBottom_toBottomOf="parent"
app:layout_constraintEnd_toEndOf="parent"
app:layout_constraintStart_toStartOf="parent"
app:layout_constraintTop_toBottomOf="@+id/goMenu"
app:navGraph="@navigation/go"/>
</androidx.constraintlayout.widget.ConstraintLayout>
</layout>
Теперь, чтобы перейти к экрану перехода, я просто использую
findNavController().navigate(MainFragmentDirections.actionMainFragmentToGoScreen(arg1, arg2))
или
findNavController().navigate(R.id. action_mainFragment_to_goScreen)
Но когда я нахожусь на фрагменте Go, navController, который я использую, findNavController()
по-прежнему остается главным фрагментом. Так чтоnavigate()
fonction не нашла идентификатор действия. Чтобы решить эту проблему, мне пришлось программно изменить navHost в GoFragment следующим образом:
override fun onCreate(savedInstanceState: Bundle?) {
Log.i(LOG_TAG, "onCreate")
super.onCreate(savedInstanceState)
if( fragmentManager != null ) {
// If the Nav Host fragment exists, return it
val existingFragment = fragmentManager!!.findFragmentByTag(FRAGMENT_TAG) as NavHostFragment?
existingFragment?.let { navHostGoFragment = it }
// Otherwise, create it and return it.
if ( navHostGoFragment == null) {
navHostGoFragment = NavHostFragment.create(R.navigation.go)
fragmentManager!!.beginTransaction()
.add(R.id.go_nav_host_fragment_container, navHostGoFragment!!, FRAGMENT_TAG)
.commit()
}
}else {
Log.e(LOG_TAG, "fragmentManager is null so I can't find the fragment host")
}
}
а затем используйте navHostGoFragment!!.navController.navigate(R.id.action_to_firstGoFragment)
для навигации в новом контейнере Но при этом мой firstFragment создается дважды (я вижу его в своих журналах), и это создает некоторые проблемы, например Cannot add the same observer with different lifecycles
например (я решаю это добавлением
if (model.getLiveData().hasObservers()) {
model.getLiveData().removeObserver(observer)
}
перед
model.getLiveData().observe(viewLifecycleOwner, observer)
) Но я думаю, что не занимаюсь навигационной частью, как следовало бы...
Не могли бы вы помочь сделать эту работу более эффективной и стабильной?
1 ответ
Попробуй использовать go_nav_host_fragment_container.findNavController().navigate()
, если вы используете расширения Kotlin для Android или findViewById(R.id.go_nav_host_fragment_container).findNavController().navigate()
и посмотрите, работает ли это.