Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/frankois944/kmp-mvvm-exploration
Kotlin for iOS using MVVM pattern with KMP ViewModel
https://github.com/frankois944/kmp-mvvm-exploration
android datastore ios kmp koin kotlin kotlin-multiplatform mvvm mvvm-architecture swiftui uikit
Last synced: about 1 month ago
JSON representation
Kotlin for iOS using MVVM pattern with KMP ViewModel
- Host: GitHub
- URL: https://github.com/frankois944/kmp-mvvm-exploration
- Owner: frankois944
- License: unlicense
- Created: 2024-05-30T09:33:22.000Z (8 months ago)
- Default Branch: main
- Last Pushed: 2024-12-08T07:52:33.000Z (about 1 month ago)
- Last Synced: 2024-12-08T08:25:07.529Z (about 1 month ago)
- Topics: android, datastore, ios, kmp, koin, kotlin, kotlin-multiplatform, mvvm, mvvm-architecture, swiftui, uikit
- Language: Kotlin
- Homepage:
- Size: 404 KB
- Stars: 18
- Watchers: 1
- Forks: 3
- Open Issues: 0
-
Metadata Files:
- Readme: README.md
- License: LICENSE
Awesome Lists containing this project
README
# Exploration of KMP MVVM and other useful features for iOS developer
I'm trying to find a good solution for using the MVVM pattern with the KMP ViewModel on SwiftUI/UIKit.
It's not that simple, I've been working on it for some time, and with the advancement of KMP, it sounds easier, but not so much :)
The KMP ViewModel approach on [Android is fully supported](https://developer.android.com/topic/libraries/architecture/viewmodel), using Kotlin multiplatform or not, it's the same implementation.
Otherwise, on iOS, it's kind of experimental (really ?); the KMP ViewModel is not made to work on this target; we need to find some workarounds for correctly using it; and the main issue is the **lifecycle**.
Other features presented in this repository are optional for using the MVVM pattern, but I think they're kind of useful.
You will find inside this repo :
- [Requirement](#requirement)
- [MVVM with a different approach](#the-viewmodel)
- [SwiftUI SKIE observable (iOS 15 and later)](#swiftui-skie-observable-ios-15-and-later)
- [SwiftUI SKIE observable (iOS 14 and earlier)](#swiftui-skie-observable-ios-14-and-earlier)
- [Custom macro](#mvvm-using-macro)
- [Pure SwiftUI MVVM](#pure-swiftui-mvvm)
- [UIKit](#uikit)
- injection with [Koin annotation](https://insert-koin.io/)
- [Getting the ViewModel or any instance from Swift/Koin](#getting-the-viewmodel-or-any-instance-from-swift-and-koin)
- Logging with [Kermit](https://kermit.touchlab.co/)
- Usage of [DataStore](https://developer.android.com/jetpack/androidx/releases/datastore)
- and more little experiences## Requirement
So, the most interesting thing is about the MVVM and what do we need :
Inspiration from this repository https://github.com/joreilly/FantasyPremierLeague and this issue https://github.com/joreilly/FantasyPremierLeague/issues/231
- First step is exporting the Kotlin MVVM dependency for accessing the KMP ViewModel from Swift
[shared Gradle file](https://github.com/frankois944/kmp-mvvm-exploration/blob/main/Shared/build.gradle.kts)
```gradle
it.binaries.framework {
//...
export(libs.androidx.lifecycle.viewmodel)
}
commonMain.dependencies {
//...
api(libs.androidx.lifecycle.viewmodel)
}
```- Then import [SKIE](https://skie.touchlab.co/) to fully access the Kotlin Flow from Swift
And activate some useful features and expand the SwiftUI capabilities.
[shared Gradle file](https://github.com/frankois944/kmp-mvvm-exploration/blob/main/Shared/build.gradle.kts)
```gradle
skie {
features {
// https://skie.touchlab.co/features/flows-in-swiftui (>= iOS15)
enableSwiftUIObservingPreview = true // (>= iOS15)
// https://skie.touchlab.co/features/combine
enableFutureCombineExtensionPreview = true
enableFlowCombineConvertorPreview = true
}
}
```- Finally, create a [SwiftUI class](https://github.com/frankois944/kmp-mvvm-exploration/blob/main/iosApp/iosApp/SharedViewModel.swift) for managing the KMP ViewModel lifecycle
By wrapping the KMP ViewModel inside an ObservableObject, the shared class is now aligned with the lifecycle of the SwiftUI view.
```swift
class SharedViewModel: ObservableObject {private let key = String(describing: type(of: VM.self))
private let viewModelStore = ViewModelStore()// Injecting the viewmodel
init(_ viewModel: VM = .init()) {
viewModelStore.put(key: key, viewModel: viewModel)
}var instance: VM {
(viewModelStore.get(key: key) as? VM)!
}deinit {
viewModelStore.clear()
}
}
```
* ### The ViewModelBased on this shared Kotlin [ViewModel](https://github.com/frankois944/kmp-mvvm-exploration/blob/main/Shared/src/commonMain/kotlin/fr/frankois944/kmpviewmodel/viewmodels/mainscreen/MainScreenViewModel.kt).
Also, you can find the Android integration [here](https://github.com/frankois944/kmp-mvvm-exploration/blob/main/androidApp/src/main/java/fr/frankois944/kmpviewmodel/android/MyFirstScreen.kt).
Note: This viewmodel doesn't represent of what a ViewModel **MUST BE**.
The most important is a [Kotlin Flow](https://kotlinlang.org/docs/flow.html) (Flow/StateFlow/SharedFlow) is transformed in [Swift async](https://www.avanderlee.com/swift/async-await/). Also, SKIE makes it easier to implement with Swift.
* ### SwiftUI SKIE observable (iOS 15 and later)
[Example with SKIE](https://github.com/frankois944/kmp-mvvm-exploration/blob/main/iosApp/iosApp/SwiftUI/MyFirstScreenWithSkie.swift).
This approach uses the [SKIE flows for SwiftUI capability](https://skie.touchlab.co/features/flows-in-swiftui), which use the `.task` SwiftUI modifier.
* ### SwiftUI SKIE observable (iOS 14 and earlier)
[Example with a customized SKIE](https://github.com/frankois944/kmp-mvvm-exploration/blob/main/iosApp/iosApp/SwiftUI/MyFirstScreenWithSkieIOS14.swift), a [copy of SKIE collect](https://github.com/frankois944/kmp-mvvm-exploration/blob/main/iosApp/iosApp/SkieCollectForiOS14.swift) methods that use the `.onAppear` SwiftUI modifier.
This approach uses the [SKIE Flow capability](https://skie.touchlab.co/features/flows) and reproduces the [SKIE flows for SwiftUI](https://skie.touchlab.co/features/flows-in-swiftui)
* ### MVVM using Macro
[Example with a macro](https://github.com/frankois944/kmp-mvvm-exploration/blob/main/iosApp/iosApp/SwiftUI/MyFirstScreenWithMacro.swift), more like an iOS dev will commonly use.
This approach uses a [macro I made](https://github.com/frankois944/KTViewModelBuilder) to automatically wrap a KMP ViewModel inside an ObservableObject, almost like a SwiftUI ViewModel, you can also add it to your project.
You can see an example of a [generated code here](https://github.com/frankois944/KTViewModelBuilder?tab=readme-ov-file#swiftui-viewmodel).
* ### Pure SwiftUI MVVM
[Example with a common SwiftUI ViewModel](https://github.com/frankois944/kmp-mvvm-exploration/blob/main/iosApp/iosApp/SwiftUI/MyFirstScreenWithSwiftViewModel.swift)
There is no usage of KMP MVVM here, just like in an MVVM SwiftUI class, but we need to use the Koin injection capacities, or you can directly use the class instead of injection.
* ### UIKit
[Example with a UIViewController](https://github.com/frankois944/kmp-mvvm-exploration/blob/main/iosApp/iosApp/UIKIt/MyFirstScreenViewController.swift)
UIKit is not dead, we can use the `SharedViewModel` class and the [SKIE combine support ](https://skie.touchlab.co/features/combine)
## Thinking
The goal of this playground is to align the behavior between the Android ViewModel and the SwiftUI ViewModel. It's not that simple, as the ViewModel model must be the same, but the lifecycle of the Viewholder is different.
Look at the logs I added to verify the lifecycle, it should be exactly the same on each approach.
## Getting the ViewModel or any instance from Swift and Koin
As this playground is using Koin, I want to get my ViewModel from it, not directly from the constructor (it's still working, but it's not great).
So we can use Koin qualifiers and parameters, like Koin for Android.
- We need to [export two Kotlin methods](https://github.com/frankois944/kmp-mvvm-exploration/blob/main/Shared/src/iosMain/kotlin/fr/frankois944/kmpviewmodel/AppInit.ios.kt) that resolve the ObjC class/protocol to the Kotlin Class from the Swift Application
- Store the Kotlin Koin Application instance somewhere and make it accessible everywhere in the Swift App
```swift
// For example: store in swift singleton the koin application
AppContext.shared.koinApplication = // instance of initialized koinapplication
```
- [Then create some swift helpers](https://github.com/frankois944/kmp-mvvm-exploration/blob/main/iosApp/iosApp/KoinHelper.swift)
```swift
private class KoinQualifier: Koin_coreQualifier {
init(value: String) {
self.value = value
}
var value: String
}extension Koin_coreKoinApplication {
// reproducing the koin `get()` method behavior
// we can set qualifier and parameters
func get(qualifier: String? = nil, parameters: [Any]? = nil) -> T {
let ktClass: KotlinKClass?
// check if T is a Class or a Protocol and get the linked kotlin class
if let protocolType = NSProtocolFromString("\(T.self)") {
ktClass = Shared.getOriginalKotlinClass(objCProtocol: protocolType)
} else {
ktClass = Shared.getOriginalKotlinClass(objCClass: T.self)
}guard let ktClass else {
// no Kotlin Class found, it's an critical error
fatalError("Cant resolve objc class \(T.self)")
}var koinQualifier: Koin_coreQualifier?
if let qualifier = qualifier {
koinQualifier = KoinQualifier(value: qualifier)
}var koinParameters: (() -> Koin_coreParametersHolder)?
if let parameters {
koinParameters = {
.init(_values: .init(array: parameters), useIndexedValues: nil)
}
}guard let instance = koin.get(clazz: ktClass,
qualifier: koinQualifier,
parameters: koinParameters) as? T else {
fatalError("Cant resolve Koin Injection \(self)")
}
return instance
}
}/// propertyWrapper like `by inject()` koin method
@propertyWrapper struct KoinInject {
var qualifier: String?
var parameters: [Any]?init(qualifier: String? = nil, parameters: [Any]? = nil) {
self.qualifier = qualifier
self.parameters = parameters
}lazy var wrappedValue: T = {
return koinGet(qualifier: qualifier, parameters: parameters)
}()
}/// like the `get()` koin method
func koinGet(qualifier: String? = nil, parameters: [Any]? = nil) -> T {
return AppContext.shared.koinApplication.get(qualifier: qualifier,
parameters: parameters)
}```
- Finally, get the instance from different ways
```swift
// lazy loading of any instance
@KoinInject private var accountService
// direct loading of any instance
private let logger: KermitLogger = koinGet(parameters: ["FirstScreenDataStore"])// get the ViewModel as example
@StateObject private var viewModel: SharedViewModel
init(param1: String? = nil) {
_viewModel = StateObject(wrappedValue: { .init(parameters: ["IOS-MyFirstScreenWithoutMacro"]) }())
}
// or
@StateObject private var viewModel: SharedViewModel = .init(koinGet())
// and many other ways
```