基于d3.js的组织结构图实现

一、效果图:
这里写图片描述

代码地址:https://codepen.io/wlei/pen/YveRmB
二、数据结构
data.json:
{
“name”:”中国”,
“number”:”2000”,
“children”:
[
{
“name”:”浙江” ,
“number”:”100”,
“children”:
[
{
“name”:”杭州” ,
“number”:”100”
},
{
“name”:”宁波” ,
“number”:”200”
},
{
“name”:”温州” ,
“number”:”300”
},
{
“name”:”绍兴”,
“number”:”400”
}
]
},
{
“name”:”广西” ,
“number”:”500”,
“children”:
[
{
“name”:”桂林”,
“number”:”100”
},
{
“name”:”南宁”,
“number”:”200”
},
{
“name”:”柳州”,
“number”:”300”
},
{
“name”:”防城港”,
“number”:”400”
}
]
},
{
“name”:”黑龙江”,
“number”:”500”,
“children”:
[
{
“name”:”哈尔滨”,
“number”:”100”
},
{
“name”:”齐齐哈尔”,
“number”:”200”
},
{
“name”:”牡丹江”,
“number”:”300”
},
{
“name”:”大庆”,
“number”:”400”
}
]
},
{
“name”:”新疆” ,
“number”:”500”,
“children”:
[
{
“name”:”乌鲁木齐”,
“number”:”100”
},
{
“name”:”克拉玛依”,
“number”:”200”
},
{
“name”:”吐鲁番”,
“number”:”300”
},
{
“name”:”哈密”,
“number”:”400”
}
]}]}

三、html页面

<!DOCTYPE html>
<html>
    <head>
        <meta charset="utf-8" />
        <title></title>
        <style>
            .node {
              font: 12px sans-serif;
            }

            .link {
              fill: none;
              stroke: #ccc;
              stroke-width: 1.5px;
            }
        </style>
    </head>
    <body>
    <script type="text/javascript" src="js/d3.min.js" ></script>
    <script>
        var width = 500,
            height = 500;

        var cluster = d3.layout.tree()
        .size([width, height - 200]);

        var diagonal = d3.svg.diagonal()
            .projection(function(d) {
                return [d.y, d.x];
            });

        var svg = d3.select("body").append("svg")
            .attr("width", width)
            .attr("height", height)
            .append("g")
            .attr("transform", "translate(40,0)");

        d3.json("data.json", function(error, root) {
          var nodes = cluster.nodes(root);
          var links = cluster.links(nodes);
          var link = svg.selectAll(".link")
              .data(links)
              .enter()
              .append("path")
              .attr("class", "link")
              .attr("d", function(d){
                return "M"+d.source.y+" "+d.source.x+
                "L"+(d.source.y+120)+" "+d.source.x+
                " L"+(d.source.y+120)+" "+d.target.x+" L"+
                d.target.y+" "+d.target.x;

              })
              .attr("style",function(){
                return "stroke:#F7881F"
              });

          var node = svg.selectAll(".node")
              .data(nodes)
              .enter()
              .append("g")
              .attr("class", "node")
              .attr("transform", function(d) { 
                return "translate(" + d.y + "," + (d.x+ -10) + ")"; 
              })

          node.append("rect")
            .attr("width",100)
            .attr("height",20)
            .attr("x",0)
            .attr("y",0)
            .attr("style","fill:#35AD5B;");

          node.append("text")
              .attr("dx", function(d) { 
                return 30;
              })
              .attr("dy", 14)
              .style("text-anchor", function(d) { 
                return "middle";
              })
              .style("fill","#fff")
              .text(function(d) { return d.name; });

           node.append("text")
              .attr("dx", function(d) { 
                return 80;
              })
              .attr("dy", 14)
              .style("text-anchor", function(d) { 
                return "middle";
              })
              .style("fill","#fff")
              .text(function(d) { return d.number; });
        });
    </script>

    </body>
</html>


优化:
一、效果图
这里写图片描述
二、html

<!DOCTYPE html>
<html>
    <head>
        <meta charset="utf-8" />
        <title>组织结构图</title>
        <style>
          .node circle {
            cursor: pointer;
          }

          .node text {
            font-size: 14px;
          }
          </style>
    </head>
    <body>
        <div id="test"></div>
        <script type="text/javascript" src="js/jquery-1.11.0.min.js" ></script>
        <script type="text/javascript" src="js/d3.min.js"></script>
        <script type="text/javascript" src="js/TreeExtend.js"></script>
        <script type="text/javascript">
            var opt = {
                colors:["#25a4f6","#92d1fa","#ae43a6","#f7881f","#f7881f"],
                width:1000,
                height:800,
                container:'#test',//树形图容器,值为选择器
                data:'new_file.json'//必填,url或对象
            };
            var TreeExtend = new TreeExtend(opt);
            TreeExtend.init();
        </script>
    </body>
</html>

二、数据结构
new_file.json

{
    "name": "总数",
    "type": "1",
    "number": "904",
    "children": [
        {
            "name": "一部",
            "type": "1",
            "children": [
                {
                    "name": "计划",
                    "type": "1",
                    "children": [
                        {
                            "name": "2017年",
                            "type": "1",
                            "children": [
                                {
                                    "name": "业务1",
                                    "type": "1",
                                    "children": [],
                                    "number": "2" }, {
                                    "name": "业务2",
                                    "type": "1",
                                    "children": [],
                                    "number": "0" }, {
                                    "name": "业务3",
                                    "type": "1",
                                    "children": [],
                                    "number": "1" }
                            ],
                            "number": "3"
                        }
                    ],
                    "number": "4"
                }, {
                    "name": "业务1",
                    "type": "1",
                    "children": [],
                    "number": "1"
                }
            ],
            "number": "616"
        }, {
            "name": "二部",
            "type": "1",
            "children": [
                {
                    "name": "业务1",
                    "type": "1",
                    "children": [],
                    "number": "10"
                }, {
                    "name": "业务2",
                    "type": "1",
                    "children": [],
                    "number": "0"
                }
            ],
            "number": "288"
        }
    ]

}

三、js –TreeExtend.js

function TreeExtend(option){

    var _defaultOpt = {
        colors:["#25a4f6","#92d1fa","#a2d403","#f7881f","#f7881f"],
        width:1000,
        height:800,
        container:'',//树形图容器,值为选择器
        data:''//必填,url或对象
    };

    option = $.extend(true, _defaultOpt,option);

    this.colors = option.colors;
    this.width  = option.width;
    this.height = option.height;
    this.container = option.container;
    this.data   = option.data;

}

TreeExtend.prototype.init = function(){
    var that = this;
    var padding = {left: 100, right:50, top: 20, bottom: 20 };
    var svg = d3.select(that.container)
            .append("svg")
            .attr("width", that.width + padding.left + padding.right)
            .attr("height", that.height + padding.top + padding.bottom)
            .append("g")
            .attr("transform","translate("+ padding.left + "," + padding.top + ")");
    //树状图布局
    var tree = d3.layout.tree()
        .size([that.height, that.width]);

    //对角线生成器
    var diagonal = d3.svg.diagonal()
        .projection(function(d) { return [d.y, d.x]; });

    if(Object.prototype.toString.call(that.data).toLowerCase()=='[object string]'){
        d3.json(that.data, function(error, root) {
            if (error) throw error;
            var dataFun = getRecursionFunc(root);
            root = dataFun(root,"key");
            render(root);
        });
    }else{
        render(that.data);
    }

    //获取给树形结构数据加字段的方法
    function getRecursionFunc(treeObj){
        var i = 0,initObj = treeObj;
        return function (treeObj,property){
            if(treeObj.children.length==0){
                treeObj[property] = i++;
                return initObj;
            }else{
                treeObj[property] = i++;
                for(var j=0;j<treeObj.children.length;j++){
                    treeObj.children[j][property] = i++;
                    if(j==(treeObj.children.length-1)){
                        return arguments.callee(treeObj.children[j],property);
                    }else{
                        arguments.callee(treeObj.children[j],property);
                    }
                }
            }
        };
    }



    function render(root){

        //给第一个节点添加初始坐标x0和x1,避免不自然效果
        root.x0 = that.height / 2;
        root.y0 = 0;

        //以第一个节点为起始节点,重绘
        redraw(root);

        //重绘函数
        function redraw(source){

            //(1)计算节点和连线的位置

            //应用布局,计算节点和连线
            var nodes = tree.nodes(root);
            var links = tree.links(nodes);
            //重新计算节点的y坐标
            nodes.forEach(function(d) { d.y = d.depth * 180; });

            //(2)节点的处理

            //获取节点的update部分
            var nodeUpdate = svg.selectAll(".node")
                .data(nodes, function(d){ return d.key; });

            //获取节点的enter部分
            var nodeEnter = nodeUpdate.enter();
            //获取节点的exit部分
            var nodeExit = nodeUpdate.exit();
            //1. 节点的 Enter 部分的处理办法
            var enterNodes = nodeEnter.append("g")
                .attr("class","node")
                .attr("transform", function(d) {
                    return "translate(" + source.y0 + "," + source.x0 + ")";
                })
                .on("click", function(d) { toggle(d); redraw(d); });

            enterNodes.append("circle")
                .attr("r", 0)
                .style("fill", function(d) { 
                    return that.colors[d.depth];
                })
                .style("stroke",function(d) {
                    return that.colors[d.depth];
                });

            enterNodes.append("text")
                .on("mouseover", function(d) {//连线mousehover事件
                    d3.select(this).style("font-size","18px")
                    .style("cursor","pointer");
                })
                .on("mouseout", function(d) {//连线mouseout事件
                    d3.select(this).style("font-size","14px")
                    .style("cursor","normal");
                })
                .attr("x", function(d) { return d.children || d._children ? -14 : 14; })
                .attr("dy", ".35em")
                .attr("text-anchor", function(d) { return d.children || d._children ? "end" : "start"; })
                .text(function(d) { return d.name +"("+d.number+")"; })
                .style("fill-opacity", 0);

            //2. 节点的 Update 部分的处理办法
            var updateNodes = nodeUpdate.transition()
                .duration(500)
                .attr("transform", function(d) { return "translate(" + d.y + "," + d.x + ")"; });

            updateNodes.select("circle")
                .attr("r", function(d){
                    return ((d.number/root.number)*40<2)?1:(d.number/root.number)*20;
                })
                .style("fill", function(d) {
                    return that.colors[d.depth];
                })
                .style("stroke",function(d) {
                    return that.colors[d.depth];
                });

            updateNodes.select("text").style("fill-opacity", 1);

            //3.节点的 Exit 部分的处理办法
            var exitNodes = nodeExit.transition()
                .duration(500)
                .attr("transform", function(d) { return "translate(" + source.y + "," + source.x + ")"; })
                .remove();
            exitNodes.select("circle").attr("r", 0);
            exitNodes.select("text").style("fill-opacity", 0);

            //(3) 连线的处理

            //获取连线的update部分
            var linkUpdate = svg.selectAll(".link")
                .data(links, function(d){ 
                    return d.target.key;
                });

            //获取连线的enter部分
            var linkEnter = linkUpdate.enter();

            //获取连线的exit部分
            var linkExit = linkUpdate.exit();

            //1. 连线的 Enter 部分的处理办法
            linkEnter.insert("path",".node")
                .on("mouseover", function(d) {//连线mousehover事件
                    d3.select(this).style("stroke-opacity","1")
                    .style("cursor","pointer");
                })
                .on("mouseout", function(d) {//连线mouseout事件
                    d3.select(this).style("stroke-opacity",(d.source.depth/10<0.2)?0.2:d.source.depth/10)
                    .style("cursor","normal");
                })
                .attr("class", "link")
                .style("fill","none")
                .style("stroke",function(d){
                    return that.colors[d.source.depth];
                })
                .style("stroke-width",function(d){
                    return ((d.target.number/root.number)*40<2)?2:(d.target.number/root.number)*40;
                })
                .style("stroke-linecap","round")
                .style("stroke-opacity",function(d) { 
                    return (d.source.depth/10<0.2)?0.2:d.source.depth/10;
                })
                .attr("d", function(d) {
                      var o = {x: source.x0, y: source.y0};
                      return diagonal({source: o, target: o});
                })
                .transition()
                .duration(500)
                .attr("d", diagonal);

            //2. 连线的 Update 部分的处理办法
            linkUpdate.transition()
                .duration(500)
                .attr("d", diagonal);

            //3.连线的 Exit 部分的处理办法
            linkExit.transition()
                .duration(500)
                .attr("d", function(d) {
                    var o = {x: source.x, y: source.y};
                    return diagonal({source: o, target: o});
                })
                .remove();


            //(4)将当前的节点坐标保存在变量x0、y0里,以备更新时使用
            nodes.forEach(function(d) {
                d.x0 = d.x;
                d.y0 = d.y;
            });

        }

        //切换开关,d 为被点击的节点
        function toggle(d){
            if(d.children){//如果有子节点
                d._children = d.children; //将该子节点保存到 _children
                d.children = null;  //将子节点设置为null
            }else{//如果没有子节点
                d.children = d._children; //从 _children 取回原来的子节点 
                d._children = null; //将 _children 设置为 null
            }
        }

    }

}
  • 9
    点赞
  • 37
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 10
    评论
### 回答1: 基于d3.js的图谱案例是指使用d3.js这个JavaScript库来创建显示图形的图谱,图谱是一种展示关系网络的图形化方式。在这种案例中,可以利用d3.js来创建各种类型的图谱,包括社交网络分析、知识图谱、组织架构图等。 利用d3.js库的强大功能,可以通过可视化方式展示数据之间的关系,从而更好地理解和分析数据。例如,在社交网络分析中,可以通过创建图谱显示不同人之间的关系,包括朋友关系、关注关系等。这样一来,就可以更好地了解人际关系的复杂性,发现隐藏在数据背后的模式和规律。 另一个例子是知识图谱,可以使用d3.js创建一个展示不同概念之间关联的图谱。这样做可以帮助人们更好地理解知识的结构,发现知识领域中的重要概念和关键因素。 在组织架构图方面,也可以利用d3.js创建一个图谱,展示不同职位之间的关系。这有助于组织内部的沟通和协作,帮助员工更好地了解组织的结构和层级关系。 总的来说,基于d3.js的图谱案例可以帮助人们更好地可视化和理解数据之间的关系,从而更好地进行数据分析和决策。这种图谱可以应用于各种领域,如社交网络分析、知识图谱和组织架构图等。 ### 回答2: 基于D3.js的图谱案例可以是一个网络关系图谱,展示网络上不同实体之间的关联和连接。 首先,我们可以使用D3.js创建一个包含节点和链接的图形容器。节点可以表示不同类型的实体,如人物、组织或地点,而链接则代表实体之间的关系。通过D3.js的强大数据绑定功能,我们可以轻松地将数据与图形元素相结合。 接下来,我们可以使用适当的数据结构表示实体之间的关系。例如,可以使用邻接矩阵、邻接表或关联列表来存储节点和链接之间的连接信息。然后,我们可以将这些数据结构转换为D3.js可以理解的格式,以便能够正确地显示图形。 在图形的设计方面,我们可以使用D3.js提供的各种图形元素和布局,来呈现一个美观而有条理的图谱。例如,我们可以使用圆形节点来代表实体,并使用不同的颜色或形状来表示不同类型的节点。链接可以用曲线或直线表示,并可以调整其样式和粗细度以突出显示不同的关系。 除此之外,我们还可以通过添加交互和动画效果来增强用户体验。例如,当用户将鼠标悬停在一个节点上时,可以显示与其关联的详细信息。当用户点击一个节点时,可以展开或折叠与其相关的链接和节点,以便更好地探索整个图谱。 总之,基于D3.js的图谱案例可以用于可视化网络关系,帮助用户更好地理解和分析复杂的数据结构和关联。通过D3.js的强大功能,我们可以创建出精美、交互性强的图形,并提供丰富的数据展示和探索功能。 ### 回答3: 根据您的指示,请允许我用300字为您回答:“基于d3.js图谱案例”是指使用d3.js这个JavaScript图形库来创建图谱可视化的应用案例。 d3.js是一个强大且灵活的数据驱动的文档操作库,它可以帮助开发人员在网页上动态地创建和操作图形。图谱是一种展示实体之间关系的可视化图形,常用于显示知识图谱、社交网络和组织结构等领域。 在基于d3.js的图谱案例中,开发人员可以使用该库的各种功能,例如数据绑定、选择元素、创建动画和事件处理等,来设计和构建图谱的可视化效果。 例如,假设我们要展示一个人物关系图谱,其中包含有多个人物节点和人物之间的关系边。我们可以使用d3.js来创建这个图谱。首先,我们可以使用d3.js的数据绑定功能将人物节点的数据与网页上的DOM元素绑定起来,然后使用选择元素功能来渲染这些节点。接着,我们可以使用d3.js的布局函数,如力导向布局或层次布局,来确定节点的位置和边的连接方式。最后,我们可以添加动画和事件处理功能,使得用户可以与图谱互动。 通过基于d3.js的图谱案例,我们可以清晰地展示各种实体之间的关系,使得数据更易于理解和分析。此外,d3.js还提供了丰富的扩展和预处理功能,使得图谱的设计和展示更加灵活和强大。 综上所述,在基于d3.js的图谱案例中,开发人员可以利用该库的强大功能来创建、显示和操作图谱,实现更好的数据可视化效果。这不仅可以提高用户对数据的理解,还可以为用户提供更好的交互体验和决策支持。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 10
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

wl_

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值