-
-
Notifications
You must be signed in to change notification settings - Fork 15
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
- Loading branch information
Showing
2 changed files
with
289 additions
and
1 deletion.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,9 +1,297 @@ | ||
[data:image/s3,"s3://crabby-images/10037/1003725a3c3a60096e5b6b3b818252c7ff75fd4c" alt="License"](https://github.com/arkivanov/Essenty/blob/master/LICENSE) | ||
[data:image/s3,"s3://crabby-images/042e9/042e98f36607ddbdae7223c971e1c03b6d8c6198" alt="Maven Central"](https://search.maven.org/search?q=g:com.arkivanov.essenty) | ||
[data:image/s3,"s3://crabby-images/17683/1768380119c0c89a3a7922cc565ecc789708079d" alt="License"](http://www.apache.org/licenses/LICENSE-2.0) | ||
[data:image/s3,"s3://crabby-images/f8253/f8253462b21c6e3c3552bf7b2f3be479cfb59cd5" alt="Twitter URL"](https://twitter.com/arkann1985) | ||
|
||
# Essenty | ||
|
||
The most essential libraries for Kotlin Multiplatform development. | ||
|
||
Supported targets: | ||
|
||
- `android` | ||
- `jvm` | ||
- `js` (`IR` and `LEGACY`) | ||
- `iosArm64`, `iosX64` | ||
- `watchosArm32`, `watchosArm64`, `watchosX64` | ||
- `tvosArm64`, `tvosX64` | ||
- `macosX64` | ||
- `linuxX64` | ||
|
||
## Lifecyle | ||
|
||
When writing Kotlin Multiplatform (common) code we often need to handle lifecycle events of a screen. For example, to stop background operations when the screen is destroyed, or to reload some data when the screen is activated. Essenty provides the `Lifecycle` API to help with lifecycle handling in the common code. It is very similar to [Android Activity lifecycle](https://developer.android.com/guide/components/activities/activity-lifecycle). | ||
|
||
### Setup | ||
|
||
Groovy: | ||
```groovy | ||
// Add the dependency, typically under the commonMain source set | ||
implementation "com.arkivanov.essenty:lifecycle:<essenty_version>" | ||
``` | ||
|
||
Kotlin: | ||
```kotlin | ||
// Add the dependency, typically under the commonMain source set | ||
implementation("com.arkivanov.essenty:lifecycle:<essenty_version>") | ||
``` | ||
|
||
### Lifecycle state transitions | ||
|
||
<img src="docs/media/LifecycleStates.png" width="512"> | ||
|
||
### Content | ||
|
||
The main [Lifecycle](https://github.com/arkivanov/Essenty/blob/master/lifecycle/src/commonMain/kotlin/com/arkivanov/essenty/lifecycle/Lifecycle.kt) interface provides ability to observe the lifecycle state changes. | ||
|
||
The [LifecycleRegistry](https://github.com/arkivanov/Essenty/blob/master/lifecycle/src/commonMain/kotlin/com/arkivanov/essenty/lifecycle/LifecycleRegistry.kt) interface extends both the `Lifecycle` and the `Lifecycle.Callbacks` at the same time. It can be used to manually control the lifecycle, for example in tests. | ||
|
||
The [LifecycleOwner](https://github.com/arkivanov/Essenty/blob/master/lifecycle/src/commonMain/kotlin/com/arkivanov/essenty/lifecycle/LifecycleOwner.kt) just holds the `Lifecyle`. It may be implemented by an arbitrary class, to provide convenient API. | ||
|
||
#### Android extensions | ||
|
||
From Android, the `Lifecycle` can be obtained by using special functions, can be found [here](https://github.com/arkivanov/Essenty/blob/master/lifecycle/src/androidMain/kotlin/com/arkivanov/essenty/lifecycle/AndroidExt.kt). | ||
|
||
|
||
### Usage example | ||
|
||
The lifecycle can be observed using its `subscribe`/`unsubscribe` methods: | ||
|
||
```kotlin | ||
import com.arkivanov.essenty.lifecycle.Lifecycle | ||
|
||
class SomeLogic(lifecycle: Lifecycle) { | ||
init { | ||
lifecycle.subscribe( | ||
object : Lifecycle.Callbacks { | ||
override fun onCreate() { | ||
// Handle lifecycle created | ||
} | ||
|
||
// onStart, onResume, onPause, onStop are also available | ||
|
||
override fun onDestroy() { | ||
// Handle lifecycle destroyed | ||
} | ||
} | ||
) | ||
} | ||
} | ||
``` | ||
|
||
Or using the extension functions: | ||
|
||
```kotlin | ||
import com.arkivanov.essenty.lifecycle.Lifecycle | ||
import com.arkivanov.essenty.lifecycle.doOnCreate | ||
import com.arkivanov.essenty.lifecycle.doOnDestroy | ||
import com.arkivanov.essenty.lifecycle.subscribe | ||
|
||
class SomeLogic(lifecycle: Lifecycle) { | ||
init { | ||
lifecycle.subscribe( | ||
onCreate = { /* Handle lifecycle created */ }, | ||
// onStart, onResume, onPause, onStop are also available | ||
onDestroy = { /* Handle lifecycle destroyed */ } | ||
) | ||
|
||
lifecycle.doOnCreate { | ||
// Handle lifecycle created | ||
} | ||
|
||
// doOnStart, doOnResume, doOnPause, doOnStop are also available | ||
|
||
lifecycle.doOnDestroy { | ||
// Handle lifecycle destroyed | ||
} | ||
} | ||
} | ||
``` | ||
|
||
## Parcelable and Parcelize | ||
|
||
Essenty brings both [Android Parcelable](https://developer.android.com/reference/android/os/Parcelable) interface and the `@Parcelize` annotation from [kotlin-parcelize](https://developer.android.com/kotlin/parcelize) compiler plugin to Kotlin Multiplatform, so they both can be used in common code. This is typically used for state/data preservation over [Android configuration changes](https://developer.android.com/guide/topics/resources/runtime-changes), when writing common code targeting Android. | ||
|
||
### Setup | ||
|
||
Groovy: | ||
```groovy | ||
plugins { | ||
id "kotlin-parcelize" // Apply the plugin for Android | ||
} | ||
// Add the dependency, typically under the commonMain source set | ||
implementation "com.arkivanov.essenty:parcelable:<essenty_version>" | ||
``` | ||
|
||
Kotlin: | ||
```kotlin | ||
plugins { | ||
id("kotlin-parcelize") // Apply the plugin for Android | ||
} | ||
|
||
// Add the dependency, typically under the commonMain source set | ||
implementation("com.arkivanov.essenty:parcelable:<essenty_version>") | ||
``` | ||
|
||
### Usage example | ||
|
||
Once the dependency is added and the plugin is applied, we can use it as follows: | ||
|
||
```kotlin | ||
import com.arkivanov.essenty.parcelable.Parcelable | ||
import com.arkivanov.essenty.parcelable.Parcelize | ||
|
||
@Parcelize | ||
data class User( | ||
val id: Long, | ||
val name: String | ||
) : Parcelable | ||
``` | ||
|
||
When compiled for Android, the `Parcelable` implementation will be generated automatically. When compiled for other targets, it will be just a regular class without any extra generated code. | ||
|
||
#### Parcelize for Darwin/Apple targets | ||
|
||
Currently there is no extra code generated when compiled for Darwin/Apple targets. However I made a proof of concept: [kotlin-parcelize-darwin](https://github.com/arkivanov/kotlin-parcelize-darwin) compiler plugin. It is not used yet by Essenty, and the applicabilty is being considered. Please raise a [Discussion](https://github.com/arkivanov/Essenty/discussions) if you are interested. | ||
|
||
## StateKeeper | ||
|
||
When writing common code targetting Android, it might be required to preserve some data over Android configuration changes or process death. For this purpose, Essenty provides the `StateKeeper` API, which is inspired by the AndroidX [SavedStateHandle](https://developer.android.com/reference/androidx/lifecycle/SavedStateHandle). The `StateKeeper` API relies on the `Parcelable` interface provided by the `parcelable` module described above. | ||
|
||
### Setup | ||
|
||
Groovy: | ||
```groovy | ||
// Add the dependency, typically under the commonMain source set | ||
implementation "com.arkivanov.essenty:state-keeper:<essenty_version>" | ||
``` | ||
|
||
Kotlin: | ||
```kotlin | ||
// Add the dependency, typically under the commonMain source set | ||
implementation("com.arkivanov.essenty:state-keeper:<essenty_version>") | ||
``` | ||
|
||
### Content | ||
|
||
The main [StateKeeper](https://github.com/arkivanov/Essenty/blob/master/state-keeper/src/commonMain/kotlin/com/arkivanov/essenty/statekeeper/StateKeeper.kt) interface provides ability to register/unregister state suppliers, and also to consume any previously saved state. | ||
|
||
The [StateKeeperDispatcher](https://github.com/arkivanov/Essenty/blob/master/state-keeper/src/commonMain/kotlin/com/arkivanov/essenty/statekeeper/StateKeeperDispatcher.kt) interface extens `StateKeeper` and allows state saving, by calling all registered state providers. | ||
|
||
The [StateKeeperOwner](https://github.com/arkivanov/Essenty/blob/master/state-keeper/src/commonMain/kotlin/com/arkivanov/essenty/statekeeper/StateKeeperOwner.kt) interface is just a holder of `StateKeeper`. It may be implemented by an arbitrary class, to provide convenient API. | ||
|
||
#### Android extensions | ||
|
||
From Android side, `StateKeeper` can be obtained by using special functions, can be found [here](https://github.com/arkivanov/Essenty/blob/master/state-keeper/src/androidMain/kotlin/com/arkivanov/essenty/statekeeper/AndroidExt.kt). | ||
|
||
#### Usage example | ||
|
||
```kotlin | ||
import com.arkivanov.essenty.parcelable.Parcelable | ||
import com.arkivanov.essenty.parcelable.Parcelize | ||
import com.arkivanov.essenty.statekeeper.StateKeeper | ||
import com.arkivanov.essenty.statekeeper.consume | ||
|
||
class SomeLogic(stateKeeper: StateKeeper) { | ||
// Use the saved State if any, otherwise create a new State | ||
private var state: State = stateKeeper.consume("SAVED_STATE") ?: State() | ||
|
||
init { | ||
// Register the State supplier | ||
stateKeeper.register("SAVED_STATE") { state } | ||
} | ||
|
||
@Parcelize | ||
private class State( | ||
val someValue: Int = 0 | ||
) : Parcelable | ||
} | ||
``` | ||
|
||
## InstanceKeeper | ||
|
||
When writing common code targetting Android, it might be required to retain objects over Android configuration changes. This use case is covered by the `InstanceKeeper` API, which is similar to the AndroidX [ViewModel](https://developer.android.com/topic/libraries/architecture/viewmodel). | ||
|
||
### Setup | ||
|
||
Groovy: | ||
```groovy | ||
// Add the dependency, typically under the commonMain source set | ||
implementation "com.arkivanov.essenty:instance-keeper:<essenty_version>" | ||
``` | ||
|
||
Kotlin: | ||
```kotlin | ||
// Add the dependency, typically under the commonMain source set | ||
implementation("com.arkivanov.essenty:instance-keeper:<essenty_version>") | ||
``` | ||
|
||
### Content | ||
|
||
The main [InstanceKeeper](https://github.com/arkivanov/Essenty/blob/master/instance-keeper/src/commonMain/kotlin/com/arkivanov/essenty/instancekeeper/InstanceKeeper.kt) interface is responsible for storing object instances, represented by the [InstanceKeeper.Instance] interface. Instances of the `InstanceKeeper.Instance` interface survive Android Configuration changes, the `InstanceKeeper.Instance.onDestroy()` method is called when `InstanceKeeper` goes out of scope (e.g. the screen is finished). | ||
|
||
The [InstanceKeeperDispatcher](https://github.com/arkivanov/Essenty/blob/master/instance-keeper/src/commonMain/kotlin/com/arkivanov/essenty/instancekeeper/InstanceKeeperDispatcher.kt) interface extens `InstanceKeeper` and adds ability to destroy all registered instances. | ||
|
||
The [InstanceKeeperOwner](https://github.com/arkivanov/Essenty/blob/master/instance-keeper/src/commonMain/kotlin/com/arkivanov/essenty/instancekeeper/InstanceKeeperOwner.kt) interface is just a holder of `InstanceKeeper`. It may be implemented by an arbitrary class, to provide convenient API. | ||
|
||
#### Android extensions | ||
|
||
From Android side, `InstanceKeeper` can be obtained by using special functions, can be found [here](https://github.com/arkivanov/Essenty/blob/master/instance-keeper/src/androidMain/kotlin/com/arkivanov/essenty/instancekeeper/AndroidExt.kt). | ||
|
||
### Usage example | ||
|
||
```kotlin | ||
import com.arkivanov.essenty.instancekeeper.InstanceKeeper | ||
import com.arkivanov.essenty.instancekeeper.getOrCreate | ||
|
||
class SomeLogic(instanceKeeper: InstanceKeeper) { | ||
// Get the existing instance or create a new one | ||
private val thing: RetainedThing = instanceKeeper.getOrCreate { RetainedThing() } | ||
} | ||
|
||
/* | ||
* Survives Android configuration changes. | ||
* ⚠️ Pay attention to not leak any dependencies. | ||
*/ | ||
class RetainedThing : InstanceKeeper.Instance { | ||
override fun onDestroy() { | ||
// Called when the screen is finished | ||
} | ||
} | ||
``` | ||
|
||
## BackPressedDispatcher | ||
|
||
The `BackPressedDispatcher` API provides ability to handle back button clicks (e.g. an Android device's back button), in common code. This API is similar to AndroidX [OnBackPressedDispatcher](https://developer.android.com/reference/androidx/activity/OnBackPressedDispatcher). | ||
|
||
### Setup | ||
|
||
Groovy: | ||
```groovy | ||
// Add the dependency, typically under the commonMain source set | ||
implementation "com.arkivanov.essenty:back-pressed:<essenty_version>" | ||
``` | ||
|
||
Kotlin: | ||
```kotlin | ||
// Add the dependency, typically under the commonMain source set | ||
implementation("com.arkivanov.essenty:back-pressed:<essenty_version>") | ||
``` | ||
|
||
### Content | ||
|
||
The [BackPressedHandler](https://github.com/arkivanov/Essenty/blob/master/back-pressed/src/commonMain/kotlin/com/arkivanov/essenty/backpressed/BackPressedHandler.kt) interface provides ability to register/unregister back button handlers. When the device's back button is clicked, all registered handlers are called in reverse order. If a handler returns `true` then the event is considered as handled and the handling process stops, the remaining handlers are not called. If none of the handlers returned `true` then the event is considered as unhandled. | ||
|
||
The [BackPressedDispatcher](https://github.com/arkivanov/Essenty/blob/master/back-pressed/src/commonMain/kotlin/com/arkivanov/essenty/backpressed/BackPressedDispatcher.kt) interface extends `BackPressedHandler` and is responsible for triggering the registered handlers. The `BackPressedDispatcher.onBackPressed()` triggers all registered handlers in reverse order, returns `true` if the event is handled, and `false` if the event is unhandled. | ||
|
||
#### Android extensions | ||
|
||
From Android side, `BackPressedDispatcher` can be obtained by using special functions, can be found [here](https://github.com/arkivanov/Essenty/blob/master/back-pressed/src/androidMain/kotlin/com/arkivanov/essenty/backpressed/AndroidExt.kt). | ||
|
||
### Usage example | ||
|
||
TBD | ||
|
||
## Author | ||
|
||
Twitter: [@arkann1985](https://twitter.com/arkann1985) | ||
|
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.