Title: A bit of website optimization every day: lazy loading of images
-closure categories: JavaScript
Student C is a front-end programmer. This day, the big boss gave him an instruction: our website pictures too much too big, too waste of network resources, I want to save traffic and do not reduce the quality of the picture, you do it. Therefore, student C came up with the idea of lazy image loading: when the image does not appear on the user screen, it is replaced by a lightweight placeholder image, and after the image appears, it is dynamically replaced by a high-quality image with JS. Next, student C picked up the keyboard and began to knock on the code……
To achieve lazy loading of images, the basic idea is to listen to the browser scroll event, constantly compare the distance between the target element and the bottom of the screen, when the target element scroll to the specified position, execute js code.
ScrollTop + offsetTop method
- The height of the page currently scrolling up scrollTop
- Distance of the target element from the top of the web page body
- Window. InnerHeight Indicates that the element is in the visible area when offsetTop -scrollTop <= innerHeight is specified
Gets the scrollTop height
- PageYOffset belongs to the Window object. Internet Explorer 9+, Firefox, Chrome, and Opera all support this method to obtain the scrolling height of the page and ignore the Doctype definition rule.
- Body is the body child node in the DOM object, the body tag
- A documentElement is the trailing node of the entire node tree, or HTML tag
However, the compatibility of the three methods varies between different browsers. The compatibility table is as follows:
Browser \ method of getting scroll height | window.pageYOffset | document.documentElement.scrollTop | document.body.scrollTop |
---|---|---|---|
chrome | Y | Y | N |
firefox | Y | Y | N |
IE11 and above | Y | Y | N |
ios | Y | N | Y |
Android (Wchat + native browser) | Y | N | Y |
Complete compatibility code
var supportPageOffset = window.pageXOffset ! = =undefined;
var isCSS1Compat = ((document.compatMode || "") = = ="CSS1Compat");
var x = supportPageOffset ? window.pageXOffset : isCSS1Compat ? document.documentElement.scrollLeft : document.body.scrollLeft;
var y = supportPageOffset ? window.pageYOffset : isCSS1Compat ? document.documentElement.scrollTop : document.body.scrollTop;
Copy the code
Page height knows how much
Image lazy loading checks if the image is visible to the user, so select window.innerHeight as the browser height.
Method implementation
let offsetTop = element.offsetTop; // The height of the element relative to the upper left corner of the body
let scrollTop = getScrollTop().top; // Page scroll height
let innerHeight = window.innerHeight; // Available window height
if(offsetTop - scrollTop <= innerHeight){
inViewFun.call(this);
}
Copy the code
The disadvantage of this approach is that offsetTop is relative to the parent element, which can be problematic if the parent element of the target image element is not body.
Use the getBoundingClientRect method
- Element. GetBoundingClientRect () method returns the element size and its position relative to the viewport
Method implementation
var rect = element.getBoundingClientRect();
if((rect && rect.top) <= window.innerHeight ){// come into view
inViewFun.call(this);
}
Copy the code
To optimize the
Add anti-shake to scroll monitoring event
Add a fixed height to preload the image
The complete code
function lazyLoad(element, inViewFun, fixedTop, a){
scrollHandler();
if(window.addEventListener){
window.addEventListener('scroll',debounce(scrollHandler));
}else{ //IE
window.attachEvent('scroll',debounce(scrollHandler));
}
// Get the scrolling height of the page
function getScrollTop(){
var supportPageOffset = window.pageXOffset ! = =undefined;
var isCSS1Compat = ((document.compatMode || "") = = ="CSS1Compat");
var x = supportPageOffset ? window.pageXOffset : isCSS1Compat ? document.documentElement.scrollLeft : document.body.scrollLeft;
var y = supportPageOffset ? window.pageYOffset : isCSS1Compat ? document.documentElement.scrollTop : document.body.scrollTop;
return {left:x, top:y};
}
// function scrollHandler(){
// let offsetTop = element.offsetTop; // The height of the element relative to the upper left corner of the body
// let scrollTop = getScrollTop().top; // Page scroll height
// let innerHeight = window.innerHeight; // Available window height
// if(offsetTop - scrollTop <= innerHeight){
// inViewFun.call(this);
/ /}
// }
function scrollHandler(){
var rect = element.getBoundingClientRect();
if((rect && rect.top) <= window.innerHeight + fixedTop ){// come into view
inViewFun.call(this); }}}Copy the code
intersection observer
The intersection observer is easier to use and much more readable than event listeners. This intersection observer is currently only available on Chrome63+ and firefox58+. Developers just need to register an Observer to monitor elements rather than writing a bunch of messy window-checking code. After registering an Observer, all we need to do is change the behavior of the element when it becomes visible.
<img class="lazy" src="placeholder-image.jpg" data-src="image-to-lazy-load-1x.jpg" data-srcset="image-to-lazy-load-2x.jpg 2x, image-to-lazy-load-1x.jpg 1x" alt="I'm an image!">
Copy the code
- Class: Used to associate elements in JavaScript
- SRC property: points to a placeholder image that is displayed on the first load of the page
- Data-src and data-srcset properties: These are placeholder properties that hold the URL of the target image
document.addEventListener("DOMContentLoaded".function() {
var lazyImages = [].slice.call(document.querySelectorAll("img.lazy"));
if ("IntersectionObserver" in window) {
let lazyImageObserver = new IntersectionObserver(function(entries, observer) {
entries.forEach(function(entry) {
if (entry.isIntersecting) {
let lazyImage = entry.target;
lazyImage.src = lazyImage.dataset.src;
lazyImage.srcset = lazyImage.dataset.srcset;
lazyImage.classList.remove("lazy"); lazyImageObserver.unobserve(lazyImage); }}); }); lazyImages.forEach(function(lazyImage) {
lazyImageObserver.observe(lazyImage);
});
} else {
// Possibly fall back to a more compatible method here}});Copy the code