Ошибка Android OpenGL UV mapping после триангуляции - PullRequest
0 голосов
/ 24 октября 2018

Я создаю приложение для Android для рендеринга 3d (wavefront.obj) модели.Используя tinyobjloader , я могу успешно загрузить модель.

Код:

std::vector<glm::vec3> vertices;
std::vector<glm::vec2> uvs;
std::vector<glm::vec3> normals;

tinyobj::attrib_t attrib;
std::vector<tinyobj::shape_t> shapes;
std::vector<tinyobj::material_t> materials;
for(size_t s =0; s < shapes.size(); s++)
    {
        size_t index_offset = 0;
        for(size_t f = 0; f < shapes[s].mesh.num_face_vertices.size(); f++)
        {
            int fv = shapes[s].mesh.num_face_vertices[f];
            for(size_t v = 0; v < fv; v++)
            {
                tinyobj::index_t idx = shapes[s].mesh.indices[index_offset + v];
                tinyobj::real_t vx = attrib.vertices[3*idx.vertex_index+0];
                tinyobj::real_t vy = attrib.vertices[3*idx.vertex_index+1];
                tinyobj::real_t vz = attrib.vertices[3*idx.vertex_index+2];
                tinyobj::real_t nx = attrib.normals[3*idx.normal_index+0];
                tinyobj::real_t ny = attrib.normals[3*idx.normal_index+1];
                tinyobj::real_t nz = attrib.normals[3*idx.normal_index+2];
                tinyobj::real_t ux = attrib.texcoords[2*idx.texcoord_index+0];
                tinyobj::real_t uy = attrib.texcoords[2*idx.texcoord_index+1];
                vertices.push_back(glm::vec3(vx,vy,vz));
                normals.push_back(glm::vec3(nx,ny,nz));
                uvs.push_back(glm::vec2(ux,uy));
            }
            index_offset += fv;
        }
    }

Поскольку исходный файл .obj имеет формат нескольких лиц, т.е.:

f 1/2/3 3/2/1 3/2/3
f 1/2/3 1/3/4 1/4/5 6/7/2

, поэтому я использую Blender Triangulate с опцией 'Beauty', чтобы преобразовать четырехугольник в треугольник.Но результат рендеринга странный result

Я построил две функции для initOpenGL (выполнить один раз) и render ().

код initOpenGL:

glGenBuffers(1, &VBO);
glBindBuffer(GL_ARRAY_BUFFER, VBO);
glBufferData(GL_ARRAY_BUFFER, vertices.size() * sizeof(glm::vec3), &vertices[0], GL_STATIC_DRAW);


glGenBuffers(1, &UVBO);
glBindBuffer(GL_ARRAY_BUFFER, UVBO);
glBufferData(GL_ARRAY_BUFFER, uvs.size() * sizeof(glm::vec2), &uvs[0], GL_STATIC_DRAW);

//Linking Vertex Attribute
glEnableVertexAttribArray(0);
glBindBuffer(GL_ARRAY_BUFFER, VBO);
glVertexAttribPointer(0, 3, GL_FLOAT, GL_FALSE, 0, (void*)0);

//bind texture
glEnableVertexAttribArray(1);
glBindBuffer(GL_ARRAY_BUFFER, UVBO);
glVertexAttribPointer(1, 2, GL_FLOAT, GL_FALSE, 0, (void*)0);

//load texture
//texture1
glGenTextures(1, &texture1);
glBindTexture(GL_TEXTURE_2D, texture1);
glTexParameteri( GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, GL_CLAMP_TO_EDGE );
glTexParameteri( GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, GL_CLAMP_TO_EDGE );
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR);
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_LINEAR);
if(patternSrc1)
{

    glTexImage2D(GL_TEXTURE_2D, 0, GL_RGBA, patWidth1, patHeight1, 0, GL_RGBA, GL_UNSIGNED_BYTE,
                 patternSrc1);
    //glGenerateMipmap(GL_TEXTURE_2D);
}

рендер () код:

glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT);
glUseProgram(shaderProgram);

camera.ProcessOneFinger(moveStpX, moveStpY);
camera.ProcessTwoFinger(move2X, move2Y);
projection = glm::perspective(camera.GetZoom(), (GLfloat)600/(GLfloat)1024, nearPlane, farPlane);
view = camera.GetViewMatrix();
glUniformMatrix4fv(glGetUniformLocation(shaderProgram, "projection"), 1, GL_FALSE, glm::value_ptr(projection));
glUniformMatrix4fv(glGetUniformLocation(shaderProgram, "view"), 1, GL_FALSE, glm::value_ptr(view));
glm::mat4 model;
model = glm::translate(model, glm::vec3(0.0f, 0.0f, 0.0f));
GLfloat angle = 20.0f;
model = glm::rotate(model, angle, glm::vec3( 1.0f, 0.3f, 0.5f));
glUniformMatrix4fv(glGetUniformLocation(shaderProgram, "model"), 1, GL_FALSE, glm::value_ptr( model ) );

glDrawArrays(GL_TRIANGLES, 0, vertices.size());

Деталь модели

Текстура: enter image description here

Файл Model.obj после триангуляции:

https://pastebin.com/vUjHv8Fr

Спасибо !!!

1 Ответ

0 голосов
/ 24 октября 2018

Мне кажется, что ваша текстура перевернута.По сути, есть две потенциальные ошибки, каждая из которых может привести к этому.Скорее всего, само изображение текстуры перевернуто, например, потому что данные изображения, переданные в glTexImage2D(), расположены в неправильном порядке.В отличие от большинства других API, OpenGL (по умолчанию) ожидает данные о пикселях в строчном порядке , начиная с нижней строки .Проверьте, как вы загружаете данные текстуры, чтобы убедиться, что они в правильном порядке.

Если это не проблема, то, возможно, ваши текстурные координаты предназначены для левой текстурной системы координат.OpenGL, однако, использует правые текстурные координаты, где источником является левый нижний угол изображения текстуры, а не левый верхний угол.Я не парень из Blender, но для этого есть настройки экспорта ...

...