KC Blog

最新記事

Compose Multiplatform 実践:CMPにおけるクロスプラットフォームAndroid、iOSコードのエントリーポイント

7 min read
CrossPlatform#CMP#Kotlin

はじめに

Compose Multiplatform (略称CMP)

昨日はCMPのプロジェクト構造について大まかに理解しました

昨日のCMPのプロジェクト構造理解とコンパイル設定

のプロジェクト構造から

CMPプロジェクトでは

commonMainで共通ロジックを記述

androidMainでAndroidプラットフォームのロジックを記述

iosMainでiOSプラットフォームのロジックを記述

desktopMainでDesktopプラットフォームのロジックを記述

YourProjectName
├── composeApp
│   ├── ...
│   ├── src
│   │   ├── commonMain
│   │   ├── commonTest
│   │   ├── iosMain
│   │   └── desktopMain
│   └── ...
├── ...
├── ...
└── ...

これから、一歩一歩

CMPのコードエントリーポイントについて理解していきましょう

クロスプラットフォーム実装に関わるため

コードがどのように動作し、各プラットフォームにどのように入っていくのかを

よく理解する必要があると思います

そこで今日はCMPのクロスプラットフォームにおけるコードエントリーポイントについて詳しく説明します

CMPコードのエントリーポイントを理解する

まずは簡単に Compose MultiplatformKotlin Multiplatform について理解しましょう

  • もちろん、CMPプロジェクト作成時にこれらのエントリーポイントは既に設定されています

    ここでは概念を理解するだけで十分です

  • CMPのcommonMainにある共通コードのエントリーポイント

    androidMain、iOSMain...などのクロスプラットフォームコードが

    この共通関数を呼び出すことを想定しています

    これによってコード共有の目的を達成します

  • ここで共通のApp()関数を作成しています

    これには以下が含まれます

    1.カスタム共通UI Theme

    2.koinを使用したviewmodelの注入

    3.カスタムCompose UIのエントリーポイント

    (これらについては後の章で、カスタムUI Themeの作成方法、koinの使用方法、カスタムCompose UIの作成方法...などのトピックを説明します)

// in ../commonMain/App.kt
@Composable
@Preview
fun App() {

    ElegantAccessComposeTheme {
        val viewModel = koinViewModel<MainViewModel>()
        ElegantAccessApp(viewModel)
    }

}
Android Appのコードエントリーポイント
  • Androidが実際にcommonMainの共通App()関数を呼び出します
// in ../androidMain/App.kt
class MainActivity : ComponentActivity() {
    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        val androidModule = module {
            single<Activity> { this@MainActivity }
            single<Context> { this@MainActivity.applicationContext }
        }

        startKoin {
            modules(appModule() + androidModule)
        }

        setContent {
            // 先ほど実装した共通App()関数を呼び出します
            App()
        }
    }
}
  • AndroidではAndroidのAndroid Manifest.xml内で、このMainActivity<activity>タグと

    アプリを起動する初期ページの<intent-filter>を宣言します

    <?xml version="1.0" encoding="utf-8"?>
      <manifest xmlns:android="http://schemas.android.com/apk/res/android">
          <application
              ...>
              <activity
                  ...
                  android:name=".MainActivity">
                  <intent-filter>
                      <action android:name="android.intent.action.MAIN" />
                      <category android:name="android.intent.category.LAUNCHER" />
                  </intent-filter>
              </activity>
          </application>
      </manifest>
    
iOS Appのコードエントリーポイント
  • iosMainが実際にcommonMainの共通App()関数を呼び出します
// in ../commonIos/MainViewController.kt

fun MainViewController() = ComposeUIViewController {

    val uiViewController = LocalUIViewController.current
    val iosModule = module {
        single<UIViewControllerWrapper> { UIViewControllerWrapperImpl(uiViewController) }
    }

    KoinApplication(application = {
        modules(appModule() + iosModule)
    }) {
        App()
    }
}
  • 実際にiOSでは上記のMainViewController.kt内の関数MainViewController()を呼び出します Cover
Desktopのコードエントリーポイント
  • desktopMainでは実際にcommonMainの共通App()関数を呼び出します

    その中でcomposeのapplication関数Windowを組み合わせてdesktop applicationを完成させます

// in ../desktopMain/main.kt
fun main() = application {
    Window(
        onCloseRequest = ::exitApplication,
        title = "App",
    ) {
        App()
    }
}
  • CMPのdesktopもJVMを通じてコンパイルされます

    Buildする場合は

    環境内で以下のgradle commandを使用します

./gradlew desktopRun -DmainClass=MainKt --quiet
  • またはIDEから直接このGradle taskをRun Configurationに追加することもできます <img src="/images/compose/046.png" alt="Cover" style="width: 90%"/ class="prose-img">

共通ロジックの開発

  • 上記のエントリーポイントを理解したら

    共通ロジックの開発を始めることができます

    1つのコードで複数のプラットフォーム向けアプリケーションを作成することができます

  • 下図からわかるように

    私たちは大部分の時間を./commonMainに費やします

    主なロジック開発はここで行われます

    ファイルシステムやファイル選択...など、各プラットフォームに依存する内容を除き

    それらはexpectactualを通じて実装されます

    (後の章でexpectとactualの使い方についても説明します)

<img src="/images/compose/047.png" alt="Cover" style="width: 80%"/ class="prose-img">

  • ただし、現時点では

    desktopプラットフォームiOSプラットフォームが独自のファイルシステムを持ち

    commonMainの共通ロジックで自分で実装する必要があるとしても

    KMMCMPでは

    すでにkotlinコードを通じて

    これらのクロスプラットフォームコンテンツを書くためのLibraryがサポートされています

    Gradleで設定するだけです

例:Kotlinを使用してdesktopのファイル関連操作を実装する:

// ../desktop/PlatformFile.desktop.kt
actual class PlatformFile actual constructor(private val path: String) {
    private val file = java.io.File(path)

    actual fun exists() = file.exists()
    actual fun createFile() = file.createNewFile()
    actual fun writeBytes(bytes: ByteArray) = file.writeBytes(bytes)
    actual fun delete() = file.delete()
    actual fun isDirectory(): Boolean = file.isDirectory
    actual fun copyTo(destination: PlatformFile, overwrite: Boolean) {
        file.copyTo(java.io.File(destination.path), overwrite)
    }
    actual fun mkdirs() {
        file.mkdirs()
    }
}

actual class PlatformZip actual constructor() {
    actual fun unzip(zipFilePath: String, destinationDir: String) {
        java.util.zip.ZipFile(zipFilePath).use { zip ->
            zip.entries().asSequence().forEach { entry ->
                val file = java.io.File(destinationDir, entry.name)
                if (entry.isDirectory) {
                    file.mkdirs()
                } else {
                    file.parentFile.mkdirs()
                    zip.getInputStream(entry).use { input ->
                        file.outputStream().use { output ->
                            input.copyTo(output)
                        }
                    }
                }
            }
        }
    }
}

関連記事

タグとカテゴリに基づく関連コンテンツ

9 min read

Compose Multiplatform 実践:CMPでのROOMによるクロスプラットフォームデータベース開発とトラブルシューティング

このシリーズのテーマはCompose Multiplatform 実践:Kotlinでゼロからクロスプラットフォームアプリを開発することです。今回はAndroidとiOSのクロスプラットフォームアプリ開発に焦点を当て、最終日には研究結果と感想を共有します。

📁 CrossPlatform
#CMP#Kotlin