MAD Topic 1 : Introduction to Android Development

The Android mobile operating system is the leading smartphone and tablet operating system. From a minority operating system only running on one phone (the T-Mobile G1) a few years ago it has grown into one of the leaders in the smartphone operating system field, along with iOS (on the iPhone), and to some extent, Windows Mobile.

Why develop for Android?

There is one very important difference between Android and many of its competitors. It is an open source operating system, which means that you are free to modify it for your own ends. Even more crucially for app developers is that as a result of its open-source nature you can develop and distribute applications without restriction. This is in contrast to some of the other contemporary mobile development options in which the operating system vendor restricts distribution to a single channel owned by themselves, and "vets" software before making it available. Android has an official distribution channel (the Play Store), but this is more liberal with accepting apps than some of the other vendors. Also, there are alternative distribution channels for Android such as F-Droid

.

Android versions

At this point it is worth elaborating on the various versions of Android. As of last May the percentages of devices running different versions are as below (this is taken from Android Studio):
Percentages of devices running different versions
The most recent version deployed on actual mobile devices is 15, however, this is a very recent release with few devices running it. Many devices are running 9 (Pie), 10 (Q), 11 (R), 12 (S), 13 (Tiramisu) and 14 (Upside Down Cake). The majority of devices are running Marshmallow (Android 6 or API level 23) upwards.

Another concept that you need to understand is the API level. The Android API is the set of classes which are used to program Android apps with. The API level denotes revisions to the Android API, in a sequence of positive integers starting from 1. Thus, the numbers used for the API levels are not the same as those used for the Android versions, but each API level corresponds to a particular version. The idea is that each time Android itself is updated, the API is updated too. For example:

When developing an Android app, you have to specify the minimum API level on which your app will run. Thus an app with minimum API level 19, for example, will only run on 4.4+. As seen above, the vast majority of devices are running at least API level 19 (Android 4.4; KitKat), so if you specify API level 19 as a minimum you will be targeting the majority of devices.

Runtime Environment

With standard Kotlin or Java, you compile to bytecode which is then run using the Java Virtual Machine (JVM). Android is similar but rather than using the standard JVM, it uses its own virtual machine and corresponding bytecode format, producing executable files known as DEX files. So "regular" Java bytecode will not run on Android and Android apps will not run on a regular JVM.

Dalvik was the original virtual machine, which versions of Android up to 4.4 used. With Android 5.0, a new virtual machine (ART : Android Runtime) (see here) is used instead. See here for more details on Android virtual machines.

General nature of Android development

Android development is generally done in either Kotlin or Java, with Kotlin the preferred language. However, because the environment differs from a standard desktop PC, the actual libraries available differ somewhat from the standard Sun/Oracle Java Development Kit. As well as the Kotlin standard library, many standard Java features from packages such as java.io and java.util are available; however (as you might expect) the standard Java GUI libraries (designed for desktop applications) are not, and also the structure of an Android application is significantly different due to the different style of interaction with a mobile device compared to a desktop computer.

What do you need to start Android development?

To get started on Android development you ideally need the Android Studio IDE. Android Studio is the recommended IDE for Android development. It is based on JetBrains' IntelliJ IDEA .

It is also possible to develop apps purely using command-line (console) tools.

Android Studio provides an IDE 'wrapper' round the core Android development environment, which contains the following components:

You can access the Android SDK and the AVD Manager either through an IDE such as Android Studio or independently, via the command-line.

Getting started

The best way to explore the different components of the Android development environment is to get started with Android Studio. Launch Studio, you will see a screen like this:


Studio main screen

Creating a project

Select "Create New project". You will then see this screen. This allows you to create a new project. You will then see a screen like this:
Project details
Select "Empty Activity", as shown.


Studio: start a new project

To explain these one by one:

Once you click "Finish", your project will be created.

Explaining the layout of a project

The Android project will open, with a screen something like this:
Android Studio project
On the right-hand side is the main code editor, containing your Kotlin code. Note that some code is auto-generated: we will explain this later.

On the left-hand side, the project structure is shown. An Android project contains of a series of files and directories, each containing different things. To explain each:

Hello World!

We're now going to start - as always in software development - with a Hello World app. Delete all the pre-generated code and replace it with this:

package com.example.helloworldapp // leave the package line as it was

import androidx.activity.ComponentActivity
import androidx.activity.compose.setContent
import android.os.Bundle
import androidx.compose.material3.Text

class MainActivity : ComponentActivity() {

    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        setContent {
            Text("Hello World!")
        }
    }
}
What does this code mean?

Setting up an Android Virtual Device

If you do not have an actual Android device, you will need to set up an Android virtual device (AVD) before starting programming. This is an emulator which you can use to test your apps as you develop them. It resembles an actual phone, and the user interface looks just like a user interface on a real Android phone so you can test your apps fairly realistically. When creating an AVD you will be prompted for various properties of the emulator such as resolution.

To set up an AVD in Studio, select Tools-Device Manager. This will launch the Android Device Manager from within Android Studio, as shown below:

AVD manager before virtual devices have been set up

This screen shows that there is one AVD already set up, called Pixel API 34. On your system there probably won't be any yet. So Set up a new AVD as follows:

Running on an actual device

You can test your apps on an actual device, though you have to enable the developer settings. Full instructions on this are available from the Android site. On Linux and Mac OS X as no driver is required. On Windows, a driver for your device is required, however Nexus devices can use the Google USB driver which comes with the SDK. For other devices, you can download a driver for your device from the Android developer site. See the Android documentation on driver installation for more details. To summarise, you have to download and then install the driver. Even with the Google USB driver, included in the SDK, the installation step is necessary.

You should also be able to use a real Android device if you have one.

The Android SDK Manager

One of the most crucial components of the Android SDK is the SDK Manager. This piece of software, which can be run within Android Studio or standalone, allows you to download and install versions of the SDK for different versions of Android, along with other items such as documentation. So if a new version of the SDK is released, the SDK manager allows you to download that new version. The default Studio download only comes with the latest versions of the SDK, so if you want to target older devices, you need to download older versions.

Starting the SDK Manager

On Android Studio, select Tools-Android-SDK Manager. This will launch an intermediate screen listing installation options; however for more control it's recommended you then click on Launch Standalone SDK Manager which will give you the SDK Manager as it appears if you launch it on its own without Android Studio.

Installing SDK versions from the Android SDK manager

You might want to install older SDK versions than those included by default by Android Studio. For example, you might want to install Android 4.4 and 4.0.3 (API levels 19 and 15 respectively). To do this, you need to start the SDK manager as described above, and then select the platforms you want, e.g:
Android SDK manager with user about to install 4.0.3 and 4.4.2
You will then need to accept the licence and it will download the individual components of these versions of the SDK.

Jetpack Compose

Android uses the Jetpack Compose UI library. We have already looked at Compose Multiplatform in OODD; Compose Multiplatform is the desktop adaptation of Jetpack Compose, which was originally an Android-only library. Everything you've learnt with Compose Multiplatform will help you develop for Jetpack Compose, and furthermore, Jetpack Compose has a number of additional features compared to Compose Multiplatform which are available on Android.

Compose Revision

You have done Compose already in OODD but the notes below are provided for revision.

Composable Funcrions

In Jetpack Compose, we define each component as a composable function. Composable functions typically represent reusable groups of UI elements which work together, e.g. a login composable function could contain username and password fields and a button which logs the user in.

Example of a composable function

This extension of Hello World uses a simple composable function containing two text elements.

package com.example.jetpackcompose2 

import androidx.activity.ComponentActivity
import androidx.activity.compose.setContent
import android.os.Bundle
import androidx.compose.material3.Text
import androidx.compose.runtime.Composable

class MainActivity : ComponentActivity() {

    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        setContent {
            HelloComposable()
        }
    }

    @Composable
    fun HelloComposable() {
        Text("Hello World from the Composable!")
    }
}
Note how we create a function HelloComposable() labelled with the annotation @Composable. Annotations are meta-language instructions which are converted by a pre-processor into more complex Kotlin or Java code. The HelloComposable function defins a Text composable which displays a Hello World message, while the original setContent() now contains the HelloComposable.

A more complex composable function

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

class MainActivity : ComponentActivity() {

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

    @Composable
    fun TwoTexts() {
        Column {
            Text("Hello World!")
            Text("Welcome to Android Development")
        }
    }
}
This example creates a composable function with two UI elements: two Text elements arranged vertically. The vertical layout is defined with Column which arranges all the elements within it in a vertical column. There is also Row which arranges all elements within it horizontally.

Controlling the appearance of UI elements

We can control the appearance of a UI element by setting various parameters on the element. For example we can control colour, font style (normal or italic), font weight (normal or bold) and font family. Here is the previous example with the appearance of the Text elements styled:

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

class MainActivity : ComponentActivity() {

    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        setContent {
            TwoTextsStyled()
        }
    }

    @Composable
    fun TwoTextsStyled() {
        Column {
            Text("Hello World!", color = Color.Red, fontStyle=FontStyle.Italic, fontFamily=FontFamily.Serif)
            Text("Welcome to Android Development", fontWeight=FontWeight.Bold)
        }
    }
}
Hopefully much of this is fairly clear but a few points:

Note how each argument to Text is labelled with an identifier, e.g. fontStyle=.... This is an interesting feature of Kotlin which is also seen in Python; we do not have to pass in the arguments to a function in order if we label them. So Text() for example has color and fontStyle parameters and these can be specified in any order as long as they are labelled.

Passing in Parameters

We can pass parameters into our composables and use those parameters to control how they display, e.g this composable will display a given message a given number of times in a column by using a loop to include a Text the given number of times displayed in a given colour (black by default):

@Composable
fun MultiMessage(message: String, nTimes: Int, colour: Color = Color.Black) {
    Column {
        for(i in 1..nTimes) {
            Text(message, color=colour)
        }
    }
}

This could then be included in your App as follows:

@Composable
fun App() {
    MultiMessage("Hello Compose World!", 3, Color.Blue)
}

Events and State

Many UI elements in Compose Multiplatform come with an accompanying event handler, which runs when the user interacts with it. Examples include Button (which comes with a click handler) and TextField (which comes with an onValueChange event handler to handle the text in the text field changing). Event handlers are typically specified as lambda functions. We will now start building a GreetingBox which allows the user to enter their name and displays "Hello" plus their name.

@Composable
fun GreetingBox() {
    Column {
        TextField(value = "Enter your name: ", onValueChange={
            /* event handler */ 
        })
        Text("Greeting will go here")
    }
}
Note how the TextField takes two compulsory arguments, the default text to place inside the text field, and the onValueChange event handler which runs when the user enters text (a lambda function). Note that here we have used named parameters in the TextField, we do not need to do this (as the value and onValueChange event handler are the first two parameters) but have added them for clarity.

How can we actually display a greeting in the Text corresponding to the name that the user entered, though? Or, in general, how can we dynamically update a Compose UI? We use composable state to do this. In event handlers, we update state variables, and then we include that state in other UI elements. When a state variable updates, composables which use that state variable will automatically be redrawn. This is called recomposition. If you have worked with React in web development, you will recognise this pattern.

So how do we setup state? We have to declare a state variable to represent the state. State variables are variables of type MutableState, for example MutableState<String> for a String state variable, though you don't normally need to declare the type as Kotlin infers it from your initialisation. For example:

val nameState = remember { mutableStateOf("") }
The remember function creates a MutableState variable called nameState which is initialised within the provided lambda to a blank string "". The state will be remembered when the composable is recomposed (redrawn).

We can then insert the state variable within the UI where we like. To reference the actual value of the MutableState, we have to use its value property. For example:

import androidx.compose.runtime.remember // new import
import androidx.compose.runtime.mutableStateOf // new import
import androidx.compose.runtime.setValue // new import
import androidx.compose.runtime.getValue // new import

@Composable
fun GreetingBox() {
    val nameState = remember { mutableStateOf("") }
    Column {
        TextField(value = nameState.value, onValueChange= { 
            nameState.value = it 
        })
        Text("Hello ${nameState.value}!")
    }
}

Buttons

Buttons in Compose Multiplatform are quite easy to setup. Here is an example:

import androidx.compose.material3.Button // new import

@Composable
fun ButtonExample() {
    val clickedState = remember { mutableStateOf("Not Clicked!") }

    Column { 
        Button ( 
            onClick = { clickedState.value="Clicked!" }
        ){
            // Text as a child composable of the button
            Text("Click me!")
        }
        Text(clickedState.value, fontWeight = FontWeight.Bold)
    }
}
The Button function takes a number of parameters including an onClick event handler, for which we usually pass a lambda. Here, the lambda updates the state variable to "Clicked!" to indicate the user has clicked the button - this is shown in the Text at the end of the composable. After specifying the other parameters of Button we specify a lambda containing the text to display on the button as a Text component. In the same way as the Button and Text are child composables of the Column, this makes the Text a child composable of the button and will, as a result, make the text appear on the button.

Application resources

Android apps consist of Kotlin or Java code plus resources - additional data which the app needs to do its job. An example of a resource is an XML layout file, as described above. Resources can be found within res, as we saw above.

One example of a resource is a string resource. In Android development, to make it easier to translate apps into different languages, much of the text that we see within the user interface is defined in a string resource file so that we can easily translate an app to a different language simply by editing the string resource file. This can be found within the values folder within res, in the file strings.xml. If you look in the strings.xml file, you will see this line:

<string name="app_name">HelloWorldApp</string>
This defines a string called app_name, which specifies the application name.

Add a new line to strings.xml:

<string name="helloworld">Geia sas Kosmos!</string>
(Sorry if I have got the Greek wrong!).

Now try changing Hello World in the code example above to resources.getString(R.string.helloworld). This will read the appropriate string from the strings.xml file.

As well as strings, the res folder can contain other types of resource. These include layout files (we can define our layout in XML; we will look at this in a future week), application menus (which we will come onto a bit later on) and images. When you distribute an app, all the resources are packed into one file along with your actual code.

The auto-generated R.java file

You might be a bit puzzled as to the meaning of the R in the code you added above, i.e.

R.string.helloworld
What, actually, is this "R"? It's a pre-generated Java class which contains "hooks" into your XML resource files. The R.java file, not directly accessible from Studio but present in your project, looks something like this:
/* AUTO-GENERATED FILE.  DO NOT MODIFY.
 *
 * This class was automatically generated by the
 * aapt tool from the resource data it found.  It
 * should not be modified by hand.
 */

package com.example.nickw.helloworld;

public final class R {
    public static final class attr {
    }
    public static final class drawable {
        public static final int icon=0x7f020000;
    }
    public static final class layout {
        public static final int activity_main=0x7f030000;
    }
    public static final class string {
        public static final int app_name=0x7f040001;
        public static final int helloworld=0x7f040000;
    }
}
Notice that it contains several static variables. These are identifiers which your Kotlin or Java code can use to access the XML, for example R.layout.activity_main has the hex value 0x7f030000 which is a "handle" for the activity_main.xml resource file. Every time you add a resource to an Android app, your R.java will automatically be updated and you will be able to use static attributes of R in your Java code to access different resources. Never edit R.java directly by the way, the system will always do it for you!

Exercises

As this is the first week, in order to start gently, this exercise is revision on Compose (which you have already done in OODD) but in the context of an Android app rather than a desktop app.

  1. Try out "Hello World" and the "TwoTexts" and "TwoTextsStyled" examples, above.
  2. Write a complete working app including a composable to allow the user to enter a name via a TextField. The composable should greet the user by name in a Text, with a message such as Hello John!
  3. Extend the app to feature two text fields to enter first name and last name. The greeting should then display the user's full name in one TextField when a button is clicked.
  4. In the same app, write a composable to implement a counter. The composable should include a state variable to hold the counter's current value (initialise it to 0) plus a Text containing the counter's value and a button which, when pressed, increases the counter by one. Display the counter's value in red.
  5. In the same app, write a composable to convert feet to metres. One foot is 0.305 metres. The user should be able to enter a value in feet, and the result should be displayed in metres. Give your TextField a label parameter, this should be set equal to a lambda containing a Text element with the label "Enter feet", i.e.:
    TextField(feetState, onValueChange={...}, label={ Text("Enter feet:") }) ...