d3.js 共享交换平台demo
今天在群里遇到一张图 遂来玩一玩,先来上图!!

点击相应按钮,开关线路,此项目的重点是计算相应图形的位置,由于是个性化项目就没有封装布局。好了直接上代码。
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>$Title$</title>
<script src="https://d3js.org/d3.v4.min.js"></script>
<style>
.container {
padding: 0 100px;
}
button {
width: 100px;
}
.rect {
stroke: #406DA7;
fill: #6693CC;
stroke-width: 2;
}
.center {
stroke: #FF3336;
fill: #FF7F7E;
stroke-width: 2;
}
.text {
fill: #ffffff;
text-anchor: middle;
font-size: 30px;
shape-rendering: crispEdges;
}
.circle {
fill: #ffffff;
stroke: #FF7F7E;
stroke-width: 2;
}
.cha {
fill: #ffffff;
stroke: #FF7F7E;
stroke-width: 2;
}
.dash1 {
stroke-dashArray: 15 5;
stroke-width: 4;
stroke: #6693CC;
}
.dash2 {
stroke-dashArray: 15 5;
stroke-width: 4;
stroke: #FF7F7E;
}
.marker1 {
fill: none;
stroke: #6693CC;
stroke-width: 4;
}
.marker2 {
fill: none;
stroke: #FF7F7E;
stroke-width: 4;
}
.line {
stroke: #6693CC;
stroke-width: 4;
}
</style>
</head>
<body>
<script>
var width = 2000;
var height = 1200;
var margin = 100;
var everyW = 280;
var leftData = ['上海市城管局','北京市城管局','天津市城管局','重庆市城管局','大庆市城管局','黑河市城管局','鸡西市城管局','鹤岗市城管局'];
var rightData = ['上海市建筑委','北京市建筑委','天津市建筑委','重庆市建筑委','大庆市建筑委','黑河市建筑委','鸡西市建筑委','鹤岗市建筑委'];
var everyH = ( height - margin * 2 ) / leftData.length / 2;
var scaleL = d3.scaleLinear().domain([0, leftData.length - 1]).range([0, height - margin * 2 - everyH])
var scaleR = d3.scaleLinear().domain([0, rightData.length - 1]).range([0, height - margin * 2 - everyH])
var text = '共享交换平台' render(); function render() {
var svg = d3.select('body')
.append('svg')
.attr('width', 1000)
.attr('height', 600)
.attr('viewBox', `0, 0, ${width}, ${height}`)
.attr("preserveAspectRatio", "xMidYMid meet")
.style('border', '1px solid #777') var defs = svg.append('defs') var cha = defs
.append('g')
.attr('id', 'cha') cha.append('circle')
.attr('cx', 20)
.attr('cy', 20)
.attr('r', 19)
.attr('class', 'circle') cha.append('path')
.attr('d', 'M 11 15 L 25 31 A 3 3 0 0 0 29 25 L 15 9 A 3 3 0 0 0 11 15 Z')
.attr('class', 'cha') cha.append('path')
.attr('d', 'M 29 15 L 15 31 A 3 3 0 0 1 11 25 L 25 9 A 3 3 0 0 1 29 15 Z')
.attr('class', 'cha') var markerA1 = defs
.append('g')
.attr('id', 'markerA1') markerA1.append('line')
.attr('x1', everyW + 2)
.attr('y1', everyH / 2)
.attr('x2', everyW * 1 / 4 + width / 4 - margin / 2)
.attr('y2', everyH / 2)
.attr('class', 'dash1') var markerA2 = defs
.append('g')
.attr('id', 'markerA2') markerA2.append('line')
.attr('x1', everyW + 2)
.attr('y1', everyH / 2)
.attr('x2', everyW * 1 / 4 + width / 4 - margin / 2)
.attr('y2', everyH / 2)
.attr('class', 'dash2') markerA2.append('use')
.attr('x', everyW * 5 / 8 + width / 8 - margin / 4 - 20)
.attr('y', everyH / 2 - 20)
.attr('xlink:href', '#cha') var markerB1 = defs
.append('g')
.attr('id', 'markerB1') markerB1.append('line')
.attr('x1', everyW * 3 / 4 - width / 4 + margin / 2)
.attr('y1', everyH / 2)
.attr('x2', -10)
.attr('y2', everyH / 2)
.attr('class', 'dash1') markerB1.append('path')
.attr('d', `M -30 ${everyH / 2 - 16} L -10 ${everyH / 2} L -30 ${everyH / 2 + 16}`)
.attr('class', 'marker1') var markerB2 = defs
.append('g')
.attr('id', 'markerB2') markerB2.append('line')
.attr('x1', everyW * 3 / 4 - width / 4 + margin / 2)
.attr('y1', everyH / 2)
.attr('x2', -10)
.attr('y2', everyH / 2)
.attr('class', 'dash2') markerB2.append('path')
.attr('d', `M -30 ${everyH / 2 - 16} L -10 ${everyH / 2} L -30 ${everyH / 2 + 16}`)
.attr('class', 'marker2') markerB2.append('use')
.attr('x', everyW * 3 / 8 - width / 8)
.attr('y', everyH / 2 - 20)
.attr('xlink:href', '#cha') var body = svg.append('g')
.attr('transform', `translate(${margin},${margin})`)
.attr('class', 'body') var leftG = body.append('g').attr('class', 'left');
var rightG = body.append('g').attr('class', 'right'); let leftObj = leftG.selectAll('g.left-g').data(leftData).enter().append('g').attr('class', 'left-g').attr('transform', (d,i) => `translate(0,${scaleL(i)})`).attr('id', (d,i) => `l_${i}`);
let rightObj = rightG.selectAll('g.right-g').data(rightData).enter().append('g').attr('class', 'right-g').attr('transform', (d,i) => `translate(${width - margin * 2 - everyW},${scaleL(i)})`).attr('id', (d,i) => `r_${i}`);; leftObj.append('rect')
.attr('width', everyW)
.attr('height', everyH)
.attr('rx', 6)
.attr('class', 'rect') leftObj.append('text')
.attr('class', 'text')
.attr('dy', '1.4em')
.text(d => d)
.attr('transform', `translate(${everyW / 2}, 0)`) leftObj.append('use')
.attr('x',0)
.attr('y', 0)
.attr('xlink:href', '#markerA1') rightObj.append('rect')
.attr('width', everyW)
.attr('height', everyH)
.attr('rx', 6)
.attr('class', 'rect') rightObj.append('text')
.attr('class', 'text')
.attr('dy', '1.4em')
.text(d => d)
.attr('transform', `translate(${everyW / 2}, 0)`) rightObj.append('use')
.attr('x',0)
.attr('y', 0)
.attr('xlink:href', '#markerB1') var centerObj = body.append('g')
.attr('transform', `translate(${width / 2 - margin - everyW / 2}, ${height / 2 - margin - everyH / 2 })`) centerObj.append('rect')
.attr('width', everyW)
.attr('height', everyH)
.attr('rx', 6)
.attr('class', 'center') centerObj.append('text')
.attr('class', 'text')
.attr('dy', '1.4em')
.text(text)
.attr('transform', `translate(${everyW / 2}, 0)`) body.append('line')
.attr('x1', everyW * 1 / 4 + width / 4 - margin / 2)
.attr('y1', everyH / 2 - 2)
.attr('x2', everyW * 1 / 4 + width / 4 - margin / 2)
.attr('y2', height - margin * 2 - everyH / 2 + 2)
.attr('class', 'line') body.append('line')
.attr('x1', width * 3 / 4 - margin * 3 / 2 - everyW * 1 / 4)
.attr('y1', everyH / 2 - 2)
.attr('x2', width * 3 / 4 - margin * 3 / 2 - everyW * 1 / 4)
.attr('y2', height - margin * 2 - everyH / 2 + 2)
.attr('class', 'line') body.append('line')
.attr('x1', everyW * 1 / 4 + width / 4 - margin / 2)
.attr('y1', height / 2 - margin)
.attr('x2', width / 2 - margin - everyW / 2 - 10)
.attr('y2', height / 2 - margin)
.attr('class', 'line') body.append('path')
.attr('d', `M ${width / 2 - margin - everyW / 2 - 30} ${height / 2 - margin - 16} L ${width / 2 - margin - everyW / 2 - 10} ${height / 2 - margin} L ${width / 2 - margin - everyW / 2 - 30} ${height / 2 - margin + 16}`)
.attr('class', 'marker1') body.append('line')
.attr('x1', (width - margin * 2 - 3 * everyW ) / 2 + 2 * everyW)
.attr('y1', height / 2 - margin)
.attr('x2', (width - margin * 2 - 3 * everyW ) * 3 / 4 + 2 * everyW)
.attr('y2', height / 2 - margin)
.attr('class', 'line')
} function clickLeft(dom, num) {
if(dom.getAttribute('off') == "true") {
dom.setAttribute('off', "false");
d3.select(".left").select(`#l_${num}`).select('use').attr('xlink:href', '#markerA2');
} else {
dom.setAttribute('off', "true");
d3.select(".left").select(`#l_${num}`).select('use').attr('xlink:href', '#markerA1');
}
} function clickRight(dom, num) {
if(dom.getAttribute('off') == "true") {
dom.setAttribute('off', "false");
d3.select(".right").select(`#r_${num}`).select('use').attr('xlink:href', '#markerB2');
} else {
dom.setAttribute('off', "true");
d3.select(".right").select(`#r_${num}`).select('use').attr('xlink:href', '#markerB1');
}
} </script> <div class="container">
<table>
<tr>
<td>左侧</td>
<td><button off="true" onclick="clickLeft(this, 0)">1</button></td>
<td><button off="true" onclick="clickLeft(this, 1)">2</button></td>
<td><button off="true" onclick="clickLeft(this, 2)">3</button></td>
<td><button off="true" onclick="clickLeft(this, 3)">4</button></td>
<td><button off="true" onclick="clickLeft(this, 4)">5</button></td>
<td><button off="true" onclick="clickLeft(this, 5)">6</button></td>
<td><button off="true" onclick="clickLeft(this, 6)">7</button></td>
<td><button off="true" onclick="clickLeft(this, 7)">8</button></td>
</tr>
<tr>
<td>右侧</td>
<td><button off="true" onclick="clickRight(this, 0)">1</button></td>
<td><button off="true" onclick="clickRight(this, 1)">2</button></td>
<td><button off="true" onclick="clickRight(this, 2)">3</button></td>
<td><button off="true" onclick="clickRight(this, 3)">4</button></td>
<td><button off="true" onclick="clickRight(this, 4)">5</button></td>
<td><button off="true" onclick="clickRight(this, 5)">6</button></td>
<td><button off="true" onclick="clickRight(this, 6)">7</button></td>
<td><button off="true" onclick="clickRight(this, 7)">8</button></td>
</tr>
</table>
</div> </body>
</html>
这里坐标原点没有挡在画布中心使得计算量增大(多思考确实可以减少好多代码量)。
由于是定制化很高的简单的demo,所有代码没有做模块化,用到的人不会很多,但是学好d3,可以让我们做更多更好的定制化项目。
想预览代码和下载的朋友请移步至 http://www.bettersmile.cn
d3.js 共享交换平台demo的更多相关文章
- 前端使用d3.js调用地图api 进行数据可视化
前段时间自己研究了demo就是把某个区域的某个位置通过经纬度在地图上可视化.其实就是使用了第三方插件,比现在比较火的可视化插件d3.js echart.js.大致思路就是,把要用到的位置的geojso ...
- 【 D3.js 进阶系列 — 6.1 】 缩放的应用(Zoom)
缩放(Zoom)是另一种重要的可视化操作,主要是使用鼠标的滚轮进行. 1. zoom 的定义 缩放是由 d3.behavior.zoom() 定义的. var zoom = d3.behavior.z ...
- D3.js 简介和安装
一.What´s D3.js D3.js是一种数据操作类型的javascript库(也可视其为插件):结合HTML,SVG和CSS,D3可以图形化的,生动的展现数据. D3 的全称是(Data-Dri ...
- 【 D3.js 高级系列 — 10.0 】 思维导图
思维导图的节点具有层级关系和隶属关系,很像枝叶从树干伸展开来的形状.在前面讲解布局的时候,提到有五个布局是由层级布局扩展来的,其中的树状图(tree layout)和集群图(cluster layou ...
- 【 D3.js 高级系列 — 9.0 】 交互式提示框
一般来说,图表中不宜存在过多文字.但是,有时需要一些文字来描述某些图形元素.那么,可以实现一种交互:当用户鼠标滑到某图形元素时,出现一个提示框,里面写有描述文字.这是一种简单.普遍的交互式,几乎适用于 ...
- 【 D3.js 高级系列 — 8.0 】 标线
有时候,需要在地图上绘制连线,表示"从某处到某处"的意思,这种时候在地图上绘制的连线,称为"标线". 1. 标线是什么 标线,是指地图上需要两个坐标以上才能表示 ...
- 【 D3.js 高级系列 — 7.0 】 标注地点
有时需要告诉用户地图上的一些目标,如果该目标是只需要一个坐标就能表示的,称其为"标注". 1. 标注是什么 标注,是指地图上只需要一个坐标即可表示的元素.例如,在经纬度(116, ...
- 【 D3.js 高级系列 — 6.0 】 值域和颜色
在[入门 - 第 10 章]作了一张中国地图,其中各省份的颜色值都是随意赋值的.如果要将一些值反映在地图上,可以利用颜色的变化来表示值的变化. 1. 思路 例如,有值域的范围为: [10, 500] ...
- 【 D3.js 高级系列 — 5.1 】 颜色插值和线性渐变
颜色插值指的是给出两个 RGB 颜色值,两个颜色之间的值通过插值函数计算得到.线性渐变是添加到 SVG 图形上的过滤器,只需给出两端的颜色值即可. 1. 颜色插值 在[高级 - 第 5.0 章]里已经 ...
随机推荐
- Android的系统框架
Android的系统架构采用了分层架构的思想,如图1所示.从上层到底层共包括四层,分别是应用程序程序层.应用框架层.系统库和Android运行时和Linux内核. 图1:Android系统架构图 每 ...
- 2018092609-2 选题 Scrum立会报告+燃尽图 02
此作业要求参见:[https://edu.cnblogs.com/campus/nenu/2019fall/homework/8683] 一.小组情况组长:迟俊文组员:宋晓丽 梁梦瑶 韩昊 刘信鹏队名 ...
- Rust 入门 (二)
我认为学习计算机语言,应该先用后学,这一节,我们来实现一个猜数字的小游戏. 先简单介绍一个这个游戏的内容:游戏先生成一个1到100之间的任意一个数字,然后我们输入自己猜测的数字,游戏会告诉我们输入的数 ...
- EasyCode实现数据库到Swagger全自动化
简介 EasyCode是基于IntelliJ IDEA开发的代码生成插件,通过自定义生成模板可以完成定制化的 Mapper Service Controller 生成,结合数据库 Comment还可以 ...
- RESTful API的理解
技术交流的时候遇到了这样的一个问题,被问及开发中用到的是不是Restful API,我说的是,我们现在用到的不属于完全是Restful API.因为我了解到的Restful API,是 通过具体的UR ...
- Java多线程——锁
Java多线系列文章是Java多线程的详解介绍,对多线程还不熟悉的同学可以先去看一下我的这篇博客Java基础系列3:多线程超详细总结,这篇博客从宏观层面介绍了多线程的整体概况,接下来的几篇文章是对多线 ...
- 小白的springboot之路(六)、跨域解决方案CORS
0-前言 前后端分离.分布式集群,经常都会涉及到跨域访问,而浏览器基于同源策略,正常情况下是不能跨域的,这就需要我们解决跨域访问问题:spring boot解决跨域也比较简单: 1-CORS跨域解决方 ...
- 利用tp5开发智慧软文发布系统中遇到的一些坑
1. PHP 计算两个时间戳之间相差的时间 假设你两个时间戳为$a,$b; 你可以用$c=$a-$b;(反正就是大的减小的),这时$c就是两个时间间隔的秒数了. 想求两个时间间隔的天数就用:$c/(6 ...
- 简单高效的端口扫描python脚本
欢迎python爱好者加入:学习交流群 667279387 最近为了获取虚拟机端口开放情况,写了一个简单脚本来查看.共享给大家.下面的代码在python2种测试通过 说明:concurrent是pyt ...
- 并查集 2019年8月10日计蒜客联盟周赛 K.数组
题目链接:https://nanti.jisuanke.com/t/40860 题意:给一个长度为n的数组a[],n<1e5,a[i]<1e5 三个操作: 1 x y:把所有值为x的数据改 ...