- iOS/Objective-C 元类和类别
- objective-c - -1001 错误,当 NSURLSession 通过 httpproxy 和/etc/hosts
- java - 使用网络类获取 url 地址
- ios - 推送通知中不播放声音
我正在尝试使用与另一个纹理不同的比例缩放在 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;
}
但我不确定当转换绑定(bind)到共享顶点位置时,我如何才能只缩放片段着色器中使用的两个纹理之一。作为引用,这里是从教程中实现的片段着色器:
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。是否可以在顶点着色器中转换纹理坐标,以便我可以将光照贴图的转换坐标传递给片段着色器?还是我误解了纹理坐标的工作原理?
最佳答案
首先,以防万一我假设 Something
代表单个 Sprite 是正确的:每次调用 batch.setProjectionMatrix()
时, Sprite 批处理被刷新,导致另一个绘制调用,如果你有超过几十个 Sprite ,这将太多了。使用将对象数组发送到 sprite 批处理的方法,您应该按 z 位置对它们进行排序,然后仅在自上次调用 batch.draw()
。我个人会为每个视差层保留一个单独的对象数组,以保持简单而不用担心排序。
从那里开始:首先,在绘制您的光照贴图时,将您的游戏支持的最小缩放比例进行绘制,这样您就不必缩小您的光照贴图并冒着显示光照贴图裁剪边缘的风险。因此,在 fbo.begin()
之后,请确保您的 cam.zoom
已更改为场景的最小 cam.zoom
。
要获得缩放的光照贴图坐标,您需要在顶点着色器中计算它并将其传递给片段着色器。首先,您需要输入的缩放级别,因此您需要将其作为制服传入。你可以这样做:
//including the changes explained way above, where objectLayers is an
//Array<LayerWrapper>, where LayerWrapper contains a zoom value and
//an Array<Something>
for(int i=0;i<objectLayers.size;i++){
LayerWrapper layer = objectLayers.get(i);
Array<Something> layerObjects = layer.objects;
camera.zoom = layer.zoom;
camera.update();
batch.setProjectionMatrix(camera.combined);
//update the zoom level in the shader you're using with the sprite batch
customShader.bind();
customShader.setUniformf("u_zoom", layer.zoom);
for(Something object : layerObjects ){
object.sprite.draw(batch);
}
}
然后更新您的顶点着色器以声明 uniform float u_zoom;
和 varying vec2 v_lightCoords;
。您的灯光坐标将根据缩放比例放大。只要我们传递一个变化,我们也应该预先计算精确的纹理坐标,这样你就不必在片段着色器中执行它,这会导致依赖纹理读取(优化可能是在开始的教程中完成)。
顶点着色器现在看起来像这样:
void main()
{
v_color = a_color;
v_texCoords = a_texCoord0;
vec4 screenSpacePosition = u_projTrans * a_position;
gl_Position = screenSpacePosition;
v_lightCoords = (screenSpacePosition.xy * u_zoom) * 0.5 + 0.5;
}
由于 v_lightCoords
已经内置了纹理坐标,您可以简化片段着色器来进行纹理查找,如下所示: vec4 Light = texture2D(u_lightmap, v_lightCoords);
并删除不再需要的 uniform vec2 resolution;
。
关于java - LibGDX - 如何在单个着色器中渲染具有不同比例的两个纹理?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/21796534/
libGDX Array类的javadoc说:一个可调整大小,有序或无序的对象数组。如果是无序的,则此类在删除元素时避免了内存复制(最后一个元素移动到删除的元素的位置)。 去除元素的改进是此类的唯一优
声音API似乎缺少指示声音播放完毕的功能。还有其他方法可以确定声音是否完成了吗? 最佳答案 据我所知,OpenAL和Android的底层声音后端甚至都没有内部跟踪信息,尽管Music API具有的is
我在应用程序启动时执行了此代码 val resolver = InternalFileHandleResolver() asset.setLoader(FreeTypeFontGenerator::c
我想要获取点 1 和点 2 之间相对于中心点的角度。如何使用 Vector2 执行此操作? Vector2 center = new Vector2((float)Gdx.graphics.getWi
我正在尝试渲染平滑的可缩放位图字体。检查后question answers之一提到使用距离场字体。 我正在按照 LibGDX wiki article 中提到的方式进行操作关于距离场字体。但是我无法让
我已经寻找答案大约 2 个小时了,但还没有找到我想要的答案。我的问题是,是否可以以及如何绘制圆形纹理,以便在圆形之外,纹理将是透明的,这甚至可能吗? 提前致谢!到目前为止,该网站提供了很大的帮助! 最
我用 libgdx 开始了一个项目,我有一个 GameStateManager 用于我的 GameStates 菜单和播放。 如果我运行该项目,它会显示菜单,然后我可以单击一个按钮来打开 Play G
我正在使用 libgdx。我需要缩放和定位文本。假设我想绘制 30 像素高的 X,并且我希望它位于屏幕中间。我想在不同的位置和不同的比例画更多的人。 有什么办法可以实现吗?我在任何地方都找不到解决方案
我正在 Libgdx 中制作纹理打包机,其中如果我使用打包机 2,048 * 2,048 的大小,只制作一个大小为 3.14 Mb 的打包机图像,如果我使用打包机的大小 1,024 * 1,024 那
我在每一帧中都进行了大量的三角函数计算。 Java 的 Math 函数是否比 Libgdx 的 MathUtils 更快? 或者我可以使用任何其他比这两个都更快的库吗? 最佳答案 com.badlog
你好,对于 LibGDX 来说有点新,目前在 pc 上的全屏模式存在问题,我想要做的是每当有人按下一个键时将我的游戏设置为全屏,而当我在 main 方法中输入一些东西时,这不会做任何事情桌面启动器.j
几年前我开始开发我的游戏,中间有很大的停顿。那时没有 gradle,只有简单的 java 安装应用程序。如何找到所使用的 LibGDX 版本? 最佳答案 干得好: Gdx.app.log("Gdx v
我正在尝试使用 libGDX 实现一个简单的动画,但目前我遇到了一件事。假设我有一堆 Sprite 需要一些时间才能完成。例如,大约 30 个 Sprite 像这个链接:https://github.
如何在游戏过程中更改窗口标题?我找不到任何方法。例如,我想在标题栏中显示我得了多少分。 最佳答案 尝试这个 : Gdx.graphics.setTitle(""+yourScore); 祝你好运 !
我正在使用 LibGdx 中的声音接口(interface)来播放 mp3 音频文件。 And when choose to loop playing the sound more than one
所以我在 LibGDX 中制作游戏,我使用 AssetManager 加载我的所有 Assets 。 我只是不确定哪种是正确的使用方法。 目前我正在第一个屏幕之前加载所有游戏 Assets 。 然后我
我一直在寻找来自 libgdx SVN 的 skinpacker,但没有成功。 然后我知道它不再存在。所以,问题是如何创建一个 skin.json 文件以在我的 libgdx 项目中使用。你知道有什么
我正在制作连点类型的游戏,我必须在屏幕上触摸的位置画一条线,所以我使用矢量来存储我触摸的各个点。 我用过 if(Gdx.input.isTouched()) { touchpos.set(Gd
我查看了测试项目中的默认皮肤文件。 我不明白为什么 uiskin.png 里面有字体图片。 uiskin.atlas 文件还包含拆分字段,我不明白为什么需要它以及如何使用它。 我在哪里可以找到所有事情
我到处搜索,但没有引用。 我想用this着色器从 shadertoy 到我的 libgdx 项目,所以我尝试首先从以下位置导入简单着色器:https://www.shadertoy.com/view/
我是一名优秀的程序员,十分优秀!