Published
Edited
May 8, 2019
23 forks
81 stars
Insert cell
Insert cell
Insert cell
Insert cell
{
const svg = d3.select(DOM.svg(width + margin.left + margin.right, height + margin.left + margin.right));
svg
.append("rect")
.attr("width", "100%")
.attr("height", "100%")
.style("fill", "#F5F5F2");
const voronoi = svg.append("g").attr("transform", "translate(" + margin.left + "," + margin.top + ")");
const labels = svg.append("g").attr("transform", "translate(" + margin.left + "," + margin.top + ")");
const pop_labels = svg.append("g").attr("transform", "translate(" + margin.left + "," + margin.top + ")");
let seed = new Math.seedrandom(20);
let voronoiTreeMap = d3.voronoiTreemap()
.prng(seed)
.clip(ellipse);
voronoiTreeMap(population_hierarchy);
colorHierarchy(population_hierarchy);
let allNodes = population_hierarchy.descendants()
.sort((a, b) => b.depth - a.depth)
.map((d, i) => Object.assign({}, d, {id: i}));
let hoveredShape = null;
//return allNodes;
voronoi.selectAll('path')
.data(allNodes)
.enter()
.append('path')
.attr('d', d => "M" + d.polygon.join("L") + "Z")
.style('fill', d => d.parent ? d.parent.color : d.color)
.attr("stroke", "#F5F5F2")
.attr("stroke-width", 0)
.style('fill-opacity', d => d.depth === 2 ? 1 : 0)
.attr('pointer-events', d => d.depth === 2 ? 'all' : 'none')
.on('mouseenter', d => {
let label = labels.select(`.label-${d.id}`);
label.attr('opacity', 1)
let pop_label = pop_labels.select(`.label-${d.id}`);
pop_label.attr('opacity', 1)
})
.on('mouseleave', d => {
let label = labels.select(`.label-${d.id}`);
label.attr('opacity', d => d.data.population > 130000000 ? 1 : 0)
let pop_label = pop_labels.select(`.label-${d.id}`);
pop_label.attr('opacity', d => d.data.population > 130000000 ? 1 : 0)
})
.transition()
.duration(1000)
.attr("stroke-width", d => 7 - d.depth*2.8)
.style('fill', d => d.color);
labels.selectAll('text')
.data(allNodes.filter(d => d.depth === 2 ))
.enter()
.append('text')
.attr('class', d => `label-${d.id}`)
.attr('text-anchor', 'middle')
.attr("transform", d => "translate("+[d.polygon.site.x, d.polygon.site.y+6]+")")
.text(d => d.data.key || d.data.countries)
//.attr('opacity', d => d.data.key === hoveredShape ? 1 : 0)
.attr('opacity', function(d) {if(d.data.key === hoveredShape){return(1);
}else if(d.data.population > 130000000) {
return(1);
}else {return(0);}})
.attr('cursor', 'default')
.attr('pointer-events', 'none')
.attr('fill', 'black')
.style('font-family', 'Montserrat');
pop_labels.selectAll('text')
.data(allNodes.filter(d => d.depth === 2 ))
.enter()
.append('text')
.attr('class', d => `label-${d.id}`)
.attr('text-anchor', 'middle')
.attr("transform", d => "translate("+[d.polygon.site.x, d.polygon.site.y+25]+")")
.text(d => bigFormat(d.data.population))
//.attr('opacity', d => d.data.key === hoveredShape ? 1 : 0)
.attr('opacity', function(d) {if(d.data.key === hoveredShape){return(1);
}else if(d.data.population > 130000000) {
return(1);
}else {return(0);}})
.attr('cursor', 'default')
.attr('pointer-events', 'none')
.attr('fill', 'black')
.style('font-size', '12px')
.style('font-family', 'Montserrat');
return svg.node();
}
Insert cell
Insert cell
height = 750 - margin.top - margin.bottom
Insert cell
width = 750 - margin.left - margin.right
Insert cell
margin = ({top: 20, right: 20, bottom: 50, left: 50})
Insert cell
bigFormat = d3.format(",.0f")
Insert cell
html`<link href="https://fonts.googleapis.com/css?family=Montserrat" rel="stylesheet">`
Insert cell
ellipse = d3
.range(100)
.map(i => [
(width * (1 + 0.99 * Math.cos((i / 50) * Math.PI))) / 2,
(height * (1 + 0.99 * Math.sin((i / 50) * Math.PI))) / 2
])
Insert cell
minYear = d3.min(freedom, d => d.year)
Insert cell
maxYear = d3.max(freedom, d => d.year)
Insert cell
regionColor = function(region) {
var colors = {
"Middle East and Africa": "#596F7E",
"Americas": "#168B98",
"Asia": "#ED5B67",
"Oceania": "#fd8f24",
"Europe": "#919c4c"
};
return colors[region];
}
Insert cell
function colorHierarchy(hierarchy) {
if(hierarchy.depth === 0) {
hierarchy.color = 'black';
} else if(hierarchy.depth === 1){
hierarchy.color = regionColor(hierarchy.data.key);
} else {
hierarchy.color = hierarchy.parent.color;
}
if(hierarchy.children) {
hierarchy.children.forEach( child => colorHierarchy(child))
}
}
Insert cell
Insert cell
freedom = d3.csv("https://gist.githubusercontent.com/will-r-chase/16827fa79e02af9e3a0651fb0d79b426/raw/92b321a8bc4d98e463156ef03a5da5cf05065704/freedom_clean.csv", d3.autoType)
Insert cell
freedom_year = freedom.filter(obj => {return obj.year === selectedYear})
Insert cell
data_nested = {
let freedom_nest = d3.nest()
.key(d => d.region_simple)
.entries(freedom_year)
return {key: "freedom_nest", values: freedom_nest}
}
Insert cell
population_hierarchy = d3.hierarchy(data_nested, d => d.values)
.sum(d => d.population)
Insert cell
Insert cell
d3 = require("d3@5", "d3-weighted-voronoi", "d3-voronoi-map", "d3-voronoi-treemap", 'seedrandom@2.4.3/seedrandom.min.js')
Insert cell
import { slider, radio, text } from '@jashkenas/inputs'
Insert cell

One platform to build and deploy the best data apps

Experiment and prototype by building visualizations in live JavaScript notebooks. Collaborate with your team and decide which concepts to build out.
Use Observable Framework to build data apps locally. Use data loaders to build in any language or library, including Python, SQL, and R.
Seamlessly deploy to Observable. Test before you ship, use automatic deploy-on-commit, and ensure your projects are always up-to-date.
Learn more