html--蛆

时间:2024-10-19 07:55:48
<!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.0 Transitional//EN"> <HTML> <HEAD> <TITLE> New Document </TITLE> <META NAME="Generator" CONTENT="EditPlus"> <META NAME="Author" CONTENT=""> <META NAME="Keywords" CONTENT=""> <META NAME="Description" CONTENT=""> <style> body, html { position: absolute; margin: 0; padding: 0; width: 100%; height: 100%; overflow: hidden; } canvas { position: absolute; width: 100%; height: 100%; background: #000; cursor: pointer; } #node { display: none; } </style> </HEAD> <BODY> <img id="node" src="" alt=""> <script> "use strict"; { const Tentacle = class { constructor() { this.length = 40 + Math.round(80 * Math.random()); this.vel = 2; this.amp = 0.2 + 0.4 * Math.random(); this.freq = 0.05 + 0.3 * Math.random(); this.dir = 2 * Math.PI * Math.random(); this.frame = 0; this.free = true; this.vDir = 0; this.nodes = []; for (let i = 0; i < this.length; i++) { this.nodes.push( new Tentacle.Node( this, i, canvas.width * Math.random(), canvas.height * Math.random() ) ); } } move() { const head = this.nodes[0]; if (head.x > canvas.width) { head.x--; this.dir += 0.1; } else if (head.x < 0) { head.x++; this.dir += 0.1; } if (head.y > canvas.height) { head.y--; this.dir += 0.1; } else if (head.y < 0) { head.y++; this.dir += 0.1; } const dx = pointer.x - head.x; const dy = pointer.y - head.y; const dist = Math.sqrt(dx * dx + dy * dy); if (dist < 300) { if (this.free) { this.dir = Math.atan2(dy, dx); this.vel = 1 + dist * 0.1; if (dist < 1) { this.free = false; } } } else { this.vel = 2; this.free = true; } this.vDir += 0.05 * (Math.random() - Math.random()); this.dir += this.vDir; this.vDir *= 0.9; head.x += this.vel * Math.cos(this.dir); head.y += this.vel * Math.sin(this.dir); this.frame += this.freq; const iDir = this.amp * Math.cos(this.frame); const iHead = this.nodes[1]; iHead.x = head.x - this.vel * Math.cos(this.dir + iDir); iHead.y = head.y - this.vel * Math.sin(this.dir + iDir); for (let i = 2; i < this.length; i++) { this.nodes[i].move(); } } }; Tentacle.Node = class { constructor(tentacle, i, x, y) { this.tentacle = tentacle; const s = tentacle.length - i; this.prev = i > 0 ? tentacle.nodes[i - 1] : null; this.pprev = i > 1 ? tentacle.nodes[i - 2] : null; this.size = 6 + s * s / tentacle.length; this.x = x; this.y = y; this.a = 0; this.img = document.getElementById("node"); } move() { const dx = this.x - this.pprev.x; const dy = this.y - this.pprev.y; this.a = Math.atan2(dy, dx); const d = Math.sqrt(dx * dx + dy * dy); this.x = this.prev.x + dx * 10 / d; this.y = this.prev.y + dy * 10 / d; ctx.save(); ctx.translate(this.x, this.y); ctx.rotate(this.a + 0.4); ctx.drawImage( this.img, -this.size * 0.5, -this.size * 0.5, this.size, this.size ); ctx.restore(); } }; const canvas = { init() { this.elem = document.createElement("canvas"); document.body.appendChild(this.elem); this.resize(); window.addEventListener("resize", () => this.resize(), false); return this.elem.getContext("2d"); }, resize() { this.width = this.elem.width = this.elem.offsetWidth; this.height = this.elem.height = this.elem.offsetHeight; } }; const pointer = { init(canvas) { this.x = -1000; this.y = 0; ["mousemove", "touchstart", "touchmove"].forEach((event, touch) => { document.addEventListener( event, e => { if (touch) { e.preventDefault(); this.x = e.targetTouches[0].clientX; this.y = e.targetTouches[0].clientY; } else { this.x = e.clientX; this.y = e.clientY; } }, false ); }); } }; const ctx = canvas.init(); pointer.init(canvas); const tentacles = []; for (let i = 0; i < 12; i++) tentacles.push(new Tentacle()); const run = () => { requestAnimationFrame(run); ctx.clearRect(0, 0, canvas.width, canvas.height); for (const tentacle of tentacles) { tentacle.move(); } }; run(); } </script> </BODY> </HTML>