原贴地址:https://www.cnblogs.com/pinking/p/6180225.html

首先,使用的库是GLUT以及GLAUX,先下载两者,添加查找路径以及链接

 
一、单文本文件  
工程openGLMFC
1、创建单文本文件
 
 
 
 
2、添加路径、链接
方法如之前篇章所示,
链接库为opengl32.lib ;glu32.lib ;glut32.lib ;glaux.lib
 
3、头文件
在stdafx.h中加入下列语句:
//openGL 所需要的头文件
#include <gl/gl.h>
#include <gl/glu.h>
#include <gl/glut.h>
#include <gl/glaux.h>
4、设置窗口风格
将窗口风格设为WS_CLIPCHILDREN和 WS_CLIPSIBLINGS,从而避免OpenGL绘制到其他窗口中去。这些应该放在PreCreateWindow()中。
BOOL CopenGLMFCView::PreCreateWindow(CREATESTRUCT& cs)
{
// TODO: Modify the Window class or styles here by modifying
// the CREATESTRUCT cs
cs.style |= WS_CLIPSIBLINGS | WS_CLIPCHILDREN;
return CView::PreCreateWindow(cs);
}
5、变量、函数声明
初始化GL,考虑到使用设备上下文、绘图上下文,
先在openGLMFCView.h中声明需要的变量以及函数
HGLRC m_hRC;    //RC 绘图上下文
CDC* m_pDC; //DC 设备上下文
BOOL InitializeOpenGL(); //初始化 OpenGL
BOOL SetupPixelFormat(); //为 DC 建立像素格式
6、创建消息响应函数
初始化需要在WM_CREATE触发
资源释放在WM_DESTROY触发
窗口大小变化时 WM_SIZE触发,需要调整绘图
由于使用opengl绘制背景,故不需要窗口自己在绘制背景了,需要改写 WM_ERASEBACKGROUND消息触发的事件
 
故,打开classWizard,在openGLMFCView类中添加上面四个消息响应函数

OnCreate   OnDestroy  OnSize  OnEraseBkground

7、初始化

将通过建立像素格式和绘制上下文来初始化OpenGL. 在InitializeOpenGL()中会创建一个设备上下文(DC),为这个DC选择一个像素格式,创建和这个DC相关的绘制上下文(RC),然后选择这个RC.这个函数会调用SetupPixelFormat()来建立像素格式。

int CopenGLMFCView::OnCreate(LPCREATESTRUCT lpCreateStruct)
{
if (CView::OnCreate(lpCreateStruct) == -)
return -; // TODO: Add your specialized creation code here
InitializeOpenGL();
return ;
} BOOL CopenGLMFCView::InitializeOpenGL()
{
//客户区获得DC
m_pDC = new CClientDC(this);
//Failure to Get DC
if (m_pDC == NULL)
{
MessageBox(L"Error Obtaining DC");
return FALSE;
}
//为DC建立像素格式
if (!SetupPixelFormat())
{
return FALSE;
}
//创建 RC
m_hRC = ::wglCreateContext(m_pDC->GetSafeHdc());
//Failure to Create Rendering Context
if (m_hRC == )
{
MessageBox(L"Error Creating RC");
return FALSE;
}
//设定OpenGL当前线程的渲染环境。
//以后这个线程所有的OpenGL调用都是在这个hdc标识的设备上绘制。
if (::wglMakeCurrent(m_pDC->GetSafeHdc(), m_hRC) == FALSE)
{
MessageBox(L"Error making RC Current");
return FALSE;
}
//背景颜色
::glClearColor(0.0f, 0.0f, 0.0f, 0.0f);
//深度缓存 1最大,让任何都能显示出来
::glClearDepth(1.0f);
//如果通过比较后深度值发生变化了,会进行更新深度缓冲区的操作
::glEnable(GL_DEPTH_TEST);
return TRUE;
}
//建立像素格式
/////////////////////////////////////////////////////////////////////////////
BOOL CopenGLMFCView::SetupPixelFormat()
{
static PIXELFORMATDESCRIPTOR pfd =
{
sizeof(PIXELFORMATDESCRIPTOR), // size of this pfd
, // version number
PFD_DRAW_TO_WINDOW | // support window
PFD_SUPPORT_OPENGL | // support OpenGL
PFD_DOUBLEBUFFER, // double buffered
PFD_TYPE_RGBA, // RGBA type
, // 24-bit color depth
, , , , , , // color bits ignored
, // no alpha buffer
, // shift bit ignored
, // no accumulation buffer
, , , , // accum bits ignored
, // 16-bit z-buffer
, // no stencil buffer
, // no auxiliary buffer
PFD_MAIN_PLANE, // main layer
, // reserved
, , // layer masks ignored
};
int m_nPixelFormat = ::ChoosePixelFormat(m_pDC->GetSafeHdc(), &pfd);
if (m_nPixelFormat == )
{
return FALSE;
}
if (::SetPixelFormat(m_pDC->GetSafeHdc(), m_nPixelFormat, &pfd) == FALSE)
{
return FALSE;
}
return TRUE;
}

8、绘制场景

在绘制场景时,一般包括如下步骤:1)清空缓存。2)绘制场景。3)Flush掉渲染流水线。4)若设置了双缓冲,则交换前后台缓冲区。

void CopenGLMFCView::OnDraw(CDC* /*pDC*/)
{
CopenGLMFCDoc* pDoc = GetDocument();
ASSERT_VALID(pDoc);
if (!pDoc)
return; // TODO: add draw code for native data here // 清除颜色、深度缓存
::glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT); //可以添加渲染函数 // Flush掉渲染流水线
::glFinish();
// 交换前后缓存区
::SwapBuffers(m_pDC->GetSafeHdc()); }

9、背景绘制修改

 

试试改变窗口的大小,你会看到很严重的闪烁,并且关闭程序后会报告内存泄露,因此我们这就来解决这两个问题吧。

发生闪烁的原因是Windows先绘制背景,然后再是OpenGL绘制,因为我们已经让OpenGL负责清空背景色,因此我们不需要Windows去清空背景了

BOOL CopenGLMFCView::OnEraseBkgnd(CDC* pDC)
{
// TODO: Add your message handler code here and/or call default return TRUE;//CView::OnEraseBkgnd(pDC);
}

内存泄露的原因是我们在InitializeOpenGL()中使用了new运算符来为CClientDC对象分配内存,因此需要显示delete掉。

void CopenGLMFCView::OnDestroy()
{
CView::OnDestroy(); // TODO: Add your message handler code here
if (::wglMakeCurrent(, ) == FALSE)
{
MessageBox(L"Could not make RC non-current");
} //Delete the rendering context
if (::wglDeleteContext(m_hRC) == FALSE)
{
MessageBox(L"Could not delete RC");
}
//Delete the DC
if (m_pDC)
{
delete m_pDC;
}
//Set it to NULL
m_pDC = NULL;
}
10、大小调整
在OnSize()中一般用来设置视口和视锥,因为这些是和窗口大小相关的。基本操作包括设置视口,选择投影矩阵,设置模型视图矩阵。
void CopenGLMFCView::OnSize(UINT nType, int cx, int cy)
{
CView::OnSize(nType, cx, cy); // TODO: Add your message handler code here
GLdouble aspect_ratio; // width/height ratio if ( >= cx || >= cy)
{
return;
}
// select the full client area
::glViewport(, , cx, cy);
// compute the aspect ratio
// this will keep all dimension scales equal
aspect_ratio = (GLdouble)cx / (GLdouble)cy;
// select the projection matrix and clear it
::glMatrixMode(GL_PROJECTION);
::glLoadIdentity();
// select the viewing volume
::gluPerspective(45.0f, aspect_ratio, .01f, 200.0f); // switch back to the modelview matrix and clear it
::glMatrixMode(GL_MODELVIEW);
::glLoadIdentity();
}

二、基于对话框

 
工程:openGLMFCDialog
1-3步骤同上
4、变量函数声明
在openGLMFCDialogDlg.h文件中,声明如上。
其余步骤同上,不同点在于,需要将OnDraw里面的写到
OnPaint函数里面,其中IsIconic()是判断是否为最小化,可以放到else里面即可。
得到结果
-----------------------------------------------------------------------------------------------------------------
原贴地址:https://www.cnblogs.com/pinking/p/6180225.html

基于MFC的OpenGL程序<转>的更多相关文章

  1. 1、基于MFC的OpenGL程序

    首先,使用的库是GLUT以及GLAUX,先下载两者,添加查找路径以及链接   一.单文本文件   工程openGLMFC 1.创建单文本文件   2.添加路径.链接 方法如之前篇章所示, 链接库为op ...

  2. 基于MFC的Opengl实现动画

    对于了解MFC程序设计的来说,就太简单了.像我这种的,还是有必要记下来. OnCreate设置定时:SetTimer(1, 10, NULL);//设置#1定时器 key point void COp ...

  3. 基于MFC和opencv的FFT

    在网上折腾了一阵子,终于把这个程序写好了,程序是基于MFC的,图像显示的部分和获取图像的像素点是用到了opencv的一些函数,不过FFT算法没有用opencv的(呵呵,老师不让),网上的二维的FFT程 ...

  4. 基于MFC的实时可视化项目中视图刷新消息的严谨使用

    在实时可视项目中,视图的实时刷新显示对软件的体验感来说非常重要,当算法的效率达到实时,比如一秒40帧,如果实时显示帧率更不上,则体验感将大大折扣,让用户感觉你的算法并没有40帧,当然最关键的是解决显示 ...

  5. MFC下OpenGL入门(可以用)

    MFC下OpenGL入门 源文件 1, 建一工程文件,我这里命名为first,现在first工程里面我们没有添加任何东西,所有的东西都是MFC自动帮我们创建的. 2, 添加链接库.这一步很关键.打开菜 ...

  6. 基于MFC的socket编程(异步非阻塞通信)

       对于许多初学者来说,网络通信程序的开发,普遍的一个现象就是觉得难以入手.许多概念,诸如:同步(Sync)/异步(Async),阻塞(Block)/非阻塞(Unblock)等,初学者往往迷惑不清, ...

  7. 用MFC实现OpenGL编程

    一.OpenGL简介 众所周知,OpenGL原先是Silicon Graphics Incorporated(SGI公司)在他们的图形工作站上开发高质量图像的接口.但最近几年它成为一个非常优秀的开放式 ...

  8. VS2012下基于Glut 矩阵变换示例程序2:

    在VS2012下基于Glut 矩阵变换示例程序:中我们在绘制甜圈或者圆柱时使用矩阵对相应的坐标进行变换后自己绘制甜圈或者圆柱.我们也可以使用glLoadMatrixf.glLoadMatrixd载入变 ...

  9. 【OpenGL】详解第一个OpenGL程序

    写在前面 OpenGL能做的事情太多了!很多程序也看起来很复杂.很多人感觉OpenGL晦涩难懂,原因大多是被OpenGL里面各种语句搞得头大,一会gen一下,一会bind一下,一会又active一下. ...

随机推荐

  1. 1122 Hamiltonian Cycle (25 分)

    1122 Hamiltonian Cycle (25 分) The "Hamilton cycle problem" is to find a simple cycle that ...

  2. [UE4]机器人射击逻辑行为树

    1.寻找敌人 2.如果没有找到,等待2秒,跳转到1 3.如果找到了,走向敌人 4.走向敌人的过程中,如果看见敌人了,则射击 5.如果敌人没有了,则跳转到1

  3. Spring+MyBatis实现数据库读写分离方案

    推荐第四种:https://github.com/shawntime/shawn-rwdb 方案1 通过MyBatis配置文件创建读写分离两个DataSource,每个SqlSessionFactor ...

  4. CentOS之文档的压缩与打包

    .rar压缩文件linux中不识别,.zip在windows和Linux中动能使用. .gz:由gzip压缩工具压缩的文件 .bz2:bzip2压缩工具压缩的文件 .tar:由tar打包程序打包的文件 ...

  5. Hadoop 2.x常用端口及查看方法

    Hadoop集群的各部分一般都会使用到多个端口,有些是daemon之间进行交互之用,有些是用于RPC访问以及HTTP访问.而随着Hadoop周边组件的增多,完全记不住哪个端口对应哪个应用,特收集记录如 ...

  6. CS229 6.16 Neurons Networks linear decoders and its implements

    Sparse AutoEncoder是一个三层结构的网络,分别为输入输出与隐层,前边自编码器的描述可知,神经网络中的神经元都采用相同的激励函数,Linear Decoders 修改了自编码器的定义,对 ...

  7. 解决IIS7下主机名灰色无法修改问题

    打开IIS ,找到网站,右击编辑绑定时: 关于导入iis   .pfx格式的证书后,编辑绑定时,主机名为灰色的问题 解决方法 : (1)打开C:\Windows\system32\inetsrv\co ...

  8. linux 乌班图 安装pycharm

    1.通过vmware安装ubuntu系统2.安装完成后,登录ubuntu,通过普通用户 s14登录,密码redhat3.下载pycharm到ubuntu系统中 -可以通过python -m http. ...

  9. 三种常见的Web安全问题

    XSS漏洞 1.XSS简介 跨站脚本(cross site script)简称为XSS,是一种经常出现在web应用中的计算机安全漏洞,也是web中最主流的攻击方式. XSS是指恶意攻击者利用网站没有对 ...

  10. 【Selenium-WebDriver自学】Selenium-IDE调试(四)

    ==================================================================================================== ...