前言

canvas 强大的功能让它成为了 HTML5 中非常重要的部分,至于它是什么,这里就不需要我多作介绍了。而可视化图表,则是 canvas 强大功能的表现之一。

现在已经有了很多成熟的图表插件都是用 canvas 实现的,Chart.js、ECharts等可以制作出好看炫酷的图表,而且几乎覆盖了所有图表的实现。

有时候自己只想画个柱状图,自己写又觉得麻烦,用别人插件又感觉累赘,最后打开百度,拷段代码,粘贴上来修修改改。还不如自己撸一个呢。

原文作者:林鑫,作者博客:https://github.com/lin-xin/blog

效果

动画效果图片显示不出来,可以到最下面找demo地址

分析

可以这个图表由 xy轴、数据条形和标题组成。

  • 轴线:可以使用 moveTo() & lineTo() 实现
  • 文字:可以使用 fillText() 实现
  • 长方形:可以使用 fillRect() 实现

这样看来,似乎并没有多难。

实现

定义画布

  1. <canvas id="canvas" width="600" height="500"></canvas>

canvas 标签只是个容器,真正实现画图的还是 JavaScript。

画坐标轴

坐标轴就是两条横线,也就是canvas里最基础的知识。

  • 由 ctx.beginPath() 开始一条新的路径
  • ctx.lineWidth=1 设置线条宽度
  • ctx.strokeStyle='#000000' 设置线条颜色
  • ctx.moveTo(x,y) 定义线条的起点
  • ctx.lineTo(x1,y1) 定义线条的终点
  • 最后 ctx.stroke() 把起点和终点连成一条线
  1. var canvas = document.getElementById('canvas');
  2. var ctx = canvas.getContext('2d');
  3. var width = canvas.width;
  4. var height = canvas.height;
  5. var padding = 50; // 坐标轴到canvas边框的边距,留边距写文字
  6. ctx.beginPath();
  7. ctx.lineWidth = 1;
  8. // y轴线
  9. ctx.moveTo(padding + 0.5, height - padding + 0.5);
  10. ctx.lineTo(padding + 0.5, padding + 0.5);
  11. ctx.stroke();
  12. // x轴线
  13. ctx.moveTo(padding + 0.5, height - padding + 0.5);
  14. ctx.lineTo(width - padding + 0.5, height - padding + 0.5);
  15. ctx.stroke();

画坐标点

y轴上多少坐标点由自己来定义,需要获取到数据的最大值来计算y轴上的坐标值。x轴的点则由传入的数据长度决定,坐标值由传入数据的 xAxis 属性决定。

  • 坐标值就是文字,由 ctx.fillText(value, x, y) 填充文字,value 为文字值,x y 为值的坐标
  • ctx.textAlign='center' 设置文字居中对齐
  • ctx.fillStyle='#000000' 设置文字填充颜色
  1. var yNumber = 5; // y轴的段数
  2. var yLength = Math.floor((height - padding * 2) / yNumber); // y轴每段的真实长度
  3. var xLength = Math.floor((width - padding * 2) / data.length); // x轴每段的真实长度
  4. ctx.beginPath();
  5. ctx.textAlign = 'center';
  6. ctx.fillStyle = '#000000';
  7. ctx.strokeStyle = '#000000';
  8. // x轴刻度和值
  9. for (var i = 0; i < data.length; i++) {
  10. var xAxis = data[i].xAxis;
  11. var xlen = xLength * (i + 1);
  12. ctx.moveTo(padding + xlen, height - padding);
  13. ctx.lineTo(padding + xlen, height - padding + 5);
  14. ctx.stroke(); // 画轴线上的刻度
  15. ctx.fillText(xAxis, padding + xlen - xLength / 2, height - padding + 15); // 填充文字
  16. }
  17. // y轴刻度和值
  18. for (var i = 0; i < yNumber; i++) {
  19. var y = yFictitious * (i + 1);
  20. var ylen = yLength * (i + 1);
  21. ctx.moveTo(padding, height - padding - ylen);
  22. ctx.lineTo(padding - 5, height - padding - ylen);
  23. ctx.stroke();
  24. ctx.fillText(y, padding - 10, height - padding - ylen + 5);
  25. }

柱状动画

接下来要把数据通过柱状的高低显示出来,这里有个动画效果,柱状会从0升到对应的值。在 canvas 上实现动画我们可以使用 setInterval、setTimeout 和 requestAnimationFrame。

requestAnimationFrame 不需要自己设置定时时间,而是跟着浏览器的绘制走。这样就不会掉帧,自然就流畅。

requestAnimationFrame 原本只支持IE10以上,不过可以通过兼容的写法实现兼容到IE6都行。

  1. function looping() {
  2. looped = requestAnimationFrame(looping);
  3. if(current < 100){
  4. // current 用来计算当前柱状的高度占最终高度的百分之几,通过不断循环实现柱状上升的动画
  5. current = (current + 3) > 100 ? 100 : (current + 3);
  6. drawAnimation();
  7. }else{
  8. window.cancelAnimationFrame(looped);
  9. looped = null;
  10. }
  11. }
  12. function drawAnimation() {
  13. for(var i = 0; i < data.length; i++) {
  14. var x = Math.ceil(data[i].value * current / 100 * yRatio);
  15. var y = height - padding - x;
  16. ctx.fillRect(padding + xLength * (i + 0.25), y, xLength/2, x);
  17. // 保存每个柱状的信息
  18. data[i].left = padding + xLength / 4 + xLength * i;
  19. data[i].top = y;
  20. data[i].right = padding + 3 * xLength / 4 + xLength * i;
  21. data[i].bottom = height - padding;
  22. }
  23. }
  24. looping();
  • 柱状即是画矩形,由 ctx.fillRect(x, y, width, height) 实现,x y 为矩形左上角的坐标,width height 为矩形的宽高,单位为像素
  • ctx.fillStyle='#1E9FFF' 设置填充颜色

到这里,一个最基本的柱状图就完成了。接下来,我们可以为他添加标题。

标题

要放置标题,就会发现我们一大早定义的 padding 内边距确实有用,总不能把标题给覆盖到柱状图上吧。但是标题有的是在顶部,有的在底部,那么就不能写死了。定一个变量 position 来判断位置去画出来。这个简单。

  1. // 标题
  2. if(title){ // 也不一定有标题
  3. ctx.textAlign = 'center';
  4. ctx.fillStyle = '#000000'; // 颜色,也可以不用写死,个性化嘛
  5. ctx.font = '16px Microsoft YaHei'
  6. if(titlePosition === 'bottom' && padding >= 40){
  7. ctx.fillText(title,width/2,height-5)
  8. }else{
  9. ctx.fillText(title,width/2,padding/2)
  10. }
  11. }

监听鼠标移动事件

我们看到,有些图表,把鼠标移上去,当前的柱状就变色了,移开之后又变回原来的颜色。这里就需要监听 mouseover 事件,当鼠标的位置位于柱状的面积内,触发事件。

那我怎么知道在柱状里啊,发现在 drawAnimation() 里会有每个柱状的坐标,那我干脆把坐标给保存到 data 里。那么鼠标在柱状里的条件应该是:

  • ev.offsetX > data[i].left
  • ev.offsetX < data[i].right
  • ev.offsetY > data[i].top
  • ev.offsetY < data[i].bottom
  1. canvas.addEventListener('mousemove',function(ev){
  2. var ev = ev||window.event;
  3. for (var i=0;i<data.length;i++){
  4. for (var i=0;i<data.length;i++){
  5. if(ev.offsetX > data[i].left &&
  6. ev.offsetX < data[i].right &&
  7. ev.offsetY > data[i].top &&
  8. ev.offsetY < data[i].bottom){
  9. console.log('我在第'+i+'个柱状里。');
  10. }
  11. }
  12. })

总结

为了更方便的使用,封装成构造函数。通过

  1. var chart = new sBarChart('canvas',data,{
  2. title: 'xxx公司年度盈利', // 标题
  3. titleColor: '#000000', // 标题颜色
  4. titlePosition: 'top', // 标题位置
  5. bgColor: '#ffffff', // 背景色
  6. fillColor: '#1E9FFF', // 柱状填充色
  7. axisColor: '#666666', // 坐标轴颜色
  8. contentColor: '#a5f0f6' // 内容横线颜色
  9. });

参数可配置,很简单就生成一个个性化的柱状图。代码地址:canvas-demo

最后加上折线图、饼图、环形图,完整封装成sChart.js插件,插件地址:sChart.js

更多文章:lin-xin/blog

HTML5 进阶系列:canvas 动态图表的更多相关文章

  1. HTML5 进阶系列:indexedDB 数据库

    前言 在 HTML5 的本地存储中,有一种叫 indexedDB 的数据库,该数据库是一种存储在客户端本地的 NoSQL 数据库,它可以存储大量的数据.从上篇:HTML5 进阶系列:web Stora ...

  2. HTML5 进阶系列:文件上传下载

    前言 HTML5 中提供的文件API在前端中有着丰富的应用,上传.下载.读取内容等在日常的交互中很常见.而且在各个浏览器的兼容也比较好,包括移动端,除了 IE 只支持 IE10 以上的版本.想要更好地 ...

  3. HTML5 进阶系列:拖放 API 实现拖放排序

    前言 HTML5 中提供了直接拖放的 API,极大的方便我们实现拖放效果,不需要去写一大堆的 js,只需要通过监听元素的拖放事件就能实现各种拖放功能. 想要拖放某个元素,必须设置该元素的 dragga ...

  4. HTML5 进阶系列:拖放 API 实现拖放排序(转载)

    HTML5之拖放API实现拖放排序 前言 HTML5 中提供了直接拖放的 API,极大的方便我们实现拖放效果,不需要去写一大堆的 js,只需要通过监听元素的拖放事件就能实现各种拖放功能. 想要拖放某个 ...

  5. 第五讲:使用html5中的canvas动态画出物理学上平抛运动

    <html> <head> <title>平抛运动</title> <script src="../js/jscex.jscexRequ ...

  6. html5进阶之canvas图像基础

    1.首先还是使用之前讲过的Image()函数来预加载图像. 在调用图像之前,需创建一个事件监听器,为其保证图像已经正确的加载. 如下图: 监听图片已经正确加载 2.把图像显示在画布上面,这里将不再使用 ...

  7. C#进阶系列——动态Lamada(二:优化)

    前言:前几天写了一篇动态Lamada的文章C#进阶系列——动态Lamada,受园友xiao99的启发,今天打算来重新优化下这个动态Lamada的工具类.在此做个笔记,以免以后忘了. 一.原理分析 上篇 ...

  8. 【转载】C#进阶系列——动态Lamada(二:优化)

    前言:前几天写了一篇动态Lamada的文章C#进阶系列——动态Lamada,受园友xiao99的启发,今天打算来重新优化下这个动态Lamada的工具类.在此做个笔记,以免以后忘了. 一.原理分析 上篇 ...

  9. 怎样用JavaScript和HTML5 Canvas绘制图表

    原文:https://code.tutsplus.com/zh-...原作:John Negoita翻译:Stypstive 在这篇教程中,我将展示用JavaScript和canvas作为手段,在饼状 ...

随机推荐

  1. 性能测试培训:sql server性能测试分析局部变量的性能影响

    poptest是国内唯一一家培养测试开发工程师的培训机构,以学员能胜任自动化测试,性能测试,测试工具开发等工作为目标.在poptest的loadrunner的培训中,为了提高学员性能优化的经验,加入了 ...

  2. 我的iOS-App

    1.PocketConfidential(密保箱) 简介 保存账号密码等敏感信息. 应用技术: sqlite.sqlcipher加密.AES数据加密.GCD https://itunes.apple. ...

  3. JavaScript基础学习(五)—其他引用类型

         JavaScript定义了两个内置对象: Global和Math. 一.Global对象 1.URI编码方法      Global对象的encodeURI()和encodeURICompo ...

  4. Andorid开发中如何去除标题栏title

    有两种方法可以去除标题栏的title. 1.在代码中实现 在setContentView()方法之前加上这一句 requestWindowFeature(Window.FEATURE_NO_TITLE ...

  5. mysql性能优化配置总结

    看了一些优化mysql运维的一些书籍,在此记录总结下:进入mysql客户端输入以下sql:1.连接设置 show variables like '%max_connection%'; show sta ...

  6. JS前端数据格式化

    当我们从后台取了数据,但是我们希望在前台统一显示格式时,我们可能需要格式化数据. 今天正好总结一下前端JS格式化数据的几个方法: 1. toFixed() 方法   可把 Number 四舍五入为指定 ...

  7. jDialects:一个从Hibernate抽取的支持70多种数据库方言的原生SQL分页工具

    jDialects(https://git.oschina.net/drinkjava2/jdialects) 是一个收集了大多数已知数据库方言的Java小项目,通常可用来创建分页SQL和建表DDL语 ...

  8. 数据仓库Hive数据导入导出

    Hive库数据导入导出 1.新建表data hive (ebank)> create table data(id int,name string) > ROW FORMAT DELIMIT ...

  9. EasyUI datagrid默认勾选checkbox时注意事项

    在使用easyui的datagrid默认选中复选框时遇到的一个问题:就是加载程序默认选中复选框时死活选不中,查了好多资料才知道是easyui的datagrid的singleSelect属性设置为‘tr ...

  10. sed命令详解-应用篇

    本篇从实用的角度讲解sed,关于sed的详细帮助文档,请参考前篇 http://www.cnblogs.com/the-capricornus/p/5279979.html 本篇用到的选项请参考前篇. ...