Lazy loading of images for website to load faster

时间:2018-02-03 09:34:47

标签: javascript image-loading

i want that before the actual image gets loaded a spinner should be shown so that my website can load faster

    window.addEventListener('load', function(){
        var allimages= document.getElementsByTagName('img');
        for (var i=0; i<allimages.length; i++) {
            if (allimages[i].getAttribute('data-src')) {
                allimages[i].setAttribute('src', allimages[i].getAttribute('data-src'));
            }
        }
    }, false)
        <img src="https://loremflickr.com/400/600" data-src="images/banguet-hall-location-icon.png" class="secEightImg" />
    <img src="http://www.jettools.com/images/animated_spinner.gif" data-src="images/banguet-hall-location-icon.png" class="secEightImg" />

    <script>

    </script>

the problem is the actual image is shown two times and instead of spinner a broken image (which is shown when no image is found) is shown first please help

2 个答案:

答案 0 :(得分:2)

If you want a spinner per image, just add the spinner URL as the initial src attribute for all of them:

<img src="spinner.gif" data-src="actual-image.png" />

Then, once the page loads, change all those src for the real URL (data-src) and listen for the load and error events on each image.

For each of them, add a .loaded class or .error class to the images and style them as you want. For example, you could hide the ones that could not be loaded, show a custom "error" image (using background-image) or style them as you wish, like in this example:

function imageLoaded(e) {
  updateImage(e.target, 'loaded');
}

function imageError(e) {
  updateImage(e.target, 'error');
}

function updateImage(img, classname) {
  // Add the right class:
  img.classList.add(classname);
  
  // Remove the data-src attribute:
  img.removeAttribute('data-src');
  
  // Remove both listeners:
  img.removeEventListener('load', imageLoaded);
  img.removeEventListener('error', imageError);
}

window.addEventListener('load', () => {
  Array.from(document.getElementsByTagName('img')).forEach(img => {
    const src = img.getAttribute('data-src');

    if (src) {
      // Listen for both events:
      img.addEventListener('load', imageLoaded);
      img.addEventListener('error', imageError);
      
      // Just to simulate a slow network:
      setTimeout(() => {
        img.setAttribute('src', src);
      }, 2000 + Math.random() * 2000);
    }
  });
})
html,
body {
  margin: 0;
  height: 100%;
}

.images {
  width: 100%;
  height: 100%;
  background: #000;
  display: flex;
  align-items: center;
  overflow-x: scroll;
}

.margin-fix {
    border-right: 1px solid transparent;
    height: 16px;
}

img {
  width: 16px;
  height: 16px;
  margin: 0 64px;
}

img.loaded {
  width: auto;
  height: 100%;
  margin: 0;
}

img.error {
  background: red;
  border-radius: 100%;
  
  /* You could add a custom "error" image here using background-image */
}
<div class="images">
  <img
    src="https://i.stack.imgur.com/RvfGz.gif"
    data-src="https://d39a3h63xew422.cloudfront.net/wp-content/uploads/2014/07/20145029/driven-by-design-the-incomparable-lancia-stratos-1476934711918-1000x573.jpg" />

  <img
    src="https://i.stack.imgur.com/RvfGz.gif"
    data-src="https://car-images.bauersecure.com/pagefiles/76591/1752x1168/ford_racing_puma_01.jpg?mode=max&quality=90&scale=down" />

  <img
    src="https://i.stack.imgur.com/RvfGz.gif"
    data-src="http://doesntexist.com/image.jpg" />
    
  <span class="margin-fix"></span>
</div>

The problem is that the spinner image you use initially might take some time to load.

One solution would be to use a data URI, so instead of:

<img src="https://i.stack.imgur.com/RvfGz.gif" data-src="actual-image.png" />

You would have:

<img src="" data-src="actual-image.png" />

As you can see, your HTML document will grow fast using this approach, which is a problem.

A better approach might be to use CSS to add the spinner so that you only include the data URI once. To do that, you need to add an empty src attribute to your images initially anyway:

<img src data-src="actual-image.png" />

Or:

<img src="" data-src="actual-image.png" />

If you don't put it, the image will have an annoying gray border you can't get rid of until you add a src attribute.

function imageLoaded(e) {
  updateImage(e.target, 'loaded');
}

function imageError(e) {
  updateImage(e.target, 'error');
}

function updateImage(img, classname) {
  // Add the right class:
  img.classList.add(classname);
  
  // Remove the data-src attribute:
  img.removeAttribute('data-src');
  
  // Remove both listeners:
  img.removeEventListener('load', imageLoaded);
  img.removeEventListener('error', imageError);
}

window.addEventListener('load', () => {
  Array.from(document.getElementsByTagName('img')).forEach(img => {
    const src = img.getAttribute('data-src');

    if (src) {
      // Listen for both events:
      img.addEventListener('load', imageLoaded);
      img.addEventListener('error', imageError);
      
      // Just to simulate a slow network:
      setTimeout(() => {
        img.setAttribute('src', src);
      }, 2000 + Math.random() * 2000);
    }
  });
})
html,
body {
  margin: 0;
  height: 100%;
}

.images {
  width: 100%;
  height: 100%;
  background: #000;
  display: flex;
  align-items: center;
  overflow-x: scroll;
}

.margin-fix {
    border-right: 1px solid transparent;
    height: 16px;
}

img {
  width: 16px;
  height: 16px;
  margin: 0 64px;
  background-image: url("");
}

img.loaded {
  width: auto;
  height: 100%;
  margin: 0;
}

img.error {
  background: red;
  border-radius: 100%;
  
  /* You could add a custom "error" image here using background-image */
}
<div class="images">
  <img
    src
    data-src="https://d39a3h63xew422.cloudfront.net/wp-content/uploads/2014/07/20145029/driven-by-design-the-incomparable-lancia-stratos-1476934711918-1000x573.jpg" />

  <img
    src
    data-src="https://car-images.bauersecure.com/pagefiles/76591/1752x1168/ford_racing_puma_01.jpg?mode=max&quality=90&scale=down" />

  <img
    src
    data-src="http://doesntexist.com/image.jpg" />
    
  <span class="margin-fix"></span>
</div>

Other approaches might be to just hide the images initially with display: none for example, in which case you might want a wrapper around them to show something like an empty box to indicate the users that something will be shown in there, or even a spinner made with other elements if you want/need to get fancy.

答案 1 :(得分:0)

What you can do is something like:

$(".sly-main-slider div img").attr('src', "#");
$(".sly-main-slider").addClass("loading");
$(".sly-main-slider div").hide();

$(".sly-main-slider div img").load(function() {
  $(".sly-main-slider div").show();
  $(".sly-main-slider").removeClass("loading");
}).attr('src', "https://loremflickr.com/400/600");
.loading {
  position: absolute;
  top: 10%;
  left: 10%;
  z-index: 2000;
  background: url(http://1.bp.blogspot.com/-nfXo9GWbDtM/VOn0vr4yLMI/AAAAAAAABCA/dDNgd7_QCFo/s1600/block-loader.gif) no-repeat center center;
  height: 32px;
  width: 32px;
}

.loading:after {
  position: absolute;
  top: 8px;
  left: 40px;
  content: 'Loading...';
}

body {
  position: relative;
  display: block;
}
<script src="https://ajax.googleapis.com/ajax/libs/jquery/2.1.1/jquery.min.js"></script>

<div class='sly-main-slider'>
  <div>
    <img src="https://loremflickr.com/400/600" alt="" />
  </div>
</div>