D3.js实现简洁实用的动态仪表盘的示例


Posted in Javascript onApril 04, 2018

本文介绍了D3.js实现简洁实用的动态仪表盘的示例,分享给大家,具体如下:

动态效果图:

D3.js实现简洁实用的动态仪表盘的示例

仪表盘效果图

细看上面的动态效果图,可以发现:

  1. 一个值变换到一个新的值时,是一个渐变的过程;
  2. 圆弧末尾有一个竖线,作为仪表盘的指针,在仪表盘数值变化时,有一个弹性的动画效果。

一开始,我是用Echarts来实现仪表盘,但是它无法满足上面的两点需求。所以后来改成用D3.js。

D3.js可以完美地实现图表的定制,从细节上,完美地满足我们的需求。

初始化仪表盘

1、首先定义一个svg元素:

<svg id="myGauge" width="80" height="108" ></svg>

然后,声明一些变量用于初始化:

var width=80, 
  height=108,  //svg的高度和宽度,也可以通过svg的width、height属性获取
  innerRadius = 22,
  outerRadius = 30, //圆弧的内外半径
  arcMin = -Math.PI*2/3,
  arcMax = Math.PI*2/3, //圆弧的起始角度和终止角度

2、创建一个 arc 方法,并设置所有的属性,除了 endAngle。在创建圆弧的时候,传递一个包含 endAngle 属性的对象到这个方法,就可以计算出一个给定角度的 SVG path。

var arc = d3.arc()
  .innerRadius(22)
  .outerRadius(30)
  .startAngle(arcMin)

圆弧角度怎么设置呢?

把一个圆圈对应到一个时钟,那么12点钟对应的角度就是0,顺时针3点钟的角度是Math.PI/2,逆时针6点钟的角度是-Math.PI。因此-Math.PI*2/3到Math.PI*2/3的圆弧形状如上面的效果图所示。更多参考API文档中的arc.startAngle。

3、获取SVG元素,并且转换原点到画布的中心,这样我们在之后创建圆弧时就不需要再单独指定它们的位置了

var svg = d3.select("#myGauge")
var g = svg.append("g").attr("transform", "translate(" + width / 2 + "," + height / 2 + ")");

4、添加仪表盘中的文字(标题,数值,单位)

//添加仪表盘的标题
g.append("text").attr("class", "gauge-title")
  .style("alignment-baseline", "central") //相对父元素对齐方式
  .style("text-anchor", "middle") //文本锚点,居中
  .attr("y", -45)  //到中心的距离
  .text("CPU占用率");
//添加仪表盘显示的数值,因为之后还要更新,所以声明一个变量
var valueLabel = g.append("text").attr("class", "gauge-value")
  .style("alignment-baseline", "central") //相对父元素对齐方式
  .style("text-anchor", "middle") //文本锚点,居中
  .attr("y", 25)  //到中心的距离
  .text(12.65); 
//添加仪表盘显示数值的单位      
g.append("text").attr("class", "gauge-unity")
  .style("alignment-baseline", "central") //相对父元素对齐方式
  .style("text-anchor", "middle") //文本锚点,居中
  .attr("y", 40)  //到中心的距离
  .text("%");

D3制作的SVG图,与Echarts绘制的Canvas比起来,很重要的一个优点是,可以用CSS定义SVG的样式。比如,此处仪表盘标题的样式如下:

.gauge-title{
  font-size: 10px;
  fill: #A1A6AD;
}

5、添加背景圆弧

//添加背景圆弧
var background = g.append("path")
  .datum({endAngle:arcMax}) //传递endAngle参数到arc方法
  .style("fill", "#444851")
  .attr("d", arc);

6、添加表示百分比的圆弧,其中percentage是要表示的百分比,0到1的小数。

//计算圆弧的结束角度
var currentAngle = percentage*(arcMax-arcMin) + arcMin
//添加另一层圆弧,用于表示百分比
var foreground = g.append("path")
  .datum({endAngle:currentAngle})
  .style("fill", "#444851")
  .attr("d", arc);

7、在圆弧末尾添加一个指针标记

var tick = g.append("line")
  .attr('class', 'gauge-tick')
  .attr("x1", 0)
  .attr("y1", -innerRadius)
  .attr("x2", 0)
  .attr("y2", -(innerRadius + 12)) //定义line位置,默认是在圆弧正中间,12是指针的长度
  .style("stroke", "#A1A6AD")
  .attr('transform', 'rotate('+ angleToDegree(currentAngle) +')')

rotate中的参数是度数,Math.PI对应180,因此需要自定义一个angleToDegree方法把currentAngle转换一下。

至此,一个SVG仪表盘就制作出来了,不过是静止的,那怎么更新这个仪表盘呢?

更新仪表盘

需要更新:表示新的百分比的圆弧;圆弧下方的数值。

修改圆弧下方的数值很简单:

valueLabel.text(newValue)

更新圆弧则稍麻烦一点,具体思路是:修改圆弧的endAngle,以及修改圆弧末尾指针的transform值。

实现的过程中,需要使用的API:

selection.transition:https://github.com/d3/d3-transition/blob/master/README.md#selection_transition
transition.attrTween:https://github.com/d3/d3-transition/blob/master/README.md#transition_attrTween
d3.interpolate:https://github.com/d3/d3-interpolate/blob/master/README.md#interpolate

1、更新圆弧,其中angle为新圆弧的结束角度。

//更新圆弧,并且设置渐变动效
foreground.transition()
  .duration(750)
  .ease(d3.easeElastic)  //设置来回弹动的效果
  .attrTween("d", arcTween(angle));

arcTween方法定义如下。它返回一个d属性的补间(渐变)动画方法,使一个圆弧从当前的角度渐变到另一个新的角度。

arcTween(newAngle) {
  let self=this
  return function(d) {
    var interpolate = d3.interpolate(d.endAngle, newAngle); //在两个值间找一个插值
    return function(t) {
      d.endAngle = interpolate(t);  //根据 transition 的时间 t 计算插值并赋值给endAngle
      return arc(d); //返回新的“d”属性值
    }; 
  };
}

这个方法更详细的说明可以参考Arc Tween中的注释。

2、更新圆弧末尾的指针的原理同上,其中oldAngle是旧圆弧的结束角度。

//更新圆弧末端的指针标记,并且设置渐变动效      
tick.transition()
  .duration(750)
  .ease(d3.easeElastic)  //设置来回弹动的效果
  .attrTween('transform', function(){ //设置“transform”属性的渐变,原理同上面的arcTween方法
    var i = d3.interpolate(angleToDegree(oldAngle), angleToDegree(newAngle));  //取插值
    return function(t) {
      return 'rotate('+ i(t) +')'
    };
  })

至此,我们就成功制作了一个动态刷新的简介美观的SVG仪表盘。

结束

每次使用D3.js都会忍不住觉得它真是太强大、太有意思了~它就像是一个百宝箱,让我们最大限度地满足需求。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持三水点靠木。

Javascript 相关文章推荐
jQuery + Flex 通过拖拽方式动态改变图片的代码
Aug 03 Javascript
jQuery仿Excel表格编辑功能的实现代码
May 01 Javascript
web css实现整站样式互相切换
Oct 29 Javascript
javascript结合ajax读取txt文件内容
Dec 05 Javascript
jQuery插件multiScroll实现全屏鼠标滚动切换页面特效
Apr 12 Javascript
关于延迟加载JavaScript
May 05 Javascript
jQuery的ajax下载blob文件
Jul 21 Javascript
jQuery简单实现列表隐藏和显示效果示例
Sep 12 Javascript
jQuery用户头像裁剪插件cropbox.js使用详解
Jun 07 jQuery
JS栈stack类的实现与使用方法示例
Jan 31 Javascript
使用vue完成微信公众号网页小记(推荐)
Apr 28 Javascript
JS如何寻找数组中心索引过程解析
Jun 01 Javascript
vue.js实现点击后动态添加class及删除同级class的实现代码
Apr 04 #Javascript
10分钟上手vue-cli 3.0 入门介绍
Apr 04 #Javascript
详解vue 单页应用(spa)前端路由实现原理
Apr 04 #Javascript
vue cli升级webapck4总结
Apr 04 #Javascript
JavaScript实现短暂提示框功能
Apr 04 #Javascript
Vue filter介绍及详细使用
Apr 04 #Javascript
在Vue项目中引入腾讯验证码服务的教程
Apr 03 #Javascript
You might like
天使彦史上最神还原,性别曝光的那一刻,百万网友恋爱了
2020/03/02 国漫
PHP 批量删除 sql语句
2009/06/05 PHP
Ajax+PHP快速上手及简单应用说明
2013/07/24 PHP
php生成N个不重复的随机数实例
2013/11/12 PHP
php检测apache mod_rewrite模块是否安装的方法
2015/03/14 PHP
php中文验证码实现方法
2015/06/18 PHP
Yii2-GridView 中让关联字段带搜索和排序功能示例
2017/01/21 PHP
Laravel 5+ .env环境配置文件详解
2020/04/06 PHP
Code:loadScript( )加载js的功能函数
2007/02/02 Javascript
js中匿名函数的N种写法
2010/09/08 Javascript
JS实现仿QQ面板的手风琴效果折叠菜单代码
2015/09/11 Javascript
JavaScript切换搜索引擎的导航网页搜索框实例代码
2017/06/11 Javascript
vue不通过路由直接获取url中参数的方法示例
2017/08/24 Javascript
node.js实现的装饰者模式示例
2017/09/06 Javascript
实现div滚动条默认最底部以及默认最右边的示例代码
2017/11/15 Javascript
深入理解 webpack 文件打包机制(小结)
2018/01/08 Javascript
jQuery实现的电子时钟效果完整示例
2018/04/28 jQuery
基于jQuery.i18n实现web前端的国际化
2018/05/04 jQuery
vue.js使用v-if实现显示与隐藏功能示例
2018/07/06 Javascript
vue.js实现的幻灯片功能示例
2019/01/18 Javascript
mui js控制开关状态、修改switch开关的值方法
2019/09/03 Javascript
vue中keep-alive、activated的探讨和使用详解
2020/07/26 Javascript
Python for Informatics 第11章之正则表达式(二)
2016/04/21 Python
Python网络爬虫实例讲解
2016/04/28 Python
python构建指数平滑预测模型示例
2019/11/21 Python
flask框架json数据的拿取和返回操作示例
2019/11/28 Python
前端canvas水印快速制作(附完整代码)
2019/09/19 HTML / CSS
联想印度官方网上商店:Lenovo India
2019/08/24 全球购物
工程班组长岗位职责
2013/12/30 职场文书
工厂总经理岗位职责
2014/02/07 职场文书
信仰心得体会
2014/09/05 职场文书
公证委托书格式
2014/09/13 职场文书
应届毕业生求职信范文
2015/03/19 职场文书
2015年城管个人工作总结
2015/05/15 职场文书
初中数学教学反思范文
2016/02/17 职场文书
pandas:get_dummies()与pd.factorize()的用法及区别说明
2021/05/21 Python