John Brainard bio photo

John Brainard

Of course! I'm not synthetic.

LinkedIn Github

// Update: I’ve posted the source code for this series on Github.

We now have a pretty functional photo frame app for our Android device. The only thing left to do to make it useful is to make sure it doesn’t shut off on us. To do that, we’re going to add a BroadcastReceiver to listen to power connect/disconnect events so the app knows whether or not to disable screen timeouts. We’re also going to use that BroadcastReceiver to launch the app when the power cable is connected.

The PowerConnectionReceiver Class

Create a new Kotlin file in the io.github.johnbrainard.kotlinphotoframe package called PowerConnectionReceiver and enter the following code:

 1 class PowerConnectionReceiver constructor() : BroadcastReceiver() {
 2 	var disconnected: () -> Unit = {}
 3 
 4 	constructor(disconnected:()->Unit) : this() {
 5 		this.disconnected = disconnected
 6 	}
 7 
 8 	override fun onReceive(context: Context, intent: Intent) {
 9 		when (intent.getAction()) {
10 			Intent.ACTION_POWER_CONNECTED -> {
11 				val activityIntent = Intent(context, javaClass<MainActivity>())
12 				activityIntent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK)
13 				context.startActivity(activityIntent)
14 			}
15 
16 			Intent.ACTION_POWER_DISCONNECTED -> {
17 				disconnected?.invoke()
18 			}
19 		}
20 	}
21 }

Notice that this class has a primary, default constructor and an additional constructor. In a moment, we’ll be adding a receiver configuration to the AndroidManifest.xml file. The second constructor takes a lambda expression as a parameter that’s called when the power is disconnected. The MainActivity will take advantage of this to allow the screen to time out while the app is running.

Configuring the PowerConnectionReceiver

In your AndroidManifest.xml file, add the following receiver configuration inside the application element:

<receiver android:name=".PowerConnectionReceiver">
	<intent-filter>
		<action android:name="android.intent.action.ACTION_POWER_CONNECTED" />
	</intent-filter>
</receiver>

isPowerConnected Extension Property

One of my favorite features of Kotlin is its extension methods and properties. They allow you to do many things that would be difficult to do without out them. We’ll look at them in more detail in a future post, including decompiled code to see how they’re implemented. For now, we’re going to create a simple extension property on Android’s Context class that we can use to determine if the power is connected.

In Android Studio, add a new file to your io.github.johnbrainard.kotlinphotoframe package called utils.kt. You can call it whatever you like and put it in whatever package you want, but for the sake of this post, we’ll choose the given package and name.

 1 package io.github.johnbrainard.kotlinphotoframe
 2 
 3 import android.content.Context
 4 import android.content.Intent
 5 import android.content.IntentFilter
 6 import android.os.BatteryManager
 7 
 8 val Context.isPowerConnected:Boolean get() {
 9 	val intent = registerReceiver(null, IntentFilter(Intent.ACTION_BATTERY_CHANGED))
10 	val plugged = intent.getIntExtra(BatteryManager.EXTRA_PLUGGED, -1)
11 	return plugged == BatteryManager.BATTERY_PLUGGED_AC || plugged == BatteryManager.BATTERY_PLUGGED_USB
12 }

Disabling Screen Timeout

Now that we have everything set up in the AndroidManifest.xml file, the PowerConnectionReceiver class created and our convenience extension property for checking the power connection state, we’ll add the final touches we need to MainActivity.

First we’re going to declare a new property and set it to a new instance of PowerConnectionReceiver. This will be how we reenable the screen timeout when the power is disconnected.

private val powerReceiver = PowerConnectionReceiver() {
	getWindow().clearFlags(WindowManager.LayoutParams.FLAG_KEEP_SCREEN_ON)
}

Now we need to register that receiver so it can receive events and disable the screen timeout if the power is connected. This is where we get to use the extension property we just created. Add the following at the end of your onResume() method.

registerReceiver(powerReceiver, IntentFilter(Intent.ACTION_POWER_DISCONNECTED))

if (isPowerConnected)
	getWindow().addFlags(WindowManager.LayoutParams.FLAG_KEEP_SCREEN_ON)

And finally, we need to do some cleanup in onPause(). We don’t want to leave our PowerConnectionReceiver registered as a listener when the activity is no longer active. So, add the following line to the end of your onPause() method.

unregisterReceiver(powerReceiver)

Wrapping Up

At this point you should have a fully functional photo frame app that, with little effort, could be more useful. Some ideas to consider:

  • Add text view to display an image title
  • Modify the ImageSource to retrieve images from your Flickr account
  • Add support for multiple image sources and randomly select

You can check out the Source Code for the app and compare yours with mine.

I would love to hear your feedback about how this app can be improved and what ideas to implement in future posts.