New Feature: Lazy Loading Images


(Robin Ward) #1

A few months ago, @Johani released an excellent theme component to lazy load images. This means images are only loaded as they scroll into view, rather than being loaded as soon as they appear anywhere on the page, even if they aren’t visible to the user yet.

This is a great thing to have on forums where topics have many images in them, especially for users on connections with limited bandwidth.

If you know some of the more technical details of Discourse, you might ask, hey @eviltrout doesn’t Discourse already kind of do this? You do infinite scrolling and posts are streamed in and out of memory as the user scrolls. Yes, that’s right! But we generally do so in “chunks”, and on some image boards a single post could have dozens of large images. Why load them all at once when they aren’t visible yet?

In the latest builds of Discourse, we now lazy load large images, no extra components or plugins are necessary. If you scroll really fast you’ll see a blurry low-res version of a large image as a placeholder. It’ll look something like this:

image

technical details: we generate a very, very small 10x10 32-color image and store it as a data URI in the post, which the browser scales to the correct size – this ends up being around ~300 bytes

It’ll soon be replaced by the correct image:

image

We will continue to tweak the feature over the next little while, so please provide feedback and notes if you have them!


(Chris Beach) #2

Should improve page loading performance nicely, especially on slow connections. Thanks for this improvement!


(AstonJ) #3

Awesome!

This is why I love Discourse - you keep adding stuff I didn’t think we needed! :+1:t3:


(Robin Ward) #4

One thing I’m looking into is applying this to non-lightboxed images, such as those included via oneboxing links.

One issue is it’s a little tricky to know when to apply the logic, since we don’t know the filesize of the destination image. Originally my plan was to append the filesize of images, but it turns out that is quite tricky since we don’t always have a downloaded copy of the image available.

I think I’ll have to come up with a magic width x height, maybe 300x300px or so, and if the image has dimensions larger than that it’ll be lazy loaded. We do have those dimensions available regardless of onebox or upload.


(Azareal) #5

Web bloat I find this article to be a good one on the subject of slow connections, do bear in mind that it has a very minimalistic theme, so don’t judge it by it’s cover so to speak. It might not quite be applicable here, but it’s always good to keep in mind.

I sometimes wish there were ways with Web APIs to detect if someone is on a connection where they pay a lot for bandwidth or one that’s slow.


(Jeff Atwood) #6

Probably fine, but to cover animated GIFs which are still quite common I would relax that to something like 200×200.


(Sam Saffron) #7

Yeah I am not sure what the downside will be of going all the way down to 50x50, as long as we do not hit emojis it should be good. Even at 50x50 we may still save 25x if you don’t end up looking at the image


(Jeff Atwood) #8

I vote for 150×150 then, as we definitely want to steer clear of anything remotely emoji-like in size…


(Orca) #11

I mean, previously, image greater than 4MB isn’t loaded properly (using mobile to access)

-edit-


(Jeff Atwood) #12

That is completely unrelated. Max allowed image size is controlled in site settings. We already lightbox and thumbnail all images that are posted above a certain size. This is about topics with dozens or hundreds of images in them, not the size of any single image.


#13

I just want to say thank you for including this feature.


(Robin Ward) #14

Here’s some improvements:


(Shawn Kirsch) #15

Man, I thought I was going crazy. This feature is great for low bandwidth, but now i’m seeing this lazy loading all over the place when I am scrolling our forum. I have so much bandwidth and I’m on a computer. Maybe there should be some sort of buffer around areas you can’t see that load right before you scroll to them? Maybe only for mobile?