ナビゲーション コードをモジュール化する

このページでは、ナビゲーション コードをモジュール化するためのガイドを提供します。このガイドは、アプリのモジュール化に関する一般的なガイダンスを補完することを目的としています。

概要

ナビゲーション コードのモジュール化とは、関連するナビゲーション キーと、それらが表すコンテンツを個別のモジュールに分割するプロセスのことです。これにより、責任が明確に分離され、アプリのさまざまな機能を切り替えることができます。

ナビゲーション コードをモジュール化する手順は次のとおりです。

  • アプリの各機能に対して apiimpl の 2 つのサブモジュールを作成します。
  • 各機能のナビゲーション キーをそれぞれの api モジュールに配置
  • 各機能の entryProviders とナビゲーション可能なコンテンツを、関連する impl モジュールに配置します。
  • メインアプリ モジュールに entryProviders を直接提供するか、依存関係の挿入を使用して提供します。

機能を api サブモジュールと実装サブモジュールに分離

アプリの各機能について、apiimpl(「実装」の略)という 2 つのサブモジュールを作成します。次の表を参考にして、ナビゲーション コードを配置する場所を決定してください。

モジュール名

以下を含む

api

ナビゲーション キー

impl

NavEntryentryProvider の定義など、その機能のコンテンツ。キーをコンテンツに解決するもご覧ください。

このアプローチでは、impl モジュールに含まれるコンテンツが、api モジュールに含まれる別のモジュールのナビゲーション キーに依存できるようにすることで、ある機能から別の機能に移動できます。

`impl` モジュールが `api` モジュールに依存する方法を示す機能モジュールの依存関係図。
図 1. 実装モジュールが API モジュールに依存する方法を示す機能モジュールの依存関係図。

拡張関数を使用してナビゲーション エントリを分離する

Navigation 3 では、ナビゲーション可能なコンテンツはナビゲーション エントリを使用して定義されます。これらのエントリを別々のモジュールに分離するには、EntryProviderScope で拡張関数を作成し、その機能の impl モジュールに移動します。これらは「エントリ ビルダー」と呼ばれます。

次のコード例は、2 つのナビゲーション エントリをビルドするエントリ ビルダーを示しています。

// import androidx.navigation3.runtime.EntryProviderScope
// import androidx.navigation3.runtime.NavKey

fun EntryProviderScope<NavKey>.featureAEntryBuilder() {
    entry<KeyA> {
        ContentRed("Screen A") {
            // Content for screen A
        }
    }
    entry<KeyA2> {
        ContentGreen("Screen A2") {
            // Content for screen A2
        }
    }
}

メインアプリ モジュールで entryProvider を定義するときに、entryProvider DSL を使用してその関数を呼び出します。

// import androidx.navigation3.runtime.entryProvider
// import androidx.navigation3.ui.NavDisplay
NavDisplay(
    entryProvider = entryProvider {
        featureAEntryBuilder()
    },
    // ...
)

依存性注入を使用してメインアプリにエントリを追加する

上記のコード例では、各エントリ ビルダーは entryProvider DSL を使用してメインアプリから直接呼び出されます。アプリに多数の画面や機能モジュールがある場合、この方法はうまくスケールしない可能性があります。

この問題を解決するには、各機能モジュールが依存性注入を使用して、エントリ ビルダーをアプリのアクティビティに提供するようにします。

たとえば、次のコードでは、Dagger マルチバインディング(具体的には @IntoSet)を使用して、MainActivity が所有する Set にエントリ ビルダーを挿入します。これらは entryProvider 内で繰り返し呼び出されるため、多数のエントリ ビルダー関数を明示的に呼び出す必要がなくなります。

機能モジュール

// import dagger.Module
// import dagger.Provides
// import dagger.hilt.InstallIn
// import dagger.hilt.android.components.ActivityRetainedComponent
// import dagger.multibindings.IntoSet

@Module
@InstallIn(ActivityRetainedComponent::class)
object FeatureAModule {

    @IntoSet
    @Provides
    fun provideFeatureAEntryBuilder() : EntryProviderScope<NavKey>.() -> Unit = {
        featureAEntryBuilder()
    }
}

アプリ モジュール

// import android.os.Bundle
// import androidx.activity.ComponentActivity
// import androidx.activity.compose.setContent
// import androidx.navigation3.runtime.EntryProviderScope
// import androidx.navigation3.runtime.NavKey
// import androidx.navigation3.runtime.entryProvider
// import androidx.navigation3.ui.NavDisplay
// import javax.inject.Inject

class MainActivity : ComponentActivity() {

    @Inject
    lateinit var entryBuilders: Set<@JvmSuppressWildcards EntryProviderScope<NavKey>.() -> Unit>

    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        setContent {
            NavDisplay(
                entryProvider = entryProvider {
                    entryBuilders.forEach { builder -> this.builder() }
                },
                // ...
            )
        }
    }
}

ナビゲーション エントリでナビゲーションが必要な場合(たとえば、新しい画面に移動する UI 要素が含まれている場合)は、アプリのナビゲーション状態を変更できるオブジェクトを各ビルダー関数に挿入します。

リソース

Navigation 3 コードをモジュール化する方法を示すコードサンプルについては、以下をご覧ください。