I'm trying to present a notification which is handled by a special activity which is not in the normal application flow, and trying to get the back stack handling "correct," meaning:
- if a notification is handled while the application is running, the notification activity should appear on the current stack, back from the notification should leave us where we were in the application. Note that this may mean the application is made open.
- if a notification is handled while the application isn't running, the notification activity should appear on the main (initial) activity of the application.
So far, the code I'm using to present the notification is:
/**
* Show (or update) a notification for the current message set.
*
* @param showNotification true to use a high priority notification which will be immediately
* displayed (as opposed to just added to the status bar)
*/
private void createOrUpdateNotification(boolean showNotification) {
Message oldest = messages.get(0);
Message newest = messages.get(messages.size() - 1);
// Create the notification
NotificationCompat.Builder builder = new NotificationCompat.Builder(context)
// Set notification data and appearance
.setContentTitle(title)
.setContentText(newest.message)
.setSmallIcon(smallIcon)
.setWhen(newest.when)
.setColor(ContextCompat.getColor(context, R.color.primary_color))
// Set notification options
.setAutoCancel(true)
.setCategory(NotificationCompat.CATEGORY_MESSAGE)
.setPriority(showNotification ? NotificationCompat.PRIORITY_HIGH : NotificationCompat.PRIORITY_LOW)
.setDefaults(NotificationCompat.DEFAULT_VIBRATE)
.setOnlyAlertOnce(!showNotification);
// Set up the action if the first (oldest) message has an intent builder
if(oldest.intent != null) {
TaskStackBuilder stackBuilder = TaskStackBuilder.create(context.getApplicationContext());
stackBuilder.addNextIntent(oldest.intent);
builder.setContentIntent(stackBuilder.getPendingIntent(0, PendingIntent.FLAG_UPDATE_CURRENT));
}
NotificationManagerCompat.from(context).notify(notificationId, builder.build());
Log.i("notification created");
}
For clarification, Message.intent
is a single intent, configured to open the notification handling activity.
My problem is that if the application is currently running and open when the notification is opened, the application is closed and the notification presented on an empty stack and the application's back stack is cleared out.
My understanding is that the desired behavior should be automatic if the content intent is a pending intent containing a single intent, which is the case here.
What am I missing?
To expand a little bit on @leco's answer, my final solution had two parts.
The first is building the
Notification
as he recommended, using aPendingIntent
directly instead of trying to use aTaskStackBuilder
:This got me the correct behavior if the application was running, but not currently open (ie., up until the user hit back all the way out of the application)
To get the "correct" behavior at that point, I modified my
Notification
handling Activity by overridingfinish()
:The gyrations with
queryIntentActivities
etc are due to the fact that I'm actually developing a drop-in component for applications, so I don't actually know what their root/home/launch activity actually is. I tried a simpler approach of just building a HOME intent:but for some reason
startActivity
was throwing an exception:even though the
queryIntentActivities
approach demonstrates that there is an appropriateActivity
for theIntent
.