我很难理解为什么在下面的代码中,当我直接渲染到屏幕时,模板测试按预期工作,而当我尝试使用帧缓冲时则不行。我没有问题使用这些帧缓冲区对纹理进行多遍后处理,所以我怀疑我的深度和深度都做错了。模板缓冲区。
首先我初始化,上下文:
gl = canvas.getContext("webgl",{stencil:true});
然后
function initFBO()
{
for (var i = 0; i < shaderPrograms.length; i++)
{
var texture = createTexture(imgWidth,imgHeight);
textures.push(texture);
var fbo = gl.createFramebuffer();
framebuffers.push(fbo);
gl.bindFramebuffer(gl.FRAMEBUFFER, fbo);
var renderbuffer = gl.createRenderbuffer();
gl.bindRenderbuffer(gl.RENDERBUFFER, renderbuffer);
gl.renderbufferStorage(gl.RENDERBUFFER, gl.DEPTH_STENCIL, imgWidth, imgHeight);
gl.framebufferTexture2D(gl.FRAMEBUFFER, gl.COLOR_ATTACHMENT0, gl.TEXTURE_2D, texture, 0);
gl.framebufferRenderbuffer(gl.FRAMEBUFFER, gl.DEPTH_STENCIL_ATTACHMENT, gl.RENDERBUFFER, renderbuffer);
}
}
function drawScene()
{
gl.enable(gl.STENCIL_TEST);
gl.enable(gl.DEPTH_TEST);
if(!useFBO)
{
computeProgFBO(9,null,imgHeight,imgHeight,textureId,null,null,false);
computeProgFBO(8,null,imgHeight,imgHeight,textureId,null,null,false);
}else{
computeProgFBO(9,framebuffers[0],imgHeight,imgHeight,textureId,null,null,false);
computeProgFBO(8,null,imgHeight,imgHeight,textures[0],null,null,false);
}
// Restore the original matrix
mvPopMatrix();
gl.disable(gl.STENCIL_TEST);
gl.disable(gl.DEPTH_TEST);
}
function computeProgFBO(progId,fboId,w,h,textureId0,textureId1,textureId2,flip)
{
currentProgram=shaderPrograms[progId];
gl.useProgram(currentProgram);
if(progId==9)
{
transform(w,h,0.0);
gl.stencilMask(1);
gl.stencilFunc(gl.ALWAYS,1,0);
gl.stencilOp(gl.KEEP,gl.KEEP,gl.REPLACE);
gl.colorMask(true, true, true, true);
gl.depthMask(false);
setMatrixUniformsView();
}else{
gl.colorMask(true, true, true, true);
gl.depthMask(true);
gl.stencilFunc(gl.EQUAL,0,1);
gl.stencilOp(gl.KEEP,gl.KEEP,gl.KEEP);
setMatrixUniforms();
}
setMyUniforms(progId,flip);
setMyTexture(textureId0, textureId1,textureId2);
gl.activeTexture(gl.TEXTURE1);
gl.bindFramebuffer(gl.FRAMEBUFFER, fboId);
//the below is commented out for now as i'm only testing the stencil buffer for now
// gl.framebufferTexture2D(gl.FRAMEBUFFER, gl.COLOR_ATTACHMENT0, gl.TEXTURE_2D, textures[progId], 0);
gl.drawElements(gl.TRIANGLES, 6, gl.UNSIGNED_SHORT, 0);
gl.bindFramebuffer(gl.FRAMEBUFFER, null);
gl.bindTexture(gl.TEXTURE_2D , null);
}
我不认为展示不同的制服和纹理功能有任何意义,但请不要让我知道。顺便说一句,progId 9绘制了一个小绿色方块,并在其上方增加了8个更大的红色方块。因此,我希望看到一个绿色正方形被一些红色边缘包围,因为它在我直接渲染到屏幕时显示。
****再到第一个答案,我修改了以下*****
if(!useFBO)
{
computeProgFBO(9,null,imgHeight,imgHeight,textureId,null,null,false,null);
computeProgFBO(8,null,imgHeight,imgHeight,textureId,null,null,false,null);
computeProgFBO(6,null,imgHeight,imgHeight,textureId,null,null,false,null);
}else{
computeProgFBO(9,framebuffers[0],imgHeight,imgHeight,textureId,null,null,false,null);
computeProgFBO(8,framebuffers[1],imgHeight,imgHeight,textures[0],null,null,false,renderbuffers[0]);
computeProgFBO(6,null,imgHeight,imgHeight,textures[1],null,null,false,null);
}
我已将函数参数更改为
computeProgFBO(progId,fboId,w,h,textureId0,textureId1,textureId2,flip,render)
并在以下评论之后
//the below is commented out for now as i'm only testing the stencil buffer for now
// gl.framebufferTexture2D(gl.FRAMEBUFFER, gl.COLOR_ATTACHMENT0, gl.TEXTURE_2D, textures[progId], 0);
我添加了
if(render!=null)
{
gl.framebufferRenderbuffer(gl.FRAMEBUFFER, gl.DEPTH_STENCIL_ATTACHMENT, gl.RENDERBUFFER, render);
}
结果略胜一筹,但还不完全正是我所期待的,即:我得到了大红色的正方形而不是内部的小绿色方块,我得到了一个白色方块。我觉得这是微不足道的,但还不能真正解决。直接渲染到屏幕仍然可以正常工作。
*****再向其他人发表评论我添加了******
我想我理解了这个问题并为它找到了一个解决方法[不是很优雅]。基本上,第一遍使用模板写入帧缓冲区[0]的纹理[0],第二遍使用framebuffers [0]的renderbuffers [0]写入帧缓冲区[1]的纹理[1],但不是纹理[0] ]。因此,我最终得到的只是一个小的白色[而不是绿色]正方形,周围是红色的。因为,我不能说,我已经意识到我可以在第一遍中画两次,如下所示:
....
gl.drawElements(gl.TRIANGLES, 6, gl.UNSIGNED_SHORT, 0);
if(progId==9)
{
gl.bindFramebuffer(gl.FRAMEBUFFER, framebuffers[1]);
gl.drawElements(gl.TRIANGLES, 6, gl.UNSIGNED_SHORT, 0);
}
这有效,但看起来很难看!还有其他办法吗?
当谈到将最终纹理渲染到屏幕时,似乎我不必做太多,即:我只将颜色缓冲区发送到屏幕(而不是模板和深度缓冲区)和所有最终纹理按预期显示。
另请注意,我的片段着色器仅执行以下操作
shader9:gl_FragColor = vec4(0.0,1.0,0.0,1.0);
shader8:gl_FragColor = vec4(1.0,vec2(0.0),1.0);
shader6:gl_FragColor = texture2D(u_Texture0,vTextureCoord);
答案 0 :(得分:2)
您不能在webgl中复制缓冲区的内容,但是您可以通过在同一个渲染缓冲区上调用gl.framebufferRenderbuffer(gl.FRAMEBUFFER, gl.DEPTH_STENCIL_ATTACHMENT, gl.RENDERBUFFER, renderbuffer);
来共享许多FBOS之间的深度模板缓冲区。
如果要在启用模板测试的情况下渲染到默认帧缓冲区,则必须先使用所需的模板渲染缓冲区附件渲染到FBO。然后使用该FBO的颜色纹理绘制到默认的帧缓冲区。