Pen Settings

HTML

CSS

CSS Base

Vendor Prefixing

Add External Stylesheets/Pens

Any URLs added here will be added as <link>s in order, and before the CSS in the editor. You can use the CSS from another Pen by using its URL and the proper URL extension.

+ add another resource

JavaScript

Babel includes JSX processing.

Add External Scripts/Pens

Any URL's added here will be added as <script>s in order, and run before the JavaScript in the editor. You can use the URL of any other Pen and it will include the JavaScript from that Pen.

+ add another resource

Packages

Add Packages

Search for and use JavaScript packages from npm here. By selecting a package, an import statement will be added to the top of the JavaScript editor for this package.

Behavior

Auto Save

If active, Pens will autosave every 30 seconds after being saved once.

Auto-Updating Preview

If enabled, the preview panel updates automatically as you code. If disabled, use the "Run" button to update.

Format on Save

If enabled, your code will be formatted when you actively save your Pen. Note: your code becomes un-folded during formatting.

Editor Settings

Code Indentation

Want to change your Syntax Highlighting theme, Fonts and more?

Visit your global Editor Settings.

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='https://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( https://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;
}
              
            
!

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
            });}
    }
};
              
            
!
999px

Console