使用 Assimp 和 OpenGL 加载和绘制 .obj 模型时遇到问题
Trouble loading and drawing a .obj model using Assimp and OpenGL
完成 OpenGL 的基础知识(创建 window、制作 2D 三角形、着色器等)后,我决定开始尝试加载简单的 .obj
模型。最推荐的库是 Assimp so I followed 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 我的屏幕如下所示:
并且从另一个角度看得更远(使用比上面那个更复杂的程序):
如果它有用,我是 运行 Ubuntu 16.04 和 Nvidia GTX 750 Ti,驱动程序版本 361.45
尝试将 ebo 的绑定移动到网格构造函数中的 vbo 之后,如下所示:
//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);
//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);
这就是他们在您链接的网格页面上的方式。我在使用不同的装载机时遇到了类似的问题。您的索引未正确加载,因此一些顶点定位正确,而另一些则不正确。
我对 OpenGL 的体验很糟糕,所以我可能会弄错。我看到你的顶点被制作成:x,y,z,nx,ny,nz 其中 xyz 是顶点坐标,nxnynz 是法线坐标。因此步幅为 6*sizeof(float)。
glVertexAttribPointer(0, 3, GL_FLOAT, GL_FALSE, 6*sizeof(float), 0);
glVertexAttribPointer(1, 3, GL_FLOAT, GL_FALSE, 6*sizeof(float), (void*) 3*sizeof(float);
只需查看此处的第二个答案:Opengl Vertex attribute stride 了解有关步幅计算的更多信息
如果这不能帮助检查索引是否正确形成
和小建议:使用立方体而不是茶壶(只是在搅拌机中制作立方体或自己在记事本中编写)
步幅应该是sizeof(Vertex)
。如果它对那个步幅不起作用,那么其他地方就错了!
完成 OpenGL 的基础知识(创建 window、制作 2D 三角形、着色器等)后,我决定开始尝试加载简单的 .obj
模型。最推荐的库是 Assimp so I followed 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 我的屏幕如下所示:
并且从另一个角度看得更远(使用比上面那个更复杂的程序):
如果它有用,我是 运行 Ubuntu 16.04 和 Nvidia GTX 750 Ti,驱动程序版本 361.45
尝试将 ebo 的绑定移动到网格构造函数中的 vbo 之后,如下所示:
//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);
//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);
这就是他们在您链接的网格页面上的方式。我在使用不同的装载机时遇到了类似的问题。您的索引未正确加载,因此一些顶点定位正确,而另一些则不正确。
我对 OpenGL 的体验很糟糕,所以我可能会弄错。我看到你的顶点被制作成:x,y,z,nx,ny,nz 其中 xyz 是顶点坐标,nxnynz 是法线坐标。因此步幅为 6*sizeof(float)。
glVertexAttribPointer(0, 3, GL_FLOAT, GL_FALSE, 6*sizeof(float), 0);
glVertexAttribPointer(1, 3, GL_FLOAT, GL_FALSE, 6*sizeof(float), (void*) 3*sizeof(float);
只需查看此处的第二个答案:Opengl Vertex attribute stride 了解有关步幅计算的更多信息
如果这不能帮助检查索引是否正确形成
和小建议:使用立方体而不是茶壶(只是在搅拌机中制作立方体或自己在记事本中编写)
步幅应该是sizeof(Vertex)
。如果它对那个步幅不起作用,那么其他地方就错了!