admin管理员组文章数量:1344179
This D3
example served as my starting point:
I wanted to change data that feeds the diagram, and I made following new example:
/
One can notice at least two problems:
- Legend is wrong. This is because it still contains 'hardcoded' names from original example.
- All nodes are colored black. This is because the color scheme is also 'hardcoded' only for node names from original example.
How to improve the original example (or my jsfiddle, it doesn't matter) so that legend and coloring are self-adjusted to the data that feeds the diagram?
This D3
example served as my starting point:
http://bl.ocks/kerryrodden/7090426
I wanted to change data that feeds the diagram, and I made following new example:
http://jsfiddle/ZGVK3/
One can notice at least two problems:
- Legend is wrong. This is because it still contains 'hardcoded' names from original example.
- All nodes are colored black. This is because the color scheme is also 'hardcoded' only for node names from original example.
How to improve the original example (or my jsfiddle, it doesn't matter) so that legend and coloring are self-adjusted to the data that feeds the diagram?
Share Improve this question edited Apr 29, 2015 at 12:08 VividD asked Jul 5, 2014 at 0:59 VividDVividD 10.5k8 gold badges66 silver badges112 bronze badges 1- 1 As far as handling colors, I deal with that here, at least to some degree. It is based on the array of page types which could be prised of any names. It is limited to a relatively small number of page types though. In any case, take a look. – FernOfTheAndes Commented Jul 5, 2014 at 2:02
1 Answer
Reset to default 10You can use an ordinal scale to map colors to the different node names. Implementing it would only require a few minor changes to your existing code.
Step 1. Create an ordinal scale for the colors
Instead of having colors
be simply a list of color names, hard-coded to specific names, use d3.scale.ordinal()
, and set the .range()
to be an array of the colors you want to use. For example:
var colors = d3.scale.ordinal()
.range(["#5687d1","#7b615c","#de783b","#6ab975","#a173d1","#bbbbbb"]);
This would create an ordinal scale that uses the same colors as the original visualization. Since your data would require more colors, you would want to add a few more to your range, otherwise colors will be repeated.
As a shortcut, you can use d3.scale.category20()
to let d3 choose a range 20 categorical colors for you.
Now when setting the fill colors for your path
element arcs and also your breadcrumbs, you would simply use colors(d.name)
instead of colors[d.name]
.
Step 2. Use your data to construct the domain of the scale
The .domain()
of this scale will be set once we have the data, since it will depend on a list of the unique names contained in the data. To do this, we can loop through the data, and create an array of the unique names. There are probably several ways to do this, but here's one that works well:
var uniqueNames = (function(a) {
var output = [];
a.forEach(function(d) {
if (output.indexOf(d.name) === -1) {
output.push(d.name);
}
});
return output;
})(nodes);
This creates an empty array, then loops through each element of the nodes
array and if the node's name doesn't already exist in the new array, it is added.
Then you can simply set the new array to be the domain of the color scale:
colors.domain(uniqueNames);
Step 3. Use the scale's domain to build the legend
Since the legend is going to depend on the domain, make sure the drawLegend()
function is called after the domain is set.
You can find the number of elements in the domain (for setting the height of the legend) by calling colors.domain().length
. Then for the legend's .data()
, you can use the domain itself. Finally, to set the fill color for the legend boxes, you call the color scale on d
since each element in the domain is a name
. Here's what those three changes to the legend look like in practice:
var legend = d3.select("#legend").append("svg:svg")
.attr("width", li.w)
.attr("height", colors.domain().length * (li.h + li.s));
var g = legend.selectAll("g")
.data(colors.domain())
.enter().append("svg:g")
.attr("transform", function(d, i) {
return "translate(0," + i * (li.h + li.s) + ")";
});
g.append("svg:rect")
.attr("rx", li.r)
.attr("ry", li.r)
.attr("width", li.w)
.attr("height", li.h)
.style("fill", function(d) { return colors(d); });
And that's about it. Hope that helps.
Here's the updated JSFiddle.
本文标签: javascriptImproving D3 Sequence Sunburst ExampleStack Overflow
版权声明:本文标题:javascript - Improving D3 Sequence Sunburst Example - Stack Overflow 内容由网友自发贡献,该文观点仅代表作者本人, 转载请联系作者并注明出处:http://www.betaflare.com/web/1743699006a2523994.html, 本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌抄袭侵权/违法违规的内容,一经查实,本站将立刻删除。
发表评论