h5跳转app指定页面及各种坑的总结

11.29更新项目中对微信内的处理方案及一些坑
12.3更新ios9后url scheme的坑及最终解决方案

最近遇到一个需求:如果用户安装了app,则跳app;如果用户没安装app,则跳app下载链接,这个在平常见的还蛮多的。查了一些资料,总结一下。

一、解决方案:URL scheme 或 universal link

a)URL scheme是在app内配置的链接,比如:weixin://,superclass://。
URL scheme的格式是[scheme]://[host]/[path]?[query]。

b)universal link是ios9之后出的功能。它是通过传统HTTP链接来启动App。它其实就是一个https开头的链接,还要满足一些特定的规则才能被识别为universal link,才能直接唤起app。

二、具体实现

分2种情况讨论:
1、ios8之前和android:使用scheme方案。
原理:不管是ios还是安卓,浏览器都不可能知道手机有没有装某个app,所以方法是首先通过URL scheme打开app,如果打不开,则跳转下载链接。

var timeout, t = 1000, hasApp = true;  
var openScript = setTimeout(function () {  
        if (!hasApp) {
               // 跳转下载链接
        }
        document.body.removeChild(ifr);  
}, 2000)  
                
var t1 = Date.now();  
var ifr = document.createElement("iframe");  
ifr.setAttribute('src', url);  
ifr.setAttribute('style', 'display:none');  
document.body.appendChild(ifr);  

timeout = setTimeout(function () {  
       var t2 = Date.now();  
       if (t2 - t1 < t + 100) {  
             hasApp = false;
       }
}, t);

之所以要用iframe打开,而不是直接跳链接,是因为如果APP唤醒失败,或者APP未安装的话,很多时候都会跳到错误页,影响用户体验。而iframe方法不会引起页面可见的变化(例如页面内容变成一个新页面),不会导致浏览器历史记录的变化。

2、ios9之后:可以使用scheme方案,也可以使用ios9的universal link方案
(1)先说scheme方案,ios9把iframe封了,所以用的是直接跳转链接

location.href = url;
setTimeout(function() {   
       // 跳转下载链接
}, 250);
setTimeout(function() {
       location.reload();
}, 1000);

这个方案在用户安装了app的时候还没什么问题,但是如果没装app,跳转失败会先弹出错误弹窗,再弹出是否在「app store」中打开链接,见下图。


失败弹窗

如果产品能接受这种弹窗,其实也行。。下面的是封装了2个函数,实现h5跳app。

通用的URL scheme方案代码:

<script>
  function detectVersion() {
          let isAndroid,isIOS,isIOS9,version,
              u = navigator.userAgent,
              ua = u.toLowerCase();

          if (u.indexOf('Android') > -1 || u.indexOf('Linux') > -1) {   //android终端或者uc浏览器
              //Android系统
              isAndroid = true
          }

          if(ua.indexOf("like mac os x") > 0){
              //ios
              var regStr_saf = /os [\d._]*/gi ;
              var verinfo = ua.match(regStr_saf) ;
              version = (verinfo+"").replace(/[^0-9|_.]/ig,"").replace(/_/ig,".");
          }
          var version_str = version+"";
          if(version_str != "undefined" && version_str.length >0){
              version = parseInt(version)
              if(version>=8){
                  // ios9以上
                  isIOS9 = true
              }
              else{
                  isIOS = true
              }
          }
          return {isAndroid,isIOS,isIOS9}
        }

        // 判断手机上是否安装了app,如果安装直接打开url,如果没安装,执行callback
        function openApp(url,callback) {
            let {isAndroid,isIOS,isIOS9} = detectVersion()
            if(isAndroid || isIOS){
                var timeout, t = 4000, hasApp = true;  
                var openScript = setTimeout(function () {  
                    if (!hasApp) {
                        callback && callback()
                    }
                    document.body.removeChild(ifr);  
                }, 5000)  
                
                var t1 = Date.now();  
                var ifr = document.createElement("iframe");  
                ifr.setAttribute('src', url);  
                ifr.setAttribute('style', 'display:none');  
                document.body.appendChild(ifr);  

                timeout = setTimeout(function () {  
                    var t2 = Date.now();  
                    if (t2 - t1 < t + 100) {  
                        hasApp = false;
                    }
                }, t);
            }

            if(isIOS9){
                location.href = url;
                setTimeout(function() {   
                    callback && callback()
                }, 250);
                setTimeout(function() {
                    location.reload();
                }, 1000);
            }  
        }
        
        //跳h5
        function goConfirmAddr(){
            window.location.href = 'http://m.gaotu100.com'
        }
        window.onload = function(){
            openApp("superclass://home",goConfirmAddr)
        }
</script>

(2)universal link
这个配置起来比较麻烦,主要是app那边配置。具体可以看看最下面的参考博客。

看了下百度知乎的方法,都是用universal link
h5的知乎有个【app打开】按钮,这个按钮就是跳转universal link链接。
a)若此时安装了app,就提示是否在app打开。可以直接跳到app


image

b)若未装app或未唤起app,跳到下载页。


image

注意看域名:oia.zhihu.com,这个就是知乎的universal link,之所以是oia开头,是因为universal link必须跨域。

参考

https://www.cnblogs.com/shadajin/p/5724117.html
http://www.cocoachina.com/ios/20170904/20463.html
//www.greatytc.com/p/03e6b7828307

11.29更新

以上只是Demo阶段的总结,在项目过程遇到了2个问题:

一、微信内的处理方案

由于公司的app达不到微信的一些要求,所以微信浏览器内用universal link直接调起app这个方案行不通。

微信内的方案是:点击「打开app」时,弹窗一个遮罩层,提示让用户点开右上角选择在浏览器内打开。之后就简单了。

遇到的问题:ios微信浏览器内url不变化

解决方案戳ios微信浏览器内vue项目url不改变

二、一些安卓手机自带浏览器不能唤起app

原因:经反复测试之后发现,在手机自带浏览器内,某个网址第一次尝试打开app时,会有个是否打开某app的选择弹窗,浏览器会记住用户的选择。如用户选择「打开」,之后每次都不出现选择弹窗,每次直接跳转app;如果用户选择「取消」,之后该网站就再也不出现是否打开某app的选择弹窗,所以就调不起app,除非用户在设置内清除浏览器的数据。

暂时只在小米手机浏览器内发现这个问题,qq浏览器都好使,暂时没找到好的解决办法。

12.3更新

测试阶段自己点出了一个bug,ios9后用url scheme会出现一个偶发bug。
bug具体描述:前面我们知道,当用户没装app时,尝试打开app时会出现一个错误弹窗【safari打不开该网页,因为网址无效】。理论上说,点击「好」确认弹窗后,应该弹出【是否在app store打开】弹窗。(是否在app store中打开是因为配置了''window.location.href = 'https://itunes.apple.com/cn/app/id**********' ")

但是!!!!!!!!!!!!!!!!!!!!

当快速点击确认错误弹窗时,页面并不会弹出【是否在app store打开】弹窗,此时地址栏闪一下itunes的地址,之后尝试打开app,都是错误弹窗,再也调不起【是否在app store打开】弹窗。

然后看了下同样是使用url scheme方案的淘宝,他们的方案是错误弹窗后,跳转到自己内部的下载页,然后一进入下载页时就会弹出【是否在app store打开】弹窗,这样就不会出现上面的问题。

最后和产品商量,解决方案是跳到内部下载页,和淘宝一样。

12.7更新

看头条的这个功能时,发现了ios里一个好玩的东西Smart App Banners

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 195,898评论 5 462
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 82,401评论 2 373
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 143,058评论 0 325
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 52,539评论 1 267
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 61,382评论 5 358
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 46,319评论 1 273
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 36,706评论 3 386
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 35,370评论 0 254
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 39,664评论 1 294
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 34,715评论 2 312
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 36,476评论 1 326
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,326评论 3 313
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 37,730评论 3 299
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,003评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,275评论 1 251
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 41,683评论 2 342
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 40,877评论 2 335

推荐阅读更多精彩内容

  • 【采桑子】剑舞惊鸿 寒光三尺青锋冷。 翩若惊鸿, 矫若游龙, 飞雪流霜岭上松。 冠绝剑舞公孙颂。 颠旭书融, 道子...
    司秋绘芳阅读 590评论 0 1
  • 燕与雀归阅读 192评论 0 1
  • “当怀旧成为一种经典的病。” 我有过这样一位小学同桌——兴许是一年级,也可能是二年级,我记得不是太清,他白白又胖胖...
    圈圈城阅读 345评论 0 0
  • 文/芦苇 记忆沿着眼眶回旋 昨日的明媚刺疼禁闭的心 清脆的声音在耳畔响起 美好瞬间变成碎片 身体承受不起如此疼痛 ...
    芦苇花开沐春风阅读 354评论 0 0
  • 话,说给谁听,字,写给谁看。人群中,找不到答案。 问,想说给谁听,字想写给谁看。孤独里,自己听、看。 2018.4...
    冷風的孤獨阅读 246评论 0 0