[ ​jump directly to the list​ ] 
 
While​ the sentiment of ​youmightnotneedjquery​ is great, developers should be aware that ditching 
libraries, like jQuery, can easily require large amounts of research on their end to avoid bugs (even in 
modern browsers). The snippets provided by ​youmightnotneedjquery​ are a starting point but hardly 
scratch the surface of being a solid robust replacement to jQuery. 
 
The great thing about an established library, like jQuery, is it’s hammered on by ​lots of talented people​, 
transparently improved​, and refined by the community.  
 
Concerned over file size? When it comes to page load time, count of HTTP requests (and placement) matter 
far more​ than total JS size. And heck, jQuery 1.9.x and 2.x allow ​custom builds​, so you can minimize what 
of jQuery you end up shipping. 
 
This line from ​youmightnotneedjquery​ is worth repeating… 
 “At the very least, make sure you know what jQuery is doing for you, and what it's not.” 
 
 
~ John-David Dalton, Paul Irish 
Feb 6, 2014 
Document history: 
● 2014 Jan 30: ​Rick Waldron (TC39)​ ​compiled a list​ of all the browser bug workarounds in jQuery’s core source. 
● 2014 Feb 2: ​Boris Zbarsky (Firefox engineer)​ looked through and ​annotated​ the items.  
● 2014 Feb 6: This cleanup and comment from​ John-David Dalton (Microsoft) & Paul Irish (Chrome) 
● 2015 Jan 8: Paul Irish added a note for how to investigate current workarounds in modern jQuery source 
● 2015 April 20: Paul Irish edited the intro and pasted in all new workarounds. John Resig updated the workaround 
count. 
 
List of browser-specific workarounds that jQuery addresses 
Update April 2015:  
You can now grep the jQuery source for their browser workarounds (noted by “​// Support:​”) : 
 
curl http://code.jquery.com/jquery‐git2.js | grep ‐A 5 ‐n Support: 
 
or: 
 
curl http://code.jquery.com/jquery‐git2.js | grep ‐n Support: | wc ‐l 
 
Currently ​88​ documented workarounds in ​jQuery 2.x​. 
 
 
Workarounds as of April 2015 
// line 40: 
// Support: Firefox 18+ 
// Can't be in strict mode, several libs including ASP.NET trace 
// the stack via arguments.caller.callee and Firefox dies if 
// you try to trace through "use strict" call chains. (#13335) 
// 
var​ arr ​=​ ​[]​; 
 
 
// line 77: 
​// Support: Android<4.1 
​// Make sure we trim BOM and NBSP 
rtrim ​=​ ​/​^[​s​uFEFFxA0​]+​|​[​s​uFEFFxA0​]+$​/​g​, 
 
​// Matches dashed string for camelizing 
rmsPrefix ​=​ ​/​^​‐ms‐​/​, 
 
 
// line 298: 
​// Support: Android<4.0 (functionish RegExp) 
​return​ ​typeof​ obj ​===​ ​"​object​"​ ​||​ ​typeof​ obj ​===​ ​"​function​"​ ​? 
class2type​[​ ​toString​.​call​(​obj​)​ ​]​ ​||​ ​"​object​"​ ​: 
​typeof​ obj​; 
​}​, 
 
 
 
// line 313: 
​// Support: IE9‐11+ 
​// Microsoft forgot to hump their vendor prefix (#9572) 
camelCase​:​ ​function​(​ string ​)​ ​{ 
​return​ string​.​replace​(​ rmsPrefix​,​ ​"​ms‐​"​ ​).​replace​(​ rdashAlpha​,​ fcamelCase ​)​; 
​}​, 
 
 
 
// line 345: 
​// Support: Android<4.1 
trim​:​ ​function​(​ text ​)​ ​{ 
​return​ text ​==​ ​null​ ​? 
​""​ ​: 
​(​ text ​+​ ​""​ ​).​replace​(​ rtrim​,​ ​""​ ​)​; 
​}​, 
 
 
// line 374: 
​// Support: Android<4.1, PhantomJS<2 
​// push.apply(_, arraylike) throws on ancient WebKit 
merge​:​ ​function​(​ first​,​ second ​)​ ​{ 
​var​ len ​=​ ​+​second​.​length​, 
j ​=​ ​0​, 
i ​=​ first​.​length​; 
 
 
// line 647: 
​// Support: Firefox<24 
​// Workaround erroneous numeric interpretation of +"0x" 
​return​ high ​!==​ high ​||​ escapedWhitespace ​? 
escaped ​: 
high ​<​ ​0​ ​? 
​// BMP codepoint 
 
 
// line 672: 
​// Support: Android<4.0 
​// Detect silently failing push.apply 
arr​[​ preferredDoc​.​childNodes​.​length​ ​].​nodeType​; 
}​ ​catch​ ​(​ e ​)​ ​{ 
push ​=​ ​{​ apply​:​ arr​.​length​ ​? 
 
 
 
// line 683: 
​// Support: IE<9 
​// Otherwise append directly 
​function​(​ target​,​ els ​)​ ​{ 
​var​ j ​=​ target​.​length​, 
i ​=​ ​0​; 
​// Can't trust NodeList.length 
 
 
// line 732: 
​// Support: IE, Opera, Webkit 
​// TODO: identify versions 
​// getElementById can match elements by name instead of ID 
​if​ ​(​ elem​.​id ​===​ m ​)​ ​{ 
results​.​push​(​ elem ​)​; 
​return​ results​; 
 
 
// line 746: 
​// Support: IE, Opera, Webkit 
​// TODO: identify versions 
​// getElementById can match elements by name instead of ID 
​if​ ​(​ newContext ​&&​ ​(​elem ​=​ newContext​.​getElementById​(​ m ​))​ ​&& 
contains​(​ context​,​ elem ​)​ ​&& 
elem​.​id ​===​ m ​)​ ​{ 
 
 
// line 783: 
​// Support: IE <=8 
​// Exclude object elements 
​}​ ​else​ ​if​ ​(​ context​.​nodeName​.​toLowerCase​()​ ​!==​ ​"​object​"​ ​)​ ​{ 
 
​// Capture the context ID, setting it first if necessary 
​if​ ​(​ ​(​nid ​=​ context​.​getAttribute​(​ ​"​id​"​ ​))​ ​)​ ​{ 
 
 
// line 1008: 
​// Support: IE 9 ‐ 11 
​// Accessing iframe documents after unload throws "permission denied" errors (jQuery #13936) 
​// Limit the fix to IE with document.documentMode and IE >=9 with document.defaultView 
​if​ ​(​ document​.​documentMode ​&&​ ​(​parent ​=​ document​.​defaultView​)​ ​&&​ parent​.​top ​!==​ parent ​)​ ​{ 
​// Support: IE 11 
​if​ ​(​ parent​.​addEventListener ​)​ ​{ 
parent​.​addEventListener​(​ ​"​unload​"​,​ unloadHandler​,​ ​false​ ​)​; 
 
​// Support: IE 9 ‐ 10 only 
​}​ ​else​ ​if​ ​(​ parent​.​attachEvent ​)​ ​{ 
parent​.​attachEvent​(​ ​"​onunload​"​,​ unloadHandler ​)​; 
​} 
​} 
 
 
 
// line 1025: 
​// Support: IE<8 
​// Verify that getAttribute really returns attributes and not properties 
​// (excepting IE8 booleans) 
support​.​attributes ​=​ assert​(​function​(​ div ​)​ ​{ 
div​.​className ​=​ ​"​i​"​; 
​return​ ​!​div​.​getAttribute​(​"​className​"​)​; 
 
 
// line 1042: 
​// Support: IE<9 
support​.​getElementsByClassName ​=​ rnative​.​test​(​ document​.​getElementsByClassName ​)​; 
 
​// Support: IE<10 
​// Check if getElementById returns elements by name 
​// The broken getElementById methods don't pick up programatically‐set names, 
​// so use a roundabout getElementsByName test 
support​.​getById ​=​ assert​(​function​(​ div ​)​ ​{ 
docElem​.​appendChild​(​ div ​).​id ​=​ expando​; 
 
 
// line 1069: 
​// Support: IE6/7 
​// getElementById is not reliable as a find shortcut 
​delete​ Expr​.​find​[​"​ID​"​]​; 
 
Expr​.​filter​[​"​ID​"​]​ ​=​  ​function​(​ id ​)​ ​{ 
​var​ attrId ​=​ id​.​replace​(​ runescape​,​ funescape ​)​; 
 
 
// line 1150: 
​// Support: IE8, Opera 11‐12.16 
​// Nothing should be selected when empty strings follow ^= or $= or *= 
​// The test attribute must be unknown in Opera but "safe" for WinRT 
​// http://msdn.microsoft.com/en‐us/library/ie/hh465388.aspx#attribute_section 
​if​ ​(​ div​.​querySelectorAll​(​"​[msallowcapture^='']​"​).​length​ ​)​ ​{ 
rbuggyQSA​.​push​(​ ​"​[*^$]=​"​ ​+​ whitespace ​+​ ​"​*(?:''|​""​)​"​ ​)​; 
 
 
// line 1158: 
​// Support: IE8 
​// Boolean attributes and "value" are not treated correctly 
​if​ ​(​ ​!​div​.​querySelectorAll​(​"​[selected]​"​).​length​ ​)​ ​{ 
rbuggyQSA​.​push​(​ ​"​​[​"​ ​+​ whitespace ​+​ ​"​*(?:value|​"​ ​+​ booleans ​+​ ​"​)​"​ ​)​; 
​} 
 
 
 
// line 1164: 
​// Support: Chrome<29, Android<4.4, Safari<7.0+, iOS<7.0+, PhantomJS<1.9.8+ 
​if​ ​(​ ​!​div​.​querySelectorAll​(​ ​"​[id~=​"​ ​+​ expando ​+​ ​"​‐]​"​ ​).​length​ ​)​ ​{ 
rbuggyQSA​.​push​(​"​~=​"​)​; 
​} 
 
​// Webkit/Opera ‐ :checked should return selected option elements 
 
 
// line 1176: 
​// Support: Safari 8+, iOS 8+ 
​// https://bugs.webkit.org/show_bug.cgi?id=136851 
​// In‐page `selector#id sibing‐combinator selector` fails 
​if​ ​(​ ​!​div​.​querySelectorAll​(​ ​"​a#​"​ ​+​ expando ​+​ ​"​+*​"​ ​).​length​ ​)​ ​{ 
rbuggyQSA​.​push​(​"​.#.+[+~]​"​)​; 
​} 
 
 
// line 1185: 
​// Support: Windows 8 Native Apps 
​// The type and name attributes are restricted during .innerHTML assignment 
​var​ ​input​ ​=​ document​.​createElement​(​"​input​"​)​; 
​input​.​setAttribute​(​ ​"​type​"​,​ ​"​hidden​"​ ​)​; 
div​.​appendChild​(​ ​input​ ​).​setAttribute​(​ ​"​name​"​,​ ​"​D​"​ ​)​; 
 
 
 
// line 1191: 
​// Support: IE8 
​// Enforce case‐sensitivity of name attribute 
​if​ ​(​ div​.​querySelectorAll​(​"​[name=d]​"​).​length​ ​)​ ​{ 
rbuggyQSA​.​push​(​ ​"​name​"​ ​+​ whitespace ​+​ ​"​*[*^$|!~]?=​"​ ​)​; 
​} 
 
 
 
// line 1687: 
​// Support: IE <9 only 
​// Defend against cloned attroperties (jQuery gh‐1709) 
uniqueCache ​=​ outerCache​[​ node​.​uniqueID ​]​ ​|| 
​(​outerCache​[​ node​.​uniqueID ​]​ ​=​ ​{}​)​; 
 
cache ​=​ uniqueCache​[​ type ​]​ ​||​ ​[]​; 
 
 
// line 1716: 
​// Support: IE <9 only 
​// Defend against cloned attroperties (jQuery gh‐1709) 
uniqueCache ​=​ outerCache​[​ node​.​uniqueID ​]​ ​|| 
​(​outerCache​[​ node​.​uniqueID ​]​ ​=​ ​{}​)​; 
 
cache ​=​ uniqueCache​[​ type ​]​ ​||​ ​[]​; 
 
 
// line 1742: 
​// Support: IE <9 only 
​// Defend against cloned attroperties (jQuery gh‐1709) 
uniqueCache ​=​ outerCache​[​ node​.​uniqueID ​]​ ​|| 
​(​outerCache​[​ node​.​uniqueID ​]​ ​=​ ​{}​)​; 
 
uniqueCache​[​ type ​]​ ​=​ ​[​ dirruns​,​ diff ​]​; 
 
 
// line 1953: 
​// Support: IE<8 
​// New HTML5 attribute values (e.g., "search") appear with elem.type === "text" 
​(​ ​(​attr ​=​ elem​.​getAttribute​(​"​type​"​))​ ​==​ ​null​ ​||​ attr​.​toLowerCase​()​ ​===​ ​"​text​"​ ​)​; 
}​, 
 
​// Position‐in‐collection 
 
 
// line 2131: 
​// Support: IE <9 only 
​// Defend against cloned attroperties (jQuery gh‐1709) 
uniqueCache ​=​ outerCache​[​ elem​.​uniqueID ​]​ ​||​ ​(​outerCache​[​ elem​.​uniqueID ​]​ ​=​ ​{}​)​; 
 
​if​ ​(​ ​(​oldCache ​=​ uniqueCache​[​ dir ​])​ ​&& 
oldCache​[​ ​0​ ​]​ ​===​ dirruns ​&&​ oldCache​[​ ​1​ ​]​ ​===​ doneName ​)​ ​{ 
 
 
// line 2371: 
​// Support: IE<9, Safari 
​// Tolerate NodeList properties (IE: "length"; Safari: <number>) matching elements by id 
​for​ ​(​ ​;​ i ​!==​ len ​&&​ ​(​elem ​=​ elems​[​i​])​ ​!=​ ​null​;​ i​++​ ​)​ ​{ 
​if​ ​(​ byElement ​&&​ elem ​)​ ​{ 
j ​=​ ​0​; 
​if​ ​(​ ​!​context ​&&​ elem​.​ownerDocument ​!==​ document ​)​ ​{ 
 
 
// line 2576: 
// Support: Chrome 14‐35+ 
// Always assume duplicates if they aren't passed to the comparison function 
support​.​detectDuplicates ​=​ ​!!​hasDuplicate​; 
 
// Initialize against the default document 
setDocument​()​; 
 
 
// line 2583: 
// Support: Webkit<537.32 ‐ Safari 6.0.3/Chrome 25 (fixed in Chrome 27) 
// Detached nodes confoundingly follow *each other* 
support​.​sortDetached ​=​ assert​(​function​(​ div1 ​)​ ​{ 
​// Should return 1, but returns 4 (following) 
​return​ div1​.​compareDocumentPosition​(​ document​.​createElement​(​"​div​"​)​ ​)​ ​&​ ​1​; 
}​)​; 
 
 
// line 2590: 
// Support: IE<8 
// Prevent attribute/property "interpolation" 
// http://msdn.microsoft.com/en‐us/library/ms536429%28VS.85%29.aspx 
if​ ​(​ ​!​assert​(​function​(​ div ​)​ ​{ 
div​.​innerHTML ​=​ ​"​<a href='#'></a>​"​; 
​return​ div​.​firstChild​.​getAttribute​(​"​href​"​)​ ​===​ ​"​#​"​ ​; 
 
 
// line 2604: 
// Support: IE<9 
// Use defaultValue in place of getAttribute("value") 
if​ ​(​ ​!​support​.​attributes ​||​ ​!​assert​(​function​(​ div ​)​ ​{ 
div​.​innerHTML ​=​ ​"​<input/>​"​; 
div​.​firstChild​.​setAttribute​(​ ​"​value​"​,​ ​""​ ​)​; 
​return​ div​.​firstChild​.​getAttribute​(​ ​"​value​"​ ​)​ ​===​ ​""​; 
 
 
// line 2618: 
// Support: IE<9 
// Use getAttributeNode to fetch booleans when getAttribute lies 
if​ ​(​ ​!​assert​(​function​(​ div ​)​ ​{ 
​return​ div​.​getAttribute​(​"​disabled​"​)​ ​==​ ​null​; 
}​)​ ​)​ ​{ 
addHandle​(​ booleans​,​ ​function​(​ elem​,​ name​,​ isXML ​)​ ​{ 
 
 
// line 3331: 
​// Support: Promises/A+ section 2.3.3.3.3 
​// https://promisesaplus.com/#point‐59 
​// Ignore double‐resolution attempts 
​if​ ​(​ depth ​<​ maxDepth ​)​ ​{ 
​return​; 
​} 
 
 
// line 3340: 
​// Support: Promises/A+ section 2.3.1 
​// https://promisesaplus.com/#point‐48 
​if​ ​(​ returned ​===​ deferred​.​promise​()​ ​)​ ​{ 
​throw​ ​new​ TypeError​(​ ​"​Thenable self‐resolution​"​ ​)​; 
​} 
 
 
 
// line 3346: 
​// Support: Promises/A+ sections 2.3.3.1, 3.5 
​// https://promisesaplus.com/#point‐54 
​// https://promisesaplus.com/#point‐75 
​// Retrieve `then` only once 
then ​=​ returned ​&& 
 
 
 
// line 3352: 
​// Support: Promises/A+ section 2.3.4 
​// https://promisesaplus.com/#point‐64 
​// Only check objects and functions for thenability 
​(​ ​typeof​ returned ​===​ ​"​object​"​ ​|| 
​typeof​ returned ​===​ ​"​function​"​ ​)​ ​&& 
returned​.​then​; 
 
 
// line 3408: 
​// Support: Promises/A+ section 2.3.3.3.4.1 
​// https://promisesaplus.com/#point‐61 
​// Ignore post‐resolution exceptions 
​if​ ​(​ depth ​+​ ​1​ ​>=​ maxDepth ​)​ ​{ 
​// Only substitue handlers pass on context 
​// and multiple values (non‐spec behavior) 
 
 
// line 3425: 
​// Support: Promises/A+ section 2.3.3.3.1 
​// https://promisesaplus.com/#point‐57 
​// Re‐resolve promises immediately to dodge false rejection from 
​// subsequent errors 
​if​ ​(​ depth ​)​ ​{ 
process​()​; 
 
 
// line 4015: 
​// Support: IE11+ 
​// The attrs elements can be null (#14894) 
​if​ ​(​ attrs​[​ i ​]​ ​)​ ​{ 
name ​=​ attrs​[​ i ​].​name​; 
​if​ ​(​ name​.​indexOf​(​ ​"​data‐​"​ ​)​ ​===​ ​0​ ​)​ ​{ 
name ​=​ jQuery​.​camelCase​(​ name​.​slice​(​5​)​ ​)​; 
 
 
// line 4315: 
​// Support: Android 4.0‐4.3 
​// Check state lost if the name is set (#11217) 
​// Support: Windows Web Apps (WWA) 
​// `name` and `type` must use .setAttribute for WWA (#14901) 
​input​.​setAttribute​(​ ​"​type​"​,​ ​"​radio​"​ ​)​; 
​input​.​setAttribute​(​ ​"​checked​"​,​ ​"​checked​"​ ​)​; 
​input​.​setAttribute​(​ ​"​name​"​,​ ​"​t​"​ ​)​; 
 
 
 
// line 4325: 
​// Support: Android<4.2 
​// Older WebKit doesn't clone checked state correctly in fragments 
support​.​checkClone ​=​ div​.​cloneNode​(​ ​true​ ​).​cloneNode​(​ ​true​ ​).​lastChild​.​checked​; 
 
​// Support: IE<=11+ 
​// Make sure textarea (and checkbox) defaultValue is properly cloned 
div​.​innerHTML ​=​ ​"​<textarea>x</textarea>​"​; 
support​.​noCloneChecked ​=​ ​!!​div​.​cloneNode​(​ ​true​ ​).​lastChild​.​defaultValue​; 
}​)​()​; 
 
 
 
// line 4868: 
​// Support: Safari 6.0+ 
​// Target should not be a text node (#504, #13143) 
​if​ ​(​ event​.​target​.​nodeType ​===​ ​3​ ​)​ ​{ 
event​.​target ​=​ event​.​target​.​parentNode​; 
​} 
 
 
 
// line 4919: 
​// Support: Firefox 20+ 
​// Firefox doesn't alert if the returnValue field is not set. 
​if​ ​(​ event​.​result ​!==​ ​undefined​ ​&&​ event​.​originalEvent ​)​ ​{ 
event​.​originalEvent​.​returnValue ​=​ event​.​result​; 
​} 
​} 
 
 
// line 4973: 
​// Support: Android<4.0 
src​.​returnValue ​===​ ​false​ ​? 
returnTrue ​: 
returnFalse​; 
 
​// Event type 
 
 
// line 5038: 
// Support: Safari<7.0 
// Safari doesn't support mouseenter/mouseleave at all. 
// 
// Support: Chrome 34+ 
// Mouseenter doesn't perform while left mouse button is pressed 
// (and initiated outside the observed element) 
// https://code.google.com/p/chromium/issues/detail?id=333868 
jQuery​.​each​(​{ 
mouseenter​:​ ​"​mouseover​"​, 
 
 
// line 5073: 
// Support: Firefox, Chrome, Safari 
// Create "bubbling" focus and blur events 
if​ ​(​ ​!​support​.​focusinBubbles ​)​ ​{ 
jQuery​.​each​(​{​ focus​:​ ​"​focusin​"​,​ blur​:​ ​"​focusout​"​ ​}​,​ ​function​(​ orig​,​ fix ​)​ ​{ 
 
​// Attach a single capturing handler on the document while someone wants 
focusin/focusout 
 
 
// line 5229: 
​// Support: IE9 
option​:​ ​[​ ​1​,​ ​"​<select multiple='multiple'>​"​,​ ​"​</select>​"​ ​], 
 
thead​:​ ​[​ ​1​,​ ​"​<table>​"​,​ ​"​</table>​"​ ​], 
 
​// Some of the following wrappers are not fully defined, because 
 
 
// line 5238: 
​// Support: Android 2.3 
​// Android browser doesn't auto‐insert colgroup 
col​:​ ​[​ ​2​,​ ​"​<table><colgroup>​"​,​ ​"​</colgroup></table>​"​ ​], 
 
​// Auto‐insert "tbody" element 
tr​:​ ​[​ ​2​,​ ​"​<table>​"​,​ ​"​</table>​"​ ​], 
 
 
// line 5251: 
// Support: IE9 
wrapMap​.​optgroup ​=​ wrapMap​.​option​; 
 
wrapMap​.​tbody ​=​ wrapMap​.​tfoot ​=​ wrapMap​.​colgroup ​=​ wrapMap​.​caption ​=​ wrapMap​.​thead​; 
wrapMap​.​th ​=​ wrapMap​.​td​; 
 
 
 
// line 5331: 
​// Support: IE9‐11+ 
​// Use typeof to avoid zero‐argument method invocation on host objects (#15151) 
​var​ ret ​=​ ​typeof​ context​.​getElementsByTagName ​!==​ ​"​undefined​"​ ​? 
context​.​getElementsByTagName​(​ tag ​||​ ​"​*​"​ ​)​ ​: 
​typeof​ context​.​querySelectorAll ​!==​ ​"​undefined​"​ ​? 
context​.​querySelectorAll​(​ tag ​||​ ​"​*​"​ ​)​ ​: 
 
 
// line 5415: 
​// Support: Android<4.1, PhantomJS<2 
​// push.apply(_, arraylike) throws on ancient WebKit 
jQuery​.​merge​(​ nodes​,​ elem​.​nodeType ​?​ ​[​ elem ​]​ ​:​ elem ​)​; 
 
​// Convert non‐html into a text node 
​}​ ​else​ ​if​ ​(​ ​!​rhtml​.​test​(​ elem ​)​ ​)​ ​{ 
 
 
// line 5438: 
​// Support: Android<4.1, PhantomJS<2 
​// push.apply(_, arraylike) throws on ancient WebKit 
jQuery​.​merge​(​ nodes​,​ tmp​.​childNodes ​)​; 
 
​// Remember the top‐level container 
tmp ​=​ fragment​.​firstChild​; 
 
 
// line 5716: 
​// Support: Android<4.1, PhantomJS<2 
​// push.apply(_, arraylike) throws on ancient WebKit 
jQuery​.​merge​(​ scripts​,​ getAll​(​ node​,​ ​"​script​"​ ​)​ ​)​; 
​} 
​} 
 
 
 
// line 5774: 
​// Support: Android<4.1, PhantomJS<2 
​// .get() because push.apply(_, arraylike) throws on ancient WebKit 
push​.​apply​(​ ret​,​ elems​.​get​()​ ​)​; 
​} 
 
​return​ ​this​.​pushStack​(​ ret ​)​; 
 
 
// line 5787: 
​// Support: Firefox 
​// We have to pre‐define these values for FF (#10227) 
HTML​:​ ​"​block​"​, 
BODY​:​ ​"​block​" 
​}​; 
 
 
 
// line 5832: 
​// Support: IE 
doc​.​write​()​; 
doc​.​close​()​; 
 
display ​=​ actualDisplay​(​ nodeName​,​ doc ​)​; 
iframe​.​detach​()​; 
 
 
// line 5851: 
​// Support: IE<=11+, Firefox<=30+ (#15098, #14150) 
​// IE throws on elements created in popups 
​// FF meanwhile throws on frame elements through "defaultView.getComputedStyle" 
​if​ ​(​ elem​.​ownerDocument​.​defaultView​.​opener ​)​ ​{ 
​return​ elem​.​ownerDocument​.​defaultView​.​getComputedStyle​(​ elem​,​ ​null​ ​)​; 
​} 
 
 
// line 5895: 
​// Support: IE9‐11+ 
​// Style of cloned element affects source element cloned (#8908) 
div​.​style​.​backgroundClip ​=​ ​"​content‐box​"​; 
div​.​cloneNode​(​ ​true​ ​).​style​.​backgroundClip ​=​ ​""​; 
support​.​clearCloneStyle ​=​ div​.​style​.​backgroundClip ​===​ ​"​content‐box​"​; 
 
 
 
// line 5909: 
​// Support: Android 2.3 
​// Vendor‐prefix box‐sizing 
​"​‐webkit‐box‐sizing:border‐box;box‐sizing:border‐box;​"​ ​+ 
​"​display:block;position:absolute;​"​ ​+ 
​"​margin:0;margin‐top:1%;margin‐right:50%;​"​ ​+ 
​"​border:1px;padding:1px;​"​ ​+ 
 
 
// line 5942: 
​// Support: Android 4.0‐4.3 
​// We're checking for boxSizingReliableVal here instead of pixelMarginRightVal 
​// since that compresses better and they're computed together anyway. 
​if​ ​(​ boxSizingReliableVal ​==​ ​null​ ​)​ ​{ 
computeStyleTests​()​; 
​} 
 
 
// line 5952: 
​// Support: Android 2.3 
​// Check if div with explicit width and no margin‐right incorrectly 
​// gets computed margin‐right based on width of container. (#3333) 
​// WebKit Bug 13343 ‐ getComputedStyle returns wrong value for margin‐right 
​// This support function is only executed once so no memoizing is needed. 
​var​ ret​, 
 
 
// line 5962: 
​// Support: Android 2.3 
​// Vendor‐prefix box‐sizing 
​"​‐webkit‐box‐sizing:content‐box;box‐sizing:content‐box;​"​ ​+ 
​"​display:block;margin:0;border:0;padding:0​"​; 
marginDiv​.​style​.​marginRight ​=​ marginDiv​.​style​.​width ​=​ ​"​0​"​; 
div​.​style​.​width ​=​ ​"​1px​"​; 
 
 
// line 5987: 
​// Support: IE9 
​// getPropertyValue is only needed for .css('filter') (#12537) 
​if​ ​(​ computed ​)​ ​{ 
ret ​=​ computed​.​getPropertyValue​(​ name ​)​ ​||​ computed​[​ name ​]​; 
​} 
 
 
 
// line 6023: 
​// Support: IE9‐11+ 
​// IE returns zIndex value as an integer. 
ret ​+​ ​""​ ​: 
ret​; 
} 
 
 
 
// line 6315: 
​// Support: IE9‐11+ 
​// background‐* props affect original clone's values 
​if​ ​(​ ​!​support​.​clearCloneStyle ​&&​ value ​===​ ​""​ ​&&​ name​.​indexOf​(​ ​"​background​"​ ​) 
===​ ​0​ ​)​ ​{ 
style​[​ name ​]​ ​=​ ​"​inherit​"​; 
​} 
 
 
 
// line 6407: 
// Support: Android 2.3 
jQuery​.​cssHooks​.​marginRight ​=​ addGetHookIf​(​ support​.​reliableMarginRight​, 
​function​(​ elem​,​ computed ​)​ ​{ 
​if​ ​(​ computed ​)​ ​{ 
​return​ swap​(​ elem​,​ ​{​ ​"​display​"​:​ ​"​inline‐block​"​ ​}​, 
curCSS​,​ ​[​ elem​,​ ​"​marginRight​"​ ​]​ ​)​; 
 
 
// line 6576: 
// Support: IE9 
// Panic based approach to setting things on disconnected nodes 
Tween​.​propHooks​.​scrollTop ​=​ Tween​.​propHooks​.​scrollLeft ​=​ ​{ 
set​:​ ​function​(​ tween ​)​ ​{ 
​if​ ​(​ tween​.​elem​.​nodeType ​&&​ tween​.​elem​.​parentNode ​)​ ​{ 
tween​.​elem​[​ tween​.​prop ​]​ ​=​ tween​.​now​; 
 
 
// line 6852: 
​// Support: Android 2.3 
​// Archaic crash bug won't allow us to use `1 ‐ ( 0.5 || 0 )` (#12497) 
temp ​=​ remaining ​/​ animation​.​duration ​||​ ​0​, 
percent ​=​ ​1​ ​‐​ temp​, 
​index​ ​=​ ​0​, 
​length​ ​=​ animation​.​tweens​.​length​; 
 
 
// line 7235: 
​// Support: Android<4.4 
​// Default value for a checkbox should be "on" 
support​.​checkOn ​=​ ​input​.​value ​!==​ ​""​; 
 
​// Support: IE<=11+ 
​// Must access selectedIndex to make default options select 
support​.​optSelected ​=​ opt​.​selected​; 
 
​// Support: Android<=2.3 
​// Options inside disabled selects are incorrectly marked as disabled 
select​.​disabled ​=​ ​true​; 
support​.​optDisabled ​=​ ​!​opt​.​disabled​; 
 
​// Support: IE<=11+ 
​// An input loses its value after becoming a radio 
​input​ ​=​ document​.​createElement​(​ ​"​input​"​ ​)​; 
​input​.​value ​=​ ​"​t​"​; 
​input​.​type ​=​ ​"​radio​"​; 
support​.​radioValue ​=​ ​input​.​value ​===​ ​"​t​"​; 
 
 
// line 7712: 
​// Support: IE<11 
​// option.value not trimmed (#14858) 
​return​ jQuery​.​trim​(​ elem​.​value ​)​; 
​} 
​}​, 
select​:​ ​{ 
 
 
// line 7848: 
// Support: Android 2.3 
// Workaround failure to string‐cast null input 
jQuery​.​parseJSON ​=​ ​function​(​ data ​)​ ​{ 
​return​ JSON​.​parse​(​ data ​+​ ​""​ ​)​; 
}; 
 
 
 
// line 7862: 
​// Support: IE9 
​try​ ​{ 
xml ​=​ ​(​ ​new​ window​.​DOMParser​()​ ​).​parseFromString​(​ data​,​ ​"​text/xml​"​ ​)​; 
​}​ ​catch​ ​(​ e ​)​ ​{ 
xml ​=​ ​undefined​; 
​} 
 
 
// line 8371: 
​// Support: IE8‐11+ 
​// IE throws exception if url is malformed, e.g. http://example.com:80x/ 
​try​ ​{ 
urlAnchor​.​href ​=​ s​.​url​; 
​// Support: IE8‐11+ 
​// Anchor's host property isn't correctly set when s.url is relative 
urlAnchor​.​href ​=​ urlAnchor​.​href​; 
s​.​crossDomain ​=​ originAnchor​.​protocol ​+​ ​"​//​"​ ​+​ originAnchor​.​host ​!== 
urlAnchor​.​protocol ​+​ ​"​//​"​ ​+​ urlAnchor​.​host​; 
​}​ ​catch​ ​(​ e ​)​ ​{ 
 
 
// line 8883: 
​// Support: IE9 
​// #1450: sometimes IE returns 1223 when it should be 204 
​1223​:​ ​204 
​}​, 
xhrSupported ​=​ jQuery​.​ajaxSettings​.​xhr​()​; 
 
 
 
// line 8889: 
// Support: IE9 
// Open requests must be manually aborted on unload (#5280) 
// See https://support.microsoft.com/kb/2856746 for more info 
if​ ​(​ window​.​attachEvent ​)​ ​{ 
window​.​attachEvent​(​ ​"​onunload​"​,​ ​function​()​ ​{ 
​for​ ​(​ ​var​ key ​in​ xhrCallbacks ​)​ ​{ 
 
 
// line 8967: 
​// Support: IE9 
​// Accessing binary‐data responseText throws an exception 
​// (#11426) 
​typeof​ xhr​.​responseText ​===​ ​"​string​"​ ​?​ ​{ 
text​:​ xhr​.​responseText 
​}​ ​:​ ​undefined​, 
 
 
// line 9478: 
// Support: Safari<7+, Chrome<37+ 
// Add the top/left cssHooks using jQuery.fn.position 
// Webkit bug: https://bugs.webkit.org/show_bug.cgi?id=29084 
// Blink bug: https://code.google.com/p/chromium/issues/detail?id=229280 
// getComputedStyle returns percent when specified for top/left/bottom/right; 
// rather than make the css module depend on the offset module, just check for it here 
 
 
Workarounds from Feb 2014 
● L40​ -- Firefox 18+ 
● L255​ -- IE < 9 
● L282​ -- Firefox <20 
● L312​ --  Android < 4, iOS < 6 / old Chrome/Safari issue 
● L692​ -- Firefox 
 
● L3483​ -- WebKit issue, fixed in recent WebKit? 
○ Android 2.3 Support 
● L3484​ -- Same as L3483 
● L3520​ -- WebKit issue, fixed in recent WebKit? 
○ Android 2.3 Support 
● L3977​ -- WebKit issue 
○ Android 2.3 Support 
● L3980​ -- WebKit issue, fixed in recent WebKit? 
○ Android 2.3 Support 
● L3985​ -- IE >=9 issue 
● L4399​ -- Firefox making the mistake of following the DOM2 Events spec  
○ There's active discussion of how this should all work 
 ​https://bugzilla.mozilla.org/show_bug.cgi?id=968265 
● L4508​ -- Buggy WebKit port issue 
● L4514​ -- WebKit issue 
● L4565​ -- Firefox issue.  Filed ​https://bugzilla.mozilla.org/show_bug.cgi?id=968065  
● L4618​ -- Android issue 
○ Android 2.3 Support 
● L4673​ -- Chrome issue 
● L4701​ -- Firefox issue. Filed ​https://bugzilla.mozilla.org/show_bug.cgi?id=687787 
● L4854​ -- IE < 10 issue 
● L4865​ -- IE < 10 issue 
● L4871​ -- Not a browser issue at all, afaict 
○ In addition to API surface, jQuery 2.x must provide the exact same semantics as jQuery 1.x, which means 
we’re forced to carry the burden of some ancient issues such as node wrapping requirements. These are 
the most frustrating, but thankfully minimal. 
● L4955​ and ​L4975​ -- IE >=9 issue 
● L5027​ -- WebKit Qt port issue 
● L5050​ -- WebKit Qt port issue 
● L5058​ -- WebKit issue 
● L5303​ -- WebKit issue 
● L5338​ -- WebKit Qt port issue 
● L5395​ -- WebKit Qt port issue 
● L5421​ -- Optimization using Firefox-only experimental API 
● L5448​ -- WebKit and Firefox issue 
○ I wrote this several years ago and honestly don’t remember what the issue I encountered was—will 
reconfirm ASAP. RW 
● L5451​ -- IE issue of some sort?  Not clear 
● L5481​ -- IE9 issue 
● L5493​ -- iOS issue 
○ Needs reconfirmation 
● L5516​ -- IE < 8 issue 
● L5545​ -- Firefox issue (fixed in v29). Filed ​https://bugzilla.mozilla.org/show_bug.cgi?id=243412​> 
○ Android 2.3 Support 
● L5563​ -- Same Firefox/Old WebKit issue as L5545 
○ Android 2.3 Support 
● L5593​ -- Android issue 
○ Android 2.3 Support 
● L5654​ -- Vendor prefixes 
○ Yeah, that part needs to die, but will take a few years.  :( 
○ Everyone is guilty. 
● L5907​ -- WebKit issue (Chrome/Safari) 
○ Needs reconfirmation 
● L5988​ -- Android issue 
○ Android 2.3 Support 
● L6156​ -- IE < 11 issue 
● L6322​ -- IE 9-10 issue 
● L6831​ -- Android/OldWebKit issue 
○ Android 2.3 Support 
● L6836​ -- IE issue (all versions) 
● L6840​ -- WebKit issue 
● L6845​ -- IE issue (all versions) 
● L6942​ -- IE< 10 issue 
● L7049​ -- IE issue (old?) 
● L7084​ -- classList cannot be used 
○ Android 2.3 support 
● L7316​ -- IE < 10 issue 
● L7372​ -- WebKit issue 
● L7425​ -- Android issue 
○ Android 2.3 Support 
● L7439​ -- IE < 10 issue 
● L7489​ -- IE< 9 issue 
● L7495​ -- IE fallback for L7489r 
● L8332​ -- Opera < 12.15 (Presto) issue 
● L8457​ -- IE < 10 issue 
● L8462​ -- IE < 10 issue 
● L8533​ -- IE9 issue 
● L8801​ -- IE issue (same origin policy?) 
● L8998​ -- WebKit issue 
● L9028​ -- Mobile Safari issue 
 
Sizzle Issues 
● L709​ -- Android WebKit bug (possibly fixed?) 
○ Android 2.3 Support 
○ Sizzle Specific 
● L720​ -- IE < 9 
○ Sizzle Specific 
● L763​ --  IE < 8/Opera/WebKit bugs 
○ Sizzle Specific 
● L799​ -- qSA issue 
● L802​ -- IE8 bug 
○ Sizzle Specific 
● L886​ -- IE6  memory leak fix 
○ Sizzle Specific 
● L917​ -- IE-specific API used for optimization 
○ Sizzle Specific 
● L997​ -- IE < 9 issue? 
○ Sizzle Specific 
● L1024​ -- IE issue 
○ Sizzle Specific 
● L1027​ -- IE < 9 issue 
○ Sizzle Specific 
● L1029​ -- IE11 removed attachEvent but old IE doesn't have addEventListener 
○ Sizzle Specific 
● L1044​ -- IE < 8 issue 
○ Sizzle Specific 
● L1064​ -- Safari < 4 issue 
○ Sizzle Specific 
● L1067​ -- Opera < 10  issue 
○ Sizzle Specific 
● L1072​ -- IE < 10 issue 
○ Sizzle Specific 
● L1098​ -- IE < 8 issue 
○ Sizzle Specific 
● L1149​ -- IE9/Opera11.5 issue 
○ Sizzle Specific 
● L1153​ -- IE < 10 qSA issue 
○ Sizzle Specific 
● L1164​ -- IE setup for line L1176 
○ Sizzle Specific 
● L1170​ -- IE 8 / Opera 10-12 qSA issue 
○ Sizzle Specific 
● L1176​ -- IE8 qSA issue 
○ Sizzle Specific 
● L1182​ -- WebKit/Opera qSA issue 
○ Sizzle Specific 
● L1184​ -- IE8 qSA issue 
○ Sizzle Specific 
● L1191​ -- Windows 8 Native Apps issue 
○ Sizzle Specific 
● L1197​ -- IE8 qSA issue 
○ Sizzle Specific 
● L1204​ -- IE8/Firefox 3.5 qSA issue 
○ Sizzle Specific 
● L1209​ -- Opera < 12 qSA issue 
○ Sizzle Specific 
● L1215​ -- Lack of unprefixed matches()/matchesSelector() in UAs? 
○ Sizzle Specific 
● L1222​ -- IE 9 issue and Firefox (old?) issue 
○ Sizzle Specific 
● L1386​ -- IE 9 issue 
○ Sizzle Specific 
● L1389​ -- IE < 10 workaround for (L1286) 
○ Sizzle Specific 
● L1869​ -- Safari issue 
○ Sizzle Specific 
● L1914​ -- IE < 8 issue 
○ Sizzle Specific 
● L2316​ -- IE < 9 issue, Safari issue 
○ Sizzle Specific 
● L2498​ -- Chrome issue, fixed in Chrome 14? 
○ Sizzle Specific 
● L2505​ -- WebKit issue, fixed in Chrome 27 
○ Sizzle Specific 
● L2512​ -- IE < 8 issue 
○ Sizzle Specific 
● L2526​ -- IE < 9 issue 
○ Sizzle Specific 
● L2540​ -- IE <9 issue 
○ Sizzle Specific 
 
 
Special Notations:  
Android 2.3 Support.  
Many Webkit issues may be fixed in the latest version however we're obligated to maintain support in jQuery 2.x for the 
code required to fix those issues as far back as 2.3 due to mobile market share.  
Sizzle Specific. 
Some issues that appear to be in the "jQuery source" are actually the Sizzle source that is added to the list of files 
concatenated together when jQuery is built. Sizzle must work the ​same way in all browsers​ as far back as IE6, but can also 
be omitted via a ​custom build​, for modern (ie. qSA without bug fixes) browser support. 
 

More Related Content

PDF
Wikis biblio
PDF
The Use of Open Source Javascript Libraries with CICS.
PDF
Distributed wikis
PDF
freeport-mcmoran copper& gold Certificate of Designations for Freeport-McMoR...
PDF
Safari Web Content Guide
PDF
Joomla 2.5 made easy (Englisch)
ODT
Trotti.ch - Webshop (Elektro Scooter) Trotti Model 2015
PDF
Merkblatt Bankleitzahlen
Wikis biblio
The Use of Open Source Javascript Libraries with CICS.
Distributed wikis
freeport-mcmoran copper& gold Certificate of Designations for Freeport-McMoR...
Safari Web Content Guide
Joomla 2.5 made easy (Englisch)
Trotti.ch - Webshop (Elektro Scooter) Trotti Model 2015
Merkblatt Bankleitzahlen

Similar to J query's browser bug workarounds (20)

PDF
Web apps of the future
PPTX
PDF
Do you need jQuery in 2019?
PDF
What is jQuery?
PDF
The Future is Static
PDF
JavaScript Libraries (Ajax Exp 2006)
PPTX
Web summit.pptx
PDF
JavaScript Library Overview (Ajax Exp West 2007)
PDF
Os php-wiki1-pdf
PDF
Selecting the Best Javascript Web Framework
PDF
Developpement informatique
PDF
Devfest09 Cschalk Gwt
PDF
jQuery - Chapter 1 - Introduction
PPTX
Starting with jQuery
PPT
Js frameworks
PDF
React Js vs Node Js_ Which Framework to Choose for Your Next Web Application
PDF
Top 10 Best Web Development Technologies
PPTX
PDF
PDF
PrairieDevCon 2014 - Web Doesn't Mean Slow
Web apps of the future
Do you need jQuery in 2019?
What is jQuery?
The Future is Static
JavaScript Libraries (Ajax Exp 2006)
Web summit.pptx
JavaScript Library Overview (Ajax Exp West 2007)
Os php-wiki1-pdf
Selecting the Best Javascript Web Framework
Developpement informatique
Devfest09 Cschalk Gwt
jQuery - Chapter 1 - Introduction
Starting with jQuery
Js frameworks
React Js vs Node Js_ Which Framework to Choose for Your Next Web Application
Top 10 Best Web Development Technologies
PrairieDevCon 2014 - Web Doesn't Mean Slow
Ad

More from Thanh Nguyen (20)

PDF
2021.02.05 Trotti.ch - E-Scooter Webshop
PDF
Trotti GmbH - Elektroroller Wintermodelle 2020-2021
PDF
GNU-Handbuch (Manual) zum Schutze der Privatsphäre
PDF
Apple URL Scheme Reference
PDF
Technics Turntable System SL-1200 MK5 Operating Instructions
PDF
Technics Turntable System SL-1210 M5G Operating Instructions
PDF
Technics Turntable System SL-1200 GLD Operating Instructions
PDF
BC-Bankenstamm
PDF
PayPal Website Payments Standard Integration Guide
PDF
Public Social Media Profile Pictures Winter 2013
PDF
Public Social Media Profile Pictures Summer 2013
PDF
Implementing Remote Procedure Calls
PDF
The beginners guide to SEO
PDF
Europass XML Schema v3.0 RC7 Documentation and Changelog
PDF
PDF
DTA Standards und Formate
PDF
Wer, wie, was - die Theorie hinter den W-Fragen
PDF
Apple Mac Pro
PDF
Shopgate Vortrag
PDF
Datenträgeraustauschverfahren (DTA)
2021.02.05 Trotti.ch - E-Scooter Webshop
Trotti GmbH - Elektroroller Wintermodelle 2020-2021
GNU-Handbuch (Manual) zum Schutze der Privatsphäre
Apple URL Scheme Reference
Technics Turntable System SL-1200 MK5 Operating Instructions
Technics Turntable System SL-1210 M5G Operating Instructions
Technics Turntable System SL-1200 GLD Operating Instructions
BC-Bankenstamm
PayPal Website Payments Standard Integration Guide
Public Social Media Profile Pictures Winter 2013
Public Social Media Profile Pictures Summer 2013
Implementing Remote Procedure Calls
The beginners guide to SEO
Europass XML Schema v3.0 RC7 Documentation and Changelog
DTA Standards und Formate
Wer, wie, was - die Theorie hinter den W-Fragen
Apple Mac Pro
Shopgate Vortrag
Datenträgeraustauschverfahren (DTA)
Ad

Recently uploaded (20)

PPTX
KSS ON CYBERSECURITY INCIDENT RESPONSE AND PLANNING MANAGEMENT.pptx
PDF
Exploring The Internet Of Things(IOT).ppt
PPTX
curriculumandpedagogyinearlychildhoodcurriculum-171021103104 - Copy.pptx
PDF
Top 8 Trusted Sources to Buy Verified Cash App Accounts.pdf
PDF
Understand the Gitlab_presentation_task.pdf
PPT
12 Things That Make People Trust a Website Instantly
PDF
Alethe Consulting Corporate Profile and Solution Aproach
PDF
The Ikigai Template _ Recalibrate How You Spend Your Time.pdf
PPT
250152213-Excitation-SystemWERRT (1).ppt
PDF
si manuel quezon at mga nagawa sa bansang pilipinas
PPTX
Introduction to cybersecurity and digital nettiquette
PPTX
module 1-Part 1.pptxdddddddddddddddddddddddddddddddddddd
PDF
BIOCHEM CH2 OVERVIEW OF MICROBIOLOGY.pdf
PPT
FIRE PREVENTION AND CONTROL PLAN- LUS.FM.MQ.OM.UTM.PLN.00014.ppt
PDF
Smart Home Technology for Health Monitoring (www.kiu.ac.ug)
PDF
mera desh ae watn.(a source of motivation and patriotism to the youth of the ...
PDF
Session 1 (Week 1)fghjmgfdsfgthyjkhfdsadfghjkhgfdsa
PPTX
Cyber Hygine IN organizations in MSME or
PDF
Alethe Consulting Corporate Profile and Solution Aproach
PPTX
Top Website Bugs That Hurt User Experience – And How Expert Web Design Fixes
KSS ON CYBERSECURITY INCIDENT RESPONSE AND PLANNING MANAGEMENT.pptx
Exploring The Internet Of Things(IOT).ppt
curriculumandpedagogyinearlychildhoodcurriculum-171021103104 - Copy.pptx
Top 8 Trusted Sources to Buy Verified Cash App Accounts.pdf
Understand the Gitlab_presentation_task.pdf
12 Things That Make People Trust a Website Instantly
Alethe Consulting Corporate Profile and Solution Aproach
The Ikigai Template _ Recalibrate How You Spend Your Time.pdf
250152213-Excitation-SystemWERRT (1).ppt
si manuel quezon at mga nagawa sa bansang pilipinas
Introduction to cybersecurity and digital nettiquette
module 1-Part 1.pptxdddddddddddddddddddddddddddddddddddd
BIOCHEM CH2 OVERVIEW OF MICROBIOLOGY.pdf
FIRE PREVENTION AND CONTROL PLAN- LUS.FM.MQ.OM.UTM.PLN.00014.ppt
Smart Home Technology for Health Monitoring (www.kiu.ac.ug)
mera desh ae watn.(a source of motivation and patriotism to the youth of the ...
Session 1 (Week 1)fghjmgfdsfgthyjkhfdsadfghjkhgfdsa
Cyber Hygine IN organizations in MSME or
Alethe Consulting Corporate Profile and Solution Aproach
Top Website Bugs That Hurt User Experience – And How Expert Web Design Fixes

J query's browser bug workarounds