OpenGL深度缓冲器搞砸了

时间:2014-03-19 00:53:45

标签: java opengl lwjgl

我正在开发一个OpenGL应用程序。正在渲染的地形似乎不遵循深度缓冲规则,所以我猜我在某处非常糟糕

我的代码中有GL11.glEnable(GL11.GL_CULL_FACE)

这是飞机在没有四边形的情况下的样子,所以即使没有zbuffer,事情看起来也很正常。请注意,这是一个“自上而下”的视图,其中“相机”位于飞机上方

enter image description here

enter image description here

现在,从不同的角度来看

enter image description here

离场景最远的四边形正在靠近计算机的四边形上呈现。这是我对glOrtho的看法,指定近距离和远距离剪裁平面。具体来说,当应该被上升平面隐藏时,最远的平面是可见的

GL11.glOrtho(0, 1000, 0, 750, 50, -50);

有趣的是,如果我删除了对GL11.glEnable(GL_DEPTH_TEST)的调用,那么zbuffer似乎只能用于几个角度

编辑:我认为这个问题与我调用glOrtho的地方有关。它只被调用一次,与我的glEnable一起在代码的开头。我应该在我的循环中重复调用它,在GL11.glLoadIdentity()之后(在循环期间反复调用)?

编辑2:添加代码以显示渲染

public void render() {
    for (int y = 0; y <  mapHeight - 1; y++) {
        for (int x = 0; x < mapWidth - 1; x++) {
            drawTile(x, y, rawMap[y][x], rawMap[y][x + 1], rawMap[y + 1][x + 1], rawMap[y + 1][x]);
        }
    }

}

private void drawTile(double xPos, double zPos, double height1, double height2, double height3, double height4) {
    GL11.glColor3ub((byte) 255, (byte) 255, (byte) 255);
    GL11.glBindTexture(GL11.GL_TEXTURE_2D, textureId);
    GL11.glBegin(GL11.GL_QUADS);
    Point3D normal = faceNormals[(int) zPos][(int) xPos];
    GL11.glNormal3d(normal.getX(), normal.getY(), normal.getZ());
    double xIn = xPos * 0.15;
    double xOut = xIn + 0.15;
    double zIn = zPos * 0.15;
    double zOut = zIn + 0.15;


    /* Original First */
    GL11.glTexCoord2d(xPos * textureFactorX, zPos * textureFactorY);
    GL11.glVertex3d(xIn, height1, zIn);

    GL11.glTexCoord2d(xPos * textureFactorX, (zPos + 1) * textureFactorY);
    GL11.glVertex3d(xIn, height4, zOut);

    GL11.glTexCoord2d((xPos + 1) * textureFactorX, (zPos + 1) * textureFactorY);
    GL11.glVertex3d(xOut, height3, zOut);

    GL11.glTexCoord2d((xPos + 1) * textureFactorY, zPos * textureFactorY);
    GL11.glVertex3d(xOut, height2, zIn);

    GL11.glEnd();
}

以下是设置所有glEnable和常量值的代码

public static void main(String[] args) {
    try {
        Display.setDisplayMode(new DisplayMode(1000, 750));
        Display.create();
        Display.setResizable(true);
    } catch (Exception ex) {
        ex.printStackTrace();
    }
    GL11.glMatrixMode(GL11.GL_PROJECTION);

    // GL11.glEnable(GL11.GL_CULL_FACE);
    GL11.glCullFace(GL11.GL_FRONT_AND_BACK);
    GL11.glEnable(GL11.GL_TEXTURE_2D);
    GL11.glEnable(GL11.GL_DEPTH_TEST);
    GL11.glEnable(GL11.GL_LIGHTING);
    GL11.glEnable(GL11.GL_LIGHT0);

    // Enable lighting
    FloatBuffer lightFloatBuffer = (FloatBuffer) BufferUtils.createFloatBuffer(4).put(5f).put(5f).put(1f).put(0f).flip();
    FloatBuffer ambientFloatBuffer = (FloatBuffer) BufferUtils.createFloatBuffer(4).put(2f).put(2f).put(2f).put(2f).flip();
    FloatBuffer diffuseFloatBuffer = (FloatBuffer) BufferUtils.createFloatBuffer(4).put(1f).put(1f).put(1f).put(1f).flip();
    FloatBuffer specularFloatBuffer = (FloatBuffer) BufferUtils.createFloatBuffer(4).put(1f).put(1f).put(1f).put(1f).flip();

    GL11.glLightModeli(GL11.GL_LIGHT_MODEL_LOCAL_VIEWER, GL11.GL_TRUE);
    GL11.glShadeModel(GL11.GL_FLAT);

    GL11.glLight(GL11.GL_LIGHT0, GL11.GL_AMBIENT, ambientFloatBuffer);
    GL11.glLight(GL11.GL_LIGHT0, GL11.GL_DIFFUSE, diffuseFloatBuffer);
    GL11.glLight(GL11.GL_LIGHT0, GL11.GL_SPECULAR, specularFloatBuffer);
    GL11.glLight(GL11.GL_LIGHT0, GL11.GL_POSITION, lightFloatBuffer);

    System.out.println("Resource Directory: " + RESOURCE_DIR.toString());

    new Game().start();
    Display.destroy();
}

1 个答案:

答案 0 :(得分:1)

注意:经过一些讨论后,使用glDepthFunc(GL_GREATER);并在清除深度缓冲区之前调用glClearDepth(0.0f);

似乎正在发生的只是你正在剔除物体的背面。对于这些平面飞机,如果您希望从后面看到它们,则必须完全禁用面部剔除。就我所见,没有任何东西可以在另一方之上绘制。

编辑:我确实看到你之前关于面部剔除的帖子。这看起来像是面部剔除的正确结果;你只是从被剔除的那一面看你画的脸。