3

我直奔主题:

我为我的顶点创建了一个结构:

struct Vertex3D
{
    Vector3D position;
    Vector2D textureCoordinate;
    Vector3D normal;
}

然后我导入一个特定的 *.dae 文件并将其绑定到一个 OpenGL 顶点缓冲区,该缓冲区由 Vertex3D 结构化顶点列表构成。一切顺利,网格导入并显示,我可以使用着色器对其进行操作,但我有一个问题。

我还要做的是加载并为其分配纹理。之后我试图以这种方式显示它:

glEnableVertexAttribArray(0);
glEnableVertexAttribArray(8);
glEnableVertexAttribArray(2);

glBindBuffer(GL_ARRAY_BUFFER, this->_entries[i].VB);
glVertexAttribPointer(0,3, GL_FLOAT, GL_FALSE, sizeof(RomCommon::Vertex3D), 0); // Vertices
glVertexAttribPointer(8,2, GL_FLOAT, GL_FALSE, sizeof(RomCommon::Vertex3D), (const GLvoid*)sizeof(RomCommon::Vector3D)); // Supposed texture position
glVertexAttribPointer(2,3, GL_FLOAT, GL_FALSE, sizeof(RomCommon::Vertex3D),  (const GLvoid*)(sizeof(RomCommon::Vector2D) + sizeof(RomCommon::Vector3D))); // Supposed normal position
glBindBuffer(GL_ELEMENT_ARRAY_BUFFER, this->_entries[i].IB);
glBindTexture(GL_TEXTURE_2D, 3);
glDrawElements(GL_TRIANGLES, this->_entries[i]._indexCount, GL_UNSIGNED_INT, 0);

glDisableVertexAttribArray(0);
glDisableVertexAttribArray(8);
glDisableVertexAttribArray(2);

我的问题是没有显示纹理。仅分配纹理第一个像素的颜色(在我的情况下为蓝色)。我已经使用gDEBugger调试了整个过程,我可以看出VertexBuffer似乎设置正确,纹理图像加载正确,但不会显示。

我已经尝试搜索和尝试调试的不同方面,这包括:

  • 确保 glEnable(GL_TEXTURE_2D) 已设置;
  • 添加 glEnable(GL_LIGHTING) 和 glDisable(GL_LIGHTING);
  • 使用 glDisable(GL_DEPTH_TEST); glDisable(GL_BLEND); 和 glDisable(GL_SCISSOR_TEST); 在绑定纹理之前
  • 获取错误
  • 搜索奇怪的消息
  • 仔细检查内存中的顶点缓冲区(实际上是通过元素并交叉比较 *.dae 文件中的内容)

编辑

我正在使用 CG 着色器。这是顶点程序:

struct vsOutput {


 float4 position : POSITION;
  float2 texCoord : TEXCOORD0;
  float3 color    : COLOR;
};

vsOutput VS_Main(      float4 position : POSITION,
                           float2 texCoord : TEXCOORD0,
                           float3 color    : COLOR,
                           uniform float4x4 ModelViewProj
                           )
{
  vsOutput OUT;

  OUT.position = mul(ModelViewProj, position);
  OUT.texCoord = texCoord;
  OUT.color = color;

  return OUT;   
}

和片段程序:

struct fsOutput {
  float4 color : COLOR;
};

fsOutput FS_Main(
    float2 texCoord : TEXCOORD0,
    uniform sampler2D decal : TEX0
)
{ 
    fsOutput OUT;
    OUT.color = tex2D(decal,texCoord);
    return OUT;
}

如有必要,我可以添加有关我的项目的更多详细信息,尽管根据我的追踪,问题似乎出在顶点缓冲区的渲染中。

编辑

我还发现因为我使用的是 CG,所以 vertexAttribPointers 应该是不同的(8 代表 TEXCOORD0)所以我在此基础上更改了一般描述。

解决方案编辑

非常感谢评论中的几个小伙子,他们打开了我的思路,从不同的角度看待这个问题,实际上做了更多的阅读。这是解决方案:

    glEnableVertexAttribArray(0);
    glEnableVertexAttribArray(8);
    glEnableVertexAttribArray(2);

    glBindBuffer(GL_ARRAY_BUFFER, this->_entries[i].VB);
    glVertexAttribPointer(0,3, GL_FLOAT, GL_FALSE, sizeof(RomCommon::Vertex3D), 0); // Vertices
    glVertexAttribPointer(8,2, GL_FLOAT, GL_FALSE, sizeof(RomCommon::Vertex3D), (const GLvoid*)sizeof(RomCommon::Vector3D)); // Supposed texture position
    glVertexAttribPointer(2,3, GL_FLOAT, GL_FALSE, sizeof(RomCommon::Vertex3D),  (const GLvoid*)(sizeof(RomCommon::Vector2D) + sizeof(RomCommon::Vector3D))); // Supposed normal position
    glBindBuffer(GL_ELEMENT_ARRAY_BUFFER, this->_entries[i].IB);


/* Solution start */
    glEnableClientState(GL_TEXTURE_COORD_ARRAY);
    glTexCoordPointer(2, GL_FLOAT, sizeof(RomCommon::Vertex3D), (const GLvoid*)12);
        glBindTexture(GL_TEXTURE_2D, 3);
        glDrawElements(GL_TRIANGLES, this->_entries[i]._indexCount, GL_UNSIGNED_INT, 0);
    glDisableClientState(GL_TEXTURE_COORD_ARRAY);
/* Solution end */


    glDisableVertexAttribArray(0);
    glDisableVertexAttribArray(8);
    glDisableVertexAttribArray(2);

我的问题是我只是将纹理坐标传递给着色器,但没有将其推送到 OpenGL 状态机(如果这是一个好的表达方式)。我认为这与我使用的是 CG 着色器而不是 GLSL 着色器这一事实有关,尽管我可能是错的。该代码大致基于我阅读的一些示例和解释,但它们都基于 GLSL 着色器并且它们正在工作,因此是奉献的。无论如何,现在已经解决了。

4

1 回答 1

1
glBindTexture(GL_TEXTURE_2D, 3);

我会警惕在 OpenGL 上强制使用纹理对象 ID。从技术上讲,它应该可以工作,但使用glGenTextures()获取纹理 ID 是一个更好的主意。

于 2013-01-29T15:46:30.030 回答