c – 不完整的球体OpenGL

我想用VBO绘制一个球体,用于纹理的顶点,颜色和UV坐标.我的问题是球体没有“封闭”,原点上有一个洞.我知道这是因为我的代码依赖于每个顶点之间的(1 /段)距离;我正在使用segments = 40.

我知道,如果我提高了这个值,那么洞会更低,但程序更慢.我不知道是否有办法在不增加变量的情况下消除孔.

这是代码:

for(int i = 0; i <= segments; i++){

    double lat0 = pi * (-0.5 + (double)(i - 1) / segments);
    double z0 = sin(lat0);
    double zr0 = cos(lat0);
    // lat1 = [-pi/2..pi/2]
    double lat1 = pi * (-0.5 + (double)i / segments);
    double z1 = sin(lat1);
    double zr1 = cos(lat1);

    for (int j = 0; j <= segments; j++){    // Longitud
        // lng = [0..2*pi]
        double lng = 2 * pi * (double)(j - 1) / segments;
        double x = cos(lng);
        double y = sin(lng);
        //glNormal3f(x * zr0, y * zr0, z0); // Normals
        ballVerts.push_back(x * zr0); //X
        ballVerts.push_back(y * zr0); //Y
        ballVerts.push_back(z0);      //Z

        ballVerts.push_back(0.0f);
        ballVerts.push_back(0.0f);
        ballVerts.push_back(0.0f);
        ballVerts.push_back(1.0f); //R,G,B,A

        texX = abs(1 - (0.5f + atan2(z0, x * zr0) / (2.0 * pi)));
        texY = 0.5f - asin(y * zr0) / pi;
        ballVerts.push_back(texX);  // Texture coords
        ballVerts.push_back(texY);  // U, V


        //glNormal3f(x * zr1, y * zr1, z1); //Normals
        ballVerts.push_back(x * zr1); //X
        ballVerts.push_back(y * zr1); //Y
        ballVerts.push_back(z1);      //Z
        ballVerts.push_back(0.0f);
        ballVerts.push_back(0.0f);
        ballVerts.push_back(1.0f);
        ballVerts.push_back(1.0f); //R,G,B,A

        texX = abs(1 - (0.5f + atan2(z1, x * zr1) / (2.0 * pi)));
        texY = 0.5f - asin(y * zr1) / pi;
        ballVerts.push_back(texX);  // Texture coords
        ballVerts.push_back(texY);

    }
}
// Create VBO....

这是我的输出:

最佳答案 我认为这不是一个漏洞.你正在绘制一个太多的段,并导致它在南极绘制额外的三角形,纹理缠绕:

for(int i = 0; i <= segments; i++){
    double lat0 = pi * (-0.5 + (double)(i - 1) / segments);

在第一次循环迭代中,当i = 0时,角度将小于-0.5 * pi,从而导致图片中显示额外的三角形.

如果要将纬度范围拆分为段片段,则只需要遍历外部循环段时间.使用上面的代码,从0到包括段的循环,您将迭代段1次.

解决这个问题的最简单方法是在1处开始循环:

for(int i = 1; i <= segments; i++){
    double lat0 = pi * (-0.5 + (double)(i - 1) / segments);

我可能会从0循环并使结束独占,并改变角度计算.但那真的相当于:

for(int i = 0; i < segments; i++){
    double lat0 = pi * (-0.5 + (double) / segments);
    ...
    double lat1 = pi * (-0.5 + (double)(i + 1) / segments);
点赞