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-09-03 19:01:11 +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-02 18:52:34 +00:00
|
|
|
|
[Example("Basic Geometry Shader", ExampleCategory.OpenGL, "2.x", Documentation = "Simple usage of EXT_geometry_shader4")]
|
2009-09-03 19:01:11 +00:00
|
|
|
|
public class SimpleGeometryShader : GameWindow
|
|
|
|
|
{
|
|
|
|
|
public SimpleGeometryShader()
|
|
|
|
|
: base(800, 600)
|
|
|
|
|
{
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int shaderProgram = 0;
|
|
|
|
|
|
2009-10-21 13:33:00 +00:00
|
|
|
|
protected override void OnLoad(EventArgs e)
|
2009-09-03 19:01:11 +00:00
|
|
|
|
{
|
|
|
|
|
if (!GL.GetString(StringName.Extensions).Contains("EXT_geometry_shader4"))
|
|
|
|
|
{
|
|
|
|
|
System.Windows.Forms.MessageBox.Show(
|
|
|
|
|
"Your video card does not support EXT_geometry_shader4. Please update your drivers.",
|
|
|
|
|
"EXT_geometry_shader4 not supported",
|
|
|
|
|
System.Windows.Forms.MessageBoxButtons.OK, System.Windows.Forms.MessageBoxIcon.Exclamation);
|
|
|
|
|
Exit();
|
|
|
|
|
throw new NotSupportedException();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// create a shader object.
|
|
|
|
|
shaderProgram = GL.CreateProgram();
|
|
|
|
|
// create shader objects for all three types.
|
|
|
|
|
int vert = GL.CreateShader(ShaderType.VertexShader);
|
|
|
|
|
int frag = GL.CreateShader(ShaderType.FragmentShader);
|
|
|
|
|
int geom = GL.CreateShader(ShaderType.GeometryShaderExt);
|
|
|
|
|
|
|
|
|
|
// GLSL for fragment shader.
|
|
|
|
|
String fragSource = @"
|
|
|
|
|
void main( void )
|
|
|
|
|
{
|
|
|
|
|
gl_FragColor = vec4(0, 1, 0, 0);
|
|
|
|
|
}
|
|
|
|
|
";
|
|
|
|
|
|
|
|
|
|
// GLSL for vertex shader.
|
|
|
|
|
String vertSource = @"
|
|
|
|
|
void main( void )
|
|
|
|
|
{
|
|
|
|
|
gl_Position = ftransform();
|
|
|
|
|
}
|
|
|
|
|
";
|
|
|
|
|
|
|
|
|
|
// GLSL for geometry shader.
|
|
|
|
|
// Note this is a version 1.20 shader
|
|
|
|
|
// Also note GL_EXT_geometry_shader4 must be enabled explicitly, correct
|
|
|
|
|
// OpenGL implementations should only have the new tokens, like
|
|
|
|
|
// EmitVertex and EndPrimitive, when this extension is enabled.
|
|
|
|
|
String geomSource = @"
|
|
|
|
|
#version 120
|
|
|
|
|
#extension GL_EXT_geometry_shader4 : enable
|
|
|
|
|
|
|
|
|
|
void main(void)
|
|
|
|
|
{
|
|
|
|
|
// variable to use in for loops
|
|
|
|
|
int i;
|
|
|
|
|
|
|
|
|
|
// Emit the original vertices without changing, making
|
|
|
|
|
// this part exactly the same as if no geometry shader
|
|
|
|
|
// was used.
|
|
|
|
|
for(i=0; i< gl_VerticesIn; i++)
|
|
|
|
|
{
|
|
|
|
|
gl_Position = gl_PositionIn[i];
|
|
|
|
|
EmitVertex();
|
|
|
|
|
}
|
|
|
|
|
// End the one primitive with the original vertices
|
|
|
|
|
EndPrimitive();
|
|
|
|
|
|
|
|
|
|
// Now we generate some more! This translates 0.2 over
|
|
|
|
|
// the positive x axis.
|
|
|
|
|
for(i=0; i< gl_VerticesIn; i++)
|
|
|
|
|
{
|
|
|
|
|
gl_Position = gl_PositionIn[i];
|
|
|
|
|
gl_Position.x += 0.2;
|
|
|
|
|
EmitVertex();
|
|
|
|
|
}
|
|
|
|
|
EndPrimitive();
|
|
|
|
|
}
|
|
|
|
|
";
|
|
|
|
|
|
|
|
|
|
// compile shaders.
|
|
|
|
|
compileShader(frag, fragSource);
|
|
|
|
|
compileShader(vert, vertSource);
|
|
|
|
|
compileShader(geom, geomSource);
|
|
|
|
|
|
|
|
|
|
// attach shaders and link the program.
|
|
|
|
|
GL.AttachShader(shaderProgram, frag);
|
|
|
|
|
GL.AttachShader(shaderProgram, vert);
|
|
|
|
|
GL.AttachShader(shaderProgram, geom);
|
|
|
|
|
|
|
|
|
|
// Set the input type of the primitives we are going to feed the geometry shader, this should be the same as
|
|
|
|
|
// the primitive type given to GL.Begin. If the types do not match a GL error will occur (todo: verify GL_INVALID_ENUM, on glBegin)
|
2014-02-22 19:55:34 +00:00
|
|
|
|
GL.Ext.ProgramParameter(shaderProgram, AssemblyProgramParameterArb.GeometryInputType, (int)BeginMode.Lines);
|
2009-09-03 19:01:11 +00:00
|
|
|
|
// Set the output type of the geometry shader. Becasue we input Lines we will output LineStrip(s).
|
2014-02-22 19:55:34 +00:00
|
|
|
|
GL.Ext.ProgramParameter(shaderProgram, AssemblyProgramParameterArb.GeometryOutputType, (int)BeginMode.LineStrip);
|
2009-09-03 19:01:11 +00:00
|
|
|
|
|
|
|
|
|
// We must tell the shader program how much vertices the geometry shader will output (at most).
|
|
|
|
|
// One simple way is to query the maximum and use that.
|
|
|
|
|
// NOTE: Make sure that the number of vertices * sum(components of active varyings) does not
|
|
|
|
|
// exceed MaxGeometryTotalOutputComponents.
|
2014-02-22 19:55:34 +00:00
|
|
|
|
GL.Ext.ProgramParameter(shaderProgram, AssemblyProgramParameterArb.GeometryVerticesOut, 50);
|
2009-09-03 19:01:11 +00:00
|
|
|
|
|
|
|
|
|
// NOTE: calls to ProgramParameter do not take effect until you call LinkProgram.
|
|
|
|
|
GL.LinkProgram(shaderProgram);
|
|
|
|
|
|
|
|
|
|
// output link info log.
|
|
|
|
|
string info;
|
|
|
|
|
GL.GetProgramInfoLog(shaderProgram, out info);
|
|
|
|
|
Debug.WriteLine(info);
|
|
|
|
|
|
|
|
|
|
// Set clearcolor and bind the shader program.
|
|
|
|
|
GL.ClearColor(0.1f, 0.1f, 0.1f, 0.1f);
|
|
|
|
|
GL.UseProgram(shaderProgram);
|
|
|
|
|
// Set color to red. If the shader fails the fixed pipeline will be used and
|
|
|
|
|
// the lines will be red, if all is ok the fragment shader is used and they will be green.
|
|
|
|
|
GL.Color3(1.0f, 0, 0);
|
|
|
|
|
|
|
|
|
|
// Clean up resources. Note the program object is not deleted.
|
|
|
|
|
if (frag != 0)
|
|
|
|
|
GL.DeleteShader(frag);
|
|
|
|
|
if (vert != 0)
|
|
|
|
|
GL.DeleteShader(vert);
|
|
|
|
|
if (geom != 0)
|
|
|
|
|
GL.DeleteShader(geom);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Helper method to avoid code duplication.
|
|
|
|
|
/// Compiles a shader and prints results using Debug.WriteLine.
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="shader">A shader object, gotten from GL.CreateShader.</param>
|
|
|
|
|
/// <param name="source">The GLSL source to compile.</param>
|
|
|
|
|
void compileShader(int shader, string source)
|
|
|
|
|
{
|
|
|
|
|
GL.ShaderSource(shader, source);
|
|
|
|
|
GL.CompileShader(shader);
|
|
|
|
|
|
|
|
|
|
string info;
|
|
|
|
|
GL.GetShaderInfoLog(shader, out info);
|
|
|
|
|
Debug.WriteLine(info);
|
|
|
|
|
|
|
|
|
|
int compileResult;
|
|
|
|
|
GL.GetShader(shader, ShaderParameter.CompileStatus, out compileResult);
|
|
|
|
|
if (compileResult != 1)
|
|
|
|
|
{
|
|
|
|
|
Debug.WriteLine("Compile Error!");
|
|
|
|
|
Debug.WriteLine(source);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2009-10-21 13:51:39 +00:00
|
|
|
|
protected override void OnUnload(EventArgs e)
|
2009-09-03 19:01:11 +00:00
|
|
|
|
{
|
|
|
|
|
if (shaderProgram != 0)
|
|
|
|
|
GL.DeleteProgram(shaderProgram);
|
|
|
|
|
base.OnUnload(e);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Sets the viewport and projection matrix for orthographic projection.
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="e">resize event args</param>
|
|
|
|
|
protected override void OnResize(EventArgs e)
|
|
|
|
|
{
|
|
|
|
|
GL.Viewport(ClientRectangle);
|
|
|
|
|
|
|
|
|
|
// Set projection matrix
|
|
|
|
|
GL.MatrixMode(MatrixMode.Projection);
|
|
|
|
|
OpenTK.Matrix4 ortho = OpenTK.Matrix4.CreateOrthographicOffCenter(-1, 1, -1, 1, 1, -1);
|
|
|
|
|
GL.LoadMatrix(ref ortho);
|
|
|
|
|
|
|
|
|
|
// Set selector state back to matrix mode
|
|
|
|
|
GL.MatrixMode(MatrixMode.Modelview);
|
|
|
|
|
|
|
|
|
|
base.OnResize(e);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
protected override void OnUpdateFrame(FrameEventArgs e)
|
|
|
|
|
{
|
|
|
|
|
base.OnUpdateFrame(e);
|
|
|
|
|
|
|
|
|
|
if (Keyboard[Key.Space])
|
|
|
|
|
{
|
|
|
|
|
ErrorCode err = GL.GetError();
|
|
|
|
|
//Console.WriteLine(err + " " + Glu.ErrorString((GluErrorCode)err));
|
|
|
|
|
Console.WriteLine("GL error code: {0}", err);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (Keyboard[Key.Escape])
|
|
|
|
|
this.Exit();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
protected override void OnRenderFrame(FrameEventArgs e)
|
|
|
|
|
{
|
|
|
|
|
GL.Clear(ClearBufferMask.ColorBufferBit | ClearBufferMask.DepthBufferBit);
|
|
|
|
|
|
|
|
|
|
// draw two vertical lines
|
2014-02-22 19:55:34 +00:00
|
|
|
|
GL.Begin(PrimitiveType.Lines);
|
2009-09-03 19:01:11 +00:00
|
|
|
|
{
|
|
|
|
|
// line one
|
|
|
|
|
GL.Vertex2(-0.5f, -0.5f);
|
|
|
|
|
GL.Vertex2(-0.5f, 0.5f);
|
|
|
|
|
// line two
|
|
|
|
|
GL.Vertex2(0.5f, 0.5f);
|
|
|
|
|
GL.Vertex2(0.5f, -0.5f);
|
|
|
|
|
}
|
|
|
|
|
GL.End();
|
|
|
|
|
|
|
|
|
|
this.SwapBuffers();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#region public static void Main()
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Entry point of this example.
|
|
|
|
|
/// </summary>
|
|
|
|
|
[STAThread]
|
|
|
|
|
public static void Main()
|
|
|
|
|
{
|
|
|
|
|
using (SimpleGeometryShader example = new SimpleGeometryShader())
|
|
|
|
|
{
|
|
|
|
|
Utilities.SetWindowTitle(example);
|
|
|
|
|
example.Run(30.0, 0.0);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#endregion
|
|
|
|
|
}
|
2009-08-20 22:15:54 +00:00
|
|
|
|
}
|