简介
the Open Tool Kit (OpenTK), 是对 OpenGL、OpenAL、OpenCL 的跨平台的封装,使用 C# 编写,它可以用在Mono、dotNet的语言:c#、VB、C++/CLI、F#、Boo等。
什么是Mono
参考:http://baike.baidu.com/subview/26639/9339264.htm?fr=aladdin
Mono是一个由Novell公司(由Ximian发起,并由Miguel de lcaza领导的,一个致力于开创.NET在Linux上使用的开源工程。它包含了一个C#语言的编译器,一个CLR的运行时,和一组类库,并实现了 ADO NET和ASP NET。能够使得开发人员在Linux用C#开发程序。)主持的项目.该项目的目标是创建一系列符合标准ECMA (Ecma-334和Ecma-335)的.Net 工具, 包括C #编译器和共同语言(CL 即 Common Language)执行平台(Platform).与微软的.Net不同, Mono项目不仅可以运行于Windows系统内,还可以运行于Linux, FreeBSD, Unix, Mac OS X和Solaris.
什么是C++/CLI
参考:http://baike.baidu.com/link?url=QPwZ5Jga7wKJJKn_NWkbvIucYtgvCHtZ3UZ9sRKD7i-eNGXg8m0nYpWVNSyt0yjQ
简而言之就是如何用C++在·NET中编程
什么是OpenAL
参考:http://baike.baidu.com/view/1355367.htm?fr=aladdin
是开源的跨平台音效API
什么是OpenCL
参考:http://baike.baidu.com/view/2056591.htm
是第一个面向异构系统通用目的并行编程的开放式、免费标准,也是一个统一的编程环境
使用
环境
dotNet IDE可以使用Visual Studio或 MonoDevelop http://monodevelop.com/Download
添加OpenTK.dll:右键“Project”,选择“Add Reference”,找到OpenTK.dll并添加它。
Windows.Forms+GLControl
参考:http://www.opentk.com/doc/chapter/2/glcontrol
向工具箱中添加GLControl控件
在工具箱的空白处右键,选择“Choose Item……”,浏览到OpenTK.GLControl.dll,添加。
创建顺序
he fact that glControl1
'sGLContext
is created in runtime is important to remember however, since you cannot
access or changeglControl1
's properties reliably until theGLContext
has been created. The same is true for anyGL.*
commands (orGlu
for that matter!).
1.运行Windows.Form的构造函数
2.加载form的事件
3.加载GLControl的事件,OK to touch glControl/GL
4.运行事件处理函数 ,ny event handler may touch glControl/GL
.
解决这个问题的一种方法是定义bool loaded=false;在GLControl加载时设置为true。
- using OpenTK.Graphics;
- using OpenTK.Graphics.OpenGL;
- public partial class Form1 : Form
- {
- bool loaded = false;
- public Form1()
- {
- InitializeComponent();
- }
- private void glControl1_Load(object sender, EventArgs e)
- {
- loaded = true;
- }
- }
由OpenTK还没有开发GLControl.Load事件,所以可以使用Form.Load事件
然后在你想使用glControl/GL的事件处理函数中检查loaded
- private void glControl1_Resize(object sender, EventArgs e)
- {
- if (!loaded)
- return;
- }
在VS中事件处理函数是很简单的:
1.在Designer中选择GLControl
2.在属性框中选择事件
3.在相应的事件中设置处理函数
- private void glControl1_Paint(object sender, PaintEventArgs e)
- {
- if (!loaded) // Play nice
- return;
- GL.Clear(ClearBufferMask.ColorBufferBit | ClearBufferMask.DepthBufferBit);
- glControl1.SwapBuffers();
- }
视口初始
- private void glControl1_Load(object sender, EventArgs e)
- {
- loaded = true;
- GL.ClearColor(Color.SkyBlue);
- SetupViewport();
- }
- private void SetupViewport()
- {
- int w = glControl1.Width;
- int h = glControl1.Height;
- GL.MatrixMode(MatrixMode.Projection);
- GL.LoadIdentity();
- GL.Ortho(0, w, 0, h, -1, 1); // Bottom-left corner pixel has coordinate (0, 0)
- GL.Viewport(0, 0, w, h); // Use all of the glControl painting area
- }
键盘输入
有两种常用的方法:(1)使用Windows.Forms的按键事件(2)使用OpenTK的KeyboardDevice。
glControl1
is not painted all the time,操作系统的窗口管理器会确保绘制事件尽可能少的发生,一般只有在resize、窗口模糊等情况下才会触发绘制事件。如果我们想手动调用重绘,使用Invalidate()
- private void glControl1_KeyDown(object sender, KeyEventArgs e)
- if (!loaded)
- return;
- if (e.KeyCode == Keys.Space)
- {
- x++;
- glControl1.Invalidate();
- }
窗口大小的改变
当GLControl的大小发生变化,我们需要更新的是viewport和projection matrix。
如果我们从右下角缩小窗口时,并不会解发重绘,这是因为窗口管理器认为(0,0)像素还存在(所以如果从左上角缩小就会重绘),解决的方法就是调用Invalidate()
- private void glControl1_Resize(object sender, EventArgs e)
- {
- SetupViewport();
- glControl1.Invalidate();
- }
添加动画
有两种方法:(1)添加一个Timer 控件,使用的其Tick事件(2)使用Thread。
这里使用第三种方式,使用Windows.Forms的Application.idle事件
- private void glControl1_Load(object sender, EventArgs e)
- {
- loaded = true;
- GL.ClearColor(Color.SkyBlue);
- SetupViewport();
- Application.Idle += Application_Idle; // press TAB twice after +=
- }
- void Application_Idle(object sender, EventArgs e)
- {
- }
当窗口比较大时,渲染(render)比较慢
原因是窗口的3d rendering比full-screen rendering一般要慢。那么可以使用一种frame-rate independent animation的技术,思想很简单:根据当前的渲染速度调整变化量,比如旋转量(渲染慢的使用的量大)。可以使用StopWatch来测量一个frame的渲染时间。StopWatch的用法:
- Stopwatch sw = new Stopwatch();
- sw.Start();
- MyAdvancedAlgorithm();
- sw.Stop();
- double milliseconds = sw.Elapsed.TotalMilliseconds;
(不要使用DataTime.Now,因为它的大小是10ms级的,和frame的渲染时间是一个数量级)
- Stopwatch sw = new Stopwatch(); // available to all event handlers
- private void glControl1_Load(object sender, EventArgs e)
- {
- ...
- sw.Start(); // start at application boot
- }
- float rotation = 0;
- void Application_Idle(object sender, EventArgs e)
- {
- // no guard needed -- we hooked into the event in Load handler
- sw.Stop(); // we've measured everything since last Idle run
- double milliseconds = sw.Elapsed.TotalMilliseconds;
- sw.Reset(); // reset stopwatch
- sw.Start(); // restart stopwatch
- // increase rotation by an amount proportional to the
- // total time since last Idle run
- float deltaRotation = (float)milliseconds / 20.0f;
- rotation += deltaRotation;
- glControl1.Invalidate();
- }
FPS计数器
我们怎么知道1s已经过去了呢?
- void Application_Idle(object sender, EventArgs e)
- {
- double milliseconds = ComputeTimeSlice();
- Accumulate(milliseconds);
- Animate(milliseconds);
- }
- private double ComputeTimeSlice()
- {
- sw.Stop();
- double timeslice = sw.Elapsed.TotalMilliseconds;
- sw.Reset();
- sw.Start();
- return timeslice;
- }
- float rotation = 0;
- private void Animate(double milliseconds)
- {
- float deltaRotation = (float)milliseconds / 20.0f;
- rotation += deltaRotation;
- glControl1.Invalidate();
- }
- double accumulator = 0;
- int idleCounter = 0;
- private void Accumulate(double milliseconds)
- {
- idleCounter++;
- accumulator += milliseconds;
- if (accumulator > 1000)
- {
- label1.Text = idleCounter.ToString();
- accumulator -= 1000;
- idleCounter = 0; // don't forget to reset the counter!
- }
- }
使用多个GLControl
假如说我们有glCtrl1和glCtrl2,想要在Paint事件处理函数中使用它们,具体看代码:
- private void glCtrl1_Paint(object sender, PaintEventArgs e)
- if (!loaded)
- return;
- glCtrl1.MakeCurrent(); // Ohh.. It's that simple?
- GL.Clear(ClearBufferMask.ColorBufferBit | ClearBufferMask.DepthBufferBit);
- ...
- }
尽管每个GLControl有它们自己的GraphicsContext,OpenTK默认是共享OpenGL资源的。你也可以禁止这个行为,能过设置属性GraphicsContext.ShareContexts。
参考:http://www.cnblogs.com/beginor/archive/2009/10/17/1585040.html