Search code examples
webgl2

Can't get a working example for WebGL2 texture using images


I've looked around and cannot find a simple example showcasing 3D Textures using multiple images to map onto any geometry.

I've found the following examples:

1) http://webglsamples.org/WebGL2Samples/#texture_3d 2) https://webgl2fundamentals.org/webgl/lessons/webgl-data-textures.html

In the first example, I cannot get it to work using images as the source for data. In the second example, it doesn't use 3D textures.

If there are any examples of showing 3D textures mapping multiple images (e.g. JPG) onto a plane, that would be super helpful


Solution

  • Here's a small sample. It's using dataURLs for the images but they could be normal images just as easily as long as they are all the same dimensions. If they are not the same dimensions you'd have to scale them so they are either offline or via canvas or something.

    Otherwise the sample doesn't set any uniforms because the default value of uniforms is 0. It just draws a single point using gl.POINTS, gl_PointSize and gl_PointCoord for coordinates so no buffers or attributes needed

    const imageURLs = [
      "",
      "",
      "",
      "",
    ];
    
    const vs = `#version 300 es
    void main() {
      gl_Position = vec4(0, 0, 0, 1);
      gl_PointSize = 150.0;
    }
    `
    
    const fs = `#version 300 es
    precision mediump float;
    uniform mediump sampler3D tex;
    out vec4 outColor;
    void main() {
      vec3 uvw = vec3(gl_PointCoord.xy, gl_PointCoord.x * gl_PointCoord.y);
      outColor = texture(tex, uvw);
    }
    `;
    
    function main() {
      const gl = document.querySelector('canvas').getContext("webgl2");
      if (!gl) { return console.log; ("need webgl2"); }
      
      const program = twgl.createProgram(gl, [vs, fs]);
      
      const tex = gl.createTexture();
      gl.bindTexture(gl.TEXTURE_3D, tex);
      gl.texParameteri(gl.TEXTURE_3D, gl.TEXTURE_MIN_FILTER, gl.LINEAR);
      // we either need to preallocate the space OR we need to wait for 
      // one texture to download to find out the size. All the sizes have
      // to be the same as all slices are required to have the same dimensions
      
      // I happen to know all the images are 16x16 so
      const levels = 1;
      const internalFormat = gl.RGBA8;
      const width = 16;
      const height = 16;
      const depth = imageURLs.length;
      gl.texStorage3D(gl.TEXTURE_3D, levels, internalFormat, width, height, depth);
      
      // load the images
      imageURLs.forEach((url, ndx) => {
        const img = new Image();
        document.body.appendChild(img);  // so we can see the images;
        img.onload = () => {
          gl.bindTexture(gl.TEXTURE_3D, tex);
          
          const level = 0;
          const x = 0;
          const y = 0;
          const z = ndx;
          const depth = 1
          gl.texSubImage3D(
               gl.TEXTURE_3D, level, 
               x, y, z, width, height, depth, 
               gl.RGBA, gl.UNSIGNED_BYTE, img);
    
          // render each time an image comes in. This means
          // the first render will be missing 3 layers or rather
          // the'll be blank (0,0,0,0). We could wait for all images
          // to load instead.
          render();
        };
        img.src = url;
      });
    
      function render() {
        gl.useProgram(program);
        gl.drawArrays(gl.POINTS, 0, 1);  // 1 point
      }
    }
    main();
    canvas {
      margin: .25em;
      border: 1px solid black;
    }
    img {
      margin: .25em;
      border: 1px solid black;
      width: 64px;
      height: 64px;
      image-rendering: pixelated;
    }
    <canvas></canvas>
    <script src="https://twgljs.org/dist/4.x/twgl.min.js"></script>