| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684 |
- /*
- * Copyright (c) 2013-2015 by appPlant UG. All rights reserved.
- *
- * @APPPLANT_LICENSE_HEADER_START@
- *
- * This file contains Original Code and/or Modifications of Original Code
- * as defined in and that are subject to the Apache License
- * Version 2.0 (the 'License'). You may not use this file except in
- * compliance with the License. Please obtain a copy of the License at
- * http://opensource.org/licenses/Apache-2.0/ and read it before using this
- * file.
- *
- * The Original Code and all software distributed under the License are
- * distributed on an 'AS IS' basis, WITHOUT WARRANTY OF ANY KIND, EITHER
- * EXPRESS OR IMPLIED, AND APPLE HEREBY DISCLAIMS ALL SUCH WARRANTIES,
- * INCLUDING WITHOUT LIMITATION, ANY WARRANTIES OF MERCHANTABILITY,
- * FITNESS FOR A PARTICULAR PURPOSE, QUIET ENJOYMENT OR NON-INFRINGEMENT.
- * Please see the License for the specific language governing rights and
- * limitations under the License.
- *
- * @APPPLANT_LICENSE_HEADER_END@
- */
- #import "APPLocalNotification.h"
- #import "APPLocalNotificationOptions.h"
- #import "AppDelegate+APPLocalNotification.h"
- #import "UIApplication+APPLocalNotification.h"
- #import "UILocalNotification+APPLocalNotification.h"
- @interface APPLocalNotification ()
- // Retrieves the application state
- @property (readonly, getter=applicationState) NSString* applicationState;
- // All events will be queued until deviceready has been fired
- @property (readwrite, assign) BOOL deviceready;
- // Event queue
- @property (readonly, nonatomic, retain) NSMutableArray* eventQueue;
- // Needed when calling `registerPermission`
- @property (nonatomic, retain) CDVInvokedUrlCommand* command;
- @end
- @implementation APPLocalNotification
- @synthesize deviceready, eventQueue;
- #pragma mark -
- #pragma mark Interface
- /**
- * Execute all queued events.
- */
- - (void) deviceready:(CDVInvokedUrlCommand*)command
- {
- deviceready = YES;
- for (NSString* js in eventQueue) {
- [self.commandDelegate evalJs:js];
- }
- [eventQueue removeAllObjects];
- }
- /**
- * Schedule a set of notifications.
- *
- * @param properties
- * A dict of properties for each notification
- */
- - (void) schedule:(CDVInvokedUrlCommand*)command
- {
- NSArray* notifications = command.arguments;
- [self.commandDelegate runInBackground:^{
- for (NSDictionary* options in notifications) {
- UILocalNotification* notification;
- notification = [[UILocalNotification alloc]
- initWithOptions:options];
- [self scheduleLocalNotification:[notification copy]];
- [self fireEvent:@"schedule" notification:notification];
- if (notifications.count > 1) {
- [NSThread sleepForTimeInterval:0.01];
- }
- }
- [self execCallback:command];
- }];
- }
- /**
- * Update a set of notifications.
- *
- * @param properties
- * A dict of properties for each notification
- */
- - (void) update:(CDVInvokedUrlCommand*)command
- {
- NSArray* notifications = command.arguments;
- [self.commandDelegate runInBackground:^{
- for (NSDictionary* options in notifications) {
- NSString* id = [options objectForKey:@"id"];
- UILocalNotification* notification;
- notification = [self.app localNotificationWithId:id];
- if (!notification)
- continue;
- [self updateLocalNotification:[notification copy]
- withOptions:options];
- [self fireEvent:@"update" notification:notification];
- if (notifications.count > 1) {
- [NSThread sleepForTimeInterval:0.01];
- }
- }
- [self execCallback:command];
- }];
- }
- /**
- * Cancel a set of notifications.
- *
- * @param ids
- * The IDs of the notifications
- */
- - (void) cancel:(CDVInvokedUrlCommand*)command
- {
- [self.commandDelegate runInBackground:^{
- for (NSString* id in command.arguments) {
- UILocalNotification* notification;
- notification = [self.app localNotificationWithId:id];
- if (!notification)
- continue;
- [self.app cancelLocalNotification:notification];
- [self fireEvent:@"cancel" notification:notification];
- }
- [self execCallback:command];
- }];
- }
- /**
- * Cancel all local notifications.
- */
- - (void) cancelAll:(CDVInvokedUrlCommand*)command
- {
- [self.commandDelegate runInBackground:^{
- [self cancelAllLocalNotifications];
- [self fireEvent:@"cancelall"];
- [self execCallback:command];
- }];
- }
- /**
- * Clear a set of notifications.
- *
- * @param ids
- * The IDs of the notifications
- */
- - (void) clear:(CDVInvokedUrlCommand*)command
- {
- [self.commandDelegate runInBackground:^{
- for (NSString* id in command.arguments) {
- UILocalNotification* notification;
- notification = [self.app localNotificationWithId:id];
- if (!notification)
- continue;
- [self.app clearLocalNotification:notification];
- [self fireEvent:@"clear" notification:notification];
- }
- [self execCallback:command];
- }];
- }
- /**
- * Clear all local notifications.
- */
- - (void) clearAll:(CDVInvokedUrlCommand*)command
- {
- [self.commandDelegate runInBackground:^{
- [self clearAllLocalNotifications];
- [self fireEvent:@"clearall"];
- [self execCallback:command];
- }];
- }
- /**
- * If a notification by ID is present.
- *
- * @param id
- * The ID of the notification
- */
- - (void) isPresent:(CDVInvokedUrlCommand *)command
- {
- [self isPresent:command type:NotifcationTypeAll];
- }
- /**
- * If a notification by ID is scheduled.
- *
- * @param id
- * The ID of the notification
- */
- - (void) isScheduled:(CDVInvokedUrlCommand*)command
- {
- [self isPresent:command type:NotifcationTypeScheduled];
- }
- /**
- * Check if a notification with an ID is triggered.
- *
- * @param id
- * The ID of the notification
- */
- - (void) isTriggered:(CDVInvokedUrlCommand*)command
- {
- [self isPresent:command type:NotifcationTypeTriggered];
- }
- /**
- * Check if a notification with an ID exists.
- *
- * @param type
- * The notification life cycle type
- */
- - (void) isPresent:(CDVInvokedUrlCommand*)command
- type:(APPLocalNotificationType)type;
- {
- [self.commandDelegate runInBackground:^{
- NSString* id = [command argumentAtIndex:0];
- BOOL exist;
- CDVPluginResult* result;
- if (type == NotifcationTypeAll) {
- exist = [self.app localNotificationExist:id];
- } else {
- exist = [self.app localNotificationExist:id type:type];
- }
- result = [CDVPluginResult resultWithStatus:CDVCommandStatus_OK
- messageAsBool:exist];
- [self.commandDelegate sendPluginResult:result
- callbackId:command.callbackId];
- }];
- }
- /**
- * List all ids from all local notifications.
- */
- - (void) getAllIds:(CDVInvokedUrlCommand*)command
- {
- [self getIds:command byType:NotifcationTypeAll];
- }
- /**
- * List all ids from all pending notifications.
- */
- - (void) getScheduledIds:(CDVInvokedUrlCommand*)command
- {
- [self getIds:command byType:NotifcationTypeScheduled];
- }
- /**
- * List all ids from all triggered notifications.
- */
- - (void) getTriggeredIds:(CDVInvokedUrlCommand*)command
- {
- [self getIds:command byType:NotifcationTypeTriggered];
- }
- /**
- * List of ids for given local notifications.
- *
- * @param type
- * Notification life cycle type
- * @param ids
- * The IDs of the notifications
- */
- - (void) getIds:(CDVInvokedUrlCommand*)command
- byType:(APPLocalNotificationType)type;
- {
- [self.commandDelegate runInBackground:^{
- CDVPluginResult* result;
- NSArray* ids;
- if (type == NotifcationTypeAll) {
- ids = [self.app localNotificationIds];
- } else {
- ids = [self.app localNotificationIdsByType:type];
- }
- result = [CDVPluginResult resultWithStatus:CDVCommandStatus_OK
- messageAsArray:ids];
- [self.commandDelegate sendPluginResult:result
- callbackId:command.callbackId];
- }];
- }
- /**
- * Property list for given local notifications.
- *
- * @param ids
- * The IDs of the notifications
- */
- - (void) getAll:(CDVInvokedUrlCommand*)command
- {
- [self getOptions:command byType:NotifcationTypeAll];
- }
- /**
- * Property list for given scheduled notifications.
- *
- * @param ids
- * The IDs of the notifications
- */
- - (void) getScheduled:(CDVInvokedUrlCommand*)command
- {
- [self getOptions:command byType:NotifcationTypeScheduled];
- }
- /**
- * Property list for given triggered notifications.
- *
- * @param ids
- * The IDs of the notifications
- */
- - (void) getTriggered:(CDVInvokedUrlCommand *)command
- {
- [self getOptions:command byType:NotifcationTypeTriggered];
- }
- /**
- * Property list for given triggered notifications.
- *
- * @param type
- * Notification life cycle type
- * @param ids
- * The IDs of the notifications
- */
- - (void) getOptions:(CDVInvokedUrlCommand*)command
- byType:(APPLocalNotificationType)type;
- {
- [self.commandDelegate runInBackground:^{
- NSArray* ids = command.arguments;
- NSArray* notifications;
- CDVPluginResult* result;
- if (type == NotifcationTypeAll && ids.count == 0) {
- notifications = [self.app localNotificationOptions];
- }
- else if (type == NotifcationTypeAll) {
- notifications = [self.app localNotificationOptionsById:ids];
- }
- else if (ids.count == 0) {
- notifications = [self.app localNotificationOptionsByType:type];
- }
- else {
- notifications = [self.app localNotificationOptionsByType:type
- andId:ids];
- }
- result = [CDVPluginResult resultWithStatus:CDVCommandStatus_OK
- messageAsArray:notifications];
- [self.commandDelegate sendPluginResult:result
- callbackId:command.callbackId];
- }];
- }
- /**
- * Inform if the app has the permission to show
- * badges and local notifications.
- */
- - (void) hasPermission:(CDVInvokedUrlCommand*)command
- {
- [self.commandDelegate runInBackground:^{
- CDVPluginResult* result;
- BOOL hasPermission;
- hasPermission = [self.app hasPermissionToScheduleLocalNotifications];
- result = [CDVPluginResult resultWithStatus:CDVCommandStatus_OK
- messageAsBool:hasPermission];
- [self.commandDelegate sendPluginResult:result
- callbackId:command.callbackId];
- }];
- }
- /**
- * Ask for permission to show badges.
- */
- - (void) registerPermission:(CDVInvokedUrlCommand*)command
- {
- if ([[UIApplication sharedApplication]
- respondsToSelector:@selector(registerUserNotificationSettings:)])
- {
- _command = command;
- [self.commandDelegate runInBackground:^{
- [self.app registerPermissionToScheduleLocalNotifications];
- }];
- } else {
- [self hasPermission:command];
- }
- }
- #pragma mark -
- #pragma mark Core Logic
- /**
- * Schedule the local notification.
- */
- - (void) scheduleLocalNotification:(UILocalNotification*)notification
- {
- [self cancelForerunnerLocalNotification:notification];
- [self.app scheduleLocalNotification:notification];
- }
- /**
- * Update the local notification.
- */
- - (void) updateLocalNotification:(UILocalNotification*)notification
- withOptions:(NSDictionary*)newOptions
- {
- NSMutableDictionary* options = [notification.userInfo mutableCopy];
- [options addEntriesFromDictionary:newOptions];
- [options setObject:[NSDate date] forKey:@"updatedAt"];
- notification = [[UILocalNotification alloc]
- initWithOptions:options];
- [self scheduleLocalNotification:notification];
- }
- /**
- * Cancel all local notifications.
- */
- - (void) cancelAllLocalNotifications
- {
- [self.app cancelAllLocalNotifications];
- [self.app setApplicationIconBadgeNumber:0];
- }
- /**
- * Clear all local notifications.
- */
- - (void) clearAllLocalNotifications
- {
- [self.app clearAllLocalNotifications];
- [self.app setApplicationIconBadgeNumber:0];
- }
- /**
- * Cancel a maybe given forerunner with the same ID.
- */
- - (void) cancelForerunnerLocalNotification:(UILocalNotification*)notification
- {
- NSString* id = notification.options.id;
- UILocalNotification* forerunner;
- forerunner = [self.app localNotificationWithId:id];
- if (!forerunner)
- return;
- [self.app cancelLocalNotification:forerunner];
- }
- /**
- * Cancels all non-repeating local notification older then
- * a specific amount of seconds
- */
- - (void) cancelAllNotificationsWhichAreOlderThen:(float)seconds
- {
- NSArray* notifications;
- notifications = [self.app localNotifications];
- for (UILocalNotification* notification in notifications)
- {
- if (![notification isRepeating]
- && notification.timeIntervalSinceFireDate > seconds)
- {
- [self.app cancelLocalNotification:notification];
- [self fireEvent:@"cancel" notification:notification];
- }
- }
- }
- #pragma mark -
- #pragma mark Delegates
- /**
- * Calls the cancel or trigger event after a local notification was received.
- * Cancels the local notification if autoCancel was set to true.
- */
- - (void) didReceiveLocalNotification:(NSNotification*)localNotification
- {
- UILocalNotification* notification = [localNotification object];
- if ([notification wasUpdated])
- return;
- NSTimeInterval timeInterval = [notification timeIntervalSinceLastTrigger];
- NSString* event = (timeInterval <= 1 && deviceready) ? @"trigger" : @"click";
- [self fireEvent:event notification:notification];
- if (![event isEqualToString:@"click"])
- return;
- if ([notification isRepeating]) {
- [self fireEvent:@"clear" notification:notification];
- } else {
- [self.app cancelLocalNotification:notification];
- [self fireEvent:@"cancel" notification:notification];
- }
- }
- /**
- * Called when app has started
- * (by clicking on a local notification).
- */
- - (void) didFinishLaunchingWithOptions:(NSNotification*)notification
- {
- NSDictionary* launchOptions = [notification userInfo];
- UILocalNotification* localNotification;
- localNotification = [launchOptions objectForKey:
- UIApplicationLaunchOptionsLocalNotificationKey];
- if (localNotification) {
- [self didReceiveLocalNotification:
- [NSNotification notificationWithName:CDVLocalNotification
- object:localNotification]];
- }
- }
- /**
- * Called on otification settings registration is completed.
- */
- - (void) didRegisterUserNotificationSettings:(UIUserNotificationSettings*)settings
- {
- if (_command)
- {
- [self hasPermission:_command];
- _command = NULL;
- }
- }
- #pragma mark -
- #pragma mark Life Cycle
- /**
- * Registers obervers after plugin was initialized.
- */
- - (void) pluginInitialize
- {
- NSNotificationCenter* center = [NSNotificationCenter
- defaultCenter];
- eventQueue = [[NSMutableArray alloc] init];
- [center addObserver:self
- selector:@selector(didReceiveLocalNotification:)
- name:CDVLocalNotification
- object:nil];
- [center addObserver:self
- selector:@selector(didFinishLaunchingWithOptions:)
- name:UIApplicationDidFinishLaunchingNotification
- object:nil];
- [center addObserver:self
- selector:@selector(didRegisterUserNotificationSettings:)
- name:UIApplicationRegisterUserNotificationSettings
- object:nil];
- }
- /**
- * Clears all single repeating notifications which are older then 5 days
- * before the app terminates.
- */
- - (void) onAppTerminate
- {
- [self cancelAllNotificationsWhichAreOlderThen:432000];
- }
- #pragma mark -
- #pragma mark Helper
- /**
- * Retrieves the application state
- *
- * @return
- * Either "background" or "foreground"
- */
- - (NSString*) applicationState
- {
- UIApplicationState state = [self.app applicationState];
- bool isActive = state == UIApplicationStateActive;
- return isActive ? @"foreground" : @"background";
- }
- /**
- * Simply invokes the callback without any parameter.
- */
- - (void) execCallback:(CDVInvokedUrlCommand*)command
- {
- CDVPluginResult *result = [CDVPluginResult
- resultWithStatus:CDVCommandStatus_OK];
- [self.commandDelegate sendPluginResult:result
- callbackId:command.callbackId];
- }
- /**
- * Short hand for shared application instance.
- */
- - (UIApplication*) app
- {
- return [UIApplication sharedApplication];
- }
- /**
- * Fire general event.
- */
- - (void) fireEvent:(NSString*)event
- {
- [self fireEvent:event notification:NULL];
- }
- /**
- * Fire event for local notification.
- */
- - (void) fireEvent:(NSString*)event notification:(UILocalNotification*)notification
- {
- NSString* js;
- NSString* params = [NSString stringWithFormat:
- @"\"%@\"", self.applicationState];
- if (notification) {
- NSString* args = [notification encodeToJSON];
- params = [NSString stringWithFormat:
- @"%@,'%@'",
- args, self.applicationState];
- }
- js = [NSString stringWithFormat:
- @"cordova.plugins.notification.local.fireEvent('%@', %@)",
- event, params];
- if (deviceready) {
- [self.commandDelegate evalJs:js];
- } else {
- [self.eventQueue addObject:js];
- }
- }
- @end
|