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.
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
.
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):
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:
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.
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.
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.
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:
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:
Select "Empty Activity", as shown.
To explain these one by one:
build.gradle
(the equivalent of pom.xml
from Maven) use? Leave as Kotlin DSL. We will discuss this more fully in the near future.Once you click "Finish", your project will be created.
The Android project will open, with a screen something like this:
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:
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?
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:
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:
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.
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.
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.
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:
You will then need to accept the licence and it will download the individual components of these versions of the SDK.
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.
You have done Compose already in OODD but the notes below are provided for revision.
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.
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
.
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.
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.
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) }
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}!") } }
value
of the TextField
to the state variable so that the text field is always in sync with the current value of the state variable.onValueChange
event handler lambda for the TextField
takes the new text within the text field as its one and only parameter (it
) so, in the event handler, we update the state variable nameState
to the current text in the text field. Then, in the Text
element, we include the state variable in our greeting, so the greeting will always sync with what the user entered in the text field.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.
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.
You might be a bit puzzled as to the meaning of the R in the code you added above, i.e.
R.string.helloworldWhat, 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!
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.
TextField
. The composable should greet the user by name in a Text
, with a message such as Hello John!Text
containing the counter's value and a button which, when pressed, increases the counter by one. Display the counter's value in red.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:") }) ...
String
state variable and use toDoubleOrNull()
(see below) when you need to do the conversion. This will be easier than storing the feet as a Double
state variable. toDoubleOrNull()
method of String
will return a null
if the user enters something which is not a number (e.g. letters). You can make use of this together with the Elvis operator to set the value to 0 if the user enters a non-numeric value for the feet. You can also use it to display an error if the user enters a non-numeric value - have a go at this if you finish everything else.