D3 Chart版本4标准化堆积条形图从垂直到水平

Sha*_*Moh 2 charts d3.js stacked-chart

这个问题非常类似于这个D3JS问题,但我使用的是最新的D3版本(// d3js.org/d3.v4.min.js).

我正在尝试将此标准化堆积条形图表 水平.在最新版本中是否有任何优化方式来实现这一目标?

我已经交换了x轴和y轴,如下所示

var svg = d3.select("svg"),
    margin = {top: 20, right: 60, bottom: 30, left: 40},
    width = +svg.attr("width") - margin.left - margin.right,
    height = +svg.attr("height") - margin.top - margin.bottom,
    g = svg.append("g").attr("transform", "translate(" + margin.left + "," + margin.top + ")");

var y = d3.scaleBand()
    .rangeRound([0, width])
    .padding(0.1)
    .align(0.1);

var x = d3.scaleLinear()
    .rangeRound([height, 0]);

var z = d3.scaleOrdinal()
    .range(["#98abc5", "#8a89a6", "#7b6888", "#6b486b", "#a05d56", "#d0743c", "#ff8c00"]);

var stack = d3.stack()
    .offset(d3.stackOffsetExpand);

d3.csv("data.csv", type, function(error, data) {
  if (error) throw error;

  data.sort(function(a, b) { return b[data.columns[1]] / b.total - a[data.columns[1]] / a.total; });

  y.domain(data.map(function(d) { return d.State; }));
  z.domain(data.columns.slice(1));

  var serie = g.selectAll(".serie")
    .data(stack.keys(data.columns.slice(1))(data))
    .enter().append("g")
      .attr("class", "serie")
      .attr("fill", function(d) { return z(d.key); });

  serie.selectAll("rect")
    .data(function(d) { return d; })
    .enter().append("rect")
      .attr("y", function(d) { return y(d.data.State); })
      .attr("x", function(d) { return x(d[1]); })
      .attr("height", function(d) { return x(d[0]) - x(d[1]); })
      .attr("width", y.bandwidth());

  g.append("g")
      .attr("class", "axis axis--y")
      .attr("transform", "translate(0," + height + ")")
      .call(d3.axisBottom(y));

  g.append("g")
      .attr("class", "axis axis--x")
      .call(d3.axisLeft(x).ticks(10, "%"));

  var legend = serie.append("g")
      .attr("class", "legend")
      .attr("transform", function(d) { var d = d[d.length - 1]; return "translate(" + (y(d.data.State) + y.bandwidth()) + "," + ((x(d[0]) + x(d[1])) / 2) + ")"; });

  legend.append("line")
      .attr("x1", -6)
      .attr("x2", 6)
      .attr("stroke", "#000");

  legend.append("text")
      .attr("y", 9)
      .attr("dy", "0.35em")
      .attr("fill", "#000")
      .style("font", "10px sans-serif")
      .text(function(d) { return d.key; });
});

function type(d, i, columns) {
  for (i = 1, t = 0; i < columns.length; ++i) t += d[columns[i]] = +d[columns[i]];
  d.total = t;
  return d;
}
Run Code Online (Sandbox Code Playgroud)

Cyr*_*ian 7

参考这个例子

您需要撤消域名:

  var y = d3.scaleBand()
    .rangeRound([0, width])
    .padding(0.1)
    .align(0.1);

  var x = d3.scaleLinear()
    .rangeRound([height, 0]);
Run Code Online (Sandbox Code Playgroud)

当域为反向时,将x交换为y,因此当您创建矩形时,x将变为y,y将变为x.

serie.selectAll("rect")
  .data(function(d) {
    return d;
  })
  .enter().append("rect")
  .attr("y", function(d) {
    return y(d.data.State);
  })
  .attr("x", function(d) {
    return x(d[1]);
  })
  .attr("width", function(d) {
    return x(d[0]) - x(d[1]);
  })
  .attr("height", y.bandwidth());
Run Code Online (Sandbox Code Playgroud)

相应地更改图例位置以将其放置在顶部栏上.

var legend = serie.append("g")
  .attr("class", "legend")
  .attr("transform", function(d) {
    var d = d[0];//get the top data for placing legends on that.
    return "translate(" +  ((x(d[0]) + x(d[1])) / 2) + ", " +(y(d.data.State) - y.bandwidth())+ ")";
  });
Run Code Online (Sandbox Code Playgroud)

最后定位图例线:

legend.append("line")
  .attr("y1", 5)
  .attr("x1", 15)
  .attr("x2", 15)
  .attr("y2", 12)
  .attr("stroke", "#000");
Run Code Online (Sandbox Code Playgroud)

这里工作代码