D3 js 从数据属性设置数据

3 javascript selection d3.js

在创建图表期间,我需要设置数据。那些数据(对象数组)我已经在 HTML 中是这样的:

<svg  class="graph-n" data-stuff="{simplified data}"></svg>
Run Code Online (Sandbox Code Playgroud)

然后使用 Javascript 和 D3 JS,我使用以下代码初始化和设置图形:

<script>
    var margin = { top: 20, right: 20, bottom: 30, left: 50},
        width = 1500 - margin.left - margin.right,
        height = 350 - margin.top - margin.bottom;

    var x = d3.scaleTime().range([0, width]);
    var y = d3.scaleLinear().range([height, 0]);


    var valueline = d3.line()
        .x(function(d) { return x(new Date(d.t)); })
        .y(function(d) { return y(d.y); });

    var svg = d3.selectAll(".graph-n")
        .attr("width", width + margin.left + margin.right)
        .attr("height", height + margin.top + margin.bottom)
        .append("g")
        .attr("transform", "translate(" + margin.left + "," + margin.top + ")");


        x.domain(d3.extent(data, function(d) { return new Date(d.t); }));
        y.domain([0, d3.max(data, function(d) { return d.y; })]);

        svg.append("path")
            .attr("class", "line")
            .attr("d", valueline);

        svg.append("g")
            .attr("transform", "translate(0," + height + ")")
            .call(d3.axisBottom(x));

        svg.append("g")
            .call(d3.axisLeft(y));
</script>
Run Code Online (Sandbox Code Playgroud)

问题是我该怎么说,在数据属性 'data-stuff' 中的选择期间,数据位于每个元素内?每个 SVG 都有要在他自己的数据属性中绘制的数据。

或者我的方法是错误的,我将使用不同的方法?

谢谢你的回复。

vep*_*oza 5

没有办法只明确告诉 d3“从这个属性中获取数据”。但是,您可以以编程方式设置数据,从您选择的属性加载它。有几种方法可以实现它,如这些选择示例所示(它们使用<ul>并且<li>为简单起见,<svg>用法是类似的):

// the pure D3 way
d3.selectAll("ul.d3-pure")                // select the element
    .datum(function() { return this.getAttribute("data-list").split(",")}) // set selection's data based on its data attribute
    .selectAll("li")                      // create new selection
        .data((d) => d)                   // set the data from the parent element
        .enter().append("li")             // create missing elements
            .text((content) => content);  // set elements' contents

// the DOM way      
var domUls = document.querySelectorAll("ul.dom");     // select elements
for(var i = 0; i < domUls.length; i++) {              // iterate over those elements
    const ul = domUls[i];
    const d3_ul = d3.select(ul);                      // create D3 object from the node
    const data = ul.getAttribute("data-list").split(",");
    d3_ul.selectAll("li").data(data)                  // create new selection and assign its data
        .enter().append("li")                         // create missing elements
            .text((content) => content)               // set elements' content
}

// the hybrid D3-DOM way
d3.selectAll("ul.d3-hybrid")                // select elements
    .each(function() {                      // iterate over each node of the selection
        const ul = d3.select(this);         // "this" is the "ul" HTML node
        const data = ul.attr("data-list").split(",");
        ul.selectAll("li").data(data)       // create new selection, assign its data
            .enter().append("li")           // create missing elements
                .text((content) => content) // set elements' content
        });
Run Code Online (Sandbox Code Playgroud)
<script src="https://cdnjs.cloudflare.com/ajax/libs/d3/3.4.11/d3.min.js"></script>
<ul class="d3-pure" data-list="1,2,3">
</ul>
<ul class="dom" data-list="a,b,c">
</ul>
<ul class="d3-hybrid" data-list="I,II,III">
</ul>
Run Code Online (Sandbox Code Playgroud)

  • 不错的答案!您也可以考虑使用鲜为人知的 [dataset](https://developer.mozilla.org/en-US/docs/Web/API/HTMLElement/dataset) 属性,它允许轻松访问元素的 *custom 数据属性*`数据-*`。你可以使用 `this.dataset.list` 代替 `this.getAttribute("data-list")`。 (4认同)