判断微信客户端的那些坑

在很多情况的开发中可能会要专门针对微信浏览器做一些特殊处理,所以需要在开发中区分出微信客户端。

在之前的项目中遇到了一些判断微信客户端的一些坑,在这里总结一下。

UserAgent 判断微信客户端

大部分开发者都会想到检测 UserAgent 中是否含有 MicroMessenger 字符来判断是否微信客户端。

一般情况下我们能拿到这样的 UserAgent 信息:

Mozilla/5.0 (iPhone; CPU iPhone OS 8_3 like Mac OS X) AppleWebKit/600.1.4 (KHTML, like Gecko) Mobile/12F70 MicroMessenger/6.1.5 NetType/WIFI

使用正则表达式就可以方便的判断:

function isWechat() {

var ua = navigator.userAgent.toLowerCase();

return /micromessenger/i.test(ua);

}

虽然这种方法很方便就能判断微信客户端,但是(重要的是但是)根据我的访问数据中得到的结果是有不少手机(主要是 Windows Phone)的微信客户端 UserAgent 是不带 MicroMessenger 字符的,所以使用 UserAgent 并不能覆盖所有的机型。

例如,我用 NOKIA 1520 获得的结果是:

Mozilla/5.0 (compatible; MSIE 10.0; Windows Phone 8.0; Trident/6.0; IEMobile/10.0; ARM; Touch; NOKIA; Lumia 1520)

所以在这里对使用 UserAgent 判断微信客户端的代码再进行优化一下:

function isWechat() {

var ua = navigator.userAgent.toLowerCase();

return /micromessenger/i.test(ua) || /windows phone/i.test(ua);

}

这样就能尽可能的覆盖所有的微信客户端了,但是问题在于在 Windows Phone 上使用非微信客户端也能绕过判断了。

Windows Phone 的判断

上面提到主要是 Windows Phone 中 UserAgent 中不包含 MicroMessenger 字符串,起初以为是一个 bug,跟开发微信 Windows Phone 的同学确认之后得知这不是一个 bug 而是Windows Phone 的 Webview 无法直接修改 UserAgent

Windows Phone 的微信客户端中的 HTTP 请求中 header 信息中会包含:

wxuserAgent:MicroMessenger

并且,Windows Phone 中的 navigator 对象中存在一个 wxuserAgent 属性。

所以改进一下上面的代码可以是这样的:

function isWechat() {

var ua = navigator.userAgent;

return /micromessenger/i.test(ua) || typeof navigator.wxuserAgent !== 'undefined';

}

当然这个办法也并不是非常方便,Windows Phone 8.1 的 JS 对象注入是异步的,所以你要能拿到 navigator.wxuserAgent 对象需要 1~2s 的延迟。

所以在页面第一次加载的使用中可能要设置一个 setTimeout 来做判断:

function isWechat() {

var ua = navigator.userAgent.toLowerCase();

return /micromessenger/i.test(ua) || typeof navigator.wxuserAgent !== 'undefined';

}

// 延迟判断

setTimeout(function() {

if(isWechat()) {

// handle is wechat...

} else {

// handle is not wechat...

}

}, 2000);

WeixinJSBridge 对象判断

之前微信客户端会内置一个 WeixinJSBridge 对象,用于 JS 调用微信内置的一些功能,但是随着微信 JSSDK 推出,该对象已经在新的版本中被废弃。

微信的JS API建立在客户端浏览器内置JS对象WeixinJSBridge上。然而WeixinJSBridge并不是WebView一打开就有了,客户端需要初始化这个对象,当这个对象准备好的时候,客户端会抛出事件"WeixinJSBridgeReady"。因此,WeixinJSBridge存在与否可以作为判断是否微信浏览器的一种办法。

// callback 函数为微信浏览器的回调

if (typeof WeixinJSBridge == "object" && typeof WeixinJSBridge.invoke == "function") {

callback();

} else {

if (document.addEventListener) {

document.addEventListener("WeixinJSBridgeReady", callback, false);

} else if (document.attachEvent) {

document.attachEvent("WeixinJSBridgeReady", callback);

document.attachEvent("onWeixinJSBridgeReady", callback);

}

}

OAuth 授权判断

通过 UserAgent 或者特殊对象以及 Header 头信息来判断微信客户端并不能保证 100% 的正确,因为客户端可能会模拟 UserAgent 或者修改 navigator 对象和 Header 头信息。

在一些并不是很重要的页面,可以考虑使用上面提到的 UserAgent 和对象判断的方法节省工作量。但是,在一些比较重要,限制必须在微信客户端内访问的网页可以使用微信的 oauth 授权来做判断会更保险。

在微信公众号的网页授权获取用户基本信息里面有提到 scope=snsapi_base 的授权方式不需要用户确认就能自动跳转并且回调获得用户的 open_id。

使用网页 oauth 授权相对比较麻烦(需要拥有一个认证过的微信公众帐号),但是更加安全可靠。 简单的梳理一下 oauth 流程大概是:

利用这个 oauth 的接口,我们可以根据后端是否能正确的获取到用户的 open_id 来判断用户是否使用微信客户端,并且还能有效的区分不同的微信用户。

第三方授权判断

有不少人是没有认证过的微信公众帐号的,所以也没有办法使用 OAuth 授权的方法判断,所以有人用自己的微信公众号做了个跟使用 OAuth 授权判断微信用户一样的思路的第三方weixingate

不过使用这种第三方的方法并不推荐,有一定风险。如果有已经认证过的微信公众号还是自己实现一个会比较安全。

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

推荐阅读更多精彩内容

  • H5页面窗口自动调整到设备宽度,并禁止用户缩放页面//一、HTML页面结构 // width 设置viewp...
    tiandashu阅读 16,658评论 0 9
  • 常见试题 行内元素:会在水平方向排列,不能包含块级元素,设置width无效,height无效(可以设置line-h...
    他大舅啊阅读 2,389评论 1 5
  • 转载自:http://www.cnblogs.com/duanguyuan/p/3534470.html 判断当前...
    白小爱阅读 869评论 1 3
  • 一、session与cookie区别 session存储在服务端,session会在一定时间内保存在服务器上,当访...
    fourzyz阅读 952评论 0 0
  • 你输了 今天在知乎上看到讨论罗辑思维是否有那么大的影响。逻辑思维是一个自媒体节目,说简单点就是把一些读书得到的心得...
    asfthj阅读 139评论 0 1