I have the following requirements. A user needs to be able to schedule a recurring reminder in my app that will trigger a push notification at an exact time every day.
This is one of those questions that I hoped I would end up not submitting as similar questions were recommended while writing it. However several team members have spent hours and hours looking through the Android Developer Docs and Stackoverflow and we seem no closer to an answer, so here we are.
If I create a reminder and set it to trigger the notification 5 minutes in the future, the notification fires just fine.
I suspect this may be a problem caused by the changes to battery saving, wake lock, etc introduced in Android P as we did not have this problem prior to updating our target SDK to 28. That being said I am not positive that this is the only problem but I can consistently reproduce the issue on a Pixel and Pixel 3 XL running Android P.
An example where the notification is not firing occurs when for example a user sets the reminder to sometime in the middle of the night (presumably when the user is asleep and thus will not have used the phone for several hours). These reminders are not firing ever.
I am currently trying to accomplish this using the Alarm Manager.
This problem appears to be similar to another question that uses the Alarm Manager's setRepeating method which we have found does not work. We are instead using the Alarm Manager's setExactAndAllowWhileIdle method. We also tried this same implementation with the Alarm Managers setAlarmClock method which according to the Android documentation "will be allowed to trigger even if the system is in a low-power idle (a.k.a. doze) mode" however this was also unsuccessful.
I suspect that the reason this is not working is because setExactAndAllowWhileIdle will not fire when the phone is in doze mode, similar to the problems expressed in this question. This question recommends using Firebase JobDispatcher but as this is an internal notification I am required to fire the notification with or without network connectivity which seems to eliminate the Firebase JobDispatcher as an option. This question also indicated that once the phone exits doze mode the user gets the notification however we are never getting the notification, they seem to be lost for lack of a better term.
I have added the wake lock permission to my AndroidManifest.xml:
<uses-permission android:name="android.permission.WAKE_LOCK" />
Here is how my receiver is registered in the AndroidManifest.xml
<receiver android:name="com.myapp.receiver.AlarmReceiver">
</receiver>
Here is my current implementation:
Pending Intent for Handling Notifications
Intent i = new Intent(context, ScheduleAllReceiver.class);
PendingIntent scheduleAllPendingIntent = PendingIntent.getBroadcast(context, SCHEDULER_DAILY_ALL, i, PendingIntent.FLAG_UPDATE_CURRENT);
I subsequently call a method "createAlarm" as follows
createAlarm(context, scheduleAllPendingIntent, calendar.getTimeInMillis());
Creating the Alarms
public static void createAlarm(Context context, PendingIntent pendingIntent, long timeinMilli) {
AlarmManager alarmManager = (AlarmManager) context.getSystemService(Context.ALARM_SERVICE);
if(alarmManager != null) {
if (Build.VERSION.SDK_INT >= 23) {
alarmManager.setExactAndAllowWhileIdle(AlarmManager.RTC_WAKEUP, timeinMilli, pendingIntent);
} else {
alarmManager.setExact(AlarmManager.RTC_WAKEUP, timeinMilli, pendingIntent);
}
}
}
In order to delete : AlarmManager alarmManager = (AlarmManager) getSystemService(Context. ALARM_SERVICE); Intent myIntent = new Intent(getApplicationContext(), SessionReceiver. class); PendingIntent pendingIntent = PendingIntent.
This example demonstrates how do I implement alarm manager in android. Step 1 − Create a new project in Android Studio, go to File ⇒ New Project and fill all required details to create a new project. Step 2 − Add the following code to res/layout/activity_main. xml.
Android AlarmManager allows you to access system alarm. By the help of Android AlarmManager in android, you can schedule your application to run at a specific time in the future. It works whether your phone is running or not.
Adding an intent Flag FLAG_RECEIVER_FOREGROUND
https://developer.android.com/reference/android/content/Intent#FLAG_RECEIVER_FOREGROUND prior to calling the broadcast receiver should do the trick
Intent intent = new Intent(context, ScheduleAllReceiver.class);
intent.addFlags(Intent.FLAG_RECEIVER_FOREGROUND);
PendingIntent scheduleAllPendingIntent = PendingIntent.getBroadcast(context, SCHEDULER_DAILY_ALL, intent, PendingIntent.FLAG_UPDATE_CURRENT);
I have faced the similar problems. I tried with work manager but the result was same. When phone is locked and is in doze mode the event are not triggered.
But for triggering event at exact time you need to use alarm manager only. It should work even in doze mode.
Method to register alarm manger(use set exact time instead of repeating)
public static void registerAlarm(Context context){
final int FIVE_MINUTES_IN_MILLI = 300000;
final int THIRTY_SECOND_IN_MILLI = 30000;
long launchTime = System.currentTimeMillis() + FIVE_MINUTES_IN_MILLI;
AlarmManager am = (AlarmManager) context.getSystemService(Context.ALARM_SERVICE);
Intent i = new Intent(context, BroadcastAlarmManger.class);
PendingIntent pi = PendingIntent.getBroadcast(context, 0, i, 0);
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M)
am.setExactAndAllowWhileIdle(AlarmManager.RTC_WAKEUP, launchTime, pi);
else am.setExact(AlarmManager.RTC_WAKEUP, launchTime, pi);
Utility.printLog("timestamp "+launchTime);
}
Broadcast receiver for alarm
public class BroadcastAlarmManger extends BroadcastReceiver {
@Override
public void onReceive(Context context, Intent intent) {
//register alarm again
registerAlarm(context);
.
.
.
.
//do your stuff
}
}
Manifest declaration
<receiver
android:name="com.taxiemall.utility.BroadcastAlarmManger"
android:enabled="true"
android:exported="true"/>
Also you have to handle Reboot manually. After every reboot registered alarm are cleared. So register a broadcast receiver for Android reboot and register you alarm again inside it.
If you love us? You can donate to us via Paypal or buy me a coffee so we can maintain and grow! Thank you!
Donate Us With