CodePen

HTML

            
              <!-- Physics shaders -->

<script id="physics-vert" type="x-vertex-shader">

    attribute vec2 aVertexPosition;

    void main() {
        gl_Position = vec4( aVertexPosition, 1, 1 );
    }

</script>

<script id="physics-frag" type="x-fragment-shader">
    
    precision mediump float;

    const vec3 TARGET = vec3( 0, 0, 0.01 );

    uniform sampler2D uParticleData;
    uniform vec2 uViewport;

    // Retrieves the texel at a given offset from the current pixel
    vec4 texelAtOffet( vec2 offset ) {
        return texture2D( uParticleData, ( gl_FragCoord.xy + offset ) / uViewport );
    }

    void main() {

        // Determine which data slot we're at (position or velocity)
        int slot = int( mod( gl_FragCoord.x, 2.0 ) );

        if ( slot == 0 ) { // position

            // Retrieve data at current and adjacent slots
            vec4 dataA = texelAtOffet( vec2( 0, 0 ) );
            vec4 dataB = texelAtOffet( vec2( 1, 0 ) );

            // Extract position and velocity data
            vec3 pos = dataA.xyz;
            vec3 vel = dataB.xyz;

            float phase = dataA.w;

            if ( phase > 0.0 ) {

                // Integrate velocity
                pos += vel * 0.005;

                // Kill the particle if it's near the target
                if ( length( TARGET - pos ) < 0.035 ) phase = 0.0;
                else phase += 0.1;

            } else {

                pos = vec3(-1);
            }

            // Write out the new position data
            gl_FragColor = vec4( pos, phase );

        } else if ( slot == 1 ) { // velocity

            // Retrieve data at current and previous slots
            vec4 dataA = texelAtOffet( vec2( -1, 0 ) );
            vec4 dataB = texelAtOffet( vec2( 0, 0 ) );

            // Extract position and velocity data
            vec3 pos = dataA.xyz;
            vec3 vel = dataB.xyz;

            float phase = dataA.w;

            if ( phase > 0.0 ) {

                // Compute a normalised vector pointing towards target
                vec3 delta = normalize( TARGET - pos );

                // Add a force along the delta vector
                vel += delta * 0.05;

                // Add a drag force
                vel *= 0.991;

            } else {

                vel = vec3(0);
            }

            // Write out the velocity data
            gl_FragColor = vec4( vel, 1.0 );
        }
    }

</script>

<!-- Render shaders -->

<script id="render-vert" type="x-vertex-shader">

    attribute vec2 aParticleUV;

    uniform sampler2D uParticleData;

    void main() {

        vec4 particle = texture2D( uParticleData, aParticleUV );
        float perspective = 1.0 + particle.z * 5.5;
        float phase = cos( particle.w ) * max( 0.5, tan( particle.z * 8.05 ) );

        gl_Position = vec4( particle.xyz, perspective );
        gl_PointSize = min( 64.0, ( 1.0 / perspective ) * ( 0.5 + phase ) );
    }

</script>

<script id="render-frag" type="x-fragment-shader">

    uniform sampler2D uParticleTexture;

    void main() {
        gl_FragColor = texture2D( uParticleTexture, gl_PointCoord );
    }

</script>

<!-- Debug shaders -->

<script id="debug-vert" type="x-vertex-shader">

    attribute vec2 aVertexPosition;
    varying vec2 vTexCoord;

    void main() {

        // Map from xy (-1 -> 1) to uv (0 -> 1)
        vTexCoord = ( aVertexPosition + 1.0 ) / 2.0;
        gl_Position = vec4( aVertexPosition, 1, 1 );
    }

</script>

<script id="debug-frag" type="x-fragment-shader">

    precision mediump float;

    uniform sampler2D uParticleData;
    varying vec2 vTexCoord;

    void main() {
        gl_FragColor = vec4( texture2D( uParticleData, vTexCoord ).xyz, 0.8 );
    }

</script>

<div class='info'>
    <hgroup class='about'>
        <h1>1,048,576 Particles</h1>
        <h2>GPU Physics with WebGL</h2>
    </hgroup>
    <nav class='more'>
        <a target="_blank" href='http://soulwire.co.uk/'>soulwire</a>
        <a target="_blank" href='https://github.com/soulwire/WebGL-GPU-Particles'>View on GitHub</a>
    </nav>
</div>
<div class="cb toggle-fbo">
    <input class="toggle" type="checkbox"/>
    <span class="button">Show FBO</span>
</div>

<div id="nogl">
    <h1>Aww, No WebGL :(</h1>
    <p>This experiment requires WebGL to run. Please enable it, or come back and visit with a <a href="http://caniuse.com/webgl" target="_blank">compatible browser</a>.</p>
</div>

<img id="particle-texture" src="">

            
          
!

CSS

            
              @import url( http://fonts.googleapis.com/css?family=Quantico );

html, body {
    font-family: 'Quantico', sans-serif;
    background: #1c3149;
    background: -moz-radial-gradient(center, ellipse cover,  #1c3149 0%, #0d121c 80%);
    background: -webkit-gradient(radial, center center, 0px, center center, 100%, color-stop(0%,#1c3149), color-stop(80%,#0d121c));
    background: -webkit-radial-gradient(center, ellipse cover,  #1c3149 0%,#0d121c 80%);
    background: -o-radial-gradient(center, ellipse cover,  #1c3149 0%,#0d121c 80%);background: -ms-radial-gradient(center, ellipse cover,  #1c3149 0%,#0d121c 80%);
    background: radial-gradient(ellipse at center,  #1c3149 0%,#0d121c 80%);
    overflow: hidden;
    margin:0;
}

#particle-texture {
    display: none;
}

/* GUI */

.cb {

    position: absolute;
    cursor: pointer;
    right: 20px;
    top: 20px;
}

.cb .button {

    -webkit-transition: all 200ms ease-out;
    -moz-transition: all 200ms ease-out;
    -o-transition: all 200ms ease-out;
    transition: all 200ms ease-out;
    text-transform: uppercase;
    font-family: Arial, "Helvetica Neue", Helvetica, sans-serif;
    border-radius: 2px;
    text-align: center;
    line-height: 25px;
    font-size: 11px;
    display: block;
    opacity: 0.8;
    height: 25px;
    width: 80px;
    color: #fff;
}

.cb .button:hover {
    opacity: 1.0;
}

.cb .toggle {

    position: absolute;
    opacity: 0.0;
    z-index: 1;
    display: block;
    cursor: pointer;
    height: 100%;
    width: 100%;
}

.cb .toggle ~ .button {
    background: #000;
}

.cb .toggle:checked ~ .button {
    background: #2ecc71;
}

/* No GL */

#nogl {
    
    -webkit-box-sizing: border-box;
    -moz-box-sizing: border-box;
    -ms-box-sizing: border-box;
    -o-box-sizing: border-box;
    box-sizing: border-box;

    border-radius: 3px;
    line-height: 1.45;
    text-align: justify;
    box-shadow: 0px 0px 0px 4px rgba(255,0,41,0.2);
    background: #FF0029;
    font-size: 13px;
    position: absolute;
    padding: 10px 20px;
    display: none;
    margin: -70px 0 0 -130px;
    height: 140px;
    width: 260px;
    color: #fff;
    left: 50%;
    top: 50%;
}

#nogl h1 {

    font-weight: 300;
    font-size: 18px;
    margin: 12px 0;
}

#nogl p {

    color: rgba(255,255,255,0.7);
}

#nogl a {

    color: #fff;
}

/* Info */

@-webkit-keyframes show-info {
    0% { -webkit-transform: rotateY(120deg); }
    100% { -webkit-transform: rotateY(0deg); }
}

@-moz-keyframes show-info {
    0% { -moz-transform: rotateY(120deg); }
    100% { -moz-transform: rotateY(0deg); }
}

@-ms-keyframes show-info {
    0% { -ms-transform: rotateY(120deg); }
    100% { -ms-transform: rotateY(0deg); }
}

@-o-keyframes show-info {
    0% { -o-transform: rotateY(120deg); }
    100% { -o-transform: rotateY(0deg); }
}

@keyframes show-info {
    0% { transform: rotateY(120deg); }
    100% { transform: rotateY(0deg); }
}

.info {

    -webkit-transition: all 180ms ease-out;
    -moz-transition: all 180ms ease-out;
    -ms-transition: all 180ms ease-out;
    -o-transition: all 180ms ease-out;
    transition: all 180ms ease-out;

    -webkit-transform-style: preserve-3d;
    -moz-transform-style: preserve-3d;
    -ms-transform-style: preserve-3d;
    -o-transform-style: preserve-3d;
    transform-style: preserve-3d;

    -webkit-transform: perspective(800);
    -moz-transform: perspective(800);
    -ms-transform: perspective(800);
    -o-transform: perspective(800);
    transform: perspective(800);

    position: absolute;
    font-size: 12px;
    opacity: 0.8;
    color: #fff;
    width: 240px;
    left: 0px;
    top: 20px;
}

.info:hover {

    box-shadow: 0 0 0 4px rgba(255,255,255,0.05);
    opacity: 1.0;
}

.info h1,
.info h2,
.info h3 {

    line-height: 1;
    margin: 5px 0;
}

.info a {

    -webkit-transition: all 200ms ease-out;
    -moz-transition: all 200ms ease-out;
    -ms-transition: all 200ms ease-out;
    -o-transition: all 200ms ease-out;
    transition: all 200ms ease-out;

    border-bottom: 1px dotted rgba(255,255,255,0.4);
    text-decoration: none;
    opacity: 0.6;
    color: #fff;
}

.info a:hover {

    opacity: 0.99;
}

.info .about, .info .more {

    -webkit-transform-origin: 0% 50%;
    -moz-transform-origin: 0% 50%;
    -ms-transform-origin: 0% 50%;
    -o-transform-origin: 0% 50%;
    transform-origin: 0% 50%;

    -webkit-transform: rotateY(120deg);
    -moz-transform: rotateY(120deg);
    -ms-transform: rotateY(120deg);
    -o-transform: rotateY(120deg);
    transform: rotateY(120deg);

    margin-bottom: 1px;
    background: rgba(0,0,0,0.8);
    padding: 12px 15px 12px 20px;
}

.info .about {

    -webkit-animation: show-info 500ms cubic-bezier(0.230, 1.000, 0.320, 1.000) 600ms 1 normal forwards;
    -moz-animation: show-info 500ms cubic-bezier(0.230, 1.000, 0.320, 1.000) 600ms 1 normal forwards;
    -ms-animation: show-info 500ms cubic-bezier(0.230, 1.000, 0.320, 1.000) 600ms 1 normal forwards;
    -o-animation: show-info 500ms cubic-bezier(0.230, 1.000, 0.320, 1.000) 600ms 1 normal forwards;
    animation: show-info 500ms cubic-bezier(0.230, 1.000, 0.320, 1.000) 600ms 1 normal forwards;

    padding-bottom: 15px;
}

.info .about a {
    
    opacity: 0.9;
}

.info .about h1 {

    letter-spacing: -1px;
    font-weight: 300;
    font-size: 19px;
    opacity: 0.95;
}

.info .about h2 {

    font-weight: 300;
    font-size: 13px;
    opacity: 0.8;
}

.info .about h3 {

    text-transform: uppercase;
    margin-top: 10px;
    font-size: 11px;
}

.info .about h3:before {

    margin-right: 2px;
    font-size: 14px;
    content: '\203A';
}

.info .more {

    -webkit-animation: show-info 500ms cubic-bezier(0.230, 1.000, 0.320, 1.000) 500ms 1 normal forwards;
    -moz-animation: show-info 500ms cubic-bezier(0.230, 1.000, 0.320, 1.000) 500ms 1 normal forwards;
    -ms-animation: show-info 500ms cubic-bezier(0.230, 1.000, 0.320, 1.000) 500ms 1 normal forwards;
    -o-animation: show-info 500ms cubic-bezier(0.230, 1.000, 0.320, 1.000) 500ms 1 normal forwards;
    animation: show-info 500ms cubic-bezier(0.230, 1.000, 0.320, 1.000) 500ms 1 normal forwards;

    padding: 5px 15px 10px 20px;
}

.info .more a {
    
    text-transform: uppercase;
    margin-right: 10px;
    font-size: 10px;
}
            
          
!
? ?
? ?
Must be a valid URL.
+ add another resource
via CSS Lint

JS

            
              
// MIT License
// @see https://github.com/soulwire/WebGL-GPU-Particles

/*
------------------------------------------------------------

Config

------------------------------------------------------------
*/

// Number of particles
var PARTICLE_COUNT = Math.pow( 1024, 2 );

// Particle count must be power of 2
var PARTICLE_COUNT_SQRT = Math.sqrt( PARTICLE_COUNT );

// How many texture slots each particle needs for it's data
var PARTICLE_DATA_SLOTS = 2;

// The required size of the FBO containing the particle data
var PARTICLE_TEXTURE_WIDTH = PARTICLE_COUNT_SQRT * PARTICLE_DATA_SLOTS;
var PARTICLE_TEXTURE_HEIGHT = PARTICLE_COUNT_SQRT;

// How many particles can be emitted at a given time
var PARTICLE_EMIT_RATE = 1000;

/*
------------------------------------------------------------

Variables

------------------------------------------------------------
*/

// Stats instance
var stats;

// Programs
var physicsProgram;
var renderProgram;
var debugProgram;

// Buffers and textures
var particleUVDataBuffer;
var particleDataTexture;
var particleFramebuffer;
var viewportQuadBuffer;
var particleData;

// Shaders
var shaders = {
    physics_vert: getShader( 'physics-vert' ),
    physics_frag: getShader( 'physics-frag' ),
    render_vert: getShader( 'render-vert' ),
    render_frag: getShader( 'render-frag' ),
    debug_vert: getShader( 'debug-vert' ),
    debug_frag: getShader( 'debug-frag' )
};

var debugToggle = document.querySelector( '.toggle-fbo .toggle' );

// Sketch instance (augmented WebGL context)
var gl = Sketch.create({
    type: Sketch.WEBGL,
    autostart: false
});

if ( !gl ) {
    nogl = document.getElementById( 'nogl' );
    nogl.style.display = 'block';
}

/*
------------------------------------------------------------

Helper methods

------------------------------------------------------------
*/

// Retrieves shader source from a script element in the DOM
function getShader( id ) {

    return document.getElementById( id ).textContent;
}

// Creates and compiles a shader from source
function createShader( source, type ) {

    var shader = gl.createShader( type );

    gl.shaderSource( shader, source );
    gl.compileShader( shader );

    if ( !gl.getShaderParameter( shader, gl.COMPILE_STATUS ) )
        throw gl.getShaderInfoLog( shader );

    return shader;
}

// Creates a shader program and creates / links shaders
function createProgram( vertexSource, fragmentSource ) {

    var vs = createShader( vertexSource, gl.VERTEX_SHADER );
    var fs = createShader( fragmentSource, gl.FRAGMENT_SHADER );

    var program = gl.createProgram();

    gl.attachShader( program, vs );
    gl.attachShader( program, fs );
    gl.linkProgram( program );

    if ( !gl.getProgramParameter( program, gl.LINK_STATUS ) )
        throw gl.getProgramInfoLog( program );

    return program;
}

// Sets `count` particles to be alive, positions them at the emission `origin` and gives them
// an initial random velocity
function emitParticles( count, origin, velocities ) {

    velocities = velocities || { x:0.0, y:0.0, z:0.0 };

    // Bind the particle data texture since we'll be updating it
    gl.activeTexture( gl.TEXTURE0 + particleDataTexture.unit );
    gl.bindTexture( gl.TEXTURE_2D, particleDataTexture );

    // Determine the x and y coordinates in the particle data texture to start writing to
    var x = ~~( ( gl.particleIndex * PARTICLE_DATA_SLOTS ) % PARTICLE_TEXTURE_WIDTH );
    var y = ~~( gl.particleIndex / PARTICLE_TEXTURE_HEIGHT );

    // An array defining the chunks of texture to be updated
    var chunks = [{
        x: x,
        y: y,
        size: count * PARTICLE_DATA_SLOTS
    }];

    // Recursively splits chunks until they will all fit into the particle data texture
    function split( chunk ) {

        var boundary = chunk.x + chunk.size;

        // If the chunk will exceed the texture size, it must be split
        if ( boundary > PARTICLE_TEXTURE_WIDTH ) {

            // How many slots fall outside the texture dimensions
            var delta = boundary - PARTICLE_TEXTURE_WIDTH;

            // Reduce the current chunk size to fit
            chunk.size -= delta;

            // Create a new chunk from the orphaned data
            chunk = {
                x: 0,
                y: ( chunk.y + 1 ) % PARTICLE_TEXTURE_HEIGHT,
                size: delta
            };

            // Store the new chunk
            chunks.push( chunk );

            // Continue splitting chunks
            split( chunk );
        }
    }

    // Start splitting chunks until we can fit them all into the texture
    split( chunks[0] );

    // Now that we have a set of chunks that fit, start updating the texture data
    var i, j, n, m, chunk, data, force = 1.0;

    for ( i = 0, n = chunks.length; i < n; i++ ) {

        chunk = chunks[i];
        data = [];

        for ( j = 0, m = chunk.size; j < m; j++ ) {

            data.push(

                // Move the particle to the emission origin
                origin.x,
                origin.y,
                origin.z,

                // Set the phase above 0, switching it on for use
                random( 10 ),

                // Populate the velocity slot with an initial random force
                velocities.x + force * random( -1.0, 1.0 ),
                velocities.y + force * random( -1.0, 1.0 ),
                velocities.z + force * random( -1.0, 1.0 ),

                // Empty data slot
                0
            );
        }

        // Now we modify the pixels of the particle data texture inside the chunk boundaries
        gl.texSubImage2D(
            // target, detail level, x, y, width, height
            gl.TEXTURE_2D, 0, chunk.x, chunk.y, chunk.size, 1,
            // data format, data type, pixels
            gl.RGBA, gl.FLOAT, new Float32Array( data )
        );
    }

    // Update the particle index
    gl.particleIndex += count;
    gl.particleIndex %= PARTICLE_COUNT;
}

/*
------------------------------------------------------------

Demo

------------------------------------------------------------
*/

gl.setup = function() {

    // Add Stats.js so we can monitor the FPS
    stats = new Stats();
    stats.domElement.style.position = 'absolute';
    stats.domElement.style.right = '20px';
    stats.domElement.style.top = '50px';
    document.body.appendChild( stats.domElement );

    // Set persistant states
    gl.clearColor( 0.0, 0.0, 0.0, 0.0 );

    // Store some extra data we'll need
    gl.particleIndex = 0;

    // Enable extensions - this won't work without floating point textures!
    if ( !gl.getExtension( 'OES_texture_float' ) ) alert( 'Float textures not supported' );

    // Create shader programs
    physicsProgram = createProgram( shaders.physics_vert, shaders.physics_frag );
    renderProgram = createProgram( shaders.render_vert, shaders.render_frag );
    debugProgram = createProgram( shaders.debug_vert, shaders.debug_frag );

    // Store physics program attribute and uniform locations
    physicsProgram.aVertexPositionLoc = gl.getAttribLocation( physicsProgram, 'aVertexPosition' );
    physicsProgram.uParticleDataLoc = gl.getUniformLocation( physicsProgram, 'uParticleData' );
    physicsProgram.uViewportLoc = gl.getUniformLocation( physicsProgram, 'uViewport' );

    // Store render program attribute and uniform locations
    renderProgram.uParticleTextureLoc = gl.getUniformLocation( renderProgram, 'uParticleTexture' );
    renderProgram.uParticleDataLoc = gl.getUniformLocation( renderProgram, 'uParticleData' );
    renderProgram.aParticleUVLoc = gl.getAttribLocation( renderProgram, 'aParticleUV' );

    // Store debug program attribute and uniform locations
    debugProgram.aVertexPositionLoc = gl.getAttribLocation( debugProgram, 'aVertexPosition' );
    debugProgram.uParticleDataLoc = gl.getUniformLocation( debugProgram, 'uParticleData' );

    // Enable array attribute slots in programs
    gl.enableVertexAttribArray( physicsProgram.aVertexPositionLoc );
    gl.enableVertexAttribArray( renderProgram.aParticleUVLoc );
    gl.enableVertexAttribArray( debugProgram.aVertexPositionLoc );

    // Setup the initial particle data (4 components per particle slot, all zeros)
    particleData = new Float32Array( 4 * PARTICLE_COUNT * PARTICLE_DATA_SLOTS );

    // Create a texture to hold the particle data
    particleDataTexture = gl.createTexture();
    particleDataTexture.unit = 0;

    // Activate the correct texture unit and bind the texture to it
    gl.activeTexture( gl.TEXTURE0 + particleDataTexture.unit );
    gl.bindTexture( gl.TEXTURE_2D, particleDataTexture );

    // Fill the texture with the initial particle data
    gl.texImage2D(
        // target, level, internal format, width, height 
        gl.TEXTURE_2D, 0, gl.RGBA, PARTICLE_TEXTURE_WIDTH, PARTICLE_TEXTURE_HEIGHT,
        // border, data format, data type, pixels
        0, gl.RGBA, gl.FLOAT, particleData
    );

    // Disable bilinear filtering when minifying / magnifying texture
    gl.texParameteri( gl.TEXTURE_2D, gl.TEXTURE_MIN_FILTER, gl.NEAREST );
    gl.texParameteri( gl.TEXTURE_2D, gl.TEXTURE_MAG_FILTER, gl.NEAREST );

    // Clamp the texture to the edge (don't repeat)
    gl.texParameteri( gl.TEXTURE_2D, gl.TEXTURE_WRAP_S, gl.CLAMP_TO_EDGE );
    gl.texParameteri( gl.TEXTURE_2D, gl.TEXTURE_WRAP_T, gl.CLAMP_TO_EDGE );

    // Create a framebuffer for the physics simulations to render updated particle data to and 
    // populate it with the initial particle data stored inside `particleDataTexture`
    particleFramebuffer = gl.createFramebuffer();
    gl.bindFramebuffer( gl.FRAMEBUFFER, particleFramebuffer );
    gl.framebufferTexture2D( gl.FRAMEBUFFER, gl.COLOR_ATTACHMENT0, gl.TEXTURE_2D, particleDataTexture, 0 );

    // Create UV coordinates for each particle (0 -> 1). This will tell the physics shader where
    // to find each particle's data inside the particle texture
    var particleUVData = new Float32Array( PARTICLE_COUNT * 2 );
    var interval = 1.0 / PARTICLE_COUNT_SQRT;

    for ( var i = 0, u = 0, v = 1; i < PARTICLE_COUNT; i++, u = i * 2, v = u + 1 ) {
        particleUVData[ u ] = interval * ~~( i % PARTICLE_COUNT_SQRT ); // u
        particleUVData[ v ] = interval * ~~( i / PARTICLE_COUNT_SQRT ); // v
    }
    // Buffer in the particle UV data for the render program
    particleUVDataBuffer = gl.createBuffer();
    gl.bindBuffer( gl.ARRAY_BUFFER, particleUVDataBuffer );
    gl.bufferData( gl.ARRAY_BUFFER, particleUVData, gl.STATIC_DRAW );
    gl.enableVertexAttribArray( renderProgram.aParticleUVLoc );

    // Create geometry for a fullscreen clipspace quad
    var viewportQuadVertices = new Float32Array([
        -1.0, -1.0, // 2----3
         1.0, -1.0, // | \  |
        -1.0,  1.0, // |  \ |
         1.0,  1.0  // 0----1
    ]);

    // Buffer in the geometry, used to fill FBOs at the full size of the viewport
    viewportQuadBuffer = gl.createBuffer();
    gl.bindBuffer( gl.ARRAY_BUFFER, viewportQuadBuffer );
    gl.bufferData( gl.ARRAY_BUFFER, viewportQuadVertices, gl.STATIC_DRAW );

    // Create a texture for the particle image asset
    var particleTexture = gl.createTexture();
    particleTexture.unit = 1;

    // Load the particle asset into the texture
    var particleTextureImage = document.getElementById( 'particle-texture' );
    
    // Activate the correct texture unit and bind the texture to it
    gl.activeTexture( gl.TEXTURE0 + particleTexture.unit );
    gl.bindTexture( gl.TEXTURE_2D, particleTexture );
    gl.texParameteri( gl.TEXTURE_2D, gl.TEXTURE_MIN_FILTER, gl.LINEAR_MIPMAP_NEAREST );
    gl.texParameteri( gl.TEXTURE_2D, gl.TEXTURE_MAG_FILTER, gl.LINEAR );
    gl.texParameteri( gl.TEXTURE_2D, gl.TEXTURE_WRAP_S, gl.CLAMP_TO_EDGE );
    gl.texParameteri( gl.TEXTURE_2D, gl.TEXTURE_WRAP_T, gl.CLAMP_TO_EDGE );
    gl.texImage2D( gl.TEXTURE_2D, 0, gl.RGBA, gl.RGBA, gl.UNSIGNED_BYTE, particleTextureImage );
    gl.generateMipmap( gl.TEXTURE_2D );

    // Set physics program uniform values
    gl.useProgram( physicsProgram );
    gl.uniform1i( physicsProgram.uParticleDataLoc, 0 );
    gl.uniform2f( physicsProgram.uViewportLoc, PARTICLE_TEXTURE_WIDTH, PARTICLE_TEXTURE_HEIGHT );

    // Set render program uniform values
    gl.useProgram( renderProgram );
    gl.uniform1i( renderProgram.uParticleDataLoc, particleDataTexture.unit );
    gl.uniform1i( renderProgram.uParticleTextureLoc, particleTexture.unit );

    // Set debug program uniform values
    gl.useProgram( debugProgram );
    gl.uniform1i( debugProgram.uParticleDataLoc, 0 );
    // Kick it off!
    gl.start();
};

gl.draw = function() {

    // Spit out some initial particles
    if ( gl.millis < 3000 ) {
        emitParticles( 800, {
            x: -1.0 + sin( gl.millis * 0.001 ) * 2.0,
            y: -0.2 + cos( gl.millis * 0.004 ) * 0.5,
            z: sin( gl.millis * 0.015 ) * -0.05
        });
    }

    stats.begin();

    // 1. Physics step

    // Set the viewport to the size of the particle data texture since we're rendering to that
    gl.viewport( 0, 0, PARTICLE_TEXTURE_WIDTH, PARTICLE_TEXTURE_HEIGHT );

    // Prepare the physics program to execute per fragment of the particle data texture
    gl.useProgram( physicsProgram );
    gl.bindBuffer( gl.ARRAY_BUFFER, viewportQuadBuffer );
    gl.vertexAttribPointer( physicsProgram.aVertexPositionLoc, 2, gl.FLOAT, gl.FALSE, 0, 0 );

    // Tell WebGL to use the particle FBO, not the front buffer for (offscreen) rendering
    gl.bindFramebuffer( gl.FRAMEBUFFER, particleFramebuffer );
    gl.drawArrays( gl.TRIANGLE_STRIP, 0, 4 );

    // Unbind the FBO, WebGL will now use the default buffer for rendering
    gl.bindFramebuffer( gl.FRAMEBUFFER, null );

    // 2. Render step

    // Clear the front buffer (if we had called clear when the particle FBO was bound, it
    // would have nixed the particle position and velocity data)
    gl.clear( gl.COLOR_BUFFER_BIT | gl.DEPTH_BUFFER_BIT );

    // Set the viewport size to the full canvas
    gl.viewport( 0, 0, gl.drawingBufferWidth, gl.drawingBufferHeight );

    // For each particle, pull out the position and render it as a point to the screen
    gl.useProgram( renderProgram );
    gl.bindBuffer( gl.ARRAY_BUFFER, particleUVDataBuffer );
    gl.vertexAttribPointer( renderProgram.aParticleUVLoc, 2, gl.FLOAT, gl.FALSE, 0, 0 );

    // Draw with additive blending
    gl.enable( gl.BLEND );
    gl.blendFunc( gl.SRC_ALPHA, gl.ONE );
    gl.drawArrays( gl.POINTS, 0, PARTICLE_COUNT );

    // 3. Debug step

    if ( debugToggle.checked ) {

        gl.viewport( 20, 20, 600, 300 );
        gl.useProgram( debugProgram );
        gl.bindBuffer( gl.ARRAY_BUFFER, viewportQuadBuffer );
        gl.vertexAttribPointer( debugProgram.aVertexPositionLoc, 2, gl.FLOAT, gl.FALSE, 0, 0 );

        // Draw with interpolative blending
        gl.blendFunc( gl.SRC_ALPHA, gl.ONE_MINUS_SRC_ALPHA );
        gl.drawArrays( gl.TRIANGLE_STRIP, 0, 4 );
    }

    gl.disable( gl.BLEND );

    stats.end();
};

gl.mousemove = function() {

    if ( gl.running && !gl.lastEmit || gl.millis - gl.lastEmit > 16 ) {

        gl.lastEmit = gl.millis;

        var i, n, x, y, touch, limit = PARTICLE_EMIT_RATE / gl.touches.length;

        for ( i = 0, n = gl.touches.length; i < n; i++ ) {
touch = gl.touches[i];

            // Convert mouse screen coordinates into clipspace coordinates
            x = map( touch.x, 0, gl.width, -1, 1 );
            y = map( touch.y, 0, gl.height, 1, -1 );

            // Emit some particles!
            emitParticles( limit, {
                x: x,
                y: y,
                z: 0.0
            });}
    }
};
            
          
!
Must be a valid URL.
+ add another resource
via JS Hint
Loading ..................