导航组件 NavDeepLinkBuilder 返回栈

时间:2021-02-10 20:37:45

标签: android navigation deep-linking android-architecture-navigation fragment-backstack

我在从深层链接(通过通知)启动应用程序时遇到了返回堆栈问题。

总体思路是我有 3 个片段(A、B、C)。 A 屏幕用作启动屏幕,因此我已将 popUpTopopUpToInclusive 放入操作中,因此我再也看不到它了。我的深层链接目的地是片段 C。为了获得正确的返回堆栈,我将 B 和 C 合并到嵌套导航中。不幸的是,当我在片段 B 中按下后退按钮时,应用程序显示的是片段 A。我猜这是因为当应用程序从深层链接本身启动时,从未调用过从 A 到 B 的带有 popUpTo 的操作。我的问题是:在这种特殊情况下,如何避免从 B 返回到 A?

提前致谢!我附上示例代码:

<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/nav_graph"
    app:startDestination="@id/splashFragment">

    <fragment
        android:id="@+id/A"
        android:name="A"
        android:label="A" >
        <action
            android:id="@+id/action_A_to_B"
            app:destination="@id/B"
            app:popUpTo="@id/A"
            app:popUpToInclusive="true" />
    </fragment>
    <navigation android:id="@+id/nested_nav"
        app:startDestination="@id/B">
        <fragment
            android:id="@+id/B"
            android:name="B"
            android:label="B">
            <action
                android:id="@+id/action_B_to_C"
                app:destination="@id/C" />
        </fragment>
        <fragment
            android:id="@+id/C"
            android:name="C"
            android:label="C">
        </fragment>
    </navigation>
</navigation>

还有我的 DeepLinkBuilder(没什么特别的,和文档一样)

NavDeepLinkBuilder(requireContext())
    .setGraph(R.navigation.nav_graph)
    .setDestination(R.id.userProfileFragment)
    .createPendingIntent()
    .send()

1 个答案:

答案 0 :(得分:0)

您已将包含启动画面的 navGraph 设置为默认 startDestination,这会在您从嵌套的 navGraph 按下后退按钮时显示它,我认为最简单的解决方案是执行以下操作: 首先编辑您的导航文件,使其只有一个导航,例如:

<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/nav_graph"
    app:startDestination="@id/splashFragment">

    <fragment
        android:id="@+id/A"
        android:name="A"
        android:label="A" >
        <action
            android:id="@+id/action_A_to_B"
            app:destination="@id/B"
            app:popUpTo="@id/A"
            app:popUpToInclusive="true" />
    </fragment>
    <fragment
            android:id="@+id/B"
            android:name="B"
            android:label="B">
            <action
                android:id="@+id/action_B_to_C"
                app:destination="@id/C" />
        </fragment>
        <fragment
            android:id="@+id/C"
            android:name="C"
            android:label="C">
        </fragment>
</navigation>

第二次设置一个监听器到 navController 来监听当目的地改变发生时,像这样:

yourNavController.setOnDestinationChangedListener(this);

覆盖 onDestinationChangedListener() 方法并检查即将显示的片段是否为 A ,然后像这样完成应用程序:

@override 
public void onDestinationChanged(NavController controller, NavDestination destination, Bundle arguments){
if(destination.getId()==R.id.splashFragmentId){
finish();
}
}