WebGL Texture resize unexpected output

When using textures in WebGL, sometimes I need to make them larger than they were originally. When I do this, it results in textures looking different, 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. This is how the image appears on two different backgrounds:

enter image description hereimage on light background

As you can see, the image displays correctly on a dark background, but when on a 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 that gets called every time the 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 causes the contour to appear and how to fix it?

NOTE. ... When I place my original image on these two backgrounds, this issue doesn't occur even when I resize the image.

I tried to disable alpha in the WebGL context (as @zfedoran said):

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

      

Now a small empty border appears around the image, like (zoomed in):

enter image description here

+3


source to share


2 answers


Have you tried turning off alpha in the context of WebGL?



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

      

+2


source


At the top of the alpha canvas, as @zfedoran mentioned, how do you make the original image?

I believe the problem is this:

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

enter image description here

Let's say the foreground color, the color of the pixels in the lower right corner, was 1.0.0 (pure red). Ideally, the pixel the arrow points to would be (1,0,0,0,5). In other words, pure red with an alpha of 0.5. But, depending on how the image was created to create this anti-aliased pixel, it could be blended with transparent pixels next to it so that it is not completely pure red. These purely transparent pixels are likely (0,0,0,0) that are transparent black .

Even if your paint program handles this correctly, GL probably doesn't. When you paint an image with texture filtering ( gl.LINEAR

etc.), GL will average pixels next to each other, some of those pixels are transparent black . Mixing black with red gives a dark red color. Hence, you end up with a dark frame.



You can see the problem here

"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>
      

Run codeHide result


There are several solutions

  • Make sure the transparent area has a color.

    In other words, if all the pixels in the upper left corner of the image are above RED with 0 alpha, then when the pixels are filtered they will blend (1,0,0,0) transparent red instead of (0,0,0,0) transparent black ... Unfortunately, there is no easy way to do this in most drawing programs.

    Here's a plugin for Photoshop that lets you do this SuperPNG It allows you to create a 4th channel for the alpha instead of using photoshop transparency.This allows you to set the alpha separately from the image.

    In your case, you will end up with an image with layers like this

    enter image description here

    Now there are no bad colors to mix with.

  • Switch to pre-multiplied alpha

    In this case, before the call gl.texImage2D

    to load the call image

    gl.pixelStorei(UNPACK_PREMULTIPLY_ALPHA_WEBGL, true);
    
          

    before calling gl.texImage2D

    . This tells WebGL to multiply the colors by their alpha when the image is loaded. Then you use the combination with

    gl.blendFunc(gl.ONE, gl.ONE_MINUS_SRC_ALPHA);
    
          

  • Disable 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 the original image doesn't have any bad colors, this means GL won't create new bad colors as it filters, but of course this also means that if you scale or rotate the image, you'll get an alias.

  • Create your own mips

    Most applications use gl.genereateMipmap

    mips to generate, but you can build them yourself offline and download them yourself. This is not ideal, but it allows you to use `gl.texParameteri (gl.TEXTURE_2D, gl.TEXTURE_MIN_FILTER, gl.LINEAR_MIPMAP_NEAREST);

Combinations of the above

+4


source







All Articles