Skip to content

Overview

A Kotlin framework for managing state and side effects. It is inspired by MVU(model, view, update) architecture and best of functional, declarative and reactive patterns. It enables building deterministic, composable, testable applications.

MVU graph

Quick example

To demonstrate some of the concepts we will make a simple counter application. It shows the current count and has two buttons to increment and decrement it.

One of the best practises when working with Formula is to first think what the UI needs for rendering and what actions the user will be able to take. This concept is called a RenderModel and is represented by a Kotlin data class.

Render Model

Render Model is an immutable representation of your view. It will be used to update Android views. Typically, it will also contain event listeners that will be invoked when user interacts with the UI.

data class CounterRenderModel(
  val count: String,
  val onDecrement: Listener<Unit>,
  val onIncrement: Listener<Unit>,
)

Once we define a Render Model, we can create a RenderView which is responsible for taking a RenderModel and applying it to Android Views.

Render View

Render View is an interface which is responsible for applying RenderModel to Android Views. This interface requires us to provide a render implementation by creating a Renderer. Renderer is a class that has an internal mechanism that checks the previous Render Model applied and only re-renders if it has changed.

class CounterRenderView(root: ViewGroup): RenderView<CounterRenderModel> {
  private val decrementButton: Button = root.findViewById(R.id.decrement_button)
  private val incrementButton: Button = root.findViewById(R.id.increment_button)
  private val countTextView: TextView = root.findViewById(R.id.count_text_view)

  override val render: Renderer<CounterRenderModel> = Renderer { model ->
    countTextView.text = model.count
    decrementButton.setOnClickListener {
      model.onDecrement()
    }
    incrementButton.setOnClickListener {
      model.onIncrement()
    }
  }
}

We now defined a single entry-point to our rendering (this makes debugging issues a lot easier). Anytime you need to update UI, just set a new Render Model.

renderView.render(renderModel)

Now that we have our rendering logic setup, let's define how we create the Render Model and handle user events. To have a dynamic UI that changes as user interacts with it requires some sort of state.

State

State is a Kotlin data class that contains all the necessary information/data to render your view. In our counter example, we need to keep track of the current count.

data class CounterState(val count: Int)

Given that this is a simple state, you could also use Int directly.

Formula

Formula is responsible for creating the Render Model. It can define an internal State class and respond to various events by transitioning to a new state.

class CounterFormula : Formula<Unit, CounterState, CounterRenderModel> {

  override fun initialState(input: Unit): CounterState = CounterState(count = 0)

  override fun evaluate(
    input: Unit,
    state: CounterState,
    context: FormulaContext<Int>
  ): Evaluation<CounterRenderModel> {
    val count = state.count
    return Evaluation(
      output = CounterRenderModel(
        title = "Count: $count",
        onDecrement = context.onEvent {
          transition(state.copy(count = count - 1))
        },
        onIncrement = context.onEvent {
          transition(state.copy(count = count + 1))
        }
      )
    )
  }
}

The most important part is the Formula.evaluate function. It gives us the current State and expects an Evaluation<RenderModel> back. Any time we transition to a new state, evaluate is called again and new Render Model is created.

There is also a special object called FormulaContext being passed. Formula Context allows us to respond to events by declaring transitions. We use context.onEvent for both onIncrement and onDecrement. Let's look at one of these functions closer.

onDecrement = context.onEvent {
  transition(state.copy(count = count - 1))
}

In response to the decrement event, we take the current count and subtract 1 from it. Then, we call transition method to create Transition<CounterState> object which we return.

If you notice, our logic currently allows user to decrement to a number below 0. We can update the transition logic to prevent this.

onDecrement = context.onEvent {
  if (count == 0) {
    none()
  } else {
    transition(state.copy(count = count - 1))
  }
}

The listener block uses a DSL to provide access to Transition.Factory which has the transition and none utility functions (take a look at that class for other utility functions).

Now that we defined our state management, let's connect it to our RenderView.

Using Formula

Formula is agnostic to other layers of abstraction. It can be used within activity or a fragment. You can convert Formula to an RxJava 3 Observable by using start extension function.

val formula = CounterFormula()
val state: Observable<CounterRenderModel> = formula.toObservable(input = Unit)

Ideally, it would be placed within a surface that survives configuration changes such as Android Components ViewModel. In this example, we will use Formula Android module. For using Formula with AndroidX ViewModel, take a look at AndroidX Guide.

Let's first define our Activity.

class MyActivity : FormulaAppCompatActivity() {
  private lateinit var counterRenderView: CounterRenderView

  override fun onCreate(state: Bundle?) {
    super.onCreate(state)
    setContentView(R.string.my_screen)

    counterRenderView = CounterRenderView(findViewById(R.id.counter))
  }

  fun render(model: CounterRenderModel) {
    counterRenderView.render(model)
  }
}

Now, let's connectCounterFormula to MyActivity.render function.

class MyApp : Application() {

  override fun onCreate() {
    super.onCreate()

    FormulaAndroid.init(this) {
      activity<MyActivity> {
        store(
          streams = {
            val formula = CounterFormula()
            update(formula.toObservable(), MyActivity::render)
          }
        )
      }
    }
  }
}

And that's it. You can see the full sample here.

To learn more about Formula Android module see Formula Android Guide.

Download

Add the library to your list of dependencies:

dependencies {
    implementation 'com.instacart.formula:formula-rxjava3:0.7.1'
    implementation 'com.instacart.formula:formula-android:0.7.1'
}

Inspiration

Formula would not have been possible without ideas from other projects such as

  • Elm
  • Cycle.js
  • React / Redux
  • Mobius
  • Square Workflows

License

The Clear BSD License

Copyright (c) 2022 Maplebear Inc. dba Instacart
All rights reserved.

Redistribution and use in source and binary forms, with or without modification, are permitted 
(subject to the limitations in the disclaimer below) provided that the following conditions are met:

* Redistributions of source code must retain the above copyright notice, this list of conditions and the following disclaimer.
* Redistributions in binary form must reproduce the above copyright notice, this list of conditions and the following disclaimer in the documentation and/or other materials provided with the distribution.
* Neither the name of Maplebear Inc. dba Instacart nor the names of its contributors may be used to endorse or promote products derived from this software without specific prior written permission.

NO EXPRESS OR IMPLIED LICENSES TO ANY PARTY'S PATENT RIGHTS ARE GRANTED BY THIS LICENSE. THIS SOFTWARE IS PROVIDED BY 
THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, 
THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE 
COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL 
DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; 
OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR 
TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED 
OF THE POSSIBILITY OF SUCH DAMAGE.