H5页面中唤起native app

时间:2022-11-18 19:18:03

现在各类app,分享出去的H5页面中,一般都会带着一个立即打开的按钮,如果本地安装了app,那么就直接唤起本地的app,如果没有安装,则跳转到下载。这是一个很正常的推广和导流量的策略,最近产品经理就提出了这样的一个需求,做一个像今日头条功能一样的带打开app的下载条。

实现这个功能,我们需要解决两个问题
1、js如何唤起本地app
2、js如何知道手机已经安装了对应的应用

js如何唤起本地app

既然是通过网页调用app,这个当然涉及到与app的通信。通过咨询ios和android的同事,ios与android都支持一种叫做schema协议的链接。这种协议的类似于我们熟悉的http协议,我们只要跟app协商好协议头,app通过拦截到这个协议头的请求就可以知道有网页要求调用。而对于js来说,我们这要像a标签的href一样来激活这个协议的链接就行了。

比如:

<a href="myapp://">调起app</a>


这种方式ios和android都可以共用

2、如何知道手机已经安装的对应的应用对于这个功能的实现,首先想到的是查询应用是否存在,但是这种方法显而易见是行不通的,比如说你在UC浏览器,微信中,我们无法主动的去查询我们系统中是否安装了该应用。所以这个判断是无法实现。所以我们就需要采用曲线救国的方式来实现。既然我们可以唤起app,那我们就可以忽略判断,直接唤起app,如果用户没有安装,我们做一个容错处理。

结合这个思路,我们基本可以得到我们的实现方案

var iframe = document.createElement('iframe');
var body = document.body;
iframe.style.display = "none";
ar timer = null;

var openapp = document.getElementById('openapp');
openapp.addEventListener('click', function() {
body.appendChild(iframe);
iframe.src = "appschema://";
timer = setTimeout(function() {
wondow.location.href = "download.html"; //容错的下载页面
}, 500);
}, false)

写完代码,做了测试,发现这样的实现有很多问题
1、微信无法调起。微信对于链接的跳转限制很严重,很多下载外链都引导到浏览器打开
2、调起app返回浏览器的时候,会跳转到下载页面,既然用户已经下载了app,再让页面跳转到下载页很不友好
3、ios9+的safari无法通过iframe跳转到其他页面

有问题就需要解决

1、对于微信或者QQ空间,在网上查找资料,如果是在应用宝上线的应用,应用宝提供了微下载来实现微信和QQ打开app,先跳转到应用宝的的下载链接,然后下载宝链接会判断打开对应的app,具体参考(http://wiki.open.qq.com/index.php?title=mobile/%E5%BA%94%E7%94%A8%E5%AE%9D%E5%BE%AE%E4%B8%8B%E8%BD%BD)但是应用宝的微下载有个问题,ios微信和QQ中无法打开对应的应用,只是会通过你应用宝配置的appstore下载链接跳转到对应的下载页面,再从appstore里面打开应用。所以这个问题还是不能完全解决,只能完美解决android的机器

2、针对问题2,网上有人通过监控页面的pagehide和visibilitychange方法来实现禁止跳转,具体的实现思路是监控页面是否隐藏,利用延时如果页面已经打开app,此时页面会是隐藏状态,触发页面的隐藏事件,clear延时事件,禁止跳转,不过这个方案会出现问题,有一些浏览器在app打开,离开浏览器之后,js事件不在执行,也就是此时无法监控的页面的隐藏,在返回页面的时候,js继续执行,但是事件监控的还是页面展示的状态,无法clear延时事件,所以该方式无法完美解决这个问题

后来找到了另外的解决方案,调起app需要唤起另外的进程,所以js的进程会挂起,导致前后有一个时间差,记录前后的事件差对比就可以判断是否调起了app了

iframe.src = "appschema://";
var timer = null,
t = Date.now();
timer = setTimeout(function() {
if (Date.now() - t > 1200) {
clearTimeout(timer);
return false;
}
}, 1000);

3、对于问题3,Apple为iOS 9发布了一个所谓的通用链接的深层链接特性,即Universal links。只要在app中授权好域名,在网页中只要打开对应域名链接,都会检测与域名绑定的app是否存在,如果存在,直接调起app,具体参考(http://*.com/questions/31891777/ios-9-safari-iframe-src-with-custom-url-scheme-not-working),并且该方法不会被微信拦截,可以在微信中使用,这样也就解决了我们在腾讯平台下ios无法通过微下载打开的问题

最后再来整理一下我们的思路
1、ios通过Universal links,针对ios9一下和以前版本没有实现Universal links,在绑定好的域名下做一个中间页,直接跳转到中间页
2、android分平台,如果是微信或者QQ(可以通过用户代理检测),直接通过微下载,其他浏览器,直接用schema协议

具体代码实现如下

var url = {
        open: 'duchuang://',
        down: 'http://a.app.qq.com/o/simple.jsp?pkgname=com.nayun.framework'
    },
    iframe = document.createElement('iframe');
iframe.style.cssText = 'display:none;width=0;height=0';
var timer = null,
    //点击第三方下载
    isAndroid = !!navigator.userAgent.match(/android/ig),
    isIos = !!navigator.userAgent.match(/iphone|ipod/ig),
    isIpad = !!navigator.userAgent.match(/ipad/ig),
    isWeixin = (/MicroMessenger/ig).test(navigator.userAgent),
    isQQ = (/qq/ig).test(navigator.userAgent),
    openapp = document.getElementById('cal-app');
openapp.addEventListener('click', function() {
    if (isIos) {
        window.location.href = "https://appdetail.netwin.cn/download.html"
    }

    if (isAndroid) {
        if (isWeixin || isQQ) { //andorid微信和QQ走微下载
            window.location.href = 'http://a.app.qq.com/o/simple.jsp?pkgname=com.nayun.framework&android_schema=' +  url;
        } else {
            body.appendChild(iframe);
            iframe.src = url.open;
            var t = Date.now();
            timer = setTimeout(function() {
                if (Date.now() - t > 1200) {
                    clearTimeout(timer);
                    return false;
                }
                if (document.webkitHidden || document.hidden) {
                    return false;
                }
                window.location.href = 'http://a.app.qq.com/o/simple.jsp?pkgname=com.nayun.framework';
            }, 1000);
        }
    }
}, false)


document.addEventListener("webkitvisibilitychange", function() {
    var tag = document.hidden || document.webkitHidden;
    if (tag) {
        clearTimeout(timer);
    }
});


window.addEventListener('pagehide', function() {
    clearTimeout(timer);
})