兼容 iOS 9 Safari 的应用跳转方案探索
- - keakon的涂鸦馆很多做 web 开发的一定遇到过这种需求:点一个链接或按钮时,如果装了应用,就用该应用打开;没装的时候,iOS 跳 App Store 下载,Android 直接下载 apk 包. 在做读读日报的时候,就被这玩意花费了好长时间;然而 iOS 9 发布后,方案又失效了,于是又折腾了我几个小时. 首先做个科普,浏览器是无法知道应用有没有安装的.
<a href="下载地址">下载或打开 app</a>
<script>
$('a').click(function() {
var ifr = document.createElement('iframe');
ifr.src = '自定义 URL scheme';
ifr.style.display = 'none';
document.body.appendChild(ifr);
setTimeout(function(){
document.body.removeChild(ifr);
}, 3000);
});
</script>
这样在点击这个 a 标签时,会先尝试打开自定义 URL scheme。如果成功,则跳到应用里去了;如果失败,则跳转到 href 属性,即下载页。 $('a').click(function() {
location.href = '自定义 URL scheme';
t = Date.now();
setTimeout(function(){
if (Date.now() - t < 1100) {
location.href = 'Android 下载地址';
}
}, 1000);
return false;
}
这里是让浏览器尝试打开自定义 URL scheme,并且忽略浏览器默认行为(跳转到 href 属性)。等待一秒后,再检查当前时间,如果超过 1100 毫秒,说明跳转 app 成功了(跳转 app 会让浏览器的定时器变慢),什么也不用干;如果没超过 1100 毫秒,很可能是没有安装应用,就跳到下载地址。 $('a').click(function() {
location.href = '自定义 URL scheme';
location.href = '下载页';
location.reload();
}
其中,下载页是一个 HTML 页面,用 JavaScript 延时 2 秒跳转到 App Store。如果直接在 HTTP 头里用 Location 跳转到 App Store,则会立刻跳到 App Store,没机会跳应用。 $('a').click(function() {
location.href = '自定义 URL scheme';
setTimeout(function() {
location.href = '下载页';
}, 250);
setTimeout(function() {
location.reload();
}, 1000);
}
如果想尝试修改这两个值,可以自己试试,失败了不要找我。它的原理大概是跳应用需要一个准备时间,在这个期间内都能被其他跳转打断。而如果跳应用失败,reload 并不会打断 App Store 的跳转。 setTimeout(function() {history.back()}, 2000);
也就是让它 2 秒后回到前一个页面,这样从应用返回时,就不会停留在下载页了。