My question is quite similar to What is the purpose of a self executing function in javascript?, however it concerns userscripts (specifically for GreaseMonkey) instead.
I see that some userscripts are distributed with this pattern, and some are not.
Example of script with the IIFE pattern: (source)
// ==UserScript==
// (...)
// ==/UserScript==
(function(){
// if <condition>
document.location.href += '?sk=h_chr';
// ...
})();
Example of script without it: (source)
// ==UserScript==
// (...)
// ==/UserScript==
window.location.href = "https://www.facebook.com/?sk=h_chr";
In addition, I also found that the "New script" template from TamperMonkey follows it, while the templates from GreaseMonkey and ViolentMonkey do not.
The question is, then, is the IIFE pattern any useful when writing userscripts?
Especially, if my script is in strict
mode, and I use let
instead of var
. In any case, as far as I know, functions and variables defined in userscripts are not made available in the global page scope.
Thanks.
In general, no; the IIFE pattern is seldom useful for wrapping a whole userscript (see edge cases below). That's a throwback to many years ago when some engines (briefly) did not wrap scripts by default.
In fact, if you include the obsolete @unwrap
directive, the script engines will all now ignore it.
Here are some reasons to use the IIFE pattern:
strict
mode in old versions of Violentmonkey (2018 or earlier) for the whole script.Parsing error: 'return' outside of function
warning if you use BOTH: (1) A script-wide return
and (2) an external LINTer.Consider this test script:
// ==UserScript==
// @name _Scope and Strict-Mode Demo script
// @match https://stackoverflow.com/*
// @unwrap
// @grant none
// ==/UserScript==
/* eslint-disable no-multi-spaces, curly */
'use strict';
if (location.pathname.includes("/users/") ) {
console.log ("Terminating script early.");
return; // In external LINTers, this will cause a harmless warning.
}
var cantSeeMeInConsole = "neener neener";
window.canSomestimesSeeMe = "Howdy";
console.log (`In Strict mode: ${bInStrictMode() }; \`cantSeeMeInConsole\`: ${cantSeeMeInConsole}`);
function bInStrictMode () {
var inStrict = false;
var dummyObj = {};
Object.defineProperty (dummyObj, 'foo', {value: "bar", writable: false } );
try { dummyObj.foo = "fee"; }
catch (e) { inStrict = true; }
return inStrict;
}
In all cases, the userscript is scoped/wrapped. The page can't see code, nor variables like cantSeeMeInConsole
.
Beware that script page conflicts can still occur in @grant none
mode.
Additional isolations apply, depending on: (a) the userscript engine, (b) the browser, and (c) the @grant
mode.
For example, using Greasemonkey, or changing the grant mode kills the page's ability to see canSomestimesSeeMe
.
'use strict';
up top like that switches the whole userscript into strict mode.In a related note, if the script does not use @run-at
settings, there is no point in using $(document).ready()
or its shorthand.