活动目标

在导航图中,目标可以是活动。虽然在您的应用中拥有单个活动是最佳实践,但应用通常会为应用内的不同组件或屏幕使用单独的活动。在这种情况下,活动目标非常有用。

Compose 和 Kotlin DSL

在 Compose 中以及使用 Kotlin DSL 和片段时,向导航图添加活动目标基本上是相同的。这是因为在将您的 NavGraph 传递给您的 NavHost 可组合项时,您使用的是相同的 createGraph() lambda。

有关更多信息,请参阅 片段和 Kotlin DSL

XML

创建 活动 目标类似于创建 片段 目标。但是,活动目标的性质却大不相同。

默认情况下,导航库NavController 附加到 Activity 布局,并且活动导航图的范围限定为活动的 Activity。如果用户导航到不同的 Activity,则当前导航图将不再处于范围内。这意味着应将 Activity 目标视为导航图中的端点。

要添加活动目标,请使用其完全限定类名指定目标 Activity

<?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"
    android:id="@+id/navigation_graph"
    app:startDestination="@id/simpleFragment">

    <activity
        android:id="@+id/sampleActivityDestination"
        android:name="com.example.android.navigation.activity.DestinationActivity"
        android:label="@string/sampleActivityTitle" />
</navigation>

此 XML 等效于对 startActivity() 的以下调用

Kotlin

startActivity(Intent(context, DestinationActivity::class.java))

Java

startActivity(new Intent(context, DestinationActivity.class));

在某些情况下,这种方法可能不适用。例如,您可能没有对活动类的编译时依赖关系,或者您可能更喜欢通过隐式意图进行间接级别的操作。目标 Activity 的清单条目中的 intent-filter 指示您需要如何构造 Activity 目标。

例如,考虑以下清单文件

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    package="com.example.android.navigation.activity">
    <application>
        <activity android:name=".DestinationActivity">
            <intent-filter>
                <action android:name="android.intent.action.VIEW" />
                <data
                    android:host="example.com"
                    android:scheme="https" />
                <category android:name="android.intent.category.BROWSABLE" />
                <category android:name="android.intent.category.DEFAULT" />
            </intent-filter>
        </activity>
    </application>
</manifest>

相应的 Activity 目标需要使用 actiondata 属性进行配置,使其与清单条目中的属性匹配

<?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"
    android:id="@+id/navigation_graph"
    app:startDestination="@id/simpleFragment">
    <activity
        android:id="@+id/localDestinationActivity"
        android:label="@string/localActivityTitle"
        app:action="android.intent.action.VIEW"
        app:data="https://example.com"
        app:targetPackage="${applicationId}" />
</navigation>

targetPackage 指定为当前 applicationId 会将范围限制为当前应用,其中包括主应用。

相同的机制可用于您希望特定应用成为目标的情况。以下示例将目标定义为 applicationIdcom.example.android.another.app 的应用。

<?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"
    android:id="@+id/navigation_graph"
    app:startDestination="@id/simpleFragment">
    <activity
        android:id="@+id/localDestinationActivity"
        android:label="@string/localActivityTitle"
        app:action="android.intent.action.VIEW"
        app:data="https://example.com"
        app:targetPackage="com.example.android.another.app" />
</navigation>

动态参数

前面的示例使用固定 URL 来导航到目标。您可能还需要支持动态 URL,其中附加信息作为 URL 的一部分发送。例如,您可能在类似于 https://example.com?userId=<actual user ID> 格式的 URL 中发送用户 ID。

在这种情况下,请使用 dataPattern 而不是 data 属性。然后,您可以提供要替换 dataPattern 值中命名占位符的参数

<?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"
    android:id="@+id/navigation_graph"
    app:startDestination="@id/simpleFragment">
    <activity
        android:id="@+id/localDestinationActivity"
        android:label="@string/localActivityTitle"
        app:action="android.intent.action.VIEW"
        app:dataPattern="https://example.com?userId={userId}"
        app:targetPackage="com.example.android.another.app">
        <argument
            android:name="userId"
            app:argType="string" />
    </activity>
</navigation>

在此示例中,您可以使用 Safe ArgsBundle 指定 userId

Kotlin

navController.navigate(
    R.id.localDestinationActivity,
    bundleOf("userId" to "someUser")
)

Java

Bundle args = new Bundle();
args.putString("userId", "someUser");
navController.navigate(R.id.localDestinationActivity, args);

此示例将 someUser 替换为 {userId} 并创建 https://example.com?userId=someUser 的 URI 值。