Here I explain how I used D3 with the mapping library Leaflet to help visualise the locations of supermarkets in the UK, to produce something like this:
If you want to find out more about Voronoi diagrams you can read my write up on visualising supermarkets with a Voronoi diagram.
Setting up the map
Here I’m creating a new map to put in the element with the id of map
with the custom Mapbox layer zetter.i73ka9hn
. I wanted to use one of the free map tile servers provided by Open Street Map but found that the bright colours on the map made the overlay hard to see. Mapbox makes it easy for you to build your own map tiles from Open Street Map data by choosing which map features to display and how they should look. The Mapbox js library is the Leaflet js library with a few additions to make working with Mapbox tiles and other Mapbox data easier.
Loading the data
This code uses D3 to turn a CSV file into javascript objects for you to work with. If you had the CSV:
Then points
would be populated with:
Adding our layer
I drew the points and Voronoi diagram onto a new layer above the base map.
To Add a layer with Leaflet we can call addLayer
with an ILayer object. This needs to at least implement onAdd
which is called by Leaflet after you add the layer. By hooking on the viewreset
(triggered when zoom changed) and moveend
(triggered after the map is moved) events we make sure the layer is redrawn when it needs to be.
Filtering and Projecting
The more points we have, the more time the points and Voronoi diagram will take to draw. So we don’t waste time drawing things we cannot see, we filter the data points to those that are in, or close to1, the part of the map that is currently displayed.
Our data has latitude and longitude data for the position of each supermarket. To be able to draw these points on our layer we have to project them onto x-y pixel coordinates, so for the latlng
of every filtered data point:
The D3 voronoi
function doesn’t work when it’s given multiple points with the same x-y coordinate2 so after projecting we must also filter to unique x-y coordinate pairs.
Drawing the points
This shows off one of the more interesting features of D3. Normally when manipulating the DOM with a library such as jQuery you would build one element at a time in a loop. D3 allows you to ‘join’ DOM manipulation to the data set to multiply any operation to multiple elements.
Here we first join to our dataset with data(filteredPoints)
and then call enter
to tell D3 that any subsequent operations should be performed for every object in filteredPoints
. So now when we call .append("g")
a new SVG <g>
node is added for every point in our dataset. Any more operations that we chain on this will also be multiplied out such as calling .append("circle")
and setting attributes on it for every point we want to draw on our map. When a function is passed to attr
, the function is called for every object in out dataset to generate the value set.
Mike Bostock, the creator of D3, has written a great post that explains more about joins.
Drawing the Voronoi paths
D3 provides a voronoi
function that creates Voronoi cells from an array of points. There is some plumbing we have to do to be able to use this. First we create our own voronoi function that knows how to get the x-y coordinate for any point passed into it:
Then we build the voronoi cells. voronoi
returns an array of cells, each with a reference back to the point it was constructed from. We loop over them to give each point a reference to the cell since points are the dataset we are working from:
Last we build a function that formats a array of coordinates that makes up a cell into a string that represents an SVG path. Then we can append all the paths to our SVG:
The Result
For me this was great chance to use D3 and Leaflet. Both libraries are well-documented and a pleasure to use. I found that D3 does take a bit of getting used to because of the breadth of features it offers and the unusual way that joins effect method chaining.
You can see all the code on github and play with the finished result. It should be easy to use the same code to display any dataset which contains latitude and longitude coordinates on a map.
References
There were two examples of using D3 by Mike Bostock that were great references when building this:
The map layer of the visualisation on this page is copyright Mapbox and OpenStreetMap.
This tutorial was updated on 2015-07-10 to be compatable with v4 of the Mapbox web API.
-
Points that are just outside the map won’t themselves visible but could produce a Voronoi cell that is in the current view. ↩
-
From the D3
voronoi
documentation: “If any vertices are coincident … the behavior of this method is undefined: most likely, invalid polygons will be returned! You should filter invalid vertices, and consolidate coincident vertices, before computing the tessellation.” ↩