第十四章 力导向图

    力导向图能表示节点之间的多对多的关系。

    初始数据如下:

    节点(nodes)和连线(edges)的数组,节点是一些城市名,连线的两端是节点的序号(序号从 0 开始)。

    这些数据是不能作图的,因为不知道节点和连线的坐标。这句话一说出来,就请想到布局。本章用到的布局是:d3.layout.force()

    定义一个力导向图的布局如下。

    1. .nodes(nodes) //指定节点数组
    2. .links(edges) //指定连线数组
    3. .size([width,height]) //指定作用域范围
    4. .linkDistance(150) //指定连线长度
    5. .charge([-400]); //相互之间的作用力

    然后,使力学作用生效:

    节点转换前后如下图。

    节点转换前后

    转换后,节点对象里多了一些变量。其意义如下:

    • index:节点的索引号
    • px, py:节点上一个时刻的坐标
    • x, y:节点的当前坐标
    • weight:节点的权重

    再来看看连线的变化。

    可以看到,连线的两个节点序号,分别变成了对应的节点对象。

    有了转换后的数据,就可以作图了。分别绘制三种图形元素:

    • line,线段,表示连线。
    • circle,圆,表示节点。
    • text,文字,描述节点。

    调用 call( force.drag ) 后节点可被拖动。force.drag() 是一个函数,将其作为 call() 的参数,相当于将当前选择的元素传到 force.drag() 函数中。

    最后,还有一段最重要的代码。由于力导向图是不断运动的,每一时刻都在发生更新,因此,必须不断更新节点和连线的位置。

    力导向图布局 force 有一个事件 tick,每进行到一个时刻,都要调用它,更新的内容就写在它的监听器里就好。

    1. force.on("tick", function(){ //对于每一个时间间隔
    2. //更新连线坐标
    3. .attr("y1",function(d){ return d.source.y; })
    4. .attr("x2",function(d){ return d.target.x; })
    5. .attr("y2",function(d){ return d.target.y; });
    6. //更新节点坐标
    7. svg_nodes.attr("cx",function(d){ return d.x; })
    8. .attr("cy",function(d){ return d.y; });
    9. //更新文字坐标
    10. svg_texts.attr("x", function(d){ return d.x; })
    11. });

    tick 的英文意思是钟表发出的嘀嗒嘀嗒声,想到这个大家应该很清楚了吧。每次触发时,都会调用后面的无名函数 function。

    结果如图:

    结果

    下载地址:rm92.zip