Aim to cover rendering and animation performance.
Both in the DOM and also general 2D and 3D animation in the web.
I will touch on many different things and hope to provide some inspiration.
The ideal feeling is that of a native experience.
The goal of performance is so that the user doesn't notice the web page.
I'll begin with rendering the DOM then move onto some practises which help for animated visuals.
The user shouldn't feel their very expensive device is struggling to display a simple a web page.
Goal framerate of 60fps, which gives us 16ms to render a single frame.
When the frame takes longer to layout and render than 16ms that is jank.
It looks like a stuttery discontinuity in the animation.
Jank shows up in the timeline as very tall bars
It's been shown that 60fps with occasional hiccoups feels much slower than a continuous 30fps.
Long frames caused by too much blocking activity in the main thread.
Can be an intense cpu bound operation or blocking IO (localstorage, sync ajax)
However most commonly on a typical webpage slowness is caused by the layouts and paints used to render the page.
A very simple view of the render pipeline is.
🐢 Layout is slowest.
🐢 An element needs to be repainted when it's appearance changes such as font-family or background colour. If an element's layout changes its also needs to be repainted.
🐰 Composite is very fast. Composite only involves the graphics card rerendering from bitmaps calculated in paint and stored in memory: it is performed when the page is being drawn to the screen. It is triggered by gfx card-only operations such as transform, clipping and opacity.
CSS triggers is a great resource for discovering how properties trigger each of these operations.
One should try to touch layout properties rarely but can be pretty free with animating compositing properties.
Writing to layout properties is free; you can write as much as you want.
It'll only be recalculated on render or when one of its layout properties is read. (scaleHeight)
One common problem is reading, then writing in a loop, which is known as layout thrashing;
The layout is invalidated and recalculated on every single read.
Interleaving reads and writes could mean multiple expensive layout operations per frame.
One fix for DOM thrashing tends to be fairly simple: batch your reads and writes all together. Especially don't change a property then immediately read it.
For example don't append to innerhtml
in a loop INSTEAD build up a string and then write that string to the innerhtml
.
In an mvc or a large framework with many independent modules it can be difficult to ensure modules do not interleave read and writes. Wilson Page's fastdom library can help with this so that reads and writes in a single frame all get done together.
It does this by allowing you to add your read and write callback functions to seperate queues these queues are flushed once per frame. So that it performs all the reads then all of the writes.
Thus giving you an async api to turn interleaved read and writes into queue of layout reads then layout writes.
Sometimes one wants to animate an element growing or sliding in.
Naively one might put a transition on an elements height or left css properties.
Animating properties like these which effect layout will trigger a relayout on every frame.
For example this awful scene here: [slide]
Although it doesn't look like much (and is probably rendering pretty smooth) it is making my fan spin up and get's my phone really hot. With more content or images this would be evident as Jank.
In this particular example I am animating the 'width' property of only a few elements. This is one of the properties which triggers layout. It is being changed and so requires a re-layout every single frame by the css animation.
A good general motto is to calculate all DOM changes first then apply them in a single step.
Changing a single element could have an effect on every single other element on the page.
Every touched element also to be repainted.
Typeface slide
Another reason is the rendering of typefaces. Type is so ubiquitous about the web it is very easily to take it for granted but it is very difficult to render especially when you appreciate that every character needs to be kerned and hinted, every word and paragraph needs to be placed and flowed correctly.
Composite slide
Compositing is great. Transforms such as Rotating or scaling are all done off the main thread.
They are performed on your phone/desktop's graphics chip.
Graphics chips are very good at applying transforms.
The transforms on a single webpage are nothing compared to a modern computer game.
Even though this modal looks isolated... the browser may still relayout the whole Dom as it's content changes.
However, we can seperate its layout invalidation from the rest of the page. Doing so doesn't accelerate the animation, but it does make the layout change less catastophic.
There is a draft css spec called containment which provides a hint to the browser that this element will not affect the rest of the DOM tree.
The containment spec requires certain properties, such as no scrolling and fixed dimensions.
{
height: <fixed value>;
width: <fixed value or a %>;
overflow: hidden;
position: absolute;
contain: strict; // In draft
}
Containment can be done in browsers currently. But it relies on undocumented browser behaviour. It requires the element to have certain properties before it can be optimised,
and is entirely dependent upon the browser's implementation.
The containment property would be really good because it makes explicit as a performance enhancement
what was previously only doable by setting a variety of properties.
The performance gains from these could not be relied upon in every browser and using styling hacks for performance feels messy.
Containment will enforce certain styling neccessary for the isolation. Which means it could potentially cause an element to be styled differently to how the user expects. E.g. no scrolling.
Layout Containment is not the same as style containment as defined in the web components spec.
layout containment is a css property in draft. It is to aid performance by allowing the browser to isolate certain elements from the rest of the DOM's layout tree.
The Web for many years was single threaded. Now we have web workers they allow us to offload some work into additional threads on the cpu. So we can leave the main thread to focus on performing the rendering operations.
We can also use workers to do complex calculations which do not rely on interacting with the DOM.
Compression library to expand data recieved in an ajax request. This data can then be processed in the worker and queried from the main thread.
On the lighter side; If your website's winter holiday theme has realtime physically correct snowfall then perhaps perform these calculations in a worker and transfer the positions/rotations of each snowflake back to the main thread for rendering.
But beware, there is a hidden cost of using workers. Transferring the data back to the main thread is not free. It needs to be deserialised in the main thread. So if you download a JSON file and parze it in a worker then send the whole object back to the main thread you gain NOTHING. The main thread still has to deserialize back into a JavaScript object.
So make sure you only send what you need for rendering back to the main thread.
function workerMessage(message) {
// This wraps the message posting/response in a promise, which will resolve if the response doesn't
// contain an error, and reject with the error if it does. If you'd prefer, it's possible to call
// controller.postMessage() and set up the onmessage handler independently of a promise, but this is
// a convenient wrapper.
return new Promise(function(resolve, reject) {
const messageChannel = new MessageChannel();
messageChannel.port1.onmessage = function(event) {
if (event.data.error) {
messageChannel.port1.onmessage = undefined;
reject(event.data.error);
} else {
messageChannel.port1.onmessage = undefined;
resolve(event.data);
}
};
// This sends the message data as well as transferring messageChannel.port2 to the service worker.
// The service worker can then use the transferred port to reply via postMessage(), which
// will in turn trigger the onmessage handler on messageChannel.port1.
// See https://html.spec.whatwg.org/multipage/workers.html#dom-worker-postmessage
myWorker.postMessage(message, [messageChannel.port2]);
});
}
workerMessage({
action: 'doThing',
myVar: 4
})
.then(data => {
console.log(data.response);
});
// Recieve messages from the client and reply back onthe same port
self.addEventListener('message', function(event) {
Promise.resolve()
.then(function () {
switch(event.data.action) {
case 'doThing':
event.data.result = expensiveThing();
default:
throw Error('Invalid Action');
}
})
.then(function () {
event.data.success = true;
}, function (err) {
console.error(err);
event.data.success = false;
if (err) {
event.data.message = err.message ? err.message : err;
}
})
.then(function () {
event.ports[0].postMessage(event.data);
});
});
Interfacing with workers is usually quite a bellyache with the postMessage API.
The MessageChannel API which recently gained support across all platforms. Allows us to create a nice pattern for sending messages to workers.
This is the important bit, the interface once can make is very simple and promise based much like the fetch API. Greatly increasing the ease of working with workers.
But it can be:
http://www.stevesouders.com/blog/2014/02/11/measuring-localstorage-performance/
Of the older webapis which are synchronous LocalStorage is probably the most commonly still used.
There are several situations where it can be slow.
My recent tests in Cr45 for mobile and desktop show that it's is about 1/3 of writing to memory.
So really not too bad.
If speed is essential it still may be best to use an async storage method such as IndexedDB.
Unlike local storage IndexedDB can also be used in workers.
On the a page's first load it is pretty common practice to cache a whole bunch of assets for later use.
Front loading your content like this can be great for later loading performance.
Unfortunately parzing JS and decoding images can also quite expensive.
This added stress can make your site's first few moments be really janky or even unresponsive.
The user can neither click nor scroll. You might as well have just sent down a pretty picture of your above the fold content.
When the page looks like it is ready to be used it should also be readily responding to a users interactions.
Especially if a site has an interstitial.
The user's very first interaction with your site, they are presented with
A barrier requiring the user to interact in order to see sone content.
Since the browser is struggling with parzing and caching the new assets.
The user's attempts to close the interstial are futile or laggy,
they will probably bounce. rather than wait for the site to start working
The interstitial looked closable but wasn't responding straight away this is not a good experience
Load just the minimum amount of content for the first use. Load more later into the site's life time.
If there is need to query some large JSON to view your site but not all the data is needed right there and then.
Don't try to eat an elephant in a single mouthful.
If you are using a Service Worker you are probably already doing precaching off thread using the cache API so good on you.
On our Six Degrees of Angela Merkel Four oh Four page project I start with just 20s worth of precalculated data for our animation to get started fast.
The remaining 10MB of data was also broken into 20s chunks which could be parzed in a single frame.
Parzing a full 10MB JSON file would have caused our smooth animation to look Janky during the first few seconds when we have the most of the user's attention.
You can compromise on performance though,
The times when the site needs to be most responsive is when interaction and animation is happening e.g. the user is touching, dragging, swiping or scrolling or when the user is being prompted for input.
But there are a few situations where you can be slow without the user noticing for example just after a click or tap interaction it takes about 100ms unresponsiveness for a site to start to feel slow.
Which is oodles of time to perform a large calculation such as precaching data or performing calculations for an upcoming animation.
[If you are reading my notes check the code for the fastdom slide]
If you are careful and nothing is being animated for example the user is reading an article take this time to deserialise a json object or precache the user's potential actions.
In WebGL to produce special effects really performantly shaders can be used.
Shaders are scripts which get compiled when instantiated (this can be slow).
They can process huge amounts of data very quickly on your computers graphics card.
This has the advantage of being off the main thread and being run on your computers graphics hardware.
Graphics cards are very powerful and specialise in vector maths can process huge amounts of data very quickly.
They can process many thousands of points in a few milliseconds.
If working with WebGL and are animating a fixed topology e.g. 3D graphs and data visualisations, fixed geometry models. Tranform eacho point in the Shader rather than setting the geometry using the cpu, you will get very good performance.
In this demo I send sound data from the microphone to the Shader.
The shader then processes this data to work out how it should effect each point on the model.
There are over 4000 vertices and it does not struggle even on a lower end phone.
Change slide
In my early prototype I used the cpu for this. It got very hot and this is a google cardboard demo. I thought it was important to set fire to the cardboard headset with the mobile's cpu. (Joke)
That's not the kind of face meltingly awesome VR one should strive for.
If you would like to process HUGE datasets with 100s of thousands of data points.
Your complex operation one can code your complex operation into a shader.
Then the data can be entered in much like I did with the bunny demo.
But instead of rendering an awesome demo.
One can draw the values out as pixels in the frame buffer.
You can look at the colour of each pixel to extract your data from Red Blue Green information.
Allowing you to use the power of your graphics card for data processing.
Kind of like NVIDIA CUDA.
But seriously don't do this. If you do I want to see it.
On the more realistic side of things there is an upcoming API which will allow you to do parallel data processing on the cpu in the web.
It is called SIMD.
It will allow one to perform simultaeneous calculations by mapping four numbers as dimensions of a 4 vector and do 4 operations at once.
In a way very similar to the way shaders do on the graphics chip.
It is still on the cpu but should be able to process a large dataset much faster.
A factor of 4, 8 or 16 faster.
It can be used now in firefox nightly
and used in wrapped webapps by shipping with chrome which has SIMD compiled in, Intel XDK can do this for you.
Or you can just wait a bit.
Any qs?