一、饼状图

  在布局的应用中,最简单的就是饼状图。

1、数据

  有如下数据,需要可视化:

  1. var dataset = [ , , , , ];

  这样的值是不能直接绘图的。例如绘制饼状图的一个部分,需要知道一段弧的起始角度和终止角度,这些值都不存在于数组 dataset 中。因此,需要用到布局,布局的作用就是:计算出适合于作图的数据。

2、布局(数据转换)

  定义一个布局:

  1. var pie = d3.layout.pie();

  返回值赋给变量 pie,此时 pie 可以当做函数使用

  1. var piedata = pie(dataset);

  将数组 dataset 作为 pie() 的参数,返回值给 piedata。如此一来:piedata 就是转换后的数据。简单吧,非常轻松就得到了。在控制台输出 piedata ,让我们来看看数据转换成什么了。

  如上图所示,5 个整数被转换成了 5 个对象(Object) ,每个对象都有变量起始角度(startAngle)和终止角度(endAngle),还有原数据(属性名称为 data)。这些都是绘图需要的数据。

  千万记住:布局不是要直接绘图,而是为了得到绘图所需的数据。

3、绘制图形

  为了根据转换后的数据 piedata 来作图,还需要一样工具:生成器

  SVG 有一个元素,叫做路径 <path>,是 SVG 中功能最强的元素,它可以表示其它任意的图形。顾名思义,路径元素就是通过定义一个段“路径”,来绘制出各种图形。但是,路径是很难计算的,通过布局转换后的数据 piedata 仍然很难手动计算得到路径值。为我们完成这项任务的,就是生成器。

  这里要用到的叫做弧生成器,能够生成弧的路径,因为饼图的每一部分都是一段弧。

  1. var outerRadius = ; //外半径
  2. var innerRadius = ; //内半径,为0则中间没有空白
  3.  
  4. var arc = d3.svg.arc() //弧生成器
  5. .innerRadius(innerRadius) //设置内半径
  6. .outerRadius(outerRadius); //设置外半径

  弧生成器返回的结果赋值给 arc。此时,arc 可以当做一个函数使用,把 piedata 作为参数传入,即可得到路径值。

  接下来,可以在 SVG 中添加图形元素了。先在 <svg> 里添加足够数量(5个)个分组元素(g),每一个分组用于存放一段弧的相关元素。

  1. var arcs = svg.selectAll("g")
  2. .data(piedata)
  3. .enter()
  4. .append("g")
  5. .attr("transform","translate("+ (width/) +","+ (width/) +")");

  接下来对每个 <g> 元素,添加 <path>

  1. arcs.append("path")
  2. .attr("fill",function(d,i){
  3. return color(i);
  4. })
  5. .attr("d",function(d){
  6. return arc(d); //调用弧生成器,得到路径值
  7. });

  因为 arcs 是同时选择了 5 个 <g> 元素的选择集,所以调用 append(“path”) 后,每个 <g> 中都有 <path> 。路径值的属性名称是 d,调用弧生成器后返回的值赋值给它。要注意,arc(d) 的参数 d 是被绑定的数据。

  另外,color 是一个颜色比例尺,它能根据传入的索引号获取相应的颜色值,定义如下。

  1. var color = d3.scale.category10(); //有十种颜色的颜色比例尺

  然后在每一个弧线中心添加文本。

  1. arcs.append("text")
  2. .attr("transform",function(d){
  3. return "translate(" + arc.centroid(d) + ")";
  4. })
  5. .attr("text-anchor","middle")
  6. .text(function(d){
  7. return d.data;
  8. });

  arc.centroid(d) 能算出弧线的中心

  要注意,text() 里返回的是 d.data ,而不是 d 。因为被绑定的数据是对象,里面有 d.startAngle、d.endAngle、d.data 等,其中 d.data 才是转换前的整数的值

二、力导向图

  力导向图中每一个节点都受到力的作用而运动,这种是一种非常绚丽的图表。

  力导向图(Force-Directed Graph),是绘图的一种算法。在二维或三维空间里配置节点,节点之间用线连接,称为连线。各连线的长度几乎相等,且尽可能不相交。节点和连线都被施加了力的作用,力是根据节点和连线的相对位置计算的。根据力的作用,来计算节点和连线的运动轨迹,并不断降低它们的能量,最终达到一种能量很低的安定状态。

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

1、数据

  初始数据如下:

  1. var nodes = [ { name: "桂林" }, { name: "广州" },
  2. { name: "厦门" }, { name: "杭州" },
  3. { name: "上海" }, { name: "青岛" },
  4. { name: "天津" } ];
  5.  
  6. var edges = [ { source : , target: } , { source : , target: } ,
  7. { source : , target: } , { source : , target: } ,
  8. { source : , target: } , { source : , target: } ];

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

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

2、布局(数据转换)

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

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

  然后,使力学作用生效:

  1. force.start(); //开始作用

  如此,数组 nodes 和 edges 的数据都发生了变化。在控制台输出一下,看看发生了什么变化。

  节点转换前后如下图

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

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

  再来看看连线的变化:

  可以看到,连线的两个节点序号,分别变成了对应的节点对象。(一个代表源,一个代表目标

3、绘制

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

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

  代码如下:

  1. //添加连线
  2. var svg_edges = svg.selectAll("line")
  3. .data(edges)
  4. .enter()
  5. .append("line")
  6. .style("stroke","#ccc")
  7. .style("stroke-width",);
  8.  
  9. var color = d3.scale.category20();//颜色比例尺
  10.  
  11. //添加节点
  12. var svg_nodes = svg.selectAll("circle")
  13. .data(nodes)
  14. .enter()
  15. .append("circle")
  16. .attr("r",20)
  17. .style("fill",function(d,i){
  18. return color(i);
  19. })
  20. .call(force.drag); //使得节点能够拖动
  21. //添加描述节点的文字
  22. var svg_texts = svg.selectAll("text")
  23. .data(nodes)
  24. .enter()
  25. .append("text")
  26. .style("fill", "black")
  27. .attr("dx", )
  28. .attr("dy", )
  29. .text(function(d){
  30. return d.name;
  31. });

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

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

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

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

  tick 的英文意思是钟表发出的嘀嗒嘀嗒声,每次触发时,都会调用后面的无名函数 function。

D3.js系列——布局:饼状图和力导向图的更多相关文章

  1. D3.js系列——布局:弦图和集群图/树状图

    一.弦图 1.弦图是什么 弦图(Chord),主要用于表示两个节点之间的联系的图表.两点之间的连线,表示谁和谁具有联系. 2.数据 初始数据为: var city_name = [ "北京& ...

  2. D3.js系列——布局:打包图和地图

    一.打包图 打包图( Pack ),用于表示包含与被包含的关系,也可表示各对象的权重,通常用一圆套一圆来表示前者,用圆的大小来表示后者. 1.布局(数据转换) var pack = d3.layout ...

  3. D3.js系列——交互式操作和布局

    一.图表交互操作 与图表的交互,指在图形元素上设置一个或多个监听器,当事件发生时,做出相应的反应. 交互,指的是用户输入了某种指令,程序接受到指令之后必须做出某种响应.对可视化图表来说,交互能使图表更 ...

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

    大家在使用D3.js中的力导向图时,基本都会遇到动态增加节点及连线的需求,这里记录一下我的实现方式. 话不多说,先放代码: <!DOCTYPE html> <html lang=&q ...

  5. D3.js力导向图(适用于其他类型图)中后添加元素遮盖已有元素的问题解决

    上一篇说了在D3.js中动态增加节点及连线的一种实现方式,但是有后添加元素遮盖原节点的现象,这一篇说一下出现这个现象的解决办法. 在D3.js中后添加的元素是会遮盖先添加的元素的,同时还有一个设定:后 ...

  6. D3.js 力导向图的制作

    力导向图中每一个节点都受到力的作用而运动,这种是一种非常绚丽的图表. 力导向图(Force-Directed Graph),是绘图的一种算法.在二维或三维空间里配置节点,节点之间用线连接,称为连线. ...

  7. D3.js 力导向图的显示优化

    D3.js 作为一个前端,说到可视化除了听过 D3.js 的大名,常见的可视化库还有 ECharts.Chart.js,这两个库功能也很强大,但是有一个共同特点是封装层次高,留给开发者可设计和控制的部 ...

  8. Vue和d3.js(v4)力导向图force结合使用,v3版本升级v4【一】

    前段时间因为参与项目涉密,所以一直没有更新博客,有些博友给我私信或者留言要部分博文的源码,因为我的电脑更换,demo的源码没有备份 所以无法提供.大家可针对具体问题问我,有空我定会回复的.另外转发文章 ...

  9. D3.js 力导向图

    花了大半天看了一个八十几行的代码..心累 力导向图是之前就有画过很多次的东西,但是这次的代码看上去很陌生,然后发现是D3更新了4.0.... 先贴代码 var svg = d3.select(&quo ...

随机推荐

  1. cobalt strike使用笔记

    启动: ./teamserver 192.168.74.1 admin #启动cs服务器.admin为密码. 监听器: windows/beacon_dns/reverse_dns_txtwindow ...

  2. python3 多态,绑定方法与非绑定方法

    多态:同一种事物的不同形态(一个抽象类有多个子类,因而多态的概念依赖于继承) 1. 序列类型有多种形态:字符串,列表,元组. 2. 动物有多种形态:人,狗,猪 多态性:多态性是指具有不同功能的函数可以 ...

  3. HDU3910(数学期望题,题目难懂)

    Liang Guo Sha Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)Tot ...

  4. [ Openstack ] OpenStack-Mitaka 高可用之 镜像服务(glance)

    目录 Openstack-Mitaka 高可用之 概述    Openstack-Mitaka 高可用之 环境初始化    Openstack-Mitaka 高可用之 Mariadb-Galera集群 ...

  5. krpano--控制热点跳转到场景的指定视角

    krpano课堂(肥宗) · 2015-07-13 19:32 有这么一种情况,假设我们用三个场景,这三个场景恰好是一条街道的同一方向的三个拍摄点.如上图. 我们可以通过设置A.B.C三个场景中的vi ...

  6. 搜索引擎--范例:django之初试牛刀

    这学期学了一门课<信息检索>,也就是传说中的搜索引擎 大作业自然而然的让我们自己做一个小型的搜索引擎.于是乎,我们本次的主题就诞生了 我也是边学边用,下面和大家一起分享我在这个过程中学到的 ...

  7. window下golang包管理glide使用说明

    golang是一门简洁高效的开发 语言,但是包管理一直是一个痛点,如图 : 很多开源项目特别是github.com中的大量应用golang.org和google.golang.org中的源码,且由于被 ...

  8. inside a shard

    fsync sync fsync/syncsync is a standard system call in the Unix operating system, which commits to d ...

  9. redis SCAN cursor

    https://redis.io/commands/scan 可以看到: sscan的返回结果,有两部分, 第一部分  是一个数字.其实代表一个游标. 第二部分  是结果. scan是以游标为基础,每 ...

  10. servletcontext.getRealPath()

    getRealPath方法已经不建议使用了: http://blog.csdn.net/lzynihao/article/details/8315796 另外:http://veryls.iteye. ...