【带着canvas去流浪(7)】绘制水球图

栏目: Html5 · 发布时间: 5年前

内容简介:[TOC]使用原生

【带着canvas去流浪(7)】绘制水球图

[TOC]

一. 任务说明

使用原生 canvasAPI 绘制水球图,这将是一个非常有意思的挑战任务。水球图是一种常见的加载动画,属于扩展图形,在 echarts 中使用时需要下载扩展库(同为扩展库的还包括文字云插件和地图插件,项目地址为 https://github.com/ecomfe/echarts-liquidfill )。

【带着canvas去流浪(7)】绘制水球图

二. 重点提示

水球图的绘制有以下几个难点:

  1. 水波的绘制

    水波的绘制实际上是运用简谐振动公式来模拟的,也就是 x = A*(wt +φ) ,其中振幅 A 决定了水波的波纹高低,角频率 w 决定了水波的快慢,相位 φ 决定了初始位移差,再加上一些y轴方向的位移偏差和颜色的差异,就可以模拟出不同的水波,接着只需要在帧动画中不断改变 φ 并重绘曲线,就可以模拟出水波效果了。

  2. 球形剪裁区域

    水波的范围是不能流出球形的外轮廓的,此处的做法是在绘制水波之前,先使用 context.clip( ) 方法将水波的可见绘图区域控制在水球之内即可,如果还有水球外的图形需要绘制,记得在每一帧绘制完水波后调用 context.restore( ) 取消掉之前的剪裁。

  3. 文字的绘制

    如果只是绘制漂浮于水球图之上的文字,是比较容易实现的,但是如果想要实现一些细节更丰富的效果,并不那么容易。我们期望实现的效果是,当文字未被水波浸入时,显示水纹的蓝色,而被水浸润的部分显示为白色,这样看起来更加生动。但是绘制起来却并不容易,如果将文字绘制成蓝色,那么被水淹没的部分就会消失在水纹中,如果绘制成白色,那么水纹高度较小时,会完全看不到文字。那么这样的渲染文字要如何实现呢?

三. 示例代码

let options = {
    value:0,
    a:20,//振幅
    pos:[300,300],//水球图位置
    r:160,//水球图半径
    color:['#2E5199','#1567c8','#1593E7','#42B8F9']//水纹颜色
};

start(options);

/**
 * 绘制水球图
 */
function start(options) {
    //移动绘图坐标至水球图左边界点
    context.translate(options.pos[0],options.pos[1]);
    context.font = 'bold 60px Arial';
    context.textAlign='center';
    context.textBaseLine = 'baseline';
    //计算水球图绘图数据
    createParams(options);
    //开启帧动画
    requestAnimationFrame(startAnim);
}

//生成水波动画参数,位置坐标公式为 y = A * (wt + φ)
function createParams(options) {
    options.w = [];//存储水波的角速度
    options.theta = [];//存储每条水波的位移
    for(let i = 0; i < 4; i++){
      options.w.push(Math.PI /(100 + 20*Math.random()));
      options.theta.push(20*Math.random());
    }
}

//绘制水波线
function drawWaterLines(options) {
   let offset;
   let A = options.a;//正弦曲线振幅
   let y,x,w,theta;
   let r = options.r;
   //遍历每一条水纹理
   for(let line = 0; line < 4; line++){ 
     context.save();
     //每次绘制时水波的偏移距离
     theta = Math.random();
     offset = r + A / 2  -  (r*19/8 + A) * (options.value / 100 ) + line * r/12;
     //获取正弦曲线计算参数
     w = options.w[line];
     theta = options.theta[line];
     context.fillStyle = options.color[line];
     context.moveTo(0,0);
     context.beginPath(); 
     //以0.1为步长绘制正弦曲线
     for(x = 0; x <= 2*r; x+=0.1){
        y = A * Math.sin(w * x + theta) + offset;
        //绘制点
        context.lineTo(x,y);
     }
      //绘制为超出水球范围的封闭图形
      context.lineTo(x,r);
      context.lineTo(x - 2 * r,r);
      context.lineTo(0, A * Math.sin(theta) - options.height);
      context.closePath();
      //填充封闭图形得到一条水波
      context.fill();
      //截取水波范围,绘制文字(此处将在后文解释)
      context.clip();
      context.fillStyle = 'white';
      context.fillText(parseInt(options.value,10) + '%',options.r + 10,10);
      context.restore();
   }
}

//绘制最底层文字
function drawText1(options) {
    context.fillStyle = options.color[0];
    context.fillText(parseInt(options.value,10) + '%',options.r + 10,10);
}

//帧动画循环
function startAnim() {
    //用位移变化模拟水波
    options.theta = options.theta.map(item=>item-0.03);
    //用百分比进度计算水波的高度
    options.value += options.value > 100 ? 0:0.1;
    context.save();
    resetClip(options);//剪切绘图区
    drawText1(options);//绘制蓝色文字
    drawWaterLines(options);//绘制水波线
    context.restore();
    requestAnimationFrame(startAnim);
}

/**设置水球范围为剪裁区域
*(本例中并没有水球以外的部分需要绘制,实际上这里不需要加入帧动画循环中,只需要在开头设置一次即可。)
*/
function resetClip(options) {
   let r = options.r;
   context.strokeStyle = '#2E5199';
   context.fillStyle = 'white';
   context.lineWidth = 10;
   context.beginPath();
   context.arc(r, 0, r + 10, 0, 2*Math.PI, false);
   context.closePath();
   context.fill();
   context.stroke();
   context.beginPath();
   context.arc(r, 0, r, 0, 2*Math.PI, true);
   context.clip();
}

浏览器中可查看效果:

【带着canvas去流浪(7)】绘制水球图

四. 文字淹水效果的实现

文字淹水效果的绘制实际上是按照如下思路来进行的:

context.clip( )

如果我们将每一层文字的绘制颜色修改一下,就比较容易理解绘制过程:

【带着canvas去流浪(7)】绘制水球图

五. 关于canvas抗锯齿

如果仔细查看上面的水球外圆,会发现水球图的外侧不是很平整,看起来会有很多锯齿。网上查到的方法大多是将画布画布尺寸( canvas.height , canvas.width )调整为元素尺寸(CSS中设置的 canvas 元素的尺寸)的3-4倍,希望利用缩放来达到抗锯齿的作用,但实测的结果却并没有明显改进,利用画布尺寸来缩放在 解决图像和填充模糊 的时候效果较好,但在抗锯齿方面的作用似乎与线条本身的尺寸仍有关系,不是一种绝对有效的方案。另一种较为有效的方案,是在绘制外圆时增加 2px-4px 的深色阴影,在视觉上可以很好地弱化锯齿感。

//在绘制外圆之前添加如下代码   
   context.shadowColor = '#2E5199';
   context.shadowBlur = 2;
   context.shadowOffsetX = 0;
   context.shadowOffsetY = 2;

六. 小结

至此,我们在这个系列中完成了所有基本图表的原生API绘制,一些相对高级的图表,其绘制过程并不一定很复杂,比如矩形树图,绘制起来实际上都是矩形方块,但却有助于我们以某种更直观更具有表现力的方式来观察数据,例如可视化呈现 webpack 的打包结果。数据可视化的基本任务就是让数据变得可视,这需要我们为想观察的数据选出恰当的表现方式,这不是纯粹靠技术能够达到的,也需要一些艺术细胞和想象力。但无论如何,这都是一个值得研究的有趣的方向。


以上所述就是小编给大家介绍的《【带着canvas去流浪(7)】绘制水球图》,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对 码农网 的支持!

查看所有标签

猜你喜欢:

本站部分资源来源于网络,本站转载出于传递更多信息之目的,版权归原作者或者来源机构所有,如转载稿涉及版权问题,请联系我们

大数据技术原理与应用

大数据技术原理与应用

林子雨 / 人民邮电出版社 / 2015-8-1 / 45.00

大数据作为继云计算、物联网之后IT行业又一颠覆性的技术,备受关注。大数据处不在,包括金融、汽车、零售、餐饮、电信、能源、政务、医疗、体育、娱乐等在内的社会各行各业,都融入了大数据的印迹,大数据对人类的社会生产和生活必将产生重大而深远的影响。 大数据时代的到来,迫切需要高校及时建立大数据技术课程体系,为社会培养和输送一大批具备大数据专业素养的高级人才,满足社会对大数据人才日益旺盛的需求。本书定......一起来看看 《大数据技术原理与应用》 这本书的介绍吧!

JSON 在线解析
JSON 在线解析

在线 JSON 格式化工具

随机密码生成器
随机密码生成器

多种字符组合密码

Markdown 在线编辑器
Markdown 在线编辑器

Markdown 在线编辑器