Visual Studio2010与OpenGL基础绘制

2024-09-01 10:32

本文主要是介绍Visual Studio2010与OpenGL基础绘制,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

本例在Visual Studio 2010环境下使用OpenGL,提供一个基本的开发应用程序框架。

第一步:OpenGL基础设置:

1.创建一个单文档的MFC应用程序,命名为TestGL,然后在TestGLView.h头文件中添加以下两条include包含语句:

#include "gl/gl.h"
#include "gl/glu.h"

2.设置程序为静态运行方式

执行菜单命令:项目->属性,弹出属性对话框,选择配置属性->常规->MFC使用->在静态库中使用MFC,然后单击确定。这样做会使程序变得很大,但是它编译生成的可执行程序可以在其他计算机中运行。


3.链接OpenGL库文件

执行菜单命令:项目->属性打开属性对话框,选择配置属性->链接器->常规->附加库路径,输入OpenGL的库文件所在的路径,单击确定。比如我的机子如下:


再选择配置属性->链接器->输入->附加依赖库,在弹出的对话框中输入OpenGL的库文件OpenGL32.lib和glu32.lib,如下:


然后确定结束属性配置。编译运行,如果没有错误则表明配置正确。


第二步:在Visual Studio下用OpenGL绘制三棱锥:

绘制步骤如下:

*通过PIXELFORMATDESCRIPTOR结构设置设备描述表DC的像素格式和属性;

*创建渲染描述表RC,并和DC建立联系;

*使用OpenGL做图;

*释放所占用的资源,包括解除DC和RC的联系,删除RC及其与之关联的DC;

为应用程序添加变量和函数,对相关变量进行初始化,然后在函数中实现DC像素格式设置,RC的创建及其与DC 的关联,图形绘制以及资源释放等功能。

1.添加成员变量和成员函数:

为CTestGLView类添加公共变量:

CClientDC* pDrawDC;     // 用于指向当前DC的指针
通过向导自动添加后,会在CTestGLView类的构造函数中将其初始化为NULL:

CTestGLView::CTestGLView(): pDrawDC(NULL)
{// TODO: 在此处添加构造代码}

然后为CTestGLView添加三个公共成员函数:

void DrawGraphics(void);// 用于后续图形绘制
BOOL PixelformatSetting(void);	// 用于设置像素格式
void GLSetting(void);	// 用于创建渲染描述表
2.添加消息响应函数:

右击CTestGLView类选择属性命令,在属性对话框中选择消息页面。然后选择WM_CREATE消息,单击其后的下拉菜单按钮并选择"<添加>OnCreate"选项,为CTestGLView类添加消息响应函数OnCreate(),如下所示:

此时在TestGLView.cpp文件中可以看到添加消息的消息映射宏ON_WM_CREATE()和对应的OnCreate()函数。该消息响应在建立一个窗体前将被调用,因此可在其中做一些初始设置。

同样的方法添加另外两个消息响应函数:

OnDestroy():响应WM_DESTROY消息,宏为ON_WM_DESTROY(),窗口销毁时响应此函数,因此应该在此释放函数中所占用的资源。

OnSize():响应WM_SIZE消息,宏为ON_WM_SIZE()。改变窗口大小时响应此函数,因此可在此函数中调整视场。

3.为成员函数和消息响应函数添加代码:

1)设置像素格式。在BOOL CTestGLView::PixelformatSetting(void)函数中添加如下代码:

// 用于设置像素格式
BOOL CTestGLView::PixelformatSetting(void)
{static PIXELFORMATDESCRIPTOR pfd={sizeof(PIXELFORMATDESCRIPTOR),  //结构体长度1,    //版本PFD_DRAW_TO_WINDOW | PFD_SUPPORT_OPENGL | PFD_DOUBLEBUFFER, //在窗口中绘图|支持进行OpenGL调用|双缓存模式PFD_TYPE_RGBA,  //RGBA颜色模式24,  //使用24位颜色0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0 ,0,32,  //深度缓冲区大小0, 0,PFD_MAIN_PLANE,0, 0, 0, 0};  //填充PIXELFORMATDESCRIPTOR像素格式int ipixelformat;//获取最佳匹配的像素格式索引if((ipixelformat = ChoosePixelFormat(pDrawDC->GetSafeHdc(),&pfd)) == 0){MessageBox(_T("Choose pixel format failed!"));return TRUE;}//把DC的像素格式设置成由索引值ipixelformat指向的像素格式if(SetPixelFormat(pDrawDC->GetSafeHdc(),ipixelformat,&pfd) == FALSE){MessageBox(_T("Set pixel format failed !"));return FALSE;}return TRUE;
}

2).创建渲染描述表:

在函数void CTestGLView::GLSetting(void)中调用PixelformatSetting(void)设置像素格式,然后创建和DC关联的渲染描述表,代码如下:

// 用于创建渲染描述表
void CTestGLView::GLSetting(void)
{HGLRC hRC;   //渲染描述表句柄pDrawDC = new CClientDC(this);  //使pDrawDC指向当前DCASSERT(pDrawDC !=NULL);  //断言pDrawDC不为空if(!PixelformatSetting())  //设置像素格式return;//创建和当前DC兼容的RC,并和当前DC关联hRC = wglCreateContext(pDrawDC->GetSafeHdc());wglMakeCurrent(pDrawDC->GetSafeHdc(),hRC);
}

3).在消息响应函数OnCreate()函数中调用GLSetting()函数,使上述设置生效,代码如下:

int CTestGLView::OnCreate(LPCREATESTRUCT lpCreateStruct)
{if (CView::OnCreate(lpCreateStruct) == -1)return -1;// TODO:  Add your specialized creation code hereGLSetting();  //该函数用来创建渲染描述表return 0;
}

4).投影变换和视口变换。

在OnSize()中添加代码设置投影变换和视口变换,如下所示:

void CTestGLView::OnSize(UINT nType, int cx, int cy)
{CView::OnSize(nType, cx, cy);// TODO: Add your message handler code hereif(cy>0){glMatrixMode(GL_PROJECTION);  //启动投影矩阵,4×4glLoadIdentity();   //初始化为单位矩阵gluPerspective(45.0f,cx/cy,0.0f,30.0f);  //设置透视投影变换,指定视场glViewport(0, 0, cx, cy);  //设置视场,即定义显示范围}RedrawWindow();   //显示更新
}

5).绘制三棱锥:

在void CTestGLView::DrawGraphics(void)函数中,添加代码绘制3个三角形,构成一个三棱锥,如下:

void CTestGLView::DrawGraphics(void)
{glTranslatef(0.0f, 0.0f, -6.0f);  //移动物体到显示区glPolygonMode(GL_FRONT_AND_BACK,GL_LINE);   //以边线方式绘制三角形//绘制3个三角形的三个顶点glBegin(GL_TRIANGLES);glVertex3f(-0.6f, 0.0f, 0.0f);glVertex3f(0.6f, 0.0f, 0.0f);glVertex3f(0.0f, 0.15f, 0.6f);glVertex3f(-0.6f, 0.0f, 0.0f);glVertex3f(0.0f, 0.15f, 0.6f);glVertex3f(0.0f, 0.9f, 0.6f);glVertex3f(0.0f, 0.9f, 0.6f);glVertex3f(0.0f, 0.15f, 0.6f);glVertex3f(0.6f, 0.0f, 0.0f);glEnd();
}

6).在OnDraw()函数中设置背景并调用DrawGraphics()函数绘制图形。

通常使用CView类的成员函数OnDraw()绘制用户界面,此外还可以在该函数中添加代码完成背景色设置,然后调用DrawGraphics()函数完成绘制。

void CTestGLView::OnDraw(CDC* /*pDC*/)
{CTestGLDoc* pDoc = GetDocument();ASSERT_VALID(pDoc);if (!pDoc)return;// TODO: 在此处为本机数据添加绘制代码static BOOL bBusy = FALSE;  //定义开关变量//绘制完成后才可以更新缓存if(bBusy)return;bBusy =TRUE;glClearColor(0.0f, 0.0f, 0.5f, 0.5f);  //设置背景颜色glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT);  //清除深度缓存和颜色缓存glMatrixMode(GL_MODELVIEW);  //启动模型矩阵glLoadIdentity();   //初始化为单位矩阵DrawGraphics();   //绘制图形SwapBuffers(wglGetCurrentDC());  //更新缓存bBusy=FALSE;
}

7).删除渲染描述表。

在消息响应函数OnDestroy()中删除渲染描述表及其绑定的设备描述表,代码如下:

void CTestGLView::OnDestroy()
{CView::OnDestroy();// TODO: Add your message handler code hereHGLRC hRC;hRC=::wglGetCurrentContext();   //获取当前RC句柄::wglMakeCurrent(NULL,NULL);  //解除当前RC和DC的关联,并把当前RC非当前化if(hRC){::wglDeleteContext(hRC);  //删除RC}if(pDrawDC){delete pDrawDC;   //删除DC}
}

编译运行项目,如果没有错,可得结果如下:

4.添加旋转功能:

实现一个交互功能,即单击鼠标开始/停止三棱锥旋转。

1).添加消息响应函数。

为CTestGLView类添加鼠标左键消息WM_LBUTTONDOWN和定时器消息WM_TIME,鼠标左键消息响应函数OnLButtonDown(),宏ON_WM_LBUTTONDOWN()。定时器消息响应函数为OnTimer(),宏ON_WM_TIMER().
2).为CTestGLView类添加两个共有属性的成员变量,

BOOL bRotate;   // 控制图形旋转
float RotateAngle;   // 旋转角度
Visual Studio会自动在CTestGLView类的构造函数中将bRotate初始化为FALSE,将RotateAngle初始化为0.手动将bRotate的初始值改为TRUE。

3).在void CTestGLView::OnLButtonDown(UINT nFlags, CPoint point)函数中添加如下代码,通过单击启动/停止定时器,

void CTestGLView::OnLButtonDown(UINT nFlags, CPoint point)
{// TODO: Add your message handler code here and/or call defaultif(bRotate){SetTimer(1,100,NULL);   //设置定时器1,100ms触发一次}else{KillTimer(1);  //移除定时器1}bRotate = !bRotate;   //更新bRotate的值CView::OnLButtonDown(nFlags, point);
}

4).设置旋转角度。在定时器消息响应函数OnTimer()中添加如下代码,实现每次定时到达时都能将图形旋转10度

void CTestGLView::OnTimer(UINT_PTR nIDEvent)
{// TODO: Add your message handler code here and/or call default//若定时器1到达预定时刻,则旋转角度增加10度if(nIDEvent == 1){RotateAngle += 10.0f;Invalidate(FALSE);  //使当前窗口失效,重新绘制}CView::OnTimer(nIDEvent);
}
在OnTimer()函数中使用的Invalidate()函数作用是将整个客户区失效,从而被重新绘制,而最终的重绘工作由OnDraw()函数完成。此处用函数Invalidate()函数调用了OnDraw()函数来绘制旋转后的图形。

5).每次按照设置的角度重绘图形。重绘窗口由OnDraw()函数完成,而OnDraw()函数调用DrawGraphics()函数绘制图形。因此在函数DrawGraphics()函数中需添加如下代码:

glRotated(RotateAngle,1.0, 1.0, 1.0);  //按设定的角度RotateAngle旋转图形
将这句代码放在glTranslatef()函数语句之后。使得每次重绘时将图形绕着从原点指向(1.0,1.0,1.0)的射线旋转RotateAngle的角度,而RotateAngle的值在定时器函数OnTimer()中更新。这样就可以实现每次单击鼠标左键,定时器开始计时,每个100ms后响应定时器消息的函数OnTimer()将RotateAngle的值增加10度,同时Invalidate()函数使界面失效而被重绘,重绘是将图形选择RotateAngle的角度,有时间间隔短,所以看上去就会产生三棱锥旋转的效果。当鼠标在此单击左键时,定时器被取消,图形停止旋转。



参考文献:张俊华.<医学图像三维重建和可视化——VC++实现>.科学出版社


这篇关于Visual Studio2010与OpenGL基础绘制的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1126681

相关文章

MySQL中my.ini文件的基础配置和优化配置方式

《MySQL中my.ini文件的基础配置和优化配置方式》文章讨论了数据库异步同步的优化思路,包括三个主要方面:幂等性、时序和延迟,作者还分享了MySQL配置文件的优化经验,并鼓励读者提供支持... 目录mysql my.ini文件的配置和优化配置优化思路MySQL配置文件优化总结MySQL my.ini文件

使用Python绘制蛇年春节祝福艺术图

《使用Python绘制蛇年春节祝福艺术图》:本文主要介绍如何使用Python的Matplotlib库绘制一幅富有创意的“蛇年有福”艺术图,这幅图结合了数字,蛇形,花朵等装饰,需要的可以参考下... 目录1. 绘图的基本概念2. 准备工作3. 实现代码解析3.1 设置绘图画布3.2 绘制数字“2025”3.3

使用Python绘制可爱的招财猫

《使用Python绘制可爱的招财猫》招财猫,也被称为“幸运猫”,是一种象征财富和好运的吉祥物,经常出现在亚洲文化的商店、餐厅和家庭中,今天,我将带你用Python和matplotlib库从零开始绘制一... 目录1. 为什么选择用 python 绘制?2. 绘图的基本概念3. 实现代码解析3.1 设置绘图画

Python绘制土地利用和土地覆盖类型图示例详解

《Python绘制土地利用和土地覆盖类型图示例详解》本文介绍了如何使用Python绘制土地利用和土地覆盖类型图,并提供了详细的代码示例,通过安装所需的库,准备地理数据,使用geopandas和matp... 目录一、所需库的安装二、数据准备三、绘制土地利用和土地覆盖类型图四、代码解释五、其他可视化形式1.

如何用Python绘制简易动态圣诞树

《如何用Python绘制简易动态圣诞树》这篇文章主要给大家介绍了关于如何用Python绘制简易动态圣诞树,文中讲解了如何通过编写代码来实现特定的效果,包括代码的编写技巧和效果的展示,需要的朋友可以参考... 目录代码:效果:总结 代码:import randomimport timefrom math

零基础学习Redis(10) -- zset类型命令使用

zset是有序集合,内部除了存储元素外,还会存储一个score,存储在zset中的元素会按照score的大小升序排列,不同元素的score可以重复,score相同的元素会按照元素的字典序排列。 1. zset常用命令 1.1 zadd  zadd key [NX | XX] [GT | LT]   [CH] [INCR] score member [score member ...]

【WebGPU Unleashed】1.1 绘制三角形

一部2024新的WebGPU教程,作者Shi Yan。内容很好,翻译过来与大家共享,内容上会有改动,加上自己的理解。更多精彩内容尽在 dt.sim3d.cn ,关注公众号【sky的数孪技术】,技术交流、源码下载请添加微信号:digital_twin123 在 3D 渲染领域,三角形是最基本的绘制元素。在这里,我们将学习如何绘制单个三角形。接下来我们将制作一个简单的着色器来定义三角形内的像素

如何在Visual Studio中调试.NET源码

今天偶然在看别人代码时,发现在他的代码里使用了Any判断List<T>是否为空。 我一般的做法是先判断是否为null,再判断Count。 看了一下Count的源码如下: 1 [__DynamicallyInvokable]2 public int Count3 {4 [__DynamicallyInvokable]5 get

【Linux 从基础到进阶】Ansible自动化运维工具使用

Ansible自动化运维工具使用 Ansible 是一款开源的自动化运维工具,采用无代理架构(agentless),基于 SSH 连接进行管理,具有简单易用、灵活强大、可扩展性高等特点。它广泛用于服务器管理、应用部署、配置管理等任务。本文将介绍 Ansible 的安装、基本使用方法及一些实际运维场景中的应用,旨在帮助运维人员快速上手并熟练运用 Ansible。 1. Ansible的核心概念

Flutter 进阶:绘制加载动画

绘制加载动画:由小圆组成的大圆 1. 定义 LoadingScreen 类2. 实现 _LoadingScreenState 类3. 定义 LoadingPainter 类4. 总结 实现加载动画 我们需要定义两个类:LoadingScreen 和 LoadingPainter。LoadingScreen 负责控制动画的状态,而 LoadingPainter 则负责绘制动画。