Skip to content

Instantly share code, notes, and snippets.

@rshaker
Last active December 7, 2023 10:41
Show Gist options
  • Star 1 You must be signed in to star a gist
  • Fork 0 You must be signed in to fork a gist
  • Save rshaker/225c6df494811f46f6ea53eba63da817 to your computer and use it in GitHub Desktop.
Save rshaker/225c6df494811f46f6ea53eba63da817 to your computer and use it in GitHub Desktop.
Update and transition of pie chart
license: mit
<!DOCTYPE html>
<html>
<head>
<script src='https://d3js.org/d3.v4.min.js'></script>
<script src='https://d3js.org/d3-scale-chromatic.v1.min.js'></script>
</head>
<body>
<div id="pie"/>
<script>
var keys = [
"White"
, "Unknown"
, "Black or African American"
, "American Indian or Alaska Native"
, "Asian"
, "Native Hawaiian or Other Pacific Islander"];
var width = 250,
height = 250,
radius = Math.min(width, height) / 2;
var svg = d3.select("#pie")
.append("svg")
.attr("width", width)
.attr("height", height)
.append("g")
.attr("transform", "translate(" + width / 2 + "," + height / 2 + ")");
svg.append("g").attr("class", "slices");
var pie = d3.pie()
.sort(null)
.value(function(d) {
return d.value;
});
var arc = d3.arc()
.outerRadius(radius * 1.0)
.innerRadius(radius * 0.0);
var outerArc = d3.arc()
.innerRadius(radius * 0.5)
.outerRadius(radius * 1);
var key = function(d) { return d.data.label; };
var color = d3.scaleOrdinal(d3.schemePastel1)
.domain(keys);
update(makeData());
var inter = setInterval(function() {
update(makeData());
}, 2000);
function mergeWithFirstEqualZero(first, second){
var secondSet = d3.set();
second.forEach(function(d) { secondSet.add(d.label); });
var onlyFirst = first
.filter(function(d){ return !secondSet.has(d.label) })
.map(function(d) { return {label: d.label, value: 0}; });
var sortedMerge = d3.merge([ second, onlyFirst ])
.sort(function(a, b) {
return d3.ascending(a.label, b.label);
});
return sortedMerge;
}
function makeData() {
var data = Array();
for (i = 0; i < keys.length; i++) {
if (Math.random() < 0.7) {
var ob = {};
ob["label"] = keys[i];
ob["value"] = randomCount(1, 100);
data.push(ob);
}
}
var sortedData = data.sort(function(a, b) {
return d3.ascending(a.label, b.label);
});
return sortedData;
}
function randomCount(min, max) {
return Math.floor(Math.random() * (max - min + 1)) + min;
}
function update(data) {
var duration = 500;
var oldData = svg.select(".slices")
.selectAll("path")
.data().map(function(d) { return d.data });
if (oldData.length == 0) oldData = data;
var was = mergeWithFirstEqualZero(data, oldData);
var is = mergeWithFirstEqualZero(oldData, data);
var slice = svg.select(".slices")
.selectAll("path")
.data(pie(was), key);
slice.enter()
.insert("path")
.attr("class", "slice")
.style("fill", function(d) { return color(d.data.label); })
.each(function(d) {
this._current = d;
});
slice = svg.select(".slices")
.selectAll("path")
.data(pie(is), key);
slice.transition()
.duration(duration)
.attrTween("d", function(d) {
var interpolate = d3.interpolate(this._current, d);
var _this = this;
return function(t) {
_this._current = interpolate(t);
return arc(_this._current);
};
});
slice = svg.select(".slices")
.selectAll("path")
.data(pie(data), key);
slice.exit()
.transition()
.delay(duration)
.duration(0)
.remove();
};
</script>
</body>
</html>
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment