Full Stack D3 and Data Visualization
Full Stack D3 and Data Visualization
© 2019 Fullstack.io
All rights reserved. No portion of the book manuscript may be reproduced, stored in a retrieval
system, or transmitted in any form or by any means beyond the number of purchased copies,
except for a single backup or archival copy. The code may be used freely in your projects,
commercial or otherwise.
The authors and publisher have taken care in preparation of this book, but make no expressed
or implied warranty of any kind and assume no responsibility for errors or omissions. No
liability is assumed for incidental or consequential damagers in connection with or arising out
of the use of the information or programs container herein.
Published by Fullstack.io.
FULLSTACK .io
Contents
Book Revision . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 1
Bug Reports . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 1
Be notified of updates via Twitter . . . . . . . . . . . . . . . . . . . . . . . . . 1
We’d love to hear from you! . . . . . . . . . . . . . . . . . . . . . . . . . . . . 1
Making a Scatterplot . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 33
Intro . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 33
Deciding the chart type . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 33
Steps in drawing any chart . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 34
Access data . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 37
Create chart dimensions . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 37
Draw canvas . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 41
Create scales . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 42
The concept behind scales . . . . . . . . . . . . . . . . . . . . . . . . . . . 43
CONTENTS
Interactions . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 139
CONTENTS
d3 events . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 139
Don’t use fat arrow functions . . . . . . . . . . . . . . . . . . . . . . . . . . . 143
Destroying d3 event listeners . . . . . . . . . . . . . . . . . . . . . . . . . 144
Bar chart . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 146
Scatter plot . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 160
Voronoi . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 165
Changing the hovered dot’s color . . . . . . . . . . . . . . . . . . . . . . 172
Line chart . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 175
d3.scan() . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 178
D3.js . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 589
What did we cover? . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 590
What did we not cover? . . . . . . . . . . . . . . . . . . . . . . . . . . . . 592
Going forward . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 595
How was your experience? . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 597
CONTENTS
Appendix . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 598
A. Generating our own weather data . . . . . . . . . . . . . . . . . . . . . . . 598
Chrome’s Color Contrast Tool . . . . . . . . . . . . . . . . . . . . . . . . . . . 599
B. Chart-drawing checklist . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 602
C. SVG elements cheat sheet . . . . . . . . . . . . . . . . . . . . . . . . . . . . 604
Changelog . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 605
Revision 8 . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 605
06-26-2019 . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 605
Revision 7 . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 606
06-14-2019 . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 606
Revision 6 . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 607
06-06-2019 . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 607
Revision 5 . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 607
06-03-2019 . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 607
Revision 4 . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 608
05-24-2019 . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 608
Revision 3 . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 608
05-17-2019 . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 608
CONTENTS 1
Book Revision
Revision 1 - 2019-05-12
Revision 2 - 2019-05-14
Revision 3 - 2019-05-17
Revision 4 - 2019-05-30
Revision 5 - 2019-06-03
Revision 6 - 2019-06-06
Revision 7 - 2019-06-14
Revision 8 - 2019-07-04
Bug Reports
If you’d like to report any bugs, typos, or suggestions just email us at: us@fullstack.io.
¹https://twitter.com/fullstackio
²mailto:us@fullstack.io
Making Your First Chart
Many books begin by talking about theory and abstract concepts. This is not
one of those books. We’ll dig in and create real charts right away! Once you’re
comfortable making your own charts, we’ll discuss how to integrate into websites,
data visualization fundamentals, and practical tips for chart design, along with other
goodies.
Getting started
To start, let’s make a line chart. Line charts are a great starting place because of their
popularity, but also because of their simplicity.
In this chapter, we’ll create a line chart that plots the daily temperature. Here’s
what our line chart will look like when we’re finished:
In the tutorial below, don’t worry too much about the details! We’re just going to
get our hands dirty and write the code to build this line chart. This will give us a
good foundation to dive deeper into each concept in Chapters 2 and 3, in which we’ll
create more complex charts.
Making Your First Chart 2
The dataset we’ll be analyzing contains 365 days of daily weather metrics. To make
it easy, we’ve provided a JSON file with this data in the code download folder named
nyc_weather_data.json. This file includes 2018 data for New York City.
We recommend that you create a dataset for your own location to keep the data
tangible, plus you’ll discover new insights about where you live! Refer to Appendix A
for instructions — it should only take a few minutes and the charts we make together
will be uniquely yours.
If you’re using the provided dataset, rename the file to my_weather_data.json.
This way, our code examples will know where to find the weather data.
Let’s get our webpage up and running. Find the index.html file and open it in your
browser. The url will start with file:///. This is a very simple webpage — we’re
rendering one element and loading two javascript files.
code/01-making-your-first-chart/completed/index.html
7 <div id="wrapper"></div>
8
9 <script src="./../../d3.v5.js"></script>
10 <script src="./chart.js"></script>
The page should be blank except for one div with an id of wrapper — this is where
our chart will live.
The first script that we’re loading is d3.js — this will give us access to the entire
library.
At this point, we want access to the whole library, but d3.js is made up of at least
thirty modules. Later, we’ll discuss how to import only the necessary modules to
keep your build lean.
Next, our index.html file loads the javascript file in which we’ll write our chart
code: chart.js.
Let’s open up the 01-making-your-first-chart/draft/chart.js file in a code
editor and dig in.
Making Your First Chart 3
If you don’t already have a code editor, any program that lets you open a file and
edit the text will do! I personally use Visual Studio Code, which I recommend —
it’s straightforward enough for beginners, but has many configuration options and
extensions for power users.
https://code.visualstudio.com/
drawLineChart()
The only thing happening so far is that we’re defining a function named drawLineChart()
and running it.
³https://github.com/d3/d3-fetch
Making Your First Chart 4
code/01-making-your-first-chart/completed/chart.js
await is a JavaScript keyword that will pause the execution of a function until
a Promise is resolved. This will only work within an async function — note that
the drawLineChart() function declaration is preceded by the keyword async.
Now when we load our webpage we should get a CORS error in the console.
CORS error
Making Your First Chart 5
a. node.js
I would recommend using this method because it has live reload built in, meaning
that our page will update when we save our changes. No page refresh necessary!
If you don’t have node.js installed, take a minute to install it (instructions here⁴).
You can check whether or not node.js is already installed by using the node -v
command in your terminal — if it responds with a version number, you’re good to
go! node.js should also come with npm, which is short for Node Package Manager
Once node.js and npm are installed, run the following command in your terminal.
This will install live-server⁵, a simple static server that has live reload built-in. To
start your server, run live-server in your terminal from the root /code folder —
it will even open a new browser window for you!
a. python
If you have python (version 3) installed already, you can use the Python 3 http server
instead. Start it up by running the command python -m http.server 8080 in your
terminal from the root /code folder.
⁴https://docs.npmjs.com/downloading-and-installing-node-js-and-npm
⁵https://github.com/tapio/live-server
Making Your First Chart 6
The particular server doesn’t matter — the key idea is that if you want to load a
file from JavaScript, you need to do it from a webserver, and these tools are an easy
solution for a development environment. Make sure that you are in the root /code
folder when you start either server.
Now we should have a server on port 8080. Load localhost:8080⁶ in your web browser
and you’ll see a directory of code for each chapter, which looks something like this:
Directory screenshot
For all of our code examples, there will be a finished version in a sibling /completed
folder — in this chapter, look at /code/01-making-your-first-chart/completed/
if you want a reference. Or view to completed chart at http://localhost:8080/01-
making-your-first-chart/completed/.
We’ll still see a blank page since we haven’t drawn anything on our page yet, but
that error should be gone!
console.log(dataset)
We can see that our dataset is array of objects, with one object per day.
Since each day seems to have the same structure, let’s delete the last line and instead
log a single data point to get a clearer view.
We can use console.table() here, which is a great function for looking at array
or object values — as long as there aren’t too many!
console.table(dataset[0])
Our dataset
We have lots of information for each day! We can see metadata (date, time,
summary) and details about that day’s weather (cloudCover, sunriseTime, temperatureMax,
etc). If you want to read more about each metric, check out The Dark Sky API docs⁸.
⁸https://darksky.net/dev/docs#data-point
Making Your First Chart 8
To grab the correct metrics from each data point, we’ll need accessor functions.
Accessor functions convert a single data point into the metric value.
Lets try it out by creating a yAccessor function that will take a data point and return
the max temperature.
If you think of a dataset as a table, a data point would be a row in that table. In this
case, a data point represents an item in our dataset array: an object that holds
the weather data for one day.
Next, we’ll need an xAccessor function that will return a point’s date, which we
will use for plotting points on the x axis.
But look closer at the data point date value - notice that it is a string (eg.
"2018-12-25"). Unfortunately, this string won’t make sense on our x axis. How
could we know how far "2018-12-25" is from "2018-12-29"?.
We need to convert the string into a JavaScript Date, which is an object that
represents a single point in time. Thankfully, d3 has a d3-time-format⁹ module with
methods for parsing and formatting dates.
The d3.timeParse() method…
For example, d3.timeParse("%Y") will parse a string with just a year (eg. "2018").
Let’s create a date parser function and use it to transform our date strings into date
objects:
code/01-making-your-first-chart/completed/chart.js
Great! Now when we call xAccessor(dataset[0]), we’ll get the first day’s date.
If you look up d3 examples, you won’t necessarily see accessor functions used.
When I first started learning d3, I never thought about using them. Since then, I’ve
learned my lesson and paid the price of painstakingly picking through old code and
updating individual lines. I want to save you that time so you can spend it making
even more wonderful charts.
Defining accessor functions might seem like unnecessary overhead right now,
especially with this simple example. However, creating a separate function to read
the values from our data points helps us in a few ways.
⁹https://github.com/d3/d3-time-format
Making Your First Chart 10
• Easy changes: every chart is likely to change at least once — whether that
change is due to business requirements, design, or data structure. These chang-
ing requirements are especially prevalent when creating dashboards with
dynamic data, where you might need to handle a new edge case two months
later. Having the accessor functions in one place at the top of a chart file makes
them easy to update throughout the chart.
• Documentation: having these functions at the top of a file can give you a quick
reminder of what metrics the chart is plotting and the structure of the data.
• Framing: sitting down with the data and planning what metrics we’ll need to
access is a great way to start making a chart. It’s tempting to rush in, then two
hours later realize that another type of chart would be better suited to the data.
Now that we know how to access our dataset, we need to prepare to draw our chart.
Chart dimensions
The wrapper contains the entire chart: the data elements, the axes, the labels, etc.
Every SVG element will be contained inside here.
Making Your First Chart 11
The bounds contain all of our data elements: in this case, our line.
This distinction will help us separate the amount of space we need for extraneous
elements (axes, labels), and let us focus on our main task: plotting our data. One
reason this is so important to define up front is the inconsistent and unfamiliar way
SVG elements are sized.
When adding a chart to a webpage, we start with the amount of space we have
available for the chart. Then we decide how much space we need for the margins,
which will accommodate the chart axes and labels. What’s left is how much space
we have for our data elements.
We will rarely have the option to decide how large our timeline is and then build up
from there. Our charts will need to be accommodating of window sizes, surrounding
text, and more.
While wrapper and bounds isn’t terminology that you’ll see in widespread use, it
will be helpful for reference in this book. Defining these concepts also helps with
thinking about chart dimensions and remembering to make space for your axes.
Let’s define a dimensions object that will contain the size of the wrapper and the
margins. We’ll have one margin defined for each side of the chart: top, right, bottom,
and left. For consistency, we’ll mimic the order used for CSS properties.
code/01-making-your-first-chart/completed/chart.js
12 let dimensions = {
13 width: window.innerWidth * 0.9,
14 height: 400,
15 margin: {
16 top: 15,
17 right: 15,
18 bottom: 40,
19 left: 60,
20 },
21 }
Making Your First Chart 12
We want a small top and right margin to give the chart some space. The line or
the y axis might overflow the chart bounds. We’ll want a larger bottom and left
margin to create room for our axes.
Let’s compute the size of our bounds and add that to our dimensions object.
code/01-making-your-first-chart/completed/chart.js
22 dimensions.boundedWidth = dimensions.width
23 - dimensions.margin.left
24 - dimensions.margin.right
25 dimensions.boundedHeight = dimensions.height
26 - dimensions.margin.top
27 - dimensions.margin.bottom
¹⁰https://github.com/d3/d3-selection
Making Your First Chart 13
If you’ve ever used jQuery or written CSS selectors, these selector strings will be
familiar.
Let’s log our new wrapper variable to the console to see what it looks like.
console.log(wrapper)
We can see that it’s a d3 selection object, with _groups and _parents keys.
chart selection
d3 selection objects are a subclass of Array. They have a lot of great methods that
we’ll explore in depth later - what’s important to us right now is the _groups list
that contains our #wrapper div.
If we log svg to the console, we’ll see that it looks like our wrapper object. However,
if we expand the _groups key, we’ll see that the linked element is our new <svg>
element.
One trick to make sure we’re grabbing the correct element is to hover the logged
DOM element. If we expand the _groups object and hover over the <svg> element,
the browser will highlight the corresponding DOM element on the webpage.
On hover, the browser will also show the element’s size: 300px by 150px. This is the
default size for SVG elements in Google Chrome, but it will vary between browsers
and even browser versions. SVG elements don’t scale the way most DOM elements
do — there are many rules that will be unfamiliar to an experienced web developer.
To maintain control, let’s tell our <svg> element what size we want it to be.
d3 selection objects have an .attr() method that will add or replace an attribute on
the selected DOM element. The first argument is the attribute name and the second
argument is the value.
• any method that selects or creates a new object will return the new selection
• any method that manipulates the current selection will return the same selec-
tion
This allows us to keep our code concise by chaining when we’re using multiple
methods. For example, we can rewrite the above code as:
In this book, we’ll follow the common d3 convention of using 4 space indents for
methods that return the same selection. This will make it easy to spot when our
selection changes.
Since we’re not going to re-use the svg variable, we can rewrite the above code as:
code/01-making-your-first-chart/completed/chart.js
When we refresh our index.html page, we should now see that our <svg> element
is the correct size. Great!
Making Your First Chart 16
We’re using backticks (`) instead of quotes (' or ") to create our translate string.
This lets us use ES6 string interpolation — if you’re unfamiliar, read more here.
https://babeljs.io/docs/en/learn/#template-strings
If we look at our Elements panel, we can see our new <g> element.
Making Your First Chart 17
g element
We can see that the <g> element size is 0px by 0px — instead of taking a width
or height attribute, a <g> element will expand to fit its contents. When we start
drawing our chart, we’ll see this in action.
We’ve all seen over-dramatized timelines with a huge drop, only to realize that the
change is relatively small. When defining an axis, we’ll often want to start at 0 to
show scale. We’ll go over this more when we talk about types of data.
As an example, let’s grab a sample day’s data — say it has a maximum temperature
of 55°F. We could draw our point 55 pixels above the bottom of the chart, but that
won’t scale with our boundedHeight.
Making Your First Chart 18
Additionally, if our lowest temperature is below 0 we would have to plot that value
below the chart! Our y axis wouldn’t be able to handle all of our temperature values.
To plot the max temperature values in the correct spot, we need to convert them
into pixel space.
d3’s d3-scale¹¹ module can create different types of scales. A scale is a function that
converts values between two domains.
For our y axis, we want to convert values from the temperature domain to the pixel
domain. If our chart needs to handle temperatures from 10°F to 100°F, a day with a
max of 55°F will be halfway up the y axis.
Let’s create a scale that converts those degrees into a y value. If our y axis is 200px
tall, the y scale should convert 55°F into 100, the halfway point on the y axis.
Our dataset
d3-scale¹² can handle many different types of scales - in this case, we want to use
d3.scaleLinear() because our y axis values will be numbers that increase linearly.
To create a new scale, we need to create an instance of d3.scaleLinear().
¹¹https://github.com/d3/d3-scale
¹²https://github.com/d3/d3-scale
Making Your First Chart 19
code/01-making-your-first-chart/completed/chart.js
Let’s start with the domain. We’ll need to create an array of the smallest and largest
numbers our y axis will need to handle — in this case the lowest and highest max
temperature in our dataset.
The d3-array¹³ module has a d3.extent() method for grabbing those numbers.
d3.extent() takes two parameters:
Let’s test this out by logging d3.extent(dataset, yAccessor) to the console. The
output should be an array of two values: the minimum and maximum temperature
in our dataset. Perfect!
Let’s plug that into our scale’s domain:
code/01-making-your-first-chart/completed/chart.js
Next, we need to specify the range. As a reminder, the range is the highest
and lowest number we want our scale to output — in this case, the maximum &
minimum number of pixels our point will be from the x axis. We want to use our
boundedHeight to stay within our margins. Remember, SVG y-values count from
top to bottom so we want our range to start at the top.
¹³https://github.com/d3/d3-array
Making Your First Chart 20
code/01-making-your-first-chart/completed/chart.js
We just made our first scale function! Let’s test it by logging some values to the
console. At what y value is the freezing point on our chart?
console.log(yScale(32))
The outputted number should tell us how far away the freezing point will be from
the bottom of the y axis.
If this returns a negative number, congratulations! You live in a lovely, warm place.
Try replacing it with a number that “feels like freezing” to you. Or highlight another
temperature that’s meaningful to you.
Let’s visualize this threshold by adding a rectangle covering all temperatures below
freezing. The SVG <rect> element can do exactly that. We just need to give it four
attributes: x, y, width, and height.
For more information about SVG elements, the MDN docs¹⁴ are a wonder-
ful resource: here is the page for <rect>¹⁵.
¹⁴https://developer.mozilla.org/en-US/docs/Web/SVG/Element
¹⁵https://developer.mozilla.org/en-US/docs/Web/SVG/Element/rect
Making Your First Chart 21
code/01-making-your-first-chart/completed/chart.js
Now we can see a black rectangle spanning the width of our bounds.
Let’s make it a frosty blue to connote “freezing” and decrease its visual importance.
You can’t style SVG elements with background or border — instead, we can use
fill and stroke respectively. We’ll discuss the differences later in more depth. As
we can see, the default fill for SVG elements is black and the default stroke color is
none with a width of 1px.
Making Your First Chart 22
code/01-making-your-first-chart/completed/chart.js
Let’s look at the rectangle in the Elements panel to see how the .attr() methods
manipulated it.
<rect
x="0"
width="1530"
y="325.7509689922481"
height="24.24903100775191"
fill="rgb(224, 243, 243)"
></rect>
Looking good!
Some SVG styles can be set with either a CSS style or an attribute value, such as
fill, stroke, and stroke-width. It’s up to you whether you want to set them with
.style() or .attr(). Once we’re familiar with styling our charts, we’ll apply
classes using .attr("class", "class-name") and add styles to a separate CSS
file.
In this code, we’re using .attr() to set the fill because an attribute has a lower CSS
precedence than linked stylesheets, which will let us overwrite the value. If we used
.style(), we’d be setting an inline style which would require an !important CSS
declaration to override.
Making Your First Chart 23
Let’s move on and create a scale for the x axis. This will look like our y axis but, since
we’re working with date objects, we’ll use a time scale which knows how to handle
date objects.
code/01-making-your-first-chart/completed/chart.js
Now that we have our scales defined, we can start drawing our chart!
The d attribute will take a few commands that can be capitalized (if giving an absolute
value) or lowercased (if giving a relative value):
Making Your First Chart 24
d shape example
More d commands exist, but thankfully we don’t need to learn them. d3’s module
d3-shape¹⁶ has a d3.line() method that will create a generator that converts data
points into a d string.
code/01-making-your-first-chart/completed/chart.js
We set these values with the x and y method, respectively, which each take one
parameter: a function to convert a data point into an x or y value.
We want to use our accessor functions, but remember: our accessor functions return
the unscaled value.
We’ll transform our data point with both the accessor function and the scale to
get the scaled value in pixel space.
code/01-making-your-first-chart/completed/chart.js
Let’s feed our dataset to our line generator to create the d attribute and tell the line
what shape to be.
Success! We have a chart with a line showing our max temperature for the whole
year.
Something looks wrong, though:
Making Your First Chart 26
Our line!
Remember that SVG elements default to a black fill and no stroke, which is why we
see this dark filled-in shape. This isn’t what we want! Let’s add some styles to get an
orange line with no fill.
code/01-making-your-first-chart/completed/chart.js
Our line!
We’re almost there, but something is missing. Let’s finish up by drawing our axes.
¹⁷https://github.com/d3/d3-axis
Making Your First Chart 28
There is one method for each orientation, which will specify the placement of labels
and tick marks:
• axisTop
• axisRight
• axisBottom
• axisLeft
Following common convention, we want the labels of our y axis to be to the left of
the axis line, so we’ll use d3.axisLeft() and pass it our y scale.
code/01-making-your-first-chart/completed/chart.js
When we call our axis generator, it will create a lot of elements — let’s create a g
element to hold all of those elements and keep our DOM organized. Then we’ll pass
that new element to our yAxisGenerator function to tell it where to draw our axis.
yAxisGenerator(yAxis)
This method works but it will break up our chained methods. To fix this, d3 selections
have a .call() method that will execute the provided function with the selection
as the first parameter.
We can use .call() to:
Note that this code does exactly the same thing as the snippet above - we are passing
the function yAxisGenerator to .call(), which then runs the function for us.
Making Your First Chart 29
code/01-making-your-first-chart/completed/chart.js
Y axis
The small notches perpendicular to the axis are called tick marks. d3 has made
behind-the-scenes decisions about how many tick marks to make and how far apart
to draw them. We’ll learn more about how to customize this later.
Making Your First Chart 30
Axis
Let’s create the x axis in the same way, this time using d3.axisBottom().
code/01-making-your-first-chart/completed/chart.js
82 const xAxisGenerator = d3.axisBottom()
83 .scale(xScale)
Alright! Now let’s create another <g> element and draw our axis.
This would create our axis directly under our bounds (in the DOM).
However, it’s a good idea to create a <g> element to contain our axis elements for
three main reasons:
2. if we want to remove or update our axis, we’ll want an easy way to target all
of the elements
3. modifying our whole axis at once, for example when we want to move it
around.
x axis on top
Why didn’t .axisBottom() draw the axis in the right place? d3’s axis generator
functions know where to place the tick marks and tick labels relative to the axis line,
but they have no idea where to place the axis itself.
To move the x axis to the bottom, we can shift the x axis group, similar to how we
shifted our chart bounds using a CSS transform.
code/01-making-your-first-chart/completed/chart.js
85 const xAxis = bounds.append("g")
86 .call(xAxisGenerator)
87 .style("transform", `translateY(${
88 dimensions.boundedHeight
89 }px)`)
And just like that we’re done making our first chart!
Making Your First Chart 32
Next, let’s dive into making a slightly more complex chart and talk more about how
d3 works for a deeper understanding of the concepts we just learned.
Making a Scatterplot
Intro
Now that we’ve created our first chart, let’s create another chart that’s a little more
complex. At the end of this chapter, we’ll have a deeper understanding of each step
required to make a chart in d3.
There are endless questions we could ask our weather dataset — many of them ask
about the relationship between different metrics. Let’s investigate these two metrics:
• dew point is the highest temperature (°F) at which dew droplets form
• humidity is the amount of water vapor in the air
I would expect them to be correlated — high humidity should cause a higher dew
point temperature, right? Let’s dive in and find out!
We’ll plot each data point (in this case, a single day) as a dot. If we wanted to involve
a third metric, we could even add another dimension by changing the color or the
size of each dot.
Making a Scatterplot 34
The great thing about scatterplots is that when we’re finished plotting the chart, we’ll
get a clear view of the relationship between the two metrics. We’ll talk more about
the potential patterns in Chapter 8.
1. Access data
Look at the data structure and declare how to access the values we’ll need
2. Create chart dimensions
Declare the physical (i.e. pixels) chart parameters
3. Draw canvas
Render the chart area and bounds element
4. Create scales
Create scales for every data-to-physical attribute in our chart
Making a Scatterplot 35
5. Draw data
Render your data elements
6. Draw peripherals
Render your axes, labels, and legends
7. Set up interactions
Initialize event listeners and create interaction behavior - we’ll get to this step
in Chapter 5
We have a Chart drawing checklist PDF in the Advanced package for easy
reference. Feel free to print it out or save it somewhere to give you an outline
in the future!
Making a Scatterplot 36
¹⁸http://localhost:8080/02-making-a-scatterplot/draft/
Making a Scatterplot 37
Access data
As we saw in Chapter 1, this step will be quick! We can utilize d3.json() to grab
the my_weather_data.json file.
code/02-making-a-scatterplot/completed/draw-scatter.js
The next part of the Access data step is to create our accessor functions. Let’s log
the first data point to the console to look at the available keys.
We can see the metrics we’re interested in as humidity and dewPoint. Let’s use
those to define our accessor functions.
code/02-making-a-scatterplot/completed/draw-scatter.js
Perfect! Now that we can access our data, we can move to the next step.
Ideally, the chart will be as large as possible while still fitting on our screen.
To fix this problem, we want to use either the height or the width of the window,
whichever one is smaller. And because we want to leave a little bit of whitespace
around the chart, we’ll multiply the value by 0.9 (so 90% of the total width or height).
d3-array can help us out here with the d3.min method. d3.min takes two arguments:
Though in this case we won’t need to specify the second parameter because it defaults
to an identity function and returns the value.
code/02-making-a-scatterplot/completed/draw-scatter.js
There is a native browser method (Math.min) that will also find the lowest number
— why wouldn’t we use that? Math.min is great, but there are a few benefits to
d3.min:
You can see how d3.min would be preferable when creating charts, especially when
using dynamic data.
Now let’s use our width variable to define the chart dimensions:
let dimensions = {
width: width,
height: width,
}
• the wrapper is your entire SVG element, containing your axes, data elements,
and legends
• the bounds live inside of the wrapper, containing just the data elements
Having margins around the bounds allows us to allocate space for our static chart
elements (axes and legends) while allowing the charting area to be dynamically sized
based on the available space.
Making a Scatterplot 40
Chart terminology
We want a small top and right margin to give the chart some space. Dots near the
top or right of the chart or the y axis’s topmost tick label might overflow our bounds
(because the position of the dot is technically the center of the dot, but the dot has a
radius).
We’ll want a larger bottom and left margin to create room for our axes.
code/02-making-a-scatterplot/completed/draw-scatter.js
16 let dimensions = {
17 width: width,
18 height: width,
19 margin: {
20 top: 10,
21 right: 10,
22 bottom: 50,
23 left: 50,
24 },
25 }
Lastly, we want to define the width and height of our bounds, calculated from the
space remaining after we add the margins.
Making a Scatterplot 41
code/02-making-a-scatterplot/completed/draw-scatter.js
26 dimensions.boundedWidth = dimensions.width
27 - dimensions.margin.left
28 - dimensions.margin.right
29 dimensions.boundedHeight = dimensions.height
30 - dimensions.margin.top
31 - dimensions.margin.bottom
You might be asking: why do we have to be explicit about the chart dimensions?
Generally when developing for the web we can let elements size themselves to fit
their contents or to fill the available space. That’s not an option here for a few
reasons:
Draw canvas
Let’s make some SVG elements! This step will look exactly like our line chart code.
First, we find an existing DOM element (#wrapper), and append an <svg> element.
Then we use attr to set the size of the <svg> to our dimensions.width and
dimensions.height. Note that these sizes are the size of the “outside” of our plot.
Everything we draw next will be within this <svg>.
Making a Scatterplot 42
code/02-making-a-scatterplot/completed/draw-scatter.js
35 const wrapper = d3.select("#wrapper")
36 .append("svg")
37 .attr("width", dimensions.width)
38 .attr("height", dimensions.height)
Next, we create our bounds and shift them to accommodate our top & left margins.
code/02-making-a-scatterplot/completed/draw-scatter.js
40 const bounds = wrapper.append("g")
41 .style("transform", `translate(${
42 dimensions.margin.left
43 }px, ${
44 dimensions.margin.top
45 }px)`)
Above, we create a <g> (think “group”) element and we use the transform CSS
property to move it to the right and down (note that the left margin pushes our
bounds to the right, and a top margin pushes our bounds down).
This bounds is the “inner” part of our chart that we will use for our data elements.
Create scales
Before we draw our data, we have to figure out how to convert numbers from
the data domain to the pixel domain.
Let’s start with the x axis. We want to decide the horizontal position of each day’s
dot based on its dew point.
To find this position we use a d3 scale object, which helps us map our data to pixels.
Let’s create a scale that will take a dew point (temperature) and tell us how far to the
right a dot needs to be.
This will be a linear scale because the input (dew point) and the output (pixels) will
be numbers that increase linearly.
Making a Scatterplot 43
For a simple example, let’s pretend that the temperatures in our dataset range from
0 to 100 degrees.
In this case, converting from temperature to pixels is easy: a temperature of 50
degrees maps to 50 pixels because both range and domain are [0,100].
But the relationship between our data and the pixel output is rarely so simple. What
if our chart was 200 pixels wide? What if we have to handle negative temperatures?
Mapping between metric values and pixels is one of the areas in which d3 scales
shine.
1. an array
2. an accessor function that extracts the metric value from a data point. If not
specified, this defaults to an identity function d => d.
We’ll pass d3.extent() our dataset and our xAccessor() function and get the min
and max temperatures we need to handle (in [min, max] format).
Making a Scatterplot 44
code/02-making-a-scatterplot/completed/draw-scatter.js
This scale will create a perfectly useable chart, but we can make it slightly friendlier.
With this x scale, our x axis will have a domain of [11.8, 77.26] — the exact min
and max values from the dataset. The resulting chart will have dots that extend all
the way to the left and right edges.
While this works, it would be easier to read the axes if the first and last tick marks
were round values. Note that d3 won’t even label the top and bottom tick marks of
an axis with a strange domain — it might be hard to reason about a chart that scales
up to 77.26 degrees. That number of decimal points gives too much unnecessary
information to the reader, making them do the next step of rounding the number to
a more tangible one.
Making a Scatterplot 45
d3 scales have a .nice() method that will round our scale’s domain, giving our x
axis friendlier bounds.
We can look at how .nice() modifies our x scale’s domain by looking at the values
before and after using .nice(). Note that calling .domain() without parameters on
an existing scale will output the scale’s existing domain instead of updating it.
console.log(xScale.domain())
xScale.nice()
console.log(xScale.domain())
With the New York City dataset, the domain changes from [11.8, 77.26] to [10,
80] — much friendlier! Let’s chain that method when we create our scale.
code/02-making-a-scatterplot/completed/draw-scatter.js
49 const xScale = d3.scaleLinear()
50 .domain(d3.extent(dataset, xAccessor))
51 .range([0, dimensions.boundedWidth])
52 .nice()
Creating our y scale will be very similar to creating our x scale. The only differences
are:
Making a Scatterplot 46
code/02-making-a-scatterplot/completed/draw-scatter.js
If we were curious about how .nice() modifies our y scale, we could log those
values.
console.log(d3.extent(dataset, yAccessor))
console.log(yScale.domain())
In this case, the domain changed from [0.27, 0.97] to [0.2, 1], which will create
a much friendlier chart.
Draw data
Here comes the fun part! Drawing our scatterplot dots will be different from how
we drew our timeline. Remember that we had one line that covered all of the data
points? For our scatter plot, we want one element per data point.
We’ll want to use the <circle>¹⁹ SVG element, which thankfully doesn’t need a
d attribute string. Instead, we’ll give it cx and cy attributes, which set its x and y
coordinates, respectively. These position the center of the circle, and the r attribute
sets the circle’s radius (half of its width or height).
Let’s draw a circle in the center of our chart to test it out.
¹⁹https://developer.mozilla.org/en-US/docs/Web/SVG/Element/circle
Making a Scatterplot 47
bounds.append("circle")
.attr("cx", dimensions.boundedWidth / 2)
.attr("cy", dimensions.boundedHeight / 2)
.attr("r", 5)
Test circle
Starting to get SVG elements mixed up? No worries! Our advanced package has an
SVG elements cheat sheet PDF to help remember what elements exist and what
attributes they want. Don’t worry if you don’t recognize any of the elements —
we’ll cover them all by the end of the book.
Making a Scatterplot 48
dataset.forEach(d => {
bounds
.append("circle")
.attr("cx", xScale(xAccessor(d)))
.attr("cy", yScale(yAccessor(d)))
.attr("r", 5)
})
Look at that! Now we’re starting to get a better sense of our data.
Making a Scatterplot 49
Dots!
While this method of drawing the dots works for now, there are a few issues we
should address.
• We’re adding a level of nesting, which makes our code harder to follow.
• If we run this function twice, we’ll end up drawing two sets of dots. When we
start updating our charts, we will want to draw and update our data with the
same code to prevent repeating ourselves.
To address these issues and keep our code clean, let’s handle the dots without using
a loop.
Data joins
Scratch that last block of code. D3 has functions that will help us address the above
problems.
We’ll start off by grabbing all <circle> elements in a d3 selection object. Instead of
using d3.selection’s .select() method, which returns one matching element, we’ll
use its .selectAll() method, which returns an array of matching elements.
Making a Scatterplot 50
This will seem strange at first — we don’t have any dots yet, why would we select
something that doesn’t exist? Don’t worry! You’ll soon become comfortable with this
pattern.
We’re creating a d3 selection that is aware of what elements already exist. If we
had already drawn part of our dataset, this selection will be aware of what dots were
already drawn, and which need to be added.
To tell the selection what our data look like, we’ll pass our dataset to the selection’s
.data() method.
When we call .data() on our selection, we’re joining our selected elements
with our array of data points. The returned selection will have a list of existing
elements, new elements that need to be added, and old elements that need to be
removed.
We’ll see these changes to our selection object in three ways:
• our selection object is updated to contain any overlap between existing DOM
elements and data points
• an _enter key is added that lists any data points that don’t already have an
element rendered
• an _exit key is added that lists any data points that are already rendered but
aren’t in the provided dataset
Making a Scatterplot 51
join schematic
Let’s get an idea of what that updated selection object looks like by logging it to the
console.
Remember, the currently selected DOM elements are located under the _groups key.
Before we join our dataset to our selection, the selection just contains an empty array.
That makes sense! There are no circles in bounds yet.
Making a Scatterplot 52
empty selection
However, the next selection object looks different. We have two new keys: _enter
and _exit, and our _groups array has an array with 365 elements — the number of
data points in our dataset.
.data selection
Let’s take a closer look at the _enter key. If we expand the array and look at one of
the values, we can see an object with a __data__ property.
Making a Scatterplot 53
For the curious, the namespaceURI key tells the browser that the element is a
SVG element and needs to be created in the “http://www.w3.org/2000/svg” names-
pace (SVG), instead of the default “http://www.w3.org/1999/xhtml” namespace
(XHTML).
Making a Scatterplot 54
If we expand the __data__ value, we will see one our data points.
Great! We can see that each value in _enter corresponds to a value in our dataset.
This is what we would expect, since all of the data points need to be added to the
DOM.
The _exit value is an empty array — if we were removing existing elements, we
would see those listed out here.
In order to act on the new elements, we can create a d3 selection object containing
just those elements with the enter method. There is a matching method (exit) for
old elements that we’ll need when we go over transitions in Chapter 4.
This looks just like any d3 selection object we’ve manipulated before. Let’s append
one <circle> for each data point. We can use the same .append() method we’ve
been using for single-node selection objects and d3 will create one element for each
data point.
Making a Scatterplot 55
When we load our webpage, we will still have a blank page. However, we will be able
to see lots of new empty <circle> elements in our bounds in the Elements panel.
Let’s set the position and size of these circles.
We can write the same code we would write for a single-node selection object. Any
attribute values that are functions will be passed each data point individually. This
helps keep our code concise and consistent.
Let’s make these dots a lighter color to help them stand out.
.attr("fill", "cornflowerblue")
dots
.enter().append("circle")
.attr("cx", d => xScale(xAccessor(d)))
.attr("cy", d => yScale(yAccessor(d)))
.attr("r", 5)
.attr("fill", color)
}
Let’s call this function with part of our dataset. The color doesn’t matter much —
let’s go with a dark grey.
grey dots
After one second, let’s call the function again with our whole dataset, this time with
a blue color. We’re adding a timeout to help distinguish between the two sets of dots.
Making a Scatterplot 57
setTimeout(() => {
drawDots(dataset, "cornflowerblue")
}, 1000)
When you refresh your webpage, you should see a set of grey dots, then a set of blue
dots one second later.
Each time we run drawDots(), we’re setting the color of only new circles. This
explains why the grey dots stay grey. If we wanted to set the color of all circles, we
could re-select all circles and set their fill on the new selection:
dots.enter().append("circle")
bounds.selectAll("circle")
.attr("cx", d => xScale(xAccessor(d)))
.attr("cy", d => yScale(yAccessor(d)))
.attr("r", 5)
.attr("fill", color)
}
Making a Scatterplot 58
In order to keep the chain going, d3 selection objects have a merge() method that will
combine the current selection with another selection. In this case, we could combine
the new enter selection with the original dots selection, which will return the full
list of dots. When we set attributes on the new merged selection, we’ll be updating
all of the dots.
dots
.enter().append("circle")
.merge(dots)
.attr("cx", d => xScale(xAccessor(d)))
.attr("cy", d => yScale(yAccessor(d)))
.attr("r", 5)
.attr("fill", color)
}
.join()
Since d3-selection version 1.4.0²⁰, there is a new .join()²¹ method that helps to
cut down on this code. .join() is a shortcut for running .enter(), .append(),
.merge(), and some other methods we haven’t covered yet. This allows us to write
the following code instead:
²⁰https://github.com/d3/d3-selection/releases/tag/v1.4.0
²¹https://github.com/d3/d3-selection/#selection_join
Making a Scatterplot 59
dots.join("circle")
.attr("cx", d => xScale(xAccessor(d)))
.attr("cy", d => yScale(yAccessor(d)))
.attr("r", 5)
.attr("fill", color)
}
While .join() is a great addition to d3, it’s still beneficial to understand the
.enter(), .append(), and .merge() methods. Most existing d3 code will use these
methods, and it’s important to understand the basics before getting fancy.
Don’t worry if this pattern still feels new — we’ll reinforce and build on what we’ve
learned when we talk about transitions. For now, let’s delete this example code,
uncomment our finished dots code, and move on with our scatter plot!
Draw peripherals
Let’s finish up our chart by drawing our axes, starting with the x axis.
We want our x axis to be:
To do this, we’ll create our axis generator using d3.axisBottom(), then pass it:
• our x scale so it knows what ticks to make (from the domain) and
• what size to be (from the range).
Making a Scatterplot 60
code/02-making-a-scatterplot/completed/draw-scatter.js
Next, we’ll use our xAxisGenerator() and call it on a new g element. Remember,
we need to translate the x axis to move it to the bottom of the chart bounds.
code/02-making-a-scatterplot/completed/draw-scatter.js
When we render our webpage, we should see our scatter plot with an x axis. As a
bonus, we can see how using .nice() on our scale ensures that our axis ends in
round values.
Let’s expand on our knowledge and create labels for our axes. Drawing text in an SVG
is fairly straightforward - we need a <text> element, which can be positioned with
an x and a y attribute. We’ll want to position it horizontally centered and slightly
above the bottom of the chart.
Making a Scatterplot 61
<text> elements will display their children as text — we can set that with our
selection’s .html() method.
code/02-making-a-scatterplot/completed/draw-scatter.js
We need to explicitly set the text fill to black because it inherits a fill value of none
that d3 sets on the axis <g> element.
Almost there! Let’s do the same thing with the y axis. First, we need an axis generator.
D3 axes can be customized in many ways. An easy way to cut down on visual clutter
Making a Scatterplot 62
is to tell our axis to aim for a certain number with the ticks method. Let’s aim for
4 ticks, which should give the viewer enough information.
code/02-making-a-scatterplot/completed/draw-scatter.js
Note that the resulting axis won’t necessarily have exactly 4 ticks. D3 will take the
number as a suggestion and aim for that many ticks, but also trying to use friendly
intervals. Check out some of the internal logic in the d3-array code — see how it’s
attempting to use intervals of 10, then 5, then 2?
There are many ways to configure the ticks for a d3 axis — find them all in the
docs. For example, you can specify their exact values by passing an array of values
to .tickValues().
https://github.com/d3/d3-array/blob/master/src/ticks.js#L43
https://github.com/d3/d3-axis#axis_ticks
To finish up, let’s draw the y axis label in the middle of the y axis, just inside the
left side of the chart wrapper. d3 selection objects also have a .text() method that
operates similarly to .html(). Let’s try using that here.
Making a Scatterplot 63
code/02-making-a-scatterplot/completed/draw-scatter.js
97 const yAxisLabel = yAxis.append("text")
98 .attr("x", -dimensions.boundedHeight / 2)
99 .attr("y", -dimensions.margin.left + 10)
100 .attr("fill", "black")
101 .style("font-size", "1.4em")
102 .text("Relative humidity")
We’ll need to rotate this label to fit next to the y axis. To rotate it around its center,
we can set its CSS property text-anchor to middle.
.style("transform", "rotate(-90deg)")
.style("text-anchor", "middle")
Finished scatterplot
Initialize interactions
The next step in our chart-drawing checklist is setting up interactions and event
listeners. We’ll go over this in detail in Chapter 5.
Making a Scatterplot 64
Finished scatterplot
Generally, it seems like we were correct in guessing that a high humidity would
likely coincide with a high dew point. We’ll discuss the different types of patterns
we might see in a scatter plot in Chapter 10.
cloud cover for that day. Let’s show how the amount of cloud cover varies based
on humidity and dew point by adding a color scale.
Our dataset
Looking at a value in our dataset, we can see that the amount of cloud cover exists
at the key cloudCover. Let’s add another data accessor function near the top of our
file:
Next up, let’s create another scale at the bottom of our Create scales step.
So far, we’ve only looked at linear scales that convert numbers to other numbers.
Scales can also convert a number into a color — we just need to replace the domain
with a range of colors.
Making a Scatterplot 66
Let’s make low cloud cover days be light blue and very cloudy days dark blue - that’s
a good semantic mapping.
Let’s test it out - if we log colorScale(0.1) to the console, we should see a color
value, such as rgb(126, 193, 219). Perfect!
Choosing colors is a complicated topic! We’ll learn about color spaces, good color
scales, and picking colors in Chapter 7.
All that’s left to do is to update how we set the fill of each dot. Let’s find where we’re
doing that now.
.attr("fill", "cornflowerblue")
Instead of making every dot blue, let’s use our colorAccessor() to grab the
precipitation value, then pass that into our colorScale().
When we refresh our webpage, we should see our finished scatter plot with dots of
various blues.
Making a Scatterplot 67
For a complete, accessible chart, it would be a good idea to add a legend to explain
what our colors mean. Stay tuned! We’ll learn how to add a color scale legend in
Chapter 6.
This chapter was jam-packed with new concepts — we learned about data joins,
<text> SVG elements, color scales, and more. Give yourself a pat on the back for
making it through! Next up, we’ll create a bar chart and learn some new concepts.
Making a Bar Chart
We’ll walk through one last “basic” chart — once finished, you’ll feel very comfort-
able with each step and we’ll move on to even more exciting concepts like animations
and interactions.
Looking at the scatter plot we just made, we can see the daily humidity values from
the dots’ vertical placement.
Making a Bar Chart 69
But it’s hard to answer our questions - do most of our dots fall close the middle of
the chart? We’re not entirely sure.
Instead, let’s make a histogram.
Histogram
A histogram is a bar chart that shows the distribution of one metric, with the metric
values on the x axis and the frequency of values on the y axis.
Making a Bar Chart 70
Histogram graphic
In order to show the frequency, values are placed in equally-sized bins (visualized
as individual bars). For example, we could make bins for dew point temperatures that
span 10 degrees - these would look something like [0-10, 10-20, 20-30, ...].
A dew point of 15 degrees would be counted in the second bin: 10-20.
The number of and size of bins is up to the implementor - you could have a histogram
with only 3 bins or one with 100 bins! There are standards that can be followed (feel
free to check out d3’s built-in formulas²²), but we can generally decide the number
based on what suits the data and what’s easy to read.
Our goal is to make a histogram of humidity values. This will show us the distribution
of humidity values and help answer our questions. Do most days stay around the
same level of humidity? Or are there two types of days: humid and dry? Are there
crazy humid days?
²²https://github.com/d3/d3-array#bin-thresholds
Making a Bar Chart 71
For extra credit, we’ll generalize our histogram function and loop through eight
metrics in our dataset - creating many histograms to compare!
Making a Bar Chart 72
Many histograms
Chart checklist
To start, let’s look over our chart-making checklist to remind ourselves of the
necessary steps.
1. Access data
2. Create dimensions
3. Draw canvas
4. Create scales
5. Draw data
6. Draw peripherals
7. Set up interactions
We’ll breeze through most of these steps, reinforcing what we’ve already learned.
Making a Bar Chart 73
Access data
As usual, make sure your node server is running (live-server) and point your
browser at http://localhost:8080/03-making-a-bar-chart/draft/. Inside the
/code/03-making-a-bar-chart/draft/draw-bars.js
file, let’s grab the data from our JSON file, waiting until it’s loaded to continue.
code/03-making-a-bar-chart/completed/draw-bars.js
As usual, the completed chart code is available if you need a hint, this time at
/code/03-making-a-bar-chart/completed/draw-bars.js.
This time, we’re only interested in one metric for the whole chart. Remember, the
y axis is plotting the frequency (i.e. the number of occurrences) of the metric whose
values are on the x axis. So instead of an xAccessor() and yAccessor(), we define
a single metricAccessor().
Create dimensions
Histograms are easiest to read when they are wider than they are tall. Let’s set the
width before defining the rest of our dimensions so we can use it to calculate the
height. We’ll also be able to quickly change the width later and keep the same aspect
ratio for our chart.
Chart design tip: Histograms are easiest to read when they are wider than
they are tall.
Making a Bar Chart 74
Instead of filling the whole window, let’s prepare for multiple histograms and keep
our chart small. That way, the charts can stack horizontally and vertically, depending
on the screen size.
code/03-making-a-bar-chart/completed/draw-bars.js
Alright! Let’s use the width to set the width and height of our chart. We’ll leave
a larger margin on the top to account for the bar labels, which we’ll position above
each bar.
code/03-making-a-bar-chart/completed/draw-bars.js
12 let dimensions = {
13 width: width,
14 height: width * 0.6,
15 margin: {
16 top: 30,
17 right: 10,
18 bottom: 50,
19 left: 50,
20 },
21 }
Remember, our wrapper encompasses the whole chart. If we subtract our margins,
we’ll get the size of our bounds which contain any data elements.
Making a Bar Chart 75
Chart terminology
Now that we know the size of our wrapper and margins, we can calculate the size
of our bounds.
code/03-making-a-bar-chart/completed/draw-bars.js
22 dimensions.boundedWidth = dimensions.width
23 - dimensions.margin.left
24 - dimensions.margin.right
25 dimensions.boundedHeight = dimensions.height
26 - dimensions.margin.top
27 - dimensions.margin.bottom
Draw canvas
Let’s create our wrapper element. Try to write this code without looking first. Like
we’ve done before, we want to select the existing element, add a new <svg> element,
and set its width and height.
Making a Bar Chart 76
code/03-making-a-bar-chart/completed/draw-bars.js
How far did you get without looking? Let’s try that again for this next part: creating
our bounds. As a reminder, our bounds are a <g> element that will contain our main
chart bits and be shifted to respect our top and left margins.
code/03-making-a-bar-chart/completed/draw-bars.js
Create scales
Our x scale should look familiar to the ones we’ve made in the past. We need a scale
that will convert humidity levels into pixels-to-the-right. Since both the domain and
the range are continuous numbers, we’ll use our friend d3.scaleLinear().
Let’s also use .nice(), which we learned in Chapter 2, to make sure our axis starts
and ends on round numbers.
Making a Bar Chart 77
code/03-making-a-bar-chart/completed/draw-bars.js
45 const xScale = d3.scaleLinear()
46 .domain(d3.extent(dataset, metricAccessor))
47 .range([0, dimensions.boundedWidth])
48 .nice()
Creating Bins
How can we split our data into bins, and what size should those bins be? We could
do this manually by looking at the domain and organizing our days into groups, but
that sounds tedious.
Thankfully, we can use d3-array’s d3.histogram() method to create a bin genera-
tor. This generator will convert our dataset into an array of bins - we can even choose
how many bins we want!
Let’s create a new generator:
code/03-making-a-bar-chart/completed/draw-bars.js
50 const binsGenerator = d3.histogram()
Similar to making a scale, we’ll pass a domain to the generator to tell it the range of
numbers we want to cover.
code/03-making-a-bar-chart/completed/draw-bars.js
50 const binsGenerator = d3.histogram()
51 .domain(xScale.domain())
Next, we’ll need to tell our generator how to get the the humidity value, since
our dataset contains objects instead of values. We can do this by passing our
metricAccessor() function to the .value() method.
Making a Bar Chart 78
code/03-making-a-bar-chart/completed/draw-bars.js
We can also tell our generator that we want it to aim for a specific number of bins.
When we create our bins, we won’t necessarily get this exact amount, but it should
be close.
Let’s aim for 13 bins — this should make sure we have enough granularity to see the
shape of our distribution without too much noise. Keep in mind that the number of
bins is the number of thresholds + 1.
code/03-making-a-bar-chart/completed/draw-bars.js
Great! Our bin generator is ready to go. Let’s create our bins by feeding it our data.
code/03-making-a-bar-chart/completed/draw-bars.js
Let’s take a look at these bins by logging them to the console: console.log(bins).
Making a Bar Chart 79
logged bins
• each item is a matching data point. For example, the first bin has no matching
days — this is likely because we used .nice() to round out our x scale.
• there is an x0 key that shows the lower bound of included humidity values
(inclusive)
• there is an x1 key that shows the upper bound of included humidity values
(exclusive). For example, a bin with a x1 value of 1 will include values up to 1,
but not 1 itself
Note how there are 16 bins in my example — our bin generator was aiming for
13 bins but decided that 16 bins were more appropriate. This was a good decision,
creating bins with a sensible size of 0.05. If our bin generator had been more
strict about the number of bins, our bins would have ended up with a size of
0.06666667, which is harder to reason about. To extract insights from a chart,
Making a Bar Chart 80
readers will mentally convert awkward numbers into rounder numbers to make
sense of them. Let’s do that work for them.
Let’s use our new accessor function and our bins to create that y scale. As usual, we’ll
want to make a linear scale. This time, however, we’ll want to start our y axis at
zero.
Previously, we wanted to represent the extent of our data since we were plotting
metrics that had no logical bounds (temperature and humidity level). But the number
of days that fall in a bin is bounded at 0 — you can’t have negative days in a bin!
Instead of using d3.extent(), we can use another method from d3-array: d3.max().
This might sound familiar — we’ve used its counterpart, d3.min() in Chapter 2.
d3.max() takes the same arguments: an array and an accessor function.
Note that we’re passing d3.max() our bins instead of our original dataset — we
want to find the maximum number of days in a bin, which is only available in our
computed bins array.
Making a Bar Chart 81
code/03-making-a-bar-chart/completed/draw-bars.js
Let’s use .nice() here as well to give our bars a round top number.
code/03-making-a-bar-chart/completed/draw-bars.js
Draw data
Here comes the fun part! Our plan is to create one bar for each bin, with a label on
top of each bar.
We’ll need one bar for each item in our bins array — this is a sign that we’ll want
to use the data bind concept we learned in Chapter 2.
Let’s first create a <g> element to contain our bins. This will help keep our code
organized and isolate our bars in the DOM.
code/03-making-a-bar-chart/completed/draw-bars.js
Because we have more than one element, we’ll bind each data point to a <g> SVG
element. This will let us group each bin’s bar and label.
To start, we’ll select all existing <g> elements within our binsGroup (there aren’t
any yet, but we’re creating a selection object that points to the right place). Then
we’ll use .data() to bind our bins to the selection.
Making a Bar Chart 82
code/03-making-a-bar-chart/completed/draw-bars.js
Next, we’ll create our <g> elements, using .enter() to target the new bins (hint: all
of them).
code/03-making-a-bar-chart/completed/draw-bars.js
The above code will create one new <g> element for each bin. We’re going to place
our bars within this group.
Next up we’ll draw our bars, but first we should calculate any constants that we’ll
need. Like a warrior going into battle, we want to prepare our weapons before things
heat up.
In this case, the only constant that we can set ahead of time is the padding between
bars. Giving them some space helps distinguish individual bars, but we don’t want
them too far apart - that will make them hard to compare and take away from the
overall shape of the distribution.
Chart design tip: putting a space between bars helps distinguish individual
bars
code/03-making-a-bar-chart/completed/draw-bars.js
70 const barPadding = 1
Now we are armed warriors and are ready to charge into battle! Each bar is a
rectangle, so we’ll append a <rect> to each of our <g> elements.
Making a Bar Chart 83
code/03-making-a-bar-chart/completed/draw-bars.js
We could create accessor functions for the x0 and x1 properties of each bin if we
were concerned about the structure of our bins changing. In this case, it would be
overkill since:
Next, we’ll specify the <rect>’s y attribute which corresponds to the top of the bar.
We’ll use our yAccessor() to grab the frequency and use our scale to convert it into
pixel space.
Making a Bar Chart 84
code/03-making-a-bar-chart/completed/draw-bars.js
To find the width of a bar, we need to subtract the x0 position of the left side of
the bar from the x1 position of the right side of the bar.
We’ll need to subtract the bar padding from the total width to account for spaces
between bars. Sometimes we’ll get a bar with a width of 0, and subtracting the
barPadding will bring us to -1. To prevent passing our <rect>s a negative width,
we’ll wrap our value with d3.max([0, width]).
code/03-making-a-bar-chart/completed/draw-bars.js
Lastly, we’ll calculate the bar’s height by subtracting the y value from the bottom of
the y axis. Since our y axis starts from 0, we can use our boundedHeight.
code/03-making-a-bar-chart/completed/draw-bars.js
Let’s put that all together and change the bar fill to blue.
Making a Bar Chart 85
code/03-making-a-bar-chart/completed/draw-bars.js
Alright! When we refresh our webpage, we’ll see the beginnings of our histogram!
Our bars
Adding Labels
Let’s add labels to show the count for each of these bars.
Making a Bar Chart 86
We can keep our chart clean by only adding labels to bins with any relevant days
— having 0s in empty spaces is unhelpful visual clutter. We can identify which bins
have no data by their lack of a bar, no need to call it out with a label.
d3 selections have a .filter() method that acts the same way the native Array
method does. .filter() accepts one parameter: a function that accepts one data
point and returns a value. Any items in our dataset who return a falsy value will be
removed.
By “falsy”, we’re referring to any value that evaluates to false. Maybe surpris-
ingly, this includes values other than false, such as 0, null, undefined, "", and
NaN. Keep in mind that empty arrays [] and object {} evaluate to truthy. If you’re
curious, read more here.
https://developer.mozilla.org/en-US/docs/Glossary/Falsy
Since these labels are just text, we’ll want to use the SVG <text> element we’ve been
using for our axis labels.
code/03-making-a-bar-chart/completed/draw-bars.js
Remember, <text> elements are positioned with x and y attributes. The label will
be centered horizontally above the bar — we can find the center of the bar by adding
half of the bar’s width (the right side minus the left side) to the left side of the bar.
Making a Bar Chart 87
code/03-making-a-bar-chart/completed/draw-bars.js
Our <text>’s y position will be similar to the <rect>’s y position, but let’s shift it
up by 5 pixels to add a little gap.
code/03-making-a-bar-chart/completed/draw-bars.js
Next, we’ll display the count of days in the bin using our yAccessor() function.
Note: again, we can use yAccessor() as shorthand for d => yAccessor(d).
code/03-making-a-bar-chart/completed/draw-bars.js
We can use the CSS text-anchor property to horizontally align our text — this is a
much simpler solution than compensating for text width when we set the x attribute.
Making a Bar Chart 88
code/03-making-a-bar-chart/completed/draw-bars.js
After adding a few styles to decrease the visual importance of our labels…
code/03-making-a-bar-chart/completed/draw-bars.js
…we should see the count of days for each of our bars!
Making a Bar Chart 89
Extra credit
When looking at the shape of a distribution, it can be helpful to know where the
mean is.
The mean is just the average — the center of a set of numbers. To calculate the mean,
you would divide the sum by the number of values. For example, the mean of [1,
2, 3, 4, 5] would be (1 + 2 + 3 + 4 + 5) / 5 = 3.
Instead of calculating the mean by hand, we can use d3.mean() to grab that value.
Like many d3 methods we’ve used, we pass the dataset as the first parameter and an
optional accessor function as the second.
code/03-making-a-bar-chart/completed/draw-bars.js
93 const mean = d3.mean(dataset, metricAccessor)
Great! Let’s see how comfortable we are with drawing an unfamiliar SVG element:
<line>. A <line> element will draw a line between two points: [x1, y1] and [x2,
y2]. Using this knowledge, let’s add a line to our bounds that is:
How close can you get before looking at the following code?
code/03-making-a-bar-chart/completed/draw-bars.js
Let’s add some styles to the line so we can see it (by default, <line>s have no stroke
color) and to distinguish it from an axis. SVG element strokes can be split into dashes
with the stroke-dasharray attribute. The lines alternate between the stroke color
and transparent, starting with transparent. We define the line lengths with a space-
separated list of values (which will be repeated until the line is drawn).
Let’s make our lines dashed with a 2px long maroon dash and a 4px long gap.
code/03-making-a-bar-chart/completed/draw-bars.js
Give yourself a pat on the back for drawing your first <line> element!
Making a Bar Chart 91
Let’s label our line to clarify to readers what it represents. We’ll want to add a <text>
element in the same position as our line, but 5 pixels higher to give a little gap.
code/03-making-a-bar-chart/completed/draw-bars.js
Hmm, we can see the text but it isn’t horizontally centered with our line.
Making a Bar Chart 92
Let’s center our text by adding the CSS property text-anchor: middle. This is a
property specifically for setting the horizontal alignment of text in SVG.
code/03-making-a-bar-chart/completed/draw-bars.js
Draw peripherals
As usual, our last task here is to draw our axes. But we’re in for a treat! Since we’re
labeling the y value of each of our bars, we won’t need a y axis. We just need an x
axis and we’re set!
We’ll start by making our axis generator — our axis will be along the bottom of the
chart so we’ll be using d3.axisBottom().
code/03-making-a-bar-chart/completed/draw-bars.js
Then we’ll use our new axis generator to create an axis, then shift it below our
bounds.
Making a Bar Chart 94
code/03-making-a-bar-chart/completed/draw-bars.js
And lastly, let’s throw a label on there to make it clear what the tick labels represent.
Set up interactions
Next, we would set up any chart interactions. We don’t have any interactions for this
chart, but stay tuned — we’ll cover this in the next chapter.
Extra credit
Let’s generalize our histogram drawing function and create a chart for each weather
metric we have access to! This will make sure that we understand what every line of
code is doing.
Generalizing our code will also help us to start thinking about handling dynamic
data — a core concept when building a dashboard. Drawing a graph with a specific
dataset can be difficult, but you get to rely on values being the same every time your
code runs. When handling data from an API, your charting functions need to be
more robust and able to handle very different datasets.
Making a Bar Chart 96
Finished histogram
Here’s the good news: we won’t need to rewrite the majority of our code! The main
difference is that we’ll wrap most of the chart drawing into a new function called
drawHistogram().
Which steps do we need to repeat for every chart? Let’s look at our checklist again.
1. Access data
2. Create dimensions
3. Draw canvas
4. Create scales
5. Draw data
6. Draw peripherals
7. Set up interactions
All of the histograms will use the same dataset, so we can skip step 1. And every
chart will be the same size, so we don’t need to repeat step 2 either. However, we
want each chart to have its own svg element, so we’ll need to wrap everything after
step 2 .
Making a Bar Chart 97
In the next section, we’ll cover ways to make our chart more accessible. We’ll be
working on the current version of our histogram - make a copy of your current
finished histogram in order to come back to it later.
Let’s do that — we’ll create a new function called drawHistogram() that contains all
of our code, starting at the point we create our svg. Note that the finished code for this
step is in the /code/03-making-a-bar-chart/completed-multiple/draw-bars.js
file if you’re unsure about any of these steps.
What parameters does our function need? The only difference between these charts
is the metric we’re plotting, so let’s add that as an argument.
But wait, we need to use the metric to update our metricAccessor(). Let’s grab
our accessor functions from our Access data step and throw them at the top of our
new function. We’ll also need our metricAccessor() to return the provided metric,
instead of hard-coding d.humidity.
Great, let’s give it a go! At the bottom of our drawBars() function, let’s run through
some of the available metrics (see code example for a list) and pass each of them to
our new generalized function.
Making a Bar Chart 98
const metrics = [
"windSpeed",
"moonPhase",
"dewPoint",
"humidity",
"uvIndex",
"windBearing",
"temperatureMin",
"temperatureMax",
]
metrics.forEach(drawHistogram)
We see multiple histograms, but something is off. Not all of these charts are showing
Humidity! Let’s find the line where we set our x axis label and update that to show
our metric instead. Here it is:
Making a Bar Chart 99
We’ll set the text to our metric instead, and we can also add a CSS text-transform
value to help format our metric names. For a production dashboard, we might want
to look up a proper label in a metric-to-label map, but this will work in a pinch.
Finished histogram
Wonderful!
Take a second and observe the variety of shapes of these histograms. What are some
insights we can discover when looking at our data in this format?
Making a Bar Chart 100
• the moon phase distribution is flat - this makes sense because it’s cyclical,
consistently going through the same steps all year.
• our wind speed is usually around 3 mph, with a long tail to the right that
represents a few very windy days. Some days have no wind at all, with an
average wind speed of 0.
• our max temperatures seem almost bimodal, with the mean falling in between
two humps. Looks like New York City spends more days with relatively extreme
temperatures (30°F - 50°F or 70°F - 90°F) than with more temperate weather
(60°F).
Accessibility
The main goal of any data visualization is for it to be readable. This generally means
that we want our elements to be easy to see, text is large enough to read, colors have
enough contrast, etc. But what about users who access web pages through screen
readers?
We can actually make our charts accessibile at a basic level, without putting a lot of
effort in. Let’s update our histogram so that it’s accessible with a screen reader.
If you want to test this out, download the ChromeVox²³ extension for chrome (or use
any other screen reader application). If we test it out on our histogram, you’ll notice
that it doesn’t give much information, other than reading all of the text in our chart.
That’s not an ideal experience.
The main standard for making websites accessible is from WAI-ARIA, the Accessible
Rich Internet Applications Suite. WAI-ARIA roles, set using a role attribute, tell the
screen reader what type of content an element is.
We’ll be updating our completed single histogram in this section. If you completed
the previous Extra credit section, either find your backup of your code or use the
completed code in the /03-making-a-bar-chart/completed/ folder.
²³https://chrome.google.com/webstore/detail/chromevox/kgejglhpjiefppelpmljglcjbhoiplfn?hl=en
Making a Bar Chart 101
The first thing we can do is to give our <svg> element a role of figure²⁴, to alert
it that this element is a chart. (This code can go at the bottom of the Draw canvas
step).
wrapper.attr("role", "figure")
Next, we can make our chart tabbable, by adding a tabindex of 0. This will make
it so that a user can hit tab to highlight our chart.
There are only two tabindex values that you should use:
wrapper.attr("role", "figure")
.attr("tabindex", "0")
When a user tabs to our chart, we want the screen reader to announce the basic layout
so the user knows what they’re “looking” at. To do this, we can add a <title> SVG
component with a short description.
wrapper.append("title")
.text("Histogram looking at the distribution of humidity in 2016")
If you have a screen reader set up, you’ll notice that it will read our <title> when
we tab to our chart. The “highlighted” state will look something like this:
²⁴https://developer.mozilla.org/en-US/docs/Web/Accessibility/ARIA/Roles/Figure_Role
Making a Bar Chart 102
Accessibility highlight
Next, we’ll want to make our binsGroup selectable by also giving it a tabindex of
0. If the user presses tab after the wrapper is focused, the browser will focus on the
binsGroup because it’s the next element (in DOM order) that is focusable.
We can also give our binsGroup a role of "list", which will make the screen
reader announce the number of items within the list. And we’ll let the user know
what the list contains by adding an aria-label.
Now when our binsGroup is highlighted, the screen reader will announce: “his-
togram bars. List with 15 items”. Perfect!
Let’s annotate each of our “list items”. After we create our binGroups, we’ll add a
few attributes to each group:
Making a Bar Chart 103
Now when we tab out of our binsGroup, it will focus the first bar group (and
subsequent ones when we tab) and announce our aria label.
We’ll tackle one last issue — you might have noticed that the screen reader reads each
of our x-axis tick labels once it’s done reading our <title>. This is pretty annoying,
and not giving the user much information. Let’s prevent that.
At the bottom of our drawBars() function, let’s select all of the text within our chart
and give it an aria-hidden attribute of "true".
wrapper.selectAll("text")
.attr("role", "presentation")
.attr("aria-hidden", "true")
Great! Now our screen reader will read only our labels and ignore any <text>
elements within our chart.
With just a little effort, we’ve made our chart accessible to any users who access the
web through a screen reader. That’s wonderful, and more than most online charts
can say!
Making a Bar Chart 104
SVG <animate>
<animate> is a native SVG element that can be defined inside of the animated
element.
Animations and Transitions 106
SVG animate
Unfortunately this won’t work for our charts. For one, <animate> is unsupported in
Internet Explorer. But the bigger issue is that we would have to set a static start and
end value. We don’t want to define static animations, instead we want our elements
to animate changes between two dynamic values. Luckily, we have other options.
Animations and Transitions 107
CSS transitions
Many of our chart changes can be transitioned with the CSS transition property.
When we update a <rect> to have a fill of red instead of blue, we can specify that
the color change take 10 seconds instead of being instantaneous. During those 10
seconds, the <rect> will continuously re-draw with intermediate colors on the way
to red.
Not all properties can be animated. For example, how would you animate changing
a label from Humidity to Dew point? However most properties can be animated,
so feel free to operate under the assumption that a property can be animated until
proven otherwise.
box transition
.box {
background: cornflowerblue;
height: 100px;
width: 100px;
}
And our styles that apply to our box when it is hovered (change the background color
and move it 30 pixels to the right).
.box:hover {
background: yellowgreen;
transform: translateX(30px);
}
To create CSS a transition, we need to specify how long we want the animation to
take with the transition-duration property. The property value accepts time CSS
data types — a number followed by either s (seconds) or ms (milliseconds).
Let’s make our box changes animate over one second.
.box {
background: cornflowerblue;
height: 100px;
width: 100px;
transition-duration: 1s;
}
When we refresh our webpage and hover over the box, we can see it slowly move to
the right and turn green. Smooth!
Now let’s say that we only want to animate our box’s movement, but we want
the color change to happen instantaneously. This is possible by specifying the
transition-property CSS property. By default, transition-property is set to
all, which animates all transitions. Instead, let’s override the default and specify a
specific CSS property name (transform).
.box {
background: cornflowerblue;
height: 100px;
width: 100px;
transition-duration: 1s;
transition-property: transform;
}
Now our box instantly turns green, but still animates to the right.
.box {
background: cornflowerblue;
height: 100px;
width: 100px;
transition: transform 1s;
}
.box {
background: cornflowerblue;
height: 100px;
width: 100px;
transition: transform 1s steps(6);
}
What if we wanted to animate the color change, but finish turning green while our
box is shifting to the right? transition will accept multiple transition statements,
we just need to separate them by a comma. Let’s add a transition for the background
color.
.box {
background: cornflowerblue;
height: 100px;
width: 100px;
transition: transform 1s steps(6),
background 2s ease-out;
}
Nice! Now our box transitions by stepping to the right, while turning green over two
seconds. Chrome’s dev tools have a great way to visualize this transition. Press esc
when looking at the Elements panel to bring up the bottom panel. In the bottom
panel, we can open up the Animations tab.
Animations and Transitions 111
If you don’t see the Animations tab, click on the kebab menu on the left and select
it from the dropdown options.
animations panel
Once we’ve triggered our box transition by hovering, we can inspect the animation.
Animations and Transitions 112
We can see the transform transition on top, with six discrete changes, and the
background animation on the bottom, easing gradually from one color to the next.
The background transition diagram is twice as wide as the transform transition
diagram, indicating that it takes twice as long.
This view can be very handy when inspecting, tweaking, and debugging transitions.
Now that we’re comfortable with CSS transition, let’s see how we might use it to
animate our charts.
bars
When we click the button, our chart re-draws with the next metric, but the change
is instantaneous.
instant change
Does this next metric have fewer bars than the previous one? Does our mean line
move to the left or the right? These questions can be answered more easily if we
transition gradually from one view to the other.
Let’s add an animation whenever our bars update (.bin rect).
Animations and Transitions 114
.bin rect {
fill: cornflowerblue;
transition: all 1s ease-out;
}
Note that a CSS transition won’t work here in Firefox. This is because Firefox uses
SVG 1.1, which sets height as an attribute. Chrome has implemented the part of
the SVG 2 spec which allows height as a CSS property, letting us animate the
transition.
https://www.w3.org/TR/SVG/styling.html#StylingUsingCSS
Now when we update our metric, our bars shift slowly to one side while changing
height — we can see that their width, height, x, and y values are animating. This
may be fun to watch, but it doesn’t really represent our mental model of bar charts. It
would make more sense for the bars to change position instantaneously and animate
any height differences. Let’s only transition the height and y values.
.bin rect {
fill: cornflowerblue;
transition: height 1s ease-out,
y 1s ease-out;
}
ease-out is a good starting point for CSS transitions — it starts quickly and slows
down near the end of the animation to ease into the final value. It won’t be ideal
in every case, but it’s generally a good choice.
That’s better! Now we can see whether each bar is increasing or decreasing.
Animations and Transitions 115
Our transitions are still looking a bit disjointed with our text changing position
instantaneously. Let’s try to animate our text position, too.
.bin text {
transition: y 1s ease-out;
}
Hmm, our text position is still not animating — it seems as if y isn’t a transition-able
property. Thankfully there is a workaround here — we can position the text using a
CSS property instead of changing its y attribute.
Our advanced package has a handy SVG element cheat sheet with green check
marks to show us what SVG elements’ attributes are animate-able with CSS
transitions. Keep it around for a quick reference when you’re transitioning your
own elements!
Animations and Transitions 116
Switching over to our updating-bars.js file, let’s position our bar labels using
translateY().
Note that we’re filling our <text> elements with empty strings instead of 0 (with
.text(d => yAccessor(d) || "")) to prevent labeling empty bars.
.bin text {
transition: transform 1s ease-out;
}
Now our bar labels are animating with our bars. Perfect!
Let’s make one last change - we want our dashed mean line to animate when it moves
left or right. We could try to transition changes to x1 and x2, but those aren’t CSS
properties, they’re SVG attributes. Let’s position the line’s horizontal position with
the transform property.
We’ll also add the transition CSS property in our styles.css file:
.mean {
stroke: maroon;
stroke-dasharray: 2px 4px;
transition: transform 1s ease-out;
}
d3.transition
CSS transitions have our back for simple property changes, but for more complex
animations we’ll need to use d3.transition() from the d3-transition²⁶ module.
When would we want to use d3.transition() instead of CSS transitions?
²⁶https://github.com/d3/d3-transition
Animations and Transitions 118
Let’s get our hands dirty by re-implementing the CSS transitions for our histogram.
Navigate to the /04-animations-and-transitions/3-draw-bars-with-d3-transition/
folder — you’ll see the same setup with our histogram and a big old Change metric
button.
bars
Let’s again start by animating any changes to our bars. Instead of adding a transition
property to our styles.css file, we’ll start in the updating-bars.js file where we
set our barRects attributes.
As a reminder, when we run:
Animations and Transitions 119
we’re creating a d3 selection object that contains all <rect> elements. Let’s log that
to the console as a refresher of what a selection object looks like.
console.log(barRects)
bars selection
We can use the .transition() method on our d3 selection object to transform our
selection object into a d3 transition object.
Animations and Transitions 120
console.log(barRects)
bars transition
d3 transition objects look a lot like selection objects, with a _groups list of relevant
DOM elements and a _parents list of ancestor elements. They have two additional
keys: _id and _name, but that’s not all that has changed.
Animations and Transitions 121
In this case, we can see that the __proto__ property contains d3-specific meth-
ods, and the nested __proto__ object contains native object methods, such as
toString().
Animations and Transitions 122
We can see that some methods are inherited from d3 selection objects (eg. .call()
and .each()), but most are overwritten by new transition methods. When we click
the Change metric button now, we can see that our bar changes are animated. This
makes sense — any .attr() updates chained after a .transition() call will use
transition’s .attr() method, which attempts to interpolate between old and new
values.
Something looks strange though - our new bars are flying in from the top left corner.
Animations and Transitions 123
Bars flying in
Note that d3 transitions animate over 0.25 seconds — we’ll learn how to change
that in a minute!
Knowing that <rect>s are drawn in the top left corner by default, this makes sense.
But how do we prevent this?
Remember how we can isolate new data points with .enter()? Let’s find the line
where we’re adding new <rect>s and set their initial values. We want them to start
in the right horizontal location, but be 0 pixels tall so we can animate them “growing”
from the x axis.
Let’s also have them be green to start to make it clear which bars we’re targeting.
We’ll need to set the fill using an inline style using .style() instead of setting the
attribute in order to override the CSS styles in styles.css.
newBinGroups.append("rect")
.attr("height", 0)
.attr("x", d => xScale(d.x0) + barPadding)
.attr("y", dimensions.boundedHeight)
.attr("width", d => d3.max([
0,
xScale(d.x1) - xScale(d.x0) - barPadding
]))
.style("fill", "yellowgreen")
Animations and Transitions 124
Why are we using .style() instead of .attr() to set the fill? We need the fill
value to be an inline style instead of an SVG attribute in order to override the
CSS styles in styles.css. The way CSS specificity works means that inline styles
override class selector styles, which override SVG attribute styles.
Once our bars are animated in, they won’t be new anymore. Let’s transition their fill
to blue. Luckily, chaining d3 transitions is really simple — to add a new transition
that starts after the first one ends, add another .transition() call.
Let’s slow things down a bit so we can bask in these fun animations. d3 transitions
default to 0.25 seconds, but we can specify how long an animation takes by chaining
.duration() with a number of milliseconds.
Smooth! Now that our bars are nicely animated, it’s jarring when our text moves to
its new position instantly. Let’s add another transition to make our text transition
with our bars.
We’ll also need to set our labels’ initial position (higher up in our code) to prevent
them from flying in from the left.
Animations and Transitions 126
newBinGroups.append("text")
.attr("x", d => xScale(d.x0)
+ (xScale(d.x1) - xScale(d.x0)) / 2
)
.attr("y", dimensions.boundedHeight)
Here’s a fun tip: we can specify a timing function (similar to CSS’s transition-timing-
function) to give our animations some life. They can look super fancy, but we only
need to chain .ease() with a d3 easing function. Check out the full list at the d3-ease
repo²⁷.
That’s looking groovy, but our animation is out of sync with our labels again. We
could ease our other transition, but there’s an easier (no pun intended) way to sync
multiple transitions.
By calling d3.transition(), we can make a transition on the root document that
can be used in multiple places. Let’s create a root transition — we’ll need to place this
definition above our existing transitions, for example after we define barPadding.
Let’s also log it to the console to take a closer look.
const barPadding = 1
console.log(updateTransition)
If we expand the _groups array, we can see that this transition is indeed targeting
our root <html> element.
²⁷https://github.com/d3/d3-ease
Animations and Transitions 127
You’ll notice errors in the dev tools console that say Error: <rect> attribute
height: A negative value is not valid.. This happens with the
d3.easeBackIn easing we’re using, which causes the bars to bounce below the
x axis when they animate.
Let’s update our bar transition to use updateTransition instead of creating a new
transition. We can do this by passing the existing transition in our .transition()
call.
We can use this transition as many times as we want — let’s also animate our mean
line when it updates.
Remember that we couldn’t animate our x axis with CSS transition? Our transition
objects are built to handle axis updates — we can see our tick marks move to fit the
new domain before the new tick marks are drawn.
But what about animating our bars when they leave? Good question - exit animations
are often difficult to implement because they involve delaying element removal.
Thankfully, d3 transition makes this pretty simple.
Let’s start by creating a transition right before we create updateTransition. Let’s
also take out the easing we added to updateTransition since it’s a little distracting.
Animations and Transitions 129
We can target only the bars that are exiting using our .exit() method. Let’s turn
them red before they animate to make it clear which bars are leaving. Then we can
use our exitTransition and animate the y and height values so the bars shrink
into the x axis.
Don’t look at the browser just yet, we’ll need to finish our exit transition first.
oldBinGroups.selectAll("text")
.transition(exitTransition)
.attr("y", dimensions.boundedHeight)
Last, we need to actually remove our bars from the DOM. We’ll use a new transition
here — not because we can animate removing the elements, but to delay their removal
until the transition is over.
oldBinGroups
.transition(exitTransition)
.remove()
Now we can look at the browser, and we can see our bars animating in and out!
Animations and Transitions 130
There is one issue, though: our bars are moving to their new positions while the bars
are still exiting and we end up with intermediate states like this one:
Transition - intermediate
To fix this, we’ll delay the update transition until the exit transition is finished.
Instead of creating a our updateTransition as a root transition, we can chain it
on our existing exitTransition.
Animations and Transitions 131
We’re chaining transitions here to run them one after the other — d3 transitions also
have a .delay() method if you need to delay a transition for a certain amount of
time. Check out the docs for more information.
https://github.com/d3/d3-transition#transition_delay
Wonderful! Now that we’ve gone through the three different ways we can animate
changes, let’s recap when each method is appropriate.
SVG <animate> is only appropriate for static animations.
CSS transition is useful for animating CSS properties. A good rule of thumb is to
use these mainly for stylistic polish — that way we can keep simpler transitions in
our stylesheets, with the main goal of making our visualizations feel smoother.
d3.transition() is what we want to use for more complex animations: whenever
we need to chain or synchronize with another transition or with DOM changes.
Lines
After animating bars, animating line transitions should be easy, right? Let’s find out!
Let’s navigate to the /04-animations-and-transitions/4-draw-line/ folder
and open the updating-line.js file.
Look familiar? This is our timeline drawing code from Chapter 1 with some small
updates.
You might need to update your “freezing” temperatures, if you live in a warm place
and are getting errors in the console.
Animations and Transitions 132
Timeline
One of the main changes is an addNewDay() function at the bottom of the script. This
exact code isn’t important — what is good to know is that addNewDay() shifts our
dataset one day in the future. To simulate a live timeline, addNewDay() runs every
1.5 seconds.
If you read through the addNewDay() code and were confused by the ...dataset.slice(1),
syntax, the ... is using ES6 spread syntax to expand the dataset (minus the first
point) in place. Read more about it in the MDN docs.
https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Operators/Spread_syntax
We can see our timeline updating when we load our webpage, but it looks jerky. We
know how to smooth the axis transitions, let’s make them nice and slow.
What’s going on here? Why is our line wriggling around instead of adding a new
point at the end?
Timeline wriggling?
Remember when we talked about how path d attributes are a string of draw-to values,
like a learn-coding turtle? d3 is transitioning each point to the next point at the same
index. Our transition’s .attr() function has no idea that we’ve just shifted our
points down one index. It’s guessing how to transition to the new d value, animating
each point to the next day’s y value.
Pretend you’re the .attr() function - how would you transition between these two
d values?
It would make the most sense to transition each point individually, interpolating
from 0 50 to 0 60 instead of moving each point to the left.
Great, we understand why our line is wriggling, but how do we shift it to the left
instead?
Animations and Transitions 134
Let’s start by figuring out how far we need to shift our line to the left. Before we
update our line, let’s grab the last two points in our dataset and find the difference
between their x values.
Now when we update our line, we can instantly shift it to the right to match the old
line.
This shift should be invisible because at the same time we’re shifting our x scale
to the left by the same amount.
Animations and Transitions 135
Then we can animate un-shifting the line to the left, to its normal position on the x
axis.
Timeline updating
Okay great! We can see the line updating before it animates to the left, but we don’t
want to see the new point until it’s within our bounds. The easiest way to hide
out-of-bounds data is to add a <clipPath>.
Animations and Transitions 136
Before we test it out, we need to learn one important SVG convention: using <defs>.
The SVG <defs> element is used to store any re-usable definitions that are used
later in the <svg>. By placing any <clipPath>s or gradients in our <defs> element,
we’ll make our code more accessible. We’ll also know where to look when we’re
debugging, similar to defining constants in one place before we use them.
Now that we know this convention, let’s create our <defs> element and add our
<clipPath> inside. We’ll want to put this definition right after we define our
bounds. Let’s also give it an id that we can reference later.
bounds.append("defs")
.append("clipPath")
.attr("id", "bounds-clip-path")
If we inspect our <clipPath> in the Elements panel, we can see that it’s not
rendering at all.
Animations and Transitions 137
clipPath
Remember, the <clipPath> element’s shape depends on its children, and it has no
children yet. Let’s add a <rect> that covers our bounds.
bounds.append("defs")
.append("clipPath")
.attr("id", "bounds-clip-path")
.append("rect")
.attr("width", dimensions.boundedWidth)
.attr("height", dimensions.boundedHeight)
To use our <clipPath> we’ll create a group with the attribute clip-path pointing
to our <clipPath>’s id. The order in which we draw SVG elements determines their
“z-index”. Keeping that in mind, let’s add our new group after we draw the freezing
<rect>.
Animations and Transitions 138
bounds.append("rect")
.attr("class", "freezing")
const clip = bounds.append("g")
.attr("clip-path", "url(#bounds-clip-path)")
Now we can update our path to sit inside of our new group, instead of the bounds.
clip.append("path")
.attr("class", "line")
Voila! When we reload our webpage, we can see that our line’s new point isn’t fully
visible until it has finished un-shifting.
Timeline transition
We can see that the first point of our dataset is being removed before our line un-
shifts. I bet you could think of a few ways around this — feel free to implement one
or two! We could save the old dataset and preserve that extra point until our line
is unshifted, or we could slice off the first data point when we define our x scale.
In a production graph, the solution would depend on how our data is updating and
what’s appropriate to show.
Now that we have the tools needed to make our chart transitions lively, we’ll learn
how to let our users interact with our charts!
Interactions
The biggest advantage of creating charts with JavaScript is the ability to respond to
user input. In this chapter, we’ll learn what ways users can interact with our graphs
and how to implement them.
d3 events
Browsers have native event listeners — using addEventListener(), we can listen
for events from a user’s:
• mouse
• keyboard
• scroll wheel
• touch
• resize
• … and more.
For example:
function onClick(event) {
// do something here...
}
addEventListener("click", onClick)
After running this code, the browser will trigger onClick() when a user clicks
anywhere on the page.
These event listeners have tons of functionality and are simple to use. We can get
even more functionality using d3’s event listener wrappers!
Interactions 140
Our d3 selection objects have an .on() method that will create event listeners on our
selected DOM elements. Let’s take a look at how to implement d3 event listeners.
First, we’ll need to get our server up and running live-server and navigate to
/code/05-interactions/1-events/draft/.
grey boxes
They don’t do much right now, let’s make it so they change to their designated color
on hover.
To add a d3 event listener, we pass the type of event we want to listen for as the first
parameter of .on(). Any DOM event type will work — see the full list of event types
on the MDN docs²⁸. To mimic a hover start, we’ll want to target mouseenter.
²⁸https://developer.mozilla.org/en-US/docs/Web/Events#Standard_events
Interactions 141
rect.on("mouseenter")
The second parameter .on() receives is a callback function that will be executed
when the specified event happens. This function will receive three parameters:
It can often be helpful to use ES6 object property shorthand for logging
multiple variables. This way, we can see the name and value of each
variable!
When we hover over a box, we can see that our mouseenter event is triggered! The
parameters passed to our function(in order) are:
1. the matching data point bound from the rectColors array (in this case, the
color)
2. the rect’s index
3. the nodes in the current selection (in this case, the list of <rect> s)
function parameters
In order to change the color of the current box, we’ll need to create a d3 selection
targeting only that box. We could find it in the list of nodes using its index, but there’s
an easier way. Let’s take a look at what this looks like in our function.
Interactions 142
Perfect! It looks like the this keyword points at the DOM element that triggered the
event.
function this
We can use this to create a d3 selection and set the box’s fill using the datum.
Now when we refresh our webpage, we can change our boxes to their related color
on hover!
Interactions 143
hovered boxes
Hmm, we’re missing something. We want our boxes to turn back to grey when our
mouse leaves them. Let’s chain another event listener that triggers on mouseout and
make our box grey again.
rects.on("mouseenter", () => {
console.log(this)
})
Oh right, this in arrow functions refer to the lexical scope, meaning that this will
always refer to the same thing inside and outside of a function.
setTimeout(() => {
}, 3000)
Removing a d3 event listener is easy — all we need to do is call .on() with null as
the triggered function.
Interactions 145
setTimeout(() => {
rects
.on("mouseenter", null)
.on("mouseout", null)
}, 3000)
Perfect! Now our hover events will stop working after 3 seconds. You might have
noticed that a box might be stuck with its hovered color if it was hovered over when
the mouse events were deleted.
setTimeout(() => {
rects
.dispatch("mouseout")
.on("mouseenter", null)
.on("mouseout", null)
}, 3000)
Perfect! Now that we have a good handle on using d3 event listeners, let’s use them
to make our charts interactive.
Interactions 146
Bar chart
Let’s add interactions to our histogram — navigate to
/code/05-interactions/2-bars/draft/ in the browser and open the
Histogram
Our goal in the section is to add an informative tooltip that shows the humidity range
and day count when a user hovers over a bar.
Interactions 147
histogram finished
We could use d3 event listeners to change the bar’s color on hover, but there’s an
alternative: CSS hover states. To add CSS properties that only apply when an element
is hovered over, add :hover after the selector name. It’s good practice to place this
selector immediately after the non-hover styles to keep all bar styles in one place.
Let’s add a new selector to the /code/05-interactions/2-bars/draft/styles.css
file.
.bin rect:hover {
}
Let’s have our bars change their fill to purple when we hover over them.
.bin rect:hover {
fill: purple;
}
Great, now our bars should turn purple when we hover over them and back to blue
when we move our mouse out.
Interactions 148
Now we know how to implement hover states in two ways: CSS hover states and
event listeners. Why would we use one over the other?
CSS hover states are good to use for more stylistic updates that don’t require DOM
changes. For example, changing colors or opacity. If we’re using a CSS preprocessor
like SASS, we can use any color variables instead of duplicating them in our
JavaScript file.
JavaScript event listeners are what we need to turn to when we need a more
complicated hover state. For example, if we want to update the text of a tooltip or
move an element, we’ll want to do that in JavaScript.
Since we need to update our tooltip text and position when we hover over a bar, let’s
add our mouseenter and mouseleave event listeners at the bottom of our bars.js
file. We can set ourselves up with named functions to keep our chained code clean
and concise.
Interactions 149
binGroups.select("rect")
.on("mouseenter", onMouseEnter)
.on("mouseleave", onMouseLeave)
function onMouseEnter(datum) {
}
function onMouseLeave(datum) {
}
Starting with our onMouseEnter() function, we’ll start by grabbing our tooltip
element. If you look in our index.html file, you can see that our template starts
with a tooltip with two children: a div to display the range and a div to display the
value. We’ll follow the common convention of using ids as hooks for JavaScript and
classes as hooks for CSS. There are two main reasons for this distinction:
1. We can use classes in multiple places (if we wanted to style multiple elements
at once) but we’ll only use an id in one place. This ensures that we’re selecting
the correct element in our chart code
2. We want to separate our chart manipulation code and our styling code — we
should be able to move our chart hook without affecting the styles.
We could create our tooltip in JavaScript, the same way we have been creating and
manipulating SVG elements with d3. We have it defined in our HTML file here,
which is generally easier to read and maintain since the tooltip layout is static.
If we open up our styles.css, we can see our basic tooltip styles, including using a
pseudo-selector .tooltip:before to add an arrow pointing down (at the hovered
bar). Also note that the tooltip is hidden (opacity: 0) and will transition any
property changes (transition: all 0.2s ease-out). It also will not receive any
mouse events (pointer-events: none) to prevent from stealing the mouse events
we’ll be implementing.
Let’s comment out the opacity: 0 property so we can get a look at our tooltip.
Interactions 150
.tooltip {
/* opacity: 0; */
We can see that our tooltip is positioned in the top left of our page.
If we position it instead at the top left of our chart, we’ll be able to shift it based on
the hovered bar’s position in the chart.
We can see that our tooltip is absolutely positioned all the way to the left and 12px
above the top (to offset the bottom triangle). So why isn’t it positioned at the top left
of our chart?
Absolutely positioned elements are placed relative to their containing block. The
default containing block is the <html> element, but will be overridden by certain
ancestor elements. The main scenario that will create a new containing block is if the
element has a position other than the default (static). There are other scenarios,
but they are much more rare (for example, if a transform is specified).
This means that our tooltip will be positioned at the top left of the nearest ancestor
element that has a set position. Let’s give our .wrapper element a position of
relative.
Interactions 151
.wrapper {
position: relative;
}
Perfect! Now our tooltip is located at the top left of our chart and ready to be shifted
into place when a bar is hovered over.
Let’s start adding our mouse events in bars.js by grabbing the existing tooltip using
its id (#tooltip). Our tooltip won’t change once we load the page, so let’s define it
outside of our onMouseEnter() function.
Now let’s start fleshing out our onMouseEnter() function by updating our tooltip
text to tell us about the hovered bar. Let’s select the nested #count element and
update it to display the y value of the bar. Remember, in our histogram the y value
is the number of days in our dataset that fall in that humidity level range.
Interactions 152
Looking good! Now our tooltip updates when we hover over a bar to show that bar’s
count.
Next, we can update our range value to match the hovered bar. The bar is covering
a range of humidity values, so let’s make an array of the values and join them with
a - (which can be easier to read than a template literal).
tooltip.select("#range")
.text([
datum.x0,
datum.x1
].join(" - "))
Our tooltip now updates to display both the count and the range, but it might be a
bit too precise.
Interactions 153
We could convert our range values to strings and slice them to a certain precision,
but there’s a better way. It’s time to meet d3.format().
The d3-format²⁹ module helps turn numbers into nicely formatted strings. Usually
when we display a number, we’ll want to parse it from its raw format. For example,
we’d rather display 32,000 than 32000 — the former is easier to read and will help
with scanning a list of numbers.
If we pass d3.format() a format specifier string, it will create a formatter function.
That formatter function will take one parameter (a number) and return a formatted
string. There are many possible format specifier strings — let’s go over the format
for the options we’ll use the most often.
[,][.precision][type]
Each of these specifiers is optional — if we use an empty string, our formatter will just
return our number as a string. Let’s talk about what each specifier tells our formatter.
,: add commas every 3 digits to the left of the decimal place
type: each specific type is declared by using a single letter or symbol. The most
handy types are:
²⁹https://github.com/d3/d3-format
Interactions 154
Run through a few examples in your terminal to get the hang of it.
Let’s create a formatter for our humidity levels. Two decimal points should be enough
to differentiate between ranges without overwhelming our user with too many 0s.
Now we can use our formatter to clean up our humidity level numbers.
Nice! An added benefit to our number formatting is that our range numbers are the
same width for every value, preventing our tooltip from jumping around.
Interactions 155
Next, we want to position our tooltip horizontally centered above a bar when we
hover over it. To calculate our tooltip’s x position, we’ll need to take three things
into account:
Remember that our tooltip is located at the top left of our wrapper - the outer
container of our chart. But since our bars are within our bounds, they are shifted
by the margins we specified.
Interactions 156
Chart terminology
Let’s add these numbers together to get the x position of our tooltip.
const x = xScale(datum.x0)
+ (xScale(datum.x1) - xScale(datum.x0)) / 2
+ dimensions.margin.left
When we calculate our tooltip’s y position, we don’t need to take into account the
bar’s dimensions because we want it placed above the bar. That means we’ll only
need to add two numbers:
const y = yScale(yAccessor(datum))
+ dimensions.margin.top
Let’s use our x and y positions to shift our tooltip. Because we’re working with a
normal xHTML div, we’ll use the CSS translate property.
Interactions 157
tooltip.style("transform", `translate(`
+ `${x}px,`
+ `${y}px`
+ `)`)
Why are we setting the transform CSS property and not left and top? A good rule
of thumb is to avoid changing (and especially animating) CSS values other than
transform and opacity. When the browser styles elements on the page, it runs
through several steps:
1. calculate style
2. layout
3. paint, and
4. layers
Most CSS properties affect steps 2 or 3, which means that the browser has to
perform that step and the subsequent steps every time that property is changed.
Transform and opacity only affect step 4, which cuts down on the amount of
work the browser has to do. Read more about each step and this distinction at
https://www.html5rocks.com/en/tutorials/speed/high-performance-animations/.
https://www.html5rocks.com/en/tutorials/speed/high-performance-animations/
Hmm, why is our tooltip in the wrong position? It looks like we’re positioning the
top left of the tooltip in the right location (above the hovered bar).
Interactions 158
We want to position the bottom, center of our tooltip (the tip of the arrow) above the
bar, instead. We could find the tooltip size by calling the .getBoundingClientRect()
method, but there’s a computationally cheaper way.
There are a few ways to shift absolutely positioned elements using CSS properties:
All of these properties can receive percentage values, but some of them are based on
different dimensions.
We’re interested in shifting the tooltip based on its own height and width, so we’ll
need to use transform: translate(). But we’re already applying a translate
value — how can we set the translate value using a pixel amount and a width?
Interactions 159
CSS calc() comes to the rescue here! We can tell CSS to calculate an offset based
on values with different units. For example, the following CSS rule would cause an
element to be 20 pixels wider than its container.
Let’s use calc() to offset our tooltip up half of its own width (-50%) and left -100%
of its own height. This is in addition to our calculated x and y values.
tooltip.style("transform", `translate(`
+ `calc( -50% + ${x}px),`
+ `calc(-100% + ${y}px)`
+ `)`)
histogram finished
We have one last task to do — hide the tooltip when we’re not hovering over a bar.
Let’s un-comment the opacity: 0 rule in styles.css so its hidden to start.
Interactions 160
.tooltip {
opacity: 0;
Jumping back to our bars.js file, we need to make our tooltip visible at the end of our
onMouseEnter() function.
tooltip.style("opacity", 1)
Lastly, we want to make our tooltip invisible again whenever our mouse leaves a bar.
Let’s add that to our onMouseLeave() function.
function onMouseLeave() {
tooltip.style("opacity", 0)
}
Look at that! You just made an interactive chart that gives users more information
when they need it. Positioning tooltips is not a simple feat, so give yourself a pat on
the back! Next up, we’ll learn an even fancier method for making it easy for users to
get tooltips even for small, close-together elements.
Scatter plot
Let’s level up and add tooltips to a scatter plot. Navigate to
/code/05-interactions/3-scatter/draft/ in your browser and open up the
/code/05-interactions/3-scatter/draft/scatter.js file.
You should see the scatter plot we made in Chapter 2 — no interactions here yet.
Interactions 161
Scatter plot
We want a tooltip to give us more information when we hover over a point in our
chart. Let’s go through the steps from last chapter.
At the bottom of the file, we’ll select all of our <circle/> elements and add a
mousenter and a mouseleave event.
bounds.selectAll("circle")
.on("mouseenter", onMouseEnter)
.on("mouseleave", onMouseLeave)
We know that we’ll need to modify our #tooltip element, so let’s assign that to a
variable. Let’s also define our onMouseEnter() and onMouseLeave() functions.
function onMouseLeave() {
}
Let’s first fill out our onMouseEnter() function. We want to display two values:
Interactions 162
For both metrics, we’ll want to define a string formatter using d3.format(). Then
we’ll use that formatter to set the text value of the relevant <span/> in our tooltip.
function onMouseEnter(datum) {
const formatHumidity = d3.format(".2f")
tooltip.select("#humidity")
.text(formatHumidity(yAccessor(datum)))
Let’s add an extra bit of information at the bottom of this function — users will
probably want to know the date of the hovered point. Our data point’s date is
formatted as a string, but not in a very human-readable format (for example, “2019-
01-01”). Let’s use d3.timeParse to turn that string into a date that we can re-format.
Now we need to turn our date object into a friendlier string. The d3-time-format³⁰
module can help us out here! d3.timeFormat() will take a date formatter string and
return a formatter function.
The date formatter string uses the same syntax as d3.timeParse — it follows four
rules:
3. usually the letter in a directive has two formats: lowerbase (abbreviated) and
uppercase (full), and
4. a dash (-) between the percent sign and the letter prevents padding of numbers.
Next, we’ll grab the x and y value of our dot , offset by the top and left margins.
³¹https://github.com/d3/d3-time-format
Interactions 164
const x = xScale(xAccessor(datum))
+ dimensions.margin.left
const y = yScale(yAccessor(datum))
+ dimensions.margin.top
Just like with our bars, we’ll use calc() to add these values to the percentage offsets
needed to shift the tooltip. Remember, this is necessary so that we’re positioning its
arrow, not the top left corner.
tooltip.style("transform", `translate(`
+ `calc( -50% + ${x}px),`
+ `calc(-100% + ${y}px)`
+ `)`)
Lastly, we’ll make our tooltip visible and hide it when we mouse out of our dot.
tooltip.style("opacity", 1)
}
function onMouseLeave() {
tooltip.style("opacity", 0)
}
Nice! Adding a tooltip was much faster the second time around, wasn’t it?
Interactions 165
Those tiny dots are hard to hover over, though. The small hover target makes us
focus really hard to move our mouse exactly over a point. To make things worse,
our tooltip disappears when moving between points, making the whole interaction
a little jerky.
Don’t worry! We have a very clever solution to this problem.
Voronoi
Let’s talk briefly about voronoi diagrams. For every location on our scatter plot,
there is a dot that is the closest. A voronoi diagram partitions a plane into regions
based on the closest point. Any location within each of these parts agrees on the
closest point.
Voronoi are useful in many fields — from creating art to detecting neuromuscular
diseases to developing predictive models for forest fires.
Let’s look at what our scatter plot would look like when split up with a voronoi
diagram.
Interactions 166
See how each point in our scatter plot is inside of a cell? If you chose any location in
that cell, that point would be the closest.
There is a voronoi generator built into the main d3 bundle: d3-voronoi³². However,
this module is deprecated and the speedier d3-delaunay³³ is recommended instead.
³²https://github.com/d3/d3-voronoi
³³https://github.com/d3/d3-delaunay
Interactions 167
Enough talking, let’s create our own diagram! Let’s add some code at the end of the
Draw data step, right before the Draw peripherals step. Because this is an external
library, the API is a bit different from other d3 code. Instead of creating a voronoi
generator, we’ll create a new Delaunay triangulation. A delaunay triangulation³⁴ is
a way to join a set of points to create a triangular mesh. To create this, we can pass
d3.Delaunay.from()³⁵ three parameters:
1. our dataset,
2. an x accessor function, and
3. a y accessor function.
Let’s bind our data and add a <path> for each of our data points with a class of
“voronoi” (for styling with our styles.css file).
³⁴https://en.wikipedia.org/wiki/Delaunay_triangulation
³⁵https://github.com/d3/d3-delaunay#delaunay_from
Interactions 168
bounds.selectAll(".voronoi")
.data(dataset)
.enter().append("path")
.attr("class", "voronoi")
bounds.selectAll(".voronoi")
// ...
.attr("d", (d,i) => voronoi.renderCell(i))
Lastly, let’s give our paths a stroke value of salmon so that we can look at them.
bounds.selectAll(".voronoi")
// ...
.attr("stroke", "salmon")
Now when we refresh our webpage, our scatter plot will be split into voronoi cells!
Interactions 169
voronoi paths
Hmm, our voronoi diagram is wider and shorter than our chart. This is because it
has no concept of the size of our bounds, and is using the default size of 960 pixels
wide and 500 pixels tall, which we can see if we log out our voronoi object.
Interactions 170
voronoi paths
Let’s specify the size of our diagram by setting our voronoi’s .xmax and .ymax
values (before we draw our <path>s).
What we want is to capture hover events for our paths instead of an individual dot.
This will be much easier to interact with because of the contiguous, large hover
targets.
Let’s remove that last line where we set the stroke (.attr("stroke", "salmon"))
so our voronoi cells are invisible. Next, we’ll update our interactions, starting by
moving our mouseenter and mouseleave events from the dots to our voronoi paths.
Interactions 172
Note that the mouse events on our dots won’t be triggered anymore, since they’re
covered by our voronoi paths.
bounds.selectAll(".voronoi")
// ...
.on("mouseenter", onMouseEnter)
.on("mouseleave", onMouseLeave)
When we refresh our webpage, notice how much easier it is to target a specific dot!
function onMouseEnter(datum) {
bounds.selectAll("circle")
.filter(d => d == datum)
.style("fill", "maroon")
However, we’ll run into an issue here. Remember that SVG elements’ z-index is
determined by their position in the DOM. We can’t change our dots’ order easily
on hover, so any dot drawn after our hovered dot will obscure it.
Interactions 173
Instead, we’ll draw a completely new dot which will appear on top.
function onMouseEnter(datum) {
const dayDot = bounds.append("circle")
.attr("class", "tooltipDot")
.attr("cx", xScale(xAccessor(datum)))
.attr("cy", yScale(yAccessor(datum)))
.attr("r", 7)
.style("fill", "maroon")
.style("pointer-events", "none")
function onMouseLeave() {
d3.selectAll(".tooltipDot")
.remove()
Now when we trigger a tooltip, we can see our hovered dot clearly!
Interactions 174
scatter plot
Making a tooltip for our scatter plot was tricker than expected, but we saw how
important encouraging interaction can be. When our hover targets were small, it felt
like work to get more information about a specific point. But now that we’re using
voronoi cells, interacting with our chart is almost fun!
Interactions 175
Line chart
Let’s go through one last example for adding tooltips. So far, we’ve added tooltips to
individual elements (bars, circles, and paths). Adding a tooltip to a timeline is a bit
different. Let’s dig in by navigating to /code/05-interactions/4-line/draft/
in the browser and /code/05-interactions/4-line/draft/line.js in your text
editor.
In this section, we’re aiming to add a tooltip to our line chart like this:
timeline
Instead of catching hover events for individual elements, we want to display a tooltip
whenever a user is hovering anywhere on the chart. Therefore, we’ll want an
element that spans our entire bounds.
To start coding up our Set up interactions step, let’s create a <rect> that covers our
bounds and add our mouse event listeners to it. This time we’ll want to listen for
mousemove events instead of mouseenter events because we’ll want to update the
tooltip’s position when a reader moves their mouse around the chart.
Note that we don’t need to define our <rect>’s x or y attributes because they both
default to 0.
Interactions 176
Perfect! We can see that our listeningRect, defaulted to a black fill, covers our
entire bounds.
.listening-rect {
fill: transparent;
}
Great! Now we can set up our tooltip variable and onMouseMove and onMouseLeave()
functions (back in our line.js file).
Interactions 177
function onMouseLeave() {
}
Let’s start fleshing out onMouseMove — how will we know the location on our line
that we are hovering over? The passed parameters we used previously (datum, index,
and nodes) won’t be helpful here, and this will just point us at the listener rect
element.
When an event listener is invoked, the d3-selection library sets a global d3.event.
d3.event will refer to the currently triggered event and will be reset when the event
listener is done. During the event listener handler, we also get access to a d3.mouse()
function which will return the x, y coordinates of the mouse event, relative to a
specified container.
Let’s see what that would look like in action and pass our listener container to
d3.mouse().
function onMouseMove() {
const mousePosition = d3.mouse(this)
console.log(mousePosition)
Now we can see our mouse position as an [x,y] array when we move our mouse
around the chart.
mouse coordinates
Test it out — what do the numbers look like when you hover over the top left of the
chart? What about the bottom right?
Interactions 178
There was a bug that will be fixed in the next version of Firefox (68) that fails
to properly offset the d3.mouse()’s reported mouse coordinates, disregarding any
CSS transform: translate() properties on ancestor elements within an <svg>
element.
Until most people have updated their Firefox to more recent versions, make sure
that you set the transform HTML attribute instead of setting a CSS transform
property if possible (for example, when shifting your bounds), if you intend on
grabbing the mouse position.
https://bugzilla.mozilla.org/show_bug.cgi?id=972041
Great, but in order to show the tooltip next to an actual data point, we need to know
which point we’re closest to. First, we’ll need to figure out what date we’re hovering
over — how do we convert an x position into a date? So far, we’ve only used our
scales to convert from the data space (in this case, JavaScript date objects) to the
pixel space.
Thankfully, d3 scales make this very simple! We can use the same xScale() we’ve
used previously, which has an .invert() method. .invert() will convert our units
backwards, from the range to the domain.
Let’s pass the x position of our mouse (mousePosition[0]) to the .invert()
method of our xScale().
Okay great, now know what date we’re hovering over — let’s figure out how to find
the closest data point.
d3.scan()
If you ever need to know where a variable will fit in a sorted list, d3.scan()³⁶ can
help you out. d3.scan() requires two parameters:
³⁶https://github.com/d3/d3-array#scan
Interactions 179
The comparator function will take two adjacent items in the passed array and return
a numerical value. d3.scan() will take those returned values and return the index
of the smallest value.
Let’s look at a few examples to get that description to click:
d3.scan([100, 0, 10], (a,b) => a - b) would create an array of values that
looks like [100, -10].
d3.scan() example
This expression would then return 1 because the second item in the array of values
is the smallest (remember, the second item is referred to as 1 when we’re looking at
zero-indexed indices).
d3.scan([100, 0, 10], (a,b) => b - a) would create the array [-100, 10]
Interactions 180
d3.scan() example
This expression would then return 0, because the first item of the array of values is
the smallest.
Let’s try it out — we’ll first create a function to find the distance between the hovered
point and a datapoint. We don’t care if the point is before or after the hovered date,
so we’ll use Math.abs() to convert that distance to an absolute distance.
Then we can use that function to compare the two data points in our d3.scan()
comparator function. This will create an array of distances from the hovered point,
and we’ll get the index of the closest data point to our hovered date.
When we move our mouse to the left of our chart, we should see dates close to the
beginning of our dataset, which increase as we move right.
Perfect! Now let’s grab the closest x and y values using our accessor functions —
Interactions 182
We can use our closestXValue to set the date in our tooltip. Let’s also format it
nicely using d3.timeFormat() with the same specifier string we used for our scatter
plot.
Next up, we can set the temperature value in our tooltip — this time our formatter
string will also add a °F suffix to clarify.
Lastly, we’ll want to grab the x and y position of our closest point, shift our tooltip,
and hide/show our tooltip appropriately. This should look like what we’ve done in
the past two sections.
const x = xScale(closestXValue)
+ dimensions.margin.left
const y = yScale(closestYValue)
+ dimensions.margin.top
tooltip.style("transform", `translate(`
+ `calc( -50% + ${x}px),`
+ `calc(-100% + ${y}px)`
+ `)`)
tooltip.style("opacity", 1)
Interactions 183
function onMouseLeave() {
tooltip.style("opacity", 0)
}
Wonderful! When we refresh our webpage, we can see a tooltip that will match the
horizontal position of our cursor, while sitting just above our line.
Extra credit
You may notice an issue that we had before with our scatter plot — it’s not
immediately clear what point we’re hovering over. Let’s solve this by positioning
a <circle> over the spot we’re hovering — this should make the interaction clearer
and the dataset more tangible.
First, we need to create our circle element — let’s draw it right after we create our
tooltip variable. We’ll hide it with an opacity of 0 to start.
Interactions 184
Now, right after we position our tooltip in onMouseEnter(), we can also position
our tooltipCircle and give it an opacity of 1.
tooltipCircle
.attr("cx", xScale(closestXValue))
.attr("cy", yScale(closestYValue))
.style("opacity", 1)
}
function onMouseLeave() {
tooltip.style("opacity", 0)
tooltipCircle.style("opacity", 0)
}
Voila! Now we should see a circle under our tooltip, right over the “hovered” point.
Interactions 185
finished timeline
Give it a spin and feel out the difference. Putting yourself in the user’s shoes, you
can see how highlighting the hovered data point makes the data feel more tangible.
Making a map
Let’s take a step back from our weather data for a chapter and replace it with a dataset
of population growth per country. We could visualize our country data as a bar chart,
which would be great for finding the countries with the highest population growth.
Instead, let’s visualize this data using a choropleth map, which is a map with shaded
areas representing a metric’s values.
Note that a map isn’t always the best way to visualize geographic data — that depends
on the goal of the visualization (and the geographic literacy of your audience). One
of the main benefit of maps is that they mimic our knowledge of the world — name
a country and the first thing that comes to mind is where it’s located in the world.
This knowledge helps to navigate the visualization — there are almost 200 countries,
but we can almost instantaneously find our own.
Maps are also uniquely good at answering several questions, such as:
Finished map
Digging in
The first step to making a map is to decide what it will be composed of. Will it have
streets, mountains, cities, or lakes? We want to visualize data at the country level, so
we need to make a map that consists of every country. Hmm, how do we find the
shape and location of each country?
To create our map, we’ll use the d3-geo³⁷ module which accepts GeoJSON data.
What is GeoJSON?
GeoJSON is a format used to represent geographic structures (a geometry, a fea-
ture, or a collection of features). A GeoJSON object can contain features of the
following types: Point, MultiPoint, LineString, MultiLineString, Polygon,
MultiPolygon, GeometryCollection, Feature, or FeatureCollection. If you’re
curious, check out the spec³⁸ for more information.
³⁷https://github.com/d3/d3-geo
³⁸https://tools.ietf.org/html/rfc7946
Making a map 188
There are many sources of GeoJSON files. We’ll use Natural Earth³⁹, which is a large
collection of public domain map data of various features, locations, and granularities.
We’ll use the Admin 0 - Countries dataset, downloadable here⁴⁰.
Feel free to either follow along and generate the GeoJSON file yourself or read
through this section as an observer. Either way, you’ll get an idea of how to create
your own GeoJSON file for your custom maps. This resource will still be here when
you want to dive deeper.
When we download the countries from Natural Earth, we’ll get a zip file of various
formats.
Thankfully, there are many tools for converting shapefiles into GeoJSON objects.
We’ll use the Geographic Data Abstraction Library (GDAL)⁴¹ - check out this
page⁴² to download it, or if you’re on a Mac, run (brew install gdal).
Once we have GDAL installed, we can convert our ne_50m_admin_0_countries.shp
shapefile into a JSON file containing a GeoJSON object. The following command
will do that and throw the output into a file called world-geojson2.json. (The
³⁹https://www.naturalearthdata.com/
⁴⁰https://www.naturalearthdata.com/downloads/50m-cultural-vectors/50m-admin-0-countries-2/
⁴¹https://www.gdal.org/
⁴²http://trac.osgeo.org/gdal/wiki/DownloadingGdalBinaries
Making a map 189
The -f flag specifies that we want the command to output GeoJSON data.
Great! Now that we have our GeoJSON data, we can import it into our map drawing
file. Let’s start our server (live-server) and open up the
/code/06-making-a-map/draft/draw-map.js file.
Access data
To start, we’ll import our country shapes from our new JSON file.
code/06-making-a-map/completed/draw-map.js
GeoJSON data
Now we can see the structure of our GeoJSON data. It has four keys: crs, features,
name, and type. All of these are metadata describing the object (eg. we’re looking
Making a map 190
Each feature has a geometry object and a properties object. The properties
object contains information about the feature — we can see that each of these features
represents one country, based on the information in here.
Making a map 191
For example, this feature has a NAME of "Zambia". If we dig into the geometry of
this feature, we’ll find an array of [latitude, longitude] coordinates.
Making a map 192
Fun fact: historically, geographers used a nation’s capital as the 0° reference for
longitude. In 1884, the International Meridian Conference decided on the universal
longitude that we use today.
Next, we want to create our accessor functions. We’ll need to access our country ID
to find the metric value in our population growth dataset (which we’ll look at later
in this chapter). We’ll also want our hovered country’s name to display in a tooltip.
We can look in a sample feature to find the relevant keys.
Making a map 193
code/06-making-a-map/completed/draw-map.js
Our dataset
Let’s dig in to the code! We’ll be working in the /code/06-making-a-map/draft/
folder, starting with the draw-map.js file.
Let’s grab our dataset that shows population growth — it’s located in the
/code/06-making-a-map/data_bank_data.csv file.
This data was pulled from The World Bank — feel free to download a more recent
dataset if you’d like. The population growth metric is the percent change of total
population in 2017.
https://databank.worldbank.org/data/source/world-development-indicators#
Note that this is a CSV (comma-separated values) instead of a JSON file. No worries!
This is just a different format of storing information — it ends up being much smaller
than a JSON file because it only mentions each data point’s properties once, similar
to a table’s header. Open the file up and take a peek!
To load data from a CSV file, we can make a simple tweak. Instead of using
d3.json(), we can use d3.csv().
code/06-making-a-map/completed/draw-map.js
Put this line of code on line 4, right after we load countryShapes to keep the file
loading in one place.
Easy peasy! Let’s log the result to the console and take a look (console.log(dataset)).
dataset
This data structure is a bit messy — a great chance to practice with a more real-world
scenario! We can see that our dataset is an array which lists each country multiple
times, each time with a different metric, named under the Series Name key.
We’re only interested in the "Population growth (annual %)" metric. Let’s define
that in a variable name.
code/06-making-a-map/completed/draw-map.js
Later on, once you’re more comfortable, play around with changing this metric
value and visualize the other metrics.
Making a map 195
We’ll want an easy way to look up the population growth number using a country
name. We could find a matching item in our dataset array, but that could be
expensive. Instead, let’s create an object with country ids as keys and the population
growth amount as values:
{
AFG: 2.49078956147291,
ALB: -0.0919722937442495,
// ...
}
14 let metricDataByCountry = {}
Then we’ll go over each item in our dataset array. If the item’s "Series Name"
doesn’t match our metric, we won’t do anything. If it does match, we’ll add a new
value to our metricDataByCountry object: the key is the item’s "Country Code"
and the value is the item’s "2017 [YR2017]" number.
Note that these values are stored as Strings — we’ll convert the value to a number
by prepending +, and default to 0 if the value doesn’t exist.
code/06-making-a-map/completed/draw-map.js
15 dataset.forEach(d => {
16 if (d["Series Name"] != metric) return
17 metricDataByCountry[d["Country Code"]] = +d["2017 [YR2017]"] || 0
18 })
That’s a messy operation, isn’t it? This is part of the reason to deal with the data
at the top of our chart-making code. We’ll polish our dataset into a cleaner format,
separating handling our data idiosyncrasies from actually drawing the chart.
Making a map 196
22 let dimensions = {
23 width: window.innerWidth * 0.9,
24 margin: {
25 top: 10,
26 right: 10,
27 bottom: 10,
28 left: 10,
29 },
30 }
31 dimensions.boundedWidth = dimensions.width
32 - dimensions.margin.left
33 - dimensions.margin.right
How tall does our map need to be? That will depend on our map’s projection.
What is a projection?
Despite what flat-earthers might say, the Earth is a sphere. When representing it on
a 2D screen, we’ll need to create some rules — these rules are the projection.
Imagine peeling an orange and turning the skin into a flat sheet — even with slicing it
in various places, it’s impossible to do perfectly. Projections will use a combination
of distortion (stretching parts of the map) and slicing to approximate the Earth’s
actual shape.
Making a map 197
There are many projections out there. d3-geo⁴³ has 15 projections built in, but many
more in a d3 module that’s not included in the core build: d3-geo-projection⁴⁴.
Play around with the different options at http://localhost:8080/06-making-a-map/-
completed-projections⁴⁵.
projections, galore!
Mercator projection
Mercator has been around for a long time — it was created in 1569, before Antarctica
had even been discovered! Its parallel lines preserve true angles, which made it easy
for sailors to use for navigation, but it sacrifices vertical distortion for horizontal
distortion.
You can see this distortion in the graticule (grid lines on the map) — see how tall the
“squares” get near the top of the map. Also note how the country sizes get skewed as
they approach the poles — Greenland is depicted as the same as Africa!
Another option is the Winkel-Tripel projection, which attempts to balance the three
types of distortion: area, direction, and distance.
Making a map 199
Land near the North and South poles is still enlarged, but the country shapes and
sizes are more accurate.
If we want to show a globe, we can use the d3.geoOrthographic() projection.
Making a map 200
Orthographic projection
Note that the type of projection matters more with maps that cover more area. The
more we’re zoomed in, the less distortion we’ll need to flatten our round Earth
shape. For example, a close-up of a city will look basically the same in two different
projections.
There really isn’t a “right” map projection to use — they all have to distort something,
mapping a 3D shape into 2D. As a general rule of thumb, a variant of the Mercator,
the Transverse Mercator (d3.geoTransverseMercator()) is a good bet for show-
ing maps that cover one country or smaller. The Winkel Tripel (d3.geoWinkel3())
or Equal Earth (d3.geoEqualEarth()) are good bets for maps covering larger areas,
such as the whole world. But this is really the tip of the iceberg - if you’re interested,
read more about how projections work⁴⁶ and specific projections⁴⁷.
If you remember from before, a GeoJSON object can contain features of the
following types: Point, MultiPoint, LineString, MultiLineString, Polygon,
MultiPolygon, GeometryCollection, Feature, or FeatureCollection. But none
of these cover the whole Earth! Worry not, d3-geo⁴⁸ adds support for a type of
"Sphere", which will cover the whole globe.
Each of the projections mentioned in the previous section (and more) are im-
plemented in either d3-geo or d3-geo-projection. We can use these projection
functions to convert from [longitude, latitude] coordinates to [x, y] pixel
coordinates. Essentially, a projection function is our scale in the geographic world.
Let’s create our projection function. We’ll use d3.geoEqualEarth() — feel free to
play around with other options once we’re finished drawing our map.
code/06-making-a-map/completed/draw-map.js
Each projection function has its own default size (think: range). But we want our
projection to be the same width as our bounds.
To update our projection’s width, we can use its .fitWidth() method, which
takes two parameters:
When we call this method, our projection will update its size so that the GeoJSON
object (2) we pass will be the specified width (1).
⁴⁸https://github.com/d3/d3-geo#_path
Making a map 202
code/06-making-a-map/completed/draw-map.js
36 const projection = d3.geoEqualEarth()
37 .fitWidth(dimensions.boundedWidth, sphere)
console.log(pathGenerator(sphere))
pathGenerator result
That looks familiar! Perfect, we got back a <path> d string. But how do we find out
how tall that path would be?
Thankfully, our pathGenerator() has a .bounds() method that will return an
array of [x, y] coordinates describing a bounding box of a specified GeoJSON
object. Let’s test that by logging the bounding box for our sphere:
⁴⁹https://github.com/d3/d3-geo/blob/master/README.md#geoPath
Making a map 203
console.log(pathGenerator.bounds(sphere))
sphere bounds
Great! When drawn on our map, the whole Earth will span from 0px to 831.4px
horizontally and from 0px to 404.7px vertically. We can use ES6 array destructuring⁵⁰
to grab the y1 value.
code/06-making-a-map/completed/draw-map.js
We want the entire Earth to fit within our bounds, so we’ll want to set our
boundedHeight to just cover our sphere.
code/06-making-a-map/completed/draw-map.js
42 dimensions.boundedHeight = y1
43 dimensions.height = dimensions.boundedHeight
44 + dimensions.margin.top
45 + dimensions.margin.bottom
Draw canvas
Now we’re back in familiar territory! Let’s create our wrapper and bounds elements.
See if you can do this step from memory.
First, we want to add a new <svg> element (our wrapper) to the existing <div> with
an id of "wrapper". Then we want to add a <g> element (our bounds) and shift it
to respect our top and left margins.
⁵⁰https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Operators/Destructuring_assignment#
Array_destructuring
Making a map 204
code/06-making-a-map/completed/draw-map.js
Create scales
Next up, we need to create our scales. Typically, we would create x and y scales,
but those are covered by our projection. We’ll only need a scale to help us turn our
metric values (population growth amounts) into color values.
Remember our object of country id and population growth values? We can grab
all of the population growth values by using Object.values(), which returns an
object’s values in an array. More info here⁵¹, for the curious.
code/06-making-a-map/completed/draw-map.js
Then we can extract the smallest and largest values by using d3.extent().
⁵¹https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_objects/Object/values
Making a map 205
code/06-making-a-map/completed/draw-map.js
Let’s take a look at the range we’re dealing with with console.log(metricValueExtent)
Aha! Our extent starts below zero, meaning that some countries had a negative
population growth in 2017. Let’s represent population declines on a red color scale
and population growths on a green color scale.
This kind of color scale is called a diverging color scale, which we’ll cover in more
detail in Chapter 7.
So far, we’ve only create scales that have one minimum and one maximum value for
both their domain and range. But we can create piecewise scales, which are basically
multiple scales in one. If we use a domain and range with more than two values, d3
will create a scale with more than two “anchors”.
In this case, we want to create a scale with a “middle” population growth of 0% that
converts to a middle color of white. Population growth amounts above 0 should
be converted with a color scale from white (0) to green (5) and population growth
amounts below 0 should be converted with a color scale from red (-2) to white (0).
Making a map 206
Color scale
Creating a piecewise scale like this is easier than it might seem: just add a middle
value to both the domain and the range.
code/06-making-a-map/completed/draw-map.js
65 const maxChange = d3.max([-metricValueExtent[0], metricValueExtent[1]\
66 ])
67 const colorScale = d3.scaleLinear()
68 .domain([-maxChange, 0, maxChange])
69 .range(["indigo", "white", "darkgreen"])
1. create a scale which scales evenly on both sides (eg. .domain([-5, 0, 5]))
Making a map 207
This approach will let the viewer see if countries are shrinking at the same pace
as countries that are growing, since a light red (-2.5%) will correlate to a light
green (2.5%).
2. create a scale which scales unevenly from white to red and white to green (eg.
.domain([-2, 0, 5]))
This approach will maximize the color scale, making distinctions between
countries with different growth rates clear. However, it can be confusing when
a light red color (-0.1%) doesn’t match the scale of a light green color (2.5%).
We’ll go with approach number 1 here, to keep things simple, but play around with
the two approaches yourself to get a feel for the differences! How might the legend
(coming up) look different with approach number 2?
Why did we choose indigo and darkgreen as our colors? A more semantic mapping
to “negative” and “positive” might be red and green, but the most common form
of color blindness is red-green color blindness, and we want our readers to be able
to see differences in our color scale. We’ll talk more about this in Chapter 7.
Draw data
Here comes the action! To start, we’ll draw our Earth outline. It makes sense to draw
this first, since SVG elements layer based on their order in the DOM, and we’ll want
all other elements to cover this outline.
Remember that our pathGenerator is aware of the projection we’re using and will
act as a scale, converting a GeoJSON object into a <path> d string.
Making a map 208
code/06-making-a-map/completed/draw-map.js
72 const earth = bounds.append("path")
73 .attr("class", "earth")
74 .attr("d", pathGenerator(sphere))
Earth outline
code/06-making-a-map/completed/draw-map.js
There are many ways to configure this GeoJSON graticule, check out the options
in the d3-geo docs.
https://github.com/d3/d3-geo#geoGraticule
graticule json
Thankfully, our pathGenerator() knows how to handle any GeoJSON type — let’s
use it to draw our graticules.
code/06-making-a-map/completed/draw-map.js
Wonderful!
Making a map 210
Finally, let’s draw our countries. We’ll use the data join method we used in Chapter
2. First, we’ll select all of the elements with a class of .country. Remember: even
though these elements don’t yet exist, this is priming our d3 selection object to bind
data to similar elements.
code/06-making-a-map/completed/draw-map.js
Next, we’ll bind our data to our selection object. Remember what our countryShapes
GeoJSON object looks like?
Making a map 211
Because d3 will create one element per item in the dataset we pass to .data(),
we’ll want to use the list of features instead of the whole object.
code/06-making-a-map/completed/draw-map.js
Next, we’ll use .enter().append("path") to tell d3 to create one new <path> for
each item in our list of countries.
We’ll give each of these <path>s a class of "country" and then pass it straight to
our pathGenerator() to get a d string, generating the shape of each country.
Making a map 212
code/06-making-a-map/completed/draw-map.js
81 const countries = bounds.selectAll(".country")
82 .data(countryShapes.features)
83 .enter().append("path")
84 .attr("class", "country")
85 .attr("d", pathGenerator)
86 .attr("fill", d => {
We have one last task — we want to color our countries based on their population
growth amount. We’ll need to set their fill by looking up the country’s id in the
metricDataByCountry object we created at the top of our file, then passing that id
to our colorScale().
Making a map 213
Since some countries might be missing from our dataset, we’ll want to indicate that
on our map by coloring them white.
code/06-making-a-map/completed/draw-map.js
87 const metricValue = metricDataByCountry[countryIdAccessor(d)]
88 if (typeof metricValue == "undefined") return "#e2e6e9"
89 return colorScale(metricValue)
It’s clear now that the highest growth countries are in Africa and the Middle East, and
that a cluster of European countries shrunk in 2017. Try to find a few other insights
that are easy to see now that we’ve visualized this data with a map.
Draw peripherals
Drawing a legend
Data visualizations should be able to stand on their own, without accompanying text.
Let’s create a legend to explain our color scale.
Making a map 214
To start, we’ll create a <g> element and position it on the left side of our map. To get
a taste of making our map work at different screen sizes, we’ll position it differently
based on the screen size. On a skinnier screen, let’s put it at the bottom of the map,
and on larger screens, we can place it halfway down the map (to the left of the
Americas).
code/06-making-a-map/completed/draw-map.js
To keep things simple, let’s align the top of our <g> element to the top of the gradient
bar. When we add our legend title, we’ll move it up 23 pixels to give the bar some
space.
Making a map 216
code/06-making-a-map/completed/draw-map.js
Note that we’re adding classes that match up with existing CSS styles — feel free to
check out styles.css to investigate. We’re using the CSS property text-anchor:
middle to horizontally center our <text>.
Making a map 217
Next up, we’ll create the bar showing the colors in our scale.
Remember when we created a <defs> SVG element for our <clipPath> in Chapter
4? Elements created within <defs> won’t be visible by themselves, but we can use
them later. Let’s create a <defs> element to store a gradient.
code/06-making-a-map/completed/draw-map.js
Since we’ll be creating a bar here and referencing it in another place, let’s define a
variable to hold the id of our gradient.
Making a map 218
code/06-making-a-map/completed/draw-map.js
Time to create our gradient! To make a gradient in SVG, we’ll need to create
a <linearGradient>⁵² SVG element. Within that, we’ll create several <stop>
SVG elements that will tell the gradient what colors to interpolate between, using
stop-color and offset attributes.
<linearGradient>
<stop stop-color="#12CBC4" offset="0%"></stop>
<stop stop-color="#FFC312" offset="50%"></stop>
<stop stop-color="#B53471" offset="100%"></stop>
</linearGradient>
Gradient example
The offset’s percentage value is in proportion to the element using the <linearGradient>.
We’ll set the id attribute to our legendGradientId so that we can reference our
gradient later.
code/06-making-a-map/completed/draw-map.js
To add our stops, we’ll use colorScale.range() to get an array of the colors in our
color scale. Then we’ll use .enter().append() to create one stop per color. We’ll
use the color’s index to calculate the percent offset (from 0% to 100%).
code/06-making-a-map/completed/draw-map.js
To use a <linearGradient>, all we need to do is set the fill or stroke of a SVG ele-
ment to url(#GRADIENT_ID) (where GRADIENT_ID matches the <linearGradient>’s
id attribute).
Let’s create a <rect> that’s centered horizontally and is filled with our gradient’s
id.
Making a map 220
code/06-making-a-map/completed/draw-map.js
Wonderful! Now we can see our color scale in our legend. Note how it highlights that
most countries are growing and the countries that are shrinking are only shrinking
a small amount.
Lastly, let’s label our color scale with the minimum and maximum percent change
in our scale. We’ll start by labeling the right side, positioning our text 10 pixels to the
right of our bar.
Making a map 221
code/06-making-a-map/completed/draw-map.js
If you’re wondering how our text is vertically centered with the gradient bar,
check out the styles.css file. We’re using the dominant-baseline: middle
CSS property.
And now we’ll label the left side — we can use the text-anchor: end CSS property
to align the right side of our text to the left of our bar.
code/06-making-a-map/completed/draw-map.js
Great! Now readers will be able to tell what our colors mean, and also the largest
decline and increase in percent of population.
Making a map 222
We have a choice here on whether or not we want to use .nice() on our color
scale to have friendlier min and max values. A color scale that spans -5% to 5% is
easier to reason about, but there are two downsides in this case:
1. With a color scale, it’s nice (no pun intended) to easily see the max change,
which are labeled on our legend.
2. Since it’s so hard to distinguish between colors, squishing the scale will make
differences harder to see.
As with many aspects of data visualizations, there is no one right answer. But it’s
important to know the pros and cons and decide based on your goals.
Want more practice creating color scale legends? Adding a legend to the scatter plot
we made in Chapter 2 could help solidify the steps!
Making a map 223
Marking my location
One of the fun parts of building in the browser is that we can interact with the user.
Let’s take advantage of this by plotting the reader’s location on our map.
Modern browsers have a global navigator object, which provides information about
the device accessing the website.
Here’s a good resource if you’re curious about present support or available keys.
https://developer.mozilla.org/en-US/docs/Web/API/Navigator
navigator.geolocation.getCurrentPosition(myPosition => {
console.log(myPosition)
})
The browser will ask to access our location the first time we load the page. The
callback will only run if we click Allow.
myPosition
code/06-making-a-map/completed/draw-map.js
Next, we’ll draw a <circle> with a class of "my-location" in that location. We’ll
also animate the circle’s radius from 0 to 10, to draw the eye.
Making a map 226
code/06-making-a-map/completed/draw-map.js
Nice! It’s good to keep in mind that users relate to our data visualizations through
their personal lens — seeing themselves in a visualization helps give it meaning.
Set up interactions
Lastly, let’s use what we learned in the last chapter to give the user more information
when they hover over a country.
First, we want to initialize our mouse events, using the .on() method for our
countries elements.
code/06-making-a-map/completed/draw-map.js
Next, we want to grab the tooltip element — this is already created in our index.html
file. Having a static tooltip reference outside of our mouse event callbacks keeps
us from having to look it up every time a country is hovered.
code/06-making-a-map/completed/draw-map.js
function onMouseEnter(datum) {
tooltip.style("opacity", 1)
}
function onMouseLeave() {
tooltip.style("opacity", 0)
}
map on hover
Next, let’s populate the text of our tooltip. Remember that the first parameter of
our mouse event (that we’ve named datum) contains the data bound to the hovered
element. We can use this to grab the country name and population growth value. We
can check the index.html file to find the ids of the elements which will contain
this text.
code/06-making-a-map/completed/draw-map.js
166 function onMouseEnter(datum) {
167 tooltip.style("opacity", 1)
168
169 const metricValue = metricDataByCountry[countryIdAccessor(datum)]
170
171 tooltip.select("#country")
172 .text(countryNameAccessor(datum))
173
174 tooltip.select("#value")
175 .text(`${d3.format(",.2f")(metricValue || 0)}%`)
Perfect! Now our tooltip updates to give us information about the currently hovered
country.
Making a map 229
But how do we know where to position the tooltip? Our pathGenerator comes to
the rescue again!
Remember how we used pathGenerator.bounds(sphere) to get the height of the
entire globe? Our pathGenerator also has a .centroid() method that will give us
the center of the passed GeoJSON object. Let’s see what we get when we ask for the
center of our hovered country’s bound data:
console.log(pathGenerator.centroid(datum))
Great! This looks like an [x, y] coordinate. Let’s grab that x and y coordinate and
position a new <circle> element to make sure this is the correct coordinate.
Making a map 230
function onMouseEnter(datum) {
// ...
const [centerX, centerY] = pathGenerator.centroid(datum)
const hoveredCircle = bounds.append("circle")
.attr("cx", centerX)
.attr("cy", centerY)
.attr("r", 3)
}
When we hover a country, we can see that a dot appears exactly in the center.
Let’s take out that hoveredCircle code and instead position our tooltip. Remember
that we want to shift it with our top and left margins, since our tooltip element
lives outside of our bounds. Just like in the last chapter, we’ll need to use calc() to
accomondate the tooltip’s own width and height.
Making a map 231
code/06-making-a-map/completed/draw-map.js
Perfect! Now our map is easy to explore, and users can dig into the exact population
growth numbers.
With this method of positioning the tooltip, it can be very hard to get more
information about small countries. An alternative is to use the voronoi method
we used for our scatter plot in the last chapter. This is implemented in the
/code/06-making-a-map/completed-voronoi/ folder. Feel free to check out the
code and compare the two methods in the browser.
Wrapping up
Way to go! We learned a ton of new concepts in this chapter. Play around with
downloading different GeoJSON files, changing the projection, and using different
data to build on the skills we learned in here. Creating maps is a whole discipline,
and this is just the tip of the iceberg!
Data Visualization Basics
Now that we’re comfortable with how to create a chart, we should zoom out a bit and
talk about what chart to create. So far, we’ve created a line, scatter, and bar chart,
but there are many more basic types to choose from, as well as charts that don’t fall
into a simple category.
Zoooming out
The format of our chart is the largest factor in what information our users take away
— even given the same dataset! Let’s look at the charts we’ve made with our weather
data.
In Chapter 1, we charted a timeline of maximum temperatures. Looking at this
chart, we could see how the temperature changed over time - how consistent was
the weather day to day or season to season?
Data Visualization Basics 234
In Chapter 2, we created a scatter plot with two metrics. Looking at this chart, we
could see at how humidity and dew point are related, answering questions such as:
does a high humidity also meant a high dew point?
Even with these three examples and a limited dataset, we can see how the type of
chart will enable the user to answer very different questions. There are also many
types of charts — thus, answering the question What type of chart is best? is both
important and overwhelmingly open-ended. Don’t worry, though — this chapter will
equip you with the tools to make that decision, and quickly!
Let’s start at the beginning: with our data.
Types of data
Given a dataset, the first task is to determine the structure of the available metrics.
Let’s look at the first item of our weather dataset.
Data Visualization Basics 236
Our dataset
There are many different values here, but two basic types: strings and numbers.
These two types can roughly be split (respectively) into two basic types of data:
qualitative and quantitative.
Qualitative data (our strings) does not have a numerical value, but it can be put into
categories. For example, precipType can either have a value of “rain” or “snow”.
Data Visualization Basics 237
Quantitative data (our numbers) is numerical and can be measured objectively. For
example, temperatureMax has values ranging from 10°F to 100°F.
Qualitative Data
Binary data can be placed into only two categories.
Data Visualization Basics 238
For example, if our weather data had an did rain metric that was either true or false,
that metric would be binary.
Nominal data can be placed multiple categories that don’t have a natural order.
For example, our weather data has the metric icon with values such as clear-day
and wind — these values can’t be ordered.
Ordinal data can be placed in multiple categories with a natural order.
Data Visualization Basics 239
For example, if our weather data instead represented wind speed values with not
windy, somewhat windy, and very windy, that metric would be ordinal.
Quantitative Data
Discrete data has numerical values that can’t be interpolated between, such as a
metric that can only be represented by an integer (whole number).
A classic example is number of kids — a family can have 1 or 2 kids, but not 1.5 kids.
With weather data, a good example would be number of tornados that happened.
Data Visualization Basics 240
Continuous data has numerical values that can be interpolated between. Usually a
metric will fall under this category — for example, max temperature can be 50°F or
51°F or 50.5°F.
This categorization is just one way to group types of data. There are other ways, but
this categorization is common and a handy place to start. When you are looking at
a new dataset, familiarize yourself by categorizing each metric.
Size
dimension - size
Position
dimension - position
Color
dimension - color
We could represent temperature by giving our elements a different color. For exam-
ple, we could make a scatter plot with dots colored from blue (lower temperatures)
to red (higher temperatures). Choosing colors can be overwhelming — don’t worry,
we have a section to help with just that coming up in this Chapter.
Humans are less adept at distinguishing two colors than distinguishing two sizes or
two positions. However, we can more easily average colors together, making color a
good choice for visualizations where the overall picture matters. If both the specific
values and the overall picture are important, try showing the overall picture and
progressively surfacing exact values with tooltips or a companion table.
There are other dimensions (for example, orientation or pattern), but these basic
ones will get you thinking on the right path. These dimensions might remind you of
a concept we’ve been using: scales. Scales have been helping us translate data values
into a physical dimension: for example, temperature into height of a bar in pixels.
Data Visualization Basics 244
When you have a dataset that is chock full of metrics like our weather data, it often
isn’t ideal to visualize all of it in one chart. Focused charts are the most effective —
sticking with 2-4 dimensions will help the user focus on what’s important without
being overwhelmed.
Putting it together
It can be tempting to jump right in and say “Oh I know, I’m going to make a bar
chart!” But let’s investigate further: what is a bar chart made out of? A bar chart is
made up of bars that vary in size (height) and are spread our horizontally (position).
In the next chapter, we’ll talk about common types of charts and how the type of
metrics and ways to visualize them apply to each. But there is a whole world of
chart types that haven’t yet been created! Now that you have this framework, you’ll
be able to look at a dataset and brainstorm unique ways to visualize it.
Chart design
Now that you understand these basics, you have the tools you need to make effective
and intuitive charts. Once you’ve decided on the what and the how, here are some
tips for keeping those charts easy to read.
Simplified chart
Annotate in-place
It can be tempting to throw a legend next to a chart to clarify a color or size scale.
While we should make sure to explain every part of our chart, ask yourself if you
can label these elements directly. In-place annotations put the description as close as
possible, which prevents forcing the reader to look back and forth between elements.
Example redesign
Let’s redesign a simple chart to see these tips in practice. The goal of this chart is to
examine how humidity changes, depending on the time of year.
We’ve decided to visualize this data using a timeline — open it up at http://localhost:8080/07-
data-visualization-basics/humidity-timeline/draft/⁵⁴. We’ll slowly work towards the
finished version at http://localhost:8080/07-data-visualization-basics/humidity-time-
line/completed/⁵⁵. Don’t peek yet!
⁵⁴http://localhost:8080/07-data-visualization-basics/humidity-timeline/draft/
⁵⁵http://localhost:8080/07-data-visualization-basics/humidity-timeline/completed/
Data Visualization Basics 247
Humidity timeline
For practice, take a minute and write down the ways you would improve on this
chart, with our goal in mind.
Goal: examine how humidity changes, depending on the time of year
Ready to dive in? There is no “correct” way to improve on this chart, but we’ll do
our best to make it easier to digest quickly.
The first problem we’ll tackle is the noisiness of the data. While the daily dips
and spikes may help answer other questions, they only distract from our goal of
investigating seasonal trends. Let’s downsample the data to one point per week and
smooth our line using d3.curveBasis().
Let’s add the original points back in, in the form of small circles. We don’t want to
lose the granularity of the original data, even when we’re getting the basic trend with
the downsampled line.
To prevent from drawing attention away from our trend line, we’ll make these dots
a lighter grey.
Data Visualization Basics 249
There’s a lot going on right now! Let’s simplify a bit and take out the grid marks and
chart background. We’ll also specify a smaller number of ticks (.ticks(3)) for our
y axis — the extra ticks aren’t giving any useful information since readers can easily
extrapolate between values.
Revisiting our goal, we realize that we want to focus on trends based on the time
of year. We’re showing the months on our x axis, but we can do some work for the
reader and highlight the different seasons. Let’s block out each season with a <rect>
underneath our main data elements.
Data Visualization Basics 250
While the legend is helpful, there are a few issues with it.
1. Both Spring and Fall are white because they are transitional seasons. This
makes it unclear which is which on the chart.
2. As we discussed earlier in the chapter, legends make the reader look back and
forth between the chart and the legend.
To fix these issues, let’s label the seasons directly on the chart instead of having an
x axis.
Let’s focus on the y axis. Rotated y axis labels are great — they’re often the only way
to fit the label to the side of our chart. However, the label is divorced from the values,
and rotating text makes it harder to read.
Data Visualization Basics 251
Instead, let’s signify the units of our y axis by incorporating it into a phrase with our
top y tick value. Human-readable labels can do some of the digesting work a reader
has to do, hinting at how to interpret a number.
While we’ve made it easier to compare trends across seasons, it’s not easy to conclude
how the seasons compare in general. We could downsample our data and only display
one point per season, but then we’d lose any insights from seasonal trends.
Instead, let’s add seasonal means as lines, which should enhance the chart but not
take away from the main picture.
Nice! Now we can easily see that the humidity is low in the Winter, but not as low
as in the Spring. Let’s look at two charts side-by-side.
Data Visualization Basics 252
While both charts display the same data and can both be helpful, the final version is
much more effective for our goal: to examine how humidity changes, depending
on the time of year.
Colors
One of the hardest parts of creating charts is choosing colors. The wrong colors can
make a chart unappealing, ineffective, and, worst of all, impossible to read. This
section will give you a good framework for choosing colors, important facts about
human perception, and simple tips.
Data Visualization Basics 253
Color scales
When choosing a color scale, you first want to identify its purpose. Going back to
what we know about data types, there are three basic use cases:
1. Representing a category
2. Representing a continuous metric
3. Representing a diverging metric
1. Representing a category
The first two data types we talked about (binary and nominal) will be best
represented with a categorical color scheme.
Since our metric values don’t have a natural order, we don’t want to use a color
scheme that has a natural order (like white to black).
d3 has built-in color schemes in its d3-scale-chromatic⁵⁶ library. We can see them
listed under Categorical if we start our server (live-server) and navigate to
http://localhost:8080/07-data-visualization-basics/scales/⁵⁷ in our browser.
⁵⁶https://github.com/d3/d3-scale-chromatic
⁵⁷http://localhost:8080/07-data-visualization-basics/scales/
Data Visualization Basics 254
These categorical color schemes have been carefully designed to have enough
contrast between colors. Each of these schemes is an array of colors — for example,
to use the first scale in this list, we would access each color at its index in
d3.schemeCategory10.
For categorical color schemes, it’s helpful for each color to have a different
descriptive name. For example, there might be two colors that could be described
as “blue”, which can be confusing to talk about.
For metrics that are continuous, we’ll want a way to interpolate in between color
values. For example, we could represent humidity values with a color scale ranging
from white to dark blue.
d3-scale-chromatic⁵⁸’s built-in continuous scales are visible under the Sequential
section.
and 1 would give us the dark blue on the right. To put it in familiar terms,
d3.interpolateBlues() is a scale with a domain of [0,1] and a range of [light
gray, dark blue].
These single-hue scales a great for basic charts and charts with multiple color scales.
However, sometimes the steps in between color values are too small and it becomes
hard to distinguish between values. In this case, d3-scale-chromatic has many
sequential color scales that cycle through another hue, increasing the difference
between values.
We can see how much easier it is to tell the difference between colors 50% and 60%
of the way through a single-hue scale (left) versus a multi-hue scale (right).
Our sequential color scales get more visually clear as they approach the end of the
scale. This is great to highlight values on the high end (for example, the highest
humidity values will be the most visible).
Sometimes, however, we want to highlight both the lowest and highest metric values.
For example, if we’re looking at temperature, we might want to highlight the coldest
days as a bright blue and the hottest days as a bright red.
Diverging scales start and end with very saturated/dark color and run through a less
intense middle range.
We can see that we have both single-hue (per side) and multi-hue diverging scales.
Again, it’s helpful to cycle through more hues so users can pick up on smaller
differences between metrics, but it can get overwhelming when we are already
cycling through two hues. When getting a feel for color scales, try a few different
options to get an idea for what will work in your specific scenario. This is not a
one-size-fits-all decision.
We’ll talk about the difference between the different color spaces (rgb, hsl, hcl) in
the next section.
d3.interpolateRgb("cyan", "tomato")
Our code has been started off with a few different custom scales.
⁵⁹https://github.com/d3/d3-interpolate
Data Visualization Basics 258
1. the scale’s name (which needs to be a string without spaces, since we’re using
it to create and reference an id), and
2. a color scale (a function that returns a color when passed a value between 0
and 1)
We’ve already created a few custom scales — check out the first three which have
the same range but use different color spaces for interpolation.
We can also create new discrete color schemes. The last two custom scales are using a
function called interpolateWithSteps() defined on line 33. interpolateWithSteps(n)
returns a new array of n elements interpolating between 0 and 1. For example,
interpolateWithSteps(3) returns [0, 0.5, 1].
We can use this function to make a new color scheme by stepping through a color
scale and returning equally spaced through the range. For example:
1 interpolateWithSteps(6).map(
2 d3.interpolateHcl("cyan", "tomato")
3 )
We can look at the color scheme by passing a unique id and our scale to addCustomScale().
Data Visualization Basics 259
addCustomScale(
"interpolate-hcl-steps",
interpolateWithSteps(6).map(
d3.interpolateHcl("cyan", "tomato")
)
)
If we wanted to create discrete color schemes with many colors, we could use one of
d3-scale-chromatic’s cyclical color scales.
interpolateWithSteps(10).map(
d3.interpolateRainbow
)
Data Visualization Basics 260
Play around and make a few color scales or color schemes to get the hang of it.
color scales
• rgb
• hsl and
• hcl
Data Visualization Basics 261
While rgb is generally the most familiar, there are good reasons to use hcl when we’re
programmatically creating color scales. Read on to find out why.
keywords
In our chart code, we’ve used a lot of keywords which map to specific, solid colors.
For example, we made our histogram bars cornflowerblue. Color keywords are great
because they are easy to remember and mirror the way we refer to colors in natural
language.
currentColor
Most color keywords are ways we would describe a color in English, but there is one
handy special case: currentColor. currentColor will set the CSS property to the
current color property. This is great for controlling one-color charts from outside
a charting component. For example, we could make a line chart component whose
stroke is currentColor. This way, we could easily create multiple line charts with
different line colors, just by setting the color higher up in the DOM.
transparent
Another useful color keyword is transparent. This is great when creating invisible
SVG elements that still capture mouse events, such as the listening rect we used to
capture mouse movement in Chapter 5.
When creating data visualizations, we’ll often need to manipulate colors. In this case,
we can’t use the color keywords because there is no way to make cornflowerblue
10% darker. In order to manipulate our colors, we’ll need to learn about color spaces.
rgb
The color space you’re most likely to come across when developing for the web is
rgb. rgb colors are composed of three values:
r: red
g: green
b: blue
Data Visualization Basics 262
For each of these values, a higher number uses more of the specified color. Then these
values are combined to create one color.
This is essentially how LCD screens work: each pixel is made up of a red, green,
and blue light. The values are combined in an additive fashion, starting at black
and moving towards white with higher values. This may be counter-intuitive to
anyone who’s used paint on paper, where more paint = a darker color. For example,
rgb(0,0,0) represents black and rgb(255,255,255) represents white.
There is another optional value for all of the color spaces we’ll talk about:
a: alpha
The alpha value sets the opacity of the color — a color with an alpha value of 0 will
be transparent. If the alpha value isn’t specified, the color will be fully opaque.
rgb can be expressed in two formats. The first, functional notation, starts with rgb
and lists each color value, in order, within parentheses. Each value will be within a
range from 0 to 255. If we wanted to specify an alpha value, we would switch the
prefix to rgba.
For example, we would represent a cyan color as a combination of green and blue
colors.
rgb colors can also be expressed with hexidecimal notation, which begins with #
and lists each value with two characters in a range from 00 to FF.
Data Visualization Basics 263
rgb can be an unintuitive color space to work in — if I have a blue color and want
an orange of the same brightness, all of the values have to change.
Next, let’s look at a color space that is closer to our mental model of color.
hsl
In the hsl color space, the values don’t refer to specific colors — instead, they refer
to color properties:
h: hue. The hue represents the angle around a circular color wheel that starts at red
(0 degrees) and cycles through orange, yellow, …., back around to red (360 degrees).
s: saturation. The saturation value starts at gray (0%) and ramps up to a completely
saturated color (100%).
l: lightness. The lightness value starts at white (0%) and ramps up to black (100%).
(a: alpha. Again, the alpha channel is optional and defaults to full opacity (100%))
Data Visualization Basics 264
In hsl, our cyan color would be partially around the color wheel, fully saturated, and
of medium lightness.
hsl more closely matches our mental model of the relationship between colors — to
switch from a blue to a similarly dark & saturated orange, we would only have to
update the hue value.
hcl
The hcl color space is similar to hsl, with the same values (c is for chroma, which is
an alternative measure of colorfulness).
Data Visualization Basics 265
Instead of being spacially uniform, the values are perceptually uniform, so that a
red and a blue with a lightness of 50% will look the same amount of light. Let’s look
at the hue spectrum at 100% saturation and 50% lightness for both hsl (top) and hcl
(bottom).
We can see bands of color in the hsl hues that look lighter than the other hues (even
though they mathematically aren’t). Those bands are not visible in the hcl spectrum.
If we create a 3d cylinder of the hsl color space, we would have a perfect cylinder.
But when we visualize the hcl color space this way, we can see that it’s not a perfect
cylinder. This is because humans can’t detect changes in saturation for all colors
equally.
Data Visualization Basics 266
This makes hcl to ideal for creating color scales — if we’re using color to represent
different categories of precipitation, those colors should have the same amount of
visibility. This will help prevent one color from dominating and skewing our read of
the data.
Data Visualization Basics 267
d3-color
While the browser will recognize colors in rgb and hsl formats, there isn’t much
native functionality for manipulating colors. The d3-color⁶⁰ module has methods
for recognizing colors in the hcl format, for converting between formats, and for
manipulating colors along the color space dimensions.
We won’t go into the nitty gritty of d3-color, but it’s good to be aware of the library.
If you find yourself needing to manipulate colors manually, check out the docs at
github.com/d3/d3-color⁶¹.
Color tips
Here are a few tips to keep in mind when choosing a color scale.
Contrast
contrast example
Make sure that your colors have enough contrast with their surroundings in all cases.
If you’re using Chrome, there is a great tool to check that your colors have enough
contrast right in your dev tools. Learn how to use it in the Appendix.
⁶⁰https://github.com/d3/d3-color
⁶¹https://github.com/d3/d3-color
Data Visualization Basics 268
Color blindness
Don’t assume that your users can see your charts as clearly as you can. Almost 8
percent of males of North American descent have red-green color blindness. To make
sure most people can encode the information from your color scale, stay away from
scales where users have to distinguish between red and green.
Here’s a simulation of what the above picture looks for people with red-green color
blindness.
It’s impossible to see which countries have negative growth and which have positive
Data Visualization Basics 269
growth if we use the red/green colors on the left for our color scale on the map we
created in Chapter 6.
There are other forms of color blindness (yellow-blue and total), but they are less
common (under 1% of the population). When creating a chart with a color scale, it
helps to have a second way to extract the information, such as adding tooltips or
in-place labels.
There are sites where you can simulate different types of color blindness to test your
data visualizations, such as Color Flip⁶².
⁶²https://www.canvasflip.com/color-blind.php
Data Visualization Basics 270
Comparing colors
Smaller areas of color need are harder to compare. When using colors as an indicator,
make sure the colored elements are large enough.
Additionally, far-apart areas of color are harder to compare. Make sure the colors
you use have enough contrast that users can easily tell the difference.
Semantics
semantics example
Data Visualization Basics 271
Choose semantically meaningful colors when possible. For example, blue for lower
and red for higher temperatures.
Gray
gray example
Gray can be the most important color. When you want to highlight part of a chart,
keeping other elements gray helps viewers focus on that element.
Wrapping up
Great work making it through this chapter! Fundamentals can be tedious to learn,
but understanding the bare bones will give you a good foundation to making your
own data visualizations.
If you want a handy way to remember these basics, we included a cheat sheet PDF
with the advanced package – feel free to print it out or store it somewhere easy-to-
find!
Data Visualization Basics 272
Cheat sheet
Common charts
Let’s talk about common chart types. This chapter can help you in a few ways:
Communication
When discussing part of your interface, it’s important to have commonly understood
names for things. Imaging trying to talk about a button without knowing the word
“button”.
Having a common language for charts makes it easier to talk about existing and
potential charts.
Quick readability
Every time a user sees a chart, they need to orient themselves to figure out how
to read it. There are ways to help speed up that process, such as adding labels and
legends. But the biggest help is using a familiar chart type.
Since most people have encountered a basic timeline before, showing data in a
timeline will help users who don’t want to spend much time learning a new
visualization.
Common charts also have implicit connotations. For example, candlestick charts are
often associated with stock data — an association that can speed up (or hinder) your
users, depending on the usage.
Common charts 274
Inspiration
Common charts are a great jumping-off point for a more complex or unique chart.
Only making the basic chart types can seem boring, but thinking about a dataset in
the frame of one of these charts helps with brainstorming. What insights might your
readers glean if you plotted your data as a histogram? What about a radar chart?
Additionally, sometimes the most basic chart is the best suited for a new feature.
Chart types
Let’s go through some of the common types and discuss examples of each.
Common charts 275
Timeline
Most dashboards have at least one timeline.
timeline
For each of these charts, we’ll practice breaking it down into its component parts
— what kinds of metrics are we visualizing and in what dimensions are they
visualized?
timeline
We can change the smoothness by using different interpolations and passing them
to d3.line().curve(). Here is our line with different interpolations (the original
points are visualized as dots):
Common charts 277
Heatmap
You might be most familiar with heatmaps from the Github contribution graph.
github heatmap
A heatmap has three dimensions: x, y, and a color scale. In this case, the x axis
corresponds to week, the y axis corresponds to day of the week, and the color scale
domain is the number of contributions per day.
Dimensions: - vertical position of a square represents the day of the week (discrete)
- horizontal position of a square represents the week of the year (discrete) - color of
a square represents a continuous or discrete metric value (often using a sequential
color scale)
In our code library’s heatmap, we can cycle through different metrics to represent
in our color scale. For example, the moon phase heatmap shows a very clear 27-day
cycle.
Common charts 280
Alternatively, if we look at the mininum temperature heatmap, we can see the much
more gradual change of temperature between the four seasons.
Heatmaps don’t necessarily need to show one metric over time. For example, the
following heatmap shows percentile for Asian countries (x axis) for different metrics
(y axis). Note that heatmap cells don’t necessarily need to be square.
Common charts 281
Heatmaps are great at showing trends over time and correlations between metrics.
It’s harder to tell small differences in color scales than, say, bar sizes, so stay away
from them or add informative tooltips if you need to show exact numbers.
Radar
Radar charts are not the most common chart type, but they can be very useful when
looking at multiple metrics.
Common charts 282
radar chart
They consist of at least three axes, where each axis represents one metric.
Dimensions: - position of a point on the line on each spoke represents a continuous
or discrete metric value
While they aren’t very helpful for making exact measurements (it’s hard to tell how
much larger variable A is on chart 1 versus chart 2), they really shine when comparing
many charts with many metrics. Each chart creates a profile of that instance, and
multiple profiles can be easily compared to find similarities and outliers.
There are a few ways to present multiple profiles:
1. Show multiple polygons of different colors, all on the same chart. This
works well when there are fewer than five profiles.
2. Show one chart with a toggle between profiles. This works well when you
want to focus on one profile at a time, potentially with more information
alongside the chart.
Common charts 283
3. Show multiple radar charts. This works well when comparing more than five
profiles, with no extra information.
If we look at a group of radar charts for our weather data, we can pick out patterns.
For example, days in June have a common pattern (up and to the right) and days in
early November have a different pattern (mostly even all around). Additionally, we
can see that November 4th is a bit of an outlier, with an unusually low wind speed.
These patterns seem obvious, given what we know about seasons — winter days
have different weather than spring days. But we can dive in deeper to see more
granular patterns — for example, the second set of days are windier and less humid.
Additionally, we know a lot about how the weather changes between seasons.
Imagine looking at radar charts for less familiar data — radar charts would help
us quickly pick up patterns across many metrics.
Scatter
Scatter plots are great for looking at the relationship between two metrics (x and y
axes).
Common charts 284
scatter plot
For example, a scatter plot of minimum temperature and maximum temperature will
have a positive pattern, since temperature varies much more across days than within
days.
Negative
The two metrics are anti-correlated and one decreases while the other increases.
Common charts 286
For example, a scatter plot of wind speed and pressure will have a negative pattern,
since windier days tend to have less more pressure.
Null
The two metrics are not related.
Common charts 287
For example, a scatter plot of wind speed and humidity will have no pattern, since
wind speed and humidity are not at all correlated.
There are many, more complex patterns that might be visible in a scatter plot. For
example, scatter plots can be exponential or u-shaped.
Common charts 288
When looking at a scatter plot, keep one of the golden rules of data analysis in mind:
correlation does not imply causation. Even if a scatter plot has a strong positive
pattern, you cannot conclude that metric A causes metric B, or vice versa.
Sometimes an extra dimension is added to a scatter plot — a color scale or different
shapes per category. For example, we gave different kinds of precipitation a different
color in Chapter 2.
Common charts 290
donut chart
• Restrict the number of slices to five or fewer. Note that we combined the last
few slices into an “other” category to keep the focus on the larger slices.
Common charts 292
Histogram
We’re already familiar with this next chart type — we made our own histograms in
Chapter 3. Histograms are unique in that they are only concerned with one metric.
histogram
histogram
Bimodal
A bimodal histogram has two clear peaks.
Symmetric
A symmetric histogram no clear peaks and looks fairly flat.
Box plot
Box plots are great for showing discrete or binned continuous metrics — usually over
time or relative to another metric.
Common charts 297
box plot
Each box shows the distribution of values for a category. For example, the first box
in this chart shows the range of all humidity values in January.
Dimensions: - horizontal position of each box represents a discrete metric -
vertical position of each box and whisker represents a continuous metric - height
of each box and whisker represents a continuous metric
The middle line represents the median (the middle value) and the box covers values
from the 25th percentile (25% of values are lower) to the 75th percentile (75% of values
are lower). The “whiskers” extend 1.5 times the inter-quartile range (IQR) from the
median in either direction. By IQR, we mean the difference between the 25th and
75th percentiles. Any outliers that lie outside of the whiskers are shown as dots.
Common charts 298
Note that there are different conventions for where to place the ends of the
whiskers. For example, some charts cover all data points with the whiskers,
whereas some charts cover one standard deviation above and below the mean. Our
definition of 1.5 times the IQR from the median is a good middle-ground that
will show the variation, but will also be resistant to outliers.
Now that we understand the different parts, let’s take another look at our weather
box plot.
Common charts 299
box plot
By looking for outlier dots, we can see that both March and April had two abnormally
hot days. Looking at the whiskers, we can see that there is no overlap between max
temperatures in March and June. Additionally, the maximum temperatures in March
are all very similar, whereas February has a lot of variability.
If we were just looking at the mean or median temperature of each month, we
wouldn’t be able to make any conclusions about the range of values. A box plot has
added complexity and can take more practice to read, but packs a lot of information
about variability.
Box plots are more complex than the charts we’ve created before — be sure to check
out the code in the /code/08-common-charts/box-plot/ folder. Note that we
create a <g> element for each of the months’ elements to keep them together and
make things easier with our data binding enter/exit pattern.
Conclusion
This is just the tip of the iceberg. There are many more types of charts and many chart
types that have yet to be discovered! Remember to use this chapter as a starting off
Common charts 300
point, or as a source of inspiration. And make sure to look through the code for each
example or try to recreate the chart on your own to solidify your skills.
Dashboard Design
So far, we’ve talked about visualizing data at the chart level. Let’s zoom out another
level and talk about how to design effective dashboards.
conceptual scopes
What is a dashboard?
Good question! There are many definitions of “dashboard” - in this chapter, I’ll be
using the word to refer to any web interface that makes sense out of dynamic data.
This is by no means the official definition of dashboard, but it is a handy definition
for our use here.
We’ll talk about ways to display a simple metric effectively, how to deal with different
data states when loading data, how to design tables, and then we’ll run through a
Dashboard Design 302
design exercise. You’ll come away with tangible strategies for displaying data in an
actionable manner. Let’s dig in!
metrics
This certainly works! Your users can read the numbers and go from there. But what
does “go from there” mean? Let’s try to anticipate some questions that need to be
answered in order to convert these numbers into insights.
If you take anything away from this chapter, let it be that context is king. A number
is meaningless if we don’t know:
1. What it means — are we looking at miles per hour or meters per second?
2. How it compares — is this an average value, or is it abnormally high?
Dashboard Design 303
metrics finished
We’ve added a gauge on top of our numbers — the range could be either:
The gauge lets the user know how the value compares to other values. Almost as
important, though, is letting the user “read” the metric more quickly.
Both the position and the color of the bubble give a fast impression of how the metric
compares. A lighter or more left-ward bubble means a low value — that’s easy to scan
for.
Adding gauges also makes metric comparisons easier — the atmospheric pressure
is high, but not as abnormally high as the wind speed.
But why did we add bubbles to the gauges? Let’s take a look at the gauges with and
without a bubble. Notice how quickly you can scan both sets.
Dashboard Design 304
With the background gradients, we’re a little overwhelmed with colors — it’s hard to
isolate the color of the actual value. Sometimes adding an extra, redundant element
helps with focusing the user on what matters.
There are many types of gauges and visual indicators that can help with quickly
interpreting numbers. This gauge is a simple example — put into terms we learned
in Chapter 7, we’re representing the metric value with both a position and color
dimension. Think of some other ways we could represent such a value, and what
the pros and cons would be of each choice.
Let’s look at the other changes we made to make our metrics easier to process:
Dashboard Design 305
metrics
metrics finished
• what the past values have been? Showing where this value lies in a distribution
could give the user lots of information.
• what the value was one week ago?
• what the value is in another location?
• what the value is likely to be in the future?
Another way to make simple numbers more insightful is to combine them into a new
metric. For example, I was recently working on a dashboard that showed two things:
Even with the historical context, these values are not very actionable. You might
think “Wow, a lot of people are reading about football”, then move on with your day.
In order to change numbers into insight, we created a new metric — views per
article.
Dashboard Design 308
views, articles viewed, and views per article, from the Parse.ly Currents dashboard
Now, a journalist could see this and think “Wow, articles about football are getting
a lot of attention (relative to other topics), they must be high in demand!” That’s
actionable — the journalist can take that information and write that football article
they’ve always wanted to write, knowing that it will interest more people than
another topic.
Dashboard Design 309
Data states
In addition to the final design, our dashboard charts will need to handle multiple
states:
• loading
• loaded
• error
• empty
But what should our charts look like during each of these states?
Loading
While this is better than nothing, we can give our users more information. In the
case of data that takes more than a few seconds to load, we should help users decide
whether they want to wait, and prepare them for when the data are available.
What extra information can we give our users without distracting them from
anything else on the page? A greyed-out version of your final chart often works
well.
Dashboard Design 311
loading state
Additionally, an animation can be helpful to indicate that this state is not final.
Another solution is to render the chart with empty or random data. This is generally
a bad idea, though, since it too closely mimics the loaded state and could confuse
the user.
Loaded
When your data have loaded, animating the final state in looks great and also alerts
your user that the wait is over!
Dashboard Design 312
loaded state
Make sure that your loading and loaded states are the same size, so your content isn’t
jumping around on the page. This is especially important if the user could be looking
at something furthur down on the page.
Error
Error states are important signals that something is wrong and the chart won’t be
appearing. Make sure these are visually distinct enough from your loading state that
a waiting user is alerted quickly.
Dashboard Design 313
error state
Error states can be jarring — we’ve all had the experience of waiting for a page to
load, just to be rejected. Make sure to tell your users what they can to do to fix the
issue: try again later, upgrade their account, change the filter, etc.
Empty
Empty states are a unique error state — instead of signalling that there was an issue,
they signal that the data does not exist.
Dashboard Design 314
empty state
Empty states require an extra check to see if there are any data values. For example,
we might receive an array of empty values for a timeline — in this case, we’ll have to
run through the whole array to check whether or not we have any non-zero values.
Dancing numbers
If we’re showing numbers on our dashboard, it’s a good idea to use a monospace font
— especially when those numbers will be changing when the data updates or filters
change. This will prevent layout changes when the numbers change, plus numbers
in a column will be easier to compare (such as in a table).
This might require bringing in a new font, although some fonts have a fixed-width
setting that can be toggled via CSS. For example, the font we use in our chart code,
Inter, can be switched to fixed width with the following CSS.
font-feature-settings: 'tnum' 1;
Dashboard Design 315
The data we use for our chart can vary wildly — the values might cluster together
tightly or we might have to display very different values. When there are extreme
outlier values in our data, the scale can be exaggerated and drown out smaller values.
As usual, the solution will depend on the goal of the chart. If the goal is to show a
birds-eye-view of the data, letting the outliers skew the whole chart might be ideal.
In most cases, though, the goal is to see the pattern for most of the data (and be
notified that outliers exist). Let’s look at a few examples.
This timeline (in the Parse.ly Analytics dashboard) is designed to show daily views
over time for a specific website.
Dashboard Design 316
Setting the y scale domain to [0, max daily views] usually works, but if the
website publishes an article that goes viral, the site might receive four times the
typical daily traffic. While exciting, this will blow out the y scale and make daily
trends hard to see.
To make the solution clear, let’s lay out the goal for this chart: to show website owners
how much traffic their site is getting — is that number increasing or decreasing over
time? Are there weekly or montly patterns? Are there any recent changes that they
need to be aware of?
With this in mind, we’ll apply two rules:
1. If an outlier exists in recent data (say, the past week), we’ll use the default y scale
domain of [0, max daily views]. This enables viewers to see the extremity
of an ongoing traffic spike — something to celebrate!
2. If an outlier exists more than 7 days ago, it will get cropped and we’ll define our
y scale domain as [0, mean daily views + 2.5 standard deviations].
We don’t want to lose this data, though! We’ll show a stripe on top of the day’s
bar, with extra stripes for more extreme outliers. This way we essentially have
two y scales — one for most of the bars and one for the outlier stripes, enabling
us to see both the general pattern and get a sense of extreme traffic patterns.
Dashboard Design 317
timeline with cropped outliers - cropped, from the Parse.ly Analytics dashboard
There are many creative ways to show outliers while preserving the rest of the chart,
but how do we detect outliers?
There are various definitions of an “outlier”. A good general-use definition is any
value that is more than 2.5 standard deviations from the mean.
Designing tables
Often we’ll want to show the exact values in our dataset. In this case, it’s hard to beat
a simple table. Although designing a table sounds straightforward, there are ways to
make a simple table more engaging and easier to read.
Let’s redesign a table of our weather data — feel free to follow along at
/code/09-dashboard-design/table/draft/.
Writing the code along with these changes is a great way to practice your d3 and
CSS skills. If you get stumped, check out the finished code at
/code/09-dashboard-design/table/completed/
Even if you’re not following along, check out the original table at http://localhost:8080/09-
dashboard-design/table/draft/⁶³.
⁶³http://localhost:8080/09-dashboard-design/table/draft/
Dashboard Design 319
We’ll start out with an unstyled table showing several metrics (columns) over
multiple days (rows).
Our table
This is pretty hard to read, with cramped cells and dark borders. To start, we’ll add
padding to each cell and take away the borders.
Unfortunately, without those borders it’s hard to separate each row. We could add
the borders back and lighten them, but that will still add visual clutter. Instead, let’s
add zebra striping: alternating rows are a different color. This will help us follow a
row from the left to the right.
Dashboard Design 320
Much better! Now we can focus on the values. There are a few good rules of thumb
for text alignment in tables:
1. Align text to the left. This helps users scan the table, since we’re used to reading
left to right.
2. Align numerical values to the right. This helps with comparing different
numbers within the column because their decimal points will line up.
3. Align the column headers with their values. This will help with scanning and
keep the label close to the metric value.
Dashboard Design 321
Our numbers are still hard to compare, though, since they have different amounts
of decimal points. Let’s use d3.format() to ensure that all of our numbers are the
same granularity. The number of decimal points we want will depend on the metric
— note that we keep one decimal point for temperature but two decimal points for
wind speed.
Let’s also format our dates to make them more human friendly and get rid of
redundant information.
These numbers are still hard to compare! The value 47.1 looks smaller than the next
Dashboard Design 322
font-feature-settings: 'tnum' 1;
If the font you’re using doesn’t have this feature, switch to a monospace font for your
numbers.
This table is much more readable than it was, but we can do more to help users
process it faster. Remember the dimensions in which we can represent data from
Chapter 7? Those dimensions aren’t restricted to charts. Let’s create a color scale
for Max Temp and Wind Speed and fill the background of each cell with its value’s
color.
We’ll make sure to use semantic colors for quicker association: blue to red for
temperature, and white to slate gray for wind speed. Additionally, having two
different color scales helps keep these columns visually distinct.
Dashboard Design 323
Great! Now we can quickly see that the first few days were warm and them cooled
down, and the days are windy in little spurts.
Another way to make our table more scannable is to convert UV Index into symbols.
Since it is a discrete metric, we can display a number of symbols to represent the
value.
There is another column that we can convert to symbols: Precipitation. But we also
notice that it’s a binary metric, and there are only two possible values. Let’s change
it into a Did Snow column and only display a symbol if the value is snow.
Dashboard Design 324
Let’s see if we can improve on Max Temp Time. Since the times span both AM and
PM, they are hard to compare quickly. Instead, we can use another data dimension:
position. Instead, let’s display a line that is further right if the max temp occured
later in the day.
This is much easier to scan! We can quickly see that the max temp was earlier in the
day for the first four rows, then in the afternoon for the next four rows. As usual, how
Dashboard Design 325
you represent this metric will depend on your goals. If the specific time is important,
maybe adding another column with the exact value would be helpful.
The last column we want to update is our Summary metric. The strings can be quite
long and take up lots of space. Let’s bump that down a font size to save space and
decrease its visual importance.
If we have a long table, we can remind our users what each column is for by sticking
the header to the top. Thankfully, modern css makes this simple, we only need these
lines of CSS:
thead th {
position: sticky;
top: 0;
}
To make the stickied header stand out while it’s stuck, let’s give it a dark background
color and light text.
Additionally, let’s highlight a row when the user hovers over it. This will allow them
to easily compare values all the way across a row.
Dashboard Design 326
Wonderful! If you haven’t been following along, make sure to check out the final
implementation at /09-dashboard-design/table-completed/.
Even though creating a table may not sound exciting, there is room to use our new
knowledge of data types and data dimensions to make it more effective and easier to
read.
Keep the number of metrics small though, to prevent making the user process a lot
of information at once. Here’s the same dashboard with eight metrics shown at once
– a user could easily feel overwhelmed with this design. What should they look at
first? When everything is important, nothing is.
Dashboard Design 329
Data density can increase with a user’s comfort – maybe add an option for a denser
layout for power users?
to show a longer explanation and/or a chart. This is a good option when the user
wants to compare multiple items.
Switch to a detail page
to show much more information. A whole page gives enough space for multiple
detailed charts and paragraphs of information. They also feel more concrete and
allow users to bookmark or share the url.
Deciding questions
There isn’t one layout that will fit any purpose, but a few choice questions can guide
you towards a more suitable layout.
For example, the layout on the left might be more accessible for unfamiliar users, but
might be totally unnecessary for a domain expert.
Dashboard Design 331
There are many ways to build a dashboard – hopefully this information will help on
your next project. Remember to choose a main focus for each screen, give metrics
context, and start with the goals of the dashboard.
Complex Data Visualizations
In these bonus chapters, we’ll combine everything we’ve learned into three complex
data visualizations. These walkthroughs will not be as in depth as the previous
chapters — instead, they are meant to show you how to build on your new knowledge.
We’ll take a higher-level look at each chart — what are the motivations, and what are
the tricky code bits that crop up when making them. When you take your new skills
and set out to make your own complicated, awesome data visualizations, you’ll run
into new challenges that we haven’t covered here.
But fear not! You’ll be able to use your solid foundation that we’ve built in this book
to tackle those new puzzles. Let’s get a sneak peak of how to approach these puzzles
in the road ahead.
Marginal Histogram
First, we’ll focus on enhancing a chart we’ve already made: our scatter plot. This chart
will have multiple goals, all exploring the daily temperature ranges in our weather
dataset:
• do most days have a similar range in temperatures? Or do colder days vary less
than warmer days?
• did we have any anomalous days? Were both the minimum and maximum
temperatures anomalous, or just one?
• how do the temperatures change throughout the year?
As you can see, more complicated charts have the ability to answer multiple
questions — we just need to make sure that they’re focused enough to answer them
well.
To help answer these questions, we’ll add two main components to our scatter plot:
• a color scale that shows when the day falls in the year
• one histogram on the x axis to show the distribution of minimum temperatures
and one histogram on the y axis to show the distribution of maximum
temperatures
Marginal Histogram 335
Marginal Histogram
To start, we can use the scatter plot code that we’ve already created. Let’s start up
our server in the terminal (live-server, if that’s your preference) and navigate
to the url http://localhost:8080/10-marginal-histogram/draft/⁶⁴. We can see our old
scatter plot friend, this time with the minimum daily temperature on the x axis and
maximum daily temperature on the y axis.
⁶⁴http://localhost:8080/10-marginal-histogram/draft/
Marginal Histogram 336
Scatter plot
1. Follow along with the code examples and try to fill the missing code gaps.
We won’t go over every line of code in this chapter, but the completed code
is available at /code/10-marginal-histogram/completed/ if you need a
reference.
2. Read through each example to follow the journey, then dive into the code and
try to re-create the chart, using the /completed/ folder as a reference.
The choice is yours, but make sure to choose whichever option will increase your
confidence in your skills so you can launch into your own crazy custom data
visualizations after.
Let’s go through each of the enhancements of the final chart.
Marginal Histogram 337
Since our bounds are rectangular, this will be as simple as creating a new <rect>
element with a white fill. Our main concern here is to draw our <rect> early on in
our chart code to make sure it doesn’t cover any other chart elements.
First, we’ll create the <rect>:
code/10-marginal-histogram/completed/scatter.js
We’ll set the <rect>’s fill in our CSS file, to keep our static styles out of the way of
the general chart logic.
.bounds-background {
fill: white;
}
Marginal Histogram 339
Equal domains
Next, we’ll notice that in our completed chart, our x and y scales have the same
domain — they both run from 0 to 100 degrees (at least in the New York City data).
This is to keep the frame of reference the same on both axes, so readers can more
easily compare minimum and maximum temperatures.
If we draw a diagonal line from the bottom left of our bounds to the top right of our
bounds, we can see the general range of temperatures in a day by looking at how far
the dots are to the left (or to the top) of this line.
Marginal Histogram 340
To find the smallest minimum temperature and the largest maximum temperature,
we can find the extent of a new array of minimum and maximum values.
code/10-marginal-histogram/completed/scatter.js
Sometimes it’s easier to hand d3 a new array with multiple values per data point
than to create several variables and choose the smallest and largest ones.
Marginal Histogram 341
We can then use temperaturesExtent as the domain for both of our axes: xScale
and yScale.
We could have used the same scale for both of our axes, since our xScale and yScale
have the same domain and range, but it’s also nice to keep them separate to be more
explicit and to be prepared for future changes.
To remind yourselves of the data structure, it’s often a good idea to temporarily
add a console.table(dataset[0]) line right after we retrieve our dataset.
Hmm, what if our dataset spans multiple years? We want to show the time of year,
not the absolute date. Let’s normalize our dates to all be in a specific year — that way
we can color them according to their month and day, without taking the year into
account.
code/10-marginal-histogram/completed/scatter.js
Great! Now we need to create a scale to convert our date objects into colors. We’ll
want to place this in our Create Scales section, probably after the other two, more
basic, scales.
Let’s use one of d3-scale-chromatic⁶⁵’s built-in scales. d3.interpolateRainbow()
will work here because our data is cyclical — we want our color scale to wrap around
seamlessly. This will ensure that days in winter are similar colors, whether they’re
in December or January.
To create a scale that uses d3.interpolateRainbow(), we can use a sequential scale
(d3.scaleSequential()) that covers all of the year 2000. Instead of setting a range,
we’ll use the .interpolator() method to use one of the built-in scales.
⁶⁵https://github.com/d3/d3-scale-chromatic
Marginal Histogram 343
Note that we’re setting our scale to cover dates in 2000. Even though our dataset has
more recent dates, our colorAccessor() will modify them to preserve their month
and day, but change their year to 2000, so that they are within our colorScale’s
range.
This scale definitely works, but remember that we want our colors to have semantic
meaning if possible.
d3.interpolateRainbow normal
The purple/blue colors in Winter work well, but burnt orange colors are often
associated with Fall, and we have them representing Spring. Instead, let’s invert our
color scale so that orange-y colors represent Fall and green-y colors represent Spring.
Marginal Histogram 344
Now that we have our color scale, we’ll use ot to color our dots. We’ll need to find
where we create our dots and set their attributes, then set their fill attribute.
code/10-marginal-histogram/completed/scatter.js
Wonderful! Now our dots are all the colors of the rainbow:
Marginal Histogram 345
Without a legend or interactions, it’s hard to tell what each color means (an important
thing to remember when designing charts). However, we can see that the purple-y
dots are near the bottom left, so we can guess that those are around January, when
Winter hits New York.
Mini histograms
Next up, we’ll tackle one of the bigger changes — adding a histogram to the top and
right of the chart. These histograms will be great for giving readers a sense of how
the minimum and maximum daily temperatures are distributed — some climates
have a normal distribution for min temperatures, but a bimodal distribution for max
temperatures!
We’ll set the dimensions of our histograms (height and margin) in our dimensions
object in the Create chart dimensions step to keep the chart size settings in
Marginal Histogram 346
one place. This way, we know where to change things if we want to update the
proportions of our chart.
We’ll also want to bump up the top and right margins to account for our
histogramHeight and histogramMargin.
let dimensions = {
width: width,
height: width,
margin: {
top: 90,
right: 90,
bottom: 50,
left: 50,
},
histogramMargin: 10,
histogramHeight: 70,
}
Let’s start with the top histogram - first we want to generate the bins using
d3.histogram(), similar to how we did in Chapter 3.
code/10-marginal-histogram/completed/scatter.js
Next, we’ll use these bins to create a y scale for our histogram.
Note that we’re creating a scale outside of our Create scales step. Now that our
charts are getting more complicated, we get to take our knowledge and decide when
to break the rules. If we want to strictly follow our chart checklist, we could move
Marginal Histogram 347
these steps up into our Create scales step. We’ll keep them in the Draw data step
here, though, to group them with the following histogram drawing code. When you
create your own charts, of course, the code organization will be totally up to you!
code/10-marginal-histogram/completed/scatter.js
Next, we’ll create a new bounds group for our top histogram and position it above
our regular bounds.
code/10-marginal-histogram/completed/scatter.js
We want to draw a <path> within these bounds, but first we need to figure out how
to create our <path>’s d attribute string. If you remember from Chapter 1, we used
d3.line() to create a d string generator to draw our timeline. We’ll do a similar
thing here, but using d3.area(), which is basically the same, but uses y0 and y1
methods instead of y to draw the top and bottom of our <path>’s shape.
code/10-marginal-histogram/completed/scatter.js
You might be wondering: why don’t we just use d3.line() and give it a fill
instead of a stroke? The reason we want to use d3.area() is because the path
created by d3.line() has no concept of the bottom of the chart. This will mostly
work here, since our histogram starts and ends at the bottom of its bounds:
But if one side of our line ended in a different place, d3.line() would end the d
string at that point, and the <path>’s fill would draw a point from the left-most
point to the right-most point, slicing our area in half.
code/10-marginal-histogram/completed/scatter.js
Great! Let’s dim the color of our histogram so we don’t overwhelm the rest of our
chart.
Marginal Histogram 350
.histogram-area {
fill: #cbd2d7;
}
Perfect! Now we can repeat these steps to draw a histogram on the right side of our
chart.
Marginal Histogram 351
code/10-marginal-histogram/completed/scatter.js
Notice that we’re drawing our histogram vertically, similar to our top histogram, and
then rotating it 90 degrees (clockwise).
Marginal Histogram 352
Hmm, we’ve rotated our histogram around the wrong axis though. Don’t worry! We
can set the rotation axis with the transform-origin css property, setting it to the
bottom, left of our histogram group (instead of the default top, left).
.right-histogram {
transform-origin: 0 70px;
}
Marginal Histogram 353
Static polish
Before we turn our attention to the interactive features, let’s do a little bit of clean-
up. Switching over to our HTML file, let’s add a title and description above our
#wrapper element. You might want to use different text — what will describe the
chart to readers as succinctly as possible, but still tell them what they need to know.
• try to mimic the styles in the following screenshot, practicing your CSS skills,
• copy the styles from the /completed/styles.css file, or
• make up your own styles!
Marginal Histogram 355
Let’s also dim the lines in our axes — the bounds of our chart aren’t very important
here, and we don’t want to grab the reader’s attention with them.
.tick line,
.domain {
color: #cfd4d8;
}
Marginal Histogram 356
This color was chosen because it’s a lighter, desaturated version of the dark color
we’re using for our text. Staying in the same “family” of grey prevents our greys
from clashing.
Adding a tooltip
Let’s move on to adding interactions! Let’s copy what we did in Chapter 5 with
adding voronoi polygons for a scatter plot tooltip. We won’t look at the code here,
Marginal Histogram 357
Alright! Let’s make one improvement here — the color of the dot is meaningful, but
we’re hiding it with an overlapping dot on hover. Let’s bump up the size of that hover
dot and update the styles to have a stroke and no fill.
.tooltip-dot {
fill: none;
stroke: #6F1E51;
stroke-width: 2px;
}
Now we can see the color of the dot we’re hovering, while still showing which dot
is hovered.
Marginal Histogram 359
After we draw our voronoi polygons, let’s add a horizontal and a vertical line that
we can move in our hover event. We’ll want to use <rect> elements, since <line>s’
attributes (x1, x2,etc) won’t respect CSS transitions, but <path>s’ d attribute will.
First, we’ll create our elements just before our onVoronoiMouseEnter() function.
Marginal Histogram 360
code/10-marginal-histogram/completed/scatter.js
Now we can update our <rect>s attributes at the bottom of our onVoronoiMouseEnter()
function, after we define our hovered x and y.
code/10-marginal-histogram/completed/scatter.js
Those black lines are a bit intense — let’s change the color by setting the fill in our
CSS file.
.hover-line {
fill: #5758BB;
}
You might notice that these lines flicker when hovering around the chart. This is due
to this chain of events:
• we hover a dot
• our onVoronoiMouseEnter() function will be triggered
• our hover lines will show
• our hover lines will capture the mouse
• our onVoronoiMouseLeave() function will be triggered, since our pointer is
now hovering a hover line
• our hover lines will be removed
• we’ll hover a dot, now that our hover lines are gone
Marginal Histogram 363
• etc
To prevent the hover lines from capturing the hover event, we can add the pointer-events:
none CSS property to our hover lines. This will prevent them from capturing any
mouse events.
We’ll also smooth the movement of our hover lines with a CSS transition and
decrease their opacity.
pointer-events: none;
transition: all 0.2s ease-out;
opacity: 0.5;
Marginal Histogram 364
Our transitions are smoother and this reduces the prominence of our bars somewhat,
but they’re still very “loud”. Here’s a fun trick: we can use a blend mode to change
how the color interacts with its “background”.
We can set the blend mode by using the mix-blend-mode CSS property.
mix-blend-mode: color-burn;
There are many blend mode options — play around with a few of them! For example,
color-dodge, overlay, hard-light, and difference. I’ll stick with color-burn
here, since it makes our line almost invisible outside of the histograms, which is
Marginal Histogram 365
great because we don’t want it obstructing other elements in our bounds. This way,
it shows where the values lie in our histograms, but doesn’t get in the reader’s way.
Adding a legend
The meaning of each color isn’t clear to our readers — let’s add a legend in the bottom,
right of our bounds. We want this legend to be a bar containing our gradient, with
a few dates called out on top.
To start, we’ll add our legend bar’s dimensions to our dimensions object.
code/10-marginal-histogram/completed/scatter.js
20 let dimensions = {
21 width: width,
22 height: width,
23 margin: {
24 top: 90,
25 right: 90,
26 bottom: 50,
27 left: 50,
28 },
29 histogramMargin: 10,
30 histogramHeight: 70,
31 legendWidth: 250,
32 legendHeight: 26,
33 }
Next, we’ll create a new <g> element to position our legend. Let’s place our legend
in the bottom, right of our chart, since there won’t be any days in our dataset with a
lower maximum temperature than minimum temperature (and therefore, no dots in
the lower, right-hand corner of our chart).
Marginal Histogram 367
This code can go at the end of our Draw peripherals step, after we’ve created our
axes.
code/10-marginal-histogram/completed/scatter.js
183 const legendGroup = bounds.append("g")
184 .attr("transform", `translate(${
185 dimensions.boundedWidth - dimensions.legendWidth - 9
186 },${
187 dimensions.boundedHeight - 37
188 })`)
Next, we need to create a gradient. This will be similar to the gradient we created in
Chapter 6 for our map, but with more <stops>.
We’ll create our gradient within a <defs> element, to keep our code organized so
we know where to find re-useable elements.
code/10-marginal-histogram/completed/scatter.js
190 const defs = wrapper.append("defs")
To create a gradient that will match our color scale, we’ll want to create an array of
equally-spaced colors within our gradient. We’ll want a good number of stops — if
we only have 3 stops, our gradient will end up looking like this.
Instead, let’s pick 10 colors within our color scale, so our gradient will better represent
our colors.
Marginal Histogram 368
d3.range(5)
will create the array [0, 1, 2, 3, 4]. Let’s create an array of the number of stops
we want (10) and normalize the indices to count up to 1.
code/10-marginal-histogram/completed/scatter.js
Now we can use our stops array to create one <stop> per item, with the matching
color and offset percent.
Note that we want to set the id of our gradient so we can reference it later.
Marginal Histogram 369
code/10-marginal-histogram/completed/scatter.js
Now we can create a <rect> to display our gradient, using the same id that we set
for it.
code/10-marginal-histogram/completed/scatter.js
Looking good! If you’re following along, try playing with different numbers of stops
to see how the granularity of the gradient affects its appearance.
Marginal Histogram 370
This gradient isn’t much help by itself — let’s add ticks, similar to our chart axes.We
could use one of d3’s axis generators (eg d3.axisBottom()), but that seems like
overkill here.
We want to label the start of a few key months within the year — because we want
to be so specific, let’s hard-code an array of a few key dates.
Marginal Histogram 371
code/10-marginal-histogram/completed/scatter.js
Next, we’ll create a scale to help position our tick <text> elements — this scale should
convert a date into an x-position on our legend.
code/10-marginal-histogram/completed/scatter.js
Now we can use our tickValues array to create <text> elements along our legend
gradient.
code/10-marginal-histogram/completed/scatter.js
We’ll also add tick lines, since it’s not very clear where March 2nd is on the gradient.
code/10-marginal-histogram/completed/scatter.js
These lines won’t show up just yet — we need to add a stroke color in our
styles.css file. While we’re in there, let’s also bump down the font size of our
ticks and make sure they’re horizontally centered with the tick.
.legend-value {
font-size: 0.76em;
text-anchor: middle;
}
.legend-tick {
stroke: #34495e;
}
Wonderful! Now our readers will have a better idea of what time of year each color
corresponds to.
Marginal Histogram 374
Right now, nothing happens when we hover the legend. Let’s add mouse events to
capture mouse events at the end of our Set up interactions step. Since we want to
do something every time the reader’s mouse moves, we’ll listen for "mousemove"
events instead of "mouseenter" events.
Marginal Histogram 376
legendGradient.on("mousemove", onLegendMouseMove)
.on("mouseleave", onLegendMouseLeave)
function onLegendMouseMove(e) {
}
function onLegendMouseLeave() {
}
Next, we’ll create our static hover elements right before we define our onLegendMouseMove()
function. The reason we want to do this outside of our function is to prevent from
creating new elements every time we move our mouse over the legend. It’s more
performant and cleaner code.
We’ll set the width of the “selected” region of our legend, then create a <g> to house
our hover elements and hide it for now. This group will serve a similar purpose to
the hoverElementsGroup we created before - we’ll only have to show and hide one
element in our mouse events, as opposed to having to remember every single element
that is visible on hover.
code/10-marginal-histogram/completed/scatter.js
Now we can create our hover elements - a “bar” that will show what region of the
legend we’re highlighting and a <text> element to show the dates.
Marginal Histogram 377
code/10-marginal-histogram/completed/scatter.js
Note that we’re shifting our text over by half of the bar width. We need to do this
in order to center the text with the bar — the group will expand to be as wide as
our bar and when we use the text-anchor: middle CSS property, we want it to
be aligned with the center of the bar, instead of the left side of the bar.
Perfect! Now let’s flesh out our onLegendMouseMove() function. First, we’ll need to
figure out what dates we’re hovering. To find the point we’re hovering, we need to
find our mouse’s x position relative to our legend.
We can use the d3.mouse() function to get back an [x, y] coordinates. These
coordinates will be relative to the element that we pass to d3.mouse() — we can
give it this, which is the element that we initialized our mouse event listener on
(legendGradient).
function onLegendMouseMove(e) {
const [x] = d3.mouse(this)
}
Marginal Histogram 378
We’re using ES6 array destructuring to grab the x value directly — this is the
equivalent of: {lang=javascript,line-numbers=off}
const coordinates = d3.mouse(this) const x = coordinates[0]
Now that we have the x position of our mouse on the legend, let’s calculate the
minimum and maximum range of dates that we we’re going to highlight. Remember
that we can use a scale’s .invert() method to convert values from the range
dimension to the domain dimension. If we use the legendTickScale we used
previously to position our legend tick values, we’ll be able to convert a legend x
position into a date.
code/10-marginal-histogram/completed/scatter.js
353 const minDateToHighlight = new Date(
354 legendTickScale.invert(x - legendHighlightBarWidth)
355 )
356 const maxDateToHighlight = new Date(
357 legendTickScale.invert(x + legendHighlightBarWidth)
358 )
If we use the x position of our mouse to position our highlighted bar, it will go out-
of-bounds when we approach the left or right side. An easy way to bound a value is
to use d3.median() which will sort the values in a passed array and pick the middle
value. This works to our favor — when we pass an array of [min, value, max],
there are three possible scenarios:
• the value is lower than the min, resulting in the sorted array [value, min,
max], making min the middle value
• the value is between the min and the max, resulting in the sorted array [min,
value, max], making min the middle value
• the value is higher than the max, resulting in the sorted array [min, max,
value], making max the middle value
Let’s use that to bound our bar’s x position by 0 (the beginning of our legend) and
the legend’s width minus the highlight bar’s width.
Marginal Histogram 379
code/10-marginal-histogram/completed/scatter.js
Lastly, let’s show and position our legendHighlightGroup and update the text of
our legendHighlightText. We want it to display the minimum and maximum
dates, separated by a hyphen.
legendHighlightGroup.style("opacity", 1)
.style("transform", `translateX(${barX}px)`)
legendHighlightText.text([
d3.timeFormat("%b %d")(minDateToHighlight),
d3.timeFormat("%b %d")(maxDateToHighlight),
].join(" - "))
Notice that we’re using the padded version of the date’s day: "%-d" will format
March 2nd as 2 whereas "%d" will zero-pad the day and format it as 02. While a
little less readable, this will keep the date from jumping around when it switches
from a 1-digit day to a 2-digit day. Try both versions to see the difference for
yourself!
Okay great, now we can see where on the legend we’re hovering, and the date range.
Marginal Histogram 380
It’s a bit hard to read, though. Let’s also dim the normal legend ticks so they don’t
distract from the highlighted portion.
legendValues.style("opacity", 0)
legendValueTicks.style("opacity", 0)
Although usually removing information is a bad idea, it works here because it’s
initiated by user input, and the user can easily get it back.
Marginal Histogram 381
Much cleaner! An additional issue is that we’re obscuring the very colors we want to
highlight - let’s add some styles to our highlight bar to show the covered colors. We’ll
also want to prevent our bar from capturing mouse events, to keep it from blocking
mouse movement on our legend bar.
.legend-highlight-bar {
fill: rgba(255, 255, 255, 0.3);
stroke: white;
stroke-width: 2px;
pointer-events: none;
}
Let’s center our text, shrink the size, and make each character the same width (to
prevent the letters from jumping around as we hover months with characters of
different widths).
Marginal Histogram 382
.legend-highlight-text {
text-anchor: middle;
font-size: 0.8em;
font-feature-settings: 'tnum' 1;
}
That feels much better when we move our mouse around the legend bar.
Now let’s update our scatter plot! First, we’ll want to dim and shrink all of our dots
— let’s animate this change, but keep the duration short to keep from making the
interaction feel laggy.
dots.transition().duration(100)
.style("opacity", 0.08)
.attr("r", 2)
Now all of our dots almost disappear when we hover our legend, perfect!
Marginal Histogram 383
Next, we want to update the days within our highlighted range. First, we’ll create
an isDayWithinRange() function that converts a single data point into a date and
returns whether or not it’s in-between our min and max highlighted dates.
There are two special edge cases we want to handle — when we hover a date near
the left edge, we want to highlight nearby dates on the right edge, and vice versa.
Marginal Histogram 384
code/10-marginal-histogram/completed/scatter.js
code/10-marginal-histogram/completed/scatter.js
405 const relevantDots = dots.filter(isDayWithinRange)
406 .transition().duration(100)
407 .style("opacity", 1)
408 .attr("r", 5)
Now when we move our mouse over the legend, we can see that we’re highlighting
a groups of dots that corresponds to our hover position.
Our dots stay highlighted, even when we move our mouse away from our legend.
Let’s update our empty onLegendMouseLeave() function to fade all of our dots back
Marginal Histogram 386
function onLegendMouseLeave() {
dots.transition().duration(500)
.style("opacity", 1)
.attr("r", 4)
legendValues.style("opacity", 1)
legendValueTicks.style("opacity", 1)
legendHighlightGroup.style("opacity", 0)
}
Note that we don’t need to hide these <path>s off the bat because they won’t be
visible without a d attribute string.
Marginal Histogram 387
Now let’s add code to update these static elements right before the end of our
onLegendMouseMove() function.
hoverTopHistogram.attr("d", d => (
topHistogramLineGenerator(topHistogramGenerator(hoveredDates))
))
Boom! Now we can see a smaller dark histogram overlaid over our top histogram,
and it updates when we move our mouse over the legend.
Marginal Histogram 388
Let’s update the styles a bit — we’ll set the fill to use our hovered date’s color
to reinforce the relationship between the legend, the highlighted dots, and the mini
histogram. We’ll also add a white stroke (to m ake the distinction between the two
histograms a bit more clear), and make our path visible.
.attr("fill", colorScale(hoveredDate))
.attr("stroke", "white")
.style("opacity", 1)
Looking great! It’s wonderful how much we can do with a few lines of code, once
most of our chart is already set up.
Marginal Histogram 389
hoverRightHistogram.attr("d", d => (
rightHistogramLineGenerator(rightHistogramGenerator(hoveredDates))
))
.attr("fill", colorScale(hoveredDate))
.attr("stroke", "white")
.style("opacity", 1)
Now we can see both histograms update when we move over our legend.
Marginal Histogram 390
Our last step is to hide our histograms when our mouse leaves the legend. We’ll set
the opacity of our highlight histograms at the end of our onLegendMouseLeave()
function.
hoverTopHistogram.style("opacity", 0)
hoverRightHistogram.style("opacity", 0)
And we’re finished! Take a minute to play around with the chart and bask in the
glory of your great accomplishment! This chart was no easy feat, but you powered
through.
Hopefully you can get a sense of how the interactions add to the effectiveness of this
Marginal Histogram 391
chart. There are many insights to be found from exploring this chart. For example, we
can see that mid-May to mid-June has some of the hottest temperatures, but the days’
minimum temperatures aren’t so hot. Most of the days with the highest minimum
temperature are actually in August.
If you have your own data, look around for interesting insights and share them to
show off your handiwork!
Radar Weather Chart
We talked about radar charts in Chapter 8. For this project, we’ll build a more
complex radar chart.
This chart will give the viewer a sense of overall weather for the whole year, and
will highlight trends such as:
• What time of the year is cloudiest, and does that correlate with the coldest days?
• Do cloudy days have lower UV indices, or are they rainier?
• How does weather vary per season?
Radar Weather Chart 394
The complexity of this chart will increase the amount of time the viewer needs to
wrap their head around it. But once the viewer understands it, they can answer more
nuanced questions than a simpler chart.
We’ll reinforce concepts we’ve already learned as well as new concepts, such as
angular math, as we build this visualization. Let’s get started!
Getting set up
Once your server is running (live-server), find the page we’ll be working on
at http://localhost:8080/11-radar-weather-chart/draft/⁶⁶, and open the code folder at
/code/11-radar-weather-chart/draft/. Feel free to update the chart title in
index.html if you’re using your own data.
⁶⁶http://localhost:8080/11-radar-weather-chart/draft/
Radar Weather Chart 395
Start
We’ll run through this example fairly quickly, so if you need a reference, the
completed chart is in the /code/11-radar-weather-chart/completed/ folder.
Our dataset
Since we already know what our final chart will look like, we can pull out all of the
metrics we’ll need. Let’s create an accessor for each of the metrics we’ll plot (min
temperature, max temperature, precipitation, cloud cover, uv, and date).
code/11-radar-weather-chart/completed/chart.js
That’s a mouthful! Thankfully, we got that out of the way and won’t need to look at
the exact structure of our data again.
73 // 4. Create scales
74
75 const angleScale = d3.scaleTime()
76 .domain(d3.extent(dataset, dateAccessor))
77 .range([0, Math.PI * 2]) // this is in radians
Note that we’re using radians, instead of degrees. Angular math is generally easier
with radians, and we’ll want to use Math.sin() and Math.cos() later, which
deals with radians. There are 2π radians in a full circle. If you want to know more
about radians, the Wikipedia entry is a good source.
https://en.wikipedia.org/wiki/Radian
Adding gridlines
To get our feet wet with this angular math, we’ll draw our peripherals before we
draw our data elements. Let’s switch those steps in our code.
Radar Weather Chart 398
// 6. Draw peripherals
// 5. Draw data
If your first thought was “but the checklist!”, here’s a reminder that our chart drawing
checklist is here as a friendly guide, and we can switch the order of steps if we need
to.
Drawing the grid lines (peripheral) first is helpful in cases like this where we want
our data elements to layer on top. If we wanted to keep our steps in order, we could
also create a <g> element first to add our grid lines to after.
Creating a group to hold our grid elements is also a good idea to keep our elements
organized – let’s do that now.
code/11-radar-weather-chart/completed/chart.js
Notice that some of the options in this list are for d3-time intervals (highlighted in
cyan), but most are inherited from prototypical Javascript objects.
For example, we could use the .floor() method to get the first “time” in the current
month:
d3.timeMonth.floor(new Date())
Radar Weather Chart 400
d3.timeMonth.floor() example
d3 time intervals also have a .range() method that will return a list of datetime
objects, spaced by the specified interval, between two dates, passed as parameters.
Let’s try it out by creating our list of months!
months array
Radar Weather Chart 401
d3-time gives us shortcut aliases that can make our code even more concise – we
can use d3.timeMonth()⁶⁸ instead of d3.timeMonth.range().
code/11-radar-weather-chart/completed/chart.js
Let’s use our array of months and draw one <line> per month.
We’ll need to find the angle for each month – let’s use our angleScale to convert
the date into an angle.
return peripherals.append("line")
})
Each spoke will start in the middle of our chart – we could start those lines at
[dimensions.boundedRadius, dimensions.boundedRadius], but most of our
element will need to be shifted in respect to the center of our chart.
Remember how we use our bounds to shift our chart according to our top and left
margins?
⁶⁸https://github.com/d3/d3-time#timeMonths
Radar Weather Chart 402
To make our math simpler, let’s instead shift our bounds to start in the center of our
chart.
Radar Weather Chart 403
This will help us when we decide where to place our data and peripheral elements –
we’ll only need to know where they lie in respect to the center of our circle.
code/11-radar-weather-chart/completed/chart.js
51 const bounds = wrapper.append("g")
52 .style("transform", `translate(${
53 dimensions.margin.left + dimensions.boundedRadius
54 }px, ${
55 dimensions.margin.top + dimensions.boundedRadius
56 }px)`)
We’ll need to convert from angle to [x, y] coordinate many times in this chart.
Let’s create a function that makes that conversion for us. Our function will take two
parameters:
Radar Weather Chart 404
1. the angle
2. the offset
and return the [x,y] coordinates of a point rotated angle radians around the center,
and offset time our circle’s radius (dimensions.boundedRadius). This will give us
the ability to draw elements at different radii (for example, to draw our precipitation
bubbles slightly outside of our temperature chart, we’ll offset them by 1.14 times our
normal radius length).
To convert an angle into a coordinate, we’ll dig into our knowledge of trigonometry⁶⁹.
Let’s look at the right-angle triangle (a triangle with a 90-degree angle) created by
connecting our origin point ([0,0]) and our destination point ([x,y]).
The numbers we already know are theta (θ) and the hypotenuse (dimensions.boundedRadius
* offset). We can use these numbers to calculate the lengths of the adjacent and
⁶⁹https://www.mathsisfun.com/algebra/trigonometry.html
Radar Weather Chart 405
opposite sides of our triangle, which will correspond to the x and y position of our
destination point.
Because our triangle has a right angle, we can multiply the sine and cosine of our
angle by the length of our hypotenuse to calculate our x and y values (remember
the soh cah toa mnenomic⁷⁰?).
Sohcahtoa
This looks great! But we have to make one more tweak to our getCoordinatesForAngle()
function – an angle of 0 would draw a line horizontally to the right of the origin
⁷⁰http://mathworld.wolfram.com/SOHCAHTOA.html
Radar Weather Chart 406
point. But our radar chart starts in the center, above our origin point. Let’s rotate our
angles by 1/4 turn to return the correct points.
π / 2 radians rotation
Remember that there are 2π radians in one full circle, so 1/4 turn would be 2π / 4,
or π / 2.
Radar Weather Chart 407
code/11-radar-weather-chart/completed/chart.js
Whew! That was a a lot of math. Now let’s use it to draw our grid lines.
If we move back down in our chart.js file, let’s grab the x and y coordinates of the
end of our spokes and set our <line>s’ x2 and y2 attributes.
We don’t need to set the x1 or y1 attributes of our line because they both default
to 0.
months.forEach(month => {
const angle = angleScale(month)
const [x, y] = getCoordinatesForAngle(angle)
peripherals.append("line")
.attr("x2", x)
.attr("y2", y)
.attr("class", "grid-line")
})
Hmm, we can’t see anything yet. Let’s give our lines a stroke color in our
styles.css file.
.grid-line {
stroke: #dadadd;
}
Finally! We have 12 spokes to show where each of the months in our chart start.
Radar Weather Chart 408
Your spokes might be rotated a bit, depending on when your dataset starts.
months.forEach(month => {
const angle = angleScale(month)
const [x, y] = getCoordinatesForAngle(angle)
peripherals.append("line")
.attr("x2", x)
.attr("y2", y)
.attr("class", "grid-line")
We can see our month labels now, but there’s one issue: the labels on the left are
closer to our spokes than the labels on the right.
Radar Weather Chart 410
This is because our <text> elements are anchored by their left side. Let’s dynamically
set their text-anchor property, depending on the label’s x position. We’ll align
labels on the left by the end of the text, and labels near the center by their middle.
Note that text-anchor is essentially the text-align CSS property for SVG
elements.
Radar Weather Chart 411
.text(d3.timeFormat("%b")(month))
.style("text-anchor",
Math.abs(labelX) < 5 ? "middle" :
labelX > 0 ? "start" :
"end"
)
Our labels also aren’t centered vertically with our spokes. Let’s center them, using
dominant-baseline, and update their styling to decrease their visual weight. We
want our labels to orient our users, but not to distract from our data.
Radar Weather Chart 412
.tick-label {
dominant-baseline: middle;
fill: #8395a7;
font-size: 0.7em;
font-weight: 900;
letter-spacing: 0.005em;
}
Higher temperatures are drawn further from the center of our chart.
Let’s add a radiusScale at the end of our Create scales section. We’ll want to use
nice() to give us friendlier minimum and maximum values, since the exact start
and end doesn’t matter. Note that we didn’t use .nice() to round the edges of our
angleScale, since we want it to start and end exactly with its range.
code/11-radar-weather-chart/completed/chart.js
We’re using the ES6 spread operator (...) to spread our arrays of min and max
temperatures so we get one flat array with both arrays concatenated. If you’re
unfamiliar with this syntax, feel free to read more here.
https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Operators/Spread_syntax
We’ll be converting a single data point into an x or y value many times – let’s create
two utility functions to help us to do just that. It seems simple enough now, but it’s
nice to have this logic in one place and not cluttering our .attr() functions.
Radar Weather Chart 414
code/11-radar-weather-chart/completed/chart.js
Let’s put this scale to use! At the end of our Draw peripherals step, let’s add a few
circle grid lines that correspond to temperatures within our radiusScale.
code/11-radar-weather-chart/completed/chart.js
Since <circle> elements default to a black fill, we won’t be able to see much yet.
Radar Weather Chart 415
Let’s add some styles to remove the fill from any .grid-line elements and add a
faint stroke.
.grid-line {
fill: none;
stroke: #dadadd;
}
Similar to our month lines, we’ll need labels to tell our viewers what temperature
each of these circles represents.
code/11-radar-weather-chart/completed/chart.js
Notice that we’re returning early if d is falsey – we don’t want to make a label for
a temperature of 0.
We’ll need to vertically center and dim our labels – let’s update our
.tick-label-temperature elements in our styles.css file.
.tick-label-temperature {
fill: #8395a7;
opacity: 0.7;
font-size: 0.7em;
dominant-baseline: middle;
}
These labels are very helpful, but they’re a little hard to read on top of our grid lines.
Radar Weather Chart 418
Let’s add a <rect> behind our labels that’s the same color as the background of our
page. We’ll need to add this code before we draw our tickLabels and after our
gridCircles, since SVG stacks elements in the order we draw them.
code/11-radar-weather-chart/completed/chart.js
149 const tickLabelBackgrounds = temperatureTicks.map(d => {
150 if (!d) return
151 return peripherals.append("rect")
152 .attr("y", -radiusScale(d) - 10)
153 .attr("width", 40)
154 .attr("height", 20)
155 .attr("fill", "#f8f9fa")
156 })
Great! We’re all set with our grid marks and ready to draw some data!
Adding freezing
Let’s ease into drawing our data elements by drawing a <circle> to show where
freezing is on our chart. We’ll want to write this code in our Draw data step.
As usual, either draw what “feels like” freezing to you or skip this step if your
weather doesn’t drop below freezing.
code/11-radar-weather-chart/completed/chart.js
If our temperatures do drop below freezing, we’ll add a <circle> whose radius ends
at 32 degrees Fahrenheit.
if (containsFreezing) {
const freezingCircle = bounds.append("circle")
.attr("r", radiusScale(32))
.attr("class", "freezing-circle")
}
Let’s set the fill color and opacity of our circle to be a light cyan.
.freezing-circle {
fill: #00d2d3;
opacity: 0.15;
}
Great! Now we can see where the freezing temperatures will lie on our chart.
Radar Weather Chart 421
⁷¹https://github.com/d3/d3-shape#areaRadial
Radar Weather Chart 422
code/11-radar-weather-chart/completed/chart.js
Like .line() and .area() generators, our areaGenerator() will return the d
attribute string for a <path> element, given a dataset. Let’s create a <path> element
and set its d attribute.
Sometimes displaying a metric in multiple ways can help focus the viewer on it and
also give them two ways to encode it. Let’s also visualize the temperature with a
gradient.
Let’s create a gradient at the end of our Draw canvas step. Creating <defs> elements
near the top helps organize our code – we’ll know where to find elements that are
re-useable.
This will look similar to drawing our legend gradient in Chapter 6, but we’ll use a
<radialGradient> instead of a <linearGradient>.
Radar Weather Chart 424
code/11-radar-weather-chart/completed/chart.js
Great! Now we can see that our gradient re-enforces the relationship between
distance from the origin and higher temperatures.
Radar Weather Chart 425
These kinds of decisions will come from your expertise as a subject matter expert.
When setting a threshold like this in your own charts, think about what might be
meaningful to the viewer.
Let’s keep our code organized and keep our UV index lines within one group.
Radar Weather Chart 426
code/11-radar-weather-chart/completed/chart.js
We want to draw our UV lines just inside the edges of our radius – let’s set their
offset to 0.95.
code/11-radar-weather-chart/completed/chart.js
Next, let’s draw one <line> per day over our threshold, drawing the outside edge
just outside of our chart’s radius.
code/11-radar-weather-chart/completed/chart.js
We won’t be able to see our <line>s until we give them a stroke – let’s add a stroke
color and width in our styles.css file.
.uv-line {
stroke: #feca57;
stroke-width: 2;
}
Now we can see that all of the days with high UV index are between April and
September, with the highest density around July.
Radar Weather Chart 427
The radius of each of our circles will depend on the amount of cloud cover.
One caveat with visualizing a linear scale with a circle’s size is that circles’ areas and
radii scale at different rates. Let’s take a circle with a radius of 100px as an example.
If we multiply its radius by 2, we’ll get a circle with a radius of 200. However, the
Radar Weather Chart 429
circle grows in every direction, making this larger circle cover four times as much
space.
Circle example
Instead, we’ll want a circle with a radius of 141 pixels to create a circle that is twice
as large as our original circle.
Radar Weather Chart 430
Since we, as humans, judge a circle by the amount of space it takes up, instead of
how wide it is, we need a way to size our circles by their area instead of their radii.
But <circle> elements are sized with their r attribute, so we need a way to scale
our radii so that our areas scale linearly.
The area of a circle is the radius multiplied by π, then squared. If we flip this equation
around, we’ll find that the radius of a circle is the square root of a circle’s area,
divided by π.
Radar Weather Chart 431
Since π is a constant, we can represent the relationship simply by using a square root
scale. How convenient!
Whenever we’re scaling a circle’s radius, we’ll want to use d3.scaleSqrt()⁷²
instead of d3.scaleLinear()⁷³ to keep the circles’ areas scaling proportionally.
Let’s create our cloud cover radius scale, making our circles’ radii range from 1 to
10 pixels.
We’ll write this code at the end of our Create scales step.
code/11-radar-weather-chart/completed/chart.js
At the end of our Draw data step, let’s create a new <g> to contain our cloud circles,
then declare their offset from the center of our chart.
⁷²https://github.com/d3/d3-scale#scaleSqrt
⁷³https://github.com/d3/d3-scale#scaleLinear
Radar Weather Chart 432
code/11-radar-weather-chart/completed/chart.js
Now we can draw one circle per day, setting each circle’s radius with our new
cloudRadiusScale.
code/11-radar-weather-chart/completed/chart.js
Great! Now we can see a ring of “clouds” around the outside of our chart.
Radar Weather Chart 433
Let’s set their fill color in our styles.css file, dimming them so they are a more
natural “cloud” color, and so they don’t visually dominate our chart.
.cloud-dot {
fill: #c8d6e5;
}
Radar Weather Chart 434
Let’s also make our cloud circles somewhat translucent, so that larger circles don’t
completely cover their smaller neighbors.
.cloud-dot {
fill: #c8d6e5;
opacity: 0.6;
}
Radar Weather Chart 435
code/11-radar-weather-chart/completed/chart.js
Next, we’ll list out the types of precipitation in our dataset, then create a color scale
mapping those types to different colors. We’ll want to use an ordinal scale, since this
is an ordinal metric, and can be placed in categories with a natural order (remember
the types of data we learned in Chapter 7?).
code/11-radar-weather-chart/completed/chart.js
Scrolling back down to the end of our Draw data step, we’ll draw our circles similarly
to how we drew our cloud circles. This time, we’ll use our precipitationTypeColorScale
to set each circle’s fill color.
code/11-radar-weather-chart/completed/chart.js
.precipitation-dot {
opacity: 0.5;
}
That’s better!
Radar Weather Chart 439
Adding annotations
Let’s take a step back and look at our charts through a new viewer’s eyes.
Radar Weather Chart 440
There’s a lot going on and not much explanation. A new viewer might wonder: what
does this blue dot represent? What are these yellow slashes?
Let’s add some annotations to help orient a new viewer. We have a lot of things that
need explanation, so let’s start by creating a function that will draw an annotation,
give three parameters:
We’ll want our <line> to have a light stroke and our <text> to be vertically centered
with the end of our <line>. Let’s add those styles to our styles.css file.
Radar Weather Chart 442
.annotation-line {
stroke: #34495e;
opacity: 0.4;
}
.annotation-text {
fill: #34495e;
font-size: 0.7em;
dominant-baseline: middle;
}
Going back to our chart.js file, let’s draw our first two annotations. To keep the
top of our chart as clean as possible, let’s create an annotation for the outer two data
elements: cloud and precipitation bubbles.
We’ll want to draw these annotations in the top right of our chart, to prevent from
stealing the show too early. If our annotations were in the top left, viewers might
read them first (since English text usually runs from left-to-right, top-to-bottom).
We’ll set the angle of these two annotations around π / 4, which is one-eight of a
turn around our chart.
And for our annotations’ offset, we can use the offsets we defined when we drew
each set of bubbles.
Wonderful! Our annotations fit in between two of our month labels, preventing any
overlap.
Radar Weather Chart 443
We’ll draw the rest of our annotations in the bottom right of our chart, making sure
to tell our viewers what the exact UV index threshold is.
Note that we had to convert our freezing point into a value relative to our bounded
radius, since our drawAnnotation() function takes an offset instead of a radius
value.
We could increase the size of our right margin, but that would un-center our
chart within our wrapper. Not a big deal, but let’s look at an alternative: prevent
overflowing svg elements from being clipped.
In our styles.css file, let’s change the overflow property of our svg from the
default of hidden.
Radar Weather Chart 445
svg {
overflow: visible;
}
Easy peasy! Now we can see the end of our annotations. Be careful, though, when
using this workaround in complicated pages – you don’t want your chart to run into
other elements on your page!
This looks great, but feel free to play around with the angle of your annotations.
Maybe you would group all of your annotation labels in the top right. Keep in mind
that irregular shapes waste more space in many page layouts.
If we again view our chart with a new viewer’s eyes, each part is way more clear!
We are missing one thing, though: the precipitation type colors are still un-labeled.
Let’s loop over each of our precipitation types, creating one <circle> to show the
Radar Weather Chart 446
Great! Now a new viewer is quickly oriented and can figure out what each data
element represents.
Radar Weather Chart 447
function onMouseMove(e) {
}
function onMouseLeave() {
}
Perfect, the black area covers exactly where we want any movement to trigger a
tooltip.
Radar Weather Chart 449
.listener-circle {
fill: transparent;
}
Next, we’ll need to create our tooltip element in our index.html file, with a spot for
each of our hovered over day’s metrics to be displayed.
Radar Weather Chart 450
Let’s also add our tooltip styles in our styles.css file, remembering to hide our
tooltip and to give our wrapper a position to create a new context.
Radar Weather Chart 451
.wrapper {
position: relative;
}
.tooltip {
opacity: 0;
position: absolute;
top: 0;
left: 0;
width: 15em;
padding: 0.6em 1em;
background: #fff;
text-align: center;
line-height: 1.4em;
font-size: 0.9em;
border: 1px solid #ddd;
z-index: 10;
pointer-events: none;
}
.tooltip-date {
margin-bottom: 0.2em;
font-weight: 600;
font-size: 1.1em;
line-height: 1.4em;
}
.tooltip-temperature {
font-feature-settings: 'tnum' 1;
}
.tooltip-metric {
display: flex;
justify-content: space-between;
width: 100%;
font-size: 0.8em;
Radar Weather Chart 452
line-height: 1.3em;
transition: all 0.1s ease-out;
}
.tooltip-metric div:first-child {
font-weight: 800;
padding-right: 1em;
}
.tooltip-metric div:nth-child(2) {
font-feature-settings: 'tnum' 1;
}
.tooltip-cloud {
color: #8395a7;
}
.tooltip-uv {
color: #feca57;
}
Switching back in our chart.js file, we’ll want to grab our tooltip element to
reference later, and also make a <path> element to highlight the hovered over day.
Now we can fill out our onMouseMove() function. Let’s start by grabbing the x and
y position of our cursor, using d3.mouse().
function onMouseMove(e) {
const [x, y] = d3.mouse(this)
// ...
We have our mouse position, but we need to know the angle from the chart origin.
Radar Weather Chart 453
How do we convert from an [x, y] position to an angle? We’ll need to use an inverse
trigonometric function: atan2. If you’re curious, read more about atan2 here⁷⁴.
code/11-radar-weather-chart/completed/chart.js
Remember that these trigometric functions originate around the horizontal, right
plane of our circle. Let’s rotate the resulting angle back one-quarter turn around the
circle to match our date scale.
To keep our angles positive, we’ll want to rotate any negative angles around our circle
by one full turn, so they fit on our angleScale.
We want to draw a line to highlight the date we’re hovering, but it needs to increase
in width as it gets further from the center of our circle. To create this shape, we’ll
use d3.arc()⁷⁵, which is the arc version of the line generators we’ve been using
(d3.line()). We can use the .innerRadius() and outerRadius() methods to
tell it how long we want our arc to be, and the .startAngle() and .endAngle()
methods to tell it how wide we want our arc to be.
code/11-radar-weather-chart/completed/chart.js
Now we can use our new arc generator to create the d attribute for our tooltip line.
⁷⁴https://en.wikipedia.org/wiki/Atan2
⁷⁵https://github.com/d3/d3-shape#arcs
Radar Weather Chart 454
tooltipLine.attr("d", tooltipArcGenerator())
.style("opacity", 1)
Perfect! Now we have a line that follows our cursor around the center of our circle.
Let’s lighten the line in our styles.css file to prevent it from covering the data we
want to highlight. We can use mix-blend-mode: multiple to make the covered
data elements stand out a little.
Radar Weather Chart 455
.tooltip-line {
fill: #8395a7;
fill-opacity: 0.2;
mix-blend-mode: multiply;
pointer-events: none;
}
Much better!
Next, we’ll want to position our tooltip at the end of our line. First, we’ll grab the
[x, y] coordinates of this point.
Radar Weather Chart 456
code/11-radar-weather-chart/completed/chart.js
Using these coordinates, we’ll set the transform CSS property of our tooltip. We
have some fancy math here, using the CSS calc() function to choose which side of
our tooltip to anchor to the outerCoordinate, based on where we are around the
circle. We don’t want our tooltip to cover our chart!
Try to work through each line to figure out what is going on, and inspect the tooltip
in the Elements tab of your dev tools to see the resulting transform value.
tooltip.style("opacity", 1)
.style("transform", `translate(calc(${
outerCoordinates[0] < - 50 ? "40px - 100" :
outerCoordinates[0] > 50 ? "-40px + 0" :
"-50"
}% + ${
outerCoordinates[0]
+ dimensions.margin.top
+ dimensions.boundedRadius
}px), calc(${
outerCoordinates[1] < - 50 ? "40px - 100" :
outerCoordinates[1] > 50 ? "-40px + 0" :
"-50"
}% + ${
outerCoordinates[1]
+ dimensions.margin.top
+ dimensions.boundedRadius
}px))`)
Wonderful! Now our tooltip follows the end of our tooltip line when we move our
mouse around our chart.
Radar Weather Chart 457
Next, we need to update the text of our tooltip to show information about the date
we’re hovering over. We can use the .invert() method of our angleScale() to
convert backwards, from its range dimension (angle) to its domain dimension (date).
If we format this date similarly to the dates in our dataset, we can look for a data
point with the same date string.
If no such date exists, we should exit this function early. This should never happen,
but is possible with a dataset that skips dates.
Radar Weather Chart 458
if (!dataPoint) return
Now that we have the data for the date we’re hovering over, we can populate the
text of our tooltip.
tooltip.select("#tooltip-temperature-min")
.html(`${d3.format(".1f")(temperatureMinAccessor(dataPoint))}°F`)
tooltip.select("#tooltip-temperature-max")
.html(`${d3.format(".1f")(temperatureMaxAccessor(dataPoint))}°F`)
tooltip.select("#tooltip-uv")
.text(uvAccessor(dataPoint))
tooltip.select("#tooltip-cloud")
.text(cloudAccessor(dataPoint))
tooltip.select("#tooltip-precipitation")
.text(d3.format(".0%")(precipitationProbabilityAccessor(dataPoint)))
tooltip.select("#tooltip-precipitation-type")
.text(precipitationTypeAccessor(dataPoint))
tooltip.select(".tooltip-precipitation-type")
.style("color", precipitationTypeAccessor(dataPoint)
? precipitationTypeColorScale(precipitationTypeAccessor(dataPoint))
: "#dadadd")
Notice that we’re also setting the color of our precipitation type label and value,
re-enforcing the relationship between the precipitation type and its color.
Radar Weather Chart 459
Let’s take this one step further! We’re using a gradient of colors to show what
temperatures each day spans. At the end of our Create scales step, let’s create a
new scale that maps temperatures to the gradient scale we’re using.
code/11-radar-weather-chart/completed/chart.js
We’re using a sequential scale here instead of a linear scale because we want to use
one of d3’s built-in color scales (d3.interpolateYlOrRd) as an .interpolator()
instead of specifying a range.
https://github.com/d3/d3-scale#sequential-scales
Radar Weather Chart 460
Now we can use this scale to color the minimum and maximum temperatures for
our hovered date.
tooltip.select("#tooltip-temperature-min")
.style("color", temperatureColorScale(
temperatureMinAccessor(dataPoint)
))
tooltip.select("#tooltip-temperature-max")
.style("color", temperatureColorScale(
temperatureMaxAccessor(dataPoint)
))
And voila! Our tooltip is chock full of helpful information, and it also helps re-enforce
some of the data visualization in our main chart.
Wrapping up
Give yourself a pat on the back, this one was a doozy! Take a step back and look at the
visualization we’ve created. The viewer gets a good sense of the annual weather, and
has the ability to explore further, but isn’t instantly overwhelmed with information.
Radar Weather Chart 461
Finished chart
Make sure to show off your chart with friends and on social media! It will be
interesting to compare these charts with weather data from different places.
Hopefully this project gave you a better idea of the process involved in making a
more complicated chart. Often, we have to research new concepts and learn new
parts of the d3.js api when making a chart – if we’re making a circular chart for the
first time, we might need to refresh our trigonometry knowledge and look up ways
to draw, for example, arcs.
Animated Sankey Diagram
In this project, we’ll be simulating real data and creating an animated diagram to
engage our viewers. We won’t be using our beloved weather data for this! We’ll be
using outcomes of a study of 10-year educational achievement for high schoolers in
the United States, based on sex and socioeconomic status. For example, 55.3% of the
males who grew up with a high socioeconomic status completed at least a Bachelor’s
degree.
We could make a static data visualization that shows these different probabilities,
but our goal here is to engage the viewer. By simulating data and animating it, the
viewer gets to “play along” and guess at the outcome before seeing it play out before
their eyes.
Finished chart
We’ll be mimicking the shape of a Sankey diagram⁷⁶, which shows flows based on
the width of paths between points.
⁷⁶https://en.wikipedia.org/wiki/Sankey_diagram
Animated Sankey Diagram 463
Our paths will all be the same width, but using the shape of a Sankey diagram helps
us communicate that we’re showing a flow between states.
Getting set up
With your server running (live-server), check out the completed chart at
http://localhost:8080/12-animated-sankey/completed/⁷⁷
Our plan is to simulate fake “people” and animate their journey from beginning
(socioeconomic status) to end (educational attainment). We’ll represent them with
a triangle (for males) or circle (for females), creating new people continuously.
Now that we know what our goal is, let’s open up our draft page at http://localhost:8080/12-
animated-sankey/draft/⁷⁸ and open the code folder at /code/12-animated-sankey/draft/.
Our data
Each object represents a possible starting point, specified by a sex and a socioeco-
nomic status (ses). The object also has the percent of people in that starting group
that attained (at most) various degrees.
If you’re curious and want more information about our dataset, follow the links
at the bottom of our chart page. They’re obtained from a longitudinal study by the
the U.S. Department of Education, studying sophomores in high school (2002) until
2012.
https://nces.ed.gov/programs/digest/d14/tables/dt14_104.91.asp
{
sex: 0,
ses: 0,
education: 0,
}
We could use strings for our values (eg. sex: "female"), but there are a two main
benefits to using indices instead.
code/12-animated-sankey/completed/chart.js
7 const sexAccessor = d => d.sex
We’ll also want a list of possible sexes, which gives us a list to sample from when
we generate a person.
code/12-animated-sankey/completed/chart.js
8 const sexes = ["female", "male"]
Lastly, we’ll want a list of ids that correspond to these options. In this case, we’ll
want the array [0, 1] that will correspond to the ["female", "male"] array,
giving females an index of 0 and males an index of 1.
We could hardcode this array, or we could use d3.range()⁷⁹ to generate an array of
indices, from 0 to the number of options.
code/12-animated-sankey/completed/chart.js
9 const sexIds = d3.range(sexes.length)
⁷⁹https://github.com/d3/d3-array#range
Animated Sankey Diagram 467
We could create a list of unique education values from our dataset, but hardcoding
our educationNames array gives us the chance to set the order of values.
Stacking probabilities
Since we’ll be creating several people per second, we’ll want our generatePerson()
function to be as simple as possible. Given their sex and socioeconomic status, we
currently know the probability of a person to fall in one of our education “buckets”.
Since these probabilities sum up to 100%, we can stack these probabilities and use a
random number to assign a person to a bucket. Let’s take the probabilities for females
who grew up in a low income household as an example.
Animated Sankey Diagram 468
Probabilities diagram
We can take these probabilities and stack them on top of each other, so that each
level instead represents the probability that a person achieves that level or lower.
The highest level (Bachelor’s and up), will get the number 1 because there is a 100%
chance that a person achieved that level or lower.
Probabilities diagram
When we have these stacked probabilities, we’ll be able to choose a random number
between 0 and 1, which we’ll locate on our number-line. For example, if we choose
the number 0.32, this person will be placed in the Some Post-secondary education
Animated Sankey Diagram 469
bucket.
Probabilities diagram
To start, we want a consistent way to access the correct set of probabilities, using a
“status key” string. Given this object:
Animated Sankey Diagram 470
{
sex: "female",
ses: "low",
}
we want to generate the string "female--low". Let’s create a function to do that for
us.
code/12-animated-sankey/completed/chart.js
const stackedProbabilities = {}
dataset.forEach(startingPoint => {
const key = getStatusKey(startingPoint)
let stackedProbability = 0
// stackedProbabilities[key] = ...
})
Next, we’ll loop over each of the education buckets (in order), adding the current
probability to the stacked probability, then returning the current sum.
const stackedProbabilities = {}
dataset.forEach(startingPoint => {
const key = getStatusKey(startingPoint)
let stackedProbability = 0
stackedProbabilities[key] = educationNames.map((education, i) => {
stackedProbability += (startingPoint[education] / 100)
return stackedProbability
})
})
Animated Sankey Diagram 471
We’ll have to add an additional check – if we’re looking at the last education bucket,
we’ll return 1 instead of our running sum. This will help account for rounding errors,
where the sum is 0.99 and doesn’t completely add up to 1.
code/12-animated-sankey/completed/chart.js
28 const stackedProbabilities = {}
29 dataset.forEach(startingPoint => {
30 const key = getStatusKey(startingPoint)
31 let stackedProbability = 0
32 stackedProbabilities[key] = educationNames.map((education, i) => {
33 stackedProbability += (startingPoint[education] / 100)
34 if (i == educationNames.length - 1) {
35 // account for rounding error
36 return 1
37 } else {
38 return stackedProbability
39 }
40 })
41 })
Generating a person
Let’s put our stacked probabilities to use and create a generatePerson() function.
We want this function to return an object with a sex, ses, and education.
If we peek at the bottom of our chart.js file, we’ll see a few utility functions
that will help us with some dirty work. For example, there is a getRandomValue()
function that takes an array of values and returns a random value. Let’s use this
function to choose a sex and ses for our person.
function generatePerson() {
const sex = getRandomValue(sexIds)
const ses = getRandomValue(sesIds)
return {
sex,
ses,
education: "?",
}
}
Animated Sankey Diagram 473
If you’re confused about the {sex} syntax, we’re using ES6 Object shorthand
notation. {sex} is the same thing as {sex: sex}. If you want to read more, here’s
a great source.
https://tylermcginnis.com/shorthand-properties/
But how do we choose our person’s education? First, we’ll generate a statusKey
and grab the matching stacked probabilities.
function generatePerson() {
const sex = getRandomValue(sexIds)
const ses = getRandomValue(sesIds)
const statusKey = getStatusKey({
sex: sexes[sex],
ses: sesNames[ses],
})
const probabilities = stackedProbabilities[statusKey]
return {
sex,
ses,
education: "?",
}
}
function generatePerson() {
const sex = getRandomValue(sexIds)
const ses = getRandomValue(sesIds)
const statusKey = getStatusKey({
sex: sexes[sex],
ses: sesNames[ses],
})
const probabilities = stackedProbabilities[statusKey]
const education = d3.bisect(probabilities, Math.random())
return {
sex,
ses,
education,
}
}
console.log(generatePerson())
console.log(generatePerson())
console.log(generatePerson())
Test people
Great! Now we can generate people who follow the rules of our dataset.
good height that will fit a lot of people at once, but not push lower categories off the
page.
let dimensions = {
width: width,
height: 500,
margin: {
top: 10,
right: 200,
bottom: 10,
left: 120,
},
pathHeight: 50,
}
We have three starting y-positions and six ending y-positions – our goal is to connect
each of these with a curved line that our markers will follow.
Paths
Animated Sankey Diagram 476
X scale
We’ll need to create some scales first in our Create scales step.
Let’s start by creating an x scale that converts a person’s progress (from left to right)
into an x-position. We’ll represent this progress with a number from 0 (not started
yet) to 1 (has reached the right side).
code/12-animated-sankey/completed/chart.js
We don’t want our markers moving beyond the left or right side of our paths, so we’ll
.clamp() this scale. This way, the smallest number our scale will return is 0 and the
largest number is 1, even if we give it a progress of 10.
code/12-animated-sankey/completed/chart.js
Y scales
Next, we’ll create a scale that will convert from a socioeconomic id to a y-position.
We want these paths to be evenly spaced between our bounds, but to still fit inside.
We can achieve this by padding our scale’s domain, making it span [-1, 3] instead
of [0, 2]. This way, our real socioeconomic status ids will fit inside our bounds.
Animated Sankey Diagram 477
y scale diagram
We’ll also want our scale’s domain to be backwards: [3, -1] instead of [-1, 3]
because we want the highest y position (closer to the bottom) to correspond to
the lowest id.
code/12-animated-sankey/completed/chart.js
Let’s create a similar scale for our educationIds on the right side of our chart.
Animated Sankey Diagram 478
code/12-animated-sankey/completed/chart.js
Example path
between each of our starting points and each of our ending points.
Since these shapes aren’t linear, we’ll draw them using <path>s by using d3.line()
to create a d string attribute generator. Our line generator will take an array of six
identical arrays. For example, we might pass it this input:
[
[0, 5],
[0, 5],
[0, 5],
[0, 5],
[0, 5],
[0, 5],
]
The first item in each of these arrays (0) is the socioeconomic status id (starting
point) and the second item (5) is the education id (ending point).
Our link generator will return the starting y position for the first 3 arrays, and the
ending position for the last 3 arrays.
Animated Sankey Diagram 480
The reason we want to repeat this array 6 times is to devote one-fifth of our horizontal
space to the y-position transition. If we only had four identical arrays, we would be
devoting half of our horizontal space to the transition, which would make our final
chart way more chaotic. Our markers would spend one third of their time moving
up and down!
Let’s create that line generator:
code/12-animated-sankey/completed/chart.js
133 const linkLineGenerator = d3.line()
134 .x((d, i) => i * (dimensions.boundedWidth / 5))
135 .y((d, i) => i <= 2
136 ? startYScale(d[0])
137 : endYScale(d[1])
138 )
We’ll need to create that six-item array for each permutation of starting and ending
ids. We’ll map over each of our starting ids and also each of our ending ids, creating
that six-item array for each loop. We’ll pass the result to d3.merge(), which will
flatten these into one array.
Animated Sankey Diagram 481
code/12-animated-sankey/completed/chart.js
lineOptions structure
Great! Now we just need to create a <path> for each of our lineOptions and
generate their d attribute strings using our line generator.
Animated Sankey Diagram 483
code/12-animated-sankey/completed/chart.js
Our paths
We’ll need to set the fill and stroke colors of our paths in the styles.css file.
.category-path {
fill: none;
stroke: white;
}
Let’s add an interpolator function to our line generator to smooth our paths.
code/12-animated-sankey/completed/chart.js
There we go!
Animated Sankey Diagram 485
Start labels
First, we’ll label each possible start, positioning our labels within a group that is
20 pixels left of the beginning of our bounds. Let’s first create a <g> element that
positions our labels’ x-positions.
code/12-animated-sankey/completed/chart.js
157 const startingLabelsGroup = bounds.append("g")
158 .style("transform", "translateX(-20px)")
Then we’ll create each individual label, using the utility function sentenceCase(),
defined at the bottom of our file, to format our socioeconomic labels.
Animated Sankey Diagram 486
code/12-animated-sankey/completed/chart.js
Let’s align them to the right of our group, using the text-anchor CSS property
in our styles.css file. We’ll also vertically center them with our paths using the
dominant-baseline CSS property.
.start-label {
text-anchor: end;
dominant-baseline: middle;
}
Starting labels
Animated Sankey Diagram 487
These labels aren’t very descriptive, though. A viewer isn’t likely to know what High
means without any context. Let’s add a title above our start labels.
There’s one issue though: the phrase Socioeconomic status is pretty long and won’t
easily fit where we want it to. SVG <text> elements don’t wrap to multiple lines the
way HTML text does. We’ll need to create two <text> elements, positioned one on
top of the other.
code/12-animated-sankey/completed/chart.js
Let’s add a few styles to align our title with our start titles, and dim the opacity to
make it visually distinct.
.start-title {
text-anchor: end;
font-size: 0.8em;
opacity: 0.6;
}
Great! Now viewers can see which each starting position means.
Animated Sankey Diagram 488
End labels
For our ending labels, we’ll want to label the paths, but also show how many “people”
of each socioeconomic status and gender ended up in each path. Let’s start with the
labels, but leave room underneath to show those numbers.
code/12-animated-sankey/completed/chart.js
Notice that we used the "label" class name on both our start and end labels –
let’s give these a little more visual weight since they’re more important than other
annotations like the title of our starting labels and the values we’ll add to our ending
labels.
.label {
font-weight: 600;
dominant-baseline: middle;
}
Ending labels
We’ll show the amount of people who made it into each ending position by showing
males in a row on the top and females in a row on the bottom. We’ll be using color
to signify socioeconomic status.
Let’s start by adding a <circle> underneath each ending position label, we’ll add
the values when we start animating our “people”. If we refer to our SVG Cheatsheet
(in Appendix C), we’ll see that <circle>s are positioned with a cx and cy attribute.
The c declares that these coordinates refer to the center of the circle, instead of the
top right.
code/12-animated-sankey/completed/chart.js
197 const maleMarkers = endingLabelsGroup.selectAll(".male-marker")
198 .data(educationIds)
199 .enter().append("circle")
200 .attr("class", "ending-marker male-marker")
201 .attr("r", 5.5)
202 .attr("cx", 5)
203 .attr("cy", d => endYScale(d) + 5)
Perfect!
Animated Sankey Diagram 491
Next, we’ll want to draw triangles to signify our second row of female counts.
However, there is no <triangle> SVG element – we’ll need to build one ourselves.
Since a triangle is a fairly basic shape, let’s define the points of a <polygon>
element. We’ll make our triangle almost equilateral, but the height will be a tiny
bit shorter than the width, to give it a friendlier appearance.
Animated Sankey Diagram 492
Triangle diagram
Let’s start with the bottom, left point and work around our triangle in a clockwise
fashion.
code/12-animated-sankey/completed/chart.js
205 const trianglePoints = [
206 "-7, 6",
207 " 0, -6",
208 " 7, 6",
209 ].join(" ")
The .join() method will combine our separate coordinates into one string:
"-7, 6 0, -6 7, 6"
We could have written the string directly, but creating an array and .join()ing the
points is easier to parse and modify, if needed.
Animated Sankey Diagram 493
Let’s draw our polygon markers – our SVG Cheatsheet tells us that there are no
position attributes for <polygon>s, so we’ll use transform instead.
code/12-animated-sankey/completed/chart.js
Looking good!
Ending markers
Those markers distract a bit from our ending labels – let’s dim them in our
styles.css file.
Animated Sankey Diagram 494
.ending-marker {
opacity: 0.6;
}
Perfect!
We’re all set with labels for now – viewers will now be able to tell what each path
signifies.
Drawing people
We’re all ready to dig in and start drawing our “people”. In out Set up interactions
step, let’s begin by initializing two variables:
1. our people list that will hold all of our simulated people
2. our <g> element that will hold all of our people markers
Animated Sankey Diagram 495
let people = []
const markersGroup = bounds.append("g")
.attr("class", "markers-group")
Next, we need to make a function called updateMarkers() that will draw our
people. We’ll use d3.timer()⁸⁰ to update the position of our people.
d3.timer()’s first parameter is a callback function that it will call until the timer is
stopped (which we can do the timer’s .stop() method). This callback function will
have access to one parameter: how many milliseconds have elapsed since the timer
started.
Let’s create our updateMarkers() function and pass it to d3.timer() to call.
To check what d3.timer() is handing updateMarkers(), let’s log out the first
parameter.
function updateMarkers(elapsed) {
console.log(elapsed)
}
d3.timer(updateMarkers)
If we look in our Dev Tools Console, we’ll see that we’re rapidly logging out the
milliseconds elapsed.
⁸⁰https://github.com/d3/d3-timer#timer
Animated Sankey Diagram 496
Perfect! Instead of logging the elapsed, let’s add a new person to our people array
every time updateMarkers() runs.
function updateMarkers(elapsed) {
people = [
...people,
generatePerson(),
]
console.log(people)
}
Wonderful – we can see that our array contains a new person on every iteration.
Animated Sankey Diagram 497
Starting with females, let’s draw a <circle> for every person in our array with a
sex of 0. We can isolate these people by .filter()ing our people array.
females.enter().append("circle")
.attr("class", "marker marker-circle")
.attr("r", 5.5)
Since we’re not positioning our <circle>s, they show up in the top, left of our
bounds.
Animated Sankey Diagram 498
Let’s draw our male markers, then update all of our markers’ positions at once.
markers.style("transform", d => {
const x = -10
const y = startYScale(sesAccessor(d))
return `translate(${ x }px, ${ y }px)`
})
Great, that’s exactly where we want our markers to start – at the beginning of our
paths.
We want our chart to be animated, though! Let’s add the elapsed value to our
markers’ x-position.
markers.style("transform", d => {
const x = elapsed
// ...
There we go! Our markers skitter off to the right. But we don’t want to draw all of
Animated Sankey Diagram 500
our people on top of each other – we want each person to start at the left when they
are created.
Let’s pass our elapsed milliseconds to each person as they are created.
people = [
...people,
generatePerson(elapsed),
]
Then we can update our generatePerson() function to record when that person
was created.
function generatePerson(elapsed) {
// ...
return {
sex,
ses,
education,
startTime: elapsed,
}
}
And now, when we draw each person, we’ll subtract their startTime from their
x-position.
markers.style("transform", d => {
const x = elapsed - d.startTime
// ...
Nice! Now we can see a steady stream of markers making their way across the page.
Animated Sankey Diagram 501
yTransitionProgressScale diagram
We’ll clamp our scale so that the lowest value it returns is 0 and the highest is 1.
code/12-animated-sankey/completed/chart.js
Let’s move back down to our updateMarkers() function. We’ll need an easy way
to get a person’s x-progress along their path. Let’s create an xProgressAccessor
that assumes that a person takes 5 seconds (5000 milliseconds) to cross the chart.
Animated Sankey Diagram 503
function updateMarkers(elapsed) {
const xProgressAccessor = d => (elapsed - d.startTime) / 5000
// ...
Now let’s update our function where we set our markers’ positions – we’ll find each
person’s x-progress and pass it to our xScale to get their correct x position. We’ll
then find their starting and ending y positions, and update their y-position based on
their progress along the route.
markers.style("transform", d => {
const x = xScale(xProgressAccessor(d))
const yStart = startYScale(sesAccessor(d))
const yEnd = endYScale(educationAccessor(d))
const yChange = yEnd - yStart
const yProgress = yTransitionProgressScale(xProgressAccessor(d))
const y = yStart
+ (yChange * yProgress)
return `translate(${ x }px, ${ y }px)`
})
Much better! Now our markers stay in their starting y-position until they reach the
middle of the chart, then they gradually transition to their ending y-position.
Animated Sankey Diagram 504
Adding jitter
How can we make it easier to follow individual dots? Because our dots are sticking
to the middle of their paths, they run into each other and are hard to distinguish.
Let’s give each person a y-jitter – we’ll want to do this when they are created,
since we want the jitter to be consistent across their journey. We can use the
getRandomNumberInRange() utility function that is defined at the bottom of our
chart.js file.
Let’s also jitter each person’s startTime, so their x positions aren’t so regular. This
will give our visualization a bit more of a natural feel.
Animated Sankey Diagram 505
function generatePerson(elapsed) {
// ...
return {
sex,
ses,
education,
startTime: elapsed + getRandomNumberInRange(-0.1, 0.1),
yJitter: getRandomNumberInRange(-15, 15),
}
}
Bingo! Now each marker has more room, making its shape easier to parse.
We’ll want to remove any markers that have completed their journey. While we’re
in our “markers drawing” code, let’s also initialize each marker with an opacity of
0, so we can fade them in at the start.
females.enter().append("circle")
.attr("class", "marker marker-circle")
.attr("r", 5.5)
.style("opacity", 0)
females.exit().remove()
males.enter().append("polygon")
.attr("class", "marker marker-triangle")
.attr("points", trianglePoints)
.style("opacity", 0)
males.exit().remove()
And once our dots are at least 10 pixels to the right of their starting position, let’s
fade them in.
Animated Sankey Diagram 507
markers.style("transform", d => {
// ...
})
.transition().duration(100)
.style("opacity", d => xScale(xProgressAccessor(d)) < 10 ? 0 : 1)
Adding color
Once our markers make it to the right side, their sex is still clear (based on their
shape), but it’s not clear what socioeconomic status they started in. Let’s add a
color scheme that to help distinguish our markers.
interpolates between two colors – by setting the domain to an array of our sesIds,
we’ll get three unique, equally-spaced colors.
code/12-animated-sankey/completed/chart.js
We’re using .interpolate() to specify that we want to use the hcl color space. If
you remember from Chapter 7, hcl manipulates colors with human perception in
mind. Play around with different color spaces by changing the .interpolation()
setting to get a sense of how the color spaces differ.
let dimensions = {
// ...
pathHeight: 50,
endsBarWidth: 15,
}
Now we can draw our bars in our Draw peripherals step, right after we label our
starting paths.
Animated Sankey Diagram 509
code/12-animated-sankey/completed/chart.js
markers.style("transform", d => {
// ...
})
.attr("fill", d => colorScale(sesAccessor(d)))
.transition().duration(100)
.style("opacity", d => xScale(xProgressAccessor(d)) < 10 ? 0 : 1)
Awesome! Now we can tell where each person started, and trends are easier to notice.
For example, most of the markers traveling along in the bottom, right are green, and
the ending paths have more and more pink markers as we move up the education
buckets.
.marker {
mix-blend-mode: multiply;
}
Even though the overall visualization doesn’t change very much, if we zoom in we
can see that each shape is more distinct.
Animated Sankey Diagram 512
let currentPersonId = 0
function generatePerson(elapsed) {
currentPersonId++
// ...
return {
id: currentPersonId,
// ...
}
}
When we create our female and male markers in our updateMarkers() function, we
can tell d3 how to distinguish people from one another. D3 selection objects’ .data()
method⁸¹ takes a second parameter: a key accessor. This key accessor defaults to
the element’s index – this explains why our markers were being recycled: items in
our filtered array are removed once they reach the end of their journey, and new
elements end up in the same position in the filtered list.
Instead, let’s set our key accessor functions to return a person’s id, which will
guarantee that they aren’t recycled.
Great! Now that our markers aren’t being recycled, they will always start with an
opacity of 0.
⁸¹https://github.com/d3/d3-selection#selection_data
Animated Sankey Diagram 514
let dimensions = {
// ...
endingBarPadding: 3,
}
Let’s create a <g> to contain our ending bars. Since we run updateMarkers()
multiple times, we’ll want to create this group outside of it, to prevent from creating
multiple groups.
function updateMarkers(elapsed) {
// ...
Next, let’s draw our ending bars. At the end of our updateMarkers() function,
let’s create an array of people who have finished their journey and fit inside of that
education bucket.
function updateMarkers(elapsed) {
// ...
const endingGroups = educationIds.map((endId, i) => (
people.filter(d => (
xProgressAccessor(d) >= 1
&& educationAccessor(d) == endId
))
))
We’ll want to draw one bar per path: each permutation of sex, socioeconomic
status, and educational attainment. Let’s create a flattened array, with one object
per permutation that contain the starting and ending positions, the count, the
percent above, and the total count in the bar.
Animated Sankey Diagram 516
Now that we have an array for each of our ending bars, we can create one <rect>
per bar.
Animated Sankey Diagram 517
endingBarGroup.selectAll(".ending-bar")
.data(endingPercentages)
.join("rect")
.attr("class", "ending-bar")
.attr("x", d => -dimensions.endsBarWidth * (d.sexId + 1)
- (d.sexId * dimensions.endingBarPadding)
)
.attr("width", dimensions.endsBarWidth)
.attr("y", d => endYScale(d.endingId)
- dimensions.pathHeight / 2
+ dimensions.pathHeight * d.percentAbove
)
.attr("height", d => d.countInBar
? dimensions.pathHeight * d.percent
: dimensions.pathHeight
)
.attr("fill", d => d.countInBar
? colorScale(d.sesId)
: "#dadadd"
)
Now once our markers finish their journey, we’ll see their colors populate the bars on
the right. After our simulation has been running for a while, our stacked bars should
start to level out, approximating the percentages in our dataset.
Animated Sankey Diagram 518
The changes in our bars’ heights can be jerky at first – let’s smooth those transitions
with a transition CSS property.
.ending-bar {
transition: all 0.3s ease-out;
}
endingLabelsGroup.selectAll(".ending-value")
.data(endingPercentages)
.join("text")
.attr("class", "ending-value")
.attr("x", d => (d.sesId) * 33
+ 47
)
.attr("y", d => endYScale(d.endingId)
- dimensions.pathHeight / 2
+ 14 * d.sexId
+ 35
)
.attr("fill", d => d.countInBar
? colorScale(d.sesId)
: "#dadadd"
)
.text(d => d.count)
Great! Now we can see the exact numbers updating as more of our markers finish.
Animated Sankey Diagram 520
Let’s make a few tweaks to our numbers’ styles: we’ll decrease their size, right-align
them, and fix their width so they horizontally align with each other.
.ending-value {
font-size: 0.7em;
text-anchor: end;
font-weight: 600;
font-feature-settings: 'tnum' 1;
}
Animated Sankey Diagram 521
Next, we’ll create another <g>, this time specifically to label the left ending bars.
We’ll move this group to the left to sit right above the center of these bars.
Animated Sankey Diagram 522
In this group, we’ll draw a female marker, <text> to label our marker, and a <line>
that connects our marker to the stacked bars.
femaleLegend.append("polygon")
.attr("points", trianglePoints)
.attr("transform", "translate(-7, 0)")
femaleLegend.append("text")
.attr("class", "legend-text-left")
.text("Male")
.attr("x", -20)
femaleLegend.append("line")
.attr("class", "legend-line")
.attr("x1", -dimensions.endsBarWidth / 2 + 1)
.attr("x2", -dimensions.endsBarWidth / 2 + 1)
.attr("y1", 12)
.attr("y2", 37)
Let’s add a few styles in our styles.css file to fix our label’s text alignment and
give our <line> a stroke.
Animated Sankey Diagram 523
.legend {
font-size: 0.8em;
opacity: 0.6;
dominant-baseline: middle;
}
.legend-text-left {
text-anchor: end;
}
.legend-line {
stroke: grey;
stroke-width: 1px;
}
Great! Now our viewers can tell what each shape means, and what each stacked bar
stands for.
Additional steps
If you were interested in building on this chart for a production visualization, you
would want to add a threshold for a maximum number of people. Left running for a
while, the list of people would grow and grow, until it was too large and crashed the
browser window. Check out the completed code for an example of how to implement
a threshold.
Wrapping up
Great job making it through this visualization, it’s the most complex one by far! Show
it off by sharing an image or a gif with friends!
We learned a lot along the way – for example, how to simulate data and how to
create an animation loop. I hope you see how simulating a dataset engages a reader
in a way that a static visualization doesn’t.
Using D3 With React
We know how to make individual charts, but you might ask: what’s the best way to
draw a chart within my current JavaScript framework? D3 can be viewed as a utility
library, but it’s also used to manipulate the DOM, which means that there is a fair
bit of overlap in functionality between a JavaScript framework like React and d3 —
let’s talk about the best way to handle that overlap.
First off, we should decide when to use d3. Should we use d3 to render a whole page?
Let’s split up d3’s functionality by concern:
With the library compartmentalized in this way, you might come up with unortho-
dox ways to utilize d3. For example, I recently used it to create a calendar date picker.
Using D3 With React 527
Date picker
This date picker doesn’t look like a chart, but d3 came in handy in a few ways.
A d3 novice might not think to utilize d3 in this way, but once you’ve read this book
you will be familiar enough to take full advantage of the d3 library.
Using D3 With React 528
React.js
React.js is a framework for building declarative, module-based user interfaces. It
helps you split your interface code into components, which each have a render
function that describes the resulting DOM structure. One of React’s greatest strengths
is its diffing algorithm, which ensures minimal DOM updates, which are relatively
expensive.
If you’re unfamiliar with React, spend some time running through an introduction
like this one⁸². Our walkthrough will assume a basic understanding of the core
concepts since we want to focus on the d3 and React bits.
In this chapter, we’ll write React components’ render methods in JSX, which is an
HTML-like syntax. We’ll also be using hooks, which were released in the v16.8
release — don’t worry about versioning here, we’ll get all set up in a minute. Hooks
are a way to use state and lifecycle methods without making a class component, and
also help share code between components. We’ll even use our own custom hook!
But wait a minute, it seems like React and d3 are both used to create elements and
update the DOM. To draw a chart, should we use both of libraries? Just one? Neither?
Having just gotten comfortable with d3, you probably aren’t going to like the answer.
Instead of using axis generators and letting d3 draw elements, we’re going to let
React handle the rendering and to use d3 as a (very powerful) utility library.
Let’s build an example chart library and see for ourselves why this makes the most
sense.
Digging in
In the /code/13-using-d3-with-react-js/ folder, you’ll find a very bare bones
React app. First, download the necessary packages with npm (or yarn, if you prefer).
⁸²https://reactjs.org/docs/hello-world.html
Using D3 With React 529
Finished dashboard
Within the src folder, we have an App component that is loading random data and
updating it every four seconds — this will help us design our chart transitions.
Our chart-making plan has four levels of components:
1. App, which will decide what our dataset is and how to access values for our
axes (accessors),
2. Timeline, ScatterPlot, or Histogram which will be re-useable components that
decide how a specific type of chart is laid out and what goes in it,
3. Chart, which will pass down chart dimensions, and
⁸³http://localhost:8090
Using D3 With React 530
4. Axis, Line, Bars, etc., which will create individual components within our
charts.
Levels 3 and 4 will be isolated in the Chart folder, creating a charting library that can
be used throughout the app to make many types of charts. Having a basic charting
library will help in many ways, such as abstracting svg components idiosyncracies
(for example, collaborators won’t need to know that you need to use a <rect> to
create a rectangle, and it takes a x attribute whereas a <circle> takes a cx attribute).
If you’re feeling lost or want to see the finished code, the completed charts and
chart library are over in src/completed to help you out.
We’ll start by fleshing out our Timeline component, running through our usual
chart making steps.
1. Access data
2. Create dimensions
3. Draw canvas
4. Create scales
5. Draw data
6. Draw peripherals
7. Set up interactions
Access data
Let’s open up our Timeline component, located in src/Timeline.jsx. There’s not
much in here: the bones of a React component, prop types, and all of the imports
we’ll need.
Using D3 With React 531
1. data
2. xAccessor
3. yAccessor
4. label
These props are flexible enough to support throwing a timeline anywhere in our
dashboard with any dataset. But we don’t have so many props that creating a new
timeline is overwhelming or allows us to create inconsistent timelines.
Create dimensions
Next up, we need to specify the size of our chart. In our dashboard, we could
have Timelines of many different sizes. Each of these Timelines are also likely to
change size based on the window size. To keep things flexible, we’ll need to grab the
dimensions of our container <div>.
We could implement this by hand by creating a React ref, querying the size of
ref.current, and instantiating a Resize Observer to update on resize. Because we’ll
use this same logic in multiple chart types, we created a custom React hook called
useChartDimensions.
{
width: 1000,
height: 1000,
marginTop: 100,
marginRight: 100,
marginBottom: 100,
marginLeft: 100,
boundedHeight: 800,
boundedWidth: 800,
}
To keep things simple, this object is flat, unlike some dimensions objects we’ve
used before. In practice, if you need to rely on a specific structure for your
dimensions object, it might be better to keep it flat instead of nesting margins
inside another object.
We won’t get into what that code looks like, but you can give it a look over in the
src/Chart/utils.js file.
First, we’ll use our hook and pull out the ref reference and the calculated dimensions.
return (
<div className="Timeline" ref={ref}>
</div>
)
}
dimensions object
Draw canvas
Next up, we need to create our canvas. Since we’ll want a canvas for all of our charts,
we can put most of this logic in the Chart component. Let’s add a Chart to our render
method and pass it our dimensions.
When we look at our dashboard again, not much has changed. Let’s open up
src/Chart/Chart.jsx to see what we’re starting with.
Chart is a very basic functional React component — it asks for only one prop:
dimensions.
The children in a Chart can be any component from our chart library (or raw SVG
elements). Each of these components might need to know the dimensions of our chart
— for example, an Axis component might need to know how tall to be. Instead of
passing dimensions to each of these components as a prop, we can create a React
Context that defines the dimensions for the whole chart.
First, we’ll use the native React createContext() to create a new context — this
code will go outside of the component, after our imports.
Next up, we can fill out our useChartDimensions() variable to create a more
descriptive, easy-to-grab function that Chart components can use.
Lastly, we need to add the context provider to our render method and specify that
we want the context consumers to receive our dimensions object.
Great! Now all our chart components need to do to access the chart dimensions is to
grab the value from useChartDimensions().
Next up, we’ll use those dimensions to create our chart wrapper and bounds. If
you remember from Chapter 1, our wrapper spans the full height and width of the
chart and the bounds respect the chart margins.
Chart dimensions
<svg
className="Chart"
width={dimensions.width}
height={dimensions.height}>
{ children }
</svg>
Lastly, we’ll create our chart bounds to shift our chart components and enforce our
top and left margins.
Using D3 With React 536
<svg
className="Chart"
width={dimensions.width}
height={dimensions.height}>
<g transform={`translate(${
dimensions.marginLeft
}, ${
dimensions.marginTop
})`}>
{ children }
</g>
</svg>
Perfect! Our Chart component is ready to go. We won’t be able to see the difference
on our webpage, but we can see our wrapper components in the Elements panel of
our dev tools.
Chart elements
Create scales
Next up, we need to create the scales to convert from the data domain to the pixel
domain. Let’s pop back to src/Timeline.jsx.
We’ll create scales just like we did in Chapter 1 — we’ll need an time-based xScale
and a linear yScale.
Using D3 With React 537
If you wanted to make creating scales easier, you could abstract the concept of a
“scale” and add ease-of-use methods to your chart library. For example, you could
make a method that takes a dimension (eg. x) and an accessor function and create a
scale. A more comprehensive chart library can abstract redundant code and make
it easier for collaborators who are less familiar with data visualization.
Next, we’ll make a scaled accessor function for both of our axes. These will take a
data point and return the pixel value. This way, our Line component won’t need any
knowledge of our scales or data structure.
Draw data
We already imported our Line component from our chart library. Let’s render one
instance inside of our Chart, passing it our data and scaled accessor functions.
Using D3 With React 538
<Line
data={data}
xAccessor={xAccessorScaled}
yAccessor={yAccessorScaled}
/>
If we inspect our webpage in the Elements panel, we can see a new <path> element.
Line element
Line accepts data and accessor props, along with a type string. A Line can have
a type of "line" or "area" — it makes more sense to combine these two types of
elements because they are more similar than they are different. There is one more
prop (interpolation), which we’ll get back to later.
Our first step is to create our lineGenerator(), which will turn our dataset into a
d string for our <path>. Since d3.line() and d3.area() mimic our type prop, we
can grab the right method with d3[prop]().
Using D3 With React 539
d3.area() uses .y0() and .y1() to decide where the top and bottom of its path
are. We’ll need to add that extra logic only if we’re creating an area.
if (type == "area") {
lineGenerator
.y0(y0Accessor)
.y1(yAccessor)
}
Now we can use our lineGenerator() to convert our data into a d string.
<path {...props}
className={`Line Line--type-${type}`}
d={lineGenerator(data)}
/>
Nice! Now when we look at our webpage, we can see a squiggly line that updates
every few seconds.
Draw peripherals
Next, we want to draw our axes. This is where even experienced d3.js and React.js
developers get confused because both libraries want to handle creating new the
DOM elements. Up until now, we’ve used d3.axisBottom() and d3.axisLeft()
to append multiple <line> and <text> elements to a manually created <g>. element.
But the core concept of React.js relies on giving it full control over the DOM.
Let’s first make a naive attempt at an Axis component, mimicking the d3.js code
we’ve written so far. Since our Axis component is already imported, we can create a
new instance in our render method. We’ll need to specify the dimension and relevant
scale of both of our axes.
<Axis
dimension="x"
scale={xScale}
/>
<Axis
dimension="y"
scale={yScale}
/>
Remember that SVG elements’ z-indices are determined by their order in the DOM.
If you want your line to overlap your axes, make sure to add the<Axis> components
before the <Line> in your render method.
Let’s head over to src/Chart/Axis-naive to flesh out our Axis component. There’s
not much going on here yet, just a basic React Component that accepts a dimension
(either x or y), a scale, and a tick formatting function.
Using D3 With React 541
Let’s start by pulling in the dimensions of our chart, using the custom React hook we
created earlier.
const axisGeneratorsByDimension = {
x: "axisBottom",
y: "axisLeft",
}
Now we can use our mapping to create a new axis generator, based on our scale
prop.
In the past, we’ve used our axisGenerator on the d3 selection of a newly created <g>
element. React gives us a way to access DOM nodes created in the render method:
Refs. To create a React Ref, we create a new variable with useRef() and add it as a
ref attribute to the element we want to target.
Using D3 With React 542
return (
<g {...props}
ref={ref}
/>
)
Now when we access ref.current, we’ll get a reference to the <g> DOM node.
Let’s transform ref.current into a d3 selection by wrapping it with d3.select(),
then transition our axis in using our axisGenerator.
Note: we’ll have to ensure that ref.current exists first, since this code will run
before the first render.
if (ref.current) {
d3.select(ref.current)
.transition()
.call(axisGenerator)
}
Right! We’ll need to shift our x axis to the bottom of the chart. Let’s add a transform
attribute to our <g> element.
<g {...props}
ref={ref}
transform={
dimension == "x"
? `translate(0, ${dimensions.boundedHeight})`
: null
}
/>
const axisComponentsByDimension = {
x: AxisHorizontal,
y: AxisVertical,
}
const Axis = ({ dimension, ...props }) => {
const dimensions = useChartDimensions()
const Component = axisComponentsByDimension[dimension]
if (!Component) return null
return (
<Component {...props}
dimensions={dimensions}
/>
)
}
The other two components, AxisHorizontal and AxisVertical, are specific Axis
implementations. Let’s start by fleshing out AxisHorizontal.
Using D3 With React 546
function AxisHorizontal (
{ dimensions, label, formatTick, scale, ...props }
) {
return (
<g className="Axis AxisHorizontal" {...props}>
</g>
)
}
Since we’re not using a d3 axis generator, we’ll need to generate the ticks ourselves.
Fortunately, many of the methods d3 uses internally are also available for external
use. d3 scales have a .ticks() method that will create an array with evenly spaced
values in the scale’s domain.
We can see this in action if we console.log(scale.ticks()).
scale.ticks()
By default, .ticks() will aim for ten ticks, but we can pass a specific count to
target. Note that .ticks() will aim for the count, but also tries to create ticks with
meaningful intervals: a week in this example.
Using D3 With React 547
The number of ticks we want will depend on the chart width, though — ten ticks will
likely crowd our x axis. Let’s aim for one tick per 100 pixels for small screens and
one tick per 250 pixels for wider screens.
function AxisHorizontal (
{ dimensions, label, formatTick, scale, ...props }
) {
const numberOfTicks = dimensions.boundedWidth < 600
? dimensions.boundedWidth / 100
: dimensions.boundedWidth / 250
Great! We’re ready to render some elements. First, we’ll shift our axis to the bottom of
the chart. Remember: our <Axis> will render within our shifted group from <Chart>,
so we don’t have to worry about the top margin.
Most charts mark the end of the bounds with a line - let’s draw a line above our axis
to make it clear where the bottom of the y axis is.
Remember that <line> elements are positioned with x1, x2, y1, and y2 attributes.
We’ll want to draw a line from [0,0] to [width, 0] — since x1, x2, and y1 will all
be 0 (the default), we can leave those attributes out.
<line
className="Axis__line"
x2={dimensions.boundedWidth}
/>
Next, we’ll create the text for each of our ticks. To do this, we want to render a
<text> element for each item in our ticks array. Let’s shift each element down by
25px to give the axis line some breathing room and shift it to the right by converting
the tick into the pixel domain using our scale.
Using D3 With React 548
{ticks.map((tick, i) => (
<text
key={tick}
className="Axis__tick"
transform={`translate(${scale(tick)}, 25)`}
>
{ formatTick(tick) }
</text>
))}
When we look at our chart, we can see a wonderful x axis with ticks:
Those dates don’t look right, though. Note that our react Axis component accepts
a formatTick prop, which will be a function that takes a tick and converts it into
a human-readable string. d3 axis generators have built-in logic that will detect date
strings and format them correctly.
Let’s override the default formatTick prop and pass formatDate defined at the top
of src/Timeline.jsx.
<Axis
dimension="x"
scale={xScale}
formatTick={formatDate}
/>
Much better!
Using D3 With React 549
In your own React chart library, it might be a good idea to detect whether or not the
tick is a date object and format it accordingly. That will depend on your use cases:
how often will you need to format dates? Will you want all dates to be formatted the
same way?
Lastly, we’ll want to render the label for our axis. Since we might not always want
an axis label, we’ll check if the label prop exists before rendering our label. We’ll
also horizontally center our label and shift it down 60 pixels to give our ticks space.
{label && (
<text
className="Axis__label"
transform={`translate(${dimensions.boundedWidth / 2}, 60)`}
>
{ label }
</text>
)}
Our AxisVertical will look very similar to AxisHorizontal. We’ll start with the
same basic component.
Using D3 With React 550
function AxisVertical (
{ dimensions, label, formatTick, scale, ...props }
) {
return (
<g className="Axis AxisVertical" {...props}>
</g>
)
}
Try to fill out the component as much as possible without looking about the
completed code below. You might want to tweak the numberOfTicks parameter —
these ticks will be stacked vertically and might have more room.
Remember that your y axis label will need to be rotated -90 degrees to fit.
function AxisVertical (
{ dimensions, label, formatTick, scale, ...props }
) {
const numberOfTicks = dimensions.boundedHeight / 70
return (
<g className="Axis AxisVertical" {...props}>
<line
className="Axis__line"
y2={dimensions.boundedHeight}
/>
{ticks.map((tick, i) => (
<text
key={tick}
className="Axis__tick"
transform={`translate(-16, ${scale(tick)})`}
>
{ formatTick(tick) }
Using D3 With React 551
</text>
))}
{label && (
<text
className="Axis__label"
style={{
transform: `translate(-56px, ${
dimensions.boundedHeight / 2
}px) rotate(-90deg)`
}}
>
{ label }
</text>
)}
</g>
)
}
How did you do? No worries if you peeked! This code will be here for you if you
need to refer back to it when you set up your own charts.
See how we could replicate the d3 axes with a small amount of code? When we
know how to do something one way (such as draw axes with a d3 axis generator), this
knowledge prevents us from finding another way. D3 has many convenient methods,
but they aren’t always the best way to draw a chart. In fact, it can often help us to
circumvent itself with smaller methods that it uses.
Another benefit of creating our own axes is that we can customize our charts however
we want. Want tick marks but no line for your y axes? No problem! We can also style
our axes in one place and ensure that our charts are consistent, even when created
by different developers.
Set up interactions
In a production app, we would next want to define our interactions. Most charts
could benefit from a tooltip - this could be implemented in various ways.
Using D3 With React 552
1. We could add a chart listener rect to our Chart component that would sit on
top of its children. We could listen to all mouse events and use d3.scan() to
find the closest point and position the tooltip using our scales (similar to our
timeline example in Chapter 5).
2. We could add a boolean property to Line that creates the listener rect, tying the
tooltip to a specific data element. This might be beneficial if we have many types
of charts that need different types of listeners (like our scatter plot example in
Chapter 5).
Finishing up
Now that we’ve created some basic chart components and a Timeline component,
we have a general idea of how to weave React.js and d3.js together. The general idea
is to use React.js for any DOM rendering and d3.js as a utility library.
Populate the rest of the dashboard by switching the import statements in src/App.jsx
to use the files in the src/completed/ folder.
When we look at our browser again, we’ll see that the whole dashboard is populated!
Using D3 With React 553
Finished dashboard
src/completed/Histogram.jsx
to see how we converted our d3 code to React + d3 code. For example, instead of
using .enter().append() we simply map over each item in our dataset.
The completed timeline has an extra area with a gradient fill - check out how that
was implemented. One important piece of information to remember here is that our
timeline component could appear multiple times on a page, so we need a unique
id per gradient instance in order to grab the right one. This is simple enough to
implement, but easy to overlook.
If you’re feeling comfortable, play around with one of the charts - what if we added
a color scale, or sized the circles by a metric? What would it look like to implement a
timeline with multiple lines? What about something radically different, like a radar
chart? Remember to let React handle the DOM changes and utilize d3 as much as
possible for data manipulation and other conveniences like scales.
Using D3 With Angular
We know how to make individual charts, but you might ask: what’s the best way to
draw a chart within my current JavaScript framework? D3 can be viewed as a utility
library, but it’s also used to manipulate the DOM. There is a fair bit of overlap in
functionality between a JavaScript framework and d3 — let’s talk about the best way
to handle that overlap.
First off, we should decide when to use d3. Should we use d3 to render a whole page?
Let’s split up d3’s functionality by concern:
With the library compartmentalized in this way, you might come up with unortho-
dox ways to utilize d3. For example, I recently used it to create a calendar date picker.
Using D3 With Angular 555
Date picker
This date picker doesn’t look like a chart, but d3 came in handy in a few ways.
A d3 novice might not think to utilize d3 in this way, but once you’ve read this book
you will be familiar enough to take full advantage of the d3 library.
Using D3 With Angular 556
Angular
Angular is a framework for building modern, component-based user interfaces in
HTML and Typescript. Typescript is a superset of Javascript: it looks very similar to
Javascript and gets compiled to Javascript, but it has extra features like static typing,
classes, and interfaces.
In this chapter, I’m going to focus on the right way to combine Angular and d3.js.
Unfortunately, that doesn’t leave much space for introducing Angular itself – if
you’re unfamiliar with Angular, you’ll want to spend some time getting familiar
first. They have a quick-start tutorial in their docs⁸⁴ that might be helpful.
Angular is a large framework that helps to run applications: Because Angular is such
a full-featured framework, it doesn’t fit nicely with d3. To draw a chart, should we
use both of libraries? Just one? Neither?
Having just gotten comfortable with d3, you probably aren’t going to like the answer.
Instead of using axis generators and letting d3 draw elements, we’re going to let
Angular handle the rendering and to use d3 as a (very powerful) utility library.
Let’s build an example chart library and see for ourselves why this makes the most
sense.
Digging in
In the /code/14-using-d3-with-angular/ folder, you’ll find a very bare bones
Angular app. Angular comes with a command-line interface tool that helps with
various development tasks, like bundling our code. First, let’s install that in the
terminal with the following command:
Once npm has finished installing that, let’s start up our app! In the terminal,
in/code/14-using-d3-with-angular/ folder, let’s run our app:
⁸⁴https://angular.io/tutorial
Using D3 With Angular 557
ng serve --open
The --open parameter will open our app (most likely at http://localhost:4200⁸⁵) in
the browser, once the server is up and running. You should see an empty dashboard
with three placeholders — one for a timeline, one for a scatter plot, and one for a
histogram.
Finished dashboard
Within the app folder, we have an app component that is loading random data and
updating it every four seconds — this will help us design our chart transitions.
Our chart-making plan has four levels of components:
1. app, which will decide what our dataset is and how to access values for our
axes (accessors),
2. timeline, scatter, or histogram which will be re-useable components that
decide how a specific type of chart is laid out and what goes in it,
3. chart, which will render our wrapper and bounds, and
4. axis, line, bars, etc., which will create individual elements within our charts.
⁸⁵http://localhost:4200
Using D3 With Angular 558
Levels 3 and 4 will be isolated in the chart folder, creating a charting library that
can be used throughout the app to make many different types of charts. Having a
basic charting library will help in many ways, such as abstracting svg components
idiosyncracies (for example, collaborators won’t need to know that you need to use
a <rect> to create a rectangle, and it takes a x attribute whereas a <circle> takes
a cx attribute).
If you’re feeling lost or want to see the finished code, the completed charts and chart
library are over in /code/14-using-d3-with-angular/app/completed to help
you out.
We’ll start by fleshing out our timeline component, running through our usual
chart making steps.
1. Access data
2. Create dimensions
3. Draw canvas
4. Create scales
5. Draw data
6. Draw peripherals
7. Set up interactions
Access data
Let’s open up our timeline component, located in
src/app/timeline/timeline.component.ts. There isn’t much in here: the bones
of an Angular component, types for our inputs, and all of the imports we’ll need.
Using D3 With Angular 559
code/14-using-d3-with-angular/src/app/timeline/timeline.component.ts
6 @Component({
7 selector: 'app-timeline',
8 templateUrl: './timeline.component.html',
9 styleUrls: ['./timeline.component.css'],
10 })
11 export class TimelineComponent {
12 @Input() data: object[]
13 @Input() label: string
14 @Input() xAccessor: AccessorType
15 @Input() yAccessor: AccessorType
16
17 }
The @Component object at the top helps configure the different parts of our compo-
nent:
code/14-using-d3-with-angular/src/app/app.component.html
5 <app-timeline
6 [data]="timelineData"
7 [xAccessor]="dateAccessor"
8 [yAccessor]="temperatureAccessor"
9 label="Temperature">
10 </app-timeline>
We can also see the different parameters that app is passing to our timeline – we’ll
look at those in more depth in a minute.
Using D3 With Angular 560
1. the templateUrl key tells our component where to find this component’s
template. We could also define our template inline, using a plain string, which
we’ll take advantage of later.
2. the styleUrls key tells our component where to find our component styles.
If we look in the timeline.component.css file, we’ll see that our styles are
already populated for us – this is to help us to focus on the Angular code. If
at any point you’re wondering why our components look the way they do, the
answer is in the linked .css files!
1. data
2. xAccessor
3. yAccessor
4. label
These properties are flexible enough to support throwing a timeline anywhere in our
dashboard with any dataset. But we don’t have so many properties that creating a
new timeline is overwhelming or allows us to create inconsistent timelines.
Create dimensions
Next up, we need to specify the size of our chart. In our dashboard, we could have
timelines of many different sizes. Each of these timelines are also likely to change size
based on the window size. To keep things flexible, we’ll need to grab the dimensions
of our container <div>.
Using D3 With Angular 561
Let’s first add a new public variable named dimensions – we can use a custom type
that we’ve imported from /chart/utils.ts. If we look inside /chart/utils.ts,
we can see our dimensions object is expected to have a height, width, and all four
margins.
code/14-using-d3-with-angular/src/app/completed/chart/utils.ts
To keep things simple, this object is flat, unlike some dimensions objects we’ve used
before. In practice, if you need to rely on a specific structure for your dimensions
object, it might be better to keep it flat instead of nesting margins inside another
object.
Let’s get back to our timeline.component.ts file and define our dimensions
variable.
code/14-using-d3-with-angular/src/app/completed/timeline/timeline.component.ts
code/14-using-d3-with-angular/src/app/completed/timeline/timeline.component.ts
27 constructor() {
28 this.dimensions = {
29 marginTop: 40,
30 marginRight: 30,
31 marginBottom: 75,
32 marginLeft: 75,
33 height: 300,
34 width: 600,
35 }
36 this.dimensions = {
37 ...this.dimensions,
38 boundedHeight: Math.max(this.dimensions.height
39 - this.dimensions.marginTop
40 - this.dimensions.marginBottom, 0),
41 boundedWidth: Math.max(this.dimensions.width
42 - this.dimensions.marginLeft
43 - this.dimensions.marginRight, 0),
44 }
45 }
While we want to specify the exact height of our timeline, we want our timeline to
stretch horizontally to fit its container. After we’ve defined our dimensions, let’s use
@ViewChild() to create an ElementRef and hook onto an element in our template.
code/14-using-d3-with-angular/src/app/completed/timeline/timeline.component.ts
Read more about this flag or about ElementRefs on the Angular docs.
https://angular.io/guide/static-query-migration
https://angular.io/api/core/ViewChild
this.container.nativeElement.
updateDimensions() {
const width = this.container.nativeElement.offsetWidth
this.dimensions.width = width
this.dimensions.boundedWidth = Math.max(
this.dimensions.width
- this.dimensions.marginLeft
- this.dimensions.marginRight,
0
)
console.log(dimensions)
}
We’ll want to execute our updateDimensions() function directly after our com-
ponent is first rendered. We’ll use the ngAfterContentInit lifecycle hook – read
about it and others in the Angular docs⁸⁶.
First, we’ll need to specify that we want to use ngAfterContentInit when we create
our class.
⁸⁶https://angular.io/guide/lifecycle-hooks
Using D3 With Angular 564
Now we can add a new function to the bottom of our TimelineComponent defini-
tion.
code/14-using-d3-with-angular/src/app/completed/timeline/timeline.component.ts
63 ngAfterContentInit() {
64 this.updateDimensions()
65 }
Now when we load our dashboard in the browser, we’ll see our dimensions object
logged in our Dev Tools console.
We can tell that we’re updating our width because dimensions.width has updated
from its original value of 600 pixels.
Thankfully, Angular makes these kinds of event listeners very easy. We can use a
@HostListener⁸⁷ to run our updateDimensions() function every time the window
resizes.
code/14-using-d3-with-angular/src/app/completed/timeline/timeline.component.ts
47 @HostListener('window:resize') windowResize() {
48 this.updateDimensions()
49 }
Great! Now when we resize our window, we’ll see our dimensions object with an
updated width number in the Dev Tools console.
⁸⁷https://angular.io/api/core/HostListener
Using D3 With Angular 566
Draw canvas
Next up, we need to create our <svg> element. Since we’ll want a canvas for all of
our charts, we’ll wrap this work into a component that we can re-use. Let’s open up
the /charts/chart.component.ts file.
We don’t need much to happen here, but we do want to pass our chart a dimensions
object so it can shift our chart bounds. Let’s define our dimensions with our custom
dimensionsType (which is already imported into this file).
code/14-using-d3-with-angular/src/app/completed/chart/chart.component.ts
4 @Component({
5 selector: 'app-chart',
6 templateUrl: './chart.component.html',
7 styleUrls: ['./chart.component.css']
8 })
9 export class ChartComponent {
10 @Input() dimensions: DimensionsType
11 }
Chart dimensions
Using D3 With Angular 567
First, we’ll add our wrapper: a <svg> element sized with our dimensions. We can
set our HTML attributes using attr., and we’ll wrap our [attr.] statement in
square brackets [] to specify that we want to pass a variable name, instead of a plain
string.
Next, we’ll add our bounds: a <g> element that is shifted by our top and left margins.
If we use a plain <g> tag, Angular will think we’re using a custom directive. To tell
Angular that we’re using the SVG namespace, we’ll need to add a svg: prefix to all
of our SVG elements’ tags.
Now we can shift our <g> element using our top and left margins. We’ll use a style.
prefix to specify that we want our transform to be applied as a CSS style.
Note that we’re .join()ing an array of strings to build our transform string.
This helps us keep our templates readable. We would normally use an ES6 template
literal (${}), but angular templates don’t recognize that grammar.
When we use our chart component, we’ll want to nest our data and peripheral
elements inside. Let’s add a <ng-content /> tag to pass any nested elements into
our bounds.
code/14-using-d3-with-angular/src/app/completed/chart/chart.component.html
<div #container>
<app-chart
[dimensions]="dimensions">
</app-chart>
</div>
We’re using the app-chart tag name here, since that’s what we specified as our
chart component’s selector in the configuration object.
Perfect! We can now see a <svg> and <g> element in the Elements panel of our dev
tools. When we resize our window, we’ll see that our <svg> updates its size to fit
perfectly
Create scales
Next up, we need to create the scales to convert from the data domain to the pixel
domain. Let’s pop back to our timeline’s typescript file:
timeline/timeline.component.ts and add some definitions.
code/14-using-d3-with-angular/src/app/completed/timeline/timeline.component.ts
At the end of our timeline class, let’s create an updateScales() function that
creates our scales. We’ll also make scaled accessor functions that we can pass to
chart components, so that they don’t need to be aware of our scales.
code/14-using-d3-with-angular/src/app/completed/timeline/timeline.component.ts
71 updateScales() {
72 this.xScale = d3.scaleTime()
73 .domain(d3.extent(this.data, this.xAccessor))
74 .range([0, this.dimensions.boundedWidth])
75
76 this.yScale = d3.scaleLinear()
77 .domain(d3.extent(this.data, this.yAccessor))
78 .range([this.dimensions.boundedHeight, 0])
79 .nice()
80
81 this.xAccessorScaled = d => this.xScale(this.xAccessor(d))
82 this.yAccessorScaled = d => this.yScale(this.yAccessor(d))
83 this.y0AccessorScaled = this.yScale(this.yScale.domain()[0])
84 }
code/14-using-d3-with-angular/src/app/completed/timeline/timeline.component.ts
51 updateDimensions() {
52 const width = this.container.nativeElement.offsetWidth
53 this.dimensions.width = width
54 this.dimensions.boundedWidth = Math.max(
55 this.dimensions.width
56 - this.dimensions.marginLeft
57 - this.dimensions.marginRight,
58 0
59 )
60 this.updateScales()
61 }
We also want our scales to update whenever our data changes. We can use the
ngOnChanges lifecycle hook. First, let’s tell our component that we’re going to use
the hook:
code/14-using-d3-with-angular/src/app/completed/timeline/timeline.component.ts
Then we’ll add a ngOnChanges() function to our class definition, specifying that we
want it to run on SimpleChanges.
code/14-using-d3-with-angular/src/app/completed/timeline/timeline.component.ts
If you’re curious, you can read more about this hook in the Angular docs.
https://angular.io/api/core/OnChanges
Using D3 With Angular 572
Draw data
Here’s the best step! Now we get to draw our data elements – in this case, a line. Let’s
prepare by creating a line component. Let’s open up the
chart/line/line.components.ts file, which has the bones of a new component.
code/14-using-d3-with-angular/src/app/chart/line/line.component.ts
4 @Component({
5 selector: '[appLine]',
6 template: ``,
7 styleUrls: ['./line.component.css']
8 })
9 export class LineComponent {
10 }
1. We’re defining our selector using square brackets ([]). This makes it an
attribute selector, which specifies that our component must be used as an
attribute on another element. For example, we’ll need to use our line
component with the code <svg:g app-line />, and not <app-line />.
2. Instead of passing in a templateUrl that links to another .html file, we’ll be
defining our template inline. This will keep our code concise, since our line
template will only be a few lines.
Let’s start fleshing our component out, starting with defining any properties we’ll
want to pass to it.
Using D3 With Angular 573
code/14-using-d3-with-angular/src/app/completed/chart/line/line.component.ts
17 @Input() type: "area" | "line" = "line"
18 @Input() data: object[]
19 @Input() xAccessor: AccessorType
20 @Input() yAccessor: AccessorType
21 @Input() y0Accessor?: AccessorType
22 @Input() interpolation?: Function = d3.curveMonotoneX
23 @Input() fill?: string
We’ll also want to define a lineString variable that will hold our d attribute string
that will tell the line what shape to be.
code/14-using-d3-with-angular/src/app/completed/chart/line/line.component.ts
24 private lineString: ""
Next, we’ll make a function that will create our lineString variable. We’ll keep
our component flexible, and handle drawing either a line or an area. d3.line()
and d3.area() are very similar, except that d3.line() uses a .y() method and
d3.area() uses a .y0() and a .y1() method (for the bottom and top of the area).
code/14-using-d3-with-angular/src/app/completed/chart/line/line.component.ts
26 updateLineString(): void {
27 const lineGenerator = d3[this.type]()
28 .x(this.xAccessor)
29 .y(this.yAccessor)
30 .curve(this.interpolation)
31
32 if (this.type == "area") {
33 lineGenerator
34 .y0(this.y0Accessor)
35 .y1(this.yAccessor)
36 }
37
38 this.lineString = lineGenerator(this.data)
39 }
Using D3 With Angular 574
We want to execute this function any time any of our properties changes. At the
top of our component definition, let’s tell our component that we want to use an
OnChanges lifecycle hook.
code/14-using-d3-with-angular/src/app/completed/chart/line/line.component.ts
Lastly, we’ll update our template string in our @Component configuration. We want
to draw one <svg:path> element and set its d attribute to our lineString. We’ll
also want to give it a class, linked to our line’s type property, so we can style our
line and area types differently, and pass down a fill style, which will come in
handy if we want to pass a specific fill color or gradient id.
code/14-using-d3-with-angular/src/app/completed/chart/line/line.component.ts
5 @Component({
6 selector: '[appLine]',
7 template: `
8 <svg:path
9 [ngClass]="type"
10 [attr.d]="lineString"
11 [style.fill]="fill">
12 </svg:path>
13 `,
14 styleUrls: ['./line.component.css']
15 })
Using D3 With Angular 575
<div #container>
<app-chart
[dimensions]="dimensions">
<svg:g appLine
[data]="data"
[xAccessor]="xAccessorScaled"
[yAccessor]="yAccessorScaled">
</svg:g>
</app-chart>
</div>
Nice! Now when we look at our webpage, we can see a squiggly line that updates
every few seconds.
Draw peripherals
Next, we want to draw our axes. This is where even experienced d3.js and Angular
developers get confused because both libraries want to handle creating new the
DOM elements. Up until now, we’ve used d3.axisBottom() and d3.axisLeft()
to append multiple <line> and <text> elements to a manually created <g>. element.
But because Angular is such a full-featured framework, it’s important to let it have
full control over the DOM.
Let’s first make a naive attempt at an axis component, mimicking the d3.js code
we’ve written so far. Since we’re already in our timeline/timeline.component.html
file, let’s start by adding a new <svg:g> element with an attribute of appAxis
(assuming that the API will be the same as our line component). We’ll make this
a y axis by giving it a dimension of "y", and we’ll pass other variables that we’ll
need: dimensions, scale, and a label.
<div #container>
<app-chart
[dimensions]="dimensions">
<svg:g appAxis
dimension="y"
[dimensions]="dimensions"
[scale]="yScale"
[label]="label">
</svg:g>
<svg:g appLine
[data]="data"
[xAccessor]="xAccessorScaled"
[yAccessor]="yAccessorScaled">
</svg:g>
</app-chart>
</div>
Using D3 With Angular 577
Remember that SVG elements’ z-indices are determined by their order in the
DOM. If you want your line to overlap your axes, make sure to add theappAxis
component before the appLine component.
5 @Component({
6 selector: '[appAxis]',
7 templateUrl: './axis.component.html',
8 styleUrls: ['./axis.component.css']
9 })
10 export class AxisComponent {
11 }
First, we’ll define all of the properties our component will handle.
code/14-using-d3-with-angular/src/app/completed/chart/axis/axis.component.ts
<svg:g #axis>
</svg:g>
updateTicks() {
const yAxisGenerator = d3.axisLeft()
.scale(this.scale)
d3.select(this.axis.nativeElement)
}
We’ll want to use a different method based on the location of our axis (d3.axisLeft()
or d3.axisBottom()), but we’ll keep things simple for now, since this is an
example.
And then we’ll create that function at the bottom of our file.
Using D3 With Angular 579
code/14-using-d3-with-angular/src/app/completed/chart/axis/axis.component.ts
Great! Now when we load our website, our timeline will have a y axis that updates
with our line.
template code. When you come back to a component you wrote a few months ago,
you’ll thank yourself for making the output obvious.
In a pinch, using Angular to create a wrapper element to modify with d3 (like we
just did) will do. You might need to do this in special situations, like animating an
arc. But try to lean towards solely creating elements with Angular and using d3 as
more of a utility library. This will keep your app speedy and less “hacky”.
Even without its DOM modifying methods, d3 is a very powerful library. In fact, we
created most of our timeline without needing to re-create a d3 generator function.
For example, creating a d string for our line component would have been tricky
without d3.line().
But how does this look in practice? Let’s re-create our axis component without using
any axis generators.
Next, we’ll need to update our updateTicks() function. Fortunately, many of the
methods d3 uses internally are also available for external use. d3 scales have a
.ticks() method that will create an array with evenly spaced values in the scale’s
domain.
By default, .ticks() will aim for ten ticks, but we can pass a specific count to
target. Note that .ticks() will aim for the count, but also tries to create ticks with
meaningful intervals: for example, a week in a time scale.
The number of ticks we want will depend on the chart width, though — ten ticks will
likely crowd our y axis. Let’s aim for one tick per 100 pixels for small screens and one
tick per 250 pixels for wider screens on the x axis, and one tick per 70 pixels on the y
axis. You’ll want to play around with these numbers yourself – they will depend on
your preference and font-sizes.
Using D3 With Angular 581
code/14-using-d3-with-angular/src/app/completed/chart/axis/axis.component.ts
18 updateTicks() {
19 if (!this.dimensions || !this.scale) return
20
21 const numberOfTicks = this.dimension == "x"
22 ? this.dimensions.boundedWidth < 600
23 ? this.dimensions.boundedWidth / 100
24 : this.dimensions.boundedWidth / 250
25 : this.dimensions.boundedHeight / 70
26
27 this.ticks = this.scale.ticks(numberOfTicks)
28 }
We’ll need to turn our raw tick numbers into human-friendly labels - we’ll add a
formatTick input near the top of our class so that the parent chart can specify
the formatting. This will be a function that takes a tick value and converts it into a
human-readable string.
code/14-using-d3-with-angular/src/app/completed/chart/axis/axis.component.ts
15 @Input() formatTick: (value: any) => (string|number) = d3.format(",")
Note that we set a default value for our dimension and formatTick inputs. This
way, we won’t need to specify these values every time we create an Axis.
Moving over to chart/axis/axis.component.html, we’ll want to update our
template code to map over our ticks. Since the position of our axis components
will vary based on the axis type (x or y), let’s create those two types separately.
First, we’ll focus on the x type, using *ngIf to only render these elements if our
dimension property equals "x". Our axis will be composed of:
code/14-using-d3-with-angular/src/app/completed/chart/axis/axis.component.html
1 <svg:g
2 *ngIf="dimension == 'x'"
3 class="x"
4 attr.transform="translate(0, {{dimensions.boundedHeight}})">
5 <svg:line
6 [attr.x2]="dimensions.boundedWidth">
7 </svg:line>
8 <svg:text
9 *ngFor="let tick of ticks"
10 class="tick"
11 attr.transform="translate({{scale(tick)}}, 25)">
12 {{ formatTick(tick) }}
13 </svg:text>
14 <svg:text
15 *ngIf="label"
16 class="label"
17 attr.transform="translate({{dimensions.boundedWidth / 2}}, 60)">
18 {{ label }}
19 </svg:text>
20 </svg:g>
Remember that <line> elements are positioned with x1, x2, y1, and y2 attributes.
We’ll want to draw a line from [0,0] to [dimensions.boundedWidth, 0] —
since x1, x2, and y1 will all be 0 (the default), we can leave those attributes out.
Our y axis will be very similar – we’ll create the same elements, but position them
differently. Note that we’ll rotate our label <text> 90 degrees counter-clockwise.
Using D3 With Angular 583
code/14-using-d3-with-angular/src/app/completed/chart/axis/axis.component.html
21 <svg:g
22 class="y"
23 *ngIf="dimension == 'y'">
24 <svg:line
25 [attr.y2]="dimensions.boundedHeight">
26 </svg:line>
27 <svg:text
28 *ngFor="let tick of ticks"
29 class="tick"
30 attr.transform="translate(-16, {{scale(tick)}})">
31 {{ formatTick(tick) }}
32 </svg:text>
33 <svg:text
34 *ngIf="label"
35 class="label"
36 [ngStyle]="{transform: [
37 'translate(-56px, ',
38 dimensions.boundedHeight / 2,
39 'px) rotate(-90deg)'
40 ].join('')}">
41 {{ label }}
42 </svg:text>
43 </svg:g>
That was a mouthful! We ended up writing way more template code – partially
because we were handling both x and y axis types, but mostly because we’re not
relying on d3 to create all of these elements for us. Cheer up, though! The axis
component is the worst part of using Angular and d3 this way.
Using D3 With Angular 584
<svg:g appAxis
dimension="x"
[dimensions]="dimensions"
[scale]="xScale"
[label]="label">
</svg:g>
When we look at our chart, we can see a wonderful x axis with ticks:
Those dates don’t look right, though. d3 axis generators have built-in logic that will
detect date strings and format them correctly.
Let’s override the default formatTick prop and pass a formatDate function that
we’ll define in a minute.
<svg:g appAxis
dimension="x"
[dimensions]="dimensions"
[scale]="xScale"
[label]="label"
[formatTick]="formatDate">
</svg:g>
In your own Angular chart library, it might be a good idea to detect whether or not
the tick is a date object and format it accordingly. That will depend on your use cases:
Using D3 With Angular 586
how often will you need to format dates? Will you want all dates to be formatted the
same way?
See how we could replicate the d3 axes with a small amount of code? When we
know how to do something one way (such as draw axes with a d3 axis generator), this
knowledge prevents us from finding another way. D3 has many convenient methods,
but they aren’t always the best way to draw a chart. In fact, it can often help us to
circumvent itself with smaller methods that it uses.
Another benefit of creating our own axes is that we can customize our charts however
we want. Want tick marks but no line for your y axes? No problem! We can also style
our axes in one place and ensure that our charts are consistent, even when created
by different developers.
Set up interactions
In a production app, we would next want to define our interactions. Most charts
could benefit from a tooltip - this could be implemented in various ways.
1. We could add a chart listener rect to our chart component that would sit on top
of its nested elements. We could listen to all mouse events and use d3.scan()
to find the closest point and position the tooltip using our scales (similar to our
timeline example in Chapter 5).
2. We could add a boolean property to our line component that creates the
listener rect, tying the tooltip to a specific data element. This might be beneficial
if we have many types of charts that need different types of listeners (like our
scatter plot example in Chapter 5).
Finishing up
Now that we’ve created some basic chart components and a timeline component,
we have a general idea of how to weave Angular and d3.js together. The general idea
is to use Angular for any DOM rendering and d3.js as a utility library.
Populate the rest of the dashboard by switching the import statements in
src/app/app.module.ts to use the files in the src/completed/ folder.
Using D3 With Angular 587
When we look at our browser again, we’ll see that the whole dashboard is populated!
Using D3 With Angular 588
Finished dashboard
src/app/completed/histogram.component.ts
to see how we converted our d3 code to Angular + d3 code. For example, instead of
using .enter().append() we simply map over each item in our dataset.
The completed timeline has an extra area with a gradient fill - check out how that
was implemented. One important piece of information to remember here is that our
timeline component could appear multiple times on a page, so we need a unique
id per gradient instance in order to grab the right one. This is simple enough to
implement, but easy to overlook.
If you’re feeling comfortable, play around with one of the charts - what if we added
a color scale, or sized the circles by a metric? What would it look like to implement a
timeline with multiple lines? What about something radically different, like a radar
chart? Remember to let Angular handle the DOM changes and utilize d3 as much as
possible for data manipulation and other conveniences like scales.
D3.js
Let’s step back and look at all of the ground we’ve covered since we started this book.
Here is a diagram of all of the d3.js modules, scaled by their minified size:
That’s pretty overwhelming! But we’ve actually talked about almost half of them —
here are the ones we’ve covered in yellow:
D3.js 590
• d3-geo⁸⁹ helps with turning GeoJSON into 2D svg paths, and d3-geo-polygon⁹⁰
adds extra projections.
• d3-shape⁹¹ helps with creating shapes — like our timeline and pie chart.
• d3-scale-chromatic⁹² provides many color scales to use and d3-color⁹³ lets us
manipulate colors to create our own color scales.
• d3-scale⁹⁴ helps us create scales to map from one domain to another. Be sure
to explore the docs⁹⁵ to see what scale types we haven’t covered, such as
d3.scaleLog(), d3.scaleThreshold(), and d3.scaleOrdinal(), among
others.
⁸⁹https://github.com/d3/d3-geo
⁹⁰https://github.com/d3/d3-geo-polygon
⁹¹https://github.com/d3/d3-shape
⁹²https://github.com/d3/d3-scale-chromatic
⁹³https://github.com/d3/d3-color
⁹⁴https://github.com/d3/d3-scale
⁹⁵https://github.com/d3/d3-scale
D3.js 592
• d3-axis¹⁰⁴ helps with creating axes — check out the docs¹⁰⁵ to see how to
customize your ticks.
• and lastly, the first module we learned: d3-fetch¹⁰⁶, which helps to fetch and
parse data files.
• d3-zoom¹⁰⁸ and d3-brush¹⁰⁹ help with creating interactive charts, where the
user can “zoom in” on a specific part.
¹⁰⁸https://github.com/d3/d3-zoom
¹⁰⁹https://github.com/d3/d3-brush
D3.js 594
Now that you understand the basics of how to create a chart with d3.js and how the
API generally works, I encourage you to explore these different modules. It’s at least
good to be aware of them — they might come in handy in the future.
Going forward
Great work learning all of that! As you can see, we covered a lot of concepts in this
book — both how to implement it and also the theory behind how to design a good
data visualization.
I’m sure you have tons of data you want to visualize — get out there are do it! Don’t
be overwhelmed by starting a chart by yourself — take it step by step (the chart steps
are available in Appendix B), apply the fundamentals we’ve learned here, and refer
back to the book when you need to pointer.
If you want ideas, here is a list of places to find inspiration:
Flowing Data¹¹² is a great, frequently-updated blog by Nathan Yau with original
data visualizations and links to other complex data visualizations.
¹¹²https://flowingdata.com/
D3.js 596
The Pudding¹¹³ is a digital publication that explores popular concepts with data
visualizations. This is a great place to find visualizations that are interactive and
easily digestable.
The New York Times Upshot¹¹⁴ often has interactive pieces that are linked to news
events. Some of my favorite examples come from here, like their You Draw It article
from 2015¹¹⁵ where you guess what the data look like.
Let’s open up the get_weather_data.py file and update the apikey value, then
grab our latitude and longitude from gps-coordinates.org¹¹⁸. Make sure to preserve
the sign of the longitude (if it’s negative, keep the minus sign).
Once we update those values we can run the script.
The script will use 365 of our 1,000 free daily api calls, so double-check the values
before we run it.
¹¹⁸gps-coordinates.org
Appendix 599
$ python get_weather_data.py
Now we should have a new my_weather_data.json file with the past 365 days of
local weather. Let’s take a closer look!
color picker
We can see the normal color picking controls, but also a Contrast ratio section with
a warning symbol. Let’s click the arrow to the right of the section to expand it.
Appendix 600
Now we can see that the dev tools are making two checks: AA and AAA. These are
checking for different contrast ratio levels — by contrast ratio, we’re talking about
the difference in luminance between the text and its surroundings (higher means
more contrast). AA is testing for at least a 4.5:1 contrast ratio (for people with 20/40
vision) and the AAA threshold is even higher at 7:1.
As we change the color by clicking around the color map at the top of the popup, we
can see that values below the curved white line pass the AA threshold (one check
mark) and values that are further below the white line pass the AAA threshold (two
check marks).
Appendix 601
When making charts for your dashboard, it’s a good idea to check your main colors
every now and then to make sure they are visible to all users.
Appendix 602
B. Chart-drawing checklist
These are the basic steps to follow to create a chart.
1. Access data
Look at the data structure and declare how to access the values we’ll need
2. Create chart dimensions
Declare the physical (i.e. pixels) chart parameters
3. Draw canvas
Render the chart area and bounds element
4. Create scales
Create scales for every data-to-physical attribute in our chart
5. Draw data
Render your data elements
6. Draw peripherals
Render your axes, labels, and legends
7. Set up interactions
Initialize event listeners and create interaction behavior
Appendix 603
Checklist
Appendix 604
06-26-2019
Chapter 1
Chapter 3
Chapter 12
Chapter 14
Revision 7
06-14-2019
Chapter 3
Chapter 6
Chapter 13
Chapter 14
Upgrade to Angular 8
Changelog 607
Revision 6
06-06-2019
Chapter 2
Chapter 5
Revision 5
06-03-2019
Chapter 5
Fix text to match the updated example using d3-delaunay instead of d3-voronoi
Thanks to Miguel C. for reporting
Chapter 6
Revision 4
05-24-2019
Chapter 1
Revision 3
05-17-2019
Chapter 6
Add data_bank_data.csv files (csv files were listed in the .gitignore file).
Thanks to Guillaume C. for reporting
Chapter 11
Chapter 13
Appendix
Add extension to python get_weather_data.py command.
Thanks to Steve for reporting