【发布时间】:2014-03-14 19:52:11
【问题描述】:
我正在尝试以与另一个纹理不同的比例缩放在 FBO 中生成的纹理,并在片段着色器中同时渲染这两个纹理,以便我可以将它们混合在一起。
我按照this tutorial 为我的游戏制作了一个光照贴图,我得到了一个完美运行的光照贴图,但我想在着色器中转换光照贴图,以便背景中的视差对象接收按比例缩放的光照贴图,以便它们的到光源的“3D”距离似乎不会影响它们的照明,但它们的 2D 距离会。
对于如何做到这一点,我最好的猜测是根据视差对象的 Z 距离在适当的位置缩放光照贴图。我游戏中的 Z 距离与物体到相机的 2D 距离成正比。
以下示例描述了常规光照贴图发生的情况以及所需的效果。此处的光照贴图由黄色径向渐变组成,该渐变位于右侧的黄色非视差“太阳”:
Description of lightmap scaling
Animated example of current lighting
Animated example of intended lighting
通常在我的游戏中,我通过在 LibGDX 中通过类似这样的方式缩放连接到 SpriteBatch 的相机来缩放视差对象:
Array<Something> objects = new Array<Something>();
SpriteBatch batch = new SpriteBatch();
OrthographicCamera camera = new OrthographicCamera(screenWidth, screenHeight);
//{start loop, add objects, run logic, etc.}
...
for(int i=0;i<objects.size;i++){
camera.zoom = objects.get(i).z;
camera.update();
batch.setProjectionMatrix(camera.combined);
objects.get(i).sprite.draw(batch);
}
根据我的研究,应该有一种方法可以在顶点着色器中进行此类转换。通过默认的 LibGDX 顶点着色器可以直接进行这样的转换,该着色器设置为将顶点位置与相机的转换矩阵相乘,如下所示:
void main() {
v_color = a_color;
v_texCoords = a_texCoord0;
gl_Position = u_projTrans * a_position;
}
但我不确定当变换绑定到共享顶点位置时,如何仅缩放片段着色器中使用的两个纹理之一。作为参考,这里是教程中实现的片段着色器:
void main() {
vec4 diffuseColor = texture2D(u_texture, v_texCoords);
vec2 lightCoord = (gl_FragCoord.xy / resolution.xy);
vec4 light = texture2D(u_lightmap, lightCoord);
vec3 ambient = ambientColor.rgb * ambientColor.a;
vec3 intensity = ambient + light.rgb;
vec3 finalColor = diffuseColor.rgb * intensity;
gl_FragColor = v_color * vec4(finalColor, diffuseColor.a);
}
我认为这与转换纹理坐标而不是位置并将它们传递给光照贴图而不是使用屏幕分辨率来获取纹理坐标有关,但是我这样做的尝试最终变得一团糟,因为我对 OpenGL ES 2 或 GLSL 不是很了解或经验丰富。
有没有一种很好的方法可以在两个纹理片段着色器中缩放一个纹理而不是另一个?或者有没有更好的方法来实现我想要用光照贴图做的事情?
编辑:
所以我可以通过为每个视差对象以不同比例重新创建 FBO 光照贴图来完成我想要做的事情,但这会给我的目标平台之一(Android)带来巨大的性能问题。
我发现 some links 表明我正在尝试做的事情是可能的,但它们都描述了将两个不同的纹理坐标传递给顶点着色器,这似乎不适用于 SpriteBatch。是否可以在顶点着色器中转换纹理坐标,以便将光照贴图的转换坐标传递给片段着色器?还是我误解了纹理坐标的工作原理?
【问题讨论】:
标签: java opengl-es-2.0 libgdx