We’re going to create a background like this in pure CSS:
Of course, the simple version would be to simply create a CSS gradient (and there are tools that make creating them easy):
But it just doesn’t look that great, simply because we can only create linear or radial gradients. We want more randomness, something that looks less geometric and predictable.
A common method is to choose a real-life photo and blur it. TinEye offers a free online tool that lets us search for CC Flickr images based on the colours they contain. I added some colours to my search and ended up choosing this picture by Flickr user “whatapar” (under the CC 2.0 license):
We don’t need most of the information in this photo so let’s scale it down to a mere 8x6 pixels:
TinyPNG turns this image into 326 bytes and we can use a simple online tool to turn it into a data URI so we can include it directly in our CSS:
body {
background-image: url("");
background-repeat: no-repeat;
background-size: cover;
}
It would be great if that was all there’s to it. And indeed, in Chrome, it looks quite nice:
But Firefox does not blur the scaled up image, at least not much:
(Update 2021: This appears to have changed as Firefox now applies the same blurring algorithm to scaled-up background images.)
One could add a separate <div>
element for the background and apply a filter: blur(10px)
to it. But I’m a bit of a purist and adding extra div’s to achieve a purely visual effect is something I try to avoid.
Most modern browsers today support SVG files and the SVG specification includes a Gaussian blur filter (which, in fact, looks even better than the primitive interpolation by Chrome seen above). Let’s create an SVG file with our 8x6 image:
<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink= "http://www.w3.org/1999/xlink" viewBox="1 1 6 4">
<image filter="url(#b)" width="8" height="6" xlink:href=""/>
<filter id="b"><feGaussianBlur stdDeviation=".5" /></filter>
</svg>
This is what it looks like:
You may have noticed the viewBox="1 1 6 4"
which crops the SVG’s area by one pixel on each side. This is because the Gaussian filter assumes that everything outside the image is white and therefore introduces a white haze at the borders. Cropping the borders somewhat makes it disappear.
Now we turn this SVG file into a data URI once more and include it in our CSS:
body {
background-image: url("");
background-repeat: no-repeat;
background-size: cover;
}
Looks good, even in Firefox:
One thing I like to do is give the background a bit of a texture. Again, using SVG here (although a PNG may work well here, too):
<svg width="3" height="3" viewBox="0 0 3 3"
xmlns:svg="http://www.w3.org/2000/svg" xmlns="http://www.w3.org/2000/svg"
xmlns:xlink="http://www.w3.org/1999/xlink">
<rect x="0" y="0" width="1" height="1" style="fill:rgba(0,0,0,.2)"/>
</svg>
It’s a one black, slightly transparent bit pixel at (1,1)
. Once more, we data-URI it and repeat it over the gradient:
html {
background-image: url("");
background-repeat: no-repeat;
background-size: cover;
}
body {
background-image: url("");
margin: 0;
}
Here we have the final product, achieved with 1,440 bytes of CSS (uncompressed):
To see the texture better, here’s an original scale screenshot:
I noticed that some browsers seem to take quite some time rendering the gradient. I assume it’s because scaling the SVG up to page size also increases the size of the Gaussian filter. So at the moment, likely until we have more GPU-based rendering of websites, there is a tradeoff to make: This solution minimizes network traffic but page render times are somewhat slow. To speed up the page render time, it’s probably better to load a gradient image.