【IOS开发高级系列】App间跳转专题

2023-10-16 11:34:38 浏览数 (1)

1 OpenUrl原理

        如果一个应用程序支持一些已知类型的URL,您就可以通过对应的URL模式和该程序进行通讯。然而,在大多数情况下,URL只是用于简单地启动一个应用程序并显示一些和调用方有关的信息。举例来说,对于一个用于管理地址信息的应用程序,您就可以在发送给它的URL中包含一个Maps程序可以处理的地址,以便显示相应的位置。这个级别的通讯为用户创造一个集成度高得多的环境,减少应用程序重新实现设备上其它程序已经实现的功能的必要性。

        苹果内置支持http、mailto、tel、和sms这些URL模式,还支持基于http的、指向Maps、YouTube、和iPod程序的URL。应用程序也可以自己注册定制的URL模式。您的应用程序可以和其它应用程序通讯,具体方法是用正确格式的内容创建一个NSURL对象,然后将它传给共享UIApplication对象openURL:方法。openURL:方法会启动注册接收该URL类型的应用程序,并将URL传给它。当用户最终退出该应用程序时,系统通常会重新启动您的应用程序,但并不总是这样。系统会考虑用户在URL处理程序中的动作及在用户看来返回您的应用程序是否合理,然后做出决定。

重要提示:如果您的URL类型包含的模式和苹果定义的一样,则启动的是苹果提供的程序,而不是您的程序。如果有多个第三方的应用程序注册处理同样的URL模式,则该类型的URL由哪个程序处理是没有定义的。

        如果您的应用程序定义了自己的URL模式,则应该实现对该模式进行处理的方法,具体信息在“实现定制的URL模式”部分中进行描述。有关系统支持的URL处理,包括如何处理URL的格式,请参见苹果的URL模式参考。

2 OpenUrl实现APP间调用

2.1 注册OpenURL协议

        在IOS中,实现一个应用启动另外一个应用,使用UIApplication的openURL:方法就可实现,这里以test跳到test02为例。(需要先创建这两个工程)

        首先被启动的应用需要向iPhone注册一个自定义URL协议。这是在info.plist文件进行的。

    1. 右键,选择“Add Row”

    2. Key值选择“URL types”

    3. 打开“Item 0″,然后为该key增加一个URL identifier。可以是任何值,但建议用“反域名”(例如 “com.fcplayer.test”)。

    4. 在“Item 0”下再加一行。

    5. 选择“URL Schemes” 作为Key。

    6. 输入你的URL协议名 (例如“test://” 应写做“test”)。如果有必要,你可以在这里加入多个协议。

    操作截图如下:

CFBundleURLTypes属性的键和值:

CFBundleURLName

        这是个字符串,表示URL类型的抽象名。为了确保其唯一性,建议您使用反向DNS风格的标识,比如com.acme.myscheme。这里提供的URL类型名是一个指向本地化字符串的键,该字符串位于本地化语言包子目录中的InfoPlist.strings文件中。本地化字符串是人类可识别的URL类型名称,用相应的语言来表示。

CFBundleURLSchemes

        这是个URL模式的数组,表示归属于这个URL类型的URL。每个模式都是一个字符串。属于指定URL类型的URL都带有它们的模式组件。

        您在对CFBundleURLTypes属性进行定义,从而注册带有定制模式的URL类型之后,可以通过下面的方式来进行测试:

    1、连编、安装、和运行您的应用程序。

    2、回到Home屏幕,启动Safari(在iPhone仿真器上,在菜单上选择Hardware > Home命令就可以回到Home屏幕)。

    3、在Safari的地址栏中,键入使用定制模式的URL。

    4、确认您的应用程序是否启动,以及应用程序委托是否收到application:handleOpenURL:消息。

2.2 访问自定义URL(在test02中)

        应用程序委托在application:handleOpenURL:方法中处理传递给应用程序的URL请求。如果您已经为自己的应用程序注册了定制的URL模式,则务必在委托中实现这个方法。

        基于定制模式的URL采用的协议是请求服务的应用程序能够理解的。URL中包含一些注册模式的应用程序期望得到的信息,这些信息是该程序在处理或响应URL请求时需要的。传递给application:handleOpenURL:方法的NSURL对象表示的是Cocoa Touch框架中的URL。NSURL遵循RFC 1808规范,该类中包含一些方法,用于返回RFC 1808定义的各个URL要素,包括用户名、密码、请求、片断、和参数字符串。与您注册的定制模式相对应的“协议”可以使用这些URL要素来传递各种信息。

        在程序清单1-2显示的application:handleOpenURL:方法实现中,传入的URL对象在其请求和片断部分带有具体应用程序的信息。应用程序委托抽出这些信息—在这个例子中,是指一个to-do任务的名称和到期日—并根据这些信息创建应用程序的模型对象。

    在主应用程序中通过访问自定义URL启动另外一个应用:(test已经安装,这段代码要写在另一个应用里面,比如test02)

2.3 自定义处理URL(在test中)

    有些时候我们除了启动还需向另外一个应用发送参数,这是也可以通过自定义的URL来实现,如:

test://

test://com.company.test

test://config=1&abar=2

    这时我们在被启动应用中就必须进行自定义处理,在AppDelegate中实现该消息(Cocos2d加在AppDelegate中),例如:

-(BOOL)application:(UIApplication *)application handleOpenURL:(NSURL*)url { 

    //Do something withthe url here 

}

    在程序清单1-2显示的application:handleOpenURL:方法实现中,传入的URL对象在其请求和片断部分带有具体应用程序的信息。应用程序委托抽出这些信息—在这个例子中,是指一个to-do任务的名称和到期日—并根据这些信息创建应用程序的模型对象。

程序清单1-2  处理基于定制模式的URL请求

- (BOOL)application:(UIApplication *)application handleOpenURL:(NSURL *)url {

if ([[url scheme] isEqualToString:@"todolist"]) {

        ToDoItem *item = [[ToDoItem alloc] init];

        NSString *taskName = [url query];

if (!taskName || ![self isValidTaskString:taskName]) { 

            // must have a task name

            [item release];

return NO;

        }

        taskName = [taskName stringByReplacingPercentEscapesUsingEncoding:NSUTF8StringEncoding];

        item.toDoTask = taskName;

        NSString *dateString = [url fragment];

if (!dateString || [dateString isEqualToString:@"today"]) {

            item.dateDue = [NSDate date];

        } else{

if(![self isValidDateString:dateString]) {

                [item release];

return NO;

            }

            // format: yyyymmddhhmm (24-hour clock)

            NSString *curStr = [dateString substringWithRange:NSMakeRange(0, 4)];

            NSInteger yeardigit = [curStr integerValue];

            curStr = [dateString substringWithRange:NSMakeRange(4, 2)];

            NSInteger monthdigit = [curStr integerValue];

            curStr = [dateString substringWithRange:NSMakeRange(6, 2)];

            NSInteger daydigit = [curStr integerValue];

            curStr = [dateString substringWithRange:NSMakeRange(8, 2)];

            NSInteger hourdigit = [curStr integerValue];

            curStr = [dateString substringWithRange:NSMakeRange(10, 2)];

            NSInteger minutedigit = [curStr integerValue];

            NSDateComponents *dateComps = [[NSDateComponents alloc] init];

            [dateComps setYear:yeardigit];

            [dateComps setMonth:monthdigit];

            [dateComps setDay:daydigit];        

            [dateComps setHour:hourdigit];

            [dateComps setMinute:minutedigit];

            NSCalendar *calendar = [NSCalendar currentCalendar];

            NSDate *itemDate = [calendar dateFromComponents:dateComps];

if(!itemDate) {

                [dateComps release];

                [item release];

return NO;

            }

            item.dateDue = itemDate;

            [dateComps release];

        }

        [(NSMutableArray *)self.list addObject:item];

        [item release];

return YES;

    }

returnNO;

3 跳转IOS自带的应用

3.1 跳转App Store方法

        在实际开发中,往往要推荐自己其他应用和推荐自己的收费软件,那么我们就需要在程序中直接连接到app store的相应页面。实际上的做法很简单,使用的还是UIApplication类的OpenURL方法: 

[[UIApplication sharedApplication] openURL:[NSURL URLWithString:@"程序的相应连接"]];

3.2 跳转其它系统应用的方法

1)调用 自带mail

[[UIApplication sharedApplication] openURL:[NSURLURLWithString: @"mailto://admin@hzlzh.com"]];

2)调用 电话phone

[[UIApplication sharedApplication] openURL: [NSURLURLWithString:@"tel://8008808888"]];

3)调用 SMS

[[UIApplication sharedApplication] openURL: [NSURLURLWithString: @"sms://800888"]];

4)调用自带 浏览器 safari

[[UIApplication sharedApplication] openURL: [NSURLURLWithString: @"http://www.hzlzh.com"]];

5)调用 Remote

[[UIApplication sharedApplication] openURL: [NSURLURLWithString: @"remote://fff"]];

上面是最基本的语句,没什么处理过程。

如:调用phone可以传递号码,调用SMS只能设定号码,不能初始化SMS内容。

4 基于Universal Links的App跳转

如何从微信浏览器中跳转到APP指定页面?

http://www.jianshu.com/p/738ac2b8865d

iOS 9学习系列:打通iOS 9的通用链接(UniversalLinks)

http://www.cocoachina.com/ios/20150902/13321.html

4.1 什么是Universal Links呢?

        Universal Links就是一个通用链接,iOS9以上的用户,可以通过点击这个链接无缝的重定向到一个app应用,而不需要通过safari打开跳转。如果用户没有安装这个app,则会在safari中打开这个链接指向的网页。

4.2 配置Universal Links

    1.创建一个名字叫做apple-app-site-association,包含固定格式的json文件  

    2.将这个文件上传到你的服务器,可以将这个文件放到服务器的根目录下,也可以放到.well-known这个子目录下。  

    3.配置app,然后在app里面添加代理方法.

4.2.1 配置流程

1.apple-app-site-association文件

{

  "applinks": {

       "apps": [],

       "details": [

           {

                "appID": "teamID.bundleId”,

                "paths": ["*"]

           }

       ]

    }

}

        创建一个包含上述格式的json文件,文件名字必须为apple-app-site-association,且不能带后缀名

4.2.2 appID

    appID 的 格式为teamID.bundleId形式。如何获取teamID呢?

    登录开发者网站 ,找到Membership选项卡。

    appID具体显示就是:xxxxxxxxxxx.com.shangxinpifa.app这种

4.2.3 paths

    paths配置,实际上就是限制哪些路径可以唤醒app,哪些路径不能唤醒app。格式如下:

"paths": [

    "/wwdc/news/", 

    "NOT /videos/wwdc/2010/*",

    "/videos/wwdc/201?/*"

]

    1.使用*配置,则整个网站都可以使用;

    2.使用特定的URL,例如/wwdc/news/来指定某一个特殊的链接;

    3.在特定URL后面添加,例如/videos/wwdc/2015/, 来指定网站的某一部分;

    4.除了使用*来匹配任意字符,你也可以使用?来匹配单个字符,你可以在路径当中结合这两个字符使用,例如 /foo/*/bar/201?/mypage;

4.2.4 配置apple-app-site-association.JSON文件

    因为主要是为了支持从微信文章中调回App,目前规划将此JSON文件配置在微信站点根目录下,为了支持后续扩展,也可以考虑采用请求拦截的方式来实现,代码如下:

4.2.5 验证apple-app-site-association文件

https://wxtest1.cmfchina.com/apple-app-site-association

    文件配置完成之后,将其上传到你的服务器根目录或者.well-known这个子目录下。

    1.确保使用https://域名.com/apple-app-site-association这个链接可以访问到。

    2.也可以使用苹果的验证网站,验证文件是否能被苹果请求到。如果是未上线的应用,使用验证网站时可能出现如下提示:

    出现该提示为apple-app-site-association文件配置正确。

    出现404错误码提示,则为apple-app-site-association文件未上传成功,或者使用 https://域名.com/apple-app-site-association路径无法访问。

4.2.6 appIDs配置

    进入开发者网站,找到你自己的bundleId,可以点击edit按钮,开启associate domains,并创建相应的provisioning Profiles,如下图:

    注意:配置完成之后,证书最好重新下载安装一遍

4.2.7 项目配置

    在项目的Capablities中开启Associated domains,如下图:

    注意domains可以添加多个,前缀必须为applinks:,applinks:后为你的服务器的域名。

4.2.8 代码接收UniversalLinks唤醒

    在appdelegate中实现上面这个方法,当使用Universal Links唤醒app时就执行这个方法。

4.2.9 验证配置

    快捷验证,在备忘录中输入https://yourdomain.com/goods/129893,长按这个链接,出现下图提示则配置成功。

4.2.10 证书申请

    阿里云提供了免费的ssl证书申请

    购买之后到控制台补全信息:

 补全信息后点击进度查看下一步的配置工作:

    按照步骤完成配置后,10分钟左右就会通过审核,服务器配置按照阿里云提供的文档继续操作即可。

4.4 常见问题

4.4.1 配置注意事项

(Good)UniversalLinks通用链接应用跳转总结以及坑

http://www.jianshu.com/p/16374288c976

一些前人踏过的坑:

·     Universal Links will notwork if you paste the link into the browser URL field.

·     Universal Links work with a user driven element click across domains. Example: if there is a Universal * Link on google.compointing to bnc.lt, it will open the app.

·      Universal Links will not work with a user driven  element click on the same domain. Example: if there is a Universal Link on google.com pointing to a different Universal Link on google.com, it will not open the app.

·      Universal Links cannot be triggered via Javascript (in window.onload or via a .click() call on an element), unless it is part of a user action.

    总的说来,就是自从9.3.X改版之后,通用链接不支持域内跳转了,跳转前后的两个domain必须是不同的,否则只会safari打开。

5  参考链接

IOS-应用之间调用

http://blog.csdn.net/likendsl/article/details/7553605

Apple URL Scheme Reference

https://developer.apple.com/library/ios/featuredarticles/iPhoneURLScheme_Reference/Introduction/Introduction.html

(Good)自定义URL Scheme完全指南

http://www.cocoachina.com/industry/20140522/8514.html

(Good)iOS App自定义URL Scheme设计

http://www.cocoachina.com/ios/20120529/4302.html

如何从微信浏览器中跳转到APP指定页面?

http://www.jianshu.com/p/738ac2b8865d

iOS 9学习系列:打通iOS 9的通用链接(UniversalLinks)

http://www.cocoachina.com/ios/20150902/13321.html

iOS 10 Universal Links(通用连接),从微信网页连接跳转到公司APP之实现流程详解--2016最新版

http://blog.csdn.net/kuangdacaikuang/article/details/52958052?locationNum=12&fps=1

(Good)UniversalLinks通用链接应用跳转总结以及坑

http://www.jianshu.com/p/16374288c976

0 人点赞