关注Uzero
公众号,更多前端小干货等着你喔!html
看到标题,你们就能想起这个需求在不少项目上都能用到。咱们部署在Web服务器上的前端应用,既能够用PC浏览器访问,也能够用手机浏览器访问,再加上如今智能设备的推广,咱们甚至能在车载系统、穿戴设备和电视平台上访问。前端
设备的多样化让用户无处不在,有时候咱们须要根据不一样的浏览器运行环境作出对应的处理。浏览器是JavaScript的承载体,咱们能够从浏览器上获取相关的信息,来进一步处理咱们的业务逻辑。linux
然而浏览器品牌众多,有些浏览器使用的标准也不太同样,造就了难以统一的判断。下面我大概罗列一下经常使用的浏览器品牌和在什么状况下使用浏览器运行环境判断。浏览器相关统计数据能够参考这里。android
国际五大浏览器品牌:按照全球使用率降序排列
Windows
、MacOS
、Linux
、Android
、iOS
MacOS
、iOS
Windows
、MacOS
、Linux
、Android
、iOS
Windows
、MacOS
、Linux
、Android
、iOS
Windows
国产经常使用浏览器品牌:按照国内使用率降序排列,广泛基于开源项目
Chromium
进行开发
2019年04月30日
宣布中止服务顺便吐槽一下这个不要脸的红芯浏览器,明明就是基于Chromium
进行二次开发再套多一层外壳,还非得说本身开发的浏览器是世界第五大浏览器,偷吃不抹嘴,仍是被眼尖的网友发现了。详情请戳one、two、three。。。。ios
使用场景
针对处理一个这样的使用场景,其实有一个比较专业的名字,叫作浏览器指纹。咱们上面谈到的需求也只是浏览器指纹方案里面的一小部分,而咱们须要使用到的浏览器指纹就是UserAgent
。git
这个UserAgent
是何方神圣呢,中文翻译过来就是用户代理。引用百度的定义,就是一个特殊字符串头,使得服务器可以识别客户使用的操做系统及版本、CPU类型、浏览器载体及版本、浏览器渲染引擎、浏览器语言、浏览器插件等。而这些信息也足够咱们去判断浏览器运行环境了。github
目前网上不少解决方法都只是针对系统是不是桌面端仍是移动端,Android端仍是iOS端,部分浏览器载体的判断和获取等等,没有一个比较完美或者终极的解决方案。web
所以我用了不少测试平台整理出一个比较全面的解决方案。这个方案包含浏览器系统及版本、浏览器平台、浏览器内核及版本、浏览器载体及版本、浏览器外壳及版本。chrome
而此方案也是基于navigator.userAgent
获取相关浏览器信息(以下),再经过系统、平台、内核、载体、外壳的特有字段进行归类统一,整理出一个完整的浏览器运行环境。shell
const ua = navigator.userAgent.toLowerCase(); // 输出 "mozilla/5.0 (iphone; cpu iphone os 11_0 like mac os x) applewebkit/604.1.38 (khtml, like gecko) version/11.0 mobile/15a372 safari/604.1"
浏览器信息:权重按照如下降序排列
Windows
、MacOS
、Linux
、Android
、iOS
Desktop
桌面端、Mobile
移动端Webkit
、Gecko
、Presto
、Trident
Chrome
、Safari
、Firefox
、Opera
、Iexplore/Edge
获取UserAgent是否包含字段:判断是否包含系统、平台、内核、载体、外壳的特有字段
const testUa = regexp => regexp.test(ua);
获取UserAgent对应字段的版本
const testVs = regexp => (ua.match(regexp) + "").replace(/[^0-9|_.]/ig, "").replace(/_/ig, ".");
上述准备工做完成后,咱们就按照权重(系统 + 系统版本 > 平台 > 内核 + 载体 + 内核版本 + 载体版本 > 外壳 + 外壳版本)根据系统、平台、内核、载体、外壳的特有字段来归类统一浏览器运行环境。
系统+系统版本
// 系统 let system = "unknown"; if (testUa(/windows|win32|win64|wow32|wow64/ig)) { system = "windows"; // window系统 } else if (testUa(/macintosh|macintel/ig)) { system = "macos"; // macos系统 } else if (testUa(/x11/ig)) { system = "linux"; // linux系统 } else if (testUa(/android|adr/ig)) { system = "android"; // android系统 } else if (testUa(/ios|iphone|ipad|ipod|iwatch/ig)) { system = "ios"; // ios系统 } // 系统版本 let systemVs = "unknown"; if (system === "windows") { if (testUa(/windows nt 5.0|windows 2000/ig)) { systemVs = "2000"; } else if (testUa(/windows nt 5.1|windows xp/ig)) { systemVs = "xp"; } else if (testUa(/windows nt 5.2|windows 2003/ig)) { systemVs = "2003"; } else if (testUa(/windows nt 6.0|windows vista/ig)) { systemVs = "vista"; } else if (testUa(/windows nt 6.1|windows 7/ig)) { systemVs = "7"; } else if (testUa(/windows nt 6.2|windows 8/ig)) { systemVs = "8"; } else if (testUa(/windows nt 6.3|windows 8.1/ig)) { systemVs = "8.1"; } else if (testUa(/windows nt 10.0|windows 10/ig)) { systemVs = "10"; } } else if (system === "macos") { systemVs = testVs(/os x [\d._]+/ig); } else if (system === "android") { systemVs = testVs(/android [\d._]+/ig); } else if (system === "ios") { systemVs = testVs(/os [\d._]+/ig); }
平台
let platform = "unknow"; if (system === "windows" || system === "macos" || system === "linux") { platform = "desktop"; // 桌面端 } else if (system === "android" || system === "ios" || testUa(/mobile/ig)) { platform = "mobile"; // 移动端 }
内核+载体
let engine = "unknow"; let supporter = "unknow"; if (testUa(/applewebkit/ig) && testUa(/safari/ig)) { engine = "webkit"; // webkit内核 if (testUa(/edge/ig)) { supporter = "edge"; // edge浏览器 } else if (testUa(/opr/ig)) { supporter = "opera"; // opera浏览器 } else if (testUa(/chrome/ig)) { supporter = "chrome"; // chrome浏览器 } else { supporter = "safari"; // safari浏览器 } } else if (testUa(/gecko/ig) && testUa(/firefox/ig)) { engine = "gecko"; // gecko内核 supporter = "firefox"; // firefox浏览器 } else if (testUa(/presto/ig)) { engine = "presto"; // presto内核 supporter = "opera"; // opera浏览器 } else if (testUa(/trident|compatible|msie/ig)) { engine = "trident"; // trident内核 supporter = "iexplore"; // iexplore浏览器 }
内核版本+载体版本
// 内核版本 let engineVs = "unknow"; if (engine === "webkit") { engineVs = testVs(/applewebkit\/[\d.]+/ig); } else if (engine === "gecko") { engineVs = testVs(/gecko\/[\d.]+/ig); } else if (engine === "presto") { engineVs = testVs(/presto\/[\d.]+/ig); } else if (engine === "trident") { engineVs = testVs(/trident\/[\d.]+/ig); } // 载体版本 let supporterVs = "unknow"; if (supporter === "chrome") { supporterVs = testVs(/chrome\/[\d.]+/ig); } else if (supporter === "safari") { supporterVs = testVs(/version\/[\d.]+/ig); } else if (supporter === "firefox") { supporterVs = testVs(/firefox\/[\d.]+/ig); } else if (supporter === "opera") { supporterVs = testVs(/opr\/[\d.]+/ig); } else if (supporter === "iexplore") { supporterVs = testVs(/(msie [\d.]+)|(rv:[\d.]+)/ig); } else if (supporter === "edge") { supporterVs = testVs(/edge\/[\d.]+/ig); }
外壳+外壳版本
let shell = "none"; let shellVs = "unknow"; if (testUa(/micromessenger/ig)) { shell = "wechat"; // 微信浏览器 shellVs = testVs(/micromessenger\/[\d.]+/ig); } else if (testUa(/qqbrowser/ig)) { shell = "qq"; // QQ浏览器 shellVs = testVs(/qqbrowser\/[\d.]+/ig); } else if (testUa(/ubrowser/ig)) { shell = "uc"; // UC浏览器 shellVs = testVs(/ubrowser\/[\d.]+/ig); } else if (testUa(/2345explorer/ig)) { shell = "2345"; // 2345浏览器 shellVs = testVs(/2345explorer\/[\d.]+/ig); } else if (testUa(/metasr/ig)) { shell = "sougou"; // 搜狗浏览器 } else if (testUa(/lbbrowser/ig)) { shell = "liebao"; // 猎豹浏览器 } else if (testUa(/maxthon/ig)) { shell = "maxthon"; // 遨游浏览器 shellVs = testVs(/maxthon\/[\d.]+/ig); } else if (testUa(/bidubrowser/ig)) { shell = "baidu"; // 百度浏览器 shellVs = testVs(/bidubrowser [\d.]+/ig); }
终极合体
根据以上的条件判断得到的变量以下,咱们能够把它们合并成一个对象输出。这样就能够输出一个清晰的浏览器运行环境,后面想干吗就干吗了,多方便。
本文重点探究方案的可行性,没有过多考虑到代码的优化,因此条件判断使用得有些多,若是有什么方法能优化下代码,减小条件判断,能够在下方评论提个建议哟。
function BrowserType() { const ua = navigator.userAgent.toLowerCase(); const testUa = regexp => regexp.test(ua); const testVs = regexp => (ua.match(regexp) + "").replace(/[^0-9|_.]/ig, "").replace(/_/ig, "."); // 接上以上if...else条件判断 // ...... // 获取到system、systemVs、platform、engine、engineVs、supporter、supporterVs、shell、shellVs return Object.assign({ engine, // webkit gecko presto trident engineVs, platform, // desktop mobile supporter, // chrome safari firefox opera iexplore edge supporterVs, system, // windows macos linux android ios systemVs }, shell === "none" ? {} : { shell, // wechat qq uc 2345 sougou liebao maxthon baidu shellVs }); }
在控制台执行BrowserType()
,该有的都出来了,哈哈!源码详情请戳这里,喜欢的能够点个赞支持下,谢谢。
写到最后总结得差很少了,后续若是我想起还有哪些判断浏览器运行环境终极方案遗漏的,会继续在这篇文章上补全,同时也但愿各位朋友对文章里的要点进行补充或者提出本身的看法。欢迎在下方进行评论或补充喔,喜欢的点个赞或收个藏,保证你在开发时用得上。
《灵活运用》系列
《随笔》系列