运动的小球

canvas绘制运动小球

知乎的网页版登录界面的背景有很多运动的小球,小球和小球运动的时候之间还有连线,给人一种三维立体变换的效果,看着十分的不错,所以用canvas试着做了个和知乎登录界面背景类似的效果,上面动图就是做好效果的截图。

实现思路

首先了解下canvas中的动画原理?canvas中的动画其实是通过不断的重绘来实现动起来的效果的,打个比方一个小球初始的时候在画布的X,Y坐标记作ball(x,y),然后每隔10毫秒更改小球的X,Y坐标为ball(x+5,y+5)(在当前X,Y坐标加5个像素) 并且清除整个画布,重新在画布上绘制更改坐标后小球,由于10毫秒非常的短,所有在视觉上给我们的感觉就是小球在不断运动着。canvas绘图的原理基本就是这样子。

  1. 定义小球对象
var ball = {
      xPointer: 100, //小球初始x坐标
      yPointer: 100, //小球初始y坐标
      vx: 1, //x方向的速度
      vy: 0.1, //y方向的速度
      x: 1, //x轴运动方向(1表示正方向,-1表示反方向)
      y: -1, //y轴运动方向
      color: "blue", //小球颜色
      radius: 10, //小球半径
};
  1. 生成小球
    demo中的小球有很多个,所以定义一个数组来装这些小球,小球的起始坐标、颜色、运动方向都不同所以这些值需要随机获取。
   var ballList = []; //小球数组
   var canvas, ctx;
   //生成多个小球
   function initBall() {
       canvas = document.getElementById("canvas");
       ctx = canvas.getContext("2d");
       //循环生成60个小球
       for (var i = 0; i < 60; i++) {
           // console.log(getIndex() + "   " + getIndex())
           var ball = {};
           ball.xPointer = getRandom(20, 980); //随机小球的X坐标
           ball.yPointer = getRandom(20, 340); //随机小球的y坐标
           ball.x = getIndex(); //随机小球x轴运动方向
           ball.y = getIndex(); //随机小球的y轴运动方向
           ball.vx = Math.random(); //随机小球x轴方向速度
           ball.vy = Math.random(); //随机小球y轴方向速度
           ball.radius = 9; //小球半径
           ball.color = "#" + ("00000" + ((Math.random() * 16777215 + 0.5) >> 0).toString(16)).slice(-6); //随机小球颜色
           ballList.push(ball);
       }
   }

   //随机一个1或者-1的方法   
   function getIndex() {
       var arr = [0, 1];
       var index = Math.floor((Math.random() * arr.length));
       if (index == 0) {
           index = -1;
       }
       return index;
   }

   //获取两数之间的一个随机数的方法
   function getRandom(first, last) {
       var choice = last - first + 1;
       return Math.floor(Math.random() * choice + first);
   }
  1. 通过canvas绘制小球
    页面canvas标签
<canvas id="canvas" width="1000" height="360" style='background-color: #EEEEEE;'></canvas>

canvas绘制小球代码

function draw(ctx) {
    ctx.clearRect(0, 0, 1000, 360); //绘制前先清除画布
    for (var i = 0; i < ballList.length; i++) {
        ctx.save();
        ctx.beginPath();
        ctx.fillStyle = ballList[i].color;
        ctx.arc(ballList[i].xPointer, ballList[i].yPointer, ballList[i].radius, 0, Math.PI * 2, false);
        ctx.closePath();
        ctx.fill();
        ctx.restore();
    }
}
  1. 运动起来
    小球运动的过程中使用了简单的碰撞检测,每次到达画布的边缘就改变小球的运动方向
//修改小球的状态,使小球动起来的方法
function update(ballList, ctx) {
    for (var i = 0; i < ballList.length; i++) {
        ballList[i].xPointer += ballList[i].vx * ballList[i].x;
        ballList[i].yPointer += ballList[i].vy * ballList[i].y;
        //碰撞检测 X轴方向
        if (ballList[i].xPointer + ballList[i].radius >= canvas.width || ballList[i].xPointer - ballList[i].radius <= 0) {
            ballList[i].x = ballList[i].x * -1;
        }
        //碰撞检测 Y轴方向
        if (ballList[i].yPointer + ballList[i].radius >= canvas.height || ballList[i].yPointer - ballList[i].radius <= 0) {
            ballList[i].y = ballList[i].y * -1;
        }
    }
}
  1. 绘制小球和小球之间的连线
//小球之间连线
function drawLine(ballList, ctx) {
    for (var i = 0; i < ballList.length; i++) {
        for (var j = 0; j < ballList.length; j++) {
            var xx = Math.pow((ballList[i].xPointer - ballList[j].xPointer), 2);
            var yy = Math.pow((ballList[i].yPointer - ballList[j].yPointer), 2);
            var zz = Math.sqrt(xx + yy);
            //判断两个小球如果之间距离在20到100之间,就绘制一条直线
            if (zz <= 100 && zz >= 20) {
                console.log(zz)
                ctx.save();
                ctx.beginPath();
                ctx.strokeStyle = "#999999";
                ctx.lineWidth = 0.1;
                // ctx.strokeStyle= "#" + ("00000" + ((Math.random() * 16777215 + 0.5) >> 0).toString(16)).slice(-6);
                ctx.moveTo(ballList[i].xPointer, ballList[i].yPointer);
                ctx.lineTo(ballList[j].xPointer, ballList[j].yPointer);
                ctx.closePath();
                ctx.stroke();
                ctx.restore();
            }
        }
    }
}
  1. 运行
(function() {
    initBall(); //生成小球
    //计时器
    setInterval(function() {
        // console.log(selectfrom(0, 600) + "         " + selectfrom(0, 600));
        draw(ctx); //绘制
        update(ballList, ctx); //修改小球状态
        drawLine(ballList, ctx); //画线
    }, 24)
})();

其它

修改小球半径后的小球运动效果

** 到此一个canvas小球运动效果的demo做完了,看着是不是有一种3d变换效果。**

文章如有误,请不吝赐教~

最后编辑于
?著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,128评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,316评论 3 388
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事?!?“怎么了?”我有些...
    开封第一讲书人阅读 159,737评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,283评论 1 287
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,384评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,458评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,467评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,251评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,688评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,980评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,155评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,818评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,492评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,142评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,382评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,020评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,044评论 2 352

推荐阅读更多精彩内容