c++ - 在 OpenGL 中以 2x 像素旋转像素艺术

标签 c++ opengl

我有一款使用像素艺术风格的游戏,经过放大,1 像素等于屏幕上的 2x2 区域。然而,当我在 OpenGl 中旋转 Sprite 时,它会使用屏幕像素来绘制它,因此它打破了低分辨率风格游戏的错觉:

Rotated sprite

如何旋转 Sprite 并使用较大的像素绘制它?现在, Sprite 是使用 Sprite 类和 Draw 方法来绘制的,代码如下所示:

void Sprite::Draw(int x, int y, int w, int h, int tx = 0, int ty = 0, int tw = 1, int th = 1, int rotation = 0, int rx = 0, int ry = 0, int sr = 0, float r = 1, float g = 1, float b = 1) {
glEnable(GL_TEXTURE_2D);

glTranslatef(x+(w/2), y+(h/2), 0);
glRotatef((float) sr, 0.0f, 0.0f, 1.0f);
glTranslatef(-x-(w/2), -y-(h/2), 0);

glTranslatef(x+(w/2)+rx, y+(h/2)+ry, 0);
glRotatef((float) rotation, 0.0f, 0.0f, 1.0f);
glTranslatef(-(w/2)-rx, -(h/2)-ry, 0);
glColor3f(r, g, b);
glBindTexture(GL_TEXTURE_2D, texture);

const float verts[] = {
    0, (float) h,
    (float) w, (float) h,
    0, 0,
    (float) w, 0
};

const float tVerts[] = {
    (float)tx/(float)width, ((float)ty+(float)th)/(float)height,
    ((float)tx+(float)tw)/(float)width, ((float)ty+(float)th)/(float)height,
    (float)tx/(float)width, (float)ty/(float)height,
    ((float)tx+(float)tw)/(float)width, (float)ty/(float)height
};

glEnableClientState(GL_VERTEX_ARRAY);
glEnableClientState(GL_TEXTURE_COORD_ARRAY);

glVertexPointer(2, GL_FLOAT, 0, verts);
glTexCoordPointer(2, GL_FLOAT, 0, tVerts);

glDrawArrays(GL_TRIANGLE_STRIP, 0, 4);

glLoadIdentity();

glDisableClientState(GL_VERTEX_ARRAY);
glDisableClientState(GL_VERTEX_ARRAY);
glDisable(GL_TEXTURE_2D);
}

谢谢!

编辑:我想我应该提到我正在使用 SDL2 进行窗口管理。

最佳答案

您可以以一半的分辨率渲染整个场景,然后将其放大。为此,您可以使用 FBO(帧缓冲区对象)进行主要渲染,然后将其 blit 到默认帧缓冲区。

在设置过程中,创建 FBO 和渲染目标,并将渲染目标附加到 FBO:

GLuint fboId = 0;
glGenFramebuffers(1, &fboId);

GLuint rbId = 0;
glGenRenderbuffers(1, &rbId);
// Need to bind this once so object is created.
glBindRenderbuffer(GL_RENDERBUFFER, rbId);
glBindRenderbuffer(GL_RENDERBUFFER, 0);

glBindFramebuffer(GL_DRAW_FRAMEBUFFER, fboId);
glFramebufferRenderbuffer(GL_DRAW_FRAMEBUFFER, GL_COLOR_ATTACHMENT0, GL_RENDERBUFFER, rbId);
glBindFramebuffer(GL_DRAW_FRAMEBUFFER, 0);

在每个窗口大小调整时,窗口大小调整为 width/height ,以窗口大小的一半分配渲染目标。如果您的窗口不可调整大小,您可以将其与设置代码结合起来:

glBindRenderbuffer(GL_RENDERBUFFER, rbId);
glRenderbufferStorage(GL_RENDERBUFFER, GL_RGBA8, width / 2, height / 2);
glBindRenderbuffer(GL_RENDERBUFFER, 0);

在每次重绘时,渲染到 FBO,并将结果传输到默认帧缓冲区:

glBindFramebuffer(GL_DRAW_FRAMEBUFFER, fboId);
glViewport(0, 0, width / 2, height / 2);

// draw content

glBindFramebuffer(GL_DRAW_FRAMEBUFFER, 0);
glBindFramebuffer(GL_READ_FRAMEBUFFER, fboId);
glBlitFramebuffer(0, 0, width / 2, height / 2,
                  0, 0, width, height,
                  GL_COLOR_BUFFER_BIT, GL_NEAREST);

glBindFramebuffer(GL_READ_FRAMEBUFFER, 0);

以上假设 widthheight是偶数。如果它们可能很奇怪,您可能需要稍微调整一下。

关于c++ - 在 OpenGL 中以 2x 像素旋转像素艺术,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/26573850/

相关文章:

c++ - 动态数组与静态数组的大小

opengl - 强制内置到其 GL 等效项

c++ - 在 XCode 5 上编译 g++?

c++ - OpenGL:切换多边形模式导致段错误

java - 全屏时视口(viewport)似乎稍微偏离屏幕

c++ - 如何简化代码,将switch合并为一个函数?

c++ - 将shared_ptr<void>转换为shared_ptr<type>时智能指针的行为

c++ - headless QT4 OpenGL 应用程序

c++ - C++11 中的递归 lambda 实现

c++ - 循环接受第一组输入,然后重用该值