#ifndef MWVR_VRVIEWER_H #define MWVR_VRVIEWER_H #include #include #include #include #include #include #include "openxrmanager.hpp" #include #include namespace MWVR { class VRFramebuffer; class VRView; class OpenXRSwapchain; /// \brief Manages stereo rendering and mirror texturing. /// /// Manipulates the osgViewer by disabling main camera rendering, and instead rendering to /// two slave cameras, each connected to and manipulated by a VRView class. class VRViewer { public: struct UpdateViewCallback : public Misc::StereoView::UpdateViewCallback { UpdateViewCallback(VRViewer* viewer) : mViewer(viewer) {}; //! Called during the update traversal of every frame to source updated stereo values. virtual void updateView(Misc::View& left, Misc::View& right) override; VRViewer* mViewer; }; class SwapBuffersCallback : public osg::GraphicsContext::SwapCallback { public: SwapBuffersCallback(VRViewer* viewer) : mViewer(viewer) {}; void swapBuffersImplementation(osg::GraphicsContext* gc) override; private: VRViewer* mViewer; }; class PredrawCallback : public osg::Camera::DrawCallback { public: PredrawCallback(VRViewer* viewer) : mViewer(viewer) {} void operator()(osg::RenderInfo& info) const override { mViewer->preDrawCallback(info); }; private: VRViewer* mViewer; }; class PostdrawCallback : public osg::Camera::DrawCallback { public: PostdrawCallback(VRViewer* viewer) : mViewer(viewer) {} void operator()(osg::RenderInfo& info) const override { mViewer->postDrawCallback(info); }; private: VRViewer* mViewer; }; class InitialDrawCallback : public osg::Camera::DrawCallback { public: InitialDrawCallback(VRViewer* viewer) : mViewer(viewer) {} void operator()(osg::RenderInfo& info) const override { mViewer->initialDrawCallback(info); }; private: VRViewer* mViewer; }; class FinaldrawCallback : public Misc::StereoView::StereoDrawCallback { public: FinaldrawCallback(VRViewer* viewer) : mViewer(viewer) {} void operator()(osg::RenderInfo& info, Misc::StereoView::StereoDrawCallback::View view) const override; private: VRViewer* mViewer; }; static const std::array sViewNames; enum class MirrorTextureEye { Left, Right, Both }; public: VRViewer( osg::ref_ptr viewer); ~VRViewer(void); void swapBuffersCallback(osg::GraphicsContext* gc); void initialDrawCallback(osg::RenderInfo& info); void preDrawCallback(osg::RenderInfo& info); void postDrawCallback(osg::RenderInfo& info); void finalDrawCallback(osg::RenderInfo& info); void blit(osg::RenderInfo& gc); void configureXR(osg::GraphicsContext* gc); void configureCallbacks(); void setupMirrorTexture(); void processChangedSettings(const std::set< std::pair >& changed); void updateView(Misc::View& left, Misc::View& right); SubImage subImage(Side side); bool xrConfigured() { return mOpenXRConfigured; }; bool callbacksConfigured() { return mCallbacksConfigured; }; private: std::mutex mMutex{}; bool mOpenXRConfigured{ false }; bool mCallbacksConfigured{ false }; osg::ref_ptr mViewer = nullptr; osg::ref_ptr mPreDraw{ nullptr }; osg::ref_ptr mPostDraw{ nullptr }; osg::ref_ptr mFinalDraw{ nullptr }; std::shared_ptr mUpdateViewCallback{ nullptr }; bool mRenderingReady{ false }; std::unique_ptr mMirrorTexture; std::vector mMirrorTextureViews; bool mMirrorTextureShouldBeCleanedUp{ false }; bool mMirrorTextureEnabled{ false }; bool mFlipMirrorTextureOrder{ false }; MirrorTextureEye mMirrorTextureEye{ MirrorTextureEye::Both }; std::unique_ptr mFramebuffer; std::unique_ptr mMsaaResolveTexture; std::unique_ptr mGammaResolveTexture; std::array, 2> mSwapchain; std::array mSubImages; std::array mSwapchainConfig; }; } #endif