2 * This file is where we decide whether to initialise the Grade A run-time.
4 * - Beware: This file MUST parse without errors on even the most ancient of browsers!
7 /* global mw, isCompatible, $VARS, $CODE */
10 * See <https://www.mediawiki.org/wiki/Compatibility#Browsers>
12 * Capabilities required for modern run-time:
14 * - DOM Level 4 & Selectors API Level 1
15 * - HTML5 & Web Storage
16 * - DOM Level 2 Events
18 * Browsers we support in our modern run-time (Grade A):
24 * - Mobile Safari 6.0+ (iOS 6+)
27 * Browsers we support in our no-javascript run-time (Grade C):
33 * - Mobile Safari 5.0+ (iOS 4+)
37 * - Symbian-based browsers
38 * - NetFront-based browser
40 * - Nokia's Ovi Browser
43 * - UC Mini (speed mode on)
45 * Other browsers that pass the check are considered Grade X.
47 * @param {string} [str] User agent, defaults to navigator.userAgent
48 * @return {boolean} User agent is compatible with MediaWiki JS
50 window
.isCompatible = function ( str
) {
51 var ua
= str
|| navigator
.userAgent
;
53 // https://caniuse.com/#feat=es5
54 // https://caniuse.com/#feat=use-strict
55 // https://caniuse.com/#feat=json / https://phabricator.wikimedia.org/T141344#2784065
58 return !this && !!Function
.prototype.bind
&& !!window
.JSON
;
61 // https://caniuse.com/#feat=queryselector
62 'querySelector' in document
&&
64 // https://caniuse.com/#feat=namevalue-storage
65 // https://developer.blackberry.com/html5/apis/v1_0/localstorage.html
66 // https://blog.whatwg.org/this-week-in-html-5-episode-30
67 'localStorage' in window
&&
69 // https://caniuse.com/#feat=addeventlistener
70 'addEventListener' in window
&&
72 // Hardcoded exceptions for browsers that pass the requirement but we don't want to
73 // support in the modern run-time.
74 // Note: Please extend the regex instead of adding new ones
76 ua
.match( /MSIE 10|webOS\/1\.[0-4]|SymbianOS|Series60|NetFront|Opera Mini|S40OviBrowser|MeeGo|Android.+Glass|^Mozilla\/5\.0 .+ Gecko\/$|googleweblight/ ) ||
77 ua
.match( /PlayStation/i )
82 // Conditional script injection
85 if ( !isCompatible() ) {
86 // Undo class swapping in case of an unsupported browser.
87 // See ResourceLoaderClientHtml::getDocumentAttributes().
88 document
.documentElement
.className
= document
.documentElement
.className
89 .replace( /(^|\s)client-js(\s|$)/, '$1client-nojs$2' );
91 NORLQ
= window
.NORLQ
|| [];
92 while ( NORLQ
.length
) {
96 push: function ( fn
) {
101 // Clear and disable the other queue
111 * The $CODE and $VARS placeholders are substituted in ResourceLoaderStartUpModule.php.
114 mw
.config
= new mw
.Map( $VARS
.wgLegacyJavaScriptGlobals
);
116 $CODE
.registrations();
118 mw
.config
.set( $VARS
.configuration
);
120 // Must be after mw.config.set because these callbacks may use mw.loader which
121 // needs to have values 'skin', 'debug' etc. from mw.config.
122 // eslint-disable-next-line vars-on-top
123 var RLQ
= window
.RLQ
|| [];
124 while ( RLQ
.length
) {
128 push: function ( fn
) {
133 // Clear and disable the other queue
140 if ( window
.performance
&& performance
.mark
) {
141 performance
.mark( 'mwStartup' );
143 script
= document
.createElement( 'script' );
144 script
.src
= $VARS
.baseModulesUri
;
145 script
.onload = function () {
147 script
.onload
= null;
152 document
.head
.appendChild( script
);