window.Modernizr = (
function
( window, document, undefined ) {
var
version =
'2.7.1'
,
Modernizr = {},
enableClasses =
true
,
docElement = document.documentElement,
mod =
'modernizr'
,
modElem = document.createElement(mod),
mStyle = modElem.style,
inputElem
= document.createElement(
'input'
)
,
smile =
':)'
,
toString = {}.toString,
prefixes =
' -webkit- -moz- -o- -ms- '
.split(
' '
),
omPrefixes =
'Webkit Moz O ms'
,
cssomPrefixes = omPrefixes.split(
' '
),
domPrefixes = omPrefixes.toLowerCase().split(
' '
),
tests = {},
inputs = {},
attrs = {},
classes = [],
slice = classes.slice,
featureName,
injectElementWithStyles =
function
( rule, callback, nodes, testnames ) {
var
style, ret, node, docOverflow,
div = document.createElement(
'div'
),
body = document.body,
fakeBody = body || document.createElement('body
');
if ( parseInt(nodes, 10) ) {
// In order not to give false positives we create a node for each test
// This also allows the method to scale for unspecified uses
while ( nodes-- ) {
node = document.createElement('
div
');
node.id = testnames ? testnames[nodes] : mod + (nodes + 1);
div.appendChild(node);
}
}
// <style> elements in IE6-9 are considered '
NoScope
' elements and therefore will be removed
// when injected with innerHTML. To get around this you need to prepend the '
NoScope
' element
// with a '
scoped
' element, in our case the soft-hyphen entity as it won'
t mess
with
our measurements.
style = [
'­'
,
'<style id="s'
, mod,
'">'
, rule,
'</style>'
].join(
''
);
div.id = mod;
(body ? div : fakeBody).innerHTML += style;
fakeBody.appendChild(div);
if
( !body ) {
fakeBody.style.background =
''
;
fakeBody.style.overflow =
'hidden'
;
docOverflow = docElement.style.overflow;
docElement.style.overflow =
'hidden'
;
docElement.appendChild(fakeBody);
}
ret = callback(div, rule);
if
( !body ) {
fakeBody.parentNode.removeChild(fakeBody);
docElement.style.overflow = docOverflow;
}
else
{
div.parentNode.removeChild(div);
}
return
!!ret;
},
testMediaQuery =
function
( mq ) {
var
matchMedia = window.matchMedia || window.msMatchMedia;
if
( matchMedia ) {
return
matchMedia(mq).matches;
}
var
bool;
injectElementWithStyles('@media
' + mq + '
{
#' + mod + ' { position: absolute; } }', function( node ) {
bool = (window.getComputedStyle ?
getComputedStyle(node,
null
) :
node.currentStyle)['position
'] == '
absolute
';
});
return bool;
},
/*>>mq*/
/*>>hasevent*/
//
// isEventSupported determines if a given element supports the given event
// kangax.github.com/iseventsupported/
//
// The following results are known incorrects:
// Modernizr.hasEvent("webkitTransitionEnd", elem) // false negative
// Modernizr.hasEvent("textInput") // in Webkit. github.com/Modernizr/Modernizr/issues/333
// ...
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
( !is(element[eventName],
'undefined'
) ) {
element[eventName] = undefined;
}
element.removeAttribute(eventName);
}
}
element =
null
;
return
isSupported;
}
return
isEventSupported;
})(),
_hasOwnProperty = ({}).hasOwnProperty, hasOwnProp;
if
( !is(_hasOwnProperty, 'undefined
') && !is(_hasOwnProperty.call, '
undefined
') ) {
hasOwnProp = function (object, property) {
return _hasOwnProperty.call(object, property);
};
}
else {
hasOwnProp = function (object, property) { /* yes, this can give false positives/negatives, but most of the time we don'
t care about those */
return
((property
in
object) && is(object.constructor.prototype[property],
'undefined'
));
};
}
if
(!Function.prototype.bind) {
Function.prototype.bind =
function
bind(that) {
var
target =
this
;
if
(
typeof
target !=
"function"
) {
throw
new
TypeError();
}
var
args = slice.call(arguments, 1),
bound =
function
() {
if
(
this
instanceof
bound) {
var
F =
function
(){};
F.prototype = target.prototype;
var
self =
new
F();
var
result = target.apply(
self,
args.concat(slice.call(arguments))
);
if
(Object(result) === result) {
return
result;
}
return
self;
}
else
{
return
target.apply(
that,
args.concat(slice.call(arguments))
);
}
};
return
bound;
};
}
function
setCss( str ) {
mStyle.cssText = str;
}
function
setCssAll( str1, str2 ) {
return
setCss(prefixes.join(str1 +
';'
) + ( str2 ||
''
));
}
function
is( obj, type ) {
return
typeof
obj === type;
}
function
contains( str, substr ) {
return
!!~(
''
+ str).indexOf(substr);
}
function
testProps( props, prefixed ) {
for
(
var
i
in
props ) {
var
prop = props[i];
if
( !contains(prop,
"-"
) && mStyle[prop] !== undefined ) {
return
prefixed == 'pfx
' ? prop : true;
}
}
return false;
}
/*>>testprop*/
// TODO :: add testDOMProps
/**
* testDOMProps is a generic DOM property test; if a browser supports
* a certain property, it won'
t
return
undefined
for
it.
*/
function
testDOMProps( props, obj, elem ) {
for
(
var
i
in
props ) {
var
item = obj[props[i]];
if
( item !== undefined) {
if
(elem ===
false
)
return
props[i];
if
(is(item, '
function
')){
// default to autobind unless override
return item.bind(elem || obj);
}
// return the unbound function or obj or value
return item;
}
}
return false;
}
/*>>testallprops*/
/**
* testPropsAll tests a list of DOM properties we want to check against.
* We specify literally ALL possible (known and/or likely) properties on
* the element including the non-vendor prefixed one, for forward-
* compatibility.
*/
function testPropsAll( prop, prefixed, elem ) {
var ucProp = prop.charAt(0).toUpperCase() + prop.slice(1),
props = (prop + '
' + cssomPrefixes.join(ucProp + '
') + ucProp).split('
');
// did they call .prefixed('
boxSizing
') or are we just testing a prop?
if(is(prefixed, "string") || is(prefixed, "undefined")) {
return testProps(props, prefixed);
// otherwise, they called .prefixed('
requestAnimationFrame
', window[, elem])
} else {
props = (prop + '
' + (domPrefixes).join(ucProp + '
') + ucProp).split('
');
return testDOMProps(props, prefixed, elem);
}
}
/*>>testallprops*/
/**
* Tests
* -----
*/
// The *new* flexbox
// dev.w3.org/csswg/css3-flexbox
tests['
flexbox
'] = function() {
return testPropsAll('
flexWrap
');
};
// The *old* flexbox
// www.w3.org/TR/2009/WD-css3-flexbox-20090723/
tests['
flexboxlegacy
'] = function() {
return testPropsAll('
boxDirection
');
};
// On the S60 and BB Storm, getContext exists, but always returns undefined
// so we actually have to call getContext() to verify
// github.com/Modernizr/Modernizr/issues/issue/97/
tests['
canvas
'] = function() {
var elem = document.createElement('
canvas
');
return !!(elem.getContext && elem.getContext('
2d
'));
};
tests['
canvastext
'] = function() {
return !!(Modernizr['
canvas
'] && is(document.createElement('
canvas
').getContext('
2d
').fillText, '
function
'));
};
// webk.it/70117 is tracking a legit WebGL feature detect proposal
// We do a soft detect which may false positive in order to avoid
// an expensive context creation: bugzil.la/732441
tests['
webgl
'] = function() {
return !!window.WebGLRenderingContext;
};
/*
* The Modernizr.touch test only indicates if the browser supports
* touch events, which does not necessarily reflect a touchscreen
* device, as evidenced by tablets running Windows 7 or, alas,
* the Palm Pre / WebOS (touch) phones.
*
* Additionally, Chrome (desktop) used to lie about its support on this,
* but that has since been rectified: crbug.com/36415
*
* We also test for Firefox 4 Multitouch Support.
*
* For more info, see: modernizr.github.com/Modernizr/touch.html
*/
tests['
touch
'] = function() {
var bool;
if(('
ontouchstart
' in window) || window.DocumentTouch && document instanceof DocumentTouch) {
bool = true;
} else {
injectElementWithStyles(['
@media (
',prefixes.join('
touch-enabled),(
'),mod,'
)
','
{
#modernizr{top:9px;position:absolute}}'].join(''), function( node ) {
bool = node.offsetTop === 9;
});
}
return
bool;
};
tests['geolocation
'] = function() {
return '
geolocation
' in navigator;
};
tests['
postmessage
'] = function() {
return !!window.postMessage;
};
// Chrome incognito mode used to throw an exception when using openDatabase
// It doesn'
t anymore.
tests[
'websqldatabase'
] =
function
() {
return
!!window.openDatabase;
};
tests[
'indexedDB'
] =
function
() {
return
!!testPropsAll(
"indexedDB"
, window);
};
tests[
'hashchange'
] =
function
() {
return
isEventSupported(
'hashchange'
, window) && (document.documentMode === undefined || document.documentMode > 7);
};
tests[
'history'
] =
function
() {
return
!!(window.history && history.pushState);
};
tests[
'draganddrop'
] =
function
() {
var
div = document.createElement(
'div'
);
return
(
'draggable'
in
div) || (
'ondragstart'
in
div &&
'ondrop'
in
div);
};
tests['websockets
'] = function() {
return '
WebSocket
' in window || '
MozWebSocket
' in window;
};
// css-tricks.com/rgba-browser-support/
tests['
rgba
'] = function() {
// Set an rgba() color and check the returned value
setCss('
background-color:rgba(150,255,150,.5)
');
return contains(mStyle.backgroundColor, '
rgba
');
};
tests['
hsla
'] = function() {
// Same as rgba(), in fact, browsers re-map hsla() to rgba() internally,
// except IE9 who retains it as hsla
setCss('
background-color:hsla(120,40%,100%,.5)
');
return contains(mStyle.backgroundColor, '
rgba
') || contains(mStyle.backgroundColor, '
hsla
');
};
tests['
multiplebgs
'] = function() {
// Setting multiple images AND a color on the background shorthand property
// and then querying the style.background property value for the number of
// occurrences of "url(" is a reliable method for detecting ACTUAL support for this!
setCss('
background:url(https:
return
(/(url\s*\(.*?){3}/).test(mStyle.background);
};
tests['backgroundsize
'] = function() {
return testPropsAll('
backgroundSize
');
};
tests['
borderimage
'] = function() {
return testPropsAll('
borderImage
');
};
// Super comprehensive table about all the unique implementations of
// border-radius: muddledramblings.com/table-of-css3-border-radius-compliance
tests['
borderradius
'] = function() {
return testPropsAll('
borderRadius
');
};
// WebOS unfortunately false positives on this test.
tests['
boxshadow
'] = function() {
return testPropsAll('
boxShadow
');
};
// FF3.0 will false positive on this test
tests['
textshadow
'] = function() {
return document.createElement('
div
').style.textShadow === '
';
};
tests['
opacity
'] = function() {
// Browsers that actually have CSS Opacity implemented have done so
// according to spec, which means their return values are within the
// range of [0.0,1.0] - including the leading zero.
setCssAll('
opacity:.55
');
// The non-literal . in this regex is intentional:
// German Chrome returns this value as 0,55
// github.com/Modernizr/Modernizr/issues/#issue/59/comment/516632
return (/^0.55$/).test(mStyle.opacity);
};
// Note, Android < 4 will pass this test, but can only animate
// a single property at a time
// daneden.me/2011/12/putting-up-with-androids-bullshit/
tests['
cssanimations
'] = function() {
return testPropsAll('
animationName
');
};
tests['
csscolumns
'] = function() {
return testPropsAll('
columnCount
');
};
tests['
cssgradients
'] = function() {
/**
* For CSS Gradients syntax, please see:
* webkit.org/blog/175/introducing-css-gradients/
* developer.mozilla.org/en/CSS/-moz-linear-gradient
* developer.mozilla.org/en/CSS/-moz-radial-gradient
* dev.w3.org/csswg/css3-images/#gradients-
*/
var str1 = '
background-image:
',
str2 = '
gradient(linear,left top,right bottom,
from
(
#9f9),to(white));',
str3 = 'linear-gradient(left top,
#9f9, white);';
setCss(
(str1 + '-webkit-
'.split('
').join(str2 + str1) +
// standard syntax // trailing '
background-image:
'
prefixes.join(str3 + str1)).slice(0, -str1.length)
);
return contains(mStyle.backgroundImage, '
gradient
');
};
tests['
cssreflections
'] = function() {
return testPropsAll('
boxReflect
');
};
tests['
csstransforms
'] = function() {
return !!testPropsAll('
transform
');
};
tests['
csstransforms3d
'] = function() {
var ret = !!testPropsAll('
perspective
');
// Webkit'
s 3D transforms are passed off to the browser
's own graphics renderer.
// It works fine in Safari on Leopard and Snow Leopard, but not in Chrome in
// some conditions. As a result, Webkit typically recognizes the syntax but
// will sometimes throw a false positive, thus we must do a more thorough check:
if ( ret && '
webkitPerspective
' in docElement.style ) {
// Webkit allows this media query to succeed only if the feature is enabled.
// `@media (transform-3d),(-webkit-transform-3d){ ... }`
injectElementWithStyles('
@media (transform-3d),(-webkit-transform-3d){
#modernizr{left:9px;position:absolute;height:3px;}}', function( node, rule ) {
ret = node.offsetLeft === 9 && node.offsetHeight === 3;
});
}
return
ret;
};
tests['csstransitions
'] = function() {
return testPropsAll('
transition
');
};
/*>>fontface*/
// @font-face detection routine by Diego Perini
// javascript.nwbox.com/CSSSupport/
// false positives:
// WebOS github.com/Modernizr/Modernizr/issues/342
// WP7 github.com/Modernizr/Modernizr/issues/538
tests['
fontface
'] = function() {
var bool;
injectElementWithStyles('
@font-face {font-family:
"font"
;src:url(
"https://"
)}
', function( node, rule ) {
var style = document.getElementById('
smodernizr
'),
sheet = style.sheet || style.styleSheet,
cssText = sheet ? (sheet.cssRules && sheet.cssRules[0] ? sheet.cssRules[0].cssText : sheet.cssText || '
') : '
';
bool = /src/i.test(cssText) && cssText.indexOf(rule.split('
')[0]) === 0;
});
return bool;
};
/*>>fontface*/
// CSS generated content detection
tests['
generatedcontent
'] = function() {
var bool;
injectElementWithStyles(['
#',mod,'{font:0/0 a}#',mod,':after{content:"',smile,'";visibility:hidden;font:3px/1 a}'].join(''), function( node ) {
bool = node.offsetHeight >= 3;
});
return
bool;
};
tests[
'video'
] =
function
() {
var
elem = document.createElement(
'video'
),
bool =
false
;
try
{
if
( bool = !!elem.canPlayType ) {
bool =
new
Boolean(bool);
bool.ogg = elem.canPlayType(
'video/ogg; codecs="theora"'
) .replace(/^no$/,
''
);
bool.h264 = elem.canPlayType(
'video/mp4; codecs="avc1.42E01E"'
) .replace(/^no$/,
''
);
bool.webm = elem.canPlayType(
'video/webm; codecs="vp8, vorbis"'
).replace(/^no$/,
''
);
}
}
catch
(e) { }
return
bool;
};
tests[
'audio'
] =
function
() {
var
elem = document.createElement(
'audio'
),
bool =
false
;
try
{
if
( bool = !!elem.canPlayType ) {
bool =
new
Boolean(bool);
bool.ogg = elem.canPlayType(
'audio/ogg; codecs="vorbis"'
).replace(/^no$/,
''
);
bool.mp3 = elem.canPlayType(
'audio/mpeg;'
) .replace(/^no$/,
''
);
bool.wav = elem.canPlayType(
'audio/wav; codecs="1"'
) .replace(/^no$/,
''
);
bool.m4a = ( elem.canPlayType(
'audio/x-m4a;'
) ||
elem.canPlayType(
'audio/aac;'
)) .replace(/^no$/,
''
);
}
}
catch
(e) { }
return
bool;
};
tests[
'localstorage'
] =
function
() {
try
{
localStorage.setItem(mod, mod);
localStorage.removeItem(mod);
return
true
;
}
catch
(e) {
return
false
;
}
};
tests[
'sessionstorage'
] =
function
() {
try
{
sessionStorage.setItem(mod, mod);
sessionStorage.removeItem(mod);
return
true
;
}
catch
(e) {
return
false
;
}
};
tests[
'webworkers'
] =
function
() {
return
!!window.Worker;
};
tests[
'applicationcache'
] =
function
() {
return
!!window.applicationCache;
};
tests[
'svg'
] =
function
() {
return
!!document.createElementNS && !!document.createElementNS(ns.svg,
'svg'
).createSVGRect;
};
tests[
'inlinesvg'
] =
function
() {
var
div = document.createElement(
'div'
);
div.innerHTML =
'<svg/>'
;
return
(div.firstChild && div.firstChild.namespaceURI) == ns.svg;
};
tests[
'smil'
] =
function
() {
return
!!document.createElementNS && /SVGAnimate/.test(toString.call(document.createElementNS(ns.svg,
'animate'
)));
};
tests[
'svgclippaths'
] =
function
() {
return
!!document.createElementNS && /SVGClipPath/.test(toString.call(document.createElementNS(ns.svg,
'clipPath'
)));
};
function
webforms() {
Modernizr['input
'] = (function( props ) {
for ( var i = 0, len = props.length; i < len; i++ ) {
attrs[ props[i] ] = !!(props[i] in inputElem);
}
if (attrs.list){
// safari false positive'
s on datalist: webk.it/74252
attrs.list = !!(document.createElement(
'datalist'
) && window.HTMLDataListElement);
}
return
attrs;
})(
'autocomplete autofocus list placeholder max min multiple pattern required step'
.split(
' '
));
Modernizr[
'inputtypes'
] = (
function
(props) {
for
(
var
i = 0, bool, inputElemType, defaultView, len = props.length; i < len; i++ ) {
inputElem.setAttribute(
'type'
, inputElemType = props[i]);
bool = inputElem.type !==
'text'
;
if
( bool ) {
inputElem.value = smile;
inputElem.style.cssText = 'position:absolute;visibility:hidden;
';
if ( /^range$/.test(inputElemType) && inputElem.style.WebkitAppearance !== undefined ) {
docElement.appendChild(inputElem);
defaultView = document.defaultView;
// Safari 2-4 allows the smiley as a value, despite making a slider
bool = defaultView.getComputedStyle &&
defaultView.getComputedStyle(inputElem, null).WebkitAppearance !== '
textfield
' &&
// Mobile android web browser has false positive, so must
// check the height to see if the widget is actually there.
(inputElem.offsetHeight !== 0);
docElement.removeChild(inputElem);
} else if ( /^(search|tel)$/.test(inputElemType) ){
// Spec doesn'
t define any special parsing or detectable UI
}
else
if
( /^(url|email)$/.test(inputElemType) ) {
bool = inputElem.checkValidity && inputElem.checkValidity() ===
false
;
}
else
{
bool = inputElem.value != smile;
}
}
inputs[ props[i] ] = !!bool;
}
return
inputs;
})('search tel url email datetime date month week time datetime-local number range color
'.split('
'));
/*>>inputtypes*/
}
/*>>webforms*/
// End of test definitions
// -----------------------
// Run through all tests and detect their support in the current UA.
// todo: hypothetically we could be doing an array of tests and use a basic loop here.
for ( var feature in tests ) {
if ( hasOwnProp(tests, feature) ) {
// run the test, throw the return value into the Modernizr,
// then based on that boolean, define an appropriate className
// and push it into an array of classes we'
ll join later.
featureName = feature.toLowerCase();
Modernizr[featureName] = tests[feature]();
classes.push((Modernizr[featureName] ?
''
:
'no-'
) + featureName);
}
}
Modernizr.input || webforms();
Modernizr.addTest =
function
( feature, test ) {
if
(
typeof
feature ==
'object'
) {
for
(
var
key
in
feature ) {
if
( hasOwnProp( feature, key ) ) {
Modernizr.addTest( key, feature[ key ] );
}
}
}
else
{
feature = feature.toLowerCase();
if
( Modernizr[feature] !== undefined ) {
return
Modernizr;
}
test =
typeof
test ==
'function'
? test() : test;
if
(
typeof
enableClasses !==
"undefined"
&& enableClasses) {
docElement.className +=
' '
+ (test ?
''
:
'no-'
) + feature;
}
Modernizr[feature] = test;
}
return
Modernizr;
};
setCss(
''
);
modElem = inputElem =
null
;
;(
function
(window, document) {
var
version =
'3.7.0'
;
var
options = window.html5 || {};
var
reSkip = /^<|^(?:button|map|select|textarea|object|iframe|option|optgroup)$/i;
var
saveClones = /^(?:a|b|code|div|fieldset|h1|h2|h3|h4|h5|h6|i|label|li|ol|p|q|span|strong|style|table|tbody|td|th|tr|ul)$/i;
var
supportsHtml5Styles;
var
expando =
'_html5shiv'
;
var
expanID = 0;
var
expandoData = {};
var
supportsUnknownElements;
(
function
() {
try
{
var
a = document.createElement(
'a'
);
a.innerHTML =
'<xyz></xyz>'
;
supportsHtml5Styles = (
'hidden'
in
a);
supportsUnknownElements = a.childNodes.length == 1 || (
function
() {
(document.createElement)(
'a'
);
var
frag = document.createDocumentFragment();
return
(
typeof
frag.cloneNode ==
'undefined'
||
typeof
frag.createDocumentFragment ==
'undefined'
||
typeof
frag.createElement ==
'undefined'
);
}());
}
catch
(e) {
supportsHtml5Styles =
true
;
supportsUnknownElements =
true
;
}
}());
function
addStyleSheet(ownerDocument, cssText) {
var
p = ownerDocument.createElement(
'p'
),
parent = ownerDocument.getElementsByTagName(
'head'
)[0] || ownerDocument.documentElement;
p.innerHTML =
'x<style>'
+ cssText +
'</style>'
;
return
parent.insertBefore(p.lastChild, parent.firstChild);
}
function
getElements() {
var
elements = html5.elements;
return
typeof
elements ==
'string'
? elements.split(
' '
) : elements;
}
function
getExpandoData(ownerDocument) {
var
data = expandoData[ownerDocument[expando]];
if
(!data) {
data = {};
expanID++;
ownerDocument[expando] = expanID;
expandoData[expanID] = data;
}
return
data;
}
function
createElement(nodeName, ownerDocument, data){
if
(!ownerDocument) {
ownerDocument = document;
}
if
(supportsUnknownElements){
return
ownerDocument.createElement(nodeName);
}
if
(!data) {
data = getExpandoData(ownerDocument);
}
var
node;
if
(data.cache[nodeName]) {
node = data.cache[nodeName].cloneNode();
}
else
if
(saveClones.test(nodeName)) {
node = (data.cache[nodeName] = data.createElem(nodeName)).cloneNode();
}
else
{
node = data.createElem(nodeName);
}
return
node.canHaveChildren && !reSkip.test(nodeName) && !node.tagUrn ? data.frag.appendChild(node) : node;
}
function
createDocumentFragment(ownerDocument, data){
if
(!ownerDocument) {
ownerDocument = document;
}
if
(supportsUnknownElements){
return
ownerDocument.createDocumentFragment();
}
data = data || getExpandoData(ownerDocument);
var
clone = data.frag.cloneNode(),
i = 0,
elems = getElements(),
l = elems.length;
for
(;i<l;i++){
clone.createElement(elems[i]);
}
return
clone;
}
function
shivMethods(ownerDocument, data) {
if
(!data.cache) {
data.cache = {};
data.createElem = ownerDocument.createElement;
data.createFrag = ownerDocument.createDocumentFragment;
data.frag = data.createFrag();
}
ownerDocument.createElement =
function
(nodeName) {
if
(!html5.shivMethods) {
return
data.createElem(nodeName);
}
return
createElement(nodeName, ownerDocument, data);
};
ownerDocument.createDocumentFragment = Function(
'h,f'
,
'return function(){'
+
'var n=f.cloneNode(),c=n.createElement;'
+
'h.shivMethods&&('
+
getElements().join().replace(/[\w\-]+/g,
function
(nodeName) {
data.createElem(nodeName);
data.frag.createElement(nodeName);
return
'c("'
+ nodeName +
'")'
;
}) +
');return n}'
)(html5, data.frag);
}
function
shivDocument(ownerDocument) {
if
(!ownerDocument) {
ownerDocument = document;
}
var
data = getExpandoData(ownerDocument);
if
(html5.shivCSS && !supportsHtml5Styles && !data.hasCSS) {
data.hasCSS = !!addStyleSheet(ownerDocument,
'article,aside,dialog,figcaption,figure,footer,header,hgroup,main,nav,section{display:block}'
+
'mark{background:#FF0;color:#000}'
+
'template{display:none}'
);
}
if
(!supportsUnknownElements) {
shivMethods(ownerDocument, data);
}
return
ownerDocument;
}
var
html5 = {
'elements'
: options.elements ||
'abbr article aside audio bdi canvas data datalist details dialog figcaption figure footer header hgroup main mark meter nav output progress section summary template time video'
,
'version'
: version,
'shivCSS'
: (options.shivCSS !==
false
),
'supportsUnknownElements'
: supportsUnknownElements,
'shivMethods
': (options.shivMethods !== false),
/**
* A string to describe the type of `html5` object ("default" or "default print").
* @memberOf html5
* @type String
*/
'
type
': '
default
',
// shivs the document according to the specified `html5` object options
'
shivDocument
': shivDocument,
//creates a shived element
createElement: createElement,
//creates a shived documentFragment
createDocumentFragment: createDocumentFragment
};
/*--------------------------------------------------------------------------*/
// expose html5
window.html5 = html5;
// shiv the document
shivDocument(document);
}(this, document));
/*>>shiv*/
// Assign private properties to the return object with prefix
Modernizr._version = version;
// expose these for the plugin API. Look in the source for how to join() them against your input
/*>>prefixes*/
Modernizr._prefixes = prefixes;
/*>>prefixes*/
/*>>domprefixes*/
Modernizr._domPrefixes = domPrefixes;
Modernizr._cssomPrefixes = cssomPrefixes;
/*>>domprefixes*/
/*>>mq*/
// Modernizr.mq tests a given media query, live against the current state of the window
// A few important notes:
// * If a browser does not support media queries at all (eg. oldIE) the mq() will always return false
// * A max-width or orientation query will be evaluated against the current state, which may change later.
// * You must specify values. Eg. If you are testing support for the min-width media query use:
// Modernizr.mq('
(min-width:0)
')
// usage:
// Modernizr.mq('
only screen and (max-width:768)
')
Modernizr.mq = testMediaQuery;
/*>>mq*/
/*>>hasevent*/
// Modernizr.hasEvent() detects support for a given event, with an optional element to test on
// Modernizr.hasEvent('
gesturestart
', elem)
Modernizr.hasEvent = isEventSupported;
/*>>hasevent*/
/*>>testprop*/
// Modernizr.testProp() investigates whether a given style property is recognized
// Note that the property names must be provided in the camelCase variant.
// Modernizr.testProp('
pointerEvents
')
Modernizr.testProp = function(prop){
return testProps([prop]);
};
/*>>testprop*/
/*>>testallprops*/
// Modernizr.testAllProps() investigates whether a given style property,
// or any of its vendor-prefixed variants, is recognized
// Note that the property names must be provided in the camelCase variant.
// Modernizr.testAllProps('
boxSizing
')
Modernizr.testAllProps = testPropsAll;
/*>>testallprops*/
/*>>teststyles*/
// Modernizr.testStyles() allows you to add custom styles to the document and test an element afterwards
// Modernizr.testStyles('
#modernizr { position:absolute }', function(elem, rule){ ... })
Modernizr.testStyles = injectElementWithStyles;
Modernizr.prefixed =
function
(prop, obj, elem){
if
(!obj) {
return
testPropsAll(prop,
'pfx'
);
}
else
{
return
testPropsAll(prop, obj, elem);
}
};
docElement.className = docElement.className.replace(/(^|\s)no-js(\s|$)/,
'$1$2'
) +
(enableClasses ?
' js '
+ classes.join(
' '
) :
''
);
return
Modernizr;
})(
this
,
this
.document);