简体   繁体   English

WebGL纹理调整意外输出的大小

[英]WebGL Texture resize unexpected output

When using textures in WebGL, sometimes I need to make them larger than they were originally. 在WebGL中使用纹理时,有时我需要使它们比原来更大。 When I do that, it causes the textures to appear differently, especially on lighter backgrounds. 当我这样做时,它会使纹理显示不同,特别是在较浅的背景上。

I have the following image (256 x 256): 我有以下图像(256 x 256):

原始图像

When rendered in WebGL, it is slightly larger than the original image. 在WebGL中渲染时,它略大于原始图像。 Here is how the image appears on two different backgrounds: 以下是图像在两个不同背景上的显示方式:

在此输入图像描述在明亮的背景上的图像

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): 我尝试在WebGL上下文中禁用alpha(由@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): 现在图像周围出现一个小的空白边框,如下所示(放大):

在此输入图像描述

On top of the canvas's alpha as mentioned by @zfedoran how do you make the original image? 在@zfedoran提到的画布上的alpha之上,你如何制作原始图像?

I believe the issue is as follows 我相信问题如下

Let's say you have an anti-aliased edge like this. 假设你有这样的抗锯齿边缘。 What color is this pixel? 这个像素是什么颜色的?

在此输入图像描述

Assume the main color, the color of the pixels in the bottom right, was 1,0,0 (pure red). 假设主要颜色(右下角的像素颜色)为1,0,0(纯红色)。 Ideally the pixel pointed to by the arrow would be (1,0,0,0.5). 理想情况下,箭头指向的像素为(1,0,0,0.5)。 In other words, pure red with an alpha of 0.5. 换句话说,纯红色的α为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 . 那些纯透明像素可能是(0,0,0,0),它是透明的黑色

Even if your drawing program handles this correctly, GL likely does not. 即使您的绘图程序正确处理此问题,GL也可能不会。 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 . 当您使用纹理过滤( gl.LINEAR等)绘制图像时,GL将平均彼此接近的像素,其中一些像素是透明的黑色 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. 换句话说,如果上面图像左上角的所有像素都是带有0 alpha的RED ,那么当像素被过滤时,它们将混合(1,0,0,0)透明红色而不是(0,0, 0,0)透明黑色。 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. 一个Photoshop插件可以让你做到这个名为SuperPNG它允许你为alpha创建第4个频道,而不是使用photoshop的透明度。 That lets you set the alpha separate from the image. 这使您可以将alpha设置为与图像分开。

    In your case you'd end up with an image with layers like this 在你的情况下,你最终会得到一个像这样的图像的图像

    在此输入图像描述

    Now there are no bad colors to blend with. 现在没有坏颜色可以混合。

  2. Switch to pre-multiplied alpha 切换到预乘alpha

    In this case before calling gl.texImage2D to upload the image call 在这种情况下,在调用gl.texImage2D之前上传图像调用

     gl.pixelStorei(UNPACK_PREMULTIPLY_ALPHA_WEBGL, true); 

    before calling gl.texImage2D . 在调用gl.texImage2D之前。 That tells WebGL to multiply the colors by their alpha when the image is loaded. 这告诉WebGL在加载图像时将颜色乘以alpha。 You then use blending with 然后使用混合

     gl.blendFunc(gl.ONE, gl.ONE_MINUS_SRC_ALPHA); 
  3. Turn off filtering in GL 在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. 假设您的源图像没有任何不良颜色,这意味着GL不会在过滤时产生新的坏颜色,但当然这也意味着如果缩放或旋转图像,您将获得锯齿。

  4. Create your own mips 创建自己的mips

    Most apps use gl.genereateMipmap to generate mips but you can generate them yourself offline and upload them yourself. 大多数应用程序使用gl.genereateMipmap生成mips,但您可以离线生成它们并gl.genereateMipmap上传。 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); 这不是一个完美的解决方案,但它确实让你使用`gl.texParameteri(gl.TEXTURE_2D,gl.TEXTURE_MIN_FILTER,gl.LINEAR_MIPMAP_NEAREST);

Combinations of the above 以上的组合

Have you tried disabling the alpha on the WebGL context? 您是否尝试在WebGL上下文中禁用alpha?

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

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM