jQuery 源码分析笔记(5) jQuery.support


其中jQuery.browser已经提供了根据UserAgent检测的浏览器信息。而jQuery.support 使用特性检测来检查浏览器的功能以及Bug。

和文档一样,首先说明一下,这个模块是很底层的代码,基本不需要在日常开发中使用,但是插件的开发者更需要。因为插件需要兼容各个浏览器。首先看一下 support模块提供了哪些浏览器特性的检测,以下结果是在Chrome 13 Dev下看到的结果。根据浏览器的不同,这里的成员可能会有变化。(PS:再次吐槽IE,大部分属性都是IE比较特别)

  • ajax: true。是否支持XMLHttpRequest对象。老版本的IE不支持,必须用ActiveX。 * appendChecked: true。 * boxModel: true。是否根据W3C CSS盒模式渲染。IE6和7的Quirks模式为False。 * changeBubbles: true。change事件是否沿着DOM树冒泡。这是W3C DOM事件模型的要求,但是IE目前为止是False。jQuery模拟了这个冒泡特性。 * checkClone: undefined。在克隆Radio Button或者CheckBox时是否保留选中状态。以前说createFragment缓存的时候提到过,WebKit内核是不保留的。 * checkOn: false。CheckBox在没有指定时是否默认为On。 * cors: true。XMLHttpRequest对象是否带有withCredentials属性。可以执行跨域请求。或者没有这个属性,但是有其他途径可以实现跨域XHR请求的,也是True(比如通过Windows Gadget)。 * cssFloat: true。支持cssFloat的CSS属性。IE是False,它使用了styleFloat。 * deleteExpando: true。 * focusinBubbles: false。 * getSetAttribute: true。 * hrefNormalized: true。对一个a元素调用getAttribute("href")时,是否返回原始指定的URL。IE会返回完整路径。比如对于href="1.html"的链接,IE会得到http://..../1.html。 * htmlSerialize: true。是否可以用innerHTML插入link元素。IE为false。 * inlineBlockNeedsLayout: false。为了让block元素表现inline-block,是否需要为inline并且hasLayout(IE下的概念,大部分布局问题的根源)。IE8以下有问题。 * leadingWhitespace: true。innerHTML属性是否严格按照代码渲染。IE6-8中会去掉前置的空白。(所以使用Markdown写的这篇文章在IE下有格式问题,因为换行不见了) * noCloneChecked: true。1.5.1新属性。检测浏览器是否克隆checked扩展属性。IE为false。 * noCloneEvent: true。拷贝的元素是否会带有事件处理函数(即Event handler是否被拷贝过去)。IE为false。 * opacity: true。是否支持opacity CSS属性(透明度)。IE为false,它使用alpha filter。 * optDisabled: true。已经disabled的select元素中的option元素是否默认为disabled。 * optSelected: true。一个默认被选中的option元素的selected属性是否正常。 * radioValue: true。 * reliableHiddenOffsets: true。在table的cells被设置为display:none的时候,仍然有offsetWidth/Height。即hidden靠不住。只有IE8有这个问题。参见4512号bug * reliableMarginRight: true。div有显式的宽度,但是没有margin-right。此时基于容器计算margin-right不正确。旧版本WebKit有问题。参见3333号bug * scriptEval(): 1.5.1版本前,这是一个属性,而现在变成函数了。检测一个inline的脚本在使用标准的DOM操作函数被添加时是否会自动执行。比如appendChild和createTextNode。IE是false,它使用text来插入脚本。 * shrinkWrapBlocks: false。元素在hasLayout的情况下是否会撑大父节点。只有IE6是true。 * style: true。一个元素的inline的style属性是否可以用DOM属性访问。比如getAttribute("style")。IE是false,它使用cssText属性。 * submitBubbles: true。submit事件是否沿着DOM树冒泡。IE为false,jQuery模拟了这个冒泡过程。 * tbody: true。一个空的table元素是否可以没有tbody元素。根据HTML规范,子元素是可选的。但是IE是false,所以会自动插入一个tbody元素。

jQuery创建了几个元素用来判断浏览器特性。如下:

复制代码 代码如下:

var div = document.createElement("div");
div.setAttribute("className", "t");
div.innerHTML = " <LINK><TABLE><TBODY></TBODY></TABLE><A style="FLOAT: left; TOP: 1px; opacity: .55" href="http://www.phpstudy.net/a">a</A><INPUT type=checkbox>";
all = div.getElementsByTagName("*");
a = div.getElementsByTagName("a")[0];
var select = document.createElement("select");
opt = select.appendChild(document.createElement("option"));
input = div.getElementsByTagName("input")[0];

其中div元素相当有内涵。包括了空白开头,空Table元素,inline style,opacity等等。除了一行行看代码和注释也没啥其他笔记了:
复制代码 代码如下:

support = {
// IE会把开头的空格去掉,所以nodeType不是3(文本)
leadingWhitespace: (div.firstChild.nodeType === 3),
// IE会自动插入tbody,所以length不同
tbody: !div.getElementsByName("tbody").length,
// IE不允许用这种方式插入link元素
htmlSerialize: !!div.getElementsByTagName("link").length,
// 如果正常获得style的话,这个正则表达式应该会正常通过
style: /top/.test(a.getAttribute("style")),
// href属性应该是原始指定的字符串,IE会修改为http开头的绝对URL
hrefNormalized: (a.getAttribute("href") === "/a"),
// 能得到opacity属性。这里使用了正则表达式,是为了绕过WebKit[5145号bug](http://dev.jquery.com/ticket/5145)
// 但是这应该是一个早期版本的问题。起码在Chrome 13中,不用正则也是对的。
opacity: /^0.55$/.test(a.style.opacity),
// IE使用styleFloat
cssFloat: !!a.style.cssFloat,
// div中的checkbox没有指定值,看默认值是否是on。WebKit是""。所以我的结果是false
checkOn: (input.value === "on"),
// 这个select只有一个option元素,所以渲染时,这个option是默认选中的。此时selected应该是true。
optSelected: opt.selected,
submitBubbles: true,
changeBubbles: true,
focusinBubbles: false,
deleteExpando: true,
noCloneEvent: true,
inlineBlockNeedsLayout: false,
shrinkWrapBlocks: false,
reliableMarginRight: true
};

下面的大部分都是按部就班的测试。其中BoxModel的测试比较有趣:
复制代码 代码如下:

div.innerHTML = ""; // 从头来
div.style.width = div.style.paddingLeft = "1px";
body = document.createElement("body");
body.style.width = 0;
body.style.height = 0;
body.style.border = 0;
body.style.margin = 0; // 全部设置为0
body.appendChild(div);
document.documentElement.insertBefore(body, document.documentElement.firstChild);
support.boxModel = div.offsetWidth === 2;
// 这个时候div嵌套在body下。而body长宽高,边框和margin都是0。所以div的偏移应该就是它自己的paddingLeft+width。如果不对,就说明盒模式不对。

然后值得注意的是事件的冒泡,注释中提到的技术参考文章链接已经失效。请自己搜索"Detecting event support without browser sniffing"
复制代码 代码如下:

// attachEvent是IE的,所以这里实际上只检查了IE,其他浏览器都是前面设置的默认值。
if(div.attachEvent) {
for(i = {
submit: 1,
change: 1
focusin: 1
}) {
eventName = "on" + i;
isSupported = (eventName in div);
if(!isSupported) {
div.setAttribute(eventName, "return;");
isSupported = (typeof div[eventName] === "function");
}
support[i + "Bubbles"] = isSupported;
}
}

PS: 浏览器检测是一个很纠结,很细节的任务。这个笔记慢慢完善中……

18:30补充:经过IE9的测试,以上大部分IE相关问题都已经不存在。以下是IE9+Win7的测试结果:
复制代码 代码如下:

$.support
{
boxModel: true,
changeBubbles: true,
checkClone: true,
checkOn: true,
cssFloat: true,
deleteExpando: true,
hrefNormalized: true,
htmlSerialize: true,
leadingWhitespace: true,
noCloneEvent: false,
opacity: true,
optSelected: false,
parentNode: true,
scriptEval: true,
style: true,
submitBubbles: true,
tbody: true
}



相关阅读:
vbs imail 密码解密
dedecms自动给已经发布过的文章中的关键词加超链接
phplib类中文详解
一个图形显示IP的PHP程序代码
(三)—共享成员
CSS的color颜色使用说明
FreeBSD网站平台建设全过程 第二步
jQuery AJAX实现调用页面后台方法和web服务定义的方法分享
了解linux中Load average负载
循环取值Request.QueryString的用法
jquery text()要注意啦
需要做特殊处理的DOM元素属性的访问
SQL存储过程初探
javascript 全选/反选,取消选择效果
快速导航
PHP MySQL HTML CSS JavaScript MSSQL AJAX .NET JSP Linux Mac ASP 服务器 CMS SQL jQuery C# C++ java Android IOS oracle MongoDB PostgreSQL SQLite 交通频道
吐鲁番-合肥  惠州-景宁、泰顺  朔州-绥德  无锡-江阴西门  石家庄-蟒石口  清远-阳山  成都-重庆  北京-上海  温州-椒江区  唐山-水厂  丽水-三门县  佛山-高明  绵阳-建中  七台河-依兰  乌海-苏米图  山南-泰州  杭州-海盐  宁波-金华  德阳-绵远  上海-张家港(沙州)  北京-商水  南阳-栗盘  沈阳-郑州  北京-高碑店  中山-德庆  天津-阳信  福州-宁德  台州-温州  深圳-广州  厦门-下伍堡  赣州-南平  河源-龙川至珠海  郑州-开封  深圳-南宁  长沙-双峰  重庆-丰盛  南阳-镇平  海西-武汉  黄石-赣州  清远-清新温泉  中山-大良  上海-敖江  邢台-毛庄  庆阳-石家庄  北京-凤山  广州-勒流  江门-广州芳村  大同-河北柴沟堡  成都-乐至  南安-石狮  北京-集宁  温州-宁波  新余-建山  承德-七家/旺上店  丽水-三门县  铁岭-杨屯(阿吉)  泸州-海潮  南京-宜兴  成都-黑水  广州-福建福州  郑州-宁陵  邢台-大鲁  遂宁-长沙  洛阳-孟州  深圳-深圳宝安国际机场  珠海-深圳  无锡-宜兴  福州-平阳  厦门-温岭  衡水-码头李  临沂-新沂  南昌-莲花  珠海-开平长沙  广州-邵阳(邵东)  汕尾-海丰县  台州-安福  广州-三灶  深圳-常平  唐山-龙山  天津-林西  宁波-柯桥  长沙-津市  北京-宁津  洛阳-郑州  无锡-张家港  广州-乔建  上海-杭州  汕头-福州  安庆-蚌埠  沈阳-法库(西)  邢台-沙村  苏州-义鸟-永康  北京-霸州  淮安-上海  济宁-梁山  武汉-宜都  重庆-永川  深圳-祁东  南阳-西张营  广州-中山沙溪  开封-周口  深圳-容奇  开封-商丘  绍兴-绍兴  邢台-高屯  焦作-济源  深圳-广州  淮安-青岛  丹东-振安区  金坛-南京  温州-义乌  日照-临沭  衡水-周胡刘  酒泉-高台  盘锦-大连  上海-昆山  江门-井岸  邢台-德归  晋城-晋庙铺镇  上海-无为  济南-新安江  杭州-德清  上海-泰兴  台州-黄岩  深圳-东莞  广州-乐民  渭南-周口  金华-永康  佛山-江门  南阳-桐柏  云浮-珠海  随州-珠海  杭州-海宁  徐州-南通  阳江-珠海  深圳-太平  惠州-惠东  唐山-滦县  无锡-张家港  唐山-马兰庄  南昌-金溪  杭州-瓜沥  韶关-始兴  潍坊-兖州  衢州-玉环  广州-湖南隆回  南昌-涟源  芜湖-溧阳  泉州-兴国  中山-惠州  江门-阳江  西双版纳-呼和浩特  驻马店-郑州  深圳-南充  白山-江源区  内江-高梁  普洱-淮安  杭州-兰溪  重庆-太和(老路)  宁波-余姚  佛山-虎门  唐山-滦县  上海-温州  广州-澳门  中山-揭阳汽车站  上海-慈溪  金华-石浦  北京-宣化区  上海-吴堡  齐齐哈尔-托力河  德州-聊城  无锡-张家港  重庆-隆盛  嘉峪关-泰州  南阳-驻马店  焦作-沂源县  北京-盐城  济源-侯马  镇江-汉府街  唐山-康关营  江门-恩平  江门-长沙(粤)  南阳-李富户  吉安-安福  丽水-三门县  淮南-炎刘  邢台-深县  广州-钦州  桂林-平乐  韶关-灵溪  丽水-三门县  郑州-鲁恭庙  济南-黄岛  南阳-沙堰  东莞-江西全南汽车站  洛阳-渑池  南京-新沂  洛阳-登封  苏州-震泽  厦门-永定

Copyright © 2016 phpStudy |