jQuery 源码分析笔记(5) jQuery.support |
本文标签:jQuery,源码分析 其中jQuery.browser已经提供了根据UserAgent检测的浏览器信息 。而jQuery.support 使用特性检测来检查浏览器的功能以及Bug 。
和文档一样,首先说明一下,这个模块是很底层的代码,基本不需要在日常开发中使用,但是插件的开发者更需要 。因为插件需要兼容各个浏览器 。首先看一下 support模块提供了哪些浏览器特性的检测,以下结果是在Chrome 13 Dev下看到的结果 。根据浏览器的不同,这里的成员可能会有变化 。(PS:再次吐槽IE,大部分属性都是IE比较特别)
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.jb51.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 } |