MAD Topic 2 : Further Jetpack Compose, including Material Design and Theming

We will look at:

Further introductory Jetpack Compose topics

Font size

First, some quick revision on a topic covered in OODD. How can we specify font size? We can do it using the fontSize parameter, e.g:

Text("Hello World!", fontSize=24.sp)
What does the font size 24.sp mean? This means 24 scalable pixels, which are related to the similar unit density-independent pixels or dp. Density-independent pixels are units which scale according to the device resolution resulting in a consistent appearance on different devices. Scalable pixels scale additionally according to the user's font size preferences. We discussed this topic in OODD; please see the OODD discussion for more details.

Modifiers

We can modify the appearance of UI elements with modifiers which allow you to control such things as padding, borders, etc. Modifiers are optional in some cases but compulsory in others. Modifiers are compulsory in the case of the Spacer, which is an element used to provide space between other elements. Here is the previous example with a Spacer:

package com.example.jetpackcompose3 

import androidx.activity.ComponentActivity
import androidx.activity.compose.setContent
import android.os.Bundle
import androidx.compose.material3.Text
import androidx.compose.runtime.Composable
import androidx.compose.foundation.layout.Column
import androidx.compose.ui.text.font.FontFamily
import androidx.compose.ui.text.font.FontStyle
import androidx.compose.ui.text.font.FontWeight
import androidx.compose.ui.graphics.Color
import androidx.compose.ui.unit.sp
import androidx.compose.ui.unit.dp
import androidx.compose.ui.Modifier
import androidx.compose.foundation.layout.Spacer
import androidx.compose.foundation.layout.height

class MainActivity : ComponentActivity() {

    override fun onCreate(savedInstanceState: Bundle?) {
        setContent {
            TwoTexts()
        }
    }

    @Composable
    fun TwoTexts() {
        Column {
            Text("Hello World!", color = Color.red, fontStyle=FontStyle.Italic, fontSize=24.sp, fontFamily=FontFamily.Serif)
            Spacer(Modifier.height(32.dp))
            Text("Welcome to Android Development", fontWeight=FontWeight.Bold, fontSize=18.sp)
        }
    }
}
This creates a spacer with a height of 32 density independent pixels (see above).

Other modifiers allow us to specify padding (the space between the border of a UI element and its content) or an element's border. For example, this surrounds a GreetingComponent component with a 2 dp wide blue border and with padding of 16dp between the border and the content. The diagram below illustrates the composable's border and padding.

Composable borders and padding

Note how Modifier contains many methods to modify different aspects of the element, and note how they can all be chained together.

GreetingComponent(
    Modifier.border(BorderStroke(2.dp, Color.Blue))
            .padding(16.dp)
)
Our GreetingComponent would now need to take a modifier as a parameter. This would then be passed onto the Column which sets the layout within the greeting component.
@Composable
fun GreetingComponent(mod: Modifier) {
    var name by remember { mutableStateOf("") }
    Column(mod) {
        ...
    }
}

Material Design

Material Design (see the website) is a published design philosophy which is adopted as the recommended standard in Android development. It revolves around the idea of screen surfaces being composed of material which has certain visual and behavioural properties, for example the ability to cast shadows when raised, and only UI elements at the top of a stack of elements being able to receive events. Material Design specifies a range of standard UI components, such as buttons, text fields and many others. Following the Material Design philosophy allows the development of clean-looking, usable and intuitive apps.

Material Design has a range of key aspects, with three particularly important components including:

Through colour, typography and shape, Material Design helps you design appealing and consistent UIs. How can it do this?

When designing a UI, you need to think about classifying your UI elements in terms of which should be particularly prominent to the user and which can be less prominent because they are less critical to the functionality of the app. This helps you break down the UI into different classes of component. The idea is that you style components of the same importance class similarly.

So for example prominent UI controls such as buttons should all have the same style (same colour and/or typography). Rather than using a random mix of styles in your app, with multiple different colours and fonts, you define colours and fonts for each class of UI element and apply them appropriately. So for colour for example, you create a palette of colours for your app and apply colours from this palette appropriately for different classes of UI element.

So in summary, by setting the properties of these colour, typography and shape classes you can ensure a consistent look and feel across similar elements, enhancing usability and aesthetics of the app.

More detail on colour roles

Material Design features a number of so-called colour roles. These are described here. Colour roles include, amongst others:

So to design the colour scheme for a Material Design app, you should pick appropriate colours for these colour roles, and ensure you pick the correct colour role for a given component.

More detail on typography

Just as for colour roles, Material Design offers a range of typography classifications. See here and here for more detail. The descriptions below are paraphrased from the second of these two articles.

Within each category, you can specify large, medium or small variants. The concept is again like colour. By defining certain fonts for each typography type, you are ensuring a consistent appearance across UI elements of the same type or class.

Using Material Design in Jetpack Compose

Having considered some of the absolute basics of Material Design, we will now see how we can use Material Design in a Jetpack Compose app.

Setting up Material Design in Jetpack Compose

Jetpack Compose integrates very well with Material Design and in fact supports the latest Material Design standard, version 3. Jetpack Compose allows you to define a theme, which is a collection of colours and typographies which will be applied to your Compose app. Android Studio pre-generates logic to create your theme for you.

Below shows the location of the pre-generated theme files for your project. They are within the ui.theme subpackage:
Compose theme source files

You can edit the theme files to set the theme for your app, in other words you can customise the different colour types (primary, secondary, tertiary, on-primary, etc) and typography. Color.kt contains colour definitions, Type.kt contains typography definitions and Theme.kt manages the theme as a whole. If you look inside the theme files, you can see that some definitions have already been setup for you.

Using a Theme

The Theme.kt file in the theme package provides a function you can use to apply your theme to your UI. This will be named <project name> plus the string Theme. For example if your project is called ComposeExample2, the theme function will be ComposeExample2Theme. Because this function takes a composable function (usually a lambda) as its argument, you can simply add composables inside it and apply it as follows:

setContent {
    ComposeExample2Theme {
        Text("Some text")
        TextField(...)
    }
}
All composables inside the theme will have the theme applied to them.

Setting theme colours and fonts

Material Design components will use default settings appropriate to that component. You can also, however. access the theme's colour and typography via the MaterialTheme class. This has a colorScheme property to access the colour roles, and a typography property to access the various font styles. For example:

You can set the color of composables directly to material colours, while the typography values give you TextStyle objects. These store multiple aspects of the text style, such as font weight and size. To set the text style of a composable, use the style property, e.g:

Text("Search results", style=MaterialTheme.typography.titleLarge)

Using the Material Theme Builder

It can be difficult to specify the precise colours and typography you want for your app manually in the theme code so that they all blend nicely. Instead, you can create your own custom theme using the graphical Material Theme Builder tool available online here. Using this tool you can pick your colour and font choices via a graphical interface, and export a Theme.kt file to use in your app.

Note that changes may have have to be made to the generated Theme.kt due to version incompatibilities, some of the parameters may not be compatible with the version of Compose running on your system. In general, any parameters showing as red (i.e. error) can be removed. Also you may need to adjust the package name so it matches your project: the generated files are likely to have the statement:

package com.example.compose

You need to adjust them so that the package matches yours. You can find the correct package name by adding ui.theme to your project package name e.g. com.example.myapplication.ui.theme. Your package name can be found at the top of your main activity source code file.

Finally if you use a custom font you need to add Google Fonts as a dependency.

Using a Surface

The most fundamental UI component of Material Design is the surface. The Surface is the UI element which your Material Design theme is applied to, so you should wrap all your other components in a Surface. Surfaces can be styled, for example you can set the shape to specify the extent of curvature at the corners. You can have a surface on top of another surface, and each surface can be styled differently.

To use a Surface, simply wrap it round the other UI elements, e.g.

import androidx.compose.material3.Surface
import your.package.ui.theme.ComposeExample2Theme // replace "your.package" and the project name!

setContent {
    ComposeExample2Theme {
        Surface(modifier=Modifier.fillMaxSize(),shape=MaterialTheme.Shapes.large, shadowElevation=1.dp) {
            Text("Some text")
            TextField(...)
       }
    }
}
This example shows:

Box Layouts

Imagine you want to display a large composable, such as a list or a map, above a row containing text fields and a button to set the location. If you think about it, this isn't so easy. If you specify the modifier for the map to be Modifier.fillMaxSize(), it will occupy the whole of the screen and there will be no room for the row. If you add the row first, it will appear on top of the screen.

What we need is another type of layout which allows us to specify exactly where each composable will be positioned, so we can position a composable at the top, the bottom, the left or the right. The Box composable allows us to do this.

In a Box, we can specify the alignment of the composables within the box. Here is an example:

Box(Modifier.fillMaxSize()) {
    Button(
        onClick = { }, modifier =
            Modifier.align(Alignment.BottomStart)
        ) { Text("Bottom Start") }

    Button(
        onClick = { }, modifier =
            Modifier.align(Alignment.BottomCenter)
        ) { Text("Bottom") }

    Button(
        onClick = { }, modifier =
            Modifier.align(Alignment.BottomEnd)
    ) { Text("Bottom End") }


    Button(
        onClick = { }, modifier =
            Modifier.align(Alignment.CenterStart)
    ) { Text("Centre Start") }

    Button(
        onClick = { }, modifier =
            Modifier.align(Alignment.Center)
    ) { Text("Centre") }

    Button(
        onClick = { }, modifier =
            Modifier.align(Alignment.CenterEnd)
    ) { Text("Centre End") }

    Button(
        onClick = { }, modifier =
            Modifier.align(Alignment.TopStart)
    ) { Text("Top Start") }

    Button(
        onClick = { }, modifier =
            Modifier.align(Alignment.TopCenter)
    ) { Text("Top") }

    Button(
        onClick = { }, modifier =
            Modifier.align(Alignment.TopEnd)
    ) { Text("Top End") }
}

This layout would look something like this:
Box Layout
Note how we use the align() method of our Modifier for each button, to specify where it is aligned within the box. The possible values include:

Implementing Lists of Data

This discussion is a repeat of what was already covered in OODD topic 8.

Creating Lists

You would implement a list by storing a list of data in state. You might think you could do something like this:

@Composable
fun ShoppingList() {
    var listState = remember { mutableStateOf(mutableListOf<String>()) }
    var currentItem = remember { mutableStateOf("") }

    TextField(value=currentItem.value, onValueChange = { currentItem.value=it } )
    Button(onClick = { listState.value.add(currentItem.value) } ) { Text("Add Item") }

    Column {
        listState.value.forEach {
            Text(it)
        }
       }
}
Note how we have a text field which allows you to enter a shopping list item, which is stored in the currentItem state variable. When the button is clicked, the current item is added to the list. As it's a mutable list, you might think this would work.

However it does not work. The reason is that composables are only re-rendered if the state variables change. Here, when we add a new item to the list, the list becomes one element longer but the actual list variable is the same variable, referring to the same location in memory.

To trigger a re-render when a new list item is added, we have to either:

This is an example of using a SnapshotStateList:
@Composable
fun ShoppingList() {
    // stateList is of type SnapshotStateList<String>
    var stateList = remember { mutableStateListOf<String>() }
    var currentItem = remember { mutableStateOf("") }

    TextField(value=currentItem.value, onValueChange = { currentItem.value=it } )
    Button(onClick = { 
        stateList.add(currentItem.value)
    } ) { Text("Add Item") }

    Column {
        stateList.forEach {
            Text(it)
        }
    }
}

Lazy Lists

One issue with a long list of items is that by default, a long list consisting of a series of Text items in a Column will not be memory efficient. Why? Let's say there are 100 items in the list, but only 10 are visible on the screen at any time. The items off the screen are still being rendered, even though they are invisible. This is clearly inefficient.

We can solve this problem through the use of lazy columns. The LazyColumn is designed to hold a series of items (i.e. a list) but is implemented with memory optimisation so that only for the items currently visible are rendered.

Creating a LazyColumn is quite straightforward, you place it in the appropriate place in your layout and then specify a lambda to control how it works. This lambda takes an object of type LazyListScope as its single parameter and this object includes an items method to specify a list of items. For example (ensure you include both imports; Alt+Enter will not be able to find items):

import androidx.compose.foundation.lazy.LazyColumn
import androidx.compose.foundation.lazy.items

@Composable
fun LazyListComposable(listItems: List<String>) {
    LazyColumn {
        items(listItems) { curItem -> Text(curItem) }
    }
}
Note how items() takes the list of items to render as its first parameter and another lambda as its last parameter. This lambda specifies how each item in the list of data should be transformed into a Compose element. So here, each item in the list is transformed to a Text element containing its details.

Exercise 1

  1. Modify your exercise last week (the feet to metres converter) so that the composable has a 4dp wide red border with a padding of 20dp.
  2. Create a new project and copy the code of your main activity into it. Apply your application's pre-generated theme to it. Change the border from red to your theme's primary colour. You will need to turn off dynamic theming - we will cover this in class.
  3. Add a Surface and place your feet-to-metres converter composable within the surface.
  4. Add a button to your feet-to-metres converter which resets the feet to 0.
  5. A bit harder! Try making the TextField occupy the whole width of the device. I have not shown you how to do this, but you might be able to figure it out. Look at the modifier used in the Surface example, above.
  6. Now try putting the TextField and Button (to clear the value) on the same line. The button should have 8dp padding. Hint: use a Row to contain both elements. It should look something like this:
    Messaging example
  7. Enhance the previous question so that the TextField and Button have a width proportion of 2:1, so that the text field has twice the width of the button. Hint: you can use Modifier.weight() to do this. Give the text field a weight of 2.0f (i.e. a Float) and the button a weight of 1.0f.

Exercise 2 - Lists and Box Layouts

  1. Implement a shopping list application. Store the list items inside a SnapshotStateList, as we did in OODD. There should be, top to bottom:
  2. Try using the Box layout to create a 3x3 grid of the numbers 1 to 9 using buttons: 1-3 for the first row, 4-6 for the second row, 7-9 for the third row.

Advanced exercise

Have a go at implementing a prototype of a simple messaging application using Jetpack Compose (lacking the communication-between-two-users aspect!). It should look something like this:
Messaging example
It should: