Skip to content

Instantly share code, notes, and snippets.

@rmarimon
Created August 13, 2011 17:10
Show Gist options
  • Star 3 You must be signed in to star a gist
  • Fork 1 You must be signed in to fork a gist
  • Save rmarimon/1144047 to your computer and use it in GitHub Desktop.
Save rmarimon/1144047 to your computer and use it in GitHub Desktop.
Dial Chart examples
Dial Chart examples. Suggestions and enhancements are welcome.
//
// DialChart
//
NBXDialChart = function() {
var w = 256,
h = 256,
m = [ 0, 0, 0, 0 ], // top right bottom left
domain = [0, 1],
range = [-135, 135],
minorTicks = 4,
minorMark = 'line'
dial = [ 1.00, 0.95, 0.92, 0.85 ],
scale = [ 0.71, 0.75, 0.76 ],
needle = [ 0.83, 0.05 ],
pivot = [ 0.10, 0.05 ];
function dialchart(g) {
g.each(function(d, i) {
var wm = w - m[1] - m[3],
hm = h - m[0] - m[2],
a = d3.scale.linear().domain(domain).range(range);
var r = Math.min(wm / 2, hm / 2);
var g = d3.select(this).select('g');
if (g.empty()) {
g = d3.select(this).append('svg:g')
.attr('transform', 'translate(' + (m[3] + wm / 2) + ',' + (m[0] + hm / 2) + ')');
createDefs(g.append('svg:defs'));
// face
g.append('svg:circle')
.attr('r', r * dial[0])
.style('fill', 'url(#outerGradient)')
.attr('filter', 'url(#dropShadow)');
g.append('svg:circle')
.attr('r', r * dial[1])
.style('fill', 'url(#innerGradient)');
g.append('svg:circle')
.attr('r', r * dial[2])
.style('fill', 'url(#faceGradient)');
// scale
var major = a.ticks(10);
var minor = a.ticks(10 * minorTicks).filter(function(d) { return major.indexOf(d) == -1; });
g.selectAll('text.label')
.data(major)
.enter().append('svg:text')
.attr('class', 'label')
.attr('x', function(d) { return Math.cos( (-90 + a(d)) / 180 * Math.PI) * r * scale[1]; })
.attr('y', function(d) { return Math.sin( (-90 + a(d)) / 180 * Math.PI) * r * scale[1]; })
.attr('text-anchor', 'middle')
.attr('dy', '0.5em')
.text(a.tickFormat());
if (minorMark == 'circle') {
g.selectAll('circle.label')
.data(minor)
.enter().append('svg:circle')
.attr('class', 'label')
.attr('cx', function(d) { return Math.cos( (-90 + a(d)) / 180 * Math.PI) * (r * scale[1]); })
.attr('cy', function(d) { return Math.sin( (-90 + a(d)) / 180 * Math.PI) * (r * scale[1]); })
.attr('r', 2);
}
if (minorMark == 'line') {
g.selectAll('line.label')
.data(minor)
.enter().append('svg:line')
.attr('class', 'label')
.attr('x1', function(d) { return Math.cos( (-90 + a(d)) / 180 * Math.PI) * (r * scale[0]); })
.attr('y1', function(d) { return Math.sin( (-90 + a(d)) / 180 * Math.PI) * (r * scale[0]); })
.attr('x2', function(d) { return Math.cos( (-90 + a(d)) / 180 * Math.PI) * (r * scale[2]); })
.attr('y2', function(d) { return Math.sin( (-90 + a(d)) / 180 * Math.PI) * (r * scale[2]); });
}
var rdial3 = r * dial[3];
g.append('svg:path')
.attr('class', 'dial-glare')
.attr('d', 'M ' + (-rdial3) + ',0 A' + rdial3 + ',' + rdial3 + ' 0 0,1 ' + rdial3 + ',0 A' + (rdial3 * 5) + ',' + (rdial3 * 5) + ' 0 0,0 ' + (-rdial3) + ',0')
.style('fill', 'url(#glareGradient)');
// needle
var n = g.append('svg:g')
.attr('class', 'needle')
.attr('filter', 'url(#dropShadow)')
.attr('transform', function(d) { return 'rotate(' + a(d) + ')'; });
n.append('svg:path')
.attr('d', 'M ' + (-r * needle[1]) + ',0 L0,' + (-r * needle[0])+ ' L' + r * needle[1] + ',0');
n.append('svg:circle')
.attr('r', r * pivot[0])
.style('fill', 'url(#outerGradient)');
n.append('svg:circle')
.attr('r', r * pivot[1])
.style('fill', 'url(#innerGradient)');
} else {
g.select('g.needle')
.transition().ease('elastic')
.attr('transform', function(d) { return 'rotate(' + a(d) + ')'; });
}
});
d3.timer.flush();
}
function createDefs(defs) {
var outerGradient = defs.append('svg:linearGradient')
.attr('id', 'outerGradient')
.attr('x1', '0%').attr('y1', '0%')
.attr('x2', '0%').attr('y2', '100%')
.attr('spreadMethod', 'pad');
outerGradient.selectAll('stop')
.data([{ o: '0%', c: '#ffffff' }, { o: '100%', c: '#d0d0d0' }])
.enter().append('svg:stop')
.attr('offset', function(d) { return d.o; })
.attr('stop-color', function(d) { return d.c; })
.attr('stop-opacity', '1');
var innerGradient = defs.append('svg:linearGradient')
.attr('id', 'innerGradient')
.attr('x1', '0%').attr('y1', '0%')
.attr('x2', '0%').attr('y2', '100%')
.attr('spreadMethod', 'pad');
innerGradient.selectAll('stop')
.data([{ o: '0%', c: '#d0d0d0' }, { o: '100%', c: '#ffffff' }])
.enter().append('svg:stop')
.attr('offset', function(d) { return d.o; })
.attr('stop-color', function(d) { return d.c; })
.attr('stop-opacity', '1');
var faceGradient = defs.append('svg:linearGradient')
.attr('id', 'faceGradient')
.attr('x1', '0%').attr('y1', '0%')
.attr('x2', '0%').attr('y2', '100%')
.attr('spreadMethod', 'pad');
faceGradient.selectAll('stop')
.data([{ o: '0%', c: '#000000' }, { o: '100%', c: '#494949' }])
.enter().append('svg:stop')
.attr('offset', function(d) { return d.o; })
.attr('stop-color', function(d) { return d.c; })
.attr('stop-opacity', '1');
var glareGradient = defs.append('svg:linearGradient')
.attr('id', 'glareGradient')
.attr('x1', '0%').attr('y1', '0%')
.attr('x2', '0%').attr('y2', '100%')
.attr('spreadMethod', 'pad');
glareGradient.selectAll('stop')
.data([{ o: '0%', c: '#ffffff', op: 0.60 }, { o: '100%', c: '#ffffff', op: 0.00 }])
.enter().append('svg:stop')
.attr('offset', function(d) { return d.o; })
.attr('stop-color', function(d) { return d.c; })
.attr('stop-opacity', function(d) { return d.op; });
var dropShadowFilter = defs.append('svg:filter')
.attr('id', 'dropShadow');
dropShadowFilter.append('svg:feGaussianBlur')
.attr('in', 'SourceAlpha')
.attr('stdDeviation', 3);
dropShadowFilter.append('svg:feOffset')
.attr('dx', 2)
.attr('dy', 2)
.attr('result', 'offsetblur');
var feMerge = dropShadowFilter.append('svg:feMerge');
feMerge.append('svg:feMergeNode');
feMerge.append('svg:feMergeNode')
.attr('in', "SourceGraphic");
}
dialchart.width = function(d) {
if (!arguments.length) return w;
w = d;
return dialchart;
};
dialchart.height = function(d) {
if (!arguments.length) return h;
h = d;
return dialchart;
};
dialchart.margin = function(d) {
if (!arguments.length) return m;
m = d;
return dialchart;
};
dialchart.domain = function(d) {
if (!arguments.length) return domain;
domain = d;
return dialchart;
};
dialchart.range = function(d) {
if (!arguments.length) return range;
range = d;
return dialchart;
};
dialchart.scale = function(d) {
if (!arguments.length) return scale;
scale = d;
return dialchart;
};
dialchart.minorTicks = function(d) {
if (!arguments.length) return minorTicks;
minorTicks = d;
return dialchart;
};
dialchart.minorMark = function(d) {
if (!arguments.length) return minorMark;
minorMark = d;
return dialchart;
};
return dialchart;
};
<!doctype html>
<html>
<head>
<meta http-equiv="Content-Type" content="text/html;charset=utf-8" />
<title>dials</title>
<script type="text/javascript" src="http://mbostock.github.com/d3/d3.min.js"></script>
<script type="text/javascript" src="dial.chart.js"></script>
<style type="text/css">
#dial-0 .needle path {
fill: beige;
}
#dial-1 .needle path {
fill: #b21f24;
}
#dial-2 .needle path {
fill: steelblue;
}
circle.label {
fill: white;
}
line.label {
stroke: white;
stroke-width: 1px;
}
text.label {
font-family: Arial;
font-size: 12px;
fill: white;
}
#dial-1 text.label {
font-size: 16px;
}
#dial-2 text.label {
font-size: 14px;
}
</style>
</head>
<body style="background-color:#009F01">
<button id="update" onclick="transition()">Update</button>
<div id="chart">
</div>
<script type="text/javascript">
(function(chartselector) {
var w = 960,
h = 500;
var layout = [
{ x: 150, y: 250, r: 80, m: 100, ticks: 2, mark: 'line' },
{ x: 460, y: 250, r: 200, m: 50, ticks: 4, mark: 'line' },
{ x: 810, y: 250, r: 120, m: 80, ticks: 2, mark: 'circle' }
];
var charts = layout.map(function(d) {
return NBXDialChart()
.width(d.r * 2)
.height(d.r * 2)
.domain([0, d.m])
.range([-150, 150])
.minorTicks(d.ticks)
.minorMark(d.mark);
});
var svg = d3.select(chartselector)
.append('svg:svg')
.attr('width', w)
.attr('height', h);
var dials = svg.selectAll('g.dial')
.data(layout)
.enter().append('svg:g')
.attr('class', 'dial')
.attr('id', function(d, i) { return 'dial-' + i; })
.attr('transform', function(d) { return 'translate(' + (d.x - d.r) + ',' + (d.y - d.r) + ')'; } );
dials.each(function(d, i) { d3.select(this).data([20]).call(charts[i]); });
window.transition = function() {
dials.each(function(d, i) {
d3.select(this)
.data([ Math.random() * charts[i].domain()[1] ])
.call(charts[i]);
});
};
})('#chart');
</script>
</body>
</html>
@guitar1999
Copy link

Thanks for this! Would it be possible to add a label on the dial to indicate what is being shown?

@somenathghosh
Copy link

Nice. Can you please show how to add text lable inside each dial ?

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment