Network Connectivity in Android — Kotlin

Rohitraj Khorwal
3 min readFeb 19, 2025

--

Networking is a crucial part of many Android apps. In this post, we’ll look at a set of Kotlin extension functions that make it easy to:

  • Check if a network connection exists
  • Determine the type of network (WiFi, Cellular, etc.)
  • Register and unregister network callbacks

1. Check for an Active Network

The isNetworkConnected() function quickly tells you whether the device is connected to the internet. It handles both modern (Android M and above) and legacy approaches.

When to use:
Before making any network requests, call this function to ensure your app doesn’t try to connect when there’s no available network.

/**
* Extension function to check if the device has an active network connection
* @return Boolean indicating if network is available
*/
@SuppressLint("ObsoleteSdkInt")
fun Context.isNetworkConnected(): Boolean {
val connectivityManager = getSystemService(Context.CONNECTIVITY_SERVICE) as? ConnectivityManager ?: return false

return when {
Build.VERSION.SDK_INT >= Build.VERSION_CODES.M -> {
connectivityManager.getNetworkCapabilities(connectivityManager.activeNetwork)?.let { capabilities ->
capabilities.hasCapability(NetworkCapabilities.NET_CAPABILITY_INTERNET) &&
capabilities.hasCapability(NetworkCapabilities.NET_CAPABILITY_VALIDATED) &&
(capabilities.hasTransport(NetworkCapabilities.TRANSPORT_WIFI) ||
capabilities.hasTransport(NetworkCapabilities.TRANSPORT_CELLULAR) ||
capabilities.hasTransport(NetworkCapabilities.TRANSPORT_ETHERNET) ||
capabilities.hasTransport(NetworkCapabilities.TRANSPORT_BLUETOOTH) ||
capabilities.hasTransport(NetworkCapabilities.TRANSPORT_VPN))
} ?: false
}
else -> {
@Suppress("DEPRECATION")
connectivityManager.activeNetworkInfo?.isConnectedOrConnecting == true
}
}
}

2. Identify the Network Type

The getNetworkType() function returns a NetworkType enum value (like WIFI, CELLULAR, or ETHERNET). Knowing the network type can help you optimize your app’s behavior.

When to use:
If your app needs to adjust behavior based on the connection type (for example, limiting heavy downloads on cellular networks), this function is your go-to.

/**
* Extension function to check network connection type
* @return NetworkType enum indicating connection type
*/
@SuppressLint("ObsoleteSdkInt")
fun Context.getNetworkType(): NetworkType {
val connectivityManager = getSystemService(Context.CONNECTIVITY_SERVICE) as? ConnectivityManager ?: return NetworkType.NONE

if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
connectivityManager.getNetworkCapabilities(connectivityManager.activeNetwork)?.let { capabilities ->
return when {
capabilities.hasTransport(NetworkCapabilities.TRANSPORT_WIFI) -> NetworkType.WIFI
capabilities.hasTransport(NetworkCapabilities.TRANSPORT_CELLULAR) -> NetworkType.CELLULAR
capabilities.hasTransport(NetworkCapabilities.TRANSPORT_ETHERNET) -> NetworkType.ETHERNET
capabilities.hasTransport(NetworkCapabilities.TRANSPORT_VPN) -> NetworkType.VPN
capabilities.hasTransport(NetworkCapabilities.TRANSPORT_BLUETOOTH) -> NetworkType.BLUETOOTH
else -> NetworkType.NONE
}
}
} else {
@Suppress("DEPRECATION")
when (connectivityManager.activeNetworkInfo?.type) {
ConnectivityManager.TYPE_WIFI -> return NetworkType.WIFI
ConnectivityManager.TYPE_MOBILE -> return NetworkType.CELLULAR
ConnectivityManager.TYPE_ETHERNET -> return NetworkType.ETHERNET
ConnectivityManager.TYPE_BLUETOOTH -> return NetworkType.BLUETOOTH
ConnectivityManager.TYPE_VPN -> return NetworkType.VPN
}
}

return NetworkType.NONE
}

/**
* Enum representing different network connection types
*/
enum class NetworkType {
WIFI, CELLULAR, ETHERNET, BLUETOOTH, VPN, NONE
}

3. Register and Unregister Network Callbacks

The registerNetworkCallback() function lets you react to network changes in real time. When the network becomes available or is lost, the corresponding callback is triggered. Don’t forget to unregister your callback with unregisterNetworkCallback() to avoid memory leaks.

When to use:
Use these functions when you need your app to be responsive to changes in connectivity — for example, updating the UI or pausing/resuming network operations as connectivity changes.

/**
* Extension function to register for network callbacks
* @param onAvailable callback when network becomes available
* @param onLost callback when network is lost
* @return NetworkCallback instance that can be used to unregister
*/
@SuppressLint("ObsoleteSdkInt")
@RequiresApi(Build.VERSION_CODES.LOLLIPOP)
fun Context.registerNetworkCallback(
onAvailable: () -> Unit,
onLost: () -> Unit
): ConnectivityManager.NetworkCallback {
val connectivityManager = getSystemService(Context.CONNECTIVITY_SERVICE) as ConnectivityManager

val networkCallback = object : ConnectivityManager.NetworkCallback() {
override fun onAvailable(network: Network) {
onAvailable()
}

override fun onLost(network: Network) {
onLost()
}
}

if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
connectivityManager.registerDefaultNetworkCallback(networkCallback)
} else {
val request = NetworkRequest.Builder()
.addCapability(NetworkCapabilities.NET_CAPABILITY_INTERNET)
.build()
connectivityManager.registerNetworkCallback(request, networkCallback)
}

return networkCallback
}

/**
* Extension function to unregister network callback
* @param callback The NetworkCallback to unregister
*/
@SuppressLint("ObsoleteSdkInt")
@RequiresApi(Build.VERSION_CODES.LOLLIPOP)
fun Context.unregisterNetworkCallback(callback: ConnectivityManager.NetworkCallback) {
val connectivityManager = getSystemService(Context.CONNECTIVITY_SERVICE) as ConnectivityManager
connectivityManager.unregisterNetworkCallback(callback)
}

Happy coding!

Free

Distraction-free reading. No ads.

Organize your knowledge with lists and highlights.

Tell your story. Find your audience.

Membership

Read member-only stories

Support writers you read most

Earn money for your writing

Listen to audio narrations

Read offline with the Medium app

--

--

No responses yet

Write a response