block by wboykinm f3cdebbbc0e6a1b4c4ae

f3cdebbbc0e6a1b4c4ae

Full Screen

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.

index.html

<!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>