Getting Started With Intents in Android Using Kotlin
An intent is an Android element that facilitates navigation from one screen to another. They are also used to pass data between activities or across applications. <!--more--> This tutorial will be a step-by-step guide on creating an application that uses intents and understanding more concepts related to them.
Objectives
By the end of this tutorial, you will be able to understand:
- What exactly is an intent.
- Different types of intents.
- How you can construct a full intent.
- How to incorporate intents into your application.
- What malicious intent is, and how to cuff it.
- What intent resolution means in Android.
This tutorial is recommended for beginners and intermediate developers looking to build their knowledge of Android intents.
Prerequisites
- Install the latest Android Studio on your machine. Download here.
Note:
Choose your best option for Android depending on the operating system running on your machine. Make sure your machine has at least 8GB of RAM otherwise, while building a project, it will strain the memory and slow down your PC. Hence, the greater the memory, the faster the project is completed.
- Have basic knowledge of the Kotlin programming language.
- Knowledge of how to start an Android project from scratch. Click here to get started.
Table of contents
- Objectives
- Prerequisites
- Table of contents
- A brief walkthrough to intents
- Types of Intents
- Intent structure
- Examples of implicit and explicit intents
- Step 1: Creating a new Android Studio project
- Step 2: Working with the main_activity.xml file
- Step 3: Working with ActivityMain.kt file
- How to identify malicious intents
- Intent resolution
- Conclusion
Let's get started!
A brief walkthrough to intents
An intent is usually a command that helps in navigating through an application's activity (screen) in either of the following ways:
- Launching a new activity
- Initiating services
- Transmitting a broadcast receiver
- Passing information between two or more screens
Intents use methods such as;
startActivity()
- Used to start a new activity or to activate an existing one.startService()
- Services are initiated using this reserved method and send modified instructions to an already running service.sendBroadcast()
- Using this method, the broadcast receivers are systematically chosen by the intent object, and the intended information is sent to them.
Intents can also be defined as sequential messages that allow application components to ask other Android features for support. Similarly, two applications on the same device maximally utilize intent to share information. An action, for example, can initiate an external activity such as capturing an image using a camera or getting data from memory.
Types of Intents
Intents are broadly categorized into;
Implicit Intents - The element in the application is not specified via implicit intent. In this situation, intent offers information on the components accessible from the system that ought to be executed.
For example, a button will take you to the intended application when you click it. Suppose your device has more than one sharing application. In that case, the options box will appear and display all applications with sharing capabilities.
App choice is made depending on which mode the user is most comfortable with, among other personal considerations like the security of the data, e.t.c.
Explicit Intents - These intents connect elements of the same application. I.e., all the components reside in the same application. Using explicit purpose, you can pass data from one activity to another. For example, a button will direct you to a device file manager when you click it, and you can view the media inside the device memory.
Intent structure
A complete intent object should have a bundle that stores data that helps the application determine which activity to start first, and information that an element in the same application uses to conduct an operation in the correct procedure.
Key terminologies used in intent structuring
Element name
- This is an additional bit of information. Still, it's the most important because it helps make an intent explicit, indicating that the intent must only be delivered to the app element defined by the element name.
If intent doesn't have an element name, it would be implicit. In this case, the system decides which element should receive the intent first following the last available intent data.
Action
- This is a string that indicates the general action that should be carried out by intent. Actions are like variables, which means you can declare your own to be used by the application's intents.
Examples of actions used when building intents include:
private const Val ACTION_EDIT = "com.davis.action.EDIT"
The code edits information only when the identifiers to be edited are declared.
Be sure to include the package name of your application before any custom actions you write, e.g.,
com.davis
is a package name.
Data
- It indicates the intent data's explicit type. In most cases, the type is deduced from the data itself.
You can disable that processing and force the explicit type by setting that attribute. The URI is the most crucial element of data because it holds the data's whereabouts to be processed by the intent of a particular activity.
Category
- It is descriptive information that states the elements appropriate to steer the intent.
Example of categories in intents:
CATEGORY_EMBED
- Functioning within a primary activity container is possible.
CATEGORY_DESK_DOCK
- Whenever the device is connected to a car dock, an activity is launched.
Extras
- It refers to collecting any extra data. It is used to supply the component with more information.
For instance, if we have an action to send e-mail messages, we can enter additional data to provide a subject, body, and other information relating to the message being sent:
private const Val EXTRA_EMAIL = "com.davis.EXTRA_EMAIL"
The code above is a String
containing the different e-mail addresses to which the message should be sent.
Flags
- In the Intent class, flags are used as an identifier for an intent.
Depending on how an activity is flagged, the Android OS may know how to begin and how to proceed with that activity.
Examples of implicit and explicit intents
We will implement the two types of intent in a simple application.
Step 1: Creating a new Android Studio project
Launch the Android Studio IDE and create a new empty activity project.
Step 2: Working with the main_activity.xml file
Add the following code to the main_activity.xml
file:
<?xml version="1.0" encoding="utf-8"?>
<androidx.constraintlayout.widget.ConstraintLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
tools:context=".MainActivity">
<ImageView
android:id="@+id/imageView"
android:layout_width="375dp"
android:layout_height="295dp"
android:layout_marginStart="10dp"
android:layout_marginTop="16dp"
android:layout_marginBottom="20dp"
android:src="@drawable/car"
app:layout_constraintBottom_toTopOf="@id/rl1"
app:layout_constraintLeft_toLeftOf="parent"
app:layout_constraintRight_toRightOf="parent"
app:layout_constraintStart_toStartOf="parent"
app:layout_constraintTop_toTopOf="parent" />
<RelativeLayout
android:id="@+id/rl1"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_marginLeft="10dp"
android:layout_marginRight="10dp"
android:layout_marginBottom="20dp"
app:layout_constraintBottom_toTopOf="@id/rl2"
app:layout_constraintLeft_toLeftOf="parent"
app:layout_constraintRight_toRightOf="parent"
tools:layout_editor_absoluteX="17dp">
<androidx.cardview.widget.CardView
android:id="@+id/cardViewY"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_marginLeft="10dp"
android:layout_marginTop="8dp"
android:layout_marginBottom="8dp"
app:cardBackgroundColor="@color/teal_200"
app:cardCornerRadius="16dp">
<LinearLayout
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:orientation="vertical">
<TextView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_margin="2dp"
android:text="@string/explicit_intent"
android:textSize="15sp"
android:textColor="@color/black" />
<LinearLayout
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_gravity="center_horizontal"
android:orientation="horizontal">
<Button
android:id="@+id/btnGallery"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_marginHorizontal="10dp"
android:text="Gallery" />
<Button
android:id="@+id/btnCamera"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_marginLeft="30dp"
android:text="Camera" />
</LinearLayout>
</LinearLayout>
</androidx.cardview.widget.CardView>
</RelativeLayout>
<RelativeLayout
android:id="@+id/rl2"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_marginLeft="10dp"
android:layout_marginRight="10dp"
android:layout_marginBottom="10dp"
android:orientation="horizontal"
app:layout_constraintBottom_toBottomOf="parent"
app:layout_constraintLeft_toLeftOf="parent"
app:layout_constraintRight_toRightOf="parent"
tools:layout_editor_absoluteX="17dp">
<TextView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_alignParentLeft="true"
android:layout_centerHorizontal="true"
android:layout_centerVertical="true"
android:layout_marginEnd="49dp"
android:layout_toStartOf="@+id/btnShare"
android:text="@string/implicit_intent"
android:textColor="@color/black"
android:textSize="15sp"/>
<Button
android:id="@+id/btnShare"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_centerHorizontal="true"
android:text="share" />
</RelativeLayout>
</androidx.constraintlayout.widget.ConstraintLayout>
Note: You may opt to use vector assets instead of buttons by adding them to the drawable resource file. Then add an image view in the XML layout and reference the image using the
src
method.
Your design should look similar to the one shown below:
Step 3: Working with ActivityMain.kt file
Before we work with the main_activity.kt
, we will add additional code to our app files to ensure that it fully meets our objective.
In the Grandle Scripts
-> build.grandle
add the following code:
buildFeatures{ding
// enable viewBinding
viewBinding true
}
Remember to sync the project.
In the Android manifests folder*
-> double click AndroidManifest.xml
and add the following permissions to be able to access the device media and the camera application:
<uses-permission android:name=
"android. permission.READ_EXTERNAL_STORAGE"/>
<uses-permission android:name="android.permission.CAMERA"/>
To make your application functional on click events, add the following code to the MainActivity.kt
:
class MainActivity : AppCompatActivity() {
private val my_request_code: Int = 0
private lateinit var imageView: ImageView
var binding: ActivityMainBinding? = null
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
binding = ActivityMainBinding.inflate(layoutInflater)
setContentView(binding!!.root)
binding!!.btnGallery.setOnClickListener {
Intent(Intent.ACTION_GET_CONTENT).also {
it.type = "image/*"
startActivityForResult(it, 0)
}
}
// Access the camera
binding!!.btnCamera.setOnClickListener {
val snapPhotoIntent = Intent(MediaStore.ACTION_IMAGE_CAPTURE)
if (snapPhotoIntent.resolveActivity(this.packageManager) != null) {
startActivityForResult(snapPhotoIntent, my_request_code)
} else {
Toast.makeText(this, "Unable to execute Camera", Toast.LENGTH_SHORT).show()
}
}
// Share plain text via intent
binding!!.btnShare.setOnClickListener {
val sendIntent: Intent = Intent().apply {
action = Intent.ACTION_SEND
putExtra(Intent.EXTRA_TEXT, "I am sharing ->")
type = "text/plain"
}
val shareIntent = Intent.createChooser(sendIntent, "Share with?")
startActivity(shareIntent)
}
}
override fun onActivityResult(requestCode: Int, resultCode: Int, data: Intent?) {
super.onActivityResult(requestCode, resultCode, data)
if (resultCode == RESULT_OK && requestCode == 0) {
val bitmap = data?.extras?.get("data") as Bitmap
val uri = data.data
imageView.setImageURI(uri)
imageView.setImageBitmap(bitmap)
}
}
}
Click here to access the full code.
How to identify malicious intents
Your app can use intents to traverse components or conduct an action on behalf of another app.
Illegal intents are realized when the application's extras of a provided intent are unparcelled by an app to create a nested intent and StrictMode
violation occurs.
In Kotlin, the following code is a programmatic way to detect an unsafe intent:
fun onCreate() {
StrictMode.setVmPolicy(StrictMode.VmPolicy.Builder()
.detectUnsafeIntentLaunch()
.penaltyDialogue //Whenever a fault happens, a pop up with violation details appears.
.penaltyLog()
.penaltyDeath() //Immediately terminate the current application when the violation condition is triggered.
.build())
}
Note: In newer versions of Android i.e., version 12, when using the
detectAll()
method in declaringVmPolicy
,detectUnsafeIntentLaunch
is involuntary invoked.
Intent resolution
The act of searching for relevant application components for your intents is known as intent resolution.
Intent resolutions are primarily used in implicit intent because they do not offer the component name, which aids a system in looking for an application component name that can conduct the activities.
The system performs the following steps to obtain the component name:
- Action Test: Here, the system ascertains whether or not the intended action and the intents are in the intent filter match. The test is successful if they are identical. Otherwise, the test fails.
- Category test: Here, the system compares the category name to that specified in the intent filter. This test is passed if both are identical, else it fails.
- Data Test: Here, the system compares the data in the Intent MIME with information in the intent filter. This check succeeds if the two are identical and vice versa.
Conclusion
In this tutorial, we have discussed intents, the existing types of intents, and their examples. We have also learned how and why intents are essential due to their numerous uses in navigating between activities and applications.
Peer Review Contributions by: Eric Gacoki