伴随着今年10月底HTML5标准版的发布,未来使用H5的场景会越来越多,这是令web开发者欢欣鼓舞的事情。然而有一个现实我们不得不看清,那就是IE系列浏览器还占有一大部分市场份额,以IE8、9为主,windows8.1的用户已经用上了IE10/11,而考虑我国的国情,IE6、7依然存留不少。在我们放手用HTML5开发的时候,新特性支持度检测就是必不可少的了。一种方式是用navigator.userAgent或navigator.appName来检测浏览器类型和版本,不过这种方式不是很可靠,浏览器对于一些新特性也是在逐渐支持,不能肯定说某个浏览器100%支持了HTML5。而且,IE11做了一个恶心的举动:在UA中去掉了“MSIE”标志,把appName改为了“Netspace”,并且开始支持-webkit-前缀的css属性,这是活生生要伪装成chrome的节奏。所以,HTML5/CSS3支持性的检测,还是靠特征检测(figure detection)或者说能力检测更好些。本篇就来介绍一下常用的检测方式都有哪些。

  补充:判断IE11仍可用此方法:/Trident/i.test(navigator.appVersion),get from 小胡子。

HTML5部分

  检测HTML5新特性的方法主要有以下几种:

1. 检查全局对象(window或navigator)上有没有相应的属性名

2. 创建一个元素,检查元素上有没有相应的属性

3. 创建一个元素,检测元素上有没有方法名称,然后调用该方法,看能否正确执行

4. 创建一个元素,为元素的相应属性赋一个值,然后再获取此属性的值,看看赋值是否生效

  由于不同浏览器的不同行为,检测一些特性的时候,可能会用到上述几个方法的组合,接下来用上面的方法做一下常用特性的检测:

canvas

function support_canvas(){
var elem = document.createElement('canvas');
return !!(elem.getContext && elem.getContext('2d'));
}

  一般来讲,创建canvas元素并检查getContext属性即可,但是在一些浏览器下不够准确,所以再检测一下elem.getContext('2d')的执行结果,就可以完全确定。以上代码摘自Modernizr:http://github.com/Modernizr/Modernizr/issues/issue/97/

  关于canvas,有一点要补充的,那就是fillText方法,尽管浏览器支持canvas,我们并不能确切的确定它支持fillText方法,因为canvas API经历了各种修改,有一些历史原因,检测支持fillText的方法如下:

function support_canvas_text(){
var elem = document.createElement('canvas');
var context = elem.getContext('2d');
return typeof context.fillText === 'function';
}

video/audio

function support_video(){
return !!document.createElement('video').canPlayType;
} function support_audio(){
return !!document.createElement('audio').canPlayType;
}

  video和audio的写法相似。

  要检测video/audio支持的资源格式,可以调用canPlayType方法来进行检查,具体如下:

unction support_video_ogg(){
var elem = document.createElement('video');
return elem.canPlayType('video/ogg; codecs="theora"');
}
function support_video_h264(){
var elem = document.createElement('video');
return elem.canPlayType('video/mp4; codecs="avc1.42E01E"');
}
function support_video_webm(){
var elem = document.createElement('video');
return elem.canPlayType('video/webm; codecs="vp8, vorbis"');
} function support_audio_ogg(){
var elem = document.createElement('audio');
return elem.canPlayType('audio/ogg; codecs="vorbis"');
}
function support_audio_mp3(){
var elem = document.createElement('audio');
return elem.canPlayType('audio/mpeg;');
}
function support_audio_wav(){
var elem = document.createElement('wav');
return elem.canPlayType('audio/wav; codecs="1"');
}

  要注意的是,canPlayType的返回值并不是布尔类型,而是字符串,取值有以下几种:

  • "probably":浏览器完全支持此格式
  • "maybe":浏览器可能支持此格式
  • "":空串,表示不支持

localStorage

  一般来讲,检查全局对象是否有localStorage属性即可,如下:

function support_localStorage(){
try {
return 'localStorage' in window && window['localStorage'] !== null;
}
catch(e){
return false;
}
}

  在一些浏览器禁用cookie,或者设置了隐私模式什么的情况,检查属性或报错,所以加在try语句中,如果报错了认为不支持。

  另外,还有一种更严格的检查方法,检查相应方法是否支持,如下:

function support_localStorage(){
try {
if('localStorage' in window && window['localStorage'] !== null){
localStorage.setItem('test_str', 'test_str');
localStorage.removeItem('test_str');
return true;
}
return false;
}
catch(e){
return false;
}
}

webWorker

function support_webWorker(){
return !!window.Worker;
}

applicationCache

function support_applicationCache(){
return !!window.applicationCache;
}

geolocation

function support_geolocation(){
return 'geolocation' in navigator;
}

input标签新属性

  input标签新增的属性包括:autocomplete、autofocus、list、placeholder、max、min、multiple、pattern、required、step,检测是否支持用上面提到的方法2即可,新建一个input标签,看是否有这些属性,以autocomplete为例:

function support_input_autocomplete(){
var elem = document.createElement('input');
return 'autocomplete' in elem;
}

  另外要特别注意list属性,因为它是与datalist标签连用的,所以检查的时候要一并检测datalist标签是否支持:

function support_input_list(){
var elem = document.createElement('input');
return !!('list' in elem && document.createElement('datalist') && window.HTMLDataListElement);
}

input标签新类型

  这里的类型就是指type的取值,input新增的type值包括:search、tel、url、email、datetime、date、month、week、time、datetime-local、number、range、color。检测这些值需要用上面提到的方法4,以number为例:

function support_input_type_number(){
var elem = document.createElement('input');
elem.setAttribute('type', 'number');
return elem.type !== 'text';
}

  这是一种较为简单的检查方法,因为严格来讲,浏览器会为不同的类型提供不同的外观或实现,例如chrome中range类型会长这样:

  

  

  我们要确切的知道浏览器为该类型提供了对应的实现,才可以认为是“支持的”,要从这个方面检测是有难度的,各浏览器实现都不一。下面贴出Modernizr中的一个实现,供参考:检测email类型的实现:

function support_input_type_email(){
var elem = document.createElement('input');
elem.setAttribute('type', 'email');
elem.value = ':)';
return elem.checkValidity && elem.checkValidity() === false;
}

  为email类型设置了一个非法的value,然后手动调用校验方法,如果返回false,说明浏览器提供了该类型的实现,认为是支持的。

history

  history虽说是HTML4就有的,但HTML5提供了新的方法,检测方法如下:

function support_history(){
return !!(window.history && history.pushState);
}

webgl

function support_webgl(){
return !!window.WebGLRenderingContext;
}

postMessage

function support_postMessage(){
return !!window.postMessage;
}

draggable

HTML5的拖拽特性:

function support_draggable(){
var div = document.createElement('div');
return ('draggable' in div) || ('ondragstart' in div && 'ondrop' in div);
}

webSocket

unction support_webSocket(){
return 'WebSocket' in window || 'MozWebSocket' in window;
}

svg

function support_svg(){
return !!document.createElementNS && !!document.createElementNS('http://www.w3.org/2000/svg', 'svg').createSVGRect;
}

事件的支持性判断

通用方法:

  检查事件的支持性,用上面提到的方法2就可以,创建一个元素,看元素上有没有对应的事件名称,下面是摘自Modernizr的实现:

isEventSupported = (function() {

      var TAGNAMES = {
'select': 'input', 'change': 'input',
'submit': 'form', 'reset': 'form',
'error': 'img', 'load': 'img', 'abort': 'img'
}; function isEventSupported( eventName, element ) { element = element || document.createElement(TAGNAMES[eventName] || 'div');
eventName = 'on' + eventName; // When using `setAttribute`, IE skips "unload", WebKit skips "unload" and "resize", whereas `in` "catches" those
var isSupported = eventName in element; if ( !isSupported ) {
// If it has no `setAttribute` (i.e. doesn't implement Node interface), try generic element
if ( !element.setAttribute ) {
element = document.createElement('div');
}
if ( element.setAttribute && element.removeAttribute ) {
element.setAttribute(eventName, '');
isSupported = is(element[eventName], 'function'); // If property was created, "remove it" (by setting value to `undefined`)
if ( !is(element[eventName], 'undefined') ) {
element[eventName] = undefined;
}
element.removeAttribute(eventName);
}
} element = null;
return isSupported;
}
return isEventSupported;
})()

touch事件

  如果仅仅是检查touch事件是否支持,就没必要写上面那么多东西了,可以简单书写如下:

function support_touch_event(){
return !!(('ontouchstart' in window) || window.DocumentTouch && document instanceof DocumentTouch);
}

  Mozilla还提供了一个媒体查询的语句用来检测touch的支持性:touch-enabled,可以在页面上插入一个带有此媒体查询的元素来判断是否支持touch事件。参考:http://www.quirksmode.org/css/mediaqueries/touch.html

  不过我们做移动开发一般都只考虑webkit内核了,Mozilla的方式就不写了,如果需要请参考Modernizr源码。

css3部分

通用方法

  css3属性支持度的检测,主要通过上面方法提到的2和4来检查,不过我们要检查的是元素的style属性。另外一个要提的就是浏览器私有前缀,在现阶段我们用css3属性大部分是要写前缀的,因为各浏览器还未完全支持。我们用到的前缀有:-webkit-、-ms-、-o-、-moz-,至于前缀-khtml-,这是Safari早期使用的,现在基本也没有人再用它了,所以进行检测的时候就把它省去了。Modernizr移除了此前缀的检测,详见:https://github.com/Modernizr/Modernizr/issues/454

  通用代码如下:

var support_css3 = (function() {
var div = document.createElement('div'),
vendors = 'Ms O Moz Webkit'.split(' '),
len = vendors.length; return function(prop) {
if ( prop in div.style ) return true; prop = prop.replace(/^[a-z]/, function(val) {
return val.toUpperCase();
}); while(len--) {
if ( vendors[len] + prop in div.style ) {
return true;
}
}
return false;
};
})();

3D变形

  css3 3D变形的检测稍微复杂些,首先要支持perspective属性,其次要支持transform-style的值为preserve-3d。用方法4的方式无法检测出来,需要借助媒体查询的方式,代码如下:

function support_css3_3d(){
var docElement = document.documentElement;
var support = support_css3('perspective');
var body = document.body;
if(support && 'webkitPerspective' in docElement.style){
var style = document.createElement('style');
style.type = 'text/css';
style.innerHTML = '@media (transform-3d),(-webkit-transform-3d){#css3_3d_test{left:9px;position:absolute;height:3px;}}';
body.appendChild(style);
var div = document.createElement('div');
div.id = 'css3_3d_test';
body.appendChild(div); support = div.offsetLeft === 9 && div.offsetHeight === 3; }
return support;
}

  需要借助上面定义好的support_css3方法来检测perspective。

  基本常用检测的就这些了,本文中一部分代码是网上搜来的,还有一部分是从Modernizr源码中抽离出来的。如文中叙述有误,欢迎指正。

  在实际开发中,推荐直接使用Modernizr进行检测,它已经封装的非常漂亮了。但是如果你仅仅检测几个属性,或者不想因加载额外的库而浪费性能,就可以使用上述代码进行单个检测。

  先写这么多,以后有想到的再进行补充。

参考资料:

http://diveintohtml5.info/detect.html

https://github.com/Modernizr/Modernizr

http://code.tutsplus.com/tutorials/quick-tip-detect-css3-support-in-browsers-with-javascript--net-16444

常用的HTML5、CSS3新特性能力检测写法的更多相关文章

  1. 使用Modernizr探测HTML5/CSS3新特性(转载)

    转载地址:http://www.cnblogs.com/TomXu/archive/2011/11/18/detecting-html5-css3-features-using-modernizr.h ...

  2. 使用Modernizr探测HTML5/CSS3新特性

    [转] HTML5, CSS3以及相关技术(例如canvas和web sockets)带来了非常有用的特性,可以让我们的web程序提升一个新的level.这些新技术允许我们只用HTML,CSS和Jav ...

  3. html5 css3新特性了解一下

    html5: 用于绘画的 canvas 元素 以及SVG 用于媒介回放的 video 和 audio 元素 拖拽(Drag 和 drop) 地理定位(Geolocation) 对本地离线存储的更好的支 ...

  4. html5.css3新特性

    块: article section, Nav aside header footer  内容分组 future fig caption main 文本级别标签   time i b  em  str ...

  5. HTML5和CSS3新特性

    1.HTML5新标签和属性 1.1 兼容性前缀与语义化 兼容低版本的写法.比较新的浏览器,可以直接写.兼容性前缀,是每个浏览器私有的. 内核 兼容性前缀 浏览器 Gecko -moz- Firefox ...

  6. 21、前端知识点--html5和css3新特性汇总

    跳转到该链接 新特性汇总版: https://www.cnblogs.com/donve/p/10697745.html HTML5和CSS3的新特性(浓缩好记版) https://blog.csdn ...

  7. 浅谈HTML5的新特性

    2014年10月29日,W3C宣布,经过接近8年的艰苦努力,HTML5标准规范终于制定完成. HTML5将会取代1999年制定的HTML 4.01.XHTML 1.0标准,使网络标准达到符合当代的网络 ...

  8. Atitti html5 h5 新特性attilax总结

    Atitti html5 h5 新特性attilax总结 Attilax觉得不错的新特性 3.语义Header和Footer (The Semantic Header and Footer) 8.占位 ...

  9. CSS3新特性(阴影、动画、渐变、变形、伪元素等)

    CSS3与页面布局学习总结(六)--CSS3新特性(阴影.动画.渐变.变形.伪元素等)   目录 一.阴影 1.1.文字阴影 1.2.盒子阴影 二.背景 2.1.背景图像尺寸 2.2.背景图像显示的原 ...

随机推荐

  1. mysql.connector操作mysql的blob值

    This tutorial shows you how to work with MySQL BLOB data in Python, with examples of updating and re ...

  2. amr转MP3

    using System; using System.Threading; using System.IO; using System.Diagnostics; using System.Securi ...

  3. 接收新信息,在会话中看不到(thread表数据插入/更新失败)

    分析原因:收到短信,sms表插入信息,触发器会自动更新thread表,更新失败导致一直有一条未读信息数量显示,但在会话列表中却看不到. (偶现,低概率. 解决方法:接收新信息插入后,立即查询threa ...

  4. 大前端学习笔记整理【五】关于JavaScript中的关键字——this

    写在前面 工作有那么一段时间了,但是在工作中,发现自己的理论知识还是有所欠缺.特别是在javascript上,很多东西其实自己属于知道要用这个,但是不知道为什么要这么用...这种情况很是尴尬了,所以写 ...

  5. archlinux 安装过程记录

    2014年安装了一次,使用U盘启动安装的,但是当时网络有问题,断断续续,没有做详细记录. 现在到了杭州,重新来一次. 使用U盘安装 下载ISO :http://mirrors.163.com/arch ...

  6. BZOJ 1047: [HAOI2007]理想的正方形

    题目 单调队列是个很神奇的东西,我以前在博客写过(吧) 我很佩服rank里那些排前几的大神,700ms做了时限10s的题,简直不能忍.(但是我还是不会写 我大概一年半没写单调队列,也有可能根本没有写过 ...

  7. C语言的选择和循环上机题目(部分)

    /*(1)某市不同车牌的出租车3公里的起步价和计费分别为:夏利7元/公里,3公里以外2.1元/公里:富康8元/公里,3公里以外2.4元/公里:桑塔纳9元,3公里以外2.7元/公里.编程:从键盘输入乘车 ...

  8. android手机调试时不能打印Logcat日志信息

    方法: 1.在拨号界面输入:*#*#2846579#*#*  进入测试菜单界面 2.Project Menu–后台设置–LOG设置 3.LOG开关–LOG打开   LOG级别设置–VERBOSE 4. ...

  9. bower

    1. bower介绍 Bower 是 twitter 推出的一款包管理工具,基于nodejs的模块化思想,把功能分散到各个模块中,让模块和模块之间存在联系,通过 Bower 来管理模块间的这种联系. ...

  10. API测试-Super Test

    框架选择 Super Test:基本的HTTP请求,返回判断 Chai:对返回的结果进行判断 grunt:集成jenkins grunt-mocha-test:grunt任务 Jenkins环境配制 ...