1、基于MFC的OpenGL程序

1、基于MFC的OpenGL程序

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

 
一、单文本文件  
工程openGLMFC
1、创建单文本文件
 
<span>1、基于MFC的OpenGL程序</span>
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) == -1)
        return -1;

    // TODO:  Add your specialized creation code here
    InitializeOpenGL();
    return 0;
}

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 == 0)
    {
        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
        1,                              // version number
        PFD_DRAW_TO_WINDOW |            // support window
        PFD_SUPPORT_OPENGL |            // support OpenGL
        PFD_DOUBLEBUFFER,                // double buffered
        PFD_TYPE_RGBA,                  // RGBA type
        24,                             // 24-bit color depth
        0, 0, 0, 0, 0, 0,               // color bits ignored
        0,                              // no alpha buffer
        0,                              // shift bit ignored
        0,                              // no accumulation buffer
        0, 0, 0, 0,                     // accum bits ignored
        16,                             // 16-bit z-buffer
        0,                              // no stencil buffer
        0,                              // no auxiliary buffer
        PFD_MAIN_PLANE,                 // main layer
        0,                              // reserved
        0, 0, 0                         // layer masks ignored
    };
    int m_nPixelFormat = ::ChoosePixelFormat(m_pDC->GetSafeHdc(), &pfd);
    if (m_nPixelFormat == 0)
    {
        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(0, 0) == 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 (0 >= cx || 0 >= cy)
    {
        return;
    }
    // select the full client area
    ::glViewport(0, 0, 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();
}

通过上面得到运行结果:

<span>1、基于MFC的OpenGL程序</span>

二、基于对话框

 
工程:openGLMFCDialog
1-3步骤同上
4、变量函数声明
在openGLMFCDialogDlg.h文件中,声明如上。
其余步骤同上,不同点在于,需要将OnDraw里面的写到
OnPaint函数里面,其中IsIconic()是判断是否为最小化,可以放到else里面即可。
得到结果
<span>1、基于MFC的OpenGL程序</span>

 
版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 举报,一经查实,本站将立刻删除。

发布者:全栈程序员-用户IM,转载请注明出处:https://javaforall.cn/119499.html原文链接:https://javaforall.cn

【正版授权,激活自己账号】: Jetbrains全家桶Ide使用,1年售后保障,每天仅需1毛

【官方授权 正版激活】: 官方授权 正版激活 支持Jetbrains家族下所有IDE 使用个人JB账号...

(0)


相关推荐

  • SAE J1939协议学习笔记

    SAE J1939协议学习笔记引用百度百科:SAEJ1939(以下简称J1939)是美国汽车工程协会(SAE)的推荐标准,用于为中重型道路车辆上电子部件间的通讯提供标准的体系结构。它由“卡车与大型客车电气与电子委员会”(Track&BusElectrical&ElectronicsCommittee)下属的“卡车与大型客车控制和通讯网络附属委员会”(Track&BusContro…

  • 线程池学习使用

    线程池学习使用线程池学习使用

  • 自动编码器重建图像及Python实现

    自动编码器重建图像及Python实现自动编码器简介自动编码器(一下简称AE)属于生成模型的一种,目前主流的生成模型有AE及其变种和生成对抗网络(GANs)及其变种。随着深度学习的出现,AE可以通过网络层堆叠形成深度自动编码器来实现数据降维。通过编码过程减少隐藏层中的单元数量,可以以分层的方式实现降维,在更深的隐藏层中获得更高级的特征,从而在解码过程中更好的重建数据。自动编码器原理自动编码器是通过无监督学习训练的神经网络,实际上…

  • let’s encrypt免费证书 制作 HTTPS证书[通俗易懂]

    let’s encrypt免费证书 制作 HTTPS证书

  • 学生成绩管理系统数据库设计–MySQL/SQL Server[通俗易懂]

    学生成绩管理系统数据库设计–MySQL/SQL Server[通俗易懂]MySQL/SQLServer数据库设计(学生成绩管理系统)设计大纲1.项目背景及需求分析1.1项目背景1.2 需求分析1.2.1 信息需求1.2.2 功能需求1.2.3 安全性与完整性需求2. 概念结构设计2.1抽象出系统实体2.2全局E-R图3. 逻辑结构设计3.1关系模式3.2函数依赖识别3.3范式4. 物理设计和实施4.1数据库及表创建4.2表数据增删改查测试4.3创建视图4.4创建函数4.5创建存储过程4.6创建触发器

  • 海思Hi3798处理器参数,Hi3798芯片详细信息介绍

    海思Hi3798处理器参数,Hi3798芯片详细信息介绍Hi3798CV200集成4核64位高性能CortexA53处理器、内置NEON加速引擎,强大的CPU处理能力可以满足各种差异化的业务需求。在码流兼容性、在线视频播放的流畅性、图像质量以及整机性能方面保持业界最好的用户体验。支持4K2KP60@10bit超高清视频解码和显示,支持H.265/HEVC、H.264/AVC、AVS+、MVC、MPEG2、MPEG4、VC-1、VP6、VP…

发表回复

您的电子邮箱地址不会被公开。

关注全栈程序员社区公众号