我如何在我的iOS
应用程序中支持Universal Links并设置我的服务器以支持通用链接?
答案 0 :(得分:76)
当您支持通用链接时,iOS 9用户可以点击指向您网站的链接,无需通过Safari即可无缝重定向到已安装的应用。如果您的应用未安装,点击指向您网站的链接即可在Safari中打开您的网站。
此处,如何设置您自己的服务器,并处理您应用中的相应链接。
您需要让服务器在线运行。为了安全地将您的iOS应用与服务器关联,Apple要求您提供名为apple-app-site-association
的配置文件。这是一个JSON
文件,用于描述域和支持的路由。
apple-app-site-association
文件需要通过HTTPS
访问,无需任何重定向,位于 https:// {domain} / apple-app-site-association 。
该文件如下所示:
{
"applinks": {
"apps": [ ],
"details": [
{
"appID": "{app_prefix}.{app_identifier}",
"paths": [ "/path/to/content", "/path/to/other/*", "NOT /path/to/exclude" ]
},
{
"appID": "TeamID.BundleID2",
"paths": [ "*" ]
}
]
}
}
注意 - 请勿将.json
附加到apple-app-site-association
文件名。
密钥如下:
apps
:应该有一个空数组作为其值,并且它必须存在。这就是Apple想要它的方式
details
:是一个字典数组,一个用于网站支持的每个iOS应用程序。每个字典都包含有关应用程序,团队和软件包ID的信息。
有三种方法可以定义路径:
Static
:整个支持的路径都经过硬编码,以识别特定的链接,例如: /静态/项
Wildcards
:A *可用于匹配动态路径,例如/ books / *可以匹配任何作者页面的路径。 ?特定路径组件内部,例如,书籍/ 1?可用于匹配ID以1开头的任何书籍
Exclusions
:在NOT前面添加路径会排除该路径的匹配。
数组中提到路径的顺序很重要。早期指数具有更高的优先级。路径匹配后,评估将停止,其他路径将被忽略。每条路径都区分大小写。
应用程序支持的每个域都需要提供自己的apple-app-site-association文件。如果每个域提供的内容不同,则文件的内容也将更改以支持相应的路径。否则,可以使用相同的文件,但需要在每个支持的域中访问它。
注意:如果您的服务器使用HTTPS
来提供内容并跳转到应用程序设置指南,则可以跳过此部分。
如果您的应用定位到iOS 9且您的服务器使用HTTPS
来投放内容,则无需对该文件进行签名。如果不是(例如,在iOS 8上支持Handoff时),则必须使用来自认可的证书颁发机构的SSL
证书对其进行签名。
注意:这不是Apple提供的将应用程序提交到App Store的证书。它应该由第三方提供,并且建议使用您用于HTTPS
服务器的相同证书(尽管不是必需的)。
要对文件进行签名,请先创建并保存简单的.txt版本。接下来,在终端中,运行以下命令:
cat <unsigned_file>.txt | openssl smime -sign -inkey example.com.key -signer example.com.pem -certfile intermediate.pem -noattr -nodetach -outform DER > apple-app-site-association
这将在当前目录中输出签名文件。 example.com.key
,example.com.pem
和intermediate.pem
是您的认证机构可以使用的文件。
注意:如果文件未签名,则其Content-Type
应为application/json
。否则,它应该是application/pkcs7-mime
。
使用Apple App搜索验证工具验证您的服务器
测试您的网页是否有iOS 9搜索API。输入网址,Applebot会抓取您的网页,并展示如何优化以获得最佳效果
https://search.developer.apple.com/appsearch-validation-tool/
网站代码可以在gh-pages分支上找到 https://github.com/vineetchoudhary/iOS-Universal-Links/tree/gh-pages
应用程序将以iOS 9为目标,并将使用带有Objective-C
的Xcode 7.2。
应用程序端的设置需要两件事:
1.配置应用程序的权利,并启用通用链接
2.处理AppDelegate
。
<强> 1。配置应用程序的权利,并启用通用链接。
配置应用程序权利的第一步是为您的应用程序ID启用它。在Apple Developer Member Center中执行此操作。点击证书,标识符和配置文件,然后是标识符。选择您的应用程序ID(如果需要,首先创建它),单击编辑并启用关联域权利。
接下来,通过单击相应的应用程序ID获取App ID前缀和后缀。
App ID前缀和后缀应与apple-app-site-association文件中的前缀匹配。
接下来在Xcode
中,选择您的应用目标,点击功能并将关联域切换为开。为您的应用支持的每个域添加一个条目,前缀为applinks:。
例如:applinks:vineetchoudhary.github.io
示例应用看起来像这样:
注意:确保您选择了相同的团队,并在会员中心输入了与注册的App ID相同的Bundle ID。同时通过选择文件确保Xcode包含权利文件,并在文件检查器中确保检查目标。
[UIApplicationDelegate application: continueUserActivity: restorationHandler:]
中的 AppDelegate.m
方法处理传入的链接。您解析此URL以确定应用程序中的正确操作。
例如,在示例应用中:
-(BOOL)application:(UIApplication *)application continueUserActivity:(NSUserActivity *)userActivity restorationHandler:(void (^)(NSArray * _Nullable))restorationHandler{
if ([userActivity.activityType isEqualToString: NSUserActivityTypeBrowsingWeb]) {
NSURL *url = userActivity.webpageURL;
UIStoryboard *storyBoard = [UIStoryboard storyboardWithName:@"Main" bundle:nil];
UINavigationController *navigationController = (UINavigationController *)_window.rootViewController;
if ([url.pathComponents containsObject:@"home"]) {
[navigationController pushViewController:[storyBoard instantiateViewControllerWithIdentifier:@"HomeScreenId"] animated:YES];
}else if ([url.pathComponents containsObject:@"about"]){
[navigationController pushViewController:[storyBoard instantiateViewControllerWithIdentifier:@"AboutScreenId"] animated:YES];
}
}
return YES;
}
func application(application: UIApplication, continueUserActivity userActivity: NSUserActivity, restorationHandler: ([AnyObject]?) -> Void) -> Bool {
if userActivity.activityType == NSUserActivityTypeBrowsingWeb {
let url = userActivity.webpageURL!
//handle url
}
return true
}
应用程序代码可以在master分支上找到 https://github.com/vineetchoudhary/iOS-Universal-Links/
答案 1 :(得分:1)
如果您完成了Vineet的所有帖子但它仍然无效,请尝试使用分发配置文件。我从上面的帖子中做了所有事情,但必须发布分发配置文件才能使其正常工作。
使用开发配置文件,应用程序成功下载了AASA文件,但在我按下链接时它从未打开我的应用程序。
希望它有所帮助。
答案 2 :(得分:0)
以下是基于the answer by Vineet Choudhary的处理Swift 3+中通用链接的代码:
func application(_ application: UIApplication, continue userActivity: NSUserActivity, restorationHandler: @escaping ([UIUserActivityRestoring]?) -> Void) -> Bool {
if userActivity.activityType == NSUserActivityTypeBrowsingWeb,
let url = userActivity.webpageURL {
//handle URL
}
return true
}