canvas实现帘子吊灯鼠标交互物理碰撞效果代码
代码语言:html
所属分类:其他
代码描述:canvas实现帘子吊灯鼠标交互物理碰撞效果代码
代码标签: canvas 帘子 吊灯 鼠标 交互 物理 碰撞
下面为部分代码预览,完整代码请点击下载或在bfwstudio webide中打开
<!DOCTYPE html> <html lang="en" > <head> <meta charset="UTF-8"> <style> html, body, canvas { -webkit-user-select: none; -moz-user-select: none; -ms-user-select: none; user-select: none; } html, body { width: 100%; height: 100%; overflow: hidden; margin: 0; display: flex; align-items: center; justify-content: center; background: #191919; } .asset-img { display: none; } </style> </head> <body > <canvas></canvas> <!-- glowing image base64 --> <img class="asset-img" id="light-img" src=""> <script src="https://cpwebassets.codepen.io/assets/common/stopExecutionOnTimeout-2c7831bb44f98c1391d6a4ffda0e1fd302503391ca806e7fcc7b9b87197aec26.js"></script> <script > class Mouse { constructor(canvas) { this.pos = new Vector(-1000, -1000) this.radius = 40 canvas.onmousemove = e => this.pos.setXY(e.clientX, e.clientY) canvas.ontouchmove = e => this.pos.setXY(e.touches[0].clientX, e.touches[0].clientY) canvas.ontouchcancel = () => this.pos.setXY(-1000, -1000) canvas.ontouchend = () => this.pos.setXY(-1000, -1000) } } class Dot { constructor(x, y) { this.id = Math.random() this.pos = new Vector(x, y) this.oldPos = new Vector(x, y) this.radius = 1 this.friction = 0.97 this.gravity = new Vector(0, 0.6) this.mass = 1 this.pinned = false this.distMouse = 1000 this.lightImg = document.querySelector('#light-img') this.lightWidth = 15 this.lightHeight = 15 } update(mouse) { if (this.pinned) return let vel = Vector.sub(this.pos, this.oldPos) vel.mult(this.friction) this.oldPos.setXY(this.pos.x, this.pos.y) vel.add(this.gravity) this.pos.add(vel) let { x: dx, y: dy } = Vector.sub(this.pos, mouse.pos) this.distMouse = Math.sqrt(dx * dx + dy * dy) if (this.distMouse > mouse.radius + this.radius) return const direction = new Vector(dx / this.distMouse, dy / this.distMouse) let force = (mouse.radius - this.distMouse) / mouse.radius if (force < 0) force = 0 if (force < 0.6) { this.pos.sub(direction.mult(force).mult(0.0001)) } else { this.pos.setXY(mouse.pos.x, mouse.pos.y) } } drawLight(ctx) { ctx.drawImage( this.lightImg, this.pos.x - this.lightWidth / 2, this.pos.y - this.lightHeight / 2, this.lightWidth, this.lightHeight ) } draw(ctx) { ctx.fillStyle = '#aaa' ctx.fillRect(this.pos.x - this.radius, this.pos.y - this.radius, this.radius * 2, this.radius * 2) } } class Stick { constructor(p1, p2) { this.startPoint = p1 this.endPoint = p2 this.tension = 1 this.color = '#999' this.length = this.startPoint.pos.dist(this.endPoint.pos) } update() { const dx = this.endPoint.pos.x - this.startPoint.pos.x const dy = this.endPoint.pos.y - this.startPoint.pos.y const dist = Math.sqrt(dx*dx + dy*dy) const diff = (this.length - dist) / dist * this.tension const offsetX = dx * diff * 0.5 const offsetY = dy * diff * 0.5 const m = this.endPoint.mass + this.startPoint.mass const m1 = this.endPoint.mass / m const m2 = this.startPoint.mass / m if (!this.startPoint.pinned) { this.startPoint.pos.x -= offsetX * m1 this.startPoint.pos.y -= offsetY * m1 } if (!this.endPoint.pinned) { this.endPoint.pos.x += offsetX * m2 this.endPoint.pos.y += offsetY * m2 } } draw(ctx) { ctx.beginPath() ctx.strokeStyle = this.color ctx.moveTo(this.startPoint.pos.x, this.startPoint.pos.y) ctx.lineTo(this.endPoint.pos.x, this.endPoint.pos.y) ctx.stroke() ctx.closePath() } } clas.........完整代码请登录后点击上方下载按钮下载查看
网友评论0