Skip to content

Commit 3511a75

Browse files
authored
add Compose Multiplatform sample, using io.github.hoc081098:solivagant-navigation:0.3.0 (#52)
* gradle wrapper 8.7 * standalone-androidApp * include(":sample:standalone-composeMultiplatform:composeApp") * include(":sample:standalone-composeMultiplatform:composeApp") * include(":sample:standalone-composeMultiplatform:composeApp") * include(":sample:standalone-composeMultiplatform:composeApp") * include(":sample:standalone-composeMultiplatform:composeApp") * format * detekt * add desktop * navigation * navigation * done compose multiplatform sample, and fix issue where bus was null before marking `collecting` state * detekt * Update sample.yml * Update build.gradle.kts * Update libs.versions.toml JetBrains/compose-multiplatform#4326
1 parent 69e7570 commit 3511a75

File tree

73 files changed

+2276
-17
lines changed

Some content is hidden

Large Commits have some content hidden by default. Use the searchbox below for content that may be hidden.

73 files changed

+2276
-17
lines changed

.github/workflows/sample.yml

+1-1
Original file line numberDiff line numberDiff line change
@@ -59,4 +59,4 @@ jobs:
5959
run: chmod +x ./gradlew
6060

6161
- name: Build
62-
run: ./gradlew :sample:app:assembleDebug --stacktrace
62+
run: ./gradlew :sample:standalone-androidApp:assembleDebug :sample:standalone-composeMultiplatform:desktopApp:packageDistributionForCurrentOS --stacktrace

.idea/deploymentTargetDropDown.xml

+1-1
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

channel-event-bus/src/commonMain/kotlin/com/hoc081098/channeleventbus/ChannelEventBus.kt

+3-1
Original file line numberDiff line numberDiff line change
@@ -201,7 +201,9 @@ private class ChannelEventBusImpl(
201201
*/
202202
private fun markAsNotCollecting(key: ChannelEventKey<*>): Unit =
203203
_busMap.synchronized {
204-
_busMap[key] = _busMap[key]!!
204+
// _busMap[key] can be null if it is removed and closed before calling this method.
205+
// Just ignore in that case.
206+
_busMap[key] = (_busMap[key] ?: return)
205207
.copy(isCollecting = false)
206208
.also { logger?.onStopCollection(key, this) }
207209
}

detekt.yml

+3
Original file line numberDiff line numberDiff line change
@@ -674,6 +674,9 @@ style:
674674
UnusedPrivateMember:
675675
active: true
676676
allowedNames: '(_|ignored|expected|serialVersionUID)'
677+
ignoreAnnotated:
678+
- Preview
679+
- androidx.compose.desktop.ui.tooling.preview.Preview
677680
UseAnyOrNoneInsteadOfFind:
678681
active: true
679682
UseArrayLiteralsInAnnotations:

gradle/libs.versions.toml

+19-12
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
[versions]
2-
kotlin = "1.9.22"
3-
coroutines = "1.8.0"
2+
kotlin = "1.9.23"
3+
coroutines = "1.8.1-Beta"
44
kotlinx-serialization-json = "1.6.3"
55
kotlinx-collections-immutable = "0.3.7"
66

@@ -12,9 +12,9 @@ java-toolchain = "17"
1212
# the Android minSdkVersion to use
1313
android-min = "21"
1414
# the Android target to use
15-
android-target = "33"
15+
android-target = "34"
1616
# the Android compileSdkVersion to use
17-
android-compile = "33"
17+
android-compile = "34"
1818

1919
android-gradle = "8.3.1"
2020

@@ -23,22 +23,23 @@ sample-android-target = "34"
2323
# the Android compileSdkVersion to use in sample
2424
sample-android-compile = "34"
2525

26-
jetbrains-compose = "1.6.1"
26+
jetbrains-compose = "1.6.2"
2727

2828
touchlab-stately = "2.0.6"
2929
napier = "2.7.1"
30-
flowExt = "0.8.0"
31-
koin = "3.5.3"
32-
koin-androidx-compose = "3.5.3"
30+
flowExt = "0.8.1-Beta"
31+
koin-core = "3.5.3"
32+
koin-compose = "1.1.2"
3333
coil = "2.6.0"
3434
compose-rules-detekt = "0.3.12"
3535

3636
androidx-lifecycle = "2.7.0"
3737
androidx-annotation = "1.7.1"
3838
androidx-activity = "1.8.2"
39-
androidx-compose-compiler = "1.5.9"
39+
androidx-compose-compiler = "1.5.11"
4040
androidx-navigation = "2.7.7"
4141
kmp-viewmodel = "0.7.1"
42+
solivagant = "0.3.0"
4243
timber = "5.0.1"
4344

4445
spotless = "6.25.0"
@@ -54,6 +55,7 @@ ktlint = "0.50.0"
5455
[libraries]
5556
coroutines-core = { module = "org.jetbrains.kotlinx:kotlinx-coroutines-core", version.ref = "coroutines" }
5657
coroutines-android = { module = "org.jetbrains.kotlinx:kotlinx-coroutines-android", version.ref = "coroutines" }
58+
coroutines-swing = { module = "org.jetbrains.kotlinx:kotlinx-coroutines-swing", version.ref = "coroutines" }
5759
coroutines-test = { module = "org.jetbrains.kotlinx:kotlinx-coroutines-test", version.ref = "coroutines" }
5860
coroutines-jdk8 = { module = "org.jetbrains.kotlinx:kotlinx-coroutines-jdk8", version.ref = "coroutines" }
5961
kotlinx-serialization-json = { module = "org.jetbrains.kotlinx:kotlinx-serialization-json", version.ref = "kotlinx-serialization-json" }
@@ -64,9 +66,10 @@ jetbrains-compose-runtime = { module = "org.jetbrains.compose.runtime:runtime",
6466
touchlab-stately-concurrency = { module = "co.touchlab:stately-concurrency", version.ref = "touchlab-stately" }
6567
napier = { module = "io.github.aakira:napier", version.ref = "napier" }
6668
flowExt = { module = "io.github.hoc081098:FlowExt", version.ref = "flowExt" }
67-
koin-core = { module = "io.insert-koin:koin-core", version.ref = "koin" }
68-
koin-android = { module = "io.insert-koin:koin-android", version.ref = "koin" }
69-
koin-androidx-compose = { module = "io.insert-koin:koin-androidx-compose", version.ref = "koin-androidx-compose" }
69+
koin-android = { module = "io.insert-koin:koin-android", version.ref = "koin-core" }
70+
koin-androidx-compose = { module = "io.insert-koin:koin-androidx-compose", version.ref = "koin-core" }
71+
koin-core = { module = "io.insert-koin:koin-core", version.ref = "koin-core" }
72+
koin-compose = { module = "io.insert-koin:koin-compose", version.ref = "koin-compose" }
7073
coil-compose = { module = "io.coil-kt:coil-compose", version.ref = "coil" }
7174
compose-rules-detekt = { module = "io.nlopez.compose.rules:detekt", version.ref = "compose-rules-detekt" }
7275

@@ -86,7 +89,11 @@ androidx-compose-material3 = { module = "androidx.compose.material3:material3" }
8689
androidx-compose-runtime = { module = "androidx.compose.runtime:runtime" }
8790
androidx-navigation-compose = { module = "androidx.navigation:navigation-compose", version.ref = "androidx-navigation" }
8891

92+
kmp-viewmodel = { module = "io.github.hoc081098:kmp-viewmodel", version.ref = "kmp-viewmodel" }
8993
kmp-viewmodel-savedstate = { module = "io.github.hoc081098:kmp-viewmodel-savedstate", version.ref = "kmp-viewmodel" }
94+
kmp-viewmodel-compose = { module = "io.github.hoc081098:kmp-viewmodel-compose", version.ref = "kmp-viewmodel" }
95+
kmp-viewmodel-koin-compose = { module = "io.github.hoc081098:kmp-viewmodel-koin-compose", version.ref = "kmp-viewmodel" }
96+
solivagant-navigation = { module = "io.github.hoc081098:solivagant-navigation", version.ref = "solivagant" }
9097
timber = { module = "com.jakewharton.timber:timber", version.ref = "timber" }
9198

9299
[plugins]
+1-1
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
distributionBase=GRADLE_USER_HOME
22
distributionPath=wrapper/dists
3-
distributionUrl=https\://services.gradle.org/distributions/gradle-8.6-bin.zip
3+
distributionUrl=https\://services.gradle.org/distributions/gradle-8.7-bin.zip
44
networkTimeout=10000
55
zipStoreBase=GRADLE_USER_HOME
66
zipStorePath=wrapper/dists
File renamed without changes.
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,109 @@
1+
plugins {
2+
alias(libs.plugins.kotlin.multiplatform)
3+
alias(libs.plugins.jetbrains.compose)
4+
alias(libs.plugins.android.library)
5+
alias(libs.plugins.kotlin.parcelize)
6+
}
7+
8+
kotlin {
9+
jvmToolchain {
10+
languageVersion = JavaLanguageVersion.of(17)
11+
vendor = JvmVendorSpec.AZUL
12+
}
13+
14+
androidTarget {
15+
compilations.all {
16+
kotlinOptions {
17+
jvmTarget = "11"
18+
}
19+
}
20+
}
21+
22+
jvm("desktop") {
23+
compilations.all {
24+
kotlinOptions {
25+
jvmTarget = "11"
26+
}
27+
}
28+
}
29+
30+
sourceSets {
31+
val desktopMain by getting
32+
33+
androidMain.dependencies {
34+
api(libs.androidx.compose.ui.tooling.preview)
35+
api(libs.androidx.activity.compose)
36+
37+
// Koin
38+
api(libs.koin.android)
39+
api(libs.koin.androidx.compose)
40+
}
41+
commonMain.dependencies {
42+
// Channel event bus
43+
implementation(project(":channel-event-bus"))
44+
45+
implementation(compose.runtime)
46+
implementation(compose.foundation)
47+
implementation(compose.material3)
48+
implementation(compose.ui)
49+
implementation(compose.components.resources)
50+
51+
// KMP View Model & Solivagant navigation
52+
api(libs.kmp.viewmodel)
53+
api(libs.kmp.viewmodel.savedstate)
54+
api(libs.kmp.viewmodel.compose)
55+
api(libs.kmp.viewmodel.koin.compose)
56+
api(libs.solivagant.navigation)
57+
58+
// Koin
59+
api(libs.koin.core)
60+
api(libs.koin.compose)
61+
62+
// Coroutines & FlowExt
63+
api(libs.coroutines.core)
64+
api(libs.flowExt)
65+
66+
// Immutable collections
67+
api(libs.kotlinx.collections.immutable)
68+
69+
// Napier logger
70+
api(libs.napier)
71+
}
72+
desktopMain.dependencies {
73+
implementation(compose.desktop.currentOs)
74+
}
75+
}
76+
}
77+
78+
android {
79+
namespace = "com.hoc081098.channeleventbus.sample.kmp.compose"
80+
compileSdk = libs.versions.sample.android.compile.get().toInt()
81+
82+
sourceSets["main"].manifest.srcFile("src/androidMain/AndroidManifest.xml")
83+
sourceSets["main"].res.srcDirs("src/androidMain/res")
84+
sourceSets["main"].resources.srcDirs("src/commonMain/resources")
85+
86+
defaultConfig {
87+
minSdk = libs.versions.android.min.get().toInt()
88+
}
89+
packaging {
90+
resources {
91+
excludes += "/META-INF/{AL2.0,LGPL2.1}"
92+
}
93+
}
94+
buildTypes {
95+
getByName("release") {
96+
isMinifyEnabled = false
97+
}
98+
}
99+
compileOptions {
100+
sourceCompatibility = JavaVersion.VERSION_11
101+
targetCompatibility = JavaVersion.VERSION_11
102+
}
103+
buildFeatures {
104+
buildConfig = true
105+
}
106+
dependencies {
107+
debugImplementation(libs.androidx.compose.ui.tooling)
108+
}
109+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,128 @@
1+
package com.hoc081098.channeleventbus.sample.kmp.compose
2+
3+
import androidx.compose.animation.AnimatedContentTransitionScope
4+
import androidx.compose.animation.fadeIn
5+
import androidx.compose.animation.fadeOut
6+
import androidx.compose.foundation.layout.consumeWindowInsets
7+
import androidx.compose.foundation.layout.fillMaxSize
8+
import androidx.compose.foundation.layout.padding
9+
import androidx.compose.material.icons.Icons
10+
import androidx.compose.material.icons.automirrored.filled.ArrowBack
11+
import androidx.compose.material3.CenterAlignedTopAppBar
12+
import androidx.compose.material3.ExperimentalMaterial3Api
13+
import androidx.compose.material3.Icon
14+
import androidx.compose.material3.IconButton
15+
import androidx.compose.material3.MaterialTheme
16+
import androidx.compose.material3.Scaffold
17+
import androidx.compose.material3.Surface
18+
import androidx.compose.material3.Text
19+
import androidx.compose.runtime.Composable
20+
import androidx.compose.runtime.Stable
21+
import androidx.compose.runtime.getValue
22+
import androidx.compose.runtime.mutableStateOf
23+
import androidx.compose.runtime.remember
24+
import androidx.compose.runtime.saveable.rememberSaveable
25+
import androidx.compose.runtime.setValue
26+
import androidx.compose.ui.Modifier
27+
import com.hoc081098.channeleventbus.sample.kmp.compose.common.MyApplicationTheme
28+
import com.hoc081098.channeleventbus.sample.kmp.compose.ui.home.detail.DetailScreenDestination
29+
import com.hoc081098.channeleventbus.sample.kmp.compose.ui.home.home.HomeScreenRouteDestination
30+
import com.hoc081098.channeleventbus.sample.kmp.compose.ui.register.stepone.RegisterStepOneScreenDestination
31+
import com.hoc081098.channeleventbus.sample.kmp.compose.ui.register.stepone.RegisterStepOneScreenRoute
32+
import com.hoc081098.channeleventbus.sample.kmp.compose.ui.register.stepthree.RegisterStepThreeScreenDestination
33+
import com.hoc081098.channeleventbus.sample.kmp.compose.ui.register.steptwo.RegisterStepTwoScreenDestination
34+
import com.hoc081098.solivagant.navigation.BaseRoute
35+
import com.hoc081098.solivagant.navigation.NavDestination
36+
import com.hoc081098.solivagant.navigation.NavEventNavigator
37+
import com.hoc081098.solivagant.navigation.NavHost
38+
import com.hoc081098.solivagant.navigation.NavHostDefaults
39+
import com.hoc081098.solivagant.navigation.NavRoot
40+
import kotlinx.collections.immutable.ImmutableSet
41+
import kotlinx.collections.immutable.adapters.ImmutableSetAdapter
42+
import org.koin.compose.KoinContext
43+
import org.koin.compose.koinInject
44+
45+
@Stable
46+
private val AllDestinations: ImmutableSet<NavDestination> by lazy {
47+
ImmutableSetAdapter(
48+
hashSetOf(
49+
RegisterStepOneScreenDestination,
50+
RegisterStepTwoScreenDestination,
51+
RegisterStepThreeScreenDestination,
52+
HomeScreenRouteDestination,
53+
DetailScreenDestination,
54+
),
55+
)
56+
}
57+
58+
@OptIn(ExperimentalMaterial3Api::class)
59+
@Suppress("LongMethod")
60+
@Composable
61+
fun ChannelEventBusSampleApp(
62+
modifier: Modifier = Modifier,
63+
navigator: NavEventNavigator = koinInject(),
64+
) {
65+
var currentRoute by rememberSaveable { mutableStateOf<BaseRoute?>(null) }
66+
67+
KoinContext {
68+
MyApplicationTheme(darkTheme = false) {
69+
Surface(
70+
modifier = modifier.fillMaxSize(),
71+
color = MaterialTheme.colorScheme.background,
72+
) {
73+
Scaffold(
74+
modifier = Modifier.fillMaxSize(),
75+
topBar = {
76+
CenterAlignedTopAppBar(
77+
title = { Text(text = currentRoute.toString()) },
78+
navigationIcon = {
79+
if (currentRoute !is NavRoot) {
80+
IconButton(onClick = remember { navigator::navigateBack }) {
81+
Icon(
82+
imageVector = Icons.AutoMirrored.Filled.ArrowBack,
83+
contentDescription = "Back",
84+
)
85+
}
86+
}
87+
},
88+
)
89+
},
90+
) { innerPadding ->
91+
NavHost(
92+
modifier = Modifier.fillMaxSize()
93+
.padding(innerPadding)
94+
.consumeWindowInsets(innerPadding),
95+
startRoute = RegisterStepOneScreenRoute,
96+
destinations = AllDestinations,
97+
navEventNavigator = navigator,
98+
destinationChangedCallback = { currentRoute = it },
99+
transitionAnimations = NavHostDefaults.transitionAnimations(
100+
enterTransition = {
101+
slideIntoContainer(
102+
towards = AnimatedContentTransitionScope.SlideDirection.Left,
103+
)
104+
},
105+
exitTransition = {
106+
slideOutOfContainer(
107+
towards = AnimatedContentTransitionScope.SlideDirection.Left,
108+
)
109+
},
110+
popEnterTransition = {
111+
slideIntoContainer(
112+
towards = AnimatedContentTransitionScope.SlideDirection.Right,
113+
)
114+
},
115+
popExitTransition = {
116+
slideOutOfContainer(
117+
towards = AnimatedContentTransitionScope.SlideDirection.Right,
118+
)
119+
},
120+
replaceEnterTransition = { fadeIn() },
121+
replaceExitTransition = { fadeOut() },
122+
),
123+
)
124+
}
125+
}
126+
}
127+
}
128+
}

0 commit comments

Comments
 (0)