5

当发生配置更改并因此重新创建我的 Activity 和 Fragment 时,我的导航 Graph 范围内的 ViewModel 不可用,而 Fragments 已经被再次创建。
似乎在 navGraph 之前重新创建了 Fragment。

我正在使用此代码从我的 Fragment 初始化我的 navGraph 范围 ViewModel:

private val myViewModel: MyViewModel by navGraphViewModels(R.id.nav_graph_id)

如果我尝试myViewModel在 FragmentsonViewCreated功能中使用,我会IllegalArgumentException在配置更改后得到一个。例外:

java.lang.IllegalArgumentException: No destination with ID <destination id> is on the NavController's back stack

我该如何处理?

我已经检查过我的 ID 没有在其他任何地方使用。

编辑1:

这是我的 activity_main.xml

<?xml version="1.0" encoding="utf-8"?>
<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=".MainActivity">
    
    <androidx.fragment.app.FragmentContainerView
            android:id="@+id/main_nav_host"
            android:name="androidx.navigation.fragment.NavHostFragment"
            android:layout_width="0dp"
            android:layout_height="0dp"
            app:defaultNavHost="true"
            app:layout_constraintTop_toTopOf="parent"
            app:layout_constraintStart_toStartOf="parent"
            app:layout_constraintEnd_toEndOf="parent"
            app:layout_constraintBottom_toBottomOf="parent"
            app:navGraph="@navigation/main_nav_graph" />

</androidx.constraintlayout.widget.ConstraintLayout>

这是我的 main_nav_graph.xml:

<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_nav_graph"
        app:startDestination="@id/main_nav_graph_1">
    
    <navigation
            android:id="@+id/main_nav_graph_1"
            android:label="@string/nav_graph_1_label"
            app:startDestination="@id/nav_graph_1_start_fragment">
        <!-- nav graph stuff -->
    </navigation>
    
    <navigation
            android:id="@+id/main_nav_graph_2"
            android:label="@string/nav_graph_2_label"
            app:startDestination="@id/nav_graph_2_start_fragment">
        
        <fragment
                android:id="@+id/nav_graph_2_start_fragment"
                android:label="@string/nav_graph_2_start_fragment_label"
                android:name="my.package.ui.NavGraph2StartFragment"
                tools:layout="@layout/fragment_nav_graph_2_start">
        </fragment>
        <!-- In here is where I get the problem -->
    </navigation>
    
</navigation>
4

3 回答 3

2

我的问题如下:

配置更改后,NavGraph 返回到其起始目的地,但最后活动的 Fragment 无论如何都会被加载。这意味着实际启动的 Fragment 和 navGraph 的当前目的地不同步。
然后,当我的 Fragment 尝试加载 navGraph 范围内的 ViewModel 时,它失败了,因为 navGraph 认为它位于与实际不同的 Fragment 上。

为了解决我的问题,我必须使用活动 savedInstanceState 保存和恢复 navController 的状态Bundle。(来源:https ://stackoverflow.com/a/59987336 )

override fun onSaveInstanceState(savedInstanceState: Bundle) {
    super.onSaveInstanceState(savedInstanceState)
    savedInstanceState.putBundle("nav_state", fragment.findNavController().saveState())
}

// restore in RestoreInstanceState
override fun onRestoreInstanceState(savedInstanceState: Bundle) {
    super.onRestoreInstanceState(savedInstanceState)
    fragment.findNavController().restoreState(savedInstanceState.getBundle("nav_state"))
}

// or restore in onCreate
override fun onCreate(savedInstanceState: Bundle?) {
    super.onCreate(savedInstanceState)
    if (savedInstanceState?.containsKey("nav_state") == true) {
        fragment.findNavController().restoreState(savedInstanceState.getBundle("nav_state"))
    }
}
于 2020-09-08T08:20:21.563 回答
1

对于其他来到这里的人,我在当前显示对话框时遇到了这个问题,我向 Fragment 发出了 navController.navigate() 调用,该调用by navGraphViewModels()在对话框被解除之前使用。

就好像导航库无法理解在显示对话框时存在具有该 ID 的目标。

为了更清楚起见,此对话框使用<dialog>我的导航图 XML 中的属性显示。

我的解决方法(目前)是确保在进一步调用以通过以下方式导航到其他地方之前关闭对话框view.postDelayed({ navController.navigate(elsewhere)}, 300)

这不是我第一次遇到 NavController 在显示对话框时尝试导航的问题。

于 2021-02-03T20:07:07.560 回答
-1

通过改变这个很容易解决:

private val myViewModel: MyViewModel by hiltNavGraphViewModels(R.id.nav_graph_id)

至:

private val viewModel: AuthViewModel by activityViewModels{ defaultViewModelProviderFactory }
于 2021-12-07T21:38:21.603 回答