#ifndef VRGUI_HPP #define VRGUI_HPP #include #include #include #include #include #include #include #include #include #include "vrview.hpp" #include "openxrmanager.hpp" namespace MyGUI { class Widget; class Window; } namespace MWGui { class Layout; class WindowBase; class ILayer; } struct XrCompositionLayerQuad; namespace MWVR { class GUICamera; class VRGUIManager; enum class TrackingMode { Menu, //!< Menu quads with fixed position based on head tracking. HudLeftHand, //!< Hud quads tracking the left hand every frame HudRightHand, //!< Hud quads tracking the right hand every frame }; // Some UI elements should occupy predefined geometries // Others should grow/shrink freely enum class SizingMode { Auto, Fixed }; /// Configuration of a VRGUILayer struct LayerConfig { int priority; //!< Higher priority shows over lower priority windows by moving higher priority layers slightly closer to the player. bool sideBySide; //!< All layers with this config will show up side by side in a partial circle around the player, and will all be resized to a predefined size. osg::Vec4 backgroundColor; //!< Background color of layer osg::Vec3 offset; //!< Offset from tracked node in meters osg::Vec2 center; //!< Model space centerpoint of menu geometry. All menu geometries have model space lengths of 1 in each dimension. Use this to affect how geometries grow with changing size. osg::Vec2 extent; //!< Spatial extent of the layer in meters when using Fixed sizing mode int spatialResolution; //!< Pixels when using the Auto sizing mode. \note Meters per pixel of the GUI viewport, not the RTT texture. osg::Vec2i pixelResolution; //!< Pixel resolution of the RTT texture osg::Vec2 myGUIViewSize; //!< Resizable elements are resized to this (fraction of full view) SizingMode sizingMode; //!< How to size the layer TrackingMode trackingMode; //!< Tracking mode std::string extraLayers; //!< Additional layers to draw (list separated by any non-alphabetic) bool ignoreModality; //!< Layer input should be enabled even if another modal window is active bool operator<(const LayerConfig& rhs) const { return priority < rhs.priority; } }; /// \brief A single VR GUI Quad. /// /// In VR menus are shown as quads within the game world. /// The behaviour of that quad is defined by the MWVR::LayerConfig struct /// Each instance of VRGUILayer is used to show one MYGUI layer. class VRGUILayer { public: VRGUILayer( osg::ref_ptr geometryRoot, osg::ref_ptr cameraRoot, MyGUI::ILayer* layer, LayerConfig config, VRGUIManager* parent); ~VRGUILayer(); void update(); protected: friend class VRGUIManager; osg::Camera* camera(); osg::ref_ptr menuTexture(); void setAngle(float angle); void updateTracking(const Pose& headPose = {}); void updatePose(); void updateRect(); void insertWidget(MWGui::Layout* widget); void removeWidget(MWGui::Layout* widget); int widgetCount() { return mWidgets.size(); } bool operator<(const VRGUILayer& rhs) const { return mConfig.priority < rhs.mConfig.priority; } public: Pose mTrackedPose{}; LayerConfig mConfig; std::string mLayerName; MyGUI::ILayer* mMyGUILayer; std::vector mWidgets; osg::ref_ptr mGeometryRoot; osg::ref_ptr mGeometry{ new osg::Geometry }; osg::ref_ptr mTransform{ new osg::PositionAttitudeTransform }; osg::ref_ptr mCameraRoot; osg::ref_ptr mGUICamera; osg::ref_ptr mMyGUICamera{ nullptr }; MyGUI::FloatRect mRealRect{}; osg::Quat mRotation{ 0,0,0,1 }; }; /// \brief osg user data used to refer back to VRGUILayer objects when intersecting with the scene graph. class VRGUILayerUserData : public osg::Referenced { public: VRGUILayerUserData(std::shared_ptr layer) : mLayer(layer) {}; std::weak_ptr mLayer; }; /// \brief Manager of VRGUILayer objects. /// /// Constructs and destructs VRGUILayer objects in response to MWGui::Layout::setVisible calls. /// Layers can also be made visible directly by calling insertLayer() directly, e.g. to show /// the video player. class VRGUIManager { public: VRGUIManager( osg::ref_ptr viewer); ~VRGUIManager(void); /// Set visibility of the layer this layout is on void setVisible(MWGui::Layout*, bool visible); /// Insert the given layer quad if it isn't already void insertLayer(MyGUI::ILayer* layer); /// Remove the given layer quad void removeLayer(MyGUI::ILayer* layer); /// Update layer quads based on current camera void updateTracking(void); /// Set camera on which to base tracking void setCamera(osg::Camera* camera); /// Check current pointer target and update focus layer bool updateFocus(); /// Update traversal void update(); /// Gui cursor coordinates to use to simulate a mouse press/move if the player is currently pointing at a vr gui layer osg::Vec2i guiCursor() { return mGuiCursor; }; /// Inject mouse click if applicable bool injectMouseClick(bool onPress); /// Notify VRGUIManager that a widget has been unlinked void notifyWidgetUnlinked(MyGUI::Widget* widget); /// Notify VRGUIManager that a window is modal void notifyModalWindow(MyGUI::Widget* window); /// Currently focus widget according to VR MyGUI::Widget* focusWidget() { return mFocusWidget; } private: void computeGuiCursor(osg::Vec3 hitPoint); void updateSideBySideLayers(); void insertWidget(MWGui::Layout* widget); void removeWidget(MWGui::Layout* widget); void setFocusLayer(VRGUILayer* layer); void setFocusWidget(MyGUI::Widget* widget); bool validateFocusWidget(); bool focusIsModalWindow(); MyGUI::Widget* widgetFromGuiCursor(int x, int y); void updateGuiCursor(int x, int y); osg::ref_ptr mOsgViewer{ nullptr }; osg::ref_ptr mGUIGeometriesRoot{ new osg::Group }; osg::ref_ptr mGUICamerasRoot{ new osg::Group }; std::map> mLayers; std::vector > mSideBySideLayers; bool mShouldUpdatePoses{ true }; Pose mHeadPose{}; osg::Vec2i mGuiCursor{}; VRGUILayer* mFocusLayer{ nullptr }; MyGUI::Widget* mFocusWidget{ nullptr }; MyGUI::Widget* mModalWindow{ nullptr }; osg::observer_ptr mCamera{ nullptr }; std::map mLayerConfigs{}; }; } #endif