10

我想用 OpenCV 作为 OpenGL 纹理加载图像(jpg 和 png)。

这是我将图像加载到 OpenGL 的方式:

glEnable(GL_TEXTURE_2D);
  textureData = loadTextureData("textures/trashbin.png");
  cv::Mat image = cv::imread("textures/trashbin.png");
  if(image.empty()){
      std::cout << "image empty" << std::endl;
  }else{
      glGenTextures( 1, &textureTrash );
      glBindTexture( GL_TEXTURE_2D, textureTrash );
      glTexParameterf(GL_TEXTURE_2D,GL_TEXTURE_MIN_FILTER,GL_LINEAR);
      glTexParameterf(GL_TEXTURE_2D,GL_TEXTURE_MAG_FILTER,GL_LINEAR);
      glTexParameterf( GL_TEXTURE_2D, GL_TEXTURE_WRAP_S , GL_REPEAT );
      glTexParameterf( GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, GL_REPEAT );
      glTexImage2D(GL_TEXTURE_2D,0,3,image.cols, image.rows,0,GL_RGB,GL_UNSIGNED_BYTE, image.data);
  }

图像已加载,因为“image.empty”总是返回 false

以下是我使用创建的纹理渲染场景的方法:

  glActiveTexture(GL_TEXTURE0);
  glBindTexture(GL_TEXTURE_2D, textureTrash);
  glm_ModelViewMatrix.top() = glm::translate(glm_ModelViewMatrix.top(),0.0f,-13.0f,-10.0f);
  glUniformMatrix4fv(uniformLocations["modelview"], 1, false, glm::value_ptr(glm_ModelViewMatrix.top()));

  std::cout << "textureShaderID: " << glGetUniformLocation(shaderProgram,"texture") << std::endl;

  glUniform1i(glGetUniformLocation(shaderProgram,"texture"), 0);
  objLoader->getMeshObj("trashbin")->render();

最后是我想将纹理应用到几何体的 fragmentShader

#version 330
in vec2 tCoord;

// texture //
// TODO: set up a texture uniform //
uniform sampler2D texture;

// this defines the fragment output //
out vec4 color;

void main() {
  // TODO: get the texel value from your texture at the position of the passed texture coordinate //
  color = texture2D(texture, tCoord);
}

纹理坐标来自顶点缓冲区对象,并从 .obj 文件中正确设置。当我在片段着色器中将颜色设置为例如红色或 vec4(tCoord,0,1); 时,我也可以在场景中看到对象。然后对象以不同的颜色着色。

不幸的是,当我想应用纹理时屏幕保持黑色......有人可以帮助我并告诉我为什么保持黑色?

4

2 回答 2

18

仅查看您的纹理加载代码,您忽略了许多关于 OpenCV 如何在内存中布置图像的考虑。我已经在这个答案glGetTexImage中解释了相反的方向(进入 OpenCV 图像),但将在此处概括为 CV-GL 方向:

首先,OpenCV 不一定会紧密存储图像行,但可能会将它们与某些字节边界对齐(不知道多少,至少 4 个,但可能是 8 个或更多?)。如果幸运的话,它将使用 4 字节对齐,并且 GL 也设置为 4 字节对齐的默认像素存储模式。但为了安全起见,最好手动调整像素存储模式:

//use fast 4-byte alignment (default anyway) if possible
glPixelStorei(GL_UNPACK_ALIGNMENT, (image.step & 3) ? 1 : 4);

//set length of one complete row in data (doesn't need to equal image.cols)
glPixelStorei(GL_UNPACK_ROW_LENGTH, image.step/image.elemSize());

然后你必须考虑 OpenCV 从上到下存储图像,而 GL 使用从下到上的事实。这可以通过适当地镜像 t-texture 坐标来解决(可能直接在着色器中),但您也可以在上传之前翻转图像:

cv::flip(image, flipped, 0);
image = flipped;              //maybe just cv::flip(image, image, 0)?

最后但并非最不重要的一点是,OpenCV 以 BGR 格式存储彩色图像,因此将其上传为 RGB 会扭曲颜色。所以GL_BGRglTexImage2D.

这些可能不是您问题的完整解决方案(因为我认为这些错误应该导致图像失真而不是黑色),但它们绝对是需要注意的问题。

编辑:您的片段着色器是否真的成功编译(在使用纹理的完整版本中)?我问是因为在 GLSL 3.30 中,您使用的单词texture也是内置函数的名称(实际上应该使用它而不是不推荐使用的texture2D函数),所以编译器可能有一些名称解析问题(也许这个错误在您的简化着色器中被忽略,因为整个统一将被优化掉,并且已知许多 GLSL 编译器不是严格符合标准的)。所以试着给那个采样器统一一个不同的名字。

于 2013-05-29T11:20:47.400 回答
9

好的,这是我的工作解决方案 - 基于“Christan Rau”的想法 - 谢谢!

cv::Mat image = cv::imread("textures/trashbin.png");
  //cv::Mat flipped;
  //cv::flip(image, flipped, 0);
  //image = flipped;
  if(image.empty()){
      std::cout << "image empty" << std::endl;
  }else{
      cv::flip(image, image, 0);
      glGenTextures(1, &textureTrash);
      glBindTexture(GL_TEXTURE_2D, textureTrash);

      glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_NEAREST);
      glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_LINEAR);

        // Set texture clamping method
      glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, GL_CLAMP);
      glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, GL_CLAMP);


      glTexImage2D(GL_TEXTURE_2D,     // Type of texture
                     0,                 // Pyramid level (for mip-mapping) - 0 is the top level
                     GL_RGB,            // Internal colour format to convert to
                     image.cols,          // Image width  i.e. 640 for Kinect in standard mode
                     image.rows,          // Image height i.e. 480 for Kinect in standard mode
                     0,                 // Border width in pixels (can either be 1 or 0)
                     GL_BGR, // Input image format (i.e. GL_RGB, GL_RGBA, GL_BGR etc.)
                     GL_UNSIGNED_BYTE,  // Image data type
                     image.ptr());        // The actual image data itself

      glGenerateMipmap(GL_TEXTURE_2D);
  }
于 2013-05-29T13:46:43.537 回答