好的,所以我这个FBO有一个深度缓冲区和4个颜色附件。我想在我拿着光标的地方和第四个颜色附件的值中读取dept缓冲区。我似乎无法管理这两者中的任何一个。 我写了这堂课:
class Picker3D{
public:
bool useAtachment;
GLenum atachment;
float PickerLastSignature;
point3f PickerLastPos;
Picker3D( bool useAtachment = true , GLenum atachment = GL_COLOR_ATTACHMENT3 ) : useAtachment(useAtachment) , atachment(atachment) {
}
void SetObjectSignature( float signature , GLSL_Prog prog ){
glUniform1f( glGetUniformLocation( prog.PR , "Signature" ) , signature );
}
float getSignature( float x , float y , float height ){
glGetError();
if( useAtachment )
glReadBuffer( atachment );
cout << " Error message1 : " << glGetError() << endl ;
float *pixels = new float[4];
glReadPixels( x , height - y , 1 , 1 , GL_RGB , GL_FLOAT , pixels );
PickerLastSignature = pixels[1];
cout << " Error message2 : " << glGetError() << endl ;
return PickerLastSignature;
}
point3f get3DPosition( float x , float y , float height ){
//glReadBuffer( GL_DEPTH_ATTACHMENT );
double depth;
glReadPixels( x , y , 1 , 1 , GL_DEPTH_COMPONENT , GL_FLOAT , &depth );
cout << depth << endl;
int viewport[4];
float *modelview;
double projection[16];
double ModelView[16];
modelview = mat4f::GetTopMatrix().returnTransposedMatrix().returnFloatArray();
for( int i = 0 ; i < 16 ; i++ ){
ModelView[i] = modelview[i];
}
glGetDoublev(GL_PROJECTION_MATRIX, projection);
glGetIntegerv(GL_VIEWPORT, viewport);
double x1 , y1 , z1;
gluUnProject( x , height - y , depth , ModelView , projection , viewport , &x1 , &y1 , &z1 );
PickerLastPos = point3f( x1 , y1 , z1 );
return PickerLastPos;
}
};
我就这样使用它
//在主循环中
BindFrameBuffer();
drawStuff();
Picker.get3DPosition( x , y , height );
Picker.getSignature( x , y , height );
UnbindBuffer();
drawSecondPass();
endDrawSecondPass();
// check Values of PickerLastPos and PickerLastSignature
SwapBuffers();
深度并没有改变,尽管它正在屏幕上运行,并且getSignature在glReadBuffer的行上给出了错误1282(我认为无效操作)。
主循环是:
glClear( GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT | GL_STENCIL_BUFFER_BIT );
/// deferred pass
DefFBO->drawFBO();
GeometryShader.UseNow();
glClear( GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT | GL_STENCIL_BUFFER_BIT );
// load the view matrix and set the camera
cam.camera_set();
cam.camera_get( GeometryShader );
// load identity for model matrix
mat4f::LoadIdentity();
/// Geometry Drawing
mat4f::PushMatrix();
mat4f::Translate( 0 , 0 , -10 );
Picker.SetObjectSignature( 1.0f , GeometryShader );
T90.obj_rend( GeometryShader );
Picker.SetObjectSignature( 2.0f , GeometryShader );
Town.obj_rend( GeometryShader );
mat4f::PopMatrix();
Picker.get3DPosition( Window.Mouse->mouse_x , Window.Mouse->mouse_y , Window.Height );
/// setting the deferred read from buffer part
// last read from framebuffer
Picker.getSignature( Window.Mouse->mouse_x , Window.Mouse->mouse_y , Window.Height );
// unbind FBO
DefFBO->readFBO();
// use fragment shader, for lights and for post processing
FragmentShader.UseNow();
// send windows width and height and bind multiple textures
DefFBO->send_to_GLSL( FragmentShader.PR );
// send view matrix to the shader
cam.camera_get( FragmentShader );
glDepthMask(GL_FALSE);
glEnable(GL_DEPTH_TEST);
glUniform3f( glGetUniformLocation( FragmentShader.PR , "CameraPosition" ) , cam.x , cam.y , cam.z );
/// final draw
DefFBO->sendGlobalLight( FragmentShader , point3f( 0 , 1 , 0 ) , point3f( 0.3 , 0.3 , 0.3 ) , point3f( 0.9 , 0.9 , 0.9 ) );
cout << Picker.PickerLastPos << endl;
DefFBO->end_read();
glEnable(GL_DEPTH_TEST);
glDepthMask(GL_TRUE);
答案 0 :(得分:0)
正如@Andon指出的那样,似乎我必须绑定fbo才能读取数据。我与GL_DRAW_FRAMEBUFFER绑定。