图形编程

OSG环境搭建(OSG3.6.2+VS2017+Qt5.10.1

2018-07-30  本文已影响3814人  downdemo

编译OSG3.6.2

  1. 编译过程参考Compiling with Visual Studio,VS要求已安装MFC,需要的安装包可在osgChinaOSG官网获取。以下是此次编译过程中所用的工具版本及其下载地址:
  1. 新建E:\OSG,将上述前三个包解压到此目录
E:\OSG
  1. 为了后续方便,将解压后的目录分别改名为3rdPartyDataOpenSceneGraph
改名后
  1. E:\OSG\OpenSceneGraph新建一个build文件夹用于存放编译内容
新建一个build文件夹
  1. 安装cmake-3.12.0-win64-x64.msi并重启,完成后打开CMake (cmake-gui)
CMake (cmake-gui)
  1. E:\OSG\OpenSceneGraph\CMakeLists.txt拖到CMake (cmake-gui)
将CmakeLists.txt拖进CMake后
  1. 将构建路径改为之前新建的E:/OSG/OpenSceneGraph/build
构建路径改为E:/OSG/OpenSceneGraph/build
  1. 点击Configure,为了生成64位版本,选择Visual Studio 15 2017 Win64。注意,如果此处选择Visual Studio 15 2017则将生成32位版本
选择Visual Studio 15 2017 Win64
  1. 选好后点击Finish
此时即可点击Finish
  1. 等待大概二十秒完成Configure
Configure中
  1. Configure完成后修改如下几个选项
    ACTUAL_3RDPARTY_DIRE:/OSG/3rdParty
    BUILD_OSG_EXAMPLES:打勾
    CMAKE_INSTALL_PREFIXE:/OSG/OpenSceneGraph/build
可用Search快速找到选项
  1. 第二次点击Configure,将BUILD_MFC_ EXAMPLE打勾,勾选Advanced
第二次点击Configure后勾选BUILD_MFC_ EXAMPLE
  1. 第三次点击Configure,完成后界面将没有红色选项
第三次点击Configure后
  1. 点击Generate
Generate中
  1. Generate完成,提示Generating done,至此可关闭cmake-gui
Generating done
  1. 用VS2017打开E:\OSG\OpenSceneGraph\build\OpenSceneGraph.sln
OpenSceneGraph.sln
  1. 等待加载完成,CMake构建的解决方案一般包含三个工程:ALL_BUILDINSTALLZERO_CHECK
  1. 打开生成 - 批生成(快捷键Alt+B+T),勾选ALL_BUILD项目的Debug|x64Release|x64配置,点击生成开始编译。若E:\OSG3.7.0\OpenSceneGraph-master\build\lib中有文件生成则说明编译过程顺利进行,大概2.5小时后完成编译,
批生成选项 生成中 生成完成
  1. ALL_BUILD项目编译完成后,再勾选INSTALL项目的Debug|x64Release|x64配置,点击生成开始编译
INSTALL批生成选项 生成结果 之前编译32位时INSTALL的生成结果 顺利的结果 更新时间为15:20的都是批生成INSTALL后改动过的

OSG环境配置

  1. 转移数据,编译完成后的文件有11G,实际只需要用到其中一部分。
image.png
  1. 添加如下环境变量,并重启电脑以使环境变量生效
  1. 打开CMD执行以下命令,以检查是否配置成功
osgviewer cow.osg

VS环境测试

  1. 打开VS并新建一个空项目,将Debug调试器设置为x64
Debug设置为x64
  1. 右键项目打开属性,展开配置属性
每个lib都有一个debug版本(后缀带d)和release版本 VS设置
  1. 添加如下源码运行,运行会先弹出一个空的控制台窗口然后显示之前出现过的那头牛
#include <osgViewer/Viewer> 
#include <osgDB/ReadFile> 
int main( int argc, char **argv ) 
{ 
    osgViewer::Viewer viewer; 
    viewer.setSceneData( osgDB::readNodeFile( "cow.osg" ) ); 
    return viewer.run(); 
} 
  1. 样例源码是学习的主要方式,源码在E:\OSG\OpenSceneGraph\examples中,每个目录是一个示例,其中的cpp文件是源码,如examples\osganimate\osganimate.cpp是两架飞机在一个平面上盘旋飞行

Qt环境测试

  1. 测试Qt5.10.1运行环境,VS2017安装好Qt Visual Studio Tools插件后,打开Qt VS Tools-Qt Options配置好Qt的msvc2017_64路径,新建Qt控制台程序
配置Qt路径 新建Qt控制台程序
  1. 测试之前的源码,配置同之前一样,运行将得到同样的结果
#include <osgViewer/Viewer> 
#include <osgDB/ReadFile> 
int main(int argc, char **argv)
{
    osgViewer::Viewer viewer;
    viewer.setSceneData(osgDB::readNodeFile("cow.osg"));
    return viewer.run();
}

osgQt编译

  1. 完成OSG的编译与环境变量的配置后,为了能在Qt界面中嵌入osg模型,还需要编译osgQt,可在此处下载
  2. 将下载的osgQt-master.zip移动到E:\OSG并解压
解压后osgQt目录中的文件
  1. 同样新建一个build文件夹用于存储编译后的文件
image.png
  1. 打开CMakeLists.txt,找到如下两行(130-131行)
FIND_PACKAGE(OpenSceneGraph 3.0.0 REQUIRED osgDB osgGA osgUtil osgText osgViewer osgWidget)
SET(OPENSCENEGRAPH_SOVERSION 145)
CMakeLists.txt
  1. 修改对应的版本号为3.6.2,157
FIND_PACKAGE(OpenSceneGraph 3.6.2 REQUIRED osgDB osgGA osgUtil osgText osgViewer osgWidget)
SET(OPENSCENEGRAPH_SOVERSION 157)
  1. 打开CMake (cmake-gui),将osgQt目录中的CMakeLists.txt拖进去,并把构建目录改为之前新建的build
将CMakeLists.txt拖进cmake
  1. 点击Configure,选择Visual Studio 15 2017 Win64
Configure
  1. 点击Finish,等待Configuring done
Configuring done
  1. 修改值
配置Qt的msvc2017_64的路径
  1. 再次点击Configure,窗口中的红色全变为白色
第二次Configure后
  1. 点击Generate,生成工程。E:\OSG\osgQt-master\build中将出现生成的osgQt工程,此时可关闭CMake (cmake-gui)
生成的osgQt工程
  1. VS2017打开osgQt.sln
VS2017打开osgQt项目后的界面
  1. 打开生成 - 批生成(快捷键Alt+B+T),勾选ALL_BUILD项目的Debug|x64Release|x64配置,点击生成开始编译。若E:\OSG\osgQt-master\build\lib中有文件生成则说明编译过程顺利进行,此步花费一分钟即可完成
批生成选项 生成中 生成完成
  1. ALL_BUILD项目编译完成后,再勾选INSTALL项目的Debug|x64Release|x64配置,点击生成开始编译
INSTALL批生成选项 INSTALL生成完成
  1. INSTALL编译完成后,在E:\OSG\osgQt\build\share\OpenSceneGraph\bin中将有四个可运行的样例程序
四个可执行样例
  1. osgqfont-qt5.exeosgqfont-qt5d.exe的运行结果是弹出一个窗口,然后显示出字体界面
osgqfont-qt5.exe运行结果
  1. osgviewerQt5.exeosgviewerQt5d.exe的运行结果是弹出一个窗口,然后显示两个窗口界面,左边一个大窗口中有四个区域,分别显示四个模型,右边一个小窗口,显示一辆卡车模型
osgviewerQt5.exe运行结果

osgQt环境配置及测试

  1. 转移数据:
osgQt的库文件
  1. 打开VS,新建Qt GUI Application
新建Qt界面程序
  1. 必须在向导中勾选OpenGL,否则后续会出现无法打开包括文件: “QGLWidget”: No such file or directory的错误,原因是后续使用的osgQt/GraphicsWindowQt中有#include <QGLWidget>
勾选OpenGL osgQt/GraphicsWindowQt
  1. 项目创建完成,右键项目打开属性-配置属性
  1. 以样例代码作为测试。样例代码在最初解压的 E:\OSG\osgQt\examples中,其中osgqfontosgviewerQt分别对应之前运行的两个程序。以第二个程序为例,打开E:\OSG\osgQt\examples\osgviewerQt\osgviewerQt.cpp
osgviewerQt.cpp
  1. 将其内容复制到Qt项目的main.cpp
// source file "main.cpp"
#include <QTimer>
#include <QApplication>
#include <QGridLayout>

#include <osgViewer/CompositeViewer>
#include <osgViewer/ViewerEventHandlers>

#include <osgGA/MultiTouchTrackballManipulator>

#include <osgDB/ReadFile>

#include <osgQt/GraphicsWindowQt>

#include <iostream>

class ViewerWidget : public QWidget, public osgViewer::CompositeViewer
{
public:
    ViewerWidget(QWidget* parent = 0, Qt::WindowFlags f = 0, osgViewer::ViewerBase::ThreadingModel threadingModel = osgViewer::CompositeViewer::SingleThreaded) : QWidget(parent, f)
    {
        setThreadingModel(threadingModel);

        // disable the default setting of viewer.done() by pressing Escape.
        setKeyEventSetsDone(0);

        QWidget* widget1 = addViewWidget(createGraphicsWindow(0, 0, 100, 100), osgDB::readRefNodeFile("cow.osgt"));
        QWidget* widget2 = addViewWidget(createGraphicsWindow(0, 0, 100, 100), osgDB::readRefNodeFile("glider.osgt"));
        QWidget* widget3 = addViewWidget(createGraphicsWindow(0, 0, 100, 100), osgDB::readRefNodeFile("axes.osgt"));
        QWidget* widget4 = addViewWidget(createGraphicsWindow(0, 0, 100, 100), osgDB::readRefNodeFile("fountain.osgt"));
        QWidget* popupWidget = addViewWidget(createGraphicsWindow(900, 100, 320, 240, "Popup window", true), osgDB::readRefNodeFile("dumptruck.osgt"));
        popupWidget->show();

        QGridLayout* grid = new QGridLayout;
        grid->addWidget(widget1, 0, 0);
        grid->addWidget(widget2, 0, 1);
        grid->addWidget(widget3, 1, 0);
        grid->addWidget(widget4, 1, 1);
        setLayout(grid);

        connect(&_timer, SIGNAL(timeout()), this, SLOT(update()));
        _timer.start(10);
    }

    QWidget* addViewWidget(osgQt::GraphicsWindowQt* gw, osg::ref_ptr<osg::Node> scene)
    {
        osgViewer::View* view = new osgViewer::View;
        addView(view);

        osg::Camera* camera = view->getCamera();
        camera->setGraphicsContext(gw);

        const osg::GraphicsContext::Traits* traits = gw->getTraits();

        camera->setClearColor(osg::Vec4(0.2, 0.2, 0.6, 1.0));
        camera->setViewport(new osg::Viewport(0, 0, traits->width, traits->height));

        // set the draw and read buffers up for a double buffered window with rendering going to back buffer
        camera->setDrawBuffer(GL_BACK);
        camera->setReadBuffer(GL_BACK);

        camera->setProjectionMatrixAsPerspective(30.0f, static_cast<double>(traits->width) / static_cast<double>(traits->height), 1.0f, 10000.0f);

        view->setSceneData(scene);
        view->addEventHandler(new osgViewer::StatsHandler);
        view->setCameraManipulator(new osgGA::MultiTouchTrackballManipulator);
        gw->setTouchEventsEnabled(true);
        return gw->getGLWidget();
    }

    osgQt::GraphicsWindowQt* createGraphicsWindow(int x, int y, int w, int h, const std::string& name = "", bool windowDecoration = false)
    {
        osg::DisplaySettings* ds = osg::DisplaySettings::instance().get();
        osg::ref_ptr<osg::GraphicsContext::Traits> traits = new osg::GraphicsContext::Traits;
        traits->windowName = name;
        traits->windowDecoration = windowDecoration;
        traits->x = x;
        traits->y = y;
        traits->width = w;
        traits->height = h;
        traits->doubleBuffer = true;
        traits->alpha = ds->getMinimumNumAlphaBits();
        traits->stencil = ds->getMinimumNumStencilBits();
        traits->sampleBuffers = ds->getMultiSamples();
        traits->samples = ds->getNumMultiSamples();

        return new osgQt::GraphicsWindowQt(traits.get());
    }

    virtual void paintEvent(QPaintEvent* /*event*/)
    {
        frame();
    }

protected:

    QTimer _timer;
};

int main(int argc, char** argv)
{
    osg::ArgumentParser arguments(&argc, argv);

#if QT_VERSION >= 0x050000
    // Qt5 is currently crashing and reporting "Cannot make QOpenGLContext current in a different thread" when the viewer is run multi-threaded, this is regression from Qt4
    osgViewer::ViewerBase::ThreadingModel threadingModel = osgViewer::ViewerBase::SingleThreaded;
#else
    osgViewer::ViewerBase::ThreadingModel threadingModel = osgViewer::ViewerBase::CullDrawThreadPerContext;
#endif

    while (arguments.read("--SingleThreaded")) threadingModel = osgViewer::ViewerBase::SingleThreaded;
    while (arguments.read("--CullDrawThreadPerContext")) threadingModel = osgViewer::ViewerBase::CullDrawThreadPerContext;
    while (arguments.read("--DrawThreadPerContext")) threadingModel = osgViewer::ViewerBase::DrawThreadPerContext;
    while (arguments.read("--CullThreadPerCameraDrawThreadPerContext")) threadingModel = osgViewer::ViewerBase::CullThreadPerCameraDrawThreadPerContext;

#if QT_VERSION >= 0x040800
    // Required for multithreaded QGLWidget on Linux/X11, see http://blog.qt.io/blog/2011/06/03/threaded-opengl-in-4-8/
    if (threadingModel != osgViewer::ViewerBase::SingleThreaded)
        QApplication::setAttribute(Qt::AA_X11InitThreads);
#endif

    QApplication app(argc, argv);
    ViewerWidget* viewWidget = new ViewerWidget(0, Qt::Widget, threadingModel);
    viewWidget->setGeometry(100, 100, 800, 600);
    viewWidget->show();
    return app.exec();
}
  1. 运行不再弹出黑窗口,而是直接显示之前osgviewerQt5d.exe的两个窗口
运行结果
上一篇 下一篇

猜你喜欢

热点阅读