This guide contains development related information to help a developer in getting better understanding of project structure.
The architecture of PrivacyCentralApp is based on clean architecture. For presentation layer, we use Model-View-Intent design pattern which is a unidirectional reactive flow pattern. We use it in conjunction to ViewModel to make our features lifecycle aware. Our android app is having single activity multiple fragments.
Clean architecture is the building block of PrivacyCentralApp. This architecture is based on the following principles:
- Independent of Frameworks. The architecture does not depend on the existence of some library of feature laden software. This allows you to use such frameworks as tools, rather than having to cram your system into their limited constraints.
- Testable. The business rules can be tested without the UI, Database, Web Server, or any other external element.
- Independent of UI. The UI can change easily, without changing the rest of the system. A Web UI could be replaced with a console UI, for example, without changing the business rules.
- Independent of Database. You can swap out Oracle or SQL Server, for Mongo, BigTable, CouchDB, or something else. Your business rules are not bound to the database.
- Independent of any external agency. In fact your business rules simply don’t know anything at all about the outside world.
MVI is used at the presentation layer of clean architecture. It is very much similar to Redux in terms of implementation and working. It has three main components.
- View: This is where activities, fragments and other android components live. It is responsible for publishing user intent/actions to the model and rendering the state returned by the model. In PrivacyCentralApp, it is just an interface which is implemented by android components.
- Intent: In context of our app, we call them actions. These are simple data classes having any extra payload like inputs, ids etc.
- Model (data layer at presentation level): This is responsible for processing the actions, communicating with domain use-cases and mutating the state of the model. It acts as a Store from redux but for our use case, we call it a Feature.
In this app, we have implemented MVI using Kotlin Flow.
Elements of a feature:
- Actor: It is just a function that takes current state, user action as input and produces an effect (result) as output. This function generally makes the call to external APIs and usecases.
- Reducer: It is also a very simple function whose inputs are current state, effect from the actor and it returns new state.
- State: Simple POJO (kotlin data class) representing various UI states of the application.
- Effect: A POJO (kotlin data class) which is returned from the actor function.
- SingleEventProducer: This is a function which is invoked by the reducer to publish single events (that can/should only be consumed once like displaying toast, snackbar message or sending an analytics event). This function takes action, effect, current state as input and it returns a
SingleEvent
. By default this function is null for any Feature.
Looking at the diagram from right to left:
- Database, Preferences and API Helpers: These are a set of classes with very specific responsibilities for each of them. They can perform some business logic, communicate with the database or make a call to third party API. These classes are implementation of abstract gateways and interfaces (see below).
- Abstract Gateways and Interfaces: An abstraction layer that sits between repositories and framework implementations. The objective of this layer is to keep repositories independent from framework (like databases).
- Respositoy Implementation: These classes provide the implementation of abstract repository defined in the domain.
- Repository: They are the core part of our domain. Each repository is a just an interface whose implementation is provided by the data layer.
- Usecase/Interactors/Observers: The act as the bridge between the presentation layer and data layer.
- Entities: These are business objects for the application. They encapsulate the most general and high-level rules. They are the least likely to change when something external changes.
- Features: MVI specific model (discussed above).
- ViewModel: arch-component lifecycle aware viewmodel.
- Views: Android high level components like activities, fragments, etc.
Imaging you have to implement a fake location feature.
- Create a new package under
features
calledfakelocation
- Create a new feature class called
FakeLocationFeature
and make it extend the BaseFeature class as below:
class FakeLocationFeature(
initialState: State,
coroutineScope: CoroutineScope,
reducer: Reducer<State, Effect>,
actor: Actor<State, Action, Effect>,
singleEventProducer: SingleEventProducer<State, Action, Effect, SingleEvent>
) : BaseFeature<FakeLocationFeature.State, FakeLocationFeature.Action, FakeLocationFeature.Effect, FakeLocationFeature.SingleEvent>(
initialState,
actor,
reducer,
coroutineScope,
{ message -> Log.d("FakeLocationFeature", message) },
singleEventProducer
) {
// Other elements goes here.
}
- Define various elements for the feature in the above class
// State to be reflected in the UI
data class State(val location: Location)
// User triggered actions
sealed class Action {
data class UpdateLocationAction(val latLng: LatLng) : Action()
object UseRealLocationAction : Action()
object UseSpecificLocationAction : Action()
data class SetFakeLocationAction(val latitude: Double, val longitude: Double) : Action()
}
// Output from the actor after processing an action
sealed class Effect {
data class LocationUpdatedEffect(val latitude: Double, val longitude: Double) : Effect()
object RealLocationSelectedEffect : Effect()
...
...
data class ErrorEffect(val message: String) : Effect()
}
- Create a static
create
function in feature which returns the feature instance:
companion object {
fun create(
initialState: State = <initial state>
coroutineScope: CoroutineScope
) = FakeLocationFeature(
initialState, coroutineScope,
reducer = { state, effect ->
when (effect) {
Effect.RealLocationSelectedEffect -> state.copy(
location = state.location.copy(
mode = LocationMode.REAL_LOCATION
)
)
is Effect.ErrorEffect, Effect.SpecificLocationSavedEffect -> state
is Effect.LocationUpdatedEffect -> state.copy(
location = state.location.copy(
latitude = effect.latitude,
longitude = effect.longitude
)
)
}
},
actor = { _, action ->
when (action) {
is Action.UpdateLocationAction -> flowOf(
Effect.LocationUpdatedEffect(
action.latLng.latitude,
action.latLng.longitude
)
)
is Action.SetFakeLocationAction -> {
val location = Location(
LocationMode.CUSTOM_LOCATION,
action.latitude,
action.longitude
)
// TODO: Call fake location api with specific coordinates here.
val success = DummyDataSource.setLocationMode(
LocationMode.CUSTOM_LOCATION,
location
)
if (success) {
flowOf(
Effect.SpecificLocationSavedEffect
)
} else {
flowOf(
Effect.ErrorEffect("Couldn't select location")
)
}
}
Action.UseRealLocationAction -> {
// TODO: Call turn off fake location api here.
val success = DummyDataSource.setLocationMode(LocationMode.REAL_LOCATION)
if (success) {
flowOf(
Effect.RealLocationSelectedEffect
)
} else {
flowOf(
Effect.ErrorEffect("Couldn't select location")
)
}
}
Action.UseSpecificLocationAction -> {
flowOf(Effect.SpecificLocationSelectedEffect)
}
}
},
singleEventProducer = { _, _, effect ->
when (effect) {
Effect.SpecificLocationSavedEffect -> SingleEvent.SpecificLocationSavedEvent
Effect.RealLocationSelectedEffect -> SingleEvent.RealLocationSelectedEvent
is Effect.ErrorEffect -> SingleEvent.ErrorEvent(effect.message)
else -> null
}
}
)
}
- Create a
viewmodel
like below:
class FakeLocationViewModel : ViewModel() {
private val _actions = MutableSharedFlow<FakeLocationFeature.Action>()
val actions = _actions.asSharedFlow()
val fakeLocationFeature: FakeLocationFeature by lazy {
FakeLocationFeature.create(coroutineScope = viewModelScope)
}
fun submitAction(action: FakeLocationFeature.Action) {
viewModelScope.launch {
_actions.emit(action)
}
}
}
- Create a
fragment
for your feature and make sure it implementsMVIView<>
interface - Initialize (or retrieve the existing) instance of viewmodel in your
fragment
class by using extension function.
private val viewModel: FakeLocationViewModel by viewModels()
- In
onCreate
method of fragment, launch a coroutine to bind the view to feature and to listen single events.
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
lifecycleScope.launchWhenStarted {
viewModel.fakeLocationFeature.takeView(this, this@FakeLocationFragment)
}
lifecycleScope.launchWhenStarted {
viewModel.fakeLocationFeature.singleEvents.collect { event ->
// Do something with event
}
}
}
- To render the state in UI, override the
render
function of MVIView. - For publishing ui actions, use
viewModel.submitAction(action)
.
Everything is lifecycle aware so we don't need to anything manually here.
This project integrates a combination of unit tests, functional test and code styling tools. To run unit tests on your machine:
./gradlew test
To run code style check and formatting tool:
./gradlew spotlessCheck
./gradlew spotlessApply
The project currently doesn't have exactly the same mentioned structure as it is just a POC and will be improved.
- Add domain layer with usecases.
- Add data layer with repository implementation.
- Add unit tests and code coverage.
- Implement Hilt DI.