• 设为首页
  • 点击收藏
  • 手机版
    手机扫一扫访问
    迪恩网络手机版
  • 关注官方公众号
    微信扫一扫关注
    公众号

ios - OpenGL 背景清晰颜色渗入透明纹理

[复制链接]
菜鸟教程小白 发表于 2022-12-13 02:56:55 | 显示全部楼层 |阅读模式 打印 上一主题 下一主题

我正在使用 OpenGL 和 C++ 编写一个简单的 2D 框架,现在遇到了与透明纹理和混合相关的问题。我已将我的问题简化为以下内容。

我有两个纹理:地砖和鱼骨。后者包含透明像素。我将我的 OpenGL 清晰颜色设置为“透明绿色”并启用深度测试和混合,如下所示:

glClearColor(0, 1, 0, 0)
glEnable(GL_DEPTH_TEST);
glEnable(GL_BLEND);
glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA);

通过两次调用 glDrawElements(),我画了鱼,然后画了地板。鱼的 Z 值较高,因此应放置在地砖的前面。 This is the result .

显然,我不希望鱼周围出现这样的绿色框。我认为发生的事情是鱼像素在绘制时与帧缓冲区的颜色渲染缓冲区中的任何内容混合,并且恰好是纯绿色(由于 glClearColorglClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT))。对于鱼纹理中的每个透明像素,我希望这种混合能够解析为透明(因为我设置了透明的清晰颜色),但正如您所看到的那样,实际情况并非如此。

如果我先画地板,然后画鱼,就可以了 as expected .

我的片段着色器非常简单:

varying lowp vec2 TexCoordOut;
uniform sampler2D Texture;

void main(void) {
    gl_FragColor = texture2D(Texture, TexCoordOut);
}

我真的必须手动管理绘图顺序(按 Z 坐标排序)还是有办法解决这个问题? OpenGL 深度缓冲系统不是为解决这个问题而设计的吗?

我正在通过 Xcode 在 iOS 模拟器上测试我的程序。



Best Answer-推荐答案


我确实相信深度排序是必要的——因为即使混合产生的像素是透明的,深度缓冲区也会被写入。因此,任何要绘制在这些透明像素后面(更高深度)的东西都会被简单地丢弃。不幸的是,深度缓冲区没有考虑透明度。

如果您的纹理中没有任何半透明像素,您可以使用 alpha 测试,如果像素是透明的,它会丢弃该像素,因此它不会写入深度缓冲区。这显然不适用于半透明像素,因为它们要么变得完全不透明,要么根据您的实现/设置被丢弃。您可以通过将着色器更改为以下方式轻松实现它:

varying lowp vec2 TexCoordOut;
uniform sampler2D Texture;

void main(void) {
   vec4 tc = texture2D(Texture, TexCoordOut);
   if (tc.a < 0.5) //for example, change to any value suitable
     discard;
   gl_FragColor = tc;
}

除此之外,除了排序之外,我没有其他方法可以通过深度缓冲修复透明度。

您可以在此处阅读有关透明度排序和 alpha 测试的更多详细信息:http://www.opengl.org/wiki/Transparency_Sorting

关于ios - OpenGL 背景清晰颜色渗入透明纹理,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/17390071/

回复

使用道具 举报

懒得打字嘛,点击右侧快捷回复 【右侧内容,后台自定义】
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

关注0

粉丝2

帖子830918

发布主题
阅读排行 更多
广告位

扫描微信二维码

查看手机版网站

随时了解更新最新资讯

139-2527-9053

在线客服(服务时间 9:00~18:00)

在线QQ客服
地址:深圳市南山区西丽大学城创智工业园
电邮:jeky_zhao#qq.com
移动电话:139-2527-9053

Powered by 互联科技 X3.4© 2001-2213 极客世界.|Sitemap