基于Canvas怎么绘制和实现连线动画的效果
Admin 2022-07-12 群英技术资讯 1681 次浏览
这篇文章给大家分享的是基于Canvas怎么绘制和实现连线动画的效果。小编觉得挺实用的,因此分享给大家做个参考,文中的介绍得很详细,而要易于理解和学习,有需要的朋友可以参考,接下来就跟随小编一起了解看看吧。前言:canvas动画入门系列之简单连线动画。虽然简单,但连线动画应用场景还挺多,因此做了个小demo,一通百通。
step1:绘制点
首先创建个标签<canvas id="canvas"></canvas>
设置几个点的坐标
const points = [
[200, 100], //上
[300, 200], //右
[100, 200], //左
[200, 100], //上
[200, 300], //下
[100, 200], //左
[300, 200], //右
[200, 300]
];
const canvas = document.querySelector("canvas");
const ctx = canvas.getContext("2d");
然后把点给画出来

points.forEach(([x, y]) => {
drawDot(x, y);
});
function drawDot(x1, y1, r) {
ctx.save();
ctx.beginPath(); //不写会和线连起来
ctx.fillStyle = "red";
//绘制成矩形
ctx.arc(x1, y1, r ? r : 2, 0, 2 * Math.PI);
ctx.fill();
ctx.restore();
}
step2:绘制线条
我们封装一个方法,传入起点终点,绘制一根线条
function drawLine(x1, y1, x2, y2) {
ctx.save();
ctx.beginPath(); //不写每次都会重绘上次的线
ctx.lineCap = "round";
ctx.lineJoin = "round";
var grd = ctx.createLinearGradient(x1, y1, x2, y2);
ctx.moveTo(x1, y1);
ctx.lineTo(x2, y2);
ctx.closePath();
ctx.strokeStyle = "rgba(255,255,255,1)";
ctx.stroke();
ctx.restore();
}
step3:线条动画
这里面需要计算两点之间的斜率,然后x坐标每次挪动±1单位,已知斜率和x偏移,即可计算出y的偏移。值得注意的是,这个坐标系和数学中的xy坐标系有点不一样,y轴是反的。然后可以引入额外的参数speed控制速度
function lineMove(points) {
if (points.length < 2) {
return;
}
const [[x1, y1], [x2, y2]] = points;
let dx = x2 - x1;
let dy = y2 - y1;
if (Math.abs(dx) < 1 && Math.abs(dy) < 1) {
points = points.slice(1);
lineMove(points);
return;
}
let x = x1,
y = y1; //线条绘制过程中的终点
if (dx === 0) {
(x = x2), (y += (speed * dy) / Math.abs(dy));
} else if (dy === 0) {
x += (speed * dx) / Math.abs(dx);
y = y2;
} else if (Math.abs(dx) >= 1) {
let rate = dy / dx;
x += (speed * dx) / Math.abs(dx);
y += (speed * rate * dx) / Math.abs(dx);
}
drawLine(x1, y1, x, y);
points[0] = [x, y];
window.requestAnimationFrame(function() {
lineMove(points);
});
}
主要代码就这么多,先看效果

完整代码
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8" />
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
<meta http-equiv="X-UA-Compatible" content="ie=edge" />
<title>canvas-连线动画</title>
</head>
<body>
<canvas id="canvas" width="400" height="400"></canvas>
<script>
//起点:10,20 终点:150,200
const points = [
[200, 100], //上
[300, 200], //右
[100, 200], //左
[200, 100], //上
[200, 300], //下
[100, 200], //左
[300, 200], //右
[200, 300]
];
const canvas = document.querySelector("canvas");
const ctx = canvas.getContext("2d");
// const img = new Image();
const speed = 10; //速度
// img.onload = function() {
// canvas.width = img.width;
// canvas.height = img.height;
animate(ctx);
// };
// img.src = "./imgs/demo.png";
function animate(ctx) {
// ctx.drawImage(img, 0, 0);
ctx.fillRect(0, 0, canvas.width, canvas.height);
points.forEach(([x, y]) => {
drawDot(x, y);
});
lineMove(points);
}
function lineMove(points) {
if (points.length < 2) {
return;
}
const [[x1, y1], [x2, y2]] = points;
let dx = x2 - x1;
let dy = y2 - y1;
if (Math.abs(dx) < 1 && Math.abs(dy) < 1) {
points = points.slice(1);
lineMove(points);
return;
}
let x = x1,
y = y1; //线条绘制过程中的终点
if (dx === 0) {
(x = x2), (y += (speed * dy) / Math.abs(dy));
} else if (dy === 0) {
x += (speed * dx) / Math.abs(dx);
y = y2;
} else if (Math.abs(dx) >= 1) {
let rate = dy / dx;
x += (speed * dx) / Math.abs(dx);
y += (speed * rate * dx) / Math.abs(dx);
}
drawLine(x1, y1, x, y);
points[0] = [x, y];
window.requestAnimationFrame(function () {
lineMove(points);
});
}
function drawLine(x1, y1, x2, y2) {
ctx.save();
ctx.beginPath(); //不写每次都会重绘上次的线
ctx.lineCap = "round";
ctx.lineJoin = "round";
var grd = ctx.createLinearGradient(x1, y1, x2, y2);
ctx.moveTo(x1, y1);
ctx.lineTo(x2, y2);
ctx.closePath();
ctx.strokeStyle = "rgba(255,255,255,1)";
ctx.stroke();
ctx.restore();
}
function drawDot(x1, y1, r) {
ctx.save();
ctx.beginPath(); //不写会和线连起来
ctx.fillStyle = "red";
//绘制成矩形
ctx.arc(x1, y1, r ? r : 2, 0, 2 * Math.PI);
ctx.fill();
ctx.restore();
}
</script>
</body>
</html>
免责声明:本站发布的内容(图片、视频和文字)以原创、转载和分享为主,文章观点不代表本网站立场,如果涉及侵权请联系站长邮箱:mmqy2019@163.com进行举报,并提供相关证据,查实之后,将立刻删除涉嫌侵权内容。
猜你喜欢
导入外部css样式表的方法使用link标签导入外部css样式表<linkrel="stylesheet"href="css/demo01.css">在样式表中import(导入)外部样式表@importurl("/crazy-html5/06css/css/demo01.css");使用内部样式表内部样式表只能作用于某一个网页,定义方式为在head头部添加style标签,在style标签中即可添加页面样式。<head><style ...
数学表达式calc()是CSS中的函数,主要用于数学运算。这篇文章给大家介绍了CSS中的数学表达式calc()的相关知识,感兴趣的朋友一起看看吧
这篇文章主要介绍了CSS3实现缺角矩形,折角矩形以及缺角边框,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
在css中,可以利用“overflow-x”属性来禁止页面的左右滚动条,当该属性的值为“hidden”时,可以将超出左右边缘的元素内容隐藏起来,进而禁止页面左右滚动条的显示;语法为“body{overflow-x:hidden;}”。
这篇文章主要介绍了Canvas图片分割效果的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
成为群英会员,开启智能安全云计算之旅
立即注册关注或联系群英网络
7x24小时售前:400-678-4567
7x24小时售后:0668-2555666
24小时QQ客服
群英微信公众号
CNNIC域名投诉举报处理平台
服务电话:010-58813000
服务邮箱:service@cnnic.cn
投诉与建议:0668-2555555
Copyright © QY Network Company Ltd. All Rights Reserved. 2003-2020 群英 版权所有
增值电信经营许可证 : B1.B2-20140078 ICP核准(ICP备案)粤ICP备09006778号 域名注册商资质 粤 D3.1-20240008