2007-11-30 00:00:36 +00:00
|
|
|
// fg_os_common.cxx -- common functions for fg_os interface
|
|
|
|
// implemented as an osgViewer
|
|
|
|
//
|
|
|
|
// Copyright (C) 2007 Tim Moore timoore@redhat.com
|
|
|
|
// Copyright (C) 2007 Mathias Froehlich
|
|
|
|
//
|
|
|
|
// This program is free software; you can redistribute it and/or
|
|
|
|
// modify it under the terms of the GNU General Public License as
|
|
|
|
// published by the Free Software Foundation; either version 2 of the
|
|
|
|
// License, or (at your option) any later version.
|
|
|
|
//
|
|
|
|
// This program is distributed in the hope that it will be useful, but
|
|
|
|
// WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
|
// General Public License for more details.
|
|
|
|
//
|
|
|
|
// You should have received a copy of the GNU General Public License
|
|
|
|
// along with this program; if not, write to the Free Software
|
|
|
|
// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
2007-06-16 16:15:49 +00:00
|
|
|
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
|
|
# include <config.h>
|
|
|
|
#endif
|
|
|
|
|
2007-05-21 17:50:02 +00:00
|
|
|
#include <stdlib.h>
|
|
|
|
|
|
|
|
#include <simgear/compiler.h>
|
|
|
|
#include <simgear/structure/exception.hxx>
|
|
|
|
#include <simgear/debug/logstream.hxx>
|
|
|
|
|
|
|
|
#include <osg/GraphicsContext>
|
|
|
|
#include <osg/Group>
|
|
|
|
#include <osg/Matrixd>
|
|
|
|
#include <osg/Viewport>
|
2007-06-08 07:14:56 +00:00
|
|
|
#include <osg/Version>
|
2008-01-25 18:44:45 +00:00
|
|
|
#include <osg/View>
|
2007-06-16 16:15:49 +00:00
|
|
|
#include <osgViewer/ViewerEventHandlers>
|
2007-05-21 17:50:02 +00:00
|
|
|
#include <osgViewer/Viewer>
|
|
|
|
#include <osgGA/MatrixManipulator>
|
|
|
|
|
2007-10-04 20:40:29 +00:00
|
|
|
#include <Include/general.hxx>
|
2007-12-14 22:51:56 +00:00
|
|
|
#include <Scenery/scenery.hxx>
|
2007-05-21 17:50:02 +00:00
|
|
|
#include "fg_os.hxx"
|
2007-05-26 13:53:46 +00:00
|
|
|
#include "fg_props.hxx"
|
2007-05-21 17:50:02 +00:00
|
|
|
#include "util.hxx"
|
|
|
|
#include "globals.hxx"
|
|
|
|
#include "renderer.hxx"
|
|
|
|
|
2007-10-04 20:40:29 +00:00
|
|
|
#if (FG_OSG_VERSION >= 19008)
|
2007-06-08 07:14:56 +00:00
|
|
|
#define OSG_HAS_MOUSE_CURSOR_PATCH
|
|
|
|
#endif
|
|
|
|
|
2007-05-21 17:50:02 +00:00
|
|
|
// fg_os implementation using OpenSceneGraph's osgViewer::Viewer class
|
|
|
|
// to create the graphics window and run the event/update/render loop.
|
|
|
|
|
|
|
|
//
|
|
|
|
// fg_os implementation
|
|
|
|
//
|
|
|
|
|
2008-01-25 18:44:45 +00:00
|
|
|
using namespace osg;
|
|
|
|
|
2007-05-21 17:50:02 +00:00
|
|
|
static osg::ref_ptr<osgViewer::Viewer> viewer;
|
2007-05-26 13:53:46 +00:00
|
|
|
static osg::ref_ptr<osg::Camera> mainCamera;
|
2007-05-21 17:50:02 +00:00
|
|
|
|
2008-01-25 18:44:45 +00:00
|
|
|
// Callback to prevent the GraphicsContext resized function from messing
|
|
|
|
// with the projection matrix of the slave
|
|
|
|
|
|
|
|
namespace
|
|
|
|
{
|
|
|
|
struct fgResizeCallback : public GraphicsContext::ResizedCallback
|
|
|
|
{
|
|
|
|
fgResizeCallback(Camera* slaveCamera)
|
|
|
|
: mainSlaveCamera(slaveCamera)
|
|
|
|
{}
|
|
|
|
|
|
|
|
virtual void resizedImplementation(GraphicsContext* gc, int x, int y,
|
|
|
|
int width, int height);
|
|
|
|
ref_ptr<Camera> mainSlaveCamera;
|
|
|
|
};
|
|
|
|
|
|
|
|
void fgResizeCallback::resizedImplementation(GraphicsContext* gc,
|
|
|
|
int x, int y,
|
|
|
|
int width, int height)
|
|
|
|
{
|
|
|
|
View* view = mainSlaveCamera->getView();
|
|
|
|
View::Slave* slave = (view ?
|
|
|
|
view->findSlaveForCamera(mainSlaveCamera.get()) : 0);
|
|
|
|
if (slave) {
|
|
|
|
Matrixd projOffset(slave->_projectionOffset);
|
|
|
|
gc->resizedImplementation(x, y, width, height);
|
|
|
|
// Restore projection offsets changed by
|
|
|
|
// GraphicsContext::resizedImplementation
|
|
|
|
slave->_projectionOffset = projOffset;
|
|
|
|
} else {
|
|
|
|
gc->resizedImplementation(x, y, width, height);
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
2007-05-21 17:50:02 +00:00
|
|
|
void fgOSOpenWindow(int w, int h, int bpp,
|
|
|
|
bool alpha, bool stencil, bool fullscreen)
|
|
|
|
{
|
2007-05-26 13:53:46 +00:00
|
|
|
osg::GraphicsContext::WindowingSystemInterface* wsi;
|
|
|
|
wsi = osg::GraphicsContext::getWindowingSystemInterface();
|
|
|
|
|
2007-05-21 17:50:02 +00:00
|
|
|
viewer = new osgViewer::Viewer;
|
2007-12-14 22:51:56 +00:00
|
|
|
viewer->setDatabasePager(FGScenery::getPagerSingleton());
|
2007-05-26 13:53:46 +00:00
|
|
|
std::string mode;
|
|
|
|
mode = fgGetString("/sim/rendering/multithreading-mode", "SingleThreaded");
|
|
|
|
if (mode == "AutomaticSelection")
|
|
|
|
viewer->setThreadingModel(osgViewer::Viewer::AutomaticSelection);
|
|
|
|
else if (mode == "CullDrawThreadPerContext")
|
|
|
|
viewer->setThreadingModel(osgViewer::Viewer::CullDrawThreadPerContext);
|
|
|
|
else if (mode == "DrawThreadPerContext")
|
|
|
|
viewer->setThreadingModel(osgViewer::Viewer::DrawThreadPerContext);
|
|
|
|
else if (mode == "CullThreadPerCameraDrawThreadPerContext")
|
|
|
|
viewer->setThreadingModel(osgViewer::Viewer::CullThreadPerCameraDrawThreadPerContext);
|
|
|
|
else
|
|
|
|
viewer->setThreadingModel(osgViewer::Viewer::SingleThreaded);
|
|
|
|
osg::ref_ptr<osg::GraphicsContext::Traits> traits;
|
|
|
|
traits = new osg::GraphicsContext::Traits;
|
2007-12-26 12:19:20 +00:00
|
|
|
traits->readDISPLAY();
|
2007-05-21 17:50:02 +00:00
|
|
|
int cbits = (bpp <= 16) ? 5 : 8;
|
|
|
|
int zbits = (bpp <= 16) ? 16 : 24;
|
|
|
|
traits->red = traits->green = traits->blue = cbits;
|
|
|
|
traits->depth = zbits;
|
|
|
|
if (alpha)
|
|
|
|
traits->alpha = 8;
|
|
|
|
if (stencil)
|
|
|
|
traits->stencil = 8;
|
2007-05-26 13:53:46 +00:00
|
|
|
traits->doubleBuffer = true;
|
|
|
|
traits->mipMapGeneration = true;
|
|
|
|
traits->windowName = "FlightGear";
|
|
|
|
traits->sampleBuffers = fgGetBool("/sim/rendering/multi-sample-buffers", traits->sampleBuffers);
|
|
|
|
traits->samples = fgGetBool("/sim/rendering/multi-samples", traits->samples);
|
|
|
|
traits->vsync = fgGetBool("/sim/rendering/vsync-enable", traits->vsync);
|
|
|
|
|
|
|
|
if (fullscreen) {
|
|
|
|
unsigned width = 0;
|
|
|
|
unsigned height = 0;
|
|
|
|
wsi->getScreenResolution(*traits, width, height);
|
2007-05-21 17:50:02 +00:00
|
|
|
traits->windowDecoration = false;
|
2007-05-26 13:53:46 +00:00
|
|
|
traits->width = width;
|
|
|
|
traits->height = height;
|
|
|
|
traits->supportsResize = false;
|
|
|
|
} else {
|
2007-05-21 17:50:02 +00:00
|
|
|
traits->windowDecoration = true;
|
2007-05-26 13:53:46 +00:00
|
|
|
traits->width = w;
|
|
|
|
traits->height = h;
|
2008-03-12 12:38:54 +00:00
|
|
|
#if defined(WIN32) || defined(__APPLE__)
|
|
|
|
// Ugly Hack, why does CW_USEDEFAULT works like phase of the moon?
|
|
|
|
// Mac also needs this to show window frame, menubar and Docks
|
|
|
|
traits->x = 100;
|
|
|
|
traits->y = 100;
|
2007-05-30 13:15:14 +00:00
|
|
|
#endif
|
2007-05-26 13:53:46 +00:00
|
|
|
traits->supportsResize = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
osg::Camera::ProjectionResizePolicy rsp = osg::Camera::VERTICAL;
|
|
|
|
|
|
|
|
// Ok, first the children.
|
|
|
|
// that achieves some magic ordering og the slaves so that we end up
|
|
|
|
// in the main window more often.
|
|
|
|
// This can be sorted out better when we got rid of glut and sdl.
|
2007-07-01 16:39:52 +00:00
|
|
|
FGManipulator* manipulator = globals->get_renderer()->getManipulator();
|
|
|
|
int nCameras = 0;
|
2007-05-26 13:53:46 +00:00
|
|
|
if (fgHasNode("/sim/rendering/camera")) {
|
|
|
|
SGPropertyNode* renderingNode = fgGetNode("/sim/rendering");
|
|
|
|
for (int i = 0; i < renderingNode->nChildren(); ++i) {
|
|
|
|
SGPropertyNode* cameraNode = renderingNode->getChild(i);
|
|
|
|
if (strcmp(cameraNode->getName(), "camera") != 0)
|
|
|
|
continue;
|
|
|
|
|
2007-07-01 16:39:52 +00:00
|
|
|
nCameras++;
|
2007-05-26 13:53:46 +00:00
|
|
|
// get a new copy of the traits struct
|
|
|
|
osg::ref_ptr<osg::GraphicsContext::Traits> cameraTraits;
|
|
|
|
cameraTraits = new osg::GraphicsContext::Traits(*traits);
|
|
|
|
|
|
|
|
double shearx = cameraNode->getDoubleValue("shear-x", 0);
|
|
|
|
double sheary = cameraNode->getDoubleValue("shear-y", 0);
|
2007-12-26 12:19:20 +00:00
|
|
|
cameraTraits->hostName
|
|
|
|
= cameraNode->getStringValue("host-name", traits->hostName.c_str());
|
|
|
|
cameraTraits->displayNum
|
|
|
|
= cameraNode->getIntValue("display", traits->displayNum);
|
|
|
|
cameraTraits->screenNum
|
|
|
|
= cameraNode->getIntValue("screen", traits->screenNum);
|
2007-05-26 13:53:46 +00:00
|
|
|
if (cameraNode->getBoolValue("fullscreen", fullscreen)) {
|
|
|
|
unsigned width = 0;
|
|
|
|
unsigned height = 0;
|
|
|
|
wsi->getScreenResolution(*cameraTraits, width, height);
|
|
|
|
cameraTraits->windowDecoration = false;
|
|
|
|
cameraTraits->width = width;
|
|
|
|
cameraTraits->height = height;
|
|
|
|
cameraTraits->supportsResize = false;
|
|
|
|
} else {
|
|
|
|
cameraTraits->windowDecoration = true;
|
|
|
|
cameraTraits->width = cameraNode->getIntValue("width", w);
|
|
|
|
cameraTraits->height = cameraNode->getIntValue("height", h);
|
|
|
|
cameraTraits->supportsResize = true;
|
|
|
|
}
|
|
|
|
// FIXME, currently this is too much of a problem to route the resize
|
|
|
|
// events. When we do no longer need sdl and such this
|
|
|
|
// can be simplified
|
|
|
|
cameraTraits->supportsResize = false;
|
|
|
|
|
|
|
|
// ok found a camera configuration, add a new slave ...
|
|
|
|
osg::ref_ptr<osg::Camera> camera = new osg::Camera;
|
|
|
|
|
|
|
|
osg::GraphicsContext* gc;
|
|
|
|
gc = osg::GraphicsContext::createGraphicsContext(cameraTraits.get());
|
|
|
|
gc->realize();
|
|
|
|
camera->setGraphicsContext(gc);
|
|
|
|
// If a viewport isn't set on the camera, then it's hard to dig it
|
|
|
|
// out of the SceneView objects in the viewer, and the coordinates
|
|
|
|
// of mouse events are somewhat bizzare.
|
|
|
|
camera->setViewport(new osg::Viewport(0, 0, cameraTraits->width, cameraTraits->height));
|
|
|
|
camera->setProjectionResizePolicy(rsp);
|
|
|
|
viewer->addSlave(camera.get(), osg::Matrix::translate(-shearx, -sheary, 0), osg::Matrix());
|
|
|
|
}
|
2007-07-01 16:39:52 +00:00
|
|
|
if (nCameras > 1)
|
|
|
|
manipulator->setResizable(false);
|
2007-05-26 13:53:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// now the main camera ...
|
|
|
|
osg::ref_ptr<osg::Camera> camera = new osg::Camera;
|
|
|
|
mainCamera = camera;
|
|
|
|
osg::GraphicsContext* gc;
|
|
|
|
gc = osg::GraphicsContext::createGraphicsContext(traits.get());
|
|
|
|
gc->realize();
|
|
|
|
gc->makeCurrent();
|
|
|
|
camera->setGraphicsContext(gc);
|
2007-05-21 17:50:02 +00:00
|
|
|
// If a viewport isn't set on the camera, then it's hard to dig it
|
|
|
|
// out of the SceneView objects in the viewer, and the coordinates
|
|
|
|
// of mouse events are somewhat bizzare.
|
2007-05-26 13:53:46 +00:00
|
|
|
camera->setViewport(new osg::Viewport(0, 0, traits->width, traits->height));
|
|
|
|
camera->setProjectionResizePolicy(rsp);
|
2008-01-25 18:44:45 +00:00
|
|
|
if (nCameras == 0) {
|
|
|
|
// Only one principal camera
|
|
|
|
gc->setResizedCallback(new fgResizeCallback(camera.get()));
|
|
|
|
}
|
|
|
|
// Why a slave? It seems to be the easiest way to assign cameras,
|
|
|
|
// for which we've created the graphics context ourselves, to the viewer.
|
2007-05-26 13:53:46 +00:00
|
|
|
viewer->addSlave(camera.get());
|
|
|
|
|
2007-05-21 17:50:02 +00:00
|
|
|
viewer->setCameraManipulator(globals->get_renderer()->getManipulator());
|
|
|
|
// Let FG handle the escape key with a confirmation
|
|
|
|
viewer->setKeyEventSetsDone(0);
|
|
|
|
osgViewer::StatsHandler* statsHandler = new osgViewer::StatsHandler;
|
|
|
|
statsHandler->setKeyEventTogglesOnScreenStats('*');
|
|
|
|
statsHandler->setKeyEventPrintsOutStats(0);
|
|
|
|
viewer->addEventHandler(statsHandler);
|
|
|
|
// The viewer won't start without some root.
|
|
|
|
viewer->setSceneData(new osg::Group);
|
|
|
|
globals->get_renderer()->setViewer(viewer.get());
|
|
|
|
}
|
|
|
|
|
|
|
|
static int status = 0;
|
|
|
|
|
|
|
|
void fgOSExit(int code)
|
|
|
|
{
|
|
|
|
viewer->setDone(true);
|
|
|
|
status = code;
|
|
|
|
}
|
|
|
|
|
|
|
|
void fgOSMainLoop()
|
|
|
|
{
|
|
|
|
viewer->run();
|
|
|
|
fgExit(status);
|
|
|
|
}
|
|
|
|
|
|
|
|
int fgGetKeyModifiers()
|
|
|
|
{
|
|
|
|
return globals->get_renderer()->getManipulator()->getCurrentModifiers();
|
|
|
|
}
|
|
|
|
|
|
|
|
void fgWarpMouse(int x, int y)
|
|
|
|
{
|
2007-08-19 05:29:00 +00:00
|
|
|
globals->get_renderer()->getManipulator()->setMouseWarped();
|
2007-08-18 22:07:11 +00:00
|
|
|
// Hack, currently the pointer is just recentered. So, we know the
|
|
|
|
// relative coordinates ...
|
|
|
|
if (!mainCamera.valid()) {
|
|
|
|
viewer->requestWarpPointer(0, 0);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
float xsize = (float)mainCamera->getGraphicsContext()->getTraits()->width;
|
|
|
|
float ysize = (float)mainCamera->getGraphicsContext()->getTraits()->height;
|
|
|
|
viewer->requestWarpPointer(2.0f * (float)x / xsize - 1.0f,
|
|
|
|
1.0f - 2.0f * (float)y / ysize);
|
2007-05-21 17:50:02 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Noop
|
|
|
|
void fgOSInit(int* argc, char** argv)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2007-05-26 13:53:46 +00:00
|
|
|
// Noop
|
2007-05-21 17:50:02 +00:00
|
|
|
void fgOSFullScreen()
|
|
|
|
{
|
2007-05-26 13:53:46 +00:00
|
|
|
}
|
2007-05-21 17:50:02 +00:00
|
|
|
|
2007-05-26 13:53:46 +00:00
|
|
|
#ifdef OSG_HAS_MOUSE_CURSOR_PATCH
|
|
|
|
static void setMouseCursor(osg::Camera* camera, int cursor)
|
|
|
|
{
|
|
|
|
if (!camera)
|
|
|
|
return;
|
|
|
|
osg::GraphicsContext* gc = camera->getGraphicsContext();
|
|
|
|
if (!gc)
|
|
|
|
return;
|
|
|
|
osgViewer::GraphicsWindow* gw;
|
|
|
|
gw = dynamic_cast<osgViewer::GraphicsWindow*>(gc);
|
|
|
|
if (!gw)
|
|
|
|
return;
|
|
|
|
|
|
|
|
osgViewer::GraphicsWindow::MouseCursor mouseCursor;
|
|
|
|
mouseCursor = osgViewer::GraphicsWindow::InheritCursor;
|
|
|
|
if (cursor == MOUSE_CURSOR_NONE)
|
|
|
|
mouseCursor = osgViewer::GraphicsWindow::NoCursor;
|
|
|
|
else if(cursor == MOUSE_CURSOR_POINTER)
|
|
|
|
mouseCursor = osgViewer::GraphicsWindow::RightArrowCursor;
|
|
|
|
else if(cursor == MOUSE_CURSOR_WAIT)
|
|
|
|
mouseCursor = osgViewer::GraphicsWindow::WaitCursor;
|
|
|
|
else if(cursor == MOUSE_CURSOR_CROSSHAIR)
|
|
|
|
mouseCursor = osgViewer::GraphicsWindow::CrosshairCursor;
|
|
|
|
else if(cursor == MOUSE_CURSOR_LEFTRIGHT)
|
|
|
|
mouseCursor = osgViewer::GraphicsWindow::LeftRightCursor;
|
|
|
|
|
|
|
|
gw->setCursor(mouseCursor);
|
2007-05-21 17:50:02 +00:00
|
|
|
}
|
2007-05-26 13:53:46 +00:00
|
|
|
#endif
|
|
|
|
|
|
|
|
static int _cursor = -1;
|
2007-05-21 17:50:02 +00:00
|
|
|
|
|
|
|
void fgSetMouseCursor(int cursor)
|
|
|
|
{
|
2007-05-26 13:53:46 +00:00
|
|
|
_cursor = cursor;
|
|
|
|
#ifdef OSG_HAS_MOUSE_CURSOR_PATCH
|
|
|
|
setMouseCursor(viewer->getCamera(), cursor);
|
|
|
|
for (unsigned i = 0; i < viewer->getNumSlaves(); ++i)
|
|
|
|
setMouseCursor(viewer->getSlave(i)._camera.get(), cursor);
|
|
|
|
#endif
|
2007-05-21 17:50:02 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int fgGetMouseCursor()
|
|
|
|
{
|
2007-05-26 13:53:46 +00:00
|
|
|
return _cursor;
|
2007-05-21 17:50:02 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void fgMakeCurrent()
|
|
|
|
{
|
2007-05-26 13:53:46 +00:00
|
|
|
if (!mainCamera.valid())
|
|
|
|
return;
|
|
|
|
osg::GraphicsContext* gc = mainCamera->getGraphicsContext();
|
|
|
|
if (!gc)
|
|
|
|
return;
|
2007-05-21 17:50:02 +00:00
|
|
|
gc->makeCurrent();
|
|
|
|
}
|
2007-05-26 13:53:46 +00:00
|
|
|
|
|
|
|
bool fgOSIsMainContext(const osg::GraphicsContext* context)
|
|
|
|
{
|
|
|
|
if (!mainCamera.valid())
|
|
|
|
return false;
|
|
|
|
return context == mainCamera->getGraphicsContext();
|
|
|
|
}
|
|
|
|
|
|
|
|
bool fgOSIsMainCamera(const osg::Camera* camera)
|
|
|
|
{
|
|
|
|
if (!camera)
|
|
|
|
return false;
|
|
|
|
if (camera == mainCamera.get())
|
|
|
|
return true;
|
|
|
|
if (!viewer.valid())
|
|
|
|
return false;
|
|
|
|
if (camera == viewer->getCamera())
|
|
|
|
return true;
|
|
|
|
return false;
|
|
|
|
}
|