如何在D3.js中的图表上设置滑块样式并显示动态y值的标签?

Rya*_*yan 5 javascript css jquery d3.js

这是一个简单图表的示例小提琴.

我是来学习D3.js,我它,我用它和它的创始人迈克·博斯托克如此深刻的印象.

我创建了一个图表网页,这些图表的价值相互影响,有点像Bostock 在纽约时报的惊人的Rent Versus Buy计算器.

我试图创建一个滑块(使用JqueryUI)来更改输入值(如下面显示的$ 250,000输入框),这会影响图表的y值.它有效,但外观令人失望,不适合移动设备.

我一直无法通过教程或文档弄清楚如何创建像Bostock这样的滑块,如下所示:

我不认为Bostock使用除D3以外的任何东西来构建他的滑块和样式.

我试图查看他的源代码,但没有运气搞清楚.

我的示例小提琴中,我使用的是JqueryUI slider($("<div class='slider'></div>").insertAfter(input).slider())和一些CSS.

我还没有找到D3的功能可以帮助我的图表看起来更像他的吗?

如果你可以指出我如何创建(和样式)滑块,y值标签和轴的正确方向,我真的很感激.

Ger*_*ado 5

这是我从头开始编写的一个非常简单的演示,因此不是基于小提琴中的代码,它展示了您想要创建的数据可视化的基本方面。

在这个演示中,我仅使用 D3,没有使用 jQuery(将 D3 与 jQuery 混合让我头疼):

var width = 500,
    height = 180,
    padding = 16;

var svg = d3.select("#svg")
    .append("svg")
    .attr("width", width)
    .attr("height", height);

var data = d3.range(1, 21);

var xScale = d3.scaleBand()
    .domain(data)
    .range([padding * 2, width - padding])
    .padding(0.2);

var yScale = d3.scaleLinear()
    .domain([0, d3.max(data, d => d)])
    .range([height - padding, padding]);

var bars = svg.selectAll(".bars")
    .data(data)
    .enter()
    .append("rect")
    .attr("x", d => xScale(d))
    .attr("width", xScale.bandwidth())
    .attr("y", d => yScale(d))
    .attr("height", d => height - padding - yScale(d))
    .attr("fill", (d => d3.select("#slider").node().value == d ? "firebrick" : "teal"));

var xAxis = d3.axisBottom(xScale);
var yAxis = d3.axisLeft(yScale);

var gX = svg.append("g")
    .attr("transform", "translate(0," + (height - padding) + ")")
    .call(xAxis);

var gY = svg.append("g")
    .attr("transform", "translate(" + padding * 2 + ",0)")
    .call(yAxis);

d3.select("#slider").on("input", function() {
    var currentValue = this.value;
    yScale.domain([0, currentValue * 2])
    bars.attr("y", d => yScale(d))
        .attr("height", d => height - padding - yScale(d))
        .attr("fill", (d => currentValue == d ? "firebrick" : "teal"));
    gY.call(yAxis);
})
Run Code Online (Sandbox Code Playgroud)
#slider {
  width: 435px;
}
#sliderdiv{
  padding-left: 40px;
 }
Run Code Online (Sandbox Code Playgroud)
<script src="https://d3js.org/d3.v4.min.js"></script>
<div id="svg"></div>
<div id="sliderdiv"><input id="slider" type="range" min="1" max="20" step="1" value="10"/></div>
Run Code Online (Sandbox Code Playgroud)

解释:

在此示例中,我对滑块使用常规 HTML 输入:

<input id="slider" type="range" min="1" max="20" step="1" value="10"/>
Run Code Online (Sandbox Code Playgroud)

然后,我们使用以下函数获取滑块中的更改:

d3.select("#slider").on("input", function() {
    var currentValue = this.value;
    yScale.domain([0, currentValue * 2])
    bars.attr("y", d => yScale(d))
        .attr("height", d => height - padding - yScale(d))
        .attr("fill", (d => currentValue == d ? "firebrick" : "teal"));
    gY.call(yAxis);
})
Run Code Online (Sandbox Code Playgroud)

在该函数中,我们有 3 个主要步骤。首先,我们获取滑块的当前值:

var currentValue = this.value;
Run Code Online (Sandbox Code Playgroud)

并且,使用该值,我们更改 y 比例:

yScale.domain([0, currentValue * 2])
Run Code Online (Sandbox Code Playgroud)

这样,选定的栏将始终保持相同的高度,就像 Bostock 的代码中一样。然后,我们更新栏:

bars.attr("y", d => yScale(d))
    .attr("height", d => height - padding - yScale(d))
    .attr("fill", (d => currentValue == d ? "firebrick" : "teal"));
Run Code Online (Sandbox Code Playgroud)

最后是 y 轴:

gY.call(yAxis);
Run Code Online (Sandbox Code Playgroud)