Skip to content

Instantly share code, notes, and snippets.

@Slashgear
Last active March 27, 2019 17:58
Show Gist options
  • Select an option

  • Save Slashgear/3f7d79480967f9a97efc79a1d0bde026 to your computer and use it in GitHub Desktop.

Select an option

Save Slashgear/3f7d79480967f9a97efc79a1d0bde026 to your computer and use it in GitHub Desktop.
LastStep
import React, { useState, useEffect } from "react";
import styled from "styled-components";
const placeHolder =
"";
const Image = styled.img`
display: block;
height: 100px;
width: 100px;
// Add a smooth animation on loading
@keyframes loaded {
0% {
opacity: 0.1;
}
100% {
opacity: 1;
}
}
// I use utilitary classes instead of props to avoid style regenerating
&.loaded:not(.has-error) {
animation: loaded 300ms ease-in-out;
}
&.has-error {
// fallback to placeholder image on error
content: url(${placeHolder});
}
`;
export const LazyImage = ({ src, alt }) => {
const [imageSrc, setImageSrc] = useState(placeHolder);
const [imageRef, setImageRef] = useState();
const onLoad = event => {
event.target.classList.add("loaded");
};
const onError = event => {
event.target.classList.add("has-error");
};
useEffect(
() => {
let observer;
let didCancel = false;
if (imageRef && imageSrc !== src) {
if (IntersectionObserver) {
observer = new IntersectionObserver(
entries => {
entries.forEach(entry => {
if (!didCancel && (entry.intersectionRatio > 0 || entry.isIntersecting)) {
setImageSrc(src);
observer.unobserve(imageRef);
}
});
},
{
threshold: 0.01,
rootMargin: "75%"
}
);
observer.observe(imageRef);
} else {
// Old browsers fallback
setImageSrc(src);
}
}
return () => {
didCancel = true;
// on component cleanup, we remove the listner
if (observer && observer.unobserve) {
observer.unobserve(imageRef);
}
};
},
[src, imageSrc, imageRef]
);
return (
<Image
ref={setImageRef}
src={imageSrc}
alt={alt}
onLoad={onLoad}
onError={onError}
/>
);
};
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment