Cloudzilla Logo

Advanced Form Operations in Jetpack Compose

When developing an application, it is always a good idea to write code that is not repetitive. This principle is known as DRY. It works hand in hand with code decoupling.

These two ensure that code is easily scalable, as a change in one part of the application does not affect other unrelated areas. A modification is done uniformly across the components/modules.

In this article, we will look at a DRY approach to working with text fields in Jetpack compose.

By the end of the article, you should:

  • Know how to work with text fields in Jetpack compose. This involves state management and validation.
  • Know how to decouple text field operations.
  • Know how to abstract the various form activities.

Prerequisites

In order to follow up with this tutorial comfortably, you will need:

  • Android studio 4.2 (Arctic Fox) and above.
  • Knowledge of Kotlin language.
  • Knowledge in Jetpack compose.
  • Physical device or emulator to run the application.

Step 1: Setup

To get started, open up Android Studio and create a new project using the "Empty Compose activity" template. Give it any name you would like. You can go ahead to customize the theme to your liking.

Step 2: Form operations

When developing mobile applications, we are likely to come across Forms or some form of input to collect data from users. With Jetpack compose, we have the TextField composable from Material.

Go ahead and add it to your screen.

@Composable
fun Screen(){
    Column {
        TextField(value = "", onValueChange = {})
    }
}

Once you run your application, you will notice no change as you type in the input field. This is because you are not updating the state of the field in the onValueChange callback.

You can read more about state management on the official documentation. So go ahead and update the Form composable.

@Composable
fun Screen(){
    var name by remember { mutableStateOf("") }
    Column {
        TextField(value = name, onValueChange = { value -> name = value })
    }
}

As you can see, you are managing the state of the field by defining a State<String> then updating it as the value changes. Let us go ahead and add another field to get the email address and button to submit our details.

@Composable
fun Screen(){
    var name by remember { mutableStateOf("") }
    var email by remember { mutableStateOf("") }

    Column {
        TextField(
            value = name,
            modifier = Modifier.padding(10.dp),
            onValueChange = { value -> name = value }
        )
        TextField(
            value = email,
            modifier = Modifier.padding(10.dp),
            onValueChange = { value -> email = value },
            keyboardOptions = KeyboardOptions(keyboardType = KeyboardType.Email),
        )
        Button(onClick = { toast(message = "Form: name is $name and email is $email")}) {
            Text("Submit")
        }
    }
}

// the toast extension function
fun Context.toast(message: String){
    Toast.makeText(this, message, Toast.LENGTH_SHORT).show()
}

It is common to run validations after submission to ensure data validity. For our case, we can run one to check that the name field is not empty and the email field matches the required regex.

We also need to show the required errors to notify the users of the issues. Let's go ahead and modify the Form composable as shown below:

@Composable
fun Screen(){
    var name by remember { mutableStateOf("") }
    var nameHasError by remember { mutableStateOf(false) }
    var nameLabel by remember { mutableStateOf("Enter your name") }

    var email by remember { mutableStateOf("") }
    var emailHasError by remember { mutableStateOf(false) }
    var emailLabel by remember { mutableStateOf("Enter your email address") }

    Column {
        TextField(
            value = name,
            isError = nameHasError,
            label = { Text(text = nameLabel) },
            modifier = Modifier.padding(10.dp),
            onValueChange = { value -> name = value }
        )
        TextField(
            value = email,
            isError = emailHasError,
            label = { Text(text = emailLabel) },
            modifier = Modifier.padding(10.dp),
            onValueChange = { value -> email = value },
            keyboardOptions = KeyboardOptions(keyboardType = KeyboardType.Email),
        )
        Button(onClick = {}) {
            Text("Submit")
        }
    }
}

In the button's onClick lambda, add the following:

when {
    name.isEmpty() -> {
        nameHasError = true
        nameLabel = "Name cannot be empty"
    }
    !Patterns.EMAIL_ADDRESS.matcher(email).matches() -> {
        emailHasError = true
        emailLabel = "Invalid email address"
    }
    else -> toast(message = "All fields are valid!")
}

That is a simple validation process for our fields. However, we have a lot of code for only two input fields. Imagine if we had ten fields, each with different validation processes. It would be worse if we had forms on different screens, as is common in most applications.

A workaround would be to create a form composable that handles form validation. We can create a state for the Form composable that handles all the fields we pass into it.

Let us go ahead and implement that.

Step 3: Validators

We can start with validators. They are all similar in terms of how they work. They return booleans based on whether the value passed meets specific criteria. Go ahead and create a sealed interface called Validator. All of our validators will be of this type.

private const val EMAIL_MESSAGE = "invalid email address"
private const val REQUIRED_MESSAGE = "this field is required"
private const val REGEX_MESSAGE = "value does not match the regex"

sealed interface Validator
open class Email(var message: String = EMAIL_MESSAGE): Validator
open class Required(var message: String = REQUIRED_MESSAGE): Validator
open class Regex(var message: String, var regex: String = REGEX_MESSAGE): Validator

Each of these will receive an optional message. This will allow us to pass in custom messages if we would like. Otherwise, we will show the default message.

We have a regex validator that receives the regex that we will use to compare with the form field value.

Step 4: Creating state for the fields

This step will create an internal state that will validate the individual input field. It will also be in charge of updating the field as the user types.

We will add more functions to help us manage the field, such as clearing the textfield, getting the value from the field, and showing/hiding errors.

class Field(val name: String, val label: String = "", val validators: List<Validator>){
    var text: String by mutableStateOf("")
    var lbl: String by mutableStateOf(label)
    var hasError: Boolean by mutableStateOf(false)

    fun clear(){ text = "" }

    private fun showError(error: String){
        hasError = true
        lbl = error
    }

    private fun hideError(){
        lbl = label
        hasError = false
    }

    @Composable
    fun Content(){
        TextField(
            value = text,
            isError = hasError,
            label = { Text(text = lbl) },
            modifier = Modifier.padding(10.dp),
            onValueChange = { value ->
                hideError()
                text = value
            }
        )
    }

    fun validate(): Boolean {
        return validators.map {
            when (it){
                is Email -> true
                is Required -> true
                is Regex -> true
            }
        }.all { it }
    }
}

We have created a class that receives three arguments. The name will be used to associate the field to the value later. The label will be used to set the field's label and error message. validators will list all the validators specified for that field.

The composable function Content will draw the field in the Form. The function validate will return a boolean to denote whether the field's value is valid or not. It will loop through the validators, checking if the value is correct based on the validator and returning true or false.

Once the user starts editing the form field, we clear the errors, so the field goes back to an invalidated state. Add the following implementations for the validators.

fun validate(): Boolean {
    return validators.map {
        when (it){
            is Email -> {
                if (!Patterns.EMAIL_ADDRESS.matcher(text).matches()){
                    showError(it.message)
                    return@map false
                }
                true
            }
            is Required -> {
                if (text.isEmpty()){
                    showError(it.message)
                    return@map  false
                }
                true
            }
            is Regex -> {
                if (!it.regex.toRegex().containsMatchIn(text)){
                    showError(it.message)
                    return@map false
                }
                true
            }
        }
    }.all { it }
}

The function should return the matching boolean from the validation process.

Step 5: Creating state for the form

This form state will draw our form fields and validate all the fields passed into it. Go ahead and create a class called FormState.

class FormState {
    var fields: List<Field> = listOf()
        set(value) { field = value }

    fun validate(): Boolean {
        var valid = true
        for (field in fields) if (!field.validate()) {
            valid = false
            break
        }
        return valid
    }

    fun getData(): Map<String, String> = fields.map { it.name to it.text }.toMap()
}

The validate function here also returns a boolean if any of the fields' validation does not pass. We break the loop at that point to save on resources. The getData function maps the form field names to the corresponding values for easy management on the receiving end.

Step 6: Putting it all together

Create a Form composable that receives two arguments.

  • state: This will be an instance of the FormState class we just created above.
  • fields: This will be a list of the form fields with their respective validators.
@Composable
fun Form(state: FormState, fields: List<Field>){
    state.fields = fields

    Column {
        fields.forEach {
            it.Content()
        }
    }
}

We set the state's fields using the setter method. We then display the form fields in a column by calling the field's Content composable function. With that, our form is complete 😁. To use it in your application, modify your UI as below.

@Composable
fun Screen(){
    val state by remember { mutableStateOf(FormState()) }

    Column {
        Form(
            state = state,
            fields = listOf(
                Field(name = "username", validators = listOf(Required())),
                Field(name = "email", validators = listOf(Required(), Email()))
            )
        )
        Button(onClick = { if (state.validate()) toast("Our form works!") }) {
            Text("Submit")
        }
    }
}

Once you run your application, everything should work as expected. To get the data, call the state.getData method, and you will receive a map of your form.

Conclusion

As you have seen, working with forms individually can lead to a lot of repetitive code. But with this approach, you encapsulate your form functions into different classes and composables, making your work easier and your code cleaner.

However, this approach has some drawbacks. For instance, it does not take care of the custom arrangement of the fields. You are restricted to a column, while you might want some fields on a row. It also does not allow modification of the fields' properties.

You can find the complete source code of the application on GitHub. If you have a solution to the issues facing this approach or a suggestion, feel free to open an issue.

Have fun with compose!


Peer Review Contributions by: Mercy Meave

Author
Linus Muema
Linus Muema is a Kotlin and Javascript developer. He has a great passion for writing code, trying out new programming paradigms, and is always ready to help other developers.
More Articles by Author
Related Articles
Cloudzilla is FREE for React and Node.js projects
No Credit Card Required

Cloudzilla is FREE for React and Node.js projects

Deploy GitHub projects across every major cloud in under 3 minutes. No credit card required.
Get Started for Free