Data Visualization workshop
Shirley Wu
(@sxywu)
data:image/s3,"s3://crabby-images/c62c1/c62c1134f0d1a8d573f232bc09baf96783cc3069" alt=""
data:image/s3,"s3://crabby-images/030f4/030f49871b8ee98f4d215a379ced9f5456379bbe" alt=""
You will learn:
SVG Paths
SVG Transforms
Selections
Data binding
Enter-append
Scales
Update & Exit
Transitions
Nesting elements
SVG Elements
rect
x: x-coordinate of top-left
y: y-coordinate of top-left
width
height
circle
cx: x-coordinate of center
cy: y-coordinate of center
r: radius
text
x: x-coordinate
y: y-coordinate
dx: x-coordinate offset
dy: y-coordinate offset
text-anchor: horizontal text alignment
Hi!
path
d: path to follow
Moveto, Lineto, Curveto, Arcto
SVG Paths
SVG Transforms
Exercise time
Create 3 unique petal shapes
data:image/s3,"s3://crabby-images/0c08f/0c08fa01775b2d6d3609bc783420f000d73f9fd4" alt=""
Selection & Data
<svg>
<rect />
<rect />
<rect />
<rect />
<rect />
</svg>
<script>
var data = [100, 250, 175, 200, 120];
d3.selectAll('rect')
.data(data)
.attr('x', (d, i) => i * rectWidth)
.attr('y', d => height - d)
.attr('width', rectWidth)
.attr('height', d => d)
.attr('fill', 'blue')
.attr('stroke', '#fff');
</script>
data:image/s3,"s3://crabby-images/c6267/c62678b6f7bcc3a256edd156e5474c079d878918" alt=""
Selection & Data
5 rectangle elements
Select all rectangle elements that exist
data:image/s3,"s3://crabby-images/12962/129623d6b659bde458ee54206befc9bbef8295cd" alt=""
<svg>
<rect />
<rect />
<rect />
<rect />
<rect />
</svg>
<script>
var data = [100, 250, 175, 200, 120];
d3.selectAll('rect')
.data(data)
.attr('x', (d, i) => i * rectWidth)
.attr('y', d => height - d)
.attr('width', rectWidth)
.attr('height', d => d)
.attr('fill', 'blue')
.attr('stroke', '#fff');
</script>
"Bind" data to the selections
data:image/s3,"s3://crabby-images/62159/6215915c02134d7a1fa77c946fa4d43f7ac40d86" alt=""
data:image/s3,"s3://crabby-images/76a2a/76a2ae8656ff021035a58a7d404817642c9adf0e" alt=""
data:image/s3,"s3://crabby-images/c5292/c52922d3c23c074e6269818a0cf6c2e9ad042ef7" alt=""
Selection & Data
<svg>
<rect />
<rect />
<rect />
<rect />
<rect />
</svg>
<script>
var data = [100, 250, 175, 200, 120];
d3.selectAll('rect')
.data(data)
.attr('x', (d, i) => i * rectWidth)
.attr('y', d => height - d)
.attr('width', rectWidth)
.attr('height', d => d)
.attr('fill', 'blue')
.attr('stroke', '#fff');
</script>
Loop through each rectangle selection
Get passed in (data, index)
data:image/s3,"s3://crabby-images/ca203/ca203b3e4fff03df39db894eca714467b2de3f42" alt=""
Selection & Data
<svg>
<rect />
<rect />
<rect />
<rect />
<rect />
</svg>
<script>
var data = [100, 250, 175, 200, 120];
d3.selectAll('rect')
.data(data)
.attr('x', (d, i) => i * rectWidth)
.attr('y', d => height - d)
.attr('width', rectWidth)
.attr('height', d => d)
.attr('fill', 'blue')
.attr('stroke', '#fff');
</script>
Selection & Data
Play with the code
data:image/s3,"s3://crabby-images/35c20/35c201c799fdb555251ba65444fdc69009b46cf1" alt=""
(Change some numbers, add/remove a rect, console log all the things)
Enter-append
<svg></svg>
<script>
var rectWidth = 100;
var height = 300;
var data = [100, 250, 175, 200, 120];
var svg = d3.select('svg');
svg.selectAll('rect')
.data(data)
.enter().append('rect')
.attr('x', (d, i) => i * rectWidth)
.attr('y', d => height - d)
.attr('width', rectWidth)
.attr('height', d => d)
.attr('fill', 'blue')
.attr('stroke', '#fff');
</script>
data:image/s3,"s3://crabby-images/c6267/c62678b6f7bcc3a256edd156e5474c079d878918" alt=""
Wut, no rectangle elements?!
So what are we even selecting?
A: an empty selection
So how are those bars appearing?
Enter-append
<svg></svg>
<script>
var rectWidth = 100;
var height = 300;
var data = [100, 250, 175, 200, 120];
var svg = d3.select('svg');
svg.selectAll('rect')
.data(data)
.enter().append('rect')
.attr('x', (d, i) => i * rectWidth)
.attr('y', d => height - d)
.attr('width', rectWidth)
.attr('height', d => d)
.attr('fill', 'blue')
.attr('stroke', '#fff');
</script>
data:image/s3,"s3://crabby-images/68d69/68d698690a4f4fe9593ec8f89a8f4e6b6fa25be9" alt=""
data:image/s3,"s3://crabby-images/ca9af/ca9af643bd3027fac4173ceb40ab5f1797ee64fc" alt=""
data:image/s3,"s3://crabby-images/8ed40/8ed40438eb590d2efb876709e84ce096adec683f" alt=""
Magic ✨
Enter-append
<svg></svg>
<script>
var rectWidth = 100;
var height = 300;
var data = [100, 250, 175, 200, 120];
var svg = d3.select('svg');
svg.selectAll('rect')
.data(data)
.enter().append('rect')
.attr('x', (d, i) => i * rectWidth)
.attr('y', d => height - d)
.attr('width', rectWidth)
.attr('height', d => d)
.attr('fill', 'blue')
.attr('stroke', '#fff');
</script>
Enter-append
Play with the code
data:image/s3,"s3://crabby-images/35c20/35c201c799fdb555251ba65444fdc69009b46cf1" alt=""
(Try adding/removing some numbers, and make sure to console log all the things)
let's make a flower
- Get data ready, and use the first movie data
- Create 6 petals using previously created paths
- Translate and rotate the petals
Scales & Axes
d3.scaleLinear()
.domain([min, max]) // input
.range([min, max]); // output
scale: mapping from
data attributes (domain)
to display (range)
data:image/s3,"s3://crabby-images/93e69/93e69ca549a599cad7e90f6c2113e13b5be9a31d" alt=""
date → x-value
value → y-value
value → opacity
etc.
Scales
// get min/max
var height = 600;
var data = [
{date: new Date('01-01-2015'), temp: 0},
{date: new Date('01-01-2017'), temp: 3}
];
var min = d3.min(data, d => d.date);
var max = d3.max(data, d => d.date);
// or use extent, which gives back [min, max]
var extent = d3.extent(data, d => d.date);
var yScale = d3.scaleLinear()
.domain([min, max])
.range([height, 0]);
Scales
// continuous
d3.scaleLinear()
d3.scaleLog()
d3.scaleTime()
// ordinal
d3.scaleBand()
Scales I use often:
Let's make a flower
Type of petal: parental guidance rating
Size of flower: IMDb rating out of 10
Number of petals: number of IMDb votes
- Create scale (scaleLinear and scaleQuantize)
- Get data ready, and use the first movie data
- Create number of petals based on data
- Rotate and scale petals
Hint: use SVG transform's translate, rotate, and scale (in that order)
update & exit
update & exit
// bars includes update selection
var bars = svg.selectAll('rect')
.data(data, d => d);
// exit
bars.exit().remove();
// enter
var enter = bars.enter().append('rect')
.attr('width', rectWidth)
.attr('stroke', '#fff');
// enter + update
bars = enter.merge(bars)
.attr('x', (d, i) => i * rectWidth)
.attr('y', d => height - d)
.attr('height', d => d)
.attr('fill', d => colors(d));
key function: controls which datum is assigned to which element
data:image/s3,"s3://crabby-images/9def9/9def98556e43c031f53a1f28ade4a33875ba2b51" alt=""
data:image/s3,"s3://crabby-images/497f1/497f1ad6ce837aac5c7ad68d2449f811cd92de43" alt=""
data:image/s3,"s3://crabby-images/da39f/da39fb76c8765260dd6bb42e23f72df00136c6e2" alt=""
data:image/s3,"s3://crabby-images/b7fc3/b7fc3027f32c9909c4c5624e63a430c44f49dc8b" alt=""
data:image/s3,"s3://crabby-images/7ff8d/7ff8d2d2badfc269ea41cf6df82f612cd73e87c7" alt=""
update & exit
// bars includes update selection
var bars = svg.selectAll('rect')
.data(data, d => d);
// exit
bars.exit().remove();
// enter
var enter = bars.enter().append('rect')
.attr('width', rectWidth)
.attr('stroke', '#fff');
// enter + update
bars = enter.merge(bars)
.attr('x', (d, i) => i * rectWidth)
.attr('y', d => height - d)
.attr('height', d => d)
.attr('fill', d => colors(d));
Enter selection: chain attributes that don't depend on data
Combines 2 selections into one
Enter+update selection: chain attrs dependent on data
Exit selection
Play with the code
data:image/s3,"s3://crabby-images/35c20/35c201c799fdb555251ba65444fdc69009b46cf1" alt=""
Update-exit
Let's update the flowers!
At every second, draw the next flower.
- Create function, within function:
- Calculate petal data for specific movie
- Bind data and draw flower with enter-update-exit pattern
Transitions
Emphasize changes in state
(object constancy)
Transitions
var t = d3.transition()
.duration(1000);
var svg = d3.select('svg');
var bars = svg.selectAll('rect')
.data(data, d => d);
// exit
bars.exit().transition(t)
.attr('y', height)
.attr('height', 0)
.remove();
// enter
var enter = bars.enter().append('rect')
.attr('width', rectWidth)
.attr('stroke', '#fff')
.attr('y', height);
// enter + update
bars = enter.merge(bars)
.attr('x', (d, i) => i * rectWidth)
.attr('fill', d => colors(d))
.transition(t)
.attr('y', d => height - d)
.attr('height', d => d);
Define transition, syncs animation everywhere it's used
Animate height down to 0 before removing
Animate remaining <rect>'s height to their next state
Transitions
var t = d3.transition()
.duration(1000);
var svg = d3.select('svg');
var bars = svg.selectAll('rect')
.data(data, d => d);
// exit
bars.exit().transition(t)
.attr('y', height)
.attr('height', 0)
.remove();
// enter
var enter = bars.enter().append('rect')
.attr('width', rectWidth)
.attr('stroke', '#fff')
.attr('y', height);
// enter + update
bars = enter.merge(bars)
.attr('x', (d, i) => i * rectWidth)
.attr('fill', d => colors(d))
.transition(t)
.attr('y', d => height - d)
.attr('height', d => d);
go from state A to state B
Everything after .transition(): state B to transition to
Same attributes set before .transition(): state A to transition from
If attributes not specified for A, d3 extrapolates from default
Transitions
Play with the code
data:image/s3,"s3://crabby-images/35c20/35c201c799fdb555251ba65444fdc69009b46cf1" alt=""
nesting elements
Nest elements when you want to apply the same styles and transforms to a group of elements
data:image/s3,"s3://crabby-images/1c675/1c675f618b57933d8c76631b7fd49ca75de7c180" alt=""
<g> element that consist of petals and leaves, has translate and scale applied
<path> element for each petal with rotate applied
nesting elements
const data = [
{fill: 'red', circle: [2, 25, 50]},
{fill: 'blue', circle: [10, 100, 150]},
];
const svg = d3.select("body").append("svg")
.attr("width", 960)
.attr("height", 500)
const groups = svg.selectAll('g')
.data(data).enter().append('g')
.attr('fill', d => d.fill)
.attr('transform', 'translate(10,10)');
const circles = groups.selectAll('circle')
.data(d => d.circle).enter().append('circle')
.attr('cx', d => d)
.attr('r', 10);
nesting elements
const data = [
{fill: 'red', circle: [2, 25, 50]},
{fill: 'blue', circle: [10, 100, 150]},
];
const svg = d3.select("body").append("svg")
.attr("width", 960)
.attr("height", 500)
const groups = svg.selectAll('g')
.data(data).enter().append('g')
.attr('fill', d => d.fill)
.attr('transform', 'translate(10,10)');
const circles = groups.selectAll('circle')
.data(d => d.circle).enter().append('circle')
.attr('cx', d => d)
.attr('r', 10);
Use <g> element to nest child elements *in SVG, only <g> elements can have children
data:image/s3,"s3://crabby-images/477d5/477d542f062c3d7d88d7d14ccf480cec28c76fbc" alt=""
nesting elements
const data = [
{fill: 'red', circle: [2, 25, 50]},
{fill: 'blue', circle: [10, 100, 150]},
];
const svg = d3.select("body").append("svg")
.attr("width", 960)
.attr("height", 500)
const groups = svg.selectAll('g')
.data(data).enter().append('g')
.attr('fill', d => d.fill)
.attr('transform', 'translate(10,10)');
const circles = groups.selectAll('circle')
.data(d => d.circle).enter().append('circle')
.attr('cx', d => d)
.attr('r', 10);
data:image/s3,"s3://crabby-images/cae1c/cae1cc6b3af191fe265824f2ffc36465caba7440" alt=""
const data = [
{fill: 'red', circle: [2, 25, 50]},
{fill: 'blue', circle: [10, 100, 150]},
];
const svg = d3.select("body").append("svg")
.attr("width", 960)
.attr("height", 500)
const groups = svg.selectAll('g')
.data(data).enter().append('g')
.attr('fill', d => d.fill)
.attr('transform', 'translate(10,10)');
const circles = groups.selectAll('circle')
.data(d => d.circle).enter().append('circle')
.attr('cx', d => d)
.attr('r', 10);
data:image/s3,"s3://crabby-images/26532/26532b2bb34bebd21f5203c34ceaddc0a2be2eb4" alt=""
nesting elements
const data = [
{fill: 'red', circle: [2, 25, 50]},
{fill: 'blue', circle: [10, 100, 150]},
];
const svg = d3.select("body").append("svg")
.attr("width", 960)
.attr("height", 500)
const groups = svg.selectAll('g')
.data(data).enter().append('g')
.attr('fill', d => d.fill)
.attr('transform', 'translate(10,10)');
const circles = groups.selectAll('circle')
.data(d => d.circle).enter().append('circle')
.attr('cx', d => d)
.attr('r', 10);
nesting elements
data:image/s3,"s3://crabby-images/ac0c2/ac0c2d4015d2cb7cf6f4e29b0dafb760bb1f9609" alt=""
Play with the code
data:image/s3,"s3://crabby-images/35c20/35c201c799fdb555251ba65444fdc69009b46cf1" alt=""
Nesting elements
Let's make all the flowers!
data:image/s3,"s3://crabby-images/030f4/030f49871b8ee98f4d215a379ced9f5456379bbe" alt=""
data:image/s3,"s3://crabby-images/45805/4580542b9c9a454f18c5660db4e0e425abac7c36" alt=""
Full-day creative data visualization workshop
By Shirley Wu
Full-day creative data visualization workshop
- 3,096