d3.js - d3 水平条形图,背景为 100%

标签 d3.js

我有这个单一水平条形图,我想进行以下调整:

  1. 将刻度值显示在条形图的右侧,而不是轴旁边。
  2. 显示条形图刻度的背景,而不是左轴和下轴。

当前版本: enter image description here

我想要达到的目标:

enter image description here

JS

var data = [
      {"yAxis":"score", "xAxis":"72"}
    ];          
    // set the dimensions and margins of the graph
    var margin = {top: 20, right: 20, bottom: 30, left: 80},
    width = 960 - margin.left - margin.right,
    height = 500 - margin.top - margin.bottom;          
    // set the ranges
    var y = d3.scaleBand()
    .range([height, 0])
    .padding(0.4);
    var x = d3.scaleLinear()
    .range([0, width]);          
    var svg = d3.select(".barChartContainer").append("svg")          
    .attr("preserveAspectRatio", "xMinYMin meet")
    .attr("viewBox", "0 0 960 500")
    .append("g")
    .attr("transform",
    "translate(" + margin.left + "," + margin.top + ")");        
    // Scale the range of the data in the domains
    x.domain([0, d3.max(data, function(d){ return d.xAxis; })])
    y.domain(data.map(function(d) { return d.yAxis; }));        
    //y.domain([0, d3.max(data, function(d) { return d.prereqs; })]);
    // append the rectangles for the bar chart
    svg.selectAll(".bar")
    .data(data)
    .enter().append("rect")
    .attr("class", "bar")          
    .attr("y", function(d) { return y(d.yAxis); })
    .attr("height", y.bandwidth())
    .transition()
    .duration(1000)
    .delay(function(d, i) {
    return i * 100
    })
    .attr("width", function(d) {return x(d.xAxis); } );
    // add the x Axis
    svg.append("g")
    .attr("transform", "translate(0," + height + ")")
    .call(d3.axisBottom(x).tickValues(d3.range(x.domain()[0], x.domain()[1] + 1, 1))
    .tickFormat(d3.format("d"))
    );
    svg.append("g")
    .attr("class", "yAxis")
    .call(d3.axisLeft(y));

我从 codepen 获得了这段代码,我一直在尝试修改它,但它已经被破坏了,所以我停下来,希望你能提供帮助。

谢谢。

最佳答案

要显示条形的背景,只需复制您的选择并为矩形选择 100% 的值,并采用浅灰色填充:

var backgroundBar = svg.selectAll(null)
    .data(data)
    .enter()
    .append("rect")
    //etc...
    .attr("width", function(d) {
        return x(100);
    });

当然,您必须更改 x 比例的范围:

var x = d3.scaleLinear()
    .domain([0, 100]);

然后,放下两个轴并使用文本选择打印标签。

最后,使用另一个文本选择作为值:

var values = svg.selectAll(null)
    .data(data)
    .enter()
    .append("text")
    //etc...
    .text(function(d) {
        return +d.xAxis
    })

如果需要,您可以补间文本:

.attrTween("text", function(d) {
    var self = this
    var i = d3.interpolateNumber(0, +d.xAxis);
    return function(t) {
        return d3.select(self).text(~~i(t));
    }
});

这是结果:

var data = [{
  "yAxis": "score",
  "xAxis": "72"
}];

var margin = {
    top: 20,
    right: 20,
    bottom: 30,
    left: 80
  },
  width = 500 - margin.left - margin.right,
  height = 200 - margin.top - margin.bottom;

var y = d3.scaleBand()
  .range([height, 0])
  .padding(0.4);

var x = d3.scaleLinear()
  .range([0, width])
  .domain([0, 100]);

var svg = d3.select("body").append("svg")
  .attr("width", 500)
  .attr("height", 200)
  .append("g")
  .attr("transform",
    "translate(" + margin.left + "," + margin.top + ")");

y.domain(data.map(function(d) {
  return d.yAxis;
}));

var backgroundBar = svg.selectAll(null)
  .data(data)
  .enter()
  .append("rect")
  .attr("fill", "lightgray")
  .attr("y", function(d) {
    return y(d.yAxis);
  })
  .attr("height", y.bandwidth())
  .attr("width", function(d) {
    return x(100);
  });

var bar = svg.selectAll(null)
  .data(data)
  .enter()
  .append("rect")
  .attr("class", "bar")
  .attr("y", function(d) {
    return y(d.yAxis);
  })
  .attr("height", y.bandwidth())
  .transition()
  .duration(2000)
  .delay(function(d, i) {
    return i * 100
  })
  .attr("width", function(d) {
    return x(d.xAxis);
  });

var labels = svg.selectAll(null)
  .data(data)
  .enter()
  .append("text")
  .attr("y", function(d) {
    return y(d.yAxis) + y.bandwidth() / 2;
  })
  .attr("x", -10)
  .attr("text-anchor", "end")
  .text(function(d) {
    return d.yAxis
  });

var values = svg.selectAll(null)
  .data(data)
  .enter()
  .append("text")
  .attr("y", function(d) {
    return y(d.yAxis) + y.bandwidth() / 2;
  })
  .attr("x", 10)
  .text(function(d) {
    return +d.xAxis
  })
  .transition()
  .duration(2000)
  .delay(function(d, i) {
    return i * 100
  })
  .attr("x", function(d) {
    return x(d.xAxis) + 10;
  })
  .attrTween("text", function(d) {
    var self = this
    var i = d3.interpolateNumber(0, +d.xAxis);
    return function(t) {
      return d3.select(self).text(~~i(t));
    }
  });
<script src="https://d3js.org/d3.v4.min.js"></script>

关于d3.js - d3 水平条形图,背景为 100%,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/44986642/

相关文章:

html - 在 D3 中鼠标悬停时突出显示表格的列的最有效方法是什么?

javascript - d3 .tsv 文件不起作用

javascript - 过滤合并两个数组,匹配两个字段/列 : Javascript

javascript - D3js hive 图 : How to label nodes?

javascript - d3.js:无法读取未定义的属性 'call'

javascript - SVG:Windows 和 Linux 上的字体间距非常不同

javascript - 在 D3 map 中显示事件列表

javascript - 具有缩放和可更新数据错误的分组条形图

javascript - D3 力布局更改显示与 slider 的链接

javascript - 尝试使用模块模式来安全地获取数据