实现功能:
1. 点击demo1屏幕,跳转到demo2的默认Page_01(橙色)界面
2. 点击demo1屏幕,跳转到demo2时传递参数,通过参数判断跳转到demo2的Page_02(绿色)界面,再从demo2跳回到demo1
模拟创建两个工程demo1和demo2
demo1简单UI搭建:
demo2简单UI搭建:
1.简单实现应用间跳转
从demo1应用跳转到demo2应用,首先需要给demo2应用创建URL Scheme:
设置路径: target - info - URL types下 ,设置自己的Url scheme
这里只是为了演示,并没有真实应用上线的ID,所以随便命名
设置好后,既然demo1需要跳转到demo2,demo1应用就需要知道demo2的URL scheme,通过URL scheme实现跳转
在demo1的touchBegan方法中
- (void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event{
// 进行应用跳转 URL Scheme格式 : scheme://
[[UIApplication sharedApplication] openURL:[NSURL URLWithString:@"demo2://"]];
}
就可以根据设置的URL scheme实现跳转:
点击屏幕,弹出提示:
点击打开,跳转到demo2:
但如果需要跳转到另外一个应用时,需要我们判断此设备上是否安装要跳转的目标应用,如果有才执行跳转操作,如果没有执行其他操作:
- (void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event{
// 判断系统上是否安装该App
if ([[UIApplication sharedApplication] canOpenURL:[NSURL URLWithString:@"demo2://"]]) {
// 如果安装 --> 跳转
// 进行应用跳转 URL Scheme格式 : scheme://
[[UIApplication sharedApplication] openURL:[NSURL URLWithString:@"demo2://"]];
}else{
// 如果未安装 --> 提示或到AppStore下载页等
NSLog(@"未安装此应用...");
}
}
加上判断后运行,会发现点击后无法实现跳转,控制台提示:
2016-07-13 12:39:29.480 demo1[1132:126924] -canOpenURL: failed for URL: "demo2://" - error: "This app is not allowed to query for scheme demo2"
2016-07-13 12:39:29.480 demo1[1132:126924] 未安装此应用...
是因为iOS进行应用跳转时,如果使用了canOpenURL进行判断,就需要设置白名单,只有设置了URL Scheme白名单,才有可能返回YES
设置白名单,在info.plist中设置对应的key: LSApplicationQueriesSchemes
<key>LSApplicationQueriesSchemes</key>
<array>
<!-- 目标应用 URL Scheme 白名单-->
<string>替换成目标app的URL Scheme</string>
</array>
这样就可以了
补充:
1).因为URL Scheme并不是唯一的,一个app也可以对应多个URL Scheme,所以会存在URL Scheme劫持的隐患
2).为了防止应用轻易获取用户隐私(通过canOpenURL遍历应用列表,就知道了用户安装的app列表,就一定程度上了解了用户的隐私),所以iOS9开始加入了白名单的限制(白名单有个数限制,最多90条)
2.跳转并传递参数
从demo1跳转到demo2后还需要返回到demo1应用,而demo2作为目标应用,并不知道谁可能会跳转到自己,所以实际使用中,不能像demo1定向跳转到demo2那样指定URL Scheme,这是就需要利用跳转传参的方式,在demo1跳转到demo2的时候,将demo1自己的URL Scheme传递给demo2,在demo2返回demo1时使用
接下来,先给demo1设置URL Scheme:
这时在跳转demo2的时候,就需要修改一下代码:
将自己的URL Scheme作为参数传递给demo2
- (void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event{
// 判断系统上是否安装该App
if ([[UIApplication sharedApplication] canOpenURL:[NSURL URLWithString:@"demo2://"]]) {
// 如果安装 --> 跳转
// 1. 进行应用跳转 URL Scheme格式 : scheme://
// [[UIApplication sharedApplication] openURL:[NSURL URLWithString:@"demo2://"]];
// 2. URL Scheme格式 : scheme://参数
[[UIApplication sharedApplication] openURL:[NSURL URLWithString:@"demo2://demo1"]];
}else{
// 如果未安装 --> 提示或到AppStore下载页等
NSLog(@"未安装此应用...");
}
}
demo2中:
/**
* 当其他APP通过应用跳转打开该APP时调用
*
* @param app 应用对象
* @param url 跳转打开该APP的url
* @param options 选项
*
* @return 是否允许打开该APP
*/
- (BOOL)application:(UIApplication *)app openURL:(NSURL *)url options:(NSDictionary<NSString *,id> *)options{
NSString *urlString = url.absoluteString;
NSLog(@"%@",urlString);
return YES;
}
因为需要从demo2工程的控制台中看打印信息,所以要保证demo2运行,先运行demo2,然后运行demo1,点击demo1屏幕,跳转到demo2,控制台打印结果:
2016-07-13 13:23:14.504 demo2[1218:164622] demo2://demo1
这样,就拿到了传递的参数,也就是demo1的URL Scheme
避免从demo1跳转到demo2后直接跳回demo1,这里加了一个延迟执行
- (BOOL)application:(UIApplication *)app openURL:(NSURL *)url options:(NSDictionary<NSString *,id> *)options{
NSString *urlString = url.absoluteString;
NSRange range = [urlString rangeOfString:@"://"];
// 截取demo1的Scheme
NSString *demo1Scheme = [urlString substringFromIndex:range.location + range.length];
dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(2 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
[[UIApplication sharedApplication] openURL:[NSURL URLWithString:[NSString stringWithFormat:@"%@://",demo1Scheme]]];
});
return YES;
}
这样从demo1跳转到demo2,再跳回的功能就实现了,接下来,当跳转到demo2时,设置demo2显示界面:
因为根控制器是一个TabBarController,跳转时设置了索引即可:
- (BOOL)application:(UIApplication *)app openURL:(NSURL *)url options:(NSDictionary<NSString *,id> *)options{
// 设置界面
UITabBarController *tabBar = (UITabBarController *)self.window.rootViewController;
tabBar.selectedIndex = 1;
return YES;
}