-
-
Save npofopr/e98abc6900dde2759f1bbee5b618c197 to your computer and use it in GitHub Desktop.
Intersection Observer + custom lazy loading demo
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
<html> | |
<head> | |
<title>Lazy loading by #Blondiecode</title> | |
<meta charset="UTF-8"> | |
<meta name="viewport" content="width=device-width, initial-scale=1"> | |
<style> | |
body { | |
font-family: Arial, "Helvetica Neue", Helvetica, serif; | |
} | |
footer { | |
padding: 20px 0; | |
text-align: center; | |
} | |
h1, h2 { | |
width: 100%; | |
text-align: center; | |
} | |
a { | |
color: #777777; | |
} | |
a:hover { | |
color: #999999; | |
} | |
.wrapper { | |
width: 100%; | |
margin: 0 auto; | |
max-width: 1200px; | |
display: flex; | |
flex-wrap: wrap; | |
} | |
.item { | |
width: 25%; | |
box-sizing: border-box; | |
min-height: 400px; | |
min-width: 300px; | |
text-align: center; | |
} | |
.item.youtube { | |
width: 50%; | |
min-height: 315px; | |
overflow: visible; | |
} | |
.item.loading { | |
border: 1px dotted #a7a7a7; | |
position: relative; | |
} | |
.item.loading:after { | |
content: ''; | |
position: absolute; | |
width: 50px; | |
left: 50%; | |
margin-left: -25px; | |
height: 50px; | |
top: 50%; | |
margin-top: -25px; | |
background: url(""); | |
} | |
.item img { | |
width: 100%; | |
height: 400px; | |
} | |
.item.youtube img { | |
height: 315px; | |
} | |
</style> | |
</head> | |
<body> | |
<main class="wrapper js"> | |
<h1>Some cool cats for you ❤</h1> | |
<div class="item loading"> | |
<img src="" data-src="http://placekitten.com/600/800" alt="meow" width="300" height="400"> | |
</div> | |
<div class="item loading"> | |
<img data-src="https://dummyimage.com/300x400/82A3A1/fff.jpg&text=cat"> | |
</div> | |
<div class="item loading"> | |
<img data-src="http://placekitten.com/g/600/800"> | |
</div> | |
<div class="item loading"> | |
<img data-src="http://placekitten.com/600/799"> | |
</div> | |
<div class="item loading"> | |
<img data-src="https://dummyimage.com/300x400/C0DFA1/fff.jpg&text=kitty"> | |
</div> | |
<div class="item loading"> | |
<img data-src="http://placekitten.com/g/300/400"> | |
</div> | |
<div class="item loading"> | |
<img data-src="http://placekitten.com/g/300/399"> | |
</div> | |
<div class="item loading"> | |
<img data-src="https://dummyimage.com/300x400/b38cc9/fff.jpg&text=purrrr"> | |
</div> | |
<div class="item loading"> | |
<img data-src="http://placekitten.com/300/399"> | |
</div> | |
<div class="item loading"> | |
<img data-src="https://dummyimage.com/300x400/9FC490/fff.jpg&text=meow"> | |
</div> | |
<div class="item loading"> | |
<img data-src="http://placekitten.com/300/400"> | |
</div> | |
<div class="item loading"> | |
<img data-src="http://placekitten.com/g/900/1200"> | |
</div> | |
<h2>Some cool videos from me with love ❤</h2> | |
<div class="item youtube loading"> | |
<iframe width="560" height="315" data-src="https://www.youtube.com/embed/pAsOU6RV0Ys" frameborder="0" allow="accelerometer; autoplay; encrypted-media; gyroscope; picture-in-picture" allowfullscreen></iframe> | |
</div> | |
<div class="item youtube loading"> | |
<iframe width="560" height="315" data-src="https://www.youtube.com/embed/fQQDeNtQW-4" frameborder="0" allow="accelerometer; autoplay; encrypted-media; gyroscope; picture-in-picture" allowfullscreen></iframe> | |
</div> | |
</main> | |
<footer> | |
<small><a href="https://www.youtube.com/c/AidaDrogan" target="_blank">#BlondieCode © 2020</a></small> | |
</footer> | |
<script> | |
//============= dynamic placeholder for images | |
const placeholder = ""; | |
const targets = document.querySelectorAll('[data-src]'); | |
targets.forEach(target => { | |
target.src = placeholder | |
}); | |
//============= IntersectionObserver | |
const options = { | |
root: null, | |
rootMargin: '0px', | |
threshold: 0.05 | |
}; | |
const loadImage = function (entries, observer) { | |
entries.forEach(entry => { | |
if (entry.isIntersecting && entry.target.parentNode.classList.contains('loading')){ | |
entry.target.src = entry.target.getAttribute('data-src'); | |
entry.target.onload = () => { | |
entry.target.parentNode.classList.remove('loading'); | |
entry.target.removeAttribute('data-src'); | |
}; | |
} | |
}); | |
}; | |
const observer = new IntersectionObserver(loadImage, options); | |
targets.forEach(target => { | |
observer.observe(target); | |
}); | |
//============= lazy loading by Aida Drogan | |
// function isIntersecting(target) { | |
// const docViewTop = window.pageYOffset; | |
// const docViewBottom = docViewTop + window.innerHeight; | |
// const elemTop = docViewTop + target.getBoundingClientRect().top; | |
// const elemBottom = elemTop + target.height; | |
// return (((elemTop <= docViewBottom) && (elemTop >= docViewTop)) | |
// || ((elemBottom <= docViewBottom) && (elemBottom >= docViewTop))); | |
// } | |
// | |
// const checkImages = function() { | |
// targets.forEach(target => { | |
// if (isIntersecting(target) && target.parentNode.classList.contains('loading')){ | |
// target.src = target.getAttribute('data-src'); | |
// target.onload = () => { | |
// target.parentNode.classList.remove('loading'); | |
// target.removeAttribute('data-src'); | |
// }; | |
// } | |
// }); | |
// }; | |
// | |
// window.onload = checkImages; | |
// window.onscroll = checkImages; | |
</script> | |
</body> | |
</html> |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment