A Spinning globe with glowing city markers in D3.
Note: This visualization constantly recalculates and reprojects the position of over 4000 markers and thus needs a lot of computation power. Older computer and mobile devices might struggle with the rendering process. Furthermore the visualization is only optimized for Chrome browsers and might look different than expected when opened in other browsers.
By Patrick Stotz, inspired by Curran Kelleher‘s idea to plot a world map in D3 by using the GeoNames database as seen in his great Introdction to D3 video.
<!DOCTYPE html>
<html>
<head>
<meta charset="utf-8">
<title>Spinning globe with glowing city markers in D3</title>
<script src="//cdnjs.cloudflare.com/ajax/libs/d3/3.5.5/d3.min.js"></script>
<style>
body {
background: #111;
}
</style>
</head>
<body>
<svg width="820" height="620"></svg>
<script>
// Map configuration
var width = 820;
var height = 620;
var rScale = d3.scale.sqrt();
var peoplePerPixel = 70000;
var max_population = [];
// Configuration for the spinning effect
var time = Date.now();
var rotate = [0, -45];
var velocity = [.005, -0];
// set projection type and paremetes
var projection = d3.geo.orthographic()
.scale(300)
.translate([(width / 2) + 100, height / 2])
.clipAngle(90)
.precision(0.3);
// create path variable, empty svg element and group container
var path = d3.geo.path()
.projection(projection);
var svg = d3.select("svg");
var g = svg.append("g");
// drawing dark grey spehere as landmass
g.append("path")
.datum({type: "Sphere"})
.attr("class", "sphere")
.attr("d", path)
.attr("fill", "rgba(10,10,10,0.9)");
// loading city locations from geoJSON
d3.json("geonames5000.geojson", function(error, data) {
// Handle errors getting and parsing the data
if (error) { return error; }
// setting the circle size (not radius!) according to the number of inhabitants per city
population_array = [];
for (i = 0; i < data.features.length; i++) {
population_array.push(data.features[i].properties.population);
}
max_population = population_array.sort(d3.descending)[0]
var rMin = 0;
var rMax = Math.sqrt(max_population / (peoplePerPixel * Math.PI));
rScale.domain([0, max_population]);
rScale.range([rMin, rMax]);
path.pointRadius(function(d) {
return d.properties ? rScale(d.properties.population) : 1;
});
// Drawing transparent circle markers for cities
g.selectAll("path.cities").data(data.features)
.enter().append("path")
.attr("class", "cities")
.attr("d", path)
.attr("fill", "#e9d362")
.attr("fill-opacity", 0.6);
// start spinning!
spinning_globe();
});
function spinning_globe(){
d3.timer(function() {
// get current time
var dt = Date.now() - time;
// get the new position from modified projection function
projection.rotate([rotate[0] + velocity[0] * dt, rotate[1] + velocity[1] * dt]);
// update cities position = redraw
svg.selectAll("path.cities").attr("d", path);
});
}
// hackish approach to get bl.ocks.org to display individual height
d3.select(self.frameElement).style("height", height + "px");
</script>
</body>
</html>