3

When using textures in WebGL, sometimes I need to make them larger than they were originally. When I do that, it causes the textures to appear differently, especially on lighter backgrounds.

I have the following image (256 x 256):

original image

When rendered in WebGL, it is slightly larger than the original image. Here is how the image appears on two different backgrounds:

enter image description here image on light background

As you can see, the image appears correctly on the dark background, but when on the light background, has a white outline.

My setup code:

gl.clearColor(0x22 / 0xFF, 0x22 / 0xFF, 0x22 / 0xFF, 1); // set background color
gl.enable(gl.BLEND); // enable transparency
gl.disable(gl.DEPTH_TEST); // disable depth test (causes problems with alpha if enabled)
gl.blendFunc(gl.SRC_ALPHA, gl.ONE_MINUS_SRC_ALPHA); //set up blending
gl.clear(gl.COLOR_BUFFER_BIT | gl.DEPTH_BUFFER_BIT); //clear the gl canvas
gl.viewport(0, 0, canvas.width, canvas.height); //set the viewport

And this is the code called every time a texture is loaded:

function handleTextureLoaded(image, texture) {
  gl.bindTexture(gl.TEXTURE_2D, texture);
  gl.texImage2D(gl.TEXTURE_2D, 0, gl.RGBA, gl.RGBA, gl.UNSIGNED_BYTE, image);
  gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MAG_FILTER, gl.LINEAR);
  gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MIN_FILTER, gl.LINEAR_MIPMAP_NEAREST);
  gl.generateMipmap(gl.TEXTURE_2D);
  gl.bindTexture(gl.TEXTURE_2D, null);
  loadCount++;
}

What is causing the outline to appear, and how do I fix it?

NOTE: When I put the original image on these same two backgrounds, this problem does not occur, even when I resize the image.

I tried disabling the alpha on the WebGL context (as told by @zfedoran):

gl = canvas.getContext('webgl', {antialias: false, alpha: false }) 
  || canvas.getContext('experimental-webgl', {antialias: false, alpha: false });

And a small blank border now appears around the image, like this (enlarged):

enter image description here

vcapra1
  • 1,988
  • 3
  • 26
  • 45

2 Answers2

5

On top of the canvas's alpha as mentioned by @zfedoran how do you make the original image?

I believe the issue is as follows

Let's say you have an anti-aliased edge like this. What color is this pixel?

enter image description here

Assume the main color, the color of the pixels in the bottom right, was 1,0,0 (pure red). Ideally the pixel pointed to by the arrow would be (1,0,0,0.5). In other words, pure red with an alpha of 0.5. But, depending how on the image was created to generate that anti-aliased pixel it might have been blended with the purely transparent pixels next to it so it no longer pure red. Those purely transparent pixels are likely (0,0,0,0) which is transparent black.

Even if your drawing program handles this correctly, GL likely does not. When you draw an image with texture filtering on (gl.LINEAR etc) GL is going to average the pixels near each other, some of those pixels are transparent black. Blending black with red gives dark red. Hence you get a dark border.

Here you can see the issue

"use strict";

function main() {
  var planeVertices = [
       -1, -1,
        1, -1,
       -1,  1,
        1,  1,
    ];
      
  var texcoords = [
     0, 1,
     1, 1,
     0, 0,
     1, 0,
  ];
    
  var indices = [
     0, 1, 2,
     2, 1, 3,
  ];
    

  var canvas = document.getElementById("c");
  var gl = canvas.getContext("webgl", {alpha:false});

  var programs = {}
  programs.normalProgram = twgl.createProgramFromScripts(
      gl, ["2d-vertex-shader", "2d-fragment-shader"], ["a_position", "a_texcoord"]);
  programs.preMultiplyAlphaProgram = twgl.createProgramFromScripts(
      gl, ["2d-vertex-shader", "pre-2d-fragment-shader"], ["a_position", "a_texcoord"]);
    
  var positionLoc = 0;  // assigned in createProgramsFromScripts
  var texcoordLoc = 1;  // assigned in createProgramsFromScripts

  var buffer = gl.createBuffer();
  gl.bindBuffer(gl.ARRAY_BUFFER, buffer);
  gl.bufferData(
      gl.ARRAY_BUFFER,
      new Float32Array(planeVertices),
      gl.STATIC_DRAW);
  gl.enableVertexAttribArray(positionLoc);
  gl.vertexAttribPointer(positionLoc, 2, gl.FLOAT, false, 0, 0);
    
  var buffer = gl.createBuffer();
  gl.bindBuffer(gl.ARRAY_BUFFER, buffer);
  gl.bufferData(
      gl.ARRAY_BUFFER,
      new Float32Array(texcoords),
      gl.STATIC_DRAW);
  gl.enableVertexAttribArray(texcoordLoc);
  gl.vertexAttribPointer(texcoordLoc, 2, gl.FLOAT, false, 0, 0);
    
  var buffer = gl.createBuffer();
  gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, buffer);
  gl.bufferData(
      gl.ELEMENT_ARRAY_BUFFER,
      new Uint16Array(indices),
      gl.STATIC_DRAW);    

  var img = new Image();
  img.onload = createTextures;
  img.src = document.getElementById("i").text;
    
  function createTexture() {
    var tex = gl.createTexture();
    gl.bindTexture(gl.TEXTURE_2D, tex);
    gl.texImage2D(gl.TEXTURE_2D, 0, gl.RGBA, gl.RGBA, gl.UNSIGNED_BYTE, img);
    gl.generateMipmap(gl.TEXTURE_2D); // assuming power-of-2 
    return tex;
  }
 
  var textures = {};    
  function createTextures() {
    gl.pixelStorei(gl.UNPACK_PREMULTIPLY_ALPHA_WEBGL, false);
    textures.unpremultipliedAlphaTexture = createTexture();
    gl.pixelStorei(gl.UNPACK_PREMULTIPLY_ALPHA_WEBGL, true);
    textures.premultipliedAlphaTexture = createTexture();
    document.body.appendChild(document.createElement("hr"));
    insert("original image");
    document.body.appendChild(img);
    render();
  }
    
  function insert(text) {
    var pre = document.createElement("pre");
    pre.appendChild(document.createTextNode(text));
    document.body.appendChild(pre);
  };
    
  function grabImage(prg, blend, texName) {
     document.body.appendChild(document.createElement("hr"));
     insert(
       "gl.useProgram(" + prg + ")\n" +
       "gl.blendFunc(gl." + blend.src + ", gl." + blend.dst + ")\n" +
       "gl.bindTexture(gl.TEXTURE2D, " + texName + ")");
     var img = new Image();
     img.src = gl.canvas.toDataURL();
     document.body.appendChild(img);
  };
  
  function render() { 
      gl.enable(gl.BLEND);
      
      Object.keys(programs).forEach(function(p, pndx) {
          gl.useProgram(programs[p]);
          
          [
              { src: "SRC_ALPHA", dst: "ONE_MINUS_SRC_ALPHA" },
              { src: "ONE", dst: "ONE_MINUS_SRC_ALPHA" },
          ].forEach(function(b, bndx) {
               gl.blendFunc(gl[b.src], gl[b.dst]);
           
               Object.keys(textures).forEach(function(texName, tndx) {
                  gl.bindTexture(gl.TEXTURE_2D, textures[texName]);
                  gl.clearColor(0x3D/0xFF, 0x87/0xFF, 0xEA/0xFF, 1);
                  gl.clear(gl.COLOR_BUFFER_BIT);
                  gl.drawElements(gl.TRIANGLES, 6, gl.UNSIGNED_SHORT, 0);
                  grabImage(p, b, texName);
              });
          });
      });
   }
}

main();
canvas {
    border: 1px solid black;
    display: none;
}
img {
    background-color: #3D87EA;
    border: 1px solid black;
    width: 256px;
    height: 256px;
}
<script src="https://twgljs.org/dist/3.x/twgl.min.js"></script>
<!-- vertex shader -->
<script id="2d-vertex-shader" type="x-shader/x-vertex">
attribute vec4 a_position;
attribute vec2 a_texcoord;

varying vec2 v_texcoord;
    
void main() {
   gl_Position = a_position;
   v_texcoord = a_texcoord;
}
</script>
<!-- fragment shaders -->
<script id="2d-fragment-shader" type="x-shader/x-fragment">
precision mediump float;
    
varying vec2 v_texcoord;

uniform sampler2D u_texture;
void main() {
    gl_FragColor = texture2D(u_texture, v_texcoord);
}
</script>
<script id="pre-2d-fragment-shader" type="x-shader/x-fragment">
precision mediump float;
    
varying vec2 v_texcoord;

uniform sampler2D u_texture;
void main() {
    vec4 textureColor = texture2D(u_texture, v_texcoord);
    gl_FragColor = vec4(textureColor.rgb * textureColor.a, textureColor.a);
}
</script>
<canvas id="c" width="32" height="32"></canvas>
<script type="not-js" id="i"></script>

There's a few solutions

  1. Make sure transparent area actually has color in.

    In other words, if all the pixels in the top left of the image above are RED with 0 alpha then when the pixels get filtered they'll be blending (1,0,0,0) transparent red instead of (0,0,0,0) transparent black. Unfortunately there's no easy way to do this in most drawing programs.

    There's a plugin for Photoshop that lets you do it called SuperPNG It lets you create a 4th channel for the alpha instead of using photoshop's transparency. That lets you set the alpha separate from the image.

    In your case you'd end up with an image with layers like this

    enter image description here

    Now there are no bad colors to blend with.

  2. Switch to pre-multiplied alpha

    In this case before calling gl.texImage2D to upload the image call

    gl.pixelStorei(UNPACK_PREMULTIPLY_ALPHA_WEBGL, true);
    

    before calling gl.texImage2D. That tells WebGL to multiply the colors by their alpha when the image is loaded. You then use blending with

    gl.blendFunc(gl.ONE, gl.ONE_MINUS_SRC_ALPHA);
    
  3. Turn off filtering in GL

    gl.texParameter(gl.TEXTURE_2D, gl.TEXTURE_MIN_FILTER, gl.NEAREST);
    gl.texParameter(gl.TEXTURE_2D, gl.TEXTURE_MAG_FILTER, gl.NEAREST);
    

    Assuming your source image doesn't have any bad colors this means GL won't making new bad colors as it filters but of course it also means if you scale or rotate the image you'll get aliasing.

  4. Create your own mips

    Most apps use gl.genereateMipmap to generate mips but you can generate them yourself offline and upload them yourself. That's not a perfect solution either but it does let you use `gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MIN_FILTER, gl.LINEAR_MIPMAP_NEAREST);

Combinations of the above

gman
  • 100,619
  • 31
  • 269
  • 393
  • This fixed the black border problem, but how can I still be able to set the texture alpha (such as in a fade animation)? – vcapra1 Aug 18 '14 at 18:15
2

Have you tried disabling the alpha on the WebGL context?

var gl = this.canvas.getContext('webgl', {antialias: false, alpha: false }) 
      || this.canvas.getContext('experimental-webgl', {antialias: false, alpha: false });
zfedoran
  • 2,986
  • 4
  • 22
  • 25
  • This made everything appear much better, but now there is a very thin, almost unnoticeable, black border. I posted what it looks like in my question. Also, Could you explain what disabling the alpha on the context does? Because I can still make things transparent. Thank you. – vcapra1 Aug 18 '14 at 15:01
  • The `alpha:true` makes the canvas opaque to the DOM elements underneath. If this is not something you want, try setting the document body color (with CSS) to the same color as your WebGL reset color. The black border may be due to how anti-aliasing is handled. – zfedoran Aug 18 '14 at 15:44
  • Thank you for explaining this. I will play around with it to try and fix the black border. – vcapra1 Aug 18 '14 at 15:53