Learning OSG programing---Multi Camera in Multi window 在多窗口中创建多相机

  这个例子演示了在多个窗口中创建多个相机,函数的代码如下:

 1 void multiWindowMultipleCameras(osgViewer::Viewer& viewer,bool multipleScreens)
 2 {
 3     osg::GraphicsContext::WindowingSystemInterface* wsi = osg::GraphicsContext::getWindowingSystemInterface();
 4     if (!wsi)
 5     {   
 6         osg::notify(osg::NOTICE)<<"Error, no WindowSystemInterface available, cannot create windows."<<std::endl;
 7         return;
 8     }
 9 
10     unsigned int width, height;
11     osg::GraphicsContext::ScreenIdentifier main_screen_id;
12 
13     main_screen_id.readDISPLAY();
14     main_screen_id.setUndefinedScreenDetailsToDefaultScreen();
15     wsi->getScreenResolution(main_screen_id, width, height);
16     unsigned int numCameras = 3;
17     double aspectRatioScale = (double)numCameras;
18     double translate_x = double(numCameras)-1;
19     for(unsigned int i=0; i<numCameras;++i, translate_x -= 2.0)
20     {
21         osg::ref_ptr<osg::GraphicsContext::Traits> traits = new osg::GraphicsContext::Traits;
22         traits->screenNum = multipleScreens ? i / 3 : 0;
23         traits->x = (i*width)/numCameras;
24         traits->y = 0;
25         traits->width = width/numCameras-1;
26         traits->height = height;
27         traits->windowDecoration = true;
28         traits->doubleBuffer = true;
29         traits->sharedContext = 0;
30         traits->readDISPLAY();
31         traits->setUndefinedScreenDetailsToDefaultScreen();
32         osg::ref_ptr<osg::GraphicsContext> gc = osg::GraphicsContext::createGraphicsContext(traits.get());
33         if (gc.valid())
34         {
35             osg::notify(osg::INFO)<<"  GraphicsWindow has been created successfully."<<std::endl;
36         }
37         else
38         {
39             osg::notify(osg::NOTICE)<<"  GraphicsWindow has not been created successfully."<<std::endl;
40         }
41 
42         osg::ref_ptr<osg::Camera> camera = new osg::Camera;
43         camera->setGraphicsContext(gc.get());
44         camera->setViewport(new osg::Viewport(0,0, width/numCameras, height));
45         GLenum buffer = traits->doubleBuffer ? GL_BACK : GL_FRONT;
46         camera->setDrawBuffer(buffer);
47         camera->setReadBuffer(buffer);
48 viewer.addSlave(camera.get(), osg::Matrix::scale(aspectRatioScale, 1.0, 1.0)*osg::Matrix::translate(translate_x, 0.0, 0.0), osg::Matrix() );
49     }
50 }

  程序中用到了addSlave函数,此函数接受三个参数,分别为摄像机地址,投影偏移矩阵,视图偏移矩阵。由48行可知,每次向观察器中添加从相机时,都对相机的投影偏移矩阵进行设置(第二个参数)。

  在这里做一个实验,验证addSlave函数第二个参数的作用。将multiWindowMultipleCameras函数中最后一行代码改为:

 

viewer.addSlave(camera.get(), osg::Matrix(), osg::Matrix() );

 

  即将函数的投影偏移矩阵设为单位矩阵。运行程序,产生的结果如下:

由上图可知,由于没有对投影矩阵进行变换,将在三个窗口中分别产生相同的模型,而没有产生共屏效果。若想达到共屏的效果,需要对投影偏移矩阵进行设置。

  首先设置投影偏移矩阵为平移,函数中有变量translate_x,其初始值为numCameras-1,在本例中即为2,随着每次循环,其值都将减少2。因此三次循环后,translate_x产生的序列为2,0,-2。为此,将添加从相机的语句改为:

viewer.addSlave(camera.get(), osg::Matrix::translate(translate_x, 0.0, 0.0), osg::Matrix() );

  运行修改后的程序,得到以下结果:

  从上图可以看到,三块屏幕之间产生联系了,模型可以跨窗口显示了。但是仍存在明显缺陷:模型在水平方向被压缩了。可能是由于每个小窗口在水平方向上被压缩,导致模型被压缩。为此考虑对模型进行水平方向上的拉伸。由于原有的整块屏幕现在被平均分割为三块,因此考虑将模型在水平方向上拉伸三倍(即屏幕数量倍)。仍然修改投影偏移矩阵:

viewer.addSlave(camera.get(), osg::Matrix::scale(aspectRatioScale, 1.0, 1.0)*osg::Matrix::translate(translate_x, 0.0, 0.0), osg::Matrix() );

运行程序,出现如下效果:

可以看到,达到了预期效果,只是模型相比于普通的用osgviewer cow.osg命令时在水平方向上略显拉伸。

  这里还有一个问题,为什么translate_x的初始值要设为numCameras-1,并且每次循环都要减少2?略加思考,就能得到答案,为了产生左右对称的偏移量:

  numCameras    translate_x

     3        2  0  -2

       4        3  1  -1  -3

       5        4  2  0  -2  -4

      ...        ...

以此类推。此外,还可以修改视图偏移矩阵来设置从不同的角度观察。

 

posted @ 2019-04-10 20:22  技术狂人djc  阅读(474)  评论(0编辑  收藏  举报