如何使OpenGL绘图在pygame中做非显示表面?

时间:2018-12-12 17:50:51

标签: python opengl pygame pyopengl

我正在尝试制作一款看起来像复古太空射击游戏的游戏,其中的线条是3d线框。 为了在Python中实现3D,我将pygame用于窗口,并使用PyOpenGL渲染对象。

我想让OpenGL渲染到一个表面(而不是显示表面),然后通过pygame放大该表面并将其渲染到显示表面上。希望这会带来低分辨率窗口的效果,同时适合在现代屏幕上工作。

阻止我执行此操作的是OpenGL渲染到显示表面,而我找不到任何允许我更改其绘制表面的选项。

因此过程应该是:OpenGL渲染到较小的表面,pygame缩放表面并将其绘制到显示屏上,重复。

这是我当前的代码:

def main():
    pygame.init()
    display = (500,500)
    pygame.display.set_mode(display, DOUBLEBUF|OPENGL) # Create display window

    gluPerspective(70, (display[0]/display[1]), 0.1, 50.0) # Setup view

    glTranslatef(0.0,0.0, -5) # Set view position

    while True:
        for event in pygame.event.get():
            if event.type == pygame.QUIT: # If X is clicked
                pygame.quit() # Close
                quit()
        glRotatef(1, 3, 1, 1) # Rotates view
        glClear(GL_COLOR_BUFFER_BIT|GL_DEPTH_BUFFER_BIT) # Clears the screen
        Cube() # Renders the cube onto the screen
        pygame.display.flip() # Updates the display
        pygame.time.wait(10)

main()

我尝试用与显示器完全相同的设置创建一个曲面,但是OpenGL STILL不会将其渲染到该曲面上。

1 个答案:

答案 0 :(得分:2)

您必须创建一个Framebuffer Object(在主循环之前),其结果小于窗口分辨率。另请参见Framebuffer Object Extension Examples

fb_size = [50, 50]

depth_buffer_obj = glGenRenderbuffers(1)
glBindRenderbuffer(GL_RENDERBUFFER, depth_buffer_obj)
glRenderbufferStorage(GL_RENDERBUFFER, GL_DEPTH_COMPONENT24, fb_size[0], fb_size[1])

color_buffer_obj = glGenRenderbuffers(1)
glBindRenderbuffer(GL_RENDERBUFFER, color_buffer_obj)
glRenderbufferStorage(GL_RENDERBUFFER, GL_RGBA, fb_size[0], fb_size[1])

fb_obj = glGenFramebuffers(1)
glBindFramebuffer(GL_FRAMEBUFFER, fb_obj)
glFramebufferRenderbuffer(GL_FRAMEBUFFER, GL_DEPTH_ATTACHMENT, GL_RENDERBUFFER, depth_buffer_obj)
glFramebufferRenderbuffer(GL_FRAMEBUFFER, GL_COLOR_ATTACHMENT0, GL_RENDERBUFFER, color_buffer_obj)

status = glCheckFramebufferStatus(GL_FRAMEBUFFER)
if status != GL_FRAMEBUFFER_COMPLETE:
    print("incomplete framebuffer object")
glBindFramebuffer(GL_FRAMEBUFFER, 0)

将视口的大小调整为帧缓冲区的大小,清除帧缓冲区,然后将多维数据集呈现为Framebuffer

while True:
    for event in pygame.event.get():
        if event.type == pygame.QUIT:
            pygame.quit() 
            quit()

    glBindFramebuffer(GL_FRAMEBUFFER, fb_obj)
    glViewport (0, 0, fb_size[0], fb_size[1])
    glClear(GL_COLOR_BUFFER_BIT|GL_DEPTH_BUFFER_BIT)

    glRotatef(1, 3, 1, 1)
    Cube()

将视口大小设置为窗口的大小,并将glBlitFramebuffer与过滤器参数 GL_NEAREST 一起使用,将像素从命名的帧缓冲区对象复制到默认帧缓冲区。注意,不必清除默认的帧缓冲区,因为它已被完全覆盖:

while True:

    # .....

    glBindFramebuffer(GL_READ_FRAMEBUFFER, fb_obj)
    glBindFramebuffer(GL_DRAW_FRAMEBUFFER, 0)
    glViewport(0, 0, 500, 500)

    glBlitFramebuffer(
        0, 0, fb_size[0], fb_size[1],
        0, 0, 500, 500,
        GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT,
        GL_NEAREST)

    pygame.display.flip()
    pygame.time.wait(10)

请注意,glBindFramebuffer(GL_READ_FRAMEBUFFER, fb_obj)行不是必需的,因为此时fb_obj势必可以读取和绘制。


如果您的系统不支持glBlitFramebuffer,则可以创建一个帧缓冲区,并在其颜色平面上附加一个纹理:

fb_size = [50, 50]

depth_buffer_obj = glGenRenderbuffers(1)
glBindRenderbuffer(GL_RENDERBUFFER, depth_buffer_obj)
glRenderbufferStorage(GL_RENDERBUFFER, GL_DEPTH_COMPONENT24, fb_size[0], fb_size[1])

#color_buffer_obj = glGenRenderbuffers(1)
#glBindRenderbuffer(GL_RENDERBUFFER, color_buffer_obj)
#glRenderbufferStorage(GL_RENDERBUFFER, GL_RGBA, fb_size[0], fb_size[1])

color_tex_obj = glGenTextures(1)
glBindTexture(GL_TEXTURE_2D, color_tex_obj)
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, GL_REPEAT)
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, GL_REPEAT)
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_NEAREST)
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_NEAREST)
glTexImage2D(GL_TEXTURE_2D, 0, GL_RGBA, fb_size[0], fb_size[1], 0, GL_RGBA, GL_UNSIGNED_BYTE, None)

fb_obj = glGenFramebuffers(1)
glBindFramebuffer(GL_FRAMEBUFFER, fb_obj)
glFramebufferRenderbuffer(GL_FRAMEBUFFER, GL_DEPTH_ATTACHMENT, GL_RENDERBUFFER, depth_buffer_obj)
#glFramebufferRenderbuffer(GL_FRAMEBUFFER, GL_COLOR_ATTACHMENT0, GL_RENDERBUFFER, color_buffer_obj)
glFramebufferTexture2D(GL_FRAMEBUFFER, GL_COLOR_ATTACHMENT0, GL_TEXTURE_2D, color_tex_obj, 0)

status = glCheckFramebufferStatus(GL_FRAMEBUFFER)
if status != GL_FRAMEBUFFER_COMPLETE:
    print("incomplete framebuffer object")
glBindFramebuffer(GL_FRAMEBUFFER, 0)
glBindTexture(GL_TEXTURE_2D, 0) 

渲染到帧缓冲区,并在整个窗口上将带有纹理的四边形绘制为默认帧缓冲区:

while True:
    for event in pygame.event.get():
        if event.type == pygame.QUIT:
            pygame.quit() 
            quit()

    glBindFramebuffer(GL_FRAMEBUFFER, fb_obj)
    glViewport (0, 0, fb_size[0], fb_size[1])
    glClear(GL_COLOR_BUFFER_BIT|GL_DEPTH_BUFFER_BIT)

    glRotatef(1, 3, 1, 1)
    gluSphere(gluNewQuadric( ), 2.0, 32, 32)

    glBindFramebuffer(GL_FRAMEBUFFER, 0)
    glViewport(0, 0, 500, 500)
    glClear(GL_COLOR_BUFFER_BIT|GL_DEPTH_BUFFER_BIT)

    glMatrixMode(GL_PROJECTION)
    glPushMatrix()
    glLoadIdentity()
    glMatrixMode(GL_MODELVIEW)
    glPushMatrix()
    glLoadIdentity()

    #glBlitFramebuffer(
    #    0, 0, fb_size[0], fb_size[1],
    #    0, 0, 500, 500,
    #    GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT,
    #    GL_NEAREST)

    glEnable(GL_TEXTURE_2D)
    glBindTexture(GL_TEXTURE_2D, color_tex_obj)
    glBegin(GL_TRIANGLE_FAN)
    glTexCoord2f(0,0)
    glVertex2f(-1,-1)
    glTexCoord2f(1,0)
    glVertex2f(1,-1)
    glTexCoord2f(1,1)
    glVertex2f(1,1)
    glTexCoord2f(0,1)
    glVertex2f(-1,1)
    glEnd()
    glDisable(GL_TEXTURE_2D)

    glMatrixMode(GL_PROJECTION)
    glPopMatrix()
    glMatrixMode(GL_MODELVIEW)
    glPopMatrix()

    pygame.display.flip()
    pygame.time.wait(10)