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