Page used to jitter when oneboxes loaded images lazily. Previously we inserted the the "shadow" loading image before the "real" image. This meant that certain styling with `firstChild` CSS selectors would apply incorrectly to the shadow image. Additionally we had special case code for onebox and quoted images that was not really needed due to this fix. We had an old fix that used computed style for image height and width in specific scenarios, we now run it all the time. On slow devices there was a possibility that the cache fetch after amending src at the end of the process would cause a flash, this is avoided using a new onload handler.
110 lines
2.9 KiB
JavaScript
110 lines
2.9 KiB
JavaScript
const OBSERVER_OPTIONS = {
|
|
rootMargin: "66%" // load images slightly before they're visible
|
|
};
|
|
|
|
// Min size in pixels for consideration for lazy loading
|
|
const MINIMUM_SIZE = 150;
|
|
|
|
const hiddenData = new WeakMap();
|
|
|
|
const LOADING_DATA =
|
|
"";
|
|
|
|
// We hide an image by replacing it with a transparent gif
|
|
function hide(image) {
|
|
image.classList.add("d-lazyload");
|
|
image.classList.add("d-lazyload-hidden");
|
|
|
|
hiddenData.set(image, {
|
|
src: image.src,
|
|
srcset: image.srcset,
|
|
width: image.width,
|
|
height: image.height,
|
|
className: image.className
|
|
});
|
|
image.removeAttribute("srcset");
|
|
|
|
image.src = image.dataset.smallUpload || LOADING_DATA;
|
|
image.removeAttribute("data-small-upload");
|
|
}
|
|
|
|
// Restore an image when onscreen
|
|
function show(image) {
|
|
let imageData = hiddenData.get(image);
|
|
|
|
if (imageData) {
|
|
const copyImg = new Image();
|
|
copyImg.onload = () => {
|
|
image.src = copyImg.src;
|
|
if (copyImg.srcset) {
|
|
image.srcset = copyImg.srcset;
|
|
}
|
|
image.classList.remove("d-lazyload-hidden");
|
|
|
|
if (image.onload) {
|
|
// don't bother fighting with existing handler
|
|
// this can mean a slight flash on mobile
|
|
image.parentNode.removeChild(copyImg);
|
|
} else {
|
|
image.onload = () => {
|
|
image.parentNode.removeChild(copyImg);
|
|
image.onload = null;
|
|
};
|
|
}
|
|
|
|
copyImg.onload = null;
|
|
};
|
|
|
|
copyImg.src = imageData.src;
|
|
|
|
if (imageData.srcset) {
|
|
copyImg.srcset = imageData.srcset;
|
|
}
|
|
|
|
// width of image may not match, use computed style which
|
|
// is the actual size of the image
|
|
const computedStyle = window.getComputedStyle(image);
|
|
const actualWidth = parseInt(computedStyle.width, 10);
|
|
const actualHeight = parseInt(computedStyle.height, 10);
|
|
|
|
copyImg.style.position = "absolute";
|
|
copyImg.style.top = `${image.offsetTop}px`;
|
|
copyImg.style.left = `${image.offsetLeft}px`;
|
|
copyImg.style.width = `${actualWidth}px`;
|
|
copyImg.style.height = `${actualHeight}px`;
|
|
|
|
copyImg.className = imageData.className;
|
|
|
|
// insert after the current element so styling still will
|
|
// apply to original image firstChild selectors
|
|
image.parentNode.insertBefore(copyImg, image.nextSibling);
|
|
} else {
|
|
image.classList.remove("d-lazyload-hidden");
|
|
}
|
|
}
|
|
|
|
export function setupLazyLoading(api) {
|
|
const observer = new IntersectionObserver(entries => {
|
|
entries.forEach(entry => {
|
|
const { target } = entry;
|
|
|
|
if (entry.isIntersecting) {
|
|
show(target);
|
|
observer.unobserve(target);
|
|
}
|
|
});
|
|
}, OBSERVER_OPTIONS);
|
|
|
|
api.decorateCooked(
|
|
$post => {
|
|
$("img", $post).each((_, img) => {
|
|
if (img.width >= MINIMUM_SIZE && img.height >= MINIMUM_SIZE) {
|
|
hide(img);
|
|
observer.observe(img);
|
|
}
|
|
});
|
|
},
|
|
{ onlyStream: true, id: "discourse-lazy-load" }
|
|
);
|
|
}
|