Skip to content

Instantly share code, notes, and snippets.

@gmaclennan
Last active February 27, 2023 01:03
Show Gist options
  • Star 2 You must be signed in to star a gist
  • Fork 1 You must be signed in to fork a gist
  • Save gmaclennan/11130600 to your computer and use it in GitHub Desktop.
Save gmaclennan/11130600 to your computer and use it in GitHub Desktop.
Fast long scrolling image grid

This is an implementation of a long-scrolling image grid. You should be able to smoothly scroll through 500,000 images from Flickr (you will see repeats because flickr only returns 4,000 images from a search). The images will delay when first loading, but once your browser has cached the images scrolling should be pretty smooth.

The trick to keeping it smooth is by only modifying CSS properties that are cheap to animate and by minimizing modifications to the DOM by reusing our exit nodes as enter nodes.

A full page of images (the same height as window height) is rendered above and below the viewable area. In addition, empty rows with a placeholder background are padded around for an additional 2 x window height. This is useful for mobile, which will not fire a scroll event until you stop scrolling.

.grid-3 {
background-image: url();
}
.grid-4 {
background-image: url();
}
.grid-5 {
background-image: url();
}
.grid-6 {
background-image: url();
}
.grid-7 {
background-image: url();
}
.grid-8 {
background-image: url();
}
.grid-9 {
background-image: url();
}
.grid-10 {
background-image: url();
}
<!DOCTYPE html>
<meta charset="utf-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<style>
@import url("image-grid.css");
body, html {
padding: 0;
margin: 0;
}
.row {
background-size: 100% auto;
position: absolute;
width: 100%;
}
.hidden {
opacity: 0;
}
img {
margin: 0;
border: 2px solid white;
-webkit-transition: opacity 500ms;
transition: opacity 500ms;
}
</style>
<body>
<div class="grid"></div>
<script src="http://d3js.org/d3.v3.js"></script>
<script>
var transformCSSProp = (function(property) {
var prefixes = ['webkit', 'ms', 'Moz', 'O'],
i = -1,
n = prefixes.length,
s = document.body.style;
if (property.toLowerCase() in s)
return property.toLowerCase();
while (++i < n)
if (prefixes[i] + property in s)
return '-' + prefixes[i].toLowerCase() + property.replace(/([A-Z])/g, '-$1').toLowerCase();
return false;
})('Transform');
var flickrApiKey = 'ea621d507593aa247dcaa792268b93d7';
var maxImageSize = 150;
var data = [];
var buffer, lastHeight, dimensions, imagesPerRow, imagesPerPage, imageSize, nest;
// var outer = d3.select('div.grid')
// .on('scroll', render);
d3.select(window)
.on('resize', setDimensions)
.on('scroll', render);
var inner = d3.select('div.grid');
// We pull down 500,000 images, but flickr only gives us 4,000 max, so there will be repeats
for (var i = 0; i < 100; i++) {
d3.json('https://api.flickr.com/services/rest/?' +
'method=flickr.photos.search&' +
'api_key=ea621d507593aa247dcaa792268b93d7&' +
'tags=mountains,forest,beach&' +
'sort=interestingness-desc&' +
'media=photos&' +
'extras=url_q&' +
'format=json&' +
'nojsoncallback=1&' +
'per_page=500' +
'page=' + i,
function(err, json) {
if (err) return console.log(err);
data = data.concat(json.photos.photo.map(function(d) {
return d.url_q;
}));
setDimensions();
})
}
function setDimensions() {
dimensions = [inner.node().clientWidth, innerHeight];
imagesPerRow = Math.ceil(dimensions[0] / maxImageSize);
imagesPerPage = Math.ceil(dimensions[1] / maxImageSize);
imageSize = dimensions[0] / imagesPerRow;
buffer = imagesPerPage;
nest = data.reduce(function(prev, item, i) {
var group = Math.floor(i / imagesPerRow);
(prev[group]) ? prev[group].value.push(item) : prev.push({
key: group,
value: [item]
});
return prev;
}, []);
var newHeight = Math.ceil(nest.length * imageSize) + 'px';
inner.style('height', newHeight);
if (newHeight > dimensions[1] && lastHeight < dimensions[1]) {
lastHeight = newHeight;
setDimensions();
}
//inner.style('background-image', 'url(grid-' + imagesPerRow + 'sm.png)');
inner.selectAll('div.row')
.call(styleRows);
inner.selectAll('img')
.call(styleImages);
render();
}
function render() {
if (!nest) return;
var scrollY = window.scrollY;
var count = imagesPerPage + buffer * 2;
var offset = Math.max(0, Math.floor(scrollY / imageSize) - buffer);
var dataSlice = nest.slice(offset, offset + count);
var pre = [],
post = [];
for (var i = 0; i < buffer * 2; i++) {
pre.push({
key: offset - i - 1,
value: []
});
post.push({
key: offset + count + i,
value: []
});
}
dataSlice = pre.concat(dataSlice, post);
var rows = inner.selectAll('div')
.data(dataSlice, function(d) {
return d.key;
});
reuseNodes.call(rows.enter(), rows.exit())
.attr('class', 'row')
.call(styleRows);
rows.exit()
.remove();
var images = rows.selectAll('img').data(function(d) {
return d.value;
});
images.enter()
.append('img')
.classed('hidden', true)
.call(styleImages);
images
.attr('src', function(d) {
return d;
})
.on('load', function() {
d3.select(this)
.classed('hidden', false);
});
images.exit()
.remove();
function reuseNodes(exitNodes) {
return this.select(function() {
var reusableNode;
for (var i = -1, n = exitNodes[0].length; ++i < n;) {
if (reusableNode = exitNodes[0][i]) {
exitNodes[0][i] = undefined;
d3.select(reusableNode)
.selectAll('img')
.classed('hidden', true);
return reusableNode;
}
}
return this.appendChild(document.createElement('div'));
});
}
}
function styleRows(selection) {
selection
.attr('class', 'row grid-' + imagesPerRow)
.style('height', imageSize + 'px')
.style(transformCSSProp, function(d, i) {
return 'translate3d(0,' + d.key * imageSize + 'px,0)';
});
}
function styleImages(selection) {
selection
.style('width', imageSize - 4 + 'px')
.style('height', imageSize - 4 + 'px');
}
//d3.select(self.frameElement).attr('scrolling', null);
</script>
</body>
<!DOCTYPE html>
<meta charset="utf-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<style>
html, body, iframe {
padding: 0;
margin: 0;
border: 0;
width: 100%;
height: 100%;
}
iframe {
position: absolute;
top: 0;
left: 0;
right: 0;
bottom: 0;
}
</style>
<!-- bl.ocks.org uses an iframe with scrolling="no"
which for some reason cannot be overridden with
self.frameSet.removeAttribute(scrolling)
so we embed another iframe with scrolling enabled
Yuck. -->
<iframe src="image-grid.html" marginwidth="0" marginheight="0"></iframe>
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment