The browser allows us to track the loading of external resources â scripts, iframes, pictures and so on.
There are two events for it:
onloadâ successful load,onerrorâ an error occurred.
Loading a script
Letâs say we need to load a third-party script and call a function that resides there.
We can load it dynamically, like this:
let script = document.createElement('script');
script.src = "my.js";
document.head.append(script);
â¦But how to run the function that is declared inside that script? We need to wait until the script loads, and only then we can call it.
For our own scripts we could use JavaScript modules here, but they are not widely adopted by third-party libraries.
script.onload
The main helper is the load event. It triggers after the script was loaded and executed.
For instance:
let script = document.createElement('script');
// can load any script, from any domain
script.src = "https://cdnjs.cloudflare.com/ajax/libs/lodash.js/4.3.0/lodash.js"
document.head.append(script);
script.onload = function() {
// the script creates a variable "_"
alert( _.VERSION ); // shows library version
};
So in onload we can use script variables, run functions etc.
â¦And what if the loading failed? For instance, thereâs no such script (error 404) or the server is down (unavailable).
script.onerror
Errors that occur during the loading of the script can be tracked in an error event.
For instance, letâs request a script that doesnât exist:
let script = document.createElement('script');
script.src = "https://example.com/404.js"; // no such script
document.head.append(script);
script.onerror = function() {
alert("Error loading " + this.src); // Error loading https://example.com/404.js
};
Please note that we canât get HTTP error details here. We donât know if it was an error 404 or 500 or something else. Just that the loading failed.
Events onload/onerror track only the loading itself.
Errors that may occur during script processing and execution are out of scope for these events. That is: if a script loaded successfully, then onload triggers, even if it has programming errors in it. To track script errors, one can use window.onerror global handler.
Other resources
The load and error events also work for other resources, basically for any resource that has an external src.
For example:
let img = document.createElement('img');
img.src = "https://js.cx/clipart/train.gif"; // (*)
img.onload = function() {
alert(`Image loaded, size ${img.width}x${img.height}`);
};
img.onerror = function() {
alert("Error occurred while loading image");
};
There are some notes though:
- Most resources start loading when they are added to the document. But
<img>is an exception. It starts loading when it gets a src(*). - For
<iframe>, theiframe.onloadevent triggers when the iframe loading finished, both for successful load and in case of an error.
Thatâs for historical reasons.
Crossorigin policy
Thereâs a rule: scripts from one site canât access contents of the other site. So, e.g. a script at https://facebook.com canât read the userâs mailbox at https://gmail.com.
Or, to be more precise, one origin (domain/port/protocol triplet) canât access the content from another one. So even if we have a subdomain, or just another port, these are different origins with no access to each other.
This rule also affects resources from other domains.
If weâre using a script from another domain, and thereâs an error in it, we canât get error details.
For example, letâs take a script error.js that consists of a single (bad) function call:
// ð error.js
noSuchFunction();
Now load it from the same site where itâs located:
<script>
window.onerror = function(message, url, line, col, errorObj) {
alert(`${message}\n${url}, ${line}:${col}`);
};
</script>
<script src="/article/onload-onerror/crossorigin/error.js"></script>
We can see a good error report, like this:
Uncaught ReferenceError: noSuchFunction is not defined
https://javascript.info/article/onload-onerror/crossorigin/error.js, 1:1
Now letâs load the same script from another domain:
<script>
window.onerror = function(message, url, line, col, errorObj) {
alert(`${message}\n${url}, ${line}:${col}`);
};
</script>
<script src="https://cors.javascript.info/article/onload-onerror/crossorigin/error.js"></script>
The report is different, like this:
Script error.
, 0:0
Details may vary depending on the browser, but the idea is the same: any information about the internals of a script, including error stack traces, is hidden. Exactly because itâs from another domain.
Why do we need error details?
There are many services (and we can build our own) that listen for global errors using window.onerror, save errors and provide an interface to access and analyze them. Thatâs great, as we can see real errors, triggered by our users. But if a script comes from another origin, then thereâs not much information about errors in it, as weâve just seen.
Similar cross-origin policy (CORS) is enforced for other types of resources as well.
To allow cross-origin access, the <script> tag needs to have the crossorigin attribute, plus the remote server must provide special headers.
There are three levels of cross-origin access:
- No
crossoriginattribute â access prohibited. crossorigin="anonymous"â access allowed if the server responds with the headerAccess-Control-Allow-Originwith*or our origin. Browser does not send authorization information and cookies to remote server.crossorigin="use-credentials"â access allowed if the server sends back the headerAccess-Control-Allow-Originwith our origin andAccess-Control-Allow-Credentials: true. Browser sends authorization information and cookies to remote server.
You can read more about cross-origin access in the chapter Fetch: Cross-Origin Requests. It describes the fetch method for network requests, but the policy is exactly the same.
Such thing as âcookiesâ is out of our current scope, but you can read about them in the chapter Cookies, document.cookie.
In our case, we didnât have any crossorigin attribute. So the cross-origin access was prohibited. Letâs add it.
We can choose between "anonymous" (no cookies sent, one server-side header needed) and "use-credentials" (sends cookies too, two server-side headers needed).
If we donât care about cookies, then "anonymous" is the way to go:
<script>
window.onerror = function(message, url, line, col, errorObj) {
alert(`${message}\n${url}, ${line}:${col}`);
};
</script>
<script crossorigin="anonymous" src="https://cors.javascript.info/article/onload-onerror/crossorigin/error.js"></script>
Now, assuming that the server provides an Access-Control-Allow-Origin header, everythingâs fine. We have the full error report.
Summary
Images <img>, external styles, scripts and other resources provide load and error events to track their loading:
loadtriggers on a successful load,errortriggers on a failed load.
The only exception is <iframe>: for historical reasons it always triggers load, for any load completion, even if the page is not found.
The readystatechange event also works for resources, but is rarely used, because load/error events are simpler.
ÙØ¸Ø±Ø§Øª
<code>Ø§Ø³ØªÙØ§Ø¯Ù Ú©ÙÛØ¯Ø Ø¨Ø±Ø§Û ÚÙØ¯Û٠خط â کد را درÙ٠تگ<pre>ÙØ±Ø§Ø± دÙÛØ¯Ø Ø¨Ø±Ø§Û Ø¨ÛØ´ از د٠خط کد â Ø§Ø² ÛÚ© جعبÙÙ Ø´ÙÛ Ø§Ø³ØªÙØ§Ø¯Ù Ú©ÙÛØ¯. (plnkrØ jsbinØ codepenâ¦)