DEV Community

Dane Stevens
Dane Stevens

Posted on • Edited on • Originally published at tueri.io

Building the React Image Optimization Component for Tueri.io

Low-quality image placeholders

This article originally appeared on Medium.com

Building the React Image Optimization Component for Tueri.io

Let’s face it, image optimization is hard. We want to make it effortless.

Full source code can be found on GitHub

When we set out to build our React Component there were a few problems we wanted to solve:

  • Automatically decide image width for any device based on the parent container.

  • Use the best possible image format the user’s browser supports.

  • Automatic image lazy loading.

  • Automatic low-quality image placeholders (LQIP).

Oh, and it had to be effortless for React Developers to use.

This is what we came up with:

<Img src={ tueriImageId } alt='Alt Text' /> 
Enter fullscreen mode Exit fullscreen mode

Easy right? Let’s dive in.

Calculating the image size:

Create a <figure /> element, detect the width and build an image URL:

class Img extends React.Component { constructor(props) { super(props) this.state = { width: 0 } this.imgRef = React.createRef() } componentDidMount() { const width = this.imgRef.current.clientWidth this.setState({ width }) } render() { // Destructure props and state const { src, alt, options = {}, ext = 'jpg' } = this.props const { width } = this.state // Create an empty query string let queryString = '' // If width is specified, otherwise use auto-detected width options['w'] = options['w'] || width // Loop through option object and build queryString Object.keys(options).map((option, i) => { return queryString += `${i < 1 ? '?' : '&'}${option}=${options[option]}` }) return( <figure ref={this.imgRef}> { // If the container width has been set, display the image else null width > 0 ? ( <img src={`https://cdn.tueri.io/${ src }/${ kebabCase(alt) }.${ ext }${ queryString }`} alt={ alt } /> ) : null } </figure> ) } } export default Img 
Enter fullscreen mode Exit fullscreen mode

This returns the following HTML:

<figure> <img src="https://cdn.tueri.io/tueriImageId/alt-text.jpg?w=autoCalculatedWidth" alt="Alt Text" /> </figure> 
Enter fullscreen mode Exit fullscreen mode

Use the best possible image format:

Next, we needed to add support for detecting WebP images and having the Tueri service return the image in the WebP format:

class Img extends React.Component { constructor(props) { // ... this.window = typeof window !== 'undefined' && window this.isWebpSupported = this.isWebpSupported.bind(this) } // ... isWebpSupported() { if (!this.window.createImageBitmap) { return false; } return true; } render() { // ... // If a format has not been specified, detect webp support // Set the fm (format) option in the image URL if (!options['fm'] && this.isWebpSupported) { options['fm'] = 'webp' } // ... return ( // ... ) } } // ... 
Enter fullscreen mode Exit fullscreen mode

This returns the following HTML:

<figure> <img src="https://cdn.tueri.io/tueriImageId/alt-text.jpg?w=autoCalculatedWidth&fm=webp" alt="Alt Text" /> </figure> 
Enter fullscreen mode Exit fullscreen mode

Automatic image lazy loading:

Now, we need to find out if the <figure /> element is in the viewport, plus we add a little buffer area so the images load just before being scrolled into view.

 class Img extends React.Component { constructor(props) { // ... this.state = { // ... isInViewport: false lqipLoaded: false } // ... this.handleViewport = this.handleViewport.bind(this) } componentDidMount() { // ... this.handleViewport() this.window.addEventListener('scroll', this.handleViewport) } handleViewport() { // Only run if the image has not already been loaded if (this.imgRef.current && !this.state.lqipLoaded) { // Get the viewport height const windowHeight = this.window.innerHeight // Get the top position of the <figure /> element const imageTopPosition = this.imgRef.current.getBoundingClientRect().top // Multiply the viewport * buffer (default buffer: 1.5) const buffer = typeof this.props.buffer === 'number' && this.props.buffer > 1 && this.props.buffer < 10 ? this.props.buffer : 1.5 // If <figure /> is in viewport if (windowHeight * buffer > imageTopPosition) { this.setState({ isInViewport: true }) } } } // ... componentWillUnmount() { this.window.removeEventListener('scroll', this.handleViewport) } render() { // Destructure props and state // ... const { isInViewport, width } = this.state // ... return ( <figure ref={this.imgRef}> { // If the container width has been set, display the image else null isInViewport && width > 0 ? ( <img onLoad={ () => { this.setState({ lqipLoaded: true }) } } // ... /> ) : null } </figure> ) } } export default Img 
Enter fullscreen mode Exit fullscreen mode

Automatic low-quality image placeholders (LQIP):

Finally, when an image is in the viewport, we want to load a 1/10 size blurred image, then fade out the placeholder image when the full-size image is loaded:

class Img extends React.Component { constructor(props) { // ... this.state = { // ... fullsizeLoaded: false } // ... } // ... render() { // Destructure props and state // ... const { isInViewport, width, fullsizeLoaded } = this.state // ... // Modify the queryString for the LQIP image: replace the width param with a value 1/10 the fullsize const lqipQueryString = queryString.replace(`w=${ width }`, `w=${ Math.round(width * 0.1) }`) // Set the default styles. The full size image should be absolutely positioned within the <figure /> element const styles = { figure: { position: 'relative', margin: 0 }, lqip: { width: '100%', filter: 'blur(5px)', opacity: 1, transition: 'all 0.5s ease-in' }, fullsize: { position: 'absolute', top: '0px', left: '0px', transition: 'all 0.5s ease-in' } } // When the fullsize image is loaded, fade out the LQIP if (fullsizeLoaded) { styles.lqip.opacity = 0 } return( <figure style={ styles.figure } // ... > { isInViewport && width > 0 ? ( <React.Fragment> {/* Load fullsize image in background */} <img onLoad={ () => { this.setState({ fullsizeLoaded: true }) } } style={ styles.fullsize } src={`https://cdn.tueri.io/${ src }/${ kebabCase(alt) }.${ ext }${ queryString }`} alt={ alt } /> {/* Load LQIP in foreground */} <img onLoad={ () => { this.setState({ lqipLoaded: true }) } } style={ styles.lqip } src={`https://cdn.tueri.io/${ src }/${ kebabCase(alt) }.${ ext }${ lqipQueryString }`} alt={ alt } /> </React.Fragment> ) : null } </figure> ) } } // ... 
Enter fullscreen mode Exit fullscreen mode

Putting it all together:

Image optimization made effortless. Just swap out your regular <img /> elements for the Tueri <Img /> and never worry about optimization again.

import React from 'react' import PropTypes from 'prop-types' import { TueriContext } from './Provider' import kebabCase from 'lodash.kebabcase' class Img extends React.Component { constructor(props) { super(props) this.state = { isInViewport: false, width: 0, height: 0, lqipLoaded: false, fullsizeLoaded: false } this.imgRef = React.createRef() this.window = typeof window !== 'undefined' && window this.handleViewport = this.handleViewport.bind(this) this.isWebpSupported = this.isWebpSupported.bind(this) } componentDidMount() { const width = this.imgRef.current.clientWidth this.setState({ width }) this.handleViewport() this.window.addEventListener('scroll', this.handleViewport) } handleViewport() { if (this.imgRef.current && !this.state.lqipLoaded) { const windowHeight = this.window.innerHeight const imageTopPosition = this.imgRef.current.getBoundingClientRect().top const buffer = typeof this.props.buffer === 'number' && this.props.buffer > 1 && this.props.buffer < 10 ? this.props.buffer : 1.5 if (windowHeight * buffer > imageTopPosition) { this.setState({ isInViewport: true }) } } } isWebpSupported() { if (!this.window.createImageBitmap) { return false; } return true; } componentWillUnmount() { this.window.removeEventListener('scroll', this.handleViewport) } render() { // Destructure props and state const { src, alt, options = {}, ext = 'jpg' } = this.props const { isInViewport, width, fullsizeLoaded } = this.state // Create an empty query string let queryString = '' // If width is specified, otherwise use auto-detected width options['w'] = options['w'] || width // If a format has not been specified, detect webp support if (!options['fm'] && this.isWebpSupported) { options['fm'] = 'webp' } // Loop through option prop and build queryString Object.keys(options).map((option, i) => { return queryString += `${i < 1 ? '?' : '&'}${option}=${options[option]}` }) // Modify the queryString for the LQIP image: replace the width param with a value 1/10 the fullsize const lqipQueryString = queryString.replace(`w=${ width }`, `w=${ Math.round(width * 0.1) }`) const styles = { figure: { position: 'relative', margin: 0 }, lqip: { width: '100%', filter: 'blur(5px)', opacity: 1, transition: 'all 0.5s ease-in' }, fullsize: { position: 'absolute', top: '0px', left: '0px', transition: 'all 0.5s ease-in' } } // When the fullsize image is loaded, fade out the LQIP if (fullsizeLoaded) { styles.lqip.opacity = 0 } const missingALt = 'ALT TEXT IS REQUIRED' return( // Return the CDN domain from the TueriProvider <TueriContext.Consumer> {({ domain }) => ( <figure style={ styles.figure } ref={this.imgRef} > { //  isInViewport && width > 0 ? ( <React.Fragment> {/* Load fullsize image in background */} <img onLoad={ () => { this.setState({ fullsizeLoaded: true }) } } style={ styles.fullsize } src={`${ domain }/${ src }/${ kebabCase(alt || missingALt) }.${ ext }${ queryString }`} alt={ alt || missingALt } /> {/* Load LQIP in foreground */} <img onLoad={ () => { this.setState({ lqipLoaded: true }) } } style={ styles.lqip } src={`${ domain }/${ src }/${ kebabCase(alt || missingALt) }.${ ext }${ lqipQueryString }`} alt={ alt || missingALt } /> </React.Fragment> ) : null } </figure> )} </TueriContext.Consumer> ) } } Img.propTypes = { src: PropTypes.string.isRequired, alt: PropTypes.string.isRequired, options: PropTypes.object, ext: PropTypes.string, buffer: PropTypes.number } export default Img 
Enter fullscreen mode Exit fullscreen mode

See it in action:

Try out a live demo on CodeSandbox:

Top comments (0)