...或者以其他方式发布通知?
是的。您可以使用触发通知(不一定是UILocalNotification
)的后台(启动)守护程序来完成这项工作。当通知向用户显示警报时,您的守护程序可以决定是否打开普通的 UI 应用程序。
构建启动守护程序。
这是我找到的最好的教程。启动守护程序在手机启动时启动,并一直作为非图形后台进程运行。从那里,您可以安排检查更新。(我有一个HelloDaemon
在方法中完成所有工作的类run:
):
int main(int argc, char *argv[]) {
@autoreleasepool {
HelloDaemon* daemon = [[HelloDaemon alloc] init];
// start a timer so that the process does not exit.
NSTimer* timer = [[NSTimer alloc] initWithFireDate: [NSDate date]
interval: 1.0
target: daemon
selector: @selector(run:)
userInfo: nil
repeats: NO];
NSRunLoop* runLoop = [NSRunLoop currentRunLoop];
[runLoop addTimer: timer forMode: NSDefaultRunLoopMode];
[runLoop run];
}
return 0;
}
守护进程可以NSTimer
正常使用,因此安排另一个计时器(在 内run:
)检查更新以随时下载。
从守护进程通知用户
如果守护进程决定应该通知用户,那么您可以:
1)打开完整的 UI 应用程序。
#include <dlfcn.h>
#define SBSERVPATH "/System/Library/PrivateFrameworks/SpringBoardServices.framework/SpringBoardServices"
-(void) openApp {
// the SpringboardServices.framework private framework can launch apps,
// so we open it dynamically and find SBSLaunchApplicationWithIdentifier()
void* sbServices = dlopen(SBSERVPATH, RTLD_LAZY);
int (*SBSLaunchApplicationWithIdentifier)(CFStringRef identifier, Boolean suspended) = dlsym(sbServices, "SBSLaunchApplicationWithIdentifier");
int result = SBSLaunchApplicationWithIdentifier(CFSTR("com.mycompany.AppName"), false);
dlclose(sbServices);
}
此代码需要com.apple.springboard.launchapplications
您的守护程序授权才能成功使用它。 有关添加权利的信息,请参见此处。您的守护程序可执行文件需要一个 entitlements.xml 文件,如下所示:
<!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
<plist version="1.0">
<dict>
<key>com.apple.springboard.launchapplications</key>
<true/>
</dict>
</plist>
2)从你的守护进程中显示一个简单的警报窗口,通知用户该事件,并提示他们打开 UI 应用程序
#include "CFUserNotification.h"
-(void) showAlert {
NSMutableDictionary* dict = [NSMutableDictionary dictionary];
[dict setObject: @"Alert!" forKey: (__bridge NSString*)kCFUserNotificationAlertHeaderKey];
[dict setObject: @"Updates Ready!" forKey: (__bridge NSString*)kCFUserNotificationAlertMessageKey];
[dict setObject: @"View" forKey:(__bridge NSString*)kCFUserNotificationDefaultButtonTitleKey];
[dict setObject: @"Cancel" forKey:(__bridge NSString*)kCFUserNotificationAlternateButtonTitleKey];
SInt32 error = 0;
CFUserNotificationRef alert =
CFUserNotificationCreate(NULL, 0, kCFUserNotificationPlainAlertLevel, &error, (__bridge CFDictionaryRef)dict);
CFOptionFlags response;
// we block, waiting for a response, for up to 10 seconds
if((error) || (CFUserNotificationReceiveResponse(alert, 10, &response))) {
NSLog(@"alert error or no user response after 10 seconds");
} else if((response & 0x3) == kCFUserNotificationAlternateResponse) {
// user clicked on Cancel ... just do nothing
NSLog(@"cancel");
} else if((response & 0x3) == kCFUserNotificationDefaultResponse) {
// user clicked on View ... so, open the UI App
NSLog(@"view");
[self openApp];
}
CFRelease(alert);
}
您需要一个CFUserNotification.h
标题才能按照我上面的方式使用代码。您可以通过谷歌搜索找到一个,或在此处查看一个。这个较旧的 wiki 文档还显示了一些有关CFUserNotification
从 iOS 应用程序使用的有用信息。
我从上面的 KennyTM 链接到的答案还显示了如何使警报弹出显示,即使设备被锁定。