-
Notifications
You must be signed in to change notification settings - Fork 3
Project Architecture (EN)
The project has three main modules:
Module | Description |
---|---|
android | The android module contains code used for the Android app. |
ios | The ios module contains code used for the iOS app. |
shared | The shared module contains common code used by the android module and the ios module. To access platform APIs, the code in this module make use of the expect/actual mechanism provided by Kotlin. The code in this module rely on multiplatform libraries such as SQLDelight and Kotlin Coroutines. |
ViewModel
s provide the data for the UI.
The repository module serves as a clean API so that that the rest of the app can retrieve data easily. It acts as a mediator for between the DB
and the API
modules. The module handle data operations by using Coroutines Suspend Functions.
The repository creates and returns a Coroutine Channel that the app "subscribes" to in order to be notified of further updates. The Channel
will be updated with relevant data and a status (loading, success or failure). The status allows the UI to update its state accordingly. For example, for the loading status, the UI can display a loading bar.
Here are the operations that are generally done by the repository module:
- The process starts when the rest of the app calls-in the repository.
- The repository starts by calling the DB module which will fetch the data stored in the DB.
- The DB module returns a Query which is a a listenable, typed query generated by SQLDelight. Notice that, if needed, this object can be easily converted to a Coroutine Channel by using the
asChannel
extension function.
Here is an example of usage with the map operator:
val query = queries.selectAll()
query.asChannel().map(context) { it.executeAsList() }
- The Query finishes and the channel is updated with cached data and a "loading" status. The app should have displayed a loading state by now.
- The repository calls the API module to start fetching data from the webservice.
- The webservice returns a response. If the call succeeded, the repository updates the DB with the new data. The
Channel
is also updated with a "success status" and new data coming from the DB. If the call failed, theChannel
is updated with a "failure" status.
To do so, use the Resource
class. It encapsulates both the data and its state.
class Resource<T> private constructor(val status: Status, val data: T?, val message: String?) {
enum class Status {
SUCCESS,
ERROR,
LOADING
}
companion object {
fun <T> success(data: T): Resource<T> = Resource(Status.SUCCESS, data, null)
fun <T> error(msg: String, data: T?): Resource<T> = Resource(Status.ERROR, data, msg)
fun <T> loading(data: T?): Resource<T> = Resource(Status.LOADING, data, null)
}
...
}
Use one of the static methods to create a Resource
instance. For example, to encapsulate a data called "myData" and a loading status, you can do like so:
Resource.loading(myData)
The database module uses SQLDelight to persist the app's data. The library let you write write SQL statements from which their Gradle plugin will generate APIs to run your queries. Please refer to their documentation for more information.
A viewModelScope extension was added for the ViewModel class. We can use it to launch coroutines inside our ViewModel
s.