DOMContentLoaded
so that enhancements can be made as an element is rendered to the screen allowing users to begin interacting with a page even if it hasn’t finished loading or if rendering is being blocked by 3rd party assets such as banner ads or tracking code. The most robust way to do this is to include an inline <script>
directly after each element you need to enhance which calls the relevant JavaScript functions — this works, but it’s not very DRY and it’s not easy to maintain.Recently, I stumbled across a hack in an article by David Walsh which uses a CSS animation and the
animationStart
event to detect when an element has been inserted into the DOM. It works by exploiting the fact that CSS animations run as soon as an element is appended to the document. Creating a simple animation that runs for a tiny duration and applying it to any elements you to want modify will fire a animationStart
event for each element as it’s inserted, essentially emulating the (now defunct) DOM mutation events. What’s great about this technique is it also works during page load!The trick
Let’s say we want to enhance all<input>
fields:<input type="text" size="50">
We’ll need to set up a CSS animation and apply it to every <input>
field with CSS: /* prefixes omitted for brevity */
input {
animation-name: nodeReady;
animation-duration: 0.001s;
}
@keyframes nodeReady {
from { clip: rect(1px, auto, auto, auto); }
to { clip: rect(0px, auto, auto, auto); }
}
Finally, we need to make our “enhancements” with JavaScript:/* prefixed variants omitted for brevity */
document.addEventListener("animationstart", function(e) {
if (e.animationName == "nodeReady") {
e.target.value = new Date();
}
}, false);
That’s it, if the browser supports CSS animations all <input>
elements will be modified as the page loads.Fallbacks
If the browser doesn’t support CSS animations we need a fallback to ensure theses elements are still modified. Handling this case is actually really simple, just useDOMContentLoaded
or the window load
event to fire the code you would have run in the animationStart
event. In this situation things will behave the way they always have.A demo
Here’s a simple demo of this techinque in action. I’ve used PHP buffering to simulate a slow connection.Credit
Credit must go to David Walsh for his article and to Daniel Buchner for the original discovery. Also, James Allardice is working on wrapping this up into something easily deployable.POST: Working with elements before the DOM is ready