从html5打开本地的app–如果本地没有app就跳转到下载页面,大家都会认为这是一项很简单的操作。网上的教程也很多,但是可行性都不高。因为手机系统和浏览器型号各不相同,所以兼容性会是让各个前端工程师头疼的问题。我们不妨看一下京东是如何解决的。京东的原代码已经混淆过了,我只能一点点反混淆并注释。

网上的文章千篇一律 都是采用window.location.href的方式打开的,但是这种方法的兼容性非常的渣。在ios的safari浏览器中无法使用,会出现还未打开app就自动跳转到下载页面的情况,影响用户的使用。那么我们来看一下京东是如何解决兼容性的问题。
下面附上我的代码翻译和注释。
京东打开app的js代码链接

(function(){// 判断浏览器var Navigator = navigator.userAgent;var ifChrome = Navigator.match(/Chrome/i) != null && Navigator.match(/Version\/\d+\.\d+(\.\d+)?\sChrome\//i) == null ? true : false;var ifAndroid = (Navigator.match(/(Android);?[\s\/]+([\d.]+)?/)) ? true : false;var ifiPad = (Navigator.match(/(iPad).*OS\s([\d_]+)/)) ? true : false;var ifiPhone = (!ifiPad && Navigator.match(/(iPhone\sOS)\s([\d_]+)/)) ? true : false;var ifSafari = (ifiPhone || ifiPad) && Navigator.match(/Safari/);var version =0;ifSafari && (version = Navigator.match(/Version\/([\d\.]+)/));version = parseFloat(version[1],10);// 是否从微信打开var ifWeixin = navigator.userAgent.indexOf("MicroMessenger") >=0; // weixinvar j = false;var iframe = "plugIn_downloadAppPlugIn_loadIframe";var t = false;var i =0;var B = {};var b = {};var selector = null;var Hquery = {};// 判断当前使用的js框架是zepto还是jqueryvar Query = window.Zepto || window.jQuery ? true : false;var g = [];// 是否存在html5的localStorage 存储var v = window.localStorage ? true : false;var o = "mdownloadAppPlugInskip";var p = null;function m() { // 打印时间 例如:2016-5-18 var M = new Date(); var N = M.getFullYear(); var O = M.getMonth() +1; var L = M.getDate(); strDate = N + "-" + O + "-" + L; return strDate }// 微信相关操作function r() { // weixin api WeixinJSBridge.invoke("getInstallState", { packageName: "com.jingdong.app.mall", packageUrl: "openApp.jdMobile://" }, function(M) { var N = M.err_msg, L =0; if (N.indexOf("get_install_state:yes") > -1) { j = true } }) }// 根据是否存在js框架进行dom和时间的绑定function bind(dom, event, fun) { // bind event if (Query) { selector("#" + dom).bind(event, fun) } else { selector("#" + dom).addEventListener(event, fun, !1) } }function z(L) { var M = (L || "mGen") + (++i); return M }// 微信操作if (ifWeixin) { // if navigitor is weixin if (window.WeixinJSBridge && WeixinJSBridge.invoke) { r() } else { document.addEventListener("WeixinJSBridgeReady", r, !1) } }// 如果存在js框架if (Query) { selector = window.$; Hquery = window.$ } else { selector = function(obj) { if (typeof obj == "object") { return obj } return document.querySelector(obj); }; if (!window.$) { window.$ = Hquery = selector } else { Hquery = window.$ } }window.onblur = function() { for (var L =0; L < g.length; L++) { clearTimeout(g[L]) } }; // 设置cookie。 function e(N) { var M = document.cookie.indexOf(N + "="); if (M == -1) { return "" } M = M + N.length +1; var L = document.cookie.indexOf(";", M); if (L == -1) { L = document.cookie.length } return document.cookie.substring(M, L) } // 设置cookie function l(N, P, L, Q, O) { var R = N + "=" + escape(P); if (L != "") { var M = new Date(); M.setTime(M.getTime() + L *24 *3600 *1000); R += ";expires=" + M.toGMTString() } if (Q != "") { R += ";path=" + Q } if (O != "") { R += ";domain=" + O } document.cookie = R } // 打开的链接集合 function F(L) { var url = { downAppURl: "http://h5.m.jd.com/active/download/download.html?channel=jd-m", downAppIos: "http://union.m.jd.com/download/go.action?to=http%3A%2F%2Fitunes.apple.com%2Fcn%2Fapp%2Fid414245413&client=apple&unionId=12532&subunionId=m-top&key=e4dd45c0f480d8a08c4621b4fff5de74", downWeixin: "http://a.app.qq.com/o/simple.jsp?pkgname=com.jingdong.app.mall&g_f=991850", downIpad: "https://itunes.apple.com/cn/app/jing-dong-hd/id434374726?mt=8", inteneUrl: "openApp.jdMobile://360buy?type=1", inteneUrlParams: null, openAppBtnId: "", closePanelBtnId: "", closePanelId: "", closeCallblack: null, closeCallblackSource: null, cookieFlag: null, noRecord: false, sourceType: "JSHOP_SOURCE_TYPE", sourceValue: "JSHOP_SOURCE_VALUE", openAppEventId: "MDownLoadFloat_OpenNow", closePanelEventId: "MDownLoadFloat_Close" }; if (L) { for (var M in L) { if (M && L[M]) { url[M] = L[M] } } } return url } // 敲黑板 重点内容。看京东是怎么解决兼容问题的。 function openApp(N, L) { // openApp var R = h(N); //获取相对应的url var O = null; if (ifWeixin) { // 如果是微信端 var M = null; if (j) { M = R } else { M = N.downWeixin } location.href = M; // 直接使用location.href打开 return } if (ifiPad) { // 如果是ipad O = N.downIpad } else { if (ifiPhone) { // 如果是iphone O = N.downAppIos } else { O = N.downAppURl } } if (ifChrome) { // 如果是chrome if (ifAndroid) { //安卓浏览器 var Q = R; R = y(Q); // 延后50毫秒 setTimeout(function() { window.location.href = R },50) } } if (ifSafari && version >=9) { // 判断safari版本 如果大于9 setTimeout(function() { // 必须要使用settimeout var S = document.createElement("a"); //创建a元素 S.setAttribute("href", R), S.style.display = "none", document.body.appendChild(S); var T = document.createEvent("HTMLEvents"); // 返回新创建的 Event 对象,具有指定的类型。 T.initEvent("click", !1, !1)// 初始化新事件对象的属性, S.dispatchEvent(T) // 绑定事件 },0) } else { document.querySelector("#" + iframe).src = R // 将iframe增加src } var P = Date.now(); setTimeout(function() { if (L) { var S = setTimeout(function() { x(P, O) },1500); g.push(S) } },100) } // x方法 function x(N, downUrl) { var L = Date.now(); if (N && (L - N) <(1500 +200)) { window.location.href = downUrl } } function h(N) { var V = []; var P = N.inteneUrlParams; var T = { category: "jump", des: "productDetail" }; if (N.sourceType && N.sourceValue) { T.sourceType = N.sourceType; T.sourceValue = N.sourceValue; if (P && !P.sourceType && !P.sourceValue) { P.sourceType = N.sourceType; P.sourceValue = N.sourceValue } } if (P) { for (var U in P) { if (U && P[U]) { V.push('"' + U + '":"' + P[U] + '"') } } } else { for (var U in T) { if (U && T[U]) { V.push('"' + U + '":"' + T[U] + '"') } } } try { var Q = MPing.EventSeries.getSeries(); if (Q) { var W = JSON.parse(Q); W.jdv = encodeURIComponent(e("__jdv")); W.unpl = encodeURIComponent(e("unpl")); W.mt_xid = encodeURIComponent(e("mt_xid")); W.mt_subsite = encodeURIComponent(e("mt_subsite")) } var S = { mt_subsite: encodeURIComponent(e("mt_subsite")), __jdv: encodeURIComponent(e("__jdv")), unpl: encodeURIComponent(e("unpl")), __jda: encodeURIComponent(e("__jda")) }; Q = JSON.stringify(W); V.push('"m_param":' + Q); V.push('"SE":' + JSON.stringify(S)) } catch (R) { V.push('"m_param":null') } var M = "{" + V.join(",") + "}"; var O = N.inteneUrl.split("?"); var L = null; if (O.length ==2) { L = O[0] + "?" + O[1] + "&params=" + M } else { L = O[0] + "?params=" + M } return L } function y(L) { return "intent://m.jd.com/#Intent;scheme=" + L + ";package=com.jingdong.app.mall;end" } function n(L) { if (L.openAppBtnId) { B[L.openAppBtnId] = L; G(L.openAppBtnId, L.openAppEventId); bind(L.openAppBtnId, "click", function() { var P = this.getAttribute("id"); var M = B[P]; if (!t) { var N = document.createElement("iframe"); N.id = iframe; document.body.appendChild(N); document.getElementById(iframe).style.display = "none"; document.getElementById(iframe).style.width = "0px"; document.getElementById(iframe).style.height = "0px"; t = true } var O = M.cookieFlag ? "downloadAppPlugIn_downCloseDate_" + M.cookieFlag : "downloadAppPlugIn_downCloseDate"; l(O, Date.now() + "_2592000000",60, "/", "m.jd.com"); l(O, Date.now() + "_2592000000",60, "/", "m.jd.hk"); openApp(M, true) }) } } function D(M) { if (M.closePanelBtnId && M.closePanelId) { B[M.closePanelBtnId] = M; G(M.closePanelBtnId, M.closePanelEventId); var Q = M.cookieFlag ? "downloadAppPlugIn_downCloseDate_" + M.cookieFlag : "downloadAppPlugIn_downCloseDate"; var O = e(Q); var P = null; if (O) { P = O.split("_"); if (P.length ==2) { P[0] = parseInt(P[0],10); P[1] = parseInt(P[1],10) } else { P = null } } var L = Date.now(); if (Html5Plus() || (!M.noRecord && P && P.length ==2 && (L - P[0]) < P[1])) { document.querySelector("#" + M.closePanelId).style.display = "none"; if (M.closeCallblack) { var N = M.closeCallblackSource ? M.closeCallblackSource : null; M.closeCallblack.call(N) } return } else { document.querySelector("#" + M.closePanelId).style.display = "block" } bind(M.closePanelBtnId, "click", function() { var U = this.getAttribute("id"); var R = B[U]; var T = R.cookieFlag ? "downloadAppPlugIn_downCloseDate_" + R.cookieFlag : "downloadAppPlugIn_downCloseDate"; if (!R.noRecord) { l(T, Date.now() + "_259200000",60, "/", "m.jd.com"); l(T, Date.now() + "_259200000",60, "/", "m.jd.hk") } document.querySelector("#" + R.closePanelId).style.display = "none"; if (R.closeCallblack) { var S = R.closeCallblackSource ? R.closeCallblackSource : null; R.closeCallblack.call(S) } }) } } function Html5Plus() { // htmlplus if (Navigator.indexOf("Html5Plus") >=0) { return true } else { return false } } function G(P, M) { try { var O = document.getElementById(P); var L = O.className; if (L) { L = L + " J_ping" } else { L = "J_ping" } O.className = L; O.setAttribute("report-eventid", M) } catch (N) {} } function C(L) { var M = F(L); n(M); D(M) } Hquery.downloadAppPlugIn = C; Hquery.downloadAppPlugInOpenApp = function(L) { var M = F(L); openApp(M); } });

京东在html5页面中打开本地app的解决方案相关推荐

  1. html5页面中打开本地app,如果没有跳转下载页面的解决方案

    需求效果 在推广网页上用户点击产品的详细信息时,判断出这个用户手机上是否安装自己的app如果安装了直接自动打开手机内的app应用,若没有则跳转app的下载页 技术实现 直接用window.locati ...

  2. iOS/Android 微信及浏览器中唤起本地APP

    title: iOS/Android 微信及浏览器中唤起本地APP date: 2017-05-10 10:19:20 tags: 需求概述 分享应用活动链接已经成为手机应用一个非常重要的推广传播形式 ...

  3. 钉钉微应用怎么进入_钉钉微应用如何打开本地app (Android)-问答-阿里云开发者社区-阿里云...

    老李归来 2016-01-28 10:14:09 Re钉钉微应用如何打开本地app (Android)经测试,免登认证已成功,执行 dd.device.launcher.checkInstalledA ...

  4. Element-UI中打开本地文件

    Element-UI中打开本地文件 一.问题 二.html部分 三.js部分 四.上一个/下一个 一.问题 有时候我们要打开本地文件直接在页面上查看,怎么用Element-UI提供的功能实现? 二.h ...

  5. H5 -- 微信h5页面中下载第三方app的方法

    需求:在微信h5页面中下载第三方app -- 安卓, 直接下载apk文件包:iphone,跳转AppStore 分析:微信不支持,在微信中屏蔽了apk文件的下载以及AppStore的跳转(且除非和TX ...

  6. html5页面中添加腾讯地图api

    html5页面中添加腾讯地图api: 点击地图出现详细的地图: 这是一个基于微信端的地图处理方案. 先看看html架构: <a id="aToMap" href=" ...

  7. android 淘宝天猫支付宝浏览器打开本地app传递参数打开应用内页

    近期由于项目需要,通过浏览器打开本地app应用.经过多方的查询反复的尝试和阅读.总结出来.有价值的知识的分享出来.虽说不是很难. 第一步: 写好相关的js方便调取如下: <!DOCTYPE ht ...

  8. htm文件在C语言中如何打开,如何在Microsoft Edge浏览器中打开本地HTML文件?

    自远古以来,如果您运行Web浏览器可执行文件,大多数Web浏览器都能打开本地文件,例如只需执行iexplore.exe file:/c:/temp/file或通过IShellDocView接口.我试图 ...

  9. android studio 本地html,android - 从当前HTML文件Android Studio中打开本地HTML文件? - 堆栈内存溢出...

    我在Android Studio上制作了一个WebView应用. 它会正确打开我的默认索引HTML页面 @Override protected void onCreate(Bundle savedIn ...

最新文章

  1. Java常用工具类---IP工具类、File文件工具类
  2. STL札记3-2(hashtable关联容器set、map)
  3. android trace文件分析ANR
  4. Spring Boot 多数据源(读写分离)入门
  5. SpringCloud配置中心内容加密
  6. LeetCode 451 根据字符出现频率排序
  7. 移动Web开发与适配笔记
  8. 图片传输_VBox6无线传输器 商业摄影师必备利器
  9. Linux基础培训笔记二
  10. 各纬度气候分布图_气候分布图纬度_世界气候气压带风带分布图要图(需要表识纬度)_滁州气象...
  11. zigbee协议栈学习(0)
  12. 2020年中国汽车后市场行业发展现状分析,汽车保有量不断提升,行业发展前景广阔「图」
  13. vmware mac os 10.11.6 安装xcode 8
  14. easypoi 语法_高中英语语法:高中英语必修1选修8全八册知识点总结
  15. ROS暑期学校与ROSCon 2018
  16. 小明对数位中含有 2、0、1、9 的数字很感兴趣,在 1 到 40 中这样的数包 括 1、2、9、10 至 32、39 和 40,共 28 个,他们的和是 574。 请问,在 1 到 2019 中,所
  17. 一行代码完成Java的Excel读写
  18. 车载DSP音频功放频率响应曲线2.5到20khz增益降太多是什么原因?
  19. 实数二分(模板及例题)
  20. 修改Firefox浏览器 user-agent 微信浏览器UA

热门文章

  1. (KALI)在U盘打造个性化PE工具箱+KALI(Persistence)+存储的工作站
  2. win10开机未能正确启动,win10首次开机无法启动是什么原因呢?
  3. thinkpad l470安装win10时系统bios配置
  4. 【江苏二级Python】8套历年真题及答案
  5. tensorflow错别字检测
  6. C# 使用SDL2实现Mp4文件播放音视频操作
  7. No Brainer 超级水
  8. 【机械臂】机械臂快速入门
  9. 将MultipartFile转换为File
  10. 第二章 lebesgue测度