c++ - 深度测试根本不起作用

标签 c++ opengl rendering depth-buffer

由于某种原因,在我的项目中我的深度测试不起作用。我已经确保它已启用并且不起作用。我知道这一点是因为我可以看到某些面被绘制在彼此之上,并且场景中的不同对象(立方体)被绘制在彼此之上。

我使用的是默认的帧缓冲区,所以应该有深度。我还检查了 gl_FragCoord.z 它返回了正确的深度。我已经彻底检查了我的代码很长时间,搜索了数十个谷歌页面,但我仍然找不到答案。

以下是与此问题相关的按执行顺序排列的代码:

Init()

void Program::Init()
{
    std::cout << "Initialising" << std::endl;
    glViewport(0, 0, Options::width, Options::height);

    glewExperimental = true; // Needed in core profile
    if (glewInit() != GLEW_OK) {
        fprintf(stderr, "Failed to initialize GLEW\n");
        return;
    }

    window.setKeyRepeatEnabled(false);

    sf::Mouse::setPosition(sf::Vector2i(Options::width / 2, Options::height / 2), window);

    LoadGameState(GameState::INGAME, false);
    Run();
}

GLInit()

void Program::GLInit() {

    lampShader = Shader("StandardVertex.shader", "LightingFragmentShader.shader");
    ourShader = Shader("VertexShader.shader", "SimpleFragmentShader.shader");
    screenShader = Shader("FrameVertexShader.shader", "FrameFragmentShader.shader");
    skyboxShader = Shader("SkyboxVertex.shader", "SkyboxFragment.shader");
    cubeDepthShader = Shader("CubeDepthVertex.shader", "CubeDepthFragment.shader", &std::string("CubeDepthGeometry.shader"));
    debugDepthQuad = Shader("SimpleVertex.shader", "DepthFragment.shader");
    blurShader = Shader("SimpleVertex.shader", "BloomFragment.shader");
    bloomShader = Shader("SimpleVertex.shader", "FinalBloom.shader");

    depthShader = Shader("VertexDepth.shader", "EmptyFragment.shader");

    geometryPass = Shader("GeometryVertex.shader", "GeometryFragment.shader");
    lightingPass = Shader("SimpleVertex.shader", "LightingFragment.shader");
    shaderSSAO = Shader("SimpleVertex.shader", "SSAOFragment.shader");
    shaderSSAOblur = Shader("SimpleVertex.shader", "SSAOBlurFragment.shader");

    colliders = Shader("VertexShader.shader", "GreenFragment.shader");


    glEnable(GL_DEPTH_TEST);
}

以及另一个重复运行的函数: RunGame()

void Program::RunGame()
{


    scene->mainCamera.Input(Options::deltaTime, window);

    if(timeSinceGameStart.getElapsedTime().asSeconds()<2)
    DoLights();

    if(timeSinceGameStart.getElapsedTime().asSeconds() > 5.0f)
    scene->DoPhysics();

    scene->CheckCollisions();

    glClearColor(0.32f, 0.5f, 0.58f, 1.0f);
    glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT);

    projection = glm::mat4();
    view = glm::mat4();
    // view/projection transformations
    projection = glm::perspective(glm::radians(Options::fov), (float)Options::width / (float)Options::height, 0.1f, 100.0f);
    view = scene->mainCamera.GetViewMatrix();

    lampShader.use();
    lampShader.setMat4("projection", projection);
    lampShader.setMat4("view", view);
    lampShader.setInt("setting", Options::settings);
    glm::mat4 model;
    for (unsigned int i = 0; i < lights.size(); i++)
    {
        model = glm::mat4();
        model = glm::translate(model, lights[i].position);
        model = glm::scale(model, glm::vec3(0.125f));
        lampShader.setMat4("model", model);
        lampShader.setVec3("lightColor", lights[i].colour);
        Scene::renderCube();
    }


    if (Options::showColliders) {
        colliders.use();
        colliders.setMat4("projection", projection);
        colliders.setMat4("view", view);
        scene->RenderColliders(colliders);
    }

}

我查看了很多页面并完成了所有建议:

  • 调用 glEnable(GL_DEPTH_TEST)
  • 清除深度缓冲区
  • 确保 zNear 不是一个很小的数字
  • 不调用会禁用深度的函数,如 glDepthMask(GL_FALSE)

如有任何帮助,欢迎。我希望这是足够的代码,如果不是,我将提供任何请求的代码。大多数未显示的功能都是不言自明的。另外,如果您认为上面的所有代码都很好,请在评论中告诉我,以便我知道问题不存在。

我显然也在使用 C++ 和 glew。我的窗口也使用 SFML。 感谢您的回答

编辑: 负责创建 sfml 窗口的代码:

Program::Program()
    :settings(24, 8, 4, 3, 0),
    window(sf::VideoMode(Options::width, Options::height), "OpenGL", sf::Style::Default, settings)
{
    Init();
}

灯光顶点着色器:

#version 330 core
layout(location = 0) in vec3 aPos;
layout(location = 1) in vec3 aNormal;
layout(location = 2) in vec2 aTexCoords;

out vec3 Normal;

uniform mat4 projection;
uniform mat4 view;
uniform mat4 model;

void main()
{
    Normal = transpose(inverse(mat3(model))) * aNormal;
    gl_Position = projection * view * model * vec4(aPos, 1.0);
}

窗口和设置的声明

sf::RenderWindow window;
sf::ContextSettings settings;

最佳答案

问题在于,在 C++ 中,类的成员按照其声明的顺序进行初始化,而不是按照它们在任何构造函数的 mem-initialiser 列表中列出的顺序。

如果您在编译器中打开了足够多的警告,它应该警告您,在您显示的 Program 构造函数中,settings 将在 window 之后初始化。这正是您的问题,因为 settings 在获取您为其指定的值之前被使用。交换类中成员的顺序即可解决。


该规则的原因是 C++ 的基本规则是“自动存储持续时间的对象总是以其构造的相反顺序被销毁”。因此,类的析构函数需要一个销毁成员的顺序,无论使用哪个构造函数来创建对象。因此,该顺序固定为声明顺序。请注意,声明顺序还用于控制其他上下文(例如函数局部变量)中的创建/销毁顺序,因此在这里使用它是一致的。

关于c++ - 深度测试根本不起作用,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/48396605/

相关文章:

c++ - 不再满足条件后,while 循环会停止验证吗?

c++ - C++负数中对1位的计数

c++ - 任何为 FUSE 提供常用结构和算法的库

c++ - 去除Wm6.5系统托盘图标

c++ - OpenGL - 无法读取/写入 2D 深度纹理数组

OpenGL:任何顶点 Z 坐标是否必须为负数?

linux - 在 Linux 上静态构建 glew 和 glfw

c - OpenGL 纹理显示为实体对象?

javascript - React.memo - 永远不会调用第二个参数 "areEqual"

html - 从 HTML/CSS 触发 Chrome 中的 GPU 光栅化以实现背景图像动画