Toggle navigation
在线编辑器
在线代码
文本比较
jQuery下载
前端库
在线手册
登录/注册
下载代码
html
css
js
分享到微信朋友圈
X
html
css
@import url("https://fonts.googleapis.com/css2?family=Open+Sans:wght@400;700&display=swap"); * { box-sizing: border-box; } body { min-height: 100vh; background: black; display: grid; place-items: center; margin: 0; padding: 0; font-family: "Open Sans", sans-serif; color: #ccc; } a { text-decoration: none; color: inherit; } a:active, a:focus { outline: none; } #me { position: absolute; bottom: 2rem; left: 50%; transform: translateX(-50%); } #me a { font-weight: bold; } #heart { color: #ff6961; }
JavaScript
let mouse, antImage, c = 0, WIDTH, HEIGHT; const n = 40, drag = 0.97, minDist = 200, ants = [], canvas = document.getElementById("canvas"), ctx = canvas.getContext("2d"); const requestAnimationFrame = window.requestAnimationFrame || window.mozRequestAnimationFrame || window.webkitRequestAnimationFrame || window.msRequestAnimationFrame; function init() { WIDTH = window.innerWidth; HEIGHT = window.innerHeight; canvas.setAttribute("width", WIDTH); canvas.setAttribute("height", HEIGHT); mouse = new Vector(WIDTH / 2, HEIGHT / 2); canvas.addEventListener("mousemove", (e) => { if (c % 10 === 0) { mouse.set(e.clientX, e.clientY); } c++; }); for (let i = 0; i < n; i++) { const pos = new Vector(random(WIDTH), random(HEIGHT)); const vel = Vector.random().setMag(random(3, 4)); ants.push(new Ant(pos, vel, Math.round(random(30, 50)))); } ctx.clearRect(0, 0, WIDTH, HEIGHT); ctx.fillStyle = "#FFDEAD"; // ctx.imageSmoothingEnabled = false; ctx.beginPath(); ctx.fillRect(0, 0, WIDTH, HEIGHT); ctx.closePath(); ani(); } function ani() { ctx.fillRect(0, 0, WIDTH, HEIGHT); for (const ant of ants) { if (Vector.dist(ant.pos, mouse) <= minDist) { ant.applyForce(Vector.sub(ant.pos, mouse).setMag(0.5)); } ant.update(); ant.borders(); ant.draw(); } requestAnimationFrame(ani); } antImage = new Image(); antImage.src = ""; class Ant { constructor(pos, vel, size) { this.pos = pos; this.vel = vel; this.maxVel = vel.mag(); this.acc = new Vector(0, 0); this.size = size; this.img = antImage; } update() { this.pos.add(this.vel); this.vel.add(this.acc); const m = this.vel.mag(); if (m > this.maxVel) { this.vel.setMag(m * drag); } this.acc.set(0, 0); } applyForce(force) { this.acc = force; } borders() { if ((this.pos.x > WIDTH) | (this.pos.x < 0)) { this.vel.x = -this.vel.x; } else if ((this.pos.y > HEIGHT) | (this.pos.y < 0)) { this.vel.y = -this.vel.y; } } draw() { ctx.save(); const { width } = antImage; ctx.translate(this.pos.x, this.pos.y); const ang = this.vel.heading(); ctx.rotate(ang + Math.PI / 2); ctx.translate(-width / 2, -width / 2); ctx.drawImage(antImage, 0, 0, this.size, this.size); ctx.restore(); } } class Vector { constructor(x = 0, y = 0) { this.x = x; this.y = y; this.m = this.mag(); this.dir = this.heading(); } copy() { return new Vector(this.x, this.y); } set(x, y) { this.x = x; this.y = y; this.update(); return this; } static add(v0, v1) { const r = new Vector(v0.x + v1.x, v0.y + v1.y); return r; } add(s0, s1 = null) { if (s0 instanceof Vector) { this.x += s0.x; this.y += s0.y; this.update(); return this; } this.x += s0; this.y += s1; this.update(); return this; } static sub(v0, v1) { const r = new Vector(v0.x - v1.x, v0.y - v1.y); return r; } sub(s0, s1) { if (s0 instanceof Vector) { this.x -= s0.x; this.y -= s0.y; this.update(); return this; } this.x -= s0; this.y -= s1; this.update(); return this; } mult(e) { if (e instanceof Vector) { this.x *= e.x; this.y *= e.y; } else { this.x *= e; this.y *= e; } this.update(); return this; } div(e) { if (e instanceof Vector) { this.x /= e.x; this.y /= e.y; } else { this.x = this.x / e; this.y = this.y / e; } this.update(); return this; } static dist(v1, v2) { const { sqrt, pow } = Math; const d = sqrt(pow(v1.x - v2.x, 2) + pow(v1.y - v2.y, 2)); return d; } mag() { const { sqrt, pow } = Math; const i = sqrt(pow(this.x, 2) + pow(this.y, 2)); this.m = i; return this.m; } setMag(m) { this.m = m; this.x = Math.cos(this.dir) * m; this.y = Math.sin(this.dir) * m; return this; } heading() { this.dir = Math.atan2(this.y, this.x); return this.dir; } static random() { // Random unit (m=1) vector const ang = random(-Math.PI, Math.PI); const x = Math.cos(ang); const y = Math.sin(ang); return new Vector(x, y); } static fromAngle(a) { const x = Math.cos(a); const y = Math.sin(a); return new Vector(x, y); } limit(max) { if (this.m > max) { this.setMag(max); } } equals(v) { if (!v instanceof Vector) { return false; } return ( this.x == v.x && this.y == v.y && this.mag() == v.mag() && this.heading() == v.heading() ); } update() { this.mag(); this.heading(); } } function random(min = 0, max = 1) { if (arguments.length === 1) { if (min != 0) { return Math.random() * min; } else { return Math.random(); } } return Math.random() * (max - min) + min; } init();
粒子
时间
文字
hover
canvas
3d
游戏
音乐
火焰
水波
轮播图
鼠标跟随
动画
css
加载动画
导航
菜单
按钮
滑块
tab
弹出层
统计图
svg
×
Close
在线代码下载提示
开通在线代码永久免费下载,需支付20jQ币
开通后,在线代码模块中所有代码可终身免费下!
您已开通在线代码永久免费下载,关闭提示框后,点下载代码可直接下载!
您已经开通过在线代码永久免费下载
对不起,您的jQ币不足!可通过发布资源 或
直接充值获取jQ币
取消
开通下载
<!doctype html> <html> <head> <meta charset="utf-8"> <title>canvas蚂蚁动画-jq22.com</title> <script src="https://www.jq22.com/jquery/jquery-1.10.2.js"></script> <style>
</style> </head> <body>
<script>
</script>
</body> </html>
2012-2021 jQuery插件库版权所有
jquery插件
|
jq22工具库
|
网页技术
|
广告合作
|
在线反馈
|
版权声明
沪ICP备13043785号-1
浙公网安备 33041102000314号