欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  IT编程

D3.js力导向图中新增节点及新增关系连线示例

程序员文章站 2022-05-18 23:03:59
大家在使用D3.js中的力导向图时,基本都会遇到动态增加节点及连线的需求,这里记录一下我的实现方式。 话不多说,先放代码: 再看效果图: 总结:从代码上看实现这个功能逻辑还是挺简单的,但是从显示效果上看后增加的连线会覆盖在原先的节点上,显示效果不友好,下一篇会说明一下这个问题应该如何解决。 ......

大家在使用d3.js中的力导向图时,基本都会遇到动态增加节点及连线的需求,这里记录一下我的实现方式。

  • 话不多说,先放代码:
<!doctype html>
<html lang="en">
<head>
    <meta charset="utf-8">
    <title>title</title>
    <script src="https://d3js.org/d3.v4.min.js"></script>
</head>
<body>
<svg width="960" height="500"></svg>
</body>
<script type="text/javascript">
    var nodes = [
        { name: "姓名1"},
        { name: "姓名2"},
        { name: "姓名3"},
        { name: "姓名4"},
        { name: "姓名5"},
    ];
 
    var links = [  { source : 0  , target: 2 } , { source : 1  , target: 2 } ,
        { source : 3  , target: 2 } , { source : 3  , target: 4 } ,
    ];
 
    var width = 1024;
    var height = 738;
    var svg = d3.select("svg")
        .attr("width",width)
        .attr("height",height);
    var circle_radius = 30;
 
    // 通过布局来转换数据,然后进行绘制
    var simulation = d3.forcesimulation(nodes)
        .force("link", d3.forcelink(links).distance(200))
        .force("charge",d3.forcemanybody().strength(-100))
        .force("center",d3.forcecenter(width/2, height/2));
 
    var color = d3.scaleordinal(d3.schemecategory20);

    // 绘制线
    var svg_links = svg.selectall("path")
        .data(links)
        .enter()
        .append("path")
        .style("stroke","#ccc")
        .style("stroke-width",3);
 
    //节点对象
    var svg_nodes = svg.selectall("circle")
        .data(nodes)
        .enter()
        .append("circle")
        .attr("r",circle_radius)
        .attr("fill","yellow")
        .call(d3.drag()
            .on("start", dragstarted)
            .on("drag", dragged)
            .on("end", dragended));
    function dragstarted(d) {
        if (!d3.event.active)
            simulation.alphatarget(0.002).restart();
        d.fx = d.x;
        d.fy = d.y;
    }
    function dragged(d) {
        d.fx = d3.event.x;
        d.fy = d3.event.y;
    }
    function dragended(d) {
        if (!d3.event.active)
            simulation.alphatarget(0);
    }
    //节点描述
    var svg_text = svg.selectall("text")
        .data(nodes)
        .enter()
        .append("text")
        .style("fill","#000")
        .attr("dominant-baseline","middle")
        .attr("text-anchor", "middle")//在圆圈中加上数据
        .text(function(d){return d.name;});
 
    //箭头
    var marker=
        svg.append("marker")
            .attr("id", "resolved")
            .attr("markerunits","userspaceonuse")
            .attr("viewbox", "0 -5 10 10")//坐标系的区域
            .attr("refx",34)//箭头坐标
            .attr("refy", -1)
            .attr("markerwidth", 12)//标识的大小
            .attr("markerheight", 12)
            .attr("orient", "auto")//绘制方向,可设定为:auto(自动确认方向)和 角度值
            .attr("stroke-width",2)//箭头宽度
            .append("path")
            .attr("d", "m0,-5l10,0l0,5")//箭头的路径
            .attr('fill','#000000');//箭头颜色
 
    function draw(){
        svg_nodes
            .attr("cx",function(d){return d.x;})
            .attr("cy",function(d){return d.y;})
            .attr("role",function (d) {
                return d.role;
            });
 
        svg_text
            .attr("x", function(d){ return d.x; })
            .attr("y", function(d){ return d.y; });
 
        svg_links
            .attr("d",function(d){
                return 'm '+d.source.x+' '+d.source.y+' l '+ d.target.x +' '+d.target.y
            })
            .attr("marker-end", "url(#resolved)");
    }
    simulation.on("tick",draw);
    svg.call(d3.zoom().scaleextent([0.05, 8]).on('zoom', () => {
        var transform = d3.event.transform;
        svg_nodes.attr('transform', transform);
        svg_links.attr("transform",transform);
        svg_text.attr("transform",transform);
    })).on('dblclick.zoom', null);
 
    var e = { name: "姓名6"};
    var f =  {source : 5  , target: 2};
    d3.timeout(function(){
        nodes.push(e);
        links.push(f);
        update()
    }, 4000);
    function update() {
        svg_nodes = svg_nodes
            .data(nodes, (d) => d.name)
            .enter()
            .append("circle")
            .attr("r",circle_radius)
            .attr("fill","yellow")
            .merge(svg_nodes).call(d3.drag()
            .on("start", dragstarted)
            .on("drag", dragged)
            .on("end", dragended));

        svg_text = svg_text.data(nodes)
            .enter()
            .append("text")
            .style("fill","#000")
            .attr("dominant-baseline","middle")
            .attr("text-anchor", "middle")
            .text(function(d){return d.name;})
            .merge(svg_text);
            
        svg_links = svg_links.data(links, (d) => { return d.source.name + "-" + d.target.name; })
            .enter()
            .append("path")
            .style("stroke","#ccc")
            .style("stroke-width",3)
            .merge(svg_links);
     
        simulation.nodes(nodes);
        simulation.force("link").links(links);
        simulation.alpha(1).restart();
    }
</script>
</html>
  • 再看效果图:

D3.js力导向图中新增节点及新增关系连线示例

  • 总结:从代码上看实现这个功能逻辑还是挺简单的,但是从显示效果上看后增加的连线会覆盖在原先的节点上,显示效果不友好,下一篇会说明一下这个问题应该如何解决。