pixi实现十几种粒子喷射模拟重力弹射动画效果代码

代码语言:html

所属分类:粒子

代码描述:pixi实现十几种粒子喷射模拟重力弹射动画效果代码,点击更换预设就可切换到其他的动画效果。

代码标签: pixi 粒子 喷射 重力 动画

下面为部分代码预览,完整代码请点击下载或在bfwstudio webide中打开

<!DOCTYPE html>
<html lang="en">

<head>

    <meta charset="UTF-8">





    <style>

        * {
          -webkit-touch-callout: none; /* iOS Safari */
          -webkit-user-select: none; /* Safari */
          -khtml-user-select: none; /* Konqueror HTML */
          -moz-user-select: none; /* Old versions of Firefox */
          -ms-user-select: none; /* Internet Explorer/Edge */
          user-select: none; /* Non-prefixed version, currently
          supported by Chrome, Edge, Opera and Firefox */
        }
        body, html {
          font-family: Roboto, sans-serif;
          padding: 0;
          margin: 0;
          height: 100vh;
          line-height: 21px;
          font-size: 16px;
          overflow: hidden;
        }
        @media only screen and (max-width : 1024px) {
          body, html {
            font-size: 14px;
            line-height: 19px;
          }
          .small {
            font-size: 13px;
          }
          .intro select {
            font-size: 14px;
          }
        }
        input {
          font-family: Fira Sans;
        }
        .small {
          font-size: 14px;
        }
        .red {
          color: #880000;
        }
        .cascade-label {
          position: absolute;
          border: 2px solid black;
          border-radius: 20px;
          width: 30px;
          height: 30px;
          display: flex;
          justify-content: center;
          align-items: center;
          background: white;
          color: black;
          font-size: 20px;
          z-index: 50;
          visibility: hidden;
          /*display: none;*/
        }
        #help-container {
          position: fixed;
          display: none;
          padding: 5vh;
          background: white;
          border: 3px solid #888;
          height: 100vh;      
          box-sizing: border-box;
          overflow: auto;
          z-index: 500;
        }
        #intro-container {
          display: grid;
          grid-template-rows: 8fr auto 2fr;
          grid-template-columns: minmax(2vw, 1fr) 20fr minmax(2vw, 1fr);
          position: fixed;
          width: 100vw;
          height: 100vh;
        }
        @media only screen and (max-width : 1024px) {
          #intro-container {
            grid-template-columns: 1rem auto 1rem;
            grid-template-rows: 8fr auto 1rem;
          }
        }
        .intro {
          grid-row: 2;
          grid-column: 2;
          border: 3px solid #AAA;
          color: #666;
          background: white;
          width: 100%;
          box-sizing: border-box;
          padding: 2rem 3rem;
        }
        @media only screen and (max-width : 1024px) {
          .intro {
            padding: 1rem;
          }
        }
        .intro button {
          margin: 1rem 0;
          background-color: #800;
          color: white;
          font-size: 1rem;
          height: 2rem;
          line-height: 2rem;
          padding: 0 1rem;
          border-radius: 20px;
          border: 0;
          cursor: pointer;
        }
        #help-button {
          text-shadow: 2px 2px 8px #888888;
          box-shadow: 2px 2px #888888;
          cursor: pointer;
          color: white;
          display: flex;
          justify-content: center; 
          align-items: center;
          position: absolute;
          bottom: 2vh;
          left: 2vh;
          width: 30px;
          height: 30px;
          border-radius: 30px;
          border: 2px solid white;
        }
        .intro select {
          font-size: 18px;
        }
    </style>


</head>


<body>
    <div data-cascade-label='1' class="cascade-label">1</div>
    <div data-cascade-label='2' class="cascade-label">2</div>
    <div data-cascade-label='3' class="cascade-label">3</div>
    <div data-cascade-label='4' class="cascade-label">4</div>
    <div id="pixi"></div>

    <canvas id='container' style='position:absolute;' width='1024' height='768'>
  </canvas>
    <input style="display: none;" id="input" type='text' value='Cascades' class='input' />

    <script type="text/javascript" src="//repo.bfw.wiki/bfwrepo/js/pixi.5.1.5.js"></script>
    <script type="text/javascript" src="//repo.bfw.wiki/bfwrepo/js/dat.gui-min.js"></script>
    <script type="text/javascript" src="//repo.bfw.wiki/bfwrepo/js/zepto.1.2.0.js"></script>

    <script>
        (function ($) {
          $.extend($.fn,
          {
            fadeIn: function (ms)
            {
              if (typeof ms === 'undefined') ms = 500;
        
              $(this).css(
              {
                display: 'block',
                opacity: 0 }).
              animate({
                opacity: 1 },
              ms);
        
              return this;
            },
        
            fadeOut: function (ms)
            {
              if (typeof ms === 'undefined') ms = 500;
        
              $(this).css(
              {
                opacity: 1 }).
              animate({
                opacity: 0 },
              ms, 'linear', function ()
              {
                $(this).css('display', 'none');
              });
        
              return this;
            } });
        
        })(Zepto);
        
        const creator = new URLSearchParams(window.location.search).get('creator');
        const viewer = new URLSearchParams(window.location.search).get('viewer');
        
        const debug = new URLSearchParams(window.location.search).get('debug');
        
        
        const circle64 = "";
        const square64 = "";
        const star64 = "";
        const flower64 = "";
        const ring64 = "";
        
        class Collider {
        
          constructor() {
        
            // control some internal vars here
            this.particleCount = 1000;
            this.hitBoxSize = 2;
        
            this.running = false;
        
            this.textCanvas = document.getElementById("container");
            this.textContext = this.textCanvas.getContext('2d');
        
            //this.cWidth = this.textCanvas.width;
            //this.cHeight = this.textCanvas.height;
        
            this.loadedFontFaces = [];
            this.fontFace = null;
        
            this.initParams();
        
            this.resize();
        
            this.hitTest = [];
            this.boxSize = this.hitBoxSize;
            this.av = 0;
        
            this.pause = 0;
        
            this.emitter1 = new Emitter(this);
            this.emitter1.i = 1;
            this.emitter1.edge = 'top';
            this.emitter1.velocity = 15;
            this.emitter1.distance = 0.3;
        
            this.emitter2 = new Emitter(this);
            this.emitter2.i = 2;
            this.emitter2.edge = 'top';
            this.emitter2.velocity = 15;
            this.emitter2.distance = 0.4;
        
            this.emitter3 = new Emitter(this);
            this.emitter3.i = 3;
            this.emitter3.edge = 'top';
            this.emitter3.velocity = 15;
            this.emitter3.distance = 0.6;
        
            this.emitter4 = new Emitter(this);
            this.emitter4.i = 4;
            this.emitter4.edge = 'top';
            this.emitter4.velocity = 15;
            this.emitter4.distance = 0.7;
        
            this.emitters = [
            this.emitter1,
            this.emitter2,
            this.emitter3,
            this.emitter4];
        
        
        
            this.init();
        
        
        
          }
        
        
          /**
           * Do a full reset of all particles
           */
          hardReset() {
            if (this.running) {
              this.particles.destroy();
              this.initParticles();
            }
        
          }
        
        
          /**
           * Initialise all the internal parameters
           * @return {[type]} [description]
           */
          initParams() {
            // todo - make textSize proportional to screen size?? No? Yes? ARGH
            this.textSize = 150;
            this.textPosition = 0.5;
            this.textBorderWidth = 0;
            this.textBorderColor = '#2222AA';
            this.font = 'Monoton';
            this.particleSize = 1;
            this.particleShape = 'circle';
            //this.leftColor = '#880000';
            //this.rightColor = '#334455';
        
            this.pal1 = '#ffc610';
            this.pal2 = '#ff5600';
            this.pal3 = '#3333EE';
            this.pal4 = '#EE3333';
            this.cyclePalette = false;
        
        
        
            this.textColor = '#2222AA';
            //this.leftPosition = 50;
            //this.rightPosition = 50;
            //this.leftVelocity = 10;
            //this.rightVelocity = 10;
            this.gravity = 1;
            this.wind = -0.25;
            //this.turbulence = 1;
            this.airDrag = 0.1;
        
        
            this.textStickiness = 25;
        
            this.shadeParticles = true;
            // should particles be mixed up in the emitters?
            this.mixPalette = false;
        
          }
        
          /**
           * Tells all sprites to regenerate on falling off the screen
           */
          regenerateSprites() {
            this.sprites.forEach(sprite => {
              sprite.regenerate = true;
            });
          }
        
        
          reset() {
            this.initParams();
            this.rebuild();
          }
        
        
          shortHalt() {
            this.pause = 1;
          }
        
          resize() {
        
            this.cWidth = window.innerWidth;
            this.cHeight = window.innerHeight;
        
            if (debug) {
              this.cWidth = 1024;
              this.cHeight = 800;
            }
        
            this.textCanvas.width = this.cWidth;
            this.textCanvas.height = this.cHeight;
        
        
            if (this.particleRenderer) {
              this.particleRenderer.resize(this.cWidth, this.cHeight);
            }
        
            this.rebuild();
        
            this.repositionLabels();
        
          }
        
          /**
           * Returns the paletter color associated with the index i
           * *really* messy. Urgh. Stop it
           * @param  {[type]} i [description]
           * @return {[type]}   [description]
           */
          getPaletteColor(i) {
        
            /*if(this.cyclePalette) {
              i += this.tOffset;
              i = i % 4;
            }*/
        
            if (i == 1) {
              return this.pal1;
            } else if (i == 2) {
              return this.pal2;
            } else if (i == 3) {
              return this.pal3;
            } else if (i == 4) {
              return this.pal4;
            }
          }
        
          buildCollisionMatrix() {
            // build collision matrix
            let iData = this.textContext.getImageData(0, 0, this.cWidth, this.cHeight);
            let data = iData.data;
            for (var i = 0; i < this.cWidth / this.boxSize; i++) {
              this.hitTest[i] = [];
              for (var j = 0; j < this.cHeight / this.boxSize; j++) {
                //if above or below a specific row, safely assume there is no text here
                if (j > (this.cHeight / 2 + this.textSize * 1.5) / this.boxSize || j < (this.cHeight / 2 - this.textSize * 1.5) / this.boxSize) {
                  this.hitTest[i][j] = 0;
                  continue;
                }
                var pixel = 0;
                // get boxSize/2 pixels to the left, right, above and below this point
                for (var x = i * this.boxSize - this.boxSize; x < i * this.boxSize + this.boxSize; x++) {
                  for (var y = j * this.boxSize - this.boxSize; y < j * this.boxSize + this.boxSize; y++) {
                    if (x < 0 || y < 0 || x > this.cWidth || y > this.cHeight) {
                      continue;
                    }
                    pixel += data[x * 4 + y * 4 * this.cWidth] + data[x * 4 + y * 4 * this.cWidth + 1] + data[x * 4 + y * 4 * this.cWidth + 2] + data[x * 4 + y * 4 * this.cWidth + 3];
                  }
                }
                this.hitTest[i][j] = pixel;
              }
            }
          }
        
          initParticles() {
            // When using the highly optimised ParticleContainer, cannot apply tint operations
            this.particles = new PIXI.ParticleContainer(100000, {
              tint: true,
              position: true,
              autoResize: true });
        
            //this.particles = new PIXI.DisplayObjectContainer();
        
            // create a texture from an image path
            this.stage.addChild(this.particles);
            this.sprites = [];
        
            for (var i = 0; i < this.particleCount; i++) {
              let sprite = this.createSprite(i);
              this.sprites.push(sprite);
              this.particles.addChild(sprite);
            }
          }
        
          init() {
        
            document.getElementById('input').value = "bfw.wiki";
        
            this.particleRenderer = PIXI.autoDetectRenderer({
              width: this.cWidth,
              height: this.cHeight,
              transparent: true });
        
            //this.particleRenderer.backgroundColor = 0x888888;
            this.particleRenderer.view.style.position = "absolute";
            this.particleRenderer.view.style.top = "0px";
            this.particleRenderer.view.style.left = "0px";
        
            document.getElementById('pixi').appendChild(this.particleRenderer.view);
            this.stage = new PIXI.Container();
        
            this.setWeightedEmitters();
        
            this.initParticles();
            this.rebuild();
            this.repositionLabels();
        
            requestAnimationFrame(this.animate.bind(this));
        
          }
        
          // thanks http://www.javascriptkit.com/javatutors/weighrandom2.shtml  
          setWeightedEmitters() {
        
            this.emitterTotalFlow = this.emitters.reduce(function (prev, cur) {
              return prev + cur.flow;
            }, 0);
        
            this.weightedEmitters = [];
            const emitterWeights = [];
        
            this.emitters.forEach(emitter => {
              emitterWeights.push(emitter.flow);
            });
        
            this.emitters.forEach(emitter => {
              // todo - oh please zero index emitter.i
              // Please.
              for (let i = 0; i < emitterWeights[emitter.i - 1]; i++) {
                this.weightedEmitters[this.weightedEmitters.length] = emitter.i - 1;
              }
            });
        
        
        
          }
        
        
          /**
           * //Add to Collider.Utils?
           * Apply luminance to a sprite. Used to get the subtle colour variation effect when choosing an emitter colour
           */
        
          lum(hex, lum) {
        
            // validate hex string
            hex = String(hex).replace(/[^0-9a-f]/gi, '');
            if (hex.length < 6) {
              hex = hex[0] + hex[0] + hex[1] + hex[1] + hex[2] + hex[2];
            }
            lum = lum || 0;
        
            // convert to decimal and change luminosity
            var rgb = "#",
            c,i;
            for (i = 0; i < 3; i++) {
              c = parseInt(hex.substr(i * 2, 2), 16);
              c = Math.round(Math.min(Math.max(0, c + c * lum), 255)).toString(16);
              rgb += ("00" + c).substr(c.length);
            }
        
            return rgb;
          }
        
          /**
           * [initSprite description]
           * @param  {[type]} sprite [description]
           * @return {[type]}        [description]
           */
          initSprite(sprite) {
        
            sprite.anchor.set(0.5);
            sprite.mass = 0.2 + Math.random() * 0.8;
        
            // Which emitter is this sprite going to come from?
        
            if (this.emitterTotalFlow == 0) {
              // can't init sprite while all emitters are turned off!
              return;
            }
        
            let emitter;
            if (!sprite.emitter || this.mixPalette) {
              let i = Math.floor(Math.random() * this.emitterTotalFlow);
              emitter = this.emitters[this.weightedEmitters[i]];
            } else {
              emitter = sprite.originalEmitter; // use the same emitter
            }
        
            emitter.emit(sprite);
        
            sprite.velocity.y = Math.random() * 4 + 1;
        
            sprite.inactive = 0;
            sprite.dropOut = false;
        
            if (sprite.regenerate) {
        
              const targetColor = this.getPaletteColor(sprite.emitter.i);
              let col;
        
              if (this.shadeParticles) {
                col = this.lum(targetColor, -0.33 + Math.random() * 0.33).replace('#', '0x');
              } else {
                col = targetColor.replace('#', '0x');
              }
        
              sprite.tint = col;
        
              let pngSizeFactor = 0.33;
        
              sprite.scale.set((sprite.scaleAdjustment * this.particleSize * sprite.mass + Math.random() * 0.5) * pngSizeFactor); // cater for later base64 data PNGs
        
              sprite.bornAt = performance.now();
              sprite.stillFor = 0;
              sprite.lastPos = {
                x: 0,
                y: 0 };
        
              sprite.regenerate = false;
        
            }
          }
        
        
          /**
           * [i description]
           * @type {[type]}
           */
          createSprite(i = false)
          {
            let sprite;
        
            if (this.particleShape == 'square') {
              sprite = PIXI.Sprite.from(square64);
            } else if (this.particleShape == 'star') {
              sprite = PIXI.Sprite.from(star64);
            } else if (this.particleShape == 'flower') {
              sprite = PIXI.Sprite.from(flower64);
            } else if (this.particleShape == 'ring') {
              sprite = PIXI.Sprite.from(ring64);
            } else {
              sprite = PIXI.Sprite.from(circle64);
            }
        
            sprite.regenerate = true;
            sprite.specular = Math.random();
            sprite.velocity = {
              x: 0,
              y: 0 };
        
            sprite.i = i;
            sprite.delay = 5000 * Math.random(); //feed sprites slowly
            return sprite;
          }
        
        
        
        
          //batchAdd();
        
          animate(t) {
        
            this.running = true;
        
            this.tOffset = Math.floor(t / 1000 * 0.25 % 4) + 1;
        
            if (this.pause > 0) {
              this.pause--;
              requestAnimationFrame(this.animate.bind(this));
              return;
            }
        
            this.particleRenderer.render(this.stage);
        
        
        
            for (var i in this.sprites) {
              var sprite = this.sprites[i];
        
              if (sprite.inactive || sprite.bornAt + sprite.delay > t) {
                sprite.alpha = 0;
                continue;
              } else {
                sprite.alpha = 1;
              }
        
        
              sprite.velocity.x *= 1 - this.airDrag * 0.25;
        
        
              let dx = sprite.velocity.x * sprite.mass;
              let dy = sprite.velocity.y * sprite.mass;
        
              sprite.position.x += dx;
              sprite.position.y += dy;
        
              // knock the sprites from the top of the letters once they've settled
              if (sprite.stillFor > this.textStickiness * (Math.random(0.4) + 0.8)) {
                sprite.dropOut = true;
              }
        
              let reverseFactor = 1;
              if (sprite.emitter) {
                reverseFactor = sprite.emitter.reverseGravity ? -1 : 1;
              }
              // 0.25 is a fudge factor to slow gravity down a bit. seems too strong
              sprite.velocity.y += this.gravity * reverseFactor * 0.25;
        
              //sprite.velocity.y += 0.3 * this.gravity;
        
              if (!sprite.stillFor) {
                sprite.position.x += this.wind;
              }
        
              var _x = Math.floor(sprite.position.x / this.boxSize);
              var _y = Math.floor(sprite.position.y / this.boxSize);
        
              if (_x >= 0 && _y >= 0 && _x < this.cWidth / this.boxSize && _y < this.cHeight / this.boxSize) {
        
                let hit;
        
                // Fixes an error sometimes where this isn't defined. Randomness
                if (this.hitTest[_x]) {
                  hit = this.hitTest[_x][_y];
                } else {
                  hit = 0;
                }
        
                if (hit > 0 && !sprite.dropOut) {
                  sprite.velocity.y = 0 - sprite.velocity.y * (0.1 + 0.3 * Math.random());
                  if (sprite.velocity.y < 0.2 && sprite.velocity.y > -0.2) {
                    sprite.velocity.y = 0; //stabilise
                  }
                  sprite.stillFor++; // assuming that the sprite lies still after settling on a surface
                  //sprite.velocity.x *= -1 + 2 * Math.random();
                  sprite.velocity.x *= -1 + 0.25 * Math.random();
                } else {
                  sprite.stillFor = 0;
                }
              } else {
                if (sprite.regenerate) {
                  // Kill the sprite. Create a new one in its place. *no one will ever know*
                  this.particles.removeChild(sprite);
                  let newSprite = this.createSprite(sprite.i);
                  this.initSprite(newSprite);
                  this.particles.addChild(newSprite);
                  this.sprites[i] = newSprite;
                } else {
                  // recycle the sprite if it falls off the edge
                  this.initSprite(sprite);
                }
        
              }
        
            }
            requestAnimationFrame(this.animate.bind(this));
          }
        
          //requestAnimationFrame(animate);
        
          /**
           * Rebuild the text and collision matrix
           * @return {[type]} [description]
           */
          rebuild() {
        
            if (!this.loadedFontFaces.includes(this.font)) {
              let fontFace = new FontFace(this.font, "url('https://assets.codepen.io/52084/" + this.font + ".ttf')");
              fontFace.load().then(_ => {
                document.fonts.add(fontFace);
                this.loadedFontFaces.push(fontFace);
                this.fontFace = fontFace;
                this.renderText();
              });
            } else {
              console.log("unused?");
              this.fontFace = fontFace;
              this.renderText();
            }
        
        
          }
        
        
          renderText()
          {
            const input = document.getElementById('input');
            let text = input.value;
            // bit of a ham-fisted way of adjusting size to always fit on screen
            let adjustmentFactor = this.cWidth / 1280;
            if (adjustmentFactor > 1) {
              adjustmentFactor = 1;
            }
            const adjustedTextSize = this.textSize * adjustmentFactor;
            this.textContext.font = adjustedTextSize + "px " + '"' + this.fontFace.family + '"';
            this.textContext.clearRect(0, 0, this.cWidth, this.cHeight);
            let textWidth = this.textContext.measureText(text).width;
        
            this.textContext.fillStyle = this.textColor;
            if (this.textBorderWidth > 0) {
              this.textContext.strokeStyle = this.textBorderColor;
              this.textContext.lineWidth = this.textBorderWidth;
              this.textContext.strokeText(text, (this.cWidth - textWidth) / 2, this.textPosition * this.cHeight);
            }
            this.textContext.fillText(text, (this.cWidth - textWidth) / 2, this.textPosition * this.cHeight);
            this.buildCollisionMatrix();
          }
        
        
          /**
           * Reposition the number labels for the cascade helper
           * @return {[type]} [description]
           */
          repositionLabels() {
        
            if (!this.emitters) {
              return;
            }
        
        
        
            this.emitters.forEach(emitter => {
        
              let origin = emitter.origin();
              let label = document.querySelectorAll('[data-cascade-label="' + emitter.i + '"]');
              let x = origin.x;
              let y = origin.y;
        
              if (emitter.edge == 'top') {
                x += 20;
                y += 40;
              }
        
              if (emitter.edge == 'right') {
                x -= 40;
                y += 20;
              }
        
              if (emitter.edge == 'bottom') {
                y -= 40;
                x -= 20;
              }
        
              if (emitter.edge == 'left') {
                x += 40;
                y += 20;
              }
        
              if (label) {
                console.log(x);
                label[0].style.left = x + "px";
                label[0].style.top = y + "px";
              }
            });
          }}
        
        
        
        
        
        /**
         * To do - get all the particle junk into this
         */
        class Particle {
        
          constructor(collider) {
            // ??
          }
        
          step() {
        
          }
        
          animate() {
        
          }}
        
        
        
        
        /**
         * Emitter class - sets up particles with initial position/velocity
         */
        
        class Emitter {
        
          constructor(collider) {
            this.collider = collider;
            this.distance = 0.5; // distance along the chosen edge
            this.edge = 'top'; // top, right, bottom, left
            this.velocity = 5; // speed at which particles are emitted
            this.reverseGravity = false;
            this.scaleAdjustment = 1;
            this.spread = 1; // 0 - 2
            this.flow = 100; // How many particles come out of this one? (0-1)
          }
        
          /**
           * Set the initial origin point
           * Pretty basic screen-space stuff, no translations happening here
           * @return {[type]} [description]
           */
          origin() {
            let x, y;
            if (this.edge == 'top') {
    .........完整代码请登录后点击上方下载按钮下载查看

网友评论0