javascript - WebGL 纹理调整大小意外输出

标签 javascript opengl-es textures webgl texture-mapping

在 WebGL 中使用纹理时,有时我需要使它们比原来更大。当我这样做时,它会导致纹理显示不同,尤其是在较亮的背景上。

我有以下图像 (256 x 256):

original image

在 WebGL 中渲染时,它比原始图像稍大。以下是图像在两种不同背景下的显示方式:

enter image description here image on light background

如您所见,图像在深色背景上显示正确,但在浅色背景上显示为白色轮廓。

我的设置代码:

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

这是每次加载纹理时调用的代码:

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++;
}

是什么导致出现轮廓,我该如何解决?

注意:当我将原始图像放在这两个相同的背景上时,即使调整图像大小也不会出现此问题。

我尝试在 WebGL 上下文中禁用 alpha(如@zfedoran 所述):

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

现在图像周围出现了一个小的空白边框,像这样(放大):

enter image description here

最佳答案

如@zfedoran 所述,在 Canvas 的 alpha 之上,您如何制作原始图像?

我认为问题如下

假设您有这样的抗锯齿边缘。这个像素是什么颜色?

enter image description here

假设主色,即右下角像素的颜色,是 1,0,0(纯红色)。理想情况下,箭头指向的像素为 (1,0,0,0.5)。换句话说,纯红色的 alpha 值为 0.5。但是,根据创建图像以生成抗锯齿像素的方式,它可能已与旁边的纯透明像素混合,因此它不再是纯红色。那些纯透明像素很可能是 (0,0,0,0),它是透明的黑色

即使您的绘图程序可以正确处理此问题,GL 也可能不会。当您使用纹理过滤(gl.LINEAR 等)绘制图像时,GL 将对彼此附近的像素进行平均,其中一些像素是透明的黑色。黑色与红色混合会产生深红色。因此你会得到一个黑色的边框。

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>

有几个解决方案

  1. 确保透明区域确实有颜色。

    换句话说,如果上图左上角的所有像素都是 RED 且 alpha 为 0,那么当像素被过滤时,它们将混合 (1,0,0,0 ) 透明红色而不是 (0,0,0,0) 透明黑色。不幸的是,在大多数绘图程序中都没有简单的方法可以做到这一点。

    a plugin for Photoshop that lets you do it called SuperPNG它允许您为 alpha 创建第四个 channel ,而不是使用 photoshop 的透明度。这使您可以将 alpha 设置为与图像分开。

    在你的情况下,你最终会得到一个像这样的图层的图像

    enter image description here

    现在没有不好的颜色可以混合了。

  2. 切换到预乘 alpha

    本例调用gl.texImage2D上传图片调用

    gl.pixelStorei(UNPACK_PREMULTIPLY_ALPHA_WEBGL, true);
    

    在调用 gl.texImage2D 之前。这告诉 WebGL 在加载图像时将颜色乘以它们的 alpha。然后你使用混合

    gl.blendFunc(gl.ONE, gl.ONE_MINUS_SRC_ALPHA);
    
  3. 关闭 GL 中的过滤

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

    假设您的源图像没有任何不良颜色,这意味着 GL 不会在过滤时产生新的不良颜色,但当然这也意味着如果您缩放或旋转图像,您将出现锯齿。

  4. 创建自己的 mips

    大多数应用程序使用 gl.genereateMipmap 生成 mip,但您可以自己离线生成它们并自己上传。这也不是一个完美的解决方案,但它确实允许您使用 `gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MIN_FILTER, gl.LINEAR_MIPMAP_NEAREST);

上述的组合

关于javascript - WebGL 纹理调整大小意外输出,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/25352627/

相关文章:

opengl-es - 具有白色轮廓的 OpenGL 彩色位图字体

java - GL_NEAREST 上的 Android OpenGL mag 过滤器,小纹理仍然模糊

c++ - 在 openGL 纹理中加载图像(使用它们的 RGB(A) 像素数据)

javascript - VueJS 连接外部数据

javascript - 使用jspdf将html页面保存为pdf,如果浏览器缩放,保存的pdf页面内容不完整,为什么?

javascript - 针对 Codeigniter 站点上 Javascript 文件通过 http 提供的内容的不安全内容警告

iphone - 如何将 OpenGL View 捕获为 UIImage?

C++ 将纹理添加到 GL_QUAD 并且它变黑了

webgl - 位移贴图 UV 贴图?

javascript - 当在选项上设置 header 时,获取 POST 内容类型不会更改