I help companies find and fix site-speed issues. Performance audits, training, consultancy, and more.
Written by Harry Roberts on CSS Wizardry.
N.B. All code can now be licensed under the permissive MIT license. Read more about licensing CSS Wizardry code samples…
Largest Contentful Paint (LCP) is my favourite Core Web Vital. It’s the easiest to optimise, and it’s the only one of the three that works the exact same in the lab as it does in the field (don’t even get me started on this…). Yet, surprisingly, it’s the least optimised CWV in CrUX—at the time of writing, only half of origins in the dataset had a Good LCP!
I help companies find and fix site-speed issues. Performance audits, training, consultancy, and more.
This genuinely surprises me, because LCP is the simplest metric to improve. So, in this post, I want to go deep and show you some interesting tricks and optimisations, as well as some pitfalls and bugs, starting with some very simple tips.
Let’s go.
Let’s start with the easy stuff. LCP is a milestone timing—it measures…
…the render time of the largest image or text block visible within the viewport, relative to when the page first started loading.
The important thing to note here is that Google doesn’t care how you get to LCP, as long as you get there fast. There are a lot of other things that could happen between the start of the page load lifecycle and its LCP. These include (but are not limited to):
If any of these are slow, you’re already on the back foot, and they’re going to have a knock-on effect on your LCP. The metrics above don’t matter in and of themselves, but it’s going to help your LCP if you can get them as low as possible.
An analogy I use with non-technical stakeholders goes a little like this:
You need to get the kids to school for 08:30. That’s all the school cares about—that the kids are there on time. You can do plenty to help make this happen: prepare their clothes the night before; prepare their lunches the night before (do the same for yourself). Set appropriate alarms. Have a morning routine that everyone follows. Leave the house with plenty of time to spare. Plan in suitable buffer time for traffic issues, etc.
The school doesn’t care if you laid out uniforms the night before. You are being judged on your ability to get the kids to school on time; it’s just common sense to do as much as you can to make that happen.
Same with your LCP. Google doesn’t (currently) care about your TTFB, but a good TTFB is going to help get closer to a good LCP.
Optimise the entire chain. Make sure you get everything beforehand as fast as possible so that you’re set up for success.
A tip that hopefully doesn’t need me to go into any real detail: if you have an image-based LCP, make sure it is well optimised—suitable format, appropriately sized, sensibly compressed, etc. Don’t have a 3MB TIFF as your LCP candidate.
This isn’t going to work for a lot, if not most, sites. But the best way to get a fast LCP is to ensure that your LCP is text-based. This, in effect, makes your FCP and LCP synonymous12. That’s it. As simple as that. If possible, avoid image-based LCP candidates and opt instead for textual LCPs.
The chances are, however, that won’t work for you. Let’s look at our other options.
Okay. Now we’re getting into the fun stuff. Let’s look at which LCP candidates we have, and whether there are any relative merits to each.
There are several potential candidates for your LCP. Taken straight from web.dev’s Largest Contentful Paint (LCP) page, these are:
<img>
elements<image>
elements inside an <svg>
element<video>
elements (the poster image is used)url()
function (as
opposed to a CSS
gradient)For the purposes of this article, I built a series of reduced demos showing how
each of the LCP types behave. Each of the demos also contains a reference to
a blocking in-<head>
JavaScript file in order to:
It’s also worth noting that each demo is very stripped back, and doesn’t necessarily represent realistic conditions in which many responses would be in-flight at the same time. Once we run into resource contention, LCP candidates’ discovery may work differently to what is exhibited in these reduced test cases. In cases like these, we might look to Priority Hints or Preload to lend a hand. All I’m interested in right now is inherent differences in how browsers treat certain resources.
The initial demos can be found at:
<img>
<img src="lcp.jpg" ... />
<image>
in <svg>
<svg xmlns="http://www.w3.org/1000/svg">
<image href="lcp.jpg" ... />
</svg>
poster
<video poster="lcp.jpg" ...></video>
background-image: url();
<div style="background-image: url(lcp.jpg)">...</div>
The WebPageTest comparison is available for you to look through, though we’ll pick apart individual waterfalls later in the article. That all comes out looking like this:
<img>
and poster
are identical in LCP; <image>
in <svg>
is the
next fastest, although there is a bug in the LCP time that Chrome reports;
background-image
-based LCPs are notably the slowest.
As we can see, not all candidates are born equal. Let’s look at each in more detail.
<img>
ElementsOf the image-based LCPs, this is probably our favourite. <img>
elements, as
long as we don’t mess things up, are quick to be discovered by the preload
scanner,
and as such, can be requested in parallel to preceding—even blocking—resources.
<picture>
and <source />
It’s worth noting that the <picture>
element behaves the same way as the <img
/>
element. This is why you need to write so much verbose syntax for your
srcset
and sizes
attributes: the idea is that you give the browser enough
information about the image that it can request the relevant file via the
preload scanner
and not have to wait until layout. (Although, I guess—technically—there must be
like a few milliseconds compute overhead working out which combination of
<source />
, srcset
, sizes
to use, but that will be mooted pretty quickly
by virtually any other moving part along the way.)
<image>
in <svg>
<image>
elements defined in <svg>
s display two very interesting behaviours.
The first of which is a simple bug in which Chrome misreports the LCP candidate,
seemingly overlooking the <image>
entirely. Depending on your context, this
could mean much more favourable and optimistic LCP scores.
Once the fix rolls out in M102 (which is Canary at the time of writing, and will reach Stable on 24 May, 2022), we can expect accurate measurements. This does mean that you may experience degraded LCP scores for your site.
Because of the current reporting bug, <image>
in <svg>
is likely to go from
being (inadvertently) one of the fastest LCP types, to one of the slowest. In
the unlikely event that you are using <image>
in <svg>
, it’s probably
something that you want to check on sooner rather than later—your scores are
likely to change.
The bug pertains only to reported LCP candidate, and does not impact how the
browser actually deals with the resources. To that end, waterfalls in all Chrome
versions look identical, and networking/scheduling behaviour remains unchanged.
Which brings me onto the second interesting thing I spotted with <image>
in
<svg>
:
<image>
elements defined in <svg>
s appear to be hidden from the preload
scanner: that is to say, the href
attribute is not parsed until the browser’s
primary parser encounters it. I can only guess that this is simply because the
preload scanner is built to scan HTML and not SVG, and that this is by design
rather than an oversight. Perhaps an optimisation that Chrome could make is to
preload scan embedded SVG in HTML…? But I’m sure that’s much more easily said
than done…
<video>
Elements’ poster
AttributeI’m pleasantly surprised by the behaviour exhibited by the <video>
’s poster
attribute. It seems to behave identically to the <img />
element, and is
discovered early by the preload scanner.
This means that poster
LCPs are inherently pretty fast, so that’s nice
news.
The other news is that it looks like there’s intent to take the first frame of
a video as the
LCP candidate if no poster
is present. That’s going to be a difficult LCP to
get under 2.5s, so either don’t have a <video>
LCP at all, or make sure you
start using a poster
image with it.
background-image: url();
Resources defined in CSS (chiefly anything requested via the url()
function) are slow by
default. The most common candidates here are background images and web fonts.
The reason these resources (in this specific case, background images) are slow is because they aren’t requested until the browser is ready to paint the DOM node that needs them. You can read more about that in this Twitter thread:
Simple yet significant thing all developers should keep in mind: CSS resources (fonts, background images) are not requested by your CSS, but by the DOM node that needs them [Note: slight oversimplification, but the correct way to think about it.]
— Harry Roberts (@csswizardry) 10 September 2021
This means that background-image
LCPs are requested at the very last moment,
which is far too late. We don’t like background-image
LCPs.
background-image
IssuesIf you currently have a site whose LCP is a background-image
, you might be
thinking of refactoring or rebuilding that component right now. But, happily,
there’s a very quick workaround that requires almost zero effort: let’s
complement the background with a hidden <img />
that the browser can discover
much earlier.
<div style="background-image: url(lcp.jpg)">
<img src="lcp.jpg" alt="" width="0" height="0" style="display: none !important;" />
</div>
This little hack allows the preload scanner to pick up the image, rather than
waiting until the browser is about to render the <div>
. This came in 1.058s
faster than the naive background-image
implementation. You’ll notice that this
waterfall almost exactly mimics the fastest <img />
option:
We could also preload
this image, rather than using an <img />
element, but
I generally feel that preload
is a bit of a code smell and should be avoided
if possible.
In summary:
<img />
and poster
LCPs are nice and fast, discoverable by the preload
scanner;<video>
without a poster
might have its first frame considered as an LCP
candidate in future versions of Chrome;<image>
in <svg>
is currently misreported but is slow because the href
is hidden from the preload scanner;background-image
s are slow by default, because of how CSS works;
<img />
.Alright! Now we know which are the best candidates, is there anything else can do (or avoid doing) to make sure we aren’t running slowly? It turns out there are plenty of things that folks do which inadvertently hold back LCP scores.
Every time I see this, my heart sinks a little. Lazy-loading your LCP is completely counter-intuitive. Please don’t do it!
Interestingly, one of the features of loading="lazy"
is that it hides the
image in question from the preload
scanner.
This means that, even if the image is in the viewport, the browser will still
late-request it. This is why you can’t safely add loading="lazy"
to all of
your images and simply hope the browser does (what you think is) the right
thing.
In my tests, lazily loading our image pushed LCP back to 4.418s: 1.274s slower
than the <img />
variant, and almost identical to the background-image
test.
Predictably, fading in our image over 500ms pushes our LCP event back by 500ms. Chrome takes the end of the animation period as the LCP measurement, moving us to a 3.767s LCP event rather than 3.144s.
Avoid fading in your LCP candidate, whether it’s image- or text-based.
Where possible, we should always self-host our static assets. This includes our LCP candidate.
It’s not uncommon for site owners to use third-party image optimisation services such as Cloudinary to serve both automated and dynamically optimised images: on the fly resizing, format switching, compression, etc. However, even when taking into account the performance improvements of of these services, the cost of heading to a different origin almost always outweighs the benefits. In testing, the time spent resolving a new origin added 509ms to overall time spend downloading our LCP image.
By all means, use third party services for non-critical, non-LCP images, but if you can, bring your LCP candidate onto the same origin as the host page. That’s exactly what I do for this site.
N.B. While preconnect
may help a little, it’s still highly unlikely
to be faster than not opening a new connection at all.
I see this all too often, and it’s part of the continued obsession with
JavaScript. Ideally, a browser will receive your HTML response, and the
reference to the LCP candidate (ideally an <img />
element) will be right
there immediately. However, if you build your LCP candidate with JS, the process
is much, much more drawn out.
Building your LCP candidate in JS
could range from a simple JS-based
image gallery, right the way through to a fully client-rendered page. The below
waterfall shows the latter:
The first response is the HTML. What we’d like to have is an <img />
right
there in the markup, waiting to be discovered almost immediately. Instead, the
HTML requests a defer
ed framework.js
at entry 12. This, in turn, eventually
requests API data about the current product, at entry 50. This response contains
information about related product imagery, which is eventually put into the
virtual DOM as an <img />
, finally initiating a request for the LCP candidate
at entry 53, well over 7s into the page load lifecycle.
This one breaks my heart every time I see it… Don’t late-load any content that accidentally becomes your LCP candidate. Usually, these are things like cookie banners or newsletter modals that cover content and get flagged as a very late LCP. I mocked up a late-loading modal for our tests, and what is important to remember is that the score is accurate, just not what we are hoping for:
Make sure your LCP candidate is what you expect it to be. Design modals and cookie banners etc. to:
Alright. We covered quite a lot there, but the takeaway is pretty simple:
text-based LCPs are the fastest, but unlikely to be possible for most. Of
the image based LCP types, <img />
and poster
are the fastest.
<image>
s defined in <svg>
s are slow because they’re hidden from the
preload scanner. Beyond that, there are several things that we need to avoid:
don’t lazy load your LCP candidate, and don’t build your LCP in JS.
You’ll need to make sure you’re using font-display: [swap|optional];
so as to avoid an initial, invisible text paint. ↩
I did discover another bug while investigating this, though. ↩
N.B. All code can now be licensed under the permissive MIT license. Read more about licensing CSS Wizardry code samples…
Harry Roberts is an independent consultant web performance engineer. He helps companies of all shapes and sizes find and fix site speed issues.
Hi there, I’m Harry Roberts. I am an award-winning Consultant Web Performance Engineer, designer, developer, writer, and speaker from the UK. I write, Tweet, speak, and share code about measuring and improving site-speed. You should hire me.
You can now find me on Mastodon.
I help teams achieve class-leading web performance, providing consultancy, guidance, and hands-on expertise.
I specialise in tackling complex, large-scale projects where speed, scalability, and reliability are critical to success.