- iOS/Objective-C 元类和类别
- objective-c - -1001 错误,当 NSURLSession 通过 httpproxy 和/etc/hosts
- java - 使用网络类获取 url 地址
- ios - 推送通知中不播放声音
我正在使用 libgdx 及其 SpriteBatch 类为 Andoid 开发 2D 动态壁纸。这是一 block watch ,所以我需要的只是从 libgdx 绘制多个纹理,将它们旋转到特定角度。
我已经使用 SpriteBatch 类解决了这个问题,一切正常。
但现在我需要在我的场景中添加放大镜效果。任务是以特定比例放大特定区域以模拟真实镜头。
我通过渲染到 FrameBuffer、从 FrameBuffer 获取纹理区域并最终使用 SpriteBatch 使用自定义着色器绘制该区域来解决此问题。
问题:在三星设备上(我在 Galaxy Note N7000 和 Galaxy Tab 10.1 上试过)在放大区域的中心有一个小矩形,大约 16x16 像素,放大率略有增加。抱歉,我现在无法发布屏幕截图,因为我没有三星设备。在其他设备上一切正常,在 HTC Vivid、Acer A500、Google Nexus One 上试过。
我认为,如果三星设备上的 Mali GPU 存在问题,但不知道如何解决。
我已将此问题中的 fragment 着色器代码改编为 SpriteBatch Add Fisheye effect to images at runtime using OpenGL ES这是它:
#ifdef GL_ES
#define LOWP lowp
precision mediump float;
#else
#define LOWP
#endif
varying LOWP vec4 v_color;
varying vec2 v_texCoords;
uniform sampler2D u_texture;
void main() {
vec2 m = vec2(0.622 , 0.4985); // lens center point, note that in live wallpaper center of texture is (0.5,0.5)
float lensSize = 0.045; //diameter of lens
float lensCut = 0.0342; //length of cut lines
vec2 d = v_texCoords - m;
float r = sqrt(dot(d, d)); // distance of pixel from mouse
float cuttop = m.y + lensCut;
float cutbottom = m.y - lensCut;
vec2 uv;
if (r >= lensSize) {
uv = v_texCoords;
} else if ( v_texCoords.y >= cuttop) {
uv = v_texCoords;
} else if (v_texCoords.y <= cutbottom) {
uv = v_texCoords;
} else {
uv = m + normalize(d) * asin(r) / (3.14159 * 0.37);
}
gl_FragColor = texture2D(u_texture, uv);
}
顶点着色器默认来自 SpriteBatch 源:
attribute vec4 a_position;
attribute vec4 a_color;
attribute vec2 a_texCoord0;
uniform mat4 u_projTrans;
varying vec4 v_color;
varying vec2 v_texCoords;
void main() {
v_color = a_color;
v_texCoords = a_texCoord0;
gl_Position = u_projTrans * a_position;
}
这是我的render()
方法:
GL20 gl = Gdx.graphics.getGL20();
gl.glEnable(GL20.GL_TEXTURE_2D);
gl.glActiveTexture(GL20.GL_TEXTURE0);
gl.glClearColor(WallpaperService.bg_red, WallpaperService.bg_green, WallpaperService.bg_blue, 1f);
gl.glClear(GL20.GL_COLOR_BUFFER_BIT);
// Creating framebuffer, if it has gone
if (mFrameBuffer == null) {
mFrameBuffer = new FrameBuffer(Format.RGBA4444, BG_WIDTH, BG_HEIGHT, true);
mFrameBufferRegion = new TextureRegion(mFrameBuffer.getColorBufferTexture());
mFrameBufferRegion.flip(false, true);
}
//Camera setting according to background texture
camera = new OrthographicCamera(BG_WIDTH, BG_HEIGHT);
camera.position.set(BG_WIDTH / 2, BG_WIDTH / 2, 0);
camera.update();
batch.setProjectionMatrix(camera.combined);
//Rendering scene to framebuffer
mFrameBuffer.begin();
batch.begin();
//main Drawing goes here
batch.end();
//Drawing frame to screen with applying shaders for needed effects
if (mFrameBuffer != null) {
mFrameBuffer.end();
camera = new OrthographicCamera(width, height);
camera.position.set(width / 2, height / 2, 0);
camera.update();
batch.setProjectionMatrix(camera.combined);
batch.begin();
batch.setShader(null);
batch.setShader(shader);
batch.draw(mFrameBufferRegion, width / 2 - (BG_WIDTH / 2) + (MARGIN_BG_LEFT * ratio), height / 2
- (BG_HEIGHT / 2) + (MARGIN_BG_TOP * ratio), (float) BG_WIDTH / 2, (float) BG_HEIGHT / 2,
(float) BG_WIDTH, (float) BG_HEIGHT, (float) ratio, (float) ratio, 0f);
batch.setShader(null);
batch.end();
}
最佳答案
我已经设法解决了这个问题。问题出在 Mali gpu 中。Mali 不支持 fragment 着色器中的高精度浮点计算。当中心到点的距离接近于零时 - r 变量变为零。
所以我在计算中添加了常数系数,这就成功了。
这是工作 fragment 着色器:
precision mediump float;
varying lowp vec4 v_color;
varying vec2 v_texCoords;
uniform sampler2D u_texture;
const float PI = 3.14159;
const float koef = 10.0;
void main() {
vec2 uv;
vec2 m = vec2(0.622 , 0.4985); // lens center point, note that in live wallpaper center of texture is (0.5,0.5)
float lensSize = 0.045; //radius of lens
float lensCut = 0.0342; //height of cut
float cuttop = m.y + lensCut;
float cutbottom = m.y - lensCut;
float cutleft = m.x-lensSize;
float cutright = m.x+lensSize;
//don't transform pixels, that aren't in lens area
if ( v_texCoords.y >= cuttop) {
uv = v_texCoords;
} else if (v_texCoords.y <= cutbottom) {
uv = v_texCoords;
} else if (v_texCoords.x <= cutleft) {
uv = v_texCoords;
} else if (v_texCoords.x >= cutright) {
uv = v_texCoords;
} else {
vec2 p = v_texCoords*koef; //current point
vec2 d = p - m*koef; //vector differnce between current point and center point
float r = distance(m*koef,p); // distance of pixel from center
if (r/koef >= lensSize) {
uv = v_texCoords;
} else {
uv =m + normalize(d) * asin(r) / (PI*0.37*koef);
}
}
gl_FragColor = texture2D(u_texture, uv);
}
关于android - libgdx:SpriteBatch,Samsung Android 设备上的 fragment 着色器工作不正确,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/14192330/
这是一个关于 XNA 中 SpriteBatches 的性能问题。当你想在 XNA 中开始绘图时,你必须使用 spritebatch.Begin()。这就是你指定一堆东西的方式,比如使用的着色器。这会
SpriteBatch.SetProjectionMatrix(cam.combined) 使 Spritebatch 不绘制 block 和我的角色,但相机移动有效,但如果我不使用这行代码,一切都被
当游戏重置但 spritebatch 没有绘制时,我试图在屏幕上绘制一个快速加载的 Sprite 。我在 game.spritebatch.begin(); 上设置了一个断点,当我逐步执行时,没有绘制
当游戏重置但 spritebatch 没有绘制时,我试图在屏幕上绘制一个快速加载的 Sprite 。我在 game.spritebatch.begin(); 上设置了一个断点,当我逐步执行时,没有绘制
我一直想知道 spriteBatch 中的变换矩阵是如何实现的。我创建了一个2D相机,变换矩阵如下: if (needUpdate) transform
但是,我有一个奇怪的问题,在绘制时,图像的外部 1px 似乎被拉伸(stretch)以适合矩形,但内部仅拉伸(stretch)到一定范围,我正在绘制 48x48 的图 block ,但绘制了 500x
我想要做的是能够将变换矩阵推送和弹出到 SpriteBatch 中。我有 2 个 Sprite ,父级和子级,子级应该相对于父级进行缩放、旋转、平移。 我目前有一个使用纹理四边形的实现,但我认为使用内
我正在使用 LibGDX 开发一款自上而下的 RPG 游戏,并正在为我的游戏创建 Ortho.. 相机。然而,这样做时,现在只有我的图 block 纹理会渲染。渲染代码如下所示: 相机初始化为new
标题说明了一切。 使用同一个 SpriteBatcher 对象调用多个 Stage 的 draw() 方法是否仅发生在单个批处理中,或者它是一个 Stage =一批? 我有一种情况,我不想使用 Act
我想在单击按钮时旋转此 SpriteBatch 本身 @Override public void render() { SpriteBatch batch = new SpriteBatch(
我是 xna 编程的新手,我需要一些帮助... 我构建了一个静态背景的 2D 游戏在我当前的屏幕中有背景图像,左侧有 4 个大图像,我需要在右侧绘制 16 个小图像(50x50 或多或少) 所以这是我
使用 C# 和 XNA Framework,在调用 SpriteBatch.DrawString 方法时,“position”和“origin”参数有什么区别? 最佳答案 原点是与位置相关的偏移量。
SpriteBatch batcher = new SpriteBatch(); batcher.draw(TextureRegion region, float x,
我有一个 2d Sprite 数组,其中包含大约。 50 个 Sprite 。我想在渲染方法中绘制它们。我不知道什么会给我最好的表现。这是我想出的两种选择: batch.begin(); for(Sp
所以我一直在浏览和试验一些东西,但不太明白如何删除屏幕上预先存在的 spriteBatch。 基本上我发起了 batcher.begin(); (blah blah blah) AssetLoader
我想知道是否可以在纹理中保存 spriteBatch。 SpriteBatch batch = new SpriteBatch(); 在批处理中绘制了一些东西之后,我想将所有包含 SpriteBat
我已经实现了一个 OpenGL spritebatch,但是当对多个 sprite 使用相同的纹理时它只绘制第一个 sprite。例如:如果我用它来呈现字符串“Hello”的字形,它只会呈现“H”。
对我来说,这是 LibGDX 中发生过的最奇怪的事情之一。我已经在游戏中的所有其他状态中使用了这些确切的规范,但在不同的名称下,它们都工作正常,除了我的 ShopState,它根本不会渲染任何东西!这
我是 XNA 的新手,想在其基础上开发一个轻量级的 2D 引擎,并将实体组织成父子层次结构。我在画 child 的时候想到了矩阵,因为他们的位置、旋转和缩放都取决于他们的 parent 。 如果我使用
我在这里绘制了一组 Sprite ,在缩小时使用较小的 (4x4) 纹理,在放大时使用较大的 (16x16) 纹理。当缩放级别达到阈值时,就会发生交换。 然而,为了实现平滑过渡, Sprite (位于
我是一名优秀的程序员,十分优秀!