javascript - 来自base64编码的纯Javascript粒子排斥器png

标签 javascript debugging animation optimization particle-system

我有一些从 youtube 教程复制的 js 代码,并适用于我自己的项目来填充标题,代码按预期工作,并且当视口(viewport)小于 1200px 左右时它工作,但是当我将 firefox 放入全屏动画时不播放并且图像被拉伸(stretch),不保持其纵横比。我确实有一个 10/15 岁的 gpu,所以我猜那是我的问题的一半。该脚本使用 100x100 像素的 png 图像文件,然后将其转换为粒子颜色值。可以对此进行优化或使其运行得更好。似乎视口(viewport)越宽,动画启动所需的时间就越长,直到它最终停止并且不起作用。全屏= [2550x1440] ...
原始教程在这里:Pure Javascript Particle Animations & 将图像转换为 base64 编码在这里:Image to base64 .
HTML:

<html>
  <body>
    <canvas id="CanV"></canvas>
  </body>
</html>
CSS:
#Canv{
  position:absolute;
  top:-1px;left:-2px;
  z-index:67;
  width:100vw !important;
  max-height: 264px !important;
  min-height: 245px !important;
  filter:blur(2.27px);
}
Javascript:
window.addEventListener("DOMContentLoaded",(e)=>{
  const canv = document.getElementById('Canv');
  const ctx = canv.getContext('2d');
  canv.width = window.innerWidth;
  canv.height = window.innerHeight/ 3.85;

  let particleArray = [];
  let mouse = {
    x: null,
    y: null,
    radius: 74
  }

   window.addEventListener('mousemove',(e)=>{
     mouse.x = event.x + canv.clientLeft/2;
     mouse.y = event.y + canv.clientTop/1.2;
   });

  function drawImage(){
    let imageWidth = png.width;      //These to values crop if / sum no.
    let imageHeight = png.height;
    const data = ctx.getImageData(0, 0, imageWidth, imageHeight); //Gets img data for particles
    ctx.clearRect(0,0, canv.width, canv.height); // Clears the original img as its now being stored in the variable data.

    class Particle {
      constructor(x, y, color, size){
        this.x = x + canv.width/2 - png.width * 174,     //Chngd Ok:74
        this.y = y + canv.height/2 - png.height * 32,     //Ch<2  Ok:16
        this.color = color,
        this.size = 2.28, // Particle Size > Changed this value. from 2 i think!.
        this.baseX = x + canv.width/1.8 - png.width * 3.1,  //Chngd ok:5.1
        this.baseY = y + canv.height/1.2 - png.height * 2.8,
        this.density = (Math.random() * 14) + 2;
      }
      draw() {
         ctx.beginPath();  // this creates the sort of force field around the mouse pointer.
         ctx.arc(this.x, this.y, this.size, 0, Math.PI * 2);
         ctx.closePath();
         ctx.fill();
      }
      update() {
        ctx.fillStyle = this.color;

        // Collision detection
        let dx = mouse.x - this.x;
        let dy = mouse.y - this.y;
        let distance = Math.sqrt(dx * dx + dy * dy);
        let forceDirectionX = dx / distance;
        let forceDirectionY = dy / distance;
        // Max distance, past that the force will be 0
        const maxDistance = 144;
        let force = (maxDistance - distance) / maxDistance;
        if (force < 0) force = 0;
        let directionX = (forceDirectionX * force * this.density * 0.6);
        let directionY = (forceDirectionY * force * this.density * 8.7); //Ch.this

        if (distance < mouse.radius + this.size) {
          this.x -= directionX;
          this.y -= directionY;
        } else {
          if (this.x !== this.baseX){
            let dx = this.x - this.baseX;
            this.x -= dx/54;  // Speed Particles return to ori
        } if (this.y !== this.baseY){
          let dy = this.y - this.baseY;
          this.y -= dy/17;  // Speed Particles return to ori
        }
       }
       this.draw();
      }
    }
    function init(){
      particleArray = [];
      for(let y = 0, y2 = data.height; y<y2; y++){
        for(let x =0, x2 = data.width; x<x2; x++){
          if(data.data[(y * 4 * data.width) + (x*4) + 3] > 128){
            let positionX = x + 25;
            let positionY = y + 45; // Co-ords on Canv
            let color = "rgb(" + data.data[(y * 4 * data.width) + (x * 4)] + "," +
                         data.data[(y * 4 * data.width) + (x * 4) + 1] + "," +
                         data.data[(y * 4 * data.width) + (x * 4) + 2] + ")";
            particleArray.push(new Particle(positionX * 2, positionY * 2, color));
          } /* These number effect png size but its to high */
        }
      }
    }
    function animate(){
      requestAnimationFrame(animate);
      ctx.fillStyle = 'rgba(0,0,0,.07)';
      ctx.fillRect(0,0, innerWidth, innerHeight);

      for(let i =0; i < particleArray.length; i++){
        particleArray[i].update();
      }
    }
    init();
    animate();
  }

  const png = new Image();

  png.src = "RemovedBase64StringToBig";

  window.addEventListener('load',(e)=>{
    console.log('page has loaded');
    ctx.drawImage(png, 0, 0);
    drawImage();
  })

});
通过缩短所有变量名称 > PartArr、ImgWidth、DirX、DirY 等,已经设法将其缩短了大约 100 个字符,但除了缩小它还有其他方法可以优化吗?并修复全屏问题?
我试图将它添加到 JSfiddle,所以我可以在这里链接到它,但我认为这不允许 base64 字符串,它无论如何都不会加载任何东西。 Canvas 加载,背景只是没有图像或动画。
我已经发现全屏问题的哪一部分,光标位置实际上在实际光标右侧大约 300px 处,但我仍然不知道如何解决这个问题或解决主要的滞后性能问题。猜测即使只有 100x100 也需要大量计算。
我能想到的让这个性能更好的一个选择是将它及其计算移动到它自己的专用网络 worker 中并将图像转换为 Webp,但我仍然不太了解网络 worker 或如何正确实现它们.. 将使用评论和答案中的所有建议,看看我能把什么放在一起。
我添加这些链接仅供将来引用,稍后我会回到这里:
MDN Canvas Optimizations
Html5Rocks Canvas Performance
Stack Question. Canv ~ Opti
Creating A blob From A Base 64 String in Js
次要奖金问题,
是否有最大文件大小或最大 px 尺寸,
可以base64编码吗?只是在 facebook 上问这个问题,最近有人向我发送了一个关于另一个具有多个 base64 编码图像的项目的问题,我不确定答案..

最佳答案

缩短代码对性能没有多大帮助。我正在使用火狐。要查看在 Firefox 中运行浏览器期间最占用您时间的内容,您可以阅读 Performance from MDN .

您的解决方案的问题是您的 fps 正在急剧下降。发生这种情况是因为您是 绘画每个 粒子 每一帧。想象一下,当每帧需要绘制数千个粒子时,它会有多迟缓。此绘制调用是从您的函数 Particle.draw 中调用的。 (调用如下:ctx.beginPath, ctx.arc, and ctx.closePath)。如前所述,此函数将被调用,因为 Particle.update对于 每个 框架。这是一项极其昂贵的操作。要大幅提高 fps,您可以尝试 不是 单独绘制每个粒子,而是收集所有粒子的 ImageData 然后在 rAQ 中仅将其放置在 Canvas 上一次(因此只发生一种油漆)。这个ImageData是一个包含 rgba 的对象对于 Canvas 上的每个像素。

在下面的解决方案中,我执行了以下操作:

  • 对于每个脏的粒子(已更新),修改 ImageData将被放入 Canvas
  • 然后,在整个 ImageData 之后已构建一帧,只画一次使用 putImageData 到 Canvas 上.这节省了大量调用函数所需的时间 Particle.update单独绘制每个粒子。

  • 另一个明显的解决方案是增加粒子的大小,以便需要处理的粒子像素更少(更改 ImageData)。我还稍微调整了代码,使图像始终至少为 100px 高;您可以调整数学,以便图像始终保持您的纵横比并响应窗口大小。

    这是一个工作示例:

    const canvas = document.querySelector('#canvas1')
    const ctx = canvas.getContext('2d')
    canvas.width = window.innerWidth
    canvas.height = window.innerHeight
    let canvasWidth = canvas.width
    let canvasHeight = canvas.height
    let particleArray = []
    let imageData = []
    
    // mouse
    let mouse = {
      x: null,
      y: null,
      radius: 40
    }
    
    window.addEventListener('mousemove', e => {
      mouse.x = event.x
      mouse.y = event.y
    })
    
    function drawImage(width, height) {
      let imageWidth = width
      let imageHeight = height
      const data = ctx.getImageData(0, 0, imageWidth, imageHeight)
      
      class Particle {
        constructor(x, y, color, size = 2) {
          this.x = Math.round(x + canvas.width / 2 - imageWidth * 2)
          this.y = Math.round(y + canvas.height / 2 - imageHeight * 2)
          this.color = color
          this.size = size
          
          // Records base and previous positions to repaint the canvas to its original background color
          this.baseX = Math.round(x + canvas.width / 2 - imageWidth * 2)
          this.baseY = Math.round(y + canvas.height / 2 - imageHeight * 2)
          this.previousX = null
          this.previousY = null
          this.density = (Math.random() * 100) + 2
        }
        
        stringifyColor() {
          return `rgba(${this.color.r}, ${this.color.g}, ${this.color.b}, ${this.color.a}`
        }
        
        update() {
          ctx.fillStyle = this.stringifyColor()
          
          // collision detection
          let dx = mouse.x - this.x
          let dy = mouse.y - this.y
          let distance = Math.sqrt(dx * dx + dy * dy)
          let forceDirectionX = dx / distance
          let forceDirectionY = dy / distance
          
          // max distance, past that the force will be 0
          const maxDistance = 100
          let force = (maxDistance - distance) / maxDistance
          if (force < 0) force = 0
          
          let directionX = (forceDirectionX * force * this.density)
          let directionY = (forceDirectionY * force * this.density)
          
          this.previousX = this.x
          this.previousY = this.y
          if (distance < mouse.radius + this.size) {
            this.x -= directionX
            this.y -= directionY
          } else {
          	// Rounded to one decimal number to as x and y cannot be the same (whole decimal-less integer) 
          	// as baseX and baseY by decreasing using a random number / 20
            if (Math.round(this.x) !== this.baseX) {
              let dx = this.x - this.baseX
              this.x -= dx / 20
            }
            if (Math.round(this.y) !== this.baseY) {
              let dy = this.y - this.baseY
              this.y -= dy / 20
            }
          }
        }
      }
      
      function createParticle(x, y, size) {
        if (data.data[(y * 4 * data.width) + (x * 4) + 3] > 128) {
          let positionX = x
          let positionY = y
          let offset = (y * 4 * data.width) + (x * 4)
          let color = {
            r: data.data[offset],
            g: data.data[offset + 1],
            b: data.data[offset + 2],
            a: data.data[offset + 3]
          }
          
          return new Particle(positionX * 4, positionY * 4, color, size)
        }
      }
      
      // Instead of drawing each Particle one by one, construct an ImageData that can be
      // painted into the canvas at once using putImageData()
      function updateImageDataWith(particle) {
        let x = particle.x
        let y = particle.y
        let prevX = particle.previousX
        let prevY = particle.previousY
        let size = particle.size
    
        if (prevX || prevY) {
          let prevMinY = Math.round(prevY - size)
          let prevMaxY = Math.round(prevY + size)
          let prevMinX = Math.round(prevX - size)
          let prevMaxX = Math.round(prevX + size)
    
          for (let y = prevMinY; y < prevMaxY; y++){
            for (let x = prevMinX; x < prevMaxX; x++) {
              if (y < 0 || y > canvasHeight) continue
              else if (x < 0 || x > canvasWidth) continue
              else {
                let offset = y * 4 * canvasWidth + x * 4
                imageData.data[offset] = 255
                imageData.data[offset + 1] = 255
                imageData.data[offset + 2] = 255
                imageData.data[offset + 3] = 255
              }
            }
          }
        }
    
        let minY = Math.round(y - size) 
        let maxY = Math.round(y + size) 
        let minX = Math.round(x - size) 
        let maxX = Math.round(x + size) 
    
        for (let y = minY; y < maxY; y++){
          for (let x = minX; x < maxX; x++) {
            if (y < 0 || y > canvasHeight) continue
            else if (x < 0 || x > canvasWidth) continue
            else {
              let offset = y * 4 * canvasWidth + x * 4
              imageData.data[offset] = particle.color.r
              imageData.data[offset + 1] = particle.color.g
              imageData.data[offset + 2] = particle.color.b
              imageData.data[offset + 3] = particle.color.a
            }
          }
        }
      }
      
      function init() {
        particleArray = []
        imageData = ctx.createImageData(canvasWidth, canvasHeight)
        // Initializing imageData to a blank white "page"
        for (let data = 1; data <= canvasWidth * canvasHeight * 4; data++) {
          imageData.data[data - 1] = data % 4 === 0 ? 255 : 255
        }
    
        const size = 2 // Min size is 2
        const step = Math.floor(size / 2)
        for (let y = 0, y2 = data.height; y < y2; y += step) {
          for (let x = 0, x2 = data.width; x < x2; x += step) {
            // If particle's alpha value is too low, don't record it
            if (data.data[(y * 4 * data.width) + (x * 4) + 3] > 128) {
              let newParticle = createParticle(x, y, size)
              particleArray.push(newParticle)
              updateImageDataWith(newParticle)
            }
          }
        }
      }
      
      function animate() {
        requestAnimationFrame(animate)
        
        for (let i = 0; i < particleArray.length; i++) {
          let imageDataCanUpdateKey = `${Math.round(particleArray[i].x)}${Math.round(particleArray[i].y)}`
          particleArray[i].update()
    
          updateImageDataWith(particleArray[i])
        }
        ctx.putImageData(imageData, 0, 0)
      }
      
      init()
      animate()
      
      window.addEventListener('resize', e => {
        canvas.width = innerWidth
        canvas.height = innerHeight
        canvasWidth = canvas.width
        canvasHeight = canvas.height
        init()
      })
    }
    
    const png = new Image()
    png.src = " "
    
    window.addEventListener('load', e => {
      // Ensuring height of image is always 100px
      let pngWidth = png.width
      let pngHeight = png.height
      
      let divisor = pngHeight / 100
      let finalWidth = pngWidth / divisor
      let finalHeight = pngHeight / divisor
      
      ctx.drawImage(png, 0, 0, finalWidth, finalHeight)
      drawImage(finalWidth, finalHeight)
    })
    #canvas1 {
      position: absolute;
      top: 0;
      left: 0;
      width: 100%;
      height: 100%;
    }
    <canvas id="canvas1"></canvas>

    关于javascript - 来自base64编码的纯Javascript粒子排斥器png,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/59092312/

    相关文章:

    javascript - Rangey:获取选定节点的新方法

    ruby - 在 REPL 中调试本地 gem

    ios - UIView transitionFromView 不在容器中设置动画

    javascript - Canvas 水平旋转图像

    javascript - 属性(property)范围问题?

    javascript - 在javascript中打印对象数组并在html标签中打印

    ios - 从另一个类加载函数在 Swift IOS 中崩溃

    python - matplotlib 保存每 n 个步骤的动画

    javascript - 更改日期格式以识别日期和月份?

    ios - iOS游戏开始时的输入延迟