目标

  在这一章,我们将学习如何使SVG坐标空间是动态的,这样我们的数据可视化不论数据是什么,都始终是可见的。

  我们会使得SVG坐标空间尺度上调或下调来适于我们的数据。

三个SVG长方形

  我们就从三个长方形作为开始:

 1   var jsonRectangles = [
2 { "x_axis": 10, "y_axis": 10, "height": 20, "width":20, "color" : "green" },
3 { "x_axis": 40, "y_axis": 40, "height": 20, "width":20, "color" : "purple" },
4 { "x_axis": 70, "y_axis": 70, "height": 20, "width":20, "color" : "red" }];
5
6 var svgContainer = d3.select("body").append("svg")
7 .attr("width", 100)
8 .attr("height", 100);
9
10 var rectangles = svgContainer.selectAll("rect")
11 .data(jsonRectangles)
12 .enter()
13 .append("rect");
14
15 var rectangleAttributes = rectangles
16 .attr("x", function (d) { return d.x_axis; })
17 .attr("y", function (d) { return d.y_axis; })
18 .attr("height", function (d) { return d.height; })
19 .attr("width", function (d) { return d.width; })
20 .style("fill", function(d) { return d.color; });

  得到的结果是:

  漂亮!

  其中SVG容器:

1   var svgContainer = d3.select("body").append("svg")
2 .attr("width", 100)
3 .attr("height", 100);

宽100单位,高100单位。

  也就是说,图中三个长方形中最右下角的点的坐标(90,90)仍然在SVG容器视窗范围内。

  但是,如果紫色的长方形的x坐标,突然增加了四倍,从40变成160,结果会怎么样呢?

 1   //原来
2 { "x_axis": 40, "y_axis": 40, "height": 20, "width":20, "color" : "purple" }
3
4 //变为
5 { "x_axis": 160, "y_axis": 40, "height": 20, "width":20, "color" : "purple" }
6
7 //因此,数据集 jsonRectangles 变成了:
8 var jsonRectangles = [
9 { "x_axis": 10, "y_axis": 10, "height": 20, "width":20, "color" : "green" },
10 { "x_axis": 160, "y_axis": 40, "height": 20, "width":20, "color" : "purple" },
11 { "x_axis": 70, "y_axis": 70, "height": 20, "width":20, "color" : "red" }];

  这也就是说,紫色长方形有坐标值(160,40)。

  这个坐标已经超出高100宽100的范围。

  这样我们的数据可视化结果成了:

  就像我们想象的那样,太糟了!

动态调整SVG容器空间

  我们真正需要的是,能够根据我们的数据,对SVG容器的width、height属性进行动态调节。

  我们打算使用基础JavaScript循环(loop)来对JSON对象数组处理,找出最大的X坐标he最大的Y坐标。

  最大的x坐标和最大的y坐标就是长方形的最右下角点的坐标。

 //新的jsonRectangle数据(其中紫色长方形的x坐标现在是160)
var jsonRectangles = [
{"x_axis":10,"y_axis":10,"height":20,"width":20,"color":"green"},
{"x_axis":160,"y_axis":40,"height":20,"width":20,"color":"purplr"},
{"x_axis":70,"y_axis":70,"height":20,"width":20,"color":"red"}
]; var max_x = 0;//用于存储最大x坐标
var max_y = 0;//用于存储最大y坐标 //在jsonRectangle数组上的循环
for(var i = 0;i<jsonRectangles.length;i++){
var temp_x,temp_y; //为了得到最右的点,我们需要把x坐标和width相加
temp_x = jsonRectangles[i].x_axis+jsonRectangles[i].width; //为了得到最下面的点,我们需要把y坐标和height相加
temp_y = jsonRectangles[i].y_axis+jsonRectangles[i].height; /**
*如果临时x坐标比max_x大,
*那么就让max_x等于temp_x
*否则,什么都不用做
*同理,max_y也一样
*/
if(temp_x>=max_x){
max_x = temp_x;
} if(temp_y>=max_y){
max_y = temp_y;
} }//循环停止 max_x;
//返回180 max_y;
//返回 90

  如果数据发生了变化,max_x和max_y将始终都是数据中的最大值。

  现在,我们可以更新我们的SVG容器:

 //原来
var svgContainer = d3.select("body").append("svg")
.attr("height",200)
.attr("width",200); //现在(使用变量max_x和max_y)
var svgContainer = d3.select("body").append("svg")
.attr("width",max_x + 20)
.attr("width",max_x + 20);
//注意 — 在这里给max_x和max_y各加了20,是为了给元素多一些文本空间

  这样,SVG容器就一直能够显示右边的最大尺寸,我们的数据也就能正确的出现在其内部。

成品

  既然问题解决了,完整的代码如下:

 1   var jsonRectangles = [
2 { "x_axis": 10, "y_axis": 10, "height": 20, "width":20, "color" : "green" },
3 { "x_axis": 160, "y_axis": 40, "height": 20, "width":20, "color" : "purple" },
4 { "x_axis": 70, "y_axis": 70, "height": 20, "width":20, "color" : "red" }];
5
6 var max_x = 0;
7 var max_y = 0;
8
9 for (var i = 0; i < jsonRectangles.length; i++) {
10 var temp_x, temp_y;
11 var temp_x = jsonRectangles[i].x_axis + jsonRectangles[i].width;
12 var temp_y = jsonRectangles[i].y_axis + jsonRectangles[i].height;
13
14 if ( temp_x >= max_x ) { max_x = temp_x; }
15
16 if ( temp_y >= max_y ) { max_y = temp_y; }
17 }
18
19 var svgContainer = d3.select("body").append("svg")
20 .attr("width", max_x)
21 .attr("height", max_y)
22
23 var rectangles = svgContainer.selectAll("rect")
24 .data(jsonRectangles)
25 .enter()
26 .append("rect");
27
28 var rectangleAttributes = rectangles
29 .attr("x", function (d) { return d.x_axis; })
30 .attr("y", function (d) { return d.y_axis; })
31 .attr("height", function (d) { return d.height; })
32 .attr("width", function (d) { return d.width; })
33 .style("fill", function(d) { return d.color; });

  现在,所有的长方形都出现啦!

  SVG视窗能够把最右的(max_x+20,max_y+20)包括进去。

  而且,SVG视窗是动态生成,不需要我们去手动的更新width和height。

  使用JavaScript的For循环,我们就能够实现动态的resize我们的SVG视窗容器来适应数据。

  如果数据再次改变,我们的视窗(容器)也能够随时包含全部的数据可视化结果。

d2.js学习笔记(七)——动态SVG坐标空间的更多相关文章

  1. JS 学习笔记 (七) 面向对象编程OOP

    1.前言 创建对象有很多种方法,最常见的是字面量创建和new Object()创建.但是在需要创建多个相同结构的对象时,这两种方法就不太方便了. 如:创建多个学生信息的对象 let tom = { n ...

  2. Node.js学习笔记(七) --- Node.js的静态文件托管、路 由、EJS 模板引擎、GET 、POST

    1 . Nodejs 静态文件托管静态 web 服务器封装 2 . 路由 官方解释:  路由(Routing)是由一个 URI(或者叫路径)和一个特定的 HTTP 方法(GET.POST 等)组成的, ...

  3. D3.js学习笔记(四)—— 使用SVG坐标空间

    目标 在这一章,你将要使用D3.js基于一些数据把SVG元素添加到你想要的坐标位置上. 我们的目标就是使用下面的数据集: var spaceCircles = [30,70,110]; 并使用D3.j ...

  4. D3.js学习笔记(六)——SVG基础图形和D3.js

    目标 在这一章,我们将会重温SVG图形,学习如何使用D3.js来创建这些图形. 这里会包括前面例子中的SVG基础图形以及如何使用D3.js设置图形的属性. 使用D3.js画一个SVG 的 圆 circ ...

  5. 系列文章--Node.js学习笔记系列

    Node.js学习笔记系列总索引 Nodejs学习笔记(一)--- 简介及安装Node.js开发环境 Nodejs学习笔记(二)--- 事件模块 Nodejs学习笔记(三)--- 模块 Nodejs学 ...

  6. js学习笔记:webpack基础入门(一)

    之前听说过webpack,今天想正式的接触一下,先跟着webpack的官方用户指南走: 在这里有: 如何安装webpack 如何使用webpack 如何使用loader 如何使用webpack的开发者 ...

  7. WebGL three.js学习笔记 使用粒子系统模拟时空隧道(虫洞)

    WebGL three.js学习笔记 使用粒子系统模拟时空隧道 本例的运行结果如图: 时空隧道demo演示 Demo地址:https://nsytsqdtn.github.io/demo/sprite ...

  8. WebGL three.js学习笔记 法向量网格材质MeshNormalMaterial的介绍和创建360度全景天空盒的方法

    WebGL学习----Three.js学习笔记(5) 点击查看demo演示 Demo地址:https://nsytsqdtn.github.io/demo/360/360 简单网格材质 MeshNor ...

  9. JS学习笔记5_DOM

    1.DOM节点的常用属性(所有节点都支持) nodeType:元素1,属性2,文本3 nodeName:元素标签名的大写形式 nodeValue:元素节点为null,文本节点为文本内容,属性节点为属性 ...

随机推荐

  1. view简写 TemplateView.as_view()

    view简写 TemplateView.as_view() https://code.ziqiangxuetang.com/django/django-generic-views.html (1)如果 ...

  2. vue中的项目目录assets和staitc的区别

    vue中的项目目录assets和staitc的区别 在进行发行正式版时,即为npm run build编译后, assets下的文件如(js.css)都会在dist文件夹下面的项目目录分别合并到一个文 ...

  3. 为Eclipse指定JVM

    运行eclipse时,报如下错误时,可以通过修改配置文件eclipse.ini来解决. Version 1.4.1_01 of the JVM is not suitable for this pro ...

  4. [转]C#读写远程共享文件夹

    1.在服务器设置一个共享文件夹,在这里我的服务器ip地址是10.200.8.73,共享文件夹名字是share,访问权限,用户名是administrator,密码是11111111. 2.新建一个控制台 ...

  5. python16_day17【Django_session、ajax】

    一.Session 1.settings.py SESSION_ENGINE = 'django.contrib.sessions.backends.db' # 引擎(默认) SESSION_COOK ...

  6. Visio Yoeman

    Visio需要确定文件位置才能运行的 Yo主要是用来生成框架的,相当于是一个框架生成器 new install -g generator-django 然后在Visio按F5,选择要创建的环境,就可以 ...

  7. spring和spirngmvc整合

    <!-- 需要进行 Spring 整合 SpringMVC 吗 ? 还是否需要再加入 Spring 的 IOC 容器 ? 是否需要再 web.xml 文件中配置启动 Spring IOC 容器的 ...

  8. hive--[ array、map、struct]使用

    复合数据类型 Structs: structs内部的数据可以通过DOT(.)来存取,例如,表中一列c的类型为STRUCT{a INT; b INT},我们可以通过c.a来访问域a Maps(K-V对) ...

  9. Mixed mode assembly is built against version 'v2.0.50727' of the runtime and cannot be loaded in the 4.0 runtime without additional configuration info

    Mixed mode assembly is built against version 'v2.0.50727' of the runtime and cannot be loaded in the ...

  10. pycharm 常用配置

    lz提示一下,pycharm中的设置是可以导入和导出的,file>export settings可以保存当前pycharm中的设置为jar文件,重装时可以直接import settings> ...