Let’s say you have a photographic image that really should be a JPG or WebP, for the best file size and quality. But what if I need transparency too? Don’t I need PNG for that? Won’t that make for either huge file sizes (PNG-24) or weird quality (PNG-8)? Let’s look at another way that ends up best-of-both-worlds.
data:image/s3,"s3://crabby-images/542ba/542ba30bb611d2fe10b2c414da63b48db74394d2" alt=""
The goal is to clip myself out of the image, removing the background. My technique for that is usually to use Photoshop and cut a clipping path manually with the Pen tool.
data:image/s3,"s3://crabby-images/a71c4/a71c46d017a5b1c493e5fc7f0ee5512c6de39258" alt=""
Now I can select the inverse of that clipping path to easily remove the background.
Attempting to save this as a 1200px wide image as PNG-24 out of Photoshop ends up as about a 1MB image!
data:image/s3,"s3://crabby-images/cdf0f/cdf0f3a13a0cfbb6375f008e76693fc2b6e3ee4f" alt=""
We could cut that by 75% using PNG-8, but then we 1) get that weird Giffy look (less photographic) and 2) have to pick a matte color for the edges because we aren’t getting nice alpha transparency here, just binary transparency.
data:image/s3,"s3://crabby-images/565b7/565b783b0a2918557915fbc6a8037c2071221b10" alt=""
Gosh what if we could just use JPG? The quality and file size is way better.
data:image/s3,"s3://crabby-images/b3068/b306891baa5d9922ed380cd350c280afaf5cccc3" alt=""
No transparency though.
But wait! Can’t we just clip this thing out? We have clip-path now. Well… yeah. We do have clip-path. It can’t take a path()
, though, and what we’ve created for vector points in Photoshop is path data. It could take a polygon()
though, if we made all the lines straight. That’s probably not ideal (I’m curvy!). Or we could make a <clipPath>
element in some inline SVG and use clip-path: url(#id_of_clipPath);
, which does support a <path>
inside.
There is masking as well, which is another possibility.
Let’s look at a third possibility though: put everything into <svg>
. That made some logical sense to me, so all this stays together and scales together.
The trick is to make two things:
- The JPG
- The clipping
<path>
The JPG is easy enough. Output that right from Photoshop. Optimize.
Now we can set up the SVG. SVG is happy to take a raster graphic. SVG is known for vector graphics, but it’s a very flexible image format.
<svg>
<image xlink:href="/images/chris.jpg" x="0" y="0">
<svg>
To get the path, we export the path we created with the Pen tool over to Illustrator.
data:image/s3,"s3://crabby-images/df6be/df6be0d8f452e33d33e50ef3425363e5ed56c50d" alt=""
Now we have the path over there, and it’s easy to export as SVG:
data:image/s3,"s3://crabby-images/faff4/faff402b9da87c4dbcc3c59fbad16faaa8150d04" alt=""
Now we have the path data we need:
data:image/s3,"s3://crabby-images/9d3f7/9d3f7de28495b308c871ad9d47561cfa9e9c965b" alt=""
Let’s use that <path>
within a <clipPath>
in the SVG we’ve started. Then also apply that clip path to the <image>
:
<svg viewBox="0 0 921.17 1409.71">
<defs>
<clipPath id="chris-clip">
<path d=" ... " />
</clipPath>
</defs>
<image xlink:href="/images/chris.jpg" clip-path="url(#chris-clip)" x="0" y="0">
<svg>
Tada!
data:image/s3,"s3://crabby-images/5521f/5521f66b1dd859b96818520d2768ac14042f915a" alt=""
A transparent JPG, essentially.
source: css-tricks