1.如果是应用未启动
则可以通过
- (BOOL)application:(UIApplication*)application didFinishLaunchingWithOptions:(NSDictionary*)launchOptions
方法来判断是点击图标启动应用还是点击推送消息启动应用,如果是点击推送启动应用的话,上面的那个方法的launchOptions必不为nil,可以通过是否为nil进行判断;然后可以通过
if(launchOptions) {
NSDictionary* remoteNotification = [launchOptionsobjectForKey:UIApplicationLaunchOptionsRemoteNotificationKey];
if(remoteNotification) {
NSLog(@"推送过来的消息是%@",remoteNotification);
//点击推送通知进入指定界面(这个肯定是相当于从后台进入的)
[self goToMssageViewControllerWith:remoteNotification];//进入相应页面的方法
}
}
返回的词典remoteNotification就是推送的消息主体,可以在此进行打印查看,然后判断是点击推送消息启动的应用之后,在goToMssageViewControllerWith:的方法里面写进入自己消息对应页面的代码(下面再讲)
2.如果应用已启动,挂在后台
这种情况下点击推送的消息进入应用时会调用
- (void)application:(UIApplication*)application didReceiveRemoteNotification:(NSDictionary*)userInfo
userInfo就是推送的消息的主体
在这个方法中可以进行自己进入消息对应页面的代码操作
3.如果推送消息到达时应用就在前台运行
这种情况依然会调用2.中的方法,但是如果你什么都没设置,用户就不会知道你推送了消息,如果你设置了进入对应页面,则应用会一下子很突然的进入对应界面,体验不好;这个时候就需要判断应用是在前台运行还是挂在后台(如果程序未启动就不用考虑这种情况)
if([UIApplicationsharedApplication].applicationState==UIApplicationStateActive)
{//前台运行时,收到推送的通知会弹出alertview提醒
NSDictionary*oneDict = [userInfoobjectForKey:@"aps"];
NSDictionary*twoDict = [oneDictobjectForKey:@"alert"];
NSString*msg = [twoDictobjectForKey:@"body"];
UIAlertView*alert = [[UIAlertViewalloc]initWithTitle:@"温馨提示"message:msgdelegate:selfcancelButtonTitle:nilotherButtonTitles:@"确定",nil];
[alertshow];
}
else if ([UIApplicationsharedApplication].applicationState==UIApplicationStateInactive)
{//点击推送通知进入界面的时候
[self goToMssageViewControllerWith:userInfo];
}
UIApplicationsharedApplication].applicationState有三个状态,分别是UIApplicationStateActive//应用正在前台运行
UIApplicationStateInactive//点击推送的通知进入应用
UIApplicationStateBackground//应用在后台挂起
这里面只须判断应用是在前台运行(2.)还是点击推送通知进入应用(3.)了,然后再进行相应的处理,我项目中所用的是如果在前台运行,推送消息来了就弹窗告知,不过感觉不是很好,读者可用自己方法,也可以用第三方MBProgressHUD中的那个浮现两三秒自动消失的控件来操作;如果是点击推送通知进入应用,我还是用的那个进入相应界面的方法
4.如何进入推送消息相应的界面
重头戏来了额,做到这块的时候我在网上搜了好多,但并没什么很多值得参考的资料,最后的话在cocoaChina论坛中找到一个值得参考的http://www.cocoachina.com/bbs/read.php?tid=257582,里面的8楼说的很有道理,我也是基于此做的;
首先关于推送的消息这块,比如你推送的消息分为三类,一类是点击消息进入A页面,一类是点击进入B页面,一类是点击进入C页面,你可以在推送的消息主体中设置一个事件值EventID,定义101就是关于A页面的消息,102就是关于B页面的消息,103就是关于C页面的消息;然后接收到消息主体后解析出来进行判断,可以使用switch case进行判断,
NSIntegerEventID = [[msgDic objectForKey:@"eventId"]integerValue];
switch(EventID) {
case101: {//写进入A页面的代码
break;
}
case102: {//写进入B页面的代码
break;
}
case103: {//写进入C页面的代码
break;
}
default:
break;
}
然后是进入相应的界面,我这里做的处理是已知消息对应的界面之后,在appdelegate文件中包含此界面的头文件,然后在对应的位置处进行初始化,然后present过去
case101: {
//进入A界面
AViewController *Avc = [[AViewController alloc] init];
[self.window.rootViewControllerpresentViewController:Avc animated:YEScompletion: nil ];
break;
}
这里的话要说明几点
第一、以上代码是只是针对与不带导航栏的简单页面,并且A页面与其上级界面并没有数据传递,也就是说,A页面的初始化并没有用到它上一级页面的数据
第二、很明显,大家想要进入的页面没有像A这么简单的;就像我这次,有一类是进入带有导航栏的页面;有一类是进入tabbar的其中一个标签页面
A.如何进入带有导航栏的页面
对于这个问题我也找了好久,试探了好几种办法,比如使用根视图的导航控制器进行push,均不行,最后使用的就是之前参考那个论坛上的方法;
case101: {
//进入带有导航栏的A页面
AViewController*Avc = [[AViewControlleralloc]init];
UINavigationController*planNav = [[UINavigationControlleralloc]initWithRootViewController:Avc];
[self.window.rootViewController presentViewController:planNav animated:YES completion: nil ];
break;
}
使用把A页面装入导航控制器中,然后present导航控制器就好了,这样点击推送消息就可以进入带有导航控制器的页面了;如果A页面是从上个界面push过来的,并且初始化的时候使用到了上级界面的一些数据,这种情况的话,就需要让推送消息把这些数据一并推送过来,然后进行解析,在上面进行初始化的时候使用这些数据进行初始化,相当于断绝它与上级界面的关系;剩下的还有一点就是导航栏上的返回键了,因为是present过去的,所以系统导航栏自带的backBarButtonItem并不会出现,即使你在那个present方法最后一个block参数中设置也不行;这里的话用的是之前说的那个论坛哥们说的方法,在viewwillappear方法里面进行判断
- (void)viewWillAppear:(BOOL)animated {
//判断是否是点击推送过来的,如果是的话设置左导航标签为返回键
NSUserDefaults*pushJudge = [NSUserDefaultsstandardUserDefaults];
if([[pushJudgeobjectForKey:@"push"]isEqualToString:@"push"]) {
//给导航栏加一个返回按钮,便于将推送进入的页面返回出去,如果不是推送进入该页面,那肯定是通过导航栏进入的,则页面导航栏肯定会有导航栏自带的leftBarButtonItem返回上一个页面
UIBarButtonItem*leftButton = [[UIBarButtonItem alloc] initWithBarButtonSystemItem:UIBarButtonSystemItemReplytarget:self action:@selector(rebackToRootViewAction)];
self.navigationItem.leftBarButtonItem= leftButton;
}else{
self.navigationItem.leftBarButtonItem=nil;
}
}
- (void)rebackToRootViewAction {
//将标示条件置空,以防通过正常情况下导航栏进入该页面时无法返回上一级页面
NSUserDefaults*pushJudge = [NSUserDefaults standardUserDefaults];
[pushJudge setObject:@""forKey:@"push"];
[pushJudge synchronize];
[self dismissViewControllerAnimated:YES completion:nil];
}
就是在本地存储NSUserDefaults中存储用来判断是否是点击推送进入页面的,这个判断值是在goToMssageViewControllerWith:(上面找)方法中存入的,即进入相应页面之前先设置这个值,
- (void)goToMssageViewControllerWith:(NSDictionary*)msgDic
{
//将字段存入本地
NSUserDefaults*pushJudge = [NSUserDefaults standardUserDefaults];
[pushJudge setObject:@"push"forKey:@"push"];
[pushJudge synchronize];
NSIntegerEventID = [[msgDic objectForKey:@"eventId"] integerValue];
switch(EventID) {
...
}
}
这样就能达到基本一样的相应页了,点击返回的话是返回之前应用退入后台时的那个页面
B.如何进入tabbar的其中一个标签页面
这个其实很简单的,因为这个肯定不会跟上级页面有什么关联,我用的方法是重新把所有的标签页面存入一个tabbar中,然后present过去就好了;
RootTabBarController*tabBarController = [RootTabBarController new];
UINavigationController*AAController = [[UINavigationController alloc] initWithRootViewController:[AAViewController new]];
UINavigationController*BBController = [[UINavigationController alloc] initWithRootViewController:[BBViewController new]];
UINavigationController*CCController = [[UINavigationController alloc] initWithRootViewController:[CCViewController new]];
NSArray*array_controllers = [NSArray arrayWithObjects:AAController,BBController,CCController,nil];
tabBarController.viewControllers= array_controllers;
tabBarController.selectedViewController= [tabBarController.viewControllersobjectAtIndex:2];
[tabBarControllersetSelBtn:2];
[self.window.rootViewControllerpresentViewController:tabBarControlleranimated:YEScompletion:nil];