#import "RNFirebaseNotifications.h" #if __has_include() #import "RNFirebaseEvents.h" #import "RNFirebaseMessaging.h" #import "RNFirebaseUtil.h" #import // For iOS 10 we need to implement UNUserNotificationCenterDelegate to receive display // notifications via APNS #if defined(__IPHONE_10_0) && __IPHONE_OS_VERSION_MAX_ALLOWED >= __IPHONE_10_0 @import UserNotifications; @interface RNFirebaseNotifications () #else @interface RNFirebaseNotifications () #endif @end @implementation RNFirebaseNotifications { NSMutableDictionary *completionHandlers; } static RNFirebaseNotifications *theRNFirebaseNotifications = nil; // PRE-BRIDGE-EVENTS: Consider enabling this to allow events built up before the bridge is built to be sent to the JS side // static NSMutableArray *pendingEvents = nil; static NSDictionary *initialNotification = nil; static bool jsReady = FALSE; static NSString *const DEFAULT_ACTION = @"com.apple.UNNotificationDefaultActionIdentifier"; + (nonnull instancetype)instance { return theRNFirebaseNotifications; } + (void)configure { // PRE-BRIDGE-EVENTS: Consider enabling this to allow events built up before the bridge is built to be sent to the JS side // pendingEvents = [[NSMutableArray alloc] init]; theRNFirebaseNotifications = [[RNFirebaseNotifications alloc] init]; } RCT_EXPORT_MODULE(); - (id)init { self = [super init]; if (self != nil) { NSLog(@"Setting up RNFirebaseNotifications instance"); [self initialise]; } return self; } - (void)initialise { // If we're on iOS 10 then we need to set this as a delegate for the UNUserNotificationCenter if (@available(iOS 10.0, *)) { [UNUserNotificationCenter currentNotificationCenter].delegate = self; } // Set static instance for use from AppDelegate theRNFirebaseNotifications = self; completionHandlers = [[NSMutableDictionary alloc] init]; } // PRE-BRIDGE-EVENTS: Consider enabling this to allow events built up before the bridge is built to be sent to the JS side // The bridge is initialised after the module is created // When the bridge is set, check if we have any pending events to send, and send them /* - (void)setValue:(nullable id)value forKey:(NSString *)key { [super setValue:value forKey:key]; if ([key isEqualToString:@"bridge"] && value) { for (NSDictionary* event in pendingEvents) { [RNFirebaseUtil sendJSEvent:self name:event[@"name"] body:event[@"body"]]; } [pendingEvents removeAllObjects]; } } */ // ******************************************************* // ** Start AppDelegate methods // ** iOS 8/9 Only // ******************************************************* - (void)didReceiveLocalNotification:(nonnull UILocalNotification *)localNotification { if ([self isIOS89]) { NSString *event; if (RCTSharedApplication().applicationState == UIApplicationStateBackground) { event = NOTIFICATIONS_NOTIFICATION_DISPLAYED; } else if (RCTSharedApplication().applicationState == UIApplicationStateInactive) { event = NOTIFICATIONS_NOTIFICATION_OPENED; } else { event = NOTIFICATIONS_NOTIFICATION_RECEIVED; } NSDictionary *notification = [self parseUILocalNotification:localNotification]; if (event == NOTIFICATIONS_NOTIFICATION_OPENED) { notification = @{ @"action": DEFAULT_ACTION, @"notification": notification }; } [self sendJSEvent:self name:event body:notification]; } } RCT_EXPORT_METHOD(complete:(NSString*)handlerKey fetchResult:(UIBackgroundFetchResult)fetchResult) { void (^completionHandler)(UIBackgroundFetchResult) = completionHandlers[handlerKey]; completionHandlers[handlerKey] = nil; if(completionHandler != nil) { completionHandler(fetchResult); } } // Listen for background messages - (void)didReceiveRemoteNotification:(NSDictionary *)userInfo fetchCompletionHandler:(void (^)(UIBackgroundFetchResult))completionHandler { // FCM Data messages come through here if they specify content-available=true // Pass them over to the RNFirebaseMessaging handler instead if (userInfo[@"aps"] && ((NSDictionary*)userInfo[@"aps"]).count == 1 && userInfo[@"aps"][@"content-available"]) { [[RNFirebaseMessaging instance] didReceiveRemoteNotification:userInfo]; completionHandler(UIBackgroundFetchResultNoData); return; } NSDictionary *notification = [self parseUserInfo:userInfo]; NSString *handlerKey = notification[@"notificationId"]; NSString *event; if (RCTSharedApplication().applicationState == UIApplicationStateBackground) { event = NOTIFICATIONS_NOTIFICATION_DISPLAYED; } else if ([self isIOS89]) { if (RCTSharedApplication().applicationState == UIApplicationStateInactive) { event = NOTIFICATIONS_NOTIFICATION_OPENED; } else { event = NOTIFICATIONS_NOTIFICATION_RECEIVED; } } else { // On IOS 10: // - foreground notifications also go through willPresentNotification // - background notification presses also go through didReceiveNotificationResponse // This prevents duplicate messages from hitting the JS app completionHandler(UIBackgroundFetchResultNoData); return; } // For onOpened events, we set the default action name as iOS 8/9 has no concept of actions if (event == NOTIFICATIONS_NOTIFICATION_OPENED) { notification = @{ @"action": DEFAULT_ACTION, @"notification": notification }; } completionHandlers[handlerKey] = completionHandler; [self sendJSEvent:self name:event body:notification]; } // ******************************************************* // ** Finish AppDelegate methods // ******************************************************* // ******************************************************* // ** Start UNUserNotificationCenterDelegate methods // ** iOS 10+ // ******************************************************* #if defined(__IPHONE_10_0) && __IPHONE_OS_VERSION_MAX_ALLOWED >= __IPHONE_10_0 // Handle incoming notification messages while app is in the foreground. - (void)userNotificationCenter:(UNUserNotificationCenter *)center willPresentNotification:(UNNotification *)notification withCompletionHandler:(void (^)(UNNotificationPresentationOptions))completionHandler NS_AVAILABLE_IOS(10_0) { UNNotificationTrigger *trigger = notification.request.trigger; BOOL isFcm = trigger && [notification.request.trigger class] == [UNPushNotificationTrigger class]; BOOL isScheduled = trigger && [notification.request.trigger class] == [UNCalendarNotificationTrigger class]; NSString *event; UNNotificationPresentationOptions options; NSDictionary *message = [self parseUNNotification:notification]; if (isFcm || isScheduled) { // If app is in the background if (RCTSharedApplication().applicationState == UIApplicationStateBackground || RCTSharedApplication().applicationState == UIApplicationStateInactive) { // display the notification options = UNNotificationPresentationOptionAlert | UNNotificationPresentationOptionBadge | UNNotificationPresentationOptionSound; // notification_displayed event = NOTIFICATIONS_NOTIFICATION_DISPLAYED; } else { // don't show notification options = UNNotificationPresentationOptionNone; // notification_received event = NOTIFICATIONS_NOTIFICATION_RECEIVED; } } else { // Triggered by `notifications().displayNotification(notification)` // Display the notification options = UNNotificationPresentationOptionAlert | UNNotificationPresentationOptionBadge | UNNotificationPresentationOptionSound; // notification_displayed event = NOTIFICATIONS_NOTIFICATION_DISPLAYED; } [self sendJSEvent:self name:event body:message]; completionHandler(options); } // Handle notification messages after display notification is tapped by the user. - (void)userNotificationCenter:(UNUserNotificationCenter *)center didReceiveNotificationResponse:(UNNotificationResponse *)response #if defined(__IPHONE_11_0) withCompletionHandler:(void(^)(void))completionHandler NS_AVAILABLE_IOS(10_0) { #else withCompletionHandler:(void(^)())completionHandler NS_AVAILABLE_IOS(10_0) { #endif NSDictionary *message = [self parseUNNotificationResponse:response]; [self sendJSEvent:self name:NOTIFICATIONS_NOTIFICATION_OPENED body:message]; completionHandler(); } #endif // ******************************************************* // ** Finish UNUserNotificationCenterDelegate methods // ******************************************************* RCT_EXPORT_METHOD(cancelAllNotifications:(RCTPromiseResolveBlock)resolve rejecter:(RCTPromiseRejectBlock)reject) { if ([self isIOS89]) { [RCTSharedApplication() cancelAllLocalNotifications]; } else { if (@available(iOS 10.0, *)) { UNUserNotificationCenter *notificationCenter = [UNUserNotificationCenter currentNotificationCenter]; if (notificationCenter != nil) { [[UNUserNotificationCenter currentNotificationCenter] removeAllPendingNotificationRequests]; } } } resolve(nil); } RCT_EXPORT_METHOD(cancelNotification:(NSString*) notificationId resolver:(RCTPromiseResolveBlock)resolve rejecter:(RCTPromiseRejectBlock)reject) { if ([self isIOS89]) { for (UILocalNotification *notification in RCTSharedApplication().scheduledLocalNotifications) { NSDictionary *notificationInfo = notification.userInfo; if ([notificationId isEqualToString:notificationInfo[@"notificationId"]]) { [RCTSharedApplication() cancelLocalNotification:notification]; } } } else { if (@available(iOS 10.0, *)) { UNUserNotificationCenter *notificationCenter = [UNUserNotificationCenter currentNotificationCenter]; if (notificationCenter != nil) { [[UNUserNotificationCenter currentNotificationCenter] removePendingNotificationRequestsWithIdentifiers:@[notificationId]]; } } } resolve(nil); } RCT_EXPORT_METHOD(displayNotification:(NSDictionary*) notification resolver:(RCTPromiseResolveBlock)resolve rejecter:(RCTPromiseRejectBlock)reject) { if ([self isIOS89]) { UILocalNotification* notif = [self buildUILocalNotification:notification withSchedule:false]; [RCTSharedApplication() presentLocalNotificationNow:notif]; resolve(nil); } else { if (@available(iOS 10.0, *)) { UNNotificationRequest* request = [self buildUNNotificationRequest:notification withSchedule:false]; [[UNUserNotificationCenter currentNotificationCenter] addNotificationRequest:request withCompletionHandler:^(NSError * _Nullable error) { if (!error) { resolve(nil); } else{ reject(@"notifications/display_notification_error", @"Failed to display notificaton", error); } }]; } } } RCT_EXPORT_METHOD(getBadge: (RCTPromiseResolveBlock)resolve rejecter:(RCTPromiseRejectBlock)reject) { dispatch_async(dispatch_get_main_queue(), ^{ resolve(@([RCTSharedApplication() applicationIconBadgeNumber])); }); } RCT_EXPORT_METHOD(getInitialNotification:(RCTPromiseResolveBlock)resolve rejecter:(RCTPromiseRejectBlock)reject) { // Check if we've cached an initial notification as this will contain the accurate action if (initialNotification) { resolve(initialNotification); } else if (self.bridge.launchOptions[UIApplicationLaunchOptionsLocalNotificationKey]) { UILocalNotification *localNotification = self.bridge.launchOptions[UIApplicationLaunchOptionsLocalNotificationKey]; resolve(@{ @"action": DEFAULT_ACTION, @"notification": [self parseUILocalNotification:localNotification] }); } else if (self.bridge.launchOptions[UIApplicationLaunchOptionsRemoteNotificationKey]) { NSDictionary *remoteNotification = [self bridge].launchOptions[UIApplicationLaunchOptionsRemoteNotificationKey]; resolve(@{ @"action": DEFAULT_ACTION, @"notification": [self parseUserInfo:remoteNotification] }); } else { resolve(nil); } } RCT_EXPORT_METHOD(getScheduledNotifications:(RCTPromiseResolveBlock)resolve rejecter:(RCTPromiseRejectBlock)reject) { if ([self isIOS89]) { NSMutableArray* notifications = [[NSMutableArray alloc] init]; for (UILocalNotification *notif in [RCTSharedApplication() scheduledLocalNotifications]){ NSDictionary *notification = [self parseUILocalNotification:notif]; [notifications addObject:notification]; } resolve(notifications); } else { if (@available(iOS 10.0, *)) { [[UNUserNotificationCenter currentNotificationCenter] getPendingNotificationRequestsWithCompletionHandler:^(NSArray * _Nonnull requests) { NSMutableArray* notifications = [[NSMutableArray alloc] init]; for (UNNotificationRequest *notif in requests){ NSDictionary *notification = [self parseUNNotificationRequest:notif]; [notifications addObject:notification]; } resolve(notifications); }]; } } } RCT_EXPORT_METHOD(removeAllDeliveredNotifications:(RCTPromiseResolveBlock)resolve rejecter:(RCTPromiseRejectBlock)reject) { if ([self isIOS89]) { // No such functionality on iOS 8/9 } else { if (@available(iOS 10.0, *)) { UNUserNotificationCenter *notificationCenter = [UNUserNotificationCenter currentNotificationCenter]; if (notificationCenter != nil) { [[UNUserNotificationCenter currentNotificationCenter] removeAllDeliveredNotifications]; } } } resolve(nil); } RCT_EXPORT_METHOD(removeDeliveredNotification:(NSString*) notificationId resolver:(RCTPromiseResolveBlock)resolve rejecter:(RCTPromiseRejectBlock)reject) { if ([self isIOS89]) { // No such functionality on iOS 8/9 } else { if (@available(iOS 10.0, *)) { UNUserNotificationCenter *notificationCenter = [UNUserNotificationCenter currentNotificationCenter]; if (notificationCenter != nil) { [[UNUserNotificationCenter currentNotificationCenter] removeDeliveredNotificationsWithIdentifiers:@[notificationId]]; } } } resolve(nil); } RCT_EXPORT_METHOD(scheduleNotification:(NSDictionary*) notification resolver:(RCTPromiseResolveBlock)resolve rejecter:(RCTPromiseRejectBlock)reject) { if ([self isIOS89]) { UILocalNotification* notif = [self buildUILocalNotification:notification withSchedule:true]; [RCTSharedApplication() scheduleLocalNotification:notif]; resolve(nil); } else { if (@available(iOS 10.0, *)) { UNNotificationRequest* request = [self buildUNNotificationRequest:notification withSchedule:true]; [[UNUserNotificationCenter currentNotificationCenter] addNotificationRequest:request withCompletionHandler:^(NSError * _Nullable error) { if (!error) { resolve(nil); } else{ reject(@"notification/schedule_notification_error", @"Failed to schedule notificaton", error); } }]; } } } RCT_EXPORT_METHOD(setBadge:(NSInteger) number resolver:(RCTPromiseResolveBlock)resolve rejecter:(RCTPromiseRejectBlock)reject) { dispatch_async(dispatch_get_main_queue(), ^{ [RCTSharedApplication() setApplicationIconBadgeNumber:number]; resolve(nil); }); } RCT_EXPORT_METHOD(jsInitialised:(RCTPromiseResolveBlock)resolve rejecter:(RCTPromiseRejectBlock)reject) { jsReady = TRUE; resolve(nil); } // Because of the time delay between the app starting and the bridge being initialised // we create a temporary instance of RNFirebaseNotifications. // With this temporary instance, we cache any events to be sent as soon as the bridge is set on the module - (void)sendJSEvent:(RCTEventEmitter *)emitter name:(NSString *)name body:(id)body { if (emitter.bridge && jsReady) { [RNFirebaseUtil sendJSEvent:emitter name:name body:body]; } else { if ([name isEqualToString:NOTIFICATIONS_NOTIFICATION_OPENED] && !initialNotification) { initialNotification = body; } else if ([name isEqualToString:NOTIFICATIONS_NOTIFICATION_OPENED]) { NSLog(@"Multiple notification open events received before the JS Notifications module has been initialised"); } // PRE-BRIDGE-EVENTS: Consider enabling this to allow events built up before the bridge is built to be sent to the JS side // [pendingEvents addObject:@{@"name":name, @"body":body}]; } } - (BOOL)isIOS89 { return floor(NSFoundationVersionNumber) <= NSFoundationVersionNumber_iOS_9_x_Max; } - (UILocalNotification*) buildUILocalNotification:(NSDictionary *) notification withSchedule:(BOOL) withSchedule { UILocalNotification *localNotification = [[UILocalNotification alloc] init]; if (notification[@"body"]) { localNotification.alertBody = notification[@"body"]; } if (notification[@"data"]) { localNotification.userInfo = notification[@"data"]; } if (notification[@"sound"]) { localNotification.soundName = notification[@"sound"]; } if (notification[@"title"]) { localNotification.alertTitle = notification[@"title"]; } if (notification[@"ios"]) { NSDictionary *ios = notification[@"ios"]; if (ios[@"alertAction"]) { localNotification.alertAction = ios[@"alertAction"]; } if (ios[@"badge"]) { NSNumber *badge = ios[@"badge"]; localNotification.applicationIconBadgeNumber = badge.integerValue; } if (ios[@"category"]) { localNotification.category = ios[@"category"]; } if (ios[@"hasAction"]) { localNotification.hasAction = ios[@"hasAction"]; } if (ios[@"launchImage"]) { localNotification.alertLaunchImage = ios[@"launchImage"]; } } if (withSchedule) { NSDictionary *schedule = notification[@"schedule"]; NSNumber *fireDateNumber = schedule[@"fireDate"]; NSDate *fireDate = [NSDate dateWithTimeIntervalSince1970:([fireDateNumber doubleValue] / 1000.0)]; localNotification.fireDate = fireDate; NSString *interval = schedule[@"repeatInterval"]; if (interval) { if ([interval isEqualToString:@"minute"]) { localNotification.repeatInterval = NSCalendarUnitMinute; } else if ([interval isEqualToString:@"hour"]) { localNotification.repeatInterval = NSCalendarUnitHour; } else if ([interval isEqualToString:@"day"]) { localNotification.repeatInterval = NSCalendarUnitDay; } else if ([interval isEqualToString:@"week"]) { localNotification.repeatInterval = NSCalendarUnitWeekday; } } } return localNotification; } - (UNNotificationRequest*) buildUNNotificationRequest:(NSDictionary *) notification withSchedule:(BOOL) withSchedule NS_AVAILABLE_IOS(10_0) { UNMutableNotificationContent *content = [[UNMutableNotificationContent alloc] init]; if (notification[@"body"]) { content.body = notification[@"body"]; } if (notification[@"data"]) { content.userInfo = notification[@"data"]; } if (notification[@"sound"]) { if ([@"default" isEqualToString:notification[@"sound"]]) { content.sound = [UNNotificationSound defaultSound]; } else { content.sound = [UNNotificationSound soundNamed:notification[@"sound"]]; } } if (notification[@"subtitle"]) { content.subtitle = notification[@"subtitle"]; } if (notification[@"title"]) { content.title = notification[@"title"]; } if (notification[@"ios"]) { NSDictionary *ios = notification[@"ios"]; if (ios[@"attachments"]) { NSMutableArray *attachments = [[NSMutableArray alloc] init]; for (NSDictionary *a in ios[@"attachments"]) { NSString *identifier = a[@"identifier"]; NSURL *url = [NSURL fileURLWithPath:a[@"url"]]; NSMutableDictionary *attachmentOptions = nil; if (a[@"options"]) { NSDictionary *options = a[@"options"]; attachmentOptions = [[NSMutableDictionary alloc] init]; for (id key in options) { if ([key isEqualToString:@"typeHint"]) { attachmentOptions[UNNotificationAttachmentOptionsTypeHintKey] = options[key]; } else if ([key isEqualToString:@"thumbnailHidden"]) { attachmentOptions[UNNotificationAttachmentOptionsThumbnailHiddenKey] = options[key]; } else if ([key isEqualToString:@"thumbnailClippingRect"]) { attachmentOptions[UNNotificationAttachmentOptionsThumbnailClippingRectKey] = options[key]; } else if ([key isEqualToString:@"thumbnailTime"]) { attachmentOptions[UNNotificationAttachmentOptionsThumbnailTimeKey] = options[key]; } } } NSError *error; UNNotificationAttachment *attachment = [UNNotificationAttachment attachmentWithIdentifier:identifier URL:url options:attachmentOptions error:&error]; if (attachment) { [attachments addObject:attachment]; } else { NSLog(@"Failed to create attachment: %@", error); } } content.attachments = attachments; } if (ios[@"badge"]) { content.badge = ios[@"badge"]; } if (ios[@"category"]) { content.categoryIdentifier = ios[@"category"]; } if (ios[@"launchImage"]) { content.launchImageName = ios[@"launchImage"]; } if (ios[@"threadIdentifier"]) { content.threadIdentifier = ios[@"threadIdentifier"]; } } if (withSchedule) { NSDictionary *schedule = notification[@"schedule"]; NSNumber *fireDateNumber = schedule[@"fireDate"]; NSString *interval = schedule[@"repeatInterval"]; NSDate *fireDate = [NSDate dateWithTimeIntervalSince1970:([fireDateNumber doubleValue] / 1000.0)]; NSCalendarUnit calendarUnit; if (interval) { if ([interval isEqualToString:@"minute"]) { calendarUnit = NSCalendarUnitSecond; } else if ([interval isEqualToString:@"hour"]) { calendarUnit = NSCalendarUnitMinute | NSCalendarUnitSecond; } else if ([interval isEqualToString:@"day"]) { calendarUnit = NSCalendarUnitHour | NSCalendarUnitMinute | NSCalendarUnitSecond; } else if ([interval isEqualToString:@"week"]) { calendarUnit = NSCalendarUnitWeekday | NSCalendarUnitHour | NSCalendarUnitMinute | NSCalendarUnitSecond; } } else { // Needs to match exactly to the secpmd calendarUnit = NSCalendarUnitYear | NSCalendarUnitMonth | NSCalendarUnitDay | NSCalendarUnitHour | NSCalendarUnitMinute | NSCalendarUnitSecond; } NSDateComponents *components = [[NSCalendar currentCalendar] components:calendarUnit fromDate:fireDate]; UNCalendarNotificationTrigger *trigger = [UNCalendarNotificationTrigger triggerWithDateMatchingComponents:components repeats:interval]; return [UNNotificationRequest requestWithIdentifier:notification[@"notificationId"] content:content trigger:trigger]; } else { return [UNNotificationRequest requestWithIdentifier:notification[@"notificationId"] content:content trigger:nil]; } } - (NSDictionary*) parseUILocalNotification:(UILocalNotification *) localNotification { NSMutableDictionary *notification = [[NSMutableDictionary alloc] init]; if (localNotification.alertBody) { notification[@"body"] = localNotification.alertBody; } if (localNotification.userInfo) { notification[@"data"] = localNotification.userInfo; } if (localNotification.soundName) { notification[@"sound"] = localNotification.soundName; } if (localNotification.alertTitle) { notification[@"title"] = localNotification.alertTitle; } NSMutableDictionary *ios = [[NSMutableDictionary alloc] init]; if (localNotification.alertAction) { ios[@"alertAction"] = localNotification.alertAction; } if (localNotification.applicationIconBadgeNumber) { ios[@"badge"] = @(localNotification.applicationIconBadgeNumber); } if (localNotification.category) { ios[@"category"] = localNotification.category; } if (localNotification.hasAction) { ios[@"hasAction"] = @(localNotification.hasAction); } if (localNotification.alertLaunchImage) { ios[@"launchImage"] = localNotification.alertLaunchImage; } notification[@"ios"] = ios; return notification; } - (NSDictionary*)parseUNNotificationResponse:(UNNotificationResponse *)response NS_AVAILABLE_IOS(10_0) { NSMutableDictionary *notificationResponse = [[NSMutableDictionary alloc] init]; NSDictionary *notification = [self parseUNNotification:response.notification]; notificationResponse[@"notification"] = notification; notificationResponse[@"action"] = response.actionIdentifier; return notificationResponse; } - (NSDictionary*)parseUNNotification:(UNNotification *)notification NS_AVAILABLE_IOS(10_0) { return [self parseUNNotificationRequest:notification.request]; } - (NSDictionary*) parseUNNotificationRequest:(UNNotificationRequest *) notificationRequest NS_AVAILABLE_IOS(10_0) { NSMutableDictionary *notification = [[NSMutableDictionary alloc] init]; notification[@"notificationId"] = notificationRequest.identifier; if (notificationRequest.content.body) { notification[@"body"] = notificationRequest.content.body; } if (notificationRequest.content.userInfo) { NSMutableDictionary *data = [[NSMutableDictionary alloc] init]; for (id k in notificationRequest.content.userInfo) { if ([k isEqualToString:@"aps"] || [k isEqualToString:@"gcm.message_id"]) { // ignore as these are handled by the OS } else { data[k] = notificationRequest.content.userInfo[k]; } } notification[@"data"] = data; } if (notificationRequest.content.sound) { notification[@"sound"] = notificationRequest.content.sound; } if (notificationRequest.content.subtitle) { notification[@"subtitle"] = notificationRequest.content.subtitle; } if (notificationRequest.content.title) { notification[@"title"] = notificationRequest.content.title; } NSMutableDictionary *ios = [[NSMutableDictionary alloc] init]; if (notificationRequest.content.attachments) { NSMutableArray *attachments = [[NSMutableArray alloc] init]; for (UNNotificationAttachment *a in notificationRequest.content.attachments) { NSMutableDictionary *attachment = [[NSMutableDictionary alloc] init]; attachment[@"identifier"] = a.identifier; attachment[@"type"] = a.type; attachment[@"url"] = [a.URL absoluteString]; [attachments addObject:attachment]; } ios[@"attachments"] = attachments; } if (notificationRequest.content.badge) { ios[@"badge"] = notificationRequest.content.badge; } if (notificationRequest.content.categoryIdentifier) { ios[@"category"] = notificationRequest.content.categoryIdentifier; } if (notificationRequest.content.launchImageName) { ios[@"launchImage"] = notificationRequest.content.launchImageName; } if (notificationRequest.content.threadIdentifier) { ios[@"threadIdentifier"] = notificationRequest.content.threadIdentifier; } notification[@"ios"] = ios; return notification; } - (NSDictionary*)parseUserInfo:(NSDictionary *)userInfo { NSMutableDictionary *notification = [[NSMutableDictionary alloc] init]; NSMutableDictionary *data = [[NSMutableDictionary alloc] init]; NSMutableDictionary *ios = [[NSMutableDictionary alloc] init]; for (id k1 in userInfo) { if ([k1 isEqualToString:@"aps"]) { NSDictionary *aps = userInfo[k1]; for (id k2 in aps) { if ([k2 isEqualToString:@"alert"]) { // alert can be a plain text string rather than a dictionary if ([aps[k2] isKindOfClass:[NSDictionary class]]) { NSDictionary *alert = aps[k2]; for (id k3 in alert) { if ([k3 isEqualToString:@"body"]) { notification[@"body"] = alert[k3]; } else if ([k3 isEqualToString:@"subtitle"]) { notification[@"subtitle"] = alert[k3]; } else if ([k3 isEqualToString:@"title"]) { notification[@"title"] = alert[k3]; } else if ([k3 isEqualToString:@"loc-args"] || [k3 isEqualToString:@"loc-key"] || [k3 isEqualToString:@"title-loc-args"] || [k3 isEqualToString:@"title-loc-key"]) { // Ignore known keys } else { NSLog(@"Unknown alert key: %@", k2); } } } else { notification[@"title"] = aps[k2]; } } else if ([k2 isEqualToString:@"badge"]) { ios[@"badge"] = aps[k2]; } else if ([k2 isEqualToString:@"category"]) { ios[@"category"] = aps[k2]; } else if ([k2 isEqualToString:@"sound"]) { notification[@"sound"] = aps[k2]; } else { NSLog(@"Unknown aps key: %@", k2); } } } else if ([k1 isEqualToString:@"gcm.message_id"]) { notification[@"notificationId"] = userInfo[k1]; } else if ([k1 isEqualToString:@"gcm.n.e"] || [k1 isEqualToString:@"gcm.notification.sound2"] || [k1 isEqualToString:@"google.c.a.c_id"] || [k1 isEqualToString:@"google.c.a.c_l"] || [k1 isEqualToString:@"google.c.a.e"] || [k1 isEqualToString:@"google.c.a.udt"] || [k1 isEqualToString:@"google.c.a.ts"]) { // Ignore known keys } else { // Assume custom data data[k1] = userInfo[k1]; } } notification[@"data"] = data; notification[@"ios"] = ios; return notification; } - (NSArray *)supportedEvents { return @[NOTIFICATIONS_NOTIFICATION_DISPLAYED, NOTIFICATIONS_NOTIFICATION_OPENED, NOTIFICATIONS_NOTIFICATION_RECEIVED]; } - (NSDictionary *) constantsToExport { return @{ @"backgroundFetchResultNoData" : @(UIBackgroundFetchResultNoData), @"backgroundFetchResultNewData" : @(UIBackgroundFetchResultNewData), @"backgroundFetchResultFailed" : @(UIBackgroundFetchResultFailed)}; } + (BOOL)requiresMainQueueSetup { return YES; } @end #else @implementation RNFirebaseNotifications @end #endif