FEATURE: Show a blurry preview when lazy loading images
This generates a 10x10 PNG thumbnail for each lightboxed image. If Image Lazy Loading is enabled (IntersectionObserver API) then we'll load the low res version when offscreen. As the image scrolls in we'll swap it for the high res version. We use a WeakMap to track the old image attributes. It's much less memory than storing them as `data-*` attributes and swapping them back and forth all the time.
This commit is contained in:
parent
e593d68beb
commit
662cfc416b
app
assets
models
lib
spec
|
@ -2,24 +2,39 @@ const OBSERVER_OPTIONS = {
|
||||||
rootMargin: "50%" // load images slightly before they're visible
|
rootMargin: "50%" // load images slightly before they're visible
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const imageSources = new WeakMap();
|
||||||
|
|
||||||
|
const LOADING_DATA =
|
||||||
|
"";
|
||||||
|
|
||||||
// We hide an image by replacing it with a transparent gif
|
// We hide an image by replacing it with a transparent gif
|
||||||
function hide(image) {
|
function hide(image) {
|
||||||
image.classList.add("d-lazyload");
|
image.classList.add("d-lazyload");
|
||||||
image.classList.add("d-lazyload-hidden");
|
image.classList.add("d-lazyload-hidden");
|
||||||
image.setAttribute("data-src", image.getAttribute("src"));
|
|
||||||
|
imageSources.set(image, {
|
||||||
|
src: image.getAttribute("src"),
|
||||||
|
srcSet: image.getAttribute("srcset")
|
||||||
|
});
|
||||||
|
image.removeAttribute("srcset");
|
||||||
|
|
||||||
image.setAttribute(
|
image.setAttribute(
|
||||||
"src",
|
"src",
|
||||||
""
|
image.getAttribute("data-small-upload") || LOADING_DATA
|
||||||
);
|
);
|
||||||
|
image.removeAttribute("data-small-upload");
|
||||||
}
|
}
|
||||||
|
|
||||||
// Restore an image from the `data-src` attribute
|
// Restore an image when onscreen
|
||||||
function show(image) {
|
function show(image) {
|
||||||
let dataSrc = image.getAttribute("data-src");
|
let sources = imageSources.get(image);
|
||||||
if (dataSrc) {
|
if (sources) {
|
||||||
image.setAttribute("src", dataSrc);
|
image.setAttribute("src", sources.src);
|
||||||
image.classList.remove("d-lazyload-hidden");
|
if (sources.srcSet) {
|
||||||
|
image.setAttribute("srcset", sources.srcSet);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
image.classList.remove("d-lazyload-hidden");
|
||||||
}
|
}
|
||||||
|
|
||||||
export function setupLazyLoading(api) {
|
export function setupLazyLoading(api) {
|
||||||
|
|
|
@ -1,6 +1,7 @@
|
||||||
.lightbox-wrapper .lightbox {
|
.lightbox-wrapper .lightbox {
|
||||||
position: relative;
|
position: relative;
|
||||||
display: inline-block;
|
display: inline-block;
|
||||||
|
overflow: hidden;
|
||||||
background: $primary-low;
|
background: $primary-low;
|
||||||
&:hover .meta {
|
&:hover .meta {
|
||||||
opacity: 0.9;
|
opacity: 0.9;
|
||||||
|
@ -9,7 +10,6 @@
|
||||||
}
|
}
|
||||||
|
|
||||||
.d-lazyload-hidden {
|
.d-lazyload-hidden {
|
||||||
opacity: 0;
|
|
||||||
box-sizing: border-box;
|
box-sizing: border-box;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -68,7 +68,7 @@ class OptimizedImage < ActiveRecord::Base
|
||||||
Rails.logger.error("Could not find file in the store located at url: #{upload.url}")
|
Rails.logger.error("Could not find file in the store located at url: #{upload.url}")
|
||||||
else
|
else
|
||||||
# create a temp file with the same extension as the original
|
# create a temp file with the same extension as the original
|
||||||
extension = ".#{upload.extension}"
|
extension = ".#{opts[:format] || upload.extension}"
|
||||||
|
|
||||||
if extension.length == 1
|
if extension.length == 1
|
||||||
return nil
|
return nil
|
||||||
|
@ -96,6 +96,7 @@ class OptimizedImage < ActiveRecord::Base
|
||||||
url: "",
|
url: "",
|
||||||
filesize: File.size(temp_path)
|
filesize: File.size(temp_path)
|
||||||
)
|
)
|
||||||
|
|
||||||
# store the optimized image and update its url
|
# store the optimized image and update its url
|
||||||
File.open(temp_path) do |file|
|
File.open(temp_path) do |file|
|
||||||
url = Discourse.store.store_optimized_image(file, thumbnail)
|
url = Discourse.store.store_optimized_image(file, thumbnail)
|
||||||
|
@ -173,7 +174,7 @@ class OptimizedImage < ActiveRecord::Base
|
||||||
IM_DECODERS ||= /\A(jpe?g|png|tiff?|bmp|ico|gif)\z/i
|
IM_DECODERS ||= /\A(jpe?g|png|tiff?|bmp|ico|gif)\z/i
|
||||||
|
|
||||||
def self.prepend_decoder!(path, ext_path = nil, opts = nil)
|
def self.prepend_decoder!(path, ext_path = nil, opts = nil)
|
||||||
extension = File.extname((opts && opts[:filename]) || ext_path || path)[1..-1]
|
extension = File.extname((opts && opts[:filename]) || path || ext_path)[1..-1]
|
||||||
raise Discourse::InvalidAccess if !extension || !extension.match?(IM_DECODERS)
|
raise Discourse::InvalidAccess if !extension || !extension.match?(IM_DECODERS)
|
||||||
"#{extension}:#{path}"
|
"#{extension}:#{path}"
|
||||||
end
|
end
|
||||||
|
@ -189,10 +190,14 @@ class OptimizedImage < ActiveRecord::Base
|
||||||
from = prepend_decoder!(from, to, opts)
|
from = prepend_decoder!(from, to, opts)
|
||||||
to = prepend_decoder!(to, to, opts)
|
to = prepend_decoder!(to, to, opts)
|
||||||
|
|
||||||
|
instructions = ['convert', "#{from}[0]"]
|
||||||
|
|
||||||
|
if opts[:colors]
|
||||||
|
instructions << "-colors" << opts[:colors].to_s
|
||||||
|
end
|
||||||
|
|
||||||
# NOTE: ORDER is important!
|
# NOTE: ORDER is important!
|
||||||
%W{
|
instructions.concat(%W{
|
||||||
convert
|
|
||||||
#{from}[0]
|
|
||||||
-auto-orient
|
-auto-orient
|
||||||
-gravity center
|
-gravity center
|
||||||
-background transparent
|
-background transparent
|
||||||
|
@ -204,7 +209,7 @@ class OptimizedImage < ActiveRecord::Base
|
||||||
-quality 98
|
-quality 98
|
||||||
-profile #{File.join(Rails.root, 'vendor', 'data', 'RT_sRGB.icm')}
|
-profile #{File.join(Rails.root, 'vendor', 'data', 'RT_sRGB.icm')}
|
||||||
#{to}
|
#{to}
|
||||||
}
|
})
|
||||||
end
|
end
|
||||||
|
|
||||||
def self.resize_instructions_animated(from, to, dimensions, opts = {})
|
def self.resize_instructions_animated(from, to, dimensions, opts = {})
|
||||||
|
@ -212,7 +217,7 @@ class OptimizedImage < ActiveRecord::Base
|
||||||
|
|
||||||
%W{
|
%W{
|
||||||
gifsicle
|
gifsicle
|
||||||
--colors=256
|
--colors=#{opts[:colors] || 256}
|
||||||
--resize-fit #{dimensions}
|
--resize-fit #{dimensions}
|
||||||
--optimize=3
|
--optimize=3
|
||||||
--output #{to}
|
--output #{to}
|
||||||
|
@ -248,7 +253,7 @@ class OptimizedImage < ActiveRecord::Base
|
||||||
%W{
|
%W{
|
||||||
gifsicle
|
gifsicle
|
||||||
--crop 0,0+#{dimensions}
|
--crop 0,0+#{dimensions}
|
||||||
--colors=256
|
--colors=#{opts[:colors] || 256}
|
||||||
--optimize=3
|
--optimize=3
|
||||||
--output #{to}
|
--output #{to}
|
||||||
#{from}
|
#{from}
|
||||||
|
|
|
@ -65,7 +65,8 @@ class Upload < ActiveRecord::Base
|
||||||
opts = opts.merge(raise_on_error: true)
|
opts = opts.merge(raise_on_error: true)
|
||||||
begin
|
begin
|
||||||
OptimizedImage.create_for(self, width, height, opts)
|
OptimizedImage.create_for(self, width, height, opts)
|
||||||
rescue
|
rescue => ex
|
||||||
|
Rails.logger.info ex if Rails.env.development?
|
||||||
opts = opts.merge(raise_on_error: false)
|
opts = opts.merge(raise_on_error: false)
|
||||||
if fix_image_extension
|
if fix_image_extension
|
||||||
OptimizedImage.create_for(self, width, height, opts)
|
OptimizedImage.create_for(self, width, height, opts)
|
||||||
|
|
|
@ -10,6 +10,8 @@ class CookedPostProcessor
|
||||||
|
|
||||||
INLINE_ONEBOX_LOADING_CSS_CLASS = "inline-onebox-loading"
|
INLINE_ONEBOX_LOADING_CSS_CLASS = "inline-onebox-loading"
|
||||||
INLINE_ONEBOX_CSS_CLASS = "inline-onebox"
|
INLINE_ONEBOX_CSS_CLASS = "inline-onebox"
|
||||||
|
LOADING_SIZE = 10
|
||||||
|
LOADING_COLORS = 32
|
||||||
|
|
||||||
attr_reader :cooking_options, :doc
|
attr_reader :cooking_options, :doc
|
||||||
|
|
||||||
|
@ -27,6 +29,8 @@ class CookedPostProcessor
|
||||||
@doc = Nokogiri::HTML::fragment(post.cook(post.raw, @cooking_options))
|
@doc = Nokogiri::HTML::fragment(post.cook(post.raw, @cooking_options))
|
||||||
@has_oneboxes = post.post_analyzer.found_oneboxes?
|
@has_oneboxes = post.post_analyzer.found_oneboxes?
|
||||||
@size_cache = {}
|
@size_cache = {}
|
||||||
|
|
||||||
|
@disable_loading_image = !!opts[:disable_loading_image]
|
||||||
end
|
end
|
||||||
|
|
||||||
def post_process(bypass_bump = false)
|
def post_process(bypass_bump = false)
|
||||||
|
@ -332,11 +336,19 @@ class CookedPostProcessor
|
||||||
upload.create_thumbnail!(resized_w, resized_h, crop: crop)
|
upload.create_thumbnail!(resized_w, resized_h, crop: crop)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
unless @disable_loading_image
|
||||||
|
upload.create_thumbnail!(LOADING_SIZE, LOADING_SIZE, format: 'png', colors: LOADING_COLORS)
|
||||||
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
add_lightbox!(img, original_width, original_height, upload, cropped: crop)
|
add_lightbox!(img, original_width, original_height, upload, cropped: crop)
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def loading_image(upload)
|
||||||
|
upload.thumbnail(LOADING_SIZE, LOADING_SIZE)
|
||||||
|
end
|
||||||
|
|
||||||
def is_a_hyperlink?(img)
|
def is_a_hyperlink?(img)
|
||||||
parent = img.parent
|
parent = img.parent
|
||||||
while parent
|
while parent
|
||||||
|
@ -398,6 +410,10 @@ class CookedPostProcessor
|
||||||
else
|
else
|
||||||
img["src"] = upload.url
|
img["src"] = upload.url
|
||||||
end
|
end
|
||||||
|
|
||||||
|
if small_upload = loading_image(upload)
|
||||||
|
img["data-small-upload"] = small_upload.url
|
||||||
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
# then, some overlay informations
|
# then, some overlay informations
|
||||||
|
|
|
@ -15,7 +15,7 @@ describe CookedPostProcessor do
|
||||||
RAW
|
RAW
|
||||||
end
|
end
|
||||||
|
|
||||||
let(:cpp) { CookedPostProcessor.new(post) }
|
let(:cpp) { CookedPostProcessor.new(post, disable_loading_image: true) }
|
||||||
let(:post_process) { sequence("post_process") }
|
let(:post_process) { sequence("post_process") }
|
||||||
|
|
||||||
it "post process in sequence" do
|
it "post process in sequence" do
|
||||||
|
@ -288,10 +288,22 @@ describe CookedPostProcessor do
|
||||||
filesize: 800
|
filesize: 800
|
||||||
)
|
)
|
||||||
|
|
||||||
|
# Fake a loading image
|
||||||
|
OptimizedImage.create!(
|
||||||
|
url: 'http://a.b.c/10x10.png',
|
||||||
|
width: CookedPostProcessor::LOADING_SIZE,
|
||||||
|
height: CookedPostProcessor::LOADING_SIZE,
|
||||||
|
upload_id: upload.id,
|
||||||
|
sha1: SecureRandom.hex,
|
||||||
|
extension: '.png',
|
||||||
|
filesize: 123
|
||||||
|
)
|
||||||
|
|
||||||
cpp = CookedPostProcessor.new(post)
|
cpp = CookedPostProcessor.new(post)
|
||||||
|
|
||||||
cpp.add_to_size_cache(upload.url, 2000, 1500)
|
cpp.add_to_size_cache(upload.url, 2000, 1500)
|
||||||
cpp.post_process_images
|
cpp.post_process_images
|
||||||
|
expect(cpp.loading_image(upload)).to be_present
|
||||||
|
|
||||||
# 1.5x is skipped cause we have a missing thumb
|
# 1.5x is skipped cause we have a missing thumb
|
||||||
expect(cpp.html).to include('srcset="http://a.b.c/666x500.jpg, http://a.b.c/1998x1500.jpg 3x"')
|
expect(cpp.html).to include('srcset="http://a.b.c/666x500.jpg, http://a.b.c/1998x1500.jpg 3x"')
|
||||||
|
@ -379,7 +391,7 @@ describe CookedPostProcessor do
|
||||||
|
|
||||||
let(:upload) { Fabricate(:upload) }
|
let(:upload) { Fabricate(:upload) }
|
||||||
let(:post) { Fabricate(:post_with_large_image) }
|
let(:post) { Fabricate(:post_with_large_image) }
|
||||||
let(:cpp) { CookedPostProcessor.new(post) }
|
let(:cpp) { CookedPostProcessor.new(post, disable_loading_image: true) }
|
||||||
|
|
||||||
before do
|
before do
|
||||||
SiteSetting.max_image_height = 2000
|
SiteSetting.max_image_height = 2000
|
||||||
|
@ -447,7 +459,7 @@ describe CookedPostProcessor do
|
||||||
|
|
||||||
let(:upload) { Fabricate(:upload) }
|
let(:upload) { Fabricate(:upload) }
|
||||||
let(:post) { Fabricate(:post_with_large_image) }
|
let(:post) { Fabricate(:post_with_large_image) }
|
||||||
let(:cpp) { CookedPostProcessor.new(post) }
|
let(:cpp) { CookedPostProcessor.new(post, disable_loading_image: true) }
|
||||||
|
|
||||||
before do
|
before do
|
||||||
SiteSetting.create_thumbnails = true
|
SiteSetting.create_thumbnails = true
|
||||||
|
@ -462,7 +474,7 @@ describe CookedPostProcessor do
|
||||||
|
|
||||||
it "crops the image" do
|
it "crops the image" do
|
||||||
cpp.post_process_images
|
cpp.post_process_images
|
||||||
expect(cpp.html).to match /width="690" height="500">/
|
expect(cpp.html).to match(/width="690" height="500">/)
|
||||||
expect(cpp).to be_dirty
|
expect(cpp).to be_dirty
|
||||||
end
|
end
|
||||||
|
|
||||||
|
@ -472,7 +484,7 @@ describe CookedPostProcessor do
|
||||||
|
|
||||||
let(:upload) { Fabricate(:upload) }
|
let(:upload) { Fabricate(:upload) }
|
||||||
let(:post) { Fabricate(:post_with_large_image) }
|
let(:post) { Fabricate(:post_with_large_image) }
|
||||||
let(:cpp) { CookedPostProcessor.new(post) }
|
let(:cpp) { CookedPostProcessor.new(post, disable_loading_image: true) }
|
||||||
|
|
||||||
before do
|
before do
|
||||||
SiteSetting.create_thumbnails = true
|
SiteSetting.create_thumbnails = true
|
||||||
|
@ -499,7 +511,7 @@ describe CookedPostProcessor do
|
||||||
|
|
||||||
let(:upload) { Fabricate(:upload) }
|
let(:upload) { Fabricate(:upload) }
|
||||||
let(:post) { Fabricate(:post_with_large_image_on_subfolder) }
|
let(:post) { Fabricate(:post_with_large_image_on_subfolder) }
|
||||||
let(:cpp) { CookedPostProcessor.new(post) }
|
let(:cpp) { CookedPostProcessor.new(post, disable_loading_image: true) }
|
||||||
let(:base_url) { "http://test.localhost/subfolder" }
|
let(:base_url) { "http://test.localhost/subfolder" }
|
||||||
let(:base_uri) { "/subfolder" }
|
let(:base_uri) { "/subfolder" }
|
||||||
|
|
||||||
|
@ -538,7 +550,7 @@ describe CookedPostProcessor do
|
||||||
|
|
||||||
let(:upload) { Fabricate(:upload) }
|
let(:upload) { Fabricate(:upload) }
|
||||||
let(:post) { Fabricate(:post_with_large_image_and_title) }
|
let(:post) { Fabricate(:post_with_large_image_and_title) }
|
||||||
let(:cpp) { CookedPostProcessor.new(post) }
|
let(:cpp) { CookedPostProcessor.new(post, disable_loading_image: true) }
|
||||||
|
|
||||||
before do
|
before do
|
||||||
SiteSetting.max_image_height = 2000
|
SiteSetting.max_image_height = 2000
|
||||||
|
|
|
@ -29,6 +29,21 @@ describe OptimizedImage do
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
describe ".resize_instructions" do
|
||||||
|
let(:image) { "#{Rails.root}/spec/fixtures/images/logo.png" }
|
||||||
|
|
||||||
|
it "doesn't return any color options by default" do
|
||||||
|
instructions = described_class.resize_instructions(image, image, "50x50")
|
||||||
|
expect(instructions).to_not include('-colors')
|
||||||
|
end
|
||||||
|
|
||||||
|
it "supports an optional color option" do
|
||||||
|
instructions = described_class.resize_instructions(image, image, "50x50", colors: 12)
|
||||||
|
expect(instructions).to include('-colors')
|
||||||
|
end
|
||||||
|
|
||||||
|
end
|
||||||
|
|
||||||
describe '.resize' do
|
describe '.resize' do
|
||||||
it 'should work correctly when extension is bad' do
|
it 'should work correctly when extension is bad' do
|
||||||
|
|
||||||
|
@ -186,7 +201,6 @@ describe OptimizedImage do
|
||||||
describe ".create_for" do
|
describe ".create_for" do
|
||||||
|
|
||||||
it "is able to 'optimize' an svg" do
|
it "is able to 'optimize' an svg" do
|
||||||
|
|
||||||
# we don't really optimize anything, we simply copy
|
# we don't really optimize anything, we simply copy
|
||||||
# but at least this confirms this actually works
|
# but at least this confirms this actually works
|
||||||
|
|
||||||
|
@ -239,6 +253,11 @@ describe OptimizedImage do
|
||||||
expect(oi.url).to eq("/internally/stored/optimized/image.png")
|
expect(oi.url).to eq("/internally/stored/optimized/image.png")
|
||||||
end
|
end
|
||||||
|
|
||||||
|
it "is able to change the format" do
|
||||||
|
oi = OptimizedImage.create_for(upload, 100, 200, format: 'gif')
|
||||||
|
expect(oi.url).to eq("/internally/stored/optimized/image.gif")
|
||||||
|
end
|
||||||
|
|
||||||
end
|
end
|
||||||
|
|
||||||
end
|
end
|
||||||
|
|
Loading…
Reference in New Issue