138

When Apple's official iOS Messages app is open and in the foreground, new messages from other contacts trigger a stock, native iOS notification alert banner. See image below.

Is this possible in 3rd party apps on the App Store? Local and/or Push Notifications for your app while your app is open and in the foreground?

When testing my app, notifications are received but no iOS alert UI is shown.

But this behavior is seen in Apple's official Messages app:

Messages is open and in the foreground. Still shows a notification alert.

The Local and Remote Notification Programming Guide says:

When the operating system delivers a local notification or remote notification and the target app is not running in the foreground, it can present the notification to the user through an alert, icon badge number, or sound.

If the app is running in the foreground when the notification is delivered, the app delegate receives a local or remote notification.

So yes, we can receive the notification data while in the foreground. But I see no way to present the native iOS notification alert UI.

-(void)application:(UIApplication *)application didReceiveRemoteNotification:(NSDictionary *)userInfo 
{
    // I know we still receive the notification `userInfo` payload in the foreground.
    // This question is about displaying the stock iOS notification alert UI.

    // Yes, one *could* use a 3rd party toast alert framework. 
    [self use3rdPartyToastAlertFrameworkFromGithub]
}

Is Messages then using a private API to display the alert while in the foreground?

For the purpose of this question, please do not suggest any 3rd party "toast" pop-up alerts on github or etc. I'm only interested if this can be done using the stock, native iOS Local or Push Notification alerts UI while your application is open and in the foreground.

0

11 Answers 11

181

iOS 10 adds the UNUserNotificationCenterDelegate protocol for handling notifications while your app is in the foreground.

The UNUserNotificationCenterDelegate protocol defines methods for receiving notifications and for handling actions. When your app is in the foreground, arriving notifications are delivered to your delegate object instead of displayed automatically using the system interfaces.

Swift:

optional func userNotificationCenter(_ center: UNUserNotificationCenter, 
                     willPresent notification: UNNotification, 
      withCompletionHandler completionHandler: (UNNotificationPresentationOptions) -> Void)

Objective-C:

- (void)userNotificationCenter:(UNUserNotificationCenter *)center 
       willPresentNotification:(UNNotification *)notification 
         withCompletionHandler:(void (^)(UNNotificationPresentationOptions options))completionHandler;

The UNNotificationPresentationOptions flags allow you to specify UNNotificationPresentationOptionAlert to display an alert using the text provided by the notification.

This is key as it allows you to display the alert while your app is open and in the foreground, which is new for iOS 10.


Sample code:

class AppDelegate: UIResponder, UIApplicationDelegate {
    
    func application(_ application: UIApplication, didFinishLaunchingWithOptions launchOptions: [UIApplication.LaunchOptionsKey: Any]?) -> Bool {
        
        // Set UNUserNotificationCenterDelegate
        UNUserNotificationCenter.current().delegate = self
        
        return true
    }
    
}

// Conform to UNUserNotificationCenterDelegate
extension AppDelegate: UNUserNotificationCenterDelegate {
    
    func userNotificationCenter(_ center: UNUserNotificationCenter,
           willPresent notification: UNNotification,
           withCompletionHandler completionHandler: @escaping (UNNotificationPresentationOptions) -> Void)
    {
        completionHandler(.alert)
    }
    
}
16
  • 4
    Could you provide a working code sample where an incoming foreground notification is displayed by the system as if the application where background?
    – azmeuk
    Oct 18, 2016 at 13:59
  • 1
    @azmeuk Check my answer...
    – chengsam
    Nov 23, 2016 at 4:32
  • 9
    Don't forget to add UNUserNotificationCenter.current().delegate = self in the application(_:willFinishLaunchingWithOptions:) or application(_:didFinishLaunchingWithOptions:) method Dec 12, 2017 at 14:13
  • 4
    Also make sure that you do not have your device in Do Not Disturb mode otherwise the notifications will be visible in the Notifications Center, but would not be presented in the foreground over your application
    – Jadent
    May 10, 2018 at 18:45
  • 2
    my 2 cents: don't miss: import UserNotifications. !
    – ingconti
    Jun 25, 2019 at 13:46
103

For displaying banner message while app is in foreground, use the following method.

iOS 10+, Swift 3+:

func userNotificationCenter(_ center: UNUserNotificationCenter, willPresent notification: UNNotification, withCompletionHandler completionHandler: @escaping (UNNotificationPresentationOptions) -> Void) {
    completionHandler([.alert, .badge, .sound])
}
13
  • 1
    Wow, everywhere I read people said this wasn't possible. I'm so glad I found this answer, it works exactly as I expected! When the app is running, push notifications now show exactly like they would if the app were in the background. Thank you! Dec 2, 2016 at 2:44
  • 1
    Where this snippet goes? In the appDelegate, or somewhere else?
    – Yoav R.
    Dec 26, 2016 at 17:41
  • @YoavR. AppDelegate
    – chengsam
    Dec 28, 2016 at 2:51
  • 4
    Chengsam you might want to correct this: It doesn't have to be put in the AppDelegate. It has to be put for whatever object that conforms to UNUserNotificationCenterDelegate. Having that said most developers make their AppDelegate conform to UNUserNotificationCenterDelegate. @YoavR.
    – mfaani
    May 29, 2018 at 13:23
  • 1
    Can this be conditional? I mean suppose I'm showing chat notifications. Can I not show the notifications if the chat screen is open but show it otherwise?
    – iOSer
    Apr 2, 2020 at 9:29
16

EDIT:

Foreground alerts are now possible in iOS 10! Please see this answer.

For iOS 9 and below:

It does not seem to be possible to show the stock iOS notification alert when your app is open and in the foreground. Messages.app must be using a private API.

The system does not display any alerts, badge the app’s icon, or play any sounds when the app is already frontmost. - UILocalNotification docs

The UIApplicationDelegate methods will still be called, allowing your app to respond to the local or remote notification:

application:didReceiveLocalNotification:
application:didReceiveRemoteNotification:

However, the stock native iOS notification alert banner UI will not be shown as it is in Apple's Messages.app, which must be using a Private API.

The best you can do is is roll your own alert banner or use an existing framework:

-(void)application:(UIApplication *)application didReceiveRemoteNotification:(NSDictionary *)userInfo 
{    
    // Use a 3rd party toast alert framework to display a banner 
    [self toastAlertFromGithub]
}

I have opened a radar for this behavior here: rdar://22313177

3
  • 3
    How does whatsapp and other famous apps do that, then?
    – Machado
    Apr 19, 2016 at 16:53
  • @tardoandre have a screenshot? I assume it's by mimicking the stock iOS alert in their own views.
    – pkamb
    Apr 19, 2016 at 17:03
  • 3
    You should include link for toastAlertFromGithub if you are suggesting this third party library to use! Dec 19, 2016 at 11:51
14

To show notifications while the App is open, we need to handle it manually. So what I am doing below is to handle the notification once received.

Add all below in AppDelegate.m

  1. Handle call for notify
  2. Create a view, add AppIcon, Notification message and show it as an animation
  3. Add Touch recogniser to remove if touched or remove in 5 seconds with animation.

Let me know if this is an ok solution. Worked well for me but am unsure if this is the right way.

- (void)application:(UIApplication *)applicationdidReceiveRemoteNotification:(NSDictionary *)userInfo fetchCompletionHandler:(void (^)(UIBackgroundFetchResult))completionHandler {


NSString *notifMessage = [[userInfo objectForKey:@"aps"] objectForKey:@"alert"];

//Define notifView as UIView in the header file
[_notifView removeFromSuperview]; //If already existing

_notifView = [[UIView alloc] initWithFrame:CGRectMake(0, -70, self.window.frame.size.width, 80)];
[_notifView setBackgroundColor:[UIColor blackColor]];

UIImageView *imageView = [[UIImageView alloc] initWithFrame:CGRectMake(10,15,30,30)];
imageView.image = [UIImage imageNamed:@"AppLogo.png"];

UILabel *myLabel = [[UILabel alloc] initWithFrame:CGRectMake(60, 15, self.window.frame.size.width - 100 , 30)];
myLabel.font = [UIFont fontWithName:@"Helvetica" size:10.0];
myLabel.text = notifMessage;

[myLabel setTextColor:[UIColor whiteColor]];
[myLabel setNumberOfLines:0];

[_notifView setAlpha:0.95];

//The Icon
[_notifView addSubview:imageView];

//The Text
[_notifView addSubview:myLabel];

//The View
[self.window addSubview:_notifView];

UITapGestureRecognizer *tapToDismissNotif = [[UITapGestureRecognizer alloc] initWithTarget:self
                                                                                    action:@selector(dismissNotifFromScreen)];
tapToDismissNotif.numberOfTapsRequired = 1;
tapToDismissNotif.numberOfTouchesRequired = 1;

[_notifView addGestureRecognizer:tapToDismissNotif];


[UIView animateWithDuration:1.0 delay:.1 usingSpringWithDamping:0.5 initialSpringVelocity:0.1 options:UIViewAnimationOptionCurveEaseIn animations:^{

    [_notifView setFrame:CGRectMake(0, 0, self.window.frame.size.width, 60)];

} completion:^(BOOL finished) {


}];


//Remove from top view after 5 seconds
[self performSelector:@selector(dismissNotifFromScreen) withObject:nil afterDelay:5.0];


return;


}

//If the user touches the view or to remove from view after 5 seconds
- (void)dismissNotifFromScreen{

[UIView animateWithDuration:1.0 delay:.1 usingSpringWithDamping:0.5 initialSpringVelocity:0.1 options:UIViewAnimationOptionCurveEaseIn animations:^{

    [_notifView setFrame:CGRectMake(0, -70, self.window.frame.size.width, 60)];

} completion:^(BOOL finished) {


}];


}
6
  • 3
    This may be a good answer for creating your own alert, but it isn't showing the Stock iOS Alert as shown in the screenshot.
    – pkamb
    Jun 7, 2016 at 20:01
  • Hi,Thank you for your kind words. Yes, it won't look like that (if you are talking about the appearance) as the one shown above in the image is iOS notification and the one via code is custom. We can maybe replicate the looks in the view. I think WhatsApp has a neat looking one with background blur etc.
    – Hafeez
    Jun 9, 2016 at 4:14
  • 1
    Someone down voted :(. Would be nice to hear why!?. Thanks.
    – Hafeez
    Jun 22, 2016 at 3:17
  • 3
    I did, as the code in this answer (nice though it may be) doesn't really answer the question asked: showing a stock iOS toast banner inside the app. There are other questions on SO where this answer would be great.
    – pkamb
    Jun 22, 2016 at 22:44
  • 2
    Thanks pkamb for clarifying, makes sense. I missed the "stock" word I guess.
    – Hafeez
    Jun 23, 2016 at 12:16
10

Here is the code to receive Push Notification when app in foreground or in open stage, iOS 10 & Swift 2.3

@available(iOS 10.0, *)
func userNotificationCenter(center: UNUserNotificationCenter, willPresentNotification notification: UNNotification, withCompletionHandler completionHandler: (UNNotificationPresentationOptions) -> Void)
{
     completionHandler([UNNotificationPresentationOptions.Alert,UNNotificationPresentationOptions.Sound,UNNotificationPresentationOptions.Badge])
}

If you need to access userInfo of notification use code: notification.request.content.userInfo

The method userNotificationCenter(_:willPresent:withCompletionHandler:) is only called if you add to payload the attribute content-available:1. The final payload should be something like:

{
     "aps":{
          "alert":"Testing.. (7)",
          "badge":1,"sound":"default"
     },
     "content-available":1
}
1
  • 2
    userNotificationCenter(_:willPresent:withCompletionHandler:) will be called when your app is running in the foreground. So it does NOT matter with "content-available", which is related to "background fetch".
    – DawnSong
    Mar 1, 2018 at 4:25
10
UNMutableNotificationContent *content = [[UNMutableNotificationContent alloc] init];
content.body = body;
content.userInfo = userInfo;
content.sound = [UNNotificationSound defaultSound];
[content setValue:@(YES) forKeyPath:@"shouldAlwaysAlertWhileAppIsForeground"];

UNNotificationRequest *request = [UNNotificationRequest requestWithIdentifier:@"Notif" content:content trigger:nil];
[[UNUserNotificationCenter currentNotificationCenter] addNotificationRequest:request withCompletionHandler:^(NSError * _Nullable error) {
    DLog(@"Error:%@", error);
}];

I can show push notification when app is active for iOS 10.

  1. The push notification from server should be silent.

  2. When receive remote notification from server, you send a local notification and set the value for keyPath: shouldAlwaysAlertWhileAppIsForeground = True

2
  • in swift 2 - it is content.setValue("YES", forKeyPath: "shouldAlwaysAlertWhileAppIsForeground") , Right? (with "Yes" and not true)
    – Yoav R.
    Dec 26, 2016 at 17:49
  • Caution: This will crash your app under iOS 12. See discussion here: stackoverflow.com/questions/41373321/… Sep 18, 2018 at 17:01
6

You can handle the notification yourself and show a custom alert. Apps such as Viber, Whatsapp, and BisPhone use this approach.

One example of a 3rd party custom alert is CRToast.

Try scheduling a local notification while your app is in the foreground, and you will see that no stock iOS alert is shown:

if (application.applicationState == UIApplicationStateActive ) {
     UILocalNotification *localNotification = [[UILocalNotification alloc] init];
    localNotification.userInfo = userInfo;
    localNotification.soundName = UILocalNotificationDefaultSoundName;
    localNotification.alertBody = message;
    localNotification.fireDate = [NSDate date];
    [[UIApplication sharedApplication] scheduleLocalNotification:localNotification];
}
3
  • Thanks, but just looking for answers using the stock iOS notification alert.
    – pkamb
    Jun 15, 2015 at 19:18
  • 1
    ok but i think you can't do it , also look at this Link : stackoverflow.com/questions/14872088/…
    – MoLowKey
    Jun 15, 2015 at 19:48
  • 6
    @matt I am attempting to keep this question on topic: showing the stock iOS alert in the foreground. Plenty of other "best toast alert framework" questions. An answer of "No, you cannot show the iOS alert in the foreground" would be perfectly acceptable, and I will accept that answer until this is possible in a future version of iOS.
    – pkamb
    Jun 15, 2015 at 21:52
5

Swift 3 version

This shows an alert when the application is in the foreground.

if #available(iOS 10.0, *)
{
    // need to setup the global notification delegate somewhere when your app starts
    //
    UNUserNotificationCenter.current().delegate = applicationDelegate

    // to show a message
    //
    let content = UNMutableNotificationContent()
    content.body = "MESSAGE"

    let request = UNNotificationRequest(identifier: "fred", content: content, trigger: nil)
    UNUserNotificationCenter.current().add(request)
    {
       error in // called when message has been sent

       debugPrint("Error: \(error)")
    }
}

ApplicationDelegate's implementation of UNUserNotificationCenterDelegate

@available(iOS 10.0, *)
public func userNotificationCenter(_ center : UNUserNotificationCenter, willPresent notification : UNNotification, withCompletionHandler completionHandler : @escaping (UNNotificationPresentationOptions) -> Void)
{
    completionHandler([.alert]) // only-always show the alert
}
3
  • just to be sure...do you mean since iOS 10 you can now also show alerts/banners/sounds in foreground just like you do in background?
    – mfaani
    Jun 27, 2017 at 19:54
  • Where I can put the above coding part? in did receiveRemoteNotification? Aug 1, 2017 at 17:19
  • @Leslie Godwin will it work for ios 8.0 for showing notifications in Foreground Sep 22, 2017 at 7:35
2

To show local Notification this is the best option . need less code to wright "BRYXBanner " https://cocoapods.org/pods/BRYXBanner

let banner = Banner(title: "title", subtitle: "subtitle", image: UIImage(named: "addContact"), backgroundColor: UIColor(red:137.0/255.0, green:172.0/255.0, blue:2.0/255.0, alpha:1.000))
    banner.dismissesOnTap = true
    banner.show(duration: 1.0)
1
  • This BRYXBanner framework does not seem to use the stock iOS notification banner.
    – pkamb
    Oct 6, 2016 at 16:26
2

Here is a version that works with SwiftUI. Modify your main App file as such:

import SwiftUI
import UserNotifications

let appDelegate = AppDelegate()

@main
struct MyApp: App {
    let persistenceController = PersistenceController.shared
    
    var body: some Scene {
        WindowGroup {
            ContentView()
                .onAppear() {
                    UNUserNotificationCenter.current().delegate = appDelegate
                }
                .environment(\.managedObjectContext, persistenceController.container.viewContext)
        }
    }
}

class AppDelegate: UIResponder, UIApplicationDelegate {
    
}

// Conform to UNUserNotificationCenterDelegate
extension AppDelegate: UNUserNotificationCenterDelegate {
    func userNotificationCenter(_ center: UNUserNotificationCenter,
                                willPresent notification: UNNotification,
                                withCompletionHandler completionHandler: @escaping (UNNotificationPresentationOptions) -> Void)
    {
        completionHandler([.alert, .badge, .sound])
    }
}
1

If your deployment target >= iOS10, use UNUserNotification like below-

func userNotificationCenter(_ center: UNUserNotificationCenter,
                                willPresent notification: UNNotification,
                                withCompletionHandler completionHandler: @escaping (UNNotificationPresentationOptions) -> Void)
    {
        // Change this to your preferred presentation option
        completionHandler([.alert, .sound])
    }
1
  • Could you please explain how's this working? like when is this function being called? Jul 3, 2021 at 11:53

Your Answer

By clicking “Post Your Answer”, you agree to our terms of service and acknowledge you have read our privacy policy.

Not the answer you're looking for? Browse other questions tagged or ask your own question.