// Viewer.cxx -- alternative flightgear viewer application
//
// Copyright (C) 2009 - 2012  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.

#ifdef HAVE_CONFIG_H
#include <config.h>
#endif

#include "Viewer.hxx"

#include <osg/Version>
#include <osg/ArgumentParser>
#include <osg/ProxyNode>
#include <osg/PagedLOD>
#include <osgDB/ReadFile>

#ifdef __linux__
#include <X11/Xlib.h>
#include <osgViewer/api/X11/GraphicsWindowX11>
#endif

#include "MEncoderCaptureOperation.hxx"

#include <cassert>

namespace fgviewer  {

Viewer::Viewer(osg::ArgumentParser& arguments) :
    osgViewer::Viewer(arguments),
    _sceneDataGroup(new osg::Group),
    _timeIncrement(SGTimeStamp::fromSec(0)),
    _simTime(SGTimeStamp::fromSec(0))
{
    /// Careful: this method really assigns the sceneDataGroup to all cameras!
    /// FIXME the 'useMasterScene' flag at the slave is able to get around that!!!
    osgViewer::Viewer::setSceneData(_sceneDataGroup.get());
    /// The only changed default that is renderer independent ...
    getCamera()->setClearColor(osg::Vec4(0, 0, 0, 0));
}

Viewer::~Viewer()
{
    stopThreading();

#if FG_HAVE_HLA
    if (_viewerFederate.valid())
        _viewerFederate->shutdown();
    _viewerFederate = 0;
#endif
}

bool
Viewer::readCameraConfig(const SGPropertyNode& viewerNode)
{
    // Collect and realize all windows
    for (int i = 0; i < viewerNode.nChildren(); ++i) {
        // FIXME support window, fullscreen, offscreen
        const SGPropertyNode* windowNode = viewerNode.getChild(i);
        if (!windowNode || windowNode->getNameString() != "window")
            continue;

        std::string name = windowNode->getStringValue("name", "");
        if (name.empty()) {
            SG_LOG(SG_VIEW, SG_ALERT, "Ignoring unnamed window!");
            return false;
        }

        Drawable* drawable = getOrCreateDrawable(name);

        osg::GraphicsContext::ScreenIdentifier screenIdentifier;
        screenIdentifier = getScreenIdentifier(windowNode->getStringValue("display", ""));
        drawable->setScreenIdentifier(screenIdentifier.displayName());

        if (windowNode->getBoolValue("fullscreen", false)) {
            osg::GraphicsContext::ScreenSettings screenSettings;
            screenSettings = getScreenSettings(screenIdentifier);
            drawable->setPosition(SGVec2i(0, 0));
            drawable->setSize(SGVec2i(screenSettings.width, screenSettings.height));
            drawable->setFullscreen(true);
            drawable->setOffscreen(false);

        } else if (windowNode->getBoolValue("video", false)) {
            drawable->setPosition(SGVec2i(0, 0));
            SGVec2i size;
            size[0] = windowNode->getIntValue("geometry/width", 1366);
            size[1] = windowNode->getIntValue("geometry/height", 768);
            drawable->setSize(size);
            drawable->setFullscreen(true);
            drawable->setOffscreen(true);

            std::string outputFile = windowNode->getStringValue("output-file", "fgviewer.avi");
            unsigned fps = windowNode->getIntValue("frames-per-second", 30);

            /// This is the case for the video writers, have a fixed time increment
            _timeIncrement = SGTimeStamp::fromSec(1.0/fps);

            MEncoderCaptureOperation* captureOperation;
            captureOperation = new MEncoderCaptureOperation(outputFile, fps);
            osgViewer::ScreenCaptureHandler* captureHandler;
            captureHandler = new osgViewer::ScreenCaptureHandler(captureOperation, -1);
            addEventHandler(captureHandler);
            captureHandler->startCapture();

        } else {

            SGVec2i position;
            position[0] = windowNode->getIntValue("geometry/x", 0);
            position[1] = windowNode->getIntValue("geometry/y", 0);
            drawable->setPosition(position);
            SGVec2i size;
            size[0] = windowNode->getIntValue("geometry/width", 1366);
            size[1] = windowNode->getIntValue("geometry/height", 768);
            drawable->setSize(size);
            drawable->setFullscreen(false);
            drawable->setOffscreen(false);
        }
    }

    for (int i = 0; i < viewerNode.nChildren(); ++i) {
        const SGPropertyNode* cameraNode = viewerNode.getChild(i);
        if (!cameraNode || cameraNode->getNameString() != "camera")
            continue;

        std::string name = cameraNode->getStringValue("name", "");
        if (name.empty()) {
            SG_LOG(SG_VIEW, SG_ALERT, "Camera configuration needs a name!");
            return false;
        }

        SlaveCamera* slaveCamera = getOrCreateSlaveCamera(name);

        std::string drawableName = cameraNode->getStringValue("window", "");
        if (drawableName.empty()) {
            SG_LOG(SG_VIEW, SG_ALERT, "Camera configuration needs an assigned window!");
            return false;
        }
        Drawable* drawable = getDrawable(drawableName);
        if (!drawable) {
            SG_LOG(SG_VIEW, SG_ALERT, "Camera configuration \"" << name << "\" needs a drawable configured!");
            return false;
        }
        slaveCamera->setDrawableName(drawableName);
        drawable->attachSlaveCamera(slaveCamera);

        SGVec2i size = drawable->getSize();
        SGVec4i viewport(0, 0, size[0], size[1]);
        viewport[0] = cameraNode->getIntValue("viewport/x", viewport[0]);
        viewport[1] = cameraNode->getIntValue("viewport/y", viewport[1]);
        viewport[2] = cameraNode->getIntValue("viewport/width", viewport[2]);
        viewport[3] = cameraNode->getIntValue("viewport/height", viewport[3]);
        slaveCamera->setViewport(viewport);

        double headingDeg = cameraNode->getDoubleValue("view-offset/heading-deg", 0);
        double pitchDeg = cameraNode->getDoubleValue("view-offset/pitch-deg", 0);
        double rollDeg = cameraNode->getDoubleValue("view-offset/roll-deg", 0);
        slaveCamera->setViewOffsetDeg(headingDeg, pitchDeg, rollDeg);

        // Care for the reference points
        if (const SGPropertyNode* referencePointsNode = cameraNode->getNode("reference-points")) {
            for (int j = 0; j < referencePointsNode->nChildren(); ++j) {
                const SGPropertyNode* referencePointNode = cameraNode->getNode("reference-point");
                if (!referencePointNode)
                    continue;
                std::string name = referencePointNode->getStringValue("name", "");
                if (name.empty())
                    continue;
                osg::Vec2 point;
                point[0] = referencePointNode->getDoubleValue("x", 0);
                point[1] = referencePointNode->getDoubleValue("y", 0);
                slaveCamera->setProjectionReferencePoint(name, point);
            }
        }
        // Define 4 reference points by monitor dimensions
        else if (const SGPropertyNode* physicalDimensionsNode = cameraNode->getNode("physical-dimensions")) {
            double physicalWidth = physicalDimensionsNode->getDoubleValue("width", viewport[2]);
            double physicalHeight = physicalDimensionsNode->getDoubleValue("height", viewport[3]);
            if (const SGPropertyNode* bezelNode = physicalDimensionsNode->getNode("bezel")) {
                double bezelHeightTop = bezelNode->getDoubleValue("top", 0);
                double bezelHeightBottom = bezelNode->getDoubleValue("bottom", 0);
                double bezelWidthLeft = bezelNode->getDoubleValue("left", 0);
                double bezelWidthRight = bezelNode->getDoubleValue("right", 0);
                slaveCamera->setMonitorProjectionReferences(physicalWidth, physicalHeight,
                                                            bezelHeightTop, bezelHeightBottom,
                                                            bezelWidthLeft, bezelWidthRight);
            }
        }

        // The frustum node takes precedence, as this is the most explicit one.
        if (const SGPropertyNode* frustumNode = cameraNode->getNode("frustum")) {
            Frustum frustum(slaveCamera->getAspectRatio());
            frustum._near = frustumNode->getDoubleValue("near", frustum._near);
            frustum._left = frustumNode->getDoubleValue("left", frustum._left);
            frustum._right = frustumNode->getDoubleValue("right", frustum._right);
            frustum._bottom = frustumNode->getDoubleValue("bottom", frustum._bottom);
            frustum._top = frustumNode->getDoubleValue("top", frustum._top);
            slaveCamera->setFrustum(frustum);

        } else if (const SGPropertyNode* perspectiveNode = cameraNode->getNode("perspective")) {
            double fieldOfViewDeg = perspectiveNode->getDoubleValue("field-of-view-deg", 55);
            slaveCamera->setFustumByFieldOfViewDeg(fieldOfViewDeg);

        } else if (const SGPropertyNode* monitorNode = cameraNode->getNode("monitor")) {

            std::string referenceCameraName;
            std::string names[2];
            std::string referenceNames[2];

            // FIXME??!!
            if (const SGPropertyNode* leftOfNode = monitorNode->getNode("left-of")) {
                referenceCameraName = leftOfNode->getStringValue("");
                names[0] = "lowerRight";
                referenceNames[0] = "lowerLeft";
                names[1] = "upperRight";
                referenceNames[1] = "upperLeft";
            } else if (const SGPropertyNode* rightOfNode = monitorNode->getNode("right-of")) {
                referenceCameraName = rightOfNode->getStringValue("");
                names[0] = "lowerLeft";
                referenceNames[0] = "lowerRight";
                names[1] = "upperLeft";
                referenceNames[1] = "upperRight";
            } else if (const SGPropertyNode* aboveNode = monitorNode->getNode("above")) {
                referenceCameraName = aboveNode->getStringValue("");
                names[0] = "lowerLeft";
                referenceNames[0] = "upperLeft";
                names[1] = "lowerRight";
                referenceNames[1] = "upperRight";
            } else if (const SGPropertyNode* belowNode = monitorNode->getNode("below")) {
                referenceCameraName = belowNode->getStringValue("");
                names[0] = "upperLeft";
                referenceNames[0] = "lowerLeft";
                names[1] = "upperRight";
                referenceNames[1] = "lowerRight";
            } else {
                // names[0] = ;
                // referenceNames[0] = ;
                // names[1] = ;
                // referenceNames[1] = ;
            }

            // If we finally found a set of reference points that should match,
            // then create a relative frustum matching these references
            if (SlaveCamera* referenceSlaveCamera = getSlaveCamera(referenceCameraName)) {
                slaveCamera->setRelativeFrustum(names, *referenceSlaveCamera, referenceNames);
            } else {
                SG_LOG(SG_VIEW, SG_ALERT, "Unable to find reference camera \"" << referenceCameraName
                       << "\" for camera \"" << name << "\"!");
            }
        } else {
            // Set a proper default taking the current aspect ratio into account
            slaveCamera->setFustumByFieldOfViewDeg(55);
        }
    }

    return true;
}

void
Viewer::setupDefaultCameraConfigIfUnset()
{
    if (getNumDrawables() || getNumSlaveCameras())
        return;

    osg::GraphicsContext::ScreenIdentifier screenIdentifier;
    screenIdentifier = getDefaultScreenIdentifier();

    Drawable* drawable = getOrCreateDrawable("fgviewer");
    drawable->setScreenIdentifier(screenIdentifier.displayName());
    drawable->setPosition(SGVec2i(0, 0));
    SGVec2i size(800, 600);
    drawable->setSize(size);
    drawable->setFullscreen(false);
    drawable->setOffscreen(false);

    SlaveCamera* slaveCamera = getOrCreateSlaveCamera(drawable->getName());
    slaveCamera->setDrawableName(drawable->getName());
    drawable->attachSlaveCamera(slaveCamera);
    slaveCamera->setViewport(SGVec4i(0, 0, size[0], size[1]));
    slaveCamera->setViewOffset(osg::Matrix::identity());
    slaveCamera->setFustumByFieldOfViewDeg(55);
}

bool
Viewer::readConfiguration(const std::string&)
{
    return false;
}

void
Viewer::setRenderer(Renderer* renderer)
{
    if (!renderer) {
        SG_LOG(SG_VIEW, SG_ALERT, "Viewer::setRenderer(): Setting the renderer to zero is not supported!");
        return;
    }
    if (_renderer.valid()) {
        SG_LOG(SG_VIEW, SG_ALERT, "Viewer::setRenderer(): Setting the renderer twice is not supported!");
        return;
    }
    _renderer = renderer;
}

Renderer*
Viewer::getRenderer()
{
    return _renderer.get();
}

Drawable*
Viewer::getOrCreateDrawable(const std::string& name)
{
    Drawable* drawable = getDrawable(name);
    if (drawable)
        return drawable;
    if (!_renderer.valid())
        return 0;
    drawable = _renderer->createDrawable(*this, name);
    if (!drawable)
        return 0;
    _drawableVector.push_back(drawable);
    return drawable;
}

Drawable*
Viewer::getDrawable(const std::string& name)
{
    return getDrawable(getDrawableIndex(name));
}

unsigned
Viewer::getDrawableIndex(const std::string& name)
{
    for (DrawableVector::size_type i = 0; i < _drawableVector.size(); ++i) {
        if (_drawableVector[i]->getName() == name)
            return i;
    }
    return ~0u;
}

Drawable*
Viewer::getDrawable(unsigned index)
{
    if (_drawableVector.size() <= index)
        return 0;
    return _drawableVector[index].get();
}

unsigned
Viewer::getNumDrawables() const
{
    return _drawableVector.size();
}

SlaveCamera*
Viewer::getOrCreateSlaveCamera(const std::string& name)
{
    SlaveCamera* slaveCamera = getSlaveCamera(name);
    if (slaveCamera)
        return slaveCamera;
    if (!_renderer.valid())
        return 0;
    slaveCamera = _renderer->createSlaveCamera(*this, name);
    if (!slaveCamera)
        return 0;
    _slaveCameraVector.push_back(slaveCamera);
    return slaveCamera;
}

SlaveCamera*
Viewer::getSlaveCamera(const std::string& name)
{
    return getSlaveCamera(getSlaveCameraIndex(name));
}

unsigned
Viewer::getSlaveCameraIndex(const std::string& name)
{
    for (SlaveCameraVector::size_type i = 0; i < _slaveCameraVector.size(); ++i) {
        if (_slaveCameraVector[i]->getName() == name)
            return i;
    }
    return ~0u;
}

SlaveCamera*
Viewer::getSlaveCamera(unsigned index)
{
    if (_slaveCameraVector.size() <= index)
        return 0;
    return _slaveCameraVector[index].get();
}

unsigned
Viewer::getNumSlaveCameras() const
{
    return _slaveCameraVector.size();
}

void
Viewer::realize()
{
    if (isRealized())
        return;

    if (!_renderer.valid())
        return;

    // Setup a default config if there is none
    setupDefaultCameraConfigIfUnset();

    // Realize
    if (!_renderer->realize(*this)) {
        SG_LOG(SG_VIEW, SG_ALERT, "Renderer::realize() failed!");
        return;
    }

    osgViewer::Viewer::realize();
}

bool
Viewer::realizeDrawables()
{
    for (DrawableVector::iterator i = _drawableVector.begin(); i != _drawableVector.end(); ++i) {
        if (!(*i)->realize(*this)) {
            SG_LOG(SG_VIEW, SG_ALERT, "Unable to realize drawable \"" << (*i)->getName() << "\"!");
            return false;
        }
    }

    return true;
}

bool
Viewer::realizeSlaveCameras()
{
    for (SlaveCameraVector::iterator i = _slaveCameraVector.begin(); i != _slaveCameraVector.end(); ++i) {
        if (!(*i)->realize(*this)) {
            SG_LOG(SG_VIEW, SG_ALERT, "Unable to realize camera \"" << (*i)->getName() << "\"!");
            return false;
        }
    }

    return true;
}

void
Viewer::advance(double)
{
    if (_timeIncrement == SGTimeStamp::fromSec(0)) {
        // Flightgears current scheme - could be improoved
        _simTime = SGTimeStamp::now();
    } else {
        // Giving an explicit time increment makes sense in presence
        // of the video capture where we need deterministic
        // frame times and object positions for each picture.
        _simTime += _timeIncrement;
    }

    // This sets the frame stamps simulation time to simTime
    // and schedules a frame event
    osgViewer::Viewer::advance(_simTime.toSecs());
}

void
Viewer::updateTraversal()
{
#if FG_HAVE_HLA
    if (_viewerFederate.valid()) {
        if (_timeIncrement == SGTimeStamp::fromSec(0)) {
            if (!_viewerFederate->timeAdvanceAvailable()) {
                SG_LOG(SG_NETWORK, SG_ALERT, "Got error from federate update!");
                _viewerFederate->shutdown();
                _viewerFederate = 0;
            }
        } else {
            osg::FrameStamp* frameStamp = getViewerFrameStamp();
            SGTimeStamp timeStamp = SGTimeStamp::fromSec(frameStamp->getSimulationTime());
            if (!_viewerFederate->timeAdvance(timeStamp)) {
                SG_LOG(SG_NETWORK, SG_ALERT, "Got error from federate update!");
                _viewerFederate->shutdown();
                _viewerFederate = 0;
            }
        }
    }
#endif

    osgViewer::Viewer::updateTraversal();

    if (!_renderer->update(*this)) {
        SG_LOG(SG_VIEW, SG_ALERT, "Renderer::update() failed!");
    }
}

bool
Viewer::updateSlaveCameras()
{
    for (SlaveCameraVector::iterator i = _slaveCameraVector.begin(); i != _slaveCameraVector.end(); ++i) {
        if (!(*i)->update(*this)) {
            SG_LOG(SG_VIEW, SG_ALERT, "SlaveCamera::update() failed!");
            return false;
        }
    }
    return true;
}

void
Viewer::setReaderWriterOptions(simgear::SGReaderWriterOptions* readerWriterOptions)
{
    _readerWriterOptions = readerWriterOptions;
}

simgear::SGReaderWriterOptions*
Viewer::getReaderWriterOptions()
{
    return _readerWriterOptions.get();
}

void
Viewer::setSceneData(osg::Node* node)
{
    _sceneDataGroup->removeChildren(0, _sceneDataGroup->getNumChildren());
    insertSceneData(node);
}

void
Viewer::insertSceneData(osg::Node* node)
{
    _sceneDataGroup->addChild(node);
}

bool
Viewer::insertSceneData(const std::string& fileName, const osgDB::Options* options)
{
#if 0
    osg::ProxyNode* proxyNode = new osg::ProxyNode;
    if (options)
        proxyNode->setDatabaseOptions(options->clone(osg::CopyOp()));
    else
        proxyNode->setDatabaseOptions(_readerWriterOptions->clone(osg::CopyOp()));
    proxyNode->setFileName(0, fileName);
    insertSceneData(proxyNode);
    return true;
#else
    osg::ref_ptr<osg::Node> node = osgDB::readRefNodeFile(fileName, options);
    if (!node.valid())
        return false;
    insertSceneData(node.get());
    return true;
#endif
}

osg::Group*
Viewer::getSceneDataGroup()
{
    return _sceneDataGroup.get();
}

class Viewer::_PurgeLevelOfDetailNodesVisitor : public osg::NodeVisitor {
public:
    _PurgeLevelOfDetailNodesVisitor() :
        osg::NodeVisitor(osg::NodeVisitor::TRAVERSE_ALL_CHILDREN)
    { }
    virtual ~_PurgeLevelOfDetailNodesVisitor()
    { }
  
    virtual void apply(osg::ProxyNode& node)
    {
        for (unsigned i = 0; i < node.getNumChildren(); ++i) {
            if (node.getFileName(i).empty())
                continue;
            node.removeChildren(i, node.getNumChildren() - i);
            break;
        }

        osg::NodeVisitor::apply(static_cast<osg::Group&>(node));
    }
    virtual void apply(osg::PagedLOD& node)
    {
        for (unsigned i = 0; i < node.getNumChildren(); ++i) {
            if (node.getFileName(i).empty())
                continue;
            node.removeChildren(i, node.getNumChildren() - i);
            break;
        }

        osg::NodeVisitor::apply(static_cast<osg::Group&>(node));
    }
};

void
Viewer::purgeLevelOfDetailNodes()
{
    _PurgeLevelOfDetailNodesVisitor purgeLevelOfDetailNodesVisitor;
    _sceneDataGroup->accept(purgeLevelOfDetailNodesVisitor);
}

osg::GraphicsContext::ScreenIdentifier
Viewer::getDefaultScreenIdentifier()
{
    osg::GraphicsContext::ScreenIdentifier screenIdentifier;
    screenIdentifier.readDISPLAY();
    if (screenIdentifier.displayNum < 0)
        screenIdentifier.displayNum = 0;
    if (screenIdentifier.screenNum < 0)
        screenIdentifier.screenNum = 0;
    return screenIdentifier;
}

osg::GraphicsContext::ScreenIdentifier
Viewer::getScreenIdentifier(const std::string& display)
{
    osg::GraphicsContext::ScreenIdentifier screenIdentifier;
    screenIdentifier.setScreenIdentifier(display);

    osg::GraphicsContext::ScreenIdentifier defaultScreenIdentifier;
    defaultScreenIdentifier = getDefaultScreenIdentifier();
    if (screenIdentifier.hostName.empty())
        screenIdentifier.hostName = defaultScreenIdentifier.hostName;
    if (screenIdentifier.displayNum < 0)
        screenIdentifier.displayNum = defaultScreenIdentifier.displayNum;
    if (screenIdentifier.screenNum < 0)
        screenIdentifier.screenNum = defaultScreenIdentifier.screenNum;
    
    return screenIdentifier;
}

osg::GraphicsContext::ScreenSettings
Viewer::getScreenSettings(const osg::GraphicsContext::ScreenIdentifier& screenIdentifier)
{
    osg::GraphicsContext::ScreenSettings screenSettings;

    osg::GraphicsContext::WindowingSystemInterface* wsi;
    wsi = osg::GraphicsContext::getWindowingSystemInterface();
    if (!wsi) {
        SG_LOG(SG_VIEW, SG_ALERT, "No windowing system interface defined!");
        return screenSettings;
    }

    wsi->getScreenSettings(screenIdentifier, screenSettings);
    return screenSettings;
}

osg::GraphicsContext::Traits*
Viewer::getTraits(const osg::GraphicsContext::ScreenIdentifier& screenIdentifier)
{
    osg::DisplaySettings* ds = _displaySettings.get();
    if (!ds)
        ds = osg::DisplaySettings::instance().get();

    osg::GraphicsContext::Traits* traits = new osg::GraphicsContext::Traits(ds);

    traits->hostName = screenIdentifier.hostName;
    traits->displayNum = screenIdentifier.displayNum;
    traits->screenNum = screenIdentifier.screenNum;
            
    // not seriously consider something different
    traits->doubleBuffer = true;

    osg::GraphicsContext::ScreenSettings screenSettings;
    screenSettings = getScreenSettings(screenIdentifier);

    traits->x = 0;
    traits->y = 0;
    traits->width = screenSettings.width;
    traits->height = screenSettings.height;

    return traits;
}

#ifdef __linux__
class Viewer::_ResetScreenSaverSwapCallback : public osg::GraphicsContext::SwapCallback {
public:
    _ResetScreenSaverSwapCallback() :
        _timeStamp(SGTimeStamp::fromSec(0))
    {
    }
    virtual ~_ResetScreenSaverSwapCallback()
    {
    }
    virtual void swapBuffersImplementation(osg::GraphicsContext* graphicsContext)
    {
        graphicsContext->swapBuffersImplementation();

        // This callback must be attached to this type of graphics context
        assert(dynamic_cast<osgViewer::GraphicsWindowX11*>(graphicsContext));

        // Reset the screen saver every 10 seconds
        SGTimeStamp timeStamp = SGTimeStamp::now();
        if (timeStamp < _timeStamp)
            return;
        _timeStamp = timeStamp + SGTimeStamp::fromSec(10);
        // Obviously runs in the draw thread. Thus, use the draw display.
        XResetScreenSaver(static_cast<osgViewer::GraphicsWindowX11*>(graphicsContext)->getDisplay());
    }
private:
    SGTimeStamp _timeStamp;
};
#endif

osg::GraphicsContext*
Viewer::createGraphicsContext(osg::GraphicsContext::Traits* traits)
{
    osg::GraphicsContext::WindowingSystemInterface* wsi;
    wsi = osg::GraphicsContext::getWindowingSystemInterface();
    if (!wsi) {
        SG_LOG(SG_VIEW, SG_ALERT, "No windowing system interface defined!");
        return 0;
    }

    osg::GraphicsContext* graphicsContext = wsi->createGraphicsContext(traits);
    if (!graphicsContext) {
        SG_LOG(SG_VIEW, SG_ALERT, "Unable to create window \"" << traits->windowName << "\"!");
        return 0;
    }

#ifdef __linux__
    if (dynamic_cast<osgViewer::GraphicsWindowX11*>(graphicsContext))
        graphicsContext->setSwapCallback(new _ResetScreenSaverSwapCallback);
#endif

    return graphicsContext;
}

#if FG_HAVE_HLA
const HLAViewerFederate*
Viewer::getViewerFederate() const
{
    return _viewerFederate.get();
}

HLAViewerFederate*
Viewer::getViewerFederate()
{
    return _viewerFederate.get();
}

void
Viewer::setViewerFederate(HLAViewerFederate* viewerFederate)
{
    if (!viewerFederate) {
        SG_LOG(SG_VIEW, SG_ALERT, "Viewer::setViewerFederate(): Setting the viewer federate to zero is not supported!");
        return;
    }
    if (_viewerFederate.valid()) {
        SG_LOG(SG_VIEW, SG_ALERT, "Viewer::setViewerFederate(): Setting the viewer federate twice is not supported!");
        return;
    }
    _viewerFederate = viewerFederate;
    _viewerFederate->attachToViewer(this);
}
#endif

} // namespace fgviewer