Im making an Alarm app, and using AlarmManager to set alarms. Im saving every alarm using Room after running the setAlarm on the AlarmManager, so I can later restore them if the phone gets turned off and o .
Im running a BroadcastReceiver after the device gets booted using the guide from Android Developer site: https://developer.android.com/training/scheduling/alarms#boot
And my idea is to get the alarms from Room on the onReceive method But Room uses a suspend fun to get the alarms, but I cant run it on the onReceive since BroadcastReceiver doesnt have a lifecycle
How could I achieve a similar result?
This section in the BroadcastReceiver documentation gives an example of how to do this.
You could clean it up a bit with an extension function:
fun BroadcastReceiver.goAsync(
context: CoroutineContext = EmptyCoroutineContext,
block: suspend CoroutineScope.(BroadcastReceiver.PendingResult) -> Unit
) {
val pendingResult = goAsync()
@OptIn(DelicateCoroutinesApi::class) // Must run globally; there's no teardown callback.
GlobalScope.launch(context) {
try {
block(pendingResult)
} finally {
pendingResult.finish()
}
}
}
Then in your receiver, you can use it like below. The code in the goAsync
block is a coroutine. Remember that you should not use Dispatchers.Main
in this coroutine and it must complete within 10 seconds.
override fun onReceive(context: Context, intent: Intent) = goAsync { pendingResult ->
val repo = MyRepository.getInstance(context)
val alarms = repo.getAlarms() // a suspend function
val result = pendingResult.resultCode
// do stuff
}