簡體   English   中英

WebGL紋理調整意外輸出的大小

[英]WebGL Texture resize unexpected output

在WebGL中使用紋理時,有時我需要使它們比原來更大。 當我這樣做時,它會使紋理顯示不同,特別是在較淺的背景上。

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

原始圖像

在WebGL中渲染時,它略大於原始圖像。 以下是圖像在兩個不同背景上的顯示方式:

在此輸入圖像描述在明亮的背景上的圖像

如您所見,圖像在深色背景上正確顯示,但在淺色背景上,則為白色輪廓。

我的設置代碼:

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

現在圖像周圍出現一個小的空白邊框,如下所示(放大):

在此輸入圖像描述

在@zfedoran提到的畫布上的alpha之上,你如何制作原始圖像?

我相信問題如下

假設你有這樣的抗鋸齒邊緣。 這個像素是什么顏色的?

在此輸入圖像描述

假設主要顏色(右下角的像素顏色)為1,0,0(純紅色)。 理想情況下,箭頭指向的像素為(1,0,0,0.5)。 換句話說,純紅色的α為0.5。 但是,根據圖像的創建方式來生成抗鋸齒像素,它可能已與其旁邊的純透明像素混合,因此它不再是純紅色。 那些純透明像素可能是(0,0,0,0),它是透明的黑色

即使您的繪圖程序正確處理此問題,GL也可能不會。 當您使用紋理過濾( gl.LINEAR等)繪制圖像時,GL將平均彼此接近的像素,其中一些像素是透明的黑色 混合黑色和紅色會產生深紅色。 因此你得到一個黑暗的邊界。

在這里你可以看到問題

 "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. 確保透明區域實際上有顏色。

    換句話說,如果上面圖像左上角的所有像素都是帶有0 alpha的RED ,那么當像素被過濾時,它們將混合(1,0,0,0)透明紅色而不是(0,0, 0,0)透明黑色。 不幸的是,在大多數繪圖程序中沒有簡單的方法可以做到這一點。

    一個Photoshop插件可以讓你做到這個名為SuperPNG它允許你為alpha創建第4個頻道,而不是使用photoshop的透明度。 這使您可以將alpha設置為與圖像分開。

    在你的情況下,你最終會得到一個像這樣的圖像的圖像

    在此輸入圖像描述

    現在沒有壞顏色可以混合。

  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生成mips,但您可以離線生成它們並gl.genereateMipmap上傳。 這不是一個完美的解決方案,但它確實讓你使用`gl.texParameteri(gl.TEXTURE_2D,gl.TEXTURE_MIN_FILTER,gl.LINEAR_MIPMAP_NEAREST);

以上的組合

您是否嘗試在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