d3.js:将样式应用于各个行

时间:2014-03-03 08:48:16

标签: javascript svg d3.js

正在试用draggable network并希望能够为不同的链接使用不同的颜色。当我注释掉这些行

/*var link = svg.append("g")
    .attr("class", "link") 
    .selectAll("line");*/

并将其替换为var link = svg.append("g");
所以我可以尝试逐个添加链接,使用if-else条件,我可以为每一行应用不同的线条样式。但是当我尝试将一种风格均匀地应用到所有线条本身时,线条没有显示在屏幕上。 Firebug也没有显示任何错误。

  link = link.data(graph.links).enter().append("line")
      .attr("x1", function(d) { return d.source.x; })
      .attr("y1", function(d) { return d.source.y; })
      .attr("x2", function(d) { return d.target.x; })
      .attr("y2", function(d) { return d.target.y; })
      .attr("class", "link");

我添加了.attr("class", "link");部分,认为它将样式应用于线条。但事实并非如此。你能帮助选择性地追加属性的正确方法吗?

这是整个代码:

<!DOCTYPE html>
<meta charset="utf-8">

<svg id="mySvg" width="20" height="20">
  <defs id="mdef">
        <pattern id="image" x="0" y="0" height="20" width="20">
          <image x="0" y="0" width="20" height="20" xlink:href="url.png"></image>
        </pattern>
  </defs>
  <defs>
  <pattern id="tile-ww" patternUnits="userSpaceOnUse" width="25" height="25">
  <image xlink:href="url.png" x="15" y="15" width="15" height="15"></image></pattern>
  </defs>
</svg>

<style>

.node {
  stroke: green;
  stroke-width: 1px;
  opacity: 0.8;
  fill: url(#image);  
}

.node .selected {
  stroke: red;
}

.link {
  stroke: orange;
  stroke-width: 3;
  stroke-dasharray: 20,10,5,5,5,10;
  opacity: 0.5;
}

.dotted {border: 1px dotted #ff0000; border-style: none none dotted; color: #fff; background-color: #fff; }

.brush .extent {
  fill-opacity: .1;
  stroke: #fff;
  shape-rendering: crispEdges;
}

</style>
<body>
<script src="d3/d3.v3.js"></script>

<script>

var width = 960, height = 500, shiftKey;

var svg = d3.select("body")
    .attr("tabindex", 1)
    .on("keydown.brush", keydown)
    .on("keyup.brush", keyup)
    .each(function() { this.focus(); })
    .append("svg")
    .attr("width", width)
    .attr("height", height);

/*var link = svg.append("g")
    .attr("class", "link") 
    .selectAll("line");*/
var link = svg.append("g");

var brush = svg.append("g")
    .datum(function() { return {selected: false, previouslySelected: false}; })
    .attr("class", "brush");

var node = svg.append("g")
    .attr("class", "node")
    .selectAll("circle");

//Add the SVG Text Element to the svgContainer
text = svg.append('text').text('This is some information about whatever')
                        .attr('x', 50)
                        .attr('y', 200)
                        .attr('fill', 'black');

//you can delete this circle
svg.append("circle")
         .attr("class", "logo")
         .attr("cx", 225)
         .attr("cy", 225)
         .attr("r", 20)
         .style("fill", "transparent")       
         .style("stroke", "black")     
         .style("stroke-width", 0.25)
         .on("mouseover", function(){ 
               d3.select(this)
                   .style("fill", "url(#image)");
         })
          .on("mouseout", function(){ 
               d3.select(this)
                   .style("fill", "transparent");
         });

//get the json file with either error messages or the entire data as "graph" object
d3.json("graph.json", function(error, graph) {

  //"links" is a property in "graph", and contains some data in it. Iterate through all of them
  graph.links.forEach(function(d) //iterate through each data object in link
  {
    d.source = graph.nodes[d.source];//we got nodes too, with graph so access its data
    d.target = graph.nodes[d.target];
    d.linkStatus = d.status;
    console.log(d.source);
  });


  link = link.data(graph.links).enter().append("line")
      .attr("x1", function(d) { return d.source.x; })
      .attr("y1", function(d) { return d.source.y; })
      .attr("x2", function(d) { return d.target.x; })
      .attr("y2", function(d) { return d.target.y; })
      .attr("class", "link");

  brush.call(d3.svg.brush()
        .x(d3.scale.identity().domain([0, width]))
        .y(d3.scale.identity().domain([0, height]))
        .on("brushstart", function(d) {
          node.each(function(d) { d.previouslySelected = shiftKey && d.selected; });
        })
        .on("brush", function() {
          var extent = d3.event.target.extent();
          node.classed("selected", function(d) {
            return d.selected = d.previouslySelected ^
                (extent[0][0] <= d.x && d.x < extent[1][0]
                && extent[0][1] <= d.y && d.y < extent[1][1]);
          });
        })
        .on("brushend", function() {
          d3.event.target.clear();
          d3.select(this).call(d3.event.target);
        }));

  node = node.data(graph.nodes).enter().append("circle")
      .attr("r", 10)//radius
      .attr("cx", function(d) { return d.x; })
      .attr("cy", function(d) { return d.y; })
      .on("mousedown", function(d) {
        if (!d.selected) { // Don't deselect on shift-drag.
          if (!shiftKey) node.classed("selected", function(p) { return p.selected = d === p; });
          else d3.select(this).classed("selected", d.selected = true);
        }
      })
      .on("mouseup", function(d) {
        if (d.selected && shiftKey) d3.select(this).classed("selected", d.selected = false);
      })
      .call(d3.behavior.drag()
        .on("drag", function(d) { nudge(d3.event.dx, d3.event.dy); }));
});

function nudge(dx, dy) 
{
  node.filter(function(d) { return d.selected; })
      .attr("cx", function(d) { return d.x += dx; })
      .attr("cy", function(d) { return d.y += dy; })

  link.filter(function(d) { return d.source.selected; })
      .attr("x1", function(d) { return d.source.x; })
      .attr("y1", function(d) { return d.source.y; });

  link.filter(function(d) { return d.target.selected; })
      .attr("x2", function(d) { return d.target.x; })
      .attr("y2", function(d) { return d.target.y; });

  d3.event.preventDefault();
}

function keydown() 
{
  if (!d3.event.metaKey) switch (d3.event.keyCode) {
    case 38: nudge( 0, -1); break; // UP
    case 40: nudge( 0, +1); break; // DOWN
    case 37: nudge(-1,  0); break; // LEFT
    case 39: nudge(+1,  0); break; // RIGHT
  }
  shiftKey = d3.event.shiftKey || d3.event.metaKey;
}

function keyup() 
{
  shiftKey = d3.event.shiftKey || d3.event.metaKey;
}

</script>

2 个答案:

答案 0 :(得分:2)

您不需要任何if-else条件,也不需要对现有代码进行大量修改。在D3中设置线颜色的方式是通过.style()命令:

link.style("stroke", ...);

参数不必是固定值,但可以是函数。然后对选择中的每个元素(在这种情况下为线条)评估此函数,允许您为它们提供不同的颜色。例如,如果您想根据源的x位置给出不同的颜色,则可以执行以下操作。

var color = d3.scale.category20();
link.style("stroke", function(d) { return color(d.source.x); });

您可以使用绑定到元素(d)的数据的任何内容来确定您想要的颜色,或者其他任何颜色。例如,根据索引为每一行设置不同的颜色,你可以这样做。

link.style("stroke", function(d, i) { return color(i); });

答案 1 :(得分:0)

使您的链接可见

这样做

var link = svg.append("g")
             .selectAll("line");

而不是

var link = svg.append("g");

现在,您可以看到橙色的所有链接。

将样式应用于各行

使用这样的条件添加类名。

link.data(graph.links).enter()
     .append("line")
     .attr("class",function(d){     
          return (d.source.x>400 && d.source.y<300)?'link-b':'link-r';
     });