data:image/s3,"s3://crabby-images/7007a/7007a7c1b37b546f2e18e3e0ed2f31fd1424c19d" alt=""
Привет!
What will we see today?
PROGRESSIVE IMAGE RENDERING
@jmperezperez
Improve perceived performance of images on the web
PROGRESSIVE IMAGE RENDERING
@jmperezperez
by
- not making requests
- optimizing images
- rendering previews
PROGRESSIVE IMAGE RENDERING
@jmperezperez
PROGRESSIVE IMAGE RENDERING
@jmperezperez
data:image/s3,"s3://crabby-images/c1ddc/c1ddc53b94747d0c74618717f6158865daeb45f5" alt=""
About me
data:image/s3,"s3://crabby-images/ee39e/ee39e64db98fa2c0bc0999242250d98cb8601a5a" alt=""
data:image/s3,"s3://crabby-images/67a95/67a95fbf2da699e5e969ca8982dc65bd3128a601" alt=""
Page Load Time
PROGRESSIVE IMAGE RENDERING
@jmperezperez
Version A
FrontFest
2017
Speakers & Schedule
0.0s
0.3s
0.6s
0.9s
1.2s
Version B
FrontFest
2017
Speakers & Schedule
0.0s
0.3s
0.6s
0.9s
1.2s
FrontFest
2017
Speakers & Schedule
FrontFest
2017
FrontFest
2017
PROGRESSIVE IMAGE RENDERING
@jmperezperez
data:image/s3,"s3://crabby-images/1d8c4/1d8c44f5a4934b825d403a5c92152a6d298caefc" alt=""
Source https://www.smashingmagazine.com/2015/11/why-performance-matters-part-2-perception-management/
PROGRESSIVE IMAGE RENDERING
@jmperezperez
PROGRESSIVE IMAGE RENDERING
@jmperezperez
end = end time in milliseconds
VC = % visually complete
PROGRESSIVE IMAGE RENDERING
@jmperezperez
WebPageTest
data:image/s3,"s3://crabby-images/3aaa1/3aaa11eec61a9038b6033c49be4d1e5dced935ad" alt=""
PROGRESSIVE IMAGE RENDERING
@jmperezperez
data:image/s3,"s3://crabby-images/8bca4/8bca46e83b72e9c9cd7b6f02d59dff9c82244ef7" alt=""
Avoiding this:
PROGRESSIVE IMAGE RENDERING
@jmperezperez
data:image/s3,"s3://crabby-images/fcd84/fcd84323b5c497527f97ca9682b508c35b8dfb50" alt=""
Techniques to improve perceived performance
- Server-side rendering
- Critical CSS
- Async JS
- Async fonts
PROGRESSIVE IMAGE RENDERING
@jmperezperez
What about images?
PROGRESSIVE IMAGE RENDERING
@jmperezperez
PROGRESSIVE IMAGE RENDERING
@jmperezperez
images
1,815kB (54%)
scripts
457kB (14%)
other
1,106kB (33%)
source: http archive 1 nov 2017
data:image/s3,"s3://crabby-images/47a29/47a29cb7fdcdccc56b99dd2b74a1d46400bb09a7" alt=""
PROGRESSIVE IMAGE RENDERING
@jmperezperez
data:image/s3,"s3://crabby-images/e919d/e919d436408912cccf08e14a71596599222a52f9" alt=""
Minimalist | Flat Design
PROGRESSIVE IMAGE RENDERING
@jmperezperez
Not that long ago
data:image/s3,"s3://crabby-images/3f9f5/3f9f5cb59e119399afbefb02b679daabc4649407" alt=""
data:image/s3,"s3://crabby-images/8e993/8e993fc247c7038eb81c18426ca5049a47a85390" alt=""
Optimise your images
PROGRESSIVE IMAGE RENDERING
@jmperezperez
TIP 1
PROGRESSIVE IMAGE RENDERING
@jmperezperez
data:image/s3,"s3://crabby-images/69a5c/69a5c9f01abdce4bad51b3f18c7b83e0b0501923" alt=""
right format
gif | png | jpg | webp | <the_next_thing>
PROGRESSIVE IMAGE RENDERING
@jmperezperez
<picture>
<source type="image/webp" srcset="2700x1209/my-image.webp 2700w,
1024x1024/my-image.webp 1024w,
600x600/my-image.webp 600w"
sizes="100vw" />
<source srcset="2700x1209/my-image.jpg 2700w,
1024x1024/my-image.jpg 1024w,
600x600/my-image.jpg 600w"
sizes="100vw" />
<img class="rsImg" src="600x600/my-image.jpg" alt="My beautiful image" />
</picture>
compression
PROGRESSIVE IMAGE RENDERING
@jmperezperez
· lossless
· perceptual
PROGRESSIVE IMAGE RENDERING
@jmperezperez
Responsive
TIP 2
data:image/s3,"s3://crabby-images/77625/77625219b433138d7b5bf1731f230210c1493550" alt=""
data:image/s3,"s3://crabby-images/77625/77625219b433138d7b5bf1731f230210c1493550" alt=""
data:image/s3,"s3://crabby-images/77625/77625219b433138d7b5bf1731f230210c1493550" alt=""
1 column
2 columns
3 columns
<img sizes="(max-width: 30em) 100vw,
(max-width: 50em) 50vw,
calc(33vw - 100px)"
srcset="red-square-200.jpg 200w,
red-square-400.jpg 400w,
red-square-800.jpg 800w,
red-square-1600.jpg 1600w"
src="red-square-400.jpg"
alt="Red Square">
PROGRESSIVE IMAGE RENDERING
@jmperezperez
Challenge: Keeping in sync markup and CSS
Example
RESPONSIVE IMAGES
Lazy-load images
PROGRESSIVE IMAGE RENDERING
@jmperezperez
TIP 3
data:image/s3,"s3://crabby-images/c3d7a/c3d7ab1d481c9b4c522af22ba2bea37be5d6ee8a" alt=""
<img> above the fold
&
Lazy below the fold
PROGRESSIVE IMAGE RENDERING
@jmperezperez
Intersection
Observer
data:image/s3,"s3://crabby-images/df33d/df33d50e462d4ac99c8e3106145243540376aaea" alt=""
PROGRESSIVE IMAGE RENDERING
@jmperezperez
// load image when it's within 100px of the viewport
const options = {
rootMargin: '100px'
}
const callback = entries => {
entries.forEach(entry => {
if (entry.intersectionRatio > 0) {
// load image
}
});
};
const observer = new IntersectionObserver(callback, options);
observer.observe(document.querySelector('.lazy-img'));
PROGRESSIVE IMAGE RENDERING
@jmperezperez
Example
INTERSECTION OBSERVER
class LazyImage extends React.Component {
constructor() {
this.observer = new IntersectionObserver(entries => {
if (entries[0].intersectionRatio > 0) {
// load!
}
});
this.element = null; /* render() will set it through a ref */
}
componentDidMount() {
this.observer.observe(this.element);
}
componentWillUnmount() {
this.observer.unobserve(this.element);
}
...
}
PROGRESSIVE IMAGE RENDERING
@jmperezperez
Encapsulating in React
INTERSECTION OBSERVER
PROGRESSIVE IMAGE RENDERING
@jmperezperez
Using image.decode()
IMAGE DECODING
// Image loading with predecoding
const img = new Image();
img.src = "image.jpg";
img.decode().then(() => {
document.body.appendChild(img);
}).catch(() => {
throw new Error('Could not load/decode big image.');
});
PROGRESSIVE IMAGE RENDERING
@jmperezperez
Using image.decode()
IMAGE DECODING
Add an image to the DOM without causing a decoding delay.
Track the proposal on https://github.com/whatwg/html/issues/1920
TIP 4
What to show while the image is loading
PROGRESSIVE IMAGE RENDERING
@jmperezperez
Options
PROGRESSIVE IMAGE RENDERING
PLACEHOLDERS
Nothing
Placeholder
Solid colour or gradient
Progressive image loading or "Blur-up"
@jmperezperez
data:image/s3,"s3://crabby-images/fa823/fa823a8c50de00139d4a4f288fd51ef3d5c88e16" alt=""
data:image/s3,"s3://crabby-images/1d65e/1d65ef6c5ea5c8e1e6dfb62534200f9bfd12849f" alt=""
data:image/s3,"s3://crabby-images/2d15f/2d15fbd82bc33258d3a43d660fe7a577c365a965" alt=""
Examples of solid color
PROGRESSIVE IMAGE RENDERING
@jmperezperez
PROGRESSIVE IMAGE RENDERING
@jmperezperez
data:image/s3,"s3://crabby-images/b72d4/b72d4b7721984e83bde68c1e8ab5b1455b503bb8" alt=""
PROGRESSIVE IMAGE RENDERING
@jmperezperez
PROGRESSIVE IMAGE RENDERING
@jmperezperez
PROGRESSIVE IMAGE RENDERING
@jmperezperez
PROGRESSIVE IMAGE RENDERING
@jmperezperez
Examples of
Progressive Image Loading
PROGRESSIVE IMAGE RENDERING
@jmperezperez
PROGRESSIVE IMAGE RENDERING
@jmperezperez
Medium
data:image/s3,"s3://crabby-images/57304/57304ce149f46d3cc90f99b5feb65991dee406ea" alt=""
PROGRESSIVE IMAGE RENDERING
@jmperezperez
Medium
PROGRESSIVE IMAGE RENDERING
@jmperezperez
Quartz (qz.com)
data:image/s3,"s3://crabby-images/c708c/c708c0408ef8e006a7398c948e87f906cf506382" alt=""
PROGRESSIVE IMAGE RENDERING
@jmperezperez
Quartz (qz.com)
PROGRESSIVE IMAGE RENDERING
@jmperezperez
Quora
data:image/s3,"s3://crabby-images/f1864/f1864cee870a282d2b91e0f5a9d2789864a2b311" alt=""
PROGRESSIVE IMAGE RENDERING
@jmperezperez
How it is done
data:image/s3,"s3://crabby-images/e1d55/e1d55389ac3a876a70629a9fb86722b67beb462a" alt=""
data:image/s3,"s3://crabby-images/7121f/7121f1b721c7e948091180bfe1ec84e77a10fb4b" alt=""
data:image/s3,"s3://crabby-images/3fbbb/3fbbb0907e11cd5aeba29ded432eb79c46cbb6da" alt=""
PROGRESSIVE IMAGE RENDERING
@jmperezperez
Markup - Eg Medium
<figure>
<div>
<div/> <!-- this div keeps the aspect ratio so
the placeholder doesn't collapse -->
<img/> <!-- this is a tiny image with a small
resolution (e.g. ~27x17) and low quality -->
<canvas/> <!-- takes the above image and applies a blur filter -->
<img/> <!-- the large image to be displayed -->
<noscript/> <!-- fallback for no JS -->
</div>
</figure>
PROGRESSIVE IMAGE RENDERING
@jmperezperez
Thumbnails
data:image/s3,"s3://crabby-images/5f2b2/5f2b2432ea3ecb863c45d2464208de1e182343e9" alt=""
data:image/s3,"s3://crabby-images/446be/446bee2b685f7c09912dbabab408bb0e4b5ae84a" alt=""
PROGRESSIVE IMAGE RENDERING
@jmperezperez
Thumbnails
data:image/s3,"s3://crabby-images/5f2b2/5f2b2432ea3ecb863c45d2464208de1e182343e9" alt=""
data:image/s3,"s3://crabby-images/446be/446bee2b685f7c09912dbabab408bb0e4b5ae84a" alt=""
data:image/s3,"s3://crabby-images/3f4e8/3f4e8b2a91fa3159b18cecc5d38451ad64d53fd2" alt=""
Getting creative with SVGs
PROGRESSIVE IMAGE RENDERING
@jmperezperez
PROGRESSIVE IMAGE RENDERING
@jmperezperez
SVG - Edges
PROGRESSIVE IMAGE RENDERING
@jmperezperez
Drawing edges with SVG
PROGRESSIVE IMAGE RENDERING
@jmperezperez
Drawing bitmap images
data:image/s3,"s3://crabby-images/9bcca/9bcca7209a1623cd7afa7de323b507e8a90d1a65" alt=""
Canny Edge Detector
PROGRESSIVE IMAGE RENDERING
@jmperezperez
<svg>
<polyline points="51,1 61,1 61,2 56,4 56,3"/>
<polyline points="52,1 50,2 51,3 50,4 50,9 46,10 46,8 48,8 48,9"/>
<polyline points="61,4 61,5 58,6"/>
...
<polyline points="62,58 61,59 61,60 50,62 50,61 51,61"/>
</svg>
- Find edges with canny edge detector
- Create lines
- Use JS and SVG to animate
How to draw bitmaps
PROGRESSIVE IMAGE RENDERING
@jmperezperez
SVG - Shapes
PROGRESSIVE IMAGE RENDERING
@jmperezperez
PROGRESSIVE IMAGE RENDERING
@jmperezperez
SVG - Shapes
data:image/s3,"s3://crabby-images/c90a3/c90a3ba481bee12b576b6658f101fa7f2c7f990b" alt=""
SVG - 10 shapes
SVG - 100 shapes
Original
0.74kB
3.5kB
101kB
PROGRESSIVE IMAGE RENDERING
@jmperezperez
SVG - Shapes + Blur
SVG - 10 shapes + Blur
SVG 100 shapes + Blur
(JPEG or WebP) + Blur
0.8kB
3.6kB
0.5kB (JPEG)
0.09 kB (WebP)
data:image/s3,"s3://crabby-images/843f0/843f01c001785673c478d105072f5e1150e4ff22" alt=""
PROGRESSIVE IMAGE RENDERING
@jmperezperez
SVG - Silhouettes / Traces
data:image/s3,"s3://crabby-images/39da5/39da59baa8bd9ac50c946ccc731c2ecede025f2a" alt=""
PROGRESSIVE IMAGE RENDERING
@jmperezperez
SVG - Silhouettes / Traces
data:image/s3,"s3://crabby-images/5109b/5109b479608cfdc3146fd286ddcb4000d3b369df" alt=""
PROGRESSIVE IMAGE RENDERING
@jmperezperez
SVG - Silhouettes / Traces
PROGRESSIVE IMAGE RENDERING
@jmperezperez
Should we do this?
Just because you can it doesn't mean you should
PROGRESSIVE IMAGE RENDERING
@jmperezperez
- Reduce requests
- Choose the right format and optimise
- Embrace responsive images
- Try to lazy load
- Think about placeholders
- Innovate!
Summary
The Web is fun.
PROGRESSIVE IMAGE RENDERING
@jmperezperez
talk with me for more info
Thanks!
@jmperezperez
Slides on https://slides.com/jmperez