c++ - 使用 Assimp 和 OpenGL 加载和绘制 .obj 模型时遇到问题

标签 c++ opengl 3d assimp

在完成了 OpenGL 的基础知识(创建窗口、制作 2D 三角形、着色器等)之后,我决定开始尝试加载简单的 .obj 模型。最推荐的库是Assimp所以我关注了some tutorials并修改了我的项目以加载模型。但不幸的是,模型的展示非常奇怪。我创建了以下代码来展示这一点:

#include <iostream>
#include <string>
#include <vector>
#include <memory>
#include <GL/glew.h>
#include <GLFW/glfw3.h>
#include <glm/glm.hpp>
#include <assimp/Importer.hpp>
#include <assimp/scene.h>
#include <assimp/postprocess.h>

struct Vertex
{
    glm::vec3 position;
    glm::vec3 normal;
};

struct Mesh
{
    //The vertex array object, vertex buffer object and element buffer object
    GLuint VAO;
    GLuint VBO;
    GLuint EBO;
    //Vectors for the vertices and indices to put in the buffers
    std::vector<Vertex> vertices;
    std::vector<GLuint> indices;

    //Constructor
    Mesh(const std::vector<Vertex>& vertices, const std::vector<GLuint>& indices)
    {
        this->vertices = vertices;
        this->indices  = indices;

        //Generate the VAO
        glGenVertexArrays(1, &VAO);

        //Generate the buffer objects
        glGenBuffers(1, &VBO);
        glGenBuffers(1, &EBO);

        //Bind the VAO
        glBindVertexArray(VAO);

        //Bind the VBO and set the vertices
        glBindBuffer(GL_ARRAY_BUFFER, VBO);
        glBufferData(GL_ARRAY_BUFFER, sizeof(Vertex) * vertices.size(), &vertices.at(0), GL_STATIC_DRAW);

        //Enable the first attribute pointer
        glEnableVertexAttribArray(0);
        //Set the attribute pointer    The stride is meant to be 'sizeof(Vertex)', but it doesn't work at all that way
        //                                              \/
        glVertexAttribPointer(0, 3, GL_FLOAT, GL_FALSE, 0, 0);

        //Enable the second attribute pointer
        glEnableVertexAttribArray(1);
        //Set the attribute pointer                   ditto
        //                                              \/
        glVertexAttribPointer(1, 3, GL_FLOAT, GL_FALSE, 0, (void*) offsetof(Vertex, normal));

        //Bind the EBO and set the indices
        glBindBuffer(GL_ELEMENT_ARRAY_BUFFER, EBO);
        glBufferData(GL_ELEMENT_ARRAY_BUFFER, sizeof(GLuint) * indices.size(), &indices.at(0), GL_STATIC_DRAW);

        //Report any errors
        GLenum error = glGetError();
        if (error != GL_NO_ERROR)
        {
            std::cerr << "Error while creating mesh!" << std::endl;
        }

        glBindVertexArray(0);
    }

    void draw()
    {
        //Bind the VAO
        glBindVertexArray(VAO);

        //Bind the ELement Buffer Object
        glBindBuffer(GL_ELEMENT_ARRAY_BUFFER, EBO);

        //Draw the mesh
        glDrawElements(GL_TRIANGLES, indices.size(), GL_UNSIGNED_INT, 0);

        //Unbind the VAO
        glBindVertexArray(0);
    }
};

int main()
{
    //Intialize GLFW (no error checking for brevity)
    glfwInit();

    //Set the OpenGL version to 3.3
    glfwWindowHint(GLFW_CONTEXT_VERSION_MAJOR, 3);
    glfwWindowHint(GLFW_CONTEXT_VERSION_MINOR, 3);
    //Create a new window
    GLFWwindow* window = glfwCreateWindow(800, 600, "Model Testing", NULL, NULL);

    glfwMakeContextCurrent(window);

    //Initialize glew (no checking again)
    glewInit();

    glViewport(0, 0, 800, 600);
    glClearColor(1.0f, 1.0f, 1.0f, 1.0f);

    //Load the model
    Assimp::Importer importer;
    const aiScene* scene = importer.ReadFile("mymodel.obj", aiProcess_Triangulate | aiProcess_GenNormals);

    //Check for errors
    if ((!scene) || (scene->mFlags == AI_SCENE_FLAGS_INCOMPLETE) || (!scene->mRootNode))
    {
        std::cerr << "Error loading mymodel.obj: " << std::string(importer.GetErrorString()) << std::endl;
        //Return fail
        return -1;
    }

    //A vector to store the meshes
    std::vector<std::unique_ptr<Mesh> > meshes;
    //Iterate over the meshes
    for (unsigned int i = 0; i < scene->mNumMeshes; ++i)
    {
        //Get the mesh
        aiMesh* mesh = scene->mMeshes[i];

        //Create vectors for the vertices and indices
        std::vector<Vertex> vertices;
        std::vector<GLuint> indices;

        //Iterate over the vertices of the mesh
        for (unsigned int j = 0; j < mesh->mNumVertices; ++j)
        {
            //Create a vertex to store the mesh's vertices temporarily
            Vertex tempVertex;

            //Set the positions
            tempVertex.position.x = mesh->mVertices[j].x;
            tempVertex.position.y = mesh->mVertices[j].y;
            tempVertex.position.z = mesh->mVertices[j].z;

            //Set the normals
            tempVertex.normal.x   = mesh->mNormals[j].x;
            tempVertex.normal.y   = mesh->mNormals[j].y;
            tempVertex.normal.z   = mesh->mNormals[j].z;

            //Add the vertex to the vertices vector
            vertices.push_back(tempVertex);
        }

        //Iterate over the faces of the mesh
        for (unsigned int j = 0; j < mesh->mNumFaces; ++j)
        {
            //Get the face
            aiFace face = mesh->mFaces[j];
            //Add the indices of the face to the vector
            for (unsigned int k = 0; k < face.mNumIndices; ++k) {indices.push_back(face.mIndices[k]);}
        }

        //Create a new mesh and add it to the vector
        meshes.push_back(std::unique_ptr<Mesh>(new Mesh(std::move(vertices), std::move(indices))));
    }

    //While the window shouldn't be closed
    while (!glfwWindowShouldClose(window))
    {
        //Clear the buffer
        glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT);

        //Draw all the meshes
        for (auto& mesh : meshes) {mesh.get()->draw();}

        //Swap the buffers
        glfwSwapBuffers(window);
    }

    //Close the window now that it's not needed anymore
    glfwDestroyWindow(window);

    return 0;
}

当程序加载时this teapot我的屏幕如下所示:

teapot

从更远的角度看(使用比上面更复杂的程序):

teapot from afar

如果它有用,我正在运行带有 Nvidia GTX 750 Ti 的 Ubuntu 16.04,驱动程序版本 361.45

最佳答案

步幅应为sizeof(Vertex)。如果它不能按照这个步幅工作,那么就有其他问题了!

关于c++ - 使用 Assimp 和 OpenGL 加载和绘制 .obj 模型时遇到问题,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/39069591/

相关文章:

c++ - Qt Creator/qmake - 从源目录加载 Assets 文件夹,而不是构建目录

c++ - 如何清除 boost 功能?

python - 线段和三角形之间的 3d 交集

c++ - 计算四边形网格的法线

c++ - 使用 glutBitmapCharacter() 绘制文本关闭 opengl 程序

c++ - 如何知道对象在 3D 中仅旋转了 15 度?

c++ - 使用 wxNotebook 创建选项卡闪烁效果(如 IM 程序)

c++ - CUDA 全局函数中的局部指针数组

OpenGL:如何获取变换后特定点的坐标?

Qt5.6:高DPI支持和OpenGL(OpenSceneGraph)