2020-05-24 16:00:42 +00:00
|
|
|
#include "vrview.hpp"
|
2020-06-26 21:02:48 +00:00
|
|
|
|
2020-05-24 16:00:42 +00:00
|
|
|
#include "openxrmanager.hpp"
|
|
|
|
#include "openxrmanagerimpl.hpp"
|
2020-06-26 21:02:48 +00:00
|
|
|
#include "vrsession.hpp"
|
2020-05-24 16:00:42 +00:00
|
|
|
#include "vrenvironment.hpp"
|
|
|
|
|
|
|
|
#include <components/debug/debuglog.hpp>
|
|
|
|
|
|
|
|
#include <osgViewer/Renderer>
|
|
|
|
|
|
|
|
namespace MWVR {
|
|
|
|
|
|
|
|
VRView::VRView(
|
|
|
|
std::string name,
|
2020-06-24 19:26:11 +00:00
|
|
|
SwapchainConfig config,
|
2020-05-24 16:00:42 +00:00
|
|
|
osg::ref_ptr<osg::State> state)
|
|
|
|
: mSwapchainConfig{ config }
|
|
|
|
, mSwapchain(new OpenXRSwapchain(state, mSwapchainConfig))
|
|
|
|
, mName(name)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
VRView::~VRView()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2020-06-03 17:46:20 +00:00
|
|
|
class CullCallback : public osg::NodeCallback
|
|
|
|
{
|
|
|
|
void operator()(osg::Node* node, osg::NodeVisitor* nv)
|
|
|
|
{
|
|
|
|
const auto& name = node->getName();
|
|
|
|
if (name == "LeftEye")
|
|
|
|
Environment::get().getSession()->beginPhase(VRSession::FramePhase::Cull);
|
|
|
|
traverse(node, nv);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2020-07-20 13:03:48 +00:00
|
|
|
osg::Camera* VRView::createCamera(int order, const osg::Vec4& clearColor, osg::GraphicsContext* gc)
|
2020-05-24 16:00:42 +00:00
|
|
|
{
|
|
|
|
osg::ref_ptr<osg::Camera> camera = new osg::Camera();
|
|
|
|
camera->setClearColor(clearColor);
|
|
|
|
camera->setClearMask(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT);
|
|
|
|
camera->setRenderTargetImplementation(osg::Camera::FRAME_BUFFER_OBJECT);
|
2020-07-20 13:03:48 +00:00
|
|
|
camera->setRenderOrder(osg::Camera::PRE_RENDER, order);
|
2020-05-24 16:00:42 +00:00
|
|
|
camera->setComputeNearFarMode(osg::CullSettings::DO_NOT_COMPUTE_NEAR_FAR);
|
|
|
|
camera->setAllowEventFocus(false);
|
|
|
|
camera->setReferenceFrame(osg::Transform::ABSOLUTE_RF);
|
|
|
|
camera->setViewport(0, 0, mSwapchain->width(), mSwapchain->height());
|
|
|
|
camera->setGraphicsContext(gc);
|
|
|
|
|
|
|
|
camera->setInitialDrawCallback(new VRView::InitialDrawCallback());
|
2020-06-03 17:46:20 +00:00
|
|
|
camera->setCullCallback(new CullCallback);
|
2020-05-24 16:00:42 +00:00
|
|
|
|
|
|
|
return camera.release();
|
|
|
|
}
|
|
|
|
|
|
|
|
void VRView::prerenderCallback(osg::RenderInfo& renderInfo)
|
|
|
|
{
|
2020-07-21 10:28:39 +00:00
|
|
|
if(Environment::get().getSession()->getFrame(VRSession::FramePhase::Draw)->mShouldRender)
|
2020-05-24 16:00:42 +00:00
|
|
|
mSwapchain->beginFrame(renderInfo.getState()->getGraphicsContext());
|
|
|
|
}
|
|
|
|
|
|
|
|
void VRView::InitialDrawCallback::operator()(osg::RenderInfo& renderInfo) const
|
|
|
|
{
|
2020-06-03 17:46:20 +00:00
|
|
|
const auto& name = renderInfo.getCurrentCamera()->getName();
|
|
|
|
if (name == "LeftEye")
|
|
|
|
Environment::get().getSession()->beginPhase(VRSession::FramePhase::Draw);
|
|
|
|
|
2020-05-24 16:00:42 +00:00
|
|
|
osg::GraphicsOperation* graphicsOperation = renderInfo.getCurrentCamera()->getRenderer();
|
|
|
|
osgViewer::Renderer* renderer = dynamic_cast<osgViewer::Renderer*>(graphicsOperation);
|
|
|
|
if (renderer != nullptr)
|
|
|
|
{
|
|
|
|
// Disable normal OSG FBO camera setup
|
|
|
|
renderer->setCameraRequiresSetUp(false);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
void VRView::UpdateSlaveCallback::updateSlave(
|
2020-06-26 21:02:48 +00:00
|
|
|
osg::View& view,
|
|
|
|
osg::View::Slave& slave)
|
2020-07-26 11:12:36 +00:00
|
|
|
{
|
|
|
|
mView->updateSlave(view, slave);
|
|
|
|
}
|
|
|
|
|
|
|
|
void VRView::postrenderCallback(osg::RenderInfo& renderInfo)
|
|
|
|
{
|
|
|
|
auto name = renderInfo.getCurrentCamera()->getName();
|
|
|
|
}
|
|
|
|
|
|
|
|
void VRView::swapBuffers(osg::GraphicsContext* gc)
|
|
|
|
{
|
|
|
|
mSwapchain->endFrame(gc);
|
|
|
|
}
|
|
|
|
void VRView::updateSlave(osg::View& view, osg::View::Slave& slave)
|
2020-05-24 16:00:42 +00:00
|
|
|
{
|
|
|
|
auto* camera = slave._camera.get();
|
|
|
|
|
2020-07-21 10:28:39 +00:00
|
|
|
// Update current cached cull mask of camera if it is active
|
|
|
|
auto mask = camera->getCullMask();
|
|
|
|
if (mask == 0)
|
|
|
|
camera->setCullMask(mCullMask);
|
|
|
|
else
|
|
|
|
mCullMask = mask;
|
2020-05-24 16:00:42 +00:00
|
|
|
|
2020-07-26 11:12:36 +00:00
|
|
|
// If the session is not active, we do not want to waste resources rendering frames.
|
2020-07-21 10:28:39 +00:00
|
|
|
if (Environment::get().getSession()->getFrame(VRSession::FramePhase::Update)->mShouldRender)
|
|
|
|
{
|
|
|
|
Side side = Side::RIGHT_SIDE;
|
2020-07-26 11:12:36 +00:00
|
|
|
if (mName == "LeftEye")
|
|
|
|
{
|
|
|
|
|
|
|
|
Environment::get().getViewer()->vrShadow().updateShadowConfig(view);
|
2020-07-21 10:28:39 +00:00
|
|
|
side = Side::LEFT_SIDE;
|
2020-07-26 11:12:36 +00:00
|
|
|
}
|
2020-07-21 10:28:39 +00:00
|
|
|
|
|
|
|
auto* session = Environment::get().getSession();
|
|
|
|
auto viewMatrix = view.getCamera()->getViewMatrix();
|
2020-07-22 18:01:56 +00:00
|
|
|
|
2020-10-21 19:22:38 +00:00
|
|
|
// If the camera does not have a view, use the VR stage directly
|
|
|
|
bool useStage = !(viewMatrix.getTrans().length() > 0.01);
|
|
|
|
|
|
|
|
// If the view matrix is still the identity matrix, conventions have to be swapped around.
|
|
|
|
bool swapConventions = viewMatrix.isIdentity();
|
|
|
|
|
|
|
|
viewMatrix = viewMatrix * session->viewMatrix(VRSession::FramePhase::Update, side, !useStage, !swapConventions);
|
|
|
|
|
2020-07-22 18:01:56 +00:00
|
|
|
camera->setViewMatrix(viewMatrix);
|
2020-05-24 16:00:42 +00:00
|
|
|
|
2020-07-22 18:01:56 +00:00
|
|
|
auto projectionMatrix = session->projectionMatrix(VRSession::FramePhase::Update, side);
|
2020-07-21 10:28:39 +00:00
|
|
|
camera->setProjectionMatrix(projectionMatrix);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
camera->setCullMask(0);
|
|
|
|
}
|
2020-05-24 16:00:42 +00:00
|
|
|
slave.updateSlaveImplementation(view);
|
|
|
|
}
|
|
|
|
}
|