注意:
UILocalNotification:本地通知,进行UI通知(iOS8以后也需要先注册通知授权)。
这两个类不在同一个框架中:
NSNotificationCenter在Foundation框架中
UILocalNotification在UIKit框架中
本地推送
iOS7:不需要授权
iOS8:以后需要授权
步骤:
1、通过在AppDelegate中判断版本注册通知授权
2、在需要的界面创建一个本地通知,设置通知的属性,最后又AppDelegate调用通知
在#import "AppDelegate.h"中
- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions {
if ([[UIDevice currentDevice].systemVersion doubleValue] >= 8.0) {
UIUserNotificationSettings *userNotification = [UIUserNotificationSettings settingsForTypes:UIUserNotificationTypeBadge | UIUserNotificationTypeSound | UIUserNotificationTypeAlert categories:nil ];
[application registerUserNotificationSettings:userNotification];
}
return YES;
}
在#import "ViewController.h"中
-(void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event
{
// 1.创建本地通知
UILocalNotification *localNote = [[UILocalNotification alloc] init];
// 2.设置本地通知的内容
// 2.1.设置通知发出的时间
localNote.fireDate = [NSDate dateWithTimeIntervalSinceNow:3.0];
// 2.2.设置通知的内容
localNote.alertBody = @"通知内容";
// 2.3.设置滑块的文字
localNote.alertAction = @"滑块文字";
// 2.4.决定alertAction是否生效
localNote.hasAction = NO;
// 2.5.设置点击通知的启动图片
localNote.alertLaunchImage = @"随便都可以";
// 2.6.设置alertTitle
localNote.alertTitle = @"123";
// 2.7.设置有通知时的音效
localNote.soundName = @"语音文件名";
// 2.8.设置应用程序图标右上角的数字
localNote.applicationIconBadgeNumber = 9;
// 2.9.设置额外信息
localNote.userInfo = @{@"type" : @1};
// 3.调用通知
[[UIApplication sharedApplication] scheduleLocalNotification:localNote];
}
Application代理方法的launchOptions属性
- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions ;
launchOptions:用户直接打开程序的时候这个参数是没有值的。
若由其他应用程序通过openURL:启动,则UIApplicationLaunchOptionsURLKey对应的对象为启动URL(NSURL),
UIApplicationLaunchOptionsSourceApplicationKey对应启动的源应用程序的bundle ID (NSString);
若由本地通知启动,则UIApplicationLaunchOptionsLocalNotificationKey对应的是为启动应用程序的的本地通知对象(UILocalNotification);
若由远程通知启动,则UIApplicationLaunchOptionsRemoteNotificationKey对应的是启动应用程序的的远程通知信息userInfo(NSDictionary)
根据通知跳转到相应的界面
1、在AppDelegate中实现方法监听通知,判断是否进入后台,根据通知设置的额外信息userInfo来打开相应的界面。
// 应用程序在进入前台,或者在前台的时候都会执行该方法
- (void)application:(UIApplication *)application didReceiveLocalNotification:(UILocalNotification *)notification
{
// 针对应用程序在后台的时候进行的跳转
if (application.applicationState == UIApplicationStateInactive) {
NSLog(@"进行界面的跳转");
NSLog(@"%@", notification.userInfo);
UIView *redView = [[UIView alloc] init];
redView.frame = CGRectMake(0, 0, 100, 100);
redView.backgroundColor = [UIColor redColor];
[self.window.rootViewController.view addSubview:redView];
}
}
2、当程序被杀死了,当再次点击进来是不会来到实现的代理方法,此时需要在程序加载完毕之后判断。
- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions {
// 界面的跳转(针对应用程序被杀死的状态下的跳转)
if (launchOptions[UIApplicationLaunchOptionsLocalNotificationKey]) {
// 跳转代码
UILabel *redView = [[UILabel alloc] init];
redView.frame = CGRectMake(0, 0, 200, 300);
redView.numberOfLines = 0;
redView.font = [UIFont systemFontOfSize:12.0];
redView.backgroundColor = [UIColor redColor];
redView.text = [NSString stringWithFormat:@"%@", launchOptions];
[self.window.rootViewController.view addSubview:redView];
}
return YES;
}
远程推送原理
一.什么是远程通知
概念:由服务器发送消息给用户弹出消息的通知(需要联网)
远程推送服务,又称为APNs(Apple Push Notification Services)
二.为什么需要远程通知
例子:淘宝最近双11搞活动,各种送红包,想告知用户.但是该用户不经常打包淘宝APP.淘宝如何通知该用户有最新的活动呢?
传统方式:只有用户打开了淘宝客户端,客户端向服务器请求是否有最新的活动,才能在APP中告知用户活动.
局限性:只要用户关闭了app,就无法跟app的服务器沟通,无法从服务器上获得最新的数据内容
远程通知的好处:不管用户打开还是关闭app,只要联网了,都能接收到服务器推送的远程通知
如何做远程通知
需要真机,配置证书。
以微信客户端为例
1、客户端将手机的唯一标识UDID和应用程序的唯一标识Bound Identifier发送给苹果的APNs服务器。
2、苹果的APNs服务器根据这两个唯一标识生成一个加密的唯一标识deviceToken,并放回给客服端。
3、在AppDelegate中实现相应的方法拦截,并将这个deviceToken和微信帐号ID发送给客户端的服务器,客户端的服务器保存(deviceToke和微信帐号对应)。
4、当别人需要发送消息给特定的微信帐号好友,微信服务器将对应的帐号的deviceToken和消息发送给苹果的APNs服务器。
5、苹果的APNs服务器根据对应的deviceToken发送推送通知给微信客户端。
6、根据推送的消息,监听用户的点击(类似本地通知)
代码
#import "AppDelegate.h"
@interface AppDelegate ()
@end
@implementation AppDelegate
- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions {
//但程序被杀死,点击推送通知进入程序来到这,根据不同iOS版本进行通知的授权与注册
if ([[UIDevice currentDevice].systemVersion doubleValue] >= 8.0) { //iOS8
UIUserNotificationSettings *settings = [UIUserNotificationSettings settingsForTypes:UIUserNotificationTypeAlert | UIUserNotificationTypeBadge | UIUserNotificationTypeSound categories:nil];
//注册通知授权
[application registerUserNotificationSettings:settings];
//注册通知
[application registerForRemoteNotifications];
} else { // iOS7
[application registerForRemoteNotificationTypes:UIRemoteNotificationTypeBadge | UIRemoteNotificationTypeNewsstandContentAvailability | UIRemoteNotificationTypeSound |UIRemoteNotificationTypeAlert];
}
if (launchOptions[UIApplicationLaunchOptionsRemoteNotificationKey]) {
// 跳转
}
return YES;
}
//拦截苹果APNs服务器返回的deviceToken
- (void)application:(UIApplication *)application didRegisterForRemoteNotificationsWithDeviceToken:(NSData *)deviceToken
{
// 将DeviceToken传给服务器
NSLog(@"%@", deviceToken.description);
}
//当程序从后台点击推送通知进入时会调用这个方法
- (void)application:(UIApplication *)application didReceiveRemoteNotification:(NSDictionary *)userInfo
{
NSLog(@"%@", userInfo);
}
//当程序进入后台,不点击推送消息,也会调用这个方法,但是发送的通知有固定的格式,1.需要打开后台模式 2.告诉系统是否有新内容的更新3、需要添加"content-available":"1";
- (void)application:(UIApplication *)application didReceiveRemoteNotification:(NSDictionary *)userInfo fetchCompletionHandler:(void (^)(UIBackgroundFetchResult))completionHandler
{
NSLog(@"11111111");
UIView *redView = [[UIView alloc] init];
redView.backgroundColor = [UIColor redColor];
redView.frame = CGRectMake(100, 100, 100, 100);
[self.window.rootViewController.view addSubview:redView];
// 1.需要打开后台模式 2.告诉系统是否有新内容的更新 3.发送的通知有固定的格式("content-available":"1")
completionHandler(UIBackgroundFetchResultNewData);
}
设置应用程序右上角显示提示数字的两种方法
方法一: [application setApplicationIconBadgeNumber:2]; (优先级高)
方法二: UILocalNotification *localNote = [[UILocalNotification alloc] init];
localNote.applicationIconBadgeNumber = 9;
远程通知证书配置
- .配置一个明确的APPID
- 选择明确的APPID,并且将远程通知功能选中
-
显示Push Notifications并非Enabled,而是Configurable.
- 需要配置对应的证书
二 证书的配置
- 在Certificates中配置证书
- 选择证书的类型(调试和发布都需要配置)
-
选择为哪一个APPID配置证书
其他步骤同真机调试和发布程序
-
配置完成后获得两个证书文件
配置描述文件
- 和真机描述文件完全一致
获取DeviceToken
1.在苹果的APNs服务器注册,以获取DeviceToken
通常在didFinishLaunchingWithOptions中添加如下代码进行注册
if ([UIDevice currentDevice].systemVersion.doubleValue >= 8.0) {
// 1.向用户请求可以给用户推送消息
UIUserNotificationSettings *settings = [UIUserNotificationSettings settingsForTypes:UIUserNotificationTypeBadge | UIUserNotificationTypeSound | UIUserNotificationTypeAlert categories:nil];
[application registerUserNotificationSettings:settings];
// 2.注册远程通知(拿到用户的DeviceToken)
[application registerForRemoteNotifications];
} else {
[application registerForRemoteNotificationTypes:UIRemoteNotificationTypeAlert | UIRemoteNotificationTypeBadge | UIRemoteNotificationTypeSound];
}
注册之后在另外一个代理方法中,拿到DeviceToken
- (void)application:(UIApplication *)application didRegisterForRemoteNotificationsWithDeviceToken:(NSData *)deviceToken
{
// 5e8cf393 9e950137 86ac8375 12185078 19eb3ebd 936777e1 f061caec a48cb236
// 将用户的用户名和deviceToken发送给服务器,让服务器进行保存备份即可
NSLog(@"%@", deviceToken);
}
- 将DeviceToken发送到服务器即可
测试远程通知
当前我们没有自己的服务器,如何测试?
可以使用一个第三方的Mac程序来测试:PushMeBaby
-
使用该程序需要修改一些内容
-
编译程序,报错的行注释掉
-
运动pushMeBaby程序
![屏幕快照 2016-06-05 下午1.00.42.png](http://upload-images.jianshu.io/upload_images/1965034-16133560df57e119.png?
imageMogr2/auto-orient/strip%7CimageView2/2/w/1240)
注意:填写的内容
填写推送给的DeviceToken
添加推送的内容:固定格式
{"aps":{"alert":"弹出的信息","badge":1,"sound":"声音","info":"额外信息"}}