2011-01-18 11:40:49 +00:00
// This code was written for the OpenTK library and has been released
// to the Public Domain.
// It is provided "as is" without express or implied warranty of any kind.
2009-02-22 10:43:35 +00:00
using System ;
using System.Diagnostics ;
using System.Drawing ;
using OpenTK ;
using OpenTK.Input ;
using OpenTK.Graphics ;
using OpenTK.Graphics.OpenGL ;
namespace Examples.Tutorial
{
2010-10-09 19:59:06 +00:00
[Example("Framebuffer Objects", ExampleCategory.OpenGL, "1.x", Documentation = "FramebufferObject")]
2009-02-22 10:43:35 +00:00
public class SimpleFBO : GameWindow
{
public SimpleFBO ( )
2010-10-28 15:58:10 +00:00
: base ( 800 , 400 )
2009-02-22 10:43:35 +00:00
{
}
uint ColorTexture ;
uint DepthTexture ;
uint FBOHandle ;
const int TextureSize = 512 ;
2010-10-28 15:58:10 +00:00
Examples . Shapes . DrawableShape Object ;
2009-02-22 10:43:35 +00:00
2009-10-21 13:33:00 +00:00
protected override void OnLoad ( EventArgs e )
2009-02-22 10:43:35 +00:00
{
2011-09-06 12:55:19 +00:00
base . OnLoad ( e ) ;
if ( ! GL . GetString ( StringName . Extensions ) . Contains ( "GL_EXT_framebuffer_object" ) )
2009-02-22 15:50:08 +00:00
{
2011-09-06 12:55:19 +00:00
throw new NotSupportedException (
"GL_EXT_framebuffer_object extension is required. Please update your drivers." ) ;
2009-02-22 15:50:08 +00:00
}
2010-10-09 19:59:06 +00:00
2010-10-28 15:58:10 +00:00
Object = new Shapes . TorusKnot ( 256 , 16 , 0.2 , 7 , 8 , 1 , true ) ;
2009-02-22 10:43:35 +00:00
GL . Enable ( EnableCap . DepthTest ) ;
2011-05-30 12:31:39 +00:00
GL . ClearDepth ( 1.0 ) ;
2009-02-22 10:43:35 +00:00
GL . DepthFunc ( DepthFunction . Lequal ) ;
2010-10-28 15:58:10 +00:00
GL . Enable ( EnableCap . CullFace ) ;
2009-02-22 10:43:35 +00:00
// Create Color Tex
GL . GenTextures ( 1 , out ColorTexture ) ;
GL . BindTexture ( TextureTarget . Texture2D , ColorTexture ) ;
GL . TexImage2D ( TextureTarget . Texture2D , 0 , PixelInternalFormat . Rgba8 , TextureSize , TextureSize , 0 , PixelFormat . Rgba , PixelType . UnsignedByte , IntPtr . Zero ) ;
GL . TexParameter ( TextureTarget . Texture2D , TextureParameterName . TextureMinFilter , ( int ) TextureMinFilter . Linear ) ;
GL . TexParameter ( TextureTarget . Texture2D , TextureParameterName . TextureMagFilter , ( int ) TextureMagFilter . Linear ) ;
GL . TexParameter ( TextureTarget . Texture2D , TextureParameterName . TextureWrapS , ( int ) TextureWrapMode . ClampToBorder ) ;
GL . TexParameter ( TextureTarget . Texture2D , TextureParameterName . TextureWrapT , ( int ) TextureWrapMode . ClampToBorder ) ;
// GL.Ext.GenerateMipmap( GenerateMipmapTarget.Texture2D );
// Create Depth Tex
GL . GenTextures ( 1 , out DepthTexture ) ;
GL . BindTexture ( TextureTarget . Texture2D , DepthTexture ) ;
GL . TexImage2D ( TextureTarget . Texture2D , 0 , ( PixelInternalFormat ) All . DepthComponent32 , TextureSize , TextureSize , 0 , PixelFormat . DepthComponent , PixelType . UnsignedInt , IntPtr . Zero ) ;
// things go horribly wrong if DepthComponent's Bitcount does not match the main Framebuffer's Depth
GL . TexParameter ( TextureTarget . Texture2D , TextureParameterName . TextureMinFilter , ( int ) TextureMinFilter . Linear ) ;
GL . TexParameter ( TextureTarget . Texture2D , TextureParameterName . TextureMagFilter , ( int ) TextureMagFilter . Linear ) ;
GL . TexParameter ( TextureTarget . Texture2D , TextureParameterName . TextureWrapS , ( int ) TextureWrapMode . ClampToBorder ) ;
GL . TexParameter ( TextureTarget . Texture2D , TextureParameterName . TextureWrapT , ( int ) TextureWrapMode . ClampToBorder ) ;
// GL.Ext.GenerateMipmap( GenerateMipmapTarget.Texture2D );
// Create a FBO and attach the textures
GL . Ext . GenFramebuffers ( 1 , out FBOHandle ) ;
GL . Ext . BindFramebuffer ( FramebufferTarget . FramebufferExt , FBOHandle ) ;
GL . Ext . FramebufferTexture2D ( FramebufferTarget . FramebufferExt , FramebufferAttachment . ColorAttachment0Ext , TextureTarget . Texture2D , ColorTexture , 0 ) ;
GL . Ext . FramebufferTexture2D ( FramebufferTarget . FramebufferExt , FramebufferAttachment . DepthAttachmentExt , TextureTarget . Texture2D , DepthTexture , 0 ) ;
#region Test for Error
switch ( GL . Ext . CheckFramebufferStatus ( FramebufferTarget . FramebufferExt ) )
{
case FramebufferErrorCode . FramebufferCompleteExt :
{
Console . WriteLine ( "FBO: The framebuffer is complete and valid for rendering." ) ;
break ;
}
case FramebufferErrorCode . FramebufferIncompleteAttachmentExt :
{
Console . WriteLine ( "FBO: One or more attachment points are not framebuffer attachment complete. This could mean there’ s no texture attached or the format isn’ t renderable. For color textures this means the base format must be RGB or RGBA and for depth textures it must be a DEPTH_COMPONENT format. Other causes of this error are that the width or height is zero or the z-offset is out of range in case of render to volume." ) ;
break ;
}
case FramebufferErrorCode . FramebufferIncompleteMissingAttachmentExt :
{
Console . WriteLine ( "FBO: There are no attachments." ) ;
break ;
}
/ * case FramebufferErrorCode . GL_FRAMEBUFFER_INCOMPLETE_DUPLICATE_ATTACHMENT_EXT :
{
Console . WriteLine ( "FBO: An object has been attached to more than one attachment point." ) ;
break ;
} * /
case FramebufferErrorCode . FramebufferIncompleteDimensionsExt :
{
Console . WriteLine ( "FBO: Attachments are of different size. All attachments must have the same width and height." ) ;
break ;
}
case FramebufferErrorCode . FramebufferIncompleteFormatsExt :
{
Console . WriteLine ( "FBO: The color attachments have different format. All color attachments must have the same format." ) ;
break ;
}
case FramebufferErrorCode . FramebufferIncompleteDrawBufferExt :
{
Console . WriteLine ( "FBO: An attachment point referenced by GL.DrawBuffers() doesn’ t have an attachment." ) ;
break ;
}
case FramebufferErrorCode . FramebufferIncompleteReadBufferExt :
{
Console . WriteLine ( "FBO: The attachment point referenced by GL.ReadBuffers() doesn’ t have an attachment." ) ;
break ;
}
case FramebufferErrorCode . FramebufferUnsupportedExt :
{
Console . WriteLine ( "FBO: This particular FBO configuration is not supported by the implementation." ) ;
break ;
}
default :
{
Console . WriteLine ( "FBO: Status unknown. (yes, this is really bad.)" ) ;
break ;
}
}
// using FBO might have changed states, e.g. the FBO might not support stereoscopic views or double buffering
int [ ] queryinfo = new int [ 6 ] ;
GL . GetInteger ( GetPName . MaxColorAttachmentsExt , out queryinfo [ 0 ] ) ;
GL . GetInteger ( GetPName . AuxBuffers , out queryinfo [ 1 ] ) ;
GL . GetInteger ( GetPName . MaxDrawBuffers , out queryinfo [ 2 ] ) ;
GL . GetInteger ( GetPName . Stereo , out queryinfo [ 3 ] ) ;
GL . GetInteger ( GetPName . Samples , out queryinfo [ 4 ] ) ;
GL . GetInteger ( GetPName . Doublebuffer , out queryinfo [ 5 ] ) ;
Console . WriteLine ( "max. ColorBuffers: " + queryinfo [ 0 ] + " max. AuxBuffers: " + queryinfo [ 1 ] + " max. DrawBuffers: " + queryinfo [ 2 ] +
"\nStereo: " + queryinfo [ 3 ] + " Samples: " + queryinfo [ 4 ] + " DoubleBuffer: " + queryinfo [ 5 ] ) ;
Console . WriteLine ( "Last GL Error: " + GL . GetError ( ) ) ;
#endregion Test for Error
GL . PushAttrib ( AttribMask . ViewportBit ) ;
{
GL . Viewport ( 0 , 0 , TextureSize , TextureSize ) ;
// clear the screen in red, to make it very obvious what the clear affected. only the FBO, not the real framebuffer
GL . ClearColor ( 1f , 0f , 0f , 0f ) ;
GL . Clear ( ClearBufferMask . ColorBufferBit | ClearBufferMask . DepthBufferBit ) ;
2010-10-28 15:58:10 +00:00
OpenTK . Matrix4 perspective = OpenTK . Matrix4 . CreatePerspectiveFieldOfView ( MathHelper . PiOver4 , TextureSize / ( float ) TextureSize , 2.5f , 6f ) ;
GL . MatrixMode ( MatrixMode . Projection ) ;
GL . LoadMatrix ( ref perspective ) ;
Matrix4 lookat = Matrix4 . LookAt ( 0f , 0f , 4.5f , 0f , 0f , 0f , 0f , 1f , 0f ) ;
GL . MatrixMode ( MatrixMode . Modelview ) ;
GL . LoadMatrix ( ref lookat ) ;
// draw some complex object into the FBO's textures
GL . Enable ( EnableCap . Lighting ) ;
GL . Enable ( EnableCap . Light0 ) ;
GL . Enable ( EnableCap . ColorMaterial ) ;
GL . Color3 ( 0f , 1f , 0f ) ;
Object . Draw ( ) ;
GL . Disable ( EnableCap . ColorMaterial ) ;
GL . Disable ( EnableCap . Light0 ) ;
GL . Disable ( EnableCap . Lighting ) ;
2009-02-22 10:43:35 +00:00
}
GL . PopAttrib ( ) ;
GL . Ext . BindFramebuffer ( FramebufferTarget . FramebufferExt , 0 ) ; // disable rendering into the FBO
GL . ClearColor ( . 1f , . 2f , . 3f , 0f ) ;
GL . Color3 ( 1f , 1f , 1f ) ;
GL . Enable ( EnableCap . Texture2D ) ; // enable Texture Mapping
GL . BindTexture ( TextureTarget . Texture2D , 0 ) ; // bind default texture
}
2009-10-21 13:51:39 +00:00
protected override void OnUnload ( EventArgs e )
2009-02-22 10:43:35 +00:00
{
2010-10-28 15:58:10 +00:00
Object . Dispose ( ) ;
2009-02-22 10:43:35 +00:00
// Clean up what we allocated before exiting
2009-02-22 16:01:29 +00:00
if ( ColorTexture ! = 0 )
GL . DeleteTextures ( 1 , ref ColorTexture ) ;
2009-02-22 10:43:35 +00:00
2009-02-22 16:01:29 +00:00
if ( DepthTexture ! = 0 )
GL . DeleteTextures ( 1 , ref DepthTexture ) ;
if ( FBOHandle ! = 0 )
GL . Ext . DeleteFramebuffers ( 1 , ref FBOHandle ) ;
2009-02-22 10:43:35 +00:00
}
2009-06-02 15:49:39 +00:00
protected override void OnResize ( EventArgs e )
2009-02-22 10:43:35 +00:00
{
GL . Viewport ( 0 , 0 , Width , Height ) ;
2010-10-09 19:59:06 +00:00
2009-08-14 13:13:28 +00:00
double aspect_ratio = Width / ( double ) Height ;
2009-08-15 17:52:49 +00:00
OpenTK . Matrix4 perspective = OpenTK . Matrix4 . CreatePerspectiveFieldOfView ( MathHelper . PiOver4 , ( float ) aspect_ratio , 1 , 64 ) ;
2009-02-22 10:43:35 +00:00
GL . MatrixMode ( MatrixMode . Projection ) ;
2009-08-14 13:13:28 +00:00
GL . LoadMatrix ( ref perspective ) ;
2009-02-22 10:43:35 +00:00
2009-08-14 13:13:28 +00:00
Matrix4 lookat = Matrix4 . LookAt ( 0 , 0 , 3 , 0 , 0 , 0 , 0 , 1 , 0 ) ;
2009-02-22 10:43:35 +00:00
GL . MatrixMode ( MatrixMode . Modelview ) ;
2009-08-14 13:13:28 +00:00
GL . LoadMatrix ( ref lookat ) ;
2009-02-22 10:43:35 +00:00
base . OnResize ( e ) ;
}
2009-06-02 15:49:39 +00:00
protected override void OnUpdateFrame ( FrameEventArgs e )
2009-02-22 10:43:35 +00:00
{
base . OnUpdateFrame ( e ) ;
2014-07-21 15:20:19 +00:00
var keyboard = OpenTK . Input . Keyboard . GetState ( ) ;
if ( keyboard [ Key . Escape ] )
2009-02-22 10:43:35 +00:00
this . Exit ( ) ;
}
2009-06-02 15:49:39 +00:00
protected override void OnRenderFrame ( FrameEventArgs e )
2009-02-22 10:43:35 +00:00
{
GL . Clear ( ClearBufferMask . ColorBufferBit | ClearBufferMask . DepthBufferBit ) ;
2010-10-09 19:59:06 +00:00
2009-02-22 10:43:35 +00:00
GL . PushMatrix ( ) ;
{
// Draw the Color Texture
GL . Translate ( - 1.1f , 0f , 0f ) ;
GL . BindTexture ( TextureTarget . Texture2D , ColorTexture ) ;
2014-02-22 19:55:34 +00:00
GL . Begin ( PrimitiveType . Quads ) ;
2009-02-22 10:43:35 +00:00
{
GL . TexCoord2 ( 0f , 1f ) ;
GL . Vertex2 ( - 1.0f , 1.0f ) ;
GL . TexCoord2 ( 0.0f , 0.0f ) ;
GL . Vertex2 ( - 1.0f , - 1.0f ) ;
GL . TexCoord2 ( 1.0f , 0.0f ) ;
GL . Vertex2 ( 1.0f , - 1.0f ) ;
GL . TexCoord2 ( 1.0f , 1.0f ) ;
GL . Vertex2 ( 1.0f , 1.0f ) ;
}
GL . End ( ) ;
// Draw the Depth Texture
GL . Translate ( + 2.2f , 0f , 0f ) ;
GL . BindTexture ( TextureTarget . Texture2D , DepthTexture ) ;
2014-02-22 19:55:34 +00:00
GL . Begin ( PrimitiveType . Quads ) ;
2009-02-22 10:43:35 +00:00
{
GL . TexCoord2 ( 0f , 1f ) ;
GL . Vertex2 ( - 1.0f , 1.0f ) ;
GL . TexCoord2 ( 0.0f , 0.0f ) ;
GL . Vertex2 ( - 1.0f , - 1.0f ) ;
GL . TexCoord2 ( 1.0f , 0.0f ) ;
GL . Vertex2 ( 1.0f , - 1.0f ) ;
GL . TexCoord2 ( 1.0f , 1.0f ) ;
GL . Vertex2 ( 1.0f , 1.0f ) ;
}
GL . End ( ) ;
}
GL . PopMatrix ( ) ;
this . SwapBuffers ( ) ;
}
#region public static void Main ( )
/// <summary>
/// Entry point of this example.
/// </summary>
[STAThread]
public static void Main ( )
{
using ( SimpleFBO example = new SimpleFBO ( ) )
{
Utilities . SetWindowTitle ( example ) ;
example . Run ( 30.0 , 0.0 ) ;
}
}
#endregion
}
2008-05-05 17:05:11 +00:00
}